typia 3.8.2 → 3.8.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 (173) hide show
  1. package/README.md +1 -1
  2. package/lib/Primitive.d.ts +10 -10
  3. package/lib/factories/MetadataFactory.d.ts +1 -1
  4. package/lib/factories/MetadataFactory.js +1 -1
  5. package/lib/factories/MetadataFactory.js.map +1 -1
  6. package/lib/factories/internal/metadata/emplace_metadata_object.js +3 -1
  7. package/lib/factories/internal/metadata/emplace_metadata_object.js.map +1 -1
  8. package/lib/factories/internal/metadata/iterate_metadata_tuple.js +2 -0
  9. package/lib/factories/internal/metadata/iterate_metadata_tuple.js.map +1 -1
  10. package/lib/metadata/IMetadata.d.ts +1 -0
  11. package/lib/metadata/Metadata.d.ts +1 -0
  12. package/lib/metadata/Metadata.js +5 -0
  13. package/lib/metadata/Metadata.js.map +1 -1
  14. package/lib/programmers/CheckerProgrammer.js +17 -6
  15. package/lib/programmers/CheckerProgrammer.js.map +1 -1
  16. package/lib/programmers/internal/application_object.js +1 -0
  17. package/lib/programmers/internal/application_object.js.map +1 -1
  18. package/lib/programmers/internal/application_tuple.js +1 -1
  19. package/lib/programmers/internal/application_tuple.js.map +1 -1
  20. package/lib/schemas/IJsonSchema.d.ts +1 -0
  21. package/package.json +1 -1
  22. package/src/Primitive.ts +123 -120
  23. package/src/executable/TypiaGenerateWizard.ts +85 -85
  24. package/src/executable/TypiaSetupWizard.ts +133 -133
  25. package/src/executable/setup/CommandExecutor.ts +8 -8
  26. package/src/executable/setup/PackageManager.ts +71 -71
  27. package/src/executable/typia.ts +52 -52
  28. package/src/factories/CommentFactory.ts +14 -14
  29. package/src/factories/ExpressionFactory.ts +70 -70
  30. package/src/factories/IdentifierFactory.ts +63 -63
  31. package/src/factories/LiteralFactory.ts +39 -39
  32. package/src/factories/MetadataCollection.ts +121 -121
  33. package/src/factories/MetadataFactory.ts +62 -62
  34. package/src/factories/MetadataTagFactory.ts +297 -297
  35. package/src/factories/StatementFactory.ts +24 -24
  36. package/src/factories/TemplateFactory.ts +58 -58
  37. package/src/factories/TypeFactory.ts +124 -124
  38. package/src/factories/internal/metadata/MetadataHelper.ts +12 -12
  39. package/src/factories/internal/metadata/emplace_metadata_object.ts +143 -140
  40. package/src/factories/internal/metadata/iterate_metadata.ts +67 -67
  41. package/src/factories/internal/metadata/iterate_metadata_map.ts +41 -41
  42. package/src/factories/internal/metadata/iterate_metadata_native.ts +219 -219
  43. package/src/factories/internal/metadata/iterate_metadata_resolve.ts +27 -27
  44. package/src/factories/internal/metadata/iterate_metadata_set.ts +33 -33
  45. package/src/factories/internal/metadata/iterate_metadata_tuple.ts +48 -44
  46. package/src/factories/internal/metadata/iterate_metadata_union.ts +57 -57
  47. package/src/functional/$any.ts +2 -2
  48. package/src/functional/$is_between.ts +2 -2
  49. package/src/functional/$is_custom.ts +14 -14
  50. package/src/functional/$is_date.ts +3 -3
  51. package/src/functional/$is_datetime.ts +2 -2
  52. package/src/functional/$is_email.ts +4 -4
  53. package/src/functional/$is_ipv4.ts +4 -4
  54. package/src/functional/$is_ipv6.ts +4 -4
  55. package/src/functional/$is_url.ts +4 -4
  56. package/src/functional/$is_uuid.ts +4 -4
  57. package/src/functional/$join.ts +46 -46
  58. package/src/functional/$number.ts +12 -12
  59. package/src/functional/$rest.ts +3 -3
  60. package/src/functional/$string.ts +37 -37
  61. package/src/functional/$tail.ts +5 -5
  62. package/src/metadata/IMetadata.ts +1 -0
  63. package/src/metadata/Metadata.ts +539 -533
  64. package/src/metadata/MetadataConstant.ts +3 -3
  65. package/src/metadata/MetadataObject.ts +127 -127
  66. package/src/module.ts +2043 -2043
  67. package/src/programmers/ApplicationProgrammer.ts +62 -62
  68. package/src/programmers/AssertCloneProgrammer.ts +79 -79
  69. package/src/programmers/AssertParseProgrammer.ts +74 -74
  70. package/src/programmers/AssertProgrammer.ts +285 -285
  71. package/src/programmers/AssertPruneProgrammer.ts +76 -76
  72. package/src/programmers/AssertStringifyProgrammer.ts +74 -74
  73. package/src/programmers/CheckerProgrammer.ts +901 -875
  74. package/src/programmers/CloneProgrammer.ts +389 -389
  75. package/src/programmers/FeatureProgrammer.ts +500 -500
  76. package/src/programmers/IsCloneProgrammer.ts +86 -86
  77. package/src/programmers/IsParseProgrammer.ts +80 -80
  78. package/src/programmers/IsProgrammer.ts +212 -212
  79. package/src/programmers/IsPruneProgrammer.ts +81 -81
  80. package/src/programmers/IsStringifyProgrammer.ts +84 -84
  81. package/src/programmers/LiteralsProgrammer.ts +65 -65
  82. package/src/programmers/PruneProgrammer.ts +347 -347
  83. package/src/programmers/RandomProgrammer.ts +423 -423
  84. package/src/programmers/StringifyProgrammer.ts +798 -798
  85. package/src/programmers/TypiaProgrammer.ts +129 -129
  86. package/src/programmers/ValidateCloneProgrammer.ts +93 -93
  87. package/src/programmers/ValidateParseProgrammer.ts +78 -78
  88. package/src/programmers/ValidateProgrammer.ts +316 -316
  89. package/src/programmers/ValidatePruneProgrammer.ts +86 -86
  90. package/src/programmers/ValidateStringifyProgrammer.ts +92 -92
  91. package/src/programmers/helpers/CloneJoiner.ts +131 -131
  92. package/src/programmers/helpers/FunctionImporeter.ts +54 -54
  93. package/src/programmers/helpers/OptionPredicator.ts +15 -15
  94. package/src/programmers/helpers/PruneJoiner.ts +144 -144
  95. package/src/programmers/helpers/StringifyJoinder.ts +113 -113
  96. package/src/programmers/helpers/StringifyPredicator.ts +12 -12
  97. package/src/programmers/helpers/UnionExplorer.ts +275 -275
  98. package/src/programmers/helpers/UnionPredicator.ts +81 -81
  99. package/src/programmers/internal/application_array.ts +36 -36
  100. package/src/programmers/internal/application_boolean.ts +17 -17
  101. package/src/programmers/internal/application_constant.ts +30 -30
  102. package/src/programmers/internal/application_default_string.ts +33 -33
  103. package/src/programmers/internal/application_native.ts +32 -32
  104. package/src/programmers/internal/application_number.ts +73 -73
  105. package/src/programmers/internal/application_object.ts +155 -154
  106. package/src/programmers/internal/application_schema.ts +213 -213
  107. package/src/programmers/internal/application_string.ts +45 -45
  108. package/src/programmers/internal/application_templates.ts +26 -26
  109. package/src/programmers/internal/application_tuple.ts +31 -29
  110. package/src/programmers/internal/check_array.ts +30 -30
  111. package/src/programmers/internal/check_custom.ts +31 -31
  112. package/src/programmers/internal/check_dynamic_properties.ts +194 -194
  113. package/src/programmers/internal/check_everything.ts +28 -28
  114. package/src/programmers/internal/check_native.ts +21 -21
  115. package/src/programmers/internal/check_string_tags.ts +67 -67
  116. package/src/programmers/internal/check_union_array_like.ts +271 -271
  117. package/src/programmers/internal/check_union_tuple.ts +33 -33
  118. package/src/programmers/internal/random_custom.ts +29 -29
  119. package/src/programmers/internal/stringify_dynamic_properties.ts +167 -167
  120. package/src/programmers/internal/stringify_native.ts +7 -7
  121. package/src/programmers/internal/stringify_regular_properties.ts +83 -83
  122. package/src/schemas/IJsonSchema.ts +130 -129
  123. package/src/transform.ts +17 -17
  124. package/src/transformers/CallExpressionTransformer.ts +177 -177
  125. package/src/transformers/FileTransformer.ts +47 -47
  126. package/src/transformers/NodeTransformer.ts +13 -13
  127. package/src/transformers/features/miscellaneous/ApplicationTransformer.ts +114 -114
  128. package/src/transformers/features/miscellaneous/AssertCloneTransformer.ts +9 -9
  129. package/src/transformers/features/miscellaneous/AssertPruneTransformer.ts +9 -9
  130. package/src/transformers/features/miscellaneous/CloneTransformer.ts +9 -9
  131. package/src/transformers/features/miscellaneous/CreateAssertCloneTransformer.ts +9 -9
  132. package/src/transformers/features/miscellaneous/CreateAssertPruneTransformer.ts +9 -9
  133. package/src/transformers/features/miscellaneous/CreateCloneTransformer.ts +9 -9
  134. package/src/transformers/features/miscellaneous/CreateIsCloneTransformer.ts +9 -9
  135. package/src/transformers/features/miscellaneous/CreateIsPruneTransformer.ts +9 -9
  136. package/src/transformers/features/miscellaneous/CreatePruneTransformer.ts +9 -9
  137. package/src/transformers/features/miscellaneous/CreateRandomTransformer.ts +39 -39
  138. package/src/transformers/features/miscellaneous/CreateValidateCloneTransformer.ts +9 -9
  139. package/src/transformers/features/miscellaneous/CreateValidatePruneTransformer.ts +9 -9
  140. package/src/transformers/features/miscellaneous/IsCloneTransformer.ts +9 -9
  141. package/src/transformers/features/miscellaneous/IsPruneTransformer.ts +9 -9
  142. package/src/transformers/features/miscellaneous/LiteralsTransformer.ts +28 -28
  143. package/src/transformers/features/miscellaneous/MetadataTransformer.ts +52 -52
  144. package/src/transformers/features/miscellaneous/PruneTransformer.ts +9 -9
  145. package/src/transformers/features/miscellaneous/RandomTransformer.ts +42 -42
  146. package/src/transformers/features/miscellaneous/ValidateCloneTransformer.ts +9 -9
  147. package/src/transformers/features/miscellaneous/ValidatePruneTransformer.ts +9 -9
  148. package/src/transformers/features/parsers/AssertParseTransformer.ts +9 -9
  149. package/src/transformers/features/parsers/CreateAssertParseTransformer.ts +9 -9
  150. package/src/transformers/features/parsers/CreateIsParseTransformer.ts +9 -9
  151. package/src/transformers/features/parsers/CreateValidateParseTransformer.ts +9 -9
  152. package/src/transformers/features/parsers/IsParseTransformer.ts +9 -9
  153. package/src/transformers/features/parsers/ValidateParseTransformer.ts +9 -9
  154. package/src/transformers/features/stringifiers/AssertStringifyTransformer.ts +10 -10
  155. package/src/transformers/features/stringifiers/CreateAssertStringifyTransformer.ts +12 -12
  156. package/src/transformers/features/stringifiers/CreateIsStringifyTransformer.ts +9 -9
  157. package/src/transformers/features/stringifiers/CreateStringifyTransformer.ts +9 -9
  158. package/src/transformers/features/stringifiers/CreateValidateStringifyProgrammer.ts +12 -12
  159. package/src/transformers/features/stringifiers/IsStringifyTransformer.ts +9 -9
  160. package/src/transformers/features/stringifiers/StringifyTransformer.ts +9 -9
  161. package/src/transformers/features/stringifiers/ValidateStringifyTransformer.ts +10 -10
  162. package/src/transformers/features/validators/AssertTransformer.ts +11 -11
  163. package/src/transformers/features/validators/CreateAssertTransformer.ts +13 -13
  164. package/src/transformers/features/validators/CreateIsTransformer.ts +11 -11
  165. package/src/transformers/features/validators/CreateValidateTransformer.ts +13 -13
  166. package/src/transformers/features/validators/IsTransformer.ts +11 -11
  167. package/src/transformers/features/validators/ValidateTransformer.ts +11 -11
  168. package/src/transformers/internal/GenericTransformer.ts +97 -97
  169. package/src/utils/ArrayUtil.ts +44 -44
  170. package/src/utils/Escaper.ts +45 -45
  171. package/src/utils/MapUtil.ts +12 -12
  172. package/src/utils/PatternUtil.ts +30 -30
  173. package/src/utils/RandomGenerator.ts +81 -81
@@ -1,347 +1,347 @@
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 { TypeFactory } from "../factories/TypeFactory";
8
-
9
- import { Metadata } from "../metadata/Metadata";
10
-
11
- import { IProject } from "../transformers/IProject";
12
-
13
- import { FeatureProgrammer } from "./FeatureProgrammer";
14
- import { IsProgrammer } from "./IsProgrammer";
15
- import { FunctionImporter } from "./helpers/FunctionImporeter";
16
- import { PruneJoiner } from "./helpers/PruneJoiner";
17
- import { UnionExplorer } from "./helpers/UnionExplorer";
18
- import { decode_union_object } from "./internal/decode_union_object";
19
-
20
- export namespace PruneProgrammer {
21
- /**
22
- * @deprecated Use `write()` function instead
23
- */
24
- export const generate =
25
- (project: IProject, modulo: ts.LeftHandSideExpression) =>
26
- (type: ts.Type, name?: string) =>
27
- write(project)(modulo)(type, name);
28
-
29
- export const write =
30
- (project: IProject) => (modulo: ts.LeftHandSideExpression) => {
31
- const importer: FunctionImporter = new FunctionImporter();
32
- return FeatureProgrammer.analyze(project)({
33
- ...configure(project)(importer),
34
- addition: (collection) => {
35
- const isFunctors =
36
- IsProgrammer.write_functors(project)(importer)(
37
- collection,
38
- );
39
- const isUnioners =
40
- IsProgrammer.write_unioners(project)(importer)(
41
- collection,
42
- );
43
-
44
- return [
45
- ...importer.declare(modulo),
46
- ...isFunctors.filter((_, i) =>
47
- importer.hasLocal(`$io${i}`),
48
- ),
49
- ...isUnioners.filter((_, i) =>
50
- importer.hasLocal(`$iu${i}`),
51
- ),
52
- ];
53
- },
54
- })(importer);
55
- };
56
-
57
- /* -----------------------------------------------------------
58
- DECODERS
59
- ----------------------------------------------------------- */
60
- const decode =
61
- (project: IProject) =>
62
- (importer: FunctionImporter) =>
63
- (
64
- input: ts.Expression,
65
- meta: Metadata,
66
- explore: FeatureProgrammer.IExplore,
67
- ): ts.ConciseBody => {
68
- if (filter(meta) === false) return ts.factory.createBlock([]);
69
-
70
- interface IUnion {
71
- type: string;
72
- is: () => ts.Expression;
73
- value: () => ts.Expression | ts.Block | ts.ReturnStatement;
74
- }
75
- const unions: IUnion[] = [];
76
-
77
- //----
78
- // LIST UP UNION TYPES
79
- //----
80
- // TUPLES
81
- for (const tuple of meta.tuples.filter((t) =>
82
- t.some((e) => filter(e.rest ?? e)),
83
- ))
84
- unions.push({
85
- type: "tuple",
86
- is: () =>
87
- IsProgrammer.decode(project, importer)(
88
- input,
89
- (() => {
90
- const partial = Metadata.initialize();
91
- partial.tuples.push(tuple);
92
- return partial;
93
- })(),
94
- explore,
95
- [],
96
- [],
97
- ),
98
- value: () =>
99
- decode_tuple(project)(importer)(input, tuple, explore),
100
- });
101
-
102
- // ARRAYS
103
- if (meta.arrays.filter(filter).length)
104
- unions.push({
105
- type: "array",
106
- is: () => ExpressionFactory.isArray(input),
107
- value: () =>
108
- explore_arrays(project)(importer)(
109
- input,
110
- meta.arrays,
111
- {
112
- ...explore,
113
- from: "array",
114
- },
115
- [],
116
- [],
117
- ),
118
- });
119
-
120
- // BUILT-IN CLASSES
121
- if (meta.natives.length)
122
- for (const native of meta.natives)
123
- unions.push({
124
- type: "native",
125
- is: () => ExpressionFactory.isInstanceOf(native)(input),
126
- value: () => ts.factory.createReturnStatement(),
127
- });
128
- if (meta.sets.length)
129
- unions.push({
130
- type: "set",
131
- is: () => ExpressionFactory.isInstanceOf("Set")(input),
132
- value: () => ts.factory.createReturnStatement(),
133
- });
134
- if (meta.maps.length)
135
- unions.push({
136
- type: "map",
137
- is: () => ExpressionFactory.isInstanceOf("Map")(input),
138
- value: () => ts.factory.createReturnStatement(),
139
- });
140
-
141
- // OBJECTS
142
- if (meta.objects.length)
143
- unions.push({
144
- type: "object",
145
- is: () =>
146
- ExpressionFactory.isObject({
147
- checkNull: true,
148
- checkArray: false,
149
- })(input),
150
- value: () =>
151
- explore_objects(importer)(input, meta, {
152
- ...explore,
153
- from: "object",
154
- }),
155
- });
156
-
157
- //----
158
- // STATEMENTS
159
- //----
160
- const converter = (
161
- v: ts.Expression | ts.Block | ts.ReturnStatement,
162
- ) =>
163
- ts.isReturnStatement(v) || ts.isBlock(v)
164
- ? v
165
- : ts.factory.createExpressionStatement(v);
166
-
167
- const statements: ts.Statement[] = unions.map((u) =>
168
- ts.factory.createIfStatement(u.is(), converter(u.value())),
169
- );
170
- return ts.factory.createBlock(statements, true);
171
- };
172
-
173
- const decode_tuple =
174
- (project: IProject) =>
175
- (importer: FunctionImporter) =>
176
- (
177
- input: ts.Expression,
178
- tuple: Metadata[],
179
- explore: FeatureProgrammer.IExplore,
180
- ): ts.Block => {
181
- const children: ts.ConciseBody[] = tuple
182
- .map((elem, index) => [elem, index] as const)
183
- .filter(([elem]) => filter(elem) && elem.rest === null)
184
- .map(([elem, index]) =>
185
- decode(project)(importer)(
186
- ts.factory.createElementAccessExpression(input, index),
187
- elem,
188
- {
189
- ...explore,
190
- from: "array",
191
- },
192
- ),
193
- );
194
- const rest = (() => {
195
- if (tuple.length === 0) return null;
196
-
197
- const last: Metadata = tuple[tuple.length - 1]!;
198
- const rest: Metadata | null = last.rest;
199
- if (rest === null || filter(rest) === false) return null;
200
-
201
- return decode(project)(importer)(
202
- ts.factory.createCallExpression(
203
- IdentifierFactory.access(input)("slice"),
204
- undefined,
205
- [ts.factory.createNumericLiteral(tuple.length - 1)],
206
- ),
207
- (() => {
208
- const wrapper: Metadata = Metadata.initialize();
209
- wrapper.arrays.push(rest);
210
- return wrapper;
211
- })(),
212
- {
213
- ...explore,
214
- start: tuple.length - 1,
215
- },
216
- );
217
- })();
218
- return PruneJoiner.tuple(children, rest);
219
- };
220
-
221
- const decode_array = (project: IProject) => (importer: FunctionImporter) =>
222
- FeatureProgrammer.decode_array(configure(project)(importer))(importer)(
223
- PruneJoiner.array,
224
- );
225
-
226
- const decode_object = (importer: FunctionImporter) =>
227
- FeatureProgrammer.decode_object({
228
- trace: false,
229
- path: false,
230
- functors: FUNCTORS,
231
- })(importer);
232
-
233
- const explore_arrays =
234
- (project: IProject) => (importer: FunctionImporter) =>
235
- UnionExplorer.array({
236
- checker: IsProgrammer.decode(project, importer),
237
- decoder: decode_array(project)(importer),
238
- empty: ts.factory.createReturnStatement(),
239
- success: ts.factory.createTrue(),
240
- failure: (input, expected) =>
241
- create_throw_error(importer)(expected)(input),
242
- });
243
-
244
- const explore_objects =
245
- (importer: FunctionImporter) =>
246
- (
247
- input: ts.Expression,
248
- meta: Metadata,
249
- explore: FeatureProgrammer.IExplore,
250
- ) => {
251
- if (meta.objects.length === 1)
252
- return decode_object(importer)(
253
- input,
254
- meta.objects[0]!,
255
- explore,
256
- );
257
-
258
- return ts.factory.createCallExpression(
259
- ts.factory.createIdentifier(`${UNIONERS}${meta.union_index!}`),
260
- undefined,
261
- [input],
262
- );
263
- };
264
-
265
- const filter = (meta: Metadata): boolean =>
266
- meta.any === false &&
267
- (meta.objects.length !== 0 ||
268
- meta.tuples.some((t) => t.some((e) => filter(e.rest ?? e))) ||
269
- meta.arrays.some((e) => filter(e)));
270
-
271
- /* -----------------------------------------------------------
272
- CONFIGURATIONS
273
- ----------------------------------------------------------- */
274
- const FUNCTORS = "$po";
275
- const UNIONERS = "$pu";
276
-
277
- const configure =
278
- (project: IProject) =>
279
- (importer: FunctionImporter): FeatureProgrammer.IConfig => ({
280
- types: {
281
- input: (type, name) =>
282
- ts.factory.createTypeReferenceNode(
283
- name ?? TypeFactory.getFullName(project.checker)(type),
284
- ),
285
- output: () => TypeFactory.keyword("void"),
286
- },
287
- functors: FUNCTORS,
288
- unioners: UNIONERS,
289
- trace: false,
290
- path: false,
291
- initializer,
292
- decoder: decode(project)(importer),
293
- objector: objector(project)(importer),
294
- });
295
-
296
- const objector =
297
- (project: IProject) =>
298
- (importer: FunctionImporter): FeatureProgrammer.IConfig.IObjector => ({
299
- checker: IsProgrammer.decode(project, importer),
300
- decoder: decode_object(importer),
301
- joiner: PruneJoiner.object,
302
- unionizer: decode_union_object(
303
- IsProgrammer.decode_object(importer),
304
- )(decode_object(importer))((exp) => exp)((value, expected) =>
305
- create_throw_error(importer)(expected)(value),
306
- ),
307
- failure: (input, expected) =>
308
- create_throw_error(importer)(expected)(input),
309
- });
310
-
311
- const initializer: FeatureProgrammer.IConfig["initializer"] =
312
- ({ checker }) =>
313
- (type) => {
314
- const collection = new MetadataCollection();
315
- const meta = MetadataFactory.analyze(checker)({
316
- resolve: false,
317
- constant: true,
318
- })(collection)(type);
319
- return [collection, meta];
320
- };
321
-
322
- const create_throw_error =
323
- (importer: FunctionImporter) =>
324
- (expected: string) =>
325
- (value: ts.Expression) =>
326
- ts.factory.createExpressionStatement(
327
- ts.factory.createCallExpression(
328
- importer.use("throws"),
329
- [],
330
- [
331
- ts.factory.createObjectLiteralExpression(
332
- [
333
- ts.factory.createPropertyAssignment(
334
- "expected",
335
- ts.factory.createStringLiteral(expected),
336
- ),
337
- ts.factory.createPropertyAssignment(
338
- "value",
339
- value,
340
- ),
341
- ],
342
- true,
343
- ),
344
- ],
345
- ),
346
- );
347
- }
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 { TypeFactory } from "../factories/TypeFactory";
8
+
9
+ import { Metadata } from "../metadata/Metadata";
10
+
11
+ import { IProject } from "../transformers/IProject";
12
+
13
+ import { FeatureProgrammer } from "./FeatureProgrammer";
14
+ import { IsProgrammer } from "./IsProgrammer";
15
+ import { FunctionImporter } from "./helpers/FunctionImporeter";
16
+ import { PruneJoiner } from "./helpers/PruneJoiner";
17
+ import { UnionExplorer } from "./helpers/UnionExplorer";
18
+ import { decode_union_object } from "./internal/decode_union_object";
19
+
20
+ export namespace PruneProgrammer {
21
+ /**
22
+ * @deprecated Use `write()` function instead
23
+ */
24
+ export const generate =
25
+ (project: IProject, modulo: ts.LeftHandSideExpression) =>
26
+ (type: ts.Type, name?: string) =>
27
+ write(project)(modulo)(type, name);
28
+
29
+ export const write =
30
+ (project: IProject) => (modulo: ts.LeftHandSideExpression) => {
31
+ const importer: FunctionImporter = new FunctionImporter();
32
+ return FeatureProgrammer.analyze(project)({
33
+ ...configure(project)(importer),
34
+ addition: (collection) => {
35
+ const isFunctors =
36
+ IsProgrammer.write_functors(project)(importer)(
37
+ collection,
38
+ );
39
+ const isUnioners =
40
+ IsProgrammer.write_unioners(project)(importer)(
41
+ collection,
42
+ );
43
+
44
+ return [
45
+ ...importer.declare(modulo),
46
+ ...isFunctors.filter((_, i) =>
47
+ importer.hasLocal(`$io${i}`),
48
+ ),
49
+ ...isUnioners.filter((_, i) =>
50
+ importer.hasLocal(`$iu${i}`),
51
+ ),
52
+ ];
53
+ },
54
+ })(importer);
55
+ };
56
+
57
+ /* -----------------------------------------------------------
58
+ DECODERS
59
+ ----------------------------------------------------------- */
60
+ const decode =
61
+ (project: IProject) =>
62
+ (importer: FunctionImporter) =>
63
+ (
64
+ input: ts.Expression,
65
+ meta: Metadata,
66
+ explore: FeatureProgrammer.IExplore,
67
+ ): ts.ConciseBody => {
68
+ if (filter(meta) === false) return ts.factory.createBlock([]);
69
+
70
+ interface IUnion {
71
+ type: string;
72
+ is: () => ts.Expression;
73
+ value: () => ts.Expression | ts.Block | ts.ReturnStatement;
74
+ }
75
+ const unions: IUnion[] = [];
76
+
77
+ //----
78
+ // LIST UP UNION TYPES
79
+ //----
80
+ // TUPLES
81
+ for (const tuple of meta.tuples.filter((t) =>
82
+ t.some((e) => filter(e.rest ?? e)),
83
+ ))
84
+ unions.push({
85
+ type: "tuple",
86
+ is: () =>
87
+ IsProgrammer.decode(project, importer)(
88
+ input,
89
+ (() => {
90
+ const partial = Metadata.initialize();
91
+ partial.tuples.push(tuple);
92
+ return partial;
93
+ })(),
94
+ explore,
95
+ [],
96
+ [],
97
+ ),
98
+ value: () =>
99
+ decode_tuple(project)(importer)(input, tuple, explore),
100
+ });
101
+
102
+ // ARRAYS
103
+ if (meta.arrays.filter(filter).length)
104
+ unions.push({
105
+ type: "array",
106
+ is: () => ExpressionFactory.isArray(input),
107
+ value: () =>
108
+ explore_arrays(project)(importer)(
109
+ input,
110
+ meta.arrays,
111
+ {
112
+ ...explore,
113
+ from: "array",
114
+ },
115
+ [],
116
+ [],
117
+ ),
118
+ });
119
+
120
+ // BUILT-IN CLASSES
121
+ if (meta.natives.length)
122
+ for (const native of meta.natives)
123
+ unions.push({
124
+ type: "native",
125
+ is: () => ExpressionFactory.isInstanceOf(native)(input),
126
+ value: () => ts.factory.createReturnStatement(),
127
+ });
128
+ if (meta.sets.length)
129
+ unions.push({
130
+ type: "set",
131
+ is: () => ExpressionFactory.isInstanceOf("Set")(input),
132
+ value: () => ts.factory.createReturnStatement(),
133
+ });
134
+ if (meta.maps.length)
135
+ unions.push({
136
+ type: "map",
137
+ is: () => ExpressionFactory.isInstanceOf("Map")(input),
138
+ value: () => ts.factory.createReturnStatement(),
139
+ });
140
+
141
+ // OBJECTS
142
+ if (meta.objects.length)
143
+ unions.push({
144
+ type: "object",
145
+ is: () =>
146
+ ExpressionFactory.isObject({
147
+ checkNull: true,
148
+ checkArray: false,
149
+ })(input),
150
+ value: () =>
151
+ explore_objects(importer)(input, meta, {
152
+ ...explore,
153
+ from: "object",
154
+ }),
155
+ });
156
+
157
+ //----
158
+ // STATEMENTS
159
+ //----
160
+ const converter = (
161
+ v: ts.Expression | ts.Block | ts.ReturnStatement,
162
+ ) =>
163
+ ts.isReturnStatement(v) || ts.isBlock(v)
164
+ ? v
165
+ : ts.factory.createExpressionStatement(v);
166
+
167
+ const statements: ts.Statement[] = unions.map((u) =>
168
+ ts.factory.createIfStatement(u.is(), converter(u.value())),
169
+ );
170
+ return ts.factory.createBlock(statements, true);
171
+ };
172
+
173
+ const decode_tuple =
174
+ (project: IProject) =>
175
+ (importer: FunctionImporter) =>
176
+ (
177
+ input: ts.Expression,
178
+ tuple: Metadata[],
179
+ explore: FeatureProgrammer.IExplore,
180
+ ): ts.Block => {
181
+ const children: ts.ConciseBody[] = tuple
182
+ .map((elem, index) => [elem, index] as const)
183
+ .filter(([elem]) => filter(elem) && elem.rest === null)
184
+ .map(([elem, index]) =>
185
+ decode(project)(importer)(
186
+ ts.factory.createElementAccessExpression(input, index),
187
+ elem,
188
+ {
189
+ ...explore,
190
+ from: "array",
191
+ },
192
+ ),
193
+ );
194
+ const rest = (() => {
195
+ if (tuple.length === 0) return null;
196
+
197
+ const last: Metadata = tuple[tuple.length - 1]!;
198
+ const rest: Metadata | null = last.rest;
199
+ if (rest === null || filter(rest) === false) return null;
200
+
201
+ return decode(project)(importer)(
202
+ ts.factory.createCallExpression(
203
+ IdentifierFactory.access(input)("slice"),
204
+ undefined,
205
+ [ts.factory.createNumericLiteral(tuple.length - 1)],
206
+ ),
207
+ (() => {
208
+ const wrapper: Metadata = Metadata.initialize();
209
+ wrapper.arrays.push(rest);
210
+ return wrapper;
211
+ })(),
212
+ {
213
+ ...explore,
214
+ start: tuple.length - 1,
215
+ },
216
+ );
217
+ })();
218
+ return PruneJoiner.tuple(children, rest);
219
+ };
220
+
221
+ const decode_array = (project: IProject) => (importer: FunctionImporter) =>
222
+ FeatureProgrammer.decode_array(configure(project)(importer))(importer)(
223
+ PruneJoiner.array,
224
+ );
225
+
226
+ const decode_object = (importer: FunctionImporter) =>
227
+ FeatureProgrammer.decode_object({
228
+ trace: false,
229
+ path: false,
230
+ functors: FUNCTORS,
231
+ })(importer);
232
+
233
+ const explore_arrays =
234
+ (project: IProject) => (importer: FunctionImporter) =>
235
+ UnionExplorer.array({
236
+ checker: IsProgrammer.decode(project, importer),
237
+ decoder: decode_array(project)(importer),
238
+ empty: ts.factory.createReturnStatement(),
239
+ success: ts.factory.createTrue(),
240
+ failure: (input, expected) =>
241
+ create_throw_error(importer)(expected)(input),
242
+ });
243
+
244
+ const explore_objects =
245
+ (importer: FunctionImporter) =>
246
+ (
247
+ input: ts.Expression,
248
+ meta: Metadata,
249
+ explore: FeatureProgrammer.IExplore,
250
+ ) => {
251
+ if (meta.objects.length === 1)
252
+ return decode_object(importer)(
253
+ input,
254
+ meta.objects[0]!,
255
+ explore,
256
+ );
257
+
258
+ return ts.factory.createCallExpression(
259
+ ts.factory.createIdentifier(`${UNIONERS}${meta.union_index!}`),
260
+ undefined,
261
+ [input],
262
+ );
263
+ };
264
+
265
+ const filter = (meta: Metadata): boolean =>
266
+ meta.any === false &&
267
+ (meta.objects.length !== 0 ||
268
+ meta.tuples.some((t) => t.some((e) => filter(e.rest ?? e))) ||
269
+ meta.arrays.some((e) => filter(e)));
270
+
271
+ /* -----------------------------------------------------------
272
+ CONFIGURATIONS
273
+ ----------------------------------------------------------- */
274
+ const FUNCTORS = "$po";
275
+ const UNIONERS = "$pu";
276
+
277
+ const configure =
278
+ (project: IProject) =>
279
+ (importer: FunctionImporter): FeatureProgrammer.IConfig => ({
280
+ types: {
281
+ input: (type, name) =>
282
+ ts.factory.createTypeReferenceNode(
283
+ name ?? TypeFactory.getFullName(project.checker)(type),
284
+ ),
285
+ output: () => TypeFactory.keyword("void"),
286
+ },
287
+ functors: FUNCTORS,
288
+ unioners: UNIONERS,
289
+ trace: false,
290
+ path: false,
291
+ initializer,
292
+ decoder: decode(project)(importer),
293
+ objector: objector(project)(importer),
294
+ });
295
+
296
+ const objector =
297
+ (project: IProject) =>
298
+ (importer: FunctionImporter): FeatureProgrammer.IConfig.IObjector => ({
299
+ checker: IsProgrammer.decode(project, importer),
300
+ decoder: decode_object(importer),
301
+ joiner: PruneJoiner.object,
302
+ unionizer: decode_union_object(
303
+ IsProgrammer.decode_object(importer),
304
+ )(decode_object(importer))((exp) => exp)((value, expected) =>
305
+ create_throw_error(importer)(expected)(value),
306
+ ),
307
+ failure: (input, expected) =>
308
+ create_throw_error(importer)(expected)(input),
309
+ });
310
+
311
+ const initializer: FeatureProgrammer.IConfig["initializer"] =
312
+ ({ checker }) =>
313
+ (type) => {
314
+ const collection = new MetadataCollection();
315
+ const meta = MetadataFactory.analyze(checker)({
316
+ resolve: false,
317
+ constant: true,
318
+ })(collection)(type);
319
+ return [collection, meta];
320
+ };
321
+
322
+ const create_throw_error =
323
+ (importer: FunctionImporter) =>
324
+ (expected: string) =>
325
+ (value: ts.Expression) =>
326
+ ts.factory.createExpressionStatement(
327
+ ts.factory.createCallExpression(
328
+ importer.use("throws"),
329
+ [],
330
+ [
331
+ ts.factory.createObjectLiteralExpression(
332
+ [
333
+ ts.factory.createPropertyAssignment(
334
+ "expected",
335
+ ts.factory.createStringLiteral(expected),
336
+ ),
337
+ ts.factory.createPropertyAssignment(
338
+ "value",
339
+ value,
340
+ ),
341
+ ],
342
+ true,
343
+ ),
344
+ ],
345
+ ),
346
+ );
347
+ }