stratal 0.0.16 → 0.0.18
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/README.md +4 -0
- package/dist/bin/cloudflare-workers-loader.mjs +33 -1
- package/dist/bin/cloudflare-workers-loader.mjs.map +1 -1
- package/dist/bin/quarry.mjs +183 -55
- package/dist/bin/quarry.mjs.map +1 -1
- package/dist/cache/index.d.mts +2 -2
- package/dist/cache/index.d.mts.map +1 -1
- package/dist/cache/index.mjs +3 -11
- package/dist/cache/index.mjs.map +1 -1
- package/dist/{colors-DJaRDXoS.mjs → colors-BTAnQRGU.mjs} +1 -1
- package/dist/{colors-DJaRDXoS.mjs.map → colors-BTAnQRGU.mjs.map} +1 -1
- package/dist/{command-B-QH-Vu3.d.mts → command-B1YuV-UZ.d.mts} +2 -2
- package/dist/{command-B-QH-Vu3.d.mts.map → command-B1YuV-UZ.d.mts.map} +1 -1
- package/dist/{command-BvCOD6df.mjs → command-DjGqCYHv.mjs} +7 -4
- package/dist/command-DjGqCYHv.mjs.map +1 -0
- package/dist/config/index.d.mts +2 -2
- package/dist/config/index.mjs +12 -20
- package/dist/config/index.mjs.map +1 -1
- package/dist/consumer-registry-BkuHXR_u.d.mts +142 -0
- package/dist/consumer-registry-BkuHXR_u.d.mts.map +1 -0
- package/dist/cron/index.d.mts +3 -116
- package/dist/cron/index.d.mts.map +1 -1
- package/dist/cron/index.mjs +1 -4
- package/dist/{cron-manager-DR7fiG6o.mjs → cron-manager-1KnZvojs.mjs} +3 -3
- package/dist/{cron-manager-DR7fiG6o.mjs.map → cron-manager-1KnZvojs.mjs.map} +1 -1
- package/dist/cron-manager-BnEZquBL.d.mts +117 -0
- package/dist/cron-manager-BnEZquBL.d.mts.map +1 -0
- package/dist/di/index.d.mts +2 -2
- package/dist/di/index.mjs +3 -4
- package/dist/email/index.d.mts +3 -3
- package/dist/email/index.mjs +8 -17
- package/dist/email/index.mjs.map +1 -1
- package/dist/{en-DaewN8hc.mjs → en-3QnZwP-u.mjs} +10 -1
- package/dist/en-3QnZwP-u.mjs.map +1 -0
- package/dist/errors/index.d.mts +2 -2
- package/dist/errors/index.mjs +2 -4
- package/dist/errors--RBIvDXr.mjs +1560 -0
- package/dist/errors--RBIvDXr.mjs.map +1 -0
- package/dist/{errors-H3TZnVeX.mjs → errors-B7hCnXgB.mjs} +2 -2
- package/dist/{errors-H3TZnVeX.mjs.map → errors-B7hCnXgB.mjs.map} +1 -1
- package/dist/events/index.d.mts +2 -2
- package/dist/events/index.mjs +1 -3
- package/dist/{events-CXl-o1Ad.mjs → events-UTJliZhl.mjs} +2 -3
- package/dist/{events-CXl-o1Ad.mjs.map → events-UTJliZhl.mjs.map} +1 -1
- package/dist/{gateway-context-BkZ4UKaX.mjs → gateway-context-BdBFoQd8.mjs} +66 -10
- package/dist/gateway-context-BdBFoQd8.mjs.map +1 -0
- package/dist/guards/index.d.mts +3 -3
- package/dist/guards/index.d.mts.map +1 -1
- package/dist/guards/index.mjs +1 -1
- package/dist/{guards-DUk_Kzst.mjs → guards-MtDgcHnF.mjs} +1 -1
- package/dist/{guards-DUk_Kzst.mjs.map → guards-MtDgcHnF.mjs.map} +1 -1
- package/dist/i18n/index.d.mts +3 -3
- package/dist/i18n/index.mjs +3 -16
- package/dist/i18n/messages/en/index.d.mts +1 -1
- package/dist/i18n/messages/en/index.mjs +1 -1
- package/dist/i18n/utils/index.d.mts +30 -0
- package/dist/i18n/utils/index.d.mts.map +1 -0
- package/dist/i18n/utils/index.mjs +2 -0
- package/dist/i18n/validation/index.d.mts +1 -1
- package/dist/i18n/validation/index.mjs +1 -1
- package/dist/i18n.module-BpLLLCTg.mjs +2462 -0
- package/dist/i18n.module-BpLLLCTg.mjs.map +1 -0
- package/dist/{index-D_w_Rmtd.d.mts → index-BDh9J2KD.d.mts} +10 -1
- package/dist/{index-D_w_Rmtd.d.mts.map → index-BDh9J2KD.d.mts.map} +1 -1
- package/dist/{index-Dp6A5ywM.d.mts → index-BR23zDMy.d.mts} +1 -1
- package/dist/{index-Dp6A5ywM.d.mts.map → index-BR23zDMy.d.mts.map} +1 -1
- package/dist/index-BrmS34sa.d.mts +4287 -0
- package/dist/index-BrmS34sa.d.mts.map +1 -0
- package/dist/{index-D9iYu2Yc.d.mts → index-DPxmo6AY.d.mts} +5 -144
- package/dist/index-DPxmo6AY.d.mts.map +1 -0
- package/dist/{index-DVhdhLvE.d.mts → index-Dfpd_ypO.d.mts} +38 -9
- package/dist/index-Dfpd_ypO.d.mts.map +1 -0
- package/dist/index.d.mts +4 -3
- package/dist/index.d.mts.map +1 -1
- package/dist/index.mjs +1 -20
- package/dist/{is-command-BfCgWAcQ.mjs → is-command-PvULqiTa.mjs} +2 -2
- package/dist/{is-command-BfCgWAcQ.mjs.map → is-command-PvULqiTa.mjs.map} +1 -1
- package/dist/{is-seeder-CebjZCDn.mjs → is-seeder-BN9Ej1r7.mjs} +1 -1
- package/dist/{is-seeder-CebjZCDn.mjs.map → is-seeder-BN9Ej1r7.mjs.map} +1 -1
- package/dist/logger/index.d.mts +1 -1
- package/dist/logger/index.mjs +1 -2
- package/dist/{logger-BR1-s1Um.mjs → logger-c0ftIK4G.mjs} +170 -4
- package/dist/logger-c0ftIK4G.mjs.map +1 -0
- package/dist/module/index.d.mts +3 -119
- package/dist/module/index.d.mts.map +1 -1
- package/dist/module/index.mjs +1 -11
- package/dist/module-C3YZ-kZN.mjs +719 -0
- package/dist/module-C3YZ-kZN.mjs.map +1 -0
- package/dist/openapi/index.d.mts +54 -54
- package/dist/openapi/index.d.mts.map +1 -1
- package/dist/openapi/index.mjs +3 -16
- package/dist/openapi-tools.service-B77QXD56.mjs +197 -0
- package/dist/openapi-tools.service-B77QXD56.mjs.map +1 -0
- package/dist/openapi.service-6yj0BUY4.d.mts +50 -0
- package/dist/openapi.service-6yj0BUY4.d.mts.map +1 -0
- package/dist/quarry/index.d.mts +124 -29
- package/dist/quarry/index.d.mts.map +1 -1
- package/dist/quarry/index.mjs +5 -7
- package/dist/quarry-registry-CQCIlYTO.mjs +686 -0
- package/dist/quarry-registry-CQCIlYTO.mjs.map +1 -0
- package/dist/queue/index.d.mts +2 -1
- package/dist/queue/index.mjs +3 -14
- package/dist/queue/index.mjs.map +1 -1
- package/dist/{queue.module-BZvmeAMj.mjs → queue.module-DIjD6nr-.mjs} +39 -42
- package/dist/queue.module-DIjD6nr-.mjs.map +1 -0
- package/dist/{resend.provider-BCCACQAU.mjs → resend.provider-Bvw36rQy.mjs} +1 -4
- package/dist/{resend.provider-BCCACQAU.mjs.map → resend.provider-Bvw36rQy.mjs.map} +1 -1
- package/dist/router/index.d.mts +2 -2
- package/dist/router/index.mjs +5 -16
- package/dist/{s3-storage.provider-BLlzQYiJ.mjs → s3-storage.provider-BAhHDMI3.mjs} +16 -9
- package/dist/s3-storage.provider-BAhHDMI3.mjs.map +1 -0
- package/dist/seeder/index.d.mts +3 -4
- package/dist/seeder/index.d.mts.map +1 -1
- package/dist/seeder/index.mjs +2 -7
- package/dist/{seeder-Cupi5jl-.mjs → seeder-D7VXULXB.mjs} +20 -17
- package/dist/seeder-D7VXULXB.mjs.map +1 -0
- package/dist/setup-BRIN-iYT.mjs +37 -0
- package/dist/setup-BRIN-iYT.mjs.map +1 -0
- package/dist/{smtp.provider-B8XtOcHU.mjs → smtp.provider-CAwpvzvD.mjs} +1 -4
- package/dist/{smtp.provider-B8XtOcHU.mjs.map → smtp.provider-CAwpvzvD.mjs.map} +1 -1
- package/dist/storage/index.d.mts +2 -195
- package/dist/storage/index.d.mts.map +1 -1
- package/dist/storage/index.mjs +2 -14
- package/dist/storage/providers/index.d.mts +273 -0
- package/dist/storage/providers/index.d.mts.map +1 -0
- package/dist/storage/providers/index.mjs +2 -0
- package/dist/{storage-By_ow2o_.mjs → storage-CJ-QOwNv.mjs} +8 -9
- package/dist/storage-CJ-QOwNv.mjs.map +1 -0
- package/dist/storage-provider.interface-YRtyYBxV.d.mts +203 -0
- package/dist/storage-provider.interface-YRtyYBxV.d.mts.map +1 -0
- package/dist/stratal-B7G4i9-N.mjs +502 -0
- package/dist/stratal-B7G4i9-N.mjs.map +1 -0
- package/dist/{types-DahElfUw.d.mts → types-CN0zONAZ.d.mts} +2 -2
- package/dist/types-CN0zONAZ.d.mts.map +1 -0
- package/dist/{usage-generator-C9hWziY4.mjs → usage-generator-Cl1HPlUp.mjs} +2 -2
- package/dist/{usage-generator-C9hWziY4.mjs.map → usage-generator-Cl1HPlUp.mjs.map} +1 -1
- package/dist/{validation-Bh875Lyg.mjs → validation-B4bePOa_.mjs} +5 -5
- package/dist/{validation-Bh875Lyg.mjs.map → validation-B4bePOa_.mjs.map} +1 -1
- package/dist/websocket/index.d.mts +2 -2
- package/dist/websocket/index.d.mts.map +1 -1
- package/dist/websocket/index.mjs +1 -5
- package/dist/workers/index.d.mts +1 -1
- package/dist/workers/index.d.mts.map +1 -1
- package/dist/workers/index.mjs +2 -20
- package/dist/workers/index.mjs.map +1 -1
- package/package.json +39 -31
- package/dist/application-zG8b-pol.d.mts +0 -116
- package/dist/application-zG8b-pol.d.mts.map +0 -1
- package/dist/command-BvCOD6df.mjs.map +0 -1
- package/dist/decorate-D5j-d9_z.mjs +0 -171
- package/dist/decorate-D5j-d9_z.mjs.map +0 -1
- package/dist/en-DaewN8hc.mjs.map +0 -1
- package/dist/errors-CtCi1wn6.mjs +0 -707
- package/dist/errors-CtCi1wn6.mjs.map +0 -1
- package/dist/gateway-context-BkZ4UKaX.mjs.map +0 -1
- package/dist/i18n.module-W8OJxg3d.mjs +0 -1791
- package/dist/i18n.module-W8OJxg3d.mjs.map +0 -1
- package/dist/index-BJWm863C.d.mts +0 -2616
- package/dist/index-BJWm863C.d.mts.map +0 -1
- package/dist/index-D9iYu2Yc.d.mts.map +0 -1
- package/dist/index-DVhdhLvE.d.mts.map +0 -1
- package/dist/logger-BR1-s1Um.mjs.map +0 -1
- package/dist/middleware/index.d.mts +0 -2
- package/dist/middleware/index.mjs +0 -6
- package/dist/middleware-C0Ebzswy.mjs +0 -362
- package/dist/middleware-C0Ebzswy.mjs.map +0 -1
- package/dist/module-BgdxxzBe.mjs +0 -370
- package/dist/module-BgdxxzBe.mjs.map +0 -1
- package/dist/quarry-registry-DCwqVcRp.mjs +0 -310
- package/dist/quarry-registry-DCwqVcRp.mjs.map +0 -1
- package/dist/queue.module-BZvmeAMj.mjs.map +0 -1
- package/dist/router-context-BEJe9HEB.mjs +0 -264
- package/dist/router-context-BEJe9HEB.mjs.map +0 -1
- package/dist/s3-storage.provider-BLlzQYiJ.mjs.map +0 -1
- package/dist/seeder-Cupi5jl-.mjs.map +0 -1
- package/dist/storage-By_ow2o_.mjs.map +0 -1
- package/dist/stratal-CE0iTz4f.mjs +0 -305
- package/dist/stratal-CE0iTz4f.mjs.map +0 -1
- package/dist/types-CLhOhYsQ.d.mts +0 -64
- package/dist/types-CLhOhYsQ.d.mts.map +0 -1
- package/dist/types-DahElfUw.d.mts.map +0 -1
|
@@ -0,0 +1,502 @@
|
|
|
1
|
+
import { A as Scope, D as runWithContainer, H as ApplicationError, V as ROUTER_TOKENS, g as DefaultExceptionHandler, i as createCronExceptionContext, j as Container, o as createQueueExceptionContext, r as createCliExceptionContext, t as StratalNotInitializedError } from "./errors--RBIvDXr.mjs";
|
|
2
|
+
import { a as __decorate, f as DI_TOKENS, i as LoggerService, l as LogLevel, n as PrettyFormatter, o as __decorateParam, r as JsonFormatter, s as __decorateMetadata, t as ConsoleTransport, u as LOGGER_TOKENS } from "./logger-c0ftIK4G.mjs";
|
|
3
|
+
import { a as getGroups, i as getGlobalMiddleware, r as getDefaultEntry, t as ModuleRegistry } from "./module-C3YZ-kZN.mjs";
|
|
4
|
+
import { r as getListenerHandlers, t as EventRegistry } from "./events-UTJliZhl.mjs";
|
|
5
|
+
import { t as Command } from "./command-DjGqCYHv.mjs";
|
|
6
|
+
import { CacheModule } from "./cache/index.mjs";
|
|
7
|
+
import { t as CronManager } from "./cron-manager-1KnZvojs.mjs";
|
|
8
|
+
import { R as OpenAPIModule, c as RouteRegistry, d as HonoApp, f as RouteRegistrationService, i as Uri, l as VersioningService, t as I18nModule, u as LocalePathService } from "./i18n.module-BpLLLCTg.mjs";
|
|
9
|
+
import { a as QueueListCommand, c as HelpCommand, d as ApiCommand, i as RouteListCommand, o as McpToolsCommand, r as ScheduleListCommand, s as McpServeCommand, t as QuarryRegistry, u as EventListCommand } from "./quarry-registry-CQCIlYTO.mjs";
|
|
10
|
+
import { t as QueueModule } from "./queue.module-DIjD6nr-.mjs";
|
|
11
|
+
import { i as SeederRegistry, n as DbSeedListCommand, r as SEEDER_TOKENS, t as DbSeedCommand } from "./seeder-D7VXULXB.mjs";
|
|
12
|
+
import { container, inject, injectable } from "tsyringe";
|
|
13
|
+
import "reflect-metadata";
|
|
14
|
+
import { writeFileSync } from "node:fs";
|
|
15
|
+
import { resolve } from "node:path";
|
|
16
|
+
//#region src/quarry/commands/route-types.command.ts
|
|
17
|
+
let RouteTypesCommand = class RouteTypesCommand extends Command {
|
|
18
|
+
static command = "route:types {--output=src/stratal.d.ts : Output file path}";
|
|
19
|
+
static description = "Generate TypeScript types for named routes";
|
|
20
|
+
constructor(registry, localePathService) {
|
|
21
|
+
super();
|
|
22
|
+
this.registry = registry;
|
|
23
|
+
this.localePathService = localePathService;
|
|
24
|
+
}
|
|
25
|
+
handle() {
|
|
26
|
+
const outputPath = resolve(this.string("output") || "src/stratal.d.ts");
|
|
27
|
+
const namedRoutes = this.registry.named();
|
|
28
|
+
if (namedRoutes.length === 0) {
|
|
29
|
+
this.warn("No named routes found. Add name to your @Route() or @Get()/@Post() decorators.");
|
|
30
|
+
return 0;
|
|
31
|
+
}
|
|
32
|
+
writeFileSync(outputPath, this.generateDeclaration(namedRoutes), "utf-8");
|
|
33
|
+
this.info(`Generated route types for ${namedRoutes.length} named routes → ${outputPath}`);
|
|
34
|
+
}
|
|
35
|
+
/**
|
|
36
|
+
* Generate the StratalRouteMap declaration content.
|
|
37
|
+
*/
|
|
38
|
+
generateDeclaration(routes) {
|
|
39
|
+
const localeConfig = this.localePathService.enabled ? this.localePathService.localePathConfig : null;
|
|
40
|
+
const localeType = localeConfig ? localeConfig.allLocales.map((l) => `'${l}'`).join(" | ") : null;
|
|
41
|
+
const localeOptional = localeConfig ? this.localePathService.prefixDefaultLocale !== true : false;
|
|
42
|
+
const entries = routes.filter((r) => r.name !== void 0).sort((a, b) => a.name.localeCompare(b.name)).map((route) => {
|
|
43
|
+
const paramEntries = [...route.paramNames.map((p) => `${p}: string`), ...route.domainParamNames.map((p) => `${p}: string`)];
|
|
44
|
+
if (localeType && route.localePaths?.length) {
|
|
45
|
+
const optionalMarker = localeOptional ? "?" : "";
|
|
46
|
+
paramEntries.push(`locale${optionalMarker}: StratalLocale`);
|
|
47
|
+
}
|
|
48
|
+
const paramsType = paramEntries.length === 0 ? "never" : `{ ${paramEntries.join("; ")} }`;
|
|
49
|
+
return ` '${route.name}': { params: ${paramsType} }`;
|
|
50
|
+
}).join("\n");
|
|
51
|
+
const lines = ["// Auto-generated by `quarry route:types` — do not edit manually", "declare module 'stratal/router' {"];
|
|
52
|
+
if (localeType) {
|
|
53
|
+
lines.push(` type StratalLocale = ${localeType}`);
|
|
54
|
+
lines.push("");
|
|
55
|
+
}
|
|
56
|
+
lines.push(" interface StratalRouteMap {", entries, " }", "}");
|
|
57
|
+
return [
|
|
58
|
+
...lines,
|
|
59
|
+
"",
|
|
60
|
+
"export {}",
|
|
61
|
+
""
|
|
62
|
+
].join("\n");
|
|
63
|
+
}
|
|
64
|
+
};
|
|
65
|
+
RouteTypesCommand = __decorate([
|
|
66
|
+
__decorateParam(0, inject(ROUTER_TOKENS.RouteRegistry)),
|
|
67
|
+
__decorateParam(1, inject(ROUTER_TOKENS.LocalePathService)),
|
|
68
|
+
__decorateMetadata("design:paramtypes", [Object, Object])
|
|
69
|
+
], RouteTypesCommand);
|
|
70
|
+
//#endregion
|
|
71
|
+
//#region src/router/router-resolver.ts
|
|
72
|
+
/**
|
|
73
|
+
* Internal resolver that computes the effective Router config for each controller.
|
|
74
|
+
*
|
|
75
|
+
* Inheritance rules:
|
|
76
|
+
* - `middleware`: parent middleware runs first, then child (concatenated)
|
|
77
|
+
* - `prefix`: concatenated (parent + child)
|
|
78
|
+
* - `name`: concatenated (parent + child)
|
|
79
|
+
* - `domain`: child overrides parent
|
|
80
|
+
* - `version`: child overrides parent
|
|
81
|
+
* - `hideFromDocs`: child overrides parent
|
|
82
|
+
*
|
|
83
|
+
* @internal — not exported from stratal/router
|
|
84
|
+
*/
|
|
85
|
+
var RouterResolver = class {
|
|
86
|
+
routers;
|
|
87
|
+
constructor(routers) {
|
|
88
|
+
this.routers = routers;
|
|
89
|
+
}
|
|
90
|
+
/**
|
|
91
|
+
* Resolve the effective config for a given controller class.
|
|
92
|
+
* Searches through all module routers to find the one owning this controller.
|
|
93
|
+
*/
|
|
94
|
+
resolveForController(controller) {
|
|
95
|
+
for (const { router, controllers: moduleControllers } of this.routers) {
|
|
96
|
+
if (!moduleControllers.includes(controller)) continue;
|
|
97
|
+
for (const group of router[getGroups]()) if (group.controllers?.includes(controller)) return this.mergeEntries(router[getDefaultEntry](), group);
|
|
98
|
+
if (!new Set(router[getGroups]().flatMap((g) => g.controllers ?? [])).has(controller)) return this.entryToConfig(router[getDefaultEntry]());
|
|
99
|
+
}
|
|
100
|
+
return { middleware: [] };
|
|
101
|
+
}
|
|
102
|
+
/**
|
|
103
|
+
* Collect all global middleware registered via `router.use()` across all modules.
|
|
104
|
+
*/
|
|
105
|
+
getGlobalMiddleware() {
|
|
106
|
+
const global = [];
|
|
107
|
+
for (const { router } of this.routers) global.push(...router[getGlobalMiddleware]());
|
|
108
|
+
return global;
|
|
109
|
+
}
|
|
110
|
+
/**
|
|
111
|
+
* Merge parent default entry with child group entry following inheritance rules.
|
|
112
|
+
*/
|
|
113
|
+
mergeEntries(parent, child) {
|
|
114
|
+
return {
|
|
115
|
+
prefix: this.concatPrefixes(parent.prefix, child.prefix),
|
|
116
|
+
domain: child.domain ?? parent.domain,
|
|
117
|
+
name: this.concatNames(parent.name, child.name),
|
|
118
|
+
middleware: [...parent.middleware, ...child.middleware],
|
|
119
|
+
version: child.version ?? parent.version,
|
|
120
|
+
hideFromDocs: child.hideFromDocs ?? parent.hideFromDocs,
|
|
121
|
+
params: this.mergeParams(parent.params, child.params)
|
|
122
|
+
};
|
|
123
|
+
}
|
|
124
|
+
entryToConfig(entry) {
|
|
125
|
+
return {
|
|
126
|
+
prefix: entry.prefix,
|
|
127
|
+
domain: entry.domain,
|
|
128
|
+
name: entry.name,
|
|
129
|
+
middleware: [...entry.middleware],
|
|
130
|
+
version: entry.version,
|
|
131
|
+
hideFromDocs: entry.hideFromDocs,
|
|
132
|
+
params: entry.params
|
|
133
|
+
};
|
|
134
|
+
}
|
|
135
|
+
mergeParams(parent, child) {
|
|
136
|
+
if (!parent && !child) return void 0;
|
|
137
|
+
if (!parent) return child;
|
|
138
|
+
if (!child) return parent;
|
|
139
|
+
return parent.extend(child.shape);
|
|
140
|
+
}
|
|
141
|
+
concatPrefixes(parent, child) {
|
|
142
|
+
if (!parent && !child) return void 0;
|
|
143
|
+
if (!parent) return child;
|
|
144
|
+
if (!child) return parent;
|
|
145
|
+
return `${parent.endsWith("/") ? parent.slice(0, -1) : parent}${child.startsWith("/") ? child : `/${child}`}`;
|
|
146
|
+
}
|
|
147
|
+
concatNames(parent, child) {
|
|
148
|
+
if (!parent && !child) return void 0;
|
|
149
|
+
if (!parent) return child;
|
|
150
|
+
if (!child) return parent;
|
|
151
|
+
return `${parent}${child}`;
|
|
152
|
+
}
|
|
153
|
+
};
|
|
154
|
+
//#endregion
|
|
155
|
+
//#region src/application.ts
|
|
156
|
+
/**
|
|
157
|
+
* Application
|
|
158
|
+
*
|
|
159
|
+
* Main application class managing the two-tier container hierarchy:
|
|
160
|
+
* - Global Container: All services (singletons via tsyringe native)
|
|
161
|
+
* - Request Container: Child of global, context-enriched instances per request
|
|
162
|
+
*
|
|
163
|
+
* @example
|
|
164
|
+
* ```typescript
|
|
165
|
+
* const app = new Application({ module: AppModule, env, ctx })
|
|
166
|
+
* await app.initialize()
|
|
167
|
+
*
|
|
168
|
+
* // Access container via getter
|
|
169
|
+
* const service = app.container.resolve(MY_TOKEN)
|
|
170
|
+
*
|
|
171
|
+
* // Handle HTTP request (via HonoApp)
|
|
172
|
+
* // Handle queue batch
|
|
173
|
+
* await app.handleQueue(batch, 'my-queue')
|
|
174
|
+
* ```
|
|
175
|
+
*/
|
|
176
|
+
var Application = class {
|
|
177
|
+
/**
|
|
178
|
+
* Unified Container - manages all DI operations
|
|
179
|
+
*/
|
|
180
|
+
_container;
|
|
181
|
+
honoApp;
|
|
182
|
+
moduleRegistry;
|
|
183
|
+
consumerRegistry;
|
|
184
|
+
cronManager;
|
|
185
|
+
quarry;
|
|
186
|
+
initialized = false;
|
|
187
|
+
env;
|
|
188
|
+
appConfig;
|
|
189
|
+
constructor({ env, ctx, ...config }) {
|
|
190
|
+
this.env = env;
|
|
191
|
+
this.appConfig = config;
|
|
192
|
+
ApplicationError.captureStackTraces = env.ENVIRONMENT !== "production";
|
|
193
|
+
this._container = new Container({ container: container.createChildContainer() });
|
|
194
|
+
this._container.registerValue(DI_TOKENS.Application, this);
|
|
195
|
+
this._container.registerValue(DI_TOKENS.CloudflareEnv, env);
|
|
196
|
+
this._container.registerValue(DI_TOKENS.ExecutionContext, ctx);
|
|
197
|
+
this.registerLoggerService();
|
|
198
|
+
this.registerCoreServices();
|
|
199
|
+
const logger = this._container.resolve(LOGGER_TOKENS.LoggerService);
|
|
200
|
+
this.moduleRegistry = new ModuleRegistry(this._container, logger);
|
|
201
|
+
this._container.registerValue(DI_TOKENS.ModuleRegistry, this.moduleRegistry);
|
|
202
|
+
}
|
|
203
|
+
/**
|
|
204
|
+
* Get the Container instance
|
|
205
|
+
*/
|
|
206
|
+
get container() {
|
|
207
|
+
return this._container;
|
|
208
|
+
}
|
|
209
|
+
/**
|
|
210
|
+
* Get the HonoApp instance
|
|
211
|
+
*/
|
|
212
|
+
get hono() {
|
|
213
|
+
return this.honoApp;
|
|
214
|
+
}
|
|
215
|
+
/**
|
|
216
|
+
* Get the application configuration
|
|
217
|
+
*/
|
|
218
|
+
get config() {
|
|
219
|
+
return this.appConfig;
|
|
220
|
+
}
|
|
221
|
+
async initialize() {
|
|
222
|
+
if (this.initialized) return;
|
|
223
|
+
await runWithContainer(this._container, () => this.initializeInternal());
|
|
224
|
+
}
|
|
225
|
+
async initializeInternal() {
|
|
226
|
+
this.moduleRegistry.registerAll([
|
|
227
|
+
I18nModule,
|
|
228
|
+
OpenAPIModule,
|
|
229
|
+
QueueModule,
|
|
230
|
+
CacheModule
|
|
231
|
+
]);
|
|
232
|
+
this.moduleRegistry.register(this.appConfig.module);
|
|
233
|
+
await this.moduleRegistry.initialize();
|
|
234
|
+
this.initializeExceptionHandler();
|
|
235
|
+
this.consumerRegistry = this._container.resolve(DI_TOKENS.ConsumerRegistry);
|
|
236
|
+
this.cronManager = this._container.resolve(DI_TOKENS.Cron);
|
|
237
|
+
this.quarry = this._container.resolve(DI_TOKENS.Quarry);
|
|
238
|
+
this.registerRoutingServices();
|
|
239
|
+
this.honoApp = this._container.resolve(ROUTER_TOKENS.HonoApp);
|
|
240
|
+
await this.honoApp.configure();
|
|
241
|
+
this.registerQueueConsumers();
|
|
242
|
+
this.registerCronJobs();
|
|
243
|
+
this.registerEventListeners();
|
|
244
|
+
this.registerSeeders();
|
|
245
|
+
this.registerCommands();
|
|
246
|
+
this.initialized = true;
|
|
247
|
+
}
|
|
248
|
+
/**
|
|
249
|
+
* Register routing services as singletons in the container.
|
|
250
|
+
* Called after module initialization so I18N_TOKENS.Options is available.
|
|
251
|
+
*/
|
|
252
|
+
registerRoutingServices() {
|
|
253
|
+
this._container.register(ROUTER_TOKENS.VersioningService, VersioningService, Scope.Singleton);
|
|
254
|
+
this._container.register(ROUTER_TOKENS.HonoApp, HonoApp, Scope.Singleton);
|
|
255
|
+
this._container.register(ROUTER_TOKENS.LocalePathService, LocalePathService, Scope.Singleton);
|
|
256
|
+
this._container.register(ROUTER_TOKENS.RouteRegistry, RouteRegistry, Scope.Singleton);
|
|
257
|
+
this._container.register(ROUTER_TOKENS.Uri, Uri, Scope.Request);
|
|
258
|
+
const routerConfigs = this.moduleRegistry.getAllRouterConfigs();
|
|
259
|
+
const routerResolver = routerConfigs.length > 0 ? new RouterResolver(routerConfigs) : null;
|
|
260
|
+
this._container.registerValue(ROUTER_TOKENS.RouterResolver, routerResolver);
|
|
261
|
+
this._container.register(RouteRegistrationService, RouteRegistrationService);
|
|
262
|
+
}
|
|
263
|
+
/**
|
|
264
|
+
* Resolve a service from the container
|
|
265
|
+
*/
|
|
266
|
+
resolve(token) {
|
|
267
|
+
try {
|
|
268
|
+
return this._container.resolve(token);
|
|
269
|
+
} catch (error) {
|
|
270
|
+
const handler = this._container.resolve(DI_TOKENS.ExceptionHandler);
|
|
271
|
+
const ctx = createCliExceptionContext("resolve");
|
|
272
|
+
handler.handle(error, ctx);
|
|
273
|
+
throw error;
|
|
274
|
+
}
|
|
275
|
+
}
|
|
276
|
+
/**
|
|
277
|
+
* Handle queue batch processing
|
|
278
|
+
*/
|
|
279
|
+
async handleQueue(batch, queueName) {
|
|
280
|
+
const locale = (batch.messages[0]?.body)?.metadata?.locale ?? "en";
|
|
281
|
+
const mockRouterContext = this.createMockRouterContext(locale);
|
|
282
|
+
await this._container.runInRequestScope(mockRouterContext, async (requestContainer) => {
|
|
283
|
+
try {
|
|
284
|
+
await requestContainer.resolve(DI_TOKENS.Queue).processBatch(queueName, batch);
|
|
285
|
+
} catch (error) {
|
|
286
|
+
await requestContainer.resolve(DI_TOKENS.ExceptionHandler).handle(error, createQueueExceptionContext(queueName));
|
|
287
|
+
throw error;
|
|
288
|
+
}
|
|
289
|
+
});
|
|
290
|
+
}
|
|
291
|
+
/**
|
|
292
|
+
* Handle scheduled cron trigger
|
|
293
|
+
*/
|
|
294
|
+
async handleScheduled(controller) {
|
|
295
|
+
const mockRouterContext = this.createMockRouterContext("en");
|
|
296
|
+
await this._container.runInRequestScope(mockRouterContext, async (requestContainer) => {
|
|
297
|
+
try {
|
|
298
|
+
await this.cronManager.executeScheduled(controller);
|
|
299
|
+
} catch (error) {
|
|
300
|
+
await requestContainer.resolve(DI_TOKENS.ExceptionHandler).handle(error, createCronExceptionContext());
|
|
301
|
+
throw error;
|
|
302
|
+
}
|
|
303
|
+
});
|
|
304
|
+
}
|
|
305
|
+
/**
|
|
306
|
+
* Create mock RouterContext for queue/cron/seeder processing
|
|
307
|
+
*/
|
|
308
|
+
createMockRouterContext(locale = "en") {
|
|
309
|
+
return {
|
|
310
|
+
getLocale: () => locale,
|
|
311
|
+
setLocale: () => {},
|
|
312
|
+
getContainer: () => this._container
|
|
313
|
+
};
|
|
314
|
+
}
|
|
315
|
+
async shutdown() {
|
|
316
|
+
if (!this.initialized) return;
|
|
317
|
+
this.initialized = false;
|
|
318
|
+
await this.moduleRegistry.shutdown();
|
|
319
|
+
this._container.resolve(LOGGER_TOKENS.LoggerService).info("Disposing container...");
|
|
320
|
+
await this._container.dispose();
|
|
321
|
+
}
|
|
322
|
+
/**
|
|
323
|
+
* Execute a command by name in a request-scoped container.
|
|
324
|
+
*/
|
|
325
|
+
async handleCommand(name, input) {
|
|
326
|
+
const mockContext = this.createMockRouterContext("en");
|
|
327
|
+
return this._container.runInRequestScope(mockContext, async () => {
|
|
328
|
+
return this.quarry.call(name, input);
|
|
329
|
+
});
|
|
330
|
+
}
|
|
331
|
+
registerCommands() {
|
|
332
|
+
const builtinCommands = [
|
|
333
|
+
HelpCommand,
|
|
334
|
+
DbSeedCommand,
|
|
335
|
+
DbSeedListCommand,
|
|
336
|
+
RouteListCommand,
|
|
337
|
+
RouteTypesCommand,
|
|
338
|
+
EventListCommand,
|
|
339
|
+
ScheduleListCommand,
|
|
340
|
+
QueueListCommand,
|
|
341
|
+
McpServeCommand,
|
|
342
|
+
McpToolsCommand,
|
|
343
|
+
ApiCommand
|
|
344
|
+
];
|
|
345
|
+
for (const Cmd of builtinCommands) {
|
|
346
|
+
injectable()(Cmd);
|
|
347
|
+
this._container.register(Cmd, Cmd, Scope.Singleton);
|
|
348
|
+
this.quarry.register(Cmd);
|
|
349
|
+
}
|
|
350
|
+
const commands = this.moduleRegistry.getAllCommands();
|
|
351
|
+
if (commands.length === 0) return;
|
|
352
|
+
for (const CommandClass of commands) this.quarry.register(CommandClass);
|
|
353
|
+
}
|
|
354
|
+
registerSeeders() {
|
|
355
|
+
const seeders = this.moduleRegistry.getAllSeeders();
|
|
356
|
+
if (seeders.length === 0) return;
|
|
357
|
+
const registry = this._container.resolve(SEEDER_TOKENS.SeederRegistry);
|
|
358
|
+
for (const SeederClass of seeders) registry.register(SeederClass);
|
|
359
|
+
}
|
|
360
|
+
registerQueueConsumers() {
|
|
361
|
+
for (const ConsumerClass of this.moduleRegistry.getAllConsumers()) {
|
|
362
|
+
const consumer = this._container.resolve(ConsumerClass);
|
|
363
|
+
this.consumerRegistry.register(consumer);
|
|
364
|
+
}
|
|
365
|
+
}
|
|
366
|
+
registerCronJobs() {
|
|
367
|
+
for (const JobClass of this.moduleRegistry.getAllJobs()) {
|
|
368
|
+
const job = this._container.resolve(JobClass);
|
|
369
|
+
this.cronManager.registerJob(job);
|
|
370
|
+
}
|
|
371
|
+
}
|
|
372
|
+
/**
|
|
373
|
+
* Auto-wire `@Listener()` classes with the EventRegistry.
|
|
374
|
+
*/
|
|
375
|
+
registerEventListeners() {
|
|
376
|
+
const listeners = this.moduleRegistry.getAllListeners();
|
|
377
|
+
if (listeners.length === 0) return;
|
|
378
|
+
const eventRegistry = this._container.resolve(DI_TOKENS.EventRegistry);
|
|
379
|
+
for (const ListenerClass of listeners) {
|
|
380
|
+
const instance = this._container.resolve(ListenerClass);
|
|
381
|
+
const handlers = getListenerHandlers(ListenerClass);
|
|
382
|
+
for (const { methodName, event, options } of handlers) eventRegistry.on(event, instance[methodName].bind(instance), options);
|
|
383
|
+
}
|
|
384
|
+
}
|
|
385
|
+
/**
|
|
386
|
+
* Register LoggerService and dependencies
|
|
387
|
+
*/
|
|
388
|
+
registerLoggerService() {
|
|
389
|
+
const logLevel = this.appConfig.logging?.level ?? LogLevel.INFO;
|
|
390
|
+
const formatter = this.appConfig.logging?.formatter ?? "json";
|
|
391
|
+
this._container.registerValue(LOGGER_TOKENS.LogLevelOptions, logLevel);
|
|
392
|
+
this._container.when(() => formatter === "pretty").use(LOGGER_TOKENS.Formatter).give(PrettyFormatter).otherwise(JsonFormatter);
|
|
393
|
+
this._container.registerSingleton(LOGGER_TOKENS.ConsoleTransport, ConsoleTransport);
|
|
394
|
+
this._container.registerFactory(LOGGER_TOKENS.Transports, (c) => [c.resolve(LOGGER_TOKENS.ConsoleTransport)]);
|
|
395
|
+
this._container.registerSingleton(LOGGER_TOKENS.LoggerService, LoggerService);
|
|
396
|
+
}
|
|
397
|
+
/**
|
|
398
|
+
* Register core services with explicit scope
|
|
399
|
+
*/
|
|
400
|
+
registerCoreServices() {
|
|
401
|
+
this._container.registerSingleton(DI_TOKENS.Cron, CronManager);
|
|
402
|
+
this._container.registerSingleton(DI_TOKENS.ExceptionHandler, this.appConfig.exceptionHandler ?? DefaultExceptionHandler);
|
|
403
|
+
this._container.registerSingleton(DI_TOKENS.EventRegistry, EventRegistry);
|
|
404
|
+
this._container.registerSingleton(DI_TOKENS.Quarry, QuarryRegistry);
|
|
405
|
+
this._container.registerValue(SEEDER_TOKENS.SeederRegistry, new SeederRegistry(this));
|
|
406
|
+
}
|
|
407
|
+
/**
|
|
408
|
+
* Initialize the ExceptionHandler: call register(), then module onException hooks.
|
|
409
|
+
*/
|
|
410
|
+
initializeExceptionHandler() {
|
|
411
|
+
const handler = this._container.resolve(DI_TOKENS.ExceptionHandler);
|
|
412
|
+
handler.register();
|
|
413
|
+
this.moduleRegistry.configureExceptionHandlers(handler);
|
|
414
|
+
}
|
|
415
|
+
};
|
|
416
|
+
//#endregion
|
|
417
|
+
//#region src/stratal.ts
|
|
418
|
+
/**
|
|
419
|
+
* Stratal — Hono-style entry point for Cloudflare Workers.
|
|
420
|
+
*
|
|
421
|
+
* Eagerly bootstraps the Application at construction time, dynamically
|
|
422
|
+
* importing `cloudflare:workers` for env and waitUntil.
|
|
423
|
+
*
|
|
424
|
+
* @example
|
|
425
|
+
* ```typescript
|
|
426
|
+
* import { Stratal } from 'stratal'
|
|
427
|
+
* import { AppModule } from './app.module'
|
|
428
|
+
*
|
|
429
|
+
* export default new Stratal({ module: AppModule })
|
|
430
|
+
* ```
|
|
431
|
+
*/
|
|
432
|
+
var Stratal = class Stratal {
|
|
433
|
+
app = null;
|
|
434
|
+
initPromise;
|
|
435
|
+
static _application = null;
|
|
436
|
+
static _generation = 0;
|
|
437
|
+
static _previousInstance = null;
|
|
438
|
+
constructor(config) {
|
|
439
|
+
this.fetch = this.fetch.bind(this);
|
|
440
|
+
this.queue = this.queue.bind(this);
|
|
441
|
+
this.scheduled = this.scheduled.bind(this);
|
|
442
|
+
const generation = ++Stratal._generation;
|
|
443
|
+
if (Stratal._previousInstance) Stratal._previousInstance.shutdown();
|
|
444
|
+
Stratal._previousInstance = this;
|
|
445
|
+
this.initPromise = this.prepareApp(config, generation);
|
|
446
|
+
Stratal._application = this.initPromise;
|
|
447
|
+
}
|
|
448
|
+
async fetch(request, env, ctx) {
|
|
449
|
+
return (await this.ensureReady()).hono.fetch(request, env, ctx);
|
|
450
|
+
}
|
|
451
|
+
async queue(batch) {
|
|
452
|
+
return (await this.ensureReady()).handleQueue(batch, batch.queue);
|
|
453
|
+
}
|
|
454
|
+
async scheduled(controller) {
|
|
455
|
+
return (await this.ensureReady()).handleScheduled(controller);
|
|
456
|
+
}
|
|
457
|
+
get hono() {
|
|
458
|
+
return this.initPromise.then((app) => app.hono);
|
|
459
|
+
}
|
|
460
|
+
async shutdown() {
|
|
461
|
+
try {
|
|
462
|
+
this.app = await this.initPromise;
|
|
463
|
+
} catch {}
|
|
464
|
+
if (this.app) {
|
|
465
|
+
await this.app.shutdown();
|
|
466
|
+
this.app = null;
|
|
467
|
+
}
|
|
468
|
+
}
|
|
469
|
+
/**
|
|
470
|
+
* @internal
|
|
471
|
+
* Resolves the Application instance from the static singleton.
|
|
472
|
+
* Used by worker base classes (DurableObject, Workflow, WorkerEntrypoint)
|
|
473
|
+
* to access the DI container without going through Cloudflare RPC.
|
|
474
|
+
*/
|
|
475
|
+
static resolveApplication() {
|
|
476
|
+
if (!Stratal._application) throw new StratalNotInitializedError();
|
|
477
|
+
return Stratal._application;
|
|
478
|
+
}
|
|
479
|
+
async ensureReady() {
|
|
480
|
+
this.app ??= await this.initPromise;
|
|
481
|
+
return this.app;
|
|
482
|
+
}
|
|
483
|
+
async prepareApp(config, generation) {
|
|
484
|
+
const { env, waitUntil } = await import("cloudflare:workers");
|
|
485
|
+
if (generation !== Stratal._generation) return new Promise(() => {});
|
|
486
|
+
const app = new Application({
|
|
487
|
+
...config,
|
|
488
|
+
env,
|
|
489
|
+
ctx: { waitUntil }
|
|
490
|
+
});
|
|
491
|
+
await app.initialize();
|
|
492
|
+
if (generation !== Stratal._generation) {
|
|
493
|
+
await app.shutdown();
|
|
494
|
+
return new Promise(() => {});
|
|
495
|
+
}
|
|
496
|
+
return app;
|
|
497
|
+
}
|
|
498
|
+
};
|
|
499
|
+
//#endregion
|
|
500
|
+
export { Application as n, Stratal as t };
|
|
501
|
+
|
|
502
|
+
//# sourceMappingURL=stratal-B7G4i9-N.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"stratal-B7G4i9-N.mjs","names":["internal.getGroups","internal.getDefaultEntry","internal.getGlobalMiddleware","tsyringeRootContainer"],"sources":["../src/quarry/commands/route-types.command.ts","../src/router/router-resolver.ts","../src/application.ts","../src/stratal.ts"],"sourcesContent":["import { writeFileSync } from 'node:fs'\nimport { resolve } from 'node:path'\nimport { inject } from 'tsyringe'\nimport type { RouteRegistry, RegisteredRoute } from '../../router/route-registry'\nimport { ROUTER_TOKENS } from '../../router/router.tokens'\nimport type { LocalePathService } from '../../router/services/locale-path.service'\nimport { Command } from '../command'\n\n/**\n * Generate TypeScript types for named routes.\n *\n * Outputs a `stratal.d.ts` file with `StratalRouteMap` augmentation\n * that provides autocomplete and type-safe params for `route()` and `ctx.route()`.\n *\n * @example\n * ```bash\n * quarry route:types # → src/stratal.d.ts\n * quarry route:types --output=types/routes.d.ts\n * ```\n */\nexport class RouteTypesCommand extends Command {\n static command = 'route:types {--output=src/stratal.d.ts : Output file path}'\n static description = 'Generate TypeScript types for named routes'\n\n constructor(\n @inject(ROUTER_TOKENS.RouteRegistry) private registry: RouteRegistry,\n @inject(ROUTER_TOKENS.LocalePathService) private localePathService: LocalePathService,\n ) {\n super()\n }\n\n handle(): number | undefined {\n const outputPath = resolve(this.string('output') || 'src/stratal.d.ts')\n const namedRoutes = this.registry.named()\n\n if (namedRoutes.length === 0) {\n this.warn('No named routes found. Add name to your @Route() or @Get()/@Post() decorators.')\n return 0\n }\n\n const content = this.generateDeclaration(namedRoutes)\n writeFileSync(outputPath, content, 'utf-8')\n this.info(`Generated route types for ${namedRoutes.length} named routes → ${outputPath}`)\n\n return undefined\n }\n\n /**\n * Generate the StratalRouteMap declaration content.\n */\n private generateDeclaration(routes: RegisteredRoute[]): string {\n const localeConfig = this.localePathService.enabled ? this.localePathService.localePathConfig : null\n const localeType = localeConfig\n ? localeConfig.allLocales.map(l => `'${l}'`).join(' | ')\n : null\n const localeOptional = localeConfig ? this.localePathService.prefixDefaultLocale !== true : false\n\n const entries = routes\n .filter((r): r is RegisteredRoute & { name: string } => r.name !== undefined)\n .sort((a, b) => a.name.localeCompare(b.name))\n .map(route => {\n const paramEntries = [\n ...route.paramNames.map(p => `${p}: string`),\n ...route.domainParamNames.map(p => `${p}: string`),\n ]\n\n if (localeType && route.localePaths?.length) {\n const optionalMarker = localeOptional ? '?' : ''\n paramEntries.push(`locale${optionalMarker}: StratalLocale`)\n }\n\n const paramsType = paramEntries.length === 0\n ? 'never'\n : `{ ${paramEntries.join('; ')} }`\n return ` '${route.name}': { params: ${paramsType} }`\n })\n .join('\\n')\n\n const lines = [\n '// Auto-generated by `quarry route:types` — do not edit manually',\n \"declare module 'stratal/router' {\",\n ]\n\n if (localeType) {\n lines.push(` type StratalLocale = ${localeType}`)\n lines.push('')\n }\n\n lines.push(\n ' interface StratalRouteMap {',\n entries,\n ' }',\n '}',\n )\n\n return [\n ...lines,\n '',\n 'export {}',\n '',\n ].join('\\n')\n }\n}\n","import type { ZodObject } from '../i18n/validation'\nimport type { Constructor } from '../types'\nimport type { Middleware } from './middleware.interface'\nimport type { Router, RouterEntry } from './router'\nimport * as internal from './router.internals'\n\n/**\n * Resolved configuration for a single controller.\n * Merges Router default entry, sub-group overrides, and inheritance rules.\n */\nexport interface ResolvedRouterConfig {\n prefix?: string\n domain?: string\n name?: string\n middleware: Constructor<Middleware>[]\n version?: string | string[]\n hideFromDocs?: boolean\n // eslint-disable-next-line @typescript-eslint/no-explicit-any -- ZodObject generics require any for flexible shape parameter\n params?: ZodObject<any>\n}\n\n/**\n * Internal resolver that computes the effective Router config for each controller.\n *\n * Inheritance rules:\n * - `middleware`: parent middleware runs first, then child (concatenated)\n * - `prefix`: concatenated (parent + child)\n * - `name`: concatenated (parent + child)\n * - `domain`: child overrides parent\n * - `version`: child overrides parent\n * - `hideFromDocs`: child overrides parent\n *\n * @internal — not exported from stratal/router\n */\nexport class RouterResolver {\n private readonly routers: { router: Router; controllers: Constructor[] }[]\n\n constructor(routers: { router: Router; controllers: Constructor[] }[]) {\n this.routers = routers\n }\n\n /**\n * Resolve the effective config for a given controller class.\n * Searches through all module routers to find the one owning this controller.\n */\n resolveForController(controller: Constructor): ResolvedRouterConfig {\n for (const { router, controllers: moduleControllers } of this.routers) {\n if (!moduleControllers.includes(controller)) continue\n\n // Check if controller is in a sub-group\n for (const group of router[internal.getGroups]()) {\n if (group.controllers?.includes(controller)) {\n return this.mergeEntries(router[internal.getDefaultEntry](), group)\n }\n }\n\n // Controller is in the default scope (not in any sub-group)\n // But only if it's not claimed by any sub-group in this module\n const groupedControllers = new Set(\n router[internal.getGroups]().flatMap(g => g.controllers ?? [])\n )\n if (!groupedControllers.has(controller)) {\n return this.entryToConfig(router[internal.getDefaultEntry]())\n }\n }\n\n // Controller not found in any module's router — return empty config\n return { middleware: [] }\n }\n\n /**\n * Collect all global middleware registered via `router.use()` across all modules.\n */\n getGlobalMiddleware(): Constructor<Middleware>[] {\n const global: Constructor<Middleware>[] = []\n for (const { router } of this.routers) {\n global.push(...router[internal.getGlobalMiddleware]())\n }\n return global\n }\n\n /**\n * Merge parent default entry with child group entry following inheritance rules.\n */\n private mergeEntries(parent: RouterEntry, child: RouterEntry): ResolvedRouterConfig {\n return {\n // Concatenate: parent prefix + child prefix\n prefix: this.concatPrefixes(parent.prefix, child.prefix),\n // Override: child domain wins\n domain: child.domain ?? parent.domain,\n // Concatenate: parent name + child name\n name: this.concatNames(parent.name, child.name),\n // Concatenate: parent middleware first, then child\n middleware: [...parent.middleware, ...child.middleware],\n // Override: child version wins\n version: child.version ?? parent.version,\n // Override: child hideFromDocs wins\n hideFromDocs: child.hideFromDocs ?? parent.hideFromDocs,\n // Extend: parent params extended with child params\n params: this.mergeParams(parent.params, child.params),\n }\n }\n\n private entryToConfig(entry: RouterEntry): ResolvedRouterConfig {\n return {\n prefix: entry.prefix,\n domain: entry.domain,\n name: entry.name,\n middleware: [...entry.middleware],\n version: entry.version,\n hideFromDocs: entry.hideFromDocs,\n params: entry.params,\n }\n }\n\n // eslint-disable-next-line @typescript-eslint/no-explicit-any -- ZodObject generics require any for flexible shape parameter\n private mergeParams(parent?: ZodObject<any>, child?: ZodObject<any>): ZodObject<any> | undefined {\n if (!parent && !child) return undefined\n if (!parent) return child\n if (!child) return parent\n // oxlint-disable-next-line typescript/no-explicit-any\n return parent.extend(child.shape) as ZodObject<any>\n }\n\n private concatPrefixes(parent?: string, child?: string): string | undefined {\n if (!parent && !child) return undefined\n if (!parent) return child\n if (!child) return parent\n // Normalize: remove trailing slash from parent, ensure child starts with /\n const p = parent.endsWith('/') ? parent.slice(0, -1) : parent\n const c = child.startsWith('/') ? child : `/${child}`\n return `${p}${c}`\n }\n\n private concatNames(parent?: string, child?: string): string | undefined {\n if (!parent && !child) return undefined\n if (!parent) return child\n if (!child) return parent\n return `${parent}${child}`\n }\n}\n","import { injectable, container as tsyringeRootContainer } from 'tsyringe'\nimport { CacheModule } from './cache'\nimport type { CronJob } from './cron/cron-job'\nimport { CronManager } from './cron/cron-manager'\nimport { Container } from './di/container'\nimport { runWithContainer } from './di/container-storage'\nimport { DI_TOKENS } from './di/tokens'\nimport { Scope } from './di/types'\nimport { type StratalEnv } from './env'\nimport { ApplicationError } from './errors'\nimport { DefaultExceptionHandler } from './errors/default-exception-handler'\nimport { createCliExceptionContext, createCronExceptionContext, createQueueExceptionContext } from './errors/exception-context'\nimport type { ExceptionHandler } from './errors/exception-handler'\nimport type { EventHandler } from './events'\nimport { EventRegistry, getListenerHandlers } from './events'\nimport type { StratalExecutionContext } from './execution-context'\nimport { I18nModule } from './i18n/i18n.module'\nimport { ConsoleTransport, JsonFormatter, LOGGER_TOKENS, LoggerService, LogLevel, PrettyFormatter } from './logger'\nimport { ModuleRegistry } from './module/module-registry'\nimport type { DynamicModule, ModuleClass } from './module/types'\nimport { OpenAPIModule } from './openapi'\nimport type { Command } from './quarry/command'\nimport { ApiCommand } from './quarry/commands/api.command'\nimport { EventListCommand } from './quarry/commands/event-list.command'\nimport { HelpCommand } from './quarry/commands/help.command'\nimport { McpServeCommand } from './quarry/commands/mcp-serve.command'\nimport { McpToolsCommand } from './quarry/commands/mcp-tools.command'\nimport { QueueListCommand } from './quarry/commands/queue-list.command'\nimport { RouteListCommand } from './quarry/commands/route-list.command'\nimport { RouteTypesCommand } from './quarry/commands/route-types.command'\nimport { ScheduleListCommand } from './quarry/commands/schedule-list.command'\nimport { QuarryRegistry } from './quarry/quarry-registry'\nimport type { CommandInput, CommandResult } from './quarry/types'\nimport { type ConsumerRegistry } from './queue/consumer-registry'\nimport type { IQueueConsumer, QueueMessage } from './queue/queue-consumer'\nimport { type QueueManager } from './queue/queue-manager'\nimport { QueueModule } from './queue/queue.module'\nimport { type RouterContext } from './router'\nimport { HonoApp } from './router/hono-app'\nimport { RouteRegistry } from './router/route-registry'\nimport { RouterResolver } from './router/router-resolver'\nimport { ROUTER_TOKENS } from './router/router.tokens'\nimport { Uri } from './router/uri'\nimport { LocalePathService } from './router/services/locale-path.service'\nimport { RouteRegistrationService } from './router/services/route-registration.service'\nimport { VersioningService } from './router/services/versioning.service'\nimport type { VersioningOptions } from './router/types'\nimport { DbSeedCommand, DbSeedListCommand, SEEDER_TOKENS, SeederRegistry, type Seeder } from './seeder'\nimport type { Constructor } from './types'\n\nexport interface ApplicationConfig {\n /** Root application module */\n module: ModuleClass | DynamicModule\n /** Logging configuration. Defaults: level=INFO, formatter='json' */\n logging?: {\n level?: LogLevel\n formatter?: 'json' | 'pretty'\n }\n /**\n * API versioning configuration.\n * When provided, enables URI-based versioning for controllers.\n */\n versioning?: VersioningOptions\n /**\n * Custom exception handler class.\n *\n * Extend {@link ExceptionHandler} and override `register()` to configure\n * custom reporting, rendering, and post-processing of exceptions.\n *\n * When not provided, {@link DefaultExceptionHandler} is used (standard\n * severity-based logging and JSON error responses).\n *\n * @example\n * ```typescript\n * new Stratal({\n * module: AppModule,\n * exceptionHandler: AppExceptionHandler,\n * })\n * ```\n */\n exceptionHandler?: Constructor<ExceptionHandler>\n}\n\nexport interface ApplicationOptions extends ApplicationConfig {\n env: StratalEnv\n ctx: StratalExecutionContext\n}\n\n/**\n * Application\n *\n * Main application class managing the two-tier container hierarchy:\n * - Global Container: All services (singletons via tsyringe native)\n * - Request Container: Child of global, context-enriched instances per request\n *\n * @example\n * ```typescript\n * const app = new Application({ module: AppModule, env, ctx })\n * await app.initialize()\n *\n * // Access container via getter\n * const service = app.container.resolve(MY_TOKEN)\n *\n * // Handle HTTP request (via HonoApp)\n * // Handle queue batch\n * await app.handleQueue(batch, 'my-queue')\n * ```\n */\nexport class Application {\n /**\n * Unified Container - manages all DI operations\n */\n private _container: Container\n\n private honoApp!: HonoApp\n private moduleRegistry: ModuleRegistry\n private consumerRegistry!: ConsumerRegistry\n private cronManager!: CronManager\n private quarry!: QuarryRegistry\n private initialized = false\n\n readonly env: StratalEnv\n private readonly appConfig: ApplicationConfig\n\n constructor({ env, ctx, ...config }: ApplicationOptions) {\n this.env = env\n this.appConfig = config\n\n ApplicationError.captureStackTraces = env.ENVIRONMENT !== 'production'\n\n // Create unified Container with explicit child container\n this._container = new Container({\n container: tsyringeRootContainer.createChildContainer()\n })\n\n // Register globally — env and ctx always available\n this._container.registerValue(DI_TOKENS.Application, this)\n this._container.registerValue(DI_TOKENS.CloudflareEnv, env)\n this._container.registerValue(DI_TOKENS.ExecutionContext, ctx)\n\n // Register core infrastructure inline\n this.registerLoggerService()\n this.registerCoreServices()\n\n // Create ModuleRegistry with our Container\n const logger = this._container.resolve<LoggerService>(LOGGER_TOKENS.LoggerService)\n this.moduleRegistry = new ModuleRegistry(this._container, logger)\n\n // Register ModuleRegistry in container so modules can access it in onInitialize\n this._container.registerValue(DI_TOKENS.ModuleRegistry, this.moduleRegistry)\n }\n\n /**\n * Get the Container instance\n */\n get container(): Container {\n return this._container\n }\n\n /**\n * Get the HonoApp instance\n */\n get hono(): HonoApp {\n return this.honoApp\n }\n\n /**\n * Get the application configuration\n */\n get config(): ApplicationConfig {\n return this.appConfig\n }\n\n async initialize(): Promise<void> {\n if (this.initialized) {\n return\n }\n\n // Wrap in AsyncLocalStorage so getContainer() works for route() and other standalone functions\n await runWithContainer(this._container, () => this.initializeInternal())\n }\n\n private async initializeInternal(): Promise<void> {\n // Phase 1: Register core infrastructure modules (internal)\n this.moduleRegistry.registerAll([\n I18nModule,\n OpenAPIModule,\n QueueModule,\n CacheModule,\n ])\n\n // Phase 2: Register user's root module (traverses imports)\n this.moduleRegistry.register(this.appConfig.module)\n\n // Phase 3: Initialize all modules\n await this.moduleRegistry.initialize()\n\n // Phase 3.5: Initialize ExceptionHandler and call module onException hooks\n this.initializeExceptionHandler()\n\n // Phase 4: Resolve managers from container\n this.consumerRegistry = this._container.resolve<ConsumerRegistry>(DI_TOKENS.ConsumerRegistry)\n this.cronManager = this._container.resolve<CronManager>(DI_TOKENS.Cron)\n this.quarry = this._container.resolve<QuarryRegistry>(DI_TOKENS.Quarry)\n\n // Phase 4.5: Register routing services in container\n // (After Phase 3 so I18N_TOKENS.Options is available for LocalePathService)\n this.registerRoutingServices()\n\n // Phase 5: Resolve & configure HonoApp\n // LocalePathService is transitively resolved via RouteRegistrationService → RouteRegistry\n // during configure(), which triggers locale middleware setup on HonoApp before route registration.\n this.honoApp = this._container.resolve<HonoApp>(ROUTER_TOKENS.HonoApp)\n await this.honoApp.configure()\n\n // Phase 6: Configure queues, cron, events, commands, seeders\n this.registerQueueConsumers()\n this.registerCronJobs()\n this.registerEventListeners()\n this.registerSeeders()\n this.registerCommands()\n\n this.initialized = true\n }\n\n /**\n * Register routing services as singletons in the container.\n * Called after module initialization so I18N_TOKENS.Options is available.\n */\n private registerRoutingServices(): void {\n // VersioningService — resolves version prefixes from appConfig.versioning\n this._container.register(ROUTER_TOKENS.VersioningService, VersioningService, Scope.Singleton)\n\n // HonoApp — the Hono application instance (must be before LocalePathService)\n this._container.register(ROUTER_TOKENS.HonoApp, HonoApp, Scope.Singleton)\n\n // LocalePathService — computes LocalePathConfig and applies locale middleware to HonoApp\n this._container.register(ROUTER_TOKENS.LocalePathService, LocalePathService, Scope.Singleton)\n\n // RouteRegistry — single source of truth, expands routes via services above\n this._container.register(ROUTER_TOKENS.RouteRegistry, RouteRegistry, Scope.Singleton)\n\n // Uri — URL generation service (request-scoped for access to RouterContext)\n this._container.register(ROUTER_TOKENS.Uri, Uri, Scope.Request)\n\n // RouterResolver — merges Router configs from modules\n const routerConfigs = this.moduleRegistry.getAllRouterConfigs()\n const routerResolver = routerConfigs.length > 0 ? new RouterResolver(routerConfigs) : null\n this._container.registerValue(ROUTER_TOKENS.RouterResolver, routerResolver)\n\n // RouteRegistrationService — transient, resolved in HonoApp.configure()\n this._container.register(RouteRegistrationService, RouteRegistrationService)\n }\n\n /**\n * Resolve a service from the container\n */\n resolve<T>(token: symbol): T {\n try {\n return this._container.resolve(token)\n } catch (error) {\n const handler = this._container.resolve<ExceptionHandler>(DI_TOKENS.ExceptionHandler)\n const ctx = createCliExceptionContext('resolve')\n // Fire-and-forget — reporting happens via waitUntil internally\n void handler.handle(error, ctx)\n throw error\n }\n }\n\n /**\n * Handle queue batch processing\n */\n async handleQueue(batch: MessageBatch, queueName: string): Promise<void> {\n const firstMessage = batch.messages[0]?.body as QueueMessage | undefined\n const locale = firstMessage?.metadata?.locale ?? 'en'\n const mockRouterContext = this.createMockRouterContext(locale)\n\n await this._container.runInRequestScope(mockRouterContext, async (requestContainer) => {\n try {\n const queueManager = requestContainer.resolve<QueueManager>(DI_TOKENS.Queue)\n await queueManager.processBatch(queueName, batch)\n } catch (error) {\n const handler = requestContainer.resolve<ExceptionHandler>(DI_TOKENS.ExceptionHandler)\n await handler.handle(error, createQueueExceptionContext(queueName))\n throw error\n }\n })\n }\n\n /**\n * Handle scheduled cron trigger\n */\n async handleScheduled(controller: ScheduledController): Promise<void> {\n const mockRouterContext = this.createMockRouterContext('en')\n\n await this._container.runInRequestScope(mockRouterContext, async (requestContainer) => {\n try {\n await this.cronManager.executeScheduled(controller)\n } catch (error) {\n const handler = requestContainer.resolve<ExceptionHandler>(DI_TOKENS.ExceptionHandler)\n await handler.handle(error, createCronExceptionContext())\n throw error\n }\n })\n }\n\n /**\n * Create mock RouterContext for queue/cron/seeder processing\n */\n createMockRouterContext(locale = 'en'): RouterContext {\n return {\n getLocale: () => locale,\n setLocale: () => { /* no-op */ },\n getContainer: () => this._container,\n } as unknown as RouterContext\n }\n\n async shutdown(): Promise<void> {\n if (!this.initialized) return\n this.initialized = false\n\n await this.moduleRegistry.shutdown()\n\n const logger = this._container.resolve<LoggerService>(LOGGER_TOKENS.LoggerService)\n logger.info('Disposing container...')\n\n await this._container.dispose()\n }\n\n /**\n * Execute a command by name in a request-scoped container.\n */\n async handleCommand(name: string, input?: CommandInput): Promise<CommandResult> {\n const mockContext = this.createMockRouterContext('en')\n return this._container.runInRequestScope(mockContext, async () => {\n return this.quarry.call(name, input)\n })\n }\n\n private registerCommands(): void {\n // Built-in commands (always available)\n const builtinCommands: Constructor<Command>[] = [\n HelpCommand,\n DbSeedCommand, DbSeedListCommand,\n RouteListCommand, RouteTypesCommand, EventListCommand,\n ScheduleListCommand, QueueListCommand,\n McpServeCommand, McpToolsCommand, ApiCommand,\n ]\n for (const Cmd of builtinCommands) {\n injectable()(Cmd)\n this._container.register(Cmd, Cmd, Scope.Singleton)\n this.quarry.register(Cmd)\n }\n\n // User commands from modules\n const commands = this.moduleRegistry.getAllCommands()\n if (commands.length === 0) {\n return\n }\n\n for (const CommandClass of commands) {\n this.quarry.register(CommandClass as Constructor<Command>)\n }\n }\n\n private registerSeeders(): void {\n const seeders = this.moduleRegistry.getAllSeeders()\n if (seeders.length === 0) return\n const registry = this._container.resolve<SeederRegistry>(SEEDER_TOKENS.SeederRegistry)\n for (const SeederClass of seeders) {\n registry.register(SeederClass as Constructor<Seeder>)\n }\n }\n\n private registerQueueConsumers(): void {\n for (const ConsumerClass of this.moduleRegistry.getAllConsumers()) {\n const consumer = this._container.resolve(ConsumerClass) as IQueueConsumer\n this.consumerRegistry.register(consumer)\n }\n }\n\n private registerCronJobs(): void {\n for (const JobClass of this.moduleRegistry.getAllJobs()) {\n const job = this._container.resolve(JobClass) as CronJob\n this.cronManager.registerJob(job)\n }\n }\n\n /**\n * Auto-wire `@Listener()` classes with the EventRegistry.\n */\n private registerEventListeners(): void {\n const listeners = this.moduleRegistry.getAllListeners()\n if (listeners.length === 0) {\n return\n }\n\n const eventRegistry = this._container.resolve<EventRegistry>(DI_TOKENS.EventRegistry)\n\n for (const ListenerClass of listeners) {\n const instance = this._container.resolve(ListenerClass) as Record<string, ((...args: unknown[]) => unknown)>\n const handlers = getListenerHandlers(ListenerClass)\n\n for (const { methodName, event, options } of handlers) {\n eventRegistry.on(event, instance[methodName].bind(instance) as EventHandler, options)\n }\n }\n }\n\n /**\n * Register LoggerService and dependencies\n */\n private registerLoggerService(): void {\n const logLevel = this.appConfig.logging?.level ?? LogLevel.INFO\n const formatter = this.appConfig.logging?.formatter ?? 'json'\n\n this._container.registerValue(LOGGER_TOKENS.LogLevelOptions, logLevel)\n\n this._container\n .when(() => formatter === 'pretty')\n .use(LOGGER_TOKENS.Formatter)\n .give(PrettyFormatter)\n .otherwise(JsonFormatter)\n\n this._container.registerSingleton(LOGGER_TOKENS.ConsoleTransport, ConsoleTransport)\n this._container.registerFactory(LOGGER_TOKENS.Transports, (c) => [c.resolve(LOGGER_TOKENS.ConsoleTransport)])\n this._container.registerSingleton(LOGGER_TOKENS.LoggerService, LoggerService)\n }\n\n /**\n * Register core services with explicit scope\n */\n private registerCoreServices(): void {\n this._container.registerSingleton(DI_TOKENS.Cron, CronManager)\n this._container.registerSingleton(\n DI_TOKENS.ExceptionHandler,\n (this.appConfig.exceptionHandler ?? DefaultExceptionHandler) as Constructor,\n )\n this._container.registerSingleton(DI_TOKENS.EventRegistry, EventRegistry)\n this._container.registerSingleton(DI_TOKENS.Quarry, QuarryRegistry)\n this._container.registerValue(SEEDER_TOKENS.SeederRegistry, new SeederRegistry(this))\n }\n\n /**\n * Initialize the ExceptionHandler: call register(), then module onException hooks.\n */\n private initializeExceptionHandler(): void {\n const handler = this._container.resolve<ExceptionHandler>(DI_TOKENS.ExceptionHandler)\n handler.register()\n this.moduleRegistry.configureExceptionHandlers(handler)\n }\n}\n","import 'reflect-metadata'\n\nimport { Application, type ApplicationConfig } from './application'\nimport type { StratalEnv } from './env'\nimport { StratalNotInitializedError } from './errors'\nimport type { HonoApp } from './router/hono-app'\n\n/**\n * Stratal — Hono-style entry point for Cloudflare Workers.\n *\n * Eagerly bootstraps the Application at construction time, dynamically\n * importing `cloudflare:workers` for env and waitUntil.\n *\n * @example\n * ```typescript\n * import { Stratal } from 'stratal'\n * import { AppModule } from './app.module'\n *\n * export default new Stratal({ module: AppModule })\n * ```\n */\nexport class Stratal<Env extends StratalEnv = StratalEnv> {\n private app: Application | null = null\n private initPromise: Promise<Application>\n\n private static _application: Promise<Application> | null = null\n private static _generation = 0\n private static _previousInstance: Stratal | null = null\n\n constructor(config: ApplicationConfig) {\n this.fetch = this.fetch.bind(this)\n this.queue = this.queue.bind(this)\n this.scheduled = this.scheduled.bind(this)\n\n // Invalidate any in-flight initialization from a previous instance (Vite HMR reload)\n const generation = ++Stratal._generation\n\n if (Stratal._previousInstance) {\n void Stratal._previousInstance.shutdown()\n }\n Stratal._previousInstance = this\n\n this.initPromise = this.prepareApp(config, generation)\n Stratal._application = this.initPromise\n }\n\n async fetch(request: Request, env: Env, ctx: ExecutionContext): Promise<Response> {\n const app = await this.ensureReady()\n return app.hono.fetch(request, env, ctx)\n }\n\n async queue(batch: MessageBatch): Promise<void> {\n const app = await this.ensureReady()\n return app.handleQueue(batch, batch.queue)\n }\n\n async scheduled(controller: ScheduledController): Promise<void> {\n const app = await this.ensureReady()\n return app.handleScheduled(controller)\n }\n\n get hono(): Promise<HonoApp> {\n return this.initPromise.then(app => app.hono)\n }\n\n async shutdown(): Promise<void> {\n try { this.app = await this.initPromise } catch { /* ignore */ }\n if (this.app) {\n await this.app.shutdown()\n this.app = null\n }\n }\n\n /**\n * @internal\n * Resolves the Application instance from the static singleton.\n * Used by worker base classes (DurableObject, Workflow, WorkerEntrypoint)\n * to access the DI container without going through Cloudflare RPC.\n */\n static resolveApplication(): Promise<Application> {\n if (!Stratal._application) {\n throw new StratalNotInitializedError()\n }\n return Stratal._application\n }\n\n private async ensureReady(): Promise<Application> {\n this.app ??= await this.initPromise;\n return this.app\n }\n\n private async prepareApp(config: ApplicationConfig, generation: number): Promise<Application> {\n const { env, waitUntil } = await import('cloudflare:workers')\n\n // After async import, check if a newer instance has replaced us (Vite HMR reload)\n if (generation !== Stratal._generation) {\n return new Promise<Application>(() => {\n //\n }) // Never resolves — avoids cross-request promise warning\n }\n\n const app = new Application({ ...config, env: env as Env, ctx: { waitUntil } })\n await app.initialize()\n\n // Check again after initialization completes\n if (generation !== Stratal._generation) {\n await app.shutdown()\n return new Promise<Application>(() => {\n //\n })\n }\n\n return app\n }\n}\n"],"mappings":"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"}
|
|
@@ -13,7 +13,7 @@
|
|
|
13
13
|
* ]
|
|
14
14
|
* ```
|
|
15
15
|
*/
|
|
16
|
-
type Constructor<T
|
|
16
|
+
type Constructor<T = object> = new (...args: any[]) => T;
|
|
17
17
|
//#endregion
|
|
18
18
|
export { Constructor as t };
|
|
19
|
-
//# sourceMappingURL=types-
|
|
19
|
+
//# sourceMappingURL=types-CN0zONAZ.d.mts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"types-CN0zONAZ.d.mts","names":[],"sources":["../src/types.ts"],"mappings":";;AAeA;;;;;;;;;;;;;KAAY,WAAA,uBAAkC,IAAA,YAAgB,CAAA"}
|
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import { t as __exportAll } from "./chunk-D1SwGrFN.mjs";
|
|
2
|
-
import { i as dimWhite, n as cyan, r as dim, s as yellow, t as bold } from "./colors-
|
|
2
|
+
import { i as dimWhite, n as cyan, r as dim, s as yellow, t as bold } from "./colors-BTAnQRGU.mjs";
|
|
3
3
|
//#region src/quarry/usage-generator.ts
|
|
4
4
|
var usage_generator_exports = /* @__PURE__ */ __exportAll({
|
|
5
5
|
generateListing: () => generateListing,
|
|
@@ -155,4 +155,4 @@ function formatTable(rows) {
|
|
|
155
155
|
//#endregion
|
|
156
156
|
export { usage_generator_exports as n, generateUsage as t };
|
|
157
157
|
|
|
158
|
-
//# sourceMappingURL=usage-generator-
|
|
158
|
+
//# sourceMappingURL=usage-generator-Cl1HPlUp.mjs.map
|