2009-05-18 12:44:01 +02:00
|
|
|
(function () {
|
2009-07-14 11:59:13 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Inherit the prototype methods from one constructor into another.
|
|
|
|
*
|
|
|
|
* The Function.prototype.inherits from lang.js rewritten as a standalone
|
|
|
|
* function (not on Function.prototype). NOTE: If this file is to be loaded
|
|
|
|
* during bootstrapping this function needs to be revritten using some native
|
|
|
|
* functions as prototype setup using normal JavaScript does not work as
|
|
|
|
* expected during bootstrapping (see mirror.js in r114903).
|
|
|
|
*
|
|
|
|
* @param {function} ctor Constructor function which needs to inherit the
|
|
|
|
* prototype
|
|
|
|
* @param {function} superCtor Constructor function to inherit prototype from
|
|
|
|
*/
|
|
|
|
function inherits(ctor, superCtor) {
|
|
|
|
var tempCtor = function(){};
|
|
|
|
tempCtor.prototype = superCtor.prototype;
|
|
|
|
ctor.super_ = superCtor.prototype;
|
|
|
|
ctor.prototype = new tempCtor();
|
|
|
|
ctor.prototype.constructor = ctor;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-05-19 13:12:46 +02:00
|
|
|
CRLF = "\r\n";
|
2009-07-14 11:59:13 +02:00
|
|
|
node.http.STATUS_CODES = {
|
2009-07-01 00:49:56 +02:00
|
|
|
100 : 'Continue',
|
|
|
|
101 : 'Switching Protocols',
|
|
|
|
200 : 'OK',
|
|
|
|
201 : 'Created',
|
|
|
|
202 : 'Accepted',
|
|
|
|
203 : 'Non-Authoritative Information',
|
|
|
|
204 : 'No Content',
|
|
|
|
205 : 'Reset Content',
|
|
|
|
206 : 'Partial Content',
|
|
|
|
300 : 'Multiple Choices',
|
|
|
|
301 : 'Moved Permanently',
|
|
|
|
302 : 'Moved Temporarily',
|
|
|
|
303 : 'See Other',
|
|
|
|
304 : 'Not Modified',
|
|
|
|
305 : 'Use Proxy',
|
|
|
|
400 : 'Bad Request',
|
|
|
|
401 : 'Unauthorized',
|
|
|
|
402 : 'Payment Required',
|
|
|
|
403 : 'Forbidden',
|
|
|
|
404 : 'Not Found',
|
|
|
|
405 : 'Method Not Allowed',
|
|
|
|
406 : 'Not Acceptable',
|
|
|
|
407 : 'Proxy Authentication Required',
|
|
|
|
408 : 'Request Time-out',
|
|
|
|
409 : 'Conflict',
|
|
|
|
410 : 'Gone',
|
|
|
|
411 : 'Length Required',
|
|
|
|
412 : 'Precondition Failed',
|
|
|
|
413 : 'Request Entity Too Large',
|
|
|
|
414 : 'Request-URI Too Large',
|
|
|
|
415 : 'Unsupported Media Type',
|
|
|
|
500 : 'Internal Server Error',
|
|
|
|
501 : 'Not Implemented',
|
|
|
|
502 : 'Bad Gateway',
|
|
|
|
503 : 'Service Unavailable',
|
|
|
|
504 : 'Gateway Time-out',
|
|
|
|
505 : 'HTTP Version not supported'
|
|
|
|
};
|
2009-05-11 19:08:29 +02:00
|
|
|
|
2009-05-18 19:33:05 +02:00
|
|
|
/*
|
|
|
|
parseUri 1.2.1
|
|
|
|
(c) 2007 Steven Levithan <stevenlevithan.com>
|
|
|
|
MIT License
|
|
|
|
*/
|
|
|
|
|
2009-06-01 20:59:33 +02:00
|
|
|
function decode (s) {
|
|
|
|
return decodeURIComponent(s.replace(/\+/g, ' '));
|
|
|
|
}
|
|
|
|
|
2009-05-19 00:01:11 +02:00
|
|
|
node.http.parseUri = function (str) {
|
|
|
|
var o = node.http.parseUri.options,
|
2009-06-22 13:12:47 +02:00
|
|
|
m = o.parser[o.strictMode ? "strict" : "loose"].exec(str),
|
|
|
|
uri = {},
|
|
|
|
i = 14;
|
2009-05-18 19:33:05 +02:00
|
|
|
|
|
|
|
while (i--) uri[o.key[i]] = m[i] || "";
|
|
|
|
|
|
|
|
uri[o.q.name] = {};
|
|
|
|
uri[o.key[12]].replace(o.q.parser, function ($0, $1, $2) {
|
2009-06-01 20:59:33 +02:00
|
|
|
if ($1) {
|
|
|
|
var key = decode($1);
|
|
|
|
var val = decode($2);
|
|
|
|
uri[o.q.name][key] = val;
|
|
|
|
}
|
2009-05-18 19:33:05 +02:00
|
|
|
});
|
|
|
|
uri.toString = function () { return str; };
|
2009-06-21 16:28:23 +02:00
|
|
|
|
2009-07-01 00:49:56 +02:00
|
|
|
for (i = o.key.length - 1; i >= 0; i--){
|
2009-06-21 16:28:23 +02:00
|
|
|
if (uri[o.key[i]] == "") delete uri[o.key[i]];
|
2009-07-01 00:49:56 +02:00
|
|
|
}
|
2009-06-21 16:28:23 +02:00
|
|
|
|
2009-05-18 19:33:05 +02:00
|
|
|
return uri;
|
|
|
|
};
|
|
|
|
|
2009-05-19 00:01:11 +02:00
|
|
|
node.http.parseUri.options = {
|
2009-05-18 19:33:05 +02:00
|
|
|
strictMode: false,
|
2009-07-01 00:49:56 +02:00
|
|
|
key: [
|
|
|
|
"source",
|
|
|
|
"protocol",
|
|
|
|
"authority",
|
|
|
|
"userInfo",
|
|
|
|
"user",
|
|
|
|
"password",
|
|
|
|
"host",
|
|
|
|
"port",
|
|
|
|
"relative",
|
|
|
|
"path",
|
|
|
|
"directory",
|
|
|
|
"file",
|
|
|
|
"query",
|
|
|
|
"anchor"
|
|
|
|
],
|
2009-05-18 19:33:05 +02:00
|
|
|
q: {
|
2009-06-01 20:59:33 +02:00
|
|
|
name: "params",
|
2009-05-18 19:33:05 +02:00
|
|
|
parser: /(?:^|&)([^&=]*)=?([^&]*)/g
|
|
|
|
},
|
|
|
|
parser: {
|
|
|
|
strict: /^(?:([^:\/?#]+):)?(?:\/\/((?:(([^:@]*):?([^:@]*))?@)?([^:\/?#]*)(?::(\d*))?))?((((?:[^?#\/]*\/)*)([^?#]*))(?:\?([^#]*))?(?:#(.*))?)/,
|
|
|
|
loose: /^(?:(?![^:@]+:[^:@\/]*@)([^:\/?#.]+):)?(?:\/\/)?((?:(([^:@]*):?([^:@]*))?@)?([^:\/?#]*)(?::(\d*))?)(((\/(?:[^?#](?![^?#\/]*\.[^?#\/.]+(?:[?#]|$)))*\/?)?([^?#\/]*))(?:\?([^#]*))?(?:#(.*))?)/
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2009-07-16 10:59:40 +02:00
|
|
|
|
2009-05-11 18:54:52 +02:00
|
|
|
var connection_expression = /Connection/i;
|
|
|
|
var transfer_encoding_expression = /Transfer-Encoding/i;
|
|
|
|
var close_expression = /close/i;
|
|
|
|
var chunk_expression = /chunk/i;
|
|
|
|
var content_length_expression = /Content-Length/i;
|
|
|
|
|
2009-06-26 18:29:57 +02:00
|
|
|
|
2009-07-14 11:59:13 +02:00
|
|
|
/* Abstract base class for ServerRequest and ClientResponse. */
|
2009-07-16 10:59:40 +02:00
|
|
|
function IncomingMessage (connection) {
|
2009-07-14 11:59:13 +02:00
|
|
|
node.EventEmitter.call(this);
|
2009-06-26 18:29:57 +02:00
|
|
|
|
2009-07-14 11:59:13 +02:00
|
|
|
this.connection = connection;
|
|
|
|
this.httpVersion = null;
|
|
|
|
this.headers = [];
|
2009-07-14 18:31:50 +02:00
|
|
|
|
|
|
|
// request (server) only
|
|
|
|
this.uri = "";
|
|
|
|
this.method = null;
|
|
|
|
|
|
|
|
// response (client) only
|
|
|
|
this.statusCode = null;
|
|
|
|
this.client = this.connection;
|
2009-07-16 10:59:40 +02:00
|
|
|
}
|
2009-07-14 11:59:13 +02:00
|
|
|
inherits(IncomingMessage, node.EventEmitter);
|
|
|
|
|
|
|
|
IncomingMessage.prototype.setBodyEncoding = function (enc) {
|
|
|
|
// TODO: Find a cleaner way of doing this.
|
|
|
|
this.connection.setEncoding(enc);
|
2009-06-26 18:29:57 +02:00
|
|
|
};
|
|
|
|
|
2009-07-12 11:48:37 +02:00
|
|
|
|
2009-07-16 10:59:40 +02:00
|
|
|
function OutgoingMessage () {
|
2009-07-14 18:31:50 +02:00
|
|
|
node.EventEmitter.call(this);
|
2009-07-14 11:59:13 +02:00
|
|
|
|
2009-07-14 18:31:50 +02:00
|
|
|
this.output = [];
|
2009-06-26 18:29:57 +02:00
|
|
|
|
2009-07-14 18:31:50 +02:00
|
|
|
this.sent_connection_header = false;
|
|
|
|
this.sent_content_length_header = false;
|
|
|
|
this.sent_transfer_encoding_header = false;
|
2009-06-26 18:29:57 +02:00
|
|
|
|
2009-07-14 18:31:50 +02:00
|
|
|
this.closeOnFinish = false;
|
|
|
|
this.chunked_encoding = false;
|
|
|
|
this.should_keep_alive = true;
|
|
|
|
this.use_chunked_encoding_by_default = true;
|
2009-07-14 11:59:13 +02:00
|
|
|
|
2009-07-14 18:31:50 +02:00
|
|
|
this.finished = false;
|
2009-07-16 10:59:40 +02:00
|
|
|
}
|
2009-07-14 18:31:50 +02:00
|
|
|
inherits(OutgoingMessage, node.EventEmitter);
|
2009-07-14 11:59:13 +02:00
|
|
|
|
2009-07-14 18:31:50 +02:00
|
|
|
OutgoingMessage.prototype.send = function (data, encoding) {
|
2009-07-16 10:59:40 +02:00
|
|
|
data.encoding = data.constructor === String ? encoding || "ascii" : "raw";
|
|
|
|
this.output.push(data);
|
2009-07-14 18:31:50 +02:00
|
|
|
};
|
2009-07-14 11:59:13 +02:00
|
|
|
|
2009-07-14 18:31:50 +02:00
|
|
|
OutgoingMessage.prototype.sendHeaderLines = function (first_line, header_lines) {
|
|
|
|
header_lines = header_lines || [];
|
2009-07-14 11:59:13 +02:00
|
|
|
|
2009-07-14 18:31:50 +02:00
|
|
|
// first_line in the case of request is: "GET /index.html HTTP/1.1\r\n"
|
|
|
|
// in the case of response it is: "HTTP/1.1 200 OK\r\n"
|
|
|
|
var header = first_line;
|
2009-07-14 11:59:13 +02:00
|
|
|
|
2009-07-14 18:31:50 +02:00
|
|
|
for (var i = 0; i < header_lines.length; i++) {
|
|
|
|
var field = header_lines[i][0];
|
|
|
|
var value = header_lines[i][1];
|
2009-06-26 18:29:57 +02:00
|
|
|
|
2009-07-14 18:31:50 +02:00
|
|
|
header += field + ": " + value + CRLF;
|
|
|
|
|
|
|
|
if (connection_expression.exec(field)) {
|
|
|
|
this.sent_connection_header = true;
|
|
|
|
if (close_expression.exec(value)) this.closeOnFinish = true;
|
2009-06-26 18:29:57 +02:00
|
|
|
|
2009-07-14 18:31:50 +02:00
|
|
|
} else if (transfer_encoding_expression.exec(field)) {
|
|
|
|
this.sent_transfer_encoding_header = true;
|
|
|
|
if (chunk_expression.exec(value)) this.chunked_encoding = true;
|
|
|
|
|
|
|
|
} else if (content_length_expression.exec(field)) {
|
|
|
|
this.sent_content_length_header = true;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// keep-alive logic
|
|
|
|
if (this.sent_connection_header == false) {
|
|
|
|
if (this.should_keep_alive) {
|
|
|
|
header += "Connection: keep-alive\r\n";
|
2009-06-26 18:29:57 +02:00
|
|
|
} else {
|
2009-07-14 18:31:50 +02:00
|
|
|
this.closeOnFinish = true;
|
|
|
|
header += "Connection: close\r\n";
|
2009-06-26 18:29:57 +02:00
|
|
|
}
|
2009-07-14 18:31:50 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (this.sent_content_length_header == false && this.sent_transfer_encoding_header == false) {
|
|
|
|
if (this.use_chunked_encoding_by_default) {
|
|
|
|
header += "Transfer-Encoding: chunked\r\n";
|
|
|
|
this.chunked_encoding = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
header += CRLF;
|
|
|
|
|
|
|
|
this.send(header);
|
2009-07-16 10:59:40 +02:00
|
|
|
// wait until the first body chunk, or finish(), is sent to flush.
|
2009-07-14 18:31:50 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
OutgoingMessage.prototype.sendBody = function (chunk, encoding) {
|
|
|
|
if (this.chunked_encoding) {
|
|
|
|
this.send(chunk.length.toString(16));
|
|
|
|
this.send(CRLF);
|
|
|
|
this.send(chunk, encoding);
|
|
|
|
this.send(CRLF);
|
|
|
|
} else {
|
|
|
|
this.send(chunk, encoding);
|
|
|
|
}
|
2009-06-26 18:29:57 +02:00
|
|
|
|
2009-07-14 18:31:50 +02:00
|
|
|
this.flush();
|
|
|
|
};
|
|
|
|
|
|
|
|
OutgoingMessage.prototype.flush = function () {
|
|
|
|
this.emit("flush");
|
|
|
|
};
|
|
|
|
|
|
|
|
OutgoingMessage.prototype.finish = function () {
|
|
|
|
if (this.chunked_encoding) this.send("0\r\n\r\n"); // last chunk
|
|
|
|
this.finished = true;
|
|
|
|
this.flush();
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2009-07-16 10:59:40 +02:00
|
|
|
function ServerResponse () {
|
2009-07-14 18:31:50 +02:00
|
|
|
OutgoingMessage.call(this);
|
|
|
|
|
|
|
|
this.should_keep_alive = true;
|
|
|
|
this.use_chunked_encoding_by_default = true;
|
2009-07-16 10:59:40 +02:00
|
|
|
}
|
2009-07-14 18:31:50 +02:00
|
|
|
inherits(ServerResponse, OutgoingMessage);
|
|
|
|
|
|
|
|
ServerResponse.prototype.sendHeader = function (statusCode, headers) {
|
|
|
|
var reason = node.http.STATUS_CODES[statusCode] || "unknown";
|
|
|
|
var status_line = "HTTP/1.1 " + statusCode.toString() + " " + reason + CRLF;
|
|
|
|
this.sendHeaderLines(status_line, headers);
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2009-07-16 10:59:40 +02:00
|
|
|
function ClientRequest (method, uri, header_lines) {
|
2009-07-14 18:31:50 +02:00
|
|
|
OutgoingMessage.call(this);
|
|
|
|
|
|
|
|
this.should_keep_alive = false;
|
|
|
|
this.use_chunked_encoding_by_default = false;
|
|
|
|
this.closeOnFinish = true;
|
|
|
|
|
|
|
|
this.sendHeaderLines(method + " " + uri + " HTTP/1.1\r\n", header_lines);
|
2009-07-16 10:59:40 +02:00
|
|
|
}
|
2009-07-14 18:31:50 +02:00
|
|
|
inherits(ClientRequest, OutgoingMessage);
|
|
|
|
|
|
|
|
ClientRequest.prototype.finish = function (responseListener) {
|
|
|
|
this.addListener("response", responseListener);
|
|
|
|
OutgoingMessage.prototype.finish.call(this);
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
function createIncomingMessageStream (connection, incoming_listener) {
|
|
|
|
var stream = new node.EventEmitter();
|
|
|
|
|
|
|
|
stream.addListener("incoming", incoming_listener);
|
|
|
|
|
|
|
|
var incoming;
|
|
|
|
var last_header_was_a_value = false;
|
2009-06-26 18:29:57 +02:00
|
|
|
|
2009-06-29 13:18:30 +02:00
|
|
|
connection.addListener("message_begin", function () {
|
2009-07-14 18:31:50 +02:00
|
|
|
incoming = new IncomingMessage(connection);
|
2009-06-26 18:29:57 +02:00
|
|
|
});
|
|
|
|
|
2009-07-14 18:31:50 +02:00
|
|
|
// Only servers will get URI events.
|
2009-06-29 13:18:30 +02:00
|
|
|
connection.addListener("uri", function (data) {
|
2009-07-14 18:31:50 +02:00
|
|
|
incoming.uri += data;
|
2009-06-26 18:29:57 +02:00
|
|
|
});
|
|
|
|
|
2009-06-29 13:18:30 +02:00
|
|
|
connection.addListener("header_field", function (data) {
|
2009-07-14 18:31:50 +02:00
|
|
|
if (incoming.headers.length > 0 && last_header_was_a_value == false) {
|
|
|
|
incoming.headers[incoming.headers.length-1][0] += data;
|
|
|
|
} else {
|
|
|
|
incoming.headers.push([data]);
|
|
|
|
}
|
|
|
|
last_header_was_a_value = false;
|
2009-06-26 18:29:57 +02:00
|
|
|
});
|
|
|
|
|
2009-06-29 13:18:30 +02:00
|
|
|
connection.addListener("header_value", function (data) {
|
2009-07-14 18:31:50 +02:00
|
|
|
var last_pair = incoming.headers[incoming.headers.length-1];
|
|
|
|
if (last_pair.length == 1) {
|
2009-06-26 18:29:57 +02:00
|
|
|
last_pair[1] = data;
|
2009-07-14 18:31:50 +02:00
|
|
|
} else {
|
2009-06-26 18:29:57 +02:00
|
|
|
last_pair[1] += data;
|
2009-07-14 18:31:50 +02:00
|
|
|
}
|
|
|
|
last_header_was_a_value = true;
|
2009-06-26 18:29:57 +02:00
|
|
|
});
|
|
|
|
|
2009-06-29 13:18:30 +02:00
|
|
|
connection.addListener("headers_complete", function (info) {
|
2009-07-14 18:31:50 +02:00
|
|
|
incoming.httpVersion = info.httpVersion;
|
2009-06-26 18:29:57 +02:00
|
|
|
|
2009-07-14 18:31:50 +02:00
|
|
|
if (info.method) {
|
|
|
|
// server only
|
|
|
|
incoming.method = info.method;
|
|
|
|
incoming.uri = node.http.parseUri(incoming.uri); // TODO parse the URI lazily?
|
|
|
|
} else {
|
|
|
|
// client only
|
|
|
|
incoming.statusCode = info.statusCode;
|
|
|
|
}
|
2009-06-26 18:29:57 +02:00
|
|
|
|
2009-07-14 18:31:50 +02:00
|
|
|
stream.emit("incoming", [incoming, info.should_keep_alive]);
|
2009-06-26 18:29:57 +02:00
|
|
|
});
|
|
|
|
|
2009-06-29 13:18:30 +02:00
|
|
|
connection.addListener("body", function (chunk) {
|
2009-07-14 18:31:50 +02:00
|
|
|
incoming.emit("body", [chunk]);
|
2009-06-26 18:29:57 +02:00
|
|
|
});
|
|
|
|
|
2009-06-29 13:18:30 +02:00
|
|
|
connection.addListener("message_complete", function () {
|
2009-07-14 18:31:50 +02:00
|
|
|
incoming.emit("complete");
|
2009-06-26 18:29:57 +02:00
|
|
|
});
|
2009-07-14 18:31:50 +02:00
|
|
|
|
|
|
|
return stream;
|
2009-07-01 00:49:56 +02:00
|
|
|
}
|
2009-06-26 18:29:57 +02:00
|
|
|
|
2009-07-14 18:31:50 +02:00
|
|
|
/* Returns true if the message queue is finished and the connection
|
|
|
|
* should be closed. */
|
|
|
|
function flushMessageQueue (connection, queue) {
|
|
|
|
if (connection.readyState === "closed" || connection.readyState === "readOnly") {
|
|
|
|
return false;
|
|
|
|
}
|
2009-05-11 18:54:52 +02:00
|
|
|
|
2009-07-14 18:31:50 +02:00
|
|
|
while (queue[0]) {
|
|
|
|
var message = queue[0];
|
2009-05-18 19:33:05 +02:00
|
|
|
|
2009-07-14 18:31:50 +02:00
|
|
|
while (message.output.length > 0) {
|
|
|
|
var out = message.output.shift();
|
2009-07-16 10:59:40 +02:00
|
|
|
connection.send(out, out.encoding);
|
2009-05-18 19:33:05 +02:00
|
|
|
}
|
2009-07-14 18:31:50 +02:00
|
|
|
|
|
|
|
if (!message.finished) break;
|
2009-05-18 19:33:05 +02:00
|
|
|
|
2009-07-14 18:31:50 +02:00
|
|
|
message.emit("sent");
|
|
|
|
queue.shift();
|
2009-05-18 19:33:05 +02:00
|
|
|
|
2009-07-14 18:31:50 +02:00
|
|
|
if (message.closeOnFinish) return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2009-05-18 19:33:05 +02:00
|
|
|
|
2009-07-16 10:59:40 +02:00
|
|
|
|
2009-07-14 18:31:50 +02:00
|
|
|
node.http.createServer = function (requestListener, options) {
|
2009-07-16 10:59:40 +02:00
|
|
|
var server = new node.http.Server();
|
2009-07-14 18:31:50 +02:00
|
|
|
//server.setOptions(options);
|
|
|
|
server.addListener("request", requestListener);
|
|
|
|
server.addListener("connection", connectionListener);
|
|
|
|
return server;
|
|
|
|
};
|
2009-05-18 19:33:05 +02:00
|
|
|
|
2009-07-14 18:31:50 +02:00
|
|
|
function connectionListener (connection) {
|
|
|
|
// An array of responses for each connection. In pipelined connections
|
|
|
|
// we need to keep track of the order they were sent.
|
|
|
|
var responses = [];
|
2009-05-18 19:33:05 +02:00
|
|
|
|
2009-07-14 18:31:50 +02:00
|
|
|
// is this really needed?
|
|
|
|
connection.addListener("eof", function () {
|
|
|
|
if (responses.length == 0) {
|
|
|
|
connection.close();
|
|
|
|
} else {
|
|
|
|
responses[responses.length-1].closeOnFinish = true;
|
2009-07-01 00:49:56 +02:00
|
|
|
}
|
2009-07-14 18:31:50 +02:00
|
|
|
});
|
2009-05-18 19:33:05 +02:00
|
|
|
|
2009-07-14 18:31:50 +02:00
|
|
|
var flushResponse = function () {
|
|
|
|
if(flushMessageQueue(connection, responses)) {
|
|
|
|
connection.fullClose();
|
2009-05-18 19:33:05 +02:00
|
|
|
}
|
|
|
|
};
|
2009-05-13 23:35:36 +02:00
|
|
|
|
2009-07-14 18:31:50 +02:00
|
|
|
createIncomingMessageStream(connection, function (incoming, should_keep_alive) {
|
|
|
|
var req = incoming;
|
2009-06-26 12:51:27 +02:00
|
|
|
|
2009-07-14 18:31:50 +02:00
|
|
|
var res = new ServerResponse(connection);
|
|
|
|
res.should_keep_alive = should_keep_alive;
|
|
|
|
res.addListener("flush", flushResponse);
|
|
|
|
responses.push(res);
|
|
|
|
|
|
|
|
connection.server.emit("request", [req, res]);
|
|
|
|
});
|
|
|
|
}
|
2009-06-26 12:51:27 +02:00
|
|
|
|
2009-06-26 18:30:55 +02:00
|
|
|
|
2009-06-26 18:29:57 +02:00
|
|
|
node.http.createClient = function (port, host) {
|
|
|
|
var client = new node.http.Client();
|
2009-06-26 18:30:55 +02:00
|
|
|
|
2009-07-14 18:31:50 +02:00
|
|
|
var requests = [];
|
2009-06-26 12:51:27 +02:00
|
|
|
|
2009-07-14 18:31:50 +02:00
|
|
|
client._pushRequest = function (req) {
|
|
|
|
req.addListener("flush", function () {
|
|
|
|
if (client.readyState == "closed") {
|
|
|
|
//node.debug("HTTP CLIENT request flush. reconnect. readyState = " + client.readyState);
|
|
|
|
client.connect(port, host); // reconnect
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (req == requests[0]) flushMessageQueue(client, [req]);
|
|
|
|
});
|
|
|
|
requests.push(req);
|
|
|
|
};
|
2009-06-26 12:51:27 +02:00
|
|
|
|
2009-06-29 13:18:30 +02:00
|
|
|
client.addListener("connect", function () {
|
2009-06-26 18:29:57 +02:00
|
|
|
//node.debug("HTTP CLIENT onConnect. readyState = " + client.readyState);
|
2009-06-26 18:30:55 +02:00
|
|
|
//node.debug("client.requests[0].uri = '" + client.requests[0].uri + "'");
|
2009-07-14 18:31:50 +02:00
|
|
|
requests[0].flush();
|
2009-06-26 18:29:57 +02:00
|
|
|
});
|
2009-05-06 14:54:28 +02:00
|
|
|
|
2009-06-29 13:18:30 +02:00
|
|
|
client.addListener("eof", function () {
|
2009-07-14 18:31:50 +02:00
|
|
|
//node.debug("client got eof closing. readyState = " + client.readyState);
|
2009-06-26 18:29:57 +02:00
|
|
|
client.close();
|
|
|
|
});
|
2009-06-26 12:51:27 +02:00
|
|
|
|
2009-06-29 13:18:30 +02:00
|
|
|
client.addListener("disconnect", function (had_error) {
|
2009-06-26 18:29:57 +02:00
|
|
|
if (had_error) {
|
2009-06-29 13:18:30 +02:00
|
|
|
client.emit("error");
|
2009-06-26 18:29:57 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
//node.debug("HTTP CLIENT onDisconnect. readyState = " + client.readyState);
|
2009-07-14 18:31:50 +02:00
|
|
|
|
2009-06-26 18:29:57 +02:00
|
|
|
// If there are more requests to handle, reconnect.
|
2009-07-14 18:31:50 +02:00
|
|
|
if (requests.length > 0 && client.readyState != "opening") {
|
|
|
|
//node.debug("HTTP CLIENT: reconnecting readyState = " + client.readyState);
|
|
|
|
client.connect(port, host); // reconnect
|
2009-06-26 18:29:57 +02:00
|
|
|
}
|
|
|
|
});
|
2009-06-26 12:51:27 +02:00
|
|
|
|
2009-07-14 18:31:50 +02:00
|
|
|
createIncomingMessageStream(client, function (res) {
|
|
|
|
//node.debug("incoming response!");
|
2009-06-26 12:51:27 +02:00
|
|
|
|
2009-07-14 18:31:50 +02:00
|
|
|
res.addListener("complete", function ( ) {
|
|
|
|
//node.debug("request complete disconnecting. readyState = " + client.readyState);
|
|
|
|
client.close();
|
|
|
|
});
|
2009-05-15 18:11:49 +02:00
|
|
|
|
2009-07-14 18:31:50 +02:00
|
|
|
var req = requests.shift();
|
2009-06-29 13:18:30 +02:00
|
|
|
req.emit("response", [res]);
|
2009-06-26 18:29:57 +02:00
|
|
|
});
|
2009-05-15 18:11:49 +02:00
|
|
|
|
2009-06-26 18:29:57 +02:00
|
|
|
return client;
|
|
|
|
};
|
2009-05-15 18:11:49 +02:00
|
|
|
|
2009-06-26 18:29:57 +02:00
|
|
|
node.http.Client.prototype.get = function (uri, headers) {
|
2009-07-14 18:31:50 +02:00
|
|
|
var req = new ClientRequest("GET", uri, headers);
|
|
|
|
this._pushRequest(req);
|
2009-06-26 18:30:55 +02:00
|
|
|
return req;
|
2009-06-26 18:29:57 +02:00
|
|
|
};
|
2009-05-15 18:11:49 +02:00
|
|
|
|
2009-06-26 18:29:57 +02:00
|
|
|
node.http.Client.prototype.head = function (uri, headers) {
|
2009-07-14 18:31:50 +02:00
|
|
|
var req = new ClientRequest("HEAD", uri, headers);
|
|
|
|
this._pushRequest(req);
|
2009-06-26 18:30:55 +02:00
|
|
|
return req;
|
2009-06-26 18:29:57 +02:00
|
|
|
};
|
2009-05-15 18:11:49 +02:00
|
|
|
|
2009-06-26 18:29:57 +02:00
|
|
|
node.http.Client.prototype.post = function (uri, headers) {
|
2009-07-14 18:31:50 +02:00
|
|
|
var req = new ClientRequest("POST", uri, headers);
|
|
|
|
this._pushRequest(req);
|
2009-06-26 18:30:55 +02:00
|
|
|
return req;
|
2009-06-26 18:29:57 +02:00
|
|
|
};
|
2009-05-15 18:11:49 +02:00
|
|
|
|
2009-06-26 18:29:57 +02:00
|
|
|
node.http.Client.prototype.del = function (uri, headers) {
|
2009-07-14 18:31:50 +02:00
|
|
|
var req = new ClientRequest("DELETE", uri, headers);
|
|
|
|
this._pushRequest(req);
|
2009-06-26 18:30:55 +02:00
|
|
|
return req;
|
2009-06-26 18:29:57 +02:00
|
|
|
};
|
2009-05-15 18:11:49 +02:00
|
|
|
|
2009-06-26 18:29:57 +02:00
|
|
|
node.http.Client.prototype.put = function (uri, headers) {
|
2009-07-14 18:31:50 +02:00
|
|
|
var req = new ClientRequest("PUT", uri, headers);
|
|
|
|
this._pushRequest(req);
|
2009-06-26 18:30:55 +02:00
|
|
|
return req;
|
2009-06-26 18:29:57 +02:00
|
|
|
};
|
2009-06-06 23:57:15 +02:00
|
|
|
|
2009-05-19 13:12:46 +02:00
|
|
|
|
2009-06-28 19:05:58 +02:00
|
|
|
node.http.cat = function (url, encoding) {
|
|
|
|
var promise = new node.Promise();
|
|
|
|
|
2009-06-22 13:12:47 +02:00
|
|
|
var uri = node.http.parseUri(url);
|
2009-06-28 19:05:58 +02:00
|
|
|
var client = node.http.createClient(uri.port || 80, uri.host);
|
|
|
|
var req = client.get(uri.path || "/");
|
|
|
|
|
2009-06-29 13:18:30 +02:00
|
|
|
client.addListener("error", function () {
|
2009-06-28 19:05:58 +02:00
|
|
|
promise.emitError();
|
|
|
|
});
|
|
|
|
|
|
|
|
var content = "";
|
|
|
|
|
2009-06-22 13:12:47 +02:00
|
|
|
req.finish(function (res) {
|
2009-06-28 19:05:58 +02:00
|
|
|
if (res.statusCode < 200 || res.statusCode >= 300) {
|
|
|
|
promise.emitError([res.statusCode]);
|
|
|
|
return;
|
|
|
|
}
|
2009-06-22 13:12:47 +02:00
|
|
|
res.setBodyEncoding(encoding);
|
2009-06-29 13:18:30 +02:00
|
|
|
res.addListener("body", function (chunk) { content += chunk; });
|
|
|
|
res.addListener("complete", function () {
|
2009-06-28 19:05:58 +02:00
|
|
|
promise.emitSuccess([content]);
|
2009-06-27 20:40:43 +02:00
|
|
|
});
|
2009-06-22 13:12:47 +02:00
|
|
|
});
|
2009-06-28 19:05:58 +02:00
|
|
|
|
|
|
|
return promise;
|
2009-06-22 13:12:47 +02:00
|
|
|
};
|
2009-06-17 08:52:47 +02:00
|
|
|
|
2009-05-18 12:44:01 +02:00
|
|
|
})(); // anonymous namespace
|