velocious 1.0.340 → 1.0.342
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/build/src/background-jobs/job-runner.d.ts.map +1 -1
- package/build/src/background-jobs/job-runner.js +4 -2
- package/build/src/background-jobs/worker.d.ts.map +1 -1
- package/build/src/background-jobs/worker.js +4 -2
- package/build/src/http-client/websocket-client.d.ts +21 -0
- package/build/src/http-client/websocket-client.d.ts.map +1 -1
- package/build/src/http-client/websocket-client.js +101 -15
- package/package.json +1 -1
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"job-runner.d.ts","sourceRoot":"","sources":["../../../src/background-jobs/job-runner.js"],"names":[],"mappings":"AAMA;;;GAGG;AACH,+CAHW,OAAO,YAAY,EAAE,oBAAoB,GACvC,OAAO,CAAC,IAAI,CAAC,
|
|
1
|
+
{"version":3,"file":"job-runner.d.ts","sourceRoot":"","sources":["../../../src/background-jobs/job-runner.js"],"names":[],"mappings":"AAMA;;;GAGG;AACH,+CAHW,OAAO,YAAY,EAAE,oBAAoB,GACvC,OAAO,CAAC,IAAI,CAAC,CA2CzB"}
|
|
@@ -18,7 +18,9 @@ export default async function runJobPayload(payload) {
|
|
|
18
18
|
/** @type {(...args: any[]) => Promise<void>} */
|
|
19
19
|
const perform = jobInstance.perform;
|
|
20
20
|
try {
|
|
21
|
-
await
|
|
21
|
+
await configuration.withConnections(async () => {
|
|
22
|
+
await perform.apply(jobInstance, payload.args || []);
|
|
23
|
+
});
|
|
22
24
|
if (payload.id) {
|
|
23
25
|
await reporter.reportWithRetry({
|
|
24
26
|
jobId: payload.id,
|
|
@@ -43,4 +45,4 @@ export default async function runJobPayload(payload) {
|
|
|
43
45
|
throw error;
|
|
44
46
|
}
|
|
45
47
|
}
|
|
46
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
48
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiam9iLXJ1bm5lci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9iYWNrZ3JvdW5kLWpvYnMvam9iLXJ1bm5lci5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxZQUFZO0FBRVosT0FBTyxxQkFBcUIsTUFBTSw4QkFBOEIsQ0FBQTtBQUNoRSxPQUFPLHFCQUFxQixNQUFNLG1CQUFtQixDQUFBO0FBQ3JELE9BQU8sNEJBQTRCLE1BQU0sc0JBQXNCLENBQUE7QUFFL0Q7OztHQUdHO0FBQ0gsTUFBTSxDQUFDLE9BQU8sQ0FBQyxLQUFLLFVBQVUsYUFBYSxDQUFDLE9BQU87SUFDakQsTUFBTSxhQUFhLEdBQUcsTUFBTSxxQkFBcUIsRUFBRSxDQUFBO0lBQ25ELGFBQWEsQ0FBQyxVQUFVLEVBQUUsQ0FBQTtJQUMxQixNQUFNLGFBQWEsQ0FBQyxVQUFVLENBQUMsRUFBQyxJQUFJLEVBQUUsd0JBQXdCLEVBQUMsQ0FBQyxDQUFBO0lBQ2hFLE1BQU0sUUFBUSxHQUFHLElBQUksNEJBQTRCLENBQUMsRUFBQyxhQUFhLEVBQUMsQ0FBQyxDQUFBO0lBRWxFLE1BQU0sUUFBUSxHQUFHLElBQUkscUJBQXFCLENBQUMsRUFBQyxhQUFhLEVBQUMsQ0FBQyxDQUFBO0lBQzNELE1BQU0sUUFBUSxDQUFDLElBQUksRUFBRSxDQUFBO0lBQ3JCLE1BQU0sUUFBUSxHQUFHLFFBQVEsQ0FBQyxZQUFZLENBQUMsT0FBTyxDQUFDLE9BQU8sQ0FBQyxDQUFBO0lBQ3ZELE1BQU0sV0FBVyxHQUFHLElBQUksUUFBUSxFQUFFLENBQUE7SUFDbEMsZ0RBQWdEO0lBQ2hELE1BQU0sT0FBTyxHQUFHLFdBQVcsQ0FBQyxPQUFPLENBQUE7SUFFbkMsSUFBSSxDQUFDO1FBQ0gsTUFBTSxhQUFhLENBQUMsZUFBZSxDQUFDLEtBQUssSUFBSSxFQUFFO1lBQzdDLE1BQU0sT0FBTyxDQUFDLEtBQUssQ0FBQyxXQUFXLEVBQUUsT0FBTyxDQUFDLElBQUksSUFBSSxFQUFFLENBQUMsQ0FBQTtRQUN0RCxDQUFDLENBQUMsQ0FBQTtRQUVGLElBQUksT0FBTyxDQUFDLEVBQUUsRUFBRSxDQUFDO1lBQ2YsTUFBTSxRQUFRLENBQUMsZUFBZSxDQUFDO2dCQUM3QixLQUFLLEVBQUUsT0FBTyxDQUFDLEVBQUU7Z0JBQ2pCLE1BQU0sRUFBRSxXQUFXO2dCQUNuQixRQUFRLEVBQUUsT0FBTyxDQUFDLFFBQVE7Z0JBQzFCLGFBQWEsRUFBRSxPQUFPLENBQUMsYUFBYTtnQkFDcEMsYUFBYSxFQUFFLEtBQUs7YUFDckIsQ0FBQyxDQUFBO1FBQ0osQ0FBQztJQUNILENBQUM7SUFBQyxPQUFPLEtBQUssRUFBRSxDQUFDO1FBQ2YsSUFBSSxPQUFPLENBQUMsRUFBRSxFQUFFLENBQUM7WUFDZixNQUFNLFFBQVEsQ0FBQyxlQUFlLENBQUM7Z0JBQzdCLEtBQUssRUFBRSxPQUFPLENBQUMsRUFBRTtnQkFDakIsTUFBTSxFQUFFLFFBQVE7Z0JBQ2hCLEtBQUs7Z0JBQ0wsUUFBUSxFQUFFLE9BQU8sQ0FBQyxRQUFRO2dCQUMxQixhQUFhLEVBQUUsT0FBTyxDQUFDLGFBQWE7Z0JBQ3BDLGFBQWEsRUFBRSxLQUFLO2FBQ3JCLENBQUMsQ0FBQTtRQUNKLENBQUM7UUFFRCxNQUFNLEtBQUssQ0FBQTtJQUNiLENBQUM7QUFDSCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLy8gQHRzLWNoZWNrXG5cbmltcG9ydCBjb25maWd1cmF0aW9uUmVzb2x2ZXIgZnJvbSBcIi4uL2NvbmZpZ3VyYXRpb24tcmVzb2x2ZXIuanNcIlxuaW1wb3J0IEJhY2tncm91bmRKb2JSZWdpc3RyeSBmcm9tIFwiLi9qb2ItcmVnaXN0cnkuanNcIlxuaW1wb3J0IEJhY2tncm91bmRKb2JzU3RhdHVzUmVwb3J0ZXIgZnJvbSBcIi4vc3RhdHVzLXJlcG9ydGVyLmpzXCJcblxuLyoqXG4gKiBAcGFyYW0ge2ltcG9ydChcIi4vdHlwZXMuanNcIikuQmFja2dyb3VuZEpvYlBheWxvYWR9IHBheWxvYWQgLSBQYXlsb2FkLlxuICogQHJldHVybnMge1Byb21pc2U8dm9pZD59IC0gUmVzb2x2ZXMgd2hlbiBjb21wbGV0ZS5cbiAqL1xuZXhwb3J0IGRlZmF1bHQgYXN5bmMgZnVuY3Rpb24gcnVuSm9iUGF5bG9hZChwYXlsb2FkKSB7XG4gIGNvbnN0IGNvbmZpZ3VyYXRpb24gPSBhd2FpdCBjb25maWd1cmF0aW9uUmVzb2x2ZXIoKVxuICBjb25maWd1cmF0aW9uLnNldEN1cnJlbnQoKVxuICBhd2FpdCBjb25maWd1cmF0aW9uLmluaXRpYWxpemUoe3R5cGU6IFwiYmFja2dyb3VuZC1qb2JzLXJ1bm5lclwifSlcbiAgY29uc3QgcmVwb3J0ZXIgPSBuZXcgQmFja2dyb3VuZEpvYnNTdGF0dXNSZXBvcnRlcih7Y29uZmlndXJhdGlvbn0pXG5cbiAgY29uc3QgcmVnaXN0cnkgPSBuZXcgQmFja2dyb3VuZEpvYlJlZ2lzdHJ5KHtjb25maWd1cmF0aW9ufSlcbiAgYXdhaXQgcmVnaXN0cnkubG9hZCgpXG4gIGNvbnN0IEpvYkNsYXNzID0gcmVnaXN0cnkuZ2V0Sm9iQnlOYW1lKHBheWxvYWQuam9iTmFtZSlcbiAgY29uc3Qgam9iSW5zdGFuY2UgPSBuZXcgSm9iQ2xhc3MoKVxuICAvKiogQHR5cGUgeyguLi5hcmdzOiBhbnlbXSkgPT4gUHJvbWlzZTx2b2lkPn0gKi9cbiAgY29uc3QgcGVyZm9ybSA9IGpvYkluc3RhbmNlLnBlcmZvcm1cblxuICB0cnkge1xuICAgIGF3YWl0IGNvbmZpZ3VyYXRpb24ud2l0aENvbm5lY3Rpb25zKGFzeW5jICgpID0+IHtcbiAgICAgIGF3YWl0IHBlcmZvcm0uYXBwbHkoam9iSW5zdGFuY2UsIHBheWxvYWQuYXJncyB8fCBbXSlcbiAgICB9KVxuXG4gICAgaWYgKHBheWxvYWQuaWQpIHtcbiAgICAgIGF3YWl0IHJlcG9ydGVyLnJlcG9ydFdpdGhSZXRyeSh7XG4gICAgICAgIGpvYklkOiBwYXlsb2FkLmlkLFxuICAgICAgICBzdGF0dXM6IFwiY29tcGxldGVkXCIsXG4gICAgICAgIHdvcmtlcklkOiBwYXlsb2FkLndvcmtlcklkLFxuICAgICAgICBoYW5kZWRPZmZBdE1zOiBwYXlsb2FkLmhhbmRlZE9mZkF0TXMsXG4gICAgICAgIG1heER1cmF0aW9uTXM6IDMwMDAwXG4gICAgICB9KVxuICAgIH1cbiAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICBpZiAocGF5bG9hZC5pZCkge1xuICAgICAgYXdhaXQgcmVwb3J0ZXIucmVwb3J0V2l0aFJldHJ5KHtcbiAgICAgICAgam9iSWQ6IHBheWxvYWQuaWQsXG4gICAgICAgIHN0YXR1czogXCJmYWlsZWRcIixcbiAgICAgICAgZXJyb3IsXG4gICAgICAgIHdvcmtlcklkOiBwYXlsb2FkLndvcmtlcklkLFxuICAgICAgICBoYW5kZWRPZmZBdE1zOiBwYXlsb2FkLmhhbmRlZE9mZkF0TXMsXG4gICAgICAgIG1heER1cmF0aW9uTXM6IDMwMDAwXG4gICAgICB9KVxuICAgIH1cblxuICAgIHRocm93IGVycm9yXG4gIH1cbn1cbiJdfQ==
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"worker.d.ts","sourceRoot":"","sources":["../../../src/background-jobs/worker.js"],"names":[],"mappings":"AAUA;IACE;;;;;OAKG;IACH,4CAJG;QAAqD,aAAa;QAC5C,IAAI;QACJ,IAAI;KAC5B,EAcA;IAZC,6DAA6D;IAC7D,sBADW,OAAO,CAAC,OAAO,qBAAqB,EAAE,OAAO,CAAC,CAC2C;IACpG,gEAAgE;IAChE,eADW,OAAO,qBAAqB,EAAE,OAAO,GAAG,SAAS,CAC9B;IAC9B,yBAAgB;IAChB,yBAAgB;IAChB,oBAAuB;IACvB,8DAA4B;IAC5B,qCAAqC;IACrC,YADW,UAAU,GAAG,SAAS,CACN;IAC3B,uDAAuD;IACvD,gBADW,4BAA4B,GAAG,SAAS,CACpB;IAGjC;;OAEG;IACH,SAFa,OAAO,CAAC,IAAI,CAAC,CAYzB;IAED;;OAEG;IACH,QAFa,OAAO,CAAC,IAAI,CAAC,CAKzB;IAED,0BA+BC;IAED;;;OAGG;IACH,oBAHW,OAAO,YAAY,EAAE,oBAAoB,GACvC,OAAO,CAAC,IAAI,CAAC,CAgCzB;IAED;;;OAGG;IACH,uBAHW,OAAO,YAAY,EAAE,oBAAoB,GACvC,OAAO,CAAC,IAAI,CAAC,
|
|
1
|
+
{"version":3,"file":"worker.d.ts","sourceRoot":"","sources":["../../../src/background-jobs/worker.js"],"names":[],"mappings":"AAUA;IACE;;;;;OAKG;IACH,4CAJG;QAAqD,aAAa;QAC5C,IAAI;QACJ,IAAI;KAC5B,EAcA;IAZC,6DAA6D;IAC7D,sBADW,OAAO,CAAC,OAAO,qBAAqB,EAAE,OAAO,CAAC,CAC2C;IACpG,gEAAgE;IAChE,eADW,OAAO,qBAAqB,EAAE,OAAO,GAAG,SAAS,CAC9B;IAC9B,yBAAgB;IAChB,yBAAgB;IAChB,oBAAuB;IACvB,8DAA4B;IAC5B,qCAAqC;IACrC,YADW,UAAU,GAAG,SAAS,CACN;IAC3B,uDAAuD;IACvD,gBADW,4BAA4B,GAAG,SAAS,CACpB;IAGjC;;OAEG;IACH,SAFa,OAAO,CAAC,IAAI,CAAC,CAYzB;IAED;;OAEG;IACH,QAFa,OAAO,CAAC,IAAI,CAAC,CAKzB;IAED,0BA+BC;IAED;;;OAGG;IACH,oBAHW,OAAO,YAAY,EAAE,oBAAoB,GACvC,OAAO,CAAC,IAAI,CAAC,CAgCzB;IAED;;;OAGG;IACH,uBAHW,OAAO,YAAY,EAAE,oBAAoB,GACvC,OAAO,CAAC,IAAI,CAAC,CAgBzB;IAED;;;OAGG;IACH,2BAHW,OAAO,YAAY,EAAE,oBAAoB,GACvC,OAAO,CAAC,IAAI,CAAC,CAuBzB;IAED;;;;;;;;OAQG;IACH,oEAPG;QAAqB,KAAK,EAAlB,MAAM;QACuB,MAAM,EAAnC,WAAW,GAAG,QAAQ;QACP,KAAK,GAApB,OAAO;QACO,aAAa;QACb,QAAQ;KAC9B,GAAU,OAAO,CAAC,IAAI,CAAC,CAUzB;CACF;uBAzLsB,kBAAkB;yCAGA,sBAAsB"}
|
|
@@ -126,7 +126,9 @@ export default class BackgroundJobsWorker {
|
|
|
126
126
|
const jobInstance = new JobClass();
|
|
127
127
|
/** @type {(...args: any[]) => Promise<void>} */
|
|
128
128
|
const perform = jobInstance.perform;
|
|
129
|
-
await
|
|
129
|
+
await configuration.withConnections(async () => {
|
|
130
|
+
await perform.apply(jobInstance, payload.args || []);
|
|
131
|
+
});
|
|
130
132
|
}
|
|
131
133
|
/**
|
|
132
134
|
* @param {import("./types.js").BackgroundJobPayload} payload - Payload.
|
|
@@ -173,4 +175,4 @@ export default class BackgroundJobsWorker {
|
|
|
173
175
|
}
|
|
174
176
|
}
|
|
175
177
|
}
|
|
176
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"worker.js","sourceRoot":"","sources":["../../../src/background-jobs/worker.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport net from \"net\"\nimport {spawn} from \"node:child_process\"\nimport JsonSocket from \"./json-socket.js\"\nimport BackgroundJobRegistry from \"./job-registry.js\"\nimport configurationResolver from \"../configuration-resolver.js\"\nimport BackgroundJobsStatusReporter from \"./status-reporter.js\"\nimport {randomUUID} from \"crypto\"\n\nexport default class BackgroundJobsWorker {\n  /**\n   * @param {object} [args] - Options.\n   * @param {import(\"../configuration.js\").default} [args.configuration] - Configuration.\n   * @param {string} [args.host] - Hostname.\n   * @param {number} [args.port] - Port.\n   */\n  constructor({configuration, host, port} = {}) {\n    /** @type {Promise<import(\"../configuration.js\").default>} */\n    this.configurationPromise = configuration ? Promise.resolve(configuration) : configurationResolver()\n    /** @type {import(\"../configuration.js\").default | undefined} */\n    this.configuration = undefined\n    this.host = host\n    this.port = port\n    this.shouldStop = false\n    this.workerId = randomUUID()\n    /** @type {JsonSocket | undefined} */\n    this.jsonSocket = undefined\n    /** @type {BackgroundJobsStatusReporter | undefined} */\n    this.statusReporter = undefined\n  }\n\n  /**\n   * @returns {Promise<void>} - Resolves when connected.\n   */\n  async start() {\n    this.configuration = await this.configurationPromise\n    this.configuration.setCurrent()\n    await this.configuration.initialize({type: \"background-jobs-worker\"})\n    this.statusReporter = new BackgroundJobsStatusReporter({\n      configuration: this.configuration,\n      host: this.host,\n      port: this.port\n    })\n    await this._connect()\n  }\n\n  /**\n   * @returns {Promise<void>} - Resolves when stopped.\n   */\n  async stop() {\n    this.shouldStop = true\n    if (this.jsonSocket) this.jsonSocket.close()\n  }\n\n  async _connect() {\n    const configuration = this.configuration\n    if (!configuration) throw new Error(\"Background jobs worker configuration not initialized\")\n\n    const config = configuration.getBackgroundJobsConfig()\n    const host = this.host || config.host\n    const port = typeof this.port === \"number\" ? this.port : config.port\n    const socket = net.createConnection({host, port})\n    const jsonSocket = new JsonSocket(socket)\n    this.jsonSocket = jsonSocket\n\n    /** @param {import(\"./types.js\").BackgroundJobSocketMessage} message - Socket message. */\n    jsonSocket.on(\"message\", async (message) => {\n      if (message?.type === \"job\") {\n        await this._handleJob(message.payload)\n      }\n    })\n\n    jsonSocket.on(\"error\", (error) => {\n      console.error(\"Background jobs worker socket error:\", error)\n    })\n\n    jsonSocket.on(\"close\", () => {\n      if (this.shouldStop) return\n      setTimeout(() => { void this._connect() }, 1000)\n    })\n\n    socket.on(\"connect\", () => {\n      jsonSocket.send({type: \"hello\", role: \"worker\", workerId: this.workerId})\n      jsonSocket.send({type: \"ready\"})\n    })\n  }\n\n  /**\n   * @param {import(\"./types.js\").BackgroundJobPayload} payload - Payload.\n   * @returns {Promise<void>} - Resolves when done.\n   */\n  async _handleJob(payload) {\n    if (!payload.id) throw new Error(\"Background job payload missing id\")\n\n    const options = payload.options || {}\n    const shouldFork = options.forked !== false\n\n    if (shouldFork) {\n      await this._spawnDetachedJob(payload)\n      this.jsonSocket?.send({type: \"ready\"})\n      return\n    }\n\n    try {\n      await this._runJobInline(payload)\n      void this._reportJobResult({\n        jobId: payload.id,\n        status: \"completed\",\n        handedOffAtMs: payload.handedOffAtMs,\n        workerId: payload.workerId || this.workerId\n      })\n    } catch (error) {\n      void this._reportJobResult({\n        jobId: payload.id,\n        status: \"failed\",\n        error,\n        handedOffAtMs: payload.handedOffAtMs,\n        workerId: payload.workerId || this.workerId\n      })\n    }\n    this.jsonSocket?.send({type: \"ready\"})\n  }\n\n  /**\n   * @param {import(\"./types.js\").BackgroundJobPayload} payload - Payload.\n   * @returns {Promise<void>} - Resolves when done.\n   */\n  async _runJobInline(payload) {\n    const configuration = this.configuration\n    if (!configuration) throw new Error(\"Background jobs worker configuration not initialized\")\n\n    const registry = new BackgroundJobRegistry({configuration})\n    await registry.load()\n    const JobClass = registry.getJobByName(payload.jobName)\n    const jobInstance = new JobClass()\n    /** @type {(...args: any[]) => Promise<void>} */\n    const perform = jobInstance.perform\n\n    await perform.apply(jobInstance, payload.args || [])\n  }\n\n  /**\n   * @param {import(\"./types.js\").BackgroundJobPayload} payload - Payload.\n   * @returns {Promise<void>} - Resolves when spawned.\n   */\n  async _spawnDetachedJob(payload) {\n    const configuration = this.configuration\n    if (!configuration) throw new Error(\"Background jobs worker configuration not initialized\")\n\n    const directory = configuration.getDirectory()\n    const argvCommand = process.argv[1]\n    const command = argvCommand ? argvCommand : `${directory}/bin/velocious.js`\n    const encodedPayload = Buffer.from(JSON.stringify(payload)).toString(\"base64\")\n    const backgroundJobsConfig = configuration.getBackgroundJobsConfig()\n    const child = spawn(process.execPath, [command, \"background-jobs-runner\"], {\n      cwd: directory,\n      detached: true,\n      stdio: \"ignore\",\n      env: Object.assign({}, process.env, {\n        VELOCIOUS_BACKGROUND_JOBS_HOST: backgroundJobsConfig.host,\n        VELOCIOUS_BACKGROUND_JOBS_PORT: `${backgroundJobsConfig.port}`,\n        VELOCIOUS_JOB_PAYLOAD: encodedPayload\n      })\n    })\n\n    child.unref()\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {string} args.jobId - Job id.\n   * @param {\"completed\" | \"failed\"} args.status - Status.\n   * @param {unknown} [args.error] - Error.\n   * @param {number} [args.handedOffAtMs] - Handed off timestamp.\n   * @param {string} [args.workerId] - Worker id.\n   * @returns {Promise<void>} - Resolves when reported.\n   */\n  async _reportJobResult({jobId, status, error, handedOffAtMs, workerId}) {\n    if (!this.statusReporter) return\n\n    try {\n      await this.statusReporter.reportWithRetry({jobId, status, error, handedOffAtMs, workerId})\n    } catch (reportError) {\n      console.error(\"Background job status reporting failed:\", reportError)\n    }\n  }\n}\n"]}
|
|
178
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"worker.js","sourceRoot":"","sources":["../../../src/background-jobs/worker.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport net from \"net\"\nimport {spawn} from \"node:child_process\"\nimport JsonSocket from \"./json-socket.js\"\nimport BackgroundJobRegistry from \"./job-registry.js\"\nimport configurationResolver from \"../configuration-resolver.js\"\nimport BackgroundJobsStatusReporter from \"./status-reporter.js\"\nimport {randomUUID} from \"crypto\"\n\nexport default class BackgroundJobsWorker {\n  /**\n   * @param {object} [args] - Options.\n   * @param {import(\"../configuration.js\").default} [args.configuration] - Configuration.\n   * @param {string} [args.host] - Hostname.\n   * @param {number} [args.port] - Port.\n   */\n  constructor({configuration, host, port} = {}) {\n    /** @type {Promise<import(\"../configuration.js\").default>} */\n    this.configurationPromise = configuration ? Promise.resolve(configuration) : configurationResolver()\n    /** @type {import(\"../configuration.js\").default | undefined} */\n    this.configuration = undefined\n    this.host = host\n    this.port = port\n    this.shouldStop = false\n    this.workerId = randomUUID()\n    /** @type {JsonSocket | undefined} */\n    this.jsonSocket = undefined\n    /** @type {BackgroundJobsStatusReporter | undefined} */\n    this.statusReporter = undefined\n  }\n\n  /**\n   * @returns {Promise<void>} - Resolves when connected.\n   */\n  async start() {\n    this.configuration = await this.configurationPromise\n    this.configuration.setCurrent()\n    await this.configuration.initialize({type: \"background-jobs-worker\"})\n    this.statusReporter = new BackgroundJobsStatusReporter({\n      configuration: this.configuration,\n      host: this.host,\n      port: this.port\n    })\n    await this._connect()\n  }\n\n  /**\n   * @returns {Promise<void>} - Resolves when stopped.\n   */\n  async stop() {\n    this.shouldStop = true\n    if (this.jsonSocket) this.jsonSocket.close()\n  }\n\n  async _connect() {\n    const configuration = this.configuration\n    if (!configuration) throw new Error(\"Background jobs worker configuration not initialized\")\n\n    const config = configuration.getBackgroundJobsConfig()\n    const host = this.host || config.host\n    const port = typeof this.port === \"number\" ? this.port : config.port\n    const socket = net.createConnection({host, port})\n    const jsonSocket = new JsonSocket(socket)\n    this.jsonSocket = jsonSocket\n\n    /** @param {import(\"./types.js\").BackgroundJobSocketMessage} message - Socket message. */\n    jsonSocket.on(\"message\", async (message) => {\n      if (message?.type === \"job\") {\n        await this._handleJob(message.payload)\n      }\n    })\n\n    jsonSocket.on(\"error\", (error) => {\n      console.error(\"Background jobs worker socket error:\", error)\n    })\n\n    jsonSocket.on(\"close\", () => {\n      if (this.shouldStop) return\n      setTimeout(() => { void this._connect() }, 1000)\n    })\n\n    socket.on(\"connect\", () => {\n      jsonSocket.send({type: \"hello\", role: \"worker\", workerId: this.workerId})\n      jsonSocket.send({type: \"ready\"})\n    })\n  }\n\n  /**\n   * @param {import(\"./types.js\").BackgroundJobPayload} payload - Payload.\n   * @returns {Promise<void>} - Resolves when done.\n   */\n  async _handleJob(payload) {\n    if (!payload.id) throw new Error(\"Background job payload missing id\")\n\n    const options = payload.options || {}\n    const shouldFork = options.forked !== false\n\n    if (shouldFork) {\n      await this._spawnDetachedJob(payload)\n      this.jsonSocket?.send({type: \"ready\"})\n      return\n    }\n\n    try {\n      await this._runJobInline(payload)\n      void this._reportJobResult({\n        jobId: payload.id,\n        status: \"completed\",\n        handedOffAtMs: payload.handedOffAtMs,\n        workerId: payload.workerId || this.workerId\n      })\n    } catch (error) {\n      void this._reportJobResult({\n        jobId: payload.id,\n        status: \"failed\",\n        error,\n        handedOffAtMs: payload.handedOffAtMs,\n        workerId: payload.workerId || this.workerId\n      })\n    }\n    this.jsonSocket?.send({type: \"ready\"})\n  }\n\n  /**\n   * @param {import(\"./types.js\").BackgroundJobPayload} payload - Payload.\n   * @returns {Promise<void>} - Resolves when done.\n   */\n  async _runJobInline(payload) {\n    const configuration = this.configuration\n    if (!configuration) throw new Error(\"Background jobs worker configuration not initialized\")\n\n    const registry = new BackgroundJobRegistry({configuration})\n    await registry.load()\n    const JobClass = registry.getJobByName(payload.jobName)\n    const jobInstance = new JobClass()\n    /** @type {(...args: any[]) => Promise<void>} */\n    const perform = jobInstance.perform\n\n    await configuration.withConnections(async () => {\n      await perform.apply(jobInstance, payload.args || [])\n    })\n  }\n\n  /**\n   * @param {import(\"./types.js\").BackgroundJobPayload} payload - Payload.\n   * @returns {Promise<void>} - Resolves when spawned.\n   */\n  async _spawnDetachedJob(payload) {\n    const configuration = this.configuration\n    if (!configuration) throw new Error(\"Background jobs worker configuration not initialized\")\n\n    const directory = configuration.getDirectory()\n    const argvCommand = process.argv[1]\n    const command = argvCommand ? argvCommand : `${directory}/bin/velocious.js`\n    const encodedPayload = Buffer.from(JSON.stringify(payload)).toString(\"base64\")\n    const backgroundJobsConfig = configuration.getBackgroundJobsConfig()\n    const child = spawn(process.execPath, [command, \"background-jobs-runner\"], {\n      cwd: directory,\n      detached: true,\n      stdio: \"ignore\",\n      env: Object.assign({}, process.env, {\n        VELOCIOUS_BACKGROUND_JOBS_HOST: backgroundJobsConfig.host,\n        VELOCIOUS_BACKGROUND_JOBS_PORT: `${backgroundJobsConfig.port}`,\n        VELOCIOUS_JOB_PAYLOAD: encodedPayload\n      })\n    })\n\n    child.unref()\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {string} args.jobId - Job id.\n   * @param {\"completed\" | \"failed\"} args.status - Status.\n   * @param {unknown} [args.error] - Error.\n   * @param {number} [args.handedOffAtMs] - Handed off timestamp.\n   * @param {string} [args.workerId] - Worker id.\n   * @returns {Promise<void>} - Resolves when reported.\n   */\n  async _reportJobResult({jobId, status, error, handedOffAtMs, workerId}) {\n    if (!this.statusReporter) return\n\n    try {\n      await this.statusReporter.reportWithRetry({jobId, status, error, handedOffAtMs, workerId})\n    } catch (reportError) {\n      console.error(\"Background job status reporting failed:\", reportError)\n    }\n  }\n}\n"]}
|
|
@@ -75,6 +75,16 @@ export default class VelociousWebsocketClient {
|
|
|
75
75
|
_sessionId: string | null;
|
|
76
76
|
/** @type {boolean} - true between a reconnect and the session-resumed / session-gone reply. */
|
|
77
77
|
_awaitingResume: boolean;
|
|
78
|
+
/** @type {boolean} - true once the current socket has an active session ready for app messages. */
|
|
79
|
+
_sessionReady: boolean;
|
|
80
|
+
/** @type {string | null} - provisional session id announced before a resume attempt finishes. */
|
|
81
|
+
_pendingSessionId: string | null;
|
|
82
|
+
/** @type {Promise<void> | null} */
|
|
83
|
+
_sessionReadyPromise: Promise<void> | null;
|
|
84
|
+
/** @type {(() => void) | null} */
|
|
85
|
+
_resolveSessionReady: (() => void) | null;
|
|
86
|
+
/** @type {unknown | null} */
|
|
87
|
+
_sessionReadyError: unknown | null;
|
|
78
88
|
/** @type {{get: () => string | null | undefined | Promise<string | null | undefined>, set: (sessionId: string) => void | Promise<void>, clear: () => void | Promise<void>} | undefined} */
|
|
79
89
|
_sessionStore: {
|
|
80
90
|
get: () => string | null | undefined | Promise<string | null | undefined>;
|
|
@@ -98,6 +108,8 @@ export default class VelociousWebsocketClient {
|
|
|
98
108
|
_waitingForOnline: boolean;
|
|
99
109
|
/** @returns {boolean} */
|
|
100
110
|
isOpen(): boolean;
|
|
111
|
+
/** @returns {boolean} */
|
|
112
|
+
isSessionReady(): boolean;
|
|
101
113
|
/**
|
|
102
114
|
* Opens a 1:1 `WebsocketConnection` of the given type against the
|
|
103
115
|
* server. Requires the socket to already be connected (call
|
|
@@ -323,6 +335,15 @@ export default class VelociousWebsocketClient {
|
|
|
323
335
|
* @returns {void}
|
|
324
336
|
*/
|
|
325
337
|
private _clearPersistedSessionId;
|
|
338
|
+
/** @returns {Promise<void>} */
|
|
339
|
+
_waitForSessionReady(): Promise<void>;
|
|
340
|
+
/** @returns {void} */
|
|
341
|
+
_markSessionReady(): void;
|
|
342
|
+
/**
|
|
343
|
+
* @param {unknown} [error]
|
|
344
|
+
* @returns {void}
|
|
345
|
+
*/
|
|
346
|
+
_resetSessionReadyState(error?: unknown): void;
|
|
326
347
|
}
|
|
327
348
|
declare class VelociousWebsocketResponse {
|
|
328
349
|
/**
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"websocket-client.d.ts","sourceRoot":"","sources":["../../../src/http-client/websocket-client.js"],"names":[],"mappings":"AAQA;;;GAGG;AACH;IAQE;;;;;;;;OAQG;IACH,2FAPG;QAAuB,aAAa;QACb,KAAK;QACgI,cAAc;0BAAnJ,MAAM,OAAO,GAAG,OAAO,CAAC,OAAO,CAAC;wBAAc,CAAC,QAAQ,EAAE,CAAC,QAAQ,EAAE,OAAO,KAAK,IAAI,KAAK,CAAC,MAAM,IAAI,CAAC,GAAG;gBAAC,MAAM,EAAE,MAAM,IAAI,CAAA;aAAC;;QAC3H,eAAe;QACyI,YAAY;iBAA9K,MAAM,MAAM,GAAG,IAAI,GAAG,SAAS,GAAG,OAAO,CAAC,MAAM,GAAG,IAAI,GAAG,SAAS,CAAC;iBAAO,CAAC,SAAS,EAAE,MAAM,KAAK,IAAI,GAAG,OAAO,CAAC,IAAI,CAAC;mBAAS,MAAM,IAAI,GAAG,OAAO,CAAC,IAAI,CAAC;;QACjJ,GAAG;KAC3B,
|
|
1
|
+
{"version":3,"file":"websocket-client.d.ts","sourceRoot":"","sources":["../../../src/http-client/websocket-client.js"],"names":[],"mappings":"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"}
|
|
@@ -58,6 +58,16 @@ export default class VelociousWebsocketClient {
|
|
|
58
58
|
this._sessionId = null;
|
|
59
59
|
/** @type {boolean} - true between a reconnect and the session-resumed / session-gone reply. */
|
|
60
60
|
this._awaitingResume = false;
|
|
61
|
+
/** @type {boolean} - true once the current socket has an active session ready for app messages. */
|
|
62
|
+
this._sessionReady = false;
|
|
63
|
+
/** @type {string | null} - provisional session id announced before a resume attempt finishes. */
|
|
64
|
+
this._pendingSessionId = null;
|
|
65
|
+
/** @type {Promise<void> | null} */
|
|
66
|
+
this._sessionReadyPromise = null;
|
|
67
|
+
/** @type {(() => void) | null} */
|
|
68
|
+
this._resolveSessionReady = null;
|
|
69
|
+
/** @type {unknown | null} */
|
|
70
|
+
this._sessionReadyError = null;
|
|
61
71
|
/** @type {{get: () => string | null | undefined | Promise<string | null | undefined>, set: (sessionId: string) => void | Promise<void>, clear: () => void | Promise<void>} | undefined} */
|
|
62
72
|
this._sessionStore = sessionStore;
|
|
63
73
|
/** @type {boolean} - true once the sessionStore has been consulted for a restored id. */
|
|
@@ -73,6 +83,10 @@ export default class VelociousWebsocketClient {
|
|
|
73
83
|
isOpen() {
|
|
74
84
|
return Boolean(this.socket && this.socket.readyState === this.socket.OPEN);
|
|
75
85
|
}
|
|
86
|
+
/** @returns {boolean} */
|
|
87
|
+
isSessionReady() {
|
|
88
|
+
return this._sessionReady;
|
|
89
|
+
}
|
|
76
90
|
/**
|
|
77
91
|
* Opens a 1:1 `WebsocketConnection` of the given type against the
|
|
78
92
|
* server. Requires the socket to already be connected (call
|
|
@@ -155,16 +169,37 @@ export default class VelociousWebsocketClient {
|
|
|
155
169
|
* @returns {void}
|
|
156
170
|
*/
|
|
157
171
|
_sendChannelSubscribe(subscription) {
|
|
158
|
-
if (!this.isOpen() || !subscription._needsSubscribe())
|
|
172
|
+
if (!this.isOpen() || !this.isSessionReady() || !subscription._needsSubscribe())
|
|
159
173
|
return;
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
174
|
+
// Send first and only mark as sent on success. If the socket
|
|
175
|
+
// closes between `isOpen()` and `send()` and `_sendMessage` throws,
|
|
176
|
+
// `_subscribeSent` must stay false so the reconnect path's
|
|
177
|
+
// `_sendPendingChannelSubscriptions()` can retry.
|
|
178
|
+
try {
|
|
179
|
+
this._sendMessage({
|
|
180
|
+
type: "channel-subscribe",
|
|
181
|
+
subscriptionId: subscription.subscriptionId,
|
|
182
|
+
channelType: subscription.channelType,
|
|
183
|
+
params: subscription.params,
|
|
184
|
+
...(subscription.lastEventId ? { lastEventId: subscription.lastEventId } : {})
|
|
185
|
+
});
|
|
186
|
+
subscription._markSubscribeSent();
|
|
187
|
+
}
|
|
188
|
+
catch (error) {
|
|
189
|
+
// Transient closed-socket race: leave the subscription
|
|
190
|
+
// retryable so `_sendPendingChannelSubscriptions()` can resend
|
|
191
|
+
// after reconnect. Rethrow so callers that expect throws on a
|
|
192
|
+
// dead socket still see them.
|
|
193
|
+
if (!this.isOpen())
|
|
194
|
+
throw error;
|
|
195
|
+
// Non-recoverable send failure on an open socket (e.g.
|
|
196
|
+
// `JSON.stringify` failing on BigInt/cyclic params). Close the
|
|
197
|
+
// subscription and remove it from the registry so it cannot
|
|
198
|
+
// poison future `_sendPendingChannelSubscriptions()` loops by
|
|
199
|
+
// throwing on every attempt and aborting later subscriptions.
|
|
200
|
+
this._channelSubscriptions.delete(subscription.subscriptionId);
|
|
201
|
+
subscription._handleClosed(`send_failed: ${error instanceof Error ? error.message : String(error)}`);
|
|
202
|
+
}
|
|
168
203
|
}
|
|
169
204
|
/** @returns {void} */
|
|
170
205
|
_sendPendingChannelSubscriptions() {
|
|
@@ -279,6 +314,7 @@ export default class VelociousWebsocketClient {
|
|
|
279
314
|
return;
|
|
280
315
|
if (this.connectPromise)
|
|
281
316
|
return this.connectPromise;
|
|
317
|
+
this._resetSessionReadyState();
|
|
282
318
|
this._waitingForOnline = false;
|
|
283
319
|
this.connectionAttempts += 1;
|
|
284
320
|
this.connectPromise = new Promise((resolve, reject) => {
|
|
@@ -336,9 +372,7 @@ export default class VelociousWebsocketClient {
|
|
|
336
372
|
if (Object.keys(this._metadata).length > 0) {
|
|
337
373
|
this._sendMessage({ type: "metadata", data: { ...this._metadata } });
|
|
338
374
|
}
|
|
339
|
-
|
|
340
|
-
this._sendPendingChannelSubscriptions();
|
|
341
|
-
}
|
|
375
|
+
await this._waitForSessionReady();
|
|
342
376
|
this.disconnectedSince = null;
|
|
343
377
|
}
|
|
344
378
|
/**
|
|
@@ -355,6 +389,7 @@ export default class VelociousWebsocketClient {
|
|
|
355
389
|
if (this.socket.readyState === this.socket.CLOSED) {
|
|
356
390
|
this.socket = undefined;
|
|
357
391
|
this.connectPromise = undefined;
|
|
392
|
+
this._resetSessionReadyState();
|
|
358
393
|
return;
|
|
359
394
|
}
|
|
360
395
|
await new Promise((resolve) => {
|
|
@@ -363,6 +398,7 @@ export default class VelociousWebsocketClient {
|
|
|
363
398
|
});
|
|
364
399
|
this.socket = undefined;
|
|
365
400
|
this.connectPromise = undefined;
|
|
401
|
+
this._resetSessionReadyState();
|
|
366
402
|
}
|
|
367
403
|
/**
|
|
368
404
|
* Disable auto-reconnect and close the websocket.
|
|
@@ -384,6 +420,7 @@ export default class VelociousWebsocketClient {
|
|
|
384
420
|
this.socket?.close();
|
|
385
421
|
});
|
|
386
422
|
this.connectPromise = undefined;
|
|
423
|
+
this._resetSessionReadyState();
|
|
387
424
|
}
|
|
388
425
|
/**
|
|
389
426
|
* Perform a POST request over the websocket.
|
|
@@ -614,16 +651,23 @@ export default class VelociousWebsocketClient {
|
|
|
614
651
|
}
|
|
615
652
|
}
|
|
616
653
|
else if (type === "session-established") {
|
|
654
|
+
this._pendingSessionId = typeof message.sessionId === "string" ? message.sessionId : null;
|
|
617
655
|
// First connect: cache sessionId for future resume attempts.
|
|
618
656
|
if (!this._awaitingResume) {
|
|
619
|
-
this._sessionId =
|
|
620
|
-
this.
|
|
657
|
+
this._sessionId = this._pendingSessionId;
|
|
658
|
+
if (this._sessionId) {
|
|
659
|
+
this._persistSessionId(this._sessionId);
|
|
660
|
+
}
|
|
661
|
+
this._markSessionReady();
|
|
662
|
+
this._sendPendingChannelSubscriptions();
|
|
621
663
|
}
|
|
622
664
|
}
|
|
623
665
|
else if (type === "session-resumed") {
|
|
624
666
|
this._awaitingResume = false;
|
|
667
|
+
this._pendingSessionId = null;
|
|
625
668
|
this._sessionId = message.sessionId;
|
|
626
669
|
this._persistSessionId(message.sessionId);
|
|
670
|
+
this._markSessionReady();
|
|
627
671
|
this._sendPendingChannelSubscriptions();
|
|
628
672
|
// Fire onResume on every live handle so user code knows the
|
|
629
673
|
// session came back with state intact.
|
|
@@ -635,6 +679,7 @@ export default class VelociousWebsocketClient {
|
|
|
635
679
|
else if (type === "session-gone") {
|
|
636
680
|
this._awaitingResume = false;
|
|
637
681
|
this._sessionId = null;
|
|
682
|
+
this._pendingSessionId = null;
|
|
638
683
|
this._clearPersistedSessionId();
|
|
639
684
|
// Tear down every live handle — their server-side counterparts
|
|
640
685
|
// are gone and nothing can bring them back.
|
|
@@ -646,6 +691,7 @@ export default class VelociousWebsocketClient {
|
|
|
646
691
|
this._channelSubscriptions.clear();
|
|
647
692
|
for (const subscription of subs)
|
|
648
693
|
subscription._handleClosed("session_gone");
|
|
694
|
+
this._markSessionReady();
|
|
649
695
|
}
|
|
650
696
|
else if (type === "error" && message.id) {
|
|
651
697
|
const pending = this.pendingRequests.get(message.id);
|
|
@@ -670,6 +716,7 @@ export default class VelociousWebsocketClient {
|
|
|
670
716
|
*/
|
|
671
717
|
onClose = () => {
|
|
672
718
|
this.disconnectedSince ||= Date.now();
|
|
719
|
+
this._resetSessionReadyState();
|
|
673
720
|
for (const [id, { reject }] of this.pendingRequests.entries()) {
|
|
674
721
|
reject(new Error(`Websocket closed before response for ${id}`));
|
|
675
722
|
}
|
|
@@ -827,6 +874,45 @@ export default class VelociousWebsocketClient {
|
|
|
827
874
|
this._debug("sessionStore.clear failed", error);
|
|
828
875
|
}
|
|
829
876
|
}
|
|
877
|
+
/** @returns {Promise<void>} */
|
|
878
|
+
_waitForSessionReady() {
|
|
879
|
+
if (this._sessionReady)
|
|
880
|
+
return Promise.resolve();
|
|
881
|
+
if (!this._sessionReadyPromise || !this._resolveSessionReady) {
|
|
882
|
+
this._sessionReadyPromise = new Promise((resolve) => {
|
|
883
|
+
this._resolveSessionReady = resolve;
|
|
884
|
+
});
|
|
885
|
+
}
|
|
886
|
+
return this._sessionReadyPromise.then(() => {
|
|
887
|
+
if (this._sessionReadyError) {
|
|
888
|
+
const error = this._sessionReadyError;
|
|
889
|
+
this._sessionReadyError = null;
|
|
890
|
+
throw error;
|
|
891
|
+
}
|
|
892
|
+
});
|
|
893
|
+
}
|
|
894
|
+
/** @returns {void} */
|
|
895
|
+
_markSessionReady() {
|
|
896
|
+
if (this._sessionReady)
|
|
897
|
+
return;
|
|
898
|
+
this._sessionReady = true;
|
|
899
|
+
this._sessionReadyError = null;
|
|
900
|
+
this._resolveSessionReady?.();
|
|
901
|
+
this._resolveSessionReady = null;
|
|
902
|
+
this._sessionReadyPromise = null;
|
|
903
|
+
}
|
|
904
|
+
/**
|
|
905
|
+
* @param {unknown} [error]
|
|
906
|
+
* @returns {void}
|
|
907
|
+
*/
|
|
908
|
+
_resetSessionReadyState(error = new Error("Websocket session readiness was reset")) {
|
|
909
|
+
this._sessionReady = false;
|
|
910
|
+
this._pendingSessionId = null;
|
|
911
|
+
this._sessionReadyError = error;
|
|
912
|
+
this._resolveSessionReady?.();
|
|
913
|
+
this._sessionReadyPromise = null;
|
|
914
|
+
this._resolveSessionReady = null;
|
|
915
|
+
}
|
|
830
916
|
}
|
|
831
917
|
class VelociousWebsocketResponse {
|
|
832
918
|
/**
|
|
@@ -849,4 +935,4 @@ class VelociousWebsocketResponse {
|
|
|
849
935
|
return deserializeFrontendModelTransportValue(JSON.parse(this.body));
|
|
850
936
|
}
|
|
851
937
|
}
|
|
852
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"websocket-client.js","sourceRoot":"","sources":["../../../src/http-client/websocket-client.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport VelociousWebsocketClientConnection from \"./websocket-connection.js\"\nimport VelociousWebsocketClientSubscription from \"./websocket-channel.js\"\nimport {deserializeFrontendModelTransportValue} from \"../frontend-models/transport-serialization.js\"\n\nconst DEFAULT_RECONNECT_DELAYS = [1000, 2000, 4000, 8000, 15000]\n\n/**\n * A small websocket client that mirrors simple HTTP-style calls and channel subscriptions.\n * Supports optional auto-reconnect with exponential backoff and listener re-subscription.\n */\nexport default class VelociousWebsocketClient {\n  /** @type {Map<string, {reject: (error: unknown) => void, resolve: (response: VelociousWebsocketResponse) => void}>} */\n  pendingRequests\n  /** @type {Map<string, {reject: (error: unknown) => void, resolve: (value?: void) => void}>} */\n  pendingSubscriptions\n  /** @type {Map<string, {callbacks: Set<(payload: any) => void>, channel: string, params: Record<string, any> | undefined, ready: Promise<void>}>} */\n  listeners\n\n  /**\n   * @param {object} [args] - Options object.\n   * @param {boolean} [args.autoReconnect] - Enable auto-reconnect with exponential backoff.\n   * @param {boolean} [args.debug] - Whether debug.\n   * @param {{getIsOnline?: () => boolean | Promise<boolean>, subscribe?: (callback: (isOnline: boolean) => void) => (() => void) | {remove: () => void}}} [args.networkMonitor] - Optional online-state adapter. When provided, auto-reconnect can wait for the network to report online before reconnecting, and open sockets are closed when the monitor reports offline.\n   * @param {number[]} [args.reconnectDelays] - Backoff delays in ms (default: [1000, 2000, 4000, 8000, 15000]).\n   * @param {{get: () => string | null | undefined | Promise<string | null | undefined>, set: (sessionId: string) => void | Promise<void>, clear: () => void | Promise<void>}} [args.sessionStore] - Optional sessionId persistence hook. When provided, the client writes every `session-established` / `session-resumed` id via `store.set(id)` and clears it on `session-gone`. Before the first `connect()`, the client reads any persisted id via `store.get()` and attempts resumption. Apps should back this by whatever persistence layer survives page reloads (localStorage, a cookie, SQLite, etc.).\n   * @param {string} [args.url] Full websocket URL (default: ws://127.0.0.1:3006/websocket)\n   */\n  constructor({autoReconnect = true, debug = false, networkMonitor, reconnectDelays, sessionStore, url} = {}) {\n    if (!globalThis.WebSocket) throw new Error(\"WebSocket global is not available\")\n\n    /** @type {boolean} */\n    this.autoReconnect = autoReconnect\n    this.debug = debug\n    /** @type {number | null} */\n    this.disconnectedSince = null\n    this.pendingRequests = new Map()\n    this.pendingSubscriptions = new Map()\n    /** @type {number} */\n    this.reconnectAttempt = 0\n    /** @type {number} */\n    this.connectionAttempts = 0\n    /** @type {number[]} */\n    this.reconnectDelays = reconnectDelays || DEFAULT_RECONNECT_DELAYS\n    /** @type {ReturnType<typeof setTimeout> | null} */\n    this.reconnectTimer = null\n    this.url = url || \"ws://127.0.0.1:3006/websocket\"\n    this.listeners = new Map()\n    this.nextID = 1\n    /** @type {(() => void | Promise<void>) | null} */\n    this.onReconnect = null\n\n    /** @type {Record<string, any>} */\n    this._metadata = {}\n\n    /** @type {Map<string, VelociousWebsocketClientConnection>} */\n    this._connections = new Map()\n\n    /** @type {Map<string, VelociousWebsocketClientSubscription>} */\n    this._channelSubscriptions = new Map()\n\n    this._nextConnectionIdSeq = 1\n    this._nextSubscriptionIdSeq = 1\n\n    /** @type {string | null} - sessionId received from `session-established`; sent on reconnect for resumption. */\n    this._sessionId = null\n\n    /** @type {boolean} - true between a reconnect and the session-resumed / session-gone reply. */\n    this._awaitingResume = false\n\n    /** @type {{get: () => string | null | undefined | Promise<string | null | undefined>, set: (sessionId: string) => void | Promise<void>, clear: () => void | Promise<void>} | undefined} */\n    this._sessionStore = sessionStore\n    /** @type {boolean} - true once the sessionStore has been consulted for a restored id. */\n    this._sessionStoreRestored = false\n\n    /** @type {{getIsOnline?: () => boolean | Promise<boolean>, subscribe?: (callback: (isOnline: boolean) => void) => (() => void) | {remove: () => void}} | undefined} */\n    this._networkMonitor = networkMonitor\n\n    /** @type {null | (() => void) | {remove: () => void}} */\n    this._networkMonitorSubscription = null\n\n    /** @type {boolean} */\n    this._waitingForOnline = false\n  }\n\n  /** @returns {boolean} */\n  isOpen() {\n    return Boolean(this.socket && this.socket.readyState === this.socket.OPEN)\n  }\n\n  /**\n   * Opens a 1:1 `WebsocketConnection` of the given type against the\n   * server. Requires the socket to already be connected (call\n   * `connect()` first).\n   *\n   * @param {string} connectionType - Name the server registered the class under.\n   * @param {{params?: Record<string, any>, onConnect?: () => void, onMessage?: (body: any) => void, onDisconnect?: () => void, onResume?: () => void, onClose?: (reason: string) => void}} [options]\n   * @returns {VelociousWebsocketClientConnection}\n   */\n  openConnection(connectionType, options = {}) {\n    if (!this.isOpen()) throw new Error(\"Websocket is not open; call connect() first\")\n\n    const connectionId = `c${this._nextConnectionIdSeq++}`\n    const connection = new VelociousWebsocketClientConnection({\n      client: this,\n      connectionId,\n      connectionType,\n      params: options.params,\n      onConnect: options.onConnect,\n      onMessage: options.onMessage,\n      onDisconnect: options.onDisconnect,\n      onResume: options.onResume,\n      onClose: options.onClose\n    })\n\n    this._connections.set(connectionId, connection)\n    this._sendMessage({\n      type: \"connection-open\",\n      connectionId,\n      connectionType,\n      params: options.params || {}\n    })\n\n    return connection\n  }\n\n  /**\n   * Drops a connection handle from the registry. Called by\n   * `VelociousWebsocketClientConnection.close()` after it notifies\n   * the server, and by the session-destroyed cleanup path.\n   *\n   * @param {string} connectionId\n   * @returns {void}\n   */\n  _removeConnection(connectionId) {\n    this._connections.delete(connectionId)\n  }\n\n  /**\n   * Subscribes to a named WebsocketChannel. If the socket is not yet\n   * open, the subscription is queued and sent once a connection is\n   * established.\n   *\n   * @param {string} channelType - Name the server registered the channel under.\n   * @param {{params?: Record<string, any>, lastEventId?: string, onMessage?: (body: any) => void, onDisconnect?: () => void, onResume?: () => void, onClose?: (reason: string) => void}} [options]\n   * @returns {VelociousWebsocketClientSubscription}\n   */\n  subscribeChannel(channelType, options = {}) {\n    const subscriptionId = `s${this._nextSubscriptionIdSeq++}`\n    const subscription = new VelociousWebsocketClientSubscription({\n      client: this,\n      subscriptionId,\n      channelType,\n      lastEventId: options.lastEventId,\n      params: options.params,\n      onMessage: options.onMessage,\n      onDisconnect: options.onDisconnect,\n      onResume: options.onResume,\n      onClose: options.onClose\n    })\n\n    this._channelSubscriptions.set(subscriptionId, subscription)\n    this._sendChannelSubscribe(subscription)\n\n    return subscription\n  }\n\n  /**\n   * @param {string} subscriptionId\n   * @returns {void}\n   */\n  _removeChannelSubscription(subscriptionId) {\n    this._channelSubscriptions.delete(subscriptionId)\n  }\n\n  /**\n   * @param {VelociousWebsocketClientSubscription} subscription\n   * @returns {void}\n   */\n  _sendChannelSubscribe(subscription) {\n    if (!this.isOpen() || !subscription._needsSubscribe()) return\n\n    subscription._markSubscribeSent()\n    this._sendMessage({\n      type: \"channel-subscribe\",\n      subscriptionId: subscription.subscriptionId,\n      channelType: subscription.channelType,\n      params: subscription.params,\n      ...(subscription.lastEventId ? {lastEventId: subscription.lastEventId} : {})\n    })\n  }\n\n  /** @returns {void} */\n  _sendPendingChannelSubscriptions() {\n    for (const subscription of this._channelSubscriptions.values()) {\n      this._sendChannelSubscribe(subscription)\n    }\n  }\n\n  /** @returns {Promise<boolean>} */\n  async _isOnline() {\n    if (!this._networkMonitor?.getIsOnline) return true\n\n    try {\n      return await this._networkMonitor.getIsOnline() !== false\n    } catch (error) {\n      this._debug(\"networkMonitor.getIsOnline failed\", error)\n      return true\n    }\n  }\n\n  /** @returns {Promise<boolean>} */\n  async _shouldWaitForOnline() {\n    if (!this._networkMonitor) return false\n\n    const isOnline = await this._isOnline()\n    if (isOnline) return false\n\n    this._waitingForOnline = true\n    this._cancelPendingReconnect()\n    return true\n  }\n\n  /** @returns {void} */\n  _ensureNetworkMonitorSubscription() {\n    if (!this._networkMonitor?.subscribe || this._networkMonitorSubscription) return\n\n    this._networkMonitorSubscription = this._networkMonitor.subscribe((isOnline) => {\n      if (!this.autoReconnect) return\n\n      if (isOnline) {\n        if (!this._waitingForOnline) return\n\n        this._waitingForOnline = false\n        void this._attemptReconnect()\n        return\n      }\n\n      this._waitingForOnline = true\n      this._cancelPendingReconnect()\n\n      if (this.isOpen()) {\n        void this.dropConnection()\n      }\n    })\n  }\n\n  /** @returns {void} */\n  _teardownNetworkMonitorSubscription() {\n    if (!this._networkMonitorSubscription) return\n\n    if (typeof this._networkMonitorSubscription === \"function\") {\n      this._networkMonitorSubscription()\n    } else {\n      this._networkMonitorSubscription.remove()\n    }\n\n    this._networkMonitorSubscription = null\n  }\n\n  /**\n   * Sets a global metadata value that is sent to the server.\n   * For WebSocket connections, a metadata update message is sent immediately.\n   * @param {string} key - Metadata key.\n   * @param {any} value - Metadata value (null to clear).\n   * @returns {void}\n   */\n  setMetadata(key, value) {\n    if (value === null || value === undefined) {\n      delete this._metadata[key]\n    } else {\n      this._metadata[key] = value\n    }\n\n    if (this.socket && this.socket.readyState === this.socket.OPEN) {\n      this._sendMessage({type: \"metadata\", data: {...this._metadata}})\n    }\n  }\n\n  /** @returns {Record<string, any>} - Current metadata. */\n  getMetadata() {\n    return {...this._metadata}\n  }\n\n  /**\n   * Ensure a websocket connection is open.\n   * Auto-reconnect and online gating are enabled by default.\n   * Pass `autoReconnect: false` or `waitForOnline: false` only when a caller\n   * explicitly needs lower-level behavior.\n   * @param {{autoReconnect?: boolean, waitForOnline?: boolean, resetReconnectState?: boolean}} [options]\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async connect({autoReconnect = this.autoReconnect, waitForOnline = true, resetReconnectState = true} = {}) {\n    this.autoReconnect = autoReconnect\n\n    if (this.autoReconnect) {\n      this._ensureNetworkMonitorSubscription()\n    } else {\n      this._waitingForOnline = false\n      this._cancelPendingReconnect()\n      this._teardownNetworkMonitorSubscription()\n    }\n\n    if (waitForOnline && this.autoReconnect && !await this._isOnline()) {\n      this._waitingForOnline = true\n      return\n    }\n\n    if (resetReconnectState) {\n      this.reconnectAttempt = 0\n    }\n\n    if (this.socket && this.socket.readyState === this.socket.OPEN) return\n    if (this.connectPromise) return this.connectPromise\n\n    this._waitingForOnline = false\n    this.connectionAttempts += 1\n\n    this.connectPromise = new Promise((resolve, reject) => {\n      this.socket = new WebSocket(this.url)\n\n      const cleanup = () => {\n        this.socket?.removeEventListener(\"open\", onOpen)\n        this.socket?.removeEventListener(\"error\", onError)\n      }\n\n      const onOpen = () => {\n        cleanup()\n        resolve(undefined)\n      }\n      const onError = (/** @type {Event & {error?: unknown}} */ event) => {\n        cleanup()\n        const error = event?.error || new Error(\"Websocket connection error\")\n        reject(error)\n      }\n\n      this.socket.addEventListener(\"open\", onOpen)\n      this.socket.addEventListener(\"error\", onError)\n      this.socket.addEventListener(\"message\", this.onMessage)\n      this.socket.addEventListener(\"close\", this.onClose)\n    })\n\n    await this.connectPromise\n\n    // Cold restore from external persistence (sessionStore) on the\n    // very first connect: apps wire this up to survive a full page\n    // reload. After the first restore attempt the in-memory cache\n    // takes over.\n    if (!this._sessionId && !this._sessionStoreRestored && this._sessionStore) {\n      this._sessionStoreRestored = true\n\n      try {\n        const storedId = await this._sessionStore.get()\n\n        if (typeof storedId === \"string\" && storedId.length > 0) {\n          this._sessionId = storedId\n        }\n      } catch (error) {\n        this._debug(\"sessionStore.get failed\", error)\n      }\n    }\n\n    // If we have a cached sessionId from a prior connect, ask the\n    // server to resume it. The server replies with either\n    // `session-resumed` (state preserved) or `session-gone` (client\n    // must start fresh); the message dispatcher fires the appropriate\n    // lifecycle hooks on live Connection / Channel handles.\n    if (this._sessionId) {\n      this._awaitingResume = true\n      this._sendMessage({type: \"session-resume\", sessionId: this._sessionId})\n      // Fire onDisconnect on live handles so apps can pause UI work\n      // until session-resumed / session-gone arrives.\n      for (const connection of this._connections.values()) connection._handleDisconnected()\n      for (const subscription of this._channelSubscriptions.values()) subscription._handleDisconnected()\n    }\n\n    if (Object.keys(this._metadata).length > 0) {\n      this._sendMessage({type: \"metadata\", data: {...this._metadata}})\n    }\n\n    if (!this._awaitingResume) {\n      this._sendPendingChannelSubscriptions()\n    }\n    this.disconnectedSince = null\n  }\n\n  /**\n   * Close the websocket and clear pending state.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async close() {\n    this.autoReconnect = false\n    this._waitingForOnline = false\n    this._cancelPendingReconnect()\n    this._teardownNetworkMonitorSubscription()\n\n    if (!this.socket) return\n\n    if (this.socket.readyState === this.socket.CLOSED) {\n      this.socket = undefined\n      this.connectPromise = undefined\n      return\n    }\n\n    await new Promise((resolve) => {\n      this.socket?.addEventListener(\"close\", () => resolve(undefined))\n      this.socket?.close()\n    })\n\n    this.socket = undefined\n    this.connectPromise = undefined\n  }\n\n  /**\n   * Disable auto-reconnect and close the websocket.\n   * @returns {Promise<void>} - Resolves when closed.\n   */\n  async disconnectAndStopReconnect() {\n    await this.close()\n  }\n\n  /**\n   * Close the raw socket without disabling auto-reconnect. Used by tests to\n   * simulate an unexpected network drop and verify reconnection behavior.\n   * @returns {Promise<void>} - Resolves when the socket has closed.\n   */\n  async dropConnection() {\n    if (!this.socket) return\n\n    await new Promise((resolve) => {\n      this.socket?.addEventListener(\"close\", () => resolve(undefined))\n      this.socket?.close()\n    })\n\n    this.connectPromise = undefined\n  }\n\n  /**\n   * Perform a POST request over the websocket.\n   * @param {string} path - Path.\n   * @param {any} [body] - Request body.\n   * @param {{headers?: Record<string, string>}} [options] - Request options such as headers.\n   * @returns {Promise<VelociousWebsocketResponse>} - Resolves with the post.\n   */\n  async post(path, body, options = {}) {\n    return await this.request(\"POST\", path, {...options, body})\n  }\n\n  /**\n   * Perform a GET request over the websocket.\n   * @param {string} path - Path.\n   * @param {{headers?: Record<string, string>}} [options] - Request options such as headers.\n   * @returns {Promise<VelociousWebsocketResponse>} - Resolves with the get.\n   */\n  async get(path, options = {}) {\n    return await this.request(\"GET\", path, options)\n  }\n\n  /**\n   * Subscribe to a channel for server-sent events.\n   * @param {string} channel - Channel name.\n   * @param {(payload: any) => void} callback - Callback function.\n   * @returns {() => void} unsubscribe function\n   */\n  on(channel, callback) {\n    return this.subscribe(channel, {}, callback)\n  }\n\n  /**\n   * Returns a snapshot of the client's connection state.\n   * @returns {{disconnectedSince: number | null, isOpen: boolean, listenerCount: number}}\n   */\n  state() {\n    return {\n      disconnectedSince: this.disconnectedSince,\n      isOpen: !!this.socket && this.socket.readyState === this.socket.OPEN,\n      listenerCount: this.listeners.size + this._channelSubscriptions.size\n    }\n  }\n\n  /**\n   * Subscribe to a channel for server-sent events with optional params.\n   * @param {string} channel - Channel name.\n   * @param {{lastEventId?: string, params?: Record<string, any>}} options - Subscription options.\n   * @param {(payload: any, message?: Record<string, any>) => void} callback - Callback function.\n   * @returns {(() => void) & {ready: Promise<void>}} - Unsubscribe function with readiness promise.\n   */\n  subscribe(channel, options, callback) {\n    const params = options?.params\n    const lastEventId = options?.lastEventId\n    const subscriptionKey = this._subscriptionKey(channel, params)\n\n    if (!this.listeners.has(subscriptionKey)) {\n      /** @type {(() => void) | undefined} */\n      /** @type {((value?: void) => void) | undefined} */\n      let resolveReady\n      /** @type {((error: unknown) => void) | undefined} */\n      let rejectReady\n      const ready = new Promise((resolve, reject) => {\n        resolveReady = resolve\n        rejectReady = reject\n      })\n\n      this.listeners.set(subscriptionKey, {\n        callbacks: new Set(),\n        channel,\n        params,\n        ready\n      })\n      this.pendingSubscriptions.set(subscriptionKey, {\n        reject: rejectReady || (() => {}),\n        resolve: resolveReady || (() => {})\n      })\n\n      void this.connect().then(() => {\n        this._sendMessage({channel, lastEventId, params, type: \"subscribe\"})\n      }).catch((error) => this._debug(\"Subscribe failed\", error))\n    }\n\n    const listenerEntry = this.listeners.get(subscriptionKey)\n\n    if (!listenerEntry) throw new Error(\"Listeners map not initialized\")\n\n    listenerEntry.callbacks.add(callback)\n\n    const unsubscribe = () => {\n      listenerEntry.callbacks.delete(callback)\n\n      if (listenerEntry.callbacks.size === 0) {\n        this.listeners.delete(subscriptionKey)\n      }\n    }\n\n    unsubscribe.ready = listenerEntry.ready\n\n    return unsubscribe\n  }\n\n  /**\n   * Subscribe to a channel and wait until the server acknowledges the subscription.\n   * @param {string} channel - Channel name.\n   * @param {{lastEventId?: string, params?: Record<string, any>}} options - Subscription options.\n   * @param {(payload: any, message?: Record<string, any>) => void} callback - Callback function.\n   * @returns {Promise<(() => void) & {ready: Promise<void>}>} - Ready unsubscribe handle.\n   */\n  async subscribeAndWait(channel, options, callback) {\n    const unsubscribe = this.subscribe(channel, options, callback)\n\n    await unsubscribe.ready\n\n    return unsubscribe\n  }\n\n  /**\n   * @private\n   * @param {string} method - HTTP method.\n   * @param {string} path - Path.\n   * @param {object} [options] - Options object.\n   * @param {any} [options.body] - Request body.\n   * @param {Record<string, string>} [options.headers] - Header list.\n   * @returns {Promise<VelociousWebsocketResponse>} - Resolves with the request.\n   */\n  async request(method, path, {body, headers} = {}) {\n    await this.connect()\n\n    const id = `ws-${this.nextID++}`\n    const payload = {\n      body,\n      headers,\n      id,\n      method,\n      path,\n      type: \"request\"\n    }\n\n    return await new Promise((resolve, reject) => {\n      this.pendingRequests.set(id, {resolve, reject})\n      this._sendMessage(payload)\n    })\n  }\n\n  /**\n   * @private\n   * @param {MessageEvent<any>} event - Event payload.\n   */\n  onMessage = (event) => {\n    const raw = typeof event.data === \"string\" ? event.data : event.data?.toString?.()\n\n    if (!raw) return\n\n    /** @type {Record<string, any>} */\n    let message\n\n    try {\n      message = JSON.parse(raw)\n    } catch (error) {\n      this._debug(\"Failed to parse websocket message\", error)\n      return\n    }\n\n    const {type} = message\n\n    if (type === \"response\") {\n      const {id} = message\n      const pending = id ? this.pendingRequests.get(id) : undefined\n\n      if (pending) {\n        this.pendingRequests.delete(id)\n        pending.resolve(new VelociousWebsocketResponse(message))\n      } else {\n        this._debug(`No pending request for response id ${id}`)\n      }\n    } else if (type === \"subscribed\") {\n      const subscriptionKey = this._subscriptionKey(message.channel, message.params)\n      const pendingSubscription = this.pendingSubscriptions.get(subscriptionKey)\n\n      if (pendingSubscription) {\n        this.pendingSubscriptions.delete(subscriptionKey)\n        pendingSubscription.resolve()\n      }\n    } else if (type === \"event\") {\n      const {channel, payload} = message\n      for (const listenerEntry of this.listeners.values()) {\n        if (listenerEntry.channel !== channel) continue\n\n        listenerEntry.callbacks.forEach((/** @type {(payload: any, message?: Record<string, any>) => void} */ callback) => {\n          try {\n            callback(payload, message)\n          } catch (error) {\n            this._debug(\"Listener error\", error)\n          }\n        })\n      }\n    } else if (type === \"replay-gap\") {\n      const subscriptionKey = this._subscriptionKey(message.channel, message.params)\n      const pendingSubscription = this.pendingSubscriptions.get(subscriptionKey)\n\n      if (pendingSubscription) {\n        this.pendingSubscriptions.delete(subscriptionKey)\n        pendingSubscription.reject(new Error(`Replay gap for ${message.channel}`))\n      }\n    } else if (type === \"connection-opened\") {\n      const connection = this._connections.get(message.connectionId)\n\n      connection?._handleOpened()\n    } else if (type === \"connection-message\") {\n      const connection = this._connections.get(message.connectionId)\n\n      connection?._handleMessage(message.body)\n    } else if (type === \"connection-closed\") {\n      const connection = this._connections.get(message.connectionId)\n\n      if (connection) {\n        this._connections.delete(message.connectionId)\n        connection._handleClosed(message.reason || \"server_close\")\n      }\n    } else if (type === \"connection-error\") {\n      const connection = this._connections.get(message.connectionId)\n\n      if (connection) {\n        this._connections.delete(message.connectionId)\n        connection._handleClosed(`error: ${message.message || \"connection-error\"}`)\n      }\n    } else if (type === \"channel-subscribed\") {\n      const sub = this._channelSubscriptions.get(message.subscriptionId)\n\n      sub?._handleSubscribed()\n    } else if (type === \"channel-message\") {\n      const sub = this._channelSubscriptions.get(message.subscriptionId)\n\n      sub?._handleMessage(message.body)\n    } else if (type === \"channel-unsubscribed\") {\n      const sub = this._channelSubscriptions.get(message.subscriptionId)\n\n      if (sub) {\n        this._channelSubscriptions.delete(message.subscriptionId)\n        sub._handleClosed(\"server_unsubscribe\")\n      }\n    } else if (type === \"channel-error\") {\n      const sub = this._channelSubscriptions.get(message.subscriptionId)\n\n      if (sub) {\n        this._channelSubscriptions.delete(message.subscriptionId)\n        sub._handleClosed(`error: ${message.message || \"channel-error\"}`)\n      }\n    } else if (type === \"session-established\") {\n      // First connect: cache sessionId for future resume attempts.\n      if (!this._awaitingResume) {\n        this._sessionId = message.sessionId\n        this._persistSessionId(message.sessionId)\n      }\n    } else if (type === \"session-resumed\") {\n      this._awaitingResume = false\n      this._sessionId = message.sessionId\n      this._persistSessionId(message.sessionId)\n      this._sendPendingChannelSubscriptions()\n      // Fire onResume on every live handle so user code knows the\n      // session came back with state intact.\n      for (const connection of this._connections.values()) connection._handleResumed()\n      for (const subscription of this._channelSubscriptions.values()) subscription._handleResumed()\n    } else if (type === \"session-gone\") {\n      this._awaitingResume = false\n      this._sessionId = null\n      this._clearPersistedSessionId()\n      // Tear down every live handle — their server-side counterparts\n      // are gone and nothing can bring them back.\n      const connections = [...this._connections.values()]\n      this._connections.clear()\n      for (const connection of connections) connection._handleClosed(\"session_gone\")\n\n      const subs = [...this._channelSubscriptions.values()]\n      this._channelSubscriptions.clear()\n      for (const subscription of subs) subscription._handleClosed(\"session_gone\")\n    } else if (type === \"error\" && message.id) {\n      const pending = this.pendingRequests.get(message.id)\n\n      if (pending) {\n        this.pendingRequests.delete(message.id)\n        pending.reject(new Error(message.error || \"Unknown websocket error\"))\n      }\n    }\n  }\n\n  /**\n   * @private\n   * @param {string} channel - Channel name.\n   * @param {Record<string, any> | undefined} params - Subscription params.\n   * @returns {string} - Stable subscription key.\n   */\n  _subscriptionKey(channel, params) {\n    return JSON.stringify([channel, params || null])\n  }\n\n  /**\n   * Reject all pending requests when the socket closes. Schedules reconnect if enabled.\n   * @private\n   */\n  onClose = () => {\n    this.disconnectedSince ||= Date.now()\n\n    for (const [id, {reject}] of this.pendingRequests.entries()) {\n      reject(new Error(`Websocket closed before response for ${id}`))\n    }\n\n    for (const {reject} of this.pendingSubscriptions.values()) {\n      reject(new Error(\"Websocket closed before subscription acknowledgement\"))\n    }\n\n    if (this._sessionId && this.autoReconnect) {\n      // Session may resume when we reconnect — keep the handles alive\n      // and fire onDisconnect so user code can pause UI work.\n      for (const connection of this._connections.values()) connection._handleDisconnected()\n      for (const subscription of this._channelSubscriptions.values()) subscription._handleDisconnected()\n    } else {\n      // No resume path: tear down every live Connection / Channel sub.\n      const connections = [...this._connections.values()]\n      this._connections.clear()\n      for (const connection of connections) {\n        connection._handleClosed(\"session_destroyed\")\n      }\n\n      const channelSubs = [...this._channelSubscriptions.values()]\n      this._channelSubscriptions.clear()\n      for (const subscription of channelSubs) {\n        subscription._handleClosed(\"session_destroyed\")\n      }\n\n      this._sessionId = null\n    }\n\n    this.pendingRequests.clear()\n    this.pendingSubscriptions.clear()\n    this.connectPromise = undefined\n\n    if (!this.autoReconnect) return\n\n    void this._shouldWaitForOnline().then((shouldWaitForOnline) => {\n      if (!shouldWaitForOnline) {\n        this._scheduleReconnect()\n      }\n    })\n  }\n\n  /**\n   * @param {Record<string, any>} payload - Payload data.\n   * @returns {void}\n   */\n  _sendMessage(payload) {\n    if (!this.socket || this.socket.readyState !== this.socket.OPEN) {\n      throw new Error(\"Websocket is not open\")\n    }\n\n    const json = JSON.stringify(payload)\n\n    this._debug(\"Sending\", json)\n    this.socket.send(json)\n  }\n\n  /** @returns {void} */\n  _cancelPendingReconnect() {\n    if (this.reconnectTimer) {\n      globalThis.clearTimeout(this.reconnectTimer)\n      this.reconnectTimer = null\n    }\n  }\n\n  /** @returns {void} */\n  _scheduleReconnect() {\n    this._cancelPendingReconnect()\n\n    const delay = this.reconnectDelays[Math.min(this.reconnectAttempt, this.reconnectDelays.length - 1)]\n\n    this.reconnectTimer = globalThis.setTimeout(() => {\n      this.reconnectTimer = null\n      void this._attemptReconnect()\n    }, delay)\n\n    this.reconnectAttempt += 1\n  }\n\n  /** @returns {Promise<void>} */\n  async _attemptReconnect() {\n    if (!this.autoReconnect) return\n\n    if (!await this._isOnline()) {\n      this._waitingForOnline = true\n      return\n    }\n\n    try {\n      this._waitingForOnline = false\n      this.connectionAttempts += 1\n      await this.connect({autoReconnect: this.autoReconnect, resetReconnectState: false, waitForOnline: false})\n      this.reconnectAttempt = 0\n      this.disconnectedSince = null\n      this._resubscribeActiveListeners()\n\n      if (typeof this.onReconnect === \"function\") {\n        await this.onReconnect()\n      }\n    } catch (error) {\n      this._debug(\"Reconnect attempt failed:\", error)\n\n      if (this.autoReconnect) {\n        this._scheduleReconnect()\n      }\n    }\n  }\n\n  /**\n   * Re-sends subscribe messages for all active listeners after reconnection.\n   * @returns {void}\n   */\n  _resubscribeActiveListeners() {\n    for (const [, listenerEntry] of this.listeners) {\n      try {\n        this._sendMessage({\n          channel: listenerEntry.channel,\n          params: listenerEntry.params,\n          type: \"subscribe\"\n        })\n      } catch (error) {\n        this._debug(\"Re-subscribe failed:\", error)\n      }\n    }\n  }\n\n  /**\n   * @private\n   * @param  {...any} args - Options object.\n   * @returns {void} - No return value.\n   */\n  _debug(...args) {\n    if (!this.debug) return\n\n    console.debug(\"[VelociousWebsocketClient]\", ...args)\n  }\n\n  /**\n   * @private\n   * @param {string} sessionId - Id to persist through the configured sessionStore.\n   * @returns {void}\n   */\n  _persistSessionId(sessionId) {\n    if (!this._sessionStore) return\n\n    try {\n      const result = this._sessionStore.set(sessionId)\n\n      if (result && typeof result.then === \"function\") {\n        result.catch((/** @type {unknown} */ error) => this._debug(\"sessionStore.set failed\", error))\n      }\n    } catch (error) {\n      this._debug(\"sessionStore.set failed\", error)\n    }\n  }\n\n  /**\n   * @private\n   * @returns {void}\n   */\n  _clearPersistedSessionId() {\n    if (!this._sessionStore) return\n\n    try {\n      const result = this._sessionStore.clear()\n\n      if (result && typeof result.then === \"function\") {\n        result.catch((/** @type {unknown} */ error) => this._debug(\"sessionStore.clear failed\", error))\n      }\n    } catch (error) {\n      this._debug(\"sessionStore.clear failed\", error)\n    }\n  }\n}\n\nclass VelociousWebsocketResponse {\n  /**\n   * @param {object} message - Message text.\n   */\n  constructor(message) {\n    const responseMessage = /** @type {{body?: any, headers?: Record<string, any>, id?: string | number | null, statusCode?: number, statusMessage?: string, type?: string}} */ (message)\n\n    this.body = responseMessage.body\n    this.headers = responseMessage.headers || {}\n    this.id = responseMessage.id\n    this.statusCode = responseMessage.statusCode || 200\n    this.statusMessage = responseMessage.statusMessage || \"OK\"\n    this.type = responseMessage.type\n  }\n\n  /** @returns {any} - The json.  */\n  json() {\n    if (typeof this.body !== \"string\") {\n      throw new Error(\"Response body is not a string\")\n    }\n\n    return deserializeFrontendModelTransportValue(JSON.parse(this.body))\n  }\n}\n"]}
|
|
938
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"websocket-client.js","sourceRoot":"","sources":["../../../src/http-client/websocket-client.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport VelociousWebsocketClientConnection from \"./websocket-connection.js\"\nimport VelociousWebsocketClientSubscription from \"./websocket-channel.js\"\nimport {deserializeFrontendModelTransportValue} from \"../frontend-models/transport-serialization.js\"\n\nconst DEFAULT_RECONNECT_DELAYS = [1000, 2000, 4000, 8000, 15000]\n\n/**\n * A small websocket client that mirrors simple HTTP-style calls and channel subscriptions.\n * Supports optional auto-reconnect with exponential backoff and listener re-subscription.\n */\nexport default class VelociousWebsocketClient {\n  /** @type {Map<string, {reject: (error: unknown) => void, resolve: (response: VelociousWebsocketResponse) => void}>} */\n  pendingRequests\n  /** @type {Map<string, {reject: (error: unknown) => void, resolve: (value?: void) => void}>} */\n  pendingSubscriptions\n  /** @type {Map<string, {callbacks: Set<(payload: any) => void>, channel: string, params: Record<string, any> | undefined, ready: Promise<void>}>} */\n  listeners\n\n  /**\n   * @param {object} [args] - Options object.\n   * @param {boolean} [args.autoReconnect] - Enable auto-reconnect with exponential backoff.\n   * @param {boolean} [args.debug] - Whether debug.\n   * @param {{getIsOnline?: () => boolean | Promise<boolean>, subscribe?: (callback: (isOnline: boolean) => void) => (() => void) | {remove: () => void}}} [args.networkMonitor] - Optional online-state adapter. When provided, auto-reconnect can wait for the network to report online before reconnecting, and open sockets are closed when the monitor reports offline.\n   * @param {number[]} [args.reconnectDelays] - Backoff delays in ms (default: [1000, 2000, 4000, 8000, 15000]).\n   * @param {{get: () => string | null | undefined | Promise<string | null | undefined>, set: (sessionId: string) => void | Promise<void>, clear: () => void | Promise<void>}} [args.sessionStore] - Optional sessionId persistence hook. When provided, the client writes every `session-established` / `session-resumed` id via `store.set(id)` and clears it on `session-gone`. Before the first `connect()`, the client reads any persisted id via `store.get()` and attempts resumption. Apps should back this by whatever persistence layer survives page reloads (localStorage, a cookie, SQLite, etc.).\n   * @param {string} [args.url] Full websocket URL (default: ws://127.0.0.1:3006/websocket)\n   */\n  constructor({autoReconnect = true, debug = false, networkMonitor, reconnectDelays, sessionStore, url} = {}) {\n    if (!globalThis.WebSocket) throw new Error(\"WebSocket global is not available\")\n\n    /** @type {boolean} */\n    this.autoReconnect = autoReconnect\n    this.debug = debug\n    /** @type {number | null} */\n    this.disconnectedSince = null\n    this.pendingRequests = new Map()\n    this.pendingSubscriptions = new Map()\n    /** @type {number} */\n    this.reconnectAttempt = 0\n    /** @type {number} */\n    this.connectionAttempts = 0\n    /** @type {number[]} */\n    this.reconnectDelays = reconnectDelays || DEFAULT_RECONNECT_DELAYS\n    /** @type {ReturnType<typeof setTimeout> | null} */\n    this.reconnectTimer = null\n    this.url = url || \"ws://127.0.0.1:3006/websocket\"\n    this.listeners = new Map()\n    this.nextID = 1\n    /** @type {(() => void | Promise<void>) | null} */\n    this.onReconnect = null\n\n    /** @type {Record<string, any>} */\n    this._metadata = {}\n\n    /** @type {Map<string, VelociousWebsocketClientConnection>} */\n    this._connections = new Map()\n\n    /** @type {Map<string, VelociousWebsocketClientSubscription>} */\n    this._channelSubscriptions = new Map()\n\n    this._nextConnectionIdSeq = 1\n    this._nextSubscriptionIdSeq = 1\n\n    /** @type {string | null} - sessionId received from `session-established`; sent on reconnect for resumption. */\n    this._sessionId = null\n\n    /** @type {boolean} - true between a reconnect and the session-resumed / session-gone reply. */\n    this._awaitingResume = false\n\n    /** @type {boolean} - true once the current socket has an active session ready for app messages. */\n    this._sessionReady = false\n\n    /** @type {string | null} - provisional session id announced before a resume attempt finishes. */\n    this._pendingSessionId = null\n\n    /** @type {Promise<void> | null} */\n    this._sessionReadyPromise = null\n\n    /** @type {(() => void) | null} */\n    this._resolveSessionReady = null\n\n    /** @type {unknown | null} */\n    this._sessionReadyError = null\n\n    /** @type {{get: () => string | null | undefined | Promise<string | null | undefined>, set: (sessionId: string) => void | Promise<void>, clear: () => void | Promise<void>} | undefined} */\n    this._sessionStore = sessionStore\n    /** @type {boolean} - true once the sessionStore has been consulted for a restored id. */\n    this._sessionStoreRestored = false\n\n    /** @type {{getIsOnline?: () => boolean | Promise<boolean>, subscribe?: (callback: (isOnline: boolean) => void) => (() => void) | {remove: () => void}} | undefined} */\n    this._networkMonitor = networkMonitor\n\n    /** @type {null | (() => void) | {remove: () => void}} */\n    this._networkMonitorSubscription = null\n\n    /** @type {boolean} */\n    this._waitingForOnline = false\n  }\n\n  /** @returns {boolean} */\n  isOpen() {\n    return Boolean(this.socket && this.socket.readyState === this.socket.OPEN)\n  }\n\n  /** @returns {boolean} */\n  isSessionReady() {\n    return this._sessionReady\n  }\n\n  /**\n   * Opens a 1:1 `WebsocketConnection` of the given type against the\n   * server. Requires the socket to already be connected (call\n   * `connect()` first).\n   *\n   * @param {string} connectionType - Name the server registered the class under.\n   * @param {{params?: Record<string, any>, onConnect?: () => void, onMessage?: (body: any) => void, onDisconnect?: () => void, onResume?: () => void, onClose?: (reason: string) => void}} [options]\n   * @returns {VelociousWebsocketClientConnection}\n   */\n  openConnection(connectionType, options = {}) {\n    if (!this.isOpen()) throw new Error(\"Websocket is not open; call connect() first\")\n\n    const connectionId = `c${this._nextConnectionIdSeq++}`\n    const connection = new VelociousWebsocketClientConnection({\n      client: this,\n      connectionId,\n      connectionType,\n      params: options.params,\n      onConnect: options.onConnect,\n      onMessage: options.onMessage,\n      onDisconnect: options.onDisconnect,\n      onResume: options.onResume,\n      onClose: options.onClose\n    })\n\n    this._connections.set(connectionId, connection)\n    this._sendMessage({\n      type: \"connection-open\",\n      connectionId,\n      connectionType,\n      params: options.params || {}\n    })\n\n    return connection\n  }\n\n  /**\n   * Drops a connection handle from the registry. Called by\n   * `VelociousWebsocketClientConnection.close()` after it notifies\n   * the server, and by the session-destroyed cleanup path.\n   *\n   * @param {string} connectionId\n   * @returns {void}\n   */\n  _removeConnection(connectionId) {\n    this._connections.delete(connectionId)\n  }\n\n  /**\n   * Subscribes to a named WebsocketChannel. If the socket is not yet\n   * open, the subscription is queued and sent once a connection is\n   * established.\n   *\n   * @param {string} channelType - Name the server registered the channel under.\n   * @param {{params?: Record<string, any>, lastEventId?: string, onMessage?: (body: any) => void, onDisconnect?: () => void, onResume?: () => void, onClose?: (reason: string) => void}} [options]\n   * @returns {VelociousWebsocketClientSubscription}\n   */\n  subscribeChannel(channelType, options = {}) {\n    const subscriptionId = `s${this._nextSubscriptionIdSeq++}`\n    const subscription = new VelociousWebsocketClientSubscription({\n      client: this,\n      subscriptionId,\n      channelType,\n      lastEventId: options.lastEventId,\n      params: options.params,\n      onMessage: options.onMessage,\n      onDisconnect: options.onDisconnect,\n      onResume: options.onResume,\n      onClose: options.onClose\n    })\n\n    this._channelSubscriptions.set(subscriptionId, subscription)\n    this._sendChannelSubscribe(subscription)\n\n    return subscription\n  }\n\n  /**\n   * @param {string} subscriptionId\n   * @returns {void}\n   */\n  _removeChannelSubscription(subscriptionId) {\n    this._channelSubscriptions.delete(subscriptionId)\n  }\n\n  /**\n   * @param {VelociousWebsocketClientSubscription} subscription\n   * @returns {void}\n   */\n  _sendChannelSubscribe(subscription) {\n    if (!this.isOpen() || !this.isSessionReady() || !subscription._needsSubscribe()) return\n\n    // Send first and only mark as sent on success. If the socket\n    // closes between `isOpen()` and `send()` and `_sendMessage` throws,\n    // `_subscribeSent` must stay false so the reconnect path's\n    // `_sendPendingChannelSubscriptions()` can retry.\n    try {\n      this._sendMessage({\n        type: \"channel-subscribe\",\n        subscriptionId: subscription.subscriptionId,\n        channelType: subscription.channelType,\n        params: subscription.params,\n        ...(subscription.lastEventId ? {lastEventId: subscription.lastEventId} : {})\n      })\n      subscription._markSubscribeSent()\n    } catch (error) {\n      // Transient closed-socket race: leave the subscription\n      // retryable so `_sendPendingChannelSubscriptions()` can resend\n      // after reconnect. Rethrow so callers that expect throws on a\n      // dead socket still see them.\n      if (!this.isOpen()) throw error\n\n      // Non-recoverable send failure on an open socket (e.g.\n      // `JSON.stringify` failing on BigInt/cyclic params). Close the\n      // subscription and remove it from the registry so it cannot\n      // poison future `_sendPendingChannelSubscriptions()` loops by\n      // throwing on every attempt and aborting later subscriptions.\n      this._channelSubscriptions.delete(subscription.subscriptionId)\n      subscription._handleClosed(`send_failed: ${error instanceof Error ? error.message : String(error)}`)\n    }\n  }\n\n  /** @returns {void} */\n  _sendPendingChannelSubscriptions() {\n    for (const subscription of this._channelSubscriptions.values()) {\n      this._sendChannelSubscribe(subscription)\n    }\n  }\n\n  /** @returns {Promise<boolean>} */\n  async _isOnline() {\n    if (!this._networkMonitor?.getIsOnline) return true\n\n    try {\n      return await this._networkMonitor.getIsOnline() !== false\n    } catch (error) {\n      this._debug(\"networkMonitor.getIsOnline failed\", error)\n      return true\n    }\n  }\n\n  /** @returns {Promise<boolean>} */\n  async _shouldWaitForOnline() {\n    if (!this._networkMonitor) return false\n\n    const isOnline = await this._isOnline()\n    if (isOnline) return false\n\n    this._waitingForOnline = true\n    this._cancelPendingReconnect()\n    return true\n  }\n\n  /** @returns {void} */\n  _ensureNetworkMonitorSubscription() {\n    if (!this._networkMonitor?.subscribe || this._networkMonitorSubscription) return\n\n    this._networkMonitorSubscription = this._networkMonitor.subscribe((isOnline) => {\n      if (!this.autoReconnect) return\n\n      if (isOnline) {\n        if (!this._waitingForOnline) return\n\n        this._waitingForOnline = false\n        void this._attemptReconnect()\n        return\n      }\n\n      this._waitingForOnline = true\n      this._cancelPendingReconnect()\n\n      if (this.isOpen()) {\n        void this.dropConnection()\n      }\n    })\n  }\n\n  /** @returns {void} */\n  _teardownNetworkMonitorSubscription() {\n    if (!this._networkMonitorSubscription) return\n\n    if (typeof this._networkMonitorSubscription === \"function\") {\n      this._networkMonitorSubscription()\n    } else {\n      this._networkMonitorSubscription.remove()\n    }\n\n    this._networkMonitorSubscription = null\n  }\n\n  /**\n   * Sets a global metadata value that is sent to the server.\n   * For WebSocket connections, a metadata update message is sent immediately.\n   * @param {string} key - Metadata key.\n   * @param {any} value - Metadata value (null to clear).\n   * @returns {void}\n   */\n  setMetadata(key, value) {\n    if (value === null || value === undefined) {\n      delete this._metadata[key]\n    } else {\n      this._metadata[key] = value\n    }\n\n    if (this.socket && this.socket.readyState === this.socket.OPEN) {\n      this._sendMessage({type: \"metadata\", data: {...this._metadata}})\n    }\n  }\n\n  /** @returns {Record<string, any>} - Current metadata. */\n  getMetadata() {\n    return {...this._metadata}\n  }\n\n  /**\n   * Ensure a websocket connection is open.\n   * Auto-reconnect and online gating are enabled by default.\n   * Pass `autoReconnect: false` or `waitForOnline: false` only when a caller\n   * explicitly needs lower-level behavior.\n   * @param {{autoReconnect?: boolean, waitForOnline?: boolean, resetReconnectState?: boolean}} [options]\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async connect({autoReconnect = this.autoReconnect, waitForOnline = true, resetReconnectState = true} = {}) {\n    this.autoReconnect = autoReconnect\n\n    if (this.autoReconnect) {\n      this._ensureNetworkMonitorSubscription()\n    } else {\n      this._waitingForOnline = false\n      this._cancelPendingReconnect()\n      this._teardownNetworkMonitorSubscription()\n    }\n\n    if (waitForOnline && this.autoReconnect && !await this._isOnline()) {\n      this._waitingForOnline = true\n      return\n    }\n\n    if (resetReconnectState) {\n      this.reconnectAttempt = 0\n    }\n\n    if (this.socket && this.socket.readyState === this.socket.OPEN) return\n    if (this.connectPromise) return this.connectPromise\n\n    this._resetSessionReadyState()\n    this._waitingForOnline = false\n    this.connectionAttempts += 1\n\n    this.connectPromise = new Promise((resolve, reject) => {\n      this.socket = new WebSocket(this.url)\n\n      const cleanup = () => {\n        this.socket?.removeEventListener(\"open\", onOpen)\n        this.socket?.removeEventListener(\"error\", onError)\n      }\n\n      const onOpen = () => {\n        cleanup()\n        resolve(undefined)\n      }\n      const onError = (/** @type {Event & {error?: unknown}} */ event) => {\n        cleanup()\n        const error = event?.error || new Error(\"Websocket connection error\")\n        reject(error)\n      }\n\n      this.socket.addEventListener(\"open\", onOpen)\n      this.socket.addEventListener(\"error\", onError)\n      this.socket.addEventListener(\"message\", this.onMessage)\n      this.socket.addEventListener(\"close\", this.onClose)\n    })\n\n    await this.connectPromise\n\n    // Cold restore from external persistence (sessionStore) on the\n    // very first connect: apps wire this up to survive a full page\n    // reload. After the first restore attempt the in-memory cache\n    // takes over.\n    if (!this._sessionId && !this._sessionStoreRestored && this._sessionStore) {\n      this._sessionStoreRestored = true\n\n      try {\n        const storedId = await this._sessionStore.get()\n\n        if (typeof storedId === \"string\" && storedId.length > 0) {\n          this._sessionId = storedId\n        }\n      } catch (error) {\n        this._debug(\"sessionStore.get failed\", error)\n      }\n    }\n\n    // If we have a cached sessionId from a prior connect, ask the\n    // server to resume it. The server replies with either\n    // `session-resumed` (state preserved) or `session-gone` (client\n    // must start fresh); the message dispatcher fires the appropriate\n    // lifecycle hooks on live Connection / Channel handles.\n    if (this._sessionId) {\n      this._awaitingResume = true\n      this._sendMessage({type: \"session-resume\", sessionId: this._sessionId})\n      // Fire onDisconnect on live handles so apps can pause UI work\n      // until session-resumed / session-gone arrives.\n      for (const connection of this._connections.values()) connection._handleDisconnected()\n      for (const subscription of this._channelSubscriptions.values()) subscription._handleDisconnected()\n    }\n\n    if (Object.keys(this._metadata).length > 0) {\n      this._sendMessage({type: \"metadata\", data: {...this._metadata}})\n    }\n\n    await this._waitForSessionReady()\n    this.disconnectedSince = null\n  }\n\n  /**\n   * Close the websocket and clear pending state.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async close() {\n    this.autoReconnect = false\n    this._waitingForOnline = false\n    this._cancelPendingReconnect()\n    this._teardownNetworkMonitorSubscription()\n\n    if (!this.socket) return\n\n    if (this.socket.readyState === this.socket.CLOSED) {\n      this.socket = undefined\n      this.connectPromise = undefined\n      this._resetSessionReadyState()\n      return\n    }\n\n    await new Promise((resolve) => {\n      this.socket?.addEventListener(\"close\", () => resolve(undefined))\n      this.socket?.close()\n    })\n\n    this.socket = undefined\n    this.connectPromise = undefined\n    this._resetSessionReadyState()\n  }\n\n  /**\n   * Disable auto-reconnect and close the websocket.\n   * @returns {Promise<void>} - Resolves when closed.\n   */\n  async disconnectAndStopReconnect() {\n    await this.close()\n  }\n\n  /**\n   * Close the raw socket without disabling auto-reconnect. Used by tests to\n   * simulate an unexpected network drop and verify reconnection behavior.\n   * @returns {Promise<void>} - Resolves when the socket has closed.\n   */\n  async dropConnection() {\n    if (!this.socket) return\n\n    await new Promise((resolve) => {\n      this.socket?.addEventListener(\"close\", () => resolve(undefined))\n      this.socket?.close()\n    })\n\n    this.connectPromise = undefined\n    this._resetSessionReadyState()\n  }\n\n  /**\n   * Perform a POST request over the websocket.\n   * @param {string} path - Path.\n   * @param {any} [body] - Request body.\n   * @param {{headers?: Record<string, string>}} [options] - Request options such as headers.\n   * @returns {Promise<VelociousWebsocketResponse>} - Resolves with the post.\n   */\n  async post(path, body, options = {}) {\n    return await this.request(\"POST\", path, {...options, body})\n  }\n\n  /**\n   * Perform a GET request over the websocket.\n   * @param {string} path - Path.\n   * @param {{headers?: Record<string, string>}} [options] - Request options such as headers.\n   * @returns {Promise<VelociousWebsocketResponse>} - Resolves with the get.\n   */\n  async get(path, options = {}) {\n    return await this.request(\"GET\", path, options)\n  }\n\n  /**\n   * Subscribe to a channel for server-sent events.\n   * @param {string} channel - Channel name.\n   * @param {(payload: any) => void} callback - Callback function.\n   * @returns {() => void} unsubscribe function\n   */\n  on(channel, callback) {\n    return this.subscribe(channel, {}, callback)\n  }\n\n  /**\n   * Returns a snapshot of the client's connection state.\n   * @returns {{disconnectedSince: number | null, isOpen: boolean, listenerCount: number}}\n   */\n  state() {\n    return {\n      disconnectedSince: this.disconnectedSince,\n      isOpen: !!this.socket && this.socket.readyState === this.socket.OPEN,\n      listenerCount: this.listeners.size + this._channelSubscriptions.size\n    }\n  }\n\n  /**\n   * Subscribe to a channel for server-sent events with optional params.\n   * @param {string} channel - Channel name.\n   * @param {{lastEventId?: string, params?: Record<string, any>}} options - Subscription options.\n   * @param {(payload: any, message?: Record<string, any>) => void} callback - Callback function.\n   * @returns {(() => void) & {ready: Promise<void>}} - Unsubscribe function with readiness promise.\n   */\n  subscribe(channel, options, callback) {\n    const params = options?.params\n    const lastEventId = options?.lastEventId\n    const subscriptionKey = this._subscriptionKey(channel, params)\n\n    if (!this.listeners.has(subscriptionKey)) {\n      /** @type {(() => void) | undefined} */\n      /** @type {((value?: void) => void) | undefined} */\n      let resolveReady\n      /** @type {((error: unknown) => void) | undefined} */\n      let rejectReady\n      const ready = new Promise((resolve, reject) => {\n        resolveReady = resolve\n        rejectReady = reject\n      })\n\n      this.listeners.set(subscriptionKey, {\n        callbacks: new Set(),\n        channel,\n        params,\n        ready\n      })\n      this.pendingSubscriptions.set(subscriptionKey, {\n        reject: rejectReady || (() => {}),\n        resolve: resolveReady || (() => {})\n      })\n\n      void this.connect().then(() => {\n        this._sendMessage({channel, lastEventId, params, type: \"subscribe\"})\n      }).catch((error) => this._debug(\"Subscribe failed\", error))\n    }\n\n    const listenerEntry = this.listeners.get(subscriptionKey)\n\n    if (!listenerEntry) throw new Error(\"Listeners map not initialized\")\n\n    listenerEntry.callbacks.add(callback)\n\n    const unsubscribe = () => {\n      listenerEntry.callbacks.delete(callback)\n\n      if (listenerEntry.callbacks.size === 0) {\n        this.listeners.delete(subscriptionKey)\n      }\n    }\n\n    unsubscribe.ready = listenerEntry.ready\n\n    return unsubscribe\n  }\n\n  /**\n   * Subscribe to a channel and wait until the server acknowledges the subscription.\n   * @param {string} channel - Channel name.\n   * @param {{lastEventId?: string, params?: Record<string, any>}} options - Subscription options.\n   * @param {(payload: any, message?: Record<string, any>) => void} callback - Callback function.\n   * @returns {Promise<(() => void) & {ready: Promise<void>}>} - Ready unsubscribe handle.\n   */\n  async subscribeAndWait(channel, options, callback) {\n    const unsubscribe = this.subscribe(channel, options, callback)\n\n    await unsubscribe.ready\n\n    return unsubscribe\n  }\n\n  /**\n   * @private\n   * @param {string} method - HTTP method.\n   * @param {string} path - Path.\n   * @param {object} [options] - Options object.\n   * @param {any} [options.body] - Request body.\n   * @param {Record<string, string>} [options.headers] - Header list.\n   * @returns {Promise<VelociousWebsocketResponse>} - Resolves with the request.\n   */\n  async request(method, path, {body, headers} = {}) {\n    await this.connect()\n\n    const id = `ws-${this.nextID++}`\n    const payload = {\n      body,\n      headers,\n      id,\n      method,\n      path,\n      type: \"request\"\n    }\n\n    return await new Promise((resolve, reject) => {\n      this.pendingRequests.set(id, {resolve, reject})\n      this._sendMessage(payload)\n    })\n  }\n\n  /**\n   * @private\n   * @param {MessageEvent<any>} event - Event payload.\n   */\n  onMessage = (event) => {\n    const raw = typeof event.data === \"string\" ? event.data : event.data?.toString?.()\n\n    if (!raw) return\n\n    /** @type {Record<string, any>} */\n    let message\n\n    try {\n      message = JSON.parse(raw)\n    } catch (error) {\n      this._debug(\"Failed to parse websocket message\", error)\n      return\n    }\n\n    const {type} = message\n\n    if (type === \"response\") {\n      const {id} = message\n      const pending = id ? this.pendingRequests.get(id) : undefined\n\n      if (pending) {\n        this.pendingRequests.delete(id)\n        pending.resolve(new VelociousWebsocketResponse(message))\n      } else {\n        this._debug(`No pending request for response id ${id}`)\n      }\n    } else if (type === \"subscribed\") {\n      const subscriptionKey = this._subscriptionKey(message.channel, message.params)\n      const pendingSubscription = this.pendingSubscriptions.get(subscriptionKey)\n\n      if (pendingSubscription) {\n        this.pendingSubscriptions.delete(subscriptionKey)\n        pendingSubscription.resolve()\n      }\n    } else if (type === \"event\") {\n      const {channel, payload} = message\n      for (const listenerEntry of this.listeners.values()) {\n        if (listenerEntry.channel !== channel) continue\n\n        listenerEntry.callbacks.forEach((/** @type {(payload: any, message?: Record<string, any>) => void} */ callback) => {\n          try {\n            callback(payload, message)\n          } catch (error) {\n            this._debug(\"Listener error\", error)\n          }\n        })\n      }\n    } else if (type === \"replay-gap\") {\n      const subscriptionKey = this._subscriptionKey(message.channel, message.params)\n      const pendingSubscription = this.pendingSubscriptions.get(subscriptionKey)\n\n      if (pendingSubscription) {\n        this.pendingSubscriptions.delete(subscriptionKey)\n        pendingSubscription.reject(new Error(`Replay gap for ${message.channel}`))\n      }\n    } else if (type === \"connection-opened\") {\n      const connection = this._connections.get(message.connectionId)\n\n      connection?._handleOpened()\n    } else if (type === \"connection-message\") {\n      const connection = this._connections.get(message.connectionId)\n\n      connection?._handleMessage(message.body)\n    } else if (type === \"connection-closed\") {\n      const connection = this._connections.get(message.connectionId)\n\n      if (connection) {\n        this._connections.delete(message.connectionId)\n        connection._handleClosed(message.reason || \"server_close\")\n      }\n    } else if (type === \"connection-error\") {\n      const connection = this._connections.get(message.connectionId)\n\n      if (connection) {\n        this._connections.delete(message.connectionId)\n        connection._handleClosed(`error: ${message.message || \"connection-error\"}`)\n      }\n    } else if (type === \"channel-subscribed\") {\n      const sub = this._channelSubscriptions.get(message.subscriptionId)\n\n      sub?._handleSubscribed()\n    } else if (type === \"channel-message\") {\n      const sub = this._channelSubscriptions.get(message.subscriptionId)\n\n      sub?._handleMessage(message.body)\n    } else if (type === \"channel-unsubscribed\") {\n      const sub = this._channelSubscriptions.get(message.subscriptionId)\n\n      if (sub) {\n        this._channelSubscriptions.delete(message.subscriptionId)\n        sub._handleClosed(\"server_unsubscribe\")\n      }\n    } else if (type === \"channel-error\") {\n      const sub = this._channelSubscriptions.get(message.subscriptionId)\n\n      if (sub) {\n        this._channelSubscriptions.delete(message.subscriptionId)\n        sub._handleClosed(`error: ${message.message || \"channel-error\"}`)\n      }\n    } else if (type === \"session-established\") {\n      this._pendingSessionId = typeof message.sessionId === \"string\" ? message.sessionId : null\n\n      // First connect: cache sessionId for future resume attempts.\n      if (!this._awaitingResume) {\n        this._sessionId = this._pendingSessionId\n        if (this._sessionId) {\n          this._persistSessionId(this._sessionId)\n        }\n\n        this._markSessionReady()\n        this._sendPendingChannelSubscriptions()\n      }\n    } else if (type === \"session-resumed\") {\n      this._awaitingResume = false\n      this._pendingSessionId = null\n      this._sessionId = message.sessionId\n      this._persistSessionId(message.sessionId)\n      this._markSessionReady()\n      this._sendPendingChannelSubscriptions()\n      // Fire onResume on every live handle so user code knows the\n      // session came back with state intact.\n      for (const connection of this._connections.values()) connection._handleResumed()\n      for (const subscription of this._channelSubscriptions.values()) subscription._handleResumed()\n    } else if (type === \"session-gone\") {\n      this._awaitingResume = false\n      this._sessionId = null\n      this._pendingSessionId = null\n      this._clearPersistedSessionId()\n\n      // Tear down every live handle — their server-side counterparts\n      // are gone and nothing can bring them back.\n      const connections = [...this._connections.values()]\n      this._connections.clear()\n      for (const connection of connections) connection._handleClosed(\"session_gone\")\n\n      const subs = [...this._channelSubscriptions.values()]\n      this._channelSubscriptions.clear()\n      for (const subscription of subs) subscription._handleClosed(\"session_gone\")\n\n      this._markSessionReady()\n    } else if (type === \"error\" && message.id) {\n      const pending = this.pendingRequests.get(message.id)\n\n      if (pending) {\n        this.pendingRequests.delete(message.id)\n        pending.reject(new Error(message.error || \"Unknown websocket error\"))\n      }\n    }\n  }\n\n  /**\n   * @private\n   * @param {string} channel - Channel name.\n   * @param {Record<string, any> | undefined} params - Subscription params.\n   * @returns {string} - Stable subscription key.\n   */\n  _subscriptionKey(channel, params) {\n    return JSON.stringify([channel, params || null])\n  }\n\n  /**\n   * Reject all pending requests when the socket closes. Schedules reconnect if enabled.\n   * @private\n   */\n  onClose = () => {\n    this.disconnectedSince ||= Date.now()\n    this._resetSessionReadyState()\n\n    for (const [id, {reject}] of this.pendingRequests.entries()) {\n      reject(new Error(`Websocket closed before response for ${id}`))\n    }\n\n    for (const {reject} of this.pendingSubscriptions.values()) {\n      reject(new Error(\"Websocket closed before subscription acknowledgement\"))\n    }\n\n    if (this._sessionId && this.autoReconnect) {\n      // Session may resume when we reconnect — keep the handles alive\n      // and fire onDisconnect so user code can pause UI work.\n      for (const connection of this._connections.values()) connection._handleDisconnected()\n      for (const subscription of this._channelSubscriptions.values()) subscription._handleDisconnected()\n    } else {\n      // No resume path: tear down every live Connection / Channel sub.\n      const connections = [...this._connections.values()]\n      this._connections.clear()\n      for (const connection of connections) {\n        connection._handleClosed(\"session_destroyed\")\n      }\n\n      const channelSubs = [...this._channelSubscriptions.values()]\n      this._channelSubscriptions.clear()\n      for (const subscription of channelSubs) {\n        subscription._handleClosed(\"session_destroyed\")\n      }\n\n      this._sessionId = null\n    }\n\n    this.pendingRequests.clear()\n    this.pendingSubscriptions.clear()\n    this.connectPromise = undefined\n\n    if (!this.autoReconnect) return\n\n    void this._shouldWaitForOnline().then((shouldWaitForOnline) => {\n      if (!shouldWaitForOnline) {\n        this._scheduleReconnect()\n      }\n    })\n  }\n\n  /**\n   * @param {Record<string, any>} payload - Payload data.\n   * @returns {void}\n   */\n  _sendMessage(payload) {\n    if (!this.socket || this.socket.readyState !== this.socket.OPEN) {\n      throw new Error(\"Websocket is not open\")\n    }\n\n    const json = JSON.stringify(payload)\n\n    this._debug(\"Sending\", json)\n    this.socket.send(json)\n  }\n\n  /** @returns {void} */\n  _cancelPendingReconnect() {\n    if (this.reconnectTimer) {\n      globalThis.clearTimeout(this.reconnectTimer)\n      this.reconnectTimer = null\n    }\n  }\n\n  /** @returns {void} */\n  _scheduleReconnect() {\n    this._cancelPendingReconnect()\n\n    const delay = this.reconnectDelays[Math.min(this.reconnectAttempt, this.reconnectDelays.length - 1)]\n\n    this.reconnectTimer = globalThis.setTimeout(() => {\n      this.reconnectTimer = null\n      void this._attemptReconnect()\n    }, delay)\n\n    this.reconnectAttempt += 1\n  }\n\n  /** @returns {Promise<void>} */\n  async _attemptReconnect() {\n    if (!this.autoReconnect) return\n\n    if (!await this._isOnline()) {\n      this._waitingForOnline = true\n      return\n    }\n\n    try {\n      this._waitingForOnline = false\n      this.connectionAttempts += 1\n      await this.connect({autoReconnect: this.autoReconnect, resetReconnectState: false, waitForOnline: false})\n      this.reconnectAttempt = 0\n      this.disconnectedSince = null\n      this._resubscribeActiveListeners()\n\n      if (typeof this.onReconnect === \"function\") {\n        await this.onReconnect()\n      }\n    } catch (error) {\n      this._debug(\"Reconnect attempt failed:\", error)\n\n      if (this.autoReconnect) {\n        this._scheduleReconnect()\n      }\n    }\n  }\n\n  /**\n   * Re-sends subscribe messages for all active listeners after reconnection.\n   * @returns {void}\n   */\n  _resubscribeActiveListeners() {\n    for (const [, listenerEntry] of this.listeners) {\n      try {\n        this._sendMessage({\n          channel: listenerEntry.channel,\n          params: listenerEntry.params,\n          type: \"subscribe\"\n        })\n      } catch (error) {\n        this._debug(\"Re-subscribe failed:\", error)\n      }\n    }\n  }\n\n  /**\n   * @private\n   * @param  {...any} args - Options object.\n   * @returns {void} - No return value.\n   */\n  _debug(...args) {\n    if (!this.debug) return\n\n    console.debug(\"[VelociousWebsocketClient]\", ...args)\n  }\n\n  /**\n   * @private\n   * @param {string} sessionId - Id to persist through the configured sessionStore.\n   * @returns {void}\n   */\n  _persistSessionId(sessionId) {\n    if (!this._sessionStore) return\n\n    try {\n      const result = this._sessionStore.set(sessionId)\n\n      if (result && typeof result.then === \"function\") {\n        result.catch((/** @type {unknown} */ error) => this._debug(\"sessionStore.set failed\", error))\n      }\n    } catch (error) {\n      this._debug(\"sessionStore.set failed\", error)\n    }\n  }\n\n  /**\n   * @private\n   * @returns {void}\n   */\n  _clearPersistedSessionId() {\n    if (!this._sessionStore) return\n\n    try {\n      const result = this._sessionStore.clear()\n\n      if (result && typeof result.then === \"function\") {\n        result.catch((/** @type {unknown} */ error) => this._debug(\"sessionStore.clear failed\", error))\n      }\n    } catch (error) {\n      this._debug(\"sessionStore.clear failed\", error)\n    }\n  }\n\n  /** @returns {Promise<void>} */\n  _waitForSessionReady() {\n    if (this._sessionReady) return Promise.resolve()\n\n    if (!this._sessionReadyPromise || !this._resolveSessionReady) {\n      this._sessionReadyPromise = new Promise((resolve) => {\n        this._resolveSessionReady = resolve\n      })\n    }\n\n    return this._sessionReadyPromise.then(() => {\n      if (this._sessionReadyError) {\n        const error = this._sessionReadyError\n        this._sessionReadyError = null\n        throw error\n      }\n    })\n  }\n\n  /** @returns {void} */\n  _markSessionReady() {\n    if (this._sessionReady) return\n\n    this._sessionReady = true\n    this._sessionReadyError = null\n    this._resolveSessionReady?.()\n    this._resolveSessionReady = null\n    this._sessionReadyPromise = null\n  }\n\n  /**\n   * @param {unknown} [error]\n   * @returns {void}\n   */\n  _resetSessionReadyState(error = new Error(\"Websocket session readiness was reset\")) {\n    this._sessionReady = false\n    this._pendingSessionId = null\n    this._sessionReadyError = error\n    this._resolveSessionReady?.()\n    this._sessionReadyPromise = null\n    this._resolveSessionReady = null\n  }\n}\n\nclass VelociousWebsocketResponse {\n  /**\n   * @param {object} message - Message text.\n   */\n  constructor(message) {\n    const responseMessage = /** @type {{body?: any, headers?: Record<string, any>, id?: string | number | null, statusCode?: number, statusMessage?: string, type?: string}} */ (message)\n\n    this.body = responseMessage.body\n    this.headers = responseMessage.headers || {}\n    this.id = responseMessage.id\n    this.statusCode = responseMessage.statusCode || 200\n    this.statusMessage = responseMessage.statusMessage || \"OK\"\n    this.type = responseMessage.type\n  }\n\n  /** @returns {any} - The json.  */\n  json() {\n    if (typeof this.body !== \"string\") {\n      throw new Error(\"Response body is not a string\")\n    }\n\n    return deserializeFrontendModelTransportValue(JSON.parse(this.body))\n  }\n}\n"]}
|