@nestia/sdk 7.0.0-dev.20250608 → 7.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 (112) hide show
  1. package/LICENSE +21 -21
  2. package/README.md +93 -92
  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/lib/executable/sdk.js +12 -12
  17. package/package.json +5 -5
  18. package/src/INestiaConfig.ts +269 -269
  19. package/src/NestiaSdkApplication.ts +307 -307
  20. package/src/NestiaSwaggerComposer.ts +138 -138
  21. package/src/analyses/AccessorAnalyzer.ts +67 -67
  22. package/src/analyses/ConfigAnalyzer.ts +155 -155
  23. package/src/analyses/ExceptionAnalyzer.ts +154 -154
  24. package/src/analyses/GenericAnalyzer.ts +49 -49
  25. package/src/analyses/ImportAnalyzer.ts +171 -171
  26. package/src/analyses/PathAnalyzer.ts +69 -69
  27. package/src/analyses/ReflectControllerAnalyzer.ts +105 -105
  28. package/src/analyses/ReflectHttpOperationAnalyzer.ts +183 -183
  29. package/src/analyses/ReflectHttpOperationExceptionAnalyzer.ts +71 -71
  30. package/src/analyses/ReflectHttpOperationParameterAnalyzer.ts +348 -348
  31. package/src/analyses/ReflectHttpOperationResponseAnalyzer.ts +127 -127
  32. package/src/analyses/ReflectMetadataAnalyzer.ts +44 -44
  33. package/src/analyses/ReflectWebSocketOperationAnalyzer.ts +172 -172
  34. package/src/analyses/SecurityAnalyzer.ts +25 -25
  35. package/src/analyses/TypedHttpRouteAnalyzer.ts +204 -204
  36. package/src/analyses/TypedWebSocketRouteAnalyzer.ts +33 -33
  37. package/src/decorators/OperationMetadata.ts +15 -15
  38. package/src/executable/internal/CommandParser.ts +15 -15
  39. package/src/executable/internal/NestiaConfigLoader.ts +78 -78
  40. package/src/executable/internal/NestiaSdkCommand.ts +103 -103
  41. package/src/executable/sdk.ts +75 -75
  42. package/src/generates/CloneGenerator.ts +66 -66
  43. package/src/generates/E2eGenerator.ts +32 -32
  44. package/src/generates/SdkGenerator.ts +160 -160
  45. package/src/generates/SwaggerGenerator.ts +284 -284
  46. package/src/generates/internal/E2eFileProgrammer.ts +205 -205
  47. package/src/generates/internal/FilePrinter.ts +53 -53
  48. package/src/generates/internal/ImportDictionary.ts +163 -163
  49. package/src/generates/internal/SdkAliasCollection.ts +255 -255
  50. package/src/generates/internal/SdkDistributionComposer.ts +103 -103
  51. package/src/generates/internal/SdkFileProgrammer.ts +116 -116
  52. package/src/generates/internal/SdkHttpCloneProgrammer.ts +124 -124
  53. package/src/generates/internal/SdkHttpCloneReferencer.ts +75 -75
  54. package/src/generates/internal/SdkHttpFunctionProgrammer.ts +276 -276
  55. package/src/generates/internal/SdkHttpNamespaceProgrammer.ts +500 -500
  56. package/src/generates/internal/SdkHttpParameterProgrammer.ts +178 -178
  57. package/src/generates/internal/SdkHttpRouteProgrammer.ts +107 -107
  58. package/src/generates/internal/SdkHttpSimulationProgrammer.ts +340 -340
  59. package/src/generates/internal/SdkImportWizard.ts +55 -55
  60. package/src/generates/internal/SdkRouteDirectory.ts +18 -18
  61. package/src/generates/internal/SdkTypeProgrammer.ts +384 -384
  62. package/src/generates/internal/SdkTypeTagProgrammer.ts +102 -102
  63. package/src/generates/internal/SdkWebSocketNamespaceProgrammer.ts +366 -366
  64. package/src/generates/internal/SdkWebSocketParameterProgrammer.ts +87 -87
  65. package/src/generates/internal/SdkWebSocketRouteProgrammer.ts +279 -279
  66. package/src/generates/internal/SwaggerDescriptionComposer.ts +64 -64
  67. package/src/generates/internal/SwaggerOperationComposer.ts +119 -119
  68. package/src/generates/internal/SwaggerOperationParameterComposer.ts +177 -177
  69. package/src/generates/internal/SwaggerOperationResponseComposer.ts +110 -110
  70. package/src/index.ts +4 -4
  71. package/src/module.ts +3 -3
  72. package/src/structures/INestiaProject.ts +13 -13
  73. package/src/structures/INestiaSdkInput.ts +20 -20
  74. package/src/structures/IReflectApplication.ts +8 -8
  75. package/src/structures/IReflectController.ts +15 -15
  76. package/src/structures/IReflectHttpOperation.ts +26 -26
  77. package/src/structures/IReflectHttpOperationException.ts +19 -19
  78. package/src/structures/IReflectHttpOperationParameter.ts +81 -81
  79. package/src/structures/IReflectHttpOperationSuccess.ts +22 -22
  80. package/src/structures/IReflectOperationError.ts +26 -26
  81. package/src/structures/IReflectType.ts +4 -4
  82. package/src/structures/IReflectTypeImport.ts +4 -4
  83. package/src/structures/IReflectWebSocketOperation.ts +17 -17
  84. package/src/structures/IReflectWebSocketOperationParameter.ts +38 -38
  85. package/src/structures/ITypedApplication.ts +11 -11
  86. package/src/structures/ITypedHttpRoute.ts +41 -41
  87. package/src/structures/ITypedHttpRouteException.ts +15 -15
  88. package/src/structures/ITypedHttpRouteParameter.ts +41 -41
  89. package/src/structures/ITypedHttpRouteSuccess.ts +22 -22
  90. package/src/structures/ITypedWebSocketRoute.ts +24 -24
  91. package/src/structures/ITypedWebSocketRouteParameter.ts +3 -3
  92. package/src/structures/MethodType.ts +5 -5
  93. package/src/structures/ParamCategory.ts +1 -1
  94. package/src/structures/TypeEntry.ts +22 -22
  95. package/src/transform.ts +9 -9
  96. package/src/transformers/IOperationMetadata.ts +44 -44
  97. package/src/transformers/ISdkOperationTransformerContext.ts +8 -8
  98. package/src/transformers/SdkOperationProgrammer.ts +209 -209
  99. package/src/transformers/SdkOperationTransformer.ts +253 -253
  100. package/src/transformers/TextPlainValidator.ts +17 -17
  101. package/src/typings/get-function-location.d.ts +7 -7
  102. package/src/utils/ArrayUtil.ts +26 -26
  103. package/src/utils/FileRetriever.ts +22 -22
  104. package/src/utils/MapUtil.ts +14 -14
  105. package/src/utils/MetadataUtil.ts +26 -26
  106. package/src/utils/PathUtil.ts +10 -10
  107. package/src/utils/SourceFinder.ts +66 -66
  108. package/src/utils/StringUtil.ts +17 -17
  109. package/src/utils/StripEnums.ts +5 -5
  110. package/src/utils/VersioningStrategy.ts +28 -28
  111. package/src/validators/HttpHeadersValidator.ts +34 -34
  112. package/src/validators/HttpQueryValidator.ts +34 -34
@@ -1,105 +1,105 @@
1
- import {
2
- HOST_METADATA,
3
- PATH_METADATA,
4
- SCOPE_OPTIONS_METADATA,
5
- } from "@nestjs/common/constants";
6
-
7
- import { INestiaProject } from "../structures/INestiaProject";
8
- import { INestiaSdkInput } from "../structures/INestiaSdkInput";
9
- import { IReflectController } from "../structures/IReflectController";
10
- import { IReflectHttpOperation } from "../structures/IReflectHttpOperation";
11
- import { IReflectWebSocketOperation } from "../structures/IReflectWebSocketOperation";
12
- import { IOperationMetadata } from "../transformers/IOperationMetadata";
13
- import { ArrayUtil } from "../utils/ArrayUtil";
14
- import { ReflectHttpOperationAnalyzer } from "./ReflectHttpOperationAnalyzer";
15
- import { ReflectMetadataAnalyzer } from "./ReflectMetadataAnalyzer";
16
- import { ReflectWebSocketOperationAnalyzer } from "./ReflectWebSocketOperationAnalyzer";
17
-
18
- export namespace ReflectControllerAnalyzer {
19
- export interface IProps {
20
- project: Omit<INestiaProject, "config">;
21
- controller: INestiaSdkInput.IController;
22
- unique: WeakSet<Function>;
23
- }
24
- /* ---------------------------------------------------------
25
- CONTROLLER
26
- --------------------------------------------------------- */
27
- export const analyze = (props: IProps): IReflectController | null => {
28
- // MUST BE TYPE OF A CREATOR WHO HAS THE CONSTRUCTOR
29
- if (
30
- ArrayUtil.has(
31
- Reflect.getMetadataKeys(props.controller.class),
32
- PATH_METADATA,
33
- HOST_METADATA,
34
- SCOPE_OPTIONS_METADATA,
35
- ) === false
36
- )
37
- return null;
38
-
39
- // BASIC INFO
40
- const controller: IReflectController = {
41
- class: props.controller.class,
42
- file: props.controller.location,
43
- operations: [],
44
- prefixes: props.controller.prefixes,
45
- paths: ReflectMetadataAnalyzer.paths(props.controller.class).filter(
46
- (str) => {
47
- if (str.includes("*") === true) {
48
- props.project.warnings.push({
49
- file: props.controller.location,
50
- class: props.controller.class.name,
51
- function: null,
52
- from: null,
53
- contents: ["@nestia/sdk does not compose wildcard controller."],
54
- });
55
- return false;
56
- }
57
- return true;
58
- },
59
- ),
60
- versions: ReflectMetadataAnalyzer.versions(props.controller.class),
61
- security: ReflectMetadataAnalyzer.securities(props.controller.class),
62
- tags:
63
- Reflect.getMetadata("swagger/apiUseTags", props.controller.class) ?? [],
64
- };
65
-
66
- // OPERATORS
67
- for (const [key, value] of _Get_prototype_entries(props.controller.class)) {
68
- if (typeof value !== "function") continue;
69
- const metadata: IOperationMetadata | undefined = Reflect.getMetadata(
70
- "nestia/OperationMetadata",
71
- props.controller.class.prototype,
72
- key,
73
- );
74
- if (metadata === undefined) continue;
75
- else if (metadata.jsDocTags.some((tag) => tag.name === "ignore"))
76
- continue;
77
- const next: ReflectHttpOperationAnalyzer.IProps = {
78
- project: props.project,
79
- controller: controller,
80
- name: key,
81
- function: value,
82
- metadata,
83
- };
84
- const child: IReflectHttpOperation | IReflectWebSocketOperation | null =
85
- ReflectWebSocketOperationAnalyzer.analyze(next) ??
86
- ReflectHttpOperationAnalyzer.analyze(next);
87
- if (child !== null) controller.operations.push(child);
88
- }
89
- return controller;
90
- };
91
-
92
- function _Get_prototype_entries(creator: any): Array<[string, unknown]> {
93
- const keyList = Object.getOwnPropertyNames(creator.prototype);
94
- const entries: Array<[string, unknown]> = keyList.map((key) => [
95
- key,
96
- creator.prototype[key],
97
- ]);
98
-
99
- const parent = Object.getPrototypeOf(creator);
100
- if (parent.prototype !== undefined)
101
- entries.push(..._Get_prototype_entries(parent));
102
-
103
- return entries;
104
- }
105
- }
1
+ import {
2
+ HOST_METADATA,
3
+ PATH_METADATA,
4
+ SCOPE_OPTIONS_METADATA,
5
+ } from "@nestjs/common/constants";
6
+
7
+ import { INestiaProject } from "../structures/INestiaProject";
8
+ import { INestiaSdkInput } from "../structures/INestiaSdkInput";
9
+ import { IReflectController } from "../structures/IReflectController";
10
+ import { IReflectHttpOperation } from "../structures/IReflectHttpOperation";
11
+ import { IReflectWebSocketOperation } from "../structures/IReflectWebSocketOperation";
12
+ import { IOperationMetadata } from "../transformers/IOperationMetadata";
13
+ import { ArrayUtil } from "../utils/ArrayUtil";
14
+ import { ReflectHttpOperationAnalyzer } from "./ReflectHttpOperationAnalyzer";
15
+ import { ReflectMetadataAnalyzer } from "./ReflectMetadataAnalyzer";
16
+ import { ReflectWebSocketOperationAnalyzer } from "./ReflectWebSocketOperationAnalyzer";
17
+
18
+ export namespace ReflectControllerAnalyzer {
19
+ export interface IProps {
20
+ project: Omit<INestiaProject, "config">;
21
+ controller: INestiaSdkInput.IController;
22
+ unique: WeakSet<Function>;
23
+ }
24
+ /* ---------------------------------------------------------
25
+ CONTROLLER
26
+ --------------------------------------------------------- */
27
+ export const analyze = (props: IProps): IReflectController | null => {
28
+ // MUST BE TYPE OF A CREATOR WHO HAS THE CONSTRUCTOR
29
+ if (
30
+ ArrayUtil.has(
31
+ Reflect.getMetadataKeys(props.controller.class),
32
+ PATH_METADATA,
33
+ HOST_METADATA,
34
+ SCOPE_OPTIONS_METADATA,
35
+ ) === false
36
+ )
37
+ return null;
38
+
39
+ // BASIC INFO
40
+ const controller: IReflectController = {
41
+ class: props.controller.class,
42
+ file: props.controller.location,
43
+ operations: [],
44
+ prefixes: props.controller.prefixes,
45
+ paths: ReflectMetadataAnalyzer.paths(props.controller.class).filter(
46
+ (str) => {
47
+ if (str.includes("*") === true) {
48
+ props.project.warnings.push({
49
+ file: props.controller.location,
50
+ class: props.controller.class.name,
51
+ function: null,
52
+ from: null,
53
+ contents: ["@nestia/sdk does not compose wildcard controller."],
54
+ });
55
+ return false;
56
+ }
57
+ return true;
58
+ },
59
+ ),
60
+ versions: ReflectMetadataAnalyzer.versions(props.controller.class),
61
+ security: ReflectMetadataAnalyzer.securities(props.controller.class),
62
+ tags:
63
+ Reflect.getMetadata("swagger/apiUseTags", props.controller.class) ?? [],
64
+ };
65
+
66
+ // OPERATORS
67
+ for (const [key, value] of _Get_prototype_entries(props.controller.class)) {
68
+ if (typeof value !== "function") continue;
69
+ const metadata: IOperationMetadata | undefined = Reflect.getMetadata(
70
+ "nestia/OperationMetadata",
71
+ props.controller.class.prototype,
72
+ key,
73
+ );
74
+ if (metadata === undefined) continue;
75
+ else if (metadata.jsDocTags.some((tag) => tag.name === "ignore"))
76
+ continue;
77
+ const next: ReflectHttpOperationAnalyzer.IProps = {
78
+ project: props.project,
79
+ controller: controller,
80
+ name: key,
81
+ function: value,
82
+ metadata,
83
+ };
84
+ const child: IReflectHttpOperation | IReflectWebSocketOperation | null =
85
+ ReflectWebSocketOperationAnalyzer.analyze(next) ??
86
+ ReflectHttpOperationAnalyzer.analyze(next);
87
+ if (child !== null) controller.operations.push(child);
88
+ }
89
+ return controller;
90
+ };
91
+
92
+ function _Get_prototype_entries(creator: any): Array<[string, unknown]> {
93
+ const keyList = Object.getOwnPropertyNames(creator.prototype);
94
+ const entries: Array<[string, unknown]> = keyList.map((key) => [
95
+ key,
96
+ creator.prototype[key],
97
+ ]);
98
+
99
+ const parent = Object.getPrototypeOf(creator);
100
+ if (parent.prototype !== undefined)
101
+ entries.push(..._Get_prototype_entries(parent));
102
+
103
+ return entries;
104
+ }
105
+ }
@@ -1,183 +1,183 @@
1
- import { METHOD_METADATA, PATH_METADATA } from "@nestjs/common/constants";
2
- import { ranges } from "tstl";
3
-
4
- import { INestiaProject } from "../structures/INestiaProject";
5
- import { IReflectController } from "../structures/IReflectController";
6
- import { IReflectHttpOperation } from "../structures/IReflectHttpOperation";
7
- import { IReflectHttpOperationParameter } from "../structures/IReflectHttpOperationParameter";
8
- import { IReflectHttpOperationSuccess } from "../structures/IReflectHttpOperationSuccess";
9
- import { IReflectOperationError } from "../structures/IReflectOperationError";
10
- import { IOperationMetadata } from "../transformers/IOperationMetadata";
11
- import { ArrayUtil } from "../utils/ArrayUtil";
12
- import { ImportAnalyzer } from "./ImportAnalyzer";
13
- import { PathAnalyzer } from "./PathAnalyzer";
14
- import { ReflectHttpOperationExceptionAnalyzer } from "./ReflectHttpOperationExceptionAnalyzer";
15
- import { ReflectHttpOperationParameterAnalyzer } from "./ReflectHttpOperationParameterAnalyzer";
16
- import { ReflectHttpOperationResponseAnalyzer } from "./ReflectHttpOperationResponseAnalyzer";
17
- import { ReflectMetadataAnalyzer } from "./ReflectMetadataAnalyzer";
18
-
19
- export namespace ReflectHttpOperationAnalyzer {
20
- export interface IProps {
21
- project: Omit<INestiaProject, "config">;
22
- controller: IReflectController;
23
- function: Function;
24
- name: string;
25
- metadata: IOperationMetadata;
26
- }
27
- export const analyze = (props: IProps): IReflectHttpOperation | null => {
28
- if (
29
- ArrayUtil.has(
30
- Reflect.getMetadataKeys(props.function),
31
- PATH_METADATA,
32
- METHOD_METADATA,
33
- ) === false
34
- )
35
- return null;
36
-
37
- const errors: IReflectOperationError[] = [];
38
- const method: string =
39
- METHODS[Reflect.getMetadata(METHOD_METADATA, props.function)];
40
- if (method === undefined || method === "OPTIONS") return null;
41
-
42
- const parameters: IReflectHttpOperationParameter[] =
43
- ReflectHttpOperationParameterAnalyzer.analyze({
44
- controller: props.controller,
45
- metadata: props.metadata,
46
- httpMethod: method,
47
- function: props.function,
48
- functionName: props.name,
49
- errors,
50
- });
51
- const success: IReflectHttpOperationSuccess | null = (() => {
52
- const localErrors: IReflectOperationError[] = [];
53
- const success = ReflectHttpOperationResponseAnalyzer.analyze({
54
- controller: props.controller,
55
- function: props.function,
56
- functionName: props.name,
57
- httpMethod: method,
58
- metadata: props.metadata,
59
- errors,
60
- });
61
- if (localErrors.length) {
62
- errors.push(...localErrors);
63
- return null;
64
- }
65
- return success;
66
- })();
67
- if (errors.length) {
68
- props.project.errors.push(...errors);
69
- return null;
70
- } else if (success === null) return null;
71
-
72
- // DO CONSTRUCT
73
- const operation: IReflectHttpOperation = {
74
- protocol: "http",
75
- function: props.function,
76
- name: props.name,
77
- method: method === "ALL" ? "POST" : method,
78
- paths: ReflectMetadataAnalyzer.paths(props.function).filter((str) => {
79
- if (str.includes("*") === true) {
80
- props.project.warnings.push({
81
- file: props.controller.file,
82
- class: props.controller.class.name,
83
- function: props.name,
84
- from: "",
85
- contents: ["@nestia/sdk does not compose wildcard method."],
86
- });
87
- return false;
88
- }
89
- return true;
90
- }),
91
- versions: ReflectMetadataAnalyzer.versions(props.function),
92
- parameters,
93
- success,
94
- security: ReflectMetadataAnalyzer.securities(props.function),
95
- exceptions: ReflectHttpOperationExceptionAnalyzer.analyze({
96
- controller: props.controller,
97
- function: props.function,
98
- functionName: props.name,
99
- httpMethod: method,
100
- metadata: props.metadata,
101
- errors,
102
- }),
103
- tags: Reflect.getMetadata("swagger/apiUseTags", props.function) ?? [],
104
- imports: ImportAnalyzer.unique(
105
- [
106
- ...props.metadata.parameters
107
- .filter((x) => parameters.some((y) => x.index === y.index))
108
- .map((x) => x.imports),
109
- ...props.metadata.success.imports,
110
- ...Object.values(props.metadata.exceptions).map((e) => e.imports),
111
- ].flat(),
112
- ),
113
- description: props.metadata.description,
114
- jsDocTags: props.metadata.jsDocTags,
115
- operationId: props.metadata.jsDocTags
116
- .find(({ name }) => name === "operationId")
117
- ?.text?.[0].text.split(" ")[0]
118
- .trim(),
119
- extensions: ReflectMetadataAnalyzer.extensions(props.function),
120
- };
121
-
122
- // VALIDATE PATH ARGUMENTS
123
- for (const controllerLocation of props.controller.paths)
124
- for (const metaLocation of operation.paths) {
125
- // NORMALIZE LOCATION
126
- const location: string = PathAnalyzer.join(
127
- controllerLocation,
128
- metaLocation,
129
- );
130
- if (location.includes("*")) continue;
131
-
132
- // LIST UP PARAMETERS
133
- const binded: string[] | null = PathAnalyzer.parameters(location);
134
- if (binded === null) {
135
- props.project.errors.push({
136
- file: props.controller.file,
137
- class: props.controller.class.name,
138
- function: props.name,
139
- from: "{parameters}",
140
- contents: [`invalid path (${JSON.stringify(location)})`],
141
- });
142
- continue;
143
- }
144
- const parameters: string[] = operation.parameters
145
- .filter((param) => param.category === "param")
146
- .map((param) => param.field!)
147
- .sort();
148
-
149
- // DO VALIDATE
150
- if (ranges.equal(binded.sort(), parameters) === false)
151
- errors.push({
152
- file: props.controller.file,
153
- class: props.controller.class.name,
154
- function: props.name,
155
- from: "{parameters}",
156
- contents: [
157
- `binded arguments in the "path" between function's decorator and parameters' decorators are different (function: [${binded.join(
158
- ", ",
159
- )}], parameters: [${parameters.join(", ")}]).`,
160
- ],
161
- });
162
- }
163
-
164
- // RETURNS
165
- if (errors.length) {
166
- props.project.errors.push(...errors);
167
- return null;
168
- }
169
- return operation;
170
- };
171
- }
172
-
173
- // node_modules/@nestjs/common/lib/enums/request-method.enum.ts
174
- const METHODS = [
175
- "GET",
176
- "POST",
177
- "PUT",
178
- "DELETE",
179
- "PATCH",
180
- "ALL",
181
- "OPTIONS",
182
- "HEAD",
183
- ];
1
+ import { METHOD_METADATA, PATH_METADATA } from "@nestjs/common/constants";
2
+ import { ranges } from "tstl";
3
+
4
+ import { INestiaProject } from "../structures/INestiaProject";
5
+ import { IReflectController } from "../structures/IReflectController";
6
+ import { IReflectHttpOperation } from "../structures/IReflectHttpOperation";
7
+ import { IReflectHttpOperationParameter } from "../structures/IReflectHttpOperationParameter";
8
+ import { IReflectHttpOperationSuccess } from "../structures/IReflectHttpOperationSuccess";
9
+ import { IReflectOperationError } from "../structures/IReflectOperationError";
10
+ import { IOperationMetadata } from "../transformers/IOperationMetadata";
11
+ import { ArrayUtil } from "../utils/ArrayUtil";
12
+ import { ImportAnalyzer } from "./ImportAnalyzer";
13
+ import { PathAnalyzer } from "./PathAnalyzer";
14
+ import { ReflectHttpOperationExceptionAnalyzer } from "./ReflectHttpOperationExceptionAnalyzer";
15
+ import { ReflectHttpOperationParameterAnalyzer } from "./ReflectHttpOperationParameterAnalyzer";
16
+ import { ReflectHttpOperationResponseAnalyzer } from "./ReflectHttpOperationResponseAnalyzer";
17
+ import { ReflectMetadataAnalyzer } from "./ReflectMetadataAnalyzer";
18
+
19
+ export namespace ReflectHttpOperationAnalyzer {
20
+ export interface IProps {
21
+ project: Omit<INestiaProject, "config">;
22
+ controller: IReflectController;
23
+ function: Function;
24
+ name: string;
25
+ metadata: IOperationMetadata;
26
+ }
27
+ export const analyze = (props: IProps): IReflectHttpOperation | null => {
28
+ if (
29
+ ArrayUtil.has(
30
+ Reflect.getMetadataKeys(props.function),
31
+ PATH_METADATA,
32
+ METHOD_METADATA,
33
+ ) === false
34
+ )
35
+ return null;
36
+
37
+ const errors: IReflectOperationError[] = [];
38
+ const method: string =
39
+ METHODS[Reflect.getMetadata(METHOD_METADATA, props.function)];
40
+ if (method === undefined || method === "OPTIONS") return null;
41
+
42
+ const parameters: IReflectHttpOperationParameter[] =
43
+ ReflectHttpOperationParameterAnalyzer.analyze({
44
+ controller: props.controller,
45
+ metadata: props.metadata,
46
+ httpMethod: method,
47
+ function: props.function,
48
+ functionName: props.name,
49
+ errors,
50
+ });
51
+ const success: IReflectHttpOperationSuccess | null = (() => {
52
+ const localErrors: IReflectOperationError[] = [];
53
+ const success = ReflectHttpOperationResponseAnalyzer.analyze({
54
+ controller: props.controller,
55
+ function: props.function,
56
+ functionName: props.name,
57
+ httpMethod: method,
58
+ metadata: props.metadata,
59
+ errors,
60
+ });
61
+ if (localErrors.length) {
62
+ errors.push(...localErrors);
63
+ return null;
64
+ }
65
+ return success;
66
+ })();
67
+ if (errors.length) {
68
+ props.project.errors.push(...errors);
69
+ return null;
70
+ } else if (success === null) return null;
71
+
72
+ // DO CONSTRUCT
73
+ const operation: IReflectHttpOperation = {
74
+ protocol: "http",
75
+ function: props.function,
76
+ name: props.name,
77
+ method: method === "ALL" ? "POST" : method,
78
+ paths: ReflectMetadataAnalyzer.paths(props.function).filter((str) => {
79
+ if (str.includes("*") === true) {
80
+ props.project.warnings.push({
81
+ file: props.controller.file,
82
+ class: props.controller.class.name,
83
+ function: props.name,
84
+ from: "",
85
+ contents: ["@nestia/sdk does not compose wildcard method."],
86
+ });
87
+ return false;
88
+ }
89
+ return true;
90
+ }),
91
+ versions: ReflectMetadataAnalyzer.versions(props.function),
92
+ parameters,
93
+ success,
94
+ security: ReflectMetadataAnalyzer.securities(props.function),
95
+ exceptions: ReflectHttpOperationExceptionAnalyzer.analyze({
96
+ controller: props.controller,
97
+ function: props.function,
98
+ functionName: props.name,
99
+ httpMethod: method,
100
+ metadata: props.metadata,
101
+ errors,
102
+ }),
103
+ tags: Reflect.getMetadata("swagger/apiUseTags", props.function) ?? [],
104
+ imports: ImportAnalyzer.unique(
105
+ [
106
+ ...props.metadata.parameters
107
+ .filter((x) => parameters.some((y) => x.index === y.index))
108
+ .map((x) => x.imports),
109
+ ...props.metadata.success.imports,
110
+ ...Object.values(props.metadata.exceptions).map((e) => e.imports),
111
+ ].flat(),
112
+ ),
113
+ description: props.metadata.description,
114
+ jsDocTags: props.metadata.jsDocTags,
115
+ operationId: props.metadata.jsDocTags
116
+ .find(({ name }) => name === "operationId")
117
+ ?.text?.[0].text.split(" ")[0]
118
+ .trim(),
119
+ extensions: ReflectMetadataAnalyzer.extensions(props.function),
120
+ };
121
+
122
+ // VALIDATE PATH ARGUMENTS
123
+ for (const controllerLocation of props.controller.paths)
124
+ for (const metaLocation of operation.paths) {
125
+ // NORMALIZE LOCATION
126
+ const location: string = PathAnalyzer.join(
127
+ controllerLocation,
128
+ metaLocation,
129
+ );
130
+ if (location.includes("*")) continue;
131
+
132
+ // LIST UP PARAMETERS
133
+ const binded: string[] | null = PathAnalyzer.parameters(location);
134
+ if (binded === null) {
135
+ props.project.errors.push({
136
+ file: props.controller.file,
137
+ class: props.controller.class.name,
138
+ function: props.name,
139
+ from: "{parameters}",
140
+ contents: [`invalid path (${JSON.stringify(location)})`],
141
+ });
142
+ continue;
143
+ }
144
+ const parameters: string[] = operation.parameters
145
+ .filter((param) => param.category === "param")
146
+ .map((param) => param.field!)
147
+ .sort();
148
+
149
+ // DO VALIDATE
150
+ if (ranges.equal(binded.sort(), parameters) === false)
151
+ errors.push({
152
+ file: props.controller.file,
153
+ class: props.controller.class.name,
154
+ function: props.name,
155
+ from: "{parameters}",
156
+ contents: [
157
+ `binded arguments in the "path" between function's decorator and parameters' decorators are different (function: [${binded.join(
158
+ ", ",
159
+ )}], parameters: [${parameters.join(", ")}]).`,
160
+ ],
161
+ });
162
+ }
163
+
164
+ // RETURNS
165
+ if (errors.length) {
166
+ props.project.errors.push(...errors);
167
+ return null;
168
+ }
169
+ return operation;
170
+ };
171
+ }
172
+
173
+ // node_modules/@nestjs/common/lib/enums/request-method.enum.ts
174
+ const METHODS = [
175
+ "GET",
176
+ "POST",
177
+ "PUT",
178
+ "DELETE",
179
+ "PATCH",
180
+ "ALL",
181
+ "OPTIONS",
182
+ "HEAD",
183
+ ];