@nestia/core 3.0.5-dev.20240418 → 3.1.0-dev.20240426

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 (139) hide show
  1. package/lib/adaptors/WebSocketAdaptor.d.ts +21 -0
  2. package/lib/adaptors/WebSocketAdaptor.js +277 -0
  3. package/lib/adaptors/WebSocketAdaptor.js.map +1 -0
  4. package/lib/decorators/DynamicModule.js +12 -59
  5. package/lib/decorators/DynamicModule.js.map +1 -1
  6. package/lib/decorators/EncryptedBody.js +36 -73
  7. package/lib/decorators/EncryptedBody.js.map +1 -1
  8. package/lib/decorators/EncryptedController.js +2 -2
  9. package/lib/decorators/EncryptedController.js.map +1 -1
  10. package/lib/decorators/EncryptedModule.js +24 -107
  11. package/lib/decorators/EncryptedModule.js.map +1 -1
  12. package/lib/decorators/EncryptedRoute.js +43 -119
  13. package/lib/decorators/EncryptedRoute.js.map +1 -1
  14. package/lib/decorators/PlainBody.js +24 -61
  15. package/lib/decorators/PlainBody.js.map +1 -1
  16. package/lib/decorators/SwaggerCustomizer.js +2 -2
  17. package/lib/decorators/SwaggerCustomizer.js.map +1 -1
  18. package/lib/decorators/TypedBody.js +12 -14
  19. package/lib/decorators/TypedBody.js.map +1 -1
  20. package/lib/decorators/TypedException.js +5 -5
  21. package/lib/decorators/TypedException.js.map +1 -1
  22. package/lib/decorators/TypedFormData.js +88 -291
  23. package/lib/decorators/TypedFormData.js.map +1 -1
  24. package/lib/decorators/TypedHeaders.js +6 -6
  25. package/lib/decorators/TypedHeaders.js.map +1 -1
  26. package/lib/decorators/TypedParam.d.ts +5 -6
  27. package/lib/decorators/TypedParam.js +14 -15
  28. package/lib/decorators/TypedParam.js.map +1 -1
  29. package/lib/decorators/TypedQuery.d.ts +2 -2
  30. package/lib/decorators/TypedQuery.js +52 -129
  31. package/lib/decorators/TypedQuery.js.map +1 -1
  32. package/lib/decorators/TypedRoute.js +31 -107
  33. package/lib/decorators/TypedRoute.js.map +1 -1
  34. package/lib/decorators/WebSocketRoute.d.ts +120 -0
  35. package/lib/decorators/WebSocketRoute.js +202 -0
  36. package/lib/decorators/WebSocketRoute.js.map +1 -0
  37. package/lib/decorators/internal/IWebSocketRouteReflect.d.ts +25 -0
  38. package/lib/decorators/internal/IWebSocketRouteReflect.js +3 -0
  39. package/lib/decorators/internal/IWebSocketRouteReflect.js.map +1 -0
  40. package/lib/decorators/internal/NoTransformConfigureError.js +1 -1
  41. package/lib/decorators/internal/NoTransformConfigureError.js.map +1 -1
  42. package/lib/decorators/internal/get_path_and_querify.js +67 -131
  43. package/lib/decorators/internal/get_path_and_querify.js.map +1 -1
  44. package/lib/decorators/internal/get_path_and_stringify.js +57 -71
  45. package/lib/decorators/internal/get_path_and_stringify.js.map +1 -1
  46. package/lib/decorators/internal/get_text_body.js +7 -46
  47. package/lib/decorators/internal/get_text_body.js.map +1 -1
  48. package/lib/decorators/internal/headers_to_object.js +3 -42
  49. package/lib/decorators/internal/headers_to_object.js.map +1 -1
  50. package/lib/decorators/internal/load_controller.js +29 -106
  51. package/lib/decorators/internal/load_controller.js.map +1 -1
  52. package/lib/decorators/internal/route_error.js +15 -65
  53. package/lib/decorators/internal/route_error.js.map +1 -1
  54. package/lib/decorators/internal/validate_request_body.js +44 -54
  55. package/lib/decorators/internal/validate_request_body.js.map +1 -1
  56. package/lib/decorators/internal/validate_request_form_data.js +36 -44
  57. package/lib/decorators/internal/validate_request_form_data.js.map +1 -1
  58. package/lib/decorators/internal/validate_request_headers.js +36 -44
  59. package/lib/decorators/internal/validate_request_headers.js.map +1 -1
  60. package/lib/decorators/internal/validate_request_query.js +36 -44
  61. package/lib/decorators/internal/validate_request_query.js.map +1 -1
  62. package/lib/index.js +1 -1
  63. package/lib/index.js.map +1 -1
  64. package/lib/module.d.ts +10 -9
  65. package/lib/module.js +10 -9
  66. package/lib/module.js.map +1 -1
  67. package/lib/programmers/PlainBodyProgrammer.js +29 -44
  68. package/lib/programmers/PlainBodyProgrammer.js.map +1 -1
  69. package/lib/programmers/TypedBodyProgrammer.js +57 -82
  70. package/lib/programmers/TypedBodyProgrammer.js.map +1 -1
  71. package/lib/programmers/TypedExceptionProgrammer.js +51 -54
  72. package/lib/programmers/TypedExceptionProgrammer.js.map +1 -1
  73. package/lib/programmers/TypedFormDataBodyProgrammer.js +50 -73
  74. package/lib/programmers/TypedFormDataBodyProgrammer.js.map +1 -1
  75. package/lib/programmers/TypedHeadersProgrammer.js +24 -43
  76. package/lib/programmers/TypedHeadersProgrammer.js.map +1 -1
  77. package/lib/programmers/TypedParamProgrammer.js +11 -28
  78. package/lib/programmers/TypedParamProgrammer.js.map +1 -1
  79. package/lib/programmers/TypedQueryBodyProgrammer.js +24 -43
  80. package/lib/programmers/TypedQueryBodyProgrammer.js.map +1 -1
  81. package/lib/programmers/TypedQueryProgrammer.js +24 -43
  82. package/lib/programmers/TypedQueryProgrammer.js.map +1 -1
  83. package/lib/programmers/TypedQueryRouteProgrammer.js +22 -39
  84. package/lib/programmers/TypedQueryRouteProgrammer.js.map +1 -1
  85. package/lib/programmers/TypedRouteProgrammer.js +22 -39
  86. package/lib/programmers/TypedRouteProgrammer.js.map +1 -1
  87. package/lib/programmers/http/HttpAssertQuerifyProgrammer.js +12 -29
  88. package/lib/programmers/http/HttpAssertQuerifyProgrammer.js.map +1 -1
  89. package/lib/programmers/http/HttpIsQuerifyProgrammer.js +10 -27
  90. package/lib/programmers/http/HttpIsQuerifyProgrammer.js.map +1 -1
  91. package/lib/programmers/http/HttpQuerifyProgrammer.js +33 -69
  92. package/lib/programmers/http/HttpQuerifyProgrammer.js.map +1 -1
  93. package/lib/programmers/http/HttpValidateQuerifyProgrammer.js +11 -28
  94. package/lib/programmers/http/HttpValidateQuerifyProgrammer.js.map +1 -1
  95. package/lib/programmers/internal/CoreMetadataUtil.js +12 -37
  96. package/lib/programmers/internal/CoreMetadataUtil.js.map +1 -1
  97. package/lib/transform.js +8 -8
  98. package/lib/transform.js.map +1 -1
  99. package/lib/transformers/FileTransformer.js +35 -66
  100. package/lib/transformers/FileTransformer.js.map +1 -1
  101. package/lib/transformers/MethodTransformer.js +36 -50
  102. package/lib/transformers/MethodTransformer.js.map +1 -1
  103. package/lib/transformers/NodeTransformer.js +8 -12
  104. package/lib/transformers/NodeTransformer.js.map +1 -1
  105. package/lib/transformers/ParameterDecoratorTransformer.js +75 -90
  106. package/lib/transformers/ParameterDecoratorTransformer.js.map +1 -1
  107. package/lib/transformers/ParameterTransformer.js +18 -24
  108. package/lib/transformers/ParameterTransformer.js.map +1 -1
  109. package/lib/transformers/TypedExceptionTransformer.js +25 -27
  110. package/lib/transformers/TypedExceptionTransformer.js.map +1 -1
  111. package/lib/transformers/TypedRouteTransformer.js +52 -88
  112. package/lib/transformers/TypedRouteTransformer.js.map +1 -1
  113. package/lib/utils/ArrayUtil.d.ts +3 -0
  114. package/lib/utils/ArrayUtil.js +11 -0
  115. package/lib/utils/ArrayUtil.js.map +1 -0
  116. package/lib/utils/ExceptionManager.js +9 -31
  117. package/lib/utils/ExceptionManager.js.map +1 -1
  118. package/lib/utils/Singleton.js +6 -7
  119. package/lib/utils/Singleton.js.map +1 -1
  120. package/lib/utils/SourceFinder.js +38 -215
  121. package/lib/utils/SourceFinder.js.map +1 -1
  122. package/lib/utils/VersioningStrategy.d.ts +9 -0
  123. package/lib/utils/VersioningStrategy.js +17 -0
  124. package/lib/utils/VersioningStrategy.js.map +1 -0
  125. package/package.json +10 -6
  126. package/src/adaptors/WebSocketAdaptor.ts +414 -0
  127. package/src/decorators/TypedParam.ts +5 -6
  128. package/src/decorators/TypedQuery.ts +251 -251
  129. package/src/decorators/WebSocketRoute.ts +249 -0
  130. package/src/decorators/internal/IWebSocketRouteReflect.ts +23 -0
  131. package/src/decorators/internal/validate_request_body.ts +72 -72
  132. package/src/decorators/internal/validate_request_form_data.ts +75 -75
  133. package/src/decorators/internal/validate_request_headers.ts +83 -83
  134. package/src/decorators/internal/validate_request_query.ts +71 -71
  135. package/src/module.ts +20 -16
  136. package/src/transformers/ParameterDecoratorTransformer.ts +129 -120
  137. package/src/typings/get-function-location.d.ts +7 -0
  138. package/src/utils/ArrayUtil.ts +7 -0
  139. package/src/utils/VersioningStrategy.ts +27 -0
@@ -0,0 +1,414 @@
1
+ /// <reference path="../typings/get-function-location.d.ts" />
2
+ import { INestApplication, VersioningType } from "@nestjs/common";
3
+ import {
4
+ HOST_METADATA,
5
+ MODULE_PATH,
6
+ PATH_METADATA,
7
+ SCOPE_OPTIONS_METADATA,
8
+ VERSION_METADATA,
9
+ } from "@nestjs/common/constants";
10
+ import { VERSION_NEUTRAL, VersionValue } from "@nestjs/common/interfaces";
11
+ import { NestContainer } from "@nestjs/core";
12
+ import { InstanceWrapper } from "@nestjs/core/injector/instance-wrapper";
13
+ import { Module } from "@nestjs/core/injector/module";
14
+ import getFunctionLocation from "get-function-location";
15
+ import { IncomingMessage, Server } from "http";
16
+ import { Path } from "path-parser";
17
+ import { Duplex } from "stream";
18
+ import { WebAcceptor } from "tgrid";
19
+ import typia from "typia";
20
+ import WebSocket from "ws";
21
+
22
+ import { IWebSocketRouteReflect } from "../decorators/internal/IWebSocketRouteReflect";
23
+ import { ArrayUtil } from "../utils/ArrayUtil";
24
+ import { VersioningStrategy } from "../utils/VersioningStrategy";
25
+
26
+ export class WebSocketAdaptor {
27
+ public static async upgrade(
28
+ app: INestApplication,
29
+ ): Promise<WebSocketAdaptor> {
30
+ return new this(app, await visitApplication(app));
31
+ }
32
+
33
+ public readonly close = async (): Promise<void> =>
34
+ new Promise((resolve) => {
35
+ this.http.off("close", this.close);
36
+ this.http.off("upgrade", this.handleUpgrade);
37
+ this.ws.close(() => resolve());
38
+ });
39
+
40
+ protected constructor(app: INestApplication, operations: IOperator[]) {
41
+ this.operators = operations;
42
+ this.ws = new WebSocket.Server({ noServer: true });
43
+ this.http = app.getHttpServer();
44
+ this.http.on("close", this.close);
45
+ this.http.on("upgrade", this.handleUpgrade);
46
+ }
47
+
48
+ private readonly handleUpgrade = (
49
+ request: IncomingMessage,
50
+ duplex: Duplex,
51
+ head: Buffer,
52
+ ) => {
53
+ this.ws.handleUpgrade(request, duplex, head, (client, request) =>
54
+ WebAcceptor.upgrade(
55
+ request,
56
+ client as any,
57
+ async (acceptor): Promise<void> => {
58
+ const path: string = (() => {
59
+ const index: number = acceptor.path.indexOf("?");
60
+ return index === -1 ? acceptor.path : acceptor.path.slice(0, index);
61
+ })();
62
+ for (const op of this.operators) {
63
+ const params: Record<string, string> | null = op.parser.test(path);
64
+ if (params !== null)
65
+ try {
66
+ await op.handler({ params, acceptor });
67
+ } catch (error) {
68
+ if (
69
+ acceptor.state === WebAcceptor.State.OPEN ||
70
+ acceptor.state === WebAcceptor.State.ACCEPTING
71
+ )
72
+ await acceptor.reject(
73
+ 1008,
74
+ error instanceof Error
75
+ ? JSON.stringify({ ...error })
76
+ : "unknown error",
77
+ );
78
+ } finally {
79
+ return;
80
+ }
81
+ }
82
+ await acceptor.reject(1002, `Cannot GET ${path}`);
83
+ },
84
+ ),
85
+ );
86
+ };
87
+
88
+ private readonly http: Server;
89
+ private readonly operators: IOperator[];
90
+ private readonly ws: WebSocket.Server;
91
+ }
92
+
93
+ const visitApplication = async (
94
+ app: INestApplication,
95
+ ): Promise<IOperator[]> => {
96
+ const operators: IOperator[] = [];
97
+ const errors: IControllerError[] = [];
98
+
99
+ const config: IConfig = {
100
+ globalPrefix:
101
+ typeof (app as any).config?.globalPrefix === "string"
102
+ ? (app as any).config.globalPrefix
103
+ : undefined,
104
+ versioning: (() => {
105
+ const versioning = (app as any).config?.versioningOptions;
106
+ return versioning === undefined || versioning.type !== VersioningType.URI
107
+ ? undefined
108
+ : {
109
+ prefix:
110
+ versioning.prefix === undefined || versioning.prefix === false
111
+ ? "v"
112
+ : versioning.prefix,
113
+ defaultVersion: versioning.defaultVersion,
114
+ };
115
+ })(),
116
+ };
117
+ const container: NestContainer = (app as any).container as NestContainer;
118
+ const modules: Module[] = [...container.getModules().values()].filter(
119
+ (m) => !!m.controllers?.size,
120
+ );
121
+ for (const m of modules) {
122
+ const modulePrefix: string =
123
+ Reflect.getMetadata(
124
+ MODULE_PATH + container.getModules().applicationId,
125
+ m.metatype,
126
+ ) ??
127
+ Reflect.getMetadata(MODULE_PATH, m.metatype) ??
128
+ "";
129
+ for (const controller of m.controllers.values())
130
+ visitController({
131
+ config,
132
+ errors,
133
+ operators,
134
+ controller,
135
+ modulePrefix,
136
+ });
137
+ }
138
+ if (errors.length) {
139
+ throw new Error(
140
+ [
141
+ `WebSocketAdaptor: ${errors.length} error(s) found:`,
142
+ ``,
143
+ ...errors.map((e) =>
144
+ [
145
+ ` - controller: ${e.name}`,
146
+ ` - methods:`,
147
+ ...e.methods.map((m) =>
148
+ [
149
+ ` - name: ${m.name}`,
150
+ ` - file: ${m.source}:${m.line}:${m.column}`,
151
+ ` - reasons:`,
152
+ ...m.messages.map(
153
+ (msg) =>
154
+ ` - ${msg
155
+ .split("\n")
156
+ .map((str) => ` ${str}`)
157
+ .join("\n")}`,
158
+ ),
159
+ ]
160
+ .map((str) => ` ${str}`)
161
+ .join("\n"),
162
+ ),
163
+ ]
164
+ .map((str) => ` ${str}`)
165
+ .join("\n"),
166
+ ),
167
+ ].join("\n"),
168
+ );
169
+ }
170
+ return operators;
171
+ };
172
+
173
+ const visitController = async (props: {
174
+ config: IConfig;
175
+ errors: IControllerError[];
176
+ operators: IOperator[];
177
+ controller: InstanceWrapper<object>;
178
+ modulePrefix: string;
179
+ }): Promise<void> => {
180
+ if (
181
+ ArrayUtil.has(
182
+ Reflect.getMetadataKeys(props.controller.metatype),
183
+ PATH_METADATA,
184
+ HOST_METADATA,
185
+ SCOPE_OPTIONS_METADATA,
186
+ ) === false
187
+ )
188
+ return;
189
+
190
+ const methodErrors: IMethodError[] = [];
191
+ const controller: IController = {
192
+ name: props.controller.name,
193
+ instance: props.controller.instance,
194
+ constructor: props.controller.metatype,
195
+ prototype: Object.getPrototypeOf(props.controller.instance),
196
+ prefixes: (() => {
197
+ const value: string | string[] = Reflect.getMetadata(
198
+ PATH_METADATA,
199
+ props.controller.metatype,
200
+ );
201
+ if (typeof value === "string") return [value];
202
+ else if (value.length === 0) return [""];
203
+ else return value;
204
+ })(),
205
+ versions: props.config.versioning
206
+ ? VersioningStrategy.cast(
207
+ Reflect.getMetadata(VERSION_METADATA, props.controller.metatype),
208
+ )
209
+ : undefined,
210
+ modulePrefix: props.modulePrefix,
211
+ };
212
+ for (const mk of Object.getOwnPropertyNames(controller.prototype).filter(
213
+ (key) =>
214
+ key !== "constructor" && typeof controller.prototype[key] === "function",
215
+ )) {
216
+ const errorMessages: string[] = [];
217
+ visitMethod({
218
+ config: props.config,
219
+ operators: props.operators,
220
+ controller,
221
+ method: {
222
+ key: mk,
223
+ value: controller.prototype[mk],
224
+ },
225
+ report: (msg) => errorMessages.push(msg),
226
+ });
227
+ if (errorMessages.length)
228
+ methodErrors.push({
229
+ name: mk,
230
+ messages: errorMessages,
231
+ ...(await getFunctionLocation(controller.prototype[mk])),
232
+ });
233
+ }
234
+
235
+ if (methodErrors.length)
236
+ props.errors.push({
237
+ name: controller.name,
238
+ methods: methodErrors,
239
+ });
240
+ };
241
+
242
+ const visitMethod = (props: {
243
+ config: IConfig;
244
+ operators: IOperator[];
245
+ controller: IController;
246
+ method: Entry<Function>;
247
+ report: (message: string) => void;
248
+ }): void => {
249
+ const route: IWebSocketRouteReflect | undefined = Reflect.getMetadata(
250
+ "nestia/WebSocketRoute",
251
+ props.method.value,
252
+ );
253
+ if (typia.is<IWebSocketRouteReflect>(route) === false) return;
254
+
255
+ const parameters: IWebSocketRouteReflect.IArgument[] = (
256
+ (Reflect.getMetadata(
257
+ "nestia/WebSocketRoute/Parameters",
258
+ props.controller.prototype,
259
+ props.method.key,
260
+ ) ?? []) as IWebSocketRouteReflect.IArgument[]
261
+ ).sort((a, b) => a.index - b.index);
262
+ // acceptor must be
263
+ if (parameters.some((p) => p.category === "acceptor") === false)
264
+ return props.report(
265
+ "@WebSocketRoute.Acceptor() decorated parameter must be.",
266
+ );
267
+ // length of parameters must be fulfilled
268
+ if (parameters.length !== props.method.value.length)
269
+ return props.report(
270
+ [
271
+ "Every parameters must be one of below:",
272
+ " - @WebSocketRoute.Acceptor()",
273
+ " - @WebSocketRoute.Driver()",
274
+ " - @WebSocketRoute.Header()",
275
+ " - @WebSocketRoute.Param()",
276
+ " - @WebSocketRoute.Query()",
277
+ ].join("\n"),
278
+ );
279
+
280
+ const versions: string[] = VersioningStrategy.merge(props.config.versioning)([
281
+ ...(props.controller.versions ?? []),
282
+ ...VersioningStrategy.cast(
283
+ Reflect.getMetadata(VERSION_METADATA, props.method.value),
284
+ ),
285
+ ]);
286
+ for (const v of versions)
287
+ for (const cp of wrapPaths(props.controller.prefixes))
288
+ for (const mp of wrapPaths(route.paths)) {
289
+ const parser: Path = new Path(
290
+ "/" +
291
+ [
292
+ props.config.globalPrefix ?? "",
293
+ v,
294
+ props.controller.modulePrefix,
295
+ cp,
296
+ mp,
297
+ ]
298
+ .filter((str) => !!str.length)
299
+ .join("/")
300
+ .split("/")
301
+ .filter((str) => str.length)
302
+ .join("/"),
303
+ );
304
+ const pathParams: IWebSocketRouteReflect.IParam[] = parameters.filter(
305
+ (p) => p.category === "param",
306
+ ) as IWebSocketRouteReflect.IParam[];
307
+ if (parser.params.length !== pathParams.length) {
308
+ props.report(
309
+ [
310
+ `Path "${parser}" must have same number of parameters with @WebSocketRoute.Param()`,
311
+ ` - path: ${JSON.stringify(parser.params)}`,
312
+ ` - arguments: ${JSON.stringify(pathParams.map((p) => p.field))}`,
313
+ ].join("\n"),
314
+ );
315
+ continue;
316
+ }
317
+ const meet: boolean = pathParams
318
+ .map((p) => {
319
+ const has: boolean = parser.params.includes(p.field);
320
+ if (has === false)
321
+ props.report(
322
+ `Path "${parser}" must have parameter "${p.field}" with @WebSocketRoute.Param()`,
323
+ );
324
+ return has;
325
+ })
326
+ .every((b) => b);
327
+ if (meet === false) continue;
328
+
329
+ props.operators.push({
330
+ parser,
331
+ handler: async (input: {
332
+ params: Record<string, string>;
333
+ acceptor: WebAcceptor<any, any, any>;
334
+ }): Promise<void> => {
335
+ const args: any[] = [];
336
+ try {
337
+ for (const p of parameters)
338
+ if (p.category === "acceptor") args.push(input.acceptor);
339
+ else if (p.category === "driver")
340
+ args.push(input.acceptor.getDriver());
341
+ else if (p.category === "header") {
342
+ const error: Error | null = p.validate(input.acceptor.header);
343
+ if (error !== null) throw error;
344
+ args.push(input.acceptor.header);
345
+ } else if (p.category === "param")
346
+ args.push(p.assert(input.params[p.field]));
347
+ else if (p.category === "query") {
348
+ const query: any | Error = p.validate(
349
+ new URLSearchParams(
350
+ input.acceptor.path.indexOf("?") !== -1
351
+ ? input.acceptor.path.split("?")[1]
352
+ : "",
353
+ ),
354
+ );
355
+ if (query instanceof Error) throw query;
356
+ args.push(query);
357
+ }
358
+ } catch (exp) {
359
+ await input.acceptor.reject(
360
+ 1003,
361
+ exp instanceof Error
362
+ ? JSON.stringify({ ...exp })
363
+ : "unknown error",
364
+ );
365
+ return;
366
+ }
367
+ await props.method.value.call(props.controller.instance, ...args);
368
+ },
369
+ });
370
+ }
371
+ };
372
+
373
+ const wrapPaths = (value: string[]) => (value.length === 0 ? [""] : value);
374
+
375
+ interface Entry<T> {
376
+ key: string;
377
+ value: T;
378
+ }
379
+
380
+ interface IController {
381
+ name: string;
382
+ versions: Array<string | typeof VERSION_NEUTRAL> | undefined;
383
+ instance: object;
384
+ constructor: Function;
385
+ prototype: any;
386
+ prefixes: string[];
387
+ modulePrefix: string;
388
+ }
389
+ interface IOperator {
390
+ parser: Path;
391
+ handler: (props: {
392
+ params: Record<string, string>;
393
+ acceptor: WebAcceptor<any, any, any>;
394
+ }) => Promise<any>;
395
+ }
396
+ interface IConfig {
397
+ globalPrefix?: string;
398
+ versioning?: {
399
+ prefix: string;
400
+ defaultVersion?: VersionValue;
401
+ };
402
+ }
403
+
404
+ interface IControllerError {
405
+ name: string;
406
+ methods: IMethodError[];
407
+ }
408
+ interface IMethodError {
409
+ name: string;
410
+ messages: string[];
411
+ source: string;
412
+ line: number;
413
+ column: number;
414
+ }
@@ -20,12 +20,11 @@ import { NoTransformConfigureError } from "./internal/NoTransformConfigureError"
20
20
  * import { tags } from "typia";
21
21
  *
22
22
  * \@TypedRoute.Get("shopping/sales/:id/:no/:paused")
23
- * public async pause
24
- * (
25
- * \@TypedParam("id", "uuid"), id: string & tags.Format<"uuid">,
26
- * \@TypedParam("no") id: number & tags.Type<"uint32">
27
- * \@TypedParam("paused") paused: boolean | null
28
- * ): Promise<void>;
23
+ * public async pause(
24
+ * \@TypedParam("id", "uuid"), id: string & tags.Format<"uuid">,
25
+ * \@TypedParam("no") id: number & tags.Type<"uint32">
26
+ * \@TypedParam("paused") paused: boolean | null
27
+ * ): Promise<void>;
29
28
  * ```
30
29
  *
31
30
  * @param name URL Parameter name