sonamu 0.8.13 → 0.8.14

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 (96) hide show
  1. package/dist/api/sonamu.d.ts.map +1 -1
  2. package/dist/api/sonamu.js +2 -3
  3. package/dist/auth/auth-generator.d.ts +8 -0
  4. package/dist/auth/auth-generator.d.ts.map +1 -1
  5. package/dist/auth/auth-generator.js +33 -1
  6. package/dist/auth/better-auth-entities.d.ts.map +1 -1
  7. package/dist/auth/better-auth-entities.js +12 -2
  8. package/dist/bin/cli.js +18 -3
  9. package/dist/cone/cone-generator.js +10 -4
  10. package/dist/database/knex.d.ts.map +1 -1
  11. package/dist/database/knex.js +64 -2
  12. package/dist/database/puri.d.ts +9 -1
  13. package/dist/database/puri.d.ts.map +1 -1
  14. package/dist/database/puri.js +42 -1
  15. package/dist/database/puri.types.d.ts +2 -0
  16. package/dist/database/puri.types.d.ts.map +1 -1
  17. package/dist/database/puri.types.js +6 -2
  18. package/dist/entity/entity-manager.d.ts +149 -1
  19. package/dist/entity/entity-manager.d.ts.map +1 -1
  20. package/dist/entity/entity-manager.js +68 -4
  21. package/dist/migration/__tests__/code-generation.search-text.test.js +435 -0
  22. package/dist/migration/code-generation.d.ts.map +1 -1
  23. package/dist/migration/code-generation.js +696 -32
  24. package/dist/migration/migration-set.js +3 -1
  25. package/dist/migration/postgresql-schema-reader.d.ts +16 -2
  26. package/dist/migration/postgresql-schema-reader.d.ts.map +1 -1
  27. package/dist/migration/postgresql-schema-reader.js +281 -7
  28. package/dist/stream/sse.js +5 -3
  29. package/dist/template/__tests__/generated.template.search-text.test.js +99 -0
  30. package/dist/template/generated.template.test-d.js +24 -0
  31. package/dist/template/implementations/generated.template.d.ts.map +1 -1
  32. package/dist/template/implementations/generated.template.js +2 -2
  33. package/dist/template/implementations/init_types.template.d.ts.map +1 -1
  34. package/dist/template/implementations/init_types.template.js +11 -3
  35. package/dist/template/zod-converter.d.ts.map +1 -1
  36. package/dist/template/zod-converter.js +6 -2
  37. package/dist/testing/dev-test-routes.d.ts.map +1 -1
  38. package/dist/testing/dev-test-routes.js +5 -3
  39. package/dist/testing/fixture-generator.d.ts +13 -0
  40. package/dist/testing/fixture-generator.d.ts.map +1 -1
  41. package/dist/testing/fixture-generator.js +105 -8
  42. package/dist/testing/fixture-manager.d.ts.map +1 -1
  43. package/dist/testing/fixture-manager.js +19 -2
  44. package/dist/types/__tests__/entity-json-schema-search-text.test.js +256 -0
  45. package/dist/types/types.d.ts +494 -1
  46. package/dist/types/types.d.ts.map +1 -1
  47. package/dist/types/types.js +117 -13
  48. package/dist/ui/api.d.ts.map +1 -1
  49. package/dist/ui/api.js +14 -2
  50. package/dist/ui/cdd-service.d.ts +16 -14
  51. package/dist/ui/cdd-service.d.ts.map +1 -1
  52. package/dist/ui/cdd-service.js +145 -37
  53. package/dist/ui/cdd-types.d.ts +60 -0
  54. package/dist/ui/cdd-types.d.ts.map +1 -0
  55. package/dist/ui/cdd-types.js +3 -0
  56. package/dist/ui-web/assets/index-D4XFBV-f.css +1 -0
  57. package/dist/ui-web/assets/{index-CQ_S40bD.js → index-D_19-Pi4.js} +87 -87
  58. package/dist/ui-web/index.html +2 -2
  59. package/package.json +7 -3
  60. package/src/api/sonamu.ts +1 -2
  61. package/src/auth/auth-generator.ts +38 -0
  62. package/src/auth/better-auth-entities.ts +18 -1
  63. package/src/bin/cli.ts +15 -1
  64. package/src/cone/cone-generator.ts +9 -3
  65. package/src/database/knex.ts +62 -4
  66. package/src/database/puri.ts +71 -0
  67. package/src/database/puri.types.ts +2 -0
  68. package/src/entity/entity-manager.ts +95 -3
  69. package/src/migration/__tests__/code-generation.search-text.test.ts +390 -0
  70. package/src/migration/code-generation.ts +848 -34
  71. package/src/migration/migration-set.ts +2 -0
  72. package/src/migration/postgresql-schema-reader.ts +366 -9
  73. package/src/skills/sonamu/auth-migration.md +80 -0
  74. package/src/skills/sonamu/cdd.md +148 -28
  75. package/src/skills/sonamu/cone.md +16 -0
  76. package/src/skills/sonamu/entity-relations.md +1 -1
  77. package/src/skills/sonamu/fixture-cli.md +4 -0
  78. package/src/skills/sonamu/frontend.md +65 -0
  79. package/src/skills/sonamu/migration.md +3 -1
  80. package/src/skills/sonamu/model.md +28 -0
  81. package/src/skills/sonamu/workflow.md +12 -5
  82. package/src/stream/sse.ts +4 -2
  83. package/src/template/__tests__/generated.template.search-text.test.ts +89 -0
  84. package/src/template/generated.template.test-d.ts +46 -0
  85. package/src/template/implementations/generated.template.ts +4 -1
  86. package/src/template/implementations/init_types.template.ts +20 -5
  87. package/src/template/zod-converter.ts +5 -0
  88. package/src/testing/dev-test-routes.ts +4 -2
  89. package/src/testing/fixture-generator.ts +157 -9
  90. package/src/testing/fixture-manager.ts +15 -1
  91. package/src/types/__tests__/entity-json-schema-search-text.test.ts +179 -0
  92. package/src/types/types.ts +168 -12
  93. package/src/ui/api.ts +24 -1
  94. package/src/ui/cdd-service.ts +195 -55
  95. package/src/ui/cdd-types.ts +73 -0
  96. package/dist/ui-web/assets/index-egkMxKos.css +0 -1
@@ -77,6 +77,12 @@ export class FixtureGenerator {
77
77
  const cone = prop.cone;
78
78
  // 1. Relation prop 처리
79
79
  if (isRelationProp(prop)) {
80
+ // BelongsToOne / OneToOne(hasJoinColumn)은 FK 컬럼명({prop.name}_id)으로도 override를 받는다
81
+ const fkColName = `${prop.name}_id`;
82
+ if (fkColName in overrides && (isBelongsToOneRelationProp(prop) || isOneToOneRelationProp(prop) && prop.hasJoinColumn)) {
83
+ fixture[fkColName] = overrides[fkColName];
84
+ continue;
85
+ }
80
86
  const relationValue = await this.generateRelationValue(entity, prop, context);
81
87
  // BelongsToOne, OneToOne(hasJoinColumn)의 경우 foreign key 컬럼명으로 저장
82
88
  if (isBelongsToOneRelationProp(prop) || isOneToOneRelationProp(prop) && prop.hasJoinColumn) {
@@ -621,14 +627,17 @@ export class FixtureGenerator {
621
627
  const apiKey = this.getApiKey();
622
628
  const { createAnthropic } = await import("@ai-sdk/anthropic");
623
629
  const { generateText } = await import("ai");
624
- const { text } = await generateText({
630
+ const rowResponse = await generateText({
625
631
  model: createAnthropic({
626
632
  apiKey
627
633
  })(this.options.llmModel || "claude-sonnet-4-5"),
628
634
  prompt: this.buildRowLLMPrompt(llmProps, entity)
629
635
  });
636
+ if (!rowResponse || typeof rowResponse.text !== "string") {
637
+ throw new Error("Invalid LLM response");
638
+ }
630
639
  // 응답을 파싱하여 각 필드에 대한 결과를 캐시에 저장
631
- const rowResult = this.parseRowLLMResponse(text, llmProps);
640
+ const rowResult = this.parseRowLLMResponse(rowResponse.text, llmProps);
632
641
  for (const [fieldName, value] of Object.entries(rowResult)){
633
642
  this.llmCache.set(`${rowKey}:${fieldName}`, value);
634
643
  }
@@ -652,13 +661,16 @@ export class FixtureGenerator {
652
661
  const apiKey = this.getApiKey();
653
662
  const { createAnthropic } = await import("@ai-sdk/anthropic");
654
663
  const { generateText } = await import("ai");
655
- const { text } = await generateText({
664
+ const singleResponse = await generateText({
656
665
  model: createAnthropic({
657
666
  apiKey
658
667
  })(this.options.llmModel || "claude-sonnet-4-5"),
659
668
  prompt: this.buildLLMPrompt(fixtureHint, prop, entity)
660
669
  });
661
- const value = this.parseLLMResponse(text, prop.type);
670
+ if (!singleResponse || typeof singleResponse.text !== "string") {
671
+ throw new Error("Invalid LLM response");
672
+ }
673
+ const value = this.parseLLMResponse(singleResponse.text, prop.type);
662
674
  if (this.options.enableLLMCache) {
663
675
  this.llmCache.set(cacheKey, value);
664
676
  }
@@ -715,7 +727,12 @@ ${outputShape}
715
727
  const jsonText = text.trim().replace(/^```json\s*/i, "").replace(/```\s*$/, "").trim();
716
728
  let parsed;
717
729
  try {
718
- parsed = JSON.parse(jsonText);
730
+ const raw = JSON.parse(jsonText);
731
+ if (typeof raw !== "object" || raw === null || Array.isArray(raw)) {
732
+ !isTest() && console.warn("[FixtureGenerator] Row LLM response is not a plain object:", text);
733
+ return {};
734
+ }
735
+ parsed = raw;
719
736
  } catch {
720
737
  !isTest() && console.warn("[FixtureGenerator] Failed to parse row LLM response:", text);
721
738
  return {};
@@ -1173,8 +1190,88 @@ Rules:
1173
1190
  }
1174
1191
  // 3. targetDb에 삽입 (FixtureManager가 의존성 정렬 처리)
1175
1192
  const results = await FixtureManager.insertFixtures(this.targetDbName, fixtureRecords);
1176
- !isTest() && console.log(chalk.green(`Generated and saved ${results.length} fixtures to ${this.targetDbName}`));
1177
- return results;
1193
+ // 4. companion fixtures 생성 (fixtureCompanions가 선언된 경우)
1194
+ const companionResults = await this.generateCompanions(specs, results);
1195
+ const total = results.length + companionResults.length;
1196
+ !isTest() && console.log(chalk.green(`Generated and saved ${total} fixtures to ${this.targetDbName}`));
1197
+ return [
1198
+ ...results,
1199
+ ...companionResults
1200
+ ];
1201
+ }
1202
+ /**
1203
+ * 부모 fixture 결과를 기반으로 fixtureCompanions에 선언된 companion Entity를 생성합니다.
1204
+ *
1205
+ * generateBatch()에서만 호출되며, companion 생성 시 재귀를 방지하기 위해
1206
+ * generateBatch()를 다시 호출하지 않고 직접 삽입합니다.
1207
+ */ async generateCompanions(specs, parentResults) {
1208
+ const allResults = [];
1209
+ const processedEntities = new Set();
1210
+ for (const spec of specs){
1211
+ if (processedEntities.has(spec.entity)) continue;
1212
+ processedEntities.add(spec.entity);
1213
+ const entity = this.entityManager.get(spec.entity);
1214
+ const idProp = entity.props.find((p)=>p.name === "id");
1215
+ const companions = idProp?.cone?.fixtureCompanions;
1216
+ if (!companions || companions.length === 0) continue;
1217
+ const entityResults = parentResults.filter((r)=>r.entityId === spec.entity);
1218
+ if (entityResults.length === 0) continue;
1219
+ for (const companion of companions){
1220
+ // companion entity에서 부모 entity로의 BelongsToOne FK 컬럼명 파악
1221
+ const companionEntity = this.entityManager.get(companion.entity);
1222
+ const fkProp = companionEntity.props.find((p)=>isRelationProp(p) && isBelongsToOneRelationProp(p) && p.with === spec.entity);
1223
+ if (!fkProp) {
1224
+ !isTest() && console.warn(chalk.yellow(`[Companion] No BelongsToOne relation from ${companion.entity} to ${spec.entity}. Skipping.`));
1225
+ continue;
1226
+ }
1227
+ const fkColName = `${fkProp.name}_id`;
1228
+ // companion의 idProp, usesSequence, count는 companion 단위로 고정
1229
+ const companionIdProp = companionEntity.props.find((p)=>p.name === "id");
1230
+ const usesSequence = companionIdProp?.type === "integer" || companionIdProp?.type === "bigInteger" || companionIdProp?.cone?.fixtureStrategy === "sequence";
1231
+ const companionCount = companion.count ?? 1;
1232
+ // 각 parent result에 대해 companion fixture 생성
1233
+ const context = this.createContext();
1234
+ const companionFixtureRecords = [];
1235
+ for (const parentResult of entityResults){
1236
+ const resolvedOverrides = this.resolveTemplateOverrides(companion.overrides ?? {}, parentResult.data);
1237
+ resolvedOverrides[fkColName] = parentResult.data.id;
1238
+ for(let i = 0; i < companionCount; i++){
1239
+ const fixture = await this.generate(companion.entity, resolvedOverrides, context);
1240
+ const dataForRecord = usesSequence ? {
1241
+ ...fixture,
1242
+ id: Math.floor(Math.random() * 1000000)
1243
+ } : fixture;
1244
+ const records = await FixtureManager.createFixtureRecord(companionEntity, dataForRecord, {
1245
+ singleRecord: true
1246
+ });
1247
+ companionFixtureRecords.push(...records);
1248
+ }
1249
+ }
1250
+ const companionResults = await FixtureManager.insertFixtures(this.targetDbName, companionFixtureRecords);
1251
+ allResults.push(...companionResults);
1252
+ !isTest() && console.log(chalk.green(`[Companion] Generated ${companionResults.length} ${companion.entity} fixtures`));
1253
+ }
1254
+ }
1255
+ return allResults;
1256
+ }
1257
+ /**
1258
+ * overrides 값의 "{{fieldName}}" 템플릿을 부모 fixture 데이터로 치환합니다.
1259
+ *
1260
+ * 예: { "account_id": "{{email}}" } → { "account_id": "user@example.com" }
1261
+ */ resolveTemplateOverrides(overrides, parentData) {
1262
+ const resolved = {};
1263
+ for (const [key, value] of Object.entries(overrides)){
1264
+ if (typeof value === "string" && value.startsWith("{{") && value.endsWith("}}")) {
1265
+ const fieldName = value.slice(2, -2).trim();
1266
+ if (!(fieldName in parentData)) {
1267
+ throw new Error(`템플릿 필드 "${fieldName}"이(가) 부모 fixture 데이터에 존재하지 않습니다 (override key: "${key}")`);
1268
+ }
1269
+ resolved[key] = parentData[fieldName];
1270
+ } else {
1271
+ resolved[key] = value;
1272
+ }
1273
+ }
1274
+ return resolved;
1178
1275
  }
1179
1276
  /**
1180
1277
  * 실제 DB(sourceDb)에서 데이터를 조회하여 fixture DB(targetDb)에 import합니다.
@@ -1235,4 +1332,4 @@ Rules:
1235
1332
  }
1236
1333
  }
1237
1334
 
1238
- //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/testing/fixture-generator.ts"],"sourcesContent":["import chalk from \"chalk\";\nimport type { Knex } from \"knex\";\nimport type { Entity } from \"../entity/entity\";\nimport type { EntityManager } from \"../entity/entity-manager\";\nimport type { EntityProp, FixtureImportResult, FixtureRecord } from \"../types/types\";\nimport { isBelongsToOneRelationProp, isOneToOneRelationProp, isRelationProp } from \"../types/types\";\nimport { isTest } from \"../utils/controller\";\nimport {\n  DataExplorer,\n  type ExploreWithRelationsOptions,\n  type ExploreWithRelationsResult,\n} from \"./data-explorer\";\nimport { type FakerMappings, fakerMappings } from \"./faker-mappings\";\nimport { FixtureManager } from \"./fixture-manager\";\n\nexport type Locale = \"ko\" | \"en\" | \"ja\";\n\nexport type FixtureGeneratorOptions = {\n  locale?: Locale;\n  useLLM?: boolean;\n  enableLLMCache?: boolean;\n  llmModel?: string;\n};\n\nexport type GeneratorContext = {\n  /** 생성 중인 fixture들 (메모리 상) */\n  fixtures: Map<string, Record<string, unknown>>;\n\n  /** 참조 데이터 캐시 (DataExplorer 결과) */\n  referenceCache: Map<string, Record<string, unknown>[]>;\n\n  /** 이미 import된 레코드를 추적하여 중복 import를 방지합니다 */\n  importedRecords: Set<string>; // \"User#123\"\n};\n\nexport class FixtureGenerator {\n  private dataExplorer: DataExplorer;\n  private locale: Locale;\n  private mappings: FakerMappings;\n  private llmCache: Map<string, unknown> = new Map();\n  private entityCache: Map<string, Entity> = new Map();\n  private options: FixtureGeneratorOptions;\n\n  constructor(\n    private sourceDb: Knex,\n    // FixtureManager.insertFixtures가 dbName 문자열을 받기 때문에 직접 사용하지 않습니다\n    // 미래 확장성을 위해 API 시그니처에는 포함시켰습니다\n    _targetDb: Knex,\n    private targetDbName: \"fixture\" | \"test\" | \"production_master\",\n    private entityManager: typeof EntityManager,\n    options?: FixtureGeneratorOptions,\n  ) {\n    this.dataExplorer = new DataExplorer(sourceDb, entityManager);\n    this.locale = options?.locale || \"ko\";\n    this.mappings = fakerMappings;\n    this.options = {\n      locale: options?.locale || \"ko\",\n      useLLM: options?.useLLM || false,\n      enableLLMCache: options?.enableLLMCache !== false,\n      llmModel: options?.llmModel || \"claude-sonnet-4-5\",\n    };\n  }\n\n  /**\n   * Fixture 생성 (단일)\n   * @returns 생성된 fixture 데이터 (메모리 상)\n   */\n  async generate(\n    entityName: string,\n    overrides: Record<string, unknown> = {},\n    context: GeneratorContext = this.createContext(),\n  ): Promise<Record<string, unknown>> {\n    // Entity 캐싱: 테스트에서 entity cone 수정이 반영되도록 보장\n    let entity = this.entityCache.get(entityName);\n    if (!entity) {\n      entity = this.entityManager.get(entityName);\n      this.entityCache.set(entityName, entity);\n    }\n\n    const tempId = `${entityName}#temp#${Date.now()}`; // 임시 ID\n\n    // LLM row 단위 생성을 위한 고유 키 (같은 row의 필드들이 동일한 rowKey를 공유)\n    const rowKey = this.options.useLLM ? `${entityName}#row#${Date.now()}` : undefined;\n\n    // 각 prop별 값 생성\n    const fixture: Record<string, unknown> = {};\n\n    for (const prop of entity.props) {\n      // Virtual prop은 스킵\n      if (\"virtual\" in prop && prop.virtual) {\n        continue;\n      }\n\n      // id prop 처리\n      if (prop.name === \"id\") {\n        if (\"cone\" in prop && prop.cone?.fixtureStrategy === \"sequence\") {\n          // DB sequence가 자동 할당하므로 스킵 (User 등)\n          continue;\n        }\n        if (prop.type === \"string\") {\n          // DB DEFAULT 없는 string PK: alphanumeric 32자 생성 (better-auth 스타일)\n          const { faker: _faker } = await import(\"@faker-js/faker\");\n          fixture[prop.name] = _faker.string.alphanumeric(32);\n          continue;\n        }\n        if (prop.type === \"uuid\") {\n          const { faker: _faker } = await import(\"@faker-js/faker\");\n          fixture[prop.name] = _faker.string.uuid();\n          continue;\n        }\n        // integer/bigInteger PK: generateBatch에서 tempId를 넣으므로 여기선 스킵\n        continue;\n      }\n\n      // override가 있으면 사용\n      if (prop.name in overrides) {\n        fixture[prop.name] = overrides[prop.name];\n        continue;\n      }\n\n      // cone에서 생성 전략 확인\n      const cone = prop.cone;\n\n      // 1. Relation prop 처리\n      if (isRelationProp(prop)) {\n        const relationValue = await this.generateRelationValue(entity, prop, context);\n        // BelongsToOne, OneToOne(hasJoinColumn)의 경우 foreign key 컬럼명으로 저장\n        if (\n          isBelongsToOneRelationProp(prop) ||\n          (isOneToOneRelationProp(prop) && prop.hasJoinColumn)\n        ) {\n          fixture[`${prop.name}_id`] = relationValue;\n        } else {\n          fixture[prop.name] = relationValue;\n        }\n        continue;\n      }\n\n      // 2. cone.note + LLM 사용 (useLLM이면 fixtureGenerator보다 우선)\n      if (cone?.note && this.options.useLLM) {\n        try {\n          fixture[prop.name] = await this.generateWithLLM(cone.note, prop, entity, rowKey);\n          continue;\n        } catch (error) {\n          console.warn(\n            `[FixtureGenerator] LLM generation failed for ${entity.id}.${prop.name}, falling back to fixtureGenerator or default`,\n            error instanceof Error ? error.message : error,\n          );\n          // fallback: fixtureGenerator → fixtureDefault → 기본값으로 계속\n        }\n      }\n\n      // 3. fixtureGenerator 사용\n      if (cone?.fixtureGenerator) {\n        fixture[prop.name] = await this.executeGenerator(\n          cone.fixtureGenerator as string,\n          prop,\n          entity,\n        );\n        continue;\n      }\n\n      // 4. fixtureDefault 사용\n      if (cone?.fixtureDefault !== undefined) {\n        fixture[prop.name] = cone.fixtureDefault;\n        continue;\n      }\n\n      // 5. 타입별 기본 생성\n      fixture[prop.name] = await this.generateDefaultValue(prop, entity);\n    }\n\n    // 6. email 필드가 있고 name 필드가 있으면, email의 로컬 파트를 name 기반으로 보정\n    if (\"email\" in fixture && typeof fixture.email === \"string\" && !(\"email\" in overrides)) {\n      const nameValue = fixture.name || fixture.username || fixture.full_name || fixture.name_en;\n      if (nameValue && typeof nameValue === \"string\") {\n        const domain = fixture.email.split(\"@\")[1] || \"example.com\";\n        const romanized = await this.romanizeName(nameValue);\n        fixture.email = `${romanized}@${domain}`;\n      }\n    }\n\n    // 7. password 필드 암호화\n    if (\"password\" in fixture && fixture.password && typeof fixture.password === \"string\") {\n      const bcrypt = await import(\"bcrypt\");\n      fixture.password = await bcrypt.hash(fixture.password, 10);\n    }\n\n    context.fixtures.set(tempId, fixture);\n    return fixture;\n  }\n\n  /**\n   * Relation 값 생성 + 자동 Import\n   */\n  private async generateRelationValue(\n    entity: Entity,\n    prop: EntityProp,\n    context: GeneratorContext,\n  ): Promise<number | null> {\n    if (!isRelationProp(prop)) {\n      throw new Error(`FixtureGenerator: ${entity.id}.${prop.name} is not a relation prop`);\n    }\n\n    // BelongsToOne, OneToOne(hasJoinColumn)만 처리\n    if (\n      !isBelongsToOneRelationProp(prop) &&\n      !(isOneToOneRelationProp(prop) && prop.hasJoinColumn)\n    ) {\n      return null;\n    }\n\n    const cone = prop.cone;\n    const dataSource = cone?.dataSource;\n\n    // DataExplorer로 참조 데이터 조회 (sourceDb)\n    // 관계 체인을 따라가기 위해 exploreWithRelations 사용\n    if (dataSource) {\n      const cacheKey = `${prop.with}:${JSON.stringify(dataSource)}`;\n\n      if (!context.referenceCache.has(cacheKey)) {\n        const exploreResult = await this.dataExplorer.exploreWithRelations(prop.with, {\n          strategy: dataSource.strategy,\n          limit:\n            ((dataSource.config as Record<string, unknown> | undefined)?.limit as\n              | number\n              | undefined) || 10,\n          includeRelations: true,\n          maxDepth: 3,\n          ...(dataSource.config as Record<string, unknown> | undefined),\n        });\n        context.referenceCache.set(cacheKey, exploreResult.main.records);\n\n        // 조회한 데이터와 관계된 모든 엔티티를 targetDb에 import\n        await this.importExploreResult(exploreResult, context);\n      }\n\n      const candidates = context.referenceCache.get(cacheKey);\n      if (candidates && candidates.length > 0) {\n        // 랜덤하게 하나 선택\n        const selected = candidates[Math.floor(Math.random() * candidates.length)];\n        return selected.id as number;\n      }\n    }\n\n    // dataSource가 없을 때 자동으로 fixture DB에서 조회 시도\n    // 관계 체인을 따라가기 위해 exploreWithRelations 사용\n    const autoKey = `${prop.with}:auto`;\n    if (!context.referenceCache.has(autoKey)) {\n      // fixture DB(sourceDb)에서 자동 조회 (관계 포함)\n      const autoExploreResult = await this.dataExplorer.exploreWithRelations(prop.with, {\n        strategy: \"random\",\n        limit: 10,\n        includeRelations: true,\n        maxDepth: 3,\n      });\n      context.referenceCache.set(autoKey, autoExploreResult.main.records);\n\n      // 조회한 데이터와 관계된 모든 엔티티를 targetDb에 import\n      if (autoExploreResult.main.records.length > 0) {\n        await this.importExploreResult(autoExploreResult, context);\n      }\n    }\n\n    const autoCandidates = context.referenceCache.get(autoKey);\n    if (autoCandidates && autoCandidates.length > 0) {\n      // 랜덤하게 하나 선택\n      const selected = autoCandidates[Math.floor(Math.random() * autoCandidates.length)];\n      return selected.id as number;\n    }\n\n    // 참조 데이터가 없으면 null 반환 (nullable인 경우)\n    if (prop.nullable) {\n      return null;\n    }\n\n    // nullable이 아니고 데이터도 없으면 에러\n    throw new Error(\n      `FixtureGenerator: ${entity.id}.${prop.name}에 필요한 ${prop.with} 데이터가 없습니다. ` +\n        `먼저 ${prop.with}를 생성하거나 cone.dataSource를 설정하세요.`,\n    );\n  }\n\n  /**\n   * ExploreWithRelations 결과를 targetDb에 import\n   *\n   * 관계 체인을 따라간 결과(main + related)를 모두 import합니다.\n   * 의존성 순서는 FixtureManager.insertFixtures가 자동으로 처리합니다.\n   */\n  private async importExploreResult(\n    exploreResult: ExploreWithRelationsResult,\n    context: GeneratorContext,\n  ): Promise<void> {\n    const allFixtureRecords: FixtureRecord[] = [];\n\n    // 1. Related entities import (Company, Department 등)\n    for (const [entityId, records] of exploreResult.related.entries()) {\n      const entity = this.entityManager.get(entityId);\n      const recordsToImport: Record<string, unknown>[] = [];\n\n      !isTest() &&\n        console.log(\n          chalk.cyan(`Importing related entity: ${entityId} (${records.length} records)`),\n        );\n\n      for (const record of records) {\n        const recordKey = `${entityId}#${record.id}`;\n        if (!context.importedRecords.has(recordKey)) {\n          recordsToImport.push(record);\n          context.importedRecords.add(recordKey);\n        }\n      }\n\n      if (recordsToImport.length > 0) {\n        for (const record of recordsToImport) {\n          !isTest() &&\n            console.log(\n              chalk.gray(\n                `  - Processing ${entityId} record:`,\n                JSON.stringify(record).slice(0, 100),\n              ),\n            );\n          const fixtureRecords = await FixtureManager.createFixtureRecord(\n            entity,\n            record as { id: number | string; [key: string]: string | number | boolean | null },\n            { _db: this.sourceDb, singleRecord: true },\n          );\n          allFixtureRecords.push(...fixtureRecords);\n        }\n      }\n    }\n\n    // 2. Main entity import (Employee 등)\n    const mainEntity = this.entityManager.get(exploreResult.main.entityId);\n    const mainRecordsToImport: Record<string, unknown>[] = [];\n\n    !isTest() &&\n      console.log(\n        chalk.cyan(\n          `Importing main entity: ${exploreResult.main.entityId} (${exploreResult.main.records.length} records)`,\n        ),\n      );\n\n    for (const record of exploreResult.main.records) {\n      const recordKey = `${exploreResult.main.entityId}#${record.id}`;\n      if (!context.importedRecords.has(recordKey)) {\n        mainRecordsToImport.push(record);\n        context.importedRecords.add(recordKey);\n      }\n    }\n\n    if (mainRecordsToImport.length > 0) {\n      for (const record of mainRecordsToImport) {\n        !isTest() &&\n          console.log(\n            chalk.gray(\n              `  - Processing ${exploreResult.main.entityId} record:`,\n              JSON.stringify(record).slice(0, 100),\n            ),\n          );\n        const fixtureRecords = await FixtureManager.createFixtureRecord(\n          mainEntity,\n          record as { id: number | string; [key: string]: string | number | boolean | null },\n          { _db: this.sourceDb, singleRecord: true },\n        );\n        allFixtureRecords.push(...fixtureRecords);\n      }\n    }\n\n    // 3. 모든 fixture를 한 번에 삽입 (의존성 순서 자동 처리)\n    if (allFixtureRecords.length > 0) {\n      await FixtureManager.insertFixtures(this.targetDbName, allFixtureRecords);\n\n      !isTest() &&\n        console.log(\n          chalk.green(\n            `Auto-imported ${exploreResult.main.entityId} with relations: ` +\n              `${exploreResult.main.records.length} main + ${exploreResult.related.size} related entities`,\n          ),\n        );\n    }\n  }\n\n  /**\n   * fixtureGenerator 실행 (Faker.js만 지원)\n   *\n   * faker.* 형식의 표현식을 안전하게 파싱하여 실행합니다.\n   * 예: \"faker.internet.email()\" → faker.internet.email()\n   * 예: \"faker.lorem.words(3)\" → faker.lorem.words(3)\n   */\n  private async executeGenerator(\n    generator: string,\n    prop: EntityProp,\n    entity: Entity,\n  ): Promise<unknown> {\n    // Faker.js 표현식만 지원\n    if (generator.startsWith(\"faker.\")) {\n      // username이나 name 필드는 한국어 faker 사용\n      const isNameField = prop.name === \"username\" || prop.name === \"name\";\n      const fakerModule = await import(\"@faker-js/faker\");\n      const faker = isNameField ? fakerModule.fakerKO : fakerModule.faker;\n      const expr = generator.slice(6); // \"faker.\" 제거\n\n      try {\n        // 함수 경로와 인자 파싱\n        const match = expr.match(/^([\\w.]+)(?:\\((.*?)\\))?$/);\n        if (!match) {\n          throw new Error(\n            `FixtureGenerator: Invalid faker expression for ${prop.name}: ${generator}`,\n          );\n        }\n\n        const [, path, argsStr] = match;\n        const parts = path.split(\".\");\n\n        // faker 객체에서 함수 찾기\n        let fn: unknown = faker;\n        for (const part of parts) {\n          if (typeof fn === \"object\" && fn !== null && part in fn) {\n            fn = (fn as Record<string, unknown>)[part];\n          } else {\n            throw new Error(`FixtureGenerator: Invalid faker path for ${prop.name}: faker.${path}`);\n          }\n        }\n\n        // 함수가 아니면 에러\n        if (typeof fn !== \"function\") {\n          throw new Error(`FixtureGenerator: faker.${path} is not a function (for ${prop.name})`);\n        }\n\n        let args: unknown[] = [];\n        if (argsStr?.trim()) {\n          args = this.parseGeneratorArgs(argsStr, prop.name);\n        }\n\n        return fn(...args);\n      } catch (error) {\n        !isTest() &&\n          console.log(\n            chalk.yellow(\n              `Failed to execute generator \"${generator}\" for ${prop.name}, falling back to default:`,\n            ),\n            error,\n          );\n        return this.generateDefaultValue(prop, entity);\n      }\n    }\n\n    // faker 이외의 표현식은 지원하지 않음\n    !isTest() &&\n      console.log(\n        chalk.yellow(\n          `Unsupported generator expression for ${prop.name}: ${generator}. Only faker.* expressions are supported. Using default value.`,\n        ),\n      );\n    return this.generateDefaultValue(prop, entity);\n  }\n\n  /**\n   * 필드의 타입과 이름을 분석하여 적절한 기본값을 생성합니다.\n   *\n   * 우선순위:\n   * 1. 필드명 패턴 매칭 (salary, budget 등 의미있는 데이터)\n   * 2. 특수 케이스 (Department name 등 도메인 지식)\n   * 3. 배열 타입 (JSON 배열)\n   * 4. Enum 타입\n   * 5. 타입별 기본값\n   */\n  private async generateDefaultValue(prop: EntityProp, entity?: Entity): Promise<unknown> {\n    const fakerModule = await import(\"@faker-js/faker\");\n    const faker = fakerModule.faker;\n    const fakerKO = fakerModule.fakerKO;\n    const fakerJA = fakerModule.fakerJA;\n\n    const localeFaker = this.locale === \"ko\" ? fakerKO : this.locale === \"ja\" ? fakerJA : faker;\n\n    /**\n     * 1. Entity-specific 특수 케이스를 먼저 처리합니다.\n     * field_patterns보다 우선하여, 특정 엔티티의 필드에 도메인에 맞는 값을 생성합니다.\n     * 예: Department.name → 한국어 부서명 (사람 이름이 아님)\n     */\n\n    /**\n     * Department name은 한국어 부서명 목록에서 선택합니다.\n     * 고유성을 위해 70% 확률로 prefix/suffix를 추가합니다.\n     */\n    if (entity?.id === \"Department\" && prop.name === \"name\") {\n      const departments = [\n        \"개발팀\",\n        \"기획팀\",\n        \"마케팅팀\",\n        \"영업팀\",\n        \"인사팀\",\n        \"총무팀\",\n        \"재무팀\",\n        \"회계팀\",\n        \"법무팀\",\n        \"디자인팀\",\n        \"IT팀\",\n        \"고객지원팀\",\n        \"품질관리팀\",\n        \"연구개발팀\",\n        \"생산팀\",\n        \"구매팀\",\n        \"물류팀\",\n      ];\n      const prefixes = [\"신규\", \"통합\", \"전략\", \"글로벌\", \"디지털\", \"핵심\"];\n      const suffixes = [\"1팀\", \"2팀\", \"3팀\", \"A팀\", \"B팀\", \"본부\", \"센터\", \"그룹\"];\n\n      const dept = faker.helpers.arrayElement(departments);\n\n      const random = Math.random();\n      if (random > 0.7) {\n        const prefix = faker.helpers.arrayElement(prefixes);\n        return `${prefix} ${dept}`;\n      }\n      if (random > 0.4) {\n        const suffix = faker.helpers.arrayElement(suffixes);\n        return `${dept} ${suffix}`;\n      }\n      return dept;\n    }\n\n    /**\n     * 2. 필드명에서 의미를 추론하여 현실적인 데이터를 생성합니다.\n     * 예: salary → 30M~150M (한국 연봉 범위)\n     *     budget → 10M~500M (프로젝트 예산 범위)\n     */\n    const localeMappings = this.mappings[this.locale] || this.mappings.en;\n    const normalizedName = prop.name.toLowerCase().replace(/_/g, \"\");\n\n    for (const [pattern, config] of Object.entries(localeMappings.field_patterns)) {\n      if (normalizedName.includes(pattern.toLowerCase())) {\n        try {\n          return await this.executeFakerExpression(config.faker, prop);\n        } catch (error) {\n          !isTest() &&\n            console.log(\n              chalk.yellow(\n                `Failed to execute field pattern \"${pattern}\" for ${prop.name}, falling back:`,\n              ),\n              error,\n            );\n          break;\n        }\n      }\n    }\n\n    /**\n     * 3. JSON 타입이면서 배열인 경우 (SonamuFile[], string[] 등)\n     * 필드명 패턴을 보고 적절한 배열 데이터를 생성합니다.\n     */\n    if (prop.type === \"json\" && \"id\" in prop && prop.id) {\n      if (prop.id.endsWith(\"[]\")) {\n        return this.generateArrayValue(prop, entity, faker, localeFaker);\n      }\n    }\n\n    /** 4. Enum 타입은 정의된 값 중 하나를 랜덤 선택합니다 */\n    if (prop.type === \"enum\") {\n      let enumValues: string[] = [];\n\n      if (\"enum\" in prop && Array.isArray(prop.enum) && prop.enum.length > 0) {\n        enumValues = prop.enum;\n      } else if (\"id\" in prop && prop.id && entity?.enumLabels?.[prop.id]) {\n        enumValues = Object.keys(entity.enumLabels[prop.id]);\n      }\n\n      if (enumValues.length > 0) {\n        return faker.helpers.arrayElement(enumValues);\n      }\n      return prop.nullable ? null : \"UNKNOWN\";\n    }\n\n    if (prop.type === \"enum[]\") {\n      let enumValues: string[] = [];\n\n      if (\"enum\" in prop && Array.isArray(prop.enum) && prop.enum.length > 0) {\n        enumValues = prop.enum;\n      } else if (\"id\" in prop && prop.id && entity?.enumLabels?.[prop.id]) {\n        enumValues = Object.keys(entity.enumLabels[prop.id]);\n      }\n\n      if (enumValues.length > 0) {\n        return [faker.helpers.arrayElement(enumValues)];\n      }\n      return [];\n    }\n\n    /**\n     * 5. Vector 타입은 현재 지원하지 않으므로 null을 반환합니다.\n     * 향후 AI embedding 생성 기능 추가 시 구현 예정입니다.\n     */\n    if (prop.type === \"vector\" || prop.type === \"vector[]\" || prop.type === \"tsvector\") {\n      return null;\n    }\n\n    /** 6. 타입별 기본 Faker 표현식을 실행합니다 */\n    const typeDefault = localeMappings.type_defaults[prop.type];\n    if (typeDefault) {\n      try {\n        return await this.executeFakerExpression(typeDefault.faker, prop);\n      } catch (error) {\n        !isTest() &&\n          console.log(\n            chalk.yellow(`Failed to execute type default for ${prop.type}, using fallback:`, error),\n          );\n      }\n    }\n\n    /** 7. 매핑되지 않은 타입은 기본 Faker 함수로 처리합니다 */\n    switch (prop.type) {\n      case \"string\":\n      case \"string[]\":\n        return faker.lorem.words(3);\n      case \"integer\":\n        return faker.number.int({ min: 1, max: 1000 });\n      case \"integer[]\":\n        return [faker.number.int({ min: 1, max: 1000 })];\n      case \"bigInteger\":\n        return faker.number.bigInt({ min: 1n, max: 1000n });\n      case \"bigInteger[]\":\n        return [faker.number.bigInt({ min: 1n, max: 1000n })];\n      case \"number\":\n      case \"numeric\":\n        return faker.number.float({ min: 0, max: 1000 });\n      case \"number[]\":\n      case \"numeric[]\":\n        return [faker.number.float({ min: 0, max: 1000 })];\n      case \"boolean\":\n        return faker.datatype.boolean();\n      case \"boolean[]\":\n        return [faker.datatype.boolean()];\n      case \"date\":\n      case \"date[]\":\n        return faker.date.past();\n      case \"json\":\n        return {};\n      case \"uuid\":\n      case \"uuid[]\":\n        return faker.string.uuid();\n      default:\n        return null;\n    }\n  }\n\n  /**\n   * 배열 타입의 값을 생성합니다.\n   *\n   * 타입 ID와 필드명 패턴을 분석하여 적절한 배열 데이터를 생성합니다.\n   * 예: image_urls → [{url, name, mime_type}, ...]\n   *     tag_ids → [1, 23, 45]\n   */\n  private generateArrayValue(\n    prop: EntityProp,\n    _entity: Entity | undefined,\n    faker: typeof import(\"@faker-js/faker\").faker,\n    _localeFaker: typeof import(\"@faker-js/faker\").faker,\n  ): unknown[] {\n    const count = faker.number.int({ min: 1, max: 3 });\n\n    /** SonamuFile[]은 Sonamu 내장 타입으로 구조가 정해져 있습니다 */\n    if (\"id\" in prop && prop.id === \"SonamuFile[]\") {\n      return Array.from({ length: count }, () => ({\n        url: faker.image.url(),\n        name: faker.system.fileName(),\n        mime_type: faker.helpers.arrayElement([\n          \"image/jpeg\",\n          \"image/png\",\n          \"image/gif\",\n          \"application/pdf\",\n        ]),\n      }));\n    }\n\n    /** 필드명에서 배열의 용도를 추론합니다 */\n    const normalizedName = prop.name.toLowerCase().replace(/_/g, \"\");\n\n    if (normalizedName.includes(\"url\") || normalizedName.includes(\"image\")) {\n      return Array.from({ length: count }, () => faker.internet.url());\n    }\n\n    if (normalizedName.includes(\"id\") && normalizedName.endsWith(\"s\")) {\n      return Array.from({ length: count }, () => faker.number.int({ min: 1, max: 100 }));\n    }\n\n    if (normalizedName.includes(\"tag\") || normalizedName.includes(\"name\")) {\n      return Array.from({ length: count }, () => faker.lorem.word());\n    }\n\n    /** 패턴 매칭되지 않으면 빈 배열을 반환합니다 */\n    return [];\n  }\n\n  /**\n   * JSON 매핑의 Faker 표현식을 파싱하여 실행합니다.\n   *\n   * 표현식 예시:\n   * - \"faker.internet.email()\" → 인자 없음\n   * - \"faker.number.int({ min: 1, max: 100 })\" → JSON 인자\n   * - \"{}\" → 리터럴 값 (JSON.parse)\n   *\n   * fakerKO, fakerJA도 지원하여 다국어 데이터를 생성합니다.\n   */\n  private async executeFakerExpression(expression: string, prop: EntityProp): Promise<unknown> {\n    const fakerModule = await import(\"@faker-js/faker\");\n    const faker = fakerModule.faker;\n    const fakerKO = fakerModule.fakerKO;\n    const fakerJA = fakerModule.fakerJA;\n\n    /** Faker 표현식이 아닌 리터럴 값은 JSON으로 파싱합니다 */\n    if (!expression.startsWith(\"faker\")) {\n      try {\n        return JSON.parse(expression);\n      } catch {\n        return expression;\n      }\n    }\n\n    /** 표현식에서 Faker 객체와 경로를 추출합니다 */\n    const match = expression.match(/^(faker|fakerKO|fakerJA)\\.(.*?)$/);\n    if (!match) {\n      throw new Error(`Invalid faker expression: ${expression}`);\n    }\n\n    const [, fakerName, expr] = match;\n    const selectedFaker =\n      fakerName === \"fakerKO\" ? fakerKO : fakerName === \"fakerJA\" ? fakerJA : faker;\n\n    const funcMatch = expr.match(/^([\\w.]+)(?:\\((.*?)\\))?$/);\n    if (!funcMatch) {\n      throw new Error(`Invalid faker expression for ${prop.name}: ${expression}`);\n    }\n\n    const [, path, argsStr] = funcMatch;\n    const parts = path.split(\".\");\n\n    /** 점 표기법(dot notation)으로 Faker 함수를 찾아갑니다 */\n    let fn: unknown = selectedFaker;\n    for (const part of parts) {\n      if (typeof fn === \"object\" && fn !== null && part in fn) {\n        fn = (fn as Record<string, unknown>)[part];\n      } else {\n        throw new Error(`Invalid faker path for ${prop.name}: ${fakerName}.${path}`);\n      }\n    }\n\n    if (typeof fn !== \"function\") {\n      throw new Error(`${fakerName}.${path} is not a function (for ${prop.name})`);\n    }\n\n    let args: unknown[] = [];\n    if (argsStr?.trim()) {\n      args = this.parseGeneratorArgs(argsStr, prop.name);\n    }\n\n    return fn(...args);\n  }\n\n  /**\n   * fixtureHint를 LLM에게 전달하여 현실적인 테스트 데이터를 생성합니다.\n   *\n   * faker.js로는 생성하기 어려운 복잡한 텍스트(자기소개, 설명문 등)를\n   * LLM을 활용하여 생성합니다. 동일한 hint에 대한 중복 호출을 방지하기 위해\n   * 캐싱을 기본으로 지원합니다 (LLM API 비용 절감).\n   *\n   * ai 패키지는 dynamic import로 불러오므로, useLLM이 false인 경우\n   * 의존성이 설치되지 않아도 fixture 생성이 정상 동작합니다.\n   */\n  private async generateWithLLM(\n    fixtureHint: string,\n    prop: EntityProp,\n    entity: Entity,\n    rowKey?: string,\n  ): Promise<unknown> {\n    // rowKey가 있으면 row 단위 생성 전략 사용\n    if (rowKey) {\n      const rowCacheKey = `${rowKey}:${prop.name}`;\n\n      // 이미 이 row에 대한 LLM 호출이 완료된 경우 캐시에서 바로 반환\n      if (this.llmCache.has(rowCacheKey)) {\n        return this.llmCache.get(rowCacheKey);\n      }\n\n      // 새 row: LLM 대상 prop 전체를 한 번에 생성\n      const llmProps = entity.props.filter((p) => {\n        if (isRelationProp(p)) return false;\n        if (p.cone?.fixtureGenerator) return false;\n        if (p.name === \"id\" && p.cone?.fixtureStrategy === \"sequence\") return false;\n        return !!p.cone?.note;\n      });\n\n      // llmProps가 비어있으면 단일 필드 방식으로 fallback\n      if (llmProps.length === 0) {\n        !isTest() &&\n          console.log(\n            `[FixtureGenerator] llmProps is empty for ${entity.id}.${prop.name}, using single field fallback`,\n          );\n        return this.generateSingleWithLLM(fixtureHint, prop, entity);\n      }\n\n      const apiKey = this.getApiKey();\n      const { createAnthropic } = await import(\"@ai-sdk/anthropic\");\n      const { generateText } = await import(\"ai\");\n\n      const { text } = await generateText({\n        model: createAnthropic({ apiKey })(this.options.llmModel || \"claude-sonnet-4-5\"),\n        prompt: this.buildRowLLMPrompt(llmProps, entity),\n      });\n\n      // 응답을 파싱하여 각 필드에 대한 결과를 캐시에 저장\n      const rowResult = this.parseRowLLMResponse(text, llmProps);\n      for (const [fieldName, value] of Object.entries(rowResult)) {\n        this.llmCache.set(`${rowKey}:${fieldName}`, value);\n      }\n\n      // 요청한 필드의 값 반환 (없으면 단일 필드 fallback)\n      if (this.llmCache.has(rowCacheKey)) {\n        return this.llmCache.get(rowCacheKey);\n      }\n\n      // 만약 row 응답에 이 필드가 누락된 경우 단일 필드 fallback\n      return this.generateSingleWithLLM(fixtureHint, prop, entity);\n    }\n\n    // rowKey가 없으면 기존 단일 필드 방식\n    return this.generateSingleWithLLM(fixtureHint, prop, entity);\n  }\n\n  /**\n   * 단일 필드를 LLM으로 생성합니다 (rowKey 없을 때 fallback용)\n   */\n  private async generateSingleWithLLM(\n    fixtureHint: string,\n    prop: EntityProp,\n    entity: Entity,\n  ): Promise<unknown> {\n    const cacheKey = `${entity.id}:${prop.name}:${fixtureHint}`;\n    if (this.options.enableLLMCache && this.llmCache.has(cacheKey)) {\n      return this.llmCache.get(cacheKey);\n    }\n\n    const apiKey = this.getApiKey();\n    const { createAnthropic } = await import(\"@ai-sdk/anthropic\");\n    const { generateText } = await import(\"ai\");\n\n    const { text } = await generateText({\n      model: createAnthropic({ apiKey })(this.options.llmModel || \"claude-sonnet-4-5\"),\n      prompt: this.buildLLMPrompt(fixtureHint, prop, entity),\n    });\n\n    const value = this.parseLLMResponse(text, prop.type);\n    if (this.options.enableLLMCache) {\n      this.llmCache.set(cacheKey, value);\n    }\n\n    return value;\n  }\n\n  /**\n   * row 전체를 한 번에 생성하는 LLM 프롬프트를 만듭니다.\n   */\n  private buildRowLLMPrompt(props: EntityProp[], entity: Entity): string {\n    const locale = this.options.locale || \"ko\";\n    const language = locale === \"ko\" ? \"Korean\" : locale === \"ja\" ? \"Japanese\" : \"English\";\n\n    const fieldDescriptions = props\n      .map((p) => {\n        let desc = `- ${p.name} (${p.type}): ${p.cone?.note ?? \"\"}`;\n        if (\n          (p.type === \"enum\" || p.type === \"enum[]\") &&\n          \"id\" in p &&\n          p.id &&\n          entity.enumLabels?.[p.id]\n        ) {\n          const values = Object.keys(entity.enumLabels[p.id]).join(\", \");\n          desc += ` [allowed values: ${values}]`;\n        }\n        return desc;\n      })\n      .join(\"\\n\");\n\n    // LLM 대상이 아닌 prop들도 맥락으로 제공 (relation 제외)\n    const otherProps = entity.props\n      .filter(\n        (p) =>\n          !props.includes(p) &&\n          !isRelationProp(p) &&\n          p.name !== \"id\" &&\n          !(\"virtual\" in p && p.virtual),\n      )\n      .map((p) => {\n        let desc = `- ${p.name} (${p.type})`;\n        if (p.cone?.note) desc += `: ${p.cone.note}`;\n        if (\n          (p.type === \"enum\" || p.type === \"enum[]\") &&\n          \"id\" in p &&\n          p.id &&\n          entity.enumLabels?.[p.id]\n        ) {\n          const values = Object.keys(entity.enumLabels[p.id]).join(\", \");\n          desc += ` [allowed values: ${values}]`;\n        }\n        return desc;\n      })\n      .join(\"\\n\");\n    const otherPropsContext = otherProps\n      ? `\\n\\nOther fields in this entity (for context, do NOT generate these):\\n${otherProps}`\n      : \"\";\n\n    const outputShape = props.map((p) => `  \"${p.name}\": <${p.type}>`).join(\",\\n\");\n\n    const entityContext = entity.cone?.note ? `\\nEntity description: ${entity.cone.note}` : \"\";\n\n    return `Generate test fixture data for the ${entity.id} entity. All fields must be coherent and consistent with each other.\n\nEntity: ${entity.id}${entityContext}\nLocale: ${locale} (use ${language} for text fields)\n\nFields to generate:\n${fieldDescriptions}${otherPropsContext}\n\nRules:\n- All fields in a single row must be logically consistent (e.g. name/name_en/name_cn should represent the same person)\n- Return ONLY valid JSON, no markdown or explanation\n- Dates in ISO 8601 format\n- Use ${language} for text unless field description says otherwise\n\nReturn exactly this JSON shape:\n{\n${outputShape}\n}`;\n  }\n\n  /**\n   * row LLM 응답을 파싱하여 필드별 값으로 변환합니다.\n   */\n  private parseRowLLMResponse(text: string, props: EntityProp[]): Record<string, unknown> {\n    const jsonText = text\n      .trim()\n      .replace(/^```json\\s*/i, \"\")\n      .replace(/```\\s*$/, \"\")\n      .trim();\n\n    let parsed: Record<string, unknown>;\n    try {\n      parsed = JSON.parse(jsonText);\n    } catch {\n      !isTest() && console.warn(\"[FixtureGenerator] Failed to parse row LLM response:\", text);\n      return {};\n    }\n\n    const result: Record<string, unknown> = {};\n    for (const prop of props) {\n      if (prop.name in parsed) {\n        result[prop.name] = this.parseLLMResponse(String(parsed[prop.name] ?? \"\"), prop.type);\n      }\n    }\n    return result;\n  }\n\n  private buildLLMPrompt(hint: string, prop: EntityProp, entity: Entity): string {\n    const locale = this.options.locale || \"ko\";\n    const language = locale === \"ko\" ? \"Korean\" : locale === \"ja\" ? \"Japanese\" : \"English\";\n\n    const entityContext = entity.cone?.note ? `\\nEntity context: ${entity.cone.note}` : \"\";\n\n    const otherFields = entity.props\n      .filter((p) => p.name !== prop.name && !isRelationProp(p) && p.cone?.note)\n      .map((p) => `- ${p.name} (${p.type}): ${p.cone?.note}`)\n      .join(\"\\n\");\n    const otherFieldsContext = otherFields\n      ? `\\n\\nOther fields in this entity (for context):\\n${otherFields}`\n      : \"\";\n\n    let prompt = `Generate test data for ${entity.id}.${prop.name} (type: ${prop.type})${entityContext}${otherFieldsContext}\n\nRequirement: ${hint}\n\nRules:\n- Return ONLY the value, no explanation or markdown\n- Use ${language} language if applicable\n- Format: ${this.getExpectedFormat(prop.type)}`;\n\n    // enum 타입인 경우 가능한 값 목록 추가\n    if (prop.type === \"enum\" || prop.type === \"enum[]\") {\n      let enumValues: string[] = [];\n\n      if (\"enum\" in prop && Array.isArray(prop.enum) && prop.enum.length > 0) {\n        enumValues = prop.enum;\n      } else if (\"id\" in prop && prop.id && entity?.enumLabels?.[prop.id]) {\n        enumValues = Object.keys(entity.enumLabels[prop.id]);\n      }\n\n      if (enumValues.length > 0) {\n        prompt += `\\n- IMPORTANT: Choose ONLY from these allowed values: ${enumValues.join(\", \")}`;\n      }\n    }\n\n    prompt += `\\n\\nExample: ${this.getExampleForType(prop.type, locale)}`;\n\n    return prompt;\n  }\n\n  private parseLLMResponse(text: string, propType: string): unknown {\n    const cleaned = text.trim();\n\n    // 배열 타입 처리\n    if (propType.endsWith(\"[]\")) {\n      try {\n        const parsed = JSON.parse(cleaned);\n        const baseType = propType.slice(0, -2); // \"integer[]\" -> \"integer\"\n\n        if (Array.isArray(parsed)) {\n          return parsed.map((item) => {\n            // null/undefined는 타입별 기본값으로\n            if (item === null || item === undefined) {\n              return this.getDefaultValueForType(baseType);\n            }\n            // 객체는 JSON.stringify 후 파싱 (json 타입인 경우)\n            if (typeof item === \"object\") {\n              return baseType === \"json\"\n                ? item\n                : this.parseScalarValue(JSON.stringify(item), baseType);\n            }\n            // primitive 값은 문자열로 변환 후 파싱\n            return this.parseScalarValue(String(item), baseType);\n          });\n        }\n\n        // 단일 값이 온 경우 배열로 감싸기\n        if (parsed === null || parsed === undefined) {\n          return [this.getDefaultValueForType(baseType)];\n        }\n        return [this.parseScalarValue(String(parsed), baseType)];\n      } catch {\n        return [];\n      }\n    }\n\n    return this.parseScalarValue(cleaned, propType);\n  }\n\n  private getDefaultValueForType(propType: string): unknown {\n    switch (propType) {\n      case \"integer\":\n        return 0;\n      case \"bigInteger\":\n        return 0n;\n      case \"float\":\n      case \"number\":\n      case \"numeric\":\n        return 0;\n      case \"boolean\":\n        return false;\n      case \"date\":\n        return new Date();\n      case \"json\":\n        return {};\n      case \"uuid\":\n        return \"00000000-0000-0000-0000-000000000000\";\n      default:\n        return \"\";\n    }\n  }\n\n  private parseScalarValue(text: string, propType: string): unknown {\n    const cleaned = text.trim();\n\n    switch (propType) {\n      case \"integer\": {\n        const num = parseInt(cleaned, 10);\n        return Number.isNaN(num) ? 0 : num;\n      }\n      case \"bigInteger\": {\n        try {\n          return BigInt(cleaned);\n        } catch {\n          return 0n;\n        }\n      }\n      case \"float\":\n      case \"number\":\n      case \"numeric\": {\n        const num = parseFloat(cleaned);\n        return Number.isNaN(num) ? 0 : num;\n      }\n      case \"boolean\":\n        return cleaned.toLowerCase() === \"true\";\n      case \"date\": {\n        const date = new Date(cleaned);\n        return Number.isNaN(date.getTime()) ? new Date() : date;\n      }\n      case \"json\":\n        try {\n          return JSON.parse(cleaned);\n        } catch {\n          return cleaned;\n        }\n      case \"uuid\":\n      case \"enum\":\n        return cleaned;\n      default:\n        return cleaned;\n    }\n  }\n\n  /**\n   * faker 함수 인자 문자열을 파싱하여 인자 배열로 반환합니다.\n   *\n   * 3단계 전략:\n   * 1. JSON 직접 파싱 (표준 JSON 표현식)\n   * 2. JS 객체 리터럴 → JSON 변환 후 재시도 (single quote, unquoted key 처리)\n   * 3. 단순 단일 인자 폴백 (숫자, 문자열)\n   */\n  private parseGeneratorArgs(argsStr: string, propName: string): unknown[] {\n    // 1. JSON 직접 파싱\n    try {\n      const parsed = JSON.parse(`[${argsStr}]`) as unknown;\n      return Array.isArray(parsed) ? parsed : [parsed];\n    } catch {\n      // 계속\n    }\n\n    // 2. JS 객체 리터럴 → JSON 변환 후 재시도\n    try {\n      const jsonStr = this.convertJsLiteralToJson(argsStr);\n      const parsed = JSON.parse(`[${jsonStr}]`) as unknown;\n      return Array.isArray(parsed) ? parsed : [parsed];\n    } catch {\n      // 계속\n    }\n\n    // 3. 단순 단일 인자 폴백\n    const trimmed = argsStr.trim();\n    if (!Number.isNaN(Number(trimmed))) {\n      return [Number(trimmed)];\n    }\n    if (\n      (trimmed.startsWith('\"') && trimmed.endsWith('\"')) ||\n      (trimmed.startsWith(\"'\") && trimmed.endsWith(\"'\"))\n    ) {\n      return [trimmed.slice(1, -1)];\n    }\n\n    throw new Error(`FixtureGenerator: Cannot parse arguments for ${propName}: ${argsStr}`);\n  }\n\n  /**\n   * JS 객체 리터럴을 JSON으로 변환합니다.\n   *\n   * 두 가지 변환:\n   * 1. Single-quoted 문자열 → double-quoted (이스케이프 처리 포함)\n   * 2. Unquoted 객체 키 → double-quoted\n   */\n  private convertJsLiteralToJson(input: string): string {\n    // 1. 'value' → \"value\" (내부 \" 이스케이프, \\' → ')\n    const withDoubleQuotes = input.replace(\n      /'([^'\\\\]*(?:\\\\.[^'\\\\]*)*)'/g,\n      (_, content: string) => `\"${content.replace(/\"/g, '\\\\\"').replace(/\\\\'/g, \"'\")}\"`,\n    );\n\n    // 2. { key: → { \"key\":\n    return withDoubleQuotes.replace(/([{,]\\s*)([a-zA-Z_$][\\w$]*)(\\s*:)/g, '$1\"$2\"$3');\n  }\n\n  /**\n   * Sonamu.secret을 우선으로 하고, 없으면 환경변수에서 API 키를 읽습니다.\n   *\n   * Sonamu.secret은 프로젝트별 설정(sonamu.config.ts)이므로 더 높은 우선순위를 가지며,\n   * 환경변수는 개발 환경이나 CI/CD에서 fallback으로 사용됩니다.\n   */\n  private getApiKey(): string {\n    let apiKey: string | undefined;\n\n    try {\n      const { Sonamu } = require(\"../api\");\n      apiKey = Sonamu.secret?.anthropic_api_key;\n    } catch {\n      // Sonamu가 초기화되지 않은 경우 (테스트 환경 등)\n    }\n\n    if (!apiKey) {\n      apiKey = process.env.ANTHROPIC_API_KEY;\n    }\n\n    if (!apiKey) {\n      throw new Error(\n        \"ANTHROPIC_API_KEY not found. Set it in environment variables or Sonamu.secret.anthropic_api_key\",\n      );\n    }\n\n    return apiKey;\n  }\n\n  private getExpectedFormat(propType: string): string {\n    // 배열 타입 처리\n    if (propType.endsWith(\"[]\")) {\n      const baseType = propType.slice(0, -2);\n      const baseFormat = this.getScalarFormat(baseType);\n      return `JSON array of ${baseFormat} (e.g., [${this.getExampleForType(baseType, \"en\")}, ...])`;\n    }\n\n    return this.getScalarFormat(propType);\n  }\n\n  private getScalarFormat(propType: string): string {\n    switch (propType) {\n      case \"integer\":\n      case \"bigInteger\":\n        return \"integer numbers\";\n      case \"float\":\n      case \"number\":\n      case \"numeric\":\n        return \"decimal numbers\";\n      case \"boolean\":\n        return \"booleans (true or false)\";\n      case \"date\":\n        return \"ISO 8601 date strings\";\n      case \"json\":\n        return \"valid JSON object or array\";\n      case \"uuid\":\n        return \"UUID strings\";\n      case \"enum\":\n        return \"one of the allowed enum values\";\n      default:\n        return \"plain text strings\";\n    }\n  }\n\n  private getExampleForType(propType: string, locale: Locale): string {\n    // 배열 타입 처리\n    if (propType.endsWith(\"[]\")) {\n      const baseType = propType.slice(0, -2);\n      const baseExample = this.getScalarExample(baseType, locale);\n      return `[${baseExample}]`;\n    }\n\n    return this.getScalarExample(propType, locale);\n  }\n\n  private getScalarExample(propType: string, locale: Locale): string {\n    const isKorean = locale === \"ko\";\n\n    switch (propType) {\n      case \"integer\":\n      case \"bigInteger\":\n        return \"42\";\n      case \"float\":\n      case \"number\":\n      case \"numeric\":\n        return \"3.14\";\n      case \"boolean\":\n        return \"true\";\n      case \"date\":\n        return \"2024-01-01\";\n      case \"json\":\n        return '{\"key\": \"value\"}';\n      case \"uuid\":\n        return \"550e8400-e29b-41d4-a716-446655440000\";\n      case \"enum\":\n        return \"ENUM_VALUE\";\n      default:\n        return isKorean ? \"안녕하세요\" : \"Hello\";\n    }\n  }\n\n  /**\n   * 이름을 이메일 로컬 파트용 로마나이즈드 문자열로 변환합니다.\n   *\n   * 한글 이름은 초성-중성-종성 분해 후 로마나이즈 처리합니다.\n   * 영문 이름은 소문자로 변환하고 공백을 점(.)\\uc73c로 치환합니다.\n   * 예: \"김철수\" → \"cheolsu.kim\", \"John Doe\" → \"john.doe\"\n   */\n  private async romanizeName(name: string): Promise<string> {\n    // 한글 포함 여부 확인\n    if (/[\\uAC00-\\uD7AF]/.test(name)) {\n      return this.romanizeKoreanName(name);\n    }\n    // 영문: 소문자 + 점 구분\n    return name\n      .toLowerCase()\n      .replace(/\\s+/g, \".\")\n      .replace(/[^a-z0-9.]/g, \"\");\n  }\n\n  /**\n   * 한글 이름을 로마나이즈 처리합니다.\n   *\n   * 초성/중성/종성 매핑 테이블을 사용하여 한글을 로마자로 변환합니다.\n   * 첫 글자를 성으로 간주하여 \"김철수\" → \"cheolsu.kim\" 형태로 출력합니다.\n   */\n  private romanizeKoreanName(name: string): string {\n    const CHOSEONG = [\n      \"g\",\n      \"kk\",\n      \"n\",\n      \"d\",\n      \"tt\",\n      \"r\",\n      \"m\",\n      \"b\",\n      \"pp\",\n      \"s\",\n      \"ss\",\n      \"\",\n      \"j\",\n      \"jj\",\n      \"ch\",\n      \"k\",\n      \"t\",\n      \"p\",\n      \"h\",\n    ];\n    const JUNGSEONG = [\n      \"a\",\n      \"ae\",\n      \"ya\",\n      \"yae\",\n      \"eo\",\n      \"e\",\n      \"yeo\",\n      \"ye\",\n      \"o\",\n      \"wa\",\n      \"wae\",\n      \"oe\",\n      \"yo\",\n      \"u\",\n      \"wo\",\n      \"we\",\n      \"wi\",\n      \"yu\",\n      \"eu\",\n      \"ui\",\n      \"i\",\n    ];\n    const JONGSEONG = [\n      \"\",\n      \"k\",\n      \"k\",\n      \"k\",\n      \"n\",\n      \"n\",\n      \"n\",\n      \"t\",\n      \"l\",\n      \"l\",\n      \"l\",\n      \"l\",\n      \"l\",\n      \"l\",\n      \"l\",\n      \"l\",\n      \"m\",\n      \"p\",\n      \"p\",\n      \"t\",\n      \"t\",\n      \"ng\",\n      \"t\",\n      \"t\",\n      \"k\",\n      \"t\",\n      \"p\",\n      \"t\",\n    ];\n\n    const romanize = (char: string): string => {\n      const code = char.charCodeAt(0);\n      if (code < 0xac00 || code > 0xd7af) return char;\n      const offset = code - 0xac00;\n      const cho = Math.floor(offset / 588);\n      const jung = Math.floor((offset % 588) / 28);\n      const jong = offset % 28;\n      return CHOSEONG[cho] + JUNGSEONG[jung] + JONGSEONG[jong];\n    };\n\n    const chars = [...name];\n    // 첫 글자를 성(姓)으로 간주\n    const familyName = romanize(chars[0]);\n    const givenName = chars.slice(1).map(romanize).join(\"\");\n\n    if (givenName) {\n      return `${givenName}.${familyName}`;\n    }\n    return familyName;\n  }\n\n  /**\n   * LLM 캐시 통계를 반환합니다.\n   */\n  getLLMCacheStats() {\n    return {\n      size: this.llmCache.size,\n      enabled: this.options.enableLLMCache,\n    };\n  }\n\n  /**\n   * LLM 캐시를 초기화합니다.\n   */\n  clearLLMCache() {\n    this.llmCache.clear();\n  }\n\n  /**\n   * 컨텍스트 생성\n   */\n  private createContext(): GeneratorContext {\n    return {\n      fixtures: new Map(),\n      referenceCache: new Map(),\n      importedRecords: new Set(),\n    };\n  }\n\n  /**\n   * 배치 생성 및 자동 저장\n   *\n   * 1. 각 spec별로 fixture 생성 (메모리)\n   * 2. FixtureRecord로 변환\n   * 3. FixtureManager.insertFixtures()로 targetDb에 저장\n   *\n   * @returns 저장된 fixture 데이터 (실제 DB ID 포함)\n   */\n  async generateBatch(\n    specs: Array<{ entity: string; count: number; overrides?: Record<string, unknown> }>,\n  ): Promise<FixtureImportResult[]> {\n    const context = this.createContext();\n    const generatedFixtures: Array<{ entity: string; data: Record<string, unknown> }> = [];\n\n    // 1. 각 spec별로 fixture 생성\n    for (const spec of specs) {\n      for (let i = 0; i < spec.count; i++) {\n        const fixture = await this.generate(spec.entity, spec.overrides || {}, context);\n        generatedFixtures.push({\n          entity: spec.entity,\n          data: fixture,\n        });\n      }\n    }\n\n    // 2. FixtureRecord로 변환\n    const fixtureRecords: FixtureRecord[] = [];\n    for (const { entity: entityName, data } of generatedFixtures) {\n      const entity = this.entityManager.get(entityName);\n\n      // integer/bigInteger PK는 임시 ID 생성 (DB 시퀀스가 실제 ID 할당)\n      // string PK는 generate()에서 이미 생성된 id 값을 그대로 사용\n      const idProp = entity.props.find((p) => p.name === \"id\");\n      const usesSequence =\n        idProp?.type === \"integer\" ||\n        idProp?.type === \"bigInteger\" ||\n        idProp?.cone?.fixtureStrategy === \"sequence\";\n\n      const dataForRecord = usesSequence\n        ? { ...data, id: Math.floor(Math.random() * 1000000) }\n        : data;\n\n      const records = await FixtureManager.createFixtureRecord(\n        entity,\n        dataForRecord as { id: number | string; [key: string]: string | number | boolean | null },\n        { singleRecord: true },\n      );\n      fixtureRecords.push(...records);\n    }\n\n    // 3. targetDb에 삽입 (FixtureManager가 의존성 정렬 처리)\n    const results = await FixtureManager.insertFixtures(this.targetDbName, fixtureRecords);\n\n    !isTest() &&\n      console.log(\n        chalk.green(`Generated and saved ${results.length} fixtures to ${this.targetDbName}`),\n      );\n    return results;\n  }\n\n  /**\n   * 실제 DB(sourceDb)에서 데이터를 조회하여 fixture DB(targetDb)에 import합니다.\n   *\n   * 1. DataExplorer로 sourceDb에서 데이터 조회 (관련 데이터 포함)\n   * 2. FixtureRecord로 변환\n   * 3. targetDb에 삽입\n   *\n   * @param entityName - 조회할 entity 이름\n   * @param options - 조회 옵션 (strategy, limit, includeRelations 등)\n   * @returns 저장된 fixture 데이터 (실제 DB ID 포함)\n   *\n   * @example\n   * // 프로덕션 DB에서 User 10명 + 관련 Employee, Department 가져오기\n   * await generator.importFromSource(\"User\", {\n   *   strategy: \"sample\",\n   *   limit: 10,\n   *   includeRelations: true,\n   *   maxDepth: 2\n   * });\n   */\n  async importFromSource(\n    entityName: string,\n    options: ExploreWithRelationsOptions,\n  ): Promise<FixtureImportResult[]> {\n    !isTest() &&\n      console.log(\n        chalk.blue(\n          `Importing ${entityName} from source DB with options: ${JSON.stringify({ strategy: options.strategy, limit: options.limit, includeRelations: options.includeRelations, maxDepth: options.maxDepth })}`,\n        ),\n      );\n\n    // 1. DataExplorer로 sourceDb에서 데이터 조회 (관련 데이터 포함)\n    const exploreResult = await this.dataExplorer.exploreWithRelations(entityName, options);\n\n    !isTest() &&\n      console.log(\n        chalk.cyan(\n          `Found ${exploreResult.main.records.length} ${entityName} records and ${exploreResult.related.size} related entities`,\n        ),\n      );\n\n    // 2. FixtureRecord로 변환\n    const fixtureRecords: FixtureRecord[] = [];\n\n    // 메인 entity의 records를 FixtureRecord로 변환\n    const mainEntity = this.entityManager.get(entityName);\n    for (const record of exploreResult.main.records) {\n      const records = await FixtureManager.createFixtureRecord(\n        mainEntity,\n        record as { id: number | string; [key: string]: string | number | boolean | null },\n        { _db: this.sourceDb, singleRecord: true },\n      );\n      fixtureRecords.push(...records);\n    }\n\n    // 관련 entity의 records를 FixtureRecord로 변환\n    for (const [relatedEntityName, relatedRecords] of exploreResult.related.entries()) {\n      const relatedEntity = this.entityManager.get(relatedEntityName);\n      for (const record of relatedRecords) {\n        const records = await FixtureManager.createFixtureRecord(\n          relatedEntity,\n          record as { id: number | string; [key: string]: string | number | boolean | null },\n          { _db: this.sourceDb, singleRecord: true },\n        );\n        fixtureRecords.push(...records);\n      }\n\n      !isTest() &&\n        console.log(chalk.gray(`  - ${relatedEntityName}: ${relatedRecords.length} records`));\n    }\n\n    // 3. targetDb에 삽입 (FixtureManager가 의존성 정렬 처리)\n    const results = await FixtureManager.insertFixtures(this.targetDbName, fixtureRecords);\n\n    !isTest() &&\n      console.log(\n        chalk.green(\n          `Successfully imported ${results.length} records to ${this.targetDbName} (${exploreResult.main.records.length} ${entityName} + ${results.length - exploreResult.main.records.length} related)`,\n        ),\n      );\n\n    return results;\n  }\n}\n"],"names":["chalk","isBelongsToOneRelationProp","isOneToOneRelationProp","isRelationProp","isTest","DataExplorer","fakerMappings","FixtureManager","FixtureGenerator","dataExplorer","locale","mappings","llmCache","Map","entityCache","options","sourceDb","_targetDb","targetDbName","entityManager","useLLM","enableLLMCache","llmModel","generate","entityName","overrides","context","createContext","entity","get","set","tempId","Date","now","rowKey","undefined","fixture","prop","props","virtual","name","cone","fixtureStrategy","type","faker","_faker","string","alphanumeric","uuid","relationValue","generateRelationValue","hasJoinColumn","note","generateWithLLM","error","console","warn","id","Error","message","fixtureGenerator","executeGenerator","fixtureDefault","generateDefaultValue","email","nameValue","username","full_name","name_en","domain","split","romanized","romanizeName","password","bcrypt","hash","fixtures","dataSource","cacheKey","with","JSON","stringify","referenceCache","has","exploreResult","exploreWithRelations","strategy","limit","config","includeRelations","maxDepth","main","records","importExploreResult","candidates","length","selected","Math","floor","random","autoKey","autoExploreResult","autoCandidates","nullable","allFixtureRecords","entityId","related","entries","recordsToImport","log","cyan","record","recordKey","importedRecords","push","add","gray","slice","fixtureRecords","createFixtureRecord","_db","singleRecord","mainEntity","mainRecordsToImport","insertFixtures","green","size","generator","startsWith","isNameField","fakerModule","fakerKO","expr","match","path","argsStr","parts","fn","part","args","trim","parseGeneratorArgs","yellow","fakerJA","localeFaker","departments","prefixes","suffixes","dept","helpers","arrayElement","prefix","suffix","localeMappings","en","normalizedName","toLowerCase","replace","pattern","Object","field_patterns","includes","executeFakerExpression","endsWith","generateArrayValue","enumValues","Array","isArray","enum","enumLabels","keys","typeDefault","type_defaults","lorem","words","number","int","min","max","bigInt","float","datatype","boolean","date","past","_entity","_localeFaker","count","from","url","image","system","fileName","mime_type","internet","word","expression","parse","fakerName","selectedFaker","funcMatch","fixtureHint","rowCacheKey","llmProps","filter","p","generateSingleWithLLM","apiKey","getApiKey","createAnthropic","generateText","text","model","prompt","buildRowLLMPrompt","rowResult","parseRowLLMResponse","fieldName","value","buildLLMPrompt","parseLLMResponse","language","fieldDescriptions","map","desc","values","join","otherProps","otherPropsContext","outputShape","entityContext","jsonText","parsed","result","String","hint","otherFields","otherFieldsContext","getExpectedFormat","getExampleForType","propType","cleaned","baseType","item","getDefaultValueForType","parseScalarValue","num","parseInt","Number","isNaN","BigInt","parseFloat","getTime","propName","jsonStr","convertJsLiteralToJson","trimmed","input","withDoubleQuotes","_","content","Sonamu","require","secret","anthropic_api_key","process","env","ANTHROPIC_API_KEY","baseFormat","getScalarFormat","baseExample","getScalarExample","isKorean","test","romanizeKoreanName","CHOSEONG","JUNGSEONG","JONGSEONG","romanize","char","code","charCodeAt","offset","cho","jung","jong","chars","familyName","givenName","getLLMCacheStats","enabled","clearLLMCache","clear","Set","generateBatch","specs","generatedFixtures","spec","i","data","idProp","find","usesSequence","dataForRecord","results","importFromSource","blue","relatedEntityName","relatedRecords","relatedEntity"],"mappings":"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"}
1335
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/testing/fixture-generator.ts"],"sourcesContent":["import chalk from \"chalk\";\nimport type { Knex } from \"knex\";\nimport type { Entity } from \"../entity/entity\";\nimport type { EntityManager } from \"../entity/entity-manager\";\nimport type { EntityProp, FixtureImportResult, FixtureRecord } from \"../types/types\";\nimport { isBelongsToOneRelationProp, isOneToOneRelationProp, isRelationProp } from \"../types/types\";\nimport { isTest } from \"../utils/controller\";\nimport {\n  DataExplorer,\n  type ExploreWithRelationsOptions,\n  type ExploreWithRelationsResult,\n} from \"./data-explorer\";\nimport { type FakerMappings, fakerMappings } from \"./faker-mappings\";\nimport { FixtureManager } from \"./fixture-manager\";\n\nexport type Locale = \"ko\" | \"en\" | \"ja\";\n\nexport type FixtureGeneratorOptions = {\n  locale?: Locale;\n  useLLM?: boolean;\n  enableLLMCache?: boolean;\n  llmModel?: string;\n};\n\nexport type GeneratorContext = {\n  /** 생성 중인 fixture들 (메모리 상) */\n  fixtures: Map<string, Record<string, unknown>>;\n\n  /** 참조 데이터 캐시 (DataExplorer 결과) */\n  referenceCache: Map<string, Record<string, unknown>[]>;\n\n  /** 이미 import된 레코드를 추적하여 중복 import를 방지합니다 */\n  importedRecords: Set<string>; // \"User#123\"\n};\n\nexport class FixtureGenerator {\n  private dataExplorer: DataExplorer;\n  private locale: Locale;\n  private mappings: FakerMappings;\n  private llmCache: Map<string, unknown> = new Map();\n  private entityCache: Map<string, Entity> = new Map();\n  private options: FixtureGeneratorOptions;\n\n  constructor(\n    private sourceDb: Knex,\n    // FixtureManager.insertFixtures가 dbName 문자열을 받기 때문에 직접 사용하지 않습니다\n    // 미래 확장성을 위해 API 시그니처에는 포함시켰습니다\n    _targetDb: Knex,\n    private targetDbName: \"fixture\" | \"test\" | \"production_master\",\n    private entityManager: typeof EntityManager,\n    options?: FixtureGeneratorOptions,\n  ) {\n    this.dataExplorer = new DataExplorer(sourceDb, entityManager);\n    this.locale = options?.locale || \"ko\";\n    this.mappings = fakerMappings;\n    this.options = {\n      locale: options?.locale || \"ko\",\n      useLLM: options?.useLLM || false,\n      enableLLMCache: options?.enableLLMCache !== false,\n      llmModel: options?.llmModel || \"claude-sonnet-4-5\",\n    };\n  }\n\n  /**\n   * Fixture 생성 (단일)\n   * @returns 생성된 fixture 데이터 (메모리 상)\n   */\n  async generate(\n    entityName: string,\n    overrides: Record<string, unknown> = {},\n    context: GeneratorContext = this.createContext(),\n  ): Promise<Record<string, unknown>> {\n    // Entity 캐싱: 테스트에서 entity cone 수정이 반영되도록 보장\n    let entity = this.entityCache.get(entityName);\n    if (!entity) {\n      entity = this.entityManager.get(entityName);\n      this.entityCache.set(entityName, entity);\n    }\n\n    const tempId = `${entityName}#temp#${Date.now()}`; // 임시 ID\n\n    // LLM row 단위 생성을 위한 고유 키 (같은 row의 필드들이 동일한 rowKey를 공유)\n    const rowKey = this.options.useLLM ? `${entityName}#row#${Date.now()}` : undefined;\n\n    // 각 prop별 값 생성\n    const fixture: Record<string, unknown> = {};\n\n    for (const prop of entity.props) {\n      // Virtual prop은 스킵\n      if (\"virtual\" in prop && prop.virtual) {\n        continue;\n      }\n\n      // id prop 처리\n      if (prop.name === \"id\") {\n        if (\"cone\" in prop && prop.cone?.fixtureStrategy === \"sequence\") {\n          // DB sequence가 자동 할당하므로 스킵 (User 등)\n          continue;\n        }\n        if (prop.type === \"string\") {\n          // DB DEFAULT 없는 string PK: alphanumeric 32자 생성 (better-auth 스타일)\n          const { faker: _faker } = await import(\"@faker-js/faker\");\n          fixture[prop.name] = _faker.string.alphanumeric(32);\n          continue;\n        }\n        if (prop.type === \"uuid\") {\n          const { faker: _faker } = await import(\"@faker-js/faker\");\n          fixture[prop.name] = _faker.string.uuid();\n          continue;\n        }\n        // integer/bigInteger PK: generateBatch에서 tempId를 넣으므로 여기선 스킵\n        continue;\n      }\n\n      // override가 있으면 사용\n      if (prop.name in overrides) {\n        fixture[prop.name] = overrides[prop.name];\n        continue;\n      }\n\n      // cone에서 생성 전략 확인\n      const cone = prop.cone;\n\n      // 1. Relation prop 처리\n      if (isRelationProp(prop)) {\n        // BelongsToOne / OneToOne(hasJoinColumn)은 FK 컬럼명({prop.name}_id)으로도 override를 받는다\n        const fkColName = `${prop.name}_id`;\n        if (\n          fkColName in overrides &&\n          (isBelongsToOneRelationProp(prop) || (isOneToOneRelationProp(prop) && prop.hasJoinColumn))\n        ) {\n          fixture[fkColName] = overrides[fkColName];\n          continue;\n        }\n\n        const relationValue = await this.generateRelationValue(entity, prop, context);\n        // BelongsToOne, OneToOne(hasJoinColumn)의 경우 foreign key 컬럼명으로 저장\n        if (\n          isBelongsToOneRelationProp(prop) ||\n          (isOneToOneRelationProp(prop) && prop.hasJoinColumn)\n        ) {\n          fixture[`${prop.name}_id`] = relationValue;\n        } else {\n          fixture[prop.name] = relationValue;\n        }\n        continue;\n      }\n\n      // 2. cone.note + LLM 사용 (useLLM이면 fixtureGenerator보다 우선)\n      if (cone?.note && this.options.useLLM) {\n        try {\n          fixture[prop.name] = await this.generateWithLLM(cone.note, prop, entity, rowKey);\n          continue;\n        } catch (error) {\n          console.warn(\n            `[FixtureGenerator] LLM generation failed for ${entity.id}.${prop.name}, falling back to fixtureGenerator or default`,\n            error instanceof Error ? error.message : error,\n          );\n          // fallback: fixtureGenerator → fixtureDefault → 기본값으로 계속\n        }\n      }\n\n      // 3. fixtureGenerator 사용\n      if (cone?.fixtureGenerator) {\n        fixture[prop.name] = await this.executeGenerator(\n          cone.fixtureGenerator as string,\n          prop,\n          entity,\n        );\n        continue;\n      }\n\n      // 4. fixtureDefault 사용\n      if (cone?.fixtureDefault !== undefined) {\n        fixture[prop.name] = cone.fixtureDefault;\n        continue;\n      }\n\n      // 5. 타입별 기본 생성\n      fixture[prop.name] = await this.generateDefaultValue(prop, entity);\n    }\n\n    // 6. email 필드가 있고 name 필드가 있으면, email의 로컬 파트를 name 기반으로 보정\n    if (\"email\" in fixture && typeof fixture.email === \"string\" && !(\"email\" in overrides)) {\n      const nameValue = fixture.name || fixture.username || fixture.full_name || fixture.name_en;\n      if (nameValue && typeof nameValue === \"string\") {\n        const domain = fixture.email.split(\"@\")[1] || \"example.com\";\n        const romanized = await this.romanizeName(nameValue);\n        fixture.email = `${romanized}@${domain}`;\n      }\n    }\n\n    // 7. password 필드 암호화\n    if (\"password\" in fixture && fixture.password && typeof fixture.password === \"string\") {\n      const bcrypt = await import(\"bcrypt\");\n      fixture.password = await bcrypt.hash(fixture.password, 10);\n    }\n\n    context.fixtures.set(tempId, fixture);\n    return fixture;\n  }\n\n  /**\n   * Relation 값 생성 + 자동 Import\n   */\n  private async generateRelationValue(\n    entity: Entity,\n    prop: EntityProp,\n    context: GeneratorContext,\n  ): Promise<number | null> {\n    if (!isRelationProp(prop)) {\n      throw new Error(`FixtureGenerator: ${entity.id}.${prop.name} is not a relation prop`);\n    }\n\n    // BelongsToOne, OneToOne(hasJoinColumn)만 처리\n    if (\n      !isBelongsToOneRelationProp(prop) &&\n      !(isOneToOneRelationProp(prop) && prop.hasJoinColumn)\n    ) {\n      return null;\n    }\n\n    const cone = prop.cone;\n    const dataSource = cone?.dataSource;\n\n    // DataExplorer로 참조 데이터 조회 (sourceDb)\n    // 관계 체인을 따라가기 위해 exploreWithRelations 사용\n    if (dataSource) {\n      const cacheKey = `${prop.with}:${JSON.stringify(dataSource)}`;\n\n      if (!context.referenceCache.has(cacheKey)) {\n        const exploreResult = await this.dataExplorer.exploreWithRelations(prop.with, {\n          strategy: dataSource.strategy,\n          limit:\n            ((dataSource.config as Record<string, unknown> | undefined)?.limit as\n              | number\n              | undefined) || 10,\n          includeRelations: true,\n          maxDepth: 3,\n          ...(dataSource.config as Record<string, unknown> | undefined),\n        });\n        context.referenceCache.set(cacheKey, exploreResult.main.records);\n\n        // 조회한 데이터와 관계된 모든 엔티티를 targetDb에 import\n        await this.importExploreResult(exploreResult, context);\n      }\n\n      const candidates = context.referenceCache.get(cacheKey);\n      if (candidates && candidates.length > 0) {\n        // 랜덤하게 하나 선택\n        const selected = candidates[Math.floor(Math.random() * candidates.length)];\n        return selected.id as number;\n      }\n    }\n\n    // dataSource가 없을 때 자동으로 fixture DB에서 조회 시도\n    // 관계 체인을 따라가기 위해 exploreWithRelations 사용\n    const autoKey = `${prop.with}:auto`;\n    if (!context.referenceCache.has(autoKey)) {\n      // fixture DB(sourceDb)에서 자동 조회 (관계 포함)\n      const autoExploreResult = await this.dataExplorer.exploreWithRelations(prop.with, {\n        strategy: \"random\",\n        limit: 10,\n        includeRelations: true,\n        maxDepth: 3,\n      });\n      context.referenceCache.set(autoKey, autoExploreResult.main.records);\n\n      // 조회한 데이터와 관계된 모든 엔티티를 targetDb에 import\n      if (autoExploreResult.main.records.length > 0) {\n        await this.importExploreResult(autoExploreResult, context);\n      }\n    }\n\n    const autoCandidates = context.referenceCache.get(autoKey);\n    if (autoCandidates && autoCandidates.length > 0) {\n      // 랜덤하게 하나 선택\n      const selected = autoCandidates[Math.floor(Math.random() * autoCandidates.length)];\n      return selected.id as number;\n    }\n\n    // 참조 데이터가 없으면 null 반환 (nullable인 경우)\n    if (prop.nullable) {\n      return null;\n    }\n\n    // nullable이 아니고 데이터도 없으면 에러\n    throw new Error(\n      `FixtureGenerator: ${entity.id}.${prop.name}에 필요한 ${prop.with} 데이터가 없습니다. ` +\n        `먼저 ${prop.with}를 생성하거나 cone.dataSource를 설정하세요.`,\n    );\n  }\n\n  /**\n   * ExploreWithRelations 결과를 targetDb에 import\n   *\n   * 관계 체인을 따라간 결과(main + related)를 모두 import합니다.\n   * 의존성 순서는 FixtureManager.insertFixtures가 자동으로 처리합니다.\n   */\n  private async importExploreResult(\n    exploreResult: ExploreWithRelationsResult,\n    context: GeneratorContext,\n  ): Promise<void> {\n    const allFixtureRecords: FixtureRecord[] = [];\n\n    // 1. Related entities import (Company, Department 등)\n    for (const [entityId, records] of exploreResult.related.entries()) {\n      const entity = this.entityManager.get(entityId);\n      const recordsToImport: Record<string, unknown>[] = [];\n\n      !isTest() &&\n        console.log(\n          chalk.cyan(`Importing related entity: ${entityId} (${records.length} records)`),\n        );\n\n      for (const record of records) {\n        const recordKey = `${entityId}#${record.id}`;\n        if (!context.importedRecords.has(recordKey)) {\n          recordsToImport.push(record);\n          context.importedRecords.add(recordKey);\n        }\n      }\n\n      if (recordsToImport.length > 0) {\n        for (const record of recordsToImport) {\n          !isTest() &&\n            console.log(\n              chalk.gray(\n                `  - Processing ${entityId} record:`,\n                JSON.stringify(record).slice(0, 100),\n              ),\n            );\n          const fixtureRecords = await FixtureManager.createFixtureRecord(\n            entity,\n            record as { id: number | string; [key: string]: string | number | boolean | null },\n            { _db: this.sourceDb, singleRecord: true },\n          );\n          allFixtureRecords.push(...fixtureRecords);\n        }\n      }\n    }\n\n    // 2. Main entity import (Employee 등)\n    const mainEntity = this.entityManager.get(exploreResult.main.entityId);\n    const mainRecordsToImport: Record<string, unknown>[] = [];\n\n    !isTest() &&\n      console.log(\n        chalk.cyan(\n          `Importing main entity: ${exploreResult.main.entityId} (${exploreResult.main.records.length} records)`,\n        ),\n      );\n\n    for (const record of exploreResult.main.records) {\n      const recordKey = `${exploreResult.main.entityId}#${record.id}`;\n      if (!context.importedRecords.has(recordKey)) {\n        mainRecordsToImport.push(record);\n        context.importedRecords.add(recordKey);\n      }\n    }\n\n    if (mainRecordsToImport.length > 0) {\n      for (const record of mainRecordsToImport) {\n        !isTest() &&\n          console.log(\n            chalk.gray(\n              `  - Processing ${exploreResult.main.entityId} record:`,\n              JSON.stringify(record).slice(0, 100),\n            ),\n          );\n        const fixtureRecords = await FixtureManager.createFixtureRecord(\n          mainEntity,\n          record as { id: number | string; [key: string]: string | number | boolean | null },\n          { _db: this.sourceDb, singleRecord: true },\n        );\n        allFixtureRecords.push(...fixtureRecords);\n      }\n    }\n\n    // 3. 모든 fixture를 한 번에 삽입 (의존성 순서 자동 처리)\n    if (allFixtureRecords.length > 0) {\n      await FixtureManager.insertFixtures(this.targetDbName, allFixtureRecords);\n\n      !isTest() &&\n        console.log(\n          chalk.green(\n            `Auto-imported ${exploreResult.main.entityId} with relations: ` +\n              `${exploreResult.main.records.length} main + ${exploreResult.related.size} related entities`,\n          ),\n        );\n    }\n  }\n\n  /**\n   * fixtureGenerator 실행 (Faker.js만 지원)\n   *\n   * faker.* 형식의 표현식을 안전하게 파싱하여 실행합니다.\n   * 예: \"faker.internet.email()\" → faker.internet.email()\n   * 예: \"faker.lorem.words(3)\" → faker.lorem.words(3)\n   */\n  private async executeGenerator(\n    generator: string,\n    prop: EntityProp,\n    entity: Entity,\n  ): Promise<unknown> {\n    // Faker.js 표현식만 지원\n    if (generator.startsWith(\"faker.\")) {\n      // username이나 name 필드는 한국어 faker 사용\n      const isNameField = prop.name === \"username\" || prop.name === \"name\";\n      const fakerModule = await import(\"@faker-js/faker\");\n      const faker = isNameField ? fakerModule.fakerKO : fakerModule.faker;\n      const expr = generator.slice(6); // \"faker.\" 제거\n\n      try {\n        // 함수 경로와 인자 파싱\n        const match = expr.match(/^([\\w.]+)(?:\\((.*?)\\))?$/);\n        if (!match) {\n          throw new Error(\n            `FixtureGenerator: Invalid faker expression for ${prop.name}: ${generator}`,\n          );\n        }\n\n        const [, path, argsStr] = match;\n        const parts = path.split(\".\");\n\n        // faker 객체에서 함수 찾기\n        let fn: unknown = faker;\n        for (const part of parts) {\n          if (typeof fn === \"object\" && fn !== null && part in fn) {\n            fn = (fn as Record<string, unknown>)[part];\n          } else {\n            throw new Error(`FixtureGenerator: Invalid faker path for ${prop.name}: faker.${path}`);\n          }\n        }\n\n        // 함수가 아니면 에러\n        if (typeof fn !== \"function\") {\n          throw new Error(`FixtureGenerator: faker.${path} is not a function (for ${prop.name})`);\n        }\n\n        let args: unknown[] = [];\n        if (argsStr?.trim()) {\n          args = this.parseGeneratorArgs(argsStr, prop.name);\n        }\n\n        return fn(...args);\n      } catch (error) {\n        !isTest() &&\n          console.log(\n            chalk.yellow(\n              `Failed to execute generator \"${generator}\" for ${prop.name}, falling back to default:`,\n            ),\n            error,\n          );\n        return this.generateDefaultValue(prop, entity);\n      }\n    }\n\n    // faker 이외의 표현식은 지원하지 않음\n    !isTest() &&\n      console.log(\n        chalk.yellow(\n          `Unsupported generator expression for ${prop.name}: ${generator}. Only faker.* expressions are supported. Using default value.`,\n        ),\n      );\n    return this.generateDefaultValue(prop, entity);\n  }\n\n  /**\n   * 필드의 타입과 이름을 분석하여 적절한 기본값을 생성합니다.\n   *\n   * 우선순위:\n   * 1. 필드명 패턴 매칭 (salary, budget 등 의미있는 데이터)\n   * 2. 특수 케이스 (Department name 등 도메인 지식)\n   * 3. 배열 타입 (JSON 배열)\n   * 4. Enum 타입\n   * 5. 타입별 기본값\n   */\n  private async generateDefaultValue(prop: EntityProp, entity?: Entity): Promise<unknown> {\n    const fakerModule = await import(\"@faker-js/faker\");\n    const faker = fakerModule.faker;\n    const fakerKO = fakerModule.fakerKO;\n    const fakerJA = fakerModule.fakerJA;\n\n    const localeFaker = this.locale === \"ko\" ? fakerKO : this.locale === \"ja\" ? fakerJA : faker;\n\n    /**\n     * 1. Entity-specific 특수 케이스를 먼저 처리합니다.\n     * field_patterns보다 우선하여, 특정 엔티티의 필드에 도메인에 맞는 값을 생성합니다.\n     * 예: Department.name → 한국어 부서명 (사람 이름이 아님)\n     */\n\n    /**\n     * Department name은 한국어 부서명 목록에서 선택합니다.\n     * 고유성을 위해 70% 확률로 prefix/suffix를 추가합니다.\n     */\n    if (entity?.id === \"Department\" && prop.name === \"name\") {\n      const departments = [\n        \"개발팀\",\n        \"기획팀\",\n        \"마케팅팀\",\n        \"영업팀\",\n        \"인사팀\",\n        \"총무팀\",\n        \"재무팀\",\n        \"회계팀\",\n        \"법무팀\",\n        \"디자인팀\",\n        \"IT팀\",\n        \"고객지원팀\",\n        \"품질관리팀\",\n        \"연구개발팀\",\n        \"생산팀\",\n        \"구매팀\",\n        \"물류팀\",\n      ];\n      const prefixes = [\"신규\", \"통합\", \"전략\", \"글로벌\", \"디지털\", \"핵심\"];\n      const suffixes = [\"1팀\", \"2팀\", \"3팀\", \"A팀\", \"B팀\", \"본부\", \"센터\", \"그룹\"];\n\n      const dept = faker.helpers.arrayElement(departments);\n\n      const random = Math.random();\n      if (random > 0.7) {\n        const prefix = faker.helpers.arrayElement(prefixes);\n        return `${prefix} ${dept}`;\n      }\n      if (random > 0.4) {\n        const suffix = faker.helpers.arrayElement(suffixes);\n        return `${dept} ${suffix}`;\n      }\n      return dept;\n    }\n\n    /**\n     * 2. 필드명에서 의미를 추론하여 현실적인 데이터를 생성합니다.\n     * 예: salary → 30M~150M (한국 연봉 범위)\n     *     budget → 10M~500M (프로젝트 예산 범위)\n     */\n    const localeMappings = this.mappings[this.locale] || this.mappings.en;\n    const normalizedName = prop.name.toLowerCase().replace(/_/g, \"\");\n\n    for (const [pattern, config] of Object.entries(localeMappings.field_patterns)) {\n      if (normalizedName.includes(pattern.toLowerCase())) {\n        try {\n          return await this.executeFakerExpression(config.faker, prop);\n        } catch (error) {\n          !isTest() &&\n            console.log(\n              chalk.yellow(\n                `Failed to execute field pattern \"${pattern}\" for ${prop.name}, falling back:`,\n              ),\n              error,\n            );\n          break;\n        }\n      }\n    }\n\n    /**\n     * 3. JSON 타입이면서 배열인 경우 (SonamuFile[], string[] 등)\n     * 필드명 패턴을 보고 적절한 배열 데이터를 생성합니다.\n     */\n    if (prop.type === \"json\" && \"id\" in prop && prop.id) {\n      if (prop.id.endsWith(\"[]\")) {\n        return this.generateArrayValue(prop, entity, faker, localeFaker);\n      }\n    }\n\n    /** 4. Enum 타입은 정의된 값 중 하나를 랜덤 선택합니다 */\n    if (prop.type === \"enum\") {\n      let enumValues: string[] = [];\n\n      if (\"enum\" in prop && Array.isArray(prop.enum) && prop.enum.length > 0) {\n        enumValues = prop.enum;\n      } else if (\"id\" in prop && prop.id && entity?.enumLabels?.[prop.id]) {\n        enumValues = Object.keys(entity.enumLabels[prop.id]);\n      }\n\n      if (enumValues.length > 0) {\n        return faker.helpers.arrayElement(enumValues);\n      }\n      return prop.nullable ? null : \"UNKNOWN\";\n    }\n\n    if (prop.type === \"enum[]\") {\n      let enumValues: string[] = [];\n\n      if (\"enum\" in prop && Array.isArray(prop.enum) && prop.enum.length > 0) {\n        enumValues = prop.enum;\n      } else if (\"id\" in prop && prop.id && entity?.enumLabels?.[prop.id]) {\n        enumValues = Object.keys(entity.enumLabels[prop.id]);\n      }\n\n      if (enumValues.length > 0) {\n        return [faker.helpers.arrayElement(enumValues)];\n      }\n      return [];\n    }\n\n    /**\n     * 5. Vector 타입은 현재 지원하지 않으므로 null을 반환합니다.\n     * 향후 AI embedding 생성 기능 추가 시 구현 예정입니다.\n     */\n    if (prop.type === \"vector\" || prop.type === \"vector[]\" || prop.type === \"tsvector\") {\n      return null;\n    }\n\n    /** 6. 타입별 기본 Faker 표현식을 실행합니다 */\n    const typeDefault = localeMappings.type_defaults[prop.type];\n    if (typeDefault) {\n      try {\n        return await this.executeFakerExpression(typeDefault.faker, prop);\n      } catch (error) {\n        !isTest() &&\n          console.log(\n            chalk.yellow(`Failed to execute type default for ${prop.type}, using fallback:`, error),\n          );\n      }\n    }\n\n    /** 7. 매핑되지 않은 타입은 기본 Faker 함수로 처리합니다 */\n    switch (prop.type) {\n      case \"string\":\n      case \"string[]\":\n        return faker.lorem.words(3);\n      case \"integer\":\n        return faker.number.int({ min: 1, max: 1000 });\n      case \"integer[]\":\n        return [faker.number.int({ min: 1, max: 1000 })];\n      case \"bigInteger\":\n        return faker.number.bigInt({ min: 1n, max: 1000n });\n      case \"bigInteger[]\":\n        return [faker.number.bigInt({ min: 1n, max: 1000n })];\n      case \"number\":\n      case \"numeric\":\n        return faker.number.float({ min: 0, max: 1000 });\n      case \"number[]\":\n      case \"numeric[]\":\n        return [faker.number.float({ min: 0, max: 1000 })];\n      case \"boolean\":\n        return faker.datatype.boolean();\n      case \"boolean[]\":\n        return [faker.datatype.boolean()];\n      case \"date\":\n      case \"date[]\":\n        return faker.date.past();\n      case \"json\":\n        return {};\n      case \"uuid\":\n      case \"uuid[]\":\n        return faker.string.uuid();\n      default:\n        return null;\n    }\n  }\n\n  /**\n   * 배열 타입의 값을 생성합니다.\n   *\n   * 타입 ID와 필드명 패턴을 분석하여 적절한 배열 데이터를 생성합니다.\n   * 예: image_urls → [{url, name, mime_type}, ...]\n   *     tag_ids → [1, 23, 45]\n   */\n  private generateArrayValue(\n    prop: EntityProp,\n    _entity: Entity | undefined,\n    faker: typeof import(\"@faker-js/faker\").faker,\n    _localeFaker: typeof import(\"@faker-js/faker\").faker,\n  ): unknown[] {\n    const count = faker.number.int({ min: 1, max: 3 });\n\n    /** SonamuFile[]은 Sonamu 내장 타입으로 구조가 정해져 있습니다 */\n    if (\"id\" in prop && prop.id === \"SonamuFile[]\") {\n      return Array.from({ length: count }, () => ({\n        url: faker.image.url(),\n        name: faker.system.fileName(),\n        mime_type: faker.helpers.arrayElement([\n          \"image/jpeg\",\n          \"image/png\",\n          \"image/gif\",\n          \"application/pdf\",\n        ]),\n      }));\n    }\n\n    /** 필드명에서 배열의 용도를 추론합니다 */\n    const normalizedName = prop.name.toLowerCase().replace(/_/g, \"\");\n\n    if (normalizedName.includes(\"url\") || normalizedName.includes(\"image\")) {\n      return Array.from({ length: count }, () => faker.internet.url());\n    }\n\n    if (normalizedName.includes(\"id\") && normalizedName.endsWith(\"s\")) {\n      return Array.from({ length: count }, () => faker.number.int({ min: 1, max: 100 }));\n    }\n\n    if (normalizedName.includes(\"tag\") || normalizedName.includes(\"name\")) {\n      return Array.from({ length: count }, () => faker.lorem.word());\n    }\n\n    /** 패턴 매칭되지 않으면 빈 배열을 반환합니다 */\n    return [];\n  }\n\n  /**\n   * JSON 매핑의 Faker 표현식을 파싱하여 실행합니다.\n   *\n   * 표현식 예시:\n   * - \"faker.internet.email()\" → 인자 없음\n   * - \"faker.number.int({ min: 1, max: 100 })\" → JSON 인자\n   * - \"{}\" → 리터럴 값 (JSON.parse)\n   *\n   * fakerKO, fakerJA도 지원하여 다국어 데이터를 생성합니다.\n   */\n  private async executeFakerExpression(expression: string, prop: EntityProp): Promise<unknown> {\n    const fakerModule = await import(\"@faker-js/faker\");\n    const faker = fakerModule.faker;\n    const fakerKO = fakerModule.fakerKO;\n    const fakerJA = fakerModule.fakerJA;\n\n    /** Faker 표현식이 아닌 리터럴 값은 JSON으로 파싱합니다 */\n    if (!expression.startsWith(\"faker\")) {\n      try {\n        return JSON.parse(expression);\n      } catch {\n        return expression;\n      }\n    }\n\n    /** 표현식에서 Faker 객체와 경로를 추출합니다 */\n    const match = expression.match(/^(faker|fakerKO|fakerJA)\\.(.*?)$/);\n    if (!match) {\n      throw new Error(`Invalid faker expression: ${expression}`);\n    }\n\n    const [, fakerName, expr] = match;\n    const selectedFaker =\n      fakerName === \"fakerKO\" ? fakerKO : fakerName === \"fakerJA\" ? fakerJA : faker;\n\n    const funcMatch = expr.match(/^([\\w.]+)(?:\\((.*?)\\))?$/);\n    if (!funcMatch) {\n      throw new Error(`Invalid faker expression for ${prop.name}: ${expression}`);\n    }\n\n    const [, path, argsStr] = funcMatch;\n    const parts = path.split(\".\");\n\n    /** 점 표기법(dot notation)으로 Faker 함수를 찾아갑니다 */\n    let fn: unknown = selectedFaker;\n    for (const part of parts) {\n      if (typeof fn === \"object\" && fn !== null && part in fn) {\n        fn = (fn as Record<string, unknown>)[part];\n      } else {\n        throw new Error(`Invalid faker path for ${prop.name}: ${fakerName}.${path}`);\n      }\n    }\n\n    if (typeof fn !== \"function\") {\n      throw new Error(`${fakerName}.${path} is not a function (for ${prop.name})`);\n    }\n\n    let args: unknown[] = [];\n    if (argsStr?.trim()) {\n      args = this.parseGeneratorArgs(argsStr, prop.name);\n    }\n\n    return fn(...args);\n  }\n\n  /**\n   * fixtureHint를 LLM에게 전달하여 현실적인 테스트 데이터를 생성합니다.\n   *\n   * faker.js로는 생성하기 어려운 복잡한 텍스트(자기소개, 설명문 등)를\n   * LLM을 활용하여 생성합니다. 동일한 hint에 대한 중복 호출을 방지하기 위해\n   * 캐싱을 기본으로 지원합니다 (LLM API 비용 절감).\n   *\n   * ai 패키지는 dynamic import로 불러오므로, useLLM이 false인 경우\n   * 의존성이 설치되지 않아도 fixture 생성이 정상 동작합니다.\n   */\n  private async generateWithLLM(\n    fixtureHint: string,\n    prop: EntityProp,\n    entity: Entity,\n    rowKey?: string,\n  ): Promise<unknown> {\n    // rowKey가 있으면 row 단위 생성 전략 사용\n    if (rowKey) {\n      const rowCacheKey = `${rowKey}:${prop.name}`;\n\n      // 이미 이 row에 대한 LLM 호출이 완료된 경우 캐시에서 바로 반환\n      if (this.llmCache.has(rowCacheKey)) {\n        return this.llmCache.get(rowCacheKey);\n      }\n\n      // 새 row: LLM 대상 prop 전체를 한 번에 생성\n      const llmProps = entity.props.filter((p) => {\n        if (isRelationProp(p)) return false;\n        if (p.cone?.fixtureGenerator) return false;\n        if (p.name === \"id\" && p.cone?.fixtureStrategy === \"sequence\") return false;\n        return !!p.cone?.note;\n      });\n\n      // llmProps가 비어있으면 단일 필드 방식으로 fallback\n      if (llmProps.length === 0) {\n        !isTest() &&\n          console.log(\n            `[FixtureGenerator] llmProps is empty for ${entity.id}.${prop.name}, using single field fallback`,\n          );\n        return this.generateSingleWithLLM(fixtureHint, prop, entity);\n      }\n\n      const apiKey = this.getApiKey();\n      const { createAnthropic } = await import(\"@ai-sdk/anthropic\");\n      const { generateText } = await import(\"ai\");\n\n      const rowResponse = await generateText({\n        model: createAnthropic({ apiKey })(this.options.llmModel || \"claude-sonnet-4-5\"),\n        prompt: this.buildRowLLMPrompt(llmProps, entity),\n      });\n      if (!rowResponse || typeof rowResponse.text !== \"string\") {\n        throw new Error(\"Invalid LLM response\");\n      }\n\n      // 응답을 파싱하여 각 필드에 대한 결과를 캐시에 저장\n      const rowResult = this.parseRowLLMResponse(rowResponse.text, llmProps);\n      for (const [fieldName, value] of Object.entries(rowResult)) {\n        this.llmCache.set(`${rowKey}:${fieldName}`, value);\n      }\n\n      // 요청한 필드의 값 반환 (없으면 단일 필드 fallback)\n      if (this.llmCache.has(rowCacheKey)) {\n        return this.llmCache.get(rowCacheKey);\n      }\n\n      // 만약 row 응답에 이 필드가 누락된 경우 단일 필드 fallback\n      return this.generateSingleWithLLM(fixtureHint, prop, entity);\n    }\n\n    // rowKey가 없으면 기존 단일 필드 방식\n    return this.generateSingleWithLLM(fixtureHint, prop, entity);\n  }\n\n  /**\n   * 단일 필드를 LLM으로 생성합니다 (rowKey 없을 때 fallback용)\n   */\n  private async generateSingleWithLLM(\n    fixtureHint: string,\n    prop: EntityProp,\n    entity: Entity,\n  ): Promise<unknown> {\n    const cacheKey = `${entity.id}:${prop.name}:${fixtureHint}`;\n    if (this.options.enableLLMCache && this.llmCache.has(cacheKey)) {\n      return this.llmCache.get(cacheKey);\n    }\n\n    const apiKey = this.getApiKey();\n    const { createAnthropic } = await import(\"@ai-sdk/anthropic\");\n    const { generateText } = await import(\"ai\");\n\n    const singleResponse = await generateText({\n      model: createAnthropic({ apiKey })(this.options.llmModel || \"claude-sonnet-4-5\"),\n      prompt: this.buildLLMPrompt(fixtureHint, prop, entity),\n    });\n    if (!singleResponse || typeof singleResponse.text !== \"string\") {\n      throw new Error(\"Invalid LLM response\");\n    }\n\n    const value = this.parseLLMResponse(singleResponse.text, prop.type);\n    if (this.options.enableLLMCache) {\n      this.llmCache.set(cacheKey, value);\n    }\n\n    return value;\n  }\n\n  /**\n   * row 전체를 한 번에 생성하는 LLM 프롬프트를 만듭니다.\n   */\n  private buildRowLLMPrompt(props: EntityProp[], entity: Entity): string {\n    const locale = this.options.locale || \"ko\";\n    const language = locale === \"ko\" ? \"Korean\" : locale === \"ja\" ? \"Japanese\" : \"English\";\n\n    const fieldDescriptions = props\n      .map((p) => {\n        let desc = `- ${p.name} (${p.type}): ${p.cone?.note ?? \"\"}`;\n        if (\n          (p.type === \"enum\" || p.type === \"enum[]\") &&\n          \"id\" in p &&\n          p.id &&\n          entity.enumLabels?.[p.id]\n        ) {\n          const values = Object.keys(entity.enumLabels[p.id]).join(\", \");\n          desc += ` [allowed values: ${values}]`;\n        }\n        return desc;\n      })\n      .join(\"\\n\");\n\n    // LLM 대상이 아닌 prop들도 맥락으로 제공 (relation 제외)\n    const otherProps = entity.props\n      .filter(\n        (p) =>\n          !props.includes(p) &&\n          !isRelationProp(p) &&\n          p.name !== \"id\" &&\n          !(\"virtual\" in p && p.virtual),\n      )\n      .map((p) => {\n        let desc = `- ${p.name} (${p.type})`;\n        if (p.cone?.note) desc += `: ${p.cone.note}`;\n        if (\n          (p.type === \"enum\" || p.type === \"enum[]\") &&\n          \"id\" in p &&\n          p.id &&\n          entity.enumLabels?.[p.id]\n        ) {\n          const values = Object.keys(entity.enumLabels[p.id]).join(\", \");\n          desc += ` [allowed values: ${values}]`;\n        }\n        return desc;\n      })\n      .join(\"\\n\");\n    const otherPropsContext = otherProps\n      ? `\\n\\nOther fields in this entity (for context, do NOT generate these):\\n${otherProps}`\n      : \"\";\n\n    const outputShape = props.map((p) => `  \"${p.name}\": <${p.type}>`).join(\",\\n\");\n\n    const entityContext = entity.cone?.note ? `\\nEntity description: ${entity.cone.note}` : \"\";\n\n    return `Generate test fixture data for the ${entity.id} entity. All fields must be coherent and consistent with each other.\n\nEntity: ${entity.id}${entityContext}\nLocale: ${locale} (use ${language} for text fields)\n\nFields to generate:\n${fieldDescriptions}${otherPropsContext}\n\nRules:\n- All fields in a single row must be logically consistent (e.g. name/name_en/name_cn should represent the same person)\n- Return ONLY valid JSON, no markdown or explanation\n- Dates in ISO 8601 format\n- Use ${language} for text unless field description says otherwise\n\nReturn exactly this JSON shape:\n{\n${outputShape}\n}`;\n  }\n\n  /**\n   * row LLM 응답을 파싱하여 필드별 값으로 변환합니다.\n   */\n  private parseRowLLMResponse(text: string, props: EntityProp[]): Record<string, unknown> {\n    const jsonText = text\n      .trim()\n      .replace(/^```json\\s*/i, \"\")\n      .replace(/```\\s*$/, \"\")\n      .trim();\n\n    let parsed: Record<string, unknown>;\n    try {\n      const raw = JSON.parse(jsonText);\n      if (typeof raw !== \"object\" || raw === null || Array.isArray(raw)) {\n        !isTest() &&\n          console.warn(\"[FixtureGenerator] Row LLM response is not a plain object:\", text);\n        return {};\n      }\n      parsed = raw as Record<string, unknown>;\n    } catch {\n      !isTest() && console.warn(\"[FixtureGenerator] Failed to parse row LLM response:\", text);\n      return {};\n    }\n\n    const result: Record<string, unknown> = {};\n    for (const prop of props) {\n      if (prop.name in parsed) {\n        result[prop.name] = this.parseLLMResponse(String(parsed[prop.name] ?? \"\"), prop.type);\n      }\n    }\n    return result;\n  }\n\n  private buildLLMPrompt(hint: string, prop: EntityProp, entity: Entity): string {\n    const locale = this.options.locale || \"ko\";\n    const language = locale === \"ko\" ? \"Korean\" : locale === \"ja\" ? \"Japanese\" : \"English\";\n\n    const entityContext = entity.cone?.note ? `\\nEntity context: ${entity.cone.note}` : \"\";\n\n    const otherFields = entity.props\n      .filter((p) => p.name !== prop.name && !isRelationProp(p) && p.cone?.note)\n      .map((p) => `- ${p.name} (${p.type}): ${p.cone?.note}`)\n      .join(\"\\n\");\n    const otherFieldsContext = otherFields\n      ? `\\n\\nOther fields in this entity (for context):\\n${otherFields}`\n      : \"\";\n\n    let prompt = `Generate test data for ${entity.id}.${prop.name} (type: ${prop.type})${entityContext}${otherFieldsContext}\n\nRequirement: ${hint}\n\nRules:\n- Return ONLY the value, no explanation or markdown\n- Use ${language} language if applicable\n- Format: ${this.getExpectedFormat(prop.type)}`;\n\n    // enum 타입인 경우 가능한 값 목록 추가\n    if (prop.type === \"enum\" || prop.type === \"enum[]\") {\n      let enumValues: string[] = [];\n\n      if (\"enum\" in prop && Array.isArray(prop.enum) && prop.enum.length > 0) {\n        enumValues = prop.enum;\n      } else if (\"id\" in prop && prop.id && entity?.enumLabels?.[prop.id]) {\n        enumValues = Object.keys(entity.enumLabels[prop.id]);\n      }\n\n      if (enumValues.length > 0) {\n        prompt += `\\n- IMPORTANT: Choose ONLY from these allowed values: ${enumValues.join(\", \")}`;\n      }\n    }\n\n    prompt += `\\n\\nExample: ${this.getExampleForType(prop.type, locale)}`;\n\n    return prompt;\n  }\n\n  private parseLLMResponse(text: string, propType: string): unknown {\n    const cleaned = text.trim();\n\n    // 배열 타입 처리\n    if (propType.endsWith(\"[]\")) {\n      try {\n        const parsed = JSON.parse(cleaned);\n        const baseType = propType.slice(0, -2); // \"integer[]\" -> \"integer\"\n\n        if (Array.isArray(parsed)) {\n          return parsed.map((item) => {\n            // null/undefined는 타입별 기본값으로\n            if (item === null || item === undefined) {\n              return this.getDefaultValueForType(baseType);\n            }\n            // 객체는 JSON.stringify 후 파싱 (json 타입인 경우)\n            if (typeof item === \"object\") {\n              return baseType === \"json\"\n                ? item\n                : this.parseScalarValue(JSON.stringify(item), baseType);\n            }\n            // primitive 값은 문자열로 변환 후 파싱\n            return this.parseScalarValue(String(item), baseType);\n          });\n        }\n\n        // 단일 값이 온 경우 배열로 감싸기\n        if (parsed === null || parsed === undefined) {\n          return [this.getDefaultValueForType(baseType)];\n        }\n        return [this.parseScalarValue(String(parsed), baseType)];\n      } catch {\n        return [];\n      }\n    }\n\n    return this.parseScalarValue(cleaned, propType);\n  }\n\n  private getDefaultValueForType(propType: string): unknown {\n    switch (propType) {\n      case \"integer\":\n        return 0;\n      case \"bigInteger\":\n        return 0n;\n      case \"float\":\n      case \"number\":\n      case \"numeric\":\n        return 0;\n      case \"boolean\":\n        return false;\n      case \"date\":\n        return new Date();\n      case \"json\":\n        return {};\n      case \"uuid\":\n        return \"00000000-0000-0000-0000-000000000000\";\n      default:\n        return \"\";\n    }\n  }\n\n  private parseScalarValue(text: string, propType: string): unknown {\n    const cleaned = text.trim();\n\n    switch (propType) {\n      case \"integer\": {\n        const num = parseInt(cleaned, 10);\n        return Number.isNaN(num) ? 0 : num;\n      }\n      case \"bigInteger\": {\n        try {\n          return BigInt(cleaned);\n        } catch {\n          return 0n;\n        }\n      }\n      case \"float\":\n      case \"number\":\n      case \"numeric\": {\n        const num = parseFloat(cleaned);\n        return Number.isNaN(num) ? 0 : num;\n      }\n      case \"boolean\":\n        return cleaned.toLowerCase() === \"true\";\n      case \"date\": {\n        const date = new Date(cleaned);\n        return Number.isNaN(date.getTime()) ? new Date() : date;\n      }\n      case \"json\":\n        try {\n          return JSON.parse(cleaned);\n        } catch {\n          return cleaned;\n        }\n      case \"uuid\":\n      case \"enum\":\n        return cleaned;\n      default:\n        return cleaned;\n    }\n  }\n\n  /**\n   * faker 함수 인자 문자열을 파싱하여 인자 배열로 반환합니다.\n   *\n   * 3단계 전략:\n   * 1. JSON 직접 파싱 (표준 JSON 표현식)\n   * 2. JS 객체 리터럴 → JSON 변환 후 재시도 (single quote, unquoted key 처리)\n   * 3. 단순 단일 인자 폴백 (숫자, 문자열)\n   */\n  private parseGeneratorArgs(argsStr: string, propName: string): unknown[] {\n    // 1. JSON 직접 파싱\n    try {\n      const parsed = JSON.parse(`[${argsStr}]`) as unknown;\n      return Array.isArray(parsed) ? parsed : [parsed];\n    } catch {\n      // 계속\n    }\n\n    // 2. JS 객체 리터럴 → JSON 변환 후 재시도\n    try {\n      const jsonStr = this.convertJsLiteralToJson(argsStr);\n      const parsed = JSON.parse(`[${jsonStr}]`) as unknown;\n      return Array.isArray(parsed) ? parsed : [parsed];\n    } catch {\n      // 계속\n    }\n\n    // 3. 단순 단일 인자 폴백\n    const trimmed = argsStr.trim();\n    if (!Number.isNaN(Number(trimmed))) {\n      return [Number(trimmed)];\n    }\n    if (\n      (trimmed.startsWith('\"') && trimmed.endsWith('\"')) ||\n      (trimmed.startsWith(\"'\") && trimmed.endsWith(\"'\"))\n    ) {\n      return [trimmed.slice(1, -1)];\n    }\n\n    throw new Error(`FixtureGenerator: Cannot parse arguments for ${propName}: ${argsStr}`);\n  }\n\n  /**\n   * JS 객체 리터럴을 JSON으로 변환합니다.\n   *\n   * 두 가지 변환:\n   * 1. Single-quoted 문자열 → double-quoted (이스케이프 처리 포함)\n   * 2. Unquoted 객체 키 → double-quoted\n   */\n  private convertJsLiteralToJson(input: string): string {\n    // 1. 'value' → \"value\" (내부 \" 이스케이프, \\' → ')\n    const withDoubleQuotes = input.replace(\n      /'([^'\\\\]*(?:\\\\.[^'\\\\]*)*)'/g,\n      (_, content: string) => `\"${content.replace(/\"/g, '\\\\\"').replace(/\\\\'/g, \"'\")}\"`,\n    );\n\n    // 2. { key: → { \"key\":\n    return withDoubleQuotes.replace(/([{,]\\s*)([a-zA-Z_$][\\w$]*)(\\s*:)/g, '$1\"$2\"$3');\n  }\n\n  /**\n   * Sonamu.secret을 우선으로 하고, 없으면 환경변수에서 API 키를 읽습니다.\n   *\n   * Sonamu.secret은 프로젝트별 설정(sonamu.config.ts)이므로 더 높은 우선순위를 가지며,\n   * 환경변수는 개발 환경이나 CI/CD에서 fallback으로 사용됩니다.\n   */\n  private getApiKey(): string {\n    let apiKey: string | undefined;\n\n    try {\n      const { Sonamu } = require(\"../api\");\n      apiKey = Sonamu.secret?.anthropic_api_key;\n    } catch {\n      // Sonamu가 초기화되지 않은 경우 (테스트 환경 등)\n    }\n\n    if (!apiKey) {\n      apiKey = process.env.ANTHROPIC_API_KEY;\n    }\n\n    if (!apiKey) {\n      throw new Error(\n        \"ANTHROPIC_API_KEY not found. Set it in environment variables or Sonamu.secret.anthropic_api_key\",\n      );\n    }\n\n    return apiKey;\n  }\n\n  private getExpectedFormat(propType: string): string {\n    // 배열 타입 처리\n    if (propType.endsWith(\"[]\")) {\n      const baseType = propType.slice(0, -2);\n      const baseFormat = this.getScalarFormat(baseType);\n      return `JSON array of ${baseFormat} (e.g., [${this.getExampleForType(baseType, \"en\")}, ...])`;\n    }\n\n    return this.getScalarFormat(propType);\n  }\n\n  private getScalarFormat(propType: string): string {\n    switch (propType) {\n      case \"integer\":\n      case \"bigInteger\":\n        return \"integer numbers\";\n      case \"float\":\n      case \"number\":\n      case \"numeric\":\n        return \"decimal numbers\";\n      case \"boolean\":\n        return \"booleans (true or false)\";\n      case \"date\":\n        return \"ISO 8601 date strings\";\n      case \"json\":\n        return \"valid JSON object or array\";\n      case \"uuid\":\n        return \"UUID strings\";\n      case \"enum\":\n        return \"one of the allowed enum values\";\n      default:\n        return \"plain text strings\";\n    }\n  }\n\n  private getExampleForType(propType: string, locale: Locale): string {\n    // 배열 타입 처리\n    if (propType.endsWith(\"[]\")) {\n      const baseType = propType.slice(0, -2);\n      const baseExample = this.getScalarExample(baseType, locale);\n      return `[${baseExample}]`;\n    }\n\n    return this.getScalarExample(propType, locale);\n  }\n\n  private getScalarExample(propType: string, locale: Locale): string {\n    const isKorean = locale === \"ko\";\n\n    switch (propType) {\n      case \"integer\":\n      case \"bigInteger\":\n        return \"42\";\n      case \"float\":\n      case \"number\":\n      case \"numeric\":\n        return \"3.14\";\n      case \"boolean\":\n        return \"true\";\n      case \"date\":\n        return \"2024-01-01\";\n      case \"json\":\n        return '{\"key\": \"value\"}';\n      case \"uuid\":\n        return \"550e8400-e29b-41d4-a716-446655440000\";\n      case \"enum\":\n        return \"ENUM_VALUE\";\n      default:\n        return isKorean ? \"안녕하세요\" : \"Hello\";\n    }\n  }\n\n  /**\n   * 이름을 이메일 로컬 파트용 로마나이즈드 문자열로 변환합니다.\n   *\n   * 한글 이름은 초성-중성-종성 분해 후 로마나이즈 처리합니다.\n   * 영문 이름은 소문자로 변환하고 공백을 점(.)\\uc73c로 치환합니다.\n   * 예: \"김철수\" → \"cheolsu.kim\", \"John Doe\" → \"john.doe\"\n   */\n  private async romanizeName(name: string): Promise<string> {\n    // 한글 포함 여부 확인\n    if (/[\\uAC00-\\uD7AF]/.test(name)) {\n      return this.romanizeKoreanName(name);\n    }\n    // 영문: 소문자 + 점 구분\n    return name\n      .toLowerCase()\n      .replace(/\\s+/g, \".\")\n      .replace(/[^a-z0-9.]/g, \"\");\n  }\n\n  /**\n   * 한글 이름을 로마나이즈 처리합니다.\n   *\n   * 초성/중성/종성 매핑 테이블을 사용하여 한글을 로마자로 변환합니다.\n   * 첫 글자를 성으로 간주하여 \"김철수\" → \"cheolsu.kim\" 형태로 출력합니다.\n   */\n  private romanizeKoreanName(name: string): string {\n    const CHOSEONG = [\n      \"g\",\n      \"kk\",\n      \"n\",\n      \"d\",\n      \"tt\",\n      \"r\",\n      \"m\",\n      \"b\",\n      \"pp\",\n      \"s\",\n      \"ss\",\n      \"\",\n      \"j\",\n      \"jj\",\n      \"ch\",\n      \"k\",\n      \"t\",\n      \"p\",\n      \"h\",\n    ];\n    const JUNGSEONG = [\n      \"a\",\n      \"ae\",\n      \"ya\",\n      \"yae\",\n      \"eo\",\n      \"e\",\n      \"yeo\",\n      \"ye\",\n      \"o\",\n      \"wa\",\n      \"wae\",\n      \"oe\",\n      \"yo\",\n      \"u\",\n      \"wo\",\n      \"we\",\n      \"wi\",\n      \"yu\",\n      \"eu\",\n      \"ui\",\n      \"i\",\n    ];\n    const JONGSEONG = [\n      \"\",\n      \"k\",\n      \"k\",\n      \"k\",\n      \"n\",\n      \"n\",\n      \"n\",\n      \"t\",\n      \"l\",\n      \"l\",\n      \"l\",\n      \"l\",\n      \"l\",\n      \"l\",\n      \"l\",\n      \"l\",\n      \"m\",\n      \"p\",\n      \"p\",\n      \"t\",\n      \"t\",\n      \"ng\",\n      \"t\",\n      \"t\",\n      \"k\",\n      \"t\",\n      \"p\",\n      \"t\",\n    ];\n\n    const romanize = (char: string): string => {\n      const code = char.charCodeAt(0);\n      if (code < 0xac00 || code > 0xd7af) return char;\n      const offset = code - 0xac00;\n      const cho = Math.floor(offset / 588);\n      const jung = Math.floor((offset % 588) / 28);\n      const jong = offset % 28;\n      return CHOSEONG[cho] + JUNGSEONG[jung] + JONGSEONG[jong];\n    };\n\n    const chars = [...name];\n    // 첫 글자를 성(姓)으로 간주\n    const familyName = romanize(chars[0]);\n    const givenName = chars.slice(1).map(romanize).join(\"\");\n\n    if (givenName) {\n      return `${givenName}.${familyName}`;\n    }\n    return familyName;\n  }\n\n  /**\n   * LLM 캐시 통계를 반환합니다.\n   */\n  getLLMCacheStats() {\n    return {\n      size: this.llmCache.size,\n      enabled: this.options.enableLLMCache,\n    };\n  }\n\n  /**\n   * LLM 캐시를 초기화합니다.\n   */\n  clearLLMCache() {\n    this.llmCache.clear();\n  }\n\n  /**\n   * 컨텍스트 생성\n   */\n  private createContext(): GeneratorContext {\n    return {\n      fixtures: new Map(),\n      referenceCache: new Map(),\n      importedRecords: new Set(),\n    };\n  }\n\n  /**\n   * 배치 생성 및 자동 저장\n   *\n   * 1. 각 spec별로 fixture 생성 (메모리)\n   * 2. FixtureRecord로 변환\n   * 3. FixtureManager.insertFixtures()로 targetDb에 저장\n   *\n   * @returns 저장된 fixture 데이터 (실제 DB ID 포함)\n   */\n  async generateBatch(\n    specs: Array<{ entity: string; count: number; overrides?: Record<string, unknown> }>,\n  ): Promise<FixtureImportResult[]> {\n    const context = this.createContext();\n    const generatedFixtures: Array<{ entity: string; data: Record<string, unknown> }> = [];\n\n    // 1. 각 spec별로 fixture 생성\n    for (const spec of specs) {\n      for (let i = 0; i < spec.count; i++) {\n        const fixture = await this.generate(spec.entity, spec.overrides || {}, context);\n        generatedFixtures.push({\n          entity: spec.entity,\n          data: fixture,\n        });\n      }\n    }\n\n    // 2. FixtureRecord로 변환\n    const fixtureRecords: FixtureRecord[] = [];\n    for (const { entity: entityName, data } of generatedFixtures) {\n      const entity = this.entityManager.get(entityName);\n\n      // integer/bigInteger PK는 임시 ID 생성 (DB 시퀀스가 실제 ID 할당)\n      // string PK는 generate()에서 이미 생성된 id 값을 그대로 사용\n      const idProp = entity.props.find((p) => p.name === \"id\");\n      const usesSequence =\n        idProp?.type === \"integer\" ||\n        idProp?.type === \"bigInteger\" ||\n        idProp?.cone?.fixtureStrategy === \"sequence\";\n\n      const dataForRecord = usesSequence\n        ? { ...data, id: Math.floor(Math.random() * 1000000) }\n        : data;\n\n      const records = await FixtureManager.createFixtureRecord(\n        entity,\n        dataForRecord as { id: number | string; [key: string]: string | number | boolean | null },\n        { singleRecord: true },\n      );\n      fixtureRecords.push(...records);\n    }\n\n    // 3. targetDb에 삽입 (FixtureManager가 의존성 정렬 처리)\n    const results = await FixtureManager.insertFixtures(this.targetDbName, fixtureRecords);\n\n    // 4. companion fixtures 생성 (fixtureCompanions가 선언된 경우)\n    const companionResults = await this.generateCompanions(specs, results);\n\n    const total = results.length + companionResults.length;\n    !isTest() &&\n      console.log(chalk.green(`Generated and saved ${total} fixtures to ${this.targetDbName}`));\n    return [...results, ...companionResults];\n  }\n\n  /**\n   * 부모 fixture 결과를 기반으로 fixtureCompanions에 선언된 companion Entity를 생성합니다.\n   *\n   * generateBatch()에서만 호출되며, companion 생성 시 재귀를 방지하기 위해\n   * generateBatch()를 다시 호출하지 않고 직접 삽입합니다.\n   */\n  private async generateCompanions(\n    specs: Array<{ entity: string; count: number; overrides?: Record<string, unknown> }>,\n    parentResults: FixtureImportResult[],\n  ): Promise<FixtureImportResult[]> {\n    const allResults: FixtureImportResult[] = [];\n    const processedEntities = new Set<string>();\n\n    for (const spec of specs) {\n      if (processedEntities.has(spec.entity)) continue;\n      processedEntities.add(spec.entity);\n\n      const entity = this.entityManager.get(spec.entity);\n      const idProp = entity.props.find((p) => p.name === \"id\");\n      const companions = idProp?.cone?.fixtureCompanions;\n      if (!companions || companions.length === 0) continue;\n\n      const entityResults = parentResults.filter((r) => r.entityId === spec.entity);\n      if (entityResults.length === 0) continue;\n\n      for (const companion of companions) {\n        // companion entity에서 부모 entity로의 BelongsToOne FK 컬럼명 파악\n        const companionEntity = this.entityManager.get(companion.entity);\n        const fkProp = companionEntity.props.find(\n          (p) => isRelationProp(p) && isBelongsToOneRelationProp(p) && p.with === spec.entity,\n        );\n        if (!fkProp) {\n          !isTest() &&\n            console.warn(\n              chalk.yellow(\n                `[Companion] No BelongsToOne relation from ${companion.entity} to ${spec.entity}. Skipping.`,\n              ),\n            );\n          continue;\n        }\n        const fkColName = `${fkProp.name}_id`;\n\n        // companion의 idProp, usesSequence, count는 companion 단위로 고정\n        const companionIdProp = companionEntity.props.find((p) => p.name === \"id\");\n        const usesSequence =\n          companionIdProp?.type === \"integer\" ||\n          companionIdProp?.type === \"bigInteger\" ||\n          companionIdProp?.cone?.fixtureStrategy === \"sequence\";\n        const companionCount = companion.count ?? 1;\n\n        // 각 parent result에 대해 companion fixture 생성\n        const context = this.createContext();\n        const companionFixtureRecords: FixtureRecord[] = [];\n\n        for (const parentResult of entityResults) {\n          const resolvedOverrides = this.resolveTemplateOverrides(\n            companion.overrides ?? {},\n            parentResult.data,\n          );\n          resolvedOverrides[fkColName] = parentResult.data.id;\n\n          for (let i = 0; i < companionCount; i++) {\n            const fixture = await this.generate(companion.entity, resolvedOverrides, context);\n\n            const dataForRecord = usesSequence\n              ? { ...fixture, id: Math.floor(Math.random() * 1000000) }\n              : fixture;\n\n            const records = await FixtureManager.createFixtureRecord(\n              companionEntity,\n              dataForRecord as {\n                id: number | string;\n                [key: string]: string | number | boolean | null;\n              },\n              { singleRecord: true },\n            );\n            companionFixtureRecords.push(...records);\n          }\n        }\n\n        const companionResults = await FixtureManager.insertFixtures(\n          this.targetDbName,\n          companionFixtureRecords,\n        );\n        allResults.push(...companionResults);\n\n        !isTest() &&\n          console.log(\n            chalk.green(\n              `[Companion] Generated ${companionResults.length} ${companion.entity} fixtures`,\n            ),\n          );\n      }\n    }\n\n    return allResults;\n  }\n\n  /**\n   * overrides 값의 \"{{fieldName}}\" 템플릿을 부모 fixture 데이터로 치환합니다.\n   *\n   * 예: { \"account_id\": \"{{email}}\" } → { \"account_id\": \"user@example.com\" }\n   */\n  private resolveTemplateOverrides(\n    overrides: Record<string, unknown>,\n    parentData: { [key: string]: string | number | boolean | Date | null },\n  ): Record<string, unknown> {\n    const resolved: Record<string, unknown> = {};\n    for (const [key, value] of Object.entries(overrides)) {\n      if (typeof value === \"string\" && value.startsWith(\"{{\") && value.endsWith(\"}}\")) {\n        const fieldName = value.slice(2, -2).trim();\n        if (!(fieldName in parentData)) {\n          throw new Error(\n            `템플릿 필드 \"${fieldName}\"이(가) 부모 fixture 데이터에 존재하지 않습니다 (override key: \"${key}\")`,\n          );\n        }\n        resolved[key] = parentData[fieldName];\n      } else {\n        resolved[key] = value;\n      }\n    }\n    return resolved;\n  }\n\n  /**\n   * 실제 DB(sourceDb)에서 데이터를 조회하여 fixture DB(targetDb)에 import합니다.\n   *\n   * 1. DataExplorer로 sourceDb에서 데이터 조회 (관련 데이터 포함)\n   * 2. FixtureRecord로 변환\n   * 3. targetDb에 삽입\n   *\n   * @param entityName - 조회할 entity 이름\n   * @param options - 조회 옵션 (strategy, limit, includeRelations 등)\n   * @returns 저장된 fixture 데이터 (실제 DB ID 포함)\n   *\n   * @example\n   * // 프로덕션 DB에서 User 10명 + 관련 Employee, Department 가져오기\n   * await generator.importFromSource(\"User\", {\n   *   strategy: \"sample\",\n   *   limit: 10,\n   *   includeRelations: true,\n   *   maxDepth: 2\n   * });\n   */\n  async importFromSource(\n    entityName: string,\n    options: ExploreWithRelationsOptions,\n  ): Promise<FixtureImportResult[]> {\n    !isTest() &&\n      console.log(\n        chalk.blue(\n          `Importing ${entityName} from source DB with options: ${JSON.stringify({ strategy: options.strategy, limit: options.limit, includeRelations: options.includeRelations, maxDepth: options.maxDepth })}`,\n        ),\n      );\n\n    // 1. DataExplorer로 sourceDb에서 데이터 조회 (관련 데이터 포함)\n    const exploreResult = await this.dataExplorer.exploreWithRelations(entityName, options);\n\n    !isTest() &&\n      console.log(\n        chalk.cyan(\n          `Found ${exploreResult.main.records.length} ${entityName} records and ${exploreResult.related.size} related entities`,\n        ),\n      );\n\n    // 2. FixtureRecord로 변환\n    const fixtureRecords: FixtureRecord[] = [];\n\n    // 메인 entity의 records를 FixtureRecord로 변환\n    const mainEntity = this.entityManager.get(entityName);\n    for (const record of exploreResult.main.records) {\n      const records = await FixtureManager.createFixtureRecord(\n        mainEntity,\n        record as { id: number | string; [key: string]: string | number | boolean | null },\n        { _db: this.sourceDb, singleRecord: true },\n      );\n      fixtureRecords.push(...records);\n    }\n\n    // 관련 entity의 records를 FixtureRecord로 변환\n    for (const [relatedEntityName, relatedRecords] of exploreResult.related.entries()) {\n      const relatedEntity = this.entityManager.get(relatedEntityName);\n      for (const record of relatedRecords) {\n        const records = await FixtureManager.createFixtureRecord(\n          relatedEntity,\n          record as { id: number | string; [key: string]: string | number | boolean | null },\n          { _db: this.sourceDb, singleRecord: true },\n        );\n        fixtureRecords.push(...records);\n      }\n\n      !isTest() &&\n        console.log(chalk.gray(`  - ${relatedEntityName}: ${relatedRecords.length} records`));\n    }\n\n    // 3. targetDb에 삽입 (FixtureManager가 의존성 정렬 처리)\n    const results = await FixtureManager.insertFixtures(this.targetDbName, fixtureRecords);\n\n    !isTest() &&\n      console.log(\n        chalk.green(\n          `Successfully imported ${results.length} records to ${this.targetDbName} (${exploreResult.main.records.length} ${entityName} + ${results.length - exploreResult.main.records.length} related)`,\n        ),\n      );\n\n    return results;\n  }\n}\n"],"names":["chalk","isBelongsToOneRelationProp","isOneToOneRelationProp","isRelationProp","isTest","DataExplorer","fakerMappings","FixtureManager","FixtureGenerator","dataExplorer","locale","mappings","llmCache","Map","entityCache","options","sourceDb","_targetDb","targetDbName","entityManager","useLLM","enableLLMCache","llmModel","generate","entityName","overrides","context","createContext","entity","get","set","tempId","Date","now","rowKey","undefined","fixture","prop","props","virtual","name","cone","fixtureStrategy","type","faker","_faker","string","alphanumeric","uuid","fkColName","hasJoinColumn","relationValue","generateRelationValue","note","generateWithLLM","error","console","warn","id","Error","message","fixtureGenerator","executeGenerator","fixtureDefault","generateDefaultValue","email","nameValue","username","full_name","name_en","domain","split","romanized","romanizeName","password","bcrypt","hash","fixtures","dataSource","cacheKey","with","JSON","stringify","referenceCache","has","exploreResult","exploreWithRelations","strategy","limit","config","includeRelations","maxDepth","main","records","importExploreResult","candidates","length","selected","Math","floor","random","autoKey","autoExploreResult","autoCandidates","nullable","allFixtureRecords","entityId","related","entries","recordsToImport","log","cyan","record","recordKey","importedRecords","push","add","gray","slice","fixtureRecords","createFixtureRecord","_db","singleRecord","mainEntity","mainRecordsToImport","insertFixtures","green","size","generator","startsWith","isNameField","fakerModule","fakerKO","expr","match","path","argsStr","parts","fn","part","args","trim","parseGeneratorArgs","yellow","fakerJA","localeFaker","departments","prefixes","suffixes","dept","helpers","arrayElement","prefix","suffix","localeMappings","en","normalizedName","toLowerCase","replace","pattern","Object","field_patterns","includes","executeFakerExpression","endsWith","generateArrayValue","enumValues","Array","isArray","enum","enumLabels","keys","typeDefault","type_defaults","lorem","words","number","int","min","max","bigInt","float","datatype","boolean","date","past","_entity","_localeFaker","count","from","url","image","system","fileName","mime_type","internet","word","expression","parse","fakerName","selectedFaker","funcMatch","fixtureHint","rowCacheKey","llmProps","filter","p","generateSingleWithLLM","apiKey","getApiKey","createAnthropic","generateText","rowResponse","model","prompt","buildRowLLMPrompt","text","rowResult","parseRowLLMResponse","fieldName","value","singleResponse","buildLLMPrompt","parseLLMResponse","language","fieldDescriptions","map","desc","values","join","otherProps","otherPropsContext","outputShape","entityContext","jsonText","parsed","raw","result","String","hint","otherFields","otherFieldsContext","getExpectedFormat","getExampleForType","propType","cleaned","baseType","item","getDefaultValueForType","parseScalarValue","num","parseInt","Number","isNaN","BigInt","parseFloat","getTime","propName","jsonStr","convertJsLiteralToJson","trimmed","input","withDoubleQuotes","_","content","Sonamu","require","secret","anthropic_api_key","process","env","ANTHROPIC_API_KEY","baseFormat","getScalarFormat","baseExample","getScalarExample","isKorean","test","romanizeKoreanName","CHOSEONG","JUNGSEONG","JONGSEONG","romanize","char","code","charCodeAt","offset","cho","jung","jong","chars","familyName","givenName","getLLMCacheStats","enabled","clearLLMCache","clear","Set","generateBatch","specs","generatedFixtures","spec","i","data","idProp","find","usesSequence","dataForRecord","results","companionResults","generateCompanions","total","parentResults","allResults","processedEntities","companions","fixtureCompanions","entityResults","r","companion","companionEntity","fkProp","companionIdProp","companionCount","companionFixtureRecords","parentResult","resolvedOverrides","resolveTemplateOverrides","parentData","resolved","key","importFromSource","blue","relatedEntityName","relatedRecords","relatedEntity"],"mappings":"AAAA,OAAOA,WAAW,QAAQ;AAK1B,SAASC,0BAA0B,EAAEC,sBAAsB,EAAEC,cAAc,QAAQ,oBAAiB;AACpG,SAASC,MAAM,QAAQ,yBAAsB;AAC7C,SACEC,YAAY,QAGP,qBAAkB;AACzB,SAA6BC,aAAa,QAAQ,sBAAmB;AACrE,SAASC,cAAc,QAAQ,uBAAoB;AAsBnD,OAAO,MAAMC;;;;IACHC,aAA2B;IAC3BC,OAAe;IACfC,SAAwB;IACxBC,WAAiC,IAAIC,MAAM;IAC3CC,cAAmC,IAAID,MAAM;IAC7CE,QAAiC;IAEzC,YACE,AAAQC,QAAc,EACtB,iEAAiE;IACjE,gCAAgC;IAChCC,SAAe,EACf,AAAQC,YAAsD,EAC9D,AAAQC,aAAmC,EAC3CJ,OAAiC,CACjC;aAPQC,WAAAA;aAIAE,eAAAA;aACAC,gBAAAA;QAGR,IAAI,CAACV,YAAY,GAAG,IAAIJ,aAAaW,UAAUG;QAC/C,IAAI,CAACT,MAAM,GAAGK,SAASL,UAAU;QACjC,IAAI,CAACC,QAAQ,GAAGL;QAChB,IAAI,CAACS,OAAO,GAAG;YACbL,QAAQK,SAASL,UAAU;YAC3BU,QAAQL,SAASK,UAAU;YAC3BC,gBAAgBN,SAASM,mBAAmB;YAC5CC,UAAUP,SAASO,YAAY;QACjC;IACF;IAEA;;;GAGC,GACD,MAAMC,SACJC,UAAkB,EAClBC,YAAqC,CAAC,CAAC,EACvCC,UAA4B,IAAI,CAACC,aAAa,EAAE,EACd;QAClC,4CAA4C;QAC5C,IAAIC,SAAS,IAAI,CAACd,WAAW,CAACe,GAAG,CAACL;QAClC,IAAI,CAACI,QAAQ;YACXA,SAAS,IAAI,CAACT,aAAa,CAACU,GAAG,CAACL;YAChC,IAAI,CAACV,WAAW,CAACgB,GAAG,CAACN,YAAYI;QACnC;QAEA,MAAMG,SAAS,GAAGP,WAAW,MAAM,EAAEQ,KAAKC,GAAG,IAAI,EAAE,QAAQ;QAE3D,uDAAuD;QACvD,MAAMC,SAAS,IAAI,CAACnB,OAAO,CAACK,MAAM,GAAG,GAAGI,WAAW,KAAK,EAAEQ,KAAKC,GAAG,IAAI,GAAGE;QAEzE,eAAe;QACf,MAAMC,UAAmC,CAAC;QAE1C,KAAK,MAAMC,QAAQT,OAAOU,KAAK,CAAE;YAC/B,mBAAmB;YACnB,IAAI,aAAaD,QAAQA,KAAKE,OAAO,EAAE;gBACrC;YACF;YAEA,aAAa;YACb,IAAIF,KAAKG,IAAI,KAAK,MAAM;gBACtB,IAAI,UAAUH,QAAQA,KAAKI,IAAI,EAAEC,oBAAoB,YAAY;oBAE/D;gBACF;gBACA,IAAIL,KAAKM,IAAI,KAAK,UAAU;oBAC1B,iEAAiE;oBACjE,MAAM,EAAEC,OAAOC,MAAM,EAAE,GAAG,MAAM,MAAM,CAAC;oBACvCT,OAAO,CAACC,KAAKG,IAAI,CAAC,GAAGK,OAAOC,MAAM,CAACC,YAAY,CAAC;oBAChD;gBACF;gBACA,IAAIV,KAAKM,IAAI,KAAK,QAAQ;oBACxB,MAAM,EAAEC,OAAOC,MAAM,EAAE,GAAG,MAAM,MAAM,CAAC;oBACvCT,OAAO,CAACC,KAAKG,IAAI,CAAC,GAAGK,OAAOC,MAAM,CAACE,IAAI;oBACvC;gBACF;gBAEA;YACF;YAEA,mBAAmB;YACnB,IAAIX,KAAKG,IAAI,IAAIf,WAAW;gBAC1BW,OAAO,CAACC,KAAKG,IAAI,CAAC,GAAGf,SAAS,CAACY,KAAKG,IAAI,CAAC;gBACzC;YACF;YAEA,kBAAkB;YAClB,MAAMC,OAAOJ,KAAKI,IAAI;YAEtB,sBAAsB;YACtB,IAAItC,eAAekC,OAAO;gBACxB,kFAAkF;gBAClF,MAAMY,YAAY,GAAGZ,KAAKG,IAAI,CAAC,GAAG,CAAC;gBACnC,IACES,aAAaxB,aACZxB,CAAAA,2BAA2BoC,SAAUnC,uBAAuBmC,SAASA,KAAKa,aAAa,GACxF;oBACAd,OAAO,CAACa,UAAU,GAAGxB,SAAS,CAACwB,UAAU;oBACzC;gBACF;gBAEA,MAAME,gBAAgB,MAAM,IAAI,CAACC,qBAAqB,CAACxB,QAAQS,MAAMX;gBACrE,iEAAiE;gBACjE,IACEzB,2BAA2BoC,SAC1BnC,uBAAuBmC,SAASA,KAAKa,aAAa,EACnD;oBACAd,OAAO,CAAC,GAAGC,KAAKG,IAAI,CAAC,GAAG,CAAC,CAAC,GAAGW;gBAC/B,OAAO;oBACLf,OAAO,CAACC,KAAKG,IAAI,CAAC,GAAGW;gBACvB;gBACA;YACF;YAEA,yDAAyD;YACzD,IAAIV,MAAMY,QAAQ,IAAI,CAACtC,OAAO,CAACK,MAAM,EAAE;gBACrC,IAAI;oBACFgB,OAAO,CAACC,KAAKG,IAAI,CAAC,GAAG,MAAM,IAAI,CAACc,eAAe,CAACb,KAAKY,IAAI,EAAEhB,MAAMT,QAAQM;oBACzE;gBACF,EAAE,OAAOqB,OAAO;oBACdC,QAAQC,IAAI,CACV,CAAC,6CAA6C,EAAE7B,OAAO8B,EAAE,CAAC,CAAC,EAAErB,KAAKG,IAAI,CAAC,6CAA6C,CAAC,EACrHe,iBAAiBI,QAAQJ,MAAMK,OAAO,GAAGL;gBAE3C,yDAAyD;gBAC3D;YACF;YAEA,yBAAyB;YACzB,IAAId,MAAMoB,kBAAkB;gBAC1BzB,OAAO,CAACC,KAAKG,IAAI,CAAC,GAAG,MAAM,IAAI,CAACsB,gBAAgB,CAC9CrB,KAAKoB,gBAAgB,EACrBxB,MACAT;gBAEF;YACF;YAEA,uBAAuB;YACvB,IAAIa,MAAMsB,mBAAmB5B,WAAW;gBACtCC,OAAO,CAACC,KAAKG,IAAI,CAAC,GAAGC,KAAKsB,cAAc;gBACxC;YACF;YAEA,eAAe;YACf3B,OAAO,CAACC,KAAKG,IAAI,CAAC,GAAG,MAAM,IAAI,CAACwB,oBAAoB,CAAC3B,MAAMT;QAC7D;QAEA,2DAA2D;QAC3D,IAAI,WAAWQ,WAAW,OAAOA,QAAQ6B,KAAK,KAAK,YAAY,CAAE,CAAA,WAAWxC,SAAQ,GAAI;YACtF,MAAMyC,YAAY9B,QAAQI,IAAI,IAAIJ,QAAQ+B,QAAQ,IAAI/B,QAAQgC,SAAS,IAAIhC,QAAQiC,OAAO;YAC1F,IAAIH,aAAa,OAAOA,cAAc,UAAU;gBAC9C,MAAMI,SAASlC,QAAQ6B,KAAK,CAACM,KAAK,CAAC,IAAI,CAAC,EAAE,IAAI;gBAC9C,MAAMC,YAAY,MAAM,IAAI,CAACC,YAAY,CAACP;gBAC1C9B,QAAQ6B,KAAK,GAAG,GAAGO,UAAU,CAAC,EAAEF,QAAQ;YAC1C;QACF;QAEA,qBAAqB;QACrB,IAAI,cAAclC,WAAWA,QAAQsC,QAAQ,IAAI,OAAOtC,QAAQsC,QAAQ,KAAK,UAAU;YACrF,MAAMC,SAAS,MAAM,MAAM,CAAC;YAC5BvC,QAAQsC,QAAQ,GAAG,MAAMC,OAAOC,IAAI,CAACxC,QAAQsC,QAAQ,EAAE;QACzD;QAEAhD,QAAQmD,QAAQ,CAAC/C,GAAG,CAACC,QAAQK;QAC7B,OAAOA;IACT;IAEA;;GAEC,GACD,MAAcgB,sBACZxB,MAAc,EACdS,IAAgB,EAChBX,OAAyB,EACD;QACxB,IAAI,CAACvB,eAAekC,OAAO;YACzB,MAAM,IAAIsB,MAAM,CAAC,kBAAkB,EAAE/B,OAAO8B,EAAE,CAAC,CAAC,EAAErB,KAAKG,IAAI,CAAC,uBAAuB,CAAC;QACtF;QAEA,4CAA4C;QAC5C,IACE,CAACvC,2BAA2BoC,SAC5B,CAAEnC,CAAAA,uBAAuBmC,SAASA,KAAKa,aAAa,AAAD,GACnD;YACA,OAAO;QACT;QAEA,MAAMT,OAAOJ,KAAKI,IAAI;QACtB,MAAMqC,aAAarC,MAAMqC;QAEzB,qCAAqC;QACrC,yCAAyC;QACzC,IAAIA,YAAY;YACd,MAAMC,WAAW,GAAG1C,KAAK2C,IAAI,CAAC,CAAC,EAAEC,KAAKC,SAAS,CAACJ,aAAa;YAE7D,IAAI,CAACpD,QAAQyD,cAAc,CAACC,GAAG,CAACL,WAAW;gBACzC,MAAMM,gBAAgB,MAAM,IAAI,CAAC5E,YAAY,CAAC6E,oBAAoB,CAACjD,KAAK2C,IAAI,EAAE;oBAC5EO,UAAUT,WAAWS,QAAQ;oBAC7BC,OACE,AAAEV,WAAWW,MAAM,EAA0CD,SAE3C;oBACpBE,kBAAkB;oBAClBC,UAAU;oBACV,GAAIb,WAAWW,MAAM;gBACvB;gBACA/D,QAAQyD,cAAc,CAACrD,GAAG,CAACiD,UAAUM,cAAcO,IAAI,CAACC,OAAO;gBAE/D,wCAAwC;gBACxC,MAAM,IAAI,CAACC,mBAAmB,CAACT,eAAe3D;YAChD;YAEA,MAAMqE,aAAarE,QAAQyD,cAAc,CAACtD,GAAG,CAACkD;YAC9C,IAAIgB,cAAcA,WAAWC,MAAM,GAAG,GAAG;gBACvC,aAAa;gBACb,MAAMC,WAAWF,UAAU,CAACG,KAAKC,KAAK,CAACD,KAAKE,MAAM,KAAKL,WAAWC,MAAM,EAAE;gBAC1E,OAAOC,SAASvC,EAAE;YACpB;QACF;QAEA,2CAA2C;QAC3C,yCAAyC;QACzC,MAAM2C,UAAU,GAAGhE,KAAK2C,IAAI,CAAC,KAAK,CAAC;QACnC,IAAI,CAACtD,QAAQyD,cAAc,CAACC,GAAG,CAACiB,UAAU;YACxC,uCAAuC;YACvC,MAAMC,oBAAoB,MAAM,IAAI,CAAC7F,YAAY,CAAC6E,oBAAoB,CAACjD,KAAK2C,IAAI,EAAE;gBAChFO,UAAU;gBACVC,OAAO;gBACPE,kBAAkB;gBAClBC,UAAU;YACZ;YACAjE,QAAQyD,cAAc,CAACrD,GAAG,CAACuE,SAASC,kBAAkBV,IAAI,CAACC,OAAO;YAElE,wCAAwC;YACxC,IAAIS,kBAAkBV,IAAI,CAACC,OAAO,CAACG,MAAM,GAAG,GAAG;gBAC7C,MAAM,IAAI,CAACF,mBAAmB,CAACQ,mBAAmB5E;YACpD;QACF;QAEA,MAAM6E,iBAAiB7E,QAAQyD,cAAc,CAACtD,GAAG,CAACwE;QAClD,IAAIE,kBAAkBA,eAAeP,MAAM,GAAG,GAAG;YAC/C,aAAa;YACb,MAAMC,WAAWM,cAAc,CAACL,KAAKC,KAAK,CAACD,KAAKE,MAAM,KAAKG,eAAeP,MAAM,EAAE;YAClF,OAAOC,SAASvC,EAAE;QACpB;QAEA,qCAAqC;QACrC,IAAIrB,KAAKmE,QAAQ,EAAE;YACjB,OAAO;QACT;QAEA,4BAA4B;QAC5B,MAAM,IAAI7C,MACR,CAAC,kBAAkB,EAAE/B,OAAO8B,EAAE,CAAC,CAAC,EAAErB,KAAKG,IAAI,CAAC,MAAM,EAAEH,KAAK2C,IAAI,CAAC,YAAY,CAAC,GACzE,CAAC,GAAG,EAAE3C,KAAK2C,IAAI,CAAC,+BAA+B,CAAC;IAEtD;IAEA;;;;;GAKC,GACD,MAAcc,oBACZT,aAAyC,EACzC3D,OAAyB,EACV;QACf,MAAM+E,oBAAqC,EAAE;QAE7C,qDAAqD;QACrD,KAAK,MAAM,CAACC,UAAUb,QAAQ,IAAIR,cAAcsB,OAAO,CAACC,OAAO,GAAI;YACjE,MAAMhF,SAAS,IAAI,CAACT,aAAa,CAACU,GAAG,CAAC6E;YACtC,MAAMG,kBAA6C,EAAE;YAErD,CAACzG,YACCoD,QAAQsD,GAAG,CACT9G,MAAM+G,IAAI,CAAC,CAAC,0BAA0B,EAAEL,SAAS,EAAE,EAAEb,QAAQG,MAAM,CAAC,SAAS,CAAC;YAGlF,KAAK,MAAMgB,UAAUnB,QAAS;gBAC5B,MAAMoB,YAAY,GAAGP,SAAS,CAAC,EAAEM,OAAOtD,EAAE,EAAE;gBAC5C,IAAI,CAAChC,QAAQwF,eAAe,CAAC9B,GAAG,CAAC6B,YAAY;oBAC3CJ,gBAAgBM,IAAI,CAACH;oBACrBtF,QAAQwF,eAAe,CAACE,GAAG,CAACH;gBAC9B;YACF;YAEA,IAAIJ,gBAAgBb,MAAM,GAAG,GAAG;gBAC9B,KAAK,MAAMgB,UAAUH,gBAAiB;oBACpC,CAACzG,YACCoD,QAAQsD,GAAG,CACT9G,MAAMqH,IAAI,CACR,CAAC,eAAe,EAAEX,SAAS,QAAQ,CAAC,EACpCzB,KAAKC,SAAS,CAAC8B,QAAQM,KAAK,CAAC,GAAG;oBAGtC,MAAMC,iBAAiB,MAAMhH,eAAeiH,mBAAmB,CAC7D5F,QACAoF,QACA;wBAAES,KAAK,IAAI,CAACzG,QAAQ;wBAAE0G,cAAc;oBAAK;oBAE3CjB,kBAAkBU,IAAI,IAAII;gBAC5B;YACF;QACF;QAEA,qCAAqC;QACrC,MAAMI,aAAa,IAAI,CAACxG,aAAa,CAACU,GAAG,CAACwD,cAAcO,IAAI,CAACc,QAAQ;QACrE,MAAMkB,sBAAiD,EAAE;QAEzD,CAACxH,YACCoD,QAAQsD,GAAG,CACT9G,MAAM+G,IAAI,CACR,CAAC,uBAAuB,EAAE1B,cAAcO,IAAI,CAACc,QAAQ,CAAC,EAAE,EAAErB,cAAcO,IAAI,CAACC,OAAO,CAACG,MAAM,CAAC,SAAS,CAAC;QAI5G,KAAK,MAAMgB,UAAU3B,cAAcO,IAAI,CAACC,OAAO,CAAE;YAC/C,MAAMoB,YAAY,GAAG5B,cAAcO,IAAI,CAACc,QAAQ,CAAC,CAAC,EAAEM,OAAOtD,EAAE,EAAE;YAC/D,IAAI,CAAChC,QAAQwF,eAAe,CAAC9B,GAAG,CAAC6B,YAAY;gBAC3CW,oBAAoBT,IAAI,CAACH;gBACzBtF,QAAQwF,eAAe,CAACE,GAAG,CAACH;YAC9B;QACF;QAEA,IAAIW,oBAAoB5B,MAAM,GAAG,GAAG;YAClC,KAAK,MAAMgB,UAAUY,oBAAqB;gBACxC,CAACxH,YACCoD,QAAQsD,GAAG,CACT9G,MAAMqH,IAAI,CACR,CAAC,eAAe,EAAEhC,cAAcO,IAAI,CAACc,QAAQ,CAAC,QAAQ,CAAC,EACvDzB,KAAKC,SAAS,CAAC8B,QAAQM,KAAK,CAAC,GAAG;gBAGtC,MAAMC,iBAAiB,MAAMhH,eAAeiH,mBAAmB,CAC7DG,YACAX,QACA;oBAAES,KAAK,IAAI,CAACzG,QAAQ;oBAAE0G,cAAc;gBAAK;gBAE3CjB,kBAAkBU,IAAI,IAAII;YAC5B;QACF;QAEA,wCAAwC;QACxC,IAAId,kBAAkBT,MAAM,GAAG,GAAG;YAChC,MAAMzF,eAAesH,cAAc,CAAC,IAAI,CAAC3G,YAAY,EAAEuF;YAEvD,CAACrG,YACCoD,QAAQsD,GAAG,CACT9G,MAAM8H,KAAK,CACT,CAAC,cAAc,EAAEzC,cAAcO,IAAI,CAACc,QAAQ,CAAC,iBAAiB,CAAC,GAC7D,GAAGrB,cAAcO,IAAI,CAACC,OAAO,CAACG,MAAM,CAAC,QAAQ,EAAEX,cAAcsB,OAAO,CAACoB,IAAI,CAAC,iBAAiB,CAAC;QAGtG;IACF;IAEA;;;;;;GAMC,GACD,MAAcjE,iBACZkE,SAAiB,EACjB3F,IAAgB,EAChBT,MAAc,EACI;QAClB,mBAAmB;QACnB,IAAIoG,UAAUC,UAAU,CAAC,WAAW;YAClC,mCAAmC;YACnC,MAAMC,cAAc7F,KAAKG,IAAI,KAAK,cAAcH,KAAKG,IAAI,KAAK;YAC9D,MAAM2F,cAAc,MAAM,MAAM,CAAC;YACjC,MAAMvF,QAAQsF,cAAcC,YAAYC,OAAO,GAAGD,YAAYvF,KAAK;YACnE,MAAMyF,OAAOL,UAAUV,KAAK,CAAC,IAAI,cAAc;YAE/C,IAAI;gBACF,eAAe;gBACf,MAAMgB,QAAQD,KAAKC,KAAK,CAAC;gBACzB,IAAI,CAACA,OAAO;oBACV,MAAM,IAAI3E,MACR,CAAC,+CAA+C,EAAEtB,KAAKG,IAAI,CAAC,EAAE,EAAEwF,WAAW;gBAE/E;gBAEA,MAAM,GAAGO,MAAMC,QAAQ,GAAGF;gBAC1B,MAAMG,QAAQF,KAAKhE,KAAK,CAAC;gBAEzB,mBAAmB;gBACnB,IAAImE,KAAc9F;gBAClB,KAAK,MAAM+F,QAAQF,MAAO;oBACxB,IAAI,OAAOC,OAAO,YAAYA,OAAO,QAAQC,QAAQD,IAAI;wBACvDA,KAAK,AAACA,EAA8B,CAACC,KAAK;oBAC5C,OAAO;wBACL,MAAM,IAAIhF,MAAM,CAAC,yCAAyC,EAAEtB,KAAKG,IAAI,CAAC,QAAQ,EAAE+F,MAAM;oBACxF;gBACF;gBAEA,aAAa;gBACb,IAAI,OAAOG,OAAO,YAAY;oBAC5B,MAAM,IAAI/E,MAAM,CAAC,wBAAwB,EAAE4E,KAAK,wBAAwB,EAAElG,KAAKG,IAAI,CAAC,CAAC,CAAC;gBACxF;gBAEA,IAAIoG,OAAkB,EAAE;gBACxB,IAAIJ,SAASK,QAAQ;oBACnBD,OAAO,IAAI,CAACE,kBAAkB,CAACN,SAASnG,KAAKG,IAAI;gBACnD;gBAEA,OAAOkG,MAAME;YACf,EAAE,OAAOrF,OAAO;gBACd,CAACnD,YACCoD,QAAQsD,GAAG,CACT9G,MAAM+I,MAAM,CACV,CAAC,6BAA6B,EAAEf,UAAU,MAAM,EAAE3F,KAAKG,IAAI,CAAC,0BAA0B,CAAC,GAEzFe;gBAEJ,OAAO,IAAI,CAACS,oBAAoB,CAAC3B,MAAMT;YACzC;QACF;QAEA,yBAAyB;QACzB,CAACxB,YACCoD,QAAQsD,GAAG,CACT9G,MAAM+I,MAAM,CACV,CAAC,qCAAqC,EAAE1G,KAAKG,IAAI,CAAC,EAAE,EAAEwF,UAAU,8DAA8D,CAAC;QAGrI,OAAO,IAAI,CAAChE,oBAAoB,CAAC3B,MAAMT;IACzC;IAEA;;;;;;;;;GASC,GACD,MAAcoC,qBAAqB3B,IAAgB,EAAET,MAAe,EAAoB;QACtF,MAAMuG,cAAc,MAAM,MAAM,CAAC;QACjC,MAAMvF,QAAQuF,YAAYvF,KAAK;QAC/B,MAAMwF,UAAUD,YAAYC,OAAO;QACnC,MAAMY,UAAUb,YAAYa,OAAO;QAEnC,MAAMC,cAAc,IAAI,CAACvI,MAAM,KAAK,OAAO0H,UAAU,IAAI,CAAC1H,MAAM,KAAK,OAAOsI,UAAUpG;QAEtF;;;;KAIC,GAED;;;KAGC,GACD,IAAIhB,QAAQ8B,OAAO,gBAAgBrB,KAAKG,IAAI,KAAK,QAAQ;YACvD,MAAM0G,cAAc;gBAClB;gBACA;gBACA;gBACA;gBACA;gBACA;gBACA;gBACA;gBACA;gBACA;gBACA;gBACA;gBACA;gBACA;gBACA;gBACA;gBACA;aACD;YACD,MAAMC,WAAW;gBAAC;gBAAM;gBAAM;gBAAM;gBAAO;gBAAO;aAAK;YACvD,MAAMC,WAAW;gBAAC;gBAAM;gBAAM;gBAAM;gBAAM;gBAAM;gBAAM;gBAAM;aAAK;YAEjE,MAAMC,OAAOzG,MAAM0G,OAAO,CAACC,YAAY,CAACL;YAExC,MAAM9C,SAASF,KAAKE,MAAM;YAC1B,IAAIA,SAAS,KAAK;gBAChB,MAAMoD,SAAS5G,MAAM0G,OAAO,CAACC,YAAY,CAACJ;gBAC1C,OAAO,GAAGK,OAAO,CAAC,EAAEH,MAAM;YAC5B;YACA,IAAIjD,SAAS,KAAK;gBAChB,MAAMqD,SAAS7G,MAAM0G,OAAO,CAACC,YAAY,CAACH;gBAC1C,OAAO,GAAGC,KAAK,CAAC,EAAEI,QAAQ;YAC5B;YACA,OAAOJ;QACT;QAEA;;;;KAIC,GACD,MAAMK,iBAAiB,IAAI,CAAC/I,QAAQ,CAAC,IAAI,CAACD,MAAM,CAAC,IAAI,IAAI,CAACC,QAAQ,CAACgJ,EAAE;QACrE,MAAMC,iBAAiBvH,KAAKG,IAAI,CAACqH,WAAW,GAAGC,OAAO,CAAC,MAAM;QAE7D,KAAK,MAAM,CAACC,SAAStE,OAAO,IAAIuE,OAAOpD,OAAO,CAAC8C,eAAeO,cAAc,EAAG;YAC7E,IAAIL,eAAeM,QAAQ,CAACH,QAAQF,WAAW,KAAK;gBAClD,IAAI;oBACF,OAAO,MAAM,IAAI,CAACM,sBAAsB,CAAC1E,OAAO7C,KAAK,EAAEP;gBACzD,EAAE,OAAOkB,OAAO;oBACd,CAACnD,YACCoD,QAAQsD,GAAG,CACT9G,MAAM+I,MAAM,CACV,CAAC,iCAAiC,EAAEgB,QAAQ,MAAM,EAAE1H,KAAKG,IAAI,CAAC,eAAe,CAAC,GAEhFe;oBAEJ;gBACF;YACF;QACF;QAEA;;;KAGC,GACD,IAAIlB,KAAKM,IAAI,KAAK,UAAU,QAAQN,QAAQA,KAAKqB,EAAE,EAAE;YACnD,IAAIrB,KAAKqB,EAAE,CAAC0G,QAAQ,CAAC,OAAO;gBAC1B,OAAO,IAAI,CAACC,kBAAkB,CAAChI,MAAMT,QAAQgB,OAAOqG;YACtD;QACF;QAEA,qCAAqC,GACrC,IAAI5G,KAAKM,IAAI,KAAK,QAAQ;YACxB,IAAI2H,aAAuB,EAAE;YAE7B,IAAI,UAAUjI,QAAQkI,MAAMC,OAAO,CAACnI,KAAKoI,IAAI,KAAKpI,KAAKoI,IAAI,CAACzE,MAAM,GAAG,GAAG;gBACtEsE,aAAajI,KAAKoI,IAAI;YACxB,OAAO,IAAI,QAAQpI,QAAQA,KAAKqB,EAAE,IAAI9B,QAAQ8I,YAAY,CAACrI,KAAKqB,EAAE,CAAC,EAAE;gBACnE4G,aAAaN,OAAOW,IAAI,CAAC/I,OAAO8I,UAAU,CAACrI,KAAKqB,EAAE,CAAC;YACrD;YAEA,IAAI4G,WAAWtE,MAAM,GAAG,GAAG;gBACzB,OAAOpD,MAAM0G,OAAO,CAACC,YAAY,CAACe;YACpC;YACA,OAAOjI,KAAKmE,QAAQ,GAAG,OAAO;QAChC;QAEA,IAAInE,KAAKM,IAAI,KAAK,UAAU;YAC1B,IAAI2H,aAAuB,EAAE;YAE7B,IAAI,UAAUjI,QAAQkI,MAAMC,OAAO,CAACnI,KAAKoI,IAAI,KAAKpI,KAAKoI,IAAI,CAACzE,MAAM,GAAG,GAAG;gBACtEsE,aAAajI,KAAKoI,IAAI;YACxB,OAAO,IAAI,QAAQpI,QAAQA,KAAKqB,EAAE,IAAI9B,QAAQ8I,YAAY,CAACrI,KAAKqB,EAAE,CAAC,EAAE;gBACnE4G,aAAaN,OAAOW,IAAI,CAAC/I,OAAO8I,UAAU,CAACrI,KAAKqB,EAAE,CAAC;YACrD;YAEA,IAAI4G,WAAWtE,MAAM,GAAG,GAAG;gBACzB,OAAO;oBAACpD,MAAM0G,OAAO,CAACC,YAAY,CAACe;iBAAY;YACjD;YACA,OAAO,EAAE;QACX;QAEA;;;KAGC,GACD,IAAIjI,KAAKM,IAAI,KAAK,YAAYN,KAAKM,IAAI,KAAK,cAAcN,KAAKM,IAAI,KAAK,YAAY;YAClF,OAAO;QACT;QAEA,+BAA+B,GAC/B,MAAMiI,cAAclB,eAAemB,aAAa,CAACxI,KAAKM,IAAI,CAAC;QAC3D,IAAIiI,aAAa;YACf,IAAI;gBACF,OAAO,MAAM,IAAI,CAACT,sBAAsB,CAACS,YAAYhI,KAAK,EAAEP;YAC9D,EAAE,OAAOkB,OAAO;gBACd,CAACnD,YACCoD,QAAQsD,GAAG,CACT9G,MAAM+I,MAAM,CAAC,CAAC,mCAAmC,EAAE1G,KAAKM,IAAI,CAAC,iBAAiB,CAAC,EAAEY;YAEvF;QACF;QAEA,sCAAsC,GACtC,OAAQlB,KAAKM,IAAI;YACf,KAAK;YACL,KAAK;gBACH,OAAOC,MAAMkI,KAAK,CAACC,KAAK,CAAC;YAC3B,KAAK;gBACH,OAAOnI,MAAMoI,MAAM,CAACC,GAAG,CAAC;oBAAEC,KAAK;oBAAGC,KAAK;gBAAK;YAC9C,KAAK;gBACH,OAAO;oBAACvI,MAAMoI,MAAM,CAACC,GAAG,CAAC;wBAAEC,KAAK;wBAAGC,KAAK;oBAAK;iBAAG;YAClD,KAAK;gBACH,OAAOvI,MAAMoI,MAAM,CAACI,MAAM,CAAC;oBAAEF,KAAK,EAAE;oBAAEC,KAAK,KAAK;gBAAC;YACnD,KAAK;gBACH,OAAO;oBAACvI,MAAMoI,MAAM,CAACI,MAAM,CAAC;wBAAEF,KAAK,EAAE;wBAAEC,KAAK,KAAK;oBAAC;iBAAG;YACvD,KAAK;YACL,KAAK;gBACH,OAAOvI,MAAMoI,MAAM,CAACK,KAAK,CAAC;oBAAEH,KAAK;oBAAGC,KAAK;gBAAK;YAChD,KAAK;YACL,KAAK;gBACH,OAAO;oBAACvI,MAAMoI,MAAM,CAACK,KAAK,CAAC;wBAAEH,KAAK;wBAAGC,KAAK;oBAAK;iBAAG;YACpD,KAAK;gBACH,OAAOvI,MAAM0I,QAAQ,CAACC,OAAO;YAC/B,KAAK;gBACH,OAAO;oBAAC3I,MAAM0I,QAAQ,CAACC,OAAO;iBAAG;YACnC,KAAK;YACL,KAAK;gBACH,OAAO3I,MAAM4I,IAAI,CAACC,IAAI;YACxB,KAAK;gBACH,OAAO,CAAC;YACV,KAAK;YACL,KAAK;gBACH,OAAO7I,MAAME,MAAM,CAACE,IAAI;YAC1B;gBACE,OAAO;QACX;IACF;IAEA;;;;;;GAMC,GACD,AAAQqH,mBACNhI,IAAgB,EAChBqJ,OAA2B,EAC3B9I,KAA6C,EAC7C+I,YAAoD,EACzC;QACX,MAAMC,QAAQhJ,MAAMoI,MAAM,CAACC,GAAG,CAAC;YAAEC,KAAK;YAAGC,KAAK;QAAE;QAEhD,8CAA8C,GAC9C,IAAI,QAAQ9I,QAAQA,KAAKqB,EAAE,KAAK,gBAAgB;YAC9C,OAAO6G,MAAMsB,IAAI,CAAC;gBAAE7F,QAAQ4F;YAAM,GAAG,IAAO,CAAA;oBAC1CE,KAAKlJ,MAAMmJ,KAAK,CAACD,GAAG;oBACpBtJ,MAAMI,MAAMoJ,MAAM,CAACC,QAAQ;oBAC3BC,WAAWtJ,MAAM0G,OAAO,CAACC,YAAY,CAAC;wBACpC;wBACA;wBACA;wBACA;qBACD;gBACH,CAAA;QACF;QAEA,wBAAwB,GACxB,MAAMK,iBAAiBvH,KAAKG,IAAI,CAACqH,WAAW,GAAGC,OAAO,CAAC,MAAM;QAE7D,IAAIF,eAAeM,QAAQ,CAAC,UAAUN,eAAeM,QAAQ,CAAC,UAAU;YACtE,OAAOK,MAAMsB,IAAI,CAAC;gBAAE7F,QAAQ4F;YAAM,GAAG,IAAMhJ,MAAMuJ,QAAQ,CAACL,GAAG;QAC/D;QAEA,IAAIlC,eAAeM,QAAQ,CAAC,SAASN,eAAeQ,QAAQ,CAAC,MAAM;YACjE,OAAOG,MAAMsB,IAAI,CAAC;gBAAE7F,QAAQ4F;YAAM,GAAG,IAAMhJ,MAAMoI,MAAM,CAACC,GAAG,CAAC;oBAAEC,KAAK;oBAAGC,KAAK;gBAAI;QACjF;QAEA,IAAIvB,eAAeM,QAAQ,CAAC,UAAUN,eAAeM,QAAQ,CAAC,SAAS;YACrE,OAAOK,MAAMsB,IAAI,CAAC;gBAAE7F,QAAQ4F;YAAM,GAAG,IAAMhJ,MAAMkI,KAAK,CAACsB,IAAI;QAC7D;QAEA,4BAA4B,GAC5B,OAAO,EAAE;IACX;IAEA;;;;;;;;;GASC,GACD,MAAcjC,uBAAuBkC,UAAkB,EAAEhK,IAAgB,EAAoB;QAC3F,MAAM8F,cAAc,MAAM,MAAM,CAAC;QACjC,MAAMvF,QAAQuF,YAAYvF,KAAK;QAC/B,MAAMwF,UAAUD,YAAYC,OAAO;QACnC,MAAMY,UAAUb,YAAYa,OAAO;QAEnC,sCAAsC,GACtC,IAAI,CAACqD,WAAWpE,UAAU,CAAC,UAAU;YACnC,IAAI;gBACF,OAAOhD,KAAKqH,KAAK,CAACD;YACpB,EAAE,OAAM;gBACN,OAAOA;YACT;QACF;QAEA,8BAA8B,GAC9B,MAAM/D,QAAQ+D,WAAW/D,KAAK,CAAC;QAC/B,IAAI,CAACA,OAAO;YACV,MAAM,IAAI3E,MAAM,CAAC,0BAA0B,EAAE0I,YAAY;QAC3D;QAEA,MAAM,GAAGE,WAAWlE,KAAK,GAAGC;QAC5B,MAAMkE,gBACJD,cAAc,YAAYnE,UAAUmE,cAAc,YAAYvD,UAAUpG;QAE1E,MAAM6J,YAAYpE,KAAKC,KAAK,CAAC;QAC7B,IAAI,CAACmE,WAAW;YACd,MAAM,IAAI9I,MAAM,CAAC,6BAA6B,EAAEtB,KAAKG,IAAI,CAAC,EAAE,EAAE6J,YAAY;QAC5E;QAEA,MAAM,GAAG9D,MAAMC,QAAQ,GAAGiE;QAC1B,MAAMhE,QAAQF,KAAKhE,KAAK,CAAC;QAEzB,0CAA0C,GAC1C,IAAImE,KAAc8D;QAClB,KAAK,MAAM7D,QAAQF,MAAO;YACxB,IAAI,OAAOC,OAAO,YAAYA,OAAO,QAAQC,QAAQD,IAAI;gBACvDA,KAAK,AAACA,EAA8B,CAACC,KAAK;YAC5C,OAAO;gBACL,MAAM,IAAIhF,MAAM,CAAC,uBAAuB,EAAEtB,KAAKG,IAAI,CAAC,EAAE,EAAE+J,UAAU,CAAC,EAAEhE,MAAM;YAC7E;QACF;QAEA,IAAI,OAAOG,OAAO,YAAY;YAC5B,MAAM,IAAI/E,MAAM,GAAG4I,UAAU,CAAC,EAAEhE,KAAK,wBAAwB,EAAElG,KAAKG,IAAI,CAAC,CAAC,CAAC;QAC7E;QAEA,IAAIoG,OAAkB,EAAE;QACxB,IAAIJ,SAASK,QAAQ;YACnBD,OAAO,IAAI,CAACE,kBAAkB,CAACN,SAASnG,KAAKG,IAAI;QACnD;QAEA,OAAOkG,MAAME;IACf;IAEA;;;;;;;;;GASC,GACD,MAActF,gBACZoJ,WAAmB,EACnBrK,IAAgB,EAChBT,MAAc,EACdM,MAAe,EACG;QAClB,8BAA8B;QAC9B,IAAIA,QAAQ;YACV,MAAMyK,cAAc,GAAGzK,OAAO,CAAC,EAAEG,KAAKG,IAAI,EAAE;YAE5C,yCAAyC;YACzC,IAAI,IAAI,CAAC5B,QAAQ,CAACwE,GAAG,CAACuH,cAAc;gBAClC,OAAO,IAAI,CAAC/L,QAAQ,CAACiB,GAAG,CAAC8K;YAC3B;YAEA,iCAAiC;YACjC,MAAMC,WAAWhL,OAAOU,KAAK,CAACuK,MAAM,CAAC,CAACC;gBACpC,IAAI3M,eAAe2M,IAAI,OAAO;gBAC9B,IAAIA,EAAErK,IAAI,EAAEoB,kBAAkB,OAAO;gBACrC,IAAIiJ,EAAEtK,IAAI,KAAK,QAAQsK,EAAErK,IAAI,EAAEC,oBAAoB,YAAY,OAAO;gBACtE,OAAO,CAAC,CAACoK,EAAErK,IAAI,EAAEY;YACnB;YAEA,sCAAsC;YACtC,IAAIuJ,SAAS5G,MAAM,KAAK,GAAG;gBACzB,CAAC5F,YACCoD,QAAQsD,GAAG,CACT,CAAC,yCAAyC,EAAElF,OAAO8B,EAAE,CAAC,CAAC,EAAErB,KAAKG,IAAI,CAAC,6BAA6B,CAAC;gBAErG,OAAO,IAAI,CAACuK,qBAAqB,CAACL,aAAarK,MAAMT;YACvD;YAEA,MAAMoL,SAAS,IAAI,CAACC,SAAS;YAC7B,MAAM,EAAEC,eAAe,EAAE,GAAG,MAAM,MAAM,CAAC;YACzC,MAAM,EAAEC,YAAY,EAAE,GAAG,MAAM,MAAM,CAAC;YAEtC,MAAMC,cAAc,MAAMD,aAAa;gBACrCE,OAAOH,gBAAgB;oBAAEF;gBAAO,GAAG,IAAI,CAACjM,OAAO,CAACO,QAAQ,IAAI;gBAC5DgM,QAAQ,IAAI,CAACC,iBAAiB,CAACX,UAAUhL;YAC3C;YACA,IAAI,CAACwL,eAAe,OAAOA,YAAYI,IAAI,KAAK,UAAU;gBACxD,MAAM,IAAI7J,MAAM;YAClB;YAEA,+BAA+B;YAC/B,MAAM8J,YAAY,IAAI,CAACC,mBAAmB,CAACN,YAAYI,IAAI,EAAEZ;YAC7D,KAAK,MAAM,CAACe,WAAWC,MAAM,IAAI5D,OAAOpD,OAAO,CAAC6G,WAAY;gBAC1D,IAAI,CAAC7M,QAAQ,CAACkB,GAAG,CAAC,GAAGI,OAAO,CAAC,EAAEyL,WAAW,EAAEC;YAC9C;YAEA,oCAAoC;YACpC,IAAI,IAAI,CAAChN,QAAQ,CAACwE,GAAG,CAACuH,cAAc;gBAClC,OAAO,IAAI,CAAC/L,QAAQ,CAACiB,GAAG,CAAC8K;YAC3B;YAEA,yCAAyC;YACzC,OAAO,IAAI,CAACI,qBAAqB,CAACL,aAAarK,MAAMT;QACvD;QAEA,0BAA0B;QAC1B,OAAO,IAAI,CAACmL,qBAAqB,CAACL,aAAarK,MAAMT;IACvD;IAEA;;GAEC,GACD,MAAcmL,sBACZL,WAAmB,EACnBrK,IAAgB,EAChBT,MAAc,EACI;QAClB,MAAMmD,WAAW,GAAGnD,OAAO8B,EAAE,CAAC,CAAC,EAAErB,KAAKG,IAAI,CAAC,CAAC,EAAEkK,aAAa;QAC3D,IAAI,IAAI,CAAC3L,OAAO,CAACM,cAAc,IAAI,IAAI,CAACT,QAAQ,CAACwE,GAAG,CAACL,WAAW;YAC9D,OAAO,IAAI,CAACnE,QAAQ,CAACiB,GAAG,CAACkD;QAC3B;QAEA,MAAMiI,SAAS,IAAI,CAACC,SAAS;QAC7B,MAAM,EAAEC,eAAe,EAAE,GAAG,MAAM,MAAM,CAAC;QACzC,MAAM,EAAEC,YAAY,EAAE,GAAG,MAAM,MAAM,CAAC;QAEtC,MAAMU,iBAAiB,MAAMV,aAAa;YACxCE,OAAOH,gBAAgB;gBAAEF;YAAO,GAAG,IAAI,CAACjM,OAAO,CAACO,QAAQ,IAAI;YAC5DgM,QAAQ,IAAI,CAACQ,cAAc,CAACpB,aAAarK,MAAMT;QACjD;QACA,IAAI,CAACiM,kBAAkB,OAAOA,eAAeL,IAAI,KAAK,UAAU;YAC9D,MAAM,IAAI7J,MAAM;QAClB;QAEA,MAAMiK,QAAQ,IAAI,CAACG,gBAAgB,CAACF,eAAeL,IAAI,EAAEnL,KAAKM,IAAI;QAClE,IAAI,IAAI,CAAC5B,OAAO,CAACM,cAAc,EAAE;YAC/B,IAAI,CAACT,QAAQ,CAACkB,GAAG,CAACiD,UAAU6I;QAC9B;QAEA,OAAOA;IACT;IAEA;;GAEC,GACD,AAAQL,kBAAkBjL,KAAmB,EAAEV,MAAc,EAAU;QACrE,MAAMlB,SAAS,IAAI,CAACK,OAAO,CAACL,MAAM,IAAI;QACtC,MAAMsN,WAAWtN,WAAW,OAAO,WAAWA,WAAW,OAAO,aAAa;QAE7E,MAAMuN,oBAAoB3L,MACvB4L,GAAG,CAAC,CAACpB;YACJ,IAAIqB,OAAO,CAAC,EAAE,EAAErB,EAAEtK,IAAI,CAAC,EAAE,EAAEsK,EAAEnK,IAAI,CAAC,GAAG,EAAEmK,EAAErK,IAAI,EAAEY,QAAQ,IAAI;YAC3D,IACE,AAACyJ,CAAAA,EAAEnK,IAAI,KAAK,UAAUmK,EAAEnK,IAAI,KAAK,QAAO,KACxC,QAAQmK,KACRA,EAAEpJ,EAAE,IACJ9B,OAAO8I,UAAU,EAAE,CAACoC,EAAEpJ,EAAE,CAAC,EACzB;gBACA,MAAM0K,SAASpE,OAAOW,IAAI,CAAC/I,OAAO8I,UAAU,CAACoC,EAAEpJ,EAAE,CAAC,EAAE2K,IAAI,CAAC;gBACzDF,QAAQ,CAAC,kBAAkB,EAAEC,OAAO,CAAC,CAAC;YACxC;YACA,OAAOD;QACT,GACCE,IAAI,CAAC;QAER,0CAA0C;QAC1C,MAAMC,aAAa1M,OAAOU,KAAK,CAC5BuK,MAAM,CACL,CAACC,IACC,CAACxK,MAAM4H,QAAQ,CAAC4C,MAChB,CAAC3M,eAAe2M,MAChBA,EAAEtK,IAAI,KAAK,QACX,CAAE,CAAA,aAAasK,KAAKA,EAAEvK,OAAO,AAAD,GAE/B2L,GAAG,CAAC,CAACpB;YACJ,IAAIqB,OAAO,CAAC,EAAE,EAAErB,EAAEtK,IAAI,CAAC,EAAE,EAAEsK,EAAEnK,IAAI,CAAC,CAAC,CAAC;YACpC,IAAImK,EAAErK,IAAI,EAAEY,MAAM8K,QAAQ,CAAC,EAAE,EAAErB,EAAErK,IAAI,CAACY,IAAI,EAAE;YAC5C,IACE,AAACyJ,CAAAA,EAAEnK,IAAI,KAAK,UAAUmK,EAAEnK,IAAI,KAAK,QAAO,KACxC,QAAQmK,KACRA,EAAEpJ,EAAE,IACJ9B,OAAO8I,UAAU,EAAE,CAACoC,EAAEpJ,EAAE,CAAC,EACzB;gBACA,MAAM0K,SAASpE,OAAOW,IAAI,CAAC/I,OAAO8I,UAAU,CAACoC,EAAEpJ,EAAE,CAAC,EAAE2K,IAAI,CAAC;gBACzDF,QAAQ,CAAC,kBAAkB,EAAEC,OAAO,CAAC,CAAC;YACxC;YACA,OAAOD;QACT,GACCE,IAAI,CAAC;QACR,MAAME,oBAAoBD,aACtB,CAAC,uEAAuE,EAAEA,YAAY,GACtF;QAEJ,MAAME,cAAclM,MAAM4L,GAAG,CAAC,CAACpB,IAAM,CAAC,GAAG,EAAEA,EAAEtK,IAAI,CAAC,IAAI,EAAEsK,EAAEnK,IAAI,CAAC,CAAC,CAAC,EAAE0L,IAAI,CAAC;QAExE,MAAMI,gBAAgB7M,OAAOa,IAAI,EAAEY,OAAO,CAAC,sBAAsB,EAAEzB,OAAOa,IAAI,CAACY,IAAI,EAAE,GAAG;QAExF,OAAO,CAAC,mCAAmC,EAAEzB,OAAO8B,EAAE,CAAC;;QAEnD,EAAE9B,OAAO8B,EAAE,GAAG+K,cAAc;QAC5B,EAAE/N,OAAO,MAAM,EAAEsN,SAAS;;;AAGlC,EAAEC,oBAAoBM,kBAAkB;;;;;;MAMlC,EAAEP,SAAS;;;;AAIjB,EAAEQ,YAAY;CACb,CAAC;IACA;IAEA;;GAEC,GACD,AAAQd,oBAAoBF,IAAY,EAAElL,KAAmB,EAA2B;QACtF,MAAMoM,WAAWlB,KACd3E,IAAI,GACJiB,OAAO,CAAC,gBAAgB,IACxBA,OAAO,CAAC,WAAW,IACnBjB,IAAI;QAEP,IAAI8F;QACJ,IAAI;YACF,MAAMC,MAAM3J,KAAKqH,KAAK,CAACoC;YACvB,IAAI,OAAOE,QAAQ,YAAYA,QAAQ,QAAQrE,MAAMC,OAAO,CAACoE,MAAM;gBACjE,CAACxO,YACCoD,QAAQC,IAAI,CAAC,8DAA8D+J;gBAC7E,OAAO,CAAC;YACV;YACAmB,SAASC;QACX,EAAE,OAAM;YACN,CAACxO,YAAYoD,QAAQC,IAAI,CAAC,wDAAwD+J;YAClF,OAAO,CAAC;QACV;QAEA,MAAMqB,SAAkC,CAAC;QACzC,KAAK,MAAMxM,QAAQC,MAAO;YACxB,IAAID,KAAKG,IAAI,IAAImM,QAAQ;gBACvBE,MAAM,CAACxM,KAAKG,IAAI,CAAC,GAAG,IAAI,CAACuL,gBAAgB,CAACe,OAAOH,MAAM,CAACtM,KAAKG,IAAI,CAAC,IAAI,KAAKH,KAAKM,IAAI;YACtF;QACF;QACA,OAAOkM;IACT;IAEQf,eAAeiB,IAAY,EAAE1M,IAAgB,EAAET,MAAc,EAAU;QAC7E,MAAMlB,SAAS,IAAI,CAACK,OAAO,CAACL,MAAM,IAAI;QACtC,MAAMsN,WAAWtN,WAAW,OAAO,WAAWA,WAAW,OAAO,aAAa;QAE7E,MAAM+N,gBAAgB7M,OAAOa,IAAI,EAAEY,OAAO,CAAC,kBAAkB,EAAEzB,OAAOa,IAAI,CAACY,IAAI,EAAE,GAAG;QAEpF,MAAM2L,cAAcpN,OAAOU,KAAK,CAC7BuK,MAAM,CAAC,CAACC,IAAMA,EAAEtK,IAAI,KAAKH,KAAKG,IAAI,IAAI,CAACrC,eAAe2M,MAAMA,EAAErK,IAAI,EAAEY,MACpE6K,GAAG,CAAC,CAACpB,IAAM,CAAC,EAAE,EAAEA,EAAEtK,IAAI,CAAC,EAAE,EAAEsK,EAAEnK,IAAI,CAAC,GAAG,EAAEmK,EAAErK,IAAI,EAAEY,MAAM,EACrDgL,IAAI,CAAC;QACR,MAAMY,qBAAqBD,cACvB,CAAC,gDAAgD,EAAEA,aAAa,GAChE;QAEJ,IAAI1B,SAAS,CAAC,uBAAuB,EAAE1L,OAAO8B,EAAE,CAAC,CAAC,EAAErB,KAAKG,IAAI,CAAC,QAAQ,EAAEH,KAAKM,IAAI,CAAC,CAAC,EAAE8L,gBAAgBQ,mBAAmB;;aAE/G,EAAEF,KAAK;;;;MAId,EAAEf,SAAS;UACP,EAAE,IAAI,CAACkB,iBAAiB,CAAC7M,KAAKM,IAAI,GAAG;QAE3C,0BAA0B;QAC1B,IAAIN,KAAKM,IAAI,KAAK,UAAUN,KAAKM,IAAI,KAAK,UAAU;YAClD,IAAI2H,aAAuB,EAAE;YAE7B,IAAI,UAAUjI,QAAQkI,MAAMC,OAAO,CAACnI,KAAKoI,IAAI,KAAKpI,KAAKoI,IAAI,CAACzE,MAAM,GAAG,GAAG;gBACtEsE,aAAajI,KAAKoI,IAAI;YACxB,OAAO,IAAI,QAAQpI,QAAQA,KAAKqB,EAAE,IAAI9B,QAAQ8I,YAAY,CAACrI,KAAKqB,EAAE,CAAC,EAAE;gBACnE4G,aAAaN,OAAOW,IAAI,CAAC/I,OAAO8I,UAAU,CAACrI,KAAKqB,EAAE,CAAC;YACrD;YAEA,IAAI4G,WAAWtE,MAAM,GAAG,GAAG;gBACzBsH,UAAU,CAAC,sDAAsD,EAAEhD,WAAW+D,IAAI,CAAC,OAAO;YAC5F;QACF;QAEAf,UAAU,CAAC,aAAa,EAAE,IAAI,CAAC6B,iBAAiB,CAAC9M,KAAKM,IAAI,EAAEjC,SAAS;QAErE,OAAO4M;IACT;IAEQS,iBAAiBP,IAAY,EAAE4B,QAAgB,EAAW;QAChE,MAAMC,UAAU7B,KAAK3E,IAAI;QAEzB,WAAW;QACX,IAAIuG,SAAShF,QAAQ,CAAC,OAAO;YAC3B,IAAI;gBACF,MAAMuE,SAAS1J,KAAKqH,KAAK,CAAC+C;gBAC1B,MAAMC,WAAWF,SAAS9H,KAAK,CAAC,GAAG,CAAC,IAAI,2BAA2B;gBAEnE,IAAIiD,MAAMC,OAAO,CAACmE,SAAS;oBACzB,OAAOA,OAAOT,GAAG,CAAC,CAACqB;wBACjB,4BAA4B;wBAC5B,IAAIA,SAAS,QAAQA,SAASpN,WAAW;4BACvC,OAAO,IAAI,CAACqN,sBAAsB,CAACF;wBACrC;wBACA,wCAAwC;wBACxC,IAAI,OAAOC,SAAS,UAAU;4BAC5B,OAAOD,aAAa,SAChBC,OACA,IAAI,CAACE,gBAAgB,CAACxK,KAAKC,SAAS,CAACqK,OAAOD;wBAClD;wBACA,4BAA4B;wBAC5B,OAAO,IAAI,CAACG,gBAAgB,CAACX,OAAOS,OAAOD;oBAC7C;gBACF;gBAEA,qBAAqB;gBACrB,IAAIX,WAAW,QAAQA,WAAWxM,WAAW;oBAC3C,OAAO;wBAAC,IAAI,CAACqN,sBAAsB,CAACF;qBAAU;gBAChD;gBACA,OAAO;oBAAC,IAAI,CAACG,gBAAgB,CAACX,OAAOH,SAASW;iBAAU;YAC1D,EAAE,OAAM;gBACN,OAAO,EAAE;YACX;QACF;QAEA,OAAO,IAAI,CAACG,gBAAgB,CAACJ,SAASD;IACxC;IAEQI,uBAAuBJ,QAAgB,EAAW;QACxD,OAAQA;YACN,KAAK;gBACH,OAAO;YACT,KAAK;gBACH,OAAO,EAAE;YACX,KAAK;YACL,KAAK;YACL,KAAK;gBACH,OAAO;YACT,KAAK;gBACH,OAAO;YACT,KAAK;gBACH,OAAO,IAAIpN;YACb,KAAK;gBACH,OAAO,CAAC;YACV,KAAK;gBACH,OAAO;YACT;gBACE,OAAO;QACX;IACF;IAEQyN,iBAAiBjC,IAAY,EAAE4B,QAAgB,EAAW;QAChE,MAAMC,UAAU7B,KAAK3E,IAAI;QAEzB,OAAQuG;YACN,KAAK;gBAAW;oBACd,MAAMM,MAAMC,SAASN,SAAS;oBAC9B,OAAOO,OAAOC,KAAK,CAACH,OAAO,IAAIA;gBACjC;YACA,KAAK;gBAAc;oBACjB,IAAI;wBACF,OAAOI,OAAOT;oBAChB,EAAE,OAAM;wBACN,OAAO,EAAE;oBACX;gBACF;YACA,KAAK;YACL,KAAK;YACL,KAAK;gBAAW;oBACd,MAAMK,MAAMK,WAAWV;oBACvB,OAAOO,OAAOC,KAAK,CAACH,OAAO,IAAIA;gBACjC;YACA,KAAK;gBACH,OAAOL,QAAQxF,WAAW,OAAO;YACnC,KAAK;gBAAQ;oBACX,MAAM2B,OAAO,IAAIxJ,KAAKqN;oBACtB,OAAOO,OAAOC,KAAK,CAACrE,KAAKwE,OAAO,MAAM,IAAIhO,SAASwJ;gBACrD;YACA,KAAK;gBACH,IAAI;oBACF,OAAOvG,KAAKqH,KAAK,CAAC+C;gBACpB,EAAE,OAAM;oBACN,OAAOA;gBACT;YACF,KAAK;YACL,KAAK;gBACH,OAAOA;YACT;gBACE,OAAOA;QACX;IACF;IAEA;;;;;;;GAOC,GACD,AAAQvG,mBAAmBN,OAAe,EAAEyH,QAAgB,EAAa;QACvE,gBAAgB;QAChB,IAAI;YACF,MAAMtB,SAAS1J,KAAKqH,KAAK,CAAC,CAAC,CAAC,EAAE9D,QAAQ,CAAC,CAAC;YACxC,OAAO+B,MAAMC,OAAO,CAACmE,UAAUA,SAAS;gBAACA;aAAO;QAClD,EAAE,OAAM;QACN,KAAK;QACP;QAEA,+BAA+B;QAC/B,IAAI;YACF,MAAMuB,UAAU,IAAI,CAACC,sBAAsB,CAAC3H;YAC5C,MAAMmG,SAAS1J,KAAKqH,KAAK,CAAC,CAAC,CAAC,EAAE4D,QAAQ,CAAC,CAAC;YACxC,OAAO3F,MAAMC,OAAO,CAACmE,UAAUA,SAAS;gBAACA;aAAO;QAClD,EAAE,OAAM;QACN,KAAK;QACP;QAEA,iBAAiB;QACjB,MAAMyB,UAAU5H,QAAQK,IAAI;QAC5B,IAAI,CAAC+G,OAAOC,KAAK,CAACD,OAAOQ,WAAW;YAClC,OAAO;gBAACR,OAAOQ;aAAS;QAC1B;QACA,IACE,AAACA,QAAQnI,UAAU,CAAC,QAAQmI,QAAQhG,QAAQ,CAAC,QAC5CgG,QAAQnI,UAAU,CAAC,QAAQmI,QAAQhG,QAAQ,CAAC,MAC7C;YACA,OAAO;gBAACgG,QAAQ9I,KAAK,CAAC,GAAG,CAAC;aAAG;QAC/B;QAEA,MAAM,IAAI3D,MAAM,CAAC,6CAA6C,EAAEsM,SAAS,EAAE,EAAEzH,SAAS;IACxF;IAEA;;;;;;GAMC,GACD,AAAQ2H,uBAAuBE,KAAa,EAAU;QACpD,4CAA4C;QAC5C,MAAMC,mBAAmBD,MAAMvG,OAAO,CACpC,+BACA,CAACyG,GAAGC,UAAoB,CAAC,CAAC,EAAEA,QAAQ1G,OAAO,CAAC,MAAM,OAAOA,OAAO,CAAC,QAAQ,KAAK,CAAC,CAAC;QAGlF,uBAAuB;QACvB,OAAOwG,iBAAiBxG,OAAO,CAAC,sCAAsC;IACxE;IAEA;;;;;GAKC,GACD,AAAQmD,YAAoB;QAC1B,IAAID;QAEJ,IAAI;YACF,MAAM,EAAEyD,MAAM,EAAE,GAAGC,QAAQ;YAC3B1D,SAASyD,OAAOE,MAAM,EAAEC;QAC1B,EAAE,OAAM;QACN,iCAAiC;QACnC;QAEA,IAAI,CAAC5D,QAAQ;YACXA,SAAS6D,QAAQC,GAAG,CAACC,iBAAiB;QACxC;QAEA,IAAI,CAAC/D,QAAQ;YACX,MAAM,IAAIrJ,MACR;QAEJ;QAEA,OAAOqJ;IACT;IAEQkC,kBAAkBE,QAAgB,EAAU;QAClD,WAAW;QACX,IAAIA,SAAShF,QAAQ,CAAC,OAAO;YAC3B,MAAMkF,WAAWF,SAAS9H,KAAK,CAAC,GAAG,CAAC;YACpC,MAAM0J,aAAa,IAAI,CAACC,eAAe,CAAC3B;YACxC,OAAO,CAAC,cAAc,EAAE0B,WAAW,SAAS,EAAE,IAAI,CAAC7B,iBAAiB,CAACG,UAAU,MAAM,OAAO,CAAC;QAC/F;QAEA,OAAO,IAAI,CAAC2B,eAAe,CAAC7B;IAC9B;IAEQ6B,gBAAgB7B,QAAgB,EAAU;QAChD,OAAQA;YACN,KAAK;YACL,KAAK;gBACH,OAAO;YACT,KAAK;YACL,KAAK;YACL,KAAK;gBACH,OAAO;YACT,KAAK;gBACH,OAAO;YACT,KAAK;gBACH,OAAO;YACT,KAAK;gBACH,OAAO;YACT,KAAK;gBACH,OAAO;YACT,KAAK;gBACH,OAAO;YACT;gBACE,OAAO;QACX;IACF;IAEQD,kBAAkBC,QAAgB,EAAE1O,MAAc,EAAU;QAClE,WAAW;QACX,IAAI0O,SAAShF,QAAQ,CAAC,OAAO;YAC3B,MAAMkF,WAAWF,SAAS9H,KAAK,CAAC,GAAG,CAAC;YACpC,MAAM4J,cAAc,IAAI,CAACC,gBAAgB,CAAC7B,UAAU5O;YACpD,OAAO,CAAC,CAAC,EAAEwQ,YAAY,CAAC,CAAC;QAC3B;QAEA,OAAO,IAAI,CAACC,gBAAgB,CAAC/B,UAAU1O;IACzC;IAEQyQ,iBAAiB/B,QAAgB,EAAE1O,MAAc,EAAU;QACjE,MAAM0Q,WAAW1Q,WAAW;QAE5B,OAAQ0O;YACN,KAAK;YACL,KAAK;gBACH,OAAO;YACT,KAAK;YACL,KAAK;YACL,KAAK;gBACH,OAAO;YACT,KAAK;gBACH,OAAO;YACT,KAAK;gBACH,OAAO;YACT,KAAK;gBACH,OAAO;YACT,KAAK;gBACH,OAAO;YACT,KAAK;gBACH,OAAO;YACT;gBACE,OAAOgC,WAAW,UAAU;QAChC;IACF;IAEA;;;;;;GAMC,GACD,MAAc3M,aAAajC,IAAY,EAAmB;QACxD,cAAc;QACd,IAAI,kBAAkB6O,IAAI,CAAC7O,OAAO;YAChC,OAAO,IAAI,CAAC8O,kBAAkB,CAAC9O;QACjC;QACA,iBAAiB;QACjB,OAAOA,KACJqH,WAAW,GACXC,OAAO,CAAC,QAAQ,KAChBA,OAAO,CAAC,eAAe;IAC5B;IAEA;;;;;GAKC,GACD,AAAQwH,mBAAmB9O,IAAY,EAAU;QAC/C,MAAM+O,WAAW;YACf;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;SACD;QACD,MAAMC,YAAY;YAChB;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;SACD;QACD,MAAMC,YAAY;YAChB;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;YACA;SACD;QAED,MAAMC,WAAW,CAACC;YAChB,MAAMC,OAAOD,KAAKE,UAAU,CAAC;YAC7B,IAAID,OAAO,UAAUA,OAAO,QAAQ,OAAOD;YAC3C,MAAMG,SAASF,OAAO;YACtB,MAAMG,MAAM7L,KAAKC,KAAK,CAAC2L,SAAS;YAChC,MAAME,OAAO9L,KAAKC,KAAK,CAAC,AAAC2L,SAAS,MAAO;YACzC,MAAMG,OAAOH,SAAS;YACtB,OAAOP,QAAQ,CAACQ,IAAI,GAAGP,SAAS,CAACQ,KAAK,GAAGP,SAAS,CAACQ,KAAK;QAC1D;QAEA,MAAMC,QAAQ;eAAI1P;SAAK;QACvB,kBAAkB;QAClB,MAAM2P,aAAaT,SAASQ,KAAK,CAAC,EAAE;QACpC,MAAME,YAAYF,MAAM5K,KAAK,CAAC,GAAG4G,GAAG,CAACwD,UAAUrD,IAAI,CAAC;QAEpD,IAAI+D,WAAW;YACb,OAAO,GAAGA,UAAU,CAAC,EAAED,YAAY;QACrC;QACA,OAAOA;IACT;IAEA;;GAEC,GACDE,mBAAmB;QACjB,OAAO;YACLtK,MAAM,IAAI,CAACnH,QAAQ,CAACmH,IAAI;YACxBuK,SAAS,IAAI,CAACvR,OAAO,CAACM,cAAc;QACtC;IACF;IAEA;;GAEC,GACDkR,gBAAgB;QACd,IAAI,CAAC3R,QAAQ,CAAC4R,KAAK;IACrB;IAEA;;GAEC,GACD,AAAQ7Q,gBAAkC;QACxC,OAAO;YACLkD,UAAU,IAAIhE;YACdsE,gBAAgB,IAAItE;YACpBqG,iBAAiB,IAAIuL;QACvB;IACF;IAEA;;;;;;;;GAQC,GACD,MAAMC,cACJC,KAAoF,EACpD;QAChC,MAAMjR,UAAU,IAAI,CAACC,aAAa;QAClC,MAAMiR,oBAA8E,EAAE;QAEtF,yBAAyB;QACzB,KAAK,MAAMC,QAAQF,MAAO;YACxB,IAAK,IAAIG,IAAI,GAAGA,IAAID,KAAKjH,KAAK,EAAEkH,IAAK;gBACnC,MAAM1Q,UAAU,MAAM,IAAI,CAACb,QAAQ,CAACsR,KAAKjR,MAAM,EAAEiR,KAAKpR,SAAS,IAAI,CAAC,GAAGC;gBACvEkR,kBAAkBzL,IAAI,CAAC;oBACrBvF,QAAQiR,KAAKjR,MAAM;oBACnBmR,MAAM3Q;gBACR;YACF;QACF;QAEA,uBAAuB;QACvB,MAAMmF,iBAAkC,EAAE;QAC1C,KAAK,MAAM,EAAE3F,QAAQJ,UAAU,EAAEuR,IAAI,EAAE,IAAIH,kBAAmB;YAC5D,MAAMhR,SAAS,IAAI,CAACT,aAAa,CAACU,GAAG,CAACL;YAEtC,qDAAqD;YACrD,8CAA8C;YAC9C,MAAMwR,SAASpR,OAAOU,KAAK,CAAC2Q,IAAI,CAAC,CAACnG,IAAMA,EAAEtK,IAAI,KAAK;YACnD,MAAM0Q,eACJF,QAAQrQ,SAAS,aACjBqQ,QAAQrQ,SAAS,gBACjBqQ,QAAQvQ,MAAMC,oBAAoB;YAEpC,MAAMyQ,gBAAgBD,eAClB;gBAAE,GAAGH,IAAI;gBAAErP,IAAIwC,KAAKC,KAAK,CAACD,KAAKE,MAAM,KAAK;YAAS,IACnD2M;YAEJ,MAAMlN,UAAU,MAAMtF,eAAeiH,mBAAmB,CACtD5F,QACAuR,eACA;gBAAEzL,cAAc;YAAK;YAEvBH,eAAeJ,IAAI,IAAItB;QACzB;QAEA,8CAA8C;QAC9C,MAAMuN,UAAU,MAAM7S,eAAesH,cAAc,CAAC,IAAI,CAAC3G,YAAY,EAAEqG;QAEvE,uDAAuD;QACvD,MAAM8L,mBAAmB,MAAM,IAAI,CAACC,kBAAkB,CAACX,OAAOS;QAE9D,MAAMG,QAAQH,QAAQpN,MAAM,GAAGqN,iBAAiBrN,MAAM;QACtD,CAAC5F,YACCoD,QAAQsD,GAAG,CAAC9G,MAAM8H,KAAK,CAAC,CAAC,oBAAoB,EAAEyL,MAAM,aAAa,EAAE,IAAI,CAACrS,YAAY,EAAE;QACzF,OAAO;eAAIkS;eAAYC;SAAiB;IAC1C;IAEA;;;;;GAKC,GACD,MAAcC,mBACZX,KAAoF,EACpFa,aAAoC,EACJ;QAChC,MAAMC,aAAoC,EAAE;QAC5C,MAAMC,oBAAoB,IAAIjB;QAE9B,KAAK,MAAMI,QAAQF,MAAO;YACxB,IAAIe,kBAAkBtO,GAAG,CAACyN,KAAKjR,MAAM,GAAG;YACxC8R,kBAAkBtM,GAAG,CAACyL,KAAKjR,MAAM;YAEjC,MAAMA,SAAS,IAAI,CAACT,aAAa,CAACU,GAAG,CAACgR,KAAKjR,MAAM;YACjD,MAAMoR,SAASpR,OAAOU,KAAK,CAAC2Q,IAAI,CAAC,CAACnG,IAAMA,EAAEtK,IAAI,KAAK;YACnD,MAAMmR,aAAaX,QAAQvQ,MAAMmR;YACjC,IAAI,CAACD,cAAcA,WAAW3N,MAAM,KAAK,GAAG;YAE5C,MAAM6N,gBAAgBL,cAAc3G,MAAM,CAAC,CAACiH,IAAMA,EAAEpN,QAAQ,KAAKmM,KAAKjR,MAAM;YAC5E,IAAIiS,cAAc7N,MAAM,KAAK,GAAG;YAEhC,KAAK,MAAM+N,aAAaJ,WAAY;gBAClC,wDAAwD;gBACxD,MAAMK,kBAAkB,IAAI,CAAC7S,aAAa,CAACU,GAAG,CAACkS,UAAUnS,MAAM;gBAC/D,MAAMqS,SAASD,gBAAgB1R,KAAK,CAAC2Q,IAAI,CACvC,CAACnG,IAAM3M,eAAe2M,MAAM7M,2BAA2B6M,MAAMA,EAAE9H,IAAI,KAAK6N,KAAKjR,MAAM;gBAErF,IAAI,CAACqS,QAAQ;oBACX,CAAC7T,YACCoD,QAAQC,IAAI,CACVzD,MAAM+I,MAAM,CACV,CAAC,0CAA0C,EAAEgL,UAAUnS,MAAM,CAAC,IAAI,EAAEiR,KAAKjR,MAAM,CAAC,WAAW,CAAC;oBAGlG;gBACF;gBACA,MAAMqB,YAAY,GAAGgR,OAAOzR,IAAI,CAAC,GAAG,CAAC;gBAErC,2DAA2D;gBAC3D,MAAM0R,kBAAkBF,gBAAgB1R,KAAK,CAAC2Q,IAAI,CAAC,CAACnG,IAAMA,EAAEtK,IAAI,KAAK;gBACrE,MAAM0Q,eACJgB,iBAAiBvR,SAAS,aAC1BuR,iBAAiBvR,SAAS,gBAC1BuR,iBAAiBzR,MAAMC,oBAAoB;gBAC7C,MAAMyR,iBAAiBJ,UAAUnI,KAAK,IAAI;gBAE1C,2CAA2C;gBAC3C,MAAMlK,UAAU,IAAI,CAACC,aAAa;gBAClC,MAAMyS,0BAA2C,EAAE;gBAEnD,KAAK,MAAMC,gBAAgBR,cAAe;oBACxC,MAAMS,oBAAoB,IAAI,CAACC,wBAAwB,CACrDR,UAAUtS,SAAS,IAAI,CAAC,GACxB4S,aAAatB,IAAI;oBAEnBuB,iBAAiB,CAACrR,UAAU,GAAGoR,aAAatB,IAAI,CAACrP,EAAE;oBAEnD,IAAK,IAAIoP,IAAI,GAAGA,IAAIqB,gBAAgBrB,IAAK;wBACvC,MAAM1Q,UAAU,MAAM,IAAI,CAACb,QAAQ,CAACwS,UAAUnS,MAAM,EAAE0S,mBAAmB5S;wBAEzE,MAAMyR,gBAAgBD,eAClB;4BAAE,GAAG9Q,OAAO;4BAAEsB,IAAIwC,KAAKC,KAAK,CAACD,KAAKE,MAAM,KAAK;wBAAS,IACtDhE;wBAEJ,MAAMyD,UAAU,MAAMtF,eAAeiH,mBAAmB,CACtDwM,iBACAb,eAIA;4BAAEzL,cAAc;wBAAK;wBAEvB0M,wBAAwBjN,IAAI,IAAItB;oBAClC;gBACF;gBAEA,MAAMwN,mBAAmB,MAAM9S,eAAesH,cAAc,CAC1D,IAAI,CAAC3G,YAAY,EACjBkT;gBAEFX,WAAWtM,IAAI,IAAIkM;gBAEnB,CAACjT,YACCoD,QAAQsD,GAAG,CACT9G,MAAM8H,KAAK,CACT,CAAC,sBAAsB,EAAEuL,iBAAiBrN,MAAM,CAAC,CAAC,EAAE+N,UAAUnS,MAAM,CAAC,SAAS,CAAC;YAGvF;QACF;QAEA,OAAO6R;IACT;IAEA;;;;GAIC,GACD,AAAQc,yBACN9S,SAAkC,EAClC+S,UAAsE,EAC7C;QACzB,MAAMC,WAAoC,CAAC;QAC3C,KAAK,MAAM,CAACC,KAAK9G,MAAM,IAAI5D,OAAOpD,OAAO,CAACnF,WAAY;YACpD,IAAI,OAAOmM,UAAU,YAAYA,MAAM3F,UAAU,CAAC,SAAS2F,MAAMxD,QAAQ,CAAC,OAAO;gBAC/E,MAAMuD,YAAYC,MAAMtG,KAAK,CAAC,GAAG,CAAC,GAAGuB,IAAI;gBACzC,IAAI,CAAE8E,CAAAA,aAAa6G,UAAS,GAAI;oBAC9B,MAAM,IAAI7Q,MACR,CAAC,QAAQ,EAAEgK,UAAU,gDAAgD,EAAE+G,IAAI,EAAE,CAAC;gBAElF;gBACAD,QAAQ,CAACC,IAAI,GAAGF,UAAU,CAAC7G,UAAU;YACvC,OAAO;gBACL8G,QAAQ,CAACC,IAAI,GAAG9G;YAClB;QACF;QACA,OAAO6G;IACT;IAEA;;;;;;;;;;;;;;;;;;;GAmBC,GACD,MAAME,iBACJnT,UAAkB,EAClBT,OAAoC,EACJ;QAChC,CAACX,YACCoD,QAAQsD,GAAG,CACT9G,MAAM4U,IAAI,CACR,CAAC,UAAU,EAAEpT,WAAW,8BAA8B,EAAEyD,KAAKC,SAAS,CAAC;YAAEK,UAAUxE,QAAQwE,QAAQ;YAAEC,OAAOzE,QAAQyE,KAAK;YAAEE,kBAAkB3E,QAAQ2E,gBAAgB;YAAEC,UAAU5E,QAAQ4E,QAAQ;QAAC,IAAI;QAI5M,iDAAiD;QACjD,MAAMN,gBAAgB,MAAM,IAAI,CAAC5E,YAAY,CAAC6E,oBAAoB,CAAC9D,YAAYT;QAE/E,CAACX,YACCoD,QAAQsD,GAAG,CACT9G,MAAM+G,IAAI,CACR,CAAC,MAAM,EAAE1B,cAAcO,IAAI,CAACC,OAAO,CAACG,MAAM,CAAC,CAAC,EAAExE,WAAW,aAAa,EAAE6D,cAAcsB,OAAO,CAACoB,IAAI,CAAC,iBAAiB,CAAC;QAI3H,uBAAuB;QACvB,MAAMR,iBAAkC,EAAE;QAE1C,wCAAwC;QACxC,MAAMI,aAAa,IAAI,CAACxG,aAAa,CAACU,GAAG,CAACL;QAC1C,KAAK,MAAMwF,UAAU3B,cAAcO,IAAI,CAACC,OAAO,CAAE;YAC/C,MAAMA,UAAU,MAAMtF,eAAeiH,mBAAmB,CACtDG,YACAX,QACA;gBAAES,KAAK,IAAI,CAACzG,QAAQ;gBAAE0G,cAAc;YAAK;YAE3CH,eAAeJ,IAAI,IAAItB;QACzB;QAEA,wCAAwC;QACxC,KAAK,MAAM,CAACgP,mBAAmBC,eAAe,IAAIzP,cAAcsB,OAAO,CAACC,OAAO,GAAI;YACjF,MAAMmO,gBAAgB,IAAI,CAAC5T,aAAa,CAACU,GAAG,CAACgT;YAC7C,KAAK,MAAM7N,UAAU8N,eAAgB;gBACnC,MAAMjP,UAAU,MAAMtF,eAAeiH,mBAAmB,CACtDuN,eACA/N,QACA;oBAAES,KAAK,IAAI,CAACzG,QAAQ;oBAAE0G,cAAc;gBAAK;gBAE3CH,eAAeJ,IAAI,IAAItB;YACzB;YAEA,CAACzF,YACCoD,QAAQsD,GAAG,CAAC9G,MAAMqH,IAAI,CAAC,CAAC,IAAI,EAAEwN,kBAAkB,EAAE,EAAEC,eAAe9O,MAAM,CAAC,QAAQ,CAAC;QACvF;QAEA,8CAA8C;QAC9C,MAAMoN,UAAU,MAAM7S,eAAesH,cAAc,CAAC,IAAI,CAAC3G,YAAY,EAAEqG;QAEvE,CAACnH,YACCoD,QAAQsD,GAAG,CACT9G,MAAM8H,KAAK,CACT,CAAC,sBAAsB,EAAEsL,QAAQpN,MAAM,CAAC,YAAY,EAAE,IAAI,CAAC9E,YAAY,CAAC,EAAE,EAAEmE,cAAcO,IAAI,CAACC,OAAO,CAACG,MAAM,CAAC,CAAC,EAAExE,WAAW,GAAG,EAAE4R,QAAQpN,MAAM,GAAGX,cAAcO,IAAI,CAACC,OAAO,CAACG,MAAM,CAAC,SAAS,CAAC;QAIpM,OAAOoN;IACT;AACF"}