@oscarpalmer/jhunal 0.16.0 → 0.18.0

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 (40) hide show
  1. package/dist/constants.d.mts +28 -15
  2. package/dist/constants.mjs +31 -14
  3. package/dist/helpers.d.mts +8 -1
  4. package/dist/helpers.mjs +68 -3
  5. package/dist/index.d.mts +283 -262
  6. package/dist/index.mjs +189 -56
  7. package/dist/models/infer.model.d.mts +66 -0
  8. package/dist/models/infer.model.mjs +1 -0
  9. package/dist/models/misc.model.d.mts +153 -0
  10. package/dist/models/misc.model.mjs +1 -0
  11. package/dist/models/schema.plain.model.d.mts +92 -0
  12. package/dist/models/schema.plain.model.mjs +1 -0
  13. package/dist/models/schema.typed.model.d.mts +96 -0
  14. package/dist/models/schema.typed.model.mjs +1 -0
  15. package/dist/models/transform.model.d.mts +59 -0
  16. package/dist/models/transform.model.mjs +1 -0
  17. package/dist/models/validation.model.d.mts +81 -0
  18. package/dist/models/validation.model.mjs +21 -0
  19. package/dist/schematic.d.mts +20 -6
  20. package/dist/schematic.mjs +7 -12
  21. package/dist/validation/property.validation.d.mts +1 -1
  22. package/dist/validation/property.validation.mjs +21 -17
  23. package/dist/validation/value.validation.d.mts +2 -2
  24. package/dist/validation/value.validation.mjs +63 -11
  25. package/package.json +3 -3
  26. package/src/constants.ts +84 -18
  27. package/src/helpers.ts +162 -4
  28. package/src/index.ts +3 -1
  29. package/src/models/infer.model.ts +105 -0
  30. package/src/models/misc.model.ts +212 -0
  31. package/src/models/schema.plain.model.ts +110 -0
  32. package/src/models/schema.typed.model.ts +109 -0
  33. package/src/models/transform.model.ts +85 -0
  34. package/src/models/validation.model.ts +123 -0
  35. package/src/schematic.ts +29 -18
  36. package/src/validation/property.validation.ts +46 -36
  37. package/src/validation/value.validation.ts +115 -15
  38. package/dist/models.d.mts +0 -507
  39. package/dist/models.mjs +0 -18
  40. package/src/models.ts +0 -691
package/dist/index.mjs CHANGED
@@ -1,23 +1,39 @@
1
1
  import { isConstructor, isPlainObject } from "@oscarpalmer/atoms/is";
2
2
  import { join } from "@oscarpalmer/atoms/string";
3
3
  //#region src/constants.ts
4
- const ERROR_NAME = "SchematicError";
5
- const EXPRESSION_PROPERTY = /(^|\.)\$(required|type|validators)(\.|$)/;
6
4
  const MESSAGE_CONSTRUCTOR = "Expected a constructor function";
7
- const MESSAGE_SCHEMA_INVALID_EMPTY = "Schema must have at least one property";
8
- const MESSAGE_SCHEMA_INVALID_PROPERTY_DISALLOWED = "'<key>.<property>' property is not allowed for schemas in $type";
9
- const MESSAGE_SCHEMA_INVALID_PROPERTY_REQUIRED = "'<>.$required' property must be a boolean";
10
- const MESSAGE_SCHEMA_INVALID_PROPERTY_TYPE = "'<>' property must be of a valid type";
11
- const MESSAGE_SCHEMA_INVALID_TYPE = "Schema must be an object";
12
- const MESSAGE_VALIDATOR_INVALID_KEY = "Validator '<>' does not exist";
13
- const MESSAGE_VALIDATOR_INVALID_TYPE = "Validators must be an object";
14
- const MESSAGE_VALIDATOR_INVALID_VALUE = "Validator '<>' must be a function or an array of functions";
5
+ const NAME_SCHEMATIC = "Schematic";
6
+ const NAME_ERROR_SCHEMATIC = "SchematicError";
7
+ const NAME_ERROR_VALIDATION = "ValidationError";
15
8
  const PROPERTY_REQUIRED = "$required";
9
+ const PROPERTY_SCHEMATIC = "$schematic";
16
10
  const PROPERTY_TYPE = "$type";
17
11
  const PROPERTY_VALIDATORS = "$validators";
18
- const SCHEMATIC_NAME = "$schematic";
19
- const TEMPLATE_PATTERN_KEY = "<key>";
20
- const TEMPLATE_PATTERN_PROPERTY = "<property>";
12
+ const VALIDATION_MESSAGE_INVALID_INPUT = "Expected 'object' as input but received <>";
13
+ const VALIDATION_MESSAGE_INVALID_REQUIRED = "Expected <> for required property '<>'";
14
+ const VALIDATION_MESSAGE_INVALID_TYPE = "Expected <> for '<>' but received <>";
15
+ const VALIDATION_MESSAGE_INVALID_VALUE = "Value does not satisfy validator for '<>' and type '<>'";
16
+ const VALIDATION_MESSAGE_INVALID_VALUE_SUFFIX = " at index <>";
17
+ const REPORTING_FIRST = "first";
18
+ const REPORTING_NONE = "none";
19
+ const REPORTING_THROW = "throw";
20
+ const REPORTING_TYPES = new Set([
21
+ "all",
22
+ REPORTING_FIRST,
23
+ REPORTING_NONE,
24
+ REPORTING_THROW
25
+ ]);
26
+ const SCHEMATIC_MESSAGE_SCHEMA_INVALID_EMPTY = "Schema must have at least one property";
27
+ const SCHEMATIC_MESSAGE_SCHEMA_INVALID_PROPERTY_DISALLOWED = "'<>.<>' property is not allowed for schemas in $type";
28
+ const SCHEMATIC_MESSAGE_SCHEMA_INVALID_PROPERTY_NULLABLE = "'<>' property must not be 'null' or 'undefined'";
29
+ const SCHEMATIC_MESSAGE_SCHEMA_INVALID_PROPERTY_REQUIRED = "'<>.$required' property must be a boolean";
30
+ const SCHEMATIC_MESSAGE_SCHEMA_INVALID_PROPERTY_TYPE = "'<>' property must be of a valid type";
31
+ const SCHEMATIC_MESSAGE_SCHEMA_INVALID_TYPE = "Schema must be an object";
32
+ const SCHEMATIC_MESSAGE_VALIDATOR_INVALID_KEY = "Validator '<>' does not exist";
33
+ const SCHEMATIC_MESSAGE_VALIDATOR_INVALID_TYPE = "Validators must be an object";
34
+ const SCHEMATIC_MESSAGE_VALIDATOR_INVALID_VALUE = "Validator '<>' must be a function or an array of functions";
35
+ const TYPE_ARRAY = "array";
36
+ const TYPE_NULL = "null";
21
37
  const TYPE_OBJECT = "object";
22
38
  const TYPE_UNDEFINED = "undefined";
23
39
  const VALIDATABLE_TYPES = new Set([
@@ -38,6 +54,53 @@ const TYPE_ALL = new Set([
38
54
  ]);
39
55
  //#endregion
40
56
  //#region src/helpers.ts
57
+ function getInvalidInputMessage(actual) {
58
+ return VALIDATION_MESSAGE_INVALID_INPUT.replace("<>", getValueType(actual));
59
+ }
60
+ function getInvalidMissingMessage(property) {
61
+ let message = VALIDATION_MESSAGE_INVALID_REQUIRED.replace("<>", renderTypes(property.types));
62
+ message = message.replace("<>", property.key.full);
63
+ return message;
64
+ }
65
+ function getInvalidTypeMessage(property, actual) {
66
+ let message = VALIDATION_MESSAGE_INVALID_TYPE.replace("<>", renderTypes(property.types));
67
+ message = message.replace("<>", property.key.full);
68
+ message = message.replace("<>", getValueType(actual));
69
+ return message;
70
+ }
71
+ function getInvalidValidatorMessage(property, type, index, length) {
72
+ let message = VALIDATION_MESSAGE_INVALID_VALUE.replace("<>", property.key.full);
73
+ message = message.replace("<>", type);
74
+ if (length > 1) message += VALIDATION_MESSAGE_INVALID_VALUE_SUFFIX.replace("<>", String(index));
75
+ return message;
76
+ }
77
+ function getPropertyType(original) {
78
+ if (typeof original === "function") return "a validated value";
79
+ if (Array.isArray(original)) return `'${TYPE_OBJECT}'`;
80
+ if (isSchematic(original)) return `a ${NAME_SCHEMATIC}`;
81
+ return `'${String(original)}'`;
82
+ }
83
+ function getReporting(value) {
84
+ const type = REPORTING_TYPES.has(value) ? value : REPORTING_NONE;
85
+ return {
86
+ ["all"]: type === "all",
87
+ [REPORTING_FIRST]: type === REPORTING_FIRST,
88
+ [REPORTING_NONE]: type === REPORTING_NONE,
89
+ [REPORTING_THROW]: type === REPORTING_THROW
90
+ };
91
+ }
92
+ function getValueType(value) {
93
+ const valueType = typeof value;
94
+ switch (true) {
95
+ case value === null: return `'${TYPE_NULL}'`;
96
+ case value === void 0: return `'${TYPE_UNDEFINED}'`;
97
+ case valueType !== TYPE_OBJECT: return `'${valueType}'`;
98
+ case Array.isArray(value): return `'${TYPE_ARRAY}'`;
99
+ case isPlainObject(value): return `'${TYPE_OBJECT}'`;
100
+ case isSchematic(value): return `a ${NAME_SCHEMATIC}`;
101
+ default: return value.constructor.name;
102
+ }
103
+ }
41
104
  /**
42
105
  * Creates a validator function for a given constructor
43
106
  * @param constructor - Constructor to check against
@@ -58,20 +121,40 @@ function instanceOf(constructor) {
58
121
  function isSchematic(value) {
59
122
  return typeof value === "object" && value !== null && "$schematic" in value && value["$schematic"] === true;
60
123
  }
124
+ function renderTypes(types) {
125
+ const unique = /* @__PURE__ */ new Set();
126
+ const parts = [];
127
+ for (let index = 0; index < types.length; index += 1) {
128
+ const rendered = getPropertyType(types[index]);
129
+ if (unique.has(rendered)) continue;
130
+ unique.add(rendered);
131
+ parts.push(rendered);
132
+ }
133
+ const { length } = parts;
134
+ let rendered = "";
135
+ for (let index = 0; index < length; index += 1) {
136
+ rendered += parts[index];
137
+ if (index < length - 2) rendered += ", ";
138
+ else if (index === length - 2) rendered += parts.length > 2 ? ", or " : " or ";
139
+ }
140
+ return rendered;
141
+ }
61
142
  //#endregion
62
- //#region src/models.ts
143
+ //#region src/models/validation.model.ts
63
144
  /**
64
- * A custom error class for schema validation failures, with its `name` set to {@link ERROR_NAME}
65
- *
66
- * @example
67
- * ```ts
68
- * throw new SchematicError('Expected a string, received a number');
69
- * ```
145
+ * A custom error class for schematic validation failures
70
146
  */
71
147
  var SchematicError = class extends Error {
72
148
  constructor(message) {
73
149
  super(message);
74
- this.name = ERROR_NAME;
150
+ this.name = NAME_ERROR_SCHEMATIC;
151
+ }
152
+ };
153
+ var ValidationError = class extends Error {
154
+ constructor(information) {
155
+ super(join(information.map((item) => item.message), "; "));
156
+ this.information = information;
157
+ this.name = NAME_ERROR_VALIDATION;
75
158
  }
76
159
  };
77
160
  //#endregion
@@ -82,32 +165,36 @@ function getDisallowedProperty(obj) {
82
165
  if ("$validators" in obj) return PROPERTY_VALIDATORS;
83
166
  }
84
167
  function getProperties(original, prefix, fromType) {
85
- if (Object.keys(original).length === 0) throw new SchematicError(MESSAGE_SCHEMA_INVALID_EMPTY);
168
+ if (Object.keys(original).length === 0) throw new SchematicError(SCHEMATIC_MESSAGE_SCHEMA_INVALID_EMPTY);
86
169
  if (fromType ?? false) {
87
170
  const property = getDisallowedProperty(original);
88
- if (property != null) throw new SchematicError(MESSAGE_SCHEMA_INVALID_PROPERTY_DISALLOWED.replace(TEMPLATE_PATTERN_KEY, prefix).replace(TEMPLATE_PATTERN_PROPERTY, property));
171
+ if (property != null) throw new SchematicError(SCHEMATIC_MESSAGE_SCHEMA_INVALID_PROPERTY_DISALLOWED.replace("<>", prefix).replace("<>", property));
89
172
  }
90
173
  const keys = Object.keys(original);
91
174
  const keysLength = keys.length;
92
175
  const properties = [];
93
176
  for (let keyIndex = 0; keyIndex < keysLength; keyIndex += 1) {
94
177
  const key = keys[keyIndex];
95
- if (EXPRESSION_PROPERTY.test(key)) continue;
178
+ const prefixed = join([prefix, key], ".");
96
179
  const value = original[key];
180
+ if (value == null) throw new SchematicError(SCHEMATIC_MESSAGE_SCHEMA_INVALID_PROPERTY_NULLABLE.replace("<>", prefixed));
97
181
  const types = [];
98
182
  let required = true;
99
183
  let validators = {};
100
184
  if (isPlainObject(value)) {
101
185
  required = getRequired(key, value) ?? required;
102
186
  validators = getValidators(value[PROPERTY_VALIDATORS]);
103
- if ("$type" in value) types.push(TYPE_OBJECT, ...getTypes(key, value[PROPERTY_TYPE], prefix, true));
104
- else types.push(TYPE_OBJECT, ...getTypes(key, value, prefix));
187
+ const hasType = PROPERTY_TYPE in value;
188
+ types.push(...getTypes(key, hasType ? value[PROPERTY_TYPE] : value, prefix, hasType));
105
189
  } else types.push(...getTypes(key, value, prefix));
106
190
  if (!required && !types.includes("undefined")) types.push(TYPE_UNDEFINED);
107
191
  properties.push({
108
- key,
109
192
  types,
110
193
  validators,
194
+ key: {
195
+ full: prefixed,
196
+ short: key
197
+ },
111
198
  required: required && !types.includes("undefined")
112
199
  });
113
200
  }
@@ -115,7 +202,7 @@ function getProperties(original, prefix, fromType) {
115
202
  }
116
203
  function getRequired(key, obj) {
117
204
  if (!("$required" in obj)) return;
118
- if (typeof obj["$required"] !== "boolean") throw new SchematicError(MESSAGE_SCHEMA_INVALID_PROPERTY_REQUIRED.replace("<>", key));
205
+ if (typeof obj["$required"] !== "boolean") throw new SchematicError(SCHEMATIC_MESSAGE_SCHEMA_INVALID_PROPERTY_REQUIRED.replace("<>", key));
119
206
  return obj[PROPERTY_REQUIRED];
120
207
  }
121
208
  function getTypes(key, original, prefix, fromType) {
@@ -129,7 +216,7 @@ function getTypes(key, original, prefix, fromType) {
129
216
  types.push(isConstructor(value) ? instanceOf(value) : value);
130
217
  break;
131
218
  case isPlainObject(value):
132
- types.push(...getProperties(value, join([prefix, key], "."), fromType));
219
+ types.push(getProperties(value, join([prefix, key], "."), fromType));
133
220
  break;
134
221
  case isSchematic(value):
135
222
  types.push(value);
@@ -137,55 +224,106 @@ function getTypes(key, original, prefix, fromType) {
137
224
  case TYPE_ALL.has(value):
138
225
  types.push(value);
139
226
  break;
140
- default: throw new SchematicError(MESSAGE_SCHEMA_INVALID_PROPERTY_TYPE.replace("<>", join([prefix, key], ".")));
227
+ default: throw new SchematicError(SCHEMATIC_MESSAGE_SCHEMA_INVALID_PROPERTY_TYPE.replace("<>", join([prefix, key], ".")));
141
228
  }
142
229
  }
143
- if (types.length === 0) throw new SchematicError(MESSAGE_SCHEMA_INVALID_PROPERTY_TYPE.replace("<>", join([prefix, key], ".")));
230
+ if (types.length === 0) throw new SchematicError(SCHEMATIC_MESSAGE_SCHEMA_INVALID_PROPERTY_TYPE.replace("<>", join([prefix, key], ".")));
144
231
  return types;
145
232
  }
146
233
  function getValidators(original) {
147
234
  const validators = {};
148
235
  if (original == null) return validators;
149
- if (!isPlainObject(original)) throw new TypeError(MESSAGE_VALIDATOR_INVALID_TYPE);
236
+ if (!isPlainObject(original)) throw new TypeError(SCHEMATIC_MESSAGE_VALIDATOR_INVALID_TYPE);
150
237
  const keys = Object.keys(original);
151
238
  const { length } = keys;
152
239
  for (let index = 0; index < length; index += 1) {
153
240
  const key = keys[index];
154
- if (!VALIDATABLE_TYPES.has(key)) throw new TypeError(MESSAGE_VALIDATOR_INVALID_KEY.replace("<>", key));
241
+ if (!VALIDATABLE_TYPES.has(key)) throw new TypeError(SCHEMATIC_MESSAGE_VALIDATOR_INVALID_KEY.replace("<>", key));
155
242
  const value = original[key];
156
- validators[key] = (Array.isArray(value) ? value : [value]).filter((item) => {
157
- if (typeof item !== "function") throw new TypeError(MESSAGE_VALIDATOR_INVALID_VALUE.replace("<>", key));
158
- return true;
243
+ validators[key] = (Array.isArray(value) ? value : [value]).map((item) => {
244
+ if (typeof item !== "function") throw new TypeError(SCHEMATIC_MESSAGE_VALIDATOR_INVALID_VALUE.replace("<>", key));
245
+ return item;
159
246
  });
160
247
  }
161
248
  return validators;
162
249
  }
163
250
  //#endregion
164
251
  //#region src/validation/value.validation.ts
165
- function validateObject(obj, properties) {
166
- if (!isPlainObject(obj)) return false;
252
+ function validateNamed(property, name, value, validation) {
253
+ if (!validators[name](value)) return false;
254
+ const propertyValidators = property.validators[name];
255
+ if (propertyValidators == null || propertyValidators.length === 0) return true;
256
+ const { length } = propertyValidators;
257
+ for (let index = 0; index < length; index += 1) {
258
+ const validator = propertyValidators[index];
259
+ if (!validator(value)) {
260
+ validation.push({
261
+ key: { ...property.key },
262
+ message: getInvalidValidatorMessage(property, name, index, length),
263
+ validator
264
+ });
265
+ return false;
266
+ }
267
+ }
268
+ return true;
269
+ }
270
+ function validateObject(obj, properties, reporting, validation) {
271
+ if (!isPlainObject(obj)) {
272
+ if (reporting.throw && validation == null) throw new ValidationError([{
273
+ key: {
274
+ full: "",
275
+ short: ""
276
+ },
277
+ message: getInvalidInputMessage(obj)
278
+ }]);
279
+ return false;
280
+ }
167
281
  const propertiesLength = properties.length;
168
282
  outer: for (let propertyIndex = 0; propertyIndex < propertiesLength; propertyIndex += 1) {
169
283
  const property = properties[propertyIndex];
170
284
  const { key, required, types } = property;
171
- const value = obj[key];
172
- if (value === void 0 && required) return false;
285
+ const value = obj[key.short];
286
+ if (value === void 0 && required) {
287
+ const information = {
288
+ key: { ...key },
289
+ message: getInvalidMissingMessage(property)
290
+ };
291
+ if (reporting.throw && validation == null) throw new ValidationError([information]);
292
+ if (validation != null) validation.push(information);
293
+ return false;
294
+ }
173
295
  const typesLength = types.length;
296
+ const information = [];
174
297
  for (let typeIndex = 0; typeIndex < typesLength; typeIndex += 1) {
175
298
  const type = types[typeIndex];
176
- if (validateValue(type, property, value)) continue outer;
299
+ if (validateValue(type, property, value, reporting, information)) continue outer;
177
300
  }
301
+ if (reporting.throw && validation == null) throw new ValidationError(information.length === 0 ? [{
302
+ key: { ...key },
303
+ message: getInvalidTypeMessage(property, value)
304
+ }] : information);
305
+ validation?.push(...information);
178
306
  return false;
179
307
  }
180
308
  return true;
181
309
  }
182
- function validateValue(type, property, value) {
310
+ function validateValue(type, property, value, reporting, validation) {
311
+ let result;
183
312
  switch (true) {
184
- case isSchematic(type): return type.is(value);
185
- case typeof type === "function": return type(value);
186
- case typeof type === "object": return validateObject(value, [type]);
187
- default: return validators[type](value) && (property.validators[type]?.every((validator) => validator(value)) ?? true);
313
+ case typeof type === "function":
314
+ result = type(value);
315
+ break;
316
+ case Array.isArray(type):
317
+ result = validateObject(value, type, reporting, validation);
318
+ break;
319
+ case isSchematic(type):
320
+ result = type.is(value, reporting);
321
+ break;
322
+ default:
323
+ result = validateNamed(property, type, value, validation);
324
+ break;
188
325
  }
326
+ return result;
189
327
  }
190
328
  const validators = {
191
329
  array: Array.isArray,
@@ -208,22 +346,17 @@ const validators = {
208
346
  var Schematic = class {
209
347
  #properties;
210
348
  constructor(properties) {
211
- Object.defineProperty(this, SCHEMATIC_NAME, { value: true });
349
+ Object.defineProperty(this, PROPERTY_SCHEMATIC, { value: true });
212
350
  this.#properties = properties;
213
351
  }
214
- /**
215
- * Does the value match the schema?
216
- * @param value - Value to validate
217
- * @returns `true` if the value matches the schema, otherwise `false`
218
- */
219
- is(value) {
220
- return validateObject(value, this.#properties);
352
+ is(value, errors) {
353
+ return validateObject(value, this.#properties, getReporting(errors));
221
354
  }
222
355
  };
223
356
  function schematic(schema) {
224
357
  if (isSchematic(schema)) return schema;
225
- if (!isPlainObject(schema)) throw new SchematicError(MESSAGE_SCHEMA_INVALID_TYPE);
358
+ if (!isPlainObject(schema)) throw new SchematicError(SCHEMATIC_MESSAGE_SCHEMA_INVALID_TYPE);
226
359
  return new Schematic(getProperties(schema));
227
360
  }
228
361
  //#endregion
229
- export { SchematicError, instanceOf, isSchematic, schematic };
362
+ export { SchematicError, ValidationError, instanceOf, isSchematic, schematic };
@@ -0,0 +1,66 @@
1
+ import { Schematic } from "../schematic.mjs";
2
+ import { PlainSchema, Schema, SchemaProperty } from "./schema.plain.model.mjs";
3
+ import { IsOptionalProperty, ValueName, Values } from "./misc.model.mjs";
4
+ import { Constructor, Simplify } from "@oscarpalmer/atoms/models";
5
+
6
+ //#region src/models/infer.model.d.ts
7
+ /**
8
+ * Infers the TypeScript type from a {@link Schema} definition
9
+ *
10
+ * @template Model - Schema to infer types from
11
+ *
12
+ * @example
13
+ * ```ts
14
+ * const userSchema = {
15
+ * name: 'string',
16
+ * age: 'number',
17
+ * address: { $required: false, $type: 'string' },
18
+ * } satisfies Schema;
19
+ *
20
+ * type User = Infer<typeof userSchema>;
21
+ * // { name: string; age: number; address?: string }
22
+ * ```
23
+ */
24
+ type Infer<Model extends Schema> = Simplify<{ [Key in InferRequiredKeys<Model>]: InferSchemaEntry<Model[Key]> } & { [Key in InferOptionalKeys<Model>]?: InferSchemaEntry<Model[Key]> }>;
25
+ /**
26
+ * Extracts keys from a {@link Schema} whose entries are optional _(i.e., `$required` is `false`)_
27
+ *
28
+ * @template Model - {@link Schema} to extract optional keys from
29
+ */
30
+ type InferOptionalKeys<Model extends Schema> = keyof { [Key in keyof Model as IsOptionalProperty<Model[Key]> extends true ? Key : never]: never };
31
+ /**
32
+ * Infers the TypeScript type of a {@link SchemaProperty}'s `$type` field, unwrapping arrays to infer their item type
33
+ *
34
+ * @template Value - `$type` value _(single or array)_
35
+ */
36
+ type InferPropertyType<Value> = Value extends (infer Item)[] ? InferPropertyValue<Item> : InferPropertyValue<Value>;
37
+ /**
38
+ * Maps a single type definition to its TypeScript equivalent
39
+ *
40
+ * Resolves, in order: {@link Constructor} instances, {@link Schematic} models, {@link ValueName} strings, and nested {@link Schema} objects
41
+ *
42
+ * @template Value - single type definition
43
+ */
44
+ type InferPropertyValue<Value> = Value extends Constructor<infer Instance> ? Instance : Value extends Schematic<infer Model> ? Model : Value extends ValueName ? Values[Value & ValueName] : Value extends Schema ? Infer<Value> : never;
45
+ /**
46
+ * Extracts keys from a {@link Schema} whose entries are required _(i.e., `$required` is not `false`)_
47
+ *
48
+ * @template Model - Schema to extract required keys from
49
+ */
50
+ type InferRequiredKeys<Model extends Schema> = keyof { [Key in keyof Model as IsOptionalProperty<Model[Key]> extends true ? never : Key]: never };
51
+ /**
52
+ * Infers the type for a top-level {@link Schema} entry, unwrapping arrays to infer their item type
53
+ *
54
+ * @template Value - Schema entry value _(single or array)_
55
+ */
56
+ type InferSchemaEntry<Value> = Value extends (infer Item)[] ? InferSchemaEntryValue<Item> : InferSchemaEntryValue<Value>;
57
+ /**
58
+ * Resolves a single schema entry to its TypeScript type
59
+ *
60
+ * Handles, in order: {@link Constructor} instances, {@link Schematic} models, {@link SchemaProperty} objects, {@link NestedSchema} objects, {@link ValueName} strings, and plain {@link Schema} objects
61
+ *
62
+ * @template Value - single schema entry
63
+ */
64
+ type InferSchemaEntryValue<Value> = Value extends Constructor<infer Instance> ? Instance : Value extends Schematic<infer Model> ? Model : Value extends SchemaProperty ? InferPropertyType<Value['$type']> : Value extends PlainSchema ? Infer<Value & Schema> : Value extends ValueName ? Values[Value & ValueName] : Value extends Schema ? Infer<Value> : never;
65
+ //#endregion
66
+ export { Infer, InferOptionalKeys, InferPropertyType, InferPropertyValue, InferRequiredKeys, InferSchemaEntry, InferSchemaEntryValue };
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,153 @@
1
+ import { SchemaProperty } from "./schema.plain.model.mjs";
2
+
3
+ //#region src/models/misc.model.d.ts
4
+ /**
5
+ * Removes duplicate types from a tuple, preserving first occurrence order
6
+ *
7
+ * @template Value - Tuple to deduplicate
8
+ * @template Seen - Accumulator for already-seen types _(internal)_
9
+ *
10
+ * @example
11
+ * ```ts
12
+ * // DeduplicateTuple<['string', 'number', 'string']>
13
+ * // => ['string', 'number']
14
+ * ```
15
+ */
16
+ type DeduplicateTuple<Value extends unknown[], Seen extends unknown[] = []> = Value extends [infer Head, ...infer Tail] ? Head extends Seen[number] ? DeduplicateTuple<Tail, Seen> : DeduplicateTuple<Tail, [...Seen, Head]> : Seen;
17
+ /**
18
+ * Recursively extracts {@link ValueName} strings from a type, unwrapping arrays and readonly arrays
19
+ *
20
+ * @template Value - Type to extract value names from
21
+ *
22
+ * @example
23
+ * ```ts
24
+ * // ExtractValueNames<'string'> => 'string'
25
+ * // ExtractValueNames<['string', 'number']> => 'string' | 'number'
26
+ * ```
27
+ */
28
+ type ExtractValueNames<Value> = Value extends ValueName ? Value : Value extends (infer Item)[] ? ExtractValueNames<Item> : Value extends readonly (infer Item)[] ? ExtractValueNames<Item> : never;
29
+ /**
30
+ * Determines whether a schema entry is optional
31
+ *
32
+ * Returns `true` if the entry is a {@link SchemaProperty} or {@link NestedSchema} with `$required` set to `false`; otherwise returns `false`
33
+ *
34
+ * @template Value - Schema entry to check
35
+ */
36
+ type IsOptionalProperty<Value> = Value extends SchemaProperty ? Value['$required'] extends false ? true : false : false;
37
+ /**
38
+ * Extracts the last member from a union type by leveraging intersection of function return types
39
+ *
40
+ * @template Value - Union type
41
+ */
42
+ type LastOfUnion<Value> = UnionToIntersection<Value extends unknown ? () => Value : never> extends (() => infer Item) ? Item : never;
43
+ /**
44
+ * Extracts keys from an object type that are optional
45
+ *
46
+ * @template Value - Object type to inspect
47
+ */
48
+ type OptionalKeys<Value> = { [Key in keyof Value]-?: {} extends Pick<Value, Key> ? Key : never }[keyof Value];
49
+ /**
50
+ * Extracts keys from an object type that are required _(i.e., not optional)_
51
+ *
52
+ * @template Value - Object type to inspect
53
+ */
54
+ type RequiredKeys<Value> = Exclude<keyof Value, OptionalKeys<Value>>;
55
+ /**
56
+ * Generates all permutations of a tuple type
57
+ *
58
+ * Used by {@link UnwrapSingle} to allow schema types in any order for small tuples _(length ≤ 5)_
59
+ *
60
+ * @template Tuple - Tuple to permute
61
+ * @template Elput - Accumulator for the current permutation _(internal; name is Tuple backwards)_
62
+ *
63
+ * @example
64
+ * ```ts
65
+ * // TuplePermutations<['string', 'number']>
66
+ * // => ['string', 'number'] | ['number', 'string']
67
+ * ```
68
+ */
69
+ type TuplePermutations<Tuple extends unknown[], Elput extends unknown[] = []> = Tuple['length'] extends 0 ? Elput : { [Key in keyof Tuple]: TuplePermutations<TupleRemoveAt<Tuple, Key & `${number}`>, [...Elput, Tuple[Key]]> }[keyof Tuple & `${number}`];
70
+ /**
71
+ * Removes the element at a given index from a tuple
72
+ *
73
+ * Used internally by {@link TuplePermutations}
74
+ *
75
+ * @template Items - Tuple to remove from
76
+ * @template Item - Stringified index to remove
77
+ * @template Prefix - Accumulator for elements before the target _(internal)_
78
+ */
79
+ type TupleRemoveAt<Items extends unknown[], Item extends string, Prefix extends unknown[] = []> = Items extends [infer Head, ...infer Tail] ? `${Prefix['length']}` extends Item ? [...Prefix, ...Tail] : TupleRemoveAt<Tail, Item, [...Prefix, Head]> : Prefix;
80
+ /**
81
+ * Converts a union type into an intersection
82
+ *
83
+ * Uses the contravariance of function parameter types to collapse a union into an intersection
84
+ *
85
+ * @template Value - Union type to convert
86
+ *
87
+ * @example
88
+ * ```ts
89
+ * // UnionToIntersection<{ a: 1 } | { b: 2 }>
90
+ * // => { a: 1 } & { b: 2 }
91
+ * ```
92
+ */
93
+ type UnionToIntersection<Value> = (Value extends unknown ? (value: Value) => void : never) extends ((value: infer Item) => void) ? Item : never;
94
+ /**
95
+ * Converts a union type into an ordered tuple
96
+ *
97
+ * Repeatedly extracts the {@link LastOfUnion} member and prepends it to the accumulator
98
+ *
99
+ * @template Value - Union type to convert
100
+ * @template Items - Accumulator for the resulting tuple _(internal)_
101
+ *
102
+ * @example
103
+ * ```ts
104
+ * // UnionToTuple<'a' | 'b' | 'c'>
105
+ * // => ['a', 'b', 'c']
106
+ * ```
107
+ */
108
+ type UnionToTuple<Value, Items extends unknown[] = []> = [Value] extends [never] ? Items : UnionToTuple<Exclude<Value, LastOfUnion<Value>>, [LastOfUnion<Value>, ...Items]>;
109
+ /**
110
+ * Unwraps a single-element tuple to its inner type
111
+ *
112
+ * For tuples of length 2–5, returns all {@link TuplePermutations} to allow types in any order. Longer tuples are returned as-is
113
+ *
114
+ * @template Value - Tuple to potentially unwrap
115
+ *
116
+ * @example
117
+ * ```ts
118
+ * // UnwrapSingle<['string']> => 'string'
119
+ * // UnwrapSingle<['string', 'number']> => ['string', 'number'] | ['number', 'string']
120
+ * ```
121
+ */
122
+ type UnwrapSingle<Value extends unknown[]> = Value extends [infer Only] ? Only : Value['length'] extends 1 | 2 | 3 | 4 | 5 ? TuplePermutations<Value> : Value;
123
+ /**
124
+ * Basic value types
125
+ */
126
+ type ValueName = keyof Values;
127
+ /**
128
+ * Maps type name strings to their TypeScript equivalents
129
+ *
130
+ * Used by the type system to resolve {@link ValueName} strings into actual types
131
+ *
132
+ * @example
133
+ * ```ts
134
+ * // Values['string'] => string
135
+ * // Values['date'] => Date
136
+ * // Values['null'] => null
137
+ * ```
138
+ */
139
+ type Values = {
140
+ array: unknown[];
141
+ bigint: bigint;
142
+ boolean: boolean;
143
+ date: Date;
144
+ function: Function;
145
+ null: null;
146
+ number: number;
147
+ object: object;
148
+ string: string;
149
+ symbol: symbol;
150
+ undefined: undefined;
151
+ };
152
+ //#endregion
153
+ export { DeduplicateTuple, ExtractValueNames, IsOptionalProperty, LastOfUnion, OptionalKeys, RequiredKeys, TuplePermutations, TupleRemoveAt, UnionToIntersection, UnionToTuple, UnwrapSingle, ValueName, Values };
@@ -0,0 +1 @@
1
+ export {};