sonamu 0.8.13 → 0.8.14

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (96) hide show
  1. package/dist/api/sonamu.d.ts.map +1 -1
  2. package/dist/api/sonamu.js +2 -3
  3. package/dist/auth/auth-generator.d.ts +8 -0
  4. package/dist/auth/auth-generator.d.ts.map +1 -1
  5. package/dist/auth/auth-generator.js +33 -1
  6. package/dist/auth/better-auth-entities.d.ts.map +1 -1
  7. package/dist/auth/better-auth-entities.js +12 -2
  8. package/dist/bin/cli.js +18 -3
  9. package/dist/cone/cone-generator.js +10 -4
  10. package/dist/database/knex.d.ts.map +1 -1
  11. package/dist/database/knex.js +64 -2
  12. package/dist/database/puri.d.ts +9 -1
  13. package/dist/database/puri.d.ts.map +1 -1
  14. package/dist/database/puri.js +42 -1
  15. package/dist/database/puri.types.d.ts +2 -0
  16. package/dist/database/puri.types.d.ts.map +1 -1
  17. package/dist/database/puri.types.js +6 -2
  18. package/dist/entity/entity-manager.d.ts +149 -1
  19. package/dist/entity/entity-manager.d.ts.map +1 -1
  20. package/dist/entity/entity-manager.js +68 -4
  21. package/dist/migration/__tests__/code-generation.search-text.test.js +435 -0
  22. package/dist/migration/code-generation.d.ts.map +1 -1
  23. package/dist/migration/code-generation.js +696 -32
  24. package/dist/migration/migration-set.js +3 -1
  25. package/dist/migration/postgresql-schema-reader.d.ts +16 -2
  26. package/dist/migration/postgresql-schema-reader.d.ts.map +1 -1
  27. package/dist/migration/postgresql-schema-reader.js +281 -7
  28. package/dist/stream/sse.js +5 -3
  29. package/dist/template/__tests__/generated.template.search-text.test.js +99 -0
  30. package/dist/template/generated.template.test-d.js +24 -0
  31. package/dist/template/implementations/generated.template.d.ts.map +1 -1
  32. package/dist/template/implementations/generated.template.js +2 -2
  33. package/dist/template/implementations/init_types.template.d.ts.map +1 -1
  34. package/dist/template/implementations/init_types.template.js +11 -3
  35. package/dist/template/zod-converter.d.ts.map +1 -1
  36. package/dist/template/zod-converter.js +6 -2
  37. package/dist/testing/dev-test-routes.d.ts.map +1 -1
  38. package/dist/testing/dev-test-routes.js +5 -3
  39. package/dist/testing/fixture-generator.d.ts +13 -0
  40. package/dist/testing/fixture-generator.d.ts.map +1 -1
  41. package/dist/testing/fixture-generator.js +105 -8
  42. package/dist/testing/fixture-manager.d.ts.map +1 -1
  43. package/dist/testing/fixture-manager.js +19 -2
  44. package/dist/types/__tests__/entity-json-schema-search-text.test.js +256 -0
  45. package/dist/types/types.d.ts +494 -1
  46. package/dist/types/types.d.ts.map +1 -1
  47. package/dist/types/types.js +117 -13
  48. package/dist/ui/api.d.ts.map +1 -1
  49. package/dist/ui/api.js +14 -2
  50. package/dist/ui/cdd-service.d.ts +16 -14
  51. package/dist/ui/cdd-service.d.ts.map +1 -1
  52. package/dist/ui/cdd-service.js +145 -37
  53. package/dist/ui/cdd-types.d.ts +60 -0
  54. package/dist/ui/cdd-types.d.ts.map +1 -0
  55. package/dist/ui/cdd-types.js +3 -0
  56. package/dist/ui-web/assets/index-D4XFBV-f.css +1 -0
  57. package/dist/ui-web/assets/{index-CQ_S40bD.js → index-D_19-Pi4.js} +87 -87
  58. package/dist/ui-web/index.html +2 -2
  59. package/package.json +7 -3
  60. package/src/api/sonamu.ts +1 -2
  61. package/src/auth/auth-generator.ts +38 -0
  62. package/src/auth/better-auth-entities.ts +18 -1
  63. package/src/bin/cli.ts +15 -1
  64. package/src/cone/cone-generator.ts +9 -3
  65. package/src/database/knex.ts +62 -4
  66. package/src/database/puri.ts +71 -0
  67. package/src/database/puri.types.ts +2 -0
  68. package/src/entity/entity-manager.ts +95 -3
  69. package/src/migration/__tests__/code-generation.search-text.test.ts +390 -0
  70. package/src/migration/code-generation.ts +848 -34
  71. package/src/migration/migration-set.ts +2 -0
  72. package/src/migration/postgresql-schema-reader.ts +366 -9
  73. package/src/skills/sonamu/auth-migration.md +80 -0
  74. package/src/skills/sonamu/cdd.md +148 -28
  75. package/src/skills/sonamu/cone.md +16 -0
  76. package/src/skills/sonamu/entity-relations.md +1 -1
  77. package/src/skills/sonamu/fixture-cli.md +4 -0
  78. package/src/skills/sonamu/frontend.md +65 -0
  79. package/src/skills/sonamu/migration.md +3 -1
  80. package/src/skills/sonamu/model.md +28 -0
  81. package/src/skills/sonamu/workflow.md +12 -5
  82. package/src/stream/sse.ts +4 -2
  83. package/src/template/__tests__/generated.template.search-text.test.ts +89 -0
  84. package/src/template/generated.template.test-d.ts +46 -0
  85. package/src/template/implementations/generated.template.ts +4 -1
  86. package/src/template/implementations/init_types.template.ts +20 -5
  87. package/src/template/zod-converter.ts +5 -0
  88. package/src/testing/dev-test-routes.ts +4 -2
  89. package/src/testing/fixture-generator.ts +157 -9
  90. package/src/testing/fixture-manager.ts +15 -1
  91. package/src/types/__tests__/entity-json-schema-search-text.test.ts +179 -0
  92. package/src/types/types.ts +168 -12
  93. package/src/ui/api.ts +24 -1
  94. package/src/ui/cdd-service.ts +195 -55
  95. package/src/ui/cdd-types.ts +73 -0
  96. package/dist/ui-web/assets/index-egkMxKos.css +0 -1
@@ -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): Promise<void>;
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,EAAE,KAAK,WAAW,EAAE,KAAK,UAAU,EAAoB,MAAM,gBAAgB,CAAC;AAGrF,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AAExD,OAAO,EAAE,MAAM,EAAE,MAAM,UAAU,CAAC;AAElC,MAAM,MAAM,iBAAiB,GAAG,MAAM,CACpC,IAAI,GAAG,UAAU,GAAG,OAAO,GAAG,aAAa,GAAG,SAAS,GAAG,eAAe,GAAG,OAAO,GAAG,UAAU,EAChG,MAAM,CACP,CAAC;AACF,MAAM,MAAM,SAAS,GAAG;IACtB,IAAI,EAAE,MAAM,CAAC;IACb,aAAa,EAAE,WAAW,EAAE,CAAC;IAC7B,WAAW,EAAE,MAAM,EAAE,CAAC;CACvB,CAAC;AACF,cAAM,kBAAkB;IACtB,OAAO,CAAC,QAAQ,CAAkC;IAC3C,WAAW,EAAE,GAAG,CAAC,MAAM,EAAE,MAAM,CAAC,CAAa;IACpD,OAAO,CAAC,UAAU,CAAqC;IAChD,YAAY,EAAE,OAAO,CAAS;IAG/B,QAAQ,CAAC,CAAC,GAAE,OAAe;IA2BjC,cAAc,CAAC,IAAI,EAAE,OAAO;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;IAKtB,MAAM,CAAC,QAAQ,GAAE,OAAe;IAShC,QAAQ,CAAC,IAAI,EAAE,UAAU,GAAG,OAAO,CAAC,IAAI,CAAC;IAO/C,GAAG,CAAC,QAAQ,EAAE,MAAM,GAAG,MAAM;IAS7B,UAAU,CAAC,KAAK,EAAE,MAAM,GAAG,MAAM;IASjC,MAAM,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO;IAKjC,SAAS,IAAI,MAAM,EAAE;IAIrB,cAAc,IAAI,MAAM,EAAE;IAI1B,eAAe,IAAI,MAAM,EAAE;IAO3B,cAAc,CAAC,QAAQ,EAAE,MAAM,GAAG,MAAM,EAAE;IAO1C,aAAa,CAAC,GAAG,EAAE,MAAM,EAAE,UAAU,EAAE,MAAM,GAAG,IAAI;IAKpD,aAAa,CAAC,GAAG,EAAE,MAAM,GAAG,MAAM;IASlC,YAAY,CAAC,SAAS,EAAE,SAAS;IAIjC,YAAY,CAAC,GAAG,EAAE,MAAM,GAAG,SAAS;IASpC,cAAc,CAAC,QAAQ,EAAE,MAAM,GAAG,iBAAiB;IAmBnD;;;;OAIG;IACH,mBAAmB,CAAC,QAAQ,EAAE,YAAY,GAAG,MAAM;YAMrC,gCAAgC;IAoB9C,OAAO,CAAC,6BAA6B;CAUtC;AAED,eAAO,MAAM,aAAa,oBAA2B,CAAC"}
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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"}