From 2b216902e121ac2205444019b6d9316654809b29 Mon Sep 17 00:00:00 2001 From: Damien Arrachequesne Date: Mon, 22 May 2017 13:01:59 +0200 Subject: [PATCH] [fix] Fix timing issues with middleware (#2948) Using a middleware could previously lead to a connecting client receiving a connect event from the server before the server triggers its own connect event. --- lib/namespace.js | 2 ++ lib/socket.js | 7 ++++--- test/socket.io.js | 29 +++++++++++++++++++++++++++++ 3 files changed, 35 insertions(+), 3 deletions(-) diff --git a/lib/namespace.js b/lib/namespace.js index 5dc7b4db7c..69d07c5c88 100644 --- a/lib/namespace.js +++ b/lib/namespace.js @@ -99,6 +99,8 @@ Namespace.prototype.initAdapter = function(){ */ Namespace.prototype.use = function(fn){ + debug('removing initial packet'); + delete this.server.eio.initialPacket; this.fns.push(fn); return this; }; diff --git a/lib/socket.js b/lib/socket.js index bb4a015330..fdb99b120a 100644 --- a/lib/socket.js +++ b/lib/socket.js @@ -297,9 +297,10 @@ Socket.prototype.onconnect = function(){ debug('socket connected - writing packet'); this.nsp.connected[this.id] = this; this.join(this.id); - // the CONNECT packet for the default namespace - // has already been sent as an `initialPacket` - if (this.nsp.name !== '/') { + var skip = this.nsp.name === '/' && this.nsp.fns.length === 0; + if (skip) { + debug('packet already sent in initial handshake'); + } else { this.packet({ type: parser.CONNECT }); } }; diff --git a/test/socket.io.js b/test/socket.io.js index 9632077524..84a252b5fa 100644 --- a/test/socket.io.js +++ b/test/socket.io.js @@ -87,6 +87,9 @@ describe('socket.io', function(){ srv.set('authorization', function(o, f) { f(null, false); }); var socket = client(httpSrv); + socket.on('connect', function(){ + expect().fail(); + }); socket.on('error', function(err) { expect(err).to.be('Not authorized'); done(); @@ -2145,6 +2148,9 @@ describe('socket.io', function(){ }); srv.listen(function(){ var socket = client(srv); + socket.on('connect', function(){ + done(new Error('nope')); + }); socket.on('error', function(err){ expect(err).to.be('Authentication error'); done(); @@ -2163,6 +2169,9 @@ describe('socket.io', function(){ }); srv.listen(function(){ var socket = client(srv); + socket.on('connect', function(){ + done(new Error('nope')); + }); socket.on('error', function(err){ expect(err).to.eql({ a: 'b', c: 3 }); done(); @@ -2186,6 +2195,26 @@ describe('socket.io', function(){ }); }); + it('should only call connection after (lengthy) fns', function(done){ + var srv = http(); + var sio = io(srv); + var authenticated = false; + + sio.use(function(socket, next){ + setTimeout(function () { + authenticated = true; + next(); + }, 300); + }); + srv.listen(function(){ + var socket = client(srv); + socket.on('connect', function(){ + expect(authenticated).to.be(true); + done(); + }); + }); + }); + it('should be ignored if socket gets closed', function(done){ var srv = http(); var sio = io(srv);