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.
- package/LICENSE.txt +759 -759
- package/README.md +211 -211
- 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 +120 -50
- package/dist/browser/solid/index.mjs.map +1 -1
- 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/index.cjs +120 -50
- package/dist/solid/index.cjs.map +1 -1
- package/dist/solid/index.mjs +120 -50
- package/dist/solid/index.mjs.map +1 -1
- 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 +1 -1
- 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 -97
- package/src/solid/connection_state.ts +6 -6
- package/src/solid/index.ts +5 -5
- package/src/solid/useProcedure.ts +57 -57
- package/src/solid/useReducer.ts +50 -50
- package/src/solid/useSpacetimeDB.ts +18 -18
- package/src/solid/useTable.ts +203 -203
- 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,330 +1,330 @@
|
|
|
1
|
-
import type {
|
|
2
|
-
QueryClient,
|
|
3
|
-
QueryKey,
|
|
4
|
-
QueryFunction,
|
|
5
|
-
} from '@tanstack/react-query';
|
|
6
|
-
import {
|
|
7
|
-
type Query,
|
|
8
|
-
toSql,
|
|
9
|
-
type BooleanExpr,
|
|
10
|
-
evaluateBooleanExpr,
|
|
11
|
-
getQueryAccessorName,
|
|
12
|
-
getQueryWhereClause,
|
|
13
|
-
} from '../lib/query';
|
|
14
|
-
|
|
15
|
-
type QueryInput = Query<any>;
|
|
16
|
-
|
|
17
|
-
const queryRegistry = new Map<
|
|
18
|
-
string,
|
|
19
|
-
{ accessorName: string; whereExpr?: BooleanExpr<any> }
|
|
20
|
-
>();
|
|
21
|
-
|
|
22
|
-
export interface SpacetimeDBQueryOptions {
|
|
23
|
-
queryKey: readonly ['spacetimedb', string, string];
|
|
24
|
-
staleTime: number;
|
|
25
|
-
}
|
|
26
|
-
|
|
27
|
-
export interface SpacetimeDBQueryOptionsSkipped
|
|
28
|
-
extends SpacetimeDBQueryOptions {
|
|
29
|
-
enabled: false;
|
|
30
|
-
}
|
|
31
|
-
|
|
32
|
-
// creates query options for useQuery/useSuspenseQuery.
|
|
33
|
-
// useQuery(spacetimeDBQuery(tables.person));
|
|
34
|
-
// useQuery(spacetimeDBQuery(tables.user.where(r => r.online.eq(true))));
|
|
35
|
-
// useQuery(spacetimeDBQuery(condition ? tables.user : 'skip'));
|
|
36
|
-
export function spacetimeDBQuery(
|
|
37
|
-
queryOrSkip: 'skip'
|
|
38
|
-
): SpacetimeDBQueryOptionsSkipped;
|
|
39
|
-
|
|
40
|
-
export function spacetimeDBQuery(query: QueryInput): SpacetimeDBQueryOptions;
|
|
41
|
-
|
|
42
|
-
export function spacetimeDBQuery(
|
|
43
|
-
queryOrSkip: QueryInput | 'skip'
|
|
44
|
-
): SpacetimeDBQueryOptions | SpacetimeDBQueryOptionsSkipped {
|
|
45
|
-
if (queryOrSkip === 'skip') {
|
|
46
|
-
return {
|
|
47
|
-
queryKey: ['spacetimedb', '', 'skip'] as const,
|
|
48
|
-
staleTime: Infinity,
|
|
49
|
-
enabled: false,
|
|
50
|
-
};
|
|
51
|
-
}
|
|
52
|
-
|
|
53
|
-
const query = queryOrSkip;
|
|
54
|
-
const accessorName = getQueryAccessorName(query);
|
|
55
|
-
const whereExpr = getQueryWhereClause(query);
|
|
56
|
-
const querySql = toSql(query);
|
|
57
|
-
|
|
58
|
-
queryRegistry.set(querySql, { accessorName, whereExpr });
|
|
59
|
-
|
|
60
|
-
return {
|
|
61
|
-
queryKey: ['spacetimedb', accessorName, querySql] as const,
|
|
62
|
-
staleTime: Infinity,
|
|
63
|
-
};
|
|
64
|
-
}
|
|
65
|
-
|
|
66
|
-
interface SpacetimeConnection {
|
|
67
|
-
db: Record<string, any>;
|
|
68
|
-
subscriptionBuilder: () => {
|
|
69
|
-
onApplied: (cb: () => void) => any;
|
|
70
|
-
subscribe: (query: string) => { unsubscribe: () => void };
|
|
71
|
-
};
|
|
72
|
-
}
|
|
73
|
-
|
|
74
|
-
interface SubscriptionState {
|
|
75
|
-
unsubscribe: () => void;
|
|
76
|
-
tableInstance: any;
|
|
77
|
-
applied: boolean;
|
|
78
|
-
}
|
|
79
|
-
|
|
80
|
-
// push updates to cache via setQueryData when SpacetimeDB data changes
|
|
81
|
-
export class SpacetimeDBQueryClient {
|
|
82
|
-
private connection: SpacetimeConnection | null = null;
|
|
83
|
-
private queryClient: QueryClient | null = null;
|
|
84
|
-
private subscriptions = new Map<string, SubscriptionState>();
|
|
85
|
-
private pendingQueries = new Map<
|
|
86
|
-
string,
|
|
87
|
-
Array<{
|
|
88
|
-
resolve: (data: any[]) => void;
|
|
89
|
-
querySql: string;
|
|
90
|
-
whereExpr?: BooleanExpr<any>;
|
|
91
|
-
}>
|
|
92
|
-
>();
|
|
93
|
-
private cacheUnsubscribe: (() => void) | null = null;
|
|
94
|
-
|
|
95
|
-
// set connection, called on onConnect callback
|
|
96
|
-
setConnection(connection: SpacetimeConnection): void {
|
|
97
|
-
this.connection = connection;
|
|
98
|
-
this.processPendingQueries();
|
|
99
|
-
this.hydrateSubscriptions();
|
|
100
|
-
}
|
|
101
|
-
|
|
102
|
-
connect(queryClient: QueryClient): void {
|
|
103
|
-
this.queryClient = queryClient;
|
|
104
|
-
|
|
105
|
-
this.cacheUnsubscribe = queryClient
|
|
106
|
-
.getQueryCache()
|
|
107
|
-
.subscribe((event: any) => {
|
|
108
|
-
if (
|
|
109
|
-
event.type === 'removed' &&
|
|
110
|
-
event.query.queryKey[0] === 'spacetimedb'
|
|
111
|
-
) {
|
|
112
|
-
const keyStr = JSON.stringify(event.query.queryKey);
|
|
113
|
-
const sub = this.subscriptions.get(keyStr);
|
|
114
|
-
if (sub) {
|
|
115
|
-
sub.unsubscribe();
|
|
116
|
-
this.subscriptions.delete(keyStr);
|
|
117
|
-
}
|
|
118
|
-
}
|
|
119
|
-
});
|
|
120
|
-
}
|
|
121
|
-
|
|
122
|
-
queryFn: QueryFunction<any[], QueryKey> = async ({
|
|
123
|
-
queryKey,
|
|
124
|
-
}: {
|
|
125
|
-
queryKey: QueryKey;
|
|
126
|
-
}) => {
|
|
127
|
-
const keyStr = JSON.stringify(queryKey);
|
|
128
|
-
const [prefix, accessorName, querySql] = queryKey as [
|
|
129
|
-
string,
|
|
130
|
-
string,
|
|
131
|
-
string,
|
|
132
|
-
];
|
|
133
|
-
|
|
134
|
-
if (prefix !== 'spacetimedb') {
|
|
135
|
-
throw new Error(
|
|
136
|
-
`SpacetimeDBQueryClient can only handle spacetimedb queries, got: ${prefix}`
|
|
137
|
-
);
|
|
138
|
-
}
|
|
139
|
-
|
|
140
|
-
const registered = queryRegistry.get(querySql);
|
|
141
|
-
const whereExpr = registered?.whereExpr;
|
|
142
|
-
|
|
143
|
-
const existingSub = this.subscriptions.get(keyStr);
|
|
144
|
-
if (existingSub?.applied) {
|
|
145
|
-
return this.getTableData(existingSub.tableInstance, whereExpr);
|
|
146
|
-
}
|
|
147
|
-
|
|
148
|
-
// queue query if connection not ready yet
|
|
149
|
-
if (!this.connection) {
|
|
150
|
-
return new Promise<any[]>(resolve => {
|
|
151
|
-
const pending = this.pendingQueries.get(keyStr) || [];
|
|
152
|
-
pending.push({ resolve, querySql, whereExpr });
|
|
153
|
-
this.pendingQueries.set(keyStr, pending);
|
|
154
|
-
});
|
|
155
|
-
}
|
|
156
|
-
|
|
157
|
-
return this.setupSubscription(queryKey, accessorName, querySql, whereExpr);
|
|
158
|
-
};
|
|
159
|
-
|
|
160
|
-
private getTableData(
|
|
161
|
-
tableInstance: any,
|
|
162
|
-
whereExpr?: BooleanExpr<any>
|
|
163
|
-
): any[] {
|
|
164
|
-
const allRows = Array.from(tableInstance.iter());
|
|
165
|
-
if (whereExpr) {
|
|
166
|
-
return allRows.filter(row =>
|
|
167
|
-
evaluateBooleanExpr(whereExpr, row as Record<string, any>)
|
|
168
|
-
);
|
|
169
|
-
}
|
|
170
|
-
return allRows;
|
|
171
|
-
}
|
|
172
|
-
|
|
173
|
-
private setupSubscription(
|
|
174
|
-
queryKey: QueryKey,
|
|
175
|
-
accessorName: string,
|
|
176
|
-
querySql: string,
|
|
177
|
-
whereExpr?: BooleanExpr<any>
|
|
178
|
-
): Promise<any[]> {
|
|
179
|
-
if (!this.connection) {
|
|
180
|
-
return Promise.resolve([]);
|
|
181
|
-
}
|
|
182
|
-
|
|
183
|
-
const keyStr = JSON.stringify(queryKey);
|
|
184
|
-
const db = this.connection.db;
|
|
185
|
-
|
|
186
|
-
const tableInstance = db[accessorName];
|
|
187
|
-
|
|
188
|
-
if (!tableInstance) {
|
|
189
|
-
console.warn(`SpacetimeDBQueryClient: table "${accessorName}" not found`);
|
|
190
|
-
return Promise.resolve([]);
|
|
191
|
-
}
|
|
192
|
-
|
|
193
|
-
// return existing data if already subscribed
|
|
194
|
-
const existingSub = this.subscriptions.get(keyStr);
|
|
195
|
-
if (existingSub) {
|
|
196
|
-
if (existingSub.applied) {
|
|
197
|
-
return Promise.resolve(
|
|
198
|
-
this.getTableData(existingSub.tableInstance, whereExpr)
|
|
199
|
-
);
|
|
200
|
-
}
|
|
201
|
-
return new Promise(resolve => {
|
|
202
|
-
const pending = this.pendingQueries.get(keyStr) || [];
|
|
203
|
-
pending.push({ resolve, querySql, whereExpr });
|
|
204
|
-
this.pendingQueries.set(keyStr, pending);
|
|
205
|
-
});
|
|
206
|
-
}
|
|
207
|
-
|
|
208
|
-
return new Promise<any[]>(resolve => {
|
|
209
|
-
const updateCache = () => {
|
|
210
|
-
if (!this.queryClient) return [];
|
|
211
|
-
const data = this.getTableData(tableInstance, whereExpr);
|
|
212
|
-
this.queryClient.setQueryData(queryKey, data);
|
|
213
|
-
return data;
|
|
214
|
-
};
|
|
215
|
-
|
|
216
|
-
const handle = this.connection!.subscriptionBuilder()
|
|
217
|
-
.onApplied(() => {
|
|
218
|
-
const sub = this.subscriptions.get(keyStr);
|
|
219
|
-
if (sub) {
|
|
220
|
-
sub.applied = true;
|
|
221
|
-
}
|
|
222
|
-
|
|
223
|
-
const data = updateCache();
|
|
224
|
-
resolve(data);
|
|
225
|
-
|
|
226
|
-
const pending = this.pendingQueries.get(keyStr);
|
|
227
|
-
if (pending) {
|
|
228
|
-
for (const p of pending) {
|
|
229
|
-
p.resolve(data);
|
|
230
|
-
}
|
|
231
|
-
this.pendingQueries.delete(keyStr);
|
|
232
|
-
}
|
|
233
|
-
})
|
|
234
|
-
.subscribe(querySql);
|
|
235
|
-
|
|
236
|
-
// push updates to cache when data changes
|
|
237
|
-
const onTableChange = () => {
|
|
238
|
-
const sub = this.subscriptions.get(keyStr);
|
|
239
|
-
if (sub?.applied) {
|
|
240
|
-
updateCache();
|
|
241
|
-
}
|
|
242
|
-
};
|
|
243
|
-
|
|
244
|
-
tableInstance.onInsert(onTableChange);
|
|
245
|
-
tableInstance.onDelete(onTableChange);
|
|
246
|
-
tableInstance.onUpdate?.(onTableChange);
|
|
247
|
-
|
|
248
|
-
this.subscriptions.set(keyStr, {
|
|
249
|
-
unsubscribe: () => {
|
|
250
|
-
handle.unsubscribe();
|
|
251
|
-
tableInstance.removeOnInsert(onTableChange);
|
|
252
|
-
tableInstance.removeOnDelete(onTableChange);
|
|
253
|
-
tableInstance.removeOnUpdate?.(onTableChange);
|
|
254
|
-
},
|
|
255
|
-
tableInstance,
|
|
256
|
-
applied: false,
|
|
257
|
-
});
|
|
258
|
-
});
|
|
259
|
-
}
|
|
260
|
-
|
|
261
|
-
private processPendingQueries(): void {
|
|
262
|
-
if (!this.connection) return;
|
|
263
|
-
|
|
264
|
-
const pendingEntries = Array.from(this.pendingQueries.entries());
|
|
265
|
-
this.pendingQueries.clear();
|
|
266
|
-
|
|
267
|
-
for (const [keyStr, pending] of pendingEntries) {
|
|
268
|
-
const queryKey = JSON.parse(keyStr) as QueryKey;
|
|
269
|
-
const [, accessorName] = queryKey as [string, string, string];
|
|
270
|
-
|
|
271
|
-
if (pending.length > 0) {
|
|
272
|
-
const first = pending[0];
|
|
273
|
-
this.setupSubscription(
|
|
274
|
-
queryKey,
|
|
275
|
-
accessorName,
|
|
276
|
-
first.querySql,
|
|
277
|
-
first.whereExpr
|
|
278
|
-
)
|
|
279
|
-
.then(data => {
|
|
280
|
-
for (const p of pending) {
|
|
281
|
-
p.resolve(data);
|
|
282
|
-
}
|
|
283
|
-
})
|
|
284
|
-
.catch(() => {
|
|
285
|
-
for (const p of pending) {
|
|
286
|
-
p.resolve([]);
|
|
287
|
-
}
|
|
288
|
-
});
|
|
289
|
-
}
|
|
290
|
-
}
|
|
291
|
-
}
|
|
292
|
-
|
|
293
|
-
// subscribe to queries with SSR cached data but no active subscription
|
|
294
|
-
private hydrateSubscriptions(): void {
|
|
295
|
-
if (!this.connection || !this.queryClient) {
|
|
296
|
-
return;
|
|
297
|
-
}
|
|
298
|
-
|
|
299
|
-
for (const [querySql, { accessorName, whereExpr }] of queryRegistry) {
|
|
300
|
-
const queryKey = ['spacetimedb', accessorName, querySql] as const;
|
|
301
|
-
const keyStr = JSON.stringify(queryKey);
|
|
302
|
-
|
|
303
|
-
if (this.subscriptions.has(keyStr)) {
|
|
304
|
-
continue;
|
|
305
|
-
}
|
|
306
|
-
if (this.queryClient.getQueryData(queryKey) === undefined) {
|
|
307
|
-
continue;
|
|
308
|
-
}
|
|
309
|
-
|
|
310
|
-
this.setupSubscription(queryKey, accessorName, querySql, whereExpr).catch(
|
|
311
|
-
() => {}
|
|
312
|
-
);
|
|
313
|
-
}
|
|
314
|
-
}
|
|
315
|
-
|
|
316
|
-
// clean up all subscriptions and disconnect
|
|
317
|
-
disconnect(): void {
|
|
318
|
-
if (this.cacheUnsubscribe) {
|
|
319
|
-
this.cacheUnsubscribe();
|
|
320
|
-
this.cacheUnsubscribe = null;
|
|
321
|
-
}
|
|
322
|
-
|
|
323
|
-
for (const sub of this.subscriptions.values()) {
|
|
324
|
-
sub.unsubscribe();
|
|
325
|
-
}
|
|
326
|
-
this.subscriptions.clear();
|
|
327
|
-
this.pendingQueries.clear();
|
|
328
|
-
this.connection = null;
|
|
329
|
-
}
|
|
330
|
-
}
|
|
1
|
+
import type {
|
|
2
|
+
QueryClient,
|
|
3
|
+
QueryKey,
|
|
4
|
+
QueryFunction,
|
|
5
|
+
} from '@tanstack/react-query';
|
|
6
|
+
import {
|
|
7
|
+
type Query,
|
|
8
|
+
toSql,
|
|
9
|
+
type BooleanExpr,
|
|
10
|
+
evaluateBooleanExpr,
|
|
11
|
+
getQueryAccessorName,
|
|
12
|
+
getQueryWhereClause,
|
|
13
|
+
} from '../lib/query';
|
|
14
|
+
|
|
15
|
+
type QueryInput = Query<any>;
|
|
16
|
+
|
|
17
|
+
const queryRegistry = new Map<
|
|
18
|
+
string,
|
|
19
|
+
{ accessorName: string; whereExpr?: BooleanExpr<any> }
|
|
20
|
+
>();
|
|
21
|
+
|
|
22
|
+
export interface SpacetimeDBQueryOptions {
|
|
23
|
+
queryKey: readonly ['spacetimedb', string, string];
|
|
24
|
+
staleTime: number;
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
export interface SpacetimeDBQueryOptionsSkipped
|
|
28
|
+
extends SpacetimeDBQueryOptions {
|
|
29
|
+
enabled: false;
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
// creates query options for useQuery/useSuspenseQuery.
|
|
33
|
+
// useQuery(spacetimeDBQuery(tables.person));
|
|
34
|
+
// useQuery(spacetimeDBQuery(tables.user.where(r => r.online.eq(true))));
|
|
35
|
+
// useQuery(spacetimeDBQuery(condition ? tables.user : 'skip'));
|
|
36
|
+
export function spacetimeDBQuery(
|
|
37
|
+
queryOrSkip: 'skip'
|
|
38
|
+
): SpacetimeDBQueryOptionsSkipped;
|
|
39
|
+
|
|
40
|
+
export function spacetimeDBQuery(query: QueryInput): SpacetimeDBQueryOptions;
|
|
41
|
+
|
|
42
|
+
export function spacetimeDBQuery(
|
|
43
|
+
queryOrSkip: QueryInput | 'skip'
|
|
44
|
+
): SpacetimeDBQueryOptions | SpacetimeDBQueryOptionsSkipped {
|
|
45
|
+
if (queryOrSkip === 'skip') {
|
|
46
|
+
return {
|
|
47
|
+
queryKey: ['spacetimedb', '', 'skip'] as const,
|
|
48
|
+
staleTime: Infinity,
|
|
49
|
+
enabled: false,
|
|
50
|
+
};
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
const query = queryOrSkip;
|
|
54
|
+
const accessorName = getQueryAccessorName(query);
|
|
55
|
+
const whereExpr = getQueryWhereClause(query);
|
|
56
|
+
const querySql = toSql(query);
|
|
57
|
+
|
|
58
|
+
queryRegistry.set(querySql, { accessorName, whereExpr });
|
|
59
|
+
|
|
60
|
+
return {
|
|
61
|
+
queryKey: ['spacetimedb', accessorName, querySql] as const,
|
|
62
|
+
staleTime: Infinity,
|
|
63
|
+
};
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
interface SpacetimeConnection {
|
|
67
|
+
db: Record<string, any>;
|
|
68
|
+
subscriptionBuilder: () => {
|
|
69
|
+
onApplied: (cb: () => void) => any;
|
|
70
|
+
subscribe: (query: string) => { unsubscribe: () => void };
|
|
71
|
+
};
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
interface SubscriptionState {
|
|
75
|
+
unsubscribe: () => void;
|
|
76
|
+
tableInstance: any;
|
|
77
|
+
applied: boolean;
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
// push updates to cache via setQueryData when SpacetimeDB data changes
|
|
81
|
+
export class SpacetimeDBQueryClient {
|
|
82
|
+
private connection: SpacetimeConnection | null = null;
|
|
83
|
+
private queryClient: QueryClient | null = null;
|
|
84
|
+
private subscriptions = new Map<string, SubscriptionState>();
|
|
85
|
+
private pendingQueries = new Map<
|
|
86
|
+
string,
|
|
87
|
+
Array<{
|
|
88
|
+
resolve: (data: any[]) => void;
|
|
89
|
+
querySql: string;
|
|
90
|
+
whereExpr?: BooleanExpr<any>;
|
|
91
|
+
}>
|
|
92
|
+
>();
|
|
93
|
+
private cacheUnsubscribe: (() => void) | null = null;
|
|
94
|
+
|
|
95
|
+
// set connection, called on onConnect callback
|
|
96
|
+
setConnection(connection: SpacetimeConnection): void {
|
|
97
|
+
this.connection = connection;
|
|
98
|
+
this.processPendingQueries();
|
|
99
|
+
this.hydrateSubscriptions();
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
connect(queryClient: QueryClient): void {
|
|
103
|
+
this.queryClient = queryClient;
|
|
104
|
+
|
|
105
|
+
this.cacheUnsubscribe = queryClient
|
|
106
|
+
.getQueryCache()
|
|
107
|
+
.subscribe((event: any) => {
|
|
108
|
+
if (
|
|
109
|
+
event.type === 'removed' &&
|
|
110
|
+
event.query.queryKey[0] === 'spacetimedb'
|
|
111
|
+
) {
|
|
112
|
+
const keyStr = JSON.stringify(event.query.queryKey);
|
|
113
|
+
const sub = this.subscriptions.get(keyStr);
|
|
114
|
+
if (sub) {
|
|
115
|
+
sub.unsubscribe();
|
|
116
|
+
this.subscriptions.delete(keyStr);
|
|
117
|
+
}
|
|
118
|
+
}
|
|
119
|
+
});
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
queryFn: QueryFunction<any[], QueryKey> = async ({
|
|
123
|
+
queryKey,
|
|
124
|
+
}: {
|
|
125
|
+
queryKey: QueryKey;
|
|
126
|
+
}) => {
|
|
127
|
+
const keyStr = JSON.stringify(queryKey);
|
|
128
|
+
const [prefix, accessorName, querySql] = queryKey as [
|
|
129
|
+
string,
|
|
130
|
+
string,
|
|
131
|
+
string,
|
|
132
|
+
];
|
|
133
|
+
|
|
134
|
+
if (prefix !== 'spacetimedb') {
|
|
135
|
+
throw new Error(
|
|
136
|
+
`SpacetimeDBQueryClient can only handle spacetimedb queries, got: ${prefix}`
|
|
137
|
+
);
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
const registered = queryRegistry.get(querySql);
|
|
141
|
+
const whereExpr = registered?.whereExpr;
|
|
142
|
+
|
|
143
|
+
const existingSub = this.subscriptions.get(keyStr);
|
|
144
|
+
if (existingSub?.applied) {
|
|
145
|
+
return this.getTableData(existingSub.tableInstance, whereExpr);
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
// queue query if connection not ready yet
|
|
149
|
+
if (!this.connection) {
|
|
150
|
+
return new Promise<any[]>(resolve => {
|
|
151
|
+
const pending = this.pendingQueries.get(keyStr) || [];
|
|
152
|
+
pending.push({ resolve, querySql, whereExpr });
|
|
153
|
+
this.pendingQueries.set(keyStr, pending);
|
|
154
|
+
});
|
|
155
|
+
}
|
|
156
|
+
|
|
157
|
+
return this.setupSubscription(queryKey, accessorName, querySql, whereExpr);
|
|
158
|
+
};
|
|
159
|
+
|
|
160
|
+
private getTableData(
|
|
161
|
+
tableInstance: any,
|
|
162
|
+
whereExpr?: BooleanExpr<any>
|
|
163
|
+
): any[] {
|
|
164
|
+
const allRows = Array.from(tableInstance.iter());
|
|
165
|
+
if (whereExpr) {
|
|
166
|
+
return allRows.filter(row =>
|
|
167
|
+
evaluateBooleanExpr(whereExpr, row as Record<string, any>)
|
|
168
|
+
);
|
|
169
|
+
}
|
|
170
|
+
return allRows;
|
|
171
|
+
}
|
|
172
|
+
|
|
173
|
+
private setupSubscription(
|
|
174
|
+
queryKey: QueryKey,
|
|
175
|
+
accessorName: string,
|
|
176
|
+
querySql: string,
|
|
177
|
+
whereExpr?: BooleanExpr<any>
|
|
178
|
+
): Promise<any[]> {
|
|
179
|
+
if (!this.connection) {
|
|
180
|
+
return Promise.resolve([]);
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
const keyStr = JSON.stringify(queryKey);
|
|
184
|
+
const db = this.connection.db;
|
|
185
|
+
|
|
186
|
+
const tableInstance = db[accessorName];
|
|
187
|
+
|
|
188
|
+
if (!tableInstance) {
|
|
189
|
+
console.warn(`SpacetimeDBQueryClient: table "${accessorName}" not found`);
|
|
190
|
+
return Promise.resolve([]);
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
// return existing data if already subscribed
|
|
194
|
+
const existingSub = this.subscriptions.get(keyStr);
|
|
195
|
+
if (existingSub) {
|
|
196
|
+
if (existingSub.applied) {
|
|
197
|
+
return Promise.resolve(
|
|
198
|
+
this.getTableData(existingSub.tableInstance, whereExpr)
|
|
199
|
+
);
|
|
200
|
+
}
|
|
201
|
+
return new Promise(resolve => {
|
|
202
|
+
const pending = this.pendingQueries.get(keyStr) || [];
|
|
203
|
+
pending.push({ resolve, querySql, whereExpr });
|
|
204
|
+
this.pendingQueries.set(keyStr, pending);
|
|
205
|
+
});
|
|
206
|
+
}
|
|
207
|
+
|
|
208
|
+
return new Promise<any[]>(resolve => {
|
|
209
|
+
const updateCache = () => {
|
|
210
|
+
if (!this.queryClient) return [];
|
|
211
|
+
const data = this.getTableData(tableInstance, whereExpr);
|
|
212
|
+
this.queryClient.setQueryData(queryKey, data);
|
|
213
|
+
return data;
|
|
214
|
+
};
|
|
215
|
+
|
|
216
|
+
const handle = this.connection!.subscriptionBuilder()
|
|
217
|
+
.onApplied(() => {
|
|
218
|
+
const sub = this.subscriptions.get(keyStr);
|
|
219
|
+
if (sub) {
|
|
220
|
+
sub.applied = true;
|
|
221
|
+
}
|
|
222
|
+
|
|
223
|
+
const data = updateCache();
|
|
224
|
+
resolve(data);
|
|
225
|
+
|
|
226
|
+
const pending = this.pendingQueries.get(keyStr);
|
|
227
|
+
if (pending) {
|
|
228
|
+
for (const p of pending) {
|
|
229
|
+
p.resolve(data);
|
|
230
|
+
}
|
|
231
|
+
this.pendingQueries.delete(keyStr);
|
|
232
|
+
}
|
|
233
|
+
})
|
|
234
|
+
.subscribe(querySql);
|
|
235
|
+
|
|
236
|
+
// push updates to cache when data changes
|
|
237
|
+
const onTableChange = () => {
|
|
238
|
+
const sub = this.subscriptions.get(keyStr);
|
|
239
|
+
if (sub?.applied) {
|
|
240
|
+
updateCache();
|
|
241
|
+
}
|
|
242
|
+
};
|
|
243
|
+
|
|
244
|
+
tableInstance.onInsert(onTableChange);
|
|
245
|
+
tableInstance.onDelete(onTableChange);
|
|
246
|
+
tableInstance.onUpdate?.(onTableChange);
|
|
247
|
+
|
|
248
|
+
this.subscriptions.set(keyStr, {
|
|
249
|
+
unsubscribe: () => {
|
|
250
|
+
handle.unsubscribe();
|
|
251
|
+
tableInstance.removeOnInsert(onTableChange);
|
|
252
|
+
tableInstance.removeOnDelete(onTableChange);
|
|
253
|
+
tableInstance.removeOnUpdate?.(onTableChange);
|
|
254
|
+
},
|
|
255
|
+
tableInstance,
|
|
256
|
+
applied: false,
|
|
257
|
+
});
|
|
258
|
+
});
|
|
259
|
+
}
|
|
260
|
+
|
|
261
|
+
private processPendingQueries(): void {
|
|
262
|
+
if (!this.connection) return;
|
|
263
|
+
|
|
264
|
+
const pendingEntries = Array.from(this.pendingQueries.entries());
|
|
265
|
+
this.pendingQueries.clear();
|
|
266
|
+
|
|
267
|
+
for (const [keyStr, pending] of pendingEntries) {
|
|
268
|
+
const queryKey = JSON.parse(keyStr) as QueryKey;
|
|
269
|
+
const [, accessorName] = queryKey as [string, string, string];
|
|
270
|
+
|
|
271
|
+
if (pending.length > 0) {
|
|
272
|
+
const first = pending[0];
|
|
273
|
+
this.setupSubscription(
|
|
274
|
+
queryKey,
|
|
275
|
+
accessorName,
|
|
276
|
+
first.querySql,
|
|
277
|
+
first.whereExpr
|
|
278
|
+
)
|
|
279
|
+
.then(data => {
|
|
280
|
+
for (const p of pending) {
|
|
281
|
+
p.resolve(data);
|
|
282
|
+
}
|
|
283
|
+
})
|
|
284
|
+
.catch(() => {
|
|
285
|
+
for (const p of pending) {
|
|
286
|
+
p.resolve([]);
|
|
287
|
+
}
|
|
288
|
+
});
|
|
289
|
+
}
|
|
290
|
+
}
|
|
291
|
+
}
|
|
292
|
+
|
|
293
|
+
// subscribe to queries with SSR cached data but no active subscription
|
|
294
|
+
private hydrateSubscriptions(): void {
|
|
295
|
+
if (!this.connection || !this.queryClient) {
|
|
296
|
+
return;
|
|
297
|
+
}
|
|
298
|
+
|
|
299
|
+
for (const [querySql, { accessorName, whereExpr }] of queryRegistry) {
|
|
300
|
+
const queryKey = ['spacetimedb', accessorName, querySql] as const;
|
|
301
|
+
const keyStr = JSON.stringify(queryKey);
|
|
302
|
+
|
|
303
|
+
if (this.subscriptions.has(keyStr)) {
|
|
304
|
+
continue;
|
|
305
|
+
}
|
|
306
|
+
if (this.queryClient.getQueryData(queryKey) === undefined) {
|
|
307
|
+
continue;
|
|
308
|
+
}
|
|
309
|
+
|
|
310
|
+
this.setupSubscription(queryKey, accessorName, querySql, whereExpr).catch(
|
|
311
|
+
() => {}
|
|
312
|
+
);
|
|
313
|
+
}
|
|
314
|
+
}
|
|
315
|
+
|
|
316
|
+
// clean up all subscriptions and disconnect
|
|
317
|
+
disconnect(): void {
|
|
318
|
+
if (this.cacheUnsubscribe) {
|
|
319
|
+
this.cacheUnsubscribe();
|
|
320
|
+
this.cacheUnsubscribe = null;
|
|
321
|
+
}
|
|
322
|
+
|
|
323
|
+
for (const sub of this.subscriptions.values()) {
|
|
324
|
+
sub.unsubscribe();
|
|
325
|
+
}
|
|
326
|
+
this.subscriptions.clear();
|
|
327
|
+
this.pendingQueries.clear();
|
|
328
|
+
this.connection = null;
|
|
329
|
+
}
|
|
330
|
+
}
|