bearychat-rtm-client
Advanced tools
Comparing version
{ | ||
"name": "bearychat-rtm-client", | ||
"version": "0.8.0", | ||
"version": "0.9.0", | ||
"description": "RTM client for BearyChat", | ||
@@ -5,0 +5,0 @@ "main": "lib/index.js", |
@@ -236,5 +236,5 @@ 'use strict'; | ||
_this4._pingTimeout = options.pingTimeout || 3000; | ||
_this4._pingTimeout = options.pingTimeout || 15 * 1000; | ||
// following options are internal to speed up testing. | ||
_this4._pingInterval = options.pingInterval || 5000; | ||
_this4._pingInterval = options.pingInterval || 15 * 1000; | ||
_this4._backoffMultiplier = options.backoffMultiplier || 1000; | ||
@@ -270,11 +270,29 @@ | ||
case 0: | ||
if (this._shouldDoReconnect()) { | ||
_context2.next = 2; | ||
break; | ||
} | ||
return _context2.abrupt('return'); | ||
case 2: | ||
this._state = _RTMClientState2.default.RECONNECT; | ||
_context2.next = 3; | ||
_context2.next = 5; | ||
return (0, _delay2.default)(generateInterval(this._reconnectAttempts, this._backoffMultiplier)); | ||
case 3: | ||
case 5: | ||
if (this._shouldDoReconnect()) { | ||
_context2.next = 7; | ||
break; | ||
} | ||
return _context2.abrupt('return'); | ||
case 7: | ||
this._reconnectAttempts++; | ||
this.connect(); | ||
case 5: | ||
case 9: | ||
case 'end': | ||
@@ -335,6 +353,8 @@ return _context2.stop(); | ||
value: function close() { | ||
var reason = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : 'unknown reason'; | ||
if (this._connection && this._state !== _RTMClientState2.default.CLOSING) { | ||
this._state = _RTMClientState2.default.CLOSING; | ||
this._forceClose = true; | ||
this._connection.close(); | ||
this._connection.close(reason); | ||
} else if (this._state !== _RTMClientState2.default.CLOSED) { | ||
@@ -375,3 +395,3 @@ this._state = _RTMClientState2.default.CLOSED; | ||
function _send(_x) { | ||
function _send(_x2) { | ||
return _ref4.apply(this, arguments); | ||
@@ -419,3 +439,3 @@ } | ||
function send(_x2, _x3) { | ||
function send(_x3, _x4) { | ||
return _ref5.apply(this, arguments); | ||
@@ -467,2 +487,7 @@ } | ||
} | ||
}, { | ||
key: '_shouldDoReconnect', | ||
value: function _shouldDoReconnect() { | ||
return this._state !== _RTMClientState2.default.CLOSING && this._state !== _RTMClientState2.default.CLOSED; | ||
} | ||
}]); | ||
@@ -489,2 +514,2 @@ | ||
module.exports = exports['default']; | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["src/RTMClient.js"],"names":["ONE_MINUTE","RTMSendTimeoutError","errorMessage","rtmMessage","Error","RTMNotConnectedError","RTMReconnectTimeoutError","constructor","__proto__","prototype","RTMClient","options","_doConnect","_state","CONNECTING","wsUrl","timeoutMessage","_reconnectTimeout","_getUrl","_reconnect","emit","ERROR","_reconnectAttempts","_setConnection","url","WebSocket","pingInterval","_pingInterval","pingTimeout","_pingTimeout","_handleConnectionOpen","CONNECTED","ONLINE","_handleConnectionClose","_removeConnection","_forceClose","CLOSED","OFFLINE","CLOSE","_handleConnectionError","error","_handleConnectionMessage","message","EVENT","window","_url","_backoffMultiplier","backoffMultiplier","reconnectTimeout","_connectionEvents","OPEN","MESSAGE","INITIAL","_connection","connect","RECONNECT","generateInterval","CLOSING","close","send","getState","timeout","Infinity","Number","isFinite","_send","sendPromise","connection","process","env","NODE_ENV","forEach","name","handler","on","removeListener","RTMClientEvents","RTMClientState","RTMMessageTypes","RTMPingTimeoutError","attempts","multiplier","maxInterval","Math","min","pow","random"],"mappings":"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","file":"RTMClient.js","sourcesContent":["import { EventEmitter } from 'events';\nimport invariant from 'invariant';\nimport delay from './delay';\nimport withTimeout from './withTimeout';\nimport RTMClientEvents from './RTMClientEvents';\nimport RTMClientState from './RTMClientState';\nimport RTMConnectionEvents from './RTMConnectionEvents';\nimport RTMConnection, { RTMPingTimeoutError } from './RTMConnection';\nimport RTMMessageTypes from './RTMMessageTypes';\n\nconst ONE_MINUTE = 60 * 1000;\n\nclass RTMSendTimeoutError extends Error {\n  constructor(errorMessage, rtmMessage) {\n    super(errorMessage);\n    this.rtmMessage = rtmMessage;\n  }\n}\n\nclass RTMNotConnectedError extends Error {\n  constructor(errorMessage, rtmMessage) {\n    super(errorMessage);\n    this.rtmMessage = rtmMessage;\n  }\n}\n\nclass RTMReconnectTimeoutError extends Error {\n  constructor(errorMessage) {\n    super(errorMessage);\n    this.constructor = RTMReconnectTimeoutError;\n    this.__proto__ = RTMReconnectTimeoutError.prototype;\n  }\n};\n\n/**\n * Keep an RTM client running with multiple.\n *\n * State diagram:\n *\n *                    INITIAL\n *                       +\n *          error        |\n *      +-------------+  |\n *      v             +  v        connect\n *  RECONNECT+------->CONNECTING<---------+CLOSED\n *      ^                +                    ^\n *      |                |                    |\n *      |    server      |                    |\n *      |    close/      v        close       +\n *      +------------+CONNECTED+---------->CLOSING\n *           error\n *\n * @constructor\n * @param {Object} options\n * @param {string|Function} options.url - A string or a function returning\n *                                        a string or a Promise resolves to\n *                                        a string.\n */\nexport default class RTMClient extends EventEmitter {\n\n  static RTMClientEvents = RTMClientEvents;\n\n  static RTMClientState = RTMClientState;\n\n  static RTMMessageTypes = RTMMessageTypes;\n\n  static RTMReconnectTimeoutError = RTMReconnectTimeoutError;\n\n  static RTMPingTimeoutError = RTMPingTimeoutError;\n\n  constructor(options) {\n    super();\n\n    options = options || {};\n    const { url } = options;\n\n    invariant(\n      url,\n      '\"url\" is required.'\n    );\n\n    invariant(\n      typeof url === 'string' || typeof url === 'function',\n      '\"url\" must be a string or a function returning a string.'\n    );\n\n    let WebSocket = options.WebSocket;\n    if (!WebSocket && typeof window !== 'undefined') {\n      WebSocket = window.WebSocket;\n    }\n\n    invariant(\n      WebSocket,\n      'A Websocket client is required.'\n    );\n\n    this._url = url;\n    this.WebSocket = WebSocket;\n\n    this._pingTimeout = options.pingTimeout || 3000;\n    // following options are internal to speed up testing.\n    this._pingInterval = options.pingInterval || 5000;\n    this._backoffMultiplier = options.backoffMultiplier || 1000;\n\n    this._reconnectTimeout = options.reconnectTimeout || ONE_MINUTE;\n\n    this._connectionEvents = [\n      [RTMConnectionEvents.OPEN, this._handleConnectionOpen],\n      [RTMConnectionEvents.CLOSE, this._handleConnectionClose],\n      [RTMConnectionEvents.ERROR, this._handleConnectionError],\n      [RTMConnectionEvents.MESSAGE, this._handleConnectionMessage],\n    ];\n\n    this._state = RTMClientState.INITIAL;\n    this._connection = null;\n    this._forceClose = false;\n    this._reconnectAttempts = 1;\n\n    this.connect();\n  }\n\n  connect() {\n    invariant(\n      this._state === RTMClientState.INITIAL ||\n      this._state === RTMClientState.CLOSED ||\n      this._state === RTMClientState.RECONNECT,\n      'Invalid state: connect() should always be called when current state ' +\n      'is \"%s\", \"%s\" or \"%s\" but the current state is \"%s\".',\n      RTMClientState.INITIAL,\n      RTMClientState.CLOSED,\n      RTMClientState.RECONNECT,\n      this._state\n    );\n\n    this._doConnect();\n  }\n\n  _doConnect = async () => {\n    this._state = RTMClientState.CONNECTING;\n\n    let wsUrl;\n    const timeoutMessage = new RTMReconnectTimeoutError('Reget websocket url error');\n    try {\n      wsUrl = await withTimeout(this._reconnectTimeout, timeoutMessage, this._getUrl());\n    } catch (e) {\n      this._reconnect(); // intentionally ignore \"await\"\n      this.emit(RTMClientEvents.ERROR, e);\n      return;\n    }\n\n    this._reconnectAttempts = 1;\n    this._setConnection(new RTMConnection({\n      url: wsUrl,\n      WebSocket: this.WebSocket,\n      pingInterval: this._pingInterval,\n      pingTimeout: this._pingTimeout\n    }));\n  };\n\n  async _reconnect() {\n    this._state = RTMClientState.RECONNECT;\n    await delay(generateInterval(this._reconnectAttempts, this._backoffMultiplier));\n    this._reconnectAttempts++;\n    this.connect();\n  }\n\n  async _getUrl() {\n    const url = this._url;\n\n    if (typeof url === 'string') {\n      return url;\n    }\n\n    // assume url is a function\n    return await url();\n  }\n\n  close() {\n    if (this._connection && this._state !== RTMClientState.CLOSING) {\n      this._state = RTMClientState.CLOSING;\n      this._forceClose = true;\n      this._connection.close();\n    } else if (this._state !== RTMClientState.CLOSED) {\n      this._state = RTMClientState.CLOSED;\n      this.emit(RTMClientEvents.CLOSE);\n    }\n  }\n\n  async _send(message) {\n    if (this._connection) {\n      return await this._connection.send(message);\n    }\n    throw new RTMNotConnectedError(\n      'Client currently not connected, the current state is: ' + this.getState()\n    );\n  }\n\n\n  async send(message, timeout) {\n    if (!timeout || timeout < 0) {\n      timeout = Infinity;\n    }\n\n    if (!Number.isFinite(timeout)) {\n      return await this._send(message);\n    }\n\n    const sendPromise = this._send(message);\n    const timeoutMessage = new RTMSendTimeoutError('RTM message send timeout.', message);\n    return withTimeout(timeout, timeoutMessage, sendPromise);\n  }\n\n  _handleConnectionOpen = () => {\n    this._state = RTMClientState.CONNECTED;\n    this.emit(RTMClientEvents.ONLINE);\n  };\n\n  _handleConnectionClose = () => {\n    this._removeConnection();\n    if (this._forceClose) {\n      // client close, close normally\n      this._state = RTMClientState.CLOSED;\n      this.emit(RTMClientEvents.OFFLINE);\n      this.emit(RTMClientEvents.CLOSE);\n      this._forceClose = false;\n    } else {\n      // server close or error, re-connect\n      this._reconnect();\n      this.emit(RTMClientEvents.OFFLINE);\n    }\n  };\n\n  _handleConnectionError = (error) => {\n    this.emit(RTMClientEvents.ERROR, error);\n  };\n\n  _handleConnectionMessage = (message) => {\n    this.emit(RTMClientEvents.EVENT, message);\n  };\n\n  getState() {\n    return this._state;\n  }\n\n  _setConnection(connection) {\n    if (process.env.NODE_ENV !== 'production') {\n      invariant(\n        !this._connection,\n        'Should not set connection when connection already exists.'\n      );\n    }\n\n    this._connectionEvents.forEach(([name, handler]) => {\n      connection.on(name, handler);\n    });\n\n    this._connection = connection;\n  }\n\n  _removeConnection() {\n    const connection = this._connection;\n\n    if (process.env.NODE_ENV !== 'production') {\n      invariant(\n        connection,\n        'Connection not set or already removed.'\n      );\n    }\n\n    this._connectionEvents.forEach(([name, handler]) => {\n      connection.removeListener(name, handler);\n    });\n\n    this._connection = null;\n  }\n}\n\n// exponential backoff, 30 seconds max\nfunction generateInterval(attempts, multiplier = 1000) {\n  const maxInterval = Math.min(30, (Math.pow(2, attempts) - 1)) * multiplier;\n  return Math.random() * maxInterval;\n}\n"]} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["src/RTMClient.js"],"names":["ONE_MINUTE","RTMSendTimeoutError","errorMessage","rtmMessage","Error","RTMNotConnectedError","RTMReconnectTimeoutError","constructor","__proto__","prototype","RTMClient","options","_doConnect","_state","CONNECTING","wsUrl","timeoutMessage","_reconnectTimeout","_getUrl","_reconnect","emit","ERROR","_reconnectAttempts","_setConnection","url","WebSocket","pingInterval","_pingInterval","pingTimeout","_pingTimeout","_handleConnectionOpen","CONNECTED","ONLINE","_handleConnectionClose","_removeConnection","_forceClose","CLOSED","OFFLINE","CLOSE","_handleConnectionError","error","_handleConnectionMessage","message","EVENT","window","_url","_backoffMultiplier","backoffMultiplier","reconnectTimeout","_connectionEvents","OPEN","MESSAGE","INITIAL","_connection","connect","RECONNECT","_shouldDoReconnect","generateInterval","reason","CLOSING","close","send","getState","timeout","Infinity","Number","isFinite","_send","sendPromise","connection","process","env","NODE_ENV","forEach","name","handler","on","removeListener","RTMClientEvents","RTMClientState","RTMMessageTypes","RTMPingTimeoutError","attempts","multiplier","maxInterval","Math","min","pow","random"],"mappings":"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","file":"RTMClient.js","sourcesContent":["import { EventEmitter } from 'events';\nimport invariant from 'invariant';\nimport delay from './delay';\nimport withTimeout from './withTimeout';\nimport RTMClientEvents from './RTMClientEvents';\nimport RTMClientState from './RTMClientState';\nimport RTMConnectionEvents from './RTMConnectionEvents';\nimport RTMConnection, { RTMPingTimeoutError } from './RTMConnection';\nimport RTMMessageTypes from './RTMMessageTypes';\n\nconst ONE_MINUTE = 60 * 1000;\n\nclass RTMSendTimeoutError extends Error {\n  constructor(errorMessage, rtmMessage) {\n    super(errorMessage);\n    this.rtmMessage = rtmMessage;\n  }\n}\n\nclass RTMNotConnectedError extends Error {\n  constructor(errorMessage, rtmMessage) {\n    super(errorMessage);\n    this.rtmMessage = rtmMessage;\n  }\n}\n\nclass RTMReconnectTimeoutError extends Error {\n  constructor(errorMessage) {\n    super(errorMessage);\n    this.constructor = RTMReconnectTimeoutError;\n    this.__proto__ = RTMReconnectTimeoutError.prototype;\n  }\n};\n\n/**\n * Keep an RTM client running with multiple.\n *\n * State diagram:\n *\n *                    INITIAL\n *                       +\n *          error        |\n *      +-------------+  |\n *      v             +  v        connect\n *  RECONNECT+------->CONNECTING<---------+CLOSED\n *      ^                +                    ^\n *      |                |                    |\n *      |    server      |                    |\n *      |    close/      v        close       +\n *      +------------+CONNECTED+---------->CLOSING\n *           error\n *\n * @constructor\n * @param {Object} options\n * @param {string|Function} options.url - A string or a function returning\n *                                        a string or a Promise resolves to\n *                                        a string.\n */\nexport default class RTMClient extends EventEmitter {\n\n  static RTMClientEvents = RTMClientEvents;\n\n  static RTMClientState = RTMClientState;\n\n  static RTMMessageTypes = RTMMessageTypes;\n\n  static RTMReconnectTimeoutError = RTMReconnectTimeoutError;\n\n  static RTMPingTimeoutError = RTMPingTimeoutError;\n\n  constructor(options) {\n    super();\n\n    options = options || {};\n    const { url } = options;\n\n    invariant(\n      url,\n      '\"url\" is required.'\n    );\n\n    invariant(\n      typeof url === 'string' || typeof url === 'function',\n      '\"url\" must be a string or a function returning a string.'\n    );\n\n    let WebSocket = options.WebSocket;\n    if (!WebSocket && typeof window !== 'undefined') {\n      WebSocket = window.WebSocket;\n    }\n\n    invariant(\n      WebSocket,\n      'A Websocket client is required.'\n    );\n\n    this._url = url;\n    this.WebSocket = WebSocket;\n\n    this._pingTimeout = options.pingTimeout || 15 * 1000;\n    // following options are internal to speed up testing.\n    this._pingInterval = options.pingInterval || 15 * 1000;\n    this._backoffMultiplier = options.backoffMultiplier || 1000;\n\n    this._reconnectTimeout = options.reconnectTimeout || ONE_MINUTE;\n\n    this._connectionEvents = [\n      [RTMConnectionEvents.OPEN, this._handleConnectionOpen],\n      [RTMConnectionEvents.CLOSE, this._handleConnectionClose],\n      [RTMConnectionEvents.ERROR, this._handleConnectionError],\n      [RTMConnectionEvents.MESSAGE, this._handleConnectionMessage],\n    ];\n\n    this._state = RTMClientState.INITIAL;\n    this._connection = null;\n    this._forceClose = false;\n    this._reconnectAttempts = 1;\n\n    this.connect();\n  }\n\n  connect() {\n    invariant(\n      this._state === RTMClientState.INITIAL ||\n      this._state === RTMClientState.CLOSED ||\n      this._state === RTMClientState.RECONNECT,\n      'Invalid state: connect() should always be called when current state ' +\n      'is \"%s\", \"%s\" or \"%s\" but the current state is \"%s\".',\n      RTMClientState.INITIAL,\n      RTMClientState.CLOSED,\n      RTMClientState.RECONNECT,\n      this._state\n    );\n\n    this._doConnect();\n  }\n\n  _doConnect = async () => {\n    this._state = RTMClientState.CONNECTING;\n\n    let wsUrl;\n    const timeoutMessage = new RTMReconnectTimeoutError('Reget websocket url error');\n    try {\n      wsUrl = await withTimeout(this._reconnectTimeout, timeoutMessage, this._getUrl());\n    } catch (e) {\n      this._reconnect(); // intentionally ignore \"await\"\n      this.emit(RTMClientEvents.ERROR, e);\n      return;\n    }\n\n    this._reconnectAttempts = 1;\n    this._setConnection(new RTMConnection({\n      url: wsUrl,\n      WebSocket: this.WebSocket,\n      pingInterval: this._pingInterval,\n      pingTimeout: this._pingTimeout\n    }));\n  };\n\n  async _reconnect() {\n    // There is possibilty that a single call of _reconnect() is followd after an asynchronous task,\n    // we should check the closing/close state before really do the reconnect.\n    if (!this._shouldDoReconnect()) return;\n\n    this._state = RTMClientState.RECONNECT;\n    await delay(generateInterval(this._reconnectAttempts, this._backoffMultiplier));\n\n    // During the time of the delay, `this.close()` may be called, so we should\n    // recheck the state before connection.\n    if (!this._shouldDoReconnect()) return;\n\n    this._reconnectAttempts++;\n    this.connect();\n  }\n\n  async _getUrl() {\n    const url = this._url;\n\n    if (typeof url === 'string') {\n      return url;\n    }\n\n    // assume url is a function\n    return await url();\n  }\n\n  close(reason = 'unknown reason') {\n    if (this._connection && this._state !== RTMClientState.CLOSING) {\n      this._state = RTMClientState.CLOSING;\n      this._forceClose = true;\n      this._connection.close(reason);\n    } else if (this._state !== RTMClientState.CLOSED) {\n      this._state = RTMClientState.CLOSED;\n      this.emit(RTMClientEvents.CLOSE);\n    }\n  }\n\n  async _send(message) {\n    if (this._connection) {\n      return await this._connection.send(message);\n    }\n    throw new RTMNotConnectedError(\n      'Client currently not connected, the current state is: ' + this.getState()\n    );\n  }\n\n\n  async send(message, timeout) {\n    if (!timeout || timeout < 0) {\n      timeout = Infinity;\n    }\n\n    if (!Number.isFinite(timeout)) {\n      return await this._send(message);\n    }\n\n    const sendPromise = this._send(message);\n    const timeoutMessage = new RTMSendTimeoutError('RTM message send timeout.', message);\n    return withTimeout(timeout, timeoutMessage, sendPromise);\n  }\n\n  _handleConnectionOpen = () => {\n    this._state = RTMClientState.CONNECTED;\n    this.emit(RTMClientEvents.ONLINE);\n  };\n\n  _handleConnectionClose = () => {\n    this._removeConnection();\n    if (this._forceClose) {\n      // client close, close normally\n      this._state = RTMClientState.CLOSED;\n      this.emit(RTMClientEvents.OFFLINE);\n      this.emit(RTMClientEvents.CLOSE);\n      this._forceClose = false;\n    } else {\n      // server close or error, re-connect\n      this._reconnect();\n      this.emit(RTMClientEvents.OFFLINE);\n    }\n  };\n\n  _handleConnectionError = (error) => {\n    this.emit(RTMClientEvents.ERROR, error);\n  };\n\n  _handleConnectionMessage = (message) => {\n    this.emit(RTMClientEvents.EVENT, message);\n  };\n\n  getState() {\n    return this._state;\n  }\n\n  _setConnection(connection) {\n    if (process.env.NODE_ENV !== 'production') {\n      invariant(\n        !this._connection,\n        'Should not set connection when connection already exists.'\n      );\n    }\n\n    this._connectionEvents.forEach(([name, handler]) => {\n      connection.on(name, handler);\n    });\n\n    this._connection = connection;\n  }\n\n  _removeConnection() {\n    const connection = this._connection;\n\n    if (process.env.NODE_ENV !== 'production') {\n      invariant(\n        connection,\n        'Connection not set or already removed.'\n      );\n    }\n\n    this._connectionEvents.forEach(([name, handler]) => {\n      connection.removeListener(name, handler);\n    });\n\n    this._connection = null;\n  }\n\n  _shouldDoReconnect() {\n    return this._state !== RTMClientState.CLOSING && this._state !== RTMClientState.CLOSED;\n  }\n}\n\n// exponential backoff, 30 seconds max\nfunction generateInterval(attempts, multiplier = 1000) {\n  const maxInterval = Math.min(30, (Math.pow(2, attempts) - 1)) * multiplier;\n  return Math.random() * maxInterval;\n}\n"]} |
@@ -38,6 +38,2 @@ 'use strict'; | ||
var _withTimeout = require('./withTimeout'); | ||
var _withTimeout2 = _interopRequireDefault(_withTimeout); | ||
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } | ||
@@ -69,4 +65,2 @@ | ||
var pingTimeoutError = new RTMPingTimeoutError('Ping timeouted.'); | ||
/** | ||
@@ -96,2 +90,3 @@ * Keep a WebSocket connection with server, handling heartbeat events, | ||
var RTMConnection = function (_EventEmitter) { | ||
@@ -145,2 +140,3 @@ _inherits(RTMConnection, _EventEmitter); | ||
_this2._startLoop = _asyncToGenerator(_regenerator2.default.mark(function _callee() { | ||
var elapsed; | ||
return _regenerator2.default.wrap(function _callee$(_context) { | ||
@@ -151,2 +147,9 @@ while (1) { | ||
if (!(_this2._state === _RTMConnectionState2.default.CONNECTED)) { | ||
_context.next = 11; | ||
break; | ||
} | ||
elapsed = Date.now() - _this2._lastSentTs; | ||
if (!(elapsed < _this2._pingInterval)) { | ||
_context.next = 6; | ||
@@ -156,11 +159,19 @@ break; | ||
_context.next = 5; | ||
return (0, _delay2.default)(_this2._pingInterval - elapsed); | ||
case 5: | ||
return _context.abrupt('continue', 0); | ||
case 6: | ||
_this2._ping(); | ||
_context.next = 4; | ||
_context.next = 9; | ||
return (0, _delay2.default)(_this2._pingInterval); | ||
case 4: | ||
case 9: | ||
_context.next = 0; | ||
break; | ||
case 6: | ||
case 11: | ||
case 'end': | ||
@@ -179,2 +190,3 @@ return _context.stop(); | ||
_this2._callbackMap = new Map(); | ||
_this2._lastSentTs = 0; | ||
@@ -209,3 +221,44 @@ _this2._ws.addEventListener('open', _this2._handleOpen); | ||
key: 'send', | ||
value: function send(message) { | ||
value: function () { | ||
var _ref3 = _asyncToGenerator(_regenerator2.default.mark(function _callee2(message) { | ||
return _regenerator2.default.wrap(function _callee2$(_context2) { | ||
while (1) { | ||
switch (_context2.prev = _context2.next) { | ||
case 0: | ||
_context2.prev = 0; | ||
_context2.next = 3; | ||
return this._send(message); | ||
case 3: | ||
return _context2.abrupt('return', _context2.sent); | ||
case 6: | ||
_context2.prev = 6; | ||
_context2.t0 = _context2['catch'](0); | ||
if (_context2.t0 instanceof RTMPingTimeoutError) { | ||
if (this._state !== _RTMConnectionState2.default.CLOSED) { | ||
this.emit(_RTMConnectionEvents2.default.ERROR, _context2.t0); | ||
this._terminate('Reply timeouted.'); | ||
} | ||
} | ||
throw _context2.t0; | ||
case 10: | ||
case 'end': | ||
return _context2.stop(); | ||
} | ||
} | ||
}, _callee2, this, [[0, 6]]); | ||
})); | ||
function send(_x) { | ||
return _ref3.apply(this, arguments); | ||
} | ||
return send; | ||
}() | ||
}, { | ||
key: '_send', | ||
value: function _send(message) { | ||
var _this4 = this; | ||
@@ -219,9 +272,18 @@ | ||
var callIdMap = this._callbackMap; | ||
var callbackMap = this._callbackMap; | ||
var callId = message.call_id; | ||
(0, _invariant2.default)(!callIdMap.has(callId), 'Duplicate call id %s', callId); | ||
(0, _invariant2.default)(!callbackMap.has(callId), 'Duplicate call id %s', callId); | ||
return new Promise(function (resolve) { | ||
callIdMap.set(callId, resolve); | ||
return new Promise(function (resolve, reject) { | ||
var timer = setTimeout(function () { | ||
reject(new RTMPingTimeoutError()); | ||
}, _this4._pingTimeout); | ||
callbackMap.set(callId, function () { | ||
clearTimeout(timer); | ||
resolve.apply(undefined, arguments); | ||
}); | ||
_this4._lastSentTs = Date.now(); | ||
_this4._ws.send(JSON.stringify(message)); | ||
@@ -233,11 +295,4 @@ }); | ||
value: function _ping() { | ||
var _this5 = this; | ||
(0, _withTimeout2.default)(this._pingTimeout, pingTimeoutError, this.send({ | ||
this.send({ | ||
type: _RTMMessageTypes2.default.PING | ||
})).catch(function (error) { | ||
if (error instanceof RTMPingTimeoutError && _this5._state !== _RTMConnectionState2.default.CLOSED) { | ||
_this5.emit(_RTMConnectionEvents2.default.ERROR, error); | ||
_this5._terminate(); | ||
} | ||
}); | ||
@@ -247,9 +302,9 @@ } | ||
key: 'close', | ||
value: function close() { | ||
value: function close(reason) { | ||
this._state = _RTMConnectionState2.default.CLOSING; | ||
this._ws.close(); | ||
this._ws.close(1000, reason); | ||
} | ||
}, { | ||
key: '_terminate', | ||
value: function _terminate() { | ||
value: function _terminate(reason) { | ||
this._ws.removeEventListener('open', this._handleOpen); | ||
@@ -262,3 +317,3 @@ this._ws.removeEventListener('close', this._handleClose); | ||
this._ws.close(); | ||
this._ws.close(1000, reason); | ||
} | ||
@@ -271,2 +326,2 @@ }]); | ||
exports.default = RTMConnection; | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["src/RTMConnection.js"],"names":["RTMPingTimeoutError","errorMessage","constructor","__proto__","prototype","Error","pingTimeoutError","RTMConnection","url","WebSocket","pingInterval","pingTimeout","state","_handleOpen","_state","CONNECTED","emit","OPEN","_startLoop","_handleClose","CLOSED","CLOSE","_handleMessage","event","message","JSON","parse","data","type","PONG","OK","REPLY","_handleReplyMessage","MESSAGE","_handleError","error","ERROR","_ping","_pingInterval","_pingTimeout","_currentCallId","INITIAL","_ws","_callbackMap","Map","addEventListener","callbackMap","callId","call_id","process","env","NODE_ENV","has","callback","get","delete","_getNextCallId","callIdMap","Promise","resolve","set","send","stringify","PING","catch","_terminate","CLOSING","close","removeEventListener"],"mappings":"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","file":"RTMConnection.js","sourcesContent":["import { EventEmitter }  from 'events';\nimport RTMConnectionState from './RTMConnectionState';\nimport RTMConnectionEvents from './RTMConnectionEvents';\nimport RTMMessageTypes from './RTMMessageTypes';\nimport delay from './delay';\nimport invariant from 'invariant';\nimport withTimeout from './withTimeout';\n\nexport class RTMPingTimeoutError extends Error {\n  constructor(errorMessage) {\n    super(errorMessage);\n    this.constructor = RTMPingTimeoutError;\n    this.__proto__ = RTMPingTimeoutError.prototype;\n  }\n}\n\nconst pingTimeoutError = new RTMPingTimeoutError('Ping timeouted.');\n\n/**\n * Keep a WebSocket connection with server, handling heartbeat events,\n * omitting obsolete message types.\n *\n * State diagram:\n *\n *    INITIAL\n *       +\n *       |\n *       |\n *       v\n *   CONNECTED +-+\n *       +       |\n * client|       |\n * close |       | server\n *       v       | close/\n *    CLOSING    | error\n *       +       |\n *       |       |\n *       |       |\n *       v       |\n *    CLOSED <---+\n */\nexport default class RTMConnection extends EventEmitter {\n\n  state = {};\n\n  constructor({ url, WebSocket, pingInterval, pingTimeout }) {\n    super();\n    this._pingTimeout = pingTimeout;\n    this._pingInterval = pingInterval;\n    this._currentCallId = 0;\n    this._state = RTMConnectionState.INITIAL;\n    this._ws = new WebSocket(url);\n    this._callbackMap = new Map();\n\n    this._ws.addEventListener('open', this._handleOpen);\n    this._ws.addEventListener('close', this._handleClose);\n    this._ws.addEventListener('message', this._handleMessage);\n    this._ws.addEventListener('error', this._handleError);\n  }\n\n  _handleOpen = () => {\n    this._state = RTMConnectionState.CONNECTED;\n    this.emit(RTMConnectionEvents.OPEN);\n    this._startLoop();\n  };\n\n  _handleClose = () => {\n    this._state = RTMConnectionState.CLOSED;\n    this.emit(RTMConnectionEvents.CLOSE);\n  };\n\n  _handleMessage = (event) => {\n    const message = JSON.parse(event.data);\n    switch (message.type) {\n      case RTMMessageTypes.PONG:\n      case RTMMessageTypes.OK:\n        // ignore deprecated events\n        break;\n      case RTMMessageTypes.REPLY:\n        this._handleReplyMessage(message);\n        break;\n      default:\n        this.emit(RTMConnectionEvents.MESSAGE, message);\n    }\n  };\n\n  _handleReplyMessage(message) {\n    const callbackMap = this._callbackMap;\n    const callId = message.call_id;\n\n    if (process.env.NODE_ENV !== 'production') {\n      invariant(\n        callbackMap.has(callId),\n        'Call id replied without sending: %s',\n        callId\n      );\n    }\n\n    const callback = callbackMap.get(callId);\n    callbackMap.delete(callId);\n    callback && callback(message);\n  }\n\n  _handleError = (error) => {\n    this.emit(RTMConnectionEvents.ERROR, error);\n  };\n\n  _getNextCallId() {\n    return this._currentCallId++;\n  }\n\n  send(message) {\n    if (!message.call_id) {\n      message = {\n        ...message,\n        call_id: this._getNextCallId()\n      };\n    }\n\n    const callIdMap = this._callbackMap;\n    const callId = message.call_id;\n\n    invariant(\n      !callIdMap.has(callId),\n      'Duplicate call id %s',\n      callId\n    );\n\n    return new Promise((resolve) => {\n      callIdMap.set(callId, resolve);\n      this._ws.send(JSON.stringify(message));\n    });\n  }\n\n  _ping() {\n    withTimeout(this._pingTimeout, pingTimeoutError, this.send({\n      type: RTMMessageTypes.PING\n    })).catch(error => {\n      if (error instanceof RTMPingTimeoutError && this._state !== RTMConnectionState.CLOSED) {\n        this.emit(RTMConnectionEvents.ERROR, error);\n        this._terminate();\n      }\n    });\n  }\n\n  _startLoop = async () => {\n    while (this._state === RTMConnectionState.CONNECTED) {\n      this._ping();\n      await delay(this._pingInterval);\n    }\n  };\n\n  close() {\n    this._state = RTMConnectionState.CLOSING;\n    this._ws.close();\n  }\n\n  _terminate() {\n    this._ws.removeEventListener('open', this._handleOpen);\n    this._ws.removeEventListener('close', this._handleClose);\n    this._ws.removeEventListener('message', this._handleMessage);\n    this._ws.removeEventListener('error', this._handleError);\n    this._state = RTMConnectionState.CLOSED;\n    this.emit(RTMConnectionEvents.CLOSE);\n\n    this._ws.close();\n  }\n}\n"]} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["src/RTMConnection.js"],"names":["RTMPingTimeoutError","errorMessage","constructor","__proto__","prototype","Error","RTMConnection","url","WebSocket","pingInterval","pingTimeout","state","_handleOpen","_state","CONNECTED","emit","OPEN","_startLoop","_handleClose","CLOSED","CLOSE","_handleMessage","event","message","JSON","parse","data","type","PONG","OK","REPLY","_handleReplyMessage","MESSAGE","_handleError","error","ERROR","elapsed","Date","now","_lastSentTs","_pingInterval","_ping","_pingTimeout","_currentCallId","INITIAL","_ws","_callbackMap","Map","addEventListener","callbackMap","callId","call_id","process","env","NODE_ENV","has","callback","get","delete","_send","_terminate","_getNextCallId","Promise","resolve","reject","timer","setTimeout","set","clearTimeout","send","stringify","PING","reason","CLOSING","close","removeEventListener"],"mappings":"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","file":"RTMConnection.js","sourcesContent":["import { EventEmitter } from 'events';\nimport RTMConnectionState from './RTMConnectionState';\nimport RTMConnectionEvents from './RTMConnectionEvents';\nimport RTMMessageTypes from './RTMMessageTypes';\nimport delay from './delay';\nimport invariant from 'invariant';\n\nexport class RTMPingTimeoutError extends Error {\n  constructor(errorMessage) {\n    super(errorMessage);\n    this.constructor = RTMPingTimeoutError;\n    this.__proto__ = RTMPingTimeoutError.prototype;\n  }\n}\n\n/**\n * Keep a WebSocket connection with server, handling heartbeat events,\n * omitting obsolete message types.\n *\n * State diagram:\n *\n *    INITIAL\n *       +\n *       |\n *       |\n *       v\n *   CONNECTED +-+\n *       +       |\n * client|       |\n * close |       | server\n *       v       | close/\n *    CLOSING    | error\n *       +       |\n *       |       |\n *       |       |\n *       v       |\n *    CLOSED <---+\n */\nexport default class RTMConnection extends EventEmitter {\n\n  state = {};\n\n  constructor({ url, WebSocket, pingInterval, pingTimeout }) {\n    super();\n    this._pingTimeout = pingTimeout;\n    this._pingInterval = pingInterval;\n    this._currentCallId = 0;\n    this._state = RTMConnectionState.INITIAL;\n    this._ws = new WebSocket(url);\n    this._callbackMap = new Map();\n    this._lastSentTs = 0;\n\n    this._ws.addEventListener('open', this._handleOpen);\n    this._ws.addEventListener('close', this._handleClose);\n    this._ws.addEventListener('message', this._handleMessage);\n    this._ws.addEventListener('error', this._handleError);\n  }\n\n  _handleOpen = () => {\n    this._state = RTMConnectionState.CONNECTED;\n    this.emit(RTMConnectionEvents.OPEN);\n    this._startLoop();\n  };\n\n  _handleClose = () => {\n    this._state = RTMConnectionState.CLOSED;\n    this.emit(RTMConnectionEvents.CLOSE);\n  };\n\n  _handleMessage = (event) => {\n    const message = JSON.parse(event.data);\n    switch (message.type) {\n      case RTMMessageTypes.PONG:\n      case RTMMessageTypes.OK:\n        // ignore deprecated events\n        break;\n      case RTMMessageTypes.REPLY:\n        this._handleReplyMessage(message);\n        break;\n      default:\n        this.emit(RTMConnectionEvents.MESSAGE, message);\n    }\n  };\n\n  _handleReplyMessage(message) {\n    const callbackMap = this._callbackMap;\n    const callId = message.call_id;\n\n    if (process.env.NODE_ENV !== 'production') {\n      invariant(\n        callbackMap.has(callId),\n        'Call id replied without sending: %s',\n        callId\n      );\n    }\n\n    const callback = callbackMap.get(callId);\n    callbackMap.delete(callId);\n    callback && callback(message);\n  }\n\n  _handleError = (error) => {\n    this.emit(RTMConnectionEvents.ERROR, error);\n  };\n\n  _getNextCallId() {\n    return this._currentCallId++;\n  }\n\n  async send(message) {\n    try {\n      return await this._send(message);\n    } catch (error) {\n      if (error instanceof RTMPingTimeoutError) {\n        if (this._state !== RTMConnectionState.CLOSED) {\n          this.emit(RTMConnectionEvents.ERROR, error);\n          this._terminate('Reply timeouted.');\n        }\n      }\n      throw error;\n    }\n  }\n\n  _send(message) {\n    if (!message.call_id) {\n      message = {\n        ...message,\n        call_id: this._getNextCallId()\n      };\n    }\n\n    const callbackMap = this._callbackMap;\n    const callId = message.call_id;\n\n    invariant(\n      !callbackMap.has(callId),\n      'Duplicate call id %s',\n      callId\n    );\n\n    return new Promise((resolve, reject) => {\n      const timer = setTimeout(() => {\n        reject(new RTMPingTimeoutError());\n      }, this._pingTimeout);\n\n      callbackMap.set(callId, (...args) => {\n        clearTimeout(timer);\n        resolve(...args);\n      });\n\n      this._lastSentTs = Date.now();\n      this._ws.send(JSON.stringify(message));\n    });\n  }\n\n  _ping() {\n    this.send({\n      type: RTMMessageTypes.PING\n    });\n  }\n\n  _startLoop = async () => {\n    while (this._state === RTMConnectionState.CONNECTED) {\n      const elapsed = Date.now() - this._lastSentTs;\n      if (elapsed < this._pingInterval) {\n        await delay(this._pingInterval - elapsed);\n        continue;\n      }\n\n      this._ping();\n      await delay(this._pingInterval);\n    }\n  };\n\n  close(reason) {\n    this._state = RTMConnectionState.CLOSING;\n    this._ws.close(1000, reason);\n  }\n\n  _terminate(reason) {\n    this._ws.removeEventListener('open', this._handleOpen);\n    this._ws.removeEventListener('close', this._handleClose);\n    this._ws.removeEventListener('message', this._handleMessage);\n    this._ws.removeEventListener('error', this._handleError);\n    this._state = RTMConnectionState.CLOSED;\n    this.emit(RTMConnectionEvents.CLOSE);\n\n    this._ws.close(1000, reason);\n  }\n}\n"]} |
Sorry, the diff of this file is not supported yet
84790
5.17%820
8.32%15
-6.25%