|
|
/** * Module requirements. */
var Transport = require('../transport'); var parser = require('engine.io-parser'); var zlib = require('zlib'); var accepts = require('accepts'); var util = require('util'); var debug = require('debug')('engine:polling');
var compressionMethods = { gzip: zlib.createGzip, deflate: zlib.createDeflate };
/** * Exports the constructor. */
module.exports = Polling;
/** * HTTP polling constructor. * * @api public. */
function Polling (req) { Transport.call(this, req);
this.closeTimeout = 30 * 1000; this.maxHttpBufferSize = null; this.httpCompression = null; }
/** * Inherits from Transport. * * @api public. */
util.inherits(Polling, Transport);
/** * Transport name * * @api public */
Polling.prototype.name = 'polling';
/** * Overrides onRequest. * * @param {http.IncomingMessage} * @api private */
Polling.prototype.onRequest = function (req) { var res = req.res;
if ('GET' === req.method) { this.onPollRequest(req, res); } else if ('POST' === req.method) { this.onDataRequest(req, res); } else { res.writeHead(500); res.end(); } };
/** * The client sends a request awaiting for us to send data. * * @api private */
Polling.prototype.onPollRequest = function (req, res) { if (this.req) { debug('request overlap'); // assert: this.res, '.req and .res should be (un)set together'
this.onError('overlap from client'); res.writeHead(500); res.end(); return; }
debug('setting request');
this.req = req; this.res = res;
var self = this;
function onClose () { self.onError('poll connection closed prematurely'); }
function cleanup () { req.removeListener('close', onClose); self.req = self.res = null; }
req.cleanup = cleanup; req.on('close', onClose);
this.writable = true; this.emit('drain');
// if we're still writable but had a pending close, trigger an empty send
if (this.writable && this.shouldClose) { debug('triggering empty send to append close packet'); this.send([{ type: 'noop' }]); } };
/** * The client sends a request with data. * * @api private */
Polling.prototype.onDataRequest = function (req, res) { if (this.dataReq) { // assert: this.dataRes, '.dataReq and .dataRes should be (un)set together'
this.onError('data request overlap from client'); res.writeHead(500); res.end(); return; }
var isBinary = 'application/octet-stream' === req.headers['content-type'];
this.dataReq = req; this.dataRes = res;
var chunks = isBinary ? new Buffer(0) : ''; var self = this;
function cleanup () { chunks = isBinary ? new Buffer(0) : ''; req.removeListener('data', onData); req.removeListener('end', onEnd); req.removeListener('close', onClose); self.dataReq = self.dataRes = null; }
function onClose () { cleanup(); self.onError('data request connection closed prematurely'); }
function onData (data) { var contentLength; if (typeof data === 'string') { chunks += data; contentLength = Buffer.byteLength(chunks); } else { chunks = Buffer.concat([chunks, data]); contentLength = chunks.length; }
if (contentLength > self.maxHttpBufferSize) { chunks = ''; req.connection.destroy(); } }
function onEnd () { self.onData(chunks);
var headers = { // text/html is required instead of text/plain to avoid an
// unwanted download dialog on certain user-agents (GH-43)
'Content-Type': 'text/html', 'Content-Length': 2 };
res.writeHead(200, self.headers(req, headers)); res.end('ok'); cleanup(); }
req.on('close', onClose); if (!isBinary) req.setEncoding('utf8'); req.on('data', onData); req.on('end', onEnd); };
/** * Processes the incoming data payload. * * @param {String} encoded payload * @api private */
Polling.prototype.onData = function (data) { debug('received "%s"', data); var self = this; var callback = function (packet) { if ('close' === packet.type) { debug('got xhr close packet'); self.onClose(); return false; }
self.onPacket(packet); };
parser.decodePayload(data, callback); };
/** * Overrides onClose. * * @api private */
Polling.prototype.onClose = function () { if (this.writable) { // close pending poll request
this.send([{ type: 'noop' }]); } Transport.prototype.onClose.call(this); };
/** * Writes a packet payload. * * @param {Object} packet * @api private */
Polling.prototype.send = function (packets) { this.writable = false;
if (this.shouldClose) { debug('appending close packet to payload'); packets.push({ type: 'close' }); this.shouldClose(); this.shouldClose = null; }
var self = this; parser.encodePayload(packets, this.supportsBinary, function (data) { var compress = packets.some(function (packet) { return packet.options && packet.options.compress; }); self.write(data, { compress: compress }); }); };
/** * Writes data as response to poll request. * * @param {String} data * @param {Object} options * @api private */
Polling.prototype.write = function (data, options) { debug('writing "%s"', data); var self = this; this.doWrite(data, options, function () { self.req.cleanup(); }); };
/** * Performs the write. * * @api private */
Polling.prototype.doWrite = function (data, options, callback) { var self = this;
// explicit UTF-8 is required for pages not served under utf
var isString = typeof data === 'string'; var contentType = isString ? 'text/plain; charset=UTF-8' : 'application/octet-stream';
var headers = { 'Content-Type': contentType };
if (!this.httpCompression || !options.compress) { respond(data); return; }
var len = isString ? Buffer.byteLength(data) : data.length; if (len < this.httpCompression.threshold) { respond(data); return; }
var encoding = accepts(this.req).encodings(['gzip', 'deflate']); if (!encoding) { respond(data); return; }
this.compress(data, encoding, function (err, data) { if (err) { self.res.writeHead(500); self.res.end(); callback(err); return; }
headers['Content-Encoding'] = encoding; respond(data); });
function respond (data) { headers['Content-Length'] = 'string' === typeof data ? Buffer.byteLength(data) : data.length; self.res.writeHead(200, self.headers(self.req, headers)); self.res.end(data); callback(); } };
/** * Comparesses data. * * @api private */
Polling.prototype.compress = function (data, encoding, callback) { debug('compressing');
var buffers = []; var nread = 0;
compressionMethods[encoding](this.httpCompression) .on('error', callback) .on('data', function (chunk) { buffers.push(chunk); nread += chunk.length; }) .on('end', function () { callback(null, Buffer.concat(buffers, nread)); }) .end(data); };
/** * Closes the transport. * * @api private */
Polling.prototype.doClose = function (fn) { debug('closing');
var self = this; var closeTimeoutTimer;
if (this.dataReq) { debug('aborting ongoing data request'); this.dataReq.destroy(); }
if (this.writable) { debug('transport writable - closing right away'); this.send([{ type: 'close' }]); onClose(); } else if (this.discarded) { debug('transport discarded - closing right away'); onClose(); } else { debug('transport not writable - buffering orderly close'); this.shouldClose = onClose; closeTimeoutTimer = setTimeout(onClose, this.closeTimeout); }
function onClose () { clearTimeout(closeTimeoutTimer); fn(); self.onClose(); } };
/** * Returns headers for a response. * * @param {http.IncomingMessage} request * @param {Object} extra headers * @api private */
Polling.prototype.headers = function (req, headers) { headers = headers || {};
// prevent XSS warnings on IE
// https://github.com/LearnBoost/socket.io/pull/1333
var ua = req.headers['user-agent']; if (ua && (~ua.indexOf(';MSIE') || ~ua.indexOf('Trident/'))) { headers['X-XSS-Protection'] = '0'; }
this.emit('headers', headers); return headers; };
|