Fix endless reconnection bug (#1389)

This commit is contained in:
Gus Caplan
2017-04-17 06:32:26 -05:00
committed by Amish Shah
parent 6bae7a135f
commit 0d83f80fc0
2 changed files with 35 additions and 16 deletions

View File

@@ -24,18 +24,18 @@ const WebSocket = (function findWebSocket() {
* Abstracts a WebSocket connection with decoding/encoding for the discord gateway * Abstracts a WebSocket connection with decoding/encoding for the discord gateway
* @private * @private
*/ */
class WebSocketConnection extends WebSocket { class WebSocketConnection extends EventEmitter {
/** /**
* @param {string} gateway Websocket gateway to connect to * @param {string} gateway Websocket gateway to connect to
*/ */
constructor(gateway) { constructor(gateway) {
super(gateway); super(gateway);
this.e = new EventEmitter(); this.ws = new WebSocket(gateway);
if (browser) this.binaryType = 'arraybuffer'; if (browser) this.ws.binaryType = 'arraybuffer';
this.onmessage = this.eventMessage.bind(this); this.ws.onmessage = this.eventMessage.bind(this);
this.onopen = this.e.emit.bind(this.e, 'open'); this.ws.onopen = this.emit.bind(this, 'open');
this.onclose = this.e.emit.bind(this.e, 'close'); this.ws.onclose = this.emit.bind(this, 'close');
this.onerror = this.e.emit.bind(this.e, 'error'); this.ws.onerror = this.emit.bind(this, 'error');
} }
/** /**
@@ -46,10 +46,10 @@ class WebSocketConnection extends WebSocket {
eventMessage(event) { eventMessage(event) {
try { try {
const data = this.unpack(event.data); const data = this.unpack(event.data);
this.e.emit('packet', data); this.emit('packet', data);
return true; return true;
} catch (err) { } catch (err) {
if (this.e.listenerCount('decodeError')) this.e.emit('decodeError', err); if (this.listenerCount('decodeError')) this.emit('decodeError', err);
return false; return false;
} }
} }
@@ -59,7 +59,7 @@ class WebSocketConnection extends WebSocket {
* @param {string|Buffer} data Data to send * @param {string|Buffer} data Data to send
*/ */
send(data) { send(data) {
super.send(this.pack(data)); this.ws.send(this.pack(data));
} }
/** /**
@@ -94,6 +94,24 @@ class WebSocketConnection extends WebSocket {
inflate(data) { inflate(data) {
return erlpack ? data : zlib.inflateSync(data).toString(); return erlpack ? data : zlib.inflateSync(data).toString();
} }
/**
* State of the WebSocket
* @type {number}
* @readonly
*/
get readyState() {
return this.ws.readyState;
}
/**
* Close the WebSocket
* @param {number} code Close code
* @param {string} [reason] Close reason
*/
close(code, reason) {
this.ws.close(code, reason);
}
} }
/** /**
@@ -101,5 +119,6 @@ class WebSocketConnection extends WebSocket {
* @type {string} * @type {string}
*/ */
WebSocketConnection.ENCODING = erlpack ? 'etf' : 'json'; WebSocketConnection.ENCODING = erlpack ? 'etf' : 'json';
WebSocketConnection.WebSocket = WebSocket;
module.exports = WebSocketConnection; module.exports = WebSocketConnection;

View File

@@ -82,10 +82,10 @@ class WebSocketManager extends EventEmitter {
this.normalReady = false; this.normalReady = false;
if (this.status !== Constants.Status.RECONNECTING) this.status = Constants.Status.CONNECTING; if (this.status !== Constants.Status.RECONNECTING) this.status = Constants.Status.CONNECTING;
this.ws = new WebSocketConnection(gateway); this.ws = new WebSocketConnection(gateway);
this.ws.e.on('open', this.eventOpen.bind(this)); this.ws.on('open', this.eventOpen.bind(this));
this.ws.e.on('packet', this.eventPacket.bind(this)); this.ws.on('packet', this.eventPacket.bind(this));
this.ws.e.on('close', this.eventClose.bind(this)); this.ws.on('close', this.eventClose.bind(this));
this.ws.e.on('error', this.eventError.bind(this)); this.ws.on('error', this.eventError.bind(this));
this._queue = []; this._queue = [];
this._remaining = 120; this._remaining = 120;
this.client.setInterval(() => { this.client.setInterval(() => {
@@ -141,7 +141,7 @@ class WebSocketManager extends EventEmitter {
} }
_send(data) { _send(data) {
if (this.ws.readyState === WebSocketConnection.OPEN) { if (this.ws.readyState === WebSocketConnection.WebSocket.OPEN) {
this.emit('send', data); this.emit('send', data);
this.ws.send(data); this.ws.send(data);
} }
@@ -149,7 +149,7 @@ class WebSocketManager extends EventEmitter {
doQueue() { doQueue() {
const item = this._queue[0]; const item = this._queue[0];
if (!(this.ws.readyState === WebSocketConnection.OPEN && item)) return; if (!(this.ws.readyState === WebSocketConnection.WebSocket.OPEN && item)) return;
if (this.remaining === 0) { if (this.remaining === 0) {
this.client.setTimeout(this.doQueue.bind(this), Date.now() - this.remainingReset); this.client.setTimeout(this.doQueue.bind(this), Date.now() - this.remainingReset);
return; return;