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,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
|
+
}
|