sonamu 0.7.3 → 0.7.5
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/api/config.d.ts +1 -4
- package/dist/api/config.d.ts.map +1 -1
- package/dist/api/config.js +1 -1
- package/dist/api/sonamu.d.ts +2 -0
- package/dist/api/sonamu.d.ts.map +1 -1
- package/dist/api/sonamu.js +19 -47
- package/dist/bin/cli.js +6 -6
- package/dist/database/base-model.d.ts +1 -1
- package/dist/database/base-model.d.ts.map +1 -1
- package/dist/database/base-model.js +15 -4
- package/dist/database/code-generator.d.ts.map +1 -1
- package/dist/database/code-generator.js +3 -3
- package/dist/database/db.d.ts.map +1 -1
- package/dist/database/db.js +1 -1
- package/dist/database/puri-wrapper.d.ts +11 -11
- package/dist/database/puri-wrapper.d.ts.map +1 -1
- package/dist/database/puri-wrapper.js +7 -11
- package/dist/database/puri.d.ts +36 -17
- package/dist/database/puri.d.ts.map +1 -1
- package/dist/database/puri.js +54 -7
- package/dist/database/puri.types.d.ts +54 -17
- package/dist/database/puri.types.d.ts.map +1 -1
- package/dist/database/puri.types.js +2 -4
- package/dist/database/puri.types.test-d.js +129 -0
- package/dist/database/upsert-builder.d.ts +16 -10
- package/dist/database/upsert-builder.d.ts.map +1 -1
- package/dist/database/upsert-builder.js +10 -19
- package/dist/entity/entity-manager.d.ts +113 -22
- package/dist/entity/entity-manager.d.ts.map +1 -1
- package/dist/entity/entity-manager.js +1 -1
- package/dist/entity/entity.d.ts +34 -0
- package/dist/entity/entity.d.ts.map +1 -1
- package/dist/entity/entity.js +110 -37
- package/dist/index.d.ts +5 -0
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +8 -2
- package/dist/migration/code-generation.d.ts.map +1 -1
- package/dist/migration/code-generation.js +341 -149
- package/dist/migration/migration-set.d.ts.map +1 -1
- package/dist/migration/migration-set.js +21 -5
- package/dist/migration/migrator.d.ts.map +1 -1
- package/dist/migration/migrator.js +7 -1
- package/dist/migration/postgresql-schema-reader.d.ts +11 -1
- package/dist/migration/postgresql-schema-reader.d.ts.map +1 -1
- package/dist/migration/postgresql-schema-reader.js +111 -10
- package/dist/syncer/syncer.d.ts.map +1 -1
- package/dist/syncer/syncer.js +5 -4
- package/dist/template/implementations/generated.template.d.ts.map +1 -1
- package/dist/template/implementations/generated.template.js +12 -2
- package/dist/template/implementations/generated_sso.template.d.ts +3 -3
- package/dist/template/implementations/generated_sso.template.d.ts.map +1 -1
- package/dist/template/implementations/generated_sso.template.js +50 -2
- package/dist/template/implementations/model.template.js +6 -6
- package/dist/template/implementations/model_test.template.js +4 -4
- package/dist/template/implementations/view_enums_dropdown.template.js +2 -2
- package/dist/template/implementations/view_enums_select.template.js +2 -2
- package/dist/template/implementations/view_form.template.d.ts.map +1 -1
- package/dist/template/implementations/view_form.template.js +12 -9
- package/dist/template/implementations/view_id_async_select.template.js +4 -4
- package/dist/template/implementations/view_list.template.d.ts.map +1 -1
- package/dist/template/implementations/view_list.template.js +12 -9
- package/dist/template/implementations/view_search_input.template.js +2 -2
- package/dist/template/template.js +2 -2
- package/dist/template/zod-converter.d.ts.map +1 -1
- package/dist/template/zod-converter.js +17 -2
- package/dist/testing/fixture-manager.d.ts +2 -1
- package/dist/testing/fixture-manager.d.ts.map +1 -1
- package/dist/testing/fixture-manager.js +29 -29
- package/dist/types/types.d.ts +593 -68
- package/dist/types/types.d.ts.map +1 -1
- package/dist/types/types.js +113 -9
- package/dist/vector/chunking.d.ts +25 -0
- package/dist/vector/chunking.d.ts.map +1 -0
- package/dist/vector/chunking.js +97 -0
- package/dist/vector/config.d.ts +12 -0
- package/dist/vector/config.d.ts.map +1 -0
- package/dist/vector/config.js +83 -0
- package/dist/vector/embedding.d.ts +42 -0
- package/dist/vector/embedding.d.ts.map +1 -0
- package/dist/vector/embedding.js +147 -0
- package/dist/vector/types.d.ts +105 -0
- package/dist/vector/types.d.ts.map +1 -0
- package/dist/vector/types.js +5 -0
- package/dist/vector/vector-search.d.ts +47 -0
- package/dist/vector/vector-search.d.ts.map +1 -0
- package/dist/vector/vector-search.js +176 -0
- package/package.json +11 -11
- package/src/api/config.ts +0 -4
- package/src/api/sonamu.ts +21 -36
- package/src/bin/cli.ts +5 -5
- package/src/database/base-model.ts +20 -11
- package/src/database/code-generator.ts +6 -2
- package/src/database/db.ts +1 -0
- package/src/database/puri-wrapper.ts +22 -16
- package/src/database/puri.ts +150 -27
- package/src/database/puri.types.test-d.ts +457 -0
- package/src/database/puri.types.ts +231 -33
- package/src/database/upsert-builder.ts +43 -34
- package/src/entity/entity-manager.ts +2 -2
- package/src/entity/entity.ts +134 -44
- package/src/index.ts +6 -0
- package/src/migration/code-generation.ts +377 -174
- package/src/migration/migration-set.ts +22 -3
- package/src/migration/migrator.ts +6 -0
- package/src/migration/postgresql-schema-reader.ts +121 -21
- package/src/syncer/syncer.ts +4 -3
- package/src/template/implementations/generated.template.ts +51 -9
- package/src/template/implementations/generated_sso.template.ts +71 -2
- package/src/template/implementations/model.template.ts +5 -5
- package/src/template/implementations/model_test.template.ts +3 -3
- package/src/template/implementations/view_enums_dropdown.template.ts +1 -1
- package/src/template/implementations/view_enums_select.template.ts +1 -1
- package/src/template/implementations/view_form.template.ts +11 -8
- package/src/template/implementations/view_id_async_select.template.ts +3 -3
- package/src/template/implementations/view_list.template.ts +11 -8
- package/src/template/implementations/view_search_input.template.ts +1 -1
- package/src/template/template.ts +1 -1
- package/src/template/zod-converter.ts +20 -0
- package/src/testing/fixture-manager.ts +31 -30
- package/src/types/types.ts +226 -48
- package/src/vector/chunking.ts +115 -0
- package/src/vector/config.ts +68 -0
- package/src/vector/embedding.ts +193 -0
- package/src/vector/types.ts +122 -0
- package/src/vector/vector-search.ts +261 -0
- package/dist/template/implementations/view_enums_buttonset.template.d.ts +0 -17
- package/dist/template/implementations/view_enums_buttonset.template.d.ts.map +0 -1
- package/dist/template/implementations/view_enums_buttonset.template.js +0 -31
- package/dist/template/implementations/view_list_columns.template.d.ts +0 -17
- package/dist/template/implementations/view_list_columns.template.d.ts.map +0 -1
- package/dist/template/implementations/view_list_columns.template.js +0 -49
- package/src/template/implementations/view_enums_buttonset.template.ts +0 -34
- package/src/template/implementations/view_list_columns.template.ts +0 -53
|
@@ -1,6 +1,4 @@
|
|
|
1
|
-
/** biome-ignore-all lint/suspicious/noExplicitAny: Puri.types.ts는 다양한 타입을 사용하고 있습니다. */ //
|
|
2
|
-
// - "nothing": DO NOTHING
|
|
3
|
-
// - { update: [...] }: DO UPDATE
|
|
1
|
+
/** biome-ignore-all lint/suspicious/noExplicitAny: Puri.types.ts는 다양한 타입을 사용하고 있습니다. */ // SelectAll 시 모든 조인된 테이블의 컬럼 포함
|
|
4
2
|
export { };
|
|
5
3
|
|
|
6
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/database/puri.types.ts"],"sourcesContent":["/** biome-ignore-all lint/suspicious/noExplicitAny: Puri.types.ts는 다양한 타입을 사용하고 있습니다. */\n\nimport type { QueryResult } from \"pg\";\nimport type { DatabaseSchemaExtend } from \"../types/types\";\nimport type { Puri } from \"./puri\";\nimport type { PuriWrapper } from \"./puri-wrapper\";\n\n// 메타데이터 컬럼 유틸\ntype MetadataColumns = \"__fulltext__\" | \"__virtual__\";\n\n// virtual 컬럼 타입 추출\ntype VirtualKeys<T> = T extends { __virtual__: readonly (infer V)[] } ? V & string : never;\n\n// virtual 컬럼 제거\ntype StripVirtual<T> = Omit<T, VirtualKeys<T>>;\n\n// 메타데이터 필드 제외한 실제 엔티티 컬럼\nexport type ColumnKeys<T> = Exclude<keyof StripVirtual<T>, MetadataColumns> & string;\n\n// virtual 컬럼 제거 후 __fulltext__ 메타데이터 유지\nexport type PuriTable<T> = Omit<StripVirtual<T>, \"__virtual__\">;\n\n// 메타데이터 컬럼 제외 타입 정의\nexport type OmitMetadataColumns<T> = Omit<T, MetadataColumns>;\n\n// TTables의 모든 테이블에서 사용 가능한 컬럼 경로\nexport type AvailableColumns<TTables extends Record<string, any>> =\n  | {\n      [TAlias in keyof TTables]: `${TAlias & string}.${ColumnKeys<TTables[TAlias]>}`;\n    }[keyof TTables]\n  | (IsSingleKey<TTables> extends true\n      ? ColumnKeys<TTables[keyof TTables]> // 단일 테이블이면 컬럼명만도 허용\n      : never);\n\n// Group By, Order By, Having 등에서 선택 가능한 컬럼\nexport type ResultAvailableColumns<TTables extends Record<string, any>, TResult = any> =\n  | AvailableColumns<TTables>\n  | `${keyof TResult & string}`;\n\n// Select 값 타입 확장\nexport type SelectValue<TTables extends Record<string, any>> =\n  | AvailableColumns<TTables>\n  | SqlExpression<\"string\" | \"number\" | \"boolean\" | \"date\">;\n\n// Select 객체 타입 (현재는 컬럼 경로만 지원)\nexport type SelectObject<TTables extends Record<string, any>> = Record<\n  string,\n  SelectValue<TTables> // AvailableColumns 대신\n>;\n\n// Select 결과 타입 추론\nexport type ParseSelectObject<\n  TTables extends Record<string, any>,\n  TSelect extends SelectObject<TTables>,\n> = {\n  [K in keyof TSelect]: TSelect[K] extends SqlExpression<infer R>\n    ? R extends \"string\"\n      ? string\n      : R extends \"number\"\n        ? number\n        : R extends \"boolean\"\n          ? boolean\n          : R extends \"date\"\n            ? Date\n            : never\n    : ExtractColumnType<TTables, TSelect[K] & string>;\n};\n\n// 컬럼 경로에서 타입 추출\nexport type ExtractColumnType<\n  TTables extends Record<string, any>,\n  Path extends string,\n> = Path extends `${infer TAlias}.${infer TColumn}`\n  ? TAlias extends keyof TTables\n    ? TColumn extends keyof TTables[TAlias]\n      ? TTables[TAlias][TColumn]\n      : never\n    : never\n  : IsSingleKey<TTables> extends true // 추가\n    ? Path extends keyof TTables[keyof TTables]\n      ? TTables[keyof TTables][Path]\n      : never\n    : never;\n// Where 조건 객체 타입\n// 예: { \"u.id\": 1, \"u.status\": \"active\" }\nexport type WhereCondition<TTables extends Record<string, any>> = {\n  [key in AvailableColumns<TTables>]?: ExtractColumnType<TTables, key & string>;\n};\n\n// Fulltext index 컬럼 추출 타입\nexport type FulltextColumns<TTables extends Record<string, any>> = {\n  [TAlias in keyof TTables]: TTables[TAlias] extends {\n    __fulltext__: readonly (infer Col)[];\n  }\n    ? Col extends string\n      ? `${TAlias & string}.${Col}`\n      : never\n    : never;\n}[keyof TTables];\n\n// 비교 연산자\nexport type ComparisonOperator = \"=\" | \">\" | \">=\" | \"<\" | \"<=\" | \"<>\" | \"!=\";\n\n// SQL Expression 타입 정의\nexport type SqlExpression<T extends \"string\" | \"number\" | \"boolean\" | \"date\"> = {\n  _type: \"sql_expression\"; // 또는 \"computed_value\"\n  _return: T;\n  _sql: string;\n};\n\n// 결과 타입 가독성을 위한 타입 확장\nexport type Expand<T> = T extends any[]\n  ? { [K in keyof T[0]]: T[0][K] }[] // 배열이면 첫 번째 요소를 Expand하고 배열로 감쌈\n  : T extends object\n    ? { [K in keyof T]: T[K] }\n    : T;\n\ntype IsSingleKey<TTables extends Record<string, any>> = keyof TTables extends infer K\n  ? K extends keyof TTables\n    ? keyof TTables extends K // 역방향 체크로 단일 키 확인\n      ? true\n      : false\n    : false\n  : false;\n\nexport type SingleTableValue<TTables extends Record<string, any>> =\n  IsSingleKey<TTables> extends true ? TTables[keyof TTables] : never;\n\n// Nullable을 Optional로 변환\ntype NullableToOptional<T> = {\n  [K in keyof T as T[K] extends null | undefined ? K : never]?: Exclude<T[K], null | undefined>;\n} & Partial<{\n  [K in keyof T as T[K] extends null | undefined ? never : K]: T[K];\n}>;\n\n// Insert 타입: id, created_at 제외\nexport type InsertData<T> = NullableToOptional<\n  Omit<PuriTable<T>, \"id\" | \"created_at\" | MetadataColumns>\n>;\n\n// Insert Result 타입\nexport type InsertResult = Pick<QueryResult<any>, \"command\" | \"rowCount\" | \"rows\" | \"oid\">;\n\n// SubsetQuery를 위한 타입 유틸리티\ntype ExtractTTables<T extends Puri<any, any, any>> = T extends Puri<any, infer TTables, any>\n  ? TTables\n  : never;\nexport type UnionExtractedTTables<\n  SubsetKey extends string,\n  SubsetQueries extends Record<\n    SubsetKey,\n    (qbWrapper: PuriWrapper<DatabaseSchemaExtend>) => Puri<any, any, any>\n  >,\n> = {\n  [K in SubsetKey]: ExtractTTables<ReturnType<SubsetQueries[K]>>;\n}[SubsetKey];\n\n// ON CONFLICT 대상 타입\n// - 단일 컬럼: \"email\"\n// - 복수 컬럼: [\"user_id\", \"product_id\"]\nexport type OnConflictTarget = string | string[];\n\n// ON CONFLICT 액션 타입\n// - \"nothing\": DO NOTHING\n// - { update: [...] }: DO UPDATE\nexport type OnConflictAction<TTables extends Record<string, unknown>> =\n  | \"nothing\"\n  | {\n      update:\n        | AvailableColumns<TTables>[] // 배열 형태 - [\"name\", \"email\"]\n        | WhereCondition<TTables>; // 객체 형태 - { name: \"John\", count: Puri.rawNumber(...) }\n    };\n"],"names":[],"mappings":"AAAA,sFAAsF,GAkKtF,oBAAoB;AACpB,0BAA0B;AAC1B,iCAAiC;AACjC,WAMM"}
|
|
4
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/database/puri.types.ts"],"sourcesContent":["/** biome-ignore-all lint/suspicious/noExplicitAny: Puri.types.ts는 다양한 타입을 사용하고 있습니다. */\n\nimport type { QueryResult } from \"pg\";\nimport type { DatabaseForeignKeys, DatabaseSchemaExtend } from \"../types/types\";\nimport type { Puri } from \"./puri\";\nimport type { PuriWrapper } from \"./puri-wrapper\";\n\n// ============================================\n// 내부 타입 키 (메타데이터)\n// ============================================\ntype FulltextKey = \"__fulltext__\";\ntype VirtualKey = \"__virtual__\";\ntype LeftJoinedKey = \"__leftJoined__\";\ntype HasDefault = \"__hasDefault__\";\ntype GeneratedKey = \"__generated__\";\n\ntype InternalTypeKeys = FulltextKey | VirtualKey | LeftJoinedKey | HasDefault | GeneratedKey;\n\n// ============================================\n// 타입 유틸리티\n// ============================================\n\n// 테이블명 타입\nexport type TableName<TSchema> = keyof TSchema & string;\n\n// virtual 컬럼 타입 추출\ntype VirtualKeys<T> = T extends { [K in VirtualKey]: readonly (infer V)[] } ? V & string : never;\n\n// virtual 컬럼 제거\ntype StripVirtual<T> = Omit<T, VirtualKeys<T>>;\n\n// LEFT JOIN 마커 - nullable FK로 조인된 테이블\n// 이 마커는 nullable FK + leftJoin 조합에서만 붙습니다.\n// join + FK nullable -> 안 붙음\n// join + FK non-nullable -> 안 붙음\n// leftJoin + FK non-nullable -> 안 붙음\n// leftJoin + FK nullable -> 붙음!\nexport type LeftJoinedMarker = { [K in LeftJoinedKey]: true };\n\n// 메타데이터 필드 제외한 실제 엔티티 컬럼\nexport type ColumnKeys<T> = Exclude<keyof StripVirtual<T>, InternalTypeKeys> & string;\n\n// virtual 컬럼 제거 후 __fulltext__ 유지\nexport type PuriTable<T> = Omit<StripVirtual<T>, VirtualKey>;\n\n// 내부 타입 키 제외 (실제 컬럼만 남김)\nexport type OmitInternalTypeKeys<T> = Omit<T, InternalTypeKeys>;\n\n// TTables의 모든 테이블에서 사용 가능한 컬럼 경로\nexport type AvailableColumns<TTables extends Record<string, any>> =\n  | {\n      [TAlias in keyof TTables]: `${TAlias & string}.${ColumnKeys<TTables[TAlias]>}`;\n    }[keyof TTables]\n  | (IsSingleKey<TTables> extends true\n      ? ColumnKeys<TTables[keyof TTables]> // 단일 테이블이면 컬럼명만도 허용\n      : never);\n\n// 숫자 타입 컬럼만 추출하는 유틸리티 타입\ntype NumericColumnKeys<T> = {\n  [K in keyof T]: T[K] extends number | bigint | null | undefined ? K : never;\n}[keyof T] &\n  string;\n\n// TTables의 모든 테이블에서 숫자 타입 컬럼만 추출\nexport type NumericColumns<TTables extends Record<string, any>> =\n  | {\n      [TAlias in keyof TTables]: `${TAlias & string}.${NumericColumnKeys<TTables[TAlias]>}`;\n    }[keyof TTables]\n  | (IsSingleKey<TTables> extends true\n      ? NumericColumnKeys<TTables[keyof TTables]> // 단일 테이블이면 컬럼명만도 허용\n      : never);\n\n// Group By, Order By, Having 등에서 선택 가능한 컬럼\nexport type ResultAvailableColumns<TTables extends Record<string, any>, TResult = any> =\n  | AvailableColumns<TTables>\n  | `${keyof TResult & string}`;\n\n// Select 값 타입 확장 (단일 컬럼 또는 SQL 표현식)\nexport type SelectValue<TTables extends Record<string, any>> =\n  | AvailableColumns<TTables>\n  | SqlExpression<\"string\" | \"number\" | \"boolean\" | \"date\">;\n\n// 중첩 Select 객체 타입 (재귀적)\n// 예: { parent: { id: \"parent.id\", name: \"parent.name\" } }\nexport type NestedSelectObject<TTables extends Record<string, any>> = {\n  [key: string]: SelectValue<TTables> | NestedSelectObject<TTables>;\n};\n\n// Select 객체 타입 (flat 또는 중첩 허용)\nexport type SelectObject<TTables extends Record<string, any>> = NestedSelectObject<TTables>;\n\n// 값이 중첩 객체인지 판별하는 헬퍼 타입\ntype IsNestedObject<T> = T extends string\n  ? false\n  : T extends SqlExpression<any>\n    ? false\n    : T extends Record<string, any>\n      ? true\n      : false;\n\n// 컬럼이 nullable인지 확인 (스키마에서 직접 추출)\n// 예: IsNullableColumn<TTables, \"employees.department_id\"> → department_id가 number | null이면 true\nexport type IsNullableColumn<\n  TTables,\n  Path extends string,\n> = Path extends `${infer TAlias}.${infer TColumn}`\n  ? TAlias extends keyof TTables\n    ? TColumn extends keyof TTables[TAlias]\n      ? null extends TTables[TAlias][TColumn]\n        ? true\n        : false\n      : false\n    : false\n  : false;\n\n// FK nullable 여부에 따른 마커 타입 결정\n// nullable FK로 leftJoin → LeftJoinedMarker (객체 자체가 null일 수 있음)\n// non-null FK로 leftJoin → 마커 없음 (부모가 있으면 자식도 반드시 있음)\nexport type LeftJoinMarkerFor<TTables, Path extends string> = IsNullableColumn<\n  TTables,\n  Path\n> extends true\n  ? LeftJoinedMarker\n  : {};\n\n// 주어진 테이블이 FK nullable로 leftJoin 된 테이블인지 확인합니다.\n// 사실 LeftJoinMarker가 붙었는지 확인하는게 다입니다.\n// 이 마커는 FK nullable + leftJoin 조합에서만 붙습니다.\ntype IsNullableJoinedTable<TTables, TableKey> = TableKey extends keyof TTables\n  ? TTables[TableKey] extends LeftJoinedMarker\n    ? true // LeftJoinedMarker가 있으면 nullable\n    : false\n  : false;\n\n// 경로 조합 헬퍼 (prefix가 없으면 key만, 있으면 prefix__key)\ntype JoinPath<Prefix extends string, Key extends string> = Prefix extends \"\"\n  ? Key\n  : `${Prefix}__${Key}`;\n\n// Select 결과 타입을 추론해주는 친구입니다.\n// 이 타입은 Puri의 select, appendSelect에서 TResult로 사용됩니다.\n//\n// Schema를 읽어서 FK의 nullability에 따라 join된 객체의 타입을 추론해주는 기능이 있습니다.\n// 이게 무슨 소리냐? FK가 nullable인데 leftJoin되었다면, 해당 객체는 nullable 해야 함을 타입 추론으로 반영해준다는 것입니다.\n// 반면 FK가 non-nullable이거나 그냥 join으로 이어졌다면 해당 객체는 non-nullable할 겁니다.\n// 물론 객체 내부의 nullability는 또 별개로 추론됩니다. \n// \n// 아래에도 ParseSelectObjectWithPath를 비롯해 ExtractColumnType, ExtractColumnTypeRaw 등의 타입이 있습니다.\n// 이들의 역할은 다음과 같습니다:\n// - Parse*: 객체 레벨에서 중첩 구조를 순회하며 객체에 | null을 붙일지 결정합니다.\n// - Extract*: 필드 레벨에서 \"table.column\" 경로로부터 실제 타입을 추출합니다.\n//\n// 예시:\n// .select({\n//   id: \"users.id\",           // ← ExtractColumnType의 결과는 number입니다.\n//   department: {             // ← ParseSelectObjectInner에 의해 nullable 객체로 추론됩니다.\n//     id: \"department.id\",    // ← ExtractColumnTypeRaw의 결과는 number입니다.\n//     name: \"department.name\" // ← ExtractColumnTypeRaw의 결과는 string입니다.\n//   }\n// })\nexport type ParseSelectObject<\n  TTables extends Record<string, any>,\n  TSelect extends SelectObject<TTables>,\n> = ParseSelectObjectWithPath<TTables, TSelect, \"\">;\n\n// 경로를 추적하면서 Select 결과 타입을 추론합니다.\ntype ParseSelectObjectWithPath<\n  TTables extends Record<string, any>,\n  TSelect extends SelectObject<TTables>,\n  Prefix extends string,\n> = Expand<{\n  [K in keyof TSelect]: TSelect[K] extends SqlExpression<infer R>\n    ? R extends \"string\"\n      ? string\n      : R extends \"number\"\n        ? number\n        : R extends \"boolean\"\n          ? boolean\n          : R extends \"date\"\n            ? Date\n            : never\n    : IsNestedObject<TSelect[K]> extends true\n      ? TSelect[K] extends NestedSelectObject<TTables>\n        ? IsNullableJoinedTable<TTables, JoinPath<Prefix, K & string>> extends true // 주어진 테이블이 FK nullable에 leftJoin되었는지 여부에 따라 select 결과 객체의 타입이 달라집니다.\n          ? Expand<ParseSelectObjectInner<TTables, TSelect[K], JoinPath<Prefix, K & string>>> | null // 만약 해당한다면 해당 객체 자체는 nullable 하며,\n          : Expand<ParseSelectObjectInner<TTables, TSelect[K], JoinPath<Prefix, K & string>>> // 그렇지 않다면 non-nullable 합니다.\n        : never\n      : ExtractColumnType<TTables, TSelect[K] & string>;\n}>;\n\n// 중첩 객체 내부용 - leftJoin nullable을 객체 레벨에서 이미 처리했으므로 필드는 원본 타입을 사용합니다.\n// ParseSelectObjectWithPath와 거의 동일하나, 마지막에 ExtractColumnType 대신 ExtractColumnTypeRaw를 사용하여\n// 필드 레벨에서 중복으로 | null이 추가되는 것을 방지합니다.\ntype ParseSelectObjectInner<\n  TTables extends Record<string, any>,\n  TSelect extends SelectObject<TTables>,\n  Prefix extends string,\n> = Expand<{\n  [K in keyof TSelect]: TSelect[K] extends SqlExpression<infer R>\n    ? R extends \"string\"\n      ? string\n      : R extends \"number\"\n        ? number\n        : R extends \"boolean\"\n          ? boolean\n          : R extends \"date\"\n            ? Date\n            : never\n    : IsNestedObject<TSelect[K]> extends true\n      ? TSelect[K] extends NestedSelectObject<TTables>\n        ? IsNullableJoinedTable<TTables, JoinPath<Prefix, K & string>> extends true\n          ? Expand<ParseSelectObjectInner<TTables, TSelect[K], JoinPath<Prefix, K & string>>> | null\n          : Expand<ParseSelectObjectInner<TTables, TSelect[K], JoinPath<Prefix, K & string>>>\n        : never\n      : ExtractColumnTypeRaw<TTables, TSelect[K] & string>; // leftJoin nullable 무시\n}>;\n\n// 컬럼 경로에서 타입을 추출합니다. LeftJoinedMarker가 있으면 | null을 추가합니다.\n// 최상위 select 필드에서 사용됩니다.\nexport type ExtractColumnType<\n  TTables extends Record<string, any>,\n  Path extends string,\n> = Path extends `${infer TAlias}.${infer TColumn}`\n  ? TAlias extends keyof TTables\n    ? TColumn extends keyof TTables[TAlias]\n      ? TTables[TAlias] extends LeftJoinedMarker\n        ? TTables[TAlias][TColumn] | null // LEFT JOIN (nullable FK) → nullable\n        : TTables[TAlias][TColumn] // INNER JOIN 또는 non-null FK leftJoin → non-nullable\n      : never\n    : never\n  : IsSingleKey<TTables> extends true\n    ? Path extends keyof TTables[keyof TTables]\n      ? TTables[keyof TTables][Path]\n      : never\n    : never;\n\n// 컬럼 경로에서 타입을 추출합니다. leftJoin 여부와 관계없이 원본 타입을 반환합니다.\n// 중첩 객체 내부 필드에서 사용됩니다. (객체 레벨에서 이미 | null 처리가 완료되었으므로)\ntype ExtractColumnTypeRaw<\n  TTables extends Record<string, any>,\n  Path extends string,\n> = Path extends `${infer TAlias}.${infer TColumn}`\n  ? TAlias extends keyof TTables\n    ? TColumn extends keyof TTables[TAlias]\n      ? TTables[TAlias][TColumn] // leftJoin 여부와 관계없이 원본 타입\n      : never\n    : never\n  : IsSingleKey<TTables> extends true\n    ? Path extends keyof TTables[keyof TTables]\n      ? TTables[keyof TTables][Path]\n      : never\n    : never;\n\n// Where 조건 객체 타입\n// 예: { \"u.id\": 1, \"u.status\": \"active\" }\nexport type WhereCondition<TTables extends Record<string, any>> = {\n  [key in AvailableColumns<TTables>]?: ExtractColumnType<TTables, key & string>;\n};\n\n// Fulltext index 컬럼 추출 타입\nexport type FulltextColumns<TTables extends Record<string, any>> = {\n  [TAlias in keyof TTables]: TTables[TAlias] extends {\n    [K in FulltextKey]: readonly (infer Col)[];\n  }\n    ? Col extends string\n      ? `${TAlias & string}.${Col}`\n      : never\n    : never;\n}[keyof TTables];\n\n// 비교 연산자\nexport type ComparisonOperator = \"=\" | \">\" | \">=\" | \"<\" | \"<=\" | \"<>\" | \"!=\";\n// 조건 연산자: 비교 연산자 + 패턴 매칭 연산자\nexport type WhereOperator = ComparisonOperator | \"like\" | \"not like\";\n\n// SQL Expression 타입 정의\nexport type SqlExpression<T extends \"string\" | \"number\" | \"boolean\" | \"date\"> = {\n  _type: \"sql_expression\"; // 또는 \"computed_value\"\n  _return: T;\n  _sql: string;\n};\n\n// 결과 타입 가독성을 위한 타입 확장\nexport type Expand<T> = T extends any[]\n  ? { [K in keyof T[0]]: T[0][K] }[] // 배열이면 첫 번째 요소를 Expand하고 배열로 감쌈\n  : T extends object\n    ? { [K in keyof T]: T[K] }\n    : T;\n\ntype IsSingleKey<TTables extends Record<string, any>> = keyof TTables extends infer K\n  ? K extends keyof TTables\n    ? keyof TTables extends K // 역방향 체크로 단일 키 확인\n      ? true\n      : false\n    : false\n  : false;\n\nexport type SingleTableValue<TTables extends Record<string, any>> =\n  IsSingleKey<TTables> extends true ? TTables[keyof TTables] : never;\n\n// __hasDefault__에 포함된 키들을 PuriTable<T>의 키로 제한\ntype HasDefaultKeys<T> = T extends { __hasDefault__: readonly (infer K)[] }\n  ? Extract<K, keyof PuriTable<T>>\n  : never;\n\n// __generated__에 포함된 키들 (INSERT 시 제외해야 함)\ntype GeneratedKeys<T> = T extends { __generated__: readonly (infer K)[] }\n  ? Extract<K, keyof PuriTable<T>>\n  : never;\n\n// Insert 타입: 메타데이터 제거 후, __hasDefault__ 컬럼들만 optional로 처리, __generated__ 컬럼은 완전히 제외\nexport type InsertData<T> = Omit<\n  PuriTable<T>,\n  InternalTypeKeys | HasDefaultKeys<T> | GeneratedKeys<T>\n> & {\n  [K in HasDefaultKeys<T>]?: PuriTable<T>[K];\n};\n\n// Insert Result 타입\nexport type InsertResult = Pick<QueryResult<any>, \"command\" | \"rowCount\" | \"rows\" | \"oid\">;\n\n// SubsetQuery를 위한 타입 유틸리티\ntype ExtractTTables<T extends Puri<any, any, any>> = T extends Puri<any, infer TTables, any>\n  ? TTables\n  : never;\nexport type UnionExtractedTTables<\n  SubsetKey extends string,\n  SubsetQueries extends Record<\n    SubsetKey,\n    (qbWrapper: PuriWrapper<DatabaseSchemaExtend>) => Puri<any, any, any>\n  >,\n> = {\n  [K in SubsetKey]: ExtractTTables<ReturnType<SubsetQueries[K]>>;\n}[SubsetKey];\n\n// ON CONFLICT 대상 타입\n// - 단일 컬럼: \"email\"\n// - 복수 컬럼: [\"user_id\", \"product_id\"]\nexport type OnConflictTarget = string | string[];\n\n// ON CONFLICT 액션 타입\n// - \"nothing\": DO NOTHING\n// - { update: [...] }: DO UPDATE\nexport type OnConflictAction<TTables extends Record<string, unknown>> =\n  | \"nothing\"\n  | {\n      update:\n        | AvailableColumns<TTables>[] // 배열 형태 - [\"name\", \"email\"]\n        | WhereCondition<TTables>; // 객체 형태 - { name: \"John\", count: Puri.rawNumber(...) }\n    };\n\n// FK 컬럼명 추출 유틸리티 타입 - DatabaseForeignKeys 활용\nexport type ForeignKeyColumns<TTable extends TableName<DatabaseSchemaExtend>> =\n  TTable extends keyof DatabaseForeignKeys ? DatabaseForeignKeys[TTable] : never;\n\n// Union을 Intersection으로 변환하는 유틸리티\ntype UnionToIntersection<U> = (U extends any ? (k: U) => void : never) extends (k: infer I) => void\n  ? I\n  : never;\n\n// SelectAll 시 모든 조인된 테이블의 컬럼 포함\nexport type SelectAllResult<TTables extends Record<string, any>> = UnionToIntersection<\n  {\n    [K in keyof TTables]: TTables[K] extends infer T\n      ? T extends LeftJoinedMarker\n        ? Partial<OmitInternalTypeKeys<T>> // LEFT JOIN은 nullable, 메타데이터 제거\n        : OmitInternalTypeKeys<T> // INNER JOIN은 non-nullable, 메타데이터 제거\n      : never;\n  }[keyof TTables]\n>;\n"],"names":[],"mappings":"AAAA,sFAAsF,GAwWtF,gCAAgC;AAChC,WAQE"}
|
|
@@ -0,0 +1,129 @@
|
|
|
1
|
+
import { describe, expectTypeOf, it } from "vitest";
|
|
2
|
+
// ============================================================================
|
|
3
|
+
// ExtractColumnType 테스트
|
|
4
|
+
// ============================================================================
|
|
5
|
+
describe("ExtractColumnType", ()=>{
|
|
6
|
+
describe("단일 테이블", ()=>{
|
|
7
|
+
it("기본 컬럼 타입을 추출한다", ()=>{
|
|
8
|
+
const result = {};
|
|
9
|
+
expectTypeOf(result).toEqualTypeOf();
|
|
10
|
+
});
|
|
11
|
+
it("nullable 컬럼 타입을 추출한다", ()=>{
|
|
12
|
+
const result = {};
|
|
13
|
+
expectTypeOf(result).toEqualTypeOf();
|
|
14
|
+
});
|
|
15
|
+
it("단일 테이블에서는 테이블명 없이 컬럼명만으로 추출 가능하다", ()=>{
|
|
16
|
+
const result = {};
|
|
17
|
+
expectTypeOf(result).toEqualTypeOf();
|
|
18
|
+
});
|
|
19
|
+
});
|
|
20
|
+
describe("innerJoin된 테이블", ()=>{
|
|
21
|
+
it("innerJoin 테이블의 컬럼은 non-null이다", ()=>{
|
|
22
|
+
const result = {};
|
|
23
|
+
expectTypeOf(result).toEqualTypeOf();
|
|
24
|
+
});
|
|
25
|
+
});
|
|
26
|
+
describe("leftJoin된 테이블", ()=>{
|
|
27
|
+
it("leftJoin 테이블의 컬럼은 nullable이다", ()=>{
|
|
28
|
+
const result = {};
|
|
29
|
+
expectTypeOf(result).toEqualTypeOf();
|
|
30
|
+
});
|
|
31
|
+
it("leftJoin 테이블의 원래 nullable 컬럼도 nullable이다", ()=>{
|
|
32
|
+
const result = {};
|
|
33
|
+
expectTypeOf(result).toEqualTypeOf();
|
|
34
|
+
});
|
|
35
|
+
});
|
|
36
|
+
describe("non-null FK로 leftJoin된 테이블", ()=>{
|
|
37
|
+
it("non-null FK로 leftJoin된 테이블의 컬럼은 non-null이다 (마커 없음)", ()=>{
|
|
38
|
+
const result = {};
|
|
39
|
+
expectTypeOf(result).toEqualTypeOf();
|
|
40
|
+
});
|
|
41
|
+
});
|
|
42
|
+
});
|
|
43
|
+
// ============================================================================
|
|
44
|
+
// ParseSelectObject 테스트
|
|
45
|
+
// ============================================================================
|
|
46
|
+
describe("ParseSelectObject", ()=>{
|
|
47
|
+
describe("단일 테이블 (flat select)", ()=>{
|
|
48
|
+
it("기본 필드를 파싱한다", ()=>{
|
|
49
|
+
const result = {};
|
|
50
|
+
expectTypeOf(result).toEqualTypeOf();
|
|
51
|
+
});
|
|
52
|
+
it("nullable 필드를 파싱한다", ()=>{
|
|
53
|
+
const result = {};
|
|
54
|
+
expectTypeOf(result).toEqualTypeOf();
|
|
55
|
+
});
|
|
56
|
+
});
|
|
57
|
+
describe("innerJoin + flat select", ()=>{
|
|
58
|
+
it("innerJoin 테이블의 필드는 non-null이다", ()=>{
|
|
59
|
+
const result = {};
|
|
60
|
+
expectTypeOf(result).toEqualTypeOf();
|
|
61
|
+
});
|
|
62
|
+
});
|
|
63
|
+
describe("leftJoin + flat select", ()=>{
|
|
64
|
+
it("leftJoin 테이블의 필드는 nullable이다", ()=>{
|
|
65
|
+
const result = {};
|
|
66
|
+
expectTypeOf(result).toEqualTypeOf();
|
|
67
|
+
});
|
|
68
|
+
});
|
|
69
|
+
// ============================================================================
|
|
70
|
+
// 핵심: 입체적 select 구조
|
|
71
|
+
// ============================================================================
|
|
72
|
+
describe("innerJoin + nested select (입체적 구조)", ()=>{
|
|
73
|
+
it("innerJoin 테이블의 중첩 객체는 non-null이다", ()=>{
|
|
74
|
+
const result = {};
|
|
75
|
+
expectTypeOf(result).toEqualTypeOf();
|
|
76
|
+
});
|
|
77
|
+
});
|
|
78
|
+
describe("leftJoin + nested select (입체적 구조)", ()=>{
|
|
79
|
+
it("leftJoin 테이블의 중첩 객체는 nullable이다 (필드는 non-null)", ()=>{
|
|
80
|
+
const result = {};
|
|
81
|
+
expectTypeOf(result).toEqualTypeOf();
|
|
82
|
+
});
|
|
83
|
+
it("leftJoin 테이블 내의 원래 nullable 필드도 non-null이다", ()=>{
|
|
84
|
+
const result = {};
|
|
85
|
+
// employee 객체가 null이 아닐 때만 접근하므로, salary의 원래 nullability 유지
|
|
86
|
+
expectTypeOf(result).toEqualTypeOf();
|
|
87
|
+
});
|
|
88
|
+
});
|
|
89
|
+
describe("non-null FK leftJoin + nested select (입체적 구조)", ()=>{
|
|
90
|
+
it("non-null FK로 leftJoin된 테이블의 중첩 객체는 non-null이다", ()=>{
|
|
91
|
+
const result = {};
|
|
92
|
+
expectTypeOf(result).toEqualTypeOf();
|
|
93
|
+
});
|
|
94
|
+
it("깊은 중첩에서도 non-null FK leftJoin은 non-null이다", ()=>{
|
|
95
|
+
const result = {};
|
|
96
|
+
expectTypeOf(result).toEqualTypeOf();
|
|
97
|
+
});
|
|
98
|
+
});
|
|
99
|
+
describe("복합 케이스", ()=>{
|
|
100
|
+
it("innerJoin + nullable FK leftJoin + non-null FK leftJoin 조합", ()=>{
|
|
101
|
+
const result = {};
|
|
102
|
+
expectTypeOf(result).toEqualTypeOf();
|
|
103
|
+
});
|
|
104
|
+
it("여러 leftJoin 관계", ()=>{
|
|
105
|
+
const result = {};
|
|
106
|
+
expectTypeOf(result).toEqualTypeOf();
|
|
107
|
+
});
|
|
108
|
+
});
|
|
109
|
+
});
|
|
110
|
+
// ============================================================================
|
|
111
|
+
// AvailableColumns 테스트
|
|
112
|
+
// ============================================================================
|
|
113
|
+
describe("AvailableColumns", ()=>{
|
|
114
|
+
it("단일 테이블에서 사용 가능한 컬럼을 추출한다", ()=>{
|
|
115
|
+
// "users.id" | "users.name" | "users.email" | "users.department_id" | "id" | "name" | "email" | "department_id"
|
|
116
|
+
const valid1 = "users.id";
|
|
117
|
+
const valid2 = "id"; // 단일 테이블이면 테이블명 생략 가능
|
|
118
|
+
expectTypeOf(valid1).toExtend();
|
|
119
|
+
expectTypeOf(valid2).toExtend();
|
|
120
|
+
});
|
|
121
|
+
it("여러 테이블에서 사용 가능한 컬럼을 추출한다", ()=>{
|
|
122
|
+
const valid1 = "users.id";
|
|
123
|
+
const valid2 = "department.name";
|
|
124
|
+
expectTypeOf(valid1).toExtend();
|
|
125
|
+
expectTypeOf(valid2).toExtend();
|
|
126
|
+
});
|
|
127
|
+
});
|
|
128
|
+
|
|
129
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/database/puri.types.test-d.ts"],"sourcesContent":["import { describe, expectTypeOf, it } from \"vitest\";\nimport type {\n  AvailableColumns,\n  ExtractColumnType,\n  LeftJoinedMarker,\n  ParseSelectObject,\n} from \"./puri.types\";\n\n// ============================================================================\n// 테스트용 Mock 스키마\n// ============================================================================\n\ntype MockSchema = {\n  users: {\n    id: number;\n    name: string;\n    email: string;\n    department_id: number | null;\n  };\n  departments: {\n    id: number;\n    name: string;\n    company_id: number;\n  };\n  companies: {\n    id: number;\n    name: string;\n  };\n  employees: {\n    id: number;\n    employee_number: string;\n    salary: string | null; // nullable 필드\n    user_id: number;\n    department_id: number | null;\n  };\n};\n\n// ============================================================================\n// ExtractColumnType 테스트\n// ============================================================================\n\ndescribe(\"ExtractColumnType\", () => {\n  describe(\"단일 테이블\", () => {\n    it(\"기본 컬럼 타입을 추출한다\", () => {\n      type Tables = { users: MockSchema[\"users\"] };\n      type Result = ExtractColumnType<Tables, \"users.id\">;\n\n      const result = {} as Result;\n      expectTypeOf(result).toEqualTypeOf<number>();\n    });\n\n    it(\"nullable 컬럼 타입을 추출한다\", () => {\n      type Tables = { users: MockSchema[\"users\"] };\n      type Result = ExtractColumnType<Tables, \"users.department_id\">;\n\n      const result = {} as Result;\n      expectTypeOf(result).toEqualTypeOf<number | null>();\n    });\n\n    it(\"단일 테이블에서는 테이블명 없이 컬럼명만으로 추출 가능하다\", () => {\n      type Tables = { users: MockSchema[\"users\"] };\n      type Result = ExtractColumnType<Tables, \"id\">;\n\n      const result = {} as Result;\n      expectTypeOf(result).toEqualTypeOf<number>();\n    });\n  });\n\n  describe(\"innerJoin된 테이블\", () => {\n    it(\"innerJoin 테이블의 컬럼은 non-null이다\", () => {\n      type Tables = {\n        users: MockSchema[\"users\"];\n        department: MockSchema[\"departments\"]; // innerJoin\n      };\n      type Result = ExtractColumnType<Tables, \"department.id\">;\n\n      const result = {} as Result;\n      expectTypeOf(result).toEqualTypeOf<number>();\n    });\n  });\n\n  describe(\"leftJoin된 테이블\", () => {\n    it(\"leftJoin 테이블의 컬럼은 nullable이다\", () => {\n      type Tables = {\n        users: MockSchema[\"users\"];\n        department: MockSchema[\"departments\"] & LeftJoinedMarker; // leftJoin\n      };\n      type Result = ExtractColumnType<Tables, \"department.id\">;\n\n      const result = {} as Result;\n      expectTypeOf(result).toEqualTypeOf<number | null>();\n    });\n\n    it(\"leftJoin 테이블의 원래 nullable 컬럼도 nullable이다\", () => {\n      type Tables = {\n        users: MockSchema[\"users\"];\n        employee: MockSchema[\"employees\"] & LeftJoinedMarker; // leftJoin\n      };\n      type Result = ExtractColumnType<Tables, \"employee.salary\">;\n\n      const result = {} as Result;\n      expectTypeOf(result).toEqualTypeOf<string | null>();\n    });\n  });\n\n  describe(\"non-null FK로 leftJoin된 테이블\", () => {\n    it(\"non-null FK로 leftJoin된 테이블의 컬럼은 non-null이다 (마커 없음)\", () => {\n      type Tables = {\n        users: MockSchema[\"users\"];\n        department: MockSchema[\"departments\"] & LeftJoinedMarker; // nullable FK\n        company: MockSchema[\"companies\"] & LeftJoinedMarker; // non-null FK → 마커 없음\n      };\n      type Result = ExtractColumnType<Tables, \"company.id\">;\n\n      const result = {} as Result;\n      expectTypeOf(result).toEqualTypeOf<number>();\n    });\n  });\n});\n\n// ============================================================================\n// ParseSelectObject 테스트\n// ============================================================================\n\ndescribe(\"ParseSelectObject\", () => {\n  describe(\"단일 테이블 (flat select)\", () => {\n    it(\"기본 필드를 파싱한다\", () => {\n      type Tables = { users: MockSchema[\"users\"] };\n      type Select = {\n        id: \"users.id\";\n        name: \"users.name\";\n      };\n      type Result = ParseSelectObject<Tables, Select>;\n\n      const result = {} as Result;\n      expectTypeOf(result).toEqualTypeOf<{\n        id: number;\n        name: string;\n      }>();\n    });\n\n    it(\"nullable 필드를 파싱한다\", () => {\n      type Tables = { users: MockSchema[\"users\"] };\n      type Select = {\n        id: \"users.id\";\n        department_id: \"users.department_id\";\n      };\n      type Result = ParseSelectObject<Tables, Select>;\n\n      const result = {} as Result;\n      expectTypeOf(result).toEqualTypeOf<{\n        id: number;\n        department_id: number | null;\n      }>();\n    });\n  });\n\n  describe(\"innerJoin + flat select\", () => {\n    it(\"innerJoin 테이블의 필드는 non-null이다\", () => {\n      type Tables = {\n        users: MockSchema[\"users\"];\n        department: MockSchema[\"departments\"]; // innerJoin\n      };\n      type Select = {\n        id: \"users.id\";\n        dept_id: \"department.id\";\n        dept_name: \"department.name\";\n      };\n      type Result = ParseSelectObject<Tables, Select>;\n\n      const result = {} as Result;\n      expectTypeOf(result).toEqualTypeOf<{\n        id: number;\n        dept_id: number;\n        dept_name: string;\n      }>();\n    });\n  });\n\n  describe(\"leftJoin + flat select\", () => {\n    it(\"leftJoin 테이블의 필드는 nullable이다\", () => {\n      type Tables = {\n        users: MockSchema[\"users\"];\n        department: MockSchema[\"departments\"] & LeftJoinedMarker; // leftJoin\n      };\n      type Select = {\n        id: \"users.id\";\n        dept_id: \"department.id\";\n        dept_name: \"department.name\";\n      };\n      type Result = ParseSelectObject<Tables, Select>;\n\n      const result = {} as Result;\n      expectTypeOf(result).toEqualTypeOf<{\n        id: number;\n        dept_id: number | null;\n        dept_name: string | null;\n      }>();\n    });\n  });\n\n  // ============================================================================\n  // 핵심: 입체적 select 구조\n  // ============================================================================\n\n  describe(\"innerJoin + nested select (입체적 구조)\", () => {\n    it(\"innerJoin 테이블의 중첩 객체는 non-null이다\", () => {\n      type Tables = {\n        users: MockSchema[\"users\"];\n        department: MockSchema[\"departments\"]; // innerJoin\n      };\n      type Select = {\n        id: \"users.id\";\n        department: {\n          id: \"department.id\";\n          name: \"department.name\";\n        };\n      };\n      type Result = ParseSelectObject<Tables, Select>;\n\n      const result = {} as Result;\n      expectTypeOf(result).toEqualTypeOf<{\n        id: number;\n        department: {\n          id: number;\n          name: string;\n        };\n      }>();\n    });\n  });\n\n  describe(\"leftJoin + nested select (입체적 구조)\", () => {\n    it(\"leftJoin 테이블의 중첩 객체는 nullable이다 (필드는 non-null)\", () => {\n      type Tables = {\n        users: MockSchema[\"users\"];\n        department: MockSchema[\"departments\"] & LeftJoinedMarker; // leftJoin\n      };\n      type Select = {\n        id: \"users.id\";\n        department: {\n          id: \"department.id\";\n          name: \"department.name\";\n        };\n      };\n      type Result = ParseSelectObject<Tables, Select>;\n\n      const result = {} as Result;\n      expectTypeOf(result).toEqualTypeOf<{\n        id: number;\n        department: {\n          id: number;\n          name: string;\n        } | null; // 객체 단위로 nullable\n      }>();\n    });\n\n    it(\"leftJoin 테이블 내의 원래 nullable 필드도 non-null이다\", () => {\n      type Tables = {\n        users: MockSchema[\"users\"];\n        employee: MockSchema[\"employees\"] & LeftJoinedMarker; // leftJoin\n      };\n      type Select = {\n        id: \"users.id\";\n        employee: {\n          id: \"employee.id\";\n          salary: \"employee.salary\"; // 스키마상 nullable이지만...\n        };\n      };\n      type Result = ParseSelectObject<Tables, Select>;\n\n      const result = {} as Result;\n      // employee 객체가 null이 아닐 때만 접근하므로, salary의 원래 nullability 유지\n      expectTypeOf(result).toEqualTypeOf<{\n        id: number;\n        employee: {\n          id: number;\n          salary: string | null; // 스키마의 원래 nullability 유지\n        } | null;\n      }>();\n    });\n  });\n\n  describe(\"non-null FK leftJoin + nested select (입체적 구조)\", () => {\n    it(\"non-null FK로 leftJoin된 테이블의 중첩 객체는 non-null이다\", () => {\n      type Tables = {\n        users: MockSchema[\"users\"];\n        department: MockSchema[\"departments\"] & LeftJoinedMarker; // nullable FK → 마커 있음\n        department__company: MockSchema[\"companies\"]; // non-null FK → 마커 없음\n      };\n      type Select = {\n        id: \"users.id\";\n        department: {\n          id: \"department.id\";\n          name: \"department.name\";\n          company: {\n            name: \"department__company.name\";\n          };\n        };\n      };\n      type Result = ParseSelectObject<Tables, Select>;\n\n      const result = {} as Result;\n      expectTypeOf(result).toEqualTypeOf<{\n        id: number;\n        department: {\n          id: number;\n          name: string;\n          company: {\n            name: string;\n          }; // non-null! (non-null FK로 조인되어 마커 없음)\n        } | null;\n      }>();\n    });\n\n    it(\"깊은 중첩에서도 non-null FK leftJoin은 non-null이다\", () => {\n      type Tables = {\n        employees: MockSchema[\"employees\"];\n        user: MockSchema[\"users\"]; // innerJoin → 마커 없음\n        user__employee: MockSchema[\"employees\"] & LeftJoinedMarker; // nullable FK\n        user__employee__department: MockSchema[\"departments\"]; // non-null FK → 마커 없음\n      };\n      type Select = {\n        id: \"employees.id\";\n        user: {\n          id: \"user.id\";\n          employee: {\n            id: \"user__employee.id\";\n            department: {\n              id: \"user__employee__department.id\";\n            };\n          };\n        };\n      };\n      type Result = ParseSelectObject<Tables, Select>;\n\n      const result = {} as Result;\n      expectTypeOf(result).toEqualTypeOf<{\n        id: number;\n        user: {\n          // user는 innerJoin이므로 무조건 존재합니다.\n          id: number;\n          employee: {\n            // user의 employee는 nullable FK leftJoin이므로 null일 수 있습니다.\n            id: number;\n            department: {\n              // employee가 존재한다면 department는 non-null FK leftJoin이므로 무조건 존재합니다.\n              id: number;\n            }; // non-null FK → non-null\n          } | null; // nullable FK leftJoin → nullable\n        }; // innerJoin → non-null\n      }>();\n    });\n  });\n\n  describe(\"복합 케이스\", () => {\n    it(\"innerJoin + nullable FK leftJoin + non-null FK leftJoin 조합\", () => {\n      type Tables = {\n        employees: MockSchema[\"employees\"];\n        user: MockSchema[\"users\"]; // innerJoin (non-null FK)\n        department: MockSchema[\"departments\"] & LeftJoinedMarker; // nullable FK leftJoin\n        department__company: MockSchema[\"companies\"]; // non-null FK leftJoin → 마커 없음\n      };\n      type Select = {\n        id: \"employees.id\";\n        employee_number: \"employees.employee_number\";\n        salary: \"employees.salary\";\n        user: {\n          id: \"user.id\";\n          username: \"user.name\";\n        };\n        department: {\n          id: \"department.id\";\n          name: \"department.name\";\n          company: {\n            name: \"department__company.name\";\n          };\n        };\n      };\n      type Result = ParseSelectObject<Tables, Select>;\n\n      const result = {} as Result;\n      expectTypeOf(result).toEqualTypeOf<{\n        id: number;\n        employee_number: string;\n        salary: string | null; // 스키마상 nullable\n        user: {\n          id: number;\n          username: string;\n        }; // innerJoin → non-null\n        department: {\n          id: number;\n          name: string;\n          company: {\n            name: string;\n          }; // non-null FK → non-null\n        } | null; // nullable FK leftJoin → nullable\n      }>();\n    });\n\n    it(\"여러 leftJoin 관계\", () => {\n      type Tables = {\n        employees: MockSchema[\"employees\"];\n        department: MockSchema[\"departments\"] & LeftJoinedMarker;\n        manager: MockSchema[\"users\"] & LeftJoinedMarker;\n      };\n      type Select = {\n        id: \"employees.id\";\n        department: {\n          name: \"department.name\";\n        };\n        manager: {\n          name: \"manager.name\";\n        };\n      };\n      type Result = ParseSelectObject<Tables, Select>;\n\n      const result = {} as Result;\n      expectTypeOf(result).toEqualTypeOf<{\n        id: number;\n        department: { name: string } | null;\n        manager: { name: string } | null;\n      }>();\n    });\n  });\n});\n\n// ============================================================================\n// AvailableColumns 테스트\n// ============================================================================\n\ndescribe(\"AvailableColumns\", () => {\n  it(\"단일 테이블에서 사용 가능한 컬럼을 추출한다\", () => {\n    type Tables = { users: MockSchema[\"users\"] };\n    type Result = AvailableColumns<Tables>;\n\n    // \"users.id\" | \"users.name\" | \"users.email\" | \"users.department_id\" | \"id\" | \"name\" | \"email\" | \"department_id\"\n    const valid1: Result = \"users.id\";\n    const valid2: Result = \"id\"; // 단일 테이블이면 테이블명 생략 가능\n\n    expectTypeOf(valid1).toExtend<Result>();\n    expectTypeOf(valid2).toExtend<Result>();\n  });\n\n  it(\"여러 테이블에서 사용 가능한 컬럼을 추출한다\", () => {\n    type Tables = {\n      users: MockSchema[\"users\"];\n      department: MockSchema[\"departments\"];\n    };\n    type Result = AvailableColumns<Tables>;\n\n    const valid1: Result = \"users.id\";\n    const valid2: Result = \"department.name\";\n\n    expectTypeOf(valid1).toExtend<Result>();\n    expectTypeOf(valid2).toExtend<Result>();\n  });\n});\n"],"names":["describe","expectTypeOf","it","result","toEqualTypeOf","valid1","valid2","toExtend"],"mappings":"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"}
|
|
@@ -1,11 +1,15 @@
|
|
|
1
1
|
import type { Knex } from "knex";
|
|
2
|
+
import type { DatabaseForeignKeys, DatabaseSchemaExtend, EntityIndex } from "../types/types";
|
|
3
|
+
import type { ForeignKeyColumns, TableName } from "./puri.types";
|
|
4
|
+
/**
|
|
5
|
+
* FK 타입 추론을 위해 DatabaseForeignKeys export
|
|
6
|
+
* (module augmentation 자동 로드 보장)
|
|
7
|
+
*/
|
|
8
|
+
export type { DatabaseForeignKeys };
|
|
2
9
|
type TableData = {
|
|
3
10
|
references: Set<string>;
|
|
4
11
|
rows: Record<string, unknown>[];
|
|
5
|
-
uniqueIndexes:
|
|
6
|
-
name?: string;
|
|
7
|
-
columns: string[];
|
|
8
|
-
}[];
|
|
12
|
+
uniqueIndexes: EntityIndex[];
|
|
9
13
|
uniquesMap: Map<string, string>;
|
|
10
14
|
};
|
|
11
15
|
export type UBRef = {
|
|
@@ -13,9 +17,12 @@ export type UBRef = {
|
|
|
13
17
|
of: string;
|
|
14
18
|
use?: string;
|
|
15
19
|
};
|
|
16
|
-
type UpsertOptions = {
|
|
20
|
+
export type UpsertOptions<TTable extends TableName<DatabaseSchemaExtend>> = {
|
|
21
|
+
chunkSize?: number;
|
|
22
|
+
cleanOrphans?: ForeignKeyColumns<TTable> | ForeignKeyColumns<TTable>[];
|
|
23
|
+
};
|
|
24
|
+
export type InsertOnlyOptions = {
|
|
17
25
|
chunkSize?: number;
|
|
18
|
-
cleanOrphans?: string | string[];
|
|
19
26
|
};
|
|
20
27
|
export declare function isRefField(field: unknown): field is UBRef;
|
|
21
28
|
export declare class UpsertBuilder {
|
|
@@ -26,9 +33,9 @@ export declare class UpsertBuilder {
|
|
|
26
33
|
register<T extends string>(tableName: string, row: {
|
|
27
34
|
[key in T]?: UBRef | string | number | boolean | bigint | null | object | unknown;
|
|
28
35
|
}): UBRef;
|
|
29
|
-
upsert(wdb: Knex, tableName:
|
|
30
|
-
insertOnly(wdb: Knex, tableName:
|
|
31
|
-
upsertOrInsert(wdb: Knex, tableName:
|
|
36
|
+
upsert<TTable extends TableName<DatabaseSchemaExtend>>(wdb: Knex, tableName: TTable, options?: UpsertOptions<TTable>): Promise<number[]>;
|
|
37
|
+
insertOnly<TTable extends TableName<DatabaseSchemaExtend>>(wdb: Knex, tableName: TTable, options?: InsertOnlyOptions): Promise<number[]>;
|
|
38
|
+
upsertOrInsert<TTable extends TableName<DatabaseSchemaExtend>>(wdb: Knex, tableName: TTable, mode: "upsert" | "insert", options?: UpsertOptions<TTable>): Promise<number[]>;
|
|
32
39
|
updateBatch(wdb: Knex, tableName: string, options?: {
|
|
33
40
|
chunkSize?: number;
|
|
34
41
|
where?: string | string[];
|
|
@@ -40,5 +47,4 @@ export declare class UpsertBuilder {
|
|
|
40
47
|
*/
|
|
41
48
|
private buildInsertLevels;
|
|
42
49
|
}
|
|
43
|
-
export {};
|
|
44
50
|
//# sourceMappingURL=upsert-builder.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"upsert-builder.d.ts","sourceRoot":"","sources":["../../src/database/upsert-builder.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;
|
|
1
|
+
{"version":3,"file":"upsert-builder.d.ts","sourceRoot":"","sources":["../../src/database/upsert-builder.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAIjC,OAAO,KAAK,EAAE,mBAAmB,EAAE,oBAAoB,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAG7F,OAAO,KAAK,EAAE,iBAAiB,EAAE,SAAS,EAAE,MAAM,cAAc,CAAC;AAEjE;;;GAGG;AACH,YAAY,EAAE,mBAAmB,EAAE,CAAC;AAGpC,KAAK,SAAS,GAAG;IACf,UAAU,EAAE,GAAG,CAAC,MAAM,CAAC,CAAC;IACxB,IAAI,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAAE,CAAC;IAChC,aAAa,EAAE,WAAW,EAAE,CAAC;IAC7B,UAAU,EAAE,GAAG,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;CACjC,CAAC;AAGF,MAAM,MAAM,KAAK,GAAG;IAClB,IAAI,EAAE,MAAM,CAAC;IACb,EAAE,EAAE,MAAM,CAAC;IACX,GAAG,CAAC,EAAE,MAAM,CAAC;CACd,CAAC;AAGF,MAAM,MAAM,aAAa,CAAC,MAAM,SAAS,SAAS,CAAC,oBAAoB,CAAC,IAAI;IAC1E,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,YAAY,CAAC,EAAE,iBAAiB,CAAC,MAAM,CAAC,GAAG,iBAAiB,CAAC,MAAM,CAAC,EAAE,CAAC;CACxE,CAAC;AAGF,MAAM,MAAM,iBAAiB,GAAG;IAC9B,SAAS,CAAC,EAAE,MAAM,CAAC;CACpB,CAAC;AAEF,wBAAgB,UAAU,CAAC,KAAK,EAAE,OAAO,GAAG,KAAK,IAAI,KAAK,CAOzD;AAED,qBAAa,aAAa;IACxB,MAAM,EAAE,GAAG,CAAC,MAAM,EAAE,SAAS,CAAC,CAAC;;IAK/B,QAAQ,CAAC,SAAS,EAAE,MAAM,GAAG,SAAS;IAwBtC,QAAQ,CAAC,SAAS,EAAE,MAAM,GAAG,OAAO;IAIpC,QAAQ,CAAC,CAAC,SAAS,MAAM,EACvB,SAAS,EAAE,MAAM,EACjB,GAAG,EAAE;SACF,GAAG,IAAI,CAAC,CAAC,CAAC,EAAE,KAAK,GAAG,MAAM,GAAG,MAAM,GAAG,OAAO,GAAG,MAAM,GAAG,IAAI,GAAG,MAAM,GAAG,OAAO;KAClF,GACA,KAAK;IAqFF,MAAM,CAAC,MAAM,SAAS,SAAS,CAAC,oBAAoB,CAAC,EACzD,GAAG,EAAE,IAAI,EACT,SAAS,EAAE,MAAM,EACjB,OAAO,CAAC,EAAE,aAAa,CAAC,MAAM,CAAC,GAC9B,OAAO,CAAC,MAAM,EAAE,CAAC;IAId,UAAU,CAAC,MAAM,SAAS,SAAS,CAAC,oBAAoB,CAAC,EAC7D,GAAG,EAAE,IAAI,EACT,SAAS,EAAE,MAAM,EACjB,OAAO,CAAC,EAAE,iBAAiB,GAC1B,OAAO,CAAC,MAAM,EAAE,CAAC;IAId,cAAc,CAAC,MAAM,SAAS,SAAS,CAAC,oBAAoB,CAAC,EACjE,GAAG,EAAE,IAAI,EACT,SAAS,EAAE,MAAM,EACjB,IAAI,EAAE,QAAQ,GAAG,QAAQ,EACzB,OAAO,CAAC,EAAE,aAAa,CAAC,MAAM,CAAC,GAC9B,OAAO,CAAC,MAAM,EAAE,CAAC;IAwMd,WAAW,CACf,GAAG,EAAE,IAAI,EACT,SAAS,EAAE,MAAM,EACjB,OAAO,CAAC,EAAE;QACR,SAAS,CAAC,EAAE,MAAM,CAAC;QACnB,KAAK,CAAC,EAAE,MAAM,GAAG,MAAM,EAAE,CAAC;KAC3B,GACA,OAAO,CAAC,IAAI,CAAC;IAyChB;;;;OAIG;IACH,OAAO,CAAC,iBAAiB;CA8D1B"}
|
|
@@ -41,11 +41,11 @@ export class UpsertBuilder {
|
|
|
41
41
|
// 해당 테이블의 unique 인덱스를 순회하며 키 생성
|
|
42
42
|
const uniqueKeys = table.uniqueIndexes.map((unqIndex)=>{
|
|
43
43
|
const uniqueKeyArray = unqIndex.columns.map((unqCol)=>{
|
|
44
|
-
const val = row[unqCol];
|
|
44
|
+
const val = row[unqCol.name];
|
|
45
45
|
if (isRefField(val)) {
|
|
46
46
|
return val.uuid;
|
|
47
47
|
} else {
|
|
48
|
-
return row[unqCol] ?? randomUUID(); // nullable인 경우 uuid로 랜덤값 삽입
|
|
48
|
+
return row[unqCol.name] ?? randomUUID(); // nullable인 경우 uuid로 랜덤값 삽입
|
|
49
49
|
}
|
|
50
50
|
});
|
|
51
51
|
// 값이 모두 null인 경우 키 생성 패스
|
|
@@ -118,17 +118,10 @@ export class UpsertBuilder {
|
|
|
118
118
|
});
|
|
119
119
|
return result;
|
|
120
120
|
}
|
|
121
|
-
async upsert(wdb, tableName,
|
|
122
|
-
// 숫자면 { chunkSize: n } 으로 변환
|
|
123
|
-
const options = typeof optionsOrChunkSize === "number" ? {
|
|
124
|
-
chunkSize: optionsOrChunkSize
|
|
125
|
-
} : optionsOrChunkSize;
|
|
121
|
+
async upsert(wdb, tableName, options) {
|
|
126
122
|
return this.upsertOrInsert(wdb, tableName, "upsert", options);
|
|
127
123
|
}
|
|
128
|
-
async insertOnly(wdb, tableName,
|
|
129
|
-
const options = typeof optionsOrChunkSize === "number" ? {
|
|
130
|
-
chunkSize: optionsOrChunkSize
|
|
131
|
-
} : optionsOrChunkSize;
|
|
124
|
+
async insertOnly(wdb, tableName, options) {
|
|
132
125
|
return this.upsertOrInsert(wdb, tableName, "insert", options);
|
|
133
126
|
}
|
|
134
127
|
async upsertOrInsert(wdb, tableName, mode, options) {
|
|
@@ -162,10 +155,6 @@ export class UpsertBuilder {
|
|
|
162
155
|
if (hasCircular) {
|
|
163
156
|
throw new Error(`${tableName}에 순환 자기 참조가 있습니다.`);
|
|
164
157
|
}
|
|
165
|
-
// upsert 모드일 때 유니크 인덱스가 없으면 에러
|
|
166
|
-
if (mode === "upsert" && table.uniqueIndexes.length === 0) {
|
|
167
|
-
throw new Error(`${tableName}에 unique index가 정의되지 않아 upsert를 할 수 없습니다.`);
|
|
168
|
-
}
|
|
169
158
|
const uuidMap = new Map();
|
|
170
159
|
const allIds = [];
|
|
171
160
|
// 레벨별로 순차 처리
|
|
@@ -213,8 +202,10 @@ export class UpsertBuilder {
|
|
|
213
202
|
// INSERT 모드 - RETURNING 사용
|
|
214
203
|
resultRows = await wdb.insert(dataForDb).into(tableName).returning(selectFields);
|
|
215
204
|
} else {
|
|
216
|
-
// UPSERT 모드 - onConflict 사용
|
|
217
|
-
const conflictColumns = table.uniqueIndexes[0].columns
|
|
205
|
+
// UPSERT 모드 - onConflict 사용 (unique index 없으면 PK fallback)
|
|
206
|
+
const conflictColumns = table.uniqueIndexes.length > 0 ? table.uniqueIndexes[0].columns.map((c)=>c.name) : [
|
|
207
|
+
"id"
|
|
208
|
+
];
|
|
218
209
|
const updateColumns = Object.keys(dataForDb[0]).filter((col)=>!conflictColumns.includes(col));
|
|
219
210
|
// updateColumns가 비어있어도 merge()를 사용하여 모든 행이 RETURNING되도록 보장
|
|
220
211
|
const mergeColumns = updateColumns.length > 0 ? updateColumns : conflictColumns;
|
|
@@ -335,7 +326,7 @@ export class UpsertBuilder {
|
|
|
335
326
|
table.uniquesMap.clear();
|
|
336
327
|
}
|
|
337
328
|
// ============================================================================
|
|
338
|
-
// Private
|
|
329
|
+
// Private Helper Methods
|
|
339
330
|
// ============================================================================
|
|
340
331
|
/**
|
|
341
332
|
* rows를 의존성 순서에 따라 레벨별로 그룹화
|
|
@@ -399,4 +390,4 @@ export class UpsertBuilder {
|
|
|
399
390
|
}
|
|
400
391
|
}
|
|
401
392
|
|
|
402
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/database/upsert-builder.ts"],"sourcesContent":["import { randomUUID } from \"crypto\";\nimport type { Knex } from \"knex\";\nimport { isArray, unique } from \"radashi\";\nimport { EntityManager } from \"../entity/entity-manager\";\nimport { Naite } from \"../naite/naite\";\nimport { assertDefined, chunk, nonNullable } from \"../utils/utils\";\nimport { batchUpdate, type RowWithId } from \"./_batch_update\";\n\ntype TableData = {\n  references: Set<string>;\n  rows: Record<string, unknown>[];\n  uniqueIndexes: { name?: string; columns: string[] }[];\n  uniquesMap: Map<string, string>;\n};\nexport type UBRef = {\n  uuid: string;\n  of: string;\n  use?: string;\n};\ntype UpsertOptions = {\n  chunkSize?: number;\n  cleanOrphans?: string | string[]; // FK 컬럼명(들)\n};\nexport function isRefField(field: unknown): field is UBRef {\n  return (\n    field !== undefined &&\n    field !== null &&\n    (field as UBRef)?.of !== undefined &&\n    (field as UBRef)?.uuid !== undefined\n  );\n}\n\nexport class UpsertBuilder {\n  tables: Map<string, TableData>;\n  constructor() {\n    this.tables = new Map();\n  }\n\n  getTable(tableName: string): TableData {\n    const table = this.tables.get(tableName);\n    if (table) {\n      return table;\n    }\n\n    const tableSpec = (() => {\n      try {\n        return EntityManager.getTableSpec(tableName);\n      } catch {\n        return null;\n      }\n    })();\n\n    const tableData = {\n      references: new Set<string>(),\n      rows: [],\n      uniqueIndexes: tableSpec?.uniqueIndexes ?? [],\n      uniquesMap: new Map<string, string>(),\n    };\n    this.tables.set(tableName, tableData);\n    return tableData;\n  }\n\n  hasTable(tableName: string): boolean {\n    return this.tables.has(tableName);\n  }\n\n  register<T extends string>(\n    tableName: string,\n    row: {\n      [key in T]?: UBRef | string | number | boolean | bigint | null | object | unknown;\n    },\n  ): UBRef {\n    const table = this.getTable(tableName);\n\n    // 해당 테이블의 unique 인덱스를 순회하며 키 생성\n    const uniqueKeys = table.uniqueIndexes\n      .map((unqIndex) => {\n        const uniqueKeyArray = unqIndex.columns.map((unqCol) => {\n          const val = row[unqCol as keyof typeof row];\n          if (isRefField(val)) {\n            return val.uuid;\n          } else {\n            return row[unqCol as keyof typeof row] ?? randomUUID(); // nullable인 경우 uuid로 랜덤값 삽입\n          }\n        });\n\n        // 값이 모두 null인 경우 키 생성 패스\n        if (uniqueKeyArray.length === 0) {\n          return null;\n        }\n        return uniqueKeyArray.join(\"---delimiter--\");\n      })\n      .filter(nonNullable);\n\n    // uuid 생성 로직\n    const { uuid, isReused } = (() => {\n      // 키를 순회하여 이미 존재하는 키가 있는지 확인\n      if (uniqueKeys.length > 0) {\n        for (const uniqueKey of uniqueKeys) {\n          if (table.uniquesMap.has(uniqueKey)) {\n            return {\n              uuid: assertDefined(table.uniquesMap.get(uniqueKey), \"Unique key not found\"),\n              isReused: true,\n            };\n          }\n        }\n      }\n\n      // 찾을 수 없는 경우 생성\n      return { uuid: randomUUID(), isReused: false };\n    })();\n\n    // 모든 유니크키에 대해 유니크맵에 uuid 저장\n    if (uniqueKeys.length > 0) {\n      for (const uniqueKey of uniqueKeys) {\n        table.uniquesMap.set(uniqueKey, uuid);\n      }\n    }\n\n    // 이 테이블에 사용된 RefField를 순회하여, 현재 테이블 정보에 어떤 필드를 참조하는지 추가\n    // 이 정보를 나중에 치환할 때 사용\n    row = Object.fromEntries(\n      Object.entries(row).map(([rowKey, rowValue]) => {\n        if (isRefField(rowValue)) {\n          rowValue.use ??= \"id\";\n          table.references.add(`${rowValue.of}.${rowValue.use}`);\n          return [rowKey, rowValue];\n        } else if (typeof rowValue === \"object\" && !(rowValue instanceof Date)) {\n          // object인 경우 JSON으로 변환\n          return [rowKey, rowValue === null ? null : JSON.stringify(rowValue)];\n        } else {\n          return [rowKey, rowValue];\n        }\n      }),\n    ) as { [key in T]?: unknown };\n\n    table.rows.push({\n      uuid,\n      ...row,\n    });\n\n    const result: UBRef = {\n      of: tableName,\n      uuid: (row as { uuid?: string }).uuid ?? uuid,\n    };\n\n    Naite.t(\"puri:ub-register\", {\n      tableName,\n      uuid: result.uuid,\n      isUuidReused: isReused,\n      row,\n    });\n\n    return result;\n  }\n\n  async upsert(\n    wdb: Knex,\n    tableName: string,\n    optionsOrChunkSize?: UpsertOptions,\n  ): Promise<number[]> {\n    // 숫자면 { chunkSize: n } 으로 변환\n    const options =\n      typeof optionsOrChunkSize === \"number\"\n        ? { chunkSize: optionsOrChunkSize }\n        : optionsOrChunkSize;\n\n    return this.upsertOrInsert(wdb, tableName, \"upsert\", options);\n  }\n  async insertOnly(\n    wdb: Knex,\n    tableName: string,\n    optionsOrChunkSize?: UpsertOptions | number,\n  ): Promise<number[]> {\n    const options =\n      typeof optionsOrChunkSize === \"number\"\n        ? { chunkSize: optionsOrChunkSize }\n        : optionsOrChunkSize;\n\n    return this.upsertOrInsert(wdb, tableName, \"insert\", options);\n  }\n\n  async upsertOrInsert(\n    wdb: Knex,\n    tableName: string,\n    mode: \"upsert\" | \"insert\",\n    options?: UpsertOptions,\n  ): Promise<number[]> {\n    if (this.hasTable(tableName) === false) {\n      return [];\n    }\n\n    const table = this.tables.get(tableName);\n    if (table === undefined) {\n      throw new Error(`존재하지 않는 테이블 ${tableName}에 upsert 요청`);\n    } else if (table.rows.length === 0) {\n      throw new Error(`${tableName}에 upsert 할 데이터가 없습니다.`);\n    }\n\n    if (\n      table.rows.some((row) =>\n        Object.entries(row).some(([, value]) => isRefField(value) && value.of !== tableName),\n      )\n    ) {\n      throw new Error(`${tableName} 해결되지 않은 참조가 있습니다.`);\n    }\n\n    // 전체 테이블 순회하여 현재 테이블 참조하는 모든 테이블 추출\n    const { references, refTables } = Array.from(this.tables).reduce(\n      (r, [, table]) => {\n        const reference = Array.from(table.references.values()).find((ref) =>\n          ref.includes(`${tableName}.`),\n        );\n        if (reference) {\n          r.references.push(reference);\n          r.refTables.push(table);\n        }\n\n        return r;\n      },\n      {\n        references: [] as string[],\n        refTables: [] as TableData[],\n      },\n    );\n    const extractFields = unique(references)\n      .map((reference) => reference.split(\".\")[1])\n      .filter((field): field is string => field !== undefined);\n\n    // 의존성 순서에 따라 레벨별 그룹화 (자기 참조가 없으면 Level 0 하나)\n    const { levels, hasCircular } = this.buildInsertLevels(table.rows, tableName);\n\n    if (hasCircular) {\n      throw new Error(`${tableName}에 순환 자기 참조가 있습니다.`);\n    }\n\n    // upsert 모드일 때 유니크 인덱스가 없으면 에러\n    if (mode === \"upsert\" && table.uniqueIndexes.length === 0) {\n      throw new Error(`${tableName}에 unique index가 정의되지 않아 upsert를 할 수 없습니다.`);\n    }\n\n    const uuidMap = new Map<string, unknown>();\n    const allIds: number[] = [];\n\n    // 레벨별로 순차 처리\n    for (const levelRows of levels) {\n      // 이전 레벨에서 얻은 ID로 자기 참조 해결\n      const resolvedRows = levelRows.map((row) => {\n        const resolved = { ...row };\n        for (const [key, value] of Object.entries(row)) {\n          if (isRefField(value) && value.of === tableName) {\n            const parent = uuidMap.get(value.uuid);\n\n            if (!parent) throw new Error(`존재하지 않는 uuid ${value.uuid} -- in ${tableName}`);\n\n            resolved[key] = (parent as Record<string, unknown>)[value.use ?? \"id\"];\n\n            Naite.t(\"puri:ub-ref-resolved\", {\n              tableName,\n              field: key,\n              from: { of: value.of, uuid: value.uuid, use: value.use ?? \"id\" },\n              to: resolved[key],\n            });\n          }\n        }\n        return resolved;\n      });\n\n      // 현재 레벨 upsert\n      const chunkSize = options?.chunkSize;\n      const levelChunks = chunkSize ? chunk(resolvedRows, chunkSize) : [resolvedRows];\n      const selectFields = unique([\"id\", ...extractFields]);\n\n      for (const dataChunk of levelChunks) {\n        if (dataChunk.length === 0) continue;\n\n        // uuid를 별도로 보관하고, DB에 저장할 데이터에서 제거\n        const originalUuids = dataChunk.map((r) => r.uuid as string);\n        const dataForDb = dataChunk.map(({ uuid, ...rest }) => rest);\n\n        let resultRows: { id: number; [key: string]: unknown }[];\n\n        if (mode === \"insert\") {\n          // INSERT 모드 - RETURNING 사용\n          resultRows = await wdb.insert(dataForDb).into(tableName).returning(selectFields);\n        } else {\n          // UPSERT 모드 - onConflict 사용\n          const conflictColumns = table.uniqueIndexes[0].columns;\n          const updateColumns = Object.keys(dataForDb[0]).filter(\n            (col) => !conflictColumns.includes(col),\n          );\n\n          // updateColumns가 비어있어도 merge()를 사용하여 모든 행이 RETURNING되도록 보장\n          const mergeColumns = updateColumns.length > 0 ? updateColumns : conflictColumns;\n\n          resultRows = await wdb\n            .insert(dataForDb)\n            .into(tableName)\n            .onConflict(conflictColumns)\n            .merge(mergeColumns)\n            .returning(selectFields);\n        }\n\n        if (originalUuids.length !== resultRows.length) {\n          throw new Error(`${tableName}: register/returning 불일치`);\n        }\n\n        for (let i = 0; i < resultRows.length; i++) {\n          uuidMap.set(originalUuids[i], resultRows[i]);\n          allIds.push(resultRows[i].id);\n        }\n      }\n    }\n\n    // 해당 테이블 참조를 실제 밸류로 변경\n    for (const table of refTables) {\n      table.rows = table.rows.map((row) => {\n        for (const key of Object.keys(row)) {\n          const prop = row[key];\n          if (isRefField(prop) && prop.of === tableName) {\n            const parent = uuidMap.get(prop.uuid);\n            if (!parent) {\n              console.error(prop);\n              throw new Error(`존재하지 않는 uuid ${prop.uuid} -- in ${tableName}`);\n            }\n            const resolvedValue = (parent as Record<string, unknown>)[prop.use ?? \"id\"];\n            row[key] = resolvedValue;\n\n            Naite.t(\"puri:ub-ref-resolved\", {\n              tableName,\n              field: key,\n              from: { of: prop.of, uuid: prop.uuid, use: prop.use ?? \"id\" },\n              to: resolvedValue,\n            });\n          }\n        }\n        return row;\n      });\n    }\n\n    if (options?.cleanOrphans) {\n      const cleanOrphans = options.cleanOrphans;\n      const fkColumns = isArray(cleanOrphans) ? cleanOrphans : [cleanOrphans];\n\n      // 현재 register된 레코드들의 FK 값들 추출\n      const fkConditions = fkColumns.map((fkCol) => {\n        const fkValues = [...new Set(table.rows.map((row) => row[fkCol]).filter((v) => v != null))];\n        return { column: fkCol, values: fkValues };\n      });\n\n      // 모든 FK 컬럼에 값이 있는 경우에만 삭제 실행\n      if (fkConditions.every((fc) => fc.values.length > 0)) {\n        let deleteQuery = wdb(tableName);\n\n        // 각 FK 컬럼에 대한 WHERE IN 조건 추가\n        for (const { column, values } of fkConditions) {\n          deleteQuery = deleteQuery.whereIn(column, values);\n        }\n\n        // 방금 upsert한 ID는 제외\n        deleteQuery = deleteQuery.whereNotIn(\"id\", allIds);\n\n        const deletedCount = await deleteQuery.delete();\n\n        Naite.t(\"puri:ub-clean-orphans\", {\n          tableName,\n          cleanOrphans: fkColumns,\n          deletedCount,\n        });\n      }\n    }\n\n    // 해당 테이블의 데이터 초기화\n    table.rows = [];\n    table.references.clear();\n    table.uniquesMap.clear();\n\n    Naite.t(\"puri:ub-upserted\", {\n      tableName,\n      mode,\n      rowCount: allIds.length,\n      returnedIds: allIds,\n    });\n\n    return allIds;\n  }\n\n  async updateBatch(\n    wdb: Knex,\n    tableName: string,\n    options?: {\n      chunkSize?: number;\n      where?: string | string[];\n    },\n  ): Promise<void> {\n    options = {\n      ...options,\n      chunkSize: options?.chunkSize ?? 500,\n      where: options?.where ?? \"id\",\n    };\n\n    if (this.hasTable(tableName) === false) {\n      return;\n    }\n    const table = this.tables.get(tableName);\n    if (!table) {\n      throw new Error(`등록되지 않은 테이블 ${tableName}에 updateBatch 요청`);\n    } else if (table.rows.length === 0) {\n      return;\n    }\n\n    const whereColumns = Array.isArray(options.where) ? options.where : [options.where ?? \"id\"];\n    const rows = table.rows.map((_row) => {\n      const { uuid: _, ...row } = _row; // uuid 제외\n      return row as RowWithId<string>;\n    });\n\n    await batchUpdate(wdb, tableName, whereColumns, rows, options.chunkSize);\n\n    Naite.t(\"puri:ub-batch-updated\", {\n      tableName,\n      rowCount: rows.length,\n      whereColumns,\n    });\n\n    // updateBatch 완료 후 처리된 데이터 제거\n    table.rows = [];\n    table.references.clear();\n    table.uniquesMap.clear();\n  }\n\n  // ============================================================================\n  // Private Helpers\n  // ============================================================================\n\n  /**\n   * rows를 의존성 순서에 따라 레벨별로 그룹화\n   * - 자기 참조 없는 경우 : 모든 rows가 Level 0\n   * - 자기 참조 있는 경우 : 자기 참조 관계를 위상 정렬하여 레벨별로 그룹화\n   */\n  private buildInsertLevels(\n    rows: Record<string, unknown>[],\n    tableName: string,\n  ): { levels: Record<string, unknown>[][]; hasCircular: boolean } {\n    // 1. 자기 참조가 없으면 한 레벨로 처리\n    const hasSelfRef = rows\n      .flatMap((row) => Object.values(row))\n      .some((value) => isRefField(value) && value.of === tableName);\n    if (!hasSelfRef) return { levels: [rows], hasCircular: false };\n\n    // 2. uuid → row 매핑 (중복 uuid 방지)\n    const rowByUuid = new Map<string, Record<string, unknown>>();\n    for (const row of rows) {\n      const uuid = row.uuid as string | undefined;\n      if (!uuid) throw new Error(`buildInsertLevels: uuid가 없는 row -- in ${tableName}`);\n      rowByUuid.set(uuid, row);\n    }\n\n    let pending = Array.from(rowByUuid.values());\n    const levels: Record<string, unknown>[][] = [];\n    const inserted = new Set<string>();\n\n    // 3. 레벨별 분류\n    while (pending.length > 0) {\n      const currentLevel: Record<string, unknown>[] = [];\n      const nextPending: Record<string, unknown>[] = [];\n\n      for (const row of pending) {\n        // 이 row가 참조하는 자기 참조들\n        const selfRefs = Object.values(row).filter(\n          (value) => isRefField(value) && value.of === tableName,\n        ) as UBRef[];\n\n        // 참조하는 모든 uuid가 이미 inserted에 있어야 이번 레벨에 포함\n        const canInsert = selfRefs.every((ref) => {\n          if (!rowByUuid.has(ref.uuid)) {\n            throw new Error(`존재하지 않는 uuid ${ref.uuid} -- in ${tableName}`);\n          }\n          return inserted.has(ref.uuid);\n        });\n\n        if (canInsert) {\n          currentLevel.push(row);\n        } else {\n          nextPending.push(row);\n        }\n      }\n\n      // 순환 참조 감지\n      if (currentLevel.length === 0) return { levels: [], hasCircular: true };\n\n      // 레벨 확정 + inserted 갱신\n      levels.push(currentLevel);\n      for (const row of currentLevel) {\n        inserted.add(row.uuid as string);\n      }\n\n      pending = nextPending;\n    }\n\n    return { levels, hasCircular: false };\n  }\n}\n"],"names":["randomUUID","isArray","unique","EntityManager","Naite","assertDefined","chunk","nonNullable","batchUpdate","isRefField","field","undefined","of","uuid","UpsertBuilder","tables","Map","getTable","tableName","table","get","tableSpec","getTableSpec","tableData","references","Set","rows","uniqueIndexes","uniquesMap","set","hasTable","has","register","row","uniqueKeys","map","unqIndex","uniqueKeyArray","columns","unqCol","val","length","join","filter","isReused","uniqueKey","Object","fromEntries","entries","rowKey","rowValue","use","add","Date","JSON","stringify","push","result","t","isUuidReused","upsert","wdb","optionsOrChunkSize","options","chunkSize","upsertOrInsert","insertOnly","mode","Error","some","value","refTables","Array","from","reduce","r","reference","values","find","ref","includes","extractFields","split","levels","hasCircular","buildInsertLevels","uuidMap","allIds","levelRows","resolvedRows","resolved","key","parent","to","levelChunks","selectFields","dataChunk","originalUuids","dataForDb","rest","resultRows","insert","into","returning","conflictColumns","updateColumns","keys","col","mergeColumns","onConflict","merge","i","id","prop","console","error","resolvedValue","cleanOrphans","fkColumns","fkConditions","fkCol","fkValues","v","column","every","fc","deleteQuery","whereIn","whereNotIn","deletedCount","delete","clear","rowCount","returnedIds","updateBatch","where","whereColumns","_row","_","hasSelfRef","flatMap","rowByUuid","pending","inserted","currentLevel","nextPending","selfRefs","canInsert"],"mappings":"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"}
|
|
393
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/database/upsert-builder.ts"],"sourcesContent":["import { randomUUID } from \"crypto\";\nimport type { Knex } from \"knex\";\nimport { isArray, unique } from \"radashi\";\nimport { EntityManager } from \"../entity/entity-manager\";\nimport { Naite } from \"../naite/naite\";\nimport type { DatabaseForeignKeys, DatabaseSchemaExtend, EntityIndex } from \"../types/types\";\nimport { assertDefined, chunk, nonNullable } from \"../utils/utils\";\nimport { batchUpdate, type RowWithId } from \"./_batch_update\";\nimport type { ForeignKeyColumns, TableName } from \"./puri.types\";\n\n/**\n * FK 타입 추론을 위해 DatabaseForeignKeys export\n * (module augmentation 자동 로드 보장)\n */\nexport type { DatabaseForeignKeys };\n\n// 테이블 데이터 타입\ntype TableData = {\n  references: Set<string>;\n  rows: Record<string, unknown>[];\n  uniqueIndexes: EntityIndex[];\n  uniquesMap: Map<string, string>;\n};\n\n// 참조 필드 타입\nexport type UBRef = {\n  uuid: string;\n  of: string;\n  use?: string;\n};\n\n// upsert 옵션\nexport type UpsertOptions<TTable extends TableName<DatabaseSchemaExtend>> = {\n  chunkSize?: number;\n  cleanOrphans?: ForeignKeyColumns<TTable> | ForeignKeyColumns<TTable>[];\n};\n\n// insertOnly 옵션\nexport type InsertOnlyOptions = {\n  chunkSize?: number;\n};\n\nexport function isRefField(field: unknown): field is UBRef {\n  return (\n    field !== undefined &&\n    field !== null &&\n    (field as UBRef)?.of !== undefined &&\n    (field as UBRef)?.uuid !== undefined\n  );\n}\n\nexport class UpsertBuilder {\n  tables: Map<string, TableData>;\n  constructor() {\n    this.tables = new Map();\n  }\n\n  getTable(tableName: string): TableData {\n    const table = this.tables.get(tableName);\n    if (table) {\n      return table;\n    }\n\n    const tableSpec = (() => {\n      try {\n        return EntityManager.getTableSpec(tableName);\n      } catch {\n        return null;\n      }\n    })();\n\n    const tableData = {\n      references: new Set<string>(),\n      rows: [],\n      uniqueIndexes: tableSpec?.uniqueIndexes ?? [],\n      uniquesMap: new Map<string, string>(),\n    };\n    this.tables.set(tableName, tableData);\n    return tableData;\n  }\n\n  hasTable(tableName: string): boolean {\n    return this.tables.has(tableName);\n  }\n\n  register<T extends string>(\n    tableName: string,\n    row: {\n      [key in T]?: UBRef | string | number | boolean | bigint | null | object | unknown;\n    },\n  ): UBRef {\n    const table = this.getTable(tableName);\n\n    // 해당 테이블의 unique 인덱스를 순회하며 키 생성\n    const uniqueKeys = table.uniqueIndexes\n      .map((unqIndex) => {\n        const uniqueKeyArray = unqIndex.columns.map((unqCol) => {\n          const val = row[unqCol.name as keyof typeof row];\n          if (isRefField(val)) {\n            return val.uuid;\n          } else {\n            return row[unqCol.name as keyof typeof row] ?? randomUUID(); // nullable인 경우 uuid로 랜덤값 삽입\n          }\n        });\n\n        // 값이 모두 null인 경우 키 생성 패스\n        if (uniqueKeyArray.length === 0) {\n          return null;\n        }\n        return uniqueKeyArray.join(\"---delimiter--\");\n      })\n      .filter(nonNullable);\n\n    // uuid 생성 로직\n    const { uuid, isReused } = (() => {\n      // 키를 순회하여 이미 존재하는 키가 있는지 확인\n      if (uniqueKeys.length > 0) {\n        for (const uniqueKey of uniqueKeys) {\n          if (table.uniquesMap.has(uniqueKey)) {\n            return {\n              uuid: assertDefined(table.uniquesMap.get(uniqueKey), \"Unique key not found\"),\n              isReused: true,\n            };\n          }\n        }\n      }\n\n      // 찾을 수 없는 경우 생성\n      return { uuid: randomUUID(), isReused: false };\n    })();\n\n    // 모든 유니크키에 대해 유니크맵에 uuid 저장\n    if (uniqueKeys.length > 0) {\n      for (const uniqueKey of uniqueKeys) {\n        table.uniquesMap.set(uniqueKey, uuid);\n      }\n    }\n\n    // 이 테이블에 사용된 RefField를 순회하여, 현재 테이블 정보에 어떤 필드를 참조하는지 추가\n    // 이 정보를 나중에 치환할 때 사용\n    row = Object.fromEntries(\n      Object.entries(row).map(([rowKey, rowValue]) => {\n        if (isRefField(rowValue)) {\n          rowValue.use ??= \"id\";\n          table.references.add(`${rowValue.of}.${rowValue.use}`);\n          return [rowKey, rowValue];\n        } else if (typeof rowValue === \"object\" && !(rowValue instanceof Date)) {\n          // object인 경우 JSON으로 변환\n          return [rowKey, rowValue === null ? null : JSON.stringify(rowValue)];\n        } else {\n          return [rowKey, rowValue];\n        }\n      }),\n    ) as { [key in T]?: unknown };\n\n    table.rows.push({\n      uuid,\n      ...row,\n    });\n\n    const result: UBRef = {\n      of: tableName,\n      uuid: (row as { uuid?: string }).uuid ?? uuid,\n    };\n\n    Naite.t(\"puri:ub-register\", {\n      tableName,\n      uuid: result.uuid,\n      isUuidReused: isReused,\n      row,\n    });\n\n    return result;\n  }\n\n  async upsert<TTable extends TableName<DatabaseSchemaExtend>>(\n    wdb: Knex,\n    tableName: TTable,\n    options?: UpsertOptions<TTable>,\n  ): Promise<number[]> {\n    return this.upsertOrInsert(wdb, tableName, \"upsert\", options);\n  }\n\n  async insertOnly<TTable extends TableName<DatabaseSchemaExtend>>(\n    wdb: Knex,\n    tableName: TTable,\n    options?: InsertOnlyOptions,\n  ): Promise<number[]> {\n    return this.upsertOrInsert(wdb, tableName, \"insert\", options);\n  }\n\n  async upsertOrInsert<TTable extends TableName<DatabaseSchemaExtend>>(\n    wdb: Knex,\n    tableName: TTable,\n    mode: \"upsert\" | \"insert\",\n    options?: UpsertOptions<TTable>,\n  ): Promise<number[]> {\n    if (this.hasTable(tableName) === false) {\n      return [];\n    }\n\n    const table = this.tables.get(tableName);\n    if (table === undefined) {\n      throw new Error(`존재하지 않는 테이블 ${tableName}에 upsert 요청`);\n    } else if (table.rows.length === 0) {\n      throw new Error(`${tableName}에 upsert 할 데이터가 없습니다.`);\n    }\n\n    if (\n      table.rows.some((row) =>\n        Object.entries(row).some(([, value]) => isRefField(value) && value.of !== tableName),\n      )\n    ) {\n      throw new Error(`${tableName} 해결되지 않은 참조가 있습니다.`);\n    }\n\n    // 전체 테이블 순회하여 현재 테이블 참조하는 모든 테이블 추출\n    const { references, refTables } = Array.from(this.tables).reduce(\n      (r, [, table]) => {\n        const reference = Array.from(table.references.values()).find((ref) =>\n          ref.includes(`${tableName}.`),\n        );\n        if (reference) {\n          r.references.push(reference);\n          r.refTables.push(table);\n        }\n\n        return r;\n      },\n      {\n        references: [] as string[],\n        refTables: [] as TableData[],\n      },\n    );\n    const extractFields = unique(references)\n      .map((reference) => reference.split(\".\")[1])\n      .filter((field): field is string => field !== undefined);\n\n    // 의존성 순서에 따라 레벨별 그룹화 (자기 참조가 없으면 Level 0 하나)\n    const { levels, hasCircular } = this.buildInsertLevels(table.rows, tableName);\n\n    if (hasCircular) {\n      throw new Error(`${tableName}에 순환 자기 참조가 있습니다.`);\n    }\n\n    const uuidMap = new Map<string, unknown>();\n    const allIds: number[] = [];\n\n    // 레벨별로 순차 처리\n    for (const levelRows of levels) {\n      // 이전 레벨에서 얻은 ID로 자기 참조 해결\n      const resolvedRows = levelRows.map((row) => {\n        const resolved = { ...row };\n        for (const [key, value] of Object.entries(row)) {\n          if (isRefField(value) && value.of === tableName) {\n            const parent = uuidMap.get(value.uuid);\n\n            if (!parent) throw new Error(`존재하지 않는 uuid ${value.uuid} -- in ${tableName}`);\n\n            resolved[key] = (parent as Record<string, unknown>)[value.use ?? \"id\"];\n\n            Naite.t(\"puri:ub-ref-resolved\", {\n              tableName,\n              field: key,\n              from: { of: value.of, uuid: value.uuid, use: value.use ?? \"id\" },\n              to: resolved[key],\n            });\n          }\n        }\n        return resolved;\n      });\n\n      // 현재 레벨 upsert\n      const chunkSize = options?.chunkSize;\n      const levelChunks = chunkSize ? chunk(resolvedRows, chunkSize) : [resolvedRows];\n      const selectFields = unique([\"id\", ...extractFields]);\n\n      for (const dataChunk of levelChunks) {\n        if (dataChunk.length === 0) continue;\n\n        // uuid를 별도로 보관하고, DB에 저장할 데이터에서 제거\n        const originalUuids = dataChunk.map((r) => r.uuid as string);\n        const dataForDb = dataChunk.map(({ uuid, ...rest }) => rest);\n\n        let resultRows: { id: number; [key: string]: unknown }[];\n\n        if (mode === \"insert\") {\n          // INSERT 모드 - RETURNING 사용\n          resultRows = await wdb.insert(dataForDb).into(tableName).returning(selectFields);\n        } else {\n          // UPSERT 모드 - onConflict 사용 (unique index 없으면 PK fallback)\n          const conflictColumns =\n            table.uniqueIndexes.length > 0\n              ? table.uniqueIndexes[0].columns.map((c) => c.name)\n              : [\"id\"];\n          const updateColumns = Object.keys(dataForDb[0]).filter(\n            (col) => !conflictColumns.includes(col),\n          );\n\n          // updateColumns가 비어있어도 merge()를 사용하여 모든 행이 RETURNING되도록 보장\n          const mergeColumns = updateColumns.length > 0 ? updateColumns : conflictColumns;\n\n          resultRows = await wdb\n            .insert(dataForDb)\n            .into(tableName)\n            .onConflict(conflictColumns)\n            .merge(mergeColumns)\n            .returning(selectFields);\n        }\n\n        if (originalUuids.length !== resultRows.length) {\n          throw new Error(`${tableName}: register/returning 불일치`);\n        }\n\n        for (let i = 0; i < resultRows.length; i++) {\n          uuidMap.set(originalUuids[i], resultRows[i]);\n          allIds.push(resultRows[i].id);\n        }\n      }\n    }\n\n    // 해당 테이블 참조를 실제 밸류로 변경\n    for (const table of refTables) {\n      table.rows = table.rows.map((row) => {\n        for (const key of Object.keys(row)) {\n          const prop = row[key];\n          if (isRefField(prop) && prop.of === tableName) {\n            const parent = uuidMap.get(prop.uuid);\n            if (!parent) {\n              console.error(prop);\n              throw new Error(`존재하지 않는 uuid ${prop.uuid} -- in ${tableName}`);\n            }\n            const resolvedValue = (parent as Record<string, unknown>)[prop.use ?? \"id\"];\n            row[key] = resolvedValue;\n\n            Naite.t(\"puri:ub-ref-resolved\", {\n              tableName,\n              field: key,\n              from: { of: prop.of, uuid: prop.uuid, use: prop.use ?? \"id\" },\n              to: resolvedValue,\n            });\n          }\n        }\n        return row;\n      });\n    }\n\n    if (options?.cleanOrphans) {\n      const cleanOrphans = options.cleanOrphans;\n      const fkColumns = isArray(cleanOrphans)\n        ? (cleanOrphans as ForeignKeyColumns<TTable>[])\n        : [cleanOrphans as ForeignKeyColumns<TTable>];\n\n      // 현재 register된 레코드들의 FK 값들 추출\n      const fkConditions = fkColumns.map((fkCol) => {\n        const fkValues = [...new Set(table.rows.map((row) => row[fkCol]).filter((v) => v != null))];\n        return { column: fkCol, values: fkValues };\n      });\n\n      // 모든 FK 컬럼에 값이 있는 경우에만 삭제 실행\n      if (fkConditions.every((fc) => fc.values.length > 0)) {\n        let deleteQuery = wdb(tableName);\n\n        // 각 FK 컬럼에 대한 WHERE IN 조건 추가\n        for (const { column, values } of fkConditions) {\n          deleteQuery = deleteQuery.whereIn(column, values);\n        }\n\n        // 방금 upsert한 ID는 제외\n        deleteQuery = deleteQuery.whereNotIn(\"id\", allIds);\n\n        const deletedCount = await deleteQuery.delete();\n\n        Naite.t(\"puri:ub-clean-orphans\", {\n          tableName,\n          cleanOrphans: fkColumns,\n          deletedCount,\n        });\n      }\n    }\n\n    // 해당 테이블의 데이터 초기화\n    table.rows = [];\n    table.references.clear();\n    table.uniquesMap.clear();\n\n    Naite.t(\"puri:ub-upserted\", {\n      tableName,\n      mode,\n      rowCount: allIds.length,\n      returnedIds: allIds,\n    });\n\n    return allIds;\n  }\n\n  async updateBatch(\n    wdb: Knex,\n    tableName: string,\n    options?: {\n      chunkSize?: number;\n      where?: string | string[];\n    },\n  ): Promise<void> {\n    options = {\n      ...options,\n      chunkSize: options?.chunkSize ?? 500,\n      where: options?.where ?? \"id\",\n    };\n\n    if (this.hasTable(tableName) === false) {\n      return;\n    }\n    const table = this.tables.get(tableName);\n    if (!table) {\n      throw new Error(`등록되지 않은 테이블 ${tableName}에 updateBatch 요청`);\n    } else if (table.rows.length === 0) {\n      return;\n    }\n\n    const whereColumns = Array.isArray(options.where) ? options.where : [options.where ?? \"id\"];\n    const rows = table.rows.map((_row) => {\n      const { uuid: _, ...row } = _row; // uuid 제외\n      return row as RowWithId<string>;\n    });\n\n    await batchUpdate(wdb, tableName, whereColumns, rows, options.chunkSize);\n\n    Naite.t(\"puri:ub-batch-updated\", {\n      tableName,\n      rowCount: rows.length,\n      whereColumns,\n    });\n\n    // updateBatch 완료 후 처리된 데이터 제거\n    table.rows = [];\n    table.references.clear();\n    table.uniquesMap.clear();\n  }\n\n  // ============================================================================\n  // Private Helper Methods\n  // ============================================================================\n\n  /**\n   * rows를 의존성 순서에 따라 레벨별로 그룹화\n   * - 자기 참조 없는 경우 : 모든 rows가 Level 0\n   * - 자기 참조 있는 경우 : 자기 참조 관계를 위상 정렬하여 레벨별로 그룹화\n   */\n  private buildInsertLevels(\n    rows: Record<string, unknown>[],\n    tableName: string,\n  ): { levels: Record<string, unknown>[][]; hasCircular: boolean } {\n    // 1. 자기 참조가 없으면 한 레벨로 처리\n    const hasSelfRef = rows\n      .flatMap((row) => Object.values(row))\n      .some((value) => isRefField(value) && value.of === tableName);\n    if (!hasSelfRef) return { levels: [rows], hasCircular: false };\n\n    // 2. uuid → row 매핑 (중복 uuid 방지)\n    const rowByUuid = new Map<string, Record<string, unknown>>();\n    for (const row of rows) {\n      const uuid = row.uuid as string | undefined;\n      if (!uuid) throw new Error(`buildInsertLevels: uuid가 없는 row -- in ${tableName}`);\n      rowByUuid.set(uuid, row);\n    }\n\n    let pending = Array.from(rowByUuid.values());\n    const levels: Record<string, unknown>[][] = [];\n    const inserted = new Set<string>();\n\n    // 3. 레벨별 분류\n    while (pending.length > 0) {\n      const currentLevel: Record<string, unknown>[] = [];\n      const nextPending: Record<string, unknown>[] = [];\n\n      for (const row of pending) {\n        // 이 row가 참조하는 자기 참조들\n        const selfRefs = Object.values(row).filter(\n          (value) => isRefField(value) && value.of === tableName,\n        ) as UBRef[];\n\n        // 참조하는 모든 uuid가 이미 inserted에 있어야 이번 레벨에 포함\n        const canInsert = selfRefs.every((ref) => {\n          if (!rowByUuid.has(ref.uuid)) {\n            throw new Error(`존재하지 않는 uuid ${ref.uuid} -- in ${tableName}`);\n          }\n          return inserted.has(ref.uuid);\n        });\n\n        if (canInsert) {\n          currentLevel.push(row);\n        } else {\n          nextPending.push(row);\n        }\n      }\n\n      // 순환 참조 감지\n      if (currentLevel.length === 0) return { levels: [], hasCircular: true };\n\n      // 레벨 확정 + inserted 갱신\n      levels.push(currentLevel);\n      for (const row of currentLevel) {\n        inserted.add(row.uuid as string);\n      }\n\n      pending = nextPending;\n    }\n\n    return { levels, hasCircular: false };\n  }\n}\n"],"names":["randomUUID","isArray","unique","EntityManager","Naite","assertDefined","chunk","nonNullable","batchUpdate","isRefField","field","undefined","of","uuid","UpsertBuilder","tables","Map","getTable","tableName","table","get","tableSpec","getTableSpec","tableData","references","Set","rows","uniqueIndexes","uniquesMap","set","hasTable","has","register","row","uniqueKeys","map","unqIndex","uniqueKeyArray","columns","unqCol","val","name","length","join","filter","isReused","uniqueKey","Object","fromEntries","entries","rowKey","rowValue","use","add","Date","JSON","stringify","push","result","t","isUuidReused","upsert","wdb","options","upsertOrInsert","insertOnly","mode","Error","some","value","refTables","Array","from","reduce","r","reference","values","find","ref","includes","extractFields","split","levels","hasCircular","buildInsertLevels","uuidMap","allIds","levelRows","resolvedRows","resolved","key","parent","to","chunkSize","levelChunks","selectFields","dataChunk","originalUuids","dataForDb","rest","resultRows","insert","into","returning","conflictColumns","c","updateColumns","keys","col","mergeColumns","onConflict","merge","i","id","prop","console","error","resolvedValue","cleanOrphans","fkColumns","fkConditions","fkCol","fkValues","v","column","every","fc","deleteQuery","whereIn","whereNotIn","deletedCount","delete","clear","rowCount","returnedIds","updateBatch","where","whereColumns","_row","_","hasSelfRef","flatMap","rowByUuid","pending","inserted","currentLevel","nextPending","selfRefs","canInsert"],"mappings":"AAAA,SAASA,UAAU,QAAQ,SAAS;AAEpC,SAASC,OAAO,EAAEC,MAAM,QAAQ,UAAU;AAC1C,SAASC,aAAa,QAAQ,8BAA2B;AACzD,SAASC,KAAK,QAAQ,oBAAiB;AAEvC,SAASC,aAAa,EAAEC,KAAK,EAAEC,WAAW,QAAQ,oBAAiB;AACnE,SAASC,WAAW,QAAwB,qBAAkB;AAmC9D,OAAO,SAASC,WAAWC,KAAc;IACvC,OACEA,UAAUC,aACVD,UAAU,QACV,AAACA,OAAiBE,OAAOD,aACzB,AAACD,OAAiBG,SAASF;AAE/B;AAEA,OAAO,MAAMG;IACXC,OAA+B;IAC/B,aAAc;QACZ,IAAI,CAACA,MAAM,GAAG,IAAIC;IACpB;IAEAC,SAASC,SAAiB,EAAa;QACrC,MAAMC,QAAQ,IAAI,CAACJ,MAAM,CAACK,GAAG,CAACF;QAC9B,IAAIC,OAAO;YACT,OAAOA;QACT;QAEA,MAAME,YAAY,AAAC,CAAA;YACjB,IAAI;gBACF,OAAOlB,cAAcmB,YAAY,CAACJ;YACpC,EAAE,OAAM;gBACN,OAAO;YACT;QACF,CAAA;QAEA,MAAMK,YAAY;YAChBC,YAAY,IAAIC;YAChBC,MAAM,EAAE;YACRC,eAAeN,WAAWM,iBAAiB,EAAE;YAC7CC,YAAY,IAAIZ;QAClB;QACA,IAAI,CAACD,MAAM,CAACc,GAAG,CAACX,WAAWK;QAC3B,OAAOA;IACT;IAEAO,SAASZ,SAAiB,EAAW;QACnC,OAAO,IAAI,CAACH,MAAM,CAACgB,GAAG,CAACb;IACzB;IAEAc,SACEd,SAAiB,EACjBe,GAEC,EACM;QACP,MAAMd,QAAQ,IAAI,CAACF,QAAQ,CAACC;QAE5B,gCAAgC;QAChC,MAAMgB,aAAaf,MAAMQ,aAAa,CACnCQ,GAAG,CAAC,CAACC;YACJ,MAAMC,iBAAiBD,SAASE,OAAO,CAACH,GAAG,CAAC,CAACI;gBAC3C,MAAMC,MAAMP,GAAG,CAACM,OAAOE,IAAI,CAAqB;gBAChD,IAAIhC,WAAW+B,MAAM;oBACnB,OAAOA,IAAI3B,IAAI;gBACjB,OAAO;oBACL,OAAOoB,GAAG,CAACM,OAAOE,IAAI,CAAqB,IAAIzC,cAAc,4BAA4B;gBAC3F;YACF;YAEA,yBAAyB;YACzB,IAAIqC,eAAeK,MAAM,KAAK,GAAG;gBAC/B,OAAO;YACT;YACA,OAAOL,eAAeM,IAAI,CAAC;QAC7B,GACCC,MAAM,CAACrC;QAEV,aAAa;QACb,MAAM,EAAEM,IAAI,EAAEgC,QAAQ,EAAE,GAAG,AAAC,CAAA;YAC1B,4BAA4B;YAC5B,IAAIX,WAAWQ,MAAM,GAAG,GAAG;gBACzB,KAAK,MAAMI,aAAaZ,WAAY;oBAClC,IAAIf,MAAMS,UAAU,CAACG,GAAG,CAACe,YAAY;wBACnC,OAAO;4BACLjC,MAAMR,cAAcc,MAAMS,UAAU,CAACR,GAAG,CAAC0B,YAAY;4BACrDD,UAAU;wBACZ;oBACF;gBACF;YACF;YAEA,gBAAgB;YAChB,OAAO;gBAAEhC,MAAMb;gBAAc6C,UAAU;YAAM;QAC/C,CAAA;QAEA,4BAA4B;QAC5B,IAAIX,WAAWQ,MAAM,GAAG,GAAG;YACzB,KAAK,MAAMI,aAAaZ,WAAY;gBAClCf,MAAMS,UAAU,CAACC,GAAG,CAACiB,WAAWjC;YAClC;QACF;QAEA,wDAAwD;QACxD,qBAAqB;QACrBoB,MAAMc,OAAOC,WAAW,CACtBD,OAAOE,OAAO,CAAChB,KAAKE,GAAG,CAAC,CAAC,CAACe,QAAQC,SAAS;YACzC,IAAI1C,WAAW0C,WAAW;gBACxBA,SAASC,GAAG,KAAK;gBACjBjC,MAAMK,UAAU,CAAC6B,GAAG,CAAC,GAAGF,SAASvC,EAAE,CAAC,CAAC,EAAEuC,SAASC,GAAG,EAAE;gBACrD,OAAO;oBAACF;oBAAQC;iBAAS;YAC3B,OAAO,IAAI,OAAOA,aAAa,YAAY,CAAEA,CAAAA,oBAAoBG,IAAG,GAAI;gBACtE,uBAAuB;gBACvB,OAAO;oBAACJ;oBAAQC,aAAa,OAAO,OAAOI,KAAKC,SAAS,CAACL;iBAAU;YACtE,OAAO;gBACL,OAAO;oBAACD;oBAAQC;iBAAS;YAC3B;QACF;QAGFhC,MAAMO,IAAI,CAAC+B,IAAI,CAAC;YACd5C;YACA,GAAGoB,GAAG;QACR;QAEA,MAAMyB,SAAgB;YACpB9C,IAAIM;YACJL,MAAM,AAACoB,IAA0BpB,IAAI,IAAIA;QAC3C;QAEAT,MAAMuD,CAAC,CAAC,oBAAoB;YAC1BzC;YACAL,MAAM6C,OAAO7C,IAAI;YACjB+C,cAAcf;YACdZ;QACF;QAEA,OAAOyB;IACT;IAEA,MAAMG,OACJC,GAAS,EACT5C,SAAiB,EACjB6C,OAA+B,EACZ;QACnB,OAAO,IAAI,CAACC,cAAc,CAACF,KAAK5C,WAAW,UAAU6C;IACvD;IAEA,MAAME,WACJH,GAAS,EACT5C,SAAiB,EACjB6C,OAA2B,EACR;QACnB,OAAO,IAAI,CAACC,cAAc,CAACF,KAAK5C,WAAW,UAAU6C;IACvD;IAEA,MAAMC,eACJF,GAAS,EACT5C,SAAiB,EACjBgD,IAAyB,EACzBH,OAA+B,EACZ;QACnB,IAAI,IAAI,CAACjC,QAAQ,CAACZ,eAAe,OAAO;YACtC,OAAO,EAAE;QACX;QAEA,MAAMC,QAAQ,IAAI,CAACJ,MAAM,CAACK,GAAG,CAACF;QAC9B,IAAIC,UAAUR,WAAW;YACvB,MAAM,IAAIwD,MAAM,CAAC,YAAY,EAAEjD,UAAU,WAAW,CAAC;QACvD,OAAO,IAAIC,MAAMO,IAAI,CAACgB,MAAM,KAAK,GAAG;YAClC,MAAM,IAAIyB,MAAM,GAAGjD,UAAU,qBAAqB,CAAC;QACrD;QAEA,IACEC,MAAMO,IAAI,CAAC0C,IAAI,CAAC,CAACnC,MACfc,OAAOE,OAAO,CAAChB,KAAKmC,IAAI,CAAC,CAAC,GAAGC,MAAM,GAAK5D,WAAW4D,UAAUA,MAAMzD,EAAE,KAAKM,aAE5E;YACA,MAAM,IAAIiD,MAAM,GAAGjD,UAAU,kBAAkB,CAAC;QAClD;QAEA,oCAAoC;QACpC,MAAM,EAAEM,UAAU,EAAE8C,SAAS,EAAE,GAAGC,MAAMC,IAAI,CAAC,IAAI,CAACzD,MAAM,EAAE0D,MAAM,CAC9D,CAACC,GAAG,GAAGvD,MAAM;YACX,MAAMwD,YAAYJ,MAAMC,IAAI,CAACrD,MAAMK,UAAU,CAACoD,MAAM,IAAIC,IAAI,CAAC,CAACC,MAC5DA,IAAIC,QAAQ,CAAC,GAAG7D,UAAU,CAAC,CAAC;YAE9B,IAAIyD,WAAW;gBACbD,EAAElD,UAAU,CAACiC,IAAI,CAACkB;gBAClBD,EAAEJ,SAAS,CAACb,IAAI,CAACtC;YACnB;YAEA,OAAOuD;QACT,GACA;YACElD,YAAY,EAAE;YACd8C,WAAW,EAAE;QACf;QAEF,MAAMU,gBAAgB9E,OAAOsB,YAC1BW,GAAG,CAAC,CAACwC,YAAcA,UAAUM,KAAK,CAAC,IAAI,CAAC,EAAE,EAC1CrC,MAAM,CAAC,CAAClC,QAA2BA,UAAUC;QAEhD,6CAA6C;QAC7C,MAAM,EAAEuE,MAAM,EAAEC,WAAW,EAAE,GAAG,IAAI,CAACC,iBAAiB,CAACjE,MAAMO,IAAI,EAAER;QAEnE,IAAIiE,aAAa;YACf,MAAM,IAAIhB,MAAM,GAAGjD,UAAU,iBAAiB,CAAC;QACjD;QAEA,MAAMmE,UAAU,IAAIrE;QACpB,MAAMsE,SAAmB,EAAE;QAE3B,aAAa;QACb,KAAK,MAAMC,aAAaL,OAAQ;YAC9B,0BAA0B;YAC1B,MAAMM,eAAeD,UAAUpD,GAAG,CAAC,CAACF;gBAClC,MAAMwD,WAAW;oBAAE,GAAGxD,GAAG;gBAAC;gBAC1B,KAAK,MAAM,CAACyD,KAAKrB,MAAM,IAAItB,OAAOE,OAAO,CAAChB,KAAM;oBAC9C,IAAIxB,WAAW4D,UAAUA,MAAMzD,EAAE,KAAKM,WAAW;wBAC/C,MAAMyE,SAASN,QAAQjE,GAAG,CAACiD,MAAMxD,IAAI;wBAErC,IAAI,CAAC8E,QAAQ,MAAM,IAAIxB,MAAM,CAAC,aAAa,EAAEE,MAAMxD,IAAI,CAAC,OAAO,EAAEK,WAAW;wBAE5EuE,QAAQ,CAACC,IAAI,GAAG,AAACC,MAAkC,CAACtB,MAAMjB,GAAG,IAAI,KAAK;wBAEtEhD,MAAMuD,CAAC,CAAC,wBAAwB;4BAC9BzC;4BACAR,OAAOgF;4BACPlB,MAAM;gCAAE5D,IAAIyD,MAAMzD,EAAE;gCAAEC,MAAMwD,MAAMxD,IAAI;gCAAEuC,KAAKiB,MAAMjB,GAAG,IAAI;4BAAK;4BAC/DwC,IAAIH,QAAQ,CAACC,IAAI;wBACnB;oBACF;gBACF;gBACA,OAAOD;YACT;YAEA,eAAe;YACf,MAAMI,YAAY9B,SAAS8B;YAC3B,MAAMC,cAAcD,YAAYvF,MAAMkF,cAAcK,aAAa;gBAACL;aAAa;YAC/E,MAAMO,eAAe7F,OAAO;gBAAC;mBAAS8E;aAAc;YAEpD,KAAK,MAAMgB,aAAaF,YAAa;gBACnC,IAAIE,UAAUtD,MAAM,KAAK,GAAG;gBAE5B,mCAAmC;gBACnC,MAAMuD,gBAAgBD,UAAU7D,GAAG,CAAC,CAACuC,IAAMA,EAAE7D,IAAI;gBACjD,MAAMqF,YAAYF,UAAU7D,GAAG,CAAC,CAAC,EAAEtB,IAAI,EAAE,GAAGsF,MAAM,GAAKA;gBAEvD,IAAIC;gBAEJ,IAAIlC,SAAS,UAAU;oBACrB,2BAA2B;oBAC3BkC,aAAa,MAAMtC,IAAIuC,MAAM,CAACH,WAAWI,IAAI,CAACpF,WAAWqF,SAAS,CAACR;gBACrE,OAAO;oBACL,2DAA2D;oBAC3D,MAAMS,kBACJrF,MAAMQ,aAAa,CAACe,MAAM,GAAG,IACzBvB,MAAMQ,aAAa,CAAC,EAAE,CAACW,OAAO,CAACH,GAAG,CAAC,CAACsE,IAAMA,EAAEhE,IAAI,IAChD;wBAAC;qBAAK;oBACZ,MAAMiE,gBAAgB3D,OAAO4D,IAAI,CAACT,SAAS,CAAC,EAAE,EAAEtD,MAAM,CACpD,CAACgE,MAAQ,CAACJ,gBAAgBzB,QAAQ,CAAC6B;oBAGrC,2DAA2D;oBAC3D,MAAMC,eAAeH,cAAchE,MAAM,GAAG,IAAIgE,gBAAgBF;oBAEhEJ,aAAa,MAAMtC,IAChBuC,MAAM,CAACH,WACPI,IAAI,CAACpF,WACL4F,UAAU,CAACN,iBACXO,KAAK,CAACF,cACNN,SAAS,CAACR;gBACf;gBAEA,IAAIE,cAAcvD,MAAM,KAAK0D,WAAW1D,MAAM,EAAE;oBAC9C,MAAM,IAAIyB,MAAM,GAAGjD,UAAU,wBAAwB,CAAC;gBACxD;gBAEA,IAAK,IAAI8F,IAAI,GAAGA,IAAIZ,WAAW1D,MAAM,EAAEsE,IAAK;oBAC1C3B,QAAQxD,GAAG,CAACoE,aAAa,CAACe,EAAE,EAAEZ,UAAU,CAACY,EAAE;oBAC3C1B,OAAO7B,IAAI,CAAC2C,UAAU,CAACY,EAAE,CAACC,EAAE;gBAC9B;YACF;QACF;QAEA,uBAAuB;QACvB,KAAK,MAAM9F,SAASmD,UAAW;YAC7BnD,MAAMO,IAAI,GAAGP,MAAMO,IAAI,CAACS,GAAG,CAAC,CAACF;gBAC3B,KAAK,MAAMyD,OAAO3C,OAAO4D,IAAI,CAAC1E,KAAM;oBAClC,MAAMiF,OAAOjF,GAAG,CAACyD,IAAI;oBACrB,IAAIjF,WAAWyG,SAASA,KAAKtG,EAAE,KAAKM,WAAW;wBAC7C,MAAMyE,SAASN,QAAQjE,GAAG,CAAC8F,KAAKrG,IAAI;wBACpC,IAAI,CAAC8E,QAAQ;4BACXwB,QAAQC,KAAK,CAACF;4BACd,MAAM,IAAI/C,MAAM,CAAC,aAAa,EAAE+C,KAAKrG,IAAI,CAAC,OAAO,EAAEK,WAAW;wBAChE;wBACA,MAAMmG,gBAAgB,AAAC1B,MAAkC,CAACuB,KAAK9D,GAAG,IAAI,KAAK;wBAC3EnB,GAAG,CAACyD,IAAI,GAAG2B;wBAEXjH,MAAMuD,CAAC,CAAC,wBAAwB;4BAC9BzC;4BACAR,OAAOgF;4BACPlB,MAAM;gCAAE5D,IAAIsG,KAAKtG,EAAE;gCAAEC,MAAMqG,KAAKrG,IAAI;gCAAEuC,KAAK8D,KAAK9D,GAAG,IAAI;4BAAK;4BAC5DwC,IAAIyB;wBACN;oBACF;gBACF;gBACA,OAAOpF;YACT;QACF;QAEA,IAAI8B,SAASuD,cAAc;YACzB,MAAMA,eAAevD,QAAQuD,YAAY;YACzC,MAAMC,YAAYtH,QAAQqH,gBACrBA,eACD;gBAACA;aAA0C;YAE/C,8BAA8B;YAC9B,MAAME,eAAeD,UAAUpF,GAAG,CAAC,CAACsF;gBAClC,MAAMC,WAAW;uBAAI,IAAIjG,IAAIN,MAAMO,IAAI,CAACS,GAAG,CAAC,CAACF,MAAQA,GAAG,CAACwF,MAAM,EAAE7E,MAAM,CAAC,CAAC+E,IAAMA,KAAK;iBAAO;gBAC3F,OAAO;oBAAEC,QAAQH;oBAAO7C,QAAQ8C;gBAAS;YAC3C;YAEA,6BAA6B;YAC7B,IAAIF,aAAaK,KAAK,CAAC,CAACC,KAAOA,GAAGlD,MAAM,CAAClC,MAAM,GAAG,IAAI;gBACpD,IAAIqF,cAAcjE,IAAI5C;gBAEtB,6BAA6B;gBAC7B,KAAK,MAAM,EAAE0G,MAAM,EAAEhD,MAAM,EAAE,IAAI4C,aAAc;oBAC7CO,cAAcA,YAAYC,OAAO,CAACJ,QAAQhD;gBAC5C;gBAEA,oBAAoB;gBACpBmD,cAAcA,YAAYE,UAAU,CAAC,MAAM3C;gBAE3C,MAAM4C,eAAe,MAAMH,YAAYI,MAAM;gBAE7C/H,MAAMuD,CAAC,CAAC,yBAAyB;oBAC/BzC;oBACAoG,cAAcC;oBACdW;gBACF;YACF;QACF;QAEA,kBAAkB;QAClB/G,MAAMO,IAAI,GAAG,EAAE;QACfP,MAAMK,UAAU,CAAC4G,KAAK;QACtBjH,MAAMS,UAAU,CAACwG,KAAK;QAEtBhI,MAAMuD,CAAC,CAAC,oBAAoB;YAC1BzC;YACAgD;YACAmE,UAAU/C,OAAO5C,MAAM;YACvB4F,aAAahD;QACf;QAEA,OAAOA;IACT;IAEA,MAAMiD,YACJzE,GAAS,EACT5C,SAAiB,EACjB6C,OAGC,EACc;QACfA,UAAU;YACR,GAAGA,OAAO;YACV8B,WAAW9B,SAAS8B,aAAa;YACjC2C,OAAOzE,SAASyE,SAAS;QAC3B;QAEA,IAAI,IAAI,CAAC1G,QAAQ,CAACZ,eAAe,OAAO;YACtC;QACF;QACA,MAAMC,QAAQ,IAAI,CAACJ,MAAM,CAACK,GAAG,CAACF;QAC9B,IAAI,CAACC,OAAO;YACV,MAAM,IAAIgD,MAAM,CAAC,YAAY,EAAEjD,UAAU,gBAAgB,CAAC;QAC5D,OAAO,IAAIC,MAAMO,IAAI,CAACgB,MAAM,KAAK,GAAG;YAClC;QACF;QAEA,MAAM+F,eAAelE,MAAMtE,OAAO,CAAC8D,QAAQyE,KAAK,IAAIzE,QAAQyE,KAAK,GAAG;YAACzE,QAAQyE,KAAK,IAAI;SAAK;QAC3F,MAAM9G,OAAOP,MAAMO,IAAI,CAACS,GAAG,CAAC,CAACuG;YAC3B,MAAM,EAAE7H,MAAM8H,CAAC,EAAE,GAAG1G,KAAK,GAAGyG,MAAM,UAAU;YAC5C,OAAOzG;QACT;QAEA,MAAMzB,YAAYsD,KAAK5C,WAAWuH,cAAc/G,MAAMqC,QAAQ8B,SAAS;QAEvEzF,MAAMuD,CAAC,CAAC,yBAAyB;YAC/BzC;YACAmH,UAAU3G,KAAKgB,MAAM;YACrB+F;QACF;QAEA,8BAA8B;QAC9BtH,MAAMO,IAAI,GAAG,EAAE;QACfP,MAAMK,UAAU,CAAC4G,KAAK;QACtBjH,MAAMS,UAAU,CAACwG,KAAK;IACxB;IAEA,+EAA+E;IAC/E,yBAAyB;IACzB,+EAA+E;IAE/E;;;;GAIC,GACD,AAAQhD,kBACN1D,IAA+B,EAC/BR,SAAiB,EAC8C;QAC/D,yBAAyB;QACzB,MAAM0H,aAAalH,KAChBmH,OAAO,CAAC,CAAC5G,MAAQc,OAAO6B,MAAM,CAAC3C,MAC/BmC,IAAI,CAAC,CAACC,QAAU5D,WAAW4D,UAAUA,MAAMzD,EAAE,KAAKM;QACrD,IAAI,CAAC0H,YAAY,OAAO;YAAE1D,QAAQ;gBAACxD;aAAK;YAAEyD,aAAa;QAAM;QAE7D,gCAAgC;QAChC,MAAM2D,YAAY,IAAI9H;QACtB,KAAK,MAAMiB,OAAOP,KAAM;YACtB,MAAMb,OAAOoB,IAAIpB,IAAI;YACrB,IAAI,CAACA,MAAM,MAAM,IAAIsD,MAAM,CAAC,sCAAsC,EAAEjD,WAAW;YAC/E4H,UAAUjH,GAAG,CAAChB,MAAMoB;QACtB;QAEA,IAAI8G,UAAUxE,MAAMC,IAAI,CAACsE,UAAUlE,MAAM;QACzC,MAAMM,SAAsC,EAAE;QAC9C,MAAM8D,WAAW,IAAIvH;QAErB,YAAY;QACZ,MAAOsH,QAAQrG,MAAM,GAAG,EAAG;YACzB,MAAMuG,eAA0C,EAAE;YAClD,MAAMC,cAAyC,EAAE;YAEjD,KAAK,MAAMjH,OAAO8G,QAAS;gBACzB,qBAAqB;gBACrB,MAAMI,WAAWpG,OAAO6B,MAAM,CAAC3C,KAAKW,MAAM,CACxC,CAACyB,QAAU5D,WAAW4D,UAAUA,MAAMzD,EAAE,KAAKM;gBAG/C,2CAA2C;gBAC3C,MAAMkI,YAAYD,SAAStB,KAAK,CAAC,CAAC/C;oBAChC,IAAI,CAACgE,UAAU/G,GAAG,CAAC+C,IAAIjE,IAAI,GAAG;wBAC5B,MAAM,IAAIsD,MAAM,CAAC,aAAa,EAAEW,IAAIjE,IAAI,CAAC,OAAO,EAAEK,WAAW;oBAC/D;oBACA,OAAO8H,SAASjH,GAAG,CAAC+C,IAAIjE,IAAI;gBAC9B;gBAEA,IAAIuI,WAAW;oBACbH,aAAaxF,IAAI,CAACxB;gBACpB,OAAO;oBACLiH,YAAYzF,IAAI,CAACxB;gBACnB;YACF;YAEA,WAAW;YACX,IAAIgH,aAAavG,MAAM,KAAK,GAAG,OAAO;gBAAEwC,QAAQ,EAAE;gBAAEC,aAAa;YAAK;YAEtE,sBAAsB;YACtBD,OAAOzB,IAAI,CAACwF;YACZ,KAAK,MAAMhH,OAAOgH,aAAc;gBAC9BD,SAAS3F,GAAG,CAACpB,IAAIpB,IAAI;YACvB;YAEAkI,UAAUG;QACZ;QAEA,OAAO;YAAEhE;YAAQC,aAAa;QAAM;IACtC;AACF"}
|