spacetimedb 2.5.0 → 2.6.0

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.
Files changed (180) hide show
  1. package/LICENSE.txt +759 -759
  2. package/README.md +211 -211
  3. package/dist/angular/index.cjs.map +1 -1
  4. package/dist/angular/index.mjs.map +1 -1
  5. package/dist/browser/angular/index.mjs.map +1 -1
  6. package/dist/browser/react/index.mjs +129 -57
  7. package/dist/browser/react/index.mjs.map +1 -1
  8. package/dist/browser/solid/index.mjs +120 -50
  9. package/dist/browser/solid/index.mjs.map +1 -1
  10. package/dist/browser/svelte/index.mjs.map +1 -1
  11. package/dist/browser/vue/index.mjs.map +1 -1
  12. package/dist/index.browser.mjs +10 -2
  13. package/dist/index.browser.mjs.map +1 -1
  14. package/dist/index.cjs +10 -2
  15. package/dist/index.cjs.map +1 -1
  16. package/dist/index.mjs +10 -2
  17. package/dist/index.mjs.map +1 -1
  18. package/dist/min/index.browser.mjs +1 -1
  19. package/dist/min/index.browser.mjs.map +1 -1
  20. package/dist/min/react/index.mjs +1 -1
  21. package/dist/min/react/index.mjs.map +1 -1
  22. package/dist/min/sdk/index.browser.mjs +1 -1
  23. package/dist/min/sdk/index.browser.mjs.map +1 -1
  24. package/dist/react/index.cjs +129 -57
  25. package/dist/react/index.cjs.map +1 -1
  26. package/dist/react/index.mjs +129 -57
  27. package/dist/react/index.mjs.map +1 -1
  28. package/dist/react/useTable.d.ts.map +1 -1
  29. package/dist/sdk/connection_manager.d.ts +8 -0
  30. package/dist/sdk/connection_manager.d.ts.map +1 -1
  31. package/dist/sdk/db_connection_impl.d.ts +7 -0
  32. package/dist/sdk/db_connection_impl.d.ts.map +1 -1
  33. package/dist/sdk/index.browser.mjs +10 -2
  34. package/dist/sdk/index.browser.mjs.map +1 -1
  35. package/dist/sdk/index.cjs +10 -2
  36. package/dist/sdk/index.cjs.map +1 -1
  37. package/dist/sdk/index.mjs +10 -2
  38. package/dist/sdk/index.mjs.map +1 -1
  39. package/dist/sdk/websocket_test_adapter.d.ts +2 -1
  40. package/dist/sdk/websocket_test_adapter.d.ts.map +1 -1
  41. package/dist/server/index.mjs.map +1 -1
  42. package/dist/server/runtime.d.ts.map +1 -1
  43. package/dist/solid/index.cjs +120 -50
  44. package/dist/solid/index.cjs.map +1 -1
  45. package/dist/solid/index.mjs +120 -50
  46. package/dist/solid/index.mjs.map +1 -1
  47. package/dist/svelte/index.cjs.map +1 -1
  48. package/dist/svelte/index.mjs.map +1 -1
  49. package/dist/tanstack/index.cjs +120 -50
  50. package/dist/tanstack/index.cjs.map +1 -1
  51. package/dist/tanstack/index.mjs +120 -50
  52. package/dist/tanstack/index.mjs.map +1 -1
  53. package/dist/vue/index.cjs.map +1 -1
  54. package/dist/vue/index.mjs.map +1 -1
  55. package/package.json +1 -1
  56. package/src/angular/connection_state.ts +19 -19
  57. package/src/angular/index.ts +3 -3
  58. package/src/angular/injectors/index.ts +4 -4
  59. package/src/angular/injectors/inject-reducer.ts +62 -62
  60. package/src/angular/injectors/inject-spacetimedb-connected.ts +13 -13
  61. package/src/angular/injectors/inject-spacetimedb.ts +10 -10
  62. package/src/angular/injectors/inject-table.ts +234 -234
  63. package/src/angular/providers/index.ts +1 -1
  64. package/src/angular/providers/provide-spacetimedb.ts +96 -96
  65. package/src/index.ts +16 -16
  66. package/src/lib/algebraic_type.ts +819 -819
  67. package/src/lib/algebraic_type_variants.ts +26 -26
  68. package/src/lib/algebraic_value.ts +10 -10
  69. package/src/lib/autogen/types.ts +746 -746
  70. package/src/lib/binary_reader.ts +188 -188
  71. package/src/lib/binary_writer.ts +213 -213
  72. package/src/lib/connection_id.ts +102 -102
  73. package/src/lib/constraints.ts +48 -48
  74. package/src/lib/errors.ts +26 -26
  75. package/src/lib/filter.ts +195 -195
  76. package/src/lib/identity.ts +83 -83
  77. package/src/lib/indexes.ts +251 -251
  78. package/src/lib/option.ts +34 -34
  79. package/src/lib/query.ts +1019 -1019
  80. package/src/lib/reducer_schema.ts +38 -38
  81. package/src/lib/reducers.ts +116 -116
  82. package/src/lib/result.ts +36 -36
  83. package/src/lib/schedule_at.ts +86 -86
  84. package/src/lib/schema.ts +420 -420
  85. package/src/lib/table.ts +548 -548
  86. package/src/lib/table_schema.ts +64 -64
  87. package/src/lib/time_duration.ts +77 -77
  88. package/src/lib/timestamp.ts +148 -148
  89. package/src/lib/type_builders.test-d.ts +128 -128
  90. package/src/lib/type_builders.ts +4014 -4014
  91. package/src/lib/type_util.ts +124 -124
  92. package/src/lib/util.ts +196 -196
  93. package/src/lib/uuid.ts +337 -337
  94. package/src/react/SpacetimeDBProvider.ts +84 -84
  95. package/src/react/connection_state.ts +6 -6
  96. package/src/react/index.ts +5 -5
  97. package/src/react/useProcedure.ts +60 -60
  98. package/src/react/useReducer.ts +53 -53
  99. package/src/react/useSpacetimeDB.ts +18 -18
  100. package/src/react/useTable.ts +256 -251
  101. package/src/sdk/client_api/index.ts +114 -114
  102. package/src/sdk/client_api/types/procedures.ts +8 -8
  103. package/src/sdk/client_api/types/reducers.ts +8 -8
  104. package/src/sdk/client_api/types.ts +288 -288
  105. package/src/sdk/client_cache.ts +129 -129
  106. package/src/sdk/client_table.ts +179 -179
  107. package/src/sdk/connection_manager.ts +352 -237
  108. package/src/sdk/db_connection_builder.ts +290 -290
  109. package/src/sdk/db_connection_impl.ts +1356 -1347
  110. package/src/sdk/db_context.ts +28 -28
  111. package/src/sdk/db_view.ts +12 -12
  112. package/src/sdk/decompress.ts +51 -51
  113. package/src/sdk/event.ts +18 -18
  114. package/src/sdk/event_context.ts +51 -51
  115. package/src/sdk/event_emitter.ts +32 -32
  116. package/src/sdk/index.ts +14 -14
  117. package/src/sdk/internal.ts +2 -2
  118. package/src/sdk/json_api.ts +46 -46
  119. package/src/sdk/logger.ts +134 -134
  120. package/src/sdk/message_types.ts +46 -46
  121. package/src/sdk/procedures.ts +83 -83
  122. package/src/sdk/reducer_event.ts +20 -20
  123. package/src/sdk/reducer_handle.ts +12 -12
  124. package/src/sdk/reducers.ts +159 -159
  125. package/src/sdk/schema.ts +45 -45
  126. package/src/sdk/spacetime_module.ts +28 -28
  127. package/src/sdk/subscription_builder_impl.ts +275 -275
  128. package/src/sdk/table_cache.ts +581 -581
  129. package/src/sdk/type_utils.ts +19 -19
  130. package/src/sdk/version.ts +133 -133
  131. package/src/sdk/websocket_decompress_adapter.ts +63 -63
  132. package/src/sdk/websocket_protocols.ts +25 -25
  133. package/src/sdk/websocket_test_adapter.ts +107 -100
  134. package/src/sdk/websocket_v3_frames.ts +126 -126
  135. package/src/sdk/ws.ts +105 -105
  136. package/src/server/console.ts +81 -81
  137. package/src/server/db_view.ts +21 -21
  138. package/src/server/errors.ts +138 -138
  139. package/src/server/http.test-d.ts +80 -80
  140. package/src/server/http.ts +14 -14
  141. package/src/server/http_handlers.ts +413 -413
  142. package/src/server/http_internal.ts +79 -79
  143. package/src/server/http_shared.ts +186 -186
  144. package/src/server/index.ts +37 -37
  145. package/src/server/polyfills.ts +4 -4
  146. package/src/server/procedures.ts +239 -239
  147. package/src/server/query.ts +1 -1
  148. package/src/server/range.ts +53 -53
  149. package/src/server/reducers.ts +113 -113
  150. package/src/server/rng.ts +113 -113
  151. package/src/server/runtime.ts +1102 -1102
  152. package/src/server/schema.test-d.ts +99 -99
  153. package/src/server/schema.ts +663 -663
  154. package/src/server/sys.d.ts +125 -125
  155. package/src/server/view.test-d.ts +194 -194
  156. package/src/server/views.ts +340 -340
  157. package/src/solid/SpacetimeDBProvider.ts +97 -97
  158. package/src/solid/connection_state.ts +6 -6
  159. package/src/solid/index.ts +5 -5
  160. package/src/solid/useProcedure.ts +57 -57
  161. package/src/solid/useReducer.ts +50 -50
  162. package/src/solid/useSpacetimeDB.ts +18 -18
  163. package/src/solid/useTable.ts +203 -203
  164. package/src/svelte/SpacetimeDBProvider.ts +101 -101
  165. package/src/svelte/connection_state.ts +16 -16
  166. package/src/svelte/index.ts +4 -4
  167. package/src/svelte/useReducer.ts +61 -61
  168. package/src/svelte/useSpacetimeDB.ts +22 -22
  169. package/src/svelte/useTable.ts +218 -218
  170. package/src/tanstack/SpacetimeDBQueryClient.ts +330 -330
  171. package/src/tanstack/hooks.ts +83 -83
  172. package/src/tanstack/index.ts +16 -16
  173. package/src/util-stub.ts +1 -1
  174. package/src/vue/SpacetimeDBProvider.ts +157 -157
  175. package/src/vue/connection_state.ts +19 -19
  176. package/src/vue/index.ts +5 -5
  177. package/src/vue/useProcedure.ts +62 -62
  178. package/src/vue/useReducer.ts +55 -55
  179. package/src/vue/useSpacetimeDB.ts +18 -18
  180. package/src/vue/useTable.ts +229 -229
@@ -1,275 +1,275 @@
1
- import type { DbConnectionImpl } from './db_connection_impl';
2
- import { INTERNAL_REMOTE_MODULE } from './internal';
3
- import type {
4
- ErrorContextInterface,
5
- SubscriptionEventContextInterface,
6
- } from './event_context';
7
- import { EventEmitter } from './event_emitter';
8
- import type { UntypedRemoteModule } from './spacetime_module';
9
- import { isRowTypedQuery, toSql, type RowTypedQuery } from '../lib/query';
10
- import type { Values } from '../lib/type_util';
11
-
12
- export class SubscriptionBuilderImpl<RemoteModule extends UntypedRemoteModule> {
13
- #onApplied?: (ctx: SubscriptionEventContextInterface<RemoteModule>) => void =
14
- undefined;
15
- #onError?: (ctx: ErrorContextInterface<RemoteModule>) => void = undefined;
16
- constructor(private db: DbConnectionImpl<RemoteModule>) {}
17
-
18
- /**
19
- * Registers `callback` to run when this query is successfully added to our subscribed set,
20
- * I.e. when its `SubscriptionApplied` message is received.
21
- *
22
- * The database state exposed via the `&EventContext` argument
23
- * includes all the rows added to the client cache as a result of the new subscription.
24
- *
25
- * The event in the `&EventContext` argument is `Event::SubscribeApplied`.
26
- *
27
- * Multiple `on_applied` callbacks for the same query may coexist.
28
- * No mechanism for un-registering `on_applied` callbacks is exposed.
29
- *
30
- * @param cb - Callback to run when the subscription is applied.
31
- * @returns The current `SubscriptionBuilder` instance.
32
- */
33
- onApplied(
34
- cb: (ctx: SubscriptionEventContextInterface<RemoteModule>) => void
35
- ): SubscriptionBuilderImpl<RemoteModule> {
36
- this.#onApplied = cb;
37
- return this;
38
- }
39
-
40
- /**
41
- * Registers `callback` to run when this query either:
42
- * - Fails to be added to our subscribed set.
43
- * - Is unexpectedly removed from our subscribed set.
44
- *
45
- * If the subscription had previously started and has been unexpectedly removed,
46
- * the database state exposed via the `&EventContext` argument contains no rows
47
- * from any subscriptions removed within the same error event.
48
- * As proposed, it must therefore contain no rows.
49
- *
50
- * The event in the `&EventContext` argument is `Event::SubscribeError`,
51
- * containing a dynamic error object with a human-readable description of the error
52
- * for diagnostic purposes.
53
- *
54
- * Multiple `on_error` callbacks for the same query may coexist.
55
- * No mechanism for un-registering `on_error` callbacks is exposed.
56
- *
57
- * @param cb - Callback to run when there is an error in subscription.
58
- * @returns The current `SubscriptionBuilder` instance.
59
- */
60
- onError(
61
- cb: (ctx: ErrorContextInterface<RemoteModule>) => void
62
- ): SubscriptionBuilderImpl<RemoteModule> {
63
- this.#onError = cb;
64
- return this;
65
- }
66
-
67
- /**
68
- * Subscribe to a single query. The results of the query will be merged into the client
69
- * cache and deduplicated on the client.
70
- *
71
- * @param query_sql A `SQL` query to subscribe to.
72
- *
73
- * @example
74
- *
75
- * ```ts
76
- * const subscription = connection.subscriptionBuilder().onApplied(() => {
77
- * console.log("SDK client cache initialized.");
78
- * }).subscribe("SELECT * FROM User");
79
- *
80
- * subscription.unsubscribe();
81
- * ```
82
- */
83
- subscribe(
84
- query_sql: string | RowTypedQuery<any, any>
85
- ): SubscriptionHandleImpl<RemoteModule>;
86
- subscribe(
87
- query_sql: Array<string | RowTypedQuery<any, any>>
88
- ): SubscriptionHandleImpl<RemoteModule>;
89
- subscribe(
90
- queryFn: (
91
- tables: Values<RemoteModule['tables']>
92
- ) => RowTypedQuery<any, any> | RowTypedQuery<any, any>[]
93
- ): SubscriptionHandleImpl<RemoteModule>;
94
- subscribe(
95
- query_sql:
96
- | string
97
- | RowTypedQuery<any, any>
98
- | Array<string | RowTypedQuery<any, any>>
99
- | ((tables: any) => RowTypedQuery<any, any> | RowTypedQuery<any, any>[])
100
- ): SubscriptionHandleImpl<RemoteModule> {
101
- let queries: Array<string | RowTypedQuery<any, any>>;
102
- if (typeof query_sql === 'function') {
103
- const tablesMap = this.db.getTablesMap?.();
104
- const result = query_sql(tablesMap);
105
- queries = Array.isArray(result) ? result : [result];
106
- } else {
107
- queries = Array.isArray(query_sql) ? query_sql : [query_sql];
108
- }
109
- if (queries.length === 0) {
110
- throw new Error('Subscriptions must have at least one query');
111
- }
112
- const queryStrings = queries.map(q => {
113
- if (typeof q === 'string') return q;
114
- if (isRowTypedQuery(q)) return toSql(q);
115
- throw new Error('Subscriptions must be SQL strings or typed queries');
116
- });
117
- return new SubscriptionHandleImpl(
118
- this.db,
119
- queryStrings,
120
- this.#onApplied,
121
- this.#onError
122
- );
123
- }
124
-
125
- /**
126
- * Subscribes to all rows from all tables.
127
- *
128
- * This method is intended as a convenience
129
- * for applications where client-side memory use and network bandwidth are not concerns.
130
- * Applications where these resources are a constraint
131
- * should register more precise queries via `subscribe`
132
- * in order to replicate only the subset of data which the client needs to function.
133
- *
134
- * This method should not be combined with `subscribe` on the same `DbConnection`.
135
- * A connection may either `subscribe` to particular queries,
136
- * or `subscribeToAllTables`, but not both.
137
- * Attempting to call `subscribe`
138
- * on a `DbConnection` that has previously used `subscribeToAllTables`,
139
- * or vice versa, may misbehave in any number of ways,
140
- * including dropping subscriptions, corrupting the client cache, or throwing errors.
141
- */
142
- subscribeToAllTables(): void {
143
- const remoteModule = this.db[INTERNAL_REMOTE_MODULE]();
144
- const queries = Object.values(remoteModule.tables).map(
145
- table => `SELECT * FROM ${table.sourceName}`
146
- );
147
- this.subscribe(queries);
148
- }
149
- }
150
-
151
- export type SubscribeEvent = 'applied' | 'error' | 'end';
152
-
153
- export class SubscriptionManager<RemoteModule extends UntypedRemoteModule> {
154
- subscriptions: Map<
155
- number,
156
- {
157
- handle: SubscriptionHandleImpl<RemoteModule>;
158
- emitter: EventEmitter<SubscribeEvent>;
159
- }
160
- > = new Map();
161
- }
162
-
163
- export class SubscriptionHandleImpl<RemoteModule extends UntypedRemoteModule> {
164
- #querySetId: number;
165
- #unsubscribeCalled: boolean = false;
166
- #endedState: boolean = false;
167
- #activeState: boolean = false;
168
- #emitter: EventEmitter<SubscribeEvent, (...args: any[]) => void> =
169
- new EventEmitter();
170
-
171
- constructor(
172
- private db: DbConnectionImpl<RemoteModule>,
173
- querySql: string[],
174
- onApplied?: (ctx: SubscriptionEventContextInterface<RemoteModule>) => void,
175
- onError?: (ctx: ErrorContextInterface<RemoteModule>, error: Error) => void
176
- ) {
177
- this.#emitter.on(
178
- 'applied',
179
- (ctx: SubscriptionEventContextInterface<RemoteModule>) => {
180
- this.#activeState = true;
181
- if (onApplied) {
182
- onApplied(ctx);
183
- }
184
- }
185
- );
186
- this.#emitter.on(
187
- 'error',
188
- (ctx: ErrorContextInterface<RemoteModule>, error: Error) => {
189
- this.#activeState = false;
190
- this.#endedState = true;
191
- if (onError) {
192
- onError(ctx, error);
193
- }
194
- }
195
- );
196
- this.#querySetId = this.db.registerSubscription(
197
- this,
198
- this.#emitter,
199
- querySql
200
- );
201
- }
202
-
203
- /**
204
- * Consumes self and issues an `Unsubscribe` message,
205
- * removing this query from the client's set of subscribed queries.
206
- * It is only valid to call this method if `is_active()` is `true`.
207
- */
208
- unsubscribe(): void {
209
- if (this.#unsubscribeCalled) {
210
- throw new Error('Unsubscribe has already been called');
211
- }
212
- this.#unsubscribeCalled = true;
213
- this.db.unregisterSubscription(this.#querySetId);
214
- this.#emitter.on(
215
- 'end',
216
- (_ctx: SubscriptionEventContextInterface<RemoteModule>) => {
217
- this.#endedState = true;
218
- this.#activeState = false;
219
- }
220
- );
221
- }
222
-
223
- /**
224
- * Unsubscribes and also registers a callback to run upon success.
225
- * I.e. when an `UnsubscribeApplied` message is received.
226
- *
227
- * If `Unsubscribe` returns an error,
228
- * or if the `on_error` callback(s) are invoked before this subscription would end normally,
229
- * the `on_end` callback is not invoked.
230
- *
231
- * @param onEnd - Callback to run upon successful unsubscribe.
232
- */
233
- unsubscribeThen(
234
- onEnd: (ctx: SubscriptionEventContextInterface<RemoteModule>) => void
235
- ): void {
236
- if (this.#endedState) {
237
- throw new Error('Subscription has already ended');
238
- }
239
- if (this.#unsubscribeCalled) {
240
- throw new Error('Unsubscribe has already been called');
241
- }
242
- this.#unsubscribeCalled = true;
243
- this.db.unregisterSubscription(this.#querySetId);
244
- this.#emitter.on(
245
- 'end',
246
- (ctx: SubscriptionEventContextInterface<RemoteModule>) => {
247
- this.#endedState = true;
248
- this.#activeState = false;
249
- onEnd(ctx);
250
- }
251
- );
252
- }
253
-
254
- /**
255
- * True if this `SubscriptionHandle` has ended,
256
- * either due to an error or a call to `unsubscribe`.
257
- *
258
- * This is initially false, and becomes true when either the `on_end` or `on_error` callback is invoked.
259
- * A subscription which has not yet been applied is not active, but is also not ended.
260
- */
261
- isEnded(): boolean {
262
- return this.#endedState;
263
- }
264
-
265
- /**
266
- * True if this `SubscriptionHandle` is active, meaning it has been successfully applied
267
- * and has not since ended, either due to an error or a complete `unsubscribe` request-response pair.
268
- *
269
- * This corresponds exactly to the interval bounded at the start by the `on_applied` callback
270
- * and at the end by either the `on_end` or `on_error` callback.
271
- */
272
- isActive(): boolean {
273
- return this.#activeState;
274
- }
275
- }
1
+ import type { DbConnectionImpl } from './db_connection_impl';
2
+ import { INTERNAL_REMOTE_MODULE } from './internal';
3
+ import type {
4
+ ErrorContextInterface,
5
+ SubscriptionEventContextInterface,
6
+ } from './event_context';
7
+ import { EventEmitter } from './event_emitter';
8
+ import type { UntypedRemoteModule } from './spacetime_module';
9
+ import { isRowTypedQuery, toSql, type RowTypedQuery } from '../lib/query';
10
+ import type { Values } from '../lib/type_util';
11
+
12
+ export class SubscriptionBuilderImpl<RemoteModule extends UntypedRemoteModule> {
13
+ #onApplied?: (ctx: SubscriptionEventContextInterface<RemoteModule>) => void =
14
+ undefined;
15
+ #onError?: (ctx: ErrorContextInterface<RemoteModule>) => void = undefined;
16
+ constructor(private db: DbConnectionImpl<RemoteModule>) {}
17
+
18
+ /**
19
+ * Registers `callback` to run when this query is successfully added to our subscribed set,
20
+ * I.e. when its `SubscriptionApplied` message is received.
21
+ *
22
+ * The database state exposed via the `&EventContext` argument
23
+ * includes all the rows added to the client cache as a result of the new subscription.
24
+ *
25
+ * The event in the `&EventContext` argument is `Event::SubscribeApplied`.
26
+ *
27
+ * Multiple `on_applied` callbacks for the same query may coexist.
28
+ * No mechanism for un-registering `on_applied` callbacks is exposed.
29
+ *
30
+ * @param cb - Callback to run when the subscription is applied.
31
+ * @returns The current `SubscriptionBuilder` instance.
32
+ */
33
+ onApplied(
34
+ cb: (ctx: SubscriptionEventContextInterface<RemoteModule>) => void
35
+ ): SubscriptionBuilderImpl<RemoteModule> {
36
+ this.#onApplied = cb;
37
+ return this;
38
+ }
39
+
40
+ /**
41
+ * Registers `callback` to run when this query either:
42
+ * - Fails to be added to our subscribed set.
43
+ * - Is unexpectedly removed from our subscribed set.
44
+ *
45
+ * If the subscription had previously started and has been unexpectedly removed,
46
+ * the database state exposed via the `&EventContext` argument contains no rows
47
+ * from any subscriptions removed within the same error event.
48
+ * As proposed, it must therefore contain no rows.
49
+ *
50
+ * The event in the `&EventContext` argument is `Event::SubscribeError`,
51
+ * containing a dynamic error object with a human-readable description of the error
52
+ * for diagnostic purposes.
53
+ *
54
+ * Multiple `on_error` callbacks for the same query may coexist.
55
+ * No mechanism for un-registering `on_error` callbacks is exposed.
56
+ *
57
+ * @param cb - Callback to run when there is an error in subscription.
58
+ * @returns The current `SubscriptionBuilder` instance.
59
+ */
60
+ onError(
61
+ cb: (ctx: ErrorContextInterface<RemoteModule>) => void
62
+ ): SubscriptionBuilderImpl<RemoteModule> {
63
+ this.#onError = cb;
64
+ return this;
65
+ }
66
+
67
+ /**
68
+ * Subscribe to a single query. The results of the query will be merged into the client
69
+ * cache and deduplicated on the client.
70
+ *
71
+ * @param query_sql A `SQL` query to subscribe to.
72
+ *
73
+ * @example
74
+ *
75
+ * ```ts
76
+ * const subscription = connection.subscriptionBuilder().onApplied(() => {
77
+ * console.log("SDK client cache initialized.");
78
+ * }).subscribe("SELECT * FROM User");
79
+ *
80
+ * subscription.unsubscribe();
81
+ * ```
82
+ */
83
+ subscribe(
84
+ query_sql: string | RowTypedQuery<any, any>
85
+ ): SubscriptionHandleImpl<RemoteModule>;
86
+ subscribe(
87
+ query_sql: Array<string | RowTypedQuery<any, any>>
88
+ ): SubscriptionHandleImpl<RemoteModule>;
89
+ subscribe(
90
+ queryFn: (
91
+ tables: Values<RemoteModule['tables']>
92
+ ) => RowTypedQuery<any, any> | RowTypedQuery<any, any>[]
93
+ ): SubscriptionHandleImpl<RemoteModule>;
94
+ subscribe(
95
+ query_sql:
96
+ | string
97
+ | RowTypedQuery<any, any>
98
+ | Array<string | RowTypedQuery<any, any>>
99
+ | ((tables: any) => RowTypedQuery<any, any> | RowTypedQuery<any, any>[])
100
+ ): SubscriptionHandleImpl<RemoteModule> {
101
+ let queries: Array<string | RowTypedQuery<any, any>>;
102
+ if (typeof query_sql === 'function') {
103
+ const tablesMap = this.db.getTablesMap?.();
104
+ const result = query_sql(tablesMap);
105
+ queries = Array.isArray(result) ? result : [result];
106
+ } else {
107
+ queries = Array.isArray(query_sql) ? query_sql : [query_sql];
108
+ }
109
+ if (queries.length === 0) {
110
+ throw new Error('Subscriptions must have at least one query');
111
+ }
112
+ const queryStrings = queries.map(q => {
113
+ if (typeof q === 'string') return q;
114
+ if (isRowTypedQuery(q)) return toSql(q);
115
+ throw new Error('Subscriptions must be SQL strings or typed queries');
116
+ });
117
+ return new SubscriptionHandleImpl(
118
+ this.db,
119
+ queryStrings,
120
+ this.#onApplied,
121
+ this.#onError
122
+ );
123
+ }
124
+
125
+ /**
126
+ * Subscribes to all rows from all tables.
127
+ *
128
+ * This method is intended as a convenience
129
+ * for applications where client-side memory use and network bandwidth are not concerns.
130
+ * Applications where these resources are a constraint
131
+ * should register more precise queries via `subscribe`
132
+ * in order to replicate only the subset of data which the client needs to function.
133
+ *
134
+ * This method should not be combined with `subscribe` on the same `DbConnection`.
135
+ * A connection may either `subscribe` to particular queries,
136
+ * or `subscribeToAllTables`, but not both.
137
+ * Attempting to call `subscribe`
138
+ * on a `DbConnection` that has previously used `subscribeToAllTables`,
139
+ * or vice versa, may misbehave in any number of ways,
140
+ * including dropping subscriptions, corrupting the client cache, or throwing errors.
141
+ */
142
+ subscribeToAllTables(): void {
143
+ const remoteModule = this.db[INTERNAL_REMOTE_MODULE]();
144
+ const queries = Object.values(remoteModule.tables).map(
145
+ table => `SELECT * FROM ${table.sourceName}`
146
+ );
147
+ this.subscribe(queries);
148
+ }
149
+ }
150
+
151
+ export type SubscribeEvent = 'applied' | 'error' | 'end';
152
+
153
+ export class SubscriptionManager<RemoteModule extends UntypedRemoteModule> {
154
+ subscriptions: Map<
155
+ number,
156
+ {
157
+ handle: SubscriptionHandleImpl<RemoteModule>;
158
+ emitter: EventEmitter<SubscribeEvent>;
159
+ }
160
+ > = new Map();
161
+ }
162
+
163
+ export class SubscriptionHandleImpl<RemoteModule extends UntypedRemoteModule> {
164
+ #querySetId: number;
165
+ #unsubscribeCalled: boolean = false;
166
+ #endedState: boolean = false;
167
+ #activeState: boolean = false;
168
+ #emitter: EventEmitter<SubscribeEvent, (...args: any[]) => void> =
169
+ new EventEmitter();
170
+
171
+ constructor(
172
+ private db: DbConnectionImpl<RemoteModule>,
173
+ querySql: string[],
174
+ onApplied?: (ctx: SubscriptionEventContextInterface<RemoteModule>) => void,
175
+ onError?: (ctx: ErrorContextInterface<RemoteModule>, error: Error) => void
176
+ ) {
177
+ this.#emitter.on(
178
+ 'applied',
179
+ (ctx: SubscriptionEventContextInterface<RemoteModule>) => {
180
+ this.#activeState = true;
181
+ if (onApplied) {
182
+ onApplied(ctx);
183
+ }
184
+ }
185
+ );
186
+ this.#emitter.on(
187
+ 'error',
188
+ (ctx: ErrorContextInterface<RemoteModule>, error: Error) => {
189
+ this.#activeState = false;
190
+ this.#endedState = true;
191
+ if (onError) {
192
+ onError(ctx, error);
193
+ }
194
+ }
195
+ );
196
+ this.#querySetId = this.db.registerSubscription(
197
+ this,
198
+ this.#emitter,
199
+ querySql
200
+ );
201
+ }
202
+
203
+ /**
204
+ * Consumes self and issues an `Unsubscribe` message,
205
+ * removing this query from the client's set of subscribed queries.
206
+ * It is only valid to call this method if `is_active()` is `true`.
207
+ */
208
+ unsubscribe(): void {
209
+ if (this.#unsubscribeCalled) {
210
+ throw new Error('Unsubscribe has already been called');
211
+ }
212
+ this.#unsubscribeCalled = true;
213
+ this.db.unregisterSubscription(this.#querySetId);
214
+ this.#emitter.on(
215
+ 'end',
216
+ (_ctx: SubscriptionEventContextInterface<RemoteModule>) => {
217
+ this.#endedState = true;
218
+ this.#activeState = false;
219
+ }
220
+ );
221
+ }
222
+
223
+ /**
224
+ * Unsubscribes and also registers a callback to run upon success.
225
+ * I.e. when an `UnsubscribeApplied` message is received.
226
+ *
227
+ * If `Unsubscribe` returns an error,
228
+ * or if the `on_error` callback(s) are invoked before this subscription would end normally,
229
+ * the `on_end` callback is not invoked.
230
+ *
231
+ * @param onEnd - Callback to run upon successful unsubscribe.
232
+ */
233
+ unsubscribeThen(
234
+ onEnd: (ctx: SubscriptionEventContextInterface<RemoteModule>) => void
235
+ ): void {
236
+ if (this.#endedState) {
237
+ throw new Error('Subscription has already ended');
238
+ }
239
+ if (this.#unsubscribeCalled) {
240
+ throw new Error('Unsubscribe has already been called');
241
+ }
242
+ this.#unsubscribeCalled = true;
243
+ this.db.unregisterSubscription(this.#querySetId);
244
+ this.#emitter.on(
245
+ 'end',
246
+ (ctx: SubscriptionEventContextInterface<RemoteModule>) => {
247
+ this.#endedState = true;
248
+ this.#activeState = false;
249
+ onEnd(ctx);
250
+ }
251
+ );
252
+ }
253
+
254
+ /**
255
+ * True if this `SubscriptionHandle` has ended,
256
+ * either due to an error or a call to `unsubscribe`.
257
+ *
258
+ * This is initially false, and becomes true when either the `on_end` or `on_error` callback is invoked.
259
+ * A subscription which has not yet been applied is not active, but is also not ended.
260
+ */
261
+ isEnded(): boolean {
262
+ return this.#endedState;
263
+ }
264
+
265
+ /**
266
+ * True if this `SubscriptionHandle` is active, meaning it has been successfully applied
267
+ * and has not since ended, either due to an error or a complete `unsubscribe` request-response pair.
268
+ *
269
+ * This corresponds exactly to the interval bounded at the start by the `on_applied` callback
270
+ * and at the end by either the `on_end` or `on_error` callback.
271
+ */
272
+ isActive(): boolean {
273
+ return this.#activeState;
274
+ }
275
+ }