UNPKG

@replit/novnc

Version:
421 lines (337 loc) 14.2 kB
"use strict"; function _typeof(obj) { "@babel/helpers - typeof"; if (typeof Symbol === "function" && typeof Symbol.iterator === "symbol") { _typeof = function _typeof(obj) { return typeof obj; }; } else { _typeof = function _typeof(obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; }; } return _typeof(obj); } Object.defineProperty(exports, "__esModule", { value: true }); exports["default"] = void 0; var Log = _interopRequireWildcard(require("./util/logging.js")); function _getRequireWildcardCache() { if (typeof WeakMap !== "function") return null; var cache = new WeakMap(); _getRequireWildcardCache = function _getRequireWildcardCache() { return cache; }; return cache; } function _interopRequireWildcard(obj) { if (obj && obj.__esModule) { return obj; } if (obj === null || _typeof(obj) !== "object" && typeof obj !== "function") { return { "default": obj }; } var cache = _getRequireWildcardCache(); if (cache && cache.has(obj)) { return cache.get(obj); } var newObj = {}; var hasPropertyDescriptor = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var key in obj) { if (Object.prototype.hasOwnProperty.call(obj, key)) { var desc = hasPropertyDescriptor ? Object.getOwnPropertyDescriptor(obj, key) : null; if (desc && (desc.get || desc.set)) { Object.defineProperty(newObj, key, desc); } else { newObj[key] = obj[key]; } } } newObj["default"] = obj; if (cache) { cache.set(obj, newObj); } return newObj; } function _toConsumableArray(arr) { return _arrayWithoutHoles(arr) || _iterableToArray(arr) || _unsupportedIterableToArray(arr) || _nonIterableSpread(); } function _nonIterableSpread() { throw new TypeError("Invalid attempt to spread non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); } function _unsupportedIterableToArray(o, minLen) { if (!o) return; if (typeof o === "string") return _arrayLikeToArray(o, minLen); var n = Object.prototype.toString.call(o).slice(8, -1); if (n === "Object" && o.constructor) n = o.constructor.name; if (n === "Map" || n === "Set") return Array.from(o); if (n === "Arguments" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return _arrayLikeToArray(o, minLen); } function _iterableToArray(iter) { if (typeof Symbol !== "undefined" && Symbol.iterator in Object(iter)) return Array.from(iter); } function _arrayWithoutHoles(arr) { if (Array.isArray(arr)) return _arrayLikeToArray(arr); } function _arrayLikeToArray(arr, len) { if (len == null || len > arr.length) len = arr.length; for (var i = 0, arr2 = new Array(len); i < len; i++) { arr2[i] = arr[i]; } return arr2; } function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } } function _defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } function _createClass(Constructor, protoProps, staticProps) { if (protoProps) _defineProperties(Constructor.prototype, protoProps); if (staticProps) _defineProperties(Constructor, staticProps); return Constructor; } // this has performance issues in some versions Chromium, and // doesn't gain a tremendous amount of performance increase in Firefox // at the moment. It may be valuable to turn it on in the future. var MAX_RQ_GROW_SIZE = 40 * 1024 * 1024; // 40 MiB // Constants pulled from RTCDataChannelState enum // https://developer.mozilla.org/en-US/docs/Web/API/RTCDataChannel/readyState#RTCDataChannelState_enum var DataChannel = { CONNECTING: "connecting", OPEN: "open", CLOSING: "closing", CLOSED: "closed" }; var ReadyStates = { CONNECTING: [WebSocket.CONNECTING, DataChannel.CONNECTING], OPEN: [WebSocket.OPEN, DataChannel.OPEN], CLOSING: [WebSocket.CLOSING, DataChannel.CLOSING], CLOSED: [WebSocket.CLOSED, DataChannel.CLOSED] }; // Properties a raw channel must have, WebSocket and RTCDataChannel are two examples var rawChannelProps = ["send", "close", "binaryType", "onerror", "onmessage", "onopen", "protocol", "readyState"]; var Websock = /*#__PURE__*/function () { function Websock() { _classCallCheck(this, Websock); this._websocket = null; // WebSocket or RTCDataChannel object this._rQi = 0; // Receive queue index this._rQlen = 0; // Next write position in the receive queue this._rQbufferSize = 1024 * 1024 * 4; // Receive queue buffer size (4 MiB) // called in init: this._rQ = new Uint8Array(this._rQbufferSize); this._rQ = null; // Receive queue this._sQbufferSize = 1024 * 10; // 10 KiB // called in init: this._sQ = new Uint8Array(this._sQbufferSize); this._sQlen = 0; this._sQ = null; // Send queue this._eventHandlers = { message: function message() {}, open: function open() {}, close: function close() {}, error: function error() {} }; } // Getters and Setters _createClass(Websock, [{ key: "sQ", get: function get() { return this._sQ; } }, { key: "rQ", get: function get() { return this._rQ; } }, { key: "rQi", get: function get() { return this._rQi; }, set: function set(val) { this._rQi = val; } // Receive Queue }, { key: "rQlen", get: function get() { return this._rQlen - this._rQi; } }, { key: "rQpeek8", value: function rQpeek8() { return this._rQ[this._rQi]; } }, { key: "rQskipBytes", value: function rQskipBytes(bytes) { this._rQi += bytes; } }, { key: "rQshift8", value: function rQshift8() { return this._rQshift(1); } }, { key: "rQshift16", value: function rQshift16() { return this._rQshift(2); } }, { key: "rQshift32", value: function rQshift32() { return this._rQshift(4); } // TODO(directxman12): test performance with these vs a DataView }, { key: "_rQshift", value: function _rQshift(bytes) { var res = 0; for (var _byte = bytes - 1; _byte >= 0; _byte--) { res += this._rQ[this._rQi++] << _byte * 8; } return res; } }, { key: "rQshiftStr", value: function rQshiftStr(len) { if (typeof len === 'undefined') { len = this.rQlen; } var str = ""; // Handle large arrays in steps to avoid long strings on the stack for (var i = 0; i < len; i += 4096) { var part = this.rQshiftBytes(Math.min(4096, len - i)); str += String.fromCharCode.apply(null, part); } return str; } }, { key: "rQshiftBytes", value: function rQshiftBytes(len) { if (typeof len === 'undefined') { len = this.rQlen; } this._rQi += len; return new Uint8Array(this._rQ.buffer, this._rQi - len, len); } }, { key: "rQshiftTo", value: function rQshiftTo(target, len) { if (len === undefined) { len = this.rQlen; } // TODO: make this just use set with views when using a ArrayBuffer to store the rQ target.set(new Uint8Array(this._rQ.buffer, this._rQi, len)); this._rQi += len; } }, { key: "rQslice", value: function rQslice(start) { var end = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : this.rQlen; return new Uint8Array(this._rQ.buffer, this._rQi + start, end - start); } // Check to see if we must wait for 'num' bytes (default to FBU.bytes) // to be available in the receive queue. Return true if we need to // wait (and possibly print a debug message), otherwise false. }, { key: "rQwait", value: function rQwait(msg, num, goback) { if (this.rQlen < num) { if (goback) { if (this._rQi < goback) { throw new Error("rQwait cannot backup " + goback + " bytes"); } this._rQi -= goback; } return true; // true means need more data } return false; } // Send Queue }, { key: "flush", value: function flush() { if (this._sQlen > 0 && ReadyStates.OPEN.indexOf(this._websocket.readyState) >= 0) { this._websocket.send(this._encodeMessage()); this._sQlen = 0; } } }, { key: "send", value: function send(arr) { this._sQ.set(arr, this._sQlen); this._sQlen += arr.length; this.flush(); } }, { key: "sendString", value: function sendString(str) { this.send(str.split('').map(function (chr) { return chr.charCodeAt(0); })); } // Event Handlers }, { key: "off", value: function off(evt) { this._eventHandlers[evt] = function () {}; } }, { key: "on", value: function on(evt, handler) { this._eventHandlers[evt] = handler; } }, { key: "_allocateBuffers", value: function _allocateBuffers() { this._rQ = new Uint8Array(this._rQbufferSize); this._sQ = new Uint8Array(this._sQbufferSize); } }, { key: "init", value: function init() { this._allocateBuffers(); this._rQi = 0; this._websocket = null; } }, { key: "open", value: function open(uri, protocols) { this.attach(new WebSocket(uri, protocols)); } }, { key: "attach", value: function attach(rawChannel) { var _this = this; this.init(); // Must get object and class methods to be compatible with the tests. var channelProps = [].concat(_toConsumableArray(Object.keys(rawChannel)), _toConsumableArray(Object.getOwnPropertyNames(Object.getPrototypeOf(rawChannel)))); for (var i = 0; i < rawChannelProps.length; i++) { var prop = rawChannelProps[i]; if (channelProps.indexOf(prop) < 0) { throw new Error('Raw channel missing property: ' + prop); } } this._websocket = rawChannel; this._websocket.binaryType = "arraybuffer"; this._websocket.onmessage = this._recvMessage.bind(this); var onOpen = function onOpen() { Log.Debug('>> WebSock.onopen'); if (_this._websocket.protocol) { Log.Info("Server choose sub-protocol: " + _this._websocket.protocol); } _this._eventHandlers.open(); Log.Debug("<< WebSock.onopen"); }; // If the readyState cannot be found this defaults to assuming it's not open. var isOpen = ReadyStates.OPEN.indexOf(this._websocket.readyState) >= 0; if (isOpen) { onOpen(); } else { this._websocket.onopen = onOpen; } this._websocket.onclose = function (e) { Log.Debug(">> WebSock.onclose"); _this._eventHandlers.close(e); Log.Debug("<< WebSock.onclose"); }; this._websocket.onerror = function (e) { Log.Debug(">> WebSock.onerror: " + e); _this._eventHandlers.error(e); Log.Debug("<< WebSock.onerror: " + e); }; } }, { key: "close", value: function close() { if (this._websocket) { if (ReadyStates.CONNECTING.indexOf(this._websocket.readyState) >= 0 || ReadyStates.OPEN.indexOf(this._websocket.readyState) >= 0) { Log.Info("Closing WebSocket connection"); this._websocket.close(); } this._websocket.onmessage = function () {}; } } // private methods }, { key: "_encodeMessage", value: function _encodeMessage() { // Put in a binary arraybuffer // according to the spec, you can send ArrayBufferViews with the send method return new Uint8Array(this._sQ.buffer, 0, this._sQlen); } // We want to move all the unread data to the start of the queue, // e.g. compacting. // The function also expands the receive que if needed, and for // performance reasons we combine these two actions to avoid // unneccessary copying. }, { key: "_expandCompactRQ", value: function _expandCompactRQ(minFit) { // if we're using less than 1/8th of the buffer even with the incoming bytes, compact in place // instead of resizing var requiredBufferSize = (this._rQlen - this._rQi + minFit) * 8; var resizeNeeded = this._rQbufferSize < requiredBufferSize; if (resizeNeeded) { // Make sure we always *at least* double the buffer size, and have at least space for 8x // the current amount of data this._rQbufferSize = Math.max(this._rQbufferSize * 2, requiredBufferSize); } // we don't want to grow unboundedly if (this._rQbufferSize > MAX_RQ_GROW_SIZE) { this._rQbufferSize = MAX_RQ_GROW_SIZE; if (this._rQbufferSize - this.rQlen < minFit) { throw new Error("Receive Queue buffer exceeded " + MAX_RQ_GROW_SIZE + " bytes, and the new message could not fit"); } } if (resizeNeeded) { var oldRQbuffer = this._rQ.buffer; this._rQ = new Uint8Array(this._rQbufferSize); this._rQ.set(new Uint8Array(oldRQbuffer, this._rQi, this._rQlen - this._rQi)); } else { this._rQ.copyWithin(0, this._rQi, this._rQlen); } this._rQlen = this._rQlen - this._rQi; this._rQi = 0; } // push arraybuffer values onto the end of the receive que }, { key: "_DecodeMessage", value: function _DecodeMessage(data) { var u8 = new Uint8Array(data); if (u8.length > this._rQbufferSize - this._rQlen) { this._expandCompactRQ(u8.length); } this._rQ.set(u8, this._rQlen); this._rQlen += u8.length; } }, { key: "_recvMessage", value: function _recvMessage(e) { this._DecodeMessage(e.data); if (this.rQlen > 0) { this._eventHandlers.message(); if (this._rQlen == this._rQi) { // All data has now been processed, this means we // can reset the receive queue. this._rQlen = 0; this._rQi = 0; } } else { Log.Debug("Ignoring empty message"); } } }]); return Websock; }(); exports["default"] = Websock;