Fix #15, Support multipart post.
This commit is contained in:
parent
e318a72006
commit
9c1cbb81c9
10 changed files with 239 additions and 30 deletions
|
@ -2,8 +2,8 @@
|
||||||
'targets': [
|
'targets': [
|
||||||
{
|
{
|
||||||
'target_name': 'node-curl',
|
'target_name': 'node-curl',
|
||||||
'cflags': ['-Wall', '-O1', '-fno-inline-functions'],
|
'cflags': ['-Wall', '-O1', '-g', '-fno-inline-functions'],
|
||||||
'cflags_cc': ['-Wall', '-O1', '-fno-inline-functions'],
|
'cflags_cc': ['-Wall', '-O1', '-g', '-fno-inline-functions'],
|
||||||
'sources': ['src/node-curl.cc'],
|
'sources': ['src/node-curl.cc'],
|
||||||
'libraries': ['-lcurl']
|
'libraries': ['-lcurl']
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
// Generated by ToffeeScript 1.4.0
|
// Generated by ToffeeScript 1.6.2
|
||||||
(function() {
|
(function() {
|
||||||
var curl, p,
|
var curl, err, p, res,
|
||||||
_this = this;
|
_this = this;
|
||||||
|
|
||||||
curl = require('../index');
|
curl = require('../index');
|
||||||
|
@ -10,10 +10,8 @@
|
||||||
curl(process.argv[2], {
|
curl(process.argv[2], {
|
||||||
VERBOSE: 1,
|
VERBOSE: 1,
|
||||||
DEBUG: 1
|
DEBUG: 1
|
||||||
}, function(_$$_err, _$$_res) {
|
}, function() {
|
||||||
var err, res;
|
err = arguments[0], res = arguments[1];
|
||||||
err = _$$_err;
|
|
||||||
res = _$$_res;
|
|
||||||
if (err) {
|
if (err) {
|
||||||
return p(err);
|
return p(err);
|
||||||
} else {
|
} else {
|
||||||
|
|
|
@ -21,7 +21,6 @@ curl.on('data', function(chunk) {
|
||||||
curl.on('error', function(e) {
|
curl.on('error', function(e) {
|
||||||
p("error: " + e.message);
|
p("error: " + e.message);
|
||||||
curl.close();
|
curl.close();
|
||||||
once();
|
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
||||||
|
|
27
examples/post-multi-part.js
Normal file
27
examples/post-multi-part.js
Normal file
|
@ -0,0 +1,27 @@
|
||||||
|
// Generated by ToffeeScript 1.6.2
|
||||||
|
(function() {
|
||||||
|
var curl, p;
|
||||||
|
|
||||||
|
curl = require('../index');
|
||||||
|
|
||||||
|
p = console.log;
|
||||||
|
|
||||||
|
p('start');
|
||||||
|
|
||||||
|
curl('127.0.0.1/upload.php', {
|
||||||
|
multipart: [
|
||||||
|
{
|
||||||
|
name: 'file',
|
||||||
|
file: '/home/miao/test.js',
|
||||||
|
type: 'text/html'
|
||||||
|
}, {
|
||||||
|
name: 'sumbit',
|
||||||
|
value: 'send'
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}, function(e) {
|
||||||
|
console.log(e);
|
||||||
|
return console.log(curl.body);
|
||||||
|
});
|
||||||
|
|
||||||
|
}).call(this);
|
12
examples/post-multi-part.toffee
Normal file
12
examples/post-multi-part.toffee
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
curl = require '../index'
|
||||||
|
p = console.log
|
||||||
|
p 'start'
|
||||||
|
|
||||||
|
curl '127.0.0.1/upload.php', {
|
||||||
|
multipart: [
|
||||||
|
{name: 'file', file: '/home/miao/test.js', type: 'text/html'},
|
||||||
|
{name: 'sumbit', value: 'send'}
|
||||||
|
],
|
||||||
|
}, (e) ->
|
||||||
|
console.log e
|
||||||
|
console.log curl.body
|
|
@ -1,6 +1,6 @@
|
||||||
// Generated by ToffeeScript 1.4.0
|
// Generated by ToffeeScript 1.6.2
|
||||||
(function() {
|
(function() {
|
||||||
var cookieFile, curl, fs, options, p, util,
|
var cookieFile, curl, err, fs, options, p, stream, util,
|
||||||
_this = this;
|
_this = this;
|
||||||
|
|
||||||
curl = require('../index');
|
curl = require('../index');
|
||||||
|
@ -25,14 +25,12 @@
|
||||||
|
|
||||||
curl.setDefaultOptions(options);
|
curl.setDefaultOptions(options);
|
||||||
|
|
||||||
curl('www.google.com', function(_$$_err) {
|
curl('www.google.com', function() {
|
||||||
var err;
|
err = arguments[0];
|
||||||
err = _$$_err;
|
|
||||||
p("\x1b[33m" + util.inspect(curl.info('COOKIELIST')) + "\x1b[0m");
|
p("\x1b[33m" + util.inspect(curl.info('COOKIELIST')) + "\x1b[0m");
|
||||||
curl.reset();
|
curl.reset();
|
||||||
return curl('www.yahoo.com', function(_$$_err) {
|
curl('www.yahoo.com', function() {
|
||||||
var stream;
|
err = arguments[0];
|
||||||
err = _$$_err;
|
|
||||||
p("\x1b[33m" + util.inspect(curl.info('COOKIELIST')) + "\x1b[0m");
|
p("\x1b[33m" + util.inspect(curl.info('COOKIELIST')) + "\x1b[0m");
|
||||||
p("body length " + curl.body.length);
|
p("body length " + curl.body.length);
|
||||||
p("\x1b[33mText in " + cookieFile + "\x1b[0m");
|
p("\x1b[33mText in " + cookieFile + "\x1b[0m");
|
||||||
|
@ -44,7 +42,7 @@
|
||||||
p("----");
|
p("----");
|
||||||
curl.close();
|
curl.close();
|
||||||
p("deleting " + cookieFile);
|
p("deleting " + cookieFile);
|
||||||
return fs.unlink(cookieFile, function() {
|
fs.unlink(cookieFile, function() {
|
||||||
return p("done.");
|
return p("done.");
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
47
lib/Curl.js
47
lib/Curl.js
|
@ -1,10 +1,12 @@
|
||||||
// Generated by ToffeeScript 1.4.0
|
// Generated by ToffeeScript 1.6.2
|
||||||
(function() {
|
(function() {
|
||||||
var Curl, curls, id, m, p;
|
var Curl, curls, e, id, m, p,
|
||||||
|
__hasProp = {}.hasOwnProperty;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
Curl = require(__dirname + '/../build/Release/node-curl').Curl;
|
Curl = require(__dirname + '/../build/Release/node-curl').Curl;
|
||||||
} catch (e) {
|
} catch (_error) {
|
||||||
|
e = _error;
|
||||||
Curl = require(__dirname + '/../build/default/node-curl').Curl;
|
Curl = require(__dirname + '/../build/default/node-curl').Curl;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -12,11 +14,45 @@
|
||||||
return this.options[option_id] = value;
|
return this.options[option_id] = value;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
Curl.prototype.setopt_httppost = function(rows) {
|
||||||
|
var cols, k, option_id, row, v;
|
||||||
|
this.httppost = (function() {
|
||||||
|
var _i, _len, _results;
|
||||||
|
_results = [];
|
||||||
|
for (_i = 0, _len = rows.length; _i < _len; _i++) {
|
||||||
|
row = rows[_i];
|
||||||
|
cols = [];
|
||||||
|
for (k in row) {
|
||||||
|
if (!__hasProp.call(row, k)) continue;
|
||||||
|
v = row[k];
|
||||||
|
k = k.toUpperCase();
|
||||||
|
if ((option_id = Curl.httppost_options[k]) != null) {
|
||||||
|
cols.push(option_id);
|
||||||
|
if (!(v instanceof Buffer)) {
|
||||||
|
v = new Buffer(v.toString());
|
||||||
|
}
|
||||||
|
cols.push(v);
|
||||||
|
} else {
|
||||||
|
throw new Error("invalid http post option " + k);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_results.push(cols);
|
||||||
|
}
|
||||||
|
return _results;
|
||||||
|
})();
|
||||||
|
this.setopt_httppost_(this.httppost);
|
||||||
|
return this;
|
||||||
|
};
|
||||||
|
|
||||||
Curl.prototype.setopt = function(option_name, value) {
|
Curl.prototype.setopt = function(option_name, value) {
|
||||||
var option, option_id;
|
var option, option_id;
|
||||||
option = option_name.toUpperCase();
|
option = option_name.toUpperCase();
|
||||||
if ((option_id = Curl.user_options[option]) != null) {
|
if ((option_id = Curl.user_options[option]) != null) {
|
||||||
this.setopt_user_(option_id, value);
|
if (option === 'MULTIPART') {
|
||||||
|
this.setopt_httppost(value);
|
||||||
|
} else {
|
||||||
|
this.setopt_user_(option_id, value);
|
||||||
|
}
|
||||||
} else if ((option_id = Curl.slist_options[option]) != null) {
|
} else if ((option_id = Curl.slist_options[option]) != null) {
|
||||||
this.setopt_slist_(option_id, value);
|
this.setopt_slist_(option_id, value);
|
||||||
} else if ((option_id = Curl.integer_options[option]) != null) {
|
} else if ((option_id = Curl.integer_options[option]) != null) {
|
||||||
|
@ -50,7 +86,8 @@
|
||||||
|
|
||||||
Curl.user_options = {
|
Curl.user_options = {
|
||||||
RAW: 'RAW',
|
RAW: 'RAW',
|
||||||
DEBUG: 'DEBUG'
|
DEBUG: 'DEBUG',
|
||||||
|
MULTIPART: 'MULTIPART'
|
||||||
};
|
};
|
||||||
|
|
||||||
id = 0;
|
id = 0;
|
||||||
|
|
|
@ -6,13 +6,38 @@ catch e
|
||||||
Curl::setopt_user_ = (option_id, value) ->
|
Curl::setopt_user_ = (option_id, value) ->
|
||||||
@options[option_id] = value
|
@options[option_id] = value
|
||||||
|
|
||||||
|
Curl::setopt_httppost = (rows) ->
|
||||||
|
# convert object-rows to array-rows
|
||||||
|
# format
|
||||||
|
# [
|
||||||
|
# [OPTION_ID, VALUE]
|
||||||
|
# ]
|
||||||
|
@httppost = for row in rows
|
||||||
|
cols = []
|
||||||
|
for own k, v of row
|
||||||
|
k = k.toUpperCase()
|
||||||
|
if (option_id = Curl.httppost_options[k])?
|
||||||
|
cols.push option_id
|
||||||
|
unless v instanceof Buffer
|
||||||
|
v = new Buffer(v.toString())
|
||||||
|
cols.push v
|
||||||
|
else
|
||||||
|
throw new Error("invalid http post option #{k}")
|
||||||
|
cols
|
||||||
|
@setopt_httppost_(@httppost)
|
||||||
|
@
|
||||||
|
|
||||||
|
|
||||||
Curl::setopt = (option_name, value) ->
|
Curl::setopt = (option_name, value) ->
|
||||||
option = option_name.toUpperCase()
|
option = option_name.toUpperCase()
|
||||||
|
|
||||||
# slist must be at the top of condition
|
# slist must be at the top of condition
|
||||||
# the option exists in string_options too
|
# the option exists in string_options too
|
||||||
if (option_id = Curl.user_options[option])?
|
if (option_id = Curl.user_options[option])?
|
||||||
@setopt_user_ option_id, value
|
if (option == 'MULTIPART')
|
||||||
|
@setopt_httppost value
|
||||||
|
else
|
||||||
|
@setopt_user_ option_id, value
|
||||||
else if (option_id = Curl.slist_options[option])?
|
else if (option_id = Curl.slist_options[option])?
|
||||||
@setopt_slist_ option_id, value
|
@setopt_slist_ option_id, value
|
||||||
else if (option_id = Curl.integer_options[option])?
|
else if (option_id = Curl.integer_options[option])?
|
||||||
|
@ -41,6 +66,7 @@ Curl::getinfo = (oinfo) ->
|
||||||
Curl.user_options =
|
Curl.user_options =
|
||||||
RAW: 'RAW'
|
RAW: 'RAW'
|
||||||
DEBUG: 'DEBUG'
|
DEBUG: 'DEBUG'
|
||||||
|
MULTIPART: 'MULTIPART'
|
||||||
|
|
||||||
id = 0
|
id = 0
|
||||||
curls = {}
|
curls = {}
|
||||||
|
|
|
@ -1,17 +1,17 @@
|
||||||
// Generated by ToffeeScript 1.4.0
|
// Generated by ToffeeScript 1.6.2
|
||||||
(function() {
|
(function() {
|
||||||
var Curl, CurlBuilder,
|
var Curl, CurlBuilder, e,
|
||||||
__hasProp = {}.hasOwnProperty,
|
__hasProp = {}.hasOwnProperty,
|
||||||
__slice = [].slice;
|
__slice = [].slice;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
Curl = require(__dirname + '/Curl');
|
Curl = require(__dirname + '/Curl');
|
||||||
} catch (e) {
|
} catch (_error) {
|
||||||
|
e = _error;
|
||||||
Curl = require(__dirname + '/Curl');
|
Curl = require(__dirname + '/Curl');
|
||||||
}
|
}
|
||||||
|
|
||||||
CurlBuilder = (function() {
|
CurlBuilder = (function() {
|
||||||
|
|
||||||
function CurlBuilder() {}
|
function CurlBuilder() {}
|
||||||
|
|
||||||
CurlBuilder.curls = {};
|
CurlBuilder.curls = {};
|
||||||
|
@ -95,7 +95,7 @@
|
||||||
curl.body = data.toString();
|
curl.body = data.toString();
|
||||||
}
|
}
|
||||||
curl.status = curl.code = c.getinfo('RESPONSE_CODE');
|
curl.status = curl.code = c.getinfo('RESPONSE_CODE');
|
||||||
return process.nextTick(function() {
|
process.nextTick(function() {
|
||||||
return cb.call(curl, null, curl);
|
return cb.call(curl, null, curl);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@ -103,7 +103,7 @@
|
||||||
var _this = this;
|
var _this = this;
|
||||||
curl.log("receive failed: " + err.message);
|
curl.log("receive failed: " + err.message);
|
||||||
curl.running = false;
|
curl.running = false;
|
||||||
return process.nextTick(function() {
|
process.nextTick(function() {
|
||||||
return cb.call(curl, err, null);
|
return cb.call(curl, err, null);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
112
src/node-curl.h
112
src/node-curl.h
|
@ -8,9 +8,84 @@
|
||||||
#include <map>
|
#include <map>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
#include <string>
|
#include <string>
|
||||||
|
#include <stdlib.h>
|
||||||
|
|
||||||
#define NODE_CURL_EXPORT(name) export_curl_options(t, #name, name, sizeof(name) / sizeof(CurlOption));
|
#define NODE_CURL_EXPORT(name) export_curl_options(t, #name, name, sizeof(name) / sizeof(CurlOption));
|
||||||
|
|
||||||
|
class NodeCurlHttppost
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
curl_httppost *first;
|
||||||
|
curl_httppost *last;
|
||||||
|
|
||||||
|
public:
|
||||||
|
NodeCurlHttppost()
|
||||||
|
: first(NULL), last(NULL)
|
||||||
|
{
|
||||||
|
reset();
|
||||||
|
}
|
||||||
|
|
||||||
|
void reset()
|
||||||
|
{
|
||||||
|
curl_httppost *cur = first;
|
||||||
|
while (cur) {
|
||||||
|
curl_httppost *next = cur->next;
|
||||||
|
if (cur->contenttype)
|
||||||
|
free(cur->contenttype);
|
||||||
|
if (cur->contents && cur->flags & HTTPPOST_FILENAME)
|
||||||
|
free(cur->contents);
|
||||||
|
if (cur->buffer)
|
||||||
|
free(cur->buffer);
|
||||||
|
if (cur->name)
|
||||||
|
free(cur->buffer);
|
||||||
|
free(cur);
|
||||||
|
cur = next;
|
||||||
|
}
|
||||||
|
first = NULL;
|
||||||
|
last = NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
void append()
|
||||||
|
{
|
||||||
|
if (!first) {
|
||||||
|
first = (curl_httppost*)calloc(1, sizeof(curl_httppost));
|
||||||
|
last = first;
|
||||||
|
} else {
|
||||||
|
last->next = (curl_httppost*)calloc(1, sizeof(curl_httppost));
|
||||||
|
last = last->next;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
enum {
|
||||||
|
NAME,
|
||||||
|
FILE,
|
||||||
|
VALUE,
|
||||||
|
TYPE
|
||||||
|
};
|
||||||
|
|
||||||
|
void set(int field, char *value, long length)
|
||||||
|
{
|
||||||
|
switch (field) {
|
||||||
|
case NAME:
|
||||||
|
value = strndup(value, length);
|
||||||
|
last->name = value;
|
||||||
|
last->namelength = length;
|
||||||
|
break;
|
||||||
|
case TYPE:
|
||||||
|
value = strndup(value, length);
|
||||||
|
last->contenttype = value;
|
||||||
|
break;
|
||||||
|
case FILE:
|
||||||
|
value = strndup(value, length);
|
||||||
|
last->flags |= HTTPPOST_FILENAME;
|
||||||
|
case VALUE:
|
||||||
|
last->contents = value;
|
||||||
|
last->contentslength = length;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
class NodeCurl
|
class NodeCurl
|
||||||
{
|
{
|
||||||
struct CurlOption
|
struct CurlOption
|
||||||
|
@ -32,6 +107,7 @@ class NodeCurl
|
||||||
bool in_curlm;
|
bool in_curlm;
|
||||||
std::vector<curl_slist*> slists;
|
std::vector<curl_slist*> slists;
|
||||||
std::map<int, std::string> strings;
|
std::map<int, std::string> strings;
|
||||||
|
NodeCurlHttppost httppost;
|
||||||
|
|
||||||
NodeCurl(v8::Handle<v8::Object> object)
|
NodeCurl(v8::Handle<v8::Object> object)
|
||||||
: in_curlm(false)
|
: in_curlm(false)
|
||||||
|
@ -242,6 +318,30 @@ class NodeCurl
|
||||||
return node_curl->setopt(args[0], slist);
|
return node_curl->setopt(args[0], slist);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static v8::Handle<v8::Value> setopt_httppost(const v8::Arguments & args)
|
||||||
|
{
|
||||||
|
NodeCurl * node_curl = unwrap(args.This());
|
||||||
|
NodeCurlHttppost &httppost = node_curl->httppost;
|
||||||
|
v8::Handle<v8::Array> rows = v8::Handle<v8::Array>::Cast(args[0]);
|
||||||
|
httppost.reset();
|
||||||
|
for (uint32_t i=0, len = rows->Length(); i<len; ++i)
|
||||||
|
{
|
||||||
|
v8::Handle<v8::Array> cols = v8::Handle<v8::Array>::Cast(rows->Get(i));
|
||||||
|
uint32_t j=0, cols_len = cols->Length();
|
||||||
|
httppost.append();
|
||||||
|
while (j<cols_len)
|
||||||
|
{
|
||||||
|
int field = cols->Get(j++)->Int32Value();
|
||||||
|
v8::Handle<v8::Object> buffer = cols->Get(j++)->ToObject();
|
||||||
|
char *value = node::Buffer::Data(buffer);
|
||||||
|
int length = node::Buffer::Length(buffer);
|
||||||
|
httppost.set(field, value, length);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
curl_easy_setopt(node_curl->curl, CURLOPT_HTTPPOST, node_curl->httppost.first);
|
||||||
|
return args.This();
|
||||||
|
}
|
||||||
|
|
||||||
static curl_slist * value_to_slist(v8::Handle<v8::Value> value)
|
static curl_slist * value_to_slist(v8::Handle<v8::Value> value)
|
||||||
{
|
{
|
||||||
curl_slist * slist = NULL;
|
curl_slist * slist = NULL;
|
||||||
|
@ -389,6 +489,7 @@ class NodeCurl
|
||||||
NODE_SET_PROTOTYPE_METHOD(t , "setopt_int_" , setopt_int);
|
NODE_SET_PROTOTYPE_METHOD(t , "setopt_int_" , setopt_int);
|
||||||
NODE_SET_PROTOTYPE_METHOD(t , "setopt_str_" , setopt_str);
|
NODE_SET_PROTOTYPE_METHOD(t , "setopt_str_" , setopt_str);
|
||||||
NODE_SET_PROTOTYPE_METHOD(t , "setopt_slist_" , setopt_slist);
|
NODE_SET_PROTOTYPE_METHOD(t , "setopt_slist_" , setopt_slist);
|
||||||
|
NODE_SET_PROTOTYPE_METHOD(t , "setopt_httppost_" , setopt_httppost);
|
||||||
|
|
||||||
NODE_SET_PROTOTYPE_METHOD(t , "getinfo_int_" , getinfo_int);
|
NODE_SET_PROTOTYPE_METHOD(t , "getinfo_int_" , getinfo_int);
|
||||||
NODE_SET_PROTOTYPE_METHOD(t , "getinfo_str_" , getinfo_str);
|
NODE_SET_PROTOTYPE_METHOD(t , "getinfo_str_" , getinfo_str);
|
||||||
|
@ -434,6 +535,15 @@ class NodeCurl
|
||||||
X(SSL_ENGINES),
|
X(SSL_ENGINES),
|
||||||
X(COOKIELIST)
|
X(COOKIELIST)
|
||||||
};
|
};
|
||||||
|
|
||||||
|
#undef X
|
||||||
|
#define X(name) {#name, NodeCurlHttppost::name}
|
||||||
|
CurlOption httppost_options[] = {
|
||||||
|
X(NAME),
|
||||||
|
X(FILE),
|
||||||
|
X(VALUE),
|
||||||
|
X(TYPE)
|
||||||
|
};
|
||||||
#undef X
|
#undef X
|
||||||
|
|
||||||
NODE_CURL_EXPORT(string_options);
|
NODE_CURL_EXPORT(string_options);
|
||||||
|
@ -445,6 +555,8 @@ class NodeCurl
|
||||||
NODE_CURL_EXPORT(double_infos);
|
NODE_CURL_EXPORT(double_infos);
|
||||||
NODE_CURL_EXPORT(slist_infos);
|
NODE_CURL_EXPORT(slist_infos);
|
||||||
|
|
||||||
|
NODE_CURL_EXPORT(httppost_options);
|
||||||
|
|
||||||
target->Set(v8::String::NewSymbol("Curl"), t->GetFunction());
|
target->Set(v8::String::NewSymbol("Curl"), t->GetFunction());
|
||||||
return target;
|
return target;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue