sonamu 0.7.39 → 0.7.41
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.
- package/dist/tasks/workflow-manager.d.ts.map +1 -1
- package/dist/tasks/workflow-manager.js +9 -1
- package/dist/template/implementations/generated_http.template.d.ts.map +1 -1
- package/dist/template/implementations/generated_http.template.js +3 -2
- package/dist/template/implementations/queries.template.d.ts.map +1 -1
- package/dist/template/implementations/queries.template.js +10 -2
- package/dist/template/implementations/sd.template.d.ts.map +1 -1
- package/dist/template/implementations/sd.template.js +34 -21
- package/dist/template/implementations/view_form.template.d.ts.map +1 -1
- package/dist/template/implementations/view_form.template.js +4 -2
- package/dist/template/implementations/view_id_async_select.template.js +2 -2
- package/dist/template/zod-converter.js +4 -2
- package/dist/testing/fixture-manager.d.ts +18 -6
- package/dist/testing/fixture-manager.d.ts.map +1 -1
- package/dist/testing/fixture-manager.js +124 -46
- package/dist/ui-web/assets/{index-Bfv7V57v.css → index-CWExqVO5.css} +1 -1
- package/dist/ui-web/assets/{index-DCf469Xl.js → index-DoZuAOiq.js} +69 -67
- package/dist/ui-web/index.html +2 -2
- package/package.json +3 -3
- package/src/tasks/workflow-manager.ts +10 -0
- package/src/template/implementations/generated_http.template.ts +2 -1
- package/src/template/implementations/queries.template.ts +23 -12
- package/src/template/implementations/sd.template.ts +33 -20
- package/src/template/implementations/view_form.template.ts +3 -1
- package/src/template/implementations/view_id_async_select.template.ts +1 -1
- package/src/template/zod-converter.ts +3 -1
- package/src/testing/fixture-manager.ts +167 -59
|
@@ -38,7 +38,6 @@ export class FixtureManagerClass {
|
|
|
38
38
|
// UpsertBuilder 기반 import를 위한 상태
|
|
39
39
|
builder = new UpsertBuilder();
|
|
40
40
|
fixtureRefMap = new Map();
|
|
41
|
-
uuidToFixtureId = new Map();
|
|
42
41
|
skippedFixtures = new Map();
|
|
43
42
|
init() {
|
|
44
43
|
if (this._tdb !== null) {
|
|
@@ -308,14 +307,18 @@ export class FixtureManagerClass {
|
|
|
308
307
|
}
|
|
309
308
|
/**
|
|
310
309
|
* 1. RelationGraph로 fixture 단위 삽입 순서 계산 (self-reference 포함)
|
|
311
|
-
* 2.
|
|
312
|
-
* 3.
|
|
310
|
+
* 2. 테이블별 레벨별로 UpsertBuilder에 등록 및 upsert 실행
|
|
311
|
+
* 3. 순서 기반 uuid→id 매핑 (UpsertBuilder가 uuid를 DB에 저장하지 않으므로)
|
|
312
|
+
*
|
|
313
|
+
* UpsertBuilder는 self-reference가 있으면 buildInsertLevels()로 재정렬하여
|
|
314
|
+
* 등록 순서와 반환 순서가 달라질 수 있습니다. 이를 방지하기 위해
|
|
315
|
+
* FixtureManager가 레벨별로 나눠서 처리하여 각 upsert 호출에서는
|
|
316
|
+
* self-reference가 없도록 합니다.
|
|
313
317
|
*/ async insertFixtures(dbName, _fixtures) {
|
|
314
318
|
const fixtures = unique(_fixtures, (f)=>f.fixtureId);
|
|
315
319
|
// 초기화
|
|
316
320
|
this.builder = new UpsertBuilder();
|
|
317
321
|
this.fixtureRefMap = new Map();
|
|
318
|
-
this.uuidToFixtureId = new Map();
|
|
319
322
|
this.skippedFixtures = new Map();
|
|
320
323
|
const db = createKnexInstance(Sonamu.dbConfig[dbName]);
|
|
321
324
|
const results = [];
|
|
@@ -323,7 +326,7 @@ export class FixtureManagerClass {
|
|
|
323
326
|
// 1. RelationGraph로 fixture 단위 삽입 순서 계산
|
|
324
327
|
this.relationGraph.buildGraph(fixtures);
|
|
325
328
|
const insertionOrder = this.relationGraph.getInsertionOrder();
|
|
326
|
-
// 2.
|
|
329
|
+
// 2. 스킵할 fixture 먼저 처리 (override 체크)
|
|
327
330
|
for (const fixtureId of insertionOrder){
|
|
328
331
|
const fixture = fixtures.find((f)=>f.fixtureId === fixtureId);
|
|
329
332
|
if (!fixture) continue;
|
|
@@ -340,35 +343,57 @@ export class FixtureManagerClass {
|
|
|
340
343
|
existingId
|
|
341
344
|
});
|
|
342
345
|
console.log(chalk.yellow(`Skipped ${fixture.entityId}#${fixture.id} (existing: #${existingId}, override: false)`));
|
|
343
|
-
continue;
|
|
344
346
|
}
|
|
345
|
-
this.registerFixture(fixture);
|
|
346
|
-
console.log(chalk.blue(`Registered ${fixture.entityId}#${fixture.id}${fixture.override ? ` (override existing: #${fixture.target?.id})` : ""}`));
|
|
347
347
|
}
|
|
348
|
-
// 3. 테이블별
|
|
349
|
-
const
|
|
348
|
+
// 3. 테이블별 fixture 그룹화 (insertionOrder 순서 기반)
|
|
349
|
+
const fixturesByTable = new Map();
|
|
350
|
+
const tableOrder = [];
|
|
351
|
+
for (const fixtureId of insertionOrder){
|
|
352
|
+
// 스킵된 fixture 제외
|
|
353
|
+
if (this.skippedFixtures.has(fixtureId)) continue;
|
|
354
|
+
const fixture = fixtures.find((f)=>f.fixtureId === fixtureId);
|
|
355
|
+
if (!fixture) continue;
|
|
356
|
+
const entity = EntityManager.get(fixture.entityId);
|
|
357
|
+
const tableName = entity.table;
|
|
358
|
+
if (!fixturesByTable.has(tableName)) {
|
|
359
|
+
fixturesByTable.set(tableName, []);
|
|
360
|
+
tableOrder.push(tableName);
|
|
361
|
+
}
|
|
362
|
+
fixturesByTable.get(tableName)?.push(fixture);
|
|
363
|
+
}
|
|
350
364
|
await db.transaction(async (trx)=>{
|
|
351
365
|
const insertedIdsByTable = new Map();
|
|
366
|
+
// 4. 테이블별 레벨별 처리
|
|
352
367
|
for (const tableName of tableOrder){
|
|
353
|
-
|
|
354
|
-
|
|
355
|
-
const
|
|
356
|
-
|
|
357
|
-
|
|
358
|
-
|
|
359
|
-
|
|
360
|
-
|
|
361
|
-
|
|
362
|
-
const
|
|
363
|
-
|
|
364
|
-
|
|
368
|
+
const tableFixtures = fixturesByTable.get(tableName) ?? [];
|
|
369
|
+
const levels = this.groupFixturesByLevel(tableFixtures);
|
|
370
|
+
for (const levelFixtures of levels){
|
|
371
|
+
// 해당 레벨의 fixture들 register
|
|
372
|
+
for (const fixture of levelFixtures){
|
|
373
|
+
this.registerFixture(fixture, insertedIdsByTable);
|
|
374
|
+
console.log(chalk.blue(`Registered ${fixture.entityId}#${fixture.id}${fixture.override ? ` (override)` : ""}`));
|
|
375
|
+
}
|
|
376
|
+
// upsert 실행 전 uuid 목록 저장
|
|
377
|
+
const table = this.builder.getTable(tableName);
|
|
378
|
+
const uuids = table.rows.map((row)=>row.uuid);
|
|
379
|
+
console.log(chalk.blue(`Upserting ${tableName} with ${uuids.length} rows (level ${levels.indexOf(levelFixtures) + 1}/${levels.length})`));
|
|
380
|
+
const ids = await this.builder.upsert(trx, tableName);
|
|
381
|
+
// 순서 기반 uuid -> id 매핑
|
|
382
|
+
// self-reference가 없으므로 등록 순서 = 반환 순서 보장
|
|
383
|
+
if (uuids.length > 0 && uuids.length === ids.length) {
|
|
384
|
+
const existingMap = insertedIdsByTable.get(tableName) ?? new Map();
|
|
385
|
+
for(let i = 0; i < uuids.length; i++){
|
|
386
|
+
existingMap.set(uuids[i], ids[i]);
|
|
387
|
+
}
|
|
388
|
+
insertedIdsByTable.set(tableName, existingMap);
|
|
389
|
+
} else if (uuids.length !== ids.length) {
|
|
390
|
+
console.warn(chalk.yellow(`Warning: uuid count (${uuids.length}) != id count (${ids.length}) for ${tableName}`));
|
|
365
391
|
}
|
|
366
|
-
insertedIdsByTable.set(tableName, uuidToId);
|
|
367
392
|
}
|
|
368
393
|
}
|
|
369
|
-
//
|
|
394
|
+
// 5. ManyToMany 관계 처리
|
|
370
395
|
await this.processManyToManyRelations(trx, fixtures, insertedIdsByTable);
|
|
371
|
-
//
|
|
396
|
+
// 6. 결과 수집
|
|
372
397
|
for (const fixture of fixtures){
|
|
373
398
|
const entity = EntityManager.get(fixture.entityId);
|
|
374
399
|
// 스킵된 fixture는 기존 레코드 정보로 결과 추가
|
|
@@ -401,7 +426,8 @@ export class FixtureManagerClass {
|
|
|
401
426
|
}
|
|
402
427
|
/**
|
|
403
428
|
* FixtureRecord를 UpsertBuilder에 등록
|
|
404
|
-
|
|
429
|
+
* @param insertedIdsByTable 이미 upsert된 테이블의 uuid→id 매핑 (레벨별 처리 시 사용)
|
|
430
|
+
*/ registerFixture(fixture, insertedIdsByTable) {
|
|
405
431
|
const entity = EntityManager.get(fixture.entityId);
|
|
406
432
|
const row = {};
|
|
407
433
|
// Override 모드 판단: target 또는 unique가 있고 override=true인 경우
|
|
@@ -412,8 +438,12 @@ export class FixtureManagerClass {
|
|
|
412
438
|
if (isVirtualProp(prop)) {
|
|
413
439
|
continue;
|
|
414
440
|
}
|
|
415
|
-
//
|
|
416
|
-
if (
|
|
441
|
+
// Generated column은 INSERT에서 제외 (DB가 자동 생성)
|
|
442
|
+
if ("generated" in prop && prop.generated) {
|
|
443
|
+
continue;
|
|
444
|
+
}
|
|
445
|
+
// id 처리: Override 모드일 때만 기존 값 사용
|
|
446
|
+
if (propName === "id") {
|
|
417
447
|
if (isOverrideMode && existingRecord) {
|
|
418
448
|
// Override: 기존 레코드의 값 사용 → UPDATE
|
|
419
449
|
row[propName] = existingRecord.columns[propName]?.value;
|
|
@@ -433,8 +463,17 @@ export class FixtureManagerClass {
|
|
|
433
463
|
} else {
|
|
434
464
|
const relatedRef = this.fixtureRefMap.get(relatedFixtureId);
|
|
435
465
|
if (relatedRef) {
|
|
436
|
-
// 이미
|
|
437
|
-
|
|
466
|
+
// 이미 upsert된 같은 테이블 fixture 확인
|
|
467
|
+
const relatedEntity = EntityManager.get(prop.with);
|
|
468
|
+
const relatedInsertedIds = insertedIdsByTable?.get(relatedEntity.table);
|
|
469
|
+
const actualId = relatedInsertedIds?.get(relatedRef.uuid);
|
|
470
|
+
if (actualId !== undefined) {
|
|
471
|
+
// 이미 upsert됨 → 실제 ID 사용
|
|
472
|
+
row[`${propName}_id`] = actualId;
|
|
473
|
+
} else {
|
|
474
|
+
// 아직 upsert 안됨 → UBRef 사용
|
|
475
|
+
row[`${propName}_id`] = relatedRef;
|
|
476
|
+
}
|
|
438
477
|
} else {
|
|
439
478
|
// fixtures에 포함되지 않은 레코드 → ID 그대로 사용
|
|
440
479
|
row[`${propName}_id`] = relatedId;
|
|
@@ -453,7 +492,6 @@ export class FixtureManagerClass {
|
|
|
453
492
|
console.log(chalk.blue(`Registering ${entity.table} - ${inspect(row, false, null, true)}`));
|
|
454
493
|
const ref = this.builder.register(entity.table, row);
|
|
455
494
|
this.fixtureRefMap.set(fixture.fixtureId, ref);
|
|
456
|
-
this.uuidToFixtureId.set(ref.uuid, fixture.fixtureId);
|
|
457
495
|
return ref;
|
|
458
496
|
}
|
|
459
497
|
/**
|
|
@@ -475,20 +513,6 @@ export class FixtureManagerClass {
|
|
|
475
513
|
return value;
|
|
476
514
|
}
|
|
477
515
|
}
|
|
478
|
-
/**
|
|
479
|
-
* 테이블 순서 추출 (fixtures에 포함된 테이블만)
|
|
480
|
-
*/ getTableOrder(fixtures) {
|
|
481
|
-
const tables = [];
|
|
482
|
-
const seen = new Set();
|
|
483
|
-
for (const fixture of fixtures){
|
|
484
|
-
const entity = EntityManager.get(fixture.entityId);
|
|
485
|
-
if (!seen.has(entity.table)) {
|
|
486
|
-
seen.add(entity.table);
|
|
487
|
-
tables.push(entity.table);
|
|
488
|
-
}
|
|
489
|
-
}
|
|
490
|
-
return tables;
|
|
491
|
-
}
|
|
492
516
|
async processManyToManyRelations(trx, fixtures, insertedIdsByTable) {
|
|
493
517
|
for (const fixture of fixtures){
|
|
494
518
|
const entity = EntityManager.get(fixture.entityId);
|
|
@@ -541,6 +565,60 @@ export class FixtureManagerClass {
|
|
|
541
565
|
}
|
|
542
566
|
}
|
|
543
567
|
}
|
|
568
|
+
/**
|
|
569
|
+
* 같은 테이블 내 fixture들을 self-reference 레벨별로 분할
|
|
570
|
+
* - self-reference가 없는 fixture들: Level 0
|
|
571
|
+
* - Level 0을 참조하는 fixture들: Level 1
|
|
572
|
+
* - 반복
|
|
573
|
+
*
|
|
574
|
+
* UpsertBuilder가 self-reference가 있으면 buildInsertLevels()로 재정렬하여
|
|
575
|
+
* 등록 순서와 반환 순서가 달라질 수 있습니다.
|
|
576
|
+
* 이를 방지하기 위해 FixtureManager가 레벨별로 나눠서 처리합니다.
|
|
577
|
+
*/ groupFixturesByLevel(fixtures) {
|
|
578
|
+
if (fixtures.length === 0) {
|
|
579
|
+
return [];
|
|
580
|
+
}
|
|
581
|
+
const entity = EntityManager.get(fixtures[0].entityId);
|
|
582
|
+
// self-reference relation prop 찾기
|
|
583
|
+
const selfRefProps = entity.props.filter((p)=>isRelationProp(p) && (isBelongsToOneRelationProp(p) || isOneToOneRelationProp(p) && p.hasJoinColumn) && p.with === entity.id);
|
|
584
|
+
if (selfRefProps.length === 0) {
|
|
585
|
+
// self-reference 없음 → 단일 레벨
|
|
586
|
+
return [
|
|
587
|
+
fixtures
|
|
588
|
+
];
|
|
589
|
+
}
|
|
590
|
+
// 레벨별 분할 (topological sort)
|
|
591
|
+
const levels = [];
|
|
592
|
+
const remaining = new Set(fixtures.map((f)=>f.fixtureId));
|
|
593
|
+
const processed = new Set();
|
|
594
|
+
while(remaining.size > 0){
|
|
595
|
+
const currentLevel = [];
|
|
596
|
+
for (const fixture of fixtures){
|
|
597
|
+
if (!remaining.has(fixture.fixtureId)) continue;
|
|
598
|
+
// self-reference가 모두 이미 처리됐거나 null인 경우
|
|
599
|
+
const canProcess = selfRefProps.every((prop)=>{
|
|
600
|
+
const refId = fixture.columns[prop.name]?.value;
|
|
601
|
+
if (refId === null || refId === undefined) return true;
|
|
602
|
+
const refFixtureId = `${prop.with}#${refId}`;
|
|
603
|
+
// 이미 처리됐거나, 현재 fixtures에 포함되지 않은 경우 (외부 참조)
|
|
604
|
+
return processed.has(refFixtureId) || !remaining.has(refFixtureId);
|
|
605
|
+
});
|
|
606
|
+
if (canProcess) {
|
|
607
|
+
currentLevel.push(fixture);
|
|
608
|
+
}
|
|
609
|
+
}
|
|
610
|
+
if (currentLevel.length === 0) {
|
|
611
|
+
const remainingIds = Array.from(remaining).join(", ");
|
|
612
|
+
throw new Error(`Circular self-reference detected in ${entity.table}. Remaining fixtures: ${remainingIds}`);
|
|
613
|
+
}
|
|
614
|
+
for (const fixture of currentLevel){
|
|
615
|
+
remaining.delete(fixture.fixtureId);
|
|
616
|
+
processed.add(fixture.fixtureId);
|
|
617
|
+
}
|
|
618
|
+
levels.push(currentLevel);
|
|
619
|
+
}
|
|
620
|
+
return levels;
|
|
621
|
+
}
|
|
544
622
|
async checkUniqueViolation(db, entity, fixture) {
|
|
545
623
|
const _uniqueIndexes = entity.indexes?.filter((i)=>i.type === "unique") ?? [];
|
|
546
624
|
const uniqueIndexes = _uniqueIndexes.filter((index)=>index.columns.every((column)=>!column.name.startsWith(`${entity.table}__`)));
|
|
@@ -610,4 +688,4 @@ export class FixtureManagerClass {
|
|
|
610
688
|
}
|
|
611
689
|
export const FixtureManager = new FixtureManagerClass();
|
|
612
690
|
|
|
613
|
-
//# 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 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 { createKnexInstance } from \"../database/knex\";\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 = createKnexInstance(Sonamu.dbConfig.test);\n    this.fdb = createKnexInstance(Sonamu.dbConfig.fixture);\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      await wdb.raw(query);\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    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 = createKnexInstance(Sonamu.dbConfig[sourceDBName]);\n    const targetDB = createKnexInstance(Sonamu.dbConfig[targetDBName]);\n\n    try {\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\"\n          ? `${field}_id`\n          : 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      return unique(fixtures, (f) => f.fixtureId);\n    } finally {\n      await targetDB.destroy();\n      await sourceDB.destroy();\n    }\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 = createKnexInstance(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","unique","inspect","Sonamu","BaseModel","createKnexInstance","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","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","raw","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","db","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","console","log","yellow","registerFixture","blue","tableOrder","getTableOrder","transaction","trx","insertedIdsByTable","tableName","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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"}
|
|
691
|
+
//# 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 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 { createKnexInstance } from \"../database/knex\";\nimport { type UBRef, UpsertBuilder } from \"../database/upsert-builder\";\nimport type { Entity } from \"../entity/entity\";\nimport { EntityManager } from \"../entity/entity-manager\";\nimport {\n  type BelongsToOneRelationProp,\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  type OneToOneRelationProp,\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 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 = createKnexInstance(Sonamu.dbConfig.test);\n    this.fdb = createKnexInstance(Sonamu.dbConfig.fixture);\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      await wdb.raw(query);\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    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 = createKnexInstance(Sonamu.dbConfig[sourceDBName]);\n    const targetDB = createKnexInstance(Sonamu.dbConfig[targetDBName]);\n\n    try {\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\"\n          ? `${field}_id`\n          : 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      return unique(fixtures, (f) => f.fixtureId);\n    } finally {\n      await targetDB.destroy();\n      await sourceDB.destroy();\n    }\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에 등록 및 upsert 실행\n   * 3. 순서 기반 uuid→id 매핑 (UpsertBuilder가 uuid를 DB에 저장하지 않으므로)\n   *\n   * UpsertBuilder는 self-reference가 있으면 buildInsertLevels()로 재정렬하여\n   * 등록 순서와 반환 순서가 달라질 수 있습니다. 이를 방지하기 위해\n   * FixtureManager가 레벨별로 나눠서 처리하여 각 upsert 호출에서는\n   * self-reference가 없도록 합니다.\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.skippedFixtures = new Map();\n\n    const db = createKnexInstance(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. 스킵할 fixture 먼저 처리 (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        }\n      }\n\n      // 3. 테이블별 fixture 그룹화 (insertionOrder 순서 기반)\n      const fixturesByTable = new Map<string, FixtureRecord[]>();\n      const tableOrder: string[] = [];\n\n      for (const fixtureId of insertionOrder) {\n        // 스킵된 fixture 제외\n        if (this.skippedFixtures.has(fixtureId)) continue;\n\n        const fixture = fixtures.find((f) => f.fixtureId === fixtureId);\n        if (!fixture) continue;\n\n        const entity = EntityManager.get(fixture.entityId);\n        const tableName = entity.table;\n\n        if (!fixturesByTable.has(tableName)) {\n          fixturesByTable.set(tableName, []);\n          tableOrder.push(tableName);\n        }\n        fixturesByTable.get(tableName)?.push(fixture);\n      }\n\n      await db.transaction(async (trx) => {\n        const insertedIdsByTable = new Map<string, Map<string, number>>();\n\n        // 4. 테이블별 레벨별 처리\n        for (const tableName of tableOrder) {\n          const tableFixtures = fixturesByTable.get(tableName) ?? [];\n          const levels = this.groupFixturesByLevel(tableFixtures);\n\n          for (const levelFixtures of levels) {\n            // 해당 레벨의 fixture들 register\n            for (const fixture of levelFixtures) {\n              this.registerFixture(fixture, insertedIdsByTable);\n              console.log(\n                chalk.blue(\n                  `Registered ${fixture.entityId}#${fixture.id}${fixture.override ? ` (override)` : \"\"}`,\n                ),\n              );\n            }\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(\n              chalk.blue(\n                `Upserting ${tableName} with ${uuids.length} rows (level ${levels.indexOf(levelFixtures) + 1}/${levels.length})`,\n              ),\n            );\n            const ids = await this.builder.upsert(trx, tableName as keyof DatabaseSchemaExtend);\n\n            // 순서 기반 uuid -> id 매핑\n            // self-reference가 없으므로 등록 순서 = 반환 순서 보장\n            if (uuids.length > 0 && uuids.length === ids.length) {\n              const existingMap = insertedIdsByTable.get(tableName) ?? new Map<string, number>();\n              for (let i = 0; i < uuids.length; i++) {\n                existingMap.set(uuids[i], ids[i]);\n              }\n              insertedIdsByTable.set(tableName, existingMap);\n            } else if (uuids.length !== ids.length) {\n              console.warn(\n                chalk.yellow(\n                  `Warning: uuid count (${uuids.length}) != id count (${ids.length}) for ${tableName}`,\n                ),\n              );\n            }\n          }\n        }\n\n        // 5. ManyToMany 관계 처리\n        await this.processManyToManyRelations(trx, fixtures, insertedIdsByTable);\n\n        // 6. 결과 수집\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   * @param insertedIdsByTable 이미 upsert된 테이블의 uuid→id 매핑 (레벨별 처리 시 사용)\n   */\n  private registerFixture(\n    fixture: FixtureRecord,\n    insertedIdsByTable?: Map<string, Map<string, number>>,\n  ): 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      // Generated column은 INSERT에서 제외 (DB가 자동 생성)\n      if (\"generated\" in prop && prop.generated) {\n        continue;\n      }\n\n      // id 처리: Override 모드일 때만 기존 값 사용\n      if (propName === \"id\") {\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                // 이미 upsert된 같은 테이블 fixture 확인\n                const relatedEntity = EntityManager.get(prop.with);\n                const relatedInsertedIds = insertedIdsByTable?.get(relatedEntity.table);\n                const actualId = relatedInsertedIds?.get(relatedRef.uuid);\n\n                if (actualId !== undefined) {\n                  // 이미 upsert됨 → 실제 ID 사용\n                  row[`${propName}_id`] = actualId;\n                } else {\n                  // 아직 upsert 안됨 → UBRef 사용\n                  row[`${propName}_id`] = relatedRef;\n                }\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\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  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  /**\n   * 같은 테이블 내 fixture들을 self-reference 레벨별로 분할\n   * - self-reference가 없는 fixture들: Level 0\n   * - Level 0을 참조하는 fixture들: Level 1\n   * - 반복\n   *\n   * UpsertBuilder가 self-reference가 있으면 buildInsertLevels()로 재정렬하여\n   * 등록 순서와 반환 순서가 달라질 수 있습니다.\n   * 이를 방지하기 위해 FixtureManager가 레벨별로 나눠서 처리합니다.\n   */\n  private groupFixturesByLevel(fixtures: FixtureRecord[]): FixtureRecord[][] {\n    if (fixtures.length === 0) {\n      return [];\n    }\n\n    const entity = EntityManager.get(fixtures[0].entityId);\n\n    // self-reference relation prop 찾기\n    const selfRefProps = entity.props.filter(\n      (p): p is BelongsToOneRelationProp | OneToOneRelationProp =>\n        isRelationProp(p) &&\n        (isBelongsToOneRelationProp(p) || (isOneToOneRelationProp(p) && p.hasJoinColumn)) &&\n        p.with === entity.id,\n    );\n\n    if (selfRefProps.length === 0) {\n      // self-reference 없음 → 단일 레벨\n      return [fixtures];\n    }\n\n    // 레벨별 분할 (topological sort)\n    const levels: FixtureRecord[][] = [];\n    const remaining = new Set(fixtures.map((f) => f.fixtureId));\n    const processed = new Set<string>();\n\n    while (remaining.size > 0) {\n      const currentLevel: FixtureRecord[] = [];\n\n      for (const fixture of fixtures) {\n        if (!remaining.has(fixture.fixtureId)) continue;\n\n        // self-reference가 모두 이미 처리됐거나 null인 경우\n        const canProcess = selfRefProps.every((prop) => {\n          const refId = fixture.columns[prop.name]?.value as number | null;\n          if (refId === null || refId === undefined) return true;\n          const refFixtureId = `${prop.with}#${refId}`;\n          // 이미 처리됐거나, 현재 fixtures에 포함되지 않은 경우 (외부 참조)\n          return processed.has(refFixtureId) || !remaining.has(refFixtureId);\n        });\n\n        if (canProcess) {\n          currentLevel.push(fixture);\n        }\n      }\n\n      if (currentLevel.length === 0) {\n        const remainingIds = Array.from(remaining).join(\", \");\n        throw new Error(\n          `Circular self-reference detected in ${entity.table}. Remaining fixtures: ${remainingIds}`,\n        );\n      }\n\n      for (const fixture of currentLevel) {\n        remaining.delete(fixture.fixtureId);\n        processed.add(fixture.fixtureId);\n      }\n\n      levels.push(currentLevel);\n    }\n\n    return levels;\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","unique","inspect","Sonamu","BaseModel","createKnexInstance","UpsertBuilder","EntityManager","isBelongsToOneRelationProp","isHasManyRelationProp","isManyToManyRelationProp","isOneToOneRelationProp","isRelationProp","isVirtualProp","RelationGraph","FixtureManagerClass","_tdb","tdb","Error","_fdb","fdb","cachedTableNames","relationGraph","builder","fixtureRefMap","Map","skippedFixtures","init","dbConfig","test","production_master","tConn","connection","pConn","host","port","database","fixture","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","raw","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","db","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","console","log","yellow","fixturesByTable","tableOrder","tableName","transaction","trx","insertedIdsByTable","tableFixtures","levels","groupFixturesByLevel","levelFixtures","registerFixture","blue","getTable","uuids","uuid","indexOf","upsert","existingMap","i","warn","processManyToManyRelations","skipped","data","ref","uuidToId","insertedId","green","existingRecord","isOverrideMode","propName","generated","relatedFixtureId","skippedExistingId","relatedRef","relatedInsertedIds","actualId","convertColumnValue","register","Date","sourceRef","sourceUuidToId","sourceId","Array","isArray","targetTable","hasTable","sourceColumn","targetColumn","targetId","relatedUuidToId","resolvedId","found","insert","selfRefProps","remaining","processed","size","currentLevel","canProcess","every","refId","refFixtureId","remainingIds","from","join","delete","_uniqueIndexes","indexes","uniqueIndexes","index","startsWith","uniqueQuery","hasCondition","containsNull","some","replace","orWhere","qb","whereIn","andWhere","uniqueFound","whereClause","dbColumn","addFixtureLoader","code","path","apiRootPath","content","toString","fixtureLoaderStart","fixtureLoaderEnd","newContent","FixtureManager"],"mappings":"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"}
|