@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,119 +1,119 @@
1
- import { OpenApi } from "@samchon/openapi";
2
- import { Metadata } from "typia/lib/schemas/metadata/Metadata";
3
-
4
- import { INestiaConfig } from "../../INestiaConfig";
5
- import { SecurityAnalyzer } from "../../analyses/SecurityAnalyzer";
6
- import { ITypedHttpRoute } from "../../structures/ITypedHttpRoute";
7
- import { SwaggerDescriptionComposer } from "./SwaggerDescriptionComposer";
8
- import { SwaggerOperationParameterComposer } from "./SwaggerOperationParameterComposer";
9
- import { SwaggerOperationResponseComposer } from "./SwaggerOperationResponseComposer";
10
-
11
- export namespace SwaggerOperationComposer {
12
- export const compose = (props: {
13
- config: Omit<INestiaConfig.ISwaggerConfig, "output">;
14
- document: OpenApi.IDocument;
15
- schema: (metadata: Metadata) => OpenApi.IJsonSchema | undefined;
16
- route: ITypedHttpRoute;
17
- }): OpenApi.IOperation => {
18
- // COMPOSE TAGS
19
- const tags: Set<string> = new Set([
20
- ...props.route.controller.tags,
21
- ...props.route.tags,
22
- ...SwaggerDescriptionComposer.getJsDocTexts({
23
- jsDocTags: props.route.jsDocTags,
24
- name: "tag",
25
- }).map((t) => t.split(" ")[0]),
26
- ]);
27
- if (tags.size) {
28
- props.document.tags ??= [];
29
- for (const t of tags)
30
- if (props.document.tags.find((elem) => elem.name === t) === undefined)
31
- props.document.tags.push({ name: t });
32
- for (const texts of SwaggerDescriptionComposer.getJsDocTexts({
33
- jsDocTags: props.route.jsDocTags,
34
- name: "tag",
35
- })) {
36
- const [name, ...description] = texts.split(" ");
37
- if (description.length)
38
- props.document.tags.find(
39
- (elem) => elem.name === name,
40
- )!.description ??= description.join(" ");
41
- }
42
- }
43
-
44
- // SECURITY
45
- const security: Record<string, string[]>[] = SecurityAnalyzer.merge(
46
- ...props.route.controller.security,
47
- ...props.route.security,
48
- ...props.route.jsDocTags
49
- .filter((tag) => tag.name === "security")
50
- .map((tag) =>
51
- tag.text === undefined
52
- ? [{}]
53
- : tag.text.map((text) => {
54
- const line: string[] = text.text
55
- .split(" ")
56
- .filter((s) => s.trim())
57
- .filter((s) => !!s.length);
58
- if (line.length === 0) return {};
59
- return {
60
- [line[0]]: line.slice(1),
61
- };
62
- }),
63
- )
64
- .flat(),
65
- );
66
-
67
- // FINALIZE
68
- return {
69
- ...SwaggerDescriptionComposer.compose({
70
- description: props.route.description,
71
- jsDocTags: props.route.jsDocTags,
72
- kind: "summary",
73
- }),
74
- deprecated: props.route.jsDocTags.some((tag) => tag.name === "deprecated")
75
- ? true
76
- : undefined,
77
- tags: Array.from(tags),
78
- operationId:
79
- props.route.operationId ??
80
- props.config.operationId?.({
81
- class: props.route.controller.class.name,
82
- function: props.route.name,
83
- method: props.route.method as "GET",
84
- path: props.route.path,
85
- }),
86
- parameters: [
87
- ...props.route.pathParameters,
88
- ...props.route.queryParameters,
89
- ...(props.route.queryObject ? [props.route.queryObject] : []),
90
- ...(props.route.headerObject ? [props.route.headerObject] : []),
91
- ]
92
- .map((p) =>
93
- SwaggerOperationParameterComposer.compose({
94
- config: props.config,
95
- document: props.document,
96
- schema: props.schema(p.metadata)!,
97
- parameter: p,
98
- jsDocTags: props.route.jsDocTags,
99
- }),
100
- )
101
- .flat(),
102
- requestBody: props.route.body
103
- ? SwaggerOperationParameterComposer.body({
104
- schema: props.schema(props.route.body.metadata)!,
105
- jsDocTags: props.route.jsDocTags,
106
- parameter: props.route.body,
107
- })
108
- : undefined,
109
- responses: SwaggerOperationResponseComposer.compose({
110
- schema: props.schema,
111
- route: props.route,
112
- }),
113
- security: security.length ? security : undefined,
114
- ...(props.route.extensions ?? {}),
115
- "x-samchon-accessor": props.route.accessor,
116
- "x-samchon-controller": props.route.controller.class.name,
117
- };
118
- };
119
- }
1
+ import { MetadataSchema } from "@typia/core";
2
+ import { OpenApi } from "@typia/interface";
3
+
4
+ import { INestiaConfig } from "../../INestiaConfig";
5
+ import { SecurityAnalyzer } from "../../analyses/SecurityAnalyzer";
6
+ import { ITypedHttpRoute } from "../../structures/ITypedHttpRoute";
7
+ import { SwaggerDescriptionComposer } from "./SwaggerDescriptionComposer";
8
+ import { SwaggerOperationParameterComposer } from "./SwaggerOperationParameterComposer";
9
+ import { SwaggerOperationResponseComposer } from "./SwaggerOperationResponseComposer";
10
+
11
+ export namespace SwaggerOperationComposer {
12
+ export const compose = (props: {
13
+ config: Omit<INestiaConfig.ISwaggerConfig, "output">;
14
+ document: OpenApi.IDocument;
15
+ schema: (metadata: MetadataSchema) => OpenApi.IJsonSchema | undefined;
16
+ route: ITypedHttpRoute;
17
+ }): OpenApi.IOperation => {
18
+ // COMPOSE TAGS
19
+ const tags: Set<string> = new Set([
20
+ ...props.route.controller.tags,
21
+ ...props.route.tags,
22
+ ...SwaggerDescriptionComposer.getJsDocTexts({
23
+ jsDocTags: props.route.jsDocTags,
24
+ name: "tag",
25
+ }).map((t) => t.split(" ")[0]!),
26
+ ]);
27
+ if (tags.size) {
28
+ props.document.tags ??= [];
29
+ for (const t of tags)
30
+ if (props.document.tags.find((elem) => elem.name === t) === undefined)
31
+ props.document.tags.push({ name: t });
32
+ for (const texts of SwaggerDescriptionComposer.getJsDocTexts({
33
+ jsDocTags: props.route.jsDocTags,
34
+ name: "tag",
35
+ })) {
36
+ const [name, ...description] = texts.split(" ");
37
+ if (description.length)
38
+ props.document.tags.find(
39
+ (elem) => elem.name === name,
40
+ )!.description ??= description.join(" ");
41
+ }
42
+ }
43
+
44
+ // SECURITY
45
+ const security: Record<string, string[]>[] = SecurityAnalyzer.merge(
46
+ ...props.route.controller.security,
47
+ ...props.route.security,
48
+ ...props.route.jsDocTags
49
+ .filter((tag) => tag.name === "security")
50
+ .map((tag) =>
51
+ tag.text === undefined
52
+ ? [{}]
53
+ : tag.text.map((text) => {
54
+ const line: string[] = text.text
55
+ .split(" ")
56
+ .filter((s) => s.trim())
57
+ .filter((s) => !!s.length);
58
+ if (line.length === 0) return {};
59
+ return {
60
+ [line[0]!]: line.slice(1),
61
+ };
62
+ }),
63
+ )
64
+ .flat(),
65
+ );
66
+
67
+ // FINALIZE
68
+ return {
69
+ ...SwaggerDescriptionComposer.compose({
70
+ description: props.route.description,
71
+ jsDocTags: props.route.jsDocTags,
72
+ kind: "summary",
73
+ }),
74
+ deprecated: props.route.jsDocTags.some((tag) => tag.name === "deprecated")
75
+ ? true
76
+ : undefined,
77
+ tags: Array.from(tags),
78
+ operationId:
79
+ props.route.operationId ??
80
+ props.config.operationId?.({
81
+ class: props.route.controller.class.name,
82
+ function: props.route.name,
83
+ method: props.route.method as "GET",
84
+ path: props.route.path,
85
+ }),
86
+ parameters: [
87
+ ...props.route.pathParameters,
88
+ ...props.route.queryParameters,
89
+ ...(props.route.queryObject ? [props.route.queryObject] : []),
90
+ ...(props.route.headerObject ? [props.route.headerObject] : []),
91
+ ]
92
+ .map((p) =>
93
+ SwaggerOperationParameterComposer.compose({
94
+ config: props.config,
95
+ document: props.document,
96
+ schema: props.schema(p.metadata)!,
97
+ parameter: p,
98
+ jsDocTags: props.route.jsDocTags,
99
+ }),
100
+ )
101
+ .flat(),
102
+ requestBody: props.route.body
103
+ ? SwaggerOperationParameterComposer.body({
104
+ schema: props.schema(props.route.body.metadata)!,
105
+ jsDocTags: props.route.jsDocTags,
106
+ parameter: props.route.body,
107
+ })
108
+ : undefined,
109
+ responses: SwaggerOperationResponseComposer.compose({
110
+ schema: props.schema,
111
+ route: props.route,
112
+ }),
113
+ security: security.length ? security : undefined,
114
+ ...(props.route.extensions ?? {}),
115
+ "x-samchon-accessor": props.route.accessor,
116
+ "x-samchon-controller": props.route.controller.class.name,
117
+ };
118
+ };
119
+ }
@@ -1,162 +1,161 @@
1
- import { OpenApi } from "@samchon/openapi";
2
- import { VariadicSingleton } from "tstl";
3
- import { IJsDocTagInfo, IJsonSchemaCollection } from "typia";
4
- import { JsonSchemasProgrammer } from "typia/lib/programmers/json/JsonSchemasProgrammer";
5
-
6
- import { INestiaConfig } from "../../INestiaConfig";
7
- import { ITypedHttpRouteParameter } from "../../structures/ITypedHttpRouteParameter";
8
- import { SwaggerDescriptionComposer } from "./SwaggerDescriptionComposer";
9
-
10
- export namespace SwaggerOperationParameterComposer {
11
- export interface IProps<Parameter extends ITypedHttpRouteParameter> {
12
- config: Omit<INestiaConfig.ISwaggerConfig, "output">;
13
- document: OpenApi.IDocument;
14
- schema: OpenApi.IJsonSchema;
15
- jsDocTags: IJsDocTagInfo[];
16
- parameter: Parameter;
17
- }
18
-
19
- export const compose = (
20
- props: IProps<ITypedHttpRouteParameter>,
21
- ): OpenApi.IOperation.IParameter[] =>
22
- props.parameter.category === "body"
23
- ? []
24
- : props.parameter.category === "param"
25
- ? [path({ ...props, parameter: props.parameter })]
26
- : props.parameter.category === "query"
27
- ? query({ ...props, parameter: props.parameter })
28
- : header({ ...props, parameter: props.parameter });
29
-
30
- export const body = (
31
- props: Omit<IProps<ITypedHttpRouteParameter.IBody>, "config" | "document">,
32
- ): OpenApi.IOperation.IRequestBody => {
33
- const description: string | undefined =
34
- SwaggerDescriptionComposer.descriptionFromJsDocTag({
35
- jsDocTags: props.jsDocTags,
36
- tag: "param",
37
- parameter: props.parameter.name,
38
- });
39
- return {
40
- description: props.parameter.encrypted
41
- ? `${warning.get(!!description)}${description ?? ""}`
42
- : description,
43
- content: {
44
- [props.parameter.contentType]: {
45
- schema: props.schema,
46
- example: props.parameter.example,
47
- examples: props.parameter.examples,
48
- },
49
- },
50
- required: props.parameter.metadata.isRequired(),
51
- ...(props.parameter.encrypted ? { "x-nestia-encrypted": true } : {}),
52
- };
53
- };
54
-
55
- export const path = (
56
- props: Omit<IProps<ITypedHttpRouteParameter.IPath>, "config" | "document">,
57
- ): OpenApi.IOperation.IParameter => ({
58
- name: props.parameter.field,
59
- in: "path",
60
- schema: props.schema,
61
- required: props.parameter.metadata.isRequired(),
62
- description: parameterDescription(props),
63
- example: props.parameter.example,
64
- examples: props.parameter.examples,
65
- });
66
-
67
- export const query = (
68
- props: IProps<ITypedHttpRouteParameter.IQuery>,
69
- ): OpenApi.IOperation.IParameter[] => decomposible(props);
70
-
71
- export const header = (
72
- props: IProps<ITypedHttpRouteParameter.IHeaders>,
73
- ): OpenApi.IOperation.IParameter[] => decomposible(props);
74
-
75
- const parameterDescription = (
76
- props: Pick<IProps<ITypedHttpRouteParameter>, "parameter" | "jsDocTags">,
77
- ): string | undefined => {
78
- return (
79
- props.parameter.description ??
80
- props.parameter.jsDocTags.find((tag) => tag.name === "description")
81
- ?.text?.[0].text ??
82
- props.jsDocTags
83
- .find(
84
- (tag) =>
85
- tag.name === "param" && tag.text?.[0].text === props.parameter.name,
86
- )
87
- ?.text?.map((e) => e.text)
88
- .join("")
89
- .substring(props.parameter.name.length)
90
- .trim()
91
- );
92
- };
93
-
94
- const decomposible = (
95
- props: IProps<
96
- ITypedHttpRouteParameter.IHeaders | ITypedHttpRouteParameter.IQuery
97
- >,
98
- ): OpenApi.IOperation.IParameter[] => {
99
- const param: OpenApi.IOperation.IParameter = {
100
- name: props.parameter.field ?? props.parameter.name,
101
- in: props.parameter.category === "query" ? "query" : "header",
102
- schema: props.schema,
103
- description: parameterDescription(props),
104
- required: props.parameter.metadata.isRequired(),
105
- example: props.parameter.example,
106
- examples: props.parameter.examples,
107
- };
108
- if (
109
- props.config.decompose === false ||
110
- props.parameter.metadata.objects.length === 0
111
- )
112
- return [param];
113
- return props.parameter.metadata.objects[0].type.properties
114
- .filter((p) =>
115
- p.jsDocTags.every(
116
- (tag) => tag.name !== "hidden" && tag.name !== "ignore",
117
- ),
118
- )
119
- .map((p) => {
120
- const json: IJsonSchemaCollection = JsonSchemasProgrammer.write({
121
- version: "3.1",
122
- metadatas: [p.value],
123
- }) as IJsonSchemaCollection;
124
- if (Object.keys(json.components.schemas ?? {}).length !== 0) {
125
- props.document.components ??= {};
126
- props.document.components.schemas ??= {};
127
- Object.assign(
128
- props.document.components.schemas,
129
- json.components.schemas,
130
- );
131
- }
132
- return {
133
- name: p.key.constants[0].values[0].value as string,
134
- in: props.parameter.category === "query" ? "query" : "header",
135
- schema: json.schemas[0],
136
- required: p.value.isRequired(),
137
- description: SwaggerDescriptionComposer.compose({
138
- description: p.description ?? null,
139
- jsDocTags: p.jsDocTags,
140
- kind: "title",
141
- }).description,
142
- };
143
- });
144
- };
145
- }
146
-
147
- const warning = new VariadicSingleton((described: boolean): string => {
148
- const summary = "Request body must be encrypted.";
149
- const component =
150
- "[EncryptedBody](https://github.com/samchon/@nestia/core#encryptedbody)";
151
- const content: string[] = [
152
- "## Warning",
153
- "",
154
- summary,
155
- "",
156
- `The request body data would be encrypted as "AES-128(256) / CBC mode / PKCS#5 Padding / Base64 Encoding", through the ${component} component.`,
157
- "",
158
- `Therefore, just utilize this swagger editor only for referencing. If you need to call the real API, using [SDK](https://github.com/samchon/nestia#software-development-kit) would be much better.`,
159
- ];
160
- if (described === true) content.push("", "----------------", "", "");
161
- return content.join("\n");
162
- });
1
+ import { JsonSchemasProgrammer } from "@typia/core";
2
+ import { OpenApi } from "@typia/interface";
3
+ import { VariadicSingleton } from "tstl";
4
+ import { IJsDocTagInfo, IJsonSchemaCollection } from "typia";
5
+
6
+ import { INestiaConfig } from "../../INestiaConfig";
7
+ import { ITypedHttpRouteParameter } from "../../structures/ITypedHttpRouteParameter";
8
+ import { SwaggerDescriptionComposer } from "./SwaggerDescriptionComposer";
9
+
10
+ export namespace SwaggerOperationParameterComposer {
11
+ export interface IProps<Parameter extends ITypedHttpRouteParameter> {
12
+ config: Omit<INestiaConfig.ISwaggerConfig, "output">;
13
+ document: OpenApi.IDocument;
14
+ schema: OpenApi.IJsonSchema;
15
+ jsDocTags: IJsDocTagInfo[];
16
+ parameter: Parameter;
17
+ }
18
+
19
+ export const compose = (
20
+ props: IProps<ITypedHttpRouteParameter>,
21
+ ): OpenApi.IOperation.IParameter[] =>
22
+ props.parameter.category === "body"
23
+ ? []
24
+ : props.parameter.category === "param"
25
+ ? [path({ ...props, parameter: props.parameter })]
26
+ : props.parameter.category === "query"
27
+ ? query({ ...props, parameter: props.parameter })
28
+ : header({ ...props, parameter: props.parameter });
29
+
30
+ export const body = (
31
+ props: Omit<IProps<ITypedHttpRouteParameter.IBody>, "config" | "document">,
32
+ ): OpenApi.IOperation.IRequestBody => {
33
+ const description: string | undefined =
34
+ SwaggerDescriptionComposer.descriptionFromJsDocTag({
35
+ jsDocTags: props.jsDocTags,
36
+ tag: "param",
37
+ parameter: props.parameter.name,
38
+ });
39
+ return {
40
+ description: props.parameter.encrypted
41
+ ? `${warning.get(!!description)}${description ?? ""}`
42
+ : description,
43
+ content: {
44
+ [props.parameter.contentType]: {
45
+ schema: props.schema,
46
+ example: props.parameter.example,
47
+ examples: props.parameter.examples,
48
+ },
49
+ },
50
+ required: props.parameter.metadata.isRequired(),
51
+ ...(props.parameter.encrypted ? { "x-nestia-encrypted": true } : {}),
52
+ };
53
+ };
54
+
55
+ export const path = (
56
+ props: Omit<IProps<ITypedHttpRouteParameter.IPath>, "config" | "document">,
57
+ ): OpenApi.IOperation.IParameter => ({
58
+ name: props.parameter.field,
59
+ in: "path",
60
+ schema: props.schema,
61
+ required: props.parameter.metadata.isRequired(),
62
+ description: parameterDescription(props),
63
+ example: props.parameter.example,
64
+ examples: props.parameter.examples,
65
+ });
66
+
67
+ export const query = (
68
+ props: IProps<ITypedHttpRouteParameter.IQuery>,
69
+ ): OpenApi.IOperation.IParameter[] => decomposible(props);
70
+
71
+ export const header = (
72
+ props: IProps<ITypedHttpRouteParameter.IHeaders>,
73
+ ): OpenApi.IOperation.IParameter[] => decomposible(props);
74
+
75
+ const parameterDescription = (
76
+ props: Pick<IProps<ITypedHttpRouteParameter>, "parameter" | "jsDocTags">,
77
+ ): string | undefined => {
78
+ return (
79
+ props.parameter.description ??
80
+ props.parameter.jsDocTags.find((tag) => tag.name === "description")
81
+ ?.text?.[0]?.text ??
82
+ props.jsDocTags
83
+ .find(
84
+ (tag) =>
85
+ tag.name === "param" &&
86
+ tag.text?.[0]?.text === props.parameter.name,
87
+ )
88
+ ?.text?.map((e) => e.text)
89
+ .join("")
90
+ .substring(props.parameter.name.length)
91
+ .trim()
92
+ );
93
+ };
94
+
95
+ const decomposible = (
96
+ props: IProps<
97
+ ITypedHttpRouteParameter.IHeaders | ITypedHttpRouteParameter.IQuery
98
+ >,
99
+ ): OpenApi.IOperation.IParameter[] => {
100
+ const param: OpenApi.IOperation.IParameter = {
101
+ name: props.parameter.field ?? props.parameter.name,
102
+ in: props.parameter.category === "query" ? "query" : "header",
103
+ schema: props.schema,
104
+ description: parameterDescription(props),
105
+ required: props.parameter.metadata.isRequired(),
106
+ example: props.parameter.example,
107
+ examples: props.parameter.examples,
108
+ };
109
+ if (
110
+ props.config.decompose === false ||
111
+ props.parameter.metadata.objects.length === 0
112
+ )
113
+ return [param];
114
+ return props.parameter.metadata.objects[0]!.type.properties.filter((p) =>
115
+ p.jsDocTags.every(
116
+ (tag) => tag.name !== "hidden" && tag.name !== "ignore",
117
+ ),
118
+ ).map((p) => {
119
+ const json: IJsonSchemaCollection = JsonSchemasProgrammer.writeSchemas({
120
+ version: "3.1",
121
+ metadatas: [p.value],
122
+ }) as IJsonSchemaCollection;
123
+ if (Object.keys(json.components.schemas ?? {}).length !== 0) {
124
+ props.document.components ??= {};
125
+ props.document.components.schemas ??= {};
126
+ Object.assign(
127
+ props.document.components.schemas,
128
+ json.components.schemas,
129
+ );
130
+ }
131
+ return {
132
+ name: p.key.constants[0]!.values[0]!.value as string,
133
+ in: props.parameter.category === "query" ? "query" : "header",
134
+ schema: json.schemas[0]!,
135
+ required: p.value.isRequired(),
136
+ description: SwaggerDescriptionComposer.compose({
137
+ description: p.description ?? null,
138
+ jsDocTags: p.jsDocTags,
139
+ kind: "title",
140
+ }).description,
141
+ };
142
+ });
143
+ };
144
+ }
145
+
146
+ const warning = new VariadicSingleton((described: boolean): string => {
147
+ const summary = "Request body must be encrypted.";
148
+ const component =
149
+ "[EncryptedBody](https://github.com/samchon/@nestia/core#encryptedbody)";
150
+ const content: string[] = [
151
+ "## Warning",
152
+ "",
153
+ summary,
154
+ "",
155
+ `The request body data would be encrypted as "AES-128(256) / CBC mode / PKCS#5 Padding / Base64 Encoding", through the ${component} component.`,
156
+ "",
157
+ `Therefore, just utilize this swagger editor only for referencing. If you need to call the real API, using [SDK](https://github.com/samchon/nestia#software-development-kit) would be much better.`,
158
+ ];
159
+ if (described === true) content.push("", "----------------", "", "");
160
+ return content.join("\n");
161
+ });