4 Commits

Author SHA1 Message Date
Roman Shtylman
c46a94b7a0 0.0.2 2012-11-03 15:33:32 -04:00
Roman Shtylman
2f692b8e29 expose client as a library
- Allows for using localtunnel from code instead of manually invoking
- add tests
- add travis config
- add travis badge
2012-11-03 15:16:30 -04:00
Roman Shtylman
51d91ce0e8 refactor server tcp handling
- limit on number of tcp connections
- preliminary support for websockets
2012-10-17 22:50:59 -04:00
Roman Shtylman
ab28444802 add server launcher to bin 2012-10-17 18:59:09 -04:00
10 changed files with 494 additions and 203 deletions

3
.travis.yml Normal file
View File

@@ -0,0 +1,3 @@
language: node_js
node_js:
- 0.8

View File

@@ -1,4 +1,4 @@
# localtunnel # # localtunnel [![Build Status](https://secure.travis-ci.org/shtylman/localtunnel.png)](http://travis-ci.org/shtylman/localtunnel) #
localtunnel exposes your localhost to the world for easy testing and sharing! No need to mess with DNS or deploy just to have others test out your changes. localtunnel exposes your localhost to the world for easy testing and sharing! No need to mess with DNS or deploy just to have others test out your changes.
@@ -21,3 +21,28 @@ lt --port 8000
Thats it! It will connect to the tunnel server, setup the tunnel, and tell you what url to use for your testing. This url will remain active for the duration of your session; so feel free to share it with others for happy fun time! Thats it! It will connect to the tunnel server, setup the tunnel, and tell you what url to use for your testing. This url will remain active for the duration of your session; so feel free to share it with others for happy fun time!
You can restart your local server all you want, ```lt``` is smart enough to detect this and reconnect once it is back. You can restart your local server all you want, ```lt``` is smart enough to detect this and reconnect once it is back.
## API ##
The localtunnel client is also usable through an API (test integration, automation, etc)
```javascript
var lt_client = require('localtunnel').client;
var client = lt_client.connect({
// the localtunnel server
host: 'http://localtunnel.com',
// your local application port
port: 12345
});
// when your are assigned a url
client.on('url', function(url) {
// you can now make http requests to the url
// they will be proxied to your local server on port [12345]
});
client.on('error', function(err) {
// uh oh!
});
```

34
bin/client Executable file
View File

@@ -0,0 +1,34 @@
#!/usr/bin/env node
var lt_client = require(__dirname + '/../client');
var argv = require('optimist')
.usage('Usage: $0 --port [num]')
.demand(['port'])
.options('host', {
default: 'http://localtunnel.me',
describe: 'upstream server providing forwarding'
})
.options('subdomain', {
describe: 'request this subdomain'
})
.describe('port', 'internal http server port')
.argv;
var opt = {
host: argv.host,
port: argv.port,
subdomain: argv.subdomain,
}
var client = lt_client.connect(opt);
// only emitted when the url changes
client.on('url', function(url) {
console.log('your url is: %s', url);
});
client.on('error', function(err) {
console.error(err);
});
// vim: ft=javascript

2
bin/lt
View File

@@ -1,2 +0,0 @@
#!/usr/bin/env node
require(__dirname + '/../client');

33
bin/server Executable file
View File

@@ -0,0 +1,33 @@
#!/usr/bin/env node
// vendor
var log = require('book');
var optimist = require('optimist');
var argv = optimist
.usage('Usage: $0 --port [num]')
.options('port', {
default: '80',
describe: 'listen on this port for outside requests'
})
.argv;
if (argv.help) {
optimist.showHelp();
process.exit();
}
process.once('uncaughtException', function(err) {
log.panic(err);
process.exit(-1);
return;
});
var server = require('../server');
server.listen(argv.port, function() {
log.info('server listening on port: %d', server.address().port);
});
// vim: ft=javascript

164
client.js
View File

@@ -2,94 +2,132 @@
var net = require('net'); var net = require('net');
var url = require('url'); var url = require('url');
var request = require('request'); var request = require('request');
var EventEmitter = require('events').EventEmitter;
var argv = require('optimist') // request upstream url and connection info
.usage('Usage: $0 --port [num]') var request_url = function(params, cb) {
.demand(['port']) request(params, function(err, res, body) {
.options('host', { if (err) {
default: 'http://localtunnel.me', cb(err);
describe: 'upstream server providing forwarding' }
})
.describe('port', 'internal http server port')
.argv;
// local port cb(null, body);
var local_port = argv.port; });
// optionally override the upstream server
var upstream = url.parse(argv.host);
// query options
var opt = {
host: upstream.hostname,
port: upstream.port || 80,
path: '/',
json: true
}; };
var base_uri = 'http://' + opt.host + ':' + opt.port + opt.path; var connect = function(opt) {
var ev = new EventEmitter();
var internal; // local port
var upstream; var local_port = opt.port;
var prev_id;
(function connect_proxy() { var base_uri = opt.host + '/';
opt.uri = base_uri + ((prev_id) ? prev_id : '?new');
// optionally override the upstream server
var upstream = url.parse(opt.host);
// no subdomain at first, maybe use requested domain
var assigned_domain = opt.subdomain;
// connect to upstream given connection parameters
var tunnel = function (remote_host, remote_port, max_conn) {
var count = 0;
// open 5 connections to the localtunnel server
// allows for resources to be served faster
for (var count = 0 ; count < max_conn ; ++count) {
var upstream = duplex(remote_host, remote_port, 'localhost', local_port);
upstream.once('end', function() {
// all upstream connections have been closed
if (--count <= 0) {
tunnel(remote_host, remote_port, max_conn);
}
});
upstream.on('error', function(err) {
console.error(err);
});
}
};
var params = {
path: '/',
json: true
};
// where to quest
params.uri = base_uri + ((assigned_domain) ? assigned_domain : '?new');
request_url(params, function(err, body) {
request(opt, function(err, res, body) {
if (err) { if (err) {
console.error('upstream not available: %s', err.message); ev.emit('error', new Error('tunnel server not available: %s, retry 1s', err.message));
return process.exit(-1);
// retry interval for id request
return setTimeout(function() {
connect_proxy(opt);
}, 1000);
} }
// our assigned hostname and tcp port // our assigned hostname and tcp port
var port = body.port; var port = body.port;
var host = opt.host; var host = upstream.hostname;
// store the id so we can try to get the same one // store the id so we can try to get the same one
prev_id = body.id; assigned_domain = body.id;
console.log('your url is: %s', body.url); tunnel(host, port, body.max_conn_count || 1);
// connect to remote tcp server ev.emit('url', body.url);
upstream = net.createConnection(port, host); });
// reconnect internal return ev;
connect_internal(); };
upstream.on('end', function() { var duplex = function(remote_host, remote_port, local_host, local_port) {
console.log('> upstream connection terminated'); var ev = new EventEmitter();
// sever connection to internal server // connect to remote tcp server
// on reconnect we will re-establish var upstream = net.createConnection(remote_port, remote_host);
internal.end(); var internal;
// when upstream connection is closed, close other associated connections
upstream.once('end', function() {
ev.emit('error', new Error('upstream connection terminated'));
// sever connection to internal server
// on reconnect we will re-establish
internal.end();
ev.emit('end');
});
upstream.on('error', function(err) {
ev.emit('error', err);
});
(function connect_internal() {
internal = net.createConnection(local_port, local_host);
internal.on('error', function() {
ev.emit('error', new Error('error connecting to local server. retrying in 1s'));
setTimeout(function() { setTimeout(function() {
connect_proxy(); connect_internal();
}, 1000); }, 1000);
}); });
});
})();
function connect_internal() { internal.on('end', function() {
ev.emit('error', new Error('disconnected from local server. retrying in 1s'));
setTimeout(function() {
connect_internal();
}, 1000);
});
internal = net.createConnection(local_port); upstream.pipe(internal).pipe(upstream);
internal.on('error', function(err) { })();
console.log('error connecting to local server. retrying in 1s');
setTimeout(function() { return ev;
connect_internal();
}, 1000);
});
internal.on('end', function() {
console.log('disconnected from local server. retrying in 1s');
setTimeout(function() {
connect_internal();
}, 1000);
});
upstream.pipe(internal);
internal.pipe(upstream);
} }
module.exports.connect = connect;

12
lib/rand_id.js Normal file
View File

@@ -0,0 +1,12 @@
var chars = 'abcdefghiklmnopqrstuvwxyz';
module.exports = function rand_id() {
var randomstring = '';
for (var i=0; i<4; ++i) {
var rnum = Math.floor(Math.random() * chars.length);
randomstring += chars[rnum];
}
return randomstring;
}

View File

@@ -2,7 +2,7 @@
"author": "Roman Shtylman <shtylman@gmail.com>", "author": "Roman Shtylman <shtylman@gmail.com>",
"name": "localtunnel", "name": "localtunnel",
"description": "expose localhost to the world", "description": "expose localhost to the world",
"version": "0.0.1", "version": "0.0.2",
"repository": { "repository": {
"type": "git", "type": "git",
"url": "git://github.com/shtylman/localtunnel.git" "url": "git://github.com/shtylman/localtunnel.git"
@@ -12,12 +12,18 @@
"book": "1.2.0", "book": "1.2.0",
"optimist": "0.3.4" "optimist": "0.3.4"
}, },
"devDependencies": {}, "devDependencies": {
"mocha": "1.6.0"
},
"optionalDependencies": {}, "optionalDependencies": {},
"engines": { "engines": {
"node": "*" "node": "*"
}, },
"scripts": {
"test": "mocha --ui qunit -- test",
"start": "./bin/server"
},
"bin": { "bin": {
"lt": "./bin/lt" "lt": "./bin/client"
} }
} }

315
server.js
View File

@@ -3,47 +3,24 @@
var http = require('http'); var http = require('http');
var net = require('net'); var net = require('net');
var url = require('url'); var url = require('url');
var FreeList = require('freelist').FreeList;
var argv = require('optimist')
.usage('Usage: $0 --port [num]')
.options('port', {
default: '80',
describe: 'listen on this port for outside requests'
})
.argv;
if (argv.help) {
require('optimist').showHelp();
process.exit();
}
// here be dragons // here be dragons
var HTTPParser = process.binding('http_parser').HTTPParser; var HTTPParser = process.binding('http_parser').HTTPParser;
var ServerResponse = http.ServerResponse; var ServerResponse = http.ServerResponse;
var IncomingMessage = http.IncomingMessage; var IncomingMessage = http.IncomingMessage;
// vendor
var log = require('book'); var log = require('book');
var chars = 'abcdefghiklmnopqrstuvwxyz'; // local
function rand_id() { var rand_id = require('./lib/rand_id');
var randomstring = '';
for (var i=0; i<4; ++i) {
var rnum = Math.floor(Math.random() * chars.length);
randomstring += chars[rnum];
}
return randomstring;
}
var server = http.createServer(); var server = http.createServer();
// id -> client http server // id -> client http server
var clients = {}; var clients = {};
// id -> list of sockets waiting for a valid response // available parsers
var wait_list = {};
var parsers = http.parsers; var parsers = http.parsers;
// data going back to a client (the last client that made a request) // data going back to a client (the last client that made a request)
@@ -52,23 +29,26 @@ function socketOnData(d, start, end) {
var socket = this; var socket = this;
var req = this._httpMessage; var req = this._httpMessage;
var current = clients[socket.subdomain].current; var response_socket = socket.respond_socket;
if (!response_socket) {
if (!current) { log.error('no response socket assigned for http response from backend');
log.error('no current for http response from backend');
return; return;
} }
// send the goodies // pass the response from our client back to the requesting socket
current.write(d.slice(start, end)); response_socket.write(d.slice(start, end));
// invoke parsing so we know when all the goodies have been sent if (socket.for_websocket) {
var parser = current.out_parser; return;
}
// invoke parsing so we know when the response is complete
var parser = response_socket.out_parser;
parser.socket = socket; parser.socket = socket;
var ret = parser.execute(d, start, end - start); var ret = parser.execute(d, start, end - start);
if (ret instanceof Error) { if (ret instanceof Error) {
debug('parse error'); log.error(ret);
freeParser(parser, req); freeParser(parser, req);
socket.destroy(ret); socket.destroy(ret);
} }
@@ -99,83 +79,103 @@ server.on('connection', function(socket) {
var self = this; var self = this;
var for_client = false; // parser handles incoming requests for the socket
var client_id; // the request is what lets us know if we proxy or not
var request;
var parser = parsers.alloc(); var parser = parsers.alloc();
parser.socket = socket; parser.socket = socket;
parser.reinitialize(HTTPParser.REQUEST); parser.reinitialize(HTTPParser.REQUEST);
function our_request(req) {
var res = new ServerResponse(req);
res.assignSocket(socket);
self.emit('request', req, res);
return;
}
// a full request is complete // a full request is complete
// we wait for the response from the server // we wait for the response from the server
parser.onIncoming = function(req, shouldKeepAlive) { parser.onIncoming = function(req, shouldKeepAlive) {
log.trace('request', req.url); log.trace('request', req.url);
request = req;
for_client = false; // default is that the data is not for the client
delete parser.sock;
delete parser.buffer;
delete parser.client;
var hostname = req.headers.host; var hostname = req.headers.host;
if (!hostname) { if (!hostname) {
log.trace('no hostname: %j', req.headers); log.trace('no hostname: %j', req.headers);
// normal processing if not proxy return our_request(req);
var res = new ServerResponse(req);
// TODO(shtylman) skip favicon for now, it caused problems
if (req.url === '/favicon.ico') {
return;
}
res.assignSocket(parser.socket);
self.emit('request', req, res);
return;
} }
var match = hostname.match(/^([a-z]{4})[.].*/); var match = hostname.match(/^([a-z]{4})[.].*/);
if (!match) { if (!match) {
// normal processing if not proxy return our_request(req);
var res = new ServerResponse(req); }
// TODO(shtylman) skip favicon for now, it caused problems var client_id = match[1];
if (req.url === '/favicon.ico') { var client = clients[client_id];
return;
}
res.assignSocket(parser.socket); // requesting a subdomain that doesn't exist
self.emit('request', req, res); if (!client) {
return socket.end();
}
parser.client = client;
// assigned socket for the client
var sock = client.sockets.shift();
// no free sockets, queue
if (!sock) {
parser.buffer = true;
return; return;
} }
client_id = match[1]; // for tcp proxying
for_client = true; parser.sock = sock;
// set who we will respond back to
sock.respond_socket = socket;
var out_parser = parsers.alloc(); var out_parser = parsers.alloc();
out_parser.reinitialize(HTTPParser.RESPONSE); out_parser.reinitialize(HTTPParser.RESPONSE);
socket.out_parser = out_parser; socket.out_parser = out_parser;
// we have a response // we have completed a response
out_parser.onIncoming = function(res) { // the tcp socket is free again
out_parser.onIncoming = function (res) {
res.on('end', function() { res.on('end', function() {
log.trace('done with response for: %s', req.url); log.trace('done with response for: %s', req.url);
// done with the parser // done with the parser
parsers.free(out_parser); parsers.free(out_parser);
var next = wait_list[client_id].shift(); // unset the response
delete sock.respond_socket;
clients[client_id].current = next;
var next = client.waiting.shift();
if (!next) { if (!next) {
// return socket to available
client.sockets.push(sock);
return; return;
} }
// write original bytes that we held cause client was busy // reuse avail socket for next connection
clients[client_id].write(next.queue); sock.respond_socket = next;
// needed to know when this response will be done
out_parser.reinitialize(HTTPParser.RESPONSE);
next.out_parser = out_parser;
// write original bytes we held cause we were busy
sock.write(next.queue);
// continue with other bytes
next.resume(); next.resume();
return;
}); });
}; };
}; };
@@ -183,68 +183,87 @@ server.on('connection', function(socket) {
// process new data on the client socket // process new data on the client socket
// we may need to forward this it the backend // we may need to forward this it the backend
socket.ondata = function(d, start, end) { socket.ondata = function(d, start, end) {
// run through request parser to determine if we should pass to tcp
// onIncoming will be run before this returns
var ret = parser.execute(d, start, end - start); var ret = parser.execute(d, start, end - start);
// invalid request from the user // invalid request from the user
if (ret instanceof Error) { if (ret instanceof Error) {
debug('parse error'); log.error(ret);
socket.destroy(ret); socket.destroy(ret);
return; return;
} }
// only write data if previous request to this client is done? // websocket stuff
log.trace('%s %s', parser.incoming && parser.incoming.upgrade, for_client);
// what if the subdomains are treated differently
// as individual channels to the backend if available?
// how can I do that?
if (parser.incoming && parser.incoming.upgrade) { if (parser.incoming && parser.incoming.upgrade) {
// websocket shit log.trace('upgrade request');
}
// wtf do you do with upgraded connections? parser.finish();
// forward the data to the backend var hostname = parser.incoming.headers.host;
if (for_client) {
var match = hostname.match(/^([a-z]{4})[.].*/);
if (!match) {
return our_request(req);
}
var client_id = match[1];
var client = clients[client_id]; var client = clients[client_id];
// requesting a subdomain that doesn't exist var sock = client.sockets.shift();
if (!client) { sock.respond_socket = socket;
return; sock.for_websocket = true;
socket.ondata = function(d, start, end) {
sock.write(d.slice(start, end));
};
socket.end = function() {
log.trace('websocket end');
delete sock.respond_socket;
client.sockets.push(sock);
} }
// if the client is already processing something sock.write(d.slice(start, end));
// then new connections need to go into pause mode
// and when they are revived, then they can send data along
if (client.current && client.current !== socket) {
log.trace('pausing', request.url);
// prevent new data from gathering for this connection
// we are waiting for a response to a previous request
socket.pause();
var copy = Buffer(end - start); return;
d.copy(copy, 0, start, end);
socket.queue = copy;
wait_list[client_id].push(socket);
return;
}
// this socket needs to receive responses
client.current = socket;
// send through tcp tunnel
client.write(d.slice(start, end));
} }
// if no available socket, buffer the request for later
if (parser.buffer) {
// pause any further data on this socket
socket.pause();
// copy the current data since we have already received it
var copy = Buffer(end - start);
d.copy(copy, 0, start, end);
socket.queue = copy;
// add socket to queue
parser.client.waiting.push(socket);
return;
}
if (!parser.sock) {
return;
}
// assert, respond socket should be set
// send through tcp tunnel
// responses will go back to the respond_socket
parser.sock.write(d.slice(start, end));
}; };
socket.onend = function() { socket.onend = function() {
var ret = parser.finish(); var ret = parser.finish();
if (ret instanceof Error) { if (ret instanceof Error) {
log.trace('parse error'); log.error(ret);
socket.destroy(ret); socket.destroy(ret);
return; return;
} }
@@ -271,8 +290,12 @@ server.on('request', function(req, res) {
if (req.url === '/' && !parsed.query.new) { if (req.url === '/' && !parsed.query.new) {
res.writeHead(301, { Location: 'http://shtylman.github.com/localtunnel/' }); res.writeHead(301, { Location: 'http://shtylman.github.com/localtunnel/' });
res.end(); 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})?/); var match = req.url.match(/\/([a-z]{4})?/);
// user can request a particular set of characters // user can request a particular set of characters
@@ -285,17 +308,17 @@ server.on('request', function(req, res) {
} }
var id = requested_id || rand_id(); var id = requested_id || rand_id();
if (wait_list[id]) {
// new id
id = rand_id();
}
// generate new shit for client // maximum number of tcp connections the client can setup
if (wait_list[id]) { // each tcp channel allows for more parallel requests
wait_list[id].forEach(function(waiting) { var max_tcp_sockets = 4;
waiting.end();
}); // sockets is a list of available sockets for the connection
} // waiting is?
var client = clients[id] = {
sockets: [],
waiting: []
};
var client_server = net.createServer(); var client_server = net.createServer();
client_server.listen(function() { client_server.listen(function() {
@@ -305,7 +328,12 @@ server.on('request', function(req, res) {
var url = 'http://' + id + '.' + req.headers.host; var url = 'http://' + id + '.' + req.headers.host;
res.writeHead(200, { 'Content-Type': 'application/json' }); res.writeHead(200, { 'Content-Type': 'application/json' });
res.end(JSON.stringify({ url: url, id: id, port: port })); res.end(JSON.stringify({
url: url,
id: id,
port: port,
max_conn_count: max_tcp_sockets
}));
}); });
// user has 5 seconds to connect before their slot is given up // user has 5 seconds to connect before their slot is given up
@@ -313,31 +341,50 @@ server.on('request', function(req, res) {
client_server.close(); client_server.close();
}, 5000); }, 5000);
// no longer accepting connections for this id
client_server.on('close', function() {
delete clients[id];
});
var count = 0;
client_server.on('connection', function(socket) { client_server.on('connection', function(socket) {
// who the info should route back to // no more socket connections allowed
socket.subdomain = id; if (count++ >= max_tcp_sockets) {
return socket.end();
}
log.trace('new connection for id: %s', id);
// multiplexes socket data out to clients // multiplexes socket data out to clients
socket.ondata = socketOnData; socket.ondata = socketOnData;
// no need to close the client server
clearTimeout(conn_timeout); clearTimeout(conn_timeout);
log.trace('new connection for id: %s', id); // add socket to pool for this id
clients[id] = socket; var idx = client.sockets.push(socket) - 1;
wait_list[id] = [];
socket.on('end', function() { socket.on('close', function(had_error) {
delete clients[id]; count--;
client.sockets.splice(idx, 1);
// no more sockets for this ident
if (client.sockets.length === 0) {
delete clients[id];
}
});
// close will be emitted after this
socket.on('error', function(err) {
log.error(err);
}); });
}); });
client_server.on('err', function(err) { client_server.on('error', function(err) {
log.error(err); log.error(err);
}); });
}); });
server.listen(argv.port, function() { module.exports = server;
log.info('server listening on port: %d', server.address().port);
});

95
test.js Normal file
View File

@@ -0,0 +1,95 @@
var http = require('http');
var url = require('url');
var assert = require('assert');
var localtunnel_server = require('./server');
var localtunnel_client = require('./client');
test('setup localtunnel server', function(done) {
localtunnel_server.listen(3000, function() {
console.log('lt server on:', 3000);
done();
});
});
test('setup local http server', function(done) {
var server = http.createServer();
server.on('request', function(req, res) {
res.write('foo');
res.end();
});
server.listen(function() {
var port = server.address().port;
test._fake_port = port;
console.log('local http on:', port);
done();
});
});
test('setup localtunnel client', function(done) {
var client = localtunnel_client.connect({
host: 'http://localhost:' + 3000,
port: test._fake_port
});
client.on('url', function(url) {
assert.ok(/^http:\/\/.*localhost:3000$/.test(url));
test._fake_url = url;
done();
});
client.on('error', function(err) {
console.error(err);
});
});
test('query localtunnel server w/ ident', function(done) {
var uri = test._fake_url;
var hostname = url.parse(uri).hostname;
var opt = {
host: 'localhost',
port: 3000,
headers: {
host: hostname
},
path: '/'
}
var req = http.request(opt, function(res) {
res.setEncoding('utf8');
var body = '';
res.on('data', function(chunk) {
body += chunk;
});
res.on('end', function() {
assert.equal('foo', body);
// TODO(shtylman) shutdown client
done();
});
});
req.end();
});
test('request specific domain', function(done) {
var client = localtunnel_client.connect({
host: 'http://localhost:' + 3000,
port: test._fake_port,
subdomain: 'abcd'
});
client.on('url', function(url) {
assert.ok(/^http:\/\/abcd.localhost:3000$/.test(url));
done();
});
client.on('error', function(err) {
console.error(err);
});
});