UNPKG

data-transport

Version:
217 lines 28.4 kB
import { __assign, __awaiter, __extends, __generator, __rest } from "tslib"; import { callbackKey, prefixKey } from '../constant'; import { getAction, Transport } from '../transport'; var connectEventName = 'sharedworker-connect'; var disconnectEventName = 'sharedworker-disconnect'; var SharedWorkerClientTransport = /** @class */ (function (_super) { __extends(SharedWorkerClientTransport, _super); function SharedWorkerClientTransport(_options) { var _this = this; var worker = _options.worker, _a = _options.listener, listener = _a === void 0 ? function (callback) { var handler = function (_a) { var data = _a.data; callback(data); }; worker.port.addEventListener('message', handler); worker.port.start(); return function () { worker.port.removeEventListener('message', handler); }; } : _a, _b = _options.sender, sender = _b === void 0 ? function (message) { var _a; var transfer = (_a = message.transfer) !== null && _a !== void 0 ? _a : []; delete message.transfer; worker.port.postMessage(message, transfer); } : _b, options = __rest(_options, ["worker", "listener", "sender"]); _this = _super.call(this, __assign(__assign({}, options), { listener: listener, sender: sender })) || this; _this._connected = false; _this._onConnectCallback = new Set(); // do not use `unload` event // https://developer.chrome.com/docs/web-platform/deprecating-unload window.addEventListener('pagehide', function () { // @ts-ignore _this.emit({ name: disconnectEventName, respond: false }, _this.id); }); // @ts-ignore _this.listen(connectEventName, function () { return __awaiter(_this, void 0, void 0, function () { var _this = this; return __generator(this, function (_a) { Promise.resolve().then(function () { _this._handleConnect(); }); return [2 /*return*/, this.id]; }); }); }); // ensure the connect event is sent when the client connect to the worker // @ts-ignore _this.emit({ name: connectEventName, respond: false, silent: true }); return _this; } SharedWorkerClientTransport.prototype._handleConnect = function () { if (this._connected) return; this._connected = true; this._onConnectCallback.forEach(function (callback) { callback(); }); }; SharedWorkerClientTransport.prototype.onConnect = function (callback) { var _this = this; this._onConnectCallback.add(callback); return function () { _this._onConnectCallback.delete(callback); }; }; return SharedWorkerClientTransport; }(Transport)); export { SharedWorkerClientTransport }; var SharedWorkerInternalTransport = /** @class */ (function (_super) { __extends(SharedWorkerInternalTransport, _super); function SharedWorkerInternalTransport(_options) { if (_options === void 0) { _options = {}; } var _this = this; var _a = _options.listener, listener = _a === void 0 ? function (callback) { var _this = this; this[callbackKey] = callback; return function () { _this.ports.forEach(function (port) { port._handler && port.removeEventListener('message', port._handler); delete port._handler; }); self.close(); }; } : _a, _b = _options.sender, sender = _b === void 0 ? function (message) { var _a, _b; var transfer = (_a = message.transfer) !== null && _a !== void 0 ? _a : []; delete message.transfer; var port = (_b = message._extra) === null || _b === void 0 ? void 0 : _b._port; // pick a client port for sender. if (port) { delete message._extra._port; port.postMessage(message, transfer); } else if (message.type === 'response' && // @ts-ignore _this.ports.has(message.requestId)) { // @ts-ignore var port_1 = _this.ports.get(message.requestId); port_1.postMessage(message, transfer); } else { _this.ports.forEach(function (port) { try { port.postMessage(message, transfer); } catch (error) { console.error(error); } }); _this.tempPorts.forEach(function (port) { try { port.postMessage(message, transfer); } catch (error) { console.error(error); } }); } } : _b, options = __rest(_options, ["listener", "sender"]); _this = _super.call(this, __assign(__assign({}, options), { listener: listener, sender: sender })) || this; _this.ports = new Map(); _this.tempPorts = new Set(); _this._onConnectCallback = new Set(); _this._onDisconnectCallback = new Set(); var disconnectAction = getAction(_this[prefixKey], disconnectEventName); var connectEvent = getAction(_this[prefixKey], connectEventName); self.addEventListener('connect', function (e) { return __awaiter(_this, void 0, void 0, function () { var port, id, err_1; var _this = this; return __generator(this, function (_a) { switch (_a.label) { case 0: port = e.ports[0]; port._handler = function (_a) { var _b; var data = _a.data; if (data.hasRespond) { data._extra = (_b = data._extra) !== null && _b !== void 0 ? _b : {}; data._extra._port = port; } if (data.action === disconnectAction && _this.ports.has(data.requestId)) { // clear port and clientId when the port's client is disconnected. _this.ports.delete(data.requestId); _this._onDisconnectCallback.forEach(function (callback) { callback(data.requestId); }); } if (data.type === 'request' && data.action === connectEvent) { _this.emit({ // @ts-ignore name: connectEventName, _extra: { _port: port }, silent: true, respond: false, }); _this._handleConnect(data.requestId, port); } _this[callbackKey](data); }; port.addEventListener('message', port._handler); port.start(); this.tempPorts.add(port); _a.label = 1; case 1: _a.trys.push([1, 3, , 4]); return [4 /*yield*/, this.emit({ // @ts-ignore name: connectEventName, _extra: { _port: port }, silent: true, })]; case 2: id = _a.sent(); this._handleConnect(id, port); return [3 /*break*/, 4]; case 3: err_1 = _a.sent(); this.tempPorts.delete(port); console.error(err_1); return [3 /*break*/, 4]; case 4: return [2 /*return*/]; } }); }); }); return _this; } SharedWorkerInternalTransport.prototype.onConnect = function (callback) { var _this = this; this._onConnectCallback.add(callback); return function () { _this._onConnectCallback.delete(callback); }; }; SharedWorkerInternalTransport.prototype._handleConnect = function (id, port) { if (id && !this.ports.has(id)) { this.ports.set(id, port); this.tempPorts.delete(port); this._onConnectCallback.forEach(function (callback) { callback(id); }); } }; SharedWorkerInternalTransport.prototype.onDisconnect = function (callback) { var _this = this; this._onDisconnectCallback.add(callback); return function () { _this._onDisconnectCallback.delete(callback); }; }; return SharedWorkerInternalTransport; }(Transport)); export { SharedWorkerInternalTransport }; export var SharedWorkerTransport = { Client: SharedWorkerClientTransport, Worker: SharedWorkerInternalTransport, }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sharedWorkerTransport.js","sourceRoot":"","sources":["../../src/transports/sharedWorkerTransport.ts"],"names":[],"mappings":"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","sourcesContent":["import { callbackKey, prefixKey } from '../constant';\nimport { listen } from '../decorators';\nimport {\n  TransportOptions,\n  TransferableWorker,\n  ListenerOptions,\n  BaseInteraction,\n} from '../interface';\nimport { getAction, Transport } from '../transport';\n\ndeclare var self: SharedWorkerGlobalScope;\ninterface SharedWorkerPort extends TransferableWorker {\n  _port?: MessagePort;\n}\n\nexport interface SharedWorkerClientTransportOptions\n  extends Partial<TransportOptions<TransferableWorker>> {\n  /**\n   * Pass a shared worker instance for data transport.\n   */\n  worker: SharedWorker;\n}\n\ntype ClientCallback = () => void | Promise<void>;\ntype WorkerCallback = (clientId: string) => void | Promise<void>;\n\nconst connectEventName = 'sharedworker-connect';\nconst disconnectEventName = 'sharedworker-disconnect';\n\nexport interface SharedWorkerInternalTransportOptions\n  extends Partial<TransportOptions<SharedWorkerPort>> {}\n\nexport abstract class SharedWorkerClientTransport<\n  T extends BaseInteraction = any\n> extends Transport<T> {\n  constructor(_options: SharedWorkerClientTransportOptions) {\n    const {\n      worker,\n      listener = (callback) => {\n        const handler = ({\n          data,\n        }: MessageEvent<ListenerOptions<TransferableWorker>>) => {\n          callback(data);\n        };\n        worker.port.addEventListener('message', handler);\n        worker.port.start();\n        return () => {\n          worker.port.removeEventListener('message', handler);\n        };\n      },\n      sender = (message) => {\n        const transfer = message.transfer ?? [];\n        delete message.transfer;\n        worker.port.postMessage(message, transfer);\n      },\n      ...options\n    } = _options;\n    super({\n      ...options,\n      listener,\n      sender,\n    });\n    // do not use `unload` event\n    // https://developer.chrome.com/docs/web-platform/deprecating-unload\n    window.addEventListener('pagehide', () => {\n      // @ts-ignore\n      this.emit({ name: disconnectEventName, respond: false }, this.id);\n    });\n    // @ts-ignore\n    this.listen(connectEventName, async () => {\n      Promise.resolve().then(() => {\n        this._handleConnect();\n      });\n      return this.id;\n    });\n    // ensure the connect event is sent when the client connect to the worker\n    // @ts-ignore\n    this.emit({ name: connectEventName, respond: false, silent: true });\n  }\n\n  private _handleConnect() {\n    if (this._connected) return;\n    this._connected = true;\n    this._onConnectCallback.forEach((callback) => {\n      callback();\n    });\n  }\n\n  private _connected = false;\n\n  private _onConnectCallback = new Set<ClientCallback>();\n\n  onConnect(callback: ClientCallback) {\n    this._onConnectCallback.add(callback);\n    return () => {\n      this._onConnectCallback.delete(callback);\n    };\n  }\n}\n\ninterface SharedWorkerTransportPort extends MessagePort {\n  _handler?: (options: MessageEvent<ListenerOptions<SharedWorkerPort>>) => void;\n}\n\nexport abstract class SharedWorkerInternalTransport<\n  T extends BaseInteraction = any\n> extends Transport<T> {\n  protected ports = new Map<string, MessagePort>();\n  protected tempPorts = new Set<MessagePort>();\n  private [callbackKey]!: (options: ListenerOptions<SharedWorkerPort>) => void;\n\n  constructor(_options: SharedWorkerInternalTransportOptions = {}) {\n    const {\n      listener = function (this: SharedWorkerInternalTransport, callback) {\n        this[callbackKey] = callback;\n        return () => {\n          this.ports.forEach((port: SharedWorkerTransportPort) => {\n            port._handler && port.removeEventListener('message', port._handler);\n            delete port._handler;\n          });\n          self.close();\n        };\n      },\n      sender = (message) => {\n        const transfer = message.transfer ?? [];\n        delete message.transfer;\n        const port = message._extra?._port;\n        // pick a client port for sender.\n        if (port) {\n          delete message._extra!._port;\n          port.postMessage(message, transfer);\n        } else if (\n          message.type === 'response' &&\n          // @ts-ignore\n          this.ports.has(message.requestId)\n        ) {\n          // @ts-ignore\n          const port = this.ports.get(message.requestId)!;\n          port.postMessage(message, transfer);\n        } else {\n          this.ports.forEach((port) => {\n            try {\n              port.postMessage(message, transfer);\n            } catch (error) {\n              console.error(error);\n            }\n          });\n          this.tempPorts.forEach((port) => {\n            try {\n              port.postMessage(message, transfer);\n            } catch (error) {\n              console.error(error);\n            }\n          });\n        }\n      },\n      ...options\n    } = _options;\n    super({\n      ...options,\n      listener,\n      sender,\n    });\n\n    const disconnectAction = getAction(this[prefixKey]!, disconnectEventName);\n    const connectEvent = getAction(this[prefixKey]!, connectEventName);\n    self.addEventListener('connect', async (e) => {\n      const port: SharedWorkerTransportPort = e.ports[0];\n      port._handler = ({\n        data,\n      }: MessageEvent<ListenerOptions<SharedWorkerPort>>) => {\n        if (data.hasRespond) {\n          data._extra = data._extra ?? {};\n          data._extra._port = port;\n        }\n        if (\n          data.action === disconnectAction &&\n          this.ports.has(data.requestId)\n        ) {\n          // clear port and clientId when the port's client is disconnected.\n          this.ports.delete(data.requestId);\n          this._onDisconnectCallback.forEach((callback) => {\n            callback(data.requestId);\n          });\n        }\n        if (data.type === 'request' && data.action === connectEvent) {\n          this.emit({\n            // @ts-ignore\n            name: connectEventName,\n            _extra: { _port: port },\n            silent: true,\n            respond: false,\n          });\n          this._handleConnect(data.requestId, port);\n        }\n        this[callbackKey](data);\n      };\n      port.addEventListener('message', port._handler);\n      port.start();\n      this.tempPorts.add(port);\n      try {\n        // because parameters is unknown\n        // @ts-ignore\n        const id: string = await this.emit({\n          // @ts-ignore\n          name: connectEventName,\n          _extra: { _port: port },\n          silent: true,\n        });\n        this._handleConnect(id, port);\n      } catch (err) {\n        this.tempPorts.delete(port);\n        console.error(err);\n      }\n    });\n  }\n\n  private _onConnectCallback = new Set<WorkerCallback>();\n\n  onConnect(callback: WorkerCallback) {\n    this._onConnectCallback.add(callback);\n    return () => {\n      this._onConnectCallback.delete(callback);\n    };\n  }\n\n  private _handleConnect(id: string, port: MessagePort) {\n    if (id && !this.ports.has(id)) {\n      this.ports.set(id, port);\n      this.tempPorts.delete(port);\n      this._onConnectCallback.forEach((callback) => {\n        callback(id);\n      });\n    }\n  }\n\n  private _onDisconnectCallback = new Set<WorkerCallback>();\n\n  onDisconnect(callback: WorkerCallback) {\n    this._onDisconnectCallback.add(callback);\n    return () => {\n      this._onDisconnectCallback.delete(callback);\n    };\n  }\n}\n\nexport const SharedWorkerTransport = {\n  Client: SharedWorkerClientTransport,\n  Worker: SharedWorkerInternalTransport,\n};\n"]}