sonamu 0.7.17 → 0.7.18

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.
@@ -1 +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"}
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;AAGvC,qBAAa,kBAAmB,SAAQ,QAAQ;;IAK9C,gBAAgB;;;;IAOhB,MAAM,CAAC,EAAE,EAAE,eAAe,CAAC,UAAU,CAAC;;;;;;;CA6OvC"}
@@ -5,6 +5,7 @@ import { Sonamu } from "../../api/sonamu.js";
5
5
  import { ApiParamType } from "../../types/types.js";
6
6
  import { assertDefined } from "../../utils/utils.js";
7
7
  import { Template } from "../template.js";
8
+ import { zodTypeToTsTypeDef } from "../zod-converter.js";
8
9
  export class Template__services extends Template {
9
10
  constructor(){
10
11
  super("services");
@@ -32,6 +33,26 @@ export class Template__services extends Template {
32
33
  for (const [modelName, modelApis] of apisByModel){
33
34
  const functions = [];
34
35
  for (const api of modelApis){
36
+ // @stream 데코레이터가 있으면 SSE 스트림 함수 생성
37
+ if (api.streamOptions) {
38
+ const paramsWithoutContext = api.parameters.filter((param)=>!ApiParamType.isContext(param.type) && !ApiParamType.isRefKnex(param.type) && !(param.optional === true && param.name.startsWith("_")));
39
+ const paramsDef = apiParamToTsCode(paramsWithoutContext, importKeys);
40
+ const apiBaseUrl = `${Sonamu.config.api.route.prefix}${api.path}`;
41
+ const methodNameStream = api.options.resourceName ? `use${inflection.camelize(api.options.resourceName)}` : `use${inflection.camelize(api.methodName)}`;
42
+ const methodNameStreamCamelized = inflection.camelize(methodNameStream, true);
43
+ const eventsTypeDef = zodTypeToTsTypeDef(api.streamOptions.events);
44
+ const paramsDefAsObject = paramsWithoutContext.length > 0 ? `{ ${paramsWithoutContext.map((p)=>p.name).join(", ")} }` : "{}";
45
+ functions.push(`
46
+ export function ${methodNameStreamCamelized}(
47
+ params: ${paramsDef ? `{ ${paramsWithoutContext.map((p)=>`${p.name}: ${apiParamTypeToTsType(p.type, importKeys)}`).join(", ")} }` : "{}"},
48
+ handlers: EventHandlers<${eventsTypeDef} & { end?: () => void }>,
49
+ options: SSEStreamOptions
50
+ ) {
51
+ return useSSEStream<${eventsTypeDef}>(\`${apiBaseUrl}\`, ${paramsDefAsObject}, handlers, options);
52
+ }
53
+ `.trim());
54
+ continue;
55
+ }
35
56
  // Context 제외한 파라미터
36
57
  const paramsWithoutContext = api.parameters.filter((param)=>!ApiParamType.isContext(param.type) && !ApiParamType.isRefKnex(param.type) && !(param.optional === true && param.name.startsWith("_")));
37
58
  // 타입 파라미터 정의
@@ -150,10 +171,10 @@ ${functions.join("\n\n")}
150
171
  `import { queryOptions, useQuery, useMutation } from '@tanstack/react-query';`,
151
172
  `import type { AxiosProgressEvent } from 'axios';`,
152
173
  `import qs from 'qs';`,
153
- `import { type ListResult, fetch } from './sonamu.shared';`
174
+ `import { type ListResult, fetch, type EventHandlers, type SSEStreamOptions, useSSEStream } from './sonamu.shared';`
154
175
  ]
155
176
  };
156
177
  }
157
178
  }
158
179
 
159
- //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../src/template/implementations/services.template.ts"],"sourcesContent":["import inflection from \"inflection\";\nimport { diff, unique } from \"radashi\";\nimport {\n  apiParamToTsCode,\n  apiParamTypeToTsType,\n  unwrapPromiseOnce,\n} from \"../../api/code-converters\";\nimport type { ExtendedApi } from \"../../api/decorators\";\nimport { Sonamu } from \"../../api/sonamu\";\nimport type { TemplateOptions } from \"../../types/types\";\nimport { ApiParamType } from \"../../types/types\";\nimport { assertDefined } from \"../../utils/utils\";\nimport { Template } from \"../template\";\n\nexport class Template__services extends Template {\n  constructor() {\n    super(\"services\");\n  }\n\n  getTargetAndPath() {\n    return {\n      target: \":target/src/services\",\n      path: `services.generated.ts`,\n    };\n  }\n\n  render({}: TemplateOptions[\"services\"]) {\n    const { apis } = Sonamu.syncer;\n\n    // 모델별로 그룹화\n    const apisByModel = new Map<string, ExtendedApi[]>();\n    for (const api of apis) {\n      const modelName = api.modelName.replace(/Model$/, \"\").replace(/Frame$/, \"\");\n      if (!apisByModel.has(modelName)) {\n        apisByModel.set(modelName, []);\n      }\n      apisByModel.get(modelName)?.push(api);\n    }\n\n    const importKeys: string[] = [];\n    const namespaces: string[] = [];\n    let typeParamNames: string[] = [];\n\n    for (const [modelName, modelApis] of apisByModel) {\n      const functions: string[] = [];\n\n      for (const api of modelApis) {\n        // Context 제외한 파라미터\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) => apiParamTypeToTsType(typeParam, importKeys))\n          .join(\", \");\n        const typeParamsDef = typeParametersAsTsType ? `<${typeParametersAsTsType}>` : \"\";\n        typeParamNames = typeParamNames.concat(api.typeParameters.map((tp) => tp.id));\n\n        // 파라미터 정의\n        const paramsDef = apiParamToTsCode(paramsWithoutContext, importKeys);\n        const paramNames = paramsWithoutContext.map((p) => p.name).join(\", \");\n\n        // 리턴 타입 정의\n        const returnTypeDef = apiParamTypeToTsType(\n          assertDefined(unwrapPromiseOnce(api.returnType)),\n          importKeys,\n        );\n\n        // 기본 URL\n        const apiBaseUrl = `${Sonamu.config.api.route.prefix}${api.path}`;\n\n        const clients = api.options.clients || [];\n\n        // 1. axios 함수 생성\n        // resourceName이 있으면 get + resourceName 형태로 함수명 생성\n        const methodName = api.options.resourceName\n          ? `get${inflection.camelize(api.options.resourceName)}`\n          : api.methodName;\n\n        // axios-multipart 처리 (파일 업로드)\n        if (clients.includes(\"axios-multipart\")) {\n          const isMultiple = api.uploadOptions?.mode === \"multiple\";\n          const fileParamName = isMultiple ? \"files\" : \"file\";\n          const fileParamType = isMultiple ? \"File[]\" : \"File\";\n\n          const formDataAppend = isMultiple\n            ? `${fileParamName}.forEach(f => { formData.append(\"${fileParamName}\", f); });`\n            : `formData.append(\"${fileParamName}\", ${fileParamName});`;\n\n          const otherParamsAppend = paramsWithoutContext\n            .map((param) => `formData.append('${param.name}', String(${param.name}));`)\n            .join(\"\\n    \");\n\n          const paramsDefComma = paramsDef !== \"\" ? \", \" : \"\";\n          functions.push(\n            `\nexport async function ${methodName}${typeParamsDef}(\n  ${paramsDef}${paramsDefComma}\n  ${fileParamName}: ${fileParamType},\n  onUploadProgress?: (pe: AxiosProgressEvent) => void\n): Promise<${returnTypeDef}> {\n  const formData = new FormData();\n  ${formDataAppend}\n  ${otherParamsAppend}\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        } else if (api.options.httpMethod === \"GET\") {\n          const hasParams = paramsWithoutContext.length > 0;\n          functions.push(\n            `\nexport async function ${methodName}${typeParamsDef}(${paramsDef}): Promise<${returnTypeDef}> {\n  return fetch({\n    method: \"GET\",\n    url: \\`${apiBaseUrl}${hasParams ? `?\\${qs.stringify({ ${paramNames} })}` : \"\"}\\`,\n    ${api.options.timeout ? `signal: AbortSignal.timeout(${api.options.timeout}),` : \"\"}\n  });\n}\n          `.trim(),\n          );\n        } else {\n          const hasParams = paramsWithoutContext.length > 0;\n          functions.push(\n            `\nexport async function ${methodName}${typeParamsDef}(${paramsDef}): Promise<${returnTypeDef}> {\n  return fetch({\n    method: \"${api.options.httpMethod}\",\n    url: \\`${apiBaseUrl}\\`,\n    ${hasParams ? `data: { ${paramNames} },` : \"\"}\n    ${api.options.timeout ? `signal: AbortSignal.timeout(${api.options.timeout}),` : \"\"}\n  });\n}\n          `.trim(),\n          );\n        }\n\n        // 2. queryOptions + useQuery (tanstack-query)\n        if (clients.includes(\"tanstack-query\")) {\n          const hookName = api.options.resourceName\n            ? inflection.camelize(api.options.resourceName, true)\n            : inflection.camelize(api.methodName, true);\n\n          // queryOptions\n          functions.push(\n            `\nexport const ${methodName}QueryOptions = ${typeParamsDef}(${paramsDef}) => queryOptions({\n  queryKey: ['${modelName}', '${methodName}'${paramNames ? `, ${paramNames}` : \"\"}],\n  queryFn: () => ${methodName}(${paramNames})\n});\n          `.trim(),\n          );\n\n          // useQuery hook\n          functions.push(\n            `\nexport const use${inflection.camelize(hookName)} = ${typeParamsDef}(${paramsDef}${\n              paramsDef ? \", \" : \"\"\n            }options?: { enabled?: boolean }) =>\n  useQuery({\n    ...${methodName}QueryOptions(${paramNames}),\n    ...options\n  });\n          `.trim(),\n          );\n        }\n\n        // 3. useMutation (tanstack-mutation)\n        if (clients.includes(\"tanstack-mutation\")) {\n          const hookName = inflection.camelize(api.methodName);\n          const mutationParamType =\n            paramsWithoutContext.length > 0\n              ? `{ ${paramsWithoutContext\n                  .map((p) => `${p.name}: ${apiParamTypeToTsType(p.type, [])}`)\n                  .join(\", \")} }`\n              : \"void\";\n          const mutationParamNames =\n            paramsWithoutContext.length > 0\n              ? paramsWithoutContext.map((p) => `params.${p.name}`).join(\", \")\n              : \"\";\n\n          functions.push(\n            `\nexport const use${hookName}Mutation = ${typeParamsDef}() => useMutation({\n  mutationFn: (params: ${mutationParamType}) => ${methodName}(${mutationParamNames})\n});\n          `.trim(),\n          );\n        }\n      }\n\n      namespaces.push(\n        `\nexport namespace ${modelName}Service {\n${functions.join(\"\\n\\n\")}\n}\n      `.trim(),\n      );\n    }\n\n    return {\n      ...this.getTargetAndPath(),\n      body: namespaces.join(\"\\n\\n\"),\n      importKeys: diff(unique(importKeys), [...typeParamNames, \"ListResult\"]),\n      customHeaders: [\n        `import { queryOptions, useQuery, useMutation } from '@tanstack/react-query';`,\n        `import type { AxiosProgressEvent } from 'axios';`,\n        `import qs from 'qs';`,\n        `import { type ListResult, fetch } from './sonamu.shared';`,\n      ],\n    };\n  }\n}\n"],"names":["inflection","diff","unique","apiParamToTsCode","apiParamTypeToTsType","unwrapPromiseOnce","Sonamu","ApiParamType","assertDefined","Template","Template__services","getTargetAndPath","target","path","render","apis","syncer","apisByModel","Map","api","modelName","replace","has","set","get","push","importKeys","namespaces","typeParamNames","modelApis","functions","paramsWithoutContext","parameters","filter","param","isContext","type","isRefKnex","optional","name","startsWith","typeParametersAsTsType","typeParameters","map","typeParam","join","typeParamsDef","concat","tp","id","paramsDef","paramNames","p","returnTypeDef","returnType","apiBaseUrl","config","route","prefix","clients","options","methodName","resourceName","camelize","includes","isMultiple","uploadOptions","mode","fileParamName","fileParamType","formDataAppend","otherParamsAppend","paramsDefComma","timeout","trim","httpMethod","hasParams","length","hookName","mutationParamType","mutationParamNames","body","customHeaders"],"mappings":"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"}
180
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../../../src/template/implementations/services.template.ts"],"sourcesContent":["import inflection from \"inflection\";\nimport { diff, unique } from \"radashi\";\nimport {\n  apiParamToTsCode,\n  apiParamTypeToTsType,\n  unwrapPromiseOnce,\n} from \"../../api/code-converters\";\nimport type { ExtendedApi } from \"../../api/decorators\";\nimport { Sonamu } from \"../../api/sonamu\";\nimport type { TemplateOptions } from \"../../types/types\";\nimport { ApiParamType } from \"../../types/types\";\nimport { assertDefined } from \"../../utils/utils\";\nimport { Template } from \"../template\";\nimport { zodTypeToTsTypeDef } from \"../zod-converter\";\n\nexport class Template__services extends Template {\n  constructor() {\n    super(\"services\");\n  }\n\n  getTargetAndPath() {\n    return {\n      target: \":target/src/services\",\n      path: `services.generated.ts`,\n    };\n  }\n\n  render({}: TemplateOptions[\"services\"]) {\n    const { apis } = Sonamu.syncer;\n\n    // 모델별로 그룹화\n    const apisByModel = new Map<string, ExtendedApi[]>();\n    for (const api of apis) {\n      const modelName = api.modelName.replace(/Model$/, \"\").replace(/Frame$/, \"\");\n      if (!apisByModel.has(modelName)) {\n        apisByModel.set(modelName, []);\n      }\n      apisByModel.get(modelName)?.push(api);\n    }\n\n    const importKeys: string[] = [];\n    const namespaces: string[] = [];\n    let typeParamNames: string[] = [];\n\n    for (const [modelName, modelApis] of apisByModel) {\n      const functions: string[] = [];\n\n      for (const api of modelApis) {\n        // @stream 데코레이터가 있으면 SSE 스트림 함수 생성\n        if (api.streamOptions) {\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          const paramsDef = apiParamToTsCode(paramsWithoutContext, importKeys);\n          const apiBaseUrl = `${Sonamu.config.api.route.prefix}${api.path}`;\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          const paramsDefAsObject =\n            paramsWithoutContext.length > 0\n              ? `{ ${paramsWithoutContext.map((p) => p.name).join(\", \")} }`\n              : \"{}\";\n\n          functions.push(\n            `\nexport function ${methodNameStreamCamelized}(\n  params: ${paramsDef ? `{ ${paramsWithoutContext.map((p) => `${p.name}: ${apiParamTypeToTsType(p.type, importKeys)}`).join(\", \")} }` : \"{}\"},\n  handlers: EventHandlers<${eventsTypeDef} & { end?: () => void }>,\n  options: SSEStreamOptions\n) {\n  return useSSEStream<${eventsTypeDef}>(\\`${apiBaseUrl}\\`, ${paramsDefAsObject}, handlers, options);\n}\n            `.trim(),\n          );\n          continue;\n        }\n\n        // Context 제외한 파라미터\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) => apiParamTypeToTsType(typeParam, importKeys))\n          .join(\", \");\n        const typeParamsDef = typeParametersAsTsType ? `<${typeParametersAsTsType}>` : \"\";\n        typeParamNames = typeParamNames.concat(api.typeParameters.map((tp) => tp.id));\n\n        // 파라미터 정의\n        const paramsDef = apiParamToTsCode(paramsWithoutContext, importKeys);\n        const paramNames = paramsWithoutContext.map((p) => p.name).join(\", \");\n\n        // 리턴 타입 정의\n        const returnTypeDef = apiParamTypeToTsType(\n          assertDefined(unwrapPromiseOnce(api.returnType)),\n          importKeys,\n        );\n\n        // 기본 URL\n        const apiBaseUrl = `${Sonamu.config.api.route.prefix}${api.path}`;\n\n        const clients = api.options.clients || [];\n\n        // 1. axios 함수 생성\n        // resourceName이 있으면 get + resourceName 형태로 함수명 생성\n        const methodName = api.options.resourceName\n          ? `get${inflection.camelize(api.options.resourceName)}`\n          : api.methodName;\n\n        // axios-multipart 처리 (파일 업로드)\n        if (clients.includes(\"axios-multipart\")) {\n          const isMultiple = api.uploadOptions?.mode === \"multiple\";\n          const fileParamName = isMultiple ? \"files\" : \"file\";\n          const fileParamType = isMultiple ? \"File[]\" : \"File\";\n\n          const formDataAppend = isMultiple\n            ? `${fileParamName}.forEach(f => { formData.append(\"${fileParamName}\", f); });`\n            : `formData.append(\"${fileParamName}\", ${fileParamName});`;\n\n          const otherParamsAppend = paramsWithoutContext\n            .map((param) => `formData.append('${param.name}', String(${param.name}));`)\n            .join(\"\\n    \");\n\n          const paramsDefComma = paramsDef !== \"\" ? \", \" : \"\";\n          functions.push(\n            `\nexport async function ${methodName}${typeParamsDef}(\n  ${paramsDef}${paramsDefComma}\n  ${fileParamName}: ${fileParamType},\n  onUploadProgress?: (pe: AxiosProgressEvent) => void\n): Promise<${returnTypeDef}> {\n  const formData = new FormData();\n  ${formDataAppend}\n  ${otherParamsAppend}\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        } else if (api.options.httpMethod === \"GET\") {\n          const hasParams = paramsWithoutContext.length > 0;\n          functions.push(\n            `\nexport async function ${methodName}${typeParamsDef}(${paramsDef}): Promise<${returnTypeDef}> {\n  return fetch({\n    method: \"GET\",\n    url: \\`${apiBaseUrl}${hasParams ? `?\\${qs.stringify({ ${paramNames} })}` : \"\"}\\`,\n    ${api.options.timeout ? `signal: AbortSignal.timeout(${api.options.timeout}),` : \"\"}\n  });\n}\n          `.trim(),\n          );\n        } else {\n          const hasParams = paramsWithoutContext.length > 0;\n          functions.push(\n            `\nexport async function ${methodName}${typeParamsDef}(${paramsDef}): Promise<${returnTypeDef}> {\n  return fetch({\n    method: \"${api.options.httpMethod}\",\n    url: \\`${apiBaseUrl}\\`,\n    ${hasParams ? `data: { ${paramNames} },` : \"\"}\n    ${api.options.timeout ? `signal: AbortSignal.timeout(${api.options.timeout}),` : \"\"}\n  });\n}\n          `.trim(),\n          );\n        }\n\n        // 2. queryOptions + useQuery (tanstack-query)\n        if (clients.includes(\"tanstack-query\")) {\n          const hookName = api.options.resourceName\n            ? inflection.camelize(api.options.resourceName, true)\n            : inflection.camelize(api.methodName, true);\n\n          // queryOptions\n          functions.push(\n            `\nexport const ${methodName}QueryOptions = ${typeParamsDef}(${paramsDef}) => queryOptions({\n  queryKey: ['${modelName}', '${methodName}'${paramNames ? `, ${paramNames}` : \"\"}],\n  queryFn: () => ${methodName}(${paramNames})\n});\n          `.trim(),\n          );\n\n          // useQuery hook\n          functions.push(\n            `\nexport const use${inflection.camelize(hookName)} = ${typeParamsDef}(${paramsDef}${\n              paramsDef ? \", \" : \"\"\n            }options?: { enabled?: boolean }) =>\n  useQuery({\n    ...${methodName}QueryOptions(${paramNames}),\n    ...options\n  });\n          `.trim(),\n          );\n        }\n\n        // 3. useMutation (tanstack-mutation)\n        if (clients.includes(\"tanstack-mutation\")) {\n          const hookName = inflection.camelize(api.methodName);\n          const mutationParamType =\n            paramsWithoutContext.length > 0\n              ? `{ ${paramsWithoutContext\n                  .map((p) => `${p.name}: ${apiParamTypeToTsType(p.type, [])}`)\n                  .join(\", \")} }`\n              : \"void\";\n          const mutationParamNames =\n            paramsWithoutContext.length > 0\n              ? paramsWithoutContext.map((p) => `params.${p.name}`).join(\", \")\n              : \"\";\n\n          functions.push(\n            `\nexport const use${hookName}Mutation = ${typeParamsDef}() => useMutation({\n  mutationFn: (params: ${mutationParamType}) => ${methodName}(${mutationParamNames})\n});\n          `.trim(),\n          );\n        }\n      }\n\n      namespaces.push(\n        `\nexport namespace ${modelName}Service {\n${functions.join(\"\\n\\n\")}\n}\n      `.trim(),\n      );\n    }\n\n    return {\n      ...this.getTargetAndPath(),\n      body: namespaces.join(\"\\n\\n\"),\n      importKeys: diff(unique(importKeys), [...typeParamNames, \"ListResult\"]),\n      customHeaders: [\n        `import { queryOptions, useQuery, useMutation } from '@tanstack/react-query';`,\n        `import type { AxiosProgressEvent } from 'axios';`,\n        `import qs from 'qs';`,\n        `import { type ListResult, fetch, type EventHandlers, type SSEStreamOptions, useSSEStream } from './sonamu.shared';`,\n      ],\n    };\n  }\n}\n"],"names":["inflection","diff","unique","apiParamToTsCode","apiParamTypeToTsType","unwrapPromiseOnce","Sonamu","ApiParamType","assertDefined","Template","zodTypeToTsTypeDef","Template__services","getTargetAndPath","target","path","render","apis","syncer","apisByModel","Map","api","modelName","replace","has","set","get","push","importKeys","namespaces","typeParamNames","modelApis","functions","streamOptions","paramsWithoutContext","parameters","filter","param","isContext","type","isRefKnex","optional","name","startsWith","paramsDef","apiBaseUrl","config","route","prefix","methodNameStream","options","resourceName","camelize","methodName","methodNameStreamCamelized","eventsTypeDef","events","paramsDefAsObject","length","map","p","join","trim","typeParametersAsTsType","typeParameters","typeParam","typeParamsDef","concat","tp","id","paramNames","returnTypeDef","returnType","clients","includes","isMultiple","uploadOptions","mode","fileParamName","fileParamType","formDataAppend","otherParamsAppend","paramsDefComma","timeout","httpMethod","hasParams","hookName","mutationParamType","mutationParamNames","body","customHeaders"],"mappings":"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"}
@@ -68,7 +68,7 @@ In order to be iterable, non-array objects must have a [Symbol.iterator]() metho
68
68
  `;break}case-3:{i=`\r
69
69
  `;break}case-2:{i=t?" ":" ";break}case-1:{if(!t&&o)continue;i=" ";break}default:i=String.fromCharCode(a)}o=a===-2,r.push(i)}return r.join("")}function xqe(e){const r={constructs:NHe([Sqe,...(e||{}).extensions||[]]),content:o(jHe),defined:[],document:o(BHe),flow:o(iqe),lazy:{},string:o(uqe),text:o(cqe)};return r;function o(a){return i;function i(s){return Eqe(r,a,s)}}}function Tqe(e){for(;!ane(e););return e}const t6=/[\0\t\n\r]/g;function Aqe(){let e=1,t="",n=!0,r;return o;function o(a,i,s){const l=[];let u,d,p,h,m;for(a=t+(typeof a=="string"?a.toString():new TextDecoder(i||void 0).decode(a)),p=0,t="",n&&(a.charCodeAt(0)===65279&&p++,n=void 0);p<a.length;){if(t6.lastIndex=p,u=t6.exec(a),h=u&&u.index!==void 0?u.index:a.length,m=a.charCodeAt(h),!u){t=a.slice(p);break}if(m===10&&p===h&&r)l.push(-3),r=void 0;else switch(r&&(l.push(-5),r=void 0),p<h&&(l.push(a.slice(p,h)),e+=h-p),m){case 0:{l.push(65533),e++;break}case 9:{for(d=Math.ceil(e/4)*4,l.push(-2);e++<d;)l.push(-1);break}case 10:{l.push(-4),e=1;break}default:r=!0,e=1}p=h+1}return s&&(r&&l.push(-5),t&&l.push(t),l.push(null)),l}}const Cqe=/\\([!-/:-@[-`{-~])|&(#(?:\d{1,7}|x[\da-f]{1,6})|[\da-z]{1,31});/gi;function Oqe(e){return e.replace(Cqe,Iqe)}function Iqe(e,t,n){if(t)return t;if(n.charCodeAt(0)===35){const o=n.charCodeAt(1),a=o===120||o===88;return tne(n.slice(a?2:1),a?16:10)}return A$(n)||e}const dne={}.hasOwnProperty;function Rqe(e,t,n){return typeof t!="string"&&(n=t,t=void 0),Nqe(n)(Tqe(xqe(n).document().write(Aqe()(e,t,!0))))}function Nqe(e){const t={transforms:[],canContainEols:["emphasis","fragment","heading","paragraph","strong"],enter:{autolink:a(Bn),autolinkProtocol:R,autolinkEmail:R,atxHeading:a(ft),blockQuote:a(pe),characterEscape:R,characterReference:R,codeFenced:a(be),codeFencedFenceInfo:i,codeFencedFenceMeta:i,codeIndented:a(be,i),codeText:a(xe,i),codeTextData:R,data:R,codeFlowValue:R,definition:a(Ie),definitionDestinationString:i,definitionLabelString:i,definitionTitleString:i,emphasis:a(rt),hardBreakEscape:a(nt),hardBreakTrailing:a(nt),htmlFlow:a(Nt,i),htmlFlowData:R,htmlText:a(Nt,i),htmlTextData:R,image:a(an),label:i,link:a(Bn),listItem:a(yn),listItemValue:h,listOrdered:a(Mt,p),listUnordered:a(Mt),paragraph:a(xn),reference:P,referenceString:i,resourceDestinationString:i,resourceTitleString:i,setextHeading:a(ft),strong:a(sr),thematicBreak:a(Gt)},exit:{atxHeading:l(),atxHeadingSequence:C,autolink:l(),autolinkEmail:ae,autolinkProtocol:te,blockQuote:l(),characterEscapeValue:$,characterReferenceMarkerHexadecimal:L,characterReferenceMarkerNumeric:L,characterReferenceValue:X,characterReference:ne,codeFenced:l(v),codeFencedFence:y,codeFencedFenceInfo:m,codeFencedFenceMeta:b,codeFlowValue:$,codeIndented:l(k),codeText:l(H),codeTextData:$,data:$,definition:l(),definitionDestinationString:T,definitionLabelString:_,definitionTitleString:S,emphasis:l(),hardBreakEscape:l(M),hardBreakTrailing:l(M),htmlFlow:l(V),htmlFlowData:$,htmlText:l(z),htmlTextData:$,image:l(G),label:Q,labelText:q,lineEnding:N,link:l(W),listItem:l(),listOrdered:l(),listUnordered:l(),paragraph:l(),referenceString:Z,resourceDestinationString:F,resourceTitleString:U,resource:K,setextHeading:l(O),setextHeadingLineSequence:A,setextHeadingText:w,strong:l(),thematicBreak:l()}};fne(t,(e||{}).mdastExtensions||[]);const n={};return r;function r(he){let ye={type:"root",children:[]};const Pe={stack:[ye],tokenStack:[],config:t,enter:s,exit:u,buffer:i,resume:d,data:n},Ke=[];let at=-1;for(;++at<he.length;)if(he[at][1].type==="listOrdered"||he[at][1].type==="listUnordered")if(he[at][0]==="enter")Ke.push(at);else{const fn=Ke.pop();at=o(he,fn,at)}for(at=-1;++at<he.length;){const fn=t[he[at][0]];dne.call(fn,he[at][1].type)&&fn[he[at][1].type].call(Object.assign({sliceSerialize:he[at][2].sliceSerialize},Pe),he[at][1])}if(Pe.tokenStack.length>0){const fn=Pe.tokenStack[Pe.tokenStack.length-1];(fn[1]||n6).call(Pe,void 0,fn[0])}for(ye.position={start:is(he.length>0?he[0][1].start:{line:1,column:1,offset:0}),end:is(he.length>0?he[he.length-2][1].end:{line:1,column:1,offset:0})},at=-1;++at<t.transforms.length;)ye=t.transforms[at](ye)||ye;return ye}function o(he,ye,Pe){let Ke=ye-1,at=-1,fn=!1,Hn,$n,Xn,qn;for(;++Ke<=Pe;){const vn=he[Ke];switch(vn[1].type){case"listUnordered":case"listOrdered":case"blockQuote":{vn[0]==="enter"?at++:at--,qn=void 0;break}case"lineEndingBlank":{vn[0]==="enter"&&(Hn&&!qn&&!at&&!Xn&&(Xn=Ke),qn=void 0);break}case"linePrefix":case"listItemValue":case"listItemMarker":case"listItemPrefix":case"listItemPrefixWhitespace":break;default:qn=void 0}if(!at&&vn[0]==="enter"&&vn[1].type==="listItemPrefix"||at===-1&&vn[0]==="exit"&&(vn[1].type==="listUnordered"||vn[1].type==="listOrdered")){if(Hn){let Qn=Ke;for($n=void 0;Qn--;){const Mn=he[Qn];if(Mn[1].type==="lineEnding"||Mn[1].type==="lineEndingBlank"){if(Mn[0]==="exit")continue;$n&&(he[$n][1].type="lineEndingBlank",fn=!0),Mn[1].type="lineEnding",$n=Qn}else if(!(Mn[1].type==="linePrefix"||Mn[1].type==="blockQuotePrefix"||Mn[1].type==="blockQuotePrefixWhitespace"||Mn[1].type==="blockQuoteMarker"||Mn[1].type==="listItemIndent"))break}Xn&&(!$n||Xn<$n)&&(Hn._spread=!0),Hn.end=Object.assign({},$n?he[$n][1].start:vn[1].end),he.splice($n||Ke,0,["exit",Hn,vn[2]]),Ke++,Pe++}if(vn[1].type==="listItemPrefix"){const Qn={type:"listItem",_spread:!1,start:Object.assign({},vn[1].start),end:void 0};Hn=Qn,he.splice(Ke,0,["enter",Qn,vn[2]]),Ke++,Pe++,Xn=void 0,qn=!0}}}return he[ye][1]._spread=fn,Pe}function a(he,ye){return Pe;function Pe(Ke){s.call(this,he(Ke),Ke),ye&&ye.call(this,Ke)}}function i(){this.stack.push({type:"fragment",children:[]})}function s(he,ye,Pe){this.stack[this.stack.length-1].children.push(he),this.stack.push(he),this.tokenStack.push([ye,Pe||void 0]),he.position={start:is(ye.start),end:void 0}}function l(he){return ye;function ye(Pe){he&&he.call(this,Pe),u.call(this,Pe)}}function u(he,ye){const Pe=this.stack.pop(),Ke=this.tokenStack.pop();if(Ke)Ke[0].type!==he.type&&(ye?ye.call(this,he,Ke[0]):(Ke[1]||n6).call(this,he,Ke[0]));else throw new Error("Cannot close `"+he.type+"` ("+fp({start:he.start,end:he.end})+"): it’s not open");Pe.position.end=is(he.end)}function d(){return IHe(this.stack.pop())}function p(){this.data.expectingFirstListItemValue=!0}function h(he){if(this.data.expectingFirstListItemValue){const ye=this.stack[this.stack.length-2];ye.start=Number.parseInt(this.sliceSerialize(he),10),this.data.expectingFirstListItemValue=void 0}}function m(){const he=this.resume(),ye=this.stack[this.stack.length-1];ye.lang=he}function b(){const he=this.resume(),ye=this.stack[this.stack.length-1];ye.meta=he}function y(){this.data.flowCodeInside||(this.buffer(),this.data.flowCodeInside=!0)}function v(){const he=this.resume(),ye=this.stack[this.stack.length-1];ye.value=he.replace(/^(\r?\n|\r)|(\r?\n|\r)$/g,""),this.data.flowCodeInside=void 0}function k(){const he=this.resume(),ye=this.stack[this.stack.length-1];ye.value=he.replace(/(\r?\n|\r)$/g,"")}function _(he){const ye=this.resume(),Pe=this.stack[this.stack.length-1];Pe.label=ye,Pe.identifier=hc(this.sliceSerialize(he)).toLowerCase()}function S(){const he=this.resume(),ye=this.stack[this.stack.length-1];ye.title=he}function T(){const he=this.resume(),ye=this.stack[this.stack.length-1];ye.url=he}function C(he){const ye=this.stack[this.stack.length-1];if(!ye.depth){const Pe=this.sliceSerialize(he).length;ye.depth=Pe}}function w(){this.data.setextHeadingSlurpLineEnding=!0}function A(he){const ye=this.stack[this.stack.length-1];ye.depth=this.sliceSerialize(he).codePointAt(0)===61?1:2}function O(){this.data.setextHeadingSlurpLineEnding=void 0}function R(he){const Pe=this.stack[this.stack.length-1].children;let Ke=Pe[Pe.length-1];(!Ke||Ke.type!=="text")&&(Ke=Pn(),Ke.position={start:is(he.start),end:void 0},Pe.push(Ke)),this.stack.push(Ke)}function $(he){const ye=this.stack.pop();ye.value+=this.sliceSerialize(he),ye.position.end=is(he.end)}function N(he){const ye=this.stack[this.stack.length-1];if(this.data.atHardBreak){const Pe=ye.children[ye.children.length-1];Pe.position.end=is(he.end),this.data.atHardBreak=void 0;return}!this.data.setextHeadingSlurpLineEnding&&t.canContainEols.includes(ye.type)&&(R.call(this,he),$.call(this,he))}function M(){this.data.atHardBreak=!0}function V(){const he=this.resume(),ye=this.stack[this.stack.length-1];ye.value=he}function z(){const he=this.resume(),ye=this.stack[this.stack.length-1];ye.value=he}function H(){const he=this.resume(),ye=this.stack[this.stack.length-1];ye.value=he}function W(){const he=this.stack[this.stack.length-1];if(this.data.inReference){const ye=this.data.referenceType||"shortcut";he.type+="Reference",he.referenceType=ye,delete he.url,delete he.title}else delete he.identifier,delete he.label;this.data.referenceType=void 0}function G(){const he=this.stack[this.stack.length-1];if(this.data.inReference){const ye=this.data.referenceType||"shortcut";he.type+="Reference",he.referenceType=ye,delete he.url,delete he.title}else delete he.identifier,delete he.label;this.data.referenceType=void 0}function q(he){const ye=this.sliceSerialize(he),Pe=this.stack[this.stack.length-2];Pe.label=Oqe(ye),Pe.identifier=hc(ye).toLowerCase()}function Q(){const he=this.stack[this.stack.length-1],ye=this.resume(),Pe=this.stack[this.stack.length-1];if(this.data.inReference=!0,Pe.type==="link"){const Ke=he.children;Pe.children=Ke}else Pe.alt=ye}function F(){const he=this.resume(),ye=this.stack[this.stack.length-1];ye.url=he}function U(){const he=this.resume(),ye=this.stack[this.stack.length-1];ye.title=he}function K(){this.data.inReference=void 0}function P(){this.data.referenceType="collapsed"}function Z(he){const ye=this.resume(),Pe=this.stack[this.stack.length-1];Pe.label=ye,Pe.identifier=hc(this.sliceSerialize(he)).toLowerCase(),this.data.referenceType="full"}function L(he){this.data.characterReferenceType=he.type}function X(he){const ye=this.sliceSerialize(he),Pe=this.data.characterReferenceType;let Ke;Pe?(Ke=tne(ye,Pe==="characterReferenceMarkerNumeric"?10:16),this.data.characterReferenceType=void 0):Ke=A$(ye);const at=this.stack[this.stack.length-1];at.value+=Ke}function ne(he){const ye=this.stack.pop();ye.position.end=is(he.end)}function te(he){$.call(this,he);const ye=this.stack[this.stack.length-1];ye.url=this.sliceSerialize(he)}function ae(he){$.call(this,he);const ye=this.stack[this.stack.length-1];ye.url="mailto:"+this.sliceSerialize(he)}function pe(){return{type:"blockquote",children:[]}}function be(){return{type:"code",lang:null,meta:null,value:""}}function xe(){return{type:"inlineCode",value:""}}function Ie(){return{type:"definition",identifier:"",label:null,title:null,url:""}}function rt(){return{type:"emphasis",children:[]}}function ft(){return{type:"heading",depth:0,children:[]}}function nt(){return{type:"break"}}function Nt(){return{type:"html",value:""}}function an(){return{type:"image",title:null,url:"",alt:null}}function Bn(){return{type:"link",title:null,url:"",children:[]}}function Mt(he){return{type:"list",ordered:he.type==="listOrdered",start:null,spread:he._spread,children:[]}}function yn(he){return{type:"listItem",spread:he._spread,checked:null,children:[]}}function xn(){return{type:"paragraph",children:[]}}function sr(){return{type:"strong",children:[]}}function Pn(){return{type:"text",value:""}}function Gt(){return{type:"thematicBreak"}}}function is(e){return{line:e.line,column:e.column,offset:e.offset}}function fne(e,t){let n=-1;for(;++n<t.length;){const r=t[n];Array.isArray(r)?fne(e,r):Dqe(e,r)}}function Dqe(e,t){let n;for(n in t)if(dne.call(t,n))switch(n){case"canContainEols":{const r=t[n];r&&e[n].push(...r);break}case"transforms":{const r=t[n];r&&e[n].push(...r);break}case"enter":case"exit":{const r=t[n];r&&Object.assign(e[n],r);break}}}function n6(e,t){throw e?new Error("Cannot close `"+e.type+"` ("+fp({start:e.start,end:e.end})+"): a different token (`"+t.type+"`, "+fp({start:t.start,end:t.end})+") is open"):new Error("Cannot close document, a token (`"+t.type+"`, "+fp({start:t.start,end:t.end})+") is still open")}function Pqe(e){const t=this;t.parser=n;function n(r){return Rqe(r,{...t.data("settings"),...e,extensions:t.data("micromarkExtensions")||[],mdastExtensions:t.data("fromMarkdownExtensions")||[]})}}function $qe(e,t){const n={type:"element",tagName:"blockquote",properties:{},children:e.wrap(e.all(t),!0)};return e.patch(t,n),e.applyData(t,n)}function Mqe(e,t){const n={type:"element",tagName:"br",properties:{},children:[]};return e.patch(t,n),[e.applyData(t,n),{type:"text",value:`
70
70
  `}]}function Lqe(e,t){const n=t.value?t.value+`
71
- `:"",r={};t.lang&&(r.className=["language-"+t.lang]);let o={type:"element",tagName:"code",properties:r,children:[{type:"text",value:n}]};return t.meta&&(o.data={meta:t.meta}),e.patch(t,o),o=e.applyData(t,o),o={type:"element",tagName:"pre",properties:{},children:[o]},e.patch(t,o),o}function zqe(e,t){const n={type:"element",tagName:"del",properties:{},children:e.all(t)};return e.patch(t,n),e.applyData(t,n)}function Fqe(e,t){const n={type:"element",tagName:"em",properties:{},children:e.all(t)};return e.patch(t,n),e.applyData(t,n)}function jqe(e,t){const n=typeof e.options.clobberPrefix=="string"?e.options.clobberPrefix:"user-content-",r=String(t.identifier).toUpperCase(),o=$d(r.toLowerCase()),a=e.footnoteOrder.indexOf(r);let i,s=e.footnoteCounts.get(r);s===void 0?(s=0,e.footnoteOrder.push(r),i=e.footnoteOrder.length):i=a+1,s+=1,e.footnoteCounts.set(r,s);const l={type:"element",tagName:"a",properties:{href:"#"+n+"fn-"+o,id:n+"fnref-"+o+(s>1?"-"+s:""),dataFootnoteRef:!0,ariaDescribedBy:["footnote-label"]},children:[{type:"text",value:String(i)}]};e.patch(t,l);const u={type:"element",tagName:"sup",properties:{},children:[l]};return e.patch(t,u),e.applyData(t,u)}function Uqe(e,t){const n={type:"element",tagName:"h"+t.depth,properties:{},children:e.all(t)};return e.patch(t,n),e.applyData(t,n)}function Bqe(e,t){if(e.options.allowDangerousHtml){const n={type:"raw",value:t.value};return e.patch(t,n),e.applyData(t,n)}}function pne(e,t){const n=t.referenceType;let r="]";if(n==="collapsed"?r+="[]":n==="full"&&(r+="["+(t.label||t.identifier)+"]"),t.type==="imageReference")return[{type:"text",value:"!["+t.alt+r}];const o=e.all(t),a=o[0];a&&a.type==="text"?a.value="["+a.value:o.unshift({type:"text",value:"["});const i=o[o.length-1];return i&&i.type==="text"?i.value+=r:o.push({type:"text",value:r}),o}function Hqe(e,t){const n=String(t.identifier).toUpperCase(),r=e.definitionById.get(n);if(!r)return pne(e,t);const o={src:$d(r.url||""),alt:t.alt};r.title!==null&&r.title!==void 0&&(o.title=r.title);const a={type:"element",tagName:"img",properties:o,children:[]};return e.patch(t,a),e.applyData(t,a)}function qqe(e,t){const n={src:$d(t.url)};t.alt!==null&&t.alt!==void 0&&(n.alt=t.alt),t.title!==null&&t.title!==void 0&&(n.title=t.title);const r={type:"element",tagName:"img",properties:n,children:[]};return e.patch(t,r),e.applyData(t,r)}function Gqe(e,t){const n={type:"text",value:t.value.replace(/\r?\n|\r/g," ")};e.patch(t,n);const r={type:"element",tagName:"code",properties:{},children:[n]};return e.patch(t,r),e.applyData(t,r)}function Wqe(e,t){const n=String(t.identifier).toUpperCase(),r=e.definitionById.get(n);if(!r)return pne(e,t);const o={href:$d(r.url||"")};r.title!==null&&r.title!==void 0&&(o.title=r.title);const a={type:"element",tagName:"a",properties:o,children:e.all(t)};return e.patch(t,a),e.applyData(t,a)}function Vqe(e,t){const n={href:$d(t.url)};t.title!==null&&t.title!==void 0&&(n.title=t.title);const r={type:"element",tagName:"a",properties:n,children:e.all(t)};return e.patch(t,r),e.applyData(t,r)}function Zqe(e,t,n){const r=e.all(t),o=n?Yqe(n):hne(t),a={},i=[];if(typeof t.checked=="boolean"){const d=r[0];let p;d&&d.type==="element"&&d.tagName==="p"?p=d:(p={type:"element",tagName:"p",properties:{},children:[]},r.unshift(p)),p.children.length>0&&p.children.unshift({type:"text",value:" "}),p.children.unshift({type:"element",tagName:"input",properties:{type:"checkbox",checked:t.checked,disabled:!0},children:[]}),a.className=["task-list-item"]}let s=-1;for(;++s<r.length;){const d=r[s];(o||s!==0||d.type!=="element"||d.tagName!=="p")&&i.push({type:"text",value:`
71
+ `:"",r={},o=t.lang?t.lang.split(/\s+/):[];o.length>0&&(r.className=["language-"+o[0]]);let a={type:"element",tagName:"code",properties:r,children:[{type:"text",value:n}]};return t.meta&&(a.data={meta:t.meta}),e.patch(t,a),a=e.applyData(t,a),a={type:"element",tagName:"pre",properties:{},children:[a]},e.patch(t,a),a}function zqe(e,t){const n={type:"element",tagName:"del",properties:{},children:e.all(t)};return e.patch(t,n),e.applyData(t,n)}function Fqe(e,t){const n={type:"element",tagName:"em",properties:{},children:e.all(t)};return e.patch(t,n),e.applyData(t,n)}function jqe(e,t){const n=typeof e.options.clobberPrefix=="string"?e.options.clobberPrefix:"user-content-",r=String(t.identifier).toUpperCase(),o=$d(r.toLowerCase()),a=e.footnoteOrder.indexOf(r);let i,s=e.footnoteCounts.get(r);s===void 0?(s=0,e.footnoteOrder.push(r),i=e.footnoteOrder.length):i=a+1,s+=1,e.footnoteCounts.set(r,s);const l={type:"element",tagName:"a",properties:{href:"#"+n+"fn-"+o,id:n+"fnref-"+o+(s>1?"-"+s:""),dataFootnoteRef:!0,ariaDescribedBy:["footnote-label"]},children:[{type:"text",value:String(i)}]};e.patch(t,l);const u={type:"element",tagName:"sup",properties:{},children:[l]};return e.patch(t,u),e.applyData(t,u)}function Uqe(e,t){const n={type:"element",tagName:"h"+t.depth,properties:{},children:e.all(t)};return e.patch(t,n),e.applyData(t,n)}function Bqe(e,t){if(e.options.allowDangerousHtml){const n={type:"raw",value:t.value};return e.patch(t,n),e.applyData(t,n)}}function pne(e,t){const n=t.referenceType;let r="]";if(n==="collapsed"?r+="[]":n==="full"&&(r+="["+(t.label||t.identifier)+"]"),t.type==="imageReference")return[{type:"text",value:"!["+t.alt+r}];const o=e.all(t),a=o[0];a&&a.type==="text"?a.value="["+a.value:o.unshift({type:"text",value:"["});const i=o[o.length-1];return i&&i.type==="text"?i.value+=r:o.push({type:"text",value:r}),o}function Hqe(e,t){const n=String(t.identifier).toUpperCase(),r=e.definitionById.get(n);if(!r)return pne(e,t);const o={src:$d(r.url||""),alt:t.alt};r.title!==null&&r.title!==void 0&&(o.title=r.title);const a={type:"element",tagName:"img",properties:o,children:[]};return e.patch(t,a),e.applyData(t,a)}function qqe(e,t){const n={src:$d(t.url)};t.alt!==null&&t.alt!==void 0&&(n.alt=t.alt),t.title!==null&&t.title!==void 0&&(n.title=t.title);const r={type:"element",tagName:"img",properties:n,children:[]};return e.patch(t,r),e.applyData(t,r)}function Gqe(e,t){const n={type:"text",value:t.value.replace(/\r?\n|\r/g," ")};e.patch(t,n);const r={type:"element",tagName:"code",properties:{},children:[n]};return e.patch(t,r),e.applyData(t,r)}function Wqe(e,t){const n=String(t.identifier).toUpperCase(),r=e.definitionById.get(n);if(!r)return pne(e,t);const o={href:$d(r.url||"")};r.title!==null&&r.title!==void 0&&(o.title=r.title);const a={type:"element",tagName:"a",properties:o,children:e.all(t)};return e.patch(t,a),e.applyData(t,a)}function Vqe(e,t){const n={href:$d(t.url)};t.title!==null&&t.title!==void 0&&(n.title=t.title);const r={type:"element",tagName:"a",properties:n,children:e.all(t)};return e.patch(t,r),e.applyData(t,r)}function Zqe(e,t,n){const r=e.all(t),o=n?Yqe(n):hne(t),a={},i=[];if(typeof t.checked=="boolean"){const d=r[0];let p;d&&d.type==="element"&&d.tagName==="p"?p=d:(p={type:"element",tagName:"p",properties:{},children:[]},r.unshift(p)),p.children.length>0&&p.children.unshift({type:"text",value:" "}),p.children.unshift({type:"element",tagName:"input",properties:{type:"checkbox",checked:t.checked,disabled:!0},children:[]}),a.className=["task-list-item"]}let s=-1;for(;++s<r.length;){const d=r[s];(o||s!==0||d.type!=="element"||d.tagName!=="p")&&i.push({type:"text",value:`
72
72
  `}),d.type==="element"&&d.tagName==="p"&&!o?i.push(...d.children):i.push(d)}const l=r[r.length-1];l&&(o||l.type!=="element"||l.tagName!=="p")&&i.push({type:"text",value:`
73
73
  `});const u={type:"element",tagName:"li",properties:a,children:i};return e.patch(t,u),e.applyData(t,u)}function Yqe(e){let t=!1;if(e.type==="list"){t=e.spread||!1;const n=e.children;let r=-1;for(;!t&&++r<n.length;)t=hne(n[r])}return t}function hne(e){const t=e.spread;return t??e.children.length>1}function Kqe(e,t){const n={},r=e.all(t);let o=-1;for(typeof t.start=="number"&&t.start!==1&&(n.start=t.start);++o<r.length;){const i=r[o];if(i.type==="element"&&i.tagName==="li"&&i.properties&&Array.isArray(i.properties.className)&&i.properties.className.includes("task-list-item")){n.className=["contains-task-list"];break}}const a={type:"element",tagName:t.ordered?"ol":"ul",properties:n,children:e.wrap(r,!0)};return e.patch(t,a),e.applyData(t,a)}function Xqe(e,t){const n={type:"element",tagName:"p",properties:{},children:e.all(t)};return e.patch(t,n),e.applyData(t,n)}function Qqe(e,t){const n={type:"root",children:e.wrap(e.all(t))};return e.patch(t,n),e.applyData(t,n)}function Jqe(e,t){const n={type:"element",tagName:"strong",properties:{},children:e.all(t)};return e.patch(t,n),e.applyData(t,n)}function e8e(e,t){const n=e.all(t),r=n.shift(),o=[];if(r){const i={type:"element",tagName:"thead",properties:{},children:e.wrap([r],!0)};e.patch(t.children[0],i),o.push(i)}if(n.length>0){const i={type:"element",tagName:"tbody",properties:{},children:e.wrap(n,!0)},s=k$(t.children[1]),l=Zte(t.children[t.children.length-1]);s&&l&&(i.position={start:s,end:l}),o.push(i)}const a={type:"element",tagName:"table",properties:{},children:e.wrap(o,!0)};return e.patch(t,a),e.applyData(t,a)}function t8e(e,t,n){const r=n?n.children:void 0,a=(r?r.indexOf(t):1)===0?"th":"td",i=n&&n.type==="table"?n.align:void 0,s=i?i.length:t.children.length;let l=-1;const u=[];for(;++l<s;){const p=t.children[l],h={},m=i?i[l]:void 0;m&&(h.align=m);let b={type:"element",tagName:a,properties:h,children:[]};p&&(b.children=e.all(p),e.patch(p,b),b=e.applyData(p,b)),u.push(b)}const d={type:"element",tagName:"tr",properties:{},children:e.wrap(u,!0)};return e.patch(t,d),e.applyData(t,d)}function n8e(e,t){const n={type:"element",tagName:"td",properties:{},children:e.all(t)};return e.patch(t,n),e.applyData(t,n)}const r6=9,o6=32;function r8e(e){const t=String(e),n=/\r?\n|\r/g;let r=n.exec(t),o=0;const a=[];for(;r;)a.push(a6(t.slice(o,r.index),o>0,!0),r[0]),o=r.index+r[0].length,r=n.exec(t);return a.push(a6(t.slice(o),o>0,!1)),a.join("")}function a6(e,t,n){let r=0,o=e.length;if(t){let a=e.codePointAt(r);for(;a===r6||a===o6;)r++,a=e.codePointAt(r)}if(n){let a=e.codePointAt(o-1);for(;a===r6||a===o6;)o--,a=e.codePointAt(o-1)}return o>r?e.slice(r,o):""}function o8e(e,t){const n={type:"text",value:r8e(String(t.value))};return e.patch(t,n),e.applyData(t,n)}function a8e(e,t){const n={type:"element",tagName:"hr",properties:{},children:[]};return e.patch(t,n),e.applyData(t,n)}const i8e={blockquote:$qe,break:Mqe,code:Lqe,delete:zqe,emphasis:Fqe,footnoteReference:jqe,heading:Uqe,html:Bqe,imageReference:Hqe,image:qqe,inlineCode:Gqe,linkReference:Wqe,link:Vqe,listItem:Zqe,list:Kqe,paragraph:Xqe,root:Qqe,strong:Jqe,table:e8e,tableCell:n8e,tableRow:t8e,text:o8e,thematicBreak:a8e,toml:Kg,yaml:Kg,definition:Kg,footnoteDefinition:Kg};function Kg(){}const mne=-1,rS=0,hp=1,Lv=2,I$=3,R$=4,N$=5,D$=6,gne=7,bne=8,i6=typeof self=="object"?self:globalThis,s8e=(e,t)=>{const n=(o,a)=>(e.set(a,o),o),r=o=>{if(e.has(o))return e.get(o);const[a,i]=t[o];switch(a){case rS:case mne:return n(i,o);case hp:{const s=n([],o);for(const l of i)s.push(r(l));return s}case Lv:{const s=n({},o);for(const[l,u]of i)s[r(l)]=r(u);return s}case I$:return n(new Date(i),o);case R$:{const{source:s,flags:l}=i;return n(new RegExp(s,l),o)}case N$:{const s=n(new Map,o);for(const[l,u]of i)s.set(r(l),r(u));return s}case D$:{const s=n(new Set,o);for(const l of i)s.add(r(l));return s}case gne:{const{name:s,message:l}=i;return n(new i6[s](l),o)}case bne:return n(BigInt(i),o);case"BigInt":return n(Object(BigInt(i)),o);case"ArrayBuffer":return n(new Uint8Array(i).buffer,i);case"DataView":{const{buffer:s}=new Uint8Array(i);return n(new DataView(s),i)}}return n(new i6[a](i),o)};return r},s6=e=>s8e(new Map,e)(0),Ku="",{toString:l8e}={},{keys:u8e}=Object,Uf=e=>{const t=typeof e;if(t!=="object"||!e)return[rS,t];const n=l8e.call(e).slice(8,-1);switch(n){case"Array":return[hp,Ku];case"Object":return[Lv,Ku];case"Date":return[I$,Ku];case"RegExp":return[R$,Ku];case"Map":return[N$,Ku];case"Set":return[D$,Ku];case"DataView":return[hp,n]}return n.includes("Array")?[hp,n]:n.includes("Error")?[gne,n]:[Lv,n]},Xg=([e,t])=>e===rS&&(t==="function"||t==="symbol"),c8e=(e,t,n,r)=>{const o=(i,s)=>{const l=r.push(i)-1;return n.set(s,l),l},a=i=>{if(n.has(i))return n.get(i);let[s,l]=Uf(i);switch(s){case rS:{let d=i;switch(l){case"bigint":s=bne,d=i.toString();break;case"function":case"symbol":if(e)throw new TypeError("unable to serialize "+l);d=null;break;case"undefined":return o([mne],i)}return o([s,d],i)}case hp:{if(l){let h=i;return l==="DataView"?h=new Uint8Array(i.buffer):l==="ArrayBuffer"&&(h=new Uint8Array(i)),o([l,[...h]],i)}const d=[],p=o([s,d],i);for(const h of i)d.push(a(h));return p}case Lv:{if(l)switch(l){case"BigInt":return o([l,i.toString()],i);case"Boolean":case"Number":case"String":return o([l,i.valueOf()],i)}if(t&&"toJSON"in i)return a(i.toJSON());const d=[],p=o([s,d],i);for(const h of u8e(i))(e||!Xg(Uf(i[h])))&&d.push([a(h),a(i[h])]);return p}case I$:return o([s,i.toISOString()],i);case R$:{const{source:d,flags:p}=i;return o([s,{source:d,flags:p}],i)}case N$:{const d=[],p=o([s,d],i);for(const[h,m]of i)(e||!(Xg(Uf(h))||Xg(Uf(m))))&&d.push([a(h),a(m)]);return p}case D$:{const d=[],p=o([s,d],i);for(const h of i)(e||!Xg(Uf(h)))&&d.push(a(h));return p}}const{message:u}=i;return o([s,{name:l,message:u}],i)};return a},l6=(e,{json:t,lossy:n}={})=>{const r=[];return c8e(!(t||n),!!t,new Map,r)(e),r},zv=typeof structuredClone=="function"?(e,t)=>t&&("json"in t||"lossy"in t)?s6(l6(e,t)):structuredClone(e):(e,t)=>s6(l6(e,t));function d8e(e,t){const n=[{type:"text",value:"↩"}];return t>1&&n.push({type:"element",tagName:"sup",properties:{},children:[{type:"text",value:String(t)}]}),n}function f8e(e,t){return"Back to reference "+(e+1)+(t>1?"-"+t:"")}function p8e(e){const t=typeof e.options.clobberPrefix=="string"?e.options.clobberPrefix:"user-content-",n=e.options.footnoteBackContent||d8e,r=e.options.footnoteBackLabel||f8e,o=e.options.footnoteLabel||"Footnotes",a=e.options.footnoteLabelTagName||"h2",i=e.options.footnoteLabelProperties||{className:["sr-only"]},s=[];let l=-1;for(;++l<e.footnoteOrder.length;){const u=e.footnoteById.get(e.footnoteOrder[l]);if(!u)continue;const d=e.all(u),p=String(u.identifier).toUpperCase(),h=$d(p.toLowerCase());let m=0;const b=[],y=e.footnoteCounts.get(p);for(;y!==void 0&&++m<=y;){b.length>0&&b.push({type:"text",value:" "});let _=typeof n=="string"?n:n(l,m);typeof _=="string"&&(_={type:"text",value:_}),b.push({type:"element",tagName:"a",properties:{href:"#"+t+"fnref-"+h+(m>1?"-"+m:""),dataFootnoteBackref:"",ariaLabel:typeof r=="string"?r:r(l,m),className:["data-footnote-backref"]},children:Array.isArray(_)?_:[_]})}const v=d[d.length-1];if(v&&v.type==="element"&&v.tagName==="p"){const _=v.children[v.children.length-1];_&&_.type==="text"?_.value+=" ":v.children.push({type:"text",value:" "}),v.children.push(...b)}else d.push(...b);const k={type:"element",tagName:"li",properties:{id:t+"fn-"+h},children:e.wrap(d,!0)};e.patch(u,k),s.push(k)}if(s.length!==0)return{type:"element",tagName:"section",properties:{dataFootnotes:!0,className:["footnotes"]},children:[{type:"element",tagName:a,properties:{...zv(i),id:"footnote-label"},children:[{type:"text",value:o}]},{type:"text",value:`
74
74
  `},{type:"element",tagName:"ol",properties:{},children:e.wrap(s,!0)},{type:"text",value:`
@@ -4,7 +4,7 @@
4
4
  <meta charset="UTF-8" />
5
5
  <meta name="viewport" content="width=device-width, initial-scale=1.0" />
6
6
  <title>{{projectName}}: Sonamu UI</title>
7
- <script type="module" crossorigin src="/sonamu-ui/assets/index-DzqUrTB-.js"></script>
7
+ <script type="module" crossorigin src="/sonamu-ui/assets/index-DFqVuxOB.js"></script>
8
8
  <link rel="stylesheet" crossorigin href="/sonamu-ui/assets/index-CpaB9P6g.css">
9
9
  </head>
10
10
  <body>
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "sonamu",
3
- "version": "0.7.17",
3
+ "version": "0.7.18",
4
4
  "description": "Sonamu — TypeScript Fullstack API Framework",
5
5
  "keywords": [
6
6
  "typescript",
@@ -81,10 +81,10 @@
81
81
  "tsicli": "^1.0.5",
82
82
  "vitest": "^4.0.10",
83
83
  "zod": "^4.1.12",
84
+ "@sonamu-kit/hmr-hook": "^0.4.1",
84
85
  "@sonamu-kit/hmr-runner": "^0.1.1",
85
- "@sonamu-kit/tasks": "^0.1.1",
86
86
  "@sonamu-kit/ts-loader": "^2.1.3",
87
- "@sonamu-kit/hmr-hook": "^0.4.1"
87
+ "@sonamu-kit/tasks": "^0.1.1"
88
88
  },
89
89
  "devDependencies": {
90
90
  "@biomejs/biome": "^2.3.7",
@@ -11,6 +11,7 @@ import type { TemplateOptions } from "../../types/types";
11
11
  import { ApiParamType } from "../../types/types";
12
12
  import { assertDefined } from "../../utils/utils";
13
13
  import { Template } from "../template";
14
+ import { zodTypeToTsTypeDef } from "../zod-converter";
14
15
 
15
16
  export class Template__services extends Template {
16
17
  constructor() {
@@ -45,6 +46,44 @@ export class Template__services extends Template {
45
46
  const functions: string[] = [];
46
47
 
47
48
  for (const api of modelApis) {
49
+ // @stream 데코레이터가 있으면 SSE 스트림 함수 생성
50
+ if (api.streamOptions) {
51
+ const paramsWithoutContext = api.parameters.filter(
52
+ (param) =>
53
+ !ApiParamType.isContext(param.type) &&
54
+ !ApiParamType.isRefKnex(param.type) &&
55
+ !(param.optional === true && param.name.startsWith("_")),
56
+ );
57
+
58
+ const paramsDef = apiParamToTsCode(paramsWithoutContext, importKeys);
59
+ const apiBaseUrl = `${Sonamu.config.api.route.prefix}${api.path}`;
60
+
61
+ const methodNameStream = api.options.resourceName
62
+ ? `use${inflection.camelize(api.options.resourceName)}`
63
+ : `use${inflection.camelize(api.methodName)}`;
64
+ const methodNameStreamCamelized = inflection.camelize(methodNameStream, true);
65
+
66
+ const eventsTypeDef = zodTypeToTsTypeDef(api.streamOptions.events);
67
+
68
+ const paramsDefAsObject =
69
+ paramsWithoutContext.length > 0
70
+ ? `{ ${paramsWithoutContext.map((p) => p.name).join(", ")} }`
71
+ : "{}";
72
+
73
+ functions.push(
74
+ `
75
+ export function ${methodNameStreamCamelized}(
76
+ params: ${paramsDef ? `{ ${paramsWithoutContext.map((p) => `${p.name}: ${apiParamTypeToTsType(p.type, importKeys)}`).join(", ")} }` : "{}"},
77
+ handlers: EventHandlers<${eventsTypeDef} & { end?: () => void }>,
78
+ options: SSEStreamOptions
79
+ ) {
80
+ return useSSEStream<${eventsTypeDef}>(\`${apiBaseUrl}\`, ${paramsDefAsObject}, handlers, options);
81
+ }
82
+ `.trim(),
83
+ );
84
+ continue;
85
+ }
86
+
48
87
  // Context 제외한 파라미터
49
88
  const paramsWithoutContext = api.parameters.filter(
50
89
  (param) =>
@@ -219,7 +258,7 @@ ${functions.join("\n\n")}
219
258
  `import { queryOptions, useQuery, useMutation } from '@tanstack/react-query';`,
220
259
  `import type { AxiosProgressEvent } from 'axios';`,
221
260
  `import qs from 'qs';`,
222
- `import { type ListResult, fetch } from './sonamu.shared';`,
261
+ `import { type ListResult, fetch, type EventHandlers, type SSEStreamOptions, useSSEStream } from './sonamu.shared';`,
223
262
  ],
224
263
  };
225
264
  }