sonamu 0.9.5 → 0.9.7

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (159) hide show
  1. package/dist/api/config.d.ts +13 -2
  2. package/dist/api/config.d.ts.map +1 -1
  3. package/dist/api/config.js +1 -1
  4. package/dist/api/context.d.ts +17 -7
  5. package/dist/api/context.d.ts.map +1 -1
  6. package/dist/api/context.js +1 -1
  7. package/dist/api/decorators.d.ts +18 -0
  8. package/dist/api/decorators.d.ts.map +1 -1
  9. package/dist/api/decorators.js +55 -4
  10. package/dist/api/index.js +8 -3
  11. package/dist/api/sonamu.d.ts +24 -9
  12. package/dist/api/sonamu.d.ts.map +1 -1
  13. package/dist/api/sonamu.js +365 -79
  14. package/dist/api/websocket-helpers.d.ts +24 -0
  15. package/dist/api/websocket-helpers.d.ts.map +1 -0
  16. package/dist/api/websocket-helpers.js +77 -0
  17. package/dist/bin/cli.js +12 -4
  18. package/dist/dict/sonamu-dictionary.js +5 -5
  19. package/dist/entity/entity-manager.js +1 -1
  20. package/dist/entity/entity.js +3 -3
  21. package/dist/index.d.ts +6 -0
  22. package/dist/index.d.ts.map +1 -1
  23. package/dist/index.js +16 -4
  24. package/dist/migration/code-generation.js +7 -7
  25. package/dist/stream/index.d.ts +6 -0
  26. package/dist/stream/index.d.ts.map +1 -1
  27. package/dist/stream/index.js +13 -2
  28. package/dist/stream/ws-audience-resolver.d.ts +15 -0
  29. package/dist/stream/ws-audience-resolver.d.ts.map +1 -0
  30. package/dist/stream/ws-audience-resolver.js +31 -0
  31. package/dist/stream/ws-audience.d.ts +28 -0
  32. package/dist/stream/ws-audience.d.ts.map +1 -0
  33. package/dist/stream/ws-audience.js +46 -0
  34. package/dist/stream/ws-cluster-bus.d.ts +23 -0
  35. package/dist/stream/ws-cluster-bus.d.ts.map +1 -0
  36. package/dist/stream/ws-cluster-bus.js +18 -0
  37. package/dist/stream/ws-core.d.ts +15 -0
  38. package/dist/stream/ws-core.d.ts.map +1 -0
  39. package/dist/stream/ws-core.js +1 -0
  40. package/dist/stream/ws-delivery.d.ts +24 -0
  41. package/dist/stream/ws-delivery.d.ts.map +1 -0
  42. package/dist/stream/ws-delivery.js +103 -0
  43. package/dist/stream/ws-local-connection-store.d.ts +10 -0
  44. package/dist/stream/ws-local-connection-store.d.ts.map +1 -0
  45. package/dist/stream/ws-local-connection-store.js +44 -0
  46. package/dist/stream/ws-presence-store.d.ts +61 -0
  47. package/dist/stream/ws-presence-store.d.ts.map +1 -0
  48. package/dist/stream/ws-presence-store.js +236 -0
  49. package/dist/stream/ws-registry.d.ts +42 -0
  50. package/dist/stream/ws-registry.d.ts.map +1 -0
  51. package/dist/stream/ws-registry.js +108 -0
  52. package/dist/stream/ws.d.ts +52 -0
  53. package/dist/stream/ws.d.ts.map +1 -0
  54. package/dist/stream/ws.js +397 -0
  55. package/dist/syncer/api-parser.d.ts.map +1 -1
  56. package/dist/syncer/api-parser.js +72 -2
  57. package/dist/syncer/checksum.d.ts.map +1 -1
  58. package/dist/syncer/checksum.js +13 -12
  59. package/dist/syncer/code-generator.d.ts.map +1 -1
  60. package/dist/syncer/code-generator.js +7 -4
  61. package/dist/syncer/event-batcher.d.ts +27 -0
  62. package/dist/syncer/event-batcher.d.ts.map +1 -0
  63. package/dist/syncer/event-batcher.js +69 -0
  64. package/dist/syncer/file-patterns.d.ts +48 -26
  65. package/dist/syncer/file-patterns.d.ts.map +1 -1
  66. package/dist/syncer/file-patterns.js +71 -23
  67. package/dist/syncer/file-tracking.d.ts +13 -0
  68. package/dist/syncer/file-tracking.d.ts.map +1 -0
  69. package/dist/syncer/file-tracking.js +33 -0
  70. package/dist/syncer/index.js +2 -2
  71. package/dist/syncer/module-loader.d.ts +2 -11
  72. package/dist/syncer/module-loader.d.ts.map +1 -1
  73. package/dist/syncer/module-loader.js +3 -3
  74. package/dist/syncer/syncer-actions.d.ts +39 -6
  75. package/dist/syncer/syncer-actions.d.ts.map +1 -1
  76. package/dist/syncer/syncer-actions.js +125 -10
  77. package/dist/syncer/syncer.d.ts +33 -19
  78. package/dist/syncer/syncer.d.ts.map +1 -1
  79. package/dist/syncer/syncer.js +168 -168
  80. package/dist/syncer/watcher.d.ts +8 -0
  81. package/dist/syncer/watcher.d.ts.map +1 -0
  82. package/dist/syncer/watcher.js +105 -0
  83. package/dist/tasks/workflow-manager.d.ts.map +1 -1
  84. package/dist/tasks/workflow-manager.js +2 -1
  85. package/dist/template/implementations/services.template.d.ts.map +1 -1
  86. package/dist/template/implementations/services.template.js +36 -1
  87. package/dist/testing/bootstrap.d.ts.map +1 -1
  88. package/dist/testing/bootstrap.js +8 -1
  89. package/dist/testing/fixture-manager.js +1 -1
  90. package/dist/types/types.d.ts +2 -1
  91. package/dist/types/types.d.ts.map +1 -1
  92. package/dist/types/types.js +2 -2
  93. package/dist/ui/api.js +1 -1
  94. package/dist/ui/cdd-service.js +1 -1
  95. package/dist/ui-web/assets/{index-DzZ7vBk4.js → index-BmThfg-s.js} +37 -37
  96. package/dist/ui-web/index.html +1 -1
  97. package/dist/utils/async-utils.d.ts +27 -3
  98. package/dist/utils/async-utils.d.ts.map +1 -1
  99. package/dist/utils/async-utils.js +56 -6
  100. package/dist/utils/formatter.d.ts +7 -1
  101. package/dist/utils/formatter.d.ts.map +1 -1
  102. package/dist/utils/formatter.js +93 -59
  103. package/dist/utils/fs-utils.d.ts +2 -0
  104. package/dist/utils/fs-utils.d.ts.map +1 -1
  105. package/dist/utils/fs-utils.js +10 -2
  106. package/dist/utils/process-utils.d.ts +10 -4
  107. package/dist/utils/process-utils.d.ts.map +1 -1
  108. package/dist/utils/process-utils.js +20 -7
  109. package/dist/utils/utils.d.ts +1 -0
  110. package/dist/utils/utils.d.ts.map +1 -1
  111. package/dist/utils/utils.js +2 -2
  112. package/package.json +3 -1
  113. package/src/api/__tests__/sonamu.websocket.test.ts +64 -0
  114. package/src/api/__tests__/websocket-context.types.test.ts +58 -0
  115. package/src/api/config.ts +28 -2
  116. package/src/api/context.ts +21 -7
  117. package/src/api/decorators.ts +103 -3
  118. package/src/api/sonamu.ts +529 -127
  119. package/src/api/websocket-helpers.ts +122 -0
  120. package/src/bin/cli.ts +10 -2
  121. package/src/dict/sonamu-dictionary.ts +2 -2
  122. package/src/entity/entity.ts +1 -1
  123. package/src/index.ts +6 -0
  124. package/src/migration/code-generation.ts +5 -5
  125. package/src/shared/app.shared.ts.txt +254 -1
  126. package/src/shared/web.shared.ts.txt +282 -1
  127. package/src/stream/__tests__/ws-contracts.test.ts +381 -0
  128. package/src/stream/__tests__/ws.test.ts +449 -0
  129. package/src/stream/index.ts +6 -0
  130. package/src/stream/ws-audience-resolver.ts +35 -0
  131. package/src/stream/ws-audience.ts +62 -0
  132. package/src/stream/ws-cluster-bus.ts +32 -0
  133. package/src/stream/ws-core.ts +16 -0
  134. package/src/stream/ws-delivery.ts +138 -0
  135. package/src/stream/ws-local-connection-store.ts +44 -0
  136. package/src/stream/ws-presence-store.ts +326 -0
  137. package/src/stream/ws-registry.ts +138 -0
  138. package/src/stream/ws.ts +591 -0
  139. package/src/syncer/__tests__/api-parser.websocket-type-ref.test.ts +78 -0
  140. package/src/syncer/api-parser.ts +112 -1
  141. package/src/syncer/checksum.ts +23 -29
  142. package/src/syncer/code-generator.ts +4 -1
  143. package/src/syncer/event-batcher.ts +72 -0
  144. package/src/syncer/file-patterns.ts +98 -30
  145. package/src/syncer/file-tracking.ts +27 -0
  146. package/src/syncer/module-loader.ts +5 -12
  147. package/src/syncer/syncer-actions.ts +179 -17
  148. package/src/syncer/syncer.ts +250 -287
  149. package/src/syncer/watcher.ts +128 -0
  150. package/src/tasks/workflow-manager.ts +1 -0
  151. package/src/template/__tests__/services.template.websocket.test.ts +79 -0
  152. package/src/template/implementations/services.template.ts +69 -0
  153. package/src/testing/bootstrap.ts +8 -1
  154. package/src/types/types.ts +20 -2
  155. package/src/utils/async-utils.ts +71 -4
  156. package/src/utils/formatter.ts +111 -74
  157. package/src/utils/fs-utils.ts +9 -0
  158. package/src/utils/process-utils.ts +21 -4
  159. package/src/utils/utils.ts +1 -1
@@ -3,6 +3,7 @@ import { type FastifyCorsOptions } from "@fastify/cors";
3
3
  import { type FastifyFormbodyOptions } from "@fastify/formbody";
4
4
  import { type FastifyMultipartOptions } from "@fastify/multipart";
5
5
  import { type FastifyStaticOptions } from "@fastify/static";
6
+ import { type WebsocketPluginOptions } from "@fastify/websocket";
6
7
  import { type BetterAuthOptions } from "better-auth";
7
8
  import { type FastifyInstance, type FastifyReply, type FastifyRequest, type FastifyServerOptions } from "fastify";
8
9
  import { type QsPluginOptions } from "fastify-qs";
@@ -12,9 +13,10 @@ import { type CacheConfig } from "../cache/types";
12
13
  import { type SonamuDBConfig } from "../database/db";
13
14
  import { type SonamuLoggingOptions } from "../logger/configure";
14
15
  import { type StorageConfig } from "../storage/types";
16
+ import { type WebSocketRuntimeOptions } from "../stream/ws";
15
17
  import { type WorkflowOptions } from "../tasks/workflow-manager";
16
18
  import { type Executable, type SonamuFastifyConfig } from "../types/types";
17
- import { type Context } from "./context";
19
+ import { type Context, type WebSocketContext } from "./context";
18
20
  export type DatabaseConfig = Omit<Knex.Config, "connection"> & {
19
21
  connection?: Knex.PgConnectionConfig;
20
22
  };
@@ -152,6 +154,7 @@ export type SonamuServerOptions = {
152
154
  qs?: boolean | QsPluginOptions;
153
155
  sse?: boolean | SsePluginOptions;
154
156
  static?: boolean | FastifyStaticOptions;
157
+ ws?: boolean | WebsocketPluginOptions;
155
158
  custom?: (server: FastifyInstance) => void;
156
159
  };
157
160
  /**
@@ -166,6 +169,13 @@ export type SonamuServerOptions = {
166
169
  }>;
167
170
  };
168
171
  };
172
+ /**
173
+ * WebSocket runtime 설정.
174
+ *
175
+ * 단일 인스턴스에서는 기본값으로 충분하며, 멀티 인스턴스/대규모 환경에서는
176
+ * presence store와 cluster bus를 여기서 주입합니다.
177
+ */
178
+ websocket?: WebSocketRuntimeOptions;
169
179
  apiConfig: SonamuFastifyConfig;
170
180
  /**
171
181
  * Storage 드라이버 설정.
@@ -213,7 +223,8 @@ export type SonamuServerOptions = {
213
223
  export type SonamuTaskOptions = {
214
224
  enableWorker?: boolean;
215
225
  workerOptions?: WorkflowOptions;
216
- contextProvider: (defaultContext: Pick<Context, "reply" | "request" | "headers" | "createSSE" | "naiteStore" | "locale" | "user" | "session">) => Context | Promise<Context>;
226
+ contextProvider: (defaultContext: Pick<Context, "transport" | "reply" | "request" | "headers" | "createSSE" | "naiteStore" | "locale" | "user" | "session">) => Context | Promise<Context>;
227
+ websocketContextProvider?: (defaultContext: Pick<WebSocketContext, "transport" | "request" | "headers" | "ws" | "naiteStore" | "locale" | "user" | "session">, request: FastifyRequest) => WebSocketContext | Promise<WebSocketContext>;
217
228
  };
218
229
  export type SonamuSSROptions = {
219
230
  /**
@@ -1 +1 @@
1
- {"version":3,"file":"config.d.ts","sourceRoot":"","sources":["../../src/api/config.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"config.d.ts","sourceRoot":"","sources":["../../src/api/config.ts"],"names":[],"mappings":"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"}
@@ -35,4 +35,4 @@ var init_config = __esmMin((() => {}));
35
35
  //#endregion
36
36
  init_config();
37
37
  export { defineConfig, init_config, loadConfig };
38
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"config.js","names":[],"sources":["../../src/api/config.ts"],"sourcesContent":["import { type FastifyCompressOptions } from \"@fastify/compress\";\nimport { type FastifyCorsOptions } from \"@fastify/cors\";\nimport { type FastifyFormbodyOptions } from \"@fastify/formbody\";\nimport { type FastifyMultipartOptions } from \"@fastify/multipart\";\nimport { type FastifyStaticOptions } from \"@fastify/static\";\nimport { type BetterAuthOptions } from \"better-auth\";\nimport {\n  type FastifyInstance,\n  type FastifyReply,\n  type FastifyRequest,\n  type FastifyServerOptions,\n} from \"fastify\";\nimport { type QsPluginOptions } from \"fastify-qs\";\nimport { type SsePluginOptions } from \"fastify-sse-v2/lib/types\";\nimport { type Knex } from \"knex\";\n\nimport { type CacheConfig } from \"../cache/types\";\nimport { type SonamuDBConfig } from \"../database/db\";\nimport { type SonamuLoggingOptions } from \"../logger/configure\";\nimport { type StorageConfig } from \"../storage/types\";\nimport { type WorkflowOptions } from \"../tasks/workflow-manager\";\nimport { type Executable, type SonamuFastifyConfig } from \"../types/types\";\nimport { type Context } from \"./context\";\n\nexport type DatabaseConfig = Omit<Knex.Config, \"connection\"> & {\n  connection?: Knex.PgConnectionConfig;\n};\n\n/**\n * i18n 설정\n */\nexport type SonamuI18nOptions = {\n  /** 기본 locale (키 정의 기준 + 런타임 기본값) */\n  defaultLocale: string;\n  /** 지원하는 locale 목록 */\n  supportedLocales: string[];\n};\n\n/**\n * Dev 서버 내 Vitest 상주 인스턴스 설정\n */\nexport type SonamuDevRunnerConfig = {\n  /** DevRunner 활성화 여부 (기본: false) */\n  enabled: boolean;\n  /** 테스트 엔드포인트 경로 접두사 (기본: /__test__) */\n  routePrefix?: string;\n  /** vitest.config.ts 경로 (api-root 상대경로) */\n  vitestConfigPath?: string;\n};\n\n/**\n * 테스트 설정\n * vitest 병렬 테스팅을 sonamu.config.ts 한 곳에서 관리하기 위한 설정입니다.\n */\nexport type SonamuTestConfig = {\n  /** 병렬 테스팅 활성화 (기본: false) */\n  parallel?: boolean;\n  /** 병렬 실행 워커 수 (기본: 4) */\n  maxWorkers?: number;\n  /** Dev 서버 내 Vitest 상주 인스턴스 설정 */\n  devRunner?: SonamuDevRunnerConfig;\n};\n\nexport type SonamuConfig<TSinkId extends string = string, TFilterId extends string = string> = {\n  projectName?: string;\n\n  api: {\n    dir: string;\n    route: {\n      prefix: string;\n    };\n    timezone?: string;\n  };\n\n  /**\n   * i18n 설정\n   *\n   * @example\n   * ```typescript\n   * i18n: {\n   *   defaultLocale: 'ko',\n   *   supportedLocales: ['ko', 'en', 'ja'],\n   * }\n   * ```\n   */\n  i18n: SonamuI18nOptions;\n\n  sync: {\n    targets: string[]; // \"web\", \"app\" 등\n  };\n\n  database: {\n    // 데이터베이스(pg는 pg 모듈, pgnative는 pg-native 모듈의 설치가 필요합니다.)\n    database?: \"pg\" | \"pgnative\";\n    // 기본 데이터베이스 이름\n    name: string;\n    // 모든 환경에 적용될 기본 Knex 옵션\n    defaultOptions: DatabaseConfig;\n    // 환경별 설정\n    environments?: {\n      development?: DatabaseConfig;\n      development_slave?: DatabaseConfig;\n      production?: DatabaseConfig;\n      production_slave?: DatabaseConfig;\n      fixture?: DatabaseConfig;\n      test?: DatabaseConfig;\n    };\n  };\n\n  logging?: false | SonamuLoggingOptions<TSinkId, TFilterId>;\n  server: SonamuServerOptions;\n  tasks?: SonamuTaskOptions;\n\n  /**\n   * 테스트 설정 (병렬 테스팅 등)\n   *\n   * @example\n   * ```typescript\n   * test: {\n   *   parallel: true,\n   *   maxWorkers: 4,\n   * }\n   * ```\n   */\n  test?: SonamuTestConfig;\n\n  /**\n   * 외부 에디터 설정 (CDD 등에서 파일 편집 시 사용)\n   * 미설정 시 \"vscode\"를 기본값으로 사용합니다.\n   *\n   * @example\n   * ```typescript\n   * externalEditor: \"cursor\"\n   * ```\n   */\n  externalEditor?: \"Visual Studio Code\" | \"Zed\" | \"Cursor\";\n\n  /**\n   * Slack 승인 설정 (Production 마이그레이션용)\n   *\n   * Production DB 마이그레이션 시 Slack을 통한 승인 프로세스를 활성화합니다.\n   * 설정이 없으면 기존 동작(바로 실행)으로 동작합니다.\n   *\n   * @example\n   * ```typescript\n   * slackConfirm: {\n   *   targets: [\"production\"],\n   *   botToken: process.env.SLACK_BOT_TOKEN ?? \"\",\n   *   channelId: process.env.SLACK_CHANNEL_ID ?? \"\",\n   * }\n   * ```\n   */\n  slackConfirm?: {\n    /** 승인이 필요한 DB 키 목록 (예: [\"production\"]) */\n    targets: (keyof SonamuDBConfig)[];\n    /** Slack Bot Token (xoxb-...) */\n    botToken: string;\n    /** Slack Channel ID (C...) */\n    channelId: string;\n  };\n};\n\nexport type SonamuServerOptions = {\n  // 프로젝트 외부에서 접근할 수 있는 URL. 기본값은 {server.listen.host}:{server.listen.port} 입니다.\n  baseUrl?: string;\n\n  fastify?: Omit<FastifyServerOptions, \"logger\">;\n\n  listen?: {\n    port: number;\n    host?: string;\n  };\n\n  plugins?: {\n    /** 응답 압축 플러그인 (@fastify/compress) */\n    compress?: boolean | FastifyCompressOptions;\n    cors?: boolean | FastifyCorsOptions;\n    formbody?: boolean | FastifyFormbodyOptions;\n    multipart?: boolean | FastifyMultipartOptions;\n    qs?: boolean | QsPluginOptions;\n    sse?: boolean | SsePluginOptions;\n    static?: boolean | FastifyStaticOptions;\n\n    custom?: (server: FastifyInstance) => void;\n  };\n\n  /**\n   * better-auth 인증 설정\n   * Sonamu 확장으로 additionalFields에 sonamuType 속성 추가 가능\n   * BetterAuth의 DBFieldAttribute를 확장할 수 없어서 우선 user만 적용\n   */\n  auth?: BetterAuthOptions & {\n    user?: {\n      additionalFields?: Record<\n        string,\n        NonNullable<NonNullable<BetterAuthOptions[\"user\"]>[\"additionalFields\"]>[string] & {\n          sonamuType?: string;\n        }\n      >;\n    };\n  };\n\n  apiConfig: SonamuFastifyConfig;\n\n  /**\n   * Storage 드라이버 설정.\n   * saveToDisk(diskName, key) 호출 시 드라이버를 명시적으로 지정합니다.\n   *\n   * @example\n   * ```typescript\n   * import { drivers } from \"sonamu/storage\";\n   *\n   * storage: {\n   *   drivers: {\n   *     fs: drivers.fs({ location: \"./uploads\", urlBuilder: { ... } }),\n   *     s3: drivers.s3({ bucket: \"my-bucket\", region: \"ap-northeast-2\", ... }),\n   *   }\n   * }\n   * ```\n   */\n  storage?: StorageConfig;\n\n  /**\n   * Cache 설정 (BentoCache 기반)\n   *\n   * @example\n   * ```typescript\n   * import { drivers, store } from \"sonamu/cache\";\n   *\n   * cache: {\n   *   default: 'main',\n   *   stores: {\n   *     main: store()\n   *       .useL1Layer(drivers.memory({ maxSize: '100mb' }))\n   *       .useL2Layer(drivers.redis({ connection: redisConnection }))\n   *       .useBus(drivers.redisBus({ connection: redisConnection }))\n   *   },\n   *   ttl: '1h',\n   * }\n   * ```\n   */\n  cache?: CacheConfig;\n\n  lifecycle?: {\n    onStart?: (server: FastifyInstance) => Promise<void> | void;\n    onShutdown?: (server: FastifyInstance) => Promise<void> | void;\n    onError?: (error: Error, request: FastifyRequest, reply: FastifyReply) => Promise<void> | void;\n  };\n};\n\nexport type SonamuTaskOptions = {\n  // worker를 사용할지 여부, 기본적으로 daemon 모드에서만 사용됨.\n  enableWorker?: boolean;\n  workerOptions?: WorkflowOptions;\n  contextProvider: (\n    defaultContext: Pick<\n      Context,\n      \"reply\" | \"request\" | \"headers\" | \"createSSE\" | \"naiteStore\" | \"locale\" | \"user\" | \"session\"\n    >,\n  ) => Context | Promise<Context>;\n};\n\nexport type SonamuSSROptions = {\n  /**\n   * Hydration 전략\n   * - 'none': 항상 새로 렌더링 (hydration mismatch 걱정 없음, 권장)\n   * - 'full': React hydration 시도 (서버 HTML 재사용, mismatch 주의)\n   * @default 'none'\n   */\n  hydration?: \"none\" | \"full\";\n};\n\n// NOTE(Haze, 251209): config에는 T, Promise<T>, () => T, () => Promise<T>가 모두 올 수 있어야 함.\nexport function defineConfig(config: Executable<SonamuConfig>): Promise<SonamuConfig> {\n  if (typeof config === \"function\") {\n    return Promise.resolve(config());\n  }\n\n  return Promise.resolve(config);\n}\n\n/**\n * sonamu.config.ts 파일을 로드합니다.\n * 이 설정 파일은 환경에 따라 다른 경로에 있을 수 있습니다.\n * dist를 빌드하는 환경이라면 dist 바로 아래에 있을 것이고(cli-wrapper.ts에서 빌드),\n * 그렇지 않은 환경이라면 프로젝트 루트에 있을 것입니다.\n *\n * 이 함수는 의도적으로 다른 의존성의 사용을 최대한 배제하였습니다.\n * 이는 실행 초기에 최대한 빠르게 설정을 읽어올 수 있도록 하기 위함입니다.\n * 따라서 경로 concat과 URL scheme 추가도 단순한 문자열 조작으로 처리하였습니다.\n *\n * @param rootPath\n * @returns\n */\nexport async function loadConfig(rootPath: string): Promise<SonamuConfig> {\n  const shouldLoadSourceConfig = process.env.HOT === \"yes\" || process.env.VITEST === \"true\";\n  const configPath = shouldLoadSourceConfig\n    ? `${rootPath}/src/sonamu.config.ts`\n    : `${rootPath}/dist/sonamu.config.js`;\n\n  if (shouldLoadSourceConfig) {\n    const { ensureTsLoaderRegistered } = await import(\"../bin/ts-loader-registration\");\n    await ensureTsLoaderRegistered(rootPath);\n  }\n\n  const { default: config } = await import(`file://${configPath}`);\n  return config;\n}\n"],"mappings":";;;AAiRA,SAAgB,aAAa,QAAyD;AACpF,KAAI,OAAO,WAAW,YAAY;AAChC,SAAO,QAAQ,QAAQ,QAAQ,CAAC;;AAGlC,QAAO,QAAQ,QAAQ,OAAO;;;;;;;;;;;;;;;AAgBhC,eAAsB,WAAW,UAAyC;CACxE,MAAM,yBAAyB,QAAQ,IAAI,QAAQ,SAAS,QAAQ,IAAI,WAAW;CACnF,MAAM,aAAa,yBACf,GAAG,SAAS,yBACZ,GAAG,SAAS;AAEhB,KAAI,wBAAwB;EAC1B,MAAM,EAAE,6BAA6B,MAAM,OAAO;AAClD,QAAM,yBAAyB,SAAS;;CAG1C,MAAM,EAAE,SAAS,WAAW,MAAM,OAAO,UAAU;AACnD,QAAO"}
38
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"config.js","names":[],"sources":["../../src/api/config.ts"],"sourcesContent":["import { type FastifyCompressOptions } from \"@fastify/compress\";\nimport { type FastifyCorsOptions } from \"@fastify/cors\";\nimport { type FastifyFormbodyOptions } from \"@fastify/formbody\";\nimport { type FastifyMultipartOptions } from \"@fastify/multipart\";\nimport { type FastifyStaticOptions } from \"@fastify/static\";\nimport { type WebsocketPluginOptions } from \"@fastify/websocket\";\nimport { type BetterAuthOptions } from \"better-auth\";\nimport {\n  type FastifyInstance,\n  type FastifyReply,\n  type FastifyRequest,\n  type FastifyServerOptions,\n} from \"fastify\";\nimport { type QsPluginOptions } from \"fastify-qs\";\nimport { type SsePluginOptions } from \"fastify-sse-v2/lib/types\";\nimport { type Knex } from \"knex\";\n\nimport { type CacheConfig } from \"../cache/types\";\nimport { type SonamuDBConfig } from \"../database/db\";\nimport { type SonamuLoggingOptions } from \"../logger/configure\";\nimport { type StorageConfig } from \"../storage/types\";\nimport { type WebSocketRuntimeOptions } from \"../stream/ws\";\nimport { type WorkflowOptions } from \"../tasks/workflow-manager\";\nimport { type Executable, type SonamuFastifyConfig } from \"../types/types\";\nimport { type Context, type WebSocketContext } from \"./context\";\n\nexport type DatabaseConfig = Omit<Knex.Config, \"connection\"> & {\n  connection?: Knex.PgConnectionConfig;\n};\n\n/**\n * i18n 설정\n */\nexport type SonamuI18nOptions = {\n  /** 기본 locale (키 정의 기준 + 런타임 기본값) */\n  defaultLocale: string;\n  /** 지원하는 locale 목록 */\n  supportedLocales: string[];\n};\n\n/**\n * Dev 서버 내 Vitest 상주 인스턴스 설정\n */\nexport type SonamuDevRunnerConfig = {\n  /** DevRunner 활성화 여부 (기본: false) */\n  enabled: boolean;\n  /** 테스트 엔드포인트 경로 접두사 (기본: /__test__) */\n  routePrefix?: string;\n  /** vitest.config.ts 경로 (api-root 상대경로) */\n  vitestConfigPath?: string;\n};\n\n/**\n * 테스트 설정\n * vitest 병렬 테스팅을 sonamu.config.ts 한 곳에서 관리하기 위한 설정입니다.\n */\nexport type SonamuTestConfig = {\n  /** 병렬 테스팅 활성화 (기본: false) */\n  parallel?: boolean;\n  /** 병렬 실행 워커 수 (기본: 4) */\n  maxWorkers?: number;\n  /** Dev 서버 내 Vitest 상주 인스턴스 설정 */\n  devRunner?: SonamuDevRunnerConfig;\n};\n\nexport type SonamuConfig<TSinkId extends string = string, TFilterId extends string = string> = {\n  projectName?: string;\n\n  api: {\n    dir: string;\n    route: {\n      prefix: string;\n    };\n    timezone?: string;\n  };\n\n  /**\n   * i18n 설정\n   *\n   * @example\n   * ```typescript\n   * i18n: {\n   *   defaultLocale: 'ko',\n   *   supportedLocales: ['ko', 'en', 'ja'],\n   * }\n   * ```\n   */\n  i18n: SonamuI18nOptions;\n\n  sync: {\n    targets: string[]; // \"web\", \"app\" 등\n  };\n\n  database: {\n    // 데이터베이스(pg는 pg 모듈, pgnative는 pg-native 모듈의 설치가 필요합니다.)\n    database?: \"pg\" | \"pgnative\";\n    // 기본 데이터베이스 이름\n    name: string;\n    // 모든 환경에 적용될 기본 Knex 옵션\n    defaultOptions: DatabaseConfig;\n    // 환경별 설정\n    environments?: {\n      development?: DatabaseConfig;\n      development_slave?: DatabaseConfig;\n      production?: DatabaseConfig;\n      production_slave?: DatabaseConfig;\n      fixture?: DatabaseConfig;\n      test?: DatabaseConfig;\n    };\n  };\n\n  logging?: false | SonamuLoggingOptions<TSinkId, TFilterId>;\n  server: SonamuServerOptions;\n  tasks?: SonamuTaskOptions;\n\n  /**\n   * 테스트 설정 (병렬 테스팅 등)\n   *\n   * @example\n   * ```typescript\n   * test: {\n   *   parallel: true,\n   *   maxWorkers: 4,\n   * }\n   * ```\n   */\n  test?: SonamuTestConfig;\n\n  /**\n   * 외부 에디터 설정 (CDD 등에서 파일 편집 시 사용)\n   * 미설정 시 \"vscode\"를 기본값으로 사용합니다.\n   *\n   * @example\n   * ```typescript\n   * externalEditor: \"cursor\"\n   * ```\n   */\n  externalEditor?: \"Visual Studio Code\" | \"Zed\" | \"Cursor\";\n\n  /**\n   * Slack 승인 설정 (Production 마이그레이션용)\n   *\n   * Production DB 마이그레이션 시 Slack을 통한 승인 프로세스를 활성화합니다.\n   * 설정이 없으면 기존 동작(바로 실행)으로 동작합니다.\n   *\n   * @example\n   * ```typescript\n   * slackConfirm: {\n   *   targets: [\"production\"],\n   *   botToken: process.env.SLACK_BOT_TOKEN ?? \"\",\n   *   channelId: process.env.SLACK_CHANNEL_ID ?? \"\",\n   * }\n   * ```\n   */\n  slackConfirm?: {\n    /** 승인이 필요한 DB 키 목록 (예: [\"production\"]) */\n    targets: (keyof SonamuDBConfig)[];\n    /** Slack Bot Token (xoxb-...) */\n    botToken: string;\n    /** Slack Channel ID (C...) */\n    channelId: string;\n  };\n};\n\nexport type SonamuServerOptions = {\n  // 프로젝트 외부에서 접근할 수 있는 URL. 기본값은 {server.listen.host}:{server.listen.port} 입니다.\n  baseUrl?: string;\n\n  fastify?: Omit<FastifyServerOptions, \"logger\">;\n\n  listen?: {\n    port: number;\n    host?: string;\n  };\n\n  plugins?: {\n    /** 응답 압축 플러그인 (@fastify/compress) */\n    compress?: boolean | FastifyCompressOptions;\n    cors?: boolean | FastifyCorsOptions;\n    formbody?: boolean | FastifyFormbodyOptions;\n    multipart?: boolean | FastifyMultipartOptions;\n    qs?: boolean | QsPluginOptions;\n    sse?: boolean | SsePluginOptions;\n    static?: boolean | FastifyStaticOptions;\n    ws?: boolean | WebsocketPluginOptions;\n\n    custom?: (server: FastifyInstance) => void;\n  };\n\n  /**\n   * better-auth 인증 설정\n   * Sonamu 확장으로 additionalFields에 sonamuType 속성 추가 가능\n   * BetterAuth의 DBFieldAttribute를 확장할 수 없어서 우선 user만 적용\n   */\n  auth?: BetterAuthOptions & {\n    user?: {\n      additionalFields?: Record<\n        string,\n        NonNullable<NonNullable<BetterAuthOptions[\"user\"]>[\"additionalFields\"]>[string] & {\n          sonamuType?: string;\n        }\n      >;\n    };\n  };\n\n  /**\n   * WebSocket runtime 설정.\n   *\n   * 단일 인스턴스에서는 기본값으로 충분하며, 멀티 인스턴스/대규모 환경에서는\n   * presence store와 cluster bus를 여기서 주입합니다.\n   */\n  websocket?: WebSocketRuntimeOptions;\n\n  apiConfig: SonamuFastifyConfig;\n\n  /**\n   * Storage 드라이버 설정.\n   * saveToDisk(diskName, key) 호출 시 드라이버를 명시적으로 지정합니다.\n   *\n   * @example\n   * ```typescript\n   * import { drivers } from \"sonamu/storage\";\n   *\n   * storage: {\n   *   drivers: {\n   *     fs: drivers.fs({ location: \"./uploads\", urlBuilder: { ... } }),\n   *     s3: drivers.s3({ bucket: \"my-bucket\", region: \"ap-northeast-2\", ... }),\n   *   }\n   * }\n   * ```\n   */\n  storage?: StorageConfig;\n\n  /**\n   * Cache 설정 (BentoCache 기반)\n   *\n   * @example\n   * ```typescript\n   * import { drivers, store } from \"sonamu/cache\";\n   *\n   * cache: {\n   *   default: 'main',\n   *   stores: {\n   *     main: store()\n   *       .useL1Layer(drivers.memory({ maxSize: '100mb' }))\n   *       .useL2Layer(drivers.redis({ connection: redisConnection }))\n   *       .useBus(drivers.redisBus({ connection: redisConnection }))\n   *   },\n   *   ttl: '1h',\n   * }\n   * ```\n   */\n  cache?: CacheConfig;\n\n  lifecycle?: {\n    onStart?: (server: FastifyInstance) => Promise<void> | void;\n    onShutdown?: (server: FastifyInstance) => Promise<void> | void;\n    onError?: (error: Error, request: FastifyRequest, reply: FastifyReply) => Promise<void> | void;\n  };\n};\n\nexport type SonamuTaskOptions = {\n  // worker를 사용할지 여부, 기본적으로 daemon 모드에서만 사용됨.\n  enableWorker?: boolean;\n  workerOptions?: WorkflowOptions;\n  contextProvider: (\n    defaultContext: Pick<\n      Context,\n      | \"transport\"\n      | \"reply\"\n      | \"request\"\n      | \"headers\"\n      | \"createSSE\"\n      | \"naiteStore\"\n      | \"locale\"\n      | \"user\"\n      | \"session\"\n    >,\n  ) => Context | Promise<Context>;\n  websocketContextProvider?: (\n    defaultContext: Pick<\n      WebSocketContext,\n      \"transport\" | \"request\" | \"headers\" | \"ws\" | \"naiteStore\" | \"locale\" | \"user\" | \"session\"\n    >,\n    request: FastifyRequest,\n  ) => WebSocketContext | Promise<WebSocketContext>;\n};\n\nexport type SonamuSSROptions = {\n  /**\n   * Hydration 전략\n   * - 'none': 항상 새로 렌더링 (hydration mismatch 걱정 없음, 권장)\n   * - 'full': React hydration 시도 (서버 HTML 재사용, mismatch 주의)\n   * @default 'none'\n   */\n  hydration?: \"none\" | \"full\";\n};\n\n// NOTE(Haze, 251209): config에는 T, Promise<T>, () => T, () => Promise<T>가 모두 올 수 있어야 함.\nexport function defineConfig(config: Executable<SonamuConfig>): Promise<SonamuConfig> {\n  if (typeof config === \"function\") {\n    return Promise.resolve(config());\n  }\n\n  return Promise.resolve(config);\n}\n\n/**\n * sonamu.config.ts 파일을 로드합니다.\n * 이 설정 파일은 환경에 따라 다른 경로에 있을 수 있습니다.\n * dist를 빌드하는 환경이라면 dist 바로 아래에 있을 것이고(cli-wrapper.ts에서 빌드),\n * 그렇지 않은 환경이라면 프로젝트 루트에 있을 것입니다.\n *\n * 이 함수는 의도적으로 다른 의존성의 사용을 최대한 배제하였습니다.\n * 이는 실행 초기에 최대한 빠르게 설정을 읽어올 수 있도록 하기 위함입니다.\n * 따라서 경로 concat과 URL scheme 추가도 단순한 문자열 조작으로 처리하였습니다.\n *\n * @param rootPath\n * @returns\n */\nexport async function loadConfig(rootPath: string): Promise<SonamuConfig> {\n  const shouldLoadSourceConfig = process.env.HOT === \"yes\" || process.env.VITEST === \"true\";\n  const configPath = shouldLoadSourceConfig\n    ? `${rootPath}/src/sonamu.config.ts`\n    : `${rootPath}/dist/sonamu.config.js`;\n\n  if (shouldLoadSourceConfig) {\n    const { ensureTsLoaderRegistered } = await import(\"../bin/ts-loader-registration\");\n    await ensureTsLoaderRegistered(rootPath);\n  }\n\n  const { default: config } = await import(`file://${configPath}`);\n  return config;\n}\n"],"mappings":";;;AA2SA,SAAgB,aAAa,QAAyD;AACpF,KAAI,OAAO,WAAW,YAAY;AAChC,SAAO,QAAQ,QAAQ,QAAQ,CAAC;;AAGlC,QAAO,QAAQ,QAAQ,OAAO;;;;;;;;;;;;;;;AAgBhC,eAAsB,WAAW,UAAyC;CACxE,MAAM,yBAAyB,QAAQ,IAAI,QAAQ,SAAS,QAAQ,IAAI,WAAW;CACnF,MAAM,aAAa,yBACf,GAAG,SAAS,yBACZ,GAAG,SAAS;AAEhB,KAAI,wBAAwB;EAC1B,MAAM,EAAE,6BAA6B,MAAM,OAAO;AAClD,QAAM,yBAAyB,SAAS;;CAG1C,MAAM,EAAE,SAAS,WAAW,MAAM,OAAO,UAAU;AACnD,QAAO"}
@@ -7,23 +7,33 @@ import { type NaiteStore } from "../naite/naite";
7
7
  import { type BufferedFile } from "../storage/buffered-file";
8
8
  import { type UploadedFile } from "../storage/uploaded-file";
9
9
  import { type createSSEFactory } from "../stream/sse";
10
+ import { type WebSocketConnection, type WebSocketEventMap } from "../stream/ws";
10
11
  export interface ContextExtend {
11
12
  }
12
- export type Context = {
13
+ type BaseContext = {
13
14
  request: FastifyRequest;
14
- reply: FastifyReply<Server, IncomingMessage, ServerResponse, RouteGenericInterface>;
15
15
  headers: IncomingHttpHeaders;
16
- createSSE: <T extends ZodObject>(events: T) => ReturnType<typeof createSSEFactory<T>>;
17
16
  naiteStore: NaiteStore;
18
17
  /** 현재 요청의 locale */
19
18
  locale: string;
20
- /** buffer 모드에서 업로드된 파일 */
21
- bufferedFiles?: BufferedFile[];
22
- /** stream 모드에서 업로드된 파일 */
23
- uploadedFiles?: UploadedFile[];
24
19
  /** 현재 로그인한 사용자 (null이면 미인증) */
25
20
  user: User | null;
26
21
  /** 현재 세션 정보 (null이면 미인증) */
27
22
  session: Session | null;
23
+ };
24
+ export type Context = BaseContext & {
25
+ transport: "http";
26
+ reply: FastifyReply<Server, IncomingMessage, ServerResponse, RouteGenericInterface>;
27
+ createSSE: <T extends ZodObject>(events: T) => ReturnType<typeof createSSEFactory<T>>;
28
+ /** buffer 모드에서 업로드된 파일 */
29
+ bufferedFiles?: BufferedFile[];
30
+ /** stream 모드에서 업로드된 파일 */
31
+ uploadedFiles?: UploadedFile[];
32
+ } & ContextExtend;
33
+ export type WebSocketContext<TOut extends WebSocketEventMap = WebSocketEventMap, TIn extends WebSocketEventMap = WebSocketEventMap> = BaseContext & {
34
+ transport: "ws";
35
+ ws: WebSocketConnection<TOut, TIn>;
28
36
  } & ContextExtend;
37
+ export type RuntimeContext = Context | WebSocketContext;
38
+ export {};
29
39
  //# sourceMappingURL=context.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"context.d.ts","sourceRoot":"","sources":["../../src/api/context.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,KAAK,mBAAmB,EACxB,KAAK,eAAe,EACpB,KAAK,MAAM,EACX,KAAK,cAAc,EACpB,MAAM,MAAM,CAAC;AAEd,OAAO,EAAE,KAAK,OAAO,EAAE,KAAK,IAAI,EAAE,MAAM,aAAa,CAAC;AACtD,OAAO,EAAE,KAAK,YAAY,EAAE,KAAK,cAAc,EAAE,MAAM,SAAS,CAAC;AACjE,OAAO,EAAE,KAAK,qBAAqB,EAAE,MAAM,qBAAqB,CAAC;AACjE,OAAO,EAAE,KAAK,SAAS,EAAE,MAAM,KAAK,CAAC;AAErC,OAAO,EAAE,KAAK,UAAU,EAAE,MAAM,gBAAgB,CAAC;AACjD,OAAO,EAAE,KAAK,YAAY,EAAE,MAAM,0BAA0B,CAAC;AAC7D,OAAO,EAAE,KAAK,YAAY,EAAE,MAAM,0BAA0B,CAAC;AAC7D,OAAO,EAAE,KAAK,gBAAgB,EAAE,MAAM,eAAe,CAAC;AAGtD,MAAM,WAAW,aAAa;CAAG;AACjC,MAAM,MAAM,OAAO,GAAG;IACpB,OAAO,EAAE,cAAc,CAAC;IACxB,KAAK,EAAE,YAAY,CAAC,MAAM,EAAE,eAAe,EAAE,cAAc,EAAE,qBAAqB,CAAC,CAAC;IACpF,OAAO,EAAE,mBAAmB,CAAC;IAC7B,SAAS,EAAE,CAAC,CAAC,SAAS,SAAS,EAAE,MAAM,EAAE,CAAC,KAAK,UAAU,CAAC,OAAO,gBAAgB,CAAC,CAAC,CAAC,CAAC,CAAC;IACtF,UAAU,EAAE,UAAU,CAAC;IACvB,oBAAoB;IACpB,MAAM,EAAE,MAAM,CAAC;IACf,0BAA0B;IAC1B,aAAa,CAAC,EAAE,YAAY,EAAE,CAAC;IAC/B,0BAA0B;IAC1B,aAAa,CAAC,EAAE,YAAY,EAAE,CAAC;IAE/B,+BAA+B;IAC/B,IAAI,EAAE,IAAI,GAAG,IAAI,CAAC;IAClB,4BAA4B;IAC5B,OAAO,EAAE,OAAO,GAAG,IAAI,CAAC;CACzB,GAAG,aAAa,CAAC"}
1
+ {"version":3,"file":"context.d.ts","sourceRoot":"","sources":["../../src/api/context.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,KAAK,mBAAmB,EACxB,KAAK,eAAe,EACpB,KAAK,MAAM,EACX,KAAK,cAAc,EACpB,MAAM,MAAM,CAAC;AAEd,OAAO,EAAE,KAAK,OAAO,EAAE,KAAK,IAAI,EAAE,MAAM,aAAa,CAAC;AACtD,OAAO,EAAE,KAAK,YAAY,EAAE,KAAK,cAAc,EAAE,MAAM,SAAS,CAAC;AACjE,OAAO,EAAE,KAAK,qBAAqB,EAAE,MAAM,qBAAqB,CAAC;AACjE,OAAO,EAAE,KAAK,SAAS,EAAE,MAAM,KAAK,CAAC;AAErC,OAAO,EAAE,KAAK,UAAU,EAAE,MAAM,gBAAgB,CAAC;AACjD,OAAO,EAAE,KAAK,YAAY,EAAE,MAAM,0BAA0B,CAAC;AAC7D,OAAO,EAAE,KAAK,YAAY,EAAE,MAAM,0BAA0B,CAAC;AAC7D,OAAO,EAAE,KAAK,gBAAgB,EAAE,MAAM,eAAe,CAAC;AACtD,OAAO,EAAE,KAAK,mBAAmB,EAAE,KAAK,iBAAiB,EAAE,MAAM,cAAc,CAAC;AAGhF,MAAM,WAAW,aAAa;CAAG;AACjC,KAAK,WAAW,GAAG;IACjB,OAAO,EAAE,cAAc,CAAC;IACxB,OAAO,EAAE,mBAAmB,CAAC;IAC7B,UAAU,EAAE,UAAU,CAAC;IACvB,oBAAoB;IACpB,MAAM,EAAE,MAAM,CAAC;IACf,+BAA+B;IAC/B,IAAI,EAAE,IAAI,GAAG,IAAI,CAAC;IAClB,4BAA4B;IAC5B,OAAO,EAAE,OAAO,GAAG,IAAI,CAAC;CACzB,CAAC;AAEF,MAAM,MAAM,OAAO,GAAG,WAAW,GAAG;IAClC,SAAS,EAAE,MAAM,CAAC;IAClB,KAAK,EAAE,YAAY,CAAC,MAAM,EAAE,eAAe,EAAE,cAAc,EAAE,qBAAqB,CAAC,CAAC;IACpF,SAAS,EAAE,CAAC,CAAC,SAAS,SAAS,EAAE,MAAM,EAAE,CAAC,KAAK,UAAU,CAAC,OAAO,gBAAgB,CAAC,CAAC,CAAC,CAAC,CAAC;IACtF,0BAA0B;IAC1B,aAAa,CAAC,EAAE,YAAY,EAAE,CAAC;IAC/B,0BAA0B;IAC1B,aAAa,CAAC,EAAE,YAAY,EAAE,CAAC;CAChC,GAAG,aAAa,CAAC;AAElB,MAAM,MAAM,gBAAgB,CAC1B,IAAI,SAAS,iBAAiB,GAAG,iBAAiB,EAClD,GAAG,SAAS,iBAAiB,GAAG,iBAAiB,IAC/C,WAAW,GAAG;IAChB,SAAS,EAAE,IAAI,CAAC;IAChB,EAAE,EAAE,mBAAmB,CAAC,IAAI,EAAE,GAAG,CAAC,CAAC;CACpC,GAAG,aAAa,CAAC;AAElB,MAAM,MAAM,cAAc,GAAG,OAAO,GAAG,gBAAgB,CAAC"}
@@ -6,4 +6,4 @@ var init_context = __esmMin((() => {}));
6
6
  //#endregion
7
7
  init_context();
8
8
  export { init_context };
9
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29udGV4dC5qcyIsIm5hbWVzIjpbXSwic291cmNlcyI6WyIuLi8uLi9zcmMvYXBpL2NvbnRleHQudHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtcbiAgdHlwZSBJbmNvbWluZ0h0dHBIZWFkZXJzLFxuICB0eXBlIEluY29taW5nTWVzc2FnZSxcbiAgdHlwZSBTZXJ2ZXIsXG4gIHR5cGUgU2VydmVyUmVzcG9uc2UsXG59IGZyb20gXCJodHRwXCI7XG5cbmltcG9ydCB7IHR5cGUgU2Vzc2lvbiwgdHlwZSBVc2VyIH0gZnJvbSBcImJldHRlci1hdXRoXCI7XG5pbXBvcnQgeyB0eXBlIEZhc3RpZnlSZXBseSwgdHlwZSBGYXN0aWZ5UmVxdWVzdCB9IGZyb20gXCJmYXN0aWZ5XCI7XG5pbXBvcnQgeyB0eXBlIFJvdXRlR2VuZXJpY0ludGVyZmFjZSB9IGZyb20gXCJmYXN0aWZ5L3R5cGVzL3JvdXRlXCI7XG5pbXBvcnQgeyB0eXBlIFpvZE9iamVjdCB9IGZyb20gXCJ6b2RcIjtcblxuaW1wb3J0IHsgdHlwZSBOYWl0ZVN0b3JlIH0gZnJvbSBcIi4uL25haXRlL25haXRlXCI7XG5pbXBvcnQgeyB0eXBlIEJ1ZmZlcmVkRmlsZSB9IGZyb20gXCIuLi9zdG9yYWdlL2J1ZmZlcmVkLWZpbGVcIjtcbmltcG9ydCB7IHR5cGUgVXBsb2FkZWRGaWxlIH0gZnJvbSBcIi4uL3N0b3JhZ2UvdXBsb2FkZWQtZmlsZVwiO1xuaW1wb3J0IHsgdHlwZSBjcmVhdGVTU0VGYWN0b3J5IH0gZnJvbSBcIi4uL3N0cmVhbS9zc2VcIjtcblxuLy8gb3hsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uby1lbXB0eS1pbnRlcmZhY2UgLS0gQ29udGV4dCDtmZXsnqUg7YOA7J6FXG5leHBvcnQgaW50ZXJmYWNlIENvbnRleHRFeHRlbmQge31cbmV4cG9ydCB0eXBlIENvbnRleHQgPSB7XG4gIHJlcXVlc3Q6IEZhc3RpZnlSZXF1ZXN0O1xuICByZXBseTogRmFzdGlmeVJlcGx5PFNlcnZlciwgSW5jb21pbmdNZXNzYWdlLCBTZXJ2ZXJSZXNwb25zZSwgUm91dGVHZW5lcmljSW50ZXJmYWNlPjtcbiAgaGVhZGVyczogSW5jb21pbmdIdHRwSGVhZGVycztcbiAgY3JlYXRlU1NFOiA8VCBleHRlbmRzIFpvZE9iamVjdD4oZXZlbnRzOiBUKSA9PiBSZXR1cm5UeXBlPHR5cGVvZiBjcmVhdGVTU0VGYWN0b3J5PFQ+PjtcbiAgbmFpdGVTdG9yZTogTmFpdGVTdG9yZTtcbiAgLyoqIO2YhOyerCDsmpTssq3snZggbG9jYWxlICovXG4gIGxvY2FsZTogc3RyaW5nO1xuICAvKiogYnVmZmVyIOuqqOuTnOyXkOyEnCDsl4XroZzrk5zrkJwg7YyM7J28ICovXG4gIGJ1ZmZlcmVkRmlsZXM/OiBCdWZmZXJlZEZpbGVbXTtcbiAgLyoqIHN0cmVhbSDrqqjrk5zsl5DshJwg7JeF66Gc65Oc65CcIO2MjOydvCAqL1xuICB1cGxvYWRlZEZpbGVzPzogVXBsb2FkZWRGaWxlW107XG5cbiAgLyoqIO2YhOyerCDroZzqt7jsnbjtlZwg7IKs7Jqp7J6QIChudWxs7J2066m0IOuvuOyduOymnSkgKi9cbiAgdXNlcjogVXNlciB8IG51bGw7XG4gIC8qKiDtmITsnqwg7IS47IWYIOygleuztCAobnVsbOydtOuptCDrr7jsnbjspp0pICovXG4gIHNlc3Npb246IFNlc3Npb24gfCBudWxsO1xufSAmIENvbnRleHRFeHRlbmQ7XG4iXSwibWFwcGluZ3MiOiIifQ==
9
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29udGV4dC5qcyIsIm5hbWVzIjpbXSwic291cmNlcyI6WyIuLi8uLi9zcmMvYXBpL2NvbnRleHQudHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtcbiAgdHlwZSBJbmNvbWluZ0h0dHBIZWFkZXJzLFxuICB0eXBlIEluY29taW5nTWVzc2FnZSxcbiAgdHlwZSBTZXJ2ZXIsXG4gIHR5cGUgU2VydmVyUmVzcG9uc2UsXG59IGZyb20gXCJodHRwXCI7XG5cbmltcG9ydCB7IHR5cGUgU2Vzc2lvbiwgdHlwZSBVc2VyIH0gZnJvbSBcImJldHRlci1hdXRoXCI7XG5pbXBvcnQgeyB0eXBlIEZhc3RpZnlSZXBseSwgdHlwZSBGYXN0aWZ5UmVxdWVzdCB9IGZyb20gXCJmYXN0aWZ5XCI7XG5pbXBvcnQgeyB0eXBlIFJvdXRlR2VuZXJpY0ludGVyZmFjZSB9IGZyb20gXCJmYXN0aWZ5L3R5cGVzL3JvdXRlXCI7XG5pbXBvcnQgeyB0eXBlIFpvZE9iamVjdCB9IGZyb20gXCJ6b2RcIjtcblxuaW1wb3J0IHsgdHlwZSBOYWl0ZVN0b3JlIH0gZnJvbSBcIi4uL25haXRlL25haXRlXCI7XG5pbXBvcnQgeyB0eXBlIEJ1ZmZlcmVkRmlsZSB9IGZyb20gXCIuLi9zdG9yYWdlL2J1ZmZlcmVkLWZpbGVcIjtcbmltcG9ydCB7IHR5cGUgVXBsb2FkZWRGaWxlIH0gZnJvbSBcIi4uL3N0b3JhZ2UvdXBsb2FkZWQtZmlsZVwiO1xuaW1wb3J0IHsgdHlwZSBjcmVhdGVTU0VGYWN0b3J5IH0gZnJvbSBcIi4uL3N0cmVhbS9zc2VcIjtcbmltcG9ydCB7IHR5cGUgV2ViU29ja2V0Q29ubmVjdGlvbiwgdHlwZSBXZWJTb2NrZXRFdmVudE1hcCB9IGZyb20gXCIuLi9zdHJlYW0vd3NcIjtcblxuLy8gb3hsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uby1lbXB0eS1pbnRlcmZhY2UgLS0gQ29udGV4dCDtmZXsnqUg7YOA7J6FXG5leHBvcnQgaW50ZXJmYWNlIENvbnRleHRFeHRlbmQge31cbnR5cGUgQmFzZUNvbnRleHQgPSB7XG4gIHJlcXVlc3Q6IEZhc3RpZnlSZXF1ZXN0O1xuICBoZWFkZXJzOiBJbmNvbWluZ0h0dHBIZWFkZXJzO1xuICBuYWl0ZVN0b3JlOiBOYWl0ZVN0b3JlO1xuICAvKiog7ZiE7J6sIOyalOyyreydmCBsb2NhbGUgKi9cbiAgbG9jYWxlOiBzdHJpbmc7XG4gIC8qKiDtmITsnqwg66Gc6re47J247ZWcIOyCrOyaqeyekCAobnVsbOydtOuptCDrr7jsnbjspp0pICovXG4gIHVzZXI6IFVzZXIgfCBudWxsO1xuICAvKiog7ZiE7J6sIOyEuOyFmCDsoJXrs7QgKG51bGzsnbTrqbQg66+47J247KadKSAqL1xuICBzZXNzaW9uOiBTZXNzaW9uIHwgbnVsbDtcbn07XG5cbmV4cG9ydCB0eXBlIENvbnRleHQgPSBCYXNlQ29udGV4dCAmIHtcbiAgdHJhbnNwb3J0OiBcImh0dHBcIjtcbiAgcmVwbHk6IEZhc3RpZnlSZXBseTxTZXJ2ZXIsIEluY29taW5nTWVzc2FnZSwgU2VydmVyUmVzcG9uc2UsIFJvdXRlR2VuZXJpY0ludGVyZmFjZT47XG4gIGNyZWF0ZVNTRTogPFQgZXh0ZW5kcyBab2RPYmplY3Q+KGV2ZW50czogVCkgPT4gUmV0dXJuVHlwZTx0eXBlb2YgY3JlYXRlU1NFRmFjdG9yeTxUPj47XG4gIC8qKiBidWZmZXIg66qo65Oc7JeQ7IScIOyXheuhnOuTnOuQnCDtjIzsnbwgKi9cbiAgYnVmZmVyZWRGaWxlcz86IEJ1ZmZlcmVkRmlsZVtdO1xuICAvKiogc3RyZWFtIOuqqOuTnOyXkOyEnCDsl4XroZzrk5zrkJwg7YyM7J28ICovXG4gIHVwbG9hZGVkRmlsZXM/OiBVcGxvYWRlZEZpbGVbXTtcbn0gJiBDb250ZXh0RXh0ZW5kO1xuXG5leHBvcnQgdHlwZSBXZWJTb2NrZXRDb250ZXh0PFxuICBUT3V0IGV4dGVuZHMgV2ViU29ja2V0RXZlbnRNYXAgPSBXZWJTb2NrZXRFdmVudE1hcCxcbiAgVEluIGV4dGVuZHMgV2ViU29ja2V0RXZlbnRNYXAgPSBXZWJTb2NrZXRFdmVudE1hcCxcbj4gPSBCYXNlQ29udGV4dCAmIHtcbiAgdHJhbnNwb3J0OiBcIndzXCI7XG4gIHdzOiBXZWJTb2NrZXRDb25uZWN0aW9uPFRPdXQsIFRJbj47XG59ICYgQ29udGV4dEV4dGVuZDtcblxuZXhwb3J0IHR5cGUgUnVudGltZUNvbnRleHQgPSBDb250ZXh0IHwgV2ViU29ja2V0Q29udGV4dDtcbiJdLCJtYXBwaW5ncyI6IiJ9
@@ -36,6 +36,21 @@ export type StreamDecoratorOptions = {
36
36
  guards?: GuardKey[];
37
37
  description?: string;
38
38
  };
39
+ export type WebSocketDecoratorOptions = {
40
+ outEvents: z.ZodObject<any>;
41
+ inEvents: z.ZodObject<any>;
42
+ path?: string;
43
+ resourceName?: string;
44
+ guards?: GuardKey[];
45
+ description?: string;
46
+ heartbeat?: number;
47
+ maxPayload?: number;
48
+ namespace?: string;
49
+ };
50
+ export type ResolvedWebSocketDecoratorOptions = WebSocketDecoratorOptions & {
51
+ outEventsTypeRef?: ApiParamType.Ref;
52
+ inEventsTypeRef?: ApiParamType.Ref;
53
+ };
39
54
  type BufferUploadOptions = {
40
55
  consume?: "buffer";
41
56
  };
@@ -58,6 +73,7 @@ export declare const registeredApis: {
58
73
  path: string;
59
74
  options: ApiDecoratorOptions;
60
75
  streamOptions?: StreamDecoratorOptions;
76
+ websocketOptions?: ResolvedWebSocketDecoratorOptions;
61
77
  uploadOptions?: UploadDecoratorOptions;
62
78
  }[];
63
79
  export type ExtendedApi = {
@@ -66,6 +82,7 @@ export type ExtendedApi = {
66
82
  path: string;
67
83
  options: ApiDecoratorOptions;
68
84
  streamOptions?: StreamDecoratorOptions;
85
+ websocketOptions?: ResolvedWebSocketDecoratorOptions;
69
86
  uploadOptions?: UploadDecoratorOptions;
70
87
  typeParameters: ApiParamType.TypeParam[];
71
88
  parameters: ApiParam[];
@@ -78,6 +95,7 @@ type DecoratorTarget = {
78
95
  };
79
96
  export declare function api(options?: ApiDecoratorOptions): (target: DecoratorTarget, propertyKey: string, descriptor: PropertyDescriptor) => void;
80
97
  export declare function stream(options: StreamDecoratorOptions): (target: DecoratorTarget, propertyKey: string, descriptor: PropertyDescriptor) => void;
98
+ export declare function websocket(options: WebSocketDecoratorOptions): (target: DecoratorTarget, propertyKey: string, descriptor: PropertyDescriptor) => void;
81
99
  export declare function transactional(options?: TransactionalOptions): (target: DecoratorTarget, propertyKey: string, descriptor: PropertyDescriptor) => PropertyDescriptor;
82
100
  /**
83
101
  * 파일 업로드 API를 생성해줍니다. (@api 데코레이터 없이 독립적으로 사용)
@@ -1 +1 @@
1
- {"version":3,"file":"decorators.d.ts","sourceRoot":"","sources":["../../src/api/decorators.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"decorators.d.ts","sourceRoot":"","sources":["../../src/api/decorators.ts"],"names":[],"mappings":"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"}
@@ -14,7 +14,7 @@ import assert from "assert";
14
14
  function checkSingleDecorator(target, propertyKey, decoratorType) {
15
15
  const method = target[propertyKey];
16
16
  if (method?.__decoratorType && method?.__decoratorType !== decoratorType) {
17
- throw new Error(`@${decoratorType.description ?? String(decoratorType)} decorator can only be used once on ${target.constructor.name}.${propertyKey}. You can use only one of @api or @stream decorator on the same method.`);
17
+ throw new Error(`@${decoratorType.description ?? String(decoratorType)} decorator can only be used once on ${target.constructor.name}.${propertyKey}. You can use only one of @api, @stream, @websocket, or @upload decorator on the same method.`);
18
18
  } else {
19
19
  method.__decoratorType = decoratorType;
20
20
  }
@@ -120,12 +120,62 @@ function stream(options) {
120
120
  };
121
121
  };
122
122
  }
123
+ function websocket(options) {
124
+ return (target, propertyKey, descriptor) => {
125
+ const modelName = target.constructor.name.match(/(.+)Class$/)?.[1];
126
+ assert(modelName, `modelName is required on @websocket decorator on ${target.constructor.name}.${propertyKey}`);
127
+ const methodName = propertyKey;
128
+ checkSingleDecorator(target, propertyKey, DECORATOR_TYPES.WEBSOCKET);
129
+ const defaultPath = `/${inflection.camelize(modelName.replace(/Model$/, "").replace(/Frame$/, ""), true)}/${inflection.camelize(propertyKey, true)}`;
130
+ const path = options.path ?? defaultPath;
131
+ const { outEvents: _outEvents, inEvents: _inEvents, ...apiOptions } = options;
132
+ const optionsWithDefaults = {
133
+ ...apiOptions,
134
+ httpMethod: "GET"
135
+ };
136
+ const existingApi = registeredApis.find((api$1) => api$1.modelName === modelName && api$1.methodName === methodName);
137
+ if (existingApi) {
138
+ assertNoConflictingPath("websocket", modelName, methodName, existingApi.path, path);
139
+ existingApi.path = path;
140
+ assertNoConflictingOptions("websocket", modelName, methodName, existingApi.options, optionsWithDefaults);
141
+ existingApi.options = {
142
+ ...existingApi.options,
143
+ ...optionsWithDefaults
144
+ };
145
+ existingApi.websocketOptions = options;
146
+ } else {
147
+ registeredApis.push({
148
+ modelName,
149
+ methodName,
150
+ path,
151
+ options: optionsWithDefaults,
152
+ websocketOptions: options
153
+ });
154
+ }
155
+ const originalMethod = descriptor.value;
156
+ descriptor.value = async function(...args) {
157
+ if (this instanceof BaseModelClass) {
158
+ getLogger(convertDomainToCategory(this.modelName, "model")).debug("websocket: {model}.{method}", {
159
+ model: modelName,
160
+ method: methodName
161
+ });
162
+ }
163
+ if (this instanceof BaseFrameClass) {
164
+ getLogger(convertDomainToCategory(this.frameName, "frame")).debug("websocket: {model}.{method}", {
165
+ model: modelName,
166
+ method: methodName
167
+ });
168
+ }
169
+ return originalMethod.apply(this, args);
170
+ };
171
+ };
172
+ }
123
173
  function transactional(options = {}) {
124
174
  const { isolation, readOnly, dbPreset = "w" } = options;
125
175
  return (target, propertyKey, descriptor) => {
126
176
  const originalMethod = descriptor.value;
127
177
  const modelName = target.constructor.name.match(/(.+)Class$/)?.[1];
128
- assert(modelName, `modelName is required on @stream decorator on ${target.constructor.name}.${propertyKey}`);
178
+ assert(modelName, `modelName is required on @transactional decorator on ${target.constructor.name}.${propertyKey}`);
129
179
  const methodName = propertyKey;
130
180
  descriptor.value = async function(...args) {
131
181
  this.logger.debug("transactional: {model}.{method}", {
@@ -250,11 +300,12 @@ var init_decorators = __esmMin((() => {
250
300
  DECORATOR_TYPES = {
251
301
  API: Symbol("api"),
252
302
  STREAM: Symbol("stream"),
303
+ WEBSOCKET: Symbol("websocket"),
253
304
  UPLOAD: Symbol("upload")
254
305
  };
255
306
  }));
256
307
 
257
308
  //#endregion
258
309
  init_decorators();
259
- export { api, init_decorators, registeredApis, stream, transactional, upload };
260
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"decorators.js","names":["api","registeredApis: {\n  /**\n   * modelName은 모델 클래스 이름입니다. (ex. \"UserModel\")\n   */\n  modelName: string;\n  methodName: string;\n  path: string;\n  options: ApiDecoratorOptions;\n  streamOptions?: StreamDecoratorOptions;\n  uploadOptions?: UploadDecoratorOptions;\n}[]"],"sources":["../../src/api/decorators.ts"],"sourcesContent":["import assert from \"assert\";\n\nimport { type FastifyMultipartBaseOptions } from \"@fastify/multipart\";\nimport { getLogger } from \"@logtape/logtape\";\nimport { type HTTPMethods } from \"fastify\";\nimport inflection from \"inflection\";\nimport { isEqual } from \"radashi\";\nimport { type z } from \"zod\";\n\nimport { type CacheControlConfig } from \"../cache-control/types\";\nimport { type CompressConfig } from \"../compress/types\";\nimport { BaseModelClass } from \"../database/base-model\";\nimport { DB } from \"../database/db\";\nimport { PuriTransactionWrapper } from \"../database/puri-wrapper\";\nimport { type TransactionalOptions } from \"../database/puri-wrapper\";\nimport { UpsertBuilder } from \"../database/upsert-builder\";\nimport { convertDomainToCategory } from \"../logger/category\";\nimport { type DriverKey } from \"../storage/drivers\";\nimport { type KeyGenerator } from \"../storage/types\";\nimport { type ApiParam, type ApiParamType } from \"../types/types\";\nimport { BaseFrameClass } from \"./base-frame\";\n\nexport interface GuardKeys {\n  query: true;\n  admin: true;\n  user: true;\n}\nexport type GuardKey = keyof GuardKeys;\nexport type ServiceClient =\n  | \"axios\"\n  | \"axios-multipart\"\n  | \"tanstack-query\"\n  | \"tanstack-mutation\"\n  | \"tanstack-mutation-multipart\"\n  | \"window-fetch\";\nexport type ApiDecoratorOptions = {\n  httpMethod?: HTTPMethods;\n  contentType?:\n    | \"text/plain\"\n    | \"text/html\"\n    | \"text/xml\"\n    | \"application/json\"\n    | \"application/octet-stream\";\n  clients?: ServiceClient[];\n  path?: string;\n  resourceName?: string;\n  guards?: GuardKey[];\n  description?: string;\n  timeout?: number;\n  /** API 응답의 Cache-Control 헤더 설정. 설정하지 않으면 cacheControlHandler 또는 기본값이 적용됩니다. */\n  cacheControl?: CacheControlConfig;\n  /** API 응답의 압축 설정. false로 설정하면 압축을 비활성화합니다. */\n  compress?: CompressConfig;\n};\nexport type StreamDecoratorOptions = {\n  type: \"sse\"; // | 'ws\n  // oxlint-disable-next-line @typescript-eslint/no-explicit-any -- 이벤트 키별로 넘겨주는 값이므로 어떤 타입이든 상관없음\n  events: z.ZodObject<any>;\n  path?: string;\n  resourceName?: string;\n  guards?: GuardKey[];\n  description?: string;\n};\n\ntype BufferUploadOptions = {\n  consume?: \"buffer\";\n};\ntype StreamUploadOptions = {\n  consume: \"stream\";\n  destination: DriverKey;\n  keyGenerator?: KeyGenerator;\n};\nexport type UploadDecoratorOptions = {\n  guards?: GuardKey[];\n  description?: string;\n  limits?: FastifyMultipartBaseOptions[\"limits\"];\n} & (BufferUploadOptions | StreamUploadOptions);\n\nexport const registeredApis: {\n  /**\n   * modelName은 모델 클래스 이름입니다. (ex. \"UserModel\")\n   */\n  modelName: string;\n  methodName: string;\n  path: string;\n  options: ApiDecoratorOptions;\n  streamOptions?: StreamDecoratorOptions;\n  uploadOptions?: UploadDecoratorOptions;\n}[] = [];\nexport type ExtendedApi = {\n  modelName: string;\n  methodName: string;\n  path: string;\n  options: ApiDecoratorOptions;\n  streamOptions?: StreamDecoratorOptions;\n  uploadOptions?: UploadDecoratorOptions;\n  typeParameters: ApiParamType.TypeParam[];\n  parameters: ApiParam[];\n  returnType: ApiParamType;\n};\ntype DecoratorTarget = { constructor: { name: string } };\n\nconst DECORATOR_TYPES = {\n  API: Symbol(\"api\"),\n  STREAM: Symbol(\"stream\"),\n  UPLOAD: Symbol(\"upload\"),\n} as const;\n\nfunction checkSingleDecorator(target: DecoratorTarget, propertyKey: string, decoratorType: symbol) {\n  const method = target[propertyKey as keyof typeof target] as { __decoratorType?: symbol };\n  if (method?.__decoratorType && method?.__decoratorType !== decoratorType) {\n    throw new Error(\n      `@${decoratorType.description ?? String(decoratorType)} decorator can only be used once on ${target.constructor.name}.${propertyKey}. You can use only one of @api or @stream decorator on the same method.`,\n    );\n  } else {\n    method.__decoratorType = decoratorType;\n  }\n}\n\nexport function api(options: ApiDecoratorOptions = {}) {\n  options = {\n    httpMethod: \"GET\",\n    contentType: \"application/json\",\n    clients: [\"axios\"],\n    ...options,\n  };\n\n  return (target: DecoratorTarget, propertyKey: string, descriptor: PropertyDescriptor) => {\n    const modelName = target.constructor.name.match(/(.+)Class$/)?.[1];\n    assert(\n      modelName,\n      `modelName is required on @api decorator on ${target.constructor.name}.${propertyKey}`,\n    );\n    const methodName = propertyKey;\n\n    // 메서드에 걸린 데코레이터 중복 체크\n    checkSingleDecorator(target, propertyKey, DECORATOR_TYPES.API);\n\n    const defaultPath = `/${inflection.camelize(\n      modelName.replace(/Model$/, \"\").replace(/Frame$/, \"\"),\n      true,\n    )}/${inflection.camelize(propertyKey, true)}`;\n    const path = options.path ?? defaultPath;\n\n    // 기존 동일한 메서드가 있는지 확인 후 있는 경우 override\n    const existingApi = registeredApis.find(\n      (api) => api.modelName === modelName && api.methodName === methodName,\n    );\n    if (existingApi) {\n      // 기존의 path와 새로운 path가 다르다면(=빈 스트링이 아니었는데 다른 스트링으로 바뀌게 된다면) 에러를 터뜨려줍니다.\n      assertNoConflictingPath(\"api\", modelName, methodName, existingApi.path, path);\n      existingApi.path = path;\n\n      // 기존의 옵션과 새로운 옵션이 겹치는 부분이 있다면 에러를 터뜨려줍니다.\n      assertNoConflictingOptions(\"api\", modelName, methodName, existingApi.options, options);\n      existingApi.options = {\n        ...existingApi.options, // 기존의 옵션을 존중하되\n        ...options, // @api 데코레이터의 옵션을 추가합니다.\n      };\n    } else {\n      registeredApis.push({\n        modelName,\n        methodName,\n        path,\n        options,\n      });\n    }\n\n    const originalMethod = descriptor.value;\n    descriptor.value = async function (this: BaseModelClass | BaseFrameClass, ...args: unknown[]) {\n      if (this instanceof BaseModelClass) {\n        getLogger(convertDomainToCategory(this.modelName, \"model\")).debug(\n          \"api: {httpMethod} {model}.{method}\",\n          {\n            httpMethod: options.httpMethod,\n            model: modelName,\n            method: methodName,\n          },\n        );\n      }\n\n      if (this instanceof BaseFrameClass) {\n        getLogger(convertDomainToCategory(this.frameName, \"frame\")).debug(\n          \"api: {httpMethod} {model}.{method}\",\n          {\n            httpMethod: options.httpMethod,\n            model: modelName,\n            method: methodName,\n          },\n        );\n      }\n\n      return originalMethod.apply(this, args);\n    };\n  };\n}\n\nexport function stream(options: StreamDecoratorOptions) {\n  return (target: DecoratorTarget, propertyKey: string, descriptor: PropertyDescriptor) => {\n    const modelName = target.constructor.name.match(/(.+)Class$/)?.[1];\n    assert(\n      modelName,\n      `modelName is required on @stream decorator on ${target.constructor.name}.${propertyKey}`,\n    );\n    const methodName = propertyKey;\n\n    // 메서드에 걸린 데코레이터 중복 체크\n    checkSingleDecorator(target, propertyKey, DECORATOR_TYPES.STREAM);\n\n    const defaultPath = `/${inflection.camelize(\n      modelName.replace(/Model$/, \"\").replace(/Frame$/, \"\"),\n      true,\n    )}/${inflection.camelize(propertyKey, true)}`;\n    const path = options.path ?? defaultPath;\n    // stream 전용 필드(events, type)는 ApiDecoratorOptions에 속하지 않으므로 제외\n    const { events: _, type: _type, ...apiOptions } = options;\n    const optionsWithDefaults = {\n      ...apiOptions,\n      httpMethod: \"GET\" as HTTPMethods,\n    };\n\n    const existingApi = registeredApis.find(\n      (api) => api.modelName === modelName && api.methodName === methodName,\n    );\n    if (existingApi) {\n      // 기존의 path와 새로운 path가 다르다면(=빈 스트링이 아니었는데 다른 스트링으로 바뀌게 된다면) 에러를 터뜨려줍니다.\n      assertNoConflictingPath(\"stream\", modelName, methodName, existingApi.path, path);\n      existingApi.path = path;\n\n      // 기존의 옵션과 새로운 옵션이 겹치는 부분이 있다면 에러를 터뜨려줍니다.\n      assertNoConflictingOptions(\n        \"stream\",\n        modelName,\n        methodName,\n        existingApi.options,\n        optionsWithDefaults,\n      );\n      existingApi.options = {\n        ...existingApi.options, // 기존의 옵션을 존중하되\n        ...optionsWithDefaults, // @stream 데코레이터의 옵션을 추가합니다.\n      };\n\n      existingApi.streamOptions = options;\n    } else {\n      registeredApis.push({\n        modelName,\n        methodName,\n        path,\n        options: optionsWithDefaults,\n        streamOptions: options,\n      });\n    }\n\n    const originalMethod = descriptor.value;\n    descriptor.value = async function (this: BaseModelClass | BaseFrameClass, ...args: unknown[]) {\n      if (this instanceof BaseModelClass) {\n        getLogger(convertDomainToCategory(this.modelName, \"model\")).debug(\n          \"stream: {model}.{method}\",\n          {\n            model: modelName,\n            method: methodName,\n          },\n        );\n      }\n\n      if (this instanceof BaseFrameClass) {\n        getLogger(convertDomainToCategory(this.frameName, \"frame\")).debug(\n          \"stream: {model}.{method}\",\n          {\n            model: modelName,\n            method: methodName,\n          },\n        );\n      }\n\n      return originalMethod.apply(this, args);\n    };\n  };\n}\n\nexport function transactional(options: TransactionalOptions = {}) {\n  const { isolation, readOnly, dbPreset = \"w\" } = options;\n\n  return (target: DecoratorTarget, propertyKey: string, descriptor: PropertyDescriptor) => {\n    const originalMethod = descriptor.value;\n    const modelName = target.constructor.name.match(/(.+)Class$/)?.[1];\n    assert(\n      modelName,\n      `modelName is required on @stream decorator on ${target.constructor.name}.${propertyKey}`,\n    );\n    const methodName = propertyKey;\n\n    descriptor.value = async function (this: BaseModelClass, ...args: unknown[]) {\n      this.logger.debug(\"transactional: {model}.{method}\", {\n        model: modelName,\n        method: methodName,\n      });\n\n      const existingContext = DB.transactionStorage.getStore();\n\n      // AsyncLocalStorage 컨텍스트 없거나 해당 preset의 트랜잭션이 없으면 새로 시작\n      const startTransaction = async () => {\n        const puri = this.getPuri(dbPreset);\n\n        return puri.knex.transaction(\n          async (trx) => {\n            this.logger.debug(\"new transaction context: {dbPreset}\", { dbPreset });\n            const trxWrapper = new PuriTransactionWrapper(trx, new UpsertBuilder());\n            // TransactionContext에 트랜잭션 저장\n            DB.getTransactionContext().setTransaction(dbPreset, trxWrapper);\n\n            try {\n              return await originalMethod.apply(this, args);\n            } finally {\n              // 트랜잭션 제거\n              this.logger.debug(\"delete transaction context: {dbPreset}\", { dbPreset });\n              DB.getTransactionContext().deleteTransaction(dbPreset);\n            }\n          },\n          { isolationLevel: isolation, readOnly },\n        );\n      };\n\n      // AsyncLocalStorage 컨텍스트가 없으면 새로 생성\n      if (!existingContext) {\n        return DB.runWithTransaction(startTransaction);\n      }\n\n      // 이미 AsyncLocalStorage 컨텍스트 안에 있는지 확인 후 해당 preset의 트랜잭션이 이미 있으면 재사용\n      if (existingContext?.getTransaction(dbPreset)) {\n        this.logger.debug(\"reuse transaction context: {dbPreset}\", { dbPreset });\n        return originalMethod.apply(this, args);\n      }\n\n      // 컨텍스트는 있지만 이 preset의 트랜잭션은 없는 경우 (같은 컨텍스트 내에서 실행)\n      return startTransaction();\n    };\n\n    return descriptor;\n  };\n}\n\n/**\n * 파일 업로드 API를 생성해줍니다. (@api 데코레이터 없이 독립적으로 사용)\n * @param options\n * @returns\n */\nexport function upload(options: UploadDecoratorOptions = { consume: \"buffer\" }) {\n  return (target: DecoratorTarget, propertyKey: string, descriptor: PropertyDescriptor) => {\n    const originalMethod = descriptor.value;\n    const modelName = target.constructor.name.match(/(.+)Class$/)?.[1];\n    assert(\n      modelName,\n      `modelName is required on @upload decorator on ${target.constructor.name}.${propertyKey}`,\n    );\n    const methodName = propertyKey;\n\n    // 메서드에 걸린 데코레이터 중복 체크\n    checkSingleDecorator(target, propertyKey, DECORATOR_TYPES.UPLOAD);\n\n    const defaultPath = `/${inflection.camelize(\n      modelName.replace(/Model$/, \"\").replace(/Frame$/, \"\"),\n      true,\n    )}/${inflection.camelize(methodName, true)}`;\n\n    // registeredApis에서 해당 API 찾아서 uploadOptions 추가\n    const existingApi = registeredApis.find(\n      (api) => api.modelName === modelName && api.methodName === methodName,\n    );\n\n    if (existingApi) {\n      // 재등록 시 업로드 옵션만 갱신\n      existingApi.uploadOptions = options;\n    } else {\n      // 새로 등록\n      registeredApis.push({\n        modelName,\n        methodName,\n        path: defaultPath,\n        options: {\n          httpMethod: \"POST\",\n          clients: [\"axios-multipart\", \"tanstack-mutation-multipart\"],\n          guards: options.guards,\n          description: options.description,\n        },\n        uploadOptions: options,\n      });\n    }\n\n    descriptor.value = async function (this: BaseModelClass | BaseFrameClass, ...args: unknown[]) {\n      if (this instanceof BaseModelClass) {\n        getLogger(convertDomainToCategory(this.modelName, \"model\")).debug(\n          \"upload: {model}.{method}\",\n          {\n            model: modelName,\n            method: methodName,\n          },\n        );\n      }\n\n      if (this instanceof BaseFrameClass) {\n        getLogger(convertDomainToCategory(this.frameName, \"frame\")).debug(\n          \"upload: {model}.{method}\",\n          {\n            model: modelName,\n            method: methodName,\n          },\n        );\n      }\n\n      return originalMethod.apply(this, args);\n    };\n\n    return descriptor;\n  };\n}\n\n/**\n * 기존의 path와 새로운 path가 다르다면(=값이 있던 스트링이 다른 값이 있는 스트링으로 바뀌게 된다면) 에러를 터뜨려줍니다.\n * @param decoratorName 데코레이터 이름\n * @param modelName 모델 이름\n * @param methodName 메서드 이름\n * @param existingPath 기존의 path\n * @param newPath 새로운 path\n */\nfunction assertNoConflictingPath(\n  decoratorName: string,\n  modelName: string,\n  methodName: string,\n  existingPath: string,\n  newPath: string,\n) {\n  if (existingPath !== \"\" && newPath !== \"\" && existingPath !== newPath) {\n    // 이것이 무슨 상황이냐면요, api.path가 덮어씌워지는 상황입니다.\n    // 가령 @api({ path: \"/api/v1/users\" }) 데코레이터가 붙어있는 메서드에\n    // @stream({ path: \"/api/v1/users/stream\" }) 같은 것이 붙어 있는 상황입니다.\n    // 이렇게 되면 두 데코레이터가 같은 api의 path 필드를 건드리게 되므로, 에러를 터뜨려줍니다.\n    throw new Error(\n      `@${decoratorName} decorator on ${modelName}.${methodName} has conflicting path: ${newPath}. The decorator is trying to override the existing path(${existingPath}) with the new path(${newPath}).`,\n    );\n  }\n}\n\n/**\n * 기존의 옵션과 새로운 옵션이 겹치는 부분이 있다면 에러를 터뜨려줍니다.\n * @param decoratorName 데코레이터 이름\n * @param modelName 모델 이름\n * @param methodName 메서드 이름\n * @param existingOptions 기존의 옵션\n * @param newOptions 새로운 옵션\n */\nfunction assertNoConflictingOptions(\n  decoratorName: string,\n  modelName: string,\n  methodName: string,\n  // oxlint-disable-next-line @typescript-eslint/no-explicit-any -- <아 쉽게쉽게 좀 갑시다>\n  existingOptions: Record<string, any>,\n  // oxlint-disable-next-line @typescript-eslint/no-explicit-any -- <이럴 때 아니면 any 언제 씁니까>\n  newOptions: Record<string, any>,\n) {\n  Object.keys(newOptions).forEach((key) => {\n    if (existingOptions[key] && !isEqual(existingOptions[key], newOptions[key])) {\n      // 이것이 무슨 상황이냐면요, api.options가 덮어씌워지는 상황입니다.\n      // 가령 @api({ resourceName: \"Users\" }) 데코레이터가 붙어있는 메서드에\n      // @stream({ resourceName: \"Posts\" }) 같은 것이 붙어 있는 상황입니다.\n      // 이렇게 되면 두 데코레이터가 같은 api의 options 속 같은 필드를 건드리게 되므로, 에러를 터뜨려줍니다.\n      throw new Error(\n        `@${decoratorName} decorator on ${modelName}.${methodName} has conflicting options: ${key}. The decorator is trying to override the existing option(${JSON.stringify(existingOptions[key])}) with the new option(${JSON.stringify(newOptions[key])}).`,\n      );\n    }\n  });\n}\n"],"mappings":"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"}
310
+ export { api, init_decorators, registeredApis, stream, transactional, upload, websocket };
311
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"decorators.js","names":["api","registeredApis: {\n  /**\n   * modelName은 모델 클래스 이름입니다. (ex. \"UserModel\")\n   */\n  modelName: string;\n  methodName: string;\n  path: string;\n  options: ApiDecoratorOptions;\n  streamOptions?: StreamDecoratorOptions;\n  websocketOptions?: ResolvedWebSocketDecoratorOptions;\n  uploadOptions?: UploadDecoratorOptions;\n}[]"],"sources":["../../src/api/decorators.ts"],"sourcesContent":["import assert from \"assert\";\n\nimport { type FastifyMultipartBaseOptions } from \"@fastify/multipart\";\nimport { getLogger } from \"@logtape/logtape\";\nimport { type HTTPMethods } from \"fastify\";\nimport inflection from \"inflection\";\nimport { isEqual } from \"radashi\";\nimport { type z } from \"zod\";\n\nimport { type CacheControlConfig } from \"../cache-control/types\";\nimport { type CompressConfig } from \"../compress/types\";\nimport { BaseModelClass } from \"../database/base-model\";\nimport { DB } from \"../database/db\";\nimport { PuriTransactionWrapper } from \"../database/puri-wrapper\";\nimport { type TransactionalOptions } from \"../database/puri-wrapper\";\nimport { UpsertBuilder } from \"../database/upsert-builder\";\nimport { convertDomainToCategory } from \"../logger/category\";\nimport { type DriverKey } from \"../storage/drivers\";\nimport { type KeyGenerator } from \"../storage/types\";\nimport { type ApiParam, type ApiParamType } from \"../types/types\";\nimport { BaseFrameClass } from \"./base-frame\";\n\nexport interface GuardKeys {\n  query: true;\n  admin: true;\n  user: true;\n}\nexport type GuardKey = keyof GuardKeys;\nexport type ServiceClient =\n  | \"axios\"\n  | \"axios-multipart\"\n  | \"tanstack-query\"\n  | \"tanstack-mutation\"\n  | \"tanstack-mutation-multipart\"\n  | \"window-fetch\";\nexport type ApiDecoratorOptions = {\n  httpMethod?: HTTPMethods;\n  contentType?:\n    | \"text/plain\"\n    | \"text/html\"\n    | \"text/xml\"\n    | \"application/json\"\n    | \"application/octet-stream\";\n  clients?: ServiceClient[];\n  path?: string;\n  resourceName?: string;\n  guards?: GuardKey[];\n  description?: string;\n  timeout?: number;\n  /** API 응답의 Cache-Control 헤더 설정. 설정하지 않으면 cacheControlHandler 또는 기본값이 적용됩니다. */\n  cacheControl?: CacheControlConfig;\n  /** API 응답의 압축 설정. false로 설정하면 압축을 비활성화합니다. */\n  compress?: CompressConfig;\n};\nexport type StreamDecoratorOptions = {\n  type: \"sse\";\n  // oxlint-disable-next-line @typescript-eslint/no-explicit-any -- 이벤트 키별로 넘겨주는 값이므로 어떤 타입이든 상관없음\n  events: z.ZodObject<any>;\n  path?: string;\n  resourceName?: string;\n  guards?: GuardKey[];\n  description?: string;\n};\nexport type WebSocketDecoratorOptions = {\n  // oxlint-disable-next-line @typescript-eslint/no-explicit-any -- 이벤트 키별로 넘겨주는 값이므로 어떤 타입이든 상관없음\n  outEvents: z.ZodObject<any>;\n  // oxlint-disable-next-line @typescript-eslint/no-explicit-any -- 이벤트 키별로 넘겨주는 값이므로 어떤 타입이든 상관없음\n  inEvents: z.ZodObject<any>;\n  path?: string;\n  resourceName?: string;\n  guards?: GuardKey[];\n  description?: string;\n  heartbeat?: number;\n  maxPayload?: number;\n  namespace?: string;\n};\nexport type ResolvedWebSocketDecoratorOptions = WebSocketDecoratorOptions & {\n  // codegen이 타입 이름을 재사용할 수 있도록 syncer가 AST에서 보강하는 메타데이터\n  outEventsTypeRef?: ApiParamType.Ref;\n  inEventsTypeRef?: ApiParamType.Ref;\n};\n\ntype BufferUploadOptions = {\n  consume?: \"buffer\";\n};\ntype StreamUploadOptions = {\n  consume: \"stream\";\n  destination: DriverKey;\n  keyGenerator?: KeyGenerator;\n};\nexport type UploadDecoratorOptions = {\n  guards?: GuardKey[];\n  description?: string;\n  limits?: FastifyMultipartBaseOptions[\"limits\"];\n} & (BufferUploadOptions | StreamUploadOptions);\n\nexport const registeredApis: {\n  /**\n   * modelName은 모델 클래스 이름입니다. (ex. \"UserModel\")\n   */\n  modelName: string;\n  methodName: string;\n  path: string;\n  options: ApiDecoratorOptions;\n  streamOptions?: StreamDecoratorOptions;\n  websocketOptions?: ResolvedWebSocketDecoratorOptions;\n  uploadOptions?: UploadDecoratorOptions;\n}[] = [];\nexport type ExtendedApi = {\n  modelName: string;\n  methodName: string;\n  path: string;\n  options: ApiDecoratorOptions;\n  streamOptions?: StreamDecoratorOptions;\n  websocketOptions?: ResolvedWebSocketDecoratorOptions;\n  uploadOptions?: UploadDecoratorOptions;\n  typeParameters: ApiParamType.TypeParam[];\n  parameters: ApiParam[];\n  returnType: ApiParamType;\n};\ntype DecoratorTarget = { constructor: { name: string } };\n\nconst DECORATOR_TYPES = {\n  API: Symbol(\"api\"),\n  STREAM: Symbol(\"stream\"),\n  WEBSOCKET: Symbol(\"websocket\"),\n  UPLOAD: Symbol(\"upload\"),\n} as const;\n\nfunction checkSingleDecorator(target: DecoratorTarget, propertyKey: string, decoratorType: symbol) {\n  const method = target[propertyKey as keyof typeof target] as { __decoratorType?: symbol };\n  if (method?.__decoratorType && method?.__decoratorType !== decoratorType) {\n    throw new Error(\n      `@${decoratorType.description ?? String(decoratorType)} decorator can only be used once on ${target.constructor.name}.${propertyKey}. You can use only one of @api, @stream, @websocket, or @upload decorator on the same method.`,\n    );\n  } else {\n    method.__decoratorType = decoratorType;\n  }\n}\n\nexport function api(options: ApiDecoratorOptions = {}) {\n  options = {\n    httpMethod: \"GET\",\n    contentType: \"application/json\",\n    clients: [\"axios\"],\n    ...options,\n  };\n\n  return (target: DecoratorTarget, propertyKey: string, descriptor: PropertyDescriptor) => {\n    const modelName = target.constructor.name.match(/(.+)Class$/)?.[1];\n    assert(\n      modelName,\n      `modelName is required on @api decorator on ${target.constructor.name}.${propertyKey}`,\n    );\n    const methodName = propertyKey;\n\n    // 메서드에 걸린 데코레이터 중복 체크\n    checkSingleDecorator(target, propertyKey, DECORATOR_TYPES.API);\n\n    const defaultPath = `/${inflection.camelize(\n      modelName.replace(/Model$/, \"\").replace(/Frame$/, \"\"),\n      true,\n    )}/${inflection.camelize(propertyKey, true)}`;\n    const path = options.path ?? defaultPath;\n\n    // 기존 동일한 메서드가 있는지 확인 후 있는 경우 override\n    const existingApi = registeredApis.find(\n      (api) => api.modelName === modelName && api.methodName === methodName,\n    );\n    if (existingApi) {\n      // 기존의 path와 새로운 path가 다르다면(=빈 스트링이 아니었는데 다른 스트링으로 바뀌게 된다면) 에러를 터뜨려줍니다.\n      assertNoConflictingPath(\"api\", modelName, methodName, existingApi.path, path);\n      existingApi.path = path;\n\n      // 기존의 옵션과 새로운 옵션이 겹치는 부분이 있다면 에러를 터뜨려줍니다.\n      assertNoConflictingOptions(\"api\", modelName, methodName, existingApi.options, options);\n      existingApi.options = {\n        ...existingApi.options, // 기존의 옵션을 존중하되\n        ...options, // @api 데코레이터의 옵션을 추가합니다.\n      };\n    } else {\n      registeredApis.push({\n        modelName,\n        methodName,\n        path,\n        options,\n      });\n    }\n\n    const originalMethod = descriptor.value;\n    descriptor.value = async function (this: BaseModelClass | BaseFrameClass, ...args: unknown[]) {\n      if (this instanceof BaseModelClass) {\n        getLogger(convertDomainToCategory(this.modelName, \"model\")).debug(\n          \"api: {httpMethod} {model}.{method}\",\n          {\n            httpMethod: options.httpMethod,\n            model: modelName,\n            method: methodName,\n          },\n        );\n      }\n\n      if (this instanceof BaseFrameClass) {\n        getLogger(convertDomainToCategory(this.frameName, \"frame\")).debug(\n          \"api: {httpMethod} {model}.{method}\",\n          {\n            httpMethod: options.httpMethod,\n            model: modelName,\n            method: methodName,\n          },\n        );\n      }\n\n      return originalMethod.apply(this, args);\n    };\n  };\n}\n\nexport function stream(options: StreamDecoratorOptions) {\n  return (target: DecoratorTarget, propertyKey: string, descriptor: PropertyDescriptor) => {\n    const modelName = target.constructor.name.match(/(.+)Class$/)?.[1];\n    assert(\n      modelName,\n      `modelName is required on @stream decorator on ${target.constructor.name}.${propertyKey}`,\n    );\n    const methodName = propertyKey;\n\n    // 메서드에 걸린 데코레이터 중복 체크\n    checkSingleDecorator(target, propertyKey, DECORATOR_TYPES.STREAM);\n\n    const defaultPath = `/${inflection.camelize(\n      modelName.replace(/Model$/, \"\").replace(/Frame$/, \"\"),\n      true,\n    )}/${inflection.camelize(propertyKey, true)}`;\n    const path = options.path ?? defaultPath;\n    // stream 전용 필드(events, type)는 ApiDecoratorOptions에 속하지 않으므로 제외\n    const { events: _, type: _type, ...apiOptions } = options;\n    const optionsWithDefaults = {\n      ...apiOptions,\n      httpMethod: \"GET\" as HTTPMethods,\n    };\n\n    const existingApi = registeredApis.find(\n      (api) => api.modelName === modelName && api.methodName === methodName,\n    );\n    if (existingApi) {\n      // 기존의 path와 새로운 path가 다르다면(=빈 스트링이 아니었는데 다른 스트링으로 바뀌게 된다면) 에러를 터뜨려줍니다.\n      assertNoConflictingPath(\"stream\", modelName, methodName, existingApi.path, path);\n      existingApi.path = path;\n\n      // 기존의 옵션과 새로운 옵션이 겹치는 부분이 있다면 에러를 터뜨려줍니다.\n      assertNoConflictingOptions(\n        \"stream\",\n        modelName,\n        methodName,\n        existingApi.options,\n        optionsWithDefaults,\n      );\n      existingApi.options = {\n        ...existingApi.options, // 기존의 옵션을 존중하되\n        ...optionsWithDefaults, // @stream 데코레이터의 옵션을 추가합니다.\n      };\n\n      existingApi.streamOptions = options;\n    } else {\n      registeredApis.push({\n        modelName,\n        methodName,\n        path,\n        options: optionsWithDefaults,\n        streamOptions: options,\n      });\n    }\n\n    const originalMethod = descriptor.value;\n    descriptor.value = async function (this: BaseModelClass | BaseFrameClass, ...args: unknown[]) {\n      if (this instanceof BaseModelClass) {\n        getLogger(convertDomainToCategory(this.modelName, \"model\")).debug(\n          \"stream: {model}.{method}\",\n          {\n            model: modelName,\n            method: methodName,\n          },\n        );\n      }\n\n      if (this instanceof BaseFrameClass) {\n        getLogger(convertDomainToCategory(this.frameName, \"frame\")).debug(\n          \"stream: {model}.{method}\",\n          {\n            model: modelName,\n            method: methodName,\n          },\n        );\n      }\n\n      return originalMethod.apply(this, args);\n    };\n  };\n}\n\nexport function websocket(options: WebSocketDecoratorOptions) {\n  return (target: DecoratorTarget, propertyKey: string, descriptor: PropertyDescriptor) => {\n    const modelName = target.constructor.name.match(/(.+)Class$/)?.[1];\n    assert(\n      modelName,\n      `modelName is required on @websocket decorator on ${target.constructor.name}.${propertyKey}`,\n    );\n    const methodName = propertyKey;\n\n    checkSingleDecorator(target, propertyKey, DECORATOR_TYPES.WEBSOCKET);\n\n    const defaultPath = `/${inflection.camelize(\n      modelName.replace(/Model$/, \"\").replace(/Frame$/, \"\"),\n      true,\n    )}/${inflection.camelize(propertyKey, true)}`;\n    const path = options.path ?? defaultPath;\n    const { outEvents: _outEvents, inEvents: _inEvents, ...apiOptions } = options;\n    const optionsWithDefaults = {\n      ...apiOptions,\n      httpMethod: \"GET\" as HTTPMethods,\n    };\n\n    const existingApi = registeredApis.find(\n      (api) => api.modelName === modelName && api.methodName === methodName,\n    );\n    if (existingApi) {\n      assertNoConflictingPath(\"websocket\", modelName, methodName, existingApi.path, path);\n      existingApi.path = path;\n\n      assertNoConflictingOptions(\n        \"websocket\",\n        modelName,\n        methodName,\n        existingApi.options,\n        optionsWithDefaults,\n      );\n      existingApi.options = {\n        ...existingApi.options,\n        ...optionsWithDefaults,\n      };\n\n      existingApi.websocketOptions = options;\n    } else {\n      registeredApis.push({\n        modelName,\n        methodName,\n        path,\n        options: optionsWithDefaults,\n        websocketOptions: options,\n      });\n    }\n\n    const originalMethod = descriptor.value;\n    descriptor.value = async function (this: BaseModelClass | BaseFrameClass, ...args: unknown[]) {\n      if (this instanceof BaseModelClass) {\n        getLogger(convertDomainToCategory(this.modelName, \"model\")).debug(\n          \"websocket: {model}.{method}\",\n          {\n            model: modelName,\n            method: methodName,\n          },\n        );\n      }\n\n      if (this instanceof BaseFrameClass) {\n        getLogger(convertDomainToCategory(this.frameName, \"frame\")).debug(\n          \"websocket: {model}.{method}\",\n          {\n            model: modelName,\n            method: methodName,\n          },\n        );\n      }\n\n      return originalMethod.apply(this, args);\n    };\n  };\n}\n\nexport function transactional(options: TransactionalOptions = {}) {\n  const { isolation, readOnly, dbPreset = \"w\" } = options;\n\n  return (target: DecoratorTarget, propertyKey: string, descriptor: PropertyDescriptor) => {\n    const originalMethod = descriptor.value;\n    const modelName = target.constructor.name.match(/(.+)Class$/)?.[1];\n    assert(\n      modelName,\n      `modelName is required on @transactional decorator on ${target.constructor.name}.${propertyKey}`,\n    );\n    const methodName = propertyKey;\n\n    descriptor.value = async function (this: BaseModelClass, ...args: unknown[]) {\n      this.logger.debug(\"transactional: {model}.{method}\", {\n        model: modelName,\n        method: methodName,\n      });\n\n      const existingContext = DB.transactionStorage.getStore();\n\n      // AsyncLocalStorage 컨텍스트 없거나 해당 preset의 트랜잭션이 없으면 새로 시작\n      const startTransaction = async () => {\n        const puri = this.getPuri(dbPreset);\n\n        return puri.knex.transaction(\n          async (trx) => {\n            this.logger.debug(\"new transaction context: {dbPreset}\", { dbPreset });\n            const trxWrapper = new PuriTransactionWrapper(trx, new UpsertBuilder());\n            // TransactionContext에 트랜잭션 저장\n            DB.getTransactionContext().setTransaction(dbPreset, trxWrapper);\n\n            try {\n              return await originalMethod.apply(this, args);\n            } finally {\n              // 트랜잭션 제거\n              this.logger.debug(\"delete transaction context: {dbPreset}\", { dbPreset });\n              DB.getTransactionContext().deleteTransaction(dbPreset);\n            }\n          },\n          { isolationLevel: isolation, readOnly },\n        );\n      };\n\n      // AsyncLocalStorage 컨텍스트가 없으면 새로 생성\n      if (!existingContext) {\n        return DB.runWithTransaction(startTransaction);\n      }\n\n      // 이미 AsyncLocalStorage 컨텍스트 안에 있는지 확인 후 해당 preset의 트랜잭션이 이미 있으면 재사용\n      if (existingContext?.getTransaction(dbPreset)) {\n        this.logger.debug(\"reuse transaction context: {dbPreset}\", { dbPreset });\n        return originalMethod.apply(this, args);\n      }\n\n      // 컨텍스트는 있지만 이 preset의 트랜잭션은 없는 경우 (같은 컨텍스트 내에서 실행)\n      return startTransaction();\n    };\n\n    return descriptor;\n  };\n}\n\n/**\n * 파일 업로드 API를 생성해줍니다. (@api 데코레이터 없이 독립적으로 사용)\n * @param options\n * @returns\n */\nexport function upload(options: UploadDecoratorOptions = { consume: \"buffer\" }) {\n  return (target: DecoratorTarget, propertyKey: string, descriptor: PropertyDescriptor) => {\n    const originalMethod = descriptor.value;\n    const modelName = target.constructor.name.match(/(.+)Class$/)?.[1];\n    assert(\n      modelName,\n      `modelName is required on @upload decorator on ${target.constructor.name}.${propertyKey}`,\n    );\n    const methodName = propertyKey;\n\n    // 메서드에 걸린 데코레이터 중복 체크\n    checkSingleDecorator(target, propertyKey, DECORATOR_TYPES.UPLOAD);\n\n    const defaultPath = `/${inflection.camelize(\n      modelName.replace(/Model$/, \"\").replace(/Frame$/, \"\"),\n      true,\n    )}/${inflection.camelize(methodName, true)}`;\n\n    // registeredApis에서 해당 API 찾아서 uploadOptions 추가\n    const existingApi = registeredApis.find(\n      (api) => api.modelName === modelName && api.methodName === methodName,\n    );\n\n    if (existingApi) {\n      // 재등록 시 업로드 옵션만 갱신\n      existingApi.uploadOptions = options;\n    } else {\n      // 새로 등록\n      registeredApis.push({\n        modelName,\n        methodName,\n        path: defaultPath,\n        options: {\n          httpMethod: \"POST\",\n          clients: [\"axios-multipart\", \"tanstack-mutation-multipart\"],\n          guards: options.guards,\n          description: options.description,\n        },\n        uploadOptions: options,\n      });\n    }\n\n    descriptor.value = async function (this: BaseModelClass | BaseFrameClass, ...args: unknown[]) {\n      if (this instanceof BaseModelClass) {\n        getLogger(convertDomainToCategory(this.modelName, \"model\")).debug(\n          \"upload: {model}.{method}\",\n          {\n            model: modelName,\n            method: methodName,\n          },\n        );\n      }\n\n      if (this instanceof BaseFrameClass) {\n        getLogger(convertDomainToCategory(this.frameName, \"frame\")).debug(\n          \"upload: {model}.{method}\",\n          {\n            model: modelName,\n            method: methodName,\n          },\n        );\n      }\n\n      return originalMethod.apply(this, args);\n    };\n\n    return descriptor;\n  };\n}\n\n/**\n * 기존의 path와 새로운 path가 다르다면(=값이 있던 스트링이 다른 값이 있는 스트링으로 바뀌게 된다면) 에러를 터뜨려줍니다.\n * @param decoratorName 데코레이터 이름\n * @param modelName 모델 이름\n * @param methodName 메서드 이름\n * @param existingPath 기존의 path\n * @param newPath 새로운 path\n */\nfunction assertNoConflictingPath(\n  decoratorName: string,\n  modelName: string,\n  methodName: string,\n  existingPath: string,\n  newPath: string,\n) {\n  if (existingPath !== \"\" && newPath !== \"\" && existingPath !== newPath) {\n    // 이것이 무슨 상황이냐면요, api.path가 덮어씌워지는 상황입니다.\n    // 가령 @api({ path: \"/api/v1/users\" }) 데코레이터가 붙어있는 메서드에\n    // @stream({ path: \"/api/v1/users/stream\" }) 같은 것이 붙어 있는 상황입니다.\n    // 이렇게 되면 두 데코레이터가 같은 api의 path 필드를 건드리게 되므로, 에러를 터뜨려줍니다.\n    throw new Error(\n      `@${decoratorName} decorator on ${modelName}.${methodName} has conflicting path: ${newPath}. The decorator is trying to override the existing path(${existingPath}) with the new path(${newPath}).`,\n    );\n  }\n}\n\n/**\n * 기존의 옵션과 새로운 옵션이 겹치는 부분이 있다면 에러를 터뜨려줍니다.\n * @param decoratorName 데코레이터 이름\n * @param modelName 모델 이름\n * @param methodName 메서드 이름\n * @param existingOptions 기존의 옵션\n * @param newOptions 새로운 옵션\n */\nfunction assertNoConflictingOptions(\n  decoratorName: string,\n  modelName: string,\n  methodName: string,\n  // oxlint-disable-next-line @typescript-eslint/no-explicit-any -- <아 쉽게쉽게 좀 갑시다>\n  existingOptions: Record<string, any>,\n  // oxlint-disable-next-line @typescript-eslint/no-explicit-any -- <이럴 때 아니면 any 언제 씁니까>\n  newOptions: Record<string, any>,\n) {\n  Object.keys(newOptions).forEach((key) => {\n    if (existingOptions[key] && !isEqual(existingOptions[key], newOptions[key])) {\n      // 이것이 무슨 상황이냐면요, api.options가 덮어씌워지는 상황입니다.\n      // 가령 @api({ resourceName: \"Users\" }) 데코레이터가 붙어있는 메서드에\n      // @stream({ resourceName: \"Posts\" }) 같은 것이 붙어 있는 상황입니다.\n      // 이렇게 되면 두 데코레이터가 같은 api의 options 속 같은 필드를 건드리게 되므로, 에러를 터뜨려줍니다.\n      throw new Error(\n        `@${decoratorName} decorator on ${modelName}.${methodName} has conflicting options: ${key}. The decorator is trying to override the existing option(${JSON.stringify(existingOptions[key])}) with the new option(${JSON.stringify(newOptions[key])}).`,\n      );\n    }\n  });\n}\n"],"mappings":";;;;;;;;;;;;;AAiIA,SAAS,qBAAqB,QAAyB,aAAqB,eAAuB;CACjG,MAAM,SAAS,OAAO;AACtB,KAAI,QAAQ,mBAAmB,QAAQ,oBAAoB,eAAe;AACxE,QAAM,IAAI,MACR,IAAI,cAAc,eAAe,OAAO,cAAc,CAAC,sCAAsC,OAAO,YAAY,KAAK,GAAG,YAAY,+FACrI;QACI;AACL,SAAO,kBAAkB;;;AAI7B,SAAgB,IAAI,UAA+B,EAAE,EAAE;AACrD,WAAU;EACR,YAAY;EACZ,aAAa;EACb,SAAS,CAAC,QAAQ;EAClB,GAAG;EACJ;AAED,SAAQ,QAAyB,aAAqB,eAAmC;EACvF,MAAM,YAAY,OAAO,YAAY,KAAK,MAAM,aAAa,GAAG;AAChE,SACE,WACA,8CAA8C,OAAO,YAAY,KAAK,GAAG,cAC1E;EACD,MAAM,aAAa;AAGnB,uBAAqB,QAAQ,aAAa,gBAAgB,IAAI;EAE9D,MAAM,cAAc,IAAI,WAAW,SACjC,UAAU,QAAQ,UAAU,GAAG,CAAC,QAAQ,UAAU,GAAG,EACrD,KACD,CAAC,GAAG,WAAW,SAAS,aAAa,KAAK;EAC3C,MAAM,OAAO,QAAQ,QAAQ;EAG7B,MAAM,cAAc,eAAe,MAChC,UAAQA,MAAI,cAAc,aAAaA,MAAI,eAAe,WAC5D;AACD,MAAI,aAAa;AAEf,2BAAwB,OAAO,WAAW,YAAY,YAAY,MAAM,KAAK;AAC7E,eAAY,OAAO;AAGnB,8BAA2B,OAAO,WAAW,YAAY,YAAY,SAAS,QAAQ;AACtF,eAAY,UAAU;IACpB,GAAG,YAAY;IACf,GAAG;IACJ;SACI;AACL,kBAAe,KAAK;IAClB;IACA;IACA;IACA;IACD,CAAC;;EAGJ,MAAM,iBAAiB,WAAW;AAClC,aAAW,QAAQ,eAAuD,GAAG,MAAiB;AAC5F,OAAI,gBAAgB,gBAAgB;AAClC,cAAU,wBAAwB,KAAK,WAAW,QAAQ,CAAC,CAAC,MAC1D,sCACA;KACE,YAAY,QAAQ;KACpB,OAAO;KACP,QAAQ;KACT,CACF;;AAGH,OAAI,gBAAgB,gBAAgB;AAClC,cAAU,wBAAwB,KAAK,WAAW,QAAQ,CAAC,CAAC,MAC1D,sCACA;KACE,YAAY,QAAQ;KACpB,OAAO;KACP,QAAQ;KACT,CACF;;AAGH,UAAO,eAAe,MAAM,MAAM,KAAK;;;;AAK7C,SAAgB,OAAO,SAAiC;AACtD,SAAQ,QAAyB,aAAqB,eAAmC;EACvF,MAAM,YAAY,OAAO,YAAY,KAAK,MAAM,aAAa,GAAG;AAChE,SACE,WACA,iDAAiD,OAAO,YAAY,KAAK,GAAG,cAC7E;EACD,MAAM,aAAa;AAGnB,uBAAqB,QAAQ,aAAa,gBAAgB,OAAO;EAEjE,MAAM,cAAc,IAAI,WAAW,SACjC,UAAU,QAAQ,UAAU,GAAG,CAAC,QAAQ,UAAU,GAAG,EACrD,KACD,CAAC,GAAG,WAAW,SAAS,aAAa,KAAK;EAC3C,MAAM,OAAO,QAAQ,QAAQ;EAE7B,MAAM,EAAE,QAAQ,GAAG,MAAM,OAAO,GAAG,eAAe;EAClD,MAAM,sBAAsB;GAC1B,GAAG;GACH,YAAY;GACb;EAED,MAAM,cAAc,eAAe,MAChC,UAAQA,MAAI,cAAc,aAAaA,MAAI,eAAe,WAC5D;AACD,MAAI,aAAa;AAEf,2BAAwB,UAAU,WAAW,YAAY,YAAY,MAAM,KAAK;AAChF,eAAY,OAAO;AAGnB,8BACE,UACA,WACA,YACA,YAAY,SACZ,oBACD;AACD,eAAY,UAAU;IACpB,GAAG,YAAY;IACf,GAAG;IACJ;AAED,eAAY,gBAAgB;SACvB;AACL,kBAAe,KAAK;IAClB;IACA;IACA;IACA,SAAS;IACT,eAAe;IAChB,CAAC;;EAGJ,MAAM,iBAAiB,WAAW;AAClC,aAAW,QAAQ,eAAuD,GAAG,MAAiB;AAC5F,OAAI,gBAAgB,gBAAgB;AAClC,cAAU,wBAAwB,KAAK,WAAW,QAAQ,CAAC,CAAC,MAC1D,4BACA;KACE,OAAO;KACP,QAAQ;KACT,CACF;;AAGH,OAAI,gBAAgB,gBAAgB;AAClC,cAAU,wBAAwB,KAAK,WAAW,QAAQ,CAAC,CAAC,MAC1D,4BACA;KACE,OAAO;KACP,QAAQ;KACT,CACF;;AAGH,UAAO,eAAe,MAAM,MAAM,KAAK;;;;AAK7C,SAAgB,UAAU,SAAoC;AAC5D,SAAQ,QAAyB,aAAqB,eAAmC;EACvF,MAAM,YAAY,OAAO,YAAY,KAAK,MAAM,aAAa,GAAG;AAChE,SACE,WACA,oDAAoD,OAAO,YAAY,KAAK,GAAG,cAChF;EACD,MAAM,aAAa;AAEnB,uBAAqB,QAAQ,aAAa,gBAAgB,UAAU;EAEpE,MAAM,cAAc,IAAI,WAAW,SACjC,UAAU,QAAQ,UAAU,GAAG,CAAC,QAAQ,UAAU,GAAG,EACrD,KACD,CAAC,GAAG,WAAW,SAAS,aAAa,KAAK;EAC3C,MAAM,OAAO,QAAQ,QAAQ;EAC7B,MAAM,EAAE,WAAW,YAAY,UAAU,WAAW,GAAG,eAAe;EACtE,MAAM,sBAAsB;GAC1B,GAAG;GACH,YAAY;GACb;EAED,MAAM,cAAc,eAAe,MAChC,UAAQA,MAAI,cAAc,aAAaA,MAAI,eAAe,WAC5D;AACD,MAAI,aAAa;AACf,2BAAwB,aAAa,WAAW,YAAY,YAAY,MAAM,KAAK;AACnF,eAAY,OAAO;AAEnB,8BACE,aACA,WACA,YACA,YAAY,SACZ,oBACD;AACD,eAAY,UAAU;IACpB,GAAG,YAAY;IACf,GAAG;IACJ;AAED,eAAY,mBAAmB;SAC1B;AACL,kBAAe,KAAK;IAClB;IACA;IACA;IACA,SAAS;IACT,kBAAkB;IACnB,CAAC;;EAGJ,MAAM,iBAAiB,WAAW;AAClC,aAAW,QAAQ,eAAuD,GAAG,MAAiB;AAC5F,OAAI,gBAAgB,gBAAgB;AAClC,cAAU,wBAAwB,KAAK,WAAW,QAAQ,CAAC,CAAC,MAC1D,+BACA;KACE,OAAO;KACP,QAAQ;KACT,CACF;;AAGH,OAAI,gBAAgB,gBAAgB;AAClC,cAAU,wBAAwB,KAAK,WAAW,QAAQ,CAAC,CAAC,MAC1D,+BACA;KACE,OAAO;KACP,QAAQ;KACT,CACF;;AAGH,UAAO,eAAe,MAAM,MAAM,KAAK;;;;AAK7C,SAAgB,cAAc,UAAgC,EAAE,EAAE;CAChE,MAAM,EAAE,WAAW,UAAU,WAAW,QAAQ;AAEhD,SAAQ,QAAyB,aAAqB,eAAmC;EACvF,MAAM,iBAAiB,WAAW;EAClC,MAAM,YAAY,OAAO,YAAY,KAAK,MAAM,aAAa,GAAG;AAChE,SACE,WACA,wDAAwD,OAAO,YAAY,KAAK,GAAG,cACpF;EACD,MAAM,aAAa;AAEnB,aAAW,QAAQ,eAAsC,GAAG,MAAiB;AAC3E,QAAK,OAAO,MAAM,mCAAmC;IACnD,OAAO;IACP,QAAQ;IACT,CAAC;GAEF,MAAM,kBAAkB,GAAG,mBAAmB,UAAU;GAGxD,MAAM,mBAAmB,YAAY;IACnC,MAAM,OAAO,KAAK,QAAQ,SAAS;AAEnC,WAAO,KAAK,KAAK,YACf,OAAO,QAAQ;AACb,UAAK,OAAO,MAAM,uCAAuC,EAAE,UAAU,CAAC;KACtE,MAAM,aAAa,IAAI,uBAAuB,KAAK,IAAI,eAAe,CAAC;AAEvE,QAAG,uBAAuB,CAAC,eAAe,UAAU,WAAW;AAE/D,SAAI;AACF,aAAO,MAAM,eAAe,MAAM,MAAM,KAAK;eACrC;AAER,WAAK,OAAO,MAAM,0CAA0C,EAAE,UAAU,CAAC;AACzE,SAAG,uBAAuB,CAAC,kBAAkB,SAAS;;OAG1D;KAAE,gBAAgB;KAAW;KAAU,CACxC;;AAIH,OAAI,CAAC,iBAAiB;AACpB,WAAO,GAAG,mBAAmB,iBAAiB;;AAIhD,OAAI,iBAAiB,eAAe,SAAS,EAAE;AAC7C,SAAK,OAAO,MAAM,yCAAyC,EAAE,UAAU,CAAC;AACxE,WAAO,eAAe,MAAM,MAAM,KAAK;;AAIzC,UAAO,kBAAkB;;AAG3B,SAAO;;;;;;;;AASX,SAAgB,OAAO,UAAkC,EAAE,SAAS,UAAU,EAAE;AAC9E,SAAQ,QAAyB,aAAqB,eAAmC;EACvF,MAAM,iBAAiB,WAAW;EAClC,MAAM,YAAY,OAAO,YAAY,KAAK,MAAM,aAAa,GAAG;AAChE,SACE,WACA,iDAAiD,OAAO,YAAY,KAAK,GAAG,cAC7E;EACD,MAAM,aAAa;AAGnB,uBAAqB,QAAQ,aAAa,gBAAgB,OAAO;EAEjE,MAAM,cAAc,IAAI,WAAW,SACjC,UAAU,QAAQ,UAAU,GAAG,CAAC,QAAQ,UAAU,GAAG,EACrD,KACD,CAAC,GAAG,WAAW,SAAS,YAAY,KAAK;EAG1C,MAAM,cAAc,eAAe,MAChC,UAAQA,MAAI,cAAc,aAAaA,MAAI,eAAe,WAC5D;AAED,MAAI,aAAa;AAEf,eAAY,gBAAgB;SACvB;AAEL,kBAAe,KAAK;IAClB;IACA;IACA,MAAM;IACN,SAAS;KACP,YAAY;KACZ,SAAS,CAAC,mBAAmB,8BAA8B;KAC3D,QAAQ,QAAQ;KAChB,aAAa,QAAQ;KACtB;IACD,eAAe;IAChB,CAAC;;AAGJ,aAAW,QAAQ,eAAuD,GAAG,MAAiB;AAC5F,OAAI,gBAAgB,gBAAgB;AAClC,cAAU,wBAAwB,KAAK,WAAW,QAAQ,CAAC,CAAC,MAC1D,4BACA;KACE,OAAO;KACP,QAAQ;KACT,CACF;;AAGH,OAAI,gBAAgB,gBAAgB;AAClC,cAAU,wBAAwB,KAAK,WAAW,QAAQ,CAAC,CAAC,MAC1D,4BACA;KACE,OAAO;KACP,QAAQ;KACT,CACF;;AAGH,UAAO,eAAe,MAAM,MAAM,KAAK;;AAGzC,SAAO;;;;;;;;;;;AAYX,SAAS,wBACP,eACA,WACA,YACA,cACA,SACA;AACA,KAAI,iBAAiB,MAAM,YAAY,MAAM,iBAAiB,SAAS;AAKrE,QAAM,IAAI,MACR,IAAI,cAAc,gBAAgB,UAAU,GAAG,WAAW,yBAAyB,QAAQ,0DAA0D,aAAa,sBAAsB,QAAQ,IACjM;;;;;;;;;;;AAYL,SAAS,2BACP,eACA,WACA,YAEA,iBAEA,YACA;AACA,QAAO,KAAK,WAAW,CAAC,SAAS,QAAQ;AACvC,MAAI,gBAAgB,QAAQ,CAAC,QAAQ,gBAAgB,MAAM,WAAW,KAAK,EAAE;AAK3E,SAAM,IAAI,MACR,IAAI,cAAc,gBAAgB,UAAU,GAAG,WAAW,4BAA4B,IAAI,4DAA4D,KAAK,UAAU,gBAAgB,KAAK,CAAC,wBAAwB,KAAK,UAAU,WAAW,KAAK,CAAC,IACpP;;GAEH;;;;kBA/iBoD;UACpB;oBAC8B;sBAEP;gBACE;kBAIf;CA4EjCC,iBAWP,EAAE;CAeF,kBAAkB;EACtB,KAAK,OAAO,MAAM;EAClB,QAAQ,OAAO,SAAS;EACxB,WAAW,OAAO,YAAY;EAC9B,QAAQ,OAAO,SAAS;EACzB"}
package/dist/api/index.js CHANGED
@@ -1,10 +1,11 @@
1
1
  import { __esmMin, __exportAll } from "../_virtual/rolldown_runtime.js";
2
2
  import { getSecrets, init_secret } from "./secret.js";
3
+ import { createWebSocketReplyStub, resolveWebSocketCloseDescriptor, resolveWebSocketPluginOptions } from "./websocket-helpers.js";
3
4
  import { Sonamu, init_sonamu } from "./sonamu.js";
4
5
  import { caster, fastifyCaster, init_caster } from "./caster.js";
5
6
  import { apiParamToTsCode, apiParamToTsCodeAsObject, apiParamTypeToTsType, getZodObjectFromApi, getZodObjectFromApiParams, getZodTypeFromApiParamType, init_code_converters, unwrapPromiseOnce } from "./code-converters.js";
6
7
  import { init_context } from "./context.js";
7
- import { api, init_decorators, registeredApis, stream, transactional, upload } from "./decorators.js";
8
+ import { api, init_decorators, registeredApis, stream, transactional, upload, websocket } from "./decorators.js";
8
9
 
9
10
  //#region src/api/index.ts
10
11
  var api_exports = /* @__PURE__ */ __exportAll({
@@ -14,16 +15,20 @@ var api_exports = /* @__PURE__ */ __exportAll({
14
15
  apiParamToTsCodeAsObject: () => apiParamToTsCodeAsObject,
15
16
  apiParamTypeToTsType: () => apiParamTypeToTsType,
16
17
  caster: () => caster,
18
+ createWebSocketReplyStub: () => createWebSocketReplyStub,
17
19
  fastifyCaster: () => fastifyCaster,
18
20
  getSecrets: () => getSecrets,
19
21
  getZodObjectFromApi: () => getZodObjectFromApi,
20
22
  getZodObjectFromApiParams: () => getZodObjectFromApiParams,
21
23
  getZodTypeFromApiParamType: () => getZodTypeFromApiParamType,
22
24
  registeredApis: () => registeredApis,
25
+ resolveWebSocketCloseDescriptor: () => resolveWebSocketCloseDescriptor,
26
+ resolveWebSocketPluginOptions: () => resolveWebSocketPluginOptions,
23
27
  stream: () => stream,
24
28
  transactional: () => transactional,
25
29
  unwrapPromiseOnce: () => unwrapPromiseOnce,
26
- upload: () => upload
30
+ upload: () => upload,
31
+ websocket: () => websocket
27
32
  });
28
33
  var init_api = __esmMin((() => {
29
34
  init_caster();
@@ -36,5 +41,5 @@ var init_api = __esmMin((() => {
36
41
 
37
42
  //#endregion
38
43
  init_api();
39
- export { Sonamu, api, apiParamToTsCode, apiParamToTsCodeAsObject, apiParamTypeToTsType, api_exports, caster, fastifyCaster, getSecrets, getZodObjectFromApi, getZodObjectFromApiParams, getZodTypeFromApiParamType, init_api, registeredApis, stream, transactional, unwrapPromiseOnce, upload };
44
+ export { Sonamu, api, apiParamToTsCode, apiParamToTsCodeAsObject, apiParamTypeToTsType, api_exports, caster, createWebSocketReplyStub, fastifyCaster, getSecrets, getZodObjectFromApi, getZodObjectFromApiParams, getZodTypeFromApiParamType, init_api, registeredApis, resolveWebSocketCloseDescriptor, resolveWebSocketPluginOptions, stream, transactional, unwrapPromiseOnce, upload, websocket };
40
45
  //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJuYW1lcyI6W10sInNvdXJjZXMiOlsiLi4vLi4vc3JjL2FwaS9pbmRleC50cyJdLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tIFwiLi9jYXN0ZXJcIjtcbmV4cG9ydCAqIGZyb20gXCIuL2NvZGUtY29udmVydGVyc1wiO1xuZXhwb3J0ICogZnJvbSBcIi4vY29udGV4dFwiO1xuZXhwb3J0ICogZnJvbSBcIi4vZGVjb3JhdG9yc1wiO1xuZXhwb3J0ICogZnJvbSBcIi4vc2VjcmV0XCI7XG5leHBvcnQgKiBmcm9tIFwiLi9zb25hbXVcIjtcbiJdLCJtYXBwaW5ncyI6IiJ9