@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,260 +1,260 @@
1
- import ts from "typescript";
2
-
3
- import { IReflectImport } from "../structures/IReflectImport";
4
- import { IReflectType } from "../structures/IReflectType";
5
- import { ImportAnalyzer } from "./ImportAnalyzer";
6
-
7
- export namespace DtoAnalyzer {
8
- export interface INodeProps {
9
- checker: ts.TypeChecker;
10
- imports: IReflectImport[];
11
- typeNode: ts.TypeNode;
12
- }
13
- export interface ITypeProps {
14
- checker: ts.TypeChecker;
15
- imports: IReflectImport[];
16
- type: ts.Type;
17
- }
18
- export interface IOutput {
19
- imports: IReflectImport[];
20
- type: IReflectType;
21
- }
22
-
23
- export const analyzeNode = (props: INodeProps): IOutput | null => {
24
- try {
25
- const container: IReflectImport[] = [];
26
- const type: IReflectType = exploreNode(
27
- {
28
- checker: props.checker,
29
- imports: props.imports,
30
- container,
31
- },
32
- props.typeNode,
33
- );
34
- return {
35
- type,
36
- imports: ImportAnalyzer.merge(container),
37
- };
38
- } catch {
39
- return null;
40
- }
41
- };
42
-
43
- export const analyzeType = (props: ITypeProps): IOutput | null => {
44
- try {
45
- const container: IReflectImport[] = [];
46
- const type: IReflectType = exploreType(
47
- {
48
- checker: props.checker,
49
- imports: props.imports,
50
- container,
51
- },
52
- props.type,
53
- );
54
- return {
55
- type,
56
- imports: ImportAnalyzer.merge(container),
57
- };
58
- } catch {
59
- return null;
60
- }
61
- };
62
-
63
- type IContext = Omit<INodeProps, "typeNode"> & {
64
- container: IReflectImport[];
65
- };
66
-
67
- const exploreNode = (ctx: IContext, typeNode: ts.TypeNode): IReflectType => {
68
- // Analyze symbol, and take special cares
69
- if (ts.isIntersectionTypeNode(typeNode))
70
- return {
71
- name: typeNode.types
72
- .map((child) => exploreNode(ctx, child))
73
- .map(getEscapedText)
74
- .join(" & "),
75
- };
76
- else if (ts.isUnionTypeNode(typeNode))
77
- return {
78
- name: typeNode.types
79
- .map((child) => exploreNode(ctx, child))
80
- .map(getEscapedText)
81
- .join(" | "),
82
- };
83
- else if (ts.isArrayTypeNode(typeNode)) {
84
- const element: IReflectType = exploreNode(ctx, typeNode.elementType);
85
- return {
86
- name: "Array",
87
- typeArguments: [element],
88
- };
89
- } else if (ts.isParenthesizedTypeNode(typeNode))
90
- return {
91
- name: `(${exploreNode(ctx, typeNode.type).name})`,
92
- };
93
- else if (ts.isTypeOperatorNode(typeNode)) {
94
- const prefix: string | null =
95
- typeNode.operator === ts.SyntaxKind.KeyOfKeyword
96
- ? "keyof"
97
- : typeNode.operator === ts.SyntaxKind.UniqueKeyword
98
- ? "unique"
99
- : typeNode.operator === ts.SyntaxKind.ReadonlyKeyword
100
- ? "readonly"
101
- : null;
102
- if (prefix === null)
103
- return exploreType(ctx, ctx.checker.getTypeFromTypeNode(typeNode));
104
- return {
105
- name: `${prefix} ${exploreNode(ctx, typeNode.type).name}`,
106
- };
107
- } else if (ts.isTypePredicateNode(typeNode) || ts.isTypeQueryNode(typeNode))
108
- return exploreType(ctx, ctx.checker.getTypeFromTypeNode(typeNode));
109
- else if (ts.isTypeReferenceNode(typeNode) === false)
110
- return {
111
- name: typeNode.getText(),
112
- };
113
-
114
- // Find matched import statement
115
- const name: string = typeNode.typeName.getText();
116
- const prefix: string = name.split(".")[0];
117
-
118
- let matched: boolean = false;
119
- const insert = (imp: IReflectImport): void => {
120
- matched ||= true;
121
- ctx.container.push(imp);
122
- };
123
- for (const imp of ctx.imports)
124
- if (prefix === imp.default)
125
- insert({
126
- file: imp.file,
127
- asterisk: null,
128
- default: prefix,
129
- elements: [],
130
- });
131
- else if (prefix === imp.asterisk)
132
- insert({
133
- file: imp.file,
134
- asterisk: prefix,
135
- default: null,
136
- elements: [],
137
- });
138
- else if (imp.elements.includes(prefix))
139
- insert({
140
- file: imp.file,
141
- asterisk: null,
142
- default: null,
143
- elements: [prefix],
144
- });
145
- if (prefix !== "Promise" && matched === false)
146
- return exploreType(ctx, ctx.checker.getTypeFromTypeNode(typeNode));
147
-
148
- // Finalize with generic arguments
149
- if (!!typeNode.typeArguments?.length) {
150
- const top: ts.TypeNode = typeNode.typeArguments[0];
151
- if (name === "Promise") return exploreNode(ctx, top);
152
- return {
153
- name,
154
- typeArguments: typeNode.typeArguments.map((child) =>
155
- exploreNode(ctx, child),
156
- ),
157
- };
158
- }
159
- return { name };
160
- };
161
-
162
- const exploreType = (ctx: IContext, type: ts.Type): IReflectType => {
163
- // Analyze symbol, and take special cares
164
- const symbol: ts.Symbol | undefined = type.aliasSymbol ?? type.symbol;
165
- if (type.aliasSymbol === undefined && type.isUnionOrIntersection()) {
166
- const joiner: string = type.isIntersection() ? " & " : " | ";
167
- return {
168
- name: type.types
169
- .map((child) => exploreType(ctx, child))
170
- .map(getEscapedText)
171
- .join(joiner),
172
- };
173
- } else if (ctx.checker.isArrayLikeType(type)) {
174
- const arrayItem: ts.Type | undefined =
175
- ctx.checker.getElementTypeOfArrayType(type);
176
- if (arrayItem === undefined) return { name: "Array<any>" };
177
- return {
178
- name: "Array",
179
- typeArguments: [exploreType(ctx, arrayItem)],
180
- };
181
- } else if (symbol === undefined)
182
- return {
183
- name: ctx.checker.typeToString(
184
- type,
185
- undefined,
186
- ts.TypeFormatFlags.NoTruncation,
187
- ),
188
- };
189
-
190
- // Find matched import statement
191
- const name: string = getNameOfSymbol(symbol);
192
- const prefix: string = name.split(".")[0];
193
-
194
- let matched: boolean = false;
195
- const insert = (imp: IReflectImport): void => {
196
- matched ||= true;
197
- ctx.container.push(imp);
198
- };
199
- for (const imp of ctx.imports)
200
- if (imp.elements.includes(prefix))
201
- insert({
202
- file: imp.file,
203
- asterisk: null,
204
- default: null,
205
- elements: [prefix],
206
- });
207
- if (prefix !== "Promise" && matched === false)
208
- emplaceSymbol(ctx, symbol, prefix);
209
-
210
- // Finalize with generic arguments
211
- const generic: readonly ts.Type[] = type.aliasSymbol
212
- ? (type.aliasTypeArguments ?? [])
213
- : ctx.checker.getTypeArguments(type as ts.TypeReference);
214
- return generic.length
215
- ? name === "Promise"
216
- ? exploreType(ctx, generic[0])
217
- : {
218
- name,
219
- typeArguments: generic.map((child) => exploreType(ctx, child)),
220
- }
221
- : { name };
222
- };
223
-
224
- const emplaceSymbol = (
225
- ctx: IContext,
226
- symbol: ts.Symbol,
227
- prefix: string,
228
- ): void => {
229
- // GET SOURCE FILE
230
- const sourceFile: ts.SourceFile | undefined =
231
- symbol.declarations?.[0]?.getSourceFile();
232
- if (sourceFile === undefined) return;
233
- if (sourceFile.fileName.indexOf("/typescript/lib") === -1)
234
- ctx.container.push({
235
- file: sourceFile.fileName,
236
- asterisk: null,
237
- default: null,
238
- elements: [prefix],
239
- });
240
- };
241
- }
242
-
243
- const getEscapedText = (type: IReflectType): string =>
244
- type.typeArguments
245
- ? `${type.name}<${type.typeArguments.map(getEscapedText).join(", ")}>`
246
- : type.name;
247
-
248
- const getNameOfSymbol = (symbol: ts.Symbol): string =>
249
- exploreName(
250
- symbol.escapedName.toString(),
251
- symbol.getDeclarations()?.[0]?.parent,
252
- );
253
-
254
- const exploreName = (name: string, decl?: ts.Node): string =>
255
- decl && ts.isModuleBlock(decl)
256
- ? exploreName(
257
- `${decl.parent.name.getFullText().trim()}.${name}`,
258
- decl.parent.parent,
259
- )
260
- : name;
1
+ import ts from "typescript";
2
+
3
+ import { IReflectImport } from "../structures/IReflectImport";
4
+ import { IReflectType } from "../structures/IReflectType";
5
+ import { ImportAnalyzer } from "./ImportAnalyzer";
6
+
7
+ export namespace DtoAnalyzer {
8
+ export interface INodeProps {
9
+ checker: ts.TypeChecker;
10
+ imports: IReflectImport[];
11
+ typeNode: ts.TypeNode;
12
+ }
13
+ export interface ITypeProps {
14
+ checker: ts.TypeChecker;
15
+ imports: IReflectImport[];
16
+ type: ts.Type;
17
+ }
18
+ export interface IOutput {
19
+ imports: IReflectImport[];
20
+ type: IReflectType;
21
+ }
22
+
23
+ export const analyzeNode = (props: INodeProps): IOutput | null => {
24
+ try {
25
+ const container: IReflectImport[] = [];
26
+ const type: IReflectType = exploreNode(
27
+ {
28
+ checker: props.checker,
29
+ imports: props.imports,
30
+ container,
31
+ },
32
+ props.typeNode,
33
+ );
34
+ return {
35
+ type,
36
+ imports: ImportAnalyzer.merge(container),
37
+ };
38
+ } catch {
39
+ return null;
40
+ }
41
+ };
42
+
43
+ export const analyzeType = (props: ITypeProps): IOutput | null => {
44
+ try {
45
+ const container: IReflectImport[] = [];
46
+ const type: IReflectType = exploreType(
47
+ {
48
+ checker: props.checker,
49
+ imports: props.imports,
50
+ container,
51
+ },
52
+ props.type,
53
+ );
54
+ return {
55
+ type,
56
+ imports: ImportAnalyzer.merge(container),
57
+ };
58
+ } catch {
59
+ return null;
60
+ }
61
+ };
62
+
63
+ type IContext = Omit<INodeProps, "typeNode"> & {
64
+ container: IReflectImport[];
65
+ };
66
+
67
+ const exploreNode = (ctx: IContext, typeNode: ts.TypeNode): IReflectType => {
68
+ // Analyze symbol, and take special cares
69
+ if (ts.isIntersectionTypeNode(typeNode))
70
+ return {
71
+ name: typeNode.types
72
+ .map((child) => exploreNode(ctx, child))
73
+ .map(getEscapedText)
74
+ .join(" & "),
75
+ };
76
+ else if (ts.isUnionTypeNode(typeNode))
77
+ return {
78
+ name: typeNode.types
79
+ .map((child) => exploreNode(ctx, child))
80
+ .map(getEscapedText)
81
+ .join(" | "),
82
+ };
83
+ else if (ts.isArrayTypeNode(typeNode)) {
84
+ const element: IReflectType = exploreNode(ctx, typeNode.elementType);
85
+ return {
86
+ name: "Array",
87
+ typeArguments: [element],
88
+ };
89
+ } else if (ts.isParenthesizedTypeNode(typeNode))
90
+ return {
91
+ name: `(${exploreNode(ctx, typeNode.type).name})`,
92
+ };
93
+ else if (ts.isTypeOperatorNode(typeNode)) {
94
+ const prefix: string | null =
95
+ typeNode.operator === ts.SyntaxKind.KeyOfKeyword
96
+ ? "keyof"
97
+ : typeNode.operator === ts.SyntaxKind.UniqueKeyword
98
+ ? "unique"
99
+ : typeNode.operator === ts.SyntaxKind.ReadonlyKeyword
100
+ ? "readonly"
101
+ : null;
102
+ if (prefix === null)
103
+ return exploreType(ctx, ctx.checker.getTypeFromTypeNode(typeNode));
104
+ return {
105
+ name: `${prefix} ${exploreNode(ctx, typeNode.type).name}`,
106
+ };
107
+ } else if (ts.isTypePredicateNode(typeNode) || ts.isTypeQueryNode(typeNode))
108
+ return exploreType(ctx, ctx.checker.getTypeFromTypeNode(typeNode));
109
+ else if (ts.isTypeReferenceNode(typeNode) === false)
110
+ return {
111
+ name: typeNode.getText(),
112
+ };
113
+
114
+ // Find matched import statement
115
+ const name: string = typeNode.typeName.getText();
116
+ const prefix: string = name.split(".")[0]!;
117
+
118
+ let matched: boolean = false;
119
+ const insert = (imp: IReflectImport): void => {
120
+ matched ||= true;
121
+ ctx.container.push(imp);
122
+ };
123
+ for (const imp of ctx.imports)
124
+ if (prefix === imp.default)
125
+ insert({
126
+ file: imp.file,
127
+ asterisk: null,
128
+ default: prefix,
129
+ elements: [],
130
+ });
131
+ else if (prefix === imp.asterisk)
132
+ insert({
133
+ file: imp.file,
134
+ asterisk: prefix,
135
+ default: null,
136
+ elements: [],
137
+ });
138
+ else if (imp.elements.includes(prefix))
139
+ insert({
140
+ file: imp.file,
141
+ asterisk: null,
142
+ default: null,
143
+ elements: [prefix],
144
+ });
145
+ if (prefix !== "Promise" && matched === false)
146
+ return exploreType(ctx, ctx.checker.getTypeFromTypeNode(typeNode));
147
+
148
+ // Finalize with generic arguments
149
+ if (!!typeNode.typeArguments?.length) {
150
+ const top: ts.TypeNode = typeNode.typeArguments[0]!;
151
+ if (name === "Promise") return exploreNode(ctx, top);
152
+ return {
153
+ name,
154
+ typeArguments: typeNode.typeArguments.map((child) =>
155
+ exploreNode(ctx, child),
156
+ ),
157
+ };
158
+ }
159
+ return { name };
160
+ };
161
+
162
+ const exploreType = (ctx: IContext, type: ts.Type): IReflectType => {
163
+ // Analyze symbol, and take special cares
164
+ const symbol: ts.Symbol | undefined = type.aliasSymbol ?? type.symbol;
165
+ if (type.aliasSymbol === undefined && type.isUnionOrIntersection()) {
166
+ const joiner: string = type.isIntersection() ? " & " : " | ";
167
+ return {
168
+ name: type.types
169
+ .map((child) => exploreType(ctx, child))
170
+ .map(getEscapedText)
171
+ .join(joiner),
172
+ };
173
+ } else if (ctx.checker.isArrayLikeType(type)) {
174
+ const arrayItem: ts.Type | undefined =
175
+ ctx.checker.getElementTypeOfArrayType(type);
176
+ if (arrayItem === undefined) return { name: "Array<any>" };
177
+ return {
178
+ name: "Array",
179
+ typeArguments: [exploreType(ctx, arrayItem)],
180
+ };
181
+ } else if (symbol === undefined)
182
+ return {
183
+ name: ctx.checker.typeToString(
184
+ type,
185
+ undefined,
186
+ ts.TypeFormatFlags.NoTruncation,
187
+ ),
188
+ };
189
+
190
+ // Find matched import statement
191
+ const name: string = getNameOfSymbol(symbol);
192
+ const prefix: string = name.split(".")[0]!;
193
+
194
+ let matched: boolean = false;
195
+ const insert = (imp: IReflectImport): void => {
196
+ matched ||= true;
197
+ ctx.container.push(imp);
198
+ };
199
+ for (const imp of ctx.imports)
200
+ if (imp.elements.includes(prefix))
201
+ insert({
202
+ file: imp.file,
203
+ asterisk: null,
204
+ default: null,
205
+ elements: [prefix],
206
+ });
207
+ if (prefix !== "Promise" && matched === false)
208
+ emplaceSymbol(ctx, symbol, prefix);
209
+
210
+ // Finalize with generic arguments
211
+ const generic: readonly ts.Type[] = type.aliasSymbol
212
+ ? (type.aliasTypeArguments ?? [])
213
+ : ctx.checker.getTypeArguments(type as ts.TypeReference);
214
+ return generic.length
215
+ ? name === "Promise"
216
+ ? exploreType(ctx, generic[0]!)
217
+ : {
218
+ name,
219
+ typeArguments: generic.map((child) => exploreType(ctx, child)),
220
+ }
221
+ : { name };
222
+ };
223
+
224
+ const emplaceSymbol = (
225
+ ctx: IContext,
226
+ symbol: ts.Symbol,
227
+ prefix: string,
228
+ ): void => {
229
+ // GET SOURCE FILE
230
+ const sourceFile: ts.SourceFile | undefined =
231
+ symbol.declarations?.[0]?.getSourceFile();
232
+ if (sourceFile === undefined) return;
233
+ if (sourceFile.fileName.indexOf("/typescript/lib") === -1)
234
+ ctx.container.push({
235
+ file: sourceFile.fileName,
236
+ asterisk: null,
237
+ default: null,
238
+ elements: [prefix],
239
+ });
240
+ };
241
+ }
242
+
243
+ const getEscapedText = (type: IReflectType): string =>
244
+ type.typeArguments
245
+ ? `${type.name}<${type.typeArguments.map(getEscapedText).join(", ")}>`
246
+ : type.name;
247
+
248
+ const getNameOfSymbol = (symbol: ts.Symbol): string =>
249
+ exploreName(
250
+ symbol.escapedName.toString(),
251
+ symbol.getDeclarations()?.[0]?.parent,
252
+ );
253
+
254
+ const exploreName = (name: string, decl?: ts.Node): string =>
255
+ decl && ts.isModuleBlock(decl)
256
+ ? exploreName(
257
+ `${decl.parent.name.getFullText().trim()}.${name}`,
258
+ decl.parent.parent,
259
+ )
260
+ : name;