sonamu 0.9.4 → 0.9.6
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/ai/providers/rtzr/utils.js +2 -2
- package/dist/api/config.d.ts +13 -2
- package/dist/api/config.d.ts.map +1 -1
- package/dist/api/config.js +1 -1
- package/dist/api/context.d.ts +17 -7
- package/dist/api/context.d.ts.map +1 -1
- package/dist/api/context.js +1 -1
- package/dist/api/decorators.d.ts +18 -0
- package/dist/api/decorators.d.ts.map +1 -1
- package/dist/api/decorators.js +54 -3
- package/dist/api/index.js +8 -3
- package/dist/api/sonamu.d.ts +24 -9
- package/dist/api/sonamu.d.ts.map +1 -1
- package/dist/api/sonamu.js +365 -79
- package/dist/api/websocket-helpers.d.ts +24 -0
- package/dist/api/websocket-helpers.d.ts.map +1 -0
- package/dist/api/websocket-helpers.js +77 -0
- package/dist/bin/cli.js +12 -4
- package/dist/database/upsert-builder.js +4 -4
- package/dist/dict/sonamu-dictionary.js +6 -6
- package/dist/entity/entity-manager.js +1 -1
- package/dist/entity/entity.js +3 -3
- package/dist/index.d.ts +6 -0
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +16 -4
- package/dist/migration/code-generation.d.ts.map +1 -1
- package/dist/migration/code-generation.js +8 -9
- package/dist/stream/index.d.ts +6 -0
- package/dist/stream/index.d.ts.map +1 -1
- package/dist/stream/index.js +13 -2
- package/dist/stream/ws-audience-resolver.d.ts +15 -0
- package/dist/stream/ws-audience-resolver.d.ts.map +1 -0
- package/dist/stream/ws-audience-resolver.js +31 -0
- package/dist/stream/ws-audience.d.ts +28 -0
- package/dist/stream/ws-audience.d.ts.map +1 -0
- package/dist/stream/ws-audience.js +46 -0
- package/dist/stream/ws-cluster-bus.d.ts +23 -0
- package/dist/stream/ws-cluster-bus.d.ts.map +1 -0
- package/dist/stream/ws-cluster-bus.js +18 -0
- package/dist/stream/ws-core.d.ts +15 -0
- package/dist/stream/ws-core.d.ts.map +1 -0
- package/dist/stream/ws-core.js +1 -0
- package/dist/stream/ws-delivery.d.ts +24 -0
- package/dist/stream/ws-delivery.d.ts.map +1 -0
- package/dist/stream/ws-delivery.js +103 -0
- package/dist/stream/ws-local-connection-store.d.ts +10 -0
- package/dist/stream/ws-local-connection-store.d.ts.map +1 -0
- package/dist/stream/ws-local-connection-store.js +44 -0
- package/dist/stream/ws-presence-store.d.ts +61 -0
- package/dist/stream/ws-presence-store.d.ts.map +1 -0
- package/dist/stream/ws-presence-store.js +236 -0
- package/dist/stream/ws-registry.d.ts +42 -0
- package/dist/stream/ws-registry.d.ts.map +1 -0
- package/dist/stream/ws-registry.js +108 -0
- package/dist/stream/ws.d.ts +52 -0
- package/dist/stream/ws.d.ts.map +1 -0
- package/dist/stream/ws.js +397 -0
- package/dist/syncer/api-parser.d.ts.map +1 -1
- package/dist/syncer/api-parser.js +72 -2
- package/dist/syncer/checksum.d.ts.map +1 -1
- package/dist/syncer/checksum.js +13 -12
- package/dist/syncer/code-generator.d.ts.map +1 -1
- package/dist/syncer/code-generator.js +7 -4
- package/dist/syncer/event-batcher.d.ts +27 -0
- package/dist/syncer/event-batcher.d.ts.map +1 -0
- package/dist/syncer/event-batcher.js +69 -0
- package/dist/syncer/file-patterns.d.ts +48 -26
- package/dist/syncer/file-patterns.d.ts.map +1 -1
- package/dist/syncer/file-patterns.js +71 -23
- package/dist/syncer/file-tracking.d.ts +13 -0
- package/dist/syncer/file-tracking.d.ts.map +1 -0
- package/dist/syncer/file-tracking.js +33 -0
- package/dist/syncer/index.js +2 -2
- package/dist/syncer/module-loader.d.ts +2 -11
- package/dist/syncer/module-loader.d.ts.map +1 -1
- package/dist/syncer/module-loader.js +3 -3
- package/dist/syncer/syncer-actions.d.ts +39 -6
- package/dist/syncer/syncer-actions.d.ts.map +1 -1
- package/dist/syncer/syncer-actions.js +125 -10
- package/dist/syncer/syncer.d.ts +33 -19
- package/dist/syncer/syncer.d.ts.map +1 -1
- package/dist/syncer/syncer.js +168 -168
- package/dist/syncer/watcher.d.ts +8 -0
- package/dist/syncer/watcher.d.ts.map +1 -0
- package/dist/syncer/watcher.js +105 -0
- package/dist/tasks/workflow-manager.d.ts.map +1 -1
- package/dist/tasks/workflow-manager.js +2 -1
- package/dist/template/implementations/services.template.d.ts.map +1 -1
- package/dist/template/implementations/services.template.js +36 -1
- package/dist/testing/bootstrap.d.ts.map +1 -1
- package/dist/testing/bootstrap.js +8 -1
- package/dist/testing/data-explorer.d.ts.map +1 -1
- package/dist/testing/data-explorer.js +5 -3
- package/dist/testing/fixture-manager.js +1 -1
- package/dist/types/types.d.ts +2 -1
- package/dist/types/types.d.ts.map +1 -1
- package/dist/types/types.js +2 -2
- package/dist/ui/api.d.ts.map +1 -1
- package/dist/ui/api.js +4 -3
- package/dist/ui/cdd-service.js +1 -1
- package/dist/ui-web/assets/{index-C5KUjXm0.js → index-BmThfg-s.js} +39 -39
- package/dist/ui-web/assets/index-D4rYm-Xz.css +1 -0
- package/dist/ui-web/index.html +2 -2
- package/dist/utils/async-utils.d.ts +27 -3
- package/dist/utils/async-utils.d.ts.map +1 -1
- package/dist/utils/async-utils.js +56 -6
- package/dist/utils/formatter.d.ts +7 -1
- package/dist/utils/formatter.d.ts.map +1 -1
- package/dist/utils/formatter.js +95 -60
- package/dist/utils/fs-utils.d.ts +2 -0
- package/dist/utils/fs-utils.d.ts.map +1 -1
- package/dist/utils/fs-utils.js +10 -2
- package/dist/utils/process-utils.d.ts +6 -0
- package/dist/utils/process-utils.d.ts.map +1 -1
- package/dist/utils/process-utils.js +16 -3
- package/dist/utils/utils.d.ts +1 -0
- package/dist/utils/utils.d.ts.map +1 -1
- package/dist/utils/utils.js +2 -2
- package/package.json +7 -5
- package/src/ai/providers/rtzr/utils.ts +1 -1
- package/src/api/__tests__/sonamu.websocket.test.ts +64 -0
- package/src/api/__tests__/websocket-context.types.test.ts +58 -0
- package/src/api/config.ts +28 -2
- package/src/api/context.ts +21 -7
- package/src/api/decorators.ts +101 -1
- package/src/api/sonamu.ts +529 -127
- package/src/api/websocket-helpers.ts +122 -0
- package/src/bin/cli.ts +10 -2
- package/src/database/upsert-builder.ts +3 -3
- package/src/dict/sonamu-dictionary.ts +3 -3
- package/src/entity/entity.ts +1 -1
- package/src/index.ts +6 -0
- package/src/migration/code-generation.ts +6 -11
- package/src/shared/app.shared.ts.txt +312 -4
- package/src/shared/web.shared.ts.txt +340 -4
- package/src/stream/__tests__/ws-contracts.test.ts +381 -0
- package/src/stream/__tests__/ws.test.ts +449 -0
- package/src/stream/index.ts +6 -0
- package/src/stream/ws-audience-resolver.ts +35 -0
- package/src/stream/ws-audience.ts +62 -0
- package/src/stream/ws-cluster-bus.ts +32 -0
- package/src/stream/ws-core.ts +16 -0
- package/src/stream/ws-delivery.ts +138 -0
- package/src/stream/ws-local-connection-store.ts +44 -0
- package/src/stream/ws-presence-store.ts +326 -0
- package/src/stream/ws-registry.ts +138 -0
- package/src/stream/ws.ts +591 -0
- package/src/syncer/__tests__/api-parser.websocket-type-ref.test.ts +78 -0
- package/src/syncer/api-parser.ts +112 -1
- package/src/syncer/checksum.ts +23 -29
- package/src/syncer/code-generator.ts +4 -1
- package/src/syncer/event-batcher.ts +72 -0
- package/src/syncer/file-patterns.ts +98 -30
- package/src/syncer/file-tracking.ts +27 -0
- package/src/syncer/module-loader.ts +5 -12
- package/src/syncer/syncer-actions.ts +179 -17
- package/src/syncer/syncer.ts +250 -287
- package/src/syncer/watcher.ts +128 -0
- package/src/tasks/workflow-manager.ts +1 -0
- package/src/template/__tests__/services.template.websocket.test.ts +79 -0
- package/src/template/implementations/services.template.ts +69 -0
- package/src/testing/bootstrap.ts +8 -1
- package/src/testing/data-explorer.ts +3 -2
- package/src/types/types.ts +20 -2
- package/src/ui/api.ts +10 -1
- package/src/utils/async-utils.ts +71 -4
- package/src/utils/formatter.ts +114 -75
- package/src/utils/fs-utils.ts +9 -0
- package/src/utils/process-utils.ts +17 -0
- package/src/utils/utils.ts +1 -1
- package/dist/ui-web/assets/index-Dr8pRJC_.css +0 -1
package/dist/api/sonamu.js
CHANGED
|
@@ -8,15 +8,18 @@ import { NotFoundException, init_so_exceptions } from "../exceptions/so-exceptio
|
|
|
8
8
|
import { BufferedFile, init_buffered_file } from "../storage/buffered-file.js";
|
|
9
9
|
import { UploadedFile, init_uploaded_file } from "../storage/uploaded-file.js";
|
|
10
10
|
import { createMockSSEFactory, init_sse } from "../stream/sse.js";
|
|
11
|
+
import { WebSocketRuntime, init_ws } from "../stream/ws.js";
|
|
12
|
+
import { centerText, init_console_util } from "../utils/console-util.js";
|
|
11
13
|
import { init_controller, isDaemonServer } from "../utils/controller.js";
|
|
12
14
|
import { exists, fileExists, init_fs_utils } from "../utils/fs-utils.js";
|
|
13
15
|
import { convertFastifyHeadersToStandard, init_utils, merge } from "../utils/utils.js";
|
|
14
16
|
import { getSecrets, init_secret } from "./secret.js";
|
|
17
|
+
import { createWebSocketReplyStub, init_websocket_helpers, resolveIntegratedViteHmrOptions, resolveWebSocketCloseDescriptor, resolveWebSocketPluginOptions } from "./websocket-helpers.js";
|
|
15
18
|
import { AsyncLocalStorage } from "async_hooks";
|
|
16
19
|
import { dispose } from "@logtape/logtape";
|
|
17
|
-
import assert from "assert";
|
|
18
20
|
import fs from "fs/promises";
|
|
19
21
|
import path from "path";
|
|
22
|
+
import chalk from "chalk";
|
|
20
23
|
import os from "os";
|
|
21
24
|
import mime, { lookup } from "mime-types";
|
|
22
25
|
|
|
@@ -37,6 +40,11 @@ function formatTime(ms) {
|
|
|
37
40
|
function isLocalHost(host) {
|
|
38
41
|
return LOCAL_HOSTS.has(host);
|
|
39
42
|
}
|
|
43
|
+
function runGuards({ guards, config, request, api }) {
|
|
44
|
+
for (const guard of guards ?? []) {
|
|
45
|
+
config.guardHandler(guard, request, api);
|
|
46
|
+
}
|
|
47
|
+
}
|
|
40
48
|
var SonamuClass, Sonamu, LOCAL_HOSTS;
|
|
41
49
|
var init_sonamu = __esmMin((() => {
|
|
42
50
|
init_better_auth_entities();
|
|
@@ -48,10 +56,13 @@ var init_sonamu = __esmMin((() => {
|
|
|
48
56
|
init_buffered_file();
|
|
49
57
|
init_uploaded_file();
|
|
50
58
|
init_sse();
|
|
59
|
+
init_ws();
|
|
60
|
+
init_console_util();
|
|
51
61
|
init_controller();
|
|
52
62
|
init_fs_utils();
|
|
53
63
|
init_utils();
|
|
54
64
|
init_secret();
|
|
65
|
+
init_websocket_helpers();
|
|
55
66
|
SonamuClass = class {
|
|
56
67
|
isInitialized = false;
|
|
57
68
|
forTesting = false;
|
|
@@ -63,10 +74,14 @@ var init_sonamu = __esmMin((() => {
|
|
|
63
74
|
}
|
|
64
75
|
if (process.env.NODE_ENV === "test") {
|
|
65
76
|
return {
|
|
77
|
+
transport: "http",
|
|
66
78
|
request: null,
|
|
67
79
|
reply: null,
|
|
68
80
|
headers: {},
|
|
69
81
|
createSSE: (schema) => createMockSSEFactory(schema),
|
|
82
|
+
locale: "",
|
|
83
|
+
user: null,
|
|
84
|
+
session: null,
|
|
70
85
|
naiteStore: new Map()
|
|
71
86
|
};
|
|
72
87
|
} else {
|
|
@@ -158,9 +173,18 @@ var init_sonamu = __esmMin((() => {
|
|
|
158
173
|
set devVitestManager(manager) {
|
|
159
174
|
this._devVitestManager = manager;
|
|
160
175
|
}
|
|
176
|
+
_websocketRuntime = null;
|
|
177
|
+
websocketPluginServers = new WeakSet();
|
|
178
|
+
get websocketRuntime() {
|
|
179
|
+
if (!this._websocketRuntime) {
|
|
180
|
+
throw new Error("WebSocket runtime has not been initialized.");
|
|
181
|
+
}
|
|
182
|
+
return this._websocketRuntime;
|
|
183
|
+
}
|
|
184
|
+
set websocketRuntime(runtime) {
|
|
185
|
+
this._websocketRuntime = runtime;
|
|
186
|
+
}
|
|
161
187
|
watcher = null;
|
|
162
|
-
pendingFiles = [];
|
|
163
|
-
hmrStartTime = 0;
|
|
164
188
|
server = null;
|
|
165
189
|
async initForTesting() {
|
|
166
190
|
await this.init(true, false, undefined, true);
|
|
@@ -214,7 +238,7 @@ var init_sonamu = __esmMin((() => {
|
|
|
214
238
|
await this.syncer.autoloadWorkflows();
|
|
215
239
|
const { TemplateManager } = await import("../template/index.js");
|
|
216
240
|
await TemplateManager.autoload();
|
|
217
|
-
await this.syncer.
|
|
241
|
+
await this.syncer.autoloadSsrRoutes();
|
|
218
242
|
const { isLocal, isTest, isHotReloadServer } = await import("../utils/controller.js");
|
|
219
243
|
if (isLocal() && !isTest() && isHotReloadServer() && enableSync) {
|
|
220
244
|
await this.syncer.sync();
|
|
@@ -238,6 +262,7 @@ var init_sonamu = __esmMin((() => {
|
|
|
238
262
|
logger: this.config.logging !== false ? getLogTapeFastifyLogger({ category: this.config.logging?.fastifyCategory ?? ["fastify"] }) : undefined
|
|
239
263
|
});
|
|
240
264
|
this.server = server;
|
|
265
|
+
this.websocketRuntime = new WebSocketRuntime(options.websocket);
|
|
241
266
|
if (options.storage) {
|
|
242
267
|
const { StorageManager } = await import("../storage/storage-manager.js");
|
|
243
268
|
this._storage = new StorageManager(options.storage);
|
|
@@ -263,6 +288,7 @@ var init_sonamu = __esmMin((() => {
|
|
|
263
288
|
await this.init(options?.doSilent, options?.enableSync);
|
|
264
289
|
}
|
|
265
290
|
this.server = server;
|
|
291
|
+
this.websocketRuntime ??= new WebSocketRuntime(this.config.server.websocket);
|
|
266
292
|
const timezone = this.config.api.timezone;
|
|
267
293
|
if (timezone) {
|
|
268
294
|
const { formatInTimeZone } = await import("date-fns-tz");
|
|
@@ -319,6 +345,15 @@ var init_sonamu = __esmMin((() => {
|
|
|
319
345
|
if (this.syncer.models[api.modelName] === undefined) {
|
|
320
346
|
throw new Error(`정의되지 않은 모델에 접근 ${api.modelName}`);
|
|
321
347
|
}
|
|
348
|
+
if (api.websocketOptions) {
|
|
349
|
+
server.route({
|
|
350
|
+
method: "GET",
|
|
351
|
+
url: this.config.api.route.prefix + api.path,
|
|
352
|
+
handler: this.createWebSocketUpgradeRequiredHandler(),
|
|
353
|
+
wsHandler: this.createWebSocketHandler(api, config)
|
|
354
|
+
});
|
|
355
|
+
continue;
|
|
356
|
+
}
|
|
322
357
|
server.route({
|
|
323
358
|
method: api.options.httpMethod ?? "GET",
|
|
324
359
|
url: this.config.api.route.prefix + api.path,
|
|
@@ -337,12 +372,22 @@ var init_sonamu = __esmMin((() => {
|
|
|
337
372
|
* 요청이 /api(정확히는 this.config.api.route.prefix)로 시작하지 않는 경우라면 null을 반환하며 끝냅니다.
|
|
338
373
|
*/
|
|
339
374
|
handleDevApiRequest(request, config) {
|
|
375
|
+
const matchedApi = this.findMatchedApi(request);
|
|
376
|
+
if (!matchedApi) {
|
|
377
|
+
throw new NotFoundException(SD("error.api.notFound"));
|
|
378
|
+
}
|
|
379
|
+
if (matchedApi.websocketOptions) {
|
|
380
|
+
return this.createWebSocketUpgradeRequiredHandler();
|
|
381
|
+
}
|
|
382
|
+
return this.createApiHandler(matchedApi, config);
|
|
383
|
+
}
|
|
384
|
+
findMatchedApi(request) {
|
|
340
385
|
const url = this.getPathnameFromUrl(request.url);
|
|
341
386
|
const method = request.method;
|
|
342
387
|
if (!url.startsWith(this.config.api.route.prefix)) {
|
|
343
|
-
return
|
|
388
|
+
return undefined;
|
|
344
389
|
}
|
|
345
|
-
|
|
390
|
+
return this.syncer.apis.find((api) => {
|
|
346
391
|
if (this.syncer.models[api.modelName] === undefined) {
|
|
347
392
|
return false;
|
|
348
393
|
}
|
|
@@ -351,19 +396,28 @@ var init_sonamu = __esmMin((() => {
|
|
|
351
396
|
const fullPath = this.config.api.route.prefix + api.path;
|
|
352
397
|
return this.isPathPatternMatch(fullPath, url);
|
|
353
398
|
});
|
|
354
|
-
if (!matchedApi) {
|
|
355
|
-
throw new NotFoundException(SD("error.api.notFound"));
|
|
356
|
-
}
|
|
357
|
-
return this.createApiHandler(matchedApi, config);
|
|
358
399
|
}
|
|
359
400
|
/**
|
|
360
401
|
* dev api 모드: Vite 없이 API 동적 라우팅만 제공합니다.
|
|
361
402
|
* HMR을 위해 catch-all에서 매 요청마다 syncer.apis를 조회합니다.
|
|
362
403
|
*/
|
|
363
404
|
setupDevServer(server, config) {
|
|
405
|
+
server.route({
|
|
406
|
+
method: "GET",
|
|
407
|
+
url: `${this.config.api.route.prefix}/*`,
|
|
408
|
+
handler: async (request, reply) => {
|
|
409
|
+
const handler = this.handleDevApiRequest(request, config);
|
|
410
|
+
if (handler) {
|
|
411
|
+
return handler(request, reply);
|
|
412
|
+
}
|
|
413
|
+
throw new NotFoundException(SD("error.api.notFound"));
|
|
414
|
+
},
|
|
415
|
+
wsHandler: async (connection, request) => {
|
|
416
|
+
await this.handleDevWebSocketRequest(connection.socket, request, config);
|
|
417
|
+
}
|
|
418
|
+
});
|
|
364
419
|
server.route({
|
|
365
420
|
method: [
|
|
366
|
-
"GET",
|
|
367
421
|
"HEAD",
|
|
368
422
|
"POST",
|
|
369
423
|
"PUT",
|
|
@@ -388,11 +442,16 @@ var init_sonamu = __esmMin((() => {
|
|
|
388
442
|
async setupDevServerWithVite(server, webPath, config) {
|
|
389
443
|
await server.register((await import("@fastify/middie")).default);
|
|
390
444
|
const vite = await import("vite");
|
|
445
|
+
const requiresDedicatedHmrServer = Boolean(this.config.server.plugins?.ws);
|
|
446
|
+
const hmr = resolveIntegratedViteHmrOptions({
|
|
447
|
+
httpServer: server.server,
|
|
448
|
+
requiresDedicatedWebSocketServer: requiresDedicatedHmrServer
|
|
449
|
+
});
|
|
391
450
|
this.viteServer = await vite.createServer({
|
|
392
451
|
root: webPath,
|
|
393
452
|
server: {
|
|
394
453
|
middlewareMode: true,
|
|
395
|
-
hmr
|
|
454
|
+
hmr
|
|
396
455
|
},
|
|
397
456
|
appType: "custom"
|
|
398
457
|
});
|
|
@@ -402,21 +461,41 @@ var init_sonamu = __esmMin((() => {
|
|
|
402
461
|
}
|
|
403
462
|
return this.viteServer.middlewares(req, res, next);
|
|
404
463
|
});
|
|
464
|
+
server.route({
|
|
465
|
+
method: "GET",
|
|
466
|
+
url: `${this.config.api.route.prefix}/*`,
|
|
467
|
+
handler: async (request, reply) => {
|
|
468
|
+
const result = this.handleDevApiRequest(request, config);
|
|
469
|
+
if (result) {
|
|
470
|
+
return result(request, reply);
|
|
471
|
+
}
|
|
472
|
+
throw new NotFoundException(SD("error.api.notFound"));
|
|
473
|
+
},
|
|
474
|
+
wsHandler: async (connection, request) => {
|
|
475
|
+
await this.handleDevWebSocketRequest(connection.socket, request, config);
|
|
476
|
+
}
|
|
477
|
+
});
|
|
405
478
|
server.route({
|
|
406
479
|
method: [
|
|
407
|
-
"GET",
|
|
408
480
|
"HEAD",
|
|
409
481
|
"POST",
|
|
410
482
|
"PUT",
|
|
411
483
|
"DELETE",
|
|
412
484
|
"PATCH"
|
|
413
485
|
],
|
|
414
|
-
url:
|
|
486
|
+
url: `${this.config.api.route.prefix}/*`,
|
|
415
487
|
handler: async (request, reply) => {
|
|
416
488
|
const result = this.handleDevApiRequest(request, config);
|
|
417
489
|
if (result) {
|
|
418
490
|
return result(request, reply);
|
|
419
491
|
}
|
|
492
|
+
throw new NotFoundException(SD("error.api.notFound"));
|
|
493
|
+
}
|
|
494
|
+
});
|
|
495
|
+
server.route({
|
|
496
|
+
method: ["GET", "HEAD"],
|
|
497
|
+
url: "/*",
|
|
498
|
+
handler: async (request, reply) => {
|
|
420
499
|
const url = request.url;
|
|
421
500
|
const { matchSSRRoute, renderSSR } = await import("../ssr/index.js");
|
|
422
501
|
const ssrMatch = matchSSRRoute(url);
|
|
@@ -443,8 +522,11 @@ var init_sonamu = __esmMin((() => {
|
|
|
443
522
|
server.addHook("onClose", async () => {
|
|
444
523
|
await this.viteServer.close();
|
|
445
524
|
});
|
|
446
|
-
const chalk = (await import("chalk")).default;
|
|
447
|
-
|
|
525
|
+
const chalk$1 = (await import("chalk")).default;
|
|
526
|
+
if ("port" in hmr) {
|
|
527
|
+
console.log(chalk$1.dim(`✓ Vite HMR using dedicated websocket port ${hmr.port} to avoid Fastify websocket conflicts`));
|
|
528
|
+
}
|
|
529
|
+
console.log(chalk$1.dim("✓ Vite dev server integrated"));
|
|
448
530
|
}
|
|
449
531
|
async setupStaticWebServer(server, config, globalCompressOptions) {
|
|
450
532
|
const webDistPath = path.join(this.apiRootPath, "web-dist", "client");
|
|
@@ -575,7 +657,12 @@ var init_sonamu = __esmMin((() => {
|
|
|
575
657
|
return async (request, reply) => {
|
|
576
658
|
const context = await this.createContext(config, request, reply);
|
|
577
659
|
return this.asyncLocalStorage.run({ context }, async () => {
|
|
578
|
-
(
|
|
660
|
+
runGuards({
|
|
661
|
+
guards: api.options.guards,
|
|
662
|
+
config,
|
|
663
|
+
request,
|
|
664
|
+
api
|
|
665
|
+
});
|
|
579
666
|
const { getZodObjectFromApi } = await import("./code-converters.js");
|
|
580
667
|
const ReqType = getZodObjectFromApi(api, this.syncer.types);
|
|
581
668
|
const which = api.options.httpMethod === "GET" ? "query" : "body";
|
|
@@ -667,6 +754,151 @@ var init_sonamu = __esmMin((() => {
|
|
|
667
754
|
});
|
|
668
755
|
};
|
|
669
756
|
}
|
|
757
|
+
createWebSocketUpgradeRequiredHandler() {
|
|
758
|
+
return async (_request, reply) => {
|
|
759
|
+
reply.header("connection", "Upgrade").header("upgrade", "websocket").status(426).send({ message: "WebSocket upgrade required" });
|
|
760
|
+
};
|
|
761
|
+
}
|
|
762
|
+
async handleDevWebSocketRequest(socket, request, config) {
|
|
763
|
+
const matchedApi = this.findMatchedApi(request);
|
|
764
|
+
if (!matchedApi?.websocketOptions) {
|
|
765
|
+
socket.close(1008, "WebSocket route not found");
|
|
766
|
+
return;
|
|
767
|
+
}
|
|
768
|
+
const handler = this.createWebSocketHandler(matchedApi, config);
|
|
769
|
+
await handler({ socket }, request);
|
|
770
|
+
}
|
|
771
|
+
createWebSocketHandler(api, config) {
|
|
772
|
+
return async (connection, request) => {
|
|
773
|
+
const socket = connection.socket;
|
|
774
|
+
let wsContext = null;
|
|
775
|
+
let rawWs = null;
|
|
776
|
+
try {
|
|
777
|
+
runGuards({
|
|
778
|
+
guards: api.options.guards,
|
|
779
|
+
config,
|
|
780
|
+
request,
|
|
781
|
+
api
|
|
782
|
+
});
|
|
783
|
+
const reqBody = await this.parseWebSocketRequestParams(api, request);
|
|
784
|
+
rawWs = this.websocketRuntime.registerConnection(socket, {
|
|
785
|
+
outEvents: api.websocketOptions.outEvents,
|
|
786
|
+
inEvents: api.websocketOptions.inEvents,
|
|
787
|
+
namespace: api.websocketOptions.namespace,
|
|
788
|
+
heartbeat: api.websocketOptions.heartbeat,
|
|
789
|
+
maxPayload: api.websocketOptions.maxPayload,
|
|
790
|
+
active: false
|
|
791
|
+
});
|
|
792
|
+
const scopedWs = this.createScopedWebSocketConnection(rawWs, () => wsContext);
|
|
793
|
+
wsContext = await this.createWebSocketContext(config, request, scopedWs);
|
|
794
|
+
this.websocketRuntime.activateConnection(rawWs.id);
|
|
795
|
+
const { ApiParamType } = await import("../types/types.js");
|
|
796
|
+
const args = api.parameters.map((param) => {
|
|
797
|
+
if (ApiParamType.isContext(param.type)) {
|
|
798
|
+
return wsContext;
|
|
799
|
+
}
|
|
800
|
+
return reqBody[param.name];
|
|
801
|
+
});
|
|
802
|
+
await this.asyncLocalStorage.run({ context: wsContext }, async () => {
|
|
803
|
+
await this.invokeModelMethod(api, args);
|
|
804
|
+
});
|
|
805
|
+
} catch (error) {
|
|
806
|
+
const closeDescriptor = resolveWebSocketCloseDescriptor(error);
|
|
807
|
+
if (rawWs) {
|
|
808
|
+
rawWs.close(closeDescriptor.code, closeDescriptor.reason);
|
|
809
|
+
} else if (socket.readyState < 2) {
|
|
810
|
+
socket.close(closeDescriptor.code, closeDescriptor.reason);
|
|
811
|
+
}
|
|
812
|
+
if (this.server?.log) {
|
|
813
|
+
const payload = {
|
|
814
|
+
err: error,
|
|
815
|
+
modelName: api.modelName,
|
|
816
|
+
methodName: api.methodName,
|
|
817
|
+
path: api.path
|
|
818
|
+
};
|
|
819
|
+
if (closeDescriptor.logLevel === "warn") {
|
|
820
|
+
this.server.log.warn(payload, closeDescriptor.reason);
|
|
821
|
+
} else {
|
|
822
|
+
this.server.log.error(payload, closeDescriptor.reason);
|
|
823
|
+
}
|
|
824
|
+
} else {
|
|
825
|
+
if (closeDescriptor.logLevel === "warn") {
|
|
826
|
+
console.warn(closeDescriptor.reason, error);
|
|
827
|
+
} else {
|
|
828
|
+
console.error(closeDescriptor.reason, error);
|
|
829
|
+
}
|
|
830
|
+
}
|
|
831
|
+
}
|
|
832
|
+
};
|
|
833
|
+
}
|
|
834
|
+
createScopedWebSocketConnection(ws, getContext) {
|
|
835
|
+
const runInContext = (callback) => {
|
|
836
|
+
const context = getContext();
|
|
837
|
+
if (!context) {
|
|
838
|
+
return callback();
|
|
839
|
+
}
|
|
840
|
+
return this.asyncLocalStorage.run({ context }, callback);
|
|
841
|
+
};
|
|
842
|
+
return {
|
|
843
|
+
get id() {
|
|
844
|
+
return ws.id;
|
|
845
|
+
},
|
|
846
|
+
get namespace() {
|
|
847
|
+
return ws.namespace;
|
|
848
|
+
},
|
|
849
|
+
get closed() {
|
|
850
|
+
return ws.closed;
|
|
851
|
+
},
|
|
852
|
+
transport: "ws",
|
|
853
|
+
publishUntyped(event, data) {
|
|
854
|
+
ws.publishUntyped(event, data);
|
|
855
|
+
},
|
|
856
|
+
close(code, reason) {
|
|
857
|
+
ws.close(code, reason);
|
|
858
|
+
},
|
|
859
|
+
onClose(callback) {
|
|
860
|
+
ws.onClose(() => runInContext(callback));
|
|
861
|
+
},
|
|
862
|
+
onMessage(event, handler) {
|
|
863
|
+
ws.onMessage(event, (data) => runInContext(() => handler(data)));
|
|
864
|
+
},
|
|
865
|
+
publish(event, data) {
|
|
866
|
+
ws.publish(event, data);
|
|
867
|
+
},
|
|
868
|
+
waitForClose() {
|
|
869
|
+
return ws.waitForClose();
|
|
870
|
+
},
|
|
871
|
+
join(roomId) {
|
|
872
|
+
ws.join(roomId);
|
|
873
|
+
},
|
|
874
|
+
leave(roomId) {
|
|
875
|
+
ws.leave(roomId);
|
|
876
|
+
},
|
|
877
|
+
setUserId(userId) {
|
|
878
|
+
ws.setUserId(userId);
|
|
879
|
+
},
|
|
880
|
+
clearUserId() {
|
|
881
|
+
ws.clearUserId();
|
|
882
|
+
}
|
|
883
|
+
};
|
|
884
|
+
}
|
|
885
|
+
async parseWebSocketRequestParams(api, request) {
|
|
886
|
+
const { getZodObjectFromApi } = await import("./code-converters.js");
|
|
887
|
+
const ReqType = getZodObjectFromApi(api, this.syncer.types);
|
|
888
|
+
try {
|
|
889
|
+
const { fastifyCaster } = await import("./caster.js");
|
|
890
|
+
return fastifyCaster(ReqType).parse(request.query ?? {});
|
|
891
|
+
} catch (e) {
|
|
892
|
+
const { ZodError } = await import("zod");
|
|
893
|
+
if (e instanceof ZodError) {
|
|
894
|
+
const { humanizeZodError } = await import("../utils/zod-error.js");
|
|
895
|
+
const messages = humanizeZodError(e).map((issue) => issue.message).join(" ");
|
|
896
|
+
const { BadRequestException } = await import("../exceptions/so-exceptions.js");
|
|
897
|
+
throw new BadRequestException(messages, { zodError: e });
|
|
898
|
+
}
|
|
899
|
+
throw e;
|
|
900
|
+
}
|
|
901
|
+
}
|
|
670
902
|
/**
|
|
671
903
|
* URL에서 path params를 추출합니다.
|
|
672
904
|
* 예: pattern="/admin/companies/:companyId", url="/admin/companies/123" → { companyId: "123" }
|
|
@@ -762,7 +994,7 @@ var init_sonamu = __esmMin((() => {
|
|
|
762
994
|
async invokeModelMethod(api, args, reply) {
|
|
763
995
|
const model = this.syncer.models[api.modelName];
|
|
764
996
|
const result = await model[api.methodName].apply(model, args);
|
|
765
|
-
reply
|
|
997
|
+
reply?.type(api.options.contentType ?? "application/json");
|
|
766
998
|
return result;
|
|
767
999
|
}
|
|
768
1000
|
async createContext(config, request, reply) {
|
|
@@ -771,7 +1003,8 @@ var init_sonamu = __esmMin((() => {
|
|
|
771
1003
|
const locale = this.detectLocale(request.headers["accept-language"], this.config.i18n.supportedLocales) ?? this.config.i18n.defaultLocale;
|
|
772
1004
|
const headers = convertFastifyHeadersToStandard(request.headers);
|
|
773
1005
|
const session = await this._auth?.api.getSession({ headers }) ?? null;
|
|
774
|
-
const context =
|
|
1006
|
+
const context = await Promise.resolve(config.contextProvider({
|
|
1007
|
+
transport: "http",
|
|
775
1008
|
request,
|
|
776
1009
|
reply,
|
|
777
1010
|
headers: request.headers,
|
|
@@ -780,9 +1013,50 @@ var init_sonamu = __esmMin((() => {
|
|
|
780
1013
|
locale,
|
|
781
1014
|
user: session?.user ?? null,
|
|
782
1015
|
session: session?.session ?? null
|
|
783
|
-
}, request, reply))
|
|
1016
|
+
}, request, reply));
|
|
784
1017
|
return context;
|
|
785
1018
|
}
|
|
1019
|
+
async createWebSocketContext(config, request, ws) {
|
|
1020
|
+
const locale = this.detectLocale(request.headers["accept-language"], this.config.i18n.supportedLocales) ?? this.config.i18n.defaultLocale;
|
|
1021
|
+
const headers = convertFastifyHeadersToStandard(request.headers);
|
|
1022
|
+
const session = await this._auth?.api.getSession({ headers }) ?? null;
|
|
1023
|
+
const defaultContext = {
|
|
1024
|
+
transport: "ws",
|
|
1025
|
+
request,
|
|
1026
|
+
headers: request.headers,
|
|
1027
|
+
ws,
|
|
1028
|
+
naiteStore: new Map(),
|
|
1029
|
+
locale,
|
|
1030
|
+
user: session?.user ?? null,
|
|
1031
|
+
session: session?.session ?? null
|
|
1032
|
+
};
|
|
1033
|
+
if (config.websocketContextProvider) {
|
|
1034
|
+
return { ...await Promise.resolve(config.websocketContextProvider(defaultContext, request)) };
|
|
1035
|
+
}
|
|
1036
|
+
const replyStub = createWebSocketReplyStub();
|
|
1037
|
+
const createSSE = (_events) => {
|
|
1038
|
+
throw new Error("createSSE is not available in websocket context. Define websocketContextProvider if your context setup depends on SSE helpers.");
|
|
1039
|
+
};
|
|
1040
|
+
const httpLikeContext = await Promise.resolve(config.contextProvider({
|
|
1041
|
+
transport: "http",
|
|
1042
|
+
request,
|
|
1043
|
+
reply: replyStub,
|
|
1044
|
+
headers: request.headers,
|
|
1045
|
+
createSSE,
|
|
1046
|
+
naiteStore: defaultContext.naiteStore,
|
|
1047
|
+
locale,
|
|
1048
|
+
user: defaultContext.user,
|
|
1049
|
+
session: defaultContext.session
|
|
1050
|
+
}, request, replyStub));
|
|
1051
|
+
const { transport: _transport, reply: _reply, createSSE: _createSSE, bufferedFiles: _bufferedFiles, uploadedFiles: _uploadedFiles, ...rest } = httpLikeContext;
|
|
1052
|
+
return {
|
|
1053
|
+
...rest,
|
|
1054
|
+
transport: "ws",
|
|
1055
|
+
request,
|
|
1056
|
+
headers: request.headers,
|
|
1057
|
+
ws
|
|
1058
|
+
};
|
|
1059
|
+
}
|
|
786
1060
|
/**
|
|
787
1061
|
* Accept-Language 헤더에서 지원하는 locale을 찾습니다.
|
|
788
1062
|
* @example "ko-KR,ko;q=0.9,en;q=0.8" → "ko"
|
|
@@ -796,33 +1070,24 @@ var init_sonamu = __esmMin((() => {
|
|
|
796
1070
|
return langs.find((lang) => supported.includes(lang));
|
|
797
1071
|
}
|
|
798
1072
|
async startWatcher() {
|
|
799
|
-
const
|
|
800
|
-
|
|
801
|
-
|
|
802
|
-
|
|
803
|
-
|
|
804
|
-
|
|
805
|
-
|
|
806
|
-
|
|
807
|
-
|
|
808
|
-
|
|
809
|
-
|
|
810
|
-
|
|
811
|
-
|
|
812
|
-
|
|
813
|
-
|
|
814
|
-
|
|
815
|
-
|
|
816
|
-
|
|
817
|
-
console.log(chalk.bold(`Detected(${event}): ${chalk.blue(relativePath)} - Restarting...`));
|
|
818
|
-
process.kill(process.pid, "SIGUSR2");
|
|
819
|
-
return;
|
|
820
|
-
}
|
|
821
|
-
await this.handleFileChange(event, absolutePath);
|
|
822
|
-
} catch (e) {
|
|
823
|
-
console.error(e);
|
|
824
|
-
}
|
|
825
|
-
});
|
|
1073
|
+
const { setupWatcher } = await import("../syncer/watcher.js");
|
|
1074
|
+
this.watcher = await setupWatcher((fileEvents) => this.runHmrSyncCycle(fileEvents));
|
|
1075
|
+
}
|
|
1076
|
+
/**
|
|
1077
|
+
* Watcher가 100ms batch로 모은 fileEvents 하나에 대해 한 번의 HMR/sync 사이클을 돕니다.
|
|
1078
|
+
* batch 큐잉 덕에 한 시점에 하나만 실행됨이 보장됩니다 (event-batcher가 직렬화).
|
|
1079
|
+
*/
|
|
1080
|
+
async runHmrSyncCycle(fileEvents) {
|
|
1081
|
+
const startedAt = Date.now();
|
|
1082
|
+
for (const [filePath, event] of fileEvents) {
|
|
1083
|
+
const relativePath = path.relative(this.appRootPath, filePath);
|
|
1084
|
+
console.log(chalk.bold(`Detected(${event}): ${chalk.blue(relativePath)}`));
|
|
1085
|
+
}
|
|
1086
|
+
await this.syncer.hmrAndSync(fileEvents);
|
|
1087
|
+
await this.syncer.renewChecksums();
|
|
1088
|
+
const totalTime = Date.now() - startedAt;
|
|
1089
|
+
const msg = `HMR Done! ${chalk.bold.white(`${totalTime}ms`)}`;
|
|
1090
|
+
console.log(chalk.black.bgGreen(centerText(msg)));
|
|
826
1091
|
}
|
|
827
1092
|
async runScript(fn) {
|
|
828
1093
|
await this.init(true, false, undefined, false);
|
|
@@ -875,10 +1140,51 @@ var init_sonamu = __esmMin((() => {
|
|
|
875
1140
|
for (const [key, pluginName] of Object.entries(pluginsModules)) {
|
|
876
1141
|
await registerPlugin(key, pluginName);
|
|
877
1142
|
}
|
|
1143
|
+
if (plugins.ws) {
|
|
1144
|
+
await this.ensureWebSocketPlugin(server);
|
|
1145
|
+
}
|
|
878
1146
|
if (plugins.custom) {
|
|
879
1147
|
plugins.custom(server);
|
|
880
1148
|
}
|
|
881
1149
|
}
|
|
1150
|
+
async ensureWebSocketPlugin(server) {
|
|
1151
|
+
if (this.websocketPluginServers.has(server)) {
|
|
1152
|
+
return;
|
|
1153
|
+
}
|
|
1154
|
+
const pluginOption = this.config.server.plugins?.ws;
|
|
1155
|
+
if (!pluginOption) {
|
|
1156
|
+
return;
|
|
1157
|
+
}
|
|
1158
|
+
const websocketPlugin = (await import("@fastify/websocket")).default;
|
|
1159
|
+
const resolvedPluginOptions = resolveWebSocketPluginOptions({
|
|
1160
|
+
rawPluginOption: pluginOption,
|
|
1161
|
+
apis: this.syncer.apis
|
|
1162
|
+
});
|
|
1163
|
+
if (resolvedPluginOptions) {
|
|
1164
|
+
await server.register(websocketPlugin, resolvedPluginOptions);
|
|
1165
|
+
} else {
|
|
1166
|
+
await server.register(websocketPlugin);
|
|
1167
|
+
}
|
|
1168
|
+
this.websocketPluginServers.add(server);
|
|
1169
|
+
this.warnOnPotentialWebSocketTimeoutConflicts(server);
|
|
1170
|
+
}
|
|
1171
|
+
warnOnPotentialWebSocketTimeoutConflicts(server) {
|
|
1172
|
+
const heartbeats = this.syncer.apis.map((api) => api.websocketOptions?.heartbeat ?? 3e4).filter((heartbeat) => heartbeat > 0);
|
|
1173
|
+
if (heartbeats.length === 0) {
|
|
1174
|
+
return;
|
|
1175
|
+
}
|
|
1176
|
+
const keepAliveTimeout = this.config.server.fastify?.keepAliveTimeout;
|
|
1177
|
+
if (!keepAliveTimeout || keepAliveTimeout <= 0) {
|
|
1178
|
+
return;
|
|
1179
|
+
}
|
|
1180
|
+
const largestHeartbeat = Math.max(...heartbeats);
|
|
1181
|
+
if (largestHeartbeat >= keepAliveTimeout) {
|
|
1182
|
+
server.log.warn({
|
|
1183
|
+
keepAliveTimeout,
|
|
1184
|
+
largestHeartbeat
|
|
1185
|
+
}, "WebSocket heartbeat is greater than or equal to keepAliveTimeout; align infrastructure idle timeouts to avoid unexpected disconnects.");
|
|
1186
|
+
}
|
|
1187
|
+
}
|
|
882
1188
|
/**
|
|
883
1189
|
* better-auth 라우트를 등록합니다.
|
|
884
1190
|
* /api/auth/* 경로로 인증 API가 자동 등록됩니다.
|
|
@@ -916,11 +1222,11 @@ var init_sonamu = __esmMin((() => {
|
|
|
916
1222
|
});
|
|
917
1223
|
}
|
|
918
1224
|
async printStartupSummary() {
|
|
919
|
-
const chalk = (await import("chalk")).default;
|
|
1225
|
+
const chalk$1 = (await import("chalk")).default;
|
|
920
1226
|
const env = process.env.NODE_ENV ?? "development";
|
|
921
1227
|
const activePreset = env === "production" ? "production_master" : "development_master";
|
|
922
|
-
const dim = (msg) => console.log(chalk.dim(`✓ ${msg}`));
|
|
923
|
-
const green = (msg) => console.log(chalk.green(`✓ ${msg}`));
|
|
1228
|
+
const dim = (msg) => console.log(chalk$1.dim(`✓ ${msg}`));
|
|
1229
|
+
const green = (msg) => console.log(chalk$1.green(`✓ ${msg}`));
|
|
924
1230
|
dim(`Config loaded${formatTime(this._configElapsed)}`);
|
|
925
1231
|
green("DB");
|
|
926
1232
|
const { isLocal } = await import("../utils/controller.js");
|
|
@@ -931,11 +1237,11 @@ var init_sonamu = __esmMin((() => {
|
|
|
931
1237
|
const host = conn?.host ?? "localhost";
|
|
932
1238
|
const addr = `@ ${host}:${conn?.port ?? 5432}/${conn?.database ?? this.config.database.name}`;
|
|
933
1239
|
const padded = name.padEnd(maxLen);
|
|
934
|
-
const remoteTag = isLocal() && !isLocalHost(host) ? chalk.yellow(` \u26a0 remote`) : "";
|
|
1240
|
+
const remoteTag = isLocal() && !isLocalHost(host) ? chalk$1.yellow(` \u26a0 remote`) : "";
|
|
935
1241
|
if (name === activePreset) {
|
|
936
|
-
console.log(chalk.green(` \u25b8 ${padded} ${addr}`) + remoteTag);
|
|
1242
|
+
console.log(chalk$1.green(` \u25b8 ${padded} ${addr}`) + remoteTag);
|
|
937
1243
|
} else {
|
|
938
|
-
console.log(chalk.dim(` ${padded} ${addr}`) + remoteTag);
|
|
1244
|
+
console.log(chalk$1.dim(` ${padded} ${addr}`) + remoteTag);
|
|
939
1245
|
}
|
|
940
1246
|
}
|
|
941
1247
|
if (this.config.server.auth) {
|
|
@@ -1011,43 +1317,23 @@ var init_sonamu = __esmMin((() => {
|
|
|
1011
1317
|
await this.workflows.startWorker();
|
|
1012
1318
|
await options.lifecycle?.onStart?.(server);
|
|
1013
1319
|
}).catch(async (err) => {
|
|
1014
|
-
const chalk = (await import("chalk")).default;
|
|
1015
|
-
console.error(chalk.red("Failed to start server:", err));
|
|
1320
|
+
const chalk$1 = (await import("chalk")).default;
|
|
1321
|
+
console.error(chalk$1.red("Failed to start server:", err));
|
|
1016
1322
|
await shutdown();
|
|
1017
1323
|
});
|
|
1018
1324
|
}
|
|
1019
|
-
async handleFileChange(event, filePath) {
|
|
1020
|
-
if (this.pendingFiles.length === 0) {
|
|
1021
|
-
this.hmrStartTime = Date.now();
|
|
1022
|
-
}
|
|
1023
|
-
this.pendingFiles.push(filePath);
|
|
1024
|
-
const relativePath = path.relative(this.apiRootPath, filePath);
|
|
1025
|
-
const chalk = (await import("chalk")).default;
|
|
1026
|
-
console.log(chalk.bold(`Detected(${event}): ${chalk.blue(relativePath)}`));
|
|
1027
|
-
await this.syncer.syncFromWatcher(event, filePath);
|
|
1028
|
-
this.pendingFiles = this.pendingFiles.slice(1);
|
|
1029
|
-
if (this.pendingFiles.length === 0) {
|
|
1030
|
-
await this.finishHMR();
|
|
1031
|
-
}
|
|
1032
|
-
}
|
|
1033
|
-
async finishHMR() {
|
|
1034
|
-
await this.syncer.renewChecksums();
|
|
1035
|
-
const endTime = Date.now();
|
|
1036
|
-
const totalTime = endTime - this.hmrStartTime;
|
|
1037
|
-
const [chalk, { centerText }] = await Promise.all([(await import("chalk")).default, import("../utils/console-util.js")]);
|
|
1038
|
-
const msg = `HMR Done! ${chalk.bold.white(`${totalTime}ms`)}`;
|
|
1039
|
-
console.log(chalk.black.bgGreen(centerText(msg)));
|
|
1040
|
-
}
|
|
1041
1325
|
async destroy() {
|
|
1042
1326
|
const { BaseModel } = await import("../database/base-model.js");
|
|
1043
1327
|
await BaseModel.destroy();
|
|
1044
1328
|
await Promise.allSettled([
|
|
1329
|
+
this._websocketRuntime?.shutdown() ?? Promise.resolve(),
|
|
1045
1330
|
this._workflows?.destroy() ?? Promise.resolve(),
|
|
1046
1331
|
this._cache?.disconnect() ?? Promise.resolve(),
|
|
1047
1332
|
this._devVitestManager?.shutdown() ?? Promise.resolve(),
|
|
1048
1333
|
this.watcher?.close() ?? Promise.resolve(),
|
|
1049
1334
|
dispose()
|
|
1050
1335
|
]);
|
|
1336
|
+
this._websocketRuntime = null;
|
|
1051
1337
|
}
|
|
1052
1338
|
};
|
|
1053
1339
|
Sonamu = new SonamuClass();
|
|
@@ -1061,5 +1347,5 @@ var init_sonamu = __esmMin((() => {
|
|
|
1061
1347
|
|
|
1062
1348
|
//#endregion
|
|
1063
1349
|
init_sonamu();
|
|
1064
|
-
export { Sonamu, init_sonamu };
|
|
1065
|
-
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"sonamu.js","names":["DB","authOptions: BetterAuthOptions","globalCompressOptions: CompressOptions | undefined","fs","cacheReq: CacheControlRequest","filePath","csrCacheReq: CacheControlRequest","mimeLookup","context: Context","reqBody: {\n          [key: string]: unknown;\n        }","files: {\n          bufferedFiles: BufferedFile[];\n          uploadedFiles: UploadedFile[];\n        }","fields: Record<string, string>","keyGenerator: KeyGenerator","params: Record<string, string>","path","logtapeDispose"],"sources":["../../src/api/sonamu.ts"],"sourcesContent":["import assert from \"assert\";\nimport { AsyncLocalStorage } from \"async_hooks\";\nimport fs from \"fs/promises\";\nimport { type IncomingMessage, type Server, type ServerResponse } from \"http\";\nimport os from \"os\";\nimport path from \"path\";\n\nimport { dispose as logtapeDispose } from \"@logtape/logtape\";\nimport { type Auth, type BetterAuthOptions } from \"better-auth\";\nimport { type FSWatcher } from \"chokidar\";\nimport { type FastifyInstance, type FastifyReply, type FastifyRequest } from \"fastify\";\nimport mime, { lookup as mimeLookup } from \"mime-types\";\nimport { type ZodObject } from \"zod\";\n\nimport { BASE_FIELD_MAPPINGS } from \"../auth/better-auth-entities\";\nimport { applyCacheHeaders, CachePresets } from \"../cache-control/cache-control\";\nimport { type CacheControlConfig, type CacheControlRequest } from \"../cache-control/types\";\nimport { type CacheConfig, type CacheManager } from \"../cache/types\";\nimport { toFastifyCompressOption } from \"../compress/compress\";\nimport { type CompressOptions } from \"../compress/types\";\nimport { DB } from \"../database/db\";\nimport { type SonamuDBConfig } from \"../database/db\";\nimport { SD, setSDConfig } from \"../dict/sd\";\nimport { type LocalizedString } from \"../dict/types\";\nimport { NotFoundException } from \"../exceptions/so-exceptions\";\nimport { BufferedFile } from \"../storage/buffered-file\";\nimport { type StorageManager } from \"../storage/storage-manager\";\nimport { type KeyGenerator } from \"../storage/types\";\nimport { UploadedFile } from \"../storage/uploaded-file\";\nimport { createMockSSEFactory } from \"../stream/sse\";\nimport { type Syncer } from \"../syncer/syncer\";\nimport { type WorkflowManager } from \"../tasks/workflow-manager\";\nimport { type DevVitestManager } from \"../testing/dev-vitest-manager\";\nimport { type SonamuFastifyConfig } from \"../types/types\";\nimport { isDaemonServer } from \"../utils/controller\";\nimport { exists, fileExists } from \"../utils/fs-utils\";\nimport { type AbsolutePath } from \"../utils/path-utils\";\nimport { convertFastifyHeadersToStandard, merge } from \"../utils/utils\";\nimport { type SonamuConfig, type SonamuServerOptions, type SonamuTaskOptions } from \"./config\";\nimport { type Context } from \"./context\";\nimport { type ExtendedApi } from \"./decorators\";\nimport { getSecrets } from \"./secret\";\nimport { type SonamuSecrets } from \"./secret\";\n\nclass SonamuClass {\n  public isInitialized: boolean = false;\n  public forTesting: boolean = false;\n  public asyncLocalStorage: AsyncLocalStorage<{\n    context: Context;\n  }> = new AsyncLocalStorage();\n\n  public getContext(): Context {\n    const store = this.asyncLocalStorage.getStore();\n    if (store?.context) {\n      return store.context;\n    }\n\n    if (process.env.NODE_ENV === \"test\") {\n      // 테스팅 환경에서 컨텍스트가 주입되지 않은 경우 빈 컨텍스트 리턴\n      return {\n        request: null,\n        reply: null,\n        headers: {},\n        createSSE: (schema: ZodObject) => createMockSSEFactory(schema),\n        // oxlint-disable-next-line @typescript-eslint/no-explicit-any -- 테스팅 환경에서 컨텍스트가 주입되지 않은 경우 빈 컨텍스트 리턴\n        naiteStore: new Map<string, any>(),\n      } as unknown as Context;\n    } else {\n      throw new Error(\"Sonamu cannot find context\");\n    }\n  }\n\n  private _apiRootPath: AbsolutePath | null = null;\n  set apiRootPath(apiRootPath: AbsolutePath) {\n    this._apiRootPath = apiRootPath;\n  }\n  get apiRootPath(): AbsolutePath {\n    if (this._apiRootPath === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n    return this._apiRootPath;\n  }\n  get appRootPath(): string {\n    return this.apiRootPath.split(path.sep).slice(0, -1).join(path.sep);\n  }\n\n  private _dbConfig: SonamuDBConfig | null = null;\n  set dbConfig(dbConfig: SonamuDBConfig) {\n    this._dbConfig = dbConfig;\n  }\n  get dbConfig(): SonamuDBConfig {\n    if (this._dbConfig === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n    return this._dbConfig;\n  }\n\n  private _syncer: Syncer | null = null;\n  set syncer(syncer: Syncer) {\n    this._syncer = syncer;\n  }\n  get syncer(): Syncer {\n    if (this._syncer === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n    return this._syncer;\n  }\n\n  private _config: SonamuConfig | null = null;\n  set config(config: SonamuConfig) {\n    this._config = config;\n  }\n  get config(): SonamuConfig {\n    if (this._config === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n    return this._config;\n  }\n\n  public readonly secrets: SonamuSecrets = getSecrets();\n\n  private _storage: StorageManager | null = null;\n  /**\n   * StorageManager 인스턴스\n   */\n  get storage(): StorageManager {\n    if (!this._storage) {\n      throw new Error(\"Storage has not been initialized. Check storage config.\");\n    }\n    return this._storage;\n  }\n\n  private _cache: CacheManager | null = null;\n  /**\n   * CacheManager 인스턴스 (BentoCache)\n   */\n  get cache(): CacheManager {\n    if (!this._cache) {\n      throw new Error(\"Cache has not been initialized. Check cache config in sonamu.config.ts.\");\n    }\n    return this._cache;\n  }\n\n  private _workflows: WorkflowManager | null = null;\n  get workflows(): WorkflowManager {\n    if (this._workflows === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n\n    return this._workflows;\n  }\n\n  private _auth: Auth<BetterAuthOptions> | null = null;\n  get auth(): Auth<BetterAuthOptions> {\n    if (!this._auth) {\n      throw new Error(\"Auth has not been initialized. Check auth config in sonamu.config.ts.\");\n    }\n    return this._auth;\n  }\n\n  private _devVitestManager: DevVitestManager | null = null;\n  get devVitestManager(): DevVitestManager | null {\n    return this._devVitestManager;\n  }\n  set devVitestManager(manager: DevVitestManager | null) {\n    this._devVitestManager = manager;\n  }\n\n  // HMR 처리\n  public watcher: FSWatcher | null = null;\n  private pendingFiles: string[] = [];\n  private hmrStartTime: number = 0;\n\n  public server: FastifyInstance | null = null;\n\n  async initForTesting() {\n    await this.init(true, false, undefined, true);\n  }\n\n  async init(\n    doSilent: boolean = false,\n    enableSync: boolean = true,\n    apiRootPath?: AbsolutePath,\n    forTesting: boolean = false,\n  ) {\n    this.forTesting = forTesting;\n\n    if (this.isInitialized) {\n      return;\n    }\n\n    const initStart = performance.now();\n\n    // API 루트 패스\n    const { findApiRootPath } = await import(\"../utils/utils\");\n    this.apiRootPath = apiRootPath ?? findApiRootPath();\n\n    // 설정을 로딩하는 것부터 시작\n    const configStart = performance.now();\n    const { loadConfig } = await import(\"./config\");\n    this.config = await loadConfig(this.apiRootPath);\n    const configTime = performance.now() - configStart;\n    setSDConfig(this.config.i18n);\n    // sonamu.config.ts 기본값 설정\n    this.config.database.database = this.config.database.database ?? \"pg\";\n    this.config.database.defaultOptions.client = this.config.database.database ?? \"pg\";\n\n    // 로깅 설정\n    const { configureLogTape } = await import(\"../logger/configure\");\n    if (this.config.logging !== false) {\n      await configureLogTape({\n        ...this.config.logging,\n      });\n    }\n\n    // DB 로드\n    const { DB } = await import(\"../database/db\");\n    this.dbConfig = DB.generateDBConfig(this.config.database);\n    DB.setConfig(this.dbConfig);\n\n    // Entity 로드\n    // 테스트에서도 Entity 정보는 필요합니다.\n    // upsert가 제대로 작동하려면 entity의 unique index 정보가 필요하기 때문입니다.\n    const { EntityManager } = await import(\"../entity/entity-manager\");\n    await EntityManager.autoload(doSilent);\n\n    // Cache 초기화\n    await this.initializeCache(this.config.server.cache, forTesting);\n\n    // BetterAuth 초기화\n    const authConfig = this.config.server.auth;\n    if (authConfig) {\n      // 사용자 설정과 기본값을 merge\n      const mergedFieldMappings = merge(BASE_FIELD_MAPPINGS, authConfig);\n\n      // better-auth 인스턴스 생성\n      const { betterAuth } = await import(\"better-auth\");\n      const { sonamuKnexAdapter } = await import(\"../auth/knex-adapter\");\n\n      const authOptions: BetterAuthOptions = {\n        database: sonamuKnexAdapter(),\n        ...mergedFieldMappings,\n      };\n      this._auth = betterAuth(authOptions);\n    }\n\n    // 테스팅인 경우 싱크 없이 중단\n    if (forTesting) {\n      this.isInitialized = true;\n      return;\n    }\n\n    // Task 등록\n    await this.initializeWorkflows(this.config.tasks);\n\n    // Syncer\n    const { Syncer } = await import(\"../syncer/syncer\");\n    this.syncer = new Syncer();\n\n    // Autoload: Models / Types / APIs / Workflows / Templates / SSR Routes\n    await this.syncer.autoloadTypes();\n    await this.syncer.autoloadModels();\n    await this.syncer.autoloadApis();\n    await this.syncer.autoloadWorkflows();\n    const { TemplateManager } = await import(\"../template\");\n    await TemplateManager.autoload();\n    await this.syncer.autoloadSSRRoutes();\n\n    const { isLocal, isTest, isHotReloadServer } = await import(\"../utils/controller\");\n    if (isLocal() && !isTest() && isHotReloadServer() && enableSync) {\n      await this.syncer.sync();\n      await this.startWatcher();\n    }\n\n    this.isInitialized = true;\n    this._initElapsed = performance.now() - initStart;\n    this._configElapsed = configTime;\n  }\n\n  private _initElapsed = 0;\n  private _configElapsed = 0;\n\n  async createServer(initOptions?: { enableSync?: boolean; doSilent?: boolean }) {\n    if (!this.isInitialized) {\n      await this.init(initOptions?.doSilent, initOptions?.enableSync);\n    }\n\n    const options = this.config.server;\n    const { default: fastify } = await import(\"fastify\");\n    const { getLogTapeFastifyLogger } = await import(\"@logtape/fastify\");\n    const server = fastify({\n      ...options.fastify,\n      logger:\n        this.config.logging !== false\n          ? getLogTapeFastifyLogger({\n              category: this.config.logging?.fastifyCategory ?? [\"fastify\"],\n            })\n          : undefined,\n    });\n    this.server = server;\n\n    // Storage 설정 → StorageManager 생성\n    if (options.storage) {\n      const { StorageManager } = await import(\"../storage/storage-manager\");\n      this._storage = new StorageManager(options.storage);\n    }\n\n    // 플러그인 등록\n    if (options.plugins) {\n      await this.registerPlugins(server, options.plugins);\n    }\n\n    if (options.auth) {\n      await this.registerBetterAuth(server, options.auth);\n    }\n\n    // API 라우팅 설정\n    await this.withFastify(server, options.apiConfig, {\n      enableSync: initOptions?.enableSync,\n      doSilent: initOptions?.doSilent,\n    });\n\n    // 서버 시작\n    await this.boot(server, options);\n\n    if (!initOptions?.doSilent) {\n      this.printStartupSummary();\n    }\n\n    return server;\n  }\n\n  async withFastify(\n    server: FastifyInstance<Server, IncomingMessage, ServerResponse>,\n    config: SonamuFastifyConfig,\n    options?: {\n      enableSync?: boolean;\n      doSilent?: boolean;\n    },\n  ) {\n    if (!this.isInitialized) {\n      await this.init(options?.doSilent, options?.enableSync);\n    }\n\n    this.server = server;\n\n    // timezone 설정\n    const timezone = this.config.api.timezone;\n    if (timezone) {\n      // 타임존에 맞게 응답 날짜 스트링을 변환해주어야 합니다.\n      // 가령 timezone이 \"Asia/Seoul\" 이면\n      // \"2025-11-21T00:00:00.000Z\" 를 \"2025-11-21T09:00:00+09:00\" 으로 변환해주어야 합니다.\n      const { formatInTimeZone } = await import(\"date-fns-tz\");\n\n      // ISO 8601 날짜 형식 정규식 (예: 2024-01-15T09:30:00.000Z)\n      const ISO_DATE_REGEX = /^\\d{4}-\\d{2}-\\d{2}T\\d{2}:\\d{2}:\\d{2}(\\.\\d{3})?Z$/;\n\n      // T를 둘러싼 작은따옴표가 없다면 \"2025-11-19176354618900018:56:29+09:00\"와 같은 결과가 나옵니다.\n      // 이는 date-fns 특입니다.\n      // 이렇게 해도 괜찮습니다. \"2025-11-19T18:56:29+09:00\" 모양으로 잘 나옵니다.\n      const DATE_FORMAT = \"yyyy-MM-dd'T'HH:mm:ssXXX\";\n\n      server.setReplySerializer((payload) => {\n        return JSON.stringify(payload, (_key, value) => {\n          if (typeof value === \"string\" && ISO_DATE_REGEX.test(value)) {\n            return formatInTimeZone(\n              new Date(value),\n              timezone as `${string}/${string}`,\n              DATE_FORMAT,\n            );\n          }\n          return value;\n        });\n      });\n      // Timezone 로그는 printStartupSummary에서 통합 출력\n    }\n\n    // 전체 라우팅 리스트\n    server.get(\n      `${this.config.api.route.prefix}/routes`,\n      async (_request, _reply): Promise<typeof this.syncer.apis> => {\n        return this.syncer.apis;\n      },\n    );\n\n    // Healthcheck API\n    server.get(\n      `${this.config.api.route.prefix}/healthcheck`,\n      async (_request, _reply): Promise<string> => {\n        return \"ok\";\n      },\n    );\n\n    // Sonamu UI API (로컬 환경에서만)\n    const { isLocal } = await import(\"../utils/controller\");\n    if (isLocal()) {\n      const { sonamuUIApiPlugin } = await import(\"../ui/api\");\n      server.register(sonamuUIApiPlugin);\n    }\n\n    // DevRunner 테스트 엔드포인트 (로컬 환경 + devRunner 활성화 시)\n    if (isLocal() && this.config.test?.devRunner?.enabled) {\n      const { registerDevTestRoutes } = await import(\"../testing/dev-test-routes\");\n      await registerDevTestRoutes(server, this.config.test.devRunner);\n    }\n\n    const webPath = path.join(this.appRootPath, \"web\");\n    const hasWeb = await exists(webPath);\n\n    // 전역 compress 옵션 계산 (route.compress: true일 때 사용)\n    const pluginCompress = this.config.server.plugins?.compress;\n    const globalCompressOptions: CompressOptions | undefined = pluginCompress\n      ? pluginCompress === true\n        ? { threshold: 1024, encodings: [\"br\", \"gzip\", \"deflate\"] }\n        : {\n            threshold: pluginCompress.threshold,\n            encodings: pluginCompress.encodings,\n            customTypes: pluginCompress.customTypes,\n          }\n      : undefined;\n\n    if (isLocal()) {\n      // 로컬 개발 환경: catch-all로 API를 동적 매칭하여 HMR을 지원합니다.\n      // SONAMU_DISABLE_INTEGRATED_WEB=yes로 설정하면 dev_api 모드에서 Vite 통합을 비활성화할 수 있습니다.\n      const disableIntegratedWeb = process.env.SONAMU_DISABLE_INTEGRATED_WEB === \"yes\";\n      if (hasWeb && !disableIntegratedWeb) {\n        await this.setupDevServerWithVite(server, webPath, config);\n      } else {\n        this.setupDevServer(server, config);\n      }\n    } else {\n      // 프로덕션 환경: 개별 API 라우트 + 정적 파일 서빙\n      for (const api of this.syncer.apis) {\n        if (this.syncer.models[api.modelName] === undefined) {\n          throw new Error(`정의되지 않은 모델에 접근 ${api.modelName}`);\n        }\n\n        server.route({\n          method: api.options.httpMethod ?? \"GET\",\n          url: this.config.api.route.prefix + api.path,\n          handler: this.createApiHandler(api, config),\n          compress: toFastifyCompressOption(api.options.compress, globalCompressOptions),\n        });\n      }\n\n      // 프로덕션에서는 web 소스(appRoot/web) 유무와 무관하게,\n      // api/web-dist 존재 여부를 setupStaticWebServer 내부에서 판단합니다.\n      await this.setupStaticWebServer(server, config, globalCompressOptions);\n    }\n  }\n\n  /**\n   * dev 모드 공통: catch-all에서 syncer.apis를 동적으로 탐색하여 API 요청을 처리합니다.\n   * server.route()로 개별 등록하면 handler가 고정되어 HMR이 동작하지 않으므로,\n   * 매 요청마다 syncer.apis를 조회하는 이 방식을 사용합니다.\n   *\n   * 요청이 /api(정확히는 this.config.api.route.prefix)로 시작하지 않는 경우라면 null을 반환하며 끝냅니다.\n   */\n  private handleDevApiRequest(\n    request: FastifyRequest,\n    config: SonamuFastifyConfig,\n  ): ((request: FastifyRequest, reply: FastifyReply) => Promise<unknown>) | null {\n    const url = this.getPathnameFromUrl(request.url);\n    const method = request.method;\n\n    if (!url.startsWith(this.config.api.route.prefix)) {\n      return null;\n    }\n\n    // syncer.apis의 path는 :param 형태를 포함할 수 있으므로 세그먼트 단위로 매칭합니다.\n    // 정규식 생성 방식은 path 문자열 내 특수문자(., +, (, [ 등)로 오작동할 수 있어 사용하지 않습니다.\n    const matchedApi = this.syncer.apis.find((api) => {\n      if (this.syncer.models[api.modelName] === undefined) {\n        return false;\n      }\n      const apiMethod = api.options.httpMethod ?? \"GET\";\n      if (apiMethod !== method) return false;\n\n      const fullPath = this.config.api.route.prefix + api.path;\n      return this.isPathPatternMatch(fullPath, url);\n    });\n\n    if (!matchedApi) {\n      throw new NotFoundException(SD(\"error.api.notFound\"));\n    }\n\n    return this.createApiHandler(matchedApi, config);\n  }\n\n  /**\n   * dev api 모드: Vite 없이 API 동적 라우팅만 제공합니다.\n   * HMR을 위해 catch-all에서 매 요청마다 syncer.apis를 조회합니다.\n   */\n  private setupDevServer(\n    server: FastifyInstance<Server, IncomingMessage, ServerResponse>,\n    config: SonamuFastifyConfig,\n  ): void {\n    server.route({\n      method: [\"GET\", \"HEAD\", \"POST\", \"PUT\", \"DELETE\", \"PATCH\"],\n      url: `${this.config.api.route.prefix}/*`,\n      handler: async (request, reply) => {\n        const handler = this.handleDevApiRequest(request, config);\n        if (handler) {\n          return handler(request, reply);\n        }\n        // 등록된 API와 일치하지 않는 요청에 대한 fallback입니다.\n        throw new NotFoundException(SD(\"error.api.notFound\"));\n      },\n    });\n  }\n\n  // oxlint-disable-next-line @typescript-eslint/no-explicit-any -- ViteDevServer 타입을 동적으로 로드해야 함\n  private viteServer: any = null;\n\n  /**\n   * dev all 모드: Vite Dev Server를 통합하여 API + SSR + CSR을 모두 제공합니다.\n   * API 동적 매칭은 handleDevApiRequest를 공유합니다.\n   */\n  private async setupDevServerWithVite(\n    server: FastifyInstance<Server, IncomingMessage, ServerResponse>,\n    webPath: string,\n    config: SonamuFastifyConfig,\n  ): Promise<void> {\n    // @fastify/middie 등록 (Connect-style middleware 지원)\n    await server.register((await import(\"@fastify/middie\")).default);\n\n    const vite = await import(\"vite\");\n\n    this.viteServer = await vite.createServer({\n      root: webPath,\n      server: {\n        middlewareMode: true,\n        hmr: {\n          server: server.server,\n        },\n      },\n      appType: \"custom\",\n    });\n\n    // Vite middleware 등록 (Vite 에셋 처리)\n    server.use((req, res, next) => {\n      // API와 Sonamu UI는 Fastify 라우트가 처리하도록 skip\n      if (req.url?.startsWith(this.config.api.route.prefix) || req.url?.startsWith(\"/sonamu-ui\")) {\n        return next();\n      }\n      // 나머지는 Vite middleware로 전달\n      return this.viteServer.middlewares(req, res, next);\n    });\n\n    // catch-all 라우트에서 동적으로 API/SSR 처리\n    // 개발 환경에서는 라우트별 compress 옵션을 포기하고 HMR 이점을 취합니다.\n    server.route({\n      method: [\"GET\", \"HEAD\", \"POST\", \"PUT\", \"DELETE\", \"PATCH\"],\n      url: \"/*\",\n      handler: async (request, reply) => {\n        // 1. API 요청 처리\n        const result = this.handleDevApiRequest(request, config);\n        if (result) {\n          return result(request, reply);\n        }\n\n        const url = request.url;\n\n        // 2. SSR 라우트 처리\n        const { matchSSRRoute, renderSSR } = await import(\"../ssr\");\n        const ssrMatch = matchSSRRoute(url);\n        if (ssrMatch) {\n          console.log(`[SSR] Matched route: ${ssrMatch.route.path}`);\n          const html = await renderSSR(\n            url,\n            ssrMatch.route,\n            ssrMatch.params,\n            request,\n            reply,\n            config,\n            this.viteServer,\n          );\n          reply.type(\"text/html\");\n          return html;\n        }\n\n        // 3. CSR fallback\n        try {\n          const fs = await import(\"node:fs/promises\");\n          let template = await fs.readFile(\n            path.join(this.viteServer.config.root, \"index.html\"),\n            \"utf-8\",\n          );\n          template = await this.viteServer.transformIndexHtml(url, template);\n\n          reply.type(\"text/html\");\n          return template;\n        } catch (e) {\n          this.viteServer.ssrFixStacktrace(e as Error);\n          console.error(e);\n          reply.status(500);\n          return (e as Error).message;\n        }\n      },\n    });\n\n    // 서버 종료 시 Vite도 종료\n    server.addHook(\"onClose\", async () => {\n      await this.viteServer.close();\n    });\n\n    const chalk = (await import(\"chalk\")).default;\n    console.log(chalk.dim(\"✓ Vite dev server integrated\"));\n  }\n\n  private async setupStaticWebServer(\n    server: FastifyInstance<Server, IncomingMessage, ServerResponse>,\n    config: SonamuFastifyConfig,\n    globalCompressOptions: CompressOptions | undefined,\n  ): Promise<void> {\n    // 경로 명확화: api/web-dist/client (정적 파일), api/web-dist/server (SSR entry), api/dist/ssr (SSR routes - API 소유)\n    const webDistPath = path.join(this.apiRootPath, \"web-dist\", \"client\");\n    const ssrPath = path.join(this.apiRootPath, \"web-dist\", \"server\");\n    const ssrEntryPath = path.join(ssrPath, \"entry-server.generated.js\");\n    const ssrRoutesPath = path.join(this.apiRootPath, \"dist\", \"ssr\", \"routes.js\");\n\n    if (!(await exists(webDistPath))) {\n      console.warn(`⚠ Web dist not found: ${webDistPath}`);\n      return;\n    }\n\n    // SSR entry 존재 여부 확인\n    const ssrAvailable = await exists(ssrEntryPath);\n\n    if (!ssrAvailable) {\n      console.warn(`⚠ SSR entry not found: ${ssrEntryPath}`);\n      console.warn(\"  SSR will be disabled. Only CSR will work.\");\n    }\n\n    // SSR 라우트 로드 (production에서만, 사용자 프로젝트의 ssr/routes.ts)\n    if (ssrAvailable) {\n      if (await exists(ssrRoutesPath)) {\n        // ts-loader라면 \"file://\"로 시작하는 fully-resolved path만 받기에 이를 처리해주는 importMembers를 사용해야 했겠지만,\n        // 여기는 프로덕션 환경에서 loader 없이 돌아가기 때문에 \"진짜 js 파일\"의 \"그냥\" 절대경로를 바로 import해도 됩니다.\n        // 이 내용은 이 함수 내에서 아래에 나올 다른 import 호출에도 동일하게 적용됩니다.\n        await import(ssrRoutesPath);\n        console.log(\"✓ SSR routes loaded\");\n      } else {\n        console.warn(`⚠ SSR routes not found: ${ssrRoutesPath}`);\n      }\n    }\n\n    // 롤링 업데이트 대응: asset hash 불일치 시 현재 버전 직접 서빙\n    server.get(\"/assets/:filename\", async (request, reply) => {\n      const requestedFile = (request.params as { filename: string }).filename;\n      const assetsDir = path.join(webDistPath, \"assets\");\n      const safeFilePath = this.resolvePathWithinBaseDir(assetsDir, requestedFile);\n      if (safeFilePath === null) {\n        reply.status(403).send();\n        return;\n      }\n      const normalizedRequestedFile = path.relative(assetsDir, safeFilePath).replace(/\\\\/g, \"/\");\n\n      const assetPath = `/assets/${normalizedRequestedFile}`;\n\n      // Cache-Control 헤더 결정\n      const getCacheControlForAsset = (): CacheControlConfig => {\n        const cacheReq: CacheControlRequest = {\n          type: \"assets\",\n          url: request.url,\n          path: assetPath,\n          method: request.method,\n        };\n\n        // 사용자 정의 핸들러 우선\n        if (config.cacheControlHandler) {\n          const result = config.cacheControlHandler(cacheReq);\n          if (result) return result;\n        }\n\n        // 기본값: immutable\n        return CachePresets.immutable;\n      };\n\n      // index-*.js 또는 index-*.css 요청인 경우\n      if (/^index-[a-f0-9]+\\.(js|css)$/.test(normalizedRequestedFile)) {\n        const ext = normalizedRequestedFile.split(\".\").pop();\n        const files = await fs.readdir(assetsDir);\n        const currentFile = files.find((f) => f.startsWith(\"index-\") && f.endsWith(`.${ext}`));\n\n        if (currentFile) {\n          const filePath = path.join(assetsDir, currentFile);\n          const content = await fs.readFile(filePath);\n          reply.type(ext === \"js\" ? \"application/javascript\" : \"text/css\");\n          applyCacheHeaders(reply, getCacheControlForAsset());\n          return reply.send(content);\n        }\n      }\n\n      // 일반 파일 서빙\n      const filePath = safeFilePath;\n      if (await exists(filePath)) {\n        const content = await fs.readFile(filePath);\n        const ext = normalizedRequestedFile.split(\".\").pop();\n        reply.type(ext === \"js\" ? \"application/javascript\" : ext === \"css\" ? \"text/css\" : \"\");\n        if (normalizedRequestedFile.includes(\"-\")) {\n          applyCacheHeaders(reply, getCacheControlForAsset());\n        }\n        return reply.send(content);\n      }\n\n      reply.status(404).send();\n    });\n\n    // SSR 라우트 개별 등록 (compress 옵션이 라우트별로 적용되도록)\n    if (ssrAvailable) {\n      const { getSSRRoutes } = await import(\"../ssr\");\n      const { renderSSR } = await import(\"../ssr/renderer\");\n      const ssrRoutes = getSSRRoutes();\n\n      for (const route of ssrRoutes) {\n        server.route({\n          method: [\"GET\", \"HEAD\"],\n          url: route.path,\n          compress: toFastifyCompressOption(route.compress ?? true, globalCompressOptions),\n          handler: async (request, reply) => {\n            const url = request.url;\n            console.log(`[SSR] Matched route: ${route.path}`);\n\n            const params = this.extractPathParams(route.path, url);\n            const html = await renderSSR(url, route, params, request, reply, config);\n\n            reply.type(\"text/html\");\n            return html;\n          },\n        });\n      }\n    }\n\n    // CSR or Static File Fallback (SSR 라우트에 매칭되지 않는 모든 요청)\n    server.route({\n      method: [\"GET\", \"HEAD\"],\n      url: \"*\",\n      handler: async (request, reply) => {\n        // /api, /sonamu-ui는 404 그대로\n        if (request.url.startsWith(\"/api\") || request.url.startsWith(\"/sonamu-ui\")) {\n          reply.status(404).send();\n          return;\n        }\n\n        // CSR용 Cache-Control 헤더 설정\n        if (config.cacheControlHandler) {\n          const csrCacheReq: CacheControlRequest = {\n            type: \"csr\",\n            url: request.url,\n            path: request.url.split(\"?\")[0],\n            method: request.method,\n          };\n          const csrCacheConfig = config.cacheControlHandler(csrCacheReq);\n\n          if (csrCacheConfig) {\n            applyCacheHeaders(reply, csrCacheConfig);\n          }\n        }\n\n        // 정적 파일이 존재할 경우, 정적 파일을 먼저 서빙해야함\n        const requestPath = this.getPathnameFromUrl(request.url);\n        const safeFilePath = this.resolvePathWithinBaseDir(webDistPath, requestPath);\n        if (safeFilePath === null) {\n          reply.status(403).send();\n          return;\n        }\n        if (await fileExists(safeFilePath)) {\n          const content = await fs.readFile(safeFilePath);\n          return reply.type(mimeLookup(safeFilePath) || \"application/octet-stream\").send(content);\n        }\n\n        // CSR fallback: index.html 서빙\n        const indexPath = path.join(webDistPath, \"index.html\");\n        return reply.type(\"text/html\").send(await fs.readFile(indexPath, \"utf-8\"));\n      },\n    });\n\n    console.log(`✓ Static web server configured with ${ssrAvailable ? \"SSR\" : \"CSR only\"} support`);\n  }\n\n  createApiHandler(\n    api: ExtendedApi,\n    config: SonamuFastifyConfig,\n  ): (request: FastifyRequest, reply: FastifyReply) => Promise<unknown> {\n    return async (request: FastifyRequest, reply: FastifyReply): Promise<unknown> => {\n      // Context 생성\n      const context: Context = await this.createContext(config, request, reply);\n\n      return this.asyncLocalStorage.run({ context }, async () => {\n        // guards 처리\n        (api.options.guards ?? []).every((guard) => config.guardHandler(guard, request, api));\n\n        // 파라미터 정보로 zod 스키마 빌드\n        const { getZodObjectFromApi } = await import(\"./code-converters\");\n        const ReqType = getZodObjectFromApi(api, this.syncer.types);\n\n        // request 파싱\n        const which = api.options.httpMethod === \"GET\" ? \"query\" : \"body\";\n        let reqBody: {\n          [key: string]: unknown;\n        };\n        // 파일 업로드 있는 경우 임시 데이터\n        const files: {\n          bufferedFiles: BufferedFile[];\n          uploadedFiles: UploadedFile[];\n        } = {\n          bufferedFiles: [],\n          uploadedFiles: [],\n        };\n\n        try {\n          const body = (request[which] ?? {}) as Record<string, unknown>;\n          if (api.uploadOptions) {\n            const parts = request.parts({\n              limits: api.uploadOptions.limits,\n            });\n\n            // FormData의 field들을 임시로 저장\n            const fields: Record<string, string> = {};\n\n            if (api.uploadOptions.consume === \"buffer\" || !api.uploadOptions.consume) {\n              // Buffer 모드: 메모리에 로드\n              for await (const part of parts) {\n                if (part.type === \"file\") {\n                  // CRITICAL: 파일 스트림을 즉시 consume해야 다음 part로 넘어갈 수 있음\n                  // 이 호출이 없으면 종종 multipart 파싱이 pending 상태로 타임아웃 발생\n                  const buffer = await part.toBuffer();\n                  files.bufferedFiles.push(new BufferedFile(part, buffer));\n                } else if (part.type === \"field\") {\n                  fields[part.fieldname] = String(part.value);\n                }\n              }\n            } else if (api.uploadOptions.consume === \"stream\") {\n              // Stream 모드: 즉시 저장소로 스트리밍\n              const diskName = api.uploadOptions.destination;\n              const disk = this.storage.use(diskName);\n\n              // 우선순위: 데코레이터 > 전역 설정 > 기본값\n              const keyGenerator: KeyGenerator =\n                api.uploadOptions.keyGenerator ??\n                this.config.server.storage?.keyGenerator ??\n                defaultKeyGenerator;\n\n              for await (const part of parts) {\n                if (part.type === \"file\") {\n                  const key = await keyGenerator({\n                    filename: part.filename,\n                    mimetype: part.mimetype,\n                  });\n\n                  await disk.putStream(key, part.file, {\n                    contentType: part.mimetype,\n                  });\n\n                  const url = await disk.getUrl(key);\n                  const signedUrl = await disk.getSignedUrl(key);\n\n                  files.uploadedFiles.push(\n                    new UploadedFile({\n                      filename: part.filename,\n                      mimetype: part.mimetype,\n                      size: part.file.bytesRead,\n                      url,\n                      signedUrl,\n                      key,\n                      diskName,\n                    }),\n                  );\n                } else if (part.type === \"field\") {\n                  fields[part.fieldname] = String(part.value);\n                }\n              }\n            }\n\n            // qs로 중첩 구조 파싱: params[category] → { params: { category: \"test\" } }\n            const qs = await import(\"qs\");\n            const parsed = qs.default.parse(fields);\n            Object.assign(body, parsed);\n          }\n\n          const { fastifyCaster } = await import(\"./caster\");\n          reqBody = fastifyCaster(ReqType).parse(body);\n        } catch (e) {\n          const { ZodError } = await import(\"zod\");\n          if (e instanceof ZodError) {\n            const { humanizeZodError } = await import(\"../utils/zod-error\");\n            const messages = humanizeZodError(e)\n              .map((issue) => issue.message)\n              .join(\" \");\n            const { BadRequestException } = await import(\"../exceptions/so-exceptions\");\n            throw new BadRequestException(messages as LocalizedString, {\n              zodError: e,\n            });\n          } else {\n            throw e;\n          }\n        }\n\n        // Content-Type\n        reply.type(api.options.contentType ?? \"application/json\");\n\n        // Cache-Control 헤더 설정\n        const apiCacheConfig = this.getApiCacheControl(api, request, config);\n        if (apiCacheConfig) {\n          applyCacheHeaders(reply, apiCacheConfig);\n        }\n\n        // 업로드 옵션이 있는 경우 파일 데이터를 Context에 추가\n        if (api.uploadOptions) {\n          const consume = api.uploadOptions.consume ?? \"buffer\";\n          if (consume === \"buffer\") {\n            context.bufferedFiles = files.bufferedFiles;\n          } else if (consume === \"stream\") {\n            context.uploadedFiles = files.uploadedFiles;\n          }\n        }\n\n        // 모델 메소드 args 생성하여 호출\n        const { ApiParamType } = await import(\"../types/types\");\n        const args = api.parameters.map((param) => {\n          // Context 인젝션\n          if (ApiParamType.isContext(param.type)) {\n            return context;\n          } else {\n            return reqBody[param.name];\n          }\n        });\n\n        return this.invokeModelMethod(api, args, reply);\n      });\n    };\n  }\n\n  /**\n   * URL에서 path params를 추출합니다.\n   * 예: pattern=\"/admin/companies/:companyId\", url=\"/admin/companies/123\" → { companyId: \"123\" }\n   */\n  private extractPathParams(pattern: string, url: string): Record<string, string> {\n    const patternParts = pattern.split(\"/\").filter(Boolean);\n    const urlParts = this.getPathnameFromUrl(url).split(\"/\").filter(Boolean);\n    const params: Record<string, string> = {};\n\n    for (let i = 0; i < patternParts.length; i++) {\n      if (patternParts[i].startsWith(\":\")) {\n        params[patternParts[i].slice(1)] = urlParts[i];\n      }\n    }\n    return params;\n  }\n\n  private isPathPatternMatch(pattern: string, url: string): boolean {\n    const patternParts = pattern.split(\"/\").filter(Boolean);\n    const urlParts = this.getPathnameFromUrl(url).split(\"/\").filter(Boolean);\n\n    if (patternParts.length !== urlParts.length) {\n      return false;\n    }\n\n    for (let i = 0; i < patternParts.length; i++) {\n      const patternPart = patternParts[i];\n      const urlPart = urlParts[i];\n      if (patternPart.startsWith(\":\")) {\n        continue;\n      }\n      if (patternPart !== urlPart) {\n        return false;\n      }\n    }\n\n    return true;\n  }\n\n  private getPathnameFromUrl(url: string): string {\n    return url.split(\"?\")[0];\n  }\n\n  private resolvePathWithinBaseDir(baseDir: string, inputPath: string): string | null {\n    try {\n      const decoded = decodeURIComponent(inputPath).replace(/\\\\/g, \"/\");\n      if (decoded.includes(\"\\0\")) {\n        return null;\n      }\n      const relativePath = decoded.replace(/^\\/+/, \"\");\n      const resolvedPath = path.resolve(baseDir, relativePath);\n      const relativeFromBase = path.relative(baseDir, resolvedPath);\n      if (relativeFromBase.startsWith(\"..\") || path.isAbsolute(relativeFromBase)) {\n        return null;\n      }\n      return resolvedPath;\n    } catch {\n      return null;\n    }\n  }\n\n  /**\n   * API 응답에 적용할 Cache-Control 설정을 결정합니다.\n   * 우선순위: 개별 지정 > cacheControlHandler\n   */\n  private getApiCacheControl(\n    api: ExtendedApi,\n    request: FastifyRequest,\n    config: SonamuFastifyConfig,\n  ) {\n    // 데코레이터 설정 우선\n    if (api.options.cacheControl) {\n      return api.options.cacheControl;\n    }\n\n    // 전역 핸들러\n    if (config.cacheControlHandler) {\n      const cacheReq: CacheControlRequest = {\n        type: \"api\",\n        url: request.url,\n        path: request.routeOptions?.url ?? request.url.split(\"?\")[0],\n        method: request.method,\n        api,\n      };\n      const result = config.cacheControlHandler(cacheReq);\n      if (result) return result;\n    }\n\n    return null;\n  }\n\n  /**\n   * SSR용 API 호출 (HTTP 오버헤드 없이 직접 호출)\n   * createApiHandler의 로직을 재사용하되, request 파싱 대신 params 직접 사용\n   */\n  async invokeApiForSSR(\n    api: ExtendedApi,\n    // oxlint-disable-next-line @typescript-eslint/no-explicit-any -- SSR에서 다양한 타입의 params를 받아야 함\n    params: any[],\n    config: SonamuFastifyConfig,\n    request: FastifyRequest,\n    reply: FastifyReply,\n  ): Promise<unknown> {\n    // Context 생성 (기존 메소드 재사용)\n    const context = await this.createContext(config, request, reply);\n\n    return this.asyncLocalStorage.run({ context }, async () => {\n      // args 생성: Context 파라미터는 주입, 나머지는 params에서 가져오기\n      const { ApiParamType } = await import(\"../types/types\");\n      let paramsIndex = 0;\n      const args = api.parameters.map((param) => {\n        if (ApiParamType.isContext(param.type)) {\n          return context;\n        }\n        return params[paramsIndex++];\n      });\n\n      // 모델 메서드 호출 (기존 메서드 재사용)\n      return this.invokeModelMethod(api, args, reply);\n    });\n  }\n\n  async invokeModelMethod(\n    api: ExtendedApi,\n    args: unknown[],\n    reply: FastifyReply,\n  ): Promise<unknown> {\n    const model = this.syncer.models[api.modelName];\n    // oxlint-disable-next-line @typescript-eslint/no-explicit-any -- model은 모델 인스턴스이므로 메서드 호출 가능\n    const result = await (model as any)[api.methodName].apply(model, args);\n    reply.type(api.options.contentType ?? \"application/json\");\n\n    return result;\n  }\n\n  async createContext(\n    config: SonamuFastifyConfig,\n    request: FastifyRequest,\n    reply: FastifyReply,\n  ): Promise<Context> {\n    // createSSEFactory 함수에 미리 request의 socket과 reply를 바인딩.\n    const { createSSEFactory } = await import(\"../stream/sse\");\n    const createSSE = (<T extends ZodObject>(\n      _request: FastifyRequest,\n      _reply: FastifyReply,\n      _events: T,\n    ) => createSSEFactory(_request.socket, _reply, _events)).bind(null, request, reply);\n\n    // locale 감지\n    const locale =\n      this.detectLocale(request.headers[\"accept-language\"], this.config.i18n.supportedLocales) ??\n      this.config.i18n.defaultLocale;\n\n    // auth context 추가\n    const headers = convertFastifyHeadersToStandard(request.headers);\n    const session = (await this._auth?.api.getSession({ headers })) ?? null;\n\n    const context: Context = {\n      ...(await Promise.resolve(\n        config.contextProvider(\n          {\n            request,\n            reply,\n            headers: request.headers,\n            createSSE,\n            naiteStore: new Map(),\n            locale,\n            // auth\n            user: session?.user ?? null,\n            session: session?.session ?? null,\n          },\n          request,\n          reply,\n        ),\n      )),\n    };\n    return context;\n  }\n\n  /**\n   * Accept-Language 헤더에서 지원하는 locale을 찾습니다.\n   * @example \"ko-KR,ko;q=0.9,en;q=0.8\" → \"ko\"\n   */\n  private detectLocale(\n    acceptLanguage: string | undefined,\n    supported: string[],\n  ): string | undefined {\n    if (!acceptLanguage) return undefined;\n\n    // Accept-Language: ko-KR,ko;q=0.9,en;q=0.8\n    const langs = acceptLanguage.split(\",\").map((lang) => {\n      const [code] = lang.split(\";\");\n      return code.trim().split(\"-\")[0]; // ko-KR → ko\n    });\n\n    return langs.find((lang) => supported.includes(lang));\n  }\n\n  async startWatcher(): Promise<void> {\n    const watchPath = [path.join(this.apiRootPath, \"src\")];\n\n    const chokidar = (await import(\"chokidar\")).default;\n    this.watcher = chokidar.watch(watchPath, {\n      ignored: (path, stats) =>\n        !!stats?.isFile() && !path.endsWith(\".ts\") && !path.endsWith(\".json\"),\n      persistent: true,\n      ignoreInitial: true,\n    });\n\n    this.watcher.on(\"all\", async (event: string, filePath: string) => {\n      const absolutePath = filePath as AbsolutePath;\n      assert(\n        absolutePath.startsWith(this.apiRootPath),\n        \"File path is not within the API root path\",\n      );\n\n      if (event !== \"change\" && event !== \"add\") {\n        return;\n      }\n\n      try {\n        // sonamu.config.ts 변경 시 재시작\n        const isConfigTs = filePath === path.join(this.apiRootPath, \"src\", \"sonamu.config.ts\");\n\n        if (isConfigTs) {\n          const relativePath = filePath.replace(this.apiRootPath, \"api\");\n          const chalk = (await import(\"chalk\")).default;\n          console.log(\n            chalk.bold(`Detected(${event}): ${chalk.blue(relativePath)} - Restarting...`),\n          );\n          process.kill(process.pid, \"SIGUSR2\");\n          return;\n        }\n\n        await this.handleFileChange(event, absolutePath);\n      } catch (e) {\n        console.error(e);\n      }\n    });\n  }\n\n  /*\n     A function that automatically handles init and destroy when using Sonamu via scripts.\n  */\n  async runScript(fn: () => Promise<void>) {\n    await this.init(true, false, undefined, false);\n    try {\n      await fn();\n    } finally {\n      await this.destroy();\n    }\n  }\n\n  private async registerPlugins(server: FastifyInstance, plugins: SonamuServerOptions[\"plugins\"]) {\n    if (!plugins) {\n      return;\n    }\n\n    // compress 플러그인은 다른 플러그인보다 먼저 등록되어야 합니다.\n    if (plugins.compress) {\n      const compressPlugin = (await import(\"@fastify/compress\")).default;\n      const defaultOptions = {\n        threshold: 1024,\n        encodings: [\"br\", \"gzip\", \"deflate\"] as (\"br\" | \"gzip\" | \"deflate\")[],\n      };\n\n      if (plugins.compress === true) {\n        server.register(compressPlugin, defaultOptions);\n      } else {\n        server.register(compressPlugin, {\n          ...defaultOptions,\n          ...plugins.compress,\n        });\n      }\n    }\n\n    const pluginsModules = {\n      cors: \"@fastify/cors\",\n      formbody: \"@fastify/formbody\",\n      multipart: \"@fastify/multipart\",\n      qs: \"fastify-qs\",\n      sse: \"fastify-sse-v2\",\n      static: \"@fastify/static\",\n    } as const;\n\n    const registerPlugin = async <K extends keyof NonNullable<typeof plugins>>(\n      key: K,\n      pluginName: string,\n    ) => {\n      const option = plugins[key];\n      if (!option) return;\n\n      if (option === true) {\n        server.register((await import(pluginName)).default);\n      } else {\n        server.register((await import(pluginName)).default, option);\n      }\n    };\n\n    for (const [key, pluginName] of Object.entries(pluginsModules)) {\n      await registerPlugin(key as keyof typeof plugins, pluginName);\n    }\n\n    if (plugins.custom) {\n      plugins.custom(server);\n    }\n  }\n\n  /**\n   * better-auth 라우트를 등록합니다.\n   * /api/auth/* 경로로 인증 API가 자동 등록됩니다.\n   */\n  private async registerBetterAuth(\n    server: FastifyInstance,\n    options: NonNullable<SonamuServerOptions[\"auth\"]>,\n  ) {\n    if (!options) return;\n\n    const basePath = options.basePath ?? \"/api/auth\";\n\n    // better-auth 라우트 등록\n    server.route({\n      method: [\"GET\", \"POST\"],\n      url: `${basePath}/*`,\n      handler: async (request, reply) => {\n        const url = new URL(request.url, `http://${request.headers.host}`);\n        const headers = convertFastifyHeadersToStandard(request.headers);\n\n        // IP 헤더 fallback: 프록시가 표준 IP 헤더를 주입하지 않는 환경에서도\n        // better-auth/infra의 getClientIpFromRequest()가 IP를 인식할 수 있도록\n        // Fastify가 resolve한 request.ip를 x-real-ip로 주입한다.\n        const IP_HEADERS = [\n          \"cf-connecting-ip\",\n          \"x-forwarded-for\",\n          \"x-real-ip\",\n          \"x-vercel-forwarded-for\",\n        ];\n        if (request.ip && !IP_HEADERS.some((h) => headers.has(h))) {\n          headers.set(\"x-real-ip\", request.ip);\n        }\n\n        const req = new Request(url.toString(), {\n          method: request.method,\n          headers,\n          ...(request.body ? { body: JSON.stringify(request.body) } : {}),\n        });\n\n        const response = await this.auth.handler(req);\n\n        reply.status(response.status);\n        response.headers.forEach((value: string, key: string) => {\n          reply.header(key, value);\n        });\n        return reply.send(response.body ? await response.text() : null);\n      },\n    });\n  }\n\n  private async printStartupSummary() {\n    const chalk = (await import(\"chalk\")).default;\n    const env = process.env.NODE_ENV ?? \"development\";\n    const activePreset = env === \"production\" ? \"production_master\" : \"development_master\";\n\n    const dim = (msg: string) => console.log(chalk.dim(`✓ ${msg}`));\n    const green = (msg: string) => console.log(chalk.green(`✓ ${msg}`));\n\n    dim(`Config loaded${formatTime(this._configElapsed)}`);\n\n    // DB preset 목록\n    green(\"DB\");\n    const { isLocal } = await import(\"../utils/controller\");\n    const presetNames = Object.keys(this.dbConfig) as (keyof SonamuDBConfig)[];\n    const maxLen = Math.max(...presetNames.map((n) => n.length));\n    for (const name of presetNames) {\n      const conn = this.dbConfig[name].connection as\n        | { host?: string; port?: number; database?: string }\n        | undefined;\n      const host = conn?.host ?? \"localhost\";\n      const addr = `@ ${host}:${conn?.port ?? 5432}/${conn?.database ?? this.config.database.name}`;\n      const padded = name.padEnd(maxLen);\n      const remoteTag = isLocal() && !isLocalHost(host) ? chalk.yellow(` \\u26a0 remote`) : \"\";\n\n      if (name === activePreset) {\n        console.log(chalk.green(`  \\u25b8 ${padded} ${addr}`) + remoteTag);\n      } else {\n        console.log(chalk.dim(`    ${padded} ${addr}`) + remoteTag);\n      }\n    }\n\n    if (this.config.server.auth) {\n      const basePath = this.config.server.auth.basePath ?? \"/api/auth\";\n      dim(`Auth: better-auth at ${basePath}/*`);\n    }\n    if (this.config.api.timezone) {\n      dim(`Timezone: ${this.config.api.timezone}`);\n    }\n    green(`Sonamu ready${formatTime(this._initElapsed)}`);\n  }\n\n  private async initializeCache(config: CacheConfig | undefined, forTesting: boolean) {\n    const { setCacheManagerRef } = await import(\"../cache/decorator\");\n\n    // 테스트 환경에서 메모리 드라이버 자동 사용\n    if (forTesting) {\n      const { createTestCacheManager } = await import(\"../cache/cache-manager\");\n      this._cache = createTestCacheManager();\n      setCacheManagerRef(this._cache);\n      return;\n    }\n\n    // 설정이 없으면 캐시 비활성화\n    if (!config) {\n      setCacheManagerRef(null);\n      return;\n    }\n\n    // 설정에 따라 CacheManager 생성\n    const { createCacheManager } = await import(\"../cache/cache-manager\");\n    this._cache = createCacheManager(config);\n    setCacheManagerRef(this._cache);\n  }\n\n  private async initializeWorkflows(options: SonamuTaskOptions | undefined) {\n    const { WorkflowManager } = await import(\"../tasks/workflow-manager\");\n    // NOTE: @sonamu-kit/tasks 안에선 knex config를 수정하기 때문에 connection이 아닌 config 째로 보냅니다.\n    this._workflows = new WorkflowManager(DB.getDBConfig(\"w\"));\n    if (!options) {\n      return;\n    }\n\n    const enableWorker = options.enableWorker ?? isDaemonServer();\n    const defaultWorkerOptions = {\n      concurrency: os.cpus().length - 1,\n      usePubSub: true,\n      listenDelay: 500,\n    };\n\n    if (enableWorker) {\n      this.workflows.setupWorker({\n        ...defaultWorkerOptions,\n        ...options.workerOptions,\n      });\n    }\n  }\n\n  private async boot(server: FastifyInstance, options: SonamuServerOptions) {\n    const port = options.listen?.port ?? 3000;\n    const host = options.listen?.host ?? \"localhost\";\n\n    server.addHook(\"onClose\", async () => {\n      await options.lifecycle?.onShutdown?.(server);\n      await this.workflows.destroy();\n      await this.destroy();\n    });\n\n    const shutdown = async () => {\n      try {\n        await server.close();\n        process.exit(0);\n      } catch (err) {\n        console.error(\"Error during shutdown:\", err);\n        process.exit(1);\n      }\n    };\n\n    process.on(\"SIGINT\", shutdown);\n    process.on(\"SIGTERM\", shutdown);\n\n    if (options.lifecycle?.onError) {\n      server.setErrorHandler(options.lifecycle?.onError);\n    }\n\n    server\n      .listen({ port, host })\n      .then(async () => {\n        await this.workflows.startWorker();\n        await options.lifecycle?.onStart?.(server);\n      })\n      .catch(async (err) => {\n        const chalk = (await import(\"chalk\")).default;\n        console.error(chalk.red(\"Failed to start server:\", err));\n        await shutdown();\n      });\n  }\n\n  private async handleFileChange(event: string, filePath: AbsolutePath): Promise<void> {\n    // 첫 번째 파일이면 HMR 시작 시간 기록\n    if (this.pendingFiles.length === 0) {\n      this.hmrStartTime = Date.now();\n    }\n    this.pendingFiles.push(filePath);\n\n    const relativePath = path.relative(this.apiRootPath, filePath);\n    const chalk = (await import(\"chalk\")).default;\n    console.log(chalk.bold(`Detected(${event}): ${chalk.blue(relativePath)}`));\n\n    await this.syncer.syncFromWatcher(event, filePath);\n\n    // 처리 완료된 파일을 대기 목록에서 제거\n    this.pendingFiles = this.pendingFiles.slice(1);\n\n    // 모든 파일 처리가 완료되면 최종 메시지 출력\n    if (this.pendingFiles.length === 0) {\n      await this.finishHMR();\n    }\n  }\n\n  private async finishHMR(): Promise<void> {\n    await this.syncer.renewChecksums();\n\n    const endTime = Date.now();\n    const totalTime = endTime - this.hmrStartTime;\n    const [chalk, { centerText }] = await Promise.all([\n      (await import(\"chalk\")).default,\n      import(\"../utils/console-util\"),\n    ]);\n    const msg = `HMR Done! ${chalk.bold.white(`${totalTime}ms`)}`;\n\n    console.log(chalk.black.bgGreen(centerText(msg)));\n  }\n\n  async destroy(): Promise<void> {\n    const { BaseModel } = await import(\"../database/base-model\");\n    // 먼저 처리해야함.\n    await BaseModel.destroy();\n    await Promise.allSettled([\n      this._workflows?.destroy() ?? Promise.resolve(),\n      this._cache?.disconnect() ?? Promise.resolve(),\n      this._devVitestManager?.shutdown() ?? Promise.resolve(),\n      this.watcher?.close() ?? Promise.resolve(),\n      logtapeDispose(),\n    ]);\n  }\n}\n\nexport const Sonamu = new SonamuClass();\n\n/**\n * stream 모드에서 키 생성 함수가 지정되지 않았을 때 사용하는 기본 함수입니다.\n */\nfunction defaultKeyGenerator(file: { filename: string; mimetype: string }): string {\n  const ext = mime.extension(file.mimetype) || \"bin\";\n  const timestamp = Date.now();\n  const random = Math.random().toString(36).slice(2, 8);\n  return `uploads/${timestamp}-${random}.${ext}`;\n}\n\nfunction formatTime(ms: number): string {\n  const formatted = ms >= 1000 ? `${(ms / 1000).toFixed(2)}s` : `${Math.round(ms)}ms`;\n  return ` (${formatted})`;\n}\n\nconst LOCAL_HOSTS = new Set([\"localhost\", \"127.0.0.1\", \"0.0.0.0\", \"::1\"]);\nfunction isLocalHost(host: string): boolean {\n  return LOCAL_HOSTS.has(host);\n}\n"],"mappings":"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"}
|
|
1350
|
+
export { Sonamu, createWebSocketReplyStub, init_sonamu, resolveWebSocketCloseDescriptor, resolveWebSocketPluginOptions };
|
|
1351
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"sonamu.js","names":["DB","authOptions: BetterAuthOptions","globalCompressOptions: CompressOptions | undefined","fs","chalk","cacheReq: CacheControlRequest","filePath","csrCacheReq: CacheControlRequest","mimeLookup","context: Context","reqBody: {\n          [key: string]: unknown;\n        }","files: {\n          bufferedFiles: BufferedFile[];\n          uploadedFiles: UploadedFile[];\n        }","fields: Record<string, string>","keyGenerator: KeyGenerator","wsContext: WebSocketContext | null","rawWs: ReturnType<WebSocketRuntime[\"registerConnection\"]> | null","params: Record<string, string>","logtapeDispose"],"sources":["../../src/api/sonamu.ts"],"sourcesContent":["import { AsyncLocalStorage } from \"async_hooks\";\nimport fs from \"fs/promises\";\nimport { type IncomingMessage, type Server, type ServerResponse } from \"http\";\nimport os from \"os\";\nimport path from \"path\";\n\nimport type {} from \"@fastify/websocket\";\nimport { dispose as logtapeDispose } from \"@logtape/logtape\";\nimport { type Auth, type BetterAuthOptions } from \"better-auth\";\nimport chalk from \"chalk\";\nimport { type FSWatcher } from \"chokidar\";\nimport { type FastifyInstance, type FastifyReply, type FastifyRequest } from \"fastify\";\nimport mime, { lookup as mimeLookup } from \"mime-types\";\nimport { type WebSocket } from \"ws\";\nimport { type ZodObject } from \"zod\";\n\nimport { BASE_FIELD_MAPPINGS } from \"../auth/better-auth-entities\";\nimport { applyCacheHeaders, CachePresets } from \"../cache-control/cache-control\";\nimport { type CacheControlConfig, type CacheControlRequest } from \"../cache-control/types\";\nimport { type CacheConfig, type CacheManager } from \"../cache/types\";\nimport { toFastifyCompressOption } from \"../compress/compress\";\nimport { type CompressOptions } from \"../compress/types\";\nimport { DB } from \"../database/db\";\nimport { type SonamuDBConfig } from \"../database/db\";\nimport { SD, setSDConfig } from \"../dict/sd\";\nimport { type LocalizedString } from \"../dict/types\";\nimport { NotFoundException } from \"../exceptions/so-exceptions\";\nimport { BufferedFile } from \"../storage/buffered-file\";\nimport { type StorageManager } from \"../storage/storage-manager\";\nimport { type KeyGenerator } from \"../storage/types\";\nimport { UploadedFile } from \"../storage/uploaded-file\";\nimport { createMockSSEFactory } from \"../stream/sse\";\nimport { WebSocketRuntime, type WebSocketConnection, type WebSocketEventMap } from \"../stream/ws\";\nimport { type Syncer } from \"../syncer/syncer\";\nimport { type WorkflowManager } from \"../tasks/workflow-manager\";\nimport { type DevVitestManager } from \"../testing/dev-vitest-manager\";\nimport { type SonamuFastifyConfig } from \"../types/types\";\nimport { centerText } from \"../utils/console-util\";\nimport { isDaemonServer } from \"../utils/controller\";\nimport { exists, fileExists } from \"../utils/fs-utils\";\nimport { type AbsolutePath } from \"../utils/path-utils\";\nimport { convertFastifyHeadersToStandard, merge } from \"../utils/utils\";\nimport { type SonamuConfig, type SonamuServerOptions, type SonamuTaskOptions } from \"./config\";\nimport { type Context, type RuntimeContext, type WebSocketContext } from \"./context\";\nimport { type ExtendedApi } from \"./decorators\";\nimport { getSecrets } from \"./secret\";\nimport { type SonamuSecrets } from \"./secret\";\nimport {\n  createWebSocketReplyStub,\n  resolveWebSocketCloseDescriptor,\n  resolveWebSocketPluginOptions,\n  resolveIntegratedViteHmrOptions,\n} from \"./websocket-helpers\";\n\nexport {\n  createWebSocketReplyStub,\n  resolveWebSocketCloseDescriptor,\n  resolveWebSocketPluginOptions,\n} from \"./websocket-helpers\";\n\nclass SonamuClass {\n  public isInitialized: boolean = false;\n  public forTesting: boolean = false;\n  public asyncLocalStorage: AsyncLocalStorage<{\n    context: RuntimeContext;\n  }> = new AsyncLocalStorage();\n\n  public getContext<T extends RuntimeContext = Context>(): T {\n    const store = this.asyncLocalStorage.getStore();\n    if (store?.context) {\n      return store.context as T;\n    }\n\n    if (process.env.NODE_ENV === \"test\") {\n      // 테스팅 환경에서 컨텍스트가 주입되지 않은 경우 빈 컨텍스트 리턴\n      return {\n        transport: \"http\",\n        request: null,\n        reply: null,\n        headers: {},\n        createSSE: (schema: ZodObject) => createMockSSEFactory(schema),\n        locale: \"\",\n        user: null,\n        session: null,\n        naiteStore: new Map<string, any>(),\n      } as unknown as T;\n    } else {\n      throw new Error(\"Sonamu cannot find context\");\n    }\n  }\n\n  private _apiRootPath: AbsolutePath | null = null;\n  set apiRootPath(apiRootPath: AbsolutePath) {\n    this._apiRootPath = apiRootPath;\n  }\n  get apiRootPath(): AbsolutePath {\n    if (this._apiRootPath === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n    return this._apiRootPath;\n  }\n  get appRootPath(): string {\n    return this.apiRootPath.split(path.sep).slice(0, -1).join(path.sep);\n  }\n\n  private _dbConfig: SonamuDBConfig | null = null;\n  set dbConfig(dbConfig: SonamuDBConfig) {\n    this._dbConfig = dbConfig;\n  }\n  get dbConfig(): SonamuDBConfig {\n    if (this._dbConfig === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n    return this._dbConfig;\n  }\n\n  private _syncer: Syncer | null = null;\n  set syncer(syncer: Syncer) {\n    this._syncer = syncer;\n  }\n  get syncer(): Syncer {\n    if (this._syncer === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n    return this._syncer;\n  }\n\n  private _config: SonamuConfig | null = null;\n  set config(config: SonamuConfig) {\n    this._config = config;\n  }\n  get config(): SonamuConfig {\n    if (this._config === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n    return this._config;\n  }\n\n  public readonly secrets: SonamuSecrets = getSecrets();\n\n  private _storage: StorageManager | null = null;\n  /**\n   * StorageManager 인스턴스\n   */\n  get storage(): StorageManager {\n    if (!this._storage) {\n      throw new Error(\"Storage has not been initialized. Check storage config.\");\n    }\n    return this._storage;\n  }\n\n  private _cache: CacheManager | null = null;\n  /**\n   * CacheManager 인스턴스 (BentoCache)\n   */\n  get cache(): CacheManager {\n    if (!this._cache) {\n      throw new Error(\"Cache has not been initialized. Check cache config in sonamu.config.ts.\");\n    }\n    return this._cache;\n  }\n\n  private _workflows: WorkflowManager | null = null;\n  get workflows(): WorkflowManager {\n    if (this._workflows === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n\n    return this._workflows;\n  }\n\n  private _auth: Auth<BetterAuthOptions> | null = null;\n  get auth(): Auth<BetterAuthOptions> {\n    if (!this._auth) {\n      throw new Error(\"Auth has not been initialized. Check auth config in sonamu.config.ts.\");\n    }\n    return this._auth;\n  }\n\n  private _devVitestManager: DevVitestManager | null = null;\n  get devVitestManager(): DevVitestManager | null {\n    return this._devVitestManager;\n  }\n  set devVitestManager(manager: DevVitestManager | null) {\n    this._devVitestManager = manager;\n  }\n\n  // Sonamu가 runtime을 직접 소유해 registry/connection lifecycle을 애플리케이션 수명주기와 동기화함\n  private _websocketRuntime: WebSocketRuntime | null = null;\n  // 같은 Fastify 인스턴스에 @fastify/websocket을 중복 등록하는 것을 WeakSet으로 차단함\n  private readonly websocketPluginServers = new WeakSet<\n    FastifyInstance<Server, IncomingMessage, ServerResponse>\n  >();\n  get websocketRuntime(): WebSocketRuntime {\n    if (!this._websocketRuntime) {\n      throw new Error(\"WebSocket runtime has not been initialized.\");\n    }\n    return this._websocketRuntime;\n  }\n  set websocketRuntime(runtime: WebSocketRuntime | null) {\n    this._websocketRuntime = runtime;\n  }\n\n  // HMR 처리: 파일 시스템 감시 + HMR/sync 사이클 실행은 watcher 모듈로 위임합니다.\n  public watcher: FSWatcher | null = null;\n\n  public server: FastifyInstance | null = null;\n\n  async initForTesting() {\n    await this.init(true, false, undefined, true);\n  }\n\n  async init(\n    doSilent: boolean = false,\n    enableSync: boolean = true,\n    apiRootPath?: AbsolutePath,\n    forTesting: boolean = false,\n  ) {\n    this.forTesting = forTesting;\n\n    if (this.isInitialized) {\n      return;\n    }\n\n    const initStart = performance.now();\n\n    // API 루트 패스\n    const { findApiRootPath } = await import(\"../utils/utils\");\n    this.apiRootPath = apiRootPath ?? findApiRootPath();\n\n    // 설정을 로딩하는 것부터 시작\n    const configStart = performance.now();\n    const { loadConfig } = await import(\"./config\");\n    this.config = await loadConfig(this.apiRootPath);\n    const configTime = performance.now() - configStart;\n    setSDConfig(this.config.i18n);\n    // sonamu.config.ts 기본값 설정\n    this.config.database.database = this.config.database.database ?? \"pg\";\n    this.config.database.defaultOptions.client = this.config.database.database ?? \"pg\";\n\n    // 로깅 설정\n    const { configureLogTape } = await import(\"../logger/configure\");\n    if (this.config.logging !== false) {\n      await configureLogTape({\n        ...this.config.logging,\n      });\n    }\n\n    // DB 로드\n    const { DB } = await import(\"../database/db\");\n    this.dbConfig = DB.generateDBConfig(this.config.database);\n    DB.setConfig(this.dbConfig);\n\n    // Entity 로드\n    // 테스트에서도 Entity 정보는 필요합니다.\n    // upsert가 제대로 작동하려면 entity의 unique index 정보가 필요하기 때문입니다.\n    const { EntityManager } = await import(\"../entity/entity-manager\");\n    await EntityManager.autoload(doSilent);\n\n    // Cache 초기화\n    await this.initializeCache(this.config.server.cache, forTesting);\n\n    // BetterAuth 초기화\n    const authConfig = this.config.server.auth;\n    if (authConfig) {\n      // 사용자 설정과 기본값을 merge\n      const mergedFieldMappings = merge(BASE_FIELD_MAPPINGS, authConfig);\n\n      // better-auth 인스턴스 생성\n      const { betterAuth } = await import(\"better-auth\");\n      const { sonamuKnexAdapter } = await import(\"../auth/knex-adapter\");\n\n      const authOptions: BetterAuthOptions = {\n        database: sonamuKnexAdapter(),\n        ...mergedFieldMappings,\n      };\n      this._auth = betterAuth(authOptions);\n    }\n\n    // 테스팅인 경우 싱크 없이 중단\n    if (forTesting) {\n      this.isInitialized = true;\n      return;\n    }\n\n    // Task 등록\n    await this.initializeWorkflows(this.config.tasks);\n\n    // Syncer\n    const { Syncer } = await import(\"../syncer/syncer\");\n    this.syncer = new Syncer();\n\n    // Autoload: Models / Types / APIs / Workflows / Templates / SSR Routes\n    await this.syncer.autoloadTypes();\n    await this.syncer.autoloadModels();\n    await this.syncer.autoloadApis();\n    await this.syncer.autoloadWorkflows();\n    const { TemplateManager } = await import(\"../template\");\n    await TemplateManager.autoload();\n    await this.syncer.autoloadSsrRoutes();\n\n    const { isLocal, isTest, isHotReloadServer } = await import(\"../utils/controller\");\n    if (isLocal() && !isTest() && isHotReloadServer() && enableSync) {\n      await this.syncer.sync();\n      await this.startWatcher();\n    }\n\n    this.isInitialized = true;\n    this._initElapsed = performance.now() - initStart;\n    this._configElapsed = configTime;\n  }\n\n  private _initElapsed = 0;\n  private _configElapsed = 0;\n\n  async createServer(initOptions?: { enableSync?: boolean; doSilent?: boolean }) {\n    if (!this.isInitialized) {\n      await this.init(initOptions?.doSilent, initOptions?.enableSync);\n    }\n\n    const options = this.config.server;\n    const { default: fastify } = await import(\"fastify\");\n    const { getLogTapeFastifyLogger } = await import(\"@logtape/fastify\");\n    const server = fastify({\n      ...options.fastify,\n      logger:\n        this.config.logging !== false\n          ? getLogTapeFastifyLogger({\n              category: this.config.logging?.fastifyCategory ?? [\"fastify\"],\n            })\n          : undefined,\n    });\n    this.server = server;\n    this.websocketRuntime = new WebSocketRuntime(options.websocket);\n\n    // Storage 설정 → StorageManager 생성\n    if (options.storage) {\n      const { StorageManager } = await import(\"../storage/storage-manager\");\n      this._storage = new StorageManager(options.storage);\n    }\n\n    // 플러그인 등록\n    if (options.plugins) {\n      await this.registerPlugins(server, options.plugins);\n    }\n\n    if (options.auth) {\n      await this.registerBetterAuth(server, options.auth);\n    }\n\n    // API 라우팅 설정\n    await this.withFastify(server, options.apiConfig, {\n      enableSync: initOptions?.enableSync,\n      doSilent: initOptions?.doSilent,\n    });\n\n    // 서버 시작\n    await this.boot(server, options);\n\n    if (!initOptions?.doSilent) {\n      this.printStartupSummary();\n    }\n\n    return server;\n  }\n\n  async withFastify(\n    server: FastifyInstance<Server, IncomingMessage, ServerResponse>,\n    config: SonamuFastifyConfig,\n    options?: {\n      enableSync?: boolean;\n      doSilent?: boolean;\n    },\n  ) {\n    if (!this.isInitialized) {\n      await this.init(options?.doSilent, options?.enableSync);\n    }\n\n    this.server = server;\n    this.websocketRuntime ??= new WebSocketRuntime(this.config.server.websocket);\n\n    // timezone 설정\n    const timezone = this.config.api.timezone;\n    if (timezone) {\n      // 타임존에 맞게 응답 날짜 스트링을 변환해주어야 합니다.\n      // 가령 timezone이 \"Asia/Seoul\" 이면\n      // \"2025-11-21T00:00:00.000Z\" 를 \"2025-11-21T09:00:00+09:00\" 으로 변환해주어야 합니다.\n      const { formatInTimeZone } = await import(\"date-fns-tz\");\n\n      // ISO 8601 날짜 형식 정규식 (예: 2024-01-15T09:30:00.000Z)\n      const ISO_DATE_REGEX = /^\\d{4}-\\d{2}-\\d{2}T\\d{2}:\\d{2}:\\d{2}(\\.\\d{3})?Z$/;\n\n      // T를 둘러싼 작은따옴표가 없다면 \"2025-11-19176354618900018:56:29+09:00\"와 같은 결과가 나옵니다.\n      // 이는 date-fns 특입니다.\n      // 이렇게 해도 괜찮습니다. \"2025-11-19T18:56:29+09:00\" 모양으로 잘 나옵니다.\n      const DATE_FORMAT = \"yyyy-MM-dd'T'HH:mm:ssXXX\";\n\n      server.setReplySerializer((payload) => {\n        return JSON.stringify(payload, (_key, value) => {\n          if (typeof value === \"string\" && ISO_DATE_REGEX.test(value)) {\n            return formatInTimeZone(\n              new Date(value),\n              timezone as `${string}/${string}`,\n              DATE_FORMAT,\n            );\n          }\n          return value;\n        });\n      });\n      // Timezone 로그는 printStartupSummary에서 통합 출력\n    }\n\n    // 전체 라우팅 리스트\n    server.get(\n      `${this.config.api.route.prefix}/routes`,\n      async (_request, _reply): Promise<typeof this.syncer.apis> => {\n        return this.syncer.apis;\n      },\n    );\n\n    // Healthcheck API\n    server.get(\n      `${this.config.api.route.prefix}/healthcheck`,\n      async (_request, _reply): Promise<string> => {\n        return \"ok\";\n      },\n    );\n\n    // Sonamu UI API (로컬 환경에서만)\n    const { isLocal } = await import(\"../utils/controller\");\n    if (isLocal()) {\n      const { sonamuUIApiPlugin } = await import(\"../ui/api\");\n      server.register(sonamuUIApiPlugin);\n    }\n\n    // DevRunner 테스트 엔드포인트 (로컬 환경 + devRunner 활성화 시)\n    if (isLocal() && this.config.test?.devRunner?.enabled) {\n      const { registerDevTestRoutes } = await import(\"../testing/dev-test-routes\");\n      await registerDevTestRoutes(server, this.config.test.devRunner);\n    }\n\n    const webPath = path.join(this.appRootPath, \"web\");\n    const hasWeb = await exists(webPath);\n\n    // 전역 compress 옵션 계산 (route.compress: true일 때 사용)\n    const pluginCompress = this.config.server.plugins?.compress;\n    const globalCompressOptions: CompressOptions | undefined = pluginCompress\n      ? pluginCompress === true\n        ? { threshold: 1024, encodings: [\"br\", \"gzip\", \"deflate\"] }\n        : {\n            threshold: pluginCompress.threshold,\n            encodings: pluginCompress.encodings,\n            customTypes: pluginCompress.customTypes,\n          }\n      : undefined;\n\n    if (isLocal()) {\n      // 로컬 개발 환경: catch-all로 API를 동적 매칭하여 HMR을 지원합니다.\n      // SONAMU_DISABLE_INTEGRATED_WEB=yes로 설정하면 dev_api 모드에서 Vite 통합을 비활성화할 수 있습니다.\n      const disableIntegratedWeb = process.env.SONAMU_DISABLE_INTEGRATED_WEB === \"yes\";\n      if (hasWeb && !disableIntegratedWeb) {\n        await this.setupDevServerWithVite(server, webPath, config);\n      } else {\n        this.setupDevServer(server, config);\n      }\n    } else {\n      // 프로덕션 환경: 개별 API 라우트 + 정적 파일 서빙\n      for (const api of this.syncer.apis) {\n        if (this.syncer.models[api.modelName] === undefined) {\n          throw new Error(`정의되지 않은 모델에 접근 ${api.modelName}`);\n        }\n\n        // @websocket route는 wsHandler로 등록하고, 같은 path의 일반 HTTP GET은 426 응답으로 upgrade를 강제함\n        if (api.websocketOptions) {\n          server.route({\n            method: \"GET\",\n            url: this.config.api.route.prefix + api.path,\n            handler: this.createWebSocketUpgradeRequiredHandler(),\n            wsHandler: this.createWebSocketHandler(api, config),\n          });\n          continue;\n        }\n\n        server.route({\n          method: api.options.httpMethod ?? \"GET\",\n          url: this.config.api.route.prefix + api.path,\n          handler: this.createApiHandler(api, config),\n          compress: toFastifyCompressOption(api.options.compress, globalCompressOptions),\n        });\n      }\n\n      // 프로덕션에서는 web 소스(appRoot/web) 유무와 무관하게,\n      // api/web-dist 존재 여부를 setupStaticWebServer 내부에서 판단합니다.\n      await this.setupStaticWebServer(server, config, globalCompressOptions);\n    }\n  }\n\n  /**\n   * dev 모드 공통: catch-all에서 syncer.apis를 동적으로 탐색하여 API 요청을 처리합니다.\n   * server.route()로 개별 등록하면 handler가 고정되어 HMR이 동작하지 않으므로,\n   * 매 요청마다 syncer.apis를 조회하는 이 방식을 사용합니다.\n   *\n   * 요청이 /api(정확히는 this.config.api.route.prefix)로 시작하지 않는 경우라면 null을 반환하며 끝냅니다.\n   */\n  private handleDevApiRequest(\n    request: FastifyRequest,\n    config: SonamuFastifyConfig,\n  ): ((request: FastifyRequest, reply: FastifyReply) => Promise<unknown>) | null {\n    const matchedApi = this.findMatchedApi(request);\n\n    if (!matchedApi) {\n      throw new NotFoundException(SD(\"error.api.notFound\"));\n    }\n\n    // websocket route를 일반 HTTP로 직접 호출한 경우 426을 돌려줘 upgrade 없이 접근하는 것을 차단함\n    if (matchedApi.websocketOptions) {\n      return this.createWebSocketUpgradeRequiredHandler();\n    }\n\n    return this.createApiHandler(matchedApi, config);\n  }\n\n  private findMatchedApi(request: FastifyRequest): ExtendedApi | undefined {\n    const url = this.getPathnameFromUrl(request.url);\n    const method = request.method;\n\n    if (!url.startsWith(this.config.api.route.prefix)) {\n      return undefined;\n    }\n\n    return this.syncer.apis.find((api) => {\n      if (this.syncer.models[api.modelName] === undefined) {\n        return false;\n      }\n      const apiMethod = api.options.httpMethod ?? \"GET\";\n      if (apiMethod !== method) return false;\n\n      const fullPath = this.config.api.route.prefix + api.path;\n      return this.isPathPatternMatch(fullPath, url);\n    });\n  }\n\n  /**\n   * dev api 모드: Vite 없이 API 동적 라우팅만 제공합니다.\n   * HMR을 위해 catch-all에서 매 요청마다 syncer.apis를 조회합니다.\n   */\n  private setupDevServer(\n    server: FastifyInstance<Server, IncomingMessage, ServerResponse>,\n    config: SonamuFastifyConfig,\n  ): void {\n    // upgrade는 실질적으로 GET에서만 성립하므로 GET + wsHandler 와 그 외 method를 별도 route로 분리함\n    server.route({\n      method: \"GET\",\n      url: `${this.config.api.route.prefix}/*`,\n      handler: async (request, reply) => {\n        const handler = this.handleDevApiRequest(request, config);\n        if (handler) {\n          return handler(request, reply);\n        }\n        // 등록된 API와 일치하지 않는 요청에 대한 fallback입니다.\n        throw new NotFoundException(SD(\"error.api.notFound\"));\n      },\n      wsHandler: async (connection, request) => {\n        await this.handleDevWebSocketRequest(connection.socket, request, config);\n      },\n    });\n\n    server.route({\n      method: [\"HEAD\", \"POST\", \"PUT\", \"DELETE\", \"PATCH\"],\n      url: `${this.config.api.route.prefix}/*`,\n      handler: async (request, reply) => {\n        const handler = this.handleDevApiRequest(request, config);\n        if (handler) {\n          return handler(request, reply);\n        }\n        throw new NotFoundException(SD(\"error.api.notFound\"));\n      },\n    });\n  }\n\n  private viteServer: any = null;\n\n  /**\n   * dev all 모드: Vite Dev Server를 통합하여 API + SSR + CSR을 모두 제공합니다.\n   * API 동적 매칭은 handleDevApiRequest를 공유합니다.\n   */\n  private async setupDevServerWithVite(\n    server: FastifyInstance<Server, IncomingMessage, ServerResponse>,\n    webPath: string,\n    config: SonamuFastifyConfig,\n  ): Promise<void> {\n    // @fastify/middie 등록 (Connect-style middleware 지원)\n    await server.register((await import(\"@fastify/middie\")).default);\n\n    const vite = await import(\"vite\");\n    // @fastify/websocket 플러그인이 활성화되면 HMR websocket과 server socket이 충돌하므로 dedicated 포트로 분리함\n    const requiresDedicatedHmrServer = Boolean(this.config.server.plugins?.ws);\n    const hmr = resolveIntegratedViteHmrOptions({\n      httpServer: server.server,\n      requiresDedicatedWebSocketServer: requiresDedicatedHmrServer,\n    });\n\n    this.viteServer = await vite.createServer({\n      root: webPath,\n      server: {\n        middlewareMode: true,\n        hmr,\n      },\n      appType: \"custom\",\n    });\n\n    // Vite middleware 등록 (Vite 에셋 처리)\n    server.use((req, res, next) => {\n      // API와 Sonamu UI는 Fastify 라우트가 처리하도록 skip\n      if (req.url?.startsWith(this.config.api.route.prefix) || req.url?.startsWith(\"/sonamu-ui\")) {\n        return next();\n      }\n      // 나머지는 Vite middleware로 전달\n      return this.viteServer.middlewares(req, res, next);\n    });\n\n    // WS upgrade 경로(GET)와 일반 HTTP 메서드를 별도 route로 분리해 websocket route가 HTML fallback에 먹히지 않도록 함\n    server.route({\n      method: \"GET\",\n      url: `${this.config.api.route.prefix}/*`,\n      handler: async (request, reply) => {\n        const result = this.handleDevApiRequest(request, config);\n        if (result) {\n          return result(request, reply);\n        }\n        throw new NotFoundException(SD(\"error.api.notFound\"));\n      },\n      wsHandler: async (connection, request) => {\n        await this.handleDevWebSocketRequest(connection.socket, request, config);\n      },\n    });\n\n    server.route({\n      method: [\"HEAD\", \"POST\", \"PUT\", \"DELETE\", \"PATCH\"],\n      url: `${this.config.api.route.prefix}/*`,\n      handler: async (request, reply) => {\n        const result = this.handleDevApiRequest(request, config);\n        if (result) {\n          return result(request, reply);\n        }\n        throw new NotFoundException(SD(\"error.api.notFound\"));\n      },\n    });\n\n    // catch-all 라우트에서 SSR/CSR 처리\n    // 개발 환경에서는 라우트별 compress 옵션을 포기하고 HMR 이점을 취합니다.\n    server.route({\n      method: [\"GET\", \"HEAD\"],\n      url: \"/*\",\n      handler: async (request, reply) => {\n        const url = request.url;\n\n        // 1. SSR 라우트 처리\n        const { matchSSRRoute, renderSSR } = await import(\"../ssr\");\n        const ssrMatch = matchSSRRoute(url);\n        if (ssrMatch) {\n          console.log(`[SSR] Matched route: ${ssrMatch.route.path}`);\n          const html = await renderSSR(\n            url,\n            ssrMatch.route,\n            ssrMatch.params,\n            request,\n            reply,\n            config,\n            this.viteServer,\n          );\n          reply.type(\"text/html\");\n          return html;\n        }\n\n        // 2. CSR fallback\n        try {\n          const fs = await import(\"node:fs/promises\");\n          let template = await fs.readFile(\n            path.join(this.viteServer.config.root, \"index.html\"),\n            \"utf-8\",\n          );\n          template = await this.viteServer.transformIndexHtml(url, template);\n\n          reply.type(\"text/html\");\n          return template;\n        } catch (e) {\n          this.viteServer.ssrFixStacktrace(e as Error);\n          console.error(e);\n          reply.status(500);\n          return (e as Error).message;\n        }\n      },\n    });\n\n    // 서버 종료 시 Vite도 종료\n    server.addHook(\"onClose\", async () => {\n      await this.viteServer.close();\n    });\n\n    const chalk = (await import(\"chalk\")).default;\n    if (\"port\" in hmr) {\n      console.log(\n        chalk.dim(\n          `✓ Vite HMR using dedicated websocket port ${hmr.port} to avoid Fastify websocket conflicts`,\n        ),\n      );\n    }\n    console.log(chalk.dim(\"✓ Vite dev server integrated\"));\n  }\n\n  private async setupStaticWebServer(\n    server: FastifyInstance<Server, IncomingMessage, ServerResponse>,\n    config: SonamuFastifyConfig,\n    globalCompressOptions: CompressOptions | undefined,\n  ): Promise<void> {\n    // 경로 명확화: api/web-dist/client (정적 파일), api/web-dist/server (SSR entry), api/dist/ssr (SSR routes - API 소유)\n    const webDistPath = path.join(this.apiRootPath, \"web-dist\", \"client\");\n    const ssrPath = path.join(this.apiRootPath, \"web-dist\", \"server\");\n    const ssrEntryPath = path.join(ssrPath, \"entry-server.generated.js\");\n    const ssrRoutesPath = path.join(this.apiRootPath, \"dist\", \"ssr\", \"routes.js\");\n\n    if (!(await exists(webDistPath))) {\n      console.warn(`⚠ Web dist not found: ${webDistPath}`);\n      return;\n    }\n\n    // SSR entry 존재 여부 확인\n    const ssrAvailable = await exists(ssrEntryPath);\n\n    if (!ssrAvailable) {\n      console.warn(`⚠ SSR entry not found: ${ssrEntryPath}`);\n      console.warn(\"  SSR will be disabled. Only CSR will work.\");\n    }\n\n    // SSR 라우트 로드 (production에서만, 사용자 프로젝트의 ssr/routes.ts)\n    if (ssrAvailable) {\n      if (await exists(ssrRoutesPath)) {\n        // ts-loader라면 \"file://\"로 시작하는 fully-resolved path만 받기에 이를 처리해주는 importMembers를 사용해야 했겠지만,\n        // 여기는 프로덕션 환경에서 loader 없이 돌아가기 때문에 \"진짜 js 파일\"의 \"그냥\" 절대경로를 바로 import해도 됩니다.\n        // 이 내용은 이 함수 내에서 아래에 나올 다른 import 호출에도 동일하게 적용됩니다.\n        await import(ssrRoutesPath);\n        console.log(\"✓ SSR routes loaded\");\n      } else {\n        console.warn(`⚠ SSR routes not found: ${ssrRoutesPath}`);\n      }\n    }\n\n    // 롤링 업데이트 대응: asset hash 불일치 시 현재 버전 직접 서빙\n    server.get(\"/assets/:filename\", async (request, reply) => {\n      const requestedFile = (request.params as { filename: string }).filename;\n      const assetsDir = path.join(webDistPath, \"assets\");\n      const safeFilePath = this.resolvePathWithinBaseDir(assetsDir, requestedFile);\n      if (safeFilePath === null) {\n        reply.status(403).send();\n        return;\n      }\n      const normalizedRequestedFile = path.relative(assetsDir, safeFilePath).replace(/\\\\/g, \"/\");\n\n      const assetPath = `/assets/${normalizedRequestedFile}`;\n\n      // Cache-Control 헤더 결정\n      const getCacheControlForAsset = (): CacheControlConfig => {\n        const cacheReq: CacheControlRequest = {\n          type: \"assets\",\n          url: request.url,\n          path: assetPath,\n          method: request.method,\n        };\n\n        // 사용자 정의 핸들러 우선\n        if (config.cacheControlHandler) {\n          const result = config.cacheControlHandler(cacheReq);\n          if (result) return result;\n        }\n\n        // 기본값: immutable\n        return CachePresets.immutable;\n      };\n\n      // index-*.js 또는 index-*.css 요청인 경우\n      if (/^index-[a-f0-9]+\\.(js|css)$/.test(normalizedRequestedFile)) {\n        const ext = normalizedRequestedFile.split(\".\").pop();\n        const files = await fs.readdir(assetsDir);\n        const currentFile = files.find((f) => f.startsWith(\"index-\") && f.endsWith(`.${ext}`));\n\n        if (currentFile) {\n          const filePath = path.join(assetsDir, currentFile);\n          const content = await fs.readFile(filePath);\n          reply.type(ext === \"js\" ? \"application/javascript\" : \"text/css\");\n          applyCacheHeaders(reply, getCacheControlForAsset());\n          return reply.send(content);\n        }\n      }\n\n      // 일반 파일 서빙\n      const filePath = safeFilePath;\n      if (await exists(filePath)) {\n        const content = await fs.readFile(filePath);\n        const ext = normalizedRequestedFile.split(\".\").pop();\n        reply.type(ext === \"js\" ? \"application/javascript\" : ext === \"css\" ? \"text/css\" : \"\");\n        if (normalizedRequestedFile.includes(\"-\")) {\n          applyCacheHeaders(reply, getCacheControlForAsset());\n        }\n        return reply.send(content);\n      }\n\n      reply.status(404).send();\n    });\n\n    // SSR 라우트 개별 등록 (compress 옵션이 라우트별로 적용되도록)\n    if (ssrAvailable) {\n      const { getSSRRoutes } = await import(\"../ssr\");\n      const { renderSSR } = await import(\"../ssr/renderer\");\n      const ssrRoutes = getSSRRoutes();\n\n      for (const route of ssrRoutes) {\n        server.route({\n          method: [\"GET\", \"HEAD\"],\n          url: route.path,\n          compress: toFastifyCompressOption(route.compress ?? true, globalCompressOptions),\n          handler: async (request, reply) => {\n            const url = request.url;\n            console.log(`[SSR] Matched route: ${route.path}`);\n\n            const params = this.extractPathParams(route.path, url);\n            const html = await renderSSR(url, route, params, request, reply, config);\n\n            reply.type(\"text/html\");\n            return html;\n          },\n        });\n      }\n    }\n\n    // CSR or Static File Fallback (SSR 라우트에 매칭되지 않는 모든 요청)\n    server.route({\n      method: [\"GET\", \"HEAD\"],\n      url: \"*\",\n      handler: async (request, reply) => {\n        // /api, /sonamu-ui는 404 그대로\n        if (request.url.startsWith(\"/api\") || request.url.startsWith(\"/sonamu-ui\")) {\n          reply.status(404).send();\n          return;\n        }\n\n        // CSR용 Cache-Control 헤더 설정\n        if (config.cacheControlHandler) {\n          const csrCacheReq: CacheControlRequest = {\n            type: \"csr\",\n            url: request.url,\n            path: request.url.split(\"?\")[0],\n            method: request.method,\n          };\n          const csrCacheConfig = config.cacheControlHandler(csrCacheReq);\n\n          if (csrCacheConfig) {\n            applyCacheHeaders(reply, csrCacheConfig);\n          }\n        }\n\n        // 정적 파일이 존재할 경우, 정적 파일을 먼저 서빙해야함\n        const requestPath = this.getPathnameFromUrl(request.url);\n        const safeFilePath = this.resolvePathWithinBaseDir(webDistPath, requestPath);\n        if (safeFilePath === null) {\n          reply.status(403).send();\n          return;\n        }\n        if (await fileExists(safeFilePath)) {\n          const content = await fs.readFile(safeFilePath);\n          return reply.type(mimeLookup(safeFilePath) || \"application/octet-stream\").send(content);\n        }\n\n        // CSR fallback: index.html 서빙\n        const indexPath = path.join(webDistPath, \"index.html\");\n        return reply.type(\"text/html\").send(await fs.readFile(indexPath, \"utf-8\"));\n      },\n    });\n\n    console.log(`✓ Static web server configured with ${ssrAvailable ? \"SSR\" : \"CSR only\"} support`);\n  }\n\n  createApiHandler(\n    api: ExtendedApi,\n    config: SonamuFastifyConfig,\n  ): (request: FastifyRequest, reply: FastifyReply) => Promise<unknown> {\n    return async (request: FastifyRequest, reply: FastifyReply): Promise<unknown> => {\n      // Context 생성\n      const context: Context = await this.createContext(config, request, reply);\n\n      return this.asyncLocalStorage.run({ context }, async () => {\n        // guards 처리\n        runGuards({\n          guards: api.options.guards,\n          config,\n          request,\n          api,\n        });\n\n        // 파라미터 정보로 zod 스키마 빌드\n        const { getZodObjectFromApi } = await import(\"./code-converters\");\n        const ReqType = getZodObjectFromApi(api, this.syncer.types);\n\n        // request 파싱\n        const which = api.options.httpMethod === \"GET\" ? \"query\" : \"body\";\n        let reqBody: {\n          [key: string]: unknown;\n        };\n        // 파일 업로드 있는 경우 임시 데이터\n        const files: {\n          bufferedFiles: BufferedFile[];\n          uploadedFiles: UploadedFile[];\n        } = {\n          bufferedFiles: [],\n          uploadedFiles: [],\n        };\n\n        try {\n          const body = (request[which] ?? {}) as Record<string, unknown>;\n          if (api.uploadOptions) {\n            const parts = request.parts({\n              limits: api.uploadOptions.limits,\n            });\n\n            // FormData의 field들을 임시로 저장\n            const fields: Record<string, string> = {};\n\n            if (api.uploadOptions.consume === \"buffer\" || !api.uploadOptions.consume) {\n              // Buffer 모드: 메모리에 로드\n              for await (const part of parts) {\n                if (part.type === \"file\") {\n                  // CRITICAL: 파일 스트림을 즉시 consume해야 다음 part로 넘어갈 수 있음\n                  // 이 호출이 없으면 종종 multipart 파싱이 pending 상태로 타임아웃 발생\n                  const buffer = await part.toBuffer();\n                  files.bufferedFiles.push(new BufferedFile(part, buffer));\n                } else if (part.type === \"field\") {\n                  fields[part.fieldname] = String(part.value);\n                }\n              }\n            } else if (api.uploadOptions.consume === \"stream\") {\n              // Stream 모드: 즉시 저장소로 스트리밍\n              const diskName = api.uploadOptions.destination;\n              const disk = this.storage.use(diskName);\n\n              // 우선순위: 데코레이터 > 전역 설정 > 기본값\n              const keyGenerator: KeyGenerator =\n                api.uploadOptions.keyGenerator ??\n                this.config.server.storage?.keyGenerator ??\n                defaultKeyGenerator;\n\n              for await (const part of parts) {\n                if (part.type === \"file\") {\n                  const key = await keyGenerator({\n                    filename: part.filename,\n                    mimetype: part.mimetype,\n                  });\n\n                  await disk.putStream(key, part.file, {\n                    contentType: part.mimetype,\n                  });\n\n                  const url = await disk.getUrl(key);\n                  const signedUrl = await disk.getSignedUrl(key);\n\n                  files.uploadedFiles.push(\n                    new UploadedFile({\n                      filename: part.filename,\n                      mimetype: part.mimetype,\n                      size: part.file.bytesRead,\n                      url,\n                      signedUrl,\n                      key,\n                      diskName,\n                    }),\n                  );\n                } else if (part.type === \"field\") {\n                  fields[part.fieldname] = String(part.value);\n                }\n              }\n            }\n\n            // qs로 중첩 구조 파싱: params[category] → { params: { category: \"test\" } }\n            const qs = await import(\"qs\");\n            const parsed = qs.default.parse(fields);\n            Object.assign(body, parsed);\n          }\n\n          const { fastifyCaster } = await import(\"./caster\");\n          reqBody = fastifyCaster(ReqType).parse(body);\n        } catch (e) {\n          const { ZodError } = await import(\"zod\");\n          if (e instanceof ZodError) {\n            const { humanizeZodError } = await import(\"../utils/zod-error\");\n            const messages = humanizeZodError(e)\n              .map((issue) => issue.message)\n              .join(\" \");\n            const { BadRequestException } = await import(\"../exceptions/so-exceptions\");\n            throw new BadRequestException(messages as LocalizedString, {\n              zodError: e,\n            });\n          } else {\n            throw e;\n          }\n        }\n\n        // Content-Type\n        reply.type(api.options.contentType ?? \"application/json\");\n\n        // Cache-Control 헤더 설정\n        const apiCacheConfig = this.getApiCacheControl(api, request, config);\n        if (apiCacheConfig) {\n          applyCacheHeaders(reply, apiCacheConfig);\n        }\n\n        // 업로드 옵션이 있는 경우 파일 데이터를 Context에 추가\n        if (api.uploadOptions) {\n          const consume = api.uploadOptions.consume ?? \"buffer\";\n          if (consume === \"buffer\") {\n            context.bufferedFiles = files.bufferedFiles;\n          } else if (consume === \"stream\") {\n            context.uploadedFiles = files.uploadedFiles;\n          }\n        }\n\n        // 모델 메소드 args 생성하여 호출\n        const { ApiParamType } = await import(\"../types/types\");\n        const args = api.parameters.map((param) => {\n          // Context 인젝션\n          if (ApiParamType.isContext(param.type)) {\n            return context;\n          } else {\n            return reqBody[param.name];\n          }\n        });\n\n        return this.invokeModelMethod(api, args, reply);\n      });\n    };\n  }\n\n  // WS path를 일반 HTTP GET으로 호출한 경우 426 + Upgrade 헤더로 명시적으로 websocket 접속을 유도함\n  private createWebSocketUpgradeRequiredHandler() {\n    return async (_request: FastifyRequest, reply: FastifyReply): Promise<void> => {\n      reply.header(\"connection\", \"Upgrade\").header(\"upgrade\", \"websocket\").status(426).send({\n        message: \"WebSocket upgrade required\",\n      });\n    };\n  }\n\n  // dev 모드의 catch-all wsHandler에서 실제 WS API로 디스패치함. 매칭되는 route가 없으면 1008로 닫음\n  private async handleDevWebSocketRequest(\n    socket: WebSocket,\n    request: FastifyRequest,\n    config: SonamuFastifyConfig,\n  ): Promise<void> {\n    const matchedApi = this.findMatchedApi(request);\n    if (!matchedApi?.websocketOptions) {\n      socket.close(1008, \"WebSocket route not found\");\n      return;\n    }\n\n    const handler = this.createWebSocketHandler(matchedApi, config);\n    await handler({ socket }, request);\n  }\n\n  // WS route 핸들러의 실행 순서를 고정함:\n  // 1) guard를 connection 등록 이전에 돌려 인증 실패 시 부분 등록 상태를 남기지 않음\n  // 2) query param 파싱도 activation 전에 끝내 handshake 실패가 registry에 노출되지 않게 함\n  // 3) `active: false`로 먼저 등록하고, context 준비가 끝난 뒤 `activate()`해 브로드캐스트가 초기화 중간 상태를 보지 못하게 함\n  // 에러 발생 시에는 resolveWebSocketCloseDescriptor 정책에 따라 close code를 매핑함\n  private createWebSocketHandler(api: ExtendedApi, config: SonamuFastifyConfig) {\n    return async (\n      connection: {\n        socket: WebSocket;\n      },\n      request: FastifyRequest,\n    ): Promise<void> => {\n      const socket = connection.socket;\n      let wsContext: WebSocketContext | null = null;\n      let rawWs: ReturnType<WebSocketRuntime[\"registerConnection\"]> | null = null;\n\n      try {\n        runGuards({\n          guards: api.options.guards,\n          config,\n          request,\n          api,\n        });\n\n        const reqBody = await this.parseWebSocketRequestParams(api, request);\n        rawWs = this.websocketRuntime.registerConnection(socket, {\n          outEvents: api.websocketOptions!.outEvents,\n          inEvents: api.websocketOptions!.inEvents,\n          namespace: api.websocketOptions!.namespace,\n          heartbeat: api.websocketOptions!.heartbeat,\n          maxPayload: api.websocketOptions!.maxPayload,\n          active: false,\n        });\n\n        const scopedWs = this.createScopedWebSocketConnection(rawWs, () => wsContext);\n        wsContext = await this.createWebSocketContext(config, request, scopedWs);\n        this.websocketRuntime.activateConnection(rawWs.id);\n\n        const { ApiParamType } = await import(\"../types/types\");\n        const args = api.parameters.map((param) => {\n          if (ApiParamType.isContext(param.type)) {\n            return wsContext;\n          }\n\n          return reqBody[param.name];\n        });\n\n        await this.asyncLocalStorage.run({ context: wsContext }, async () => {\n          await this.invokeModelMethod(api, args);\n        });\n      } catch (error) {\n        const closeDescriptor = resolveWebSocketCloseDescriptor(error);\n        if (rawWs) {\n          rawWs.close(closeDescriptor.code, closeDescriptor.reason);\n        } else if (socket.readyState < 2) {\n          socket.close(closeDescriptor.code, closeDescriptor.reason);\n        }\n\n        if (this.server?.log) {\n          const payload = {\n            err: error,\n            modelName: api.modelName,\n            methodName: api.methodName,\n            path: api.path,\n          };\n          if (closeDescriptor.logLevel === \"warn\") {\n            this.server.log.warn(payload, closeDescriptor.reason);\n          } else {\n            this.server.log.error(payload, closeDescriptor.reason);\n          }\n        } else {\n          if (closeDescriptor.logLevel === \"warn\") {\n            console.warn(closeDescriptor.reason, error);\n          } else {\n            console.error(closeDescriptor.reason, error);\n          }\n        }\n      }\n    };\n  }\n\n  // onMessage/onClose처럼 다른 tick에서 실행되는 callback은 ALS context가 끊기므로 wrapper에서 `asyncLocalStorage.run`으로 다시 감싸 복원함\n  // publish/join/leave/setUserId 같은 즉시 실행 API는 단순 위임만 하고, deferred callback에만 context 복원을 적용함\n  private createScopedWebSocketConnection<\n    TOut extends WebSocketEventMap,\n    TIn extends WebSocketEventMap,\n  >(\n    ws: WebSocketConnection<TOut, TIn>,\n    getContext: () => WebSocketContext | null,\n  ): WebSocketConnection<TOut, TIn> {\n    const runInContext = <T>(callback: () => T): T => {\n      const context = getContext();\n      if (!context) {\n        return callback();\n      }\n\n      return this.asyncLocalStorage.run({ context }, callback);\n    };\n\n    return {\n      get id() {\n        return ws.id;\n      },\n      get namespace() {\n        return ws.namespace;\n      },\n      get closed() {\n        return ws.closed;\n      },\n      transport: \"ws\",\n      publishUntyped(event, data) {\n        ws.publishUntyped(event, data);\n      },\n      close(code, reason) {\n        ws.close(code, reason);\n      },\n      onClose(callback) {\n        ws.onClose(() => runInContext(callback));\n      },\n      onMessage(event, handler) {\n        ws.onMessage(event, (data) => runInContext(() => handler(data)));\n      },\n      publish(event, data) {\n        ws.publish(event, data);\n      },\n      waitForClose() {\n        return ws.waitForClose();\n      },\n      join(roomId) {\n        ws.join(roomId);\n      },\n      leave(roomId) {\n        ws.leave(roomId);\n      },\n      setUserId(userId) {\n        ws.setUserId(userId);\n      },\n      clearUserId() {\n        ws.clearUserId();\n      },\n    };\n  }\n\n  private async parseWebSocketRequestParams(\n    api: ExtendedApi,\n    request: FastifyRequest,\n  ): Promise<Record<string, unknown>> {\n    const { getZodObjectFromApi } = await import(\"./code-converters\");\n    const ReqType = getZodObjectFromApi(api, this.syncer.types);\n\n    try {\n      const { fastifyCaster } = await import(\"./caster\");\n      return fastifyCaster(ReqType).parse((request.query ?? {}) as Record<string, unknown>);\n    } catch (e) {\n      const { ZodError } = await import(\"zod\");\n      if (e instanceof ZodError) {\n        const { humanizeZodError } = await import(\"../utils/zod-error\");\n        const messages = humanizeZodError(e)\n          .map((issue) => issue.message)\n          .join(\" \");\n        const { BadRequestException } = await import(\"../exceptions/so-exceptions\");\n        throw new BadRequestException(messages as LocalizedString, {\n          zodError: e,\n        });\n      }\n\n      throw e;\n    }\n  }\n\n  /**\n   * URL에서 path params를 추출합니다.\n   * 예: pattern=\"/admin/companies/:companyId\", url=\"/admin/companies/123\" → { companyId: \"123\" }\n   */\n  private extractPathParams(pattern: string, url: string): Record<string, string> {\n    const patternParts = pattern.split(\"/\").filter(Boolean);\n    const urlParts = this.getPathnameFromUrl(url).split(\"/\").filter(Boolean);\n    const params: Record<string, string> = {};\n\n    for (let i = 0; i < patternParts.length; i++) {\n      if (patternParts[i].startsWith(\":\")) {\n        params[patternParts[i].slice(1)] = urlParts[i];\n      }\n    }\n    return params;\n  }\n\n  private isPathPatternMatch(pattern: string, url: string): boolean {\n    const patternParts = pattern.split(\"/\").filter(Boolean);\n    const urlParts = this.getPathnameFromUrl(url).split(\"/\").filter(Boolean);\n\n    if (patternParts.length !== urlParts.length) {\n      return false;\n    }\n\n    for (let i = 0; i < patternParts.length; i++) {\n      const patternPart = patternParts[i];\n      const urlPart = urlParts[i];\n      if (patternPart.startsWith(\":\")) {\n        continue;\n      }\n      if (patternPart !== urlPart) {\n        return false;\n      }\n    }\n\n    return true;\n  }\n\n  private getPathnameFromUrl(url: string): string {\n    return url.split(\"?\")[0];\n  }\n\n  private resolvePathWithinBaseDir(baseDir: string, inputPath: string): string | null {\n    try {\n      const decoded = decodeURIComponent(inputPath).replace(/\\\\/g, \"/\");\n      if (decoded.includes(\"\\0\")) {\n        return null;\n      }\n      const relativePath = decoded.replace(/^\\/+/, \"\");\n      const resolvedPath = path.resolve(baseDir, relativePath);\n      const relativeFromBase = path.relative(baseDir, resolvedPath);\n      if (relativeFromBase.startsWith(\"..\") || path.isAbsolute(relativeFromBase)) {\n        return null;\n      }\n      return resolvedPath;\n    } catch {\n      return null;\n    }\n  }\n\n  /**\n   * API 응답에 적용할 Cache-Control 설정을 결정합니다.\n   * 우선순위: 개별 지정 > cacheControlHandler\n   */\n  private getApiCacheControl(\n    api: ExtendedApi,\n    request: FastifyRequest,\n    config: SonamuFastifyConfig,\n  ) {\n    // 데코레이터 설정 우선\n    if (api.options.cacheControl) {\n      return api.options.cacheControl;\n    }\n\n    // 전역 핸들러\n    if (config.cacheControlHandler) {\n      const cacheReq: CacheControlRequest = {\n        type: \"api\",\n        url: request.url,\n        path: request.routeOptions?.url ?? request.url.split(\"?\")[0],\n        method: request.method,\n        api,\n      };\n      const result = config.cacheControlHandler(cacheReq);\n      if (result) return result;\n    }\n\n    return null;\n  }\n\n  /**\n   * SSR용 API 호출 (HTTP 오버헤드 없이 직접 호출)\n   * createApiHandler의 로직을 재사용하되, request 파싱 대신 params 직접 사용\n   */\n  async invokeApiForSSR(\n    api: ExtendedApi,\n    params: any[],\n    config: SonamuFastifyConfig,\n    request: FastifyRequest,\n    reply: FastifyReply,\n  ): Promise<unknown> {\n    // Context 생성 (기존 메소드 재사용)\n    const context = await this.createContext(config, request, reply);\n\n    return this.asyncLocalStorage.run({ context }, async () => {\n      // args 생성: Context 파라미터는 주입, 나머지는 params에서 가져오기\n      const { ApiParamType } = await import(\"../types/types\");\n      let paramsIndex = 0;\n      const args = api.parameters.map((param) => {\n        if (ApiParamType.isContext(param.type)) {\n          return context;\n        }\n        return params[paramsIndex++];\n      });\n\n      // 모델 메서드 호출 (기존 메서드 재사용)\n      return this.invokeModelMethod(api, args, reply);\n    });\n  }\n\n  // WS 경로에서는 HTTP reply가 없으므로 reply를 optional로 받아 공통 호출 경로를 유지함\n  async invokeModelMethod(\n    api: ExtendedApi,\n    args: unknown[],\n    reply?: FastifyReply,\n  ): Promise<unknown> {\n    const model = this.syncer.models[api.modelName];\n    const result = await (model as any)[api.methodName].apply(model, args);\n    reply?.type(api.options.contentType ?? \"application/json\");\n\n    return result;\n  }\n\n  async createContext(\n    config: SonamuFastifyConfig,\n    request: FastifyRequest,\n    reply: FastifyReply,\n  ): Promise<Context> {\n    // createSSEFactory 함수에 미리 request의 socket과 reply를 바인딩.\n    const { createSSEFactory } = await import(\"../stream/sse\");\n    const createSSE = (<T extends ZodObject>(\n      _request: FastifyRequest,\n      _reply: FastifyReply,\n      _events: T,\n    ) => createSSEFactory(_request.socket, _reply, _events)).bind(null, request, reply);\n\n    // locale 감지\n    const locale =\n      this.detectLocale(request.headers[\"accept-language\"], this.config.i18n.supportedLocales) ??\n      this.config.i18n.defaultLocale;\n\n    // auth context 추가\n    const headers = convertFastifyHeadersToStandard(request.headers);\n    const session = (await this._auth?.api.getSession({ headers })) ?? null;\n\n    const context: Context = await Promise.resolve(\n      config.contextProvider(\n        {\n          transport: \"http\",\n          request,\n          reply,\n          headers: request.headers,\n          createSSE,\n          naiteStore: new Map(),\n          locale,\n          // auth\n          user: session?.user ?? null,\n          session: session?.session ?? null,\n        },\n        request,\n        reply,\n      ),\n    );\n    return context;\n  }\n\n  // session/locale/store 같은 공통 state는 HTTP context와 공유하되, reply/createSSE 같은 HTTP 전용 helper는 노출하지 않음\n  // 사용자가 websocketContextProvider를 주면 그대로 위임하고, 없으면 기존 contextProvider를 reply/SSE stub과 함께 재활용함\n  async createWebSocketContext(\n    config: SonamuFastifyConfig,\n    request: FastifyRequest,\n    ws: WebSocketContext[\"ws\"],\n  ): Promise<WebSocketContext> {\n    const locale =\n      this.detectLocale(request.headers[\"accept-language\"], this.config.i18n.supportedLocales) ??\n      this.config.i18n.defaultLocale;\n\n    const headers = convertFastifyHeadersToStandard(request.headers);\n    const session = (await this._auth?.api.getSession({ headers })) ?? null;\n\n    const defaultContext = {\n      transport: \"ws\" as const,\n      request,\n      headers: request.headers,\n      ws,\n      naiteStore: new Map(),\n      locale,\n      user: session?.user ?? null,\n      session: session?.session ?? null,\n    };\n\n    if (config.websocketContextProvider) {\n      return {\n        ...(await Promise.resolve(config.websocketContextProvider(defaultContext, request))),\n      };\n    }\n\n    // reply/createSSE에 의존하는 contextProvider가 있으면 즉시 에러를 던져 transport misuse를 빨리 드러냄\n    const replyStub = createWebSocketReplyStub();\n    const createSSE = <T extends ZodObject>(_events: T) => {\n      throw new Error(\n        \"createSSE is not available in websocket context. Define websocketContextProvider if your context setup depends on SSE helpers.\",\n      );\n    };\n    const httpLikeContext = await Promise.resolve(\n      config.contextProvider(\n        {\n          transport: \"http\",\n          request,\n          reply: replyStub,\n          headers: request.headers,\n          createSSE,\n          naiteStore: defaultContext.naiteStore,\n          locale,\n          user: defaultContext.user,\n          session: defaultContext.session,\n        },\n        request,\n        replyStub,\n      ),\n    );\n\n    const {\n      transport: _transport,\n      reply: _reply,\n      createSSE: _createSSE,\n      bufferedFiles: _bufferedFiles,\n      uploadedFiles: _uploadedFiles,\n      ...rest\n    } = httpLikeContext;\n\n    return {\n      ...rest,\n      transport: \"ws\",\n      request,\n      headers: request.headers,\n      ws,\n    };\n  }\n\n  /**\n   * Accept-Language 헤더에서 지원하는 locale을 찾습니다.\n   * @example \"ko-KR,ko;q=0.9,en;q=0.8\" → \"ko\"\n   */\n  private detectLocale(\n    acceptLanguage: string | undefined,\n    supported: string[],\n  ): string | undefined {\n    if (!acceptLanguage) return undefined;\n\n    // Accept-Language: ko-KR,ko;q=0.9,en;q=0.8\n    const langs = acceptLanguage.split(\",\").map((lang) => {\n      const [code] = lang.split(\";\");\n      return code.trim().split(\"-\")[0]; // ko-KR → ko\n    });\n\n    return langs.find((lang) => supported.includes(lang));\n  }\n\n  async startWatcher(): Promise<void> {\n    // watcher 모듈은 file-patterns → Sonamu 순환을 피하기 위해 dynamic import 합니다.\n    const { setupWatcher } = await import(\"../syncer/watcher\");\n    this.watcher = await setupWatcher((fileEvents) => this.runHmrSyncCycle(fileEvents));\n  }\n\n  /**\n   * Watcher가 100ms batch로 모은 fileEvents 하나에 대해 한 번의 HMR/sync 사이클을 돕니다.\n   * batch 큐잉 덕에 한 시점에 하나만 실행됨이 보장됩니다 (event-batcher가 직렬화).\n   */\n  private async runHmrSyncCycle(fileEvents: Map<AbsolutePath, \"change\" | \"add\">): Promise<void> {\n    const startedAt = Date.now();\n\n    for (const [filePath, event] of fileEvents) {\n      const relativePath = path.relative(this.appRootPath, filePath);\n      console.log(chalk.bold(`Detected(${event}): ${chalk.blue(relativePath)}`));\n    }\n\n    // 본체: 변경 흡수 + 체크섬 갱신.\n    await this.syncer.hmrAndSync(fileEvents);\n    await this.syncer.renewChecksums();\n\n    const totalTime = Date.now() - startedAt;\n    const msg = `HMR Done! ${chalk.bold.white(`${totalTime}ms`)}`;\n    console.log(chalk.black.bgGreen(centerText(msg)));\n  }\n\n  /*\n     A function that automatically handles init and destroy when using Sonamu via scripts.\n  */\n  async runScript(fn: () => Promise<void>) {\n    await this.init(true, false, undefined, false);\n    try {\n      await fn();\n    } finally {\n      await this.destroy();\n    }\n  }\n\n  private async registerPlugins(server: FastifyInstance, plugins: SonamuServerOptions[\"plugins\"]) {\n    if (!plugins) {\n      return;\n    }\n\n    // compress 플러그인은 다른 플러그인보다 먼저 등록되어야 합니다.\n    if (plugins.compress) {\n      const compressPlugin = (await import(\"@fastify/compress\")).default;\n      const defaultOptions = {\n        threshold: 1024,\n        encodings: [\"br\", \"gzip\", \"deflate\"] as (\"br\" | \"gzip\" | \"deflate\")[],\n      };\n\n      if (plugins.compress === true) {\n        server.register(compressPlugin, defaultOptions);\n      } else {\n        server.register(compressPlugin, {\n          ...defaultOptions,\n          ...plugins.compress,\n        });\n      }\n    }\n\n    const pluginsModules = {\n      cors: \"@fastify/cors\",\n      formbody: \"@fastify/formbody\",\n      multipart: \"@fastify/multipart\",\n      qs: \"fastify-qs\",\n      sse: \"fastify-sse-v2\",\n      static: \"@fastify/static\",\n    } as const;\n\n    const registerPlugin = async <K extends keyof NonNullable<typeof plugins>>(\n      key: K,\n      pluginName: string,\n    ) => {\n      const option = plugins[key];\n      if (!option) return;\n\n      if (option === true) {\n        server.register((await import(pluginName)).default);\n      } else {\n        server.register((await import(pluginName)).default, option);\n      }\n    };\n\n    for (const [key, pluginName] of Object.entries(pluginsModules)) {\n      await registerPlugin(key as keyof typeof plugins, pluginName);\n    }\n\n    if (plugins.ws) {\n      await this.ensureWebSocketPlugin(server);\n    }\n\n    if (plugins.custom) {\n      plugins.custom(server);\n    }\n  }\n\n  // @fastify/websocket은 plugins.ws가 설정된 경우에만 등록하고, 같은 server에 중복 등록되지 않도록 WeakSet으로 기록함\n  private async ensureWebSocketPlugin(\n    server: FastifyInstance<Server, IncomingMessage, ServerResponse>,\n  ): Promise<void> {\n    if (this.websocketPluginServers.has(server)) {\n      return;\n    }\n\n    const pluginOption = this.config.server.plugins?.ws;\n    if (!pluginOption) {\n      return;\n    }\n\n    const websocketPlugin = (await import(\"@fastify/websocket\")).default;\n    const resolvedPluginOptions = resolveWebSocketPluginOptions({\n      rawPluginOption: pluginOption,\n      apis: this.syncer.apis,\n    });\n    if (resolvedPluginOptions) {\n      await server.register(websocketPlugin, resolvedPluginOptions);\n    } else {\n      await server.register(websocketPlugin);\n    }\n\n    this.websocketPluginServers.add(server);\n    this.warnOnPotentialWebSocketTimeoutConflicts(server);\n  }\n\n  // heartbeat interval이 Fastify keepAliveTimeout 이상이면 인프라가 먼저 idle 연결을 끊을 수 있어 경고만 남기고 넘어감\n  private warnOnPotentialWebSocketTimeoutConflicts(\n    server: FastifyInstance<Server, IncomingMessage, ServerResponse>,\n  ): void {\n    const heartbeats = this.syncer.apis\n      .map((api) => api.websocketOptions?.heartbeat ?? 30000)\n      .filter((heartbeat) => heartbeat > 0);\n\n    if (heartbeats.length === 0) {\n      return;\n    }\n\n    const keepAliveTimeout = this.config.server.fastify?.keepAliveTimeout;\n    if (!keepAliveTimeout || keepAliveTimeout <= 0) {\n      return;\n    }\n\n    const largestHeartbeat = Math.max(...heartbeats);\n    if (largestHeartbeat >= keepAliveTimeout) {\n      server.log.warn(\n        {\n          keepAliveTimeout,\n          largestHeartbeat,\n        },\n        \"WebSocket heartbeat is greater than or equal to keepAliveTimeout; align infrastructure idle timeouts to avoid unexpected disconnects.\",\n      );\n    }\n  }\n\n  /**\n   * better-auth 라우트를 등록합니다.\n   * /api/auth/* 경로로 인증 API가 자동 등록됩니다.\n   */\n  private async registerBetterAuth(\n    server: FastifyInstance,\n    options: NonNullable<SonamuServerOptions[\"auth\"]>,\n  ) {\n    if (!options) return;\n\n    const basePath = options.basePath ?? \"/api/auth\";\n\n    // better-auth 라우트 등록\n    server.route({\n      method: [\"GET\", \"POST\"],\n      url: `${basePath}/*`,\n      handler: async (request, reply) => {\n        const url = new URL(request.url, `http://${request.headers.host}`);\n        const headers = convertFastifyHeadersToStandard(request.headers);\n\n        // IP 헤더 fallback: 프록시가 표준 IP 헤더를 주입하지 않는 환경에서도\n        // better-auth/infra의 getClientIpFromRequest()가 IP를 인식할 수 있도록\n        // Fastify가 resolve한 request.ip를 x-real-ip로 주입한다.\n        const IP_HEADERS = [\n          \"cf-connecting-ip\",\n          \"x-forwarded-for\",\n          \"x-real-ip\",\n          \"x-vercel-forwarded-for\",\n        ];\n        if (request.ip && !IP_HEADERS.some((h) => headers.has(h))) {\n          headers.set(\"x-real-ip\", request.ip);\n        }\n\n        const req = new Request(url.toString(), {\n          method: request.method,\n          headers,\n          ...(request.body ? { body: JSON.stringify(request.body) } : {}),\n        });\n\n        const response = await this.auth.handler(req);\n\n        reply.status(response.status);\n        response.headers.forEach((value: string, key: string) => {\n          reply.header(key, value);\n        });\n        return reply.send(response.body ? await response.text() : null);\n      },\n    });\n  }\n\n  private async printStartupSummary() {\n    const chalk = (await import(\"chalk\")).default;\n    const env = process.env.NODE_ENV ?? \"development\";\n    const activePreset = env === \"production\" ? \"production_master\" : \"development_master\";\n\n    const dim = (msg: string) => console.log(chalk.dim(`✓ ${msg}`));\n    const green = (msg: string) => console.log(chalk.green(`✓ ${msg}`));\n\n    dim(`Config loaded${formatTime(this._configElapsed)}`);\n\n    // DB preset 목록\n    green(\"DB\");\n    const { isLocal } = await import(\"../utils/controller\");\n    const presetNames = Object.keys(this.dbConfig) as (keyof SonamuDBConfig)[];\n    const maxLen = Math.max(...presetNames.map((n) => n.length));\n    for (const name of presetNames) {\n      const conn = this.dbConfig[name].connection as\n        | { host?: string; port?: number; database?: string }\n        | undefined;\n      const host = conn?.host ?? \"localhost\";\n      const addr = `@ ${host}:${conn?.port ?? 5432}/${conn?.database ?? this.config.database.name}`;\n      const padded = name.padEnd(maxLen);\n      const remoteTag = isLocal() && !isLocalHost(host) ? chalk.yellow(` \\u26a0 remote`) : \"\";\n\n      if (name === activePreset) {\n        console.log(chalk.green(`  \\u25b8 ${padded} ${addr}`) + remoteTag);\n      } else {\n        console.log(chalk.dim(`    ${padded} ${addr}`) + remoteTag);\n      }\n    }\n\n    if (this.config.server.auth) {\n      const basePath = this.config.server.auth.basePath ?? \"/api/auth\";\n      dim(`Auth: better-auth at ${basePath}/*`);\n    }\n    if (this.config.api.timezone) {\n      dim(`Timezone: ${this.config.api.timezone}`);\n    }\n    green(`Sonamu ready${formatTime(this._initElapsed)}`);\n  }\n\n  private async initializeCache(config: CacheConfig | undefined, forTesting: boolean) {\n    const { setCacheManagerRef } = await import(\"../cache/decorator\");\n\n    // 테스트 환경에서 메모리 드라이버 자동 사용\n    if (forTesting) {\n      const { createTestCacheManager } = await import(\"../cache/cache-manager\");\n      this._cache = createTestCacheManager();\n      setCacheManagerRef(this._cache);\n      return;\n    }\n\n    // 설정이 없으면 캐시 비활성화\n    if (!config) {\n      setCacheManagerRef(null);\n      return;\n    }\n\n    // 설정에 따라 CacheManager 생성\n    const { createCacheManager } = await import(\"../cache/cache-manager\");\n    this._cache = createCacheManager(config);\n    setCacheManagerRef(this._cache);\n  }\n\n  private async initializeWorkflows(options: SonamuTaskOptions | undefined) {\n    const { WorkflowManager } = await import(\"../tasks/workflow-manager\");\n    // NOTE: @sonamu-kit/tasks 안에선 knex config를 수정하기 때문에 connection이 아닌 config 째로 보냅니다.\n    this._workflows = new WorkflowManager(DB.getDBConfig(\"w\"));\n    if (!options) {\n      return;\n    }\n\n    const enableWorker = options.enableWorker ?? isDaemonServer();\n    const defaultWorkerOptions = {\n      concurrency: os.cpus().length - 1,\n      usePubSub: true,\n      listenDelay: 500,\n    };\n\n    if (enableWorker) {\n      this.workflows.setupWorker({\n        ...defaultWorkerOptions,\n        ...options.workerOptions,\n      });\n    }\n  }\n\n  private async boot(server: FastifyInstance, options: SonamuServerOptions) {\n    const port = options.listen?.port ?? 3000;\n    const host = options.listen?.host ?? \"localhost\";\n\n    server.addHook(\"onClose\", async () => {\n      await options.lifecycle?.onShutdown?.(server);\n      await this.workflows.destroy();\n      await this.destroy();\n    });\n\n    const shutdown = async () => {\n      try {\n        await server.close();\n        process.exit(0);\n      } catch (err) {\n        console.error(\"Error during shutdown:\", err);\n        process.exit(1);\n      }\n    };\n\n    process.on(\"SIGINT\", shutdown);\n    process.on(\"SIGTERM\", shutdown);\n\n    if (options.lifecycle?.onError) {\n      server.setErrorHandler(options.lifecycle?.onError);\n    }\n\n    server\n      .listen({ port, host })\n      .then(async () => {\n        await this.workflows.startWorker();\n        await options.lifecycle?.onStart?.(server);\n      })\n      .catch(async (err) => {\n        const chalk = (await import(\"chalk\")).default;\n        console.error(chalk.red(\"Failed to start server:\", err));\n        await shutdown();\n      });\n  }\n\n  async destroy(): Promise<void> {\n    const { BaseModel } = await import(\"../database/base-model\");\n    // 먼저 처리해야함.\n    await BaseModel.destroy();\n    // 프로세스 종료 시 살아있는 WS 연결을 먼저 정리해 이후 다른 리소스 해제 과정에서 잔여 callback이 튀지 않게 함\n    await Promise.allSettled([\n      this._websocketRuntime?.shutdown() ?? Promise.resolve(),\n      this._workflows?.destroy() ?? Promise.resolve(),\n      this._cache?.disconnect() ?? Promise.resolve(),\n      this._devVitestManager?.shutdown() ?? Promise.resolve(),\n      this.watcher?.close() ?? Promise.resolve(),\n      logtapeDispose(),\n    ]);\n    this._websocketRuntime = null;\n  }\n}\n\nexport const Sonamu = new SonamuClass();\n\n/**\n * stream 모드에서 키 생성 함수가 지정되지 않았을 때 사용하는 기본 함수입니다.\n */\nfunction defaultKeyGenerator(file: { filename: string; mimetype: string }): string {\n  const ext = mime.extension(file.mimetype) || \"bin\";\n  const timestamp = Date.now();\n  const random = Math.random().toString(36).slice(2, 8);\n  return `uploads/${timestamp}-${random}.${ext}`;\n}\n\nfunction formatTime(ms: number): string {\n  const formatted = ms >= 1000 ? `${(ms / 1000).toFixed(2)}s` : `${Math.round(ms)}ms`;\n  return ` (${formatted})`;\n}\n\nconst LOCAL_HOSTS = new Set([\"localhost\", \"127.0.0.1\", \"0.0.0.0\", \"::1\"]);\nfunction isLocalHost(host: string): boolean {\n  return LOCAL_HOSTS.has(host);\n}\n\n// `.every()`가 첫 guard 이후 순회를 멈추는 문제가 있어 `for...of`로 모든 guard를 순서대로 실행하도록 고정함\nfunction runGuards({\n  guards,\n  config,\n  request,\n  api,\n}: {\n  guards: ExtendedApi[\"options\"][\"guards\"] | undefined;\n  config: Pick<SonamuFastifyConfig, \"guardHandler\">;\n  request: FastifyRequest;\n  api: ExtendedApi;\n}): void {\n  for (const guard of guards ?? []) {\n    config.guardHandler(guard, request, api);\n  }\n}\n"],"mappings":"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"}
|