sonamu 0.7.17 → 0.7.19

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 (58) hide show
  1. package/dist/api/config.d.ts +19 -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 +3 -3
  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.map +1 -1
  8. package/dist/api/decorators.js +4 -8
  9. package/dist/api/index.d.ts +0 -2
  10. package/dist/api/index.d.ts.map +1 -1
  11. package/dist/api/index.js +1 -3
  12. package/dist/api/sonamu.d.ts +5 -3
  13. package/dist/api/sonamu.d.ts.map +1 -1
  14. package/dist/api/sonamu.js +10 -8
  15. package/dist/index.d.ts +0 -1
  16. package/dist/index.d.ts.map +1 -1
  17. package/dist/index.js +1 -2
  18. package/dist/storage/drivers.d.ts +14 -0
  19. package/dist/storage/drivers.d.ts.map +1 -0
  20. package/dist/storage/drivers.js +11 -0
  21. package/dist/storage/index.d.ts +5 -0
  22. package/dist/storage/index.d.ts.map +1 -0
  23. package/dist/storage/index.js +6 -0
  24. package/dist/storage/storage-manager.d.ts +21 -0
  25. package/dist/storage/storage-manager.d.ts.map +1 -0
  26. package/dist/storage/storage-manager.js +33 -0
  27. package/dist/storage/types.d.ts +12 -0
  28. package/dist/storage/types.d.ts.map +1 -0
  29. package/dist/storage/types.js +5 -0
  30. package/dist/storage/uploaded-file.d.ts +35 -0
  31. package/dist/storage/uploaded-file.d.ts.map +1 -0
  32. package/dist/storage/uploaded-file.js +58 -0
  33. package/dist/template/implementations/services.template.d.ts.map +1 -1
  34. package/dist/template/implementations/services.template.js +23 -2
  35. package/dist/ui-web/assets/{index-DzqUrTB-.js → index-DFqVuxOB.js} +1 -1
  36. package/dist/ui-web/index.html +1 -1
  37. package/package.json +8 -3
  38. package/src/api/config.ts +19 -2
  39. package/src/api/context.ts +3 -3
  40. package/src/api/decorators.ts +3 -8
  41. package/src/api/index.ts +0 -2
  42. package/src/api/sonamu.ts +12 -9
  43. package/src/index.ts +0 -1
  44. package/src/storage/drivers.ts +15 -0
  45. package/src/storage/index.ts +5 -0
  46. package/src/storage/storage-manager.ts +39 -0
  47. package/src/storage/types.ts +12 -0
  48. package/src/storage/uploaded-file.ts +81 -0
  49. package/src/template/implementations/service.template.ts.txt +328 -0
  50. package/src/template/implementations/services.template.ts +40 -1
  51. package/dist/file-storage/driver.d.ts +0 -48
  52. package/dist/file-storage/driver.d.ts.map +0 -1
  53. package/dist/file-storage/driver.js +0 -79
  54. package/dist/file-storage/file-storage.d.ts +0 -50
  55. package/dist/file-storage/file-storage.d.ts.map +0 -1
  56. package/dist/file-storage/file-storage.js +0 -75
  57. package/src/file-storage/driver.ts +0 -131
  58. package/src/file-storage/file-storage.ts +0 -100
@@ -85,10 +85,12 @@ class SonamuClass {
85
85
  return this._secrets;
86
86
  }
87
87
  _storage = null;
88
- set storage(storage) {
89
- this._storage = storage;
90
- }
91
- get storage() {
88
+ /**
89
+ * StorageManager 인스턴스
90
+ */ get storage() {
91
+ if (!this._storage) {
92
+ throw new Error("Storage has not been initialized. Check storage config.");
93
+ }
92
94
  return this._storage;
93
95
  }
94
96
  _workflows = null;
@@ -188,9 +190,10 @@ class SonamuClass {
188
190
  const fastify = (await import("fastify")).default;
189
191
  const server = fastify(options.fastify);
190
192
  this.server = server;
191
- // Storage 설정 저장
193
+ // Storage 설정 → StorageManager 생성
192
194
  if (options.storage) {
193
- this.storage = options.storage;
195
+ const { StorageManager } = await import("../storage/storage-manager.js");
196
+ this._storage = new StorageManager(options.storage);
194
197
  }
195
198
  // 플러그인 등록
196
199
  if (options.plugins) {
@@ -532,9 +535,8 @@ class SonamuClass {
532
535
  await BaseModel.destroy();
533
536
  await this._workflows?.destroy();
534
537
  await this.watcher?.close();
535
- this.storage?.destroy();
536
538
  }
537
539
  }
538
540
  export const Sonamu = new SonamuClass();
539
541
 
540
- //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/api/sonamu.ts"],"sourcesContent":["import assert from \"assert\";\nimport { AsyncLocalStorage } from \"async_hooks\";\nimport type { FSWatcher } from \"chokidar\";\nimport type { FastifyInstance, FastifyReply, FastifyRequest } from \"fastify\";\nimport type { IncomingMessage, Server, ServerResponse } from \"http\";\nimport os from \"os\";\nimport path from \"path\";\nimport type { ZodObject } from \"zod\";\nimport { createMockSSEFactory, DB, isDaemonServer } from \"..\";\nimport type { SonamuDBConfig } from \"../database/db\";\nimport type { Driver } from \"../file-storage/driver\";\nimport { Naite } from \"../naite/naite\";\nimport type { Syncer } from \"../syncer/syncer\";\nimport type { WorkflowManager } from \"../tasks/workflow-manager\";\nimport type { SonamuFastifyConfig } from \"../types/types\";\nimport type { AbsolutePath } from \"../utils/path-utils\";\nimport type { SonamuConfig, SonamuServerOptions, SonamuTaskOptions } from \"./config\";\nimport type { AuthContext, Context, UploadContext } from \"./context\";\nimport type { ExtendedApi } from \"./decorators\";\n\nexport type SonamuSecrets = {\n  anthropic_api_key?: string;\n  voyage_api_key?: string;\n  openai_api_key?: string;\n};\nclass SonamuClass {\n  public isInitialized: boolean = false;\n  public asyncLocalStorage: AsyncLocalStorage<{\n    context: Context;\n  }> = new AsyncLocalStorage();\n\n  public uploadStorage: AsyncLocalStorage<{\n    uploadContext: UploadContext;\n  }> = new AsyncLocalStorage();\n\n  public getContext(): Context {\n    const store = this.asyncLocalStorage.getStore();\n    if (store?.context) {\n      return store.context;\n    }\n\n    if (process.env.NODE_ENV === \"test\") {\n      // 테스팅 환경에서 컨텍스트가 주입되지 않은 경우 빈 컨텍스트 리턴\n      return {\n        request: null,\n        reply: null,\n        headers: {},\n        createSSE: (schema: ZodObject) => createMockSSEFactory(schema),\n        // biome-ignore lint/suspicious/noExplicitAny: 테스팅 환경에서 컨텍스트가 주입되지 않은 경우 빈 컨텍스트 리턴\n        naiteStore: new Map<string, any>(),\n      } as unknown as Context;\n    } else {\n      throw new Error(\"Sonamu cannot find context\");\n    }\n  }\n\n  public getUploadContext(): UploadContext {\n    const store = this.uploadStorage.getStore();\n    if (store?.uploadContext) {\n      return store.uploadContext;\n    }\n    throw new Error(\"Sonamu cannot find upload context. Did you use @upload decorator?\");\n  }\n\n  private _apiRootPath: AbsolutePath | null = null;\n  set apiRootPath(apiRootPath: AbsolutePath) {\n    this._apiRootPath = apiRootPath;\n  }\n  get apiRootPath(): AbsolutePath {\n    if (this._apiRootPath === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n    return this._apiRootPath;\n  }\n  get appRootPath(): string {\n    return this.apiRootPath.split(path.sep).slice(0, -1).join(path.sep);\n  }\n\n  private _dbConfig: SonamuDBConfig | null = null;\n  set dbConfig(dbConfig: SonamuDBConfig) {\n    this._dbConfig = dbConfig;\n  }\n  get dbConfig(): SonamuDBConfig {\n    if (this._dbConfig === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n    return this._dbConfig;\n  }\n\n  private _syncer: Syncer | null = null;\n  set syncer(syncer: Syncer) {\n    this._syncer = syncer;\n  }\n  get syncer(): Syncer {\n    if (this._syncer === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n    return this._syncer;\n  }\n\n  private _config: SonamuConfig | null = null;\n  set config(config: SonamuConfig) {\n    this._config = config;\n  }\n  get config(): SonamuConfig {\n    if (this._config === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n    return this._config;\n  }\n\n  private _secrets: SonamuSecrets | null = null;\n  set secrets(secrets: SonamuSecrets) {\n    this._secrets = secrets;\n  }\n  get secrets(): SonamuSecrets | null {\n    return this._secrets;\n  }\n\n  private _storage: Driver | null = null;\n  set storage(storage: Driver) {\n    this._storage = storage;\n  }\n  get storage(): Driver | null {\n    return this._storage;\n  }\n\n  private _workflows: WorkflowManager | null = null;\n  get workflows(): WorkflowManager {\n    if (this._workflows === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n\n    return this._workflows;\n  }\n\n  // HMR 처리\n  public watcher: FSWatcher | null = null;\n  private pendingFiles: string[] = [];\n  private hmrStartTime: number = 0;\n\n  public server: FastifyInstance | null = null;\n\n  async initForTesting() {\n    await this.init(true, false, undefined, true);\n  }\n\n  async init(\n    doSilent: boolean = false,\n    enableSync: boolean = true,\n    apiRootPath?: AbsolutePath,\n    forTesting: boolean = false,\n  ) {\n    if (this.isInitialized) {\n      return;\n    }\n\n    if (!doSilent) {\n      const chalk = (await import(\"chalk\")).default;\n      console.time(chalk.cyan(`Sonamu.init${forTesting ? \" for testing\" : \"\"}`));\n    }\n\n    // API 루트 패스\n    const { findApiRootPath } = await import(\"../utils/utils\");\n    this.apiRootPath = apiRootPath ?? findApiRootPath();\n\n    const { loadConfig } = await import(\"./config\");\n    this.config = await loadConfig(this.apiRootPath);\n    // sonamu.config.ts 기본값 설정\n    this.config.database.database = this.config.database.database ?? \"postgresql\";\n\n    // API 키 환경변수 로드\n    const secrets: SonamuSecrets = {};\n    if (process.env.ANTHROPIC_API_KEY) {\n      secrets.anthropic_api_key = process.env.ANTHROPIC_API_KEY;\n    }\n    if (process.env.VOYAGE_API_KEY) {\n      secrets.voyage_api_key = process.env.VOYAGE_API_KEY;\n    }\n    if (process.env.OPENAI_API_KEY) {\n      secrets.openai_api_key = process.env.OPENAI_API_KEY;\n    }\n    if (Object.keys(secrets).length > 0) {\n      this.secrets = secrets;\n    }\n\n    // DB 로드\n    const { DB } = await import(\"../database/db\");\n    this.dbConfig = DB.generateDBConfig(this.config.database);\n    if (!doSilent) {\n      const chalk = (await import(\"chalk\")).default;\n      console.log(chalk.green(\"DB Config Loaded!\"));\n    }\n\n    // Entity 로드\n    // 테스트에서도 Entity 정보는 필요합니다.\n    // upsert가 제대로 작동하려면 entity의 unique index 정보가 필요하기 때문입니다.\n    const { EntityManager } = await import(\"../entity/entity-manager\");\n    await EntityManager.autoload(doSilent);\n\n    // 테스팅인 경우 싱크 없이 중단\n    if (forTesting) {\n      this.isInitialized = true;\n      return;\n    }\n\n    // Task 등록\n    await this.initializeWorkflows(this.config.tasks);\n\n    // Syncer\n    const { Syncer } = await import(\"../syncer/syncer\");\n    this.syncer = new Syncer();\n\n    // Autoload: Models / Types / APIs\n    await this.syncer.autoloadTypes();\n    await this.syncer.autoloadModels();\n    await this.syncer.autoloadApis();\n    await this.syncer.autoloadWorkflows();\n\n    const { TemplateManager } = await import(\"../template\");\n    await TemplateManager.autoload();\n\n    const { isLocal, isTest } = await import(\"../utils/controller\");\n    if (isLocal()) {\n      // 로컬에서는 코드 생성을 위해 Biome 셋업이 필요함 (현재 apiRootPath 전달하여 실행)\n      (await import(\"../utils/formatter\")).setupBiome(this.apiRootPath);\n    }\n\n    const { isHotReloadServer } = await import(\"../utils/controller\");\n    if (isLocal() && !isTest() && isHotReloadServer() && enableSync) {\n      await this.syncer.sync();\n\n      await this.startWatcher();\n    }\n\n    this.isInitialized = true;\n    if (!doSilent) {\n      const chalk = (await import(\"chalk\")).default;\n      console.timeEnd(chalk.cyan(\"Sonamu.init\"));\n    }\n  }\n\n  async createServer(initOptions?: { enableSync?: boolean; doSilent?: boolean }) {\n    if (this.isInitialized === false) {\n      await this.init(initOptions?.doSilent, initOptions?.enableSync);\n    }\n\n    const options = this.config.server;\n    const fastify = (await import(\"fastify\")).default;\n    const server = fastify(options.fastify);\n    this.server = server;\n\n    // Storage 설정 저장\n    if (options.storage) {\n      this.storage = options.storage;\n    }\n\n    // 플러그인 등록\n    if (options.plugins) {\n      await this.registerPlugins(server, options.plugins);\n    }\n\n    if (options.auth) {\n      if (!options.plugins?.session) {\n        throw new Error(\"Auth requires session plugin. Please add plugins.session configuration.\");\n      }\n\n      await this.registerAuth(server, options.auth);\n    }\n\n    // API 라우팅 설정\n    await this.withFastify(server, options.apiConfig, {\n      enableSync: initOptions?.enableSync,\n      doSilent: initOptions?.doSilent,\n    });\n\n    // 서버 시작\n    await this.boot(server, options);\n\n    return server;\n  }\n\n  async withFastify(\n    server: FastifyInstance<Server, IncomingMessage, ServerResponse>,\n    config: SonamuFastifyConfig,\n    options?: {\n      enableSync?: boolean;\n      doSilent?: boolean;\n    },\n  ) {\n    if (this.isInitialized === false) {\n      await this.init(options?.doSilent, options?.enableSync);\n    }\n\n    this.server = server;\n\n    // timezone 설정\n    const timezone = this.config.api.timezone;\n    if (timezone) {\n      // 타임존에 맞게 응답 날짜 스트링을 변환해주어야 합니다.\n      // 가령 timezone이 \"Asia/Seoul\" 이면\n      // \"2025-11-21T00:00:00.000Z\" 를 \"2025-11-21T09:00:00+09:00\" 으로 변환해주어야 합니다.\n      const { formatInTimeZone } = await import(\"date-fns-tz\");\n\n      // ISO 8601 날짜 형식 정규식 (예: 2024-01-15T09:30:00.000Z)\n      const ISO_DATE_REGEX = /^\\d{4}-\\d{2}-\\d{2}T\\d{2}:\\d{2}:\\d{2}(\\.\\d{3})?Z$/;\n\n      // T를 둘러싼 작은따옴표가 없다면 \"2025-11-19176354618900018:56:29+09:00\"와 같은 결과가 나옵니다.\n      // 이는 date-fns 특입니다.\n      // 이렇게 해도 괜찮습니다. \"2025-11-19T18:56:29+09:00\" 모양으로 잘 나옵니다.\n      const DATE_FORMAT = \"yyyy-MM-dd'T'HH:mm:ssXXX\";\n\n      server.setReplySerializer((payload) => {\n        return JSON.stringify(payload, (_key, value) => {\n          if (typeof value === \"string\" && ISO_DATE_REGEX.test(value)) {\n            return formatInTimeZone(\n              new Date(value),\n              timezone as `${string}/${string}`,\n              DATE_FORMAT,\n            );\n          }\n          return value;\n        });\n      });\n      if (!options?.doSilent) {\n        const chalk = (await import(\"chalk\")).default;\n        console.log(chalk.green(`Timezone set to ${timezone}`));\n      }\n    }\n\n    // 전체 라우팅 리스트\n    server.get(\n      `${this.config.api.route.prefix}/routes`,\n      async (_request, _reply): Promise<typeof this.syncer.apis> => {\n        return this.syncer.apis;\n      },\n    );\n\n    // Healthcheck API\n    server.get(\n      `${this.config.api.route.prefix}/healthcheck`,\n      async (_request, _reply): Promise<string> => {\n        return \"ok\";\n      },\n    );\n\n    // Sonamu UI API\n    const { sonamuUIApiPlugin } = await import(\"../ui/api\");\n    server.register(sonamuUIApiPlugin);\n\n    // API 라우팅 (로컬HMR 상태와 구분)\n    const { isLocal } = await import(\"../utils/controller\");\n    if (isLocal()) {\n      server.all(\"*\", async (request, reply) => {\n        // Sonamu UI\n        if (request.url.startsWith(\"/sonamu-ui\")) {\n          return;\n        }\n\n        const found = this.syncer.apis.find(\n          (api) =>\n            this.config.api.route.prefix + api.path === request.url.split(\"?\")[0] &&\n            (api.options.httpMethod ?? \"GET\") === request.method.toUpperCase(),\n        );\n        if (found) {\n          return this.createApiHandler(found, config)(request, reply);\n        }\n\n        if (request.url.startsWith(\"/api/\")) {\n          const { NotFoundException } = await import(\"../exceptions/so-exceptions\");\n          throw new NotFoundException(`존재하지 않는 API 접근입니다. ${request.url}`);\n        }\n\n        // 일반 파일 접근시 별도의 에러 출력하지 않음\n        return;\n      });\n    } else {\n      for (const api of this.syncer.apis) {\n        // model\n        if (this.syncer.models[api.modelName] === undefined) {\n          throw new Error(`정의되지 않은 모델에 접근 ${api.modelName}`);\n        }\n\n        // route\n        server.route({\n          method: api.options.httpMethod ?? \"GET\",\n          url: this.config.api.route.prefix + api.path,\n          handler: this.createApiHandler(api, config),\n        }); // END server.route\n      }\n    }\n  }\n\n  createApiHandler(\n    api: ExtendedApi,\n    config: SonamuFastifyConfig,\n  ): (request: FastifyRequest, reply: FastifyReply) => Promise<unknown> {\n    return async (request: FastifyRequest, reply: FastifyReply): Promise<unknown> => {\n      (api.options.guards ?? []).every((guard) => config.guardHandler(guard, request, api));\n\n      // 파라미터 정보로 zod 스키마 빌드\n      const { getZodObjectFromApi } = await import(\"./code-converters\");\n      const ReqType = getZodObjectFromApi(api, this.syncer.types);\n\n      // request 파싱\n      const which = api.options.httpMethod === \"GET\" ? \"query\" : \"body\";\n      let reqBody: {\n        [key: string]: unknown;\n      };\n      try {\n        const { fastifyCaster } = await import(\"./caster\");\n        reqBody = fastifyCaster(ReqType).parse(request[which] ?? {});\n      } catch (e) {\n        const { ZodError } = await import(\"zod\");\n        if (e instanceof ZodError) {\n          const { humanizeZodError } = await import(\"../utils/zod-error\");\n          const messages = humanizeZodError(e)\n            .map((issue) => issue.message)\n            .join(\" \");\n          const { BadRequestException } = await import(\"../exceptions/so-exceptions\");\n          throw new BadRequestException(messages, {\n            zodError: e,\n          });\n        } else {\n          throw e;\n        }\n      }\n\n      // Content-Type\n      reply.type(api.options.contentType ?? \"application/json\");\n\n      // Context 생성\n      const context: Context = await this.createContext(config, request, reply);\n\n      // 모델 메소드 args 생성하여 호출\n      const { ApiParamType } = await import(\"../types/types\");\n      const args = api.parameters.map((param) => {\n        // Context 인젝션\n        if (ApiParamType.isContext(param.type)) {\n          return context;\n        } else {\n          return reqBody[param.name];\n        }\n      });\n      return this.invokeModelMethod(api, args, context, reply);\n    };\n  }\n\n  async invokeModelMethod(\n    api: ExtendedApi,\n    args: unknown[],\n    context: Context,\n    reply: FastifyReply,\n  ): Promise<unknown> {\n    const model = this.syncer.models[api.modelName];\n    return this.asyncLocalStorage.run({ context }, async () => {\n      // biome-ignore lint/suspicious/noExplicitAny: model은 모델 인스턴스이므로 메서드 호출 가능\n      const result = await (model as any)[api.methodName].apply(model, args);\n      reply.type(api.options.contentType ?? \"application/json\");\n\n      return result;\n    });\n  }\n\n  async createContext(\n    config: SonamuFastifyConfig,\n    request: FastifyRequest,\n    reply: FastifyReply,\n  ): Promise<Context> {\n    // createSSEFactory 함수에 미리 request의 socket과 reply를 바인딩.\n    const { createSSEFactory } = await import(\"../stream/sse\");\n    const createSSE = (<T extends ZodObject>(\n      _request: FastifyRequest,\n      _reply: FastifyReply,\n      _events: T,\n    ) => createSSEFactory(_request.socket, _reply, _events)).bind(null, request, reply);\n\n    const context: Context = {\n      ...(await Promise.resolve(\n        config.contextProvider(\n          {\n            request,\n            reply,\n            headers: request.headers,\n            createSSE,\n            naiteStore: Naite.createStore(),\n            // auth\n            user: request.user ?? null,\n            passport: {\n              login: request.login.bind(request) as AuthContext[\"passport\"][\"login\"],\n              logout: request.logout.bind(request) as AuthContext[\"passport\"][\"logout\"],\n            },\n          },\n          request,\n          reply,\n        ),\n      )),\n    };\n    return context;\n  }\n\n  async startWatcher(): Promise<void> {\n    const watchPath = [path.join(this.apiRootPath, \"src\")];\n\n    const chokidar = (await import(\"chokidar\")).default;\n    this.watcher = chokidar.watch(watchPath, {\n      ignored: (path, stats) =>\n        !!stats?.isFile() && !path.endsWith(\".ts\") && !path.endsWith(\".json\"),\n      persistent: true,\n      ignoreInitial: true,\n    });\n\n    this.watcher.on(\"all\", async (event: string, filePath: string) => {\n      const absolutePath = filePath as AbsolutePath;\n      assert(\n        absolutePath.startsWith(this.apiRootPath),\n        \"File path is not within the API root path\",\n      );\n\n      if (event !== \"change\" && event !== \"add\") {\n        return;\n      }\n\n      try {\n        // sonamu.config.ts 변경 시 재시작\n        const isConfigTs = filePath === path.join(this.apiRootPath, \"src\", \"sonamu.config.ts\");\n\n        if (isConfigTs) {\n          const relativePath = filePath.replace(this.apiRootPath, \"api\");\n          const chalk = (await import(\"chalk\")).default;\n          console.log(\n            chalk.bold(`Detected(${event}): ${chalk.blue(relativePath)} - Restarting...`),\n          );\n          process.kill(process.pid, \"SIGUSR2\");\n          return;\n        }\n\n        await this.handleFileChange(event, absolutePath);\n      } catch (e) {\n        console.error(e);\n      }\n    });\n  }\n\n  /*\n     A function that automatically handles init and destroy when using Sonamu via scripts.    \n  */\n  async runScript(fn: () => Promise<void>) {\n    await this.init(true, false, undefined, false);\n    try {\n      await fn();\n    } finally {\n      await this.destroy();\n    }\n  }\n\n  private async registerPlugins(server: FastifyInstance, plugins: SonamuServerOptions[\"plugins\"]) {\n    if (!plugins) {\n      return;\n    }\n\n    const pluginsModules = {\n      cors: \"@fastify/cors\",\n      formbody: \"@fastify/formbody\",\n      multipart: \"@fastify/multipart\",\n      qs: \"fastify-qs\",\n      sse: \"fastify-sse-v2\",\n      static: \"@fastify/static\",\n      session: \"@fastify/secure-session\",\n    } as const;\n\n    const registerPlugin = async <K extends keyof NonNullable<typeof plugins>>(\n      key: K,\n      pluginName: string,\n    ) => {\n      const option = plugins[key];\n      if (!option) return;\n\n      if (option === true) {\n        server.register((await import(pluginName)).default);\n      } else {\n        server.register((await import(pluginName)).default, option);\n      }\n    };\n\n    for (const [key, pluginName] of Object.entries(pluginsModules)) {\n      await registerPlugin(key as keyof typeof plugins, pluginName);\n    }\n\n    if (plugins.custom) {\n      plugins.custom(server);\n    }\n  }\n\n  private async registerAuth(\n    server: FastifyInstance,\n    options: NonNullable<SonamuServerOptions[\"auth\"]>,\n  ) {\n    // await import(\"fastify\");\n    const fastifyPassport = (await import(\"@fastify/passport\")).default;\n    server.register(fastifyPassport.initialize());\n    server.register(fastifyPassport.secureSession());\n\n    if (typeof options === \"boolean\") {\n      fastifyPassport.registerUserSerializer(async (user, _request) => user);\n      fastifyPassport.registerUserDeserializer(async (serialized, _request) => serialized);\n    } else {\n      fastifyPassport.registerUserSerializer(options.userSerializer);\n      fastifyPassport.registerUserDeserializer(options.userDeserializer);\n    }\n  }\n\n  private async initializeWorkflows(options: SonamuTaskOptions | undefined) {\n    const { WorkflowManager } = await import(\"../tasks/workflow-manager\");\n    // NOTE: @sonamu-kit/tasks 안에선 knex config를 수정하기 때문에 connection이 아닌 config 째로 보냅니다.\n    this._workflows = new WorkflowManager(DB.getDBConfig(\"w\"));\n    if (!options) {\n      return;\n    }\n\n    const enableWorker = options.enableWorker ?? isDaemonServer();\n    const defaultWorkerOptions = {\n      concurrency: os.cpus().length - 1,\n      usePubSub: true,\n      listenDelay: 500,\n    };\n\n    if (enableWorker) {\n      this.workflows.setupWorker({\n        ...defaultWorkerOptions,\n        ...options.workerOptions,\n      });\n    }\n  }\n\n  private async boot(server: FastifyInstance, options: SonamuServerOptions) {\n    const port = options.listen?.port ?? 3000;\n    const host = options.listen?.host ?? \"localhost\";\n\n    server.addHook(\"onClose\", async () => {\n      await options.lifecycle?.onShutdown?.(server);\n      await this.workflows.destroy();\n      await this.destroy();\n    });\n\n    const shutdown = async () => {\n      try {\n        await server.close();\n        process.exit(0);\n      } catch (err) {\n        console.error(\"Error during shutdown:\", err);\n        process.exit(1);\n      }\n    };\n\n    process.on(\"SIGINT\", shutdown);\n    process.on(\"SIGTERM\", shutdown);\n\n    if (options.lifecycle?.onError) {\n      server.setErrorHandler(options.lifecycle?.onError);\n    }\n\n    server\n      .listen({ port, host })\n      .then(async () => {\n        await this.workflows.startWorker();\n        await options.lifecycle?.onStart?.(server);\n      })\n      .catch(async (err) => {\n        const chalk = (await import(\"chalk\")).default;\n        console.error(chalk.red(\"Failed to start server:\", err));\n        await shutdown();\n      });\n  }\n\n  private async handleFileChange(event: string, filePath: AbsolutePath): Promise<void> {\n    // 첫 번째 파일이면 HMR 시작 시간 기록\n    if (this.pendingFiles.length === 0) {\n      this.hmrStartTime = Date.now();\n    }\n    this.pendingFiles.push(filePath);\n\n    const relativePath = path.relative(this.apiRootPath, filePath);\n    const chalk = (await import(\"chalk\")).default;\n    console.log(chalk.bold(`Detected(${event}): ${chalk.blue(relativePath)}`));\n\n    await this.syncer.syncFromWatcher(event, filePath);\n\n    // 처리 완료된 파일을 대기 목록에서 제거\n    this.pendingFiles = this.pendingFiles.slice(1);\n\n    // 모든 파일 처리가 완료되면 최종 메시지 출력\n    if (this.pendingFiles.length === 0) {\n      await this.finishHMR();\n    }\n  }\n\n  private async finishHMR(): Promise<void> {\n    await this.syncer.renewChecksums();\n\n    const endTime = Date.now();\n    const totalTime = endTime - this.hmrStartTime;\n    const [chalk, { centerText }] = await Promise.all([\n      (await import(\"chalk\")).default,\n      import(\"../utils/console-util\"),\n    ]);\n    const msg = `HMR Done! ${chalk.bold.white(`${totalTime}ms`)}`;\n\n    console.log(chalk.black.bgGreen(centerText(msg)));\n  }\n\n  async destroy(): Promise<void> {\n    const { BaseModel } = await import(\"../database/base-model\");\n    await BaseModel.destroy();\n    await this._workflows?.destroy();\n    await this.watcher?.close();\n    this.storage?.destroy();\n  }\n}\nexport const Sonamu = new SonamuClass();\n"],"names":["assert","AsyncLocalStorage","os","path","createMockSSEFactory","DB","isDaemonServer","Naite","SonamuClass","isInitialized","asyncLocalStorage","uploadStorage","getContext","store","getStore","context","process","env","NODE_ENV","request","reply","headers","createSSE","schema","naiteStore","Map","Error","getUploadContext","uploadContext","_apiRootPath","apiRootPath","appRootPath","split","sep","slice","join","_dbConfig","dbConfig","_syncer","syncer","_config","config","_secrets","secrets","_storage","storage","_workflows","workflows","watcher","pendingFiles","hmrStartTime","server","initForTesting","init","undefined","doSilent","enableSync","forTesting","chalk","default","console","time","cyan","findApiRootPath","loadConfig","database","ANTHROPIC_API_KEY","anthropic_api_key","VOYAGE_API_KEY","voyage_api_key","OPENAI_API_KEY","openai_api_key","Object","keys","length","generateDBConfig","log","green","EntityManager","autoload","initializeWorkflows","tasks","Syncer","autoloadTypes","autoloadModels","autoloadApis","autoloadWorkflows","TemplateManager","isLocal","isTest","setupBiome","isHotReloadServer","sync","startWatcher","timeEnd","createServer","initOptions","options","fastify","plugins","registerPlugins","auth","session","registerAuth","withFastify","apiConfig","boot","timezone","api","formatInTimeZone","ISO_DATE_REGEX","DATE_FORMAT","setReplySerializer","payload","JSON","stringify","_key","value","test","Date","get","route","prefix","_request","_reply","apis","sonamuUIApiPlugin","register","all","url","startsWith","found","find","httpMethod","method","toUpperCase","createApiHandler","NotFoundException","models","modelName","handler","guards","every","guard","guardHandler","getZodObjectFromApi","ReqType","types","which","reqBody","fastifyCaster","parse","e","ZodError","humanizeZodError","messages","map","issue","message","BadRequestException","zodError","type","contentType","createContext","ApiParamType","args","parameters","param","isContext","name","invokeModelMethod","model","run","result","methodName","apply","createSSEFactory","_events","socket","bind","Promise","resolve","contextProvider","createStore","user","passport","login","logout","watchPath","chokidar","watch","ignored","stats","isFile","endsWith","persistent","ignoreInitial","on","event","filePath","absolutePath","isConfigTs","relativePath","replace","bold","blue","kill","pid","handleFileChange","error","runScript","fn","destroy","pluginsModules","cors","formbody","multipart","qs","sse","static","registerPlugin","key","pluginName","option","entries","custom","fastifyPassport","initialize","secureSession","registerUserSerializer","registerUserDeserializer","serialized","userSerializer","userDeserializer","WorkflowManager","getDBConfig","enableWorker","defaultWorkerOptions","concurrency","cpus","usePubSub","listenDelay","setupWorker","workerOptions","port","listen","host","addHook","lifecycle","onShutdown","shutdown","close","exit","err","onError","setErrorHandler","then","startWorker","onStart","catch","red","now","push","relative","syncFromWatcher","finishHMR","renewChecksums","endTime","totalTime","centerText","msg","white","black","bgGreen","BaseModel","Sonamu"],"mappings":"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"}
542
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/api/sonamu.ts"],"sourcesContent":["import assert from \"assert\";\nimport { AsyncLocalStorage } from \"async_hooks\";\nimport type { FSWatcher } from \"chokidar\";\nimport type { FastifyInstance, FastifyReply, FastifyRequest } from \"fastify\";\nimport type { IncomingMessage, Server, ServerResponse } from \"http\";\nimport os from \"os\";\nimport path from \"path\";\nimport type { ZodObject } from \"zod\";\nimport { createMockSSEFactory, DB, isDaemonServer } from \"..\";\nimport type { SonamuDBConfig } from \"../database/db\";\nimport { Naite } from \"../naite/naite\";\nimport type { StorageManager } from \"../storage/storage-manager\";\nimport type { Syncer } from \"../syncer/syncer\";\nimport type { WorkflowManager } from \"../tasks/workflow-manager\";\nimport type { SonamuFastifyConfig } from \"../types/types\";\nimport type { AbsolutePath } from \"../utils/path-utils\";\nimport type { SonamuConfig, SonamuServerOptions, SonamuTaskOptions } from \"./config\";\nimport type { AuthContext, Context, UploadContext } from \"./context\";\nimport type { ExtendedApi } from \"./decorators\";\n\nexport type SonamuSecrets = {\n  anthropic_api_key?: string;\n  voyage_api_key?: string;\n  openai_api_key?: string;\n};\nclass SonamuClass {\n  public isInitialized: boolean = false;\n  public asyncLocalStorage: AsyncLocalStorage<{\n    context: Context;\n  }> = new AsyncLocalStorage();\n\n  public uploadStorage: AsyncLocalStorage<{\n    uploadContext: UploadContext;\n  }> = new AsyncLocalStorage();\n\n  public getContext(): Context {\n    const store = this.asyncLocalStorage.getStore();\n    if (store?.context) {\n      return store.context;\n    }\n\n    if (process.env.NODE_ENV === \"test\") {\n      // 테스팅 환경에서 컨텍스트가 주입되지 않은 경우 빈 컨텍스트 리턴\n      return {\n        request: null,\n        reply: null,\n        headers: {},\n        createSSE: (schema: ZodObject) => createMockSSEFactory(schema),\n        // biome-ignore lint/suspicious/noExplicitAny: 테스팅 환경에서 컨텍스트가 주입되지 않은 경우 빈 컨텍스트 리턴\n        naiteStore: new Map<string, any>(),\n      } as unknown as Context;\n    } else {\n      throw new Error(\"Sonamu cannot find context\");\n    }\n  }\n\n  public getUploadContext(): UploadContext {\n    const store = this.uploadStorage.getStore();\n    if (store?.uploadContext) {\n      return store.uploadContext;\n    }\n    throw new Error(\"Sonamu cannot find upload context. Did you use @upload decorator?\");\n  }\n\n  private _apiRootPath: AbsolutePath | null = null;\n  set apiRootPath(apiRootPath: AbsolutePath) {\n    this._apiRootPath = apiRootPath;\n  }\n  get apiRootPath(): AbsolutePath {\n    if (this._apiRootPath === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n    return this._apiRootPath;\n  }\n  get appRootPath(): string {\n    return this.apiRootPath.split(path.sep).slice(0, -1).join(path.sep);\n  }\n\n  private _dbConfig: SonamuDBConfig | null = null;\n  set dbConfig(dbConfig: SonamuDBConfig) {\n    this._dbConfig = dbConfig;\n  }\n  get dbConfig(): SonamuDBConfig {\n    if (this._dbConfig === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n    return this._dbConfig;\n  }\n\n  private _syncer: Syncer | null = null;\n  set syncer(syncer: Syncer) {\n    this._syncer = syncer;\n  }\n  get syncer(): Syncer {\n    if (this._syncer === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n    return this._syncer;\n  }\n\n  private _config: SonamuConfig | null = null;\n  set config(config: SonamuConfig) {\n    this._config = config;\n  }\n  get config(): SonamuConfig {\n    if (this._config === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n    return this._config;\n  }\n\n  private _secrets: SonamuSecrets | null = null;\n  set secrets(secrets: SonamuSecrets) {\n    this._secrets = secrets;\n  }\n  get secrets(): SonamuSecrets | null {\n    return this._secrets;\n  }\n\n  private _storage: StorageManager | null = null;\n  /**\n   * StorageManager 인스턴스\n   */\n  get storage(): StorageManager {\n    if (!this._storage) {\n      throw new Error(\"Storage has not been initialized. Check storage config.\");\n    }\n    return this._storage;\n  }\n\n  private _workflows: WorkflowManager | null = null;\n  get workflows(): WorkflowManager {\n    if (this._workflows === null) {\n      throw new Error(\"Sonamu has not been initialized\");\n    }\n\n    return this._workflows;\n  }\n\n  // HMR 처리\n  public watcher: FSWatcher | null = null;\n  private pendingFiles: string[] = [];\n  private hmrStartTime: number = 0;\n\n  public server: FastifyInstance | null = null;\n\n  async initForTesting() {\n    await this.init(true, false, undefined, true);\n  }\n\n  async init(\n    doSilent: boolean = false,\n    enableSync: boolean = true,\n    apiRootPath?: AbsolutePath,\n    forTesting: boolean = false,\n  ) {\n    if (this.isInitialized) {\n      return;\n    }\n\n    if (!doSilent) {\n      const chalk = (await import(\"chalk\")).default;\n      console.time(chalk.cyan(`Sonamu.init${forTesting ? \" for testing\" : \"\"}`));\n    }\n\n    // API 루트 패스\n    const { findApiRootPath } = await import(\"../utils/utils\");\n    this.apiRootPath = apiRootPath ?? findApiRootPath();\n\n    const { loadConfig } = await import(\"./config\");\n    this.config = await loadConfig(this.apiRootPath);\n    // sonamu.config.ts 기본값 설정\n    this.config.database.database = this.config.database.database ?? \"postgresql\";\n\n    // API 키 환경변수 로드\n    const secrets: SonamuSecrets = {};\n    if (process.env.ANTHROPIC_API_KEY) {\n      secrets.anthropic_api_key = process.env.ANTHROPIC_API_KEY;\n    }\n    if (process.env.VOYAGE_API_KEY) {\n      secrets.voyage_api_key = process.env.VOYAGE_API_KEY;\n    }\n    if (process.env.OPENAI_API_KEY) {\n      secrets.openai_api_key = process.env.OPENAI_API_KEY;\n    }\n    if (Object.keys(secrets).length > 0) {\n      this.secrets = secrets;\n    }\n\n    // DB 로드\n    const { DB } = await import(\"../database/db\");\n    this.dbConfig = DB.generateDBConfig(this.config.database);\n    if (!doSilent) {\n      const chalk = (await import(\"chalk\")).default;\n      console.log(chalk.green(\"DB Config Loaded!\"));\n    }\n\n    // Entity 로드\n    // 테스트에서도 Entity 정보는 필요합니다.\n    // upsert가 제대로 작동하려면 entity의 unique index 정보가 필요하기 때문입니다.\n    const { EntityManager } = await import(\"../entity/entity-manager\");\n    await EntityManager.autoload(doSilent);\n\n    // 테스팅인 경우 싱크 없이 중단\n    if (forTesting) {\n      this.isInitialized = true;\n      return;\n    }\n\n    // Task 등록\n    await this.initializeWorkflows(this.config.tasks);\n\n    // Syncer\n    const { Syncer } = await import(\"../syncer/syncer\");\n    this.syncer = new Syncer();\n\n    // Autoload: Models / Types / APIs\n    await this.syncer.autoloadTypes();\n    await this.syncer.autoloadModels();\n    await this.syncer.autoloadApis();\n    await this.syncer.autoloadWorkflows();\n\n    const { TemplateManager } = await import(\"../template\");\n    await TemplateManager.autoload();\n\n    const { isLocal, isTest } = await import(\"../utils/controller\");\n    if (isLocal()) {\n      // 로컬에서는 코드 생성을 위해 Biome 셋업이 필요함 (현재 apiRootPath 전달하여 실행)\n      (await import(\"../utils/formatter\")).setupBiome(this.apiRootPath);\n    }\n\n    const { isHotReloadServer } = await import(\"../utils/controller\");\n    if (isLocal() && !isTest() && isHotReloadServer() && enableSync) {\n      await this.syncer.sync();\n\n      await this.startWatcher();\n    }\n\n    this.isInitialized = true;\n    if (!doSilent) {\n      const chalk = (await import(\"chalk\")).default;\n      console.timeEnd(chalk.cyan(\"Sonamu.init\"));\n    }\n  }\n\n  async createServer(initOptions?: { enableSync?: boolean; doSilent?: boolean }) {\n    if (this.isInitialized === false) {\n      await this.init(initOptions?.doSilent, initOptions?.enableSync);\n    }\n\n    const options = this.config.server;\n    const fastify = (await import(\"fastify\")).default;\n    const server = fastify(options.fastify);\n    this.server = server;\n\n    // Storage 설정 → StorageManager 생성\n    if (options.storage) {\n      const { StorageManager } = await import(\"../storage/storage-manager\");\n      this._storage = new StorageManager(options.storage);\n    }\n\n    // 플러그인 등록\n    if (options.plugins) {\n      await this.registerPlugins(server, options.plugins);\n    }\n\n    if (options.auth) {\n      if (!options.plugins?.session) {\n        throw new Error(\"Auth requires session plugin. Please add plugins.session configuration.\");\n      }\n\n      await this.registerAuth(server, options.auth);\n    }\n\n    // API 라우팅 설정\n    await this.withFastify(server, options.apiConfig, {\n      enableSync: initOptions?.enableSync,\n      doSilent: initOptions?.doSilent,\n    });\n\n    // 서버 시작\n    await this.boot(server, options);\n\n    return server;\n  }\n\n  async withFastify(\n    server: FastifyInstance<Server, IncomingMessage, ServerResponse>,\n    config: SonamuFastifyConfig,\n    options?: {\n      enableSync?: boolean;\n      doSilent?: boolean;\n    },\n  ) {\n    if (this.isInitialized === false) {\n      await this.init(options?.doSilent, options?.enableSync);\n    }\n\n    this.server = server;\n\n    // timezone 설정\n    const timezone = this.config.api.timezone;\n    if (timezone) {\n      // 타임존에 맞게 응답 날짜 스트링을 변환해주어야 합니다.\n      // 가령 timezone이 \"Asia/Seoul\" 이면\n      // \"2025-11-21T00:00:00.000Z\" 를 \"2025-11-21T09:00:00+09:00\" 으로 변환해주어야 합니다.\n      const { formatInTimeZone } = await import(\"date-fns-tz\");\n\n      // ISO 8601 날짜 형식 정규식 (예: 2024-01-15T09:30:00.000Z)\n      const ISO_DATE_REGEX = /^\\d{4}-\\d{2}-\\d{2}T\\d{2}:\\d{2}:\\d{2}(\\.\\d{3})?Z$/;\n\n      // T를 둘러싼 작은따옴표가 없다면 \"2025-11-19176354618900018:56:29+09:00\"와 같은 결과가 나옵니다.\n      // 이는 date-fns 특입니다.\n      // 이렇게 해도 괜찮습니다. \"2025-11-19T18:56:29+09:00\" 모양으로 잘 나옵니다.\n      const DATE_FORMAT = \"yyyy-MM-dd'T'HH:mm:ssXXX\";\n\n      server.setReplySerializer((payload) => {\n        return JSON.stringify(payload, (_key, value) => {\n          if (typeof value === \"string\" && ISO_DATE_REGEX.test(value)) {\n            return formatInTimeZone(\n              new Date(value),\n              timezone as `${string}/${string}`,\n              DATE_FORMAT,\n            );\n          }\n          return value;\n        });\n      });\n      if (!options?.doSilent) {\n        const chalk = (await import(\"chalk\")).default;\n        console.log(chalk.green(`Timezone set to ${timezone}`));\n      }\n    }\n\n    // 전체 라우팅 리스트\n    server.get(\n      `${this.config.api.route.prefix}/routes`,\n      async (_request, _reply): Promise<typeof this.syncer.apis> => {\n        return this.syncer.apis;\n      },\n    );\n\n    // Healthcheck API\n    server.get(\n      `${this.config.api.route.prefix}/healthcheck`,\n      async (_request, _reply): Promise<string> => {\n        return \"ok\";\n      },\n    );\n\n    // Sonamu UI API\n    const { sonamuUIApiPlugin } = await import(\"../ui/api\");\n    server.register(sonamuUIApiPlugin);\n\n    // API 라우팅 (로컬HMR 상태와 구분)\n    const { isLocal } = await import(\"../utils/controller\");\n    if (isLocal()) {\n      server.all(\"*\", async (request, reply) => {\n        // Sonamu UI\n        if (request.url.startsWith(\"/sonamu-ui\")) {\n          return;\n        }\n\n        const found = this.syncer.apis.find(\n          (api) =>\n            this.config.api.route.prefix + api.path === request.url.split(\"?\")[0] &&\n            (api.options.httpMethod ?? \"GET\") === request.method.toUpperCase(),\n        );\n        if (found) {\n          return this.createApiHandler(found, config)(request, reply);\n        }\n\n        if (request.url.startsWith(\"/api/\")) {\n          const { NotFoundException } = await import(\"../exceptions/so-exceptions\");\n          throw new NotFoundException(`존재하지 않는 API 접근입니다. ${request.url}`);\n        }\n\n        // 일반 파일 접근시 별도의 에러 출력하지 않음\n        return;\n      });\n    } else {\n      for (const api of this.syncer.apis) {\n        // model\n        if (this.syncer.models[api.modelName] === undefined) {\n          throw new Error(`정의되지 않은 모델에 접근 ${api.modelName}`);\n        }\n\n        // route\n        server.route({\n          method: api.options.httpMethod ?? \"GET\",\n          url: this.config.api.route.prefix + api.path,\n          handler: this.createApiHandler(api, config),\n        }); // END server.route\n      }\n    }\n  }\n\n  createApiHandler(\n    api: ExtendedApi,\n    config: SonamuFastifyConfig,\n  ): (request: FastifyRequest, reply: FastifyReply) => Promise<unknown> {\n    return async (request: FastifyRequest, reply: FastifyReply): Promise<unknown> => {\n      (api.options.guards ?? []).every((guard) => config.guardHandler(guard, request, api));\n\n      // 파라미터 정보로 zod 스키마 빌드\n      const { getZodObjectFromApi } = await import(\"./code-converters\");\n      const ReqType = getZodObjectFromApi(api, this.syncer.types);\n\n      // request 파싱\n      const which = api.options.httpMethod === \"GET\" ? \"query\" : \"body\";\n      let reqBody: {\n        [key: string]: unknown;\n      };\n      try {\n        const { fastifyCaster } = await import(\"./caster\");\n        reqBody = fastifyCaster(ReqType).parse(request[which] ?? {});\n      } catch (e) {\n        const { ZodError } = await import(\"zod\");\n        if (e instanceof ZodError) {\n          const { humanizeZodError } = await import(\"../utils/zod-error\");\n          const messages = humanizeZodError(e)\n            .map((issue) => issue.message)\n            .join(\" \");\n          const { BadRequestException } = await import(\"../exceptions/so-exceptions\");\n          throw new BadRequestException(messages, {\n            zodError: e,\n          });\n        } else {\n          throw e;\n        }\n      }\n\n      // Content-Type\n      reply.type(api.options.contentType ?? \"application/json\");\n\n      // Context 생성\n      const context: Context = await this.createContext(config, request, reply);\n\n      // 모델 메소드 args 생성하여 호출\n      const { ApiParamType } = await import(\"../types/types\");\n      const args = api.parameters.map((param) => {\n        // Context 인젝션\n        if (ApiParamType.isContext(param.type)) {\n          return context;\n        } else {\n          return reqBody[param.name];\n        }\n      });\n      return this.invokeModelMethod(api, args, context, reply);\n    };\n  }\n\n  async invokeModelMethod(\n    api: ExtendedApi,\n    args: unknown[],\n    context: Context,\n    reply: FastifyReply,\n  ): Promise<unknown> {\n    const model = this.syncer.models[api.modelName];\n    return this.asyncLocalStorage.run({ context }, async () => {\n      // biome-ignore lint/suspicious/noExplicitAny: model은 모델 인스턴스이므로 메서드 호출 가능\n      const result = await (model as any)[api.methodName].apply(model, args);\n      reply.type(api.options.contentType ?? \"application/json\");\n\n      return result;\n    });\n  }\n\n  async createContext(\n    config: SonamuFastifyConfig,\n    request: FastifyRequest,\n    reply: FastifyReply,\n  ): Promise<Context> {\n    // createSSEFactory 함수에 미리 request의 socket과 reply를 바인딩.\n    const { createSSEFactory } = await import(\"../stream/sse\");\n    const createSSE = (<T extends ZodObject>(\n      _request: FastifyRequest,\n      _reply: FastifyReply,\n      _events: T,\n    ) => createSSEFactory(_request.socket, _reply, _events)).bind(null, request, reply);\n\n    const context: Context = {\n      ...(await Promise.resolve(\n        config.contextProvider(\n          {\n            request,\n            reply,\n            headers: request.headers,\n            createSSE,\n            naiteStore: Naite.createStore(),\n            // auth\n            user: request.user ?? null,\n            passport: {\n              login: request.login.bind(request) as AuthContext[\"passport\"][\"login\"],\n              logout: request.logout.bind(request) as AuthContext[\"passport\"][\"logout\"],\n            },\n          },\n          request,\n          reply,\n        ),\n      )),\n    };\n    return context;\n  }\n\n  async startWatcher(): Promise<void> {\n    const watchPath = [path.join(this.apiRootPath, \"src\")];\n\n    const chokidar = (await import(\"chokidar\")).default;\n    this.watcher = chokidar.watch(watchPath, {\n      ignored: (path, stats) =>\n        !!stats?.isFile() && !path.endsWith(\".ts\") && !path.endsWith(\".json\"),\n      persistent: true,\n      ignoreInitial: true,\n    });\n\n    this.watcher.on(\"all\", async (event: string, filePath: string) => {\n      const absolutePath = filePath as AbsolutePath;\n      assert(\n        absolutePath.startsWith(this.apiRootPath),\n        \"File path is not within the API root path\",\n      );\n\n      if (event !== \"change\" && event !== \"add\") {\n        return;\n      }\n\n      try {\n        // sonamu.config.ts 변경 시 재시작\n        const isConfigTs = filePath === path.join(this.apiRootPath, \"src\", \"sonamu.config.ts\");\n\n        if (isConfigTs) {\n          const relativePath = filePath.replace(this.apiRootPath, \"api\");\n          const chalk = (await import(\"chalk\")).default;\n          console.log(\n            chalk.bold(`Detected(${event}): ${chalk.blue(relativePath)} - Restarting...`),\n          );\n          process.kill(process.pid, \"SIGUSR2\");\n          return;\n        }\n\n        await this.handleFileChange(event, absolutePath);\n      } catch (e) {\n        console.error(e);\n      }\n    });\n  }\n\n  /*\n     A function that automatically handles init and destroy when using Sonamu via scripts.    \n  */\n  async runScript(fn: () => Promise<void>) {\n    await this.init(true, false, undefined, false);\n    try {\n      await fn();\n    } finally {\n      await this.destroy();\n    }\n  }\n\n  private async registerPlugins(server: FastifyInstance, plugins: SonamuServerOptions[\"plugins\"]) {\n    if (!plugins) {\n      return;\n    }\n\n    const pluginsModules = {\n      cors: \"@fastify/cors\",\n      formbody: \"@fastify/formbody\",\n      multipart: \"@fastify/multipart\",\n      qs: \"fastify-qs\",\n      sse: \"fastify-sse-v2\",\n      static: \"@fastify/static\",\n      session: \"@fastify/secure-session\",\n    } as const;\n\n    const registerPlugin = async <K extends keyof NonNullable<typeof plugins>>(\n      key: K,\n      pluginName: string,\n    ) => {\n      const option = plugins[key];\n      if (!option) return;\n\n      if (option === true) {\n        server.register((await import(pluginName)).default);\n      } else {\n        server.register((await import(pluginName)).default, option);\n      }\n    };\n\n    for (const [key, pluginName] of Object.entries(pluginsModules)) {\n      await registerPlugin(key as keyof typeof plugins, pluginName);\n    }\n\n    if (plugins.custom) {\n      plugins.custom(server);\n    }\n  }\n\n  private async registerAuth(\n    server: FastifyInstance,\n    options: NonNullable<SonamuServerOptions[\"auth\"]>,\n  ) {\n    // await import(\"fastify\");\n    const fastifyPassport = (await import(\"@fastify/passport\")).default;\n    server.register(fastifyPassport.initialize());\n    server.register(fastifyPassport.secureSession());\n\n    if (typeof options === \"boolean\") {\n      fastifyPassport.registerUserSerializer(async (user, _request) => user);\n      fastifyPassport.registerUserDeserializer(async (serialized, _request) => serialized);\n    } else {\n      fastifyPassport.registerUserSerializer(options.userSerializer);\n      fastifyPassport.registerUserDeserializer(options.userDeserializer);\n    }\n  }\n\n  private async initializeWorkflows(options: SonamuTaskOptions | undefined) {\n    const { WorkflowManager } = await import(\"../tasks/workflow-manager\");\n    // NOTE: @sonamu-kit/tasks 안에선 knex config를 수정하기 때문에 connection이 아닌 config 째로 보냅니다.\n    this._workflows = new WorkflowManager(DB.getDBConfig(\"w\"));\n    if (!options) {\n      return;\n    }\n\n    const enableWorker = options.enableWorker ?? isDaemonServer();\n    const defaultWorkerOptions = {\n      concurrency: os.cpus().length - 1,\n      usePubSub: true,\n      listenDelay: 500,\n    };\n\n    if (enableWorker) {\n      this.workflows.setupWorker({\n        ...defaultWorkerOptions,\n        ...options.workerOptions,\n      });\n    }\n  }\n\n  private async boot(server: FastifyInstance, options: SonamuServerOptions) {\n    const port = options.listen?.port ?? 3000;\n    const host = options.listen?.host ?? \"localhost\";\n\n    server.addHook(\"onClose\", async () => {\n      await options.lifecycle?.onShutdown?.(server);\n      await this.workflows.destroy();\n      await this.destroy();\n    });\n\n    const shutdown = async () => {\n      try {\n        await server.close();\n        process.exit(0);\n      } catch (err) {\n        console.error(\"Error during shutdown:\", err);\n        process.exit(1);\n      }\n    };\n\n    process.on(\"SIGINT\", shutdown);\n    process.on(\"SIGTERM\", shutdown);\n\n    if (options.lifecycle?.onError) {\n      server.setErrorHandler(options.lifecycle?.onError);\n    }\n\n    server\n      .listen({ port, host })\n      .then(async () => {\n        await this.workflows.startWorker();\n        await options.lifecycle?.onStart?.(server);\n      })\n      .catch(async (err) => {\n        const chalk = (await import(\"chalk\")).default;\n        console.error(chalk.red(\"Failed to start server:\", err));\n        await shutdown();\n      });\n  }\n\n  private async handleFileChange(event: string, filePath: AbsolutePath): Promise<void> {\n    // 첫 번째 파일이면 HMR 시작 시간 기록\n    if (this.pendingFiles.length === 0) {\n      this.hmrStartTime = Date.now();\n    }\n    this.pendingFiles.push(filePath);\n\n    const relativePath = path.relative(this.apiRootPath, filePath);\n    const chalk = (await import(\"chalk\")).default;\n    console.log(chalk.bold(`Detected(${event}): ${chalk.blue(relativePath)}`));\n\n    await this.syncer.syncFromWatcher(event, filePath);\n\n    // 처리 완료된 파일을 대기 목록에서 제거\n    this.pendingFiles = this.pendingFiles.slice(1);\n\n    // 모든 파일 처리가 완료되면 최종 메시지 출력\n    if (this.pendingFiles.length === 0) {\n      await this.finishHMR();\n    }\n  }\n\n  private async finishHMR(): Promise<void> {\n    await this.syncer.renewChecksums();\n\n    const endTime = Date.now();\n    const totalTime = endTime - this.hmrStartTime;\n    const [chalk, { centerText }] = await Promise.all([\n      (await import(\"chalk\")).default,\n      import(\"../utils/console-util\"),\n    ]);\n    const msg = `HMR Done! ${chalk.bold.white(`${totalTime}ms`)}`;\n\n    console.log(chalk.black.bgGreen(centerText(msg)));\n  }\n\n  async destroy(): Promise<void> {\n    const { BaseModel } = await import(\"../database/base-model\");\n    await BaseModel.destroy();\n    await this._workflows?.destroy();\n    await this.watcher?.close();\n  }\n}\nexport const Sonamu = new SonamuClass();\n"],"names":["assert","AsyncLocalStorage","os","path","createMockSSEFactory","DB","isDaemonServer","Naite","SonamuClass","isInitialized","asyncLocalStorage","uploadStorage","getContext","store","getStore","context","process","env","NODE_ENV","request","reply","headers","createSSE","schema","naiteStore","Map","Error","getUploadContext","uploadContext","_apiRootPath","apiRootPath","appRootPath","split","sep","slice","join","_dbConfig","dbConfig","_syncer","syncer","_config","config","_secrets","secrets","_storage","storage","_workflows","workflows","watcher","pendingFiles","hmrStartTime","server","initForTesting","init","undefined","doSilent","enableSync","forTesting","chalk","default","console","time","cyan","findApiRootPath","loadConfig","database","ANTHROPIC_API_KEY","anthropic_api_key","VOYAGE_API_KEY","voyage_api_key","OPENAI_API_KEY","openai_api_key","Object","keys","length","generateDBConfig","log","green","EntityManager","autoload","initializeWorkflows","tasks","Syncer","autoloadTypes","autoloadModels","autoloadApis","autoloadWorkflows","TemplateManager","isLocal","isTest","setupBiome","isHotReloadServer","sync","startWatcher","timeEnd","createServer","initOptions","options","fastify","StorageManager","plugins","registerPlugins","auth","session","registerAuth","withFastify","apiConfig","boot","timezone","api","formatInTimeZone","ISO_DATE_REGEX","DATE_FORMAT","setReplySerializer","payload","JSON","stringify","_key","value","test","Date","get","route","prefix","_request","_reply","apis","sonamuUIApiPlugin","register","all","url","startsWith","found","find","httpMethod","method","toUpperCase","createApiHandler","NotFoundException","models","modelName","handler","guards","every","guard","guardHandler","getZodObjectFromApi","ReqType","types","which","reqBody","fastifyCaster","parse","e","ZodError","humanizeZodError","messages","map","issue","message","BadRequestException","zodError","type","contentType","createContext","ApiParamType","args","parameters","param","isContext","name","invokeModelMethod","model","run","result","methodName","apply","createSSEFactory","_events","socket","bind","Promise","resolve","contextProvider","createStore","user","passport","login","logout","watchPath","chokidar","watch","ignored","stats","isFile","endsWith","persistent","ignoreInitial","on","event","filePath","absolutePath","isConfigTs","relativePath","replace","bold","blue","kill","pid","handleFileChange","error","runScript","fn","destroy","pluginsModules","cors","formbody","multipart","qs","sse","static","registerPlugin","key","pluginName","option","entries","custom","fastifyPassport","initialize","secureSession","registerUserSerializer","registerUserDeserializer","serialized","userSerializer","userDeserializer","WorkflowManager","getDBConfig","enableWorker","defaultWorkerOptions","concurrency","cpus","usePubSub","listenDelay","setupWorker","workerOptions","port","listen","host","addHook","lifecycle","onShutdown","shutdown","close","exit","err","onError","setErrorHandler","then","startWorker","onStart","catch","red","now","push","relative","syncFromWatcher","finishHMR","renewChecksums","endTime","totalTime","centerText","msg","white","black","bgGreen","BaseModel","Sonamu"],"mappings":"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"}
package/dist/index.d.ts CHANGED
@@ -15,7 +15,6 @@ export * from "./entity/entity";
15
15
  export * from "./entity/entity-manager";
16
16
  export * from "./exceptions/error-handler";
17
17
  export * from "./exceptions/so-exceptions";
18
- export * from "./file-storage/driver";
19
18
  export * from "./migration/migration-set";
20
19
  export * from "./migration/migrator";
21
20
  export * from "./migration/postgresql-schema-reader";
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,cAAc,kBAAkB,CAAC;AACjC,cAAc,cAAc,CAAC;AAC7B,mBAAmB,eAAe,CAAC;AACnC,cAAc,kBAAkB,CAAC;AACjC,cAAc,cAAc,CAAC;AAC7B,cAAc,uBAAuB,CAAC;AACtC,cAAc,6BAA6B,CAAC;AAC5C,cAAc,eAAe,CAAC;AAC9B,cAAc,iBAAiB,CAAC;AAChC,cAAc,uBAAuB,CAAC;AACtC,cAAc,8BAA8B,CAAC;AAC7C,cAAc,yBAAyB,CAAC;AACxC,cAAc,2BAA2B,CAAC;AAC1C,cAAc,iBAAiB,CAAC;AAChC,cAAc,yBAAyB,CAAC;AACxC,cAAc,4BAA4B,CAAC;AAC3C,cAAc,4BAA4B,CAAC;AAC3C,cAAc,uBAAuB,CAAC;AACtC,cAAc,2BAA2B,CAAC;AAC1C,cAAc,sBAAsB,CAAC;AACrC,cAAc,sCAAsC,CAAC;AACrD,cAAc,mBAAmB,CAAC;AAClC,cAAc,eAAe,CAAC;AAC9B,cAAc,wBAAwB,CAAC;AACvC,cAAc,cAAc,CAAC;AAC7B,cAAc,mBAAmB,CAAC;AAClC,cAAc,qBAAqB,CAAC;AACpC,cAAc,6BAA6B,CAAC;AAC5C,cAAc,2BAA2B,CAAC;AAC1C,cAAc,eAAe,CAAC;AAC9B,cAAc,oBAAoB,CAAC;AACnC,cAAc,eAAe,CAAC;AAC9B,cAAc,oBAAoB,CAAC;AACnC,cAAc,eAAe,CAAC"}
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,cAAc,kBAAkB,CAAC;AACjC,cAAc,cAAc,CAAC;AAC7B,mBAAmB,eAAe,CAAC;AACnC,cAAc,kBAAkB,CAAC;AACjC,cAAc,cAAc,CAAC;AAC7B,cAAc,uBAAuB,CAAC;AACtC,cAAc,6BAA6B,CAAC;AAC5C,cAAc,eAAe,CAAC;AAC9B,cAAc,iBAAiB,CAAC;AAChC,cAAc,uBAAuB,CAAC;AACtC,cAAc,8BAA8B,CAAC;AAC7C,cAAc,yBAAyB,CAAC;AACxC,cAAc,2BAA2B,CAAC;AAC1C,cAAc,iBAAiB,CAAC;AAChC,cAAc,yBAAyB,CAAC;AACxC,cAAc,4BAA4B,CAAC;AAC3C,cAAc,4BAA4B,CAAC;AAC3C,cAAc,2BAA2B,CAAC;AAC1C,cAAc,sBAAsB,CAAC;AACrC,cAAc,sCAAsC,CAAC;AACrD,cAAc,mBAAmB,CAAC;AAClC,cAAc,eAAe,CAAC;AAC9B,cAAc,wBAAwB,CAAC;AACvC,cAAc,cAAc,CAAC;AAC7B,cAAc,mBAAmB,CAAC;AAClC,cAAc,qBAAqB,CAAC;AACpC,cAAc,6BAA6B,CAAC;AAC5C,cAAc,2BAA2B,CAAC;AAC1C,cAAc,eAAe,CAAC;AAC9B,cAAc,oBAAoB,CAAC;AACnC,cAAc,eAAe,CAAC;AAC9B,cAAc,oBAAoB,CAAC;AACnC,cAAc,eAAe,CAAC"}
package/dist/index.js CHANGED
@@ -14,7 +14,6 @@ export * from "./entity/entity.js";
14
14
  export * from "./entity/entity-manager.js";
15
15
  export * from "./exceptions/error-handler.js";
16
16
  export * from "./exceptions/so-exceptions.js";
17
- export * from "./file-storage/driver.js";
18
17
  export * from "./migration/migration-set.js";
19
18
  export * from "./migration/migrator.js";
20
19
  export * from "./migration/postgresql-schema-reader.js";
@@ -33,4 +32,4 @@ export * from "./utils/type-utils.js";
33
32
  export * from "./utils/utils.js"; // export * from "./api/code-converters";
34
33
  // export * from "./syncer/syncer";
35
34
 
36
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uL3NyYy9pbmRleC50cyJdLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tIFwiLi9hcGkvYmFzZS1mcmFtZVwiO1xuZXhwb3J0ICogZnJvbSBcIi4vYXBpL2NvbmZpZ1wiO1xuZXhwb3J0IHR5cGUgKiBmcm9tIFwiLi9hcGkvY29udGV4dFwiO1xuZXhwb3J0ICogZnJvbSBcIi4vYXBpL2RlY29yYXRvcnNcIjtcbmV4cG9ydCAqIGZyb20gXCIuL2FwaS9zb25hbXVcIjtcbmV4cG9ydCAqIGZyb20gXCIuL2RhdGFiYXNlL2Jhc2UtbW9kZWxcIjtcbmV4cG9ydCAqIGZyb20gXCIuL2RhdGFiYXNlL2Jhc2UtbW9kZWwudHlwZXNcIjtcbmV4cG9ydCAqIGZyb20gXCIuL2RhdGFiYXNlL2RiXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9kYXRhYmFzZS9wdXJpXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9kYXRhYmFzZS9wdXJpLnR5cGVzXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9kYXRhYmFzZS9wdXJpLXN1YnNldC50eXBlc1wiO1xuZXhwb3J0ICogZnJvbSBcIi4vZGF0YWJhc2UvcHVyaS13cmFwcGVyXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9kYXRhYmFzZS91cHNlcnQtYnVpbGRlclwiO1xuZXhwb3J0ICogZnJvbSBcIi4vZW50aXR5L2VudGl0eVwiO1xuZXhwb3J0ICogZnJvbSBcIi4vZW50aXR5L2VudGl0eS1tYW5hZ2VyXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9leGNlcHRpb25zL2Vycm9yLWhhbmRsZXJcIjtcbmV4cG9ydCAqIGZyb20gXCIuL2V4Y2VwdGlvbnMvc28tZXhjZXB0aW9uc1wiO1xuZXhwb3J0ICogZnJvbSBcIi4vZmlsZS1zdG9yYWdlL2RyaXZlclwiO1xuZXhwb3J0ICogZnJvbSBcIi4vbWlncmF0aW9uL21pZ3JhdGlvbi1zZXRcIjtcbmV4cG9ydCAqIGZyb20gXCIuL21pZ3JhdGlvbi9taWdyYXRvclwiO1xuZXhwb3J0ICogZnJvbSBcIi4vbWlncmF0aW9uL3Bvc3RncmVzcWwtc2NoZW1hLXJlYWRlclwiO1xuZXhwb3J0ICogZnJvbSBcIi4vbWlncmF0aW9uL3R5cGVzXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9uYWl0ZS9uYWl0ZVwiO1xuZXhwb3J0ICogZnJvbSBcIi4vbmFpdGUvbmFpdGUtcmVwb3J0ZXJcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3N0cmVhbS9zc2VcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3Rhc2tzL2RlY29yYXRvclwiO1xuZXhwb3J0ICogZnJvbSBcIi4vdGVtcGxhdGUvdGVtcGxhdGVcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3RlbXBsYXRlL3RlbXBsYXRlLW1hbmFnZXJcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3Rlc3RpbmcvZml4dHVyZS1tYW5hZ2VyXCI7XG5leHBvcnQgKiBmcm9tIFwiLi90eXBlcy90eXBlc1wiO1xuZXhwb3J0ICogZnJvbSBcIi4vdXRpbHMvY29udHJvbGxlclwiO1xuZXhwb3J0ICogZnJvbSBcIi4vdXRpbHMvbW9kZWxcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3V0aWxzL3R5cGUtdXRpbHNcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3V0aWxzL3V0aWxzXCI7XG5cbi8vIGV4cG9ydCAqIGZyb20gXCIuL2FwaS9jb2RlLWNvbnZlcnRlcnNcIjtcbi8vIGV4cG9ydCAqIGZyb20gXCIuL3N5bmNlci9zeW5jZXJcIjtcbiJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLHNCQUFtQjtBQUNqQyxjQUFjLGtCQUFlO0FBRTdCLGNBQWMsc0JBQW1CO0FBQ2pDLGNBQWMsa0JBQWU7QUFDN0IsY0FBYywyQkFBd0I7QUFDdEMsY0FBYyxpQ0FBOEI7QUFDNUMsY0FBYyxtQkFBZ0I7QUFDOUIsY0FBYyxxQkFBa0I7QUFDaEMsY0FBYywyQkFBd0I7QUFDdEMsY0FBYyxrQ0FBK0I7QUFDN0MsY0FBYyw2QkFBMEI7QUFDeEMsY0FBYywrQkFBNEI7QUFDMUMsY0FBYyxxQkFBa0I7QUFDaEMsY0FBYyw2QkFBMEI7QUFDeEMsY0FBYyxnQ0FBNkI7QUFDM0MsY0FBYyxnQ0FBNkI7QUFDM0MsY0FBYywyQkFBd0I7QUFDdEMsY0FBYywrQkFBNEI7QUFDMUMsY0FBYywwQkFBdUI7QUFDckMsY0FBYywwQ0FBdUM7QUFDckQsY0FBYyx1QkFBb0I7QUFDbEMsY0FBYyxtQkFBZ0I7QUFDOUIsY0FBYyw0QkFBeUI7QUFDdkMsY0FBYyxrQkFBZTtBQUM3QixjQUFjLHVCQUFvQjtBQUNsQyxjQUFjLHlCQUFzQjtBQUNwQyxjQUFjLGlDQUE4QjtBQUM1QyxjQUFjLCtCQUE0QjtBQUMxQyxjQUFjLG1CQUFnQjtBQUM5QixjQUFjLHdCQUFxQjtBQUNuQyxjQUFjLG1CQUFnQjtBQUM5QixjQUFjLHdCQUFxQjtBQUNuQyxjQUFjLG1CQUFnQixDQUU5Qix5Q0FBeUM7Q0FDekMsbUNBQW1DIn0=
35
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uL3NyYy9pbmRleC50cyJdLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tIFwiLi9hcGkvYmFzZS1mcmFtZVwiO1xuZXhwb3J0ICogZnJvbSBcIi4vYXBpL2NvbmZpZ1wiO1xuZXhwb3J0IHR5cGUgKiBmcm9tIFwiLi9hcGkvY29udGV4dFwiO1xuZXhwb3J0ICogZnJvbSBcIi4vYXBpL2RlY29yYXRvcnNcIjtcbmV4cG9ydCAqIGZyb20gXCIuL2FwaS9zb25hbXVcIjtcbmV4cG9ydCAqIGZyb20gXCIuL2RhdGFiYXNlL2Jhc2UtbW9kZWxcIjtcbmV4cG9ydCAqIGZyb20gXCIuL2RhdGFiYXNlL2Jhc2UtbW9kZWwudHlwZXNcIjtcbmV4cG9ydCAqIGZyb20gXCIuL2RhdGFiYXNlL2RiXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9kYXRhYmFzZS9wdXJpXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9kYXRhYmFzZS9wdXJpLnR5cGVzXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9kYXRhYmFzZS9wdXJpLXN1YnNldC50eXBlc1wiO1xuZXhwb3J0ICogZnJvbSBcIi4vZGF0YWJhc2UvcHVyaS13cmFwcGVyXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9kYXRhYmFzZS91cHNlcnQtYnVpbGRlclwiO1xuZXhwb3J0ICogZnJvbSBcIi4vZW50aXR5L2VudGl0eVwiO1xuZXhwb3J0ICogZnJvbSBcIi4vZW50aXR5L2VudGl0eS1tYW5hZ2VyXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9leGNlcHRpb25zL2Vycm9yLWhhbmRsZXJcIjtcbmV4cG9ydCAqIGZyb20gXCIuL2V4Y2VwdGlvbnMvc28tZXhjZXB0aW9uc1wiO1xuZXhwb3J0ICogZnJvbSBcIi4vbWlncmF0aW9uL21pZ3JhdGlvbi1zZXRcIjtcbmV4cG9ydCAqIGZyb20gXCIuL21pZ3JhdGlvbi9taWdyYXRvclwiO1xuZXhwb3J0ICogZnJvbSBcIi4vbWlncmF0aW9uL3Bvc3RncmVzcWwtc2NoZW1hLXJlYWRlclwiO1xuZXhwb3J0ICogZnJvbSBcIi4vbWlncmF0aW9uL3R5cGVzXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9uYWl0ZS9uYWl0ZVwiO1xuZXhwb3J0ICogZnJvbSBcIi4vbmFpdGUvbmFpdGUtcmVwb3J0ZXJcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3N0cmVhbS9zc2VcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3Rhc2tzL2RlY29yYXRvclwiO1xuZXhwb3J0ICogZnJvbSBcIi4vdGVtcGxhdGUvdGVtcGxhdGVcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3RlbXBsYXRlL3RlbXBsYXRlLW1hbmFnZXJcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3Rlc3RpbmcvZml4dHVyZS1tYW5hZ2VyXCI7XG5leHBvcnQgKiBmcm9tIFwiLi90eXBlcy90eXBlc1wiO1xuZXhwb3J0ICogZnJvbSBcIi4vdXRpbHMvY29udHJvbGxlclwiO1xuZXhwb3J0ICogZnJvbSBcIi4vdXRpbHMvbW9kZWxcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3V0aWxzL3R5cGUtdXRpbHNcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3V0aWxzL3V0aWxzXCI7XG5cbi8vIGV4cG9ydCAqIGZyb20gXCIuL2FwaS9jb2RlLWNvbnZlcnRlcnNcIjtcbi8vIGV4cG9ydCAqIGZyb20gXCIuL3N5bmNlci9zeW5jZXJcIjtcbiJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLHNCQUFtQjtBQUNqQyxjQUFjLGtCQUFlO0FBRTdCLGNBQWMsc0JBQW1CO0FBQ2pDLGNBQWMsa0JBQWU7QUFDN0IsY0FBYywyQkFBd0I7QUFDdEMsY0FBYyxpQ0FBOEI7QUFDNUMsY0FBYyxtQkFBZ0I7QUFDOUIsY0FBYyxxQkFBa0I7QUFDaEMsY0FBYywyQkFBd0I7QUFDdEMsY0FBYyxrQ0FBK0I7QUFDN0MsY0FBYyw2QkFBMEI7QUFDeEMsY0FBYywrQkFBNEI7QUFDMUMsY0FBYyxxQkFBa0I7QUFDaEMsY0FBYyw2QkFBMEI7QUFDeEMsY0FBYyxnQ0FBNkI7QUFDM0MsY0FBYyxnQ0FBNkI7QUFDM0MsY0FBYywrQkFBNEI7QUFDMUMsY0FBYywwQkFBdUI7QUFDckMsY0FBYywwQ0FBdUM7QUFDckQsY0FBYyx1QkFBb0I7QUFDbEMsY0FBYyxtQkFBZ0I7QUFDOUIsY0FBYyw0QkFBeUI7QUFDdkMsY0FBYyxrQkFBZTtBQUM3QixjQUFjLHVCQUFvQjtBQUNsQyxjQUFjLHlCQUFzQjtBQUNwQyxjQUFjLGlDQUE4QjtBQUM1QyxjQUFjLCtCQUE0QjtBQUMxQyxjQUFjLG1CQUFnQjtBQUM5QixjQUFjLHdCQUFxQjtBQUNuQyxjQUFjLG1CQUFnQjtBQUM5QixjQUFjLHdCQUFxQjtBQUNuQyxjQUFjLG1CQUFnQixDQUU5Qix5Q0FBeUM7Q0FDekMsbUNBQW1DIn0=
@@ -0,0 +1,14 @@
1
+ import { FSDriver } from "flydrive/drivers/fs";
2
+ import type { FSDriverOptions } from "flydrive/drivers/fs/types";
3
+ import { S3Driver } from "flydrive/drivers/s3";
4
+ import type { S3DriverOptions } from "flydrive/drivers/s3/types";
5
+ /**
6
+ * 드라이버 팩토리 함수
7
+ * 설정 → 드라이버 인스턴스 생성 함수 변환
8
+ */
9
+ export declare const drivers: {
10
+ fs: (config: FSDriverOptions) => () => FSDriver;
11
+ s3: (config: S3DriverOptions) => () => S3Driver;
12
+ };
13
+ export type DriverKey = keyof typeof drivers;
14
+ //# sourceMappingURL=drivers.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"drivers.d.ts","sourceRoot":"","sources":["../../src/storage/drivers.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,qBAAqB,CAAC;AAC/C,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,2BAA2B,CAAC;AACjE,OAAO,EAAE,QAAQ,EAAE,MAAM,qBAAqB,CAAC;AAC/C,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,2BAA2B,CAAC;AAEjE;;;GAGG;AACH,eAAO,MAAM,OAAO;iBACL,eAAe;iBACf,eAAe;CAC7B,CAAC;AAEF,MAAM,MAAM,SAAS,GAAG,MAAM,OAAO,OAAO,CAAC"}
@@ -0,0 +1,11 @@
1
+ import { FSDriver } from "flydrive/drivers/fs";
2
+ import { S3Driver } from "flydrive/drivers/s3";
3
+ /**
4
+ * 드라이버 팩토리 함수
5
+ * 설정 → 드라이버 인스턴스 생성 함수 변환
6
+ */ export const drivers = {
7
+ fs: (config)=>()=>new FSDriver(config),
8
+ s3: (config)=>()=>new S3Driver(config)
9
+ };
10
+
11
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9zdG9yYWdlL2RyaXZlcnMudHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgRlNEcml2ZXIgfSBmcm9tIFwiZmx5ZHJpdmUvZHJpdmVycy9mc1wiO1xuaW1wb3J0IHR5cGUgeyBGU0RyaXZlck9wdGlvbnMgfSBmcm9tIFwiZmx5ZHJpdmUvZHJpdmVycy9mcy90eXBlc1wiO1xuaW1wb3J0IHsgUzNEcml2ZXIgfSBmcm9tIFwiZmx5ZHJpdmUvZHJpdmVycy9zM1wiO1xuaW1wb3J0IHR5cGUgeyBTM0RyaXZlck9wdGlvbnMgfSBmcm9tIFwiZmx5ZHJpdmUvZHJpdmVycy9zMy90eXBlc1wiO1xuXG4vKipcbiAqIOuTnOudvOydtOuyhCDtjKnthqDrpqwg7ZWo7IiYXG4gKiDshKTsoJUg4oaSIOuTnOudvOydtOuyhCDsnbjsiqTthLTsiqQg7IOd7ISxIO2VqOyImCDrs4DtmZhcbiAqL1xuZXhwb3J0IGNvbnN0IGRyaXZlcnMgPSB7XG4gIGZzOiAoY29uZmlnOiBGU0RyaXZlck9wdGlvbnMpID0+ICgpID0+IG5ldyBGU0RyaXZlcihjb25maWcpLFxuICBzMzogKGNvbmZpZzogUzNEcml2ZXJPcHRpb25zKSA9PiAoKSA9PiBuZXcgUzNEcml2ZXIoY29uZmlnKSxcbn07XG5cbmV4cG9ydCB0eXBlIERyaXZlcktleSA9IGtleW9mIHR5cGVvZiBkcml2ZXJzO1xuIl0sIm5hbWVzIjpbIkZTRHJpdmVyIiwiUzNEcml2ZXIiLCJkcml2ZXJzIiwiZnMiLCJjb25maWciLCJzMyJdLCJtYXBwaW5ncyI6IkFBQUEsU0FBU0EsUUFBUSxRQUFRLHNCQUFzQjtBQUUvQyxTQUFTQyxRQUFRLFFBQVEsc0JBQXNCO0FBRy9DOzs7Q0FHQyxHQUNELE9BQU8sTUFBTUMsVUFBVTtJQUNyQkMsSUFBSSxDQUFDQyxTQUE0QixJQUFNLElBQUlKLFNBQVNJO0lBQ3BEQyxJQUFJLENBQUNELFNBQTRCLElBQU0sSUFBSUgsU0FBU0c7QUFDdEQsRUFBRSJ9
@@ -0,0 +1,5 @@
1
+ export { type DriverKey, drivers } from "./drivers";
2
+ export { StorageManager } from "./storage-manager";
3
+ export type { StorageConfig } from "./types";
4
+ export { UploadedFile } from "./uploaded-file";
5
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/storage/index.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,KAAK,SAAS,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACpD,OAAO,EAAE,cAAc,EAAE,MAAM,mBAAmB,CAAC;AACnD,YAAY,EAAE,aAAa,EAAE,MAAM,SAAS,CAAC;AAC7C,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC"}
@@ -0,0 +1,6 @@
1
+ // Storage 서브모듈 exports
2
+ export { drivers } from "./drivers.js";
3
+ export { StorageManager } from "./storage-manager.js";
4
+ export { UploadedFile } from "./uploaded-file.js";
5
+
6
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9zdG9yYWdlL2luZGV4LnRzIl0sInNvdXJjZXNDb250ZW50IjpbIi8vIFN0b3JhZ2Ug7ISc67iM66qo65OIIGV4cG9ydHNcbmV4cG9ydCB7IHR5cGUgRHJpdmVyS2V5LCBkcml2ZXJzIH0gZnJvbSBcIi4vZHJpdmVyc1wiO1xuZXhwb3J0IHsgU3RvcmFnZU1hbmFnZXIgfSBmcm9tIFwiLi9zdG9yYWdlLW1hbmFnZXJcIjtcbmV4cG9ydCB0eXBlIHsgU3RvcmFnZUNvbmZpZyB9IGZyb20gXCIuL3R5cGVzXCI7XG5leHBvcnQgeyBVcGxvYWRlZEZpbGUgfSBmcm9tIFwiLi91cGxvYWRlZC1maWxlXCI7XG4iXSwibmFtZXMiOlsiZHJpdmVycyIsIlN0b3JhZ2VNYW5hZ2VyIiwiVXBsb2FkZWRGaWxlIl0sIm1hcHBpbmdzIjoiQUFBQSx1QkFBdUI7QUFDdkIsU0FBeUJBLE9BQU8sUUFBUSxlQUFZO0FBQ3BELFNBQVNDLGNBQWMsUUFBUSx1QkFBb0I7QUFFbkQsU0FBU0MsWUFBWSxRQUFRLHFCQUFrQiJ9
@@ -0,0 +1,21 @@
1
+ import { Disk } from "flydrive";
2
+ import type { DriverKey } from "./drivers";
3
+ import type { StorageConfig } from "./types";
4
+ /**
5
+ * 여러 디스크를 관리하는 매니저
6
+ */
7
+ export declare class StorageManager {
8
+ private config;
9
+ private disks;
10
+ constructor(config: StorageConfig);
11
+ /**
12
+ * 디스크 인스턴스 반환 (lazy initialization)
13
+ * @param diskName 디스크 이름 (없으면 default)
14
+ */
15
+ use(diskName?: DriverKey): Disk;
16
+ /**
17
+ * 기본 디스크 이름 반환
18
+ */
19
+ get defaultDisk(): string;
20
+ }
21
+ //# sourceMappingURL=storage-manager.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"storage-manager.d.ts","sourceRoot":"","sources":["../../src/storage/storage-manager.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,UAAU,CAAC;AAEhC,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,WAAW,CAAC;AAC3C,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,SAAS,CAAC;AAE7C;;GAEG;AACH,qBAAa,cAAc;IAGb,OAAO,CAAC,MAAM;IAF1B,OAAO,CAAC,KAAK,CAAmC;gBAE5B,MAAM,EAAE,aAAa;IAEzC;;;OAGG;IACH,GAAG,CAAC,QAAQ,CAAC,EAAE,SAAS,GAAG,IAAI;IAe/B;;OAEG;IACH,IAAI,WAAW,IAAI,MAAM,CAExB;CACF"}
@@ -0,0 +1,33 @@
1
+ import { Disk } from "flydrive";
2
+ import { assertDefined } from "../utils/utils.js";
3
+ /**
4
+ * 여러 디스크를 관리하는 매니저
5
+ */ export class StorageManager {
6
+ config;
7
+ disks = new Map();
8
+ constructor(config){
9
+ this.config = config;
10
+ }
11
+ /**
12
+ * 디스크 인스턴스 반환 (lazy initialization)
13
+ * @param diskName 디스크 이름 (없으면 default)
14
+ */ use(diskName) {
15
+ const name = diskName ?? this.config.default;
16
+ if (!this.disks.has(name)) {
17
+ const factory = this.config.drivers[name];
18
+ if (!factory) {
19
+ const available = Object.keys(this.config.drivers).join(", ");
20
+ throw new Error(`Unknown disk: "${name}". Available: ${available}`);
21
+ }
22
+ this.disks.set(name, new Disk(factory()));
23
+ }
24
+ return assertDefined(this.disks.get(name), `Disk ${name} not found`);
25
+ }
26
+ /**
27
+ * 기본 디스크 이름 반환
28
+ */ get defaultDisk() {
29
+ return this.config.default;
30
+ }
31
+ }
32
+
33
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9zdG9yYWdlL3N0b3JhZ2UtbWFuYWdlci50cyJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBEaXNrIH0gZnJvbSBcImZseWRyaXZlXCI7XG5pbXBvcnQgeyBhc3NlcnREZWZpbmVkIH0gZnJvbSBcIi4uL3V0aWxzL3V0aWxzXCI7XG5pbXBvcnQgdHlwZSB7IERyaXZlcktleSB9IGZyb20gXCIuL2RyaXZlcnNcIjtcbmltcG9ydCB0eXBlIHsgU3RvcmFnZUNvbmZpZyB9IGZyb20gXCIuL3R5cGVzXCI7XG5cbi8qKlxuICog7Jes65+sIOuUlOyKpO2BrOulvCDqtIDrpqztlZjripQg66ek64uI7KCAXG4gKi9cbmV4cG9ydCBjbGFzcyBTdG9yYWdlTWFuYWdlciB7XG4gIHByaXZhdGUgZGlza3M6IE1hcDxEcml2ZXJLZXksIERpc2s+ID0gbmV3IE1hcCgpO1xuXG4gIGNvbnN0cnVjdG9yKHByaXZhdGUgY29uZmlnOiBTdG9yYWdlQ29uZmlnKSB7fVxuXG4gIC8qKlxuICAgKiDrlJTsiqTtgawg7J247Iqk7YS07IqkIOuwmO2ZmCAobGF6eSBpbml0aWFsaXphdGlvbilcbiAgICogQHBhcmFtIGRpc2tOYW1lIOuUlOyKpO2BrCDsnbTrpoQgKOyXhuycvOuptCBkZWZhdWx0KVxuICAgKi9cbiAgdXNlKGRpc2tOYW1lPzogRHJpdmVyS2V5KTogRGlzayB7XG4gICAgY29uc3QgbmFtZSA9IGRpc2tOYW1lID8/ICh0aGlzLmNvbmZpZy5kZWZhdWx0IGFzIERyaXZlcktleSk7XG5cbiAgICBpZiAoIXRoaXMuZGlza3MuaGFzKG5hbWUpKSB7XG4gICAgICBjb25zdCBmYWN0b3J5ID0gdGhpcy5jb25maWcuZHJpdmVyc1tuYW1lXTtcbiAgICAgIGlmICghZmFjdG9yeSkge1xuICAgICAgICBjb25zdCBhdmFpbGFibGUgPSBPYmplY3Qua2V5cyh0aGlzLmNvbmZpZy5kcml2ZXJzKS5qb2luKFwiLCBcIik7XG4gICAgICAgIHRocm93IG5ldyBFcnJvcihgVW5rbm93biBkaXNrOiBcIiR7bmFtZX1cIi4gQXZhaWxhYmxlOiAke2F2YWlsYWJsZX1gKTtcbiAgICAgIH1cbiAgICAgIHRoaXMuZGlza3Muc2V0KG5hbWUsIG5ldyBEaXNrKGZhY3RvcnkoKSkpO1xuICAgIH1cblxuICAgIHJldHVybiBhc3NlcnREZWZpbmVkKHRoaXMuZGlza3MuZ2V0KG5hbWUpLCBgRGlzayAke25hbWV9IG5vdCBmb3VuZGApO1xuICB9XG5cbiAgLyoqXG4gICAqIOq4sOuzuCDrlJTsiqTtgawg7J2066aEIOuwmO2ZmFxuICAgKi9cbiAgZ2V0IGRlZmF1bHREaXNrKCk6IHN0cmluZyB7XG4gICAgcmV0dXJuIHRoaXMuY29uZmlnLmRlZmF1bHQ7XG4gIH1cbn1cbiJdLCJuYW1lcyI6WyJEaXNrIiwiYXNzZXJ0RGVmaW5lZCIsIlN0b3JhZ2VNYW5hZ2VyIiwiZGlza3MiLCJNYXAiLCJjb25maWciLCJ1c2UiLCJkaXNrTmFtZSIsIm5hbWUiLCJkZWZhdWx0IiwiaGFzIiwiZmFjdG9yeSIsImRyaXZlcnMiLCJhdmFpbGFibGUiLCJPYmplY3QiLCJrZXlzIiwiam9pbiIsIkVycm9yIiwic2V0IiwiZ2V0IiwiZGVmYXVsdERpc2siXSwibWFwcGluZ3MiOiJBQUFBLFNBQVNBLElBQUksUUFBUSxXQUFXO0FBQ2hDLFNBQVNDLGFBQWEsUUFBUSxvQkFBaUI7QUFJL0M7O0NBRUMsR0FDRCxPQUFPLE1BQU1DOztJQUNIQyxRQUE4QixJQUFJQyxNQUFNO0lBRWhELFlBQVksQUFBUUMsTUFBcUIsQ0FBRTthQUF2QkEsU0FBQUE7SUFBd0I7SUFFNUM7OztHQUdDLEdBQ0RDLElBQUlDLFFBQW9CLEVBQVE7UUFDOUIsTUFBTUMsT0FBT0QsWUFBYSxJQUFJLENBQUNGLE1BQU0sQ0FBQ0ksT0FBTztRQUU3QyxJQUFJLENBQUMsSUFBSSxDQUFDTixLQUFLLENBQUNPLEdBQUcsQ0FBQ0YsT0FBTztZQUN6QixNQUFNRyxVQUFVLElBQUksQ0FBQ04sTUFBTSxDQUFDTyxPQUFPLENBQUNKLEtBQUs7WUFDekMsSUFBSSxDQUFDRyxTQUFTO2dCQUNaLE1BQU1FLFlBQVlDLE9BQU9DLElBQUksQ0FBQyxJQUFJLENBQUNWLE1BQU0sQ0FBQ08sT0FBTyxFQUFFSSxJQUFJLENBQUM7Z0JBQ3hELE1BQU0sSUFBSUMsTUFBTSxDQUFDLGVBQWUsRUFBRVQsS0FBSyxjQUFjLEVBQUVLLFdBQVc7WUFDcEU7WUFDQSxJQUFJLENBQUNWLEtBQUssQ0FBQ2UsR0FBRyxDQUFDVixNQUFNLElBQUlSLEtBQUtXO1FBQ2hDO1FBRUEsT0FBT1YsY0FBYyxJQUFJLENBQUNFLEtBQUssQ0FBQ2dCLEdBQUcsQ0FBQ1gsT0FBTyxDQUFDLEtBQUssRUFBRUEsS0FBSyxVQUFVLENBQUM7SUFDckU7SUFFQTs7R0FFQyxHQUNELElBQUlZLGNBQXNCO1FBQ3hCLE9BQU8sSUFBSSxDQUFDZixNQUFNLENBQUNJLE9BQU87SUFDNUI7QUFDRiJ9
@@ -0,0 +1,12 @@
1
+ import type { DriverContract } from "flydrive/types";
2
+ import type { DriverKey } from "./drivers";
3
+ /**
4
+ * Storage 설정 타입
5
+ */
6
+ export type StorageConfig = {
7
+ /** 기본 디스크 이름 */
8
+ default: string;
9
+ /** 디스크별 드라이버 팩토리 */
10
+ drivers: Record<DriverKey, () => DriverContract>;
11
+ };
12
+ //# sourceMappingURL=types.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../src/storage/types.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,gBAAgB,CAAC;AACrD,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,WAAW,CAAC;AAE3C;;GAEG;AACH,MAAM,MAAM,aAAa,GAAG;IAC1B,gBAAgB;IAChB,OAAO,EAAE,MAAM,CAAC;IAChB,oBAAoB;IACpB,OAAO,EAAE,MAAM,CAAC,SAAS,EAAE,MAAM,cAAc,CAAC,CAAC;CAClD,CAAC"}
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Storage 설정 타입
3
+ */ export { };
4
+
5
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9zdG9yYWdlL3R5cGVzLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB0eXBlIHsgRHJpdmVyQ29udHJhY3QgfSBmcm9tIFwiZmx5ZHJpdmUvdHlwZXNcIjtcbmltcG9ydCB0eXBlIHsgRHJpdmVyS2V5IH0gZnJvbSBcIi4vZHJpdmVyc1wiO1xuXG4vKipcbiAqIFN0b3JhZ2Ug7ISk7KCVIO2DgOyehVxuICovXG5leHBvcnQgdHlwZSBTdG9yYWdlQ29uZmlnID0ge1xuICAvKiog6riw67O4IOuUlOyKpO2BrCDsnbTrpoQgKi9cbiAgZGVmYXVsdDogc3RyaW5nO1xuICAvKiog65SU7Iqk7YGs67OEIOuTnOudvOydtOuyhCDtjKnthqDrpqwgKi9cbiAgZHJpdmVyczogUmVjb3JkPERyaXZlcktleSwgKCkgPT4gRHJpdmVyQ29udHJhY3Q+O1xufTtcbiJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFHQTs7Q0FFQyxHQUNELFdBS0UifQ==
@@ -0,0 +1,35 @@
1
+ import type { MultipartFile } from "@fastify/multipart";
2
+ import type { DriverKey } from "./drivers";
3
+ /**
4
+ * 업로드된 파일 래퍼
5
+ */
6
+ export declare class UploadedFile {
7
+ private _file;
8
+ private _buffer?;
9
+ private _url?;
10
+ constructor(file: MultipartFile);
11
+ /** 원본 파일명 */
12
+ get filename(): string;
13
+ /** MIME 타입 */
14
+ get mimetype(): string;
15
+ /** 파일 크기 (bytes) */
16
+ get size(): number;
17
+ /** 확장자 (점 제외) */
18
+ get extname(): string | false;
19
+ /** saveToDisk 후 저장된 URL */
20
+ get url(): string | undefined;
21
+ /** Buffer로 변환 (캐싱됨) */
22
+ toBuffer(): Promise<Buffer>;
23
+ /** MD5 해시 계산 */
24
+ md5(): Promise<string>;
25
+ /**
26
+ * 파일을 디스크에 저장
27
+ * @param key 저장 경로 (예: 'uploads/avatar.png')
28
+ * @param diskName 디스크 이름 (기본: default disk)
29
+ * @returns 저장된 파일의 URL
30
+ */
31
+ saveToDisk(key: string, diskName?: DriverKey): Promise<string>;
32
+ /** 원본 MultipartFile 접근 */
33
+ get raw(): MultipartFile;
34
+ }
35
+ //# sourceMappingURL=uploaded-file.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"uploaded-file.d.ts","sourceRoot":"","sources":["../../src/storage/uploaded-file.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,oBAAoB,CAAC;AAGxD,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,WAAW,CAAC;AAE3C;;GAEG;AACH,qBAAa,YAAY;IACvB,OAAO,CAAC,KAAK,CAAgB;IAC7B,OAAO,CAAC,OAAO,CAAC,CAAS;IACzB,OAAO,CAAC,IAAI,CAAC,CAAS;gBAEV,IAAI,EAAE,aAAa;IAI/B,aAAa;IACb,IAAI,QAAQ,IAAI,MAAM,CAErB;IAED,cAAc;IACd,IAAI,QAAQ,IAAI,MAAM,CAErB;IAED,oBAAoB;IACpB,IAAI,IAAI,IAAI,MAAM,CAEjB;IAED,iBAAiB;IACjB,IAAI,OAAO,IAAI,MAAM,GAAG,KAAK,CAE5B;IAED,2BAA2B;IAC3B,IAAI,GAAG,IAAI,MAAM,GAAG,SAAS,CAE5B;IAED,uBAAuB;IACjB,QAAQ,IAAI,OAAO,CAAC,MAAM,CAAC;IAOjC,gBAAgB;IACV,GAAG,IAAI,OAAO,CAAC,MAAM,CAAC;IAK5B;;;;;OAKG;IACG,UAAU,CAAC,GAAG,EAAE,MAAM,EAAE,QAAQ,CAAC,EAAE,SAAS,GAAG,OAAO,CAAC,MAAM,CAAC;IAcpE,0BAA0B;IAC1B,IAAI,GAAG,IAAI,aAAa,CAEvB;CACF"}
@@ -0,0 +1,58 @@
1
+ import { createHash } from "crypto";
2
+ import mime from "mime-types";
3
+ /**
4
+ * 업로드된 파일 래퍼
5
+ */ export class UploadedFile {
6
+ _file;
7
+ _buffer;
8
+ _url;
9
+ constructor(file){
10
+ this._file = file;
11
+ }
12
+ /** 원본 파일명 */ get filename() {
13
+ return this._file.filename;
14
+ }
15
+ /** MIME 타입 */ get mimetype() {
16
+ return this._file.mimetype;
17
+ }
18
+ /** 파일 크기 (bytes) */ get size() {
19
+ return this._file.file.bytesRead;
20
+ }
21
+ /** 확장자 (점 제외) */ get extname() {
22
+ return mime.extension(this._file.mimetype);
23
+ }
24
+ /** saveToDisk 후 저장된 URL */ get url() {
25
+ return this._url;
26
+ }
27
+ /** Buffer로 변환 (캐싱됨) */ async toBuffer() {
28
+ if (!this._buffer) {
29
+ this._buffer = await this._file.toBuffer();
30
+ }
31
+ return this._buffer;
32
+ }
33
+ /** MD5 해시 계산 */ async md5() {
34
+ const buffer = await this.toBuffer();
35
+ return createHash("md5").update(buffer).digest("hex");
36
+ }
37
+ /**
38
+ * 파일을 디스크에 저장
39
+ * @param key 저장 경로 (예: 'uploads/avatar.png')
40
+ * @param diskName 디스크 이름 (기본: default disk)
41
+ * @returns 저장된 파일의 URL
42
+ */ async saveToDisk(key, diskName) {
43
+ // 순환 의존성 방지를 위해 동적 import
44
+ const { Sonamu } = await import("../api/sonamu.js");
45
+ const disk = Sonamu.storage.use(diskName);
46
+ const buffer = await this.toBuffer();
47
+ await disk.put(key, new Uint8Array(buffer), {
48
+ contentType: this.mimetype
49
+ });
50
+ this._url = await disk.getSignedUrl(key);
51
+ return this._url;
52
+ }
53
+ /** 원본 MultipartFile 접근 */ get raw() {
54
+ return this._file;
55
+ }
56
+ }
57
+
58
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9zdG9yYWdlL3VwbG9hZGVkLWZpbGUudHMiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHR5cGUgeyBNdWx0aXBhcnRGaWxlIH0gZnJvbSBcIkBmYXN0aWZ5L211bHRpcGFydFwiO1xuaW1wb3J0IHsgY3JlYXRlSGFzaCB9IGZyb20gXCJjcnlwdG9cIjtcbmltcG9ydCBtaW1lIGZyb20gXCJtaW1lLXR5cGVzXCI7XG5pbXBvcnQgdHlwZSB7IERyaXZlcktleSB9IGZyb20gXCIuL2RyaXZlcnNcIjtcblxuLyoqXG4gKiDsl4XroZzrk5zrkJwg7YyM7J28IOuemO2NvFxuICovXG5leHBvcnQgY2xhc3MgVXBsb2FkZWRGaWxlIHtcbiAgcHJpdmF0ZSBfZmlsZTogTXVsdGlwYXJ0RmlsZTtcbiAgcHJpdmF0ZSBfYnVmZmVyPzogQnVmZmVyO1xuICBwcml2YXRlIF91cmw/OiBzdHJpbmc7XG5cbiAgY29uc3RydWN0b3IoZmlsZTogTXVsdGlwYXJ0RmlsZSkge1xuICAgIHRoaXMuX2ZpbGUgPSBmaWxlO1xuICB9XG5cbiAgLyoqIOybkOuzuCDtjIzsnbzrqoUgKi9cbiAgZ2V0IGZpbGVuYW1lKCk6IHN0cmluZyB7XG4gICAgcmV0dXJuIHRoaXMuX2ZpbGUuZmlsZW5hbWU7XG4gIH1cblxuICAvKiogTUlNRSDtg4DsnoUgKi9cbiAgZ2V0IG1pbWV0eXBlKCk6IHN0cmluZyB7XG4gICAgcmV0dXJuIHRoaXMuX2ZpbGUubWltZXR5cGU7XG4gIH1cblxuICAvKiog7YyM7J28IO2BrOq4sCAoYnl0ZXMpICovXG4gIGdldCBzaXplKCk6IG51bWJlciB7XG4gICAgcmV0dXJuIHRoaXMuX2ZpbGUuZmlsZS5ieXRlc1JlYWQ7XG4gIH1cblxuICAvKiog7ZmV7J6l7J6QICjsoJAg7KCc7Jm4KSAqL1xuICBnZXQgZXh0bmFtZSgpOiBzdHJpbmcgfCBmYWxzZSB7XG4gICAgcmV0dXJuIG1pbWUuZXh0ZW5zaW9uKHRoaXMuX2ZpbGUubWltZXR5cGUpO1xuICB9XG5cbiAgLyoqIHNhdmVUb0Rpc2sg7ZuEIOyggOyepeuQnCBVUkwgKi9cbiAgZ2V0IHVybCgpOiBzdHJpbmcgfCB1bmRlZmluZWQge1xuICAgIHJldHVybiB0aGlzLl91cmw7XG4gIH1cblxuICAvKiogQnVmZmVy66GcIOuzgO2ZmCAo7LqQ7Iux65CoKSAqL1xuICBhc3luYyB0b0J1ZmZlcigpOiBQcm9taXNlPEJ1ZmZlcj4ge1xuICAgIGlmICghdGhpcy5fYnVmZmVyKSB7XG4gICAgICB0aGlzLl9idWZmZXIgPSBhd2FpdCB0aGlzLl9maWxlLnRvQnVmZmVyKCk7XG4gICAgfVxuICAgIHJldHVybiB0aGlzLl9idWZmZXI7XG4gIH1cblxuICAvKiogTUQ1IO2VtOyLnCDqs4TsgrAgKi9cbiAgYXN5bmMgbWQ1KCk6IFByb21pc2U8c3RyaW5nPiB7XG4gICAgY29uc3QgYnVmZmVyID0gYXdhaXQgdGhpcy50b0J1ZmZlcigpO1xuICAgIHJldHVybiBjcmVhdGVIYXNoKFwibWQ1XCIpLnVwZGF0ZShidWZmZXIpLmRpZ2VzdChcImhleFwiKTtcbiAgfVxuXG4gIC8qKlxuICAgKiDtjIzsnbzsnYQg65SU7Iqk7YGs7JeQIOyggOyepVxuICAgKiBAcGFyYW0ga2V5IOyggOyepSDqsr3roZwgKOyYiDogJ3VwbG9hZHMvYXZhdGFyLnBuZycpXG4gICAqIEBwYXJhbSBkaXNrTmFtZSDrlJTsiqTtgawg7J2066aEICjquLDrs7g6IGRlZmF1bHQgZGlzaylcbiAgICogQHJldHVybnMg7KCA7J6l65CcIO2MjOydvOydmCBVUkxcbiAgICovXG4gIGFzeW5jIHNhdmVUb0Rpc2soa2V5OiBzdHJpbmcsIGRpc2tOYW1lPzogRHJpdmVyS2V5KTogUHJvbWlzZTxzdHJpbmc+IHtcbiAgICAvLyDsiJztmZgg7J2Y7KG07ISxIOuwqeyngOulvCDsnITtlbQg64+Z7KCBIGltcG9ydFxuICAgIGNvbnN0IHsgU29uYW11IH0gPSBhd2FpdCBpbXBvcnQoXCIuLi9hcGkvc29uYW11XCIpO1xuICAgIGNvbnN0IGRpc2sgPSBTb25hbXUuc3RvcmFnZS51c2UoZGlza05hbWUpO1xuICAgIGNvbnN0IGJ1ZmZlciA9IGF3YWl0IHRoaXMudG9CdWZmZXIoKTtcblxuICAgIGF3YWl0IGRpc2sucHV0KGtleSwgbmV3IFVpbnQ4QXJyYXkoYnVmZmVyKSwge1xuICAgICAgY29udGVudFR5cGU6IHRoaXMubWltZXR5cGUsXG4gICAgfSk7XG5cbiAgICB0aGlzLl91cmwgPSBhd2FpdCBkaXNrLmdldFNpZ25lZFVybChrZXkpO1xuICAgIHJldHVybiB0aGlzLl91cmw7XG4gIH1cblxuICAvKiog7JuQ67O4IE11bHRpcGFydEZpbGUg7KCR6re8ICovXG4gIGdldCByYXcoKTogTXVsdGlwYXJ0RmlsZSB7XG4gICAgcmV0dXJuIHRoaXMuX2ZpbGU7XG4gIH1cbn1cbiJdLCJuYW1lcyI6WyJjcmVhdGVIYXNoIiwibWltZSIsIlVwbG9hZGVkRmlsZSIsIl9maWxlIiwiX2J1ZmZlciIsIl91cmwiLCJmaWxlIiwiZmlsZW5hbWUiLCJtaW1ldHlwZSIsInNpemUiLCJieXRlc1JlYWQiLCJleHRuYW1lIiwiZXh0ZW5zaW9uIiwidXJsIiwidG9CdWZmZXIiLCJtZDUiLCJidWZmZXIiLCJ1cGRhdGUiLCJkaWdlc3QiLCJzYXZlVG9EaXNrIiwia2V5IiwiZGlza05hbWUiLCJTb25hbXUiLCJkaXNrIiwic3RvcmFnZSIsInVzZSIsInB1dCIsIlVpbnQ4QXJyYXkiLCJjb250ZW50VHlwZSIsImdldFNpZ25lZFVybCIsInJhdyJdLCJtYXBwaW5ncyI6IkFBQ0EsU0FBU0EsVUFBVSxRQUFRLFNBQVM7QUFDcEMsT0FBT0MsVUFBVSxhQUFhO0FBRzlCOztDQUVDLEdBQ0QsT0FBTyxNQUFNQztJQUNIQyxNQUFxQjtJQUNyQkMsUUFBaUI7SUFDakJDLEtBQWM7SUFFdEIsWUFBWUMsSUFBbUIsQ0FBRTtRQUMvQixJQUFJLENBQUNILEtBQUssR0FBR0c7SUFDZjtJQUVBLFdBQVcsR0FDWCxJQUFJQyxXQUFtQjtRQUNyQixPQUFPLElBQUksQ0FBQ0osS0FBSyxDQUFDSSxRQUFRO0lBQzVCO0lBRUEsWUFBWSxHQUNaLElBQUlDLFdBQW1CO1FBQ3JCLE9BQU8sSUFBSSxDQUFDTCxLQUFLLENBQUNLLFFBQVE7SUFDNUI7SUFFQSxrQkFBa0IsR0FDbEIsSUFBSUMsT0FBZTtRQUNqQixPQUFPLElBQUksQ0FBQ04sS0FBSyxDQUFDRyxJQUFJLENBQUNJLFNBQVM7SUFDbEM7SUFFQSxlQUFlLEdBQ2YsSUFBSUMsVUFBMEI7UUFDNUIsT0FBT1YsS0FBS1csU0FBUyxDQUFDLElBQUksQ0FBQ1QsS0FBSyxDQUFDSyxRQUFRO0lBQzNDO0lBRUEseUJBQXlCLEdBQ3pCLElBQUlLLE1BQTBCO1FBQzVCLE9BQU8sSUFBSSxDQUFDUixJQUFJO0lBQ2xCO0lBRUEscUJBQXFCLEdBQ3JCLE1BQU1TLFdBQTRCO1FBQ2hDLElBQUksQ0FBQyxJQUFJLENBQUNWLE9BQU8sRUFBRTtZQUNqQixJQUFJLENBQUNBLE9BQU8sR0FBRyxNQUFNLElBQUksQ0FBQ0QsS0FBSyxDQUFDVyxRQUFRO1FBQzFDO1FBQ0EsT0FBTyxJQUFJLENBQUNWLE9BQU87SUFDckI7SUFFQSxjQUFjLEdBQ2QsTUFBTVcsTUFBdUI7UUFDM0IsTUFBTUMsU0FBUyxNQUFNLElBQUksQ0FBQ0YsUUFBUTtRQUNsQyxPQUFPZCxXQUFXLE9BQU9pQixNQUFNLENBQUNELFFBQVFFLE1BQU0sQ0FBQztJQUNqRDtJQUVBOzs7OztHQUtDLEdBQ0QsTUFBTUMsV0FBV0MsR0FBVyxFQUFFQyxRQUFvQixFQUFtQjtRQUNuRSwwQkFBMEI7UUFDMUIsTUFBTSxFQUFFQyxNQUFNLEVBQUUsR0FBRyxNQUFNLE1BQU0sQ0FBQztRQUNoQyxNQUFNQyxPQUFPRCxPQUFPRSxPQUFPLENBQUNDLEdBQUcsQ0FBQ0o7UUFDaEMsTUFBTUwsU0FBUyxNQUFNLElBQUksQ0FBQ0YsUUFBUTtRQUVsQyxNQUFNUyxLQUFLRyxHQUFHLENBQUNOLEtBQUssSUFBSU8sV0FBV1gsU0FBUztZQUMxQ1ksYUFBYSxJQUFJLENBQUNwQixRQUFRO1FBQzVCO1FBRUEsSUFBSSxDQUFDSCxJQUFJLEdBQUcsTUFBTWtCLEtBQUtNLFlBQVksQ0FBQ1Q7UUFDcEMsT0FBTyxJQUFJLENBQUNmLElBQUk7SUFDbEI7SUFFQSx3QkFBd0IsR0FDeEIsSUFBSXlCLE1BQXFCO1FBQ3ZCLE9BQU8sSUFBSSxDQUFDM0IsS0FBSztJQUNuQjtBQUNGIn0=
@@ -1 +1 @@
1
- {"version":3,"file":"services.template.d.ts","sourceRoot":"","sources":["../../../src/template/implementations/services.template.ts"],"names":[],"mappings":"AASA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,mBAAmB,CAAC;AAGzD,OAAO,EAAE,QAAQ,EAAE,MAAM,aAAa,CAAC;AAEvC,qBAAa,kBAAmB,SAAQ,QAAQ;;IAK9C,gBAAgB;;;;IAOhB,MAAM,CAAC,EAAE,EAAE,eAAe,CAAC,UAAU,CAAC;;;;;;;CAuMvC"}
1
+ {"version":3,"file":"services.template.d.ts","sourceRoot":"","sources":["../../../src/template/implementations/services.template.ts"],"names":[],"mappings":"AASA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,mBAAmB,CAAC;AAGzD,OAAO,EAAE,QAAQ,EAAE,MAAM,aAAa,CAAC;AAGvC,qBAAa,kBAAmB,SAAQ,QAAQ;;IAK9C,gBAAgB;;;;IAOhB,MAAM,CAAC,EAAE,EAAE,eAAe,CAAC,UAAU,CAAC;;;;;;;CA6OvC"}