typia 4.2.3-dev.20230814 → 4.3.0-dev.20230821

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 (260) hide show
  1. package/README.md +1 -1
  2. package/lib/factories/ExpressionFactory.d.ts +1 -0
  3. package/lib/factories/ExpressionFactory.js +3 -0
  4. package/lib/factories/ExpressionFactory.js.map +1 -1
  5. package/lib/factories/MetadataTagFactory.js +10 -4
  6. package/lib/factories/MetadataTagFactory.js.map +1 -1
  7. package/lib/metadata/IMetadataTag.d.ts +1 -1
  8. package/lib/metadata/MetadataTuple.js.map +1 -1
  9. package/lib/programmers/CheckerProgrammer.js +2 -2
  10. package/lib/programmers/CheckerProgrammer.js.map +1 -1
  11. package/lib/programmers/LiteralsProgrammer.js +2 -1
  12. package/lib/programmers/LiteralsProgrammer.js.map +1 -1
  13. package/lib/programmers/RandomProgrammer.js +19 -10
  14. package/lib/programmers/RandomProgrammer.js.map +1 -1
  15. package/lib/programmers/helpers/RandomRanger.js +2 -2
  16. package/lib/programmers/helpers/RandomRanger.js.map +1 -1
  17. package/lib/programmers/helpers/UnionExplorer.js +11 -7
  18. package/lib/programmers/helpers/UnionExplorer.js.map +1 -1
  19. package/lib/programmers/internal/application_number.js +10 -3
  20. package/lib/programmers/internal/application_number.js.map +1 -1
  21. package/lib/programmers/internal/check_bigint.js +16 -12
  22. package/lib/programmers/internal/check_bigint.js.map +1 -1
  23. package/lib/programmers/internal/check_number.js +38 -5
  24. package/lib/programmers/internal/check_number.js.map +1 -1
  25. package/package.json +2 -2
  26. package/src/CustomValidatorMap.ts +126 -126
  27. package/src/IRandomGenerator.ts +34 -34
  28. package/src/IValidation.ts +21 -21
  29. package/src/Primitive.ts +131 -131
  30. package/src/TypeGuardError.ts +36 -36
  31. package/src/executable/TypiaGenerateWizard.ts +85 -85
  32. package/src/executable/TypiaSetupWizard.ts +153 -153
  33. package/src/executable/setup/ArgumentParser.ts +45 -45
  34. package/src/executable/setup/CommandExecutor.ts +8 -8
  35. package/src/executable/setup/FileRetriever.ts +22 -22
  36. package/src/executable/setup/PackageManager.ts +71 -71
  37. package/src/executable/setup/PluginConfigurator.ts +70 -70
  38. package/src/executable/typia.ts +52 -52
  39. package/src/factories/CommentFactory.ts +84 -84
  40. package/src/factories/ExpressionFactory.ts +77 -70
  41. package/src/factories/IdentifierFactory.ts +59 -59
  42. package/src/factories/LiteralFactory.ts +39 -39
  43. package/src/factories/MetadataCollection.ts +269 -269
  44. package/src/factories/MetadataFactory.ts +34 -34
  45. package/src/factories/MetadataTagFactory.ts +361 -355
  46. package/src/factories/StatementFactory.ts +24 -24
  47. package/src/factories/TemplateFactory.ts +58 -58
  48. package/src/factories/TypeFactory.ts +124 -124
  49. package/src/factories/ValueFactory.ts +12 -12
  50. package/src/factories/internal/metadata/MetadataHelper.ts +12 -12
  51. package/src/factories/internal/metadata/emend_metadata_atomics.ts +33 -33
  52. package/src/factories/internal/metadata/emplace_metadata_alias.ts +40 -40
  53. package/src/factories/internal/metadata/emplace_metadata_array.ts +34 -34
  54. package/src/factories/internal/metadata/emplace_metadata_object.ts +136 -136
  55. package/src/factories/internal/metadata/emplace_metadata_tuple.ts +50 -50
  56. package/src/factories/internal/metadata/explore_metadata.ts +38 -38
  57. package/src/factories/internal/metadata/iterate_metadata.ts +81 -81
  58. package/src/factories/internal/metadata/iterate_metadata_alias.ts +30 -30
  59. package/src/factories/internal/metadata/iterate_metadata_array.ts +24 -24
  60. package/src/factories/internal/metadata/iterate_metadata_atomic.ts +59 -59
  61. package/src/factories/internal/metadata/iterate_metadata_coalesce.ts +33 -33
  62. package/src/factories/internal/metadata/iterate_metadata_collection.ts +133 -133
  63. package/src/factories/internal/metadata/iterate_metadata_constant.ts +58 -58
  64. package/src/factories/internal/metadata/iterate_metadata_intersection.ts +84 -84
  65. package/src/factories/internal/metadata/iterate_metadata_map.ts +41 -41
  66. package/src/factories/internal/metadata/iterate_metadata_native.ts +219 -219
  67. package/src/factories/internal/metadata/iterate_metadata_object.ts +43 -43
  68. package/src/factories/internal/metadata/iterate_metadata_resolve.ts +49 -49
  69. package/src/factories/internal/metadata/iterate_metadata_set.ts +33 -33
  70. package/src/factories/internal/metadata/iterate_metadata_sort.ts +69 -69
  71. package/src/factories/internal/metadata/iterate_metadata_tag.ts +31 -31
  72. package/src/factories/internal/metadata/iterate_metadata_template.ts +38 -38
  73. package/src/factories/internal/metadata/iterate_metadata_tuple.ts +24 -24
  74. package/src/factories/internal/metadata/iterate_metadata_union.ts +24 -24
  75. package/src/functional/$any.ts +2 -2
  76. package/src/functional/$dictionary.ts +25 -25
  77. package/src/functional/$every.ts +11 -11
  78. package/src/functional/$guard.ts +35 -35
  79. package/src/functional/$is_between.ts +2 -2
  80. package/src/functional/$is_custom.ts +14 -14
  81. package/src/functional/$is_date.ts +3 -3
  82. package/src/functional/$is_datetime.ts +2 -2
  83. package/src/functional/$is_email.ts +4 -4
  84. package/src/functional/$is_ipv4.ts +4 -4
  85. package/src/functional/$is_ipv6.ts +4 -4
  86. package/src/functional/$is_url.ts +4 -4
  87. package/src/functional/$is_uuid.ts +4 -4
  88. package/src/functional/$join.ts +46 -46
  89. package/src/functional/$number.ts +12 -12
  90. package/src/functional/$report.ts +15 -15
  91. package/src/functional/$rest.ts +3 -3
  92. package/src/functional/$string.ts +50 -50
  93. package/src/functional/$tail.ts +5 -5
  94. package/src/functional/Namespace.ts +127 -127
  95. package/src/index.ts +4 -4
  96. package/src/metadata/ICommentTag.ts +4 -4
  97. package/src/metadata/IJsDocTagInfo.ts +10 -10
  98. package/src/metadata/IMetadata.ts +28 -28
  99. package/src/metadata/IMetadataAlias.ts +14 -14
  100. package/src/metadata/IMetadataApplication.ts +7 -7
  101. package/src/metadata/IMetadataArray.ts +10 -10
  102. package/src/metadata/IMetadataCollection.ts +11 -11
  103. package/src/metadata/IMetadataConstant.ts +16 -16
  104. package/src/metadata/IMetadataDictionary.ts +14 -14
  105. package/src/metadata/IMetadataEntry.ts +6 -6
  106. package/src/metadata/IMetadataObject.ts +18 -18
  107. package/src/metadata/IMetadataProperty.ts +11 -11
  108. package/src/metadata/IMetadataResolved.ts +6 -6
  109. package/src/metadata/IMetadataTag.ts +112 -105
  110. package/src/metadata/IMetadataTuple.ts +10 -10
  111. package/src/metadata/Metadata.ts +607 -607
  112. package/src/metadata/MetadataAlias.ts +66 -66
  113. package/src/metadata/MetadataArray.ts +55 -55
  114. package/src/metadata/MetadataConstant.ts +3 -3
  115. package/src/metadata/MetadataObject.ts +129 -129
  116. package/src/metadata/MetadataProperty.ts +64 -64
  117. package/src/metadata/MetadataResolved.ts +51 -51
  118. package/src/metadata/MetadataTuple.ts +58 -53
  119. package/src/module.ts +2038 -2038
  120. package/src/programmers/ApplicationProgrammer.ts +47 -47
  121. package/src/programmers/AssertCloneProgrammer.ts +71 -71
  122. package/src/programmers/AssertParseProgrammer.ts +66 -66
  123. package/src/programmers/AssertProgrammer.ts +279 -279
  124. package/src/programmers/AssertPruneProgrammer.ts +68 -68
  125. package/src/programmers/AssertStringifyProgrammer.ts +66 -66
  126. package/src/programmers/CheckerProgrammer.ts +1173 -1173
  127. package/src/programmers/CloneProgrammer.ts +587 -587
  128. package/src/programmers/FeatureProgrammer.ts +495 -495
  129. package/src/programmers/IsCloneProgrammer.ts +78 -78
  130. package/src/programmers/IsParseProgrammer.ts +72 -72
  131. package/src/programmers/IsProgrammer.ts +239 -239
  132. package/src/programmers/IsPruneProgrammer.ts +73 -73
  133. package/src/programmers/IsStringifyProgrammer.ts +76 -76
  134. package/src/programmers/LiteralsProgrammer.ts +64 -63
  135. package/src/programmers/PruneProgrammer.ts +542 -542
  136. package/src/programmers/RandomProgrammer.ts +578 -579
  137. package/src/programmers/StringifyProgrammer.ts +986 -986
  138. package/src/programmers/TypiaProgrammer.ts +129 -129
  139. package/src/programmers/ValidateCloneProgrammer.ts +85 -85
  140. package/src/programmers/ValidateParseProgrammer.ts +70 -70
  141. package/src/programmers/ValidateProgrammer.ts +305 -305
  142. package/src/programmers/ValidatePruneProgrammer.ts +78 -78
  143. package/src/programmers/ValidateStringifyProgrammer.ts +84 -84
  144. package/src/programmers/helpers/AtomicPredicator.ts +31 -31
  145. package/src/programmers/helpers/CloneJoiner.ts +131 -131
  146. package/src/programmers/helpers/FunctionImporeter.ts +78 -78
  147. package/src/programmers/helpers/ICheckEntry.ts +12 -12
  148. package/src/programmers/helpers/IExpressionEntry.ts +12 -12
  149. package/src/programmers/helpers/OptionPredicator.ts +15 -15
  150. package/src/programmers/helpers/PruneJoiner.ts +143 -143
  151. package/src/programmers/helpers/RandomJoiner.ts +173 -173
  152. package/src/programmers/helpers/RandomRanger.ts +208 -216
  153. package/src/programmers/helpers/StringifyJoinder.ts +113 -113
  154. package/src/programmers/helpers/StringifyPredicator.ts +13 -13
  155. package/src/programmers/helpers/UnionExplorer.ts +305 -301
  156. package/src/programmers/helpers/UnionPredicator.ts +81 -81
  157. package/src/programmers/helpers/disable_function_importer_declare.ts +26 -26
  158. package/src/programmers/internal/JSON_SCHEMA_PREFIX.ts +1 -1
  159. package/src/programmers/internal/application_alias.ts +66 -66
  160. package/src/programmers/internal/application_array.ts +30 -30
  161. package/src/programmers/internal/application_boolean.ts +15 -15
  162. package/src/programmers/internal/application_constant.ts +26 -26
  163. package/src/programmers/internal/application_default.ts +17 -17
  164. package/src/programmers/internal/application_default_string.ts +33 -33
  165. package/src/programmers/internal/application_native.ts +39 -39
  166. package/src/programmers/internal/application_number.ts +80 -74
  167. package/src/programmers/internal/application_object.ts +165 -165
  168. package/src/programmers/internal/application_resolved.ts +55 -55
  169. package/src/programmers/internal/application_schema.ts +157 -157
  170. package/src/programmers/internal/application_string.ts +44 -44
  171. package/src/programmers/internal/application_templates.ts +25 -25
  172. package/src/programmers/internal/application_tuple.ts +57 -57
  173. package/src/programmers/internal/check_array.ts +30 -30
  174. package/src/programmers/internal/check_array_length.ts +35 -35
  175. package/src/programmers/internal/check_bigint.ts +113 -82
  176. package/src/programmers/internal/check_custom.ts +31 -31
  177. package/src/programmers/internal/check_dynamic_properties.ts +195 -195
  178. package/src/programmers/internal/check_everything.ts +28 -28
  179. package/src/programmers/internal/check_native.ts +21 -21
  180. package/src/programmers/internal/check_number.ts +258 -177
  181. package/src/programmers/internal/check_object.ts +55 -55
  182. package/src/programmers/internal/check_string.ts +25 -25
  183. package/src/programmers/internal/check_string_tags.ts +67 -67
  184. package/src/programmers/internal/check_template.ts +56 -56
  185. package/src/programmers/internal/check_union_array_like.ts +329 -329
  186. package/src/programmers/internal/decode_union_object.ts +73 -73
  187. package/src/programmers/internal/feature_object_entries.ts +63 -63
  188. package/src/programmers/internal/get_comment_tags.ts +23 -23
  189. package/src/programmers/internal/metadata_to_pattern.ts +34 -34
  190. package/src/programmers/internal/prune_object_properties.ts +60 -60
  191. package/src/programmers/internal/random_custom.ts +29 -29
  192. package/src/programmers/internal/stringify_dynamic_properties.ts +171 -171
  193. package/src/programmers/internal/stringify_native.ts +7 -7
  194. package/src/programmers/internal/stringify_regular_properties.ts +83 -83
  195. package/src/programmers/internal/template_to_pattern.ts +15 -15
  196. package/src/programmers/internal/wrap_metadata_rest_tuple.ts +16 -16
  197. package/src/schemas/IJsonApplication.ts +8 -8
  198. package/src/schemas/IJsonComponents.ts +33 -33
  199. package/src/schemas/IJsonSchema.ts +133 -133
  200. package/src/transform.ts +27 -27
  201. package/src/transformers/CallExpressionTransformer.ts +179 -179
  202. package/src/transformers/FileTransformer.ts +47 -47
  203. package/src/transformers/IProject.ts +11 -11
  204. package/src/transformers/ITransformOptions.ts +62 -62
  205. package/src/transformers/ImportTransformer.ts +66 -66
  206. package/src/transformers/NodeTransformer.ts +13 -13
  207. package/src/transformers/features/miscellaneous/ApplicationTransformer.ts +112 -112
  208. package/src/transformers/features/miscellaneous/AssertCloneTransformer.ts +9 -9
  209. package/src/transformers/features/miscellaneous/AssertPruneTransformer.ts +9 -9
  210. package/src/transformers/features/miscellaneous/CloneTransformer.ts +9 -9
  211. package/src/transformers/features/miscellaneous/CreateAssertCloneTransformer.ts +9 -9
  212. package/src/transformers/features/miscellaneous/CreateAssertPruneTransformer.ts +9 -9
  213. package/src/transformers/features/miscellaneous/CreateCloneTransformer.ts +9 -9
  214. package/src/transformers/features/miscellaneous/CreateIsCloneTransformer.ts +9 -9
  215. package/src/transformers/features/miscellaneous/CreateIsPruneTransformer.ts +9 -9
  216. package/src/transformers/features/miscellaneous/CreatePruneTransformer.ts +9 -9
  217. package/src/transformers/features/miscellaneous/CreateRandomTransformer.ts +39 -39
  218. package/src/transformers/features/miscellaneous/CreateValidateCloneTransformer.ts +9 -9
  219. package/src/transformers/features/miscellaneous/CreateValidatePruneTransformer.ts +9 -9
  220. package/src/transformers/features/miscellaneous/IsCloneTransformer.ts +9 -9
  221. package/src/transformers/features/miscellaneous/IsPruneTransformer.ts +9 -9
  222. package/src/transformers/features/miscellaneous/LiteralsTransformer.ts +28 -28
  223. package/src/transformers/features/miscellaneous/MetadataTransformer.ts +53 -53
  224. package/src/transformers/features/miscellaneous/PruneTransformer.ts +9 -9
  225. package/src/transformers/features/miscellaneous/RandomTransformer.ts +42 -42
  226. package/src/transformers/features/miscellaneous/ValidateCloneTransformer.ts +9 -9
  227. package/src/transformers/features/miscellaneous/ValidatePruneTransformer.ts +9 -9
  228. package/src/transformers/features/parsers/AssertParseTransformer.ts +9 -9
  229. package/src/transformers/features/parsers/CreateAssertParseTransformer.ts +9 -9
  230. package/src/transformers/features/parsers/CreateIsParseTransformer.ts +9 -9
  231. package/src/transformers/features/parsers/CreateValidateParseTransformer.ts +9 -9
  232. package/src/transformers/features/parsers/IsParseTransformer.ts +9 -9
  233. package/src/transformers/features/parsers/ValidateParseTransformer.ts +9 -9
  234. package/src/transformers/features/stringifiers/AssertStringifyTransformer.ts +10 -10
  235. package/src/transformers/features/stringifiers/CreateAssertStringifyTransformer.ts +12 -12
  236. package/src/transformers/features/stringifiers/CreateIsStringifyTransformer.ts +9 -9
  237. package/src/transformers/features/stringifiers/CreateStringifyTransformer.ts +9 -9
  238. package/src/transformers/features/stringifiers/CreateValidateStringifyProgrammer.ts +12 -12
  239. package/src/transformers/features/stringifiers/IsStringifyTransformer.ts +9 -9
  240. package/src/transformers/features/stringifiers/StringifyTransformer.ts +9 -9
  241. package/src/transformers/features/stringifiers/ValidateStringifyTransformer.ts +10 -10
  242. package/src/transformers/features/validators/AssertTransformer.ts +11 -11
  243. package/src/transformers/features/validators/CreateAssertTransformer.ts +13 -13
  244. package/src/transformers/features/validators/CreateIsTransformer.ts +11 -11
  245. package/src/transformers/features/validators/CreateValidateTransformer.ts +13 -13
  246. package/src/transformers/features/validators/IsTransformer.ts +11 -11
  247. package/src/transformers/features/validators/ValidateTransformer.ts +11 -11
  248. package/src/transformers/internal/GenericTransformer.ts +97 -97
  249. package/src/typings/Atomic.ts +17 -17
  250. package/src/typings/ClassProperties.ts +5 -5
  251. package/src/typings/Customizable.ts +5 -5
  252. package/src/typings/OmitNever.ts +3 -3
  253. package/src/typings/SpecialFields.ts +3 -3
  254. package/src/typings/Writable.ts +11 -11
  255. package/src/utils/ArrayUtil.ts +45 -45
  256. package/src/utils/Escaper.ts +46 -46
  257. package/src/utils/MapUtil.ts +12 -12
  258. package/src/utils/PatternUtil.ts +33 -33
  259. package/src/utils/RandomGenerator.ts +81 -81
  260. package/src/utils/Singleton.ts +17 -17
@@ -1,355 +1,361 @@
1
- import ts from "typescript";
2
-
3
- import { IMetadataTag } from "../metadata/IMetadataTag";
4
- import { Metadata } from "../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
- return has_atomic("number")(new Set())(metadata) &&
88
- (text === "int" || text === "uint")
89
- ? { kind: "type", value: text }
90
- : text === "{int}" || text === "{uint}"
91
- ? { kind: "type", value: text.slice(1, -1) as "int" | "uint" }
92
- : null;
93
- },
94
- minimum: (identifier, metadata, text, output) => {
95
- validate(identifier, metadata, output, "minimum", "number", [
96
- "exclusiveMinimum",
97
- ]);
98
- return {
99
- kind: "minimum",
100
- value: parse_number(identifier, text),
101
- };
102
- },
103
- maximum: (identifier, metadata, text, output) => {
104
- validate(identifier, metadata, output, "maximum", "number", [
105
- "exclusiveMaximum",
106
- ]);
107
- return {
108
- kind: "maximum",
109
- value: parse_number(identifier, text),
110
- };
111
- },
112
- exclusiveMinimum: (identifier, metadata, text, output) => {
113
- validate(
114
- identifier,
115
- metadata,
116
- output,
117
- "exclusiveMinimum",
118
- "number",
119
- ["minimum"],
120
- );
121
- return {
122
- kind: "exclusiveMinimum",
123
- value: parse_number(identifier, text),
124
- };
125
- },
126
- exclusiveMaximum: (identifier, metadata, text, output) => {
127
- validate(
128
- identifier,
129
- metadata,
130
- output,
131
- "exclusiveMaximum",
132
- "number",
133
- ["maximum"],
134
- );
135
- return {
136
- kind: "exclusiveMaximum",
137
- value: parse_number(identifier, text),
138
- };
139
- },
140
- multipleOf: (identifier, metadata, text, output) => {
141
- validate(identifier, metadata, output, "multipleOf", "number", [
142
- "step",
143
- ]);
144
- return {
145
- kind: "multipleOf",
146
- value: parse_number(identifier, text),
147
- };
148
- },
149
- step: (identifier, metadata, text, output) => {
150
- validate(identifier, metadata, output, "step", "number", [
151
- "multipleOf",
152
- ]);
153
-
154
- const minimum: boolean = output.some(
155
- (tag) =>
156
- tag.kind === "minimum" || tag.kind === "exclusiveMinimum",
157
- );
158
- if (minimum === undefined)
159
- throw new Error(
160
- `${LABEL}: step requires minimum or exclusiveMinimum tag on "${identifier()}".`,
161
- );
162
-
163
- return {
164
- kind: "step",
165
- value: parse_number(identifier, text),
166
- };
167
- },
168
-
169
- /* -----------------------------------------------------------
170
- STRING
171
- ----------------------------------------------------------- */
172
- // Ignore arbitrary @format values in the internal metadata,
173
- // these are currently only supported on the typia.application() API.
174
- format: (identifier, metadata, str, output) => {
175
- const value: IMetadataTag.IFormat["value"] | undefined =
176
- FORMATS.get(str);
177
- validate(
178
- identifier,
179
- metadata,
180
- output,
181
- "format",
182
- value === "date" || value === "datetime" ? "Date" : "string",
183
- ["pattern"],
184
- );
185
- if (value === undefined) return null;
186
- return {
187
- kind: "format",
188
- value,
189
- };
190
- },
191
- pattern: (identifier, metadata, value, output) => {
192
- validate(identifier, metadata, output, "pattern", "string", [
193
- "format",
194
- ]);
195
- return {
196
- kind: "pattern",
197
- value,
198
- };
199
- },
200
- length: (identifier, metadata, text, output) => {
201
- validate(identifier, metadata, output, "length", "string", [
202
- "minLength",
203
- "maxLength",
204
- ]);
205
- return {
206
- kind: "length",
207
- value: parse_number(identifier, text),
208
- };
209
- },
210
- minLength: (identifier, metadata, text, output) => {
211
- validate(identifier, metadata, output, "minLength", "string", [
212
- "length",
213
- ]);
214
- return {
215
- kind: "minLength",
216
- value: parse_number(identifier, text),
217
- };
218
- },
219
- maxLength: (identifier, metadata, text, output) => {
220
- validate(identifier, metadata, output, "maxLength", "string", [
221
- "length",
222
- ]);
223
- return {
224
- kind: "maxLength",
225
- value: parse_number(identifier, text),
226
- };
227
- },
228
- };
229
- }
230
-
231
- const parse_number = (identifier: () => string, str: string): number => {
232
- const value: number = Number(str);
233
- if (isNaN(value) === true)
234
- throw new Error(`${LABEL}: invalid number on "${identifier()}".`);
235
- return value;
236
- };
237
-
238
- const LABEL = "Error on typia.MetadataTagFactory.generate()";
239
- const FORMATS: Map<string, IMetadataTag.IFormat["value"]> = new Map([
240
- ["uuid", "uuid"],
241
- ["email", "email"],
242
- ["url", "url"],
243
- ["ipv4", "ipv4"],
244
- ["ipv6", "ipv6"],
245
- ["date", "date"],
246
- ["datetime", "datetime"],
247
- ["date-time", "datetime"],
248
- ["dateTime", "datetime"],
249
- ]);
250
-
251
- const WRONG_TYPE = (
252
- tag: string,
253
- type: "string" | "number" | "array",
254
- identifier: () => string,
255
- ) => `${LABEL}: ${tag} requires ${type} type, but no "${identifier()}".`;
256
-
257
- const validate = (
258
- identifier: () => string,
259
- metadata: Metadata,
260
- output: IMetadataTag[],
261
- kind: IMetadataTag["kind"],
262
- type: "array" | "string" | "number" | "Date",
263
- neighbors: IMetadataTag["kind"][],
264
- ): void => {
265
- // TYPE CHECKING
266
- if (type === "array") {
267
- if (has_array(new Set())(metadata) === false)
268
- throw new Error(WRONG_TYPE(kind, "array", identifier));
269
- } else if (type === "Date") {
270
- if (
271
- has_native("Date")(new Set())(metadata) === false &&
272
- has_atomic("string")(new Set())(metadata) === false
273
- )
274
- throw new Error(WRONG_TYPE(kind, "string", identifier));
275
- } else if (has_atomic(type)(new Set())(metadata) === false)
276
- throw new Error(WRONG_TYPE(kind, type, identifier));
277
-
278
- // DUPLICATED TAG
279
- if (output.some((tag) => tag.kind === kind))
280
- throw new Error(
281
- `${LABEL}: duplicated ${kind} tags on "${identifier()}".`,
282
- );
283
-
284
- // NEIGHBOR TAG
285
- for (const name of neighbors)
286
- if (output.some((tag) => tag.kind === name))
287
- throw new Error(
288
- `${LABEL}: ${kind} and ${name} tags on "${identifier()}".`,
289
- );
290
- };
291
-
292
- // @todo: must block repeated array and tuple type
293
- const has_atomic =
294
- (type: "string" | "number") =>
295
- (visited: Set<Metadata>) =>
296
- (metadata: Metadata): boolean => {
297
- if (visited.has(metadata)) return false;
298
- visited.add(metadata);
299
- return (
300
- metadata.atomics.find(
301
- type === "number"
302
- ? (atom: string) => atom === type || atom === "bigint"
303
- : (atom: string) => atom === type,
304
- ) !== undefined ||
305
- metadata.arrays.some((array) =>
306
- has_atomic(type)(visited)(array.value),
307
- ) ||
308
- metadata.tuples.some((tuple) =>
309
- tuple.elements.some(has_atomic(type)(visited)),
310
- ) ||
311
- metadata.aliases.some((alias) =>
312
- has_atomic(type)(visited)(alias.value),
313
- ) ||
314
- (metadata.resolved !== null &&
315
- has_atomic(type)(visited)(metadata.resolved.returns))
316
- );
317
- };
318
-
319
- const has_native =
320
- (type: string) =>
321
- (visited: Set<Metadata>) =>
322
- (metadata: Metadata): boolean => {
323
- if (visited.has(metadata)) return false;
324
- visited.add(metadata);
325
- return (
326
- metadata.natives.find((native) => native === type) !== undefined ||
327
- metadata.arrays.some((child) =>
328
- has_native(type)(visited)(child.value),
329
- ) ||
330
- metadata.tuples.some((tuple) =>
331
- tuple.elements.some(has_native(type)(visited)),
332
- ) ||
333
- metadata.aliases.some((alias) =>
334
- has_native(type)(visited)(alias.value),
335
- ) ||
336
- (metadata.resolved !== null &&
337
- has_native(type)(visited)(metadata.resolved.returns))
338
- );
339
- };
340
-
341
- const has_array =
342
- (visited: Set<Metadata>) =>
343
- (metadata: Metadata): boolean => {
344
- if (visited.has(metadata)) return false;
345
- visited.add(metadata);
346
- return (
347
- metadata.arrays.length !== 0 ||
348
- metadata.tuples.some((tuple) =>
349
- tuple.elements.some(has_array(visited)),
350
- ) ||
351
- metadata.aliases.some((alias) => has_array(visited)(alias.value)) ||
352
- (metadata.resolved !== null &&
353
- has_array(visited)(metadata.resolved.returns))
354
- );
355
- };
1
+ import ts from "typescript";
2
+
3
+ import { IMetadataTag } from "../metadata/IMetadataTag";
4
+ import { Metadata } from "../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.aliases.some((alias) =>
318
+ has_atomic(type)(visited)(alias.value),
319
+ ) ||
320
+ (metadata.resolved !== null &&
321
+ has_atomic(type)(visited)(metadata.resolved.returns))
322
+ );
323
+ };
324
+
325
+ const has_native =
326
+ (type: string) =>
327
+ (visited: Set<Metadata>) =>
328
+ (metadata: Metadata): boolean => {
329
+ if (visited.has(metadata)) return false;
330
+ visited.add(metadata);
331
+ return (
332
+ metadata.natives.find((native) => native === type) !== undefined ||
333
+ metadata.arrays.some((child) =>
334
+ has_native(type)(visited)(child.value),
335
+ ) ||
336
+ metadata.tuples.some((tuple) =>
337
+ tuple.elements.some(has_native(type)(visited)),
338
+ ) ||
339
+ metadata.aliases.some((alias) =>
340
+ has_native(type)(visited)(alias.value),
341
+ ) ||
342
+ (metadata.resolved !== null &&
343
+ has_native(type)(visited)(metadata.resolved.returns))
344
+ );
345
+ };
346
+
347
+ const has_array =
348
+ (visited: Set<Metadata>) =>
349
+ (metadata: Metadata): boolean => {
350
+ if (visited.has(metadata)) return false;
351
+ visited.add(metadata);
352
+ return (
353
+ metadata.arrays.length !== 0 ||
354
+ metadata.tuples.some((tuple) =>
355
+ tuple.elements.some(has_array(visited)),
356
+ ) ||
357
+ metadata.aliases.some((alias) => has_array(visited)(alias.value)) ||
358
+ (metadata.resolved !== null &&
359
+ has_array(visited)(metadata.resolved.returns))
360
+ );
361
+ };