typia 5.0.0-dev.20230823 → 5.0.0-dev.2023084

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 (217) hide show
  1. package/lib/Primitive.d.ts +14 -12
  2. package/lib/Resolved.d.ts +46 -0
  3. package/lib/{schemas/metadata/IMetadataResolved.js → Resolved.js} +1 -1
  4. package/lib/Resolved.js.map +1 -0
  5. package/lib/factories/MetadataFactory.d.ts +1 -1
  6. package/lib/factories/MetadataTagFactory.js +10 -8
  7. package/lib/factories/MetadataTagFactory.js.map +1 -1
  8. package/lib/factories/ProtobufFactory.js +4 -3
  9. package/lib/factories/ProtobufFactory.js.map +1 -1
  10. package/lib/factories/internal/metadata/emend_metadata_atomics.js +6 -6
  11. package/lib/factories/internal/metadata/emend_metadata_atomics.js.map +1 -1
  12. package/lib/factories/internal/metadata/explore_metadata.js +3 -3
  13. package/lib/factories/internal/metadata/explore_metadata.js.map +1 -1
  14. package/lib/factories/internal/metadata/iterate_metadata_atomic.js +1 -1
  15. package/lib/factories/internal/metadata/iterate_metadata_atomic.js.map +1 -1
  16. package/lib/factories/internal/metadata/iterate_metadata_collection.js +6 -6
  17. package/lib/factories/internal/metadata/iterate_metadata_collection.js.map +1 -1
  18. package/lib/factories/internal/metadata/iterate_metadata_native.js +2 -10
  19. package/lib/factories/internal/metadata/iterate_metadata_native.js.map +1 -1
  20. package/lib/factories/internal/metadata/iterate_metadata_resolve.js +6 -6
  21. package/lib/factories/internal/metadata/iterate_metadata_resolve.js.map +1 -1
  22. package/lib/factories/internal/metadata/iterate_metadata_sort.js +2 -2
  23. package/lib/factories/internal/metadata/iterate_metadata_sort.js.map +1 -1
  24. package/lib/functional/$ProtobufWriter.js.map +1 -1
  25. package/lib/json.d.ts +15 -15
  26. package/lib/misc.d.ts +42 -42
  27. package/lib/module.d.ts +18 -17
  28. package/lib/module.js +1 -0
  29. package/lib/module.js.map +1 -1
  30. package/lib/programmers/AssertProgrammer.js +1 -1
  31. package/lib/programmers/AssertProgrammer.js.map +1 -1
  32. package/lib/programmers/CheckerProgrammer.js +7 -7
  33. package/lib/programmers/CheckerProgrammer.js.map +1 -1
  34. package/lib/programmers/IsProgrammer.js +2 -2
  35. package/lib/programmers/IsProgrammer.js.map +1 -1
  36. package/lib/programmers/RandomProgrammer.js +311 -136
  37. package/lib/programmers/RandomProgrammer.js.map +1 -1
  38. package/lib/programmers/ValidateProgrammer.js +1 -1
  39. package/lib/programmers/ValidateProgrammer.js.map +1 -1
  40. package/lib/programmers/helpers/AtomicPredicator.js +2 -2
  41. package/lib/programmers/helpers/AtomicPredicator.js.map +1 -1
  42. package/lib/programmers/helpers/FunctionImporeter.d.ts +2 -0
  43. package/lib/programmers/helpers/FunctionImporeter.js +2 -1
  44. package/lib/programmers/helpers/FunctionImporeter.js.map +1 -1
  45. package/lib/programmers/helpers/ProtobufUtil.js +2 -2
  46. package/lib/programmers/helpers/ProtobufUtil.js.map +1 -1
  47. package/lib/programmers/helpers/StringifyPredicator.js +1 -2
  48. package/lib/programmers/helpers/StringifyPredicator.js.map +1 -1
  49. package/lib/programmers/helpers/disable_function_importer_declare.js +1 -0
  50. package/lib/programmers/helpers/disable_function_importer_declare.js.map +1 -1
  51. package/lib/programmers/internal/application_default_string.js +2 -2
  52. package/lib/programmers/internal/application_default_string.js.map +1 -1
  53. package/lib/programmers/internal/application_resolved.d.ts +2 -2
  54. package/lib/programmers/internal/application_resolved.js.map +1 -1
  55. package/lib/programmers/internal/application_schema.js +7 -7
  56. package/lib/programmers/internal/application_schema.js.map +1 -1
  57. package/lib/programmers/internal/metadata_to_pattern.js +4 -4
  58. package/lib/programmers/internal/metadata_to_pattern.js.map +1 -1
  59. package/lib/programmers/internal/stringify_dynamic_properties.js +2 -1
  60. package/lib/programmers/internal/stringify_dynamic_properties.js.map +1 -1
  61. package/lib/programmers/json/JsonStringifyProgrammer.js +20 -14
  62. package/lib/programmers/json/JsonStringifyProgrammer.js.map +1 -1
  63. package/lib/programmers/misc/MiscAssertCloneProgrammer.js +1 -1
  64. package/lib/programmers/misc/MiscAssertCloneProgrammer.js.map +1 -1
  65. package/lib/programmers/misc/MiscCloneProgrammer.js +114 -33
  66. package/lib/programmers/misc/MiscCloneProgrammer.js.map +1 -1
  67. package/lib/programmers/misc/MiscIsCloneProgrammer.js +1 -1
  68. package/lib/programmers/misc/MiscIsCloneProgrammer.js.map +1 -1
  69. package/lib/programmers/misc/MiscLiteralsProgrammer.js +3 -3
  70. package/lib/programmers/misc/MiscLiteralsProgrammer.js.map +1 -1
  71. package/lib/programmers/misc/MiscPruneProgrammer.js +2 -2
  72. package/lib/programmers/misc/MiscPruneProgrammer.js.map +1 -1
  73. package/lib/programmers/misc/MiscValidateCloneProgrammer.js +1 -1
  74. package/lib/programmers/misc/MiscValidateCloneProgrammer.js.map +1 -1
  75. package/lib/programmers/protobuf/ProtobufAssertDecodeProgrammer.js +3 -3
  76. package/lib/programmers/protobuf/ProtobufAssertDecodeProgrammer.js.map +1 -1
  77. package/lib/programmers/protobuf/ProtobufDecodeProgrammer.js +3 -3
  78. package/lib/programmers/protobuf/ProtobufDecodeProgrammer.js.map +1 -1
  79. package/lib/programmers/protobuf/ProtobufEncodeProgrammer.js +1 -1
  80. package/lib/programmers/protobuf/ProtobufEncodeProgrammer.js.map +1 -1
  81. package/lib/programmers/protobuf/ProtobufIsDecodeProgrammer.js +2 -2
  82. package/lib/programmers/protobuf/ProtobufIsDecodeProgrammer.js.map +1 -1
  83. package/lib/programmers/protobuf/ProtobufValidateDecodeProgrammer.js +1 -1
  84. package/lib/programmers/protobuf/ProtobufValidateDecodeProgrammer.js.map +1 -1
  85. package/lib/protobuf.d.ts +631 -10
  86. package/lib/protobuf.js +13 -13
  87. package/lib/protobuf.js.map +1 -1
  88. package/lib/schemas/metadata/IMetadata.d.ts +4 -4
  89. package/lib/schemas/metadata/IMetadataAtomic.d.ts +12 -0
  90. package/lib/schemas/metadata/IMetadataAtomic.js +3 -0
  91. package/lib/schemas/metadata/IMetadataAtomic.js.map +1 -0
  92. package/lib/schemas/metadata/{IMetadataResolved.d.ts → IMetadataEscaped.d.ts} +1 -1
  93. package/lib/schemas/metadata/IMetadataEscaped.js +3 -0
  94. package/lib/schemas/metadata/IMetadataEscaped.js.map +1 -0
  95. package/lib/schemas/metadata/Metadata.d.ts +4 -4
  96. package/lib/schemas/metadata/Metadata.js +64 -59
  97. package/lib/schemas/metadata/Metadata.js.map +1 -1
  98. package/lib/schemas/metadata/{MetadataResolved.d.ts → MetadataEscaped.d.ts} +3 -3
  99. package/lib/schemas/metadata/{MetadataResolved.js → MetadataEscaped.js} +11 -11
  100. package/lib/schemas/metadata/MetadataEscaped.js.map +1 -0
  101. package/lib/tags/ExclusiveMaximum.d.ts +8 -0
  102. package/lib/tags/ExclusiveMaximum.js +3 -0
  103. package/lib/tags/ExclusiveMaximum.js.map +1 -0
  104. package/lib/tags/ExclusiveMinimum.d.ts +8 -0
  105. package/lib/tags/ExclusiveMinimum.js +3 -0
  106. package/lib/tags/ExclusiveMinimum.js.map +1 -0
  107. package/lib/tags/Format.d.ts +8 -0
  108. package/lib/tags/Format.js +3 -0
  109. package/lib/tags/Format.js.map +1 -0
  110. package/lib/tags/MaxItems.d.ts +8 -0
  111. package/lib/tags/MaxItems.js +3 -0
  112. package/lib/tags/MaxItems.js.map +1 -0
  113. package/lib/tags/MaxLength.d.ts +8 -0
  114. package/lib/tags/MaxLength.js +3 -0
  115. package/lib/tags/MaxLength.js.map +1 -0
  116. package/lib/tags/Maximum.d.ts +8 -0
  117. package/lib/tags/Maximum.js +3 -0
  118. package/lib/tags/Maximum.js.map +1 -0
  119. package/lib/tags/MinItems.d.ts +8 -0
  120. package/lib/tags/MinItems.js +3 -0
  121. package/lib/tags/MinItems.js.map +1 -0
  122. package/lib/tags/MinLength.d.ts +8 -0
  123. package/lib/tags/MinLength.js +3 -0
  124. package/lib/tags/MinLength.js.map +1 -0
  125. package/lib/tags/Minimum.d.ts +8 -0
  126. package/lib/tags/Minimum.js +3 -0
  127. package/lib/tags/Minimum.js.map +1 -0
  128. package/lib/tags/MultipleOf.d.ts +8 -0
  129. package/lib/tags/MultipleOf.js +3 -0
  130. package/lib/tags/MultipleOf.js.map +1 -0
  131. package/lib/tags/Pattern.d.ts +8 -0
  132. package/lib/tags/Pattern.js +3 -0
  133. package/lib/tags/Pattern.js.map +1 -0
  134. package/lib/tags/TagBase.d.ts +10 -0
  135. package/lib/tags/TagBase.js +3 -0
  136. package/lib/tags/TagBase.js.map +1 -0
  137. package/lib/tags/Type.d.ts +13 -0
  138. package/lib/tags/Type.js +3 -0
  139. package/lib/tags/Type.js.map +1 -0
  140. package/lib/tags/index.d.ts +12 -0
  141. package/lib/tags/index.js +29 -0
  142. package/lib/tags/index.js.map +1 -0
  143. package/lib/transformers/features/json/JsonApplicationTransformer.js +2 -2
  144. package/lib/transformers/features/json/JsonApplicationTransformer.js.map +1 -1
  145. package/lib/transformers/features/misc/MetadataTransformer.js +1 -1
  146. package/lib/transformers/features/misc/MetadataTransformer.js.map +1 -1
  147. package/lib/utils/RandomGenerator.js +3 -2
  148. package/lib/utils/RandomGenerator.js.map +1 -1
  149. package/package.json +2 -1
  150. package/src/Primitive.ts +135 -131
  151. package/src/Resolved.ts +116 -0
  152. package/src/factories/MetadataFactory.ts +46 -46
  153. package/src/factories/MetadataTagFactory.ts +366 -364
  154. package/src/factories/ProtobufFactory.ts +268 -266
  155. package/src/factories/internal/metadata/emend_metadata_atomics.ts +35 -33
  156. package/src/factories/internal/metadata/explore_metadata.ts +38 -38
  157. package/src/factories/internal/metadata/iterate_metadata_atomic.ts +63 -59
  158. package/src/factories/internal/metadata/iterate_metadata_collection.ts +133 -133
  159. package/src/factories/internal/metadata/iterate_metadata_native.ts +210 -219
  160. package/src/factories/internal/metadata/iterate_metadata_resolve.ts +49 -49
  161. package/src/factories/internal/metadata/iterate_metadata_sort.ts +69 -69
  162. package/src/functional/$ProtobufWriter.ts +151 -151
  163. package/src/json.ts +648 -648
  164. package/src/misc.ts +651 -651
  165. package/src/module.ts +709 -708
  166. package/src/programmers/AssertProgrammer.ts +281 -279
  167. package/src/programmers/CheckerProgrammer.ts +1174 -1173
  168. package/src/programmers/IsProgrammer.ts +241 -239
  169. package/src/programmers/RandomProgrammer.ts +874 -578
  170. package/src/programmers/ValidateProgrammer.ts +307 -305
  171. package/src/programmers/helpers/AtomicPredicator.ts +31 -31
  172. package/src/programmers/helpers/FunctionImporeter.ts +91 -89
  173. package/src/programmers/helpers/ProtobufUtil.ts +29 -29
  174. package/src/programmers/helpers/StringifyPredicator.ts +12 -13
  175. package/src/programmers/helpers/disable_function_importer_declare.ts +32 -27
  176. package/src/programmers/internal/application_default_string.ts +37 -33
  177. package/src/programmers/internal/application_resolved.ts +55 -55
  178. package/src/programmers/internal/application_schema.ts +157 -157
  179. package/src/programmers/internal/metadata_to_pattern.ts +34 -34
  180. package/src/programmers/internal/stringify_dynamic_properties.ts +171 -171
  181. package/src/programmers/json/JsonStringifyProgrammer.ts +995 -987
  182. package/src/programmers/misc/MiscAssertCloneProgrammer.ts +71 -71
  183. package/src/programmers/misc/MiscCloneProgrammer.ts +775 -587
  184. package/src/programmers/misc/MiscIsCloneProgrammer.ts +78 -78
  185. package/src/programmers/misc/MiscLiteralsProgrammer.ts +64 -64
  186. package/src/programmers/misc/MiscPruneProgrammer.ts +544 -542
  187. package/src/programmers/misc/MiscValidateCloneProgrammer.ts +85 -85
  188. package/src/programmers/protobuf/ProtobufAssertDecodeProgrammer.ts +75 -66
  189. package/src/programmers/protobuf/ProtobufDecodeProgrammer.ts +673 -669
  190. package/src/programmers/protobuf/ProtobufEncodeProgrammer.ts +814 -814
  191. package/src/programmers/protobuf/ProtobufIsDecodeProgrammer.ts +85 -75
  192. package/src/programmers/protobuf/ProtobufValidateDecodeProgrammer.ts +75 -75
  193. package/src/protobuf.ts +881 -249
  194. package/src/schemas/metadata/IMetadata.ts +27 -28
  195. package/src/schemas/metadata/IMetadataAtomic.ts +13 -0
  196. package/src/schemas/metadata/{IMetadataResolved.ts → IMetadataEscaped.ts} +6 -6
  197. package/src/schemas/metadata/Metadata.ts +643 -637
  198. package/src/schemas/metadata/{MetadataResolved.ts → MetadataEscaped.ts} +51 -51
  199. package/src/tags/ExclusiveMaximum.ts +8 -0
  200. package/src/tags/ExclusiveMinimum.ts +8 -0
  201. package/src/tags/Format.ts +29 -0
  202. package/src/tags/MaxItems.ts +8 -0
  203. package/src/tags/MaxLength.ts +8 -0
  204. package/src/tags/Maximum.ts +8 -0
  205. package/src/tags/MinItems.ts +8 -0
  206. package/src/tags/MinLength.ts +8 -0
  207. package/src/tags/Minimum.ts +8 -0
  208. package/src/tags/MultipleOf.ts +10 -0
  209. package/src/tags/Pattern.ts +8 -0
  210. package/src/tags/TagBase.ts +17 -0
  211. package/src/tags/Type.ts +30 -0
  212. package/src/tags/index.ts +12 -0
  213. package/src/transformers/features/json/JsonApplicationTransformer.ts +111 -111
  214. package/src/transformers/features/misc/MetadataTransformer.ts +53 -53
  215. package/src/utils/RandomGenerator.ts +83 -81
  216. package/lib/schemas/metadata/IMetadataResolved.js.map +0 -1
  217. package/lib/schemas/metadata/MetadataResolved.js.map +0 -1
@@ -1,364 +1,366 @@
1
- import ts from "typescript";
2
-
3
- import { IMetadataTag } from "../schemas/metadata/IMetadataTag";
4
- import { Metadata } from "../schemas/metadata/Metadata";
5
-
6
- export namespace MetadataTagFactory {
7
- export const generate =
8
- (metadata: Metadata) =>
9
- (tagList: ts.JSDocTagInfo[]) =>
10
- (identifier: () => string): IMetadataTag[] => {
11
- const output: IMetadataTag[] = [];
12
- for (const tag of tagList) {
13
- const elem: IMetadataTag | null = parse(
14
- identifier,
15
- metadata,
16
- tag,
17
- output,
18
- );
19
- if (elem !== null) output.push(elem);
20
- }
21
- return output;
22
- };
23
-
24
- const parse = (
25
- identifier: () => string,
26
- metadata: Metadata,
27
- tag: ts.JSDocTagInfo,
28
- output: IMetadataTag[],
29
- ): IMetadataTag | null => {
30
- const closure = _PARSER[tag.name];
31
- if (closure === undefined) return null;
32
-
33
- const text = (tag.text || [])[0]?.text;
34
- if (text === undefined)
35
- throw new Error(`${LABEL}: no tag value on ${identifier()}`);
36
-
37
- return closure(identifier, metadata, text, output);
38
- };
39
-
40
- /**
41
- * @internal
42
- */
43
- export const _PARSER: Record<
44
- string,
45
- (
46
- identifier: () => string,
47
- metadata: Metadata,
48
- text: string,
49
- output: IMetadataTag[],
50
- ) => IMetadataTag | null
51
- > = {
52
- /* -----------------------------------------------------------
53
- ARRAY
54
- ----------------------------------------------------------- */
55
- items: (identifier, metadata, text, output) => {
56
- validate(identifier, metadata, output, "items", "array", [
57
- "minItems",
58
- ]);
59
- return {
60
- kind: "items",
61
- value: parse_number(identifier, text),
62
- };
63
- },
64
- minItems: (identifier, metadata, text, output) => {
65
- validate(identifier, metadata, output, "minItems", "array", [
66
- "items",
67
- ]);
68
- return {
69
- kind: "minItems",
70
- value: parse_number(identifier, text),
71
- };
72
- },
73
- maxItems: (identifier, metadata, text, output) => {
74
- validate(identifier, metadata, output, "maxItems", "array", [
75
- "items",
76
- ]);
77
- return {
78
- kind: "maxItems",
79
- value: parse_number(identifier, text),
80
- };
81
- },
82
-
83
- /* -----------------------------------------------------------
84
- NUMBER
85
- ----------------------------------------------------------- */
86
- type: (_identifier, metadata, text, _output) => {
87
- if (text.startsWith("{") && text.endsWith("}"))
88
- text = text.substring(1, text.length - 1);
89
- return has_atomic("number")(new Set())(metadata) &&
90
- (text === "int" ||
91
- text === "uint" ||
92
- text === "int32" ||
93
- text === "uint32" ||
94
- text === "int64" ||
95
- text === "uint64" ||
96
- text === "float")
97
- ? { kind: "type", value: text }
98
- : null;
99
- },
100
- minimum: (identifier, metadata, text, output) => {
101
- validate(identifier, metadata, output, "minimum", "number", [
102
- "exclusiveMinimum",
103
- ]);
104
- return {
105
- kind: "minimum",
106
- value: parse_number(identifier, text),
107
- };
108
- },
109
- maximum: (identifier, metadata, text, output) => {
110
- validate(identifier, metadata, output, "maximum", "number", [
111
- "exclusiveMaximum",
112
- ]);
113
- return {
114
- kind: "maximum",
115
- value: parse_number(identifier, text),
116
- };
117
- },
118
- exclusiveMinimum: (identifier, metadata, text, output) => {
119
- validate(
120
- identifier,
121
- metadata,
122
- output,
123
- "exclusiveMinimum",
124
- "number",
125
- ["minimum"],
126
- );
127
- return {
128
- kind: "exclusiveMinimum",
129
- value: parse_number(identifier, text),
130
- };
131
- },
132
- exclusiveMaximum: (identifier, metadata, text, output) => {
133
- validate(
134
- identifier,
135
- metadata,
136
- output,
137
- "exclusiveMaximum",
138
- "number",
139
- ["maximum"],
140
- );
141
- return {
142
- kind: "exclusiveMaximum",
143
- value: parse_number(identifier, text),
144
- };
145
- },
146
- multipleOf: (identifier, metadata, text, output) => {
147
- validate(identifier, metadata, output, "multipleOf", "number", [
148
- "step",
149
- ]);
150
- return {
151
- kind: "multipleOf",
152
- value: parse_number(identifier, text),
153
- };
154
- },
155
- step: (identifier, metadata, text, output) => {
156
- validate(identifier, metadata, output, "step", "number", [
157
- "multipleOf",
158
- ]);
159
-
160
- const minimum: boolean = output.some(
161
- (tag) =>
162
- tag.kind === "minimum" || tag.kind === "exclusiveMinimum",
163
- );
164
- if (minimum === undefined)
165
- throw new Error(
166
- `${LABEL}: step requires minimum or exclusiveMinimum tag on "${identifier()}".`,
167
- );
168
-
169
- return {
170
- kind: "step",
171
- value: parse_number(identifier, text),
172
- };
173
- },
174
-
175
- /* -----------------------------------------------------------
176
- STRING
177
- ----------------------------------------------------------- */
178
- // Ignore arbitrary @format values in the internal metadata,
179
- // these are currently only supported on the typia.application() API.
180
- format: (identifier, metadata, str, output) => {
181
- const value: IMetadataTag.IFormat["value"] | undefined =
182
- FORMATS.get(str);
183
- validate(
184
- identifier,
185
- metadata,
186
- output,
187
- "format",
188
- value === "date" || value === "datetime" ? "Date" : "string",
189
- ["pattern"],
190
- );
191
- if (value === undefined) return null;
192
- return {
193
- kind: "format",
194
- value,
195
- };
196
- },
197
- pattern: (identifier, metadata, value, output) => {
198
- validate(identifier, metadata, output, "pattern", "string", [
199
- "format",
200
- ]);
201
- return {
202
- kind: "pattern",
203
- value,
204
- };
205
- },
206
- length: (identifier, metadata, text, output) => {
207
- validate(identifier, metadata, output, "length", "string", [
208
- "minLength",
209
- "maxLength",
210
- ]);
211
- return {
212
- kind: "length",
213
- value: parse_number(identifier, text),
214
- };
215
- },
216
- minLength: (identifier, metadata, text, output) => {
217
- validate(identifier, metadata, output, "minLength", "string", [
218
- "length",
219
- ]);
220
- return {
221
- kind: "minLength",
222
- value: parse_number(identifier, text),
223
- };
224
- },
225
- maxLength: (identifier, metadata, text, output) => {
226
- validate(identifier, metadata, output, "maxLength", "string", [
227
- "length",
228
- ]);
229
- return {
230
- kind: "maxLength",
231
- value: parse_number(identifier, text),
232
- };
233
- },
234
- };
235
- }
236
-
237
- const parse_number = (identifier: () => string, str: string): number => {
238
- const value: number = Number(str);
239
- if (isNaN(value) === true)
240
- throw new Error(`${LABEL}: invalid number on "${identifier()}".`);
241
- return value;
242
- };
243
-
244
- const LABEL = "Error on typia.MetadataTagFactory.generate()";
245
- const FORMATS: Map<string, IMetadataTag.IFormat["value"]> = new Map([
246
- ["uuid", "uuid"],
247
- ["email", "email"],
248
- ["url", "url"],
249
- ["ipv4", "ipv4"],
250
- ["ipv6", "ipv6"],
251
- ["date", "date"],
252
- ["datetime", "datetime"],
253
- ["date-time", "datetime"],
254
- ["dateTime", "datetime"],
255
- ]);
256
-
257
- const WRONG_TYPE = (
258
- tag: string,
259
- type: "string" | "number" | "array",
260
- identifier: () => string,
261
- ) => `${LABEL}: ${tag} requires ${type} type, but no "${identifier()}".`;
262
-
263
- const validate = (
264
- identifier: () => string,
265
- metadata: Metadata,
266
- output: IMetadataTag[],
267
- kind: IMetadataTag["kind"],
268
- type: "array" | "string" | "number" | "Date",
269
- neighbors: IMetadataTag["kind"][],
270
- ): void => {
271
- // TYPE CHECKING
272
- if (type === "array") {
273
- if (has_array(new Set())(metadata) === false)
274
- throw new Error(WRONG_TYPE(kind, "array", identifier));
275
- } else if (type === "Date") {
276
- if (
277
- has_native("Date")(new Set())(metadata) === false &&
278
- has_atomic("string")(new Set())(metadata) === false
279
- )
280
- throw new Error(WRONG_TYPE(kind, "string", identifier));
281
- } else if (has_atomic(type)(new Set())(metadata) === false)
282
- throw new Error(WRONG_TYPE(kind, type, identifier));
283
-
284
- // DUPLICATED TAG
285
- if (output.some((tag) => tag.kind === kind))
286
- throw new Error(
287
- `${LABEL}: duplicated ${kind} tags on "${identifier()}".`,
288
- );
289
-
290
- // NEIGHBOR TAG
291
- for (const name of neighbors)
292
- if (output.some((tag) => tag.kind === name))
293
- throw new Error(
294
- `${LABEL}: ${kind} and ${name} tags on "${identifier()}".`,
295
- );
296
- };
297
-
298
- // @todo: must block repeated array and tuple type
299
- const has_atomic =
300
- (type: "string" | "number" | "bigint") =>
301
- (visited: Set<Metadata>) =>
302
- (metadata: Metadata): boolean => {
303
- if (visited.has(metadata)) return false;
304
- visited.add(metadata);
305
- return (
306
- metadata.atomics.find(
307
- type === "number"
308
- ? (atom: string) => atom === type || atom === "bigint"
309
- : (atom: string) => atom === type,
310
- ) !== undefined ||
311
- metadata.arrays.some((array) =>
312
- has_atomic(type)(visited)(array.value),
313
- ) ||
314
- metadata.tuples.some((tuple) =>
315
- tuple.elements.some(has_atomic(type)(visited)),
316
- ) ||
317
- metadata.maps.some((map) => has_atomic(type)(visited)(map.value)) ||
318
- metadata.aliases.some((alias) =>
319
- has_atomic(type)(visited)(alias.value),
320
- ) ||
321
- (metadata.resolved !== null &&
322
- has_atomic(type)(visited)(metadata.resolved.returns))
323
- );
324
- };
325
-
326
- const has_native =
327
- (type: string) =>
328
- (visited: Set<Metadata>) =>
329
- (metadata: Metadata): boolean => {
330
- if (visited.has(metadata)) return false;
331
- visited.add(metadata);
332
- return (
333
- metadata.natives.find((native) => native === type) !== undefined ||
334
- metadata.arrays.some((child) =>
335
- has_native(type)(visited)(child.value),
336
- ) ||
337
- metadata.tuples.some((tuple) =>
338
- tuple.elements.some(has_native(type)(visited)),
339
- ) ||
340
- metadata.maps.some((map) => has_native(type)(visited)(map.value)) ||
341
- metadata.aliases.some((alias) =>
342
- has_native(type)(visited)(alias.value),
343
- ) ||
344
- (metadata.resolved !== null &&
345
- has_native(type)(visited)(metadata.resolved.returns))
346
- );
347
- };
348
-
349
- const has_array =
350
- (visited: Set<Metadata>) =>
351
- (metadata: Metadata): boolean => {
352
- if (visited.has(metadata)) return false;
353
- visited.add(metadata);
354
- return (
355
- metadata.arrays.length !== 0 ||
356
- metadata.tuples.some((tuple) =>
357
- tuple.elements.some(has_array(visited)),
358
- ) ||
359
- metadata.maps.some((map) => has_array(visited)(map.value)) ||
360
- metadata.aliases.some((alias) => has_array(visited)(alias.value)) ||
361
- (metadata.resolved !== null &&
362
- has_array(visited)(metadata.resolved.returns))
363
- );
364
- };
1
+ import ts from "typescript";
2
+
3
+ import { IMetadataAtomic } from "../schemas/metadata/IMetadataAtomic";
4
+ import { IMetadataTag } from "../schemas/metadata/IMetadataTag";
5
+ import { Metadata } from "../schemas/metadata/Metadata";
6
+
7
+ export namespace MetadataTagFactory {
8
+ export const generate =
9
+ (metadata: Metadata) =>
10
+ (tagList: ts.JSDocTagInfo[]) =>
11
+ (identifier: () => string): IMetadataTag[] => {
12
+ const output: IMetadataTag[] = [];
13
+ for (const tag of tagList) {
14
+ const elem: IMetadataTag | null = parse(
15
+ identifier,
16
+ metadata,
17
+ tag,
18
+ output,
19
+ );
20
+ if (elem !== null) output.push(elem);
21
+ }
22
+ return output;
23
+ };
24
+
25
+ const parse = (
26
+ identifier: () => string,
27
+ metadata: Metadata,
28
+ tag: ts.JSDocTagInfo,
29
+ output: IMetadataTag[],
30
+ ): IMetadataTag | null => {
31
+ const closure = _PARSER[tag.name];
32
+ if (closure === undefined) return null;
33
+
34
+ const text = (tag.text || [])[0]?.text;
35
+ if (text === undefined)
36
+ throw new Error(`${LABEL}: no tag value on ${identifier()}`);
37
+
38
+ return closure(identifier, metadata, text, output);
39
+ };
40
+
41
+ /**
42
+ * @internal
43
+ */
44
+ export const _PARSER: Record<
45
+ string,
46
+ (
47
+ identifier: () => string,
48
+ metadata: Metadata,
49
+ text: string,
50
+ output: IMetadataTag[],
51
+ ) => IMetadataTag | null
52
+ > = {
53
+ /* -----------------------------------------------------------
54
+ ARRAY
55
+ ----------------------------------------------------------- */
56
+ items: (identifier, metadata, text, output) => {
57
+ validate(identifier, metadata, output, "items", "array", [
58
+ "minItems",
59
+ ]);
60
+ return {
61
+ kind: "items",
62
+ value: parse_number(identifier, text),
63
+ };
64
+ },
65
+ minItems: (identifier, metadata, text, output) => {
66
+ validate(identifier, metadata, output, "minItems", "array", [
67
+ "items",
68
+ ]);
69
+ return {
70
+ kind: "minItems",
71
+ value: parse_number(identifier, text),
72
+ };
73
+ },
74
+ maxItems: (identifier, metadata, text, output) => {
75
+ validate(identifier, metadata, output, "maxItems", "array", [
76
+ "items",
77
+ ]);
78
+ return {
79
+ kind: "maxItems",
80
+ value: parse_number(identifier, text),
81
+ };
82
+ },
83
+
84
+ /* -----------------------------------------------------------
85
+ NUMBER
86
+ ----------------------------------------------------------- */
87
+ type: (_identifier, metadata, text, _output) => {
88
+ if (text.startsWith("{") && text.endsWith("}"))
89
+ text = text.substring(1, text.length - 1);
90
+ return has_atomic("number")(new Set())(metadata) &&
91
+ (text === "int" ||
92
+ text === "uint" ||
93
+ text === "int32" ||
94
+ text === "uint32" ||
95
+ text === "int64" ||
96
+ text === "uint64" ||
97
+ text === "float")
98
+ ? { kind: "type", value: text }
99
+ : null;
100
+ },
101
+ minimum: (identifier, metadata, text, output) => {
102
+ validate(identifier, metadata, output, "minimum", "number", [
103
+ "exclusiveMinimum",
104
+ ]);
105
+ return {
106
+ kind: "minimum",
107
+ value: parse_number(identifier, text),
108
+ };
109
+ },
110
+ maximum: (identifier, metadata, text, output) => {
111
+ validate(identifier, metadata, output, "maximum", "number", [
112
+ "exclusiveMaximum",
113
+ ]);
114
+ return {
115
+ kind: "maximum",
116
+ value: parse_number(identifier, text),
117
+ };
118
+ },
119
+ exclusiveMinimum: (identifier, metadata, text, output) => {
120
+ validate(
121
+ identifier,
122
+ metadata,
123
+ output,
124
+ "exclusiveMinimum",
125
+ "number",
126
+ ["minimum"],
127
+ );
128
+ return {
129
+ kind: "exclusiveMinimum",
130
+ value: parse_number(identifier, text),
131
+ };
132
+ },
133
+ exclusiveMaximum: (identifier, metadata, text, output) => {
134
+ validate(
135
+ identifier,
136
+ metadata,
137
+ output,
138
+ "exclusiveMaximum",
139
+ "number",
140
+ ["maximum"],
141
+ );
142
+ return {
143
+ kind: "exclusiveMaximum",
144
+ value: parse_number(identifier, text),
145
+ };
146
+ },
147
+ multipleOf: (identifier, metadata, text, output) => {
148
+ validate(identifier, metadata, output, "multipleOf", "number", [
149
+ "step",
150
+ ]);
151
+ return {
152
+ kind: "multipleOf",
153
+ value: parse_number(identifier, text),
154
+ };
155
+ },
156
+ step: (identifier, metadata, text, output) => {
157
+ validate(identifier, metadata, output, "step", "number", [
158
+ "multipleOf",
159
+ ]);
160
+
161
+ const minimum: boolean = output.some(
162
+ (tag) =>
163
+ tag.kind === "minimum" || tag.kind === "exclusiveMinimum",
164
+ );
165
+ if (minimum === undefined)
166
+ throw new Error(
167
+ `${LABEL}: step requires minimum or exclusiveMinimum tag on "${identifier()}".`,
168
+ );
169
+
170
+ return {
171
+ kind: "step",
172
+ value: parse_number(identifier, text),
173
+ };
174
+ },
175
+
176
+ /* -----------------------------------------------------------
177
+ STRING
178
+ ----------------------------------------------------------- */
179
+ // Ignore arbitrary @format values in the internal metadata,
180
+ // these are currently only supported on the typia.application() API.
181
+ format: (identifier, metadata, str, output) => {
182
+ const value: IMetadataTag.IFormat["value"] | undefined =
183
+ FORMATS.get(str);
184
+ validate(
185
+ identifier,
186
+ metadata,
187
+ output,
188
+ "format",
189
+ value === "date" || value === "datetime" ? "Date" : "string",
190
+ ["pattern"],
191
+ );
192
+ if (value === undefined) return null;
193
+ return {
194
+ kind: "format",
195
+ value,
196
+ };
197
+ },
198
+ pattern: (identifier, metadata, value, output) => {
199
+ validate(identifier, metadata, output, "pattern", "string", [
200
+ "format",
201
+ ]);
202
+ return {
203
+ kind: "pattern",
204
+ value,
205
+ };
206
+ },
207
+ length: (identifier, metadata, text, output) => {
208
+ validate(identifier, metadata, output, "length", "string", [
209
+ "minLength",
210
+ "maxLength",
211
+ ]);
212
+ return {
213
+ kind: "length",
214
+ value: parse_number(identifier, text),
215
+ };
216
+ },
217
+ minLength: (identifier, metadata, text, output) => {
218
+ validate(identifier, metadata, output, "minLength", "string", [
219
+ "length",
220
+ ]);
221
+ return {
222
+ kind: "minLength",
223
+ value: parse_number(identifier, text),
224
+ };
225
+ },
226
+ maxLength: (identifier, metadata, text, output) => {
227
+ validate(identifier, metadata, output, "maxLength", "string", [
228
+ "length",
229
+ ]);
230
+ return {
231
+ kind: "maxLength",
232
+ value: parse_number(identifier, text),
233
+ };
234
+ },
235
+ };
236
+ }
237
+
238
+ const parse_number = (identifier: () => string, str: string): number => {
239
+ const value: number = Number(str);
240
+ if (isNaN(value) === true)
241
+ throw new Error(`${LABEL}: invalid number on "${identifier()}".`);
242
+ return value;
243
+ };
244
+
245
+ const LABEL = "Error on typia.MetadataTagFactory.generate()";
246
+ const FORMATS: Map<string, IMetadataTag.IFormat["value"]> = new Map([
247
+ ["uuid", "uuid"],
248
+ ["email", "email"],
249
+ ["url", "url"],
250
+ ["ipv4", "ipv4"],
251
+ ["ipv6", "ipv6"],
252
+ ["date", "date"],
253
+ ["datetime", "datetime"],
254
+ ["date-time", "datetime"],
255
+ ["dateTime", "datetime"],
256
+ ]);
257
+
258
+ const WRONG_TYPE = (
259
+ tag: string,
260
+ type: "string" | "number" | "array",
261
+ identifier: () => string,
262
+ ) => `${LABEL}: ${tag} requires ${type} type, but no "${identifier()}".`;
263
+
264
+ const validate = (
265
+ identifier: () => string,
266
+ metadata: Metadata,
267
+ output: IMetadataTag[],
268
+ kind: IMetadataTag["kind"],
269
+ type: "array" | "string" | "number" | "Date",
270
+ neighbors: IMetadataTag["kind"][],
271
+ ): void => {
272
+ // TYPE CHECKING
273
+ if (type === "array") {
274
+ if (has_array(new Set())(metadata) === false)
275
+ throw new Error(WRONG_TYPE(kind, "array", identifier));
276
+ } else if (type === "Date") {
277
+ if (
278
+ has_native("Date")(new Set())(metadata) === false &&
279
+ has_atomic("string")(new Set())(metadata) === false
280
+ )
281
+ throw new Error(WRONG_TYPE(kind, "string", identifier));
282
+ } else if (has_atomic(type)(new Set())(metadata) === false)
283
+ throw new Error(WRONG_TYPE(kind, type, identifier));
284
+
285
+ // DUPLICATED TAG
286
+ if (output.some((tag) => tag.kind === kind))
287
+ throw new Error(
288
+ `${LABEL}: duplicated ${kind} tags on "${identifier()}".`,
289
+ );
290
+
291
+ // NEIGHBOR TAG
292
+ for (const name of neighbors)
293
+ if (output.some((tag) => tag.kind === name))
294
+ throw new Error(
295
+ `${LABEL}: ${kind} and ${name} tags on "${identifier()}".`,
296
+ );
297
+ };
298
+
299
+ // @todo: must block repeated array and tuple type
300
+ const has_atomic =
301
+ (type: "string" | "number" | "bigint") =>
302
+ (visited: Set<Metadata>) =>
303
+ (metadata: Metadata): boolean => {
304
+ if (visited.has(metadata)) return false;
305
+ visited.add(metadata);
306
+ return (
307
+ metadata.atomics.find(
308
+ type === "number"
309
+ ? (atom: IMetadataAtomic) =>
310
+ atom.type === type || atom.type === "bigint"
311
+ : (atom: IMetadataAtomic) => atom.type === type,
312
+ ) !== undefined ||
313
+ metadata.arrays.some((array) =>
314
+ has_atomic(type)(visited)(array.value),
315
+ ) ||
316
+ metadata.tuples.some((tuple) =>
317
+ tuple.elements.some(has_atomic(type)(visited)),
318
+ ) ||
319
+ metadata.maps.some((map) => has_atomic(type)(visited)(map.value)) ||
320
+ metadata.aliases.some((alias) =>
321
+ has_atomic(type)(visited)(alias.value),
322
+ ) ||
323
+ (metadata.escaped !== null &&
324
+ has_atomic(type)(visited)(metadata.escaped.returns))
325
+ );
326
+ };
327
+
328
+ const has_native =
329
+ (type: string) =>
330
+ (visited: Set<Metadata>) =>
331
+ (metadata: Metadata): boolean => {
332
+ if (visited.has(metadata)) return false;
333
+ visited.add(metadata);
334
+ return (
335
+ metadata.natives.find((native) => native === type) !== undefined ||
336
+ metadata.arrays.some((child) =>
337
+ has_native(type)(visited)(child.value),
338
+ ) ||
339
+ metadata.tuples.some((tuple) =>
340
+ tuple.elements.some(has_native(type)(visited)),
341
+ ) ||
342
+ metadata.maps.some((map) => has_native(type)(visited)(map.value)) ||
343
+ metadata.aliases.some((alias) =>
344
+ has_native(type)(visited)(alias.value),
345
+ ) ||
346
+ (metadata.escaped !== null &&
347
+ has_native(type)(visited)(metadata.escaped.returns))
348
+ );
349
+ };
350
+
351
+ const has_array =
352
+ (visited: Set<Metadata>) =>
353
+ (metadata: Metadata): boolean => {
354
+ if (visited.has(metadata)) return false;
355
+ visited.add(metadata);
356
+ return (
357
+ metadata.arrays.length !== 0 ||
358
+ metadata.tuples.some((tuple) =>
359
+ tuple.elements.some(has_array(visited)),
360
+ ) ||
361
+ metadata.maps.some((map) => has_array(visited)(map.value)) ||
362
+ metadata.aliases.some((alias) => has_array(visited)(alias.value)) ||
363
+ (metadata.escaped !== null &&
364
+ has_array(visited)(metadata.escaped.returns))
365
+ );
366
+ };