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.
Files changed (96) hide show
  1. package/dist/ai/providers/rtzr/error.d.ts +1 -1
  2. package/dist/ai/providers/rtzr/error.d.ts.map +1 -1
  3. package/dist/api/config.d.ts +1 -0
  4. package/dist/api/config.d.ts.map +1 -1
  5. package/dist/api/config.js +1 -1
  6. package/dist/api/decorators.d.ts +1 -1
  7. package/dist/api/decorators.d.ts.map +1 -1
  8. package/dist/api/decorators.js +1 -1
  9. package/dist/api/sonamu.d.ts +3 -1
  10. package/dist/api/sonamu.d.ts.map +1 -1
  11. package/dist/api/sonamu.js +51 -40
  12. package/dist/database/base-model.d.ts +16 -6
  13. package/dist/database/base-model.d.ts.map +1 -1
  14. package/dist/database/base-model.js +44 -3
  15. package/dist/database/base-model.types.d.ts +29 -48
  16. package/dist/database/base-model.types.d.ts.map +1 -1
  17. package/dist/database/base-model.types.js +12 -2
  18. package/dist/database/puri.d.ts +2 -1
  19. package/dist/database/puri.d.ts.map +1 -1
  20. package/dist/database/puri.js +2 -1
  21. package/dist/database/puri.types.d.ts +3 -3
  22. package/dist/database/puri.types.d.ts.map +1 -1
  23. package/dist/database/puri.types.js +1 -1
  24. package/dist/entity/entity-manager.d.ts +8 -4
  25. package/dist/entity/entity-manager.d.ts.map +1 -1
  26. package/dist/entity/entity.d.ts +10 -1
  27. package/dist/entity/entity.d.ts.map +1 -1
  28. package/dist/entity/entity.js +84 -39
  29. package/dist/index.d.ts +1 -0
  30. package/dist/index.d.ts.map +1 -1
  31. package/dist/index.js +2 -1
  32. package/dist/syncer/checksum.d.ts +8 -3
  33. package/dist/syncer/checksum.d.ts.map +1 -1
  34. package/dist/syncer/checksum.js +17 -9
  35. package/dist/syncer/code-generator.js +7 -2
  36. package/dist/syncer/syncer.d.ts +6 -6
  37. package/dist/syncer/syncer.d.ts.map +1 -1
  38. package/dist/syncer/syncer.js +27 -13
  39. package/dist/tasks/workflow-manager.d.ts +3 -3
  40. package/dist/tasks/workflow-manager.d.ts.map +1 -1
  41. package/dist/tasks/workflow-manager.js +15 -11
  42. package/dist/template/implementations/generated.template.d.ts.map +1 -1
  43. package/dist/template/implementations/generated.template.js +8 -6
  44. package/dist/template/implementations/model.template.js +5 -5
  45. package/dist/template/implementations/services.template.d.ts +17 -0
  46. package/dist/template/implementations/services.template.d.ts.map +1 -0
  47. package/dist/template/implementations/services.template.js +159 -0
  48. package/dist/template/implementations/view_form.template.js +2 -2
  49. package/dist/template/implementations/view_id_async_select.template.js +2 -2
  50. package/dist/template/implementations/view_list.template.js +5 -5
  51. package/dist/types/types.d.ts +43 -25
  52. package/dist/types/types.d.ts.map +1 -1
  53. package/dist/types/types.js +29 -17
  54. package/dist/ui/ai-api.d.ts +2 -0
  55. package/dist/ui/ai-api.d.ts.map +1 -1
  56. package/dist/ui/ai-api.js +43 -49
  57. package/dist/ui/ai-client.d.ts +10 -0
  58. package/dist/ui/ai-client.d.ts.map +1 -1
  59. package/dist/ui/ai-client.js +457 -437
  60. package/dist/ui/api.d.ts.map +1 -1
  61. package/dist/ui/api.js +14 -3
  62. package/dist/ui-web/assets/{index-J9MCfjCd.js → index-DzqUrTB-.js} +56 -59
  63. package/dist/ui-web/index.html +1 -1
  64. package/package.json +12 -8
  65. package/src/api/config.ts +3 -0
  66. package/src/api/decorators.ts +6 -1
  67. package/src/api/sonamu.ts +71 -52
  68. package/src/database/base-model.ts +66 -11
  69. package/src/database/base-model.types.ts +79 -76
  70. package/src/database/puri.ts +5 -1
  71. package/src/database/puri.types.ts +3 -6
  72. package/src/entity/entity.ts +83 -34
  73. package/src/index.ts +1 -0
  74. package/src/shared/app.shared.ts.txt +1 -1
  75. package/src/shared/web.shared.ts.txt +0 -43
  76. package/src/syncer/checksum.ts +31 -9
  77. package/src/syncer/code-generator.ts +8 -1
  78. package/src/syncer/syncer.ts +38 -26
  79. package/src/tasks/workflow-manager.ts +16 -12
  80. package/src/template/implementations/generated.template.ts +17 -3
  81. package/src/template/implementations/model.template.ts +4 -4
  82. package/src/template/implementations/services.template.ts +226 -0
  83. package/src/template/implementations/view_form.template.ts +1 -1
  84. package/src/template/implementations/view_id_async_select.template.ts +1 -1
  85. package/src/template/implementations/view_list.template.ts +4 -4
  86. package/src/types/types.ts +33 -16
  87. package/src/ui/ai-api.ts +61 -60
  88. package/src/ui/ai-client.ts +535 -499
  89. package/src/ui/api.ts +14 -2
  90. package/src/ui/entity.instructions.md +536 -0
  91. package/dist/template/implementations/service.template.d.ts +0 -29
  92. package/dist/template/implementations/service.template.d.ts.map +0 -1
  93. package/dist/template/implementations/service.template.js +0 -202
  94. package/dist/ui-web/assets/provider-utils_false-BKJD46kk.js +0 -1
  95. package/dist/ui-web/assets/provider-utils_false-Bu5lmX18.js +0 -1
  96. package/src/template/implementations/service.template.ts +0 -328
@@ -19,7 +19,7 @@ export type WorkflowCreateOptions<Input, Output, TSchema extends StandardSchemaV
19
19
  };
20
20
  export declare class WorkflowManager {
21
21
  #private;
22
- private constructor();
22
+ constructor(dbConf: Knex.Config, runMigrations?: boolean);
23
23
  /**
24
24
  * 정의된 워크플로우 및 워크플로우에 대한 scheduled tasks를 동기화합니다.
25
25
  */
@@ -29,12 +29,12 @@ export declare class WorkflowManager {
29
29
  unscheduleTask(name: string): Promise<void>;
30
30
  registerWorkflow<Input, Output, TSchema extends StandardSchemaV1 | undefined = undefined>(options: WorkflowCreateOptions<Input, Output, TSchema>): RunnableWorkflow<SchemaOutput<TSchema, Input>, Output, SchemaInput<TSchema, Input>>;
31
31
  unregisterWorkflow(workflow: Pick<WorkflowMetadata, "name" | "version" | "id">): Promise<void>;
32
- setupWorker(options: WorkflowOptions): Promise<void>;
32
+ setupWorker(options: WorkflowOptions): void;
33
+ startWorker(): Promise<void>;
33
34
  stopWorker(): Promise<void>;
34
35
  stopSchedules(): Promise<void>;
35
36
  destroySchedules(): Promise<void>;
36
37
  destroy(): Promise<void>;
37
38
  [Symbol.asyncDispose](): Promise<void>;
38
- static create(dbConf: Knex.Config, runMigrations?: boolean): Promise<WorkflowManager>;
39
39
  }
40
40
  //# sourceMappingURL=workflow-manager.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"workflow-manager.d.ts","sourceRoot":"","sources":["../../src/tasks/workflow-manager.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EACV,gBAAgB,EAChB,WAAW,EACX,YAAY,EACZ,gBAAgB,EAEhB,iBAAiB,EACjB,YAAY,EACb,MAAM,4BAA4B,CAAC;AAEpC,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAQjC,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,aAAa,CAAC;AACpD,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAE7C,MAAM,WAAW,eAAe;IAE9B,WAAW,CAAC,EAAE,MAAM,CAAC;IAGrB,SAAS,CAAC,EAAE,OAAO,CAAC;IAGpB,WAAW,CAAC,EAAE,MAAM,CAAC;CACtB;AAGD,MAAM,MAAM,gBAAgB,CAAC,KAAK,EAAE,MAAM,IAAI,CAC5C,MAAM,EAAE,QAAQ,CAAC;IACf,KAAK,EAAE,KAAK,CAAC;IACb,IAAI,EAAE,WAAW,CAAC;IAClB,OAAO,EAAE,MAAM,GAAG,IAAI,CAAC;CACxB,CAAC,KACC,OAAO,CAAC,MAAM,CAAC,GAAG,MAAM,CAAC;AAG9B,MAAM,MAAM,qBAAqB,CAC/B,KAAK,EACL,MAAM,EACN,OAAO,SAAS,gBAAgB,GAAG,SAAS,GAAG,SAAS,IACtD,IAAI,CACN,YAAY,CAAC,YAAY,CAAC,OAAO,EAAE,KAAK,CAAC,EAAE,MAAM,EAAE,WAAW,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC,EAC/E,SAAS,CACV,GAAG;IACF,EAAE,EAAE,MAAM,CAAC;IACX,OAAO,EAAE,MAAM,GAAG,IAAI,CAAC;IACvB,QAAQ,EAAE,gBAAgB,CAAC,YAAY,CAAC,OAAO,EAAE,KAAK,CAAC,EAAE,MAAM,CAAC,CAAC;CAClE,CAAC;AAEF,qBAAa,eAAe;;IAuB1B,OAAO;IAQP;;OAEG;IACG,WAAW,CAAC,WAAW,EAAE,GAAG,CAAC,MAAM,EAAE,gBAAgB,EAAE,CAAC;IAqF9D,GAAG,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,SAAS,gBAAgB,GAAG,SAAS,GAAG,SAAS,EACzE,OAAO,EAAE,IAAI,CAAC,qBAAqB,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,UAAU,GAAG,QAAQ,GAAG,IAAI,CAAC,EAC1F,KAAK,EAAE,WAAW,CAAC,OAAO,EAAE,KAAK,CAAC,GACjC,OAAO,CAAC,iBAAiB,CAAC,MAAM,CAAC,CAAC;IAW/B,YAAY,CAChB,QAAQ,EAAE,IAAI,CAAC,gBAAgB,EAAE,IAAI,GAAG,MAAM,GAAG,SAAS,CAAC,EAC3D,QAAQ,EAAE,gBAAgB,CAAC,WAAW,CAAC,CAAC,MAAM,CAAC;IA8B3C,cAAc,CAAC,IAAI,EAAE,MAAM;IAajC,gBAAgB,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,SAAS,gBAAgB,GAAG,SAAS,GAAG,SAAS,EACtF,OAAO,EAAE,qBAAqB,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,CAAC,GACrD,gBAAgB,CAAC,YAAY,CAAC,OAAO,EAAE,KAAK,CAAC,EAAE,MAAM,EAAE,WAAW,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC;IA4ChF,kBAAkB,CAAC,QAAQ,EAAE,IAAI,CAAC,gBAAgB,EAAE,MAAM,GAAG,SAAS,GAAG,IAAI,CAAC;IAK9E,WAAW,CAAC,OAAO,EAAE,eAAe;IAMpC,UAAU;IAKV,aAAa;IAOb,gBAAgB;IAOhB,OAAO;IAOb,CAAC,MAAM,CAAC,YAAY,CAAC;WAKR,MAAM,CACjB,MAAM,EAAE,IAAI,CAAC,MAAM,EACnB,aAAa,GAAE,OAAc,GAC5B,OAAO,CAAC,eAAe,CAAC;CAI5B"}
1
+ {"version":3,"file":"workflow-manager.d.ts","sourceRoot":"","sources":["../../src/tasks/workflow-manager.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EACV,gBAAgB,EAChB,WAAW,EACX,YAAY,EACZ,gBAAgB,EAEhB,iBAAiB,EACjB,YAAY,EACb,MAAM,4BAA4B,CAAC;AAEpC,OAAO,KAAK,EAAE,IAAI,EAAE,MAAM,MAAM,CAAC;AAQjC,OAAO,KAAK,EAAE,gBAAgB,EAAE,MAAM,aAAa,CAAC;AACpD,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAE7C,MAAM,WAAW,eAAe;IAE9B,WAAW,CAAC,EAAE,MAAM,CAAC;IAGrB,SAAS,CAAC,EAAE,OAAO,CAAC;IAGpB,WAAW,CAAC,EAAE,MAAM,CAAC;CACtB;AAGD,MAAM,MAAM,gBAAgB,CAAC,KAAK,EAAE,MAAM,IAAI,CAC5C,MAAM,EAAE,QAAQ,CAAC;IACf,KAAK,EAAE,KAAK,CAAC;IACb,IAAI,EAAE,WAAW,CAAC;IAClB,OAAO,EAAE,MAAM,GAAG,IAAI,CAAC;CACxB,CAAC,KACC,OAAO,CAAC,MAAM,CAAC,GAAG,MAAM,CAAC;AAG9B,MAAM,MAAM,qBAAqB,CAC/B,KAAK,EACL,MAAM,EACN,OAAO,SAAS,gBAAgB,GAAG,SAAS,GAAG,SAAS,IACtD,IAAI,CACN,YAAY,CAAC,YAAY,CAAC,OAAO,EAAE,KAAK,CAAC,EAAE,MAAM,EAAE,WAAW,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC,EAC/E,SAAS,CACV,GAAG;IACF,EAAE,EAAE,MAAM,CAAC;IACX,OAAO,EAAE,MAAM,GAAG,IAAI,CAAC;IACvB,QAAQ,EAAE,gBAAgB,CAAC,YAAY,CAAC,OAAO,EAAE,KAAK,CAAC,EAAE,MAAM,CAAC,CAAC;CAClE,CAAC;AAEF,qBAAa,eAAe;;gBAwBd,MAAM,EAAE,IAAI,CAAC,MAAM,EAAE,aAAa,GAAE,OAAc;IAW9D;;OAEG;IACG,WAAW,CAAC,WAAW,EAAE,GAAG,CAAC,MAAM,EAAE,gBAAgB,EAAE,CAAC;IAqF9D,GAAG,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,SAAS,gBAAgB,GAAG,SAAS,GAAG,SAAS,EACzE,OAAO,EAAE,IAAI,CAAC,qBAAqB,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,UAAU,GAAG,QAAQ,GAAG,IAAI,CAAC,EAC1F,KAAK,EAAE,WAAW,CAAC,OAAO,EAAE,KAAK,CAAC,GACjC,OAAO,CAAC,iBAAiB,CAAC,MAAM,CAAC,CAAC;IAW/B,YAAY,CAChB,QAAQ,EAAE,IAAI,CAAC,gBAAgB,EAAE,IAAI,GAAG,MAAM,GAAG,SAAS,CAAC,EAC3D,QAAQ,EAAE,gBAAgB,CAAC,WAAW,CAAC,CAAC,MAAM,CAAC;IA8B3C,cAAc,CAAC,IAAI,EAAE,MAAM;IAajC,gBAAgB,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,SAAS,gBAAgB,GAAG,SAAS,GAAG,SAAS,EACtF,OAAO,EAAE,qBAAqB,CAAC,KAAK,EAAE,MAAM,EAAE,OAAO,CAAC,GACrD,gBAAgB,CAAC,YAAY,CAAC,OAAO,EAAE,KAAK,CAAC,EAAE,MAAM,EAAE,WAAW,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC;IA4ChF,kBAAkB,CAAC,QAAQ,EAAE,IAAI,CAAC,gBAAgB,EAAE,MAAM,GAAG,SAAS,GAAG,IAAI,CAAC;IAKpF,WAAW,CAAC,OAAO,EAAE,eAAe;IAK9B,WAAW;IAUX,UAAU;IAKV,aAAa;IAOb,gBAAgB;IAOhB,OAAO;IAOb,CAAC,MAAM,CAAC,YAAY,CAAC;CAGtB"}
@@ -16,7 +16,11 @@ export class WorkflowManager {
16
16
  #workflowsMap;
17
17
  // Task 이름 -> Task 정보 및 Input 값, Workflow ID
18
18
  #scheduledTasks;
19
- constructor(backend){
19
+ // BackendPostgres에서 처리하는 것들이 있어서 Knex 커넥션이 아니라 설정값을 넣어줘야함.
20
+ constructor(dbConf, runMigrations = true){
21
+ const backend = new BackendPostgres(dbConf, {
22
+ runMigrations
23
+ });
20
24
  this.#backend = backend;
21
25
  this.#ow = new OpenWorkflow({
22
26
  backend
@@ -155,9 +159,16 @@ export class WorkflowManager {
155
159
  this.#ow.unregisterWorkflow(workflow.name, workflow.version ?? null);
156
160
  }
157
161
  // Worker를 설정 후 시작
158
- async setupWorker(options) {
162
+ setupWorker(options) {
159
163
  this.#worker = this.#ow.newWorker(options);
160
- await this.#worker.start();
164
+ }
165
+ // Worker를 초기화
166
+ async startWorker() {
167
+ if (!this.#worker) {
168
+ return;
169
+ }
170
+ await this.#backend.initialize();
171
+ await this.#worker?.start();
161
172
  }
162
173
  // Worker를 중지
163
174
  async stopWorker() {
@@ -181,13 +192,6 @@ export class WorkflowManager {
181
192
  [Symbol.asyncDispose]() {
182
193
  return this.destroy();
183
194
  }
184
- // BackendPostgres에서 처리하는 것들이 있어서 Knex 커넥션이 아니라 설정값을 넣어줘야함.
185
- static async create(dbConf, runMigrations = true) {
186
- const backend = await BackendPostgres.connect(dbConf, {
187
- runMigrations
188
- });
189
- return new WorkflowManager(backend);
190
- }
191
195
  }
192
196
 
193
- //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/tasks/workflow-manager.ts"],"sourcesContent":["import { BackendPostgres, OpenWorkflow, type Worker } from \"@sonamu-kit/tasks\";\nimport type {\n  RunnableWorkflow,\n  SchemaInput,\n  SchemaOutput,\n  StandardSchemaV1,\n  StepApi,\n  WorkflowRunHandle,\n  WorkflowSpec,\n} from \"@sonamu-kit/tasks/internal\";\nimport assert from \"assert\";\nimport type { Knex } from \"knex\";\nimport { schedule as cronSchedule, type ScheduledTask } from \"node-cron\";\nimport type { ZodObject } from \"zod\";\nimport type { Context } from \"../api/context\";\nimport { Sonamu } from \"../api/sonamu\";\nimport { Naite } from \"../naite/naite\";\nimport { createMockSSEFactory } from \"../stream/sse\";\nimport type { Executable } from \"../types/types\";\nimport type { WorkflowMetadata } from \"./decorator\";\nimport { StepWrapper } from \"./step-wrapper\";\n\nexport interface WorkflowOptions {\n  // worker에서 동시 실행할 태스크 수, 기본은 CPU 코어 수 - 1개\n  concurrency?: number;\n\n  // worker에서 사용할 pub/sub 여부, 기본은 true\n  usePubSub?: boolean;\n\n  // pub/sub으로 태스크를 수신했을 때 워크플로우 실행까지 지연 시간, 기본은 500ms\n  listenDelay?: number;\n}\n\n// Workflow 함수의 타입, @sonamu-kit/tasks와 다른 점은 step을 한번 감싼 형태.\nexport type WorkflowFunction<Input, Output> = (\n  params: Readonly<{\n    input: Input;\n    step: StepWrapper;\n    version: string | null;\n  }>,\n) => Promise<Output> | Output;\n\n// Workflow를 등록할 때를 위한 타입\nexport type WorkflowCreateOptions<\n  Input,\n  Output,\n  TSchema extends StandardSchemaV1 | undefined = undefined,\n> = Omit<\n  WorkflowSpec<SchemaOutput<TSchema, Input>, Output, SchemaInput<TSchema, Input>>,\n  \"version\"\n> & {\n  id: string;\n  version: string | null;\n  function: WorkflowFunction<SchemaOutput<TSchema, Input>, Output>;\n};\n\nexport class WorkflowManager {\n  // backend 인스턴스\n  readonly #backend: BackendPostgres;\n\n  // OpenWorkflow 인스턴스\n  readonly #ow: OpenWorkflow;\n\n  // Worker 인스턴스 (없을 수 있으며, 이 때는 분산된 서버 환경에서 DB에 publish만 한다고 가정함)\n  #worker: Worker | null;\n\n  // 파일 경로 -> 파일에 정의된 워크플로우 메타데이터 목록\n  #workflowsMap: Map<string, WorkflowMetadata[]>;\n\n  // Task 이름 -> Task 정보 및 Input 값, Workflow ID\n  #scheduledTasks: Map<\n    string,\n    {\n      task: ScheduledTask;\n      inputFn: Executable<SchemaInput<unknown, unknown>>;\n      workflowId: string;\n    }\n  >;\n\n  private constructor(backend: BackendPostgres) {\n    this.#backend = backend;\n    this.#ow = new OpenWorkflow({ backend });\n    this.#worker = null;\n    this.#workflowsMap = new Map();\n    this.#scheduledTasks = new Map();\n  }\n\n  /**\n   * 정의된 워크플로우 및 워크플로우에 대한 scheduled tasks를 동기화합니다.\n   */\n  async synchronize(workflowMap: Map<string, WorkflowMetadata[]>) {\n    // 1. 삭제된 파일은 일괄 삭제\n    await Promise.allSettled(\n      Array.from(this.#workflowsMap.entries())\n        .filter(([key]) => !workflowMap.has(key))\n        .flatMap(([_, workflows]) =>\n          workflows.map((workflow) => {\n            return Promise.allSettled([\n              ...workflow.schedules.map((schedule) => this.unscheduleTask(schedule.name)),\n              this.unregisterWorkflow(workflow),\n            ]);\n          }),\n        ),\n    );\n\n    // 2. 새로 추가된 파일은 일괄 등록\n    await Promise.allSettled(\n      Array.from(workflowMap.entries())\n        .filter(([key]) => !this.#workflowsMap.has(key))\n        .flatMap(([_, workflows]) =>\n          workflows.map((workflow) => {\n            this.registerWorkflow({\n              id: workflow.id,\n              name: workflow.name,\n              version: workflow.version ?? null,\n              schema: workflow.schema,\n              function: workflow.fn,\n            });\n\n            return Promise.allSettled(\n              workflow.schedules.map((schedule) => this.scheduleTask(workflow, schedule)),\n            );\n          }),\n        ),\n    );\n\n    // 3. 기존과 다른 것을 diff친 다음, 삭제 후 재등록\n    await Promise.allSettled(\n      Array.from(workflowMap.entries())\n        .filter(([key]) => this.#workflowsMap.has(key))\n        .map<[string, [WorkflowMetadata[], WorkflowMetadata[]]]>(([key, newWorkflows]) => {\n          const previousWorkflows = this.#workflowsMap.get(key);\n          assert(previousWorkflows, \"previous workflows not found\");\n          return [key, [previousWorkflows, newWorkflows]];\n        })\n        .map(async ([_, [previousWorkflows, newWorkflows]]) => {\n          // 기존 것들을 삭제부터 해야함.\n          await Promise.allSettled(\n            previousWorkflows\n              .filter((prevItem) => !newWorkflows.some((newItem) => newItem.id === prevItem.id))\n              .map((prevItem) => {\n                return Promise.allSettled([\n                  ...prevItem.schedules.map((schedule) => this.unscheduleTask(schedule.name)),\n                  this.unregisterWorkflow(prevItem),\n                ]);\n              }),\n          );\n\n          // 새로 추가된 것들을 등록\n          await Promise.allSettled(\n            newWorkflows\n              .filter(\n                (newItem) => !previousWorkflows.some((prevItem) => prevItem.id === newItem.id),\n              )\n              .map(async (newItem) => {\n                this.registerWorkflow({\n                  id: newItem.id,\n                  name: newItem.name,\n                  version: newItem.version ?? null,\n                  schema: newItem.schema,\n                  function: newItem.fn,\n                });\n\n                return Promise.allSettled(\n                  newItem.schedules.map((schedule) => this.scheduleTask(newItem, schedule)),\n                );\n              }),\n          );\n        }),\n    );\n\n    this.#workflowsMap = workflowMap;\n  }\n\n  // 워크플로우를 실행\n  run<Input, Output, TSchema extends StandardSchemaV1 | undefined = undefined>(\n    options: Omit<WorkflowCreateOptions<Input, Output, TSchema>, \"function\" | \"schema\" | \"id\">,\n    input: SchemaInput<TSchema, Input>,\n  ): Promise<WorkflowRunHandle<Output>> {\n    return this.#ow.runWorkflow(\n      {\n        name: options.name,\n        version: options.version ?? undefined,\n      },\n      input,\n    );\n  }\n\n  // cron task를 등록\n  async scheduleTask(\n    workflow: Pick<WorkflowMetadata, \"id\" | \"name\" | \"version\">,\n    schedule: WorkflowMetadata[\"schedules\"][number],\n  ) {\n    const task = cronSchedule(\n      schedule.expression,\n      (async (\n        { name, version }: Pick<WorkflowMetadata, \"name\" | \"version\">,\n        { input }: WorkflowMetadata[\"schedules\"][number],\n      ) => {\n        const inputData = await (typeof input === \"function\"\n          ? Promise.resolve(input())\n          : Promise.resolve(input));\n\n        return this.run({ name, version: version ?? null }, inputData);\n      }).bind(this, workflow, schedule),\n      {\n        name: schedule.name,\n        timezone: Sonamu.config.api.timezone,\n        noOverlap: false,\n      },\n    );\n\n    this.#scheduledTasks.set(schedule.name, {\n      task,\n      inputFn: schedule.input,\n      workflowId: workflow.id,\n    });\n    await task.start();\n  }\n\n  // cron task를 중지\n  async unscheduleTask(name: string) {\n    const taskItem = this.#scheduledTasks.get(name);\n    if (!taskItem) {\n      console.error(\"scheduled task not found\", name);\n      return;\n    }\n\n    this.#scheduledTasks.delete(name);\n    await taskItem.task.stop();\n    await taskItem.task.destroy();\n  }\n\n  // 워크플로우를 등록, 관련된 스케줄은 별도로 등록해야 함.\n  registerWorkflow<Input, Output, TSchema extends StandardSchemaV1 | undefined = undefined>(\n    options: WorkflowCreateOptions<Input, Output, TSchema>,\n  ): RunnableWorkflow<SchemaOutput<TSchema, Input>, Output, SchemaInput<TSchema, Input>> {\n    const fn = async (\n      params: Readonly<{\n        input: SchemaOutput<TSchema, Input>;\n        step: StepApi;\n        version: string | null;\n      }>,\n    ) => {\n      const baseContext = {\n        request: null,\n        reply: null,\n        headers: {},\n        createSSE: (schema: ZodObject) => createMockSSEFactory(schema),\n        naiteStore: Naite.createStore(),\n        user: null,\n        passport: {\n          login: async () => {},\n          logout: () => {},\n        },\n      } as unknown as Context;\n\n      const contextProvider = Sonamu.config.tasks?.contextProvider;\n      const context: Context = contextProvider\n        ? await Promise.resolve(contextProvider(baseContext))\n        : baseContext;\n\n      const step = new StepWrapper(params.step);\n      return Sonamu.asyncLocalStorage.run({ context }, () =>\n        options.function({ input: params.input, step, version: params.version }),\n      );\n    };\n\n    const workflow = this.#ow.defineWorkflow(\n      {\n        name: options.name,\n        version: options.version ?? undefined,\n        schema: options.schema,\n      },\n      fn,\n    );\n    return workflow;\n  }\n\n  // 워크플로우를 등록 해제, 관련된 스케줄은 별도로 해제해야 함.\n  async unregisterWorkflow(workflow: Pick<WorkflowMetadata, \"name\" | \"version\" | \"id\">) {\n    this.#ow.unregisterWorkflow(workflow.name, workflow.version ?? null);\n  }\n\n  // Worker를 설정 후 시작\n  async setupWorker(options: WorkflowOptions) {\n    this.#worker = this.#ow.newWorker(options);\n    await this.#worker.start();\n  }\n\n  // Worker를 중지\n  async stopWorker() {\n    await this.#worker?.stop();\n  }\n\n  // cron task들을 모두 중지\n  async stopSchedules() {\n    await Promise.allSettled(\n      Array.from(this.#scheduledTasks.values()).map(({ task }) => Promise.resolve(task.stop())),\n    );\n  }\n\n  // cron task들을 모두 정리\n  async destroySchedules() {\n    await Promise.allSettled(\n      Array.from(this.#scheduledTasks.values()).map(({ task }) => Promise.resolve(task.destroy())),\n    );\n    this.#scheduledTasks.clear();\n  }\n\n  async destroy() {\n    await this.stopSchedules();\n    await this.stopWorker();\n    await this.destroySchedules();\n    await this.#backend.stop();\n  }\n\n  [Symbol.asyncDispose]() {\n    return this.destroy();\n  }\n\n  // BackendPostgres에서 처리하는 것들이 있어서 Knex 커넥션이 아니라 설정값을 넣어줘야함.\n  static async create(\n    dbConf: Knex.Config,\n    runMigrations: boolean = true,\n  ): Promise<WorkflowManager> {\n    const backend = await BackendPostgres.connect(dbConf, { runMigrations });\n    return new WorkflowManager(backend);\n  }\n}\n"],"names":["BackendPostgres","OpenWorkflow","assert","schedule","cronSchedule","Sonamu","Naite","createMockSSEFactory","StepWrapper","WorkflowManager","backend","Map","synchronize","workflowMap","Promise","allSettled","Array","from","entries","filter","key","has","flatMap","_","workflows","map","workflow","schedules","unscheduleTask","name","unregisterWorkflow","registerWorkflow","id","version","schema","function","fn","scheduleTask","newWorkflows","previousWorkflows","get","prevItem","some","newItem","run","options","input","runWorkflow","undefined","task","expression","inputData","resolve","bind","timezone","config","api","noOverlap","set","inputFn","workflowId","start","taskItem","console","error","delete","stop","destroy","params","baseContext","request","reply","headers","createSSE","naiteStore","createStore","user","passport","login","logout","contextProvider","tasks","context","step","asyncLocalStorage","defineWorkflow","setupWorker","newWorker","stopWorker","stopSchedules","values","destroySchedules","clear","Symbol","asyncDispose","create","dbConf","runMigrations","connect"],"mappings":"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"}
197
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../src/tasks/workflow-manager.ts"],"sourcesContent":["import { BackendPostgres, OpenWorkflow, type Worker } from \"@sonamu-kit/tasks\";\nimport type {\n  RunnableWorkflow,\n  SchemaInput,\n  SchemaOutput,\n  StandardSchemaV1,\n  StepApi,\n  WorkflowRunHandle,\n  WorkflowSpec,\n} from \"@sonamu-kit/tasks/internal\";\nimport assert from \"assert\";\nimport type { Knex } from \"knex\";\nimport { schedule as cronSchedule, type ScheduledTask } from \"node-cron\";\nimport type { ZodObject } from \"zod\";\nimport type { Context } from \"../api/context\";\nimport { Sonamu } from \"../api/sonamu\";\nimport { Naite } from \"../naite/naite\";\nimport { createMockSSEFactory } from \"../stream/sse\";\nimport type { Executable } from \"../types/types\";\nimport type { WorkflowMetadata } from \"./decorator\";\nimport { StepWrapper } from \"./step-wrapper\";\n\nexport interface WorkflowOptions {\n  // worker에서 동시 실행할 태스크 수, 기본은 CPU 코어 수 - 1개\n  concurrency?: number;\n\n  // worker에서 사용할 pub/sub 여부, 기본은 true\n  usePubSub?: boolean;\n\n  // pub/sub으로 태스크를 수신했을 때 워크플로우 실행까지 지연 시간, 기본은 500ms\n  listenDelay?: number;\n}\n\n// Workflow 함수의 타입, @sonamu-kit/tasks와 다른 점은 step을 한번 감싼 형태.\nexport type WorkflowFunction<Input, Output> = (\n  params: Readonly<{\n    input: Input;\n    step: StepWrapper;\n    version: string | null;\n  }>,\n) => Promise<Output> | Output;\n\n// Workflow를 등록할 때를 위한 타입\nexport type WorkflowCreateOptions<\n  Input,\n  Output,\n  TSchema extends StandardSchemaV1 | undefined = undefined,\n> = Omit<\n  WorkflowSpec<SchemaOutput<TSchema, Input>, Output, SchemaInput<TSchema, Input>>,\n  \"version\"\n> & {\n  id: string;\n  version: string | null;\n  function: WorkflowFunction<SchemaOutput<TSchema, Input>, Output>;\n};\n\nexport class WorkflowManager {\n  // backend 인스턴스\n  readonly #backend: BackendPostgres;\n\n  // OpenWorkflow 인스턴스\n  readonly #ow: OpenWorkflow;\n\n  // Worker 인스턴스 (없을 수 있으며, 이 때는 분산된 서버 환경에서 DB에 publish만 한다고 가정함)\n  #worker: Worker | null;\n\n  // 파일 경로 -> 파일에 정의된 워크플로우 메타데이터 목록\n  #workflowsMap: Map<string, WorkflowMetadata[]>;\n\n  // Task 이름 -> Task 정보 및 Input 값, Workflow ID\n  #scheduledTasks: Map<\n    string,\n    {\n      task: ScheduledTask;\n      inputFn: Executable<SchemaInput<unknown, unknown>>;\n      workflowId: string;\n    }\n  >;\n\n  // BackendPostgres에서 처리하는 것들이 있어서 Knex 커넥션이 아니라 설정값을 넣어줘야함.\n  constructor(dbConf: Knex.Config, runMigrations: boolean = true) {\n    const backend = new BackendPostgres(dbConf, { runMigrations });\n\n    this.#backend = backend;\n    this.#ow = new OpenWorkflow({ backend });\n    this.#worker = null;\n\n    this.#workflowsMap = new Map();\n    this.#scheduledTasks = new Map();\n  }\n\n  /**\n   * 정의된 워크플로우 및 워크플로우에 대한 scheduled tasks를 동기화합니다.\n   */\n  async synchronize(workflowMap: Map<string, WorkflowMetadata[]>) {\n    // 1. 삭제된 파일은 일괄 삭제\n    await Promise.allSettled(\n      Array.from(this.#workflowsMap.entries())\n        .filter(([key]) => !workflowMap.has(key))\n        .flatMap(([_, workflows]) =>\n          workflows.map((workflow) => {\n            return Promise.allSettled([\n              ...workflow.schedules.map((schedule) => this.unscheduleTask(schedule.name)),\n              this.unregisterWorkflow(workflow),\n            ]);\n          }),\n        ),\n    );\n\n    // 2. 새로 추가된 파일은 일괄 등록\n    await Promise.allSettled(\n      Array.from(workflowMap.entries())\n        .filter(([key]) => !this.#workflowsMap.has(key))\n        .flatMap(([_, workflows]) =>\n          workflows.map((workflow) => {\n            this.registerWorkflow({\n              id: workflow.id,\n              name: workflow.name,\n              version: workflow.version ?? null,\n              schema: workflow.schema,\n              function: workflow.fn,\n            });\n\n            return Promise.allSettled(\n              workflow.schedules.map((schedule) => this.scheduleTask(workflow, schedule)),\n            );\n          }),\n        ),\n    );\n\n    // 3. 기존과 다른 것을 diff친 다음, 삭제 후 재등록\n    await Promise.allSettled(\n      Array.from(workflowMap.entries())\n        .filter(([key]) => this.#workflowsMap.has(key))\n        .map<[string, [WorkflowMetadata[], WorkflowMetadata[]]]>(([key, newWorkflows]) => {\n          const previousWorkflows = this.#workflowsMap.get(key);\n          assert(previousWorkflows, \"previous workflows not found\");\n          return [key, [previousWorkflows, newWorkflows]];\n        })\n        .map(async ([_, [previousWorkflows, newWorkflows]]) => {\n          // 기존 것들을 삭제부터 해야함.\n          await Promise.allSettled(\n            previousWorkflows\n              .filter((prevItem) => !newWorkflows.some((newItem) => newItem.id === prevItem.id))\n              .map((prevItem) => {\n                return Promise.allSettled([\n                  ...prevItem.schedules.map((schedule) => this.unscheduleTask(schedule.name)),\n                  this.unregisterWorkflow(prevItem),\n                ]);\n              }),\n          );\n\n          // 새로 추가된 것들을 등록\n          await Promise.allSettled(\n            newWorkflows\n              .filter(\n                (newItem) => !previousWorkflows.some((prevItem) => prevItem.id === newItem.id),\n              )\n              .map(async (newItem) => {\n                this.registerWorkflow({\n                  id: newItem.id,\n                  name: newItem.name,\n                  version: newItem.version ?? null,\n                  schema: newItem.schema,\n                  function: newItem.fn,\n                });\n\n                return Promise.allSettled(\n                  newItem.schedules.map((schedule) => this.scheduleTask(newItem, schedule)),\n                );\n              }),\n          );\n        }),\n    );\n\n    this.#workflowsMap = workflowMap;\n  }\n\n  // 워크플로우를 실행\n  run<Input, Output, TSchema extends StandardSchemaV1 | undefined = undefined>(\n    options: Omit<WorkflowCreateOptions<Input, Output, TSchema>, \"function\" | \"schema\" | \"id\">,\n    input: SchemaInput<TSchema, Input>,\n  ): Promise<WorkflowRunHandle<Output>> {\n    return this.#ow.runWorkflow(\n      {\n        name: options.name,\n        version: options.version ?? undefined,\n      },\n      input,\n    );\n  }\n\n  // cron task를 등록\n  async scheduleTask(\n    workflow: Pick<WorkflowMetadata, \"id\" | \"name\" | \"version\">,\n    schedule: WorkflowMetadata[\"schedules\"][number],\n  ) {\n    const task = cronSchedule(\n      schedule.expression,\n      (async (\n        { name, version }: Pick<WorkflowMetadata, \"name\" | \"version\">,\n        { input }: WorkflowMetadata[\"schedules\"][number],\n      ) => {\n        const inputData = await (typeof input === \"function\"\n          ? Promise.resolve(input())\n          : Promise.resolve(input));\n\n        return this.run({ name, version: version ?? null }, inputData);\n      }).bind(this, workflow, schedule),\n      {\n        name: schedule.name,\n        timezone: Sonamu.config.api.timezone,\n        noOverlap: false,\n      },\n    );\n\n    this.#scheduledTasks.set(schedule.name, {\n      task,\n      inputFn: schedule.input,\n      workflowId: workflow.id,\n    });\n    await task.start();\n  }\n\n  // cron task를 중지\n  async unscheduleTask(name: string) {\n    const taskItem = this.#scheduledTasks.get(name);\n    if (!taskItem) {\n      console.error(\"scheduled task not found\", name);\n      return;\n    }\n\n    this.#scheduledTasks.delete(name);\n    await taskItem.task.stop();\n    await taskItem.task.destroy();\n  }\n\n  // 워크플로우를 등록, 관련된 스케줄은 별도로 등록해야 함.\n  registerWorkflow<Input, Output, TSchema extends StandardSchemaV1 | undefined = undefined>(\n    options: WorkflowCreateOptions<Input, Output, TSchema>,\n  ): RunnableWorkflow<SchemaOutput<TSchema, Input>, Output, SchemaInput<TSchema, Input>> {\n    const fn = async (\n      params: Readonly<{\n        input: SchemaOutput<TSchema, Input>;\n        step: StepApi;\n        version: string | null;\n      }>,\n    ) => {\n      const baseContext = {\n        request: null,\n        reply: null,\n        headers: {},\n        createSSE: (schema: ZodObject) => createMockSSEFactory(schema),\n        naiteStore: Naite.createStore(),\n        user: null,\n        passport: {\n          login: async () => {},\n          logout: () => {},\n        },\n      } as unknown as Context;\n\n      const contextProvider = Sonamu.config.tasks?.contextProvider;\n      const context: Context = contextProvider\n        ? await Promise.resolve(contextProvider(baseContext))\n        : baseContext;\n\n      const step = new StepWrapper(params.step);\n      return Sonamu.asyncLocalStorage.run({ context }, () =>\n        options.function({ input: params.input, step, version: params.version }),\n      );\n    };\n\n    const workflow = this.#ow.defineWorkflow(\n      {\n        name: options.name,\n        version: options.version ?? undefined,\n        schema: options.schema,\n      },\n      fn,\n    );\n    return workflow;\n  }\n\n  // 워크플로우를 등록 해제, 관련된 스케줄은 별도로 해제해야 함.\n  async unregisterWorkflow(workflow: Pick<WorkflowMetadata, \"name\" | \"version\" | \"id\">) {\n    this.#ow.unregisterWorkflow(workflow.name, workflow.version ?? null);\n  }\n\n  // Worker를 설정 후 시작\n  setupWorker(options: WorkflowOptions) {\n    this.#worker = this.#ow.newWorker(options);\n  }\n\n  // Worker를 초기화\n  async startWorker() {\n    if (!this.#worker) {\n      return;\n    }\n\n    await this.#backend.initialize();\n    await this.#worker?.start();\n  }\n\n  // Worker를 중지\n  async stopWorker() {\n    await this.#worker?.stop();\n  }\n\n  // cron task들을 모두 중지\n  async stopSchedules() {\n    await Promise.allSettled(\n      Array.from(this.#scheduledTasks.values()).map(({ task }) => Promise.resolve(task.stop())),\n    );\n  }\n\n  // cron task들을 모두 정리\n  async destroySchedules() {\n    await Promise.allSettled(\n      Array.from(this.#scheduledTasks.values()).map(({ task }) => Promise.resolve(task.destroy())),\n    );\n    this.#scheduledTasks.clear();\n  }\n\n  async destroy() {\n    await this.stopSchedules();\n    await this.stopWorker();\n    await this.destroySchedules();\n    await this.#backend.stop();\n  }\n\n  [Symbol.asyncDispose]() {\n    return this.destroy();\n  }\n}\n"],"names":["BackendPostgres","OpenWorkflow","assert","schedule","cronSchedule","Sonamu","Naite","createMockSSEFactory","StepWrapper","WorkflowManager","dbConf","runMigrations","backend","Map","synchronize","workflowMap","Promise","allSettled","Array","from","entries","filter","key","has","flatMap","_","workflows","map","workflow","schedules","unscheduleTask","name","unregisterWorkflow","registerWorkflow","id","version","schema","function","fn","scheduleTask","newWorkflows","previousWorkflows","get","prevItem","some","newItem","run","options","input","runWorkflow","undefined","task","expression","inputData","resolve","bind","timezone","config","api","noOverlap","set","inputFn","workflowId","start","taskItem","console","error","delete","stop","destroy","params","baseContext","request","reply","headers","createSSE","naiteStore","createStore","user","passport","login","logout","contextProvider","tasks","context","step","asyncLocalStorage","defineWorkflow","setupWorker","newWorker","startWorker","initialize","stopWorker","stopSchedules","values","destroySchedules","clear","Symbol","asyncDispose"],"mappings":"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"}
@@ -1 +1 @@
1
- {"version":3,"file":"generated.template.d.ts","sourceRoot":"","sources":["../../../src/template/implementations/generated.template.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,qBAAqB,CAAC;AAKlD,OAAO,EAAE,QAAQ,EAAE,MAAM,aAAa,CAAC;AAGvC,MAAM,MAAM,UAAU,GAAG;IACvB,KAAK,EAAE,MAAM,CAAC;IACd,KAAK,EAAE,MAAM,EAAE,CAAC;IAChB,UAAU,EAAE,MAAM,EAAE,CAAC;CACtB,CAAC;AACF,qBAAa,mBAAoB,SAAQ,QAAQ;;IAK/C,gBAAgB;;;;IAOhB,MAAM;;;;;;;IAmGN,kBAAkB,CAAC,MAAM,EAAE,MAAM,GAAG,UAAU,GAAG,IAAI;IAqBrD,uBAAuB,CAAC,MAAM,EAAE,MAAM,EAAE,UAAU,GAAE,MAAM,EAAO,GAAG,UAAU;IA6G9E,2BAA2B,CAAC,MAAM,EAAE,MAAM,GAAG,UAAU,GAAG,IAAI;IAgD9D,mBAAmB,CAAC,MAAM,EAAE,MAAM,GAAG,UAAU,GAAG,IAAI;CA8CvD"}
1
+ {"version":3,"file":"generated.template.d.ts","sourceRoot":"","sources":["../../../src/template/implementations/generated.template.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,qBAAqB,CAAC;AAUlD,OAAO,EAAE,QAAQ,EAAE,MAAM,aAAa,CAAC;AAGvC,MAAM,MAAM,UAAU,GAAG;IACvB,KAAK,EAAE,MAAM,CAAC;IACd,KAAK,EAAE,MAAM,EAAE,CAAC;IAChB,UAAU,EAAE,MAAM,EAAE,CAAC;CACtB,CAAC;AACF,qBAAa,mBAAoB,SAAQ,QAAQ;;IAK/C,gBAAgB;;;;IAOhB,MAAM;;;;;;;IAmGN,kBAAkB,CAAC,MAAM,EAAE,MAAM,GAAG,UAAU,GAAG,IAAI;IAqBrD,uBAAuB,CAAC,MAAM,EAAE,MAAM,EAAE,UAAU,GAAE,MAAM,EAAO,GAAG,UAAU;IAsH9E,2BAA2B,CAAC,MAAM,EAAE,MAAM,GAAG,UAAU,GAAG,IAAI;IAgD9D,mBAAmB,CAAC,MAAM,EAAE,MAAM,GAAG,UAAU,GAAG,IAAI;CA8CvD"}
@@ -3,7 +3,7 @@ import { unique } from "radashi";
3
3
  import { Sonamu } from "../../api/index.js";
4
4
  import { EntityManager } from "../../entity/entity-manager.js";
5
5
  import { Naite } from "../../naite/naite.js";
6
- import { isVirtualProp } from "../../types/types.js";
6
+ import { isVirtualCodeProp, isVirtualQueryProp } from "../../types/types.js";
7
7
  import { nonNullable } from "../../utils/utils.js";
8
8
  import { Template } from "../template.js";
9
9
  import { propNodeToZodTypeDef, zodTypeToZodCode } from "../zod-converter.js";
@@ -155,8 +155,10 @@ export class Template__generated extends Template {
155
155
  // fulltext index에 포함된 컬럼들 추출
156
156
  // TODO: GIN/GiST 인덱스 생성된 컬럼 추출
157
157
  const fulltextColumns = [];
158
- // virtual props
159
- const virtualProps = entity.props.filter((prop)=>isVirtualProp(prop)).map((prop)=>prop.name);
158
+ // virtual props (virtualType: "code" 또는 undefined인 것만 포함)
159
+ const virtualProps = entity.props.filter((prop)=>isVirtualCodeProp(prop)).map((prop)=>prop.name);
160
+ // query virtual props (virtualType: "query"인 것만 포함)
161
+ const virtualQueryProps = entity.props.filter((prop)=>isVirtualQueryProp(prop)).map((prop)=>prop.name);
160
162
  /**
161
163
  * hasDefault props
162
164
  * - nullable 또는 dbDefault가 있는 컬럼 (id 포함)
@@ -170,10 +172,10 @@ export class Template__generated extends Template {
170
172
  * generated props
171
173
  * - generated 속성이 있는 컬럼 (INSERT/UPDATE 시 값 제공 불가)
172
174
  */ const generatedColumns = entity.props.filter((prop)=>prop.type !== "relation" && prop.generated !== undefined).map((prop)=>prop.name);
173
- const hasMetadata = fulltextColumns.length > 0 || virtualProps.length > 0 || hasDefaultColumns.length > 0 || generatedColumns.length > 0 || hasVectorColumns.length > 0;
175
+ const hasMetadata = fulltextColumns.length > 0 || virtualProps.length > 0 || virtualQueryProps.length > 0 || hasDefaultColumns.length > 0 || generatedColumns.length > 0 || hasVectorColumns.length > 0;
174
176
  const lines = [
175
177
  `export const ${schemaName} = ${schemaBody};`,
176
- `export type ${schemaName} = z.infer<typeof ${schemaName}>` + (hasMetadata ? ` & {${(fulltextColumns.length > 0 ? `readonly __fulltext__: readonly [${fulltextColumns.map((col)=>`"${col}"`).join(", ")}],` : "") + (virtualProps.length > 0 ? `readonly __virtual__: readonly [${virtualProps.map((prop)=>`"${prop}"`).join(", ")}],` : "") + (hasDefaultColumns.length > 0 ? `readonly __hasDefault__: readonly [${hasDefaultColumns.map((col)=>`"${col}"`).join(", ")}],` : "") + (generatedColumns.length > 0 ? `readonly __generated__: readonly [${generatedColumns.map((col)=>`"${col}"`).join(", ")}],` : "") + (hasVectorColumns.length > 0 ? `readonly __vector__: readonly [${hasVectorColumns.map((col)=>`"${col}"`).join(", ")}],` : "")}}` : "") + ";"
178
+ `export type ${schemaName} = z.infer<typeof ${schemaName}>` + (hasMetadata ? ` & {${(fulltextColumns.length > 0 ? `readonly __fulltext__: readonly [${fulltextColumns.map((col)=>`"${col}"`).join(", ")}],` : "") + (virtualProps.length > 0 ? `readonly __virtual__: readonly [${virtualProps.map((prop)=>`"${prop}"`).join(", ")}],` : "") + (virtualQueryProps.length > 0 ? `readonly __virtual_query__: readonly [${virtualQueryProps.map((prop)=>`"${prop}"`).join(", ")}],` : "") + (hasDefaultColumns.length > 0 ? `readonly __hasDefault__: readonly [${hasDefaultColumns.map((col)=>`"${col}"`).join(", ")}],` : "") + (generatedColumns.length > 0 ? `readonly __generated__: readonly [${generatedColumns.map((col)=>`"${col}"`).join(", ")}],` : "") + (hasVectorColumns.length > 0 ? `readonly __vector__: readonly [${hasVectorColumns.map((col)=>`"${col}"`).join(", ")}],` : "")}}` : "") + ";"
177
179
  ];
178
180
  return {
179
181
  label: `BaseSchema: ${entity.id}`,
@@ -261,4 +263,4 @@ z.object({
261
263
  }
262
264
  }
263
265
 
264
- //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../src/template/implementations/generated.template.ts"],"sourcesContent":["import assert from \"assert\";\nimport { unique } from \"radashi\";\nimport { Sonamu } from \"../../api\";\nimport type { Entity } from \"../../entity/entity\";\nimport { EntityManager } from \"../../entity/entity-manager\";\nimport { Naite } from \"../../naite/naite\";\nimport { type EntityIndex, type EntityPropNode, isVirtualProp } from \"../../types/types\";\nimport { nonNullable } from \"../../utils/utils\";\nimport { Template } from \"../template\";\nimport { propNodeToZodTypeDef, zodTypeToZodCode } from \"../zod-converter\";\n\nexport type SourceCode = {\n  label: string;\n  lines: string[];\n  importKeys: string[];\n};\nexport class Template__generated extends Template {\n  constructor() {\n    super(\"generated\");\n  }\n\n  getTargetAndPath() {\n    const { dir } = Sonamu.config.api;\n    return {\n      target: `${dir}/src/application`,\n      path: `sonamu.generated.ts`,\n    };\n  }\n  render() {\n    const entityIds = EntityManager.getAllIds();\n    const entities = entityIds.map((id) => EntityManager.get(id));\n\n    // 전체 SourceCode 생성\n    const sourceCodes = entities.flatMap((entity) => {\n      return [\n        this.getEnumsSourceCode(entity),\n        this.getBaseSchemaSourceCode(entity),\n        this.getBaseListParamsSourceCode(entity),\n        this.getSubsetSourceCode(entity),\n      ].filter(nonNullable);\n    });\n    Naite.t(\"Template__generated:sourceCodes\", sourceCodes);\n\n    // Sort\n    const LABEL_KEY_ORDER = [\"Enums\", \"BaseSchema\", \"BaseListParams\", \"Subsets\", \"SubsetQueries\"];\n    sourceCodes.sort((a, b) => {\n      const [aKey] = a.label.split(\":\");\n      const [bKey] = b.label.split(\":\");\n      const aIndex = LABEL_KEY_ORDER.indexOf(aKey);\n      const bIndex = LABEL_KEY_ORDER.indexOf(bKey);\n      if (aIndex > bIndex) {\n        return 1;\n      } else if (aIndex < bIndex) {\n        return -1;\n      } else {\n        return 0;\n      }\n    });\n\n    const sourceCode = sourceCodes.reduce(\n      (result, ts) => {\n        if (ts === null) {\n          return result;\n        }\n        return {\n          lines: [...result.lines, `// ${ts.label}`, ...ts.lines, \"\"],\n          importKeys: unique([...result.importKeys, ...ts.importKeys].sort()),\n        };\n      },\n      {\n        lines: [],\n        importKeys: [],\n      } as Omit<SourceCode, \"label\">,\n    );\n\n    // .types.ts의 타입을 참조하는 경우 순환참조(상호참조)가 발생하므로 타입을 가져와 인라인 처리\n    const allTypeKeys = entities.flatMap((entity) => Object.keys(entity.types));\n    const cdImportKeys = sourceCode.importKeys.filter((importKey) =>\n      allTypeKeys.includes(importKey),\n    );\n    if (cdImportKeys.length > 0) {\n      const customScalarLines = cdImportKeys.flatMap((importKey) => {\n        const entity = entities.find((entity) => entity.types[importKey]);\n        if (!entity) {\n          throw new Error(`ZodType not found ${importKey}`);\n        }\n        const zodType = entity.types[importKey];\n        assert(zodType);\n\n        return [\n          `// CustomScalar: ${importKey}`,\n          `const ${importKey} = ${zodTypeToZodCode(zodType)};`,\n          `type ${importKey} = z.infer<typeof ${importKey}>`,\n          \"\",\n        ];\n      });\n      sourceCode.lines = [...customScalarLines, ...sourceCode.lines];\n      sourceCode.importKeys = sourceCode.importKeys.filter(\n        (importKey) => !cdImportKeys.includes(importKey),\n      );\n    }\n\n    const body = sourceCode.lines.join(\"\\n\");\n    Naite.t(\"Template__generated:body\", body);\n\n    // import\n    const sonamuImports = [\n      \"zArrayable\",\n      \"SQLDateTimeString\",\n      \"SubsetQuery\",\n      \"SonamuQueryMode\",\n    ].filter((mod) => body.includes(mod));\n\n    return {\n      ...this.getTargetAndPath(),\n      body,\n      importKeys: sourceCode.importKeys,\n      customHeaders: [\n        \"/** biome-ignore-all lint: generated는 무시 */\",\n        \"/** biome-ignore-all assist: generated는 무시 */\",\n        \"\",\n        `import { z } from 'zod';`,\n        `import { ${sonamuImports.join(\",\")} } from \"sonamu\";`,\n      ],\n    };\n  }\n\n  getEnumsSourceCode(entity: Entity): SourceCode | null {\n    if (Object.keys(entity.enumLabels).length === 0) {\n      return null;\n    }\n    return {\n      label: `Enums: ${entity.id}`,\n      lines: [\n        ...Object.entries(entity.enumLabels)\n          .filter(([_, enumLabel]) => Object.keys(enumLabel).length > 0)\n          .flatMap(([enumId, enumLabel]) => [\n            `export const ${enumId} = z.enum([${Object.keys(enumLabel).map(\n              (el) => `\"${el}\"`,\n            )}]).describe(\"${enumId}\");`,\n            `export type ${enumId} = z.infer<typeof ${enumId}>;`,\n            `export const ${enumId}Label = ${JSON.stringify(enumLabel)};`,\n          ]),\n      ],\n      importKeys: [],\n    };\n  }\n\n  getBaseSchemaSourceCode(entity: Entity, importKeys: string[] = []): SourceCode {\n    const schemaName = `${entity.names.module}BaseSchema`;\n    const propNode: EntityPropNode = {\n      nodeType: \"object\",\n      children: entity.props.map((prop) => {\n        return {\n          nodeType: \"plain\",\n          prop,\n        };\n      }),\n    };\n\n    const schemaBody = (() => {\n      const result = propNodeToZodTypeDef(propNode, importKeys);\n      if (result.endsWith(\",\")) {\n        return result.slice(0, -1);\n      }\n\n      return result;\n    })();\n\n    // fulltext index에 포함된 컬럼들 추출\n    // TODO: GIN/GiST 인덱스 생성된 컬럼 추출\n    const fulltextColumns: EntityIndex[\"columns\"][] = [];\n\n    // virtual props\n    const virtualProps = entity.props\n      .filter((prop) => isVirtualProp(prop))\n      .map((prop) => prop.name);\n\n    /**\n     * hasDefault props\n     * - nullable 또는 dbDefault가 있는 컬럼 (id 포함)\n     * - relation이 아니거나, relation이어도 nullable이면 포함\n     */\n    const hasDefaultColumns = entity.props\n      .filter(\n        (prop) =>\n          (prop.type !== \"relation\" || prop.nullable === true) &&\n          (prop.nullable === true || (prop.type !== \"relation\" && prop.dbDefault !== undefined)),\n      )\n      .map((prop) => (prop.type === \"relation\" ? `${prop.name}_id` : prop.name))\n      .concat(\"id\");\n\n    /**\n     * hasVector props\n     * - vector 타입인 컬럼\n     */\n    const hasVectorColumns = entity.props\n      .filter((prop) => prop.type === \"vector\" || prop.type === \"vector[]\")\n      .map((prop) => prop.name);\n\n    /**\n     * generated props\n     * - generated 속성이 있는 컬럼 (INSERT/UPDATE 시 값 제공 불가)\n     */\n    const generatedColumns = entity.props\n      .filter((prop) => prop.type !== \"relation\" && prop.generated !== undefined)\n      .map((prop) => prop.name);\n\n    const hasMetadata =\n      fulltextColumns.length > 0 ||\n      virtualProps.length > 0 ||\n      hasDefaultColumns.length > 0 ||\n      generatedColumns.length > 0 ||\n      hasVectorColumns.length > 0;\n\n    const lines = [\n      `export const ${schemaName} = ${schemaBody};`,\n      `export type ${schemaName} = z.infer<typeof ${schemaName}>` +\n        (hasMetadata\n          ? ` & {${\n              (fulltextColumns.length > 0\n                ? `readonly __fulltext__: readonly [${fulltextColumns\n                    .map((col) => `\"${col}\"`)\n                    .join(\", \")}],`\n                : \"\") +\n              (virtualProps.length > 0\n                ? `readonly __virtual__: readonly [${virtualProps.map((prop) => `\"${prop}\"`).join(\", \")}],`\n                : \"\") +\n              (hasDefaultColumns.length > 0\n                ? `readonly __hasDefault__: readonly [${hasDefaultColumns\n                    .map((col) => `\"${col}\"`)\n                    .join(\", \")}],`\n                : \"\") +\n              (\n                generatedColumns.length > 0\n                  ? `readonly __generated__: readonly [${generatedColumns\n                      .map((col) => `\"${col}\"`)\n                      .join(\", \")}],`\n                  : \"\"\n              ) +\n              (hasVectorColumns.length > 0\n                ? `readonly __vector__: readonly [${hasVectorColumns\n                    .map((col) => `\"${col}\"`)\n                    .join(\", \")}],`\n                : \"\")\n            }}`\n          : \"\") +\n        \";\",\n    ];\n\n    return {\n      label: `BaseSchema: ${entity.id}`,\n      importKeys,\n      lines,\n    };\n  }\n\n  getBaseListParamsSourceCode(entity: Entity): SourceCode | null {\n    // Prop 없는 MD인 경우 생성 제외\n    if (entity.props.length === 0) {\n      return null;\n    } else if (entity.parentId !== undefined) {\n      return null;\n    }\n\n    const schemaName = `${entity.names.module}BaseListParams`;\n\n    const filterProps = entity.props.filter((prop) => prop.toFilter === true);\n\n    const propNodes: EntityPropNode[] = filterProps.map((prop) => {\n      return {\n        nodeType: \"plain\" as const,\n        prop,\n        children: [],\n      };\n    });\n\n    const importKeys: string[] = [];\n    const filterBody = propNodes\n      .map((propNode) => propNodeToZodTypeDef(propNode, importKeys))\n      .join(\"\\n\");\n    const schemaBody = `\nz.object({\n  num: z.number().int().nonnegative(),\n  page: z.number().int().min(1),\n  search: ${entity.id}SearchField,\n  keyword: z.string(),\n  orderBy: ${entity.id}OrderBy,\n  queryMode: SonamuQueryMode,\n  id: zArrayable(z.number().int().positive()),${filterBody}\n}).partial();\n`.trim();\n\n    const lines = [\n      `export const ${schemaName} = ${schemaBody}`,\n      `export type ${schemaName} = z.infer<typeof ${schemaName}>;`,\n    ];\n\n    return {\n      label: `BaseListParams: ${entity.id}`,\n      importKeys,\n      lines,\n    };\n  }\n\n  getSubsetSourceCode(entity: Entity): SourceCode | null {\n    if (Object.keys(entity.subsets).length === 0) {\n      return null;\n    } else if (entity.parentId !== undefined) {\n      return null;\n    }\n\n    const subsetKeys = Object.keys(entity.subsets);\n    const importKeys: string[] = [];\n    const lines: string[] = [\n      ...subsetKeys.flatMap((subsetKey) => {\n        // 서브셋에서 FieldExpr[] 가져옴\n        const fieldExprs = entity.subsets[subsetKey];\n\n        // FieldExpr[]로 EntityPropNode[] 가져옴\n        const propNodes = entity.fieldExprsToPropNodes(fieldExprs);\n        const schemaName = `${entity.names.module}Subset${subsetKey}`;\n        const propNode: EntityPropNode = {\n          nodeType: \"object\",\n          children: propNodes,\n        };\n\n        // EntityPropNode[]로 ZodTypeDef(string)을 가져옴\n        const body = propNodeToZodTypeDef(propNode, importKeys);\n\n        return [\n          `export const ${schemaName} = ${body.replace(/,$/, \"\")};`,\n          `export type ${schemaName} = z.infer<typeof ${schemaName}>;`,\n        ];\n      }),\n      `export type ${entity.names.module}SubsetMapping = {`,\n      ...subsetKeys.map((subsetKey) => `  ${subsetKey}: ${entity.names.module}Subset${subsetKey};`),\n      \"};\",\n      `export const ${entity.names.module}SubsetKey = z.enum([${subsetKeys\n        .map((k) => `\"${k}\"`)\n        .join(\",\")}]);`,\n      `export type ${entity.names.module}SubsetKey = z.infer<typeof ${entity.names.module}SubsetKey>;`,\n      \"\",\n    ];\n\n    return {\n      label: `Subsets: ${entity.id}`,\n      lines,\n      importKeys: unique(importKeys),\n    };\n  }\n}\n"],"names":["assert","unique","Sonamu","EntityManager","Naite","isVirtualProp","nonNullable","Template","propNodeToZodTypeDef","zodTypeToZodCode","Template__generated","getTargetAndPath","dir","config","api","target","path","render","entityIds","getAllIds","entities","map","id","get","sourceCodes","flatMap","entity","getEnumsSourceCode","getBaseSchemaSourceCode","getBaseListParamsSourceCode","getSubsetSourceCode","filter","t","LABEL_KEY_ORDER","sort","a","b","aKey","label","split","bKey","aIndex","indexOf","bIndex","sourceCode","reduce","result","ts","lines","importKeys","allTypeKeys","Object","keys","types","cdImportKeys","importKey","includes","length","customScalarLines","find","Error","zodType","body","join","sonamuImports","mod","customHeaders","enumLabels","entries","_","enumLabel","enumId","el","JSON","stringify","schemaName","names","module","propNode","nodeType","children","props","prop","schemaBody","endsWith","slice","fulltextColumns","virtualProps","name","hasDefaultColumns","type","nullable","dbDefault","undefined","concat","hasVectorColumns","generatedColumns","generated","hasMetadata","col","parentId","filterProps","toFilter","propNodes","filterBody","trim","subsets","subsetKeys","subsetKey","fieldExprs","fieldExprsToPropNodes","replace","k"],"mappings":"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"}
266
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../src/template/implementations/generated.template.ts"],"sourcesContent":["import assert from \"assert\";\nimport { unique } from \"radashi\";\nimport { Sonamu } from \"../../api\";\nimport type { Entity } from \"../../entity/entity\";\nimport { EntityManager } from \"../../entity/entity-manager\";\nimport { Naite } from \"../../naite/naite\";\nimport {\n  type EntityIndex,\n  type EntityPropNode,\n  isVirtualCodeProp,\n  isVirtualQueryProp,\n} from \"../../types/types\";\nimport { nonNullable } from \"../../utils/utils\";\nimport { Template } from \"../template\";\nimport { propNodeToZodTypeDef, zodTypeToZodCode } from \"../zod-converter\";\n\nexport type SourceCode = {\n  label: string;\n  lines: string[];\n  importKeys: string[];\n};\nexport class Template__generated extends Template {\n  constructor() {\n    super(\"generated\");\n  }\n\n  getTargetAndPath() {\n    const { dir } = Sonamu.config.api;\n    return {\n      target: `${dir}/src/application`,\n      path: `sonamu.generated.ts`,\n    };\n  }\n  render() {\n    const entityIds = EntityManager.getAllIds();\n    const entities = entityIds.map((id) => EntityManager.get(id));\n\n    // 전체 SourceCode 생성\n    const sourceCodes = entities.flatMap((entity) => {\n      return [\n        this.getEnumsSourceCode(entity),\n        this.getBaseSchemaSourceCode(entity),\n        this.getBaseListParamsSourceCode(entity),\n        this.getSubsetSourceCode(entity),\n      ].filter(nonNullable);\n    });\n    Naite.t(\"Template__generated:sourceCodes\", sourceCodes);\n\n    // Sort\n    const LABEL_KEY_ORDER = [\"Enums\", \"BaseSchema\", \"BaseListParams\", \"Subsets\", \"SubsetQueries\"];\n    sourceCodes.sort((a, b) => {\n      const [aKey] = a.label.split(\":\");\n      const [bKey] = b.label.split(\":\");\n      const aIndex = LABEL_KEY_ORDER.indexOf(aKey);\n      const bIndex = LABEL_KEY_ORDER.indexOf(bKey);\n      if (aIndex > bIndex) {\n        return 1;\n      } else if (aIndex < bIndex) {\n        return -1;\n      } else {\n        return 0;\n      }\n    });\n\n    const sourceCode = sourceCodes.reduce(\n      (result, ts) => {\n        if (ts === null) {\n          return result;\n        }\n        return {\n          lines: [...result.lines, `// ${ts.label}`, ...ts.lines, \"\"],\n          importKeys: unique([...result.importKeys, ...ts.importKeys].sort()),\n        };\n      },\n      {\n        lines: [],\n        importKeys: [],\n      } as Omit<SourceCode, \"label\">,\n    );\n\n    // .types.ts의 타입을 참조하는 경우 순환참조(상호참조)가 발생하므로 타입을 가져와 인라인 처리\n    const allTypeKeys = entities.flatMap((entity) => Object.keys(entity.types));\n    const cdImportKeys = sourceCode.importKeys.filter((importKey) =>\n      allTypeKeys.includes(importKey),\n    );\n    if (cdImportKeys.length > 0) {\n      const customScalarLines = cdImportKeys.flatMap((importKey) => {\n        const entity = entities.find((entity) => entity.types[importKey]);\n        if (!entity) {\n          throw new Error(`ZodType not found ${importKey}`);\n        }\n        const zodType = entity.types[importKey];\n        assert(zodType);\n\n        return [\n          `// CustomScalar: ${importKey}`,\n          `const ${importKey} = ${zodTypeToZodCode(zodType)};`,\n          `type ${importKey} = z.infer<typeof ${importKey}>`,\n          \"\",\n        ];\n      });\n      sourceCode.lines = [...customScalarLines, ...sourceCode.lines];\n      sourceCode.importKeys = sourceCode.importKeys.filter(\n        (importKey) => !cdImportKeys.includes(importKey),\n      );\n    }\n\n    const body = sourceCode.lines.join(\"\\n\");\n    Naite.t(\"Template__generated:body\", body);\n\n    // import\n    const sonamuImports = [\n      \"zArrayable\",\n      \"SQLDateTimeString\",\n      \"SubsetQuery\",\n      \"SonamuQueryMode\",\n    ].filter((mod) => body.includes(mod));\n\n    return {\n      ...this.getTargetAndPath(),\n      body,\n      importKeys: sourceCode.importKeys,\n      customHeaders: [\n        \"/** biome-ignore-all lint: generated는 무시 */\",\n        \"/** biome-ignore-all assist: generated는 무시 */\",\n        \"\",\n        `import { z } from 'zod';`,\n        `import { ${sonamuImports.join(\",\")} } from \"sonamu\";`,\n      ],\n    };\n  }\n\n  getEnumsSourceCode(entity: Entity): SourceCode | null {\n    if (Object.keys(entity.enumLabels).length === 0) {\n      return null;\n    }\n    return {\n      label: `Enums: ${entity.id}`,\n      lines: [\n        ...Object.entries(entity.enumLabels)\n          .filter(([_, enumLabel]) => Object.keys(enumLabel).length > 0)\n          .flatMap(([enumId, enumLabel]) => [\n            `export const ${enumId} = z.enum([${Object.keys(enumLabel).map(\n              (el) => `\"${el}\"`,\n            )}]).describe(\"${enumId}\");`,\n            `export type ${enumId} = z.infer<typeof ${enumId}>;`,\n            `export const ${enumId}Label = ${JSON.stringify(enumLabel)};`,\n          ]),\n      ],\n      importKeys: [],\n    };\n  }\n\n  getBaseSchemaSourceCode(entity: Entity, importKeys: string[] = []): SourceCode {\n    const schemaName = `${entity.names.module}BaseSchema`;\n    const propNode: EntityPropNode = {\n      nodeType: \"object\",\n      children: entity.props.map((prop) => {\n        return {\n          nodeType: \"plain\",\n          prop,\n        };\n      }),\n    };\n\n    const schemaBody = (() => {\n      const result = propNodeToZodTypeDef(propNode, importKeys);\n      if (result.endsWith(\",\")) {\n        return result.slice(0, -1);\n      }\n\n      return result;\n    })();\n\n    // fulltext index에 포함된 컬럼들 추출\n    // TODO: GIN/GiST 인덱스 생성된 컬럼 추출\n    const fulltextColumns: EntityIndex[\"columns\"][] = [];\n\n    // virtual props (virtualType: \"code\" 또는 undefined인 것만 포함)\n    const virtualProps = entity.props\n      .filter((prop) => isVirtualCodeProp(prop))\n      .map((prop) => prop.name);\n\n    // query virtual props (virtualType: \"query\"인 것만 포함)\n    const virtualQueryProps = entity.props\n      .filter((prop) => isVirtualQueryProp(prop))\n      .map((prop) => prop.name);\n\n    /**\n     * hasDefault props\n     * - nullable 또는 dbDefault가 있는 컬럼 (id 포함)\n     * - relation이 아니거나, relation이어도 nullable이면 포함\n     */\n    const hasDefaultColumns = entity.props\n      .filter(\n        (prop) =>\n          (prop.type !== \"relation\" || prop.nullable === true) &&\n          (prop.nullable === true || (prop.type !== \"relation\" && prop.dbDefault !== undefined)),\n      )\n      .map((prop) => (prop.type === \"relation\" ? `${prop.name}_id` : prop.name))\n      .concat(\"id\");\n\n    /**\n     * hasVector props\n     * - vector 타입인 컬럼\n     */\n    const hasVectorColumns = entity.props\n      .filter((prop) => prop.type === \"vector\" || prop.type === \"vector[]\")\n      .map((prop) => prop.name);\n\n    /**\n     * generated props\n     * - generated 속성이 있는 컬럼 (INSERT/UPDATE 시 값 제공 불가)\n     */\n    const generatedColumns = entity.props\n      .filter((prop) => prop.type !== \"relation\" && prop.generated !== undefined)\n      .map((prop) => prop.name);\n\n    const hasMetadata =\n      fulltextColumns.length > 0 ||\n      virtualProps.length > 0 ||\n      virtualQueryProps.length > 0 ||\n      hasDefaultColumns.length > 0 ||\n      generatedColumns.length > 0 ||\n      hasVectorColumns.length > 0;\n\n    const lines = [\n      `export const ${schemaName} = ${schemaBody};`,\n      `export type ${schemaName} = z.infer<typeof ${schemaName}>` +\n        (hasMetadata\n          ? ` & {${\n              (fulltextColumns.length > 0\n                ? `readonly __fulltext__: readonly [${fulltextColumns\n                    .map((col) => `\"${col}\"`)\n                    .join(\", \")}],`\n                : \"\") +\n              (virtualProps.length > 0\n                ? `readonly __virtual__: readonly [${virtualProps.map((prop) => `\"${prop}\"`).join(\", \")}],`\n                : \"\") +\n              (virtualQueryProps.length > 0\n                ? `readonly __virtual_query__: readonly [${virtualQueryProps.map((prop) => `\"${prop}\"`).join(\", \")}],`\n                : \"\") +\n              (hasDefaultColumns.length > 0\n                ? `readonly __hasDefault__: readonly [${hasDefaultColumns\n                    .map((col) => `\"${col}\"`)\n                    .join(\", \")}],`\n                : \"\") +\n              (\n                generatedColumns.length > 0\n                  ? `readonly __generated__: readonly [${generatedColumns\n                      .map((col) => `\"${col}\"`)\n                      .join(\", \")}],`\n                  : \"\"\n              ) +\n              (hasVectorColumns.length > 0\n                ? `readonly __vector__: readonly [${hasVectorColumns\n                    .map((col) => `\"${col}\"`)\n                    .join(\", \")}],`\n                : \"\")\n            }}`\n          : \"\") +\n        \";\",\n    ];\n\n    return {\n      label: `BaseSchema: ${entity.id}`,\n      importKeys,\n      lines,\n    };\n  }\n\n  getBaseListParamsSourceCode(entity: Entity): SourceCode | null {\n    // Prop 없는 MD인 경우 생성 제외\n    if (entity.props.length === 0) {\n      return null;\n    } else if (entity.parentId !== undefined) {\n      return null;\n    }\n\n    const schemaName = `${entity.names.module}BaseListParams`;\n\n    const filterProps = entity.props.filter((prop) => prop.toFilter === true);\n\n    const propNodes: EntityPropNode[] = filterProps.map((prop) => {\n      return {\n        nodeType: \"plain\" as const,\n        prop,\n        children: [],\n      };\n    });\n\n    const importKeys: string[] = [];\n    const filterBody = propNodes\n      .map((propNode) => propNodeToZodTypeDef(propNode, importKeys))\n      .join(\"\\n\");\n    const schemaBody = `\nz.object({\n  num: z.number().int().nonnegative(),\n  page: z.number().int().min(1),\n  search: ${entity.id}SearchField,\n  keyword: z.string(),\n  orderBy: ${entity.id}OrderBy,\n  queryMode: SonamuQueryMode,\n  id: zArrayable(z.number().int().positive()),${filterBody}\n}).partial();\n`.trim();\n\n    const lines = [\n      `export const ${schemaName} = ${schemaBody}`,\n      `export type ${schemaName} = z.infer<typeof ${schemaName}>;`,\n    ];\n\n    return {\n      label: `BaseListParams: ${entity.id}`,\n      importKeys,\n      lines,\n    };\n  }\n\n  getSubsetSourceCode(entity: Entity): SourceCode | null {\n    if (Object.keys(entity.subsets).length === 0) {\n      return null;\n    } else if (entity.parentId !== undefined) {\n      return null;\n    }\n\n    const subsetKeys = Object.keys(entity.subsets);\n    const importKeys: string[] = [];\n    const lines: string[] = [\n      ...subsetKeys.flatMap((subsetKey) => {\n        // 서브셋에서 FieldExpr[] 가져옴\n        const fieldExprs = entity.subsets[subsetKey];\n\n        // FieldExpr[]로 EntityPropNode[] 가져옴\n        const propNodes = entity.fieldExprsToPropNodes(fieldExprs);\n        const schemaName = `${entity.names.module}Subset${subsetKey}`;\n        const propNode: EntityPropNode = {\n          nodeType: \"object\",\n          children: propNodes,\n        };\n\n        // EntityPropNode[]로 ZodTypeDef(string)을 가져옴\n        const body = propNodeToZodTypeDef(propNode, importKeys);\n\n        return [\n          `export const ${schemaName} = ${body.replace(/,$/, \"\")};`,\n          `export type ${schemaName} = z.infer<typeof ${schemaName}>;`,\n        ];\n      }),\n      `export type ${entity.names.module}SubsetMapping = {`,\n      ...subsetKeys.map((subsetKey) => `  ${subsetKey}: ${entity.names.module}Subset${subsetKey};`),\n      \"};\",\n      `export const ${entity.names.module}SubsetKey = z.enum([${subsetKeys\n        .map((k) => `\"${k}\"`)\n        .join(\",\")}]);`,\n      `export type ${entity.names.module}SubsetKey = z.infer<typeof ${entity.names.module}SubsetKey>;`,\n      \"\",\n    ];\n\n    return {\n      label: `Subsets: ${entity.id}`,\n      lines,\n      importKeys: unique(importKeys),\n    };\n  }\n}\n"],"names":["assert","unique","Sonamu","EntityManager","Naite","isVirtualCodeProp","isVirtualQueryProp","nonNullable","Template","propNodeToZodTypeDef","zodTypeToZodCode","Template__generated","getTargetAndPath","dir","config","api","target","path","render","entityIds","getAllIds","entities","map","id","get","sourceCodes","flatMap","entity","getEnumsSourceCode","getBaseSchemaSourceCode","getBaseListParamsSourceCode","getSubsetSourceCode","filter","t","LABEL_KEY_ORDER","sort","a","b","aKey","label","split","bKey","aIndex","indexOf","bIndex","sourceCode","reduce","result","ts","lines","importKeys","allTypeKeys","Object","keys","types","cdImportKeys","importKey","includes","length","customScalarLines","find","Error","zodType","body","join","sonamuImports","mod","customHeaders","enumLabels","entries","_","enumLabel","enumId","el","JSON","stringify","schemaName","names","module","propNode","nodeType","children","props","prop","schemaBody","endsWith","slice","fulltextColumns","virtualProps","name","virtualQueryProps","hasDefaultColumns","type","nullable","dbDefault","undefined","concat","hasVectorColumns","generatedColumns","generated","hasMetadata","col","parentId","filterProps","toFilter","propNodes","filterBody","trim","subsets","subsetKeys","subsetKey","fieldExprs","fieldExprsToPropNodes","replace","k"],"mappings":"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"}
@@ -55,7 +55,7 @@ class ${entityId}ModelClass extends BaseModelClass<
55
55
  > {
56
56
  modelName = "${entityId}";
57
57
 
58
- @api({ httpMethod: "GET", clients: ["axios", "swr"], resourceName: "${entityId}" })
58
+ @api({ httpMethod: "GET", clients: ["axios", "tanstack-query"], resourceName: "${entityId}" })
59
59
  async findById<T extends ${entityId}SubsetKey>(
60
60
  subset: T,
61
61
  id: number
@@ -85,7 +85,7 @@ class ${entityId}ModelClass extends BaseModelClass<
85
85
  return rows[0] ?? null;
86
86
  }
87
87
 
88
- @api({ httpMethod: "GET", clients: ["axios", "swr"], resourceName: "${names.capitalPlural}" })
88
+ @api({ httpMethod: "GET", clients: ["axios", "tanstack-query"], resourceName: "${names.capitalPlural}" })
89
89
  async findMany<T extends ${entityId}SubsetKey, LP extends ${entityId}ListParams>(
90
90
  subset: T,
91
91
  rawParams?: LP,
@@ -144,7 +144,7 @@ class ${entityId}ModelClass extends BaseModelClass<
144
144
  });
145
145
  }
146
146
 
147
- @api({ httpMethod: "POST" })
147
+ @api({ httpMethod: "POST", clients: ["axios", "tanstack-mutation"] })
148
148
  async save(
149
149
  spa: ${entityId}SaveParams[]
150
150
  ): Promise<number[]> {
@@ -163,7 +163,7 @@ class ${entityId}ModelClass extends BaseModelClass<
163
163
  });
164
164
  }
165
165
 
166
- @api({ httpMethod: "POST", guards: [ "admin" ] })
166
+ @api({ httpMethod: "POST", clients: ["axios", "tanstack-mutation"], guards: [ "admin" ] })
167
167
  async del(ids: number[]): Promise<number> {
168
168
  const wdb = this.getPuri("w");
169
169
 
@@ -183,4 +183,4 @@ export const ${entityId}Model = new ${entityId}ModelClass(${names.camel}SubsetQu
183
183
  }
184
184
  }
185
185
 
186
- //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../src/template/implementations/model.template.ts"],"sourcesContent":["import type z from \"zod\";\nimport { Sonamu } from \"../../api\";\nimport { EntityManager, type EntityNamesRecord } from \"../../entity/entity-manager\";\nimport { Naite } from \"../../naite/naite\";\nimport type { TemplateOptions } from \"../../types/types\";\nimport { Template } from \"../template\";\nimport { getZodTypeById, zodTypeToRenderingNode } from \"../zod-converter\";\nimport { Template__view_list } from \"./view_list.template\";\nexport class Template__model extends Template {\n  constructor() {\n    super(\"model\");\n  }\n\n  getTargetAndPath(names: EntityNamesRecord) {\n    const { dir } = Sonamu.config.api;\n\n    return {\n      target: `${dir}/src/application`,\n      path: `${names.fs}/${names.fs}.model.ts`,\n    };\n  }\n\n  async render({ entityId }: TemplateOptions[\"model\"]) {\n    Naite.t(\"render\", { entityId });\n\n    const listParamsZodType = await getZodTypeById(`${entityId}ListParams`);\n    const listParamsNode = zodTypeToRenderingNode(listParamsZodType);\n\n    const subsetKeyZodType = await getZodTypeById(`${entityId}SubsetKey`);\n    const subsetKeys = (subsetKeyZodType as z.ZodEnum).enum;\n\n    const names = EntityManager.getNamesFromId(entityId);\n    const entity = EntityManager.get(entityId);\n\n    const vlTpl = new Template__view_list();\n    if (listParamsNode?.children === undefined) {\n      throw new Error(`listParamsNode가 없습니다. ${entityId}`);\n    }\n    const def = vlTpl.getDefault(listParamsNode.children);\n\n    return {\n      ...this.getTargetAndPath(names),\n      body: `\nimport { BaseModelClass, type ListResult, asArray, NotFoundException, BadRequestException, api, exhaustive } from 'sonamu';\nimport {\n  ${entityId}SubsetKey,\n  ${entityId}SubsetMapping,\n} from \"../sonamu.generated\";\nimport {\n  ${names.camel}SubsetQueries,\n  ${names.camel}LoaderQueries,\n} from \"../sonamu.generated.sso\";\nimport { ${entityId}ListParams, ${entityId}SaveParams } from \"./${names.fs}.types\";\n\n/*\n  ${entityId} Model\n*/\nclass ${entityId}ModelClass extends BaseModelClass<\n  ${entityId}SubsetKey,\n  ${entityId}SubsetMapping,\n  typeof ${names.camel}SubsetQueries,\n  typeof ${names.camel}LoaderQueries\n> {\n  modelName = \"${entityId}\";\n\n  @api({ httpMethod: \"GET\", clients: [\"axios\", \"swr\"], resourceName: \"${entityId}\" })\n  async findById<T extends ${entityId}SubsetKey>(\n    subset: T,\n    id: number\n  ): Promise<${entityId}SubsetMapping[T]> {\n    const { rows } = await this.findMany(subset, {\n      id,\n      num: 1,\n      page: 1,\n    });\n    if (!rows[0]) {\n      throw new NotFoundException(\\`존재하지 않는 ${names.capital} ID \\${id}\\`);\n    }\n\n    return rows[0];\n  }\n\n  async findOne<T extends ${entityId}SubsetKey>(\n    subset: T,\n    listParams: ${entityId}ListParams\n  ): Promise<${entityId}SubsetMapping[T] | null> {\n    const { rows } = await this.findMany(subset, {\n      ...listParams,\n      num: 1,\n      page: 1,\n    });\n\n    return rows[0] ?? null;\n  }\n\n  @api({ httpMethod: \"GET\", clients: [\"axios\", \"swr\"], resourceName: \"${names.capitalPlural}\" })\n  async findMany<T extends ${entityId}SubsetKey, LP extends ${entityId}ListParams>(\n    subset: T,\n    rawParams?: LP,\n  ): Promise<ListResult<LP, ${entityId}SubsetMapping[T]>> {\n    // params with defaults\n    const params = {\n      num: 24,\n      page: 1,\n      search: \"${def.search}\" as const,\n      orderBy: \"${def.orderBy}\" as const,\n      ...rawParams,\n    } satisfies ${entityId}ListParams;\n\n    // build queries\n    const { qb, onSubset: _ } = this.getSubsetQueries(subset);\n    \n    // id\n    if (params.id) {\n      qb.whereIn(\"${entity.table}.id\", asArray(params.id));\n    }\n\n    // search-keyword\n    if (params.search && params.keyword && params.keyword.length > 0) {\n      if (params.search === \"id\") {\n        qb.where(\"${entity.table}.id\", Number(params.keyword));\n        // } else if (params.search === \"field\") {\n        //   qb.where(\"${entity.table}.field\", \"like\", \\`%\\${params.keyword}%\\`);\n      } else {\n        throw new BadRequestException(\\`구현되지 않은 검색 필드 \\${params.search}\\`);\n      }\n    }\n      \n    // orderBy\n    if (params.orderBy) {\n      // default orderBy\n      if (params.orderBy === \"id-desc\") {\n        qb.orderBy(\"${entity.table}.id\", \"desc\");\n      } else {\n        exhaustive(params.orderBy);\n      }\n    }\n\n    const enhancers = this.createEnhancers({\n      ${Object.keys(subsetKeys)\n        .map(\n          (key) => `${key}: (row) => ({\n          ...row,\n          // 서브셋별로 virtual 필드 계산로직 추가\n        }),`,\n        )\n        .join(\"\\n\")}\n    });\n\n    return this.executeSubsetQuery({\n      subset,\n      qb,\n      params,\n      enhancers,\n      debug: false,\n    });\n  }\n\n  @api({ httpMethod: \"POST\" })\n  async save(\n    spa: ${entityId}SaveParams[]\n  ): Promise<number[]> {\n    const wdb = this.getPuri(\"w\");\n\n    // register\n    spa.forEach((sp) => {\n      wdb.ubRegister(\"${entity.table}\", sp);\n    });\n\n    // transaction\n    return wdb.transaction(async (trx) => {\n      const ids = await trx.ubUpsert(\"${entity.table}\");\n\n      return ids;\n    });\n  }\n\n  @api({ httpMethod: \"POST\", guards: [ \"admin\" ] })\n  async del(ids: number[]): Promise<number> {\n    const wdb = this.getPuri(\"w\");\n\n    // transaction\n    await wdb.transaction(async (trx) => {\n      return trx.table(\"${entity.table}\").whereIn(\"${entity.table}.id\", ids).delete();\n    });\n\n    return ids.length;\n  }\n}\n\nexport const ${entityId}Model = new ${entityId}ModelClass(${names.camel}SubsetQueries, ${names.camel}LoaderQueries);\n      `.trim(),\n      importKeys: [],\n    };\n  }\n}\n"],"names":["Sonamu","EntityManager","Naite","Template","getZodTypeById","zodTypeToRenderingNode","Template__view_list","Template__model","getTargetAndPath","names","dir","config","api","target","path","fs","render","entityId","t","listParamsZodType","listParamsNode","subsetKeyZodType","subsetKeys","enum","getNamesFromId","entity","get","vlTpl","children","undefined","Error","def","getDefault","body","camel","capital","capitalPlural","search","orderBy","table","Object","keys","map","key","join","trim","importKeys"],"mappings":"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"}
186
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../src/template/implementations/model.template.ts"],"sourcesContent":["import type z from \"zod\";\nimport { Sonamu } from \"../../api\";\nimport { EntityManager, type EntityNamesRecord } from \"../../entity/entity-manager\";\nimport { Naite } from \"../../naite/naite\";\nimport type { TemplateOptions } from \"../../types/types\";\nimport { Template } from \"../template\";\nimport { getZodTypeById, zodTypeToRenderingNode } from \"../zod-converter\";\nimport { Template__view_list } from \"./view_list.template\";\nexport class Template__model extends Template {\n  constructor() {\n    super(\"model\");\n  }\n\n  getTargetAndPath(names: EntityNamesRecord) {\n    const { dir } = Sonamu.config.api;\n\n    return {\n      target: `${dir}/src/application`,\n      path: `${names.fs}/${names.fs}.model.ts`,\n    };\n  }\n\n  async render({ entityId }: TemplateOptions[\"model\"]) {\n    Naite.t(\"render\", { entityId });\n\n    const listParamsZodType = await getZodTypeById(`${entityId}ListParams`);\n    const listParamsNode = zodTypeToRenderingNode(listParamsZodType);\n\n    const subsetKeyZodType = await getZodTypeById(`${entityId}SubsetKey`);\n    const subsetKeys = (subsetKeyZodType as z.ZodEnum).enum;\n\n    const names = EntityManager.getNamesFromId(entityId);\n    const entity = EntityManager.get(entityId);\n\n    const vlTpl = new Template__view_list();\n    if (listParamsNode?.children === undefined) {\n      throw new Error(`listParamsNode가 없습니다. ${entityId}`);\n    }\n    const def = vlTpl.getDefault(listParamsNode.children);\n\n    return {\n      ...this.getTargetAndPath(names),\n      body: `\nimport { BaseModelClass, type ListResult, asArray, NotFoundException, BadRequestException, api, exhaustive } from 'sonamu';\nimport {\n  ${entityId}SubsetKey,\n  ${entityId}SubsetMapping,\n} from \"../sonamu.generated\";\nimport {\n  ${names.camel}SubsetQueries,\n  ${names.camel}LoaderQueries,\n} from \"../sonamu.generated.sso\";\nimport { ${entityId}ListParams, ${entityId}SaveParams } from \"./${names.fs}.types\";\n\n/*\n  ${entityId} Model\n*/\nclass ${entityId}ModelClass extends BaseModelClass<\n  ${entityId}SubsetKey,\n  ${entityId}SubsetMapping,\n  typeof ${names.camel}SubsetQueries,\n  typeof ${names.camel}LoaderQueries\n> {\n  modelName = \"${entityId}\";\n\n  @api({ httpMethod: \"GET\", clients: [\"axios\", \"tanstack-query\"], resourceName: \"${entityId}\" })\n  async findById<T extends ${entityId}SubsetKey>(\n    subset: T,\n    id: number\n  ): Promise<${entityId}SubsetMapping[T]> {\n    const { rows } = await this.findMany(subset, {\n      id,\n      num: 1,\n      page: 1,\n    });\n    if (!rows[0]) {\n      throw new NotFoundException(\\`존재하지 않는 ${names.capital} ID \\${id}\\`);\n    }\n\n    return rows[0];\n  }\n\n  async findOne<T extends ${entityId}SubsetKey>(\n    subset: T,\n    listParams: ${entityId}ListParams\n  ): Promise<${entityId}SubsetMapping[T] | null> {\n    const { rows } = await this.findMany(subset, {\n      ...listParams,\n      num: 1,\n      page: 1,\n    });\n\n    return rows[0] ?? null;\n  }\n\n  @api({ httpMethod: \"GET\", clients: [\"axios\", \"tanstack-query\"], resourceName: \"${names.capitalPlural}\" })\n  async findMany<T extends ${entityId}SubsetKey, LP extends ${entityId}ListParams>(\n    subset: T,\n    rawParams?: LP,\n  ): Promise<ListResult<LP, ${entityId}SubsetMapping[T]>> {\n    // params with defaults\n    const params = {\n      num: 24,\n      page: 1,\n      search: \"${def.search}\" as const,\n      orderBy: \"${def.orderBy}\" as const,\n      ...rawParams,\n    } satisfies ${entityId}ListParams;\n\n    // build queries\n    const { qb, onSubset: _ } = this.getSubsetQueries(subset);\n    \n    // id\n    if (params.id) {\n      qb.whereIn(\"${entity.table}.id\", asArray(params.id));\n    }\n\n    // search-keyword\n    if (params.search && params.keyword && params.keyword.length > 0) {\n      if (params.search === \"id\") {\n        qb.where(\"${entity.table}.id\", Number(params.keyword));\n        // } else if (params.search === \"field\") {\n        //   qb.where(\"${entity.table}.field\", \"like\", \\`%\\${params.keyword}%\\`);\n      } else {\n        throw new BadRequestException(\\`구현되지 않은 검색 필드 \\${params.search}\\`);\n      }\n    }\n      \n    // orderBy\n    if (params.orderBy) {\n      // default orderBy\n      if (params.orderBy === \"id-desc\") {\n        qb.orderBy(\"${entity.table}.id\", \"desc\");\n      } else {\n        exhaustive(params.orderBy);\n      }\n    }\n\n    const enhancers = this.createEnhancers({\n      ${Object.keys(subsetKeys)\n        .map(\n          (key) => `${key}: (row) => ({\n          ...row,\n          // 서브셋별로 virtual 필드 계산로직 추가\n        }),`,\n        )\n        .join(\"\\n\")}\n    });\n\n    return this.executeSubsetQuery({\n      subset,\n      qb,\n      params,\n      enhancers,\n      debug: false,\n    });\n  }\n\n  @api({ httpMethod: \"POST\", clients: [\"axios\", \"tanstack-mutation\"] })\n  async save(\n    spa: ${entityId}SaveParams[]\n  ): Promise<number[]> {\n    const wdb = this.getPuri(\"w\");\n\n    // register\n    spa.forEach((sp) => {\n      wdb.ubRegister(\"${entity.table}\", sp);\n    });\n\n    // transaction\n    return wdb.transaction(async (trx) => {\n      const ids = await trx.ubUpsert(\"${entity.table}\");\n\n      return ids;\n    });\n  }\n\n  @api({ httpMethod: \"POST\", clients: [\"axios\", \"tanstack-mutation\"], guards: [ \"admin\" ] })\n  async del(ids: number[]): Promise<number> {\n    const wdb = this.getPuri(\"w\");\n\n    // transaction\n    await wdb.transaction(async (trx) => {\n      return trx.table(\"${entity.table}\").whereIn(\"${entity.table}.id\", ids).delete();\n    });\n\n    return ids.length;\n  }\n}\n\nexport const ${entityId}Model = new ${entityId}ModelClass(${names.camel}SubsetQueries, ${names.camel}LoaderQueries);\n      `.trim(),\n      importKeys: [],\n    };\n  }\n}\n"],"names":["Sonamu","EntityManager","Naite","Template","getZodTypeById","zodTypeToRenderingNode","Template__view_list","Template__model","getTargetAndPath","names","dir","config","api","target","path","fs","render","entityId","t","listParamsZodType","listParamsNode","subsetKeyZodType","subsetKeys","enum","getNamesFromId","entity","get","vlTpl","children","undefined","Error","def","getDefault","body","camel","capital","capitalPlural","search","orderBy","table","Object","keys","map","key","join","trim","importKeys"],"mappings":"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"}
@@ -0,0 +1,17 @@
1
+ import type { TemplateOptions } from "../../types/types";
2
+ import { Template } from "../template";
3
+ export declare class Template__services extends Template {
4
+ constructor();
5
+ getTargetAndPath(): {
6
+ target: string;
7
+ path: string;
8
+ };
9
+ render({}: TemplateOptions["services"]): {
10
+ body: string;
11
+ importKeys: string[];
12
+ customHeaders: string[];
13
+ target: string;
14
+ path: string;
15
+ };
16
+ }
17
+ //# sourceMappingURL=services.template.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"services.template.d.ts","sourceRoot":"","sources":["../../../src/template/implementations/services.template.ts"],"names":[],"mappings":"AASA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,mBAAmB,CAAC;AAGzD,OAAO,EAAE,QAAQ,EAAE,MAAM,aAAa,CAAC;AAEvC,qBAAa,kBAAmB,SAAQ,QAAQ;;IAK9C,gBAAgB;;;;IAOhB,MAAM,CAAC,EAAE,EAAE,eAAe,CAAC,UAAU,CAAC;;;;;;;CAuMvC"}