spacetimedb 2.5.0 → 2.6.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (180) hide show
  1. package/LICENSE.txt +759 -759
  2. package/README.md +211 -211
  3. package/dist/angular/index.cjs.map +1 -1
  4. package/dist/angular/index.mjs.map +1 -1
  5. package/dist/browser/angular/index.mjs.map +1 -1
  6. package/dist/browser/react/index.mjs +129 -57
  7. package/dist/browser/react/index.mjs.map +1 -1
  8. package/dist/browser/solid/index.mjs +120 -50
  9. package/dist/browser/solid/index.mjs.map +1 -1
  10. package/dist/browser/svelte/index.mjs.map +1 -1
  11. package/dist/browser/vue/index.mjs.map +1 -1
  12. package/dist/index.browser.mjs +10 -2
  13. package/dist/index.browser.mjs.map +1 -1
  14. package/dist/index.cjs +10 -2
  15. package/dist/index.cjs.map +1 -1
  16. package/dist/index.mjs +10 -2
  17. package/dist/index.mjs.map +1 -1
  18. package/dist/min/index.browser.mjs +1 -1
  19. package/dist/min/index.browser.mjs.map +1 -1
  20. package/dist/min/react/index.mjs +1 -1
  21. package/dist/min/react/index.mjs.map +1 -1
  22. package/dist/min/sdk/index.browser.mjs +1 -1
  23. package/dist/min/sdk/index.browser.mjs.map +1 -1
  24. package/dist/react/index.cjs +129 -57
  25. package/dist/react/index.cjs.map +1 -1
  26. package/dist/react/index.mjs +129 -57
  27. package/dist/react/index.mjs.map +1 -1
  28. package/dist/react/useTable.d.ts.map +1 -1
  29. package/dist/sdk/connection_manager.d.ts +8 -0
  30. package/dist/sdk/connection_manager.d.ts.map +1 -1
  31. package/dist/sdk/db_connection_impl.d.ts +7 -0
  32. package/dist/sdk/db_connection_impl.d.ts.map +1 -1
  33. package/dist/sdk/index.browser.mjs +10 -2
  34. package/dist/sdk/index.browser.mjs.map +1 -1
  35. package/dist/sdk/index.cjs +10 -2
  36. package/dist/sdk/index.cjs.map +1 -1
  37. package/dist/sdk/index.mjs +10 -2
  38. package/dist/sdk/index.mjs.map +1 -1
  39. package/dist/sdk/websocket_test_adapter.d.ts +2 -1
  40. package/dist/sdk/websocket_test_adapter.d.ts.map +1 -1
  41. package/dist/server/index.mjs.map +1 -1
  42. package/dist/server/runtime.d.ts.map +1 -1
  43. package/dist/solid/index.cjs +120 -50
  44. package/dist/solid/index.cjs.map +1 -1
  45. package/dist/solid/index.mjs +120 -50
  46. package/dist/solid/index.mjs.map +1 -1
  47. package/dist/svelte/index.cjs.map +1 -1
  48. package/dist/svelte/index.mjs.map +1 -1
  49. package/dist/tanstack/index.cjs +120 -50
  50. package/dist/tanstack/index.cjs.map +1 -1
  51. package/dist/tanstack/index.mjs +120 -50
  52. package/dist/tanstack/index.mjs.map +1 -1
  53. package/dist/vue/index.cjs.map +1 -1
  54. package/dist/vue/index.mjs.map +1 -1
  55. package/package.json +1 -1
  56. package/src/angular/connection_state.ts +19 -19
  57. package/src/angular/index.ts +3 -3
  58. package/src/angular/injectors/index.ts +4 -4
  59. package/src/angular/injectors/inject-reducer.ts +62 -62
  60. package/src/angular/injectors/inject-spacetimedb-connected.ts +13 -13
  61. package/src/angular/injectors/inject-spacetimedb.ts +10 -10
  62. package/src/angular/injectors/inject-table.ts +234 -234
  63. package/src/angular/providers/index.ts +1 -1
  64. package/src/angular/providers/provide-spacetimedb.ts +96 -96
  65. package/src/index.ts +16 -16
  66. package/src/lib/algebraic_type.ts +819 -819
  67. package/src/lib/algebraic_type_variants.ts +26 -26
  68. package/src/lib/algebraic_value.ts +10 -10
  69. package/src/lib/autogen/types.ts +746 -746
  70. package/src/lib/binary_reader.ts +188 -188
  71. package/src/lib/binary_writer.ts +213 -213
  72. package/src/lib/connection_id.ts +102 -102
  73. package/src/lib/constraints.ts +48 -48
  74. package/src/lib/errors.ts +26 -26
  75. package/src/lib/filter.ts +195 -195
  76. package/src/lib/identity.ts +83 -83
  77. package/src/lib/indexes.ts +251 -251
  78. package/src/lib/option.ts +34 -34
  79. package/src/lib/query.ts +1019 -1019
  80. package/src/lib/reducer_schema.ts +38 -38
  81. package/src/lib/reducers.ts +116 -116
  82. package/src/lib/result.ts +36 -36
  83. package/src/lib/schedule_at.ts +86 -86
  84. package/src/lib/schema.ts +420 -420
  85. package/src/lib/table.ts +548 -548
  86. package/src/lib/table_schema.ts +64 -64
  87. package/src/lib/time_duration.ts +77 -77
  88. package/src/lib/timestamp.ts +148 -148
  89. package/src/lib/type_builders.test-d.ts +128 -128
  90. package/src/lib/type_builders.ts +4014 -4014
  91. package/src/lib/type_util.ts +124 -124
  92. package/src/lib/util.ts +196 -196
  93. package/src/lib/uuid.ts +337 -337
  94. package/src/react/SpacetimeDBProvider.ts +84 -84
  95. package/src/react/connection_state.ts +6 -6
  96. package/src/react/index.ts +5 -5
  97. package/src/react/useProcedure.ts +60 -60
  98. package/src/react/useReducer.ts +53 -53
  99. package/src/react/useSpacetimeDB.ts +18 -18
  100. package/src/react/useTable.ts +256 -251
  101. package/src/sdk/client_api/index.ts +114 -114
  102. package/src/sdk/client_api/types/procedures.ts +8 -8
  103. package/src/sdk/client_api/types/reducers.ts +8 -8
  104. package/src/sdk/client_api/types.ts +288 -288
  105. package/src/sdk/client_cache.ts +129 -129
  106. package/src/sdk/client_table.ts +179 -179
  107. package/src/sdk/connection_manager.ts +352 -237
  108. package/src/sdk/db_connection_builder.ts +290 -290
  109. package/src/sdk/db_connection_impl.ts +1356 -1347
  110. package/src/sdk/db_context.ts +28 -28
  111. package/src/sdk/db_view.ts +12 -12
  112. package/src/sdk/decompress.ts +51 -51
  113. package/src/sdk/event.ts +18 -18
  114. package/src/sdk/event_context.ts +51 -51
  115. package/src/sdk/event_emitter.ts +32 -32
  116. package/src/sdk/index.ts +14 -14
  117. package/src/sdk/internal.ts +2 -2
  118. package/src/sdk/json_api.ts +46 -46
  119. package/src/sdk/logger.ts +134 -134
  120. package/src/sdk/message_types.ts +46 -46
  121. package/src/sdk/procedures.ts +83 -83
  122. package/src/sdk/reducer_event.ts +20 -20
  123. package/src/sdk/reducer_handle.ts +12 -12
  124. package/src/sdk/reducers.ts +159 -159
  125. package/src/sdk/schema.ts +45 -45
  126. package/src/sdk/spacetime_module.ts +28 -28
  127. package/src/sdk/subscription_builder_impl.ts +275 -275
  128. package/src/sdk/table_cache.ts +581 -581
  129. package/src/sdk/type_utils.ts +19 -19
  130. package/src/sdk/version.ts +133 -133
  131. package/src/sdk/websocket_decompress_adapter.ts +63 -63
  132. package/src/sdk/websocket_protocols.ts +25 -25
  133. package/src/sdk/websocket_test_adapter.ts +107 -100
  134. package/src/sdk/websocket_v3_frames.ts +126 -126
  135. package/src/sdk/ws.ts +105 -105
  136. package/src/server/console.ts +81 -81
  137. package/src/server/db_view.ts +21 -21
  138. package/src/server/errors.ts +138 -138
  139. package/src/server/http.test-d.ts +80 -80
  140. package/src/server/http.ts +14 -14
  141. package/src/server/http_handlers.ts +413 -413
  142. package/src/server/http_internal.ts +79 -79
  143. package/src/server/http_shared.ts +186 -186
  144. package/src/server/index.ts +37 -37
  145. package/src/server/polyfills.ts +4 -4
  146. package/src/server/procedures.ts +239 -239
  147. package/src/server/query.ts +1 -1
  148. package/src/server/range.ts +53 -53
  149. package/src/server/reducers.ts +113 -113
  150. package/src/server/rng.ts +113 -113
  151. package/src/server/runtime.ts +1102 -1102
  152. package/src/server/schema.test-d.ts +99 -99
  153. package/src/server/schema.ts +663 -663
  154. package/src/server/sys.d.ts +125 -125
  155. package/src/server/view.test-d.ts +194 -194
  156. package/src/server/views.ts +340 -340
  157. package/src/solid/SpacetimeDBProvider.ts +97 -97
  158. package/src/solid/connection_state.ts +6 -6
  159. package/src/solid/index.ts +5 -5
  160. package/src/solid/useProcedure.ts +57 -57
  161. package/src/solid/useReducer.ts +50 -50
  162. package/src/solid/useSpacetimeDB.ts +18 -18
  163. package/src/solid/useTable.ts +203 -203
  164. package/src/svelte/SpacetimeDBProvider.ts +101 -101
  165. package/src/svelte/connection_state.ts +16 -16
  166. package/src/svelte/index.ts +4 -4
  167. package/src/svelte/useReducer.ts +61 -61
  168. package/src/svelte/useSpacetimeDB.ts +22 -22
  169. package/src/svelte/useTable.ts +218 -218
  170. package/src/tanstack/SpacetimeDBQueryClient.ts +330 -330
  171. package/src/tanstack/hooks.ts +83 -83
  172. package/src/tanstack/index.ts +16 -16
  173. package/src/util-stub.ts +1 -1
  174. package/src/vue/SpacetimeDBProvider.ts +157 -157
  175. package/src/vue/connection_state.ts +19 -19
  176. package/src/vue/index.ts +5 -5
  177. package/src/vue/useProcedure.ts +62 -62
  178. package/src/vue/useReducer.ts +55 -55
  179. package/src/vue/useSpacetimeDB.ts +18 -18
  180. package/src/vue/useTable.ts +229 -229
@@ -1,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
+ }