123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407 |
- /**
- * 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 ? Buffer.concat([]) : '';
- var self = this;
- function cleanup () {
- req.removeListener('data', onData);
- req.removeListener('end', onEnd);
- req.removeListener('close', onClose);
- self.dataReq = self.dataRes = chunks = null;
- }
- function onClose () {
- cleanup();
- self.onError('data request connection closed prematurely');
- }
- function onData (data) {
- var contentLength;
- if (isBinary) {
- chunks = Buffer.concat([chunks, data]);
- contentLength = chunks.length;
- } else {
- chunks += data;
- contentLength = Buffer.byteLength(chunks);
- }
- if (contentLength > self.maxHttpBufferSize) {
- chunks = isBinary ? Buffer.concat([]) : '';
- 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();
- }
- };
- /**
- * Compresses 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;
- };
|