2010-10-11 23:04:09 +02:00
|
|
|
var util = require('util');
|
2010-03-20 05:49:00 +01:00
|
|
|
|
2010-05-27 02:59:55 +02:00
|
|
|
var debug;
|
|
|
|
var debugLevel = parseInt(process.env.NODE_DEBUG, 16);
|
|
|
|
if (debugLevel & 0x4) {
|
2010-10-11 23:04:09 +02:00
|
|
|
debug = function (x) { util.error('HTTP: ' + x); };
|
2010-05-27 02:59:55 +02:00
|
|
|
} else {
|
|
|
|
debug = function () { };
|
2010-03-20 05:49:00 +01:00
|
|
|
}
|
|
|
|
|
2010-03-20 03:22:04 +01:00
|
|
|
var net = require('net');
|
2010-10-11 04:38:57 +02:00
|
|
|
var stream = require('stream');
|
2009-09-28 12:36:36 +02:00
|
|
|
|
2010-04-12 18:27:32 +02:00
|
|
|
var FreeList = require('freelist').FreeList;
|
2010-03-20 03:22:04 +01:00
|
|
|
var HTTPParser = process.binding('http_parser').HTTPParser;
|
|
|
|
|
2010-04-12 18:27:32 +02:00
|
|
|
var parsers = new FreeList('parsers', 1000, function () {
|
|
|
|
var parser = new HTTPParser('request');
|
2010-03-20 03:22:04 +01:00
|
|
|
|
2010-04-12 18:27:32 +02:00
|
|
|
parser.onMessageBegin = function () {
|
|
|
|
parser.incoming = new IncomingMessage(parser.socket);
|
|
|
|
parser.field = null;
|
|
|
|
parser.value = null;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Only servers will get URL events.
|
|
|
|
parser.onURL = function (b, start, len) {
|
|
|
|
var slice = b.toString('ascii', start, start+len);
|
|
|
|
if (parser.incoming.url) {
|
|
|
|
parser.incoming.url += slice;
|
|
|
|
} else {
|
|
|
|
// Almost always will branch here.
|
|
|
|
parser.incoming.url = slice;
|
|
|
|
}
|
|
|
|
};
|
2010-03-20 03:22:04 +01:00
|
|
|
|
2010-04-12 18:27:32 +02:00
|
|
|
parser.onHeaderField = function (b, start, len) {
|
|
|
|
var slice = b.toString('ascii', start, start+len).toLowerCase();
|
2010-07-07 18:43:39 +02:00
|
|
|
if (parser.value != undefined) {
|
2010-04-12 18:27:32 +02:00
|
|
|
parser.incoming._addHeaderLine(parser.field, parser.value);
|
2010-03-20 03:22:04 +01:00
|
|
|
parser.field = null;
|
|
|
|
parser.value = null;
|
2010-04-12 18:27:32 +02:00
|
|
|
}
|
|
|
|
if (parser.field) {
|
|
|
|
parser.field += slice;
|
|
|
|
} else {
|
|
|
|
parser.field = slice;
|
|
|
|
}
|
|
|
|
};
|
2010-03-20 03:22:04 +01:00
|
|
|
|
2010-04-12 18:27:32 +02:00
|
|
|
parser.onHeaderValue = function (b, start, len) {
|
|
|
|
var slice = b.toString('ascii', start, start+len);
|
|
|
|
if (parser.value) {
|
|
|
|
parser.value += slice;
|
|
|
|
} else {
|
|
|
|
parser.value = slice;
|
|
|
|
}
|
|
|
|
};
|
2010-03-20 03:22:04 +01:00
|
|
|
|
2010-04-12 18:27:32 +02:00
|
|
|
parser.onHeadersComplete = function (info) {
|
2010-07-07 18:43:39 +02:00
|
|
|
if (parser.field && (parser.value != undefined)) {
|
2010-04-12 18:27:32 +02:00
|
|
|
parser.incoming._addHeaderLine(parser.field, parser.value);
|
2010-09-29 11:38:48 +02:00
|
|
|
parser.field = null;
|
|
|
|
parser.value = null;
|
2010-04-12 18:27:32 +02:00
|
|
|
}
|
2010-03-20 03:22:04 +01:00
|
|
|
|
2010-04-12 18:27:32 +02:00
|
|
|
parser.incoming.httpVersionMajor = info.versionMajor;
|
|
|
|
parser.incoming.httpVersionMinor = info.versionMinor;
|
|
|
|
parser.incoming.httpVersion = info.versionMajor
|
|
|
|
+ '.'
|
|
|
|
+ info.versionMinor ;
|
2010-03-20 03:22:04 +01:00
|
|
|
|
2010-04-12 18:27:32 +02:00
|
|
|
if (info.method) {
|
|
|
|
// server only
|
|
|
|
parser.incoming.method = info.method;
|
|
|
|
} else {
|
|
|
|
// client only
|
2010-09-10 03:56:35 +02:00
|
|
|
parser.incoming.statusCode = info.statusCode;
|
2010-04-12 18:27:32 +02:00
|
|
|
}
|
2010-03-20 03:22:04 +01:00
|
|
|
|
2010-04-14 12:52:15 +02:00
|
|
|
parser.incoming.upgrade = info.upgrade;
|
|
|
|
|
2010-05-26 04:24:30 +02:00
|
|
|
var isHeadResponse = false;
|
|
|
|
|
2010-04-14 12:52:15 +02:00
|
|
|
if (!info.upgrade) {
|
|
|
|
// For upgraded connections, we'll emit this after parser.execute
|
|
|
|
// so that we can capture the first part of the new protocol
|
2010-05-26 04:24:30 +02:00
|
|
|
isHeadResponse = parser.onIncoming(parser.incoming, info.shouldKeepAlive);
|
2010-04-14 12:52:15 +02:00
|
|
|
}
|
2010-05-26 03:39:23 +02:00
|
|
|
|
2010-05-26 04:24:30 +02:00
|
|
|
return isHeadResponse;
|
2010-04-12 18:27:32 +02:00
|
|
|
};
|
2010-03-20 03:22:04 +01:00
|
|
|
|
2010-04-12 18:27:32 +02:00
|
|
|
parser.onBody = function (b, start, len) {
|
|
|
|
// TODO body encoding?
|
2010-06-16 03:19:25 +02:00
|
|
|
var slice = b.slice(start, start+len);
|
|
|
|
if (parser.incoming._decoder) {
|
|
|
|
var string = parser.incoming._decoder.write(slice);
|
|
|
|
if (string.length) parser.incoming.emit('data', string);
|
2010-04-12 18:27:32 +02:00
|
|
|
} else {
|
2010-06-16 03:19:25 +02:00
|
|
|
parser.incoming.emit('data', slice);
|
2010-04-12 18:27:32 +02:00
|
|
|
}
|
|
|
|
};
|
2010-03-20 03:22:04 +01:00
|
|
|
|
2010-04-12 18:27:32 +02:00
|
|
|
parser.onMessageComplete = function () {
|
2010-09-29 11:38:48 +02:00
|
|
|
this.incoming.complete = true;
|
|
|
|
if (parser.field && (parser.value != undefined)) {
|
|
|
|
parser.incoming._addHeaderLine(parser.field, parser.value);
|
|
|
|
}
|
2010-04-14 12:52:15 +02:00
|
|
|
if (!parser.incoming.upgrade) {
|
|
|
|
// For upgraded connections, also emit this after parser.execute
|
|
|
|
parser.incoming.emit("end");
|
|
|
|
}
|
2010-04-12 18:27:32 +02:00
|
|
|
};
|
2010-03-20 03:22:04 +01:00
|
|
|
|
|
|
|
return parser;
|
2010-04-12 18:27:32 +02:00
|
|
|
});
|
2010-10-08 01:17:06 +02:00
|
|
|
exports.parsers = parsers;
|
2010-03-15 21:48:03 +01:00
|
|
|
|
|
|
|
|
2009-09-28 12:36:36 +02:00
|
|
|
var CRLF = "\r\n";
|
2009-12-08 01:18:43 +01:00
|
|
|
var STATUS_CODES = exports.STATUS_CODES = {
|
2009-07-01 00:49:56 +02:00
|
|
|
100 : 'Continue',
|
|
|
|
101 : 'Switching Protocols',
|
2010-05-23 22:48:50 +02:00
|
|
|
102 : 'Processing', // RFC 2518, obsoleted by RFC 4918
|
2009-07-01 00:49:56 +02:00
|
|
|
200 : 'OK',
|
|
|
|
201 : 'Created',
|
|
|
|
202 : 'Accepted',
|
|
|
|
203 : 'Non-Authoritative Information',
|
|
|
|
204 : 'No Content',
|
|
|
|
205 : 'Reset Content',
|
|
|
|
206 : 'Partial Content',
|
2010-05-23 22:48:50 +02:00
|
|
|
207 : 'Multi-Status', // RFC 4918
|
2009-07-01 00:49:56 +02:00
|
|
|
300 : 'Multiple Choices',
|
|
|
|
301 : 'Moved Permanently',
|
|
|
|
302 : 'Moved Temporarily',
|
|
|
|
303 : 'See Other',
|
|
|
|
304 : 'Not Modified',
|
|
|
|
305 : 'Use Proxy',
|
2010-05-23 22:48:50 +02:00
|
|
|
307 : 'Temporary Redirect',
|
2009-07-01 00:49:56 +02:00
|
|
|
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',
|
2010-05-23 22:48:50 +02:00
|
|
|
416 : 'Requested Range Not Satisfiable',
|
|
|
|
417 : 'Expectation Failed',
|
|
|
|
418 : 'I\'m a teapot', // RFC 2324
|
|
|
|
422 : 'Unprocessable Entity', // RFC 4918
|
|
|
|
423 : 'Locked', // RFC 4918
|
|
|
|
424 : 'Failed Dependency', // RFC 4918
|
|
|
|
425 : 'Unordered Collection', // RFC 4918
|
|
|
|
426 : 'Upgrade Required', // RFC 2817
|
2009-07-01 00:49:56 +02:00
|
|
|
500 : 'Internal Server Error',
|
|
|
|
501 : 'Not Implemented',
|
|
|
|
502 : 'Bad Gateway',
|
|
|
|
503 : 'Service Unavailable',
|
|
|
|
504 : 'Gateway Time-out',
|
2010-05-23 22:48:50 +02:00
|
|
|
505 : 'HTTP Version not supported',
|
|
|
|
506 : 'Variant Also Negotiates', // RFC 2295
|
|
|
|
507 : 'Insufficient Storage', // RFC 4918
|
|
|
|
509 : 'Bandwidth Limit Exceeded',
|
|
|
|
510 : 'Not Extended' // RFC 2774
|
2009-07-01 00:49:56 +02:00
|
|
|
};
|
2009-05-11 19:08:29 +02:00
|
|
|
|
2010-03-24 05:31:44 +01:00
|
|
|
var connectionExpression = /Connection/i;
|
|
|
|
var transferEncodingExpression = /Transfer-Encoding/i;
|
|
|
|
var closeExpression = /close/i;
|
|
|
|
var chunkExpression = /chunk/i;
|
|
|
|
var contentLengthExpression = /Content-Length/i;
|
2010-09-10 03:56:35 +02:00
|
|
|
var expectExpression = /Expect/i;
|
|
|
|
var continueExpression = /100-continue/i;
|
2009-05-11 18:54:52 +02:00
|
|
|
|
2009-06-26 18:29:57 +02:00
|
|
|
|
2009-07-14 11:59:13 +02:00
|
|
|
/* Abstract base class for ServerRequest and ClientResponse. */
|
2010-03-20 03:22:04 +01:00
|
|
|
function IncomingMessage (socket) {
|
2010-10-11 04:38:57 +02:00
|
|
|
stream.Stream.call(this);
|
2009-06-26 18:29:57 +02:00
|
|
|
|
2010-03-24 15:20:56 +01:00
|
|
|
// TODO Remove one of these eventually.
|
2010-03-20 03:22:04 +01:00
|
|
|
this.socket = socket;
|
2010-03-24 15:20:56 +01:00
|
|
|
this.connection = socket;
|
|
|
|
|
2009-07-14 11:59:13 +02:00
|
|
|
this.httpVersion = null;
|
2010-09-29 11:38:48 +02:00
|
|
|
this.complete = false;
|
2009-08-23 12:20:25 +02:00
|
|
|
this.headers = {};
|
2010-09-29 11:38:48 +02:00
|
|
|
this.trailers = {};
|
2009-07-14 18:31:50 +02:00
|
|
|
|
2010-10-11 10:22:38 +02:00
|
|
|
this.readable = true;
|
|
|
|
|
2009-08-26 22:03:19 +02:00
|
|
|
// request (server) only
|
2010-01-05 06:07:50 +01:00
|
|
|
this.url = "";
|
2009-10-04 12:17:50 +02:00
|
|
|
|
2009-07-14 18:31:50 +02:00
|
|
|
this.method = null;
|
|
|
|
|
|
|
|
// response (client) only
|
|
|
|
this.statusCode = null;
|
2010-03-20 03:22:04 +01:00
|
|
|
this.client = this.socket;
|
2009-07-16 10:59:40 +02:00
|
|
|
}
|
2010-10-11 23:04:09 +02:00
|
|
|
util.inherits(IncomingMessage, stream.Stream);
|
2009-11-05 00:02:15 +01:00
|
|
|
exports.IncomingMessage = IncomingMessage;
|
2009-07-14 11:59:13 +02:00
|
|
|
|
2010-10-12 01:36:12 +02:00
|
|
|
|
|
|
|
IncomingMessage.prototype.destroy = function (error) {
|
|
|
|
this.socket.destroy(error);
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2010-06-16 03:19:25 +02:00
|
|
|
IncomingMessage.prototype.setEncoding = function (encoding) {
|
|
|
|
var StringDecoder = require("string_decoder").StringDecoder; // lazy load
|
|
|
|
this._decoder = new StringDecoder(encoding);
|
2009-06-26 18:29:57 +02:00
|
|
|
};
|
|
|
|
|
2009-08-09 19:16:25 +02:00
|
|
|
IncomingMessage.prototype.pause = function () {
|
2010-03-20 03:22:04 +01:00
|
|
|
this.socket.pause();
|
2009-08-09 19:16:25 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
IncomingMessage.prototype.resume = function () {
|
2010-03-20 03:22:04 +01:00
|
|
|
this.socket.resume();
|
2009-08-09 19:16:25 +02:00
|
|
|
};
|
|
|
|
|
2010-06-21 20:53:17 +02:00
|
|
|
// Add the given (field, value) pair to the message
|
|
|
|
//
|
|
|
|
// Per RFC2616, section 4.2 it is acceptable to join multiple instances of the
|
|
|
|
// same header with a ', ' if the header in question supports specification of
|
|
|
|
// multiple values this way. If not, we declare the first instance the winner
|
|
|
|
// and drop the second. Extended header fields (those beginning with 'x-') are
|
|
|
|
// always joined.
|
2009-08-23 12:20:25 +02:00
|
|
|
IncomingMessage.prototype._addHeaderLine = function (field, value) {
|
2010-09-29 11:38:48 +02:00
|
|
|
var dest;
|
|
|
|
if (this.complete) {
|
2010-10-07 05:05:23 +02:00
|
|
|
dest = this.trailers;
|
2010-09-29 11:38:48 +02:00
|
|
|
} else {
|
2010-10-07 05:05:23 +02:00
|
|
|
dest = this.headers;
|
2010-09-29 11:38:48 +02:00
|
|
|
}
|
2010-06-21 20:53:17 +02:00
|
|
|
switch (field) {
|
2010-08-22 01:34:38 +02:00
|
|
|
// Array headers:
|
|
|
|
case 'set-cookie':
|
2010-09-29 11:38:48 +02:00
|
|
|
if (field in dest) {
|
|
|
|
dest[field].push(value);
|
2010-08-22 01:34:38 +02:00
|
|
|
} else {
|
2010-09-29 11:38:48 +02:00
|
|
|
dest[field] = [value];
|
2010-08-22 01:34:38 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
// Comma separate. Maybe make these arrays?
|
|
|
|
case 'accept':
|
|
|
|
case 'accept-charset':
|
|
|
|
case 'accept-encoding':
|
|
|
|
case 'accept-language':
|
|
|
|
case 'connection':
|
|
|
|
case 'cookie':
|
2010-09-29 11:38:48 +02:00
|
|
|
if (field in dest) {
|
|
|
|
dest[field] += ', ' + value;
|
2010-08-22 01:34:38 +02:00
|
|
|
} else {
|
2010-09-29 11:38:48 +02:00
|
|
|
dest[field] = value;
|
2010-08-22 01:34:38 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
if (field.slice(0,2) == 'x-') {
|
|
|
|
// except for x-
|
2010-09-29 11:38:48 +02:00
|
|
|
if (field in dest) {
|
|
|
|
dest[field] += ', ' + value;
|
2010-08-22 01:34:38 +02:00
|
|
|
} else {
|
2010-09-29 11:38:48 +02:00
|
|
|
dest[field] = value;
|
2010-08-22 01:34:38 +02:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// drop duplicates
|
2010-09-29 11:38:48 +02:00
|
|
|
if (!(field in dest)) dest[field] = value;
|
2010-08-22 01:34:38 +02:00
|
|
|
}
|
|
|
|
break;
|
2009-08-23 12:20:25 +02:00
|
|
|
}
|
|
|
|
};
|
2009-07-12 11:48:37 +02:00
|
|
|
|
2010-03-20 03:22:04 +01:00
|
|
|
function OutgoingMessage (socket) {
|
2010-10-11 04:38:57 +02:00
|
|
|
stream.Stream.call(this);
|
2010-03-03 22:06:19 +01:00
|
|
|
|
2010-03-24 15:20:56 +01:00
|
|
|
// TODO Remove one of these eventually.
|
2010-03-20 03:22:04 +01:00
|
|
|
this.socket = socket;
|
2010-03-24 15:20:56 +01:00
|
|
|
this.connection = socket;
|
2009-07-14 11:59:13 +02:00
|
|
|
|
2009-07-14 18:31:50 +02:00
|
|
|
this.output = [];
|
2009-09-13 12:38:59 +02:00
|
|
|
this.outputEncodings = [];
|
2009-06-26 18:29:57 +02:00
|
|
|
|
2010-10-11 10:22:38 +02:00
|
|
|
this.writable = true;
|
|
|
|
|
2010-05-27 02:59:55 +02:00
|
|
|
this._last = false;
|
2010-03-24 05:31:44 +01:00
|
|
|
this.chunkedEncoding = false;
|
|
|
|
this.shouldKeepAlive = true;
|
|
|
|
this.useChunkedEncodingByDefault = true;
|
2009-07-14 11:59:13 +02:00
|
|
|
|
2010-04-22 00:15:21 +02:00
|
|
|
this._hasBody = true;
|
2010-09-09 08:29:35 +02:00
|
|
|
this._trailer = '';
|
2010-04-22 00:15:21 +02:00
|
|
|
|
2009-07-14 18:31:50 +02:00
|
|
|
this.finished = false;
|
2009-07-16 10:59:40 +02:00
|
|
|
}
|
2010-10-11 23:04:09 +02:00
|
|
|
util.inherits(OutgoingMessage, stream.Stream);
|
2009-11-05 00:02:15 +01:00
|
|
|
exports.OutgoingMessage = OutgoingMessage;
|
2009-07-14 11:59:13 +02:00
|
|
|
|
2010-10-12 01:36:12 +02:00
|
|
|
|
|
|
|
OutgoingMessage.prototype.destroy = function (error) {
|
|
|
|
this.socket.destroy(error);
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2010-05-27 02:59:55 +02:00
|
|
|
// This abstract either writing directly to the socket or buffering it.
|
2010-02-17 07:16:29 +01:00
|
|
|
OutgoingMessage.prototype._send = function (data, encoding) {
|
2010-05-28 05:28:12 +02:00
|
|
|
// This is a shameful hack to get the headers and first body chunk onto
|
|
|
|
// the same packet. Future versions of Node are going to take care of
|
|
|
|
// this at a lower level and in a more general way.
|
|
|
|
if (!this._headerSent) {
|
|
|
|
if (typeof data === 'string') {
|
|
|
|
data = this._header + data;
|
|
|
|
} else {
|
|
|
|
this.output.unshift(this._header);
|
|
|
|
this.outputEncodings.unshift('ascii');
|
|
|
|
}
|
|
|
|
this._headerSent = true;
|
|
|
|
}
|
2010-10-12 01:00:36 +02:00
|
|
|
return this._writeRaw(data, encoding);
|
2010-10-07 05:05:23 +02:00
|
|
|
};
|
2010-05-28 05:28:12 +02:00
|
|
|
|
2010-09-10 03:56:35 +02:00
|
|
|
OutgoingMessage.prototype._writeRaw = function(data, encoding) {
|
2010-05-28 05:28:12 +02:00
|
|
|
if (this.connection._outgoing[0] === this && this.connection.writable) {
|
|
|
|
// There might be pending data in the this.output buffer.
|
|
|
|
while (this.output.length) {
|
|
|
|
if (!this.connection.writable) {
|
|
|
|
this._buffer(data, encoding);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
var c = this.output.shift();
|
|
|
|
var e = this.outputEncodings.shift();
|
|
|
|
this.connection.write(c, e);
|
|
|
|
}
|
|
|
|
|
2010-05-27 02:59:55 +02:00
|
|
|
// Directly write to socket.
|
|
|
|
return this.connection.write(data, encoding);
|
|
|
|
} else {
|
2010-05-28 05:28:12 +02:00
|
|
|
this._buffer(data, encoding);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
};
|
2010-05-27 02:59:55 +02:00
|
|
|
|
2010-05-28 05:28:12 +02:00
|
|
|
OutgoingMessage.prototype._buffer = function (data, encoding) {
|
|
|
|
if (data.length === 0) return;
|
2010-05-27 02:59:55 +02:00
|
|
|
|
2010-05-28 05:28:12 +02:00
|
|
|
var length = this.output.length;
|
2009-09-14 16:25:41 +02:00
|
|
|
|
2010-05-28 05:28:12 +02:00
|
|
|
if (length === 0 || typeof data != 'string') {
|
2009-09-14 16:25:41 +02:00
|
|
|
this.output.push(data);
|
2009-09-21 12:27:22 +02:00
|
|
|
encoding = encoding || "ascii";
|
2009-09-14 16:25:41 +02:00
|
|
|
this.outputEncodings.push(encoding);
|
2010-05-28 05:28:12 +02:00
|
|
|
return false;
|
|
|
|
}
|
2009-09-14 16:25:41 +02:00
|
|
|
|
2010-05-28 05:28:12 +02:00
|
|
|
var lastEncoding = this.outputEncodings[length-1];
|
|
|
|
var lastData = this.output[length-1];
|
|
|
|
|
|
|
|
if ((lastEncoding === encoding) ||
|
|
|
|
(!encoding && data.constructor === lastData.constructor)) {
|
|
|
|
this.output[length-1] = lastData + data;
|
2010-05-27 02:59:55 +02:00
|
|
|
return false;
|
2009-09-14 16:25:41 +02:00
|
|
|
}
|
2010-05-28 05:28:12 +02:00
|
|
|
|
|
|
|
this.output.push(data);
|
|
|
|
encoding = encoding || "ascii";
|
|
|
|
this.outputEncodings.push(encoding);
|
|
|
|
|
|
|
|
return false;
|
2009-07-14 18:31:50 +02:00
|
|
|
};
|
2009-07-14 11:59:13 +02:00
|
|
|
|
2010-05-27 02:59:55 +02:00
|
|
|
|
|
|
|
OutgoingMessage.prototype._storeHeader = function (firstLine, headers) {
|
2010-03-24 05:31:44 +01:00
|
|
|
var sentConnectionHeader = false;
|
|
|
|
var sentContentLengthHeader = false;
|
|
|
|
var sentTransferEncodingHeader = false;
|
2010-09-10 03:56:35 +02:00
|
|
|
var sentExpect = false;
|
2009-07-31 18:34:27 +02:00
|
|
|
|
2010-03-24 05:31:44 +01:00
|
|
|
// firstLine in the case of request is: "GET /index.html HTTP/1.1\r\n"
|
2009-07-14 18:31:50 +02:00
|
|
|
// in the case of response it is: "HTTP/1.1 200 OK\r\n"
|
2010-03-24 05:31:44 +01:00
|
|
|
var messageHeader = firstLine;
|
2009-08-23 12:20:25 +02:00
|
|
|
var field, value;
|
2010-09-15 16:52:23 +02:00
|
|
|
var self = this;
|
|
|
|
|
|
|
|
function store(field, value) {
|
|
|
|
messageHeader += field + ": " + value + CRLF;
|
|
|
|
|
|
|
|
if (connectionExpression.test(field)) {
|
|
|
|
sentConnectionHeader = true;
|
|
|
|
if (closeExpression.test(value)) {
|
|
|
|
self._last = true;
|
|
|
|
} else {
|
|
|
|
self.shouldKeepAlive = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
} else if (transferEncodingExpression.test(field)) {
|
|
|
|
sentTransferEncodingHeader = true;
|
|
|
|
if (chunkExpression.test(value)) self.chunkedEncoding = true;
|
|
|
|
|
|
|
|
} else if (contentLengthExpression.test(field)) {
|
|
|
|
sentContentLengthHeader = true;
|
|
|
|
|
2010-09-10 03:56:35 +02:00
|
|
|
} else if (expectExpression.test(field)) {
|
|
|
|
sentExpect = true;
|
|
|
|
|
2010-09-15 16:52:23 +02:00
|
|
|
}
|
|
|
|
}
|
2009-06-26 18:29:57 +02:00
|
|
|
|
2010-04-12 18:57:24 +02:00
|
|
|
if (headers) {
|
|
|
|
var keys = Object.keys(headers);
|
2010-07-14 23:29:28 +02:00
|
|
|
var isArray = (Array.isArray(headers));
|
2010-09-15 16:52:23 +02:00
|
|
|
|
2010-10-04 19:50:35 +02:00
|
|
|
for (var i = 0, l = keys.length; i < l; i++) {
|
2010-04-12 18:57:24 +02:00
|
|
|
var key = keys[i];
|
|
|
|
if (isArray) {
|
|
|
|
field = headers[key][0];
|
|
|
|
value = headers[key][1];
|
|
|
|
} else {
|
|
|
|
field = key;
|
|
|
|
value = headers[key];
|
|
|
|
}
|
2009-08-26 22:03:19 +02:00
|
|
|
|
2010-09-15 16:52:23 +02:00
|
|
|
if (Array.isArray(value)) {
|
2010-10-04 19:50:35 +02:00
|
|
|
for (var j = 0; j < value.length; j++) {
|
|
|
|
store(field, value[j]);
|
2010-08-02 07:22:20 +02:00
|
|
|
}
|
2010-09-15 16:52:23 +02:00
|
|
|
} else {
|
|
|
|
store(field, value);
|
2010-04-12 18:57:24 +02:00
|
|
|
}
|
2009-07-14 18:31:50 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-08-26 22:03:19 +02:00
|
|
|
// keep-alive logic
|
2010-03-24 05:31:44 +01:00
|
|
|
if (sentConnectionHeader == false) {
|
|
|
|
if (this.shouldKeepAlive &&
|
|
|
|
(sentContentLengthHeader || this.useChunkedEncodingByDefault)) {
|
|
|
|
messageHeader += "Connection: keep-alive\r\n";
|
2009-06-26 18:29:57 +02:00
|
|
|
} else {
|
2010-05-27 02:59:55 +02:00
|
|
|
this._last = true;
|
2010-03-24 05:31:44 +01:00
|
|
|
messageHeader += "Connection: close\r\n";
|
2009-06-26 18:29:57 +02:00
|
|
|
}
|
2009-07-14 18:31:50 +02:00
|
|
|
}
|
|
|
|
|
2010-03-24 05:31:44 +01:00
|
|
|
if (sentContentLengthHeader == false && sentTransferEncodingHeader == false) {
|
2010-04-22 00:15:21 +02:00
|
|
|
if (this._hasBody) {
|
|
|
|
if (this.useChunkedEncodingByDefault) {
|
|
|
|
messageHeader += "Transfer-Encoding: chunked\r\n";
|
|
|
|
this.chunkedEncoding = true;
|
|
|
|
} else {
|
2010-05-27 02:59:55 +02:00
|
|
|
this._last = true;
|
2010-04-22 00:15:21 +02:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Make sure we don't end the 0\r\n\r\n at the end of the message.
|
|
|
|
this.chunkedEncoding = false;
|
2009-12-23 21:48:14 +01:00
|
|
|
}
|
2009-07-14 18:31:50 +02:00
|
|
|
}
|
|
|
|
|
2010-05-28 05:28:12 +02:00
|
|
|
this._header = messageHeader + CRLF;
|
|
|
|
this._headerSent = false;
|
2010-09-10 03:56:35 +02:00
|
|
|
// wait until the first body chunk, or close(), is sent to flush,
|
|
|
|
// UNLESS we're sending Expect: 100-continue.
|
|
|
|
if (sentExpect) {
|
|
|
|
this._send("");
|
|
|
|
}
|
2009-07-14 18:31:50 +02:00
|
|
|
};
|
|
|
|
|
2010-02-17 07:16:29 +01:00
|
|
|
OutgoingMessage.prototype.write = function (chunk, encoding) {
|
2010-05-27 02:59:55 +02:00
|
|
|
if (!this._header) {
|
2010-05-28 05:28:12 +02:00
|
|
|
throw new Error("You have to call writeHead() before write()");
|
2010-03-14 04:36:45 +01:00
|
|
|
}
|
|
|
|
|
2010-04-22 00:15:21 +02:00
|
|
|
if (!this._hasBody) {
|
2010-08-12 19:06:52 +02:00
|
|
|
console.error("This type of response MUST NOT have a body. Ignoring write() calls.");
|
|
|
|
return true;
|
2010-04-22 00:15:21 +02:00
|
|
|
}
|
|
|
|
|
2010-05-05 03:28:49 +02:00
|
|
|
if (typeof chunk !== "string"
|
2010-07-15 23:37:03 +02:00
|
|
|
&& !Buffer.isBuffer(chunk)
|
2010-05-05 03:28:49 +02:00
|
|
|
&& !Array.isArray(chunk)) {
|
|
|
|
throw new TypeError("first argument must be a string, Array, or Buffer");
|
|
|
|
}
|
|
|
|
|
2010-05-27 02:59:55 +02:00
|
|
|
if (chunk.length === 0) return false;
|
|
|
|
|
|
|
|
var len, ret;
|
2010-03-24 05:31:44 +01:00
|
|
|
if (this.chunkedEncoding) {
|
2010-05-27 02:59:55 +02:00
|
|
|
if (typeof(chunk) === 'string') {
|
|
|
|
len = Buffer.byteLength(chunk, encoding);
|
|
|
|
var chunk = len.toString(16) + CRLF + chunk + CRLF;
|
2010-10-11 23:04:09 +02:00
|
|
|
debug('string chunk = ' + util.inspect(chunk));
|
2010-05-27 02:59:55 +02:00
|
|
|
ret = this._send(chunk, encoding);
|
|
|
|
} else {
|
|
|
|
// buffer
|
|
|
|
len = chunk.length;
|
|
|
|
this._send(len.toString(16) + CRLF);
|
|
|
|
this._send(chunk);
|
|
|
|
ret = this._send(CRLF);
|
2010-03-20 03:22:04 +01:00
|
|
|
}
|
2009-07-14 18:31:50 +02:00
|
|
|
} else {
|
2010-05-27 02:59:55 +02:00
|
|
|
ret = this._send(chunk, encoding);
|
2009-07-14 18:31:50 +02:00
|
|
|
}
|
2009-06-26 18:29:57 +02:00
|
|
|
|
2010-05-27 02:59:55 +02:00
|
|
|
debug('write ret = ' + ret);
|
|
|
|
return ret;
|
2009-07-14 18:31:50 +02:00
|
|
|
};
|
|
|
|
|
2010-09-09 08:29:35 +02:00
|
|
|
|
|
|
|
OutgoingMessage.prototype.addTrailers = function (headers) {
|
|
|
|
this._trailer = "";
|
|
|
|
var keys = Object.keys(headers);
|
|
|
|
var isArray = (Array.isArray(headers));
|
|
|
|
for (var i = 0, l = keys.length; i < l; i++) {
|
|
|
|
var key = keys[i];
|
|
|
|
if (isArray) {
|
|
|
|
field = headers[key][0];
|
|
|
|
value = headers[key][1];
|
|
|
|
} else {
|
|
|
|
field = key;
|
|
|
|
value = headers[key];
|
|
|
|
}
|
|
|
|
|
|
|
|
this._trailer += field + ": " + value + CRLF;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2010-04-08 19:44:22 +02:00
|
|
|
OutgoingMessage.prototype.end = function (data, encoding) {
|
2010-05-27 02:59:55 +02:00
|
|
|
var ret;
|
|
|
|
|
2010-05-28 05:28:12 +02:00
|
|
|
var hot = this._headerSent === false
|
|
|
|
&& typeof(data) === "string"
|
|
|
|
&& data.length > 0
|
|
|
|
&& this.output.length === 0
|
|
|
|
&& this.connection.writable
|
|
|
|
&& this.connection._outgoing[0] === this
|
|
|
|
;
|
|
|
|
|
|
|
|
if (hot) {
|
|
|
|
// Hot path. They're doing
|
|
|
|
// res.writeHead();
|
|
|
|
// res.end(blah);
|
|
|
|
// HACKY.
|
|
|
|
if (this.chunkedEncoding) {
|
|
|
|
var l = Buffer.byteLength(data, encoding).toString(16);
|
|
|
|
ret = this.connection.write( this._header
|
|
|
|
+ l
|
|
|
|
+ CRLF
|
|
|
|
+ data
|
2010-09-09 08:29:35 +02:00
|
|
|
+ "\r\n0\r\n"
|
|
|
|
+ this._trailer
|
|
|
|
+ "\r\n"
|
2010-05-28 05:28:12 +02:00
|
|
|
, encoding
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
ret = this.connection.write(this._header + data, encoding);
|
|
|
|
}
|
|
|
|
this._headerSent = true;
|
|
|
|
|
|
|
|
} else if (data) {
|
|
|
|
// Normal body write.
|
2010-05-27 02:59:55 +02:00
|
|
|
ret = this.write(data, encoding);
|
|
|
|
}
|
|
|
|
|
2010-05-28 05:28:12 +02:00
|
|
|
if (!hot) {
|
|
|
|
if (this.chunkedEncoding) {
|
2010-09-09 08:29:35 +02:00
|
|
|
ret = this._send('0\r\n' + this._trailer + '\r\n'); // Last chunk.
|
2010-10-01 02:12:56 +02:00
|
|
|
} else {
|
2010-05-28 05:28:12 +02:00
|
|
|
// Force a flush, HACK.
|
|
|
|
ret = this._send('');
|
|
|
|
}
|
2010-05-27 02:59:55 +02:00
|
|
|
}
|
|
|
|
|
2010-05-28 05:28:12 +02:00
|
|
|
this.finished = true;
|
|
|
|
|
2010-05-27 02:59:55 +02:00
|
|
|
// There is the first message on the outgoing queue, and we've sent
|
|
|
|
// everything to the socket.
|
|
|
|
if (this.output.length === 0 && this.connection._outgoing[0] === this) {
|
|
|
|
debug('outgoing message end. shifting because was flushed');
|
|
|
|
this.connection._onOutgoingSent();
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
2009-07-14 18:31:50 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2009-12-05 08:37:46 +01:00
|
|
|
function ServerResponse (req) {
|
2010-03-20 03:22:04 +01:00
|
|
|
OutgoingMessage.call(this, req.socket);
|
2009-07-14 18:31:50 +02:00
|
|
|
|
2010-06-02 23:46:29 +02:00
|
|
|
if (req.method === 'HEAD') this._hasBody = false;
|
|
|
|
|
2009-12-05 08:37:46 +01:00
|
|
|
if (req.httpVersionMajor < 1 || req.httpVersionMinor < 1) {
|
2010-03-24 05:31:44 +01:00
|
|
|
this.useChunkedEncodingByDefault = false;
|
|
|
|
this.shouldKeepAlive = false;
|
2009-12-05 08:37:46 +01:00
|
|
|
}
|
2009-07-16 10:59:40 +02:00
|
|
|
}
|
2010-10-11 23:04:09 +02:00
|
|
|
util.inherits(ServerResponse, OutgoingMessage);
|
2009-11-04 12:37:52 +01:00
|
|
|
exports.ServerResponse = ServerResponse;
|
2009-07-14 18:31:50 +02:00
|
|
|
|
2010-09-10 03:56:35 +02:00
|
|
|
ServerResponse.prototype.writeContinue = function () {
|
|
|
|
this._writeRaw("HTTP/1.1 100 Continue" + CRLF + CRLF, 'ascii');
|
|
|
|
this._sent100 = true;
|
2010-10-07 05:05:23 +02:00
|
|
|
};
|
2010-02-22 01:53:20 +01:00
|
|
|
|
2010-02-25 21:54:48 +01:00
|
|
|
ServerResponse.prototype.writeHead = function (statusCode) {
|
2010-02-22 01:53:20 +01:00
|
|
|
var reasonPhrase, headers, headerIndex;
|
|
|
|
|
|
|
|
if (typeof arguments[1] == 'string') {
|
|
|
|
reasonPhrase = arguments[1];
|
|
|
|
headerIndex = 2;
|
|
|
|
} else {
|
|
|
|
reasonPhrase = STATUS_CODES[statusCode] || "unknown";
|
|
|
|
headerIndex = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typeof arguments[headerIndex] == 'object') {
|
|
|
|
headers = arguments[headerIndex];
|
|
|
|
} else {
|
|
|
|
headers = {};
|
|
|
|
}
|
|
|
|
|
2010-03-24 05:31:44 +01:00
|
|
|
var statusLine = "HTTP/1.1 " + statusCode.toString() + " "
|
2010-02-22 01:53:20 +01:00
|
|
|
+ reasonPhrase + CRLF;
|
2010-04-22 00:15:21 +02:00
|
|
|
|
2010-09-10 03:56:35 +02:00
|
|
|
if ( statusCode === 204
|
|
|
|
|| statusCode === 304
|
|
|
|
|| (statusCode >= 100 && statusCode <= 199)
|
|
|
|
) {
|
2010-04-22 00:15:21 +02:00
|
|
|
// RFC 2616, 10.2.5:
|
|
|
|
// The 204 response MUST NOT include a message-body, and thus is always
|
|
|
|
// terminated by the first empty line after the header fields.
|
|
|
|
// RFC 2616, 10.3.5:
|
|
|
|
// The 304 response MUST NOT contain a message-body, and thus is always
|
|
|
|
// terminated by the first empty line after the header fields.
|
2010-09-10 03:56:35 +02:00
|
|
|
// RFC 2616, 10.1 Informational 1xx:
|
|
|
|
// This class of status code indicates a provisional response,
|
|
|
|
// consisting only of the Status-Line and optional headers, and is
|
|
|
|
// terminated by an empty line.
|
2010-04-22 00:15:21 +02:00
|
|
|
this._hasBody = false;
|
|
|
|
}
|
|
|
|
|
2010-09-10 03:56:35 +02:00
|
|
|
// don't keep alive connections where the client expects 100 Continue
|
|
|
|
// but we sent a final status; they may put extra bytes on the wire.
|
|
|
|
if (this._expect_continue && ! this._sent100) {
|
2010-10-18 07:14:29 +02:00
|
|
|
this.shouldKeepAlive = false;
|
2010-09-10 03:56:35 +02:00
|
|
|
}
|
2010-04-22 00:15:21 +02:00
|
|
|
|
2010-05-27 02:59:55 +02:00
|
|
|
this._storeHeader(statusLine, headers);
|
2009-07-14 18:31:50 +02:00
|
|
|
};
|
|
|
|
|
2010-09-16 23:49:12 +02:00
|
|
|
|
2010-05-12 21:15:58 +02:00
|
|
|
ServerResponse.prototype.writeHeader = function () {
|
|
|
|
this.writeHead.apply(this, arguments);
|
|
|
|
};
|
2009-07-14 18:31:50 +02:00
|
|
|
|
2010-03-20 03:22:04 +01:00
|
|
|
function ClientRequest (socket, method, url, headers) {
|
|
|
|
OutgoingMessage.call(this, socket);
|
2009-07-14 18:31:50 +02:00
|
|
|
|
2010-08-10 06:23:52 +02:00
|
|
|
this.method = method = method.toUpperCase();
|
2010-03-24 05:31:44 +01:00
|
|
|
this.shouldKeepAlive = false;
|
2009-07-31 18:34:27 +02:00
|
|
|
if (method === "GET" || method === "HEAD") {
|
2010-03-24 05:31:44 +01:00
|
|
|
this.useChunkedEncodingByDefault = false;
|
2009-07-31 18:34:27 +02:00
|
|
|
} else {
|
2010-03-24 05:31:44 +01:00
|
|
|
this.useChunkedEncodingByDefault = true;
|
2009-07-31 18:34:27 +02:00
|
|
|
}
|
2010-05-27 02:59:55 +02:00
|
|
|
this._last = true;
|
2009-07-14 18:31:50 +02:00
|
|
|
|
2010-05-27 02:59:55 +02:00
|
|
|
this._storeHeader(method + " " + url + " HTTP/1.1\r\n", headers);
|
2009-07-16 10:59:40 +02:00
|
|
|
}
|
2010-10-11 23:04:09 +02:00
|
|
|
util.inherits(ClientRequest, OutgoingMessage);
|
2009-11-04 12:37:52 +01:00
|
|
|
exports.ClientRequest = ClientRequest;
|
2009-07-14 18:31:50 +02:00
|
|
|
|
2010-05-27 02:59:55 +02:00
|
|
|
function outgoingFlush (socket) {
|
|
|
|
// This logic is probably a bit confusing. Let me explain a bit:
|
|
|
|
//
|
|
|
|
// In both HTTP servers and clients it is possible to queue up several
|
|
|
|
// outgoing messages. This is easiest to imagine in the case of a client.
|
|
|
|
// Take the following situation:
|
|
|
|
//
|
|
|
|
// req1 = client.request('GET', '/');
|
|
|
|
// req2 = client.request('POST', '/');
|
|
|
|
//
|
|
|
|
// The question is what happens when the user does
|
|
|
|
//
|
|
|
|
// req2.write("hello world\n");
|
|
|
|
//
|
|
|
|
// It's possible that the first request has not been completely flushed to
|
|
|
|
// the socket yet. Thus the outgoing messages need to be prepared to queue
|
|
|
|
// up data internally before sending it on further to the socket's queue.
|
|
|
|
//
|
|
|
|
// This function, outgoingFlush(), is called by both the Server
|
|
|
|
// implementation and the Client implementation to attempt to flush any
|
|
|
|
// pending messages out to the socket.
|
|
|
|
var message = socket._outgoing[0];
|
|
|
|
|
|
|
|
if (!message) return;
|
|
|
|
|
|
|
|
var ret;
|
|
|
|
|
|
|
|
while (message.output.length) {
|
|
|
|
if (!socket.writable) return; // XXX Necessary?
|
|
|
|
|
|
|
|
var data = message.output.shift();
|
|
|
|
var encoding = message.outputEncodings.shift();
|
|
|
|
|
|
|
|
ret = socket.write(data, encoding);
|
|
|
|
}
|
2009-09-13 12:38:59 +02:00
|
|
|
|
2010-05-27 02:59:55 +02:00
|
|
|
if (message.finished) {
|
|
|
|
socket._onOutgoingSent();
|
|
|
|
} else if (ret) {
|
|
|
|
message.emit('drain');
|
|
|
|
}
|
|
|
|
}
|
2009-08-26 22:03:19 +02:00
|
|
|
|
2009-05-18 19:33:05 +02:00
|
|
|
|
2010-05-27 02:59:55 +02:00
|
|
|
function httpSocketSetup (socket) {
|
|
|
|
// An array of outgoing messages for the socket. In pipelined connections
|
|
|
|
// we need to keep track of the order they were sent.
|
|
|
|
socket._outgoing = [];
|
2010-05-28 05:28:12 +02:00
|
|
|
socket.__destroyOnDrain = false;
|
2009-05-18 19:33:05 +02:00
|
|
|
|
2010-05-27 02:59:55 +02:00
|
|
|
// NOTE: be sure not to use ondrain elsewhere in this file!
|
|
|
|
socket.ondrain = function () {
|
|
|
|
var message = socket._outgoing[0];
|
|
|
|
if (message) message.emit('drain');
|
2010-05-28 05:28:12 +02:00
|
|
|
if (socket.__destroyOnDrain) socket.destroy();
|
2010-05-27 02:59:55 +02:00
|
|
|
};
|
2009-07-14 18:31:50 +02:00
|
|
|
}
|
2009-05-18 19:33:05 +02:00
|
|
|
|
2009-07-16 10:59:40 +02:00
|
|
|
|
2010-05-27 02:59:55 +02:00
|
|
|
|
2010-03-20 03:22:04 +01:00
|
|
|
function Server (requestListener) {
|
2010-08-27 22:58:52 +02:00
|
|
|
if (!(this instanceof Server)) return new Server(requestListener);
|
2010-10-26 07:04:39 +02:00
|
|
|
net.Server.call(this, { allowHalfOpen: true });
|
2010-05-02 03:01:06 +02:00
|
|
|
|
2010-10-26 07:04:39 +02:00
|
|
|
if (requestListener) {
|
2010-04-30 15:35:51 +02:00
|
|
|
this.addListener("request", requestListener);
|
|
|
|
}
|
2010-05-02 03:01:06 +02:00
|
|
|
|
2010-03-20 03:22:04 +01:00
|
|
|
this.addListener("connection", connectionListener);
|
|
|
|
}
|
2010-10-11 23:04:09 +02:00
|
|
|
util.inherits(Server, net.Server);
|
2010-03-20 03:22:04 +01:00
|
|
|
|
|
|
|
exports.Server = Server;
|
|
|
|
|
|
|
|
exports.createServer = function (requestListener) {
|
|
|
|
return new Server(requestListener);
|
2009-07-14 18:31:50 +02:00
|
|
|
};
|
2009-05-18 19:33:05 +02:00
|
|
|
|
2010-03-20 03:22:04 +01:00
|
|
|
function connectionListener (socket) {
|
|
|
|
var self = this;
|
2010-05-27 02:59:55 +02:00
|
|
|
|
|
|
|
debug("new http connection");
|
|
|
|
|
|
|
|
httpSocketSetup(socket);
|
2009-05-18 19:33:05 +02:00
|
|
|
|
2010-05-12 20:42:01 +02:00
|
|
|
socket.setTimeout(2*60*1000); // 2 minute timeout
|
|
|
|
socket.addListener('timeout', function () {
|
|
|
|
socket.destroy();
|
|
|
|
});
|
|
|
|
|
2010-04-12 18:27:32 +02:00
|
|
|
var parser = parsers.alloc();
|
|
|
|
parser.reinitialize('request');
|
|
|
|
parser.socket = socket;
|
2010-03-20 03:22:04 +01:00
|
|
|
|
2010-04-23 02:22:03 +02:00
|
|
|
socket.addListener('error', function (e) {
|
2010-04-30 19:28:18 +02:00
|
|
|
self.emit('clientError', e);
|
2010-04-23 02:22:03 +02:00
|
|
|
});
|
|
|
|
|
2010-03-20 03:22:04 +01:00
|
|
|
socket.ondata = function (d, start, end) {
|
2010-05-21 00:21:40 +02:00
|
|
|
var ret = parser.execute(d, start, end - start);
|
|
|
|
if (ret instanceof Error) {
|
|
|
|
socket.destroy(ret);
|
|
|
|
} else if (parser.incoming && parser.incoming.upgrade) {
|
|
|
|
var bytesParsed = ret;
|
2010-04-14 12:52:15 +02:00
|
|
|
socket.ondata = null;
|
|
|
|
socket.onend = null;
|
2010-05-02 03:01:06 +02:00
|
|
|
|
2010-04-30 15:33:11 +02:00
|
|
|
var req = parser.incoming;
|
2010-05-02 03:01:06 +02:00
|
|
|
|
|
|
|
// This is start + byteParsed + 1 due to the error of getting \n
|
|
|
|
// in the upgradeHead from the closing lines of the headers
|
|
|
|
var upgradeHead = d.slice(start + bytesParsed + 1, end);
|
|
|
|
|
2010-05-03 20:23:36 +02:00
|
|
|
if (self.listeners("upgrade").length) {
|
2010-05-02 03:01:06 +02:00
|
|
|
self.emit('upgrade', req, req.socket, upgradeHead);
|
2010-04-30 15:33:11 +02:00
|
|
|
} else {
|
2010-05-03 20:23:36 +02:00
|
|
|
// Got upgrade header, but have no handler.
|
|
|
|
socket.destroy();
|
2010-04-30 15:33:11 +02:00
|
|
|
}
|
2010-04-14 12:52:15 +02:00
|
|
|
}
|
2010-03-20 03:22:04 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
socket.onend = function () {
|
|
|
|
parser.finish();
|
2010-05-21 00:21:40 +02:00
|
|
|
|
2010-05-27 02:59:55 +02:00
|
|
|
if (socket._outgoing.length) {
|
|
|
|
socket._outgoing[socket._outgoing.length-1]._last = true;
|
|
|
|
outgoingFlush(socket);
|
|
|
|
} else {
|
|
|
|
socket.end();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
socket.addListener('close', function () {
|
2010-03-20 03:22:04 +01:00
|
|
|
// unref the parser for easy gc
|
2010-04-12 18:27:32 +02:00
|
|
|
parsers.free(parser);
|
2010-05-27 02:59:55 +02:00
|
|
|
});
|
2009-11-21 16:27:02 +01:00
|
|
|
|
2010-05-27 02:59:55 +02:00
|
|
|
// At the end of each response message, after it has been flushed to the
|
|
|
|
// socket. Here we insert logic about what to do next.
|
|
|
|
socket._onOutgoingSent = function (message) {
|
|
|
|
var message = socket._outgoing.shift();
|
|
|
|
if (message._last) {
|
|
|
|
// No more messages to be pushed out.
|
2010-05-28 05:28:12 +02:00
|
|
|
|
|
|
|
// HACK: need way to do this with socket interface
|
2010-11-21 05:49:44 +01:00
|
|
|
if (socket._writeQueue.length) {
|
|
|
|
socket.__destroyOnDrain = true; //socket.end();
|
2010-05-28 05:28:12 +02:00
|
|
|
} else {
|
|
|
|
socket.destroy();
|
|
|
|
}
|
|
|
|
|
2010-05-27 02:59:55 +02:00
|
|
|
} else if (socket._outgoing.length) {
|
|
|
|
// Push out the next message.
|
|
|
|
outgoingFlush(socket);
|
2009-07-01 00:49:56 +02:00
|
|
|
}
|
2010-03-20 03:22:04 +01:00
|
|
|
};
|
2009-11-06 12:44:20 +01:00
|
|
|
|
2010-03-20 03:22:04 +01:00
|
|
|
// The following callback is issued after the headers have been read on a
|
|
|
|
// new message. In this callback we setup the response object and pass it
|
|
|
|
// to the user.
|
2010-03-24 05:31:44 +01:00
|
|
|
parser.onIncoming = function (req, shouldKeepAlive) {
|
2009-12-05 08:37:46 +01:00
|
|
|
var res = new ServerResponse(req);
|
2010-05-27 02:59:55 +02:00
|
|
|
debug('server response shouldKeepAlive: ' + shouldKeepAlive);
|
2010-03-24 05:31:44 +01:00
|
|
|
res.shouldKeepAlive = shouldKeepAlive;
|
2010-05-27 02:59:55 +02:00
|
|
|
socket._outgoing.push(res);
|
2009-08-26 22:03:19 +02:00
|
|
|
|
2010-09-10 03:56:35 +02:00
|
|
|
if ('expect' in req.headers
|
|
|
|
&& (req.httpVersionMajor == 1 && req.httpVersionMinor == 1)
|
|
|
|
&& continueExpression.test(req.headers['expect'])) {
|
|
|
|
res._expect_continue = true;
|
|
|
|
if (self.listeners("checkContinue").length) {
|
2010-10-07 05:05:23 +02:00
|
|
|
self.emit("checkContinue", req, res);
|
2010-09-10 03:56:35 +02:00
|
|
|
} else {
|
|
|
|
res.writeContinue();
|
|
|
|
self.emit('request', req, res);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
self.emit('request', req, res);
|
|
|
|
}
|
2010-05-26 04:24:30 +02:00
|
|
|
return false; // Not a HEAD response. (Not even a response!)
|
2010-03-20 03:22:04 +01:00
|
|
|
};
|
2009-07-14 18:31:50 +02:00
|
|
|
}
|
2009-06-26 12:51:27 +02:00
|
|
|
|
2009-06-26 18:30:55 +02:00
|
|
|
|
2010-03-20 03:22:04 +01:00
|
|
|
function Client ( ) {
|
2010-08-27 22:58:52 +02:00
|
|
|
if (!(this instanceof Client)) return new Client();
|
2010-10-26 07:04:39 +02:00
|
|
|
net.Stream.call(this, { allowHalfOpen: true });
|
2010-03-20 03:22:04 +01:00
|
|
|
var self = this;
|
2009-06-26 18:30:55 +02:00
|
|
|
|
2010-05-27 02:59:55 +02:00
|
|
|
httpSocketSetup(self);
|
|
|
|
|
2010-09-22 14:51:53 +02:00
|
|
|
function onData(d, start, end) {
|
2010-09-30 20:41:50 +02:00
|
|
|
if (!self.parser) {
|
2010-05-05 10:55:04 +02:00
|
|
|
throw new Error("parser not initialized prior to Client.ondata call");
|
|
|
|
}
|
2010-09-30 20:41:50 +02:00
|
|
|
var ret = self.parser.execute(d, start, end - start);
|
2010-05-21 00:21:40 +02:00
|
|
|
if (ret instanceof Error) {
|
|
|
|
self.destroy(ret);
|
2010-09-30 20:41:50 +02:00
|
|
|
} else if (self.parser.incoming && self.parser.incoming.upgrade) {
|
2010-05-21 00:21:40 +02:00
|
|
|
var bytesParsed = ret;
|
2010-04-14 12:52:15 +02:00
|
|
|
self.ondata = null;
|
2010-10-07 05:05:23 +02:00
|
|
|
self.onend = null;
|
2010-06-15 19:35:49 +02:00
|
|
|
|
2010-09-30 20:41:50 +02:00
|
|
|
var req = self.parser.incoming;
|
2010-06-15 19:35:49 +02:00
|
|
|
|
|
|
|
var upgradeHead = d.slice(start + bytesParsed + 1, end);
|
|
|
|
|
|
|
|
if (self.listeners('upgrade').length) {
|
|
|
|
self.emit('upgrade', req, self, upgradeHead);
|
|
|
|
} else {
|
|
|
|
self.destroy();
|
|
|
|
}
|
2010-04-14 12:52:15 +02:00
|
|
|
}
|
2010-03-20 03:22:04 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
self.addListener("connect", function () {
|
2010-05-27 02:59:55 +02:00
|
|
|
debug('client connected');
|
2010-09-22 14:51:53 +02:00
|
|
|
|
|
|
|
self.ondata = onData;
|
|
|
|
self.onend = onEnd;
|
|
|
|
|
2010-09-30 20:41:50 +02:00
|
|
|
self._initParser();
|
2010-10-11 23:04:09 +02:00
|
|
|
debug('requests: ' + util.inspect(self._outgoing));
|
2010-05-27 02:59:55 +02:00
|
|
|
outgoingFlush(self);
|
2009-06-26 18:29:57 +02:00
|
|
|
});
|
2009-05-06 14:54:28 +02:00
|
|
|
|
2010-09-22 14:51:53 +02:00
|
|
|
function onEnd() {
|
2010-09-30 20:41:50 +02:00
|
|
|
if (self.parser) self.parser.finish();
|
2010-03-20 05:49:00 +01:00
|
|
|
debug("self got end closing. readyState = " + self.readyState);
|
2010-04-08 19:44:22 +02:00
|
|
|
self.end();
|
2010-04-14 12:52:15 +02:00
|
|
|
};
|
2009-06-26 12:51:27 +02:00
|
|
|
|
2010-03-24 05:31:44 +01:00
|
|
|
self.addListener("close", function (e) {
|
2010-05-03 19:53:52 +02:00
|
|
|
if (e) return;
|
2009-08-26 22:03:19 +02:00
|
|
|
|
2010-03-20 05:49:00 +01:00
|
|
|
debug("HTTP CLIENT onClose. readyState = " + self.readyState);
|
2009-07-14 18:31:50 +02:00
|
|
|
|
2010-05-27 02:59:55 +02:00
|
|
|
// finally done with the request
|
|
|
|
self._outgoing.shift();
|
|
|
|
|
2009-06-26 18:29:57 +02:00
|
|
|
// If there are more requests to handle, reconnect.
|
2010-05-27 02:59:55 +02:00
|
|
|
if (self._outgoing.length) {
|
2010-03-20 03:22:04 +01:00
|
|
|
self._reconnect();
|
2010-09-30 20:41:50 +02:00
|
|
|
} else if (self.parser) {
|
|
|
|
parsers.free(self.parser);
|
|
|
|
self.parser = null;
|
2009-06-26 18:29:57 +02:00
|
|
|
}
|
|
|
|
});
|
|
|
|
};
|
2010-10-11 23:04:09 +02:00
|
|
|
util.inherits(Client, net.Stream);
|
2010-03-20 03:22:04 +01:00
|
|
|
|
|
|
|
exports.Client = Client;
|
|
|
|
|
2010-04-12 19:57:22 +02:00
|
|
|
exports.createClient = function (port, host, https, credentials) {
|
2010-05-27 02:59:55 +02:00
|
|
|
var c = new Client();
|
2010-03-20 03:22:04 +01:00
|
|
|
c.port = port;
|
|
|
|
c.host = host;
|
2010-04-12 19:57:22 +02:00
|
|
|
c.https = https;
|
|
|
|
c.credentials = credentials;
|
2010-03-20 03:22:04 +01:00
|
|
|
return c;
|
2010-05-27 02:59:55 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2010-09-30 20:41:50 +02:00
|
|
|
Client.prototype._initParser = function () {
|
|
|
|
var self = this;
|
|
|
|
if (!self.parser) self.parser = parsers.alloc();
|
|
|
|
self.parser.reinitialize('response');
|
|
|
|
self.parser.socket = self;
|
|
|
|
self.parser.onIncoming = function (res) {
|
|
|
|
debug("incoming response!");
|
|
|
|
|
|
|
|
var req = self._outgoing[0];
|
|
|
|
|
|
|
|
// Responses to HEAD requests are AWFUL. Ask Ryan.
|
|
|
|
// A major oversight in HTTP. Hence this nastiness.
|
|
|
|
var isHeadResponse = req.method == "HEAD";
|
|
|
|
debug('isHeadResponse ' + isHeadResponse);
|
2010-09-10 03:56:35 +02:00
|
|
|
|
|
|
|
if (res.statusCode == 100) {
|
|
|
|
// restart the parser, as this is a continue message.
|
|
|
|
req.emit("continue");
|
|
|
|
return true;
|
|
|
|
}
|
2010-09-30 20:41:50 +02:00
|
|
|
|
|
|
|
if (req.shouldKeepAlive && res.headers.connection === 'close') {
|
|
|
|
req.shouldKeepAlive = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
res.addListener('end', function ( ) {
|
|
|
|
debug("request complete disconnecting. readyState = " + self.readyState);
|
|
|
|
// For the moment we reconnect for every request. FIXME!
|
|
|
|
// All that should be required for keep-alive is to not reconnect,
|
|
|
|
// but outgoingFlush instead.
|
|
|
|
if (req.shouldKeepAlive) {
|
2010-10-07 05:05:23 +02:00
|
|
|
outgoingFlush(self);
|
|
|
|
self._outgoing.shift();
|
|
|
|
outgoingFlush(self);
|
2010-09-30 20:41:50 +02:00
|
|
|
} else {
|
|
|
|
self.end();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
req.emit("response", res);
|
|
|
|
|
|
|
|
return isHeadResponse;
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2010-05-27 02:59:55 +02:00
|
|
|
// This is called each time a request has been pushed completely to the
|
|
|
|
// socket. The message that was sent is still sitting at client._outgoing[0]
|
|
|
|
// it is our responsibility to shift it off.
|
|
|
|
//
|
|
|
|
// We have to be careful when it we shift it because once we do any writes
|
|
|
|
// to other requests will be flushed directly to the socket.
|
|
|
|
//
|
|
|
|
// At the moment we're implement a client which connects and disconnects on
|
|
|
|
// each request/response cycle so we cannot shift off the request from
|
|
|
|
// client._outgoing until we're completely disconnected after the response
|
|
|
|
// comes back.
|
|
|
|
Client.prototype._onOutgoingSent = function (message) {
|
|
|
|
// We've just finished a message. We don't end/shutdown the connection here
|
|
|
|
// because HTTP servers typically cannot handle half-closed connections
|
|
|
|
// (Node servers can).
|
|
|
|
//
|
|
|
|
// Instead, we just check if the connection is closed, and if so
|
|
|
|
// reconnect if we have pending messages.
|
|
|
|
if (this._outgoing.length && this.readyState == "closed") {
|
|
|
|
debug("HTTP client request flush. reconnect. readyState = " + this.readyState);
|
|
|
|
this._reconnect();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
Client.prototype._reconnect = function () {
|
|
|
|
if (this.readyState === "closed") {
|
|
|
|
debug("HTTP CLIENT: reconnecting readyState = " + this.readyState);
|
|
|
|
this.connect(this.port, this.host);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2010-03-20 03:22:04 +01:00
|
|
|
Client.prototype.request = function (method, url, headers) {
|
2010-05-27 02:59:55 +02:00
|
|
|
if (typeof(url) != "string") {
|
|
|
|
// assume method was omitted, shift arguments
|
2010-01-05 06:07:50 +01:00
|
|
|
headers = url;
|
|
|
|
url = method;
|
2010-05-27 02:59:55 +02:00
|
|
|
method = "GET";
|
2009-12-02 23:25:56 +01:00
|
|
|
}
|
2010-05-27 02:59:55 +02:00
|
|
|
var req = new ClientRequest(this, method, url, headers);
|
|
|
|
this._outgoing.push(req);
|
|
|
|
if (this.readyState === 'closed') this._reconnect();
|
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
|
|
|
|
2010-02-20 01:26:48 +01:00
|
|
|
exports.cat = function (url, encoding_, headers_) {
|
2010-03-20 03:22:04 +01:00
|
|
|
var encoding = 'utf8',
|
2010-02-20 01:26:48 +01:00
|
|
|
headers = {},
|
|
|
|
callback = null;
|
|
|
|
|
|
|
|
// parse the arguments for the various options... very ugly
|
|
|
|
if (typeof(arguments[1]) == 'string') {
|
|
|
|
encoding = arguments[1];
|
|
|
|
if (typeof(arguments[2]) == 'object') {
|
|
|
|
headers = arguments[2];
|
|
|
|
if (typeof(arguments[3]) == 'function') callback = arguments[3];
|
|
|
|
} else {
|
|
|
|
if (typeof(arguments[2]) == 'function') callback = arguments[2];
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// didn't specify encoding
|
|
|
|
if (typeof(arguments[1]) == 'object') {
|
|
|
|
headers = arguments[1];
|
|
|
|
callback = arguments[2];
|
|
|
|
} else {
|
|
|
|
callback = arguments[1];
|
|
|
|
}
|
|
|
|
}
|
2009-09-15 22:41:25 +02:00
|
|
|
|
2010-01-05 06:07:50 +01:00
|
|
|
var url = require("url").parse(url);
|
2010-05-05 06:35:46 +02:00
|
|
|
|
2009-12-12 00:46:32 +01:00
|
|
|
var hasHost = false;
|
2010-07-14 23:29:28 +02:00
|
|
|
if (Array.isArray(headers)) {
|
2010-04-12 18:57:24 +02:00
|
|
|
for (var i = 0, l = headers.length; i < l; i++) {
|
|
|
|
if (headers[i][0].toLowerCase() === 'host') {
|
|
|
|
hasHost = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (typeof headers === "Object") {
|
|
|
|
var keys = Object.keys(headers);
|
|
|
|
for (var i = 0, l = keys.length; i < l; i++) {
|
|
|
|
var key = keys[i];
|
|
|
|
if (key.toLowerCase() == 'host') {
|
|
|
|
hasHost = true;
|
|
|
|
break;
|
|
|
|
}
|
2010-02-20 01:26:48 +01:00
|
|
|
}
|
2009-09-30 11:54:14 +02:00
|
|
|
}
|
2010-02-20 01:26:48 +01:00
|
|
|
if (!hasHost) headers["Host"] = url.hostname;
|
2010-02-17 20:10:10 +01:00
|
|
|
|
|
|
|
var content = "";
|
2010-05-05 06:35:46 +02:00
|
|
|
|
2010-01-05 06:07:50 +01:00
|
|
|
var client = exports.createClient(url.port || 80, url.hostname);
|
|
|
|
var req = client.request((url.pathname || "/")+(url.search || "")+(url.hash || ""), headers);
|
2009-06-28 19:05:58 +02:00
|
|
|
|
2010-04-12 19:57:22 +02:00
|
|
|
if (url.protocol=="https:") {
|
|
|
|
client.https = true;
|
|
|
|
}
|
|
|
|
|
2010-05-12 20:42:01 +02:00
|
|
|
var callbackSent = false;
|
|
|
|
|
2010-02-17 20:10:10 +01:00
|
|
|
req.addListener('response', function (res) {
|
2009-06-28 19:05:58 +02:00
|
|
|
if (res.statusCode < 200 || res.statusCode >= 300) {
|
2010-05-12 20:42:01 +02:00
|
|
|
if (callback && !callbackSent) {
|
|
|
|
callback(res.statusCode);
|
|
|
|
callbackSent = true;
|
|
|
|
}
|
2010-04-08 19:44:22 +02:00
|
|
|
client.end();
|
2009-06-28 19:05:58 +02:00
|
|
|
return;
|
|
|
|
}
|
2010-05-28 05:28:12 +02:00
|
|
|
res.setEncoding(encoding);
|
2010-02-17 07:16:29 +01:00
|
|
|
res.addListener('data', function (chunk) { content += chunk; });
|
|
|
|
res.addListener('end', function () {
|
2010-05-12 20:42:01 +02:00
|
|
|
if (callback && !callbackSent) {
|
|
|
|
callback(null, content);
|
|
|
|
callbackSent = true;
|
|
|
|
}
|
2009-06-27 20:40:43 +02:00
|
|
|
});
|
2009-06-22 13:12:47 +02:00
|
|
|
});
|
2009-06-28 19:05:58 +02:00
|
|
|
|
2010-02-20 01:26:48 +01:00
|
|
|
client.addListener("error", function (err) {
|
2010-05-12 20:42:01 +02:00
|
|
|
if (callback && !callbackSent) {
|
|
|
|
callback(err);
|
|
|
|
callbackSent = true;
|
|
|
|
}
|
2010-02-17 20:10:10 +01:00
|
|
|
});
|
|
|
|
|
2010-05-12 20:42:01 +02:00
|
|
|
client.addListener("close", function () {
|
|
|
|
if (callback && !callbackSent) {
|
|
|
|
callback(new Error('Connection closed unexpectedly'));
|
|
|
|
callbackSent = true;
|
|
|
|
}
|
|
|
|
});
|
2010-04-08 19:44:22 +02:00
|
|
|
req.end();
|
2009-06-22 13:12:47 +02:00
|
|
|
};
|