@nestia/sdk 10.0.2 → 11.0.0-dev.20260312

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 (151) hide show
  1. package/LICENSE +21 -21
  2. package/README.md +93 -93
  3. package/assets/bundle/api/HttpError.ts +1 -1
  4. package/assets/bundle/api/IConnection.ts +1 -1
  5. package/assets/bundle/api/Primitive.ts +1 -1
  6. package/assets/bundle/api/Resolved.ts +1 -1
  7. package/assets/bundle/api/index.ts +4 -4
  8. package/assets/bundle/api/module.ts +6 -6
  9. package/assets/bundle/distribute/README.md +37 -37
  10. package/assets/bundle/distribute/package.json +28 -28
  11. package/assets/bundle/distribute/tsconfig.json +109 -109
  12. package/assets/bundle/e2e/index.ts +42 -42
  13. package/assets/config/nestia.config.ts +97 -97
  14. package/lib/INestiaConfig.d.ts +3 -3
  15. package/lib/NestiaSdkApplication.js.map +1 -1
  16. package/lib/NestiaSwaggerComposer.d.ts +3 -2
  17. package/lib/NestiaSwaggerComposer.js +5 -6
  18. package/lib/NestiaSwaggerComposer.js.map +1 -1
  19. package/lib/analyses/AccessorAnalyzer.js +2 -2
  20. package/lib/analyses/AccessorAnalyzer.js.map +1 -1
  21. package/lib/analyses/DtoAnalyzer.js.map +1 -1
  22. package/lib/analyses/ImportAnalyzer.js.map +1 -1
  23. package/lib/analyses/ReflectHttpOperationAnalyzer.js +3 -3
  24. package/lib/analyses/ReflectHttpOperationAnalyzer.js.map +1 -1
  25. package/lib/analyses/ReflectHttpOperationExceptionAnalyzer.js +4 -2
  26. package/lib/analyses/ReflectHttpOperationExceptionAnalyzer.js.map +1 -1
  27. package/lib/analyses/ReflectHttpOperationParameterAnalyzer.js +7 -11
  28. package/lib/analyses/ReflectHttpOperationParameterAnalyzer.js.map +1 -1
  29. package/lib/analyses/ReflectHttpOperationResponseAnalyzer.js +6 -6
  30. package/lib/analyses/ReflectHttpOperationResponseAnalyzer.js.map +1 -1
  31. package/lib/analyses/TypedHttpRouteAnalyzer.d.ts +1 -1
  32. package/lib/analyses/TypedHttpRouteAnalyzer.js +6 -8
  33. package/lib/analyses/TypedHttpRouteAnalyzer.js.map +1 -1
  34. package/lib/executable/internal/NestiaConfigLoader.js +111 -52
  35. package/lib/executable/internal/NestiaConfigLoader.js.map +1 -1
  36. package/lib/executable/internal/NestiaSdkCommand.js.map +1 -1
  37. package/lib/executable/sdk.js +0 -0
  38. package/lib/generates/SwaggerGenerator.d.ts +1 -1
  39. package/lib/generates/SwaggerGenerator.js +11 -12
  40. package/lib/generates/SwaggerGenerator.js.map +1 -1
  41. package/lib/generates/internal/E2eFileProgrammer.js +8 -9
  42. package/lib/generates/internal/E2eFileProgrammer.js.map +1 -1
  43. package/lib/generates/internal/SdkAliasCollection.d.ts +2 -2
  44. package/lib/generates/internal/SdkAliasCollection.js +2 -2
  45. package/lib/generates/internal/SdkAliasCollection.js.map +1 -1
  46. package/lib/generates/internal/SdkDistributionComposer.js +1 -1
  47. package/lib/generates/internal/SdkHttpCloneProgrammer.js.map +1 -1
  48. package/lib/generates/internal/SdkHttpCloneReferencer.js.map +1 -1
  49. package/lib/generates/internal/SdkHttpFunctionProgrammer.js +11 -12
  50. package/lib/generates/internal/SdkHttpFunctionProgrammer.js.map +1 -1
  51. package/lib/generates/internal/SdkHttpNamespaceProgrammer.js +13 -16
  52. package/lib/generates/internal/SdkHttpNamespaceProgrammer.js.map +1 -1
  53. package/lib/generates/internal/SdkHttpRouteProgrammer.js +4 -4
  54. package/lib/generates/internal/SdkHttpRouteProgrammer.js.map +1 -1
  55. package/lib/generates/internal/SdkHttpSimulationProgrammer.js +12 -16
  56. package/lib/generates/internal/SdkHttpSimulationProgrammer.js.map +1 -1
  57. package/lib/generates/internal/SdkImportWizard.js +2 -2
  58. package/lib/generates/internal/SdkImportWizard.js.map +1 -1
  59. package/lib/generates/internal/SdkTypeProgrammer.d.ts +2 -3
  60. package/lib/generates/internal/SdkTypeProgrammer.js +6 -7
  61. package/lib/generates/internal/SdkTypeProgrammer.js.map +1 -1
  62. package/lib/generates/internal/SdkTypeTagProgrammer.d.ts +1 -1
  63. package/lib/generates/internal/SdkTypeTagProgrammer.js +11 -11
  64. package/lib/generates/internal/SdkTypeTagProgrammer.js.map +1 -1
  65. package/lib/generates/internal/SdkWebSocketNamespaceProgrammer.js +6 -8
  66. package/lib/generates/internal/SdkWebSocketNamespaceProgrammer.js.map +1 -1
  67. package/lib/generates/internal/SdkWebSocketRouteProgrammer.js +2 -2
  68. package/lib/generates/internal/SdkWebSocketRouteProgrammer.js.map +1 -1
  69. package/lib/generates/internal/SwaggerOperationComposer.d.ts +3 -3
  70. package/lib/generates/internal/SwaggerOperationComposer.js.map +1 -1
  71. package/lib/generates/internal/SwaggerOperationParameterComposer.d.ts +1 -1
  72. package/lib/generates/internal/SwaggerOperationParameterComposer.js +10 -8
  73. package/lib/generates/internal/SwaggerOperationParameterComposer.js.map +1 -1
  74. package/lib/generates/internal/SwaggerOperationResponseComposer.d.ts +3 -3
  75. package/lib/generates/internal/SwaggerOperationResponseComposer.js.map +1 -1
  76. package/lib/module.d.ts +1 -0
  77. package/lib/module.js +1 -0
  78. package/lib/module.js.map +1 -1
  79. package/lib/structures/IReflectHttpOperationException.d.ts +3 -4
  80. package/lib/structures/IReflectHttpOperationParameter.d.ts +3 -5
  81. package/lib/structures/IReflectHttpOperationSuccess.d.ts +3 -4
  82. package/lib/structures/ITypedApplication.d.ts +1 -1
  83. package/lib/structures/ITypedHttpRouteException.d.ts +2 -2
  84. package/lib/structures/ITypedHttpRouteParameter.d.ts +2 -2
  85. package/lib/structures/ITypedHttpRouteSuccess.d.ts +2 -2
  86. package/lib/transformers/IOperationMetadata.d.ts +2 -4
  87. package/lib/transformers/ISdkOperationTransformerContext.d.ts +1 -1
  88. package/lib/transformers/SdkOperationProgrammer.js +8 -10
  89. package/lib/transformers/SdkOperationProgrammer.js.map +1 -1
  90. package/lib/transformers/SdkOperationTransformer.js +6 -8
  91. package/lib/transformers/SdkOperationTransformer.js.map +1 -1
  92. package/lib/transformers/TextPlainValidator.d.ts +2 -2
  93. package/lib/transformers/TextPlainValidator.js.map +1 -1
  94. package/lib/utils/MetadataUtil.d.ts +2 -2
  95. package/lib/utils/MetadataUtil.js.map +1 -1
  96. package/lib/validators/HttpHeadersValidator.d.ts +2 -3
  97. package/lib/validators/HttpHeadersValidator.js +2 -2
  98. package/lib/validators/HttpHeadersValidator.js.map +1 -1
  99. package/lib/validators/HttpQueryValidator.d.ts +2 -3
  100. package/lib/validators/HttpQueryValidator.js +2 -2
  101. package/lib/validators/HttpQueryValidator.js.map +1 -1
  102. package/package.json +44 -30
  103. package/src/INestiaConfig.ts +267 -267
  104. package/src/NestiaSdkApplication.ts +307 -307
  105. package/src/NestiaSwaggerComposer.ts +143 -138
  106. package/src/analyses/AccessorAnalyzer.ts +67 -67
  107. package/src/analyses/DtoAnalyzer.ts +260 -260
  108. package/src/analyses/ImportAnalyzer.ts +126 -126
  109. package/src/analyses/ReflectHttpOperationAnalyzer.ts +183 -183
  110. package/src/analyses/ReflectHttpOperationExceptionAnalyzer.ts +72 -71
  111. package/src/analyses/ReflectHttpOperationParameterAnalyzer.ts +350 -348
  112. package/src/analyses/ReflectHttpOperationResponseAnalyzer.ts +126 -127
  113. package/src/analyses/TypedHttpRouteAnalyzer.ts +208 -204
  114. package/src/executable/internal/NestiaConfigLoader.ts +85 -78
  115. package/src/executable/internal/NestiaSdkCommand.ts +107 -103
  116. package/src/generates/SwaggerGenerator.ts +291 -284
  117. package/src/generates/internal/E2eFileProgrammer.ts +196 -197
  118. package/src/generates/internal/FilePrinter.ts +64 -64
  119. package/src/generates/internal/ImportDictionary.ts +192 -192
  120. package/src/generates/internal/SdkAliasCollection.ts +260 -261
  121. package/src/generates/internal/SdkFileProgrammer.ts +110 -110
  122. package/src/generates/internal/SdkHttpCloneProgrammer.ts +126 -124
  123. package/src/generates/internal/SdkHttpCloneReferencer.ts +77 -77
  124. package/src/generates/internal/SdkHttpFunctionProgrammer.ts +278 -279
  125. package/src/generates/internal/SdkHttpNamespaceProgrammer.ts +502 -500
  126. package/src/generates/internal/SdkHttpRouteProgrammer.ts +109 -108
  127. package/src/generates/internal/SdkHttpSimulationProgrammer.ts +312 -310
  128. package/src/generates/internal/SdkImportWizard.ts +62 -62
  129. package/src/generates/internal/SdkTypeProgrammer.ts +388 -385
  130. package/src/generates/internal/SdkTypeTagProgrammer.ts +114 -104
  131. package/src/generates/internal/SdkWebSocketNamespaceProgrammer.ts +379 -381
  132. package/src/generates/internal/SdkWebSocketRouteProgrammer.ts +302 -302
  133. package/src/generates/internal/SwaggerOperationComposer.ts +119 -119
  134. package/src/generates/internal/SwaggerOperationParameterComposer.ts +161 -162
  135. package/src/generates/internal/SwaggerOperationResponseComposer.ts +110 -110
  136. package/src/module.ts +4 -3
  137. package/src/structures/IReflectHttpOperationException.ts +18 -19
  138. package/src/structures/IReflectHttpOperationParameter.ts +79 -77
  139. package/src/structures/IReflectHttpOperationSuccess.ts +21 -22
  140. package/src/structures/ITypedApplication.ts +11 -11
  141. package/src/structures/ITypedHttpRouteException.ts +15 -15
  142. package/src/structures/ITypedHttpRouteParameter.ts +41 -41
  143. package/src/structures/ITypedHttpRouteSuccess.ts +22 -22
  144. package/src/transformers/IOperationMetadata.ts +46 -44
  145. package/src/transformers/ISdkOperationTransformerContext.ts +8 -8
  146. package/src/transformers/SdkOperationProgrammer.ts +240 -238
  147. package/src/transformers/SdkOperationTransformer.ts +248 -252
  148. package/src/transformers/TextPlainValidator.ts +17 -17
  149. package/src/utils/MetadataUtil.ts +26 -26
  150. package/src/validators/HttpHeadersValidator.ts +36 -34
  151. package/src/validators/HttpQueryValidator.ts +36 -34
@@ -1,284 +1,291 @@
1
- import { SwaggerCustomizer } from "@nestia/core";
2
- import { OpenApi, OpenApiV3, SwaggerV2 } from "@samchon/openapi";
3
- import fs from "fs";
4
- import path from "path";
5
- import { Singleton } from "tstl";
6
- import typia, { IJsonSchemaCollection } from "typia";
7
- import { JsonSchemasProgrammer } from "typia/lib/programmers/json/JsonSchemasProgrammer";
8
- import { Metadata } from "typia/lib/schemas/metadata/Metadata";
9
-
10
- import { INestiaConfig } from "../INestiaConfig";
11
- import { ITypedApplication } from "../structures/ITypedApplication";
12
- import { ITypedHttpRoute } from "../structures/ITypedHttpRoute";
13
- import { FileRetriever } from "../utils/FileRetriever";
14
- import { SdkHttpParameterProgrammer } from "./internal/SdkHttpParameterProgrammer";
15
- import { SwaggerOperationComposer } from "./internal/SwaggerOperationComposer";
16
-
17
- export namespace SwaggerGenerator {
18
- export const generate = async (app: ITypedApplication): Promise<void> => {
19
- // GET CONFIGURATION
20
- console.log("Generating Swagger Document");
21
- if (app.project.config.swagger === undefined)
22
- throw new Error("Swagger configuration is not defined.");
23
- const config: INestiaConfig.ISwaggerConfig = app.project.config.swagger;
24
-
25
- // TARGET LOCATION
26
- const parsed: path.ParsedPath = path.parse(config.output);
27
- const directory: string = path.dirname(parsed.dir);
28
- if (fs.existsSync(directory) === false)
29
- try {
30
- await fs.promises.mkdir(directory);
31
- } catch {}
32
- if (fs.existsSync(directory) === false)
33
- throw new Error(
34
- `Error on NestiaApplication.swagger(): failed to create output directory: ${directory}`,
35
- );
36
- const location: string = !!parsed.ext
37
- ? path.resolve(config.output)
38
- : path.join(path.resolve(config.output), "swagger.json");
39
-
40
- // COMPOSE SWAGGER DOCUMENT
41
- const document: OpenApi.IDocument = compose({
42
- config,
43
- routes: app.routes.filter((route) => route.protocol === "http"),
44
- document: await initialize(config),
45
- });
46
- const specified:
47
- | OpenApi.IDocument
48
- | SwaggerV2.IDocument
49
- | OpenApiV3.IDocument =
50
- config.openapi === "2.0"
51
- ? OpenApi.downgrade(document, config.openapi as "2.0")
52
- : config.openapi === "3.0"
53
- ? OpenApi.downgrade(document, config.openapi as "3.0")
54
- : document;
55
- await fs.promises.writeFile(
56
- location,
57
- !config.beautify
58
- ? JSON.stringify(specified)
59
- : JSON.stringify(
60
- specified,
61
- null,
62
- typeof config.beautify === "number" ? config.beautify : 2,
63
- ),
64
- "utf8",
65
- );
66
- };
67
-
68
- export const compose = (props: {
69
- config: Omit<INestiaConfig.ISwaggerConfig, "output">;
70
- routes: ITypedHttpRoute[];
71
- document: OpenApi.IDocument;
72
- }): OpenApi.IDocument => {
73
- // GATHER METADATA
74
- const routes: ITypedHttpRoute[] = props.routes.filter((r) =>
75
- r.jsDocTags.every(
76
- (tag) => tag.name !== "internal" && tag.name !== "hidden",
77
- ),
78
- );
79
- const metadatas: Metadata[] = routes
80
- .map((r) => [
81
- r.success.metadata,
82
- ...SdkHttpParameterProgrammer.getAll(r).map((p) => p.metadata),
83
- ...Object.values(r.exceptions).map((e) => e.metadata),
84
- ])
85
- .flat()
86
- .filter((m) => m.size() !== 0);
87
-
88
- // COMPOSE JSON SCHEMAS
89
- const json: IJsonSchemaCollection = JsonSchemasProgrammer.write({
90
- version: "3.1",
91
- metadatas,
92
- });
93
- const dict: WeakMap<Metadata, OpenApi.IJsonSchema> = new WeakMap();
94
- json.schemas.forEach((schema, i) => dict.set(metadatas[i], schema));
95
- const schema = (metadata: Metadata): OpenApi.IJsonSchema | undefined =>
96
- dict.get(metadata);
97
-
98
- // COMPOSE DOCUMENT
99
- const document: OpenApi.IDocument = props.document;
100
- document.components.schemas ??= {};
101
- Object.assign(document.components.schemas, json.components.schemas);
102
- fillPaths({
103
- ...props,
104
- routes,
105
- schema,
106
- document,
107
- });
108
- return document;
109
- };
110
-
111
- export const initialize = async (
112
- config: Omit<INestiaConfig.ISwaggerConfig, "output">,
113
- ): Promise<OpenApi.IDocument> => {
114
- const pack = new Singleton(
115
- async (): Promise<Partial<OpenApi.IDocument.IInfo> | null> => {
116
- const location: string | null = await FileRetriever.file(
117
- "package.json",
118
- )(process.cwd());
119
- if (location === null) return null;
120
-
121
- try {
122
- const content: string = await fs.promises.readFile(location, "utf8");
123
- const data = typia.json.assertParse<{
124
- name?: string;
125
- version?: string;
126
- description?: string;
127
- license?:
128
- | string
129
- | {
130
- type: string;
131
- /** @format uri */
132
- url: string;
133
- };
134
- }>(content);
135
- return {
136
- title: data.name,
137
- version: data.version,
138
- description: data.description,
139
- license: data.license
140
- ? typeof data.license === "string"
141
- ? { name: data.license }
142
- : typeof data.license === "object"
143
- ? {
144
- name: data.license.type,
145
- url: data.license.url,
146
- }
147
- : undefined
148
- : undefined,
149
- };
150
- } catch {
151
- return null;
152
- }
153
- },
154
- );
155
-
156
- return {
157
- openapi: "3.1.0",
158
- servers: config.servers ?? [
159
- {
160
- url: "https://github.com/samchon/nestia",
161
- description: "insert your server url",
162
- },
163
- ],
164
- info: {
165
- ...(config.info ?? {}),
166
- version: config.info?.version ?? (await pack.get())?.version ?? "0.1.0",
167
- title:
168
- config.info?.title ??
169
- (await pack.get())?.title ??
170
- "Swagger Documents",
171
- description:
172
- config.info?.description ??
173
- (await pack.get())?.description ??
174
- "Generated by nestia - https://github.com/samchon/nestia",
175
- license: config.info?.license ?? (await pack.get())?.license,
176
- },
177
- paths: {},
178
- components: {
179
- schemas: {},
180
- securitySchemes: config.security,
181
- },
182
- tags: config.tags ?? [],
183
- "x-samchon-emended-v4": true,
184
- };
185
- };
186
-
187
- const fillPaths = (props: {
188
- config: Omit<INestiaConfig.ISwaggerConfig, "output">;
189
- document: OpenApi.IDocument;
190
- schema: (metadata: Metadata) => OpenApi.IJsonSchema | undefined;
191
- routes: ITypedHttpRoute[];
192
- }): void => {
193
- // SWAGGER CUSTOMIZER
194
- const customizers: Array<() => void> = [];
195
- const neighbor = {
196
- at: new Singleton(() => {
197
- const functor: Map<Function, Endpoint> = new Map();
198
- for (const r of props.routes) {
199
- const method: OpenApi.Method =
200
- r.method.toLowerCase() as OpenApi.Method;
201
- const path: string = getPath(r);
202
- const operation: OpenApi.IOperation | undefined =
203
- props.document.paths?.[path]?.[method];
204
- if (operation === undefined) continue;
205
- functor.set(r.function, {
206
- method,
207
- path,
208
- route: operation,
209
- });
210
- }
211
- return functor;
212
- }),
213
- get: new Singleton(
214
- () =>
215
- (key: Accessor): OpenApi.IOperation | undefined => {
216
- const method: OpenApi.Method =
217
- key.method.toLowerCase() as OpenApi.Method;
218
- const path: string =
219
- "/" +
220
- key.path
221
- .split("/")
222
- .filter((str) => !!str.length)
223
- .map((str) =>
224
- str.startsWith(":") ? `{${str.substring(1)}}` : str,
225
- )
226
- .join("/");
227
- return props.document.paths?.[path]?.[method];
228
- },
229
- ),
230
- };
231
-
232
- // COMPOSE OPERATIONS
233
- for (const r of props.routes) {
234
- const operation: OpenApi.IOperation = SwaggerOperationComposer.compose({
235
- ...props,
236
- route: r,
237
- });
238
- const path: string = getPath(r);
239
- props.document.paths ??= {};
240
- props.document.paths[path] ??= {};
241
- props.document.paths[path][r.method.toLowerCase() as "get"] = operation;
242
-
243
- const closure: Function | Function[] | undefined = Reflect.getMetadata(
244
- "nestia/SwaggerCustomizer",
245
- r.controller.class.prototype,
246
- r.name,
247
- );
248
- if (closure !== undefined) {
249
- const array: Function[] = Array.isArray(closure) ? closure : [closure];
250
- customizers.push(() => {
251
- for (const closure of array)
252
- closure({
253
- swagger: props.document,
254
- method: r.method,
255
- path,
256
- route: operation,
257
- at: (func: Function) => neighbor.at.get().get(func),
258
- get: (accessor: Accessor) => neighbor.get.get()(accessor),
259
- } satisfies SwaggerCustomizer.IProps);
260
- });
261
- }
262
- }
263
-
264
- // DO CUSTOMIZE
265
- for (const fn of customizers) fn();
266
- };
267
-
268
- const getPath = (route: ITypedHttpRoute): string => {
269
- let str: string = route.path;
270
- for (const param of route.pathParameters)
271
- str = str.replace(`:${param.field}`, `{${param.field}}`);
272
- return str;
273
- };
274
- }
275
-
276
- interface Accessor {
277
- method: string;
278
- path: string;
279
- }
280
- interface Endpoint {
281
- method: string;
282
- path: string;
283
- route: OpenApi.IOperation;
284
- }
1
+ import { SwaggerCustomizer } from "@nestia/core";
2
+ import { JsonSchemasProgrammer, MetadataSchema } from "@typia/core";
3
+ import {
4
+ OpenApi,
5
+ OpenApiV3,
6
+ OpenApiV3_1,
7
+ OpenApiV3_2,
8
+ SwaggerV2,
9
+ } from "@typia/interface";
10
+ import { OpenApiConverter } from "@typia/utils";
11
+ import fs from "fs";
12
+ import path from "path";
13
+ import { Singleton } from "tstl";
14
+ import typia, { IJsonSchemaCollection } from "typia";
15
+
16
+ import { INestiaConfig } from "../INestiaConfig";
17
+ import { ITypedApplication } from "../structures/ITypedApplication";
18
+ import { ITypedHttpRoute } from "../structures/ITypedHttpRoute";
19
+ import { FileRetriever } from "../utils/FileRetriever";
20
+ import { SdkHttpParameterProgrammer } from "./internal/SdkHttpParameterProgrammer";
21
+ import { SwaggerOperationComposer } from "./internal/SwaggerOperationComposer";
22
+
23
+ export namespace SwaggerGenerator {
24
+ export const generate = async (app: ITypedApplication): Promise<void> => {
25
+ // GET CONFIGURATION
26
+ console.log("Generating Swagger Document");
27
+ if (app.project.config.swagger === undefined)
28
+ throw new Error("Swagger configuration is not defined.");
29
+ const config: INestiaConfig.ISwaggerConfig = app.project.config.swagger;
30
+
31
+ // TARGET LOCATION
32
+ const parsed: path.ParsedPath = path.parse(config.output);
33
+ const directory: string = path.dirname(parsed.dir);
34
+ if (fs.existsSync(directory) === false)
35
+ try {
36
+ await fs.promises.mkdir(directory);
37
+ } catch {}
38
+ if (fs.existsSync(directory) === false)
39
+ throw new Error(
40
+ `Error on NestiaApplication.swagger(): failed to create output directory: ${directory}`,
41
+ );
42
+ const location: string = !!parsed.ext
43
+ ? path.resolve(config.output)
44
+ : path.join(path.resolve(config.output), "swagger.json");
45
+
46
+ // COMPOSE SWAGGER DOCUMENT
47
+ const document: OpenApi.IDocument = compose({
48
+ config,
49
+ routes: app.routes.filter((route) => route.protocol === "http"),
50
+ document: await initialize(config),
51
+ });
52
+ const specified:
53
+ | OpenApi.IDocument
54
+ | SwaggerV2.IDocument
55
+ | OpenApiV3.IDocument
56
+ | OpenApiV3_1.IDocument
57
+ | OpenApiV3_2.IDocument =
58
+ (config.openapi ?? "3.2") === "3.2"
59
+ ? document
60
+ : OpenApiConverter.downgradeDocument(document, config.openapi as "2.0");
61
+ await fs.promises.writeFile(
62
+ location,
63
+ !config.beautify
64
+ ? JSON.stringify(specified)
65
+ : JSON.stringify(
66
+ specified,
67
+ null,
68
+ typeof config.beautify === "number" ? config.beautify : 2,
69
+ ),
70
+ "utf8",
71
+ );
72
+ };
73
+
74
+ export const compose = (props: {
75
+ config: Omit<INestiaConfig.ISwaggerConfig, "output">;
76
+ routes: ITypedHttpRoute[];
77
+ document: OpenApi.IDocument;
78
+ }): OpenApi.IDocument => {
79
+ // GATHER METADATA
80
+ const routes: ITypedHttpRoute[] = props.routes.filter((r) =>
81
+ r.jsDocTags.every(
82
+ (tag) => tag.name !== "internal" && tag.name !== "hidden",
83
+ ),
84
+ );
85
+ const metadatas: MetadataSchema[] = routes
86
+ .map((r) => [
87
+ r.success.metadata,
88
+ ...SdkHttpParameterProgrammer.getAll(r).map((p) => p.metadata),
89
+ ...Object.values(r.exceptions).map((e) => e.metadata),
90
+ ])
91
+ .flat()
92
+ .filter((m) => m.size() !== 0);
93
+
94
+ // COMPOSE JSON SCHEMAS
95
+ const json: IJsonSchemaCollection = JsonSchemasProgrammer.writeSchemas({
96
+ version: "3.1",
97
+ metadatas,
98
+ });
99
+ const dict: WeakMap<MetadataSchema, OpenApi.IJsonSchema> = new WeakMap();
100
+ json.schemas.forEach((schema, i) => dict.set(metadatas[i]!, schema));
101
+ const schema = (
102
+ metadata: MetadataSchema,
103
+ ): OpenApi.IJsonSchema | undefined => dict.get(metadata);
104
+
105
+ // COMPOSE DOCUMENT
106
+ const document: OpenApi.IDocument = props.document;
107
+ document.components.schemas ??= {};
108
+ Object.assign(document.components.schemas, json.components.schemas);
109
+ fillPaths({
110
+ ...props,
111
+ routes,
112
+ schema,
113
+ document,
114
+ });
115
+ return document;
116
+ };
117
+
118
+ export const initialize = async (
119
+ config: Omit<INestiaConfig.ISwaggerConfig, "output">,
120
+ ): Promise<OpenApi.IDocument> => {
121
+ const pack = new Singleton(
122
+ async (): Promise<Partial<OpenApi.IDocument.IInfo> | null> => {
123
+ const location: string | null = await FileRetriever.file(
124
+ "package.json",
125
+ )(process.cwd());
126
+ if (location === null) return null;
127
+
128
+ try {
129
+ const content: string = await fs.promises.readFile(location, "utf8");
130
+ const data = typia.json.assertParse<{
131
+ name?: string;
132
+ version?: string;
133
+ description?: string;
134
+ license?:
135
+ | string
136
+ | {
137
+ type: string;
138
+ /** @format uri */
139
+ url: string;
140
+ };
141
+ }>(content);
142
+ return {
143
+ title: data.name,
144
+ version: data.version,
145
+ description: data.description,
146
+ license: data.license
147
+ ? typeof data.license === "string"
148
+ ? { name: data.license }
149
+ : typeof data.license === "object"
150
+ ? {
151
+ name: data.license.type,
152
+ url: data.license.url,
153
+ }
154
+ : undefined
155
+ : undefined,
156
+ };
157
+ } catch {
158
+ return null;
159
+ }
160
+ },
161
+ );
162
+
163
+ return {
164
+ openapi: "3.2.0",
165
+ servers: config.servers ?? [
166
+ {
167
+ url: "https://github.com/samchon/nestia",
168
+ description: "insert your server url",
169
+ },
170
+ ],
171
+ info: {
172
+ ...(config.info ?? {}),
173
+ version: config.info?.version ?? (await pack.get())?.version ?? "0.1.0",
174
+ title:
175
+ config.info?.title ??
176
+ (await pack.get())?.title ??
177
+ "Swagger Documents",
178
+ description:
179
+ config.info?.description ??
180
+ (await pack.get())?.description ??
181
+ "Generated by nestia - https://github.com/samchon/nestia",
182
+ license: config.info?.license ?? (await pack.get())?.license,
183
+ },
184
+ paths: {},
185
+ components: {
186
+ schemas: {},
187
+ securitySchemes: config.security,
188
+ },
189
+ tags: config.tags ?? [],
190
+ "x-typia-emended-v12": true,
191
+ };
192
+ };
193
+
194
+ const fillPaths = (props: {
195
+ config: Omit<INestiaConfig.ISwaggerConfig, "output">;
196
+ document: OpenApi.IDocument;
197
+ schema: (metadata: MetadataSchema) => OpenApi.IJsonSchema | undefined;
198
+ routes: ITypedHttpRoute[];
199
+ }): void => {
200
+ // SWAGGER CUSTOMIZER
201
+ const customizers: Array<() => void> = [];
202
+ const neighbor = {
203
+ at: new Singleton(() => {
204
+ const functor: Map<Function, Endpoint> = new Map();
205
+ for (const r of props.routes) {
206
+ const method: OpenApi.Method =
207
+ r.method.toLowerCase() as OpenApi.Method;
208
+ const path: string = getPath(r);
209
+ const operation: OpenApi.IOperation | undefined =
210
+ props.document.paths?.[path]?.[method];
211
+ if (operation === undefined) continue;
212
+ functor.set(r.function, {
213
+ method,
214
+ path,
215
+ route: operation,
216
+ });
217
+ }
218
+ return functor;
219
+ }),
220
+ get: new Singleton(
221
+ () =>
222
+ (key: Accessor): OpenApi.IOperation | undefined => {
223
+ const method: OpenApi.Method =
224
+ key.method.toLowerCase() as OpenApi.Method;
225
+ const path: string =
226
+ "/" +
227
+ key.path
228
+ .split("/")
229
+ .filter((str) => !!str.length)
230
+ .map((str) =>
231
+ str.startsWith(":") ? `{${str.substring(1)}}` : str,
232
+ )
233
+ .join("/");
234
+ return props.document.paths?.[path]?.[method];
235
+ },
236
+ ),
237
+ };
238
+
239
+ // COMPOSE OPERATIONS
240
+ for (const r of props.routes) {
241
+ const operation: OpenApi.IOperation = SwaggerOperationComposer.compose({
242
+ ...props,
243
+ route: r,
244
+ });
245
+ const path: string = getPath(r);
246
+ props.document.paths ??= {};
247
+ props.document.paths[path] ??= {};
248
+ props.document.paths[path][r.method.toLowerCase() as "get"] = operation;
249
+
250
+ const closure: Function | Function[] | undefined = Reflect.getMetadata(
251
+ "nestia/SwaggerCustomizer",
252
+ r.controller.class.prototype,
253
+ r.name,
254
+ );
255
+ if (closure !== undefined) {
256
+ const array: Function[] = Array.isArray(closure) ? closure : [closure];
257
+ customizers.push(() => {
258
+ for (const closure of array)
259
+ closure({
260
+ swagger: props.document,
261
+ method: r.method,
262
+ path,
263
+ route: operation,
264
+ at: (func: Function) => neighbor.at.get().get(func),
265
+ get: (accessor: Accessor) => neighbor.get.get()(accessor),
266
+ } satisfies SwaggerCustomizer.IProps);
267
+ });
268
+ }
269
+ }
270
+
271
+ // DO CUSTOMIZE
272
+ for (const fn of customizers) fn();
273
+ };
274
+
275
+ const getPath = (route: ITypedHttpRoute): string => {
276
+ let str: string = route.path;
277
+ for (const param of route.pathParameters)
278
+ str = str.replace(`:${param.field}`, `{${param.field}}`);
279
+ return str;
280
+ };
281
+ }
282
+
283
+ interface Accessor {
284
+ method: string;
285
+ path: string;
286
+ }
287
+ interface Endpoint {
288
+ method: string;
289
+ path: string;
290
+ route: OpenApi.IOperation;
291
+ }