typia 7.0.0-dev.20241007 → 7.0.0-dev.20241008

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 (240) hide show
  1. package/lib/factories/ExpressionFactory.d.ts +1 -1
  2. package/lib/factories/ValueFactory.d.ts +1 -1
  3. package/lib/programmers/AssertProgrammer.d.ts +5 -2
  4. package/lib/programmers/AssertProgrammer.js +12 -5
  5. package/lib/programmers/AssertProgrammer.js.map +1 -1
  6. package/lib/programmers/ImportProgrammer.d.ts +25 -4
  7. package/lib/programmers/ImportProgrammer.js +67 -31
  8. package/lib/programmers/ImportProgrammer.js.map +1 -1
  9. package/lib/programmers/RandomProgrammer.js +48 -32
  10. package/lib/programmers/RandomProgrammer.js.map +1 -1
  11. package/lib/programmers/ValidateProgrammer.js +10 -4
  12. package/lib/programmers/ValidateProgrammer.js.map +1 -1
  13. package/lib/programmers/functional/FunctionalAssertFunctionProgrammer.js +1 -1
  14. package/lib/programmers/functional/FunctionalAssertFunctionProgrammer.js.map +1 -1
  15. package/lib/programmers/functional/FunctionalValidateFunctionProgrammer.d.ts +1 -0
  16. package/lib/programmers/functional/FunctionalValidateFunctionProgrammer.js +14 -5
  17. package/lib/programmers/functional/FunctionalValidateFunctionProgrammer.js.map +1 -1
  18. package/lib/programmers/functional/FunctionalValidateParametersProgrammer.js +5 -1
  19. package/lib/programmers/functional/FunctionalValidateParametersProgrammer.js.map +1 -1
  20. package/lib/programmers/functional/FunctionalValidateReturnProgrammer.js +1 -0
  21. package/lib/programmers/functional/FunctionalValidateReturnProgrammer.js.map +1 -1
  22. package/lib/programmers/http/HttpAssertFormDataProgrammer.js +4 -1
  23. package/lib/programmers/http/HttpAssertFormDataProgrammer.js.map +1 -1
  24. package/lib/programmers/http/HttpAssertHeadersProgrammer.js +4 -1
  25. package/lib/programmers/http/HttpAssertHeadersProgrammer.js.map +1 -1
  26. package/lib/programmers/http/HttpAssertQueryProgrammer.js +4 -1
  27. package/lib/programmers/http/HttpAssertQueryProgrammer.js.map +1 -1
  28. package/lib/programmers/http/HttpFormDataProgrammer.js +10 -6
  29. package/lib/programmers/http/HttpFormDataProgrammer.js.map +1 -1
  30. package/lib/programmers/http/HttpHeadersProgrammer.js +10 -6
  31. package/lib/programmers/http/HttpHeadersProgrammer.js.map +1 -1
  32. package/lib/programmers/http/HttpQueryProgrammer.js +10 -6
  33. package/lib/programmers/http/HttpQueryProgrammer.js.map +1 -1
  34. package/lib/programmers/http/HttpValidateFormDataProgrammer.js +5 -3
  35. package/lib/programmers/http/HttpValidateFormDataProgrammer.js.map +1 -1
  36. package/lib/programmers/http/HttpValidateHeadersProgrammer.js +5 -3
  37. package/lib/programmers/http/HttpValidateHeadersProgrammer.js.map +1 -1
  38. package/lib/programmers/http/HttpValidateQueryProgrammer.js +5 -3
  39. package/lib/programmers/http/HttpValidateQueryProgrammer.js.map +1 -1
  40. package/lib/programmers/json/JsonAssertParseProgrammer.js +14 -7
  41. package/lib/programmers/json/JsonAssertParseProgrammer.js.map +1 -1
  42. package/lib/programmers/json/JsonAssertStringifyProgrammer.js +4 -1
  43. package/lib/programmers/json/JsonAssertStringifyProgrammer.js.map +1 -1
  44. package/lib/programmers/json/JsonIsParseProgrammer.js +10 -6
  45. package/lib/programmers/json/JsonIsParseProgrammer.js.map +1 -1
  46. package/lib/programmers/json/JsonValidateParseProgrammer.js +16 -4
  47. package/lib/programmers/json/JsonValidateParseProgrammer.js.map +1 -1
  48. package/lib/programmers/json/JsonValidateStringifyProgrammer.js +7 -3
  49. package/lib/programmers/json/JsonValidateStringifyProgrammer.js.map +1 -1
  50. package/lib/programmers/misc/MiscAssertCloneProgrammer.js +4 -1
  51. package/lib/programmers/misc/MiscAssertCloneProgrammer.js.map +1 -1
  52. package/lib/programmers/misc/MiscAssertPruneProgrammer.js +4 -1
  53. package/lib/programmers/misc/MiscAssertPruneProgrammer.js.map +1 -1
  54. package/lib/programmers/misc/MiscCloneProgrammer.js +10 -6
  55. package/lib/programmers/misc/MiscCloneProgrammer.js.map +1 -1
  56. package/lib/programmers/misc/MiscValidateCloneProgrammer.js +5 -3
  57. package/lib/programmers/misc/MiscValidateCloneProgrammer.js.map +1 -1
  58. package/lib/programmers/misc/MiscValidatePruneProgrammer.js +10 -4
  59. package/lib/programmers/misc/MiscValidatePruneProgrammer.js.map +1 -1
  60. package/lib/programmers/notations/NotationAssertGeneralProgrammer.js +4 -1
  61. package/lib/programmers/notations/NotationAssertGeneralProgrammer.js.map +1 -1
  62. package/lib/programmers/notations/NotationGeneralProgrammer.d.ts +2 -1
  63. package/lib/programmers/notations/NotationGeneralProgrammer.js +13 -5
  64. package/lib/programmers/notations/NotationGeneralProgrammer.js.map +1 -1
  65. package/lib/programmers/notations/NotationValidateGeneralProgrammer.js +5 -3
  66. package/lib/programmers/notations/NotationValidateGeneralProgrammer.js.map +1 -1
  67. package/lib/programmers/protobuf/ProtobufAssertDecodeProgrammer.js +4 -1
  68. package/lib/programmers/protobuf/ProtobufAssertDecodeProgrammer.js.map +1 -1
  69. package/lib/programmers/protobuf/ProtobufAssertEncodeProgrammer.js +4 -1
  70. package/lib/programmers/protobuf/ProtobufAssertEncodeProgrammer.js.map +1 -1
  71. package/lib/programmers/protobuf/ProtobufDecodeProgrammer.js +10 -6
  72. package/lib/programmers/protobuf/ProtobufDecodeProgrammer.js.map +1 -1
  73. package/lib/programmers/protobuf/ProtobufValidateDecodeProgrammer.js +5 -3
  74. package/lib/programmers/protobuf/ProtobufValidateDecodeProgrammer.js.map +1 -1
  75. package/lib/programmers/protobuf/ProtobufValidateEncodeProgrammer.js +7 -3
  76. package/lib/programmers/protobuf/ProtobufValidateEncodeProgrammer.js.map +1 -1
  77. package/package.json +1 -1
  78. package/src/factories/JsonMetadataFactory.ts +63 -63
  79. package/src/factories/ProtobufFactory.ts +275 -275
  80. package/src/programmers/AssertProgrammer.ts +454 -456
  81. package/src/programmers/CheckerProgrammer.ts +1610 -1610
  82. package/src/programmers/ImportProgrammer.ts +184 -109
  83. package/src/programmers/IsProgrammer.ts +273 -273
  84. package/src/programmers/RandomProgrammer.ts +168 -149
  85. package/src/programmers/TypiaProgrammer.ts +171 -171
  86. package/src/programmers/ValidateProgrammer.ts +434 -430
  87. package/src/programmers/functional/FunctionalAssertFunctionProgrammer.ts +1 -1
  88. package/src/programmers/functional/FunctionalValidateFunctionProgrammer.ts +12 -16
  89. package/src/programmers/functional/FunctionalValidateParametersProgrammer.ts +5 -12
  90. package/src/programmers/functional/FunctionalValidateReturnProgrammer.ts +1 -0
  91. package/src/programmers/helpers/OptionPredicator.ts +15 -15
  92. package/src/programmers/http/HttpAssertFormDataProgrammer.ts +99 -96
  93. package/src/programmers/http/HttpAssertHeadersProgrammer.ts +99 -96
  94. package/src/programmers/http/HttpAssertQueryProgrammer.ts +105 -102
  95. package/src/programmers/http/HttpFormDataProgrammer.ts +304 -308
  96. package/src/programmers/http/HttpHeadersProgrammer.ts +400 -404
  97. package/src/programmers/http/HttpIsFormDataProgrammer.ts +108 -108
  98. package/src/programmers/http/HttpIsHeadersProgrammer.ts +108 -108
  99. package/src/programmers/http/HttpIsQueryProgrammer.ts +114 -114
  100. package/src/programmers/http/HttpParameterProgrammer.ts +115 -115
  101. package/src/programmers/http/HttpQueryProgrammer.ts +329 -333
  102. package/src/programmers/http/HttpValidateFormDataProgrammer.ts +92 -90
  103. package/src/programmers/http/HttpValidateHeadersProgrammer.ts +92 -90
  104. package/src/programmers/http/HttpValidateQueryProgrammer.ts +98 -96
  105. package/src/programmers/json/JsonApplicationProgrammer.ts +92 -92
  106. package/src/programmers/json/JsonAssertParseProgrammer.ts +103 -104
  107. package/src/programmers/json/JsonAssertStringifyProgrammer.ts +115 -112
  108. package/src/programmers/json/JsonIsParseProgrammer.ts +114 -118
  109. package/src/programmers/json/JsonIsStringifyProgrammer.ts +108 -108
  110. package/src/programmers/json/JsonStringifyProgrammer.ts +1124 -1124
  111. package/src/programmers/json/JsonValidateParseProgrammer.ts +105 -95
  112. package/src/programmers/json/JsonValidateStringifyProgrammer.ts +124 -119
  113. package/src/programmers/misc/MiscAssertCloneProgrammer.ts +95 -92
  114. package/src/programmers/misc/MiscAssertPruneProgrammer.ts +116 -113
  115. package/src/programmers/misc/MiscCloneProgrammer.ts +1025 -1029
  116. package/src/programmers/misc/MiscIsCloneProgrammer.ts +99 -99
  117. package/src/programmers/misc/MiscIsPruneProgrammer.ts +97 -97
  118. package/src/programmers/misc/MiscLiteralsProgrammer.ts +80 -80
  119. package/src/programmers/misc/MiscPruneProgrammer.ts +725 -725
  120. package/src/programmers/misc/MiscValidateCloneProgrammer.ts +111 -109
  121. package/src/programmers/misc/MiscValidatePruneProgrammer.ts +113 -109
  122. package/src/programmers/notations/NotationAssertGeneralProgrammer.ts +101 -98
  123. package/src/programmers/notations/NotationGeneralProgrammer.ts +977 -973
  124. package/src/programmers/notations/NotationIsGeneralProgrammer.ts +105 -105
  125. package/src/programmers/notations/NotationValidateGeneralProgrammer.ts +119 -117
  126. package/src/programmers/protobuf/ProtobufAssertDecodeProgrammer.ts +98 -95
  127. package/src/programmers/protobuf/ProtobufAssertEncodeProgrammer.ts +102 -99
  128. package/src/programmers/protobuf/ProtobufDecodeProgrammer.ts +708 -711
  129. package/src/programmers/protobuf/ProtobufEncodeProgrammer.ts +1019 -1019
  130. package/src/programmers/protobuf/ProtobufIsDecodeProgrammer.ts +109 -109
  131. package/src/programmers/protobuf/ProtobufIsEncodeProgrammer.ts +98 -98
  132. package/src/programmers/protobuf/ProtobufValidateDecodeProgrammer.ts +92 -90
  133. package/src/programmers/protobuf/ProtobufValidateEncodeProgrammer.ts +119 -114
  134. package/src/transform.ts +35 -35
  135. package/src/transformers/CallExpressionTransformer.ts +540 -540
  136. package/src/transformers/FileTransformer.ts +120 -120
  137. package/src/transformers/IProgrammerProps.ts +11 -11
  138. package/src/transformers/ITransformOptions.ts +62 -62
  139. package/src/transformers/ITransformProps.ts +9 -9
  140. package/src/transformers/ITypiaContext.ts +18 -18
  141. package/src/transformers/ImportTransformer.ts +66 -66
  142. package/src/transformers/NodeTransformer.ts +17 -17
  143. package/src/transformers/TransformerError.ts +59 -59
  144. package/src/transformers/features/AssertTransformer.ts +24 -24
  145. package/src/transformers/features/CreateAssertTransformer.ts +24 -24
  146. package/src/transformers/features/CreateIsTransformer.ts +18 -18
  147. package/src/transformers/features/CreateRandomTransformer.ts +43 -43
  148. package/src/transformers/features/CreateValidateTransformer.ts +18 -18
  149. package/src/transformers/features/IsTransformer.ts +18 -18
  150. package/src/transformers/features/RandomTransformer.ts +41 -41
  151. package/src/transformers/features/ValidateTransformer.ts +18 -18
  152. package/src/transformers/features/functional/FunctionalGenericTransformer.ts +57 -57
  153. package/src/transformers/features/http/CreateHttpAssertFormDataTransformer.ts +13 -13
  154. package/src/transformers/features/http/CreateHttpAssertHeadersTransformer.ts +13 -13
  155. package/src/transformers/features/http/CreateHttpAssertQueryTransformer.ts +13 -13
  156. package/src/transformers/features/http/CreateHttpFormDataTransformer.ts +13 -13
  157. package/src/transformers/features/http/CreateHttpHeadersTransformer.ts +13 -13
  158. package/src/transformers/features/http/CreateHttpIsFormDataTransformer.ts +13 -13
  159. package/src/transformers/features/http/CreateHttpIsHeadersTransformer.ts +13 -13
  160. package/src/transformers/features/http/CreateHttpIsQueryTransformer.ts +13 -13
  161. package/src/transformers/features/http/CreateHttpParameterTransformer.ts +13 -13
  162. package/src/transformers/features/http/CreateHttpQueryTransformer.ts +13 -13
  163. package/src/transformers/features/http/CreateHttpValidateFormDataTransformer.ts +13 -13
  164. package/src/transformers/features/http/CreateHttpValidateHeadersTransformer.ts +13 -13
  165. package/src/transformers/features/http/CreateHttpValidateQueryTransformer.ts +13 -13
  166. package/src/transformers/features/http/HttpAssertFormDataTransformer.ts +13 -13
  167. package/src/transformers/features/http/HttpAssertHeadersTransformer.ts +13 -13
  168. package/src/transformers/features/http/HttpAssertQueryTransformer.ts +13 -13
  169. package/src/transformers/features/http/HttpFormDataTransformer.ts +13 -13
  170. package/src/transformers/features/http/HttpHeadersTransformer.ts +13 -13
  171. package/src/transformers/features/http/HttpIsFormDataTransformer.ts +13 -13
  172. package/src/transformers/features/http/HttpIsHeadersTransformer.ts +13 -13
  173. package/src/transformers/features/http/HttpIsQueryTransformer.ts +13 -13
  174. package/src/transformers/features/http/HttpParameterTransformer.ts +13 -13
  175. package/src/transformers/features/http/HttpQueryTransformer.ts +13 -13
  176. package/src/transformers/features/http/HttpValidateFormDataTransformer.ts +13 -13
  177. package/src/transformers/features/http/HttpValidateHeadersTransformer.ts +13 -13
  178. package/src/transformers/features/http/HttpValidateQueryTransformer.ts +13 -13
  179. package/src/transformers/features/json/JsonApplicationTransformer.ts +130 -130
  180. package/src/transformers/features/json/JsonAssertParseTransformer.ts +13 -13
  181. package/src/transformers/features/json/JsonAssertStringifyTransformer.ts +13 -13
  182. package/src/transformers/features/json/JsonCreateAssertParseTransformer.ts +13 -13
  183. package/src/transformers/features/json/JsonCreateAssertStringifyTransformer.ts +13 -13
  184. package/src/transformers/features/json/JsonCreateIsParseTransformer.ts +13 -13
  185. package/src/transformers/features/json/JsonCreateIsStringifyTransformer.ts +13 -13
  186. package/src/transformers/features/json/JsonCreateStringifyTransformer.ts +13 -13
  187. package/src/transformers/features/json/JsonCreateValidateParseTransformer.ts +13 -13
  188. package/src/transformers/features/json/JsonCreateValidateStringifyProgrammer.ts +13 -13
  189. package/src/transformers/features/json/JsonIsParseTransformer.ts +13 -13
  190. package/src/transformers/features/json/JsonIsStringifyTransformer.ts +13 -13
  191. package/src/transformers/features/json/JsonStringifyTransformer.ts +13 -13
  192. package/src/transformers/features/json/JsonValidateParseTransformer.ts +13 -13
  193. package/src/transformers/features/json/JsonValidateStringifyTransformer.ts +13 -13
  194. package/src/transformers/features/llm/LlmApplicationTransformer.ts +86 -86
  195. package/src/transformers/features/llm/LlmSchemaTransformer.ts +59 -59
  196. package/src/transformers/features/misc/MiscAssertCloneTransformer.ts +13 -13
  197. package/src/transformers/features/misc/MiscAssertPruneTransformer.ts +13 -13
  198. package/src/transformers/features/misc/MiscCloneTransformer.ts +13 -13
  199. package/src/transformers/features/misc/MiscCreateAssertCloneTransformer.ts +13 -13
  200. package/src/transformers/features/misc/MiscCreateAssertPruneTransformer.ts +13 -13
  201. package/src/transformers/features/misc/MiscCreateCloneTransformer.ts +13 -13
  202. package/src/transformers/features/misc/MiscCreateIsCloneTransformer.ts +13 -13
  203. package/src/transformers/features/misc/MiscCreateIsPruneTransformer.ts +13 -13
  204. package/src/transformers/features/misc/MiscCreatePruneTransformer.ts +13 -13
  205. package/src/transformers/features/misc/MiscCreateValidateCloneTransformer.ts +13 -13
  206. package/src/transformers/features/misc/MiscCreateValidatePruneTransformer.ts +13 -13
  207. package/src/transformers/features/misc/MiscIsCloneTransformer.ts +13 -13
  208. package/src/transformers/features/misc/MiscIsPruneTransformer.ts +13 -13
  209. package/src/transformers/features/misc/MiscLiteralsTransformer.ts +35 -35
  210. package/src/transformers/features/misc/MiscPruneTransformer.ts +13 -13
  211. package/src/transformers/features/misc/MiscValidateCloneTransformer.ts +13 -13
  212. package/src/transformers/features/misc/MiscValidatePruneTransformer.ts +13 -13
  213. package/src/transformers/features/notations/NotationAssertGeneralTransformer.ts +20 -20
  214. package/src/transformers/features/notations/NotationCreateAssertGeneralTransformer.ts +20 -20
  215. package/src/transformers/features/notations/NotationCreateGeneralTransformer.ts +20 -20
  216. package/src/transformers/features/notations/NotationCreateIsGeneralTransformer.ts +20 -20
  217. package/src/transformers/features/notations/NotationCreateValidateGeneralTransformer.ts +20 -20
  218. package/src/transformers/features/notations/NotationGeneralTransformer.ts +18 -18
  219. package/src/transformers/features/notations/NotationIsGeneralTransformer.ts +20 -20
  220. package/src/transformers/features/notations/NotationValidateGeneralTransformer.ts +20 -20
  221. package/src/transformers/features/protobuf/ProtobufAssertDecodeTransformer.ts +13 -13
  222. package/src/transformers/features/protobuf/ProtobufAssertEncodeTransformer.ts +13 -13
  223. package/src/transformers/features/protobuf/ProtobufCreateAssertDecodeTransformer.ts +13 -13
  224. package/src/transformers/features/protobuf/ProtobufCreateAssertEncodeTransformer.ts +13 -13
  225. package/src/transformers/features/protobuf/ProtobufCreateDecodeTransformer.ts +13 -13
  226. package/src/transformers/features/protobuf/ProtobufCreateEncodeTransformer.ts +13 -13
  227. package/src/transformers/features/protobuf/ProtobufCreateIsDecodeTransformer.ts +13 -13
  228. package/src/transformers/features/protobuf/ProtobufCreateIsEncodeTransformer.ts +13 -13
  229. package/src/transformers/features/protobuf/ProtobufCreateValidateDecodeTransformer.ts +13 -13
  230. package/src/transformers/features/protobuf/ProtobufCreateValidateEncodeTransformer.ts +13 -13
  231. package/src/transformers/features/protobuf/ProtobufDecodeTransformer.ts +13 -13
  232. package/src/transformers/features/protobuf/ProtobufEncodeTransformer.ts +13 -13
  233. package/src/transformers/features/protobuf/ProtobufIsDecodeTransformer.ts +13 -13
  234. package/src/transformers/features/protobuf/ProtobufIsEncodeTransformer.ts +13 -13
  235. package/src/transformers/features/protobuf/ProtobufMessageTransformer.ts +35 -35
  236. package/src/transformers/features/protobuf/ProtobufValidateDecodeTransformer.ts +13 -13
  237. package/src/transformers/features/protobuf/ProtobufValidateEncodeTransformer.ts +13 -13
  238. package/src/transformers/features/reflect/ReflectMetadataTransformer.ts +69 -69
  239. package/src/transformers/features/reflect/ReflectNameTransformer.ts +82 -82
  240. package/src/transformers/internal/GenericTransformer.ts +101 -101
@@ -1,973 +1,977 @@
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 { Metadata } from "../../schemas/metadata/Metadata";
11
- import { MetadataArray } from "../../schemas/metadata/MetadataArray";
12
- import { MetadataObject } from "../../schemas/metadata/MetadataObject";
13
- import { MetadataTuple } from "../../schemas/metadata/MetadataTuple";
14
- import { MetadataTupleType } from "../../schemas/metadata/MetadataTupleType";
15
-
16
- import { IProgrammerProps } from "../../transformers/IProgrammerProps";
17
- import { ITypiaContext } from "../../transformers/ITypiaContext";
18
- import { TransformerError } from "../../transformers/TransformerError";
19
-
20
- import { StringUtil } from "../../utils/StringUtil";
21
-
22
- import { FeatureProgrammer } from "../FeatureProgrammer";
23
- import { IsProgrammer } from "../IsProgrammer";
24
- import { FunctionProgrammer } from "../helpers/FunctionProgrammer";
25
- import { NotationJoiner } from "../helpers/NotationJoiner";
26
- import { UnionExplorer } from "../helpers/UnionExplorer";
27
- import { decode_union_object } from "../internal/decode_union_object";
28
- import { postfix_of_tuple } from "../internal/postfix_of_tuple";
29
- import { wrap_metadata_rest_tuple } from "../internal/wrap_metadata_rest_tuple";
30
-
31
- export namespace NotationGeneralProgrammer {
32
- export interface IProps extends IProgrammerProps {
33
- rename: (str: string) => string;
34
- }
35
-
36
- export const returnType = (props: {
37
- rename: (str: string) => string;
38
- type: string;
39
- }) => `typia.${StringUtil.capitalize(props.rename.name)}Case<${props.type}>`;
40
-
41
- export const decompose = (props: {
42
- rename: (str: string) => string;
43
- validated: boolean;
44
- context: ITypiaContext;
45
- functor: FunctionProgrammer;
46
- type: ts.Type;
47
- name: string | undefined;
48
- }): FeatureProgrammer.IDecomposed => {
49
- const config = configure(props);
50
- if (props.validated === false)
51
- config.addition = (collection) =>
52
- IsProgrammer.write_function_statements({
53
- context: props.context,
54
- functor: props.functor,
55
- collection,
56
- });
57
- const composed: FeatureProgrammer.IComposed = FeatureProgrammer.compose({
58
- ...props,
59
- config,
60
- });
61
- return {
62
- functions: composed.functions,
63
- statements: composed.statements,
64
- arrow: ts.factory.createArrowFunction(
65
- undefined,
66
- undefined,
67
- composed.parameters,
68
- ts.factory.createTypeReferenceNode(
69
- returnType({
70
- rename: props.rename,
71
- type:
72
- props.name ??
73
- TypeFactory.getFullName({
74
- checker: props.context.checker,
75
- type: props.type,
76
- }),
77
- }),
78
- ),
79
- undefined,
80
- composed.body,
81
- ),
82
- };
83
- };
84
-
85
- export const write = (props: IProps) => {
86
- const functor: FunctionProgrammer = new FunctionProgrammer(
87
- props.modulo.getText(),
88
- );
89
- const result: FeatureProgrammer.IDecomposed = decompose({
90
- ...props,
91
- functor,
92
- validated: true,
93
- });
94
- return FeatureProgrammer.writeDecomposed({
95
- functor,
96
- modulo: props.modulo,
97
- result,
98
- });
99
- };
100
-
101
- const write_array_functions = (props: {
102
- config: FeatureProgrammer.IConfig;
103
- functor: FunctionProgrammer;
104
- collection: MetadataCollection;
105
- }): ts.VariableStatement[] =>
106
- props.collection
107
- .arrays()
108
- .filter((a) => a.recursive)
109
- .map((type, i) =>
110
- StatementFactory.constant({
111
- name: `${props.config.prefix}a${i}`,
112
- value: ts.factory.createArrowFunction(
113
- undefined,
114
- undefined,
115
- FeatureProgrammer.parameterDeclarations({
116
- config: props.config,
117
- type: TypeFactory.keyword("any"),
118
- input: ts.factory.createIdentifier("input"),
119
- }),
120
- TypeFactory.keyword("any"),
121
- undefined,
122
- decode_array_inline({
123
- ...props,
124
- input: ts.factory.createIdentifier("input"),
125
- array: MetadataArray.create({
126
- type,
127
- tags: [],
128
- }),
129
- explore: {
130
- tracable: props.config.trace,
131
- source: "function",
132
- from: "array",
133
- postfix: "",
134
- },
135
- }),
136
- ),
137
- }),
138
- );
139
-
140
- const write_tuple_functions = (props: {
141
- config: FeatureProgrammer.IConfig;
142
- rename: (str: string) => string;
143
- context: ITypiaContext;
144
- functor: FunctionProgrammer;
145
- collection: MetadataCollection;
146
- }): ts.VariableStatement[] =>
147
- props.collection
148
- .tuples()
149
- .filter((t) => t.recursive)
150
- .map((tuple, i) =>
151
- StatementFactory.constant({
152
- name: `${props.config.prefix}t${i}`,
153
- value: ts.factory.createArrowFunction(
154
- undefined,
155
- undefined,
156
- FeatureProgrammer.parameterDeclarations({
157
- config: props.config,
158
- type: TypeFactory.keyword("any"),
159
- input: ts.factory.createIdentifier("input"),
160
- }),
161
- TypeFactory.keyword("any"),
162
- undefined,
163
- decode_tuple_inline({
164
- ...props,
165
- input: ts.factory.createIdentifier("input"),
166
- tuple,
167
- explore: {
168
- tracable: props.config.trace,
169
- source: "function",
170
- from: "array",
171
- postfix: "",
172
- },
173
- }),
174
- ),
175
- }),
176
- );
177
-
178
- /* -----------------------------------------------------------
179
- DECODERS
180
- ----------------------------------------------------------- */
181
- const decode = (props: {
182
- config: FeatureProgrammer.IConfig;
183
- rename: (str: string) => string;
184
- context: ITypiaContext;
185
- functor: FunctionProgrammer;
186
- metadata: Metadata;
187
- explore: FeatureProgrammer.IExplore;
188
- input: ts.Expression;
189
- }): ts.Expression => {
190
- // ANY TYPE
191
- if (
192
- props.metadata.any ||
193
- props.metadata.arrays.some((a) => a.type.value.any) ||
194
- props.metadata.tuples.some(
195
- (t) => !!t.type.elements.length && t.type.elements.every((e) => e.any),
196
- )
197
- )
198
- return ExpressionFactory.currying({
199
- function: props.context.importer.internal("notationAny"),
200
- arguments: [
201
- props.context.importer.internal(
202
- `notation${StringUtil.capitalize(props.rename.name)}`,
203
- ),
204
- props.input,
205
- ],
206
- });
207
-
208
- interface IUnion {
209
- type: string;
210
- is: () => ts.Expression;
211
- value: () => ts.Expression;
212
- }
213
- const unions: IUnion[] = [];
214
-
215
- //----
216
- // LIST UP UNION TYPES
217
- //----
218
- // FUNCTIONAL
219
- if (props.metadata.functions.length)
220
- unions.push({
221
- type: "functional",
222
- is: () =>
223
- ts.factory.createStrictEquality(
224
- ts.factory.createStringLiteral("function"),
225
- ts.factory.createTypeOfExpression(props.input),
226
- ),
227
- value: () => ts.factory.createIdentifier("undefined"),
228
- });
229
-
230
- // TUPLES
231
- for (const tuple of props.metadata.tuples)
232
- unions.push({
233
- type: "tuple",
234
- is: () =>
235
- IsProgrammer.decode({
236
- ...props,
237
- metadata: (() => {
238
- const partial = Metadata.initialize();
239
- partial.tuples.push(tuple);
240
- return partial;
241
- })(),
242
- }),
243
- value: () =>
244
- decode_tuple({
245
- ...props,
246
- tuple,
247
- }),
248
- });
249
-
250
- // ARRAYS
251
- if (props.metadata.arrays.length)
252
- unions.push({
253
- type: "array",
254
- is: () => ExpressionFactory.isArray(props.input),
255
- value: () =>
256
- explore_arrays({
257
- ...props,
258
- arrays: props.metadata.arrays,
259
- explore: {
260
- ...props.explore,
261
- from: "array",
262
- },
263
- }),
264
- });
265
-
266
- // NATIVE TYPES
267
- if (props.metadata.sets.length)
268
- unions.push({
269
- type: "set",
270
- is: () => ExpressionFactory.isInstanceOf("Set", props.input),
271
- value: () =>
272
- explore_sets({
273
- ...props,
274
- sets: props.metadata.sets,
275
- explore: {
276
- ...props.explore,
277
- from: "array",
278
- },
279
- }),
280
- });
281
- if (props.metadata.maps.length)
282
- unions.push({
283
- type: "map",
284
- is: () => ExpressionFactory.isInstanceOf("Map", props.input),
285
- value: () =>
286
- explore_maps({
287
- ...props,
288
- maps: props.metadata.maps,
289
- explore: {
290
- ...props.explore,
291
- from: "array",
292
- },
293
- }),
294
- });
295
- for (const native of props.metadata.natives) {
296
- if (native === "WeakSet" || native === "WeakMap") continue;
297
- unions.push({
298
- type: "native",
299
- is: () => ExpressionFactory.isInstanceOf(native, props.input),
300
- value: () =>
301
- native === "Boolean" || native === "Number" || native === "String"
302
- ? ts.factory.createCallExpression(
303
- IdentifierFactory.access(props.input, "valueOf"),
304
- undefined,
305
- undefined,
306
- )
307
- : decode_native({
308
- name: native,
309
- input: props.input,
310
- }),
311
- });
312
- }
313
-
314
- // OBJECTS
315
- if (props.metadata.objects.length)
316
- unions.push({
317
- type: "object",
318
- is: () =>
319
- ExpressionFactory.isObject({
320
- checkNull: true,
321
- checkArray: false,
322
- input: props.input,
323
- }),
324
- value: () =>
325
- explore_objects({
326
- ...props,
327
- explore: {
328
- ...props.explore,
329
- from: "object",
330
- },
331
- }),
332
- });
333
-
334
- // COMPOSITION
335
- if (unions.length === 0) return props.input;
336
- else if (unions.length === 1 && props.metadata.size() === 1) {
337
- const value: ts.Expression =
338
- (props.metadata.nullable || props.metadata.isRequired() === false) &&
339
- is_instance(props.metadata)
340
- ? ts.factory.createConditionalExpression(
341
- props.input,
342
- undefined,
343
- unions[0]!.value(),
344
- undefined,
345
- props.input,
346
- )
347
- : unions[0]!.value();
348
- return ts.factory.createAsExpression(value, TypeFactory.keyword("any"));
349
- } else {
350
- let last: ts.Expression = props.input;
351
- for (const u of unions.reverse())
352
- last = ts.factory.createConditionalExpression(
353
- u.is(),
354
- undefined,
355
- u.value(),
356
- undefined,
357
- last,
358
- );
359
- return ts.factory.createAsExpression(last, TypeFactory.keyword("any"));
360
- }
361
- };
362
-
363
- const decode_object = (props: {
364
- functor: FunctionProgrammer;
365
- object: MetadataObject;
366
- input: ts.Expression;
367
- explore: FeatureProgrammer.IExplore;
368
- }) =>
369
- FeatureProgrammer.decode_object({
370
- config: {
371
- trace: false,
372
- path: false,
373
- prefix: PREFIX,
374
- },
375
- functor: props.functor,
376
- object: props.object,
377
- input: props.input,
378
- explore: props.explore,
379
- });
380
-
381
- const decode_array = (props: {
382
- config: FeatureProgrammer.IConfig;
383
- functor: FunctionProgrammer;
384
- input: ts.Expression;
385
- array: MetadataArray;
386
- explore: FeatureProgrammer.IExplore;
387
- }) =>
388
- props.array.type.recursive
389
- ? ts.factory.createCallExpression(
390
- ts.factory.createIdentifier(
391
- props.functor.useLocal(
392
- `${props.config.prefix}a${props.array.type.index}`,
393
- ),
394
- ),
395
- undefined,
396
- FeatureProgrammer.argumentsArray({
397
- config: props.config,
398
- explore: {
399
- ...props.explore,
400
- source: "function",
401
- from: "array",
402
- },
403
- input: props.input,
404
- }),
405
- )
406
- : decode_array_inline(props);
407
-
408
- const decode_array_inline = (props: {
409
- config: FeatureProgrammer.IConfig;
410
- functor: FunctionProgrammer;
411
- input: ts.Expression;
412
- array: MetadataArray;
413
- explore: FeatureProgrammer.IExplore;
414
- }) =>
415
- FeatureProgrammer.decode_array({
416
- config: props.config,
417
- functor: props.functor,
418
- combiner: NotationJoiner.array,
419
- array: props.array,
420
- input: props.input,
421
- explore: props.explore,
422
- });
423
-
424
- const decode_tuple = (props: {
425
- config: FeatureProgrammer.IConfig;
426
- rename: (str: string) => string;
427
- context: ITypiaContext;
428
- functor: FunctionProgrammer;
429
- tuple: MetadataTuple;
430
- explore: FeatureProgrammer.IExplore;
431
- input: ts.Expression;
432
- }): ts.Expression =>
433
- props.tuple.type.recursive
434
- ? ts.factory.createCallExpression(
435
- ts.factory.createIdentifier(
436
- props.functor.useLocal(
437
- `${props.config.prefix}t${props.tuple.type.index}`,
438
- ),
439
- ),
440
- undefined,
441
- FeatureProgrammer.argumentsArray({
442
- config: props.config,
443
- explore: {
444
- ...props.explore,
445
- source: "function",
446
- },
447
- input: props.input,
448
- }),
449
- )
450
- : decode_tuple_inline({
451
- ...props,
452
- tuple: props.tuple.type,
453
- });
454
-
455
- const decode_tuple_inline = (props: {
456
- context: ITypiaContext;
457
- rename: (str: string) => string;
458
- config: FeatureProgrammer.IConfig;
459
- functor: FunctionProgrammer;
460
- explore: FeatureProgrammer.IExplore;
461
- tuple: MetadataTupleType;
462
- input: ts.Expression;
463
- }): ts.Expression => {
464
- const elements: ts.Expression[] = props.tuple.elements
465
- .filter((m) => m.rest === null)
466
- .map((elem, index) =>
467
- decode({
468
- ...props,
469
- input: ts.factory.createElementAccessExpression(props.input, index),
470
- metadata: elem,
471
- explore: {
472
- ...props.explore,
473
- from: "array",
474
- postfix: props.explore.postfix.length
475
- ? `${postfix_of_tuple(props.explore.postfix)}[${index}]"`
476
- : `"[${index}]"`,
477
- },
478
- }),
479
- );
480
- const rest = (() => {
481
- if (props.tuple.elements.length === 0) return null;
482
-
483
- const last: Metadata = props.tuple.elements.at(-1)!;
484
- const rest: Metadata | null = last.rest;
485
- if (rest === null) return null;
486
-
487
- return decode({
488
- ...props,
489
- input: ts.factory.createCallExpression(
490
- IdentifierFactory.access(props.input, "slice"),
491
- undefined,
492
- [ExpressionFactory.number(props.tuple.elements.length - 1)],
493
- ),
494
- metadata: wrap_metadata_rest_tuple(props.tuple.elements.at(-1)!.rest!),
495
- explore: {
496
- ...props.explore,
497
- start: props.tuple.elements.length - 1,
498
- },
499
- });
500
- })();
501
- return NotationJoiner.tuple({
502
- elements,
503
- rest,
504
- });
505
- };
506
-
507
- /* -----------------------------------------------------------
508
- NATIVE CLASSES
509
- ----------------------------------------------------------- */
510
- const decode_native = (props: { name: string; input: ts.Expression }) =>
511
- props.name === "Date"
512
- ? ts.factory.createNewExpression(
513
- ts.factory.createIdentifier(props.name),
514
- undefined,
515
- [props.input],
516
- )
517
- : props.input;
518
-
519
- /* -----------------------------------------------------------
520
- EXPLORERS FOR UNION TYPES
521
- ----------------------------------------------------------- */
522
- const explore_sets = (props: {
523
- context: ITypiaContext;
524
- config: FeatureProgrammer.IConfig;
525
- functor: FunctionProgrammer;
526
- input: ts.Expression;
527
- explore: FeatureProgrammer.IExplore;
528
- sets: Metadata[];
529
- }): ts.Expression =>
530
- ts.factory.createCallExpression(
531
- UnionExplorer.set({
532
- config: {
533
- checker: (v) =>
534
- IsProgrammer.decode({
535
- context: props.context,
536
- functor: props.functor,
537
- input: v.input,
538
- metadata: v.definition,
539
- explore: v.explore,
540
- }),
541
- decoder: (v) =>
542
- ts.factory.createNewExpression(
543
- ts.factory.createIdentifier("Set"),
544
- [TypeFactory.keyword("any")],
545
- [
546
- decode_array({
547
- config: props.config,
548
- functor: props.functor,
549
- input: v.input,
550
- array: v.definition,
551
- explore: v.explore,
552
- }),
553
- ],
554
- ),
555
- empty: ts.factory.createNewExpression(
556
- ts.factory.createIdentifier("Set"),
557
- [TypeFactory.keyword("any")],
558
- [],
559
- ),
560
- success: ts.factory.createTrue(),
561
- failure: (v) =>
562
- create_throw_error({
563
- context: props.context,
564
- functor: props.functor,
565
- expected: v.expected,
566
- input: v.input,
567
- }),
568
- },
569
- parameters: [],
570
- input: props.input,
571
- sets: props.sets,
572
- explore: props.explore,
573
- }),
574
- undefined,
575
- undefined,
576
- );
577
-
578
- const explore_maps = (props: {
579
- context: ITypiaContext;
580
- config: FeatureProgrammer.IConfig;
581
- functor: FunctionProgrammer;
582
- input: ts.Expression;
583
- maps: Metadata.Entry[];
584
- explore: FeatureProgrammer.IExplore;
585
- }): ts.Expression =>
586
- ts.factory.createCallExpression(
587
- UnionExplorer.map({
588
- config: {
589
- checker: (v) =>
590
- ts.factory.createLogicalAnd(
591
- IsProgrammer.decode({
592
- context: props.context,
593
- functor: props.functor,
594
- input: ts.factory.createElementAccessExpression(v.input, 0),
595
- metadata: v.definition[0],
596
- explore: {
597
- ...props.explore,
598
- postfix: `${v.explore.postfix}[0]`,
599
- },
600
- }),
601
- IsProgrammer.decode({
602
- context: props.context,
603
- functor: props.functor,
604
- input: ts.factory.createElementAccessExpression(v.input, 1),
605
- metadata: v.definition[1],
606
- explore: {
607
- ...props.explore,
608
- postfix: `${props.explore.postfix}[1]`,
609
- },
610
- }),
611
- ),
612
- decoder: (v) =>
613
- ts.factory.createNewExpression(
614
- ts.factory.createIdentifier("Map"),
615
- [TypeFactory.keyword("any"), TypeFactory.keyword("any")],
616
- [
617
- decode_array({
618
- config: props.config,
619
- functor: props.functor,
620
- input: v.input,
621
- array: v.definition,
622
- explore: v.explore,
623
- }),
624
- ],
625
- ),
626
- empty: ts.factory.createNewExpression(
627
- ts.factory.createIdentifier("Map"),
628
- [TypeFactory.keyword("any"), TypeFactory.keyword("any")],
629
- [],
630
- ),
631
- success: ts.factory.createTrue(),
632
- failure: (v) =>
633
- create_throw_error({
634
- context: props.context,
635
- functor: props.functor,
636
- expected: v.expected,
637
- input: v.input,
638
- }),
639
- },
640
- parameters: [],
641
- input: props.input,
642
- maps: props.maps,
643
- explore: props.explore,
644
- }),
645
- undefined,
646
- undefined,
647
- );
648
-
649
- const explore_objects = (props: {
650
- config: FeatureProgrammer.IConfig;
651
- functor: FunctionProgrammer;
652
- input: ts.Expression;
653
- metadata: Metadata;
654
- explore: FeatureProgrammer.IExplore;
655
- }) => {
656
- if (props.metadata.objects.length === 1)
657
- return decode_object({
658
- functor: props.functor,
659
- object: props.metadata.objects[0]!,
660
- input: props.input,
661
- explore: props.explore,
662
- });
663
- return ts.factory.createCallExpression(
664
- ts.factory.createIdentifier(
665
- props.functor.useLocal(`${PREFIX}u${props.metadata.union_index!}`),
666
- ),
667
- undefined,
668
- FeatureProgrammer.argumentsArray(props),
669
- );
670
- };
671
-
672
- const explore_arrays = (props: {
673
- context: ITypiaContext;
674
- config: FeatureProgrammer.IConfig;
675
- functor: FunctionProgrammer;
676
- input: ts.Expression;
677
- arrays: MetadataArray[];
678
- explore: FeatureProgrammer.IExplore;
679
- }): ts.Expression =>
680
- explore_array_like_union_types({
681
- ...props,
682
- factory: (next) =>
683
- UnionExplorer.array({
684
- config: {
685
- checker: (v) =>
686
- IsProgrammer.decode({
687
- context: props.context,
688
- functor: props.functor,
689
- input: v.input,
690
- metadata: v.definition,
691
- explore: v.explore,
692
- }),
693
- decoder: (v) =>
694
- decode_array({
695
- config: props.config,
696
- functor: props.functor,
697
- input: v.input,
698
- array: v.definition,
699
- explore: v.explore,
700
- }),
701
- empty: ts.factory.createIdentifier("[]"),
702
- success: ts.factory.createTrue(),
703
- failure: (v) =>
704
- create_throw_error({
705
- context: props.context,
706
- functor: props.functor,
707
- expected: v.expected,
708
- input: v.input,
709
- }),
710
- },
711
- parameters: next.parameters,
712
- input: next.input,
713
- arrays: next.definitions,
714
- explore: next.explore,
715
- }),
716
- definitions: props.arrays,
717
- input: props.input,
718
- explore: props.explore,
719
- });
720
-
721
- const explore_array_like_union_types = <
722
- T extends MetadataArray | MetadataTuple,
723
- >(props: {
724
- config: FeatureProgrammer.IConfig;
725
- functor: FunctionProgrammer;
726
- factory: (next: {
727
- parameters: ts.ParameterDeclaration[];
728
- input: ts.Expression;
729
- definitions: T[];
730
- explore: FeatureProgrammer.IExplore;
731
- }) => ts.ArrowFunction;
732
- input: ts.Expression;
733
- definitions: T[];
734
- explore: FeatureProgrammer.IExplore;
735
- }): ts.Expression => {
736
- const arrow = (next: {
737
- parameters: ts.ParameterDeclaration[];
738
- explore: FeatureProgrammer.IExplore;
739
- input: ts.Expression;
740
- }): ts.ArrowFunction =>
741
- props.factory({
742
- parameters: next.parameters,
743
- definitions: props.definitions,
744
- explore: next.explore,
745
- input: next.input,
746
- });
747
- if (props.definitions.every((e) => e.type.recursive === false))
748
- ts.factory.createCallExpression(
749
- arrow({
750
- parameters: [],
751
- explore: props.explore,
752
- input: props.input,
753
- }),
754
- undefined,
755
- [],
756
- );
757
-
758
- const arrayExplore: FeatureProgrammer.IExplore = {
759
- ...props.explore,
760
- source: "function",
761
- from: "array",
762
- };
763
- return ts.factory.createCallExpression(
764
- ts.factory.createIdentifier(
765
- props.functor.emplaceUnion(
766
- props.config.prefix,
767
- props.definitions.map((e) => e.type.name).join(" | "),
768
- () =>
769
- arrow({
770
- parameters: FeatureProgrammer.parameterDeclarations({
771
- config: props.config,
772
- type: TypeFactory.keyword("any"),
773
- input: ts.factory.createIdentifier("input"),
774
- }),
775
- explore: {
776
- ...arrayExplore,
777
- postfix: "",
778
- },
779
- input: ts.factory.createIdentifier("input"),
780
- }),
781
- ),
782
- ),
783
- undefined,
784
- FeatureProgrammer.argumentsArray({
785
- config: props.config,
786
- explore: arrayExplore,
787
- input: props.input,
788
- }),
789
- );
790
- };
791
-
792
- /* -----------------------------------------------------------
793
- CONFIGURATIONS
794
- ----------------------------------------------------------- */
795
- const PREFIX = "$c";
796
-
797
- const configure = (props: {
798
- rename: (str: string) => string;
799
- context: ITypiaContext;
800
- functor: FunctionProgrammer;
801
- }): FeatureProgrammer.IConfig<ts.Expression> => {
802
- const config: FeatureProgrammer.IConfig<ts.Expression> = {
803
- types: {
804
- input: (type, name) =>
805
- ts.factory.createTypeReferenceNode(
806
- name ??
807
- TypeFactory.getFullName({ checker: props.context.checker, type }),
808
- ),
809
- output: (type, name) =>
810
- ts.factory.createTypeReferenceNode(
811
- returnType({
812
- rename: props.rename,
813
- type:
814
- name ??
815
- TypeFactory.getFullName({
816
- checker: props.context.checker,
817
- type,
818
- }),
819
- }),
820
- ),
821
- },
822
- prefix: PREFIX,
823
- trace: false,
824
- path: false,
825
- initializer,
826
- decoder: (next) =>
827
- decode({
828
- config,
829
- rename: props.rename,
830
- context: props.context,
831
- functor: props.functor,
832
- metadata: next.metadata,
833
- explore: next.explore,
834
- input: next.input,
835
- }),
836
- objector: {
837
- checker: (next) =>
838
- IsProgrammer.decode({
839
- context: props.context,
840
- functor: props.functor,
841
- input: next.input,
842
- metadata: next.metadata,
843
- explore: next.explore,
844
- }),
845
- decoder: (next) =>
846
- decode_object({
847
- functor: props.functor,
848
- object: next.object,
849
- input: next.input,
850
- explore: next.explore,
851
- }),
852
- joiner: (next) =>
853
- NotationJoiner.object({
854
- rename: props.rename,
855
- input: next.input!,
856
- entries: next.entries,
857
- }),
858
- unionizer: (next) =>
859
- decode_union_object({
860
- checker: (v) =>
861
- IsProgrammer.decode_object({
862
- context: props.context,
863
- functor: props.functor,
864
- object: v.object,
865
- input: v.input,
866
- explore: v.explore,
867
- }),
868
- decoder: (v) =>
869
- decode_object({
870
- functor: props.functor,
871
- object: v.object,
872
- input: v.input,
873
- explore: v.explore,
874
- }),
875
- success: (exp) => exp,
876
- escaper: (v) =>
877
- create_throw_error({
878
- context: props.context,
879
- functor: props.functor,
880
- expected: v.expected,
881
- input: v.input,
882
- }),
883
- input: next.input,
884
- objects: next.objects,
885
- explore: next.explore,
886
- }),
887
- failure: (next) =>
888
- create_throw_error({
889
- context: props.context,
890
- functor: props.functor,
891
- expected: next.expected,
892
- input: next.input,
893
- }),
894
- },
895
- generator: {
896
- arrays: (collection) =>
897
- write_array_functions({
898
- functor: props.functor,
899
- config,
900
- collection,
901
- }),
902
- tuples: (collection) =>
903
- write_tuple_functions({
904
- rename: props.rename,
905
- context: props.context,
906
- functor: props.functor,
907
- config,
908
- collection,
909
- }),
910
- },
911
- };
912
- return config;
913
- };
914
-
915
- const initializer: FeatureProgrammer.IConfig["initializer"] = (props) => {
916
- const collection = new MetadataCollection();
917
- const result = MetadataFactory.analyze({
918
- checker: props.context.checker,
919
- transformer: props.context.transformer,
920
- options: {
921
- escape: false,
922
- constant: true,
923
- absorb: true,
924
- },
925
- collection,
926
- type: props.type,
927
- });
928
- if (result.success === false)
929
- throw TransformerError.from({
930
- code: props.functor.method,
931
- errors: result.errors,
932
- });
933
- return [collection, result.data];
934
- };
935
-
936
- const create_throw_error = (props: {
937
- context: ITypiaContext;
938
- functor: FunctionProgrammer;
939
- expected: string;
940
- input: ts.Expression;
941
- }) =>
942
- ts.factory.createExpressionStatement(
943
- ts.factory.createCallExpression(
944
- props.context.importer.internal("throwTypeGuardError"),
945
- undefined,
946
- [
947
- ts.factory.createObjectLiteralExpression(
948
- [
949
- ts.factory.createPropertyAssignment(
950
- "method",
951
- ts.factory.createStringLiteral(props.functor.method),
952
- ),
953
- ts.factory.createPropertyAssignment(
954
- "expected",
955
- ts.factory.createStringLiteral(props.expected),
956
- ),
957
- ts.factory.createPropertyAssignment("value", props.input),
958
- ],
959
- true,
960
- ),
961
- ],
962
- ),
963
- );
964
-
965
- const is_instance = (metadata: Metadata): boolean =>
966
- !!metadata.objects.length ||
967
- !!metadata.arrays.length ||
968
- !!metadata.tuples.length ||
969
- !!metadata.sets.length ||
970
- !!metadata.maps.length ||
971
- !!metadata.natives.length ||
972
- (metadata.rest !== null && is_instance(metadata.rest));
973
- }
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 { Metadata } from "../../schemas/metadata/Metadata";
11
+ import { MetadataArray } from "../../schemas/metadata/MetadataArray";
12
+ import { MetadataObject } from "../../schemas/metadata/MetadataObject";
13
+ import { MetadataTuple } from "../../schemas/metadata/MetadataTuple";
14
+ import { MetadataTupleType } from "../../schemas/metadata/MetadataTupleType";
15
+
16
+ import { IProgrammerProps } from "../../transformers/IProgrammerProps";
17
+ import { ITypiaContext } from "../../transformers/ITypiaContext";
18
+ import { TransformerError } from "../../transformers/TransformerError";
19
+
20
+ import { StringUtil } from "../../utils/StringUtil";
21
+
22
+ import { FeatureProgrammer } from "../FeatureProgrammer";
23
+ import { IsProgrammer } from "../IsProgrammer";
24
+ import { FunctionProgrammer } from "../helpers/FunctionProgrammer";
25
+ import { NotationJoiner } from "../helpers/NotationJoiner";
26
+ import { UnionExplorer } from "../helpers/UnionExplorer";
27
+ import { decode_union_object } from "../internal/decode_union_object";
28
+ import { postfix_of_tuple } from "../internal/postfix_of_tuple";
29
+ import { wrap_metadata_rest_tuple } from "../internal/wrap_metadata_rest_tuple";
30
+
31
+ export namespace NotationGeneralProgrammer {
32
+ export interface IProps extends IProgrammerProps {
33
+ rename: (str: string) => string;
34
+ }
35
+
36
+ export const returnType = (props: {
37
+ rename: (str: string) => string;
38
+ context: ITypiaContext;
39
+ type: string;
40
+ }) =>
41
+ props.context.importer.type({
42
+ file: "typia",
43
+ name: `${StringUtil.capitalize(props.rename.name)}Case`,
44
+ arguments: [ts.factory.createTypeReferenceNode(props.type)],
45
+ });
46
+
47
+ export const decompose = (props: {
48
+ rename: (str: string) => string;
49
+ validated: boolean;
50
+ context: ITypiaContext;
51
+ functor: FunctionProgrammer;
52
+ type: ts.Type;
53
+ name: string | undefined;
54
+ }): FeatureProgrammer.IDecomposed => {
55
+ const config = configure(props);
56
+ if (props.validated === false)
57
+ config.addition = (collection) =>
58
+ IsProgrammer.write_function_statements({
59
+ context: props.context,
60
+ functor: props.functor,
61
+ collection,
62
+ });
63
+ const composed: FeatureProgrammer.IComposed = FeatureProgrammer.compose({
64
+ ...props,
65
+ config,
66
+ });
67
+ return {
68
+ functions: composed.functions,
69
+ statements: composed.statements,
70
+ arrow: ts.factory.createArrowFunction(
71
+ undefined,
72
+ undefined,
73
+ composed.parameters,
74
+ returnType({
75
+ rename: props.rename,
76
+ context: props.context,
77
+ type:
78
+ props.name ??
79
+ TypeFactory.getFullName({
80
+ checker: props.context.checker,
81
+ type: props.type,
82
+ }),
83
+ }),
84
+ undefined,
85
+ composed.body,
86
+ ),
87
+ };
88
+ };
89
+
90
+ export const write = (props: IProps) => {
91
+ const functor: FunctionProgrammer = new FunctionProgrammer(
92
+ props.modulo.getText(),
93
+ );
94
+ const result: FeatureProgrammer.IDecomposed = decompose({
95
+ ...props,
96
+ functor,
97
+ validated: true,
98
+ });
99
+ return FeatureProgrammer.writeDecomposed({
100
+ functor,
101
+ modulo: props.modulo,
102
+ result,
103
+ });
104
+ };
105
+
106
+ const write_array_functions = (props: {
107
+ config: FeatureProgrammer.IConfig;
108
+ functor: FunctionProgrammer;
109
+ collection: MetadataCollection;
110
+ }): ts.VariableStatement[] =>
111
+ props.collection
112
+ .arrays()
113
+ .filter((a) => a.recursive)
114
+ .map((type, i) =>
115
+ StatementFactory.constant({
116
+ name: `${props.config.prefix}a${i}`,
117
+ value: ts.factory.createArrowFunction(
118
+ undefined,
119
+ undefined,
120
+ FeatureProgrammer.parameterDeclarations({
121
+ config: props.config,
122
+ type: TypeFactory.keyword("any"),
123
+ input: ts.factory.createIdentifier("input"),
124
+ }),
125
+ TypeFactory.keyword("any"),
126
+ undefined,
127
+ decode_array_inline({
128
+ ...props,
129
+ input: ts.factory.createIdentifier("input"),
130
+ array: MetadataArray.create({
131
+ type,
132
+ tags: [],
133
+ }),
134
+ explore: {
135
+ tracable: props.config.trace,
136
+ source: "function",
137
+ from: "array",
138
+ postfix: "",
139
+ },
140
+ }),
141
+ ),
142
+ }),
143
+ );
144
+
145
+ const write_tuple_functions = (props: {
146
+ config: FeatureProgrammer.IConfig;
147
+ rename: (str: string) => string;
148
+ context: ITypiaContext;
149
+ functor: FunctionProgrammer;
150
+ collection: MetadataCollection;
151
+ }): ts.VariableStatement[] =>
152
+ props.collection
153
+ .tuples()
154
+ .filter((t) => t.recursive)
155
+ .map((tuple, i) =>
156
+ StatementFactory.constant({
157
+ name: `${props.config.prefix}t${i}`,
158
+ value: ts.factory.createArrowFunction(
159
+ undefined,
160
+ undefined,
161
+ FeatureProgrammer.parameterDeclarations({
162
+ config: props.config,
163
+ type: TypeFactory.keyword("any"),
164
+ input: ts.factory.createIdentifier("input"),
165
+ }),
166
+ TypeFactory.keyword("any"),
167
+ undefined,
168
+ decode_tuple_inline({
169
+ ...props,
170
+ input: ts.factory.createIdentifier("input"),
171
+ tuple,
172
+ explore: {
173
+ tracable: props.config.trace,
174
+ source: "function",
175
+ from: "array",
176
+ postfix: "",
177
+ },
178
+ }),
179
+ ),
180
+ }),
181
+ );
182
+
183
+ /* -----------------------------------------------------------
184
+ DECODERS
185
+ ----------------------------------------------------------- */
186
+ const decode = (props: {
187
+ config: FeatureProgrammer.IConfig;
188
+ rename: (str: string) => string;
189
+ context: ITypiaContext;
190
+ functor: FunctionProgrammer;
191
+ metadata: Metadata;
192
+ explore: FeatureProgrammer.IExplore;
193
+ input: ts.Expression;
194
+ }): ts.Expression => {
195
+ // ANY TYPE
196
+ if (
197
+ props.metadata.any ||
198
+ props.metadata.arrays.some((a) => a.type.value.any) ||
199
+ props.metadata.tuples.some(
200
+ (t) => !!t.type.elements.length && t.type.elements.every((e) => e.any),
201
+ )
202
+ )
203
+ return ExpressionFactory.currying({
204
+ function: props.context.importer.internal("notationAny"),
205
+ arguments: [
206
+ props.context.importer.internal(
207
+ `notation${StringUtil.capitalize(props.rename.name)}`,
208
+ ),
209
+ props.input,
210
+ ],
211
+ });
212
+
213
+ interface IUnion {
214
+ type: string;
215
+ is: () => ts.Expression;
216
+ value: () => ts.Expression;
217
+ }
218
+ const unions: IUnion[] = [];
219
+
220
+ //----
221
+ // LIST UP UNION TYPES
222
+ //----
223
+ // FUNCTIONAL
224
+ if (props.metadata.functions.length)
225
+ unions.push({
226
+ type: "functional",
227
+ is: () =>
228
+ ts.factory.createStrictEquality(
229
+ ts.factory.createStringLiteral("function"),
230
+ ts.factory.createTypeOfExpression(props.input),
231
+ ),
232
+ value: () => ts.factory.createIdentifier("undefined"),
233
+ });
234
+
235
+ // TUPLES
236
+ for (const tuple of props.metadata.tuples)
237
+ unions.push({
238
+ type: "tuple",
239
+ is: () =>
240
+ IsProgrammer.decode({
241
+ ...props,
242
+ metadata: (() => {
243
+ const partial = Metadata.initialize();
244
+ partial.tuples.push(tuple);
245
+ return partial;
246
+ })(),
247
+ }),
248
+ value: () =>
249
+ decode_tuple({
250
+ ...props,
251
+ tuple,
252
+ }),
253
+ });
254
+
255
+ // ARRAYS
256
+ if (props.metadata.arrays.length)
257
+ unions.push({
258
+ type: "array",
259
+ is: () => ExpressionFactory.isArray(props.input),
260
+ value: () =>
261
+ explore_arrays({
262
+ ...props,
263
+ arrays: props.metadata.arrays,
264
+ explore: {
265
+ ...props.explore,
266
+ from: "array",
267
+ },
268
+ }),
269
+ });
270
+
271
+ // NATIVE TYPES
272
+ if (props.metadata.sets.length)
273
+ unions.push({
274
+ type: "set",
275
+ is: () => ExpressionFactory.isInstanceOf("Set", props.input),
276
+ value: () =>
277
+ explore_sets({
278
+ ...props,
279
+ sets: props.metadata.sets,
280
+ explore: {
281
+ ...props.explore,
282
+ from: "array",
283
+ },
284
+ }),
285
+ });
286
+ if (props.metadata.maps.length)
287
+ unions.push({
288
+ type: "map",
289
+ is: () => ExpressionFactory.isInstanceOf("Map", props.input),
290
+ value: () =>
291
+ explore_maps({
292
+ ...props,
293
+ maps: props.metadata.maps,
294
+ explore: {
295
+ ...props.explore,
296
+ from: "array",
297
+ },
298
+ }),
299
+ });
300
+ for (const native of props.metadata.natives) {
301
+ if (native === "WeakSet" || native === "WeakMap") continue;
302
+ unions.push({
303
+ type: "native",
304
+ is: () => ExpressionFactory.isInstanceOf(native, props.input),
305
+ value: () =>
306
+ native === "Boolean" || native === "Number" || native === "String"
307
+ ? ts.factory.createCallExpression(
308
+ IdentifierFactory.access(props.input, "valueOf"),
309
+ undefined,
310
+ undefined,
311
+ )
312
+ : decode_native({
313
+ name: native,
314
+ input: props.input,
315
+ }),
316
+ });
317
+ }
318
+
319
+ // OBJECTS
320
+ if (props.metadata.objects.length)
321
+ unions.push({
322
+ type: "object",
323
+ is: () =>
324
+ ExpressionFactory.isObject({
325
+ checkNull: true,
326
+ checkArray: false,
327
+ input: props.input,
328
+ }),
329
+ value: () =>
330
+ explore_objects({
331
+ ...props,
332
+ explore: {
333
+ ...props.explore,
334
+ from: "object",
335
+ },
336
+ }),
337
+ });
338
+
339
+ // COMPOSITION
340
+ if (unions.length === 0) return props.input;
341
+ else if (unions.length === 1 && props.metadata.size() === 1) {
342
+ const value: ts.Expression =
343
+ (props.metadata.nullable || props.metadata.isRequired() === false) &&
344
+ is_instance(props.metadata)
345
+ ? ts.factory.createConditionalExpression(
346
+ props.input,
347
+ undefined,
348
+ unions[0]!.value(),
349
+ undefined,
350
+ props.input,
351
+ )
352
+ : unions[0]!.value();
353
+ return ts.factory.createAsExpression(value, TypeFactory.keyword("any"));
354
+ } else {
355
+ let last: ts.Expression = props.input;
356
+ for (const u of unions.reverse())
357
+ last = ts.factory.createConditionalExpression(
358
+ u.is(),
359
+ undefined,
360
+ u.value(),
361
+ undefined,
362
+ last,
363
+ );
364
+ return ts.factory.createAsExpression(last, TypeFactory.keyword("any"));
365
+ }
366
+ };
367
+
368
+ const decode_object = (props: {
369
+ functor: FunctionProgrammer;
370
+ object: MetadataObject;
371
+ input: ts.Expression;
372
+ explore: FeatureProgrammer.IExplore;
373
+ }) =>
374
+ FeatureProgrammer.decode_object({
375
+ config: {
376
+ trace: false,
377
+ path: false,
378
+ prefix: PREFIX,
379
+ },
380
+ functor: props.functor,
381
+ object: props.object,
382
+ input: props.input,
383
+ explore: props.explore,
384
+ });
385
+
386
+ const decode_array = (props: {
387
+ config: FeatureProgrammer.IConfig;
388
+ functor: FunctionProgrammer;
389
+ input: ts.Expression;
390
+ array: MetadataArray;
391
+ explore: FeatureProgrammer.IExplore;
392
+ }) =>
393
+ props.array.type.recursive
394
+ ? ts.factory.createCallExpression(
395
+ ts.factory.createIdentifier(
396
+ props.functor.useLocal(
397
+ `${props.config.prefix}a${props.array.type.index}`,
398
+ ),
399
+ ),
400
+ undefined,
401
+ FeatureProgrammer.argumentsArray({
402
+ config: props.config,
403
+ explore: {
404
+ ...props.explore,
405
+ source: "function",
406
+ from: "array",
407
+ },
408
+ input: props.input,
409
+ }),
410
+ )
411
+ : decode_array_inline(props);
412
+
413
+ const decode_array_inline = (props: {
414
+ config: FeatureProgrammer.IConfig;
415
+ functor: FunctionProgrammer;
416
+ input: ts.Expression;
417
+ array: MetadataArray;
418
+ explore: FeatureProgrammer.IExplore;
419
+ }) =>
420
+ FeatureProgrammer.decode_array({
421
+ config: props.config,
422
+ functor: props.functor,
423
+ combiner: NotationJoiner.array,
424
+ array: props.array,
425
+ input: props.input,
426
+ explore: props.explore,
427
+ });
428
+
429
+ const decode_tuple = (props: {
430
+ config: FeatureProgrammer.IConfig;
431
+ rename: (str: string) => string;
432
+ context: ITypiaContext;
433
+ functor: FunctionProgrammer;
434
+ tuple: MetadataTuple;
435
+ explore: FeatureProgrammer.IExplore;
436
+ input: ts.Expression;
437
+ }): ts.Expression =>
438
+ props.tuple.type.recursive
439
+ ? ts.factory.createCallExpression(
440
+ ts.factory.createIdentifier(
441
+ props.functor.useLocal(
442
+ `${props.config.prefix}t${props.tuple.type.index}`,
443
+ ),
444
+ ),
445
+ undefined,
446
+ FeatureProgrammer.argumentsArray({
447
+ config: props.config,
448
+ explore: {
449
+ ...props.explore,
450
+ source: "function",
451
+ },
452
+ input: props.input,
453
+ }),
454
+ )
455
+ : decode_tuple_inline({
456
+ ...props,
457
+ tuple: props.tuple.type,
458
+ });
459
+
460
+ const decode_tuple_inline = (props: {
461
+ context: ITypiaContext;
462
+ rename: (str: string) => string;
463
+ config: FeatureProgrammer.IConfig;
464
+ functor: FunctionProgrammer;
465
+ explore: FeatureProgrammer.IExplore;
466
+ tuple: MetadataTupleType;
467
+ input: ts.Expression;
468
+ }): ts.Expression => {
469
+ const elements: ts.Expression[] = props.tuple.elements
470
+ .filter((m) => m.rest === null)
471
+ .map((elem, index) =>
472
+ decode({
473
+ ...props,
474
+ input: ts.factory.createElementAccessExpression(props.input, index),
475
+ metadata: elem,
476
+ explore: {
477
+ ...props.explore,
478
+ from: "array",
479
+ postfix: props.explore.postfix.length
480
+ ? `${postfix_of_tuple(props.explore.postfix)}[${index}]"`
481
+ : `"[${index}]"`,
482
+ },
483
+ }),
484
+ );
485
+ const rest = (() => {
486
+ if (props.tuple.elements.length === 0) return null;
487
+
488
+ const last: Metadata = props.tuple.elements.at(-1)!;
489
+ const rest: Metadata | null = last.rest;
490
+ if (rest === null) return null;
491
+
492
+ return decode({
493
+ ...props,
494
+ input: ts.factory.createCallExpression(
495
+ IdentifierFactory.access(props.input, "slice"),
496
+ undefined,
497
+ [ExpressionFactory.number(props.tuple.elements.length - 1)],
498
+ ),
499
+ metadata: wrap_metadata_rest_tuple(props.tuple.elements.at(-1)!.rest!),
500
+ explore: {
501
+ ...props.explore,
502
+ start: props.tuple.elements.length - 1,
503
+ },
504
+ });
505
+ })();
506
+ return NotationJoiner.tuple({
507
+ elements,
508
+ rest,
509
+ });
510
+ };
511
+
512
+ /* -----------------------------------------------------------
513
+ NATIVE CLASSES
514
+ ----------------------------------------------------------- */
515
+ const decode_native = (props: { name: string; input: ts.Expression }) =>
516
+ props.name === "Date"
517
+ ? ts.factory.createNewExpression(
518
+ ts.factory.createIdentifier(props.name),
519
+ undefined,
520
+ [props.input],
521
+ )
522
+ : props.input;
523
+
524
+ /* -----------------------------------------------------------
525
+ EXPLORERS FOR UNION TYPES
526
+ ----------------------------------------------------------- */
527
+ const explore_sets = (props: {
528
+ context: ITypiaContext;
529
+ config: FeatureProgrammer.IConfig;
530
+ functor: FunctionProgrammer;
531
+ input: ts.Expression;
532
+ explore: FeatureProgrammer.IExplore;
533
+ sets: Metadata[];
534
+ }): ts.Expression =>
535
+ ts.factory.createCallExpression(
536
+ UnionExplorer.set({
537
+ config: {
538
+ checker: (v) =>
539
+ IsProgrammer.decode({
540
+ context: props.context,
541
+ functor: props.functor,
542
+ input: v.input,
543
+ metadata: v.definition,
544
+ explore: v.explore,
545
+ }),
546
+ decoder: (v) =>
547
+ ts.factory.createNewExpression(
548
+ ts.factory.createIdentifier("Set"),
549
+ [TypeFactory.keyword("any")],
550
+ [
551
+ decode_array({
552
+ config: props.config,
553
+ functor: props.functor,
554
+ input: v.input,
555
+ array: v.definition,
556
+ explore: v.explore,
557
+ }),
558
+ ],
559
+ ),
560
+ empty: ts.factory.createNewExpression(
561
+ ts.factory.createIdentifier("Set"),
562
+ [TypeFactory.keyword("any")],
563
+ [],
564
+ ),
565
+ success: ts.factory.createTrue(),
566
+ failure: (v) =>
567
+ create_throw_error({
568
+ context: props.context,
569
+ functor: props.functor,
570
+ expected: v.expected,
571
+ input: v.input,
572
+ }),
573
+ },
574
+ parameters: [],
575
+ input: props.input,
576
+ sets: props.sets,
577
+ explore: props.explore,
578
+ }),
579
+ undefined,
580
+ undefined,
581
+ );
582
+
583
+ const explore_maps = (props: {
584
+ context: ITypiaContext;
585
+ config: FeatureProgrammer.IConfig;
586
+ functor: FunctionProgrammer;
587
+ input: ts.Expression;
588
+ maps: Metadata.Entry[];
589
+ explore: FeatureProgrammer.IExplore;
590
+ }): ts.Expression =>
591
+ ts.factory.createCallExpression(
592
+ UnionExplorer.map({
593
+ config: {
594
+ checker: (v) =>
595
+ ts.factory.createLogicalAnd(
596
+ IsProgrammer.decode({
597
+ context: props.context,
598
+ functor: props.functor,
599
+ input: ts.factory.createElementAccessExpression(v.input, 0),
600
+ metadata: v.definition[0],
601
+ explore: {
602
+ ...props.explore,
603
+ postfix: `${v.explore.postfix}[0]`,
604
+ },
605
+ }),
606
+ IsProgrammer.decode({
607
+ context: props.context,
608
+ functor: props.functor,
609
+ input: ts.factory.createElementAccessExpression(v.input, 1),
610
+ metadata: v.definition[1],
611
+ explore: {
612
+ ...props.explore,
613
+ postfix: `${props.explore.postfix}[1]`,
614
+ },
615
+ }),
616
+ ),
617
+ decoder: (v) =>
618
+ ts.factory.createNewExpression(
619
+ ts.factory.createIdentifier("Map"),
620
+ [TypeFactory.keyword("any"), TypeFactory.keyword("any")],
621
+ [
622
+ decode_array({
623
+ config: props.config,
624
+ functor: props.functor,
625
+ input: v.input,
626
+ array: v.definition,
627
+ explore: v.explore,
628
+ }),
629
+ ],
630
+ ),
631
+ empty: ts.factory.createNewExpression(
632
+ ts.factory.createIdentifier("Map"),
633
+ [TypeFactory.keyword("any"), TypeFactory.keyword("any")],
634
+ [],
635
+ ),
636
+ success: ts.factory.createTrue(),
637
+ failure: (v) =>
638
+ create_throw_error({
639
+ context: props.context,
640
+ functor: props.functor,
641
+ expected: v.expected,
642
+ input: v.input,
643
+ }),
644
+ },
645
+ parameters: [],
646
+ input: props.input,
647
+ maps: props.maps,
648
+ explore: props.explore,
649
+ }),
650
+ undefined,
651
+ undefined,
652
+ );
653
+
654
+ const explore_objects = (props: {
655
+ config: FeatureProgrammer.IConfig;
656
+ functor: FunctionProgrammer;
657
+ input: ts.Expression;
658
+ metadata: Metadata;
659
+ explore: FeatureProgrammer.IExplore;
660
+ }) => {
661
+ if (props.metadata.objects.length === 1)
662
+ return decode_object({
663
+ functor: props.functor,
664
+ object: props.metadata.objects[0]!,
665
+ input: props.input,
666
+ explore: props.explore,
667
+ });
668
+ return ts.factory.createCallExpression(
669
+ ts.factory.createIdentifier(
670
+ props.functor.useLocal(`${PREFIX}u${props.metadata.union_index!}`),
671
+ ),
672
+ undefined,
673
+ FeatureProgrammer.argumentsArray(props),
674
+ );
675
+ };
676
+
677
+ const explore_arrays = (props: {
678
+ context: ITypiaContext;
679
+ config: FeatureProgrammer.IConfig;
680
+ functor: FunctionProgrammer;
681
+ input: ts.Expression;
682
+ arrays: MetadataArray[];
683
+ explore: FeatureProgrammer.IExplore;
684
+ }): ts.Expression =>
685
+ explore_array_like_union_types({
686
+ ...props,
687
+ factory: (next) =>
688
+ UnionExplorer.array({
689
+ config: {
690
+ checker: (v) =>
691
+ IsProgrammer.decode({
692
+ context: props.context,
693
+ functor: props.functor,
694
+ input: v.input,
695
+ metadata: v.definition,
696
+ explore: v.explore,
697
+ }),
698
+ decoder: (v) =>
699
+ decode_array({
700
+ config: props.config,
701
+ functor: props.functor,
702
+ input: v.input,
703
+ array: v.definition,
704
+ explore: v.explore,
705
+ }),
706
+ empty: ts.factory.createIdentifier("[]"),
707
+ success: ts.factory.createTrue(),
708
+ failure: (v) =>
709
+ create_throw_error({
710
+ context: props.context,
711
+ functor: props.functor,
712
+ expected: v.expected,
713
+ input: v.input,
714
+ }),
715
+ },
716
+ parameters: next.parameters,
717
+ input: next.input,
718
+ arrays: next.definitions,
719
+ explore: next.explore,
720
+ }),
721
+ definitions: props.arrays,
722
+ input: props.input,
723
+ explore: props.explore,
724
+ });
725
+
726
+ const explore_array_like_union_types = <
727
+ T extends MetadataArray | MetadataTuple,
728
+ >(props: {
729
+ config: FeatureProgrammer.IConfig;
730
+ functor: FunctionProgrammer;
731
+ factory: (next: {
732
+ parameters: ts.ParameterDeclaration[];
733
+ input: ts.Expression;
734
+ definitions: T[];
735
+ explore: FeatureProgrammer.IExplore;
736
+ }) => ts.ArrowFunction;
737
+ input: ts.Expression;
738
+ definitions: T[];
739
+ explore: FeatureProgrammer.IExplore;
740
+ }): ts.Expression => {
741
+ const arrow = (next: {
742
+ parameters: ts.ParameterDeclaration[];
743
+ explore: FeatureProgrammer.IExplore;
744
+ input: ts.Expression;
745
+ }): ts.ArrowFunction =>
746
+ props.factory({
747
+ parameters: next.parameters,
748
+ definitions: props.definitions,
749
+ explore: next.explore,
750
+ input: next.input,
751
+ });
752
+ if (props.definitions.every((e) => e.type.recursive === false))
753
+ ts.factory.createCallExpression(
754
+ arrow({
755
+ parameters: [],
756
+ explore: props.explore,
757
+ input: props.input,
758
+ }),
759
+ undefined,
760
+ [],
761
+ );
762
+
763
+ const arrayExplore: FeatureProgrammer.IExplore = {
764
+ ...props.explore,
765
+ source: "function",
766
+ from: "array",
767
+ };
768
+ return ts.factory.createCallExpression(
769
+ ts.factory.createIdentifier(
770
+ props.functor.emplaceUnion(
771
+ props.config.prefix,
772
+ props.definitions.map((e) => e.type.name).join(" | "),
773
+ () =>
774
+ arrow({
775
+ parameters: FeatureProgrammer.parameterDeclarations({
776
+ config: props.config,
777
+ type: TypeFactory.keyword("any"),
778
+ input: ts.factory.createIdentifier("input"),
779
+ }),
780
+ explore: {
781
+ ...arrayExplore,
782
+ postfix: "",
783
+ },
784
+ input: ts.factory.createIdentifier("input"),
785
+ }),
786
+ ),
787
+ ),
788
+ undefined,
789
+ FeatureProgrammer.argumentsArray({
790
+ config: props.config,
791
+ explore: arrayExplore,
792
+ input: props.input,
793
+ }),
794
+ );
795
+ };
796
+
797
+ /* -----------------------------------------------------------
798
+ CONFIGURATIONS
799
+ ----------------------------------------------------------- */
800
+ const PREFIX = "$c";
801
+
802
+ const configure = (props: {
803
+ rename: (str: string) => string;
804
+ context: ITypiaContext;
805
+ functor: FunctionProgrammer;
806
+ }): FeatureProgrammer.IConfig<ts.Expression> => {
807
+ const config: FeatureProgrammer.IConfig<ts.Expression> = {
808
+ types: {
809
+ input: (type, name) =>
810
+ ts.factory.createTypeReferenceNode(
811
+ name ??
812
+ TypeFactory.getFullName({ checker: props.context.checker, type }),
813
+ ),
814
+ output: (type, name) =>
815
+ returnType({
816
+ rename: props.rename,
817
+ context: props.context,
818
+ type:
819
+ name ??
820
+ TypeFactory.getFullName({
821
+ checker: props.context.checker,
822
+ type,
823
+ }),
824
+ }),
825
+ },
826
+ prefix: PREFIX,
827
+ trace: false,
828
+ path: false,
829
+ initializer,
830
+ decoder: (next) =>
831
+ decode({
832
+ config,
833
+ rename: props.rename,
834
+ context: props.context,
835
+ functor: props.functor,
836
+ metadata: next.metadata,
837
+ explore: next.explore,
838
+ input: next.input,
839
+ }),
840
+ objector: {
841
+ checker: (next) =>
842
+ IsProgrammer.decode({
843
+ context: props.context,
844
+ functor: props.functor,
845
+ input: next.input,
846
+ metadata: next.metadata,
847
+ explore: next.explore,
848
+ }),
849
+ decoder: (next) =>
850
+ decode_object({
851
+ functor: props.functor,
852
+ object: next.object,
853
+ input: next.input,
854
+ explore: next.explore,
855
+ }),
856
+ joiner: (next) =>
857
+ NotationJoiner.object({
858
+ rename: props.rename,
859
+ input: next.input!,
860
+ entries: next.entries,
861
+ }),
862
+ unionizer: (next) =>
863
+ decode_union_object({
864
+ checker: (v) =>
865
+ IsProgrammer.decode_object({
866
+ context: props.context,
867
+ functor: props.functor,
868
+ object: v.object,
869
+ input: v.input,
870
+ explore: v.explore,
871
+ }),
872
+ decoder: (v) =>
873
+ decode_object({
874
+ functor: props.functor,
875
+ object: v.object,
876
+ input: v.input,
877
+ explore: v.explore,
878
+ }),
879
+ success: (exp) => exp,
880
+ escaper: (v) =>
881
+ create_throw_error({
882
+ context: props.context,
883
+ functor: props.functor,
884
+ expected: v.expected,
885
+ input: v.input,
886
+ }),
887
+ input: next.input,
888
+ objects: next.objects,
889
+ explore: next.explore,
890
+ }),
891
+ failure: (next) =>
892
+ create_throw_error({
893
+ context: props.context,
894
+ functor: props.functor,
895
+ expected: next.expected,
896
+ input: next.input,
897
+ }),
898
+ },
899
+ generator: {
900
+ arrays: (collection) =>
901
+ write_array_functions({
902
+ functor: props.functor,
903
+ config,
904
+ collection,
905
+ }),
906
+ tuples: (collection) =>
907
+ write_tuple_functions({
908
+ rename: props.rename,
909
+ context: props.context,
910
+ functor: props.functor,
911
+ config,
912
+ collection,
913
+ }),
914
+ },
915
+ };
916
+ return config;
917
+ };
918
+
919
+ const initializer: FeatureProgrammer.IConfig["initializer"] = (props) => {
920
+ const collection = new MetadataCollection();
921
+ const result = MetadataFactory.analyze({
922
+ checker: props.context.checker,
923
+ transformer: props.context.transformer,
924
+ options: {
925
+ escape: false,
926
+ constant: true,
927
+ absorb: true,
928
+ },
929
+ collection,
930
+ type: props.type,
931
+ });
932
+ if (result.success === false)
933
+ throw TransformerError.from({
934
+ code: props.functor.method,
935
+ errors: result.errors,
936
+ });
937
+ return [collection, result.data];
938
+ };
939
+
940
+ const create_throw_error = (props: {
941
+ context: ITypiaContext;
942
+ functor: FunctionProgrammer;
943
+ expected: string;
944
+ input: ts.Expression;
945
+ }) =>
946
+ ts.factory.createExpressionStatement(
947
+ ts.factory.createCallExpression(
948
+ props.context.importer.internal("throwTypeGuardError"),
949
+ undefined,
950
+ [
951
+ ts.factory.createObjectLiteralExpression(
952
+ [
953
+ ts.factory.createPropertyAssignment(
954
+ "method",
955
+ ts.factory.createStringLiteral(props.functor.method),
956
+ ),
957
+ ts.factory.createPropertyAssignment(
958
+ "expected",
959
+ ts.factory.createStringLiteral(props.expected),
960
+ ),
961
+ ts.factory.createPropertyAssignment("value", props.input),
962
+ ],
963
+ true,
964
+ ),
965
+ ],
966
+ ),
967
+ );
968
+
969
+ const is_instance = (metadata: Metadata): boolean =>
970
+ !!metadata.objects.length ||
971
+ !!metadata.arrays.length ||
972
+ !!metadata.tuples.length ||
973
+ !!metadata.sets.length ||
974
+ !!metadata.maps.length ||
975
+ !!metadata.natives.length ||
976
+ (metadata.rest !== null && is_instance(metadata.rest));
977
+ }