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/index.d.ts
CHANGED
|
@@ -4,6 +4,7 @@ export type * from "./api/context";
|
|
|
4
4
|
export * from "./api/decorators";
|
|
5
5
|
export * from "./api/sonamu";
|
|
6
6
|
export * from "./database/base-model";
|
|
7
|
+
export * from "./database/base-model.types";
|
|
7
8
|
export * from "./database/db";
|
|
8
9
|
export * from "./database/puri";
|
|
9
10
|
export * from "./database/puri.types";
|
package/dist/index.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,cAAc,kBAAkB,CAAC;AACjC,cAAc,cAAc,CAAC;AAC7B,mBAAmB,eAAe,CAAC;AACnC,cAAc,kBAAkB,CAAC;AACjC,cAAc,cAAc,CAAC;AAC7B,cAAc,uBAAuB,CAAC;AACtC,cAAc,eAAe,CAAC;AAC9B,cAAc,iBAAiB,CAAC;AAChC,cAAc,uBAAuB,CAAC;AACtC,cAAc,8BAA8B,CAAC;AAC7C,cAAc,yBAAyB,CAAC;AACxC,cAAc,2BAA2B,CAAC;AAC1C,cAAc,iBAAiB,CAAC;AAChC,cAAc,yBAAyB,CAAC;AACxC,cAAc,4BAA4B,CAAC;AAC3C,cAAc,4BAA4B,CAAC;AAC3C,cAAc,uBAAuB,CAAC;AACtC,cAAc,2BAA2B,CAAC;AAC1C,cAAc,sBAAsB,CAAC;AACrC,cAAc,sCAAsC,CAAC;AACrD,cAAc,mBAAmB,CAAC;AAClC,cAAc,eAAe,CAAC;AAC9B,cAAc,wBAAwB,CAAC;AACvC,cAAc,cAAc,CAAC;AAC7B,cAAc,mBAAmB,CAAC;AAClC,cAAc,qBAAqB,CAAC;AACpC,cAAc,6BAA6B,CAAC;AAC5C,cAAc,2BAA2B,CAAC;AAC1C,cAAc,eAAe,CAAC;AAC9B,cAAc,oBAAoB,CAAC;AACnC,cAAc,eAAe,CAAC;AAC9B,cAAc,oBAAoB,CAAC;AACnC,cAAc,eAAe,CAAC"}
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,cAAc,kBAAkB,CAAC;AACjC,cAAc,cAAc,CAAC;AAC7B,mBAAmB,eAAe,CAAC;AACnC,cAAc,kBAAkB,CAAC;AACjC,cAAc,cAAc,CAAC;AAC7B,cAAc,uBAAuB,CAAC;AACtC,cAAc,6BAA6B,CAAC;AAC5C,cAAc,eAAe,CAAC;AAC9B,cAAc,iBAAiB,CAAC;AAChC,cAAc,uBAAuB,CAAC;AACtC,cAAc,8BAA8B,CAAC;AAC7C,cAAc,yBAAyB,CAAC;AACxC,cAAc,2BAA2B,CAAC;AAC1C,cAAc,iBAAiB,CAAC;AAChC,cAAc,yBAAyB,CAAC;AACxC,cAAc,4BAA4B,CAAC;AAC3C,cAAc,4BAA4B,CAAC;AAC3C,cAAc,uBAAuB,CAAC;AACtC,cAAc,2BAA2B,CAAC;AAC1C,cAAc,sBAAsB,CAAC;AACrC,cAAc,sCAAsC,CAAC;AACrD,cAAc,mBAAmB,CAAC;AAClC,cAAc,eAAe,CAAC;AAC9B,cAAc,wBAAwB,CAAC;AACvC,cAAc,cAAc,CAAC;AAC7B,cAAc,mBAAmB,CAAC;AAClC,cAAc,qBAAqB,CAAC;AACpC,cAAc,6BAA6B,CAAC;AAC5C,cAAc,2BAA2B,CAAC;AAC1C,cAAc,eAAe,CAAC;AAC9B,cAAc,oBAAoB,CAAC;AACnC,cAAc,eAAe,CAAC;AAC9B,cAAc,oBAAoB,CAAC;AACnC,cAAc,eAAe,CAAC"}
|
package/dist/index.js
CHANGED
|
@@ -3,6 +3,7 @@ export * from "./api/config.js";
|
|
|
3
3
|
export * from "./api/decorators.js";
|
|
4
4
|
export * from "./api/sonamu.js";
|
|
5
5
|
export * from "./database/base-model.js";
|
|
6
|
+
export * from "./database/base-model.types.js";
|
|
6
7
|
export * from "./database/db.js";
|
|
7
8
|
export * from "./database/puri.js";
|
|
8
9
|
export * from "./database/puri.types.js";
|
|
@@ -32,4 +33,4 @@ export * from "./utils/type-utils.js";
|
|
|
32
33
|
export * from "./utils/utils.js"; // export * from "./api/code-converters";
|
|
33
34
|
// export * from "./syncer/syncer";
|
|
34
35
|
|
|
35
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
36
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uL3NyYy9pbmRleC50cyJdLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tIFwiLi9hcGkvYmFzZS1mcmFtZVwiO1xuZXhwb3J0ICogZnJvbSBcIi4vYXBpL2NvbmZpZ1wiO1xuZXhwb3J0IHR5cGUgKiBmcm9tIFwiLi9hcGkvY29udGV4dFwiO1xuZXhwb3J0ICogZnJvbSBcIi4vYXBpL2RlY29yYXRvcnNcIjtcbmV4cG9ydCAqIGZyb20gXCIuL2FwaS9zb25hbXVcIjtcbmV4cG9ydCAqIGZyb20gXCIuL2RhdGFiYXNlL2Jhc2UtbW9kZWxcIjtcbmV4cG9ydCAqIGZyb20gXCIuL2RhdGFiYXNlL2Jhc2UtbW9kZWwudHlwZXNcIjtcbmV4cG9ydCAqIGZyb20gXCIuL2RhdGFiYXNlL2RiXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9kYXRhYmFzZS9wdXJpXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9kYXRhYmFzZS9wdXJpLnR5cGVzXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9kYXRhYmFzZS9wdXJpLXN1YnNldC50eXBlc1wiO1xuZXhwb3J0ICogZnJvbSBcIi4vZGF0YWJhc2UvcHVyaS13cmFwcGVyXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9kYXRhYmFzZS91cHNlcnQtYnVpbGRlclwiO1xuZXhwb3J0ICogZnJvbSBcIi4vZW50aXR5L2VudGl0eVwiO1xuZXhwb3J0ICogZnJvbSBcIi4vZW50aXR5L2VudGl0eS1tYW5hZ2VyXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9leGNlcHRpb25zL2Vycm9yLWhhbmRsZXJcIjtcbmV4cG9ydCAqIGZyb20gXCIuL2V4Y2VwdGlvbnMvc28tZXhjZXB0aW9uc1wiO1xuZXhwb3J0ICogZnJvbSBcIi4vZmlsZS1zdG9yYWdlL2RyaXZlclwiO1xuZXhwb3J0ICogZnJvbSBcIi4vbWlncmF0aW9uL21pZ3JhdGlvbi1zZXRcIjtcbmV4cG9ydCAqIGZyb20gXCIuL21pZ3JhdGlvbi9taWdyYXRvclwiO1xuZXhwb3J0ICogZnJvbSBcIi4vbWlncmF0aW9uL3Bvc3RncmVzcWwtc2NoZW1hLXJlYWRlclwiO1xuZXhwb3J0ICogZnJvbSBcIi4vbWlncmF0aW9uL3R5cGVzXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9uYWl0ZS9uYWl0ZVwiO1xuZXhwb3J0ICogZnJvbSBcIi4vbmFpdGUvbmFpdGUtcmVwb3J0ZXJcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3N0cmVhbS9zc2VcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3Rhc2tzL2RlY29yYXRvclwiO1xuZXhwb3J0ICogZnJvbSBcIi4vdGVtcGxhdGUvdGVtcGxhdGVcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3RlbXBsYXRlL3RlbXBsYXRlLW1hbmFnZXJcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3Rlc3RpbmcvZml4dHVyZS1tYW5hZ2VyXCI7XG5leHBvcnQgKiBmcm9tIFwiLi90eXBlcy90eXBlc1wiO1xuZXhwb3J0ICogZnJvbSBcIi4vdXRpbHMvY29udHJvbGxlclwiO1xuZXhwb3J0ICogZnJvbSBcIi4vdXRpbHMvbW9kZWxcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3V0aWxzL3R5cGUtdXRpbHNcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3V0aWxzL3V0aWxzXCI7XG5cbi8vIGV4cG9ydCAqIGZyb20gXCIuL2FwaS9jb2RlLWNvbnZlcnRlcnNcIjtcbi8vIGV4cG9ydCAqIGZyb20gXCIuL3N5bmNlci9zeW5jZXJcIjtcbiJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLHNCQUFtQjtBQUNqQyxjQUFjLGtCQUFlO0FBRTdCLGNBQWMsc0JBQW1CO0FBQ2pDLGNBQWMsa0JBQWU7QUFDN0IsY0FBYywyQkFBd0I7QUFDdEMsY0FBYyxpQ0FBOEI7QUFDNUMsY0FBYyxtQkFBZ0I7QUFDOUIsY0FBYyxxQkFBa0I7QUFDaEMsY0FBYywyQkFBd0I7QUFDdEMsY0FBYyxrQ0FBK0I7QUFDN0MsY0FBYyw2QkFBMEI7QUFDeEMsY0FBYywrQkFBNEI7QUFDMUMsY0FBYyxxQkFBa0I7QUFDaEMsY0FBYyw2QkFBMEI7QUFDeEMsY0FBYyxnQ0FBNkI7QUFDM0MsY0FBYyxnQ0FBNkI7QUFDM0MsY0FBYywyQkFBd0I7QUFDdEMsY0FBYywrQkFBNEI7QUFDMUMsY0FBYywwQkFBdUI7QUFDckMsY0FBYywwQ0FBdUM7QUFDckQsY0FBYyx1QkFBb0I7QUFDbEMsY0FBYyxtQkFBZ0I7QUFDOUIsY0FBYyw0QkFBeUI7QUFDdkMsY0FBYyxrQkFBZTtBQUM3QixjQUFjLHVCQUFvQjtBQUNsQyxjQUFjLHlCQUFzQjtBQUNwQyxjQUFjLGlDQUE4QjtBQUM1QyxjQUFjLCtCQUE0QjtBQUMxQyxjQUFjLG1CQUFnQjtBQUM5QixjQUFjLHdCQUFxQjtBQUNuQyxjQUFjLG1CQUFnQjtBQUM5QixjQUFjLHdCQUFxQjtBQUNuQyxjQUFjLG1CQUFnQixDQUU5Qix5Q0FBeUM7Q0FDekMsbUNBQW1DIn0=
|
|
@@ -10,12 +10,17 @@ export declare function findChangedFilesUsingChecksums(): Promise<AbsolutePath[]
|
|
|
10
10
|
* 현재 파일들의 체크섬을 계산해서 구한 다음, 체크섬 파일에 저장된 내용과 다르면 체크섬 파일을 갱신합니다.
|
|
11
11
|
*/
|
|
12
12
|
export declare function renewChecksums(): Promise<void>;
|
|
13
|
+
export type FileOrData = {
|
|
14
|
+
path: PathLike;
|
|
15
|
+
} | {
|
|
16
|
+
data: string;
|
|
17
|
+
};
|
|
13
18
|
/**
|
|
14
19
|
* 두 파일의 내용이 같은지 체크섬으로 비교합니다.
|
|
15
20
|
* 만약 파일이 둘 중 하나라도 없다면 비교 불가로 false 반환합니다.
|
|
16
|
-
* @param one 파일 경로
|
|
17
|
-
* @param two 파일 경로
|
|
21
|
+
* @param one 파일 경로 혹은 데이터
|
|
22
|
+
* @param two 파일 경로 혹은 데이터
|
|
18
23
|
* @returns boolean
|
|
19
24
|
*/
|
|
20
|
-
export declare function areFilesSame(
|
|
25
|
+
export declare function areFilesSame(...files: FileOrData[]): Promise<boolean>;
|
|
21
26
|
//# sourceMappingURL=checksum.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"checksum.d.ts","sourceRoot":"","sources":["../../src/syncer/checksum.ts"],"names":[],"mappings":"AAEA,OAAO,EAAoB,KAAK,QAAQ,EAAE,MAAM,IAAI,CAAC;AAOrD,OAAO,KAAK,EAAE,YAAY,EAAmB,MAAM,qBAAqB,CAAC;AASzE;;;GAGG;AACH,wBAAsB,8BAA8B,IAAI,OAAO,CAAC,YAAY,EAAE,CAAC,CAU9E;AAED;;;GAGG;AACH,wBAAsB,cAAc,IAAI,OAAO,CAAC,IAAI,CAAC,CAUpD;AAED;;;;;;GAMG;AACH,wBAAsB,YAAY,CAAC,GAAG,
|
|
1
|
+
{"version":3,"file":"checksum.d.ts","sourceRoot":"","sources":["../../src/syncer/checksum.ts"],"names":[],"mappings":"AAEA,OAAO,EAAoB,KAAK,QAAQ,EAAE,MAAM,IAAI,CAAC;AAOrD,OAAO,KAAK,EAAE,YAAY,EAAmB,MAAM,qBAAqB,CAAC;AASzE;;;GAGG;AACH,wBAAsB,8BAA8B,IAAI,OAAO,CAAC,YAAY,EAAE,CAAC,CAU9E;AAED;;;GAGG;AACH,wBAAsB,cAAc,IAAI,OAAO,CAAC,IAAI,CAAC,CAUpD;AAED,MAAM,MAAM,UAAU,GAClB;IACE,IAAI,EAAE,QAAQ,CAAC;CAChB,GACD;IACE,IAAI,EAAE,MAAM,CAAC;CACd,CAAC;AAEN;;;;;;GAMG;AACH,wBAAsB,YAAY,CAAC,GAAG,KAAK,EAAE,UAAU,EAAE,GAAG,OAAO,CAAC,OAAO,CAAC,CAiB3E"}
|
package/dist/syncer/checksum.js
CHANGED
|
@@ -36,16 +36,19 @@ import { getChecksumPatternGroupInAbsolutePath } from "./file-patterns.js";
|
|
|
36
36
|
/**
|
|
37
37
|
* 두 파일의 내용이 같은지 체크섬으로 비교합니다.
|
|
38
38
|
* 만약 파일이 둘 중 하나라도 없다면 비교 불가로 false 반환합니다.
|
|
39
|
-
* @param one 파일 경로
|
|
40
|
-
* @param two 파일 경로
|
|
39
|
+
* @param one 파일 경로 혹은 데이터
|
|
40
|
+
* @param two 파일 경로 혹은 데이터
|
|
41
41
|
* @returns boolean
|
|
42
|
-
*/ export async function areFilesSame(
|
|
43
|
-
|
|
44
|
-
|
|
42
|
+
*/ export async function areFilesSame(...files) {
|
|
43
|
+
const checksums = [];
|
|
44
|
+
for (const file of files){
|
|
45
|
+
if ("path" in file && !await exists(file.path)) {
|
|
46
|
+
return false;
|
|
47
|
+
}
|
|
48
|
+
checksums.push("path" in file ? await getChecksumOfFile(file.path) : getChecksumOfData(file.data));
|
|
45
49
|
}
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
return oneChecksum === twoChecksum;
|
|
50
|
+
return checksums.every(// 다음 체크섬과 비교, 만약 마지막 체크섬일 때는 첫 번째 체크섬과 비교
|
|
51
|
+
(checksum, index)=>checksum === checksums[index === checksums.length - 1 ? 0 : index + 1]);
|
|
49
52
|
}
|
|
50
53
|
async function getCurrentChecksums() {
|
|
51
54
|
const filePaths = (await Promise.all(Object.entries(getChecksumPatternGroupInAbsolutePath()).map(async ([_fileType, pattern])=>{
|
|
@@ -81,6 +84,11 @@ async function saveChecksums(checksums) {
|
|
|
81
84
|
})), null, 2), "utf-8");
|
|
82
85
|
console.log("checksum saved", checksumFilePath);
|
|
83
86
|
}
|
|
87
|
+
function getChecksumOfData(data) {
|
|
88
|
+
const hash = crypto.createHash("sha1");
|
|
89
|
+
hash.update(data);
|
|
90
|
+
return hash.digest("hex");
|
|
91
|
+
}
|
|
84
92
|
async function getChecksumOfFile(filePath) {
|
|
85
93
|
return new Promise((resolve, reject)=>{
|
|
86
94
|
const hash = crypto.createHash("sha1");
|
|
@@ -95,4 +103,4 @@ async function getChecksumOfFile(filePath) {
|
|
|
95
103
|
});
|
|
96
104
|
}
|
|
97
105
|
|
|
98
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/syncer/checksum.ts"],"sourcesContent":["import crypto, { type BinaryLike } from \"crypto\";\nimport equal from \"fast-deep-equal\";\nimport { createReadStream, type PathLike } from \"fs\";\nimport { readFile, writeFile } from \"fs/promises\";\nimport path from \"path\";\nimport { isEqual } from \"radashi\";\nimport { Sonamu } from \"../api/sonamu\";\nimport { globAsync } from \"../utils/async-utils\";\nimport { exists } from \"../utils/fs-utils\";\nimport type { AbsolutePath, ApiRelativePath } from \"../utils/path-utils\";\nimport { differenceWith } from \"../utils/utils\";\nimport { getChecksumPatternGroupInAbsolutePath } from \"./file-patterns\";\n\ntype PathAndChecksum = {\n  path: AbsolutePath;\n  checksum: string;\n};\n\n/**\n * 체크섬 파일에 저장된 내용과 현재 실제 파일의 체크섬을 비교하여 변경된 파일을 찾습니다.\n * @returns 변경된 파일 경로 배열. 프로젝트 루트부터 슬래시로 시작합니다. 예시: \"/src/application/user/user.model.ts\"\n */\nexport async function findChangedFilesUsingChecksums(): Promise<AbsolutePath[]> {\n  const calculatedChecksums = await getCurrentChecksums();\n  const savedChecksums = await getPreviousChecksums();\n\n  const isSame = equal(calculatedChecksums, savedChecksums);\n  if (isSame) {\n    return [];\n  }\n\n  return differenceWith(calculatedChecksums, savedChecksums, isEqual).map((r) => r.path);\n}\n\n/**\n * 체크섬을 갱신합니다.\n * 현재 파일들의 체크섬을 계산해서 구한 다음, 체크섬 파일에 저장된 내용과 다르면 체크섬 파일을 갱신합니다.\n */\nexport async function renewChecksums(): Promise<void> {\n  const calculatedChecksums = await getCurrentChecksums();\n  const savedChecksums = await getPreviousChecksums();\n\n  const isSame = equal(calculatedChecksums, savedChecksums);\n  if (isSame) {\n    return;\n  }\n\n  await saveChecksums(calculatedChecksums);\n}\n\n/**\n * 두 파일의 내용이 같은지 체크섬으로 비교합니다.\n * 만약 파일이 둘 중 하나라도 없다면 비교 불가로 false 반환합니다.\n * @param one 파일 경로\n * @param two 파일 경로\n * @returns boolean\n */\nexport async function areFilesSame(one: PathLike, two: PathLike): Promise<boolean> {\n  if (!(await exists(one)) || !(await exists(two))) {\n    return false;\n  }\n\n  const oneChecksum = await getChecksumOfFile(one);\n  const twoChecksum = await getChecksumOfFile(two);\n\n  return oneChecksum === twoChecksum;\n}\n\nasync function getCurrentChecksums(): Promise<PathAndChecksum[]> {\n  const filePaths = (\n    await Promise.all(\n      Object.entries(getChecksumPatternGroupInAbsolutePath()).map(async ([_fileType, pattern]) => {\n        return globAsync(pattern) as Promise<AbsolutePath[]>;\n      }),\n    )\n  )\n    .flat()\n    .sort();\n\n  const fileChecksums = await Promise.all(\n    filePaths.map(async (filePath) => {\n      return {\n        path: filePath,\n        checksum: await getChecksumOfFile(filePath),\n      };\n    }),\n  );\n\n  return fileChecksums;\n}\n\nasync function getPreviousChecksums(): Promise<PathAndChecksum[]> {\n  const checksumFilePath = getChecksumFilePath();\n  if (!(await exists(checksumFilePath))) {\n    return [];\n  }\n\n  const previousChecksums = JSON.parse(await readFile(checksumFilePath, \"utf-8\")).map(\n    (r: { path: ApiRelativePath; checksum: string }) => ({\n      path: path.join(Sonamu.apiRootPath, r.path), // 체크섬 파일에서 읽을 때: API 상대 경로 → 절대 경로\n      checksum: r.checksum,\n    }),\n  ) as PathAndChecksum[];\n  return previousChecksums;\n}\n\nfunction getChecksumFilePath(): AbsolutePath {\n  return path.join(Sonamu.apiRootPath, \"sonamu.lock\") as AbsolutePath;\n}\n\nasync function saveChecksums(checksums: PathAndChecksum[]): Promise<void> {\n  const checksumFilePath = getChecksumFilePath();\n  await writeFile(\n    checksumFilePath,\n    JSON.stringify(\n      checksums.map((r) => ({\n        path: path.relative(Sonamu.apiRootPath, r.path), // 체크섬 파일에 저장할 때: 절대 경로 → API 상대 경로\n        checksum: r.checksum,\n      })),\n      null,\n      2,\n    ),\n    \"utf-8\",\n  );\n  console.log(\"checksum saved\", checksumFilePath);\n}\n\nasync function getChecksumOfFile(filePath: PathLike): Promise<string> {\n  return new Promise<string>((resolve, reject) => {\n    const hash = crypto.createHash(\"sha1\");\n    const input = createReadStream(filePath);\n    input.on(\"error\", reject);\n    input.on(\"data\", (chunk: BinaryLike) => {\n      hash.update(chunk);\n    });\n    input.on(\"close\", () => {\n      resolve(hash.digest(\"hex\"));\n    });\n  });\n}\n"],"names":["crypto","equal","createReadStream","readFile","writeFile","path","isEqual","Sonamu","globAsync","exists","differenceWith","getChecksumPatternGroupInAbsolutePath","findChangedFilesUsingChecksums","calculatedChecksums","getCurrentChecksums","savedChecksums","getPreviousChecksums","isSame","map","r","renewChecksums","saveChecksums","areFilesSame","one","two","oneChecksum","getChecksumOfFile","twoChecksum","filePaths","Promise","all","Object","entries","_fileType","pattern","flat","sort","fileChecksums","filePath","checksum","checksumFilePath","getChecksumFilePath","previousChecksums","JSON","parse","join","apiRootPath","checksums","stringify","relative","console","log","resolve","reject","hash","createHash","input","on","chunk","update","digest"],"mappings":"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"}
|
|
106
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/syncer/checksum.ts"],"sourcesContent":["import crypto, { type BinaryLike } from \"crypto\";\nimport equal from \"fast-deep-equal\";\nimport { createReadStream, type PathLike } from \"fs\";\nimport { readFile, writeFile } from \"fs/promises\";\nimport path from \"path\";\nimport { isEqual } from \"radashi\";\nimport { Sonamu } from \"../api/sonamu\";\nimport { globAsync } from \"../utils/async-utils\";\nimport { exists } from \"../utils/fs-utils\";\nimport type { AbsolutePath, ApiRelativePath } from \"../utils/path-utils\";\nimport { differenceWith } from \"../utils/utils\";\nimport { getChecksumPatternGroupInAbsolutePath } from \"./file-patterns\";\n\ntype PathAndChecksum = {\n  path: AbsolutePath;\n  checksum: string;\n};\n\n/**\n * 체크섬 파일에 저장된 내용과 현재 실제 파일의 체크섬을 비교하여 변경된 파일을 찾습니다.\n * @returns 변경된 파일 경로 배열. 프로젝트 루트부터 슬래시로 시작합니다. 예시: \"/src/application/user/user.model.ts\"\n */\nexport async function findChangedFilesUsingChecksums(): Promise<AbsolutePath[]> {\n  const calculatedChecksums = await getCurrentChecksums();\n  const savedChecksums = await getPreviousChecksums();\n\n  const isSame = equal(calculatedChecksums, savedChecksums);\n  if (isSame) {\n    return [];\n  }\n\n  return differenceWith(calculatedChecksums, savedChecksums, isEqual).map((r) => r.path);\n}\n\n/**\n * 체크섬을 갱신합니다.\n * 현재 파일들의 체크섬을 계산해서 구한 다음, 체크섬 파일에 저장된 내용과 다르면 체크섬 파일을 갱신합니다.\n */\nexport async function renewChecksums(): Promise<void> {\n  const calculatedChecksums = await getCurrentChecksums();\n  const savedChecksums = await getPreviousChecksums();\n\n  const isSame = equal(calculatedChecksums, savedChecksums);\n  if (isSame) {\n    return;\n  }\n\n  await saveChecksums(calculatedChecksums);\n}\n\nexport type FileOrData =\n  | {\n      path: PathLike;\n    }\n  | {\n      data: string;\n    };\n\n/**\n * 두 파일의 내용이 같은지 체크섬으로 비교합니다.\n * 만약 파일이 둘 중 하나라도 없다면 비교 불가로 false 반환합니다.\n * @param one 파일 경로 혹은 데이터\n * @param two 파일 경로 혹은 데이터\n * @returns boolean\n */\nexport async function areFilesSame(...files: FileOrData[]): Promise<boolean> {\n  const checksums: string[] = [];\n\n  for (const file of files) {\n    if (\"path\" in file && !(await exists(file.path))) {\n      return false;\n    }\n\n    checksums.push(\n      \"path\" in file ? await getChecksumOfFile(file.path) : getChecksumOfData(file.data),\n    );\n  }\n\n  return checksums.every(\n    // 다음 체크섬과 비교, 만약 마지막 체크섬일 때는 첫 번째 체크섬과 비교\n    (checksum, index) => checksum === checksums[index === checksums.length - 1 ? 0 : index + 1],\n  );\n}\n\nasync function getCurrentChecksums(): Promise<PathAndChecksum[]> {\n  const filePaths = (\n    await Promise.all(\n      Object.entries(getChecksumPatternGroupInAbsolutePath()).map(async ([_fileType, pattern]) => {\n        return globAsync(pattern) as Promise<AbsolutePath[]>;\n      }),\n    )\n  )\n    .flat()\n    .sort();\n\n  const fileChecksums = await Promise.all(\n    filePaths.map(async (filePath) => {\n      return {\n        path: filePath,\n        checksum: await getChecksumOfFile(filePath),\n      };\n    }),\n  );\n\n  return fileChecksums;\n}\n\nasync function getPreviousChecksums(): Promise<PathAndChecksum[]> {\n  const checksumFilePath = getChecksumFilePath();\n  if (!(await exists(checksumFilePath))) {\n    return [];\n  }\n\n  const previousChecksums = JSON.parse(await readFile(checksumFilePath, \"utf-8\")).map(\n    (r: { path: ApiRelativePath; checksum: string }) => ({\n      path: path.join(Sonamu.apiRootPath, r.path), // 체크섬 파일에서 읽을 때: API 상대 경로 → 절대 경로\n      checksum: r.checksum,\n    }),\n  ) as PathAndChecksum[];\n  return previousChecksums;\n}\n\nfunction getChecksumFilePath(): AbsolutePath {\n  return path.join(Sonamu.apiRootPath, \"sonamu.lock\") as AbsolutePath;\n}\n\nasync function saveChecksums(checksums: PathAndChecksum[]): Promise<void> {\n  const checksumFilePath = getChecksumFilePath();\n  await writeFile(\n    checksumFilePath,\n    JSON.stringify(\n      checksums.map((r) => ({\n        path: path.relative(Sonamu.apiRootPath, r.path), // 체크섬 파일에 저장할 때: 절대 경로 → API 상대 경로\n        checksum: r.checksum,\n      })),\n      null,\n      2,\n    ),\n    \"utf-8\",\n  );\n  console.log(\"checksum saved\", checksumFilePath);\n}\n\nfunction getChecksumOfData(data: string): string {\n  const hash = crypto.createHash(\"sha1\");\n  hash.update(data);\n  return hash.digest(\"hex\");\n}\n\nasync function getChecksumOfFile(filePath: PathLike): Promise<string> {\n  return new Promise<string>((resolve, reject) => {\n    const hash = crypto.createHash(\"sha1\");\n    const input = createReadStream(filePath);\n    input.on(\"error\", reject);\n    input.on(\"data\", (chunk: BinaryLike) => {\n      hash.update(chunk);\n    });\n    input.on(\"close\", () => {\n      resolve(hash.digest(\"hex\"));\n    });\n  });\n}\n"],"names":["crypto","equal","createReadStream","readFile","writeFile","path","isEqual","Sonamu","globAsync","exists","differenceWith","getChecksumPatternGroupInAbsolutePath","findChangedFilesUsingChecksums","calculatedChecksums","getCurrentChecksums","savedChecksums","getPreviousChecksums","isSame","map","r","renewChecksums","saveChecksums","areFilesSame","files","checksums","file","push","getChecksumOfFile","getChecksumOfData","data","every","checksum","index","length","filePaths","Promise","all","Object","entries","_fileType","pattern","flat","sort","fileChecksums","filePath","checksumFilePath","getChecksumFilePath","previousChecksums","JSON","parse","join","apiRootPath","stringify","relative","console","log","hash","createHash","update","digest","resolve","reject","input","on","chunk"],"mappings":"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"}
|
|
@@ -87,7 +87,12 @@ async function resolveRenderedTemplate(key, result) {
|
|
|
87
87
|
const { target, path: filePath, body, importKeys, customHeaders } = result;
|
|
88
88
|
// import 할 대상의 대상 path 추출
|
|
89
89
|
const importDefs = importKeys.reduce((r, importKey)=>{
|
|
90
|
-
|
|
90
|
+
let modulePath = importKey;
|
|
91
|
+
try {
|
|
92
|
+
modulePath = EntityManager.getModulePath(importKey);
|
|
93
|
+
} catch (error) {
|
|
94
|
+
throw new Error(`[resolveRenderedTemplate:${key}] ${importKey} 모듈 경로 찾기 실패: ${error}`);
|
|
95
|
+
}
|
|
91
96
|
let importPath = modulePath;
|
|
92
97
|
if (modulePath.includes("/") || modulePath.includes(".")) {
|
|
93
98
|
importPath = wrapIf(path.relative(path.dirname(filePath), modulePath), (p)=>[
|
|
@@ -158,4 +163,4 @@ async function writeCodeToPathEachTarget(pathAndCode) {
|
|
|
158
163
|
}));
|
|
159
164
|
}
|
|
160
165
|
|
|
161
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/syncer/code-generator.ts"],"sourcesContent":["import chalk from \"chalk\";\nimport { mkdir, writeFile } from \"fs/promises\";\nimport path from \"path\";\nimport { unique } from \"radashi\";\nimport { Sonamu } from \"../api/sonamu\";\nimport { EntityManager } from \"../entity/entity-manager\";\nimport { AlreadyProcessedException } from \"../exceptions/so-exceptions\";\nimport { Naite } from \"../naite/naite\";\nimport type { RenderedTemplate } from \"../template/template\";\nimport { TemplateManager } from \"../template/template-manager\";\nimport type { GenerateOptions, PathAndCode, TemplateKey, TemplateOptions } from \"../types/types\";\nimport { everyAsync, filterAsync } from \"../utils/async-utils\";\nimport { isTest } from \"../utils/controller\";\nimport { formatCode } from \"../utils/formatter\";\nimport { exists } from \"../utils/fs-utils\";\nimport { wrapIf } from \"../utils/lodash-able\";\nimport type { AbsolutePath } from \"../utils/path-utils\";\n\n/**\n * 템플릿을 렌더링하고 파일로 생성합니다.\n * overwrite 옵션이 false인 경우, 이미 존재하는 파일은 건너뜁니다.\n * @param key - 템플릿 키 (예: \"entity\", \"model\", \"service\" 등)\n * @param templateOptions - 템플릿 렌더링에 필요한 옵션\n * @param _generateOptions - 생성 옵션 (overwrite 여부)\n * @returns 생성된 파일 경로 배열\n */\nexport async function generateTemplate<T extends TemplateKey>(\n  key: T,\n  templateOptions: TemplateOptions[T],\n  _generateOptions?: GenerateOptions,\n): Promise<AbsolutePath[]> {\n  const generateOptions = {\n    overwrite: false,\n    ..._generateOptions,\n  };\n  Naite.t(\"generateTemplate\", { key, templateOptions, generateOptions });\n\n  // 키 children\n  const keys: TemplateKey[] = [key];\n\n  // 템플릿 렌더\n  const pathAndCodes = (\n    await Promise.all(\n      keys.map(async (key) => {\n        return await renderTemplate(key, templateOptions);\n      }),\n    )\n  ).flat();\n\n  const filteredPathAndCodes: PathAndCode[] = await (async () => {\n    if (generateOptions.overwrite === true) {\n      return pathAndCodes;\n    } else {\n      return await filterAsync(pathAndCodes, async (pathAndCode) => {\n        const { targets } = Sonamu.config.sync;\n        const filePath = `${Sonamu.appRootPath}/${pathAndCode.path}`;\n        const dstFilePaths = targets.map((target) => filePath.replace(\"/:target/\", `/${target}/`));\n        return await everyAsync(dstFilePaths, async (dstPath) => !(await exists(dstPath)));\n      });\n    }\n  })();\n\n  if (filteredPathAndCodes.length === 0) {\n    throw new AlreadyProcessedException(\"이미 경로에 모든 파일이 존재합니다.\");\n  }\n\n  return (\n    await Promise.all(\n      filteredPathAndCodes.map((pathAndCode) => writeCodeToPathEachTarget(pathAndCode)),\n    )\n  ).flat();\n}\n\n/**\n * 템플릿을 렌더링하여 PathAndCode 객체를 반환합니다.\n * 파일로 쓰지 않고 메모리상에서만 렌더링합니다.\n * @param key - 템플릿 키\n * @param options - 템플릿 렌더링 옵션\n * @returns 경로와 코드 쌍의 배열\n */\nexport async function renderTemplate<T extends keyof TemplateOptions>(\n  key: T,\n  options: TemplateOptions[T],\n): Promise<PathAndCode[]> {\n  Naite.t(\"renderTemplate\", { key, options });\n\n  const template = TemplateManager.get(key);\n\n  const rendered = await template.render(options);\n  const resolved = await resolveRenderedTemplate(key, rendered);\n\n  let preTemplateResolved: PathAndCode[] = [];\n  if (rendered.preTemplates) {\n    preTemplateResolved = (\n      await Promise.all(\n        rendered.preTemplates.map(({ key, options }) => {\n          return renderTemplate(key, options);\n        }),\n      )\n    ).flat();\n  }\n\n  return [resolved, ...preTemplateResolved];\n}\n\nasync function resolveRenderedTemplate(\n  key: TemplateKey,\n  result: RenderedTemplate,\n): Promise<PathAndCode> {\n  Naite.t(`resolveRenderedTemplate${key}`, { key, result });\n\n  const { target, path: filePath, body, importKeys, customHeaders } = result;\n\n  // import 할 대상의 대상 path 추출\n  const importDefs = importKeys\n    .reduce(\n      (r, importKey) => {\n        const modulePath = EntityManager.getModulePath(importKey);\n        let importPath = modulePath;\n        if (modulePath.includes(\"/\") || modulePath.includes(\".\")) {\n          importPath = wrapIf(path.relative(path.dirname(filePath), modulePath), (p) => [\n            p.startsWith(\".\") === false,\n            `./${p}`,\n          ]);\n        }\n\n        // 같은 파일에서 import 하는 경우 keys 로 나열 처리\n        const existsOne = r.find((importDef) => importDef.from === importPath);\n        if (existsOne) {\n          existsOne.keys = unique(existsOne.keys.concat(importKey));\n        } else {\n          r.push({\n            keys: [importKey],\n            from: importPath,\n          });\n        }\n        return r;\n      },\n      [] as {\n        keys: string[];\n        from: string;\n      }[],\n    )\n    // 셀프 참조 방지\n    .filter((importDef) => filePath.endsWith(`${importDef.from.replace(\"./\", \"\")}.ts`) === false);\n\n  // 커스텀 헤더 포함하여 헤더 생성\n  const header = [\n    ...(customHeaders ?? []),\n    ...importDefs.map(\n      (importDef) => `import { ${importDef.keys.join(\", \")} } from '${importDef.from}'`,\n    ),\n  ].join(\"\\n\");\n\n  const formatted = await (async () => {\n    if (key === \"generated_http\") {\n      return [header, body].join(\"\\n\\n\");\n    } else {\n      Naite.t(\"resolveRenderedTemplate:beforeFormat\", { key, header, body });\n      const formatted = formatCode(\n        [header, body].join(\"\\n\\n\"),\n        key === \"entity\" ? \"json\" : \"typescript\",\n        `${Sonamu.appRootPath}/${filePath}`,\n      );\n      Naite.t(`resolveRenderedTemplate:formatted:${key}`, formatted);\n      return formatted;\n    }\n  })();\n\n  return {\n    path: `${target}/${filePath}`,\n    code: formatted,\n  };\n}\n\nasync function writeCodeToPathEachTarget(pathAndCode: PathAndCode): Promise<AbsolutePath[]> {\n  const { targets } = Sonamu.config.sync;\n  const { appRootPath } = Sonamu;\n  const filePath = `${Sonamu.appRootPath}/${pathAndCode.path}` as AbsolutePath;\n\n  const dstFilePaths = unique(\n    targets.map((target) => filePath.replace(\"/:target/\", `/${target}/`)) as AbsolutePath[],\n  );\n  return await Promise.all(\n    dstFilePaths.map(async (dstFilePath) => {\n      const dir = path.dirname(dstFilePath);\n      if (!(await exists(dir))) {\n        await mkdir(dir, { recursive: true });\n      }\n      await writeFile(dstFilePath, pathAndCode.code);\n      !isTest() &&\n        console.log(\n          chalk.bold(\"Generated: \") + chalk.blue(`${dstFilePath.replace(`${appRootPath}/`, \"\")}`),\n        );\n      return dstFilePath;\n    }),\n  );\n}\n"],"names":["chalk","mkdir","writeFile","path","unique","Sonamu","EntityManager","AlreadyProcessedException","Naite","TemplateManager","everyAsync","filterAsync","isTest","formatCode","exists","wrapIf","generateTemplate","key","templateOptions","_generateOptions","generateOptions","overwrite","t","keys","pathAndCodes","Promise","all","map","renderTemplate","flat","filteredPathAndCodes","pathAndCode","targets","config","sync","filePath","appRootPath","dstFilePaths","target","replace","dstPath","length","writeCodeToPathEachTarget","options","template","get","rendered","render","resolved","resolveRenderedTemplate","preTemplateResolved","preTemplates","result","body","importKeys","customHeaders","importDefs","reduce","r","importKey","modulePath","getModulePath","importPath","includes","relative","dirname","p","startsWith","existsOne","find","importDef","from","concat","push","filter","endsWith","header","join","formatted","code","dstFilePath","dir","recursive","console","log","bold","blue"],"mappings":"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"}
|
|
166
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/syncer/code-generator.ts"],"sourcesContent":["import chalk from \"chalk\";\nimport { mkdir, writeFile } from \"fs/promises\";\nimport path from \"path\";\nimport { unique } from \"radashi\";\nimport { Sonamu } from \"../api/sonamu\";\nimport { EntityManager } from \"../entity/entity-manager\";\nimport { AlreadyProcessedException } from \"../exceptions/so-exceptions\";\nimport { Naite } from \"../naite/naite\";\nimport type { RenderedTemplate } from \"../template/template\";\nimport { TemplateManager } from \"../template/template-manager\";\nimport type { GenerateOptions, PathAndCode, TemplateKey, TemplateOptions } from \"../types/types\";\nimport { everyAsync, filterAsync } from \"../utils/async-utils\";\nimport { isTest } from \"../utils/controller\";\nimport { formatCode } from \"../utils/formatter\";\nimport { exists } from \"../utils/fs-utils\";\nimport { wrapIf } from \"../utils/lodash-able\";\nimport type { AbsolutePath } from \"../utils/path-utils\";\n\n/**\n * 템플릿을 렌더링하고 파일로 생성합니다.\n * overwrite 옵션이 false인 경우, 이미 존재하는 파일은 건너뜁니다.\n * @param key - 템플릿 키 (예: \"entity\", \"model\", \"service\" 등)\n * @param templateOptions - 템플릿 렌더링에 필요한 옵션\n * @param _generateOptions - 생성 옵션 (overwrite 여부)\n * @returns 생성된 파일 경로 배열\n */\nexport async function generateTemplate<T extends TemplateKey>(\n  key: T,\n  templateOptions: TemplateOptions[T],\n  _generateOptions?: GenerateOptions,\n): Promise<AbsolutePath[]> {\n  const generateOptions = {\n    overwrite: false,\n    ..._generateOptions,\n  };\n  Naite.t(\"generateTemplate\", { key, templateOptions, generateOptions });\n\n  // 키 children\n  const keys: TemplateKey[] = [key];\n\n  // 템플릿 렌더\n  const pathAndCodes = (\n    await Promise.all(\n      keys.map(async (key) => {\n        return await renderTemplate(key, templateOptions);\n      }),\n    )\n  ).flat();\n\n  const filteredPathAndCodes: PathAndCode[] = await (async () => {\n    if (generateOptions.overwrite === true) {\n      return pathAndCodes;\n    } else {\n      return await filterAsync(pathAndCodes, async (pathAndCode) => {\n        const { targets } = Sonamu.config.sync;\n        const filePath = `${Sonamu.appRootPath}/${pathAndCode.path}`;\n        const dstFilePaths = targets.map((target) => filePath.replace(\"/:target/\", `/${target}/`));\n        return await everyAsync(dstFilePaths, async (dstPath) => !(await exists(dstPath)));\n      });\n    }\n  })();\n\n  if (filteredPathAndCodes.length === 0) {\n    throw new AlreadyProcessedException(\"이미 경로에 모든 파일이 존재합니다.\");\n  }\n\n  return (\n    await Promise.all(\n      filteredPathAndCodes.map((pathAndCode) => writeCodeToPathEachTarget(pathAndCode)),\n    )\n  ).flat();\n}\n\n/**\n * 템플릿을 렌더링하여 PathAndCode 객체를 반환합니다.\n * 파일로 쓰지 않고 메모리상에서만 렌더링합니다.\n * @param key - 템플릿 키\n * @param options - 템플릿 렌더링 옵션\n * @returns 경로와 코드 쌍의 배열\n */\nexport async function renderTemplate<T extends keyof TemplateOptions>(\n  key: T,\n  options: TemplateOptions[T],\n): Promise<PathAndCode[]> {\n  Naite.t(\"renderTemplate\", { key, options });\n\n  const template = TemplateManager.get(key);\n\n  const rendered = await template.render(options);\n  const resolved = await resolveRenderedTemplate(key, rendered);\n\n  let preTemplateResolved: PathAndCode[] = [];\n  if (rendered.preTemplates) {\n    preTemplateResolved = (\n      await Promise.all(\n        rendered.preTemplates.map(({ key, options }) => {\n          return renderTemplate(key, options);\n        }),\n      )\n    ).flat();\n  }\n\n  return [resolved, ...preTemplateResolved];\n}\n\nasync function resolveRenderedTemplate(\n  key: TemplateKey,\n  result: RenderedTemplate,\n): Promise<PathAndCode> {\n  Naite.t(`resolveRenderedTemplate${key}`, { key, result });\n\n  const { target, path: filePath, body, importKeys, customHeaders } = result;\n\n  // import 할 대상의 대상 path 추출\n  const importDefs = importKeys\n    .reduce(\n      (r, importKey) => {\n        let modulePath = importKey;\n        try {\n          modulePath = EntityManager.getModulePath(importKey);\n        } catch (error) {\n          throw new Error(\n            `[resolveRenderedTemplate:${key}] ${importKey} 모듈 경로 찾기 실패: ${error}`,\n          );\n        }\n        let importPath = modulePath;\n        if (modulePath.includes(\"/\") || modulePath.includes(\".\")) {\n          importPath = wrapIf(path.relative(path.dirname(filePath), modulePath), (p) => [\n            p.startsWith(\".\") === false,\n            `./${p}`,\n          ]);\n        }\n\n        // 같은 파일에서 import 하는 경우 keys 로 나열 처리\n        const existsOne = r.find((importDef) => importDef.from === importPath);\n        if (existsOne) {\n          existsOne.keys = unique(existsOne.keys.concat(importKey));\n        } else {\n          r.push({\n            keys: [importKey],\n            from: importPath,\n          });\n        }\n        return r;\n      },\n      [] as {\n        keys: string[];\n        from: string;\n      }[],\n    )\n    // 셀프 참조 방지\n    .filter((importDef) => filePath.endsWith(`${importDef.from.replace(\"./\", \"\")}.ts`) === false);\n\n  // 커스텀 헤더 포함하여 헤더 생성\n  const header = [\n    ...(customHeaders ?? []),\n    ...importDefs.map(\n      (importDef) => `import { ${importDef.keys.join(\", \")} } from '${importDef.from}'`,\n    ),\n  ].join(\"\\n\");\n\n  const formatted = await (async () => {\n    if (key === \"generated_http\") {\n      return [header, body].join(\"\\n\\n\");\n    } else {\n      Naite.t(\"resolveRenderedTemplate:beforeFormat\", { key, header, body });\n      const formatted = formatCode(\n        [header, body].join(\"\\n\\n\"),\n        key === \"entity\" ? \"json\" : \"typescript\",\n        `${Sonamu.appRootPath}/${filePath}`,\n      );\n      Naite.t(`resolveRenderedTemplate:formatted:${key}`, formatted);\n      return formatted;\n    }\n  })();\n\n  return {\n    path: `${target}/${filePath}`,\n    code: formatted,\n  };\n}\n\nasync function writeCodeToPathEachTarget(pathAndCode: PathAndCode): Promise<AbsolutePath[]> {\n  const { targets } = Sonamu.config.sync;\n  const { appRootPath } = Sonamu;\n  const filePath = `${Sonamu.appRootPath}/${pathAndCode.path}` as AbsolutePath;\n\n  const dstFilePaths = unique(\n    targets.map((target) => filePath.replace(\"/:target/\", `/${target}/`)) as AbsolutePath[],\n  );\n  return await Promise.all(\n    dstFilePaths.map(async (dstFilePath) => {\n      const dir = path.dirname(dstFilePath);\n      if (!(await exists(dir))) {\n        await mkdir(dir, { recursive: true });\n      }\n      await writeFile(dstFilePath, pathAndCode.code);\n      !isTest() &&\n        console.log(\n          chalk.bold(\"Generated: \") + chalk.blue(`${dstFilePath.replace(`${appRootPath}/`, \"\")}`),\n        );\n      return dstFilePath;\n    }),\n  );\n}\n"],"names":["chalk","mkdir","writeFile","path","unique","Sonamu","EntityManager","AlreadyProcessedException","Naite","TemplateManager","everyAsync","filterAsync","isTest","formatCode","exists","wrapIf","generateTemplate","key","templateOptions","_generateOptions","generateOptions","overwrite","t","keys","pathAndCodes","Promise","all","map","renderTemplate","flat","filteredPathAndCodes","pathAndCode","targets","config","sync","filePath","appRootPath","dstFilePaths","target","replace","dstPath","length","writeCodeToPathEachTarget","options","template","get","rendered","render","resolved","resolveRenderedTemplate","preTemplateResolved","preTemplates","result","body","importKeys","customHeaders","importDefs","reduce","r","importKey","modulePath","getModulePath","error","Error","importPath","includes","relative","dirname","p","startsWith","existsOne","find","importDef","from","concat","push","filter","endsWith","header","join","formatted","code","dstFilePath","dir","recursive","console","log","bold","blue"],"mappings":"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"}
|
package/dist/syncer/syncer.d.ts
CHANGED
|
@@ -52,18 +52,18 @@ export declare class Syncer {
|
|
|
52
52
|
handleModelOrFrameChange(diffGroups: DiffGroups): Promise<void>;
|
|
53
53
|
actionSyncConfig(): Promise<void>;
|
|
54
54
|
/**
|
|
55
|
-
*
|
|
56
|
-
* @returns 생성된 파일 경로 배열.
|
|
57
|
-
*/
|
|
58
|
-
actionGenerateSchemas(): Promise<AbsolutePath[]>;
|
|
59
|
-
/**
|
|
60
|
-
* *.service.ts를 생성합니다.
|
|
55
|
+
* services.generated.ts를 생성합니다.
|
|
61
56
|
* @param paramsArray
|
|
62
57
|
* @returns 생성된 파일 경로 배열.
|
|
63
58
|
*/
|
|
64
59
|
actionGenerateServices(paramsArray: {
|
|
65
60
|
namesRecord: EntityNamesRecord;
|
|
66
61
|
}[]): Promise<string[]>;
|
|
62
|
+
/**
|
|
63
|
+
* sonamu.generated.ts와 sonamu.generated.sso.ts를 생성합니다.
|
|
64
|
+
* @returns 생성된 파일 경로 배열.
|
|
65
|
+
*/
|
|
66
|
+
actionGenerateSchemas(): Promise<AbsolutePath[]>;
|
|
67
67
|
/**
|
|
68
68
|
* sonamu.generated.http를 생성합니다.
|
|
69
69
|
* @returns 생성된 파일 경로.
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"syncer.d.ts","sourceRoot":"","sources":["../../src/syncer/syncer.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,YAAY,EAAE,MAAM,QAAQ,CAAC;AAMtC,OAAO,KAAK,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AAC7B,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,IAAI,CAAC;AAC3C,OAAO,EAAE,cAAc,EAAE,MAAM,mBAAmB,CAAC;AAEnD,OAAO,EAAiB,KAAK,iBAAiB,EAAE,MAAM,0BAA0B,CAAC;AAGjF,OAAO,KAAK,EAAE,eAAe,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AACnE,OAAO,EAAE,WAAW,EAAE,KAAK,eAAe,EAAE,MAAM,gBAAgB,CAAC;AAKnE,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AAKxD,OAAO,EAAE,KAAK,QAAQ,EAAyC,MAAM,iBAAiB,CAAC;AACvF,OAAO,EACL,KAAK,UAAU,EACf,KAAK,YAAY,EACjB,KAAK,WAAW,EAKjB,MAAM,iBAAiB,CAAC;AAEzB,KAAK,UAAU,GAAG;KACf,GAAG,IAAI,QAAQ,GAAG,YAAY,EAAE;CAClC,CAAC;AAEF,qBAAa,MAAM;IACjB,IAAI,EAAE,UAAU,CAAM;IACtB,KAAK,EAAE,WAAW,CAAM;IACxB,MAAM,EAAE,YAAY,CAAM;IAC1B,SAAS,EAAE,GAAG,CAAC,MAAM,EAAE,gBAAgB,EAAE,CAAC,CAAa;IACvD,SAAS,EAAE,OAAO,CAAS;IAC3B,YAAY,EAAE,YAAY,CAAsB;IAEhD;;;;OAIG;IACG,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC;IA2B3B;;;;;OAKG;IACG,eAAe,CAAC,KAAK,EAAE,MAAM,EAAE,YAAY,EAAE,YAAY,GAAG,OAAO,CAAC,IAAI,CAAC;IA0D/E,+BAA+B,CAC7B,eAAe,EAAE,YAAY,GAC5B,CAAC,OAAO,cAAc,CAAC,CAAC,MAAM,CAAC,EAAE;IAc9B,mBAAmB,CAAC,OAAO,EAAE,MAAM,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC;
|
|
1
|
+
{"version":3,"file":"syncer.d.ts","sourceRoot":"","sources":["../../src/syncer/syncer.ts"],"names":[],"mappings":"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"}
|
package/dist/syncer/syncer.js
CHANGED
|
@@ -118,6 +118,10 @@ export class Syncer {
|
|
|
118
118
|
return toRemove;
|
|
119
119
|
}
|
|
120
120
|
async copySharedToTargets(targets) {
|
|
121
|
+
// 특정 변수 치환을 위해서 사용합니다.
|
|
122
|
+
const convertMap = {
|
|
123
|
+
baseUrl: Sonamu.config.server.baseUrl ?? `http://${Sonamu.config.server.listen?.host ?? "localhost"}:${Sonamu.config.server.listen?.port ?? 3000}`
|
|
124
|
+
};
|
|
121
125
|
for (const target of targets){
|
|
122
126
|
// 지금 가져가려는 이 파일은 Sonamu 코드베이스의 일부입니다.
|
|
123
127
|
// 그런데 dist 속 빌드된 소스 코드 파일이 필요한 것이 아니고, src에만 있는 텍스트 파일이 필요합니다.
|
|
@@ -129,6 +133,8 @@ export class Syncer {
|
|
|
129
133
|
if (!await exists(path.join(Sonamu.appRootPath, target))) {
|
|
130
134
|
throw new Error(`Tried to copy sonamu.shared.ts to target '${target}' but the target directory does not exist. Please check your project directory structure.`);
|
|
131
135
|
}
|
|
136
|
+
const fullText = await readFile(srcPath, "utf-8");
|
|
137
|
+
const convertedText = Object.entries(convertMap).reduce((acc, [key, value])=>acc.replace(`$[[${key}]]`, value), fullText);
|
|
132
138
|
// 이건 프로젝트에 .ts 소스 코드 파일을 생성하는 것이므로 src의 .ts 경로로 갑니다.
|
|
133
139
|
const destPath = path.join(Sonamu.appRootPath, target, "src/services/sonamu.shared.ts");
|
|
134
140
|
// 정말 혹시나지만 target 디렉토리는 있어도 src/services 디렉토리는 없을 수 있으므로 미리 생성해줍니다.
|
|
@@ -138,10 +144,14 @@ export class Syncer {
|
|
|
138
144
|
});
|
|
139
145
|
console.warn(`Created directory '${path.dirname(destPath)}' because it did not exist.`);
|
|
140
146
|
}
|
|
141
|
-
if (await areFilesSame(
|
|
147
|
+
if (await areFilesSame({
|
|
148
|
+
data: convertedText
|
|
149
|
+
}, {
|
|
150
|
+
path: destPath
|
|
151
|
+
})) {
|
|
142
152
|
continue;
|
|
143
153
|
}
|
|
144
|
-
await writeFile(destPath,
|
|
154
|
+
await writeFile(destPath, convertedText);
|
|
145
155
|
!isTest() && console.log(chalk.bold("Copied: ") + chalk.blue(path.relative(Sonamu.appRootPath, destPath)));
|
|
146
156
|
}
|
|
147
157
|
}
|
|
@@ -293,6 +303,20 @@ export class Syncer {
|
|
|
293
303
|
}));
|
|
294
304
|
}
|
|
295
305
|
/**
|
|
306
|
+
* services.generated.ts를 생성합니다.
|
|
307
|
+
* @param paramsArray
|
|
308
|
+
* @returns 생성된 파일 경로 배열.
|
|
309
|
+
*/ async actionGenerateServices(paramsArray) {
|
|
310
|
+
Naite.t("actionGenerateServices", paramsArray);
|
|
311
|
+
// services.generated.ts 통합 파일 생성
|
|
312
|
+
const servicesFile = await generateTemplate("services", {}, {
|
|
313
|
+
overwrite: true
|
|
314
|
+
});
|
|
315
|
+
return [
|
|
316
|
+
...servicesFile
|
|
317
|
+
];
|
|
318
|
+
}
|
|
319
|
+
/**
|
|
296
320
|
* sonamu.generated.ts와 sonamu.generated.sso.ts를 생성합니다.
|
|
297
321
|
* @returns 생성된 파일 경로 배열.
|
|
298
322
|
*/ async actionGenerateSchemas() {
|
|
@@ -306,16 +330,6 @@ export class Syncer {
|
|
|
306
330
|
])).flat().flat();
|
|
307
331
|
}
|
|
308
332
|
/**
|
|
309
|
-
* *.service.ts를 생성합니다.
|
|
310
|
-
* @param paramsArray
|
|
311
|
-
* @returns 생성된 파일 경로 배열.
|
|
312
|
-
*/ async actionGenerateServices(paramsArray) {
|
|
313
|
-
Naite.t("actionGenerateServices", paramsArray);
|
|
314
|
-
return (await Promise.all(paramsArray.map(async (params)=>generateTemplate("service", params, {
|
|
315
|
-
overwrite: true
|
|
316
|
-
})))).flat().flat();
|
|
317
|
-
}
|
|
318
|
-
/**
|
|
319
333
|
* sonamu.generated.http를 생성합니다.
|
|
320
334
|
* @returns 생성된 파일 경로.
|
|
321
335
|
*/ async actionGenerateHttps() {
|
|
@@ -441,4 +455,4 @@ export class Syncer {
|
|
|
441
455
|
}
|
|
442
456
|
}
|
|
443
457
|
|
|
444
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/syncer/syncer.ts"],"sourcesContent":["import { hot } from \"@sonamu-kit/hmr-hook\";\nimport assert from \"assert\";\nimport chalk from \"chalk\";\nimport { EventEmitter } from \"events\";\nimport { mkdir, readFile, writeFile } from \"fs/promises\";\nimport inflection from \"inflection\";\nimport { minimatch } from \"minimatch\";\nimport path, { dirname } from \"path\";\nimport { group, unique } from \"radashi\";\nimport type { z } from \"zod\";\nimport type { WorkflowMetadata } from \"..\";\nimport { registeredApis } from \"../api/decorators\";\nimport { Sonamu } from \"../api/sonamu\";\nimport { EntityManager, type EntityNamesRecord } from \"../entity/entity-manager\";\nimport { Naite } from \"../naite/naite\";\nimport { TemplateManager } from \"../template/template-manager\";\nimport type { GenerateOptions, PathAndCode } from \"../types/types\";\nimport { TemplateKey, type TemplateOptions } from \"../types/types\";\nimport { mapAsync, reduceAsync } from \"../utils/async-utils\";\nimport { centerText } from \"../utils/console-util\";\nimport { isTest } from \"../utils/controller\";\nimport { exists } from \"../utils/fs-utils\";\nimport type { AbsolutePath } from \"../utils/path-utils\";\nimport { runWithGracefulShutdown } from \"../utils/process-utils\";\nimport { areFilesSame, findChangedFilesUsingChecksums, renewChecksums } from \"./checksum\";\nimport { generateTemplate, renderTemplate } from \"./code-generator\";\nimport { createEntity, delEntity } from \"./entity-operations\";\nimport { type FileType, getChecksumPatternGroupInAbsolutePath } from \"./file-patterns\";\nimport {\n  type LoadedApis,\n  type LoadedModels,\n  type LoadedTypes,\n  loadApis,\n  loadModels,\n  loadTypes,\n  loadWorkflows,\n} from \"./module-loader\";\n\ntype DiffGroups = {\n  [key in FileType]: AbsolutePath[];\n};\n\nexport class Syncer {\n  apis: LoadedApis = [];\n  types: LoadedTypes = {};\n  models: LoadedModels = {};\n  workflows: Map<string, WorkflowMetadata[]> = new Map();\n  isSyncing: boolean = false;\n  eventEmitter: EventEmitter = new EventEmitter();\n\n  /**\n   * 체크섬이 변경된 부분에 대해 싱크를 진행합니다.\n   * 다만 sonamu.shared.ts는 체크섬 비교 없이 무조건 싱크(복사)합니다.\n   * @returns\n   */\n  async sync(): Promise<void> {\n    const { targets } = Sonamu.config.sync;\n\n    // sonamu.shared.ts는 무조건 싱크(복사)합니다.\n    await this.copySharedToTargets(targets);\n\n    // 그 다음부터는 변경된 파일을 찾아서 동기화 작업을 실행합니다.\n    const changedFiles = await findChangedFilesUsingChecksums();\n    if (changedFiles.length === 0) {\n      console.log(chalk.black.bgGreen(centerText(\"All files are synced!\")));\n      return;\n    }\n\n    // 만약 싱크 중에 프로세스가 죽으면 꼬여버리기 때문에,\n    // 시그널에도 잠시 버틸 수 있는 환경 속에서 싱크를 실행합니다.\n    await runWithGracefulShutdown(\n      async () => {\n        // 얘가 싱크 작업 수행하는 본체입니다.\n        await this.doSyncActions(changedFiles);\n\n        // 싱크 액션이 끝나면 항상 체크섬을 다시 갱신합니다.\n        await renewChecksums();\n      },\n      { whenThisHappens: \"SIGUSR2\", waitForUpTo: 20000 },\n    );\n  }\n\n  /**\n   * Watcher가 감지한 파일 변경 사항에 대해 싱크를 진행합니다.\n   * 주어진 변경 파일들 중 체크섬 관리 대상인 것들만 가져다가 싱크를 진행합니다.\n   * 체크섬 파일 업데이트는 여기에서 하지 않습니다. 호출자가 합니다.\n   * @param diffFilePath - 변경 파일들. 프로젝트 루트부터 \"src/\" 또는 \"dist/\"로 시작하는 상대 경로입니다. 예시: \"src/application/user/user.model.ts\"\n   */\n  async syncFromWatcher(event: string, diffFilePath: AbsolutePath): Promise<void> {\n    if (event !== \"change\" && event !== \"add\" && event !== \"unlink\") {\n      return;\n    }\n\n    // 일단 변경된 파일과 dependent 파일들을 invalidate 합니다.\n    // 한 번 이상 import된 친구들에 대해서만 실제 작업이 일어납니다.\n    // 그러니 안심하고 invalidate 해도 됩니다.\n    // 테스트 환경에서는 hot.invalidateFile시 초기 에러가 발생하기 때문에 invalidate 하지 않습니다.\n    if (!isTest()) {\n      const invalidatedPaths = (await hot.invalidateFile(diffFilePath, event)) as AbsolutePath[];\n\n      if (invalidatedPaths.length > 0) {\n        console.log(chalk.bold(`🔄 Invalidated:`));\n\n        for (const invalidatedPath of invalidatedPaths) {\n          try {\n            // 만약 model.ts 파일이 변경(invalidate)되었다? 그러면 registeredApis 중에서 이 모델에 해당하는 api들은 지워줘요.\n            // registeredApis는 통으로 다 날려버릴 수 없습니다. registeredApis에 올라오는 친구들은 초기 로드시 또는 HMR시에만 등록되기 때문입니다.\n            // 따라서 model.ts 파일의 변경으로 다음번 새로운 eval이 예상되는 이 시점에서만, 이 모델에서 나온 registeredApis들을 지워줄 수 있습니다.\n            const removedApis = this.removeInvalidatedRegisteredApis(invalidatedPath);\n            if (removedApis.length > 0) {\n              console.log(\n                chalk.blue(`- ${path.relative(Sonamu.apiRootPath, invalidatedPath)}`),\n                chalk.gray(`(with ${removedApis.length} APIs)`),\n              );\n            } else {\n              console.log(chalk.blue(`- ${path.relative(Sonamu.apiRootPath, invalidatedPath)}`));\n            }\n          } catch (e) {\n            console.error(e);\n            console.error(\n              chalk.red(`Failed to remove invalidated registered APIs for ${invalidatedPath}`),\n            );\n          }\n        }\n      }\n    }\n\n    const isInCheckPatternGroup = Object.values(getChecksumPatternGroupInAbsolutePath()).some(\n      (pattern) => minimatch(diffFilePath, pattern),\n    );\n\n    // 할 일(sync)이 있으면 합니다.\n    if (isInCheckPatternGroup) {\n      await this.doSyncActions([diffFilePath]);\n    }\n\n    // 싱크 작업이 끝나면 모든 모듈을 로드합니다.\n    // hmr-hook에 의해 invalidate된 부분들이 아니라면 캐시 그대로 유지합니다.\n    await this.autoloadTypes();\n    await this.autoloadModels();\n    await this.autoloadApis();\n    await this.autoloadWorkflows();\n\n    this.eventEmitter.emit(\"onHMRCompleted\");\n  }\n\n  removeInvalidatedRegisteredApis(\n    invalidatedPath: AbsolutePath,\n  ): (typeof registeredApis)[number][] {\n    if (!invalidatedPath.endsWith(\".model.ts\" /*소스 코드를 다루는 상황이니 .ts 경로로 봅니다.*/)) {\n      return [];\n    }\n\n    const entityId = EntityManager.getEntityIdFromPath(invalidatedPath);\n    const toRemove = registeredApis.filter((api) => api.modelName === `${entityId}Model`);\n    for (const api of toRemove) {\n      registeredApis.splice(registeredApis.indexOf(api), 1);\n    }\n\n    return toRemove;\n  }\n\n  async copySharedToTargets(targets: string[]): Promise<void> {\n    for (const target of targets) {\n      // 지금 가져가려는 이 파일은 Sonamu 코드베이스의 일부입니다.\n      // 그런데 dist 속 빌드된 소스 코드 파일이 필요한 것이 아니고, src에만 있는 텍스트 파일이 필요합니다.\n      // 따라서 /src/에서 찾습니다.\n      const srcPath = path.join(\n        import.meta.dirname.replace(\"/dist/\", \"/src/\"),\n        `../shared/${target}.shared.ts.txt`,\n      );\n      if (!(await exists(srcPath))) {\n        continue;\n      }\n      if (!(await exists(path.join(Sonamu.appRootPath, target)))) {\n        throw new Error(\n          `Tried to copy sonamu.shared.ts to target '${target}' but the target directory does not exist. Please check your project directory structure.`,\n        );\n      }\n\n      // 이건 프로젝트에 .ts 소스 코드 파일을 생성하는 것이므로 src의 .ts 경로로 갑니다.\n      const destPath = path.join(Sonamu.appRootPath, target, \"src/services/sonamu.shared.ts\");\n\n      // 정말 혹시나지만 target 디렉토리는 있어도 src/services 디렉토리는 없을 수 있으므로 미리 생성해줍니다.\n      if (!(await exists(path.dirname(destPath)))) {\n        await mkdir(path.dirname(destPath), { recursive: true });\n        console.warn(`Created directory '${path.dirname(destPath)}' because it did not exist.`);\n      }\n\n      if (await areFilesSame(srcPath, destPath)) {\n        continue;\n      }\n\n      await writeFile(destPath, await readFile(srcPath));\n\n      !isTest() &&\n        console.log(\n          chalk.bold(\"Copied: \") + chalk.blue(path.relative(Sonamu.appRootPath, destPath)),\n        );\n    }\n  }\n\n  async autoloadTypes() {\n    this.types = await loadTypes();\n  }\n\n  async autoloadModels() {\n    this.models = await loadModels();\n  }\n\n  async autoloadApis() {\n    this.apis = await loadApis();\n  }\n\n  async autoloadWorkflows() {\n    this.workflows = await loadWorkflows();\n    await Sonamu.workflows.synchronize(this.workflows);\n  }\n\n  /**\n   * 실제 싱크를 수행하는 본체입니다.\n   * 변경된 파일들을 타입별로 분류하고 각 타입에 맞는 액션을 실행합니다.\n   * @param diffFilePaths - 변경된 파일들의 절대 경로 목록\n   * @returns diffTypes - 변경된 파일의 타입 목록 (entity, types, model 등)\n   */\n  async doSyncActions(diffFilePaths: AbsolutePath[]): Promise<{ diffTypes: string[] }> {\n    const diffGroups = this.calculateDiffGroups(diffFilePaths);\n    const diffTypes = Object.keys(diffGroups);\n\n    // 트리거: entity, types\n    // 액션: 스키마 생성\n    if (diffTypes.includes(\"entity\")) {\n      await this.handleEntityChange(diffGroups, diffTypes);\n    }\n\n    // 트리거: types, enums, generated 변경시\n    // 액션: 파일 싱크 types, enums, generated\n    if (\n      diffTypes.includes(\"types\") ||\n      diffTypes.includes(\"functions\") ||\n      diffTypes.includes(\"generated\")\n    ) {\n      await this.handleTypesOrFunctionsOrGeneratedChange(diffGroups);\n    }\n\n    // 트리거: model\n    if (diffTypes.includes(\"model\") || diffTypes.includes(\"frame\")) {\n      await this.handleModelOrFrameChange(diffGroups);\n    }\n\n    // 트리거: config\n    if (diffTypes.includes(\"config\")) {\n      await this.actionSyncConfig();\n    }\n\n    // 트리거: workflow\n    if (diffTypes.includes(\"workflow\")) {\n      await this.autoloadWorkflows();\n    }\n\n    return {\n      diffTypes,\n    };\n  }\n\n  calculateDiffGroups(diffFiles: AbsolutePath[]): DiffGroups {\n    return group(diffFiles, (r) => {\n      const matched = r.match(/\\.(model|types|functions|entity|generated|frame|config)\\.[tj]s/);\n      return matched?.[1] ?? \"unknown\";\n    }) as unknown as DiffGroups;\n  }\n\n  async handleEntityChange(diffGroups: DiffGroups, diffTypes: string[]): Promise<void> {\n    Naite.t(\"handleEntityChange\", { diffGroups, diffTypes });\n\n    await EntityManager.reload();\n\n    // types 생성(entity 새로 추가된 경우)\n    // parentId가 없고, types가 없는 경우에만 생성\n    const entityId = EntityManager.getEntityIdFromPath(diffGroups.entity?.[0]);\n\n    if (entityId) {\n      const entity = EntityManager.get(entityId);\n      // 프로젝트에 생성되어야 하는 .ts 파일의 경로입니다.\n      const typeFilePath = path.join(\n        Sonamu.apiRootPath,\n        `src/application/${entity.names.fs}/${entity.names.fs}.types.ts`,\n      );\n      if (entity.parentId === undefined && !(await exists(typeFilePath))) {\n        await generateTemplate(\"init_types\", { entityId });\n      }\n    }\n\n    await this.actionGenerateSchemas();\n\n    diffGroups.generated = unique([\n      ...(diffGroups.generated ?? []),\n      path.join(Sonamu.apiRootPath, \"src/application/sonamu.generated.ts\") as AbsolutePath,\n    ]);\n    diffTypes.push(\"generated\");\n  }\n\n  async handleTypesOrFunctionsOrGeneratedChange(diffGroups: DiffGroups): Promise<FileType[]> {\n    const tsPaths = unique([\n      ...(diffGroups.types ?? []),\n      ...(diffGroups.functions ?? []),\n      ...(diffGroups.generated ?? []),\n    ]);\n    Naite.t(\"handleTypesOrFunctionsOrGeneratedChange\", { diffGroups });\n\n    // console.log(\n    //   chalk.gray(\n    //     `[Processing] Handling types/functions/generated changes: ${tsPaths.map((p) => path.relative(Sonamu.apiRootPath, p)).join(\", \")}`\n    //   )\n    // );\n\n    await this.actionSyncFilesToTargets(tsPaths);\n\n    return [];\n  }\n\n  async handleModelOrFrameChange(diffGroups: DiffGroups): Promise<void> {\n    Naite.t(\"handleModelOrFrameChange\", { diffGroups });\n    const mergedGroup = [...(diffGroups.model ?? []), ...(diffGroups.frame ?? [])];\n\n    // console.log(\n    //   chalk.gray(\n    //     `[Processing] Handling model/frame changes: ${mergedGroup.map((p) => path.relative(Sonamu.apiRootPath, p)).join(\", \")}`\n    //   )\n    // );\n\n    // generated_http.template.ts에서 syncer.types를 씁니다.\n    // service.template.ts에서 syncer.apis를 씁니다.\n    await this.autoloadModels();\n    await this.autoloadTypes();\n    await this.autoloadApis();\n\n    const params: {\n      namesRecord: EntityNamesRecord;\n    }[] = mergedGroup.map((modelPath) => {\n      if (modelPath.endsWith(\".model.ts\")) {\n        const entityId = EntityManager.getEntityIdFromPath(modelPath);\n        assert(entityId);\n        return {\n          namesRecord: EntityManager.getNamesFromId(entityId),\n        };\n      }\n      if (modelPath.endsWith(\".frame.ts\")) {\n        const [, frameName] = modelPath.match(/.+\\/(.+)\\.frame\\.ts$/) ?? [];\n        assert(frameName);\n        // frameName을 PascalCase로 변환 (dashboard -> Dashboard)\n        const frameId = inflection.camelize(frameName);\n        return {\n          namesRecord: EntityManager.getNamesFromId(frameId),\n        };\n      }\n      throw new Error(\"not reachable\");\n    });\n\n    await this.actionGenerateServices(params);\n    await this.actionGenerateHttps();\n  }\n\n  // web/.sonamu.env 에 현재 설정값 저장\n  async actionSyncConfig() {\n    const { host, port } = Sonamu.config.server.listen ?? {};\n    const content = `API_HOST=${host ?? \"localhost\"}\\nAPI_PORT=${port ?? 3000}`;\n\n    Naite.t(\"actionSyncConfig\", { content });\n    await Promise.all(\n      Sonamu.config.sync.targets.map(async (target) => {\n        await writeFile(path.join(Sonamu.appRootPath, target, \".sonamu.env\"), content);\n      }),\n    );\n  }\n\n  /**\n   * sonamu.generated.ts와 sonamu.generated.sso.ts를 생성합니다.\n   * @returns 생성된 파일 경로 배열.\n   */\n  async actionGenerateSchemas(): Promise<AbsolutePath[]> {\n    return (\n      await Promise.all([\n        generateTemplate(\"generated_sso\", {}, { overwrite: true }),\n        generateTemplate(\"generated\", {}, { overwrite: true }),\n      ])\n    )\n      .flat()\n      .flat();\n  }\n\n  /**\n   * *.service.ts를 생성합니다.\n   * @param paramsArray\n   * @returns 생성된 파일 경로 배열.\n   */\n  async actionGenerateServices(\n    paramsArray: {\n      namesRecord: EntityNamesRecord;\n    }[],\n  ): Promise<string[]> {\n    Naite.t(\"actionGenerateServices\", paramsArray);\n    return (\n      await Promise.all(\n        paramsArray.map(async (params) =>\n          generateTemplate(\"service\", params as TemplateOptions[\"service\"], {\n            overwrite: true,\n          }),\n        ),\n      )\n    )\n      .flat()\n      .flat();\n  }\n\n  /**\n   * sonamu.generated.http를 생성합니다.\n   * @returns 생성된 파일 경로.\n   */\n  async actionGenerateHttps(): Promise<AbsolutePath> {\n    const [res] = await generateTemplate(\n      \"generated_http\",\n      { entityId: \"dummy\" },\n      { overwrite: true },\n    );\n    assert(res);\n    return res;\n  }\n\n  /**\n   * *.types.ts, *.functions.ts, *.generated.ts를 타겟 디렉토리에 복사합니다.\n   * @param tsPaths\n   * @returns 복사된 파일 경로 배열.\n   */\n  async actionSyncFilesToTargets(tsPaths: AbsolutePath[]): Promise<string[]> {\n    const { targets } = Sonamu.config.sync;\n    const { dir: apiDir } = Sonamu.config.api;\n\n    return (\n      await Promise.all(\n        targets.map(async (target) =>\n          Promise.all(\n            tsPaths.map(async (realSrc) => {\n              const dst = realSrc\n                .replace(`/${apiDir}/`, `/${target}/`)\n                .replace(\"/application/\", \"/services/\");\n              const dir = dirname(dst);\n              if (!(await exists(dir))) {\n                await mkdir(dir, { recursive: true });\n              }\n              !isTest() &&\n                console.log(\n                  chalk.bold(\"Copied: \") + chalk.blue(dst.replace(`${Sonamu.appRootPath}/`, \"\")),\n                );\n              await this.copyFileWithReplaceCoreToShared(realSrc, dst);\n              return dst;\n            }),\n          ),\n        ),\n      )\n    ).flat();\n  }\n\n  private async copyFileWithReplaceCoreToShared(fromPath: string, toPath: string) {\n    if (!(await exists(fromPath))) {\n      return;\n    }\n\n    const oldFileContent = (await readFile(fromPath)).toString();\n\n    const newFileContent = (() => {\n      // web이나 app 등에는 sonamu가 없습니다.\n      // 따라서 sonamu에 대한 import는 함께 복사되는 sonamu.shared.ts에 대한 import로 치환해야 합니다.\n      // 문제는 리소스 종류에 따라 sonamu.shared.ts로 가는 경로가 다르다는 점입니다.\n      // 예를 들어 sonamu.generated.ts 입장에서 sonamu.shared.ts는 같은 디렉토리에 있으니 ./sonamu.shared로 치환하면 되지만,\n      // user.types.ts 입장에서 sonamu.shared.ts는 상위 디렉토리에 있으니 ../sonamu.shared로 치환해야 합니다.\n      // 이 문제를 해결하기 위해 복사하고자 하는 리소스의 경로(toPath)를 기준으로 sonamu.shared.ts가 있는 디렉토리를 찾아서 상대 경로를 계산하도록 하였습니다.\n      const servicesDir = toPath.replace(/\\/services\\/.*$/, \"/services\");\n      const fileDir = dirname(toPath);\n      const relativePath = path.relative(fileDir, servicesDir);\n      const sharedPath = relativePath === \"\" ? \"./sonamu.shared\" : `${relativePath}/sonamu.shared`;\n\n      const nfc = oldFileContent.replace(/from \"sonamu\"/g, `from \"${sharedPath}\"`);\n      return nfc;\n    })();\n    return writeFile(toPath, newFileContent);\n  }\n\n  /**\n   * 주어진 엔티티와 템플릿 키에 대해, 생성된 코드가 존재하는지 확인합니다.\n   * @param entityId 엔티티 ID\n   * @param templateKey 템플릿 키\n   * @param enumId 열거형 ID\n   * @returns 생성된 코드가 존재하는지 여부\n   */\n  async checkExistsGenCode(\n    entityId: string,\n    templateKey: TemplateKey,\n    enumId?: string,\n  ): Promise<{ subPath: string; fullPath: string; isExists: boolean }> {\n    const { target, path: genPath } = TemplateManager.get(templateKey).getTargetAndPath(\n      EntityManager.getNamesFromId(entityId),\n      enumId,\n    );\n\n    const subPath = path.join(target, genPath);\n    const fullPath = path.join(Sonamu.appRootPath, subPath);\n    return {\n      subPath,\n      fullPath,\n      isExists: await exists(fullPath),\n    };\n  }\n\n  /**\n   * 주어진 엔티티와 열거형에 대해, 생성된 코드가 존재하는지 확인합니다.\n   * @param entityId 엔티티 ID\n   * @param enums 열거형 레이블\n   * @returns 생성된 코드가 존재하는지 여부\n   */\n  async checkExists(\n    entityId: string,\n    enums: {\n      [name: string]: z.ZodEnum<Readonly<Record<string, string | number>>>;\n    },\n  ): Promise<Record<`${TemplateKey}${string}`, boolean>> {\n    const keys: TemplateKey[] = TemplateKey.options;\n    const names = EntityManager.getNamesFromId(entityId);\n    const enumsKeys = Object.keys(enums).filter((name) => name !== names.constant);\n\n    return await reduceAsync(\n      keys,\n      async (result, key) => {\n        const tpl = TemplateManager.get(key);\n        if (key.startsWith(\"view_enums\")) {\n          await mapAsync(enumsKeys, async (componentId) => {\n            const { target, path: p } = tpl.getTargetAndPath(names, componentId);\n            result[`${key}__${componentId}`] = await exists(\n              path.join(Sonamu.appRootPath, target, p),\n            );\n          });\n          return result;\n        }\n\n        const { target, path: p } = tpl.getTargetAndPath(names);\n        const { targets } = Sonamu.config.sync;\n        if (target.includes(\":target\")) {\n          await mapAsync(targets, async (t) => {\n            result[`${key}__${t}`] = await exists(\n              path.join(Sonamu.appRootPath, target.replace(\":target\", t), p),\n            );\n          });\n        } else {\n          result[key] = await exists(path.join(Sonamu.appRootPath, target, p));\n        }\n\n        return result;\n      },\n      {} as Record<`${TemplateKey}${string}`, boolean>,\n    );\n  }\n\n  /**\n   * 하위호환용 프록시 메소드입니다.\n   */\n  async createEntity(form: TemplateOptions[\"entity\"]) {\n    return await createEntity(form);\n  }\n\n  /**\n   * 하위호환용 프록시 메소드입니다.\n   */\n  async delEntity(entityId: string): Promise<{ delPaths: string[] }> {\n    return await delEntity(entityId);\n  }\n\n  /**\n   * 하위호환용 프록시 메소드입니다.\n   */\n  async generateTemplate<T extends TemplateKey>(\n    key: T,\n    templateOptions: TemplateOptions[T],\n    _generateOptions?: GenerateOptions,\n  ): Promise<AbsolutePath[]> {\n    return await generateTemplate(key, templateOptions, _generateOptions);\n  }\n\n  /**\n   * 하위호환용 프록시 메소드입니다.\n   */\n  async renderTemplate<T extends keyof TemplateOptions>(\n    key: T,\n    templateOptions: TemplateOptions[T],\n  ): Promise<PathAndCode[]> {\n    return await renderTemplate(key, templateOptions);\n  }\n\n  /**\n   * 하위호환용 프록시 메소드입니다.\n   */\n  async renewChecksums(): Promise<void> {\n    return await renewChecksums();\n  }\n}\n"],"names":["hot","assert","chalk","EventEmitter","mkdir","readFile","writeFile","inflection","minimatch","path","dirname","group","unique","registeredApis","Sonamu","EntityManager","Naite","TemplateManager","TemplateKey","mapAsync","reduceAsync","centerText","isTest","exists","runWithGracefulShutdown","areFilesSame","findChangedFilesUsingChecksums","renewChecksums","generateTemplate","renderTemplate","createEntity","delEntity","getChecksumPatternGroupInAbsolutePath","loadApis","loadModels","loadTypes","loadWorkflows","Syncer","apis","types","models","workflows","Map","isSyncing","eventEmitter","sync","targets","config","copySharedToTargets","changedFiles","length","console","log","black","bgGreen","doSyncActions","whenThisHappens","waitForUpTo","syncFromWatcher","event","diffFilePath","invalidatedPaths","invalidateFile","bold","invalidatedPath","removedApis","removeInvalidatedRegisteredApis","blue","relative","apiRootPath","gray","e","error","red","isInCheckPatternGroup","Object","values","some","pattern","autoloadTypes","autoloadModels","autoloadApis","autoloadWorkflows","emit","endsWith","entityId","getEntityIdFromPath","toRemove","filter","api","modelName","splice","indexOf","target","srcPath","join","replace","appRootPath","Error","destPath","recursive","warn","synchronize","diffFilePaths","diffGroups","calculateDiffGroups","diffTypes","keys","includes","handleEntityChange","handleTypesOrFunctionsOrGeneratedChange","handleModelOrFrameChange","actionSyncConfig","diffFiles","r","matched","match","t","reload","entity","get","typeFilePath","names","fs","parentId","undefined","actionGenerateSchemas","generated","push","tsPaths","functions","actionSyncFilesToTargets","mergedGroup","model","frame","params","map","modelPath","namesRecord","getNamesFromId","frameName","frameId","camelize","actionGenerateServices","actionGenerateHttps","host","port","server","listen","content","Promise","all","overwrite","flat","paramsArray","res","dir","apiDir","realSrc","dst","copyFileWithReplaceCoreToShared","fromPath","toPath","oldFileContent","toString","newFileContent","servicesDir","fileDir","relativePath","sharedPath","nfc","checkExistsGenCode","templateKey","enumId","genPath","getTargetAndPath","subPath","fullPath","isExists","checkExists","enums","options","enumsKeys","name","constant","result","key","tpl","startsWith","componentId","p","form","templateOptions","_generateOptions"],"mappings":"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"}
|
|
458
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/syncer/syncer.ts"],"sourcesContent":["import { hot } from \"@sonamu-kit/hmr-hook\";\nimport assert from \"assert\";\nimport chalk from \"chalk\";\nimport { EventEmitter } from \"events\";\nimport { mkdir, readFile, writeFile } from \"fs/promises\";\nimport inflection from \"inflection\";\nimport { minimatch } from \"minimatch\";\nimport path, { dirname } from \"path\";\nimport { group, unique } from \"radashi\";\nimport type { z } from \"zod\";\nimport type { WorkflowMetadata } from \"..\";\nimport { registeredApis } from \"../api/decorators\";\nimport { Sonamu } from \"../api/sonamu\";\nimport { EntityManager, type EntityNamesRecord } from \"../entity/entity-manager\";\nimport { Naite } from \"../naite/naite\";\nimport { TemplateManager } from \"../template/template-manager\";\nimport type { GenerateOptions, PathAndCode } from \"../types/types\";\nimport { TemplateKey, type TemplateOptions } from \"../types/types\";\nimport { mapAsync, reduceAsync } from \"../utils/async-utils\";\nimport { centerText } from \"../utils/console-util\";\nimport { isTest } from \"../utils/controller\";\nimport { exists } from \"../utils/fs-utils\";\nimport type { AbsolutePath } from \"../utils/path-utils\";\nimport { runWithGracefulShutdown } from \"../utils/process-utils\";\nimport { areFilesSame, findChangedFilesUsingChecksums, renewChecksums } from \"./checksum\";\nimport { generateTemplate, renderTemplate } from \"./code-generator\";\nimport { createEntity, delEntity } from \"./entity-operations\";\nimport { type FileType, getChecksumPatternGroupInAbsolutePath } from \"./file-patterns\";\nimport {\n  type LoadedApis,\n  type LoadedModels,\n  type LoadedTypes,\n  loadApis,\n  loadModels,\n  loadTypes,\n  loadWorkflows,\n} from \"./module-loader\";\n\ntype DiffGroups = {\n  [key in FileType]: AbsolutePath[];\n};\n\nexport class Syncer {\n  apis: LoadedApis = [];\n  types: LoadedTypes = {};\n  models: LoadedModels = {};\n  workflows: Map<string, WorkflowMetadata[]> = new Map();\n  isSyncing: boolean = false;\n  eventEmitter: EventEmitter = new EventEmitter();\n\n  /**\n   * 체크섬이 변경된 부분에 대해 싱크를 진행합니다.\n   * 다만 sonamu.shared.ts는 체크섬 비교 없이 무조건 싱크(복사)합니다.\n   * @returns\n   */\n  async sync(): Promise<void> {\n    const { targets } = Sonamu.config.sync;\n\n    // sonamu.shared.ts는 무조건 싱크(복사)합니다.\n    await this.copySharedToTargets(targets);\n\n    // 그 다음부터는 변경된 파일을 찾아서 동기화 작업을 실행합니다.\n    const changedFiles = await findChangedFilesUsingChecksums();\n    if (changedFiles.length === 0) {\n      console.log(chalk.black.bgGreen(centerText(\"All files are synced!\")));\n      return;\n    }\n\n    // 만약 싱크 중에 프로세스가 죽으면 꼬여버리기 때문에,\n    // 시그널에도 잠시 버틸 수 있는 환경 속에서 싱크를 실행합니다.\n    await runWithGracefulShutdown(\n      async () => {\n        // 얘가 싱크 작업 수행하는 본체입니다.\n        await this.doSyncActions(changedFiles);\n\n        // 싱크 액션이 끝나면 항상 체크섬을 다시 갱신합니다.\n        await renewChecksums();\n      },\n      { whenThisHappens: \"SIGUSR2\", waitForUpTo: 20000 },\n    );\n  }\n\n  /**\n   * Watcher가 감지한 파일 변경 사항에 대해 싱크를 진행합니다.\n   * 주어진 변경 파일들 중 체크섬 관리 대상인 것들만 가져다가 싱크를 진행합니다.\n   * 체크섬 파일 업데이트는 여기에서 하지 않습니다. 호출자가 합니다.\n   * @param diffFilePath - 변경 파일들. 프로젝트 루트부터 \"src/\" 또는 \"dist/\"로 시작하는 상대 경로입니다. 예시: \"src/application/user/user.model.ts\"\n   */\n  async syncFromWatcher(event: string, diffFilePath: AbsolutePath): Promise<void> {\n    if (event !== \"change\" && event !== \"add\" && event !== \"unlink\") {\n      return;\n    }\n\n    // 일단 변경된 파일과 dependent 파일들을 invalidate 합니다.\n    // 한 번 이상 import된 친구들에 대해서만 실제 작업이 일어납니다.\n    // 그러니 안심하고 invalidate 해도 됩니다.\n    // 테스트 환경에서는 hot.invalidateFile시 초기 에러가 발생하기 때문에 invalidate 하지 않습니다.\n    if (!isTest()) {\n      const invalidatedPaths = (await hot.invalidateFile(diffFilePath, event)) as AbsolutePath[];\n\n      if (invalidatedPaths.length > 0) {\n        console.log(chalk.bold(`🔄 Invalidated:`));\n\n        for (const invalidatedPath of invalidatedPaths) {\n          try {\n            // 만약 model.ts 파일이 변경(invalidate)되었다? 그러면 registeredApis 중에서 이 모델에 해당하는 api들은 지워줘요.\n            // registeredApis는 통으로 다 날려버릴 수 없습니다. registeredApis에 올라오는 친구들은 초기 로드시 또는 HMR시에만 등록되기 때문입니다.\n            // 따라서 model.ts 파일의 변경으로 다음번 새로운 eval이 예상되는 이 시점에서만, 이 모델에서 나온 registeredApis들을 지워줄 수 있습니다.\n            const removedApis = this.removeInvalidatedRegisteredApis(invalidatedPath);\n            if (removedApis.length > 0) {\n              console.log(\n                chalk.blue(`- ${path.relative(Sonamu.apiRootPath, invalidatedPath)}`),\n                chalk.gray(`(with ${removedApis.length} APIs)`),\n              );\n            } else {\n              console.log(chalk.blue(`- ${path.relative(Sonamu.apiRootPath, invalidatedPath)}`));\n            }\n          } catch (e) {\n            console.error(e);\n            console.error(\n              chalk.red(`Failed to remove invalidated registered APIs for ${invalidatedPath}`),\n            );\n          }\n        }\n      }\n    }\n\n    const isInCheckPatternGroup = Object.values(getChecksumPatternGroupInAbsolutePath()).some(\n      (pattern) => minimatch(diffFilePath, pattern),\n    );\n\n    // 할 일(sync)이 있으면 합니다.\n    if (isInCheckPatternGroup) {\n      await this.doSyncActions([diffFilePath]);\n    }\n\n    // 싱크 작업이 끝나면 모든 모듈을 로드합니다.\n    // hmr-hook에 의해 invalidate된 부분들이 아니라면 캐시 그대로 유지합니다.\n    await this.autoloadTypes();\n    await this.autoloadModels();\n    await this.autoloadApis();\n    await this.autoloadWorkflows();\n\n    this.eventEmitter.emit(\"onHMRCompleted\");\n  }\n\n  removeInvalidatedRegisteredApis(\n    invalidatedPath: AbsolutePath,\n  ): (typeof registeredApis)[number][] {\n    if (!invalidatedPath.endsWith(\".model.ts\" /*소스 코드를 다루는 상황이니 .ts 경로로 봅니다.*/)) {\n      return [];\n    }\n\n    const entityId = EntityManager.getEntityIdFromPath(invalidatedPath);\n    const toRemove = registeredApis.filter((api) => api.modelName === `${entityId}Model`);\n    for (const api of toRemove) {\n      registeredApis.splice(registeredApis.indexOf(api), 1);\n    }\n\n    return toRemove;\n  }\n\n  async copySharedToTargets(targets: string[]): Promise<void> {\n    // 특정 변수 치환을 위해서 사용합니다.\n    const convertMap = {\n      baseUrl:\n        Sonamu.config.server.baseUrl ??\n        `http://${Sonamu.config.server.listen?.host ?? \"localhost\"}:${Sonamu.config.server.listen?.port ?? 3000}`,\n    };\n\n    for (const target of targets) {\n      // 지금 가져가려는 이 파일은 Sonamu 코드베이스의 일부입니다.\n      // 그런데 dist 속 빌드된 소스 코드 파일이 필요한 것이 아니고, src에만 있는 텍스트 파일이 필요합니다.\n      // 따라서 /src/에서 찾습니다.\n      const srcPath = path.join(\n        import.meta.dirname.replace(\"/dist/\", \"/src/\"),\n        `../shared/${target}.shared.ts.txt`,\n      );\n      if (!(await exists(srcPath))) {\n        continue;\n      }\n      if (!(await exists(path.join(Sonamu.appRootPath, target)))) {\n        throw new Error(\n          `Tried to copy sonamu.shared.ts to target '${target}' but the target directory does not exist. Please check your project directory structure.`,\n        );\n      }\n\n      const fullText = await readFile(srcPath, \"utf-8\");\n      const convertedText = Object.entries(convertMap).reduce(\n        (acc, [key, value]) => acc.replace(`$[[${key}]]`, value),\n        fullText,\n      );\n\n      // 이건 프로젝트에 .ts 소스 코드 파일을 생성하는 것이므로 src의 .ts 경로로 갑니다.\n      const destPath = path.join(Sonamu.appRootPath, target, \"src/services/sonamu.shared.ts\");\n\n      // 정말 혹시나지만 target 디렉토리는 있어도 src/services 디렉토리는 없을 수 있으므로 미리 생성해줍니다.\n      if (!(await exists(path.dirname(destPath)))) {\n        await mkdir(path.dirname(destPath), { recursive: true });\n        console.warn(`Created directory '${path.dirname(destPath)}' because it did not exist.`);\n      }\n\n      if (await areFilesSame({ data: convertedText }, { path: destPath })) {\n        continue;\n      }\n\n      await writeFile(destPath, convertedText);\n      !isTest() &&\n        console.log(\n          chalk.bold(\"Copied: \") + chalk.blue(path.relative(Sonamu.appRootPath, destPath)),\n        );\n    }\n  }\n\n  async autoloadTypes() {\n    this.types = await loadTypes();\n  }\n\n  async autoloadModels() {\n    this.models = await loadModels();\n  }\n\n  async autoloadApis() {\n    this.apis = await loadApis();\n  }\n\n  async autoloadWorkflows() {\n    this.workflows = await loadWorkflows();\n    await Sonamu.workflows.synchronize(this.workflows);\n  }\n\n  /**\n   * 실제 싱크를 수행하는 본체입니다.\n   * 변경된 파일들을 타입별로 분류하고 각 타입에 맞는 액션을 실행합니다.\n   * @param diffFilePaths - 변경된 파일들의 절대 경로 목록\n   * @returns diffTypes - 변경된 파일의 타입 목록 (entity, types, model 등)\n   */\n  async doSyncActions(diffFilePaths: AbsolutePath[]): Promise<{ diffTypes: string[] }> {\n    const diffGroups = this.calculateDiffGroups(diffFilePaths);\n    const diffTypes = Object.keys(diffGroups);\n\n    // 트리거: entity, types\n    // 액션: 스키마 생성\n    if (diffTypes.includes(\"entity\")) {\n      await this.handleEntityChange(diffGroups, diffTypes);\n    }\n\n    // 트리거: types, enums, generated 변경시\n    // 액션: 파일 싱크 types, enums, generated\n    if (\n      diffTypes.includes(\"types\") ||\n      diffTypes.includes(\"functions\") ||\n      diffTypes.includes(\"generated\")\n    ) {\n      await this.handleTypesOrFunctionsOrGeneratedChange(diffGroups);\n    }\n\n    // 트리거: model\n    if (diffTypes.includes(\"model\") || diffTypes.includes(\"frame\")) {\n      await this.handleModelOrFrameChange(diffGroups);\n    }\n\n    // 트리거: config\n    if (diffTypes.includes(\"config\")) {\n      await this.actionSyncConfig();\n    }\n\n    // 트리거: workflow\n    if (diffTypes.includes(\"workflow\")) {\n      await this.autoloadWorkflows();\n    }\n\n    return {\n      diffTypes,\n    };\n  }\n\n  calculateDiffGroups(diffFiles: AbsolutePath[]): DiffGroups {\n    return group(diffFiles, (r) => {\n      const matched = r.match(/\\.(model|types|functions|entity|generated|frame|config)\\.[tj]s/);\n      return matched?.[1] ?? \"unknown\";\n    }) as unknown as DiffGroups;\n  }\n\n  async handleEntityChange(diffGroups: DiffGroups, diffTypes: string[]): Promise<void> {\n    Naite.t(\"handleEntityChange\", { diffGroups, diffTypes });\n\n    await EntityManager.reload();\n\n    // types 생성(entity 새로 추가된 경우)\n    // parentId가 없고, types가 없는 경우에만 생성\n    const entityId = EntityManager.getEntityIdFromPath(diffGroups.entity?.[0]);\n\n    if (entityId) {\n      const entity = EntityManager.get(entityId);\n      // 프로젝트에 생성되어야 하는 .ts 파일의 경로입니다.\n      const typeFilePath = path.join(\n        Sonamu.apiRootPath,\n        `src/application/${entity.names.fs}/${entity.names.fs}.types.ts`,\n      );\n      if (entity.parentId === undefined && !(await exists(typeFilePath))) {\n        await generateTemplate(\"init_types\", { entityId });\n      }\n    }\n\n    await this.actionGenerateSchemas();\n\n    diffGroups.generated = unique([\n      ...(diffGroups.generated ?? []),\n      path.join(Sonamu.apiRootPath, \"src/application/sonamu.generated.ts\") as AbsolutePath,\n    ]);\n    diffTypes.push(\"generated\");\n  }\n\n  async handleTypesOrFunctionsOrGeneratedChange(diffGroups: DiffGroups): Promise<FileType[]> {\n    const tsPaths = unique([\n      ...(diffGroups.types ?? []),\n      ...(diffGroups.functions ?? []),\n      ...(diffGroups.generated ?? []),\n    ]);\n    Naite.t(\"handleTypesOrFunctionsOrGeneratedChange\", { diffGroups });\n\n    // console.log(\n    //   chalk.gray(\n    //     `[Processing] Handling types/functions/generated changes: ${tsPaths.map((p) => path.relative(Sonamu.apiRootPath, p)).join(\", \")}`\n    //   )\n    // );\n\n    await this.actionSyncFilesToTargets(tsPaths);\n\n    return [];\n  }\n\n  async handleModelOrFrameChange(diffGroups: DiffGroups): Promise<void> {\n    Naite.t(\"handleModelOrFrameChange\", { diffGroups });\n    const mergedGroup = [...(diffGroups.model ?? []), ...(diffGroups.frame ?? [])];\n\n    // console.log(\n    //   chalk.gray(\n    //     `[Processing] Handling model/frame changes: ${mergedGroup.map((p) => path.relative(Sonamu.apiRootPath, p)).join(\", \")}`\n    //   )\n    // );\n\n    // generated_http.template.ts에서 syncer.types를 씁니다.\n    // service.template.ts에서 syncer.apis를 씁니다.\n    await this.autoloadModels();\n    await this.autoloadTypes();\n    await this.autoloadApis();\n\n    const params: {\n      namesRecord: EntityNamesRecord;\n    }[] = mergedGroup.map((modelPath) => {\n      if (modelPath.endsWith(\".model.ts\")) {\n        const entityId = EntityManager.getEntityIdFromPath(modelPath);\n        assert(entityId);\n        return {\n          namesRecord: EntityManager.getNamesFromId(entityId),\n        };\n      }\n      if (modelPath.endsWith(\".frame.ts\")) {\n        const [, frameName] = modelPath.match(/.+\\/(.+)\\.frame\\.ts$/) ?? [];\n        assert(frameName);\n        // frameName을 PascalCase로 변환 (dashboard -> Dashboard)\n        const frameId = inflection.camelize(frameName);\n        return {\n          namesRecord: EntityManager.getNamesFromId(frameId),\n        };\n      }\n      throw new Error(\"not reachable\");\n    });\n\n    await this.actionGenerateServices(params);\n    await this.actionGenerateHttps();\n  }\n\n  // web/.sonamu.env 에 현재 설정값 저장\n  async actionSyncConfig() {\n    const { host, port } = Sonamu.config.server.listen ?? {};\n    const content = `API_HOST=${host ?? \"localhost\"}\\nAPI_PORT=${port ?? 3000}`;\n\n    Naite.t(\"actionSyncConfig\", { content });\n    await Promise.all(\n      Sonamu.config.sync.targets.map(async (target) => {\n        await writeFile(path.join(Sonamu.appRootPath, target, \".sonamu.env\"), content);\n      }),\n    );\n  }\n\n  /**\n   * services.generated.ts를 생성합니다.\n   * @param paramsArray\n   * @returns 생성된 파일 경로 배열.\n   */\n  async actionGenerateServices(\n    paramsArray: {\n      namesRecord: EntityNamesRecord;\n    }[],\n  ): Promise<string[]> {\n    Naite.t(\"actionGenerateServices\", paramsArray);\n\n    // services.generated.ts 통합 파일 생성\n    const servicesFile = await generateTemplate(\n      \"services\",\n      {},\n      {\n        overwrite: true,\n      },\n    );\n\n    return [...servicesFile];\n  }\n\n  /**\n   * sonamu.generated.ts와 sonamu.generated.sso.ts를 생성합니다.\n   * @returns 생성된 파일 경로 배열.\n   */\n  async actionGenerateSchemas(): Promise<AbsolutePath[]> {\n    return (\n      await Promise.all([\n        generateTemplate(\"generated_sso\", {}, { overwrite: true }),\n        generateTemplate(\"generated\", {}, { overwrite: true }),\n      ])\n    )\n      .flat()\n      .flat();\n  }\n\n  /**\n   * sonamu.generated.http를 생성합니다.\n   * @returns 생성된 파일 경로.\n   */\n  async actionGenerateHttps(): Promise<AbsolutePath> {\n    const [res] = await generateTemplate(\n      \"generated_http\",\n      { entityId: \"dummy\" },\n      { overwrite: true },\n    );\n    assert(res);\n    return res;\n  }\n\n  /**\n   * *.types.ts, *.functions.ts, *.generated.ts를 타겟 디렉토리에 복사합니다.\n   * @param tsPaths\n   * @returns 복사된 파일 경로 배열.\n   */\n  async actionSyncFilesToTargets(tsPaths: AbsolutePath[]): Promise<string[]> {\n    const { targets } = Sonamu.config.sync;\n    const { dir: apiDir } = Sonamu.config.api;\n\n    return (\n      await Promise.all(\n        targets.map(async (target) =>\n          Promise.all(\n            tsPaths.map(async (realSrc) => {\n              const dst = realSrc\n                .replace(`/${apiDir}/`, `/${target}/`)\n                .replace(\"/application/\", \"/services/\");\n              const dir = dirname(dst);\n              if (!(await exists(dir))) {\n                await mkdir(dir, { recursive: true });\n              }\n              !isTest() &&\n                console.log(\n                  chalk.bold(\"Copied: \") + chalk.blue(dst.replace(`${Sonamu.appRootPath}/`, \"\")),\n                );\n              await this.copyFileWithReplaceCoreToShared(realSrc, dst);\n              return dst;\n            }),\n          ),\n        ),\n      )\n    ).flat();\n  }\n\n  private async copyFileWithReplaceCoreToShared(fromPath: string, toPath: string) {\n    if (!(await exists(fromPath))) {\n      return;\n    }\n\n    const oldFileContent = (await readFile(fromPath)).toString();\n\n    const newFileContent = (() => {\n      // web이나 app 등에는 sonamu가 없습니다.\n      // 따라서 sonamu에 대한 import는 함께 복사되는 sonamu.shared.ts에 대한 import로 치환해야 합니다.\n      // 문제는 리소스 종류에 따라 sonamu.shared.ts로 가는 경로가 다르다는 점입니다.\n      // 예를 들어 sonamu.generated.ts 입장에서 sonamu.shared.ts는 같은 디렉토리에 있으니 ./sonamu.shared로 치환하면 되지만,\n      // user.types.ts 입장에서 sonamu.shared.ts는 상위 디렉토리에 있으니 ../sonamu.shared로 치환해야 합니다.\n      // 이 문제를 해결하기 위해 복사하고자 하는 리소스의 경로(toPath)를 기준으로 sonamu.shared.ts가 있는 디렉토리를 찾아서 상대 경로를 계산하도록 하였습니다.\n      const servicesDir = toPath.replace(/\\/services\\/.*$/, \"/services\");\n      const fileDir = dirname(toPath);\n      const relativePath = path.relative(fileDir, servicesDir);\n      const sharedPath = relativePath === \"\" ? \"./sonamu.shared\" : `${relativePath}/sonamu.shared`;\n\n      const nfc = oldFileContent.replace(/from \"sonamu\"/g, `from \"${sharedPath}\"`);\n      return nfc;\n    })();\n    return writeFile(toPath, newFileContent);\n  }\n\n  /**\n   * 주어진 엔티티와 템플릿 키에 대해, 생성된 코드가 존재하는지 확인합니다.\n   * @param entityId 엔티티 ID\n   * @param templateKey 템플릿 키\n   * @param enumId 열거형 ID\n   * @returns 생성된 코드가 존재하는지 여부\n   */\n  async checkExistsGenCode(\n    entityId: string,\n    templateKey: TemplateKey,\n    enumId?: string,\n  ): Promise<{ subPath: string; fullPath: string; isExists: boolean }> {\n    const { target, path: genPath } = TemplateManager.get(templateKey).getTargetAndPath(\n      EntityManager.getNamesFromId(entityId),\n      enumId,\n    );\n\n    const subPath = path.join(target, genPath);\n    const fullPath = path.join(Sonamu.appRootPath, subPath);\n    return {\n      subPath,\n      fullPath,\n      isExists: await exists(fullPath),\n    };\n  }\n\n  /**\n   * 주어진 엔티티와 열거형에 대해, 생성된 코드가 존재하는지 확인합니다.\n   * @param entityId 엔티티 ID\n   * @param enums 열거형 레이블\n   * @returns 생성된 코드가 존재하는지 여부\n   */\n  async checkExists(\n    entityId: string,\n    enums: {\n      [name: string]: z.ZodEnum<Readonly<Record<string, string | number>>>;\n    },\n  ): Promise<Record<`${TemplateKey}${string}`, boolean>> {\n    const keys: TemplateKey[] = TemplateKey.options;\n    const names = EntityManager.getNamesFromId(entityId);\n    const enumsKeys = Object.keys(enums).filter((name) => name !== names.constant);\n\n    return await reduceAsync(\n      keys,\n      async (result, key) => {\n        const tpl = TemplateManager.get(key);\n        if (key.startsWith(\"view_enums\")) {\n          await mapAsync(enumsKeys, async (componentId) => {\n            const { target, path: p } = tpl.getTargetAndPath(names, componentId);\n            result[`${key}__${componentId}`] = await exists(\n              path.join(Sonamu.appRootPath, target, p),\n            );\n          });\n          return result;\n        }\n\n        const { target, path: p } = tpl.getTargetAndPath(names);\n        const { targets } = Sonamu.config.sync;\n        if (target.includes(\":target\")) {\n          await mapAsync(targets, async (t) => {\n            result[`${key}__${t}`] = await exists(\n              path.join(Sonamu.appRootPath, target.replace(\":target\", t), p),\n            );\n          });\n        } else {\n          result[key] = await exists(path.join(Sonamu.appRootPath, target, p));\n        }\n\n        return result;\n      },\n      {} as Record<`${TemplateKey}${string}`, boolean>,\n    );\n  }\n\n  /**\n   * 하위호환용 프록시 메소드입니다.\n   */\n  async createEntity(form: TemplateOptions[\"entity\"]) {\n    return await createEntity(form);\n  }\n\n  /**\n   * 하위호환용 프록시 메소드입니다.\n   */\n  async delEntity(entityId: string): Promise<{ delPaths: string[] }> {\n    return await delEntity(entityId);\n  }\n\n  /**\n   * 하위호환용 프록시 메소드입니다.\n   */\n  async generateTemplate<T extends TemplateKey>(\n    key: T,\n    templateOptions: TemplateOptions[T],\n    _generateOptions?: GenerateOptions,\n  ): Promise<AbsolutePath[]> {\n    return await generateTemplate(key, templateOptions, _generateOptions);\n  }\n\n  /**\n   * 하위호환용 프록시 메소드입니다.\n   */\n  async renderTemplate<T extends keyof TemplateOptions>(\n    key: T,\n    templateOptions: TemplateOptions[T],\n  ): Promise<PathAndCode[]> {\n    return await renderTemplate(key, templateOptions);\n  }\n\n  /**\n   * 하위호환용 프록시 메소드입니다.\n   */\n  async renewChecksums(): Promise<void> {\n    return await renewChecksums();\n  }\n}\n"],"names":["hot","assert","chalk","EventEmitter","mkdir","readFile","writeFile","inflection","minimatch","path","dirname","group","unique","registeredApis","Sonamu","EntityManager","Naite","TemplateManager","TemplateKey","mapAsync","reduceAsync","centerText","isTest","exists","runWithGracefulShutdown","areFilesSame","findChangedFilesUsingChecksums","renewChecksums","generateTemplate","renderTemplate","createEntity","delEntity","getChecksumPatternGroupInAbsolutePath","loadApis","loadModels","loadTypes","loadWorkflows","Syncer","apis","types","models","workflows","Map","isSyncing","eventEmitter","sync","targets","config","copySharedToTargets","changedFiles","length","console","log","black","bgGreen","doSyncActions","whenThisHappens","waitForUpTo","syncFromWatcher","event","diffFilePath","invalidatedPaths","invalidateFile","bold","invalidatedPath","removedApis","removeInvalidatedRegisteredApis","blue","relative","apiRootPath","gray","e","error","red","isInCheckPatternGroup","Object","values","some","pattern","autoloadTypes","autoloadModels","autoloadApis","autoloadWorkflows","emit","endsWith","entityId","getEntityIdFromPath","toRemove","filter","api","modelName","splice","indexOf","convertMap","baseUrl","server","listen","host","port","target","srcPath","join","replace","appRootPath","Error","fullText","convertedText","entries","reduce","acc","key","value","destPath","recursive","warn","data","synchronize","diffFilePaths","diffGroups","calculateDiffGroups","diffTypes","keys","includes","handleEntityChange","handleTypesOrFunctionsOrGeneratedChange","handleModelOrFrameChange","actionSyncConfig","diffFiles","r","matched","match","t","reload","entity","get","typeFilePath","names","fs","parentId","undefined","actionGenerateSchemas","generated","push","tsPaths","functions","actionSyncFilesToTargets","mergedGroup","model","frame","params","map","modelPath","namesRecord","getNamesFromId","frameName","frameId","camelize","actionGenerateServices","actionGenerateHttps","content","Promise","all","paramsArray","servicesFile","overwrite","flat","res","dir","apiDir","realSrc","dst","copyFileWithReplaceCoreToShared","fromPath","toPath","oldFileContent","toString","newFileContent","servicesDir","fileDir","relativePath","sharedPath","nfc","checkExistsGenCode","templateKey","enumId","genPath","getTargetAndPath","subPath","fullPath","isExists","checkExists","enums","options","enumsKeys","name","constant","result","tpl","startsWith","componentId","p","form","templateOptions","_generateOptions"],"mappings":"AAAA,SAASA,GAAG,QAAQ,uBAAuB;AAC3C,OAAOC,YAAY,SAAS;AAC5B,OAAOC,WAAW,QAAQ;AAC1B,SAASC,YAAY,QAAQ,SAAS;AACtC,SAASC,KAAK,EAAEC,QAAQ,EAAEC,SAAS,QAAQ,mBAAc;AACzD,OAAOC,gBAAgB,aAAa;AACpC,SAASC,SAAS,QAAQ,YAAY;AACtC,OAAOC,QAAQC,OAAO,QAAQ,OAAO;AACrC,SAASC,KAAK,EAAEC,MAAM,QAAQ,UAAU;AAGxC,SAASC,cAAc,QAAQ,uBAAoB;AACnD,SAASC,MAAM,QAAQ,mBAAgB;AACvC,SAASC,aAAa,QAAgC,8BAA2B;AACjF,SAASC,KAAK,QAAQ,oBAAiB;AACvC,SAASC,eAAe,QAAQ,kCAA+B;AAE/D,SAASC,WAAW,QAA8B,oBAAiB;AACnE,SAASC,QAAQ,EAAEC,WAAW,QAAQ,0BAAuB;AAC7D,SAASC,UAAU,QAAQ,2BAAwB;AACnD,SAASC,MAAM,QAAQ,yBAAsB;AAC7C,SAASC,MAAM,QAAQ,uBAAoB;AAE3C,SAASC,uBAAuB,QAAQ,4BAAyB;AACjE,SAASC,YAAY,EAAEC,8BAA8B,EAAEC,cAAc,QAAQ,gBAAa;AAC1F,SAASC,gBAAgB,EAAEC,cAAc,QAAQ,sBAAmB;AACpE,SAASC,YAAY,EAAEC,SAAS,QAAQ,yBAAsB;AAC9D,SAAwBC,qCAAqC,QAAQ,qBAAkB;AACvF,SAIEC,QAAQ,EACRC,UAAU,EACVC,SAAS,EACTC,aAAa,QACR,qBAAkB;AAMzB,OAAO,MAAMC;IACXC,OAAmB,EAAE,CAAC;IACtBC,QAAqB,CAAC,EAAE;IACxBC,SAAuB,CAAC,EAAE;IAC1BC,YAA6C,IAAIC,MAAM;IACvDC,YAAqB,MAAM;IAC3BC,eAA6B,IAAIzC,eAAe;IAEhD;;;;GAIC,GACD,MAAM0C,OAAsB;QAC1B,MAAM,EAAEC,OAAO,EAAE,GAAGhC,OAAOiC,MAAM,CAACF,IAAI;QAEtC,mCAAmC;QACnC,MAAM,IAAI,CAACG,mBAAmB,CAACF;QAE/B,qCAAqC;QACrC,MAAMG,eAAe,MAAMvB;QAC3B,IAAIuB,aAAaC,MAAM,KAAK,GAAG;YAC7BC,QAAQC,GAAG,CAAClD,MAAMmD,KAAK,CAACC,OAAO,CAACjC,WAAW;YAC3C;QACF;QAEA,gCAAgC;QAChC,qCAAqC;QACrC,MAAMG,wBACJ;YACE,uBAAuB;YACvB,MAAM,IAAI,CAAC+B,aAAa,CAACN;YAEzB,+BAA+B;YAC/B,MAAMtB;QACR,GACA;YAAE6B,iBAAiB;YAAWC,aAAa;QAAM;IAErD;IAEA;;;;;GAKC,GACD,MAAMC,gBAAgBC,KAAa,EAAEC,YAA0B,EAAiB;QAC9E,IAAID,UAAU,YAAYA,UAAU,SAASA,UAAU,UAAU;YAC/D;QACF;QAEA,4CAA4C;QAC5C,yCAAyC;QACzC,8BAA8B;QAC9B,oEAAoE;QACpE,IAAI,CAACrC,UAAU;YACb,MAAMuC,mBAAoB,MAAM7D,IAAI8D,cAAc,CAACF,cAAcD;YAEjE,IAAIE,iBAAiBX,MAAM,GAAG,GAAG;gBAC/BC,QAAQC,GAAG,CAAClD,MAAM6D,IAAI,CAAC,CAAC,eAAe,CAAC;gBAExC,KAAK,MAAMC,mBAAmBH,iBAAkB;oBAC9C,IAAI;wBACF,mFAAmF;wBACnF,4FAA4F;wBAC5F,2FAA2F;wBAC3F,MAAMI,cAAc,IAAI,CAACC,+BAA+B,CAACF;wBACzD,IAAIC,YAAYf,MAAM,GAAG,GAAG;4BAC1BC,QAAQC,GAAG,CACTlD,MAAMiE,IAAI,CAAC,CAAC,EAAE,EAAE1D,KAAK2D,QAAQ,CAACtD,OAAOuD,WAAW,EAAEL,kBAAkB,GACpE9D,MAAMoE,IAAI,CAAC,CAAC,MAAM,EAAEL,YAAYf,MAAM,CAAC,MAAM,CAAC;wBAElD,OAAO;4BACLC,QAAQC,GAAG,CAAClD,MAAMiE,IAAI,CAAC,CAAC,EAAE,EAAE1D,KAAK2D,QAAQ,CAACtD,OAAOuD,WAAW,EAAEL,kBAAkB;wBAClF;oBACF,EAAE,OAAOO,GAAG;wBACVpB,QAAQqB,KAAK,CAACD;wBACdpB,QAAQqB,KAAK,CACXtE,MAAMuE,GAAG,CAAC,CAAC,iDAAiD,EAAET,iBAAiB;oBAEnF;gBACF;YACF;QACF;QAEA,MAAMU,wBAAwBC,OAAOC,MAAM,CAAC5C,yCAAyC6C,IAAI,CACvF,CAACC,UAAYtE,UAAUoD,cAAckB;QAGvC,sBAAsB;QACtB,IAAIJ,uBAAuB;YACzB,MAAM,IAAI,CAACnB,aAAa,CAAC;gBAACK;aAAa;QACzC;QAEA,2BAA2B;QAC3B,mDAAmD;QACnD,MAAM,IAAI,CAACmB,aAAa;QACxB,MAAM,IAAI,CAACC,cAAc;QACzB,MAAM,IAAI,CAACC,YAAY;QACvB,MAAM,IAAI,CAACC,iBAAiB;QAE5B,IAAI,CAACtC,YAAY,CAACuC,IAAI,CAAC;IACzB;IAEAjB,gCACEF,eAA6B,EACM;QACnC,IAAI,CAACA,gBAAgBoB,QAAQ,CAAC,YAAY,8BAA8B,MAAK;YAC3E,OAAO,EAAE;QACX;QAEA,MAAMC,WAAWtE,cAAcuE,mBAAmB,CAACtB;QACnD,MAAMuB,WAAW1E,eAAe2E,MAAM,CAAC,CAACC,MAAQA,IAAIC,SAAS,KAAK,GAAGL,SAAS,KAAK,CAAC;QACpF,KAAK,MAAMI,OAAOF,SAAU;YAC1B1E,eAAe8E,MAAM,CAAC9E,eAAe+E,OAAO,CAACH,MAAM;QACrD;QAEA,OAAOF;IACT;IAEA,MAAMvC,oBAAoBF,OAAiB,EAAiB;QAC1D,uBAAuB;QACvB,MAAM+C,aAAa;YACjBC,SACEhF,OAAOiC,MAAM,CAACgD,MAAM,CAACD,OAAO,IAC5B,CAAC,OAAO,EAAEhF,OAAOiC,MAAM,CAACgD,MAAM,CAACC,MAAM,EAAEC,QAAQ,YAAY,CAAC,EAAEnF,OAAOiC,MAAM,CAACgD,MAAM,CAACC,MAAM,EAAEE,QAAQ,MAAM;QAC7G;QAEA,KAAK,MAAMC,UAAUrD,QAAS;YAC5B,sCAAsC;YACtC,+DAA+D;YAC/D,oBAAoB;YACpB,MAAMsD,UAAU3F,KAAK4F,IAAI,CACvB,YAAY3F,OAAO,CAAC4F,OAAO,CAAC,UAAU,UACtC,CAAC,UAAU,EAAEH,OAAO,cAAc,CAAC;YAErC,IAAI,CAAE,MAAM5E,OAAO6E,UAAW;gBAC5B;YACF;YACA,IAAI,CAAE,MAAM7E,OAAOd,KAAK4F,IAAI,CAACvF,OAAOyF,WAAW,EAAEJ,UAAW;gBAC1D,MAAM,IAAIK,MACR,CAAC,0CAA0C,EAAEL,OAAO,yFAAyF,CAAC;YAElJ;YAEA,MAAMM,WAAW,MAAMpG,SAAS+F,SAAS;YACzC,MAAMM,gBAAgB/B,OAAOgC,OAAO,CAACd,YAAYe,MAAM,CACrD,CAACC,KAAK,CAACC,KAAKC,MAAM,GAAKF,IAAIP,OAAO,CAAC,CAAC,GAAG,EAAEQ,IAAI,EAAE,CAAC,EAAEC,QAClDN;YAGF,qDAAqD;YACrD,MAAMO,WAAWvG,KAAK4F,IAAI,CAACvF,OAAOyF,WAAW,EAAEJ,QAAQ;YAEvD,oEAAoE;YACpE,IAAI,CAAE,MAAM5E,OAAOd,KAAKC,OAAO,CAACsG,YAAa;gBAC3C,MAAM5G,MAAMK,KAAKC,OAAO,CAACsG,WAAW;oBAAEC,WAAW;gBAAK;gBACtD9D,QAAQ+D,IAAI,CAAC,CAAC,mBAAmB,EAAEzG,KAAKC,OAAO,CAACsG,UAAU,2BAA2B,CAAC;YACxF;YAEA,IAAI,MAAMvF,aAAa;gBAAE0F,MAAMT;YAAc,GAAG;gBAAEjG,MAAMuG;YAAS,IAAI;gBACnE;YACF;YAEA,MAAM1G,UAAU0G,UAAUN;YAC1B,CAACpF,YACC6B,QAAQC,GAAG,CACTlD,MAAM6D,IAAI,CAAC,cAAc7D,MAAMiE,IAAI,CAAC1D,KAAK2D,QAAQ,CAACtD,OAAOyF,WAAW,EAAES;QAE5E;IACF;IAEA,MAAMjC,gBAAgB;QACpB,IAAI,CAACxC,KAAK,GAAG,MAAMJ;IACrB;IAEA,MAAM6C,iBAAiB;QACrB,IAAI,CAACxC,MAAM,GAAG,MAAMN;IACtB;IAEA,MAAM+C,eAAe;QACnB,IAAI,CAAC3C,IAAI,GAAG,MAAML;IACpB;IAEA,MAAMiD,oBAAoB;QACxB,IAAI,CAACzC,SAAS,GAAG,MAAML;QACvB,MAAMtB,OAAO2B,SAAS,CAAC2E,WAAW,CAAC,IAAI,CAAC3E,SAAS;IACnD;IAEA;;;;;GAKC,GACD,MAAMc,cAAc8D,aAA6B,EAAoC;QACnF,MAAMC,aAAa,IAAI,CAACC,mBAAmB,CAACF;QAC5C,MAAMG,YAAY7C,OAAO8C,IAAI,CAACH;QAE9B,qBAAqB;QACrB,aAAa;QACb,IAAIE,UAAUE,QAAQ,CAAC,WAAW;YAChC,MAAM,IAAI,CAACC,kBAAkB,CAACL,YAAYE;QAC5C;QAEA,mCAAmC;QACnC,oCAAoC;QACpC,IACEA,UAAUE,QAAQ,CAAC,YACnBF,UAAUE,QAAQ,CAAC,gBACnBF,UAAUE,QAAQ,CAAC,cACnB;YACA,MAAM,IAAI,CAACE,uCAAuC,CAACN;QACrD;QAEA,aAAa;QACb,IAAIE,UAAUE,QAAQ,CAAC,YAAYF,UAAUE,QAAQ,CAAC,UAAU;YAC9D,MAAM,IAAI,CAACG,wBAAwB,CAACP;QACtC;QAEA,cAAc;QACd,IAAIE,UAAUE,QAAQ,CAAC,WAAW;YAChC,MAAM,IAAI,CAACI,gBAAgB;QAC7B;QAEA,gBAAgB;QAChB,IAAIN,UAAUE,QAAQ,CAAC,aAAa;YAClC,MAAM,IAAI,CAACxC,iBAAiB;QAC9B;QAEA,OAAO;YACLsC;QACF;IACF;IAEAD,oBAAoBQ,SAAyB,EAAc;QACzD,OAAOpH,MAAMoH,WAAW,CAACC;YACvB,MAAMC,UAAUD,EAAEE,KAAK,CAAC;YACxB,OAAOD,SAAS,CAAC,EAAE,IAAI;QACzB;IACF;IAEA,MAAMN,mBAAmBL,UAAsB,EAAEE,SAAmB,EAAiB;QACnFxG,MAAMmH,CAAC,CAAC,sBAAsB;YAAEb;YAAYE;QAAU;QAEtD,MAAMzG,cAAcqH,MAAM;QAE1B,6BAA6B;QAC7B,kCAAkC;QAClC,MAAM/C,WAAWtE,cAAcuE,mBAAmB,CAACgC,WAAWe,MAAM,EAAE,CAAC,EAAE;QAEzE,IAAIhD,UAAU;YACZ,MAAMgD,SAAStH,cAAcuH,GAAG,CAACjD;YACjC,gCAAgC;YAChC,MAAMkD,eAAe9H,KAAK4F,IAAI,CAC5BvF,OAAOuD,WAAW,EAClB,CAAC,gBAAgB,EAAEgE,OAAOG,KAAK,CAACC,EAAE,CAAC,CAAC,EAAEJ,OAAOG,KAAK,CAACC,EAAE,CAAC,SAAS,CAAC;YAElE,IAAIJ,OAAOK,QAAQ,KAAKC,aAAa,CAAE,MAAMpH,OAAOgH,eAAgB;gBAClE,MAAM3G,iBAAiB,cAAc;oBAAEyD;gBAAS;YAClD;QACF;QAEA,MAAM,IAAI,CAACuD,qBAAqB;QAEhCtB,WAAWuB,SAAS,GAAGjI,OAAO;eACxB0G,WAAWuB,SAAS,IAAI,EAAE;YAC9BpI,KAAK4F,IAAI,CAACvF,OAAOuD,WAAW,EAAE;SAC/B;QACDmD,UAAUsB,IAAI,CAAC;IACjB;IAEA,MAAMlB,wCAAwCN,UAAsB,EAAuB;QACzF,MAAMyB,UAAUnI,OAAO;eACjB0G,WAAW/E,KAAK,IAAI,EAAE;eACtB+E,WAAW0B,SAAS,IAAI,EAAE;eAC1B1B,WAAWuB,SAAS,IAAI,EAAE;SAC/B;QACD7H,MAAMmH,CAAC,CAAC,2CAA2C;YAAEb;QAAW;QAEhE,eAAe;QACf,gBAAgB;QAChB,wIAAwI;QACxI,MAAM;QACN,KAAK;QAEL,MAAM,IAAI,CAAC2B,wBAAwB,CAACF;QAEpC,OAAO,EAAE;IACX;IAEA,MAAMlB,yBAAyBP,UAAsB,EAAiB;QACpEtG,MAAMmH,CAAC,CAAC,4BAA4B;YAAEb;QAAW;QACjD,MAAM4B,cAAc;eAAK5B,WAAW6B,KAAK,IAAI,EAAE;eAAO7B,WAAW8B,KAAK,IAAI,EAAE;SAAE;QAE9E,eAAe;QACf,gBAAgB;QAChB,8HAA8H;QAC9H,MAAM;QACN,KAAK;QAEL,kDAAkD;QAClD,0CAA0C;QAC1C,MAAM,IAAI,CAACpE,cAAc;QACzB,MAAM,IAAI,CAACD,aAAa;QACxB,MAAM,IAAI,CAACE,YAAY;QAEvB,MAAMoE,SAEAH,YAAYI,GAAG,CAAC,CAACC;YACrB,IAAIA,UAAUnE,QAAQ,CAAC,cAAc;gBACnC,MAAMC,WAAWtE,cAAcuE,mBAAmB,CAACiE;gBACnDtJ,OAAOoF;gBACP,OAAO;oBACLmE,aAAazI,cAAc0I,cAAc,CAACpE;gBAC5C;YACF;YACA,IAAIkE,UAAUnE,QAAQ,CAAC,cAAc;gBACnC,MAAM,GAAGsE,UAAU,GAAGH,UAAUrB,KAAK,CAAC,2BAA2B,EAAE;gBACnEjI,OAAOyJ;gBACP,qDAAqD;gBACrD,MAAMC,UAAUpJ,WAAWqJ,QAAQ,CAACF;gBACpC,OAAO;oBACLF,aAAazI,cAAc0I,cAAc,CAACE;gBAC5C;YACF;YACA,MAAM,IAAInD,MAAM;QAClB;QAEA,MAAM,IAAI,CAACqD,sBAAsB,CAACR;QAClC,MAAM,IAAI,CAACS,mBAAmB;IAChC;IAEA,8BAA8B;IAC9B,MAAMhC,mBAAmB;QACvB,MAAM,EAAE7B,IAAI,EAAEC,IAAI,EAAE,GAAGpF,OAAOiC,MAAM,CAACgD,MAAM,CAACC,MAAM,IAAI,CAAC;QACvD,MAAM+D,UAAU,CAAC,SAAS,EAAE9D,QAAQ,YAAY,WAAW,EAAEC,QAAQ,MAAM;QAE3ElF,MAAMmH,CAAC,CAAC,oBAAoB;YAAE4B;QAAQ;QACtC,MAAMC,QAAQC,GAAG,CACfnJ,OAAOiC,MAAM,CAACF,IAAI,CAACC,OAAO,CAACwG,GAAG,CAAC,OAAOnD;YACpC,MAAM7F,UAAUG,KAAK4F,IAAI,CAACvF,OAAOyF,WAAW,EAAEJ,QAAQ,gBAAgB4D;QACxE;IAEJ;IAEA;;;;GAIC,GACD,MAAMF,uBACJK,WAEG,EACgB;QACnBlJ,MAAMmH,CAAC,CAAC,0BAA0B+B;QAElC,iCAAiC;QACjC,MAAMC,eAAe,MAAMvI,iBACzB,YACA,CAAC,GACD;YACEwI,WAAW;QACb;QAGF,OAAO;eAAID;SAAa;IAC1B;IAEA;;;GAGC,GACD,MAAMvB,wBAAiD;QACrD,OAAO,AACL,CAAA,MAAMoB,QAAQC,GAAG,CAAC;YAChBrI,iBAAiB,iBAAiB,CAAC,GAAG;gBAAEwI,WAAW;YAAK;YACxDxI,iBAAiB,aAAa,CAAC,GAAG;gBAAEwI,WAAW;YAAK;SACrD,CAAA,EAEAC,IAAI,GACJA,IAAI;IACT;IAEA;;;GAGC,GACD,MAAMP,sBAA6C;QACjD,MAAM,CAACQ,IAAI,GAAG,MAAM1I,iBAClB,kBACA;YAAEyD,UAAU;QAAQ,GACpB;YAAE+E,WAAW;QAAK;QAEpBnK,OAAOqK;QACP,OAAOA;IACT;IAEA;;;;GAIC,GACD,MAAMrB,yBAAyBF,OAAuB,EAAqB;QACzE,MAAM,EAAEjG,OAAO,EAAE,GAAGhC,OAAOiC,MAAM,CAACF,IAAI;QACtC,MAAM,EAAE0H,KAAKC,MAAM,EAAE,GAAG1J,OAAOiC,MAAM,CAAC0C,GAAG;QAEzC,OAAO,AACL,CAAA,MAAMuE,QAAQC,GAAG,CACfnH,QAAQwG,GAAG,CAAC,OAAOnD,SACjB6D,QAAQC,GAAG,CACTlB,QAAQO,GAAG,CAAC,OAAOmB;gBACjB,MAAMC,MAAMD,QACTnE,OAAO,CAAC,CAAC,CAAC,EAAEkE,OAAO,CAAC,CAAC,EAAE,CAAC,CAAC,EAAErE,OAAO,CAAC,CAAC,EACpCG,OAAO,CAAC,iBAAiB;gBAC5B,MAAMiE,MAAM7J,QAAQgK;gBACpB,IAAI,CAAE,MAAMnJ,OAAOgJ,MAAO;oBACxB,MAAMnK,MAAMmK,KAAK;wBAAEtD,WAAW;oBAAK;gBACrC;gBACA,CAAC3F,YACC6B,QAAQC,GAAG,CACTlD,MAAM6D,IAAI,CAAC,cAAc7D,MAAMiE,IAAI,CAACuG,IAAIpE,OAAO,CAAC,GAAGxF,OAAOyF,WAAW,CAAC,CAAC,CAAC,EAAE;gBAE9E,MAAM,IAAI,CAACoE,+BAA+B,CAACF,SAASC;gBACpD,OAAOA;YACT,KAGN,EACAL,IAAI;IACR;IAEA,MAAcM,gCAAgCC,QAAgB,EAAEC,MAAc,EAAE;QAC9E,IAAI,CAAE,MAAMtJ,OAAOqJ,WAAY;YAC7B;QACF;QAEA,MAAME,iBAAiB,AAAC,CAAA,MAAMzK,SAASuK,SAAQ,EAAGG,QAAQ;QAE1D,MAAMC,iBAAiB,AAAC,CAAA;YACtB,8BAA8B;YAC9B,wEAAwE;YACxE,qDAAqD;YACrD,2FAA2F;YAC3F,gFAAgF;YAChF,kGAAkG;YAClG,MAAMC,cAAcJ,OAAOvE,OAAO,CAAC,mBAAmB;YACtD,MAAM4E,UAAUxK,QAAQmK;YACxB,MAAMM,eAAe1K,KAAK2D,QAAQ,CAAC8G,SAASD;YAC5C,MAAMG,aAAaD,iBAAiB,KAAK,oBAAoB,GAAGA,aAAa,cAAc,CAAC;YAE5F,MAAME,MAAMP,eAAexE,OAAO,CAAC,kBAAkB,CAAC,MAAM,EAAE8E,WAAW,CAAC,CAAC;YAC3E,OAAOC;QACT,CAAA;QACA,OAAO/K,UAAUuK,QAAQG;IAC3B;IAEA;;;;;;GAMC,GACD,MAAMM,mBACJjG,QAAgB,EAChBkG,WAAwB,EACxBC,MAAe,EACoD;QACnE,MAAM,EAAErF,MAAM,EAAE1F,MAAMgL,OAAO,EAAE,GAAGxK,gBAAgBqH,GAAG,CAACiD,aAAaG,gBAAgB,CACjF3K,cAAc0I,cAAc,CAACpE,WAC7BmG;QAGF,MAAMG,UAAUlL,KAAK4F,IAAI,CAACF,QAAQsF;QAClC,MAAMG,WAAWnL,KAAK4F,IAAI,CAACvF,OAAOyF,WAAW,EAAEoF;QAC/C,OAAO;YACLA;YACAC;YACAC,UAAU,MAAMtK,OAAOqK;QACzB;IACF;IAEA;;;;;GAKC,GACD,MAAME,YACJzG,QAAgB,EAChB0G,KAEC,EACoD;QACrD,MAAMtE,OAAsBvG,YAAY8K,OAAO;QAC/C,MAAMxD,QAAQzH,cAAc0I,cAAc,CAACpE;QAC3C,MAAM4G,YAAYtH,OAAO8C,IAAI,CAACsE,OAAOvG,MAAM,CAAC,CAAC0G,OAASA,SAAS1D,MAAM2D,QAAQ;QAE7E,OAAO,MAAM/K,YACXqG,MACA,OAAO2E,QAAQtF;YACb,MAAMuF,MAAMpL,gBAAgBqH,GAAG,CAACxB;YAChC,IAAIA,IAAIwF,UAAU,CAAC,eAAe;gBAChC,MAAMnL,SAAS8K,WAAW,OAAOM;oBAC/B,MAAM,EAAEpG,MAAM,EAAE1F,MAAM+L,CAAC,EAAE,GAAGH,IAAIX,gBAAgB,CAAClD,OAAO+D;oBACxDH,MAAM,CAAC,GAAGtF,IAAI,EAAE,EAAEyF,aAAa,CAAC,GAAG,MAAMhL,OACvCd,KAAK4F,IAAI,CAACvF,OAAOyF,WAAW,EAAEJ,QAAQqG;gBAE1C;gBACA,OAAOJ;YACT;YAEA,MAAM,EAAEjG,MAAM,EAAE1F,MAAM+L,CAAC,EAAE,GAAGH,IAAIX,gBAAgB,CAAClD;YACjD,MAAM,EAAE1F,OAAO,EAAE,GAAGhC,OAAOiC,MAAM,CAACF,IAAI;YACtC,IAAIsD,OAAOuB,QAAQ,CAAC,YAAY;gBAC9B,MAAMvG,SAAS2B,SAAS,OAAOqF;oBAC7BiE,MAAM,CAAC,GAAGtF,IAAI,EAAE,EAAEqB,GAAG,CAAC,GAAG,MAAM5G,OAC7Bd,KAAK4F,IAAI,CAACvF,OAAOyF,WAAW,EAAEJ,OAAOG,OAAO,CAAC,WAAW6B,IAAIqE;gBAEhE;YACF,OAAO;gBACLJ,MAAM,CAACtF,IAAI,GAAG,MAAMvF,OAAOd,KAAK4F,IAAI,CAACvF,OAAOyF,WAAW,EAAEJ,QAAQqG;YACnE;YAEA,OAAOJ;QACT,GACA,CAAC;IAEL;IAEA;;GAEC,GACD,MAAMtK,aAAa2K,IAA+B,EAAE;QAClD,OAAO,MAAM3K,aAAa2K;IAC5B;IAEA;;GAEC,GACD,MAAM1K,UAAUsD,QAAgB,EAAmC;QACjE,OAAO,MAAMtD,UAAUsD;IACzB;IAEA;;GAEC,GACD,MAAMzD,iBACJkF,GAAM,EACN4F,eAAmC,EACnCC,gBAAkC,EACT;QACzB,OAAO,MAAM/K,iBAAiBkF,KAAK4F,iBAAiBC;IACtD;IAEA;;GAEC,GACD,MAAM9K,eACJiF,GAAM,EACN4F,eAAmC,EACX;QACxB,OAAO,MAAM7K,eAAeiF,KAAK4F;IACnC;IAEA;;GAEC,GACD,MAAM/K,iBAAgC;QACpC,OAAO,MAAMA;IACf;AACF"}
|