spacetimedb 2.4.1 → 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.
- package/LICENSE.txt +759 -759
- package/README.md +211 -120
- package/dist/angular/index.cjs.map +1 -1
- package/dist/angular/index.mjs.map +1 -1
- package/dist/browser/angular/index.mjs.map +1 -1
- package/dist/browser/react/index.mjs +129 -57
- package/dist/browser/react/index.mjs.map +1 -1
- package/dist/browser/solid/index.mjs +1933 -0
- package/dist/browser/solid/index.mjs.map +1 -0
- package/dist/browser/svelte/index.mjs.map +1 -1
- package/dist/browser/vue/index.mjs.map +1 -1
- package/dist/index.browser.mjs +10 -2
- package/dist/index.browser.mjs.map +1 -1
- package/dist/index.cjs +10 -2
- package/dist/index.cjs.map +1 -1
- package/dist/index.mjs +10 -2
- package/dist/index.mjs.map +1 -1
- package/dist/min/index.browser.mjs +1 -1
- package/dist/min/index.browser.mjs.map +1 -1
- package/dist/min/react/index.mjs +1 -1
- package/dist/min/react/index.mjs.map +1 -1
- package/dist/min/sdk/index.browser.mjs +1 -1
- package/dist/min/sdk/index.browser.mjs.map +1 -1
- package/dist/react/index.cjs +129 -57
- package/dist/react/index.cjs.map +1 -1
- package/dist/react/index.mjs +129 -57
- package/dist/react/index.mjs.map +1 -1
- package/dist/react/useTable.d.ts.map +1 -1
- package/dist/sdk/connection_manager.d.ts +8 -0
- package/dist/sdk/connection_manager.d.ts.map +1 -1
- package/dist/sdk/db_connection_impl.d.ts +7 -0
- package/dist/sdk/db_connection_impl.d.ts.map +1 -1
- package/dist/sdk/index.browser.mjs +10 -2
- package/dist/sdk/index.browser.mjs.map +1 -1
- package/dist/sdk/index.cjs +10 -2
- package/dist/sdk/index.cjs.map +1 -1
- package/dist/sdk/index.mjs +10 -2
- package/dist/sdk/index.mjs.map +1 -1
- package/dist/sdk/websocket_test_adapter.d.ts +2 -1
- package/dist/sdk/websocket_test_adapter.d.ts.map +1 -1
- package/dist/server/index.mjs.map +1 -1
- package/dist/server/runtime.d.ts.map +1 -1
- package/dist/solid/SpacetimeDBProvider.d.ts +7 -0
- package/dist/solid/SpacetimeDBProvider.d.ts.map +1 -0
- package/dist/solid/connection_state.d.ts +6 -0
- package/dist/solid/connection_state.d.ts.map +1 -0
- package/dist/solid/index.cjs +1939 -0
- package/dist/solid/index.cjs.map +1 -0
- package/dist/solid/index.d.ts +6 -0
- package/dist/solid/index.d.ts.map +1 -0
- package/dist/solid/index.mjs +1933 -0
- package/dist/solid/index.mjs.map +1 -0
- package/dist/solid/useProcedure.d.ts +4 -0
- package/dist/solid/useProcedure.d.ts.map +1 -0
- package/dist/solid/useReducer.d.ts +4 -0
- package/dist/solid/useReducer.d.ts.map +1 -0
- package/dist/solid/useSpacetimeDB.d.ts +4 -0
- package/dist/solid/useSpacetimeDB.d.ts.map +1 -0
- package/dist/solid/useTable.d.ts +32 -0
- package/dist/solid/useTable.d.ts.map +1 -0
- package/dist/svelte/index.cjs.map +1 -1
- package/dist/svelte/index.mjs.map +1 -1
- package/dist/tanstack/index.cjs +120 -50
- package/dist/tanstack/index.cjs.map +1 -1
- package/dist/tanstack/index.mjs +120 -50
- package/dist/tanstack/index.mjs.map +1 -1
- package/dist/vue/index.cjs.map +1 -1
- package/dist/vue/index.mjs.map +1 -1
- package/package.json +13 -3
- package/src/angular/connection_state.ts +19 -19
- package/src/angular/index.ts +3 -3
- package/src/angular/injectors/index.ts +4 -4
- package/src/angular/injectors/inject-reducer.ts +62 -62
- package/src/angular/injectors/inject-spacetimedb-connected.ts +13 -13
- package/src/angular/injectors/inject-spacetimedb.ts +10 -10
- package/src/angular/injectors/inject-table.ts +234 -234
- package/src/angular/providers/index.ts +1 -1
- package/src/angular/providers/provide-spacetimedb.ts +96 -96
- package/src/index.ts +16 -16
- package/src/lib/algebraic_type.ts +819 -819
- package/src/lib/algebraic_type_variants.ts +26 -26
- package/src/lib/algebraic_value.ts +10 -10
- package/src/lib/autogen/types.ts +746 -746
- package/src/lib/binary_reader.ts +188 -188
- package/src/lib/binary_writer.ts +213 -213
- package/src/lib/connection_id.ts +102 -102
- package/src/lib/constraints.ts +48 -48
- package/src/lib/errors.ts +26 -26
- package/src/lib/filter.ts +195 -195
- package/src/lib/identity.ts +83 -83
- package/src/lib/indexes.ts +251 -251
- package/src/lib/option.ts +34 -34
- package/src/lib/query.ts +1019 -1019
- package/src/lib/reducer_schema.ts +38 -38
- package/src/lib/reducers.ts +116 -116
- package/src/lib/result.ts +36 -36
- package/src/lib/schedule_at.ts +86 -86
- package/src/lib/schema.ts +420 -420
- package/src/lib/table.ts +548 -548
- package/src/lib/table_schema.ts +64 -64
- package/src/lib/time_duration.ts +77 -77
- package/src/lib/timestamp.ts +148 -148
- package/src/lib/type_builders.test-d.ts +128 -128
- package/src/lib/type_builders.ts +4014 -4014
- package/src/lib/type_util.ts +124 -124
- package/src/lib/util.ts +196 -196
- package/src/lib/uuid.ts +337 -337
- package/src/react/SpacetimeDBProvider.ts +84 -84
- package/src/react/connection_state.ts +6 -6
- package/src/react/index.ts +5 -5
- package/src/react/useProcedure.ts +60 -60
- package/src/react/useReducer.ts +53 -53
- package/src/react/useSpacetimeDB.ts +18 -18
- package/src/react/useTable.ts +256 -251
- package/src/sdk/client_api/index.ts +114 -114
- package/src/sdk/client_api/types/procedures.ts +8 -8
- package/src/sdk/client_api/types/reducers.ts +8 -8
- package/src/sdk/client_api/types.ts +288 -288
- package/src/sdk/client_cache.ts +129 -129
- package/src/sdk/client_table.ts +179 -179
- package/src/sdk/connection_manager.ts +352 -237
- package/src/sdk/db_connection_builder.ts +290 -290
- package/src/sdk/db_connection_impl.ts +1356 -1347
- package/src/sdk/db_context.ts +28 -28
- package/src/sdk/db_view.ts +12 -12
- package/src/sdk/decompress.ts +51 -51
- package/src/sdk/event.ts +18 -18
- package/src/sdk/event_context.ts +51 -51
- package/src/sdk/event_emitter.ts +32 -32
- package/src/sdk/index.ts +14 -14
- package/src/sdk/internal.ts +2 -2
- package/src/sdk/json_api.ts +46 -46
- package/src/sdk/logger.ts +134 -134
- package/src/sdk/message_types.ts +46 -46
- package/src/sdk/procedures.ts +83 -83
- package/src/sdk/reducer_event.ts +20 -20
- package/src/sdk/reducer_handle.ts +12 -12
- package/src/sdk/reducers.ts +159 -159
- package/src/sdk/schema.ts +45 -45
- package/src/sdk/spacetime_module.ts +28 -28
- package/src/sdk/subscription_builder_impl.ts +275 -275
- package/src/sdk/table_cache.ts +581 -581
- package/src/sdk/type_utils.ts +19 -19
- package/src/sdk/version.ts +133 -133
- package/src/sdk/websocket_decompress_adapter.ts +63 -63
- package/src/sdk/websocket_protocols.ts +25 -25
- package/src/sdk/websocket_test_adapter.ts +107 -100
- package/src/sdk/websocket_v3_frames.ts +126 -126
- package/src/sdk/ws.ts +105 -105
- package/src/server/console.ts +81 -81
- package/src/server/db_view.ts +21 -21
- package/src/server/errors.ts +138 -138
- package/src/server/http.test-d.ts +80 -80
- package/src/server/http.ts +14 -14
- package/src/server/http_handlers.ts +413 -413
- package/src/server/http_internal.ts +79 -79
- package/src/server/http_shared.ts +186 -186
- package/src/server/index.ts +37 -37
- package/src/server/polyfills.ts +4 -4
- package/src/server/procedures.ts +239 -239
- package/src/server/query.ts +1 -1
- package/src/server/range.ts +53 -53
- package/src/server/reducers.ts +113 -113
- package/src/server/rng.ts +113 -113
- package/src/server/runtime.ts +1102 -1102
- package/src/server/schema.test-d.ts +99 -99
- package/src/server/schema.ts +663 -663
- package/src/server/sys.d.ts +125 -125
- package/src/server/view.test-d.ts +194 -194
- package/src/server/views.ts +340 -340
- package/src/solid/SpacetimeDBProvider.ts +97 -0
- package/src/solid/connection_state.ts +6 -0
- package/src/solid/index.ts +5 -0
- package/src/solid/useProcedure.ts +57 -0
- package/src/solid/useReducer.ts +50 -0
- package/src/solid/useSpacetimeDB.ts +18 -0
- package/src/solid/useTable.ts +203 -0
- package/src/svelte/SpacetimeDBProvider.ts +101 -101
- package/src/svelte/connection_state.ts +16 -16
- package/src/svelte/index.ts +4 -4
- package/src/svelte/useReducer.ts +61 -61
- package/src/svelte/useSpacetimeDB.ts +22 -22
- package/src/svelte/useTable.ts +218 -218
- package/src/tanstack/SpacetimeDBQueryClient.ts +330 -330
- package/src/tanstack/hooks.ts +83 -83
- package/src/tanstack/index.ts +16 -16
- package/src/util-stub.ts +1 -1
- package/src/vue/SpacetimeDBProvider.ts +157 -157
- package/src/vue/connection_state.ts +19 -19
- package/src/vue/index.ts +5 -5
- package/src/vue/useProcedure.ts +62 -62
- package/src/vue/useReducer.ts +55 -55
- package/src/vue/useSpacetimeDB.ts +18 -18
- 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
|
+
}
|