sonamu 0.8.10 → 0.8.11

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 (55) hide show
  1. package/dist/api/config.d.ts +10 -0
  2. package/dist/api/config.d.ts.map +1 -1
  3. package/dist/api/config.js +1 -1
  4. package/dist/bin/test-command.d.ts.map +1 -1
  5. package/dist/bin/test-command.js +53 -10
  6. package/dist/migration/migrator.d.ts.map +1 -1
  7. package/dist/migration/migrator.js +43 -37
  8. package/dist/migration/postgresql-schema-reader.d.ts.map +1 -1
  9. package/dist/migration/postgresql-schema-reader.js +8 -2
  10. package/dist/stream/sse.js +2 -1
  11. package/dist/testing/fixture-manager.d.ts.map +1 -1
  12. package/dist/testing/fixture-manager.js +5 -3
  13. package/dist/ui/api.d.ts.map +1 -1
  14. package/dist/ui/api.js +14 -1
  15. package/dist/ui/cdd-service.d.ts +24 -0
  16. package/dist/ui/cdd-service.d.ts.map +1 -0
  17. package/dist/ui/cdd-service.js +179 -0
  18. package/dist/ui-web/assets/index-BwnPfLKr.css +1 -0
  19. package/dist/ui-web/assets/{index-C3UTzGvY.js → index-CcRgePSa.js} +78 -78
  20. package/dist/ui-web/index.html +2 -2
  21. package/package.json +3 -3
  22. package/src/api/config.ts +11 -0
  23. package/src/bin/test-command.ts +73 -9
  24. package/src/migration/migrator.ts +48 -44
  25. package/src/migration/postgresql-schema-reader.ts +7 -1
  26. package/src/shared/app.shared.ts.txt +11 -68
  27. package/src/skills/AGENTS.md +6 -0
  28. package/src/skills/sonamu/SKILL.md +16 -0
  29. package/src/skills/sonamu/ai-agents.md +204 -0
  30. package/src/skills/sonamu/api.md +51 -0
  31. package/src/skills/sonamu/auth-migration.md +200 -310
  32. package/src/skills/sonamu/auth-plugins.md +297 -0
  33. package/src/skills/sonamu/cone.md +36 -0
  34. package/src/skills/sonamu/config.md +62 -1
  35. package/src/skills/sonamu/create-sonamu.md +33 -0
  36. package/src/skills/sonamu/entity-basic.md +74 -355
  37. package/src/skills/sonamu/entity-relations.md +10 -0
  38. package/src/skills/sonamu/entity-validation-checklist.md +1 -1
  39. package/src/skills/sonamu/fixture-cli.md +19 -1
  40. package/src/skills/sonamu/frontend.md +149 -451
  41. package/src/skills/sonamu/migration.md +12 -1
  42. package/src/skills/sonamu/model.md +32 -1
  43. package/src/skills/sonamu/naite.md +264 -0
  44. package/src/skills/sonamu/skill-contribution.md +241 -0
  45. package/src/skills/sonamu/tasks.md +221 -0
  46. package/src/skills/sonamu/testing-devrunner.md +390 -0
  47. package/src/skills/sonamu/testing.md +396 -1146
  48. package/src/skills/sonamu/upsert.md +18 -0
  49. package/src/skills/sonamu/vector.md +221 -0
  50. package/src/skills/sonamu/workflow.md +4 -2
  51. package/src/stream/sse.ts +2 -0
  52. package/src/testing/fixture-manager.ts +1 -2
  53. package/src/ui/api.ts +16 -0
  54. package/src/ui/cdd-service.ts +208 -0
  55. package/dist/ui-web/assets/index-B3NU_uZ2.css +0 -1
@@ -102,6 +102,16 @@ export type SonamuConfig<TSinkId extends string = string, TFilterId extends stri
102
102
  * ```
103
103
  */
104
104
  test?: SonamuTestConfig;
105
+ /**
106
+ * 외부 에디터 설정 (CDD 등에서 파일 편집 시 사용)
107
+ * 미설정 시 "vscode"를 기본값으로 사용합니다.
108
+ *
109
+ * @example
110
+ * ```typescript
111
+ * externalEditor: "cursor"
112
+ * ```
113
+ */
114
+ externalEditor?: "Visual Studio Code" | "Zed" | "Cursor";
105
115
  /**
106
116
  * Slack 승인 설정 (Production 마이그레이션용)
107
117
  *
@@ -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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"}
@@ -26,4 +26,4 @@ export function defineConfig(config) {
26
26
  return config;
27
27
  }
28
28
 
29
- //# sourceMappingURL=data:application/json;base64,{"version":3,"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 type { FastifyInstance, FastifyReply, FastifyRequest, FastifyServerOptions } from \"fastify\";\nimport type { QsPluginOptions } from \"fastify-qs\";\nimport type { SsePluginOptions } from \"fastify-sse-v2/lib/types\";\nimport type { Knex } from \"knex\";\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, 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   * 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   * DRIVE_DISK 환경변수로 사용할 드라이버를 선택합니다. (기본값: default 키)\n   *\n   * @example\n   * ```typescript\n   * import { drivers } from \"sonamu/storage\";\n   *\n   * storage: {\n   *   default: process.env.DRIVE_DISK ?? \"fs\",\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 start = performance.now();\n  const configPath =\n    process.env.HOT === \"yes\" || process.env.VITEST === \"true\"\n      ? `${rootPath}/src/sonamu.config.ts`\n      : `${rootPath}/dist/sonamu.config.js`;\n  const { default: config } = await import(`file://${configPath}`);\n  const importTime = performance.now() - start;\n  process.env.NODE_ENV !== \"test\" &&\n    console.log(`[TIMING] loadConfig took ${importTime.toFixed(2)}ms`);\n  return config;\n}\n"],"names":["defineConfig","config","Promise","resolve","loadConfig","rootPath","start","performance","now","configPath","process","env","HOT","VITEST","default","importTime","NODE_ENV","console","log","toFixed"],"mappings":"AAgQA,uFAAuF;AACvF,OAAO,SAASA,aAAaC,MAAgC;IAC3D,IAAI,OAAOA,WAAW,YAAY;QAChC,OAAOC,QAAQC,OAAO,CAACF;IACzB;IAEA,OAAOC,QAAQC,OAAO,CAACF;AACzB;AAEA;;;;;;;;;;;;CAYC,GACD,OAAO,eAAeG,WAAWC,QAAgB;IAC/C,MAAMC,QAAQC,YAAYC,GAAG;IAC7B,MAAMC,aACJC,QAAQC,GAAG,CAACC,GAAG,KAAK,SAASF,QAAQC,GAAG,CAACE,MAAM,KAAK,SAChD,GAAGR,SAAS,qBAAqB,CAAC,GAClC,GAAGA,SAAS,sBAAsB,CAAC;IACzC,MAAM,EAAES,SAASb,MAAM,EAAE,GAAG,MAAM,MAAM,CAAC,CAAC,OAAO,EAAEQ,YAAY;IAC/D,MAAMM,aAAaR,YAAYC,GAAG,KAAKF;IACvCI,QAAQC,GAAG,CAACK,QAAQ,KAAK,UACvBC,QAAQC,GAAG,CAAC,CAAC,yBAAyB,EAAEH,WAAWI,OAAO,CAAC,GAAG,EAAE,CAAC;IACnE,OAAOlB;AACT"}
29
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"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 type { FastifyInstance, FastifyReply, FastifyRequest, FastifyServerOptions } from \"fastify\";\nimport type { QsPluginOptions } from \"fastify-qs\";\nimport type { SsePluginOptions } from \"fastify-sse-v2/lib/types\";\nimport type { Knex } from \"knex\";\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, 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   * DRIVE_DISK 환경변수로 사용할 드라이버를 선택합니다. (기본값: default 키)\n   *\n   * @example\n   * ```typescript\n   * import { drivers } from \"sonamu/storage\";\n   *\n   * storage: {\n   *   default: process.env.DRIVE_DISK ?? \"fs\",\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 start = performance.now();\n  const configPath =\n    process.env.HOT === \"yes\" || process.env.VITEST === \"true\"\n      ? `${rootPath}/src/sonamu.config.ts`\n      : `${rootPath}/dist/sonamu.config.js`;\n  const { default: config } = await import(`file://${configPath}`);\n  const importTime = performance.now() - start;\n  process.env.NODE_ENV !== \"test\" &&\n    console.log(`[TIMING] loadConfig took ${importTime.toFixed(2)}ms`);\n  return config;\n}\n"],"names":["defineConfig","config","Promise","resolve","loadConfig","rootPath","start","performance","now","configPath","process","env","HOT","VITEST","default","importTime","NODE_ENV","console","log","toFixed"],"mappings":"AA2QA,uFAAuF;AACvF,OAAO,SAASA,aAAaC,MAAgC;IAC3D,IAAI,OAAOA,WAAW,YAAY;QAChC,OAAOC,QAAQC,OAAO,CAACF;IACzB;IAEA,OAAOC,QAAQC,OAAO,CAACF;AACzB;AAEA;;;;;;;;;;;;CAYC,GACD,OAAO,eAAeG,WAAWC,QAAgB;IAC/C,MAAMC,QAAQC,YAAYC,GAAG;IAC7B,MAAMC,aACJC,QAAQC,GAAG,CAACC,GAAG,KAAK,SAASF,QAAQC,GAAG,CAACE,MAAM,KAAK,SAChD,GAAGR,SAAS,qBAAqB,CAAC,GAClC,GAAGA,SAAS,sBAAsB,CAAC;IACzC,MAAM,EAAES,SAASb,MAAM,EAAE,GAAG,MAAM,MAAM,CAAC,CAAC,OAAO,EAAEQ,YAAY;IAC/D,MAAMM,aAAaR,YAAYC,GAAG,KAAKF;IACvCI,QAAQC,GAAG,CAACK,QAAQ,KAAK,UACvBC,QAAQC,GAAG,CAAC,CAAC,yBAAyB,EAAEH,WAAWI,OAAO,CAAC,GAAG,EAAE,CAAC;IACnE,OAAOlB;AACT"}
@@ -1 +1 @@
1
- {"version":3,"file":"test-command.d.ts","sourceRoot":"","sources":["../../src/bin/test-command.ts"],"names":[],"mappings":"AAOA,wBAAsB,WAAW,IAAI,OAAO,CAAC,IAAI,CAAC,CA0IjD"}
1
+ {"version":3,"file":"test-command.d.ts","sourceRoot":"","sources":["../../src/bin/test-command.ts"],"names":[],"mappings":"AAkCA,wBAAsB,WAAW,IAAI,OAAO,CAAC,IAAI,CAAC,CAmIjD"}
@@ -2,14 +2,12 @@ import path from "node:path";
2
2
  import chalk from "chalk";
3
3
  import { loadConfig } from "../api/config.js";
4
4
  import { findApiRootPath } from "../utils/utils.js";
5
- export async function testCommand() {
6
- // VITEST=true 임시 설정으로 loadConfig가 src/sonamu.config.ts 경로를 사용하도록 함
5
+ async function loadTestConfig() {
7
6
  const prevVitest = process.env.VITEST;
8
7
  process.env.VITEST = "true";
9
- let config;
10
8
  try {
11
9
  const apiRootPath = findApiRootPath();
12
- config = await loadConfig(apiRootPath);
10
+ return await loadConfig(apiRootPath);
13
11
  } finally{
14
12
  if (prevVitest === undefined) {
15
13
  delete process.env.VITEST;
@@ -17,29 +15,47 @@ export async function testCommand() {
17
15
  process.env.VITEST = prevVitest;
18
16
  }
19
17
  }
20
- // process.argv 파싱: sonamu test [file...] --pattern "이름" --traces
18
+ }
19
+ function resolveTestBaseUrl(config) {
20
+ const port = config.server.listen?.port ?? 3000;
21
+ const host = config.server.listen?.host ?? "localhost";
22
+ const routePrefix = config.test?.devRunner?.routePrefix ?? "/__test__";
23
+ return {
24
+ host,
25
+ port,
26
+ routePrefix,
27
+ baseUrl: `http://${host}:${port}${routePrefix}`
28
+ };
29
+ }
30
+ export async function testCommand() {
21
31
  const args = process.argv.slice(3);
32
+ const config = await loadTestConfig();
33
+ // process.argv 파싱: sonamu test [file...] --pattern "이름" --traces --status
22
34
  const files = [];
23
35
  let pattern;
24
36
  let showTraces = false;
37
+ let showStatus = false;
25
38
  for(let i = 0; i < args.length; i++){
26
39
  const arg = args[i];
27
40
  if (arg === "--pattern" || arg === "-p") {
28
41
  pattern = args[++i];
29
42
  } else if (arg === "--traces" || arg === "-t") {
30
43
  showTraces = true;
44
+ } else if (arg === "--status" || arg === "-s") {
45
+ showStatus = true;
31
46
  } else if (!arg.startsWith("-")) {
32
47
  files.push(arg);
33
48
  }
34
49
  }
50
+ if (showStatus) {
51
+ return testStatusCommand(config);
52
+ }
35
53
  if (!config.test?.devRunner?.enabled) {
36
54
  console.error(chalk.red("devRunner가 활성화되지 않았습니다. sonamu.config.ts에서 test.devRunner.enabled: true 설정이 필요합니다"));
37
55
  process.exit(1);
38
56
  }
39
- const port = config.server.listen?.port ?? 3000;
40
- const host = config.server.listen?.host ?? "localhost";
41
- const routePrefix = config.test?.devRunner?.routePrefix ?? "/__test__";
42
- const url = `http://${host}:${port}${routePrefix}/run`;
57
+ const { baseUrl } = resolveTestBaseUrl(config);
58
+ const url = `${baseUrl}/run`;
43
59
  const payload = {};
44
60
  if (files.length > 0) {
45
61
  payload.files = files;
@@ -143,5 +159,32 @@ function collectTracesFromResults(nodes) {
143
159
  }
144
160
  return result;
145
161
  }
162
+ async function testStatusCommand(config) {
163
+ if (!config.test?.devRunner?.enabled) {
164
+ console.error(chalk.red("devRunner가 활성화되지 않았습니다. sonamu.config.ts에서 test.devRunner.enabled: true 설정이 필요합니다"));
165
+ process.exit(1);
166
+ }
167
+ const { baseUrl } = resolveTestBaseUrl(config);
168
+ const url = `${baseUrl}/status`;
169
+ try {
170
+ const response = await fetch(url);
171
+ if (!response.ok) {
172
+ console.error(chalk.red(`예상하지 못한 응답: ${response.status}`));
173
+ process.exit(1);
174
+ }
175
+ const status = await response.json();
176
+ console.log(chalk.bold("DevRunner 상태:"));
177
+ console.log(` ready: ${status.ready ? chalk.green("true") : chalk.red("false")}`);
178
+ console.log(` running: ${status.running ? chalk.yellow("true") : "false"}`);
179
+ console.log(` lastRunAt: ${status.lastRunAt ?? chalk.dim("없음")}`);
180
+ console.log(` sseAvailable: ${status.sseAvailable ? chalk.green("true") : "false"}`);
181
+ } catch (err) {
182
+ if (err instanceof TypeError && err.cause) {
183
+ console.error(chalk.red("dev 서버에 연결할 수 없습니다. sonamu dev가 실행 중인지 확인하세요"));
184
+ process.exit(1);
185
+ }
186
+ throw err;
187
+ }
188
+ }
146
189
 
147
- //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/bin/test-command.ts"],"sourcesContent":["import path from \"node:path\";\nimport chalk from \"chalk\";\nimport type { SonamuConfig } from \"../api/config\";\nimport { loadConfig } from \"../api/config\";\nimport type { RunResult, TestCaseResult } from \"../testing\";\nimport { findApiRootPath } from \"../utils/utils\";\n\nexport async function testCommand(): Promise<void> {\n  // VITEST=true 임시 설정으로 loadConfig가 src/sonamu.config.ts 경로를 사용하도록 함\n  const prevVitest = process.env.VITEST;\n  process.env.VITEST = \"true\";\n  let config: SonamuConfig;\n  try {\n    const apiRootPath = findApiRootPath();\n    config = await loadConfig(apiRootPath);\n  } finally {\n    if (prevVitest === undefined) {\n      delete process.env.VITEST;\n    } else {\n      process.env.VITEST = prevVitest;\n    }\n  }\n\n  // process.argv 파싱: sonamu test [file...] --pattern \"이름\" --traces\n  const args = process.argv.slice(3);\n  const files: string[] = [];\n  let pattern: string | undefined;\n  let showTraces = false;\n\n  for (let i = 0; i < args.length; i++) {\n    const arg = args[i];\n    if (arg === \"--pattern\" || arg === \"-p\") {\n      pattern = args[++i];\n    } else if (arg === \"--traces\" || arg === \"-t\") {\n      showTraces = true;\n    } else if (!arg.startsWith(\"-\")) {\n      files.push(arg);\n    }\n  }\n\n  if (!config.test?.devRunner?.enabled) {\n    console.error(\n      chalk.red(\n        \"devRunner가 활성화되지 않았습니다. sonamu.config.ts에서 test.devRunner.enabled: true 설정이 필요합니다\",\n      ),\n    );\n    process.exit(1);\n  }\n\n  const port = config.server.listen?.port ?? 3000;\n  const host = config.server.listen?.host ?? \"localhost\";\n  const routePrefix = config.test?.devRunner?.routePrefix ?? \"/__test__\";\n  const url = `http://${host}:${port}${routePrefix}/run`;\n\n  const payload: { files?: string[]; pattern?: string } = {};\n  if (files.length > 0) {\n    payload.files = files;\n  }\n  if (pattern) {\n    payload.pattern = pattern;\n  }\n\n  try {\n    const response = await fetch(url, {\n      method: \"POST\",\n      headers: { \"Content-Type\": \"application/json\" },\n      body: JSON.stringify(payload),\n    });\n\n    if (response.status === 404) {\n      console.error(\n        chalk.red(\n          \"devRunner가 활성화되지 않았습니다. sonamu.config.ts에서 test.devRunner.enabled: true 설정이 필요합니다\",\n        ),\n      );\n      process.exit(1);\n    }\n\n    if (response.status === 500) {\n      const errorBody = (await response.json()) as { error?: string };\n      console.error(chalk.red(\"Vitest 인스턴스가 아직 준비되지 않았습니다\"));\n      if (errorBody.error) {\n        console.error(chalk.red(errorBody.error));\n      }\n      process.exit(1);\n    }\n\n    if (!response.ok) {\n      console.error(chalk.red(`예상하지 못한 응답: ${response.status}`));\n      process.exit(1);\n    }\n\n    const result = (await response.json()) as RunResult;\n\n    const { passed, failed: failedCount, total, durationMs } = result.summary;\n    const passedStr = chalk.green(`${passed} passed`);\n    const failedStr =\n      failedCount > 0 ? chalk.red(`${failedCount} failed`) : `${failedCount} failed`;\n    console.log(`\\nTests: ${passedStr}, ${failedStr}, ${total} total`);\n    console.log(chalk.dim(`Duration: ${durationMs}ms`));\n\n    const failedTests = collectFailedFromResults(result.results);\n    if (failedTests.length > 0) {\n      console.log(chalk.red.bold(\"\\nFailed tests:\"));\n      for (const f of failedTests) {\n        console.log(`  ${chalk.red(\"x\")} ${f.fullName} ${chalk.dim(`(${f.file})`)}`);\n        if (f.error) {\n          console.log(`    ${chalk.red(f.error.message)}`);\n        }\n      }\n    }\n\n    if (showTraces) {\n      const testsWithTraces = collectTracesFromResults(result.results);\n      if (testsWithTraces.length > 0) {\n        console.log(chalk.cyan.bold(\"\\nTraces:\"));\n        for (const { testName, file, traces } of testsWithTraces) {\n          console.log(`\\n  ${chalk.bold(testName)}`);\n          console.log(`  ${chalk.dim(path.basename(file))}`);\n          for (const trace of traces) {\n            const loc = `${path.basename(trace.filePath)}:${trace.lineNumber}`;\n            const valueStr =\n              typeof trace.value === \"string\"\n                ? trace.value\n                : (JSON.stringify(trace.value, null, 2) ?? \"undefined\");\n            console.log(`\\n    ${chalk.yellow(`[${trace.key}]`)} ${chalk.dim(loc)}`);\n            const indented = valueStr.split(\"\\n\").join(\"\\n    \");\n            console.log(`    ${indented}`);\n          }\n        }\n      }\n    }\n\n    if (!result.ok) {\n      process.exit(1);\n    }\n  } catch (err) {\n    if (err instanceof TypeError && err.cause) {\n      console.error(\n        chalk.red(\"dev 서버에 연결할 수 없습니다. sonamu dev가 실행 중인지 확인하세요\"),\n      );\n      process.exit(1);\n    }\n    throw err;\n  }\n}\n\nfunction collectFailedFromResults(nodes: TestCaseResult[]): TestCaseResult[] {\n  const result: TestCaseResult[] = [];\n  for (const node of nodes) {\n    if (node.kind === \"test\" && node.state === \"failed\") {\n      result.push(node);\n    }\n    if (node.children.length > 0) {\n      result.push(...collectFailedFromResults(node.children));\n    }\n  }\n  return result;\n}\n\nfunction collectTracesFromResults(\n  nodes: TestCaseResult[],\n): { testName: string; file: string; traces: TestCaseResult[\"traces\"] }[] {\n  const result: { testName: string; file: string; traces: TestCaseResult[\"traces\"] }[] = [];\n  for (const node of nodes) {\n    if (node.kind === \"test\" && node.traces.length > 0) {\n      result.push({ testName: node.fullName, file: node.file, traces: node.traces });\n    }\n    if (node.children.length > 0) {\n      result.push(...collectTracesFromResults(node.children));\n    }\n  }\n  return result;\n}\n"],"names":["path","chalk","loadConfig","findApiRootPath","testCommand","prevVitest","process","env","VITEST","config","apiRootPath","undefined","args","argv","slice","files","pattern","showTraces","i","length","arg","startsWith","push","test","devRunner","enabled","console","error","red","exit","port","server","listen","host","routePrefix","url","payload","response","fetch","method","headers","body","JSON","stringify","status","errorBody","json","ok","result","passed","failed","failedCount","total","durationMs","summary","passedStr","green","failedStr","log","dim","failedTests","collectFailedFromResults","results","bold","f","fullName","file","message","testsWithTraces","collectTracesFromResults","cyan","testName","traces","basename","trace","loc","filePath","lineNumber","valueStr","value","yellow","key","indented","split","join","err","TypeError","cause","nodes","node","kind","state","children"],"mappings":"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"}
190
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/bin/test-command.ts"],"sourcesContent":["import path from \"node:path\";\nimport chalk from \"chalk\";\nimport type { SonamuConfig } from \"../api/config\";\nimport { loadConfig } from \"../api/config\";\nimport type { RunResult, TestCaseResult } from \"../testing\";\nimport { findApiRootPath } from \"../utils/utils\";\n\nasync function loadTestConfig(): Promise<SonamuConfig> {\n  const prevVitest = process.env.VITEST;\n  process.env.VITEST = \"true\";\n  try {\n    const apiRootPath = findApiRootPath();\n    return await loadConfig(apiRootPath);\n  } finally {\n    if (prevVitest === undefined) {\n      delete process.env.VITEST;\n    } else {\n      process.env.VITEST = prevVitest;\n    }\n  }\n}\n\nfunction resolveTestBaseUrl(config: SonamuConfig): {\n  host: string;\n  port: number;\n  routePrefix: string;\n  baseUrl: string;\n} {\n  const port = config.server.listen?.port ?? 3000;\n  const host = config.server.listen?.host ?? \"localhost\";\n  const routePrefix = config.test?.devRunner?.routePrefix ?? \"/__test__\";\n  return { host, port, routePrefix, baseUrl: `http://${host}:${port}${routePrefix}` };\n}\n\nexport async function testCommand(): Promise<void> {\n  const args = process.argv.slice(3);\n\n  const config = await loadTestConfig();\n\n  // process.argv 파싱: sonamu test [file...] --pattern \"이름\" --traces --status\n  const files: string[] = [];\n  let pattern: string | undefined;\n  let showTraces = false;\n  let showStatus = false;\n\n  for (let i = 0; i < args.length; i++) {\n    const arg = args[i];\n    if (arg === \"--pattern\" || arg === \"-p\") {\n      pattern = args[++i];\n    } else if (arg === \"--traces\" || arg === \"-t\") {\n      showTraces = true;\n    } else if (arg === \"--status\" || arg === \"-s\") {\n      showStatus = true;\n    } else if (!arg.startsWith(\"-\")) {\n      files.push(arg);\n    }\n  }\n\n  if (showStatus) {\n    return testStatusCommand(config);\n  }\n\n  if (!config.test?.devRunner?.enabled) {\n    console.error(\n      chalk.red(\n        \"devRunner가 활성화되지 않았습니다. sonamu.config.ts에서 test.devRunner.enabled: true 설정이 필요합니다\",\n      ),\n    );\n    process.exit(1);\n  }\n\n  const { baseUrl } = resolveTestBaseUrl(config);\n  const url = `${baseUrl}/run`;\n\n  const payload: { files?: string[]; pattern?: string } = {};\n  if (files.length > 0) {\n    payload.files = files;\n  }\n  if (pattern) {\n    payload.pattern = pattern;\n  }\n\n  try {\n    const response = await fetch(url, {\n      method: \"POST\",\n      headers: { \"Content-Type\": \"application/json\" },\n      body: JSON.stringify(payload),\n    });\n\n    if (response.status === 404) {\n      console.error(\n        chalk.red(\n          \"devRunner가 활성화되지 않았습니다. sonamu.config.ts에서 test.devRunner.enabled: true 설정이 필요합니다\",\n        ),\n      );\n      process.exit(1);\n    }\n\n    if (response.status === 500) {\n      const errorBody = (await response.json()) as { error?: string };\n      console.error(chalk.red(\"Vitest 인스턴스가 아직 준비되지 않았습니다\"));\n      if (errorBody.error) {\n        console.error(chalk.red(errorBody.error));\n      }\n      process.exit(1);\n    }\n\n    if (!response.ok) {\n      console.error(chalk.red(`예상하지 못한 응답: ${response.status}`));\n      process.exit(1);\n    }\n\n    const result = (await response.json()) as RunResult;\n\n    const { passed, failed: failedCount, total, durationMs } = result.summary;\n    const passedStr = chalk.green(`${passed} passed`);\n    const failedStr =\n      failedCount > 0 ? chalk.red(`${failedCount} failed`) : `${failedCount} failed`;\n    console.log(`\\nTests: ${passedStr}, ${failedStr}, ${total} total`);\n    console.log(chalk.dim(`Duration: ${durationMs}ms`));\n\n    const failedTests = collectFailedFromResults(result.results);\n    if (failedTests.length > 0) {\n      console.log(chalk.red.bold(\"\\nFailed tests:\"));\n      for (const f of failedTests) {\n        console.log(`  ${chalk.red(\"x\")} ${f.fullName} ${chalk.dim(`(${f.file})`)}`);\n        if (f.error) {\n          console.log(`    ${chalk.red(f.error.message)}`);\n        }\n      }\n    }\n\n    if (showTraces) {\n      const testsWithTraces = collectTracesFromResults(result.results);\n      if (testsWithTraces.length > 0) {\n        console.log(chalk.cyan.bold(\"\\nTraces:\"));\n        for (const { testName, file, traces } of testsWithTraces) {\n          console.log(`\\n  ${chalk.bold(testName)}`);\n          console.log(`  ${chalk.dim(path.basename(file))}`);\n          for (const trace of traces) {\n            const loc = `${path.basename(trace.filePath)}:${trace.lineNumber}`;\n            const valueStr =\n              typeof trace.value === \"string\"\n                ? trace.value\n                : (JSON.stringify(trace.value, null, 2) ?? \"undefined\");\n            console.log(`\\n    ${chalk.yellow(`[${trace.key}]`)} ${chalk.dim(loc)}`);\n            const indented = valueStr.split(\"\\n\").join(\"\\n    \");\n            console.log(`    ${indented}`);\n          }\n        }\n      }\n    }\n\n    if (!result.ok) {\n      process.exit(1);\n    }\n  } catch (err) {\n    if (err instanceof TypeError && err.cause) {\n      console.error(\n        chalk.red(\"dev 서버에 연결할 수 없습니다. sonamu dev가 실행 중인지 확인하세요\"),\n      );\n      process.exit(1);\n    }\n    throw err;\n  }\n}\n\nfunction collectFailedFromResults(nodes: TestCaseResult[]): TestCaseResult[] {\n  const result: TestCaseResult[] = [];\n  for (const node of nodes) {\n    if (node.kind === \"test\" && node.state === \"failed\") {\n      result.push(node);\n    }\n    if (node.children.length > 0) {\n      result.push(...collectFailedFromResults(node.children));\n    }\n  }\n  return result;\n}\n\nfunction collectTracesFromResults(\n  nodes: TestCaseResult[],\n): { testName: string; file: string; traces: TestCaseResult[\"traces\"] }[] {\n  const result: { testName: string; file: string; traces: TestCaseResult[\"traces\"] }[] = [];\n  for (const node of nodes) {\n    if (node.kind === \"test\" && node.traces.length > 0) {\n      result.push({ testName: node.fullName, file: node.file, traces: node.traces });\n    }\n    if (node.children.length > 0) {\n      result.push(...collectTracesFromResults(node.children));\n    }\n  }\n  return result;\n}\n\nasync function testStatusCommand(config: SonamuConfig): Promise<void> {\n  if (!config.test?.devRunner?.enabled) {\n    console.error(\n      chalk.red(\n        \"devRunner가 활성화되지 않았습니다. sonamu.config.ts에서 test.devRunner.enabled: true 설정이 필요합니다\",\n      ),\n    );\n    process.exit(1);\n  }\n\n  const { baseUrl } = resolveTestBaseUrl(config);\n  const url = `${baseUrl}/status`;\n\n  try {\n    const response = await fetch(url);\n\n    if (!response.ok) {\n      console.error(chalk.red(`예상하지 못한 응답: ${response.status}`));\n      process.exit(1);\n    }\n\n    const status = (await response.json()) as {\n      ready: boolean;\n      running: boolean;\n      lastRunAt: string | null;\n      sseAvailable: boolean;\n    };\n\n    console.log(chalk.bold(\"DevRunner 상태:\"));\n    console.log(`  ready:        ${status.ready ? chalk.green(\"true\") : chalk.red(\"false\")}`);\n    console.log(`  running:      ${status.running ? chalk.yellow(\"true\") : \"false\"}`);\n    console.log(`  lastRunAt:    ${status.lastRunAt ?? chalk.dim(\"없음\")}`);\n    console.log(`  sseAvailable: ${status.sseAvailable ? chalk.green(\"true\") : \"false\"}`);\n  } catch (err) {\n    if (err instanceof TypeError && err.cause) {\n      console.error(\n        chalk.red(\"dev 서버에 연결할 수 없습니다. sonamu dev가 실행 중인지 확인하세요\"),\n      );\n      process.exit(1);\n    }\n    throw err;\n  }\n}\n"],"names":["path","chalk","loadConfig","findApiRootPath","loadTestConfig","prevVitest","process","env","VITEST","apiRootPath","undefined","resolveTestBaseUrl","config","port","server","listen","host","routePrefix","test","devRunner","baseUrl","testCommand","args","argv","slice","files","pattern","showTraces","showStatus","i","length","arg","startsWith","push","testStatusCommand","enabled","console","error","red","exit","url","payload","response","fetch","method","headers","body","JSON","stringify","status","errorBody","json","ok","result","passed","failed","failedCount","total","durationMs","summary","passedStr","green","failedStr","log","dim","failedTests","collectFailedFromResults","results","bold","f","fullName","file","message","testsWithTraces","collectTracesFromResults","cyan","testName","traces","basename","trace","loc","filePath","lineNumber","valueStr","value","yellow","key","indented","split","join","err","TypeError","cause","nodes","node","kind","state","children","ready","running","lastRunAt","sseAvailable"],"mappings":"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"}
@@ -1 +1 @@
1
- {"version":3,"file":"migrator.d.ts","sourceRoot":"","sources":["../../src/migration/migrator.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAIjC,OAAO,EAAM,KAAK,cAAc,EAAE,MAAM,gBAAgB,CAAC;AAMzD,OAAO,KAAK,EAAE,gBAAgB,EAAgB,MAAM,gBAAgB,CAAC;AAMrE,OAAO,KAAK,EAA6B,eAAe,EAAE,MAAM,SAAS,CAAC;AAE1E,MAAM,MAAM,eAAe,GAAG;IAC5B,OAAO,EAAE,MAAM,CAAC;IAChB,OAAO,EAAE,MAAM,CAAC;IAChB,OAAO,EAAE,MAAM,EAAE,CAAC;CACnB,EAAE,CAAC;AAEJ,qBAAa,QAAQ;YACL,iBAAiB;IAqB/B;;;;;;;;OAQG;IACG,SAAS,IAAI,OAAO,CAAC,eAAe,CAAC;IAiG3C;;;;;;;;;OASG;IACG,SAAS,CACb,MAAM,EAAE,OAAO,GAAG,UAAU,EAC5B,OAAO,EAAE,CAAC,MAAM,cAAc,CAAC,EAAE,GAChC,OAAO,CAAC,eAAe,CAAC;IAmE3B;;;;;;OAMG;IACH,iBAAiB,CAAC,KAAK,EAAE,eAAe,CAAC,OAAO,CAAC,EAAE,SAAS,EAAE,MAAM,EAAE;;;;IAWtE;;;;;;;OAOG;IACG,QAAQ,CAAC,SAAS,EAAE,MAAM,EAAE,GAAG,OAAO,CAAC,MAAM,CAAC;IAuBpD,OAAO,CAAC,UAAU;IAalB;;;;;;OAMG;IACG,qBAAqB,IAAI,OAAO,CAAC,MAAM,CAAC;IAuBxC,iBAAiB,CAAC,SAAS,EAAE,IAAI,GAAG,OAAO,CAAC,gBAAgB,EAAE,CAAC;IAgErE;;;;;;OAMG;IACG,aAAa,IAAI,OAAO,CAAC,eAAe,CAAC;CAkEhD"}
1
+ {"version":3,"file":"migrator.d.ts","sourceRoot":"","sources":["../../src/migration/migrator.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAIjC,OAAO,EAAM,KAAK,cAAc,EAAE,MAAM,gBAAgB,CAAC;AAMzD,OAAO,KAAK,EAAE,gBAAgB,EAAgB,MAAM,gBAAgB,CAAC;AAMrE,OAAO,KAAK,EAA6B,eAAe,EAAE,MAAM,SAAS,CAAC;AAE1E,MAAM,MAAM,eAAe,GAAG;IAC5B,OAAO,EAAE,MAAM,CAAC;IAChB,OAAO,EAAE,MAAM,CAAC;IAChB,OAAO,EAAE,MAAM,EAAE,CAAC;CACnB,EAAE,CAAC;AAEJ,qBAAa,QAAQ;YACL,iBAAiB;IAqB/B;;;;;;;;OAQG;IACG,SAAS,IAAI,OAAO,CAAC,eAAe,CAAC;IAiG3C;;;;;;;;;OASG;IACG,SAAS,CACb,MAAM,EAAE,OAAO,GAAG,UAAU,EAC5B,OAAO,EAAE,CAAC,MAAM,cAAc,CAAC,EAAE,GAChC,OAAO,CAAC,eAAe,CAAC;IAmE3B;;;;;;OAMG;IACH,iBAAiB,CAAC,KAAK,EAAE,eAAe,CAAC,OAAO,CAAC,EAAE,SAAS,EAAE,MAAM,EAAE;;;;IAWtE;;;;;;;OAOG;IACG,QAAQ,CAAC,SAAS,EAAE,MAAM,EAAE,GAAG,OAAO,CAAC,MAAM,CAAC;IAuBpD,OAAO,CAAC,UAAU;IAalB;;;;;;OAMG;IACG,qBAAqB,IAAI,OAAO,CAAC,MAAM,CAAC;IAuBxC,iBAAiB,CAAC,SAAS,EAAE,IAAI,GAAG,OAAO,CAAC,gBAAgB,EAAE,CAAC;IAgErE;;;;;;OAMG;IACG,aAAa,IAAI,OAAO,CAAC,eAAe,CAAC;CAsEhD"}
@@ -297,51 +297,57 @@ export class Migrator {
297
297
  }
298
298
  // 기존 Shadow DB 삭제 후 Shadow DB 생성
299
299
  const tdb = createKnexInstance(Sonamu.dbConfig.test);
300
- !isTest() && console.log(chalk.magenta(`${shadowDatabase} 삭제`));
301
- await tdb.raw(`DROP DATABASE IF EXISTS ${shadowDatabase}`);
302
- await tdb.raw(`
303
- SELECT pg_terminate_backend(pg_stat_activity.pid)
304
- FROM pg_stat_activity
305
- WHERE datname = '${tdbConn.database}'
306
- AND pid <> pg_backend_pid();
307
- `);
308
- await tdb.raw(`CREATE DATABASE ${shadowDatabase} TEMPLATE ${tdbConn.database}`);
309
- // Shadow DB에 연결
310
- const sdb = createKnexInstance({
311
- ...Sonamu.dbConfig.test,
312
- connection: {
313
- ...tdbConn,
314
- database: shadowDatabase,
315
- password: tdbConn.password
316
- }
317
- });
318
- // shadow DB 테스트 진행
319
300
  try {
320
- const [batchNo, applied] = await sdb.migrate.latest();
321
- !isTest() && console.log(chalk.green("Shadow DB 테스트에 성공했습니다!"), {
322
- batchNo,
323
- applied
301
+ !isTest() && console.log(chalk.magenta(`${shadowDatabase} 삭제`));
302
+ await tdb.raw(`DROP DATABASE IF EXISTS ${shadowDatabase}`);
303
+ await tdb.raw(`
304
+ SELECT pg_terminate_backend(pg_stat_activity.pid)
305
+ FROM pg_stat_activity
306
+ WHERE datname = '${tdbConn.database}'
307
+ AND pid <> pg_backend_pid();
308
+ `);
309
+ await tdb.raw(`CREATE DATABASE ${shadowDatabase} TEMPLATE ${tdbConn.database}`);
310
+ // Shadow DB에 연결
311
+ const sdb = createKnexInstance({
312
+ ...Sonamu.dbConfig.test,
313
+ connection: {
314
+ ...tdbConn,
315
+ database: shadowDatabase,
316
+ password: tdbConn.password
317
+ }
324
318
  });
325
- return [
326
- {
327
- connKey: "shadow",
319
+ // shadow DB 테스트 진행
320
+ try {
321
+ const [batchNo, applied] = await sdb.migrate.latest();
322
+ !isTest() && console.log(chalk.green("Shadow DB 테스트에 성공했습니다!"), {
328
323
  batchNo,
329
324
  applied
330
- }
331
- ];
332
- } catch (e) {
333
- console.error(e);
334
- throw new ServiceUnavailableException(SD("sonamu.error.shadowDbTestFailed"));
325
+ });
326
+ return [
327
+ {
328
+ connKey: "shadow",
329
+ batchNo,
330
+ applied
331
+ }
332
+ ];
333
+ } catch (e) {
334
+ console.error(e);
335
+ throw new ServiceUnavailableException(SD("sonamu.error.shadowDbTestFailed"));
336
+ } finally{
337
+ await sdb.destroy();
338
+ }
335
339
  } finally{
336
- // Shadow DB 연결 종료
337
- await sdb.destroy();
338
340
  // Shadow DB 삭제
339
341
  !isTest() && console.log(chalk.magenta(`${shadowDatabase} 삭제`));
340
- await tdb.raw(`DROP DATABASE IF EXISTS ${shadowDatabase}`);
341
- // Test DB 연결 종료
342
- await tdb.destroy();
342
+ try {
343
+ await tdb.raw(`DROP DATABASE IF EXISTS ${shadowDatabase}`);
344
+ } catch (e) {
345
+ console.error("Shadow DB 정리 실패:", e); // 이게 없으면 조용히 누수
346
+ } finally{
347
+ await tdb.destroy();
348
+ }
343
349
  }
344
350
  }
345
351
  }
346
352
 
347
- //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/migration/migrator.ts"],"sourcesContent":["import assert from \"assert\";\nimport chalk from \"chalk\";\nimport { mkdir, readdir, unlink, writeFile } from \"fs/promises\";\nimport type { Knex } from \"knex\";\nimport path from \"path\";\nimport { group, sum, unique } from \"radashi\";\nimport { Sonamu } from \"../api\";\nimport { DB, type SonamuDBConfig } from \"../database/db\";\nimport { createKnexInstance } from \"../database/knex\";\nimport { SD } from \"../dict/sd\";\nimport { EntityManager } from \"../entity/entity-manager\";\nimport { ServiceUnavailableException } from \"../exceptions/so-exceptions\";\nimport { Naite } from \"../naite/naite\";\nimport type { GenMigrationCode, MigrationSet } from \"../types/types\";\nimport { isTest } from \"../utils/controller\";\nimport { exists } from \"../utils/fs-utils\";\nimport { generateAlterCode, generateCreateCode } from \"./code-generation\";\nimport { getMigrationSetFromEntity } from \"./migration-set\";\nimport { PostgreSQLSchemaReader } from \"./postgresql-schema-reader\";\nimport type { ConnString, MigrationCode, MigrationStatus } from \"./types\";\n\nexport type MigrationResult = {\n  connKey: string;\n  batchNo: number;\n  applied: string[];\n}[];\n\nexport class Migrator {\n  private async getMigrationCodes(): Promise<MigrationCode[]> {\n    const srcMigrationsDir = path.join(Sonamu.apiRootPath, \"src\", \"migrations\"); // 이건 환경에 관계없이 항상 src에서 찾아야 해요.\n\n    if (!(await exists(srcMigrationsDir))) {\n      await mkdir(srcMigrationsDir, {\n        recursive: true,\n      });\n    }\n\n    const codes = (await readdir(srcMigrationsDir))\n      .filter((f) => f.endsWith(\".ts\"))\n      .map((f) => ({\n        name: f.replace(\".ts\", \"\"),\n        path: path.join(srcMigrationsDir, f),\n      }))\n      .sort((a, b) => (a.name < b.name ? 1 : -1)); // 이름 내림차순 정렬(최신순)\n\n    Naite.t(\"migrator:getMigrationCodes:results\", codes);\n    return codes;\n  }\n\n  /**\n   * 타겟별 마이그레이션 상태와 코드 생성/준비 상태를 구해옵니다.\n   * 실제로 DB에 접근도 하고 마이그레이션 코드 파일도 확인하고,\n   * 필요하다면 적용할 수 있는 코드를 생성까지 해옵니다.\n   *\n   * CLI와 Sonamu UI에서 사용됩니다.\n   *\n   * @returns\n   */\n  async getStatus(): Promise<MigrationStatus> {\n    const codes = await this.getMigrationCodes();\n    Naite.t(\"migrator:getStatus:codes\", codes);\n\n    const connKeys = Object.keys(Sonamu.dbConfig).filter(\n      (key) => key.endsWith(\"_slave\") === false,\n    ) as (keyof typeof Sonamu.dbConfig)[];\n\n    let migrationStatusError: string | undefined;\n\n    const statuses = await Promise.all(\n      connKeys.map(async (connKey) => {\n        const knexOptions = Sonamu.dbConfig[connKey];\n        const tConn = createKnexInstance(knexOptions);\n\n        try {\n          const status = await (async () => {\n            try {\n              return await tConn.migrate.status();\n            } catch (err) {\n              console.warn(\n                chalk.yellow(\n                  `${connKey}의 마이그레이션 상태를 가져오는 데에 실패하였습니다. 데이터베이스가 올바르게 구성되지 않은 것 같습니다. 확인하시고 다시 시도해주세요.\\n시도한 연결 설정:\\n${JSON.stringify(knexOptions.connection, null, 2)}\\n발생한 에러:\\n${err}\\n`,\n                ),\n              );\n              migrationStatusError = err instanceof Error ? err.message : String(err);\n              return \"error\";\n            }\n          })();\n          const pending: string[] = await (async () => {\n            try {\n              const [, fdList] = await tConn.migrate.list();\n              return fdList.map((fd: { file: string }) => fd.file.replace(\".ts\", \"\"));\n            } catch (err) {\n              migrationStatusError = err instanceof Error ? err.message : String(err);\n              return [];\n            }\n          })();\n          const currentVersion = await (async () => {\n            try {\n              return await tConn.migrate.currentVersion();\n            } catch (_err) {\n              migrationStatusError = _err instanceof Error ? _err.message : String(_err);\n              return \"error\";\n            }\n          })();\n          Naite.t(\"migrator:getStatus:status\", status);\n\n          const connection = knexOptions.connection as Knex.PgConnectionConfig;\n\n          return {\n            name: connKey.replace(\"_master\", \"\"),\n            connKey,\n            connString: `pg://${connection.user ?? \"\"}@${connection.host}:${\n              connection.port\n            }/${connection.database}` as ConnString,\n            currentVersion,\n            status: status as number | \"error\",\n            pending,\n          };\n        } finally {\n          await tConn.destroy();\n        }\n      }),\n    );\n\n    Naite.t(\"migrator:getStatus:conns\", statuses);\n\n    const preparedCodes: GenMigrationCode[] = await (async () => {\n      const status0conn = statuses.find((status) => status.status === 0);\n      if (status0conn === undefined) {\n        console.warn(\n          chalk.yellow(\n            `While trying to prepare migration codes, we found that there is no database to compare migrations. We need at least one database where every migration is applied(status === 0). You might want to apply your existing migrations to one of the databases.`,\n          ),\n        );\n        return [];\n      }\n\n      const compareDBconn = createKnexInstance(Sonamu.dbConfig[status0conn.connKey]);\n      try {\n        return await this.compareMigrations(compareDBconn);\n      } finally {\n        await compareDBconn.destroy();\n      }\n    })();\n\n    Naite.t(\"migrator:getStatus:preparedCodes\", preparedCodes);\n\n    return {\n      conns: statuses,\n      codes,\n      preparedCodes,\n      error: migrationStatusError,\n    };\n  }\n\n  /**\n   * 마이그레이션을 적용하거나 롤백합니다.\n   * Sonamu UI에서 마이그레이션 작업을 수행할 때 사용됩니다.\n   *\n   * CLI와 Sonamu UI에서 사용됩니다.\n   *\n   * @param action 작업 유형 (apply/rollback)\n   * @param targets 작업 대상 DB 설정 키 (keyof SonamuDBConfig)\n   * @returns 작업 결과\n   */\n  async runAction(\n    action: \"apply\" | \"rollback\",\n    targets: (keyof SonamuDBConfig)[],\n  ): Promise<MigrationResult> {\n    Naite.t(\"migrator:runAction:action\", action);\n    Naite.t(\"migrator:runAction:targets\", targets);\n\n    // get uniq knex configs\n    const configs = unique(\n      targets\n        .map((target) => ({\n          connKey: target,\n          options: Sonamu.dbConfig[target as keyof typeof Sonamu.dbConfig],\n        }))\n        .filter((c) => c.options !== undefined),\n      ({ options }) =>\n        `${(options.connection as Knex.PgConnectionConfig).host}:${\n          (options.connection as Knex.PgConnectionConfig).port ?? 5432\n        }/${(options.connection as Knex.PgConnectionConfig).database}`,\n    );\n\n    // get connections\n    const conns = await Promise.all(\n      configs.map(async (config) => ({\n        connKey: config.connKey,\n        knex: createKnexInstance(config.options),\n      })),\n    );\n\n    try {\n      // action\n      const result = await (async () => {\n        switch (action) {\n          case \"apply\":\n            return Promise.all(\n              conns.map(async ({ connKey, knex }) => {\n                const [batchNo, applied] = await knex.migrate.latest();\n                return {\n                  connKey,\n                  batchNo,\n                  applied, // 이번 latest 호출로 인해 \"up\"이 적용된 마이그레이션 이름(e.g. \"20251124233557_create__companies.ts\")들의 배열입니다. 참고: https://github.com/knex/knex/blob/01b177c485d696f1b72858dee728ba143c4fad76/lib/migrations/migrate/Migrator.js#L560\n                };\n              }),\n            );\n          case \"rollback\":\n            return Promise.all(\n              conns.map(async ({ connKey, knex }) => {\n                const [batchNo, applied] = await knex.migrate.rollback();\n                return {\n                  connKey,\n                  batchNo,\n                  applied, // 이번 rollback 호출로 인해 \"down\"이 적용된(=롤백된) 마이그레이션 이름(e.g. \"20251124233557_create__companies.ts\")들의 배열입니다. 참고: https://github.com/knex/knex/blob/01b177c485d696f1b72858dee728ba143c4fad76/lib/migrations/migrate/Migrator.js#L611\n                };\n              }),\n            );\n        }\n      })();\n\n      Naite.t(\"migrator:runAction:result\", result);\n\n      return result;\n    } finally {\n      await Promise.all(\n        conns.map(({ knex }) => {\n          return knex.destroy();\n        }),\n      );\n    }\n  }\n\n  /**\n   * 삭제 가능한 마이그레이션 코드 파일을 검증합니다.\n   *\n   * @param conns 마이그레이션 상태 배열\n   * @param codeNames 삭제할 마이그레이션 코드 파일 이름 배열\n   * @returns 삭제 가능 여부 및 적용된 마이그레이션 코드 파일 이름\n   */\n  validateDeletable(conns: MigrationStatus[\"conns\"], codeNames: string[]) {\n    const appliedCodes = codeNames.filter((codeName) =>\n      conns.some((conn) => conn.pending.includes(codeName) === false),\n    );\n\n    return {\n      canDelete: appliedCodes.length === 0,\n      appliedCodes,\n    };\n  }\n\n  /**\n   * 마이그레이션 코드 파일을 삭제합니다.\n   *\n   * Sonamu UI에서 사용됩니다.\n   *\n   * @param codeNames 삭제할 마이그레이션 코드 파일 이름 배열\n   * @returns 삭제된 마이그레이션 코드 파일 개수\n   */\n  async delCodes(codeNames: string[]): Promise<number> {\n    const { conns } = await this.getStatus();\n    const { canDelete, appliedCodes } = this.validateDeletable(conns, codeNames);\n    if (!canDelete) {\n      throw new Error(\n        `You cannot delete a migration file if there is already applied. Applied codes: ${appliedCodes.join(\", \")}`,\n      );\n    }\n\n    return sum(\n      await Promise.all(\n        codeNames.map(async (codeName) => {\n          const filePath = `${Sonamu.apiRootPath}/src/migrations/${codeName}.ts`;\n          if (await exists(filePath)) {\n            await unlink(filePath);\n            return 1;\n          }\n          return 0;\n        }),\n      ),\n    );\n  }\n\n  private genDateTag(index: number, baseDate: Date = new Date()): string {\n    const date = new Date(baseDate.getTime() + index * 1000);\n    const pad = (num: number, size: number = 2) => num.toString().padStart(size, \"0\");\n    return (\n      date.getFullYear().toString() +\n      pad(date.getMonth() + 1) +\n      pad(date.getDate()) +\n      pad(date.getHours()) +\n      pad(date.getMinutes()) +\n      pad(date.getSeconds())\n    );\n  }\n\n  /**\n   * 마이그레이션 코드 파일을 생성합니다.\n   *\n   * Sonamu UI에서 사용됩니다.\n   *\n   * @returns 생성된 마이그레이션 코드 파일 개수\n   */\n  async generatePreparedCodes(): Promise<number> {\n    const { preparedCodes } = await this.getStatus();\n    Naite.t(\"migrator:generatePreparedCodes:preparedCodes\", preparedCodes);\n    if (preparedCodes.length === 0) {\n      console.log(chalk.green(\"\\n현재 모두 싱크된 상태입니다.\"));\n      return 0;\n    }\n\n    // 실제 코드 생성\n    const migrationsDir = `${Sonamu.apiRootPath}/src/migrations`;\n\n    for (const [index, pcode] of preparedCodes.entries()) {\n      if (pcode.formatted) {\n        const dateTag = this.genDateTag(index);\n        const filePath = `${migrationsDir}/${dateTag}_${pcode.title}.ts`;\n        await writeFile(filePath, pcode.formatted);\n        !isTest() && console.log(chalk.green(`MIGRATION CREATED ${filePath}`));\n      }\n    }\n\n    return preparedCodes.length;\n  }\n\n  async compareMigrations(compareDB: Knex): Promise<GenMigrationCode[]> {\n    // Entity 순회하여 싱크\n    const entityIds = EntityManager.getAllIds();\n\n    // 조인테이블 포함하여 Entity에서 MigrationSet 추출\n    const entitySetsWithJoinTable = entityIds\n      .filter((entityId) => EntityManager.get(entityId).props.length > 0)\n      .map((entityId) => getMigrationSetFromEntity(EntityManager.get(entityId)));\n\n    // 조인테이블만 추출\n    const joinTablesWithDup = entitySetsWithJoinTable.flatMap((entitySet) => entitySet.joinTables);\n    // 중복 제거 (중복인 경우 indexes를 병합)\n    const joinTables = Object.values(group(joinTablesWithDup, (jt) => jt.table)).map((tables) => {\n      assert(tables !== undefined, \"tables is undefined\");\n      if (tables.length === 1) {\n        return tables[0];\n      }\n      return {\n        ...tables[0],\n        indexes: unique(\n          tables.flatMap((t) => t.indexes),\n          (index) => [index.type, ...index.columns].join(\"-\"),\n        ),\n      };\n    });\n\n    // 조인테이블 포함하여 MigrationSet 배열\n    const entitySets: MigrationSet[] = [...entitySetsWithJoinTable, ...joinTables];\n\n    const codes: GenMigrationCode[] = (\n      await Promise.all(\n        entitySets.map(async (entitySet) => {\n          const dbSet = await PostgreSQLSchemaReader.getMigrationSetFromDB(\n            compareDB,\n            entitySet.table,\n          );\n          Naite.t(`migrator:compareMigrations:entitySet:${entitySet.table}`, entitySet);\n          Naite.t(`migrator:compareMigrations:dbSet:${entitySet.table}`, dbSet);\n\n          if (dbSet === null) {\n            // 기존 테이블 없음, 새로 테이블 생성\n            return await generateCreateCode(entitySet);\n          } else {\n            // 기존 테이블 존재하는 케이스\n            return await generateAlterCode(entitySet, dbSet, compareDB);\n          }\n        }),\n      )\n    ).flat();\n\n    // normal 타입이 앞으로, foreign이 뒤로\n    codes.sort((codeA, codeB) => {\n      if (codeA.type === \"foreign\" && codeB.type === \"normal\") {\n        return 1;\n      } else if (codeA.type === \"normal\" && codeB.type === \"foreign\") {\n        return -1;\n      } else {\n        return 0;\n      }\n    });\n\n    return codes;\n  }\n\n  /**\n   * Shadow DB 테스트를 진행합니다.\n   *\n   * Sonamu UI에서 사용됩니다.\n   *\n   * @returns Shadow DB 테스트 결과\n   */\n  async runShadowTest(): Promise<MigrationResult> {\n    const tdbConn = Sonamu.dbConfig.test.connection as Knex.PgConnectionConfig;\n    const shadowDatabase = `${tdbConn.database}__migration_shadow`;\n\n    // 테스트 상황에서는 트랜잭션을 초기화하고, 새 데이터베이스 커넥션을 가져와야 함\n    if (isTest()) {\n      await DB.clearTestTransaction();\n      // 병렬 테스트 모드에서는 worker DB 연결 유지\n      if (process.env.SONAMU_WORKER_DB !== \"true\") {\n        await DB.destroy();\n      }\n    }\n\n    // 기존 Shadow DB 삭제 후 Shadow DB 생성\n    const tdb = createKnexInstance(Sonamu.dbConfig.test);\n    !isTest() && console.log(chalk.magenta(`${shadowDatabase} 삭제`));\n    await tdb.raw(`DROP DATABASE IF EXISTS ${shadowDatabase}`);\n    await tdb.raw(`\n      SELECT pg_terminate_backend(pg_stat_activity.pid)\n      FROM pg_stat_activity\n      WHERE datname = '${tdbConn.database}'\n        AND pid <> pg_backend_pid();\n    `);\n    await tdb.raw(`CREATE DATABASE ${shadowDatabase} TEMPLATE ${tdbConn.database}`);\n\n    // Shadow DB에 연결\n    const sdb = createKnexInstance({\n      ...Sonamu.dbConfig.test,\n      connection: {\n        ...tdbConn,\n        database: shadowDatabase,\n        password: tdbConn.password,\n      },\n    });\n\n    // shadow DB 테스트 진행\n    try {\n      const [batchNo, applied] = await sdb.migrate.latest();\n      !isTest() &&\n        console.log(chalk.green(\"Shadow DB 테스트에 성공했습니다!\"), {\n          batchNo,\n          applied,\n        });\n\n      return [\n        {\n          connKey: \"shadow\",\n          batchNo,\n          applied,\n        },\n      ];\n    } catch (e) {\n      console.error(e);\n      throw new ServiceUnavailableException(SD(\"sonamu.error.shadowDbTestFailed\"));\n    } finally {\n      // Shadow DB 연결 종료\n      await sdb.destroy();\n\n      // Shadow DB 삭제\n      !isTest() && console.log(chalk.magenta(`${shadowDatabase} 삭제`));\n      await tdb.raw(`DROP DATABASE IF EXISTS ${shadowDatabase}`);\n\n      // Test DB 연결 종료\n      await tdb.destroy();\n    }\n  }\n}\n"],"names":["assert","chalk","mkdir","readdir","unlink","writeFile","path","group","sum","unique","Sonamu","DB","createKnexInstance","SD","EntityManager","ServiceUnavailableException","Naite","isTest","exists","generateAlterCode","generateCreateCode","getMigrationSetFromEntity","PostgreSQLSchemaReader","Migrator","getMigrationCodes","srcMigrationsDir","join","apiRootPath","recursive","codes","filter","f","endsWith","map","name","replace","sort","a","b","t","getStatus","connKeys","Object","keys","dbConfig","key","migrationStatusError","statuses","Promise","all","connKey","knexOptions","tConn","status","migrate","err","console","warn","yellow","JSON","stringify","connection","Error","message","String","pending","fdList","list","fd","file","currentVersion","_err","connString","user","host","port","database","destroy","preparedCodes","status0conn","find","undefined","compareDBconn","compareMigrations","conns","error","runAction","action","targets","configs","target","options","c","config","knex","result","batchNo","applied","latest","rollback","validateDeletable","codeNames","appliedCodes","codeName","some","conn","includes","canDelete","length","delCodes","filePath","genDateTag","index","baseDate","Date","date","getTime","pad","num","size","toString","padStart","getFullYear","getMonth","getDate","getHours","getMinutes","getSeconds","generatePreparedCodes","log","green","migrationsDir","pcode","entries","formatted","dateTag","title","compareDB","entityIds","getAllIds","entitySetsWithJoinTable","entityId","get","props","joinTablesWithDup","flatMap","entitySet","joinTables","values","jt","table","tables","indexes","type","columns","entitySets","dbSet","getMigrationSetFromDB","flat","codeA","codeB","runShadowTest","tdbConn","test","shadowDatabase","clearTestTransaction","process","env","SONAMU_WORKER_DB","tdb","magenta","raw","sdb","password","e"],"mappings":"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"}
353
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/migration/migrator.ts"],"sourcesContent":["import assert from \"assert\";\nimport chalk from \"chalk\";\nimport { mkdir, readdir, unlink, writeFile } from \"fs/promises\";\nimport type { Knex } from \"knex\";\nimport path from \"path\";\nimport { group, sum, unique } from \"radashi\";\nimport { Sonamu } from \"../api\";\nimport { DB, type SonamuDBConfig } from \"../database/db\";\nimport { createKnexInstance } from \"../database/knex\";\nimport { SD } from \"../dict/sd\";\nimport { EntityManager } from \"../entity/entity-manager\";\nimport { ServiceUnavailableException } from \"../exceptions/so-exceptions\";\nimport { Naite } from \"../naite/naite\";\nimport type { GenMigrationCode, MigrationSet } from \"../types/types\";\nimport { isTest } from \"../utils/controller\";\nimport { exists } from \"../utils/fs-utils\";\nimport { generateAlterCode, generateCreateCode } from \"./code-generation\";\nimport { getMigrationSetFromEntity } from \"./migration-set\";\nimport { PostgreSQLSchemaReader } from \"./postgresql-schema-reader\";\nimport type { ConnString, MigrationCode, MigrationStatus } from \"./types\";\n\nexport type MigrationResult = {\n  connKey: string;\n  batchNo: number;\n  applied: string[];\n}[];\n\nexport class Migrator {\n  private async getMigrationCodes(): Promise<MigrationCode[]> {\n    const srcMigrationsDir = path.join(Sonamu.apiRootPath, \"src\", \"migrations\"); // 이건 환경에 관계없이 항상 src에서 찾아야 해요.\n\n    if (!(await exists(srcMigrationsDir))) {\n      await mkdir(srcMigrationsDir, {\n        recursive: true,\n      });\n    }\n\n    const codes = (await readdir(srcMigrationsDir))\n      .filter((f) => f.endsWith(\".ts\"))\n      .map((f) => ({\n        name: f.replace(\".ts\", \"\"),\n        path: path.join(srcMigrationsDir, f),\n      }))\n      .sort((a, b) => (a.name < b.name ? 1 : -1)); // 이름 내림차순 정렬(최신순)\n\n    Naite.t(\"migrator:getMigrationCodes:results\", codes);\n    return codes;\n  }\n\n  /**\n   * 타겟별 마이그레이션 상태와 코드 생성/준비 상태를 구해옵니다.\n   * 실제로 DB에 접근도 하고 마이그레이션 코드 파일도 확인하고,\n   * 필요하다면 적용할 수 있는 코드를 생성까지 해옵니다.\n   *\n   * CLI와 Sonamu UI에서 사용됩니다.\n   *\n   * @returns\n   */\n  async getStatus(): Promise<MigrationStatus> {\n    const codes = await this.getMigrationCodes();\n    Naite.t(\"migrator:getStatus:codes\", codes);\n\n    const connKeys = Object.keys(Sonamu.dbConfig).filter(\n      (key) => key.endsWith(\"_slave\") === false,\n    ) as (keyof typeof Sonamu.dbConfig)[];\n\n    let migrationStatusError: string | undefined;\n\n    const statuses = await Promise.all(\n      connKeys.map(async (connKey) => {\n        const knexOptions = Sonamu.dbConfig[connKey];\n        const tConn = createKnexInstance(knexOptions);\n\n        try {\n          const status = await (async () => {\n            try {\n              return await tConn.migrate.status();\n            } catch (err) {\n              console.warn(\n                chalk.yellow(\n                  `${connKey}의 마이그레이션 상태를 가져오는 데에 실패하였습니다. 데이터베이스가 올바르게 구성되지 않은 것 같습니다. 확인하시고 다시 시도해주세요.\\n시도한 연결 설정:\\n${JSON.stringify(knexOptions.connection, null, 2)}\\n발생한 에러:\\n${err}\\n`,\n                ),\n              );\n              migrationStatusError = err instanceof Error ? err.message : String(err);\n              return \"error\";\n            }\n          })();\n          const pending: string[] = await (async () => {\n            try {\n              const [, fdList] = await tConn.migrate.list();\n              return fdList.map((fd: { file: string }) => fd.file.replace(\".ts\", \"\"));\n            } catch (err) {\n              migrationStatusError = err instanceof Error ? err.message : String(err);\n              return [];\n            }\n          })();\n          const currentVersion = await (async () => {\n            try {\n              return await tConn.migrate.currentVersion();\n            } catch (_err) {\n              migrationStatusError = _err instanceof Error ? _err.message : String(_err);\n              return \"error\";\n            }\n          })();\n          Naite.t(\"migrator:getStatus:status\", status);\n\n          const connection = knexOptions.connection as Knex.PgConnectionConfig;\n\n          return {\n            name: connKey.replace(\"_master\", \"\"),\n            connKey,\n            connString: `pg://${connection.user ?? \"\"}@${connection.host}:${\n              connection.port\n            }/${connection.database}` as ConnString,\n            currentVersion,\n            status: status as number | \"error\",\n            pending,\n          };\n        } finally {\n          await tConn.destroy();\n        }\n      }),\n    );\n\n    Naite.t(\"migrator:getStatus:conns\", statuses);\n\n    const preparedCodes: GenMigrationCode[] = await (async () => {\n      const status0conn = statuses.find((status) => status.status === 0);\n      if (status0conn === undefined) {\n        console.warn(\n          chalk.yellow(\n            `While trying to prepare migration codes, we found that there is no database to compare migrations. We need at least one database where every migration is applied(status === 0). You might want to apply your existing migrations to one of the databases.`,\n          ),\n        );\n        return [];\n      }\n\n      const compareDBconn = createKnexInstance(Sonamu.dbConfig[status0conn.connKey]);\n      try {\n        return await this.compareMigrations(compareDBconn);\n      } finally {\n        await compareDBconn.destroy();\n      }\n    })();\n\n    Naite.t(\"migrator:getStatus:preparedCodes\", preparedCodes);\n\n    return {\n      conns: statuses,\n      codes,\n      preparedCodes,\n      error: migrationStatusError,\n    };\n  }\n\n  /**\n   * 마이그레이션을 적용하거나 롤백합니다.\n   * Sonamu UI에서 마이그레이션 작업을 수행할 때 사용됩니다.\n   *\n   * CLI와 Sonamu UI에서 사용됩니다.\n   *\n   * @param action 작업 유형 (apply/rollback)\n   * @param targets 작업 대상 DB 설정 키 (keyof SonamuDBConfig)\n   * @returns 작업 결과\n   */\n  async runAction(\n    action: \"apply\" | \"rollback\",\n    targets: (keyof SonamuDBConfig)[],\n  ): Promise<MigrationResult> {\n    Naite.t(\"migrator:runAction:action\", action);\n    Naite.t(\"migrator:runAction:targets\", targets);\n\n    // get uniq knex configs\n    const configs = unique(\n      targets\n        .map((target) => ({\n          connKey: target,\n          options: Sonamu.dbConfig[target as keyof typeof Sonamu.dbConfig],\n        }))\n        .filter((c) => c.options !== undefined),\n      ({ options }) =>\n        `${(options.connection as Knex.PgConnectionConfig).host}:${\n          (options.connection as Knex.PgConnectionConfig).port ?? 5432\n        }/${(options.connection as Knex.PgConnectionConfig).database}`,\n    );\n\n    // get connections\n    const conns = await Promise.all(\n      configs.map(async (config) => ({\n        connKey: config.connKey,\n        knex: createKnexInstance(config.options),\n      })),\n    );\n\n    try {\n      // action\n      const result = await (async () => {\n        switch (action) {\n          case \"apply\":\n            return Promise.all(\n              conns.map(async ({ connKey, knex }) => {\n                const [batchNo, applied] = await knex.migrate.latest();\n                return {\n                  connKey,\n                  batchNo,\n                  applied, // 이번 latest 호출로 인해 \"up\"이 적용된 마이그레이션 이름(e.g. \"20251124233557_create__companies.ts\")들의 배열입니다. 참고: https://github.com/knex/knex/blob/01b177c485d696f1b72858dee728ba143c4fad76/lib/migrations/migrate/Migrator.js#L560\n                };\n              }),\n            );\n          case \"rollback\":\n            return Promise.all(\n              conns.map(async ({ connKey, knex }) => {\n                const [batchNo, applied] = await knex.migrate.rollback();\n                return {\n                  connKey,\n                  batchNo,\n                  applied, // 이번 rollback 호출로 인해 \"down\"이 적용된(=롤백된) 마이그레이션 이름(e.g. \"20251124233557_create__companies.ts\")들의 배열입니다. 참고: https://github.com/knex/knex/blob/01b177c485d696f1b72858dee728ba143c4fad76/lib/migrations/migrate/Migrator.js#L611\n                };\n              }),\n            );\n        }\n      })();\n\n      Naite.t(\"migrator:runAction:result\", result);\n\n      return result;\n    } finally {\n      await Promise.all(\n        conns.map(({ knex }) => {\n          return knex.destroy();\n        }),\n      );\n    }\n  }\n\n  /**\n   * 삭제 가능한 마이그레이션 코드 파일을 검증합니다.\n   *\n   * @param conns 마이그레이션 상태 배열\n   * @param codeNames 삭제할 마이그레이션 코드 파일 이름 배열\n   * @returns 삭제 가능 여부 및 적용된 마이그레이션 코드 파일 이름\n   */\n  validateDeletable(conns: MigrationStatus[\"conns\"], codeNames: string[]) {\n    const appliedCodes = codeNames.filter((codeName) =>\n      conns.some((conn) => conn.pending.includes(codeName) === false),\n    );\n\n    return {\n      canDelete: appliedCodes.length === 0,\n      appliedCodes,\n    };\n  }\n\n  /**\n   * 마이그레이션 코드 파일을 삭제합니다.\n   *\n   * Sonamu UI에서 사용됩니다.\n   *\n   * @param codeNames 삭제할 마이그레이션 코드 파일 이름 배열\n   * @returns 삭제된 마이그레이션 코드 파일 개수\n   */\n  async delCodes(codeNames: string[]): Promise<number> {\n    const { conns } = await this.getStatus();\n    const { canDelete, appliedCodes } = this.validateDeletable(conns, codeNames);\n    if (!canDelete) {\n      throw new Error(\n        `You cannot delete a migration file if there is already applied. Applied codes: ${appliedCodes.join(\", \")}`,\n      );\n    }\n\n    return sum(\n      await Promise.all(\n        codeNames.map(async (codeName) => {\n          const filePath = `${Sonamu.apiRootPath}/src/migrations/${codeName}.ts`;\n          if (await exists(filePath)) {\n            await unlink(filePath);\n            return 1;\n          }\n          return 0;\n        }),\n      ),\n    );\n  }\n\n  private genDateTag(index: number, baseDate: Date = new Date()): string {\n    const date = new Date(baseDate.getTime() + index * 1000);\n    const pad = (num: number, size: number = 2) => num.toString().padStart(size, \"0\");\n    return (\n      date.getFullYear().toString() +\n      pad(date.getMonth() + 1) +\n      pad(date.getDate()) +\n      pad(date.getHours()) +\n      pad(date.getMinutes()) +\n      pad(date.getSeconds())\n    );\n  }\n\n  /**\n   * 마이그레이션 코드 파일을 생성합니다.\n   *\n   * Sonamu UI에서 사용됩니다.\n   *\n   * @returns 생성된 마이그레이션 코드 파일 개수\n   */\n  async generatePreparedCodes(): Promise<number> {\n    const { preparedCodes } = await this.getStatus();\n    Naite.t(\"migrator:generatePreparedCodes:preparedCodes\", preparedCodes);\n    if (preparedCodes.length === 0) {\n      console.log(chalk.green(\"\\n현재 모두 싱크된 상태입니다.\"));\n      return 0;\n    }\n\n    // 실제 코드 생성\n    const migrationsDir = `${Sonamu.apiRootPath}/src/migrations`;\n\n    for (const [index, pcode] of preparedCodes.entries()) {\n      if (pcode.formatted) {\n        const dateTag = this.genDateTag(index);\n        const filePath = `${migrationsDir}/${dateTag}_${pcode.title}.ts`;\n        await writeFile(filePath, pcode.formatted);\n        !isTest() && console.log(chalk.green(`MIGRATION CREATED ${filePath}`));\n      }\n    }\n\n    return preparedCodes.length;\n  }\n\n  async compareMigrations(compareDB: Knex): Promise<GenMigrationCode[]> {\n    // Entity 순회하여 싱크\n    const entityIds = EntityManager.getAllIds();\n\n    // 조인테이블 포함하여 Entity에서 MigrationSet 추출\n    const entitySetsWithJoinTable = entityIds\n      .filter((entityId) => EntityManager.get(entityId).props.length > 0)\n      .map((entityId) => getMigrationSetFromEntity(EntityManager.get(entityId)));\n\n    // 조인테이블만 추출\n    const joinTablesWithDup = entitySetsWithJoinTable.flatMap((entitySet) => entitySet.joinTables);\n    // 중복 제거 (중복인 경우 indexes를 병합)\n    const joinTables = Object.values(group(joinTablesWithDup, (jt) => jt.table)).map((tables) => {\n      assert(tables !== undefined, \"tables is undefined\");\n      if (tables.length === 1) {\n        return tables[0];\n      }\n      return {\n        ...tables[0],\n        indexes: unique(\n          tables.flatMap((t) => t.indexes),\n          (index) => [index.type, ...index.columns].join(\"-\"),\n        ),\n      };\n    });\n\n    // 조인테이블 포함하여 MigrationSet 배열\n    const entitySets: MigrationSet[] = [...entitySetsWithJoinTable, ...joinTables];\n\n    const codes: GenMigrationCode[] = (\n      await Promise.all(\n        entitySets.map(async (entitySet) => {\n          const dbSet = await PostgreSQLSchemaReader.getMigrationSetFromDB(\n            compareDB,\n            entitySet.table,\n          );\n          Naite.t(`migrator:compareMigrations:entitySet:${entitySet.table}`, entitySet);\n          Naite.t(`migrator:compareMigrations:dbSet:${entitySet.table}`, dbSet);\n\n          if (dbSet === null) {\n            // 기존 테이블 없음, 새로 테이블 생성\n            return await generateCreateCode(entitySet);\n          } else {\n            // 기존 테이블 존재하는 케이스\n            return await generateAlterCode(entitySet, dbSet, compareDB);\n          }\n        }),\n      )\n    ).flat();\n\n    // normal 타입이 앞으로, foreign이 뒤로\n    codes.sort((codeA, codeB) => {\n      if (codeA.type === \"foreign\" && codeB.type === \"normal\") {\n        return 1;\n      } else if (codeA.type === \"normal\" && codeB.type === \"foreign\") {\n        return -1;\n      } else {\n        return 0;\n      }\n    });\n\n    return codes;\n  }\n\n  /**\n   * Shadow DB 테스트를 진행합니다.\n   *\n   * Sonamu UI에서 사용됩니다.\n   *\n   * @returns Shadow DB 테스트 결과\n   */\n  async runShadowTest(): Promise<MigrationResult> {\n    const tdbConn = Sonamu.dbConfig.test.connection as Knex.PgConnectionConfig;\n    const shadowDatabase = `${tdbConn.database}__migration_shadow`;\n\n    // 테스트 상황에서는 트랜잭션을 초기화하고, 새 데이터베이스 커넥션을 가져와야 함\n    if (isTest()) {\n      await DB.clearTestTransaction();\n      // 병렬 테스트 모드에서는 worker DB 연결 유지\n      if (process.env.SONAMU_WORKER_DB !== \"true\") {\n        await DB.destroy();\n      }\n    }\n\n    // 기존 Shadow DB 삭제 후 Shadow DB 생성\n    const tdb = createKnexInstance(Sonamu.dbConfig.test);\n    try {\n      !isTest() && console.log(chalk.magenta(`${shadowDatabase} 삭제`));\n      await tdb.raw(`DROP DATABASE IF EXISTS ${shadowDatabase}`);\n      await tdb.raw(`\n        SELECT pg_terminate_backend(pg_stat_activity.pid)\n        FROM pg_stat_activity\n        WHERE datname = '${tdbConn.database}'\n          AND pid <> pg_backend_pid();\n      `);\n      await tdb.raw(`CREATE DATABASE ${shadowDatabase} TEMPLATE ${tdbConn.database}`);\n\n      // Shadow DB에 연결\n      const sdb = createKnexInstance({\n        ...Sonamu.dbConfig.test,\n        connection: {\n          ...tdbConn,\n          database: shadowDatabase,\n          password: tdbConn.password,\n        },\n      });\n\n      // shadow DB 테스트 진행\n      try {\n        const [batchNo, applied] = await sdb.migrate.latest();\n        !isTest() &&\n          console.log(chalk.green(\"Shadow DB 테스트에 성공했습니다!\"), {\n            batchNo,\n            applied,\n          });\n\n        return [\n          {\n            connKey: \"shadow\",\n            batchNo,\n            applied,\n          },\n        ];\n      } catch (e) {\n        console.error(e);\n        throw new ServiceUnavailableException(SD(\"sonamu.error.shadowDbTestFailed\"));\n      } finally {\n        await sdb.destroy();\n      }\n    } finally {\n      // Shadow DB 삭제\n      !isTest() && console.log(chalk.magenta(`${shadowDatabase} 삭제`));\n      try {\n        await tdb.raw(`DROP DATABASE IF EXISTS ${shadowDatabase}`);\n      } catch (e) {\n        console.error(\"Shadow DB 정리 실패:\", e); // 이게 없으면 조용히 누수\n      } finally {\n        await tdb.destroy();\n      }\n    }\n  }\n}\n"],"names":["assert","chalk","mkdir","readdir","unlink","writeFile","path","group","sum","unique","Sonamu","DB","createKnexInstance","SD","EntityManager","ServiceUnavailableException","Naite","isTest","exists","generateAlterCode","generateCreateCode","getMigrationSetFromEntity","PostgreSQLSchemaReader","Migrator","getMigrationCodes","srcMigrationsDir","join","apiRootPath","recursive","codes","filter","f","endsWith","map","name","replace","sort","a","b","t","getStatus","connKeys","Object","keys","dbConfig","key","migrationStatusError","statuses","Promise","all","connKey","knexOptions","tConn","status","migrate","err","console","warn","yellow","JSON","stringify","connection","Error","message","String","pending","fdList","list","fd","file","currentVersion","_err","connString","user","host","port","database","destroy","preparedCodes","status0conn","find","undefined","compareDBconn","compareMigrations","conns","error","runAction","action","targets","configs","target","options","c","config","knex","result","batchNo","applied","latest","rollback","validateDeletable","codeNames","appliedCodes","codeName","some","conn","includes","canDelete","length","delCodes","filePath","genDateTag","index","baseDate","Date","date","getTime","pad","num","size","toString","padStart","getFullYear","getMonth","getDate","getHours","getMinutes","getSeconds","generatePreparedCodes","log","green","migrationsDir","pcode","entries","formatted","dateTag","title","compareDB","entityIds","getAllIds","entitySetsWithJoinTable","entityId","get","props","joinTablesWithDup","flatMap","entitySet","joinTables","values","jt","table","tables","indexes","type","columns","entitySets","dbSet","getMigrationSetFromDB","flat","codeA","codeB","runShadowTest","tdbConn","test","shadowDatabase","clearTestTransaction","process","env","SONAMU_WORKER_DB","tdb","magenta","raw","sdb","password","e"],"mappings":"AAAA,OAAOA,YAAY,SAAS;AAC5B,OAAOC,WAAW,QAAQ;AAC1B,SAASC,KAAK,EAAEC,OAAO,EAAEC,MAAM,EAAEC,SAAS,QAAQ,mBAAc;AAEhE,OAAOC,UAAU,OAAO;AACxB,SAASC,KAAK,EAAEC,GAAG,EAAEC,MAAM,QAAQ,UAAU;AAC7C,SAASC,MAAM,QAAQ,kBAAS;AAChC,SAASC,EAAE,QAA6B,oBAAiB;AACzD,SAASC,kBAAkB,QAAQ,sBAAmB;AACtD,SAASC,EAAE,QAAQ,gBAAa;AAChC,SAASC,aAAa,QAAQ,8BAA2B;AACzD,SAASC,2BAA2B,QAAQ,iCAA8B;AAC1E,SAASC,KAAK,QAAQ,oBAAiB;AAEvC,SAASC,MAAM,QAAQ,yBAAsB;AAC7C,SAASC,MAAM,QAAQ,uBAAoB;AAC3C,SAASC,iBAAiB,EAAEC,kBAAkB,QAAQ,uBAAoB;AAC1E,SAASC,yBAAyB,QAAQ,qBAAkB;AAC5D,SAASC,sBAAsB,QAAQ,gCAA6B;AASpE,OAAO,MAAMC;IACX,MAAcC,oBAA8C;QAC1D,MAAMC,mBAAmBnB,KAAKoB,IAAI,CAAChB,OAAOiB,WAAW,EAAE,OAAO,eAAe,+BAA+B;QAE5G,IAAI,CAAE,MAAMT,OAAOO,mBAAoB;YACrC,MAAMvB,MAAMuB,kBAAkB;gBAC5BG,WAAW;YACb;QACF;QAEA,MAAMC,QAAQ,AAAC,CAAA,MAAM1B,QAAQsB,iBAAgB,EAC1CK,MAAM,CAAC,CAACC,IAAMA,EAAEC,QAAQ,CAAC,QACzBC,GAAG,CAAC,CAACF,IAAO,CAAA;gBACXG,MAAMH,EAAEI,OAAO,CAAC,OAAO;gBACvB7B,MAAMA,KAAKoB,IAAI,CAACD,kBAAkBM;YACpC,CAAA,GACCK,IAAI,CAAC,CAACC,GAAGC,IAAOD,EAAEH,IAAI,GAAGI,EAAEJ,IAAI,GAAG,IAAI,CAAC,IAAK,kBAAkB;QAEjElB,MAAMuB,CAAC,CAAC,sCAAsCV;QAC9C,OAAOA;IACT;IAEA;;;;;;;;GAQC,GACD,MAAMW,YAAsC;QAC1C,MAAMX,QAAQ,MAAM,IAAI,CAACL,iBAAiB;QAC1CR,MAAMuB,CAAC,CAAC,4BAA4BV;QAEpC,MAAMY,WAAWC,OAAOC,IAAI,CAACjC,OAAOkC,QAAQ,EAAEd,MAAM,CAClD,CAACe,MAAQA,IAAIb,QAAQ,CAAC,cAAc;QAGtC,IAAIc;QAEJ,MAAMC,WAAW,MAAMC,QAAQC,GAAG,CAChCR,SAASR,GAAG,CAAC,OAAOiB;YAClB,MAAMC,cAAczC,OAAOkC,QAAQ,CAACM,QAAQ;YAC5C,MAAME,QAAQxC,mBAAmBuC;YAEjC,IAAI;gBACF,MAAME,SAAS,MAAM,AAAC,CAAA;oBACpB,IAAI;wBACF,OAAO,MAAMD,MAAME,OAAO,CAACD,MAAM;oBACnC,EAAE,OAAOE,KAAK;wBACZC,QAAQC,IAAI,CACVxD,MAAMyD,MAAM,CACV,GAAGR,QAAQ,yFAAyF,EAAES,KAAKC,SAAS,CAACT,YAAYU,UAAU,EAAE,MAAM,GAAG,WAAW,EAAEN,IAAI,EAAE,CAAC;wBAG9KT,uBAAuBS,eAAeO,QAAQP,IAAIQ,OAAO,GAAGC,OAAOT;wBACnE,OAAO;oBACT;gBACF,CAAA;gBACA,MAAMU,UAAoB,MAAM,AAAC,CAAA;oBAC/B,IAAI;wBACF,MAAM,GAAGC,OAAO,GAAG,MAAMd,MAAME,OAAO,CAACa,IAAI;wBAC3C,OAAOD,OAAOjC,GAAG,CAAC,CAACmC,KAAyBA,GAAGC,IAAI,CAAClC,OAAO,CAAC,OAAO;oBACrE,EAAE,OAAOoB,KAAK;wBACZT,uBAAuBS,eAAeO,QAAQP,IAAIQ,OAAO,GAAGC,OAAOT;wBACnE,OAAO,EAAE;oBACX;gBACF,CAAA;gBACA,MAAMe,iBAAiB,MAAM,AAAC,CAAA;oBAC5B,IAAI;wBACF,OAAO,MAAMlB,MAAME,OAAO,CAACgB,cAAc;oBAC3C,EAAE,OAAOC,MAAM;wBACbzB,uBAAuByB,gBAAgBT,QAAQS,KAAKR,OAAO,GAAGC,OAAOO;wBACrE,OAAO;oBACT;gBACF,CAAA;gBACAvD,MAAMuB,CAAC,CAAC,6BAA6Bc;gBAErC,MAAMQ,aAAaV,YAAYU,UAAU;gBAEzC,OAAO;oBACL3B,MAAMgB,QAAQf,OAAO,CAAC,WAAW;oBACjCe;oBACAsB,YAAY,CAAC,KAAK,EAAEX,WAAWY,IAAI,IAAI,GAAG,CAAC,EAAEZ,WAAWa,IAAI,CAAC,CAAC,EAC5Db,WAAWc,IAAI,CAChB,CAAC,EAAEd,WAAWe,QAAQ,EAAE;oBACzBN;oBACAjB,QAAQA;oBACRY;gBACF;YACF,SAAU;gBACR,MAAMb,MAAMyB,OAAO;YACrB;QACF;QAGF7D,MAAMuB,CAAC,CAAC,4BAA4BQ;QAEpC,MAAM+B,gBAAoC,MAAM,AAAC,CAAA;YAC/C,MAAMC,cAAchC,SAASiC,IAAI,CAAC,CAAC3B,SAAWA,OAAOA,MAAM,KAAK;YAChE,IAAI0B,gBAAgBE,WAAW;gBAC7BzB,QAAQC,IAAI,CACVxD,MAAMyD,MAAM,CACV,CAAC,0PAA0P,CAAC;gBAGhQ,OAAO,EAAE;YACX;YAEA,MAAMwB,gBAAgBtE,mBAAmBF,OAAOkC,QAAQ,CAACmC,YAAY7B,OAAO,CAAC;YAC7E,IAAI;gBACF,OAAO,MAAM,IAAI,CAACiC,iBAAiB,CAACD;YACtC,SAAU;gBACR,MAAMA,cAAcL,OAAO;YAC7B;QACF,CAAA;QAEA7D,MAAMuB,CAAC,CAAC,oCAAoCuC;QAE5C,OAAO;YACLM,OAAOrC;YACPlB;YACAiD;YACAO,OAAOvC;QACT;IACF;IAEA;;;;;;;;;GASC,GACD,MAAMwC,UACJC,MAA4B,EAC5BC,OAAiC,EACP;QAC1BxE,MAAMuB,CAAC,CAAC,6BAA6BgD;QACrCvE,MAAMuB,CAAC,CAAC,8BAA8BiD;QAEtC,wBAAwB;QACxB,MAAMC,UAAUhF,OACd+E,QACGvD,GAAG,CAAC,CAACyD,SAAY,CAAA;gBAChBxC,SAASwC;gBACTC,SAASjF,OAAOkC,QAAQ,CAAC8C,OAAuC;YAClE,CAAA,GACC5D,MAAM,CAAC,CAAC8D,IAAMA,EAAED,OAAO,KAAKV,YAC/B,CAAC,EAAEU,OAAO,EAAE,GACV,GAAG,AAACA,QAAQ9B,UAAU,CAA6Ba,IAAI,CAAC,CAAC,EACvD,AAACiB,QAAQ9B,UAAU,CAA6Bc,IAAI,IAAI,KACzD,CAAC,EAAE,AAACgB,QAAQ9B,UAAU,CAA6Be,QAAQ,EAAE;QAGlE,kBAAkB;QAClB,MAAMQ,QAAQ,MAAMpC,QAAQC,GAAG,CAC7BwC,QAAQxD,GAAG,CAAC,OAAO4D,SAAY,CAAA;gBAC7B3C,SAAS2C,OAAO3C,OAAO;gBACvB4C,MAAMlF,mBAAmBiF,OAAOF,OAAO;YACzC,CAAA;QAGF,IAAI;YACF,SAAS;YACT,MAAMI,SAAS,MAAM,AAAC,CAAA;gBACpB,OAAQR;oBACN,KAAK;wBACH,OAAOvC,QAAQC,GAAG,CAChBmC,MAAMnD,GAAG,CAAC,OAAO,EAAEiB,OAAO,EAAE4C,IAAI,EAAE;4BAChC,MAAM,CAACE,SAASC,QAAQ,GAAG,MAAMH,KAAKxC,OAAO,CAAC4C,MAAM;4BACpD,OAAO;gCACLhD;gCACA8C;gCACAC;4BACF;wBACF;oBAEJ,KAAK;wBACH,OAAOjD,QAAQC,GAAG,CAChBmC,MAAMnD,GAAG,CAAC,OAAO,EAAEiB,OAAO,EAAE4C,IAAI,EAAE;4BAChC,MAAM,CAACE,SAASC,QAAQ,GAAG,MAAMH,KAAKxC,OAAO,CAAC6C,QAAQ;4BACtD,OAAO;gCACLjD;gCACA8C;gCACAC;4BACF;wBACF;gBAEN;YACF,CAAA;YAEAjF,MAAMuB,CAAC,CAAC,6BAA6BwD;YAErC,OAAOA;QACT,SAAU;YACR,MAAM/C,QAAQC,GAAG,CACfmC,MAAMnD,GAAG,CAAC,CAAC,EAAE6D,IAAI,EAAE;gBACjB,OAAOA,KAAKjB,OAAO;YACrB;QAEJ;IACF;IAEA;;;;;;GAMC,GACDuB,kBAAkBhB,KAA+B,EAAEiB,SAAmB,EAAE;QACtE,MAAMC,eAAeD,UAAUvE,MAAM,CAAC,CAACyE,WACrCnB,MAAMoB,IAAI,CAAC,CAACC,OAASA,KAAKxC,OAAO,CAACyC,QAAQ,CAACH,cAAc;QAG3D,OAAO;YACLI,WAAWL,aAAaM,MAAM,KAAK;YACnCN;QACF;IACF;IAEA;;;;;;;GAOC,GACD,MAAMO,SAASR,SAAmB,EAAmB;QACnD,MAAM,EAAEjB,KAAK,EAAE,GAAG,MAAM,IAAI,CAAC5C,SAAS;QACtC,MAAM,EAAEmE,SAAS,EAAEL,YAAY,EAAE,GAAG,IAAI,CAACF,iBAAiB,CAAChB,OAAOiB;QAClE,IAAI,CAACM,WAAW;YACd,MAAM,IAAI7C,MACR,CAAC,+EAA+E,EAAEwC,aAAa5E,IAAI,CAAC,OAAO;QAE/G;QAEA,OAAOlB,IACL,MAAMwC,QAAQC,GAAG,CACfoD,UAAUpE,GAAG,CAAC,OAAOsE;YACnB,MAAMO,WAAW,GAAGpG,OAAOiB,WAAW,CAAC,gBAAgB,EAAE4E,SAAS,GAAG,CAAC;YACtE,IAAI,MAAMrF,OAAO4F,WAAW;gBAC1B,MAAM1G,OAAO0G;gBACb,OAAO;YACT;YACA,OAAO;QACT;IAGN;IAEQC,WAAWC,KAAa,EAAEC,WAAiB,IAAIC,MAAM,EAAU;QACrE,MAAMC,OAAO,IAAID,KAAKD,SAASG,OAAO,KAAKJ,QAAQ;QACnD,MAAMK,MAAM,CAACC,KAAaC,OAAe,CAAC,GAAKD,IAAIE,QAAQ,GAAGC,QAAQ,CAACF,MAAM;QAC7E,OACEJ,KAAKO,WAAW,GAAGF,QAAQ,KAC3BH,IAAIF,KAAKQ,QAAQ,KAAK,KACtBN,IAAIF,KAAKS,OAAO,MAChBP,IAAIF,KAAKU,QAAQ,MACjBR,IAAIF,KAAKW,UAAU,MACnBT,IAAIF,KAAKY,UAAU;IAEvB;IAEA;;;;;;GAMC,GACD,MAAMC,wBAAyC;QAC7C,MAAM,EAAElD,aAAa,EAAE,GAAG,MAAM,IAAI,CAACtC,SAAS;QAC9CxB,MAAMuB,CAAC,CAAC,gDAAgDuC;QACxD,IAAIA,cAAc8B,MAAM,KAAK,GAAG;YAC9BpD,QAAQyE,GAAG,CAAChI,MAAMiI,KAAK,CAAC;YACxB,OAAO;QACT;QAEA,WAAW;QACX,MAAMC,gBAAgB,GAAGzH,OAAOiB,WAAW,CAAC,eAAe,CAAC;QAE5D,KAAK,MAAM,CAACqF,OAAOoB,MAAM,IAAItD,cAAcuD,OAAO,GAAI;YACpD,IAAID,MAAME,SAAS,EAAE;gBACnB,MAAMC,UAAU,IAAI,CAACxB,UAAU,CAACC;gBAChC,MAAMF,WAAW,GAAGqB,cAAc,CAAC,EAAEI,QAAQ,CAAC,EAAEH,MAAMI,KAAK,CAAC,GAAG,CAAC;gBAChE,MAAMnI,UAAUyG,UAAUsB,MAAME,SAAS;gBACzC,CAACrH,YAAYuC,QAAQyE,GAAG,CAAChI,MAAMiI,KAAK,CAAC,CAAC,kBAAkB,EAAEpB,UAAU;YACtE;QACF;QAEA,OAAOhC,cAAc8B,MAAM;IAC7B;IAEA,MAAMzB,kBAAkBsD,SAAe,EAA+B;QACpE,iBAAiB;QACjB,MAAMC,YAAY5H,cAAc6H,SAAS;QAEzC,sCAAsC;QACtC,MAAMC,0BAA0BF,UAC7B5G,MAAM,CAAC,CAAC+G,WAAa/H,cAAcgI,GAAG,CAACD,UAAUE,KAAK,CAACnC,MAAM,GAAG,GAChE3E,GAAG,CAAC,CAAC4G,WAAaxH,0BAA0BP,cAAcgI,GAAG,CAACD;QAEjE,YAAY;QACZ,MAAMG,oBAAoBJ,wBAAwBK,OAAO,CAAC,CAACC,YAAcA,UAAUC,UAAU;QAC7F,6BAA6B;QAC7B,MAAMA,aAAazG,OAAO0G,MAAM,CAAC7I,MAAMyI,mBAAmB,CAACK,KAAOA,GAAGC,KAAK,GAAGrH,GAAG,CAAC,CAACsH;YAChFvJ,OAAOuJ,WAAWtE,WAAW;YAC7B,IAAIsE,OAAO3C,MAAM,KAAK,GAAG;gBACvB,OAAO2C,MAAM,CAAC,EAAE;YAClB;YACA,OAAO;gBACL,GAAGA,MAAM,CAAC,EAAE;gBACZC,SAAS/I,OACP8I,OAAON,OAAO,CAAC,CAAC1G,IAAMA,EAAEiH,OAAO,GAC/B,CAACxC,QAAU;wBAACA,MAAMyC,IAAI;2BAAKzC,MAAM0C,OAAO;qBAAC,CAAChI,IAAI,CAAC;YAEnD;QACF;QAEA,6BAA6B;QAC7B,MAAMiI,aAA6B;eAAIf;eAA4BO;SAAW;QAE9E,MAAMtH,QAA4B,AAChC,CAAA,MAAMmB,QAAQC,GAAG,CACf0G,WAAW1H,GAAG,CAAC,OAAOiH;YACpB,MAAMU,QAAQ,MAAMtI,uBAAuBuI,qBAAqB,CAC9DpB,WACAS,UAAUI,KAAK;YAEjBtI,MAAMuB,CAAC,CAAC,CAAC,qCAAqC,EAAE2G,UAAUI,KAAK,EAAE,EAAEJ;YACnElI,MAAMuB,CAAC,CAAC,CAAC,iCAAiC,EAAE2G,UAAUI,KAAK,EAAE,EAAEM;YAE/D,IAAIA,UAAU,MAAM;gBAClB,uBAAuB;gBACvB,OAAO,MAAMxI,mBAAmB8H;YAClC,OAAO;gBACL,kBAAkB;gBAClB,OAAO,MAAM/H,kBAAkB+H,WAAWU,OAAOnB;YACnD;QACF,GACF,EACAqB,IAAI;QAEN,8BAA8B;QAC9BjI,MAAMO,IAAI,CAAC,CAAC2H,OAAOC;YACjB,IAAID,MAAMN,IAAI,KAAK,aAAaO,MAAMP,IAAI,KAAK,UAAU;gBACvD,OAAO;YACT,OAAO,IAAIM,MAAMN,IAAI,KAAK,YAAYO,MAAMP,IAAI,KAAK,WAAW;gBAC9D,OAAO,CAAC;YACV,OAAO;gBACL,OAAO;YACT;QACF;QAEA,OAAO5H;IACT;IAEA;;;;;;GAMC,GACD,MAAMoI,gBAA0C;QAC9C,MAAMC,UAAUxJ,OAAOkC,QAAQ,CAACuH,IAAI,CAACtG,UAAU;QAC/C,MAAMuG,iBAAiB,GAAGF,QAAQtF,QAAQ,CAAC,kBAAkB,CAAC;QAE9D,8CAA8C;QAC9C,IAAI3D,UAAU;YACZ,MAAMN,GAAG0J,oBAAoB;YAC7B,+BAA+B;YAC/B,IAAIC,QAAQC,GAAG,CAACC,gBAAgB,KAAK,QAAQ;gBAC3C,MAAM7J,GAAGkE,OAAO;YAClB;QACF;QAEA,iCAAiC;QACjC,MAAM4F,MAAM7J,mBAAmBF,OAAOkC,QAAQ,CAACuH,IAAI;QACnD,IAAI;YACF,CAAClJ,YAAYuC,QAAQyE,GAAG,CAAChI,MAAMyK,OAAO,CAAC,GAAGN,eAAe,GAAG,CAAC;YAC7D,MAAMK,IAAIE,GAAG,CAAC,CAAC,wBAAwB,EAAEP,gBAAgB;YACzD,MAAMK,IAAIE,GAAG,CAAC,CAAC;;;yBAGI,EAAET,QAAQtF,QAAQ,CAAC;;MAEtC,CAAC;YACD,MAAM6F,IAAIE,GAAG,CAAC,CAAC,gBAAgB,EAAEP,eAAe,UAAU,EAAEF,QAAQtF,QAAQ,EAAE;YAE9E,gBAAgB;YAChB,MAAMgG,MAAMhK,mBAAmB;gBAC7B,GAAGF,OAAOkC,QAAQ,CAACuH,IAAI;gBACvBtG,YAAY;oBACV,GAAGqG,OAAO;oBACVtF,UAAUwF;oBACVS,UAAUX,QAAQW,QAAQ;gBAC5B;YACF;YAEA,mBAAmB;YACnB,IAAI;gBACF,MAAM,CAAC7E,SAASC,QAAQ,GAAG,MAAM2E,IAAItH,OAAO,CAAC4C,MAAM;gBACnD,CAACjF,YACCuC,QAAQyE,GAAG,CAAChI,MAAMiI,KAAK,CAAC,2BAA2B;oBACjDlC;oBACAC;gBACF;gBAEF,OAAO;oBACL;wBACE/C,SAAS;wBACT8C;wBACAC;oBACF;iBACD;YACH,EAAE,OAAO6E,GAAG;gBACVtH,QAAQ6B,KAAK,CAACyF;gBACd,MAAM,IAAI/J,4BAA4BF,GAAG;YAC3C,SAAU;gBACR,MAAM+J,IAAI/F,OAAO;YACnB;QACF,SAAU;YACR,eAAe;YACf,CAAC5D,YAAYuC,QAAQyE,GAAG,CAAChI,MAAMyK,OAAO,CAAC,GAAGN,eAAe,GAAG,CAAC;YAC7D,IAAI;gBACF,MAAMK,IAAIE,GAAG,CAAC,CAAC,wBAAwB,EAAEP,gBAAgB;YAC3D,EAAE,OAAOU,GAAG;gBACVtH,QAAQ6B,KAAK,CAAC,oBAAoByF,IAAI,gBAAgB;YACxD,SAAU;gBACR,MAAML,IAAI5F,OAAO;YACnB;QACF;IACF;AACF"}
@@ -1 +1 @@
1
- {"version":3,"file":"postgresql-schema-reader.d.ts","sourceRoot":"","sources":["../../src/migration/postgresql-schema-reader.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAEjC,OAAO,KAAK,EACV,eAAe,EAGf,YAAY,EACZ,UAAU,EACX,MAAM,gBAAgB,CAAC;AAExB;;;GAGG;AACH,MAAM,MAAM,qBAAqB,GAAG;IAClC,mBAAmB;IACnB,SAAS,EAAE,MAAM,CAAC;IAClB,iBAAiB;IACjB,cAAc,EAAE,MAAM,CAAC;IACvB,aAAa;IACb,UAAU,EAAE,MAAM,CAAC;IACnB,6BAA6B;IAC7B,mBAAmB,EAAE,MAAM,CAAC;IAC5B,yBAAyB;IACzB,oBAAoB,EAAE,MAAM,CAAC;IAC7B,mBAAmB;IACnB,QAAQ,EAAE,UAAU,CAAC;IACrB,mBAAmB;IACnB,QAAQ,EAAE,UAAU,CAAC;CACtB,CAAC;AAEF,MAAM,MAAM,QAAQ,GAAG;IACrB,WAAW,EAAE,MAAM,CAAC;IACpB,SAAS,EAAE,MAAM,CAAC;IAClB,QAAQ,EAAE,MAAM,CAAC;IACjB,wBAAwB,EAAE,MAAM,GAAG,IAAI,CAAC;IACxC,SAAS,EAAE,MAAM,GAAG,IAAI,CAAC;IACzB,aAAa,EAAE,MAAM,GAAG,IAAI,CAAC;IAC7B,WAAW,EAAE,MAAM,CAAC;IACpB,cAAc,EAAE,MAAM,GAAG,IAAI,CAAC;IAC9B,YAAY,EAAE,MAAM,CAAC;IACrB,qBAAqB,EAAE,MAAM,GAAG,IAAI,CAAC;CACtC,CAAC;AAEF,KAAK,OAAO,GAAG;IACb,UAAU,EAAE,MAAM,CAAC;IACnB,WAAW,EAAE,MAAM,CAAC;IACpB,SAAS,EAAE,OAAO,CAAC;IACnB,UAAU,EAAE,OAAO,CAAC;IACpB,UAAU,EAAE,MAAM,CAAC;IACnB,WAAW,EAAE,OAAO,CAAC;IACrB,UAAU,EAAE,KAAK,GAAG,MAAM,CAAC;IAC3B,kBAAkB,EAAE,OAAO,CAAC;CAC7B,CAAC;AAEF,KAAK,SAAS,GAAG;IACf,eAAe,EAAE,MAAM,CAAC;IACxB,WAAW,EAAE,MAAM,CAAC;IACpB,kBAAkB,EAAE,MAAM,CAAC;IAC3B,mBAAmB,EAAE,MAAM,CAAC;IAC5B,WAAW,EAAE,MAAM,CAAC;IACpB,WAAW,EAAE,MAAM,CAAC;CACrB,CAAC;AAEF,cAAM,2BAA2B;IAC/B;;;;;OAKG;IACG,qBAAqB,CAAC,SAAS,EAAE,IAAI,EAAE,KAAK,EAAE,MAAM,GAAG,OAAO,CAAC,YAAY,GAAG,IAAI,CAAC;IA8HzF;;OAEG;IACH,OAAO,CAAC,mBAAmB;IAW3B;;OAEG;IACG,SAAS,CACb,SAAS,EAAE,IAAI,EACf,SAAS,EAAE,MAAM,GAChB,OAAO,CAAC,CAAC,QAAQ,EAAE,EAAE,OAAO,EAAE,EAAE,SAAS,EAAE,CAAC,CAAC;IAqHhD;;;OAGG;IACG,yBAAyB,CAAC,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,MAAM,GAAG,OAAO,CAAC,qBAAqB,EAAE,CAAC;IA+C9F;;;OAGG;YACW,mBAAmB;IAwBjC;;OAEG;IACH,gBAAgB,CACd,QAAQ,EAAE,QAAQ,GACjB,IAAI,CACL,eAAe,EACf,MAAM,GAAG,QAAQ,GAAG,WAAW,GAAG,OAAO,GAAG,YAAY,GAAG,YAAY,CACxE;CA+FF;AAED,eAAO,MAAM,sBAAsB,6BAAoC,CAAC"}
1
+ {"version":3,"file":"postgresql-schema-reader.d.ts","sourceRoot":"","sources":["../../src/migration/postgresql-schema-reader.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAEjC,OAAO,KAAK,EACV,eAAe,EAGf,YAAY,EACZ,UAAU,EACX,MAAM,gBAAgB,CAAC;AAExB;;;GAGG;AACH,MAAM,MAAM,qBAAqB,GAAG;IAClC,mBAAmB;IACnB,SAAS,EAAE,MAAM,CAAC;IAClB,iBAAiB;IACjB,cAAc,EAAE,MAAM,CAAC;IACvB,aAAa;IACb,UAAU,EAAE,MAAM,CAAC;IACnB,6BAA6B;IAC7B,mBAAmB,EAAE,MAAM,CAAC;IAC5B,yBAAyB;IACzB,oBAAoB,EAAE,MAAM,CAAC;IAC7B,mBAAmB;IACnB,QAAQ,EAAE,UAAU,CAAC;IACrB,mBAAmB;IACnB,QAAQ,EAAE,UAAU,CAAC;CACtB,CAAC;AAEF,MAAM,MAAM,QAAQ,GAAG;IACrB,WAAW,EAAE,MAAM,CAAC;IACpB,SAAS,EAAE,MAAM,CAAC;IAClB,QAAQ,EAAE,MAAM,CAAC;IACjB,wBAAwB,EAAE,MAAM,GAAG,IAAI,CAAC;IACxC,SAAS,EAAE,MAAM,GAAG,IAAI,CAAC;IACzB,aAAa,EAAE,MAAM,GAAG,IAAI,CAAC;IAC7B,WAAW,EAAE,MAAM,CAAC;IACpB,cAAc,EAAE,MAAM,GAAG,IAAI,CAAC;IAC9B,YAAY,EAAE,MAAM,CAAC;IACrB,qBAAqB,EAAE,MAAM,GAAG,IAAI,CAAC;CACtC,CAAC;AAEF,KAAK,OAAO,GAAG;IACb,UAAU,EAAE,MAAM,CAAC;IACnB,WAAW,EAAE,MAAM,CAAC;IACpB,SAAS,EAAE,OAAO,CAAC;IACnB,UAAU,EAAE,OAAO,CAAC;IACpB,UAAU,EAAE,MAAM,CAAC;IACnB,WAAW,EAAE,OAAO,CAAC;IACrB,UAAU,EAAE,KAAK,GAAG,MAAM,CAAC;IAC3B,kBAAkB,EAAE,OAAO,CAAC;CAC7B,CAAC;AAEF,KAAK,SAAS,GAAG;IACf,eAAe,EAAE,MAAM,CAAC;IACxB,WAAW,EAAE,MAAM,CAAC;IACpB,kBAAkB,EAAE,MAAM,CAAC;IAC3B,mBAAmB,EAAE,MAAM,CAAC;IAC5B,WAAW,EAAE,MAAM,CAAC;IACpB,WAAW,EAAE,MAAM,CAAC;CACrB,CAAC;AAEF,cAAM,2BAA2B;IAC/B;;;;;OAKG;IACG,qBAAqB,CAAC,SAAS,EAAE,IAAI,EAAE,KAAK,EAAE,MAAM,GAAG,OAAO,CAAC,YAAY,GAAG,IAAI,CAAC;IA8HzF;;OAEG;IACH,OAAO,CAAC,mBAAmB;IAW3B;;OAEG;IACG,SAAS,CACb,SAAS,EAAE,IAAI,EACf,SAAS,EAAE,MAAM,GAChB,OAAO,CAAC,CAAC,QAAQ,EAAE,EAAE,OAAO,EAAE,EAAE,SAAS,EAAE,CAAC,CAAC;IA2HhD;;;OAGG;IACG,yBAAyB,CAAC,EAAE,EAAE,IAAI,EAAE,SAAS,EAAE,MAAM,GAAG,OAAO,CAAC,qBAAqB,EAAE,CAAC;IA+C9F;;;OAGG;YACW,mBAAmB;IAwBjC;;OAEG;IACH,gBAAgB,CACd,QAAQ,EAAE,QAAQ,GACjB,IAAI,CACL,eAAe,EACf,MAAM,GAAG,QAAQ,GAAG,WAAW,GAAG,OAAO,GAAG,YAAY,GAAG,YAAY,CACxE;CA+FF;AAED,eAAO,MAAM,sBAAsB,6BAAoC,CAAC"}