sonamu 0.8.13 → 0.8.14
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/api/sonamu.d.ts.map +1 -1
- package/dist/api/sonamu.js +2 -3
- package/dist/auth/auth-generator.d.ts +8 -0
- package/dist/auth/auth-generator.d.ts.map +1 -1
- package/dist/auth/auth-generator.js +33 -1
- package/dist/auth/better-auth-entities.d.ts.map +1 -1
- package/dist/auth/better-auth-entities.js +12 -2
- package/dist/bin/cli.js +18 -3
- package/dist/cone/cone-generator.js +10 -4
- package/dist/database/knex.d.ts.map +1 -1
- package/dist/database/knex.js +64 -2
- package/dist/database/puri.d.ts +9 -1
- package/dist/database/puri.d.ts.map +1 -1
- package/dist/database/puri.js +42 -1
- package/dist/database/puri.types.d.ts +2 -0
- package/dist/database/puri.types.d.ts.map +1 -1
- package/dist/database/puri.types.js +6 -2
- package/dist/entity/entity-manager.d.ts +149 -1
- package/dist/entity/entity-manager.d.ts.map +1 -1
- package/dist/entity/entity-manager.js +68 -4
- package/dist/migration/__tests__/code-generation.search-text.test.js +435 -0
- package/dist/migration/code-generation.d.ts.map +1 -1
- package/dist/migration/code-generation.js +696 -32
- package/dist/migration/migration-set.js +3 -1
- package/dist/migration/postgresql-schema-reader.d.ts +16 -2
- package/dist/migration/postgresql-schema-reader.d.ts.map +1 -1
- package/dist/migration/postgresql-schema-reader.js +281 -7
- package/dist/stream/sse.js +5 -3
- package/dist/template/__tests__/generated.template.search-text.test.js +99 -0
- package/dist/template/generated.template.test-d.js +24 -0
- package/dist/template/implementations/generated.template.d.ts.map +1 -1
- package/dist/template/implementations/generated.template.js +2 -2
- package/dist/template/implementations/init_types.template.d.ts.map +1 -1
- package/dist/template/implementations/init_types.template.js +11 -3
- package/dist/template/zod-converter.d.ts.map +1 -1
- package/dist/template/zod-converter.js +6 -2
- package/dist/testing/dev-test-routes.d.ts.map +1 -1
- package/dist/testing/dev-test-routes.js +5 -3
- package/dist/testing/fixture-generator.d.ts +13 -0
- package/dist/testing/fixture-generator.d.ts.map +1 -1
- package/dist/testing/fixture-generator.js +105 -8
- package/dist/testing/fixture-manager.d.ts.map +1 -1
- package/dist/testing/fixture-manager.js +19 -2
- package/dist/types/__tests__/entity-json-schema-search-text.test.js +256 -0
- package/dist/types/types.d.ts +494 -1
- package/dist/types/types.d.ts.map +1 -1
- package/dist/types/types.js +117 -13
- package/dist/ui/api.d.ts.map +1 -1
- package/dist/ui/api.js +14 -2
- package/dist/ui/cdd-service.d.ts +16 -14
- package/dist/ui/cdd-service.d.ts.map +1 -1
- package/dist/ui/cdd-service.js +145 -37
- package/dist/ui/cdd-types.d.ts +60 -0
- package/dist/ui/cdd-types.d.ts.map +1 -0
- package/dist/ui/cdd-types.js +3 -0
- package/dist/ui-web/assets/index-D4XFBV-f.css +1 -0
- package/dist/ui-web/assets/{index-CQ_S40bD.js → index-D_19-Pi4.js} +87 -87
- package/dist/ui-web/index.html +2 -2
- package/package.json +7 -3
- package/src/api/sonamu.ts +1 -2
- package/src/auth/auth-generator.ts +38 -0
- package/src/auth/better-auth-entities.ts +18 -1
- package/src/bin/cli.ts +15 -1
- package/src/cone/cone-generator.ts +9 -3
- package/src/database/knex.ts +62 -4
- package/src/database/puri.ts +71 -0
- package/src/database/puri.types.ts +2 -0
- package/src/entity/entity-manager.ts +95 -3
- package/src/migration/__tests__/code-generation.search-text.test.ts +390 -0
- package/src/migration/code-generation.ts +848 -34
- package/src/migration/migration-set.ts +2 -0
- package/src/migration/postgresql-schema-reader.ts +366 -9
- package/src/skills/sonamu/auth-migration.md +80 -0
- package/src/skills/sonamu/cdd.md +148 -28
- package/src/skills/sonamu/cone.md +16 -0
- package/src/skills/sonamu/entity-relations.md +1 -1
- package/src/skills/sonamu/fixture-cli.md +4 -0
- package/src/skills/sonamu/frontend.md +65 -0
- package/src/skills/sonamu/migration.md +3 -1
- package/src/skills/sonamu/model.md +28 -0
- package/src/skills/sonamu/workflow.md +12 -5
- package/src/stream/sse.ts +4 -2
- package/src/template/__tests__/generated.template.search-text.test.ts +89 -0
- package/src/template/generated.template.test-d.ts +46 -0
- package/src/template/implementations/generated.template.ts +4 -1
- package/src/template/implementations/init_types.template.ts +20 -5
- package/src/template/zod-converter.ts +5 -0
- package/src/testing/dev-test-routes.ts +4 -2
- package/src/testing/fixture-generator.ts +157 -9
- package/src/testing/fixture-manager.ts +15 -1
- package/src/types/__tests__/entity-json-schema-search-text.test.ts +179 -0
- package/src/types/types.ts +168 -12
- package/src/ui/api.ts +24 -1
- package/src/ui/cdd-service.ts +195 -55
- package/src/ui/cdd-types.ts +73 -0
- package/dist/ui-web/assets/index-egkMxKos.css +0 -1
|
@@ -1,3 +1,7 @@
|
|
|
1
|
-
/** biome-ignore-all lint/suspicious/noExplicitAny: Puri.types.ts는 다양한 타입을 사용하고 있습니다. */ export
|
|
1
|
+
/** biome-ignore-all lint/suspicious/noExplicitAny: Puri.types.ts는 다양한 타입을 사용하고 있습니다. */ export const FUZZY_OPERATORS = [
|
|
2
|
+
"<%",
|
|
3
|
+
"%",
|
|
4
|
+
"<<%"
|
|
5
|
+
];
|
|
2
6
|
|
|
3
|
-
//# 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 { PuriSubsetFn } from \"./puri-subset.types\";\n\n// ============================================\n// 내부 타입 키 (메타데이터)\n// ============================================\ntype FulltextKey = \"__fulltext__\";\ntype VirtualKey = \"__virtual__\";\ntype LeftJoinedKey = \"__leftJoined__\";\ntype HasDefault = \"__hasDefault__\";\ntype GeneratedKey = \"__generated__\";\ntype VectorKey = \"__vector__\";\ntype VirtualQueryKey = \"__virtual_query__\";\n\ntype InternalTypeKeys =\n  | FulltextKey\n  | VirtualKey\n  | LeftJoinedKey\n  | HasDefault\n  | GeneratedKey\n  | VectorKey\n  | VirtualQueryKey;\n\n// ============================================\n// 타입 유틸리티\n// ============================================\n\n// __vector__ 메타데이터에서 벡터 컬럼 추출\ntype VectorColumnKeys<T> = T extends { [K in VectorKey]: readonly (infer V)[] }\n  ? V & string\n  : never;\n\nexport type VectorColumns<TTables extends Record<string, any>> =\n  | {\n      [TAlias in keyof TTables]: `${TAlias & string}.${VectorColumnKeys<TTables[TAlias]>}`;\n    }[keyof TTables]\n  | (IsSingleKey<TTables> extends true ? VectorColumnKeys<TTables[keyof TTables]> : never);\n\n// 테이블명 타입\nexport type TableName<TSchema> = keyof TSchema & string;\n\n// 테이블의 id(PK) 타입을 추출\nexport type IdType<TSchema, TTable extends keyof TSchema> = TSchema[TTable] extends { id: infer I }\n  ? I\n  : number;\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\" | \"string[]\">;\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            : R extends \"string[]\"\n              ? string[]\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            : R extends \"string[]\"\n              ? string[]\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\" | \"ilike\" | \"not ilike\";\n\n// SQL Expression 타입 정의\nexport type SqlExpression<T extends \"string\" | \"number\" | \"boolean\" | \"date\" | \"string[]\"> = {\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를 위한 타입 유틸리티\nexport type ExtractTTables<T extends Puri<any, any, any>> =\n  T extends Puri<any, infer TTables, any> ? TTables : never;\nexport type UnionExtractedTTables<\n  SubsetKey extends string,\n  SubsetQueries extends Record<SubsetKey, PuriSubsetFn>,\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\n// FTS 타입\ntype TsQueryParser = \"to_tsquery\" | \"plainto_tsquery\" | \"phraseto_tsquery\" | \"websearch_to_tsquery\";\nexport type TsQueryConfig = \"simple\" | \"english\";\nexport type TsQueryOptions = {\n  parser?: TsQueryParser;\n  config?: TsQueryConfig;\n};\n\nexport type TsHighlightOptions = {\n  /** 쿼리 변환 함수 (기본값: \"websearch_to_tsquery\") */\n  parser?: TsQueryParser;\n  /** 텍스트 검색 설정 (기본값: \"simple\") */\n  config?: TsQueryConfig;\n  /** 최대 단어 수 (기본값: 35) */\n  maxWords?: number;\n  /** 최소 단어 수 (기본값: 15) */\n  minWords?: number;\n  /** 헤드라인 시작/끝에서 제거할 짧은 단어 길이 (기본값: 3) */\n  shortWord?: number;\n  /** true면 전체 문서를 헤드라인으로 사용 (기본값: false) */\n  highlightAll?: boolean;\n  /** 표시할 최대 텍스트 조각 수 (기본값: 0, 조각 미사용) */\n  maxFragments?: number;\n  /** 쿼리 단어 시작 구분자 (기본값: \"<b>\") */\n  startSel?: string;\n  /** 쿼리 단어 끝 구분자 (기본값: \"</b>\") */\n  stopSel?: string;\n  /** 조각 구분자 (기본값: \" ... \") */\n  fragmentDelimiter?: string;\n};\n\nexport type TsRankOptions = {\n  parser?: TsQueryParser;\n  config?: TsQueryConfig;\n  /** 가중치 배열 [D, C, B, A] (기본값: [0.1, 0.2, 0.4, 1.0]) */\n  weights?: [number, number, number, number];\n  /**\n   * 정규화 옵션\n   * 0: 문서 길이 무시 (기본값)\n   * 1: 1 + log(문서 길이)로 나눔\n   * 2: 문서 길이로 나눔\n   * 4: 평균 조화 거리로 나눔 (ts_rank_cd만)\n   * 8: 고유 단어 수로 나눔\n   * 16: 1 + log(고유 단어 수)로 나눔\n   * 32: rank/(rank+1) -> 0~1 사이의 값으로 스케일링\n   *\n   * 비트마스크를 사용하여 옵션을 조합할 수 있음\n   * 예: 8 | 32 -> 고유 단어 수로 나누고 0~1 스케일링\n   */\n  normalization?: number;\n};\n"],"names":[],"mappings":"AAAA,sFAAsF,GA2atF,WAmBE"}
|
|
7
|
+
//# 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 { PuriSubsetFn } from \"./puri-subset.types\";\n\n// ============================================\n// 내부 타입 키 (메타데이터)\n// ============================================\ntype FulltextKey = \"__fulltext__\";\ntype VirtualKey = \"__virtual__\";\ntype LeftJoinedKey = \"__leftJoined__\";\ntype HasDefault = \"__hasDefault__\";\ntype GeneratedKey = \"__generated__\";\ntype VectorKey = \"__vector__\";\ntype VirtualQueryKey = \"__virtual_query__\";\n\ntype InternalTypeKeys =\n  | FulltextKey\n  | VirtualKey\n  | LeftJoinedKey\n  | HasDefault\n  | GeneratedKey\n  | VectorKey\n  | VirtualQueryKey;\n\n// ============================================\n// 타입 유틸리티\n// ============================================\n\n// __vector__ 메타데이터에서 벡터 컬럼 추출\ntype VectorColumnKeys<T> = T extends { [K in VectorKey]: readonly (infer V)[] }\n  ? V & string\n  : never;\n\nexport type VectorColumns<TTables extends Record<string, any>> =\n  | {\n      [TAlias in keyof TTables]: `${TAlias & string}.${VectorColumnKeys<TTables[TAlias]>}`;\n    }[keyof TTables]\n  | (IsSingleKey<TTables> extends true ? VectorColumnKeys<TTables[keyof TTables]> : never);\n\n// 테이블명 타입\nexport type TableName<TSchema> = keyof TSchema & string;\n\n// 테이블의 id(PK) 타입을 추출\nexport type IdType<TSchema, TTable extends keyof TSchema> = TSchema[TTable] extends { id: infer I }\n  ? I\n  : number;\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\" | \"string[]\">;\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            : R extends \"string[]\"\n              ? string[]\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            : R extends \"string[]\"\n              ? string[]\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\" | \"ilike\" | \"not ilike\";\nexport const FUZZY_OPERATORS = [\"<%\", \"%\", \"<<%\"] as const;\nexport type FuzzyOperator = (typeof FUZZY_OPERATORS)[number];\n\n// SQL Expression 타입 정의\nexport type SqlExpression<T extends \"string\" | \"number\" | \"boolean\" | \"date\" | \"string[]\"> = {\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를 위한 타입 유틸리티\nexport type ExtractTTables<T extends Puri<any, any, any>> =\n  T extends Puri<any, infer TTables, any> ? TTables : never;\nexport type UnionExtractedTTables<\n  SubsetKey extends string,\n  SubsetQueries extends Record<SubsetKey, PuriSubsetFn>,\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\n// FTS 타입\ntype TsQueryParser = \"to_tsquery\" | \"plainto_tsquery\" | \"phraseto_tsquery\" | \"websearch_to_tsquery\";\nexport type TsQueryConfig = \"simple\" | \"english\";\nexport type TsQueryOptions = {\n  parser?: TsQueryParser;\n  config?: TsQueryConfig;\n};\n\nexport type TsHighlightOptions = {\n  /** 쿼리 변환 함수 (기본값: \"websearch_to_tsquery\") */\n  parser?: TsQueryParser;\n  /** 텍스트 검색 설정 (기본값: \"simple\") */\n  config?: TsQueryConfig;\n  /** 최대 단어 수 (기본값: 35) */\n  maxWords?: number;\n  /** 최소 단어 수 (기본값: 15) */\n  minWords?: number;\n  /** 헤드라인 시작/끝에서 제거할 짧은 단어 길이 (기본값: 3) */\n  shortWord?: number;\n  /** true면 전체 문서를 헤드라인으로 사용 (기본값: false) */\n  highlightAll?: boolean;\n  /** 표시할 최대 텍스트 조각 수 (기본값: 0, 조각 미사용) */\n  maxFragments?: number;\n  /** 쿼리 단어 시작 구분자 (기본값: \"<b>\") */\n  startSel?: string;\n  /** 쿼리 단어 끝 구분자 (기본값: \"</b>\") */\n  stopSel?: string;\n  /** 조각 구분자 (기본값: \" ... \") */\n  fragmentDelimiter?: string;\n};\n\nexport type TsRankOptions = {\n  parser?: TsQueryParser;\n  config?: TsQueryConfig;\n  /** 가중치 배열 [D, C, B, A] (기본값: [0.1, 0.2, 0.4, 1.0]) */\n  weights?: [number, number, number, number];\n  /**\n   * 정규화 옵션\n   * 0: 문서 길이 무시 (기본값)\n   * 1: 1 + log(문서 길이)로 나눔\n   * 2: 문서 길이로 나눔\n   * 4: 평균 조화 거리로 나눔 (ts_rank_cd만)\n   * 8: 고유 단어 수로 나눔\n   * 16: 1 + log(고유 단어 수)로 나눔\n   * 32: rank/(rank+1) -> 0~1 사이의 값으로 스케일링\n   *\n   * 비트마스크를 사용하여 옵션을 조합할 수 있음\n   * 예: 8 | 32 -> 고유 단어 수로 나누고 0~1 스케일링\n   */\n  normalization?: number;\n};\n"],"names":["FUZZY_OPERATORS"],"mappings":"AAAA,sFAAsF,GA+StF,OAAO,MAAMA,kBAAkB;IAAC;IAAM;IAAK;CAAM,CAAU"}
|
|
@@ -36,6 +36,11 @@ declare class EntityManagerClass {
|
|
|
36
36
|
fixtureGenerator?: string | undefined;
|
|
37
37
|
fixtureDefault?: unknown;
|
|
38
38
|
fixtureStrategy?: "sequence" | undefined;
|
|
39
|
+
fixtureCompanions?: {
|
|
40
|
+
entity: string;
|
|
41
|
+
overrides?: Record<string, unknown> | undefined;
|
|
42
|
+
count?: number | undefined;
|
|
43
|
+
}[] | undefined;
|
|
39
44
|
dataSource?: {
|
|
40
45
|
strategy: "file" | "sample" | "ids" | "query" | "recent" | "random";
|
|
41
46
|
config?: unknown;
|
|
@@ -61,6 +66,11 @@ declare class EntityManagerClass {
|
|
|
61
66
|
fixtureGenerator?: string | undefined;
|
|
62
67
|
fixtureDefault?: unknown;
|
|
63
68
|
fixtureStrategy?: "sequence" | undefined;
|
|
69
|
+
fixtureCompanions?: {
|
|
70
|
+
entity: string;
|
|
71
|
+
overrides?: Record<string, unknown> | undefined;
|
|
72
|
+
count?: number | undefined;
|
|
73
|
+
}[] | undefined;
|
|
64
74
|
dataSource?: {
|
|
65
75
|
strategy: "file" | "sample" | "ids" | "query" | "recent" | "random";
|
|
66
76
|
config?: unknown;
|
|
@@ -86,6 +96,11 @@ declare class EntityManagerClass {
|
|
|
86
96
|
fixtureGenerator?: string | undefined;
|
|
87
97
|
fixtureDefault?: unknown;
|
|
88
98
|
fixtureStrategy?: "sequence" | undefined;
|
|
99
|
+
fixtureCompanions?: {
|
|
100
|
+
entity: string;
|
|
101
|
+
overrides?: Record<string, unknown> | undefined;
|
|
102
|
+
count?: number | undefined;
|
|
103
|
+
}[] | undefined;
|
|
89
104
|
dataSource?: {
|
|
90
105
|
strategy: "file" | "sample" | "ids" | "query" | "recent" | "random";
|
|
91
106
|
config?: unknown;
|
|
@@ -111,6 +126,11 @@ declare class EntityManagerClass {
|
|
|
111
126
|
fixtureGenerator?: string | undefined;
|
|
112
127
|
fixtureDefault?: unknown;
|
|
113
128
|
fixtureStrategy?: "sequence" | undefined;
|
|
129
|
+
fixtureCompanions?: {
|
|
130
|
+
entity: string;
|
|
131
|
+
overrides?: Record<string, unknown> | undefined;
|
|
132
|
+
count?: number | undefined;
|
|
133
|
+
}[] | undefined;
|
|
114
134
|
dataSource?: {
|
|
115
135
|
strategy: "file" | "sample" | "ids" | "query" | "recent" | "random";
|
|
116
136
|
config?: unknown;
|
|
@@ -135,6 +155,11 @@ declare class EntityManagerClass {
|
|
|
135
155
|
fixtureGenerator?: string | undefined;
|
|
136
156
|
fixtureDefault?: unknown;
|
|
137
157
|
fixtureStrategy?: "sequence" | undefined;
|
|
158
|
+
fixtureCompanions?: {
|
|
159
|
+
entity: string;
|
|
160
|
+
overrides?: Record<string, unknown> | undefined;
|
|
161
|
+
count?: number | undefined;
|
|
162
|
+
}[] | undefined;
|
|
138
163
|
dataSource?: {
|
|
139
164
|
strategy: "file" | "sample" | "ids" | "query" | "recent" | "random";
|
|
140
165
|
config?: unknown;
|
|
@@ -161,6 +186,11 @@ declare class EntityManagerClass {
|
|
|
161
186
|
fixtureGenerator?: string | undefined;
|
|
162
187
|
fixtureDefault?: unknown;
|
|
163
188
|
fixtureStrategy?: "sequence" | undefined;
|
|
189
|
+
fixtureCompanions?: {
|
|
190
|
+
entity: string;
|
|
191
|
+
overrides?: Record<string, unknown> | undefined;
|
|
192
|
+
count?: number | undefined;
|
|
193
|
+
}[] | undefined;
|
|
164
194
|
dataSource?: {
|
|
165
195
|
strategy: "file" | "sample" | "ids" | "query" | "recent" | "random";
|
|
166
196
|
config?: unknown;
|
|
@@ -187,6 +217,11 @@ declare class EntityManagerClass {
|
|
|
187
217
|
fixtureGenerator?: string | undefined;
|
|
188
218
|
fixtureDefault?: unknown;
|
|
189
219
|
fixtureStrategy?: "sequence" | undefined;
|
|
220
|
+
fixtureCompanions?: {
|
|
221
|
+
entity: string;
|
|
222
|
+
overrides?: Record<string, unknown> | undefined;
|
|
223
|
+
count?: number | undefined;
|
|
224
|
+
}[] | undefined;
|
|
190
225
|
dataSource?: {
|
|
191
226
|
strategy: "file" | "sample" | "ids" | "query" | "recent" | "random";
|
|
192
227
|
config?: unknown;
|
|
@@ -211,6 +246,11 @@ declare class EntityManagerClass {
|
|
|
211
246
|
fixtureGenerator?: string | undefined;
|
|
212
247
|
fixtureDefault?: unknown;
|
|
213
248
|
fixtureStrategy?: "sequence" | undefined;
|
|
249
|
+
fixtureCompanions?: {
|
|
250
|
+
entity: string;
|
|
251
|
+
overrides?: Record<string, unknown> | undefined;
|
|
252
|
+
count?: number | undefined;
|
|
253
|
+
}[] | undefined;
|
|
214
254
|
dataSource?: {
|
|
215
255
|
strategy: "file" | "sample" | "ids" | "query" | "recent" | "random";
|
|
216
256
|
config?: unknown;
|
|
@@ -235,6 +275,11 @@ declare class EntityManagerClass {
|
|
|
235
275
|
fixtureGenerator?: string | undefined;
|
|
236
276
|
fixtureDefault?: unknown;
|
|
237
277
|
fixtureStrategy?: "sequence" | undefined;
|
|
278
|
+
fixtureCompanions?: {
|
|
279
|
+
entity: string;
|
|
280
|
+
overrides?: Record<string, unknown> | undefined;
|
|
281
|
+
count?: number | undefined;
|
|
282
|
+
}[] | undefined;
|
|
238
283
|
dataSource?: {
|
|
239
284
|
strategy: "file" | "sample" | "ids" | "query" | "recent" | "random";
|
|
240
285
|
config?: unknown;
|
|
@@ -260,6 +305,11 @@ declare class EntityManagerClass {
|
|
|
260
305
|
fixtureGenerator?: string | undefined;
|
|
261
306
|
fixtureDefault?: unknown;
|
|
262
307
|
fixtureStrategy?: "sequence" | undefined;
|
|
308
|
+
fixtureCompanions?: {
|
|
309
|
+
entity: string;
|
|
310
|
+
overrides?: Record<string, unknown> | undefined;
|
|
311
|
+
count?: number | undefined;
|
|
312
|
+
}[] | undefined;
|
|
263
313
|
dataSource?: {
|
|
264
314
|
strategy: "file" | "sample" | "ids" | "query" | "recent" | "random";
|
|
265
315
|
config?: unknown;
|
|
@@ -285,6 +335,11 @@ declare class EntityManagerClass {
|
|
|
285
335
|
fixtureGenerator?: string | undefined;
|
|
286
336
|
fixtureDefault?: unknown;
|
|
287
337
|
fixtureStrategy?: "sequence" | undefined;
|
|
338
|
+
fixtureCompanions?: {
|
|
339
|
+
entity: string;
|
|
340
|
+
overrides?: Record<string, unknown> | undefined;
|
|
341
|
+
count?: number | undefined;
|
|
342
|
+
}[] | undefined;
|
|
288
343
|
dataSource?: {
|
|
289
344
|
strategy: "file" | "sample" | "ids" | "query" | "recent" | "random";
|
|
290
345
|
config?: unknown;
|
|
@@ -309,6 +364,43 @@ declare class EntityManagerClass {
|
|
|
309
364
|
fixtureGenerator?: string | undefined;
|
|
310
365
|
fixtureDefault?: unknown;
|
|
311
366
|
fixtureStrategy?: "sequence" | undefined;
|
|
367
|
+
fixtureCompanions?: {
|
|
368
|
+
entity: string;
|
|
369
|
+
overrides?: Record<string, unknown> | undefined;
|
|
370
|
+
count?: number | undefined;
|
|
371
|
+
}[] | undefined;
|
|
372
|
+
dataSource?: {
|
|
373
|
+
strategy: "file" | "sample" | "ids" | "query" | "recent" | "random";
|
|
374
|
+
config?: unknown;
|
|
375
|
+
} | undefined;
|
|
376
|
+
} | undefined;
|
|
377
|
+
} | {
|
|
378
|
+
type: "searchText";
|
|
379
|
+
sourceColumns: {
|
|
380
|
+
name: string;
|
|
381
|
+
caseInsensitive?: boolean | undefined;
|
|
382
|
+
}[];
|
|
383
|
+
name: string;
|
|
384
|
+
desc?: string | undefined;
|
|
385
|
+
nullable?: boolean | undefined;
|
|
386
|
+
toFilter?: boolean | undefined;
|
|
387
|
+
dbDefault?: string | number | boolean | undefined;
|
|
388
|
+
generated?: {
|
|
389
|
+
type: "STORED" | "VIRTUAL";
|
|
390
|
+
expression: string;
|
|
391
|
+
} | undefined;
|
|
392
|
+
cone?: {
|
|
393
|
+
[x: string]: unknown;
|
|
394
|
+
note?: string | undefined;
|
|
395
|
+
tags?: string[] | undefined;
|
|
396
|
+
fixtureGenerator?: string | undefined;
|
|
397
|
+
fixtureDefault?: unknown;
|
|
398
|
+
fixtureStrategy?: "sequence" | undefined;
|
|
399
|
+
fixtureCompanions?: {
|
|
400
|
+
entity: string;
|
|
401
|
+
overrides?: Record<string, unknown> | undefined;
|
|
402
|
+
count?: number | undefined;
|
|
403
|
+
}[] | undefined;
|
|
312
404
|
dataSource?: {
|
|
313
405
|
strategy: "file" | "sample" | "ids" | "query" | "recent" | "random";
|
|
314
406
|
config?: unknown;
|
|
@@ -334,6 +426,11 @@ declare class EntityManagerClass {
|
|
|
334
426
|
fixtureGenerator?: string | undefined;
|
|
335
427
|
fixtureDefault?: unknown;
|
|
336
428
|
fixtureStrategy?: "sequence" | undefined;
|
|
429
|
+
fixtureCompanions?: {
|
|
430
|
+
entity: string;
|
|
431
|
+
overrides?: Record<string, unknown> | undefined;
|
|
432
|
+
count?: number | undefined;
|
|
433
|
+
}[] | undefined;
|
|
337
434
|
dataSource?: {
|
|
338
435
|
strategy: "file" | "sample" | "ids" | "query" | "recent" | "random";
|
|
339
436
|
config?: unknown;
|
|
@@ -358,6 +455,11 @@ declare class EntityManagerClass {
|
|
|
358
455
|
fixtureGenerator?: string | undefined;
|
|
359
456
|
fixtureDefault?: unknown;
|
|
360
457
|
fixtureStrategy?: "sequence" | undefined;
|
|
458
|
+
fixtureCompanions?: {
|
|
459
|
+
entity: string;
|
|
460
|
+
overrides?: Record<string, unknown> | undefined;
|
|
461
|
+
count?: number | undefined;
|
|
462
|
+
}[] | undefined;
|
|
361
463
|
dataSource?: {
|
|
362
464
|
strategy: "file" | "sample" | "ids" | "query" | "recent" | "random";
|
|
363
465
|
config?: unknown;
|
|
@@ -382,6 +484,11 @@ declare class EntityManagerClass {
|
|
|
382
484
|
fixtureGenerator?: string | undefined;
|
|
383
485
|
fixtureDefault?: unknown;
|
|
384
486
|
fixtureStrategy?: "sequence" | undefined;
|
|
487
|
+
fixtureCompanions?: {
|
|
488
|
+
entity: string;
|
|
489
|
+
overrides?: Record<string, unknown> | undefined;
|
|
490
|
+
count?: number | undefined;
|
|
491
|
+
}[] | undefined;
|
|
385
492
|
dataSource?: {
|
|
386
493
|
strategy: "file" | "sample" | "ids" | "query" | "recent" | "random";
|
|
387
494
|
config?: unknown;
|
|
@@ -411,6 +518,11 @@ declare class EntityManagerClass {
|
|
|
411
518
|
fixtureGenerator?: string | undefined;
|
|
412
519
|
fixtureDefault?: unknown;
|
|
413
520
|
fixtureStrategy?: "sequence" | undefined;
|
|
521
|
+
fixtureCompanions?: {
|
|
522
|
+
entity: string;
|
|
523
|
+
overrides?: Record<string, unknown> | undefined;
|
|
524
|
+
count?: number | undefined;
|
|
525
|
+
}[] | undefined;
|
|
414
526
|
dataSource?: {
|
|
415
527
|
strategy: "file" | "sample" | "ids" | "query" | "recent" | "random";
|
|
416
528
|
config?: unknown;
|
|
@@ -438,6 +550,11 @@ declare class EntityManagerClass {
|
|
|
438
550
|
fixtureGenerator?: string | undefined;
|
|
439
551
|
fixtureDefault?: unknown;
|
|
440
552
|
fixtureStrategy?: "sequence" | undefined;
|
|
553
|
+
fixtureCompanions?: {
|
|
554
|
+
entity: string;
|
|
555
|
+
overrides?: Record<string, unknown> | undefined;
|
|
556
|
+
count?: number | undefined;
|
|
557
|
+
}[] | undefined;
|
|
441
558
|
dataSource?: {
|
|
442
559
|
strategy: "file" | "sample" | "ids" | "query" | "recent" | "random";
|
|
443
560
|
config?: unknown;
|
|
@@ -466,6 +583,11 @@ declare class EntityManagerClass {
|
|
|
466
583
|
fixtureGenerator?: string | undefined;
|
|
467
584
|
fixtureDefault?: unknown;
|
|
468
585
|
fixtureStrategy?: "sequence" | undefined;
|
|
586
|
+
fixtureCompanions?: {
|
|
587
|
+
entity: string;
|
|
588
|
+
overrides?: Record<string, unknown> | undefined;
|
|
589
|
+
count?: number | undefined;
|
|
590
|
+
}[] | undefined;
|
|
469
591
|
dataSource?: {
|
|
470
592
|
strategy: "file" | "sample" | "ids" | "query" | "recent" | "random";
|
|
471
593
|
config?: unknown;
|
|
@@ -496,6 +618,11 @@ declare class EntityManagerClass {
|
|
|
496
618
|
fixtureGenerator?: string | undefined;
|
|
497
619
|
fixtureDefault?: unknown;
|
|
498
620
|
fixtureStrategy?: "sequence" | undefined;
|
|
621
|
+
fixtureCompanions?: {
|
|
622
|
+
entity: string;
|
|
623
|
+
overrides?: Record<string, unknown> | undefined;
|
|
624
|
+
count?: number | undefined;
|
|
625
|
+
}[] | undefined;
|
|
499
626
|
dataSource?: {
|
|
500
627
|
strategy: "file" | "sample" | "ids" | "query" | "recent" | "random";
|
|
501
628
|
config?: unknown;
|
|
@@ -509,6 +636,7 @@ declare class EntityManagerClass {
|
|
|
509
636
|
nullsFirst?: boolean | undefined;
|
|
510
637
|
sortOrder?: "ASC" | "DESC" | undefined;
|
|
511
638
|
vectorOps?: "vector_cosine_ops" | "vector_ip_ops" | "vector_l2_ops" | undefined;
|
|
639
|
+
opclass?: string | undefined;
|
|
512
640
|
}[];
|
|
513
641
|
name: string;
|
|
514
642
|
using?: "btree" | "hash" | "gin" | "gist" | "pgroonga" | undefined;
|
|
@@ -532,6 +660,11 @@ declare class EntityManagerClass {
|
|
|
532
660
|
fixtureGenerator?: string | undefined;
|
|
533
661
|
fixtureDefault?: unknown;
|
|
534
662
|
fixtureStrategy?: "sequence" | undefined;
|
|
663
|
+
fixtureCompanions?: {
|
|
664
|
+
entity: string;
|
|
665
|
+
overrides?: Record<string, unknown> | undefined;
|
|
666
|
+
count?: number | undefined;
|
|
667
|
+
}[] | undefined;
|
|
535
668
|
dataSource?: {
|
|
536
669
|
strategy: "file" | "sample" | "ids" | "query" | "recent" | "random";
|
|
537
670
|
config?: unknown;
|
|
@@ -547,6 +680,11 @@ declare class EntityManagerClass {
|
|
|
547
680
|
fixtureGenerator?: string | undefined;
|
|
548
681
|
fixtureDefault?: unknown;
|
|
549
682
|
fixtureStrategy?: "sequence" | undefined;
|
|
683
|
+
fixtureCompanions?: {
|
|
684
|
+
entity: string;
|
|
685
|
+
overrides?: Record<string, unknown> | undefined;
|
|
686
|
+
count?: number | undefined;
|
|
687
|
+
}[] | undefined;
|
|
550
688
|
dataSource?: {
|
|
551
689
|
strategy: "file" | "sample" | "ids" | "query" | "recent" | "random";
|
|
552
690
|
config?: unknown;
|
|
@@ -561,6 +699,11 @@ declare class EntityManagerClass {
|
|
|
561
699
|
fixtureGenerator?: string | undefined;
|
|
562
700
|
fixtureDefault?: unknown;
|
|
563
701
|
fixtureStrategy?: "sequence" | undefined;
|
|
702
|
+
fixtureCompanions?: {
|
|
703
|
+
entity: string;
|
|
704
|
+
overrides?: Record<string, unknown> | undefined;
|
|
705
|
+
count?: number | undefined;
|
|
706
|
+
}[] | undefined;
|
|
564
707
|
dataSource?: {
|
|
565
708
|
strategy: "file" | "sample" | "ids" | "query" | "recent" | "random";
|
|
566
709
|
config?: unknown;
|
|
@@ -568,7 +711,12 @@ declare class EntityManagerClass {
|
|
|
568
711
|
} | undefined;
|
|
569
712
|
}> | null;
|
|
570
713
|
reload(doSilent?: boolean): Promise<void>;
|
|
571
|
-
register(json: EntityJson
|
|
714
|
+
register(json: EntityJson, options?: {
|
|
715
|
+
deferSearchTextJsonSourceValidation?: boolean;
|
|
716
|
+
}): Promise<void>;
|
|
717
|
+
validateAllRegisteredSearchTextJsonSources(): Promise<void>;
|
|
718
|
+
private validateSearchTextJsonSources;
|
|
719
|
+
private resolveSearchTextJsonSourceType;
|
|
572
720
|
get(entityId: string): Entity;
|
|
573
721
|
getByTable(table: string): Entity;
|
|
574
722
|
exists(entityId: string): boolean;
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"entity-manager.d.ts","sourceRoot":"","sources":["../../src/entity/entity-manager.ts"],"names":[],"mappings":"AAKA,OAAO,EAAiB,CAAC,EAAE,MAAM,KAAK,CAAC;AAEvC,OAAO,
|
|
1
|
+
{"version":3,"file":"entity-manager.d.ts","sourceRoot":"","sources":["../../src/entity/entity-manager.ts"],"names":[],"mappings":"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"}
|
|
@@ -5,7 +5,7 @@ import inflection from "inflection";
|
|
|
5
5
|
import path from "path";
|
|
6
6
|
import { prettifyError, z } from "zod";
|
|
7
7
|
import { Sonamu } from "../api/sonamu.js";
|
|
8
|
-
import { EntityJsonSchema } from "../types/types.js";
|
|
8
|
+
import { EntityJsonSchema, isSearchTextJsonSourceZodType, isSearchTextProp, SonamuFileArraySchema, SonamuFileSchema } from "../types/types.js";
|
|
9
9
|
import { globAsync } from "../utils/async-utils.js";
|
|
10
10
|
import { importMembers } from "../utils/esm-utils.js";
|
|
11
11
|
import { runtimePath } from "../utils/path-utils.js";
|
|
@@ -30,9 +30,12 @@ class EntityManagerClass {
|
|
|
30
30
|
const errorMessage = prettifyError(error);
|
|
31
31
|
console.error(chalk.red(`Invalid entity.json schema: ${relativePath}\n${chalk.yellow(errorMessage)}`));
|
|
32
32
|
}
|
|
33
|
-
await this.register(json
|
|
33
|
+
await this.register(json, {
|
|
34
|
+
deferSearchTextJsonSourceValidation: true
|
|
35
|
+
});
|
|
34
36
|
}
|
|
35
37
|
await this.registerNonEntityTypeModulePaths();
|
|
38
|
+
await this.validateAllRegisteredSearchTextJsonSources();
|
|
36
39
|
this.isAutoloaded = true;
|
|
37
40
|
}
|
|
38
41
|
schemaValidate(json) {
|
|
@@ -46,12 +49,73 @@ class EntityManagerClass {
|
|
|
46
49
|
this.isAutoloaded = false;
|
|
47
50
|
return await this.autoload(doSilent);
|
|
48
51
|
}
|
|
49
|
-
async register(json) {
|
|
52
|
+
async register(json, options = {}) {
|
|
50
53
|
const entity = new Entity(json);
|
|
51
54
|
await entity.registerModulePaths();
|
|
55
|
+
if (!options.deferSearchTextJsonSourceValidation) {
|
|
56
|
+
await this.validateSearchTextJsonSources(entity);
|
|
57
|
+
}
|
|
52
58
|
entity.registerTableSpecs();
|
|
53
59
|
this.entities.set(json.id, entity);
|
|
54
60
|
}
|
|
61
|
+
async validateAllRegisteredSearchTextJsonSources() {
|
|
62
|
+
for (const entity of this.entities.values()){
|
|
63
|
+
await this.validateSearchTextJsonSources(entity);
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
async validateSearchTextJsonSources(entity) {
|
|
67
|
+
const propsByName = new Map(entity.props.map((prop)=>[
|
|
68
|
+
prop.name,
|
|
69
|
+
prop
|
|
70
|
+
]));
|
|
71
|
+
for (const prop of entity.props){
|
|
72
|
+
if (!isSearchTextProp(prop)) {
|
|
73
|
+
continue;
|
|
74
|
+
}
|
|
75
|
+
for (const source of prop.sourceColumns){
|
|
76
|
+
const sourceProp = propsByName.get(source.name);
|
|
77
|
+
if (!sourceProp || sourceProp.type !== "json") {
|
|
78
|
+
continue;
|
|
79
|
+
}
|
|
80
|
+
const zodType = await this.resolveSearchTextJsonSourceType(entity, sourceProp.id);
|
|
81
|
+
if (!zodType) {
|
|
82
|
+
throw new Error(`searchText source "${source.name}"의 json 타입 "${sourceProp.id}"을(를) 로드할 수 없습니다.`);
|
|
83
|
+
}
|
|
84
|
+
if (!isSearchTextJsonSourceZodType(zodType)) {
|
|
85
|
+
throw new Error(`searchText source "${source.name}"의 json 타입 "${sourceProp.id}"은(는) unwrap 후 z.array(z.string()) 이어야 합니다.`);
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
}
|
|
90
|
+
async resolveSearchTextJsonSourceType(entity, typeId) {
|
|
91
|
+
const localType = entity.types[typeId];
|
|
92
|
+
if (localType instanceof z.ZodType) {
|
|
93
|
+
return localType;
|
|
94
|
+
}
|
|
95
|
+
for (const registeredEntity of this.entities.values()){
|
|
96
|
+
const registeredType = registeredEntity.types[typeId];
|
|
97
|
+
if (registeredType instanceof z.ZodType) {
|
|
98
|
+
return registeredType;
|
|
99
|
+
}
|
|
100
|
+
}
|
|
101
|
+
if (typeId === "SonamuFile") {
|
|
102
|
+
return SonamuFileSchema;
|
|
103
|
+
}
|
|
104
|
+
if (typeId === "SonamuFile[]") {
|
|
105
|
+
return SonamuFileArraySchema;
|
|
106
|
+
}
|
|
107
|
+
const modulePath = this.modulePaths.get(typeId);
|
|
108
|
+
if (!modulePath) {
|
|
109
|
+
return null;
|
|
110
|
+
}
|
|
111
|
+
const moduleFilePath = path.join(Sonamu.apiRootPath, runtimePath(`dist/application/${modulePath}.js`));
|
|
112
|
+
const importedMembers = await importMembers(moduleFilePath);
|
|
113
|
+
const matched = importedMembers.find(({ name })=>name === typeId);
|
|
114
|
+
if (!matched || !(matched.value instanceof z.ZodType)) {
|
|
115
|
+
return null;
|
|
116
|
+
}
|
|
117
|
+
return matched.value;
|
|
118
|
+
}
|
|
55
119
|
get(entityId) {
|
|
56
120
|
const entity = this.entities.get(entityId);
|
|
57
121
|
if (entity === undefined) {
|
|
@@ -159,4 +223,4 @@ class EntityManagerClass {
|
|
|
159
223
|
}
|
|
160
224
|
export const EntityManager = new EntityManagerClass();
|
|
161
225
|
|
|
162
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/entity/entity-manager.ts"],"sourcesContent":["import assert from \"assert\";\nimport chalk from \"chalk\";\nimport { glob, readFile } from \"fs/promises\";\nimport inflection from \"inflection\";\nimport path from \"path\";\nimport { prettifyError, z } from \"zod\";\nimport { Sonamu } from \"../api/sonamu\";\nimport { type EntityIndex, type EntityJson, EntityJsonSchema } from \"../types/types\";\nimport { globAsync } from \"../utils/async-utils\";\nimport { importMembers } from \"../utils/esm-utils\";\nimport type { AbsolutePath } from \"../utils/path-utils\";\nimport { runtimePath } from \"../utils/path-utils\";\nimport { Entity } from \"./entity\";\n\nexport type EntityNamesRecord = Record<\n  \"fs\" | \"fsPlural\" | \"camel\" | \"camelPlural\" | \"capital\" | \"capitalPlural\" | \"upper\" | \"constant\",\n  string\n>;\nexport type TableSpec = {\n  name: string;\n  uniqueIndexes: EntityIndex[];\n  jsonColumns: string[];\n};\nclass EntityManagerClass {\n  private entities: Map<string, Entity> = new Map();\n  public modulePaths: Map<string, string> = new Map();\n  private tableSpecs: Map<string, TableSpec> = new Map();\n  public isAutoloaded: boolean = false;\n\n  // 경로 전달받아 모든 entity.json 파일 로드\n  async autoload(_: boolean = false) {\n    if (this.isAutoloaded) {\n      return;\n    }\n    const pathPattern = path.join(Sonamu.apiRootPath, \"/src/application/**/*.entity.json\");\n\n    for await (const file of glob(path.resolve(pathPattern))) {\n      const json = JSON.parse((await readFile(file)).toString());\n\n      // entity.json 스키마 검증\n      const error = this.schemaValidate(json);\n      if (error) {\n        const relativePath = path.relative(Sonamu.apiRootPath, file);\n        const errorMessage = prettifyError(error);\n        console.error(\n          chalk.red(`Invalid entity.json schema: ${relativePath}\\n${chalk.yellow(errorMessage)}`),\n        );\n      }\n\n      await this.register(json);\n    }\n\n    await this.registerNonEntityTypeModulePaths();\n\n    this.isAutoloaded = true;\n  }\n\n  schemaValidate(json: unknown) {\n    const result = EntityJsonSchema.safeParse(json);\n    return result.success ? null : result.error;\n  }\n\n  async reload(doSilent: boolean = false) {\n    this.entities.clear();\n    this.modulePaths.clear();\n    this.tableSpecs.clear();\n    this.isAutoloaded = false;\n\n    return await this.autoload(doSilent);\n  }\n\n  async register(json: EntityJson): Promise<void> {\n    const entity = new Entity(json);\n    await entity.registerModulePaths();\n    entity.registerTableSpecs();\n    this.entities.set(json.id, entity);\n  }\n\n  get(entityId: string): Entity {\n    const entity = this.entities.get(entityId);\n    if (entity === undefined) {\n      throw new Error(`존재하지 않는 Entity 요청 ${entityId}`);\n    }\n\n    return entity;\n  }\n\n  getByTable(table: string): Entity {\n    const entity = Array.from(this.entities.values()).find((entity) => entity.table === table);\n    if (entity === undefined) {\n      throw new Error(`존재하지 않는 Entity 요청 ${table}`);\n    }\n\n    return entity;\n  }\n\n  exists(entityId: string): boolean {\n    const entity = this.entities.get(entityId);\n    return entity !== undefined;\n  }\n\n  getAllIds(): string[] {\n    return Array.from(EntityManager.entities.keys()).sort();\n  }\n\n  getAllEntities(): Entity[] {\n    return Array.from(this.entities.values());\n  }\n\n  getAllParentIds(): string[] {\n    return this.getAllIds().filter((entityId) => {\n      const entity = this.get(entityId);\n      return entity.parentId === undefined;\n    });\n  }\n\n  getChildrenIds(parentId: string): string[] {\n    return this.getAllIds().filter((entityId) => {\n      const entity = this.get(entityId);\n      return entity.parentId === parentId;\n    });\n  }\n\n  setModulePath(key: string, modulePath: string): void {\n    // console.debug(chalk.cyan(`setModulePath :: ${key} :: ${modulePath}`));\n    this.modulePaths.set(key, modulePath);\n  }\n\n  getModulePath(key: string): string {\n    const modulePath = this.modulePaths.get(key);\n    if (modulePath === undefined) {\n      throw new Error(`존재하지 않는 모듈 패스 요청 ${key}`);\n    }\n\n    return modulePath;\n  }\n\n  setTableSpec(tableSpec: TableSpec) {\n    this.tableSpecs.set(tableSpec.name, tableSpec);\n  }\n\n  getTableSpec(key: string): TableSpec {\n    const tableSpec = this.tableSpecs.get(key);\n    if (tableSpec === undefined) {\n      throw new Error(`존재하지 않는 테이블 스펙 요청 ${key}`);\n    }\n\n    return tableSpec;\n  }\n\n  getNamesFromId(entityId: string): EntityNamesRecord {\n    // entityId가 단복수 동형 단어인 경우 List 붙여서 생성\n    const pluralized =\n      inflection.pluralize(entityId) === entityId\n        ? `${entityId}List`\n        : inflection.pluralize(entityId);\n\n    return {\n      fs: inflection.dasherize(inflection.underscore(entityId)).toLowerCase(),\n      fsPlural: inflection.dasherize(inflection.underscore(pluralized)).toLowerCase(),\n      camel: inflection.camelize(entityId, true),\n      camelPlural: inflection.camelize(pluralized, true),\n      capital: entityId,\n      capitalPlural: pluralized,\n      upper: entityId.toUpperCase(),\n      constant: inflection.underscore(entityId).toUpperCase(),\n    };\n  }\n\n  /**\n   * EntityId는 Model을 제외한 PascalCase 이름입니다. (ex. \"User\")\n   * @param filePath\n   * @returns\n   */\n  getEntityIdFromPath(filePath: AbsolutePath): string {\n    const matched = filePath.match(/application\\/(.+)\\//);\n    assert(matched?.[1]);\n    return inflection.camelize(matched[1].replace(/-/g, \"_\"));\n  }\n\n  private async registerNonEntityTypeModulePaths(): Promise<void> {\n    const typePathsPatterns = [\n      path.join(Sonamu.apiRootPath, runtimePath(\"src/application/**/*.types.ts\")),\n      path.join(Sonamu.apiRootPath, runtimePath(\"src/application/**/*.generated.ts\")),\n    ];\n    const typePaths = (\n      await Promise.all(typePathsPatterns.map((pattern) => globAsync(pattern)))\n    ).flat();\n\n    for (const filePath of typePaths) {\n      const modulePath = this.getModulePathFromTypeFilePath(filePath);\n      const importedMembers = await importMembers<unknown>(filePath);\n      for (const { name, value } of importedMembers) {\n        if (value instanceof z.ZodType) {\n          this.setModulePath(name, modulePath);\n        }\n      }\n    }\n  }\n\n  private getModulePathFromTypeFilePath(filePath: string): string {\n    const normalizedPath = filePath.replaceAll(\"\\\\\", \"/\");\n    const matched = normalizedPath.match(/\\/(?:src|dist)\\/application\\/(.+)\\.(?:ts|js)$/);\n\n    if (!matched?.[1]) {\n      throw new Error(`타입 파일의 모듈 경로를 계산할 수 없습니다: ${filePath}`);\n    }\n\n    return matched[1];\n  }\n}\n\nexport const EntityManager = new EntityManagerClass();\n"],"names":["assert","chalk","glob","readFile","inflection","path","prettifyError","z","Sonamu","EntityJsonSchema","globAsync","importMembers","runtimePath","Entity","EntityManagerClass","entities","Map","modulePaths","tableSpecs","isAutoloaded","autoload","_","pathPattern","join","apiRootPath","file","resolve","json","JSON","parse","toString","error","schemaValidate","relativePath","relative","errorMessage","console","red","yellow","register","registerNonEntityTypeModulePaths","result","safeParse","success","reload","doSilent","clear","entity","registerModulePaths","registerTableSpecs","set","id","get","entityId","undefined","Error","getByTable","table","Array","from","values","find","exists","getAllIds","EntityManager","keys","sort","getAllEntities","getAllParentIds","filter","parentId","getChildrenIds","setModulePath","key","modulePath","getModulePath","setTableSpec","tableSpec","name","getTableSpec","getNamesFromId","pluralized","pluralize","fs","dasherize","underscore","toLowerCase","fsPlural","camel","camelize","camelPlural","capital","capitalPlural","upper","toUpperCase","constant","getEntityIdFromPath","filePath","matched","match","replace","typePathsPatterns","typePaths","Promise","all","map","pattern","flat","getModulePathFromTypeFilePath","importedMembers","value","ZodType","normalizedPath","replaceAll"],"mappings":"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"}
|
|
226
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/entity/entity-manager.ts"],"sourcesContent":["import assert from \"assert\";\nimport chalk from \"chalk\";\nimport { glob, readFile } from \"fs/promises\";\nimport inflection from \"inflection\";\nimport path from \"path\";\nimport { prettifyError, z } from \"zod\";\nimport { Sonamu } from \"../api/sonamu\";\nimport {\n  type EntityIndex,\n  type EntityJson,\n  EntityJsonSchema,\n  isSearchTextJsonSourceZodType,\n  isSearchTextProp,\n  SonamuFileArraySchema,\n  SonamuFileSchema,\n} from \"../types/types\";\nimport { globAsync } from \"../utils/async-utils\";\nimport { importMembers } from \"../utils/esm-utils\";\nimport type { AbsolutePath } from \"../utils/path-utils\";\nimport { runtimePath } from \"../utils/path-utils\";\nimport { Entity } from \"./entity\";\n\nexport type EntityNamesRecord = Record<\n  \"fs\" | \"fsPlural\" | \"camel\" | \"camelPlural\" | \"capital\" | \"capitalPlural\" | \"upper\" | \"constant\",\n  string\n>;\nexport type TableSpec = {\n  name: string;\n  uniqueIndexes: EntityIndex[];\n  jsonColumns: string[];\n};\nclass EntityManagerClass {\n  private entities: Map<string, Entity> = new Map();\n  public modulePaths: Map<string, string> = new Map();\n  private tableSpecs: Map<string, TableSpec> = new Map();\n  public isAutoloaded: boolean = false;\n\n  // 경로 전달받아 모든 entity.json 파일 로드\n  async autoload(_: boolean = false) {\n    if (this.isAutoloaded) {\n      return;\n    }\n    const pathPattern = path.join(Sonamu.apiRootPath, \"/src/application/**/*.entity.json\");\n\n    for await (const file of glob(path.resolve(pathPattern))) {\n      const json = JSON.parse((await readFile(file)).toString());\n\n      // entity.json 스키마 검증\n      const error = this.schemaValidate(json);\n      if (error) {\n        const relativePath = path.relative(Sonamu.apiRootPath, file);\n        const errorMessage = prettifyError(error);\n        console.error(\n          chalk.red(`Invalid entity.json schema: ${relativePath}\\n${chalk.yellow(errorMessage)}`),\n        );\n      }\n\n      await this.register(json, { deferSearchTextJsonSourceValidation: true });\n    }\n\n    await this.registerNonEntityTypeModulePaths();\n    await this.validateAllRegisteredSearchTextJsonSources();\n\n    this.isAutoloaded = true;\n  }\n\n  schemaValidate(json: unknown) {\n    const result = EntityJsonSchema.safeParse(json);\n    return result.success ? null : result.error;\n  }\n\n  async reload(doSilent: boolean = false) {\n    this.entities.clear();\n    this.modulePaths.clear();\n    this.tableSpecs.clear();\n    this.isAutoloaded = false;\n\n    return await this.autoload(doSilent);\n  }\n\n  async register(\n    json: EntityJson,\n    options: { deferSearchTextJsonSourceValidation?: boolean } = {},\n  ): Promise<void> {\n    const entity = new Entity(json);\n    await entity.registerModulePaths();\n    if (!options.deferSearchTextJsonSourceValidation) {\n      await this.validateSearchTextJsonSources(entity);\n    }\n    entity.registerTableSpecs();\n    this.entities.set(json.id, entity);\n  }\n\n  async validateAllRegisteredSearchTextJsonSources(): Promise<void> {\n    for (const entity of this.entities.values()) {\n      await this.validateSearchTextJsonSources(entity);\n    }\n  }\n\n  private async validateSearchTextJsonSources(entity: Entity): Promise<void> {\n    const propsByName = new Map(entity.props.map((prop) => [prop.name, prop]));\n\n    for (const prop of entity.props) {\n      if (!isSearchTextProp(prop)) {\n        continue;\n      }\n\n      for (const source of prop.sourceColumns) {\n        const sourceProp = propsByName.get(source.name);\n        if (!sourceProp || sourceProp.type !== \"json\") {\n          continue;\n        }\n\n        const zodType = await this.resolveSearchTextJsonSourceType(entity, sourceProp.id);\n        if (!zodType) {\n          throw new Error(\n            `searchText source \"${source.name}\"의 json 타입 \"${sourceProp.id}\"을(를) 로드할 수 없습니다.`,\n          );\n        }\n\n        if (!isSearchTextJsonSourceZodType(zodType)) {\n          throw new Error(\n            `searchText source \"${source.name}\"의 json 타입 \"${sourceProp.id}\"은(는) unwrap 후 z.array(z.string()) 이어야 합니다.`,\n          );\n        }\n      }\n    }\n  }\n\n  private async resolveSearchTextJsonSourceType(\n    entity: Entity,\n    typeId: string,\n  ): Promise<z.ZodTypeAny | null> {\n    const localType = entity.types[typeId];\n    if (localType instanceof z.ZodType) {\n      return localType;\n    }\n\n    for (const registeredEntity of this.entities.values()) {\n      const registeredType = registeredEntity.types[typeId];\n      if (registeredType instanceof z.ZodType) {\n        return registeredType;\n      }\n    }\n\n    if (typeId === \"SonamuFile\") {\n      return SonamuFileSchema;\n    }\n    if (typeId === \"SonamuFile[]\") {\n      return SonamuFileArraySchema;\n    }\n\n    const modulePath = this.modulePaths.get(typeId);\n    if (!modulePath) {\n      return null;\n    }\n\n    const moduleFilePath = path.join(\n      Sonamu.apiRootPath,\n      runtimePath(`dist/application/${modulePath}.js`),\n    );\n    const importedMembers = await importMembers<unknown>(moduleFilePath);\n    const matched = importedMembers.find(({ name }) => name === typeId);\n    if (!matched || !(matched.value instanceof z.ZodType)) {\n      return null;\n    }\n\n    return matched.value;\n  }\n\n  get(entityId: string): Entity {\n    const entity = this.entities.get(entityId);\n    if (entity === undefined) {\n      throw new Error(`존재하지 않는 Entity 요청 ${entityId}`);\n    }\n\n    return entity;\n  }\n\n  getByTable(table: string): Entity {\n    const entity = Array.from(this.entities.values()).find((entity) => entity.table === table);\n    if (entity === undefined) {\n      throw new Error(`존재하지 않는 Entity 요청 ${table}`);\n    }\n\n    return entity;\n  }\n\n  exists(entityId: string): boolean {\n    const entity = this.entities.get(entityId);\n    return entity !== undefined;\n  }\n\n  getAllIds(): string[] {\n    return Array.from(EntityManager.entities.keys()).sort();\n  }\n\n  getAllEntities(): Entity[] {\n    return Array.from(this.entities.values());\n  }\n\n  getAllParentIds(): string[] {\n    return this.getAllIds().filter((entityId) => {\n      const entity = this.get(entityId);\n      return entity.parentId === undefined;\n    });\n  }\n\n  getChildrenIds(parentId: string): string[] {\n    return this.getAllIds().filter((entityId) => {\n      const entity = this.get(entityId);\n      return entity.parentId === parentId;\n    });\n  }\n\n  setModulePath(key: string, modulePath: string): void {\n    // console.debug(chalk.cyan(`setModulePath :: ${key} :: ${modulePath}`));\n    this.modulePaths.set(key, modulePath);\n  }\n\n  getModulePath(key: string): string {\n    const modulePath = this.modulePaths.get(key);\n    if (modulePath === undefined) {\n      throw new Error(`존재하지 않는 모듈 패스 요청 ${key}`);\n    }\n\n    return modulePath;\n  }\n\n  setTableSpec(tableSpec: TableSpec) {\n    this.tableSpecs.set(tableSpec.name, tableSpec);\n  }\n\n  getTableSpec(key: string): TableSpec {\n    const tableSpec = this.tableSpecs.get(key);\n    if (tableSpec === undefined) {\n      throw new Error(`존재하지 않는 테이블 스펙 요청 ${key}`);\n    }\n\n    return tableSpec;\n  }\n\n  getNamesFromId(entityId: string): EntityNamesRecord {\n    // entityId가 단복수 동형 단어인 경우 List 붙여서 생성\n    const pluralized =\n      inflection.pluralize(entityId) === entityId\n        ? `${entityId}List`\n        : inflection.pluralize(entityId);\n\n    return {\n      fs: inflection.dasherize(inflection.underscore(entityId)).toLowerCase(),\n      fsPlural: inflection.dasherize(inflection.underscore(pluralized)).toLowerCase(),\n      camel: inflection.camelize(entityId, true),\n      camelPlural: inflection.camelize(pluralized, true),\n      capital: entityId,\n      capitalPlural: pluralized,\n      upper: entityId.toUpperCase(),\n      constant: inflection.underscore(entityId).toUpperCase(),\n    };\n  }\n\n  /**\n   * EntityId는 Model을 제외한 PascalCase 이름입니다. (ex. \"User\")\n   * @param filePath\n   * @returns\n   */\n  getEntityIdFromPath(filePath: AbsolutePath): string {\n    const matched = filePath.match(/application\\/(.+)\\//);\n    assert(matched?.[1]);\n    return inflection.camelize(matched[1].replace(/-/g, \"_\"));\n  }\n\n  private async registerNonEntityTypeModulePaths(): Promise<void> {\n    const typePathsPatterns = [\n      path.join(Sonamu.apiRootPath, runtimePath(\"src/application/**/*.types.ts\")),\n      path.join(Sonamu.apiRootPath, runtimePath(\"src/application/**/*.generated.ts\")),\n    ];\n    const typePaths = (\n      await Promise.all(typePathsPatterns.map((pattern) => globAsync(pattern)))\n    ).flat();\n\n    for (const filePath of typePaths) {\n      const modulePath = this.getModulePathFromTypeFilePath(filePath);\n      const importedMembers = await importMembers<unknown>(filePath);\n      for (const { name, value } of importedMembers) {\n        if (value instanceof z.ZodType) {\n          this.setModulePath(name, modulePath);\n        }\n      }\n    }\n  }\n\n  private getModulePathFromTypeFilePath(filePath: string): string {\n    const normalizedPath = filePath.replaceAll(\"\\\\\", \"/\");\n    const matched = normalizedPath.match(/\\/(?:src|dist)\\/application\\/(.+)\\.(?:ts|js)$/);\n\n    if (!matched?.[1]) {\n      throw new Error(`타입 파일의 모듈 경로를 계산할 수 없습니다: ${filePath}`);\n    }\n\n    return matched[1];\n  }\n}\n\nexport const EntityManager = new EntityManagerClass();\n"],"names":["assert","chalk","glob","readFile","inflection","path","prettifyError","z","Sonamu","EntityJsonSchema","isSearchTextJsonSourceZodType","isSearchTextProp","SonamuFileArraySchema","SonamuFileSchema","globAsync","importMembers","runtimePath","Entity","EntityManagerClass","entities","Map","modulePaths","tableSpecs","isAutoloaded","autoload","_","pathPattern","join","apiRootPath","file","resolve","json","JSON","parse","toString","error","schemaValidate","relativePath","relative","errorMessage","console","red","yellow","register","deferSearchTextJsonSourceValidation","registerNonEntityTypeModulePaths","validateAllRegisteredSearchTextJsonSources","result","safeParse","success","reload","doSilent","clear","options","entity","registerModulePaths","validateSearchTextJsonSources","registerTableSpecs","set","id","values","propsByName","props","map","prop","name","source","sourceColumns","sourceProp","get","type","zodType","resolveSearchTextJsonSourceType","Error","typeId","localType","types","ZodType","registeredEntity","registeredType","modulePath","moduleFilePath","importedMembers","matched","find","value","entityId","undefined","getByTable","table","Array","from","exists","getAllIds","EntityManager","keys","sort","getAllEntities","getAllParentIds","filter","parentId","getChildrenIds","setModulePath","key","getModulePath","setTableSpec","tableSpec","getTableSpec","getNamesFromId","pluralized","pluralize","fs","dasherize","underscore","toLowerCase","fsPlural","camel","camelize","camelPlural","capital","capitalPlural","upper","toUpperCase","constant","getEntityIdFromPath","filePath","match","replace","typePathsPatterns","typePaths","Promise","all","pattern","flat","getModulePathFromTypeFilePath","normalizedPath","replaceAll"],"mappings":"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"}
|