2009-05-06 14:54:28 +02:00
|
|
|
/* This is a wrapper around the LowLevelServer interface. It provides
|
|
|
|
* connection handling, overflow checking, and some data buffering.
|
|
|
|
*/
|
|
|
|
node.http.Server = function (RequestHandler, options) {
|
|
|
|
var MAX_FIELD_SIZE = 80*1024;
|
|
|
|
function Protocol (connection) {
|
2009-05-06 15:03:13 +02:00
|
|
|
function fillField (obj, field, data) {
|
|
|
|
obj[field] = (obj[field] || "") + data;
|
|
|
|
if (obj[field].length > MAX_FIELD_SIZE) {
|
2009-05-06 14:54:28 +02:00
|
|
|
connection.fullClose();
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
2009-05-03 21:06:20 +02:00
|
|
|
}
|
2009-05-06 14:54:28 +02:00
|
|
|
|
|
|
|
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.
|
|
|
|
*/
|
|
|
|
|
2009-05-07 12:15:01 +02:00
|
|
|
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.
|
2009-05-06 14:54:28 +02:00
|
|
|
function send (data) {
|
2009-05-07 12:15:01 +02:00
|
|
|
if (output.length == 0) {
|
|
|
|
output.push(data);
|
|
|
|
return;
|
2009-05-06 14:54:28 +02:00
|
|
|
}
|
2009-05-07 12:15:01 +02:00
|
|
|
|
|
|
|
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 14:54:28 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
this.sendStatus = function (status, reason) {
|
2009-05-06 19:07:43 +02:00
|
|
|
// XXX http/1.0 until i get the keep-alive logic below working.
|
2009-05-07 12:15:01 +02:00
|
|
|
send("HTTP/1.0 " + status.toString() + " " + reason + "\r\n");
|
2009-05-06 14:54:28 +02:00
|
|
|
};
|
|
|
|
|
2009-05-06 19:07:43 +02:00
|
|
|
var chunked_encoding = false;
|
|
|
|
var connection_close = false;
|
|
|
|
|
2009-05-06 14:54:28 +02:00
|
|
|
this.sendHeader = function (field, value) {
|
2009-05-07 12:15:01 +02:00
|
|
|
send(field + ": " + value.toString() + "\r\n");
|
|
|
|
|
2009-05-06 19:07:43 +02:00
|
|
|
if (/Connection/i.exec(field) && /close/i.exec(value))
|
|
|
|
connection_close = true;
|
2009-05-07 12:15:01 +02:00
|
|
|
|
2009-05-06 19:07:43 +02:00
|
|
|
else if (/Transfer-Encoding/i.exec(field) && /chunk/i.exec(value))
|
|
|
|
chunked_encoding = true;
|
2009-05-06 14:54:28 +02:00
|
|
|
};
|
|
|
|
|
2009-05-06 19:07:43 +02:00
|
|
|
var bodyBegan = false;
|
|
|
|
|
2009-05-06 14:54:28 +02:00
|
|
|
this.sendBody = function (chunk) {
|
2009-05-06 19:07:43 +02:00
|
|
|
if (bodyBegan === false) {
|
2009-05-07 12:15:01 +02:00
|
|
|
send("\r\n");
|
2009-05-06 19:07:43 +02:00
|
|
|
bodyBegan = true;
|
2009-05-06 14:54:28 +02:00
|
|
|
}
|
2009-05-06 19:07:43 +02:00
|
|
|
|
2009-05-07 12:15:01 +02:00
|
|
|
if (chunked_encoding) {
|
|
|
|
send(chunk.length.toString(16));
|
|
|
|
send("\r\n");
|
|
|
|
send(chunk);
|
|
|
|
send("\r\n");
|
|
|
|
} else {
|
|
|
|
send(chunk);
|
2009-05-06 14:54:28 +02:00
|
|
|
}
|
2009-05-07 12:15:01 +02:00
|
|
|
|
|
|
|
this.flush();
|
2009-05-06 14:54:28 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
var finished = false;
|
|
|
|
this.finish = function () {
|
2009-05-06 19:07:43 +02:00
|
|
|
if (chunked_encoding)
|
2009-05-07 12:15:01 +02:00
|
|
|
send("0\r\n\r\n"); // last chunk
|
2009-05-06 19:07:43 +02:00
|
|
|
|
2009-05-06 14:54:28 +02:00
|
|
|
this.finished = true;
|
2009-05-06 19:07:43 +02:00
|
|
|
|
2009-05-06 14:54:28 +02:00
|
|
|
while (responses.length > 0 && responses[0].finished) {
|
2009-05-07 12:15:01 +02:00
|
|
|
var res = responses[0];
|
|
|
|
res.flush();
|
|
|
|
responses.shift();
|
2009-05-06 14:54:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
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();
|
2009-05-07 12:15:01 +02:00
|
|
|
var req = new RequestHandler(res, connection);
|
2009-05-06 14:54:28 +02:00
|
|
|
|
|
|
|
this.encoding = req.encoding || "raw";
|
|
|
|
|
2009-05-06 15:03:13 +02:00
|
|
|
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); };
|
2009-05-06 14:54:28 +02:00
|
|
|
|
|
|
|
this.onHeaderField = function (data) {
|
|
|
|
if (req.hasOwnProperty("headers")) {
|
|
|
|
var last_pair = req.headers[req.headers.length-1];
|
|
|
|
if (last_pair.length == 1)
|
2009-05-06 15:03:13 +02:00
|
|
|
return fillField(last_pair, 0, data);
|
2009-05-06 14:54:28 +02:00
|
|
|
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
|
2009-05-06 15:03:13 +02:00
|
|
|
return fillField(last_pair, 1, data);
|
2009-05-06 14:54:28 +02:00
|
|
|
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(); };
|
|
|
|
};
|
2009-05-03 21:06:20 +02:00
|
|
|
}
|
2009-05-06 14:54:28 +02:00
|
|
|
|
|
|
|
var server = new node.http.LowLevelServer(Protocol, options);
|
|
|
|
this.listen = function (port, host) { server.listen(port, host); }
|
|
|
|
this.close = function () { server.close(); }
|
2009-05-03 21:06:20 +02:00
|
|
|
};
|