@nestia/sdk 11.0.0-dev.20260316 → 11.0.1

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 (65) 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/executable/internal/NestiaConfigLoader.js +5 -5
  15. package/lib/executable/internal/NestiaConfigLoader.js.map +1 -1
  16. package/package.json +8 -8
  17. package/src/INestiaConfig.ts +267 -267
  18. package/src/NestiaSdkApplication.ts +307 -307
  19. package/src/NestiaSwaggerComposer.ts +143 -143
  20. package/src/analyses/AccessorAnalyzer.ts +67 -67
  21. package/src/analyses/DtoAnalyzer.ts +260 -260
  22. package/src/analyses/ImportAnalyzer.ts +126 -126
  23. package/src/analyses/ReflectHttpOperationAnalyzer.ts +183 -183
  24. package/src/analyses/ReflectHttpOperationExceptionAnalyzer.ts +72 -72
  25. package/src/analyses/ReflectHttpOperationParameterAnalyzer.ts +350 -350
  26. package/src/analyses/ReflectHttpOperationResponseAnalyzer.ts +126 -126
  27. package/src/analyses/TypedHttpRouteAnalyzer.ts +208 -208
  28. package/src/executable/internal/NestiaConfigLoader.ts +85 -85
  29. package/src/executable/internal/NestiaSdkCommand.ts +107 -107
  30. package/src/generates/SwaggerGenerator.ts +291 -291
  31. package/src/generates/internal/E2eFileProgrammer.ts +196 -196
  32. package/src/generates/internal/FilePrinter.ts +64 -64
  33. package/src/generates/internal/ImportDictionary.ts +192 -192
  34. package/src/generates/internal/SdkAliasCollection.ts +260 -260
  35. package/src/generates/internal/SdkFileProgrammer.ts +110 -110
  36. package/src/generates/internal/SdkHttpCloneProgrammer.ts +126 -126
  37. package/src/generates/internal/SdkHttpCloneReferencer.ts +77 -77
  38. package/src/generates/internal/SdkHttpFunctionProgrammer.ts +278 -278
  39. package/src/generates/internal/SdkHttpNamespaceProgrammer.ts +502 -502
  40. package/src/generates/internal/SdkHttpRouteProgrammer.ts +109 -109
  41. package/src/generates/internal/SdkHttpSimulationProgrammer.ts +312 -312
  42. package/src/generates/internal/SdkImportWizard.ts +62 -62
  43. package/src/generates/internal/SdkTypeProgrammer.ts +388 -388
  44. package/src/generates/internal/SdkTypeTagProgrammer.ts +114 -114
  45. package/src/generates/internal/SdkWebSocketNamespaceProgrammer.ts +379 -379
  46. package/src/generates/internal/SdkWebSocketRouteProgrammer.ts +302 -302
  47. package/src/generates/internal/SwaggerOperationComposer.ts +119 -119
  48. package/src/generates/internal/SwaggerOperationParameterComposer.ts +161 -161
  49. package/src/generates/internal/SwaggerOperationResponseComposer.ts +110 -110
  50. package/src/module.ts +4 -4
  51. package/src/structures/IReflectHttpOperationException.ts +18 -18
  52. package/src/structures/IReflectHttpOperationParameter.ts +79 -79
  53. package/src/structures/IReflectHttpOperationSuccess.ts +21 -21
  54. package/src/structures/ITypedApplication.ts +11 -11
  55. package/src/structures/ITypedHttpRouteException.ts +15 -15
  56. package/src/structures/ITypedHttpRouteParameter.ts +41 -41
  57. package/src/structures/ITypedHttpRouteSuccess.ts +22 -22
  58. package/src/transformers/IOperationMetadata.ts +46 -46
  59. package/src/transformers/ISdkOperationTransformerContext.ts +8 -8
  60. package/src/transformers/SdkOperationProgrammer.ts +240 -240
  61. package/src/transformers/SdkOperationTransformer.ts +248 -248
  62. package/src/transformers/TextPlainValidator.ts +17 -17
  63. package/src/utils/MetadataUtil.ts +26 -26
  64. package/src/validators/HttpHeadersValidator.ts +40 -40
  65. package/src/validators/HttpQueryValidator.ts +40 -40
@@ -1,350 +1,350 @@
1
- import { SwaggerExample } from "@nestia/core";
2
- import { ROUTE_ARGS_METADATA } from "@nestjs/common/constants";
3
- import { RouteParamtypes } from "@nestjs/common/enums/route-paramtypes.enum";
4
- import {
5
- HttpFormDataProgrammer,
6
- HttpHeadersProgrammer,
7
- HttpParameterProgrammer,
8
- HttpQueryProgrammer,
9
- JsonMetadataFactory,
10
- } from "@typia/core";
11
-
12
- import { IReflectController } from "../structures/IReflectController";
13
- import { IReflectHttpOperationParameter } from "../structures/IReflectHttpOperationParameter";
14
- import { IReflectOperationError } from "../structures/IReflectOperationError";
15
- import { IOperationMetadata } from "../transformers/IOperationMetadata";
16
- import { TextPlainValidator } from "../transformers/TextPlainValidator";
17
- import { HttpHeadersValidator } from "../validators/HttpHeadersValidator";
18
- import { HttpQueryValidator } from "../validators/HttpQueryValidator";
19
-
20
- export namespace ReflectHttpOperationParameterAnalyzer {
21
- export interface IContext {
22
- controller: IReflectController;
23
- function: Function;
24
- functionName: string;
25
- httpMethod: string;
26
- metadata: IOperationMetadata;
27
- errors: IReflectOperationError[];
28
- }
29
- export const analyze = (ctx: IContext): IReflectHttpOperationParameter[] => {
30
- const preconfigured: IReflectHttpOperationParameter.IPreconfigured[] =
31
- analyzePreconfigured(ctx);
32
- const errors: IReflectOperationError[] = [];
33
-
34
- //----
35
- // FIND CONTRADICTIONS
36
- //----
37
- // GET AND HEAD METHOD
38
- const contradictErrors: string[] = [];
39
- const contradict = (message: string) => {
40
- contradictErrors.push(message);
41
- };
42
- if (
43
- (ctx.httpMethod === "GET" || ctx.httpMethod === "HEAD") &&
44
- preconfigured.some((x) => x.category === "body")
45
- )
46
- contradict(`@Body() is not allowed in the ${ctx.httpMethod} method.`);
47
-
48
- // FIND DUPLICATED BODY
49
- if (
50
- preconfigured.filter(
51
- (x) => x.category === "body" && x.field === undefined,
52
- ).length > 1
53
- )
54
- contradict(`Duplicated @Body() is not allowed.`);
55
- if (
56
- preconfigured.filter(
57
- (x) => x.category === "query" && x.field === undefined,
58
- ).length > 1
59
- )
60
- contradict(`Duplicated @Query() without field name is not allowed.`);
61
- if (
62
- preconfigured.filter(
63
- (x) => x.category === "headers" && x.field === undefined,
64
- ).length > 1
65
- )
66
- contradict(`Duplicated @Headers() without field name is not allowed.`);
67
-
68
- // FIND DUPLICATED FIELDS
69
- if (
70
- isUnique(
71
- preconfigured
72
- .filter((x) => x.category === "param")
73
- .map((x) => x.field)
74
- .filter((field) => field !== undefined),
75
- ) === false
76
- )
77
- contradict(`Duplicated field names of path are not allowed.`);
78
- if (
79
- isUnique(
80
- preconfigured
81
- .filter((x) => x.category === "query")
82
- .map((x) => x.field)
83
- .filter((field) => field !== undefined),
84
- ) === false
85
- )
86
- contradict(`Duplicated field names of query are not allowed.`);
87
- if (
88
- isUnique(
89
- preconfigured
90
- .filter((x) => x.category === "headers")
91
- .map((x) => x.field)
92
- .filter((field) => field !== undefined),
93
- ) === false
94
- )
95
- contradict(`Duplicated field names of headers are not allowed.`);
96
- if (contradictErrors.length)
97
- errors.push({
98
- file: ctx.controller.file,
99
- class: ctx.controller.class.name,
100
- function: ctx.functionName,
101
- from: "",
102
- contents: contradictErrors,
103
- });
104
-
105
- //----
106
- // COMPOSE PARAMETERS
107
- //----
108
- const parameters: IReflectHttpOperationParameter[] = preconfigured
109
- .map((p): IReflectHttpOperationParameter | null => {
110
- // METADATA INFO
111
- const pErrorContents: Array<string | IOperationMetadata.IError> = [];
112
- const matched: IOperationMetadata.IParameter | undefined =
113
- ctx.metadata.parameters.find((x) => x.index === p.index);
114
- const report = () => {
115
- errors.push({
116
- file: ctx.controller.file,
117
- class: ctx.controller.class.name,
118
- function: ctx.functionName,
119
- from: `parameter ${matched ? JSON.stringify(matched.name) : `of ${p.index} th`}`,
120
- contents: pErrorContents,
121
- });
122
- return null;
123
- };
124
-
125
- // VALIDATE TYPE
126
- if (matched === undefined)
127
- pErrorContents.push(`Unable to find parameter type.`);
128
- else if (matched.type === null)
129
- pErrorContents.push(`Failed to get the type info.`);
130
-
131
- // CONSIDER KIND
132
- const schema: IOperationMetadata.ISchema | null = (() => {
133
- if (matched === undefined) return null;
134
- const result =
135
- p.category === "body" &&
136
- (p.contentType === "application/json" || p.encrypted === true)
137
- ? matched.primitive
138
- : matched.resolved;
139
- return result.success ? result.data : null;
140
- })();
141
- if (p.category === "body" && p.field !== undefined)
142
- pErrorContents.push(`@Body() must not have a field name.`);
143
- else if (p.category === "param" && p.field === undefined)
144
- pErrorContents.push(`@Param() must have a field name.`);
145
-
146
- if (pErrorContents.length) return report();
147
- else if (
148
- matched === undefined ||
149
- matched.type === null ||
150
- schema === null
151
- )
152
- return null; // unreachable
153
-
154
- const example: SwaggerExample.IData<any> | undefined = (
155
- Reflect.getMetadata(
156
- "nestia/SwaggerExample/Parameters",
157
- ctx.controller.class.prototype,
158
- ctx.functionName,
159
- ) ?? []
160
- ).find((x: SwaggerExample.IData<any>) => x.index === matched.index);
161
-
162
- // COMPOSITION
163
- if (p.category === "param")
164
- return {
165
- category: p.category,
166
- index: p.index,
167
- field: p.field!,
168
- name: matched.name,
169
- type: matched.type,
170
- validate: HttpParameterProgrammer.validate,
171
- description: matched.description,
172
- jsDocTags: matched.jsDocTags,
173
- example: example?.example,
174
- examples: example?.examples,
175
- ...schema,
176
- };
177
- else if (p.category === "query")
178
- return {
179
- category: p.category,
180
- index: p.index,
181
- field: p.field ?? null,
182
- name: matched.name,
183
- type: matched.type,
184
- validate: p.field
185
- ? HttpQueryValidator.validate
186
- : HttpQueryProgrammer.validate,
187
- description: matched.description,
188
- jsDocTags: matched.jsDocTags,
189
- example: example?.example,
190
- examples: example?.examples,
191
- ...schema,
192
- };
193
- else if (p.category === "headers")
194
- return {
195
- category: p.category,
196
- index: p.index,
197
- field: p.field ?? null,
198
- name: matched.name,
199
- type: matched.type,
200
- validate: p.field
201
- ? HttpHeadersValidator.validate
202
- : HttpHeadersProgrammer.validate,
203
- description: matched.description,
204
- jsDocTags: matched.jsDocTags,
205
- example: example?.example,
206
- examples: example?.examples,
207
- ...schema,
208
- };
209
- else if (p.category === "body")
210
- return {
211
- category: p.category,
212
- index: p.index,
213
- encrypted: !!p.encrypted,
214
- contentType: p.contentType,
215
- name: matched.name,
216
- type: matched.type,
217
- validate:
218
- p.contentType === "application/json" || p.encrypted === true
219
- ? JsonMetadataFactory.validate
220
- : p.contentType === "application/x-www-form-urlencoded"
221
- ? HttpQueryProgrammer.validate
222
- : p.contentType === "multipart/form-data"
223
- ? HttpFormDataProgrammer.validate
224
- : TextPlainValidator.validate,
225
- description: matched.description,
226
- jsDocTags: matched.jsDocTags,
227
- example: example?.example,
228
- examples: example?.examples,
229
- ...schema,
230
- };
231
- else {
232
- pErrorContents.push(`Unknown kind of the parameter.`);
233
- return report();
234
- }
235
- })
236
- .filter((x): x is IReflectHttpOperationParameter => x !== null);
237
-
238
- if (errors.length) ctx.errors.push(...errors);
239
- return parameters;
240
- };
241
-
242
- const analyzePreconfigured = (
243
- props: IContext,
244
- ): IReflectHttpOperationParameter.IPreconfigured[] => {
245
- const dict: NestParameters | undefined = Reflect.getMetadata(
246
- ROUTE_ARGS_METADATA,
247
- props.controller.class,
248
- props.functionName,
249
- );
250
- if (dict === undefined) return [];
251
- return Object.entries(dict)
252
- .map(([key, param]) => analyzeHttpParameter(key, param))
253
- .filter(
254
- (x): x is IReflectHttpOperationParameter.IPreconfigured => x !== null,
255
- )
256
- .sort((x, y) => x.index - y.index);
257
- };
258
-
259
- const analyzeHttpParameter = (
260
- key: string,
261
- param: INestParam,
262
- ): IReflectHttpOperationParameter.IPreconfigured | null => {
263
- const symbol: string = key.split(":")[0]!;
264
- if (symbol.indexOf("__custom") !== -1) return analyzeCustomParameter(param);
265
-
266
- const category:
267
- | IReflectHttpOperationParameter.IPreconfigured["category"]
268
- | null = getNestParamType(Number(symbol[0]) as RouteParamtypes);
269
- if (category === null) return null;
270
- if (category === "body")
271
- return {
272
- category: "body",
273
- index: param.index,
274
- field: param.data,
275
- contentType: "application/json",
276
- };
277
- else
278
- return {
279
- category,
280
- index: param.index,
281
- field: param.data,
282
- };
283
- };
284
-
285
- const analyzeCustomParameter = (
286
- param: INestParam,
287
- ): IReflectHttpOperationParameter.IPreconfigured | null => {
288
- if (param.factory === undefined) return null;
289
- else if (
290
- param.factory.name === "EncryptedBody" ||
291
- param.factory.name === "PlainBody" ||
292
- param.factory.name === "TypedQueryBody" ||
293
- param.factory.name === "TypedBody" ||
294
- param.factory.name === "TypedFormDataBody"
295
- )
296
- return {
297
- category: "body",
298
- index: param.index,
299
- encrypted: param.factory.name === "EncryptedBody",
300
- contentType:
301
- param.factory.name === "PlainBody" ||
302
- param.factory.name === "EncryptedBody"
303
- ? "text/plain"
304
- : param.factory.name === "TypedQueryBody"
305
- ? "application/x-www-form-urlencoded"
306
- : param.factory.name === "TypedFormDataBody"
307
- ? "multipart/form-data"
308
- : "application/json",
309
- };
310
- else if (param.factory.name === "TypedHeaders")
311
- return {
312
- category: "headers",
313
- index: param.index,
314
- field: param.data,
315
- };
316
- else if (param.factory.name === "TypedParam")
317
- return {
318
- category: "param",
319
- index: param.index,
320
- field: param.data,
321
- };
322
- else if (param.factory.name === "TypedQuery")
323
- return {
324
- category: "query",
325
- index: param.index,
326
- field: undefined,
327
- };
328
- else return null;
329
- };
330
-
331
- const isUnique = (values: string[]) => new Set(values).size === values.length;
332
- }
333
-
334
- type NestParameters = {
335
- [key: string]: INestParam;
336
- };
337
- interface INestParam {
338
- name: string;
339
- index: number;
340
- factory?: (...args: any) => any;
341
- data: string | undefined;
342
- }
343
-
344
- const getNestParamType = (value: RouteParamtypes) => {
345
- if (value === RouteParamtypes.BODY) return "body";
346
- else if (value === RouteParamtypes.HEADERS) return "headers";
347
- else if (value === RouteParamtypes.QUERY) return "query";
348
- else if (value === RouteParamtypes.PARAM) return "param";
349
- return null;
350
- };
1
+ import { SwaggerExample } from "@nestia/core";
2
+ import { ROUTE_ARGS_METADATA } from "@nestjs/common/constants";
3
+ import { RouteParamtypes } from "@nestjs/common/enums/route-paramtypes.enum";
4
+ import {
5
+ HttpFormDataProgrammer,
6
+ HttpHeadersProgrammer,
7
+ HttpParameterProgrammer,
8
+ HttpQueryProgrammer,
9
+ JsonMetadataFactory,
10
+ } from "@typia/core";
11
+
12
+ import { IReflectController } from "../structures/IReflectController";
13
+ import { IReflectHttpOperationParameter } from "../structures/IReflectHttpOperationParameter";
14
+ import { IReflectOperationError } from "../structures/IReflectOperationError";
15
+ import { IOperationMetadata } from "../transformers/IOperationMetadata";
16
+ import { TextPlainValidator } from "../transformers/TextPlainValidator";
17
+ import { HttpHeadersValidator } from "../validators/HttpHeadersValidator";
18
+ import { HttpQueryValidator } from "../validators/HttpQueryValidator";
19
+
20
+ export namespace ReflectHttpOperationParameterAnalyzer {
21
+ export interface IContext {
22
+ controller: IReflectController;
23
+ function: Function;
24
+ functionName: string;
25
+ httpMethod: string;
26
+ metadata: IOperationMetadata;
27
+ errors: IReflectOperationError[];
28
+ }
29
+ export const analyze = (ctx: IContext): IReflectHttpOperationParameter[] => {
30
+ const preconfigured: IReflectHttpOperationParameter.IPreconfigured[] =
31
+ analyzePreconfigured(ctx);
32
+ const errors: IReflectOperationError[] = [];
33
+
34
+ //----
35
+ // FIND CONTRADICTIONS
36
+ //----
37
+ // GET AND HEAD METHOD
38
+ const contradictErrors: string[] = [];
39
+ const contradict = (message: string) => {
40
+ contradictErrors.push(message);
41
+ };
42
+ if (
43
+ (ctx.httpMethod === "GET" || ctx.httpMethod === "HEAD") &&
44
+ preconfigured.some((x) => x.category === "body")
45
+ )
46
+ contradict(`@Body() is not allowed in the ${ctx.httpMethod} method.`);
47
+
48
+ // FIND DUPLICATED BODY
49
+ if (
50
+ preconfigured.filter(
51
+ (x) => x.category === "body" && x.field === undefined,
52
+ ).length > 1
53
+ )
54
+ contradict(`Duplicated @Body() is not allowed.`);
55
+ if (
56
+ preconfigured.filter(
57
+ (x) => x.category === "query" && x.field === undefined,
58
+ ).length > 1
59
+ )
60
+ contradict(`Duplicated @Query() without field name is not allowed.`);
61
+ if (
62
+ preconfigured.filter(
63
+ (x) => x.category === "headers" && x.field === undefined,
64
+ ).length > 1
65
+ )
66
+ contradict(`Duplicated @Headers() without field name is not allowed.`);
67
+
68
+ // FIND DUPLICATED FIELDS
69
+ if (
70
+ isUnique(
71
+ preconfigured
72
+ .filter((x) => x.category === "param")
73
+ .map((x) => x.field)
74
+ .filter((field) => field !== undefined),
75
+ ) === false
76
+ )
77
+ contradict(`Duplicated field names of path are not allowed.`);
78
+ if (
79
+ isUnique(
80
+ preconfigured
81
+ .filter((x) => x.category === "query")
82
+ .map((x) => x.field)
83
+ .filter((field) => field !== undefined),
84
+ ) === false
85
+ )
86
+ contradict(`Duplicated field names of query are not allowed.`);
87
+ if (
88
+ isUnique(
89
+ preconfigured
90
+ .filter((x) => x.category === "headers")
91
+ .map((x) => x.field)
92
+ .filter((field) => field !== undefined),
93
+ ) === false
94
+ )
95
+ contradict(`Duplicated field names of headers are not allowed.`);
96
+ if (contradictErrors.length)
97
+ errors.push({
98
+ file: ctx.controller.file,
99
+ class: ctx.controller.class.name,
100
+ function: ctx.functionName,
101
+ from: "",
102
+ contents: contradictErrors,
103
+ });
104
+
105
+ //----
106
+ // COMPOSE PARAMETERS
107
+ //----
108
+ const parameters: IReflectHttpOperationParameter[] = preconfigured
109
+ .map((p): IReflectHttpOperationParameter | null => {
110
+ // METADATA INFO
111
+ const pErrorContents: Array<string | IOperationMetadata.IError> = [];
112
+ const matched: IOperationMetadata.IParameter | undefined =
113
+ ctx.metadata.parameters.find((x) => x.index === p.index);
114
+ const report = () => {
115
+ errors.push({
116
+ file: ctx.controller.file,
117
+ class: ctx.controller.class.name,
118
+ function: ctx.functionName,
119
+ from: `parameter ${matched ? JSON.stringify(matched.name) : `of ${p.index} th`}`,
120
+ contents: pErrorContents,
121
+ });
122
+ return null;
123
+ };
124
+
125
+ // VALIDATE TYPE
126
+ if (matched === undefined)
127
+ pErrorContents.push(`Unable to find parameter type.`);
128
+ else if (matched.type === null)
129
+ pErrorContents.push(`Failed to get the type info.`);
130
+
131
+ // CONSIDER KIND
132
+ const schema: IOperationMetadata.ISchema | null = (() => {
133
+ if (matched === undefined) return null;
134
+ const result =
135
+ p.category === "body" &&
136
+ (p.contentType === "application/json" || p.encrypted === true)
137
+ ? matched.primitive
138
+ : matched.resolved;
139
+ return result.success ? result.data : null;
140
+ })();
141
+ if (p.category === "body" && p.field !== undefined)
142
+ pErrorContents.push(`@Body() must not have a field name.`);
143
+ else if (p.category === "param" && p.field === undefined)
144
+ pErrorContents.push(`@Param() must have a field name.`);
145
+
146
+ if (pErrorContents.length) return report();
147
+ else if (
148
+ matched === undefined ||
149
+ matched.type === null ||
150
+ schema === null
151
+ )
152
+ return null; // unreachable
153
+
154
+ const example: SwaggerExample.IData<any> | undefined = (
155
+ Reflect.getMetadata(
156
+ "nestia/SwaggerExample/Parameters",
157
+ ctx.controller.class.prototype,
158
+ ctx.functionName,
159
+ ) ?? []
160
+ ).find((x: SwaggerExample.IData<any>) => x.index === matched.index);
161
+
162
+ // COMPOSITION
163
+ if (p.category === "param")
164
+ return {
165
+ category: p.category,
166
+ index: p.index,
167
+ field: p.field!,
168
+ name: matched.name,
169
+ type: matched.type,
170
+ validate: HttpParameterProgrammer.validate,
171
+ description: matched.description,
172
+ jsDocTags: matched.jsDocTags,
173
+ example: example?.example,
174
+ examples: example?.examples,
175
+ ...schema,
176
+ };
177
+ else if (p.category === "query")
178
+ return {
179
+ category: p.category,
180
+ index: p.index,
181
+ field: p.field ?? null,
182
+ name: matched.name,
183
+ type: matched.type,
184
+ validate: p.field
185
+ ? HttpQueryValidator.validate
186
+ : HttpQueryProgrammer.validate,
187
+ description: matched.description,
188
+ jsDocTags: matched.jsDocTags,
189
+ example: example?.example,
190
+ examples: example?.examples,
191
+ ...schema,
192
+ };
193
+ else if (p.category === "headers")
194
+ return {
195
+ category: p.category,
196
+ index: p.index,
197
+ field: p.field ?? null,
198
+ name: matched.name,
199
+ type: matched.type,
200
+ validate: p.field
201
+ ? HttpHeadersValidator.validate
202
+ : HttpHeadersProgrammer.validate,
203
+ description: matched.description,
204
+ jsDocTags: matched.jsDocTags,
205
+ example: example?.example,
206
+ examples: example?.examples,
207
+ ...schema,
208
+ };
209
+ else if (p.category === "body")
210
+ return {
211
+ category: p.category,
212
+ index: p.index,
213
+ encrypted: !!p.encrypted,
214
+ contentType: p.contentType,
215
+ name: matched.name,
216
+ type: matched.type,
217
+ validate:
218
+ p.contentType === "application/json" || p.encrypted === true
219
+ ? JsonMetadataFactory.validate
220
+ : p.contentType === "application/x-www-form-urlencoded"
221
+ ? HttpQueryProgrammer.validate
222
+ : p.contentType === "multipart/form-data"
223
+ ? HttpFormDataProgrammer.validate
224
+ : TextPlainValidator.validate,
225
+ description: matched.description,
226
+ jsDocTags: matched.jsDocTags,
227
+ example: example?.example,
228
+ examples: example?.examples,
229
+ ...schema,
230
+ };
231
+ else {
232
+ pErrorContents.push(`Unknown kind of the parameter.`);
233
+ return report();
234
+ }
235
+ })
236
+ .filter((x): x is IReflectHttpOperationParameter => x !== null);
237
+
238
+ if (errors.length) ctx.errors.push(...errors);
239
+ return parameters;
240
+ };
241
+
242
+ const analyzePreconfigured = (
243
+ props: IContext,
244
+ ): IReflectHttpOperationParameter.IPreconfigured[] => {
245
+ const dict: NestParameters | undefined = Reflect.getMetadata(
246
+ ROUTE_ARGS_METADATA,
247
+ props.controller.class,
248
+ props.functionName,
249
+ );
250
+ if (dict === undefined) return [];
251
+ return Object.entries(dict)
252
+ .map(([key, param]) => analyzeHttpParameter(key, param))
253
+ .filter(
254
+ (x): x is IReflectHttpOperationParameter.IPreconfigured => x !== null,
255
+ )
256
+ .sort((x, y) => x.index - y.index);
257
+ };
258
+
259
+ const analyzeHttpParameter = (
260
+ key: string,
261
+ param: INestParam,
262
+ ): IReflectHttpOperationParameter.IPreconfigured | null => {
263
+ const symbol: string = key.split(":")[0]!;
264
+ if (symbol.indexOf("__custom") !== -1) return analyzeCustomParameter(param);
265
+
266
+ const category:
267
+ | IReflectHttpOperationParameter.IPreconfigured["category"]
268
+ | null = getNestParamType(Number(symbol[0]) as RouteParamtypes);
269
+ if (category === null) return null;
270
+ if (category === "body")
271
+ return {
272
+ category: "body",
273
+ index: param.index,
274
+ field: param.data,
275
+ contentType: "application/json",
276
+ };
277
+ else
278
+ return {
279
+ category,
280
+ index: param.index,
281
+ field: param.data,
282
+ };
283
+ };
284
+
285
+ const analyzeCustomParameter = (
286
+ param: INestParam,
287
+ ): IReflectHttpOperationParameter.IPreconfigured | null => {
288
+ if (param.factory === undefined) return null;
289
+ else if (
290
+ param.factory.name === "EncryptedBody" ||
291
+ param.factory.name === "PlainBody" ||
292
+ param.factory.name === "TypedQueryBody" ||
293
+ param.factory.name === "TypedBody" ||
294
+ param.factory.name === "TypedFormDataBody"
295
+ )
296
+ return {
297
+ category: "body",
298
+ index: param.index,
299
+ encrypted: param.factory.name === "EncryptedBody",
300
+ contentType:
301
+ param.factory.name === "PlainBody" ||
302
+ param.factory.name === "EncryptedBody"
303
+ ? "text/plain"
304
+ : param.factory.name === "TypedQueryBody"
305
+ ? "application/x-www-form-urlencoded"
306
+ : param.factory.name === "TypedFormDataBody"
307
+ ? "multipart/form-data"
308
+ : "application/json",
309
+ };
310
+ else if (param.factory.name === "TypedHeaders")
311
+ return {
312
+ category: "headers",
313
+ index: param.index,
314
+ field: param.data,
315
+ };
316
+ else if (param.factory.name === "TypedParam")
317
+ return {
318
+ category: "param",
319
+ index: param.index,
320
+ field: param.data,
321
+ };
322
+ else if (param.factory.name === "TypedQuery")
323
+ return {
324
+ category: "query",
325
+ index: param.index,
326
+ field: undefined,
327
+ };
328
+ else return null;
329
+ };
330
+
331
+ const isUnique = (values: string[]) => new Set(values).size === values.length;
332
+ }
333
+
334
+ type NestParameters = {
335
+ [key: string]: INestParam;
336
+ };
337
+ interface INestParam {
338
+ name: string;
339
+ index: number;
340
+ factory?: (...args: any) => any;
341
+ data: string | undefined;
342
+ }
343
+
344
+ const getNestParamType = (value: RouteParamtypes) => {
345
+ if (value === RouteParamtypes.BODY) return "body";
346
+ else if (value === RouteParamtypes.HEADERS) return "headers";
347
+ else if (value === RouteParamtypes.QUERY) return "query";
348
+ else if (value === RouteParamtypes.PARAM) return "param";
349
+ return null;
350
+ };