sonamu 0.7.18 → 0.7.20

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 (67) hide show
  1. package/dist/api/config.d.ts +21 -3
  2. package/dist/api/config.d.ts.map +1 -1
  3. package/dist/api/config.js +1 -1
  4. package/dist/api/context.d.ts +3 -3
  5. package/dist/api/context.d.ts.map +1 -1
  6. package/dist/api/context.js +1 -1
  7. package/dist/api/decorators.d.ts.map +1 -1
  8. package/dist/api/decorators.js +4 -8
  9. package/dist/api/index.d.ts +0 -2
  10. package/dist/api/index.d.ts.map +1 -1
  11. package/dist/api/index.js +1 -3
  12. package/dist/api/sonamu.d.ts +5 -3
  13. package/dist/api/sonamu.d.ts.map +1 -1
  14. package/dist/api/sonamu.js +10 -8
  15. package/dist/bin/cli.js +3 -3
  16. package/dist/database/db.d.ts +2 -2
  17. package/dist/database/db.d.ts.map +1 -1
  18. package/dist/database/db.js +23 -30
  19. package/dist/index.d.ts +0 -1
  20. package/dist/index.d.ts.map +1 -1
  21. package/dist/index.js +1 -2
  22. package/dist/storage/drivers.d.ts +14 -0
  23. package/dist/storage/drivers.d.ts.map +1 -0
  24. package/dist/storage/drivers.js +11 -0
  25. package/dist/storage/index.d.ts +5 -0
  26. package/dist/storage/index.d.ts.map +1 -0
  27. package/dist/storage/index.js +6 -0
  28. package/dist/storage/storage-manager.d.ts +21 -0
  29. package/dist/storage/storage-manager.d.ts.map +1 -0
  30. package/dist/storage/storage-manager.js +33 -0
  31. package/dist/storage/types.d.ts +12 -0
  32. package/dist/storage/types.d.ts.map +1 -0
  33. package/dist/storage/types.js +5 -0
  34. package/dist/storage/uploaded-file.d.ts +35 -0
  35. package/dist/storage/uploaded-file.d.ts.map +1 -0
  36. package/dist/storage/uploaded-file.js +58 -0
  37. package/dist/template/implementations/services.template.d.ts.map +1 -1
  38. package/dist/template/implementations/services.template.js +8 -5
  39. package/dist/testing/fixture-manager.d.ts.map +1 -1
  40. package/dist/testing/fixture-manager.js +4 -4
  41. package/dist/ui-web/assets/{index-DFqVuxOB.js → index-B87IyofX.js} +1 -1
  42. package/dist/ui-web/index.html +1 -1
  43. package/package.json +6 -1
  44. package/src/api/config.ts +21 -3
  45. package/src/api/context.ts +3 -3
  46. package/src/api/decorators.ts +3 -8
  47. package/src/api/index.ts +0 -2
  48. package/src/api/sonamu.ts +12 -9
  49. package/src/bin/cli.ts +2 -2
  50. package/src/database/db.ts +40 -43
  51. package/src/index.ts +0 -1
  52. package/src/storage/drivers.ts +15 -0
  53. package/src/storage/index.ts +5 -0
  54. package/src/storage/storage-manager.ts +39 -0
  55. package/src/storage/types.ts +12 -0
  56. package/src/storage/uploaded-file.ts +81 -0
  57. package/src/template/implementations/service.template.ts.txt +328 -0
  58. package/src/template/implementations/services.template.ts +7 -4
  59. package/src/testing/fixture-manager.ts +3 -4
  60. package/dist/file-storage/driver.d.ts +0 -48
  61. package/dist/file-storage/driver.d.ts.map +0 -1
  62. package/dist/file-storage/driver.js +0 -79
  63. package/dist/file-storage/file-storage.d.ts +0 -50
  64. package/dist/file-storage/file-storage.d.ts.map +0 -1
  65. package/dist/file-storage/file-storage.js +0 -75
  66. package/src/file-storage/driver.ts +0 -131
  67. package/src/file-storage/file-storage.ts +0 -100
@@ -52,7 +52,7 @@ export class FixtureManagerClass {
52
52
  }
53
53
  }
54
54
  this.tdb = knex(Sonamu.dbConfig.test);
55
- this.fdb = knex(Sonamu.dbConfig.fixture_remote);
55
+ this.fdb = knex(Sonamu.dbConfig.fixture);
56
56
  }
57
57
  async getChecksum(db, tableName) {
58
58
  const [[checksumRow]] = await db.raw(`CHECKSUM TABLE ${tableName}`);
@@ -62,7 +62,7 @@ export class FixtureManagerClass {
62
62
  원격 fixture DB를 로컬 test DB로 복사합니다.
63
63
  pg_dump로 원격 DB를 덤프하고, pg_restore로 로컬에 복원합니다.
64
64
  */ async sync() {
65
- const fixtureConn = Sonamu.dbConfig.fixture_remote.connection;
65
+ const fixtureConn = Sonamu.dbConfig.fixture.connection;
66
66
  const testConn = Sonamu.dbConfig.test.connection;
67
67
  // 1. 로컬 test DB 연결 종료 및 재생성
68
68
  const testPgEnv = {
@@ -145,7 +145,7 @@ export class FixtureManagerClass {
145
145
  throw new Error(`${entityId}#${id} row를 찾을 수 없습니다.`);
146
146
  }
147
147
  // 픽스쳐DB, 실DB
148
- const fixtureDatabase = Sonamu.dbConfig.fixture_remote.connection.database;
148
+ const fixtureDatabase = Sonamu.dbConfig.fixture.connection.database;
149
149
  const realDatabase = Sonamu.dbConfig.production_master.connection.database;
150
150
  const selfQuery = `INSERT IGNORE INTO \`${fixtureDatabase}\`.\`${entity.table}\` (SELECT * FROM \`${realDatabase}\`.\`${entity.table}\` WHERE \`id\` = ${id})`;
151
151
  const args = Object.entries(entity.relations).filter(([, relation])=>isBelongsToOneRelationProp(relation) || isOneToOneRelationProp(relation) && relation.customJoinClause === undefined).map(([, relation])=>{
@@ -620,4 +620,4 @@ export class FixtureManagerClass {
620
620
  }
621
621
  export const FixtureManager = new FixtureManagerClass();
622
622
 
623
- //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/testing/fixture-manager.ts"],"sourcesContent":["import assert from \"assert\";\nimport chalk from \"chalk\";\nimport { execSync } from \"child_process\";\nimport { readFileSync, writeFileSync } from \"fs\";\nimport inflection from \"inflection\";\nimport knex, { type Knex } from \"knex\";\nimport { unique } from \"radashi\";\nimport { inspect } from \"util\";\nimport { Sonamu } from \"../api\";\nimport { BaseModel } from \"../database/base-model\";\nimport type { SonamuDBConfig } from \"../database/db\";\nimport { type UBRef, UpsertBuilder } from \"../database/upsert-builder\";\nimport type { Entity } from \"../entity/entity\";\nimport { EntityManager } from \"../entity/entity-manager\";\nimport {\n  type DatabaseSchemaExtend,\n  type EntityProp,\n  type FixtureImportResult,\n  type FixtureRecord,\n  type FixtureSearchOptions,\n  isBelongsToOneRelationProp,\n  isHasManyRelationProp,\n  isManyToManyRelationProp,\n  isOneToOneRelationProp,\n  isRelationProp,\n  isVirtualProp,\n  type ManyToManyRelationProp,\n} from \"../types/types\";\nimport { RelationGraph } from \"./_relation-graph\";\n\n/** 사용자 지정 중복 확인 컬럼 (entityId별로 지정) */\nexport interface DuplicateCheckOptions {\n  columns?: {\n    [entityId: string]: string[];\n  };\n}\n\nexport class FixtureManagerClass {\n  private _tdb: Knex | null = null;\n  set tdb(tdb: Knex) {\n    this._tdb = tdb;\n  }\n  get tdb(): Knex {\n    if (this._tdb === null) {\n      throw new Error(\"FixtureManager has not been initialized\");\n    }\n    return this._tdb;\n  }\n\n  private _fdb: Knex | null = null;\n  set fdb(fdb: Knex) {\n    this._fdb = fdb;\n  }\n  get fdb(): Knex {\n    if (this._fdb === null) {\n      throw new Error(\"FixtureManager has not been initialized\");\n    }\n    return this._fdb;\n  }\n  cachedTableNames: string[] | null = null;\n\n  private relationGraph = new RelationGraph();\n\n  // UpsertBuilder 기반 import를 위한 상태\n  private builder: UpsertBuilder = new UpsertBuilder();\n  private fixtureRefMap: Map<string, UBRef> = new Map();\n  private uuidToFixtureId: Map<string, string> = new Map();\n  private skippedFixtures: Map<string, { entityId: string; existingId: number }> = new Map();\n\n  init() {\n    if (this._tdb !== null) {\n      return;\n    }\n    if (Sonamu.dbConfig.test && Sonamu.dbConfig.production_master) {\n      const tConn = Sonamu.dbConfig.test.connection as Knex.ConnectionConfig & {\n        port?: number;\n      };\n      const pConn = Sonamu.dbConfig.production_master.connection as Knex.ConnectionConfig & {\n        port?: number;\n      };\n      if (\n        `${tConn.host ?? \"localhost\"}:${tConn.port ?? 5432}/${tConn.database}` ===\n        `${pConn.host ?? \"localhost\"}:${pConn.port ?? 5432}/${pConn.database}`\n      ) {\n        throw new Error(`테스트DB와 프로덕션DB에 동일한 데이터베이스가 사용되었습니다.`);\n      }\n    }\n\n    this.tdb = knex(Sonamu.dbConfig.test);\n    this.fdb = knex(Sonamu.dbConfig.fixture_remote);\n  }\n\n  async getChecksum(db: Knex, tableName: string) {\n    const [[checksumRow]] = await db.raw(`CHECKSUM TABLE ${tableName}`);\n    return checksumRow.Checksum;\n  }\n\n  /**\n    원격 fixture DB를 로컬 test DB로 복사합니다.\n    pg_dump로 원격 DB를 덤프하고, pg_restore로 로컬에 복원합니다.\n  */\n  async sync() {\n    const fixtureConn = Sonamu.dbConfig.fixture_remote.connection as Knex.PgConnectionConfig;\n    const testConn = Sonamu.dbConfig.test.connection as Knex.PgConnectionConfig;\n\n    // 1. 로컬 test DB 연결 종료 및 재생성\n    const testPgEnv = { PGPASSWORD: testConn.password || \"\" };\n    execSync(\n      `psql -h ${testConn.host} -p ${testConn.port ?? 5432} -U ${testConn.user} -d postgres -c \"\n        SELECT pg_terminate_backend(pg_stat_activity.pid)\n        FROM pg_stat_activity\n        WHERE datname = '${testConn.database}'\n          AND pid <> pg_backend_pid();\n      \"`,\n      { stdio: \"inherit\", env: { ...process.env, ...testPgEnv } as NodeJS.ProcessEnv },\n    );\n\n    execSync(\n      `psql -h ${testConn.host} -p ${testConn.port ?? 5432} -U ${testConn.user} -d postgres -c \"DROP DATABASE IF EXISTS \\\\\"${testConn.database}\\\\\"\"`,\n      { stdio: \"inherit\", env: { ...process.env, ...testPgEnv } as NodeJS.ProcessEnv },\n    );\n\n    execSync(\n      `psql -h ${testConn.host} -p ${testConn.port ?? 5432} -U ${testConn.user} -d postgres -c \"CREATE DATABASE \\\\\"${testConn.database}\\\\\"\"`,\n      { stdio: \"inherit\", env: { ...process.env, ...testPgEnv } as NodeJS.ProcessEnv },\n    );\n\n    // 2. 원격 fixture DB → 로컬 test DB로 복사 (pg_dump | pg_restore)\n    const fixturePgEnv = { PGPASSWORD: fixtureConn.password || \"\" };\n    const dumpCmd = `pg_dump -h ${fixtureConn.host} -p ${fixtureConn.port ?? 5432} -U ${fixtureConn.user} -d ${fixtureConn.database} -Fc`;\n    const restoreCmd = `pg_restore -h ${testConn.host} -p ${testConn.port ?? 5432} -U ${testConn.user} -d ${testConn.database} --no-owner --no-acl`;\n\n    execSync(`${dumpCmd} | PGPASSWORD=\"${testConn.password || \"\"}\" ${restoreCmd}`, {\n      stdio: \"inherit\",\n      env: { ...process.env, ...fixturePgEnv } as NodeJS.ProcessEnv,\n      shell: \"/bin/bash\",\n    });\n  }\n\n  private visitedRecords = new Set<string>();\n  async importFixture(entityId: string, ids: number[]) {\n    // 방문 기록 초기화 (새로운 import 작업 시작)\n    this.visitedRecords.clear();\n\n    const queries = unique(\n      (\n        await Promise.all(\n          ids.map(async (id) => {\n            return await this.getImportQueries(entityId, \"id\", id);\n          }),\n        )\n      ).flat(),\n    );\n\n    const wdb = BaseModel.getDB(\"w\");\n    for (const query of queries) {\n      const [rsh] = await wdb.raw(query);\n      console.log({\n        query,\n        info: rsh.info,\n      });\n    }\n  }\n\n  async getImportQueries(entityId: string, field: string, id: number): Promise<string[]> {\n    const recordKey = `${entityId}#${field}#${id}`;\n\n    // 순환 참조 방지: 이미 방문한 레코드는 스킵\n    if (this.visitedRecords.has(recordKey)) {\n      return [];\n    }\n    this.visitedRecords.add(recordKey);\n\n    console.log({ entityId, field, id });\n    const entity = EntityManager.get(entityId);\n    const wdb = BaseModel.getDB(\"w\");\n\n    // 여기서 실DB의 row 가져옴\n    const [row] = await wdb(entity.table).where(field, id).limit(1);\n    if (row === undefined) {\n      throw new Error(`${entityId}#${id} row를 찾을 수 없습니다.`);\n    }\n\n    // 픽스쳐DB, 실DB\n    const fixtureDatabase = (Sonamu.dbConfig.fixture_remote.connection as Knex.ConnectionConfig)\n      .database;\n    const realDatabase = (Sonamu.dbConfig.production_master.connection as Knex.ConnectionConfig)\n      .database;\n\n    const selfQuery = `INSERT IGNORE INTO \\`${fixtureDatabase}\\`.\\`${entity.table}\\` (SELECT * FROM \\`${realDatabase}\\`.\\`${entity.table}\\` WHERE \\`id\\` = ${id})`;\n\n    const args = Object.entries(entity.relations)\n      .filter(\n        ([, relation]) =>\n          isBelongsToOneRelationProp(relation) ||\n          (isOneToOneRelationProp(relation) && relation.customJoinClause === undefined),\n      )\n      .map(([, relation]) => {\n        /*\n        BelongsToOne인 경우\n          Category / 'id' / row[category_id] 호출\n        OneToOne에 joinColumn === true 인 경우\n          Profile / 'id' / row[profile_id] 호출\n        OneToOne에 joinColumn === false 인 경우\n          Profile / 'profile_id' / row['id'] 호출\n        */\n        let field: string;\n        let id: number;\n        if (isOneToOneRelationProp(relation) && !relation.hasJoinColumn) {\n          const relatedEntity = EntityManager.get(relation.with);\n          const relatedIdColumnName = relatedEntity.props.find(\n            (p) => isRelationProp(p) && p.with === entity.id,\n          )?.name;\n          if (!relatedIdColumnName) {\n            throw new Error(`${relatedEntity.id}의 ${entity.id} 관계 프롭을 찾을 수 없습니다.`);\n          }\n          field = `${relatedIdColumnName}_id`;\n          id = row.id;\n        } else {\n          field = \"id\";\n          id = row[`${relation.name}_id`];\n        }\n        return {\n          entityId: relation.with,\n          field,\n          id,\n        };\n      })\n      .filter((arg) => arg.id !== null);\n\n    const relQueries = await Promise.all(\n      args.map(async (args) => {\n        return this.getImportQueries(args.entityId, args.field, args.id);\n      }),\n    );\n\n    return [...unique(relQueries.reverse().flat()), selfQuery];\n  }\n\n  async destroy() {\n    if (this._tdb) {\n      await this._tdb.destroy();\n      this._tdb = null;\n    }\n    if (this._fdb) {\n      await this._fdb.destroy();\n      this._fdb = null;\n    }\n    await BaseModel.destroy();\n  }\n\n  async getFixtures(\n    sourceDBName: keyof SonamuDBConfig,\n    targetDBName: keyof SonamuDBConfig,\n    searchOptions: FixtureSearchOptions,\n    duplicateCheck?: DuplicateCheckOptions,\n  ) {\n    const sourceDB = knex(Sonamu.dbConfig[sourceDBName]);\n    const targetDB = knex(Sonamu.dbConfig[targetDBName]);\n\n    const { entityId, field, value, searchType } = searchOptions;\n\n    const entity = EntityManager.get(entityId);\n    const column =\n      entity.props.find((prop) => prop.name === field)?.type === \"relation\" ? `${field}_id` : field;\n\n    let query = sourceDB(entity.table);\n    if (searchType === \"equals\") {\n      query = query.where(column, value);\n    } else if (searchType === \"like\") {\n      query = query.where(column, \"like\", `%${value}%`);\n    }\n\n    const rows = await query;\n    if (rows.length === 0) {\n      throw new Error(\"No records found\");\n    }\n\n    const fixtures: FixtureRecord[] = [];\n    for (const row of rows) {\n      const initialRecordsLength = fixtures.length;\n      const newRecords = await this.createFixtureRecord(entity, row, {\n        _db: sourceDB,\n      });\n      fixtures.push(...newRecords);\n      const currentFixtureRecord = fixtures.find((r) => r.fixtureId === `${entityId}#${row.id}`);\n\n      if (currentFixtureRecord) {\n        // 현재 fixture로부터 생성된 fetchedRecords 설정\n        currentFixtureRecord.fetchedRecords = fixtures\n          .filter((r) => r.fixtureId !== currentFixtureRecord.fixtureId)\n          .slice(initialRecordsLength)\n          .map((r) => r.fixtureId);\n      }\n    }\n\n    for await (const fixture of fixtures) {\n      const entity = EntityManager.get(fixture.entityId);\n\n      // 사용자 지정 컬럼 기준 중복 확인 → target\n      const customColumns = duplicateCheck?.columns?.[fixture.entityId];\n      if (customColumns && customColumns.length > 0) {\n        const customDuplicateRow = await this.checkDuplicateByColumns(\n          targetDB,\n          entity,\n          fixture,\n          customColumns,\n        );\n        if (customDuplicateRow) {\n          const [record] = await this.createFixtureRecord(entity, customDuplicateRow, {\n            singleRecord: true,\n            _db: targetDB,\n          });\n          fixture.target = record;\n        }\n      }\n\n      // Unique index 기준 중복 확인 → fixture.unique\n      const uniqueRow = await this.checkUniqueViolation(targetDB, entity, fixture);\n      if (uniqueRow) {\n        const [record] = await this.createFixtureRecord(entity, uniqueRow, {\n          singleRecord: true,\n          _db: targetDB,\n        });\n        fixture.unique = record;\n      }\n    }\n\n    await targetDB.destroy();\n    await sourceDB.destroy();\n\n    return unique(fixtures, (f) => f.fixtureId);\n  }\n\n  async createFixtureRecord(\n    entity: Entity,\n    row: {\n      id: number;\n      [key: string]: string | number | boolean | null;\n    },\n    options?: {\n      singleRecord?: boolean;\n      _db?: Knex;\n    },\n  ): Promise<FixtureRecord[]> {\n    const records: FixtureRecord[] = [];\n    const visitedEntities = new Set<string>();\n\n    const create = async (\n      entity: Entity,\n      row: {\n        id: number;\n        [key: string]: string | number | boolean | null;\n      },\n    ) => {\n      const fixtureId = `${entity.id}#${row.id}`;\n      if (visitedEntities.has(fixtureId)) {\n        return;\n      }\n      visitedEntities.add(fixtureId);\n\n      const record: FixtureRecord = {\n        fixtureId,\n        entityId: entity.id,\n        id: row.id,\n        columns: {},\n        fetchedRecords: [],\n        belongsRecords: [],\n      };\n\n      for (const prop of entity.props) {\n        if (isVirtualProp(prop)) {\n          continue;\n        }\n\n        record.columns[prop.name] = {\n          prop: prop,\n          value: row[prop.name],\n        };\n\n        const db = options?._db ?? BaseModel.getDB(\"w\");\n        if (isManyToManyRelationProp(prop)) {\n          const relatedEntity = EntityManager.get(prop.with);\n          const throughTable = prop.joinTable;\n          const fromColumn = `${inflection.singularize(entity.table)}_id`;\n          const toColumn = `${inflection.singularize(relatedEntity.table)}_id`;\n\n          const relatedIds = await db(throughTable).where(fromColumn, row.id).pluck(toColumn);\n          record.columns[prop.name].value = relatedIds;\n        } else if (isHasManyRelationProp(prop)) {\n          const relatedEntity = EntityManager.get(prop.with);\n          const relatedIds = await db(relatedEntity.table)\n            .where(prop.joinColumn, row.id)\n            .pluck(\"id\");\n          record.columns[prop.name].value = relatedIds;\n        } else if (isOneToOneRelationProp(prop) && !prop.hasJoinColumn) {\n          const relatedEntity = EntityManager.get(prop.with);\n          const relatedProp = relatedEntity.props.find(\n            (p) => isRelationProp(p) && p.with === entity.id,\n          );\n          if (relatedProp) {\n            const relatedRow = await db(relatedEntity.table).where(\"id\", row.id).first();\n            record.columns[prop.name].value = relatedRow?.id;\n          }\n        } else if (isRelationProp(prop)) {\n          const relatedId = row[`${prop.name}_id`];\n          record.columns[prop.name].value = relatedId;\n          if (relatedId) {\n            record.belongsRecords.push(`${prop.with}#${relatedId}`);\n          }\n          if (!options?.singleRecord && relatedId) {\n            const relatedEntity = EntityManager.get(prop.with);\n            const relatedRow = await db(relatedEntity.table).where(\"id\", relatedId).first();\n            if (relatedRow) {\n              await create(relatedEntity, relatedRow);\n            }\n          }\n        }\n      }\n\n      records.push(record);\n    };\n\n    await create(entity, row);\n\n    return records;\n  }\n\n  /**\n   * 1. RelationGraph로 fixture 단위 삽입 순서 계산 (self-reference 포함)\n   * 2. 순서대로 UpsertBuilder에 등록 (UBRef로 참조 관계 표현)\n   * 3. 테이블별 upsert 실행 (ID는 DB가 자동 할당)\n   */\n  async insertFixtures(\n    dbName: keyof SonamuDBConfig,\n    _fixtures: FixtureRecord[],\n  ): Promise<FixtureImportResult[]> {\n    const fixtures = unique(_fixtures, (f) => f.fixtureId);\n\n    // 초기화\n    this.builder = new UpsertBuilder();\n    this.fixtureRefMap = new Map();\n    this.uuidToFixtureId = new Map();\n    this.skippedFixtures = new Map();\n\n    const db = knex(Sonamu.dbConfig[dbName]);\n    const results: FixtureImportResult[] = [];\n\n    try {\n      // 1. RelationGraph로 fixture 단위 삽입 순서 계산\n      this.relationGraph.buildGraph(fixtures);\n      const insertionOrder = this.relationGraph.getInsertionOrder();\n\n      // 2. 순서대로 UpsertBuilder에 등록 (override 체크)\n      for (const fixtureId of insertionOrder) {\n        const fixture = fixtures.find((f) => f.fixtureId === fixtureId);\n        if (!fixture) continue;\n\n        const hasTarget = !!fixture.target;\n        const hasUnique = !!fixture.unique;\n        const hasDuplicate = hasTarget || hasUnique;\n\n        // 중복이 있고 override=false인 경우: 스킵\n        if (hasDuplicate && !fixture.override) {\n          // 기존 레코드 ID 저장 (unique 우선, 없으면 target)\n          const existingId = fixture.unique?.id ?? fixture.target?.id;\n          assert(existingId);\n          this.skippedFixtures.set(fixtureId, {\n            entityId: fixture.entityId,\n            existingId,\n          });\n\n          console.log(\n            chalk.yellow(\n              `Skipped ${fixture.entityId}#${fixture.id} (existing: #${existingId}, override: false)`,\n            ),\n          );\n          continue;\n        }\n\n        this.registerFixture(fixture);\n        console.log(\n          chalk.blue(\n            `Registered ${fixture.entityId}#${fixture.id}${fixture.override ? ` (override existing: #${fixture.target?.id})` : \"\"}`,\n          ),\n        );\n      }\n\n      // 3. 테이블별 upsert 실행\n      const tableOrder = this.getTableOrder(fixtures);\n\n      await db.transaction(async (trx) => {\n        const insertedIdsByTable = new Map<string, Map<string, number>>();\n\n        for (const tableName of tableOrder) {\n          if (!this.builder.hasTable(tableName)) continue;\n\n          // upsert 실행 전 uuid 목록 저장\n          const table = this.builder.getTable(tableName);\n          const uuids = table.rows.map((row) => row.uuid as string);\n\n          console.log(chalk.blue(`Upserting ${tableName} with ${uuids.length} rows`));\n          await this.builder.upsert(trx, tableName);\n\n          // upsert된 row들의 uuid -> id 매핑 구축\n          if (uuids.length > 0) {\n            const uuidToId = new Map<string, number>();\n            const rows = await trx(tableName as string)\n              .select(\"uuid\", \"id\")\n              .whereIn(\"uuid\", uuids);\n\n            for (const row of rows) {\n              uuidToId.set(row.uuid, row.id);\n            }\n\n            insertedIdsByTable.set(tableName, uuidToId);\n          }\n        }\n\n        // 4. ManyToMany 관계 처리\n        await this.processManyToManyRelations(trx, fixtures, insertedIdsByTable);\n\n        // 5. 결과 수집\n        for (const fixture of fixtures) {\n          const entity = EntityManager.get(fixture.entityId);\n\n          // 스킵된 fixture는 기존 레코드 정보로 결과 추가\n          const skipped = this.skippedFixtures.get(fixture.fixtureId);\n          if (skipped) {\n            results.push({\n              entityId: fixture.entityId,\n              data: await trx(entity.table).where(\"id\", skipped.existingId).first(),\n            });\n            continue;\n          }\n\n          const ref = this.fixtureRefMap.get(fixture.fixtureId);\n          if (ref) {\n            const uuidToId = insertedIdsByTable.get(entity.table);\n            const insertedId = uuidToId?.get(ref.uuid);\n\n            if (insertedId !== undefined) {\n              results.push({\n                entityId: fixture.entityId,\n                data: await trx(entity.table).where(\"id\", insertedId).first(),\n              });\n\n              console.log(\n                chalk.green(`Inserted into ${entity.table}: #${fixture.id} -> #${insertedId}`),\n              );\n            }\n          }\n        }\n      });\n    } finally {\n      await db.destroy();\n    }\n\n    return unique(results, (r) => `${r.entityId}#${r.data.id}`);\n  }\n\n  /**\n   * FixtureRecord를 UpsertBuilder에 등록\n   */\n  private registerFixture(fixture: FixtureRecord): UBRef {\n    const entity = EntityManager.get(fixture.entityId);\n    const row: Record<string, unknown> = {};\n\n    // Override 모드 판단: target 또는 unique가 있고 override=true인 경우\n    const existingRecord = fixture.target ?? fixture.unique;\n    const isOverrideMode = fixture.override && existingRecord;\n\n    for (const [propName, column] of Object.entries(fixture.columns)) {\n      const prop = column.prop;\n\n      if (isVirtualProp(prop)) {\n        continue;\n      }\n\n      // id/uuid 처리: Override 모드일 때만 기존 값 사용\n      if (propName === \"id\" || propName === \"uuid\") {\n        if (isOverrideMode && existingRecord) {\n          // Override: 기존 레코드의 값 사용 → UPDATE\n          row[propName] = existingRecord.columns[propName]?.value;\n        }\n        // 새 레코드: 제외 → INSERT (DB/UpsertBuilder가 생성)\n        continue;\n      }\n\n      if (isRelationProp(prop)) {\n        if (\n          isBelongsToOneRelationProp(prop) ||\n          (isOneToOneRelationProp(prop) && prop.hasJoinColumn)\n        ) {\n          const relatedId = column.value as number | null;\n          if (relatedId !== null && relatedId !== undefined) {\n            const relatedFixtureId = `${prop.with}#${relatedId}`;\n\n            // 먼저 skip된 fixture인지 확인\n            const skippedExistingId = this.skippedFixtures.get(relatedFixtureId)?.existingId;\n            if (skippedExistingId !== undefined) {\n              // skip된 fixture → target DB의 기존 레코드 id 사용\n              row[`${propName}_id`] = skippedExistingId;\n            } else {\n              const relatedRef = this.fixtureRefMap.get(relatedFixtureId);\n              if (relatedRef) {\n                // 이미 등록된 fixture 참조 → UBRef 사용\n                row[`${propName}_id`] = relatedRef;\n              } else {\n                // fixtures에 포함되지 않은 레코드 → ID 그대로 사용\n                row[`${propName}_id`] = relatedId;\n              }\n            }\n          } else {\n            row[`${propName}_id`] = null;\n          }\n        }\n        // HasMany, ManyToMany는 별도 처리\n      } else {\n        // 일반 컬럼\n        row[propName] = this.convertColumnValue(prop as EntityProp, column.value);\n      }\n    }\n\n    console.log(chalk.blue(`Registering ${entity.table} - ${inspect(row, false, null, true)}`));\n    const ref = this.builder.register(entity.table, row);\n    this.fixtureRefMap.set(fixture.fixtureId, ref);\n    this.uuidToFixtureId.set(ref.uuid, fixture.fixtureId);\n\n    return ref;\n  }\n\n  /**\n   * 컬럼 값 변환\n   */\n  private convertColumnValue(prop: EntityProp, value: unknown): unknown {\n    if (value === null || value === undefined) {\n      return null;\n    }\n\n    switch (prop.type) {\n      case \"json\":\n        // UpsertBuilder.register에서 JSON.stringify 처리하므로 object 그대로 전달\n        return value;\n\n      case \"date\":\n        if (typeof value === \"string\" || typeof value === \"number\") {\n          return new Date(value);\n        }\n        return value;\n\n      default:\n        return value;\n    }\n  }\n\n  /**\n   * 테이블 순서 추출 (fixtures에 포함된 테이블만)\n   */\n  private getTableOrder(fixtures: FixtureRecord[]): (keyof DatabaseSchemaExtend)[] {\n    const tables: string[] = [];\n    const seen = new Set<string>();\n\n    for (const fixture of fixtures) {\n      const entity = EntityManager.get(fixture.entityId);\n      if (!seen.has(entity.table)) {\n        seen.add(entity.table);\n        tables.push(entity.table);\n      }\n    }\n\n    return tables as (keyof DatabaseSchemaExtend)[];\n  }\n\n  private async processManyToManyRelations(\n    trx: Knex.Transaction,\n    fixtures: FixtureRecord[],\n    insertedIdsByTable: Map<string, Map<string, number>>,\n  ): Promise<void> {\n    for (const fixture of fixtures) {\n      const entity = EntityManager.get(fixture.entityId);\n      const sourceRef = this.fixtureRefMap.get(fixture.fixtureId);\n\n      if (!sourceRef) continue;\n\n      const sourceUuidToId = insertedIdsByTable.get(entity.table);\n      const sourceId = sourceUuidToId?.get(sourceRef.uuid);\n\n      if (sourceId === undefined) continue;\n\n      for (const [, column] of Object.entries(fixture.columns)) {\n        const prop = column.prop;\n\n        if (isManyToManyRelationProp(prop) && Array.isArray(column.value)) {\n          // 선택되지 않은 ManyToMany 관계는 저장하지 않음\n          const targetTable = EntityManager.get(prop.with);\n          if (this.builder.hasTable(targetTable.table) === false) continue;\n\n          const relatedIds = column.value as number[];\n          if (relatedIds.length === 0) continue;\n\n          const joinTable = (prop as ManyToManyRelationProp).joinTable;\n          const relatedEntity = EntityManager.get(prop.with);\n\n          const sourceColumn = `${inflection.singularize(entity.table)}_id`;\n          const targetColumn = `${inflection.singularize(relatedEntity.table)}_id`;\n\n          for (const relatedId of relatedIds) {\n            const relatedFixtureId = `${prop.with}#${relatedId}`;\n            const relatedRef = this.fixtureRefMap.get(relatedFixtureId);\n\n            let targetId: number;\n\n            if (relatedRef) {\n              const relatedUuidToId = insertedIdsByTable.get(relatedEntity.table);\n              const resolvedId = relatedUuidToId?.get(relatedRef.uuid);\n\n              if (resolvedId === undefined) {\n                console.warn(\n                  `Related fixture ${relatedFixtureId} not found in insertedIds, skipping`,\n                );\n                continue;\n              }\n              targetId = resolvedId;\n            } else {\n              targetId = relatedId;\n            }\n\n            // JoinTable에 삽입\n            const [found] = await trx(joinTable)\n              .where({\n                [sourceColumn]: sourceId,\n                [targetColumn]: targetId,\n              })\n              .limit(1);\n\n            if (!found) {\n              await trx(joinTable).insert({\n                [sourceColumn]: sourceId,\n                [targetColumn]: targetId,\n              });\n\n              console.log(\n                chalk.green(\n                  `Inserted into ${joinTable}: ${entity.table}(${sourceId}) - ${relatedEntity.table}(${targetId})`,\n                ),\n              );\n            }\n          }\n        }\n      }\n    }\n  }\n\n  private async checkUniqueViolation(db: Knex, entity: Entity, fixture: FixtureRecord) {\n    const _uniqueIndexes = entity.indexes?.filter((i) => i.type === \"unique\") ?? [];\n\n    const uniqueIndexes = _uniqueIndexes.filter((index) =>\n      index.columns.every((column) => !column.name.startsWith(`${entity.table}__`)),\n    );\n    if (uniqueIndexes.length === 0) {\n      return null;\n    }\n\n    let uniqueQuery = db(entity.table);\n    let hasCondition = false;\n\n    for (const index of uniqueIndexes) {\n      // 컬럼 중 하나라도 null이면 유니크 제약을 위반하지 않기 때문에 해당 인덱스는 무시\n      const containsNull = index.columns.some((column) => {\n        const field = column.name.replace(/_id$/, \"\");\n        return fixture.columns[field]?.value === null;\n      });\n      if (containsNull) {\n        continue;\n      }\n\n      uniqueQuery = uniqueQuery.orWhere((qb) => {\n        for (const column of index.columns) {\n          const field = column.name.replace(/_id$/, \"\");\n\n          if (Array.isArray(fixture.columns[field]?.value)) {\n            qb.whereIn(column.name, fixture.columns[field].value);\n          } else {\n            qb.andWhere(column.name, fixture.columns[field]?.value);\n          }\n        }\n      });\n      hasCondition = true;\n    }\n\n    if (!hasCondition) {\n      return null;\n    }\n\n    const [uniqueFound] = await uniqueQuery;\n    return uniqueFound;\n  }\n\n  private async checkDuplicateByColumns(\n    db: Knex,\n    entity: Entity,\n    fixture: FixtureRecord,\n    columns: string[],\n  ) {\n    if (columns.length === 0) {\n      return null;\n    }\n\n    const whereClause: Record<string, unknown> = {};\n\n    for (const column of columns) {\n      // relation 필드인 경우 _id 붙이기\n      const prop = entity.props.find((p) => p.name === column);\n      const dbColumn = prop && isRelationProp(prop) ? `${column}_id` : column;\n      const value = fixture.columns[column]?.value;\n\n      // null 값이 포함된 경우 중복 확인 스킵\n      if (value === null || value === undefined) {\n        return null;\n      }\n\n      whereClause[dbColumn] = value;\n    }\n\n    const [found] = await db(entity.table).where(whereClause).limit(1);\n    return found;\n  }\n\n  async addFixtureLoader(code: string) {\n    const path = `${Sonamu.apiRootPath}/src/testing/fixture.ts`;\n    const content = readFileSync(path).toString();\n\n    const fixtureLoaderStart = content.indexOf(\"const fixtureLoader = {\");\n    const fixtureLoaderEnd = content.indexOf(\"};\", fixtureLoaderStart);\n\n    if (fixtureLoaderStart !== -1 && fixtureLoaderEnd !== -1) {\n      const newContent = `${content.slice(0, fixtureLoaderEnd)}  ${code}\\n${content.slice(fixtureLoaderEnd)}`;\n\n      writeFileSync(path, newContent);\n    } else {\n      throw new Error(\"Failed to find fixtureLoader in fixture.ts\");\n    }\n  }\n}\n\nexport const FixtureManager = new FixtureManagerClass();\n"],"names":["assert","chalk","execSync","readFileSync","writeFileSync","inflection","knex","unique","inspect","Sonamu","BaseModel","UpsertBuilder","EntityManager","isBelongsToOneRelationProp","isHasManyRelationProp","isManyToManyRelationProp","isOneToOneRelationProp","isRelationProp","isVirtualProp","RelationGraph","FixtureManagerClass","_tdb","tdb","Error","_fdb","fdb","cachedTableNames","relationGraph","builder","fixtureRefMap","Map","uuidToFixtureId","skippedFixtures","init","dbConfig","test","production_master","tConn","connection","pConn","host","port","database","fixture_remote","getChecksum","db","tableName","checksumRow","raw","Checksum","sync","fixtureConn","testConn","testPgEnv","PGPASSWORD","password","user","stdio","env","process","fixturePgEnv","dumpCmd","restoreCmd","shell","visitedRecords","Set","importFixture","entityId","ids","clear","queries","Promise","all","map","id","getImportQueries","flat","wdb","getDB","query","rsh","console","log","info","field","recordKey","has","add","entity","get","row","table","where","limit","undefined","fixtureDatabase","realDatabase","selfQuery","args","Object","entries","relations","filter","relation","customJoinClause","hasJoinColumn","relatedEntity","with","relatedIdColumnName","props","find","p","name","arg","relQueries","reverse","destroy","getFixtures","sourceDBName","targetDBName","searchOptions","duplicateCheck","sourceDB","targetDB","value","searchType","column","prop","type","rows","length","fixtures","initialRecordsLength","newRecords","createFixtureRecord","_db","push","currentFixtureRecord","r","fixtureId","fetchedRecords","slice","fixture","customColumns","columns","customDuplicateRow","checkDuplicateByColumns","record","singleRecord","target","uniqueRow","checkUniqueViolation","f","options","records","visitedEntities","create","belongsRecords","throughTable","joinTable","fromColumn","singularize","toColumn","relatedIds","pluck","joinColumn","relatedProp","relatedRow","first","relatedId","insertFixtures","dbName","_fixtures","results","buildGraph","insertionOrder","getInsertionOrder","hasTarget","hasUnique","hasDuplicate","override","existingId","set","yellow","registerFixture","blue","tableOrder","getTableOrder","transaction","trx","insertedIdsByTable","hasTable","getTable","uuids","uuid","upsert","uuidToId","select","whereIn","processManyToManyRelations","skipped","data","ref","insertedId","green","existingRecord","isOverrideMode","propName","relatedFixtureId","skippedExistingId","relatedRef","convertColumnValue","register","Date","tables","seen","sourceRef","sourceUuidToId","sourceId","Array","isArray","targetTable","sourceColumn","targetColumn","targetId","relatedUuidToId","resolvedId","warn","found","insert","_uniqueIndexes","indexes","i","uniqueIndexes","index","every","startsWith","uniqueQuery","hasCondition","containsNull","some","replace","orWhere","qb","andWhere","uniqueFound","whereClause","dbColumn","addFixtureLoader","code","path","apiRootPath","content","toString","fixtureLoaderStart","indexOf","fixtureLoaderEnd","newContent","FixtureManager"],"mappings":"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"}
623
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/testing/fixture-manager.ts"],"sourcesContent":["import assert from \"assert\";\nimport chalk from \"chalk\";\nimport { execSync } from \"child_process\";\nimport { readFileSync, writeFileSync } from \"fs\";\nimport inflection from \"inflection\";\nimport knex, { type Knex } from \"knex\";\nimport { unique } from \"radashi\";\nimport { inspect } from \"util\";\nimport { Sonamu } from \"../api\";\nimport { BaseModel } from \"../database/base-model\";\nimport type { SonamuDBConfig } from \"../database/db\";\nimport { type UBRef, UpsertBuilder } from \"../database/upsert-builder\";\nimport type { Entity } from \"../entity/entity\";\nimport { EntityManager } from \"../entity/entity-manager\";\nimport {\n  type DatabaseSchemaExtend,\n  type EntityProp,\n  type FixtureImportResult,\n  type FixtureRecord,\n  type FixtureSearchOptions,\n  isBelongsToOneRelationProp,\n  isHasManyRelationProp,\n  isManyToManyRelationProp,\n  isOneToOneRelationProp,\n  isRelationProp,\n  isVirtualProp,\n  type ManyToManyRelationProp,\n} from \"../types/types\";\nimport { RelationGraph } from \"./_relation-graph\";\n\n/** 사용자 지정 중복 확인 컬럼 (entityId별로 지정) */\nexport interface DuplicateCheckOptions {\n  columns?: {\n    [entityId: string]: string[];\n  };\n}\n\nexport class FixtureManagerClass {\n  private _tdb: Knex | null = null;\n  set tdb(tdb: Knex) {\n    this._tdb = tdb;\n  }\n  get tdb(): Knex {\n    if (this._tdb === null) {\n      throw new Error(\"FixtureManager has not been initialized\");\n    }\n    return this._tdb;\n  }\n\n  private _fdb: Knex | null = null;\n  set fdb(fdb: Knex) {\n    this._fdb = fdb;\n  }\n  get fdb(): Knex {\n    if (this._fdb === null) {\n      throw new Error(\"FixtureManager has not been initialized\");\n    }\n    return this._fdb;\n  }\n  cachedTableNames: string[] | null = null;\n\n  private relationGraph = new RelationGraph();\n\n  // UpsertBuilder 기반 import를 위한 상태\n  private builder: UpsertBuilder = new UpsertBuilder();\n  private fixtureRefMap: Map<string, UBRef> = new Map();\n  private uuidToFixtureId: Map<string, string> = new Map();\n  private skippedFixtures: Map<string, { entityId: string; existingId: number }> = new Map();\n\n  init() {\n    if (this._tdb !== null) {\n      return;\n    }\n    if (Sonamu.dbConfig.test && Sonamu.dbConfig.production_master) {\n      const tConn = Sonamu.dbConfig.test.connection as Knex.ConnectionConfig & {\n        port?: number;\n      };\n      const pConn = Sonamu.dbConfig.production_master.connection as Knex.ConnectionConfig & {\n        port?: number;\n      };\n      if (\n        `${tConn.host ?? \"localhost\"}:${tConn.port ?? 5432}/${tConn.database}` ===\n        `${pConn.host ?? \"localhost\"}:${pConn.port ?? 5432}/${pConn.database}`\n      ) {\n        throw new Error(`테스트DB와 프로덕션DB에 동일한 데이터베이스가 사용되었습니다.`);\n      }\n    }\n\n    this.tdb = knex(Sonamu.dbConfig.test);\n    this.fdb = knex(Sonamu.dbConfig.fixture);\n  }\n\n  async getChecksum(db: Knex, tableName: string) {\n    const [[checksumRow]] = await db.raw(`CHECKSUM TABLE ${tableName}`);\n    return checksumRow.Checksum;\n  }\n\n  /**\n    원격 fixture DB를 로컬 test DB로 복사합니다.\n    pg_dump로 원격 DB를 덤프하고, pg_restore로 로컬에 복원합니다.\n  */\n  async sync() {\n    const fixtureConn = Sonamu.dbConfig.fixture.connection as Knex.PgConnectionConfig;\n    const testConn = Sonamu.dbConfig.test.connection as Knex.PgConnectionConfig;\n\n    // 1. 로컬 test DB 연결 종료 및 재생성\n    const testPgEnv = { PGPASSWORD: testConn.password || \"\" };\n    execSync(\n      `psql -h ${testConn.host} -p ${testConn.port ?? 5432} -U ${testConn.user} -d postgres -c \"\n        SELECT pg_terminate_backend(pg_stat_activity.pid)\n        FROM pg_stat_activity\n        WHERE datname = '${testConn.database}'\n          AND pid <> pg_backend_pid();\n      \"`,\n      { stdio: \"inherit\", env: { ...process.env, ...testPgEnv } as NodeJS.ProcessEnv },\n    );\n\n    execSync(\n      `psql -h ${testConn.host} -p ${testConn.port ?? 5432} -U ${testConn.user} -d postgres -c \"DROP DATABASE IF EXISTS \\\\\"${testConn.database}\\\\\"\"`,\n      { stdio: \"inherit\", env: { ...process.env, ...testPgEnv } as NodeJS.ProcessEnv },\n    );\n\n    execSync(\n      `psql -h ${testConn.host} -p ${testConn.port ?? 5432} -U ${testConn.user} -d postgres -c \"CREATE DATABASE \\\\\"${testConn.database}\\\\\"\"`,\n      { stdio: \"inherit\", env: { ...process.env, ...testPgEnv } as NodeJS.ProcessEnv },\n    );\n\n    // 2. 원격 fixture DB → 로컬 test DB로 복사 (pg_dump | pg_restore)\n    const fixturePgEnv = { PGPASSWORD: fixtureConn.password || \"\" };\n    const dumpCmd = `pg_dump -h ${fixtureConn.host} -p ${fixtureConn.port ?? 5432} -U ${fixtureConn.user} -d ${fixtureConn.database} -Fc`;\n    const restoreCmd = `pg_restore -h ${testConn.host} -p ${testConn.port ?? 5432} -U ${testConn.user} -d ${testConn.database} --no-owner --no-acl`;\n\n    execSync(`${dumpCmd} | PGPASSWORD=\"${testConn.password || \"\"}\" ${restoreCmd}`, {\n      stdio: \"inherit\",\n      env: { ...process.env, ...fixturePgEnv } as NodeJS.ProcessEnv,\n      shell: \"/bin/bash\",\n    });\n  }\n\n  private visitedRecords = new Set<string>();\n  async importFixture(entityId: string, ids: number[]) {\n    // 방문 기록 초기화 (새로운 import 작업 시작)\n    this.visitedRecords.clear();\n\n    const queries = unique(\n      (\n        await Promise.all(\n          ids.map(async (id) => {\n            return await this.getImportQueries(entityId, \"id\", id);\n          }),\n        )\n      ).flat(),\n    );\n\n    const wdb = BaseModel.getDB(\"w\");\n    for (const query of queries) {\n      const [rsh] = await wdb.raw(query);\n      console.log({\n        query,\n        info: rsh.info,\n      });\n    }\n  }\n\n  async getImportQueries(entityId: string, field: string, id: number): Promise<string[]> {\n    const recordKey = `${entityId}#${field}#${id}`;\n\n    // 순환 참조 방지: 이미 방문한 레코드는 스킵\n    if (this.visitedRecords.has(recordKey)) {\n      return [];\n    }\n    this.visitedRecords.add(recordKey);\n\n    console.log({ entityId, field, id });\n    const entity = EntityManager.get(entityId);\n    const wdb = BaseModel.getDB(\"w\");\n\n    // 여기서 실DB의 row 가져옴\n    const [row] = await wdb(entity.table).where(field, id).limit(1);\n    if (row === undefined) {\n      throw new Error(`${entityId}#${id} row를 찾을 수 없습니다.`);\n    }\n\n    // 픽스쳐DB, 실DB\n    const fixtureDatabase = (Sonamu.dbConfig.fixture.connection as Knex.ConnectionConfig).database;\n    const realDatabase = (Sonamu.dbConfig.production_master.connection as Knex.ConnectionConfig)\n      .database;\n\n    const selfQuery = `INSERT IGNORE INTO \\`${fixtureDatabase}\\`.\\`${entity.table}\\` (SELECT * FROM \\`${realDatabase}\\`.\\`${entity.table}\\` WHERE \\`id\\` = ${id})`;\n\n    const args = Object.entries(entity.relations)\n      .filter(\n        ([, relation]) =>\n          isBelongsToOneRelationProp(relation) ||\n          (isOneToOneRelationProp(relation) && relation.customJoinClause === undefined),\n      )\n      .map(([, relation]) => {\n        /*\n        BelongsToOne인 경우\n          Category / 'id' / row[category_id] 호출\n        OneToOne에 joinColumn === true 인 경우\n          Profile / 'id' / row[profile_id] 호출\n        OneToOne에 joinColumn === false 인 경우\n          Profile / 'profile_id' / row['id'] 호출\n        */\n        let field: string;\n        let id: number;\n        if (isOneToOneRelationProp(relation) && !relation.hasJoinColumn) {\n          const relatedEntity = EntityManager.get(relation.with);\n          const relatedIdColumnName = relatedEntity.props.find(\n            (p) => isRelationProp(p) && p.with === entity.id,\n          )?.name;\n          if (!relatedIdColumnName) {\n            throw new Error(`${relatedEntity.id}의 ${entity.id} 관계 프롭을 찾을 수 없습니다.`);\n          }\n          field = `${relatedIdColumnName}_id`;\n          id = row.id;\n        } else {\n          field = \"id\";\n          id = row[`${relation.name}_id`];\n        }\n        return {\n          entityId: relation.with,\n          field,\n          id,\n        };\n      })\n      .filter((arg) => arg.id !== null);\n\n    const relQueries = await Promise.all(\n      args.map(async (args) => {\n        return this.getImportQueries(args.entityId, args.field, args.id);\n      }),\n    );\n\n    return [...unique(relQueries.reverse().flat()), selfQuery];\n  }\n\n  async destroy() {\n    if (this._tdb) {\n      await this._tdb.destroy();\n      this._tdb = null;\n    }\n    if (this._fdb) {\n      await this._fdb.destroy();\n      this._fdb = null;\n    }\n    await BaseModel.destroy();\n  }\n\n  async getFixtures(\n    sourceDBName: keyof SonamuDBConfig,\n    targetDBName: keyof SonamuDBConfig,\n    searchOptions: FixtureSearchOptions,\n    duplicateCheck?: DuplicateCheckOptions,\n  ) {\n    const sourceDB = knex(Sonamu.dbConfig[sourceDBName]);\n    const targetDB = knex(Sonamu.dbConfig[targetDBName]);\n\n    const { entityId, field, value, searchType } = searchOptions;\n\n    const entity = EntityManager.get(entityId);\n    const column =\n      entity.props.find((prop) => prop.name === field)?.type === \"relation\" ? `${field}_id` : field;\n\n    let query = sourceDB(entity.table);\n    if (searchType === \"equals\") {\n      query = query.where(column, value);\n    } else if (searchType === \"like\") {\n      query = query.where(column, \"like\", `%${value}%`);\n    }\n\n    const rows = await query;\n    if (rows.length === 0) {\n      throw new Error(\"No records found\");\n    }\n\n    const fixtures: FixtureRecord[] = [];\n    for (const row of rows) {\n      const initialRecordsLength = fixtures.length;\n      const newRecords = await this.createFixtureRecord(entity, row, {\n        _db: sourceDB,\n      });\n      fixtures.push(...newRecords);\n      const currentFixtureRecord = fixtures.find((r) => r.fixtureId === `${entityId}#${row.id}`);\n\n      if (currentFixtureRecord) {\n        // 현재 fixture로부터 생성된 fetchedRecords 설정\n        currentFixtureRecord.fetchedRecords = fixtures\n          .filter((r) => r.fixtureId !== currentFixtureRecord.fixtureId)\n          .slice(initialRecordsLength)\n          .map((r) => r.fixtureId);\n      }\n    }\n\n    for await (const fixture of fixtures) {\n      const entity = EntityManager.get(fixture.entityId);\n\n      // 사용자 지정 컬럼 기준 중복 확인 → target\n      const customColumns = duplicateCheck?.columns?.[fixture.entityId];\n      if (customColumns && customColumns.length > 0) {\n        const customDuplicateRow = await this.checkDuplicateByColumns(\n          targetDB,\n          entity,\n          fixture,\n          customColumns,\n        );\n        if (customDuplicateRow) {\n          const [record] = await this.createFixtureRecord(entity, customDuplicateRow, {\n            singleRecord: true,\n            _db: targetDB,\n          });\n          fixture.target = record;\n        }\n      }\n\n      // Unique index 기준 중복 확인 → fixture.unique\n      const uniqueRow = await this.checkUniqueViolation(targetDB, entity, fixture);\n      if (uniqueRow) {\n        const [record] = await this.createFixtureRecord(entity, uniqueRow, {\n          singleRecord: true,\n          _db: targetDB,\n        });\n        fixture.unique = record;\n      }\n    }\n\n    await targetDB.destroy();\n    await sourceDB.destroy();\n\n    return unique(fixtures, (f) => f.fixtureId);\n  }\n\n  async createFixtureRecord(\n    entity: Entity,\n    row: {\n      id: number;\n      [key: string]: string | number | boolean | null;\n    },\n    options?: {\n      singleRecord?: boolean;\n      _db?: Knex;\n    },\n  ): Promise<FixtureRecord[]> {\n    const records: FixtureRecord[] = [];\n    const visitedEntities = new Set<string>();\n\n    const create = async (\n      entity: Entity,\n      row: {\n        id: number;\n        [key: string]: string | number | boolean | null;\n      },\n    ) => {\n      const fixtureId = `${entity.id}#${row.id}`;\n      if (visitedEntities.has(fixtureId)) {\n        return;\n      }\n      visitedEntities.add(fixtureId);\n\n      const record: FixtureRecord = {\n        fixtureId,\n        entityId: entity.id,\n        id: row.id,\n        columns: {},\n        fetchedRecords: [],\n        belongsRecords: [],\n      };\n\n      for (const prop of entity.props) {\n        if (isVirtualProp(prop)) {\n          continue;\n        }\n\n        record.columns[prop.name] = {\n          prop: prop,\n          value: row[prop.name],\n        };\n\n        const db = options?._db ?? BaseModel.getDB(\"w\");\n        if (isManyToManyRelationProp(prop)) {\n          const relatedEntity = EntityManager.get(prop.with);\n          const throughTable = prop.joinTable;\n          const fromColumn = `${inflection.singularize(entity.table)}_id`;\n          const toColumn = `${inflection.singularize(relatedEntity.table)}_id`;\n\n          const relatedIds = await db(throughTable).where(fromColumn, row.id).pluck(toColumn);\n          record.columns[prop.name].value = relatedIds;\n        } else if (isHasManyRelationProp(prop)) {\n          const relatedEntity = EntityManager.get(prop.with);\n          const relatedIds = await db(relatedEntity.table)\n            .where(prop.joinColumn, row.id)\n            .pluck(\"id\");\n          record.columns[prop.name].value = relatedIds;\n        } else if (isOneToOneRelationProp(prop) && !prop.hasJoinColumn) {\n          const relatedEntity = EntityManager.get(prop.with);\n          const relatedProp = relatedEntity.props.find(\n            (p) => isRelationProp(p) && p.with === entity.id,\n          );\n          if (relatedProp) {\n            const relatedRow = await db(relatedEntity.table).where(\"id\", row.id).first();\n            record.columns[prop.name].value = relatedRow?.id;\n          }\n        } else if (isRelationProp(prop)) {\n          const relatedId = row[`${prop.name}_id`];\n          record.columns[prop.name].value = relatedId;\n          if (relatedId) {\n            record.belongsRecords.push(`${prop.with}#${relatedId}`);\n          }\n          if (!options?.singleRecord && relatedId) {\n            const relatedEntity = EntityManager.get(prop.with);\n            const relatedRow = await db(relatedEntity.table).where(\"id\", relatedId).first();\n            if (relatedRow) {\n              await create(relatedEntity, relatedRow);\n            }\n          }\n        }\n      }\n\n      records.push(record);\n    };\n\n    await create(entity, row);\n\n    return records;\n  }\n\n  /**\n   * 1. RelationGraph로 fixture 단위 삽입 순서 계산 (self-reference 포함)\n   * 2. 순서대로 UpsertBuilder에 등록 (UBRef로 참조 관계 표현)\n   * 3. 테이블별 upsert 실행 (ID는 DB가 자동 할당)\n   */\n  async insertFixtures(\n    dbName: keyof SonamuDBConfig,\n    _fixtures: FixtureRecord[],\n  ): Promise<FixtureImportResult[]> {\n    const fixtures = unique(_fixtures, (f) => f.fixtureId);\n\n    // 초기화\n    this.builder = new UpsertBuilder();\n    this.fixtureRefMap = new Map();\n    this.uuidToFixtureId = new Map();\n    this.skippedFixtures = new Map();\n\n    const db = knex(Sonamu.dbConfig[dbName]);\n    const results: FixtureImportResult[] = [];\n\n    try {\n      // 1. RelationGraph로 fixture 단위 삽입 순서 계산\n      this.relationGraph.buildGraph(fixtures);\n      const insertionOrder = this.relationGraph.getInsertionOrder();\n\n      // 2. 순서대로 UpsertBuilder에 등록 (override 체크)\n      for (const fixtureId of insertionOrder) {\n        const fixture = fixtures.find((f) => f.fixtureId === fixtureId);\n        if (!fixture) continue;\n\n        const hasTarget = !!fixture.target;\n        const hasUnique = !!fixture.unique;\n        const hasDuplicate = hasTarget || hasUnique;\n\n        // 중복이 있고 override=false인 경우: 스킵\n        if (hasDuplicate && !fixture.override) {\n          // 기존 레코드 ID 저장 (unique 우선, 없으면 target)\n          const existingId = fixture.unique?.id ?? fixture.target?.id;\n          assert(existingId);\n          this.skippedFixtures.set(fixtureId, {\n            entityId: fixture.entityId,\n            existingId,\n          });\n\n          console.log(\n            chalk.yellow(\n              `Skipped ${fixture.entityId}#${fixture.id} (existing: #${existingId}, override: false)`,\n            ),\n          );\n          continue;\n        }\n\n        this.registerFixture(fixture);\n        console.log(\n          chalk.blue(\n            `Registered ${fixture.entityId}#${fixture.id}${fixture.override ? ` (override existing: #${fixture.target?.id})` : \"\"}`,\n          ),\n        );\n      }\n\n      // 3. 테이블별 upsert 실행\n      const tableOrder = this.getTableOrder(fixtures);\n\n      await db.transaction(async (trx) => {\n        const insertedIdsByTable = new Map<string, Map<string, number>>();\n\n        for (const tableName of tableOrder) {\n          if (!this.builder.hasTable(tableName)) continue;\n\n          // upsert 실행 전 uuid 목록 저장\n          const table = this.builder.getTable(tableName);\n          const uuids = table.rows.map((row) => row.uuid as string);\n\n          console.log(chalk.blue(`Upserting ${tableName} with ${uuids.length} rows`));\n          await this.builder.upsert(trx, tableName);\n\n          // upsert된 row들의 uuid -> id 매핑 구축\n          if (uuids.length > 0) {\n            const uuidToId = new Map<string, number>();\n            const rows = await trx(tableName as string)\n              .select(\"uuid\", \"id\")\n              .whereIn(\"uuid\", uuids);\n\n            for (const row of rows) {\n              uuidToId.set(row.uuid, row.id);\n            }\n\n            insertedIdsByTable.set(tableName, uuidToId);\n          }\n        }\n\n        // 4. ManyToMany 관계 처리\n        await this.processManyToManyRelations(trx, fixtures, insertedIdsByTable);\n\n        // 5. 결과 수집\n        for (const fixture of fixtures) {\n          const entity = EntityManager.get(fixture.entityId);\n\n          // 스킵된 fixture는 기존 레코드 정보로 결과 추가\n          const skipped = this.skippedFixtures.get(fixture.fixtureId);\n          if (skipped) {\n            results.push({\n              entityId: fixture.entityId,\n              data: await trx(entity.table).where(\"id\", skipped.existingId).first(),\n            });\n            continue;\n          }\n\n          const ref = this.fixtureRefMap.get(fixture.fixtureId);\n          if (ref) {\n            const uuidToId = insertedIdsByTable.get(entity.table);\n            const insertedId = uuidToId?.get(ref.uuid);\n\n            if (insertedId !== undefined) {\n              results.push({\n                entityId: fixture.entityId,\n                data: await trx(entity.table).where(\"id\", insertedId).first(),\n              });\n\n              console.log(\n                chalk.green(`Inserted into ${entity.table}: #${fixture.id} -> #${insertedId}`),\n              );\n            }\n          }\n        }\n      });\n    } finally {\n      await db.destroy();\n    }\n\n    return unique(results, (r) => `${r.entityId}#${r.data.id}`);\n  }\n\n  /**\n   * FixtureRecord를 UpsertBuilder에 등록\n   */\n  private registerFixture(fixture: FixtureRecord): UBRef {\n    const entity = EntityManager.get(fixture.entityId);\n    const row: Record<string, unknown> = {};\n\n    // Override 모드 판단: target 또는 unique가 있고 override=true인 경우\n    const existingRecord = fixture.target ?? fixture.unique;\n    const isOverrideMode = fixture.override && existingRecord;\n\n    for (const [propName, column] of Object.entries(fixture.columns)) {\n      const prop = column.prop;\n\n      if (isVirtualProp(prop)) {\n        continue;\n      }\n\n      // id/uuid 처리: Override 모드일 때만 기존 값 사용\n      if (propName === \"id\" || propName === \"uuid\") {\n        if (isOverrideMode && existingRecord) {\n          // Override: 기존 레코드의 값 사용 → UPDATE\n          row[propName] = existingRecord.columns[propName]?.value;\n        }\n        // 새 레코드: 제외 → INSERT (DB/UpsertBuilder가 생성)\n        continue;\n      }\n\n      if (isRelationProp(prop)) {\n        if (\n          isBelongsToOneRelationProp(prop) ||\n          (isOneToOneRelationProp(prop) && prop.hasJoinColumn)\n        ) {\n          const relatedId = column.value as number | null;\n          if (relatedId !== null && relatedId !== undefined) {\n            const relatedFixtureId = `${prop.with}#${relatedId}`;\n\n            // 먼저 skip된 fixture인지 확인\n            const skippedExistingId = this.skippedFixtures.get(relatedFixtureId)?.existingId;\n            if (skippedExistingId !== undefined) {\n              // skip된 fixture → target DB의 기존 레코드 id 사용\n              row[`${propName}_id`] = skippedExistingId;\n            } else {\n              const relatedRef = this.fixtureRefMap.get(relatedFixtureId);\n              if (relatedRef) {\n                // 이미 등록된 fixture 참조 → UBRef 사용\n                row[`${propName}_id`] = relatedRef;\n              } else {\n                // fixtures에 포함되지 않은 레코드 → ID 그대로 사용\n                row[`${propName}_id`] = relatedId;\n              }\n            }\n          } else {\n            row[`${propName}_id`] = null;\n          }\n        }\n        // HasMany, ManyToMany는 별도 처리\n      } else {\n        // 일반 컬럼\n        row[propName] = this.convertColumnValue(prop as EntityProp, column.value);\n      }\n    }\n\n    console.log(chalk.blue(`Registering ${entity.table} - ${inspect(row, false, null, true)}`));\n    const ref = this.builder.register(entity.table, row);\n    this.fixtureRefMap.set(fixture.fixtureId, ref);\n    this.uuidToFixtureId.set(ref.uuid, fixture.fixtureId);\n\n    return ref;\n  }\n\n  /**\n   * 컬럼 값 변환\n   */\n  private convertColumnValue(prop: EntityProp, value: unknown): unknown {\n    if (value === null || value === undefined) {\n      return null;\n    }\n\n    switch (prop.type) {\n      case \"json\":\n        // UpsertBuilder.register에서 JSON.stringify 처리하므로 object 그대로 전달\n        return value;\n\n      case \"date\":\n        if (typeof value === \"string\" || typeof value === \"number\") {\n          return new Date(value);\n        }\n        return value;\n\n      default:\n        return value;\n    }\n  }\n\n  /**\n   * 테이블 순서 추출 (fixtures에 포함된 테이블만)\n   */\n  private getTableOrder(fixtures: FixtureRecord[]): (keyof DatabaseSchemaExtend)[] {\n    const tables: string[] = [];\n    const seen = new Set<string>();\n\n    for (const fixture of fixtures) {\n      const entity = EntityManager.get(fixture.entityId);\n      if (!seen.has(entity.table)) {\n        seen.add(entity.table);\n        tables.push(entity.table);\n      }\n    }\n\n    return tables as (keyof DatabaseSchemaExtend)[];\n  }\n\n  private async processManyToManyRelations(\n    trx: Knex.Transaction,\n    fixtures: FixtureRecord[],\n    insertedIdsByTable: Map<string, Map<string, number>>,\n  ): Promise<void> {\n    for (const fixture of fixtures) {\n      const entity = EntityManager.get(fixture.entityId);\n      const sourceRef = this.fixtureRefMap.get(fixture.fixtureId);\n\n      if (!sourceRef) continue;\n\n      const sourceUuidToId = insertedIdsByTable.get(entity.table);\n      const sourceId = sourceUuidToId?.get(sourceRef.uuid);\n\n      if (sourceId === undefined) continue;\n\n      for (const [, column] of Object.entries(fixture.columns)) {\n        const prop = column.prop;\n\n        if (isManyToManyRelationProp(prop) && Array.isArray(column.value)) {\n          // 선택되지 않은 ManyToMany 관계는 저장하지 않음\n          const targetTable = EntityManager.get(prop.with);\n          if (this.builder.hasTable(targetTable.table) === false) continue;\n\n          const relatedIds = column.value as number[];\n          if (relatedIds.length === 0) continue;\n\n          const joinTable = (prop as ManyToManyRelationProp).joinTable;\n          const relatedEntity = EntityManager.get(prop.with);\n\n          const sourceColumn = `${inflection.singularize(entity.table)}_id`;\n          const targetColumn = `${inflection.singularize(relatedEntity.table)}_id`;\n\n          for (const relatedId of relatedIds) {\n            const relatedFixtureId = `${prop.with}#${relatedId}`;\n            const relatedRef = this.fixtureRefMap.get(relatedFixtureId);\n\n            let targetId: number;\n\n            if (relatedRef) {\n              const relatedUuidToId = insertedIdsByTable.get(relatedEntity.table);\n              const resolvedId = relatedUuidToId?.get(relatedRef.uuid);\n\n              if (resolvedId === undefined) {\n                console.warn(\n                  `Related fixture ${relatedFixtureId} not found in insertedIds, skipping`,\n                );\n                continue;\n              }\n              targetId = resolvedId;\n            } else {\n              targetId = relatedId;\n            }\n\n            // JoinTable에 삽입\n            const [found] = await trx(joinTable)\n              .where({\n                [sourceColumn]: sourceId,\n                [targetColumn]: targetId,\n              })\n              .limit(1);\n\n            if (!found) {\n              await trx(joinTable).insert({\n                [sourceColumn]: sourceId,\n                [targetColumn]: targetId,\n              });\n\n              console.log(\n                chalk.green(\n                  `Inserted into ${joinTable}: ${entity.table}(${sourceId}) - ${relatedEntity.table}(${targetId})`,\n                ),\n              );\n            }\n          }\n        }\n      }\n    }\n  }\n\n  private async checkUniqueViolation(db: Knex, entity: Entity, fixture: FixtureRecord) {\n    const _uniqueIndexes = entity.indexes?.filter((i) => i.type === \"unique\") ?? [];\n\n    const uniqueIndexes = _uniqueIndexes.filter((index) =>\n      index.columns.every((column) => !column.name.startsWith(`${entity.table}__`)),\n    );\n    if (uniqueIndexes.length === 0) {\n      return null;\n    }\n\n    let uniqueQuery = db(entity.table);\n    let hasCondition = false;\n\n    for (const index of uniqueIndexes) {\n      // 컬럼 중 하나라도 null이면 유니크 제약을 위반하지 않기 때문에 해당 인덱스는 무시\n      const containsNull = index.columns.some((column) => {\n        const field = column.name.replace(/_id$/, \"\");\n        return fixture.columns[field]?.value === null;\n      });\n      if (containsNull) {\n        continue;\n      }\n\n      uniqueQuery = uniqueQuery.orWhere((qb) => {\n        for (const column of index.columns) {\n          const field = column.name.replace(/_id$/, \"\");\n\n          if (Array.isArray(fixture.columns[field]?.value)) {\n            qb.whereIn(column.name, fixture.columns[field].value);\n          } else {\n            qb.andWhere(column.name, fixture.columns[field]?.value);\n          }\n        }\n      });\n      hasCondition = true;\n    }\n\n    if (!hasCondition) {\n      return null;\n    }\n\n    const [uniqueFound] = await uniqueQuery;\n    return uniqueFound;\n  }\n\n  private async checkDuplicateByColumns(\n    db: Knex,\n    entity: Entity,\n    fixture: FixtureRecord,\n    columns: string[],\n  ) {\n    if (columns.length === 0) {\n      return null;\n    }\n\n    const whereClause: Record<string, unknown> = {};\n\n    for (const column of columns) {\n      // relation 필드인 경우 _id 붙이기\n      const prop = entity.props.find((p) => p.name === column);\n      const dbColumn = prop && isRelationProp(prop) ? `${column}_id` : column;\n      const value = fixture.columns[column]?.value;\n\n      // null 값이 포함된 경우 중복 확인 스킵\n      if (value === null || value === undefined) {\n        return null;\n      }\n\n      whereClause[dbColumn] = value;\n    }\n\n    const [found] = await db(entity.table).where(whereClause).limit(1);\n    return found;\n  }\n\n  async addFixtureLoader(code: string) {\n    const path = `${Sonamu.apiRootPath}/src/testing/fixture.ts`;\n    const content = readFileSync(path).toString();\n\n    const fixtureLoaderStart = content.indexOf(\"const fixtureLoader = {\");\n    const fixtureLoaderEnd = content.indexOf(\"};\", fixtureLoaderStart);\n\n    if (fixtureLoaderStart !== -1 && fixtureLoaderEnd !== -1) {\n      const newContent = `${content.slice(0, fixtureLoaderEnd)}  ${code}\\n${content.slice(fixtureLoaderEnd)}`;\n\n      writeFileSync(path, newContent);\n    } else {\n      throw new Error(\"Failed to find fixtureLoader in fixture.ts\");\n    }\n  }\n}\n\nexport const FixtureManager = new FixtureManagerClass();\n"],"names":["assert","chalk","execSync","readFileSync","writeFileSync","inflection","knex","unique","inspect","Sonamu","BaseModel","UpsertBuilder","EntityManager","isBelongsToOneRelationProp","isHasManyRelationProp","isManyToManyRelationProp","isOneToOneRelationProp","isRelationProp","isVirtualProp","RelationGraph","FixtureManagerClass","_tdb","tdb","Error","_fdb","fdb","cachedTableNames","relationGraph","builder","fixtureRefMap","Map","uuidToFixtureId","skippedFixtures","init","dbConfig","test","production_master","tConn","connection","pConn","host","port","database","fixture","getChecksum","db","tableName","checksumRow","raw","Checksum","sync","fixtureConn","testConn","testPgEnv","PGPASSWORD","password","user","stdio","env","process","fixturePgEnv","dumpCmd","restoreCmd","shell","visitedRecords","Set","importFixture","entityId","ids","clear","queries","Promise","all","map","id","getImportQueries","flat","wdb","getDB","query","rsh","console","log","info","field","recordKey","has","add","entity","get","row","table","where","limit","undefined","fixtureDatabase","realDatabase","selfQuery","args","Object","entries","relations","filter","relation","customJoinClause","hasJoinColumn","relatedEntity","with","relatedIdColumnName","props","find","p","name","arg","relQueries","reverse","destroy","getFixtures","sourceDBName","targetDBName","searchOptions","duplicateCheck","sourceDB","targetDB","value","searchType","column","prop","type","rows","length","fixtures","initialRecordsLength","newRecords","createFixtureRecord","_db","push","currentFixtureRecord","r","fixtureId","fetchedRecords","slice","customColumns","columns","customDuplicateRow","checkDuplicateByColumns","record","singleRecord","target","uniqueRow","checkUniqueViolation","f","options","records","visitedEntities","create","belongsRecords","throughTable","joinTable","fromColumn","singularize","toColumn","relatedIds","pluck","joinColumn","relatedProp","relatedRow","first","relatedId","insertFixtures","dbName","_fixtures","results","buildGraph","insertionOrder","getInsertionOrder","hasTarget","hasUnique","hasDuplicate","override","existingId","set","yellow","registerFixture","blue","tableOrder","getTableOrder","transaction","trx","insertedIdsByTable","hasTable","getTable","uuids","uuid","upsert","uuidToId","select","whereIn","processManyToManyRelations","skipped","data","ref","insertedId","green","existingRecord","isOverrideMode","propName","relatedFixtureId","skippedExistingId","relatedRef","convertColumnValue","register","Date","tables","seen","sourceRef","sourceUuidToId","sourceId","Array","isArray","targetTable","sourceColumn","targetColumn","targetId","relatedUuidToId","resolvedId","warn","found","insert","_uniqueIndexes","indexes","i","uniqueIndexes","index","every","startsWith","uniqueQuery","hasCondition","containsNull","some","replace","orWhere","qb","andWhere","uniqueFound","whereClause","dbColumn","addFixtureLoader","code","path","apiRootPath","content","toString","fixtureLoaderStart","indexOf","fixtureLoaderEnd","newContent","FixtureManager"],"mappings":"AAAA,OAAOA,YAAY,SAAS;AAC5B,OAAOC,WAAW,QAAQ;AAC1B,SAASC,QAAQ,QAAQ,gBAAgB;AACzC,SAASC,YAAY,EAAEC,aAAa,QAAQ,KAAK;AACjD,OAAOC,gBAAgB,aAAa;AACpC,OAAOC,UAAyB,OAAO;AACvC,SAASC,MAAM,QAAQ,UAAU;AACjC,SAASC,OAAO,QAAQ,OAAO;AAC/B,SAASC,MAAM,QAAQ,kBAAS;AAChC,SAASC,SAAS,QAAQ,4BAAyB;AAEnD,SAAqBC,aAAa,QAAQ,gCAA6B;AAEvE,SAASC,aAAa,QAAQ,8BAA2B;AACzD,SAMEC,0BAA0B,EAC1BC,qBAAqB,EACrBC,wBAAwB,EACxBC,sBAAsB,EACtBC,cAAc,EACdC,aAAa,QAER,oBAAiB;AACxB,SAASC,aAAa,QAAQ,uBAAoB;AASlD,OAAO,MAAMC;IACHC,OAAoB,KAAK;IACjC,IAAIC,IAAIA,GAAS,EAAE;QACjB,IAAI,CAACD,IAAI,GAAGC;IACd;IACA,IAAIA,MAAY;QACd,IAAI,IAAI,CAACD,IAAI,KAAK,MAAM;YACtB,MAAM,IAAIE,MAAM;QAClB;QACA,OAAO,IAAI,CAACF,IAAI;IAClB;IAEQG,OAAoB,KAAK;IACjC,IAAIC,IAAIA,GAAS,EAAE;QACjB,IAAI,CAACD,IAAI,GAAGC;IACd;IACA,IAAIA,MAAY;QACd,IAAI,IAAI,CAACD,IAAI,KAAK,MAAM;YACtB,MAAM,IAAID,MAAM;QAClB;QACA,OAAO,IAAI,CAACC,IAAI;IAClB;IACAE,mBAAoC,KAAK;IAEjCC,gBAAgB,IAAIR,gBAAgB;IAE5C,iCAAiC;IACzBS,UAAyB,IAAIjB,gBAAgB;IAC7CkB,gBAAoC,IAAIC,MAAM;IAC9CC,kBAAuC,IAAID,MAAM;IACjDE,kBAAyE,IAAIF,MAAM;IAE3FG,OAAO;QACL,IAAI,IAAI,CAACZ,IAAI,KAAK,MAAM;YACtB;QACF;QACA,IAAIZ,OAAOyB,QAAQ,CAACC,IAAI,IAAI1B,OAAOyB,QAAQ,CAACE,iBAAiB,EAAE;YAC7D,MAAMC,QAAQ5B,OAAOyB,QAAQ,CAACC,IAAI,CAACG,UAAU;YAG7C,MAAMC,QAAQ9B,OAAOyB,QAAQ,CAACE,iBAAiB,CAACE,UAAU;YAG1D,IACE,GAAGD,MAAMG,IAAI,IAAI,YAAY,CAAC,EAAEH,MAAMI,IAAI,IAAI,KAAK,CAAC,EAAEJ,MAAMK,QAAQ,EAAE,KACtE,GAAGH,MAAMC,IAAI,IAAI,YAAY,CAAC,EAAED,MAAME,IAAI,IAAI,KAAK,CAAC,EAAEF,MAAMG,QAAQ,EAAE,EACtE;gBACA,MAAM,IAAInB,MAAM,CAAC,mCAAmC,CAAC;YACvD;QACF;QAEA,IAAI,CAACD,GAAG,GAAGhB,KAAKG,OAAOyB,QAAQ,CAACC,IAAI;QACpC,IAAI,CAACV,GAAG,GAAGnB,KAAKG,OAAOyB,QAAQ,CAACS,OAAO;IACzC;IAEA,MAAMC,YAAYC,EAAQ,EAAEC,SAAiB,EAAE;QAC7C,MAAM,CAAC,CAACC,YAAY,CAAC,GAAG,MAAMF,GAAGG,GAAG,CAAC,CAAC,eAAe,EAAEF,WAAW;QAClE,OAAOC,YAAYE,QAAQ;IAC7B;IAEA;;;EAGA,GACA,MAAMC,OAAO;QACX,MAAMC,cAAc1C,OAAOyB,QAAQ,CAACS,OAAO,CAACL,UAAU;QACtD,MAAMc,WAAW3C,OAAOyB,QAAQ,CAACC,IAAI,CAACG,UAAU;QAEhD,4BAA4B;QAC5B,MAAMe,YAAY;YAAEC,YAAYF,SAASG,QAAQ,IAAI;QAAG;QACxDrD,SACE,CAAC,QAAQ,EAAEkD,SAASZ,IAAI,CAAC,IAAI,EAAEY,SAASX,IAAI,IAAI,KAAK,IAAI,EAAEW,SAASI,IAAI,CAAC;;;yBAGtD,EAAEJ,SAASV,QAAQ,CAAC;;OAEtC,CAAC,EACF;YAAEe,OAAO;YAAWC,KAAK;gBAAE,GAAGC,QAAQD,GAAG;gBAAE,GAAGL,SAAS;YAAC;QAAuB;QAGjFnD,SACE,CAAC,QAAQ,EAAEkD,SAASZ,IAAI,CAAC,IAAI,EAAEY,SAASX,IAAI,IAAI,KAAK,IAAI,EAAEW,SAASI,IAAI,CAAC,4CAA4C,EAAEJ,SAASV,QAAQ,CAAC,IAAI,CAAC,EAC9I;YAAEe,OAAO;YAAWC,KAAK;gBAAE,GAAGC,QAAQD,GAAG;gBAAE,GAAGL,SAAS;YAAC;QAAuB;QAGjFnD,SACE,CAAC,QAAQ,EAAEkD,SAASZ,IAAI,CAAC,IAAI,EAAEY,SAASX,IAAI,IAAI,KAAK,IAAI,EAAEW,SAASI,IAAI,CAAC,oCAAoC,EAAEJ,SAASV,QAAQ,CAAC,IAAI,CAAC,EACtI;YAAEe,OAAO;YAAWC,KAAK;gBAAE,GAAGC,QAAQD,GAAG;gBAAE,GAAGL,SAAS;YAAC;QAAuB;QAGjF,2DAA2D;QAC3D,MAAMO,eAAe;YAAEN,YAAYH,YAAYI,QAAQ,IAAI;QAAG;QAC9D,MAAMM,UAAU,CAAC,WAAW,EAAEV,YAAYX,IAAI,CAAC,IAAI,EAAEW,YAAYV,IAAI,IAAI,KAAK,IAAI,EAAEU,YAAYK,IAAI,CAAC,IAAI,EAAEL,YAAYT,QAAQ,CAAC,IAAI,CAAC;QACrI,MAAMoB,aAAa,CAAC,cAAc,EAAEV,SAASZ,IAAI,CAAC,IAAI,EAAEY,SAASX,IAAI,IAAI,KAAK,IAAI,EAAEW,SAASI,IAAI,CAAC,IAAI,EAAEJ,SAASV,QAAQ,CAAC,oBAAoB,CAAC;QAE/IxC,SAAS,GAAG2D,QAAQ,eAAe,EAAET,SAASG,QAAQ,IAAI,GAAG,EAAE,EAAEO,YAAY,EAAE;YAC7EL,OAAO;YACPC,KAAK;gBAAE,GAAGC,QAAQD,GAAG;gBAAE,GAAGE,YAAY;YAAC;YACvCG,OAAO;QACT;IACF;IAEQC,iBAAiB,IAAIC,MAAc;IAC3C,MAAMC,cAAcC,QAAgB,EAAEC,GAAa,EAAE;QACnD,+BAA+B;QAC/B,IAAI,CAACJ,cAAc,CAACK,KAAK;QAEzB,MAAMC,UAAU/D,OACd,AACE,CAAA,MAAMgE,QAAQC,GAAG,CACfJ,IAAIK,GAAG,CAAC,OAAOC;YACb,OAAO,MAAM,IAAI,CAACC,gBAAgB,CAACR,UAAU,MAAMO;QACrD,GACF,EACAE,IAAI;QAGR,MAAMC,MAAMnE,UAAUoE,KAAK,CAAC;QAC5B,KAAK,MAAMC,SAAST,QAAS;YAC3B,MAAM,CAACU,IAAI,GAAG,MAAMH,IAAI7B,GAAG,CAAC+B;YAC5BE,QAAQC,GAAG,CAAC;gBACVH;gBACAI,MAAMH,IAAIG,IAAI;YAChB;QACF;IACF;IAEA,MAAMR,iBAAiBR,QAAgB,EAAEiB,KAAa,EAAEV,EAAU,EAAqB;QACrF,MAAMW,YAAY,GAAGlB,SAAS,CAAC,EAAEiB,MAAM,CAAC,EAAEV,IAAI;QAE9C,2BAA2B;QAC3B,IAAI,IAAI,CAACV,cAAc,CAACsB,GAAG,CAACD,YAAY;YACtC,OAAO,EAAE;QACX;QACA,IAAI,CAACrB,cAAc,CAACuB,GAAG,CAACF;QAExBJ,QAAQC,GAAG,CAAC;YAAEf;YAAUiB;YAAOV;QAAG;QAClC,MAAMc,SAAS5E,cAAc6E,GAAG,CAACtB;QACjC,MAAMU,MAAMnE,UAAUoE,KAAK,CAAC;QAE5B,mBAAmB;QACnB,MAAM,CAACY,IAAI,GAAG,MAAMb,IAAIW,OAAOG,KAAK,EAAEC,KAAK,CAACR,OAAOV,IAAImB,KAAK,CAAC;QAC7D,IAAIH,QAAQI,WAAW;YACrB,MAAM,IAAIvE,MAAM,GAAG4C,SAAS,CAAC,EAAEO,GAAG,gBAAgB,CAAC;QACrD;QAEA,aAAa;QACb,MAAMqB,kBAAkB,AAACtF,OAAOyB,QAAQ,CAACS,OAAO,CAACL,UAAU,CAA2BI,QAAQ;QAC9F,MAAMsD,eAAe,AAACvF,OAAOyB,QAAQ,CAACE,iBAAiB,CAACE,UAAU,CAC/DI,QAAQ;QAEX,MAAMuD,YAAY,CAAC,qBAAqB,EAAEF,gBAAgB,KAAK,EAAEP,OAAOG,KAAK,CAAC,oBAAoB,EAAEK,aAAa,KAAK,EAAER,OAAOG,KAAK,CAAC,kBAAkB,EAAEjB,GAAG,CAAC,CAAC;QAE9J,MAAMwB,OAAOC,OAAOC,OAAO,CAACZ,OAAOa,SAAS,EACzCC,MAAM,CACL,CAAC,GAAGC,SAAS,GACX1F,2BAA2B0F,aAC1BvF,uBAAuBuF,aAAaA,SAASC,gBAAgB,KAAKV,WAEtErB,GAAG,CAAC,CAAC,GAAG8B,SAAS;YAChB;;;;;;;QAOA,GACA,IAAInB;YACJ,IAAIV;YACJ,IAAI1D,uBAAuBuF,aAAa,CAACA,SAASE,aAAa,EAAE;gBAC/D,MAAMC,gBAAgB9F,cAAc6E,GAAG,CAACc,SAASI,IAAI;gBACrD,MAAMC,sBAAsBF,cAAcG,KAAK,CAACC,IAAI,CAClD,CAACC,IAAM9F,eAAe8F,MAAMA,EAAEJ,IAAI,KAAKnB,OAAOd,EAAE,GAC/CsC;gBACH,IAAI,CAACJ,qBAAqB;oBACxB,MAAM,IAAIrF,MAAM,GAAGmF,cAAchC,EAAE,CAAC,EAAE,EAAEc,OAAOd,EAAE,CAAC,kBAAkB,CAAC;gBACvE;gBACAU,QAAQ,GAAGwB,oBAAoB,GAAG,CAAC;gBACnClC,KAAKgB,IAAIhB,EAAE;YACb,OAAO;gBACLU,QAAQ;gBACRV,KAAKgB,GAAG,CAAC,GAAGa,SAASS,IAAI,CAAC,GAAG,CAAC,CAAC;YACjC;YACA,OAAO;gBACL7C,UAAUoC,SAASI,IAAI;gBACvBvB;gBACAV;YACF;QACF,GACC4B,MAAM,CAAC,CAACW,MAAQA,IAAIvC,EAAE,KAAK;QAE9B,MAAMwC,aAAa,MAAM3C,QAAQC,GAAG,CAClC0B,KAAKzB,GAAG,CAAC,OAAOyB;YACd,OAAO,IAAI,CAACvB,gBAAgB,CAACuB,KAAK/B,QAAQ,EAAE+B,KAAKd,KAAK,EAAEc,KAAKxB,EAAE;QACjE;QAGF,OAAO;eAAInE,OAAO2G,WAAWC,OAAO,GAAGvC,IAAI;YAAKqB;SAAU;IAC5D;IAEA,MAAMmB,UAAU;QACd,IAAI,IAAI,CAAC/F,IAAI,EAAE;YACb,MAAM,IAAI,CAACA,IAAI,CAAC+F,OAAO;YACvB,IAAI,CAAC/F,IAAI,GAAG;QACd;QACA,IAAI,IAAI,CAACG,IAAI,EAAE;YACb,MAAM,IAAI,CAACA,IAAI,CAAC4F,OAAO;YACvB,IAAI,CAAC5F,IAAI,GAAG;QACd;QACA,MAAMd,UAAU0G,OAAO;IACzB;IAEA,MAAMC,YACJC,YAAkC,EAClCC,YAAkC,EAClCC,aAAmC,EACnCC,cAAsC,EACtC;QACA,MAAMC,WAAWpH,KAAKG,OAAOyB,QAAQ,CAACoF,aAAa;QACnD,MAAMK,WAAWrH,KAAKG,OAAOyB,QAAQ,CAACqF,aAAa;QAEnD,MAAM,EAAEpD,QAAQ,EAAEiB,KAAK,EAAEwC,KAAK,EAAEC,UAAU,EAAE,GAAGL;QAE/C,MAAMhC,SAAS5E,cAAc6E,GAAG,CAACtB;QACjC,MAAM2D,SACJtC,OAAOqB,KAAK,CAACC,IAAI,CAAC,CAACiB,OAASA,KAAKf,IAAI,KAAK5B,QAAQ4C,SAAS,aAAa,GAAG5C,MAAM,GAAG,CAAC,GAAGA;QAE1F,IAAIL,QAAQ2C,SAASlC,OAAOG,KAAK;QACjC,IAAIkC,eAAe,UAAU;YAC3B9C,QAAQA,MAAMa,KAAK,CAACkC,QAAQF;QAC9B,OAAO,IAAIC,eAAe,QAAQ;YAChC9C,QAAQA,MAAMa,KAAK,CAACkC,QAAQ,QAAQ,CAAC,CAAC,EAAEF,MAAM,CAAC,CAAC;QAClD;QAEA,MAAMK,OAAO,MAAMlD;QACnB,IAAIkD,KAAKC,MAAM,KAAK,GAAG;YACrB,MAAM,IAAI3G,MAAM;QAClB;QAEA,MAAM4G,WAA4B,EAAE;QACpC,KAAK,MAAMzC,OAAOuC,KAAM;YACtB,MAAMG,uBAAuBD,SAASD,MAAM;YAC5C,MAAMG,aAAa,MAAM,IAAI,CAACC,mBAAmB,CAAC9C,QAAQE,KAAK;gBAC7D6C,KAAKb;YACP;YACAS,SAASK,IAAI,IAAIH;YACjB,MAAMI,uBAAuBN,SAASrB,IAAI,CAAC,CAAC4B,IAAMA,EAAEC,SAAS,KAAK,GAAGxE,SAAS,CAAC,EAAEuB,IAAIhB,EAAE,EAAE;YAEzF,IAAI+D,sBAAsB;gBACxB,sCAAsC;gBACtCA,qBAAqBG,cAAc,GAAGT,SACnC7B,MAAM,CAAC,CAACoC,IAAMA,EAAEC,SAAS,KAAKF,qBAAqBE,SAAS,EAC5DE,KAAK,CAACT,sBACN3D,GAAG,CAAC,CAACiE,IAAMA,EAAEC,SAAS;YAC3B;QACF;QAEA,WAAW,MAAMhG,WAAWwF,SAAU;YACpC,MAAM3C,SAAS5E,cAAc6E,GAAG,CAAC9C,QAAQwB,QAAQ;YAEjD,8BAA8B;YAC9B,MAAM2E,gBAAgBrB,gBAAgBsB,SAAS,CAACpG,QAAQwB,QAAQ,CAAC;YACjE,IAAI2E,iBAAiBA,cAAcZ,MAAM,GAAG,GAAG;gBAC7C,MAAMc,qBAAqB,MAAM,IAAI,CAACC,uBAAuB,CAC3DtB,UACAnC,QACA7C,SACAmG;gBAEF,IAAIE,oBAAoB;oBACtB,MAAM,CAACE,OAAO,GAAG,MAAM,IAAI,CAACZ,mBAAmB,CAAC9C,QAAQwD,oBAAoB;wBAC1EG,cAAc;wBACdZ,KAAKZ;oBACP;oBACAhF,QAAQyG,MAAM,GAAGF;gBACnB;YACF;YAEA,yCAAyC;YACzC,MAAMG,YAAY,MAAM,IAAI,CAACC,oBAAoB,CAAC3B,UAAUnC,QAAQ7C;YACpE,IAAI0G,WAAW;gBACb,MAAM,CAACH,OAAO,GAAG,MAAM,IAAI,CAACZ,mBAAmB,CAAC9C,QAAQ6D,WAAW;oBACjEF,cAAc;oBACdZ,KAAKZ;gBACP;gBACAhF,QAAQpC,MAAM,GAAG2I;YACnB;QACF;QAEA,MAAMvB,SAASP,OAAO;QACtB,MAAMM,SAASN,OAAO;QAEtB,OAAO7G,OAAO4H,UAAU,CAACoB,IAAMA,EAAEZ,SAAS;IAC5C;IAEA,MAAML,oBACJ9C,MAAc,EACdE,GAGC,EACD8D,OAGC,EACyB;QAC1B,MAAMC,UAA2B,EAAE;QACnC,MAAMC,kBAAkB,IAAIzF;QAE5B,MAAM0F,SAAS,OACbnE,QACAE;YAKA,MAAMiD,YAAY,GAAGnD,OAAOd,EAAE,CAAC,CAAC,EAAEgB,IAAIhB,EAAE,EAAE;YAC1C,IAAIgF,gBAAgBpE,GAAG,CAACqD,YAAY;gBAClC;YACF;YACAe,gBAAgBnE,GAAG,CAACoD;YAEpB,MAAMO,SAAwB;gBAC5BP;gBACAxE,UAAUqB,OAAOd,EAAE;gBACnBA,IAAIgB,IAAIhB,EAAE;gBACVqE,SAAS,CAAC;gBACVH,gBAAgB,EAAE;gBAClBgB,gBAAgB,EAAE;YACpB;YAEA,KAAK,MAAM7B,QAAQvC,OAAOqB,KAAK,CAAE;gBAC/B,IAAI3F,cAAc6G,OAAO;oBACvB;gBACF;gBAEAmB,OAAOH,OAAO,CAAChB,KAAKf,IAAI,CAAC,GAAG;oBAC1Be,MAAMA;oBACNH,OAAOlC,GAAG,CAACqC,KAAKf,IAAI,CAAC;gBACvB;gBAEA,MAAMnE,KAAK2G,SAASjB,OAAO7H,UAAUoE,KAAK,CAAC;gBAC3C,IAAI/D,yBAAyBgH,OAAO;oBAClC,MAAMrB,gBAAgB9F,cAAc6E,GAAG,CAACsC,KAAKpB,IAAI;oBACjD,MAAMkD,eAAe9B,KAAK+B,SAAS;oBACnC,MAAMC,aAAa,GAAG1J,WAAW2J,WAAW,CAACxE,OAAOG,KAAK,EAAE,GAAG,CAAC;oBAC/D,MAAMsE,WAAW,GAAG5J,WAAW2J,WAAW,CAACtD,cAAcf,KAAK,EAAE,GAAG,CAAC;oBAEpE,MAAMuE,aAAa,MAAMrH,GAAGgH,cAAcjE,KAAK,CAACmE,YAAYrE,IAAIhB,EAAE,EAAEyF,KAAK,CAACF;oBAC1Ef,OAAOH,OAAO,CAAChB,KAAKf,IAAI,CAAC,CAACY,KAAK,GAAGsC;gBACpC,OAAO,IAAIpJ,sBAAsBiH,OAAO;oBACtC,MAAMrB,gBAAgB9F,cAAc6E,GAAG,CAACsC,KAAKpB,IAAI;oBACjD,MAAMuD,aAAa,MAAMrH,GAAG6D,cAAcf,KAAK,EAC5CC,KAAK,CAACmC,KAAKqC,UAAU,EAAE1E,IAAIhB,EAAE,EAC7ByF,KAAK,CAAC;oBACTjB,OAAOH,OAAO,CAAChB,KAAKf,IAAI,CAAC,CAACY,KAAK,GAAGsC;gBACpC,OAAO,IAAIlJ,uBAAuB+G,SAAS,CAACA,KAAKtB,aAAa,EAAE;oBAC9D,MAAMC,gBAAgB9F,cAAc6E,GAAG,CAACsC,KAAKpB,IAAI;oBACjD,MAAM0D,cAAc3D,cAAcG,KAAK,CAACC,IAAI,CAC1C,CAACC,IAAM9F,eAAe8F,MAAMA,EAAEJ,IAAI,KAAKnB,OAAOd,EAAE;oBAElD,IAAI2F,aAAa;wBACf,MAAMC,aAAa,MAAMzH,GAAG6D,cAAcf,KAAK,EAAEC,KAAK,CAAC,MAAMF,IAAIhB,EAAE,EAAE6F,KAAK;wBAC1ErB,OAAOH,OAAO,CAAChB,KAAKf,IAAI,CAAC,CAACY,KAAK,GAAG0C,YAAY5F;oBAChD;gBACF,OAAO,IAAIzD,eAAe8G,OAAO;oBAC/B,MAAMyC,YAAY9E,GAAG,CAAC,GAAGqC,KAAKf,IAAI,CAAC,GAAG,CAAC,CAAC;oBACxCkC,OAAOH,OAAO,CAAChB,KAAKf,IAAI,CAAC,CAACY,KAAK,GAAG4C;oBAClC,IAAIA,WAAW;wBACbtB,OAAOU,cAAc,CAACpB,IAAI,CAAC,GAAGT,KAAKpB,IAAI,CAAC,CAAC,EAAE6D,WAAW;oBACxD;oBACA,IAAI,CAAChB,SAASL,gBAAgBqB,WAAW;wBACvC,MAAM9D,gBAAgB9F,cAAc6E,GAAG,CAACsC,KAAKpB,IAAI;wBACjD,MAAM2D,aAAa,MAAMzH,GAAG6D,cAAcf,KAAK,EAAEC,KAAK,CAAC,MAAM4E,WAAWD,KAAK;wBAC7E,IAAID,YAAY;4BACd,MAAMX,OAAOjD,eAAe4D;wBAC9B;oBACF;gBACF;YACF;YAEAb,QAAQjB,IAAI,CAACU;QACf;QAEA,MAAMS,OAAOnE,QAAQE;QAErB,OAAO+D;IACT;IAEA;;;;GAIC,GACD,MAAMgB,eACJC,MAA4B,EAC5BC,SAA0B,EACM;QAChC,MAAMxC,WAAW5H,OAAOoK,WAAW,CAACpB,IAAMA,EAAEZ,SAAS;QAErD,MAAM;QACN,IAAI,CAAC/G,OAAO,GAAG,IAAIjB;QACnB,IAAI,CAACkB,aAAa,GAAG,IAAIC;QACzB,IAAI,CAACC,eAAe,GAAG,IAAID;QAC3B,IAAI,CAACE,eAAe,GAAG,IAAIF;QAE3B,MAAMe,KAAKvC,KAAKG,OAAOyB,QAAQ,CAACwI,OAAO;QACvC,MAAME,UAAiC,EAAE;QAEzC,IAAI;YACF,wCAAwC;YACxC,IAAI,CAACjJ,aAAa,CAACkJ,UAAU,CAAC1C;YAC9B,MAAM2C,iBAAiB,IAAI,CAACnJ,aAAa,CAACoJ,iBAAiB;YAE3D,0CAA0C;YAC1C,KAAK,MAAMpC,aAAamC,eAAgB;gBACtC,MAAMnI,UAAUwF,SAASrB,IAAI,CAAC,CAACyC,IAAMA,EAAEZ,SAAS,KAAKA;gBACrD,IAAI,CAAChG,SAAS;gBAEd,MAAMqI,YAAY,CAAC,CAACrI,QAAQyG,MAAM;gBAClC,MAAM6B,YAAY,CAAC,CAACtI,QAAQpC,MAAM;gBAClC,MAAM2K,eAAeF,aAAaC;gBAElC,gCAAgC;gBAChC,IAAIC,gBAAgB,CAACvI,QAAQwI,QAAQ,EAAE;oBACrC,uCAAuC;oBACvC,MAAMC,aAAazI,QAAQpC,MAAM,EAAEmE,MAAM/B,QAAQyG,MAAM,EAAE1E;oBACzD1E,OAAOoL;oBACP,IAAI,CAACpJ,eAAe,CAACqJ,GAAG,CAAC1C,WAAW;wBAClCxE,UAAUxB,QAAQwB,QAAQ;wBAC1BiH;oBACF;oBAEAnG,QAAQC,GAAG,CACTjF,MAAMqL,MAAM,CACV,CAAC,QAAQ,EAAE3I,QAAQwB,QAAQ,CAAC,CAAC,EAAExB,QAAQ+B,EAAE,CAAC,aAAa,EAAE0G,WAAW,kBAAkB,CAAC;oBAG3F;gBACF;gBAEA,IAAI,CAACG,eAAe,CAAC5I;gBACrBsC,QAAQC,GAAG,CACTjF,MAAMuL,IAAI,CACR,CAAC,WAAW,EAAE7I,QAAQwB,QAAQ,CAAC,CAAC,EAAExB,QAAQ+B,EAAE,GAAG/B,QAAQwI,QAAQ,GAAG,CAAC,sBAAsB,EAAExI,QAAQyG,MAAM,EAAE1E,GAAG,CAAC,CAAC,GAAG,IAAI;YAG7H;YAEA,oBAAoB;YACpB,MAAM+G,aAAa,IAAI,CAACC,aAAa,CAACvD;YAEtC,MAAMtF,GAAG8I,WAAW,CAAC,OAAOC;gBAC1B,MAAMC,qBAAqB,IAAI/J;gBAE/B,KAAK,MAAMgB,aAAa2I,WAAY;oBAClC,IAAI,CAAC,IAAI,CAAC7J,OAAO,CAACkK,QAAQ,CAAChJ,YAAY;oBAEvC,yBAAyB;oBACzB,MAAM6C,QAAQ,IAAI,CAAC/D,OAAO,CAACmK,QAAQ,CAACjJ;oBACpC,MAAMkJ,QAAQrG,MAAMsC,IAAI,CAACxD,GAAG,CAAC,CAACiB,MAAQA,IAAIuG,IAAI;oBAE9ChH,QAAQC,GAAG,CAACjF,MAAMuL,IAAI,CAAC,CAAC,UAAU,EAAE1I,UAAU,MAAM,EAAEkJ,MAAM9D,MAAM,CAAC,KAAK,CAAC;oBACzE,MAAM,IAAI,CAACtG,OAAO,CAACsK,MAAM,CAACN,KAAK9I;oBAE/B,iCAAiC;oBACjC,IAAIkJ,MAAM9D,MAAM,GAAG,GAAG;wBACpB,MAAMiE,WAAW,IAAIrK;wBACrB,MAAMmG,OAAO,MAAM2D,IAAI9I,WACpBsJ,MAAM,CAAC,QAAQ,MACfC,OAAO,CAAC,QAAQL;wBAEnB,KAAK,MAAMtG,OAAOuC,KAAM;4BACtBkE,SAASd,GAAG,CAAC3F,IAAIuG,IAAI,EAAEvG,IAAIhB,EAAE;wBAC/B;wBAEAmH,mBAAmBR,GAAG,CAACvI,WAAWqJ;oBACpC;gBACF;gBAEA,sBAAsB;gBACtB,MAAM,IAAI,CAACG,0BAA0B,CAACV,KAAKzD,UAAU0D;gBAErD,WAAW;gBACX,KAAK,MAAMlJ,WAAWwF,SAAU;oBAC9B,MAAM3C,SAAS5E,cAAc6E,GAAG,CAAC9C,QAAQwB,QAAQ;oBAEjD,gCAAgC;oBAChC,MAAMoI,UAAU,IAAI,CAACvK,eAAe,CAACyD,GAAG,CAAC9C,QAAQgG,SAAS;oBAC1D,IAAI4D,SAAS;wBACX3B,QAAQpC,IAAI,CAAC;4BACXrE,UAAUxB,QAAQwB,QAAQ;4BAC1BqI,MAAM,MAAMZ,IAAIpG,OAAOG,KAAK,EAAEC,KAAK,CAAC,MAAM2G,QAAQnB,UAAU,EAAEb,KAAK;wBACrE;wBACA;oBACF;oBAEA,MAAMkC,MAAM,IAAI,CAAC5K,aAAa,CAAC4D,GAAG,CAAC9C,QAAQgG,SAAS;oBACpD,IAAI8D,KAAK;wBACP,MAAMN,WAAWN,mBAAmBpG,GAAG,CAACD,OAAOG,KAAK;wBACpD,MAAM+G,aAAaP,UAAU1G,IAAIgH,IAAIR,IAAI;wBAEzC,IAAIS,eAAe5G,WAAW;4BAC5B8E,QAAQpC,IAAI,CAAC;gCACXrE,UAAUxB,QAAQwB,QAAQ;gCAC1BqI,MAAM,MAAMZ,IAAIpG,OAAOG,KAAK,EAAEC,KAAK,CAAC,MAAM8G,YAAYnC,KAAK;4BAC7D;4BAEAtF,QAAQC,GAAG,CACTjF,MAAM0M,KAAK,CAAC,CAAC,cAAc,EAAEnH,OAAOG,KAAK,CAAC,GAAG,EAAEhD,QAAQ+B,EAAE,CAAC,KAAK,EAAEgI,YAAY;wBAEjF;oBACF;gBACF;YACF;QACF,SAAU;YACR,MAAM7J,GAAGuE,OAAO;QAClB;QAEA,OAAO7G,OAAOqK,SAAS,CAAClC,IAAM,GAAGA,EAAEvE,QAAQ,CAAC,CAAC,EAAEuE,EAAE8D,IAAI,CAAC9H,EAAE,EAAE;IAC5D;IAEA;;GAEC,GACD,AAAQ6G,gBAAgB5I,OAAsB,EAAS;QACrD,MAAM6C,SAAS5E,cAAc6E,GAAG,CAAC9C,QAAQwB,QAAQ;QACjD,MAAMuB,MAA+B,CAAC;QAEtC,yDAAyD;QACzD,MAAMkH,iBAAiBjK,QAAQyG,MAAM,IAAIzG,QAAQpC,MAAM;QACvD,MAAMsM,iBAAiBlK,QAAQwI,QAAQ,IAAIyB;QAE3C,KAAK,MAAM,CAACE,UAAUhF,OAAO,IAAI3B,OAAOC,OAAO,CAACzD,QAAQoG,OAAO,EAAG;YAChE,MAAMhB,OAAOD,OAAOC,IAAI;YAExB,IAAI7G,cAAc6G,OAAO;gBACvB;YACF;YAEA,sCAAsC;YACtC,IAAI+E,aAAa,QAAQA,aAAa,QAAQ;gBAC5C,IAAID,kBAAkBD,gBAAgB;oBACpC,kCAAkC;oBAClClH,GAAG,CAACoH,SAAS,GAAGF,eAAe7D,OAAO,CAAC+D,SAAS,EAAElF;gBACpD;gBAEA;YACF;YAEA,IAAI3G,eAAe8G,OAAO;gBACxB,IACElH,2BAA2BkH,SAC1B/G,uBAAuB+G,SAASA,KAAKtB,aAAa,EACnD;oBACA,MAAM+D,YAAY1C,OAAOF,KAAK;oBAC9B,IAAI4C,cAAc,QAAQA,cAAc1E,WAAW;wBACjD,MAAMiH,mBAAmB,GAAGhF,KAAKpB,IAAI,CAAC,CAAC,EAAE6D,WAAW;wBAEpD,wBAAwB;wBACxB,MAAMwC,oBAAoB,IAAI,CAAChL,eAAe,CAACyD,GAAG,CAACsH,mBAAmB3B;wBACtE,IAAI4B,sBAAsBlH,WAAW;4BACnC,0CAA0C;4BAC1CJ,GAAG,CAAC,GAAGoH,SAAS,GAAG,CAAC,CAAC,GAAGE;wBAC1B,OAAO;4BACL,MAAMC,aAAa,IAAI,CAACpL,aAAa,CAAC4D,GAAG,CAACsH;4BAC1C,IAAIE,YAAY;gCACd,+BAA+B;gCAC/BvH,GAAG,CAAC,GAAGoH,SAAS,GAAG,CAAC,CAAC,GAAGG;4BAC1B,OAAO;gCACL,oCAAoC;gCACpCvH,GAAG,CAAC,GAAGoH,SAAS,GAAG,CAAC,CAAC,GAAGtC;4BAC1B;wBACF;oBACF,OAAO;wBACL9E,GAAG,CAAC,GAAGoH,SAAS,GAAG,CAAC,CAAC,GAAG;oBAC1B;gBACF;YACA,6BAA6B;YAC/B,OAAO;gBACL,QAAQ;gBACRpH,GAAG,CAACoH,SAAS,GAAG,IAAI,CAACI,kBAAkB,CAACnF,MAAoBD,OAAOF,KAAK;YAC1E;QACF;QAEA3C,QAAQC,GAAG,CAACjF,MAAMuL,IAAI,CAAC,CAAC,YAAY,EAAEhG,OAAOG,KAAK,CAAC,GAAG,EAAEnF,QAAQkF,KAAK,OAAO,MAAM,OAAO;QACzF,MAAM+G,MAAM,IAAI,CAAC7K,OAAO,CAACuL,QAAQ,CAAC3H,OAAOG,KAAK,EAAED;QAChD,IAAI,CAAC7D,aAAa,CAACwJ,GAAG,CAAC1I,QAAQgG,SAAS,EAAE8D;QAC1C,IAAI,CAAC1K,eAAe,CAACsJ,GAAG,CAACoB,IAAIR,IAAI,EAAEtJ,QAAQgG,SAAS;QAEpD,OAAO8D;IACT;IAEA;;GAEC,GACD,AAAQS,mBAAmBnF,IAAgB,EAAEH,KAAc,EAAW;QACpE,IAAIA,UAAU,QAAQA,UAAU9B,WAAW;YACzC,OAAO;QACT;QAEA,OAAQiC,KAAKC,IAAI;YACf,KAAK;gBACH,8DAA8D;gBAC9D,OAAOJ;YAET,KAAK;gBACH,IAAI,OAAOA,UAAU,YAAY,OAAOA,UAAU,UAAU;oBAC1D,OAAO,IAAIwF,KAAKxF;gBAClB;gBACA,OAAOA;YAET;gBACE,OAAOA;QACX;IACF;IAEA;;GAEC,GACD,AAAQ8D,cAAcvD,QAAyB,EAAkC;QAC/E,MAAMkF,SAAmB,EAAE;QAC3B,MAAMC,OAAO,IAAIrJ;QAEjB,KAAK,MAAMtB,WAAWwF,SAAU;YAC9B,MAAM3C,SAAS5E,cAAc6E,GAAG,CAAC9C,QAAQwB,QAAQ;YACjD,IAAI,CAACmJ,KAAKhI,GAAG,CAACE,OAAOG,KAAK,GAAG;gBAC3B2H,KAAK/H,GAAG,CAACC,OAAOG,KAAK;gBACrB0H,OAAO7E,IAAI,CAAChD,OAAOG,KAAK;YAC1B;QACF;QAEA,OAAO0H;IACT;IAEA,MAAcf,2BACZV,GAAqB,EACrBzD,QAAyB,EACzB0D,kBAAoD,EACrC;QACf,KAAK,MAAMlJ,WAAWwF,SAAU;YAC9B,MAAM3C,SAAS5E,cAAc6E,GAAG,CAAC9C,QAAQwB,QAAQ;YACjD,MAAMoJ,YAAY,IAAI,CAAC1L,aAAa,CAAC4D,GAAG,CAAC9C,QAAQgG,SAAS;YAE1D,IAAI,CAAC4E,WAAW;YAEhB,MAAMC,iBAAiB3B,mBAAmBpG,GAAG,CAACD,OAAOG,KAAK;YAC1D,MAAM8H,WAAWD,gBAAgB/H,IAAI8H,UAAUtB,IAAI;YAEnD,IAAIwB,aAAa3H,WAAW;YAE5B,KAAK,MAAM,GAAGgC,OAAO,IAAI3B,OAAOC,OAAO,CAACzD,QAAQoG,OAAO,EAAG;gBACxD,MAAMhB,OAAOD,OAAOC,IAAI;gBAExB,IAAIhH,yBAAyBgH,SAAS2F,MAAMC,OAAO,CAAC7F,OAAOF,KAAK,GAAG;oBACjE,iCAAiC;oBACjC,MAAMgG,cAAchN,cAAc6E,GAAG,CAACsC,KAAKpB,IAAI;oBAC/C,IAAI,IAAI,CAAC/E,OAAO,CAACkK,QAAQ,CAAC8B,YAAYjI,KAAK,MAAM,OAAO;oBAExD,MAAMuE,aAAapC,OAAOF,KAAK;oBAC/B,IAAIsC,WAAWhC,MAAM,KAAK,GAAG;oBAE7B,MAAM4B,YAAY,AAAC/B,KAAgC+B,SAAS;oBAC5D,MAAMpD,gBAAgB9F,cAAc6E,GAAG,CAACsC,KAAKpB,IAAI;oBAEjD,MAAMkH,eAAe,GAAGxN,WAAW2J,WAAW,CAACxE,OAAOG,KAAK,EAAE,GAAG,CAAC;oBACjE,MAAMmI,eAAe,GAAGzN,WAAW2J,WAAW,CAACtD,cAAcf,KAAK,EAAE,GAAG,CAAC;oBAExE,KAAK,MAAM6E,aAAaN,WAAY;wBAClC,MAAM6C,mBAAmB,GAAGhF,KAAKpB,IAAI,CAAC,CAAC,EAAE6D,WAAW;wBACpD,MAAMyC,aAAa,IAAI,CAACpL,aAAa,CAAC4D,GAAG,CAACsH;wBAE1C,IAAIgB;wBAEJ,IAAId,YAAY;4BACd,MAAMe,kBAAkBnC,mBAAmBpG,GAAG,CAACiB,cAAcf,KAAK;4BAClE,MAAMsI,aAAaD,iBAAiBvI,IAAIwH,WAAWhB,IAAI;4BAEvD,IAAIgC,eAAenI,WAAW;gCAC5Bb,QAAQiJ,IAAI,CACV,CAAC,gBAAgB,EAAEnB,iBAAiB,mCAAmC,CAAC;gCAE1E;4BACF;4BACAgB,WAAWE;wBACb,OAAO;4BACLF,WAAWvD;wBACb;wBAEA,gBAAgB;wBAChB,MAAM,CAAC2D,MAAM,GAAG,MAAMvC,IAAI9B,WACvBlE,KAAK,CAAC;4BACL,CAACiI,aAAa,EAAEJ;4BAChB,CAACK,aAAa,EAAEC;wBAClB,GACClI,KAAK,CAAC;wBAET,IAAI,CAACsI,OAAO;4BACV,MAAMvC,IAAI9B,WAAWsE,MAAM,CAAC;gCAC1B,CAACP,aAAa,EAAEJ;gCAChB,CAACK,aAAa,EAAEC;4BAClB;4BAEA9I,QAAQC,GAAG,CACTjF,MAAM0M,KAAK,CACT,CAAC,cAAc,EAAE7C,UAAU,EAAE,EAAEtE,OAAOG,KAAK,CAAC,CAAC,EAAE8H,SAAS,IAAI,EAAE/G,cAAcf,KAAK,CAAC,CAAC,EAAEoI,SAAS,CAAC,CAAC;wBAGtG;oBACF;gBACF;YACF;QACF;IACF;IAEA,MAAczE,qBAAqBzG,EAAQ,EAAE2C,MAAc,EAAE7C,OAAsB,EAAE;QACnF,MAAM0L,iBAAiB7I,OAAO8I,OAAO,EAAEhI,OAAO,CAACiI,IAAMA,EAAEvG,IAAI,KAAK,aAAa,EAAE;QAE/E,MAAMwG,gBAAgBH,eAAe/H,MAAM,CAAC,CAACmI,QAC3CA,MAAM1F,OAAO,CAAC2F,KAAK,CAAC,CAAC5G,SAAW,CAACA,OAAOd,IAAI,CAAC2H,UAAU,CAAC,GAAGnJ,OAAOG,KAAK,CAAC,EAAE,CAAC;QAE7E,IAAI6I,cAActG,MAAM,KAAK,GAAG;YAC9B,OAAO;QACT;QAEA,IAAI0G,cAAc/L,GAAG2C,OAAOG,KAAK;QACjC,IAAIkJ,eAAe;QAEnB,KAAK,MAAMJ,SAASD,cAAe;YACjC,kDAAkD;YAClD,MAAMM,eAAeL,MAAM1F,OAAO,CAACgG,IAAI,CAAC,CAACjH;gBACvC,MAAM1C,QAAQ0C,OAAOd,IAAI,CAACgI,OAAO,CAAC,QAAQ;gBAC1C,OAAOrM,QAAQoG,OAAO,CAAC3D,MAAM,EAAEwC,UAAU;YAC3C;YACA,IAAIkH,cAAc;gBAChB;YACF;YAEAF,cAAcA,YAAYK,OAAO,CAAC,CAACC;gBACjC,KAAK,MAAMpH,UAAU2G,MAAM1F,OAAO,CAAE;oBAClC,MAAM3D,QAAQ0C,OAAOd,IAAI,CAACgI,OAAO,CAAC,QAAQ;oBAE1C,IAAItB,MAAMC,OAAO,CAAChL,QAAQoG,OAAO,CAAC3D,MAAM,EAAEwC,QAAQ;wBAChDsH,GAAG7C,OAAO,CAACvE,OAAOd,IAAI,EAAErE,QAAQoG,OAAO,CAAC3D,MAAM,CAACwC,KAAK;oBACtD,OAAO;wBACLsH,GAAGC,QAAQ,CAACrH,OAAOd,IAAI,EAAErE,QAAQoG,OAAO,CAAC3D,MAAM,EAAEwC;oBACnD;gBACF;YACF;YACAiH,eAAe;QACjB;QAEA,IAAI,CAACA,cAAc;YACjB,OAAO;QACT;QAEA,MAAM,CAACO,YAAY,GAAG,MAAMR;QAC5B,OAAOQ;IACT;IAEA,MAAcnG,wBACZpG,EAAQ,EACR2C,MAAc,EACd7C,OAAsB,EACtBoG,OAAiB,EACjB;QACA,IAAIA,QAAQb,MAAM,KAAK,GAAG;YACxB,OAAO;QACT;QAEA,MAAMmH,cAAuC,CAAC;QAE9C,KAAK,MAAMvH,UAAUiB,QAAS;YAC5B,0BAA0B;YAC1B,MAAMhB,OAAOvC,OAAOqB,KAAK,CAACC,IAAI,CAAC,CAACC,IAAMA,EAAEC,IAAI,KAAKc;YACjD,MAAMwH,WAAWvH,QAAQ9G,eAAe8G,QAAQ,GAAGD,OAAO,GAAG,CAAC,GAAGA;YACjE,MAAMF,QAAQjF,QAAQoG,OAAO,CAACjB,OAAO,EAAEF;YAEvC,0BAA0B;YAC1B,IAAIA,UAAU,QAAQA,UAAU9B,WAAW;gBACzC,OAAO;YACT;YAEAuJ,WAAW,CAACC,SAAS,GAAG1H;QAC1B;QAEA,MAAM,CAACuG,MAAM,GAAG,MAAMtL,GAAG2C,OAAOG,KAAK,EAAEC,KAAK,CAACyJ,aAAaxJ,KAAK,CAAC;QAChE,OAAOsI;IACT;IAEA,MAAMoB,iBAAiBC,IAAY,EAAE;QACnC,MAAMC,OAAO,GAAGhP,OAAOiP,WAAW,CAAC,uBAAuB,CAAC;QAC3D,MAAMC,UAAUxP,aAAasP,MAAMG,QAAQ;QAE3C,MAAMC,qBAAqBF,QAAQG,OAAO,CAAC;QAC3C,MAAMC,mBAAmBJ,QAAQG,OAAO,CAAC,MAAMD;QAE/C,IAAIA,uBAAuB,CAAC,KAAKE,qBAAqB,CAAC,GAAG;YACxD,MAAMC,aAAa,GAAGL,QAAQ9G,KAAK,CAAC,GAAGkH,kBAAkB,EAAE,EAAEP,KAAK,EAAE,EAAEG,QAAQ9G,KAAK,CAACkH,mBAAmB;YAEvG3P,cAAcqP,MAAMO;QACtB,OAAO;YACL,MAAM,IAAIzO,MAAM;QAClB;IACF;AACF;AAEA,OAAO,MAAM0O,iBAAiB,IAAI7O,sBAAsB"}
@@ -89,4 +89,4 @@ In order to be iterable, non-array objects must have a [Symbol.iterator]() metho
89
89
  `):String(m);try{const v=document.createElement("textarea");v.value=y,document.body.appendChild(v),v.select(),document.execCommand("copy"),document.body.removeChild(v),d(!0),setTimeout(()=>d(!1),1500)}catch(v){console.error("Failed to copy text: ",v)}};return j.useEffect(()=>{i(new Array(e.split(`
90
90
  `).length).fill(!1))},[e]),x.jsx(eGe,{children:`\`\`\`${t} ${n?`title="${n}"`:""}
91
91
  ${e}
92
- \`\`\``,components:{code({children:m,className:b,node:y,ref:v,...k}){const _=String(m).trimEnd();return x.jsxs("div",{className:"code",children:[x.jsxs("div",{className:"code-header",children:[x.jsx("span",{children:n??t}),x.jsxs("div",{children:[o&&x.jsx(ar,{label:a.every(S=>S)?"전체 해제":"전체 선택",checked:a.every(S=>S),onChange:()=>{const S=a.every(T=>T);i(a.map(()=>!S))}}),x.jsxs(Le,{icon:!0,onClick:()=>h(_),size:"tiny",children:[x.jsx(mt,{name:u?"check circle outline":"clipboard outline"}),u?"복사 완료":"복사"]})]})]}),x.jsx(AQe,{...k,children:_,language:t,style:Bne[r??"materialDark"],renderer:({rows:S,stylesheet:T})=>x.jsx("div",{style:{position:"relative"},children:S.map((C,w)=>{const A=a[w]??!1,O=s===w;return x.jsxs("div",{className:`code-line ${O?"hovered":""}`,style:A?{backgroundColor:"rgba(0, 123, 255, 0.1)"}:{},onMouseEnter:()=>l(w),onMouseLeave:()=>l(null),onClick:()=>o&&p(w),children:[o&&x.jsx(ar,{checked:A,onClick:R=>R.stopPropagation(),onChange:()=>p(w)}),x.jsx("span",{children:C.children?.map((R,$)=>R.type==="element"?x.jsx("span",{className:R.properties.className.join(" "),style:{...R.properties.className.reduce((N,M)=>T[M]?{...N,...T[M]}:N,{}),fontWeight:"normal"},children:R.children.map((N,M)=>x.jsx("span",{children:N.value},M))},$):x.jsx("span",{children:R.value},$))})]},w)})})})]})}}})};function IQe({fixtureRecords:e,onRelationToggle:t,selectedIds:n,setFixtureRecords:r}){const o=Qv(e,a=>a.entityId);return x.jsx("div",{className:"fixture-record-viewer",children:Object.entries(o).map(([a,i])=>x.jsx(jte,{fixtures:i??[],selectedIds:n,onRelationToggle:t,setFixtureRecords:r},a))})}const sG=["development_master","production_master","fixture_remote","test"];function RQe(){const{data:e,isLoading:t}=Xe.useEntities(),[n,r]=j.useState("development_master"),[o,a]=j.useState("test"),[i,s]=j.useState([]),[l,u]=j.useState([]),[d,p]=j.useState(new Set),[h,m]=j.useState(0),[b,y]=j.useState("table"),[v,k]=j.useState({}),[_,S]=j.useState(""),[T,C]=j.useState([]),[w,A]=j.useState(!1),[O,R]=j.useState("chat"),{form:$,register:N}=em(Ve({entityId:ie(),field:ie(),value:ie(),searchType:Br(["equals","like"])}),{entityId:"",field:"id",value:"",searchType:"equals"}),[M,V]=j.useState(null),z=e?.entities?.find(U=>U.id===_)??null,H=()=>{if(!$.entityId||!$.field||!$.value)return;m(0),s([]),u([]),p(new Set);const U=Object.keys(v).length>0?{columns:v}:void 0;Xe.getFixtures(n,o,$,U).then(K=>{s(K),p(new Set(K.map(P=>P.fixtureId)))}).catch(Ht)},W=()=>{i.length!==0&&(m(1),Xe.importFixtures(o,i).then(U=>{u(U)}).catch(Ht))},G=async(U,K,P,Z)=>{const L=`${K}#${P}`,X=Object.keys(v).length>0?{columns:v}:void 0;if(Z)Xe.getFixtures(n,o,{entityId:K,field:"id",value:String(P),searchType:"equals"},X).then(ne=>{const te=i.find(pe=>pe.fixtureId===U);te&&te.fetchedRecords.push(L);const ae=ne.filter(pe=>!i.some(be=>be.fixtureId===pe.fixtureId));s(pe=>Array.from([...pe,...ae])),p(pe=>{const be=new Set(pe);return ae.forEach(xe=>{be.add(xe.fixtureId)}),be})}).catch(Ht);else{const ne=i.find(pe=>pe.fixtureId===U);if(!ne)return;ne.fetchedRecords=ne.fetchedRecords.filter(pe=>pe!==L);const te=new Set([L]),ae=i.find(pe=>pe.fixtureId===L);if(ae?.fetchedRecords.length){const pe=new Set([U]);i.forEach(be=>{be.fixtureId!==L&&be.fetchedRecords.forEach(xe=>{pe.add(xe)})}),ae?.fetchedRecords.forEach(be=>{pe.has(be)||te.add(be)})}else{const pe=new Set,be=xe=>{pe.has(xe)||(pe.add(xe),i.forEach(Ie=>{Ie.belongsRecords.includes(xe)&&be(Ie.fixtureId)}))};be(L)}p(pe=>{const be=new Set(pe);return te.forEach(xe=>{be.delete(xe)}),be}),s(pe=>pe.filter(be=>!te.has(be.fixtureId)))}},q=()=>{!_||T.length===0||(k(U=>({...U,[_]:T})),S(""),C([]))},Q=U=>{k(K=>{const{[U]:P,...Z}=K;return Z})};j.useEffect(()=>{if($.entityId&&e?.entities){const U=e.entities.find(K=>K.id===$.entityId);U&&V(U)}},[$.entityId,e]),j.useEffect(()=>{C([])},[]);const F=[{menuItem:"Fixture Record Viewer",render:()=>x.jsx(ps.Pane,{children:b==="table"?x.jsx(IQe,{fixtureRecords:i,onRelationToggle:G,selectedIds:d,setFixtureRecords:s}):x.jsx(z5e,{fixtures:i,selectedIds:d,onRelationToggle:G,setFixtureRecords:s})})},{menuItem:"Fixture Code Viewer",render:()=>x.jsx(ps.Pane,{children:e?.entities&&l.length>0&&x.jsx(CQe,{entities:e.entities,fixtureResults:l,targetDB:o})})}];return x.jsxs("div",{className:"fixture-index",children:[x.jsx("div",{className:"fixture-sidebar",children:x.jsxs(un,{className:"fixture-header",children:[x.jsxs("div",{className:"search-section",children:[x.jsxs("div",{className:"search-title",children:[x.jsx(mt,{name:"search",style:{marginRight:"5px"}}),"검색 대상 설정"]}),x.jsx("div",{className:"db-dropdown-wrapper",children:x.jsx(Ct,{fluid:!0,placeholder:"검색할 DB 선택",selection:!0,options:sG.map(U=>({key:U,value:U,text:U.replace("_master","")})),value:n,onChange:(U,{value:K})=>r(K)})}),x.jsx("div",{style:{flexGrow:1,minWidth:"200px"},children:x.jsx(Ct,{fluid:!0,placeholder:"엔티티 선택",search:!0,selection:!0,loading:t,options:e?.entities?.map(U=>({key:U.id,value:U.id,text:U.id}))||[],...N("entityId")})}),M&&x.jsxs(x.Fragment,{children:[x.jsx(Ct,{placeholder:"컬럼 선택",selection:!0,options:M.props.filter(U=>U.type==="virtual"?!1:U.type==="relation"?!!(U.relationType==="BelongsToOne"||U.relationType==="OneToOne"&&U.hasJoinColumn):!0).map(U=>({key:U.name,value:U.name,text:U.name})),...N("field")}),x.jsx(vr,{placeholder:"검색 값 입력",...N("value"),style:{flexGrow:1}}),x.jsx(Ct,{selection:!0,options:[{key:"equals",text:"Equals",value:"equals"},{key:"like",text:"Like",value:"like"}],...N("searchType")})]}),x.jsx(Le,{onClick:H,disabled:!$.entityId||!$.field||!$.value||t,loading:t,primary:!0,content:"검색"})]}),x.jsxs("div",{className:"save-section",children:[x.jsxs("button",{type:"button",className:"save-title",style:{cursor:"pointer",background:"none",border:"none",padding:0,display:"flex",alignItems:"center",width:"100%",textAlign:"left"},onClick:()=>A(!w),children:[x.jsx(mt,{name:w?"chevron down":"chevron right"}),x.jsx(mt,{name:"database",style:{marginRight:"5px"}}),"저장 DB 설정",i.length>0&&(()=>{const U=i.filter(K=>{const P=!!K.target,Z=!!K.unique;return!!(!P&&!Z||K.override&&(P||Z))});return x.jsxs(Yt,{color:"green",size:"small",style:{marginLeft:"auto"},children:[U.length,"개 저장 예정"]})})()]}),x.jsx("div",{className:"db-dropdown-wrapper",children:x.jsx(Ct,{fluid:!0,placeholder:"저장할 대상 DB 선택",header:"Fixture Target DB",selection:!0,options:sG.map(U=>({key:U,value:U,text:U})),value:o,onChange:(U,{value:K})=>a(K)})}),x.jsx(Le,{onClick:W,color:"blue",content:"저장",disabled:i.length===0}),w&&i.length>0&&(()=>{const K=i.filter(P=>{const Z=!!P.target,L=!!P.unique;return!!(!Z&&!L||P.override&&(Z||L))}).reduce((P,Z)=>(P[Z.entityId]||(P[Z.entityId]=[]),P[Z.entityId].push(Z),P),{});return x.jsxs("div",{style:{marginTop:"10px",marginBottom:"10px"},children:[x.jsx("p",{style:{color:"#666",fontSize:"12px",marginBottom:"10px"},children:"저장될 픽스쳐 목록"}),x.jsx("div",{className:"fixture-record-group",children:Object.entries(K).map(([P,Z])=>x.jsxs("div",{style:{padding:"8px",backgroundColor:"#f9f9f9",borderRadius:"4px"},children:[x.jsxs("div",{style:{fontWeight:"bold",marginBottom:"4px"},children:[P," (",Z.length,"개)"]}),x.jsx("div",{style:{fontSize:"12px",color:"#555"},children:Z.map(L=>{const X=!!L.target,ne=!!L.unique;let te="신규";return L.override&&(X||ne)&&(te="덮어쓰기"),x.jsxs(Yt,{size:"tiny",style:{margin:"2px"},children:["#",L.id,x.jsx(Yt.Detail,{children:te})]},L.fixtureId)})})]},P))})]})})()]}),O==="chat"?x.jsx(Gze,{fixtureRecords:i,onUpdateFixtures:U=>{s(U);const K=new Set(i.map(Z=>Z.fixtureId)),P=U.filter(Z=>!K.has(Z.fixtureId)).map(Z=>Z.fixtureId);P.length>0&&p(Z=>new Set([...Z,...P]))}}):x.jsxs("div",{className:"duplicate-check-section",children:[x.jsxs("p",{style:{color:"#666",fontSize:"11px"},children:["엔티티별로 중복 확인에 사용할 컬럼을 지정합니다. ",x.jsx("br",{}),"지정하지 않으면 unique index만 사용합니다."]}),x.jsx(Ct,{placeholder:"엔티티 선택",search:!0,selection:!0,clearable:!0,loading:t,options:e?.entities?.filter(U=>!v[U.id]).map(U=>({key:U.id,value:U.id,text:U.id}))||[],value:_,onChange:(U,{value:K})=>S(K)}),x.jsx(Ct,{placeholder:"중복 확인 컬럼 선택",multiple:!0,selection:!0,disabled:!z,options:z?.props.filter(U=>U.type==="virtual"?!1:U.type==="relation"?!!(U.relationType==="BelongsToOne"||U.relationType==="OneToOne"&&U.hasJoinColumn):!0).map(U=>({key:U.name,value:U.name,text:U.name}))||[],value:T,onChange:(U,{value:K})=>C(K)}),x.jsx(Le,{icon:"plus",color:"blue",size:"small",disabled:!_||T.length===0,onClick:q}),Object.keys(v).length>0&&x.jsx("div",{style:{display:"flex",flexWrap:"wrap",gap:"8px"},children:Object.entries(v).map(([U,K])=>x.jsxs(Yt,{size:"medium",style:{display:"flex",alignItems:"center",gap:"6px",padding:"8px 12px"},children:[x.jsx("span",{style:{fontWeight:"bold"},children:U}),x.jsxs("span",{style:{color:"#666"},children:["(",K.join(", "),")"]}),x.jsx(mt,{name:"delete",style:{cursor:"pointer",marginLeft:"4px"},onClick:()=>Q(U)})]},U))})]})]})}),x.jsx("div",{className:"fixture-main",children:x.jsxs("div",{className:"fixture-viewer",children:[x.jsx("div",{style:{display:"flex",justifyContent:"flex-end",marginBottom:"15px"},children:x.jsx(Le,{onClick:()=>y(b==="table"?"graph":"table"),content:b==="table"?"그래프 보기":"테이블 보기",icon:b==="table"?"sitemap":"table",basic:!0,color:"grey"})}),x.jsx(ps,{panes:F,activeIndex:h,onTabChange:(U,{activeIndex:K})=>{typeof K=="number"&&m(K)},style:{boxShadow:"0 5px 15px rgba(0, 0, 0, 0.08)",borderRadius:"12px",overflow:"hidden"}})]})})]})}function NQe({action:e,targets:t,conns:n}){const[r,o]=j.useState(!1),{doneModal:a}=Ls(),{form:i,register:s}=em(Ve({doShadowDbTesting:ct()}),{doShadowDbTesting:e==="apply"}),l=async()=>{o(!0);try{i.doShadowDbTesting&&await Xe.migrationsRunAction("shadow",t),await Xe.migrationsRunAction(e,t),a()}catch(u){Ht(u)}finally{o(!1)}};return x.jsx("div",{className:"form migration-commit-form",children:x.jsx(un,{padded:!0,basic:!0,loading:r,children:x.jsxs(un,{padded:!0,color:"green",children:[x.jsx("div",{className:"header-row",children:x.jsx(Or,{children:"Migrations Action Form"})}),x.jsxs(un,{basic:!0,children:[x.jsxs("div",{children:[x.jsxs("h4",{children:["Action: ",e.toUpperCase()]}),x.jsx("p",{children:" "})]}),x.jsxs("div",{className:"targets",children:[x.jsx("h4",{children:"Targets"}),x.jsx("div",{className:"conns",children:n.map(u=>x.jsxs("div",{className:gi("conn",{"is-targeted":t.includes(u.connKey)}),children:[t.includes(u.connKey)&&x.jsx(mt,{name:"check"}),u.name]},u.name))})]}),e==="apply"&&x.jsxs("div",{className:"shadow-db-testing",children:[x.jsx("h4",{children:"Shadow DB Testing"}),x.jsx(Kf,{...s("doShadowDbTesting")})]}),x.jsx("div",{className:"text-center",style:{marginTop:"2em"},children:x.jsx(Le,{color:"green",onClick:()=>l(),icon:"play",content:"Commit"})})]})]})})})}function DQe(e){const{data:t,error:n,refetch:r}=Xe.useMigrationStatus(),{status:o}=t??{},{preparedCodes:a,conns:i,codes:s}=o??{},l=o?.error,{openModal:u}=Ls(),d=!n&&!t,[p,h]=j.useState(!1),[m,b]=j.useState([]),[y,v]=j.useState([]),[k,_]=j.useState(!1),S=O=>{const R=(()=>{switch(O){case"ALL":return["test","fixture_remote","development_master","production_master"];case"LOCAL":return["test"];case"REMOTE":return["fixture_remote","development_master","production_master"];case"TESTING":return["test","fixture_remote"];case"FIXTURE":return["fixture_remote"]}})();R.filter($=>m.includes($)).length===R.length?b(m.filter($=>!R.includes($))):Qpe(R,m).length>0?b(R):b(ap([...m,...R]))},T=()=>{y.length===0||!confirm(`Are you sure to delete the selected ${y.length} migration codes?`)||(h(!0),Xe.migrationsDelCodes(y).then(()=>{r()}).catch(Ht).finally(()=>{h(!1)}))},C=()=>{h(!0),Xe.migrationsGeneratePreparedCodes().then(()=>{setTimeout(()=>{r()},500)}).catch(Ht).finally(()=>{h(!1)})},w=(O,R)=>{if(!i)return;const $=m;u(x.jsx(NQe,{action:O,targets:$,conns:i}),{onCompleted:()=>{r()}})},A=()=>{s&&(y.length===0?v(s.map(O=>O.name)):v([]))};return n?x.jsx("div",{className:"migrations-index",children:x.jsx("div",{className:"message-box error",children:n.message})}):x.jsx("div",{className:"migrations-index",children:x.jsxs(un,{className:"migrations-index",loading:p||d,children:[a&&x.jsxs("div",{className:"prepared",children:[x.jsxs("h3",{children:["Prepared Migration Codes"," ",x.jsxs("div",{className:"buttons",children:[x.jsx(Le,{icon:`toggle ${k?"on":"off"}`,size:"mini",color:"olive",content:"Toggle codes",onClick:()=>_(!k)}),x.jsx(Fl,{vertical:!0}),x.jsx(Le,{size:"mini",color:"green",icon:"play",content:"Generate",onClick:()=>C()})]})]}),x.jsxs(me,{celled:!0,selectable:!0,children:[x.jsx(me.Header,{children:x.jsxs(me.Row,{children:[x.jsx(me.HeaderCell,{children:"Type"}),x.jsx(me.HeaderCell,{children:"Table"}),x.jsx(me.HeaderCell,{children:"Name"}),x.jsx(me.HeaderCell,{children:"Code"})]})}),x.jsxs(me.Body,{children:[l&&x.jsx(me.Row,{className:"table-empty",children:x.jsx(me.Cell,{colSpan:6,children:l})}),!l&&a.length===0&&x.jsx(me.Row,{className:"table-empty",children:x.jsx(me.Cell,{colSpan:6,children:"No prepared migration codes."})}),a.map((O,R)=>x.jsxs(me.Row,{className:"prepared-code",children:[x.jsx(me.Cell,{collapsing:!0,children:O.type}),x.jsx(me.Cell,{collapsing:!0,children:O.table}),x.jsx(me.Cell,{collapsing:!0,children:O.title}),x.jsx(me.Cell,{style:{padding:0,width:700,textAlign:"center"},children:x.jsx(PQe,{code:O.formatted??"",open:k})})]},R))]})]}),x.jsx(Fl,{})]}),x.jsxs("div",{className:"codes",children:[x.jsx("h3",{children:"Migration Code Files"}),x.jsxs("div",{className:"tools",children:[x.jsx("div",{className:"code-buttons",children:x.jsx(Le,{size:"tiny",color:"red",icon:"trash",content:"Delete codes",disabled:y.length===0,onClick:()=>T()})}),x.jsx("div",{className:"conn-preset-buttons",children:["ALL","LOCAL","REMOTE","TESTING","FIXTURE"].map(O=>x.jsx(Le,{color:"black",size:"tiny",content:O,onClick:()=>S(O)},O))}),x.jsxs("div",{className:"conn-action-buttons",children:[x.jsx(Le,{size:"tiny",color:"green",icon:"play",content:"Apply to Latest",disabled:m.length===0||!!l,onClick:()=>w("apply")}),x.jsx(Le,{size:"tiny",color:"red",icon:"refresh",content:"Rollback!!",disabled:m.length===0||!!l,onClick:()=>w("rollback")})]})]}),i&&s&&x.jsxs(me,{celled:!0,selectable:!0,children:[x.jsx(me.Header,{children:x.jsxs(me.Row,{children:[x.jsxs(me.HeaderCell,{children:["Name"," ",x.jsx(Le,{icon:"check",size:"mini",color:"blue",onClick:()=>A()})]}),i.map((O,R)=>x.jsx(me.HeaderCell,{width:"2",className:gi({"conn-selected":m.includes(O.connKey)}),children:x.jsx(ar,{label:`${O.name} / ${O.status}`,disabled:O.status==="error"||!!l,checked:m.includes(O.connKey),onChange:($,N)=>{N.checked?b(ap([...m,O.connKey])):b(m.filter(M=>M!==O.connKey))}})},R))]})}),x.jsxs(me.Body,{children:[i.some(O=>O.status==="error"||!!l)&&x.jsx(me.Row,{className:"table-empty",children:x.jsx(me.Cell,{colSpan:6,children:x.jsx("b",{children:"Some connections are in error state. Please check the connection settings and try again."})})}),s.length===0&&x.jsx(me.Row,{className:"table-empty",children:x.jsx(me.Cell,{colSpan:6,children:"No migration code files"})}),s.map((O,R)=>x.jsxs(me.Row,{children:[x.jsxs(me.Cell,{children:[x.jsx(ar,{label:O.name,checked:y.includes(O.name),onChange:($,N)=>{N.checked?v(ap([...y,O.name])):v(y.filter(M=>M!==O.name))}})," "," ",x.jsx(Le,{size:"mini",icon:"code",onClick:()=>{Xe.openVscode({absPath:O.path})}})]}),i.map(($,N)=>x.jsx(me.Cell,{className:gi("conn-status",{"conn-selected":m.includes($.connKey)}),children:$.pending.includes(O.name)?x.jsx(Yt,{size:"mini",color:"yellow",icon:"minus",content:"PENDING"}):$.status==="error"||l?x.jsx(Yt,{size:"mini",color:"red",icon:"times",content:"ERROR"}):x.jsx(Yt,{size:"mini",color:"green",icon:"check",content:"APPLIED"})},N))]},R))]})]})]})]})})}function PQe({code:e,open:t}){return x.jsx("div",{className:"code-viewer",children:t?x.jsx("code",{children:e}):x.jsx("div",{children:"Code is collapsed"})})}function $Qe({}){const{data:e}=Xe.useEntities(),{entities:t}=e??{},[n,r]=j.useState({templateGroupName:"Entity",entityIds:[],templateKeys:[],enumIds:[]}),[o,a]=j.useState({open:!1,pathAndCodes:null}),[i,s]=j.useState({}),l=(t??[]).filter(w=>!w.parentId),u=[{name:"Entity",templateKeys:["model","model_test","view_list","view_search_input","view_form","view_id_async_select"]},{name:"Enums",templateKeys:["view_enums_select","view_enums_dropdown"]}],d=(t??[]).filter(w=>n.entityIds.includes(w.id)||n.entityIds.includes(w.parentId??"")).flatMap(w=>Object.keys(w.enumLabels)),p=w=>{r({...n,entityIds:w,enumIds:d.filter(A=>n.enumIds.includes(A))})},h=(w,A)=>{const O=u.find(R=>R.name===w);O&&r({...n,templateGroupName:w,templateKeys:O.templateKeys.filter(R=>A.includes(R)),enumIds:w==="Entity"?[]:n.enumIds})},m=w=>{r({...n,enumIds:d.filter(A=>w.includes(A))})},{data:b,isLoading:y,refetch:v}=Xe.useScaffoldingStatus(n),{statuses:k}=b??{},_=w=>[w.entityId,w.templateKey,w.enumId].join("///"),S=()=>{if(!k)return;const w=k.filter(O=>O.isExists),A=w.every(O=>i[_(O)]?.overwrite??!1);s(A?{}:w.reduce((O,R)=>(O[_(R)]={overwrite:!0},O),{}))},T=()=>{if(!k)return;const w=k.map(A=>({entityId:A.entityId,templateKey:A.templateKey,enumId:A.enumId,overwrite:i[_(A)]?.overwrite??!1}));Xe.scaffoldingGenerate(w).then(()=>{v()}).catch(Ht)},C=w=>{Xe.scaffoldingPreview(w).then(({pathAndCodes:A})=>{a({open:!0,pathAndCodes:A})}).catch(Ht)};return x.jsxs("div",{className:"scaffolding-index",children:[x.jsxs("div",{className:"entities",children:[x.jsx("h3",{children:"Entities"}),x.jsx("div",{className:"button-set",children:n.entityIds.length!==l.length?x.jsx(Le,{size:"mini",icon:"check",content:"Check all entities",onClick:()=>p(l.map(w=>w.id))}):x.jsx(Le,{size:"mini",icon:"check",content:"Uncheck all entities",onClick:()=>p([])})}),l.map(w=>x.jsx("div",{className:"entity",id:w.id,children:x.jsx(ar,{label:w.id,checked:n.entityIds.includes(w.id),onChange:(A,{checked:O})=>{p(O?[...n.entityIds,w.id]:n.entityIds.filter(R=>R!==w.id))}})},w.id))]}),x.jsx("div",{className:"template-groups",children:u.map(w=>x.jsxs("div",{className:"template-group",children:[x.jsxs("h4",{children:["Template: ",w.name]}),x.jsx("div",{className:"button-set",children:n.templateGroupName!==w.name||n.templateKeys.length!==w.templateKeys.length?x.jsx(Le,{size:"mini",icon:"check",content:"Check all",onClick:()=>h(w.name,w.templateKeys)}):x.jsx(Le,{size:"mini",icon:"check",content:"Uncheck all",onClick:()=>h(w.name,[])})}),w.templateKeys.map(A=>x.jsx("div",{className:"template-key",children:x.jsx(ar,{label:A,checked:n.templateGroupName===w.name&&n.templateKeys.includes(A),onChange:(O,{checked:R})=>{R?h(w.name,[...n.templateKeys,A]):h(w.name,n.templateKeys.filter($=>$!==A))}})},A))]},w.name))}),n.templateGroupName==="Enums"&&x.jsxs("div",{className:"enums-list",children:[x.jsx("h4",{children:"Enums"}),x.jsx("div",{className:"button-set",children:n.enumIds.length!==d.length?x.jsx(Le,{size:"mini",icon:"check",content:"Check all enums",onClick:()=>m(d)}):x.jsx(Le,{size:"mini",icon:"check",content:"Uncheck all enums",onClick:()=>m([])})}),d.map(w=>x.jsx("div",{className:"enums",children:x.jsx(ar,{label:w,checked:n.enumIds.includes(w),onChange:(A,{checked:O})=>{m(O?[...n.enumIds,w]:n.enumIds.filter(R=>R!==w))}})},w))]}),x.jsx("div",{className:"content",children:x.jsxs(Ce,{children:[!k&&!y&&x.jsxs("div",{className:"message-box warning",children:["Please select EntityIDs / TemplateKeys",n.templateGroupName==="Enums"?" / EnumIDs":""," to generate"]}),k&&x.jsxs("div",{className:"statuses",children:[k.length>0&&x.jsx(Le,{size:"small",color:"green",icon:"play",content:`Generate ${k.length} template(s) — ${Object.keys(i).length} overwrite`,onClick:()=>T()}),x.jsxs(me,{celled:!0,selectable:!0,children:[x.jsx(me.Header,{children:x.jsxs(me.Row,{children:[x.jsx(me.HeaderCell,{children:"Entity"}),x.jsx(me.HeaderCell,{children:"TemplateKey"}),n.templateGroupName==="Enums"&&x.jsx(me.HeaderCell,{children:"EnumId"}),x.jsx(me.HeaderCell,{children:"Path"}),x.jsx(me.HeaderCell,{children:"IsExists"}),x.jsx(me.HeaderCell,{collapsing:!0,children:x.jsx(Le,{size:"mini",icon:"check",content:"Overwrite",onClick:()=>S()})}),x.jsx(me.HeaderCell,{children:"Preview"})]})}),x.jsx(me.Body,{children:k.map((w,A)=>x.jsxs(me.Row,{positive:!w.isExists,negative:w.isExists,children:[x.jsx(me.Cell,{collapsing:!0,children:w.entityId}),x.jsx(me.Cell,{collapsing:!0,children:w.templateKey}),n.templateGroupName==="Enums"&&x.jsx(me.Cell,{collapsing:!0,children:w.enumId}),x.jsx(me.Cell,{children:w.subPath}),x.jsx(me.Cell,{children:w.isExists?x.jsx(Le,{icon:"code",size:"mini",color:"blue",onClick:()=>{Xe.openVscode({absPath:w.fullPath})}}):x.jsx(mt,{name:"x"})}),x.jsx(me.Cell,{children:w.isExists&&x.jsx(Ce.Group,{children:x.jsx(Ce.Field,{children:x.jsx(ar,{checked:i[_(w)]?.overwrite??!1,onChange:(O,{checked:R})=>{s({...i,[_(w)]:{overwrite:R??!1}})}})})})}),x.jsx(me.Cell,{collapsing:!0,children:x.jsx(Le,{size:"mini",content:"Preview",color:"purple",onClick:()=>C(w)})})]},A))})]})]})]})}),x.jsxs(Yn,{open:o.open,onClose:()=>{a({open:!1,pathAndCodes:null})},children:[x.jsx(Yn.Header,{children:"Preview"}),x.jsx(Yn.Content,{children:x.jsx(Yn.Description,{children:o.pathAndCodes?.map(w=>x.jsxs("div",{className:"preview-item",children:[x.jsx("h4",{children:w.path}),x.jsx("code",{children:w.code})]},w.path))})})]})]})}const MQe=new Ehe({defaultOptions:{queries:{refetchOnWindowFocus:!0,retry:3,retryDelay:3e3}}});Roe.createRoot(document.getElementById("root")).render(x.jsx(_he,{client:MQe,children:x.jsx(lae,{basename:"/sonamu-ui/",children:x.jsxs(aae,{children:[x.jsxs(ss,{path:"/",element:x.jsx(mge,{}),children:[x.jsx(ss,{path:"/entities",element:x.jsx(Lze,{}),children:x.jsx(ss,{path:":entityId",element:x.jsx(qze,{})})}),x.jsx(ss,{path:"/migrations",element:x.jsx(DQe,{})}),x.jsx(ss,{path:"/scaffolding",element:x.jsx($Qe,{})}),x.jsx(ss,{path:"/fixture",element:x.jsx(RQe,{})})]}),x.jsx(ss,{path:"*",element:x.jsx("div",{children:"Page Not Found"})})]})})}))});export default LQe();
92
+ \`\`\``,components:{code({children:m,className:b,node:y,ref:v,...k}){const _=String(m).trimEnd();return x.jsxs("div",{className:"code",children:[x.jsxs("div",{className:"code-header",children:[x.jsx("span",{children:n??t}),x.jsxs("div",{children:[o&&x.jsx(ar,{label:a.every(S=>S)?"전체 해제":"전체 선택",checked:a.every(S=>S),onChange:()=>{const S=a.every(T=>T);i(a.map(()=>!S))}}),x.jsxs(Le,{icon:!0,onClick:()=>h(_),size:"tiny",children:[x.jsx(mt,{name:u?"check circle outline":"clipboard outline"}),u?"복사 완료":"복사"]})]})]}),x.jsx(AQe,{...k,children:_,language:t,style:Bne[r??"materialDark"],renderer:({rows:S,stylesheet:T})=>x.jsx("div",{style:{position:"relative"},children:S.map((C,w)=>{const A=a[w]??!1,O=s===w;return x.jsxs("div",{className:`code-line ${O?"hovered":""}`,style:A?{backgroundColor:"rgba(0, 123, 255, 0.1)"}:{},onMouseEnter:()=>l(w),onMouseLeave:()=>l(null),onClick:()=>o&&p(w),children:[o&&x.jsx(ar,{checked:A,onClick:R=>R.stopPropagation(),onChange:()=>p(w)}),x.jsx("span",{children:C.children?.map((R,$)=>R.type==="element"?x.jsx("span",{className:R.properties.className.join(" "),style:{...R.properties.className.reduce((N,M)=>T[M]?{...N,...T[M]}:N,{}),fontWeight:"normal"},children:R.children.map((N,M)=>x.jsx("span",{children:N.value},M))},$):x.jsx("span",{children:R.value},$))})]},w)})})})]})}}})};function IQe({fixtureRecords:e,onRelationToggle:t,selectedIds:n,setFixtureRecords:r}){const o=Qv(e,a=>a.entityId);return x.jsx("div",{className:"fixture-record-viewer",children:Object.entries(o).map(([a,i])=>x.jsx(jte,{fixtures:i??[],selectedIds:n,onRelationToggle:t,setFixtureRecords:r},a))})}const sG=["development_master","production_master","fixture_remote","test"];function RQe(){const{data:e,isLoading:t}=Xe.useEntities(),[n,r]=j.useState("development_master"),[o,a]=j.useState("test"),[i,s]=j.useState([]),[l,u]=j.useState([]),[d,p]=j.useState(new Set),[h,m]=j.useState(0),[b,y]=j.useState("table"),[v,k]=j.useState({}),[_,S]=j.useState(""),[T,C]=j.useState([]),[w,A]=j.useState(!1),[O,R]=j.useState("chat"),{form:$,register:N}=em(Ve({entityId:ie(),field:ie(),value:ie(),searchType:Br(["equals","like"])}),{entityId:"",field:"id",value:"",searchType:"equals"}),[M,V]=j.useState(null),z=e?.entities?.find(U=>U.id===_)??null,H=()=>{if(!$.entityId||!$.field||!$.value)return;m(0),s([]),u([]),p(new Set);const U=Object.keys(v).length>0?{columns:v}:void 0;Xe.getFixtures(n,o,$,U).then(K=>{s(K),p(new Set(K.map(P=>P.fixtureId)))}).catch(Ht)},W=()=>{i.length!==0&&(m(1),Xe.importFixtures(o,i).then(U=>{u(U)}).catch(Ht))},G=async(U,K,P,Z)=>{const L=`${K}#${P}`,X=Object.keys(v).length>0?{columns:v}:void 0;if(Z)Xe.getFixtures(n,o,{entityId:K,field:"id",value:String(P),searchType:"equals"},X).then(ne=>{const te=i.find(pe=>pe.fixtureId===U);te&&te.fetchedRecords.push(L);const ae=ne.filter(pe=>!i.some(be=>be.fixtureId===pe.fixtureId));s(pe=>Array.from([...pe,...ae])),p(pe=>{const be=new Set(pe);return ae.forEach(xe=>{be.add(xe.fixtureId)}),be})}).catch(Ht);else{const ne=i.find(pe=>pe.fixtureId===U);if(!ne)return;ne.fetchedRecords=ne.fetchedRecords.filter(pe=>pe!==L);const te=new Set([L]),ae=i.find(pe=>pe.fixtureId===L);if(ae?.fetchedRecords.length){const pe=new Set([U]);i.forEach(be=>{be.fixtureId!==L&&be.fetchedRecords.forEach(xe=>{pe.add(xe)})}),ae?.fetchedRecords.forEach(be=>{pe.has(be)||te.add(be)})}else{const pe=new Set,be=xe=>{pe.has(xe)||(pe.add(xe),i.forEach(Ie=>{Ie.belongsRecords.includes(xe)&&be(Ie.fixtureId)}))};be(L)}p(pe=>{const be=new Set(pe);return te.forEach(xe=>{be.delete(xe)}),be}),s(pe=>pe.filter(be=>!te.has(be.fixtureId)))}},q=()=>{!_||T.length===0||(k(U=>({...U,[_]:T})),S(""),C([]))},Q=U=>{k(K=>{const{[U]:P,...Z}=K;return Z})};j.useEffect(()=>{if($.entityId&&e?.entities){const U=e.entities.find(K=>K.id===$.entityId);U&&V(U)}},[$.entityId,e]),j.useEffect(()=>{C([])},[]);const F=[{menuItem:"Fixture Record Viewer",render:()=>x.jsx(ps.Pane,{children:b==="table"?x.jsx(IQe,{fixtureRecords:i,onRelationToggle:G,selectedIds:d,setFixtureRecords:s}):x.jsx(z5e,{fixtures:i,selectedIds:d,onRelationToggle:G,setFixtureRecords:s})})},{menuItem:"Fixture Code Viewer",render:()=>x.jsx(ps.Pane,{children:e?.entities&&l.length>0&&x.jsx(CQe,{entities:e.entities,fixtureResults:l,targetDB:o})})}];return x.jsxs("div",{className:"fixture-index",children:[x.jsx("div",{className:"fixture-sidebar",children:x.jsxs(un,{className:"fixture-header",children:[x.jsxs("div",{className:"search-section",children:[x.jsxs("div",{className:"search-title",children:[x.jsx(mt,{name:"search",style:{marginRight:"5px"}}),"검색 대상 설정"]}),x.jsx("div",{className:"db-dropdown-wrapper",children:x.jsx(Ct,{fluid:!0,placeholder:"검색할 DB 선택",selection:!0,options:sG.map(U=>({key:U,value:U,text:U.replace("_master","")})),value:n,onChange:(U,{value:K})=>r(K)})}),x.jsx("div",{style:{flexGrow:1,minWidth:"200px"},children:x.jsx(Ct,{fluid:!0,placeholder:"엔티티 선택",search:!0,selection:!0,loading:t,options:e?.entities?.map(U=>({key:U.id,value:U.id,text:U.id}))||[],...N("entityId")})}),M&&x.jsxs(x.Fragment,{children:[x.jsx(Ct,{placeholder:"컬럼 선택",selection:!0,options:M.props.filter(U=>U.type==="virtual"?!1:U.type==="relation"?!!(U.relationType==="BelongsToOne"||U.relationType==="OneToOne"&&U.hasJoinColumn):!0).map(U=>({key:U.name,value:U.name,text:U.name})),...N("field")}),x.jsx(vr,{placeholder:"검색 값 입력",...N("value"),style:{flexGrow:1}}),x.jsx(Ct,{selection:!0,options:[{key:"equals",text:"Equals",value:"equals"},{key:"like",text:"Like",value:"like"}],...N("searchType")})]}),x.jsx(Le,{onClick:H,disabled:!$.entityId||!$.field||!$.value||t,loading:t,primary:!0,content:"검색"})]}),x.jsxs("div",{className:"save-section",children:[x.jsxs("button",{type:"button",className:"save-title",style:{cursor:"pointer",background:"none",border:"none",padding:0,display:"flex",alignItems:"center",width:"100%",textAlign:"left"},onClick:()=>A(!w),children:[x.jsx(mt,{name:w?"chevron down":"chevron right"}),x.jsx(mt,{name:"database",style:{marginRight:"5px"}}),"저장 DB 설정",i.length>0&&(()=>{const U=i.filter(K=>{const P=!!K.target,Z=!!K.unique;return!!(!P&&!Z||K.override&&(P||Z))});return x.jsxs(Yt,{color:"green",size:"small",style:{marginLeft:"auto"},children:[U.length,"개 저장 예정"]})})()]}),x.jsx("div",{className:"db-dropdown-wrapper",children:x.jsx(Ct,{fluid:!0,placeholder:"저장할 대상 DB 선택",header:"Fixture Target DB",selection:!0,options:sG.map(U=>({key:U,value:U,text:U})),value:o,onChange:(U,{value:K})=>a(K)})}),x.jsx(Le,{onClick:W,color:"blue",content:"저장",disabled:i.length===0}),w&&i.length>0&&(()=>{const K=i.filter(P=>{const Z=!!P.target,L=!!P.unique;return!!(!Z&&!L||P.override&&(Z||L))}).reduce((P,Z)=>(P[Z.entityId]||(P[Z.entityId]=[]),P[Z.entityId].push(Z),P),{});return x.jsxs("div",{style:{marginTop:"10px",marginBottom:"10px"},children:[x.jsx("p",{style:{color:"#666",fontSize:"12px",marginBottom:"10px"},children:"저장될 픽스쳐 목록"}),x.jsx("div",{className:"fixture-record-group",children:Object.entries(K).map(([P,Z])=>x.jsxs("div",{style:{padding:"8px",backgroundColor:"#f9f9f9",borderRadius:"4px"},children:[x.jsxs("div",{style:{fontWeight:"bold",marginBottom:"4px"},children:[P," (",Z.length,"개)"]}),x.jsx("div",{style:{fontSize:"12px",color:"#555"},children:Z.map(L=>{const X=!!L.target,ne=!!L.unique;let te="신규";return L.override&&(X||ne)&&(te="덮어쓰기"),x.jsxs(Yt,{size:"tiny",style:{margin:"2px"},children:["#",L.id,x.jsx(Yt.Detail,{children:te})]},L.fixtureId)})})]},P))})]})})()]}),O==="chat"?x.jsx(Gze,{fixtureRecords:i,onUpdateFixtures:U=>{s(U);const K=new Set(i.map(Z=>Z.fixtureId)),P=U.filter(Z=>!K.has(Z.fixtureId)).map(Z=>Z.fixtureId);P.length>0&&p(Z=>new Set([...Z,...P]))}}):x.jsxs("div",{className:"duplicate-check-section",children:[x.jsxs("p",{style:{color:"#666",fontSize:"11px"},children:["엔티티별로 중복 확인에 사용할 컬럼을 지정합니다. ",x.jsx("br",{}),"지정하지 않으면 unique index만 사용합니다."]}),x.jsx(Ct,{placeholder:"엔티티 선택",search:!0,selection:!0,clearable:!0,loading:t,options:e?.entities?.filter(U=>!v[U.id]).map(U=>({key:U.id,value:U.id,text:U.id}))||[],value:_,onChange:(U,{value:K})=>S(K)}),x.jsx(Ct,{placeholder:"중복 확인 컬럼 선택",multiple:!0,selection:!0,disabled:!z,options:z?.props.filter(U=>U.type==="virtual"?!1:U.type==="relation"?!!(U.relationType==="BelongsToOne"||U.relationType==="OneToOne"&&U.hasJoinColumn):!0).map(U=>({key:U.name,value:U.name,text:U.name}))||[],value:T,onChange:(U,{value:K})=>C(K)}),x.jsx(Le,{icon:"plus",color:"blue",size:"small",disabled:!_||T.length===0,onClick:q}),Object.keys(v).length>0&&x.jsx("div",{style:{display:"flex",flexWrap:"wrap",gap:"8px"},children:Object.entries(v).map(([U,K])=>x.jsxs(Yt,{size:"medium",style:{display:"flex",alignItems:"center",gap:"6px",padding:"8px 12px"},children:[x.jsx("span",{style:{fontWeight:"bold"},children:U}),x.jsxs("span",{style:{color:"#666"},children:["(",K.join(", "),")"]}),x.jsx(mt,{name:"delete",style:{cursor:"pointer",marginLeft:"4px"},onClick:()=>Q(U)})]},U))})]})]})}),x.jsx("div",{className:"fixture-main",children:x.jsxs("div",{className:"fixture-viewer",children:[x.jsx("div",{style:{display:"flex",justifyContent:"flex-end",marginBottom:"15px"},children:x.jsx(Le,{onClick:()=>y(b==="table"?"graph":"table"),content:b==="table"?"그래프 보기":"테이블 보기",icon:b==="table"?"sitemap":"table",basic:!0,color:"grey"})}),x.jsx(ps,{panes:F,activeIndex:h,onTabChange:(U,{activeIndex:K})=>{typeof K=="number"&&m(K)},style:{boxShadow:"0 5px 15px rgba(0, 0, 0, 0.08)",borderRadius:"12px",overflow:"hidden"}})]})})]})}function NQe({action:e,targets:t,conns:n}){const[r,o]=j.useState(!1),{doneModal:a}=Ls(),{form:i,register:s}=em(Ve({doShadowDbTesting:ct()}),{doShadowDbTesting:e==="apply"}),l=async()=>{o(!0);try{i.doShadowDbTesting&&await Xe.migrationsRunAction("shadow",t),await Xe.migrationsRunAction(e,t),a()}catch(u){Ht(u)}finally{o(!1)}};return x.jsx("div",{className:"form migration-commit-form",children:x.jsx(un,{padded:!0,basic:!0,loading:r,children:x.jsxs(un,{padded:!0,color:"green",children:[x.jsx("div",{className:"header-row",children:x.jsx(Or,{children:"Migrations Action Form"})}),x.jsxs(un,{basic:!0,children:[x.jsxs("div",{children:[x.jsxs("h4",{children:["Action: ",e.toUpperCase()]}),x.jsx("p",{children:" "})]}),x.jsxs("div",{className:"targets",children:[x.jsx("h4",{children:"Targets"}),x.jsx("div",{className:"conns",children:n.map(u=>x.jsxs("div",{className:gi("conn",{"is-targeted":t.includes(u.connKey)}),children:[t.includes(u.connKey)&&x.jsx(mt,{name:"check"}),u.name]},u.name))})]}),e==="apply"&&x.jsxs("div",{className:"shadow-db-testing",children:[x.jsx("h4",{children:"Shadow DB Testing"}),x.jsx(Kf,{...s("doShadowDbTesting")})]}),x.jsx("div",{className:"text-center",style:{marginTop:"2em"},children:x.jsx(Le,{color:"green",onClick:()=>l(),icon:"play",content:"Commit"})})]})]})})})}function DQe(e){const{data:t,error:n,refetch:r}=Xe.useMigrationStatus(),{status:o}=t??{},{preparedCodes:a,conns:i,codes:s}=o??{},l=o?.error,{openModal:u}=Ls(),d=!n&&!t,[p,h]=j.useState(!1),[m,b]=j.useState([]),[y,v]=j.useState([]),[k,_]=j.useState(!1),S=O=>{const R=(()=>{switch(O){case"ALL":return["test","fixture","development_master","production_master"];case"LOCAL":return["test"];case"REMOTE":return["fixture","development_master","production_master"];case"TESTING":return["test","fixture"];case"FIXTURE":return["fixture"]}})();R.filter($=>m.includes($)).length===R.length?b(m.filter($=>!R.includes($))):Qpe(R,m).length>0?b(R):b(ap([...m,...R]))},T=()=>{y.length===0||!confirm(`Are you sure to delete the selected ${y.length} migration codes?`)||(h(!0),Xe.migrationsDelCodes(y).then(()=>{r()}).catch(Ht).finally(()=>{h(!1)}))},C=()=>{h(!0),Xe.migrationsGeneratePreparedCodes().then(()=>{setTimeout(()=>{r()},500)}).catch(Ht).finally(()=>{h(!1)})},w=(O,R)=>{if(!i)return;const $=m;u(x.jsx(NQe,{action:O,targets:$,conns:i}),{onCompleted:()=>{r()}})},A=()=>{s&&(y.length===0?v(s.map(O=>O.name)):v([]))};return n?x.jsx("div",{className:"migrations-index",children:x.jsx("div",{className:"message-box error",children:n.message})}):x.jsx("div",{className:"migrations-index",children:x.jsxs(un,{className:"migrations-index",loading:p||d,children:[a&&x.jsxs("div",{className:"prepared",children:[x.jsxs("h3",{children:["Prepared Migration Codes"," ",x.jsxs("div",{className:"buttons",children:[x.jsx(Le,{icon:`toggle ${k?"on":"off"}`,size:"mini",color:"olive",content:"Toggle codes",onClick:()=>_(!k)}),x.jsx(Fl,{vertical:!0}),x.jsx(Le,{size:"mini",color:"green",icon:"play",content:"Generate",onClick:()=>C()})]})]}),x.jsxs(me,{celled:!0,selectable:!0,children:[x.jsx(me.Header,{children:x.jsxs(me.Row,{children:[x.jsx(me.HeaderCell,{children:"Type"}),x.jsx(me.HeaderCell,{children:"Table"}),x.jsx(me.HeaderCell,{children:"Name"}),x.jsx(me.HeaderCell,{children:"Code"})]})}),x.jsxs(me.Body,{children:[l&&x.jsx(me.Row,{className:"table-empty",children:x.jsx(me.Cell,{colSpan:6,children:l})}),!l&&a.length===0&&x.jsx(me.Row,{className:"table-empty",children:x.jsx(me.Cell,{colSpan:6,children:"No prepared migration codes."})}),a.map((O,R)=>x.jsxs(me.Row,{className:"prepared-code",children:[x.jsx(me.Cell,{collapsing:!0,children:O.type}),x.jsx(me.Cell,{collapsing:!0,children:O.table}),x.jsx(me.Cell,{collapsing:!0,children:O.title}),x.jsx(me.Cell,{style:{padding:0,width:700,textAlign:"center"},children:x.jsx(PQe,{code:O.formatted??"",open:k})})]},R))]})]}),x.jsx(Fl,{})]}),x.jsxs("div",{className:"codes",children:[x.jsx("h3",{children:"Migration Code Files"}),x.jsxs("div",{className:"tools",children:[x.jsx("div",{className:"code-buttons",children:x.jsx(Le,{size:"tiny",color:"red",icon:"trash",content:"Delete codes",disabled:y.length===0,onClick:()=>T()})}),x.jsx("div",{className:"conn-preset-buttons",children:["ALL","LOCAL","REMOTE","TESTING","FIXTURE"].map(O=>x.jsx(Le,{color:"black",size:"tiny",content:O,onClick:()=>S(O)},O))}),x.jsxs("div",{className:"conn-action-buttons",children:[x.jsx(Le,{size:"tiny",color:"green",icon:"play",content:"Apply to Latest",disabled:m.length===0||!!l,onClick:()=>w("apply")}),x.jsx(Le,{size:"tiny",color:"red",icon:"refresh",content:"Rollback!!",disabled:m.length===0||!!l,onClick:()=>w("rollback")})]})]}),i&&s&&x.jsxs(me,{celled:!0,selectable:!0,children:[x.jsx(me.Header,{children:x.jsxs(me.Row,{children:[x.jsxs(me.HeaderCell,{children:["Name"," ",x.jsx(Le,{icon:"check",size:"mini",color:"blue",onClick:()=>A()})]}),i.map((O,R)=>x.jsx(me.HeaderCell,{width:"2",className:gi({"conn-selected":m.includes(O.connKey)}),children:x.jsx(ar,{label:`${O.name} / ${O.status}`,disabled:O.status==="error"||!!l,checked:m.includes(O.connKey),onChange:($,N)=>{N.checked?b(ap([...m,O.connKey])):b(m.filter(M=>M!==O.connKey))}})},R))]})}),x.jsxs(me.Body,{children:[i.some(O=>O.status==="error"||!!l)&&x.jsx(me.Row,{className:"table-empty",children:x.jsx(me.Cell,{colSpan:6,children:x.jsx("b",{children:"Some connections are in error state. Please check the connection settings and try again."})})}),s.length===0&&x.jsx(me.Row,{className:"table-empty",children:x.jsx(me.Cell,{colSpan:6,children:"No migration code files"})}),s.map((O,R)=>x.jsxs(me.Row,{children:[x.jsxs(me.Cell,{children:[x.jsx(ar,{label:O.name,checked:y.includes(O.name),onChange:($,N)=>{N.checked?v(ap([...y,O.name])):v(y.filter(M=>M!==O.name))}})," "," ",x.jsx(Le,{size:"mini",icon:"code",onClick:()=>{Xe.openVscode({absPath:O.path})}})]}),i.map(($,N)=>x.jsx(me.Cell,{className:gi("conn-status",{"conn-selected":m.includes($.connKey)}),children:$.pending.includes(O.name)?x.jsx(Yt,{size:"mini",color:"yellow",icon:"minus",content:"PENDING"}):$.status==="error"||l?x.jsx(Yt,{size:"mini",color:"red",icon:"times",content:"ERROR"}):x.jsx(Yt,{size:"mini",color:"green",icon:"check",content:"APPLIED"})},N))]},R))]})]})]})]})})}function PQe({code:e,open:t}){return x.jsx("div",{className:"code-viewer",children:t?x.jsx("code",{children:e}):x.jsx("div",{children:"Code is collapsed"})})}function $Qe({}){const{data:e}=Xe.useEntities(),{entities:t}=e??{},[n,r]=j.useState({templateGroupName:"Entity",entityIds:[],templateKeys:[],enumIds:[]}),[o,a]=j.useState({open:!1,pathAndCodes:null}),[i,s]=j.useState({}),l=(t??[]).filter(w=>!w.parentId),u=[{name:"Entity",templateKeys:["model","model_test","view_list","view_search_input","view_form","view_id_async_select"]},{name:"Enums",templateKeys:["view_enums_select","view_enums_dropdown"]}],d=(t??[]).filter(w=>n.entityIds.includes(w.id)||n.entityIds.includes(w.parentId??"")).flatMap(w=>Object.keys(w.enumLabels)),p=w=>{r({...n,entityIds:w,enumIds:d.filter(A=>n.enumIds.includes(A))})},h=(w,A)=>{const O=u.find(R=>R.name===w);O&&r({...n,templateGroupName:w,templateKeys:O.templateKeys.filter(R=>A.includes(R)),enumIds:w==="Entity"?[]:n.enumIds})},m=w=>{r({...n,enumIds:d.filter(A=>w.includes(A))})},{data:b,isLoading:y,refetch:v}=Xe.useScaffoldingStatus(n),{statuses:k}=b??{},_=w=>[w.entityId,w.templateKey,w.enumId].join("///"),S=()=>{if(!k)return;const w=k.filter(O=>O.isExists),A=w.every(O=>i[_(O)]?.overwrite??!1);s(A?{}:w.reduce((O,R)=>(O[_(R)]={overwrite:!0},O),{}))},T=()=>{if(!k)return;const w=k.map(A=>({entityId:A.entityId,templateKey:A.templateKey,enumId:A.enumId,overwrite:i[_(A)]?.overwrite??!1}));Xe.scaffoldingGenerate(w).then(()=>{v()}).catch(Ht)},C=w=>{Xe.scaffoldingPreview(w).then(({pathAndCodes:A})=>{a({open:!0,pathAndCodes:A})}).catch(Ht)};return x.jsxs("div",{className:"scaffolding-index",children:[x.jsxs("div",{className:"entities",children:[x.jsx("h3",{children:"Entities"}),x.jsx("div",{className:"button-set",children:n.entityIds.length!==l.length?x.jsx(Le,{size:"mini",icon:"check",content:"Check all entities",onClick:()=>p(l.map(w=>w.id))}):x.jsx(Le,{size:"mini",icon:"check",content:"Uncheck all entities",onClick:()=>p([])})}),l.map(w=>x.jsx("div",{className:"entity",id:w.id,children:x.jsx(ar,{label:w.id,checked:n.entityIds.includes(w.id),onChange:(A,{checked:O})=>{p(O?[...n.entityIds,w.id]:n.entityIds.filter(R=>R!==w.id))}})},w.id))]}),x.jsx("div",{className:"template-groups",children:u.map(w=>x.jsxs("div",{className:"template-group",children:[x.jsxs("h4",{children:["Template: ",w.name]}),x.jsx("div",{className:"button-set",children:n.templateGroupName!==w.name||n.templateKeys.length!==w.templateKeys.length?x.jsx(Le,{size:"mini",icon:"check",content:"Check all",onClick:()=>h(w.name,w.templateKeys)}):x.jsx(Le,{size:"mini",icon:"check",content:"Uncheck all",onClick:()=>h(w.name,[])})}),w.templateKeys.map(A=>x.jsx("div",{className:"template-key",children:x.jsx(ar,{label:A,checked:n.templateGroupName===w.name&&n.templateKeys.includes(A),onChange:(O,{checked:R})=>{R?h(w.name,[...n.templateKeys,A]):h(w.name,n.templateKeys.filter($=>$!==A))}})},A))]},w.name))}),n.templateGroupName==="Enums"&&x.jsxs("div",{className:"enums-list",children:[x.jsx("h4",{children:"Enums"}),x.jsx("div",{className:"button-set",children:n.enumIds.length!==d.length?x.jsx(Le,{size:"mini",icon:"check",content:"Check all enums",onClick:()=>m(d)}):x.jsx(Le,{size:"mini",icon:"check",content:"Uncheck all enums",onClick:()=>m([])})}),d.map(w=>x.jsx("div",{className:"enums",children:x.jsx(ar,{label:w,checked:n.enumIds.includes(w),onChange:(A,{checked:O})=>{m(O?[...n.enumIds,w]:n.enumIds.filter(R=>R!==w))}})},w))]}),x.jsx("div",{className:"content",children:x.jsxs(Ce,{children:[!k&&!y&&x.jsxs("div",{className:"message-box warning",children:["Please select EntityIDs / TemplateKeys",n.templateGroupName==="Enums"?" / EnumIDs":""," to generate"]}),k&&x.jsxs("div",{className:"statuses",children:[k.length>0&&x.jsx(Le,{size:"small",color:"green",icon:"play",content:`Generate ${k.length} template(s) — ${Object.keys(i).length} overwrite`,onClick:()=>T()}),x.jsxs(me,{celled:!0,selectable:!0,children:[x.jsx(me.Header,{children:x.jsxs(me.Row,{children:[x.jsx(me.HeaderCell,{children:"Entity"}),x.jsx(me.HeaderCell,{children:"TemplateKey"}),n.templateGroupName==="Enums"&&x.jsx(me.HeaderCell,{children:"EnumId"}),x.jsx(me.HeaderCell,{children:"Path"}),x.jsx(me.HeaderCell,{children:"IsExists"}),x.jsx(me.HeaderCell,{collapsing:!0,children:x.jsx(Le,{size:"mini",icon:"check",content:"Overwrite",onClick:()=>S()})}),x.jsx(me.HeaderCell,{children:"Preview"})]})}),x.jsx(me.Body,{children:k.map((w,A)=>x.jsxs(me.Row,{positive:!w.isExists,negative:w.isExists,children:[x.jsx(me.Cell,{collapsing:!0,children:w.entityId}),x.jsx(me.Cell,{collapsing:!0,children:w.templateKey}),n.templateGroupName==="Enums"&&x.jsx(me.Cell,{collapsing:!0,children:w.enumId}),x.jsx(me.Cell,{children:w.subPath}),x.jsx(me.Cell,{children:w.isExists?x.jsx(Le,{icon:"code",size:"mini",color:"blue",onClick:()=>{Xe.openVscode({absPath:w.fullPath})}}):x.jsx(mt,{name:"x"})}),x.jsx(me.Cell,{children:w.isExists&&x.jsx(Ce.Group,{children:x.jsx(Ce.Field,{children:x.jsx(ar,{checked:i[_(w)]?.overwrite??!1,onChange:(O,{checked:R})=>{s({...i,[_(w)]:{overwrite:R??!1}})}})})})}),x.jsx(me.Cell,{collapsing:!0,children:x.jsx(Le,{size:"mini",content:"Preview",color:"purple",onClick:()=>C(w)})})]},A))})]})]})]})}),x.jsxs(Yn,{open:o.open,onClose:()=>{a({open:!1,pathAndCodes:null})},children:[x.jsx(Yn.Header,{children:"Preview"}),x.jsx(Yn.Content,{children:x.jsx(Yn.Description,{children:o.pathAndCodes?.map(w=>x.jsxs("div",{className:"preview-item",children:[x.jsx("h4",{children:w.path}),x.jsx("code",{children:w.code})]},w.path))})})]})]})}const MQe=new Ehe({defaultOptions:{queries:{refetchOnWindowFocus:!0,retry:3,retryDelay:3e3}}});Roe.createRoot(document.getElementById("root")).render(x.jsx(_he,{client:MQe,children:x.jsx(lae,{basename:"/sonamu-ui/",children:x.jsxs(aae,{children:[x.jsxs(ss,{path:"/",element:x.jsx(mge,{}),children:[x.jsx(ss,{path:"/entities",element:x.jsx(Lze,{}),children:x.jsx(ss,{path:":entityId",element:x.jsx(qze,{})})}),x.jsx(ss,{path:"/migrations",element:x.jsx(DQe,{})}),x.jsx(ss,{path:"/scaffolding",element:x.jsx($Qe,{})}),x.jsx(ss,{path:"/fixture",element:x.jsx(RQe,{})})]}),x.jsx(ss,{path:"*",element:x.jsx("div",{children:"Page Not Found"})})]})})}))});export default LQe();
@@ -4,7 +4,7 @@
4
4
  <meta charset="UTF-8" />
5
5
  <meta name="viewport" content="width=device-width, initial-scale=1.0" />
6
6
  <title>{{projectName}}: Sonamu UI</title>
7
- <script type="module" crossorigin src="/sonamu-ui/assets/index-DFqVuxOB.js"></script>
7
+ <script type="module" crossorigin src="/sonamu-ui/assets/index-B87IyofX.js"></script>
8
8
  <link rel="stylesheet" crossorigin href="/sonamu-ui/assets/index-CpaB9P6g.css">
9
9
  </head>
10
10
  <body>
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "sonamu",
3
- "version": "0.7.18",
3
+ "version": "0.7.20",
4
4
  "description": "Sonamu — TypeScript Fullstack API Framework",
5
5
  "keywords": [
6
6
  "typescript",
@@ -21,6 +21,10 @@
21
21
  "import": "./dist/vector/index.js",
22
22
  "types": "./dist/vector/index.d.ts"
23
23
  },
24
+ "./storage": {
25
+ "import": "./dist/storage/index.js",
26
+ "types": "./dist/storage/index.d.ts"
27
+ },
24
28
  "./ai/providers/rtzr": {
25
29
  "import": "./dist/ai/providers/rtzr/index.js",
26
30
  "types": "./dist/ai/providers/rtzr/index.d.ts"
@@ -68,6 +72,7 @@
68
72
  "fastify": "^4.23.2",
69
73
  "fastify-qs": "^4.0.0",
70
74
  "fastify-sse-v2": "^4.2.1",
75
+ "flydrive": "^1.3.0",
71
76
  "inflection": "^1.13.2",
72
77
  "knex": "^3.1.0",
73
78
  "mime-types": "^3.0.1",
package/src/api/config.ts CHANGED
@@ -8,7 +8,7 @@ import type { FastifyInstance, FastifyReply, FastifyRequest, FastifyServerOption
8
8
  import type { QsPluginOptions } from "fastify-qs";
9
9
  import type { SsePluginOptions } from "fastify-sse-v2/lib/types";
10
10
  import type { Knex } from "knex";
11
- import type { Driver } from "../file-storage/driver";
11
+ import type { StorageConfig } from "../storage/types";
12
12
  import type { WorkflowOptions } from "../tasks/workflow-manager";
13
13
  import type { Executable, SonamuFastifyConfig } from "../types/types";
14
14
  import type { AuthContext, Context } from "./context";
@@ -44,7 +44,8 @@ export type SonamuConfig = {
44
44
  development_slave?: DatabaseConfig;
45
45
  production?: DatabaseConfig;
46
46
  production_slave?: DatabaseConfig;
47
- remote_fixture?: DatabaseConfig;
47
+ fixture?: DatabaseConfig;
48
+ test?: DatabaseConfig;
48
49
  };
49
50
  };
50
51
 
@@ -84,7 +85,24 @@ export type SonamuServerOptions = {
84
85
 
85
86
  apiConfig: SonamuFastifyConfig;
86
87
 
87
- storage?: Driver;
88
+ /**
89
+ * Storage 드라이버 설정.
90
+ * DRIVE_DISK 환경변수로 사용할 드라이버를 선택합니다. (기본값: default 키)
91
+ *
92
+ * @example
93
+ * ```typescript
94
+ * import { drivers } from "sonamu/storage";
95
+ *
96
+ * storage: {
97
+ * default: process.env.DRIVE_DISK ?? "fs",
98
+ * drivers: {
99
+ * fs: drivers.fs({ location: "./uploads", urlBuilder: { ... } }),
100
+ * s3: drivers.s3({ bucket: "my-bucket", region: "ap-northeast-2", ... }),
101
+ * }
102
+ * }
103
+ * ```
104
+ */
105
+ storage?: StorageConfig;
88
106
 
89
107
  lifecycle?: {
90
108
  onStart?: (server: FastifyInstance) => Promise<void> | void;
@@ -2,8 +2,8 @@ import type { FastifyReply, FastifyRequest, PassportUser } from "fastify";
2
2
  import type { RouteGenericInterface } from "fastify/types/route";
3
3
  import type { IncomingHttpHeaders, IncomingMessage, Server, ServerResponse } from "http";
4
4
  import type { ZodObject } from "zod";
5
- import type { FileStorage } from "../file-storage/file-storage";
6
5
  import type { NaiteStore } from "../naite/naite";
6
+ import type { UploadedFile } from "../storage/uploaded-file";
7
7
  import type { createSSEFactory } from "../stream/sse";
8
8
 
9
9
  // biome-ignore lint/suspicious/noEmptyInterface: Context 확장 타입
@@ -26,6 +26,6 @@ export type AuthContext = {
26
26
  };
27
27
 
28
28
  export type UploadContext = {
29
- file?: FileStorage;
30
- files: FileStorage[];
29
+ file?: UploadedFile;
30
+ files: UploadedFile[];
31
31
  };