UNPKG

@twilio/voice-sdk

Version:
511 lines (506 loc) 39.9 kB
'use strict'; Object.defineProperty(exports, '__esModule', { value: true }); var tslib = require('tslib'); var events = require('events'); var backoff = require('./backoff.js'); require('./errors/index.js'); var log = require('./log.js'); var generated = require('./errors/generated.js'); var WebSocket = globalThis.WebSocket; var CONNECT_SUCCESS_TIMEOUT = 10000; var CONNECT_TIMEOUT = 5000; var HEARTBEAT_TIMEOUT = 15000; var MAX_PREFERRED_DURATION = 15000; var MAX_PRIMARY_DURATION = Infinity; var MAX_RETRY_AFTER_DURATION = 75000; var MAX_PREFERRED_DELAY = 1000; var MAX_PRIMARY_DELAY = 20000; var MAX_RETRY_AFTER_DELAY = 60000; /** * All possible states of WSTransport. */ exports.WSTransportState = void 0; (function (WSTransportState) { /** * The WebSocket is not open but is trying to connect. */ WSTransportState["Connecting"] = "connecting"; /** * The WebSocket is not open and is not trying to connect. */ WSTransportState["Closed"] = "closed"; /** * The underlying WebSocket is open and active. */ WSTransportState["Open"] = "open"; })(exports.WSTransportState || (exports.WSTransportState = {})); /** * WebSocket Transport */ var WSTransport = /** @class */ (function (_super) { tslib.__extends(WSTransport, _super); /** * @constructor * @param uris - List of URI of the endpoints to connect to. * @param [options] - Constructor options. */ function WSTransport(uris, options) { if (options === void 0) { options = {}; } var _this = _super.call(this) || this; /** * The current state of the WSTransport. */ _this.state = exports.WSTransportState.Closed; /** * Start timestamp values for backoffs. */ _this._backoffStartTime = { preferred: null, primary: null, }; /** * The URI that the transport is connecting or connected to. The value of this * property is `null` if a connection attempt has not been made yet. */ _this._connectedUri = null; /** * An instance of Logger to use. */ _this._log = new log.default('WSTransport'); /** * The retryAfter value from signaling error, in seconds. */ _this._retryAfter = null; /** * Whether we should attempt to fallback if we receive an applicable error * when trying to connect to a signaling endpoint. */ _this._shouldFallback = false; /** * The current uri index that the transport is connected to. */ _this._uriIndex = 0; /** * Move the uri index to the next index * If the index is at the end, the index goes back to the first one. */ _this._moveUriIndex = function () { _this._uriIndex++; if (_this._uriIndex >= _this._uris.length) { _this._uriIndex = 0; } }; /** * Called in response to WebSocket#close event. */ _this._onSocketClose = function (event) { _this._log.error("Received websocket close event code: ".concat(event.code, ". Reason: ").concat(event.reason)); // 1006: Abnormal close. When the server is unreacheable // 1015: TLS Handshake error if (event.code === 1006 || event.code === 1015) { _this.emit('error', { code: 31005, message: event.reason || 'Websocket connection to Twilio\'s signaling servers were ' + 'unexpectedly ended. If this is happening consistently, there may ' + 'be an issue resolving the hostname provided. If a region or an ' + 'edge is being specified in Device setup, ensure it is valid.', twilioError: new generated.SignalingErrors.ConnectionError(), }); var wasConnected = ( // Only in Safari and certain Firefox versions, on network interruption, websocket drops right away with 1006 // Let's check current state if it's open, meaning we should not fallback // because we're coming from a previously connected session _this.state === exports.WSTransportState.Open || // But on other browsers, websocket doesn't drop // but our heartbeat catches it, setting the internal state to "Connecting". // With this, we should check the previous state instead. _this._previousState === exports.WSTransportState.Open); // Only fallback if this is not the first error // and if we were not connected previously if (_this._shouldFallback || !wasConnected) { _this._moveUriIndex(); } _this._shouldFallback = true; } _this._closeSocket(); }; /** * Called in response to WebSocket#error event. */ _this._onSocketError = function (err) { _this._log.error("WebSocket received error: ".concat(err.message)); _this.emit('error', { code: 31000, message: err.message || 'WSTransport socket error', twilioError: new generated.SignalingErrors.ConnectionDisconnected(), }); }; /** * Called in response to WebSocket#message event. */ _this._onSocketMessage = function (message) { // Clear heartbeat timeout on any incoming message, as they // all indicate an active connection. _this._setHeartbeatTimeout(); // Filter and respond to heartbeats if (_this._socket && message.data === '\n') { _this._socket.send('\n'); _this._log.debug('heartbeat'); return; } if (message && typeof message.data === 'string') { _this._log.debug("Received: ".concat(message.data)); var _a = JSON.parse(message.data), type = _a.type, _b = _a.payload, payload = _b === void 0 ? {} : _b; if (type === 'error' && payload.error && payload.error.retryAfter) { _this._retryAfter = payload.error.retryAfter * 1000; // convert to milliseconds } } _this.emit('message', message); }; /** * Called in response to WebSocket#open event. */ _this._onSocketOpen = function () { _this._log.info('WebSocket opened successfully.'); _this._timeOpened = Date.now(); _this._shouldFallback = false; _this._setState(exports.WSTransportState.Open); clearTimeout(_this._connectTimeout); if (_this._backoff) { _this._resetBackoffs(); } _this._setHeartbeatTimeout(); _this.emit('open'); }; _this._options = tslib.__assign(tslib.__assign({}, WSTransport.defaultConstructorOptions), options); _this._uris = uris; _this._backoff = null; return _this; } /** * Close the WebSocket, and don't try to reconnect. */ WSTransport.prototype.close = function () { this._log.info('WSTransport.close() called...'); this._close(); }; /** * Attempt to open a WebSocket connection. */ WSTransport.prototype.open = function () { this._log.info('WSTransport.open() called...'); if (this._socket && (this._socket.readyState === WebSocket.CONNECTING || this._socket.readyState === WebSocket.OPEN)) { this._log.info('WebSocket already open.'); return; } if (this._preferredUri) { this._connect(this._preferredUri); } else { this._connect(this._uris[this._uriIndex]); } }; /** * Send a message through the WebSocket connection. * @param message - A message to send to the endpoint. * @returns Whether the message was sent. */ WSTransport.prototype.send = function (message) { this._log.debug("Sending: ".concat(message)); // We can't send the message if the WebSocket isn't open if (!this._socket || this._socket.readyState !== WebSocket.OPEN) { this._log.debug('Cannot send message. WebSocket is not open.'); return false; } try { this._socket.send(message); } catch (e) { // Some unknown error occurred. Reset the socket to get a fresh session. this._log.error('Error while sending message:', e.message); this._closeSocket(); return false; } return true; }; /** * Update the preferred URI to connect to. Useful for Call signaling * reconnection, which requires connecting on the same edge. If `null` is * passed, the preferred URI is unset and the original `uris` array and * `uriIndex` is used to determine the signaling URI to connect to. * @param uri */ WSTransport.prototype.updatePreferredURI = function (uri) { this._preferredUri = uri; }; /** * Update acceptable URIs to reconnect to. Resets the URI index to 0. */ WSTransport.prototype.updateURIs = function (uris) { if (typeof uris === 'string') { uris = [uris]; } this._uris = uris; this._uriIndex = 0; }; /** * Close the WebSocket, and don't try to reconnect. */ WSTransport.prototype._close = function () { this._setState(exports.WSTransportState.Closed); this._closeSocket(); }; /** * Close the WebSocket and remove all event listeners. */ WSTransport.prototype._closeSocket = function () { clearTimeout(this._connectTimeout); clearTimeout(this._heartbeatTimeout); this._log.info('Closing and cleaning up WebSocket...'); if (!this._socket) { this._log.info('No WebSocket to clean up.'); return; } this._socket.removeEventListener('close', this._onSocketClose); this._socket.removeEventListener('error', this._onSocketError); this._socket.removeEventListener('message', this._onSocketMessage); this._socket.removeEventListener('open', this._onSocketOpen); if (this._socket.readyState === WebSocket.CONNECTING || this._socket.readyState === WebSocket.OPEN) { this._socket.close(); } // Reset backoff counter if connection was open for long enough to be considered successful if (this._backoff && this._timeOpened && ((Date.now() - this._timeOpened) > CONNECT_SUCCESS_TIMEOUT)) { this._resetBackoffs(); } if (this.state !== exports.WSTransportState.Closed) { if (!this._backoff) { this._backoff = this._setupBackoffs(); } this._performBackoff(); } delete this._socket; this.emit('close'); }; /** * Attempt to connect to the endpoint via WebSocket. * @param [uri] - URI string to connect to. * @param [retryCount] - Retry number, if this is a retry. Undefined if * first attempt, 1+ if a retry. */ WSTransport.prototype._connect = function (uri, retryCount) { var _this = this; this._log.info(typeof retryCount === 'number' ? "Attempting to reconnect (retry #".concat(retryCount, ")...") : 'Attempting to connect...'); this._closeSocket(); this._setState(exports.WSTransportState.Connecting); this._connectedUri = uri; try { this._socket = new this._options.WebSocket(this._connectedUri); } catch (e) { this._log.error('Could not connect to endpoint:', e.message); this._close(); this.emit('error', { code: 31000, message: e.message || "Could not connect to ".concat(this._connectedUri), twilioError: new generated.SignalingErrors.ConnectionDisconnected(), }); return; } this._socket.addEventListener('close', this._onSocketClose); this._socket.addEventListener('error', this._onSocketError); this._socket.addEventListener('message', this._onSocketMessage); this._socket.addEventListener('open', this._onSocketOpen); delete this._timeOpened; this._connectTimeout = setTimeout(function () { _this._log.info('WebSocket connection attempt timed out.'); _this._moveUriIndex(); _this._closeSocket(); }, this._options.connectTimeoutMs); }; /** * Perform a backoff. If a preferred URI is set (not null), then backoff * using the preferred mechanism. Otherwise, use the primary mechanism. */ WSTransport.prototype._performBackoff = function () { if (!this._backoff) { this._log.info('No backoff instance to perform backoff.'); return; } if (this._preferredUri) { this._log.info('Preferred URI set; backing off.'); this._backoff.preferred.backoff(); } else { this._log.info('Preferred URI not set; backing off.'); this._backoff.primary.backoff(); } }; /** * Reset both primary and preferred backoff mechanisms. */ WSTransport.prototype._resetBackoffs = function () { if (!this._backoff) { this._log.info('No backoff instance to reset.'); return; } this._backoff.preferred.removeAllListeners('backoff'); this._backoff.preferred.removeAllListeners('ready'); this._backoff.primary.removeAllListeners('backoff'); this._backoff.primary.removeAllListeners('ready'); this._backoff = null; this._retryAfter = null; this._backoffStartTime.preferred = null; this._backoffStartTime.primary = null; }; /** * Set a timeout to reconnect after HEARTBEAT_TIMEOUT milliseconds * have passed without receiving a message over the WebSocket. */ WSTransport.prototype._setHeartbeatTimeout = function () { var _this = this; clearTimeout(this._heartbeatTimeout); this._heartbeatTimeout = setTimeout(function () { _this._log.info("No messages received in ".concat(HEARTBEAT_TIMEOUT / 1000, " seconds. Reconnecting...")); _this._shouldFallback = true; _this._closeSocket(); }, HEARTBEAT_TIMEOUT); }; /** * Set the current and previous state */ WSTransport.prototype._setState = function (state) { this._previousState = this.state; this.state = state; }; /** * Set up the primary and preferred backoff mechanisms. */ WSTransport.prototype._setupBackoffs = function () { var _this = this; var preferredRetryAfter = this._retryAfter !== null && this._preferredUri ? this._retryAfter : null; if (preferredRetryAfter) { this._log.info("Setting initial preferred backoff value to retryAfter: ".concat(preferredRetryAfter, "ms")); } var maxPreferredDurationMs = preferredRetryAfter ? this._options.maxRetryAfterDurationMs : this._options.maxPreferredDurationMs; var preferredBackoffConfig = { factor: 2.0, jitter: 0.40, max: preferredRetryAfter ? this._options.maxRetryAfterDelayMs : this._options.maxPreferredDelayMs, min: preferredRetryAfter || 100, useInitialValue: Boolean(preferredRetryAfter), }; this._log.info('Initializing preferred transport backoff using config: ', preferredBackoffConfig); var preferredBackoff = new backoff.default(preferredBackoffConfig); preferredBackoff.on('backoff', function (attempt, delay) { if (_this.state === exports.WSTransportState.Closed) { _this._log.info('Preferred backoff initiated but transport state is closed; not attempting a connection.'); return; } _this._log.info("Will attempt to reconnect Websocket to preferred URI in ".concat(delay, "ms")); if (attempt === 0) { _this._backoffStartTime.preferred = Date.now(); _this._log.info("Preferred backoff start; ".concat(_this._backoffStartTime.preferred)); } }); preferredBackoff.on('ready', function (attempt, _delay) { if (_this.state === exports.WSTransportState.Closed) { _this._log.info('Preferred backoff ready but transport state is closed; not attempting a connection.'); return; } if (_this._backoffStartTime.preferred === null) { _this._log.info('Preferred backoff start time invalid; not attempting a connection.'); return; } if (!_this._backoff) { _this._log.info('Preferred backoff instance invalid; not attempting a connection.'); return; } if (Date.now() - _this._backoffStartTime.preferred > maxPreferredDurationMs) { _this._log.info('Max preferred backoff attempt time exceeded; falling back to primary backoff.'); _this._preferredUri = null; _this._backoff.primary.backoff(); return; } if (typeof _this._preferredUri !== 'string') { _this._log.info('Preferred URI cleared; falling back to primary backoff.'); _this._preferredUri = null; _this._backoff.primary.backoff(); return; } _this._connect(_this._preferredUri, attempt + 1); }); var primaryBackoffConfig = { factor: 2.0, jitter: 0.40, max: this._options.maxPrimaryDelayMs, // We only want a random initial delay if there are any fallback edges // Initial delay between 1s and 5s both inclusive min: this._uris && this._uris.length > 1 ? Math.floor(Math.random() * (5000 - 1000 + 1)) + 1000 : 100, }; this._log.info('Initializing primary transport backoff using config: ', primaryBackoffConfig); var primaryBackoff = new backoff.default(primaryBackoffConfig); primaryBackoff.on('backoff', function (attempt, delay) { if (_this.state === exports.WSTransportState.Closed) { _this._log.info('Primary backoff initiated but transport state is closed; not attempting a connection.'); return; } _this._log.info("Will attempt to reconnect WebSocket in ".concat(delay, "ms")); if (attempt === 0) { _this._backoffStartTime.primary = Date.now(); _this._log.info("Primary backoff start; ".concat(_this._backoffStartTime.primary)); } }); primaryBackoff.on('ready', function (attempt, _delay) { if (_this.state === exports.WSTransportState.Closed) { _this._log.info('Primary backoff ready but transport state is closed; not attempting a connection.'); return; } if (_this._backoffStartTime.primary === null) { _this._log.info('Primary backoff start time invalid; not attempting a connection.'); return; } if (Date.now() - _this._backoffStartTime.primary > _this._options.maxPrimaryDurationMs) { _this._log.info('Max primary backoff attempt time exceeded; not attempting a connection.'); return; } _this._connect(_this._uris[_this._uriIndex], attempt + 1); }); return { preferred: preferredBackoff, primary: primaryBackoff, }; }; Object.defineProperty(WSTransport.prototype, "uri", { /** * The uri the transport is currently connected to */ get: function () { return this._connectedUri; }, enumerable: false, configurable: true }); WSTransport.defaultConstructorOptions = { WebSocket: WebSocket, connectTimeoutMs: CONNECT_TIMEOUT, maxPreferredDelayMs: MAX_PREFERRED_DELAY, maxPreferredDurationMs: MAX_PREFERRED_DURATION, maxPrimaryDelayMs: MAX_PRIMARY_DELAY, maxPrimaryDurationMs: MAX_PRIMARY_DURATION, maxRetryAfterDelayMs: MAX_RETRY_AFTER_DELAY, maxRetryAfterDurationMs: MAX_RETRY_AFTER_DURATION, }; return WSTransport; }(events.EventEmitter)); exports.default = WSTransport; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"wstransport.js","sources":["../../lib/twilio/wstransport.ts"],"sourcesContent":[null],"names":["WSTransportState","__extends","Log","SignalingErrors","__assign","Backoff","EventEmitter"],"mappings":"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"}