@canmi/seam-server 0.2.14 → 0.4.3
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/dist/index.d.ts +62 -3
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +324 -60
- package/dist/index.js.map +1 -1
- package/package.json +2 -2
package/dist/index.d.ts
CHANGED
|
@@ -106,13 +106,30 @@ type LoaderFn = (params: Record<string, string>) => LoaderResult;
|
|
|
106
106
|
interface LayoutDef {
|
|
107
107
|
id: string;
|
|
108
108
|
template: string;
|
|
109
|
+
localeTemplates?: Record<string, string>;
|
|
109
110
|
loaders: Record<string, LoaderFn>;
|
|
111
|
+
i18nKeys?: string[];
|
|
110
112
|
}
|
|
111
113
|
interface PageDef {
|
|
112
114
|
template: string;
|
|
115
|
+
localeTemplates?: Record<string, string>;
|
|
113
116
|
loaders: Record<string, LoaderFn>;
|
|
114
117
|
layoutChain?: LayoutDef[];
|
|
115
118
|
headMeta?: string;
|
|
119
|
+
dataId?: string;
|
|
120
|
+
i18nKeys?: string[];
|
|
121
|
+
}
|
|
122
|
+
interface I18nConfig {
|
|
123
|
+
locales: string[];
|
|
124
|
+
default: string;
|
|
125
|
+
mode: "memory" | "paged";
|
|
126
|
+
cache: boolean;
|
|
127
|
+
routeHashes: Record<string, string>;
|
|
128
|
+
contentHashes: Record<string, Record<string, string>>;
|
|
129
|
+
/** Memory mode: locale → routeHash → messages */
|
|
130
|
+
messages: Record<string, Record<string, Record<string, string>>>;
|
|
131
|
+
/** Paged mode: base directory for on-demand reads */
|
|
132
|
+
distDir?: string;
|
|
116
133
|
}
|
|
117
134
|
declare function definePage(config: PageDef): PageDef;
|
|
118
135
|
//#endregion
|
|
@@ -128,6 +145,38 @@ interface HandlePageResult {
|
|
|
128
145
|
html: string;
|
|
129
146
|
timing?: PageTiming;
|
|
130
147
|
}
|
|
148
|
+
interface I18nOpts {
|
|
149
|
+
locale: string;
|
|
150
|
+
config: I18nConfig;
|
|
151
|
+
/** Route pattern for hash-based message lookup */
|
|
152
|
+
routePattern: string;
|
|
153
|
+
}
|
|
154
|
+
//#endregion
|
|
155
|
+
//#region src/resolve.d.ts
|
|
156
|
+
interface ResolveStrategy {
|
|
157
|
+
readonly kind: string;
|
|
158
|
+
resolve(data: ResolveData): string | null;
|
|
159
|
+
}
|
|
160
|
+
interface ResolveData {
|
|
161
|
+
readonly url: string;
|
|
162
|
+
readonly pathLocale: string | null;
|
|
163
|
+
readonly cookie: string | undefined;
|
|
164
|
+
readonly acceptLanguage: string | undefined;
|
|
165
|
+
readonly locales: string[];
|
|
166
|
+
readonly defaultLocale: string;
|
|
167
|
+
}
|
|
168
|
+
/** URL prefix strategy: trusts pathLocale if it is a known locale */
|
|
169
|
+
declare function fromUrlPrefix(): ResolveStrategy;
|
|
170
|
+
/** Cookie strategy: reads a named cookie and validates against known locales */
|
|
171
|
+
declare function fromCookie(name?: string): ResolveStrategy;
|
|
172
|
+
/** Accept-Language strategy: parses header with q-value sorting + prefix match */
|
|
173
|
+
declare function fromAcceptLanguage(): ResolveStrategy;
|
|
174
|
+
/** URL query strategy: reads a query parameter and validates against known locales */
|
|
175
|
+
declare function fromUrlQuery(param?: string): ResolveStrategy;
|
|
176
|
+
/** Run strategies in order; first non-null wins, otherwise defaultLocale */
|
|
177
|
+
declare function resolveChain(strategies: ResolveStrategy[], data: ResolveData): string;
|
|
178
|
+
/** Default strategy chain: url_prefix -> cookie -> accept_language */
|
|
179
|
+
declare function defaultStrategies(): ResolveStrategy[];
|
|
131
180
|
//#endregion
|
|
132
181
|
//#region src/router/index.d.ts
|
|
133
182
|
interface ProcedureDef<TIn = unknown, TOut = unknown> {
|
|
@@ -148,7 +197,14 @@ interface SubscriptionDef<TIn = unknown, TOut = unknown> {
|
|
|
148
197
|
type DefinitionMap = Record<string, ProcedureDef<any, any> | SubscriptionDef<any, any>>;
|
|
149
198
|
interface RouterOptions {
|
|
150
199
|
pages?: Record<string, PageDef>;
|
|
200
|
+
i18n?: I18nConfig | null;
|
|
151
201
|
validateOutput?: boolean;
|
|
202
|
+
resolve?: ResolveStrategy[];
|
|
203
|
+
}
|
|
204
|
+
interface PageRequestHeaders {
|
|
205
|
+
url?: string;
|
|
206
|
+
cookie?: string;
|
|
207
|
+
acceptLanguage?: string;
|
|
152
208
|
}
|
|
153
209
|
interface Router<T extends DefinitionMap> {
|
|
154
210
|
manifest(): ProcedureManifest;
|
|
@@ -157,7 +213,7 @@ interface Router<T extends DefinitionMap> {
|
|
|
157
213
|
results: BatchResultItem[];
|
|
158
214
|
}>;
|
|
159
215
|
handleSubscription(name: string, input: unknown): AsyncIterable<unknown>;
|
|
160
|
-
handlePage(path: string): Promise<HandlePageResult | null>;
|
|
216
|
+
handlePage(path: string, headers?: PageRequestHeaders): Promise<HandlePageResult | null>;
|
|
161
217
|
readonly hasPages: boolean;
|
|
162
218
|
/** Exposed for adapter access to the definitions */
|
|
163
219
|
readonly procedures: T;
|
|
@@ -183,6 +239,7 @@ interface HttpRequest {
|
|
|
183
239
|
method: string;
|
|
184
240
|
url: string;
|
|
185
241
|
body: () => Promise<unknown>;
|
|
242
|
+
header?: (name: string) => string | null;
|
|
186
243
|
}
|
|
187
244
|
interface HttpBodyResponse {
|
|
188
245
|
status: number;
|
|
@@ -221,8 +278,10 @@ declare function toWebResponse(result: HttpResponse): Response;
|
|
|
221
278
|
//#region src/page/build-loader.d.ts
|
|
222
279
|
/** Load the RPC hash map from build output (returns undefined when obfuscation is off) */
|
|
223
280
|
declare function loadRpcHashMap(distDir: string): RpcHashMap | undefined;
|
|
281
|
+
/** Load i18n config and messages from build output */
|
|
282
|
+
declare function loadI18nMessages(distDir: string): I18nConfig | null;
|
|
224
283
|
declare function loadBuildOutput(distDir: string): Record<string, PageDef>;
|
|
225
|
-
/** Load build output with lazy template getters
|
|
284
|
+
/** Load build output with lazy template getters -- templates re-read from disk on each access */
|
|
226
285
|
declare function loadBuildOutputDev(distDir: string): Record<string, PageDef>;
|
|
227
286
|
//#endregion
|
|
228
287
|
//#region src/subscription.d.ts
|
|
@@ -264,5 +323,5 @@ interface ReloadWatcher {
|
|
|
264
323
|
}
|
|
265
324
|
declare function watchReloadTrigger(distDir: string, onReload: () => void): ReloadWatcher;
|
|
266
325
|
//#endregion
|
|
267
|
-
export { type BatchCall, type BatchResultItem, type CallbackSink, type DefinitionMap, type DevProxyOptions, type ErrorCode, type HandlePageResult, type HandleResult, type HttpBodyResponse, type HttpHandler, type HttpHandlerOptions, type HttpRequest, type HttpResponse, type HttpStreamResponse, type Infer, type LayoutDef, type LoaderFn, type OptionalSchemaNode, type PageDef, type PageTiming, type ProcedureDef, type ProcedureEntry, type ProcedureManifest, type ProcedureType, type ReloadWatcher, type Router, type RouterOptions, type RpcHashMap, type SchemaNode, SeamError, type StaticHandlerOptions, type SubscriptionDef, createDevProxy, createHttpHandler, createRouter, createStaticHandler, definePage, drainStream, fromCallback, loadBuildOutput, loadBuildOutputDev, loadRpcHashMap, serialize, sseCompleteEvent, sseDataEvent, sseErrorEvent, t, toWebResponse, watchReloadTrigger };
|
|
326
|
+
export { type BatchCall, type BatchResultItem, type CallbackSink, type DefinitionMap, type DevProxyOptions, type ErrorCode, type HandlePageResult, type HandleResult, type HttpBodyResponse, type HttpHandler, type HttpHandlerOptions, type HttpRequest, type HttpResponse, type HttpStreamResponse, type I18nConfig, type I18nOpts, type Infer, type LayoutDef, type LoaderFn, type OptionalSchemaNode, type PageDef, type PageRequestHeaders, type PageTiming, type ProcedureDef, type ProcedureEntry, type ProcedureManifest, type ProcedureType, type ReloadWatcher, type ResolveData, type ResolveStrategy, type Router, type RouterOptions, type RpcHashMap, type SchemaNode, SeamError, type StaticHandlerOptions, type SubscriptionDef, createDevProxy, createHttpHandler, createRouter, createStaticHandler, defaultStrategies, definePage, drainStream, fromAcceptLanguage, fromCallback, fromCookie, fromUrlPrefix, fromUrlQuery, loadBuildOutput, loadBuildOutputDev, loadI18nMessages, loadRpcHashMap, resolveChain, serialize, sseCompleteEvent, sseDataEvent, sseErrorEvent, t, toWebResponse, watchReloadTrigger };
|
|
268
327
|
//# sourceMappingURL=index.d.ts.map
|
package/dist/index.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.d.ts","names":[],"sources":["../src/types/schema.ts","../src/types/primitives.ts","../src/types/composites.ts","../src/types/index.ts","../src/manifest/index.ts","../src/procedure.ts","../src/router/handler.ts","../src/page/index.ts","../src/page/handler.ts","../src/router/index.ts","../src/errors.ts","../src/http.ts","../src/page/build-loader.ts","../src/subscription.ts","../src/proxy.ts","../src/dev/reload-watcher.ts"],"mappings":"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
|
|
1
|
+
{"version":3,"file":"index.d.ts","names":[],"sources":["../src/types/schema.ts","../src/types/primitives.ts","../src/types/composites.ts","../src/types/index.ts","../src/manifest/index.ts","../src/procedure.ts","../src/router/handler.ts","../src/page/index.ts","../src/page/handler.ts","../src/resolve.ts","../src/router/index.ts","../src/errors.ts","../src/http.ts","../src/page/build-loader.ts","../src/subscription.ts","../src/proxy.ts","../src/dev/reload-watcher.ts"],"mappings":"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"}
|
package/dist/index.js
CHANGED
|
@@ -1,8 +1,8 @@
|
|
|
1
|
+
import { existsSync, readFileSync, watch } from "node:fs";
|
|
2
|
+
import { extname, join } from "node:path";
|
|
1
3
|
import { validate } from "jtd";
|
|
2
|
-
import { escapeHtml,
|
|
4
|
+
import { escapeHtml, renderPage } from "@canmi/seam-engine";
|
|
3
5
|
import { readFile } from "node:fs/promises";
|
|
4
|
-
import { extname, join } from "node:path";
|
|
5
|
-
import { readFileSync, watch } from "node:fs";
|
|
6
6
|
|
|
7
7
|
//#region \0rolldown/runtime.js
|
|
8
8
|
var __defProp = Object.defineProperty;
|
|
@@ -266,12 +266,6 @@ async function* handleSubscription(subscriptions, name, rawInput, validateOutput
|
|
|
266
266
|
|
|
267
267
|
//#endregion
|
|
268
268
|
//#region src/page/handler.ts
|
|
269
|
-
/** Flatten keyed loader results: spread object values into a flat map for slot resolution */
|
|
270
|
-
function flattenForSlots(keyed) {
|
|
271
|
-
const merged = { ...keyed };
|
|
272
|
-
for (const value of Object.values(keyed)) if (value && typeof value === "object" && !Array.isArray(value)) Object.assign(merged, value);
|
|
273
|
-
return merged;
|
|
274
|
-
}
|
|
275
269
|
/** Execute loaders, returning keyed results */
|
|
276
270
|
async function executeLoaders(loaders, params, procedures) {
|
|
277
271
|
const entries = Object.entries(loaders);
|
|
@@ -283,47 +277,61 @@ async function executeLoaders(loaders, params, procedures) {
|
|
|
283
277
|
}));
|
|
284
278
|
return Object.fromEntries(results);
|
|
285
279
|
}
|
|
286
|
-
/**
|
|
287
|
-
function
|
|
288
|
-
|
|
289
|
-
|
|
290
|
-
|
|
291
|
-
|
|
292
|
-
|
|
293
|
-
const
|
|
294
|
-
|
|
295
|
-
|
|
296
|
-
|
|
280
|
+
/** Select the template for a given locale, falling back to the default template */
|
|
281
|
+
function selectTemplate(defaultTemplate, localeTemplates, locale) {
|
|
282
|
+
if (locale && localeTemplates) return localeTemplates[locale] ?? defaultTemplate;
|
|
283
|
+
return defaultTemplate;
|
|
284
|
+
}
|
|
285
|
+
/** Look up pre-resolved messages for a route + locale. Zero merge, zero filter. */
|
|
286
|
+
function lookupMessages(config, routePattern, locale) {
|
|
287
|
+
const routeHash = config.routeHashes[routePattern];
|
|
288
|
+
if (!routeHash) return {};
|
|
289
|
+
if (config.mode === "paged" && config.distDir) {
|
|
290
|
+
const filePath = join(config.distDir, "i18n", routeHash, `${locale}.json`);
|
|
291
|
+
if (existsSync(filePath)) return JSON.parse(readFileSync(filePath, "utf-8"));
|
|
292
|
+
return {};
|
|
293
|
+
}
|
|
294
|
+
return config.messages[locale]?.[routeHash] ?? {};
|
|
295
|
+
}
|
|
296
|
+
async function handlePageRequest(page, params, procedures, i18nOpts) {
|
|
297
297
|
try {
|
|
298
298
|
const t0 = performance.now();
|
|
299
299
|
const layoutChain = page.layoutChain ?? [];
|
|
300
|
+
const locale = i18nOpts?.locale;
|
|
300
301
|
const loaderResults = await Promise.all([...layoutChain.map((layout) => executeLoaders(layout.loaders, params, procedures)), executeLoaders(page.loaders, params, procedures)]);
|
|
301
302
|
const t1 = performance.now();
|
|
302
|
-
const
|
|
303
|
-
const
|
|
304
|
-
let
|
|
305
|
-
const layoutKeyed = {};
|
|
303
|
+
const allData = {};
|
|
304
|
+
for (const result of loaderResults) Object.assign(allData, result);
|
|
305
|
+
let composedTemplate = selectTemplate(page.template, page.localeTemplates, locale);
|
|
306
306
|
for (let i = layoutChain.length - 1; i >= 0; i--) {
|
|
307
307
|
const layout = layoutChain[i];
|
|
308
|
-
|
|
309
|
-
layoutKeyed[layout.id] = data;
|
|
310
|
-
innerContent = injectLayout(layout.template, flattenForSlots(data), innerContent);
|
|
308
|
+
composedTemplate = selectTemplate(layout.template, layout.localeTemplates, locale).replace("<!--seam:outlet-->", composedTemplate);
|
|
311
309
|
}
|
|
312
|
-
|
|
313
|
-
|
|
314
|
-
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
|
|
310
|
+
const config = {
|
|
311
|
+
layout_chain: layoutChain.map((l) => ({
|
|
312
|
+
id: l.id,
|
|
313
|
+
loader_keys: Object.keys(l.loaders)
|
|
314
|
+
})),
|
|
315
|
+
data_id: page.dataId ?? "__SEAM_DATA__",
|
|
316
|
+
head_meta: page.headMeta
|
|
317
|
+
};
|
|
318
|
+
let i18nOptsJson;
|
|
319
|
+
if (i18nOpts) {
|
|
320
|
+
const { config: i18nConfig, routePattern } = i18nOpts;
|
|
321
|
+
const messages = lookupMessages(i18nConfig, routePattern, i18nOpts.locale);
|
|
322
|
+
const routeHash = i18nConfig.routeHashes[routePattern];
|
|
323
|
+
const i18nData = {
|
|
324
|
+
locale: i18nOpts.locale,
|
|
325
|
+
default_locale: i18nConfig.default,
|
|
326
|
+
messages
|
|
327
|
+
};
|
|
328
|
+
if (i18nConfig.cache && routeHash) {
|
|
329
|
+
i18nData.hash = i18nConfig.contentHashes[routeHash]?.[i18nOpts.locale];
|
|
330
|
+
i18nData.router = i18nConfig.contentHashes;
|
|
318
331
|
}
|
|
332
|
+
i18nOptsJson = JSON.stringify(i18nData);
|
|
319
333
|
}
|
|
320
|
-
const
|
|
321
|
-
if (Object.keys(layoutKeyed).length > 0) seamData._layouts = layoutKeyed;
|
|
322
|
-
const script = `<script id="__SEAM_DATA__" type="application/json">${JSON.stringify(seamData)}<\/script>`;
|
|
323
|
-
const bodyClose = innerContent.lastIndexOf("</body>");
|
|
324
|
-
let html;
|
|
325
|
-
if (bodyClose !== -1) html = innerContent.slice(0, bodyClose) + script + innerContent.slice(bodyClose);
|
|
326
|
-
else html = innerContent + script;
|
|
334
|
+
const html = renderPage(composedTemplate, JSON.stringify(allData), JSON.stringify(config), i18nOptsJson);
|
|
327
335
|
const t2 = performance.now();
|
|
328
336
|
return {
|
|
329
337
|
status: 200,
|
|
@@ -367,6 +375,7 @@ var RouteMatcher = class {
|
|
|
367
375
|
routes = [];
|
|
368
376
|
add(pattern, value) {
|
|
369
377
|
this.routes.push({
|
|
378
|
+
pattern,
|
|
370
379
|
compiled: compileRoute(pattern),
|
|
371
380
|
value
|
|
372
381
|
});
|
|
@@ -377,18 +386,138 @@ var RouteMatcher = class {
|
|
|
377
386
|
const params = matchRoute(route.compiled.segments, parts);
|
|
378
387
|
if (params) return {
|
|
379
388
|
value: route.value,
|
|
380
|
-
params
|
|
389
|
+
params,
|
|
390
|
+
pattern: route.pattern
|
|
381
391
|
};
|
|
382
392
|
}
|
|
383
393
|
return null;
|
|
384
394
|
}
|
|
385
395
|
};
|
|
386
396
|
|
|
397
|
+
//#endregion
|
|
398
|
+
//#region src/resolve.ts
|
|
399
|
+
/** URL prefix strategy: trusts pathLocale if it is a known locale */
|
|
400
|
+
function fromUrlPrefix() {
|
|
401
|
+
return {
|
|
402
|
+
kind: "url_prefix",
|
|
403
|
+
resolve(data) {
|
|
404
|
+
if (data.pathLocale && data.locales.includes(data.pathLocale)) return data.pathLocale;
|
|
405
|
+
return null;
|
|
406
|
+
}
|
|
407
|
+
};
|
|
408
|
+
}
|
|
409
|
+
/** Cookie strategy: reads a named cookie and validates against known locales */
|
|
410
|
+
function fromCookie(name = "seam-locale") {
|
|
411
|
+
return {
|
|
412
|
+
kind: "cookie",
|
|
413
|
+
resolve(data) {
|
|
414
|
+
if (!data.cookie) return null;
|
|
415
|
+
for (const pair of data.cookie.split(";")) {
|
|
416
|
+
const [k, v] = pair.trim().split("=");
|
|
417
|
+
if (k === name && v && data.locales.includes(v)) return v;
|
|
418
|
+
}
|
|
419
|
+
return null;
|
|
420
|
+
}
|
|
421
|
+
};
|
|
422
|
+
}
|
|
423
|
+
/** Accept-Language strategy: parses header with q-value sorting + prefix match */
|
|
424
|
+
function fromAcceptLanguage() {
|
|
425
|
+
return {
|
|
426
|
+
kind: "accept_language",
|
|
427
|
+
resolve(data) {
|
|
428
|
+
if (!data.acceptLanguage) return null;
|
|
429
|
+
const entries = [];
|
|
430
|
+
for (const part of data.acceptLanguage.split(",")) {
|
|
431
|
+
const [lang, ...rest] = part.trim().split(";");
|
|
432
|
+
let q = 1;
|
|
433
|
+
for (const r of rest) {
|
|
434
|
+
const match = r.trim().match(/^q=(\d+(?:\.\d+)?)$/);
|
|
435
|
+
if (match) q = parseFloat(match[1]);
|
|
436
|
+
}
|
|
437
|
+
entries.push({
|
|
438
|
+
lang: lang.trim(),
|
|
439
|
+
q
|
|
440
|
+
});
|
|
441
|
+
}
|
|
442
|
+
entries.sort((a, b) => b.q - a.q);
|
|
443
|
+
const localeSet = new Set(data.locales);
|
|
444
|
+
for (const { lang } of entries) {
|
|
445
|
+
if (localeSet.has(lang)) return lang;
|
|
446
|
+
const prefix = lang.split("-")[0];
|
|
447
|
+
if (prefix !== lang && localeSet.has(prefix)) return prefix;
|
|
448
|
+
}
|
|
449
|
+
return null;
|
|
450
|
+
}
|
|
451
|
+
};
|
|
452
|
+
}
|
|
453
|
+
/** URL query strategy: reads a query parameter and validates against known locales */
|
|
454
|
+
function fromUrlQuery(param = "lang") {
|
|
455
|
+
return {
|
|
456
|
+
kind: "url_query",
|
|
457
|
+
resolve(data) {
|
|
458
|
+
if (!data.url) return null;
|
|
459
|
+
try {
|
|
460
|
+
const value = new URL(data.url, "http://localhost").searchParams.get(param);
|
|
461
|
+
if (value && data.locales.includes(value)) return value;
|
|
462
|
+
} catch {}
|
|
463
|
+
return null;
|
|
464
|
+
}
|
|
465
|
+
};
|
|
466
|
+
}
|
|
467
|
+
/** Run strategies in order; first non-null wins, otherwise defaultLocale */
|
|
468
|
+
function resolveChain(strategies, data) {
|
|
469
|
+
for (const s of strategies) {
|
|
470
|
+
const result = s.resolve(data);
|
|
471
|
+
if (result !== null) return result;
|
|
472
|
+
}
|
|
473
|
+
return data.defaultLocale;
|
|
474
|
+
}
|
|
475
|
+
/** Default strategy chain: url_prefix -> cookie -> accept_language */
|
|
476
|
+
function defaultStrategies() {
|
|
477
|
+
return [
|
|
478
|
+
fromUrlPrefix(),
|
|
479
|
+
fromCookie(),
|
|
480
|
+
fromAcceptLanguage()
|
|
481
|
+
];
|
|
482
|
+
}
|
|
483
|
+
|
|
387
484
|
//#endregion
|
|
388
485
|
//#region src/router/index.ts
|
|
389
486
|
function isSubscriptionDef(def) {
|
|
390
487
|
return "type" in def && def.type === "subscription";
|
|
391
488
|
}
|
|
489
|
+
/** Build the resolve strategy list from options */
|
|
490
|
+
function buildStrategies(opts) {
|
|
491
|
+
const strategies = opts?.resolve ?? defaultStrategies();
|
|
492
|
+
return {
|
|
493
|
+
strategies,
|
|
494
|
+
hasUrlPrefix: strategies.some((s) => s.kind === "url_prefix")
|
|
495
|
+
};
|
|
496
|
+
}
|
|
497
|
+
/** Register built-in __seam_i18n_query procedure (route-hash-based lookup) */
|
|
498
|
+
function registerI18nQuery(procedureMap, config) {
|
|
499
|
+
procedureMap.set("__seam_i18n_query", {
|
|
500
|
+
inputSchema: {},
|
|
501
|
+
outputSchema: {},
|
|
502
|
+
handler: ({ input }) => {
|
|
503
|
+
const { route, locale } = input;
|
|
504
|
+
const messages = lookupI18nMessages(config, route, locale);
|
|
505
|
+
return {
|
|
506
|
+
hash: config.contentHashes[route]?.[locale] ?? "",
|
|
507
|
+
messages
|
|
508
|
+
};
|
|
509
|
+
}
|
|
510
|
+
});
|
|
511
|
+
}
|
|
512
|
+
/** Look up messages by route hash + locale for RPC query */
|
|
513
|
+
function lookupI18nMessages(config, routeHash, locale) {
|
|
514
|
+
if (config.mode === "paged" && config.distDir) {
|
|
515
|
+
const filePath = join(config.distDir, "i18n", routeHash, `${locale}.json`);
|
|
516
|
+
if (existsSync(filePath)) return JSON.parse(readFileSync(filePath, "utf-8"));
|
|
517
|
+
return {};
|
|
518
|
+
}
|
|
519
|
+
return config.messages[locale]?.[routeHash] ?? {};
|
|
520
|
+
}
|
|
392
521
|
function createRouter(procedures, opts) {
|
|
393
522
|
const procedureMap = /* @__PURE__ */ new Map();
|
|
394
523
|
const subscriptionMap = /* @__PURE__ */ new Map();
|
|
@@ -406,6 +535,9 @@ function createRouter(procedures, opts) {
|
|
|
406
535
|
const pageMatcher = new RouteMatcher();
|
|
407
536
|
const pages = opts?.pages;
|
|
408
537
|
if (pages) for (const [pattern, page] of Object.entries(pages)) pageMatcher.add(pattern, page);
|
|
538
|
+
const i18nConfig = opts?.i18n ?? null;
|
|
539
|
+
const { strategies, hasUrlPrefix } = buildStrategies(opts);
|
|
540
|
+
if (i18nConfig) registerI18nQuery(procedureMap, i18nConfig);
|
|
409
541
|
return {
|
|
410
542
|
procedures,
|
|
411
543
|
hasPages: !!pages && Object.keys(pages).length > 0,
|
|
@@ -421,10 +553,34 @@ function createRouter(procedures, opts) {
|
|
|
421
553
|
handleSubscription(name, input) {
|
|
422
554
|
return handleSubscription(subscriptionMap, name, input, shouldValidateOutput);
|
|
423
555
|
},
|
|
424
|
-
async handlePage(path) {
|
|
425
|
-
|
|
556
|
+
async handlePage(path, headers) {
|
|
557
|
+
let pathLocale = null;
|
|
558
|
+
let routePath = path;
|
|
559
|
+
if (hasUrlPrefix && i18nConfig) {
|
|
560
|
+
const segments = path.split("/").filter(Boolean);
|
|
561
|
+
const localeSet = new Set(i18nConfig.locales);
|
|
562
|
+
if (segments.length > 0 && localeSet.has(segments[0])) {
|
|
563
|
+
pathLocale = segments[0];
|
|
564
|
+
routePath = "/" + segments.slice(1).join("/") || "/";
|
|
565
|
+
}
|
|
566
|
+
}
|
|
567
|
+
let locale;
|
|
568
|
+
if (i18nConfig) locale = resolveChain(strategies, {
|
|
569
|
+
url: headers?.url ?? "",
|
|
570
|
+
pathLocale,
|
|
571
|
+
cookie: headers?.cookie,
|
|
572
|
+
acceptLanguage: headers?.acceptLanguage,
|
|
573
|
+
locales: i18nConfig.locales,
|
|
574
|
+
defaultLocale: i18nConfig.default
|
|
575
|
+
});
|
|
576
|
+
const match = pageMatcher.match(routePath);
|
|
426
577
|
if (!match) return null;
|
|
427
|
-
|
|
578
|
+
const i18nOpts = locale && i18nConfig ? {
|
|
579
|
+
locale,
|
|
580
|
+
config: i18nConfig,
|
|
581
|
+
routePattern: match.pattern
|
|
582
|
+
} : void 0;
|
|
583
|
+
return handlePageRequest(match.value, match.params, procedureMap, i18nOpts);
|
|
428
584
|
}
|
|
429
585
|
};
|
|
430
586
|
}
|
|
@@ -543,6 +699,7 @@ async function handleBatchHttp(req, router, hashToName) {
|
|
|
543
699
|
}
|
|
544
700
|
function createHttpHandler(router, opts) {
|
|
545
701
|
const hashToName = opts?.rpcHashMap ? new Map(Object.entries(opts.rpcHashMap.procedures).map(([n, h]) => [h, n])) : null;
|
|
702
|
+
if (hashToName) hashToName.set("__seam_i18n_query", "__seam_i18n_query");
|
|
546
703
|
const batchHash = opts?.rpcHashMap?.batch ?? null;
|
|
547
704
|
return async (req) => {
|
|
548
705
|
const url = new URL(req.url, "http://localhost");
|
|
@@ -592,7 +749,12 @@ function createHttpHandler(router, opts) {
|
|
|
592
749
|
}
|
|
593
750
|
if (req.method === "GET" && pathname.startsWith(PAGE_PREFIX) && router.hasPages) {
|
|
594
751
|
const pagePath = "/" + pathname.slice(12);
|
|
595
|
-
const
|
|
752
|
+
const headers = req.header ? {
|
|
753
|
+
url: req.url,
|
|
754
|
+
cookie: req.header("cookie") ?? void 0,
|
|
755
|
+
acceptLanguage: req.header("accept-language") ?? void 0
|
|
756
|
+
} : void 0;
|
|
757
|
+
const result = await router.handlePage(pagePath, headers);
|
|
596
758
|
if (result) return {
|
|
597
759
|
status: result.status,
|
|
598
760
|
headers: HTML_HEADER,
|
|
@@ -655,8 +817,25 @@ function buildLoaderFns(configs) {
|
|
|
655
817
|
for (const [key, config] of Object.entries(configs)) fns[key] = buildLoaderFn(config);
|
|
656
818
|
return fns;
|
|
657
819
|
}
|
|
820
|
+
function resolveTemplatePath(entry, defaultLocale) {
|
|
821
|
+
if (entry.template) return entry.template;
|
|
822
|
+
if (entry.templates) {
|
|
823
|
+
const locale = defaultLocale ?? Object.keys(entry.templates)[0];
|
|
824
|
+
const path = entry.templates[locale];
|
|
825
|
+
if (!path) throw new Error(`No template for locale "${locale}"`);
|
|
826
|
+
return path;
|
|
827
|
+
}
|
|
828
|
+
throw new Error("Manifest entry has neither 'template' nor 'templates'");
|
|
829
|
+
}
|
|
830
|
+
/** Load all locale templates for a manifest entry, keyed by locale */
|
|
831
|
+
function loadLocaleTemplates(entry, distDir) {
|
|
832
|
+
if (!entry.templates) return void 0;
|
|
833
|
+
const result = {};
|
|
834
|
+
for (const [locale, relPath] of Object.entries(entry.templates)) result[locale] = readFileSync(join(distDir, relPath), "utf-8");
|
|
835
|
+
return result;
|
|
836
|
+
}
|
|
658
837
|
/** Resolve parent chain for a layout, returning outer-to-inner order */
|
|
659
|
-
function resolveLayoutChain(layoutId, layoutEntries, templates) {
|
|
838
|
+
function resolveLayoutChain(layoutId, layoutEntries, templates, localeTemplatesMap) {
|
|
660
839
|
const chain = [];
|
|
661
840
|
let currentId = layoutId;
|
|
662
841
|
while (currentId) {
|
|
@@ -665,6 +844,7 @@ function resolveLayoutChain(layoutId, layoutEntries, templates) {
|
|
|
665
844
|
chain.push({
|
|
666
845
|
id: currentId,
|
|
667
846
|
template: templates[currentId],
|
|
847
|
+
localeTemplates: localeTemplatesMap[currentId],
|
|
668
848
|
loaders: buildLoaderFns(entry.loaders ?? {})
|
|
669
849
|
});
|
|
670
850
|
currentId = entry.parent;
|
|
@@ -673,16 +853,17 @@ function resolveLayoutChain(layoutId, layoutEntries, templates) {
|
|
|
673
853
|
return chain;
|
|
674
854
|
}
|
|
675
855
|
/** Resolve layout chain with lazy template getters (re-read from disk on each access) */
|
|
676
|
-
function resolveLayoutChainDev(layoutId, layoutEntries, distDir) {
|
|
856
|
+
function resolveLayoutChainDev(layoutId, layoutEntries, distDir, defaultLocale) {
|
|
677
857
|
const chain = [];
|
|
678
858
|
let currentId = layoutId;
|
|
679
859
|
while (currentId) {
|
|
680
860
|
const entry = layoutEntries[currentId];
|
|
681
861
|
if (!entry) break;
|
|
682
|
-
const layoutTemplatePath = join(distDir, entry
|
|
862
|
+
const layoutTemplatePath = join(distDir, resolveTemplatePath(entry, defaultLocale));
|
|
683
863
|
const def = {
|
|
684
864
|
id: currentId,
|
|
685
865
|
template: "",
|
|
866
|
+
localeTemplates: entry.templates ? makeLocaleTemplateGetters(entry.templates, distDir) : void 0,
|
|
686
867
|
loaders: buildLoaderFns(entry.loaders ?? {})
|
|
687
868
|
};
|
|
688
869
|
Object.defineProperty(def, "template", {
|
|
@@ -695,6 +876,33 @@ function resolveLayoutChainDev(layoutId, layoutEntries, distDir) {
|
|
|
695
876
|
chain.reverse();
|
|
696
877
|
return chain;
|
|
697
878
|
}
|
|
879
|
+
/** Create a proxy object that lazily reads locale templates from disk */
|
|
880
|
+
function makeLocaleTemplateGetters(templates, distDir) {
|
|
881
|
+
const obj = {};
|
|
882
|
+
for (const [locale, relPath] of Object.entries(templates)) {
|
|
883
|
+
const fullPath = join(distDir, relPath);
|
|
884
|
+
Object.defineProperty(obj, locale, {
|
|
885
|
+
get: () => readFileSync(fullPath, "utf-8"),
|
|
886
|
+
enumerable: true
|
|
887
|
+
});
|
|
888
|
+
}
|
|
889
|
+
return obj;
|
|
890
|
+
}
|
|
891
|
+
/** Merge i18n_keys from route + layout chain into a single list */
|
|
892
|
+
function mergeI18nKeys(route, layoutEntries) {
|
|
893
|
+
const keys = [];
|
|
894
|
+
if (route.layout) {
|
|
895
|
+
let currentId = route.layout;
|
|
896
|
+
while (currentId) {
|
|
897
|
+
const entry = layoutEntries[currentId];
|
|
898
|
+
if (!entry) break;
|
|
899
|
+
if (entry.i18n_keys) keys.push(...entry.i18n_keys);
|
|
900
|
+
currentId = entry.parent;
|
|
901
|
+
}
|
|
902
|
+
}
|
|
903
|
+
if (route.i18n_keys) keys.push(...route.i18n_keys);
|
|
904
|
+
return keys.length > 0 ? keys : void 0;
|
|
905
|
+
}
|
|
698
906
|
/** Load the RPC hash map from build output (returns undefined when obfuscation is off) */
|
|
699
907
|
function loadRpcHashMap(distDir) {
|
|
700
908
|
const hashMapPath = join(distDir, "rpc-hash-map.json");
|
|
@@ -704,33 +912,89 @@ function loadRpcHashMap(distDir) {
|
|
|
704
912
|
return;
|
|
705
913
|
}
|
|
706
914
|
}
|
|
915
|
+
/** Load i18n config and messages from build output */
|
|
916
|
+
function loadI18nMessages(distDir) {
|
|
917
|
+
const manifestPath = join(distDir, "route-manifest.json");
|
|
918
|
+
try {
|
|
919
|
+
const manifest = JSON.parse(readFileSync(manifestPath, "utf-8"));
|
|
920
|
+
if (!manifest.i18n) return null;
|
|
921
|
+
const mode = manifest.i18n.mode ?? "memory";
|
|
922
|
+
const cache = manifest.i18n.cache ?? false;
|
|
923
|
+
const routeHashes = manifest.i18n.route_hashes ?? {};
|
|
924
|
+
const contentHashes = manifest.i18n.content_hashes ?? {};
|
|
925
|
+
const messages = {};
|
|
926
|
+
if (mode === "memory") {
|
|
927
|
+
const i18nDir = join(distDir, "i18n");
|
|
928
|
+
for (const locale of manifest.i18n.locales) {
|
|
929
|
+
const localePath = join(i18nDir, `${locale}.json`);
|
|
930
|
+
if (existsSync(localePath)) messages[locale] = JSON.parse(readFileSync(localePath, "utf-8"));
|
|
931
|
+
else messages[locale] = {};
|
|
932
|
+
}
|
|
933
|
+
}
|
|
934
|
+
return {
|
|
935
|
+
locales: manifest.i18n.locales,
|
|
936
|
+
default: manifest.i18n.default,
|
|
937
|
+
mode,
|
|
938
|
+
cache,
|
|
939
|
+
routeHashes,
|
|
940
|
+
contentHashes,
|
|
941
|
+
messages,
|
|
942
|
+
distDir: mode === "paged" ? distDir : void 0
|
|
943
|
+
};
|
|
944
|
+
} catch {
|
|
945
|
+
return null;
|
|
946
|
+
}
|
|
947
|
+
}
|
|
707
948
|
function loadBuildOutput(distDir) {
|
|
708
949
|
const raw = readFileSync(join(distDir, "route-manifest.json"), "utf-8");
|
|
709
950
|
const manifest = JSON.parse(raw);
|
|
951
|
+
const defaultLocale = manifest.i18n?.default;
|
|
710
952
|
const layoutTemplates = {};
|
|
953
|
+
const layoutLocaleTemplates = {};
|
|
711
954
|
const layoutEntries = manifest.layouts ?? {};
|
|
712
|
-
for (const [id, entry] of Object.entries(layoutEntries))
|
|
955
|
+
for (const [id, entry] of Object.entries(layoutEntries)) {
|
|
956
|
+
layoutTemplates[id] = readFileSync(join(distDir, resolveTemplatePath(entry, defaultLocale)), "utf-8");
|
|
957
|
+
const lt = loadLocaleTemplates(entry, distDir);
|
|
958
|
+
if (lt) layoutLocaleTemplates[id] = lt;
|
|
959
|
+
}
|
|
713
960
|
const pages = {};
|
|
714
|
-
for (const [path, entry] of Object.entries(manifest.routes))
|
|
715
|
-
template
|
|
716
|
-
loaders
|
|
717
|
-
layoutChain
|
|
718
|
-
|
|
719
|
-
|
|
961
|
+
for (const [path, entry] of Object.entries(manifest.routes)) {
|
|
962
|
+
const template = readFileSync(join(distDir, resolveTemplatePath(entry, defaultLocale)), "utf-8");
|
|
963
|
+
const loaders = buildLoaderFns(entry.loaders);
|
|
964
|
+
const layoutChain = entry.layout ? resolveLayoutChain(entry.layout, layoutEntries, layoutTemplates, layoutLocaleTemplates) : [];
|
|
965
|
+
const i18nKeys = mergeI18nKeys(entry, layoutEntries);
|
|
966
|
+
pages[path] = {
|
|
967
|
+
template,
|
|
968
|
+
localeTemplates: loadLocaleTemplates(entry, distDir),
|
|
969
|
+
loaders,
|
|
970
|
+
layoutChain,
|
|
971
|
+
headMeta: entry.head_meta,
|
|
972
|
+
dataId: manifest.data_id,
|
|
973
|
+
i18nKeys
|
|
974
|
+
};
|
|
975
|
+
}
|
|
720
976
|
return pages;
|
|
721
977
|
}
|
|
722
|
-
/** Load build output with lazy template getters
|
|
978
|
+
/** Load build output with lazy template getters -- templates re-read from disk on each access */
|
|
723
979
|
function loadBuildOutputDev(distDir) {
|
|
724
980
|
const raw = readFileSync(join(distDir, "route-manifest.json"), "utf-8");
|
|
725
981
|
const manifest = JSON.parse(raw);
|
|
982
|
+
const defaultLocale = manifest.i18n?.default;
|
|
726
983
|
const layoutEntries = manifest.layouts ?? {};
|
|
727
984
|
const pages = {};
|
|
728
985
|
for (const [path, entry] of Object.entries(manifest.routes)) {
|
|
729
|
-
const templatePath = join(distDir, entry
|
|
986
|
+
const templatePath = join(distDir, resolveTemplatePath(entry, defaultLocale));
|
|
987
|
+
const loaders = buildLoaderFns(entry.loaders);
|
|
988
|
+
const layoutChain = entry.layout ? resolveLayoutChainDev(entry.layout, layoutEntries, distDir, defaultLocale) : [];
|
|
989
|
+
const localeTemplates = entry.templates ? makeLocaleTemplateGetters(entry.templates, distDir) : void 0;
|
|
990
|
+
const i18nKeys = mergeI18nKeys(entry, layoutEntries);
|
|
730
991
|
const page = {
|
|
731
992
|
template: "",
|
|
732
|
-
|
|
733
|
-
|
|
993
|
+
localeTemplates,
|
|
994
|
+
loaders,
|
|
995
|
+
layoutChain,
|
|
996
|
+
dataId: manifest.data_id,
|
|
997
|
+
i18nKeys
|
|
734
998
|
};
|
|
735
999
|
Object.defineProperty(page, "template", {
|
|
736
1000
|
get: () => readFileSync(templatePath, "utf-8"),
|
|
@@ -888,5 +1152,5 @@ function watchReloadTrigger(distDir, onReload) {
|
|
|
888
1152
|
}
|
|
889
1153
|
|
|
890
1154
|
//#endregion
|
|
891
|
-
export { SeamError, createDevProxy, createHttpHandler, createRouter, createStaticHandler, definePage, drainStream, fromCallback, loadBuildOutput, loadBuildOutputDev, loadRpcHashMap, serialize, sseCompleteEvent, sseDataEvent, sseErrorEvent, t, toWebResponse, watchReloadTrigger };
|
|
1155
|
+
export { SeamError, createDevProxy, createHttpHandler, createRouter, createStaticHandler, defaultStrategies, definePage, drainStream, fromAcceptLanguage, fromCallback, fromCookie, fromUrlPrefix, fromUrlQuery, loadBuildOutput, loadBuildOutputDev, loadI18nMessages, loadRpcHashMap, resolveChain, serialize, sseCompleteEvent, sseDataEvent, sseErrorEvent, t, toWebResponse, watchReloadTrigger };
|
|
892
1156
|
//# sourceMappingURL=index.js.map
|
package/dist/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.js","names":["primitives"],"sources":["../src/types/schema.ts","../src/types/primitives.ts","../src/types/composites.ts","../src/types/index.ts","../src/manifest/index.ts","../src/errors.ts","../src/validation/index.ts","../src/router/handler.ts","../src/page/handler.ts","../src/page/route-matcher.ts","../src/router/index.ts","../src/page/index.ts","../src/mime.ts","../src/http.ts","../src/page/build-loader.ts","../src/subscription.ts","../src/proxy.ts","../src/dev/reload-watcher.ts"],"sourcesContent":["/* packages/server/core/typescript/src/types/schema.ts */\n\nimport type { Schema } from \"jtd\";\n\nexport type JTDSchema = Schema;\n\nexport interface SchemaNode<TOutput = unknown> {\n readonly _schema: JTDSchema;\n /** Phantom type marker — never exists at runtime */\n readonly _output: TOutput;\n}\n\nexport interface OptionalSchemaNode<TOutput = unknown> extends SchemaNode<TOutput> {\n readonly _optional: true;\n}\n\nexport type Infer<T extends SchemaNode> = T[\"_output\"];\n\nexport function createSchemaNode<T>(schema: JTDSchema): SchemaNode<T> {\n return { _schema: schema } as SchemaNode<T>;\n}\n\nexport function createOptionalSchemaNode<T>(schema: JTDSchema): OptionalSchemaNode<T> {\n return { _schema: schema, _optional: true } as OptionalSchemaNode<T>;\n}\n","/* packages/server/core/typescript/src/types/primitives.ts */\n\nimport type { SchemaNode } from \"./schema.js\";\nimport { createSchemaNode } from \"./schema.js\";\n\nexport function string(): SchemaNode<string> {\n return createSchemaNode<string>({ type: \"string\" });\n}\n\nexport function boolean(): SchemaNode<boolean> {\n return createSchemaNode<boolean>({ type: \"boolean\" });\n}\n\nexport function int8(): SchemaNode<number> {\n return createSchemaNode<number>({ type: \"int8\" });\n}\n\nexport function int16(): SchemaNode<number> {\n return createSchemaNode<number>({ type: \"int16\" });\n}\n\nexport function int32(): SchemaNode<number> {\n return createSchemaNode<number>({ type: \"int32\" });\n}\n\nexport function uint8(): SchemaNode<number> {\n return createSchemaNode<number>({ type: \"uint8\" });\n}\n\nexport function uint16(): SchemaNode<number> {\n return createSchemaNode<number>({ type: \"uint16\" });\n}\n\nexport function uint32(): SchemaNode<number> {\n return createSchemaNode<number>({ type: \"uint32\" });\n}\n\nexport function float32(): SchemaNode<number> {\n return createSchemaNode<number>({ type: \"float32\" });\n}\n\nexport function float64(): SchemaNode<number> {\n return createSchemaNode<number>({ type: \"float64\" });\n}\n\nexport function timestamp(): SchemaNode<string> {\n return createSchemaNode<string>({ type: \"timestamp\" });\n}\n\nexport function html(): SchemaNode<string> {\n return createSchemaNode<string>({ type: \"string\", metadata: { format: \"html\" } });\n}\n","/* packages/server/core/typescript/src/types/composites.ts */\n\nimport type { SchemaNode, OptionalSchemaNode, Infer, JTDSchema } from \"./schema.js\";\nimport { createSchemaNode, createOptionalSchemaNode } from \"./schema.js\";\n\n// -- Type-level utilities --\n\ntype Simplify<T> = { [K in keyof T]: T[K] } & {};\n\ntype RequiredKeys<T extends Record<string, SchemaNode>> = {\n [K in keyof T]: T[K] extends OptionalSchemaNode ? never : K;\n}[keyof T];\n\ntype OptionalKeys<T extends Record<string, SchemaNode>> = {\n [K in keyof T]: T[K] extends OptionalSchemaNode ? K : never;\n}[keyof T];\n\ntype InferObject<T extends Record<string, SchemaNode>> = Simplify<\n { [K in RequiredKeys<T>]: Infer<T[K]> } & { [K in OptionalKeys<T>]?: Infer<T[K]> }\n>;\n\n// -- Builders --\n\nexport function object<T extends Record<string, SchemaNode>>(\n fields: T,\n): SchemaNode<InferObject<T>> {\n const properties: Record<string, JTDSchema> = {};\n const optionalProperties: Record<string, JTDSchema> = {};\n\n for (const [key, node] of Object.entries(fields)) {\n if (\"_optional\" in node && node._optional === true) {\n optionalProperties[key] = node._schema;\n } else {\n properties[key] = node._schema;\n }\n }\n\n const schema: Record<string, unknown> = {};\n if (Object.keys(properties).length > 0 || Object.keys(optionalProperties).length === 0) {\n schema.properties = properties;\n }\n if (Object.keys(optionalProperties).length > 0) {\n schema.optionalProperties = optionalProperties;\n }\n\n return createSchemaNode<InferObject<T>>(schema as JTDSchema);\n}\n\nexport function optional<T>(node: SchemaNode<T>): OptionalSchemaNode<T> {\n return createOptionalSchemaNode<T>(node._schema);\n}\n\nexport function array<T>(node: SchemaNode<T>): SchemaNode<T[]> {\n return createSchemaNode<T[]>({ elements: node._schema });\n}\n\nexport function nullable<T>(node: SchemaNode<T>): SchemaNode<T | null> {\n return createSchemaNode<T | null>({ ...node._schema, nullable: true } as JTDSchema);\n}\n\nexport function enumType<const T extends readonly string[]>(values: T): SchemaNode<T[number]> {\n return createSchemaNode<T[number]>({ enum: [...values] } as JTDSchema);\n}\n\nexport function values<T>(node: SchemaNode<T>): SchemaNode<Record<string, T>> {\n return createSchemaNode<Record<string, T>>({ values: node._schema });\n}\n\ntype DiscriminatorUnion<TTag extends string, TMapping extends Record<string, SchemaNode>> = {\n [K in keyof TMapping & string]: Simplify<{ [P in TTag]: K } & Infer<TMapping[K]>>;\n}[keyof TMapping & string];\n\nexport function discriminator<\n TTag extends string,\n TMapping extends Record<string, SchemaNode<Record<string, unknown>>>,\n>(tag: TTag, mapping: TMapping): SchemaNode<DiscriminatorUnion<TTag, TMapping>> {\n const jtdMapping: Record<string, JTDSchema> = {};\n for (const [key, node] of Object.entries(mapping)) {\n jtdMapping[key] = node._schema;\n }\n return createSchemaNode<DiscriminatorUnion<TTag, TMapping>>({\n discriminator: tag,\n mapping: jtdMapping,\n } as JTDSchema);\n}\n","/* packages/server/core/typescript/src/types/index.ts */\n\nimport * as primitives from \"./primitives.js\";\nimport {\n object,\n optional,\n array,\n nullable,\n enumType,\n values,\n discriminator,\n} from \"./composites.js\";\n\nexport const t = {\n ...primitives,\n object,\n optional,\n array,\n nullable,\n enum: enumType,\n values,\n discriminator,\n} as const;\n","/* packages/server/core/typescript/src/manifest/index.ts */\n\nimport type { Schema } from \"jtd\";\nimport type { SchemaNode } from \"../types/schema.js\";\n\nexport type ProcedureType = \"query\" | \"subscription\";\n\nexport interface ProcedureEntry {\n type: ProcedureType;\n input: Schema;\n output: Schema;\n}\n\nexport interface ProcedureManifest {\n version: string;\n procedures: Record<string, ProcedureEntry>;\n}\n\nexport function buildManifest(\n definitions: Record<string, { input: SchemaNode; output: SchemaNode; type?: string }>,\n): ProcedureManifest {\n const mapped: ProcedureManifest[\"procedures\"] = {};\n\n for (const [name, def] of Object.entries(definitions)) {\n mapped[name] = {\n type: def.type === \"subscription\" ? \"subscription\" : \"query\",\n input: def.input._schema,\n output: def.output._schema,\n };\n }\n\n return { version: \"0.1.0\", procedures: mapped };\n}\n","/* packages/server/core/typescript/src/errors.ts */\n\nexport type ErrorCode =\n | \"VALIDATION_ERROR\"\n | \"NOT_FOUND\"\n | \"UNAUTHORIZED\"\n | \"FORBIDDEN\"\n | \"RATE_LIMITED\"\n | \"INTERNAL_ERROR\"\n | (string & {});\n\nexport const DEFAULT_STATUS: Record<string, number> = {\n VALIDATION_ERROR: 400,\n UNAUTHORIZED: 401,\n FORBIDDEN: 403,\n NOT_FOUND: 404,\n RATE_LIMITED: 429,\n INTERNAL_ERROR: 500,\n};\n\nexport class SeamError extends Error {\n readonly code: string;\n readonly status: number;\n\n constructor(code: string, message: string, status?: number) {\n super(message);\n this.code = code;\n this.status = status ?? DEFAULT_STATUS[code] ?? 500;\n this.name = \"SeamError\";\n }\n\n toJSON() {\n return {\n error: {\n code: this.code,\n message: this.message,\n },\n };\n }\n}\n","/* packages/server/core/typescript/src/validation/index.ts */\n\nimport { validate } from \"jtd\";\nimport type { Schema, ValidationError as JTDValidationError } from \"jtd\";\n\nexport interface ValidationResult {\n valid: boolean;\n errors: JTDValidationError[];\n}\n\nexport function validateInput(schema: Schema, data: unknown): ValidationResult {\n const errors = validate(schema, data, { maxDepth: 32, maxErrors: 10 });\n return {\n valid: errors.length === 0,\n errors,\n };\n}\n\nexport function formatValidationErrors(errors: JTDValidationError[]): string {\n return errors\n .map((e) => {\n const path = e.instancePath.length > 0 ? e.instancePath.join(\"/\") : \"(root)\";\n const schema = e.schemaPath.join(\"/\");\n return `${path} (schema: ${schema})`;\n })\n .join(\"; \");\n}\n","/* packages/server/core/typescript/src/router/handler.ts */\n\nimport { SeamError } from \"../errors.js\";\nimport type { HandleResult, InternalProcedure, InternalSubscription } from \"../procedure.js\";\nimport { validateInput, formatValidationErrors } from \"../validation/index.js\";\n\nexport type { HandleResult, InternalProcedure } from \"../procedure.js\";\n\nexport async function handleRequest(\n procedures: Map<string, InternalProcedure>,\n procedureName: string,\n rawBody: unknown,\n validateOutput?: boolean,\n): Promise<HandleResult> {\n const procedure = procedures.get(procedureName);\n if (!procedure) {\n return {\n status: 404,\n body: new SeamError(\"NOT_FOUND\", `Procedure '${procedureName}' not found`).toJSON(),\n };\n }\n\n const validation = validateInput(procedure.inputSchema, rawBody);\n if (!validation.valid) {\n const details = formatValidationErrors(validation.errors);\n return {\n status: 400,\n body: new SeamError(\"VALIDATION_ERROR\", `Input validation failed: ${details}`).toJSON(),\n };\n }\n\n try {\n const result = await procedure.handler({ input: rawBody });\n\n if (validateOutput) {\n const outValidation = validateInput(procedure.outputSchema, result);\n if (!outValidation.valid) {\n const details = formatValidationErrors(outValidation.errors);\n return {\n status: 500,\n body: new SeamError(\"INTERNAL_ERROR\", `Output validation failed: ${details}`).toJSON(),\n };\n }\n }\n\n return { status: 200, body: result };\n } catch (error) {\n if (error instanceof SeamError) {\n return { status: error.status, body: error.toJSON() };\n }\n const message = error instanceof Error ? error.message : \"Unknown error\";\n return {\n status: 500,\n body: new SeamError(\"INTERNAL_ERROR\", message).toJSON(),\n };\n }\n}\n\nexport interface BatchCall {\n procedure: string;\n input: unknown;\n}\n\nexport type BatchResultItem =\n | { ok: true; data: unknown }\n | { ok: false; error: { code: string; message: string } };\n\nexport async function handleBatchRequest(\n procedures: Map<string, InternalProcedure>,\n calls: BatchCall[],\n validateOutput?: boolean,\n): Promise<{ results: BatchResultItem[] }> {\n const results = await Promise.all(\n calls.map(async (call) => {\n const result = await handleRequest(procedures, call.procedure, call.input, validateOutput);\n if (result.status === 200) {\n return { ok: true as const, data: result.body };\n }\n const envelope = result.body as { error: { code: string; message: string } };\n return { ok: false as const, error: envelope.error };\n }),\n );\n return { results };\n}\n\nexport async function* handleSubscription(\n subscriptions: Map<string, InternalSubscription>,\n name: string,\n rawInput: unknown,\n validateOutput?: boolean,\n): AsyncIterable<unknown> {\n const sub = subscriptions.get(name);\n if (!sub) {\n throw new SeamError(\"NOT_FOUND\", `Subscription '${name}' not found`);\n }\n\n const validation = validateInput(sub.inputSchema, rawInput);\n if (!validation.valid) {\n const details = formatValidationErrors(validation.errors);\n throw new SeamError(\"VALIDATION_ERROR\", `Input validation failed: ${details}`);\n }\n\n for await (const value of sub.handler({ input: rawInput })) {\n if (validateOutput) {\n const outValidation = validateInput(sub.outputSchema, value);\n if (!outValidation.valid) {\n const details = formatValidationErrors(outValidation.errors);\n throw new SeamError(\"INTERNAL_ERROR\", `Output validation failed: ${details}`);\n }\n }\n yield value;\n }\n}\n","/* packages/server/core/typescript/src/page/handler.ts */\n\nimport { inject, escapeHtml } from \"@canmi/seam-injector\";\nimport { SeamError } from \"../errors.js\";\nimport type { InternalProcedure } from \"../procedure.js\";\nimport type { PageDef, LoaderFn } from \"./index.js\";\n\nexport interface PageTiming {\n /** Procedure execution time in milliseconds */\n dataFetch: number;\n /** Template injection time in milliseconds */\n inject: number;\n}\n\nexport interface HandlePageResult {\n status: number;\n html: string;\n timing?: PageTiming;\n}\n\n/** Flatten keyed loader results: spread object values into a flat map for slot resolution */\nfunction flattenForSlots(keyed: Record<string, unknown>): Record<string, unknown> {\n const merged: Record<string, unknown> = { ...keyed };\n for (const value of Object.values(keyed)) {\n if (value && typeof value === \"object\" && !Array.isArray(value)) {\n Object.assign(merged, value as Record<string, unknown>);\n }\n }\n return merged;\n}\n\n/** Execute loaders, returning keyed results */\nasync function executeLoaders(\n loaders: Record<string, LoaderFn>,\n params: Record<string, string>,\n procedures: Map<string, InternalProcedure>,\n): Promise<Record<string, unknown>> {\n const entries = Object.entries(loaders);\n const results = await Promise.all(\n entries.map(async ([key, loader]) => {\n const { procedure, input } = loader(params);\n const proc = procedures.get(procedure);\n if (!proc) throw new SeamError(\"INTERNAL_ERROR\", `Procedure '${procedure}' not found`);\n // Skip JTD validation -- loader input is trusted server-side code\n const result = await proc.handler({ input });\n return [key, result] as const;\n }),\n );\n return Object.fromEntries(results);\n}\n\n/** Split-inject a layout template around its outlet marker */\nfunction injectLayout(template: string, data: Record<string, unknown>, inner: string): string {\n const outletMarker = \"<!--seam:outlet-->\";\n const outletIdx = template.indexOf(outletMarker);\n if (outletIdx === -1) {\n return inject(template, data, { skipDataScript: true });\n }\n const before = template.slice(0, outletIdx);\n const after = template.slice(outletIdx + outletMarker.length);\n const injectedBefore = inject(before, data, { skipDataScript: true });\n const injectedAfter = inject(after, data, { skipDataScript: true });\n return injectedBefore + inner + injectedAfter;\n}\n\nexport async function handlePageRequest(\n page: PageDef,\n params: Record<string, string>,\n procedures: Map<string, InternalProcedure>,\n): Promise<HandlePageResult> {\n try {\n const t0 = performance.now();\n const layoutChain = page.layoutChain ?? [];\n\n // Execute all loaders (layout chain + page) in parallel\n const loaderResults = await Promise.all([\n ...layoutChain.map((layout) => executeLoaders(layout.loaders, params, procedures)),\n executeLoaders(page.loaders, params, procedures),\n ]);\n\n const t1 = performance.now();\n\n // Partition: first N results are layout, last is page\n const layoutResults = loaderResults.slice(0, layoutChain.length);\n const pageKeyed = loaderResults[loaderResults.length - 1];\n\n // Inject page template\n let innerContent = inject(page.template, flattenForSlots(pageKeyed), { skipDataScript: true });\n\n // Compose layouts from innermost to outermost\n const layoutKeyed: Record<string, Record<string, unknown>> = {};\n for (let i = layoutChain.length - 1; i >= 0; i--) {\n const layout = layoutChain[i];\n const data = layoutResults[i];\n layoutKeyed[layout.id] = data;\n innerContent = injectLayout(layout.template, flattenForSlots(data), innerContent);\n }\n\n // Inject page-level metadata (<title>, <meta>, <link>) into <head>.\n // These were extracted from the page fragment at build time so they don't\n // end up inside the root div via <!--seam:outlet--> substitution.\n // Inserted after <meta charset=\"utf-8\"> so page <title> wins over layout metadata\n // (first <title> wins in HTML5).\n if (page.headMeta) {\n const injectedMeta = inject(page.headMeta, flattenForSlots(pageKeyed), {\n skipDataScript: true,\n });\n const charset = '<meta charset=\"utf-8\">';\n const charsetIdx = innerContent.indexOf(charset);\n if (charsetIdx !== -1) {\n const insertAt = charsetIdx + charset.length;\n innerContent =\n innerContent.slice(0, insertAt) + injectedMeta + innerContent.slice(insertAt);\n }\n }\n\n // Build __SEAM_DATA__: page data at top level, layout data under _layouts\n const seamData: Record<string, unknown> = { ...pageKeyed };\n if (Object.keys(layoutKeyed).length > 0) {\n seamData._layouts = layoutKeyed;\n }\n\n const script = `<script id=\"__SEAM_DATA__\" type=\"application/json\">${JSON.stringify(seamData)}</script>`;\n const bodyClose = innerContent.lastIndexOf(\"</body>\");\n let html: string;\n if (bodyClose !== -1) {\n html = innerContent.slice(0, bodyClose) + script + innerContent.slice(bodyClose);\n } else {\n html = innerContent + script;\n }\n\n const t2 = performance.now();\n\n return {\n status: 200,\n html,\n timing: { dataFetch: t1 - t0, inject: t2 - t1 },\n };\n } catch (error) {\n const message = error instanceof Error ? error.message : \"Unknown error\";\n return {\n status: 500,\n html: `<!DOCTYPE html><html><body><h1>500 Internal Server Error</h1><p>${escapeHtml(message)}</p></body></html>`,\n };\n }\n}\n","/* packages/server/core/typescript/src/page/route-matcher.ts */\n\ninterface CompiledRoute {\n segments: RouteSegment[];\n}\n\ntype RouteSegment = { kind: \"static\"; value: string } | { kind: \"param\"; name: string };\n\nfunction compileRoute(pattern: string): CompiledRoute {\n const segments: RouteSegment[] = pattern\n .split(\"/\")\n .filter(Boolean)\n .map((seg) =>\n seg.startsWith(\":\") ? { kind: \"param\", name: seg.slice(1) } : { kind: \"static\", value: seg },\n );\n return { segments };\n}\n\nfunction matchRoute(segments: RouteSegment[], pathParts: string[]): Record<string, string> | null {\n if (segments.length !== pathParts.length) return null;\n const params: Record<string, string> = {};\n for (let i = 0; i < segments.length; i++) {\n const seg = segments[i];\n if (seg.kind === \"static\") {\n if (seg.value !== pathParts[i]) return null;\n } else {\n params[seg.name] = pathParts[i];\n }\n }\n return params;\n}\n\nexport class RouteMatcher<T> {\n private routes: { compiled: CompiledRoute; value: T }[] = [];\n\n add(pattern: string, value: T): void {\n this.routes.push({ compiled: compileRoute(pattern), value });\n }\n\n match(path: string): { value: T; params: Record<string, string> } | null {\n const parts = path.split(\"/\").filter(Boolean);\n for (const route of this.routes) {\n const params = matchRoute(route.compiled.segments, parts);\n if (params) return { value: route.value, params };\n }\n return null;\n }\n}\n","/* packages/server/core/typescript/src/router/index.ts */\n\nimport type { SchemaNode } from \"../types/schema.js\";\nimport type { ProcedureManifest } from \"../manifest/index.js\";\nimport type { HandleResult, InternalProcedure } from \"./handler.js\";\nimport type { InternalSubscription } from \"../procedure.js\";\nimport type { HandlePageResult } from \"../page/handler.js\";\nimport type { PageDef } from \"../page/index.js\";\nimport { buildManifest } from \"../manifest/index.js\";\nimport { handleRequest, handleSubscription, handleBatchRequest } from \"./handler.js\";\nimport type { BatchCall, BatchResultItem } from \"./handler.js\";\nimport { handlePageRequest } from \"../page/handler.js\";\nimport { RouteMatcher } from \"../page/route-matcher.js\";\n\nexport interface ProcedureDef<TIn = unknown, TOut = unknown> {\n input: SchemaNode<TIn>;\n output: SchemaNode<TOut>;\n handler: (params: { input: TIn }) => TOut | Promise<TOut>;\n}\n\nexport interface SubscriptionDef<TIn = unknown, TOut = unknown> {\n type: \"subscription\";\n input: SchemaNode<TIn>;\n output: SchemaNode<TOut>;\n handler: (params: { input: TIn }) => AsyncIterable<TOut>;\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nexport type DefinitionMap = Record<string, ProcedureDef<any, any> | SubscriptionDef<any, any>>;\n\nfunction isSubscriptionDef(def: ProcedureDef | SubscriptionDef): def is SubscriptionDef {\n return \"type\" in def && def.type === \"subscription\";\n}\n\nexport interface RouterOptions {\n pages?: Record<string, PageDef>;\n validateOutput?: boolean;\n}\n\nexport interface Router<T extends DefinitionMap> {\n manifest(): ProcedureManifest;\n handle(procedureName: string, body: unknown): Promise<HandleResult>;\n handleBatch(calls: BatchCall[]): Promise<{ results: BatchResultItem[] }>;\n handleSubscription(name: string, input: unknown): AsyncIterable<unknown>;\n handlePage(path: string): Promise<HandlePageResult | null>;\n readonly hasPages: boolean;\n /** Exposed for adapter access to the definitions */\n readonly procedures: T;\n}\n\nexport function createRouter<T extends DefinitionMap>(\n procedures: T,\n opts?: RouterOptions,\n): Router<T> {\n const procedureMap = new Map<string, InternalProcedure>();\n const subscriptionMap = new Map<string, InternalSubscription>();\n\n for (const [name, def] of Object.entries(procedures)) {\n if (isSubscriptionDef(def)) {\n subscriptionMap.set(name, {\n inputSchema: def.input._schema,\n outputSchema: def.output._schema,\n handler: def.handler as InternalSubscription[\"handler\"],\n });\n } else {\n procedureMap.set(name, {\n inputSchema: def.input._schema,\n outputSchema: def.output._schema,\n handler: def.handler as InternalProcedure[\"handler\"],\n });\n }\n }\n\n const shouldValidateOutput =\n opts?.validateOutput ??\n (typeof process !== \"undefined\" && process.env.NODE_ENV !== \"production\");\n\n const pageMatcher = new RouteMatcher<PageDef>();\n const pages = opts?.pages;\n if (pages) {\n for (const [pattern, page] of Object.entries(pages)) {\n pageMatcher.add(pattern, page);\n }\n }\n\n return {\n procedures,\n hasPages: !!pages && Object.keys(pages).length > 0,\n manifest() {\n return buildManifest(procedures);\n },\n handle(procedureName, body) {\n return handleRequest(procedureMap, procedureName, body, shouldValidateOutput);\n },\n handleBatch(calls) {\n return handleBatchRequest(procedureMap, calls, shouldValidateOutput);\n },\n handleSubscription(name, input) {\n return handleSubscription(subscriptionMap, name, input, shouldValidateOutput);\n },\n async handlePage(path) {\n const match = pageMatcher.match(path);\n if (!match) return null;\n return handlePageRequest(match.value, match.params, procedureMap);\n },\n };\n}\n","/* packages/server/core/typescript/src/page/index.ts */\n\nexport interface LoaderResult {\n procedure: string;\n input: unknown;\n}\n\nexport type LoaderFn = (params: Record<string, string>) => LoaderResult;\n\nexport interface LayoutDef {\n id: string;\n template: string;\n loaders: Record<string, LoaderFn>;\n}\n\nexport interface PageDef {\n template: string;\n loaders: Record<string, LoaderFn>;\n layoutChain?: LayoutDef[];\n headMeta?: string;\n}\n\nexport function definePage(config: PageDef): PageDef {\n return { ...config, layoutChain: config.layoutChain ?? [] };\n}\n","/* packages/server/core/typescript/src/mime.ts */\n\nexport const MIME_TYPES: Record<string, string> = {\n \".js\": \"application/javascript\",\n \".mjs\": \"application/javascript\",\n \".css\": \"text/css\",\n \".html\": \"text/html\",\n \".json\": \"application/json\",\n \".svg\": \"image/svg+xml\",\n \".png\": \"image/png\",\n \".jpg\": \"image/jpeg\",\n \".jpeg\": \"image/jpeg\",\n \".gif\": \"image/gif\",\n \".woff\": \"font/woff\",\n \".woff2\": \"font/woff2\",\n \".ttf\": \"font/ttf\",\n \".ico\": \"image/x-icon\",\n \".map\": \"application/json\",\n \".ts\": \"application/javascript\",\n \".tsx\": \"application/javascript\",\n};\n","/* packages/server/core/typescript/src/http.ts */\n\nimport { readFile } from \"node:fs/promises\";\nimport { join, extname } from \"node:path\";\nimport type { Router, DefinitionMap } from \"./router/index.js\";\nimport { SeamError } from \"./errors.js\";\nimport { MIME_TYPES } from \"./mime.js\";\n\nexport interface HttpRequest {\n method: string;\n url: string;\n body: () => Promise<unknown>;\n}\n\nexport interface HttpBodyResponse {\n status: number;\n headers: Record<string, string>;\n body: unknown;\n}\n\nexport interface HttpStreamResponse {\n status: number;\n headers: Record<string, string>;\n stream: AsyncIterable<string>;\n}\n\nexport type HttpResponse = HttpBodyResponse | HttpStreamResponse;\n\nexport type HttpHandler = (req: HttpRequest) => Promise<HttpResponse>;\n\nexport interface RpcHashMap {\n procedures: Record<string, string>;\n batch: string;\n}\n\nexport interface HttpHandlerOptions {\n staticDir?: string;\n fallback?: HttpHandler;\n rpcHashMap?: RpcHashMap;\n}\n\nconst RPC_PREFIX = \"/_seam/rpc/\";\nconst PAGE_PREFIX = \"/_seam/page/\";\nconst STATIC_PREFIX = \"/_seam/static/\";\nconst SUBSCRIBE_PREFIX = \"/_seam/subscribe/\";\nconst MANIFEST_PATH = \"/_seam/manifest.json\";\n\nconst JSON_HEADER = { \"Content-Type\": \"application/json\" };\nconst HTML_HEADER = { \"Content-Type\": \"text/html; charset=utf-8\" };\nconst SSE_HEADER = {\n \"Content-Type\": \"text/event-stream\",\n \"Cache-Control\": \"no-cache\",\n Connection: \"keep-alive\",\n};\nconst IMMUTABLE_CACHE = \"public, max-age=31536000, immutable\";\n\nfunction jsonResponse(status: number, body: unknown): HttpBodyResponse {\n return { status, headers: JSON_HEADER, body };\n}\n\nfunction errorResponse(status: number, code: string, message: string): HttpBodyResponse {\n return jsonResponse(status, new SeamError(code, message).toJSON());\n}\n\nasync function handleStaticAsset(assetPath: string, staticDir: string): Promise<HttpBodyResponse> {\n if (assetPath.includes(\"..\")) {\n return errorResponse(403, \"VALIDATION_ERROR\", \"Forbidden\");\n }\n\n const filePath = join(staticDir, assetPath);\n try {\n const content = await readFile(filePath, \"utf-8\");\n const ext = extname(filePath);\n const contentType = MIME_TYPES[ext] || \"application/octet-stream\";\n return {\n status: 200,\n headers: {\n \"Content-Type\": contentType,\n \"Cache-Control\": IMMUTABLE_CACHE,\n },\n body: content,\n };\n } catch {\n return errorResponse(404, \"NOT_FOUND\", \"Asset not found\");\n }\n}\n\n/** Format a single SSE data event */\nexport function sseDataEvent(data: unknown): string {\n return `event: data\\ndata: ${JSON.stringify(data)}\\n\\n`;\n}\n\n/** Format an SSE error event */\nexport function sseErrorEvent(code: string, message: string): string {\n return `event: error\\ndata: ${JSON.stringify({ code, message })}\\n\\n`;\n}\n\n/** Format an SSE complete event */\nexport function sseCompleteEvent(): string {\n return \"event: complete\\ndata: {}\\n\\n\";\n}\n\nasync function* sseStream<T extends DefinitionMap>(\n router: Router<T>,\n name: string,\n input: unknown,\n): AsyncIterable<string> {\n try {\n for await (const value of router.handleSubscription(name, input)) {\n yield sseDataEvent(value);\n }\n yield sseCompleteEvent();\n } catch (error) {\n if (error instanceof SeamError) {\n yield sseErrorEvent(error.code, error.message);\n } else {\n const message = error instanceof Error ? error.message : \"Unknown error\";\n yield sseErrorEvent(\"INTERNAL_ERROR\", message);\n }\n }\n}\n\nasync function handleBatchHttp<T extends DefinitionMap>(\n req: HttpRequest,\n router: Router<T>,\n hashToName: Map<string, string> | null,\n): Promise<HttpBodyResponse> {\n let body: unknown;\n try {\n body = await req.body();\n } catch {\n return errorResponse(400, \"VALIDATION_ERROR\", \"Invalid JSON body\");\n }\n if (!body || typeof body !== \"object\" || !Array.isArray((body as { calls?: unknown }).calls)) {\n return errorResponse(400, \"VALIDATION_ERROR\", \"Batch request must have a 'calls' array\");\n }\n const calls = (body as { calls: Array<{ procedure?: unknown; input?: unknown }> }).calls.map(\n (c) => ({\n procedure:\n typeof c.procedure === \"string\" ? (hashToName?.get(c.procedure) ?? c.procedure) : \"\",\n input: c.input ?? {},\n }),\n );\n const result = await router.handleBatch(calls);\n return jsonResponse(200, result);\n}\n\nexport function createHttpHandler<T extends DefinitionMap>(\n router: Router<T>,\n opts?: HttpHandlerOptions,\n): HttpHandler {\n // Build reverse lookup (hash -> original name) when obfuscation is active\n const hashToName: Map<string, string> | null = opts?.rpcHashMap\n ? new Map(Object.entries(opts.rpcHashMap.procedures).map(([n, h]) => [h, n]))\n : null;\n const batchHash = opts?.rpcHashMap?.batch ?? null;\n\n return async (req) => {\n const url = new URL(req.url, \"http://localhost\");\n const { pathname } = url;\n\n if (req.method === \"GET\" && pathname === MANIFEST_PATH) {\n if (opts?.rpcHashMap) return errorResponse(403, \"FORBIDDEN\", \"Manifest disabled\");\n return jsonResponse(200, router.manifest());\n }\n\n if (req.method === \"POST\" && pathname.startsWith(RPC_PREFIX)) {\n let name = pathname.slice(RPC_PREFIX.length);\n if (!name) {\n return errorResponse(404, \"NOT_FOUND\", \"Empty procedure name\");\n }\n\n // Batch: match both original \"_batch\" and hashed batch endpoint\n if (name === \"_batch\" || (batchHash && name === batchHash)) {\n return handleBatchHttp(req, router, hashToName);\n }\n\n // Resolve hash -> original name when obfuscation is active\n if (hashToName) {\n const resolved = hashToName.get(name);\n if (!resolved) return errorResponse(404, \"NOT_FOUND\", \"Not found\");\n name = resolved;\n }\n\n let body: unknown;\n try {\n body = await req.body();\n } catch {\n return errorResponse(400, \"VALIDATION_ERROR\", \"Invalid JSON body\");\n }\n\n const result = await router.handle(name, body);\n return jsonResponse(result.status, result.body);\n }\n\n if (req.method === \"GET\" && pathname.startsWith(SUBSCRIBE_PREFIX)) {\n let name = pathname.slice(SUBSCRIBE_PREFIX.length);\n if (!name) {\n return errorResponse(404, \"NOT_FOUND\", \"Empty subscription name\");\n }\n\n // Resolve hash -> original name for subscriptions\n if (hashToName) {\n const resolved = hashToName.get(name);\n if (!resolved) return errorResponse(404, \"NOT_FOUND\", \"Not found\");\n name = resolved;\n }\n\n const rawInput = url.searchParams.get(\"input\");\n let input: unknown;\n try {\n input = rawInput ? JSON.parse(rawInput) : {};\n } catch {\n return errorResponse(400, \"VALIDATION_ERROR\", \"Invalid input query parameter\");\n }\n\n return { status: 200, headers: SSE_HEADER, stream: sseStream(router, name, input) };\n }\n\n if (req.method === \"GET\" && pathname.startsWith(PAGE_PREFIX) && router.hasPages) {\n const pagePath = \"/\" + pathname.slice(PAGE_PREFIX.length);\n const result = await router.handlePage(pagePath);\n if (result) {\n return { status: result.status, headers: HTML_HEADER, body: result.html };\n }\n }\n\n if (req.method === \"GET\" && pathname.startsWith(STATIC_PREFIX) && opts?.staticDir) {\n const assetPath = pathname.slice(STATIC_PREFIX.length);\n return handleStaticAsset(assetPath, opts.staticDir);\n }\n\n if (opts?.fallback) return opts.fallback(req);\n return errorResponse(404, \"NOT_FOUND\", \"Not found\");\n };\n}\n\nexport function serialize(body: unknown): string {\n return typeof body === \"string\" ? body : JSON.stringify(body);\n}\n\n/** Consume an async stream chunk-by-chunk; return false from write to stop early. */\nexport async function drainStream(\n stream: AsyncIterable<string>,\n write: (chunk: string) => boolean | void,\n): Promise<void> {\n try {\n for await (const chunk of stream) {\n if (write(chunk) === false) break;\n }\n } catch {\n // Client disconnected\n }\n}\n\n/** Convert an HttpResponse to a Web API Response (for adapters using fetch-compatible runtimes) */\nexport function toWebResponse(result: HttpResponse): Response {\n if (\"stream\" in result) {\n const stream = result.stream;\n const encoder = new TextEncoder();\n const readable = new ReadableStream({\n async start(controller) {\n await drainStream(stream, (chunk) => {\n controller.enqueue(encoder.encode(chunk));\n });\n controller.close();\n },\n });\n return new Response(readable, { status: result.status, headers: result.headers });\n }\n return new Response(serialize(result.body), { status: result.status, headers: result.headers });\n}\n","/* packages/server/core/typescript/src/page/build-loader.ts */\n\nimport { readFileSync } from \"node:fs\";\nimport { join } from \"node:path\";\nimport type { PageDef, LayoutDef, LoaderFn, LoaderResult } from \"./index.js\";\nimport type { RpcHashMap } from \"../http.js\";\n\ninterface RouteManifest {\n layouts?: Record<string, LayoutManifestEntry>;\n routes: Record<string, RouteManifestEntry>;\n}\n\ninterface LayoutManifestEntry {\n template: string;\n loaders?: Record<string, LoaderConfig>;\n parent?: string;\n}\n\ninterface RouteManifestEntry {\n template: string;\n layout?: string;\n loaders: Record<string, LoaderConfig>;\n head_meta?: string;\n}\n\ninterface LoaderConfig {\n procedure: string;\n params?: Record<string, ParamConfig>;\n}\n\ninterface ParamConfig {\n from: \"route\";\n type?: \"string\" | \"int\";\n}\n\nfunction buildLoaderFn(config: LoaderConfig): LoaderFn {\n return (params: Record<string, string>): LoaderResult => {\n const input: Record<string, unknown> = {};\n if (config.params) {\n for (const [key, mapping] of Object.entries(config.params)) {\n const raw = params[key];\n input[key] = mapping.type === \"int\" ? Number(raw) : raw;\n }\n }\n return { procedure: config.procedure, input };\n };\n}\n\nfunction buildLoaderFns(configs: Record<string, LoaderConfig>): Record<string, LoaderFn> {\n const fns: Record<string, LoaderFn> = {};\n for (const [key, config] of Object.entries(configs)) {\n fns[key] = buildLoaderFn(config);\n }\n return fns;\n}\n\n/** Resolve parent chain for a layout, returning outer-to-inner order */\nfunction resolveLayoutChain(\n layoutId: string,\n layoutEntries: Record<string, LayoutManifestEntry>,\n templates: Record<string, string>,\n): LayoutDef[] {\n const chain: LayoutDef[] = [];\n let currentId: string | undefined = layoutId;\n\n while (currentId) {\n const entry: LayoutManifestEntry | undefined = layoutEntries[currentId];\n if (!entry) break;\n chain.push({\n id: currentId,\n template: templates[currentId],\n loaders: buildLoaderFns(entry.loaders ?? {}),\n });\n currentId = entry.parent;\n }\n\n // Reverse: we walked inner→outer, but want outer→inner\n chain.reverse();\n return chain;\n}\n\n/** Resolve layout chain with lazy template getters (re-read from disk on each access) */\nfunction resolveLayoutChainDev(\n layoutId: string,\n layoutEntries: Record<string, LayoutManifestEntry>,\n distDir: string,\n): LayoutDef[] {\n const chain: LayoutDef[] = [];\n let currentId: string | undefined = layoutId;\n\n while (currentId) {\n const entry: LayoutManifestEntry | undefined = layoutEntries[currentId];\n if (!entry) break;\n const layoutTemplatePath = join(distDir, entry.template);\n const def: LayoutDef = {\n id: currentId,\n template: \"\", // placeholder, overridden by getter\n loaders: buildLoaderFns(entry.loaders ?? {}),\n };\n Object.defineProperty(def, \"template\", {\n get: () => readFileSync(layoutTemplatePath, \"utf-8\"),\n enumerable: true,\n });\n chain.push(def);\n currentId = entry.parent;\n }\n\n chain.reverse();\n return chain;\n}\n\n/** Load the RPC hash map from build output (returns undefined when obfuscation is off) */\nexport function loadRpcHashMap(distDir: string): RpcHashMap | undefined {\n const hashMapPath = join(distDir, \"rpc-hash-map.json\");\n try {\n return JSON.parse(readFileSync(hashMapPath, \"utf-8\")) as RpcHashMap;\n } catch {\n return undefined;\n }\n}\n\nexport function loadBuildOutput(distDir: string): Record<string, PageDef> {\n const manifestPath = join(distDir, \"route-manifest.json\");\n const raw = readFileSync(manifestPath, \"utf-8\");\n const manifest = JSON.parse(raw) as RouteManifest;\n\n // Load layout templates\n const layoutTemplates: Record<string, string> = {};\n const layoutEntries = manifest.layouts ?? {};\n for (const [id, entry] of Object.entries(layoutEntries)) {\n layoutTemplates[id] = readFileSync(join(distDir, entry.template), \"utf-8\");\n }\n\n const pages: Record<string, PageDef> = {};\n for (const [path, entry] of Object.entries(manifest.routes)) {\n const templatePath = join(distDir, entry.template);\n const template = readFileSync(templatePath, \"utf-8\");\n\n const loaders = buildLoaderFns(entry.loaders);\n const layoutChain = entry.layout\n ? resolveLayoutChain(entry.layout, layoutEntries, layoutTemplates)\n : [];\n\n pages[path] = { template, loaders, layoutChain, headMeta: entry.head_meta };\n }\n return pages;\n}\n\n/** Load build output with lazy template getters — templates re-read from disk on each access */\nexport function loadBuildOutputDev(distDir: string): Record<string, PageDef> {\n const manifestPath = join(distDir, \"route-manifest.json\");\n const raw = readFileSync(manifestPath, \"utf-8\");\n const manifest = JSON.parse(raw) as RouteManifest;\n\n const layoutEntries = manifest.layouts ?? {};\n\n const pages: Record<string, PageDef> = {};\n for (const [path, entry] of Object.entries(manifest.routes)) {\n const templatePath = join(distDir, entry.template);\n const loaders = buildLoaderFns(entry.loaders);\n const layoutChain = entry.layout\n ? resolveLayoutChainDev(entry.layout, layoutEntries, distDir)\n : [];\n\n const page: PageDef = {\n template: \"\", // placeholder, overridden by getter\n loaders,\n layoutChain,\n };\n Object.defineProperty(page, \"template\", {\n get: () => readFileSync(templatePath, \"utf-8\"),\n enumerable: true,\n });\n pages[path] = page;\n }\n return pages;\n}\n","/* packages/server/core/typescript/src/subscription.ts */\n\n/**\n * Bridge callback-style event sources to an AsyncGenerator.\n *\n * Usage:\n * const stream = fromCallback<string>(({ emit, end, error }) => {\n * emitter.on(\"data\", emit);\n * emitter.on(\"end\", end);\n * emitter.on(\"error\", error);\n * return () => emitter.removeAllListeners();\n * });\n */\nexport interface CallbackSink<T> {\n emit: (value: T) => void;\n end: () => void;\n error: (err: Error) => void;\n}\n\ntype QueueItem<T> = { type: \"value\"; value: T } | { type: \"end\" } | { type: \"error\"; error: Error };\n\nexport function fromCallback<T>(\n setup: (sink: CallbackSink<T>) => (() => void) | void,\n): AsyncGenerator<T, void, undefined> {\n const queue: QueueItem<T>[] = [];\n let resolve: (() => void) | null = null;\n let done = false;\n function notify() {\n if (resolve) {\n const r = resolve;\n resolve = null;\n r();\n }\n }\n\n const sink: CallbackSink<T> = {\n emit(value) {\n if (done) return;\n queue.push({ type: \"value\", value });\n notify();\n },\n end() {\n if (done) return;\n done = true;\n queue.push({ type: \"end\" });\n notify();\n },\n error(err) {\n if (done) return;\n done = true;\n queue.push({ type: \"error\", error: err });\n notify();\n },\n };\n\n const cleanup = setup(sink);\n\n async function* generate(): AsyncGenerator<T, void, undefined> {\n try {\n while (true) {\n if (queue.length === 0) {\n await new Promise<void>((r) => {\n resolve = r;\n });\n }\n\n while (queue.length > 0) {\n const item = queue.shift()!;\n if (item.type === \"value\") {\n yield item.value;\n } else if (item.type === \"error\") {\n throw item.error;\n } else {\n return;\n }\n }\n }\n } finally {\n done = true;\n if (cleanup) cleanup();\n }\n }\n\n return generate();\n}\n","/* packages/server/core/typescript/src/proxy.ts */\n\nimport { readFile } from \"node:fs/promises\";\nimport { join, extname } from \"node:path\";\nimport type { HttpHandler, HttpBodyResponse } from \"./http.js\";\nimport { MIME_TYPES } from \"./mime.js\";\n\nexport interface DevProxyOptions {\n /** Target URL to forward requests to (e.g. \"http://localhost:5173\") */\n target: string;\n}\n\nexport interface StaticHandlerOptions {\n /** Directory to serve static files from */\n dir: string;\n}\n\n/** Forward non-seam requests to a dev server (e.g. Vite) */\nexport function createDevProxy(opts: DevProxyOptions): HttpHandler {\n const target = opts.target.replace(/\\/$/, \"\");\n\n return async (req) => {\n const url = new URL(req.url, \"http://localhost\");\n const proxyUrl = `${target}${url.pathname}${url.search}`;\n\n try {\n const resp = await fetch(proxyUrl, {\n method: req.method,\n headers: { Accept: \"*/*\" },\n });\n\n const body = await resp.text();\n const headers: Record<string, string> = {};\n resp.headers.forEach((value, key) => {\n headers[key] = value;\n });\n\n return { status: resp.status, headers, body };\n } catch {\n return {\n status: 502,\n headers: { \"Content-Type\": \"text/plain\" },\n body: `Bad Gateway: failed to connect to ${target}`,\n };\n }\n };\n}\n\n/** Serve static files from a directory, with index.html fallback for directories */\nexport function createStaticHandler(opts: StaticHandlerOptions): HttpHandler {\n const dir = opts.dir;\n\n return async (req): Promise<HttpBodyResponse> => {\n const url = new URL(req.url, \"http://localhost\");\n let filePath = url.pathname;\n\n if (filePath.includes(\"..\")) {\n return {\n status: 403,\n headers: { \"Content-Type\": \"text/plain\" },\n body: \"Forbidden\",\n };\n }\n\n // Serve index.html for directory paths\n if (filePath.endsWith(\"/\")) {\n filePath += \"index.html\";\n }\n\n const fullPath = join(dir, filePath);\n try {\n const content = await readFile(fullPath);\n const ext = extname(fullPath);\n const contentType = MIME_TYPES[ext] || \"application/octet-stream\";\n return {\n status: 200,\n headers: { \"Content-Type\": contentType },\n body: content.toString(),\n };\n } catch {\n return {\n status: 404,\n headers: { \"Content-Type\": \"text/plain\" },\n body: \"Not found\",\n };\n }\n };\n}\n","/* packages/server/core/typescript/src/dev/reload-watcher.ts */\n\nimport { watch, type FSWatcher } from \"node:fs\";\nimport { join } from \"node:path\";\n\nexport interface ReloadWatcher {\n close(): void;\n}\n\nexport function watchReloadTrigger(distDir: string, onReload: () => void): ReloadWatcher {\n const triggerPath = join(distDir, \".reload-trigger\");\n let watcher: FSWatcher | null = null;\n try {\n watcher = watch(triggerPath, () => onReload());\n } catch {\n // Trigger file may not exist yet; watch directory until it appears\n const dirWatcher = watch(distDir, (_event, filename) => {\n if (filename === \".reload-trigger\") {\n dirWatcher.close();\n watcher = watch(triggerPath, () => onReload());\n // First creation IS the reload signal -- fire immediately\n onReload();\n }\n });\n return {\n close() {\n dirWatcher.close();\n watcher?.close();\n },\n };\n }\n return {\n close() {\n watcher?.close();\n },\n };\n}\n"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"index.js","names":["primitives"],"sources":["../src/types/schema.ts","../src/types/primitives.ts","../src/types/composites.ts","../src/types/index.ts","../src/manifest/index.ts","../src/errors.ts","../src/validation/index.ts","../src/router/handler.ts","../src/page/handler.ts","../src/page/route-matcher.ts","../src/resolve.ts","../src/router/index.ts","../src/page/index.ts","../src/mime.ts","../src/http.ts","../src/page/build-loader.ts","../src/subscription.ts","../src/proxy.ts","../src/dev/reload-watcher.ts"],"sourcesContent":["/* packages/server/core/typescript/src/types/schema.ts */\n\nimport type { Schema } from \"jtd\";\n\nexport type JTDSchema = Schema;\n\nexport interface SchemaNode<TOutput = unknown> {\n readonly _schema: JTDSchema;\n /** Phantom type marker — never exists at runtime */\n readonly _output: TOutput;\n}\n\nexport interface OptionalSchemaNode<TOutput = unknown> extends SchemaNode<TOutput> {\n readonly _optional: true;\n}\n\nexport type Infer<T extends SchemaNode> = T[\"_output\"];\n\nexport function createSchemaNode<T>(schema: JTDSchema): SchemaNode<T> {\n return { _schema: schema } as SchemaNode<T>;\n}\n\nexport function createOptionalSchemaNode<T>(schema: JTDSchema): OptionalSchemaNode<T> {\n return { _schema: schema, _optional: true } as OptionalSchemaNode<T>;\n}\n","/* packages/server/core/typescript/src/types/primitives.ts */\n\nimport type { SchemaNode } from \"./schema.js\";\nimport { createSchemaNode } from \"./schema.js\";\n\nexport function string(): SchemaNode<string> {\n return createSchemaNode<string>({ type: \"string\" });\n}\n\nexport function boolean(): SchemaNode<boolean> {\n return createSchemaNode<boolean>({ type: \"boolean\" });\n}\n\nexport function int8(): SchemaNode<number> {\n return createSchemaNode<number>({ type: \"int8\" });\n}\n\nexport function int16(): SchemaNode<number> {\n return createSchemaNode<number>({ type: \"int16\" });\n}\n\nexport function int32(): SchemaNode<number> {\n return createSchemaNode<number>({ type: \"int32\" });\n}\n\nexport function uint8(): SchemaNode<number> {\n return createSchemaNode<number>({ type: \"uint8\" });\n}\n\nexport function uint16(): SchemaNode<number> {\n return createSchemaNode<number>({ type: \"uint16\" });\n}\n\nexport function uint32(): SchemaNode<number> {\n return createSchemaNode<number>({ type: \"uint32\" });\n}\n\nexport function float32(): SchemaNode<number> {\n return createSchemaNode<number>({ type: \"float32\" });\n}\n\nexport function float64(): SchemaNode<number> {\n return createSchemaNode<number>({ type: \"float64\" });\n}\n\nexport function timestamp(): SchemaNode<string> {\n return createSchemaNode<string>({ type: \"timestamp\" });\n}\n\nexport function html(): SchemaNode<string> {\n return createSchemaNode<string>({ type: \"string\", metadata: { format: \"html\" } });\n}\n","/* packages/server/core/typescript/src/types/composites.ts */\n\nimport type { SchemaNode, OptionalSchemaNode, Infer, JTDSchema } from \"./schema.js\";\nimport { createSchemaNode, createOptionalSchemaNode } from \"./schema.js\";\n\n// -- Type-level utilities --\n\ntype Simplify<T> = { [K in keyof T]: T[K] } & {};\n\ntype RequiredKeys<T extends Record<string, SchemaNode>> = {\n [K in keyof T]: T[K] extends OptionalSchemaNode ? never : K;\n}[keyof T];\n\ntype OptionalKeys<T extends Record<string, SchemaNode>> = {\n [K in keyof T]: T[K] extends OptionalSchemaNode ? K : never;\n}[keyof T];\n\ntype InferObject<T extends Record<string, SchemaNode>> = Simplify<\n { [K in RequiredKeys<T>]: Infer<T[K]> } & { [K in OptionalKeys<T>]?: Infer<T[K]> }\n>;\n\n// -- Builders --\n\nexport function object<T extends Record<string, SchemaNode>>(\n fields: T,\n): SchemaNode<InferObject<T>> {\n const properties: Record<string, JTDSchema> = {};\n const optionalProperties: Record<string, JTDSchema> = {};\n\n for (const [key, node] of Object.entries(fields)) {\n if (\"_optional\" in node && node._optional === true) {\n optionalProperties[key] = node._schema;\n } else {\n properties[key] = node._schema;\n }\n }\n\n const schema: Record<string, unknown> = {};\n if (Object.keys(properties).length > 0 || Object.keys(optionalProperties).length === 0) {\n schema.properties = properties;\n }\n if (Object.keys(optionalProperties).length > 0) {\n schema.optionalProperties = optionalProperties;\n }\n\n return createSchemaNode<InferObject<T>>(schema as JTDSchema);\n}\n\nexport function optional<T>(node: SchemaNode<T>): OptionalSchemaNode<T> {\n return createOptionalSchemaNode<T>(node._schema);\n}\n\nexport function array<T>(node: SchemaNode<T>): SchemaNode<T[]> {\n return createSchemaNode<T[]>({ elements: node._schema });\n}\n\nexport function nullable<T>(node: SchemaNode<T>): SchemaNode<T | null> {\n return createSchemaNode<T | null>({ ...node._schema, nullable: true } as JTDSchema);\n}\n\nexport function enumType<const T extends readonly string[]>(values: T): SchemaNode<T[number]> {\n return createSchemaNode<T[number]>({ enum: [...values] } as JTDSchema);\n}\n\nexport function values<T>(node: SchemaNode<T>): SchemaNode<Record<string, T>> {\n return createSchemaNode<Record<string, T>>({ values: node._schema });\n}\n\ntype DiscriminatorUnion<TTag extends string, TMapping extends Record<string, SchemaNode>> = {\n [K in keyof TMapping & string]: Simplify<{ [P in TTag]: K } & Infer<TMapping[K]>>;\n}[keyof TMapping & string];\n\nexport function discriminator<\n TTag extends string,\n TMapping extends Record<string, SchemaNode<Record<string, unknown>>>,\n>(tag: TTag, mapping: TMapping): SchemaNode<DiscriminatorUnion<TTag, TMapping>> {\n const jtdMapping: Record<string, JTDSchema> = {};\n for (const [key, node] of Object.entries(mapping)) {\n jtdMapping[key] = node._schema;\n }\n return createSchemaNode<DiscriminatorUnion<TTag, TMapping>>({\n discriminator: tag,\n mapping: jtdMapping,\n } as JTDSchema);\n}\n","/* packages/server/core/typescript/src/types/index.ts */\n\nimport * as primitives from \"./primitives.js\";\nimport {\n object,\n optional,\n array,\n nullable,\n enumType,\n values,\n discriminator,\n} from \"./composites.js\";\n\nexport const t = {\n ...primitives,\n object,\n optional,\n array,\n nullable,\n enum: enumType,\n values,\n discriminator,\n} as const;\n","/* packages/server/core/typescript/src/manifest/index.ts */\n\nimport type { Schema } from \"jtd\";\nimport type { SchemaNode } from \"../types/schema.js\";\n\nexport type ProcedureType = \"query\" | \"subscription\";\n\nexport interface ProcedureEntry {\n type: ProcedureType;\n input: Schema;\n output: Schema;\n}\n\nexport interface ProcedureManifest {\n version: string;\n procedures: Record<string, ProcedureEntry>;\n}\n\nexport function buildManifest(\n definitions: Record<string, { input: SchemaNode; output: SchemaNode; type?: string }>,\n): ProcedureManifest {\n const mapped: ProcedureManifest[\"procedures\"] = {};\n\n for (const [name, def] of Object.entries(definitions)) {\n mapped[name] = {\n type: def.type === \"subscription\" ? \"subscription\" : \"query\",\n input: def.input._schema,\n output: def.output._schema,\n };\n }\n\n return { version: \"0.1.0\", procedures: mapped };\n}\n","/* packages/server/core/typescript/src/errors.ts */\n\nexport type ErrorCode =\n | \"VALIDATION_ERROR\"\n | \"NOT_FOUND\"\n | \"UNAUTHORIZED\"\n | \"FORBIDDEN\"\n | \"RATE_LIMITED\"\n | \"INTERNAL_ERROR\"\n | (string & {});\n\nexport const DEFAULT_STATUS: Record<string, number> = {\n VALIDATION_ERROR: 400,\n UNAUTHORIZED: 401,\n FORBIDDEN: 403,\n NOT_FOUND: 404,\n RATE_LIMITED: 429,\n INTERNAL_ERROR: 500,\n};\n\nexport class SeamError extends Error {\n readonly code: string;\n readonly status: number;\n\n constructor(code: string, message: string, status?: number) {\n super(message);\n this.code = code;\n this.status = status ?? DEFAULT_STATUS[code] ?? 500;\n this.name = \"SeamError\";\n }\n\n toJSON() {\n return {\n error: {\n code: this.code,\n message: this.message,\n },\n };\n }\n}\n","/* packages/server/core/typescript/src/validation/index.ts */\n\nimport { validate } from \"jtd\";\nimport type { Schema, ValidationError as JTDValidationError } from \"jtd\";\n\nexport interface ValidationResult {\n valid: boolean;\n errors: JTDValidationError[];\n}\n\nexport function validateInput(schema: Schema, data: unknown): ValidationResult {\n const errors = validate(schema, data, { maxDepth: 32, maxErrors: 10 });\n return {\n valid: errors.length === 0,\n errors,\n };\n}\n\nexport function formatValidationErrors(errors: JTDValidationError[]): string {\n return errors\n .map((e) => {\n const path = e.instancePath.length > 0 ? e.instancePath.join(\"/\") : \"(root)\";\n const schema = e.schemaPath.join(\"/\");\n return `${path} (schema: ${schema})`;\n })\n .join(\"; \");\n}\n","/* packages/server/core/typescript/src/router/handler.ts */\n\nimport { SeamError } from \"../errors.js\";\nimport type { HandleResult, InternalProcedure, InternalSubscription } from \"../procedure.js\";\nimport { validateInput, formatValidationErrors } from \"../validation/index.js\";\n\nexport type { HandleResult, InternalProcedure } from \"../procedure.js\";\n\nexport async function handleRequest(\n procedures: Map<string, InternalProcedure>,\n procedureName: string,\n rawBody: unknown,\n validateOutput?: boolean,\n): Promise<HandleResult> {\n const procedure = procedures.get(procedureName);\n if (!procedure) {\n return {\n status: 404,\n body: new SeamError(\"NOT_FOUND\", `Procedure '${procedureName}' not found`).toJSON(),\n };\n }\n\n const validation = validateInput(procedure.inputSchema, rawBody);\n if (!validation.valid) {\n const details = formatValidationErrors(validation.errors);\n return {\n status: 400,\n body: new SeamError(\"VALIDATION_ERROR\", `Input validation failed: ${details}`).toJSON(),\n };\n }\n\n try {\n const result = await procedure.handler({ input: rawBody });\n\n if (validateOutput) {\n const outValidation = validateInput(procedure.outputSchema, result);\n if (!outValidation.valid) {\n const details = formatValidationErrors(outValidation.errors);\n return {\n status: 500,\n body: new SeamError(\"INTERNAL_ERROR\", `Output validation failed: ${details}`).toJSON(),\n };\n }\n }\n\n return { status: 200, body: result };\n } catch (error) {\n if (error instanceof SeamError) {\n return { status: error.status, body: error.toJSON() };\n }\n const message = error instanceof Error ? error.message : \"Unknown error\";\n return {\n status: 500,\n body: new SeamError(\"INTERNAL_ERROR\", message).toJSON(),\n };\n }\n}\n\nexport interface BatchCall {\n procedure: string;\n input: unknown;\n}\n\nexport type BatchResultItem =\n | { ok: true; data: unknown }\n | { ok: false; error: { code: string; message: string } };\n\nexport async function handleBatchRequest(\n procedures: Map<string, InternalProcedure>,\n calls: BatchCall[],\n validateOutput?: boolean,\n): Promise<{ results: BatchResultItem[] }> {\n const results = await Promise.all(\n calls.map(async (call) => {\n const result = await handleRequest(procedures, call.procedure, call.input, validateOutput);\n if (result.status === 200) {\n return { ok: true as const, data: result.body };\n }\n const envelope = result.body as { error: { code: string; message: string } };\n return { ok: false as const, error: envelope.error };\n }),\n );\n return { results };\n}\n\nexport async function* handleSubscription(\n subscriptions: Map<string, InternalSubscription>,\n name: string,\n rawInput: unknown,\n validateOutput?: boolean,\n): AsyncIterable<unknown> {\n const sub = subscriptions.get(name);\n if (!sub) {\n throw new SeamError(\"NOT_FOUND\", `Subscription '${name}' not found`);\n }\n\n const validation = validateInput(sub.inputSchema, rawInput);\n if (!validation.valid) {\n const details = formatValidationErrors(validation.errors);\n throw new SeamError(\"VALIDATION_ERROR\", `Input validation failed: ${details}`);\n }\n\n for await (const value of sub.handler({ input: rawInput })) {\n if (validateOutput) {\n const outValidation = validateInput(sub.outputSchema, value);\n if (!outValidation.valid) {\n const details = formatValidationErrors(outValidation.errors);\n throw new SeamError(\"INTERNAL_ERROR\", `Output validation failed: ${details}`);\n }\n }\n yield value;\n }\n}\n","/* packages/server/core/typescript/src/page/handler.ts */\n\nimport { readFileSync, existsSync } from \"node:fs\";\nimport { join } from \"node:path\";\nimport { renderPage, escapeHtml } from \"@canmi/seam-engine\";\nimport { SeamError } from \"../errors.js\";\nimport type { InternalProcedure } from \"../procedure.js\";\nimport type { PageDef, LoaderFn, I18nConfig } from \"./index.js\";\n\nexport interface PageTiming {\n /** Procedure execution time in milliseconds */\n dataFetch: number;\n /** Template injection time in milliseconds */\n inject: number;\n}\n\nexport interface HandlePageResult {\n status: number;\n html: string;\n timing?: PageTiming;\n}\n\nexport interface I18nOpts {\n locale: string;\n config: I18nConfig;\n /** Route pattern for hash-based message lookup */\n routePattern: string;\n}\n\n/** Execute loaders, returning keyed results */\nasync function executeLoaders(\n loaders: Record<string, LoaderFn>,\n params: Record<string, string>,\n procedures: Map<string, InternalProcedure>,\n): Promise<Record<string, unknown>> {\n const entries = Object.entries(loaders);\n const results = await Promise.all(\n entries.map(async ([key, loader]) => {\n const { procedure, input } = loader(params);\n const proc = procedures.get(procedure);\n if (!proc) throw new SeamError(\"INTERNAL_ERROR\", `Procedure '${procedure}' not found`);\n // Skip JTD validation -- loader input is trusted server-side code\n const result = await proc.handler({ input });\n return [key, result] as const;\n }),\n );\n return Object.fromEntries(results);\n}\n\n/** Select the template for a given locale, falling back to the default template */\nfunction selectTemplate(\n defaultTemplate: string,\n localeTemplates: Record<string, string> | undefined,\n locale: string | undefined,\n): string {\n if (locale && localeTemplates) {\n return localeTemplates[locale] ?? defaultTemplate;\n }\n return defaultTemplate;\n}\n\n/** Look up pre-resolved messages for a route + locale. Zero merge, zero filter. */\nfunction lookupMessages(\n config: I18nConfig,\n routePattern: string,\n locale: string,\n): Record<string, string> {\n const routeHash = config.routeHashes[routePattern];\n if (!routeHash) return {};\n\n if (config.mode === \"paged\" && config.distDir) {\n const filePath = join(config.distDir, \"i18n\", routeHash, `${locale}.json`);\n if (existsSync(filePath)) {\n return JSON.parse(readFileSync(filePath, \"utf-8\")) as Record<string, string>;\n }\n return {};\n }\n\n return config.messages[locale]?.[routeHash] ?? {};\n}\n\nexport async function handlePageRequest(\n page: PageDef,\n params: Record<string, string>,\n procedures: Map<string, InternalProcedure>,\n i18nOpts?: I18nOpts,\n): Promise<HandlePageResult> {\n try {\n const t0 = performance.now();\n const layoutChain = page.layoutChain ?? [];\n const locale = i18nOpts?.locale;\n\n // Execute all loaders (layout chain + page) in parallel\n const loaderResults = await Promise.all([\n ...layoutChain.map((layout) => executeLoaders(layout.loaders, params, procedures)),\n executeLoaders(page.loaders, params, procedures),\n ]);\n\n const t1 = performance.now();\n\n // Merge all loader data into a single object\n const allData: Record<string, unknown> = {};\n for (const result of loaderResults) {\n Object.assign(allData, result);\n }\n\n // Compose template: nest page inside layouts via outlet substitution\n const pageTemplate = selectTemplate(page.template, page.localeTemplates, locale);\n let composedTemplate = pageTemplate;\n for (let i = layoutChain.length - 1; i >= 0; i--) {\n const layout = layoutChain[i];\n const layoutTemplate = selectTemplate(layout.template, layout.localeTemplates, locale);\n composedTemplate = layoutTemplate.replace(\"<!--seam:outlet-->\", composedTemplate);\n }\n\n // Build PageConfig for engine\n const config = {\n layout_chain: layoutChain.map((l) => ({\n id: l.id,\n loader_keys: Object.keys(l.loaders),\n })),\n data_id: page.dataId ?? \"__SEAM_DATA__\",\n head_meta: page.headMeta,\n };\n\n // Build I18nOpts for engine (hash-based lookup — zero merge, zero filter)\n let i18nOptsJson: string | undefined;\n if (i18nOpts) {\n const { config: i18nConfig, routePattern } = i18nOpts;\n const messages = lookupMessages(i18nConfig, routePattern, i18nOpts.locale);\n const routeHash = i18nConfig.routeHashes[routePattern];\n const i18nData: Record<string, unknown> = {\n locale: i18nOpts.locale,\n default_locale: i18nConfig.default,\n messages,\n };\n // Inject content hash and router table when cache is enabled\n if (i18nConfig.cache && routeHash) {\n i18nData.hash = i18nConfig.contentHashes[routeHash]?.[i18nOpts.locale];\n i18nData.router = i18nConfig.contentHashes;\n }\n i18nOptsJson = JSON.stringify(i18nData);\n }\n\n // Single WASM call: inject slots, compose data script, apply locale/meta\n const html = renderPage(\n composedTemplate,\n JSON.stringify(allData),\n JSON.stringify(config),\n i18nOptsJson,\n );\n\n const t2 = performance.now();\n\n return {\n status: 200,\n html,\n timing: { dataFetch: t1 - t0, inject: t2 - t1 },\n };\n } catch (error) {\n const message = error instanceof Error ? error.message : \"Unknown error\";\n return {\n status: 500,\n html: `<!DOCTYPE html><html><body><h1>500 Internal Server Error</h1><p>${escapeHtml(message)}</p></body></html>`,\n };\n }\n}\n","/* packages/server/core/typescript/src/page/route-matcher.ts */\n\ninterface CompiledRoute {\n segments: RouteSegment[];\n}\n\ntype RouteSegment = { kind: \"static\"; value: string } | { kind: \"param\"; name: string };\n\nfunction compileRoute(pattern: string): CompiledRoute {\n const segments: RouteSegment[] = pattern\n .split(\"/\")\n .filter(Boolean)\n .map((seg) =>\n seg.startsWith(\":\") ? { kind: \"param\", name: seg.slice(1) } : { kind: \"static\", value: seg },\n );\n return { segments };\n}\n\nfunction matchRoute(segments: RouteSegment[], pathParts: string[]): Record<string, string> | null {\n if (segments.length !== pathParts.length) return null;\n const params: Record<string, string> = {};\n for (let i = 0; i < segments.length; i++) {\n const seg = segments[i];\n if (seg.kind === \"static\") {\n if (seg.value !== pathParts[i]) return null;\n } else {\n params[seg.name] = pathParts[i];\n }\n }\n return params;\n}\n\nexport class RouteMatcher<T> {\n private routes: { pattern: string; compiled: CompiledRoute; value: T }[] = [];\n\n add(pattern: string, value: T): void {\n this.routes.push({ pattern, compiled: compileRoute(pattern), value });\n }\n\n match(path: string): { value: T; params: Record<string, string>; pattern: string } | null {\n const parts = path.split(\"/\").filter(Boolean);\n for (const route of this.routes) {\n const params = matchRoute(route.compiled.segments, parts);\n if (params) return { value: route.value, params, pattern: route.pattern };\n }\n return null;\n }\n}\n","/* packages/server/core/typescript/src/resolve.ts */\n\n// -- New strategy-based API --\n\nexport interface ResolveStrategy {\n readonly kind: string;\n resolve(data: ResolveData): string | null;\n}\n\nexport interface ResolveData {\n readonly url: string;\n readonly pathLocale: string | null;\n readonly cookie: string | undefined;\n readonly acceptLanguage: string | undefined;\n readonly locales: string[];\n readonly defaultLocale: string;\n}\n\n/** URL prefix strategy: trusts pathLocale if it is a known locale */\nexport function fromUrlPrefix(): ResolveStrategy {\n return {\n kind: \"url_prefix\",\n resolve(data) {\n if (data.pathLocale && data.locales.includes(data.pathLocale)) {\n return data.pathLocale;\n }\n return null;\n },\n };\n}\n\n/** Cookie strategy: reads a named cookie and validates against known locales */\nexport function fromCookie(name = \"seam-locale\"): ResolveStrategy {\n return {\n kind: \"cookie\",\n resolve(data) {\n if (!data.cookie) return null;\n for (const pair of data.cookie.split(\";\")) {\n const [k, v] = pair.trim().split(\"=\");\n if (k === name && v && data.locales.includes(v)) return v;\n }\n return null;\n },\n };\n}\n\n/** Accept-Language strategy: parses header with q-value sorting + prefix match */\nexport function fromAcceptLanguage(): ResolveStrategy {\n return {\n kind: \"accept_language\",\n resolve(data) {\n if (!data.acceptLanguage) return null;\n const entries: { lang: string; q: number }[] = [];\n for (const part of data.acceptLanguage.split(\",\")) {\n const trimmed = part.trim();\n const [lang, ...rest] = trimmed.split(\";\");\n let q = 1;\n for (const r of rest) {\n const match = r.trim().match(/^q=(\\d+(?:\\.\\d+)?)$/);\n if (match) q = parseFloat(match[1]);\n }\n entries.push({ lang: lang.trim(), q });\n }\n entries.sort((a, b) => b.q - a.q);\n const localeSet = new Set(data.locales);\n for (const { lang } of entries) {\n if (localeSet.has(lang)) return lang;\n // Prefix match: zh-CN -> zh\n const prefix = lang.split(\"-\")[0];\n if (prefix !== lang && localeSet.has(prefix)) return prefix;\n }\n return null;\n },\n };\n}\n\n/** URL query strategy: reads a query parameter and validates against known locales */\nexport function fromUrlQuery(param = \"lang\"): ResolveStrategy {\n return {\n kind: \"url_query\",\n resolve(data) {\n if (!data.url) return null;\n try {\n const url = new URL(data.url, \"http://localhost\");\n const value = url.searchParams.get(param);\n if (value && data.locales.includes(value)) return value;\n } catch {\n // Invalid URL — skip\n }\n return null;\n },\n };\n}\n\n/** Run strategies in order; first non-null wins, otherwise defaultLocale */\nexport function resolveChain(strategies: ResolveStrategy[], data: ResolveData): string {\n for (const s of strategies) {\n const result = s.resolve(data);\n if (result !== null) return result;\n }\n return data.defaultLocale;\n}\n\n/** Default strategy chain: url_prefix -> cookie -> accept_language */\nexport function defaultStrategies(): ResolveStrategy[] {\n return [fromUrlPrefix(), fromCookie(), fromAcceptLanguage()];\n}\n","/* packages/server/core/typescript/src/router/index.ts */\n\nimport { existsSync, readFileSync } from \"node:fs\";\nimport { join } from \"node:path\";\nimport type { SchemaNode } from \"../types/schema.js\";\nimport type { ProcedureManifest } from \"../manifest/index.js\";\nimport type { HandleResult, InternalProcedure } from \"./handler.js\";\nimport type { InternalSubscription } from \"../procedure.js\";\nimport type { HandlePageResult } from \"../page/handler.js\";\nimport type { PageDef, I18nConfig } from \"../page/index.js\";\nimport { buildManifest } from \"../manifest/index.js\";\nimport { handleRequest, handleSubscription, handleBatchRequest } from \"./handler.js\";\nimport type { BatchCall, BatchResultItem } from \"./handler.js\";\nimport { handlePageRequest } from \"../page/handler.js\";\nimport { RouteMatcher } from \"../page/route-matcher.js\";\nimport { defaultStrategies, resolveChain } from \"../resolve.js\";\nimport type { ResolveStrategy } from \"../resolve.js\";\n\nexport interface ProcedureDef<TIn = unknown, TOut = unknown> {\n input: SchemaNode<TIn>;\n output: SchemaNode<TOut>;\n handler: (params: { input: TIn }) => TOut | Promise<TOut>;\n}\n\nexport interface SubscriptionDef<TIn = unknown, TOut = unknown> {\n type: \"subscription\";\n input: SchemaNode<TIn>;\n output: SchemaNode<TOut>;\n handler: (params: { input: TIn }) => AsyncIterable<TOut>;\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nexport type DefinitionMap = Record<string, ProcedureDef<any, any> | SubscriptionDef<any, any>>;\n\nfunction isSubscriptionDef(def: ProcedureDef | SubscriptionDef): def is SubscriptionDef {\n return \"type\" in def && def.type === \"subscription\";\n}\n\nexport interface RouterOptions {\n pages?: Record<string, PageDef>;\n i18n?: I18nConfig | null;\n validateOutput?: boolean;\n resolve?: ResolveStrategy[];\n}\n\nexport interface PageRequestHeaders {\n url?: string;\n cookie?: string;\n acceptLanguage?: string;\n}\n\nexport interface Router<T extends DefinitionMap> {\n manifest(): ProcedureManifest;\n handle(procedureName: string, body: unknown): Promise<HandleResult>;\n handleBatch(calls: BatchCall[]): Promise<{ results: BatchResultItem[] }>;\n handleSubscription(name: string, input: unknown): AsyncIterable<unknown>;\n handlePage(path: string, headers?: PageRequestHeaders): Promise<HandlePageResult | null>;\n readonly hasPages: boolean;\n /** Exposed for adapter access to the definitions */\n readonly procedures: T;\n}\n\n/** Build the resolve strategy list from options */\nfunction buildStrategies(opts?: RouterOptions): {\n strategies: ResolveStrategy[];\n hasUrlPrefix: boolean;\n} {\n const strategies = opts?.resolve ?? defaultStrategies();\n return {\n strategies,\n hasUrlPrefix: strategies.some((s) => s.kind === \"url_prefix\"),\n };\n}\n\n/** Register built-in __seam_i18n_query procedure (route-hash-based lookup) */\nfunction registerI18nQuery(procedureMap: Map<string, InternalProcedure>, config: I18nConfig): void {\n procedureMap.set(\"__seam_i18n_query\", {\n inputSchema: {},\n outputSchema: {},\n handler: ({ input }) => {\n const { route, locale } = input as { route: string; locale: string };\n const messages = lookupI18nMessages(config, route, locale);\n const hash = config.contentHashes[route]?.[locale] ?? \"\";\n return { hash, messages };\n },\n });\n}\n\n/** Look up messages by route hash + locale for RPC query */\nfunction lookupI18nMessages(\n config: I18nConfig,\n routeHash: string,\n locale: string,\n): Record<string, string> {\n if (config.mode === \"paged\" && config.distDir) {\n const filePath = join(config.distDir, \"i18n\", routeHash, `${locale}.json`);\n if (existsSync(filePath)) {\n return JSON.parse(readFileSync(filePath, \"utf-8\")) as Record<string, string>;\n }\n return {};\n }\n return config.messages[locale]?.[routeHash] ?? {};\n}\n\nexport function createRouter<T extends DefinitionMap>(\n procedures: T,\n opts?: RouterOptions,\n): Router<T> {\n const procedureMap = new Map<string, InternalProcedure>();\n const subscriptionMap = new Map<string, InternalSubscription>();\n\n for (const [name, def] of Object.entries(procedures)) {\n if (isSubscriptionDef(def)) {\n subscriptionMap.set(name, {\n inputSchema: def.input._schema,\n outputSchema: def.output._schema,\n handler: def.handler as InternalSubscription[\"handler\"],\n });\n } else {\n procedureMap.set(name, {\n inputSchema: def.input._schema,\n outputSchema: def.output._schema,\n handler: def.handler as InternalProcedure[\"handler\"],\n });\n }\n }\n\n const shouldValidateOutput =\n opts?.validateOutput ??\n (typeof process !== \"undefined\" && process.env.NODE_ENV !== \"production\");\n\n const pageMatcher = new RouteMatcher<PageDef>();\n const pages = opts?.pages;\n if (pages) {\n for (const [pattern, page] of Object.entries(pages)) {\n pageMatcher.add(pattern, page);\n }\n }\n\n const i18nConfig = opts?.i18n ?? null;\n const { strategies, hasUrlPrefix } = buildStrategies(opts);\n if (i18nConfig) registerI18nQuery(procedureMap, i18nConfig);\n\n return {\n procedures,\n hasPages: !!pages && Object.keys(pages).length > 0,\n manifest() {\n return buildManifest(procedures);\n },\n handle(procedureName, body) {\n return handleRequest(procedureMap, procedureName, body, shouldValidateOutput);\n },\n handleBatch(calls) {\n return handleBatchRequest(procedureMap, calls, shouldValidateOutput);\n },\n handleSubscription(name, input) {\n return handleSubscription(subscriptionMap, name, input, shouldValidateOutput);\n },\n async handlePage(path, headers) {\n let pathLocale: string | null = null;\n let routePath = path;\n\n if (hasUrlPrefix && i18nConfig) {\n const segments = path.split(\"/\").filter(Boolean);\n const localeSet = new Set(i18nConfig.locales);\n if (segments.length > 0 && localeSet.has(segments[0])) {\n pathLocale = segments[0];\n routePath = \"/\" + segments.slice(1).join(\"/\") || \"/\";\n }\n }\n\n let locale: string | undefined;\n if (i18nConfig) {\n locale = resolveChain(strategies, {\n url: headers?.url ?? \"\",\n pathLocale,\n cookie: headers?.cookie,\n acceptLanguage: headers?.acceptLanguage,\n locales: i18nConfig.locales,\n defaultLocale: i18nConfig.default,\n });\n }\n\n const match = pageMatcher.match(routePath);\n if (!match) return null;\n\n const i18nOpts =\n locale && i18nConfig\n ? { locale, config: i18nConfig, routePattern: match.pattern }\n : undefined;\n return handlePageRequest(match.value, match.params, procedureMap, i18nOpts);\n },\n };\n}\n","/* packages/server/core/typescript/src/page/index.ts */\n\nexport interface LoaderResult {\n procedure: string;\n input: unknown;\n}\n\nexport type LoaderFn = (params: Record<string, string>) => LoaderResult;\n\nexport interface LayoutDef {\n id: string;\n template: string;\n localeTemplates?: Record<string, string>;\n loaders: Record<string, LoaderFn>;\n i18nKeys?: string[];\n}\n\nexport interface PageDef {\n template: string;\n localeTemplates?: Record<string, string>;\n loaders: Record<string, LoaderFn>;\n layoutChain?: LayoutDef[];\n headMeta?: string;\n dataId?: string;\n i18nKeys?: string[];\n}\n\nexport interface I18nConfig {\n locales: string[];\n default: string;\n mode: \"memory\" | \"paged\";\n cache: boolean;\n routeHashes: Record<string, string>;\n contentHashes: Record<string, Record<string, string>>;\n /** Memory mode: locale → routeHash → messages */\n messages: Record<string, Record<string, Record<string, string>>>;\n /** Paged mode: base directory for on-demand reads */\n distDir?: string;\n}\n\nexport function definePage(config: PageDef): PageDef {\n return { ...config, layoutChain: config.layoutChain ?? [] };\n}\n","/* packages/server/core/typescript/src/mime.ts */\n\nexport const MIME_TYPES: Record<string, string> = {\n \".js\": \"application/javascript\",\n \".mjs\": \"application/javascript\",\n \".css\": \"text/css\",\n \".html\": \"text/html\",\n \".json\": \"application/json\",\n \".svg\": \"image/svg+xml\",\n \".png\": \"image/png\",\n \".jpg\": \"image/jpeg\",\n \".jpeg\": \"image/jpeg\",\n \".gif\": \"image/gif\",\n \".woff\": \"font/woff\",\n \".woff2\": \"font/woff2\",\n \".ttf\": \"font/ttf\",\n \".ico\": \"image/x-icon\",\n \".map\": \"application/json\",\n \".ts\": \"application/javascript\",\n \".tsx\": \"application/javascript\",\n};\n","/* packages/server/core/typescript/src/http.ts */\n\nimport { readFile } from \"node:fs/promises\";\nimport { join, extname } from \"node:path\";\nimport type { Router, DefinitionMap } from \"./router/index.js\";\nimport { SeamError } from \"./errors.js\";\nimport { MIME_TYPES } from \"./mime.js\";\n\nexport interface HttpRequest {\n method: string;\n url: string;\n body: () => Promise<unknown>;\n header?: (name: string) => string | null;\n}\n\nexport interface HttpBodyResponse {\n status: number;\n headers: Record<string, string>;\n body: unknown;\n}\n\nexport interface HttpStreamResponse {\n status: number;\n headers: Record<string, string>;\n stream: AsyncIterable<string>;\n}\n\nexport type HttpResponse = HttpBodyResponse | HttpStreamResponse;\n\nexport type HttpHandler = (req: HttpRequest) => Promise<HttpResponse>;\n\nexport interface RpcHashMap {\n procedures: Record<string, string>;\n batch: string;\n}\n\nexport interface HttpHandlerOptions {\n staticDir?: string;\n fallback?: HttpHandler;\n rpcHashMap?: RpcHashMap;\n}\n\nconst RPC_PREFIX = \"/_seam/rpc/\";\nconst PAGE_PREFIX = \"/_seam/page/\";\nconst STATIC_PREFIX = \"/_seam/static/\";\nconst SUBSCRIBE_PREFIX = \"/_seam/subscribe/\";\nconst MANIFEST_PATH = \"/_seam/manifest.json\";\n\nconst JSON_HEADER = { \"Content-Type\": \"application/json\" };\nconst HTML_HEADER = { \"Content-Type\": \"text/html; charset=utf-8\" };\nconst SSE_HEADER = {\n \"Content-Type\": \"text/event-stream\",\n \"Cache-Control\": \"no-cache\",\n Connection: \"keep-alive\",\n};\nconst IMMUTABLE_CACHE = \"public, max-age=31536000, immutable\";\n\nfunction jsonResponse(status: number, body: unknown): HttpBodyResponse {\n return { status, headers: JSON_HEADER, body };\n}\n\nfunction errorResponse(status: number, code: string, message: string): HttpBodyResponse {\n return jsonResponse(status, new SeamError(code, message).toJSON());\n}\n\nasync function handleStaticAsset(assetPath: string, staticDir: string): Promise<HttpBodyResponse> {\n if (assetPath.includes(\"..\")) {\n return errorResponse(403, \"VALIDATION_ERROR\", \"Forbidden\");\n }\n\n const filePath = join(staticDir, assetPath);\n try {\n const content = await readFile(filePath, \"utf-8\");\n const ext = extname(filePath);\n const contentType = MIME_TYPES[ext] || \"application/octet-stream\";\n return {\n status: 200,\n headers: {\n \"Content-Type\": contentType,\n \"Cache-Control\": IMMUTABLE_CACHE,\n },\n body: content,\n };\n } catch {\n return errorResponse(404, \"NOT_FOUND\", \"Asset not found\");\n }\n}\n\n/** Format a single SSE data event */\nexport function sseDataEvent(data: unknown): string {\n return `event: data\\ndata: ${JSON.stringify(data)}\\n\\n`;\n}\n\n/** Format an SSE error event */\nexport function sseErrorEvent(code: string, message: string): string {\n return `event: error\\ndata: ${JSON.stringify({ code, message })}\\n\\n`;\n}\n\n/** Format an SSE complete event */\nexport function sseCompleteEvent(): string {\n return \"event: complete\\ndata: {}\\n\\n\";\n}\n\nasync function* sseStream<T extends DefinitionMap>(\n router: Router<T>,\n name: string,\n input: unknown,\n): AsyncIterable<string> {\n try {\n for await (const value of router.handleSubscription(name, input)) {\n yield sseDataEvent(value);\n }\n yield sseCompleteEvent();\n } catch (error) {\n if (error instanceof SeamError) {\n yield sseErrorEvent(error.code, error.message);\n } else {\n const message = error instanceof Error ? error.message : \"Unknown error\";\n yield sseErrorEvent(\"INTERNAL_ERROR\", message);\n }\n }\n}\n\nasync function handleBatchHttp<T extends DefinitionMap>(\n req: HttpRequest,\n router: Router<T>,\n hashToName: Map<string, string> | null,\n): Promise<HttpBodyResponse> {\n let body: unknown;\n try {\n body = await req.body();\n } catch {\n return errorResponse(400, \"VALIDATION_ERROR\", \"Invalid JSON body\");\n }\n if (!body || typeof body !== \"object\" || !Array.isArray((body as { calls?: unknown }).calls)) {\n return errorResponse(400, \"VALIDATION_ERROR\", \"Batch request must have a 'calls' array\");\n }\n const calls = (body as { calls: Array<{ procedure?: unknown; input?: unknown }> }).calls.map(\n (c) => ({\n procedure:\n typeof c.procedure === \"string\" ? (hashToName?.get(c.procedure) ?? c.procedure) : \"\",\n input: c.input ?? {},\n }),\n );\n const result = await router.handleBatch(calls);\n return jsonResponse(200, result);\n}\n\nexport function createHttpHandler<T extends DefinitionMap>(\n router: Router<T>,\n opts?: HttpHandlerOptions,\n): HttpHandler {\n // Build reverse lookup (hash -> original name) when obfuscation is active\n const hashToName: Map<string, string> | null = opts?.rpcHashMap\n ? new Map(Object.entries(opts.rpcHashMap.procedures).map(([n, h]) => [h, n]))\n : null;\n // Built-in procedures bypass hash obfuscation (identity mapping)\n if (hashToName) {\n hashToName.set(\"__seam_i18n_query\", \"__seam_i18n_query\");\n }\n const batchHash = opts?.rpcHashMap?.batch ?? null;\n\n return async (req) => {\n const url = new URL(req.url, \"http://localhost\");\n const { pathname } = url;\n\n if (req.method === \"GET\" && pathname === MANIFEST_PATH) {\n if (opts?.rpcHashMap) return errorResponse(403, \"FORBIDDEN\", \"Manifest disabled\");\n return jsonResponse(200, router.manifest());\n }\n\n if (req.method === \"POST\" && pathname.startsWith(RPC_PREFIX)) {\n let name = pathname.slice(RPC_PREFIX.length);\n if (!name) {\n return errorResponse(404, \"NOT_FOUND\", \"Empty procedure name\");\n }\n\n // Batch: match both original \"_batch\" and hashed batch endpoint\n if (name === \"_batch\" || (batchHash && name === batchHash)) {\n return handleBatchHttp(req, router, hashToName);\n }\n\n // Resolve hash -> original name when obfuscation is active\n if (hashToName) {\n const resolved = hashToName.get(name);\n if (!resolved) return errorResponse(404, \"NOT_FOUND\", \"Not found\");\n name = resolved;\n }\n\n let body: unknown;\n try {\n body = await req.body();\n } catch {\n return errorResponse(400, \"VALIDATION_ERROR\", \"Invalid JSON body\");\n }\n\n const result = await router.handle(name, body);\n return jsonResponse(result.status, result.body);\n }\n\n if (req.method === \"GET\" && pathname.startsWith(SUBSCRIBE_PREFIX)) {\n let name = pathname.slice(SUBSCRIBE_PREFIX.length);\n if (!name) {\n return errorResponse(404, \"NOT_FOUND\", \"Empty subscription name\");\n }\n\n // Resolve hash -> original name for subscriptions\n if (hashToName) {\n const resolved = hashToName.get(name);\n if (!resolved) return errorResponse(404, \"NOT_FOUND\", \"Not found\");\n name = resolved;\n }\n\n const rawInput = url.searchParams.get(\"input\");\n let input: unknown;\n try {\n input = rawInput ? JSON.parse(rawInput) : {};\n } catch {\n return errorResponse(400, \"VALIDATION_ERROR\", \"Invalid input query parameter\");\n }\n\n return { status: 200, headers: SSE_HEADER, stream: sseStream(router, name, input) };\n }\n\n // Pages are served under /_seam/page/* prefix only.\n // Root-path serving is the application's responsibility — see the\n // github-dashboard ts-hono example for the fallback pattern.\n if (req.method === \"GET\" && pathname.startsWith(PAGE_PREFIX) && router.hasPages) {\n const pagePath = \"/\" + pathname.slice(PAGE_PREFIX.length);\n const headers = req.header\n ? {\n url: req.url,\n cookie: req.header(\"cookie\") ?? undefined,\n acceptLanguage: req.header(\"accept-language\") ?? undefined,\n }\n : undefined;\n const result = await router.handlePage(pagePath, headers);\n if (result) {\n return { status: result.status, headers: HTML_HEADER, body: result.html };\n }\n }\n\n if (req.method === \"GET\" && pathname.startsWith(STATIC_PREFIX) && opts?.staticDir) {\n const assetPath = pathname.slice(STATIC_PREFIX.length);\n return handleStaticAsset(assetPath, opts.staticDir);\n }\n\n if (opts?.fallback) return opts.fallback(req);\n return errorResponse(404, \"NOT_FOUND\", \"Not found\");\n };\n}\n\nexport function serialize(body: unknown): string {\n return typeof body === \"string\" ? body : JSON.stringify(body);\n}\n\n/** Consume an async stream chunk-by-chunk; return false from write to stop early. */\nexport async function drainStream(\n stream: AsyncIterable<string>,\n write: (chunk: string) => boolean | void,\n): Promise<void> {\n try {\n for await (const chunk of stream) {\n if (write(chunk) === false) break;\n }\n } catch {\n // Client disconnected\n }\n}\n\n/** Convert an HttpResponse to a Web API Response (for adapters using fetch-compatible runtimes) */\nexport function toWebResponse(result: HttpResponse): Response {\n if (\"stream\" in result) {\n const stream = result.stream;\n const encoder = new TextEncoder();\n const readable = new ReadableStream({\n async start(controller) {\n await drainStream(stream, (chunk) => {\n controller.enqueue(encoder.encode(chunk));\n });\n controller.close();\n },\n });\n return new Response(readable, { status: result.status, headers: result.headers });\n }\n return new Response(serialize(result.body), { status: result.status, headers: result.headers });\n}\n","/* packages/server/core/typescript/src/page/build-loader.ts */\n\nimport { existsSync, readFileSync } from \"node:fs\";\nimport { join } from \"node:path\";\nimport type { PageDef, LayoutDef, LoaderFn, LoaderResult, I18nConfig } from \"./index.js\";\nimport type { RpcHashMap } from \"../http.js\";\n\ninterface RouteManifest {\n layouts?: Record<string, LayoutManifestEntry>;\n routes: Record<string, RouteManifestEntry>;\n data_id?: string;\n i18n?: {\n locales: string[];\n default: string;\n mode?: string;\n cache?: boolean;\n route_hashes?: Record<string, string>;\n content_hashes?: Record<string, Record<string, string>>;\n };\n}\n\ninterface LayoutManifestEntry {\n template?: string;\n templates?: Record<string, string>;\n loaders?: Record<string, LoaderConfig>;\n parent?: string;\n i18n_keys?: string[];\n}\n\ninterface RouteManifestEntry {\n template?: string;\n templates?: Record<string, string>;\n layout?: string;\n loaders: Record<string, LoaderConfig>;\n head_meta?: string;\n i18n_keys?: string[];\n}\n\ninterface LoaderConfig {\n procedure: string;\n params?: Record<string, ParamConfig>;\n}\n\ninterface ParamConfig {\n from: \"route\";\n type?: \"string\" | \"int\";\n}\n\nfunction buildLoaderFn(config: LoaderConfig): LoaderFn {\n return (params: Record<string, string>): LoaderResult => {\n const input: Record<string, unknown> = {};\n if (config.params) {\n for (const [key, mapping] of Object.entries(config.params)) {\n const raw = params[key];\n input[key] = mapping.type === \"int\" ? Number(raw) : raw;\n }\n }\n return { procedure: config.procedure, input };\n };\n}\n\nfunction buildLoaderFns(configs: Record<string, LoaderConfig>): Record<string, LoaderFn> {\n const fns: Record<string, LoaderFn> = {};\n for (const [key, config] of Object.entries(configs)) {\n fns[key] = buildLoaderFn(config);\n }\n return fns;\n}\n\nfunction resolveTemplatePath(\n entry: { template?: string; templates?: Record<string, string> },\n defaultLocale: string | undefined,\n): string {\n if (entry.template) return entry.template;\n if (entry.templates) {\n const locale = defaultLocale ?? Object.keys(entry.templates)[0];\n const path = entry.templates[locale];\n if (!path) throw new Error(`No template for locale \"${locale}\"`);\n return path;\n }\n throw new Error(\"Manifest entry has neither 'template' nor 'templates'\");\n}\n\n/** Load all locale templates for a manifest entry, keyed by locale */\nfunction loadLocaleTemplates(\n entry: { templates?: Record<string, string> },\n distDir: string,\n): Record<string, string> | undefined {\n if (!entry.templates) return undefined;\n const result: Record<string, string> = {};\n for (const [locale, relPath] of Object.entries(entry.templates)) {\n result[locale] = readFileSync(join(distDir, relPath), \"utf-8\");\n }\n return result;\n}\n\n/** Resolve parent chain for a layout, returning outer-to-inner order */\nfunction resolveLayoutChain(\n layoutId: string,\n layoutEntries: Record<string, LayoutManifestEntry>,\n templates: Record<string, string>,\n localeTemplatesMap: Record<string, Record<string, string>>,\n): LayoutDef[] {\n const chain: LayoutDef[] = [];\n let currentId: string | undefined = layoutId;\n\n while (currentId) {\n const entry: LayoutManifestEntry | undefined = layoutEntries[currentId];\n if (!entry) break;\n chain.push({\n id: currentId,\n template: templates[currentId],\n localeTemplates: localeTemplatesMap[currentId],\n loaders: buildLoaderFns(entry.loaders ?? {}),\n });\n currentId = entry.parent;\n }\n\n // Reverse: we walked inner->outer, but want outer->inner\n chain.reverse();\n return chain;\n}\n\n/** Resolve layout chain with lazy template getters (re-read from disk on each access) */\nfunction resolveLayoutChainDev(\n layoutId: string,\n layoutEntries: Record<string, LayoutManifestEntry>,\n distDir: string,\n defaultLocale: string | undefined,\n): LayoutDef[] {\n const chain: LayoutDef[] = [];\n let currentId: string | undefined = layoutId;\n\n while (currentId) {\n const entry: LayoutManifestEntry | undefined = layoutEntries[currentId];\n if (!entry) break;\n const layoutTemplatePath = join(distDir, resolveTemplatePath(entry, defaultLocale));\n const def: LayoutDef = {\n id: currentId,\n template: \"\", // placeholder, overridden by getter\n localeTemplates: entry.templates\n ? makeLocaleTemplateGetters(entry.templates, distDir)\n : undefined,\n loaders: buildLoaderFns(entry.loaders ?? {}),\n };\n Object.defineProperty(def, \"template\", {\n get: () => readFileSync(layoutTemplatePath, \"utf-8\"),\n enumerable: true,\n });\n chain.push(def);\n currentId = entry.parent;\n }\n\n chain.reverse();\n return chain;\n}\n\n/** Create a proxy object that lazily reads locale templates from disk */\nfunction makeLocaleTemplateGetters(\n templates: Record<string, string>,\n distDir: string,\n): Record<string, string> {\n const obj: Record<string, string> = {};\n for (const [locale, relPath] of Object.entries(templates)) {\n const fullPath = join(distDir, relPath);\n Object.defineProperty(obj, locale, {\n get: () => readFileSync(fullPath, \"utf-8\"),\n enumerable: true,\n });\n }\n return obj;\n}\n\n/** Merge i18n_keys from route + layout chain into a single list */\nfunction mergeI18nKeys(\n route: RouteManifestEntry,\n layoutEntries: Record<string, LayoutManifestEntry>,\n): string[] | undefined {\n const keys: string[] = [];\n if (route.layout) {\n let currentId: string | undefined = route.layout;\n while (currentId) {\n const entry: LayoutManifestEntry | undefined = layoutEntries[currentId];\n if (!entry) break;\n if (entry.i18n_keys) keys.push(...entry.i18n_keys);\n currentId = entry.parent;\n }\n }\n if (route.i18n_keys) keys.push(...route.i18n_keys);\n return keys.length > 0 ? keys : undefined;\n}\n\n/** Load the RPC hash map from build output (returns undefined when obfuscation is off) */\nexport function loadRpcHashMap(distDir: string): RpcHashMap | undefined {\n const hashMapPath = join(distDir, \"rpc-hash-map.json\");\n try {\n return JSON.parse(readFileSync(hashMapPath, \"utf-8\")) as RpcHashMap;\n } catch {\n return undefined;\n }\n}\n\n/** Load i18n config and messages from build output */\nexport function loadI18nMessages(distDir: string): I18nConfig | null {\n const manifestPath = join(distDir, \"route-manifest.json\");\n try {\n const manifest = JSON.parse(readFileSync(manifestPath, \"utf-8\")) as RouteManifest;\n if (!manifest.i18n) return null;\n\n const mode = (manifest.i18n.mode ?? \"memory\") as \"memory\" | \"paged\";\n const cache = manifest.i18n.cache ?? false;\n const routeHashes = manifest.i18n.route_hashes ?? {};\n const contentHashes = manifest.i18n.content_hashes ?? {};\n\n // Memory mode: preload all route messages per locale\n // Paged mode: store distDir for on-demand reads\n const messages: Record<string, Record<string, Record<string, string>>> = {};\n if (mode === \"memory\") {\n const i18nDir = join(distDir, \"i18n\");\n for (const locale of manifest.i18n.locales) {\n const localePath = join(i18nDir, `${locale}.json`);\n if (existsSync(localePath)) {\n messages[locale] = JSON.parse(readFileSync(localePath, \"utf-8\")) as Record<\n string,\n Record<string, string>\n >;\n } else {\n messages[locale] = {};\n }\n }\n }\n\n return {\n locales: manifest.i18n.locales,\n default: manifest.i18n.default,\n mode,\n cache,\n routeHashes,\n contentHashes,\n messages,\n distDir: mode === \"paged\" ? distDir : undefined,\n };\n } catch {\n return null;\n }\n}\n\nexport function loadBuildOutput(distDir: string): Record<string, PageDef> {\n const manifestPath = join(distDir, \"route-manifest.json\");\n const raw = readFileSync(manifestPath, \"utf-8\");\n const manifest = JSON.parse(raw) as RouteManifest;\n const defaultLocale = manifest.i18n?.default;\n\n // Load layout templates (default + all locales)\n const layoutTemplates: Record<string, string> = {};\n const layoutLocaleTemplates: Record<string, Record<string, string>> = {};\n const layoutEntries = manifest.layouts ?? {};\n for (const [id, entry] of Object.entries(layoutEntries)) {\n layoutTemplates[id] = readFileSync(\n join(distDir, resolveTemplatePath(entry, defaultLocale)),\n \"utf-8\",\n );\n const lt = loadLocaleTemplates(entry, distDir);\n if (lt) layoutLocaleTemplates[id] = lt;\n }\n\n const pages: Record<string, PageDef> = {};\n for (const [path, entry] of Object.entries(manifest.routes)) {\n const templatePath = join(distDir, resolveTemplatePath(entry, defaultLocale));\n const template = readFileSync(templatePath, \"utf-8\");\n\n const loaders = buildLoaderFns(entry.loaders);\n const layoutChain = entry.layout\n ? resolveLayoutChain(entry.layout, layoutEntries, layoutTemplates, layoutLocaleTemplates)\n : [];\n\n // Merge i18n_keys from layout chain + route\n const i18nKeys = mergeI18nKeys(entry, layoutEntries);\n\n pages[path] = {\n template,\n localeTemplates: loadLocaleTemplates(entry, distDir),\n loaders,\n layoutChain,\n headMeta: entry.head_meta,\n dataId: manifest.data_id,\n i18nKeys,\n };\n }\n return pages;\n}\n\n/** Load build output with lazy template getters -- templates re-read from disk on each access */\nexport function loadBuildOutputDev(distDir: string): Record<string, PageDef> {\n const manifestPath = join(distDir, \"route-manifest.json\");\n const raw = readFileSync(manifestPath, \"utf-8\");\n const manifest = JSON.parse(raw) as RouteManifest;\n const defaultLocale = manifest.i18n?.default;\n\n const layoutEntries = manifest.layouts ?? {};\n\n const pages: Record<string, PageDef> = {};\n for (const [path, entry] of Object.entries(manifest.routes)) {\n const templatePath = join(distDir, resolveTemplatePath(entry, defaultLocale));\n const loaders = buildLoaderFns(entry.loaders);\n const layoutChain = entry.layout\n ? resolveLayoutChainDev(entry.layout, layoutEntries, distDir, defaultLocale)\n : [];\n\n const localeTemplates = entry.templates\n ? makeLocaleTemplateGetters(entry.templates, distDir)\n : undefined;\n\n // Merge i18n_keys from layout chain + route\n const i18nKeys = mergeI18nKeys(entry, layoutEntries);\n\n const page: PageDef = {\n template: \"\", // placeholder, overridden by getter\n localeTemplates,\n loaders,\n layoutChain,\n dataId: manifest.data_id,\n i18nKeys,\n };\n Object.defineProperty(page, \"template\", {\n get: () => readFileSync(templatePath, \"utf-8\"),\n enumerable: true,\n });\n pages[path] = page;\n }\n return pages;\n}\n","/* packages/server/core/typescript/src/subscription.ts */\n\n/**\n * Bridge callback-style event sources to an AsyncGenerator.\n *\n * Usage:\n * const stream = fromCallback<string>(({ emit, end, error }) => {\n * emitter.on(\"data\", emit);\n * emitter.on(\"end\", end);\n * emitter.on(\"error\", error);\n * return () => emitter.removeAllListeners();\n * });\n */\nexport interface CallbackSink<T> {\n emit: (value: T) => void;\n end: () => void;\n error: (err: Error) => void;\n}\n\ntype QueueItem<T> = { type: \"value\"; value: T } | { type: \"end\" } | { type: \"error\"; error: Error };\n\nexport function fromCallback<T>(\n setup: (sink: CallbackSink<T>) => (() => void) | void,\n): AsyncGenerator<T, void, undefined> {\n const queue: QueueItem<T>[] = [];\n let resolve: (() => void) | null = null;\n let done = false;\n function notify() {\n if (resolve) {\n const r = resolve;\n resolve = null;\n r();\n }\n }\n\n const sink: CallbackSink<T> = {\n emit(value) {\n if (done) return;\n queue.push({ type: \"value\", value });\n notify();\n },\n end() {\n if (done) return;\n done = true;\n queue.push({ type: \"end\" });\n notify();\n },\n error(err) {\n if (done) return;\n done = true;\n queue.push({ type: \"error\", error: err });\n notify();\n },\n };\n\n const cleanup = setup(sink);\n\n async function* generate(): AsyncGenerator<T, void, undefined> {\n try {\n while (true) {\n if (queue.length === 0) {\n await new Promise<void>((r) => {\n resolve = r;\n });\n }\n\n while (queue.length > 0) {\n const item = queue.shift()!;\n if (item.type === \"value\") {\n yield item.value;\n } else if (item.type === \"error\") {\n throw item.error;\n } else {\n return;\n }\n }\n }\n } finally {\n done = true;\n if (cleanup) cleanup();\n }\n }\n\n return generate();\n}\n","/* packages/server/core/typescript/src/proxy.ts */\n\nimport { readFile } from \"node:fs/promises\";\nimport { join, extname } from \"node:path\";\nimport type { HttpHandler, HttpBodyResponse } from \"./http.js\";\nimport { MIME_TYPES } from \"./mime.js\";\n\nexport interface DevProxyOptions {\n /** Target URL to forward requests to (e.g. \"http://localhost:5173\") */\n target: string;\n}\n\nexport interface StaticHandlerOptions {\n /** Directory to serve static files from */\n dir: string;\n}\n\n/** Forward non-seam requests to a dev server (e.g. Vite) */\nexport function createDevProxy(opts: DevProxyOptions): HttpHandler {\n const target = opts.target.replace(/\\/$/, \"\");\n\n return async (req) => {\n const url = new URL(req.url, \"http://localhost\");\n const proxyUrl = `${target}${url.pathname}${url.search}`;\n\n try {\n const resp = await fetch(proxyUrl, {\n method: req.method,\n headers: { Accept: \"*/*\" },\n });\n\n const body = await resp.text();\n const headers: Record<string, string> = {};\n resp.headers.forEach((value, key) => {\n headers[key] = value;\n });\n\n return { status: resp.status, headers, body };\n } catch {\n return {\n status: 502,\n headers: { \"Content-Type\": \"text/plain\" },\n body: `Bad Gateway: failed to connect to ${target}`,\n };\n }\n };\n}\n\n/** Serve static files from a directory, with index.html fallback for directories */\nexport function createStaticHandler(opts: StaticHandlerOptions): HttpHandler {\n const dir = opts.dir;\n\n return async (req): Promise<HttpBodyResponse> => {\n const url = new URL(req.url, \"http://localhost\");\n let filePath = url.pathname;\n\n if (filePath.includes(\"..\")) {\n return {\n status: 403,\n headers: { \"Content-Type\": \"text/plain\" },\n body: \"Forbidden\",\n };\n }\n\n // Serve index.html for directory paths\n if (filePath.endsWith(\"/\")) {\n filePath += \"index.html\";\n }\n\n const fullPath = join(dir, filePath);\n try {\n const content = await readFile(fullPath);\n const ext = extname(fullPath);\n const contentType = MIME_TYPES[ext] || \"application/octet-stream\";\n return {\n status: 200,\n headers: { \"Content-Type\": contentType },\n body: content.toString(),\n };\n } catch {\n return {\n status: 404,\n headers: { \"Content-Type\": \"text/plain\" },\n body: \"Not found\",\n };\n }\n };\n}\n","/* packages/server/core/typescript/src/dev/reload-watcher.ts */\n\nimport { watch, type FSWatcher } from \"node:fs\";\nimport { join } from \"node:path\";\n\nexport interface ReloadWatcher {\n close(): void;\n}\n\nexport function watchReloadTrigger(distDir: string, onReload: () => void): ReloadWatcher {\n const triggerPath = join(distDir, \".reload-trigger\");\n let watcher: FSWatcher | null = null;\n try {\n watcher = watch(triggerPath, () => onReload());\n } catch {\n // Trigger file may not exist yet; watch directory until it appears\n const dirWatcher = watch(distDir, (_event, filename) => {\n if (filename === \".reload-trigger\") {\n dirWatcher.close();\n watcher = watch(triggerPath, () => onReload());\n // First creation IS the reload signal -- fire immediately\n onReload();\n }\n });\n return {\n close() {\n dirWatcher.close();\n watcher?.close();\n },\n };\n }\n return {\n close() {\n watcher?.close();\n },\n };\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;AAkBA,SAAgB,iBAAoB,QAAkC;AACpE,QAAO,EAAE,SAAS,QAAQ;;AAG5B,SAAgB,yBAA4B,QAA0C;AACpF,QAAO;EAAE,SAAS;EAAQ,WAAW;EAAM;;;;;;;;;;;;;;;;;;;AClB7C,SAAgB,SAA6B;AAC3C,QAAO,iBAAyB,EAAE,MAAM,UAAU,CAAC;;AAGrD,SAAgB,UAA+B;AAC7C,QAAO,iBAA0B,EAAE,MAAM,WAAW,CAAC;;AAGvD,SAAgB,OAA2B;AACzC,QAAO,iBAAyB,EAAE,MAAM,QAAQ,CAAC;;AAGnD,SAAgB,QAA4B;AAC1C,QAAO,iBAAyB,EAAE,MAAM,SAAS,CAAC;;AAGpD,SAAgB,QAA4B;AAC1C,QAAO,iBAAyB,EAAE,MAAM,SAAS,CAAC;;AAGpD,SAAgB,QAA4B;AAC1C,QAAO,iBAAyB,EAAE,MAAM,SAAS,CAAC;;AAGpD,SAAgB,SAA6B;AAC3C,QAAO,iBAAyB,EAAE,MAAM,UAAU,CAAC;;AAGrD,SAAgB,SAA6B;AAC3C,QAAO,iBAAyB,EAAE,MAAM,UAAU,CAAC;;AAGrD,SAAgB,UAA8B;AAC5C,QAAO,iBAAyB,EAAE,MAAM,WAAW,CAAC;;AAGtD,SAAgB,UAA8B;AAC5C,QAAO,iBAAyB,EAAE,MAAM,WAAW,CAAC;;AAGtD,SAAgB,YAAgC;AAC9C,QAAO,iBAAyB,EAAE,MAAM,aAAa,CAAC;;AAGxD,SAAgB,OAA2B;AACzC,QAAO,iBAAyB;EAAE,MAAM;EAAU,UAAU,EAAE,QAAQ,QAAQ;EAAE,CAAC;;;;;AC3BnF,SAAgB,OACd,QAC4B;CAC5B,MAAM,aAAwC,EAAE;CAChD,MAAM,qBAAgD,EAAE;AAExD,MAAK,MAAM,CAAC,KAAK,SAAS,OAAO,QAAQ,OAAO,CAC9C,KAAI,eAAe,QAAQ,KAAK,cAAc,KAC5C,oBAAmB,OAAO,KAAK;KAE/B,YAAW,OAAO,KAAK;CAI3B,MAAM,SAAkC,EAAE;AAC1C,KAAI,OAAO,KAAK,WAAW,CAAC,SAAS,KAAK,OAAO,KAAK,mBAAmB,CAAC,WAAW,EACnF,QAAO,aAAa;AAEtB,KAAI,OAAO,KAAK,mBAAmB,CAAC,SAAS,EAC3C,QAAO,qBAAqB;AAG9B,QAAO,iBAAiC,OAAoB;;AAG9D,SAAgB,SAAY,MAA4C;AACtE,QAAO,yBAA4B,KAAK,QAAQ;;AAGlD,SAAgB,MAAS,MAAsC;AAC7D,QAAO,iBAAsB,EAAE,UAAU,KAAK,SAAS,CAAC;;AAG1D,SAAgB,SAAY,MAA2C;AACrE,QAAO,iBAA2B;EAAE,GAAG,KAAK;EAAS,UAAU;EAAM,CAAc;;AAGrF,SAAgB,SAA4C,QAAkC;AAC5F,QAAO,iBAA4B,EAAE,MAAM,CAAC,GAAG,OAAO,EAAE,CAAc;;AAGxE,SAAgB,OAAU,MAAoD;AAC5E,QAAO,iBAAoC,EAAE,QAAQ,KAAK,SAAS,CAAC;;AAOtE,SAAgB,cAGd,KAAW,SAAmE;CAC9E,MAAM,aAAwC,EAAE;AAChD,MAAK,MAAM,CAAC,KAAK,SAAS,OAAO,QAAQ,QAAQ,CAC/C,YAAW,OAAO,KAAK;AAEzB,QAAO,iBAAqD;EAC1D,eAAe;EACf,SAAS;EACV,CAAc;;;;;ACtEjB,MAAa,IAAI;CACf,GAAGA;CACH;CACA;CACA;CACA;CACA,MAAM;CACN;CACA;CACD;;;;ACJD,SAAgB,cACd,aACmB;CACnB,MAAM,SAA0C,EAAE;AAElD,MAAK,MAAM,CAAC,MAAM,QAAQ,OAAO,QAAQ,YAAY,CACnD,QAAO,QAAQ;EACb,MAAM,IAAI,SAAS,iBAAiB,iBAAiB;EACrD,OAAO,IAAI,MAAM;EACjB,QAAQ,IAAI,OAAO;EACpB;AAGH,QAAO;EAAE,SAAS;EAAS,YAAY;EAAQ;;;;;ACpBjD,MAAa,iBAAyC;CACpD,kBAAkB;CAClB,cAAc;CACd,WAAW;CACX,WAAW;CACX,cAAc;CACd,gBAAgB;CACjB;AAED,IAAa,YAAb,cAA+B,MAAM;CACnC,AAAS;CACT,AAAS;CAET,YAAY,MAAc,SAAiB,QAAiB;AAC1D,QAAM,QAAQ;AACd,OAAK,OAAO;AACZ,OAAK,SAAS,UAAU,eAAe,SAAS;AAChD,OAAK,OAAO;;CAGd,SAAS;AACP,SAAO,EACL,OAAO;GACL,MAAM,KAAK;GACX,SAAS,KAAK;GACf,EACF;;;;;;AC3BL,SAAgB,cAAc,QAAgB,MAAiC;CAC7E,MAAM,SAAS,SAAS,QAAQ,MAAM;EAAE,UAAU;EAAI,WAAW;EAAI,CAAC;AACtE,QAAO;EACL,OAAO,OAAO,WAAW;EACzB;EACD;;AAGH,SAAgB,uBAAuB,QAAsC;AAC3E,QAAO,OACJ,KAAK,MAAM;AAGV,SAAO,GAFM,EAAE,aAAa,SAAS,IAAI,EAAE,aAAa,KAAK,IAAI,GAAG,SAErD,YADA,EAAE,WAAW,KAAK,IAAI,CACH;GAClC,CACD,KAAK,KAAK;;;;;ACjBf,eAAsB,cACpB,YACA,eACA,SACA,gBACuB;CACvB,MAAM,YAAY,WAAW,IAAI,cAAc;AAC/C,KAAI,CAAC,UACH,QAAO;EACL,QAAQ;EACR,MAAM,IAAI,UAAU,aAAa,cAAc,cAAc,aAAa,CAAC,QAAQ;EACpF;CAGH,MAAM,aAAa,cAAc,UAAU,aAAa,QAAQ;AAChE,KAAI,CAAC,WAAW,MAEd,QAAO;EACL,QAAQ;EACR,MAAM,IAAI,UAAU,oBAAoB,4BAH1B,uBAAuB,WAAW,OAAO,GAGuB,CAAC,QAAQ;EACxF;AAGH,KAAI;EACF,MAAM,SAAS,MAAM,UAAU,QAAQ,EAAE,OAAO,SAAS,CAAC;AAE1D,MAAI,gBAAgB;GAClB,MAAM,gBAAgB,cAAc,UAAU,cAAc,OAAO;AACnE,OAAI,CAAC,cAAc,MAEjB,QAAO;IACL,QAAQ;IACR,MAAM,IAAI,UAAU,kBAAkB,6BAHxB,uBAAuB,cAAc,OAAO,GAGmB,CAAC,QAAQ;IACvF;;AAIL,SAAO;GAAE,QAAQ;GAAK,MAAM;GAAQ;UAC7B,OAAO;AACd,MAAI,iBAAiB,UACnB,QAAO;GAAE,QAAQ,MAAM;GAAQ,MAAM,MAAM,QAAQ;GAAE;AAGvD,SAAO;GACL,QAAQ;GACR,MAAM,IAAI,UAAU,kBAHN,iBAAiB,QAAQ,MAAM,UAAU,gBAGT,CAAC,QAAQ;GACxD;;;AAaL,eAAsB,mBACpB,YACA,OACA,gBACyC;AAWzC,QAAO,EAAE,SAVO,MAAM,QAAQ,IAC5B,MAAM,IAAI,OAAO,SAAS;EACxB,MAAM,SAAS,MAAM,cAAc,YAAY,KAAK,WAAW,KAAK,OAAO,eAAe;AAC1F,MAAI,OAAO,WAAW,IACpB,QAAO;GAAE,IAAI;GAAe,MAAM,OAAO;GAAM;AAGjD,SAAO;GAAE,IAAI;GAAgB,OADZ,OAAO,KACqB;GAAO;GACpD,CACH,EACiB;;AAGpB,gBAAuB,mBACrB,eACA,MACA,UACA,gBACwB;CACxB,MAAM,MAAM,cAAc,IAAI,KAAK;AACnC,KAAI,CAAC,IACH,OAAM,IAAI,UAAU,aAAa,iBAAiB,KAAK,aAAa;CAGtE,MAAM,aAAa,cAAc,IAAI,aAAa,SAAS;AAC3D,KAAI,CAAC,WAAW,MAEd,OAAM,IAAI,UAAU,oBAAoB,4BADxB,uBAAuB,WAAW,OAAO,GACqB;AAGhF,YAAW,MAAM,SAAS,IAAI,QAAQ,EAAE,OAAO,UAAU,CAAC,EAAE;AAC1D,MAAI,gBAAgB;GAClB,MAAM,gBAAgB,cAAc,IAAI,cAAc,MAAM;AAC5D,OAAI,CAAC,cAAc,MAEjB,OAAM,IAAI,UAAU,kBAAkB,6BADtB,uBAAuB,cAAc,OAAO,GACiB;;AAGjF,QAAM;;;;;;;AChFV,eAAe,eACb,SACA,QACA,YACkC;CAClC,MAAM,UAAU,OAAO,QAAQ,QAAQ;CACvC,MAAM,UAAU,MAAM,QAAQ,IAC5B,QAAQ,IAAI,OAAO,CAAC,KAAK,YAAY;EACnC,MAAM,EAAE,WAAW,UAAU,OAAO,OAAO;EAC3C,MAAM,OAAO,WAAW,IAAI,UAAU;AACtC,MAAI,CAAC,KAAM,OAAM,IAAI,UAAU,kBAAkB,cAAc,UAAU,aAAa;AAGtF,SAAO,CAAC,KADO,MAAM,KAAK,QAAQ,EAAE,OAAO,CAAC,CACxB;GACpB,CACH;AACD,QAAO,OAAO,YAAY,QAAQ;;;AAIpC,SAAS,eACP,iBACA,iBACA,QACQ;AACR,KAAI,UAAU,gBACZ,QAAO,gBAAgB,WAAW;AAEpC,QAAO;;;AAIT,SAAS,eACP,QACA,cACA,QACwB;CACxB,MAAM,YAAY,OAAO,YAAY;AACrC,KAAI,CAAC,UAAW,QAAO,EAAE;AAEzB,KAAI,OAAO,SAAS,WAAW,OAAO,SAAS;EAC7C,MAAM,WAAW,KAAK,OAAO,SAAS,QAAQ,WAAW,GAAG,OAAO,OAAO;AAC1E,MAAI,WAAW,SAAS,CACtB,QAAO,KAAK,MAAM,aAAa,UAAU,QAAQ,CAAC;AAEpD,SAAO,EAAE;;AAGX,QAAO,OAAO,SAAS,UAAU,cAAc,EAAE;;AAGnD,eAAsB,kBACpB,MACA,QACA,YACA,UAC2B;AAC3B,KAAI;EACF,MAAM,KAAK,YAAY,KAAK;EAC5B,MAAM,cAAc,KAAK,eAAe,EAAE;EAC1C,MAAM,SAAS,UAAU;EAGzB,MAAM,gBAAgB,MAAM,QAAQ,IAAI,CACtC,GAAG,YAAY,KAAK,WAAW,eAAe,OAAO,SAAS,QAAQ,WAAW,CAAC,EAClF,eAAe,KAAK,SAAS,QAAQ,WAAW,CACjD,CAAC;EAEF,MAAM,KAAK,YAAY,KAAK;EAG5B,MAAM,UAAmC,EAAE;AAC3C,OAAK,MAAM,UAAU,cACnB,QAAO,OAAO,SAAS,OAAO;EAKhC,IAAI,mBADiB,eAAe,KAAK,UAAU,KAAK,iBAAiB,OAAO;AAEhF,OAAK,IAAI,IAAI,YAAY,SAAS,GAAG,KAAK,GAAG,KAAK;GAChD,MAAM,SAAS,YAAY;AAE3B,sBADuB,eAAe,OAAO,UAAU,OAAO,iBAAiB,OAAO,CACpD,QAAQ,sBAAsB,iBAAiB;;EAInF,MAAM,SAAS;GACb,cAAc,YAAY,KAAK,OAAO;IACpC,IAAI,EAAE;IACN,aAAa,OAAO,KAAK,EAAE,QAAQ;IACpC,EAAE;GACH,SAAS,KAAK,UAAU;GACxB,WAAW,KAAK;GACjB;EAGD,IAAI;AACJ,MAAI,UAAU;GACZ,MAAM,EAAE,QAAQ,YAAY,iBAAiB;GAC7C,MAAM,WAAW,eAAe,YAAY,cAAc,SAAS,OAAO;GAC1E,MAAM,YAAY,WAAW,YAAY;GACzC,MAAM,WAAoC;IACxC,QAAQ,SAAS;IACjB,gBAAgB,WAAW;IAC3B;IACD;AAED,OAAI,WAAW,SAAS,WAAW;AACjC,aAAS,OAAO,WAAW,cAAc,aAAa,SAAS;AAC/D,aAAS,SAAS,WAAW;;AAE/B,kBAAe,KAAK,UAAU,SAAS;;EAIzC,MAAM,OAAO,WACX,kBACA,KAAK,UAAU,QAAQ,EACvB,KAAK,UAAU,OAAO,EACtB,aACD;EAED,MAAM,KAAK,YAAY,KAAK;AAE5B,SAAO;GACL,QAAQ;GACR;GACA,QAAQ;IAAE,WAAW,KAAK;IAAI,QAAQ,KAAK;IAAI;GAChD;UACM,OAAO;AAEd,SAAO;GACL,QAAQ;GACR,MAAM,mEAAmE,WAH3D,iBAAiB,QAAQ,MAAM,UAAU,gBAGqC,CAAC;GAC9F;;;;;;AC5JL,SAAS,aAAa,SAAgC;AAOpD,QAAO,EAAE,UANwB,QAC9B,MAAM,IAAI,CACV,OAAO,QAAQ,CACf,KAAK,QACJ,IAAI,WAAW,IAAI,GAAG;EAAE,MAAM;EAAS,MAAM,IAAI,MAAM,EAAE;EAAE,GAAG;EAAE,MAAM;EAAU,OAAO;EAAK,CAC7F,EACgB;;AAGrB,SAAS,WAAW,UAA0B,WAAoD;AAChG,KAAI,SAAS,WAAW,UAAU,OAAQ,QAAO;CACjD,MAAM,SAAiC,EAAE;AACzC,MAAK,IAAI,IAAI,GAAG,IAAI,SAAS,QAAQ,KAAK;EACxC,MAAM,MAAM,SAAS;AACrB,MAAI,IAAI,SAAS,UACf;OAAI,IAAI,UAAU,UAAU,GAAI,QAAO;QAEvC,QAAO,IAAI,QAAQ,UAAU;;AAGjC,QAAO;;AAGT,IAAa,eAAb,MAA6B;CAC3B,AAAQ,SAAmE,EAAE;CAE7E,IAAI,SAAiB,OAAgB;AACnC,OAAK,OAAO,KAAK;GAAE;GAAS,UAAU,aAAa,QAAQ;GAAE;GAAO,CAAC;;CAGvE,MAAM,MAAoF;EACxF,MAAM,QAAQ,KAAK,MAAM,IAAI,CAAC,OAAO,QAAQ;AAC7C,OAAK,MAAM,SAAS,KAAK,QAAQ;GAC/B,MAAM,SAAS,WAAW,MAAM,SAAS,UAAU,MAAM;AACzD,OAAI,OAAQ,QAAO;IAAE,OAAO,MAAM;IAAO;IAAQ,SAAS,MAAM;IAAS;;AAE3E,SAAO;;;;;;;AC1BX,SAAgB,gBAAiC;AAC/C,QAAO;EACL,MAAM;EACN,QAAQ,MAAM;AACZ,OAAI,KAAK,cAAc,KAAK,QAAQ,SAAS,KAAK,WAAW,CAC3D,QAAO,KAAK;AAEd,UAAO;;EAEV;;;AAIH,SAAgB,WAAW,OAAO,eAAgC;AAChE,QAAO;EACL,MAAM;EACN,QAAQ,MAAM;AACZ,OAAI,CAAC,KAAK,OAAQ,QAAO;AACzB,QAAK,MAAM,QAAQ,KAAK,OAAO,MAAM,IAAI,EAAE;IACzC,MAAM,CAAC,GAAG,KAAK,KAAK,MAAM,CAAC,MAAM,IAAI;AACrC,QAAI,MAAM,QAAQ,KAAK,KAAK,QAAQ,SAAS,EAAE,CAAE,QAAO;;AAE1D,UAAO;;EAEV;;;AAIH,SAAgB,qBAAsC;AACpD,QAAO;EACL,MAAM;EACN,QAAQ,MAAM;AACZ,OAAI,CAAC,KAAK,eAAgB,QAAO;GACjC,MAAM,UAAyC,EAAE;AACjD,QAAK,MAAM,QAAQ,KAAK,eAAe,MAAM,IAAI,EAAE;IAEjD,MAAM,CAAC,MAAM,GAAG,QADA,KAAK,MAAM,CACK,MAAM,IAAI;IAC1C,IAAI,IAAI;AACR,SAAK,MAAM,KAAK,MAAM;KACpB,MAAM,QAAQ,EAAE,MAAM,CAAC,MAAM,sBAAsB;AACnD,SAAI,MAAO,KAAI,WAAW,MAAM,GAAG;;AAErC,YAAQ,KAAK;KAAE,MAAM,KAAK,MAAM;KAAE;KAAG,CAAC;;AAExC,WAAQ,MAAM,GAAG,MAAM,EAAE,IAAI,EAAE,EAAE;GACjC,MAAM,YAAY,IAAI,IAAI,KAAK,QAAQ;AACvC,QAAK,MAAM,EAAE,UAAU,SAAS;AAC9B,QAAI,UAAU,IAAI,KAAK,CAAE,QAAO;IAEhC,MAAM,SAAS,KAAK,MAAM,IAAI,CAAC;AAC/B,QAAI,WAAW,QAAQ,UAAU,IAAI,OAAO,CAAE,QAAO;;AAEvD,UAAO;;EAEV;;;AAIH,SAAgB,aAAa,QAAQ,QAAyB;AAC5D,QAAO;EACL,MAAM;EACN,QAAQ,MAAM;AACZ,OAAI,CAAC,KAAK,IAAK,QAAO;AACtB,OAAI;IAEF,MAAM,QADM,IAAI,IAAI,KAAK,KAAK,mBAAmB,CAC/B,aAAa,IAAI,MAAM;AACzC,QAAI,SAAS,KAAK,QAAQ,SAAS,MAAM,CAAE,QAAO;WAC5C;AAGR,UAAO;;EAEV;;;AAIH,SAAgB,aAAa,YAA+B,MAA2B;AACrF,MAAK,MAAM,KAAK,YAAY;EAC1B,MAAM,SAAS,EAAE,QAAQ,KAAK;AAC9B,MAAI,WAAW,KAAM,QAAO;;AAE9B,QAAO,KAAK;;;AAId,SAAgB,oBAAuC;AACrD,QAAO;EAAC,eAAe;EAAE,YAAY;EAAE,oBAAoB;EAAC;;;;;ACvE9D,SAAS,kBAAkB,KAA6D;AACtF,QAAO,UAAU,OAAO,IAAI,SAAS;;;AA4BvC,SAAS,gBAAgB,MAGvB;CACA,MAAM,aAAa,MAAM,WAAW,mBAAmB;AACvD,QAAO;EACL;EACA,cAAc,WAAW,MAAM,MAAM,EAAE,SAAS,aAAa;EAC9D;;;AAIH,SAAS,kBAAkB,cAA8C,QAA0B;AACjG,cAAa,IAAI,qBAAqB;EACpC,aAAa,EAAE;EACf,cAAc,EAAE;EAChB,UAAU,EAAE,YAAY;GACtB,MAAM,EAAE,OAAO,WAAW;GAC1B,MAAM,WAAW,mBAAmB,QAAQ,OAAO,OAAO;AAE1D,UAAO;IAAE,MADI,OAAO,cAAc,SAAS,WAAW;IACvC;IAAU;;EAE5B,CAAC;;;AAIJ,SAAS,mBACP,QACA,WACA,QACwB;AACxB,KAAI,OAAO,SAAS,WAAW,OAAO,SAAS;EAC7C,MAAM,WAAW,KAAK,OAAO,SAAS,QAAQ,WAAW,GAAG,OAAO,OAAO;AAC1E,MAAI,WAAW,SAAS,CACtB,QAAO,KAAK,MAAM,aAAa,UAAU,QAAQ,CAAC;AAEpD,SAAO,EAAE;;AAEX,QAAO,OAAO,SAAS,UAAU,cAAc,EAAE;;AAGnD,SAAgB,aACd,YACA,MACW;CACX,MAAM,+BAAe,IAAI,KAAgC;CACzD,MAAM,kCAAkB,IAAI,KAAmC;AAE/D,MAAK,MAAM,CAAC,MAAM,QAAQ,OAAO,QAAQ,WAAW,CAClD,KAAI,kBAAkB,IAAI,CACxB,iBAAgB,IAAI,MAAM;EACxB,aAAa,IAAI,MAAM;EACvB,cAAc,IAAI,OAAO;EACzB,SAAS,IAAI;EACd,CAAC;KAEF,cAAa,IAAI,MAAM;EACrB,aAAa,IAAI,MAAM;EACvB,cAAc,IAAI,OAAO;EACzB,SAAS,IAAI;EACd,CAAC;CAIN,MAAM,uBACJ,MAAM,mBACL,OAAO,YAAY,eAAe,QAAQ,IAAI,aAAa;CAE9D,MAAM,cAAc,IAAI,cAAuB;CAC/C,MAAM,QAAQ,MAAM;AACpB,KAAI,MACF,MAAK,MAAM,CAAC,SAAS,SAAS,OAAO,QAAQ,MAAM,CACjD,aAAY,IAAI,SAAS,KAAK;CAIlC,MAAM,aAAa,MAAM,QAAQ;CACjC,MAAM,EAAE,YAAY,iBAAiB,gBAAgB,KAAK;AAC1D,KAAI,WAAY,mBAAkB,cAAc,WAAW;AAE3D,QAAO;EACL;EACA,UAAU,CAAC,CAAC,SAAS,OAAO,KAAK,MAAM,CAAC,SAAS;EACjD,WAAW;AACT,UAAO,cAAc,WAAW;;EAElC,OAAO,eAAe,MAAM;AAC1B,UAAO,cAAc,cAAc,eAAe,MAAM,qBAAqB;;EAE/E,YAAY,OAAO;AACjB,UAAO,mBAAmB,cAAc,OAAO,qBAAqB;;EAEtE,mBAAmB,MAAM,OAAO;AAC9B,UAAO,mBAAmB,iBAAiB,MAAM,OAAO,qBAAqB;;EAE/E,MAAM,WAAW,MAAM,SAAS;GAC9B,IAAI,aAA4B;GAChC,IAAI,YAAY;AAEhB,OAAI,gBAAgB,YAAY;IAC9B,MAAM,WAAW,KAAK,MAAM,IAAI,CAAC,OAAO,QAAQ;IAChD,MAAM,YAAY,IAAI,IAAI,WAAW,QAAQ;AAC7C,QAAI,SAAS,SAAS,KAAK,UAAU,IAAI,SAAS,GAAG,EAAE;AACrD,kBAAa,SAAS;AACtB,iBAAY,MAAM,SAAS,MAAM,EAAE,CAAC,KAAK,IAAI,IAAI;;;GAIrD,IAAI;AACJ,OAAI,WACF,UAAS,aAAa,YAAY;IAChC,KAAK,SAAS,OAAO;IACrB;IACA,QAAQ,SAAS;IACjB,gBAAgB,SAAS;IACzB,SAAS,WAAW;IACpB,eAAe,WAAW;IAC3B,CAAC;GAGJ,MAAM,QAAQ,YAAY,MAAM,UAAU;AAC1C,OAAI,CAAC,MAAO,QAAO;GAEnB,MAAM,WACJ,UAAU,aACN;IAAE;IAAQ,QAAQ;IAAY,cAAc,MAAM;IAAS,GAC3D;AACN,UAAO,kBAAkB,MAAM,OAAO,MAAM,QAAQ,cAAc,SAAS;;EAE9E;;;;;ACxJH,SAAgB,WAAW,QAA0B;AACnD,QAAO;EAAE,GAAG;EAAQ,aAAa,OAAO,eAAe,EAAE;EAAE;;;;;ACvC7D,MAAa,aAAqC;CAChD,OAAO;CACP,QAAQ;CACR,QAAQ;CACR,SAAS;CACT,SAAS;CACT,QAAQ;CACR,QAAQ;CACR,QAAQ;CACR,SAAS;CACT,QAAQ;CACR,SAAS;CACT,UAAU;CACV,QAAQ;CACR,QAAQ;CACR,QAAQ;CACR,OAAO;CACP,QAAQ;CACT;;;;ACsBD,MAAM,aAAa;AACnB,MAAM,cAAc;AACpB,MAAM,gBAAgB;AACtB,MAAM,mBAAmB;AACzB,MAAM,gBAAgB;AAEtB,MAAM,cAAc,EAAE,gBAAgB,oBAAoB;AAC1D,MAAM,cAAc,EAAE,gBAAgB,4BAA4B;AAClE,MAAM,aAAa;CACjB,gBAAgB;CAChB,iBAAiB;CACjB,YAAY;CACb;AACD,MAAM,kBAAkB;AAExB,SAAS,aAAa,QAAgB,MAAiC;AACrE,QAAO;EAAE;EAAQ,SAAS;EAAa;EAAM;;AAG/C,SAAS,cAAc,QAAgB,MAAc,SAAmC;AACtF,QAAO,aAAa,QAAQ,IAAI,UAAU,MAAM,QAAQ,CAAC,QAAQ,CAAC;;AAGpE,eAAe,kBAAkB,WAAmB,WAA8C;AAChG,KAAI,UAAU,SAAS,KAAK,CAC1B,QAAO,cAAc,KAAK,oBAAoB,YAAY;CAG5D,MAAM,WAAW,KAAK,WAAW,UAAU;AAC3C,KAAI;EACF,MAAM,UAAU,MAAM,SAAS,UAAU,QAAQ;EAEjD,MAAM,cAAc,WADR,QAAQ,SAAS,KACU;AACvC,SAAO;GACL,QAAQ;GACR,SAAS;IACP,gBAAgB;IAChB,iBAAiB;IAClB;GACD,MAAM;GACP;SACK;AACN,SAAO,cAAc,KAAK,aAAa,kBAAkB;;;;AAK7D,SAAgB,aAAa,MAAuB;AAClD,QAAO,sBAAsB,KAAK,UAAU,KAAK,CAAC;;;AAIpD,SAAgB,cAAc,MAAc,SAAyB;AACnE,QAAO,uBAAuB,KAAK,UAAU;EAAE;EAAM;EAAS,CAAC,CAAC;;;AAIlE,SAAgB,mBAA2B;AACzC,QAAO;;AAGT,gBAAgB,UACd,QACA,MACA,OACuB;AACvB,KAAI;AACF,aAAW,MAAM,SAAS,OAAO,mBAAmB,MAAM,MAAM,CAC9D,OAAM,aAAa,MAAM;AAE3B,QAAM,kBAAkB;UACjB,OAAO;AACd,MAAI,iBAAiB,UACnB,OAAM,cAAc,MAAM,MAAM,MAAM,QAAQ;MAG9C,OAAM,cAAc,kBADJ,iBAAiB,QAAQ,MAAM,UAAU,gBACX;;;AAKpD,eAAe,gBACb,KACA,QACA,YAC2B;CAC3B,IAAI;AACJ,KAAI;AACF,SAAO,MAAM,IAAI,MAAM;SACjB;AACN,SAAO,cAAc,KAAK,oBAAoB,oBAAoB;;AAEpE,KAAI,CAAC,QAAQ,OAAO,SAAS,YAAY,CAAC,MAAM,QAAS,KAA6B,MAAM,CAC1F,QAAO,cAAc,KAAK,oBAAoB,0CAA0C;CAE1F,MAAM,QAAS,KAAoE,MAAM,KACtF,OAAO;EACN,WACE,OAAO,EAAE,cAAc,WAAY,YAAY,IAAI,EAAE,UAAU,IAAI,EAAE,YAAa;EACpF,OAAO,EAAE,SAAS,EAAE;EACrB,EACF;AAED,QAAO,aAAa,KADL,MAAM,OAAO,YAAY,MAAM,CACd;;AAGlC,SAAgB,kBACd,QACA,MACa;CAEb,MAAM,aAAyC,MAAM,aACjD,IAAI,IAAI,OAAO,QAAQ,KAAK,WAAW,WAAW,CAAC,KAAK,CAAC,GAAG,OAAO,CAAC,GAAG,EAAE,CAAC,CAAC,GAC3E;AAEJ,KAAI,WACF,YAAW,IAAI,qBAAqB,oBAAoB;CAE1D,MAAM,YAAY,MAAM,YAAY,SAAS;AAE7C,QAAO,OAAO,QAAQ;EACpB,MAAM,MAAM,IAAI,IAAI,IAAI,KAAK,mBAAmB;EAChD,MAAM,EAAE,aAAa;AAErB,MAAI,IAAI,WAAW,SAAS,aAAa,eAAe;AACtD,OAAI,MAAM,WAAY,QAAO,cAAc,KAAK,aAAa,oBAAoB;AACjF,UAAO,aAAa,KAAK,OAAO,UAAU,CAAC;;AAG7C,MAAI,IAAI,WAAW,UAAU,SAAS,WAAW,WAAW,EAAE;GAC5D,IAAI,OAAO,SAAS,MAAM,GAAkB;AAC5C,OAAI,CAAC,KACH,QAAO,cAAc,KAAK,aAAa,uBAAuB;AAIhE,OAAI,SAAS,YAAa,aAAa,SAAS,UAC9C,QAAO,gBAAgB,KAAK,QAAQ,WAAW;AAIjD,OAAI,YAAY;IACd,MAAM,WAAW,WAAW,IAAI,KAAK;AACrC,QAAI,CAAC,SAAU,QAAO,cAAc,KAAK,aAAa,YAAY;AAClE,WAAO;;GAGT,IAAI;AACJ,OAAI;AACF,WAAO,MAAM,IAAI,MAAM;WACjB;AACN,WAAO,cAAc,KAAK,oBAAoB,oBAAoB;;GAGpE,MAAM,SAAS,MAAM,OAAO,OAAO,MAAM,KAAK;AAC9C,UAAO,aAAa,OAAO,QAAQ,OAAO,KAAK;;AAGjD,MAAI,IAAI,WAAW,SAAS,SAAS,WAAW,iBAAiB,EAAE;GACjE,IAAI,OAAO,SAAS,MAAM,GAAwB;AAClD,OAAI,CAAC,KACH,QAAO,cAAc,KAAK,aAAa,0BAA0B;AAInE,OAAI,YAAY;IACd,MAAM,WAAW,WAAW,IAAI,KAAK;AACrC,QAAI,CAAC,SAAU,QAAO,cAAc,KAAK,aAAa,YAAY;AAClE,WAAO;;GAGT,MAAM,WAAW,IAAI,aAAa,IAAI,QAAQ;GAC9C,IAAI;AACJ,OAAI;AACF,YAAQ,WAAW,KAAK,MAAM,SAAS,GAAG,EAAE;WACtC;AACN,WAAO,cAAc,KAAK,oBAAoB,gCAAgC;;AAGhF,UAAO;IAAE,QAAQ;IAAK,SAAS;IAAY,QAAQ,UAAU,QAAQ,MAAM,MAAM;IAAE;;AAMrF,MAAI,IAAI,WAAW,SAAS,SAAS,WAAW,YAAY,IAAI,OAAO,UAAU;GAC/E,MAAM,WAAW,MAAM,SAAS,MAAM,GAAmB;GACzD,MAAM,UAAU,IAAI,SAChB;IACE,KAAK,IAAI;IACT,QAAQ,IAAI,OAAO,SAAS,IAAI;IAChC,gBAAgB,IAAI,OAAO,kBAAkB,IAAI;IAClD,GACD;GACJ,MAAM,SAAS,MAAM,OAAO,WAAW,UAAU,QAAQ;AACzD,OAAI,OACF,QAAO;IAAE,QAAQ,OAAO;IAAQ,SAAS;IAAa,MAAM,OAAO;IAAM;;AAI7E,MAAI,IAAI,WAAW,SAAS,SAAS,WAAW,cAAc,IAAI,MAAM,UAEtE,QAAO,kBADW,SAAS,MAAM,GAAqB,EAClB,KAAK,UAAU;AAGrD,MAAI,MAAM,SAAU,QAAO,KAAK,SAAS,IAAI;AAC7C,SAAO,cAAc,KAAK,aAAa,YAAY;;;AAIvD,SAAgB,UAAU,MAAuB;AAC/C,QAAO,OAAO,SAAS,WAAW,OAAO,KAAK,UAAU,KAAK;;;AAI/D,eAAsB,YACpB,QACA,OACe;AACf,KAAI;AACF,aAAW,MAAM,SAAS,OACxB,KAAI,MAAM,MAAM,KAAK,MAAO;SAExB;;;AAMV,SAAgB,cAAc,QAAgC;AAC5D,KAAI,YAAY,QAAQ;EACtB,MAAM,SAAS,OAAO;EACtB,MAAM,UAAU,IAAI,aAAa;EACjC,MAAM,WAAW,IAAI,eAAe,EAClC,MAAM,MAAM,YAAY;AACtB,SAAM,YAAY,SAAS,UAAU;AACnC,eAAW,QAAQ,QAAQ,OAAO,MAAM,CAAC;KACzC;AACF,cAAW,OAAO;KAErB,CAAC;AACF,SAAO,IAAI,SAAS,UAAU;GAAE,QAAQ,OAAO;GAAQ,SAAS,OAAO;GAAS,CAAC;;AAEnF,QAAO,IAAI,SAAS,UAAU,OAAO,KAAK,EAAE;EAAE,QAAQ,OAAO;EAAQ,SAAS,OAAO;EAAS,CAAC;;;;;AC7OjG,SAAS,cAAc,QAAgC;AACrD,SAAQ,WAAiD;EACvD,MAAM,QAAiC,EAAE;AACzC,MAAI,OAAO,OACT,MAAK,MAAM,CAAC,KAAK,YAAY,OAAO,QAAQ,OAAO,OAAO,EAAE;GAC1D,MAAM,MAAM,OAAO;AACnB,SAAM,OAAO,QAAQ,SAAS,QAAQ,OAAO,IAAI,GAAG;;AAGxD,SAAO;GAAE,WAAW,OAAO;GAAW;GAAO;;;AAIjD,SAAS,eAAe,SAAiE;CACvF,MAAM,MAAgC,EAAE;AACxC,MAAK,MAAM,CAAC,KAAK,WAAW,OAAO,QAAQ,QAAQ,CACjD,KAAI,OAAO,cAAc,OAAO;AAElC,QAAO;;AAGT,SAAS,oBACP,OACA,eACQ;AACR,KAAI,MAAM,SAAU,QAAO,MAAM;AACjC,KAAI,MAAM,WAAW;EACnB,MAAM,SAAS,iBAAiB,OAAO,KAAK,MAAM,UAAU,CAAC;EAC7D,MAAM,OAAO,MAAM,UAAU;AAC7B,MAAI,CAAC,KAAM,OAAM,IAAI,MAAM,2BAA2B,OAAO,GAAG;AAChE,SAAO;;AAET,OAAM,IAAI,MAAM,wDAAwD;;;AAI1E,SAAS,oBACP,OACA,SACoC;AACpC,KAAI,CAAC,MAAM,UAAW,QAAO;CAC7B,MAAM,SAAiC,EAAE;AACzC,MAAK,MAAM,CAAC,QAAQ,YAAY,OAAO,QAAQ,MAAM,UAAU,CAC7D,QAAO,UAAU,aAAa,KAAK,SAAS,QAAQ,EAAE,QAAQ;AAEhE,QAAO;;;AAIT,SAAS,mBACP,UACA,eACA,WACA,oBACa;CACb,MAAM,QAAqB,EAAE;CAC7B,IAAI,YAAgC;AAEpC,QAAO,WAAW;EAChB,MAAM,QAAyC,cAAc;AAC7D,MAAI,CAAC,MAAO;AACZ,QAAM,KAAK;GACT,IAAI;GACJ,UAAU,UAAU;GACpB,iBAAiB,mBAAmB;GACpC,SAAS,eAAe,MAAM,WAAW,EAAE,CAAC;GAC7C,CAAC;AACF,cAAY,MAAM;;AAIpB,OAAM,SAAS;AACf,QAAO;;;AAIT,SAAS,sBACP,UACA,eACA,SACA,eACa;CACb,MAAM,QAAqB,EAAE;CAC7B,IAAI,YAAgC;AAEpC,QAAO,WAAW;EAChB,MAAM,QAAyC,cAAc;AAC7D,MAAI,CAAC,MAAO;EACZ,MAAM,qBAAqB,KAAK,SAAS,oBAAoB,OAAO,cAAc,CAAC;EACnF,MAAM,MAAiB;GACrB,IAAI;GACJ,UAAU;GACV,iBAAiB,MAAM,YACnB,0BAA0B,MAAM,WAAW,QAAQ,GACnD;GACJ,SAAS,eAAe,MAAM,WAAW,EAAE,CAAC;GAC7C;AACD,SAAO,eAAe,KAAK,YAAY;GACrC,WAAW,aAAa,oBAAoB,QAAQ;GACpD,YAAY;GACb,CAAC;AACF,QAAM,KAAK,IAAI;AACf,cAAY,MAAM;;AAGpB,OAAM,SAAS;AACf,QAAO;;;AAIT,SAAS,0BACP,WACA,SACwB;CACxB,MAAM,MAA8B,EAAE;AACtC,MAAK,MAAM,CAAC,QAAQ,YAAY,OAAO,QAAQ,UAAU,EAAE;EACzD,MAAM,WAAW,KAAK,SAAS,QAAQ;AACvC,SAAO,eAAe,KAAK,QAAQ;GACjC,WAAW,aAAa,UAAU,QAAQ;GAC1C,YAAY;GACb,CAAC;;AAEJ,QAAO;;;AAIT,SAAS,cACP,OACA,eACsB;CACtB,MAAM,OAAiB,EAAE;AACzB,KAAI,MAAM,QAAQ;EAChB,IAAI,YAAgC,MAAM;AAC1C,SAAO,WAAW;GAChB,MAAM,QAAyC,cAAc;AAC7D,OAAI,CAAC,MAAO;AACZ,OAAI,MAAM,UAAW,MAAK,KAAK,GAAG,MAAM,UAAU;AAClD,eAAY,MAAM;;;AAGtB,KAAI,MAAM,UAAW,MAAK,KAAK,GAAG,MAAM,UAAU;AAClD,QAAO,KAAK,SAAS,IAAI,OAAO;;;AAIlC,SAAgB,eAAe,SAAyC;CACtE,MAAM,cAAc,KAAK,SAAS,oBAAoB;AACtD,KAAI;AACF,SAAO,KAAK,MAAM,aAAa,aAAa,QAAQ,CAAC;SAC/C;AACN;;;;AAKJ,SAAgB,iBAAiB,SAAoC;CACnE,MAAM,eAAe,KAAK,SAAS,sBAAsB;AACzD,KAAI;EACF,MAAM,WAAW,KAAK,MAAM,aAAa,cAAc,QAAQ,CAAC;AAChE,MAAI,CAAC,SAAS,KAAM,QAAO;EAE3B,MAAM,OAAQ,SAAS,KAAK,QAAQ;EACpC,MAAM,QAAQ,SAAS,KAAK,SAAS;EACrC,MAAM,cAAc,SAAS,KAAK,gBAAgB,EAAE;EACpD,MAAM,gBAAgB,SAAS,KAAK,kBAAkB,EAAE;EAIxD,MAAM,WAAmE,EAAE;AAC3E,MAAI,SAAS,UAAU;GACrB,MAAM,UAAU,KAAK,SAAS,OAAO;AACrC,QAAK,MAAM,UAAU,SAAS,KAAK,SAAS;IAC1C,MAAM,aAAa,KAAK,SAAS,GAAG,OAAO,OAAO;AAClD,QAAI,WAAW,WAAW,CACxB,UAAS,UAAU,KAAK,MAAM,aAAa,YAAY,QAAQ,CAAC;QAKhE,UAAS,UAAU,EAAE;;;AAK3B,SAAO;GACL,SAAS,SAAS,KAAK;GACvB,SAAS,SAAS,KAAK;GACvB;GACA;GACA;GACA;GACA;GACA,SAAS,SAAS,UAAU,UAAU;GACvC;SACK;AACN,SAAO;;;AAIX,SAAgB,gBAAgB,SAA0C;CAExE,MAAM,MAAM,aADS,KAAK,SAAS,sBAAsB,EAClB,QAAQ;CAC/C,MAAM,WAAW,KAAK,MAAM,IAAI;CAChC,MAAM,gBAAgB,SAAS,MAAM;CAGrC,MAAM,kBAA0C,EAAE;CAClD,MAAM,wBAAgE,EAAE;CACxE,MAAM,gBAAgB,SAAS,WAAW,EAAE;AAC5C,MAAK,MAAM,CAAC,IAAI,UAAU,OAAO,QAAQ,cAAc,EAAE;AACvD,kBAAgB,MAAM,aACpB,KAAK,SAAS,oBAAoB,OAAO,cAAc,CAAC,EACxD,QACD;EACD,MAAM,KAAK,oBAAoB,OAAO,QAAQ;AAC9C,MAAI,GAAI,uBAAsB,MAAM;;CAGtC,MAAM,QAAiC,EAAE;AACzC,MAAK,MAAM,CAAC,MAAM,UAAU,OAAO,QAAQ,SAAS,OAAO,EAAE;EAE3D,MAAM,WAAW,aADI,KAAK,SAAS,oBAAoB,OAAO,cAAc,CAAC,EACjC,QAAQ;EAEpD,MAAM,UAAU,eAAe,MAAM,QAAQ;EAC7C,MAAM,cAAc,MAAM,SACtB,mBAAmB,MAAM,QAAQ,eAAe,iBAAiB,sBAAsB,GACvF,EAAE;EAGN,MAAM,WAAW,cAAc,OAAO,cAAc;AAEpD,QAAM,QAAQ;GACZ;GACA,iBAAiB,oBAAoB,OAAO,QAAQ;GACpD;GACA;GACA,UAAU,MAAM;GAChB,QAAQ,SAAS;GACjB;GACD;;AAEH,QAAO;;;AAIT,SAAgB,mBAAmB,SAA0C;CAE3E,MAAM,MAAM,aADS,KAAK,SAAS,sBAAsB,EAClB,QAAQ;CAC/C,MAAM,WAAW,KAAK,MAAM,IAAI;CAChC,MAAM,gBAAgB,SAAS,MAAM;CAErC,MAAM,gBAAgB,SAAS,WAAW,EAAE;CAE5C,MAAM,QAAiC,EAAE;AACzC,MAAK,MAAM,CAAC,MAAM,UAAU,OAAO,QAAQ,SAAS,OAAO,EAAE;EAC3D,MAAM,eAAe,KAAK,SAAS,oBAAoB,OAAO,cAAc,CAAC;EAC7E,MAAM,UAAU,eAAe,MAAM,QAAQ;EAC7C,MAAM,cAAc,MAAM,SACtB,sBAAsB,MAAM,QAAQ,eAAe,SAAS,cAAc,GAC1E,EAAE;EAEN,MAAM,kBAAkB,MAAM,YAC1B,0BAA0B,MAAM,WAAW,QAAQ,GACnD;EAGJ,MAAM,WAAW,cAAc,OAAO,cAAc;EAEpD,MAAM,OAAgB;GACpB,UAAU;GACV;GACA;GACA;GACA,QAAQ,SAAS;GACjB;GACD;AACD,SAAO,eAAe,MAAM,YAAY;GACtC,WAAW,aAAa,cAAc,QAAQ;GAC9C,YAAY;GACb,CAAC;AACF,QAAM,QAAQ;;AAEhB,QAAO;;;;;ACrTT,SAAgB,aACd,OACoC;CACpC,MAAM,QAAwB,EAAE;CAChC,IAAI,UAA+B;CACnC,IAAI,OAAO;CACX,SAAS,SAAS;AAChB,MAAI,SAAS;GACX,MAAM,IAAI;AACV,aAAU;AACV,MAAG;;;CAwBP,MAAM,UAAU,MApBc;EAC5B,KAAK,OAAO;AACV,OAAI,KAAM;AACV,SAAM,KAAK;IAAE,MAAM;IAAS;IAAO,CAAC;AACpC,WAAQ;;EAEV,MAAM;AACJ,OAAI,KAAM;AACV,UAAO;AACP,SAAM,KAAK,EAAE,MAAM,OAAO,CAAC;AAC3B,WAAQ;;EAEV,MAAM,KAAK;AACT,OAAI,KAAM;AACV,UAAO;AACP,SAAM,KAAK;IAAE,MAAM;IAAS,OAAO;IAAK,CAAC;AACzC,WAAQ;;EAEX,CAE0B;CAE3B,gBAAgB,WAA+C;AAC7D,MAAI;AACF,UAAO,MAAM;AACX,QAAI,MAAM,WAAW,EACnB,OAAM,IAAI,SAAe,MAAM;AAC7B,eAAU;MACV;AAGJ,WAAO,MAAM,SAAS,GAAG;KACvB,MAAM,OAAO,MAAM,OAAO;AAC1B,SAAI,KAAK,SAAS,QAChB,OAAM,KAAK;cACF,KAAK,SAAS,QACvB,OAAM,KAAK;SAEX;;;YAIE;AACR,UAAO;AACP,OAAI,QAAS,UAAS;;;AAI1B,QAAO,UAAU;;;;;;ACjEnB,SAAgB,eAAe,MAAoC;CACjE,MAAM,SAAS,KAAK,OAAO,QAAQ,OAAO,GAAG;AAE7C,QAAO,OAAO,QAAQ;EACpB,MAAM,MAAM,IAAI,IAAI,IAAI,KAAK,mBAAmB;EAChD,MAAM,WAAW,GAAG,SAAS,IAAI,WAAW,IAAI;AAEhD,MAAI;GACF,MAAM,OAAO,MAAM,MAAM,UAAU;IACjC,QAAQ,IAAI;IACZ,SAAS,EAAE,QAAQ,OAAO;IAC3B,CAAC;GAEF,MAAM,OAAO,MAAM,KAAK,MAAM;GAC9B,MAAM,UAAkC,EAAE;AAC1C,QAAK,QAAQ,SAAS,OAAO,QAAQ;AACnC,YAAQ,OAAO;KACf;AAEF,UAAO;IAAE,QAAQ,KAAK;IAAQ;IAAS;IAAM;UACvC;AACN,UAAO;IACL,QAAQ;IACR,SAAS,EAAE,gBAAgB,cAAc;IACzC,MAAM,qCAAqC;IAC5C;;;;;AAMP,SAAgB,oBAAoB,MAAyC;CAC3E,MAAM,MAAM,KAAK;AAEjB,QAAO,OAAO,QAAmC;EAE/C,IAAI,WADQ,IAAI,IAAI,IAAI,KAAK,mBAAmB,CAC7B;AAEnB,MAAI,SAAS,SAAS,KAAK,CACzB,QAAO;GACL,QAAQ;GACR,SAAS,EAAE,gBAAgB,cAAc;GACzC,MAAM;GACP;AAIH,MAAI,SAAS,SAAS,IAAI,CACxB,aAAY;EAGd,MAAM,WAAW,KAAK,KAAK,SAAS;AACpC,MAAI;GACF,MAAM,UAAU,MAAM,SAAS,SAAS;GAExC,MAAM,cAAc,WADR,QAAQ,SAAS,KACU;AACvC,UAAO;IACL,QAAQ;IACR,SAAS,EAAE,gBAAgB,aAAa;IACxC,MAAM,QAAQ,UAAU;IACzB;UACK;AACN,UAAO;IACL,QAAQ;IACR,SAAS,EAAE,gBAAgB,cAAc;IACzC,MAAM;IACP;;;;;;;AC3EP,SAAgB,mBAAmB,SAAiB,UAAqC;CACvF,MAAM,cAAc,KAAK,SAAS,kBAAkB;CACpD,IAAI,UAA4B;AAChC,KAAI;AACF,YAAU,MAAM,mBAAmB,UAAU,CAAC;SACxC;EAEN,MAAM,aAAa,MAAM,UAAU,QAAQ,aAAa;AACtD,OAAI,aAAa,mBAAmB;AAClC,eAAW,OAAO;AAClB,cAAU,MAAM,mBAAmB,UAAU,CAAC;AAE9C,cAAU;;IAEZ;AACF,SAAO,EACL,QAAQ;AACN,cAAW,OAAO;AAClB,YAAS,OAAO;KAEnB;;AAEH,QAAO,EACL,QAAQ;AACN,WAAS,OAAO;IAEnB"}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@canmi/seam-server",
|
|
3
|
-
"version": "0.
|
|
3
|
+
"version": "0.4.3",
|
|
4
4
|
"files": [
|
|
5
5
|
"dist"
|
|
6
6
|
],
|
|
@@ -16,7 +16,7 @@
|
|
|
16
16
|
"test": "vitest run"
|
|
17
17
|
},
|
|
18
18
|
"dependencies": {
|
|
19
|
-
"@canmi/seam-
|
|
19
|
+
"@canmi/seam-engine": "workspace:*",
|
|
20
20
|
"jtd": "^0.1.1"
|
|
21
21
|
},
|
|
22
22
|
"devDependencies": {
|