123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267 |
- var util = require('./util');
- var express = require('express');
- var http = require('http');
- var EventEmitter = require('events').EventEmitter;
- var WebSocketServer = require('ws').Server;
- var url = require('url');
- function PeerServer(options) {
- if (!(this instanceof PeerServer)) return new PeerServer(options);
- EventEmitter.call(this);
- this._app = express();
- this._httpServer = http.createServer(this._app);
- this._app.use(express.bodyParser());
- this._app.use(this._allowCrossDomain);
- options = util.extend({
- port: 80
- }, options);
- util.debug = options.debug;
- // Listen on user-specified port and create WebSocket server as well.
- this._httpServer.listen(options.port);
- this._wss = new WebSocketServer({ path: '/ws', server: this._httpServer });
- // WebSockets that are opened or HTTP responses (which are paired with
- // something in timeouts.
- this._clients = {};
- // Timeouts for HTTP responses.
- this._timeouts = {};
- // Connections waiting for another peer.
- this._outstandingOffers = {};
- // Initailize WebSocket server handlers.
- this._initializeWSS();
- // Initialize HTTP routes. This is only used for the first few milliseconds
- // before a socket is opened for a Peer.
- this._initializeHTTP();
- };
- util.inherits(PeerServer, EventEmitter);
- /** Handle CORS */
- PeerServer.prototype._allowCrossDomain = function(req, res, next) {
- res.setHeader('Access-Control-Allow-Origin', '*');
- res.setHeader('Access-Control-Allow-Methods', 'GET,PUT,POST,DELETE');
- res.setHeader('Access-Control-Allow-Headers', 'Content-Type');
- next();
- }
- /** Initialize WebSocket server. */
- PeerServer.prototype._initializeWSS = function() {
- var self = this;
- this._wss.on('connection', function(socket) {
- var id = url.parse(socket.upgradeReq.url, true).query.id;
- if (!!id && !!self._clients[id]) {
- // If response client and timeout exist, overwrite and clear.
- if (!!self._timeouts[id]) {
- clearTimeout(self._timeouts[id]);
- delete self._timeouts[id];
- self._clients[id].end('socket');
- } else {
- socket.send(JSON.stringify({ type: 'ERROR', msg: 'ID is taken' }));
- return;
- }
- } else if (!id) {
- id = self._generateClientId();
- socket.send(JSON.stringify({ type: 'ID', id: id }));
- }
- // Save the socket for this id.
- self._clients[id] = socket;
- self._processOutstandingOffers(id);
- socket.on('message', function(data) {
- try {
- var message = JSON.parse(data);
- util.log(message);
- switch (message.type) {
- case 'LEAVE':
- // Clean up if a Peer sends a LEAVE.
- if (!message.dst) {
- delete self._clients[message.src];
- delete self._timeouts[message.src];
- break;
- }
- // ICE candidates
- case 'CANDIDATE':
- // Offer or answer between peers.
- case 'OFFER':
- case 'ANSWER':
- // Firefoxism (connectDataConnection ports)
- case 'PORT':
- self._handleTransmission(message.type, message.src, message.dst, data);
- break;
- default:
- util.prettyError('message unrecognized');
- }
- } catch(e) {
- util.log('invalid message');
- }
- });
- });
- };
- /** Process outstanding peer offers. */
- PeerServer.prototype._processOutstandingOffers = function(id) {
- var offers = this._outstandingOffers[id];
- for (var source in offers) {
- if (offers.hasOwnProperty(source)) {
- var messages = offers[source]
- for (var i = 0; i < messages.length; i += 1)
- this._handleTransmission.apply(this, messages[i]);
- delete this._outstandingOffers[id][source];
- }
- }
- };
- /** Initialize HTTP server routes. */
- PeerServer.prototype._initializeHTTP = function() {
- var self = this;
- this._app.options('/*', function(req, res, next) {
- res.send(200);
- });
- // Server sets up HTTP streaming whether you get or post an ID.
- // Retrieve guaranteed random ID.
- this._app.get('/id', function(req, res) {
- var clientId = util.randomId();
- while (!!self._clients[clientId]) {
- clientId = util.randomId();
- }
- self._startStreaming(res, clientId, function() {
- // Chrome hacks.
- res.write('{"id":"' + clientId + '"}\n');
- });
- });
- this._app.post('/id', function(req, res) {
- var id = req.body.id;
- self._startStreaming(res, id);
- });
- this._app.post('/offer', function(req, res) {
- var src = req.body.src;
- var dst = req.body.dst;
- self._handleTransmission('OFFER', src, dst, JSON.stringify(req.body), res);
- });
- this._app.post('/ice', function(req, res) {
- var src = req.body.src;
- var dst = req.body.dst;
- self._handleTransmission('ICE', src, dst, JSON.stringify(req.body), res);
- });
- this._app.post('/answer', function(req, res) {
- var src = req.body.src;
- var dst = req.body.dst;
- self._handleTransmission('ANSWER', src, dst, JSON.stringify(req.body), res);
- });
- this._app.post('/leave', function(req, res) {
- var src = req.body.src;
- var dst = req.body.dst;
- self._handleTransmission('LEAVE', src, dst, JSON.stringify(req.body), res);
- });
- this._app.post('/port', function(req, res) {
- var src = req.body.src;
- var dst = req.body.dst;
- self._handleTransmission('PORT', src, dst, JSON.stringify(req.body), res);
- });
- };
- /** Saves a streaming response and takes care of timeouts and headers. */
- PeerServer.prototype._startStreaming = function(res, id, write) {
- res.writeHead(200, {'Content-Type': 'application/octet-stream'});
- if (!!write) {
- write();
- }
- var pad = '00';
- var iterations = 10;
- for (var i = 0; i < iterations; i++) {
- pad += pad;
- }
- res.write(pad + '\n');
- // Save res so we can write to it.
- if (!this._clients[id]) {
- this._clients[id] = res;
- // Set timeout to expire.
- this._timeouts[id] = setTimeout(function() { res.end('end') }, 10000);
- } else {
- res.write(JSON.stringify({ type: 'ERROR', msg: 'ID is taken' }) + '\n');
- res.end('error');
- }
- };
- /** Handles passing on a message. */
- PeerServer.prototype._handleTransmission = function(type, src, dst, data, res) {
- var destination = this._clients[dst];
- if (!!destination) {
- try {
- if (this._timeouts[dst]) {
- data += '\n';
- }
- // We have to let the source peer know that the offer was sent
- // successfully so that ice can start being processed.
- if (type === 'OFFER') {
- if (!!res) {
- res.send(200);
- } else if (!this._timeouts[src] && !!this._clients[src]) {
- this._clients[src].send(JSON.stringify({ type: 'PEER_READY', src: dst, dst: src }));
- }
- }
- destination.send(data);
- } catch (e) {
- util.prettyError(e);
- // This really shouldn't happen given correct client browsers.
- // 501: Server does not support this functionality.
- if (!!res) res.send(501);
- }
- } else {
- if (type === 'OFFER' && (!this._outstandingOffers[dst] || !this._outstandingOffers[dst][src])) {
- // Wait 5 seconds for this client to connect.
- var self = this;
- if (!this._outstandingOffers[dst])
- this._outstandingOffers[dst] = {};
- this._outstandingOffers[dst][src] = [];
- this._outstandingOffers[dst][src].push(Array.prototype.slice.apply(arguments));
- setTimeout(function() {
- delete self._outstandingOffers[dst][src]
- }, 5000);
- } else if (type === 'ICE' && !!this._outstandingOffers[dst][src]) {
- this._outstandingOffers[dst][src].push(Array.prototype.slice.apply(arguments));
- } else {
- // Assume a disconnect if the client no longer exists.
- util.log('destination does not exist');
- this._handleTransmission('LEAVE', dst, src, JSON.stringify({ type: 'LEAVE', dst: src, src: dst }));
- // 410: Resource not available.
- if (!!res) res.send(410);
- }
- }
- };
- PeerServer.prototype._generateClientId = function() {
- var clientId = util.randomId();
- while (!!self._clients[clientId]) {
- clientId = util.randomId();
- }
- };
- exports.PeerServer = PeerServer;
|