typia 3.4.13 → 3.4.14

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 (157) hide show
  1. package/README.md +9 -8
  2. package/lib/factories/internal/iterate_metadata_object.js +4 -2
  3. package/lib/factories/internal/iterate_metadata_object.js.map +1 -1
  4. package/package.json +1 -1
  5. package/src/IValidation.ts +21 -21
  6. package/src/Primitive.ts +82 -82
  7. package/src/TypeGuardError.ts +36 -36
  8. package/src/executable/internal/CommandParser.ts +15 -15
  9. package/src/executable/internal/TypiaSetupWizard.ts +175 -175
  10. package/src/executable/typia.ts +46 -46
  11. package/src/factories/CommentFactory.ts +10 -10
  12. package/src/factories/ExpressionFactory.ts +66 -66
  13. package/src/factories/IdentifierFactory.ts +72 -72
  14. package/src/factories/LiteralFactory.ts +44 -44
  15. package/src/factories/MetadataCollection.ts +122 -122
  16. package/src/factories/MetadataFactory.ts +46 -46
  17. package/src/factories/MetadataTagFactory.ts +347 -347
  18. package/src/factories/StatementFactory.ts +60 -60
  19. package/src/factories/TemplateFactory.ts +56 -56
  20. package/src/factories/TypeFactory.ts +101 -101
  21. package/src/factories/ValueFactory.ts +12 -12
  22. package/src/factories/internal/MetadataHelper.ts +12 -12
  23. package/src/factories/internal/emplace_metadata_object.ts +140 -140
  24. package/src/factories/internal/explore_metadata.ts +91 -91
  25. package/src/factories/internal/iterate_metadata.ts +80 -80
  26. package/src/factories/internal/iterate_metadata_array.ts +29 -29
  27. package/src/factories/internal/iterate_metadata_atomic.ts +59 -59
  28. package/src/factories/internal/iterate_metadata_coalesce.ts +33 -33
  29. package/src/factories/internal/iterate_metadata_constant.ts +58 -58
  30. package/src/factories/internal/iterate_metadata_map.ts +41 -41
  31. package/src/factories/internal/iterate_metadata_native.ts +227 -227
  32. package/src/factories/internal/iterate_metadata_object.ts +48 -45
  33. package/src/factories/internal/iterate_metadata_resolve.ts +27 -27
  34. package/src/factories/internal/iterate_metadata_set.ts +33 -33
  35. package/src/factories/internal/iterate_metadata_template.ts +38 -38
  36. package/src/factories/internal/iterate_metadata_tuple.ts +45 -45
  37. package/src/factories/internal/iterate_metadata_union.ts +59 -59
  38. package/src/functional/$every.ts +11 -11
  39. package/src/functional/$guard.ts +35 -35
  40. package/src/functional/$is_email.ts +5 -5
  41. package/src/functional/$is_ipv4.ts +5 -5
  42. package/src/functional/$is_ipv6.ts +5 -5
  43. package/src/functional/$is_url.ts +5 -5
  44. package/src/functional/$is_uuid.ts +5 -5
  45. package/src/functional/$join.ts +50 -50
  46. package/src/functional/$number.ts +19 -19
  47. package/src/functional/$report.ts +15 -15
  48. package/src/functional/$rest.ts +3 -3
  49. package/src/functional/$string.ts +37 -37
  50. package/src/functional/$tail.ts +6 -6
  51. package/src/index.ts +4 -4
  52. package/src/metadata/IJsDocTagInfo.ts +10 -10
  53. package/src/metadata/IMetadata.ts +25 -25
  54. package/src/metadata/IMetadataApplication.ts +7 -7
  55. package/src/metadata/IMetadataConstant.ts +16 -16
  56. package/src/metadata/IMetadataEntry.ts +6 -6
  57. package/src/metadata/IMetadataObject.ts +29 -29
  58. package/src/metadata/IMetadataProperty.ts +11 -11
  59. package/src/metadata/IMetadataTag.ts +122 -122
  60. package/src/metadata/Metadata.ts +477 -477
  61. package/src/metadata/MetadataConstant.ts +3 -3
  62. package/src/metadata/MetadataObject.ts +131 -131
  63. package/src/metadata/MetadataProperty.ts +64 -64
  64. package/src/module.ts +1535 -1535
  65. package/src/programmers/ApplicationProgrammer.ts +55 -55
  66. package/src/programmers/AssertParseProgrammer.ts +45 -45
  67. package/src/programmers/AssertProgrammer.ts +444 -444
  68. package/src/programmers/AssertStringifyProgrammer.ts +45 -45
  69. package/src/programmers/CheckerProgrammer.ts +804 -804
  70. package/src/programmers/FeatureProgrammer.ts +327 -327
  71. package/src/programmers/IsParseProgrammer.ts +51 -51
  72. package/src/programmers/IsProgrammer.ts +172 -172
  73. package/src/programmers/IsStringifyProgrammer.ts +49 -49
  74. package/src/programmers/StringifyProgrammer.ts +756 -756
  75. package/src/programmers/ValidateParseProgrammer.ts +49 -49
  76. package/src/programmers/ValidateProgrammer.ts +236 -236
  77. package/src/programmers/ValidateStringifyProgrammer.ts +60 -60
  78. package/src/programmers/helpers/AtomicPredicator.ts +15 -15
  79. package/src/programmers/helpers/FunctionImporeter.ts +31 -31
  80. package/src/programmers/helpers/IExpressionEntry.ts +10 -10
  81. package/src/programmers/helpers/OptionPredicator.ts +11 -11
  82. package/src/programmers/helpers/StringifyJoinder.ts +111 -111
  83. package/src/programmers/helpers/StringifyPredicator.ts +18 -18
  84. package/src/programmers/helpers/UnionExplorer.ts +437 -437
  85. package/src/programmers/helpers/UnionPredicator.ts +81 -81
  86. package/src/programmers/internal/application_array.ts +45 -45
  87. package/src/programmers/internal/application_boolean.ts +17 -17
  88. package/src/programmers/internal/application_constant.ts +29 -29
  89. package/src/programmers/internal/application_default.ts +17 -17
  90. package/src/programmers/internal/application_default_string.ts +32 -32
  91. package/src/programmers/internal/application_native.ts +29 -29
  92. package/src/programmers/internal/application_number.ts +76 -76
  93. package/src/programmers/internal/application_object.ts +103 -103
  94. package/src/programmers/internal/application_schema.ts +221 -221
  95. package/src/programmers/internal/application_string.ts +49 -49
  96. package/src/programmers/internal/application_templates.ts +27 -27
  97. package/src/programmers/internal/application_tuple.ts +25 -25
  98. package/src/programmers/internal/check_array.ts +44 -44
  99. package/src/programmers/internal/check_dynamic_properties.ts +146 -146
  100. package/src/programmers/internal/check_everything.ts +25 -25
  101. package/src/programmers/internal/check_length.ts +46 -46
  102. package/src/programmers/internal/check_native.ts +9 -9
  103. package/src/programmers/internal/check_number.ts +178 -178
  104. package/src/programmers/internal/check_object.ts +42 -42
  105. package/src/programmers/internal/check_string.ts +24 -24
  106. package/src/programmers/internal/check_string_tags.ts +63 -63
  107. package/src/programmers/internal/check_template.ts +50 -50
  108. package/src/programmers/internal/decode_union_object.ts +73 -73
  109. package/src/programmers/internal/feature_object_entries.ts +49 -49
  110. package/src/programmers/internal/metadata_to_pattern.ts +31 -31
  111. package/src/programmers/internal/stringify_dynamic_properties.ts +164 -164
  112. package/src/programmers/internal/stringify_native.ts +8 -8
  113. package/src/programmers/internal/stringify_regular_properties.ts +81 -81
  114. package/src/programmers/internal/template_to_pattern.ts +15 -15
  115. package/src/schemas/IJsonApplication.ts +9 -9
  116. package/src/schemas/IJsonComponents.ts +24 -24
  117. package/src/schemas/IJsonSchema.ts +92 -92
  118. package/src/transform.ts +20 -20
  119. package/src/transformers/CallExpressionTransformer.ts +124 -124
  120. package/src/transformers/ExpressionWithArgumentTransformer.ts +66 -66
  121. package/src/transformers/FileTransformer.ts +49 -49
  122. package/src/transformers/IProject.ts +11 -11
  123. package/src/transformers/ITransformOptions.ts +4 -4
  124. package/src/transformers/NodeTransformer.ts +19 -19
  125. package/src/transformers/features/miscellaneous/ApplicationTransformer.ts +114 -114
  126. package/src/transformers/features/miscellaneous/CreateInstanceTransformer.ts +41 -41
  127. package/src/transformers/features/miscellaneous/MetadataTransformer.ts +55 -55
  128. package/src/transformers/features/parsers/AssertParseTransformer.ts +36 -36
  129. package/src/transformers/features/parsers/CreateAssertParseTransformer.ts +32 -32
  130. package/src/transformers/features/parsers/CreateIsParseTransformer.ts +32 -32
  131. package/src/transformers/features/parsers/CreateValidateParseTransformer.ts +32 -32
  132. package/src/transformers/features/parsers/IsParseTransformer.ts +36 -36
  133. package/src/transformers/features/parsers/ValidateParseTransformer.ts +36 -36
  134. package/src/transformers/features/stringifiers/AssertStringifyTransformer.ts +38 -38
  135. package/src/transformers/features/stringifiers/CreateAssertStringifyTransformer.ts +32 -32
  136. package/src/transformers/features/stringifiers/CreateIsStringifyTransformer.ts +32 -32
  137. package/src/transformers/features/stringifiers/CreateStringifyTransformer.ts +31 -31
  138. package/src/transformers/features/stringifiers/CreateValidateStringifyProgrammer.ts +32 -32
  139. package/src/transformers/features/stringifiers/IsStringifyTransformer.ts +38 -38
  140. package/src/transformers/features/stringifiers/StringifyTransformer.ts +36 -36
  141. package/src/transformers/features/stringifiers/ValidateStringifyTransformer.ts +38 -38
  142. package/src/transformers/features/validators/AssertTransformer.ts +43 -43
  143. package/src/transformers/features/validators/CreateAssertTransformer.ts +35 -35
  144. package/src/transformers/features/validators/CreateIsTransformer.ts +35 -35
  145. package/src/transformers/features/validators/CreateValidateTransformer.ts +35 -35
  146. package/src/transformers/features/validators/IsTransformer.ts +43 -43
  147. package/src/transformers/features/validators/ValidateTransformer.ts +43 -43
  148. package/src/typings/Atomic.ts +17 -17
  149. package/src/typings/ClassProperties.ts +5 -5
  150. package/src/typings/OmitNever.ts +3 -3
  151. package/src/typings/SpecialFields.ts +3 -3
  152. package/src/typings/Writable.ts +11 -11
  153. package/src/utils/ArrayUtil.ts +49 -49
  154. package/src/utils/Escaper.ts +50 -50
  155. package/src/utils/MapUtil.ts +14 -14
  156. package/src/utils/PatternUtil.ts +30 -30
  157. package/src/utils/Singleton.ts +17 -17
@@ -1,804 +1,804 @@
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 { ValueFactory } from "../factories/ValueFactory";
8
-
9
- import { IMetadataTag } from "../metadata/IMetadataTag";
10
- import { Metadata } from "../metadata/Metadata";
11
- import { MetadataObject } from "../metadata/MetadataObject";
12
-
13
- import { IProject } from "../transformers/IProject";
14
-
15
- import { FeatureProgrammer } from "./FeatureProgrammer";
16
- import { FunctionImporter } from "./helpers/FunctionImporeter";
17
- import { IExpressionEntry } from "./helpers/IExpressionEntry";
18
- import { OptionPredicator } from "./helpers/OptionPredicator";
19
- import { UnionExplorer } from "./helpers/UnionExplorer";
20
- import { check_array } from "./internal/check_array";
21
- import { check_native } from "./internal/check_native";
22
- import { check_number } from "./internal/check_number";
23
- import { check_string } from "./internal/check_string";
24
- import { check_template } from "./internal/check_template";
25
- import { decode_union_object } from "./internal/decode_union_object";
26
-
27
- export namespace CheckerProgrammer {
28
- export interface IConfig {
29
- functors: string;
30
- unioners: string;
31
- path: boolean;
32
- trace: boolean;
33
- equals: boolean;
34
- numeric: boolean;
35
- combiner: IConfig.Combiner;
36
- decoder?: FeatureProgrammer.Decoder<Metadata>;
37
- joiner: IConfig.IJoiner;
38
- success: ts.Expression;
39
- }
40
- export namespace IConfig {
41
- export interface Combiner {
42
- (explorer: IExplore): {
43
- (logic: "and" | "or"): {
44
- (
45
- input: ts.Expression,
46
- binaries: IBinary[],
47
- expected: string,
48
- ): ts.Expression;
49
- };
50
- };
51
- }
52
- export interface IJoiner {
53
- object(entries: IExpressionEntry[]): ts.Expression;
54
- array(input: ts.Expression, arrow: ts.ArrowFunction): ts.Expression;
55
- tuple?(exprs: ts.Expression[]): ts.Expression;
56
-
57
- failure(
58
- value: ts.Expression,
59
- expected: string,
60
- explore?: FeatureProgrammer.IExplore,
61
- ): ts.Expression;
62
- is?(expression: ts.Expression): ts.Expression;
63
- required?(exp: ts.Expression): ts.Expression;
64
- full?: (
65
- condition: ts.Expression,
66
- ) => (
67
- input: ts.Expression,
68
- expected: string,
69
- explore: IExplore,
70
- ) => ts.Expression;
71
- }
72
- }
73
- export import IExplore = FeatureProgrammer.IExplore;
74
-
75
- export interface IBinary {
76
- expression: ts.Expression;
77
- combined: boolean;
78
- }
79
-
80
- /* -----------------------------------------------------------
81
- GENERATORS
82
- ----------------------------------------------------------- */
83
- export function generate(
84
- project: IProject,
85
- config: IConfig,
86
- importer: FunctionImporter,
87
- addition?: () => ts.Statement[],
88
- ) {
89
- return FeatureProgrammer.generate(
90
- project,
91
- CONFIG(project, config, importer),
92
- importer,
93
- () => (addition ? (addition ? addition() : []) : undefined),
94
- );
95
- }
96
-
97
- export const generate_functors = (
98
- project: IProject,
99
- config: IConfig,
100
- importer: FunctionImporter,
101
- ) =>
102
- FeatureProgrammer.generate_functors(
103
- CONFIG(project, config, importer),
104
- importer,
105
- );
106
-
107
- export const generate_unioners = (
108
- project: IProject,
109
- config: IConfig,
110
- importer: FunctionImporter,
111
- ) =>
112
- FeatureProgrammer.generate_unioners(
113
- CONFIG(project, { ...config, numeric: false }, importer),
114
- );
115
-
116
- function CONFIG(
117
- project: IProject,
118
- config: IConfig,
119
- importer: FunctionImporter,
120
- ): FeatureProgrammer.IConfig {
121
- const output: FeatureProgrammer.IConfig = {
122
- trace: config.trace,
123
- path: config.path,
124
- functors: config.functors,
125
- unioners: config.unioners,
126
- initializer: ({ checker }, type) => {
127
- const collection: MetadataCollection = new MetadataCollection();
128
- const meta: Metadata = MetadataFactory.generate(
129
- checker,
130
- collection,
131
- type,
132
- {
133
- resolve: false,
134
- constant: true,
135
- },
136
- );
137
- return [collection, meta];
138
- },
139
- decoder: config.decoder || decode(project, config, importer),
140
- objector: {
141
- checker: config.decoder || decode(project, config, importer),
142
- decoder: decode_object(config),
143
- joiner: config.joiner.object,
144
- unionizer: config.equals
145
- ? decode_union_object(decode_object(config))(
146
- (input, obj, explore) =>
147
- decode_object(config)(input, obj, {
148
- ...explore,
149
- tracable: true,
150
- }),
151
- )(config.joiner.is || ((expr) => expr))(
152
- (value, expected) =>
153
- ts.factory.createReturnStatement(
154
- config.joiner.failure(value, expected),
155
- ),
156
- )
157
- : (input, targets, explore) =>
158
- config.combiner(explore)("or")(
159
- input,
160
- targets.map((obj) => ({
161
- expression: decode_object(config)(
162
- input,
163
- obj,
164
- explore,
165
- ),
166
- combined: true,
167
- })),
168
- `(${targets.map((t) => t.name).join(" | ")})`,
169
- ),
170
- failure: (value, expected) =>
171
- ts.factory.createReturnStatement(
172
- config.joiner.failure(value, expected),
173
- ),
174
- is: config.joiner.is,
175
- required: config.joiner.required,
176
- full: config.joiner.full,
177
- },
178
- };
179
- if (config.numeric === true)
180
- output.generator = {
181
- unioners: FeatureProgrammer.generate_unioners(
182
- CONFIG(project, { ...config, numeric: false }, importer),
183
- ),
184
- };
185
- return output;
186
- }
187
-
188
- /* -----------------------------------------------------------
189
- DECODERS
190
- ----------------------------------------------------------- */
191
- export function decode(
192
- project: IProject,
193
- config: IConfig,
194
- importer: FunctionImporter,
195
- ): (
196
- input: ts.Expression,
197
- meta: Metadata,
198
- explore: IExplore,
199
- tags: IMetadataTag[],
200
- ) => ts.Expression;
201
-
202
- /**
203
- * @internal
204
- */
205
- export function decode(
206
- project: IProject,
207
- config: IConfig,
208
- importer: FunctionImporter,
209
- checkTupleLength: boolean,
210
- ): (
211
- input: ts.Expression,
212
- meta: Metadata,
213
- explore: IExplore,
214
- tags: IMetadataTag[],
215
- ) => ts.Expression;
216
-
217
- /**
218
- * @internal
219
- */
220
- export function decode(
221
- project: IProject,
222
- config: IConfig,
223
- importer: FunctionImporter,
224
- checkTupleLength: boolean = true,
225
- ) {
226
- return function (
227
- input: ts.Expression,
228
- meta: Metadata,
229
- explore: IExplore,
230
- tags: IMetadataTag[],
231
- ): ts.Expression {
232
- if (meta.any) return config.success;
233
-
234
- const top: IBinary[] = [];
235
- const binaries: IBinary[] = [];
236
- const add = create_add(binaries)(input);
237
- const getConstantValue = (
238
- value: number | string | bigint | boolean,
239
- ) =>
240
- typeof value === "string"
241
- ? ts.factory.createStringLiteral(value)
242
- : ts.factory.createIdentifier(value.toString());
243
-
244
- //----
245
- // CHECK OPTIONAL
246
- //----
247
- const checkOptional: boolean = meta.empty() || meta.isUnionBucket();
248
-
249
- // NULLABLE
250
- if (
251
- checkOptional ||
252
- meta.nullable
253
- // || (meta.objects.length && meta.size() !== meta.objects.length)
254
- )
255
- (meta.nullable ? add : create_add(top)(input))(
256
- meta.nullable,
257
- ValueFactory.NULL(),
258
- );
259
-
260
- // UNDEFINDABLE
261
- if (checkOptional || !meta.required)
262
- (meta.required ? create_add(top)(input) : add)(
263
- !meta.required,
264
- ValueFactory.UNDEFINED(),
265
- );
266
-
267
- // FUNCTIONAL
268
- if (meta.functional === true)
269
- if (
270
- OptionPredicator.functional(project.options) ||
271
- meta.size() !== 1
272
- )
273
- add(
274
- true,
275
- ts.factory.createStringLiteral("function"),
276
- ValueFactory.TYPEOF(input),
277
- );
278
- else
279
- binaries.push({
280
- combined: false,
281
- expression: config.success,
282
- });
283
-
284
- //----
285
- // VALUES
286
- //----
287
- // CONSTANT VALUES
288
- for (const constant of meta.constants)
289
- for (const val of constant.values)
290
- add(true, getConstantValue(val));
291
-
292
- // ATOMIC VALUES
293
- for (const type of meta.atomics)
294
- if (type === "number")
295
- binaries.push({
296
- expression: check_number(project, config.numeric)(
297
- input,
298
- tags,
299
- ),
300
- combined: false,
301
- });
302
- else if (type === "string")
303
- binaries.push({
304
- expression: check_string(importer)(input, tags),
305
- combined: false,
306
- });
307
- else
308
- add(
309
- true,
310
- ts.factory.createStringLiteral(type),
311
- ValueFactory.TYPEOF(input),
312
- );
313
-
314
- // TEMPLATE LITERAL VALUES
315
- if (meta.templates.length)
316
- binaries.push({
317
- expression: check_template(importer)(
318
- input,
319
- meta.templates,
320
- tags,
321
- ),
322
- combined: false,
323
- });
324
-
325
- //----
326
- // INSTANCES
327
- //----
328
- // TUPLE
329
- if (meta.tuples.length > 0) {
330
- const inner: ts.Expression[] = [];
331
- for (const tuple of meta.tuples)
332
- inner.push(
333
- decode_tuple(
334
- project,
335
- config,
336
- importer,
337
- checkTupleLength,
338
- )(input, tuple, explore, tags),
339
- );
340
-
341
- // ADD
342
- binaries.push({
343
- expression: config.combiner(explore)("and")(
344
- input,
345
- [
346
- ...(checkTupleLength
347
- ? [
348
- {
349
- expression:
350
- ExpressionFactory.isArray(input),
351
- combined: false,
352
- },
353
- ]
354
- : []),
355
- ...inner.map((expression) => ({
356
- expression,
357
- combined: true,
358
- })),
359
- ],
360
- meta.getName(),
361
- ),
362
- combined: true,
363
- });
364
- }
365
-
366
- // ARRAY
367
- if (meta.arrays.length > 0)
368
- if (meta.arrays.every((elem) => elem.any))
369
- binaries.push({
370
- expression: check_array(input, tags),
371
- combined: false,
372
- });
373
- else
374
- binaries.push({
375
- expression: config.combiner(explore)("and")(
376
- input,
377
- [
378
- ...(checkTupleLength
379
- ? [
380
- {
381
- expression: check_array(
382
- input,
383
- tags,
384
- ),
385
- combined: false,
386
- },
387
- ]
388
- : []),
389
- {
390
- expression: explore_array(
391
- project,
392
- config,
393
- importer,
394
- )(
395
- input,
396
- meta.arrays,
397
- {
398
- ...explore,
399
- from: "array",
400
- },
401
- tags,
402
- ),
403
- combined: true,
404
- },
405
- ],
406
- meta.getName(),
407
- ),
408
- combined: true,
409
- });
410
-
411
- // OBJECT
412
- if (meta.objects.length > 0)
413
- binaries.push({
414
- expression: config.combiner(explore)("and")(
415
- input,
416
- [
417
- {
418
- expression: ExpressionFactory.isObject(input, {
419
- checkNull: true,
420
- checkArray: meta.objects.some((obj) =>
421
- obj.properties.every(
422
- (prop) =>
423
- !prop.key.isSoleLiteral() ||
424
- !prop.value.required,
425
- ),
426
- ),
427
- }),
428
- combined: false,
429
- },
430
- {
431
- expression: explore_objects(config)(
432
- input,
433
- meta,
434
- {
435
- ...explore,
436
- from: "object",
437
- },
438
- ),
439
- combined: true,
440
- },
441
- ],
442
- meta.getName(),
443
- ),
444
- combined: true,
445
- });
446
-
447
- // NATIVE CLASSES
448
- for (const native of meta.natives)
449
- binaries.push({
450
- expression: check_native(native)(input),
451
- combined: false,
452
- });
453
-
454
- // SETS
455
- if (meta.sets.length)
456
- if (meta.sets.every((elem) => elem.any))
457
- binaries.push({
458
- combined: false,
459
- expression: check_native("Set")(input),
460
- });
461
- else
462
- binaries.push({
463
- combined: true,
464
- expression: config.combiner(explore)("and")(
465
- input,
466
- [
467
- {
468
- combined: false,
469
- expression: check_native("Set")(input),
470
- },
471
- {
472
- combined: true,
473
- expression: explore_set(
474
- project,
475
- config,
476
- importer,
477
- )(
478
- input,
479
- meta.sets,
480
- {
481
- ...explore,
482
- from: "array",
483
- },
484
- [],
485
- ),
486
- },
487
- ],
488
- meta.getName(),
489
- ),
490
- });
491
-
492
- // MAPS
493
- if (meta.maps.length)
494
- if (meta.maps.every((elem) => elem.key.any && elem.value.any))
495
- binaries.push({
496
- expression: check_native("Map")(input),
497
- combined: false,
498
- });
499
- else
500
- binaries.push({
501
- combined: true,
502
- expression: config.combiner(explore)("and")(
503
- input,
504
- [
505
- {
506
- combined: false,
507
- expression: check_native("Map")(input),
508
- },
509
- {
510
- combined: true,
511
- expression: explore_map(
512
- project,
513
- config,
514
- importer,
515
- )(
516
- input,
517
- meta.maps.map((m) => [m.key, m.value]),
518
- {
519
- ...explore,
520
- from: "array",
521
- },
522
- [],
523
- ),
524
- },
525
- ],
526
- meta.getName(),
527
- ),
528
- });
529
-
530
- // COMBINE CONDITIONS
531
- return top.length && binaries.length
532
- ? config.combiner(explore)("and")(
533
- input,
534
- [
535
- ...top,
536
- {
537
- expression: config.combiner(explore)("or")(
538
- input,
539
- binaries,
540
- meta.getName(),
541
- ),
542
- combined: true,
543
- },
544
- ],
545
- meta.getName(),
546
- )
547
- : binaries.length
548
- ? config.combiner(explore)("or")(
549
- input,
550
- binaries,
551
- meta.getName(),
552
- )
553
- : config.success;
554
- };
555
- }
556
-
557
- function decode_tuple(
558
- project: IProject,
559
- config: IConfig,
560
- importer: FunctionImporter,
561
- checkLength: boolean,
562
- ) {
563
- return function (
564
- input: ts.Expression,
565
- tuple: Array<Metadata>,
566
- explore: IExplore,
567
- tagList: IMetadataTag[],
568
- ): ts.Expression {
569
- const binaries: ts.Expression[] = tuple
570
- .filter((meta) => meta.rest === null)
571
- .map((meta, index) =>
572
- decode(project, config, importer)(
573
- ts.factory.createElementAccessExpression(input, index),
574
- meta,
575
- {
576
- ...explore,
577
- from: "array",
578
- postfix: explore.postfix.length
579
- ? `${explore.postfix.slice(0, -1)}[${index}]"`
580
- : `[${index}]`,
581
- },
582
- tagList,
583
- ),
584
- );
585
- const rest: ts.Expression | null =
586
- tuple.length && tuple[tuple.length - 1]!.rest !== null
587
- ? decode(project, config, importer, false)(
588
- ts.factory.createCallExpression(
589
- IdentifierFactory.join(input, "slice"),
590
- undefined,
591
- [
592
- ts.factory.createNumericLiteral(
593
- tuple.length - 1,
594
- ),
595
- ],
596
- ),
597
- (() => {
598
- const wrapper: Metadata = Metadata.initialize();
599
- wrapper.arrays.push(
600
- tuple[tuple.length - 1]!.rest!,
601
- );
602
- return wrapper;
603
- })(),
604
- {
605
- ...explore,
606
- start: tuple.length - 1,
607
- },
608
- tagList,
609
- )
610
- : null;
611
-
612
- return config.combiner(explore)("and")(
613
- input,
614
- [
615
- ...(checkLength && rest === null
616
- ? [
617
- {
618
- combined: false,
619
- expression: ts.factory.createStrictEquality(
620
- ts.factory.createPropertyAccessExpression(
621
- input,
622
- "length",
623
- ),
624
- ts.factory.createNumericLiteral(
625
- tuple.length,
626
- ),
627
- ),
628
- },
629
- ]
630
- : []),
631
- ...(config.joiner.tuple
632
- ? [
633
- {
634
- expression: config.joiner.tuple(binaries),
635
- combined: true,
636
- },
637
- ]
638
- : binaries.map((expression) => ({
639
- expression,
640
- combined: true,
641
- }))),
642
- ...(rest !== null
643
- ? [
644
- {
645
- expression: rest,
646
- combined: true,
647
- },
648
- ]
649
- : []),
650
- ],
651
- `[${tuple.map((t) => t.getName()).join(", ")}]`,
652
- );
653
- };
654
- }
655
-
656
- function decode_array(
657
- project: IProject,
658
- config: IConfig,
659
- importer: FunctionImporter,
660
- checkTupleLength: boolean,
661
- ) {
662
- return FeatureProgrammer.decode_array(
663
- {
664
- trace: config.trace,
665
- path: config.path,
666
- decoder: decode(project, config, importer, checkTupleLength),
667
- },
668
- importer,
669
- config.joiner.array,
670
- );
671
- }
672
-
673
- export function decode_object(config: IConfig) {
674
- const func = FeatureProgrammer.decode_object(config);
675
- return function (
676
- input: ts.Expression,
677
- obj: MetadataObject,
678
- explore: IExplore,
679
- ) {
680
- obj.validated = true;
681
- return func(input, obj, explore);
682
- };
683
- }
684
-
685
- const explore_array = (
686
- project: IProject,
687
- config: IConfig,
688
- importer: FunctionImporter,
689
- ) =>
690
- UnionExplorer.array({
691
- checker: decode(project, config, importer),
692
- decoder: decode_array(project, config, importer, true),
693
- empty: config.success,
694
- success: config.success,
695
- failure: (input, expected, explore) =>
696
- ts.factory.createReturnStatement(
697
- config.joiner.failure(input, expected, explore),
698
- ),
699
- });
700
-
701
- const explore_set = (
702
- project: IProject,
703
- config: IConfig,
704
- importer: FunctionImporter,
705
- ) =>
706
- UnionExplorer.set({
707
- checker: decode(project, config, importer),
708
- decoder: decode_array(project, config, importer, true),
709
- empty: config.success,
710
- success: config.success,
711
- failure: (input, expected, explore) =>
712
- ts.factory.createReturnStatement(
713
- config.joiner.failure(input, expected, explore),
714
- ),
715
- });
716
-
717
- const explore_map = (
718
- project: IProject,
719
- config: IConfig,
720
- importer: FunctionImporter,
721
- ) =>
722
- UnionExplorer.map({
723
- checker: (input, entry, explore) => {
724
- const func = decode(project, config, importer);
725
- return ts.factory.createLogicalAnd(
726
- func(
727
- ts.factory.createElementAccessExpression(input, 0),
728
- entry[0],
729
- { ...explore, postfix: `${explore.postfix}[0]` },
730
- [],
731
- ),
732
- func(
733
- ts.factory.createElementAccessExpression(input, 1),
734
- entry[1],
735
- { ...explore, postfix: `${explore.postfix}[1]` },
736
- [],
737
- ),
738
- );
739
- },
740
- decoder: (input, target, explore) =>
741
- decode_array(project, config, importer, false)(
742
- input,
743
- Metadata.create({
744
- any: false,
745
- nullable: false,
746
- required: true,
747
- functional: false,
748
- resolved: null,
749
- constants: [],
750
- atomics: [],
751
- templates: [],
752
- rest: null,
753
- arrays: [],
754
- tuples: [target],
755
- objects: [],
756
- natives: [],
757
- sets: [],
758
- maps: [],
759
- }),
760
- explore,
761
- [],
762
- ),
763
- empty: config.success,
764
- success: config.success,
765
- failure: (input, expected, explore) =>
766
- ts.factory.createReturnStatement(
767
- config.joiner.failure(input, expected, explore),
768
- ),
769
- });
770
-
771
- const explore_objects = (config: IConfig) => {
772
- const objector = decode_object(config);
773
-
774
- return (input: ts.Expression, meta: Metadata, explore: IExplore) => {
775
- if (meta.objects.length === 1)
776
- return objector(input, meta.objects[0]!, explore);
777
-
778
- return ts.factory.createCallExpression(
779
- ts.factory.createIdentifier(
780
- `${config.unioners}${meta.union_index!}`,
781
- ),
782
- undefined,
783
- FeatureProgrammer.get_object_arguments(config)(explore)(input),
784
- );
785
- };
786
- };
787
- }
788
-
789
- const create_add =
790
- (binaries: CheckerProgrammer.IBinary[]) =>
791
- (defaultInput: ts.Expression) =>
792
- (
793
- exact: boolean,
794
- left: ts.Expression,
795
- right: ts.Expression = defaultInput,
796
- ) => {
797
- const factory = exact
798
- ? ts.factory.createStrictEquality
799
- : ts.factory.createStrictInequality;
800
- binaries.push({
801
- expression: factory(left, right),
802
- combined: false,
803
- });
804
- };
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 { ValueFactory } from "../factories/ValueFactory";
8
+
9
+ import { IMetadataTag } from "../metadata/IMetadataTag";
10
+ import { Metadata } from "../metadata/Metadata";
11
+ import { MetadataObject } from "../metadata/MetadataObject";
12
+
13
+ import { IProject } from "../transformers/IProject";
14
+
15
+ import { FeatureProgrammer } from "./FeatureProgrammer";
16
+ import { FunctionImporter } from "./helpers/FunctionImporeter";
17
+ import { IExpressionEntry } from "./helpers/IExpressionEntry";
18
+ import { OptionPredicator } from "./helpers/OptionPredicator";
19
+ import { UnionExplorer } from "./helpers/UnionExplorer";
20
+ import { check_array } from "./internal/check_array";
21
+ import { check_native } from "./internal/check_native";
22
+ import { check_number } from "./internal/check_number";
23
+ import { check_string } from "./internal/check_string";
24
+ import { check_template } from "./internal/check_template";
25
+ import { decode_union_object } from "./internal/decode_union_object";
26
+
27
+ export namespace CheckerProgrammer {
28
+ export interface IConfig {
29
+ functors: string;
30
+ unioners: string;
31
+ path: boolean;
32
+ trace: boolean;
33
+ equals: boolean;
34
+ numeric: boolean;
35
+ combiner: IConfig.Combiner;
36
+ decoder?: FeatureProgrammer.Decoder<Metadata>;
37
+ joiner: IConfig.IJoiner;
38
+ success: ts.Expression;
39
+ }
40
+ export namespace IConfig {
41
+ export interface Combiner {
42
+ (explorer: IExplore): {
43
+ (logic: "and" | "or"): {
44
+ (
45
+ input: ts.Expression,
46
+ binaries: IBinary[],
47
+ expected: string,
48
+ ): ts.Expression;
49
+ };
50
+ };
51
+ }
52
+ export interface IJoiner {
53
+ object(entries: IExpressionEntry[]): ts.Expression;
54
+ array(input: ts.Expression, arrow: ts.ArrowFunction): ts.Expression;
55
+ tuple?(exprs: ts.Expression[]): ts.Expression;
56
+
57
+ failure(
58
+ value: ts.Expression,
59
+ expected: string,
60
+ explore?: FeatureProgrammer.IExplore,
61
+ ): ts.Expression;
62
+ is?(expression: ts.Expression): ts.Expression;
63
+ required?(exp: ts.Expression): ts.Expression;
64
+ full?: (
65
+ condition: ts.Expression,
66
+ ) => (
67
+ input: ts.Expression,
68
+ expected: string,
69
+ explore: IExplore,
70
+ ) => ts.Expression;
71
+ }
72
+ }
73
+ export import IExplore = FeatureProgrammer.IExplore;
74
+
75
+ export interface IBinary {
76
+ expression: ts.Expression;
77
+ combined: boolean;
78
+ }
79
+
80
+ /* -----------------------------------------------------------
81
+ GENERATORS
82
+ ----------------------------------------------------------- */
83
+ export function generate(
84
+ project: IProject,
85
+ config: IConfig,
86
+ importer: FunctionImporter,
87
+ addition?: () => ts.Statement[],
88
+ ) {
89
+ return FeatureProgrammer.generate(
90
+ project,
91
+ CONFIG(project, config, importer),
92
+ importer,
93
+ () => (addition ? (addition ? addition() : []) : undefined),
94
+ );
95
+ }
96
+
97
+ export const generate_functors = (
98
+ project: IProject,
99
+ config: IConfig,
100
+ importer: FunctionImporter,
101
+ ) =>
102
+ FeatureProgrammer.generate_functors(
103
+ CONFIG(project, config, importer),
104
+ importer,
105
+ );
106
+
107
+ export const generate_unioners = (
108
+ project: IProject,
109
+ config: IConfig,
110
+ importer: FunctionImporter,
111
+ ) =>
112
+ FeatureProgrammer.generate_unioners(
113
+ CONFIG(project, { ...config, numeric: false }, importer),
114
+ );
115
+
116
+ function CONFIG(
117
+ project: IProject,
118
+ config: IConfig,
119
+ importer: FunctionImporter,
120
+ ): FeatureProgrammer.IConfig {
121
+ const output: FeatureProgrammer.IConfig = {
122
+ trace: config.trace,
123
+ path: config.path,
124
+ functors: config.functors,
125
+ unioners: config.unioners,
126
+ initializer: ({ checker }, type) => {
127
+ const collection: MetadataCollection = new MetadataCollection();
128
+ const meta: Metadata = MetadataFactory.generate(
129
+ checker,
130
+ collection,
131
+ type,
132
+ {
133
+ resolve: false,
134
+ constant: true,
135
+ },
136
+ );
137
+ return [collection, meta];
138
+ },
139
+ decoder: config.decoder || decode(project, config, importer),
140
+ objector: {
141
+ checker: config.decoder || decode(project, config, importer),
142
+ decoder: decode_object(config),
143
+ joiner: config.joiner.object,
144
+ unionizer: config.equals
145
+ ? decode_union_object(decode_object(config))(
146
+ (input, obj, explore) =>
147
+ decode_object(config)(input, obj, {
148
+ ...explore,
149
+ tracable: true,
150
+ }),
151
+ )(config.joiner.is || ((expr) => expr))(
152
+ (value, expected) =>
153
+ ts.factory.createReturnStatement(
154
+ config.joiner.failure(value, expected),
155
+ ),
156
+ )
157
+ : (input, targets, explore) =>
158
+ config.combiner(explore)("or")(
159
+ input,
160
+ targets.map((obj) => ({
161
+ expression: decode_object(config)(
162
+ input,
163
+ obj,
164
+ explore,
165
+ ),
166
+ combined: true,
167
+ })),
168
+ `(${targets.map((t) => t.name).join(" | ")})`,
169
+ ),
170
+ failure: (value, expected) =>
171
+ ts.factory.createReturnStatement(
172
+ config.joiner.failure(value, expected),
173
+ ),
174
+ is: config.joiner.is,
175
+ required: config.joiner.required,
176
+ full: config.joiner.full,
177
+ },
178
+ };
179
+ if (config.numeric === true)
180
+ output.generator = {
181
+ unioners: FeatureProgrammer.generate_unioners(
182
+ CONFIG(project, { ...config, numeric: false }, importer),
183
+ ),
184
+ };
185
+ return output;
186
+ }
187
+
188
+ /* -----------------------------------------------------------
189
+ DECODERS
190
+ ----------------------------------------------------------- */
191
+ export function decode(
192
+ project: IProject,
193
+ config: IConfig,
194
+ importer: FunctionImporter,
195
+ ): (
196
+ input: ts.Expression,
197
+ meta: Metadata,
198
+ explore: IExplore,
199
+ tags: IMetadataTag[],
200
+ ) => ts.Expression;
201
+
202
+ /**
203
+ * @internal
204
+ */
205
+ export function decode(
206
+ project: IProject,
207
+ config: IConfig,
208
+ importer: FunctionImporter,
209
+ checkTupleLength: boolean,
210
+ ): (
211
+ input: ts.Expression,
212
+ meta: Metadata,
213
+ explore: IExplore,
214
+ tags: IMetadataTag[],
215
+ ) => ts.Expression;
216
+
217
+ /**
218
+ * @internal
219
+ */
220
+ export function decode(
221
+ project: IProject,
222
+ config: IConfig,
223
+ importer: FunctionImporter,
224
+ checkTupleLength: boolean = true,
225
+ ) {
226
+ return function (
227
+ input: ts.Expression,
228
+ meta: Metadata,
229
+ explore: IExplore,
230
+ tags: IMetadataTag[],
231
+ ): ts.Expression {
232
+ if (meta.any) return config.success;
233
+
234
+ const top: IBinary[] = [];
235
+ const binaries: IBinary[] = [];
236
+ const add = create_add(binaries)(input);
237
+ const getConstantValue = (
238
+ value: number | string | bigint | boolean,
239
+ ) =>
240
+ typeof value === "string"
241
+ ? ts.factory.createStringLiteral(value)
242
+ : ts.factory.createIdentifier(value.toString());
243
+
244
+ //----
245
+ // CHECK OPTIONAL
246
+ //----
247
+ const checkOptional: boolean = meta.empty() || meta.isUnionBucket();
248
+
249
+ // NULLABLE
250
+ if (
251
+ checkOptional ||
252
+ meta.nullable
253
+ // || (meta.objects.length && meta.size() !== meta.objects.length)
254
+ )
255
+ (meta.nullable ? add : create_add(top)(input))(
256
+ meta.nullable,
257
+ ValueFactory.NULL(),
258
+ );
259
+
260
+ // UNDEFINDABLE
261
+ if (checkOptional || !meta.required)
262
+ (meta.required ? create_add(top)(input) : add)(
263
+ !meta.required,
264
+ ValueFactory.UNDEFINED(),
265
+ );
266
+
267
+ // FUNCTIONAL
268
+ if (meta.functional === true)
269
+ if (
270
+ OptionPredicator.functional(project.options) ||
271
+ meta.size() !== 1
272
+ )
273
+ add(
274
+ true,
275
+ ts.factory.createStringLiteral("function"),
276
+ ValueFactory.TYPEOF(input),
277
+ );
278
+ else
279
+ binaries.push({
280
+ combined: false,
281
+ expression: config.success,
282
+ });
283
+
284
+ //----
285
+ // VALUES
286
+ //----
287
+ // CONSTANT VALUES
288
+ for (const constant of meta.constants)
289
+ for (const val of constant.values)
290
+ add(true, getConstantValue(val));
291
+
292
+ // ATOMIC VALUES
293
+ for (const type of meta.atomics)
294
+ if (type === "number")
295
+ binaries.push({
296
+ expression: check_number(project, config.numeric)(
297
+ input,
298
+ tags,
299
+ ),
300
+ combined: false,
301
+ });
302
+ else if (type === "string")
303
+ binaries.push({
304
+ expression: check_string(importer)(input, tags),
305
+ combined: false,
306
+ });
307
+ else
308
+ add(
309
+ true,
310
+ ts.factory.createStringLiteral(type),
311
+ ValueFactory.TYPEOF(input),
312
+ );
313
+
314
+ // TEMPLATE LITERAL VALUES
315
+ if (meta.templates.length)
316
+ binaries.push({
317
+ expression: check_template(importer)(
318
+ input,
319
+ meta.templates,
320
+ tags,
321
+ ),
322
+ combined: false,
323
+ });
324
+
325
+ //----
326
+ // INSTANCES
327
+ //----
328
+ // TUPLE
329
+ if (meta.tuples.length > 0) {
330
+ const inner: ts.Expression[] = [];
331
+ for (const tuple of meta.tuples)
332
+ inner.push(
333
+ decode_tuple(
334
+ project,
335
+ config,
336
+ importer,
337
+ checkTupleLength,
338
+ )(input, tuple, explore, tags),
339
+ );
340
+
341
+ // ADD
342
+ binaries.push({
343
+ expression: config.combiner(explore)("and")(
344
+ input,
345
+ [
346
+ ...(checkTupleLength
347
+ ? [
348
+ {
349
+ expression:
350
+ ExpressionFactory.isArray(input),
351
+ combined: false,
352
+ },
353
+ ]
354
+ : []),
355
+ ...inner.map((expression) => ({
356
+ expression,
357
+ combined: true,
358
+ })),
359
+ ],
360
+ meta.getName(),
361
+ ),
362
+ combined: true,
363
+ });
364
+ }
365
+
366
+ // ARRAY
367
+ if (meta.arrays.length > 0)
368
+ if (meta.arrays.every((elem) => elem.any))
369
+ binaries.push({
370
+ expression: check_array(input, tags),
371
+ combined: false,
372
+ });
373
+ else
374
+ binaries.push({
375
+ expression: config.combiner(explore)("and")(
376
+ input,
377
+ [
378
+ ...(checkTupleLength
379
+ ? [
380
+ {
381
+ expression: check_array(
382
+ input,
383
+ tags,
384
+ ),
385
+ combined: false,
386
+ },
387
+ ]
388
+ : []),
389
+ {
390
+ expression: explore_array(
391
+ project,
392
+ config,
393
+ importer,
394
+ )(
395
+ input,
396
+ meta.arrays,
397
+ {
398
+ ...explore,
399
+ from: "array",
400
+ },
401
+ tags,
402
+ ),
403
+ combined: true,
404
+ },
405
+ ],
406
+ meta.getName(),
407
+ ),
408
+ combined: true,
409
+ });
410
+
411
+ // OBJECT
412
+ if (meta.objects.length > 0)
413
+ binaries.push({
414
+ expression: config.combiner(explore)("and")(
415
+ input,
416
+ [
417
+ {
418
+ expression: ExpressionFactory.isObject(input, {
419
+ checkNull: true,
420
+ checkArray: meta.objects.some((obj) =>
421
+ obj.properties.every(
422
+ (prop) =>
423
+ !prop.key.isSoleLiteral() ||
424
+ !prop.value.required,
425
+ ),
426
+ ),
427
+ }),
428
+ combined: false,
429
+ },
430
+ {
431
+ expression: explore_objects(config)(
432
+ input,
433
+ meta,
434
+ {
435
+ ...explore,
436
+ from: "object",
437
+ },
438
+ ),
439
+ combined: true,
440
+ },
441
+ ],
442
+ meta.getName(),
443
+ ),
444
+ combined: true,
445
+ });
446
+
447
+ // NATIVE CLASSES
448
+ for (const native of meta.natives)
449
+ binaries.push({
450
+ expression: check_native(native)(input),
451
+ combined: false,
452
+ });
453
+
454
+ // SETS
455
+ if (meta.sets.length)
456
+ if (meta.sets.every((elem) => elem.any))
457
+ binaries.push({
458
+ combined: false,
459
+ expression: check_native("Set")(input),
460
+ });
461
+ else
462
+ binaries.push({
463
+ combined: true,
464
+ expression: config.combiner(explore)("and")(
465
+ input,
466
+ [
467
+ {
468
+ combined: false,
469
+ expression: check_native("Set")(input),
470
+ },
471
+ {
472
+ combined: true,
473
+ expression: explore_set(
474
+ project,
475
+ config,
476
+ importer,
477
+ )(
478
+ input,
479
+ meta.sets,
480
+ {
481
+ ...explore,
482
+ from: "array",
483
+ },
484
+ [],
485
+ ),
486
+ },
487
+ ],
488
+ meta.getName(),
489
+ ),
490
+ });
491
+
492
+ // MAPS
493
+ if (meta.maps.length)
494
+ if (meta.maps.every((elem) => elem.key.any && elem.value.any))
495
+ binaries.push({
496
+ expression: check_native("Map")(input),
497
+ combined: false,
498
+ });
499
+ else
500
+ binaries.push({
501
+ combined: true,
502
+ expression: config.combiner(explore)("and")(
503
+ input,
504
+ [
505
+ {
506
+ combined: false,
507
+ expression: check_native("Map")(input),
508
+ },
509
+ {
510
+ combined: true,
511
+ expression: explore_map(
512
+ project,
513
+ config,
514
+ importer,
515
+ )(
516
+ input,
517
+ meta.maps.map((m) => [m.key, m.value]),
518
+ {
519
+ ...explore,
520
+ from: "array",
521
+ },
522
+ [],
523
+ ),
524
+ },
525
+ ],
526
+ meta.getName(),
527
+ ),
528
+ });
529
+
530
+ // COMBINE CONDITIONS
531
+ return top.length && binaries.length
532
+ ? config.combiner(explore)("and")(
533
+ input,
534
+ [
535
+ ...top,
536
+ {
537
+ expression: config.combiner(explore)("or")(
538
+ input,
539
+ binaries,
540
+ meta.getName(),
541
+ ),
542
+ combined: true,
543
+ },
544
+ ],
545
+ meta.getName(),
546
+ )
547
+ : binaries.length
548
+ ? config.combiner(explore)("or")(
549
+ input,
550
+ binaries,
551
+ meta.getName(),
552
+ )
553
+ : config.success;
554
+ };
555
+ }
556
+
557
+ function decode_tuple(
558
+ project: IProject,
559
+ config: IConfig,
560
+ importer: FunctionImporter,
561
+ checkLength: boolean,
562
+ ) {
563
+ return function (
564
+ input: ts.Expression,
565
+ tuple: Array<Metadata>,
566
+ explore: IExplore,
567
+ tagList: IMetadataTag[],
568
+ ): ts.Expression {
569
+ const binaries: ts.Expression[] = tuple
570
+ .filter((meta) => meta.rest === null)
571
+ .map((meta, index) =>
572
+ decode(project, config, importer)(
573
+ ts.factory.createElementAccessExpression(input, index),
574
+ meta,
575
+ {
576
+ ...explore,
577
+ from: "array",
578
+ postfix: explore.postfix.length
579
+ ? `${explore.postfix.slice(0, -1)}[${index}]"`
580
+ : `[${index}]`,
581
+ },
582
+ tagList,
583
+ ),
584
+ );
585
+ const rest: ts.Expression | null =
586
+ tuple.length && tuple[tuple.length - 1]!.rest !== null
587
+ ? decode(project, config, importer, false)(
588
+ ts.factory.createCallExpression(
589
+ IdentifierFactory.join(input, "slice"),
590
+ undefined,
591
+ [
592
+ ts.factory.createNumericLiteral(
593
+ tuple.length - 1,
594
+ ),
595
+ ],
596
+ ),
597
+ (() => {
598
+ const wrapper: Metadata = Metadata.initialize();
599
+ wrapper.arrays.push(
600
+ tuple[tuple.length - 1]!.rest!,
601
+ );
602
+ return wrapper;
603
+ })(),
604
+ {
605
+ ...explore,
606
+ start: tuple.length - 1,
607
+ },
608
+ tagList,
609
+ )
610
+ : null;
611
+
612
+ return config.combiner(explore)("and")(
613
+ input,
614
+ [
615
+ ...(checkLength && rest === null
616
+ ? [
617
+ {
618
+ combined: false,
619
+ expression: ts.factory.createStrictEquality(
620
+ ts.factory.createPropertyAccessExpression(
621
+ input,
622
+ "length",
623
+ ),
624
+ ts.factory.createNumericLiteral(
625
+ tuple.length,
626
+ ),
627
+ ),
628
+ },
629
+ ]
630
+ : []),
631
+ ...(config.joiner.tuple
632
+ ? [
633
+ {
634
+ expression: config.joiner.tuple(binaries),
635
+ combined: true,
636
+ },
637
+ ]
638
+ : binaries.map((expression) => ({
639
+ expression,
640
+ combined: true,
641
+ }))),
642
+ ...(rest !== null
643
+ ? [
644
+ {
645
+ expression: rest,
646
+ combined: true,
647
+ },
648
+ ]
649
+ : []),
650
+ ],
651
+ `[${tuple.map((t) => t.getName()).join(", ")}]`,
652
+ );
653
+ };
654
+ }
655
+
656
+ function decode_array(
657
+ project: IProject,
658
+ config: IConfig,
659
+ importer: FunctionImporter,
660
+ checkTupleLength: boolean,
661
+ ) {
662
+ return FeatureProgrammer.decode_array(
663
+ {
664
+ trace: config.trace,
665
+ path: config.path,
666
+ decoder: decode(project, config, importer, checkTupleLength),
667
+ },
668
+ importer,
669
+ config.joiner.array,
670
+ );
671
+ }
672
+
673
+ export function decode_object(config: IConfig) {
674
+ const func = FeatureProgrammer.decode_object(config);
675
+ return function (
676
+ input: ts.Expression,
677
+ obj: MetadataObject,
678
+ explore: IExplore,
679
+ ) {
680
+ obj.validated = true;
681
+ return func(input, obj, explore);
682
+ };
683
+ }
684
+
685
+ const explore_array = (
686
+ project: IProject,
687
+ config: IConfig,
688
+ importer: FunctionImporter,
689
+ ) =>
690
+ UnionExplorer.array({
691
+ checker: decode(project, config, importer),
692
+ decoder: decode_array(project, config, importer, true),
693
+ empty: config.success,
694
+ success: config.success,
695
+ failure: (input, expected, explore) =>
696
+ ts.factory.createReturnStatement(
697
+ config.joiner.failure(input, expected, explore),
698
+ ),
699
+ });
700
+
701
+ const explore_set = (
702
+ project: IProject,
703
+ config: IConfig,
704
+ importer: FunctionImporter,
705
+ ) =>
706
+ UnionExplorer.set({
707
+ checker: decode(project, config, importer),
708
+ decoder: decode_array(project, config, importer, true),
709
+ empty: config.success,
710
+ success: config.success,
711
+ failure: (input, expected, explore) =>
712
+ ts.factory.createReturnStatement(
713
+ config.joiner.failure(input, expected, explore),
714
+ ),
715
+ });
716
+
717
+ const explore_map = (
718
+ project: IProject,
719
+ config: IConfig,
720
+ importer: FunctionImporter,
721
+ ) =>
722
+ UnionExplorer.map({
723
+ checker: (input, entry, explore) => {
724
+ const func = decode(project, config, importer);
725
+ return ts.factory.createLogicalAnd(
726
+ func(
727
+ ts.factory.createElementAccessExpression(input, 0),
728
+ entry[0],
729
+ { ...explore, postfix: `${explore.postfix}[0]` },
730
+ [],
731
+ ),
732
+ func(
733
+ ts.factory.createElementAccessExpression(input, 1),
734
+ entry[1],
735
+ { ...explore, postfix: `${explore.postfix}[1]` },
736
+ [],
737
+ ),
738
+ );
739
+ },
740
+ decoder: (input, target, explore) =>
741
+ decode_array(project, config, importer, false)(
742
+ input,
743
+ Metadata.create({
744
+ any: false,
745
+ nullable: false,
746
+ required: true,
747
+ functional: false,
748
+ resolved: null,
749
+ constants: [],
750
+ atomics: [],
751
+ templates: [],
752
+ rest: null,
753
+ arrays: [],
754
+ tuples: [target],
755
+ objects: [],
756
+ natives: [],
757
+ sets: [],
758
+ maps: [],
759
+ }),
760
+ explore,
761
+ [],
762
+ ),
763
+ empty: config.success,
764
+ success: config.success,
765
+ failure: (input, expected, explore) =>
766
+ ts.factory.createReturnStatement(
767
+ config.joiner.failure(input, expected, explore),
768
+ ),
769
+ });
770
+
771
+ const explore_objects = (config: IConfig) => {
772
+ const objector = decode_object(config);
773
+
774
+ return (input: ts.Expression, meta: Metadata, explore: IExplore) => {
775
+ if (meta.objects.length === 1)
776
+ return objector(input, meta.objects[0]!, explore);
777
+
778
+ return ts.factory.createCallExpression(
779
+ ts.factory.createIdentifier(
780
+ `${config.unioners}${meta.union_index!}`,
781
+ ),
782
+ undefined,
783
+ FeatureProgrammer.get_object_arguments(config)(explore)(input),
784
+ );
785
+ };
786
+ };
787
+ }
788
+
789
+ const create_add =
790
+ (binaries: CheckerProgrammer.IBinary[]) =>
791
+ (defaultInput: ts.Expression) =>
792
+ (
793
+ exact: boolean,
794
+ left: ts.Expression,
795
+ right: ts.Expression = defaultInput,
796
+ ) => {
797
+ const factory = exact
798
+ ? ts.factory.createStrictEquality
799
+ : ts.factory.createStrictInequality;
800
+ binaries.push({
801
+ expression: factory(left, right),
802
+ combined: false,
803
+ });
804
+ };