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.
Files changed (194) hide show
  1. package/LICENSE.txt +759 -759
  2. package/README.md +211 -120
  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 +1933 -0
  9. package/dist/browser/solid/index.mjs.map +1 -0
  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/SpacetimeDBProvider.d.ts +7 -0
  44. package/dist/solid/SpacetimeDBProvider.d.ts.map +1 -0
  45. package/dist/solid/connection_state.d.ts +6 -0
  46. package/dist/solid/connection_state.d.ts.map +1 -0
  47. package/dist/solid/index.cjs +1939 -0
  48. package/dist/solid/index.cjs.map +1 -0
  49. package/dist/solid/index.d.ts +6 -0
  50. package/dist/solid/index.d.ts.map +1 -0
  51. package/dist/solid/index.mjs +1933 -0
  52. package/dist/solid/index.mjs.map +1 -0
  53. package/dist/solid/useProcedure.d.ts +4 -0
  54. package/dist/solid/useProcedure.d.ts.map +1 -0
  55. package/dist/solid/useReducer.d.ts +4 -0
  56. package/dist/solid/useReducer.d.ts.map +1 -0
  57. package/dist/solid/useSpacetimeDB.d.ts +4 -0
  58. package/dist/solid/useSpacetimeDB.d.ts.map +1 -0
  59. package/dist/solid/useTable.d.ts +32 -0
  60. package/dist/solid/useTable.d.ts.map +1 -0
  61. package/dist/svelte/index.cjs.map +1 -1
  62. package/dist/svelte/index.mjs.map +1 -1
  63. package/dist/tanstack/index.cjs +120 -50
  64. package/dist/tanstack/index.cjs.map +1 -1
  65. package/dist/tanstack/index.mjs +120 -50
  66. package/dist/tanstack/index.mjs.map +1 -1
  67. package/dist/vue/index.cjs.map +1 -1
  68. package/dist/vue/index.mjs.map +1 -1
  69. package/package.json +13 -3
  70. package/src/angular/connection_state.ts +19 -19
  71. package/src/angular/index.ts +3 -3
  72. package/src/angular/injectors/index.ts +4 -4
  73. package/src/angular/injectors/inject-reducer.ts +62 -62
  74. package/src/angular/injectors/inject-spacetimedb-connected.ts +13 -13
  75. package/src/angular/injectors/inject-spacetimedb.ts +10 -10
  76. package/src/angular/injectors/inject-table.ts +234 -234
  77. package/src/angular/providers/index.ts +1 -1
  78. package/src/angular/providers/provide-spacetimedb.ts +96 -96
  79. package/src/index.ts +16 -16
  80. package/src/lib/algebraic_type.ts +819 -819
  81. package/src/lib/algebraic_type_variants.ts +26 -26
  82. package/src/lib/algebraic_value.ts +10 -10
  83. package/src/lib/autogen/types.ts +746 -746
  84. package/src/lib/binary_reader.ts +188 -188
  85. package/src/lib/binary_writer.ts +213 -213
  86. package/src/lib/connection_id.ts +102 -102
  87. package/src/lib/constraints.ts +48 -48
  88. package/src/lib/errors.ts +26 -26
  89. package/src/lib/filter.ts +195 -195
  90. package/src/lib/identity.ts +83 -83
  91. package/src/lib/indexes.ts +251 -251
  92. package/src/lib/option.ts +34 -34
  93. package/src/lib/query.ts +1019 -1019
  94. package/src/lib/reducer_schema.ts +38 -38
  95. package/src/lib/reducers.ts +116 -116
  96. package/src/lib/result.ts +36 -36
  97. package/src/lib/schedule_at.ts +86 -86
  98. package/src/lib/schema.ts +420 -420
  99. package/src/lib/table.ts +548 -548
  100. package/src/lib/table_schema.ts +64 -64
  101. package/src/lib/time_duration.ts +77 -77
  102. package/src/lib/timestamp.ts +148 -148
  103. package/src/lib/type_builders.test-d.ts +128 -128
  104. package/src/lib/type_builders.ts +4014 -4014
  105. package/src/lib/type_util.ts +124 -124
  106. package/src/lib/util.ts +196 -196
  107. package/src/lib/uuid.ts +337 -337
  108. package/src/react/SpacetimeDBProvider.ts +84 -84
  109. package/src/react/connection_state.ts +6 -6
  110. package/src/react/index.ts +5 -5
  111. package/src/react/useProcedure.ts +60 -60
  112. package/src/react/useReducer.ts +53 -53
  113. package/src/react/useSpacetimeDB.ts +18 -18
  114. package/src/react/useTable.ts +256 -251
  115. package/src/sdk/client_api/index.ts +114 -114
  116. package/src/sdk/client_api/types/procedures.ts +8 -8
  117. package/src/sdk/client_api/types/reducers.ts +8 -8
  118. package/src/sdk/client_api/types.ts +288 -288
  119. package/src/sdk/client_cache.ts +129 -129
  120. package/src/sdk/client_table.ts +179 -179
  121. package/src/sdk/connection_manager.ts +352 -237
  122. package/src/sdk/db_connection_builder.ts +290 -290
  123. package/src/sdk/db_connection_impl.ts +1356 -1347
  124. package/src/sdk/db_context.ts +28 -28
  125. package/src/sdk/db_view.ts +12 -12
  126. package/src/sdk/decompress.ts +51 -51
  127. package/src/sdk/event.ts +18 -18
  128. package/src/sdk/event_context.ts +51 -51
  129. package/src/sdk/event_emitter.ts +32 -32
  130. package/src/sdk/index.ts +14 -14
  131. package/src/sdk/internal.ts +2 -2
  132. package/src/sdk/json_api.ts +46 -46
  133. package/src/sdk/logger.ts +134 -134
  134. package/src/sdk/message_types.ts +46 -46
  135. package/src/sdk/procedures.ts +83 -83
  136. package/src/sdk/reducer_event.ts +20 -20
  137. package/src/sdk/reducer_handle.ts +12 -12
  138. package/src/sdk/reducers.ts +159 -159
  139. package/src/sdk/schema.ts +45 -45
  140. package/src/sdk/spacetime_module.ts +28 -28
  141. package/src/sdk/subscription_builder_impl.ts +275 -275
  142. package/src/sdk/table_cache.ts +581 -581
  143. package/src/sdk/type_utils.ts +19 -19
  144. package/src/sdk/version.ts +133 -133
  145. package/src/sdk/websocket_decompress_adapter.ts +63 -63
  146. package/src/sdk/websocket_protocols.ts +25 -25
  147. package/src/sdk/websocket_test_adapter.ts +107 -100
  148. package/src/sdk/websocket_v3_frames.ts +126 -126
  149. package/src/sdk/ws.ts +105 -105
  150. package/src/server/console.ts +81 -81
  151. package/src/server/db_view.ts +21 -21
  152. package/src/server/errors.ts +138 -138
  153. package/src/server/http.test-d.ts +80 -80
  154. package/src/server/http.ts +14 -14
  155. package/src/server/http_handlers.ts +413 -413
  156. package/src/server/http_internal.ts +79 -79
  157. package/src/server/http_shared.ts +186 -186
  158. package/src/server/index.ts +37 -37
  159. package/src/server/polyfills.ts +4 -4
  160. package/src/server/procedures.ts +239 -239
  161. package/src/server/query.ts +1 -1
  162. package/src/server/range.ts +53 -53
  163. package/src/server/reducers.ts +113 -113
  164. package/src/server/rng.ts +113 -113
  165. package/src/server/runtime.ts +1102 -1102
  166. package/src/server/schema.test-d.ts +99 -99
  167. package/src/server/schema.ts +663 -663
  168. package/src/server/sys.d.ts +125 -125
  169. package/src/server/view.test-d.ts +194 -194
  170. package/src/server/views.ts +340 -340
  171. package/src/solid/SpacetimeDBProvider.ts +97 -0
  172. package/src/solid/connection_state.ts +6 -0
  173. package/src/solid/index.ts +5 -0
  174. package/src/solid/useProcedure.ts +57 -0
  175. package/src/solid/useReducer.ts +50 -0
  176. package/src/solid/useSpacetimeDB.ts +18 -0
  177. package/src/solid/useTable.ts +203 -0
  178. package/src/svelte/SpacetimeDBProvider.ts +101 -101
  179. package/src/svelte/connection_state.ts +16 -16
  180. package/src/svelte/index.ts +4 -4
  181. package/src/svelte/useReducer.ts +61 -61
  182. package/src/svelte/useSpacetimeDB.ts +22 -22
  183. package/src/svelte/useTable.ts +218 -218
  184. package/src/tanstack/SpacetimeDBQueryClient.ts +330 -330
  185. package/src/tanstack/hooks.ts +83 -83
  186. package/src/tanstack/index.ts +16 -16
  187. package/src/util-stub.ts +1 -1
  188. package/src/vue/SpacetimeDBProvider.ts +157 -157
  189. package/src/vue/connection_state.ts +19 -19
  190. package/src/vue/index.ts +5 -5
  191. package/src/vue/useProcedure.ts +62 -62
  192. package/src/vue/useReducer.ts +55 -55
  193. package/src/vue/useSpacetimeDB.ts +18 -18
  194. package/src/vue/useTable.ts +229 -229
@@ -1,413 +1,413 @@
1
- import type { Identity } from '../lib/identity';
2
- import type {
3
- HttpMethod,
4
- HttpVersion,
5
- MethodOrAny,
6
- } from '../lib/autogen/types';
7
- import type { UntypedSchemaDef } from '../lib/schema';
8
- import type { Timestamp } from '../lib/timestamp';
9
- import type { Uuid } from '../lib/uuid';
10
- import type { TransactionCtx } from './procedures';
11
- import type { HttpClient } from './http_internal';
12
- import type { Random } from './rng';
13
- import {
14
- exportContext,
15
- registerExport,
16
- type ModuleExport,
17
- type SchemaInner,
18
- } from './schema';
19
- import {
20
- Headers,
21
- makeResponse,
22
- SyncResponse,
23
- textDecoder,
24
- textEncoder,
25
- type BodyInit,
26
- type HeadersInit,
27
- type ResponseInit,
28
- } from './http_shared';
29
-
30
- export { Headers };
31
- export { SyncResponse };
32
- export type { BodyInit, HeadersInit, ResponseInit };
33
- export { makeResponse };
34
- export const httpHandlerFn = Symbol('SpacetimeDB.httpHandlerFn');
35
-
36
- export interface RequestInit {
37
- body?: BodyInit | null;
38
- headers?: HeadersInit;
39
- method?: string;
40
- version?: HttpVersion;
41
- }
42
-
43
- type RequestInner = {
44
- headers: Headers;
45
- method: string;
46
- uri: string;
47
- version: HttpVersion;
48
- };
49
-
50
- type RouteSpec = {
51
- handler: HttpHandlerExport<any>;
52
- method: MethodOrAny;
53
- path: string;
54
- };
55
-
56
- const ACCEPTABLE_ROUTE_PATH_CHARS_HUMAN_DESCRIPTION =
57
- 'ASCII lowercase letters, digits and `-_~/`';
58
-
59
- export const makeRequest = Symbol('makeRequest');
60
-
61
- function coerceRequestBody(body?: BodyInit | null): string | Uint8Array | null {
62
- if (body == null) {
63
- return null;
64
- }
65
- if (typeof body === 'string') {
66
- return body;
67
- }
68
- return new Uint8Array(body as any);
69
- }
70
-
71
- function requestBodyToBytes(body: string | Uint8Array | null): Uint8Array {
72
- if (body == null) {
73
- return new Uint8Array();
74
- }
75
- if (typeof body === 'string') {
76
- return textEncoder.encode(body);
77
- }
78
- return body;
79
- }
80
-
81
- function requestBodyToText(body: string | Uint8Array | null): string {
82
- if (body == null) {
83
- return '';
84
- }
85
- if (typeof body === 'string') {
86
- return body;
87
- }
88
- return textDecoder.decode(body);
89
- }
90
-
91
- function characterIsAcceptableForRoutePath(c: string) {
92
- return (
93
- (c >= 'a' && c <= 'z') ||
94
- (c >= '0' && c <= '9') ||
95
- c === '-' ||
96
- c === '_' ||
97
- c === '~' ||
98
- c === '/'
99
- );
100
- }
101
-
102
- function assertValidPath(path: string) {
103
- if (path !== '' && !path.startsWith('/')) {
104
- throw new TypeError(`Route paths must start with \`/\`: ${path}`);
105
- }
106
- if (![...path].every(characterIsAcceptableForRoutePath)) {
107
- throw new TypeError(
108
- `Route paths may contain only ${ACCEPTABLE_ROUTE_PATH_CHARS_HUMAN_DESCRIPTION}: ${path}`
109
- );
110
- }
111
- }
112
-
113
- function routesOverlap(a: RouteSpec, b: RouteSpec) {
114
- const methodsMatch = (left: HttpMethod, right: HttpMethod) => {
115
- if (left.tag !== right.tag) {
116
- return false;
117
- }
118
- if (left.tag === 'Extension' && right.tag === 'Extension') {
119
- return left.value === right.value;
120
- }
121
- return true;
122
- };
123
-
124
- return (
125
- a.path === b.path &&
126
- (a.method.tag === 'Any' ||
127
- b.method.tag === 'Any' ||
128
- (a.method.tag === 'Method' &&
129
- b.method.tag === 'Method' &&
130
- methodsMatch(a.method.value, b.method.value)))
131
- );
132
- }
133
-
134
- function joinPaths(prefix: string, suffix: string) {
135
- if (prefix === '/') {
136
- return suffix;
137
- }
138
- if (suffix === '/') {
139
- return prefix;
140
- }
141
- let prefixEnd = prefix.length;
142
- while (prefixEnd > 0 && prefix[prefixEnd - 1] === '/') {
143
- prefixEnd--;
144
- }
145
-
146
- let suffixStart = 0;
147
- while (suffixStart < suffix.length && suffix[suffixStart] === '/') {
148
- suffixStart++;
149
- }
150
-
151
- const joinedPrefix = prefix.slice(0, prefixEnd);
152
- const joinedSuffix = suffix.slice(suffixStart);
153
- return `${joinedPrefix}/${joinedSuffix}`;
154
- }
155
-
156
- export class Request {
157
- #body: string | Uint8Array | null;
158
- #inner: RequestInner;
159
-
160
- constructor(url: URL | string, init: RequestInit = {}) {
161
- this.#body = coerceRequestBody(init.body);
162
- this.#inner = {
163
- headers: new Headers(init.headers as any),
164
- method: init.method ?? 'GET',
165
- uri: '' + url,
166
- version: init.version ?? { tag: 'Http11' },
167
- };
168
- }
169
-
170
- static [makeRequest](body: BodyInit | null, inner: RequestInner) {
171
- const me = new Request(inner.uri);
172
- me.#body = coerceRequestBody(body);
173
- me.#inner = inner;
174
- return me;
175
- }
176
-
177
- get headers(): Headers {
178
- return this.#inner.headers;
179
- }
180
-
181
- get method(): string {
182
- return this.#inner.method;
183
- }
184
-
185
- get uri(): string {
186
- return this.#inner.uri;
187
- }
188
-
189
- get url(): string {
190
- return this.#inner.uri;
191
- }
192
-
193
- get version(): HttpVersion {
194
- return this.#inner.version;
195
- }
196
-
197
- arrayBuffer(): ArrayBuffer {
198
- return this.bytes().buffer as ArrayBuffer;
199
- }
200
-
201
- bytes(): Uint8Array {
202
- return requestBodyToBytes(this.#body);
203
- }
204
-
205
- json(): any {
206
- return JSON.parse(this.text());
207
- }
208
-
209
- text(): string {
210
- return requestBodyToText(this.#body);
211
- }
212
- }
213
-
214
- export interface HandlerContext<S extends UntypedSchemaDef = UntypedSchemaDef> {
215
- readonly timestamp: Timestamp;
216
- readonly http: HttpClient;
217
- readonly identity: Identity;
218
- readonly random: Random;
219
- withTx<T>(body: (ctx: TransactionCtx<S>) => T): T;
220
- newUuidV4(): Uuid;
221
- newUuidV7(): Uuid;
222
- }
223
-
224
- export type HandlerFn<S extends UntypedSchemaDef = UntypedSchemaDef> = (
225
- ctx: HandlerContext<S>,
226
- req: Request
227
- ) => SyncResponse;
228
-
229
- export interface HttpHandlerExport<
230
- S extends UntypedSchemaDef = UntypedSchemaDef,
231
- > extends ModuleExport {
232
- [httpHandlerFn]: HandlerFn<S>;
233
- }
234
-
235
- const exportedHttpHandlerObjects = new WeakSet<object>();
236
-
237
- export interface HttpHandlerOpts {
238
- name: string;
239
- }
240
-
241
- export class Router {
242
- #routes: RouteSpec[];
243
-
244
- constructor(routes: RouteSpec[] = []) {
245
- this.#routes = routes;
246
- }
247
-
248
- get(path: string, handler: HttpHandlerExport<any>) {
249
- return this.addRoute(
250
- { tag: 'Method', value: { tag: 'Get' } },
251
- path,
252
- handler
253
- );
254
- }
255
-
256
- head(path: string, handler: HttpHandlerExport<any>) {
257
- return this.addRoute(
258
- { tag: 'Method', value: { tag: 'Head' } },
259
- path,
260
- handler
261
- );
262
- }
263
-
264
- options(path: string, handler: HttpHandlerExport<any>) {
265
- return this.addRoute(
266
- { tag: 'Method', value: { tag: 'Options' } },
267
- path,
268
- handler
269
- );
270
- }
271
-
272
- put(path: string, handler: HttpHandlerExport<any>) {
273
- return this.addRoute(
274
- { tag: 'Method', value: { tag: 'Put' } },
275
- path,
276
- handler
277
- );
278
- }
279
-
280
- delete(path: string, handler: HttpHandlerExport<any>) {
281
- return this.addRoute(
282
- { tag: 'Method', value: { tag: 'Delete' } },
283
- path,
284
- handler
285
- );
286
- }
287
-
288
- post(path: string, handler: HttpHandlerExport<any>) {
289
- return this.addRoute(
290
- { tag: 'Method', value: { tag: 'Post' } },
291
- path,
292
- handler
293
- );
294
- }
295
-
296
- patch(path: string, handler: HttpHandlerExport<any>) {
297
- return this.addRoute(
298
- { tag: 'Method', value: { tag: 'Patch' } },
299
- path,
300
- handler
301
- );
302
- }
303
-
304
- any(path: string, handler: HttpHandlerExport<any>) {
305
- return this.addRoute({ tag: 'Any' }, path, handler);
306
- }
307
-
308
- nest(path: string, subRouter: Router) {
309
- assertValidPath(path);
310
- if (this.#routes.some(route => route.path.startsWith(path))) {
311
- throw new TypeError(
312
- `Cannot nest router at \`${path}\`; existing routes overlap with nested path`
313
- );
314
- }
315
- let merged = new Router(this.#routes);
316
- for (const route of subRouter.#routes) {
317
- merged = merged.addRoute(
318
- route.method,
319
- joinPaths(path, route.path),
320
- route.handler
321
- );
322
- }
323
- return merged;
324
- }
325
-
326
- merge(otherRouter: Router) {
327
- let merged = new Router(this.#routes);
328
- for (const route of otherRouter.#routes) {
329
- merged = merged.addRoute(route.method, route.path, route.handler);
330
- }
331
- return merged;
332
- }
333
-
334
- intoRoutes() {
335
- return this.#routes.slice();
336
- }
337
-
338
- private addRoute(
339
- method: MethodOrAny,
340
- path: string,
341
- handler: HttpHandlerExport<any>
342
- ) {
343
- assertValidPath(path);
344
- const candidate = { method, path, handler };
345
- if (this.#routes.some(route => routesOverlap(route, candidate))) {
346
- throw new TypeError(`Route conflict for \`${path}\``);
347
- }
348
- return new Router([...this.#routes, candidate]);
349
- }
350
- }
351
-
352
- export function makeHttpHandlerExport<S extends UntypedSchemaDef>(
353
- ctx: SchemaInner,
354
- opts: HttpHandlerOpts | undefined,
355
- fn: HandlerFn<S>
356
- ): HttpHandlerExport<S> {
357
- const handlerExport = {
358
- [httpHandlerFn]: fn,
359
- [exportContext]: ctx,
360
- [registerExport](ctx: SchemaInner, exportName: string) {
361
- if (exportedHttpHandlerObjects.has(handlerExport)) {
362
- throw new TypeError(
363
- `HTTP handler '${exportName}' was exported more than once`
364
- );
365
- }
366
- exportedHttpHandlerObjects.add(handlerExport);
367
- registerHttpHandler(ctx, exportName, fn, opts);
368
- ctx.httpHandlerExports.set(
369
- handlerExport as HttpHandlerExport<UntypedSchemaDef>,
370
- exportName
371
- );
372
- },
373
- };
374
- return handlerExport as HttpHandlerExport<S>;
375
- }
376
-
377
- export function makeHttpRouterExport(
378
- ctx: SchemaInner,
379
- router: Router
380
- ): ModuleExport {
381
- return {
382
- [exportContext]: ctx,
383
- [registerExport](ctx: SchemaInner) {
384
- ctx.pendingHttpRoutes.push(...router.intoRoutes());
385
- },
386
- };
387
- }
388
-
389
- function registerHttpHandler<S extends UntypedSchemaDef>(
390
- ctx: SchemaInner,
391
- exportName: string,
392
- fn: HandlerFn<S>,
393
- opts?: HttpHandlerOpts
394
- ) {
395
- ctx.defineHttpHandler(exportName);
396
- ctx.moduleDef.httpHandlers.push({ sourceName: exportName });
397
-
398
- if (opts?.name != null) {
399
- ctx.moduleDef.explicitNames.entries.push({
400
- tag: 'Function',
401
- value: {
402
- sourceName: exportName,
403
- canonicalName: opts.name,
404
- },
405
- });
406
- }
407
-
408
- if (!fn.name) {
409
- Object.defineProperty(fn, 'name', { value: exportName, writable: false });
410
- }
411
-
412
- ctx.httpHandlers.push(fn as HandlerFn<UntypedSchemaDef>);
413
- }
1
+ import type { Identity } from '../lib/identity';
2
+ import type {
3
+ HttpMethod,
4
+ HttpVersion,
5
+ MethodOrAny,
6
+ } from '../lib/autogen/types';
7
+ import type { UntypedSchemaDef } from '../lib/schema';
8
+ import type { Timestamp } from '../lib/timestamp';
9
+ import type { Uuid } from '../lib/uuid';
10
+ import type { TransactionCtx } from './procedures';
11
+ import type { HttpClient } from './http_internal';
12
+ import type { Random } from './rng';
13
+ import {
14
+ exportContext,
15
+ registerExport,
16
+ type ModuleExport,
17
+ type SchemaInner,
18
+ } from './schema';
19
+ import {
20
+ Headers,
21
+ makeResponse,
22
+ SyncResponse,
23
+ textDecoder,
24
+ textEncoder,
25
+ type BodyInit,
26
+ type HeadersInit,
27
+ type ResponseInit,
28
+ } from './http_shared';
29
+
30
+ export { Headers };
31
+ export { SyncResponse };
32
+ export type { BodyInit, HeadersInit, ResponseInit };
33
+ export { makeResponse };
34
+ export const httpHandlerFn = Symbol('SpacetimeDB.httpHandlerFn');
35
+
36
+ export interface RequestInit {
37
+ body?: BodyInit | null;
38
+ headers?: HeadersInit;
39
+ method?: string;
40
+ version?: HttpVersion;
41
+ }
42
+
43
+ type RequestInner = {
44
+ headers: Headers;
45
+ method: string;
46
+ uri: string;
47
+ version: HttpVersion;
48
+ };
49
+
50
+ type RouteSpec = {
51
+ handler: HttpHandlerExport<any>;
52
+ method: MethodOrAny;
53
+ path: string;
54
+ };
55
+
56
+ const ACCEPTABLE_ROUTE_PATH_CHARS_HUMAN_DESCRIPTION =
57
+ 'ASCII lowercase letters, digits and `-_~/`';
58
+
59
+ export const makeRequest = Symbol('makeRequest');
60
+
61
+ function coerceRequestBody(body?: BodyInit | null): string | Uint8Array | null {
62
+ if (body == null) {
63
+ return null;
64
+ }
65
+ if (typeof body === 'string') {
66
+ return body;
67
+ }
68
+ return new Uint8Array(body as any);
69
+ }
70
+
71
+ function requestBodyToBytes(body: string | Uint8Array | null): Uint8Array {
72
+ if (body == null) {
73
+ return new Uint8Array();
74
+ }
75
+ if (typeof body === 'string') {
76
+ return textEncoder.encode(body);
77
+ }
78
+ return body;
79
+ }
80
+
81
+ function requestBodyToText(body: string | Uint8Array | null): string {
82
+ if (body == null) {
83
+ return '';
84
+ }
85
+ if (typeof body === 'string') {
86
+ return body;
87
+ }
88
+ return textDecoder.decode(body);
89
+ }
90
+
91
+ function characterIsAcceptableForRoutePath(c: string) {
92
+ return (
93
+ (c >= 'a' && c <= 'z') ||
94
+ (c >= '0' && c <= '9') ||
95
+ c === '-' ||
96
+ c === '_' ||
97
+ c === '~' ||
98
+ c === '/'
99
+ );
100
+ }
101
+
102
+ function assertValidPath(path: string) {
103
+ if (path !== '' && !path.startsWith('/')) {
104
+ throw new TypeError(`Route paths must start with \`/\`: ${path}`);
105
+ }
106
+ if (![...path].every(characterIsAcceptableForRoutePath)) {
107
+ throw new TypeError(
108
+ `Route paths may contain only ${ACCEPTABLE_ROUTE_PATH_CHARS_HUMAN_DESCRIPTION}: ${path}`
109
+ );
110
+ }
111
+ }
112
+
113
+ function routesOverlap(a: RouteSpec, b: RouteSpec) {
114
+ const methodsMatch = (left: HttpMethod, right: HttpMethod) => {
115
+ if (left.tag !== right.tag) {
116
+ return false;
117
+ }
118
+ if (left.tag === 'Extension' && right.tag === 'Extension') {
119
+ return left.value === right.value;
120
+ }
121
+ return true;
122
+ };
123
+
124
+ return (
125
+ a.path === b.path &&
126
+ (a.method.tag === 'Any' ||
127
+ b.method.tag === 'Any' ||
128
+ (a.method.tag === 'Method' &&
129
+ b.method.tag === 'Method' &&
130
+ methodsMatch(a.method.value, b.method.value)))
131
+ );
132
+ }
133
+
134
+ function joinPaths(prefix: string, suffix: string) {
135
+ if (prefix === '/') {
136
+ return suffix;
137
+ }
138
+ if (suffix === '/') {
139
+ return prefix;
140
+ }
141
+ let prefixEnd = prefix.length;
142
+ while (prefixEnd > 0 && prefix[prefixEnd - 1] === '/') {
143
+ prefixEnd--;
144
+ }
145
+
146
+ let suffixStart = 0;
147
+ while (suffixStart < suffix.length && suffix[suffixStart] === '/') {
148
+ suffixStart++;
149
+ }
150
+
151
+ const joinedPrefix = prefix.slice(0, prefixEnd);
152
+ const joinedSuffix = suffix.slice(suffixStart);
153
+ return `${joinedPrefix}/${joinedSuffix}`;
154
+ }
155
+
156
+ export class Request {
157
+ #body: string | Uint8Array | null;
158
+ #inner: RequestInner;
159
+
160
+ constructor(url: URL | string, init: RequestInit = {}) {
161
+ this.#body = coerceRequestBody(init.body);
162
+ this.#inner = {
163
+ headers: new Headers(init.headers as any),
164
+ method: init.method ?? 'GET',
165
+ uri: '' + url,
166
+ version: init.version ?? { tag: 'Http11' },
167
+ };
168
+ }
169
+
170
+ static [makeRequest](body: BodyInit | null, inner: RequestInner) {
171
+ const me = new Request(inner.uri);
172
+ me.#body = coerceRequestBody(body);
173
+ me.#inner = inner;
174
+ return me;
175
+ }
176
+
177
+ get headers(): Headers {
178
+ return this.#inner.headers;
179
+ }
180
+
181
+ get method(): string {
182
+ return this.#inner.method;
183
+ }
184
+
185
+ get uri(): string {
186
+ return this.#inner.uri;
187
+ }
188
+
189
+ get url(): string {
190
+ return this.#inner.uri;
191
+ }
192
+
193
+ get version(): HttpVersion {
194
+ return this.#inner.version;
195
+ }
196
+
197
+ arrayBuffer(): ArrayBuffer {
198
+ return this.bytes().buffer as ArrayBuffer;
199
+ }
200
+
201
+ bytes(): Uint8Array {
202
+ return requestBodyToBytes(this.#body);
203
+ }
204
+
205
+ json(): any {
206
+ return JSON.parse(this.text());
207
+ }
208
+
209
+ text(): string {
210
+ return requestBodyToText(this.#body);
211
+ }
212
+ }
213
+
214
+ export interface HandlerContext<S extends UntypedSchemaDef = UntypedSchemaDef> {
215
+ readonly timestamp: Timestamp;
216
+ readonly http: HttpClient;
217
+ readonly identity: Identity;
218
+ readonly random: Random;
219
+ withTx<T>(body: (ctx: TransactionCtx<S>) => T): T;
220
+ newUuidV4(): Uuid;
221
+ newUuidV7(): Uuid;
222
+ }
223
+
224
+ export type HandlerFn<S extends UntypedSchemaDef = UntypedSchemaDef> = (
225
+ ctx: HandlerContext<S>,
226
+ req: Request
227
+ ) => SyncResponse;
228
+
229
+ export interface HttpHandlerExport<
230
+ S extends UntypedSchemaDef = UntypedSchemaDef,
231
+ > extends ModuleExport {
232
+ [httpHandlerFn]: HandlerFn<S>;
233
+ }
234
+
235
+ const exportedHttpHandlerObjects = new WeakSet<object>();
236
+
237
+ export interface HttpHandlerOpts {
238
+ name: string;
239
+ }
240
+
241
+ export class Router {
242
+ #routes: RouteSpec[];
243
+
244
+ constructor(routes: RouteSpec[] = []) {
245
+ this.#routes = routes;
246
+ }
247
+
248
+ get(path: string, handler: HttpHandlerExport<any>) {
249
+ return this.addRoute(
250
+ { tag: 'Method', value: { tag: 'Get' } },
251
+ path,
252
+ handler
253
+ );
254
+ }
255
+
256
+ head(path: string, handler: HttpHandlerExport<any>) {
257
+ return this.addRoute(
258
+ { tag: 'Method', value: { tag: 'Head' } },
259
+ path,
260
+ handler
261
+ );
262
+ }
263
+
264
+ options(path: string, handler: HttpHandlerExport<any>) {
265
+ return this.addRoute(
266
+ { tag: 'Method', value: { tag: 'Options' } },
267
+ path,
268
+ handler
269
+ );
270
+ }
271
+
272
+ put(path: string, handler: HttpHandlerExport<any>) {
273
+ return this.addRoute(
274
+ { tag: 'Method', value: { tag: 'Put' } },
275
+ path,
276
+ handler
277
+ );
278
+ }
279
+
280
+ delete(path: string, handler: HttpHandlerExport<any>) {
281
+ return this.addRoute(
282
+ { tag: 'Method', value: { tag: 'Delete' } },
283
+ path,
284
+ handler
285
+ );
286
+ }
287
+
288
+ post(path: string, handler: HttpHandlerExport<any>) {
289
+ return this.addRoute(
290
+ { tag: 'Method', value: { tag: 'Post' } },
291
+ path,
292
+ handler
293
+ );
294
+ }
295
+
296
+ patch(path: string, handler: HttpHandlerExport<any>) {
297
+ return this.addRoute(
298
+ { tag: 'Method', value: { tag: 'Patch' } },
299
+ path,
300
+ handler
301
+ );
302
+ }
303
+
304
+ any(path: string, handler: HttpHandlerExport<any>) {
305
+ return this.addRoute({ tag: 'Any' }, path, handler);
306
+ }
307
+
308
+ nest(path: string, subRouter: Router) {
309
+ assertValidPath(path);
310
+ if (this.#routes.some(route => route.path.startsWith(path))) {
311
+ throw new TypeError(
312
+ `Cannot nest router at \`${path}\`; existing routes overlap with nested path`
313
+ );
314
+ }
315
+ let merged = new Router(this.#routes);
316
+ for (const route of subRouter.#routes) {
317
+ merged = merged.addRoute(
318
+ route.method,
319
+ joinPaths(path, route.path),
320
+ route.handler
321
+ );
322
+ }
323
+ return merged;
324
+ }
325
+
326
+ merge(otherRouter: Router) {
327
+ let merged = new Router(this.#routes);
328
+ for (const route of otherRouter.#routes) {
329
+ merged = merged.addRoute(route.method, route.path, route.handler);
330
+ }
331
+ return merged;
332
+ }
333
+
334
+ intoRoutes() {
335
+ return this.#routes.slice();
336
+ }
337
+
338
+ private addRoute(
339
+ method: MethodOrAny,
340
+ path: string,
341
+ handler: HttpHandlerExport<any>
342
+ ) {
343
+ assertValidPath(path);
344
+ const candidate = { method, path, handler };
345
+ if (this.#routes.some(route => routesOverlap(route, candidate))) {
346
+ throw new TypeError(`Route conflict for \`${path}\``);
347
+ }
348
+ return new Router([...this.#routes, candidate]);
349
+ }
350
+ }
351
+
352
+ export function makeHttpHandlerExport<S extends UntypedSchemaDef>(
353
+ ctx: SchemaInner,
354
+ opts: HttpHandlerOpts | undefined,
355
+ fn: HandlerFn<S>
356
+ ): HttpHandlerExport<S> {
357
+ const handlerExport = {
358
+ [httpHandlerFn]: fn,
359
+ [exportContext]: ctx,
360
+ [registerExport](ctx: SchemaInner, exportName: string) {
361
+ if (exportedHttpHandlerObjects.has(handlerExport)) {
362
+ throw new TypeError(
363
+ `HTTP handler '${exportName}' was exported more than once`
364
+ );
365
+ }
366
+ exportedHttpHandlerObjects.add(handlerExport);
367
+ registerHttpHandler(ctx, exportName, fn, opts);
368
+ ctx.httpHandlerExports.set(
369
+ handlerExport as HttpHandlerExport<UntypedSchemaDef>,
370
+ exportName
371
+ );
372
+ },
373
+ };
374
+ return handlerExport as HttpHandlerExport<S>;
375
+ }
376
+
377
+ export function makeHttpRouterExport(
378
+ ctx: SchemaInner,
379
+ router: Router
380
+ ): ModuleExport {
381
+ return {
382
+ [exportContext]: ctx,
383
+ [registerExport](ctx: SchemaInner) {
384
+ ctx.pendingHttpRoutes.push(...router.intoRoutes());
385
+ },
386
+ };
387
+ }
388
+
389
+ function registerHttpHandler<S extends UntypedSchemaDef>(
390
+ ctx: SchemaInner,
391
+ exportName: string,
392
+ fn: HandlerFn<S>,
393
+ opts?: HttpHandlerOpts
394
+ ) {
395
+ ctx.defineHttpHandler(exportName);
396
+ ctx.moduleDef.httpHandlers.push({ sourceName: exportName });
397
+
398
+ if (opts?.name != null) {
399
+ ctx.moduleDef.explicitNames.entries.push({
400
+ tag: 'Function',
401
+ value: {
402
+ sourceName: exportName,
403
+ canonicalName: opts.name,
404
+ },
405
+ });
406
+ }
407
+
408
+ if (!fn.name) {
409
+ Object.defineProperty(fn, 'name', { value: exportName, writable: false });
410
+ }
411
+
412
+ ctx.httpHandlers.push(fn as HandlerFn<UntypedSchemaDef>);
413
+ }