@decaf-ts/decorator-validation 1.7.7 → 1.7.8
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.
- package/dist/decorator-validation.cjs +629 -254
- package/dist/decorator-validation.esm.cjs +622 -255
- package/lib/constants/validation.cjs +3 -2
- package/lib/constants/validation.d.ts +1 -0
- package/lib/esm/constants/validation.d.ts +1 -0
- package/lib/esm/constants/validation.js +2 -1
- package/lib/esm/index.d.ts +1 -1
- package/lib/esm/index.js +1 -1
- package/lib/esm/mcp/ModelContextProtocol.d.ts +14 -22
- package/lib/esm/mcp/ModelContextProtocol.js +55 -51
- package/lib/esm/model/Model.d.ts +11 -8
- package/lib/esm/model/Model.js +12 -6
- package/lib/esm/model/decorators.js +1 -1
- package/lib/esm/model/types.d.ts +37 -9
- package/lib/esm/model/types.js +1 -1
- package/lib/esm/model/utils.d.ts +14 -1
- package/lib/esm/model/utils.js +15 -1
- package/lib/esm/model/validation.d.ts +107 -5
- package/lib/esm/model/validation.js +329 -124
- package/lib/esm/types/index.d.ts +1 -0
- package/lib/esm/types/index.js +2 -0
- package/lib/esm/types/validation.d.ts +25 -0
- package/lib/esm/types/validation.js +2 -0
- package/lib/esm/validation/Validators/AsyncValidator.d.ts +72 -0
- package/lib/esm/validation/Validators/AsyncValidator.js +61 -0
- package/lib/esm/validation/Validators/BaseValidator.d.ts +118 -0
- package/lib/esm/validation/Validators/BaseValidator.js +117 -0
- package/lib/esm/validation/Validators/ListValidator.js +2 -2
- package/lib/esm/validation/Validators/MinLengthValidator.d.ts +1 -1
- package/lib/esm/validation/Validators/MinLengthValidator.js +2 -2
- package/lib/esm/validation/Validators/TypeValidator.js +1 -1
- package/lib/esm/validation/Validators/Validator.d.ts +28 -68
- package/lib/esm/validation/Validators/Validator.js +30 -86
- package/lib/esm/validation/Validators/constants.d.ts +12 -11
- package/lib/esm/validation/Validators/constants.js +14 -12
- package/lib/esm/validation/Validators/index.d.ts +1 -0
- package/lib/esm/validation/Validators/index.js +2 -1
- package/lib/esm/validation/decorators.d.ts +2 -1
- package/lib/esm/validation/decorators.js +31 -8
- package/lib/esm/validation/types.d.ts +19 -0
- package/lib/esm/validation/types.js +1 -1
- package/lib/index.cjs +1 -1
- package/lib/index.d.ts +1 -1
- package/lib/mcp/ModelContextProtocol.cjs +55 -51
- package/lib/mcp/ModelContextProtocol.d.ts +14 -22
- package/lib/model/Model.cjs +11 -5
- package/lib/model/Model.d.ts +11 -8
- package/lib/model/decorators.cjs +1 -1
- package/lib/model/types.cjs +1 -1
- package/lib/model/types.d.ts +37 -9
- package/lib/model/utils.cjs +16 -1
- package/lib/model/utils.d.ts +14 -1
- package/lib/model/validation.cjs +334 -125
- package/lib/model/validation.d.ts +107 -5
- package/lib/types/index.cjs +18 -0
- package/lib/types/index.d.ts +1 -0
- package/lib/types/validation.cjs +3 -0
- package/lib/types/validation.d.ts +25 -0
- package/lib/validation/Validators/AsyncValidator.cjs +65 -0
- package/lib/validation/Validators/AsyncValidator.d.ts +72 -0
- package/lib/validation/Validators/BaseValidator.cjs +121 -0
- package/lib/validation/Validators/BaseValidator.d.ts +118 -0
- package/lib/validation/Validators/ListValidator.cjs +2 -2
- package/lib/validation/Validators/MinLengthValidator.cjs +2 -2
- package/lib/validation/Validators/MinLengthValidator.d.ts +1 -1
- package/lib/validation/Validators/TypeValidator.cjs +1 -1
- package/lib/validation/Validators/Validator.cjs +30 -86
- package/lib/validation/Validators/Validator.d.ts +28 -68
- package/lib/validation/Validators/constants.cjs +14 -12
- package/lib/validation/Validators/constants.d.ts +12 -11
- package/lib/validation/Validators/index.cjs +2 -1
- package/lib/validation/Validators/index.d.ts +1 -0
- package/lib/validation/decorators.cjs +32 -8
- package/lib/validation/decorators.d.ts +2 -1
- package/lib/validation/types.cjs +1 -1
- package/lib/validation/types.d.ts +19 -0
- package/package.json +1 -1
|
@@ -1,149 +1,354 @@
|
|
|
1
1
|
import { ModelErrorDefinition } from "./ModelErrorDefinition.js";
|
|
2
|
-
import { Reflection, } from "@decaf-ts/reflection";
|
|
3
2
|
import { ModelKeys } from "./../utils/constants.js";
|
|
4
|
-
import { ReservedModels } from "./constants.js";
|
|
5
|
-
import { VALIDATION_PARENT_KEY } from "./../constants/index.js";
|
|
6
3
|
import { Model } from "./Model.js";
|
|
7
4
|
import { Validation } from "./../validation/Validation.js";
|
|
8
5
|
import { ValidationKeys } from "./../validation/Validators/constants.js";
|
|
9
6
|
import { PathProxyEngine } from "./../utils/PathProxy.js";
|
|
7
|
+
import { ASYNC_META_KEY, VALIDATION_PARENT_KEY } from "./../constants/index.js";
|
|
8
|
+
import { Reflection } from "@decaf-ts/reflection";
|
|
9
|
+
import { toConditionalPromise } from "./utils.js";
|
|
10
10
|
/**
|
|
11
|
-
*
|
|
11
|
+
* Retrieves the validation metadata decorators associated with a specific property of a model,
|
|
12
|
+
* using the reflective metadata key.
|
|
12
13
|
*
|
|
13
|
-
* @
|
|
14
|
-
* @
|
|
15
|
-
* @
|
|
14
|
+
* @param model - The model instance or class containing the decorated property.
|
|
15
|
+
* @param {string} prop - The name of the property whose decorators should be retrieved.
|
|
16
|
+
* @param {string} reflectKey - The metadata key used to retrieve the decorators.
|
|
17
|
+
* Defaults to `ValidationKeys.REFLECT`.
|
|
16
18
|
*
|
|
19
|
+
* @returns The validation decorators applied to the property
|
|
20
|
+
*/
|
|
21
|
+
export function getValidationDecorators(model, prop, reflectKey = ValidationKeys.REFLECT) {
|
|
22
|
+
return Reflection.getPropertyDecorators(reflectKey, model, prop);
|
|
23
|
+
}
|
|
24
|
+
/**
|
|
25
|
+
* @description
|
|
26
|
+
* Retrieves all validatable property decorators from a given model, excluding specified properties.
|
|
27
|
+
*
|
|
28
|
+
* @summary
|
|
29
|
+
* Iterates through the own enumerable properties of a model instance, filtering out any properties
|
|
30
|
+
* listed in the `propsToIgnore` array. For each remaining property, it checks whether validation
|
|
31
|
+
* decorators are present using `getValidationDecorators`, and if so, collects them in the result array.
|
|
32
|
+
*
|
|
33
|
+
* @template M - A generic parameter extending the `Model` class, representing the model type being inspected.
|
|
34
|
+
*
|
|
35
|
+
* @param {M} model - An instance of a class extending `Model` from which validatable properties will be extracted.
|
|
36
|
+
* @param {string[]} propsToIgnore - An array of property names that should be excluded from validation inspection.
|
|
37
|
+
*
|
|
38
|
+
* @return {ValidationPropertyDecoratorDefinition[]} An array of validation decorator definitions
|
|
39
|
+
* associated with the model's properties, excluding those listed in `propsToIgnore`.
|
|
40
|
+
*
|
|
41
|
+
* @function getValidatableProperties
|
|
42
|
+
*/
|
|
43
|
+
export function getValidatableProperties(model, propsToIgnore) {
|
|
44
|
+
const decoratedProperties = [];
|
|
45
|
+
for (const prop in model) {
|
|
46
|
+
if (Object.prototype.hasOwnProperty.call(model, prop) &&
|
|
47
|
+
!propsToIgnore.includes(prop)) {
|
|
48
|
+
const dec = getValidationDecorators(model, prop);
|
|
49
|
+
if (dec)
|
|
50
|
+
decoratedProperties.push(dec);
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
return decoratedProperties;
|
|
54
|
+
}
|
|
55
|
+
/**
|
|
56
|
+
* Safely sets temporary metadata on an object
|
|
57
|
+
*/
|
|
58
|
+
function setTemporaryContext(target, key, value) {
|
|
59
|
+
if (!Object.hasOwnProperty.call(target, key))
|
|
60
|
+
target[key] = value;
|
|
61
|
+
}
|
|
62
|
+
/**
|
|
63
|
+
* Safely removes temporary metadata from an object
|
|
64
|
+
*/
|
|
65
|
+
function cleanupTemporaryContext(target, key) {
|
|
66
|
+
if (Object.hasOwnProperty.call(target, key))
|
|
67
|
+
delete target[key];
|
|
68
|
+
}
|
|
69
|
+
/**
|
|
70
|
+
* Executes validation with temporary context and returns the validation result
|
|
71
|
+
*
|
|
72
|
+
* @param nestedModel - The instance to validate
|
|
73
|
+
* @param parentModel - Reference to a parent object for nested validation
|
|
74
|
+
* @param isAsync - Whether to perform async validation
|
|
75
|
+
* @returns Validation result from hasErrors()
|
|
76
|
+
*/
|
|
77
|
+
function getNestedValidationErrors(nestedModel, parentModel, isAsync) {
|
|
78
|
+
// Set temporary context for nested models
|
|
79
|
+
if (parentModel) {
|
|
80
|
+
setTemporaryContext(nestedModel, VALIDATION_PARENT_KEY, parentModel);
|
|
81
|
+
}
|
|
82
|
+
setTemporaryContext(nestedModel, ASYNC_META_KEY, !!isAsync);
|
|
83
|
+
const errs = nestedModel.hasErrors();
|
|
84
|
+
cleanupTemporaryContext(nestedModel, VALIDATION_PARENT_KEY);
|
|
85
|
+
cleanupTemporaryContext(nestedModel, ASYNC_META_KEY);
|
|
86
|
+
return errs;
|
|
87
|
+
}
|
|
88
|
+
export function validateDecorator(model, value, decorator, async) {
|
|
89
|
+
const validator = Validation.get(decorator.key);
|
|
90
|
+
if (!validator) {
|
|
91
|
+
throw new Error(`Missing validator for ${decorator.key}`);
|
|
92
|
+
}
|
|
93
|
+
// skip async decorators if validateDecorators is called synchronously (async = false)
|
|
94
|
+
if (!async && decorator.props.async)
|
|
95
|
+
return undefined;
|
|
96
|
+
const decoratorProps = decorator.key === ModelKeys.TYPE
|
|
97
|
+
? [decorator.props]
|
|
98
|
+
: decorator.props || {};
|
|
99
|
+
const context = PathProxyEngine.create(model, {
|
|
100
|
+
ignoreUndefined: true,
|
|
101
|
+
ignoreNull: true,
|
|
102
|
+
});
|
|
103
|
+
const maybeAsyncErrors = validator.hasErrors(value, decoratorProps, context);
|
|
104
|
+
return toConditionalPromise(maybeAsyncErrors, async);
|
|
105
|
+
}
|
|
106
|
+
/**
|
|
107
|
+
* @description
|
|
108
|
+
* Executes validation logic for a set of decorators applied to a model's property, handling both
|
|
109
|
+
* synchronous and asynchronous validations, including support for nested validations and lists.
|
|
110
|
+
*
|
|
111
|
+
* @summary
|
|
112
|
+
* Iterates over an array of decorator metadata objects and applies each validation rule to the
|
|
113
|
+
* provided value. For list decorators (`ValidationKeys.LIST`), it performs element-wise validation,
|
|
114
|
+
* supporting nested model validation and type checks. If the `async` flag is set, asynchronous
|
|
115
|
+
* validation is supported using `Promise.all`. The result is a record mapping validation keys to
|
|
116
|
+
* error messages, or `undefined` if no errors are found.
|
|
117
|
+
*
|
|
118
|
+
* @template M - A type parameter extending `Model`, representing the model type being validated.
|
|
119
|
+
* @template Async - A boolean indicating whether validation should be performed asynchronously.
|
|
120
|
+
*
|
|
121
|
+
* @param {M} model - The model instance that the validation is associated with.
|
|
122
|
+
* @param {any} value - The value to be validated against the provided decorators.
|
|
123
|
+
* @param {DecoratorMetadataAsync[]} decorators - An array of metadata objects representing validation decorators.
|
|
124
|
+
* @param {Async} [async] - Optional flag indicating whether validation should be performed asynchronously.
|
|
125
|
+
*
|
|
126
|
+
* @return {ConditionalAsync<Async, Record<string, string>> | undefined}
|
|
127
|
+
* Returns either a record of validation errors (keyed by the decorator key) or `undefined` if no errors are found.
|
|
128
|
+
* If `async` is true, the return value is a Promise resolving to the same structure.
|
|
129
|
+
*
|
|
130
|
+
* @function validateDecorators
|
|
131
|
+
*/
|
|
132
|
+
export function validateDecorators(model, value, decorators, async) {
|
|
133
|
+
const result = {};
|
|
134
|
+
for (const decorator of decorators) {
|
|
135
|
+
// skip async decorators if validateDecorators is called synchronously (async = false)
|
|
136
|
+
if (!async && decorator.props.async)
|
|
137
|
+
continue;
|
|
138
|
+
let validationErrors = validateDecorator(model, value, decorator, async);
|
|
139
|
+
/*
|
|
140
|
+
If the decorator is a list, each element must be checked.
|
|
141
|
+
When 'async' is true, the 'err' will always be a pending promise initially,
|
|
142
|
+
so the '!err' check will evaluate to false (even if the promise later resolves with no errors)
|
|
143
|
+
*/
|
|
144
|
+
if (decorator.key === ValidationKeys.LIST && (!validationErrors || async)) {
|
|
145
|
+
const values = value instanceof Set ? [...value] : value;
|
|
146
|
+
if (values && values.length > 0) {
|
|
147
|
+
const types = decorator.props.class ||
|
|
148
|
+
decorator.props.clazz ||
|
|
149
|
+
decorator.props.customTypes;
|
|
150
|
+
const allowedTypes = [types].flat().map((t) => String(t).toLowerCase());
|
|
151
|
+
// const reserved = Object.values(ReservedModels).map((v) => v.toLowerCase()) as string[];
|
|
152
|
+
const errs = values.map((childValue) => {
|
|
153
|
+
// if (Model.isModel(v) && !reserved.includes(v) {
|
|
154
|
+
if (Model.isModel(childValue)) {
|
|
155
|
+
return getNestedValidationErrors(childValue, model, async);
|
|
156
|
+
}
|
|
157
|
+
return allowedTypes.includes(typeof childValue)
|
|
158
|
+
? undefined
|
|
159
|
+
: "Value has no validatable type";
|
|
160
|
+
});
|
|
161
|
+
if (async) {
|
|
162
|
+
validationErrors = Promise.all(errs).then((result) => {
|
|
163
|
+
const allEmpty = result.every((r) => !r);
|
|
164
|
+
return allEmpty ? undefined : result;
|
|
165
|
+
});
|
|
166
|
+
}
|
|
167
|
+
else {
|
|
168
|
+
const allEmpty = errs.every((r) => !r);
|
|
169
|
+
validationErrors = errs.length > 0 && !allEmpty ? errs : undefined;
|
|
170
|
+
}
|
|
171
|
+
}
|
|
172
|
+
}
|
|
173
|
+
if (validationErrors)
|
|
174
|
+
result[decorator.key] = validationErrors;
|
|
175
|
+
}
|
|
176
|
+
if (!async)
|
|
177
|
+
return Object.keys(result).length > 0
|
|
178
|
+
? result
|
|
179
|
+
: undefined;
|
|
180
|
+
const keys = Object.keys(result);
|
|
181
|
+
const promises = Object.values(result);
|
|
182
|
+
return Promise.all(promises).then((resolvedValues) => {
|
|
183
|
+
const res = {};
|
|
184
|
+
for (let i = 0; i < resolvedValues.length; i++) {
|
|
185
|
+
const val = resolvedValues[i];
|
|
186
|
+
if (val !== undefined) {
|
|
187
|
+
res[keys[i]] = val;
|
|
188
|
+
}
|
|
189
|
+
}
|
|
190
|
+
return Object.keys(res).length > 0 ? res : undefined;
|
|
191
|
+
});
|
|
192
|
+
}
|
|
193
|
+
/**
|
|
17
194
|
* @function validate
|
|
195
|
+
* @template M
|
|
196
|
+
* @template Async
|
|
18
197
|
* @memberOf module:decorator-validation
|
|
19
198
|
* @category Model
|
|
199
|
+
*
|
|
200
|
+
* @description
|
|
201
|
+
* Validates the properties of a {@link Model} instance using registered decorators.
|
|
202
|
+
* Supports both synchronous and asynchronous validation flows, depending on the `async` flag.
|
|
203
|
+
*
|
|
204
|
+
* @summary
|
|
205
|
+
* This function inspects a given model object, identifies decorated properties that require validation,
|
|
206
|
+
* and applies the corresponding validation rules. It also supports nested model validation and gracefully
|
|
207
|
+
* merges any validation errors. For collections (Array/Set), it enforces the presence of the `@list` decorator
|
|
208
|
+
* and checks the type of elements. If a property is a nested model, it will call `hasErrors` on it and flatten
|
|
209
|
+
* the nested error keys using dot notation.
|
|
210
|
+
*
|
|
211
|
+
* @param {M} model - The model instance to be validated. Must extend from {@link Model}.
|
|
212
|
+
* @param {Async} [async] - A flag indicating whether validation should be asynchronous.
|
|
213
|
+
* @param {...string} propsToIgnore - A variadic list of property names that should be skipped during validation.
|
|
214
|
+
*
|
|
215
|
+
* @returns {ConditionalAsync<Async, ModelErrorDefinition | undefined>}
|
|
216
|
+
* Returns either a {@link ModelErrorDefinition} containing validation errors,
|
|
217
|
+
* or `undefined` if no errors are found. When `async` is `true`, returns a Promise.
|
|
218
|
+
*
|
|
219
|
+
* @see {@link Model}
|
|
220
|
+
* @see {@link ModelErrorDefinition}
|
|
221
|
+
* @see {@link validateDecorators}
|
|
222
|
+
* @see {@link getValidatableProperties}
|
|
223
|
+
*
|
|
224
|
+
* @mermaid
|
|
225
|
+
* sequenceDiagram
|
|
226
|
+
* participant Caller
|
|
227
|
+
* participant validate
|
|
228
|
+
* participant getValidatableProperties
|
|
229
|
+
* participant validateDecorators
|
|
230
|
+
* participant ModelInstance
|
|
231
|
+
* Caller->>validate: call with obj, async, propsToIgnore
|
|
232
|
+
* validate->>getValidatableProperties: retrieve decorated props
|
|
233
|
+
* loop for each property
|
|
234
|
+
* validate->>validateDecorators: validate using decorators
|
|
235
|
+
* alt is nested model
|
|
236
|
+
* validate->>ModelInstance: call hasErrors()
|
|
237
|
+
* end
|
|
238
|
+
* end
|
|
239
|
+
* alt async
|
|
240
|
+
* validate->>validate: Promise.allSettled for errors
|
|
241
|
+
* end
|
|
242
|
+
* validate-->>Caller: return ModelErrorDefinition | undefined
|
|
20
243
|
*/
|
|
21
|
-
export function validate(
|
|
22
|
-
const decoratedProperties =
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
244
|
+
export function validate(model, async, ...propsToIgnore) {
|
|
245
|
+
const decoratedProperties = getValidatableProperties(model, propsToIgnore);
|
|
246
|
+
const result = {};
|
|
247
|
+
const nestedErrors = {};
|
|
248
|
+
for (const { prop, decorators } of decoratedProperties) {
|
|
249
|
+
const propKey = String(prop);
|
|
250
|
+
let propValue = model[prop];
|
|
251
|
+
if (!decorators?.length)
|
|
252
|
+
continue;
|
|
253
|
+
// Get the default type validator
|
|
254
|
+
const designTypeDec = decorators.find((d) => {
|
|
255
|
+
return [ModelKeys.TYPE, ValidationKeys.TYPE].includes(d.key);
|
|
256
|
+
});
|
|
257
|
+
if (!designTypeDec)
|
|
34
258
|
continue;
|
|
35
|
-
const
|
|
36
|
-
//
|
|
37
|
-
if (
|
|
38
|
-
if (d.key === ValidationKeys.
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
259
|
+
const designType = designTypeDec.props.name;
|
|
260
|
+
// Handle array or Set types and enforce the presence of @list decorator
|
|
261
|
+
if ([Array.name, Set.name].includes(designType)) {
|
|
262
|
+
if (!decorators.some((d) => d.key === ValidationKeys.LIST)) {
|
|
263
|
+
result[propKey] = {
|
|
264
|
+
[ValidationKeys.TYPE]: `Array or Set property '${propKey}' requires a @list decorator`,
|
|
265
|
+
};
|
|
266
|
+
continue;
|
|
267
|
+
}
|
|
268
|
+
if (propValue &&
|
|
269
|
+
!(Array.isArray(propValue) || propValue instanceof Set)) {
|
|
270
|
+
result[propKey] = {
|
|
271
|
+
[ValidationKeys.TYPE]: `Property '${String(prop)}' must be either an Array or a Set`,
|
|
272
|
+
};
|
|
273
|
+
continue;
|
|
274
|
+
}
|
|
275
|
+
// Remove design:type decorator, since @list decorator already ensures type
|
|
276
|
+
for (let i = decorators.length - 1; i >= 0; i--) {
|
|
277
|
+
if (decorators[i].key === ModelKeys.TYPE) {
|
|
278
|
+
decorators.splice(i, 1);
|
|
279
|
+
}
|
|
280
|
+
}
|
|
281
|
+
propValue = propValue instanceof Set ? [...propValue] : propValue;
|
|
43
282
|
}
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
283
|
+
const propErrors = validateDecorators(model, propValue, decorators, async) || {};
|
|
284
|
+
// Check for nested properties.
|
|
285
|
+
// To prevent unnecessary processing, "propValue" must be defined and validatable
|
|
286
|
+
// let nestedErrors: Record<string, any> = {};
|
|
287
|
+
const isConstr = Model.isPropertyModel(model, propKey);
|
|
288
|
+
// if propValue !== undefined, null
|
|
289
|
+
if (propValue && isConstr) {
|
|
290
|
+
const instance = propValue;
|
|
291
|
+
const isInvalidModel = typeof instance !== "object" ||
|
|
292
|
+
!instance.hasErrors ||
|
|
293
|
+
typeof instance.hasErrors !== "function";
|
|
294
|
+
if (isInvalidModel) {
|
|
295
|
+
// propErrors[ValidationKeys.TYPE] = "Model should be validatable but it's not.";
|
|
296
|
+
console.warn("Model should be validatable but it's not.");
|
|
49
297
|
}
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
: decorator.props || {};
|
|
53
|
-
const err = validator.hasErrors(obj[prop.toString()], decoratorProps, PathProxyEngine.create(obj, { ignoreUndefined: true, ignoreNull: true }));
|
|
54
|
-
if (err) {
|
|
55
|
-
errs = errs || {};
|
|
56
|
-
errs[decorator.key] = err;
|
|
298
|
+
else {
|
|
299
|
+
nestedErrors[propKey] = getNestedValidationErrors(instance, model, async);
|
|
57
300
|
}
|
|
58
301
|
}
|
|
59
|
-
if
|
|
60
|
-
|
|
61
|
-
|
|
302
|
+
// Add to the result if we have any errors
|
|
303
|
+
// Async mode returns a Promise that resolves to undefined when no errors exist
|
|
304
|
+
if (Object.keys(propErrors).length > 0 || async)
|
|
305
|
+
result[propKey] = propErrors;
|
|
306
|
+
// Then merge any nested errors
|
|
307
|
+
if (!async) {
|
|
308
|
+
Object.entries(nestedErrors[propKey] || {}).forEach(([key, error]) => {
|
|
309
|
+
if (error !== undefined) {
|
|
310
|
+
result[`${propKey}.${key}`] = error;
|
|
311
|
+
}
|
|
312
|
+
});
|
|
62
313
|
}
|
|
63
314
|
}
|
|
64
|
-
//
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
? ValidationKeys.LIST
|
|
83
|
-
: ValidationKeys.TYPE;
|
|
84
|
-
const types = allDecorators.find((d) => d.key === typeDecoratorKey) || {};
|
|
85
|
-
let allowedTypes = [];
|
|
86
|
-
if (types && types.props) {
|
|
87
|
-
const customTypes = Array.isArray(obj[prop])
|
|
88
|
-
? types.props.class
|
|
89
|
-
: types.props.customTypes;
|
|
90
|
-
if (customTypes)
|
|
91
|
-
allowedTypes = Array.isArray(customTypes)
|
|
92
|
-
? customTypes.map((t) => `${t}`.toLowerCase())
|
|
93
|
-
: [customTypes.toLowerCase()];
|
|
94
|
-
}
|
|
95
|
-
const validate = (prop, value) => {
|
|
96
|
-
if (typeof value !== "object" && typeof value !== "function")
|
|
97
|
-
return undefined;
|
|
98
|
-
try {
|
|
99
|
-
if (value && !value[VALIDATION_PARENT_KEY])
|
|
100
|
-
value[VALIDATION_PARENT_KEY] = obj; // TODO: freeze?
|
|
101
|
-
return Model.isModel(value)
|
|
102
|
-
? value.hasErrors()
|
|
103
|
-
: allowedTypes.includes(typeof value)
|
|
104
|
-
? undefined
|
|
105
|
-
: "Value has no validatable type";
|
|
106
|
-
}
|
|
107
|
-
finally {
|
|
108
|
-
if (value && value[VALIDATION_PARENT_KEY])
|
|
109
|
-
delete value[VALIDATION_PARENT_KEY];
|
|
315
|
+
// Synchronous return
|
|
316
|
+
if (!async) {
|
|
317
|
+
return (Object.keys(result).length > 0
|
|
318
|
+
? new ModelErrorDefinition(result)
|
|
319
|
+
: undefined);
|
|
320
|
+
}
|
|
321
|
+
const merged = result; // TODO: apply filtering
|
|
322
|
+
const keys = Object.keys(merged);
|
|
323
|
+
const promises = Object.values(merged);
|
|
324
|
+
return Promise.allSettled(promises).then(async (results) => {
|
|
325
|
+
const result = {};
|
|
326
|
+
for (const [parentProp, nestedErrPromise] of Object.entries(nestedErrors)) {
|
|
327
|
+
const nestedPropDecErrors = (await nestedErrPromise);
|
|
328
|
+
if (nestedPropDecErrors)
|
|
329
|
+
Object.entries(nestedPropDecErrors).forEach(([nestedProp, nestedPropDecError]) => {
|
|
330
|
+
if (nestedPropDecError !== undefined) {
|
|
331
|
+
const nestedKey = [parentProp, nestedProp].join(".");
|
|
332
|
+
result[nestedKey] = nestedPropDecError;
|
|
110
333
|
}
|
|
111
|
-
};
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
err = (c === Array.name
|
|
119
|
-
? obj[prop]
|
|
120
|
-
: // If it's a Set
|
|
121
|
-
obj[prop].values())
|
|
122
|
-
.map((v) => validate(prop, v))
|
|
123
|
-
.filter((e) => !!e);
|
|
124
|
-
if (!err?.length) {
|
|
125
|
-
// if the result is an empty list...
|
|
126
|
-
err = undefined;
|
|
127
|
-
}
|
|
128
|
-
}
|
|
129
|
-
}
|
|
130
|
-
break;
|
|
131
|
-
default:
|
|
132
|
-
try {
|
|
133
|
-
if (obj[prop])
|
|
134
|
-
err = validate(prop, obj[prop]);
|
|
135
|
-
}
|
|
136
|
-
catch (e) {
|
|
137
|
-
console.warn(`Model should be validatable but its not: ${e}`);
|
|
138
|
-
}
|
|
139
|
-
}
|
|
334
|
+
});
|
|
335
|
+
}
|
|
336
|
+
for (let i = 0; i < results.length; i++) {
|
|
337
|
+
const key = keys[i];
|
|
338
|
+
const res = results[i];
|
|
339
|
+
if (res.status === "fulfilled" && res.value !== undefined) {
|
|
340
|
+
result[key] = res.value;
|
|
140
341
|
}
|
|
141
|
-
if (
|
|
142
|
-
result =
|
|
143
|
-
|
|
342
|
+
else if (res.status === "rejected") {
|
|
343
|
+
result[key] =
|
|
344
|
+
res.reason instanceof Error
|
|
345
|
+
? res.reason.message
|
|
346
|
+
: String(res.reason || "Validation failed");
|
|
144
347
|
}
|
|
145
348
|
}
|
|
146
|
-
|
|
147
|
-
|
|
349
|
+
return Object.keys(result).length > 0
|
|
350
|
+
? new ModelErrorDefinition(result)
|
|
351
|
+
: undefined;
|
|
352
|
+
});
|
|
148
353
|
}
|
|
149
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"validation.js","sourceRoot":"","sources":["../../../src/model/validation.ts"],"names":[],"mappings":"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","sourcesContent":["import { ModelErrorDefinition } from \"./ModelErrorDefinition\";\nimport {\n  DecoratorMetadata,\n  Reflection,\n  FullPropertyDecoratorList,\n} from \"@decaf-ts/reflection\";\nimport { ModelKeys } from \"../utils/constants\";\nimport { ReservedModels } from \"./constants\";\nimport { VALIDATION_PARENT_KEY } from \"../constants\";\nimport { Validatable } from \"./types\";\nimport { Model } from \"./Model\";\nimport { Validation } from \"../validation/Validation\";\nimport { ValidationKeys } from \"../validation/Validators/constants\";\nimport {\n  ModelErrors,\n  ValidationPropertyDecoratorDefinition,\n  ValidatorOptions,\n} from \"../validation/types\";\nimport { PathProxyEngine } from \"../utils/PathProxy\";\n\n/**\n * @summary Analyses the decorations of the properties and validates the obj according to them\n *\n * @typedef M extends Model\n * @prop {M} obj Model object to validate\n * @prop {string[]} [propsToIgnore] object properties to ignore in the validation\n *\n * @function validate\n * @memberOf module:decorator-validation\n * @category Model\n */\nexport function validate<M extends Model>(\n  obj: M,\n  ...propsToIgnore: string[]\n): ModelErrorDefinition | undefined {\n  const decoratedProperties: ValidationPropertyDecoratorDefinition[] = [];\n  for (const prop in obj)\n    if (\n      Object.prototype.hasOwnProperty.call(obj, prop) &&\n      propsToIgnore.indexOf(prop) === -1\n    ) {\n      decoratedProperties.push(\n        // @ts-ignore\n        Reflection.getPropertyDecorators(\n          ValidationKeys.REFLECT,\n          obj,\n          prop\n        ) as FullPropertyDecoratorList\n      );\n    }\n\n  let result: ModelErrors | undefined = undefined;\n\n  for (const decoratedProperty of decoratedProperties) {\n    const { prop, decorators } = decoratedProperty;\n\n    if (!decorators || !decorators.length) continue;\n\n    const defaultTypeDecorator: DecoratorMetadata = decorators[0];\n\n    // tries to find any type decorators or other decorators that already enforce type (the ones with the allowed types property defined). if so, skip the default type verification\n    if (\n      decorators.find((d) => {\n        if (d.key === ValidationKeys.TYPE) return true;\n        return !!d.props.types?.find(\n          (t) => t === defaultTypeDecorator.props.name\n        );\n      })\n    ) {\n      decorators.shift(); // remove the design:type decorator, since the type will already be checked\n    }\n\n    let errs: Record<string, string | undefined> | undefined = undefined;\n\n    for (const decorator of decorators) {\n      const validator = Validation.get(decorator.key);\n      if (!validator) {\n        throw new Error(`Missing validator for ${decorator.key}`);\n      }\n\n      const decoratorProps =\n        decorator.key === ModelKeys.TYPE\n          ? [decorator.props]\n          : decorator.props || {};\n\n      const err: string | undefined = validator.hasErrors(\n        (obj as any)[prop.toString()],\n        decoratorProps as ValidatorOptions,\n        PathProxyEngine.create(obj, { ignoreUndefined: true, ignoreNull: true })\n      );\n\n      if (err) {\n        errs = errs || {};\n        errs[decorator.key] = err;\n      }\n    }\n\n    if (errs) {\n      result = result || {};\n      result[decoratedProperty.prop.toString()] = errs;\n    }\n  }\n\n  // tests nested classes\n  for (const prop of Object.keys(obj).filter((k) => !result || !result[k])) {\n    let err: string | undefined;\n    // if a nested Model\n    const allDecorators = Reflection.getPropertyDecorators(\n      ValidationKeys.REFLECT,\n      obj,\n      prop\n    ).decorators;\n    const decorators = Reflection.getPropertyDecorators(\n      ValidationKeys.REFLECT,\n      obj,\n      prop\n    ).decorators.filter(\n      (d: { key: string }) =>\n        [ModelKeys.TYPE, ValidationKeys.TYPE as string].indexOf(d.key) !== -1\n    );\n    if (!decorators || !decorators.length) continue;\n    const dec = decorators.pop() as DecoratorMetadata;\n    const clazz = dec.props.name\n      ? [dec.props.name]\n      : Array.isArray(dec.props.customTypes)\n        ? dec.props.customTypes\n        : [dec.props.customTypes];\n    const reserved = Object.values(ReservedModels).map((v) =>\n      v.toLowerCase()\n    ) as string[];\n\n    for (const c of clazz) {\n      if (reserved.indexOf(c.toLowerCase()) === -1) {\n        const typeDecoratorKey = Array.isArray((obj as any)[prop])\n          ? ValidationKeys.LIST\n          : ValidationKeys.TYPE;\n        const types: any =\n          allDecorators.find(\n            (d: { key: string }) => d.key === typeDecoratorKey\n          ) || {};\n        let allowedTypes: string[] = [];\n        if (types && types.props) {\n          const customTypes = Array.isArray((obj as any)[prop])\n            ? types.props.class\n            : types.props.customTypes;\n          if (customTypes)\n            allowedTypes = Array.isArray(customTypes)\n              ? customTypes.map((t) => `${t}`.toLowerCase())\n              : [customTypes.toLowerCase()];\n        }\n\n        const validate = (prop: string, value: any): any => {\n          if (typeof value !== \"object\" && typeof value !== \"function\")\n            return undefined;\n\n          try {\n            if (value && !value[VALIDATION_PARENT_KEY])\n              value[VALIDATION_PARENT_KEY] = obj; // TODO: freeze?\n\n            return Model.isModel(value)\n              ? value.hasErrors()\n              : allowedTypes.includes(typeof value)\n                ? undefined\n                : \"Value has no validatable type\";\n          } finally {\n            if (value && value[VALIDATION_PARENT_KEY])\n              delete value[VALIDATION_PARENT_KEY];\n          }\n        };\n\n        switch (c) {\n          case Array.name:\n          case Set.name:\n            if (allDecorators.length) {\n              const listDec = allDecorators.find(\n                (d: { key: string }) => d.key === ValidationKeys.LIST\n              );\n              if (listDec) {\n                err = (\n                  c === Array.name\n                    ? (obj as Record<string, any>)[prop]\n                    : // If it's a Set\n                      (obj as Record<string, any>)[prop].values()\n                )\n                  .map((v: Validatable) => validate(prop, v))\n                  .filter((e: any) => !!e) as any;\n                if (!err?.length) {\n                  // if the result is an empty list...\n                  err = undefined;\n                }\n              }\n            }\n            break;\n          default:\n            try {\n              if ((obj as Record<string, any>)[prop])\n                err = validate(prop, (obj as any)[prop]);\n            } catch (e: unknown) {\n              console.warn(`Model should be validatable but its not: ${e}`);\n            }\n        }\n      }\n      if (err) {\n        result = result || {};\n        result[prop] = err as any;\n      }\n    }\n  }\n\n  return result ? new ModelErrorDefinition(result) : undefined;\n}\n"]}
|
|
354
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"validation.js","sourceRoot":"","sources":["../../../src/model/validation.ts"],"names":[],"mappings":"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","sourcesContent":["import { ModelErrorDefinition } from \"./ModelErrorDefinition\";\nimport { ModelKeys } from \"../utils/constants\";\nimport { Model } from \"./Model\";\nimport { Validation } from \"../validation/Validation\";\nimport { ValidationKeys } from \"../validation/Validators/constants\";\nimport {\n  ModelErrors,\n  ValidationPropertyDecoratorDefinition,\n  ValidatorOptions,\n} from \"../validation\";\nimport { PathProxyEngine } from \"../utils/PathProxy\";\nimport { ASYNC_META_KEY, VALIDATION_PARENT_KEY } from \"../constants\";\nimport { ConditionalAsync, DecoratorMetadataAsync } from \"../types\";\nimport { Reflection } from \"@decaf-ts/reflection\";\nimport { toConditionalPromise } from \"./utils\";\n\n/**\n * Retrieves the validation metadata decorators associated with a specific property of a model,\n * using the reflective metadata key.\n *\n * @param model - The model instance or class containing the decorated property.\n * @param {string} prop - The name of the property whose decorators should be retrieved.\n * @param {string} reflectKey - The metadata key used to retrieve the decorators.\n *                     Defaults to `ValidationKeys.REFLECT`.\n *\n * @returns The validation decorators applied to the property\n */\nexport function getValidationDecorators(\n  model: Record<string, any>,\n  prop: string,\n  reflectKey: string = ValidationKeys.REFLECT\n): ValidationPropertyDecoratorDefinition {\n  return Reflection.getPropertyDecorators(\n    reflectKey,\n    model,\n    prop\n  ) as unknown as ValidationPropertyDecoratorDefinition;\n}\n\n/**\n * @description\n * Retrieves all validatable property decorators from a given model, excluding specified properties.\n *\n * @summary\n * Iterates through the own enumerable properties of a model instance, filtering out any properties\n * listed in the `propsToIgnore` array. For each remaining property, it checks whether validation\n * decorators are present using `getValidationDecorators`, and if so, collects them in the result array.\n *\n * @template M - A generic parameter extending the `Model` class, representing the model type being inspected.\n *\n * @param {M} model - An instance of a class extending `Model` from which validatable properties will be extracted.\n * @param {string[]} propsToIgnore - An array of property names that should be excluded from validation inspection.\n *\n * @return {ValidationPropertyDecoratorDefinition[]} An array of validation decorator definitions\n * associated with the model's properties, excluding those listed in `propsToIgnore`.\n *\n * @function getValidatableProperties\n */\nexport function getValidatableProperties<M extends Model>(\n  model: M,\n  propsToIgnore: string[]\n): ValidationPropertyDecoratorDefinition[] {\n  const decoratedProperties: ValidationPropertyDecoratorDefinition[] = [];\n\n  for (const prop in model) {\n    if (\n      Object.prototype.hasOwnProperty.call(model, prop) &&\n      !propsToIgnore.includes(prop)\n    ) {\n      const dec = getValidationDecorators(model, prop);\n      if (dec) decoratedProperties.push(dec);\n    }\n  }\n\n  return decoratedProperties;\n}\n\n/**\n * Safely sets temporary metadata on an object\n */\nfunction setTemporaryContext(\n  target: any,\n  key: symbol | string,\n  value: unknown\n): void {\n  if (!Object.hasOwnProperty.call(target, key)) target[key] = value;\n}\n\n/**\n * Safely removes temporary metadata from an object\n */\nfunction cleanupTemporaryContext(target: any, key: symbol | string): void {\n  if (Object.hasOwnProperty.call(target, key)) delete target[key];\n}\n\n/**\n * Executes validation with temporary context and returns the validation result\n *\n * @param nestedModel - The instance to validate\n * @param parentModel - Reference to a parent object for nested validation\n * @param isAsync - Whether to perform async validation\n * @returns Validation result from hasErrors()\n */\nfunction getNestedValidationErrors<\n  M extends Model,\n  Async extends boolean = false,\n>(\n  nestedModel: M,\n  parentModel?: M,\n  isAsync?: Async\n): ConditionalAsync<Async, ModelErrorDefinition | undefined> {\n  // Set temporary context for nested models\n  if (parentModel) {\n    setTemporaryContext(nestedModel, VALIDATION_PARENT_KEY, parentModel);\n  }\n  setTemporaryContext(nestedModel, ASYNC_META_KEY, !!isAsync);\n\n  const errs = nestedModel.hasErrors();\n  cleanupTemporaryContext(nestedModel, VALIDATION_PARENT_KEY);\n  cleanupTemporaryContext(nestedModel, ASYNC_META_KEY);\n  return errs as any;\n}\n\nexport function validateDecorator<\n  M extends Model,\n  Async extends boolean = false,\n>(\n  model: M,\n  value: any,\n  decorator: DecoratorMetadataAsync,\n  async?: Async\n): ConditionalAsync<Async, string | undefined> {\n  const validator = Validation.get(decorator.key);\n  if (!validator) {\n    throw new Error(`Missing validator for ${decorator.key}`);\n  }\n\n  // skip async decorators if validateDecorators is called synchronously (async = false)\n  if (!async && decorator.props.async) return undefined as any;\n\n  const decoratorProps =\n    decorator.key === ModelKeys.TYPE\n      ? [decorator.props]\n      : decorator.props || {};\n\n  const context = PathProxyEngine.create(model, {\n    ignoreUndefined: true,\n    ignoreNull: true,\n  });\n\n  const maybeAsyncErrors = validator.hasErrors(\n    value,\n    decoratorProps as ValidatorOptions,\n    context\n  );\n\n  return toConditionalPromise(maybeAsyncErrors, async);\n}\n\n/**\n * @description\n * Executes validation logic for a set of decorators applied to a model's property, handling both\n * synchronous and asynchronous validations, including support for nested validations and lists.\n *\n * @summary\n * Iterates over an array of decorator metadata objects and applies each validation rule to the\n * provided value. For list decorators (`ValidationKeys.LIST`), it performs element-wise validation,\n * supporting nested model validation and type checks. If the `async` flag is set, asynchronous\n * validation is supported using `Promise.all`. The result is a record mapping validation keys to\n * error messages, or `undefined` if no errors are found.\n *\n * @template M - A type parameter extending `Model`, representing the model type being validated.\n * @template Async - A boolean indicating whether validation should be performed asynchronously.\n *\n * @param {M} model - The model instance that the validation is associated with.\n * @param {any} value - The value to be validated against the provided decorators.\n * @param {DecoratorMetadataAsync[]} decorators - An array of metadata objects representing validation decorators.\n * @param {Async} [async] - Optional flag indicating whether validation should be performed asynchronously.\n *\n * @return {ConditionalAsync<Async, Record<string, string>> | undefined}\n * Returns either a record of validation errors (keyed by the decorator key) or `undefined` if no errors are found.\n * If `async` is true, the return value is a Promise resolving to the same structure.\n *\n * @function validateDecorators\n */\nexport function validateDecorators<\n  M extends Model,\n  Async extends boolean = false,\n>(\n  model: M,\n  value: any,\n  decorators: DecoratorMetadataAsync[],\n  async?: Async\n): ConditionalAsync<Async, Record<string, string> | undefined> {\n  const result: Record<string, string | Promise<string>> = {};\n\n  for (const decorator of decorators) {\n    // skip async decorators if validateDecorators is called synchronously (async = false)\n    if (!async && decorator.props.async) continue;\n\n    let validationErrors = validateDecorator(model, value, decorator, async);\n\n    /*\n    If the decorator is a list, each element must be checked.\n    When 'async' is true, the 'err' will always be a pending promise initially,\n    so the '!err' check will evaluate to false (even if the promise later resolves with no errors)\n    */\n    if (decorator.key === ValidationKeys.LIST && (!validationErrors || async)) {\n      const values = value instanceof Set ? [...value] : value;\n      if (values && values.length > 0) {\n        const types =\n          decorator.props.class ||\n          decorator.props.clazz ||\n          decorator.props.customTypes;\n\n        const allowedTypes = [types].flat().map((t) => String(t).toLowerCase());\n        // const reserved = Object.values(ReservedModels).map((v) => v.toLowerCase()) as string[];\n\n        const errs = values.map((childValue: any) => {\n          // if (Model.isModel(v) && !reserved.includes(v) {\n          if (Model.isModel(childValue)) {\n            return getNestedValidationErrors(childValue, model, async);\n          }\n\n          return allowedTypes.includes(typeof childValue)\n            ? undefined\n            : \"Value has no validatable type\";\n        });\n\n        if (async) {\n          validationErrors = Promise.all(errs).then((result) => {\n            const allEmpty = result.every((r) => !r);\n            return allEmpty ? undefined : result;\n          }) as any;\n        } else {\n          const allEmpty = errs.every((r: string | undefined) => !r);\n          validationErrors = errs.length > 0 && !allEmpty ? errs : undefined;\n        }\n      }\n    }\n\n    if (validationErrors) (result as any)[decorator.key] = validationErrors;\n  }\n\n  if (!async)\n    return Object.keys(result).length > 0\n      ? (result as any)\n      : (undefined as any);\n\n  const keys = Object.keys(result);\n  const promises = Object.values(result) as Promise<string | undefined>[];\n  return Promise.all(promises).then((resolvedValues) => {\n    const res: Record<string, string> = {};\n    for (let i = 0; i < resolvedValues.length; i++) {\n      const val = resolvedValues[i];\n      if (val !== undefined) {\n        res[keys[i]] = val;\n      }\n    }\n    return Object.keys(res).length > 0 ? res : undefined;\n  }) as any;\n}\n\n/**\n * @function validate\n * @template M\n * @template Async\n * @memberOf module:decorator-validation\n * @category Model\n *\n * @description\n * Validates the properties of a {@link Model} instance using registered decorators.\n * Supports both synchronous and asynchronous validation flows, depending on the `async` flag.\n *\n * @summary\n * This function inspects a given model object, identifies decorated properties that require validation,\n * and applies the corresponding validation rules. It also supports nested model validation and gracefully\n * merges any validation errors. For collections (Array/Set), it enforces the presence of the `@list` decorator\n * and checks the type of elements. If a property is a nested model, it will call `hasErrors` on it and flatten\n * the nested error keys using dot notation.\n *\n * @param {M} model - The model instance to be validated. Must extend from {@link Model}.\n * @param {Async} [async] - A flag indicating whether validation should be asynchronous.\n * @param {...string} propsToIgnore - A variadic list of property names that should be skipped during validation.\n *\n * @returns {ConditionalAsync<Async, ModelErrorDefinition | undefined>}\n * Returns either a {@link ModelErrorDefinition} containing validation errors,\n * or `undefined` if no errors are found. When `async` is `true`, returns a Promise.\n *\n * @see {@link Model}\n * @see {@link ModelErrorDefinition}\n * @see {@link validateDecorators}\n * @see {@link getValidatableProperties}\n *\n * @mermaid\n * sequenceDiagram\n *     participant Caller\n *     participant validate\n *     participant getValidatableProperties\n *     participant validateDecorators\n *     participant ModelInstance\n *     Caller->>validate: call with obj, async, propsToIgnore\n *     validate->>getValidatableProperties: retrieve decorated props\n *     loop for each property\n *         validate->>validateDecorators: validate using decorators\n *         alt is nested model\n *             validate->>ModelInstance: call hasErrors()\n *         end\n *     end\n *     alt async\n *         validate->>validate: Promise.allSettled for errors\n *     end\n *     validate-->>Caller: return ModelErrorDefinition | undefined\n */\nexport function validate<\n  M extends Model<boolean>,\n  Async extends boolean = false,\n>(\n  model: M,\n  async: Async,\n  ...propsToIgnore: string[]\n): ConditionalAsync<Async, ModelErrorDefinition | undefined> {\n  const decoratedProperties: ValidationPropertyDecoratorDefinition[] =\n    getValidatableProperties(model, propsToIgnore);\n\n  const result: Record<string, any> = {};\n  const nestedErrors: Record<string, any> = {};\n\n  for (const { prop, decorators } of decoratedProperties) {\n    const propKey = String(prop);\n    let propValue = (model as any)[prop];\n\n    if (!decorators?.length) continue;\n\n    // Get the default type validator\n    const designTypeDec = decorators.find((d) => {\n      return [ModelKeys.TYPE, ValidationKeys.TYPE].includes(d.key as any);\n    });\n\n    if (!designTypeDec) continue;\n\n    const designType = designTypeDec.props.name;\n\n    // Handle array or Set types and enforce the presence of @list decorator\n    if ([Array.name, Set.name].includes(designType)) {\n      if (!decorators.some((d) => d.key === ValidationKeys.LIST)) {\n        result[propKey] = {\n          [ValidationKeys.TYPE]: `Array or Set property '${propKey}' requires a @list decorator`,\n        };\n        continue;\n      }\n\n      if (\n        propValue &&\n        !(Array.isArray(propValue) || propValue instanceof Set)\n      ) {\n        result[propKey] = {\n          [ValidationKeys.TYPE]: `Property '${String(prop)}' must be either an Array or a Set`,\n        };\n        continue;\n      }\n\n      // Remove design:type decorator, since @list decorator already ensures type\n      for (let i = decorators.length - 1; i >= 0; i--) {\n        if (decorators[i].key === ModelKeys.TYPE) {\n          decorators.splice(i, 1);\n        }\n      }\n      propValue = propValue instanceof Set ? [...propValue] : propValue;\n    }\n\n    const propErrors: Record<string, any> =\n      validateDecorators(model, propValue, decorators, async) || {};\n\n    // Check for nested properties.\n    // To prevent unnecessary processing, \"propValue\" must be defined and validatable\n    // let nestedErrors: Record<string, any> = {};\n    const isConstr = Model.isPropertyModel(model, propKey);\n    // if propValue !== undefined, null\n    if (propValue && isConstr) {\n      const instance: Model = propValue;\n      const isInvalidModel =\n        typeof instance !== \"object\" ||\n        !instance.hasErrors ||\n        typeof instance.hasErrors !== \"function\";\n\n      if (isInvalidModel) {\n        // propErrors[ValidationKeys.TYPE] = \"Model should be validatable but it's not.\";\n        console.warn(\"Model should be validatable but it's not.\");\n      } else {\n        nestedErrors[propKey] = getNestedValidationErrors(\n          instance,\n          model,\n          async\n        );\n      }\n    }\n\n    // Add to the result if we have any errors\n    // Async mode returns a Promise that resolves to undefined when no errors exist\n    if (Object.keys(propErrors).length > 0 || async)\n      result[propKey] = propErrors;\n\n    // Then merge any nested errors\n    if (!async) {\n      Object.entries(nestedErrors[propKey] || {}).forEach(([key, error]) => {\n        if (error !== undefined) {\n          result[`${propKey}.${key}`] = error;\n        }\n      });\n    }\n  }\n\n  // Synchronous return\n  if (!async) {\n    return (\n      Object.keys(result).length > 0\n        ? new ModelErrorDefinition(result)\n        : undefined\n    ) as any;\n  }\n\n  const merged: any = result; // TODO: apply filtering\n\n  const keys = Object.keys(merged);\n  const promises = Object.values(merged);\n  return Promise.allSettled(promises).then(async (results) => {\n    const result: ModelErrors = {};\n\n    for (const [parentProp, nestedErrPromise] of Object.entries(nestedErrors)) {\n      const nestedPropDecErrors = (await nestedErrPromise) as Record<\n        string,\n        any\n      >;\n\n      if (nestedPropDecErrors)\n        Object.entries(nestedPropDecErrors).forEach(\n          ([nestedProp, nestedPropDecError]) => {\n            if (nestedPropDecError !== undefined) {\n              const nestedKey = [parentProp, nestedProp].join(\".\");\n              result[nestedKey] = nestedPropDecError;\n            }\n          }\n        );\n    }\n\n    for (let i = 0; i < results.length; i++) {\n      const key = keys[i];\n      const res = results[i];\n\n      if (res.status === \"fulfilled\" && res.value !== undefined) {\n        (result as any)[key] = res.value;\n      } else if (res.status === \"rejected\") {\n        (result as any)[key] =\n          res.reason instanceof Error\n            ? res.reason.message\n            : String(res.reason || \"Validation failed\");\n      }\n    }\n\n    return Object.keys(result).length > 0\n      ? new ModelErrorDefinition(result)\n      : undefined;\n  }) as any;\n}\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export * from "./validation";
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export * from "./validation.js";
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvdHlwZXMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsZ0NBQTZCIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSBcIi4vdmFsaWRhdGlvblwiO1xuIl19
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
import { DecoratorMetadata } from "@decaf-ts/reflection";
|
|
2
|
+
import { Model, ModelErrorDefinition } from "../model";
|
|
3
|
+
/**
|
|
4
|
+
* @description Conditionally wraps a type in a `Promise` based on the `Async` flag.
|
|
5
|
+
* @summary Utility type that resolves to `T` if `Async` is `false`, or to `Promise<T>` if `Async` is `true`.
|
|
6
|
+
* Used to abstract the return type of functions that may be either synchronous or asynchronous depending on a flag.
|
|
7
|
+
*
|
|
8
|
+
* @template Async A boolean flag indicating whether the result should be asynchronous (`true`) or synchronous (`false`).
|
|
9
|
+
* @template T The base type to return directly or to wrap in a `Promise`.
|
|
10
|
+
*
|
|
11
|
+
* @example
|
|
12
|
+
* ```typescript
|
|
13
|
+
* // Synchronous result
|
|
14
|
+
* type SyncResult: string | number = ConditionalAsync<false, string | number>;
|
|
15
|
+
*
|
|
16
|
+
* // Asynchronous result
|
|
17
|
+
* type AsyncResult: Promise<string | number> = ConditionalAsync<true, string | number>;
|
|
18
|
+
* ```
|
|
19
|
+
* @memberOf module:decorator-validation
|
|
20
|
+
*/
|
|
21
|
+
export type ConditionalAsync<Async extends boolean, T> = Async extends true ? Promise<T> : T;
|
|
22
|
+
export type DecoratorMetadataAsync = DecoratorMetadata & {
|
|
23
|
+
async?: boolean;
|
|
24
|
+
};
|
|
25
|
+
export type ModelConditionalAsync<M> = M extends Model<true> ? Promise<ModelErrorDefinition | undefined> : ModelErrorDefinition | undefined;
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export {};
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmFsaWRhdGlvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy90eXBlcy92YWxpZGF0aW9uLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBEZWNvcmF0b3JNZXRhZGF0YSB9IGZyb20gXCJAZGVjYWYtdHMvcmVmbGVjdGlvblwiO1xuaW1wb3J0IHsgTW9kZWwsIE1vZGVsRXJyb3JEZWZpbml0aW9uIH0gZnJvbSBcIi4uL21vZGVsXCI7XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uIENvbmRpdGlvbmFsbHkgd3JhcHMgYSB0eXBlIGluIGEgYFByb21pc2VgIGJhc2VkIG9uIHRoZSBgQXN5bmNgIGZsYWcuXG4gKiBAc3VtbWFyeSBVdGlsaXR5IHR5cGUgdGhhdCByZXNvbHZlcyB0byBgVGAgaWYgYEFzeW5jYCBpcyBgZmFsc2VgLCBvciB0byBgUHJvbWlzZTxUPmAgaWYgYEFzeW5jYCBpcyBgdHJ1ZWAuXG4gKiBVc2VkIHRvIGFic3RyYWN0IHRoZSByZXR1cm4gdHlwZSBvZiBmdW5jdGlvbnMgdGhhdCBtYXkgYmUgZWl0aGVyIHN5bmNocm9ub3VzIG9yIGFzeW5jaHJvbm91cyBkZXBlbmRpbmcgb24gYSBmbGFnLlxuICpcbiAqIEB0ZW1wbGF0ZSBBc3luYyBBIGJvb2xlYW4gZmxhZyBpbmRpY2F0aW5nIHdoZXRoZXIgdGhlIHJlc3VsdCBzaG91bGQgYmUgYXN5bmNocm9ub3VzIChgdHJ1ZWApIG9yIHN5bmNocm9ub3VzIChgZmFsc2VgKS5cbiAqIEB0ZW1wbGF0ZSBUIFRoZSBiYXNlIHR5cGUgdG8gcmV0dXJuIGRpcmVjdGx5IG9yIHRvIHdyYXAgaW4gYSBgUHJvbWlzZWAuXG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIC8vIFN5bmNocm9ub3VzIHJlc3VsdFxuICogdHlwZSBTeW5jUmVzdWx0OiBzdHJpbmcgfCBudW1iZXIgPSBDb25kaXRpb25hbEFzeW5jPGZhbHNlLCBzdHJpbmcgfCBudW1iZXI+O1xuICpcbiAqIC8vIEFzeW5jaHJvbm91cyByZXN1bHRcbiAqIHR5cGUgQXN5bmNSZXN1bHQ6IFByb21pc2U8c3RyaW5nIHwgbnVtYmVyPiA9IENvbmRpdGlvbmFsQXN5bmM8dHJ1ZSwgc3RyaW5nIHwgbnVtYmVyPjtcbiAqIGBgYFxuICogQG1lbWJlck9mIG1vZHVsZTpkZWNvcmF0b3ItdmFsaWRhdGlvblxuICovXG5leHBvcnQgdHlwZSBDb25kaXRpb25hbEFzeW5jPEFzeW5jIGV4dGVuZHMgYm9vbGVhbiwgVD4gPSBBc3luYyBleHRlbmRzIHRydWVcbiAgPyBQcm9taXNlPFQ+XG4gIDogVDtcblxuZXhwb3J0IHR5cGUgRGVjb3JhdG9yTWV0YWRhdGFBc3luYyA9IERlY29yYXRvck1ldGFkYXRhICYgeyBhc3luYz86IGJvb2xlYW4gfTtcblxuZXhwb3J0IHR5cGUgTW9kZWxDb25kaXRpb25hbEFzeW5jPE0+ID1cbiAgTSBleHRlbmRzIE1vZGVsPHRydWU+XG4gICAgPyBQcm9taXNlPE1vZGVsRXJyb3JEZWZpbml0aW9uIHwgdW5kZWZpbmVkPlxuICAgIDogTW9kZWxFcnJvckRlZmluaXRpb24gfCB1bmRlZmluZWQ7XG4iXX0=
|