@maximegris/node-websockify
Version:
node-websockify is a WebSocket-to-TCP proxy/bridge you can use in a NodeJS program
205 lines (177 loc) • 5.34 kB
JavaScript
var net = require('net'),
http = require('http'),
https = require('https'),
url = require('url'),
path = require('path'),
fs = require('fs'),
mime = require('mime'),
Buffer = require('buffer').Buffer,
WebSocketServer = require('ws').Server,
webServer, wsServer,
source_host, source_port, target_host, target_port,
argv = null,
onConnectedCallback = null,
onDisconnectedCallback = null;
// Handle new WebSocket client
var new_client = function(client, req) {
var clientAddr = client._socket.remoteAddress, log;
console.log(req ? req.url : client.upgradeReq.url);
log = function (msg) {
console.log(' ' + clientAddr + ': '+ msg);
};
log('WebSocket connection from : ' + clientAddr);
log('Version ' + client.protocolVersion + ', subprotocol: ' + client.protocol);
var target = net.createConnection(target_port,target_host, function() {
log('connected to target');
if (onConnectedCallback)
{
try {
onConnectedCallback(client, target);
} catch(e) {
log("onConnectedCallback failed, cleaning up target");
target.end();
}
}
});
target.on('data', function(data) {
//log("sending message: " + data);
try {
client.send(data);
} catch(e) {
log("Client closed, cleaning up target");
target.end();
}
});
target.on('end', function() {
log('target disconnected');
client.close();
});
target.on('error', function() {
log('target connection error');
target.end();
client.close();
});
client.on('message', function(msg) {
//log('got message: ' + msg);
target.write(msg);
});
client.on('close', function(code, reason) {
if (onDisconnectedCallback)
{
try {
onDisconnectedCallback(client, code, reason);
} catch(e) {
log("onDisconnectedCallback failed");
}
}
log('WebSocket client disconnected: ' + code + ' [' + reason + ']');
target.end();
});
client.on('error', function(a) {
log('WebSocket client error: ' + a);
target.end();
});
};
// Send an HTTP error response
var http_error = function (response, code, msg) {
response.writeHead(code, {"Content-Type": "text/plain"});
response.write(msg + "\n");
response.end();
return;
}
// Process an HTTP static file request
var http_request = function (request, response) {
// console.log("pathname: " + url.parse(req.url).pathname);
// res.writeHead(200, {'Content-Type': 'text/plain'});
// res.end('okay');
if (!argv.web) {
return http_error(response, 403, "403 Permission Denied");
}
var uri = url.parse(request.url).pathname
, filename = path.join(argv.web, uri);
fs.exists(filename, function(exists) {
if(!exists) {
return http_error(response, 404, "404 Not Found");
}
if (fs.statSync(filename).isDirectory()) {
filename += '/index.html';
}
fs.readFile(filename, "binary", function(err, file) {
if(err) {
return http_error(response, 500, err);
}
response.setHeader('Content-type', mime.getType(path.parse(uri).ext));
response.writeHead(200);
response.write(file, "binary");
response.end();
});
});
};
function initWsServer(_argv, callbacks = undefined) {
argv = _argv;
var source_arg = argv.source;
var target_arg = argv.target;
if (!callbacks)
{
console.log("no callbacks");
}
else
{
if (callbacks.onConnected)
{
onConnectedCallback = callbacks.onConnected;
console.log("onConnectedCallback registered");
}
if (callbacks.onDisconnected)
{
onDisconnectedCallback = callbacks.onDisconnected;
console.log("onDisconnectedCallback registered");
}
}
// parse source and target arguments into parts
try {
var idx;
idx = source_arg.indexOf(":");
if (idx >= 0) {
source_host = source_arg.slice(0, idx);
source_port = parseInt(source_arg.slice(idx+1), 10);
} else {
source_host = "";
source_port = parseInt(source_arg, 10);
}
idx = target_arg.indexOf(":");
if (idx < 0) {
throw("target must be host:port");
}
target_host = target_arg.slice(0, idx);
target_port = parseInt(target_arg.slice(idx+1), 10);
if (isNaN(source_port) || isNaN(target_port)) {
throw("illegal port");
}
} catch(e) {
console.error("websockify.js [--web web_dir] [--cert cert.pem [--key key.pem]] [source_addr:]source_port target_addr:target_port");
process.exit(2);
}
console.log("WebSocket settings: ");
console.log(" - proxying from " + source_host + ":" + source_port +
" to " + target_host + ":" + target_port);
if (argv.web) {
console.log(" - Web server active. Serving: " + argv.web);
}
if (argv.cert) {
argv.key = argv.key || argv.cert;
var cert = fs.readFileSync(argv.cert),
key = fs.readFileSync(argv.key);
console.log(" - Running in encrypted HTTPS (wss://) mode using: " + argv.cert + ", " + argv.key);
webServer = https.createServer({cert: cert, key: key}, http_request);
} else {
console.log(" - Running in unencrypted HTTP (ws://) mode");
webServer = http.createServer(http_request);
}
webServer.listen(source_port, function() {
wsServer = new WebSocketServer({server: webServer});
wsServer.on('connection', new_client);
});
}
module.exports = initWsServer;