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
@@ -1,202 +0,0 @@
1
- import assert from "assert";
2
- import inflection from "inflection";
3
- import { diff, group, sort, unique } from "radashi";
4
- import { apiParamToTsCode, apiParamToTsCodeAsObject, apiParamTypeToTsType, unwrapPromiseOnce } from "../../api/code-converters.js";
5
- import { Sonamu } from "../../api/sonamu.js";
6
- import { Naite } from "../../naite/naite.js";
7
- import { ApiParamType } from "../../types/types.js";
8
- import { assertDefined } from "../../utils/utils.js";
9
- import { Template } from "../template.js";
10
- import { zodTypeToTsTypeDef } from "../zod-converter.js";
11
- export class Template__service extends Template {
12
- constructor(){
13
- super("service");
14
- }
15
- getTargetAndPath(names) {
16
- return {
17
- target: ":target/src/services",
18
- path: `${names.fs}/${names.fs}.service.ts`
19
- };
20
- }
21
- render({ namesRecord }) {
22
- Naite.t("render", {
23
- namesRecord
24
- });
25
- const { syncer: { apis } } = Sonamu;
26
- const apisForThisModel = apis.filter((api)=>api.modelName === `${namesRecord.capital}Model` || api.modelName === `${namesRecord.capital}Frame`);
27
- // 서비스 TypeSource
28
- const { lines, importKeys } = this.getTypeSource(apisForThisModel);
29
- // AxiosProgressEvent 있는지 확인
30
- const hasAxiosProgressEvent = apis.find((api)=>(api.options.clients ?? []).includes("axios-multipart"));
31
- return {
32
- ...this.getTargetAndPath(namesRecord),
33
- body: lines.join("\n"),
34
- importKeys: importKeys.filter((key)=>[
35
- "ListResult"
36
- ].includes(key) === false),
37
- customHeaders: [
38
- `import { z } from 'zod';`,
39
- `import qs from "qs";`,
40
- `import useSWR, { type SWRResponse } from "swr";`,
41
- `import { fetch, ListResult, SWRError, SwrOptions, handleConditional, swrPostFetcher, EventHandlers, SSEStreamOptions, useSSEStream } from '../sonamu.shared';`,
42
- ...hasAxiosProgressEvent ? [
43
- `import { type AxiosProgressEvent } from 'axios';`
44
- ] : []
45
- ]
46
- };
47
- }
48
- getTypeSource(apis) {
49
- const importKeys = [];
50
- // 제네릭에서 선언한 타입, importKeys에서 제외 필요
51
- let typeParamNames = [];
52
- const groups = group(apis, (api)=>api.modelName);
53
- const body = Object.keys(groups).map((modelName)=>{
54
- const methods = groups[modelName];
55
- assert(methods);
56
- const methodCodes = methods.map((api)=>{
57
- // 컨텍스트 제외된 파라미터 리스트
58
- const paramsWithoutContext = api.parameters.filter((param)=>!ApiParamType.isContext(param.type) && !ApiParamType.isRefKnex(param.type) && !(param.optional === true && param.name.startsWith("_")));
59
- // 파라미터 타입 정의
60
- const typeParametersAsTsType = api.typeParameters.map((typeParam)=>{
61
- return apiParamTypeToTsType(typeParam, importKeys);
62
- }).join(", ");
63
- const typeParamsDef = typeParametersAsTsType ? `<${typeParametersAsTsType}>` : "";
64
- typeParamNames = typeParamNames.concat(api.typeParameters.map((typeParam)=>typeParam.id));
65
- // 파라미터 정의
66
- const paramsDef = apiParamToTsCode(paramsWithoutContext, importKeys);
67
- // 파라미터 정의 (객체 형태)
68
- const paramsDefAsObject = apiParamToTsCodeAsObject(paramsWithoutContext, importKeys);
69
- // 리턴 타입 정의
70
- const returnTypeDef = apiParamTypeToTsType(assertDefined(unwrapPromiseOnce(api.returnType)), importKeys);
71
- // 페이로드 데이터 정의
72
- const payloadDef = `{ ${paramsWithoutContext.map((param)=>param.name).join(", ")} }`;
73
- // 기본 URL
74
- const apiBaseUrl = `${Sonamu.config.api.route.prefix}${api.path}`;
75
- const clients = api.options.clients ?? [];
76
- return [
77
- // 클라이언트별로 생성
78
- ...sort(clients, (client)=>client === "swr" ? 0 : 1).map((client)=>{
79
- switch(client){
80
- case "axios":
81
- return this.renderAxios(api, apiBaseUrl, typeParamsDef, paramsDef, returnTypeDef, payloadDef);
82
- case "axios-multipart":
83
- return this.renderAxiosMultipart(api, apiBaseUrl, typeParamsDef, paramsDef, returnTypeDef, paramsWithoutContext);
84
- case "swr":
85
- return this.renderSwr(api, apiBaseUrl, typeParamsDef, paramsDef, returnTypeDef, payloadDef);
86
- case "window-fetch":
87
- return this.renderWindowFetch(api, apiBaseUrl, typeParamsDef, paramsDef, payloadDef);
88
- default:
89
- return `// Not supported ${inflection.camelize(client, true)} yet.`;
90
- }
91
- }),
92
- // 스트리밍인 경우
93
- ...api.streamOptions ? [
94
- this.renderStream(api, apiBaseUrl, paramsDefAsObject)
95
- ] : []
96
- ].join("\n");
97
- }).join("\n\n");
98
- return `export namespace ${modelName.replace(/Model$/, "Service").replace(/Frame$/, "Service")} {
99
- ${methodCodes}
100
- }`;
101
- }).join("\n\n");
102
- return {
103
- lines: [
104
- body
105
- ],
106
- importKeys: diff(unique(importKeys), typeParamNames)
107
- };
108
- }
109
- renderAxios(api, apiBaseUrl, typeParamsDef, paramsDef, returnTypeDef, payloadDef) {
110
- const methodNameAxios = api.options.resourceName ? `get${inflection.camelize(api.options.resourceName)}` : api.methodName;
111
- if (api.options.httpMethod === "GET") {
112
- return `
113
- export async function ${methodNameAxios}${typeParamsDef}(${paramsDef}): Promise<${returnTypeDef}> {
114
- return fetch({
115
- method: "GET",
116
- url: \`${apiBaseUrl}?\${qs.stringify(${payloadDef})}\`,
117
- ${api.options.timeout ? `signal: AbortSignal.timeout(${api.options.timeout}),` : ""}
118
- });
119
- }
120
- `.trim();
121
- } else {
122
- return `
123
- export async function ${methodNameAxios}${typeParamsDef}(${paramsDef}): Promise<${returnTypeDef}> {
124
- return fetch({
125
- method: '${api.options.httpMethod}',
126
- url: \`${apiBaseUrl}\`,
127
- data: ${payloadDef},
128
- ${api.options.timeout ? `signal: AbortSignal.timeout(${api.options.timeout}),` : ""}
129
- });
130
- }
131
- `.trim();
132
- }
133
- }
134
- renderAxiosMultipart(api, apiBaseUrl, typeParamsDef, paramsDef, returnTypeDef, paramsWithoutContext) {
135
- const isMultiple = api.uploadOptions?.mode === "multiple";
136
- const fileParamName = isMultiple ? "files" : "file";
137
- const fileParamType = isMultiple ? "File[]" : "File";
138
- const formDataDef = isMultiple ? [
139
- `${fileParamName}.forEach(f => { formData.append("${fileParamName}", f) } ); `,
140
- ...paramsWithoutContext.map((param)=>`formData.append('${param.name}', String(${param.name}));`)
141
- ].join("\n") : [
142
- `formData.append("${fileParamName}", ${fileParamName});`,
143
- ...paramsWithoutContext.map((param)=>`formData.append('${param.name}', String(${param.name}));`)
144
- ].join("\n");
145
- const paramsDefComma = paramsDef !== "" ? ", " : "";
146
- return `
147
- export async function ${api.methodName}${typeParamsDef}(
148
- ${paramsDef}${paramsDefComma}
149
- ${fileParamName}: ${fileParamType},
150
- onUploadProgress?: (pe:AxiosProgressEvent) => void
151
- ): Promise<${returnTypeDef}> {
152
- const formData = new FormData();
153
- ${formDataDef}
154
- return fetch({
155
- method: 'POST',
156
- url: \`${apiBaseUrl}\`,
157
- headers: {
158
- "Content-Type": "multipart/form-data",
159
- },
160
- onUploadProgress,
161
- data: formData,
162
- ${api.options.timeout ? `signal: AbortSignal.timeout(${api.options.timeout}),` : ""}
163
- });
164
- }
165
- `.trim();
166
- }
167
- renderSwr(api, apiBaseUrl, typeParamsDef, paramsDef, returnTypeDef, payloadDef) {
168
- const methodNameSwr = api.options.resourceName ? `use${inflection.camelize(api.options.resourceName)}` : `use${inflection.camelize(api.methodName)}`;
169
- return ` export function ${inflection.camelize(methodNameSwr, true)}${typeParamsDef}(${[
170
- paramsDef,
171
- "swrOptions?: SwrOptions"
172
- ].filter((p)=>p !== "").join(",")}, ): SWRResponse<${returnTypeDef}, SWRError> {
173
- return useSWR(handleConditional([
174
- \`${apiBaseUrl}\`,
175
- ${payloadDef},
176
- ], swrOptions?.conditional)${api.options.httpMethod === "POST" ? ", swrPostFetcher" : ""}${api.options.timeout ? `, { loadingTimeout: ${api.options.timeout} }` : ""});
177
- }`;
178
- }
179
- renderWindowFetch(api, apiBaseUrl, typeParamsDef, paramsDef, payloadDef) {
180
- return `
181
- export async function ${api.methodName}${typeParamsDef}(${paramsDef}): Promise<Response> {
182
- return window.fetch(\`${apiBaseUrl}?\${qs.stringify(${payloadDef})}\`${api.options.timeout ? `, { signal: AbortSignal.timeout(${api.options.timeout}) }` : ""});
183
- }
184
- `.trim();
185
- }
186
- renderStream(api, apiBaseUrl, paramsDefAsObject) {
187
- if (!api.streamOptions) {
188
- return "// streamOptions not found";
189
- }
190
- const methodNameStream = api.options.resourceName ? `use${inflection.camelize(api.options.resourceName)}` : `use${inflection.camelize(api.methodName)}`;
191
- const methodNameStreamCamelized = inflection.camelize(methodNameStream, true);
192
- const eventsTypeDef = zodTypeToTsTypeDef(api.streamOptions.events);
193
- return ` export function ${methodNameStreamCamelized}(
194
- params: ${paramsDefAsObject},
195
- handlers: EventHandlers<${eventsTypeDef} & { end?: () => void }>,
196
- options: SSEStreamOptions) {
197
- return useSSEStream<${eventsTypeDef}>(\`${apiBaseUrl}\`, params, handlers, options);
198
- }`;
199
- }
200
- }
201
-
202
- //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../src/template/implementations/service.template.ts"],"sourcesContent":["import assert from \"assert\";\nimport inflection from \"inflection\";\nimport { diff, group, sort, unique } from \"radashi\";\nimport {\n  apiParamToTsCode,\n  apiParamToTsCodeAsObject,\n  apiParamTypeToTsType,\n  unwrapPromiseOnce,\n} from \"../../api/code-converters\";\nimport type { ExtendedApi } from \"../../api/decorators\";\nimport { Sonamu } from \"../../api/sonamu\";\nimport type { EntityNamesRecord } from \"../../entity/entity-manager\";\nimport { Naite } from \"../../naite/naite\";\nimport type { TemplateOptions } from \"../../types/types\";\nimport { type ApiParam, ApiParamType } from \"../../types/types\";\nimport { assertDefined } from \"../../utils/utils\";\nimport { Template } from \"../template\";\nimport { zodTypeToTsTypeDef } from \"../zod-converter\";\n\nexport class Template__service extends Template {\n  constructor() {\n    super(\"service\");\n  }\n\n  getTargetAndPath(names: EntityNamesRecord) {\n    return {\n      target: \":target/src/services\",\n      path: `${names.fs}/${names.fs}.service.ts`,\n    };\n  }\n\n  render({ namesRecord }: TemplateOptions[\"service\"]) {\n    Naite.t(\"render\", { namesRecord });\n\n    const {\n      syncer: { apis },\n    } = Sonamu;\n\n    const apisForThisModel = apis.filter(\n      (api) =>\n        api.modelName === `${namesRecord.capital}Model` ||\n        api.modelName === `${namesRecord.capital}Frame`,\n    );\n\n    // 서비스 TypeSource\n    const { lines, importKeys } = this.getTypeSource(apisForThisModel);\n\n    // AxiosProgressEvent 있는지 확인\n    const hasAxiosProgressEvent = apis.find((api) =>\n      (api.options.clients ?? []).includes(\"axios-multipart\"),\n    );\n\n    return {\n      ...this.getTargetAndPath(namesRecord),\n      body: lines.join(\"\\n\"),\n      importKeys: importKeys.filter((key) => [\"ListResult\"].includes(key) === false),\n      customHeaders: [\n        `import { z } from 'zod';`,\n        `import qs from \"qs\";`,\n        `import useSWR, { type SWRResponse } from \"swr\";`,\n        `import { fetch, ListResult, SWRError, SwrOptions, handleConditional, swrPostFetcher, EventHandlers, SSEStreamOptions, useSSEStream } from '../sonamu.shared';`,\n        ...(hasAxiosProgressEvent ? [`import { type AxiosProgressEvent } from 'axios';`] : []),\n      ],\n    };\n  }\n\n  getTypeSource(apis: ExtendedApi[]): {\n    lines: string[];\n    importKeys: string[];\n  } {\n    const importKeys: string[] = [];\n\n    // 제네릭에서 선언한 타입, importKeys에서 제외 필요\n    let typeParamNames: string[] = [];\n\n    const groups = group(apis, (api) => api.modelName);\n    const body = Object.keys(groups)\n      .map((modelName) => {\n        const methods = groups[modelName];\n        assert(methods);\n        const methodCodes = methods\n          .map((api) => {\n            // 컨텍스트 제외된 파라미터 리스트\n            const paramsWithoutContext = api.parameters.filter(\n              (param) =>\n                !ApiParamType.isContext(param.type) &&\n                !ApiParamType.isRefKnex(param.type) &&\n                !(param.optional === true && param.name.startsWith(\"_\")), // _로 시작하는 파라미터는 제외\n            );\n\n            // 파라미터 타입 정의\n            const typeParametersAsTsType = api.typeParameters\n              .map((typeParam) => {\n                return apiParamTypeToTsType(typeParam, importKeys);\n              })\n              .join(\", \");\n            const typeParamsDef = typeParametersAsTsType ? `<${typeParametersAsTsType}>` : \"\";\n            typeParamNames = typeParamNames.concat(\n              api.typeParameters.map((typeParam) => typeParam.id),\n            );\n\n            // 파라미터 정의\n            const paramsDef = apiParamToTsCode(paramsWithoutContext, importKeys);\n\n            // 파라미터 정의 (객체 형태)\n            const paramsDefAsObject = apiParamToTsCodeAsObject(paramsWithoutContext, importKeys);\n\n            // 리턴 타입 정의\n            const returnTypeDef = apiParamTypeToTsType(\n              assertDefined(unwrapPromiseOnce(api.returnType)),\n              importKeys,\n            );\n\n            // 페이로드 데이터 정의\n            const payloadDef = `{ ${paramsWithoutContext.map((param) => param.name).join(\", \")} }`;\n\n            // 기본 URL\n            const apiBaseUrl = `${Sonamu.config.api.route.prefix}${api.path}`;\n\n            const clients = api.options.clients ?? [];\n            return [\n              // 클라이언트별로 생성\n              ...sort(clients, (client) => (client === \"swr\" ? 0 : 1)).map((client) => {\n                switch (client) {\n                  case \"axios\":\n                    return this.renderAxios(\n                      api,\n                      apiBaseUrl,\n                      typeParamsDef,\n                      paramsDef,\n                      returnTypeDef,\n                      payloadDef,\n                    );\n                  case \"axios-multipart\":\n                    return this.renderAxiosMultipart(\n                      api,\n                      apiBaseUrl,\n                      typeParamsDef,\n                      paramsDef,\n                      returnTypeDef,\n                      paramsWithoutContext,\n                    );\n                  case \"swr\":\n                    return this.renderSwr(\n                      api,\n                      apiBaseUrl,\n                      typeParamsDef,\n                      paramsDef,\n                      returnTypeDef,\n                      payloadDef,\n                    );\n                  case \"window-fetch\":\n                    return this.renderWindowFetch(\n                      api,\n                      apiBaseUrl,\n                      typeParamsDef,\n                      paramsDef,\n                      payloadDef,\n                    );\n                  default:\n                    return `// Not supported ${inflection.camelize(client, true)} yet.`;\n                }\n              }),\n              // 스트리밍인 경우\n              ...(api.streamOptions ? [this.renderStream(api, apiBaseUrl, paramsDefAsObject)] : []),\n            ].join(\"\\n\");\n          })\n          .join(\"\\n\\n\");\n\n        return `export namespace ${modelName.replace(/Model$/, \"Service\").replace(/Frame$/, \"Service\")} {\n${methodCodes}\n}`;\n      })\n      .join(\"\\n\\n\");\n\n    return {\n      lines: [body],\n      importKeys: diff(unique(importKeys), typeParamNames),\n    };\n  }\n\n  renderAxios(\n    api: ExtendedApi,\n    apiBaseUrl: string,\n    typeParamsDef: string,\n    paramsDef: string,\n    returnTypeDef: string,\n    payloadDef: string,\n  ) {\n    const methodNameAxios = api.options.resourceName\n      ? `get${inflection.camelize(api.options.resourceName)}`\n      : api.methodName;\n\n    if (api.options.httpMethod === \"GET\") {\n      return `\nexport async function ${methodNameAxios}${typeParamsDef}(${paramsDef}): Promise<${returnTypeDef}> {\n    return fetch({\n      method: \"GET\",\n      url: \\`${apiBaseUrl}?\\${qs.stringify(${payloadDef})}\\`,\n      ${api.options.timeout ? `signal: AbortSignal.timeout(${api.options.timeout}),` : \"\"}\n    });\n}\n    `.trim();\n    } else {\n      return `\nexport async function ${methodNameAxios}${typeParamsDef}(${paramsDef}): Promise<${returnTypeDef}> {\n    return fetch({\n      method: '${api.options.httpMethod}',\n      url: \\`${apiBaseUrl}\\`,\n      data: ${payloadDef},\n      ${api.options.timeout ? `signal: AbortSignal.timeout(${api.options.timeout}),` : \"\"}\n    });\n}\n      `.trim();\n    }\n  }\n\n  renderAxiosMultipart(\n    api: ExtendedApi,\n    apiBaseUrl: string,\n    typeParamsDef: string,\n    paramsDef: string,\n    returnTypeDef: string,\n    paramsWithoutContext: ApiParam[],\n  ) {\n    const isMultiple = api.uploadOptions?.mode === \"multiple\";\n    const fileParamName = isMultiple ? \"files\" : \"file\";\n    const fileParamType = isMultiple ? \"File[]\" : \"File\";\n\n    const formDataDef = isMultiple\n      ? [\n          `${fileParamName}.forEach(f => { formData.append(\"${fileParamName}\", f) } ); `,\n          ...paramsWithoutContext.map(\n            (param) => `formData.append('${param.name}', String(${param.name}));`,\n          ),\n        ].join(\"\\n\")\n      : [\n          `formData.append(\"${fileParamName}\", ${fileParamName});`,\n          ...paramsWithoutContext.map(\n            (param) => `formData.append('${param.name}', String(${param.name}));`,\n          ),\n        ].join(\"\\n\");\n\n    const paramsDefComma = paramsDef !== \"\" ? \", \" : \"\";\n    return `\nexport async function ${api.methodName}${typeParamsDef}(\n  ${paramsDef}${paramsDefComma}\n  ${fileParamName}: ${fileParamType},\n  onUploadProgress?: (pe:AxiosProgressEvent) => void\n  ): Promise<${returnTypeDef}> {\n    const formData = new FormData();\n    ${formDataDef}\n    return fetch({\n      method: 'POST',\n      url: \\`${apiBaseUrl}\\`,\n      headers: {\n        \"Content-Type\": \"multipart/form-data\",\n      },\n      onUploadProgress,\n      data: formData,\n      ${api.options.timeout ? `signal: AbortSignal.timeout(${api.options.timeout}),` : \"\"}\n    });\n  }\n  `.trim();\n  }\n\n  renderSwr(\n    api: ExtendedApi,\n    apiBaseUrl: string,\n    typeParamsDef: string,\n    paramsDef: string,\n    returnTypeDef: string,\n    payloadDef: string,\n  ) {\n    const methodNameSwr = api.options.resourceName\n      ? `use${inflection.camelize(api.options.resourceName)}`\n      : `use${inflection.camelize(api.methodName)}`;\n    return `  export function ${inflection.camelize(methodNameSwr, true)}${typeParamsDef}(${[\n      paramsDef,\n      \"swrOptions?: SwrOptions\",\n    ]\n      .filter((p) => p !== \"\")\n      .join(\",\")}, ): SWRResponse<${returnTypeDef}, SWRError> {\n    return useSWR(handleConditional([\n      \\`${apiBaseUrl}\\`,\n      ${payloadDef},\n    ], swrOptions?.conditional)${api.options.httpMethod === \"POST\" ? \", swrPostFetcher\" : \"\"}${\n      api.options.timeout ? `, { loadingTimeout: ${api.options.timeout} }` : \"\"\n    });\n  }`;\n  }\n\n  renderWindowFetch(\n    api: ExtendedApi,\n    apiBaseUrl: string,\n    typeParamsDef: string,\n    paramsDef: string,\n    payloadDef: string,\n  ) {\n    return `\nexport async function ${api.methodName}${typeParamsDef}(${paramsDef}): Promise<Response> {\n    return window.fetch(\\`${apiBaseUrl}?\\${qs.stringify(${payloadDef})}\\`${\n      api.options.timeout ? `, { signal: AbortSignal.timeout(${api.options.timeout}) }` : \"\"\n    });\n}\n    `.trim();\n  }\n\n  renderStream(api: ExtendedApi, apiBaseUrl: string, paramsDefAsObject: string) {\n    if (!api.streamOptions) {\n      return \"// streamOptions not found\";\n    }\n\n    const methodNameStream = api.options.resourceName\n      ? `use${inflection.camelize(api.options.resourceName)}`\n      : `use${inflection.camelize(api.methodName)}`;\n    const methodNameStreamCamelized = inflection.camelize(methodNameStream, true);\n\n    const eventsTypeDef = zodTypeToTsTypeDef(api.streamOptions.events);\n\n    return `  export function ${methodNameStreamCamelized}(\n  params: ${paramsDefAsObject},\n  handlers: EventHandlers<${eventsTypeDef} & { end?: () => void }>,\n  options: SSEStreamOptions) {\n    return useSSEStream<${eventsTypeDef}>(\\`${apiBaseUrl}\\`, params, handlers, options);\n  }`;\n  }\n}\n"],"names":["assert","inflection","diff","group","sort","unique","apiParamToTsCode","apiParamToTsCodeAsObject","apiParamTypeToTsType","unwrapPromiseOnce","Sonamu","Naite","ApiParamType","assertDefined","Template","zodTypeToTsTypeDef","Template__service","getTargetAndPath","names","target","path","fs","render","namesRecord","t","syncer","apis","apisForThisModel","filter","api","modelName","capital","lines","importKeys","getTypeSource","hasAxiosProgressEvent","find","options","clients","includes","body","join","key","customHeaders","typeParamNames","groups","Object","keys","map","methods","methodCodes","paramsWithoutContext","parameters","param","isContext","type","isRefKnex","optional","name","startsWith","typeParametersAsTsType","typeParameters","typeParam","typeParamsDef","concat","id","paramsDef","paramsDefAsObject","returnTypeDef","returnType","payloadDef","apiBaseUrl","config","route","prefix","client","renderAxios","renderAxiosMultipart","renderSwr","renderWindowFetch","camelize","streamOptions","renderStream","replace","methodNameAxios","resourceName","methodName","httpMethod","timeout","trim","isMultiple","uploadOptions","mode","fileParamName","fileParamType","formDataDef","paramsDefComma","methodNameSwr","p","methodNameStream","methodNameStreamCamelized","eventsTypeDef","events"],"mappings":"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"}
@@ -1 +0,0 @@
1
- throw new Error('Could not resolve "@valibot/to-json-schema" imported by "@ai-sdk/provider-utils".');
@@ -1 +0,0 @@
1
- throw new Error('Could not resolve "effect" imported by "@ai-sdk/provider-utils".');
@@ -1,328 +0,0 @@
1
- import assert from "assert";
2
- import inflection from "inflection";
3
- import { diff, group, sort, unique } from "radashi";
4
- import {
5
- apiParamToTsCode,
6
- apiParamToTsCodeAsObject,
7
- apiParamTypeToTsType,
8
- unwrapPromiseOnce,
9
- } from "../../api/code-converters";
10
- import type { ExtendedApi } from "../../api/decorators";
11
- import { Sonamu } from "../../api/sonamu";
12
- import type { EntityNamesRecord } from "../../entity/entity-manager";
13
- import { Naite } from "../../naite/naite";
14
- import type { TemplateOptions } from "../../types/types";
15
- import { type ApiParam, ApiParamType } from "../../types/types";
16
- import { assertDefined } from "../../utils/utils";
17
- import { Template } from "../template";
18
- import { zodTypeToTsTypeDef } from "../zod-converter";
19
-
20
- export class Template__service extends Template {
21
- constructor() {
22
- super("service");
23
- }
24
-
25
- getTargetAndPath(names: EntityNamesRecord) {
26
- return {
27
- target: ":target/src/services",
28
- path: `${names.fs}/${names.fs}.service.ts`,
29
- };
30
- }
31
-
32
- render({ namesRecord }: TemplateOptions["service"]) {
33
- Naite.t("render", { namesRecord });
34
-
35
- const {
36
- syncer: { apis },
37
- } = Sonamu;
38
-
39
- const apisForThisModel = apis.filter(
40
- (api) =>
41
- api.modelName === `${namesRecord.capital}Model` ||
42
- api.modelName === `${namesRecord.capital}Frame`,
43
- );
44
-
45
- // 서비스 TypeSource
46
- const { lines, importKeys } = this.getTypeSource(apisForThisModel);
47
-
48
- // AxiosProgressEvent 있는지 확인
49
- const hasAxiosProgressEvent = apis.find((api) =>
50
- (api.options.clients ?? []).includes("axios-multipart"),
51
- );
52
-
53
- return {
54
- ...this.getTargetAndPath(namesRecord),
55
- body: lines.join("\n"),
56
- importKeys: importKeys.filter((key) => ["ListResult"].includes(key) === false),
57
- customHeaders: [
58
- `import { z } from 'zod';`,
59
- `import qs from "qs";`,
60
- `import useSWR, { type SWRResponse } from "swr";`,
61
- `import { fetch, ListResult, SWRError, SwrOptions, handleConditional, swrPostFetcher, EventHandlers, SSEStreamOptions, useSSEStream } from '../sonamu.shared';`,
62
- ...(hasAxiosProgressEvent ? [`import { type AxiosProgressEvent } from 'axios';`] : []),
63
- ],
64
- };
65
- }
66
-
67
- getTypeSource(apis: ExtendedApi[]): {
68
- lines: string[];
69
- importKeys: string[];
70
- } {
71
- const importKeys: string[] = [];
72
-
73
- // 제네릭에서 선언한 타입, importKeys에서 제외 필요
74
- let typeParamNames: string[] = [];
75
-
76
- const groups = group(apis, (api) => api.modelName);
77
- const body = Object.keys(groups)
78
- .map((modelName) => {
79
- const methods = groups[modelName];
80
- assert(methods);
81
- const methodCodes = methods
82
- .map((api) => {
83
- // 컨텍스트 제외된 파라미터 리스트
84
- const paramsWithoutContext = api.parameters.filter(
85
- (param) =>
86
- !ApiParamType.isContext(param.type) &&
87
- !ApiParamType.isRefKnex(param.type) &&
88
- !(param.optional === true && param.name.startsWith("_")), // _로 시작하는 파라미터는 제외
89
- );
90
-
91
- // 파라미터 타입 정의
92
- const typeParametersAsTsType = api.typeParameters
93
- .map((typeParam) => {
94
- return apiParamTypeToTsType(typeParam, importKeys);
95
- })
96
- .join(", ");
97
- const typeParamsDef = typeParametersAsTsType ? `<${typeParametersAsTsType}>` : "";
98
- typeParamNames = typeParamNames.concat(
99
- api.typeParameters.map((typeParam) => typeParam.id),
100
- );
101
-
102
- // 파라미터 정의
103
- const paramsDef = apiParamToTsCode(paramsWithoutContext, importKeys);
104
-
105
- // 파라미터 정의 (객체 형태)
106
- const paramsDefAsObject = apiParamToTsCodeAsObject(paramsWithoutContext, importKeys);
107
-
108
- // 리턴 타입 정의
109
- const returnTypeDef = apiParamTypeToTsType(
110
- assertDefined(unwrapPromiseOnce(api.returnType)),
111
- importKeys,
112
- );
113
-
114
- // 페이로드 데이터 정의
115
- const payloadDef = `{ ${paramsWithoutContext.map((param) => param.name).join(", ")} }`;
116
-
117
- // 기본 URL
118
- const apiBaseUrl = `${Sonamu.config.api.route.prefix}${api.path}`;
119
-
120
- const clients = api.options.clients ?? [];
121
- return [
122
- // 클라이언트별로 생성
123
- ...sort(clients, (client) => (client === "swr" ? 0 : 1)).map((client) => {
124
- switch (client) {
125
- case "axios":
126
- return this.renderAxios(
127
- api,
128
- apiBaseUrl,
129
- typeParamsDef,
130
- paramsDef,
131
- returnTypeDef,
132
- payloadDef,
133
- );
134
- case "axios-multipart":
135
- return this.renderAxiosMultipart(
136
- api,
137
- apiBaseUrl,
138
- typeParamsDef,
139
- paramsDef,
140
- returnTypeDef,
141
- paramsWithoutContext,
142
- );
143
- case "swr":
144
- return this.renderSwr(
145
- api,
146
- apiBaseUrl,
147
- typeParamsDef,
148
- paramsDef,
149
- returnTypeDef,
150
- payloadDef,
151
- );
152
- case "window-fetch":
153
- return this.renderWindowFetch(
154
- api,
155
- apiBaseUrl,
156
- typeParamsDef,
157
- paramsDef,
158
- payloadDef,
159
- );
160
- default:
161
- return `// Not supported ${inflection.camelize(client, true)} yet.`;
162
- }
163
- }),
164
- // 스트리밍인 경우
165
- ...(api.streamOptions ? [this.renderStream(api, apiBaseUrl, paramsDefAsObject)] : []),
166
- ].join("\n");
167
- })
168
- .join("\n\n");
169
-
170
- return `export namespace ${modelName.replace(/Model$/, "Service").replace(/Frame$/, "Service")} {
171
- ${methodCodes}
172
- }`;
173
- })
174
- .join("\n\n");
175
-
176
- return {
177
- lines: [body],
178
- importKeys: diff(unique(importKeys), typeParamNames),
179
- };
180
- }
181
-
182
- renderAxios(
183
- api: ExtendedApi,
184
- apiBaseUrl: string,
185
- typeParamsDef: string,
186
- paramsDef: string,
187
- returnTypeDef: string,
188
- payloadDef: string,
189
- ) {
190
- const methodNameAxios = api.options.resourceName
191
- ? `get${inflection.camelize(api.options.resourceName)}`
192
- : api.methodName;
193
-
194
- if (api.options.httpMethod === "GET") {
195
- return `
196
- export async function ${methodNameAxios}${typeParamsDef}(${paramsDef}): Promise<${returnTypeDef}> {
197
- return fetch({
198
- method: "GET",
199
- url: \`${apiBaseUrl}?\${qs.stringify(${payloadDef})}\`,
200
- ${api.options.timeout ? `signal: AbortSignal.timeout(${api.options.timeout}),` : ""}
201
- });
202
- }
203
- `.trim();
204
- } else {
205
- return `
206
- export async function ${methodNameAxios}${typeParamsDef}(${paramsDef}): Promise<${returnTypeDef}> {
207
- return fetch({
208
- method: '${api.options.httpMethod}',
209
- url: \`${apiBaseUrl}\`,
210
- data: ${payloadDef},
211
- ${api.options.timeout ? `signal: AbortSignal.timeout(${api.options.timeout}),` : ""}
212
- });
213
- }
214
- `.trim();
215
- }
216
- }
217
-
218
- renderAxiosMultipart(
219
- api: ExtendedApi,
220
- apiBaseUrl: string,
221
- typeParamsDef: string,
222
- paramsDef: string,
223
- returnTypeDef: string,
224
- paramsWithoutContext: ApiParam[],
225
- ) {
226
- const isMultiple = api.uploadOptions?.mode === "multiple";
227
- const fileParamName = isMultiple ? "files" : "file";
228
- const fileParamType = isMultiple ? "File[]" : "File";
229
-
230
- const formDataDef = isMultiple
231
- ? [
232
- `${fileParamName}.forEach(f => { formData.append("${fileParamName}", f) } ); `,
233
- ...paramsWithoutContext.map(
234
- (param) => `formData.append('${param.name}', String(${param.name}));`,
235
- ),
236
- ].join("\n")
237
- : [
238
- `formData.append("${fileParamName}", ${fileParamName});`,
239
- ...paramsWithoutContext.map(
240
- (param) => `formData.append('${param.name}', String(${param.name}));`,
241
- ),
242
- ].join("\n");
243
-
244
- const paramsDefComma = paramsDef !== "" ? ", " : "";
245
- return `
246
- export async function ${api.methodName}${typeParamsDef}(
247
- ${paramsDef}${paramsDefComma}
248
- ${fileParamName}: ${fileParamType},
249
- onUploadProgress?: (pe:AxiosProgressEvent) => void
250
- ): Promise<${returnTypeDef}> {
251
- const formData = new FormData();
252
- ${formDataDef}
253
- return fetch({
254
- method: 'POST',
255
- url: \`${apiBaseUrl}\`,
256
- headers: {
257
- "Content-Type": "multipart/form-data",
258
- },
259
- onUploadProgress,
260
- data: formData,
261
- ${api.options.timeout ? `signal: AbortSignal.timeout(${api.options.timeout}),` : ""}
262
- });
263
- }
264
- `.trim();
265
- }
266
-
267
- renderSwr(
268
- api: ExtendedApi,
269
- apiBaseUrl: string,
270
- typeParamsDef: string,
271
- paramsDef: string,
272
- returnTypeDef: string,
273
- payloadDef: string,
274
- ) {
275
- const methodNameSwr = api.options.resourceName
276
- ? `use${inflection.camelize(api.options.resourceName)}`
277
- : `use${inflection.camelize(api.methodName)}`;
278
- return ` export function ${inflection.camelize(methodNameSwr, true)}${typeParamsDef}(${[
279
- paramsDef,
280
- "swrOptions?: SwrOptions",
281
- ]
282
- .filter((p) => p !== "")
283
- .join(",")}, ): SWRResponse<${returnTypeDef}, SWRError> {
284
- return useSWR(handleConditional([
285
- \`${apiBaseUrl}\`,
286
- ${payloadDef},
287
- ], swrOptions?.conditional)${api.options.httpMethod === "POST" ? ", swrPostFetcher" : ""}${
288
- api.options.timeout ? `, { loadingTimeout: ${api.options.timeout} }` : ""
289
- });
290
- }`;
291
- }
292
-
293
- renderWindowFetch(
294
- api: ExtendedApi,
295
- apiBaseUrl: string,
296
- typeParamsDef: string,
297
- paramsDef: string,
298
- payloadDef: string,
299
- ) {
300
- return `
301
- export async function ${api.methodName}${typeParamsDef}(${paramsDef}): Promise<Response> {
302
- return window.fetch(\`${apiBaseUrl}?\${qs.stringify(${payloadDef})}\`${
303
- api.options.timeout ? `, { signal: AbortSignal.timeout(${api.options.timeout}) }` : ""
304
- });
305
- }
306
- `.trim();
307
- }
308
-
309
- renderStream(api: ExtendedApi, apiBaseUrl: string, paramsDefAsObject: string) {
310
- if (!api.streamOptions) {
311
- return "// streamOptions not found";
312
- }
313
-
314
- const methodNameStream = api.options.resourceName
315
- ? `use${inflection.camelize(api.options.resourceName)}`
316
- : `use${inflection.camelize(api.methodName)}`;
317
- const methodNameStreamCamelized = inflection.camelize(methodNameStream, true);
318
-
319
- const eventsTypeDef = zodTypeToTsTypeDef(api.streamOptions.events);
320
-
321
- return ` export function ${methodNameStreamCamelized}(
322
- params: ${paramsDefAsObject},
323
- handlers: EventHandlers<${eventsTypeDef} & { end?: () => void }>,
324
- options: SSEStreamOptions) {
325
- return useSSEStream<${eventsTypeDef}>(\`${apiBaseUrl}\`, params, handlers, options);
326
- }`;
327
- }
328
- }