#include
#include
#include
#include
#include
#include /* strcasecmp() */
#include /* strdup() */
#include /* free() */
// This is a binding to http_parser (http://github.com/ry/http-parser)
// The goal is to decouple sockets from parsing for more javascript-level
// agility. A Buffer is read from a socket and passed to parser.execute().
// The parser then issues callbacks with slices of the data
// parser.onMessageBegin
// parser.onPath
// parser.onBody
// ...
// No copying is performed when slicing the buffer, only small reference
// allocations.
namespace node {
using namespace v8;
static Persistent on_message_begin_sym;
static Persistent on_path_sym;
static Persistent on_query_string_sym;
static Persistent on_url_sym;
static Persistent on_fragment_sym;
static Persistent on_header_field_sym;
static Persistent on_header_value_sym;
static Persistent on_headers_complete_sym;
static Persistent on_body_sym;
static Persistent on_message_complete_sym;
static Persistent delete_sym;
static Persistent get_sym;
static Persistent head_sym;
static Persistent post_sym;
static Persistent put_sym;
static Persistent connect_sym;
static Persistent options_sym;
static Persistent trace_sym;
static Persistent copy_sym;
static Persistent lock_sym;
static Persistent mkcol_sym;
static Persistent move_sym;
static Persistent propfind_sym;
static Persistent proppatch_sym;
static Persistent unlock_sym;
static Persistent report_sym;
static Persistent mkactivity_sym;
static Persistent checkout_sym;
static Persistent merge_sym;
static Persistent msearch_sym;
static Persistent notify_sym;
static Persistent subscribe_sym;
static Persistent unsubscribe_sym;
static Persistent unknown_method_sym;
static Persistent method_sym;
static Persistent status_code_sym;
static Persistent http_version_sym;
static Persistent version_major_sym;
static Persistent version_minor_sym;
static Persistent should_keep_alive_sym;
static Persistent upgrade_sym;
static struct http_parser_settings settings;
// This is a hack to get the current_buffer to the callbacks with the least
// amount of overhead. Nothing else will run while http_parser_execute()
// runs, therefore this pointer can be set and used for the execution.
static Local* current_buffer;
static char* current_buffer_data;
static size_t current_buffer_len;
// Callback prototype for http_cb
#define DEFINE_HTTP_CB(name) \
static int name(http_parser *p) { \
Parser *parser = static_cast(p->data); \
Local cb_value = parser->handle_->Get(name##_sym); \
if (!cb_value->IsFunction()) return 0; \
Local cb = Local::Cast(cb_value); \
Local ret = cb->Call(parser->handle_, 0, NULL); \
if (ret.IsEmpty()) { \
parser->got_exception_ = true; \
return -1; \
} else { \
return 0; \
} \
}
// Callback prototype for http_data_cb
#define DEFINE_HTTP_DATA_CB(name) \
static int name(http_parser *p, const char *at, size_t length) { \
Parser *parser = static_cast(p->data); \
assert(current_buffer); \
Local cb_value = parser->handle_->Get(name##_sym); \
if (!cb_value->IsFunction()) return 0; \
Local cb = Local::Cast(cb_value); \
Local argv[3] = { *current_buffer \
, Integer::New(at - current_buffer_data) \
, Integer::New(length) \
}; \
Local ret = cb->Call(parser->handle_, 3, argv); \
assert(current_buffer); \
if (ret.IsEmpty()) { \
parser->got_exception_ = true; \
return -1; \
} else { \
return 0; \
} \
}
static inline Persistent
method_to_str(unsigned short m) {
switch (m) {
case HTTP_DELETE: return delete_sym;
case HTTP_GET: return get_sym;
case HTTP_HEAD: return head_sym;
case HTTP_POST: return post_sym;
case HTTP_PUT: return put_sym;
case HTTP_CONNECT: return connect_sym;
case HTTP_OPTIONS: return options_sym;
case HTTP_TRACE: return trace_sym;
case HTTP_COPY: return copy_sym;
case HTTP_LOCK: return lock_sym;
case HTTP_MKCOL: return mkcol_sym;
case HTTP_MOVE: return move_sym;
case HTTP_PROPFIND: return propfind_sym;
case HTTP_PROPPATCH: return proppatch_sym;
case HTTP_UNLOCK: return unlock_sym;
case HTTP_REPORT: return report_sym;
case HTTP_MKACTIVITY: return mkactivity_sym;
case HTTP_CHECKOUT: return checkout_sym;
case HTTP_MERGE: return merge_sym;
case HTTP_MSEARCH: return msearch_sym;
case HTTP_NOTIFY: return notify_sym;
case HTTP_SUBSCRIBE: return subscribe_sym;
case HTTP_UNSUBSCRIBE:return unsubscribe_sym;
default: return unknown_method_sym;
}
}
class Parser : public ObjectWrap {
public:
Parser(enum http_parser_type type) : ObjectWrap() {
Init(type);
}
~Parser() {
}
DEFINE_HTTP_CB(on_message_begin)
DEFINE_HTTP_CB(on_message_complete)
DEFINE_HTTP_DATA_CB(on_path)
DEFINE_HTTP_DATA_CB(on_url)
DEFINE_HTTP_DATA_CB(on_fragment)
DEFINE_HTTP_DATA_CB(on_query_string)
DEFINE_HTTP_DATA_CB(on_header_field)
DEFINE_HTTP_DATA_CB(on_header_value)
DEFINE_HTTP_DATA_CB(on_body)
static int on_headers_complete(http_parser *p) {
Parser *parser = static_cast(p->data);
Local cb_value = parser->handle_->Get(on_headers_complete_sym);
if (!cb_value->IsFunction()) return 0;
Local cb = Local::Cast(cb_value);
Local