@nestia/core 12.0.0-dev.20260601.1 → 12.0.0-dev.20260612.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 (116) hide show
  1. package/LICENSE +21 -21
  2. package/MIGRATION.md +169 -169
  3. package/README.md +93 -93
  4. package/lib/adaptors/McpAdaptor.d.ts +75 -0
  5. package/lib/adaptors/McpAdaptor.js +257 -0
  6. package/lib/adaptors/McpAdaptor.js.map +1 -0
  7. package/lib/adaptors/WebSocketAdaptor.js +4 -4
  8. package/lib/adaptors/WebSocketAdaptor.js.map +1 -1
  9. package/lib/decorators/McpRoute.d.ts +69 -0
  10. package/lib/decorators/McpRoute.js +58 -0
  11. package/lib/decorators/McpRoute.js.map +1 -0
  12. package/lib/decorators/TypedParam.js +4 -4
  13. package/lib/decorators/TypedParam.js.map +1 -1
  14. package/lib/decorators/TypedRoute.js +1 -1
  15. package/lib/decorators/TypedRoute.js.map +1 -1
  16. package/lib/decorators/internal/IMcpRouteReflect.d.ts +2 -0
  17. package/lib/decorators/internal/IMcpRouteReflect.js +3 -0
  18. package/lib/decorators/internal/IMcpRouteReflect.js.map +1 -0
  19. package/lib/decorators/internal/get_path_and_querify.js +4 -4
  20. package/lib/decorators/internal/get_path_and_querify.js.map +1 -1
  21. package/lib/decorators/internal/get_path_and_stringify.js +4 -4
  22. package/lib/decorators/internal/get_path_and_stringify.js.map +1 -1
  23. package/lib/decorators/internal/load_controller.js +34 -65
  24. package/lib/decorators/internal/load_controller.js.map +1 -1
  25. package/lib/decorators/internal/validate_request_body.js +4 -4
  26. package/lib/decorators/internal/validate_request_body.js.map +1 -1
  27. package/lib/decorators/internal/validate_request_form_data.js +4 -4
  28. package/lib/decorators/internal/validate_request_form_data.js.map +1 -1
  29. package/lib/decorators/internal/validate_request_headers.js +4 -4
  30. package/lib/decorators/internal/validate_request_headers.js.map +1 -1
  31. package/lib/decorators/internal/validate_request_query.js +4 -4
  32. package/lib/decorators/internal/validate_request_query.js.map +1 -1
  33. package/lib/module.d.ts +2 -0
  34. package/lib/module.js +2 -0
  35. package/lib/module.js.map +1 -1
  36. package/native/cmd/ttsc-nestia/main.go +11 -11
  37. package/native/go.mod +32 -32
  38. package/native/go.sum +54 -54
  39. package/native/plugin/plan.go +102 -102
  40. package/native/transform/ast.go +32 -32
  41. package/native/transform/build.go +380 -444
  42. package/native/transform/cleanup.go +408 -408
  43. package/native/transform/contributor.go +97 -68
  44. package/native/transform/core_querify.go +231 -227
  45. package/native/transform/core_transform.go +1996 -1713
  46. package/native/transform/core_websocket.go +115 -115
  47. package/native/transform/exports.go +13 -13
  48. package/native/transform/mcp_transform.go +414 -0
  49. package/native/transform/node_transform.go +357 -0
  50. package/native/transform/path_rewrite.go +285 -285
  51. package/native/transform/printer.go +244 -244
  52. package/native/transform/rewrite.go +668 -662
  53. package/native/transform/run.go +73 -73
  54. package/native/transform/transform.go +336 -403
  55. package/native/transform/typia_fast.go +352 -326
  56. package/native/transform/typia_replacement.go +24 -24
  57. package/native/transform.cjs +43 -43
  58. package/package.json +15 -8
  59. package/src/adaptors/McpAdaptor.ts +276 -0
  60. package/src/adaptors/WebSocketAdaptor.ts +429 -429
  61. package/src/decorators/DynamicModule.ts +44 -44
  62. package/src/decorators/EncryptedBody.ts +97 -97
  63. package/src/decorators/EncryptedController.ts +40 -40
  64. package/src/decorators/EncryptedModule.ts +98 -98
  65. package/src/decorators/EncryptedRoute.ts +213 -213
  66. package/src/decorators/HumanRoute.ts +21 -21
  67. package/src/decorators/McpRoute.ts +154 -0
  68. package/src/decorators/NoTransformConfigurationError.ts +40 -40
  69. package/src/decorators/PlainBody.ts +76 -76
  70. package/src/decorators/SwaggerCustomizer.ts +97 -97
  71. package/src/decorators/SwaggerExample.ts +180 -180
  72. package/src/decorators/TypedBody.ts +57 -57
  73. package/src/decorators/TypedException.ts +147 -147
  74. package/src/decorators/TypedFormData.ts +187 -187
  75. package/src/decorators/TypedHeaders.ts +66 -66
  76. package/src/decorators/TypedParam.ts +77 -77
  77. package/src/decorators/TypedQuery.ts +234 -234
  78. package/src/decorators/TypedRoute.ts +198 -196
  79. package/src/decorators/WebSocketRoute.ts +242 -242
  80. package/src/decorators/doNotThrowTransformError.ts +5 -5
  81. package/src/decorators/internal/EncryptedConstant.ts +2 -2
  82. package/src/decorators/internal/IMcpRouteReflect.ts +40 -0
  83. package/src/decorators/internal/IWebSocketRouteReflect.ts +23 -23
  84. package/src/decorators/internal/get_path_and_querify.ts +94 -94
  85. package/src/decorators/internal/get_path_and_stringify.ts +110 -110
  86. package/src/decorators/internal/get_text_body.ts +16 -16
  87. package/src/decorators/internal/headers_to_object.ts +11 -11
  88. package/src/decorators/internal/is_request_body_undefined.ts +12 -12
  89. package/src/decorators/internal/load_controller.ts +91 -76
  90. package/src/decorators/internal/route_error.ts +43 -43
  91. package/src/decorators/internal/validate_request_body.ts +64 -64
  92. package/src/decorators/internal/validate_request_form_data.ts +67 -67
  93. package/src/decorators/internal/validate_request_headers.ts +76 -76
  94. package/src/decorators/internal/validate_request_query.ts +83 -83
  95. package/src/index.ts +5 -5
  96. package/src/module.ts +25 -23
  97. package/src/options/IRequestBodyValidator.ts +20 -20
  98. package/src/options/IRequestFormDataProps.ts +27 -27
  99. package/src/options/IRequestHeadersValidator.ts +22 -22
  100. package/src/options/IRequestQueryValidator.ts +20 -20
  101. package/src/options/IResponseBodyQuerifier.ts +25 -25
  102. package/src/options/IResponseBodyStringifier.ts +30 -30
  103. package/src/transform.ts +101 -101
  104. package/src/typings/Creator.ts +3 -3
  105. package/src/typings/get-function-location.d.ts +7 -7
  106. package/src/utils/ArrayUtil.ts +7 -7
  107. package/src/utils/ExceptionManager.ts +115 -115
  108. package/src/utils/Singleton.ts +16 -16
  109. package/src/utils/SourceFinder.ts +54 -54
  110. package/src/utils/VersioningStrategy.ts +27 -27
  111. package/native/transform/cleanup_test.go +0 -76
  112. package/native/transform/commonjs_import_alias_test.go +0 -49
  113. package/native/transform/core_dispatch_test.go +0 -127
  114. package/native/transform/path_rewrite_test.go +0 -243
  115. package/native/transform/rewrite_test.go +0 -118
  116. package/native/transform/rewrite_unique_base_test.go +0 -48
@@ -1,180 +1,180 @@
1
- /**
2
- * Attach example values to Swagger documents.
3
- *
4
- * `SwaggerExample` is a namespace of decorators that attach example values
5
- * to controller methods (request/response bodies, parameters), so that
6
- * `@nestia/sdk`'s Swagger generator can populate the `example` / `examples`
7
- * fields of the generated OpenAPI document.
8
- *
9
- * The decorators only affect Swagger document generation. They do not change
10
- * runtime behavior, validation, or SDK function signatures.
11
- *
12
- * @example
13
- *
14
- * ```typescript
15
- * import core from "@nestia/core";
16
- * import { Controller } from "@nestjs/common";
17
- * import typia from "typia";
18
- *
19
- * @Controller("bbs/articles")
20
- * export class BbsArticlesController {
21
- * // Single response example.
22
- * @core.SwaggerExample.Response(typia.random<IBbsArticle>())
23
- * @core.TypedRoute.Post()
24
- * public async create(
25
- * // Multiple named parameter examples plus a default one.
26
- * @core.SwaggerExample.Parameter(typia.random<IBbsArticle.ICreate>())
27
- * @core.SwaggerExample.Parameter("x", typia.random<IBbsArticle.ICreate>())
28
- * @core.SwaggerExample.Parameter("y", typia.random<IBbsArticle.ICreate>())
29
- * @core.TypedBody()
30
- * input: IBbsArticle.ICreate,
31
- * ): Promise<IBbsArticle> {
32
- * // ...
33
- * }
34
- * }
35
- * ```
36
- *
37
- * @author Jeongho Nam - https://github.com/samchon
38
- */
39
- export namespace SwaggerExample {
40
- /**
41
- * Attach an example value to the response body of a controller method.
42
- *
43
- * Two forms are supported:
44
- *
45
- * - {@link Response | `Response(value)`}: registers `value` as the single
46
- * default `example`.
47
- * - {@link Response | `Response(key, value)`}: registers `value` under
48
- * `examples[key]`. May be applied multiple times to attach several
49
- * named examples.
50
- *
51
- * Both forms can coexist on the same method — the default `example` and
52
- * any number of named `examples` will all surface in the generated
53
- * Swagger document.
54
- *
55
- * @template T Type of the example value (typically the route's response DTO).
56
- */
57
- export function Response<T>(value: T): MethodDecorator;
58
- export function Response<T>(key: string, value: T): MethodDecorator;
59
- export function Response(...args: any[]): MethodDecorator {
60
- return function SwaggerExampleResponse(
61
- _target: Object,
62
- _propertyKey: string | symbol,
63
- descriptor: TypedPropertyDescriptor<any>,
64
- ): TypedPropertyDescriptor<any> {
65
- emplaceValue(emplaceOfResponse(descriptor))(args);
66
- return descriptor;
67
- };
68
- }
69
-
70
- /**
71
- * Attach an example value to a request parameter (body, path, query, etc.)
72
- * of a controller method.
73
- *
74
- * Two forms are supported:
75
- *
76
- * - {@link Parameter | `Parameter(value)`}: registers `value` as the single
77
- * default `example` for that parameter.
78
- * - {@link Parameter | `Parameter(key, value)`}: registers `value` under
79
- * `examples[key]`. May be applied multiple times to attach several
80
- * named examples to the same parameter.
81
- *
82
- * Apply alongside the actual parameter decorator
83
- * (`@TypedBody()`, `@TypedParam()`, `@TypedQuery()`, etc.); decorator
84
- * order does not matter.
85
- *
86
- * @template T Type of the example value (typically the parameter's DTO).
87
- */
88
- export function Parameter<T>(value: T): ParameterDecorator;
89
- export function Parameter<T>(key: string, value: T): ParameterDecorator;
90
- export function Parameter(...args: any[]): ParameterDecorator {
91
- return function SwaggerExampleParameter(
92
- target: Object,
93
- propertyKey: string | symbol | undefined,
94
- index: number,
95
- ): void {
96
- emplaceValue(emplaceOfParameter(target, propertyKey ?? "", index))(args);
97
- };
98
- }
99
-
100
- /**
101
- * Internal storage shape for `SwaggerExample` metadata.
102
- *
103
- * Reflects the OpenAPI spec's two example fields: a single default
104
- * `example`, and/or a named map `examples`. `index` is used internally
105
- * to associate parameter examples with the right parameter position.
106
- */
107
- export interface IData<T> {
108
- examples?: Record<string, T>;
109
- example?: T;
110
- index?: number;
111
- }
112
- }
113
-
114
- const emplaceValue =
115
- <T>(data: SwaggerExample.IData<T>) =>
116
- (args: any[]) => {
117
- if (args.length === 1) data.example = args[0];
118
- else {
119
- const key: string = args[0];
120
- const value: T = args[1];
121
- data.examples ??= {};
122
- data.examples[key] = value;
123
- }
124
- };
125
-
126
- const emplaceOfResponse = <T>(
127
- descriptor: TypedPropertyDescriptor<any>,
128
- ): SwaggerExample.IData<T> => {
129
- const oldbie: SwaggerExample.IData<T> | undefined = Reflect.getMetadata(
130
- "nestia/SwaggerExample/Response",
131
- descriptor.value,
132
- );
133
- if (oldbie !== undefined) return oldbie;
134
- const newbie: SwaggerExample.IData<T> = {};
135
- Reflect.defineMetadata(
136
- "nestia/SwaggerExample/Response",
137
- newbie,
138
- descriptor.value,
139
- );
140
- return newbie;
141
- };
142
-
143
- const emplaceOfParameter = (
144
- target: Object,
145
- propertyKey: string | symbol,
146
- index: number,
147
- ): SwaggerExample.IData<any> => {
148
- const array: SwaggerExample.IData<any>[] = emplaceArrayOfParameters(
149
- target,
150
- propertyKey,
151
- );
152
- const oldibe: SwaggerExample.IData<any> | undefined = array.find(
153
- (e) => e.index === index,
154
- );
155
- if (oldibe !== undefined) return oldibe;
156
-
157
- const data: SwaggerExample.IData<any> = { index };
158
- array.push(data);
159
- return data;
160
- };
161
-
162
- const emplaceArrayOfParameters = (
163
- target: Object,
164
- propertyKey: string | symbol,
165
- ): SwaggerExample.IData<any>[] => {
166
- const array: SwaggerExample.IData<any>[] | undefined = Reflect.getMetadata(
167
- "nestia/SwaggerExample/Parameters",
168
- target,
169
- propertyKey,
170
- );
171
- if (array !== undefined) return array;
172
- const newbie: SwaggerExample.IData<any>[] = [];
173
- Reflect.defineMetadata(
174
- "nestia/SwaggerExample/Parameters",
175
- newbie,
176
- target,
177
- propertyKey,
178
- );
179
- return newbie;
180
- };
1
+ /**
2
+ * Attach example values to Swagger documents.
3
+ *
4
+ * `SwaggerExample` is a namespace of decorators that attach example values
5
+ * to controller methods (request/response bodies, parameters), so that
6
+ * `@nestia/sdk`'s Swagger generator can populate the `example` / `examples`
7
+ * fields of the generated OpenAPI document.
8
+ *
9
+ * The decorators only affect Swagger document generation. They do not change
10
+ * runtime behavior, validation, or SDK function signatures.
11
+ *
12
+ * @example
13
+ *
14
+ * ```typescript
15
+ * import core from "@nestia/core";
16
+ * import { Controller } from "@nestjs/common";
17
+ * import typia from "typia";
18
+ *
19
+ * @Controller("bbs/articles")
20
+ * export class BbsArticlesController {
21
+ * // Single response example.
22
+ * @core.SwaggerExample.Response(typia.random<IBbsArticle>())
23
+ * @core.TypedRoute.Post()
24
+ * public async create(
25
+ * // Multiple named parameter examples plus a default one.
26
+ * @core.SwaggerExample.Parameter(typia.random<IBbsArticle.ICreate>())
27
+ * @core.SwaggerExample.Parameter("x", typia.random<IBbsArticle.ICreate>())
28
+ * @core.SwaggerExample.Parameter("y", typia.random<IBbsArticle.ICreate>())
29
+ * @core.TypedBody()
30
+ * input: IBbsArticle.ICreate,
31
+ * ): Promise<IBbsArticle> {
32
+ * // ...
33
+ * }
34
+ * }
35
+ * ```
36
+ *
37
+ * @author Jeongho Nam - https://github.com/samchon
38
+ */
39
+ export namespace SwaggerExample {
40
+ /**
41
+ * Attach an example value to the response body of a controller method.
42
+ *
43
+ * Two forms are supported:
44
+ *
45
+ * - {@link Response | `Response(value)`}: registers `value` as the single
46
+ * default `example`.
47
+ * - {@link Response | `Response(key, value)`}: registers `value` under
48
+ * `examples[key]`. May be applied multiple times to attach several
49
+ * named examples.
50
+ *
51
+ * Both forms can coexist on the same method — the default `example` and
52
+ * any number of named `examples` will all surface in the generated
53
+ * Swagger document.
54
+ *
55
+ * @template T Type of the example value (typically the route's response DTO).
56
+ */
57
+ export function Response<T>(value: T): MethodDecorator;
58
+ export function Response<T>(key: string, value: T): MethodDecorator;
59
+ export function Response(...args: any[]): MethodDecorator {
60
+ return function SwaggerExampleResponse(
61
+ _target: Object,
62
+ _propertyKey: string | symbol,
63
+ descriptor: TypedPropertyDescriptor<any>,
64
+ ): TypedPropertyDescriptor<any> {
65
+ emplaceValue(emplaceOfResponse(descriptor))(args);
66
+ return descriptor;
67
+ };
68
+ }
69
+
70
+ /**
71
+ * Attach an example value to a request parameter (body, path, query, etc.)
72
+ * of a controller method.
73
+ *
74
+ * Two forms are supported:
75
+ *
76
+ * - {@link Parameter | `Parameter(value)`}: registers `value` as the single
77
+ * default `example` for that parameter.
78
+ * - {@link Parameter | `Parameter(key, value)`}: registers `value` under
79
+ * `examples[key]`. May be applied multiple times to attach several
80
+ * named examples to the same parameter.
81
+ *
82
+ * Apply alongside the actual parameter decorator
83
+ * (`@TypedBody()`, `@TypedParam()`, `@TypedQuery()`, etc.); decorator
84
+ * order does not matter.
85
+ *
86
+ * @template T Type of the example value (typically the parameter's DTO).
87
+ */
88
+ export function Parameter<T>(value: T): ParameterDecorator;
89
+ export function Parameter<T>(key: string, value: T): ParameterDecorator;
90
+ export function Parameter(...args: any[]): ParameterDecorator {
91
+ return function SwaggerExampleParameter(
92
+ target: Object,
93
+ propertyKey: string | symbol | undefined,
94
+ index: number,
95
+ ): void {
96
+ emplaceValue(emplaceOfParameter(target, propertyKey ?? "", index))(args);
97
+ };
98
+ }
99
+
100
+ /**
101
+ * Internal storage shape for `SwaggerExample` metadata.
102
+ *
103
+ * Reflects the OpenAPI spec's two example fields: a single default
104
+ * `example`, and/or a named map `examples`. `index` is used internally
105
+ * to associate parameter examples with the right parameter position.
106
+ */
107
+ export interface IData<T> {
108
+ examples?: Record<string, T>;
109
+ example?: T;
110
+ index?: number;
111
+ }
112
+ }
113
+
114
+ const emplaceValue =
115
+ <T>(data: SwaggerExample.IData<T>) =>
116
+ (args: any[]) => {
117
+ if (args.length === 1) data.example = args[0];
118
+ else {
119
+ const key: string = args[0];
120
+ const value: T = args[1];
121
+ data.examples ??= {};
122
+ data.examples[key] = value;
123
+ }
124
+ };
125
+
126
+ const emplaceOfResponse = <T>(
127
+ descriptor: TypedPropertyDescriptor<any>,
128
+ ): SwaggerExample.IData<T> => {
129
+ const oldbie: SwaggerExample.IData<T> | undefined = Reflect.getMetadata(
130
+ "nestia/SwaggerExample/Response",
131
+ descriptor.value,
132
+ );
133
+ if (oldbie !== undefined) return oldbie;
134
+ const newbie: SwaggerExample.IData<T> = {};
135
+ Reflect.defineMetadata(
136
+ "nestia/SwaggerExample/Response",
137
+ newbie,
138
+ descriptor.value,
139
+ );
140
+ return newbie;
141
+ };
142
+
143
+ const emplaceOfParameter = (
144
+ target: Object,
145
+ propertyKey: string | symbol,
146
+ index: number,
147
+ ): SwaggerExample.IData<any> => {
148
+ const array: SwaggerExample.IData<any>[] = emplaceArrayOfParameters(
149
+ target,
150
+ propertyKey,
151
+ );
152
+ const oldibe: SwaggerExample.IData<any> | undefined = array.find(
153
+ (e) => e.index === index,
154
+ );
155
+ if (oldibe !== undefined) return oldibe;
156
+
157
+ const data: SwaggerExample.IData<any> = { index };
158
+ array.push(data);
159
+ return data;
160
+ };
161
+
162
+ const emplaceArrayOfParameters = (
163
+ target: Object,
164
+ propertyKey: string | symbol,
165
+ ): SwaggerExample.IData<any>[] => {
166
+ const array: SwaggerExample.IData<any>[] | undefined = Reflect.getMetadata(
167
+ "nestia/SwaggerExample/Parameters",
168
+ target,
169
+ propertyKey,
170
+ );
171
+ if (array !== undefined) return array;
172
+ const newbie: SwaggerExample.IData<any>[] = [];
173
+ Reflect.defineMetadata(
174
+ "nestia/SwaggerExample/Parameters",
175
+ newbie,
176
+ target,
177
+ propertyKey,
178
+ );
179
+ return newbie;
180
+ };
@@ -1,57 +1,57 @@
1
- import {
2
- BadRequestException,
3
- ExecutionContext,
4
- createParamDecorator,
5
- } from "@nestjs/common";
6
- import type express from "express";
7
- import type { FastifyRequest } from "fastify";
8
-
9
- import { IRequestBodyValidator } from "../options/IRequestBodyValidator";
10
- import { is_request_body_undefined } from "./internal/is_request_body_undefined";
11
- import { validate_request_body } from "./internal/validate_request_body";
12
-
13
- /**
14
- * Type safe body decorator.
15
- *
16
- * `TypedBody` is a decorator function getting `application/json` typed data
17
- * from request body. Also, it validates the request body data type through
18
- * [typia](https://github.com/samchon/typia) and the validation speed is maximum
19
- * 20,000x times faster than `class-validator`.
20
- *
21
- * For reference, when the request body data is not following the promised type
22
- * `T`, `BadRequestException` error (status code: 400) would be thrown.
23
- *
24
- * @author Jeongho Nam - https://github.com/samchon
25
- * @param validator Custom validator if required. Default is `typia.assert()`
26
- */
27
- export function TypedBody<T>(
28
- validator?: IRequestBodyValidator<T>,
29
- ): ParameterDecorator {
30
- const checker = validate_request_body("TypedBody")(validator);
31
- return createParamDecorator(function TypedBody(
32
- _unknown: any,
33
- context: ExecutionContext,
34
- ) {
35
- const request: express.Request | FastifyRequest = context
36
- .switchToHttp()
37
- .getRequest();
38
- if (is_request_body_undefined(request) && checker(undefined as T) === null)
39
- return undefined;
40
- else if (isApplicationJson(request.headers["content-type"]) === false)
41
- throw new BadRequestException(
42
- `Request body type is not "application/json".`,
43
- );
44
-
45
- const error: Error | null = checker(request.body);
46
- if (error !== null) throw error;
47
- return request.body;
48
- })();
49
- }
50
-
51
- /** @internal */
52
- const isApplicationJson = (text?: string): boolean =>
53
- text !== undefined &&
54
- text
55
- .split(";")
56
- .map((str) => str.trim())
57
- .some((str) => str === "application/json");
1
+ import {
2
+ BadRequestException,
3
+ ExecutionContext,
4
+ createParamDecorator,
5
+ } from "@nestjs/common";
6
+ import type express from "express";
7
+ import type { FastifyRequest } from "fastify";
8
+
9
+ import { IRequestBodyValidator } from "../options/IRequestBodyValidator";
10
+ import { is_request_body_undefined } from "./internal/is_request_body_undefined";
11
+ import { validate_request_body } from "./internal/validate_request_body";
12
+
13
+ /**
14
+ * Type safe body decorator.
15
+ *
16
+ * `TypedBody` is a decorator function getting `application/json` typed data
17
+ * from request body. Also, it validates the request body data type through
18
+ * [typia](https://github.com/samchon/typia) and the validation speed is maximum
19
+ * 20,000x times faster than `class-validator`.
20
+ *
21
+ * For reference, when the request body data is not following the promised type
22
+ * `T`, `BadRequestException` error (status code: 400) would be thrown.
23
+ *
24
+ * @author Jeongho Nam - https://github.com/samchon
25
+ * @param validator Custom validator if required. Default is `typia.assert()`
26
+ */
27
+ export function TypedBody<T>(
28
+ validator?: IRequestBodyValidator<T>,
29
+ ): ParameterDecorator {
30
+ const checker = validate_request_body("TypedBody")(validator);
31
+ return createParamDecorator(function TypedBody(
32
+ _unknown: any,
33
+ context: ExecutionContext,
34
+ ) {
35
+ const request: express.Request | FastifyRequest = context
36
+ .switchToHttp()
37
+ .getRequest();
38
+ if (is_request_body_undefined(request) && checker(undefined as T) === null)
39
+ return undefined;
40
+ else if (isApplicationJson(request.headers["content-type"]) === false)
41
+ throw new BadRequestException(
42
+ `Request body type is not "application/json".`,
43
+ );
44
+
45
+ const error: Error | null = checker(request.body);
46
+ if (error !== null) throw error;
47
+ return request.body;
48
+ })();
49
+ }
50
+
51
+ /** @internal */
52
+ const isApplicationJson = (text?: string): boolean =>
53
+ text !== undefined &&
54
+ text
55
+ .split(";")
56
+ .map((str) => str.trim())
57
+ .some((str) => str === "application/json");