mirror of
https://github.com/bitinflow/server.git
synced 2026-03-13 13:35:53 +00:00
- http raw exposes a socket to the req/res pair - cleanup client to be more resilient - add test for queued requests
323 lines
8.0 KiB
JavaScript
323 lines
8.0 KiB
JavaScript
|
|
// builtin
|
|
var http = require('http');
|
|
var net = require('net');
|
|
var url = require('url');
|
|
|
|
// here be dragons
|
|
var HTTPParser = process.binding('http_parser').HTTPParser;
|
|
|
|
// vendor
|
|
var log = require('book');
|
|
var createRawServer = require('http-raw');
|
|
|
|
// local
|
|
var rand_id = require('./lib/rand_id');
|
|
|
|
// id -> client http server
|
|
var clients = {};
|
|
|
|
// available parsers
|
|
var parsers = http.parsers;
|
|
|
|
// send this request to the appropriate client
|
|
// in -> incoming request stream
|
|
function proxy_request(client, req, res, rs, ws) {
|
|
|
|
rs = rs || req.createRawStream();
|
|
ws = ws || res.createRawStream();
|
|
|
|
// socket is a tcp connection back to the user hosting the site
|
|
var sock = client.sockets.shift();
|
|
|
|
// queue request
|
|
if (!sock) {
|
|
log.info('no more clients, queued: %s', req.url);
|
|
rs.pause();
|
|
client.waiting.push([req, res, rs, ws]);
|
|
return;
|
|
}
|
|
|
|
log.info('handle req: %s', req.url);
|
|
|
|
// pipe incoming request into tcp socket
|
|
// incoming request isn't allowed to end the socket back to lt client
|
|
rs.pipe(sock, { end: false });
|
|
|
|
sock.ws = ws;
|
|
sock.req = req;
|
|
|
|
// since tcp connection to upstream are kept open
|
|
// invoke parsing so we know when the response is complete
|
|
var parser = sock.parser;
|
|
parser.reinitialize(HTTPParser.RESPONSE);
|
|
parser.socket = sock;
|
|
|
|
// we have completed a response
|
|
// the tcp socket is free again
|
|
parser.onIncoming = function (res) {
|
|
parser.onMessageComplete = function() {
|
|
log.info('ended response: %s', req.url);
|
|
|
|
// any request we had going on is now done
|
|
ws.end();
|
|
|
|
// no more forwarding
|
|
delete sock.ws;
|
|
delete parser.onIncoming;
|
|
|
|
// return socket to available pool
|
|
client.sockets.push(sock);
|
|
|
|
var next = client.waiting.shift();
|
|
if (next) {
|
|
log.trace('popped');
|
|
proxy_request(client, next[0], next[1], next[2], next[3]);
|
|
}
|
|
};
|
|
};
|
|
|
|
rs.resume();
|
|
}
|
|
|
|
function upstream_response(d, start, end) {
|
|
var socket = this;
|
|
|
|
var ws = socket.ws;
|
|
if (!ws) {
|
|
log.warn('no stream set for req:', socket.req.url);
|
|
return;
|
|
}
|
|
|
|
ws.write(d.slice(start, end));
|
|
|
|
if (socket.upgraded) {
|
|
return;
|
|
}
|
|
|
|
var ret = socket.parser.execute(d, start, end - start);
|
|
if (ret instanceof Error) {
|
|
log.error(ret);
|
|
parsers.free(parser);
|
|
socket.destroy(ret);
|
|
}
|
|
}
|
|
|
|
var handle_req = function (req, res) {
|
|
|
|
var max_tcp_sockets = req.socket.server.max_tcp_sockets;
|
|
|
|
// ignore favicon
|
|
if (req.url === '/favicon.ico') {
|
|
res.writeHead(404);
|
|
return res.end();
|
|
}
|
|
|
|
var hostname = req.headers.host;
|
|
if (!hostname) {
|
|
log.trace('no hostname: %j', req.headers);
|
|
return res.end();
|
|
}
|
|
|
|
var match = hostname.match(/^([a-z]{4})[.].*/);
|
|
if (match) {
|
|
var client_id = match[1];
|
|
var client = clients[client_id];
|
|
|
|
// no such subdomain
|
|
if (!client) {
|
|
log.trace('no client found for id: ' + client_id);
|
|
res.statusCode = 404;
|
|
return res.end();
|
|
}
|
|
|
|
return proxy_request(client, req, res);
|
|
}
|
|
|
|
// redirect main page to github reference
|
|
if (req.url === '/' && !parsed.query.new) {
|
|
res.writeHead(301, { Location: 'http://shtylman.github.com/localtunnel/' });
|
|
res.end();
|
|
return;
|
|
}
|
|
|
|
// at this point, the client is requesting a new tunnel setup
|
|
// either generate an id or use the one they requested
|
|
|
|
var match = req.url.match(/\/([a-z]{4})?/);
|
|
|
|
// user can request a particular set of characters
|
|
// will be given if not already taken
|
|
// this is useful when the main server is restarted
|
|
// users can keep testing with their expected ids
|
|
var requested_id;
|
|
if (match && match[1]) {
|
|
requested_id = match[1];
|
|
}
|
|
|
|
var id = requested_id || rand_id();
|
|
|
|
// if the id already exists, this client must use something else
|
|
if (clients[id]) {
|
|
id = rand_id();
|
|
}
|
|
|
|
// sockets is a list of available sockets for the connection
|
|
// waiting is?
|
|
var client = clients[id] = {
|
|
sockets: [],
|
|
waiting: []
|
|
};
|
|
|
|
var client_server = net.createServer();
|
|
client_server.listen(function() {
|
|
var port = client_server.address().port;
|
|
log.info('tcp server listening on port: %d', port);
|
|
|
|
var url = 'http://' + id + '.' + req.headers.host;
|
|
|
|
res.writeHead(200, { 'Content-Type': 'application/json' });
|
|
res.end(JSON.stringify({
|
|
url: url,
|
|
id: id,
|
|
port: port,
|
|
max_conn_count: max_tcp_sockets
|
|
}));
|
|
});
|
|
|
|
var conn_timeout;
|
|
|
|
// user has 5 seconds to connect before their slot is given up
|
|
function maybe_tcp_close() {
|
|
conn_timeout = setTimeout(client_server.close.bind(client_server), 5000);
|
|
}
|
|
|
|
maybe_tcp_close();
|
|
|
|
// no longer accepting connections for this id
|
|
client_server.on('close', function() {
|
|
log.trace('closed tcp socket for client(%s)', id);
|
|
clearTimeout(conn_timeout);
|
|
delete clients[id];
|
|
});
|
|
|
|
client_server.on('connection', function(socket) {
|
|
|
|
// no more socket connections allowed
|
|
if (client.sockets.length >= max_tcp_sockets) {
|
|
return socket.end();
|
|
}
|
|
|
|
log.trace('new connection for id: %s', id);
|
|
|
|
// no need to close the client server
|
|
clearTimeout(conn_timeout);
|
|
|
|
// allocate a response parser for the socket
|
|
// it only needs one since it will reuse it
|
|
socket.parser = parsers.alloc();
|
|
|
|
socket._orig_ondata = socket.ondata;
|
|
socket.ondata = upstream_response;
|
|
|
|
client.sockets.push(socket);
|
|
|
|
socket.once('close', function(had_error) {
|
|
log.trace('client %s closed socket', id);
|
|
|
|
// remove this socket
|
|
var idx = client.sockets.indexOf(socket);
|
|
client.sockets.splice(idx, 1);
|
|
|
|
log.trace('remaining client sockets: %s', client.sockets.length);
|
|
|
|
// no more sockets for this ident
|
|
if (client.sockets.length === 0) {
|
|
log.trace('all client(%s) sockets disconnected', id);
|
|
maybe_tcp_close();
|
|
}
|
|
});
|
|
|
|
// close will be emitted after this
|
|
socket.on('error', function(err) {
|
|
log.error(err);
|
|
socket.end();
|
|
});
|
|
});
|
|
|
|
client_server.on('error', function(err) {
|
|
log.error(err);
|
|
});
|
|
};
|
|
|
|
var handle_upgrade = function(req, ws) {
|
|
|
|
if (req.headers.connection !== 'Upgrade') {
|
|
return;
|
|
}
|
|
|
|
var hostname = req.headers.host;
|
|
if (!hostname) {
|
|
return res.end();
|
|
}
|
|
|
|
var match = hostname.match(/^([a-z]{4})[.].*/);
|
|
|
|
// not a valid client
|
|
if (!match) {
|
|
return res.end();
|
|
}
|
|
|
|
var client_id = match[1];
|
|
var client = clients[client_id];
|
|
|
|
if (!client) {
|
|
// no such subdomain
|
|
return res.end();
|
|
}
|
|
|
|
var socket = client.sockets.shift();
|
|
if (!socket) {
|
|
// no available sockets to upgrade to
|
|
return res.end();
|
|
}
|
|
|
|
var stream = req.createRawStream();
|
|
|
|
socket.ws = ws;
|
|
socket.upgraded = true;
|
|
|
|
stream.once('end', function() {
|
|
delete socket.ws;
|
|
|
|
// when this ends, we just reset the socket to the lt client
|
|
// this is easier than trying to figure anything else out
|
|
socket.end();
|
|
|
|
// put socket back into available pool
|
|
client.sockets.push(socket);
|
|
|
|
var next = client.waiting.shift();
|
|
if (next) {
|
|
log.trace('popped');
|
|
proxy_request(client, next[0], next[1], next[2], next[3]);
|
|
}
|
|
});
|
|
|
|
stream.pipe(socket, {end: false});
|
|
socket.once('end', ws.end.bind(ws));
|
|
};
|
|
|
|
module.exports = function(opt) {
|
|
opt = opt || {};
|
|
|
|
var server = createRawServer();
|
|
|
|
server.max_tcp_sockets = opt.max_tcp_sockets || 5;
|
|
server.on('request', handle_req);
|
|
server.on('upgrade', handle_upgrade);
|
|
|
|
return server;
|
|
};
|
|
|