sonamu 0.7.15 → 0.7.17
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/ai/providers/rtzr/error.d.ts +1 -1
- package/dist/ai/providers/rtzr/error.d.ts.map +1 -1
- package/dist/api/config.d.ts +1 -0
- package/dist/api/config.d.ts.map +1 -1
- package/dist/api/config.js +1 -1
- package/dist/api/decorators.d.ts +1 -1
- package/dist/api/decorators.d.ts.map +1 -1
- package/dist/api/decorators.js +1 -1
- package/dist/api/sonamu.d.ts +3 -1
- package/dist/api/sonamu.d.ts.map +1 -1
- package/dist/api/sonamu.js +51 -40
- package/dist/database/base-model.d.ts +16 -6
- package/dist/database/base-model.d.ts.map +1 -1
- package/dist/database/base-model.js +44 -3
- package/dist/database/base-model.types.d.ts +29 -48
- package/dist/database/base-model.types.d.ts.map +1 -1
- package/dist/database/base-model.types.js +12 -2
- package/dist/database/puri.d.ts +2 -1
- package/dist/database/puri.d.ts.map +1 -1
- package/dist/database/puri.js +2 -1
- package/dist/database/puri.types.d.ts +3 -3
- package/dist/database/puri.types.d.ts.map +1 -1
- package/dist/database/puri.types.js +1 -1
- package/dist/entity/entity-manager.d.ts +8 -4
- package/dist/entity/entity-manager.d.ts.map +1 -1
- package/dist/entity/entity.d.ts +10 -1
- package/dist/entity/entity.d.ts.map +1 -1
- package/dist/entity/entity.js +84 -39
- package/dist/index.d.ts +1 -0
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +2 -1
- package/dist/syncer/checksum.d.ts +8 -3
- package/dist/syncer/checksum.d.ts.map +1 -1
- package/dist/syncer/checksum.js +17 -9
- package/dist/syncer/code-generator.js +7 -2
- package/dist/syncer/syncer.d.ts +6 -6
- package/dist/syncer/syncer.d.ts.map +1 -1
- package/dist/syncer/syncer.js +27 -13
- package/dist/tasks/workflow-manager.d.ts +3 -3
- package/dist/tasks/workflow-manager.d.ts.map +1 -1
- package/dist/tasks/workflow-manager.js +15 -11
- package/dist/template/implementations/generated.template.d.ts.map +1 -1
- package/dist/template/implementations/generated.template.js +8 -6
- package/dist/template/implementations/model.template.js +5 -5
- package/dist/template/implementations/services.template.d.ts +17 -0
- package/dist/template/implementations/services.template.d.ts.map +1 -0
- package/dist/template/implementations/services.template.js +159 -0
- package/dist/template/implementations/view_form.template.js +2 -2
- package/dist/template/implementations/view_id_async_select.template.js +2 -2
- package/dist/template/implementations/view_list.template.js +5 -5
- package/dist/types/types.d.ts +43 -25
- package/dist/types/types.d.ts.map +1 -1
- package/dist/types/types.js +29 -17
- package/dist/ui/ai-api.d.ts +2 -0
- package/dist/ui/ai-api.d.ts.map +1 -1
- package/dist/ui/ai-api.js +43 -49
- package/dist/ui/ai-client.d.ts +10 -0
- package/dist/ui/ai-client.d.ts.map +1 -1
- package/dist/ui/ai-client.js +457 -437
- package/dist/ui/api.d.ts.map +1 -1
- package/dist/ui/api.js +14 -3
- package/dist/ui-web/assets/{index-J9MCfjCd.js → index-DzqUrTB-.js} +56 -59
- package/dist/ui-web/index.html +1 -1
- package/package.json +12 -8
- package/src/api/config.ts +3 -0
- package/src/api/decorators.ts +6 -1
- package/src/api/sonamu.ts +71 -52
- package/src/database/base-model.ts +66 -11
- package/src/database/base-model.types.ts +79 -76
- package/src/database/puri.ts +5 -1
- package/src/database/puri.types.ts +3 -6
- package/src/entity/entity.ts +83 -34
- package/src/index.ts +1 -0
- package/src/shared/app.shared.ts.txt +1 -1
- package/src/shared/web.shared.ts.txt +0 -43
- package/src/syncer/checksum.ts +31 -9
- package/src/syncer/code-generator.ts +8 -1
- package/src/syncer/syncer.ts +38 -26
- package/src/tasks/workflow-manager.ts +16 -12
- package/src/template/implementations/generated.template.ts +17 -3
- package/src/template/implementations/model.template.ts +4 -4
- package/src/template/implementations/services.template.ts +226 -0
- package/src/template/implementations/view_form.template.ts +1 -1
- package/src/template/implementations/view_id_async_select.template.ts +1 -1
- package/src/template/implementations/view_list.template.ts +4 -4
- package/src/types/types.ts +33 -16
- package/src/ui/ai-api.ts +61 -60
- package/src/ui/ai-client.ts +535 -499
- package/src/ui/api.ts +14 -2
- package/src/ui/entity.instructions.md +536 -0
- package/dist/template/implementations/service.template.d.ts +0 -29
- package/dist/template/implementations/service.template.d.ts.map +0 -1
- package/dist/template/implementations/service.template.js +0 -202
- package/dist/ui-web/assets/provider-utils_false-BKJD46kk.js +0 -1
- package/dist/ui-web/assets/provider-utils_false-Bu5lmX18.js +0 -1
- package/src/template/implementations/service.template.ts +0 -328
package/dist/entity/entity.js
CHANGED
|
@@ -5,7 +5,7 @@ import path from "path";
|
|
|
5
5
|
import { group, unique } from "radashi";
|
|
6
6
|
import { z } from "zod";
|
|
7
7
|
import { Sonamu } from "../api/sonamu.js";
|
|
8
|
-
import { isBelongsToOneRelationProp, isEnumProp, isHasManyRelationProp, isManyToManyRelationProp, isOneToOneRelationProp, isRelationProp, isVirtualProp } from "../types/types.js";
|
|
8
|
+
import { isBelongsToOneRelationProp, isEnumProp, isHasManyRelationProp, isInternalSubsetField, isManyToManyRelationProp, isOneToOneRelationProp, isRelationProp, isVirtualCodeProp, isVirtualProp, normalizeSubsetField } from "../types/types.js";
|
|
9
9
|
import { importMembers } from "../utils/esm-utils.js";
|
|
10
10
|
import { formatCode } from "../utils/formatter.js";
|
|
11
11
|
import { exists } from "../utils/fs-utils.js";
|
|
@@ -23,6 +23,7 @@ export class Entity {
|
|
|
23
23
|
relations;
|
|
24
24
|
indexes;
|
|
25
25
|
subsets;
|
|
26
|
+
subsetsInternal;
|
|
26
27
|
types = {};
|
|
27
28
|
enums = {};
|
|
28
29
|
enumLabels = {};
|
|
@@ -60,8 +61,13 @@ export class Entity {
|
|
|
60
61
|
}
|
|
61
62
|
// indexes
|
|
62
63
|
this.indexes = indexes ?? [];
|
|
63
|
-
// subsets
|
|
64
|
-
this.subsets =
|
|
64
|
+
// subsets: SubsetField[]를 파싱하여 subsets(일반)와 subsetsInternal(internal)로 분리
|
|
65
|
+
this.subsets = {};
|
|
66
|
+
this.subsetsInternal = {};
|
|
67
|
+
for (const [key, fields] of Object.entries(subsets ?? {})){
|
|
68
|
+
this.subsets[key] = fields.filter((f)=>!isInternalSubsetField(f)).map(normalizeSubsetField);
|
|
69
|
+
this.subsetsInternal[key] = fields.filter(isInternalSubsetField).map(normalizeSubsetField);
|
|
70
|
+
}
|
|
65
71
|
// enums
|
|
66
72
|
this.enumLabels = enums ?? {};
|
|
67
73
|
this.enums = Object.fromEntries(Object.entries(this.enumLabels).map(([key, enumLabel])=>{
|
|
@@ -78,9 +84,17 @@ export class Entity {
|
|
|
78
84
|
};
|
|
79
85
|
}
|
|
80
86
|
/**
|
|
87
|
+
* 쿼리용 서브셋 필드를 반환합니다 (subsets + subsetsInternal 합침)
|
|
88
|
+
*/ getSubsetFieldsForQuery(subsetKey) {
|
|
89
|
+
return [
|
|
90
|
+
...this.subsets[subsetKey] ?? [],
|
|
91
|
+
...this.subsetsInternal[subsetKey] ?? []
|
|
92
|
+
];
|
|
93
|
+
}
|
|
94
|
+
/**
|
|
81
95
|
* 주어진 이름(subsetKey)의 subset을 실제로 가져오는 Puri 코드 구현체 string을 반환합니다.
|
|
82
96
|
*/ getPuriSubsetQuery(subsetKey) {
|
|
83
|
-
const subset = this.
|
|
97
|
+
const subset = this.getSubsetFieldsForQuery(subsetKey);
|
|
84
98
|
const subsetQuery = this.resolveSubsetQuery("", subset);
|
|
85
99
|
const lines = [];
|
|
86
100
|
// from
|
|
@@ -190,7 +204,7 @@ export class Entity {
|
|
|
190
204
|
}
|
|
191
205
|
}
|
|
192
206
|
getPuriLoaderQuery(subsetKey) {
|
|
193
|
-
const subset = this.
|
|
207
|
+
const subset = this.getSubsetFieldsForQuery(subsetKey);
|
|
194
208
|
const { loaders } = this.resolveSubsetQuery("", subset);
|
|
195
209
|
const lines = [
|
|
196
210
|
`[`
|
|
@@ -253,7 +267,7 @@ export class Entity {
|
|
|
253
267
|
/*
|
|
254
268
|
subset SELECT/JOIN/LOADER 결과 리턴
|
|
255
269
|
*/ getSubsetQuery(subsetKey) {
|
|
256
|
-
const subset = this.
|
|
270
|
+
const subset = this.getSubsetFieldsForQuery(subsetKey);
|
|
257
271
|
const result = this.resolveSubsetQuery("", subset);
|
|
258
272
|
return result;
|
|
259
273
|
}
|
|
@@ -276,11 +290,13 @@ export class Entity {
|
|
|
276
290
|
// 현재 테이블 필드셋은 select, virtual에 추가하고 리턴
|
|
277
291
|
if (groupKey === "") {
|
|
278
292
|
const realFields = fields.filter((field)=>!isVirtualProp(this.propsDict[field]));
|
|
279
|
-
|
|
293
|
+
// virtualType: "code" (또는 undefined)인 virtual prop만 r.virtual에 추가
|
|
294
|
+
// virtualType: "query"인 경우 사용자가 appendSelect로 직접 추가하므로 제외
|
|
295
|
+
const virtualCodeFields = fields.filter((field)=>isVirtualCodeProp(this.propsDict[field]));
|
|
280
296
|
if (prefix === "") {
|
|
281
297
|
// 현재 테이블인 경우
|
|
282
298
|
r.select = r.select.concat(realFields.map((field)=>`${this.table}.${field}`));
|
|
283
|
-
r.virtual = r.virtual.concat(
|
|
299
|
+
r.virtual = r.virtual.concat(virtualCodeFields);
|
|
284
300
|
} else {
|
|
285
301
|
// 넘어온 테이블인 경우
|
|
286
302
|
r.select = r.select.concat(realFields.map((field)=>`${prefix}.${field} as ${prefix}__${field}`));
|
|
@@ -448,18 +464,6 @@ export class Entity {
|
|
|
448
464
|
// 일반 prop 처리
|
|
449
465
|
if (key === "") {
|
|
450
466
|
return group.map((propName)=>{
|
|
451
|
-
// FIXME: 이거 나중에 없애야함
|
|
452
|
-
if (propName === "말도안되는프롭명__이거왜타입처리가꼬여서이러지?") {
|
|
453
|
-
return {
|
|
454
|
-
nodeType: "plain",
|
|
455
|
-
prop: {
|
|
456
|
-
type: "string",
|
|
457
|
-
name: "uuid",
|
|
458
|
-
length: 128
|
|
459
|
-
},
|
|
460
|
-
children: []
|
|
461
|
-
};
|
|
462
|
-
}
|
|
463
467
|
const prop = entity.props.find((p)=>p.name === propName);
|
|
464
468
|
if (prop === undefined) {
|
|
465
469
|
console.log({
|
|
@@ -470,8 +474,7 @@ export class Entity {
|
|
|
470
474
|
}
|
|
471
475
|
return {
|
|
472
476
|
nodeType: "plain",
|
|
473
|
-
prop
|
|
474
|
-
children: []
|
|
477
|
+
prop
|
|
475
478
|
};
|
|
476
479
|
});
|
|
477
480
|
}
|
|
@@ -492,8 +495,7 @@ export class Entity {
|
|
|
492
495
|
...idProp,
|
|
493
496
|
name: `${key}_id`,
|
|
494
497
|
nullable: prop.nullable
|
|
495
|
-
}
|
|
496
|
-
children: []
|
|
498
|
+
}
|
|
497
499
|
};
|
|
498
500
|
}
|
|
499
501
|
}
|
|
@@ -503,9 +505,9 @@ export class Entity {
|
|
|
503
505
|
const children = this.fieldExprsToPropNodes(group, relEntity);
|
|
504
506
|
const nodeType = isBelongsToOneRelationProp(prop) || isOneToOneRelationProp(prop) ? "object" : "array";
|
|
505
507
|
return {
|
|
508
|
+
nodeType,
|
|
506
509
|
prop,
|
|
507
|
-
children
|
|
508
|
-
nodeType
|
|
510
|
+
children
|
|
509
511
|
};
|
|
510
512
|
});
|
|
511
513
|
}
|
|
@@ -608,6 +610,19 @@ export class Entity {
|
|
|
608
610
|
});
|
|
609
611
|
}
|
|
610
612
|
toJson() {
|
|
613
|
+
// subsets와 subsetsInternal을 SubsetField[] 형태로 복원
|
|
614
|
+
const subsets = {};
|
|
615
|
+
for (const key of Object.keys(this.subsets)){
|
|
616
|
+
const normalFields = this.subsets[key];
|
|
617
|
+
const internalFields = (this.subsetsInternal[key] ?? []).map((field)=>({
|
|
618
|
+
field,
|
|
619
|
+
internal: true
|
|
620
|
+
}));
|
|
621
|
+
subsets[key] = [
|
|
622
|
+
...normalFields,
|
|
623
|
+
...internalFields
|
|
624
|
+
];
|
|
625
|
+
}
|
|
611
626
|
return {
|
|
612
627
|
id: this.id,
|
|
613
628
|
parentId: this.parentId,
|
|
@@ -615,7 +630,7 @@ export class Entity {
|
|
|
615
630
|
title: this.title,
|
|
616
631
|
props: this.props,
|
|
617
632
|
indexes: this.indexes,
|
|
618
|
-
subsets
|
|
633
|
+
subsets,
|
|
619
634
|
enums: this.enumLabels
|
|
620
635
|
};
|
|
621
636
|
}
|
|
@@ -625,7 +640,13 @@ export class Entity {
|
|
|
625
640
|
this.subsets = Object.fromEntries(Object.entries(this.subsets).map(([subsetKey])=>{
|
|
626
641
|
return [
|
|
627
642
|
subsetKey,
|
|
628
|
-
this.subsetRowsToSubsetFields(subsetRows, subsetKey)
|
|
643
|
+
this.subsetRowsToSubsetFields(subsetRows, subsetKey, false)
|
|
644
|
+
];
|
|
645
|
+
}));
|
|
646
|
+
this.subsetsInternal = Object.fromEntries(Object.entries(this.subsetsInternal).map(([subsetKey])=>{
|
|
647
|
+
return [
|
|
648
|
+
subsetKey,
|
|
649
|
+
this.subsetRowsToSubsetFields(subsetRows, subsetKey, true)
|
|
629
650
|
];
|
|
630
651
|
}));
|
|
631
652
|
// save
|
|
@@ -635,20 +656,27 @@ export class Entity {
|
|
|
635
656
|
// reload
|
|
636
657
|
await EntityManager.register(json);
|
|
637
658
|
}
|
|
638
|
-
getSubsetRows(_subsets, prefixes = []) {
|
|
659
|
+
getSubsetRows(_subsets, _subsetsInternal, prefixes = []) {
|
|
639
660
|
if (prefixes.length > 10) {
|
|
640
661
|
return [];
|
|
641
662
|
}
|
|
642
663
|
const subsets = _subsets ?? this.subsets;
|
|
664
|
+
const subsetsInternal = _subsetsInternal ?? this.subsetsInternal;
|
|
643
665
|
const subsetKeys = Object.keys(subsets);
|
|
644
666
|
const allFields = unique(subsetKeys.flatMap((key)=>subsets[key]));
|
|
667
|
+
// internal 필드도 allFields에 포함 (relation 탐색용)
|
|
668
|
+
const allInternalFields = unique(subsetKeys.flatMap((key)=>subsetsInternal[key] ?? []));
|
|
669
|
+
const combinedFields = unique([
|
|
670
|
+
...allFields,
|
|
671
|
+
...allInternalFields
|
|
672
|
+
]);
|
|
645
673
|
return this.props.map((prop)=>{
|
|
646
|
-
if (prop.type === "relation" &&
|
|
674
|
+
if (prop.type === "relation" && combinedFields.find((f)=>f.startsWith(`${[
|
|
647
675
|
...prefixes,
|
|
648
676
|
prop.name
|
|
649
677
|
].join(".")}.`))) {
|
|
650
678
|
const relEntity = EntityManager.get(prop.with);
|
|
651
|
-
const children = relEntity.getSubsetRows(subsets, [
|
|
679
|
+
const children = relEntity.getSubsetRows(subsets, subsetsInternal, [
|
|
652
680
|
...prefixes,
|
|
653
681
|
`${prop.name}`
|
|
654
682
|
]);
|
|
@@ -663,9 +691,19 @@ export class Entity {
|
|
|
663
691
|
subsetKey,
|
|
664
692
|
children.every((child)=>child.has[subsetKey] === true)
|
|
665
693
|
];
|
|
694
|
+
})),
|
|
695
|
+
isInternal: Object.fromEntries(subsetKeys.map((subsetKey)=>{
|
|
696
|
+
return [
|
|
697
|
+
subsetKey,
|
|
698
|
+
children.every((child)=>child.isInternal[subsetKey] === true)
|
|
699
|
+
];
|
|
666
700
|
}))
|
|
667
701
|
};
|
|
668
702
|
}
|
|
703
|
+
const field = [
|
|
704
|
+
...prefixes,
|
|
705
|
+
prop.name
|
|
706
|
+
].join(".");
|
|
669
707
|
return {
|
|
670
708
|
field: prop.name,
|
|
671
709
|
children: [],
|
|
@@ -674,25 +712,32 @@ export class Entity {
|
|
|
674
712
|
has: Object.fromEntries(subsetKeys.map((subsetKey)=>{
|
|
675
713
|
const subsetFields = subsets[subsetKey];
|
|
676
714
|
const has = subsetFields.some((f)=>{
|
|
677
|
-
const field = [
|
|
678
|
-
...prefixes,
|
|
679
|
-
prop.name
|
|
680
|
-
].join(".");
|
|
681
715
|
return f === field || f.startsWith(`${field}.`);
|
|
682
716
|
});
|
|
683
717
|
return [
|
|
684
718
|
subsetKey,
|
|
685
719
|
has
|
|
686
720
|
];
|
|
721
|
+
})),
|
|
722
|
+
isInternal: Object.fromEntries(subsetKeys.map((subsetKey)=>{
|
|
723
|
+
const internalFields = subsetsInternal[subsetKey] ?? [];
|
|
724
|
+
const isInternal = internalFields.some((f)=>{
|
|
725
|
+
return f === field || f.startsWith(`${field}.`);
|
|
726
|
+
});
|
|
727
|
+
return [
|
|
728
|
+
subsetKey,
|
|
729
|
+
isInternal
|
|
730
|
+
];
|
|
687
731
|
}))
|
|
688
732
|
};
|
|
689
733
|
});
|
|
690
734
|
}
|
|
691
|
-
subsetRowsToSubsetFields(subsetRows, subsetKey) {
|
|
735
|
+
subsetRowsToSubsetFields(subsetRows, subsetKey, internal = false) {
|
|
736
|
+
const hasKey = internal ? "isInternal" : "has";
|
|
692
737
|
return subsetRows.map((subsetRow)=>{
|
|
693
738
|
if (subsetRow.children.length > 0) {
|
|
694
|
-
return this.subsetRowsToSubsetFields(subsetRow.children, subsetKey);
|
|
695
|
-
} else if (subsetRow
|
|
739
|
+
return this.subsetRowsToSubsetFields(subsetRow.children, subsetKey, internal);
|
|
740
|
+
} else if (subsetRow[hasKey][subsetKey]) {
|
|
696
741
|
return subsetRow.prefixes.concat(subsetRow.field).join(".");
|
|
697
742
|
} else {
|
|
698
743
|
return null;
|
|
@@ -835,4 +880,4 @@ export class Entity {
|
|
|
835
880
|
}
|
|
836
881
|
}
|
|
837
882
|
|
|
838
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/entity/entity.ts"],"sourcesContent":["import assert from \"assert\";\nimport { writeFile } from \"fs/promises\";\nimport inflection from \"inflection\";\nimport path from \"path\";\nimport { group, unique } from \"radashi\";\nimport { z } from \"zod\";\nimport { Sonamu } from \"../api/sonamu\";\nimport {\n  type EntityIndex,\n  type EntityJson,\n  type EntityProp,\n  type EntityPropNode,\n  type EntitySubsetRow,\n  isBelongsToOneRelationProp,\n  isEnumProp,\n  isHasManyRelationProp,\n  isManyToManyRelationProp,\n  isOneToOneRelationProp,\n  isRelationProp,\n  isVirtualProp,\n  type RelationProp,\n  type StringProp,\n  type SubsetQuery,\n} from \"../types/types\";\nimport { importMembers } from \"../utils/esm-utils\";\nimport { formatCode } from \"../utils/formatter\";\nimport { exists } from \"../utils/fs-utils\";\nimport { runtimePath } from \"../utils/path-utils\";\nimport { assertDefined, nonNullable } from \"../utils/utils\";\nimport { EntityManager } from \"./entity-manager\";\n\nexport class Entity {\n  id: string;\n  parentId?: string;\n  table: string;\n  title: string;\n  names: {\n    parentFs: string;\n    fs: string;\n    module: string;\n  };\n  props: EntityProp[];\n  propsDict: {\n    [key: string]: EntityProp;\n  };\n  relations: {\n    [key: string]: RelationProp;\n  };\n  indexes: EntityIndex[];\n  subsets: {\n    [key: string]: string[];\n  };\n  types: {\n    [name: string]: z.ZodTypeAny;\n  } = {};\n  enums: {\n    [enumId: string]: z.ZodEnum<Readonly<Record<string, string>>>;\n  } = {};\n  enumLabels: {\n    [enumId: string]: {\n      [key: string]: string;\n    };\n  } = {};\n\n  constructor({ id, parentId, table, title, props, indexes, subsets, enums }: EntityJson) {\n    // id\n    this.id = id;\n    this.parentId = parentId;\n    this.title = title ?? this.id;\n    this.table = table ?? inflection.underscore(inflection.pluralize(id));\n\n    // props\n    if (props) {\n      this.props = props.map((prop) => {\n        if (isEnumProp(prop)) {\n          if (prop.id.includes(\"$Model\")) {\n            prop.id = prop.id.replace(\"$Model\", id);\n          }\n        }\n        return prop;\n      });\n      this.propsDict = Object.fromEntries(\n        props.map((prop) => {\n          return [prop.name, prop];\n        }),\n      );\n\n      // relations\n      this.relations = Object.fromEntries(\n        props.filter((prop) => isRelationProp(prop)).map((prop) => [prop.name, prop]),\n      );\n    } else {\n      this.props = [];\n      this.propsDict = {};\n      this.relations = {};\n    }\n\n    // indexes\n    this.indexes = indexes ?? [];\n\n    // subsets\n    this.subsets = subsets ?? {};\n\n    // enums\n    this.enumLabels = enums ?? {};\n    this.enums = Object.fromEntries(\n      Object.entries(this.enumLabels).map(([key, enumLabel]) => {\n        return [key, z.enum(Object.keys(enumLabel) as unknown as readonly [string, ...string[]])];\n      }),\n    );\n\n    // names\n    this.names = {\n      parentFs: inflection.dasherize(inflection.underscore(parentId ?? id)).toLowerCase(),\n      fs: inflection.dasherize(inflection.underscore(id)).toLowerCase(),\n      module: id,\n    };\n  }\n\n  /**\n   * 주어진 이름(subsetKey)의 subset을 실제로 가져오는 Puri 코드 구현체 string을 반환합니다.\n   */\n  getPuriSubsetQuery(subsetKey: string): string {\n    const subset = this.subsets[subsetKey];\n    const subsetQuery = this.resolveSubsetQuery(\"\", subset);\n\n    const lines: string[] = [];\n\n    // from\n    lines.push(`return qbWrapper`);\n    lines.push(`.from(\"${this.table}\")`);\n\n    // join\n    for (const join of subsetQuery.joins) {\n      // join 메서드 결정: inner → join, outer → leftJoin\n      // FK nullable 여부는 leftJoin 타입 시그니처에서 자동으로 판단됨\n      const joinMethod = join.join === \"inner\" ? \"join\" : \"leftJoin\";\n\n      if (\"custom\" in join) {\n        // custom join clause는 raw 사용\n        lines.push(\n          `.${joinMethod}({ ${join.as}: \"${join.table}\" }, qbWrapper.knex.raw(\\`${join.custom}\\`))`,\n        );\n      } else {\n        lines.push(`.${joinMethod}({ ${join.as}: \"${join.table}\" }, \"${join.from}\", \"${join.to}\")`);\n      }\n    }\n\n    // select - 입체적 구조로 생성\n    const selectObj = this.buildNestedSelectObject(subsetQuery.select);\n\n    lines.push(`.select(${this.stringifyNestedSelectObject(selectObj)});`);\n\n    return lines.join(\"\\n\");\n  }\n\n  /**\n   * *.entity.json의 subset에 들어있는 필드 배열을 받아서,\n   * Puri의 SelectObject 타입으로 변환합니다.\n   *\n   * 예: [\"users.id\", \"parent.id\", \"parent.name\"]\n   *   → { id: \"users.id\", parent: { id: \"parent.id\", name: \"parent.name\" } }\n   *\n   * 언더바가 아닌 중첩 객체로 변환함에 유의하세요.\n   * 이렇게 중첩 객체로 변환하여 select에 넘겨주면 ParseSelectObject 타입이 join된 객체의 타입을 잘 잡아줄 수 있습니다.\n   * 즉, enhancer에서 row를 받았을 때 hydrate된 객체 자체의 nullity와 그 안쪽 필드의 nullity가 fk nullable 여부에 따라 잘 추론됩니다.\n   */\n  private buildNestedSelectObject(\n    selectItems: string[],\n    // biome-ignore lint/suspicious/noExplicitAny: 반환 오브젝트의 값은 string일 수도 있고 또다른 오브젝트일 수도 있는데, 이를 재귀 타입으로 나타낼 수 없어 any로 처리합니다.\n  ): Record<string, any> {\n    const result: ReturnType<typeof this.buildNestedSelectObject> = {};\n\n    for (const selectItem of selectItems) {\n      // \"users.id\" 또는 \"users.id as user__id\" 형태 파싱\n      const match = selectItem.match(/^(.+?)(?: as (.+))?$/);\n      if (!match) continue;\n\n      const [, column, alias] = match;\n      const columnValue = `\"${column.trim()}\"`;\n\n      if (!alias || !alias.includes(\"__\")) {\n        // alias가 없거나 __를 포함하지 않으면 최상위 필드\n        const key = alias ?? assertDefined(column.split(\".\").pop());\n        result[key] = columnValue;\n      } else {\n        // alias가 __를 포함하면 입체 구조로 그룹화\n        const parts = alias.split(\"__\");\n        let current = result;\n\n        // 마지막 파트 전까지 중첩 객체 생성\n        for (let i = 0; i < parts.length - 1; i++) {\n          const part = parts[i];\n          if (part in current) {\n            if (typeof current[part] === \"string\") {\n              // 입력이 [\"user\", \"user__id\"] 같은 경우!\n              // 애초에 말도 안 되지만 안전하게 예외를 던집니다.\n              throw new Error(\n                `Conflict detected in select items: parent path \"${parts.slice(0, i + 1).join(\"__\")}\" is already set as a field, cannot nest \"${alias}\" under it.`,\n              );\n            }\n          } else {\n            current[part] = {};\n          }\n          current = current[part];\n        }\n\n        // 마지막 파트에 값 설정\n        const lastPart = parts[parts.length - 1];\n        current[lastPart] = columnValue;\n      }\n    }\n\n    return result;\n  }\n\n  /**\n   * JSON.stringify와 유사한 일을 합니다.\n   * 다만 주어진 객체를 JSON이 아닌 TypeScript 객체 리터럴 스트링으로 만들어줍니다.\n   * key에 따옴표가 없어요.\n   * 출력 예시:\n   * ```typescript\n   * {\n   *   id: \"users.id\",\n   *   parent: {\n   *     id: \"parent.id\",\n   *     name: \"parent.name\",\n   *   },\n   * }\n   * ```\n   * @param obj 변환할 객체\n   * @param indent 들여쓰기 레벨\n   * @param withBraces true면 중괄호 포함, false면 내용만 반환\n   */\n  private stringifyNestedSelectObject(\n    // biome-ignore lint/suspicious/noExplicitAny: 중첩 오브젝트의 값은 string일 수도 있고 또다른 오브젝트일 수도 있는데, 이를 재귀 타입으로 나타낼 수 없어 any로 처리합니다.\n    obj: Record<string, any>,\n    indent: number = 0,\n    withBraces: boolean = true,\n  ): string {\n    const spaces = \"  \".repeat(indent);\n    const innerSpaces = \"  \".repeat(indent + 1);\n\n    const entries = Object.entries(obj);\n    if (entries.length === 0) return withBraces ? \"{}\" : \"\";\n\n    const lines = entries.map(([key, value]) => {\n      if (typeof value === \"string\") {\n        // 컬럼 경로 (이미 따옴표 포함)\n        return `${innerSpaces}${key}: ${value},`;\n      } else {\n        // 중첩 객체 (항상 중괄호 포함)\n        return `${innerSpaces}${key}: ${this.stringifyNestedSelectObject(value, indent + 1, true)},`;\n      }\n    });\n\n    if (withBraces) {\n      return `{\\n${lines.join(\"\\n\")}\\n${spaces}}`;\n    } else {\n      // 중괄호 없이 내용만 반환 (앞뒤 개행 제외)\n      return lines.join(\"\\n\");\n    }\n  }\n\n  getPuriLoaderQuery(subsetKey: string): string {\n    const subset = this.subsets[subsetKey];\n    const { loaders } = this.resolveSubsetQuery(\"\", subset);\n\n    const lines: string[] = [`[`];\n\n    // 재귀적으로 loader 생성하는 헬퍼 함수\n    const generateLoaderCode = (loaders: SubsetQuery[\"loaders\"]): string[] => {\n      const loaderLines: string[] = [];\n\n      for (const loader of loaders) {\n        const { toTable, toCol, through } = loader.manyJoin;\n        loaderLines.push(\n          \"{\",\n          `as: \"${loader.as}\",`,\n          `refId: \"${loader.manyJoin.idField}\",`,\n          `qb: (qbWrapper: PuriWrapper<DatabaseSchemaExtend>, fromIds: number[]) => {`,\n        );\n\n        if (through === undefined) {\n          // HasMany\n          loaderLines.push(\n            //\n            \"return qbWrapper\",\n            `.from(\"${toTable}\")`,\n          );\n\n          loader.oneJoins.forEach((join: SubsetQuery[\"joins\"][number]) => {\n            // FK nullable 여부는 leftJoin 타입 시그니처에서 자동으로 판단됨\n            const joinMethod = join.join === \"inner\" ? \"join\" : \"leftJoin\";\n            if (\"custom\" in join) {\n              // FIXME: 검증 필요\n              loaderLines.push(\n                `.${joinMethod}({ ${join.as}: \"${join.table}\" }, (j) => {`,\n                `j.on(Puri.rawString(\"${join.custom}\"));`,\n                \"})\",\n              );\n            } else {\n              loaderLines.push(\n                `.${joinMethod}({ ${join.as}: \"${join.table}\" }, \"${join.from}\", \"${join.to}\")`,\n              );\n            }\n          });\n\n          // 입체적 select 구조 생성 (refId 포함)\n          const selectObj = this.buildNestedSelectObject(loader.select);\n          selectObj.refId = `\"${toTable}.${toCol}\"`;\n          loaderLines.push(\n            `.whereIn(\"${toTable}.${toCol}\", fromIds)`,\n            `.select(${this.stringifyNestedSelectObject(selectObj)});`,\n          );\n        } else {\n          // ManyToMany\n          loaderLines.push(\n            \"return qbWrapper\",\n            `.from(\"${through.table}\")`,\n            `.join(\"${toTable}\", \"${through.table}.${through.toCol}\", \"${toTable}.${toCol}\")`,\n          );\n\n          loader.oneJoins.forEach((join: SubsetQuery[\"joins\"][number]) => {\n            // FK nullable 여부는 leftJoin 타입 시그니처에서 자동으로 판단됨\n            const joinMethod = join.join === \"inner\" ? \"join\" : \"leftJoin\";\n            if (\"custom\" in join) {\n              // FIXME: 검증 필요\n              loaderLines.push(\n                `.${joinMethod}({ ${join.as}: \"${join.table}\" }, (j) => {`,\n                `j.on(Puri.rawString(\"${join.custom}\"));`,\n                \"})\",\n              );\n            } else {\n              loaderLines.push(\n                `.${joinMethod}({ ${join.as}: \"${join.table}\" }, \"${join.from}\", \"${join.to}\")`,\n              );\n            }\n          });\n\n          // 입체적 select 구조 생성 (refId 포함)\n          const selectObj = this.buildNestedSelectObject(loader.select);\n          selectObj.refId = `\"${through.table}.${through.fromCol}\"`;\n          loaderLines.push(\n            `.whereIn(\"${through.table}.${through.fromCol}\", fromIds)`,\n            `.select(${this.stringifyNestedSelectObject(selectObj)});`,\n          );\n        }\n\n        loaderLines.push(`},`);\n\n        // 중첩 loaders 처리\n        if (loader.loaders && loader.loaders.length > 0) {\n          loaderLines.push(\"loaders: [\", ...generateLoaderCode(loader.loaders), \"],\");\n        }\n\n        loaderLines.push(\"},\");\n      }\n\n      return loaderLines;\n    };\n\n    lines.push(...generateLoaderCode(loaders));\n    lines.push(`]`);\n\n    return lines.join(\"\\n\");\n  }\n\n  /*\n    subset SELECT/JOIN/LOADER 결과 리턴\n  */\n  getSubsetQuery(subsetKey: string): SubsetQuery {\n    const subset = this.subsets[subsetKey];\n\n    const result: SubsetQuery = this.resolveSubsetQuery(\"\", subset);\n    return result;\n  }\n\n  /*\n   */\n  resolveSubsetQuery(\n    prefix: string,\n    fields: string[],\n    isAlreadyOuterJoined: boolean = false,\n  ): SubsetQuery {\n    // prefix 치환 (prefix는 ToOneRelation이 복수로 붙은 경우 모두 __로 변경됨)\n    prefix = prefix.replace(/\\./g, \"__\");\n\n    // 서브셋을 1뎁스만 분리하여 그룹핑\n    const subsetGroup = group(fields, (field) => {\n      if (field.includes(\".\")) {\n        const [rel] = field.split(\".\");\n        return rel;\n      } else {\n        return \"\";\n      }\n    });\n\n    const result = Object.keys(subsetGroup).reduce(\n      (r, groupKey) => {\n        const fields = subsetGroup[groupKey];\n        assert(fields !== undefined, \"fields is undefined\");\n\n        // 현재 테이블 필드셋은 select, virtual에 추가하고 리턴\n        if (groupKey === \"\") {\n          const realFields = fields.filter((field) => !isVirtualProp(this.propsDict[field]));\n          const virtualFields = fields.filter((field) => isVirtualProp(this.propsDict[field]));\n\n          if (prefix === \"\") {\n            // 현재 테이블인 경우\n            r.select = r.select.concat(realFields.map((field) => `${this.table}.${field}`));\n            r.virtual = r.virtual.concat(virtualFields);\n          } else {\n            // 넘어온 테이블인 경우\n            r.select = r.select.concat(\n              realFields.map((field) => `${prefix}.${field} as ${prefix}__${field}`),\n            );\n          }\n\n          return r;\n        }\n\n        const relation = this.relations[groupKey];\n        if (relation === undefined) {\n          throw new Error(`존재하지 않는 relation 참조 ${groupKey}`);\n        }\n        const relEntity = EntityManager.get(relation.with);\n\n        if (isOneToOneRelationProp(relation) || isBelongsToOneRelationProp(relation)) {\n          // -One Relation: JOIN 으로 처리\n          const relFields = fields.map((field) => field.split(\".\").slice(1).join(\".\"));\n\n          // -One Relation에서 id 필드만 참조하는 경우 릴레이션 넘기지 않고 리턴\n          if (relFields.length === 1 && relFields[0] === \"id\") {\n            if (prefix === \"\") {\n              r.select = r.select.concat(`${this.table}.${groupKey}_id`);\n            } else {\n              r.select = r.select.concat(`${prefix}.${groupKey}_id as ${prefix}__${groupKey}_id`);\n            }\n            return r;\n          }\n\n          // innerOrOuter\n          const innerOrOuter = (() => {\n            if (isAlreadyOuterJoined) {\n              return \"outer\";\n            }\n\n            if (isOneToOneRelationProp(relation)) {\n              if (relation.hasJoinColumn === true && (relation.nullable ?? false) === false) {\n                return \"inner\";\n              } else {\n                return \"outer\";\n              }\n            } else {\n              if (relation.nullable) {\n                return \"outer\";\n              } else {\n                return \"inner\";\n              }\n            }\n          })();\n          const relSubsetQuery = relEntity.resolveSubsetQuery(\n            `${prefix !== \"\" ? `${prefix}.` : \"\"}${groupKey}`,\n            relFields,\n            innerOrOuter === \"outer\",\n          );\n          r.select = r.select.concat(relSubsetQuery.select);\n          r.virtual = r.virtual.concat(relSubsetQuery.virtual);\n\n          const joinAs = prefix === \"\" ? groupKey : `${prefix}__${groupKey}`;\n          const fromTable = prefix === \"\" ? this.table : prefix;\n\n          let joinClause:\n            | {\n                from: string;\n                to: string;\n              }\n            | {\n                custom: string;\n              };\n          if (relation.customJoinClause) {\n            joinClause = {\n              custom: relation.customJoinClause,\n            };\n          } else {\n            let from: string, to: string;\n            if (isOneToOneRelationProp(relation)) {\n              if (relation.hasJoinColumn) {\n                from = `${fromTable}.${relation.name}_id`;\n                to = `${joinAs}.id`;\n              } else {\n                from = `${fromTable}.id`;\n                to = `${joinAs}.${inflection.underscore(this.names.fs.replace(/-/g, \"_\"))}_id`;\n              }\n            } else {\n              from = `${fromTable}.${relation.name}_id`;\n              to = `${joinAs}.id`;\n            }\n            joinClause = {\n              from,\n              to,\n            };\n          }\n\n          r.joins.push({\n            as: joinAs,\n            join: innerOrOuter,\n            table: relEntity.table,\n            ...joinClause,\n          });\n\n          // BelongsToOne 밑에 HasMany가 붙은 경우\n          if (relSubsetQuery.loaders.length > 0) {\n            const convertedLoaders = relSubsetQuery.loaders.map((loader) => {\n              const newAs = [groupKey, loader.as].join(\"__\");\n              return {\n                as: newAs,\n                table: loader.table,\n                manyJoin: loader.manyJoin,\n                oneJoins: loader.oneJoins,\n                select: loader.select,\n                loaders: loader.loaders,\n              };\n            });\n\n            r.loaders = [...r.loaders, ...convertedLoaders];\n          }\n\n          r.joins = r.joins.concat(relSubsetQuery.joins);\n        } else if (isHasManyRelationProp(relation) || isManyToManyRelationProp(relation)) {\n          // -Many Relation: Loader 로 처리\n          const relFields = fields.map((field) => field.split(\".\").slice(1).join(\".\"));\n          const relSubsetQuery = relEntity.resolveSubsetQuery(\"\", relFields);\n\n          let manyJoin: SubsetQuery[\"loaders\"][number][\"manyJoin\"];\n          if (isHasManyRelationProp(relation)) {\n            const fromCol = relation?.fromColumn ?? \"id\";\n            manyJoin = {\n              fromTable: this.table,\n              fromCol,\n              idField: prefix === \"\" ? `${fromCol}` : `${prefix}__${fromCol}`,\n              toTable: relEntity.table,\n              toCol: relation.joinColumn,\n            };\n          } else if (isManyToManyRelationProp(relation)) {\n            manyJoin = {\n              fromTable: this.table,\n              fromCol: \"id\",\n              idField: prefix === \"\" ? `id` : `${prefix}__id`,\n              through: {\n                table: relation.joinTable,\n                fromCol: `${inflection.singularize(this.table)}_id`,\n                toCol: `${inflection.singularize(relEntity.table)}_id`,\n              },\n              toTable: relEntity.table,\n              toCol: \"id\",\n            };\n          } else {\n            throw new Error();\n          }\n\n          r.loaders.push({\n            as: groupKey,\n            table: relEntity.table,\n            manyJoin,\n            oneJoins: relSubsetQuery.joins,\n            select: relSubsetQuery.select,\n            loaders: relSubsetQuery.loaders,\n          });\n        }\n\n        return r;\n      },\n      {\n        select: [],\n        virtual: [],\n        joins: [],\n        loaders: [],\n      } as SubsetQuery,\n    );\n    return result;\n  }\n\n  /*\n    FieldExpr[] 을 EntityPropNode[] 로 변환\n  */\n  fieldExprsToPropNodes(fieldExprs: string[], entity: Entity = this): EntityPropNode[] {\n    const groups = fieldExprs.reduce(\n      (result, fieldExpr) => {\n        let key: string, value: string, elseExpr: string[];\n        if (fieldExpr.includes(\".\")) {\n          [key, ...elseExpr] = fieldExpr.split(\".\");\n          value = elseExpr.join(\".\");\n        } else {\n          key = \"\";\n          value = fieldExpr;\n        }\n        result[key] = (result[key] ?? []).concat(value);\n\n        return result;\n      },\n      {} as {\n        [k: string]: string[];\n      },\n    );\n\n    return Object.keys(groups).flatMap((key) => {\n      const group = groups[key];\n\n      // 일반 prop 처리\n      if (key === \"\") {\n        return group.map((propName) => {\n          // FIXME: 이거 나중에 없애야함\n          if (propName === \"말도안되는프롭명__이거왜타입처리가꼬여서이러지?\") {\n            return {\n              nodeType: \"plain\" as const,\n              prop: {\n                type: \"string\",\n                name: \"uuid\",\n                length: 128,\n              } as StringProp,\n              children: [],\n            } as EntityPropNode;\n          }\n\n          const prop = entity.props.find((p) => p.name === propName);\n          if (prop === undefined) {\n            console.log({ propName, groups });\n            throw new Error(`${entity.id} -- 잘못된 FieldExpr ${propName}`);\n          }\n          return {\n            nodeType: \"plain\" as const,\n            prop,\n            children: [],\n          };\n        });\n      }\n\n      // relation prop 처리\n      const prop = entity.propsDict[key];\n      if (!isRelationProp(prop)) {\n        throw new Error(`잘못된 FieldExpr ${key}.${group[0]}`);\n      }\n      const relEntity = EntityManager.get(prop.with);\n\n      // relation -One 에 id 필드 하나인 경우\n      if (isBelongsToOneRelationProp(prop) || isOneToOneRelationProp(prop)) {\n        if (group.length === 1 && (group[0] === \"id\" || group[0] === \"id?\")) {\n          // id 하나만 있는지 체크해서, 하나만 있으면 상위 prop으로 id를 리턴\n          const idProp = relEntity.propsDict.id;\n          return {\n            nodeType: \"plain\" as const,\n            prop: {\n              ...idProp,\n              name: `${key}_id`,\n              nullable: prop.nullable,\n            },\n            children: [],\n          };\n        }\n      }\n\n      // -One 그외의 경우 object로 리턴\n      // -Many의 경우 array로 리턴\n      // Recursive 로 뎁스 처리\n      const children = this.fieldExprsToPropNodes(group, relEntity);\n      const nodeType =\n        isBelongsToOneRelationProp(prop) || isOneToOneRelationProp(prop)\n          ? (\"object\" as const)\n          : (\"array\" as const);\n\n      return {\n        prop,\n        children,\n        nodeType,\n      };\n    });\n  }\n\n  getFieldExprs(prefix = \"\", maxDepth: number = 3, froms: string[] = []): string[] {\n    return this.props\n      .flatMap((prop) => {\n        const propName = [prefix, prop.name].filter((v) => v !== \"\").join(\".\");\n        if (propName === prefix) {\n          return null;\n        }\n        if (isRelationProp(prop)) {\n          if (maxDepth < 0) {\n            return null;\n          }\n          if (froms.includes(prop.with)) {\n            // 역방향 relation인 경우 제외\n            return null;\n          }\n          // 정방향 relation인 경우 recursive 콜\n          const relMd = EntityManager.get(prop.with);\n          return relMd.getFieldExprs(propName, maxDepth - 1, [...froms, this.id]);\n        }\n        return propName;\n      })\n      .filter((f) => f !== null) as string[];\n  }\n\n  getTableColumns(): { name: string; type: string }[] {\n    return this.props\n      .map((prop) => {\n        if (prop.type === \"relation\") {\n          if (\n            prop.relationType === \"BelongsToOne\" ||\n            (prop.relationType === \"OneToOne\" && prop.hasJoinColumn === true)\n          ) {\n            return { name: `${prop.name}_id`, type: \"int_unsigned\" };\n          } else {\n            return null;\n          }\n        }\n        return { name: prop.name, type: prop.type };\n      })\n      .filter(nonNullable);\n  }\n\n  /**\n   * Entity에 정의된 모든 vector 타입 컬럼 반환\n   */\n  getVectorColumns(): EntityProp[] {\n    return this.props.filter((p) => p.type === \"vector\");\n  }\n\n  /**\n   * 특정 vector 컬럼 반환\n   * @param columnName - 컬럼명 (생략 시 첫 번째 vector 컬럼)\n   */\n  getVectorColumn(columnName?: string): EntityProp | undefined {\n    const vectorProps = this.getVectorColumns();\n    if (columnName) {\n      return vectorProps.find((p) => p.name === columnName);\n    }\n    return vectorProps[0];\n  }\n\n  async registerModulePaths() {\n    const basePath = `${this.names.parentFs}`;\n\n    // base-scheme\n    EntityManager.setModulePath(`${this.id}BaseSchema`, `sonamu.generated`);\n\n    // subset\n    if (Object.keys(this.subsets).length > 0) {\n      EntityManager.setModulePath(`${this.id}SubsetKey`, `sonamu.generated`);\n      EntityManager.setModulePath(`${this.id}SubsetMapping`, `sonamu.generated`);\n      for (const subsetKey of Object.keys(this.subsets)) {\n        EntityManager.setModulePath(\n          `${this.id}Subset${subsetKey.toUpperCase()}`,\n          `sonamu.generated`,\n        );\n      }\n    }\n\n    // enums\n    for (const enumId of Object.keys(this.enumLabels)) {\n      EntityManager.setModulePath(enumId, `sonamu.generated`);\n    }\n\n    // types\n    const typesModulePath = `${basePath}/${this.names.parentFs}.types`;\n    const typesFilePath = path.join(\n      Sonamu.apiRootPath,\n      runtimePath(`dist/application/${typesModulePath}.js`),\n    );\n\n    if (await exists(typesFilePath)) {\n      const importedMembers = await importMembers<z.ZodTypeAny>(typesFilePath);\n      this.types = Object.fromEntries(\n        importedMembers.map(({ name, value }) => {\n          EntityManager.setModulePath(name, typesModulePath);\n          return [name, value];\n        }),\n      ) as { [name: string]: z.ZodTypeAny };\n    }\n  }\n\n  registerTableSpecs(): void {\n    // 조인 테이블 인덱스 제외 (컬럼 이름에 '.'이 포함된 경우)\n    const uniqueIndexes = this.indexes\n      .filter((idx) => idx.type === \"unique\")\n      .filter((idx) => idx.columns.every((col) => !col.name.includes(\".\")));\n\n    EntityManager.setTableSpec({\n      name: this.table,\n      uniqueIndexes,\n    });\n  }\n\n  toJson(): EntityJson {\n    return {\n      id: this.id,\n      parentId: this.parentId,\n      table: this.table,\n      title: this.title,\n      props: this.props,\n      indexes: this.indexes,\n      subsets: this.subsets,\n      enums: this.enumLabels,\n    };\n  }\n\n  async save(): Promise<void> {\n    // sort: subsets\n    const subsetRows = this.getSubsetRows();\n    this.subsets = Object.fromEntries(\n      Object.entries(this.subsets).map(([subsetKey]) => {\n        return [subsetKey, this.subsetRowsToSubsetFields(subsetRows, subsetKey)];\n      }),\n    );\n\n    // save\n    const jsonPath = path.join(\n      Sonamu.apiRootPath,\n      `src/application/${this.names.parentFs}/${this.names.fs}.entity.json`,\n    );\n    const json = this.toJson();\n    await writeFile(jsonPath, formatCode(JSON.stringify(json), \"json\", jsonPath));\n\n    // reload\n    await EntityManager.register(json);\n  }\n\n  getSubsetRows(\n    _subsets?: { [key: string]: string[] },\n    prefixes: string[] = [],\n  ): EntitySubsetRow[] {\n    if (prefixes.length > 10) {\n      return [];\n    }\n\n    const subsets = _subsets ?? this.subsets;\n    const subsetKeys = Object.keys(subsets);\n    const allFields = unique(subsetKeys.flatMap((key) => subsets[key]));\n\n    return this.props.map((prop) => {\n      if (\n        prop.type === \"relation\" &&\n        allFields.find((f) => f.startsWith(`${[...prefixes, prop.name].join(\".\")}.`))\n      ) {\n        const relEntity = EntityManager.get(prop.with);\n        const children = relEntity.getSubsetRows(subsets, [...prefixes, `${prop.name}`]);\n\n        return {\n          field: prop.name,\n          children,\n          relationEntity: prop.with,\n          prefixes,\n          isOpen: children.length > 0,\n          has: Object.fromEntries(\n            subsetKeys.map((subsetKey) => {\n              return [subsetKey, children.every((child) => child.has[subsetKey] === true)];\n            }),\n          ),\n        };\n      }\n\n      return {\n        field: prop.name,\n        children: [],\n        relationEntity: prop.type === \"relation\" ? prop.with : undefined,\n        prefixes,\n        has: Object.fromEntries(\n          subsetKeys.map((subsetKey) => {\n            const subsetFields = subsets[subsetKey];\n            const has = subsetFields.some((f) => {\n              const field = [...prefixes, prop.name].join(\".\");\n              return f === field || f.startsWith(`${field}.`);\n            });\n            return [subsetKey, has];\n          }),\n        ),\n      };\n    });\n  }\n\n  subsetRowsToSubsetFields(subsetRows: EntitySubsetRow[], subsetKey: string): string[] {\n    return subsetRows\n      .map((subsetRow) => {\n        if (subsetRow.children.length > 0) {\n          return this.subsetRowsToSubsetFields(subsetRow.children, subsetKey);\n        } else if (subsetRow.has[subsetKey]) {\n          return subsetRow.prefixes.concat(subsetRow.field).join(\".\");\n        } else {\n          return null;\n        }\n      })\n      .filter(nonNullable)\n      .flat();\n  }\n\n  async createProp(prop: EntityProp, at?: number): Promise<void> {\n    if (!at) {\n      this.props.push(prop);\n    } else {\n      this.props.splice(at, 0, prop);\n    }\n    await this.save();\n  }\n\n  analyzeSubsetField(subsetField: string): {\n    entityId: string;\n    propName: string;\n  }[] {\n    const arr = subsetField.split(\".\");\n\n    let entityId = this.id;\n    const result: {\n      entityId: string;\n      propName: string;\n    }[] = [];\n    for (let i = 0; i < arr.length; i++) {\n      const propName = arr[i];\n      result.push({\n        entityId,\n        propName,\n      });\n\n      const prop = EntityManager.get(entityId).props.find((p) => p.name === propName);\n      if (!prop) {\n        throw new Error(`${entityId}의 잘못된 서브셋키 ${subsetField}`);\n      }\n      if (isRelationProp(prop)) {\n        entityId = prop.with;\n      }\n    }\n    return result;\n  }\n\n  async modifyProp(newProp: EntityProp, at: number): Promise<void> {\n    // 이전 프롭 이름 저장\n    const oldName = this.props[at].name;\n\n    // 저장할 엔티티\n    const entities: Entity[] = [this];\n\n    // 이름이 바뀐 경우\n    if (oldName !== newProp.name) {\n      // 전체 엔티티에서 현재 수정된 프롭을 참조하고 있는 모든 서브셋필드 찾아서 수정\n      const allEntityIds = EntityManager.getAllIds();\n      for (const relEntityId of allEntityIds) {\n        const relEntity = EntityManager.get(relEntityId);\n        const relEntitySubsetKeys = Object.keys(relEntity.subsets);\n        for (const subsetKey of relEntitySubsetKeys) {\n          const subset = relEntity.subsets[subsetKey];\n\n          // 서브셋 필드를 순회하며, 엔티티-프롭 단위로 분석한 후 현재 엔티티-프롭과 일치하는 경우 수정 처리\n          const modifiedSubsetFields = subset.map((subsetField) => {\n            const analyzed = relEntity.analyzeSubsetField(subsetField);\n            const modified = analyzed.map((a) =>\n              a.propName === oldName && a.entityId === this.id\n                ? {\n                    ...a,\n                    propName: newProp.name,\n                  }\n                : a,\n            );\n            // 분석한 필드를 다시 서브셋 필드로 복구\n            return modified.map((a) => a.propName).join(\".\");\n          });\n\n          if (subset.join(\",\") !== modifiedSubsetFields.join(\",\")) {\n            relEntity.subsets[subsetKey] = modifiedSubsetFields;\n            entities.push(relEntity);\n          }\n        }\n      }\n    }\n\n    // 프롭 수정\n    this.props[at] = newProp;\n\n    await Promise.all(entities.map(async (entity) => entity.save()));\n  }\n\n  async delProp(at: number): Promise<void> {\n    // 이전 프롭 이름 저장\n    const oldName = this.props[at].name;\n\n    // 저장할 엔티티\n    const entities: Entity[] = [this];\n\n    // 전체 엔티티에서 현재 삭제된 프롭을 참조하고 있는 모든 서브셋필드 찾아서 제외\n    const allEntityIds = EntityManager.getAllIds();\n    for (const relEntityId of allEntityIds) {\n      const relEntity = EntityManager.get(relEntityId);\n      const relEntitySubsetKeys = Object.keys(relEntity.subsets);\n      for (const subsetKey of relEntitySubsetKeys) {\n        const subset = relEntity.subsets[subsetKey];\n        // 서브셋 필드를 순회하며, 엔티티-프롭 단위로 분석한 후 현재 엔티티-프롭과 일치하는 경우 이후의 필드를 제외\n        const modifiedSubsetFields = subset\n          .map((subsetField) => {\n            const analyzed = relEntity.analyzeSubsetField(subsetField);\n            if (analyzed.find((a) => a.propName === oldName && a.entityId === this.id)) {\n              return null;\n            } else {\n              return subsetField;\n            }\n          })\n          .filter(nonNullable);\n\n        if (subset.join(\",\") !== modifiedSubsetFields.join(\",\")) {\n          relEntity.subsets[subsetKey] = modifiedSubsetFields;\n          entities.push(relEntity);\n        }\n      }\n    }\n\n    // 현재 엔티티의 인덱스에서 제외\n    for (const index of EntityManager.get(this.id).indexes) {\n      index.columns = index.columns.filter((col) => col.name !== oldName);\n    }\n\n    // 프롭 삭제\n    this.props.splice(at, 1);\n\n    await Promise.all(entities.map(async (entity) => entity.save()));\n  }\n\n  getEntityIdFromSubsetField(subsetField: string): string {\n    if (subsetField.includes(\".\") === false) {\n      return this.id;\n    }\n\n    // 서브셋 필드의 마지막은 프롭이므로 제외\n    const arr = subsetField.split(\".\").slice(0, -1);\n\n    // 서브셋 필드를 내려가면서 마지막으로 relation된 엔티티를 찾음\n    const lastEntityId = arr.reduce((entityId, field) => {\n      const relProp = EntityManager.get(entityId).props.find((p) => p.name === field);\n      if (!relProp || relProp.type !== \"relation\") {\n        console.debug({ arr, thisId: this.id, entityId, field });\n        throw new Error(`잘못된 서브셋키 ${subsetField}`);\n      }\n      return relProp.with;\n    }, this.id);\n    return lastEntityId;\n  }\n\n  async moveProp(at: number, to: number): Promise<void> {\n    const prop = this.props[at];\n    const newProps = [...this.props];\n    newProps.splice(to, 0, prop);\n    newProps.splice(at < to ? at : at + 1, 1);\n    this.props = newProps;\n\n    await this.save();\n  }\n}\n"],"names":["assert","writeFile","inflection","path","group","unique","z","Sonamu","isBelongsToOneRelationProp","isEnumProp","isHasManyRelationProp","isManyToManyRelationProp","isOneToOneRelationProp","isRelationProp","isVirtualProp","importMembers","formatCode","exists","runtimePath","assertDefined","nonNullable","EntityManager","Entity","id","parentId","table","title","names","props","propsDict","relations","indexes","subsets","types","enums","enumLabels","underscore","pluralize","map","prop","includes","replace","Object","fromEntries","name","filter","entries","key","enumLabel","enum","keys","parentFs","dasherize","toLowerCase","fs","module","getPuriSubsetQuery","subsetKey","subset","subsetQuery","resolveSubsetQuery","lines","push","join","joins","joinMethod","as","custom","from","to","selectObj","buildNestedSelectObject","select","stringifyNestedSelectObject","selectItems","result","selectItem","match","column","alias","columnValue","trim","split","pop","parts","current","i","length","part","Error","slice","lastPart","obj","indent","withBraces","spaces","repeat","innerSpaces","value","getPuriLoaderQuery","loaders","generateLoaderCode","loaderLines","loader","toTable","toCol","through","manyJoin","idField","undefined","oneJoins","forEach","refId","fromCol","getSubsetQuery","prefix","fields","isAlreadyOuterJoined","subsetGroup","field","rel","reduce","r","groupKey","realFields","virtualFields","concat","virtual","relation","relEntity","get","with","relFields","innerOrOuter","hasJoinColumn","nullable","relSubsetQuery","joinAs","fromTable","joinClause","customJoinClause","convertedLoaders","newAs","fromColumn","joinColumn","joinTable","singularize","fieldExprsToPropNodes","fieldExprs","entity","groups","fieldExpr","elseExpr","flatMap","propName","nodeType","type","children","find","p","console","log","idProp","getFieldExprs","maxDepth","froms","v","relMd","f","getTableColumns","relationType","getVectorColumns","getVectorColumn","columnName","vectorProps","registerModulePaths","basePath","setModulePath","toUpperCase","enumId","typesModulePath","typesFilePath","apiRootPath","importedMembers","registerTableSpecs","uniqueIndexes","idx","columns","every","col","setTableSpec","toJson","save","subsetRows","getSubsetRows","subsetRowsToSubsetFields","jsonPath","json","JSON","stringify","register","_subsets","prefixes","subsetKeys","allFields","startsWith","relationEntity","isOpen","has","child","subsetFields","some","subsetRow","flat","createProp","at","splice","analyzeSubsetField","subsetField","arr","entityId","modifyProp","newProp","oldName","entities","allEntityIds","getAllIds","relEntityId","relEntitySubsetKeys","modifiedSubsetFields","analyzed","modified","a","Promise","all","delProp","index","getEntityIdFromSubsetField","lastEntityId","relProp","debug","thisId","moveProp","newProps"],"mappings":"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"}
|
|
883
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/entity/entity.ts"],"sourcesContent":["import assert from \"assert\";\nimport { writeFile } from \"fs/promises\";\nimport inflection from \"inflection\";\nimport path from \"path\";\nimport { group, unique } from \"radashi\";\nimport { z } from \"zod\";\nimport { Sonamu } from \"../api/sonamu\";\nimport {\n  type EntityIndex,\n  type EntityJson,\n  type EntityProp,\n  type EntityPropNode,\n  type EntitySubsetRow,\n  isBelongsToOneRelationProp,\n  isEnumProp,\n  isHasManyRelationProp,\n  isInternalSubsetField,\n  isManyToManyRelationProp,\n  isOneToOneRelationProp,\n  isRelationProp,\n  isVirtualCodeProp,\n  isVirtualProp,\n  normalizeSubsetField,\n  type RelationProp,\n  type SubsetField,\n  type SubsetQuery,\n} from \"../types/types\";\nimport { importMembers } from \"../utils/esm-utils\";\nimport { formatCode } from \"../utils/formatter\";\nimport { exists } from \"../utils/fs-utils\";\nimport { runtimePath } from \"../utils/path-utils\";\nimport { assertDefined, nonNullable } from \"../utils/utils\";\nimport { EntityManager } from \"./entity-manager\";\n\nexport class Entity {\n  id: string;\n  parentId?: string;\n  table: string;\n  title: string;\n  names: {\n    parentFs: string;\n    fs: string;\n    module: string;\n  };\n  props: EntityProp[];\n  propsDict: {\n    [key: string]: EntityProp;\n  };\n  relations: {\n    [key: string]: RelationProp;\n  };\n  indexes: EntityIndex[];\n  subsets: {\n    [key: string]: string[];\n  };\n  subsetsInternal: {\n    [key: string]: string[];\n  };\n  types: {\n    [name: string]: z.ZodTypeAny;\n  } = {};\n  enums: {\n    [enumId: string]: z.ZodEnum<Readonly<Record<string, string>>>;\n  } = {};\n  enumLabels: {\n    [enumId: string]: {\n      [key: string]: string;\n    };\n  } = {};\n\n  constructor({ id, parentId, table, title, props, indexes, subsets, enums }: EntityJson) {\n    // id\n    this.id = id;\n    this.parentId = parentId;\n    this.title = title ?? this.id;\n    this.table = table ?? inflection.underscore(inflection.pluralize(id));\n\n    // props\n    if (props) {\n      this.props = props.map((prop) => {\n        if (isEnumProp(prop)) {\n          if (prop.id.includes(\"$Model\")) {\n            prop.id = prop.id.replace(\"$Model\", id);\n          }\n        }\n        return prop;\n      });\n      this.propsDict = Object.fromEntries(\n        props.map((prop) => {\n          return [prop.name, prop];\n        }),\n      );\n\n      // relations\n      this.relations = Object.fromEntries(\n        props.filter((prop) => isRelationProp(prop)).map((prop) => [prop.name, prop]),\n      );\n    } else {\n      this.props = [];\n      this.propsDict = {};\n      this.relations = {};\n    }\n\n    // indexes\n    this.indexes = indexes ?? [];\n\n    // subsets: SubsetField[]를 파싱하여 subsets(일반)와 subsetsInternal(internal)로 분리\n    this.subsets = {};\n    this.subsetsInternal = {};\n    for (const [key, fields] of Object.entries(subsets ?? {})) {\n      this.subsets[key] = fields.filter((f) => !isInternalSubsetField(f)).map(normalizeSubsetField);\n      this.subsetsInternal[key] = fields.filter(isInternalSubsetField).map(normalizeSubsetField);\n    }\n\n    // enums\n    this.enumLabels = enums ?? {};\n    this.enums = Object.fromEntries(\n      Object.entries(this.enumLabels).map(([key, enumLabel]) => {\n        return [key, z.enum(Object.keys(enumLabel) as unknown as readonly [string, ...string[]])];\n      }),\n    );\n\n    // names\n    this.names = {\n      parentFs: inflection.dasherize(inflection.underscore(parentId ?? id)).toLowerCase(),\n      fs: inflection.dasherize(inflection.underscore(id)).toLowerCase(),\n      module: id,\n    };\n  }\n\n  /**\n   * 쿼리용 서브셋 필드를 반환합니다 (subsets + subsetsInternal 합침)\n   */\n  getSubsetFieldsForQuery(subsetKey: string): string[] {\n    return [...(this.subsets[subsetKey] ?? []), ...(this.subsetsInternal[subsetKey] ?? [])];\n  }\n\n  /**\n   * 주어진 이름(subsetKey)의 subset을 실제로 가져오는 Puri 코드 구현체 string을 반환합니다.\n   */\n  getPuriSubsetQuery(subsetKey: string): string {\n    const subset = this.getSubsetFieldsForQuery(subsetKey);\n    const subsetQuery = this.resolveSubsetQuery(\"\", subset);\n\n    const lines: string[] = [];\n\n    // from\n    lines.push(`return qbWrapper`);\n    lines.push(`.from(\"${this.table}\")`);\n\n    // join\n    for (const join of subsetQuery.joins) {\n      // join 메서드 결정: inner → join, outer → leftJoin\n      // FK nullable 여부는 leftJoin 타입 시그니처에서 자동으로 판단됨\n      const joinMethod = join.join === \"inner\" ? \"join\" : \"leftJoin\";\n\n      if (\"custom\" in join) {\n        // custom join clause는 raw 사용\n        lines.push(\n          `.${joinMethod}({ ${join.as}: \"${join.table}\" }, qbWrapper.knex.raw(\\`${join.custom}\\`))`,\n        );\n      } else {\n        lines.push(`.${joinMethod}({ ${join.as}: \"${join.table}\" }, \"${join.from}\", \"${join.to}\")`);\n      }\n    }\n\n    // select - 입체적 구조로 생성\n    const selectObj = this.buildNestedSelectObject(subsetQuery.select);\n    lines.push(`.select(${this.stringifyNestedSelectObject(selectObj)});`);\n\n    return lines.join(\"\\n\");\n  }\n\n  /**\n   * *.entity.json의 subset에 들어있는 필드 배열을 받아서,\n   * Puri의 SelectObject 타입으로 변환합니다.\n   *\n   * 예: [\"users.id\", \"parent.id\", \"parent.name\"]\n   *   → { id: \"users.id\", parent: { id: \"parent.id\", name: \"parent.name\" } }\n   *\n   * 언더바가 아닌 중첩 객체로 변환함에 유의하세요.\n   * 이렇게 중첩 객체로 변환하여 select에 넘겨주면 ParseSelectObject 타입이 join된 객체의 타입을 잘 잡아줄 수 있습니다.\n   * 즉, enhancer에서 row를 받았을 때 hydrate된 객체 자체의 nullity와 그 안쪽 필드의 nullity가 fk nullable 여부에 따라 잘 추론됩니다.\n   */\n  private buildNestedSelectObject(\n    selectItems: string[],\n    // biome-ignore lint/suspicious/noExplicitAny: 반환 오브젝트의 값은 string일 수도 있고 또다른 오브젝트일 수도 있는데, 이를 재귀 타입으로 나타낼 수 없어 any로 처리합니다.\n  ): Record<string, any> {\n    const result: ReturnType<typeof this.buildNestedSelectObject> = {};\n\n    for (const selectItem of selectItems) {\n      // \"users.id\" 또는 \"users.id as user__id\" 형태 파싱\n      const match = selectItem.match(/^(.+?)(?: as (.+))?$/);\n      if (!match) continue;\n\n      const [, column, alias] = match;\n      const columnValue = `\"${column.trim()}\"`;\n\n      if (!alias || !alias.includes(\"__\")) {\n        // alias가 없거나 __를 포함하지 않으면 최상위 필드\n        const key = alias ?? assertDefined(column.split(\".\").pop());\n        result[key] = columnValue;\n      } else {\n        // alias가 __를 포함하면 입체 구조로 그룹화\n        const parts = alias.split(\"__\");\n        let current = result;\n\n        // 마지막 파트 전까지 중첩 객체 생성\n        for (let i = 0; i < parts.length - 1; i++) {\n          const part = parts[i];\n          if (part in current) {\n            if (typeof current[part] === \"string\") {\n              // 입력이 [\"user\", \"user__id\"] 같은 경우!\n              // 애초에 말도 안 되지만 안전하게 예외를 던집니다.\n              throw new Error(\n                `Conflict detected in select items: parent path \"${parts.slice(0, i + 1).join(\"__\")}\" is already set as a field, cannot nest \"${alias}\" under it.`,\n              );\n            }\n          } else {\n            current[part] = {};\n          }\n          current = current[part];\n        }\n\n        // 마지막 파트에 값 설정\n        const lastPart = parts[parts.length - 1];\n        current[lastPart] = columnValue;\n      }\n    }\n\n    return result;\n  }\n\n  /**\n   * JSON.stringify와 유사한 일을 합니다.\n   * 다만 주어진 객체를 JSON이 아닌 TypeScript 객체 리터럴 스트링으로 만들어줍니다.\n   * key에 따옴표가 없어요.\n   * 출력 예시:\n   * ```typescript\n   * {\n   *   id: \"users.id\",\n   *   parent: {\n   *     id: \"parent.id\",\n   *     name: \"parent.name\",\n   *   },\n   * }\n   * ```\n   * @param obj 변환할 객체\n   * @param indent 들여쓰기 레벨\n   * @param withBraces true면 중괄호 포함, false면 내용만 반환\n   */\n  private stringifyNestedSelectObject(\n    // biome-ignore lint/suspicious/noExplicitAny: 중첩 오브젝트의 값은 string일 수도 있고 또다른 오브젝트일 수도 있는데, 이를 재귀 타입으로 나타낼 수 없어 any로 처리합니다.\n    obj: Record<string, any>,\n    indent: number = 0,\n    withBraces: boolean = true,\n  ): string {\n    const spaces = \"  \".repeat(indent);\n    const innerSpaces = \"  \".repeat(indent + 1);\n\n    const entries = Object.entries(obj);\n    if (entries.length === 0) return withBraces ? \"{}\" : \"\";\n\n    const lines = entries.map(([key, value]) => {\n      if (typeof value === \"string\") {\n        // 컬럼 경로 (이미 따옴표 포함)\n        return `${innerSpaces}${key}: ${value},`;\n      } else {\n        // 중첩 객체 (항상 중괄호 포함)\n        return `${innerSpaces}${key}: ${this.stringifyNestedSelectObject(value, indent + 1, true)},`;\n      }\n    });\n\n    if (withBraces) {\n      return `{\\n${lines.join(\"\\n\")}\\n${spaces}}`;\n    } else {\n      // 중괄호 없이 내용만 반환 (앞뒤 개행 제외)\n      return lines.join(\"\\n\");\n    }\n  }\n\n  getPuriLoaderQuery(subsetKey: string): string {\n    const subset = this.getSubsetFieldsForQuery(subsetKey);\n    const { loaders } = this.resolveSubsetQuery(\"\", subset);\n\n    const lines: string[] = [`[`];\n\n    // 재귀적으로 loader 생성하는 헬퍼 함수\n    const generateLoaderCode = (loaders: SubsetQuery[\"loaders\"]): string[] => {\n      const loaderLines: string[] = [];\n\n      for (const loader of loaders) {\n        const { toTable, toCol, through } = loader.manyJoin;\n        loaderLines.push(\n          \"{\",\n          `as: \"${loader.as}\",`,\n          `refId: \"${loader.manyJoin.idField}\",`,\n          `qb: (qbWrapper: PuriWrapper<DatabaseSchemaExtend>, fromIds: number[]) => {`,\n        );\n\n        if (through === undefined) {\n          // HasMany\n          loaderLines.push(\n            //\n            \"return qbWrapper\",\n            `.from(\"${toTable}\")`,\n          );\n\n          loader.oneJoins.forEach((join: SubsetQuery[\"joins\"][number]) => {\n            // FK nullable 여부는 leftJoin 타입 시그니처에서 자동으로 판단됨\n            const joinMethod = join.join === \"inner\" ? \"join\" : \"leftJoin\";\n            if (\"custom\" in join) {\n              // FIXME: 검증 필요\n              loaderLines.push(\n                `.${joinMethod}({ ${join.as}: \"${join.table}\" }, (j) => {`,\n                `j.on(Puri.rawString(\"${join.custom}\"));`,\n                \"})\",\n              );\n            } else {\n              loaderLines.push(\n                `.${joinMethod}({ ${join.as}: \"${join.table}\" }, \"${join.from}\", \"${join.to}\")`,\n              );\n            }\n          });\n\n          // 입체적 select 구조 생성 (refId 포함)\n          const selectObj = this.buildNestedSelectObject(loader.select);\n          selectObj.refId = `\"${toTable}.${toCol}\"`;\n          loaderLines.push(\n            `.whereIn(\"${toTable}.${toCol}\", fromIds)`,\n            `.select(${this.stringifyNestedSelectObject(selectObj)});`,\n          );\n        } else {\n          // ManyToMany\n          loaderLines.push(\n            \"return qbWrapper\",\n            `.from(\"${through.table}\")`,\n            `.join(\"${toTable}\", \"${through.table}.${through.toCol}\", \"${toTable}.${toCol}\")`,\n          );\n\n          loader.oneJoins.forEach((join: SubsetQuery[\"joins\"][number]) => {\n            // FK nullable 여부는 leftJoin 타입 시그니처에서 자동으로 판단됨\n            const joinMethod = join.join === \"inner\" ? \"join\" : \"leftJoin\";\n            if (\"custom\" in join) {\n              // FIXME: 검증 필요\n              loaderLines.push(\n                `.${joinMethod}({ ${join.as}: \"${join.table}\" }, (j) => {`,\n                `j.on(Puri.rawString(\"${join.custom}\"));`,\n                \"})\",\n              );\n            } else {\n              loaderLines.push(\n                `.${joinMethod}({ ${join.as}: \"${join.table}\" }, \"${join.from}\", \"${join.to}\")`,\n              );\n            }\n          });\n\n          // 입체적 select 구조 생성 (refId 포함)\n          const selectObj = this.buildNestedSelectObject(loader.select);\n          selectObj.refId = `\"${through.table}.${through.fromCol}\"`;\n          loaderLines.push(\n            `.whereIn(\"${through.table}.${through.fromCol}\", fromIds)`,\n            `.select(${this.stringifyNestedSelectObject(selectObj)});`,\n          );\n        }\n\n        loaderLines.push(`},`);\n\n        // 중첩 loaders 처리\n        if (loader.loaders && loader.loaders.length > 0) {\n          loaderLines.push(\"loaders: [\", ...generateLoaderCode(loader.loaders), \"],\");\n        }\n\n        loaderLines.push(\"},\");\n      }\n\n      return loaderLines;\n    };\n\n    lines.push(...generateLoaderCode(loaders));\n    lines.push(`]`);\n\n    return lines.join(\"\\n\");\n  }\n\n  /*\n    subset SELECT/JOIN/LOADER 결과 리턴\n  */\n  getSubsetQuery(subsetKey: string): SubsetQuery {\n    const subset = this.getSubsetFieldsForQuery(subsetKey);\n\n    const result: SubsetQuery = this.resolveSubsetQuery(\"\", subset);\n    return result;\n  }\n\n  /*\n   */\n  resolveSubsetQuery(\n    prefix: string,\n    fields: string[],\n    isAlreadyOuterJoined: boolean = false,\n  ): SubsetQuery {\n    // prefix 치환 (prefix는 ToOneRelation이 복수로 붙은 경우 모두 __로 변경됨)\n    prefix = prefix.replace(/\\./g, \"__\");\n\n    // 서브셋을 1뎁스만 분리하여 그룹핑\n    const subsetGroup = group(fields, (field) => {\n      if (field.includes(\".\")) {\n        const [rel] = field.split(\".\");\n        return rel;\n      } else {\n        return \"\";\n      }\n    });\n\n    const result = Object.keys(subsetGroup).reduce(\n      (r, groupKey) => {\n        const fields = subsetGroup[groupKey];\n        assert(fields !== undefined, \"fields is undefined\");\n\n        // 현재 테이블 필드셋은 select, virtual에 추가하고 리턴\n        if (groupKey === \"\") {\n          const realFields = fields.filter((field) => !isVirtualProp(this.propsDict[field]));\n          // virtualType: \"code\" (또는 undefined)인 virtual prop만 r.virtual에 추가\n          // virtualType: \"query\"인 경우 사용자가 appendSelect로 직접 추가하므로 제외\n          const virtualCodeFields = fields.filter((field) =>\n            isVirtualCodeProp(this.propsDict[field]),\n          );\n\n          if (prefix === \"\") {\n            // 현재 테이블인 경우\n            r.select = r.select.concat(realFields.map((field) => `${this.table}.${field}`));\n            r.virtual = r.virtual.concat(virtualCodeFields);\n          } else {\n            // 넘어온 테이블인 경우\n            r.select = r.select.concat(\n              realFields.map((field) => `${prefix}.${field} as ${prefix}__${field}`),\n            );\n          }\n\n          return r;\n        }\n\n        const relation = this.relations[groupKey];\n        if (relation === undefined) {\n          throw new Error(`존재하지 않는 relation 참조 ${groupKey}`);\n        }\n        const relEntity = EntityManager.get(relation.with);\n\n        if (isOneToOneRelationProp(relation) || isBelongsToOneRelationProp(relation)) {\n          // -One Relation: JOIN 으로 처리\n          const relFields = fields.map((field) => field.split(\".\").slice(1).join(\".\"));\n\n          // -One Relation에서 id 필드만 참조하는 경우 릴레이션 넘기지 않고 리턴\n          if (relFields.length === 1 && relFields[0] === \"id\") {\n            if (prefix === \"\") {\n              r.select = r.select.concat(`${this.table}.${groupKey}_id`);\n            } else {\n              r.select = r.select.concat(`${prefix}.${groupKey}_id as ${prefix}__${groupKey}_id`);\n            }\n            return r;\n          }\n\n          // innerOrOuter\n          const innerOrOuter = (() => {\n            if (isAlreadyOuterJoined) {\n              return \"outer\";\n            }\n\n            if (isOneToOneRelationProp(relation)) {\n              if (relation.hasJoinColumn === true && (relation.nullable ?? false) === false) {\n                return \"inner\";\n              } else {\n                return \"outer\";\n              }\n            } else {\n              if (relation.nullable) {\n                return \"outer\";\n              } else {\n                return \"inner\";\n              }\n            }\n          })();\n          const relSubsetQuery = relEntity.resolveSubsetQuery(\n            `${prefix !== \"\" ? `${prefix}.` : \"\"}${groupKey}`,\n            relFields,\n            innerOrOuter === \"outer\",\n          );\n          r.select = r.select.concat(relSubsetQuery.select);\n          r.virtual = r.virtual.concat(relSubsetQuery.virtual);\n\n          const joinAs = prefix === \"\" ? groupKey : `${prefix}__${groupKey}`;\n          const fromTable = prefix === \"\" ? this.table : prefix;\n\n          let joinClause:\n            | {\n                from: string;\n                to: string;\n              }\n            | {\n                custom: string;\n              };\n          if (relation.customJoinClause) {\n            joinClause = {\n              custom: relation.customJoinClause,\n            };\n          } else {\n            let from: string, to: string;\n            if (isOneToOneRelationProp(relation)) {\n              if (relation.hasJoinColumn) {\n                from = `${fromTable}.${relation.name}_id`;\n                to = `${joinAs}.id`;\n              } else {\n                from = `${fromTable}.id`;\n                to = `${joinAs}.${inflection.underscore(this.names.fs.replace(/-/g, \"_\"))}_id`;\n              }\n            } else {\n              from = `${fromTable}.${relation.name}_id`;\n              to = `${joinAs}.id`;\n            }\n            joinClause = {\n              from,\n              to,\n            };\n          }\n\n          r.joins.push({\n            as: joinAs,\n            join: innerOrOuter,\n            table: relEntity.table,\n            ...joinClause,\n          });\n\n          // BelongsToOne 밑에 HasMany가 붙은 경우\n          if (relSubsetQuery.loaders.length > 0) {\n            const convertedLoaders = relSubsetQuery.loaders.map((loader) => {\n              const newAs = [groupKey, loader.as].join(\"__\");\n              return {\n                as: newAs,\n                table: loader.table,\n                manyJoin: loader.manyJoin,\n                oneJoins: loader.oneJoins,\n                select: loader.select,\n                loaders: loader.loaders,\n              };\n            });\n\n            r.loaders = [...r.loaders, ...convertedLoaders];\n          }\n\n          r.joins = r.joins.concat(relSubsetQuery.joins);\n        } else if (isHasManyRelationProp(relation) || isManyToManyRelationProp(relation)) {\n          // -Many Relation: Loader 로 처리\n          const relFields = fields.map((field) => field.split(\".\").slice(1).join(\".\"));\n          const relSubsetQuery = relEntity.resolveSubsetQuery(\"\", relFields);\n\n          let manyJoin: SubsetQuery[\"loaders\"][number][\"manyJoin\"];\n          if (isHasManyRelationProp(relation)) {\n            const fromCol = relation?.fromColumn ?? \"id\";\n            manyJoin = {\n              fromTable: this.table,\n              fromCol,\n              idField: prefix === \"\" ? `${fromCol}` : `${prefix}__${fromCol}`,\n              toTable: relEntity.table,\n              toCol: relation.joinColumn,\n            };\n          } else if (isManyToManyRelationProp(relation)) {\n            manyJoin = {\n              fromTable: this.table,\n              fromCol: \"id\",\n              idField: prefix === \"\" ? `id` : `${prefix}__id`,\n              through: {\n                table: relation.joinTable,\n                fromCol: `${inflection.singularize(this.table)}_id`,\n                toCol: `${inflection.singularize(relEntity.table)}_id`,\n              },\n              toTable: relEntity.table,\n              toCol: \"id\",\n            };\n          } else {\n            throw new Error();\n          }\n\n          r.loaders.push({\n            as: groupKey,\n            table: relEntity.table,\n            manyJoin,\n            oneJoins: relSubsetQuery.joins,\n            select: relSubsetQuery.select,\n            loaders: relSubsetQuery.loaders,\n          });\n        }\n\n        return r;\n      },\n      {\n        select: [],\n        virtual: [],\n        joins: [],\n        loaders: [],\n      } as SubsetQuery,\n    );\n    return result;\n  }\n\n  /*\n    FieldExpr[] 을 EntityPropNode[] 로 변환\n  */\n  fieldExprsToPropNodes(fieldExprs: string[], entity: Entity = this): EntityPropNode[] {\n    const groups = fieldExprs.reduce(\n      (result, fieldExpr) => {\n        let key: string, value: string, elseExpr: string[];\n        if (fieldExpr.includes(\".\")) {\n          [key, ...elseExpr] = fieldExpr.split(\".\");\n          value = elseExpr.join(\".\");\n        } else {\n          key = \"\";\n          value = fieldExpr;\n        }\n        result[key] = (result[key] ?? []).concat(value);\n\n        return result;\n      },\n      {} as {\n        [k: string]: string[];\n      },\n    );\n\n    return Object.keys(groups).flatMap<EntityPropNode, EntityPropNode[]>((key) => {\n      const group = groups[key];\n\n      // 일반 prop 처리\n      if (key === \"\") {\n        return group.map((propName) => {\n          const prop = entity.props.find((p) => p.name === propName);\n          if (prop === undefined) {\n            console.log({ propName, groups });\n            throw new Error(`${entity.id} -- 잘못된 FieldExpr ${propName}`);\n          }\n          return {\n            nodeType: \"plain\" as const,\n            prop,\n          };\n        });\n      }\n\n      // relation prop 처리\n      const prop = entity.propsDict[key];\n      if (!isRelationProp(prop)) {\n        throw new Error(`잘못된 FieldExpr ${key}.${group[0]}`);\n      }\n      const relEntity = EntityManager.get(prop.with);\n\n      // relation -One 에 id 필드 하나인 경우\n      if (isBelongsToOneRelationProp(prop) || isOneToOneRelationProp(prop)) {\n        if (group.length === 1 && (group[0] === \"id\" || group[0] === \"id?\")) {\n          // id 하나만 있는지 체크해서, 하나만 있으면 상위 prop으로 id를 리턴\n          const idProp = relEntity.propsDict.id;\n          return {\n            nodeType: \"plain\" as const,\n            prop: {\n              ...idProp,\n              name: `${key}_id`,\n              nullable: prop.nullable,\n            },\n          };\n        }\n      }\n\n      // -One 그외의 경우 object로 리턴\n      // -Many의 경우 array로 리턴\n      // Recursive 로 뎁스 처리\n      const children = this.fieldExprsToPropNodes(group, relEntity);\n      const nodeType =\n        isBelongsToOneRelationProp(prop) || isOneToOneRelationProp(prop)\n          ? (\"object\" as const)\n          : (\"array\" as const);\n\n      return {\n        nodeType,\n        prop,\n        children,\n      };\n    });\n  }\n\n  getFieldExprs(prefix = \"\", maxDepth: number = 3, froms: string[] = []): string[] {\n    return this.props\n      .flatMap((prop) => {\n        const propName = [prefix, prop.name].filter((v) => v !== \"\").join(\".\");\n        if (propName === prefix) {\n          return null;\n        }\n        if (isRelationProp(prop)) {\n          if (maxDepth < 0) {\n            return null;\n          }\n          if (froms.includes(prop.with)) {\n            // 역방향 relation인 경우 제외\n            return null;\n          }\n          // 정방향 relation인 경우 recursive 콜\n          const relMd = EntityManager.get(prop.with);\n          return relMd.getFieldExprs(propName, maxDepth - 1, [...froms, this.id]);\n        }\n        return propName;\n      })\n      .filter((f) => f !== null) as string[];\n  }\n\n  getTableColumns(): { name: string; type: string }[] {\n    return this.props\n      .map((prop) => {\n        if (prop.type === \"relation\") {\n          if (\n            prop.relationType === \"BelongsToOne\" ||\n            (prop.relationType === \"OneToOne\" && prop.hasJoinColumn === true)\n          ) {\n            return { name: `${prop.name}_id`, type: \"int_unsigned\" };\n          } else {\n            return null;\n          }\n        }\n        return { name: prop.name, type: prop.type };\n      })\n      .filter(nonNullable);\n  }\n\n  /**\n   * Entity에 정의된 모든 vector 타입 컬럼 반환\n   */\n  getVectorColumns(): EntityProp[] {\n    return this.props.filter((p) => p.type === \"vector\");\n  }\n\n  /**\n   * 특정 vector 컬럼 반환\n   * @param columnName - 컬럼명 (생략 시 첫 번째 vector 컬럼)\n   */\n  getVectorColumn(columnName?: string): EntityProp | undefined {\n    const vectorProps = this.getVectorColumns();\n    if (columnName) {\n      return vectorProps.find((p) => p.name === columnName);\n    }\n    return vectorProps[0];\n  }\n\n  async registerModulePaths() {\n    const basePath = `${this.names.parentFs}`;\n\n    // base-scheme\n    EntityManager.setModulePath(`${this.id}BaseSchema`, `sonamu.generated`);\n\n    // subset\n    if (Object.keys(this.subsets).length > 0) {\n      EntityManager.setModulePath(`${this.id}SubsetKey`, `sonamu.generated`);\n      EntityManager.setModulePath(`${this.id}SubsetMapping`, `sonamu.generated`);\n      for (const subsetKey of Object.keys(this.subsets)) {\n        EntityManager.setModulePath(\n          `${this.id}Subset${subsetKey.toUpperCase()}`,\n          `sonamu.generated`,\n        );\n      }\n    }\n\n    // enums\n    for (const enumId of Object.keys(this.enumLabels)) {\n      EntityManager.setModulePath(enumId, `sonamu.generated`);\n    }\n\n    // types\n    const typesModulePath = `${basePath}/${this.names.parentFs}.types`;\n    const typesFilePath = path.join(\n      Sonamu.apiRootPath,\n      runtimePath(`dist/application/${typesModulePath}.js`),\n    );\n\n    if (await exists(typesFilePath)) {\n      const importedMembers = await importMembers<z.ZodTypeAny>(typesFilePath);\n      this.types = Object.fromEntries(\n        importedMembers.map(({ name, value }) => {\n          EntityManager.setModulePath(name, typesModulePath);\n          return [name, value];\n        }),\n      ) as { [name: string]: z.ZodTypeAny };\n    }\n  }\n\n  registerTableSpecs(): void {\n    // 조인 테이블 인덱스 제외 (컬럼 이름에 '.'이 포함된 경우)\n    const uniqueIndexes = this.indexes\n      .filter((idx) => idx.type === \"unique\")\n      .filter((idx) => idx.columns.every((col) => !col.name.includes(\".\")));\n\n    EntityManager.setTableSpec({\n      name: this.table,\n      uniqueIndexes,\n    });\n  }\n\n  toJson(): EntityJson {\n    // subsets와 subsetsInternal을 SubsetField[] 형태로 복원\n    const subsets: { [key: string]: SubsetField[] } = {};\n    for (const key of Object.keys(this.subsets)) {\n      const normalFields: SubsetField[] = this.subsets[key];\n      const internalFields: SubsetField[] = (this.subsetsInternal[key] ?? []).map((field) => ({\n        field,\n        internal: true,\n      }));\n      subsets[key] = [...normalFields, ...internalFields];\n    }\n\n    return {\n      id: this.id,\n      parentId: this.parentId,\n      table: this.table,\n      title: this.title,\n      props: this.props,\n      indexes: this.indexes,\n      subsets,\n      enums: this.enumLabels,\n    };\n  }\n\n  async save(): Promise<void> {\n    // sort: subsets\n    const subsetRows = this.getSubsetRows();\n    this.subsets = Object.fromEntries(\n      Object.entries(this.subsets).map(([subsetKey]) => {\n        return [subsetKey, this.subsetRowsToSubsetFields(subsetRows, subsetKey, false)];\n      }),\n    );\n    this.subsetsInternal = Object.fromEntries(\n      Object.entries(this.subsetsInternal).map(([subsetKey]) => {\n        return [subsetKey, this.subsetRowsToSubsetFields(subsetRows, subsetKey, true)];\n      }),\n    );\n\n    // save\n    const jsonPath = path.join(\n      Sonamu.apiRootPath,\n      `src/application/${this.names.parentFs}/${this.names.fs}.entity.json`,\n    );\n    const json = this.toJson();\n    await writeFile(jsonPath, formatCode(JSON.stringify(json), \"json\", jsonPath));\n\n    // reload\n    await EntityManager.register(json);\n  }\n\n  getSubsetRows(\n    _subsets?: { [key: string]: string[] },\n    _subsetsInternal?: { [key: string]: string[] },\n    prefixes: string[] = [],\n  ): EntitySubsetRow[] {\n    if (prefixes.length > 10) {\n      return [];\n    }\n\n    const subsets = _subsets ?? this.subsets;\n    const subsetsInternal = _subsetsInternal ?? this.subsetsInternal;\n    const subsetKeys = Object.keys(subsets);\n    const allFields = unique(subsetKeys.flatMap((key) => subsets[key]));\n    // internal 필드도 allFields에 포함 (relation 탐색용)\n    const allInternalFields = unique(subsetKeys.flatMap((key) => subsetsInternal[key] ?? []));\n    const combinedFields = unique([...allFields, ...allInternalFields]);\n\n    return this.props.map((prop) => {\n      if (\n        prop.type === \"relation\" &&\n        combinedFields.find((f) => f.startsWith(`${[...prefixes, prop.name].join(\".\")}.`))\n      ) {\n        const relEntity = EntityManager.get(prop.with);\n        const children = relEntity.getSubsetRows(subsets, subsetsInternal, [\n          ...prefixes,\n          `${prop.name}`,\n        ]);\n\n        return {\n          field: prop.name,\n          children,\n          relationEntity: prop.with,\n          prefixes,\n          isOpen: children.length > 0,\n          has: Object.fromEntries(\n            subsetKeys.map((subsetKey) => {\n              return [subsetKey, children.every((child) => child.has[subsetKey] === true)];\n            }),\n          ),\n          isInternal: Object.fromEntries(\n            subsetKeys.map((subsetKey) => {\n              return [subsetKey, children.every((child) => child.isInternal[subsetKey] === true)];\n            }),\n          ),\n        };\n      }\n\n      const field = [...prefixes, prop.name].join(\".\");\n      return {\n        field: prop.name,\n        children: [],\n        relationEntity: prop.type === \"relation\" ? prop.with : undefined,\n        prefixes,\n        has: Object.fromEntries(\n          subsetKeys.map((subsetKey) => {\n            const subsetFields = subsets[subsetKey];\n            const has = subsetFields.some((f) => {\n              return f === field || f.startsWith(`${field}.`);\n            });\n            return [subsetKey, has];\n          }),\n        ),\n        isInternal: Object.fromEntries(\n          subsetKeys.map((subsetKey) => {\n            const internalFields = subsetsInternal[subsetKey] ?? [];\n            const isInternal = internalFields.some((f) => {\n              return f === field || f.startsWith(`${field}.`);\n            });\n            return [subsetKey, isInternal];\n          }),\n        ),\n      };\n    });\n  }\n\n  subsetRowsToSubsetFields(\n    subsetRows: EntitySubsetRow[],\n    subsetKey: string,\n    internal: boolean = false,\n  ): string[] {\n    const hasKey = internal ? \"isInternal\" : \"has\";\n    return subsetRows\n      .map((subsetRow) => {\n        if (subsetRow.children.length > 0) {\n          return this.subsetRowsToSubsetFields(subsetRow.children, subsetKey, internal);\n        } else if (subsetRow[hasKey][subsetKey]) {\n          return subsetRow.prefixes.concat(subsetRow.field).join(\".\");\n        } else {\n          return null;\n        }\n      })\n      .filter(nonNullable)\n      .flat();\n  }\n\n  async createProp(prop: EntityProp, at?: number): Promise<void> {\n    if (!at) {\n      this.props.push(prop);\n    } else {\n      this.props.splice(at, 0, prop);\n    }\n    await this.save();\n  }\n\n  analyzeSubsetField(subsetField: string): {\n    entityId: string;\n    propName: string;\n  }[] {\n    const arr = subsetField.split(\".\");\n\n    let entityId = this.id;\n    const result: {\n      entityId: string;\n      propName: string;\n    }[] = [];\n    for (let i = 0; i < arr.length; i++) {\n      const propName = arr[i];\n      result.push({\n        entityId,\n        propName,\n      });\n\n      const prop = EntityManager.get(entityId).props.find((p) => p.name === propName);\n      if (!prop) {\n        throw new Error(`${entityId}의 잘못된 서브셋키 ${subsetField}`);\n      }\n      if (isRelationProp(prop)) {\n        entityId = prop.with;\n      }\n    }\n    return result;\n  }\n\n  async modifyProp(newProp: EntityProp, at: number): Promise<void> {\n    // 이전 프롭 이름 저장\n    const oldName = this.props[at].name;\n\n    // 저장할 엔티티\n    const entities: Entity[] = [this];\n\n    // 이름이 바뀐 경우\n    if (oldName !== newProp.name) {\n      // 전체 엔티티에서 현재 수정된 프롭을 참조하고 있는 모든 서브셋필드 찾아서 수정\n      const allEntityIds = EntityManager.getAllIds();\n      for (const relEntityId of allEntityIds) {\n        const relEntity = EntityManager.get(relEntityId);\n        const relEntitySubsetKeys = Object.keys(relEntity.subsets);\n        for (const subsetKey of relEntitySubsetKeys) {\n          const subset = relEntity.subsets[subsetKey];\n\n          // 서브셋 필드를 순회하며, 엔티티-프롭 단위로 분석한 후 현재 엔티티-프롭과 일치하는 경우 수정 처리\n          const modifiedSubsetFields = subset.map((subsetField) => {\n            const analyzed = relEntity.analyzeSubsetField(subsetField);\n            const modified = analyzed.map((a) =>\n              a.propName === oldName && a.entityId === this.id\n                ? {\n                    ...a,\n                    propName: newProp.name,\n                  }\n                : a,\n            );\n            // 분석한 필드를 다시 서브셋 필드로 복구\n            return modified.map((a) => a.propName).join(\".\");\n          });\n\n          if (subset.join(\",\") !== modifiedSubsetFields.join(\",\")) {\n            relEntity.subsets[subsetKey] = modifiedSubsetFields;\n            entities.push(relEntity);\n          }\n        }\n      }\n    }\n\n    // 프롭 수정\n    this.props[at] = newProp;\n\n    await Promise.all(entities.map(async (entity) => entity.save()));\n  }\n\n  async delProp(at: number): Promise<void> {\n    // 이전 프롭 이름 저장\n    const oldName = this.props[at].name;\n\n    // 저장할 엔티티\n    const entities: Entity[] = [this];\n\n    // 전체 엔티티에서 현재 삭제된 프롭을 참조하고 있는 모든 서브셋필드 찾아서 제외\n    const allEntityIds = EntityManager.getAllIds();\n    for (const relEntityId of allEntityIds) {\n      const relEntity = EntityManager.get(relEntityId);\n      const relEntitySubsetKeys = Object.keys(relEntity.subsets);\n      for (const subsetKey of relEntitySubsetKeys) {\n        const subset = relEntity.subsets[subsetKey];\n        // 서브셋 필드를 순회하며, 엔티티-프롭 단위로 분석한 후 현재 엔티티-프롭과 일치하는 경우 이후의 필드를 제외\n        const modifiedSubsetFields = subset\n          .map((subsetField) => {\n            const analyzed = relEntity.analyzeSubsetField(subsetField);\n            if (analyzed.find((a) => a.propName === oldName && a.entityId === this.id)) {\n              return null;\n            } else {\n              return subsetField;\n            }\n          })\n          .filter(nonNullable);\n\n        if (subset.join(\",\") !== modifiedSubsetFields.join(\",\")) {\n          relEntity.subsets[subsetKey] = modifiedSubsetFields;\n          entities.push(relEntity);\n        }\n      }\n    }\n\n    // 현재 엔티티의 인덱스에서 제외\n    for (const index of EntityManager.get(this.id).indexes) {\n      index.columns = index.columns.filter((col) => col.name !== oldName);\n    }\n\n    // 프롭 삭제\n    this.props.splice(at, 1);\n\n    await Promise.all(entities.map(async (entity) => entity.save()));\n  }\n\n  getEntityIdFromSubsetField(subsetField: string): string {\n    if (subsetField.includes(\".\") === false) {\n      return this.id;\n    }\n\n    // 서브셋 필드의 마지막은 프롭이므로 제외\n    const arr = subsetField.split(\".\").slice(0, -1);\n\n    // 서브셋 필드를 내려가면서 마지막으로 relation된 엔티티를 찾음\n    const lastEntityId = arr.reduce((entityId, field) => {\n      const relProp = EntityManager.get(entityId).props.find((p) => p.name === field);\n      if (!relProp || relProp.type !== \"relation\") {\n        console.debug({ arr, thisId: this.id, entityId, field });\n        throw new Error(`잘못된 서브셋키 ${subsetField}`);\n      }\n      return relProp.with;\n    }, this.id);\n    return lastEntityId;\n  }\n\n  async moveProp(at: number, to: number): Promise<void> {\n    const prop = this.props[at];\n    const newProps = [...this.props];\n    newProps.splice(to, 0, prop);\n    newProps.splice(at < to ? at : at + 1, 1);\n    this.props = newProps;\n\n    await this.save();\n  }\n}\n"],"names":["assert","writeFile","inflection","path","group","unique","z","Sonamu","isBelongsToOneRelationProp","isEnumProp","isHasManyRelationProp","isInternalSubsetField","isManyToManyRelationProp","isOneToOneRelationProp","isRelationProp","isVirtualCodeProp","isVirtualProp","normalizeSubsetField","importMembers","formatCode","exists","runtimePath","assertDefined","nonNullable","EntityManager","Entity","id","parentId","table","title","names","props","propsDict","relations","indexes","subsets","subsetsInternal","types","enums","enumLabels","underscore","pluralize","map","prop","includes","replace","Object","fromEntries","name","filter","key","fields","entries","f","enumLabel","enum","keys","parentFs","dasherize","toLowerCase","fs","module","getSubsetFieldsForQuery","subsetKey","getPuriSubsetQuery","subset","subsetQuery","resolveSubsetQuery","lines","push","join","joins","joinMethod","as","custom","from","to","selectObj","buildNestedSelectObject","select","stringifyNestedSelectObject","selectItems","result","selectItem","match","column","alias","columnValue","trim","split","pop","parts","current","i","length","part","Error","slice","lastPart","obj","indent","withBraces","spaces","repeat","innerSpaces","value","getPuriLoaderQuery","loaders","generateLoaderCode","loaderLines","loader","toTable","toCol","through","manyJoin","idField","undefined","oneJoins","forEach","refId","fromCol","getSubsetQuery","prefix","isAlreadyOuterJoined","subsetGroup","field","rel","reduce","r","groupKey","realFields","virtualCodeFields","concat","virtual","relation","relEntity","get","with","relFields","innerOrOuter","hasJoinColumn","nullable","relSubsetQuery","joinAs","fromTable","joinClause","customJoinClause","convertedLoaders","newAs","fromColumn","joinColumn","joinTable","singularize","fieldExprsToPropNodes","fieldExprs","entity","groups","fieldExpr","elseExpr","flatMap","propName","find","p","console","log","nodeType","idProp","children","getFieldExprs","maxDepth","froms","v","relMd","getTableColumns","type","relationType","getVectorColumns","getVectorColumn","columnName","vectorProps","registerModulePaths","basePath","setModulePath","toUpperCase","enumId","typesModulePath","typesFilePath","apiRootPath","importedMembers","registerTableSpecs","uniqueIndexes","idx","columns","every","col","setTableSpec","toJson","normalFields","internalFields","internal","save","subsetRows","getSubsetRows","subsetRowsToSubsetFields","jsonPath","json","JSON","stringify","register","_subsets","_subsetsInternal","prefixes","subsetKeys","allFields","allInternalFields","combinedFields","startsWith","relationEntity","isOpen","has","child","isInternal","subsetFields","some","hasKey","subsetRow","flat","createProp","at","splice","analyzeSubsetField","subsetField","arr","entityId","modifyProp","newProp","oldName","entities","allEntityIds","getAllIds","relEntityId","relEntitySubsetKeys","modifiedSubsetFields","analyzed","modified","a","Promise","all","delProp","index","getEntityIdFromSubsetField","lastEntityId","relProp","debug","thisId","moveProp","newProps"],"mappings":"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"}
|