mirror of
https://github.com/socketio/socket.io.git
synced 2026-04-30 03:00:39 -04:00
There were two issues here. 1. When Socket.send called with or without callback alternately, the trigger order is incorrect. 2. The 'drain' event from transport is one per packet for transports supporting framing like websocket and is all in one for those without framing like polling.
361 lines
8.1 KiB
JavaScript
361 lines
8.1 KiB
JavaScript
/**
|
|
* Module dependencies.
|
|
*/
|
|
|
|
var EventEmitter = require('events').EventEmitter
|
|
, debug = require('debug')('engine:socket')
|
|
|
|
/**
|
|
* Module exports.
|
|
*/
|
|
|
|
module.exports = Socket;
|
|
|
|
/**
|
|
* Client class (abstract).
|
|
*
|
|
* @api private
|
|
*/
|
|
|
|
function Socket (id, server, transport) {
|
|
this.id = id;
|
|
this.server = server;
|
|
this.upgraded = false;
|
|
this.readyState = 'opening';
|
|
this.writeBuffer = [];
|
|
this.packetsFn = [];
|
|
|
|
this.setTransport(transport);
|
|
this.onOpen();
|
|
}
|
|
|
|
/**
|
|
* Inherits from EventEmitter.
|
|
*/
|
|
|
|
Socket.prototype.__proto__ = EventEmitter.prototype;
|
|
|
|
/**
|
|
* Accessor for request that originated socket.
|
|
*
|
|
* @api public
|
|
*/
|
|
|
|
Socket.prototype.__defineGetter__('request', function () {
|
|
return this.transport.request;
|
|
});
|
|
|
|
/**
|
|
* Called upon transport considered open.
|
|
*
|
|
* @api private
|
|
*/
|
|
|
|
Socket.prototype.onOpen = function () {
|
|
this.readyState = 'open';
|
|
|
|
// sends an `open` packet
|
|
this.transport.sid = this.id;
|
|
this.sendPacket('open', JSON.stringify({
|
|
sid: this.id
|
|
, upgrades: this.getAvailableUpgrades()
|
|
, pingInterval: this.server.pingInterval
|
|
, pingTimeout: this.server.pingTimeout
|
|
}));
|
|
|
|
this.emit('open');
|
|
this.setPingTimeout();
|
|
};
|
|
|
|
/**
|
|
* Called upon transport packet.
|
|
*
|
|
* @param {Object} packet
|
|
* @api private
|
|
*/
|
|
|
|
Socket.prototype.onPacket = function (packet) {
|
|
if ('open' == this.readyState) {
|
|
// export packet event
|
|
debug('packet');
|
|
this.emit('packet', packet);
|
|
|
|
// Reset ping timeout on any packet, incoming data is a good sign of
|
|
// other side's liveness
|
|
this.setPingTimeout();
|
|
|
|
switch (packet.type) {
|
|
|
|
case 'ping':
|
|
debug('got ping');
|
|
this.sendPacket('pong');
|
|
this.emit('heartbeat');
|
|
break;
|
|
|
|
case 'error':
|
|
this.onClose('parse error');
|
|
break;
|
|
|
|
case 'message':
|
|
this.emit('data', packet.data);
|
|
this.emit('message', packet.data);
|
|
break;
|
|
}
|
|
} else {
|
|
debug('packet received with closed socket');
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Called upon transport error.
|
|
*
|
|
* @param {Error} error object
|
|
* @api private
|
|
*/
|
|
|
|
Socket.prototype.onError = function (err) {
|
|
debug('transport error');
|
|
this.onClose('transport error', err);
|
|
};
|
|
|
|
/**
|
|
* Sets and resets ping timeout timer based on client pings.
|
|
*
|
|
* @api private
|
|
*/
|
|
|
|
Socket.prototype.setPingTimeout = function () {
|
|
var self = this;
|
|
clearTimeout(self.pingTimeoutTimer);
|
|
self.pingTimeoutTimer = setTimeout(function () {
|
|
self.onClose('ping timeout');
|
|
}, self.server.pingInterval + self.server.pingTimeout);
|
|
};
|
|
|
|
/**
|
|
* Attaches handlers for the given transport.
|
|
*
|
|
* @param {Transport} transport
|
|
* @api private
|
|
*/
|
|
|
|
Socket.prototype.setTransport = function (transport) {
|
|
this.transport = transport;
|
|
this.transport.once('error', this.onError.bind(this));
|
|
this.transport.on('packet', this.onPacket.bind(this));
|
|
this.transport.on('drain', this.flush.bind(this));
|
|
this.transport.once('close', this.onClose.bind(this, 'transport close'));
|
|
//this function will manage packet events (also message callbacks)
|
|
this.setupSendCallback();
|
|
};
|
|
|
|
/**
|
|
* Upgrades socket to the given transport
|
|
*
|
|
* @param {Transport} transport
|
|
* @api private
|
|
*/
|
|
|
|
Socket.prototype.maybeUpgrade = function (transport) {
|
|
debug('might upgrade socket transport from "%s" to "%s"'
|
|
, this.transport.name, transport.name);
|
|
|
|
// set transport upgrade timer
|
|
var checkInterval;
|
|
var upgradeTimeout = setTimeout(function () {
|
|
debug('client did not complete upgrade - closing transport');
|
|
clearInterval(checkInterval);
|
|
if ('open' == transport.readyState) {
|
|
transport.close();
|
|
}
|
|
}, this.server.upgradeTimeout);
|
|
|
|
var self = this;
|
|
function onPacket (packet) {
|
|
if ('ping' == packet.type && 'probe' == packet.data) {
|
|
transport.send([{ type: 'pong', data: 'probe' }]);
|
|
|
|
// we force a polling cycle to ensure a fast upgrade
|
|
function check () {
|
|
if ('polling' == self.transport.name && self.transport.writable) {
|
|
debug('writing a noop packet to polling for fast upgrade');
|
|
self.transport.send([{ type: 'noop' }]);
|
|
}
|
|
}
|
|
|
|
checkInterval = setInterval(check, 100);
|
|
} else if ('upgrade' == packet.type) {
|
|
debug('got upgrade packet - upgrading');
|
|
self.upgraded = true;
|
|
self.emit('upgrade', transport);
|
|
self.clearTransport();
|
|
self.setTransport(transport);
|
|
self.setPingTimeout();
|
|
self.flush();
|
|
clearInterval(checkInterval);
|
|
clearTimeout(upgradeTimeout);
|
|
transport.removeListener('packet', onPacket);
|
|
} else {
|
|
transport.close();
|
|
}
|
|
}
|
|
transport.on('packet', onPacket);
|
|
};
|
|
|
|
/**
|
|
* Clears listeners and timers associated with current transport.
|
|
*
|
|
* @api private
|
|
*/
|
|
|
|
Socket.prototype.clearTransport = function () {
|
|
// silence further transport errors and prevent uncaught exceptions
|
|
this.transport.on('error', function(){
|
|
debug('error triggered by discarded transport');
|
|
});
|
|
clearTimeout(this.pingIntervalTimer);
|
|
clearTimeout(this.pingTimeoutTimer);
|
|
};
|
|
|
|
/**
|
|
* Called upon transport considered closed.
|
|
* Possible reasons: `ping timeout`, `client error`, `parse error`,
|
|
* `transport error`, `server close`, `transport close`
|
|
*/
|
|
|
|
Socket.prototype.onClose = function (reason, description) {
|
|
if ('closed' != this.readyState) {
|
|
this.packetsFn = [];
|
|
this.clearTransport();
|
|
this.readyState = 'closed';
|
|
this.emit('close', reason, description);
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Setup and manage send callback
|
|
*
|
|
* @api private
|
|
*/
|
|
|
|
Socket.prototype.setupSendCallback = function () {
|
|
var self = this;
|
|
//the message was sent successfully, execute the callback
|
|
this.transport.on('drain', function() {
|
|
if (self.packetsFn.length > 0) {
|
|
var seqFn = self.packetsFn.splice(0,1)[0];
|
|
if ('function' == typeof seqFn) {
|
|
debug('executing send callback');
|
|
seqFn(self.transport);
|
|
} else if (Array.isArray(seqFn)) {
|
|
debug('executing batch send callback');
|
|
for (var i in seqFn) {
|
|
if ('function' == typeof seqFn) {
|
|
seqFn[i](self.transport);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
});
|
|
};
|
|
|
|
/**
|
|
* Sends a message packet.
|
|
*
|
|
* @param {String} message
|
|
* @param {Function} callback
|
|
* @return {Socket} for chaining
|
|
* @api public
|
|
*/
|
|
|
|
Socket.prototype.send =
|
|
Socket.prototype.write = function(data, callback){
|
|
this.sendPacket('message', data, callback);
|
|
return this;
|
|
};
|
|
|
|
/**
|
|
* Sends a packet.
|
|
*
|
|
* @param {String} packet type
|
|
* @param {String} optional, data
|
|
* @api private
|
|
*/
|
|
|
|
Socket.prototype.sendPacket = function (type, data, callback) {
|
|
if ('closing' != this.readyState) {
|
|
debug('sending packet "%s" (%s)', type, data);
|
|
|
|
var packet = { type: type };
|
|
if (data) packet.data = data;
|
|
|
|
// exports packetCreate event
|
|
this.emit('packetCreate', packet);
|
|
|
|
this.writeBuffer.push(packet);
|
|
|
|
//add send callback to object
|
|
this.packetsFn.push(callback);
|
|
|
|
this.flush();
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Attempts to flush the packets buffer.
|
|
*
|
|
* @api private
|
|
*/
|
|
|
|
Socket.prototype.flush = function () {
|
|
if ('closed' != this.readyState && this.transport.writable
|
|
&& this.writeBuffer.length) {
|
|
debug('flushing buffer to transport');
|
|
this.emit('flush', this.writeBuffer);
|
|
this.server.emit('flush', this, this.writeBuffer);
|
|
var wbuf = this.writeBuffer;
|
|
this.writeBuffer = [];
|
|
if (!this.transport.supportsFraming) {
|
|
this.packetsFn = [this.packetsFn];
|
|
}
|
|
this.transport.send(wbuf);
|
|
this.emit('drain');
|
|
this.server.emit('drain', this);
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Get available upgrades for this socket.
|
|
*
|
|
* @api private
|
|
*/
|
|
|
|
Socket.prototype.getAvailableUpgrades = function () {
|
|
var availableUpgrades = [];
|
|
var allUpgrades = this.server.upgrades(this.transport.name);
|
|
for (var i = 0, l = allUpgrades.length; i < l; ++i) {
|
|
var upg = allUpgrades[i];
|
|
if (this.server.transports.indexOf(upg) != -1) {
|
|
availableUpgrades.push(upg);
|
|
}
|
|
}
|
|
return availableUpgrades;
|
|
};
|
|
|
|
/**
|
|
* Closes the socket and underlying transport.
|
|
*
|
|
* @return {Socket} for chaining
|
|
* @api public
|
|
*/
|
|
|
|
Socket.prototype.close = function () {
|
|
if ('open' == this.readyState) {
|
|
this.readyState = 'closing';
|
|
var self = this;
|
|
this.transport.close(function () {
|
|
self.onClose('forced close');
|
|
});
|
|
}
|
|
};
|