data-transport
Version:
A simple and responsive transport
191 lines • 26.6 kB
JavaScript
import { __assign, __awaiter, __extends, __generator, __rest } from "tslib";
import { callbackKey } from '../constant';
import { global } from '../global';
import { Transport } from '../transport';
var transportName = '__DATA_TRANSPORT_BROWSER_EXTENSIONS__';
var BrowserExtensionsGenericTransport = /** @class */ (function (_super) {
__extends(BrowserExtensionsGenericTransport, _super);
function BrowserExtensionsGenericTransport(_options) {
if (_options === void 0) { _options = {}; }
var _a;
var _b = _options.browser, browser = _b === void 0 ? (_a = global.browser) !== null && _a !== void 0 ? _a : global.chrome : _b, _c = _options.listener, listener = _c === void 0 ? function (callback) {
this[callbackKey] = callback;
var handler = function (data, sender, sendResponse) {
data._sendResponse = sendResponse;
callback(data);
// https://developer.chrome.com/docs/extensions/develop/concepts/messaging#simple
// support async callback
return true;
};
browser.runtime.onMessage.addListener(handler);
return function () {
browser.runtime.onMessage.removeListener(handler);
};
} : _c, _d = _options.sender, sender = _d === void 0 ? function (message) {
if (message._sendResponse) {
var sendResponse = message._sendResponse;
delete message._sendResponse;
sendResponse(message);
}
else {
// @ts-ignore
browser.runtime.sendMessage(message, {}, this[callbackKey]);
}
} : _d, options = __rest(_options, ["browser", "listener", "sender"]);
return _super.call(this, __assign(__assign({}, options), { listener: listener, sender: sender })) || this;
}
return BrowserExtensionsGenericTransport;
}(Transport));
export { BrowserExtensionsGenericTransport };
var connectEventName = 'sharedworker-connect';
var BrowserExtensionsMainTransport = /** @class */ (function (_super) {
__extends(BrowserExtensionsMainTransport, _super);
function BrowserExtensionsMainTransport(_options) {
if (_options === void 0) { _options = {}; }
var _this = this;
var _a;
var _b = _options.browser, browser = _b === void 0 ? (_a = global.browser) !== null && _a !== void 0 ? _a : global.chrome : _b, _c = _options.listener, listener = _c === void 0 ? function (callback) {
var _this = this;
this[callbackKey] = callback;
return function () {
_this.ports.forEach(function (port) {
port.disconnect();
});
};
} : _c, _d = _options.sender, sender = _d === void 0 ? function (message) {
var _a;
var port = (_a = message._extra) === null || _a === void 0 ? void 0 : _a._port;
if (port) {
delete message._port;
port.postMessage(message);
}
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);
}
else {
this.ports.forEach(function (port) {
try {
port.postMessage(message);
}
catch (error) {
console.error(error);
}
});
}
} : _d, options = __rest(_options, ["browser", "listener", "sender"]);
_this = _super.call(this, __assign(__assign({}, options), { listener: listener, sender: sender })) || this;
_this.ports = new Map();
_this._onConnectCallback = new Set();
_this._onDisconnectCallback = new Set();
browser.runtime.onConnect.addListener(function (port) { return __awaiter(_this, void 0, void 0, function () {
var handler_1, id_1;
var _this = this;
return __generator(this, function (_a) {
switch (_a.label) {
case 0:
if (!(port.name === transportName)) return [3 /*break*/, 2];
handler_1 = function (data) {
var _a;
data._extra = (_a = data._extra) !== null && _a !== void 0 ? _a : {};
data._extra._port = port;
_this[callbackKey](data);
};
port.onMessage.addListener(handler_1);
port.onDisconnect.addListener(function () {
port.onMessage.removeListener(handler_1);
_this.ports.forEach(function (_port, id) {
if (_port === port) {
_this.ports.delete(id);
}
});
_this._onDisconnectCallback.forEach(function (callback) {
callback(id_1);
});
});
return [4 /*yield*/, this.emit({
// @ts-ignore
name: connectEventName,
_extra: { _port: port },
})];
case 1:
id_1 = _a.sent();
this.ports.set(id_1, port);
this._onConnectCallback.forEach(function (callback) {
callback(id_1);
});
_a.label = 2;
case 2: return [2 /*return*/];
}
});
}); });
return _this;
}
BrowserExtensionsMainTransport.prototype.onConnect = function (callback) {
var _this = this;
this._onConnectCallback.add(callback);
return function () {
_this._onConnectCallback.delete(callback);
};
};
BrowserExtensionsMainTransport.prototype.onDisconnect = function (callback) {
var _this = this;
this._onDisconnectCallback.add(callback);
return function () {
_this._onDisconnectCallback.delete(callback);
};
};
return BrowserExtensionsMainTransport;
}(Transport));
export { BrowserExtensionsMainTransport };
var BrowserExtensionsClientTransport = /** @class */ (function (_super) {
__extends(BrowserExtensionsClientTransport, _super);
function BrowserExtensionsClientTransport(_options) {
if (_options === void 0) { _options = {}; }
var _this = this;
var _a;
var _b = _options.browser, browser = _b === void 0 ? (_a = global.browser) !== null && _a !== void 0 ? _a : global.chrome : _b, _c = _options.port, port = _c === void 0 ? browser.runtime.connect({ name: transportName }) : _c, _d = _options.listener, listener = _d === void 0 ? function (callback) {
var handler = function (options) {
callback(options);
};
port.onMessage.addListener(handler);
return function () {
port.onMessage.removeListener(handler);
};
} : _d, _e = _options.sender, sender = _e === void 0 ? function (message) {
port.postMessage(message);
} : _e, options = __rest(_options, ["browser", "port", "listener", "sender"]);
_this = _super.call(this, __assign(__assign({}, options), { listener: listener, sender: sender })) || this;
_this._onConnectCallback = new Set();
// @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._onConnectCallback.forEach(function (callback) {
callback();
});
});
return [2 /*return*/, this.id];
});
}); });
return _this;
}
BrowserExtensionsClientTransport.prototype.onConnect = function (callback) {
var _this = this;
this._onConnectCallback.add(callback);
return function () {
_this._onConnectCallback.delete(callback);
};
};
return BrowserExtensionsClientTransport;
}(Transport));
export { BrowserExtensionsClientTransport };
export var BrowserExtensionsTransport = {
Main: BrowserExtensionsMainTransport,
Client: BrowserExtensionsClientTransport,
};
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"browserExtensionsTransport.js","sourceRoot":"","sources":["../../src/transports/browserExtensionsTransport.ts"],"names":[],"mappings":"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","sourcesContent":["import { callbackKey } from '../constant';\nimport { global } from '../global';\nimport type {\n  BaseInteraction,\n  IRequest,\n  ListenerOptions,\n  SendOptions,\n  TransportOptions,\n} from '../interface';\nimport { Transport } from '../transport';\n\nconst transportName = '__DATA_TRANSPORT_BROWSER_EXTENSIONS__';\n\ntype Browser = typeof global.browser | typeof global.chrome;\n\ntype Port = browser.runtime.Port | chrome.runtime.Port;\n\ninterface SendResponse {\n  _sendResponse?: (response?: SendOptions<SendResponse>) => void;\n}\n\nexport interface BrowserExtensionsGenericTransportOptions\n  extends Partial<TransportOptions<SendResponse>> {\n  /**\n   * browser extension API.\n   */\n  browser?: Browser;\n}\n\nexport interface BrowserExtensionsMainTransportOptions\n  extends Partial<TransportOptions<BrowserExtensionsMainPort>> {\n  /**\n   * browser extension API.\n   */\n  browser?: Browser;\n}\n\nexport interface BrowserExtensionsClientTransportOptions\n  extends Partial<TransportOptions> {\n  /**\n   * browser extension API.\n   */\n  browser?: Browser;\n  /**\n   * browser extension client port.\n   */\n  port?: Port;\n}\n\ninterface BrowserExtensionsMainPort {\n  _port?: Port;\n}\n\ntype ClientCallback = () => void | Promise<void>;\ntype MainCallback = (clientId: string) => void | Promise<void>;\n\nexport abstract class BrowserExtensionsGenericTransport<\n  T extends BaseInteraction = any\n> extends Transport<T> {\n  private [callbackKey]!: (options: ListenerOptions<SendResponse>) => void;\n\n  constructor(_options: BrowserExtensionsGenericTransportOptions = {}) {\n    const {\n      browser = global.browser ?? global.chrome,\n      listener = function (this: BrowserExtensionsGenericTransport, callback) {\n        this[callbackKey] = callback;\n        const handler = (\n          data: ListenerOptions<SendResponse>,\n          sender: browser.runtime.MessageSender | chrome.runtime.MessageSender,\n          sendResponse: (response?: SendOptions<SendResponse>) => void\n        ) => {\n          data._sendResponse = sendResponse;\n          callback(data);\n          // https://developer.chrome.com/docs/extensions/develop/concepts/messaging#simple\n          // support async callback\n          return true;\n        };\n        browser.runtime.onMessage.addListener(handler);\n        return () => {\n          browser.runtime.onMessage.removeListener(handler);\n        };\n      },\n      sender = function (this: BrowserExtensionsGenericTransport, message) {\n        if (message._sendResponse) {\n          const sendResponse = message._sendResponse;\n          delete message._sendResponse;\n          sendResponse(message);\n        } else {\n          // @ts-ignore\n          browser.runtime.sendMessage(\n            message,\n            {},\n            this[callbackKey] as (response: IRequest<SendResponse>) => void\n          );\n        }\n      },\n      ...options\n    } = _options;\n    super({\n      ...options,\n      listener,\n      sender,\n    });\n  }\n}\n\nconst connectEventName = 'sharedworker-connect';\n\nexport abstract class BrowserExtensionsMainTransport<\n  T extends BaseInteraction = any\n> extends Transport<T> {\n  protected ports = new Map<string, Port>();\n\n  private [callbackKey]!: (\n    options: ListenerOptions<BrowserExtensionsMainPort>\n  ) => void;\n\n  constructor(_options: BrowserExtensionsMainTransportOptions = {}) {\n    const {\n      browser = global.browser ?? global.chrome,\n      listener = function (this: BrowserExtensionsMainTransport, callback) {\n        this[callbackKey] = callback;\n        return () => {\n          this.ports.forEach((port) => {\n            port.disconnect();\n          });\n        };\n      },\n      sender = function (this: BrowserExtensionsMainTransport, message) {\n        const port = message._extra?._port;\n        if (port) {\n          delete message._port;\n          port.postMessage(message);\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);\n        } else {\n          this.ports.forEach((port) => {\n            try {\n              port.postMessage(message);\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    browser.runtime.onConnect.addListener(async (port: Port) => {\n      if (port.name === transportName) {\n        const handler = (data: any) => {\n          data._extra = data._extra ?? {};\n          data._extra._port = port;\n          this[callbackKey](data as ListenerOptions<BrowserExtensionsMainPort>);\n        };\n        port.onMessage.addListener(handler);\n        port.onDisconnect.addListener(() => {\n          port.onMessage.removeListener(handler);\n          this.ports.forEach((_port, id) => {\n            if (_port === port) {\n              this.ports.delete(id);\n            }\n          });\n          this._onDisconnectCallback.forEach((callback) => {\n            callback(id);\n          });\n        });\n\n        // @ts-ignore\n        const id: string = await this.emit({\n          // @ts-ignore\n          name: connectEventName,\n          _extra: { _port: port },\n        });\n        this.ports.set(id, port);\n        this._onConnectCallback.forEach((callback) => {\n          callback(id);\n        });\n      }\n    });\n  }\n\n  private _onConnectCallback = new Set<MainCallback>();\n\n  onConnect(callback: MainCallback) {\n    this._onConnectCallback.add(callback);\n    return () => {\n      this._onConnectCallback.delete(callback);\n    };\n  }\n\n  private _onDisconnectCallback = new Set<MainCallback>();\n\n  onDisconnect(callback: MainCallback) {\n    this._onDisconnectCallback.add(callback);\n    return () => {\n      this._onDisconnectCallback.delete(callback);\n    };\n  }\n}\n\nexport abstract class BrowserExtensionsClientTransport<\n  T extends BaseInteraction = any\n> extends Transport<T> {\n  constructor(_options: BrowserExtensionsClientTransportOptions = {}) {\n    const {\n      browser = global.browser ?? global.chrome,\n      port = browser.runtime.connect({ name: transportName }),\n      listener = (callback) => {\n        const handler = (options: object) => {\n          callback(options as ListenerOptions<{}>);\n        };\n        port.onMessage.addListener(handler);\n        return () => {\n          port.onMessage.removeListener(handler);\n        };\n      },\n      sender = (message) => {\n        port.postMessage(message);\n      },\n      ...options\n    } = _options;\n    super({\n      ...options,\n      listener,\n      sender,\n    });\n    // @ts-ignore\n    this.listen(connectEventName, async () => {\n      Promise.resolve().then(() => {\n        this._onConnectCallback.forEach((callback) => {\n          callback();\n        });\n      });\n      return this.id;\n    });\n  }\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\nexport const BrowserExtensionsTransport = {\n  Main: BrowserExtensionsMainTransport,\n  Client: BrowserExtensionsClientTransport,\n};\n"]}