typia 4.3.2 → 4.3.3

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 (241) hide show
  1. package/lib/factories/internal/metadata/iterate_metadata_intersection.js +2 -0
  2. package/lib/factories/internal/metadata/iterate_metadata_intersection.js.map +1 -1
  3. package/lib/programmers/helpers/CloneJoiner.js +10 -2
  4. package/lib/programmers/helpers/CloneJoiner.js.map +1 -1
  5. package/lib/programmers/internal/check_dynamic_properties.js +1 -1
  6. package/lib/programmers/internal/check_dynamic_properties.js.map +1 -1
  7. package/package.json +2 -2
  8. package/src/CustomValidatorMap.ts +126 -126
  9. package/src/IRandomGenerator.ts +34 -34
  10. package/src/IValidation.ts +21 -21
  11. package/src/Primitive.ts +131 -131
  12. package/src/TypeGuardError.ts +36 -36
  13. package/src/executable/TypiaGenerateWizard.ts +85 -85
  14. package/src/executable/TypiaSetupWizard.ts +153 -153
  15. package/src/executable/setup/ArgumentParser.ts +45 -45
  16. package/src/executable/setup/CommandExecutor.ts +8 -8
  17. package/src/executable/setup/FileRetriever.ts +22 -22
  18. package/src/executable/setup/PackageManager.ts +71 -71
  19. package/src/executable/setup/PluginConfigurator.ts +70 -70
  20. package/src/executable/typia.ts +52 -52
  21. package/src/factories/CommentFactory.ts +84 -84
  22. package/src/factories/ExpressionFactory.ts +77 -77
  23. package/src/factories/IdentifierFactory.ts +59 -59
  24. package/src/factories/LiteralFactory.ts +39 -39
  25. package/src/factories/MetadataCollection.ts +269 -269
  26. package/src/factories/MetadataFactory.ts +34 -34
  27. package/src/factories/MetadataTagFactory.ts +361 -361
  28. package/src/factories/StatementFactory.ts +24 -24
  29. package/src/factories/TemplateFactory.ts +58 -58
  30. package/src/factories/TypeFactory.ts +124 -124
  31. package/src/factories/ValueFactory.ts +12 -12
  32. package/src/factories/internal/metadata/MetadataHelper.ts +12 -12
  33. package/src/factories/internal/metadata/emend_metadata_atomics.ts +33 -33
  34. package/src/factories/internal/metadata/emplace_metadata_alias.ts +40 -40
  35. package/src/factories/internal/metadata/emplace_metadata_array.ts +34 -34
  36. package/src/factories/internal/metadata/emplace_metadata_object.ts +136 -136
  37. package/src/factories/internal/metadata/emplace_metadata_tuple.ts +50 -50
  38. package/src/factories/internal/metadata/explore_metadata.ts +38 -38
  39. package/src/factories/internal/metadata/iterate_metadata.ts +81 -81
  40. package/src/factories/internal/metadata/iterate_metadata_alias.ts +30 -30
  41. package/src/factories/internal/metadata/iterate_metadata_array.ts +24 -24
  42. package/src/factories/internal/metadata/iterate_metadata_atomic.ts +59 -59
  43. package/src/factories/internal/metadata/iterate_metadata_coalesce.ts +33 -33
  44. package/src/factories/internal/metadata/iterate_metadata_collection.ts +133 -133
  45. package/src/factories/internal/metadata/iterate_metadata_constant.ts +58 -58
  46. package/src/factories/internal/metadata/iterate_metadata_intersection.ts +88 -84
  47. package/src/factories/internal/metadata/iterate_metadata_map.ts +41 -41
  48. package/src/factories/internal/metadata/iterate_metadata_native.ts +219 -219
  49. package/src/factories/internal/metadata/iterate_metadata_object.ts +43 -43
  50. package/src/factories/internal/metadata/iterate_metadata_resolve.ts +49 -49
  51. package/src/factories/internal/metadata/iterate_metadata_set.ts +33 -33
  52. package/src/factories/internal/metadata/iterate_metadata_sort.ts +69 -69
  53. package/src/factories/internal/metadata/iterate_metadata_tag.ts +31 -31
  54. package/src/factories/internal/metadata/iterate_metadata_template.ts +38 -38
  55. package/src/factories/internal/metadata/iterate_metadata_tuple.ts +24 -24
  56. package/src/factories/internal/metadata/iterate_metadata_union.ts +24 -24
  57. package/src/functional/$any.ts +2 -2
  58. package/src/functional/$dictionary.ts +25 -25
  59. package/src/functional/$every.ts +11 -11
  60. package/src/functional/$guard.ts +35 -35
  61. package/src/functional/$is_between.ts +2 -2
  62. package/src/functional/$is_custom.ts +14 -14
  63. package/src/functional/$is_date.ts +3 -3
  64. package/src/functional/$is_datetime.ts +2 -2
  65. package/src/functional/$is_email.ts +4 -4
  66. package/src/functional/$is_ipv4.ts +4 -4
  67. package/src/functional/$is_ipv6.ts +4 -4
  68. package/src/functional/$is_url.ts +4 -4
  69. package/src/functional/$is_uuid.ts +4 -4
  70. package/src/functional/$join.ts +46 -46
  71. package/src/functional/$number.ts +12 -12
  72. package/src/functional/$report.ts +15 -15
  73. package/src/functional/$rest.ts +3 -3
  74. package/src/functional/$string.ts +50 -50
  75. package/src/functional/$tail.ts +5 -5
  76. package/src/functional/Namespace.ts +127 -127
  77. package/src/index.ts +4 -4
  78. package/src/metadata/ICommentTag.ts +4 -4
  79. package/src/metadata/IJsDocTagInfo.ts +10 -10
  80. package/src/metadata/IMetadata.ts +28 -28
  81. package/src/metadata/IMetadataAlias.ts +14 -14
  82. package/src/metadata/IMetadataApplication.ts +7 -7
  83. package/src/metadata/IMetadataArray.ts +10 -10
  84. package/src/metadata/IMetadataCollection.ts +11 -11
  85. package/src/metadata/IMetadataConstant.ts +16 -16
  86. package/src/metadata/IMetadataDictionary.ts +14 -14
  87. package/src/metadata/IMetadataEntry.ts +6 -6
  88. package/src/metadata/IMetadataObject.ts +18 -18
  89. package/src/metadata/IMetadataProperty.ts +11 -11
  90. package/src/metadata/IMetadataResolved.ts +6 -6
  91. package/src/metadata/IMetadataTag.ts +112 -112
  92. package/src/metadata/IMetadataTuple.ts +10 -10
  93. package/src/metadata/Metadata.ts +607 -607
  94. package/src/metadata/MetadataAlias.ts +66 -66
  95. package/src/metadata/MetadataArray.ts +55 -55
  96. package/src/metadata/MetadataConstant.ts +3 -3
  97. package/src/metadata/MetadataObject.ts +129 -129
  98. package/src/metadata/MetadataProperty.ts +64 -64
  99. package/src/metadata/MetadataResolved.ts +51 -51
  100. package/src/metadata/MetadataTuple.ts +58 -58
  101. package/src/module.ts +2038 -2038
  102. package/src/programmers/ApplicationProgrammer.ts +47 -47
  103. package/src/programmers/AssertCloneProgrammer.ts +71 -71
  104. package/src/programmers/AssertParseProgrammer.ts +66 -66
  105. package/src/programmers/AssertProgrammer.ts +279 -279
  106. package/src/programmers/AssertPruneProgrammer.ts +68 -68
  107. package/src/programmers/AssertStringifyProgrammer.ts +66 -66
  108. package/src/programmers/CheckerProgrammer.ts +1173 -1173
  109. package/src/programmers/CloneProgrammer.ts +587 -587
  110. package/src/programmers/FeatureProgrammer.ts +495 -495
  111. package/src/programmers/IsCloneProgrammer.ts +78 -78
  112. package/src/programmers/IsParseProgrammer.ts +72 -72
  113. package/src/programmers/IsProgrammer.ts +239 -239
  114. package/src/programmers/IsPruneProgrammer.ts +73 -73
  115. package/src/programmers/IsStringifyProgrammer.ts +76 -76
  116. package/src/programmers/LiteralsProgrammer.ts +64 -64
  117. package/src/programmers/PruneProgrammer.ts +542 -542
  118. package/src/programmers/RandomProgrammer.ts +578 -578
  119. package/src/programmers/StringifyProgrammer.ts +986 -986
  120. package/src/programmers/TypiaProgrammer.ts +129 -129
  121. package/src/programmers/ValidateCloneProgrammer.ts +85 -85
  122. package/src/programmers/ValidateParseProgrammer.ts +70 -70
  123. package/src/programmers/ValidateProgrammer.ts +305 -305
  124. package/src/programmers/ValidatePruneProgrammer.ts +78 -78
  125. package/src/programmers/ValidateStringifyProgrammer.ts +84 -84
  126. package/src/programmers/helpers/AtomicPredicator.ts +31 -31
  127. package/src/programmers/helpers/CloneJoiner.ts +168 -131
  128. package/src/programmers/helpers/FunctionImporeter.ts +78 -78
  129. package/src/programmers/helpers/ICheckEntry.ts +12 -12
  130. package/src/programmers/helpers/IExpressionEntry.ts +12 -12
  131. package/src/programmers/helpers/OptionPredicator.ts +15 -15
  132. package/src/programmers/helpers/PruneJoiner.ts +143 -143
  133. package/src/programmers/helpers/RandomJoiner.ts +173 -173
  134. package/src/programmers/helpers/RandomRanger.ts +208 -208
  135. package/src/programmers/helpers/StringifyJoinder.ts +113 -113
  136. package/src/programmers/helpers/StringifyPredicator.ts +13 -13
  137. package/src/programmers/helpers/UnionExplorer.ts +305 -305
  138. package/src/programmers/helpers/UnionPredicator.ts +81 -81
  139. package/src/programmers/helpers/disable_function_importer_declare.ts +26 -26
  140. package/src/programmers/internal/JSON_SCHEMA_PREFIX.ts +1 -1
  141. package/src/programmers/internal/application_alias.ts +66 -66
  142. package/src/programmers/internal/application_array.ts +30 -30
  143. package/src/programmers/internal/application_boolean.ts +15 -15
  144. package/src/programmers/internal/application_constant.ts +26 -26
  145. package/src/programmers/internal/application_default.ts +17 -17
  146. package/src/programmers/internal/application_default_string.ts +33 -33
  147. package/src/programmers/internal/application_native.ts +39 -39
  148. package/src/programmers/internal/application_number.ts +80 -80
  149. package/src/programmers/internal/application_object.ts +165 -165
  150. package/src/programmers/internal/application_resolved.ts +55 -55
  151. package/src/programmers/internal/application_schema.ts +157 -157
  152. package/src/programmers/internal/application_string.ts +44 -44
  153. package/src/programmers/internal/application_templates.ts +25 -25
  154. package/src/programmers/internal/application_tuple.ts +57 -57
  155. package/src/programmers/internal/check_array.ts +30 -30
  156. package/src/programmers/internal/check_array_length.ts +35 -35
  157. package/src/programmers/internal/check_bigint.ts +110 -110
  158. package/src/programmers/internal/check_custom.ts +31 -31
  159. package/src/programmers/internal/check_dynamic_properties.ts +194 -195
  160. package/src/programmers/internal/check_everything.ts +28 -28
  161. package/src/programmers/internal/check_native.ts +21 -21
  162. package/src/programmers/internal/check_object.ts +55 -55
  163. package/src/programmers/internal/check_string.ts +25 -25
  164. package/src/programmers/internal/check_string_tags.ts +67 -67
  165. package/src/programmers/internal/check_template.ts +56 -56
  166. package/src/programmers/internal/check_union_array_like.ts +329 -329
  167. package/src/programmers/internal/decode_union_object.ts +73 -73
  168. package/src/programmers/internal/feature_object_entries.ts +63 -63
  169. package/src/programmers/internal/get_comment_tags.ts +23 -23
  170. package/src/programmers/internal/metadata_to_pattern.ts +34 -34
  171. package/src/programmers/internal/prune_object_properties.ts +60 -60
  172. package/src/programmers/internal/random_custom.ts +29 -29
  173. package/src/programmers/internal/stringify_dynamic_properties.ts +171 -171
  174. package/src/programmers/internal/stringify_native.ts +7 -7
  175. package/src/programmers/internal/stringify_regular_properties.ts +83 -83
  176. package/src/programmers/internal/template_to_pattern.ts +15 -15
  177. package/src/programmers/internal/wrap_metadata_rest_tuple.ts +16 -16
  178. package/src/schemas/IJsonApplication.ts +8 -8
  179. package/src/schemas/IJsonComponents.ts +33 -33
  180. package/src/schemas/IJsonSchema.ts +133 -133
  181. package/src/transform.ts +27 -27
  182. package/src/transformers/CallExpressionTransformer.ts +179 -179
  183. package/src/transformers/FileTransformer.ts +47 -47
  184. package/src/transformers/IProject.ts +11 -11
  185. package/src/transformers/ITransformOptions.ts +62 -62
  186. package/src/transformers/ImportTransformer.ts +66 -66
  187. package/src/transformers/NodeTransformer.ts +13 -13
  188. package/src/transformers/features/miscellaneous/ApplicationTransformer.ts +112 -112
  189. package/src/transformers/features/miscellaneous/AssertCloneTransformer.ts +9 -9
  190. package/src/transformers/features/miscellaneous/AssertPruneTransformer.ts +9 -9
  191. package/src/transformers/features/miscellaneous/CloneTransformer.ts +9 -9
  192. package/src/transformers/features/miscellaneous/CreateAssertCloneTransformer.ts +9 -9
  193. package/src/transformers/features/miscellaneous/CreateAssertPruneTransformer.ts +9 -9
  194. package/src/transformers/features/miscellaneous/CreateCloneTransformer.ts +9 -9
  195. package/src/transformers/features/miscellaneous/CreateIsCloneTransformer.ts +9 -9
  196. package/src/transformers/features/miscellaneous/CreateIsPruneTransformer.ts +9 -9
  197. package/src/transformers/features/miscellaneous/CreatePruneTransformer.ts +9 -9
  198. package/src/transformers/features/miscellaneous/CreateRandomTransformer.ts +39 -39
  199. package/src/transformers/features/miscellaneous/CreateValidateCloneTransformer.ts +9 -9
  200. package/src/transformers/features/miscellaneous/CreateValidatePruneTransformer.ts +9 -9
  201. package/src/transformers/features/miscellaneous/IsCloneTransformer.ts +9 -9
  202. package/src/transformers/features/miscellaneous/IsPruneTransformer.ts +9 -9
  203. package/src/transformers/features/miscellaneous/LiteralsTransformer.ts +28 -28
  204. package/src/transformers/features/miscellaneous/MetadataTransformer.ts +53 -53
  205. package/src/transformers/features/miscellaneous/PruneTransformer.ts +9 -9
  206. package/src/transformers/features/miscellaneous/RandomTransformer.ts +42 -42
  207. package/src/transformers/features/miscellaneous/ValidateCloneTransformer.ts +9 -9
  208. package/src/transformers/features/miscellaneous/ValidatePruneTransformer.ts +9 -9
  209. package/src/transformers/features/parsers/AssertParseTransformer.ts +9 -9
  210. package/src/transformers/features/parsers/CreateAssertParseTransformer.ts +9 -9
  211. package/src/transformers/features/parsers/CreateIsParseTransformer.ts +9 -9
  212. package/src/transformers/features/parsers/CreateValidateParseTransformer.ts +9 -9
  213. package/src/transformers/features/parsers/IsParseTransformer.ts +9 -9
  214. package/src/transformers/features/parsers/ValidateParseTransformer.ts +9 -9
  215. package/src/transformers/features/stringifiers/AssertStringifyTransformer.ts +10 -10
  216. package/src/transformers/features/stringifiers/CreateAssertStringifyTransformer.ts +12 -12
  217. package/src/transformers/features/stringifiers/CreateIsStringifyTransformer.ts +9 -9
  218. package/src/transformers/features/stringifiers/CreateStringifyTransformer.ts +9 -9
  219. package/src/transformers/features/stringifiers/CreateValidateStringifyProgrammer.ts +12 -12
  220. package/src/transformers/features/stringifiers/IsStringifyTransformer.ts +9 -9
  221. package/src/transformers/features/stringifiers/StringifyTransformer.ts +9 -9
  222. package/src/transformers/features/stringifiers/ValidateStringifyTransformer.ts +10 -10
  223. package/src/transformers/features/validators/AssertTransformer.ts +11 -11
  224. package/src/transformers/features/validators/CreateAssertTransformer.ts +13 -13
  225. package/src/transformers/features/validators/CreateIsTransformer.ts +11 -11
  226. package/src/transformers/features/validators/CreateValidateTransformer.ts +13 -13
  227. package/src/transformers/features/validators/IsTransformer.ts +11 -11
  228. package/src/transformers/features/validators/ValidateTransformer.ts +11 -11
  229. package/src/transformers/internal/GenericTransformer.ts +97 -97
  230. package/src/typings/Atomic.ts +17 -17
  231. package/src/typings/ClassProperties.ts +5 -5
  232. package/src/typings/Customizable.ts +5 -5
  233. package/src/typings/OmitNever.ts +3 -3
  234. package/src/typings/SpecialFields.ts +3 -3
  235. package/src/typings/Writable.ts +11 -11
  236. package/src/utils/ArrayUtil.ts +45 -45
  237. package/src/utils/Escaper.ts +46 -46
  238. package/src/utils/MapUtil.ts +12 -12
  239. package/src/utils/PatternUtil.ts +33 -33
  240. package/src/utils/RandomGenerator.ts +81 -81
  241. package/src/utils/Singleton.ts +17 -17
@@ -1,542 +1,542 @@
1
- import ts from "typescript";
2
-
3
- import { ExpressionFactory } from "../factories/ExpressionFactory";
4
- import { IdentifierFactory } from "../factories/IdentifierFactory";
5
- import { MetadataCollection } from "../factories/MetadataCollection";
6
- import { MetadataFactory } from "../factories/MetadataFactory";
7
- import { StatementFactory } from "../factories/StatementFactory";
8
- import { TypeFactory } from "../factories/TypeFactory";
9
-
10
- import { IJsDocTagInfo } from "../metadata/IJsDocTagInfo";
11
- import { IMetadataTag } from "../metadata/IMetadataTag";
12
- import { Metadata } from "../metadata/Metadata";
13
- import { MetadataArray } from "../metadata/MetadataArray";
14
- import { MetadataTuple } from "../metadata/MetadataTuple";
15
-
16
- import { IProject } from "../transformers/IProject";
17
-
18
- import { FeatureProgrammer } from "./FeatureProgrammer";
19
- import { IsProgrammer } from "./IsProgrammer";
20
- import { FunctionImporter } from "./helpers/FunctionImporeter";
21
- import { PruneJoiner } from "./helpers/PruneJoiner";
22
- import { UnionExplorer } from "./helpers/UnionExplorer";
23
- import { decode_union_object } from "./internal/decode_union_object";
24
- import { wrap_metadata_rest_tuple } from "./internal/wrap_metadata_rest_tuple";
25
-
26
- export namespace PruneProgrammer {
27
- export const write =
28
- (project: IProject) => (modulo: ts.LeftHandSideExpression) => {
29
- const importer: FunctionImporter = new FunctionImporter();
30
- return FeatureProgrammer.write(project)({
31
- ...configure(project)(importer),
32
- addition: (collection) => [
33
- ...IsProgrammer.write_function_statements(project)(
34
- importer,
35
- )(collection),
36
- ...importer.declare(modulo),
37
- ],
38
- })(importer);
39
- };
40
-
41
- const write_array_functions =
42
- (config: FeatureProgrammer.IConfig) =>
43
- (importer: FunctionImporter) =>
44
- (collection: MetadataCollection): ts.VariableStatement[] =>
45
- collection
46
- .arrays()
47
- .filter((a) => a.recursive)
48
- .map((array, i) =>
49
- StatementFactory.constant(
50
- `${config.prefix}a${i}`,
51
- ts.factory.createArrowFunction(
52
- undefined,
53
- undefined,
54
- FeatureProgrammer.parameterDeclarations(config)(
55
- TypeFactory.keyword("any"),
56
- )(ts.factory.createIdentifier("input")),
57
- TypeFactory.keyword("any"),
58
- undefined,
59
- decode_array_inline(config)(importer)(
60
- ts.factory.createIdentifier("input"),
61
- array,
62
- {
63
- tracable: config.trace,
64
- source: "function",
65
- from: "array",
66
- postfix: "",
67
- },
68
- ),
69
- ),
70
- ),
71
- );
72
-
73
- const write_tuple_functions =
74
- (project: IProject) =>
75
- (config: FeatureProgrammer.IConfig) =>
76
- (importer: FunctionImporter) =>
77
- (collection: MetadataCollection): ts.VariableStatement[] =>
78
- collection
79
- .tuples()
80
- .filter((t) => t.recursive)
81
- .map((tuple, i) =>
82
- StatementFactory.constant(
83
- `${config.prefix}t${i}`,
84
- ts.factory.createArrowFunction(
85
- undefined,
86
- undefined,
87
- FeatureProgrammer.parameterDeclarations(config)(
88
- TypeFactory.keyword("any"),
89
- )(ts.factory.createIdentifier("input")),
90
- TypeFactory.keyword("any"),
91
- undefined,
92
- decode_tuple_inline(project)(config)(importer)(
93
- ts.factory.createIdentifier("input"),
94
- tuple,
95
- {
96
- tracable: config.trace,
97
- source: "function",
98
- from: "array",
99
- postfix: "",
100
- },
101
- ),
102
- ),
103
- ),
104
- );
105
-
106
- /* -----------------------------------------------------------
107
- DECODERS
108
- ----------------------------------------------------------- */
109
- const decode =
110
- (project: IProject) =>
111
- (config: FeatureProgrammer.IConfig) =>
112
- (importer: FunctionImporter) =>
113
- (
114
- input: ts.Expression,
115
- meta: Metadata,
116
- explore: FeatureProgrammer.IExplore,
117
- ): ts.ConciseBody => {
118
- if (filter(meta) === false) return ts.factory.createBlock([]);
119
-
120
- interface IUnion {
121
- type: string;
122
- is: () => ts.Expression;
123
- value: () => ts.Expression | ts.Block | ts.ReturnStatement;
124
- }
125
- const unions: IUnion[] = [];
126
-
127
- //----
128
- // LIST UP UNION TYPES
129
- //----
130
- // TUPLES
131
- for (const tuple of meta.tuples.filter((t) =>
132
- t.elements.some((e) => filter(e.rest ?? e)),
133
- ))
134
- unions.push({
135
- type: "tuple",
136
- is: () =>
137
- IsProgrammer.decode(project)(importer)(
138
- input,
139
- (() => {
140
- const partial = Metadata.initialize();
141
- partial.tuples.push(tuple);
142
- return partial;
143
- })(),
144
- explore,
145
- [],
146
- [],
147
- ),
148
- value: () =>
149
- decode_tuple(project)(config)(importer)(
150
- input,
151
- tuple,
152
- explore,
153
- ),
154
- });
155
-
156
- // ARRAYS
157
- if (meta.arrays.filter((a) => filter(a.value)).length)
158
- unions.push({
159
- type: "array",
160
- is: () => ExpressionFactory.isArray(input),
161
- value: () =>
162
- explore_arrays(project)(config)(importer)(
163
- input,
164
- meta.arrays,
165
- {
166
- ...explore,
167
- from: "array",
168
- },
169
- ),
170
- });
171
-
172
- // BUILT-IN CLASSES
173
- if (meta.natives.length)
174
- for (const native of meta.natives)
175
- unions.push({
176
- type: "native",
177
- is: () => ExpressionFactory.isInstanceOf(native)(input),
178
- value: () => ts.factory.createReturnStatement(),
179
- });
180
- if (meta.sets.length)
181
- unions.push({
182
- type: "set",
183
- is: () => ExpressionFactory.isInstanceOf("Set")(input),
184
- value: () => ts.factory.createReturnStatement(),
185
- });
186
- if (meta.maps.length)
187
- unions.push({
188
- type: "map",
189
- is: () => ExpressionFactory.isInstanceOf("Map")(input),
190
- value: () => ts.factory.createReturnStatement(),
191
- });
192
-
193
- // OBJECTS
194
- if (meta.objects.length)
195
- unions.push({
196
- type: "object",
197
- is: () =>
198
- ExpressionFactory.isObject({
199
- checkNull: true,
200
- checkArray: false,
201
- })(input),
202
- value: () =>
203
- explore_objects(config)(importer)(input, meta, {
204
- ...explore,
205
- from: "object",
206
- }),
207
- });
208
-
209
- //----
210
- // STATEMENTS
211
- //----
212
- const converter = (
213
- v: ts.Expression | ts.Block | ts.ReturnStatement,
214
- ) =>
215
- ts.isReturnStatement(v) || ts.isBlock(v)
216
- ? v
217
- : ts.factory.createExpressionStatement(v);
218
-
219
- const statements: ts.Statement[] = unions.map((u) =>
220
- ts.factory.createIfStatement(u.is(), converter(u.value())),
221
- );
222
- return ts.factory.createBlock(statements, true);
223
- };
224
-
225
- const decode_object = (importer: FunctionImporter) =>
226
- FeatureProgrammer.decode_object({
227
- trace: false,
228
- path: false,
229
- prefix: PREFIX,
230
- })(importer);
231
-
232
- const decode_array =
233
- (config: FeatureProgrammer.IConfig) =>
234
- (importer: FunctionImporter) =>
235
- (
236
- input: ts.Expression,
237
- array: MetadataArray,
238
- explore: FeatureProgrammer.IExplore,
239
- ) =>
240
- array.recursive
241
- ? ts.factory.createCallExpression(
242
- ts.factory.createIdentifier(
243
- importer.useLocal(`${config.prefix}a${array.index}`),
244
- ),
245
- undefined,
246
- FeatureProgrammer.argumentsArray(config)({
247
- ...explore,
248
- source: "function",
249
- from: "array",
250
- })(input),
251
- )
252
- : decode_array_inline(config)(importer)(input, array, explore);
253
-
254
- const decode_array_inline =
255
- (config: FeatureProgrammer.IConfig) =>
256
- (importer: FunctionImporter) =>
257
- (
258
- input: ts.Expression,
259
- array: MetadataArray,
260
- explore: FeatureProgrammer.IExplore,
261
- ): ts.Expression =>
262
- FeatureProgrammer.decode_array(config)(importer)(PruneJoiner.array)(
263
- input,
264
- array,
265
- explore,
266
- [],
267
- [],
268
- );
269
-
270
- const decode_tuple =
271
- (project: IProject) =>
272
- (config: FeatureProgrammer.IConfig) =>
273
- (importer: FunctionImporter) =>
274
- (
275
- input: ts.Expression,
276
- tuple: MetadataTuple,
277
- explore: FeatureProgrammer.IExplore,
278
- ): ts.Expression | ts.Block =>
279
- tuple.recursive
280
- ? ts.factory.createCallExpression(
281
- ts.factory.createIdentifier(
282
- importer.useLocal(`${config.prefix}t${tuple.index}`),
283
- ),
284
- undefined,
285
- FeatureProgrammer.argumentsArray(config)({
286
- ...explore,
287
- source: "function",
288
- })(input),
289
- )
290
- : decode_tuple_inline(project)(config)(importer)(
291
- input,
292
- tuple,
293
- explore,
294
- );
295
-
296
- const decode_tuple_inline =
297
- (project: IProject) =>
298
- (config: FeatureProgrammer.IConfig) =>
299
- (importer: FunctionImporter) =>
300
- (
301
- input: ts.Expression,
302
- tuple: MetadataTuple,
303
- explore: FeatureProgrammer.IExplore,
304
- ): ts.Block => {
305
- const children: ts.ConciseBody[] = tuple.elements
306
- .map((elem, index) => [elem, index] as const)
307
- .filter(([elem]) => filter(elem) && elem.rest === null)
308
- .map(([elem, index]) =>
309
- decode(project)(config)(importer)(
310
- ts.factory.createElementAccessExpression(input, index),
311
- elem,
312
- {
313
- ...explore,
314
- from: "array",
315
- postfix: explore.postfix.length
316
- ? `${explore.postfix.slice(0, -1)}[${index}]"`
317
- : `"[${index}]"`,
318
- },
319
- ),
320
- );
321
- const rest = (() => {
322
- if (tuple.elements.length === 0) return null;
323
-
324
- const last: Metadata = tuple.elements.at(-1)!;
325
- const rest: Metadata | null = last.rest;
326
- if (rest === null || filter(rest) === false) return null;
327
-
328
- return decode(project)(config)(importer)(
329
- ts.factory.createCallExpression(
330
- IdentifierFactory.access(input)("slice"),
331
- undefined,
332
- [
333
- ts.factory.createNumericLiteral(
334
- tuple.elements.length - 1,
335
- ),
336
- ],
337
- ),
338
- wrap_metadata_rest_tuple(tuple.elements.at(-1)!.rest!),
339
- {
340
- ...explore,
341
- start: tuple.elements.length - 1,
342
- },
343
- );
344
- })();
345
- return PruneJoiner.tuple(children, rest);
346
- };
347
-
348
- /* -----------------------------------------------------------
349
- UNION TYPE EXPLORERS
350
- ----------------------------------------------------------- */
351
- const explore_objects =
352
- (config: FeatureProgrammer.IConfig) =>
353
- (importer: FunctionImporter) =>
354
- (
355
- input: ts.Expression,
356
- meta: Metadata,
357
- explore: FeatureProgrammer.IExplore,
358
- ) => {
359
- if (meta.objects.length === 1)
360
- return decode_object(importer)(
361
- input,
362
- meta.objects[0]!,
363
- explore,
364
- );
365
-
366
- return ts.factory.createCallExpression(
367
- ts.factory.createIdentifier(
368
- importer.useLocal(`${PREFIX}u${meta.union_index!}`),
369
- ),
370
- undefined,
371
- FeatureProgrammer.argumentsArray(config)(explore)(input),
372
- );
373
- };
374
-
375
- const explore_arrays =
376
- (project: IProject) =>
377
- (config: FeatureProgrammer.IConfig) =>
378
- (importer: FunctionImporter) =>
379
- (
380
- input: ts.Expression,
381
- elements: MetadataArray[],
382
- explore: FeatureProgrammer.IExplore,
383
- ): ts.Expression =>
384
- explore_array_like_union_types(config)(importer)(
385
- UnionExplorer.array({
386
- checker: IsProgrammer.decode(project)(importer),
387
- decoder: decode_array(config)(importer),
388
- empty: ts.factory.createStringLiteral("[]"),
389
- success: ts.factory.createTrue(),
390
- failure: (input, expected) =>
391
- create_throw_error(importer)(expected)(input),
392
- }),
393
- )(input, elements, explore);
394
-
395
- const explore_array_like_union_types =
396
- (config: FeatureProgrammer.IConfig) =>
397
- (importer: FunctionImporter) =>
398
- <T extends MetadataArray | MetadataTuple>(
399
- factory: (
400
- parameters: ts.ParameterDeclaration[],
401
- ) => (
402
- input: ts.Expression,
403
- elements: T[],
404
- explore: FeatureProgrammer.IExplore,
405
- tags: IMetadataTag[],
406
- jsDocTags: IJsDocTagInfo[],
407
- ) => ts.ArrowFunction,
408
- ) =>
409
- (
410
- input: ts.Expression,
411
- elements: T[],
412
- explore: FeatureProgrammer.IExplore,
413
- ): ts.Expression => {
414
- const arrow =
415
- (parameters: ts.ParameterDeclaration[]) =>
416
- (explore: FeatureProgrammer.IExplore) =>
417
- (input: ts.Expression): ts.ArrowFunction =>
418
- factory(parameters)(input, elements, explore, [], []);
419
- if (elements.every((e) => e.recursive === false))
420
- ts.factory.createCallExpression(
421
- arrow([])(explore)(input),
422
- undefined,
423
- [],
424
- );
425
-
426
- explore = {
427
- ...explore,
428
- source: "function",
429
- from: "array",
430
- };
431
- return ts.factory.createCallExpression(
432
- ts.factory.createIdentifier(
433
- importer.emplaceUnion(
434
- config.prefix,
435
- elements.map((e) => e.name).join(" | "),
436
- () =>
437
- arrow(
438
- FeatureProgrammer.parameterDeclarations(config)(
439
- TypeFactory.keyword("any"),
440
- )(ts.factory.createIdentifier("input")),
441
- )({
442
- ...explore,
443
- postfix: "",
444
- })(ts.factory.createIdentifier("input")),
445
- ),
446
- ),
447
- undefined,
448
- FeatureProgrammer.argumentsArray(config)(explore)(input),
449
- );
450
- };
451
-
452
- // @todo -> must filter out recursive visit
453
- const filter = (meta: Metadata): boolean =>
454
- meta.any === false &&
455
- (meta.objects.length !== 0 ||
456
- meta.tuples.some((t) =>
457
- t.elements.some((e) => filter(e.rest ?? e)),
458
- ) ||
459
- meta.arrays.some((e) => filter(e.value)));
460
-
461
- /* -----------------------------------------------------------
462
- CONFIGURATIONS
463
- ----------------------------------------------------------- */
464
- const PREFIX = "$p";
465
-
466
- const configure =
467
- (project: IProject) =>
468
- (importer: FunctionImporter): FeatureProgrammer.IConfig => {
469
- const config: FeatureProgrammer.IConfig = {
470
- types: {
471
- input: (type, name) =>
472
- ts.factory.createTypeReferenceNode(
473
- name ??
474
- TypeFactory.getFullName(project.checker)(type),
475
- ),
476
- output: () => TypeFactory.keyword("void"),
477
- },
478
- prefix: PREFIX,
479
- trace: false,
480
- path: false,
481
- initializer,
482
- decoder: () => decode(project)(config)(importer),
483
- objector: {
484
- checker: () => IsProgrammer.decode(project)(importer),
485
- decoder: () => decode_object(importer),
486
- joiner: PruneJoiner.object,
487
- unionizer: decode_union_object(
488
- IsProgrammer.decode_object(importer),
489
- )(decode_object(importer))((exp) => exp)(
490
- (value, expected) =>
491
- create_throw_error(importer)(expected)(value),
492
- ),
493
- failure: (input, expected) =>
494
- create_throw_error(importer)(expected)(input),
495
- },
496
- generator: {
497
- arrays: () => write_array_functions(config)(importer),
498
- tuples: () =>
499
- write_tuple_functions(project)(config)(importer),
500
- },
501
- };
502
- return config;
503
- };
504
-
505
- const initializer: FeatureProgrammer.IConfig["initializer"] =
506
- ({ checker }) =>
507
- (type) => {
508
- const collection = new MetadataCollection();
509
- const meta = MetadataFactory.analyze(checker)({
510
- resolve: false,
511
- constant: true,
512
- absorb: true,
513
- })(collection)(type);
514
- return [collection, meta];
515
- };
516
-
517
- const create_throw_error =
518
- (importer: FunctionImporter) =>
519
- (expected: string) =>
520
- (value: ts.Expression) =>
521
- ts.factory.createExpressionStatement(
522
- ts.factory.createCallExpression(
523
- importer.use("throws"),
524
- [],
525
- [
526
- ts.factory.createObjectLiteralExpression(
527
- [
528
- ts.factory.createPropertyAssignment(
529
- "expected",
530
- ts.factory.createStringLiteral(expected),
531
- ),
532
- ts.factory.createPropertyAssignment(
533
- "value",
534
- value,
535
- ),
536
- ],
537
- true,
538
- ),
539
- ],
540
- ),
541
- );
542
- }
1
+ import ts from "typescript";
2
+
3
+ import { ExpressionFactory } from "../factories/ExpressionFactory";
4
+ import { IdentifierFactory } from "../factories/IdentifierFactory";
5
+ import { MetadataCollection } from "../factories/MetadataCollection";
6
+ import { MetadataFactory } from "../factories/MetadataFactory";
7
+ import { StatementFactory } from "../factories/StatementFactory";
8
+ import { TypeFactory } from "../factories/TypeFactory";
9
+
10
+ import { IJsDocTagInfo } from "../metadata/IJsDocTagInfo";
11
+ import { IMetadataTag } from "../metadata/IMetadataTag";
12
+ import { Metadata } from "../metadata/Metadata";
13
+ import { MetadataArray } from "../metadata/MetadataArray";
14
+ import { MetadataTuple } from "../metadata/MetadataTuple";
15
+
16
+ import { IProject } from "../transformers/IProject";
17
+
18
+ import { FeatureProgrammer } from "./FeatureProgrammer";
19
+ import { IsProgrammer } from "./IsProgrammer";
20
+ import { FunctionImporter } from "./helpers/FunctionImporeter";
21
+ import { PruneJoiner } from "./helpers/PruneJoiner";
22
+ import { UnionExplorer } from "./helpers/UnionExplorer";
23
+ import { decode_union_object } from "./internal/decode_union_object";
24
+ import { wrap_metadata_rest_tuple } from "./internal/wrap_metadata_rest_tuple";
25
+
26
+ export namespace PruneProgrammer {
27
+ export const write =
28
+ (project: IProject) => (modulo: ts.LeftHandSideExpression) => {
29
+ const importer: FunctionImporter = new FunctionImporter();
30
+ return FeatureProgrammer.write(project)({
31
+ ...configure(project)(importer),
32
+ addition: (collection) => [
33
+ ...IsProgrammer.write_function_statements(project)(
34
+ importer,
35
+ )(collection),
36
+ ...importer.declare(modulo),
37
+ ],
38
+ })(importer);
39
+ };
40
+
41
+ const write_array_functions =
42
+ (config: FeatureProgrammer.IConfig) =>
43
+ (importer: FunctionImporter) =>
44
+ (collection: MetadataCollection): ts.VariableStatement[] =>
45
+ collection
46
+ .arrays()
47
+ .filter((a) => a.recursive)
48
+ .map((array, i) =>
49
+ StatementFactory.constant(
50
+ `${config.prefix}a${i}`,
51
+ ts.factory.createArrowFunction(
52
+ undefined,
53
+ undefined,
54
+ FeatureProgrammer.parameterDeclarations(config)(
55
+ TypeFactory.keyword("any"),
56
+ )(ts.factory.createIdentifier("input")),
57
+ TypeFactory.keyword("any"),
58
+ undefined,
59
+ decode_array_inline(config)(importer)(
60
+ ts.factory.createIdentifier("input"),
61
+ array,
62
+ {
63
+ tracable: config.trace,
64
+ source: "function",
65
+ from: "array",
66
+ postfix: "",
67
+ },
68
+ ),
69
+ ),
70
+ ),
71
+ );
72
+
73
+ const write_tuple_functions =
74
+ (project: IProject) =>
75
+ (config: FeatureProgrammer.IConfig) =>
76
+ (importer: FunctionImporter) =>
77
+ (collection: MetadataCollection): ts.VariableStatement[] =>
78
+ collection
79
+ .tuples()
80
+ .filter((t) => t.recursive)
81
+ .map((tuple, i) =>
82
+ StatementFactory.constant(
83
+ `${config.prefix}t${i}`,
84
+ ts.factory.createArrowFunction(
85
+ undefined,
86
+ undefined,
87
+ FeatureProgrammer.parameterDeclarations(config)(
88
+ TypeFactory.keyword("any"),
89
+ )(ts.factory.createIdentifier("input")),
90
+ TypeFactory.keyword("any"),
91
+ undefined,
92
+ decode_tuple_inline(project)(config)(importer)(
93
+ ts.factory.createIdentifier("input"),
94
+ tuple,
95
+ {
96
+ tracable: config.trace,
97
+ source: "function",
98
+ from: "array",
99
+ postfix: "",
100
+ },
101
+ ),
102
+ ),
103
+ ),
104
+ );
105
+
106
+ /* -----------------------------------------------------------
107
+ DECODERS
108
+ ----------------------------------------------------------- */
109
+ const decode =
110
+ (project: IProject) =>
111
+ (config: FeatureProgrammer.IConfig) =>
112
+ (importer: FunctionImporter) =>
113
+ (
114
+ input: ts.Expression,
115
+ meta: Metadata,
116
+ explore: FeatureProgrammer.IExplore,
117
+ ): ts.ConciseBody => {
118
+ if (filter(meta) === false) return ts.factory.createBlock([]);
119
+
120
+ interface IUnion {
121
+ type: string;
122
+ is: () => ts.Expression;
123
+ value: () => ts.Expression | ts.Block | ts.ReturnStatement;
124
+ }
125
+ const unions: IUnion[] = [];
126
+
127
+ //----
128
+ // LIST UP UNION TYPES
129
+ //----
130
+ // TUPLES
131
+ for (const tuple of meta.tuples.filter((t) =>
132
+ t.elements.some((e) => filter(e.rest ?? e)),
133
+ ))
134
+ unions.push({
135
+ type: "tuple",
136
+ is: () =>
137
+ IsProgrammer.decode(project)(importer)(
138
+ input,
139
+ (() => {
140
+ const partial = Metadata.initialize();
141
+ partial.tuples.push(tuple);
142
+ return partial;
143
+ })(),
144
+ explore,
145
+ [],
146
+ [],
147
+ ),
148
+ value: () =>
149
+ decode_tuple(project)(config)(importer)(
150
+ input,
151
+ tuple,
152
+ explore,
153
+ ),
154
+ });
155
+
156
+ // ARRAYS
157
+ if (meta.arrays.filter((a) => filter(a.value)).length)
158
+ unions.push({
159
+ type: "array",
160
+ is: () => ExpressionFactory.isArray(input),
161
+ value: () =>
162
+ explore_arrays(project)(config)(importer)(
163
+ input,
164
+ meta.arrays,
165
+ {
166
+ ...explore,
167
+ from: "array",
168
+ },
169
+ ),
170
+ });
171
+
172
+ // BUILT-IN CLASSES
173
+ if (meta.natives.length)
174
+ for (const native of meta.natives)
175
+ unions.push({
176
+ type: "native",
177
+ is: () => ExpressionFactory.isInstanceOf(native)(input),
178
+ value: () => ts.factory.createReturnStatement(),
179
+ });
180
+ if (meta.sets.length)
181
+ unions.push({
182
+ type: "set",
183
+ is: () => ExpressionFactory.isInstanceOf("Set")(input),
184
+ value: () => ts.factory.createReturnStatement(),
185
+ });
186
+ if (meta.maps.length)
187
+ unions.push({
188
+ type: "map",
189
+ is: () => ExpressionFactory.isInstanceOf("Map")(input),
190
+ value: () => ts.factory.createReturnStatement(),
191
+ });
192
+
193
+ // OBJECTS
194
+ if (meta.objects.length)
195
+ unions.push({
196
+ type: "object",
197
+ is: () =>
198
+ ExpressionFactory.isObject({
199
+ checkNull: true,
200
+ checkArray: false,
201
+ })(input),
202
+ value: () =>
203
+ explore_objects(config)(importer)(input, meta, {
204
+ ...explore,
205
+ from: "object",
206
+ }),
207
+ });
208
+
209
+ //----
210
+ // STATEMENTS
211
+ //----
212
+ const converter = (
213
+ v: ts.Expression | ts.Block | ts.ReturnStatement,
214
+ ) =>
215
+ ts.isReturnStatement(v) || ts.isBlock(v)
216
+ ? v
217
+ : ts.factory.createExpressionStatement(v);
218
+
219
+ const statements: ts.Statement[] = unions.map((u) =>
220
+ ts.factory.createIfStatement(u.is(), converter(u.value())),
221
+ );
222
+ return ts.factory.createBlock(statements, true);
223
+ };
224
+
225
+ const decode_object = (importer: FunctionImporter) =>
226
+ FeatureProgrammer.decode_object({
227
+ trace: false,
228
+ path: false,
229
+ prefix: PREFIX,
230
+ })(importer);
231
+
232
+ const decode_array =
233
+ (config: FeatureProgrammer.IConfig) =>
234
+ (importer: FunctionImporter) =>
235
+ (
236
+ input: ts.Expression,
237
+ array: MetadataArray,
238
+ explore: FeatureProgrammer.IExplore,
239
+ ) =>
240
+ array.recursive
241
+ ? ts.factory.createCallExpression(
242
+ ts.factory.createIdentifier(
243
+ importer.useLocal(`${config.prefix}a${array.index}`),
244
+ ),
245
+ undefined,
246
+ FeatureProgrammer.argumentsArray(config)({
247
+ ...explore,
248
+ source: "function",
249
+ from: "array",
250
+ })(input),
251
+ )
252
+ : decode_array_inline(config)(importer)(input, array, explore);
253
+
254
+ const decode_array_inline =
255
+ (config: FeatureProgrammer.IConfig) =>
256
+ (importer: FunctionImporter) =>
257
+ (
258
+ input: ts.Expression,
259
+ array: MetadataArray,
260
+ explore: FeatureProgrammer.IExplore,
261
+ ): ts.Expression =>
262
+ FeatureProgrammer.decode_array(config)(importer)(PruneJoiner.array)(
263
+ input,
264
+ array,
265
+ explore,
266
+ [],
267
+ [],
268
+ );
269
+
270
+ const decode_tuple =
271
+ (project: IProject) =>
272
+ (config: FeatureProgrammer.IConfig) =>
273
+ (importer: FunctionImporter) =>
274
+ (
275
+ input: ts.Expression,
276
+ tuple: MetadataTuple,
277
+ explore: FeatureProgrammer.IExplore,
278
+ ): ts.Expression | ts.Block =>
279
+ tuple.recursive
280
+ ? ts.factory.createCallExpression(
281
+ ts.factory.createIdentifier(
282
+ importer.useLocal(`${config.prefix}t${tuple.index}`),
283
+ ),
284
+ undefined,
285
+ FeatureProgrammer.argumentsArray(config)({
286
+ ...explore,
287
+ source: "function",
288
+ })(input),
289
+ )
290
+ : decode_tuple_inline(project)(config)(importer)(
291
+ input,
292
+ tuple,
293
+ explore,
294
+ );
295
+
296
+ const decode_tuple_inline =
297
+ (project: IProject) =>
298
+ (config: FeatureProgrammer.IConfig) =>
299
+ (importer: FunctionImporter) =>
300
+ (
301
+ input: ts.Expression,
302
+ tuple: MetadataTuple,
303
+ explore: FeatureProgrammer.IExplore,
304
+ ): ts.Block => {
305
+ const children: ts.ConciseBody[] = tuple.elements
306
+ .map((elem, index) => [elem, index] as const)
307
+ .filter(([elem]) => filter(elem) && elem.rest === null)
308
+ .map(([elem, index]) =>
309
+ decode(project)(config)(importer)(
310
+ ts.factory.createElementAccessExpression(input, index),
311
+ elem,
312
+ {
313
+ ...explore,
314
+ from: "array",
315
+ postfix: explore.postfix.length
316
+ ? `${explore.postfix.slice(0, -1)}[${index}]"`
317
+ : `"[${index}]"`,
318
+ },
319
+ ),
320
+ );
321
+ const rest = (() => {
322
+ if (tuple.elements.length === 0) return null;
323
+
324
+ const last: Metadata = tuple.elements.at(-1)!;
325
+ const rest: Metadata | null = last.rest;
326
+ if (rest === null || filter(rest) === false) return null;
327
+
328
+ return decode(project)(config)(importer)(
329
+ ts.factory.createCallExpression(
330
+ IdentifierFactory.access(input)("slice"),
331
+ undefined,
332
+ [
333
+ ts.factory.createNumericLiteral(
334
+ tuple.elements.length - 1,
335
+ ),
336
+ ],
337
+ ),
338
+ wrap_metadata_rest_tuple(tuple.elements.at(-1)!.rest!),
339
+ {
340
+ ...explore,
341
+ start: tuple.elements.length - 1,
342
+ },
343
+ );
344
+ })();
345
+ return PruneJoiner.tuple(children, rest);
346
+ };
347
+
348
+ /* -----------------------------------------------------------
349
+ UNION TYPE EXPLORERS
350
+ ----------------------------------------------------------- */
351
+ const explore_objects =
352
+ (config: FeatureProgrammer.IConfig) =>
353
+ (importer: FunctionImporter) =>
354
+ (
355
+ input: ts.Expression,
356
+ meta: Metadata,
357
+ explore: FeatureProgrammer.IExplore,
358
+ ) => {
359
+ if (meta.objects.length === 1)
360
+ return decode_object(importer)(
361
+ input,
362
+ meta.objects[0]!,
363
+ explore,
364
+ );
365
+
366
+ return ts.factory.createCallExpression(
367
+ ts.factory.createIdentifier(
368
+ importer.useLocal(`${PREFIX}u${meta.union_index!}`),
369
+ ),
370
+ undefined,
371
+ FeatureProgrammer.argumentsArray(config)(explore)(input),
372
+ );
373
+ };
374
+
375
+ const explore_arrays =
376
+ (project: IProject) =>
377
+ (config: FeatureProgrammer.IConfig) =>
378
+ (importer: FunctionImporter) =>
379
+ (
380
+ input: ts.Expression,
381
+ elements: MetadataArray[],
382
+ explore: FeatureProgrammer.IExplore,
383
+ ): ts.Expression =>
384
+ explore_array_like_union_types(config)(importer)(
385
+ UnionExplorer.array({
386
+ checker: IsProgrammer.decode(project)(importer),
387
+ decoder: decode_array(config)(importer),
388
+ empty: ts.factory.createStringLiteral("[]"),
389
+ success: ts.factory.createTrue(),
390
+ failure: (input, expected) =>
391
+ create_throw_error(importer)(expected)(input),
392
+ }),
393
+ )(input, elements, explore);
394
+
395
+ const explore_array_like_union_types =
396
+ (config: FeatureProgrammer.IConfig) =>
397
+ (importer: FunctionImporter) =>
398
+ <T extends MetadataArray | MetadataTuple>(
399
+ factory: (
400
+ parameters: ts.ParameterDeclaration[],
401
+ ) => (
402
+ input: ts.Expression,
403
+ elements: T[],
404
+ explore: FeatureProgrammer.IExplore,
405
+ tags: IMetadataTag[],
406
+ jsDocTags: IJsDocTagInfo[],
407
+ ) => ts.ArrowFunction,
408
+ ) =>
409
+ (
410
+ input: ts.Expression,
411
+ elements: T[],
412
+ explore: FeatureProgrammer.IExplore,
413
+ ): ts.Expression => {
414
+ const arrow =
415
+ (parameters: ts.ParameterDeclaration[]) =>
416
+ (explore: FeatureProgrammer.IExplore) =>
417
+ (input: ts.Expression): ts.ArrowFunction =>
418
+ factory(parameters)(input, elements, explore, [], []);
419
+ if (elements.every((e) => e.recursive === false))
420
+ ts.factory.createCallExpression(
421
+ arrow([])(explore)(input),
422
+ undefined,
423
+ [],
424
+ );
425
+
426
+ explore = {
427
+ ...explore,
428
+ source: "function",
429
+ from: "array",
430
+ };
431
+ return ts.factory.createCallExpression(
432
+ ts.factory.createIdentifier(
433
+ importer.emplaceUnion(
434
+ config.prefix,
435
+ elements.map((e) => e.name).join(" | "),
436
+ () =>
437
+ arrow(
438
+ FeatureProgrammer.parameterDeclarations(config)(
439
+ TypeFactory.keyword("any"),
440
+ )(ts.factory.createIdentifier("input")),
441
+ )({
442
+ ...explore,
443
+ postfix: "",
444
+ })(ts.factory.createIdentifier("input")),
445
+ ),
446
+ ),
447
+ undefined,
448
+ FeatureProgrammer.argumentsArray(config)(explore)(input),
449
+ );
450
+ };
451
+
452
+ // @todo -> must filter out recursive visit
453
+ const filter = (meta: Metadata): boolean =>
454
+ meta.any === false &&
455
+ (meta.objects.length !== 0 ||
456
+ meta.tuples.some((t) =>
457
+ t.elements.some((e) => filter(e.rest ?? e)),
458
+ ) ||
459
+ meta.arrays.some((e) => filter(e.value)));
460
+
461
+ /* -----------------------------------------------------------
462
+ CONFIGURATIONS
463
+ ----------------------------------------------------------- */
464
+ const PREFIX = "$p";
465
+
466
+ const configure =
467
+ (project: IProject) =>
468
+ (importer: FunctionImporter): FeatureProgrammer.IConfig => {
469
+ const config: FeatureProgrammer.IConfig = {
470
+ types: {
471
+ input: (type, name) =>
472
+ ts.factory.createTypeReferenceNode(
473
+ name ??
474
+ TypeFactory.getFullName(project.checker)(type),
475
+ ),
476
+ output: () => TypeFactory.keyword("void"),
477
+ },
478
+ prefix: PREFIX,
479
+ trace: false,
480
+ path: false,
481
+ initializer,
482
+ decoder: () => decode(project)(config)(importer),
483
+ objector: {
484
+ checker: () => IsProgrammer.decode(project)(importer),
485
+ decoder: () => decode_object(importer),
486
+ joiner: PruneJoiner.object,
487
+ unionizer: decode_union_object(
488
+ IsProgrammer.decode_object(importer),
489
+ )(decode_object(importer))((exp) => exp)(
490
+ (value, expected) =>
491
+ create_throw_error(importer)(expected)(value),
492
+ ),
493
+ failure: (input, expected) =>
494
+ create_throw_error(importer)(expected)(input),
495
+ },
496
+ generator: {
497
+ arrays: () => write_array_functions(config)(importer),
498
+ tuples: () =>
499
+ write_tuple_functions(project)(config)(importer),
500
+ },
501
+ };
502
+ return config;
503
+ };
504
+
505
+ const initializer: FeatureProgrammer.IConfig["initializer"] =
506
+ ({ checker }) =>
507
+ (type) => {
508
+ const collection = new MetadataCollection();
509
+ const meta = MetadataFactory.analyze(checker)({
510
+ resolve: false,
511
+ constant: true,
512
+ absorb: true,
513
+ })(collection)(type);
514
+ return [collection, meta];
515
+ };
516
+
517
+ const create_throw_error =
518
+ (importer: FunctionImporter) =>
519
+ (expected: string) =>
520
+ (value: ts.Expression) =>
521
+ ts.factory.createExpressionStatement(
522
+ ts.factory.createCallExpression(
523
+ importer.use("throws"),
524
+ [],
525
+ [
526
+ ts.factory.createObjectLiteralExpression(
527
+ [
528
+ ts.factory.createPropertyAssignment(
529
+ "expected",
530
+ ts.factory.createStringLiteral(expected),
531
+ ),
532
+ ts.factory.createPropertyAssignment(
533
+ "value",
534
+ value,
535
+ ),
536
+ ],
537
+ true,
538
+ ),
539
+ ],
540
+ ),
541
+ );
542
+ }