velocious 1.0.299 → 1.0.301
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/configuration-types.d.ts +1 -1
- package/build/src/configuration.d.ts +1 -1
- package/build/src/database/drivers/base.d.ts +29 -0
- package/build/src/database/drivers/base.d.ts.map +1 -1
- package/build/src/database/drivers/base.js +25 -1
- package/build/src/database/drivers/mssql/index.d.ts.map +1 -1
- package/build/src/database/drivers/mssql/index.js +10 -1
- package/build/src/database/drivers/mssql/sql/upsert.d.ts +5 -0
- package/build/src/database/drivers/mssql/sql/upsert.d.ts.map +1 -0
- package/build/src/database/drivers/mssql/sql/upsert.js +19 -0
- package/build/src/database/drivers/mysql/index.d.ts.map +1 -1
- package/build/src/database/drivers/mysql/index.js +10 -1
- package/build/src/database/drivers/mysql/sql/upsert.d.ts +5 -0
- package/build/src/database/drivers/mysql/sql/upsert.d.ts.map +1 -0
- package/build/src/database/drivers/mysql/sql/upsert.js +11 -0
- package/build/src/database/drivers/pgsql/index.d.ts.map +1 -1
- package/build/src/database/drivers/pgsql/index.js +10 -1
- package/build/src/database/drivers/pgsql/sql/upsert.d.ts +5 -0
- package/build/src/database/drivers/pgsql/sql/upsert.d.ts.map +1 -0
- package/build/src/database/drivers/pgsql/sql/upsert.js +12 -0
- package/build/src/database/drivers/sqlite/base.d.ts.map +1 -1
- package/build/src/database/drivers/sqlite/base.js +7 -1
- package/build/src/database/drivers/sqlite/sql/upsert.d.ts +5 -0
- package/build/src/database/drivers/sqlite/sql/upsert.d.ts.map +1 -0
- package/build/src/database/drivers/sqlite/sql/upsert.js +12 -0
- package/build/src/database/query/upsert-base.d.ts +59 -0
- package/build/src/database/query/upsert-base.d.ts.map +1 -0
- package/build/src/database/query/upsert-base.js +87 -0
- package/build/src/frontend-model-controller.d.ts.map +1 -1
- package/build/src/frontend-model-controller.js +8 -3
- package/build/src/frontend-models/websocket-channel.d.ts +9 -1
- package/build/src/frontend-models/websocket-channel.d.ts.map +1 -1
- package/build/src/frontend-models/websocket-channel.js +17 -3
- package/build/src/http-client/websocket-client.d.ts +8 -6
- package/build/src/http-client/websocket-client.d.ts.map +1 -1
- package/build/src/http-client/websocket-client.js +17 -8
- package/build/src/http-server/client/websocket-session.d.ts +62 -5
- package/build/src/http-server/client/websocket-session.d.ts.map +1 -1
- package/build/src/http-server/client/websocket-session.js +129 -14
- package/build/src/http-server/websocket-channel.d.ts +18 -3
- package/build/src/http-server/websocket-channel.d.ts.map +1 -1
- package/build/src/http-server/websocket-channel.js +18 -6
- package/build/src/http-server/websocket-event-log-store.d.ts +198 -0
- package/build/src/http-server/websocket-event-log-store.d.ts.map +1 -0
- package/build/src/http-server/websocket-event-log-store.js +308 -0
- package/build/src/http-server/websocket-events-host.d.ts +14 -0
- package/build/src/http-server/websocket-events-host.d.ts.map +1 -1
- package/build/src/http-server/websocket-events-host.js +38 -4
- package/build/src/http-server/worker-handler/in-process.d.ts +5 -1
- package/build/src/http-server/worker-handler/in-process.d.ts.map +1 -1
- package/build/src/http-server/worker-handler/in-process.js +10 -3
- package/build/src/http-server/worker-handler/index.d.ts +5 -1
- package/build/src/http-server/worker-handler/index.d.ts.map +1 -1
- package/build/src/http-server/worker-handler/index.js +5 -3
- package/build/src/http-server/worker-handler/worker-thread.d.ts +9 -1
- package/build/src/http-server/worker-handler/worker-thread.d.ts.map +1 -1
- package/build/src/http-server/worker-handler/worker-thread.js +16 -6
- package/build/tsconfig.tsbuildinfo +1 -1
- package/package.json +1 -1
|
@@ -5,13 +5,17 @@ export default class VelociousHttpServerWebsocketChannel {
|
|
|
5
5
|
* @param {import("../configuration.js").default} args.configuration - Configuration instance.
|
|
6
6
|
* @param {import("./client/request.js").default | import("./client/websocket-request.js").default | undefined} args.request - Request instance.
|
|
7
7
|
* @param {import("./client/index.js").default} args.client - Client instance.
|
|
8
|
+
* @param {string} [args.lastEventId] - Last received event id.
|
|
9
|
+
* @param {string} [args.subscriptionChannel] - Client-facing subscription channel.
|
|
8
10
|
* @param {import("./client/websocket-session.js").default} args.websocketSession - Websocket session.
|
|
9
11
|
* @param {Record<string, unknown>} [args.subscriptionParams] - Params from subscribe message.
|
|
10
12
|
*/
|
|
11
|
-
constructor({ configuration, request, client, websocketSession, subscriptionParams }) {
|
|
13
|
+
constructor({ configuration, request, client, lastEventId, subscriptionChannel, websocketSession, subscriptionParams }) {
|
|
12
14
|
this.configuration = configuration;
|
|
13
15
|
this.request = request;
|
|
14
16
|
this.client = client;
|
|
17
|
+
this.lastEventId = lastEventId;
|
|
18
|
+
this.subscriptionChannel = subscriptionChannel;
|
|
15
19
|
this.websocketSession = websocketSession;
|
|
16
20
|
this.subscriptionParams = subscriptionParams;
|
|
17
21
|
this._params = this._buildParams();
|
|
@@ -33,24 +37,32 @@ export default class VelociousHttpServerWebsocketChannel {
|
|
|
33
37
|
/**
|
|
34
38
|
* Subscribe this connection to a broadcast channel.
|
|
35
39
|
* @param {string} channel - Channel name.
|
|
36
|
-
* @param {{acknowledge?: boolean}} [options] - Subscription options.
|
|
40
|
+
* @param {{acknowledge?: boolean, lastEventId?: string}} [options] - Subscription options.
|
|
37
41
|
* @returns {Promise<boolean>} - Whether the subscription succeeded.
|
|
38
42
|
*/
|
|
39
43
|
async streamFrom(channel, options = {}) {
|
|
44
|
+
const lastEventId = options.lastEventId ?? this.lastEventId;
|
|
40
45
|
return await this.websocketSession.subscribeToChannel(channel, {
|
|
41
46
|
acknowledge: options.acknowledge ?? true,
|
|
42
|
-
channelHandler: this
|
|
47
|
+
channelHandler: this,
|
|
48
|
+
lastEventId,
|
|
49
|
+
params: this.subscriptionParams,
|
|
50
|
+
subscriptionChannel: this.subscriptionChannel
|
|
43
51
|
});
|
|
44
52
|
}
|
|
45
53
|
/**
|
|
46
54
|
* Called when a broadcast event is delivered for one of this channel instance's subscriptions.
|
|
47
55
|
* @param {object} args - Event args.
|
|
48
56
|
* @param {string} args.channel - Broadcast channel name.
|
|
57
|
+
* @param {string} [args.createdAt] - Event creation timestamp.
|
|
58
|
+
* @param {string} [args.eventId] - Event id.
|
|
49
59
|
* @param {any} args.payload - Broadcast payload.
|
|
60
|
+
* @param {boolean} [args.replayed] - Whether this event was replayed.
|
|
61
|
+
* @param {number} [args.sequence] - Event sequence.
|
|
50
62
|
* @returns {Promise<void>} - Resolves when complete.
|
|
51
63
|
*/
|
|
52
|
-
async receivedBroadcast({ channel, payload }) {
|
|
53
|
-
this.websocketSession.sendJson({ channel, payload, type: "event" });
|
|
64
|
+
async receivedBroadcast({ channel, createdAt, eventId, payload, replayed, sequence }) {
|
|
65
|
+
this.websocketSession.sendJson({ channel, createdAt, eventId, payload, replayed, sequence, type: "event" });
|
|
54
66
|
}
|
|
55
67
|
/**
|
|
56
68
|
* @returns {Record<string, unknown>} - Parsed params.
|
|
@@ -80,4 +92,4 @@ export default class VelociousHttpServerWebsocketChannel {
|
|
|
80
92
|
return params;
|
|
81
93
|
}
|
|
82
94
|
}
|
|
83
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
95
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"websocket-channel.js","sourceRoot":"","sources":["../../../src/http-server/websocket-channel.js"],"names":[],"mappings":"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","sourcesContent":["// @ts-check\n\nexport default class VelociousHttpServerWebsocketChannel {\n  /**\n   * @param {object} args - Options object.\n   * @param {import(\"../configuration.js\").default} args.configuration - Configuration instance.\n   * @param {import(\"./client/request.js\").default | import(\"./client/websocket-request.js\").default | undefined} args.request - Request instance.\n   * @param {import(\"./client/index.js\").default} args.client - Client instance.\n   * @param {string} [args.lastEventId] - Last received event id.\n   * @param {string} [args.subscriptionChannel] - Client-facing subscription channel.\n   * @param {import(\"./client/websocket-session.js\").default} args.websocketSession - Websocket session.\n   * @param {Record<string, unknown>} [args.subscriptionParams] - Params from subscribe message.\n   */\n  constructor({configuration, request, client, lastEventId, subscriptionChannel, websocketSession, subscriptionParams}) {\n    this.configuration = configuration\n    this.request = request\n    this.client = client\n    this.lastEventId = lastEventId\n    this.subscriptionChannel = subscriptionChannel\n    this.websocketSession = websocketSession\n    this.subscriptionParams = subscriptionParams\n    this._params = this._buildParams()\n  }\n\n  /**\n   * @returns {Record<string, unknown>} - Params for the websocket connection.\n   */\n  params() { return this._params }\n\n  /**\n   * Called when the channel is created for a websocket connection.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async subscribed() {}\n\n  /**\n   * Called when the websocket disconnects.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async unsubscribed() {}\n\n  /**\n   * Subscribe this connection to a broadcast channel.\n   * @param {string} channel - Channel name.\n   * @param {{acknowledge?: boolean, lastEventId?: string}} [options] - Subscription options.\n   * @returns {Promise<boolean>} - Whether the subscription succeeded.\n   */\n  async streamFrom(channel, options = {}) {\n    const lastEventId = options.lastEventId ?? this.lastEventId\n\n    return await this.websocketSession.subscribeToChannel(channel, {\n      acknowledge: options.acknowledge ?? true,\n      channelHandler: this,\n      lastEventId,\n      params: this.subscriptionParams,\n      subscriptionChannel: this.subscriptionChannel\n    })\n  }\n\n  /**\n   * Called when a broadcast event is delivered for one of this channel instance's subscriptions.\n   * @param {object} args - Event args.\n   * @param {string} args.channel - Broadcast channel name.\n   * @param {string} [args.createdAt] - Event creation timestamp.\n   * @param {string} [args.eventId] - Event id.\n   * @param {any} args.payload - Broadcast payload.\n   * @param {boolean} [args.replayed] - Whether this event was replayed.\n   * @param {number} [args.sequence] - Event sequence.\n   * @returns {Promise<void>} - Resolves when complete.\n   */\n  async receivedBroadcast({channel, createdAt, eventId, payload, replayed, sequence}) {\n    this.websocketSession.sendJson({channel, createdAt, eventId, payload, replayed, sequence, type: \"event\"})\n  }\n\n  /**\n   * @returns {Record<string, unknown>} - Parsed params.\n   */\n  _buildParams() {\n    /** @type {Record<string, unknown>} */\n    const params = {}\n\n    if (this.request?.params) {\n      const requestParams = this.request.params()\n\n      if (requestParams && typeof requestParams === \"object\") {\n        Object.assign(params, requestParams)\n      }\n    }\n\n    const pathValue = this.request?.path?.()\n    const query = pathValue?.split(\"?\")[1]\n\n    if (query) {\n      const searchParams = new URLSearchParams(query)\n\n      for (const [key, value] of searchParams.entries()) {\n        if (params[key] === undefined) {\n          params[key] = value\n        }\n      }\n    }\n\n    if (this.subscriptionParams && typeof this.subscriptionParams === \"object\") {\n      Object.assign(params, this.subscriptionParams)\n    }\n\n    return params\n  }\n}\n"]}
|
|
@@ -0,0 +1,198 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @typedef {object} WebsocketEventRow
|
|
3
|
+
* @property {string} channel - Channel name.
|
|
4
|
+
* @property {Date | string} created_at - Creation time.
|
|
5
|
+
* @property {string} id - Event id.
|
|
6
|
+
* @property {string} payload_json - Serialized payload.
|
|
7
|
+
* @property {number | string} sequence - Sequence number.
|
|
8
|
+
*/
|
|
9
|
+
/**
|
|
10
|
+
* @typedef {object} WebsocketReplayChannelRow
|
|
11
|
+
* @property {string} channel - Channel name.
|
|
12
|
+
*/
|
|
13
|
+
/**
|
|
14
|
+
* @param {import("../configuration.js").default} configuration - Configuration.
|
|
15
|
+
* @returns {VelociousHttpServerWebsocketEventLogStore} - Shared store instance.
|
|
16
|
+
*/
|
|
17
|
+
export function websocketEventLogStoreForConfiguration(configuration: import("../configuration.js").default): VelociousHttpServerWebsocketEventLogStore;
|
|
18
|
+
export default class VelociousHttpServerWebsocketEventLogStore {
|
|
19
|
+
/**
|
|
20
|
+
* @param {object} args - Options.
|
|
21
|
+
* @param {import("../configuration.js").default} args.configuration - Configuration.
|
|
22
|
+
* @param {string} [args.databaseIdentifier] - Database identifier.
|
|
23
|
+
* @param {number} [args.retentionMs] - Event retention in milliseconds.
|
|
24
|
+
*/
|
|
25
|
+
constructor({ configuration, databaseIdentifier, retentionMs }: {
|
|
26
|
+
configuration: import("../configuration.js").default;
|
|
27
|
+
databaseIdentifier?: string | undefined;
|
|
28
|
+
retentionMs?: number | undefined;
|
|
29
|
+
});
|
|
30
|
+
configuration: import("../configuration.js").default;
|
|
31
|
+
databaseIdentifier: string;
|
|
32
|
+
retentionMs: number;
|
|
33
|
+
logger: Logger;
|
|
34
|
+
_isReady: boolean;
|
|
35
|
+
_readyPromise: Promise<void> | null;
|
|
36
|
+
/**
|
|
37
|
+
* @returns {Promise<void>} - Resolves when ready.
|
|
38
|
+
*/
|
|
39
|
+
ensureReady(): Promise<void>;
|
|
40
|
+
/**
|
|
41
|
+
* @param {object} args - Options.
|
|
42
|
+
* @param {string} args.channel - Channel name.
|
|
43
|
+
* @param {unknown} args.payload - Event payload.
|
|
44
|
+
* @returns {Promise<{channel: string, createdAt: string, id: string, payload: unknown}>} - Persisted event row.
|
|
45
|
+
*/
|
|
46
|
+
appendEvent({ channel, payload }: {
|
|
47
|
+
channel: string;
|
|
48
|
+
payload: unknown;
|
|
49
|
+
}): Promise<{
|
|
50
|
+
channel: string;
|
|
51
|
+
createdAt: string;
|
|
52
|
+
id: string;
|
|
53
|
+
payload: unknown;
|
|
54
|
+
}>;
|
|
55
|
+
/**
|
|
56
|
+
* @param {string} channel - Channel name.
|
|
57
|
+
* @returns {Promise<void>} - Resolves when the channel interest was persisted.
|
|
58
|
+
*/
|
|
59
|
+
markChannelInterested(channel: string): Promise<void>;
|
|
60
|
+
/**
|
|
61
|
+
* @param {string} channel - Channel name.
|
|
62
|
+
* @returns {Promise<boolean>} - Whether the channel should be persisted for replay.
|
|
63
|
+
*/
|
|
64
|
+
shouldPersistChannel(channel: string): Promise<boolean>;
|
|
65
|
+
/**
|
|
66
|
+
* @param {object} args - Options.
|
|
67
|
+
* @param {string} args.channel - Channel name.
|
|
68
|
+
* @param {string} args.id - Event id.
|
|
69
|
+
* @returns {Promise<{channel: string, createdAt: string, id: string, payload: unknown, sequence: number} | null>} - Event row or null.
|
|
70
|
+
*/
|
|
71
|
+
getEventById({ channel, id }: {
|
|
72
|
+
channel: string;
|
|
73
|
+
id: string;
|
|
74
|
+
}): Promise<{
|
|
75
|
+
channel: string;
|
|
76
|
+
createdAt: string;
|
|
77
|
+
id: string;
|
|
78
|
+
payload: unknown;
|
|
79
|
+
sequence: number;
|
|
80
|
+
} | null>;
|
|
81
|
+
/**
|
|
82
|
+
* @param {string} channel - Channel name.
|
|
83
|
+
* @returns {Promise<number | null>} - Latest channel sequence.
|
|
84
|
+
*/
|
|
85
|
+
latestSequence(channel: string): Promise<number | null>;
|
|
86
|
+
/**
|
|
87
|
+
* @param {object} args - Options.
|
|
88
|
+
* @param {string} args.channel - Channel name.
|
|
89
|
+
* @param {number} args.sequence - Lower bound sequence.
|
|
90
|
+
* @param {number | null | undefined} [args.upToSequence] - Inclusive ceiling sequence.
|
|
91
|
+
* @returns {Promise<Array<{channel: string, createdAt: string, id: string, payload: unknown, sequence: number}>>} - Ordered events.
|
|
92
|
+
*/
|
|
93
|
+
getEventsAfter({ channel, sequence, upToSequence }: {
|
|
94
|
+
channel: string;
|
|
95
|
+
sequence: number;
|
|
96
|
+
upToSequence?: number | null | undefined;
|
|
97
|
+
}): Promise<Array<{
|
|
98
|
+
channel: string;
|
|
99
|
+
createdAt: string;
|
|
100
|
+
id: string;
|
|
101
|
+
payload: unknown;
|
|
102
|
+
sequence: number;
|
|
103
|
+
}>>;
|
|
104
|
+
/**
|
|
105
|
+
* @param {object} [args] - Options.
|
|
106
|
+
* @param {Date} [args.now] - Cleanup reference time.
|
|
107
|
+
* @returns {Promise<void>} - Resolves when cleanup completes.
|
|
108
|
+
*/
|
|
109
|
+
cleanupExpired({ now }?: {
|
|
110
|
+
now?: Date | undefined;
|
|
111
|
+
}): Promise<void>;
|
|
112
|
+
_ensureSchema(): Promise<void>;
|
|
113
|
+
/**
|
|
114
|
+
* @param {import("../database/drivers/base.js").default} db - Database connection.
|
|
115
|
+
* @returns {Promise<void>} - Resolves when complete.
|
|
116
|
+
*/
|
|
117
|
+
_ensureEventsTable(db: import("../database/drivers/base.js").default): Promise<void>;
|
|
118
|
+
/**
|
|
119
|
+
* @param {import("../database/drivers/base.js").default} db - Database connection.
|
|
120
|
+
* @returns {Promise<void>} - Resolves when complete.
|
|
121
|
+
*/
|
|
122
|
+
_ensureReplayChannelsTable(db: import("../database/drivers/base.js").default): Promise<void>;
|
|
123
|
+
/**
|
|
124
|
+
* @param {object} args - Options.
|
|
125
|
+
* @param {string} args.channel - Channel name.
|
|
126
|
+
* @param {import("../database/drivers/base.js").default} args.db - Database connection.
|
|
127
|
+
* @param {string} args.id - Event id.
|
|
128
|
+
* @returns {Promise<{channel: string, createdAt: string, id: string, payload: unknown, sequence: number} | null>} - Event row or null.
|
|
129
|
+
*/
|
|
130
|
+
_getEventById({ channel, db, id }: {
|
|
131
|
+
channel: string;
|
|
132
|
+
db: import("../database/drivers/base.js").default;
|
|
133
|
+
id: string;
|
|
134
|
+
}): Promise<{
|
|
135
|
+
channel: string;
|
|
136
|
+
createdAt: string;
|
|
137
|
+
id: string;
|
|
138
|
+
payload: unknown;
|
|
139
|
+
sequence: number;
|
|
140
|
+
} | null>;
|
|
141
|
+
/**
|
|
142
|
+
* @param {WebsocketEventRow} row - Raw row.
|
|
143
|
+
* @returns {{channel: string, createdAt: string, id: string, payload: unknown, sequence: number}} - Normalized row.
|
|
144
|
+
*/
|
|
145
|
+
_normalizeEventRow(row: WebsocketEventRow): {
|
|
146
|
+
channel: string;
|
|
147
|
+
createdAt: string;
|
|
148
|
+
id: string;
|
|
149
|
+
payload: unknown;
|
|
150
|
+
sequence: number;
|
|
151
|
+
};
|
|
152
|
+
/**
|
|
153
|
+
* @param {import("../database/drivers/base.js").default} db - Database connection.
|
|
154
|
+
* @param {object} args - Options.
|
|
155
|
+
* @param {string} args.channel - Channel name.
|
|
156
|
+
* @param {Date} args.interestedUntil - Retention deadline.
|
|
157
|
+
* @returns {Promise<void>} - Resolves when the replay-channel row was upserted.
|
|
158
|
+
*/
|
|
159
|
+
_upsertReplayChannelInterest(db: import("../database/drivers/base.js").default, { channel, interestedUntil }: {
|
|
160
|
+
channel: string;
|
|
161
|
+
interestedUntil: Date;
|
|
162
|
+
}): Promise<void>;
|
|
163
|
+
/**
|
|
164
|
+
* @param {(db: import("../database/drivers/base.js").default) => Promise<any>} callback - Callback.
|
|
165
|
+
* @returns {Promise<any>} - Callback result.
|
|
166
|
+
*/
|
|
167
|
+
_withDb(callback: (db: import("../database/drivers/base.js").default) => Promise<any>): Promise<any>;
|
|
168
|
+
}
|
|
169
|
+
export type WebsocketEventRow = {
|
|
170
|
+
/**
|
|
171
|
+
* - Channel name.
|
|
172
|
+
*/
|
|
173
|
+
channel: string;
|
|
174
|
+
/**
|
|
175
|
+
* - Creation time.
|
|
176
|
+
*/
|
|
177
|
+
created_at: Date | string;
|
|
178
|
+
/**
|
|
179
|
+
* - Event id.
|
|
180
|
+
*/
|
|
181
|
+
id: string;
|
|
182
|
+
/**
|
|
183
|
+
* - Serialized payload.
|
|
184
|
+
*/
|
|
185
|
+
payload_json: string;
|
|
186
|
+
/**
|
|
187
|
+
* - Sequence number.
|
|
188
|
+
*/
|
|
189
|
+
sequence: number | string;
|
|
190
|
+
};
|
|
191
|
+
export type WebsocketReplayChannelRow = {
|
|
192
|
+
/**
|
|
193
|
+
* - Channel name.
|
|
194
|
+
*/
|
|
195
|
+
channel: string;
|
|
196
|
+
};
|
|
197
|
+
import Logger from "../logger.js";
|
|
198
|
+
//# sourceMappingURL=websocket-event-log-store.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
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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"}
|
|
@@ -0,0 +1,308 @@
|
|
|
1
|
+
// @ts-check
|
|
2
|
+
import { randomUUID } from "crypto";
|
|
3
|
+
import TableData from "../database/table-data/index.js";
|
|
4
|
+
import Logger from "../logger.js";
|
|
5
|
+
const EVENTS_TABLE = "websocket_channel_events";
|
|
6
|
+
const REPLAY_CHANNELS_TABLE = "websocket_replay_channels";
|
|
7
|
+
const DEFAULT_RETENTION_MS = 10 * 60 * 1000;
|
|
8
|
+
const stores = new WeakMap();
|
|
9
|
+
/**
|
|
10
|
+
* @typedef {object} WebsocketEventRow
|
|
11
|
+
* @property {string} channel - Channel name.
|
|
12
|
+
* @property {Date | string} created_at - Creation time.
|
|
13
|
+
* @property {string} id - Event id.
|
|
14
|
+
* @property {string} payload_json - Serialized payload.
|
|
15
|
+
* @property {number | string} sequence - Sequence number.
|
|
16
|
+
*/
|
|
17
|
+
/**
|
|
18
|
+
* @typedef {object} WebsocketReplayChannelRow
|
|
19
|
+
* @property {string} channel - Channel name.
|
|
20
|
+
*/
|
|
21
|
+
/**
|
|
22
|
+
* @param {import("../configuration.js").default} configuration - Configuration.
|
|
23
|
+
* @returns {VelociousHttpServerWebsocketEventLogStore} - Shared store instance.
|
|
24
|
+
*/
|
|
25
|
+
export function websocketEventLogStoreForConfiguration(configuration) {
|
|
26
|
+
let store = stores.get(configuration);
|
|
27
|
+
if (!store) {
|
|
28
|
+
store = new VelociousHttpServerWebsocketEventLogStore({ configuration });
|
|
29
|
+
stores.set(configuration, store);
|
|
30
|
+
}
|
|
31
|
+
return store;
|
|
32
|
+
}
|
|
33
|
+
export default class VelociousHttpServerWebsocketEventLogStore {
|
|
34
|
+
/**
|
|
35
|
+
* @param {object} args - Options.
|
|
36
|
+
* @param {import("../configuration.js").default} args.configuration - Configuration.
|
|
37
|
+
* @param {string} [args.databaseIdentifier] - Database identifier.
|
|
38
|
+
* @param {number} [args.retentionMs] - Event retention in milliseconds.
|
|
39
|
+
*/
|
|
40
|
+
constructor({ configuration, databaseIdentifier = "default", retentionMs = DEFAULT_RETENTION_MS }) {
|
|
41
|
+
this.configuration = configuration;
|
|
42
|
+
this.databaseIdentifier = databaseIdentifier;
|
|
43
|
+
this.retentionMs = retentionMs;
|
|
44
|
+
this.logger = new Logger(this);
|
|
45
|
+
this._isReady = false;
|
|
46
|
+
this._readyPromise = null;
|
|
47
|
+
}
|
|
48
|
+
/**
|
|
49
|
+
* @returns {Promise<void>} - Resolves when ready.
|
|
50
|
+
*/
|
|
51
|
+
async ensureReady() {
|
|
52
|
+
if (this._isReady)
|
|
53
|
+
return;
|
|
54
|
+
if (this._readyPromise)
|
|
55
|
+
return await this._readyPromise;
|
|
56
|
+
this._readyPromise = (async () => {
|
|
57
|
+
this.configuration.setCurrent();
|
|
58
|
+
await this._ensureSchema();
|
|
59
|
+
this._isReady = true;
|
|
60
|
+
})();
|
|
61
|
+
try {
|
|
62
|
+
await this._readyPromise;
|
|
63
|
+
}
|
|
64
|
+
finally {
|
|
65
|
+
if (!this._isReady) {
|
|
66
|
+
this._readyPromise = null;
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
/**
|
|
71
|
+
* @param {object} args - Options.
|
|
72
|
+
* @param {string} args.channel - Channel name.
|
|
73
|
+
* @param {unknown} args.payload - Event payload.
|
|
74
|
+
* @returns {Promise<{channel: string, createdAt: string, id: string, payload: unknown}>} - Persisted event row.
|
|
75
|
+
*/
|
|
76
|
+
async appendEvent({ channel, payload }) {
|
|
77
|
+
await this.ensureReady();
|
|
78
|
+
const id = randomUUID();
|
|
79
|
+
const createdAt = new Date();
|
|
80
|
+
return await this._withDb(async (db) => {
|
|
81
|
+
await db.insert({
|
|
82
|
+
tableName: EVENTS_TABLE,
|
|
83
|
+
data: {
|
|
84
|
+
channel,
|
|
85
|
+
created_at: createdAt,
|
|
86
|
+
id,
|
|
87
|
+
payload_json: JSON.stringify(payload)
|
|
88
|
+
}
|
|
89
|
+
});
|
|
90
|
+
return { channel, createdAt: createdAt.toISOString(), id, payload };
|
|
91
|
+
});
|
|
92
|
+
}
|
|
93
|
+
/**
|
|
94
|
+
* @param {string} channel - Channel name.
|
|
95
|
+
* @returns {Promise<void>} - Resolves when the channel interest was persisted.
|
|
96
|
+
*/
|
|
97
|
+
async markChannelInterested(channel) {
|
|
98
|
+
await this.ensureReady();
|
|
99
|
+
const interestedUntil = new Date(Date.now() + this.retentionMs);
|
|
100
|
+
await this._withDb(async (db) => {
|
|
101
|
+
await this._upsertReplayChannelInterest(db, { channel, interestedUntil });
|
|
102
|
+
});
|
|
103
|
+
}
|
|
104
|
+
/**
|
|
105
|
+
* @param {string} channel - Channel name.
|
|
106
|
+
* @returns {Promise<boolean>} - Whether the channel should be persisted for replay.
|
|
107
|
+
*/
|
|
108
|
+
async shouldPersistChannel(channel) {
|
|
109
|
+
await this.ensureReady();
|
|
110
|
+
return await this._withDb(async (db) => {
|
|
111
|
+
const rows = await db
|
|
112
|
+
.newQuery()
|
|
113
|
+
.from(REPLAY_CHANNELS_TABLE)
|
|
114
|
+
.where({ channel })
|
|
115
|
+
.where(`interested_until > ${db.quote(new Date())}`)
|
|
116
|
+
.limit(1)
|
|
117
|
+
.results();
|
|
118
|
+
return rows.length > 0;
|
|
119
|
+
});
|
|
120
|
+
}
|
|
121
|
+
/**
|
|
122
|
+
* @param {object} args - Options.
|
|
123
|
+
* @param {string} args.channel - Channel name.
|
|
124
|
+
* @param {string} args.id - Event id.
|
|
125
|
+
* @returns {Promise<{channel: string, createdAt: string, id: string, payload: unknown, sequence: number} | null>} - Event row or null.
|
|
126
|
+
*/
|
|
127
|
+
async getEventById({ channel, id }) {
|
|
128
|
+
await this.ensureReady();
|
|
129
|
+
return await this._withDb(async (db) => {
|
|
130
|
+
return await this._getEventById({ channel, db, id });
|
|
131
|
+
});
|
|
132
|
+
}
|
|
133
|
+
/**
|
|
134
|
+
* @param {string} channel - Channel name.
|
|
135
|
+
* @returns {Promise<number | null>} - Latest channel sequence.
|
|
136
|
+
*/
|
|
137
|
+
async latestSequence(channel) {
|
|
138
|
+
await this.ensureReady();
|
|
139
|
+
return await this._withDb(async (db) => {
|
|
140
|
+
const rows = await db
|
|
141
|
+
.newQuery()
|
|
142
|
+
.from(EVENTS_TABLE)
|
|
143
|
+
.where({ channel })
|
|
144
|
+
.order("sequence DESC")
|
|
145
|
+
.limit(1)
|
|
146
|
+
.results();
|
|
147
|
+
const row = /** @type {Record<string, any> | undefined} */ (rows[0]);
|
|
148
|
+
if (!row)
|
|
149
|
+
return null;
|
|
150
|
+
return Number(row.sequence);
|
|
151
|
+
});
|
|
152
|
+
}
|
|
153
|
+
/**
|
|
154
|
+
* @param {object} args - Options.
|
|
155
|
+
* @param {string} args.channel - Channel name.
|
|
156
|
+
* @param {number} args.sequence - Lower bound sequence.
|
|
157
|
+
* @param {number | null | undefined} [args.upToSequence] - Inclusive ceiling sequence.
|
|
158
|
+
* @returns {Promise<Array<{channel: string, createdAt: string, id: string, payload: unknown, sequence: number}>>} - Ordered events.
|
|
159
|
+
*/
|
|
160
|
+
async getEventsAfter({ channel, sequence, upToSequence }) {
|
|
161
|
+
await this.ensureReady();
|
|
162
|
+
return await this._withDb(async (db) => {
|
|
163
|
+
const query = db
|
|
164
|
+
.newQuery()
|
|
165
|
+
.from(EVENTS_TABLE)
|
|
166
|
+
.where({ channel })
|
|
167
|
+
.where(`sequence > ${db.quote(sequence)}`)
|
|
168
|
+
.order("sequence ASC");
|
|
169
|
+
if (typeof upToSequence === "number") {
|
|
170
|
+
query.where(`sequence <= ${db.quote(upToSequence)}`);
|
|
171
|
+
}
|
|
172
|
+
const rows = /** @type {WebsocketEventRow[]} */ (await query.results());
|
|
173
|
+
return rows.map((row) => this._normalizeEventRow(row));
|
|
174
|
+
});
|
|
175
|
+
}
|
|
176
|
+
/**
|
|
177
|
+
* @param {object} [args] - Options.
|
|
178
|
+
* @param {Date} [args.now] - Cleanup reference time.
|
|
179
|
+
* @returns {Promise<void>} - Resolves when cleanup completes.
|
|
180
|
+
*/
|
|
181
|
+
async cleanupExpired({ now = new Date() } = {}) {
|
|
182
|
+
await this.ensureReady();
|
|
183
|
+
const cutoff = new Date(now.getTime() - this.retentionMs);
|
|
184
|
+
await this._withDb(async (db) => {
|
|
185
|
+
const expiredEventRows = /** @type {Array<{id: string}>} */ (await db
|
|
186
|
+
.newQuery()
|
|
187
|
+
.from(EVENTS_TABLE)
|
|
188
|
+
.where(`created_at <= ${db.quote(cutoff)}`)
|
|
189
|
+
.results());
|
|
190
|
+
const expiredReplayChannelRows = /** @type {WebsocketReplayChannelRow[]} */ (await db
|
|
191
|
+
.newQuery()
|
|
192
|
+
.from(REPLAY_CHANNELS_TABLE)
|
|
193
|
+
.where(`interested_until <= ${db.quote(now)}`)
|
|
194
|
+
.results());
|
|
195
|
+
for (const expiredEventRow of expiredEventRows) {
|
|
196
|
+
await db.delete({
|
|
197
|
+
tableName: EVENTS_TABLE,
|
|
198
|
+
conditions: { id: expiredEventRow.id }
|
|
199
|
+
});
|
|
200
|
+
}
|
|
201
|
+
for (const expiredReplayChannelRow of expiredReplayChannelRows) {
|
|
202
|
+
await db.delete({
|
|
203
|
+
tableName: REPLAY_CHANNELS_TABLE,
|
|
204
|
+
conditions: { channel: expiredReplayChannelRow.channel }
|
|
205
|
+
});
|
|
206
|
+
}
|
|
207
|
+
});
|
|
208
|
+
}
|
|
209
|
+
async _ensureSchema() {
|
|
210
|
+
await this._withDb(async (db) => {
|
|
211
|
+
await this._ensureEventsTable(db);
|
|
212
|
+
await this._ensureReplayChannelsTable(db);
|
|
213
|
+
});
|
|
214
|
+
}
|
|
215
|
+
/**
|
|
216
|
+
* @param {import("../database/drivers/base.js").default} db - Database connection.
|
|
217
|
+
* @returns {Promise<void>} - Resolves when complete.
|
|
218
|
+
*/
|
|
219
|
+
async _ensureEventsTable(db) {
|
|
220
|
+
this.logger.info("Applying websocket event-log schema");
|
|
221
|
+
if (await db.tableExists(EVENTS_TABLE)) {
|
|
222
|
+
this.logger.info("Websocket event-log table already exists - skipping create");
|
|
223
|
+
return;
|
|
224
|
+
}
|
|
225
|
+
const eventTable = new TableData(EVENTS_TABLE, { ifNotExists: true });
|
|
226
|
+
eventTable.integer("sequence", { autoIncrement: true, null: false, primaryKey: true });
|
|
227
|
+
eventTable.string("id", { index: true, null: false });
|
|
228
|
+
eventTable.string("channel", { index: true, null: false });
|
|
229
|
+
eventTable.text("payload_json", { null: false });
|
|
230
|
+
eventTable.datetime("created_at", { index: true, null: false });
|
|
231
|
+
await db.createTable(eventTable);
|
|
232
|
+
}
|
|
233
|
+
/**
|
|
234
|
+
* @param {import("../database/drivers/base.js").default} db - Database connection.
|
|
235
|
+
* @returns {Promise<void>} - Resolves when complete.
|
|
236
|
+
*/
|
|
237
|
+
async _ensureReplayChannelsTable(db) {
|
|
238
|
+
if (await db.tableExists(REPLAY_CHANNELS_TABLE))
|
|
239
|
+
return;
|
|
240
|
+
const replayChannelTable = new TableData(REPLAY_CHANNELS_TABLE, { ifNotExists: true });
|
|
241
|
+
replayChannelTable.string("channel", { null: false, primaryKey: true });
|
|
242
|
+
replayChannelTable.datetime("interested_until", { index: true, null: false });
|
|
243
|
+
await db.createTable(replayChannelTable);
|
|
244
|
+
}
|
|
245
|
+
/**
|
|
246
|
+
* @param {object} args - Options.
|
|
247
|
+
* @param {string} args.channel - Channel name.
|
|
248
|
+
* @param {import("../database/drivers/base.js").default} args.db - Database connection.
|
|
249
|
+
* @param {string} args.id - Event id.
|
|
250
|
+
* @returns {Promise<{channel: string, createdAt: string, id: string, payload: unknown, sequence: number} | null>} - Event row or null.
|
|
251
|
+
*/
|
|
252
|
+
async _getEventById({ channel, db, id }) {
|
|
253
|
+
const rows = /** @type {WebsocketEventRow[]} */ (await db
|
|
254
|
+
.newQuery()
|
|
255
|
+
.from(EVENTS_TABLE)
|
|
256
|
+
.where({ channel, id })
|
|
257
|
+
.limit(1)
|
|
258
|
+
.results());
|
|
259
|
+
if (!rows[0])
|
|
260
|
+
return null;
|
|
261
|
+
return this._normalizeEventRow(rows[0]);
|
|
262
|
+
}
|
|
263
|
+
/**
|
|
264
|
+
* @param {WebsocketEventRow} row - Raw row.
|
|
265
|
+
* @returns {{channel: string, createdAt: string, id: string, payload: unknown, sequence: number}} - Normalized row.
|
|
266
|
+
*/
|
|
267
|
+
_normalizeEventRow(row) {
|
|
268
|
+
const createdAtValue = row.created_at;
|
|
269
|
+
return {
|
|
270
|
+
channel: row.channel,
|
|
271
|
+
createdAt: createdAtValue instanceof Date ? createdAtValue.toISOString() : new Date(createdAtValue).toISOString(),
|
|
272
|
+
id: row.id,
|
|
273
|
+
payload: JSON.parse(row.payload_json),
|
|
274
|
+
sequence: Number(row.sequence)
|
|
275
|
+
};
|
|
276
|
+
}
|
|
277
|
+
/**
|
|
278
|
+
* @param {import("../database/drivers/base.js").default} db - Database connection.
|
|
279
|
+
* @param {object} args - Options.
|
|
280
|
+
* @param {string} args.channel - Channel name.
|
|
281
|
+
* @param {Date} args.interestedUntil - Retention deadline.
|
|
282
|
+
* @returns {Promise<void>} - Resolves when the replay-channel row was upserted.
|
|
283
|
+
*/
|
|
284
|
+
async _upsertReplayChannelInterest(db, { channel, interestedUntil }) {
|
|
285
|
+
await db.upsert({
|
|
286
|
+
conflictColumns: ["channel"],
|
|
287
|
+
data: {
|
|
288
|
+
channel,
|
|
289
|
+
interested_until: interestedUntil
|
|
290
|
+
},
|
|
291
|
+
tableName: REPLAY_CHANNELS_TABLE,
|
|
292
|
+
updateColumns: ["interested_until"]
|
|
293
|
+
});
|
|
294
|
+
}
|
|
295
|
+
/**
|
|
296
|
+
* @param {(db: import("../database/drivers/base.js").default) => Promise<any>} callback - Callback.
|
|
297
|
+
* @returns {Promise<any>} - Callback result.
|
|
298
|
+
*/
|
|
299
|
+
async _withDb(callback) {
|
|
300
|
+
return await this.configuration.ensureConnections(async (dbs) => {
|
|
301
|
+
const db = dbs[this.databaseIdentifier];
|
|
302
|
+
if (!db)
|
|
303
|
+
throw new Error(`No database connection available for identifier: ${this.databaseIdentifier}`);
|
|
304
|
+
return await callback(db);
|
|
305
|
+
});
|
|
306
|
+
}
|
|
307
|
+
}
|
|
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"]}
|