sonamu 0.7.11 → 0.7.13
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/api/config.d.ts +10 -6
- package/dist/api/config.d.ts.map +1 -1
- package/dist/api/config.js +2 -1
- package/dist/api/sonamu.d.ts +4 -0
- package/dist/api/sonamu.d.ts.map +1 -1
- package/dist/api/sonamu.js +49 -5
- package/dist/bin/cli.js +118 -170
- package/dist/database/base-model.d.ts +10 -50
- package/dist/database/base-model.d.ts.map +1 -1
- package/dist/database/base-model.js +19 -84
- package/dist/database/base-model.types.d.ts +4 -4
- package/dist/database/base-model.types.d.ts.map +1 -1
- package/dist/database/base-model.types.js +1 -1
- package/dist/database/db.d.ts +1 -0
- package/dist/database/db.d.ts.map +1 -1
- package/dist/database/db.js +24 -13
- package/dist/database/puri-subset.test-d.js +1 -1
- package/dist/database/puri-subset.types.d.ts +1 -0
- package/dist/database/puri-subset.types.d.ts.map +1 -1
- package/dist/database/puri-subset.types.js +2 -2
- package/dist/database/puri.d.ts +82 -3
- package/dist/database/puri.d.ts.map +1 -1
- package/dist/database/puri.js +180 -14
- package/dist/database/puri.types.d.ts +33 -6
- package/dist/database/puri.types.d.ts.map +1 -1
- package/dist/database/puri.types.js +1 -1
- package/dist/database/puri.types.test-d.js +1 -1
- package/dist/entity/entity-manager.d.ts +5 -4
- package/dist/entity/entity-manager.d.ts.map +1 -1
- package/dist/entity/entity-manager.js +8 -1
- package/dist/index.d.ts +1 -1
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +3 -3
- package/dist/migration/code-generation.d.ts.map +1 -1
- package/dist/migration/code-generation.js +33 -2
- package/dist/migration/postgresql-schema-reader.d.ts.map +1 -1
- package/dist/migration/postgresql-schema-reader.js +53 -22
- package/dist/naite/messaging-types.d.ts.map +1 -1
- package/dist/naite/messaging-types.js +1 -1
- package/dist/naite/naite.js +2 -2
- package/dist/stream/sse.d.ts +2 -6
- package/dist/stream/sse.d.ts.map +1 -1
- package/dist/stream/sse.js +9 -3
- package/dist/syncer/api-parser.d.ts.map +1 -1
- package/dist/syncer/api-parser.js +7 -2
- package/dist/syncer/file-patterns.d.ts +1 -1
- package/dist/syncer/file-patterns.d.ts.map +1 -1
- package/dist/syncer/file-patterns.js +6 -5
- package/dist/syncer/module-loader.d.ts +5 -0
- package/dist/syncer/module-loader.d.ts.map +1 -1
- package/dist/syncer/module-loader.js +17 -1
- package/dist/syncer/syncer.d.ts +5 -1
- package/dist/syncer/syncer.d.ts.map +1 -1
- package/dist/syncer/syncer.js +28 -19
- package/dist/tasks/decorator.d.ts +26 -0
- package/dist/tasks/decorator.d.ts.map +1 -0
- package/dist/tasks/decorator.js +28 -0
- package/dist/tasks/step-wrapper.d.ts +18 -0
- package/dist/tasks/step-wrapper.d.ts.map +1 -0
- package/dist/tasks/step-wrapper.js +38 -0
- package/dist/tasks/workflow-manager.d.ts +40 -0
- package/dist/tasks/workflow-manager.d.ts.map +1 -0
- package/dist/tasks/workflow-manager.js +193 -0
- package/dist/template/implementations/generated.template.d.ts.map +1 -1
- package/dist/template/implementations/generated.template.js +7 -3
- package/dist/types/types.d.ts +26 -10
- package/dist/types/types.d.ts.map +1 -1
- package/dist/types/types.js +15 -2
- package/dist/ui/ai-api.d.ts +1 -0
- package/dist/ui/ai-api.d.ts.map +1 -0
- package/dist/ui/ai-api.js +50 -0
- package/dist/ui/ai-client.d.ts +1 -0
- package/dist/ui/ai-client.d.ts.map +1 -0
- package/dist/ui/ai-client.js +438 -0
- package/dist/ui/api.d.ts +3 -0
- package/dist/ui/api.d.ts.map +1 -0
- package/dist/ui/api.js +680 -0
- package/dist/ui-web/assets/brand-icons-Cu_C0hZ4.svg +1008 -0
- package/dist/ui-web/assets/brand-icons-F3SPCeH1.woff +0 -0
- package/dist/ui-web/assets/brand-icons-XL9sxUpA.woff2 +0 -0
- package/dist/ui-web/assets/brand-icons-sqJ2Pg7a.eot +0 -0
- package/dist/ui-web/assets/brand-icons-ubhWoxly.ttf +0 -0
- package/dist/ui-web/assets/flags-DOLqOU7Y.png +0 -0
- package/dist/ui-web/assets/icons-BOCtAERH.woff +0 -0
- package/dist/ui-web/assets/icons-CHzK1VD9.eot +0 -0
- package/dist/ui-web/assets/icons-D29ZQHHw.ttf +0 -0
- package/dist/ui-web/assets/icons-Du6TOHnR.woff2 +0 -0
- package/dist/ui-web/assets/icons-RwhydX30.svg +1518 -0
- package/dist/ui-web/assets/index-CpaB9P6g.css +1 -0
- package/dist/ui-web/assets/index-J9MCfjCd.js +95 -0
- package/dist/ui-web/assets/outline-icons-BfdLr8tr.svg +366 -0
- package/dist/ui-web/assets/outline-icons-DD8jm0uy.ttf +0 -0
- package/dist/ui-web/assets/outline-icons-DInHoiqI.woff2 +0 -0
- package/dist/ui-web/assets/outline-icons-LX8adJ4n.eot +0 -0
- package/dist/ui-web/assets/outline-icons-aQ88nltS.woff +0 -0
- package/dist/ui-web/assets/provider-utils_false-BKJD46kk.js +1 -0
- package/dist/ui-web/assets/provider-utils_false-Bu5lmX18.js +1 -0
- package/dist/ui-web/index.html +13 -0
- package/dist/ui-web/vite.svg +1 -0
- package/dist/utils/formatter.d.ts.map +1 -1
- package/dist/utils/formatter.js +10 -2
- package/dist/utils/model.d.ts +9 -2
- package/dist/utils/model.d.ts.map +1 -1
- package/dist/utils/model.js +16 -1
- package/dist/utils/type-utils.d.ts.map +1 -1
- package/dist/utils/type-utils.js +3 -1
- package/dist/vector/embedding.d.ts +2 -5
- package/dist/vector/embedding.d.ts.map +1 -1
- package/dist/vector/embedding.js +9 -13
- package/dist/vector/types.d.ts.map +1 -1
- package/dist/vector/types.js +1 -1
- package/package.json +9 -5
- package/src/api/config.ts +15 -11
- package/src/api/sonamu.ts +60 -6
- package/src/bin/cli.ts +57 -119
- package/src/database/base-model.ts +21 -128
- package/src/database/base-model.types.ts +3 -4
- package/src/database/db.ts +28 -18
- package/src/database/puri-subset.test-d.ts +1 -0
- package/src/database/puri-subset.types.ts +2 -0
- package/src/database/puri.ts +238 -27
- package/src/database/puri.types.test-d.ts +1 -1
- package/src/database/puri.types.ts +49 -6
- package/src/entity/entity-manager.ts +9 -0
- package/src/index.ts +1 -1
- package/src/migration/code-generation.ts +40 -1
- package/src/migration/postgresql-schema-reader.ts +53 -22
- package/src/naite/messaging-types.ts +43 -44
- package/src/naite/naite.ts +1 -1
- package/src/shared/app.shared.ts.txt +13 -0
- package/src/shared/web.shared.ts.txt +13 -0
- package/src/stream/sse.ts +15 -3
- package/src/syncer/api-parser.ts +6 -1
- package/src/syncer/file-patterns.ts +11 -9
- package/src/syncer/module-loader.ts +35 -0
- package/src/syncer/syncer.ts +34 -21
- package/src/tasks/decorator.ts +71 -0
- package/src/tasks/step-wrapper.ts +84 -0
- package/src/tasks/workflow-manager.ts +330 -0
- package/src/template/implementations/generated.template.ts +19 -6
- package/src/types/types.ts +20 -4
- package/src/ui/ai-api.ts +60 -0
- package/src/ui/ai-client.ts +499 -0
- package/src/ui/api.ts +786 -0
- package/src/utils/formatter.ts +8 -1
- package/src/utils/model.ts +26 -2
- package/src/utils/type-utils.ts +2 -0
- package/src/vector/embedding.ts +10 -14
- package/src/vector/types.ts +1 -2
- package/dist/vector/vector-search.d.ts +0 -47
- package/dist/vector/vector-search.d.ts.map +0 -1
- package/dist/vector/vector-search.js +0 -176
- package/src/vector/vector-search.ts +0 -261
package/dist/ui/api.js
ADDED
|
@@ -0,0 +1,680 @@
|
|
|
1
|
+
import { execSync } from "child_process";
|
|
2
|
+
import fs from "fs";
|
|
3
|
+
import inflection from "inflection";
|
|
4
|
+
import path from "path";
|
|
5
|
+
import { range } from "radashi";
|
|
6
|
+
import { Sonamu } from "../api/sonamu.js";
|
|
7
|
+
import { EntityManager } from "../entity/entity-manager.js";
|
|
8
|
+
import { BadRequestException, isSoException, ServiceUnavailableException } from "../exceptions/so-exceptions.js";
|
|
9
|
+
import { Migrator } from "../migration/migrator.js";
|
|
10
|
+
import { FixtureManager } from "../testing/fixture-manager.js";
|
|
11
|
+
import { TemplateKey } from "../types/types.js";
|
|
12
|
+
import { nonNullable } from "../utils/utils.js";
|
|
13
|
+
export async function sonamuUIApiPlugin(fastify) {
|
|
14
|
+
fastify.register(async (server)=>{
|
|
15
|
+
// migrator
|
|
16
|
+
const migrator = new Migrator();
|
|
17
|
+
// waitForHMRCompleted
|
|
18
|
+
async function waitForHMRCompleted(fn) {
|
|
19
|
+
const waitPromise = new Promise((resolve)=>{
|
|
20
|
+
const timeout = setTimeout(()=>{
|
|
21
|
+
resolve();
|
|
22
|
+
}, 1500);
|
|
23
|
+
const handler = ()=>{
|
|
24
|
+
clearTimeout(timeout);
|
|
25
|
+
Sonamu.syncer.eventEmitter.off("onHMRCompleted", handler);
|
|
26
|
+
resolve();
|
|
27
|
+
};
|
|
28
|
+
Sonamu.syncer.eventEmitter.once("onHMRCompleted", handler);
|
|
29
|
+
});
|
|
30
|
+
const result = await fn();
|
|
31
|
+
await waitPromise;
|
|
32
|
+
return result;
|
|
33
|
+
}
|
|
34
|
+
server.get("/api/sonamu/config", async ()=>{
|
|
35
|
+
return Sonamu.config;
|
|
36
|
+
});
|
|
37
|
+
server.get("/api/tools/openVscode", async (request)=>{
|
|
38
|
+
const { entityId, preset, absPath } = request.query;
|
|
39
|
+
const targetPath = (()=>{
|
|
40
|
+
if (entityId && preset) {
|
|
41
|
+
const entity = EntityManager.get(entityId);
|
|
42
|
+
const { names } = entity;
|
|
43
|
+
const { apiRootPath } = Sonamu;
|
|
44
|
+
const filename = (()=>{
|
|
45
|
+
switch(preset){
|
|
46
|
+
case "types":
|
|
47
|
+
return `${names.fs}.types.ts`;
|
|
48
|
+
case "entity.json":
|
|
49
|
+
return `${names.fs}.entity.json`;
|
|
50
|
+
case "generated":
|
|
51
|
+
return `${names.fs}.generated.ts`;
|
|
52
|
+
}
|
|
53
|
+
})();
|
|
54
|
+
return `${apiRootPath}/src/application/${entity.names.parentFs}/${filename}`;
|
|
55
|
+
} else {
|
|
56
|
+
if (!absPath) {
|
|
57
|
+
throw new BadRequestException("preset or absPath must be provided");
|
|
58
|
+
}
|
|
59
|
+
return absPath;
|
|
60
|
+
}
|
|
61
|
+
})();
|
|
62
|
+
execSync(`code ${targetPath}`);
|
|
63
|
+
});
|
|
64
|
+
server.get("/api/tools/getSuggestion", async (request)=>{
|
|
65
|
+
const { origin, entityId } = request.query;
|
|
66
|
+
// 치환 용어집
|
|
67
|
+
const glossary = new Map([
|
|
68
|
+
[
|
|
69
|
+
"status",
|
|
70
|
+
"상태"
|
|
71
|
+
],
|
|
72
|
+
[
|
|
73
|
+
"type",
|
|
74
|
+
"타입"
|
|
75
|
+
],
|
|
76
|
+
[
|
|
77
|
+
"image",
|
|
78
|
+
"이미지"
|
|
79
|
+
],
|
|
80
|
+
[
|
|
81
|
+
"images",
|
|
82
|
+
"이미지리스트"
|
|
83
|
+
],
|
|
84
|
+
[
|
|
85
|
+
"url",
|
|
86
|
+
"URL"
|
|
87
|
+
],
|
|
88
|
+
[
|
|
89
|
+
"id",
|
|
90
|
+
"ID"
|
|
91
|
+
],
|
|
92
|
+
[
|
|
93
|
+
"name",
|
|
94
|
+
`{EntityID}명`
|
|
95
|
+
],
|
|
96
|
+
[
|
|
97
|
+
"title",
|
|
98
|
+
"{EntityID}명"
|
|
99
|
+
],
|
|
100
|
+
[
|
|
101
|
+
"parent",
|
|
102
|
+
"상위{EntityID}"
|
|
103
|
+
],
|
|
104
|
+
[
|
|
105
|
+
"desc",
|
|
106
|
+
"설명"
|
|
107
|
+
],
|
|
108
|
+
[
|
|
109
|
+
"at",
|
|
110
|
+
"일시"
|
|
111
|
+
],
|
|
112
|
+
[
|
|
113
|
+
"created",
|
|
114
|
+
"등록"
|
|
115
|
+
],
|
|
116
|
+
[
|
|
117
|
+
"updated",
|
|
118
|
+
"수정"
|
|
119
|
+
],
|
|
120
|
+
[
|
|
121
|
+
"deleted",
|
|
122
|
+
"삭제"
|
|
123
|
+
],
|
|
124
|
+
[
|
|
125
|
+
"by",
|
|
126
|
+
"유저"
|
|
127
|
+
],
|
|
128
|
+
[
|
|
129
|
+
"date",
|
|
130
|
+
"일자"
|
|
131
|
+
],
|
|
132
|
+
[
|
|
133
|
+
"time",
|
|
134
|
+
"시간"
|
|
135
|
+
],
|
|
136
|
+
[
|
|
137
|
+
"ko",
|
|
138
|
+
"(한글)"
|
|
139
|
+
],
|
|
140
|
+
[
|
|
141
|
+
"en",
|
|
142
|
+
"(영문)"
|
|
143
|
+
],
|
|
144
|
+
[
|
|
145
|
+
"krw",
|
|
146
|
+
"(원)"
|
|
147
|
+
],
|
|
148
|
+
[
|
|
149
|
+
"usd",
|
|
150
|
+
"(USD)"
|
|
151
|
+
],
|
|
152
|
+
[
|
|
153
|
+
"color",
|
|
154
|
+
"컬러"
|
|
155
|
+
],
|
|
156
|
+
[
|
|
157
|
+
"code",
|
|
158
|
+
"코드"
|
|
159
|
+
],
|
|
160
|
+
[
|
|
161
|
+
"x",
|
|
162
|
+
"X좌표"
|
|
163
|
+
],
|
|
164
|
+
[
|
|
165
|
+
"y",
|
|
166
|
+
"Y좌표"
|
|
167
|
+
],
|
|
168
|
+
[
|
|
169
|
+
"current",
|
|
170
|
+
"현재"
|
|
171
|
+
],
|
|
172
|
+
[
|
|
173
|
+
"stock",
|
|
174
|
+
"재고"
|
|
175
|
+
],
|
|
176
|
+
[
|
|
177
|
+
"total",
|
|
178
|
+
"총"
|
|
179
|
+
],
|
|
180
|
+
[
|
|
181
|
+
"admin",
|
|
182
|
+
"관리자"
|
|
183
|
+
],
|
|
184
|
+
[
|
|
185
|
+
"group",
|
|
186
|
+
"그룹"
|
|
187
|
+
],
|
|
188
|
+
[
|
|
189
|
+
"item",
|
|
190
|
+
"아이템"
|
|
191
|
+
],
|
|
192
|
+
[
|
|
193
|
+
"cnt",
|
|
194
|
+
"수량"
|
|
195
|
+
],
|
|
196
|
+
[
|
|
197
|
+
"price",
|
|
198
|
+
"가격"
|
|
199
|
+
],
|
|
200
|
+
[
|
|
201
|
+
"preset",
|
|
202
|
+
"프리셋"
|
|
203
|
+
],
|
|
204
|
+
[
|
|
205
|
+
"acct",
|
|
206
|
+
"계좌"
|
|
207
|
+
],
|
|
208
|
+
[
|
|
209
|
+
"tel",
|
|
210
|
+
"전화번호"
|
|
211
|
+
],
|
|
212
|
+
[
|
|
213
|
+
"no",
|
|
214
|
+
"번호"
|
|
215
|
+
],
|
|
216
|
+
[
|
|
217
|
+
"body",
|
|
218
|
+
"내용"
|
|
219
|
+
],
|
|
220
|
+
[
|
|
221
|
+
"content",
|
|
222
|
+
"내용"
|
|
223
|
+
],
|
|
224
|
+
[
|
|
225
|
+
"orderno",
|
|
226
|
+
"정렬순서"
|
|
227
|
+
],
|
|
228
|
+
[
|
|
229
|
+
"priority",
|
|
230
|
+
"우선순위"
|
|
231
|
+
],
|
|
232
|
+
[
|
|
233
|
+
"text",
|
|
234
|
+
"텍스트"
|
|
235
|
+
],
|
|
236
|
+
[
|
|
237
|
+
"key",
|
|
238
|
+
"키"
|
|
239
|
+
],
|
|
240
|
+
[
|
|
241
|
+
"sum",
|
|
242
|
+
"합산"
|
|
243
|
+
],
|
|
244
|
+
[
|
|
245
|
+
"expected",
|
|
246
|
+
"예상"
|
|
247
|
+
],
|
|
248
|
+
[
|
|
249
|
+
"actual",
|
|
250
|
+
"실제"
|
|
251
|
+
]
|
|
252
|
+
]);
|
|
253
|
+
// 전체 엔티티 순회하며, 엔티티 타이틀과 프롭 설명을 치환 용어집에 추가
|
|
254
|
+
for (const entityId of EntityManager.getAllIds()){
|
|
255
|
+
const entity = EntityManager.get(entityId);
|
|
256
|
+
if ((entity.title ?? "") !== "") {
|
|
257
|
+
glossary.set(inflection.underscore(entity.id), entity.title);
|
|
258
|
+
glossary.set(inflection.underscore(inflection.pluralize(entity.id)), `${entity.title}리스트`);
|
|
259
|
+
}
|
|
260
|
+
entity.props.forEach((prop)=>{
|
|
261
|
+
if (glossary.has(prop.name)) {
|
|
262
|
+
return;
|
|
263
|
+
}
|
|
264
|
+
if (prop.desc) {
|
|
265
|
+
glossary.set(prop.name, prop.desc.replace(entity.title ?? "", "{EntityID}"));
|
|
266
|
+
}
|
|
267
|
+
});
|
|
268
|
+
}
|
|
269
|
+
const suggested = (()=>{
|
|
270
|
+
// 단어 분리, 가능한 조합 생성
|
|
271
|
+
const words = origin.split("_");
|
|
272
|
+
const combinations = [
|
|
273
|
+
...range(words.length, 0, -1)
|
|
274
|
+
].flatMap((len)=>{
|
|
275
|
+
return [
|
|
276
|
+
...range(0, words.length - len + 1, (idx)=>{
|
|
277
|
+
return {
|
|
278
|
+
len,
|
|
279
|
+
w: words.slice(idx, idx + len).join("_")
|
|
280
|
+
};
|
|
281
|
+
})
|
|
282
|
+
];
|
|
283
|
+
});
|
|
284
|
+
// 조합을 순회하며, 치환 용어집에 있는 단어가 포함된 경우, 치환 용어로 치환
|
|
285
|
+
const REPLACED_PREFIX = "#REPLACED//"; // 치환된 단어를 join 이후에도 식별하기 위해 prefix 추가
|
|
286
|
+
let remainArr = [
|
|
287
|
+
...words
|
|
288
|
+
];
|
|
289
|
+
for (const comb of combinations){
|
|
290
|
+
const remainStr = remainArr.join("_");
|
|
291
|
+
if (remainStr.includes(comb.w) && glossary.has(comb.w)) {
|
|
292
|
+
remainArr = remainStr.replace(comb.w, REPLACED_PREFIX + glossary.get(comb.w)).split("_");
|
|
293
|
+
}
|
|
294
|
+
}
|
|
295
|
+
return remainArr.map((r)=>{
|
|
296
|
+
if (r.startsWith(REPLACED_PREFIX)) {
|
|
297
|
+
return r.replace(REPLACED_PREFIX, "");
|
|
298
|
+
} else {
|
|
299
|
+
return r.toUpperCase();
|
|
300
|
+
}
|
|
301
|
+
}).join("").replace(/{EntityID}/g, entityId ? EntityManager.get(entityId).title : "");
|
|
302
|
+
})();
|
|
303
|
+
console.log({
|
|
304
|
+
entityId,
|
|
305
|
+
origin,
|
|
306
|
+
suggested
|
|
307
|
+
});
|
|
308
|
+
return {
|
|
309
|
+
suggested
|
|
310
|
+
};
|
|
311
|
+
});
|
|
312
|
+
server.get("/api/entity/findMany", async ()=>{
|
|
313
|
+
const entityIds = EntityManager.getAllIds();
|
|
314
|
+
function flattenSubsetRows(subsetRows) {
|
|
315
|
+
return subsetRows.flatMap((subsetRow)=>{
|
|
316
|
+
const { children, ...sRow } = subsetRow;
|
|
317
|
+
return [
|
|
318
|
+
sRow,
|
|
319
|
+
...flattenSubsetRows(children)
|
|
320
|
+
];
|
|
321
|
+
});
|
|
322
|
+
}
|
|
323
|
+
const entities = await Promise.all(entityIds.map((entityId)=>{
|
|
324
|
+
const entity = EntityManager.get(entityId);
|
|
325
|
+
const subsetRows = entity.getSubsetRows();
|
|
326
|
+
return {
|
|
327
|
+
...entity,
|
|
328
|
+
flattenSubsetRows: flattenSubsetRows(subsetRows)
|
|
329
|
+
};
|
|
330
|
+
}));
|
|
331
|
+
entities.sort((a, b)=>{
|
|
332
|
+
const aId = a.parentId ?? a.id;
|
|
333
|
+
const bId = b.parentId ?? b.id;
|
|
334
|
+
if (aId < bId) return -1;
|
|
335
|
+
if (aId > bId) return 1;
|
|
336
|
+
if (aId === bId) {
|
|
337
|
+
if (a.parentId === undefined) return -1;
|
|
338
|
+
if (b.parentId === undefined) return 1;
|
|
339
|
+
return 0;
|
|
340
|
+
}
|
|
341
|
+
return 0;
|
|
342
|
+
});
|
|
343
|
+
return {
|
|
344
|
+
entities
|
|
345
|
+
};
|
|
346
|
+
});
|
|
347
|
+
server.get("/api/entity/typeIds", async (request)=>{
|
|
348
|
+
const { filter, reload } = request.query;
|
|
349
|
+
if (reload === "1") {
|
|
350
|
+
await Sonamu.syncer.autoloadTypes();
|
|
351
|
+
}
|
|
352
|
+
const typeIds = (()=>{
|
|
353
|
+
const typeIds = Object.entries(Sonamu.syncer.types).filter(([_typeId, zodType])=>zodType.def.type !== "enum").map(([typeId, _zodType])=>typeId);
|
|
354
|
+
if (filter === "types") {
|
|
355
|
+
return typeIds;
|
|
356
|
+
}
|
|
357
|
+
const enumIds = EntityManager.getAllIds().flatMap((entityId)=>{
|
|
358
|
+
const entity = EntityManager.get(entityId);
|
|
359
|
+
return Object.keys(entity.enumLabels);
|
|
360
|
+
});
|
|
361
|
+
if (filter === "enums") {
|
|
362
|
+
return enumIds;
|
|
363
|
+
} else {
|
|
364
|
+
return [
|
|
365
|
+
...typeIds,
|
|
366
|
+
...enumIds
|
|
367
|
+
];
|
|
368
|
+
}
|
|
369
|
+
})();
|
|
370
|
+
return {
|
|
371
|
+
typeIds
|
|
372
|
+
};
|
|
373
|
+
});
|
|
374
|
+
server.post("/api/entity/create", async (request)=>{
|
|
375
|
+
return await waitForHMRCompleted(async ()=>{
|
|
376
|
+
const { form } = request.body;
|
|
377
|
+
await Sonamu.syncer.createEntity({
|
|
378
|
+
...form,
|
|
379
|
+
entityId: form.id
|
|
380
|
+
});
|
|
381
|
+
return 1;
|
|
382
|
+
});
|
|
383
|
+
});
|
|
384
|
+
server.post("/api/entity/del", async (request)=>{
|
|
385
|
+
return await waitForHMRCompleted(async ()=>{
|
|
386
|
+
const { entityId } = request.body;
|
|
387
|
+
return await Sonamu.syncer.delEntity(entityId);
|
|
388
|
+
});
|
|
389
|
+
});
|
|
390
|
+
server.post("/api/entity/modifyEntityBase", async (request)=>{
|
|
391
|
+
return await waitForHMRCompleted(async ()=>{
|
|
392
|
+
const { entityId, newValues } = request.body;
|
|
393
|
+
const entity = EntityManager.get(entityId);
|
|
394
|
+
entity.title = newValues.title;
|
|
395
|
+
entity.table = newValues.table;
|
|
396
|
+
entity.parentId = newValues.parentId;
|
|
397
|
+
await entity.save();
|
|
398
|
+
return 1;
|
|
399
|
+
});
|
|
400
|
+
});
|
|
401
|
+
server.post("/api/entity/modifySubset", async (request)=>{
|
|
402
|
+
return await waitForHMRCompleted(async ()=>{
|
|
403
|
+
const { entityId, subsetKey, fields } = request.body;
|
|
404
|
+
const entity = EntityManager.get(entityId);
|
|
405
|
+
entity.subsets[subsetKey] = fields;
|
|
406
|
+
await entity.save();
|
|
407
|
+
return {
|
|
408
|
+
updated: fields
|
|
409
|
+
};
|
|
410
|
+
});
|
|
411
|
+
});
|
|
412
|
+
server.post("/api/entity/delSubset", async (request)=>{
|
|
413
|
+
return await waitForHMRCompleted(async ()=>{
|
|
414
|
+
const { entityId, subsetKey } = request.body;
|
|
415
|
+
const entity = EntityManager.get(entityId);
|
|
416
|
+
delete entity.subsets[subsetKey];
|
|
417
|
+
await entity.save();
|
|
418
|
+
return 1;
|
|
419
|
+
});
|
|
420
|
+
});
|
|
421
|
+
server.post("/api/entity/createProp", async (request)=>{
|
|
422
|
+
return await waitForHMRCompleted(async ()=>{
|
|
423
|
+
const { entityId, at, newProp } = request.body;
|
|
424
|
+
const entity = EntityManager.get(entityId);
|
|
425
|
+
await entity.createProp(newProp, at);
|
|
426
|
+
return true;
|
|
427
|
+
});
|
|
428
|
+
});
|
|
429
|
+
server.post("/api/entity/modifyProp", async (request)=>{
|
|
430
|
+
return await waitForHMRCompleted(async ()=>{
|
|
431
|
+
const { entityId, at, newProp } = request.body;
|
|
432
|
+
const entity = EntityManager.get(entityId);
|
|
433
|
+
entity.modifyProp(newProp, at);
|
|
434
|
+
return true;
|
|
435
|
+
});
|
|
436
|
+
});
|
|
437
|
+
server.post("/api/entity/delProp", async (request)=>{
|
|
438
|
+
return await waitForHMRCompleted(async ()=>{
|
|
439
|
+
const { entityId, at } = request.body;
|
|
440
|
+
const entity = EntityManager.get(entityId);
|
|
441
|
+
entity.delProp(at);
|
|
442
|
+
return true;
|
|
443
|
+
});
|
|
444
|
+
});
|
|
445
|
+
server.post("/api/entity/moveProp", async (request)=>{
|
|
446
|
+
return await waitForHMRCompleted(async ()=>{
|
|
447
|
+
const { entityId, at, to } = request.body;
|
|
448
|
+
const entity = EntityManager.get(entityId);
|
|
449
|
+
entity.moveProp(at, to);
|
|
450
|
+
return true;
|
|
451
|
+
});
|
|
452
|
+
});
|
|
453
|
+
server.post("/api/entity/modifyIndexes", async (request)=>{
|
|
454
|
+
return await waitForHMRCompleted(async ()=>{
|
|
455
|
+
const { entityId, indexes } = request.body;
|
|
456
|
+
const entity = EntityManager.get(entityId);
|
|
457
|
+
entity.indexes = indexes;
|
|
458
|
+
await entity.save();
|
|
459
|
+
return {
|
|
460
|
+
updated: indexes
|
|
461
|
+
};
|
|
462
|
+
});
|
|
463
|
+
});
|
|
464
|
+
server.post("/api/entity/modifyEnumLabels", async (request)=>{
|
|
465
|
+
return await waitForHMRCompleted(async ()=>{
|
|
466
|
+
const { entityId, enumLabels } = request.body;
|
|
467
|
+
const entity = EntityManager.get(entityId);
|
|
468
|
+
entity.enumLabels = enumLabels;
|
|
469
|
+
await entity.save();
|
|
470
|
+
return {
|
|
471
|
+
updated: enumLabels
|
|
472
|
+
};
|
|
473
|
+
});
|
|
474
|
+
});
|
|
475
|
+
server.post("/api/entity/createEnumId", async (request)=>{
|
|
476
|
+
return await waitForHMRCompleted(async ()=>{
|
|
477
|
+
const { entityId, newEnumId } = request.body;
|
|
478
|
+
const entity = EntityManager.get(entityId);
|
|
479
|
+
if (entity.enumLabels[newEnumId]) {
|
|
480
|
+
throw new Error(`이미 존재하는 enumId입니다: ${newEnumId}`);
|
|
481
|
+
}
|
|
482
|
+
entity.enumLabels[newEnumId] = {
|
|
483
|
+
...newEnumId.endsWith("Status") ? {
|
|
484
|
+
active: "노출",
|
|
485
|
+
hidden: "숨김"
|
|
486
|
+
} : {
|
|
487
|
+
"": ""
|
|
488
|
+
}
|
|
489
|
+
};
|
|
490
|
+
await entity.save();
|
|
491
|
+
return 1;
|
|
492
|
+
});
|
|
493
|
+
});
|
|
494
|
+
server.post("/api/entity/modifyEnumId", async (request)=>{
|
|
495
|
+
return await waitForHMRCompleted(async ()=>{
|
|
496
|
+
const { entityId, enumId } = request.body;
|
|
497
|
+
const entityIds = EntityManager.getAllIds();
|
|
498
|
+
const isExists = entityIds.some((entityId)=>{
|
|
499
|
+
const entity = EntityManager.get(entityId);
|
|
500
|
+
return Object.keys(entity.enumLabels).includes(enumId.after);
|
|
501
|
+
});
|
|
502
|
+
if (isExists) {
|
|
503
|
+
throw new Error(`이미 존재하는 EnumId입니다: ${enumId.after}`);
|
|
504
|
+
}
|
|
505
|
+
const entity = EntityManager.get(entityId);
|
|
506
|
+
entity.enumLabels[enumId.after] = entity.enumLabels[enumId.before];
|
|
507
|
+
delete entity.enumLabels[enumId.before];
|
|
508
|
+
await entity.save();
|
|
509
|
+
for (const entityId of entityIds){
|
|
510
|
+
const entity = EntityManager.get(entityId);
|
|
511
|
+
for (const prop of entity.props){
|
|
512
|
+
if (prop.type === "enum" && prop.id === enumId.before) {
|
|
513
|
+
prop.id = enumId.after;
|
|
514
|
+
}
|
|
515
|
+
}
|
|
516
|
+
await entity.save();
|
|
517
|
+
}
|
|
518
|
+
});
|
|
519
|
+
});
|
|
520
|
+
server.post("/api/entity/deleteEnumId", async (request)=>{
|
|
521
|
+
return await waitForHMRCompleted(async ()=>{
|
|
522
|
+
const { entityId, enumId } = request.body;
|
|
523
|
+
const entityIds = EntityManager.getAllIds();
|
|
524
|
+
const isReferenced = entityIds.flatMap((entityId)=>EntityManager.get(entityId).props).some((prop)=>prop.type === "enum" && prop.id === enumId);
|
|
525
|
+
if (isReferenced) {
|
|
526
|
+
throw new Error(`${enumId}를 참조하는 프로퍼티가 존재합니다.`);
|
|
527
|
+
}
|
|
528
|
+
const entity = EntityManager.get(entityId);
|
|
529
|
+
delete entity.enumLabels[enumId];
|
|
530
|
+
await entity.save();
|
|
531
|
+
});
|
|
532
|
+
});
|
|
533
|
+
server.get("/api/entity/getTableColumns", async (request)=>{
|
|
534
|
+
const { entityId } = request.query;
|
|
535
|
+
const entity = EntityManager.get(entityId);
|
|
536
|
+
const columns = entity.getTableColumns();
|
|
537
|
+
return {
|
|
538
|
+
columns
|
|
539
|
+
};
|
|
540
|
+
});
|
|
541
|
+
server.get("/api/migrations/status", async ()=>{
|
|
542
|
+
const status = await migrator.getStatus();
|
|
543
|
+
return {
|
|
544
|
+
status
|
|
545
|
+
};
|
|
546
|
+
});
|
|
547
|
+
server.post("/api/migrations/runAction", async (request)=>{
|
|
548
|
+
const { action, targets } = request.body;
|
|
549
|
+
if (action === "shadow") {
|
|
550
|
+
return migrator.runShadowTest();
|
|
551
|
+
} else {
|
|
552
|
+
return migrator.runAction(action, targets);
|
|
553
|
+
}
|
|
554
|
+
});
|
|
555
|
+
server.post("/api/migrations/delCodes", async (request)=>{
|
|
556
|
+
const { codeNames } = request.body;
|
|
557
|
+
return await migrator.delCodes(codeNames);
|
|
558
|
+
});
|
|
559
|
+
server.post("/api/migrations/generatePreparedCodes", async (_requestt)=>{
|
|
560
|
+
return await migrator.generatePreparedCodes();
|
|
561
|
+
});
|
|
562
|
+
server.post("/api/scaffolding/getStatus", async (request)=>{
|
|
563
|
+
const { templateGroupName, entityIds, templateKeys: _templateKeys, enumIds } = request.body;
|
|
564
|
+
if ((entityIds ?? []).length === 0) {
|
|
565
|
+
throw new BadRequestException("entityIds must be provided");
|
|
566
|
+
} else if ((_templateKeys ?? []).length === 0) {
|
|
567
|
+
throw new BadRequestException("templateKeys must be provided");
|
|
568
|
+
} else if (templateGroupName === "Enums" && (enumIds ?? []).length === 0) {
|
|
569
|
+
throw new BadRequestException("enumIds must be provided");
|
|
570
|
+
}
|
|
571
|
+
// sorting
|
|
572
|
+
entityIds.sort((a, b)=>a < b ? -1 : a > b ? 1 : 0);
|
|
573
|
+
const templateKeys = TemplateKey.options.filter((tk)=>_templateKeys.includes(tk));
|
|
574
|
+
const combinations = entityIds.flatMap((entityId)=>{
|
|
575
|
+
if (templateGroupName === "Enums") {
|
|
576
|
+
const entityIds = [
|
|
577
|
+
entityId,
|
|
578
|
+
...EntityManager.getChildrenIds(entityId)
|
|
579
|
+
];
|
|
580
|
+
const allEnumIds = entityIds.flatMap((entityId)=>Object.keys(EntityManager.get(entityId).enumLabels));
|
|
581
|
+
return templateKeys.flatMap((templateKey)=>allEnumIds.filter((enumId)=>enumIds.includes(enumId)).map((enumId)=>[
|
|
582
|
+
entityId,
|
|
583
|
+
templateKey,
|
|
584
|
+
enumId
|
|
585
|
+
]));
|
|
586
|
+
} else {
|
|
587
|
+
return templateKeys.map((templateKey)=>[
|
|
588
|
+
entityId,
|
|
589
|
+
templateKey
|
|
590
|
+
]);
|
|
591
|
+
}
|
|
592
|
+
});
|
|
593
|
+
const statuses = await Promise.all(combinations.map(async ([entityId, templateKey, enumId])=>{
|
|
594
|
+
const { subPath, fullPath, isExists } = await Sonamu.syncer.checkExistsGenCode(entityId, templateKey, enumId);
|
|
595
|
+
return {
|
|
596
|
+
entityId,
|
|
597
|
+
templateGroupName,
|
|
598
|
+
templateKey,
|
|
599
|
+
enumId,
|
|
600
|
+
subPath,
|
|
601
|
+
fullPath,
|
|
602
|
+
isExists
|
|
603
|
+
};
|
|
604
|
+
}));
|
|
605
|
+
return {
|
|
606
|
+
statuses
|
|
607
|
+
};
|
|
608
|
+
});
|
|
609
|
+
server.post("/api/scaffolding/generate", async (request)=>{
|
|
610
|
+
const { options } = request.body;
|
|
611
|
+
if (options.length === 0) {
|
|
612
|
+
throw new BadRequestException("options must be provided");
|
|
613
|
+
}
|
|
614
|
+
const result = await Promise.all(options.map(async ({ entityId, templateKey, enumId, overwrite })=>{
|
|
615
|
+
try {
|
|
616
|
+
return await Sonamu.syncer.generateTemplate(templateKey, {
|
|
617
|
+
entityId,
|
|
618
|
+
enumId
|
|
619
|
+
}, {
|
|
620
|
+
overwrite
|
|
621
|
+
});
|
|
622
|
+
} catch (e) {
|
|
623
|
+
if (isSoException(e) && e.statusCode === 541) {
|
|
624
|
+
return null;
|
|
625
|
+
} else {
|
|
626
|
+
console.error(e);
|
|
627
|
+
throw e;
|
|
628
|
+
}
|
|
629
|
+
}
|
|
630
|
+
}));
|
|
631
|
+
console.log(result);
|
|
632
|
+
if (result.filter(nonNullable).length === 0) {
|
|
633
|
+
throw new ServiceUnavailableException("이미 모든 파일이 생성된 상태입니다.");
|
|
634
|
+
}
|
|
635
|
+
return result;
|
|
636
|
+
});
|
|
637
|
+
server.post("/api/scaffolding/preview", async (request)=>{
|
|
638
|
+
const { option } = request.body;
|
|
639
|
+
try {
|
|
640
|
+
const { templateKey, ...templateOptions } = option;
|
|
641
|
+
const pathAndCodes = await Sonamu.syncer.renderTemplate(templateKey, templateOptions);
|
|
642
|
+
return {
|
|
643
|
+
pathAndCodes
|
|
644
|
+
};
|
|
645
|
+
} catch (e) {
|
|
646
|
+
console.error(e);
|
|
647
|
+
throw e;
|
|
648
|
+
}
|
|
649
|
+
});
|
|
650
|
+
server.post("/api/fixture", async (request)=>{
|
|
651
|
+
const { sourceDB, targetDB, search, duplicateCheck } = request.body;
|
|
652
|
+
return FixtureManager.getFixtures(sourceDB, targetDB, search, duplicateCheck);
|
|
653
|
+
});
|
|
654
|
+
server.post("/api/fixture/import", async (request)=>{
|
|
655
|
+
const { db, fixtures } = request.body;
|
|
656
|
+
return FixtureManager.insertFixtures(db, fixtures);
|
|
657
|
+
});
|
|
658
|
+
server.post("/api/fixture/addFixtureLoader", async (request)=>{
|
|
659
|
+
const { code } = request.body;
|
|
660
|
+
return FixtureManager.addFixtureLoader(code);
|
|
661
|
+
});
|
|
662
|
+
// ui-web 빌드 파일 서빙
|
|
663
|
+
const uiDistPath = path.resolve(import.meta.dirname, "../ui-web");
|
|
664
|
+
server.register(await import("@fastify/static"), {
|
|
665
|
+
root: path.join(uiDistPath, "assets"),
|
|
666
|
+
prefix: "/assets",
|
|
667
|
+
decorateReply: false
|
|
668
|
+
});
|
|
669
|
+
// SPA fallback - /sonamu-ui/* 경로는 전부 index.html로
|
|
670
|
+
server.get("*", async (_request, reply)=>{
|
|
671
|
+
reply.headers({
|
|
672
|
+
"Content-type": "text/html"
|
|
673
|
+
}).send(fs.readFileSync(path.resolve(import.meta.dirname, "../ui-web/index.html")).toString().replace("{{projectName}}", Sonamu.config.projectName ?? "UnknownSonamuProject"));
|
|
674
|
+
});
|
|
675
|
+
}, {
|
|
676
|
+
prefix: "/sonamu-ui"
|
|
677
|
+
});
|
|
678
|
+
}
|
|
679
|
+
|
|
680
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/ui/api.ts"],"sourcesContent":["import { execSync } from \"child_process\";\nimport type { FastifyInstance } from \"fastify\";\nimport fs from \"fs\";\nimport inflection from \"inflection\";\nimport path from \"path\";\nimport { range } from \"radashi\";\nimport { Sonamu } from \"../api/sonamu\";\nimport type { SonamuDBConfig } from \"../database/db\";\nimport type { Entity } from \"../entity/entity\";\nimport { EntityManager } from \"../entity/entity-manager\";\nimport {\n  BadRequestException,\n  isSoException,\n  ServiceUnavailableException,\n} from \"../exceptions/so-exceptions\";\nimport { type MigrationResult, Migrator } from \"../migration/migrator\";\nimport { type DuplicateCheckOptions, FixtureManager } from \"../testing/fixture-manager\";\nimport {\n  type EntityIndex,\n  type EntityProp,\n  type EntitySubsetRow,\n  type FixtureRecord,\n  type FixtureSearchOptions,\n  type FlattenSubsetRow,\n  type PathAndCode,\n  TemplateKey,\n} from \"../types/types\";\nimport { nonNullable } from \"../utils/utils\";\n\nexport async function sonamuUIApiPlugin(fastify: FastifyInstance) {\n  fastify.register(\n    async (server) => {\n      // migrator\n      const migrator = new Migrator();\n\n      // waitForHMRCompleted\n      async function waitForHMRCompleted<T>(fn: () => Promise<T>): Promise<T> {\n        const waitPromise = new Promise<void>((resolve) => {\n          const timeout = setTimeout(() => {\n            resolve();\n          }, 1500);\n\n          const handler = () => {\n            clearTimeout(timeout);\n            Sonamu.syncer.eventEmitter.off(\"onHMRCompleted\", handler);\n            resolve();\n          };\n\n          Sonamu.syncer.eventEmitter.once(\"onHMRCompleted\", handler);\n        });\n\n        const result = await fn();\n        await waitPromise;\n        return result;\n      }\n\n      server.get(\"/api/sonamu/config\", async () => {\n        return Sonamu.config;\n      });\n\n      server.get<{\n        Querystring: {\n          entityId?: string;\n          preset?: \"types\" | \"entity.json\" | \"generated\";\n          absPath?: string;\n        };\n      }>(\"/api/tools/openVscode\", async (request) => {\n        const { entityId, preset, absPath } = request.query;\n\n        const targetPath = (() => {\n          if (entityId && preset) {\n            const entity = EntityManager.get(entityId);\n            const { names } = entity;\n\n            const { apiRootPath } = Sonamu;\n            const filename = (() => {\n              switch (preset) {\n                case \"types\":\n                  return `${names.fs}.types.ts`;\n                case \"entity.json\":\n                  return `${names.fs}.entity.json`;\n                case \"generated\":\n                  return `${names.fs}.generated.ts`;\n              }\n            })();\n            return `${apiRootPath}/src/application/${entity.names.parentFs}/${filename}`;\n          } else {\n            if (!absPath) {\n              throw new BadRequestException(\"preset or absPath must be provided\");\n            }\n            return absPath;\n          }\n        })();\n        execSync(`code ${targetPath}`);\n      });\n\n      server.get<{\n        Querystring: {\n          origin: string;\n          entityId?: string;\n        };\n      }>(\"/api/tools/getSuggestion\", async (request) => {\n        const { origin, entityId } = request.query;\n\n        // 치환 용어집\n        const glossary = new Map<string, string>([\n          [\"status\", \"상태\"],\n          [\"type\", \"타입\"],\n          [\"image\", \"이미지\"],\n          [\"images\", \"이미지리스트\"],\n          [\"url\", \"URL\"],\n          [\"id\", \"ID\"],\n          [\"name\", `{EntityID}명`],\n          [\"title\", \"{EntityID}명\"],\n          [\"parent\", \"상위{EntityID}\"],\n          [\"desc\", \"설명\"],\n          [\"at\", \"일시\"],\n          [\"created\", \"등록\"],\n          [\"updated\", \"수정\"],\n          [\"deleted\", \"삭제\"],\n          [\"by\", \"유저\"],\n          [\"date\", \"일자\"],\n          [\"time\", \"시간\"],\n          [\"ko\", \"(한글)\"],\n          [\"en\", \"(영문)\"],\n          [\"krw\", \"(원)\"],\n          [\"usd\", \"(USD)\"],\n          [\"color\", \"컬러\"],\n          [\"code\", \"코드\"],\n          [\"x\", \"X좌표\"],\n          [\"y\", \"Y좌표\"],\n          [\"current\", \"현재\"],\n          [\"stock\", \"재고\"],\n          [\"total\", \"총\"],\n          [\"admin\", \"관리자\"],\n          [\"group\", \"그룹\"],\n          [\"item\", \"아이템\"],\n          [\"cnt\", \"수량\"],\n          [\"price\", \"가격\"],\n          [\"preset\", \"프리셋\"],\n          [\"acct\", \"계좌\"],\n          [\"tel\", \"전화번호\"],\n          [\"no\", \"번호\"],\n          [\"body\", \"내용\"],\n          [\"content\", \"내용\"],\n          [\"orderno\", \"정렬순서\"],\n          [\"priority\", \"우선순위\"],\n          [\"text\", \"텍스트\"],\n          [\"key\", \"키\"],\n          [\"sum\", \"합산\"],\n          [\"expected\", \"예상\"],\n          [\"actual\", \"실제\"],\n        ]);\n        // 전체 엔티티 순회하며, 엔티티 타이틀과 프롭 설명을 치환 용어집에 추가\n        for (const entityId of EntityManager.getAllIds()) {\n          const entity = EntityManager.get(entityId);\n          if ((entity.title ?? \"\") !== \"\") {\n            glossary.set(inflection.underscore(entity.id), entity.title);\n            glossary.set(\n              inflection.underscore(inflection.pluralize(entity.id)),\n              `${entity.title}리스트`,\n            );\n          }\n\n          entity.props.forEach((prop) => {\n            if (glossary.has(prop.name)) {\n              return;\n            }\n            if (prop.desc) {\n              glossary.set(prop.name, prop.desc.replace(entity.title ?? \"\", \"{EntityID}\"));\n            }\n          });\n        }\n\n        const suggested = (() => {\n          // 단어 분리, 가능한 조합 생성\n          const words = origin.split(\"_\");\n          const combinations = [...range(words.length, 0, -1)].flatMap((len) => {\n            return [\n              ...range(0, words.length - len + 1, (idx) => {\n                return {\n                  len,\n                  w: words.slice(idx, idx + len).join(\"_\"),\n                };\n              }),\n            ];\n          });\n\n          // 조합을 순회하며, 치환 용어집에 있는 단어가 포함된 경우, 치환 용어로 치환\n          const REPLACED_PREFIX = \"#REPLACED//\"; // 치환된 단어를 join 이후에도 식별하기 위해 prefix 추가\n          let remainArr: string[] = [...words];\n          for (const comb of combinations) {\n            const remainStr = remainArr.join(\"_\");\n            if (remainStr.includes(comb.w) && glossary.has(comb.w)) {\n              remainArr = remainStr\n                .replace(comb.w, REPLACED_PREFIX + glossary.get(comb.w))\n                .split(\"_\");\n            }\n          }\n\n          return remainArr\n            .map((r) => {\n              if (r.startsWith(REPLACED_PREFIX)) {\n                return r.replace(REPLACED_PREFIX, \"\");\n              } else {\n                return r.toUpperCase();\n              }\n            })\n            .join(\"\")\n            .replace(/{EntityID}/g, entityId ? EntityManager.get(entityId).title : \"\");\n        })();\n\n        console.log({ entityId, origin, suggested });\n        return { suggested };\n      });\n\n      server.get(\"/api/entity/findMany\", async () => {\n        const entityIds = EntityManager.getAllIds();\n\n        function flattenSubsetRows(subsetRows: EntitySubsetRow[]): FlattenSubsetRow[] {\n          return subsetRows.flatMap((subsetRow) => {\n            const { children, ...sRow } = subsetRow;\n            return [sRow, ...flattenSubsetRows(children)];\n          });\n        }\n\n        const entities = await Promise.all(\n          entityIds.map((entityId) => {\n            const entity = EntityManager.get(entityId);\n            const subsetRows = entity.getSubsetRows();\n\n            return {\n              ...entity,\n              flattenSubsetRows: flattenSubsetRows(subsetRows),\n            };\n          }),\n        );\n\n        entities.sort((a, b) => {\n          const aId = a.parentId ?? a.id;\n          const bId = b.parentId ?? b.id;\n          if (aId < bId) return -1;\n          if (aId > bId) return 1;\n          if (aId === bId) {\n            if (a.parentId === undefined) return -1;\n            if (b.parentId === undefined) return 1;\n            return 0;\n          }\n          return 0;\n        });\n        return { entities };\n      });\n\n      server.get<{\n        Querystring: {\n          filter?: \"enums\" | \"types\";\n          reload?: \"1\";\n        };\n      }>(\"/api/entity/typeIds\", async (request): Promise<{ typeIds: string[] }> => {\n        const { filter, reload } = request.query;\n\n        if (reload === \"1\") {\n          await Sonamu.syncer.autoloadTypes();\n        }\n\n        const typeIds = (() => {\n          const typeIds = Object.entries(Sonamu.syncer.types)\n            .filter(([_typeId, zodType]) => (zodType.def.type as string) !== \"enum\")\n            .map(([typeId, _zodType]) => typeId);\n\n          if (filter === \"types\") {\n            return typeIds;\n          }\n\n          const enumIds = EntityManager.getAllIds().flatMap((entityId) => {\n            const entity = EntityManager.get(entityId);\n            return Object.keys(entity.enumLabels);\n          });\n\n          if (filter === \"enums\") {\n            return enumIds;\n          } else {\n            return [...typeIds, ...enumIds];\n          }\n        })();\n\n        return {\n          typeIds,\n        };\n      });\n\n      server.post<{\n        Body: {\n          form: {\n            id: string;\n            title: string;\n            table: string;\n            parentId?: string;\n          };\n        };\n      }>(\"/api/entity/create\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { form } = request.body;\n          await Sonamu.syncer.createEntity({ ...form, entityId: form.id });\n\n          return 1;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n        };\n      }>(\"/api/entity/del\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId } = request.body;\n          return await Sonamu.syncer.delEntity(entityId);\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          newValues: {\n            title: string;\n            table: string;\n            parentId?: string;\n          };\n        };\n      }>(\"/api/entity/modifyEntityBase\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, newValues } = request.body;\n          const entity = EntityManager.get(entityId);\n          entity.title = newValues.title;\n          entity.table = newValues.table;\n          entity.parentId = newValues.parentId;\n          await entity.save();\n\n          return 1;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          subsetKey: string;\n          fields: string[];\n        };\n      }>(\"/api/entity/modifySubset\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, subsetKey, fields } = request.body;\n          const entity = EntityManager.get(entityId);\n          entity.subsets[subsetKey] = fields;\n          await entity.save();\n\n          return { updated: fields };\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          subsetKey: string;\n        };\n      }>(\"/api/entity/delSubset\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, subsetKey } = request.body;\n          const entity = EntityManager.get(entityId);\n          delete entity.subsets[subsetKey];\n          await entity.save();\n\n          return 1;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          newProp: EntityProp;\n          at?: number;\n        };\n      }>(\"/api/entity/createProp\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, at, newProp } = request.body;\n          const entity = EntityManager.get(entityId);\n          await entity.createProp(newProp, at);\n          return true;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          newProp: EntityProp;\n          at: number;\n        };\n      }>(\"/api/entity/modifyProp\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, at, newProp } = request.body;\n\n          const entity = EntityManager.get(entityId);\n          entity.modifyProp(newProp, at);\n\n          return true;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          at: number;\n        };\n      }>(\"/api/entity/delProp\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, at } = request.body;\n\n          const entity = EntityManager.get(entityId);\n          entity.delProp(at);\n          return true;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          at: number;\n          to: number;\n        };\n      }>(\"/api/entity/moveProp\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, at, to } = request.body;\n\n          const entity = EntityManager.get(entityId);\n          entity.moveProp(at, to);\n\n          return true;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          indexes: EntityIndex[];\n        };\n      }>(\"/api/entity/modifyIndexes\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, indexes } = request.body;\n          const entity = EntityManager.get(entityId);\n          entity.indexes = indexes;\n          await entity.save();\n\n          return { updated: indexes };\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          enumLabels: Entity[\"enumLabels\"];\n        };\n      }>(\"/api/entity/modifyEnumLabels\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, enumLabels } = request.body;\n          const entity = EntityManager.get(entityId);\n          entity.enumLabels = enumLabels;\n          await entity.save();\n\n          return { updated: enumLabels };\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          newEnumId: string;\n        };\n      }>(\"/api/entity/createEnumId\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, newEnumId } = request.body;\n          const entity = EntityManager.get(entityId);\n\n          if (entity.enumLabels[newEnumId]) {\n            throw new Error(`이미 존재하는 enumId입니다: ${newEnumId}`);\n          }\n\n          entity.enumLabels[newEnumId] = {\n            ...(newEnumId.endsWith(\"Status\")\n              ? {\n                  active: \"노출\",\n                  hidden: \"숨김\",\n                }\n              : {\n                  \"\": \"\",\n                }),\n          };\n          await entity.save();\n\n          return 1;\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          enumId: {\n            before: string;\n            after: string;\n          };\n        };\n      }>(\"/api/entity/modifyEnumId\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, enumId } = request.body;\n          const entityIds = EntityManager.getAllIds();\n          const isExists = entityIds.some((entityId) => {\n            const entity = EntityManager.get(entityId);\n            return Object.keys(entity.enumLabels).includes(enumId.after);\n          });\n          if (isExists) {\n            throw new Error(`이미 존재하는 EnumId입니다: ${enumId.after}`);\n          }\n\n          const entity = EntityManager.get(entityId);\n          entity.enumLabels[enumId.after] = entity.enumLabels[enumId.before];\n          delete entity.enumLabels[enumId.before];\n\n          await entity.save();\n\n          for (const entityId of entityIds) {\n            const entity = EntityManager.get(entityId);\n            for (const prop of entity.props) {\n              if (prop.type === \"enum\" && prop.id === enumId.before) {\n                prop.id = enumId.after;\n              }\n            }\n            await entity.save();\n          }\n        });\n      });\n\n      server.post<{\n        Body: {\n          entityId: string;\n          enumId: string;\n        };\n      }>(\"/api/entity/deleteEnumId\", async (request) => {\n        return await waitForHMRCompleted(async () => {\n          const { entityId, enumId } = request.body;\n\n          const entityIds = EntityManager.getAllIds();\n          const isReferenced = entityIds\n            .flatMap((entityId) => EntityManager.get(entityId).props)\n            .some((prop) => prop.type === \"enum\" && prop.id === enumId);\n          if (isReferenced) {\n            throw new Error(`${enumId}를 참조하는 프로퍼티가 존재합니다.`);\n          }\n\n          const entity = EntityManager.get(entityId);\n          delete entity.enumLabels[enumId];\n          await entity.save();\n        });\n      });\n\n      server.get<{\n        Querystring: {\n          entityId: string;\n        };\n      }>(\"/api/entity/getTableColumns\", async (request) => {\n        const { entityId } = request.query;\n        const entity = EntityManager.get(entityId);\n        const columns = entity.getTableColumns();\n        return { columns };\n      });\n\n      server.get(\"/api/migrations/status\", async () => {\n        const status = await migrator.getStatus();\n\n        return { status };\n      });\n\n      server.post<{\n        Body: {\n          action: \"apply\" | \"rollback\" | \"shadow\";\n          targets: (keyof SonamuDBConfig)[];\n        };\n      }>(\"/api/migrations/runAction\", async (request): Promise<MigrationResult> => {\n        const { action, targets } = request.body;\n\n        if (action === \"shadow\") {\n          return migrator.runShadowTest();\n        } else {\n          return migrator.runAction(action, targets);\n        }\n      });\n\n      server.post<{\n        Body: {\n          codeNames: string[];\n        };\n      }>(\"/api/migrations/delCodes\", async (request) => {\n        const { codeNames } = request.body;\n        return await migrator.delCodes(codeNames);\n      });\n\n      server.post(\"/api/migrations/generatePreparedCodes\", async (_requestt) => {\n        return await migrator.generatePreparedCodes();\n      });\n\n      server.post<{\n        Body: {\n          templateGroupName: \"Entity\" | \"Enums\";\n          entityIds: string[];\n          templateKeys: string[];\n          enumIds: string[];\n        };\n      }>(\"/api/scaffolding/getStatus\", async (request) => {\n        const { templateGroupName, entityIds, templateKeys: _templateKeys, enumIds } = request.body;\n        if ((entityIds ?? []).length === 0) {\n          throw new BadRequestException(\"entityIds must be provided\");\n        } else if ((_templateKeys ?? []).length === 0) {\n          throw new BadRequestException(\"templateKeys must be provided\");\n        } else if (templateGroupName === \"Enums\" && (enumIds ?? []).length === 0) {\n          throw new BadRequestException(\"enumIds must be provided\");\n        }\n\n        // sorting\n        entityIds.sort((a, b) => (a < b ? -1 : a > b ? 1 : 0));\n        const templateKeys = TemplateKey.options.filter((tk) => _templateKeys.includes(tk));\n\n        const combinations = entityIds.flatMap((entityId) => {\n          if (templateGroupName === \"Enums\") {\n            const entityIds = [entityId, ...EntityManager.getChildrenIds(entityId)];\n            const allEnumIds = entityIds.flatMap((entityId) =>\n              Object.keys(EntityManager.get(entityId).enumLabels),\n            );\n            return templateKeys.flatMap((templateKey) =>\n              allEnumIds\n                .filter((enumId) => enumIds.includes(enumId))\n                .map((enumId) => [entityId, templateKey, enumId]),\n            );\n          } else {\n            return templateKeys.map((templateKey) => [entityId, templateKey]);\n          }\n        });\n\n        const statuses = await Promise.all(\n          combinations.map(async ([entityId, templateKey, enumId]) => {\n            const { subPath, fullPath, isExists } = await Sonamu.syncer.checkExistsGenCode(\n              entityId,\n              templateKey as TemplateKey,\n              enumId,\n            );\n            return {\n              entityId,\n              templateGroupName,\n              templateKey,\n              enumId,\n              subPath,\n              fullPath,\n              isExists,\n            };\n          }),\n        );\n        return { statuses };\n      });\n\n      server.post<{\n        Body: {\n          options: {\n            entityId: string;\n            templateKey: string;\n            enumId?: string;\n            overwrite?: boolean;\n          }[];\n        };\n      }>(\"/api/scaffolding/generate\", async (request) => {\n        const { options } = request.body;\n        if (options.length === 0) {\n          throw new BadRequestException(\"options must be provided\");\n        }\n\n        const result = await Promise.all(\n          options.map(async ({ entityId, templateKey, enumId, overwrite }) => {\n            try {\n              return await Sonamu.syncer.generateTemplate(\n                templateKey as TemplateKey,\n                {\n                  entityId,\n                  enumId,\n                } as {\n                  entityId: string;\n                  enumId?: string;\n                },\n                {\n                  overwrite,\n                },\n              );\n            } catch (e) {\n              if (isSoException(e) && e.statusCode === 541) {\n                return null;\n              } else {\n                console.error(e);\n                throw e;\n              }\n            }\n          }),\n        );\n        console.log(result);\n\n        if (result.filter(nonNullable).length === 0) {\n          throw new ServiceUnavailableException(\"이미 모든 파일이 생성된 상태입니다.\");\n        }\n        return result;\n      });\n\n      server.post<{\n        Body: {\n          option: {\n            entityId: string;\n            templateKey: string;\n            enumId?: string;\n          };\n        };\n      }>(\"/api/scaffolding/preview\", async (request): Promise<{ pathAndCodes: PathAndCode[] }> => {\n        const { option } = request.body;\n\n        try {\n          const { templateKey, ...templateOptions } = option;\n          const pathAndCodes = await Sonamu.syncer.renderTemplate(\n            templateKey as TemplateKey,\n            templateOptions,\n          );\n\n          return { pathAndCodes };\n        } catch (e) {\n          console.error(e);\n          throw e;\n        }\n      });\n\n      server.post(\"/api/fixture\", async (request) => {\n        const { sourceDB, targetDB, search, duplicateCheck } = request.body as {\n          sourceDB: keyof SonamuDBConfig;\n          targetDB: keyof SonamuDBConfig;\n          search: FixtureSearchOptions;\n          duplicateCheck?: DuplicateCheckOptions;\n        };\n\n        return FixtureManager.getFixtures(sourceDB, targetDB, search, duplicateCheck);\n      });\n\n      server.post(\"/api/fixture/import\", async (request) => {\n        const { db, fixtures } = request.body as {\n          db: keyof SonamuDBConfig;\n          fixtures: FixtureRecord[];\n        };\n\n        return FixtureManager.insertFixtures(db, fixtures);\n      });\n\n      server.post(\"/api/fixture/addFixtureLoader\", async (request) => {\n        const { code } = request.body as { code: string };\n\n        return FixtureManager.addFixtureLoader(code);\n      });\n\n      // ui-web 빌드 파일 서빙\n      const uiDistPath = path.resolve(import.meta.dirname, \"../ui-web\");\n      server.register(await import(\"@fastify/static\"), {\n        root: path.join(uiDistPath, \"assets\"),\n        prefix: \"/assets\",\n        decorateReply: false,\n      });\n\n      // SPA fallback - /sonamu-ui/* 경로는 전부 index.html로\n      server.get(\"*\", async (_request, reply) => {\n        reply.headers({ \"Content-type\": \"text/html\" }).send(\n          fs\n            .readFileSync(path.resolve(import.meta.dirname, \"../ui-web/index.html\"))\n            .toString()\n            .replace(\"{{projectName}}\", Sonamu.config.projectName ?? \"UnknownSonamuProject\"),\n        );\n      });\n    },\n    { prefix: \"/sonamu-ui\" },\n  );\n}\n"],"names":["execSync","fs","inflection","path","range","Sonamu","EntityManager","BadRequestException","isSoException","ServiceUnavailableException","Migrator","FixtureManager","TemplateKey","nonNullable","sonamuUIApiPlugin","fastify","register","server","migrator","waitForHMRCompleted","fn","waitPromise","Promise","resolve","timeout","setTimeout","handler","clearTimeout","syncer","eventEmitter","off","once","result","get","config","request","entityId","preset","absPath","query","targetPath","entity","names","apiRootPath","filename","parentFs","origin","glossary","Map","getAllIds","title","set","underscore","id","pluralize","props","forEach","prop","has","name","desc","replace","suggested","words","split","combinations","length","flatMap","len","idx","w","slice","join","REPLACED_PREFIX","remainArr","comb","remainStr","includes","map","r","startsWith","toUpperCase","console","log","entityIds","flattenSubsetRows","subsetRows","subsetRow","children","sRow","entities","all","getSubsetRows","sort","a","b","aId","parentId","bId","undefined","filter","reload","autoloadTypes","typeIds","Object","entries","types","_typeId","zodType","def","type","typeId","_zodType","enumIds","keys","enumLabels","post","form","body","createEntity","delEntity","newValues","table","save","subsetKey","fields","subsets","updated","at","newProp","createProp","modifyProp","delProp","to","moveProp","indexes","newEnumId","Error","endsWith","active","hidden","enumId","isExists","some","after","before","isReferenced","columns","getTableColumns","status","getStatus","action","targets","runShadowTest","runAction","codeNames","delCodes","_requestt","generatePreparedCodes","templateGroupName","templateKeys","_templateKeys","options","tk","getChildrenIds","allEnumIds","templateKey","statuses","subPath","fullPath","checkExistsGenCode","overwrite","generateTemplate","e","statusCode","error","option","templateOptions","pathAndCodes","renderTemplate","sourceDB","targetDB","search","duplicateCheck","getFixtures","db","fixtures","insertFixtures","code","addFixtureLoader","uiDistPath","dirname","root","prefix","decorateReply","_request","reply","headers","send","readFileSync","toString","projectName"],"mappings":"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"}
|