@nestia/sdk 7.1.1-dev.20250714 → 7.2.0

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 (162) 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/analyses/DtoAnalyzer.d.ts +21 -0
  15. package/lib/analyses/DtoAnalyzer.js +208 -0
  16. package/lib/analyses/DtoAnalyzer.js.map +1 -0
  17. package/lib/analyses/ImportAnalyzer.d.ts +3 -8
  18. package/lib/analyses/ImportAnalyzer.js +84 -102
  19. package/lib/analyses/ImportAnalyzer.js.map +1 -1
  20. package/lib/analyses/ReflectHttpOperationAnalyzer.js +1 -1
  21. package/lib/analyses/ReflectHttpOperationAnalyzer.js.map +1 -1
  22. package/lib/analyses/ReflectWebSocketOperationAnalyzer.js +1 -1
  23. package/lib/analyses/ReflectWebSocketOperationAnalyzer.js.map +1 -1
  24. package/lib/executable/internal/NestiaConfigLoader.js +4 -4
  25. package/lib/executable/sdk.js +12 -12
  26. package/lib/generates/internal/E2eFileProgrammer.js +3 -9
  27. package/lib/generates/internal/E2eFileProgrammer.js.map +1 -1
  28. package/lib/generates/internal/ImportDictionary.d.ts +9 -11
  29. package/lib/generates/internal/ImportDictionary.js +55 -48
  30. package/lib/generates/internal/ImportDictionary.js.map +1 -1
  31. package/lib/generates/internal/SdkAliasCollection.js +24 -18
  32. package/lib/generates/internal/SdkAliasCollection.js.map +1 -1
  33. package/lib/generates/internal/SdkFileProgrammer.js +1 -7
  34. package/lib/generates/internal/SdkFileProgrammer.js.map +1 -1
  35. package/lib/generates/internal/SdkHttpCloneReferencer.js +3 -1
  36. package/lib/generates/internal/SdkHttpCloneReferencer.js.map +1 -1
  37. package/lib/generates/internal/SdkHttpSimulationProgrammer.js +4 -3
  38. package/lib/generates/internal/SdkHttpSimulationProgrammer.js.map +1 -1
  39. package/lib/generates/internal/SdkImportWizard.js +28 -21
  40. package/lib/generates/internal/SdkImportWizard.js.map +1 -1
  41. package/lib/generates/internal/SdkTypeProgrammer.js +3 -2
  42. package/lib/generates/internal/SdkTypeProgrammer.js.map +1 -1
  43. package/lib/generates/internal/SdkTypeTagProgrammer.js +10 -8
  44. package/lib/generates/internal/SdkTypeTagProgrammer.js.map +1 -1
  45. package/lib/generates/internal/SdkWebSocketNamespaceProgrammer.js +8 -6
  46. package/lib/generates/internal/SdkWebSocketNamespaceProgrammer.js.map +1 -1
  47. package/lib/generates/internal/SdkWebSocketRouteProgrammer.js +12 -9
  48. package/lib/generates/internal/SdkWebSocketRouteProgrammer.js.map +1 -1
  49. package/lib/structures/IReflectHttpOperation.d.ts +2 -2
  50. package/lib/structures/IReflectImport.d.ts +6 -0
  51. package/lib/structures/{IReflectTypeImport.js → IReflectImport.js} +1 -1
  52. package/lib/structures/IReflectImport.js.map +1 -0
  53. package/lib/structures/IReflectWebSocketOperation.d.ts +2 -2
  54. package/lib/structures/IReflectWebSocketOperationParameter.d.ts +2 -2
  55. package/lib/structures/ITypedHttpRoute.d.ts +2 -2
  56. package/lib/structures/ITypedWebSocketRoute.d.ts +2 -2
  57. package/lib/transformers/IOperationMetadata.d.ts +3 -3
  58. package/lib/transformers/SdkOperationProgrammer.d.ts +3 -1
  59. package/lib/transformers/SdkOperationProgrammer.js +57 -26
  60. package/lib/transformers/SdkOperationProgrammer.js.map +1 -1
  61. package/lib/transformers/SdkOperationTransformer.js +3 -3
  62. package/lib/transformers/SdkOperationTransformer.js.map +1 -1
  63. package/package.json +4 -4
  64. package/src/INestiaConfig.ts +269 -269
  65. package/src/NestiaSdkApplication.ts +307 -307
  66. package/src/NestiaSwaggerComposer.ts +138 -138
  67. package/src/analyses/AccessorAnalyzer.ts +67 -67
  68. package/src/analyses/ConfigAnalyzer.ts +155 -155
  69. package/src/analyses/DtoAnalyzer.ts +250 -0
  70. package/src/analyses/ExceptionAnalyzer.ts +154 -154
  71. package/src/analyses/GenericAnalyzer.ts +49 -49
  72. package/src/analyses/ImportAnalyzer.ts +126 -171
  73. package/src/analyses/PathAnalyzer.ts +69 -69
  74. package/src/analyses/ReflectControllerAnalyzer.ts +105 -105
  75. package/src/analyses/ReflectHttpOperationAnalyzer.ts +183 -183
  76. package/src/analyses/ReflectHttpOperationExceptionAnalyzer.ts +71 -71
  77. package/src/analyses/ReflectHttpOperationParameterAnalyzer.ts +348 -348
  78. package/src/analyses/ReflectHttpOperationResponseAnalyzer.ts +127 -127
  79. package/src/analyses/ReflectMetadataAnalyzer.ts +44 -44
  80. package/src/analyses/ReflectWebSocketOperationAnalyzer.ts +172 -172
  81. package/src/analyses/SecurityAnalyzer.ts +25 -25
  82. package/src/analyses/TypedHttpRouteAnalyzer.ts +204 -204
  83. package/src/analyses/TypedWebSocketRouteAnalyzer.ts +33 -33
  84. package/src/decorators/OperationMetadata.ts +15 -15
  85. package/src/executable/internal/CommandParser.ts +15 -15
  86. package/src/executable/internal/NestiaConfigLoader.ts +78 -78
  87. package/src/executable/internal/NestiaSdkCommand.ts +103 -103
  88. package/src/executable/sdk.ts +75 -75
  89. package/src/generates/CloneGenerator.ts +66 -66
  90. package/src/generates/E2eGenerator.ts +32 -32
  91. package/src/generates/SdkGenerator.ts +160 -160
  92. package/src/generates/SwaggerGenerator.ts +284 -284
  93. package/src/generates/internal/E2eFileProgrammer.ts +197 -205
  94. package/src/generates/internal/FilePrinter.ts +53 -53
  95. package/src/generates/internal/ImportDictionary.ts +190 -163
  96. package/src/generates/internal/SdkAliasCollection.ts +261 -255
  97. package/src/generates/internal/SdkDistributionComposer.ts +103 -103
  98. package/src/generates/internal/SdkFileProgrammer.ts +110 -116
  99. package/src/generates/internal/SdkHttpCloneProgrammer.ts +124 -124
  100. package/src/generates/internal/SdkHttpCloneReferencer.ts +77 -75
  101. package/src/generates/internal/SdkHttpFunctionProgrammer.ts +279 -279
  102. package/src/generates/internal/SdkHttpNamespaceProgrammer.ts +500 -500
  103. package/src/generates/internal/SdkHttpParameterProgrammer.ts +178 -178
  104. package/src/generates/internal/SdkHttpRouteProgrammer.ts +107 -107
  105. package/src/generates/internal/SdkHttpSimulationProgrammer.ts +310 -309
  106. package/src/generates/internal/SdkImportWizard.ts +62 -55
  107. package/src/generates/internal/SdkRouteDirectory.ts +18 -18
  108. package/src/generates/internal/SdkTypeProgrammer.ts +385 -384
  109. package/src/generates/internal/SdkTypeTagProgrammer.ts +104 -102
  110. package/src/generates/internal/SdkWebSocketNamespaceProgrammer.ts +368 -366
  111. package/src/generates/internal/SdkWebSocketParameterProgrammer.ts +87 -87
  112. package/src/generates/internal/SdkWebSocketRouteProgrammer.ts +282 -279
  113. package/src/generates/internal/SwaggerDescriptionComposer.ts +64 -64
  114. package/src/generates/internal/SwaggerOperationComposer.ts +119 -119
  115. package/src/generates/internal/SwaggerOperationParameterComposer.ts +177 -177
  116. package/src/generates/internal/SwaggerOperationResponseComposer.ts +110 -110
  117. package/src/index.ts +4 -4
  118. package/src/module.ts +3 -3
  119. package/src/structures/INestiaProject.ts +13 -13
  120. package/src/structures/INestiaSdkInput.ts +20 -20
  121. package/src/structures/IReflectApplication.ts +8 -8
  122. package/src/structures/IReflectController.ts +15 -15
  123. package/src/structures/IReflectHttpOperation.ts +26 -26
  124. package/src/structures/IReflectHttpOperationException.ts +19 -19
  125. package/src/structures/IReflectHttpOperationParameter.ts +77 -81
  126. package/src/structures/IReflectHttpOperationSuccess.ts +22 -22
  127. package/src/structures/IReflectImport.ts +6 -0
  128. package/src/structures/IReflectOperationError.ts +26 -26
  129. package/src/structures/IReflectType.ts +4 -4
  130. package/src/structures/IReflectWebSocketOperation.ts +17 -17
  131. package/src/structures/IReflectWebSocketOperationParameter.ts +36 -38
  132. package/src/structures/ITypedApplication.ts +11 -11
  133. package/src/structures/ITypedHttpRoute.ts +41 -41
  134. package/src/structures/ITypedHttpRouteException.ts +15 -15
  135. package/src/structures/ITypedHttpRouteParameter.ts +41 -41
  136. package/src/structures/ITypedHttpRouteSuccess.ts +22 -22
  137. package/src/structures/ITypedWebSocketRoute.ts +24 -24
  138. package/src/structures/ITypedWebSocketRouteParameter.ts +3 -3
  139. package/src/structures/MethodType.ts +5 -5
  140. package/src/structures/ParamCategory.ts +1 -1
  141. package/src/structures/TypeEntry.ts +22 -22
  142. package/src/transform.ts +9 -9
  143. package/src/transformers/IOperationMetadata.ts +44 -44
  144. package/src/transformers/ISdkOperationTransformerContext.ts +8 -8
  145. package/src/transformers/SdkOperationProgrammer.ts +238 -209
  146. package/src/transformers/SdkOperationTransformer.ts +252 -253
  147. package/src/transformers/TextPlainValidator.ts +17 -17
  148. package/src/typings/get-function-location.d.ts +7 -7
  149. package/src/utils/ArrayUtil.ts +26 -26
  150. package/src/utils/FileRetriever.ts +22 -22
  151. package/src/utils/MapUtil.ts +14 -14
  152. package/src/utils/MetadataUtil.ts +26 -26
  153. package/src/utils/PathUtil.ts +10 -10
  154. package/src/utils/SourceFinder.ts +66 -66
  155. package/src/utils/StringUtil.ts +17 -17
  156. package/src/utils/StripEnums.ts +5 -5
  157. package/src/utils/VersioningStrategy.ts +28 -28
  158. package/src/validators/HttpHeadersValidator.ts +34 -34
  159. package/src/validators/HttpQueryValidator.ts +34 -34
  160. package/lib/structures/IReflectTypeImport.d.ts +0 -4
  161. package/lib/structures/IReflectTypeImport.js.map +0 -1
  162. package/src/structures/IReflectTypeImport.ts +0 -4
@@ -1,154 +1,154 @@
1
- // import path from "path";
2
- // import ts from "typescript";
3
- // import { MetadataCollection } from "typia/lib/factories/MetadataCollection";
4
- // import { MetadataFactory } from "typia/lib/factories/MetadataFactory";
5
- // import { Metadata } from "typia/lib/schemas/metadata/Metadata";
6
-
7
- // import { INestiaProject } from "../structures/INestiaProject";
8
- // import { IReflectController } from "../structures/IReflectController";
9
- // import { IReflectHttpOperation } from "../structures/IReflectHttpOperation";
10
- // import { ITypeTuple } from "../structures/ITypeTuple";
11
- // import { ITypedHttpRoute } from "../structures/ITypedHttpRoute";
12
- // import { GenericAnalyzer } from "./GenericAnalyzer";
13
- // import { ImportAnalyzer } from "./ImportAnalyzer";
14
-
15
- // export namespace ExceptionAnalyzer {
16
- // export const analyze =
17
- // (project: INestiaProject) =>
18
- // (props: {
19
- // generics: GenericAnalyzer.Dictionary;
20
- // imports: ImportAnalyzer.Dictionary;
21
- // controller: IReflectController;
22
- // operation: IReflectHttpOperation;
23
- // declaration: ts.MethodDeclaration;
24
- // }): Record<
25
- // number | "2XX" | "3XX" | "4XX" | "5XX",
26
- // ITypedHttpRoute.IOutput
27
- // > => {
28
- // const output: Record<
29
- // number | "2XX" | "3XX" | "4XX" | "5XX",
30
- // ITypedHttpRoute.IOutput
31
- // > = {} as any;
32
- // for (const decorator of props.declaration.modifiers ?? [])
33
- // if (ts.isDecorator(decorator))
34
- // analyzeTyped(project)({
35
- // ...props,
36
- // output,
37
- // decorator,
38
- // });
39
- // return output;
40
- // };
41
-
42
- // const analyzeTyped =
43
- // (project: INestiaProject) =>
44
- // (props: {
45
- // generics: GenericAnalyzer.Dictionary;
46
- // imports: ImportAnalyzer.Dictionary;
47
- // controller: IReflectController;
48
- // operation: IReflectHttpOperation;
49
- // output: Record<
50
- // number | "2XX" | "3XX" | "4XX" | "5XX",
51
- // ITypedHttpRoute.IOutput
52
- // >;
53
- // decorator: ts.Decorator;
54
- // }): boolean => {
55
- // // CHECK DECORATOR
56
- // if (!ts.isCallExpression(props.decorator.expression)) return false;
57
- // else if ((props.decorator.expression.typeArguments ?? []).length !== 1)
58
- // return false;
59
-
60
- // // CHECK SIGNATURE
61
- // const signature: ts.Signature | undefined =
62
- // project.checker.getResolvedSignature(props.decorator.expression);
63
- // if (!signature || !signature.declaration) return false;
64
- // else if (
65
- // path
66
- // .resolve(signature.declaration.getSourceFile().fileName)
67
- // .indexOf(TYPED_EXCEPTION_PATH) === -1
68
- // )
69
- // return false;
70
-
71
- // // GET TYPE INFO
72
- // const status: string | null = getStatus(project.checker)(
73
- // props.decorator.expression.arguments[0] ?? null,
74
- // );
75
- // if (status === null) return false;
76
-
77
- // const node: ts.TypeNode = props.decorator.expression.typeArguments![0];
78
- // const type: ts.Type = project.checker.getTypeFromTypeNode(node);
79
- // if (type.isTypeParameter()) {
80
- // project.errors.push({
81
- // file: props.controller.file,
82
- // controller: props.controller.name,
83
- // function: props.operation.name,
84
- // message: "TypedException() without generic argument specification.",
85
- // });
86
- // return false;
87
- // }
88
-
89
- // const tuple: ITypeTuple | null = ImportAnalyzer.analyze(project.checker)({
90
- // generics: props.generics,
91
- // imports: props.imports,
92
- // type,
93
- // });
94
- // if (tuple === null) {
95
- // project.errors.push({
96
- // file: props.controller.file,
97
- // controller: props.controller.name,
98
- // function: props.operation.name,
99
- // message: `TypeException() with unknown type on ${status} status.`,
100
- // });
101
- // return false;
102
- // }
103
-
104
- // // DO ASSIGN
105
- // const matched: IReflectHttpOperation.IException[] = Object.entries(
106
- // props.operation.exceptions,
107
- // )
108
- // .filter(([key]) => status === key)
109
- // .map(([_key, value]) => value);
110
- // for (const m of matched)
111
- // props.output[m.status] = {
112
- // type: tuple.type,
113
- // typeName: tuple.typeName,
114
- // contentType: "application/json",
115
- // description: m.description,
116
- // };
117
- // return true;
118
- // };
119
-
120
- // const getStatus =
121
- // (checker: ts.TypeChecker) =>
122
- // (expression: ts.Expression | null): string | null => {
123
- // if (expression === null) return null;
124
-
125
- // const type: ts.Type = checker.getTypeAtLocation(expression);
126
- // const result = MetadataFactory.analyze(checker)({
127
- // escape: true,
128
- // constant: true,
129
- // absorb: true,
130
- // })(new MetadataCollection())(type);
131
- // if (false === result.success) return null;
132
-
133
- // const meta: Metadata = result.data;
134
- // if (meta.constants.length === 1)
135
- // return meta.constants[0].values[0].value.toString();
136
- // else if (meta.escaped && meta.escaped.returns.constants.length === 1)
137
- // return meta.escaped.returns.constants[0].values[0].value.toString();
138
- // else if (ts.isStringLiteral(expression)) return expression.text;
139
- // else if (ts.isNumericLiteral(expression)) {
140
- // const value: number = Number(expression.text.split("_").join(""));
141
- // if (false === isNaN(value)) return value.toString();
142
- // }
143
- // return null;
144
- // };
145
- // }
146
-
147
- // const TYPED_EXCEPTION_PATH = path.join(
148
- // "node_modules",
149
- // "@nestia",
150
- // "core",
151
- // "lib",
152
- // "decorators",
153
- // "TypedException.d.ts",
154
- // );
1
+ // import path from "path";
2
+ // import ts from "typescript";
3
+ // import { MetadataCollection } from "typia/lib/factories/MetadataCollection";
4
+ // import { MetadataFactory } from "typia/lib/factories/MetadataFactory";
5
+ // import { Metadata } from "typia/lib/schemas/metadata/Metadata";
6
+
7
+ // import { INestiaProject } from "../structures/INestiaProject";
8
+ // import { IReflectController } from "../structures/IReflectController";
9
+ // import { IReflectHttpOperation } from "../structures/IReflectHttpOperation";
10
+ // import { ITypeTuple } from "../structures/ITypeTuple";
11
+ // import { ITypedHttpRoute } from "../structures/ITypedHttpRoute";
12
+ // import { GenericAnalyzer } from "./GenericAnalyzer";
13
+ // import { ImportAnalyzer } from "./ImportAnalyzer";
14
+
15
+ // export namespace ExceptionAnalyzer {
16
+ // export const analyze =
17
+ // (project: INestiaProject) =>
18
+ // (props: {
19
+ // generics: GenericAnalyzer.Dictionary;
20
+ // imports: ImportAnalyzer.Dictionary;
21
+ // controller: IReflectController;
22
+ // operation: IReflectHttpOperation;
23
+ // declaration: ts.MethodDeclaration;
24
+ // }): Record<
25
+ // number | "2XX" | "3XX" | "4XX" | "5XX",
26
+ // ITypedHttpRoute.IOutput
27
+ // > => {
28
+ // const output: Record<
29
+ // number | "2XX" | "3XX" | "4XX" | "5XX",
30
+ // ITypedHttpRoute.IOutput
31
+ // > = {} as any;
32
+ // for (const decorator of props.declaration.modifiers ?? [])
33
+ // if (ts.isDecorator(decorator))
34
+ // analyzeTyped(project)({
35
+ // ...props,
36
+ // output,
37
+ // decorator,
38
+ // });
39
+ // return output;
40
+ // };
41
+
42
+ // const analyzeTyped =
43
+ // (project: INestiaProject) =>
44
+ // (props: {
45
+ // generics: GenericAnalyzer.Dictionary;
46
+ // imports: ImportAnalyzer.Dictionary;
47
+ // controller: IReflectController;
48
+ // operation: IReflectHttpOperation;
49
+ // output: Record<
50
+ // number | "2XX" | "3XX" | "4XX" | "5XX",
51
+ // ITypedHttpRoute.IOutput
52
+ // >;
53
+ // decorator: ts.Decorator;
54
+ // }): boolean => {
55
+ // // CHECK DECORATOR
56
+ // if (!ts.isCallExpression(props.decorator.expression)) return false;
57
+ // else if ((props.decorator.expression.typeArguments ?? []).length !== 1)
58
+ // return false;
59
+
60
+ // // CHECK SIGNATURE
61
+ // const signature: ts.Signature | undefined =
62
+ // project.checker.getResolvedSignature(props.decorator.expression);
63
+ // if (!signature || !signature.declaration) return false;
64
+ // else if (
65
+ // path
66
+ // .resolve(signature.declaration.getSourceFile().fileName)
67
+ // .indexOf(TYPED_EXCEPTION_PATH) === -1
68
+ // )
69
+ // return false;
70
+
71
+ // // GET TYPE INFO
72
+ // const status: string | null = getStatus(project.checker)(
73
+ // props.decorator.expression.arguments[0] ?? null,
74
+ // );
75
+ // if (status === null) return false;
76
+
77
+ // const node: ts.TypeNode = props.decorator.expression.typeArguments![0];
78
+ // const type: ts.Type = project.checker.getTypeFromTypeNode(node);
79
+ // if (type.isTypeParameter()) {
80
+ // project.errors.push({
81
+ // file: props.controller.file,
82
+ // controller: props.controller.name,
83
+ // function: props.operation.name,
84
+ // message: "TypedException() without generic argument specification.",
85
+ // });
86
+ // return false;
87
+ // }
88
+
89
+ // const tuple: ITypeTuple | null = ImportAnalyzer.analyze(project.checker)({
90
+ // generics: props.generics,
91
+ // imports: props.imports,
92
+ // type,
93
+ // });
94
+ // if (tuple === null) {
95
+ // project.errors.push({
96
+ // file: props.controller.file,
97
+ // controller: props.controller.name,
98
+ // function: props.operation.name,
99
+ // message: `TypeException() with unknown type on ${status} status.`,
100
+ // });
101
+ // return false;
102
+ // }
103
+
104
+ // // DO ASSIGN
105
+ // const matched: IReflectHttpOperation.IException[] = Object.entries(
106
+ // props.operation.exceptions,
107
+ // )
108
+ // .filter(([key]) => status === key)
109
+ // .map(([_key, value]) => value);
110
+ // for (const m of matched)
111
+ // props.output[m.status] = {
112
+ // type: tuple.type,
113
+ // typeName: tuple.typeName,
114
+ // contentType: "application/json",
115
+ // description: m.description,
116
+ // };
117
+ // return true;
118
+ // };
119
+
120
+ // const getStatus =
121
+ // (checker: ts.TypeChecker) =>
122
+ // (expression: ts.Expression | null): string | null => {
123
+ // if (expression === null) return null;
124
+
125
+ // const type: ts.Type = checker.getTypeAtLocation(expression);
126
+ // const result = MetadataFactory.analyze(checker)({
127
+ // escape: true,
128
+ // constant: true,
129
+ // absorb: true,
130
+ // })(new MetadataCollection())(type);
131
+ // if (false === result.success) return null;
132
+
133
+ // const meta: Metadata = result.data;
134
+ // if (meta.constants.length === 1)
135
+ // return meta.constants[0].values[0].value.toString();
136
+ // else if (meta.escaped && meta.escaped.returns.constants.length === 1)
137
+ // return meta.escaped.returns.constants[0].values[0].value.toString();
138
+ // else if (ts.isStringLiteral(expression)) return expression.text;
139
+ // else if (ts.isNumericLiteral(expression)) {
140
+ // const value: number = Number(expression.text.split("_").join(""));
141
+ // if (false === isNaN(value)) return value.toString();
142
+ // }
143
+ // return null;
144
+ // };
145
+ // }
146
+
147
+ // const TYPED_EXCEPTION_PATH = path.join(
148
+ // "node_modules",
149
+ // "@nestia",
150
+ // "core",
151
+ // "lib",
152
+ // "decorators",
153
+ // "TypedException.d.ts",
154
+ // );
@@ -1,49 +1,49 @@
1
- import ts from "typescript";
2
-
3
- export namespace GenericAnalyzer {
4
- export function analyze(
5
- checker: ts.TypeChecker,
6
- classNode: ts.ClassDeclaration,
7
- ): WeakMap<ts.Type, ts.Type> {
8
- const dict: WeakMap<ts.Type, ts.Type> = new WeakMap();
9
- explore(checker, dict, classNode);
10
- return dict;
11
- }
12
-
13
- function explore(
14
- checker: ts.TypeChecker,
15
- dict: WeakMap<ts.Type, ts.Type>,
16
- classNode: ts.ClassDeclaration,
17
- ): void {
18
- if (classNode.heritageClauses === undefined) return;
19
-
20
- for (const heritage of classNode.heritageClauses)
21
- for (const hType of heritage.types) {
22
- // MUST BE CLASS
23
- const expression: ts.Type = checker.getTypeAtLocation(hType.expression);
24
- const superNode: ts.Declaration | undefined =
25
- expression.symbol?.getDeclarations?.()?.[0];
26
-
27
- if (superNode === undefined || !ts.isClassDeclaration(superNode))
28
- continue;
29
-
30
- // SPECIFY GENERICS
31
- const usages: ReadonlyArray<ts.TypeNode> = hType.typeArguments || [];
32
- const parameters: ReadonlyArray<ts.TypeParameterDeclaration> =
33
- superNode.typeParameters || [];
34
-
35
- parameters.forEach((param, index) => {
36
- const paramType: ts.Type = checker.getTypeAtLocation(param);
37
- const usageType: ts.Type =
38
- usages[index] !== undefined
39
- ? checker.getTypeAtLocation(usages[index])
40
- : checker.getTypeAtLocation(param.default!);
41
-
42
- dict.set(paramType, usageType);
43
- });
44
-
45
- // RECURSIVE EXPLORATION
46
- explore(checker, dict, superNode);
47
- }
48
- }
49
- }
1
+ import ts from "typescript";
2
+
3
+ export namespace GenericAnalyzer {
4
+ export function analyze(
5
+ checker: ts.TypeChecker,
6
+ classNode: ts.ClassDeclaration,
7
+ ): WeakMap<ts.Type, ts.Type> {
8
+ const dict: WeakMap<ts.Type, ts.Type> = new WeakMap();
9
+ explore(checker, dict, classNode);
10
+ return dict;
11
+ }
12
+
13
+ function explore(
14
+ checker: ts.TypeChecker,
15
+ dict: WeakMap<ts.Type, ts.Type>,
16
+ classNode: ts.ClassDeclaration,
17
+ ): void {
18
+ if (classNode.heritageClauses === undefined) return;
19
+
20
+ for (const heritage of classNode.heritageClauses)
21
+ for (const hType of heritage.types) {
22
+ // MUST BE CLASS
23
+ const expression: ts.Type = checker.getTypeAtLocation(hType.expression);
24
+ const superNode: ts.Declaration | undefined =
25
+ expression.symbol?.getDeclarations?.()?.[0];
26
+
27
+ if (superNode === undefined || !ts.isClassDeclaration(superNode))
28
+ continue;
29
+
30
+ // SPECIFY GENERICS
31
+ const usages: ReadonlyArray<ts.TypeNode> = hType.typeArguments || [];
32
+ const parameters: ReadonlyArray<ts.TypeParameterDeclaration> =
33
+ superNode.typeParameters || [];
34
+
35
+ parameters.forEach((param, index) => {
36
+ const paramType: ts.Type = checker.getTypeAtLocation(param);
37
+ const usageType: ts.Type =
38
+ usages[index] !== undefined
39
+ ? checker.getTypeAtLocation(usages[index])
40
+ : checker.getTypeAtLocation(param.default!);
41
+
42
+ dict.set(paramType, usageType);
43
+ });
44
+
45
+ // RECURSIVE EXPLORATION
46
+ explore(checker, dict, superNode);
47
+ }
48
+ }
49
+ }