0
0
mirror of https://github.com/nodejs/node.git synced 2024-11-30 07:27:22 +01:00
nodejs/src/http.js

219 lines
6.6 KiB
JavaScript
Raw Normal View History

/* This is a wrapper around the LowLevelServer interface. It provides
* connection handling, overflow checking, and some data buffering.
*/
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;
node.http.Server = function (RequestHandler, options) {
var MAX_FIELD_SIZE = 80*1024;
function Protocol (connection) {
function fillField (obj, field, data) {
obj[field] = (obj[field] || "") + data;
if (obj[field].length > MAX_FIELD_SIZE) {
connection.fullClose();
return false;
}
return true;
}
var responses = [];
function Response () {
responses.push(this);
/* This annoying output buisness is necessary for the case that users
* are writing to responses out of order! HTTP requires that responses
* are returned in the same order the requests come.
*/
var output = [];
function toRaw(string) {
var a = [];
for (var i = 0; i < string.length; i++)
a.push(string.charCodeAt(i));
return a;
}
// The send method appends data onto the output array. The deal is,
// the data is either an array of integer, representing binary or it
// is a string in which case it's UTF8 encoded.
// Two things to considered:
// - we should be able to send mixed encodings.
// - we don't want to call connection.send("smallstring") because that
// is wasteful. *I think* its rather faster to concat inside of JS
// Thus I attempt to concat as much as possible.
function send (data) {
if (output.length == 0) {
output.push(data);
return;
}
var li = output.length-1;
if (data.constructor == String && output[li].constructor == String) {
output[li] += data;
return;
}
if (data.constructor == Array && output[li].constructor == Array) {
output[li] = output[li].concat(data);
return;
}
// If the string is small enough, just convert it to binary
if (data.constructor == String
&& data.length < 128
&& output[li].constructor == Array)
{
output[li] = output[li].concat(toRaw(data));
return;
}
output.push(data);
};
this.flush = function () {
if (responses.length > 0 && responses[0] === this)
while (output.length > 0)
connection.send(output.shift());
};
2009-05-06 19:07:43 +02:00
var chunked_encoding = false;
var connection_close = false;
this.sendHeader = function (status, headers) {
var sent_connection_header = false;
var sent_transfer_encoding_header = false;
var sent_content_length_header = false;
var reason = "Ok"; // FIXME
var header = "HTTP/1.0 ";
header += status.toString() + " " + reason + "\r\n";
for (var i = 0; i < headers.length; i++) {
var field = headers[i][0];
var value = headers[i][1];
header += field + ": " + value + "\r\n";
if (connection_expression.exec(field)) {
sent_connection_header = true;
if (close_expression.exec(value))
connection_close = true;
} else if (transfer_encoding_expression.exec(field)) {
sent_transfer_encoding_header = true;
if (chunk_expression.exec(value))
chunked_encoding = true;
} else if (content_length_expression.exec(field)) {
sent_content_length_header = true;
}
}
if (sent_connection_header == false)
header += "Connection: keep-alive\r\n";
if (sent_content_length_header == false && sent_transfer_encoding_header == false) {
header += "Transfer-Encoding: chunked\r\n";
chunked_encoding = true;
}
header += "\r\n";
send(header);
};
2009-05-06 19:07:43 +02:00
var bodyBegan = false;
this.sendBody = function (chunk) {
if (chunked_encoding) {
send(chunk.length.toString(16));
send("\r\n");
send(chunk);
send("\r\n");
} else {
send(chunk);
}
this.flush();
};
var finished = false;
this.finish = function () {
2009-05-06 19:07:43 +02:00
if (chunked_encoding)
send("0\r\n\r\n"); // last chunk
2009-05-06 19:07:43 +02:00
this.finished = true;
2009-05-06 19:07:43 +02:00
while (responses.length > 0 && responses[0].finished) {
var res = responses[0];
res.flush();
responses.shift();
}
if (responses.length == 0) {
connection.fullClose();
// TODO keep-alive logic
// if http/1.0 without "Connection: keep-alive" header - yes
// if http/1.1 if the request was not GET or HEAD - yes
// if http/1.1 without "Connection: close" header - yes
}
};
}
this.onMessage = function ( ) {
var res = new Response();
var req = new RequestHandler(res, connection);
this.encoding = req.encoding || "raw";
this.onPath = function (data) { return fillField(req, "path", data); };
this.onURI = function (data) { return fillField(req, "uri", data); };
this.onQueryString = function (data) { return fillField(req, "query_string", data); };
this.onFragment = function (data) { return fillField(req, "fragment", data); };
this.onHeaderField = function (data) {
if (req.hasOwnProperty("headers")) {
var last_pair = req.headers[req.headers.length-1];
if (last_pair.length == 1)
return fillField(last_pair, 0, data);
else
req.headers.push([data]);
} else {
req.headers = [[data]];
}
return true;
};
this.onHeaderValue = function (data) {
var last_pair = req.headers[req.headers.length-1];
if (last_pair.length == 1)
last_pair[1] = data;
else
return fillField(last_pair, 1, data);
return true;
};
this.onHeadersComplete = function () {
req.http_version = this.http_version;
req.method = this.method;
return req.onHeadersComplete();
};
this.onBody = function (chunk) { return req.onBody(chunk); };
this.onBodyComplete = function () { return req.onBodyComplete(); };
};
}
var server = new node.http.LowLevelServer(Protocol, options);
this.listen = function (port, host) { server.listen(port, host); }
this.close = function () { server.close(); }
};