velocious 1.0.308 → 1.0.309
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/application.d.ts.map +1 -1
- package/build/src/application.js +1 -1
- package/build/src/http-server/websocket-event-log-store.d.ts +2 -0
- package/build/src/http-server/websocket-event-log-store.d.ts.map +1 -1
- package/build/src/http-server/websocket-event-log-store.js +6 -1
- package/package.json +1 -1
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"application.d.ts","sourceRoot":"","sources":["../../src/application.js"],"names":[],"mappings":"AAQA;;;;;;GAMG;AAEH;IACE;;;;;OAKG;IACH,8DAJG;QAAmD,aAAa,EAAxD,OAAO,oBAAoB,EAAE,OAAO;QACL,UAAU;QAC5B,IAAI,EAAjB,MAAM;KAChB,EAaA;IAPC,oDAAkC;IAElC,sCAAsC;IACtC,yBADW,uBAAuB,CAC4B;IAE9D,eAA8B;IAC9B,cAAiB;IAGnB,qCAAqC;IACrC,WADc,MAAM,CACW;IAE/B,0DAA0D;IAC1D,cADc,OAAO,CAAC,IAAI,CAAC,
|
|
1
|
+
{"version":3,"file":"application.d.ts","sourceRoot":"","sources":["../../src/application.js"],"names":[],"mappings":"AAQA;;;;;;GAMG;AAEH;IACE;;;;;OAKG;IACH,8DAJG;QAAmD,aAAa,EAAxD,OAAO,oBAAoB,EAAE,OAAO;QACL,UAAU;QAC5B,IAAI,EAAjB,MAAM;KAChB,EAaA;IAPC,oDAAkC;IAElC,sCAAsC;IACtC,yBADW,uBAAuB,CAC4B;IAE9D,eAA8B;IAC9B,cAAiB;IAGnB,qCAAqC;IACrC,WADc,MAAM,CACW;IAE/B,0DAA0D;IAC1D,cADc,OAAO,CAAC,IAAI,CAAC,CAY1B;IAED,4CAA4C;IAC5C,YADc,OAAO,CAOpB;IAED;;;OAGG;IACH,cAHW,MAAa,IAAI,GACf,OAAO,CAAC,IAAI,CAAC,CAUzB;IAED,0DAA0D;IAC1D,mBADc,OAAO,CAAC,IAAI,CAAC,CAe1B;IAJC,mCAAqG;IAMvG,0DAA0D;IAC1D,QADc,OAAO,CAAC,IAAI,CAAC,CAK1B;IAED,0CAA0C;IAC1C,yBADc,IAAI,CAOjB;IAED,0DAA0D;IAC1D,QADc,OAAO,CAAC,IAAI,CAAC,CAK1B;IAFG,qEAA0B;CAG/B;;;;;;;;;;;;;;;;;;;mBAjHkB,aAAa;uBACT,wBAAwB"}
|
package/build/src/application.js
CHANGED
|
@@ -91,4 +91,4 @@ export default class VelociousApplication {
|
|
|
91
91
|
});
|
|
92
92
|
}
|
|
93
93
|
}
|
|
94
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
94
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"application.js","sourceRoot":"","sources":["../../src/application.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport AppRoutes from \"./routes/app-routes.js\"\nimport Logger from \"./logger.js\"\nimport HttpServer from \"./http-server/index.js\"\nimport websocketEventsHost from \"./http-server/websocket-events-host.js\"\nimport restArgsError from \"./utils/rest-args-error.js\"\n\n/**\n * @typedef {object} HttpServerConfiguration\n * @property {boolean} [inProcess] - Run HTTP handlers in the main thread instead of worker threads.\n * @property {number} [maxWorkers] - Max worker threads for the HTTP server.\n * @property {string} [host] - Hostname to bind the HTTP server to.\n * @property {number} [port] - Port to bind the HTTP server to.\n */\n\nexport default class VelociousApplication {\n  /**\n   * @param {object} args - Options object.\n   * @param {import(\"./configuration.js\").default} args.configuration - Configuration instance.\n   * @param {HttpServerConfiguration} [args.httpServer] - Http server.\n   * @param {string} args.type - Type identifier.\n   */\n  constructor({configuration, httpServer, type, ...restArgs}) {\n    restArgsError(restArgs)\n\n    if (!configuration) throw new Error(\"configuration is required\")\n\n    this.configuration = configuration\n\n    /** @type {HttpServerConfiguration} */\n    this.httpServerConfiguration = httpServer ?? {port: undefined}\n\n    this.logger = new Logger(this)\n    this._type = type\n  }\n\n  /** @returns {string} - The type.  */\n  getType() { return this._type }\n\n  /** @returns {Promise<void>} - Resolves when complete.  */\n  async initialize() {\n    const routes = await AppRoutes.getRoutes(this.configuration)\n\n    await this.configuration.initialize({type: this.getType()})\n\n    this.configuration.setRoutes(routes)\n\n    if (!this.configuration.isDatabasePoolInitialized()) {\n      await this.configuration.initializeDatabasePool()\n    }\n\n  }\n\n  /** @returns {boolean} - Whether active.  */\n  isActive() {\n    if (this.httpServer) {\n      return this.httpServer?.isActive()\n    }\n\n    return false\n  }\n\n  /**\n   * @param {function() : void} callback - Callback function.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async run(callback) {\n    await this.startHttpServer()\n\n    try {\n      await callback()\n    } finally {\n      this.stop()\n    }\n  }\n\n  /** @returns {Promise<void>} - Resolves when complete.  */\n  async startHttpServer() {\n    const {configuration, httpServerConfiguration} = this\n    const port = httpServerConfiguration.port || 3006\n\n    await this.logger.debug(`Starting server on port ${port}`)\n\n    if (!configuration.getWebsocketEvents()) {\n      configuration.setWebsocketEvents(/** @type {any} */ (websocketEventsHost))\n    }\n\n    this.httpServer = new HttpServer({configuration, inProcess: httpServerConfiguration.inProcess, port})\n    this.httpServer.events.on(\"close\", this.onHttpServerClose)\n\n    await this.httpServer.start()\n  }\n\n  /** @returns {Promise<void>} - Resolves when complete.  */\n  async stop() {\n    await this.logger.debug(\"Stopping server\")\n    await this.httpServer?.stop()\n    await this.configuration.closeDatabaseConnections()\n  }\n\n  /** @returns {void} - No return value.  */\n  onHttpServerClose = () => {\n    this.logger.debug(\"HTTP server closed\")\n\n    if (this.waitResolve) {\n      this.waitResolve()\n    }\n  }\n\n  /** @returns {Promise<void>} - Resolves when complete.  */\n  wait() {\n    return new Promise((resolve) => {\n      this.waitResolve = resolve\n    })\n  }\n}\n"]}
|
|
@@ -33,6 +33,8 @@ export default class VelociousHttpServerWebsocketEventLogStore {
|
|
|
33
33
|
logger: Logger;
|
|
34
34
|
_isReady: boolean;
|
|
35
35
|
_readyPromise: Promise<void> | null;
|
|
36
|
+
/** @type {Set<string>} */
|
|
37
|
+
_interestedChannels: Set<string>;
|
|
36
38
|
/**
|
|
37
39
|
* @returns {Promise<void>} - Resolves when ready.
|
|
38
40
|
*/
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"websocket-event-log-store.d.ts","sourceRoot":"","sources":["../../../src/http-server/websocket-event-log-store.js"],"names":[],"mappings":"AAWA;;;;;;;GAOG;AAEH;;;GAGG;AAEH;;;GAGG;AACH,sEAHW,OAAO,qBAAqB,EAAE,OAAO,GACnC,yCAAyC,CAWrD;AAED;IACE;;;;;OAKG;IACH,gEAJG;QAAoD,aAAa,EAAzD,OAAO,qBAAqB,EAAE,OAAO;QACvB,kBAAkB;QAClB,WAAW;KACnC,
|
|
1
|
+
{"version":3,"file":"websocket-event-log-store.d.ts","sourceRoot":"","sources":["../../../src/http-server/websocket-event-log-store.js"],"names":[],"mappings":"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"}
|
|
@@ -44,6 +44,8 @@ export default class VelociousHttpServerWebsocketEventLogStore {
|
|
|
44
44
|
this.logger = new Logger(this);
|
|
45
45
|
this._isReady = false;
|
|
46
46
|
this._readyPromise = null;
|
|
47
|
+
/** @type {Set<string>} */
|
|
48
|
+
this._interestedChannels = new Set();
|
|
47
49
|
}
|
|
48
50
|
/**
|
|
49
51
|
* @returns {Promise<void>} - Resolves when ready.
|
|
@@ -96,6 +98,7 @@ export default class VelociousHttpServerWebsocketEventLogStore {
|
|
|
96
98
|
*/
|
|
97
99
|
async markChannelInterested(channel) {
|
|
98
100
|
await this.ensureReady();
|
|
101
|
+
this._interestedChannels.add(channel);
|
|
99
102
|
const interestedUntil = new Date(Date.now() + this.retentionMs);
|
|
100
103
|
await this._withDb(async (db) => {
|
|
101
104
|
await this._upsertReplayChannelInterest(db, { channel, interestedUntil });
|
|
@@ -106,6 +109,8 @@ export default class VelociousHttpServerWebsocketEventLogStore {
|
|
|
106
109
|
* @returns {Promise<boolean>} - Whether the channel should be persisted for replay.
|
|
107
110
|
*/
|
|
108
111
|
async shouldPersistChannel(channel) {
|
|
112
|
+
if (this._interestedChannels.size === 0)
|
|
113
|
+
return false;
|
|
109
114
|
await this.ensureReady();
|
|
110
115
|
return await this._withDb(async (db) => {
|
|
111
116
|
const rows = await db
|
|
@@ -305,4 +310,4 @@ export default class VelociousHttpServerWebsocketEventLogStore {
|
|
|
305
310
|
});
|
|
306
311
|
}
|
|
307
312
|
}
|
|
308
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"websocket-event-log-store.js","sourceRoot":"","sources":["../../../src/http-server/websocket-event-log-store.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport {randomUUID} from \"crypto\"\nimport TableData from \"../database/table-data/index.js\"\nimport Logger from \"../logger.js\"\n\nconst EVENTS_TABLE = \"websocket_channel_events\"\nconst REPLAY_CHANNELS_TABLE = \"websocket_replay_channels\"\nconst DEFAULT_RETENTION_MS = 10 * 60 * 1000\nconst stores = new WeakMap()\n\n/**\n * @typedef {object} WebsocketEventRow\n * @property {string} channel - Channel name.\n * @property {Date | string} created_at - Creation time.\n * @property {string} id - Event id.\n * @property {string} payload_json - Serialized payload.\n * @property {number | string} sequence - Sequence number.\n */\n\n/**\n * @typedef {object} WebsocketReplayChannelRow\n * @property {string} channel - Channel name.\n */\n\n/**\n * @param {import(\"../configuration.js\").default} configuration - Configuration.\n * @returns {VelociousHttpServerWebsocketEventLogStore} - Shared store instance.\n */\nexport function websocketEventLogStoreForConfiguration(configuration) {\n  let store = stores.get(configuration)\n\n  if (!store) {\n    store = new VelociousHttpServerWebsocketEventLogStore({configuration})\n    stores.set(configuration, store)\n  }\n\n  return store\n}\n\nexport default class VelociousHttpServerWebsocketEventLogStore {\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../configuration.js\").default} args.configuration - Configuration.\n   * @param {string} [args.databaseIdentifier] - Database identifier.\n   * @param {number} [args.retentionMs] - Event retention in milliseconds.\n   */\n  constructor({configuration, databaseIdentifier = \"default\", retentionMs = DEFAULT_RETENTION_MS}) {\n    this.configuration = configuration\n    this.databaseIdentifier = databaseIdentifier\n    this.retentionMs = retentionMs\n    this.logger = new Logger(this)\n    this._isReady = false\n    this._readyPromise = null\n  }\n\n  /**\n   * @returns {Promise<void>} - Resolves when ready.\n   */\n  async ensureReady() {\n    if (this._isReady) return\n    if (this._readyPromise) return await this._readyPromise\n\n    this._readyPromise = (async () => {\n      this.configuration.setCurrent()\n      await this._ensureSchema()\n      this._isReady = true\n    })()\n\n    try {\n      await this._readyPromise\n    } finally {\n      if (!this._isReady) {\n        this._readyPromise = null\n      }\n    }\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {string} args.channel - Channel name.\n   * @param {unknown} args.payload - Event payload.\n   * @returns {Promise<{channel: string, createdAt: string, id: string, payload: unknown}>} - Persisted event row.\n   */\n  async appendEvent({channel, payload}) {\n    await this.ensureReady()\n\n    const id = randomUUID()\n    const createdAt = new Date()\n\n    return await this._withDb(async (db) => {\n      await db.insert({\n        tableName: EVENTS_TABLE,\n        data: {\n          channel,\n          created_at: createdAt,\n          id,\n          payload_json: JSON.stringify(payload)\n        }\n      })\n      return {channel, createdAt: createdAt.toISOString(), id, payload}\n    })\n  }\n\n  /**\n   * @param {string} channel - Channel name.\n   * @returns {Promise<void>} - Resolves when the channel interest was persisted.\n   */\n  async markChannelInterested(channel) {\n    await this.ensureReady()\n\n    const interestedUntil = new Date(Date.now() + this.retentionMs)\n\n    await this._withDb(async (db) => {\n      await this._upsertReplayChannelInterest(db, {channel, interestedUntil})\n    })\n  }\n\n  /**\n   * @param {string} channel - Channel name.\n   * @returns {Promise<boolean>} - Whether the channel should be persisted for replay.\n   */\n  async shouldPersistChannel(channel) {\n    await this.ensureReady()\n\n    return await this._withDb(async (db) => {\n      const rows = await db\n        .newQuery()\n        .from(REPLAY_CHANNELS_TABLE)\n        .where({channel})\n        .where(`interested_until > ${db.quote(new Date())}`)\n        .limit(1)\n        .results()\n\n      return rows.length > 0\n    })\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {string} args.channel - Channel name.\n   * @param {string} args.id - Event id.\n   * @returns {Promise<{channel: string, createdAt: string, id: string, payload: unknown, sequence: number} | null>} - Event row or null.\n   */\n  async getEventById({channel, id}) {\n    await this.ensureReady()\n\n    return await this._withDb(async (db) => {\n      return await this._getEventById({channel, db, id})\n    })\n  }\n\n  /**\n   * @param {string} channel - Channel name.\n   * @returns {Promise<number | null>} - Latest channel sequence.\n   */\n  async latestSequence(channel) {\n    await this.ensureReady()\n\n    return await this._withDb(async (db) => {\n      const rows = await db\n        .newQuery()\n        .from(EVENTS_TABLE)\n        .where({channel})\n        .order(\"sequence DESC\")\n        .limit(1)\n        .results()\n      const row = /** @type {Record<string, any> | undefined} */ (rows[0])\n\n      if (!row) return null\n\n      return Number(row.sequence)\n    })\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {string} args.channel - Channel name.\n   * @param {number} args.sequence - Lower bound sequence.\n   * @param {number | null | undefined} [args.upToSequence] - Inclusive ceiling sequence.\n   * @returns {Promise<Array<{channel: string, createdAt: string, id: string, payload: unknown, sequence: number}>>} - Ordered events.\n   */\n  async getEventsAfter({channel, sequence, upToSequence}) {\n    await this.ensureReady()\n\n    return await this._withDb(async (db) => {\n      const query = db\n        .newQuery()\n        .from(EVENTS_TABLE)\n        .where({channel})\n        .where(`sequence > ${db.quote(sequence)}`)\n        .order(\"sequence ASC\")\n\n      if (typeof upToSequence === \"number\") {\n        query.where(`sequence <= ${db.quote(upToSequence)}`)\n      }\n\n      const rows = /** @type {WebsocketEventRow[]} */ (await query.results())\n\n      return rows.map((row) => this._normalizeEventRow(row))\n    })\n  }\n\n  /**\n   * @param {object} [args] - Options.\n   * @param {Date} [args.now] - Cleanup reference time.\n   * @returns {Promise<void>} - Resolves when cleanup completes.\n   */\n  async cleanupExpired({now = new Date()} = {}) {\n    await this.ensureReady()\n\n    const cutoff = new Date(now.getTime() - this.retentionMs)\n\n    await this._withDb(async (db) => {\n      const expiredEventRows = /** @type {Array<{id: string}>} */ (await db\n        .newQuery()\n        .from(EVENTS_TABLE)\n        .where(`created_at <= ${db.quote(cutoff)}`)\n        .results())\n      const expiredReplayChannelRows = /** @type {WebsocketReplayChannelRow[]} */ (await db\n        .newQuery()\n        .from(REPLAY_CHANNELS_TABLE)\n        .where(`interested_until <= ${db.quote(now)}`)\n        .results())\n\n      for (const expiredEventRow of expiredEventRows) {\n        await db.delete({\n          tableName: EVENTS_TABLE,\n          conditions: {id: expiredEventRow.id}\n        })\n      }\n\n      for (const expiredReplayChannelRow of expiredReplayChannelRows) {\n        await db.delete({\n          tableName: REPLAY_CHANNELS_TABLE,\n          conditions: {channel: expiredReplayChannelRow.channel}\n        })\n      }\n    })\n  }\n\n  async _ensureSchema() {\n    await this._withDb(async (db) => {\n      await this._ensureEventsTable(db)\n      await this._ensureReplayChannelsTable(db)\n    })\n  }\n\n  /**\n   * @param {import(\"../database/drivers/base.js\").default} db - Database connection.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async _ensureEventsTable(db) {\n    this.logger.info(\"Applying websocket event-log schema\")\n\n    if (await db.tableExists(EVENTS_TABLE)) {\n      this.logger.info(\"Websocket event-log table already exists - skipping create\")\n      return\n    }\n\n    const eventTable = new TableData(EVENTS_TABLE, {ifNotExists: true})\n\n    eventTable.integer(\"sequence\", {autoIncrement: true, null: false, primaryKey: true})\n    eventTable.string(\"id\", {index: true, null: false})\n    eventTable.string(\"channel\", {index: true, null: false})\n    eventTable.text(\"payload_json\", {null: false})\n    eventTable.datetime(\"created_at\", {index: true, null: false})\n\n    await db.createTable(eventTable)\n  }\n\n  /**\n   * @param {import(\"../database/drivers/base.js\").default} db - Database connection.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async _ensureReplayChannelsTable(db) {\n    if (await db.tableExists(REPLAY_CHANNELS_TABLE)) return\n\n    const replayChannelTable = new TableData(REPLAY_CHANNELS_TABLE, {ifNotExists: true})\n\n    replayChannelTable.string(\"channel\", {null: false, primaryKey: true})\n    replayChannelTable.datetime(\"interested_until\", {index: true, null: false})\n\n    await db.createTable(replayChannelTable)\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {string} args.channel - Channel name.\n   * @param {import(\"../database/drivers/base.js\").default} args.db - Database connection.\n   * @param {string} args.id - Event id.\n   * @returns {Promise<{channel: string, createdAt: string, id: string, payload: unknown, sequence: number} | null>} - Event row or null.\n   */\n  async _getEventById({channel, db, id}) {\n    const rows = /** @type {WebsocketEventRow[]} */ (await db\n      .newQuery()\n      .from(EVENTS_TABLE)\n      .where({channel, id})\n      .limit(1)\n      .results())\n\n    if (!rows[0]) return null\n\n    return this._normalizeEventRow(rows[0])\n  }\n\n  /**\n   * @param {WebsocketEventRow} row - Raw row.\n   * @returns {{channel: string, createdAt: string, id: string, payload: unknown, sequence: number}} - Normalized row.\n   */\n  _normalizeEventRow(row) {\n    const createdAtValue = row.created_at\n\n    return {\n      channel: row.channel,\n      createdAt: createdAtValue instanceof Date ? createdAtValue.toISOString() : new Date(createdAtValue).toISOString(),\n      id: row.id,\n      payload: JSON.parse(row.payload_json),\n      sequence: Number(row.sequence)\n    }\n  }\n\n  /**\n   * @param {import(\"../database/drivers/base.js\").default} db - Database connection.\n   * @param {object} args - Options.\n   * @param {string} args.channel - Channel name.\n   * @param {Date} args.interestedUntil - Retention deadline.\n   * @returns {Promise<void>} - Resolves when the replay-channel row was upserted.\n   */\n  async _upsertReplayChannelInterest(db, {channel, interestedUntil}) {\n    await db.upsert({\n      conflictColumns: [\"channel\"],\n      data: {\n        channel,\n        interested_until: interestedUntil\n      },\n      tableName: REPLAY_CHANNELS_TABLE,\n      updateColumns: [\"interested_until\"]\n    })\n  }\n\n  /**\n   * @param {(db: import(\"../database/drivers/base.js\").default) => Promise<any>} callback - Callback.\n   * @returns {Promise<any>} - Callback result.\n   */\n  async _withDb(callback) {\n    return await this.configuration.ensureConnections(async (dbs) => {\n      const db = dbs[this.databaseIdentifier]\n\n      if (!db) throw new Error(`No database connection available for identifier: ${this.databaseIdentifier}`)\n\n      return await callback(db)\n    })\n  }\n}\n"]}
|
|
313
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"websocket-event-log-store.js","sourceRoot":"","sources":["../../../src/http-server/websocket-event-log-store.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nimport {randomUUID} from \"crypto\"\nimport TableData from \"../database/table-data/index.js\"\nimport Logger from \"../logger.js\"\n\nconst EVENTS_TABLE = \"websocket_channel_events\"\nconst REPLAY_CHANNELS_TABLE = \"websocket_replay_channels\"\nconst DEFAULT_RETENTION_MS = 10 * 60 * 1000\nconst stores = new WeakMap()\n\n/**\n * @typedef {object} WebsocketEventRow\n * @property {string} channel - Channel name.\n * @property {Date | string} created_at - Creation time.\n * @property {string} id - Event id.\n * @property {string} payload_json - Serialized payload.\n * @property {number | string} sequence - Sequence number.\n */\n\n/**\n * @typedef {object} WebsocketReplayChannelRow\n * @property {string} channel - Channel name.\n */\n\n/**\n * @param {import(\"../configuration.js\").default} configuration - Configuration.\n * @returns {VelociousHttpServerWebsocketEventLogStore} - Shared store instance.\n */\nexport function websocketEventLogStoreForConfiguration(configuration) {\n  let store = stores.get(configuration)\n\n  if (!store) {\n    store = new VelociousHttpServerWebsocketEventLogStore({configuration})\n    stores.set(configuration, store)\n  }\n\n  return store\n}\n\nexport default class VelociousHttpServerWebsocketEventLogStore {\n  /**\n   * @param {object} args - Options.\n   * @param {import(\"../configuration.js\").default} args.configuration - Configuration.\n   * @param {string} [args.databaseIdentifier] - Database identifier.\n   * @param {number} [args.retentionMs] - Event retention in milliseconds.\n   */\n  constructor({configuration, databaseIdentifier = \"default\", retentionMs = DEFAULT_RETENTION_MS}) {\n    this.configuration = configuration\n    this.databaseIdentifier = databaseIdentifier\n    this.retentionMs = retentionMs\n    this.logger = new Logger(this)\n    this._isReady = false\n    this._readyPromise = null\n    /** @type {Set<string>} */\n    this._interestedChannels = new Set()\n  }\n\n  /**\n   * @returns {Promise<void>} - Resolves when ready.\n   */\n  async ensureReady() {\n    if (this._isReady) return\n    if (this._readyPromise) return await this._readyPromise\n\n    this._readyPromise = (async () => {\n      this.configuration.setCurrent()\n      await this._ensureSchema()\n      this._isReady = true\n    })()\n\n    try {\n      await this._readyPromise\n    } finally {\n      if (!this._isReady) {\n        this._readyPromise = null\n      }\n    }\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {string} args.channel - Channel name.\n   * @param {unknown} args.payload - Event payload.\n   * @returns {Promise<{channel: string, createdAt: string, id: string, payload: unknown}>} - Persisted event row.\n   */\n  async appendEvent({channel, payload}) {\n    await this.ensureReady()\n\n    const id = randomUUID()\n    const createdAt = new Date()\n\n    return await this._withDb(async (db) => {\n      await db.insert({\n        tableName: EVENTS_TABLE,\n        data: {\n          channel,\n          created_at: createdAt,\n          id,\n          payload_json: JSON.stringify(payload)\n        }\n      })\n      return {channel, createdAt: createdAt.toISOString(), id, payload}\n    })\n  }\n\n  /**\n   * @param {string} channel - Channel name.\n   * @returns {Promise<void>} - Resolves when the channel interest was persisted.\n   */\n  async markChannelInterested(channel) {\n    await this.ensureReady()\n\n    this._interestedChannels.add(channel)\n    const interestedUntil = new Date(Date.now() + this.retentionMs)\n\n    await this._withDb(async (db) => {\n      await this._upsertReplayChannelInterest(db, {channel, interestedUntil})\n    })\n  }\n\n  /**\n   * @param {string} channel - Channel name.\n   * @returns {Promise<boolean>} - Whether the channel should be persisted for replay.\n   */\n  async shouldPersistChannel(channel) {\n    if (this._interestedChannels.size === 0) return false\n\n    await this.ensureReady()\n\n    return await this._withDb(async (db) => {\n      const rows = await db\n        .newQuery()\n        .from(REPLAY_CHANNELS_TABLE)\n        .where({channel})\n        .where(`interested_until > ${db.quote(new Date())}`)\n        .limit(1)\n        .results()\n\n      return rows.length > 0\n    })\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {string} args.channel - Channel name.\n   * @param {string} args.id - Event id.\n   * @returns {Promise<{channel: string, createdAt: string, id: string, payload: unknown, sequence: number} | null>} - Event row or null.\n   */\n  async getEventById({channel, id}) {\n    await this.ensureReady()\n\n    return await this._withDb(async (db) => {\n      return await this._getEventById({channel, db, id})\n    })\n  }\n\n  /**\n   * @param {string} channel - Channel name.\n   * @returns {Promise<number | null>} - Latest channel sequence.\n   */\n  async latestSequence(channel) {\n    await this.ensureReady()\n\n    return await this._withDb(async (db) => {\n      const rows = await db\n        .newQuery()\n        .from(EVENTS_TABLE)\n        .where({channel})\n        .order(\"sequence DESC\")\n        .limit(1)\n        .results()\n      const row = /** @type {Record<string, any> | undefined} */ (rows[0])\n\n      if (!row) return null\n\n      return Number(row.sequence)\n    })\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {string} args.channel - Channel name.\n   * @param {number} args.sequence - Lower bound sequence.\n   * @param {number | null | undefined} [args.upToSequence] - Inclusive ceiling sequence.\n   * @returns {Promise<Array<{channel: string, createdAt: string, id: string, payload: unknown, sequence: number}>>} - Ordered events.\n   */\n  async getEventsAfter({channel, sequence, upToSequence}) {\n    await this.ensureReady()\n\n    return await this._withDb(async (db) => {\n      const query = db\n        .newQuery()\n        .from(EVENTS_TABLE)\n        .where({channel})\n        .where(`sequence > ${db.quote(sequence)}`)\n        .order(\"sequence ASC\")\n\n      if (typeof upToSequence === \"number\") {\n        query.where(`sequence <= ${db.quote(upToSequence)}`)\n      }\n\n      const rows = /** @type {WebsocketEventRow[]} */ (await query.results())\n\n      return rows.map((row) => this._normalizeEventRow(row))\n    })\n  }\n\n  /**\n   * @param {object} [args] - Options.\n   * @param {Date} [args.now] - Cleanup reference time.\n   * @returns {Promise<void>} - Resolves when cleanup completes.\n   */\n  async cleanupExpired({now = new Date()} = {}) {\n    await this.ensureReady()\n\n    const cutoff = new Date(now.getTime() - this.retentionMs)\n\n    await this._withDb(async (db) => {\n      const expiredEventRows = /** @type {Array<{id: string}>} */ (await db\n        .newQuery()\n        .from(EVENTS_TABLE)\n        .where(`created_at <= ${db.quote(cutoff)}`)\n        .results())\n      const expiredReplayChannelRows = /** @type {WebsocketReplayChannelRow[]} */ (await db\n        .newQuery()\n        .from(REPLAY_CHANNELS_TABLE)\n        .where(`interested_until <= ${db.quote(now)}`)\n        .results())\n\n      for (const expiredEventRow of expiredEventRows) {\n        await db.delete({\n          tableName: EVENTS_TABLE,\n          conditions: {id: expiredEventRow.id}\n        })\n      }\n\n      for (const expiredReplayChannelRow of expiredReplayChannelRows) {\n        await db.delete({\n          tableName: REPLAY_CHANNELS_TABLE,\n          conditions: {channel: expiredReplayChannelRow.channel}\n        })\n      }\n    })\n  }\n\n  async _ensureSchema() {\n    await this._withDb(async (db) => {\n      await this._ensureEventsTable(db)\n      await this._ensureReplayChannelsTable(db)\n    })\n  }\n\n  /**\n   * @param {import(\"../database/drivers/base.js\").default} db - Database connection.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async _ensureEventsTable(db) {\n    this.logger.info(\"Applying websocket event-log schema\")\n\n    if (await db.tableExists(EVENTS_TABLE)) {\n      this.logger.info(\"Websocket event-log table already exists - skipping create\")\n      return\n    }\n\n    const eventTable = new TableData(EVENTS_TABLE, {ifNotExists: true})\n\n    eventTable.integer(\"sequence\", {autoIncrement: true, null: false, primaryKey: true})\n    eventTable.string(\"id\", {index: true, null: false})\n    eventTable.string(\"channel\", {index: true, null: false})\n    eventTable.text(\"payload_json\", {null: false})\n    eventTable.datetime(\"created_at\", {index: true, null: false})\n\n    await db.createTable(eventTable)\n  }\n\n  /**\n   * @param {import(\"../database/drivers/base.js\").default} db - Database connection.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async _ensureReplayChannelsTable(db) {\n    if (await db.tableExists(REPLAY_CHANNELS_TABLE)) return\n\n    const replayChannelTable = new TableData(REPLAY_CHANNELS_TABLE, {ifNotExists: true})\n\n    replayChannelTable.string(\"channel\", {null: false, primaryKey: true})\n    replayChannelTable.datetime(\"interested_until\", {index: true, null: false})\n\n    await db.createTable(replayChannelTable)\n  }\n\n  /**\n   * @param {object} args - Options.\n   * @param {string} args.channel - Channel name.\n   * @param {import(\"../database/drivers/base.js\").default} args.db - Database connection.\n   * @param {string} args.id - Event id.\n   * @returns {Promise<{channel: string, createdAt: string, id: string, payload: unknown, sequence: number} | null>} - Event row or null.\n   */\n  async _getEventById({channel, db, id}) {\n    const rows = /** @type {WebsocketEventRow[]} */ (await db\n      .newQuery()\n      .from(EVENTS_TABLE)\n      .where({channel, id})\n      .limit(1)\n      .results())\n\n    if (!rows[0]) return null\n\n    return this._normalizeEventRow(rows[0])\n  }\n\n  /**\n   * @param {WebsocketEventRow} row - Raw row.\n   * @returns {{channel: string, createdAt: string, id: string, payload: unknown, sequence: number}} - Normalized row.\n   */\n  _normalizeEventRow(row) {\n    const createdAtValue = row.created_at\n\n    return {\n      channel: row.channel,\n      createdAt: createdAtValue instanceof Date ? createdAtValue.toISOString() : new Date(createdAtValue).toISOString(),\n      id: row.id,\n      payload: JSON.parse(row.payload_json),\n      sequence: Number(row.sequence)\n    }\n  }\n\n  /**\n   * @param {import(\"../database/drivers/base.js\").default} db - Database connection.\n   * @param {object} args - Options.\n   * @param {string} args.channel - Channel name.\n   * @param {Date} args.interestedUntil - Retention deadline.\n   * @returns {Promise<void>} - Resolves when the replay-channel row was upserted.\n   */\n  async _upsertReplayChannelInterest(db, {channel, interestedUntil}) {\n    await db.upsert({\n      conflictColumns: [\"channel\"],\n      data: {\n        channel,\n        interested_until: interestedUntil\n      },\n      tableName: REPLAY_CHANNELS_TABLE,\n      updateColumns: [\"interested_until\"]\n    })\n  }\n\n  /**\n   * @param {(db: import(\"../database/drivers/base.js\").default) => Promise<any>} callback - Callback.\n   * @returns {Promise<any>} - Callback result.\n   */\n  async _withDb(callback) {\n    return await this.configuration.ensureConnections(async (dbs) => {\n      const db = dbs[this.databaseIdentifier]\n\n      if (!db) throw new Error(`No database connection available for identifier: ${this.databaseIdentifier}`)\n\n      return await callback(db)\n    })\n  }\n}\n"]}
|