@decaf-ts/decorator-validation 1.7.11 → 1.7.13

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 (41) hide show
  1. package/dist/decorator-validation.cjs +274 -104
  2. package/dist/decorator-validation.esm.cjs +274 -105
  3. package/lib/esm/index.d.ts +1 -1
  4. package/lib/esm/index.js +1 -1
  5. package/lib/esm/model/Model.js +28 -6
  6. package/lib/esm/model/decorators.d.ts +2 -12
  7. package/lib/esm/model/decorators.js +57 -37
  8. package/lib/esm/model/validation.d.ts +3 -2
  9. package/lib/esm/model/validation.js +58 -21
  10. package/lib/esm/utils/Decoration.d.ts +15 -3
  11. package/lib/esm/utils/Decoration.js +49 -12
  12. package/lib/esm/utils/decorators.d.ts +1 -1
  13. package/lib/esm/utils/decorators.js +13 -4
  14. package/lib/esm/utils/types.d.ts +3 -2
  15. package/lib/esm/utils/types.js +1 -1
  16. package/lib/esm/validation/Validators/ListValidator.js +8 -4
  17. package/lib/esm/validation/Validators/TypeValidator.js +12 -8
  18. package/lib/esm/validation/decorators.d.ts +3 -3
  19. package/lib/esm/validation/decorators.js +56 -18
  20. package/lib/esm/validation/types.d.ts +2 -1
  21. package/lib/esm/validation/types.js +1 -1
  22. package/lib/index.cjs +1 -1
  23. package/lib/index.d.ts +1 -1
  24. package/lib/model/Model.cjs +28 -6
  25. package/lib/model/decorators.cjs +58 -37
  26. package/lib/model/decorators.d.ts +2 -12
  27. package/lib/model/validation.cjs +58 -21
  28. package/lib/model/validation.d.ts +3 -2
  29. package/lib/utils/Decoration.cjs +49 -12
  30. package/lib/utils/Decoration.d.ts +15 -3
  31. package/lib/utils/decorators.cjs +13 -4
  32. package/lib/utils/decorators.d.ts +1 -1
  33. package/lib/utils/types.cjs +1 -1
  34. package/lib/utils/types.d.ts +3 -2
  35. package/lib/validation/Validators/ListValidator.cjs +8 -4
  36. package/lib/validation/Validators/TypeValidator.cjs +12 -8
  37. package/lib/validation/decorators.cjs +56 -18
  38. package/lib/validation/decorators.d.ts +3 -3
  39. package/lib/validation/types.cjs +1 -1
  40. package/lib/validation/types.d.ts +2 -1
  41. package/package.json +1 -1
@@ -82,30 +82,32 @@ function cleanupTemporaryContext(target, key) {
82
82
  * @param isAsync - Whether to perform async validation
83
83
  * @returns Validation result from hasErrors()
84
84
  */
85
- function getNestedValidationErrors(nestedModel, parentModel, isAsync) {
85
+ function getNestedValidationErrors(nestedModel, parentModel, isAsync, ...propsToIgnore) {
86
86
  // Set temporary context for nested models
87
87
  if (parentModel) {
88
88
  setTemporaryContext(nestedModel, constants_3.VALIDATION_PARENT_KEY, parentModel);
89
89
  }
90
90
  setTemporaryContext(nestedModel, constants_3.ASYNC_META_KEY, !!isAsync);
91
- const errs = nestedModel.hasErrors();
91
+ const errs = nestedModel.hasErrors(...propsToIgnore);
92
92
  cleanupTemporaryContext(nestedModel, constants_3.VALIDATION_PARENT_KEY);
93
93
  cleanupTemporaryContext(nestedModel, constants_3.ASYNC_META_KEY);
94
94
  return errs;
95
95
  }
96
- function validateChildValue(childValue, parentModel, allowedTypes, async) {
96
+ function validateChildValue(prop, childValue, parentModel, allowedTypes, async, ...propsToIgnore) {
97
97
  let err = undefined;
98
98
  let atLeastOneMatched = false;
99
99
  for (const allowedType of allowedTypes) {
100
100
  const Constr = Model_1.Model.get(allowedType);
101
101
  if (!Constr) {
102
102
  err = new ModelErrorDefinition_1.ModelErrorDefinition({
103
- [constants_2.ValidationKeys.TYPE]: `Unable to verify type consistency, missing model registry for ${allowedType}`,
103
+ [prop]: {
104
+ [constants_2.ValidationKeys.TYPE]: `Unable to verify type consistency, missing model registry for ${allowedType}`,
105
+ },
104
106
  });
105
107
  }
106
108
  if (childValue instanceof Constr) {
107
109
  atLeastOneMatched = true;
108
- err = getNestedValidationErrors(childValue, parentModel, async);
110
+ err = getNestedValidationErrors(childValue, parentModel, async, ...propsToIgnore);
109
111
  break;
110
112
  }
111
113
  }
@@ -113,7 +115,9 @@ function validateChildValue(childValue, parentModel, allowedTypes, async) {
113
115
  return err;
114
116
  return (err ||
115
117
  new ModelErrorDefinition_1.ModelErrorDefinition({
116
- [constants_2.ValidationKeys.TYPE]: `Value must be an instance of one of the expected types: ${allowedTypes.join(", ")}`,
118
+ [prop]: {
119
+ [constants_2.ValidationKeys.TYPE]: `Value must be an instance of one of the expected types: ${allowedTypes.join(", ")}`,
120
+ },
117
121
  }));
118
122
  }
119
123
  function validateDecorator(model, value, decorator, async) {
@@ -131,7 +135,9 @@ function validateDecorator(model, value, decorator, async) {
131
135
  ignoreUndefined: true,
132
136
  ignoreNull: true,
133
137
  });
134
- const maybeAsyncErrors = validator.hasErrors(value, decoratorProps, context);
138
+ const maybeAsyncErrors = validator.hasErrors(value, decorator.key === constants_1.ModelKeys.TYPE
139
+ ? { types: decoratorProps[0].name }
140
+ : decoratorProps, context);
135
141
  return (0, utils_1.toConditionalPromise)(maybeAsyncErrors, async);
136
142
  }
137
143
  /**
@@ -150,6 +156,7 @@ function validateDecorator(model, value, decorator, async) {
150
156
  * @template Async - A boolean indicating whether validation should be performed asynchronously.
151
157
  *
152
158
  * @param {M} model - The model instance that the validation is associated with.
159
+ * @param {string} prop - The model field name
153
160
  * @param {any} value - The value to be validated against the provided decorators.
154
161
  * @param {DecoratorMetadataAsync[]} decorators - An array of metadata objects representing validation decorators.
155
162
  * @param {Async} [async] - Optional flag indicating whether validation should be performed asynchronously.
@@ -160,7 +167,7 @@ function validateDecorator(model, value, decorator, async) {
160
167
  *
161
168
  * @function validateDecorators
162
169
  */
163
- function validateDecorators(model, value, decorators, async) {
170
+ function validateDecorators(model, prop, value, decorators, async, ...propsToIgnore) {
164
171
  const result = {};
165
172
  for (const decorator of decorators) {
166
173
  // skip async decorators if validateDecorators is called synchronously (async = false)
@@ -175,15 +182,19 @@ function validateDecorators(model, value, decorators, async) {
175
182
  if (decorator.key === constants_2.ValidationKeys.LIST && (!validationErrors || async)) {
176
183
  const values = value instanceof Set ? [...value] : value;
177
184
  if (values && values.length > 0) {
178
- const types = (decorator.props.class ||
185
+ let types = (decorator.props.class ||
179
186
  decorator.props.clazz ||
180
187
  decorator.props.customTypes);
188
+ types = (Array.isArray(types) ? types : [types]).map((e) => {
189
+ e = typeof e === "function" && !e.name ? e() : e;
190
+ return e.name ? e.name : e;
191
+ });
181
192
  const allowedTypes = [types].flat().map((t) => String(t).toLowerCase());
182
193
  // const reserved = Object.values(ReservedModels).map((v) => v.toLowerCase()) as string[];
183
194
  const errs = values.map((childValue) => {
184
195
  // if (Model.isModel(v) && !reserved.includes(v) {
185
196
  if (Model_1.Model.isModel(childValue)) {
186
- return validateChildValue(childValue, model, [types].flat(), !!async);
197
+ return validateChildValue(prop, childValue, model, types.flat(), !!async, ...propsToIgnore);
187
198
  // return getNestedValidationErrors(childValue, model, async);
188
199
  }
189
200
  return allowedTypes.includes(typeof childValue)
@@ -202,8 +213,9 @@ function validateDecorators(model, value, decorators, async) {
202
213
  }
203
214
  }
204
215
  }
216
+ const name = decorator.key === constants_1.ModelKeys.TYPE ? constants_2.ValidationKeys.TYPE : decorator.key;
205
217
  if (validationErrors)
206
- result[decorator.key] = validationErrors;
218
+ result[name] = validationErrors;
207
219
  }
208
220
  if (!async)
209
221
  return Object.keys(result).length > 0
@@ -283,14 +295,32 @@ function validate(model, async, ...propsToIgnore) {
283
295
  if (!decorators?.length)
284
296
  continue;
285
297
  // Get the default type validator
286
- const designTypeDec = decorators.find((d) => {
287
- return [constants_1.ModelKeys.TYPE, constants_2.ValidationKeys.TYPE].includes(d.key);
288
- });
298
+ const priority = [constants_2.ValidationKeys.TYPE, constants_1.ModelKeys.TYPE];
299
+ const designTypeDec = priority
300
+ .map((key) => decorators.find((d) => d.key === key))
301
+ .find(Boolean);
302
+ // Ensures that only one type decorator remains.
303
+ if (designTypeDec?.key === constants_2.ValidationKeys.TYPE) {
304
+ decorators.splice(0, decorators.length, ...decorators.filter((d) => d.key !== constants_1.ModelKeys.TYPE));
305
+ }
289
306
  if (!designTypeDec)
290
307
  continue;
291
- const designType = designTypeDec.props.name;
308
+ const designType = designTypeDec.props.class ||
309
+ designTypeDec.props.clazz ||
310
+ designTypeDec.props.customTypes ||
311
+ designTypeDec.props.name;
312
+ // TS emits "Object" as design:type for unions (string | number) and intersections (A & B).
313
+ // Since this metadata is ambiguous for validation, skip design:type checks in these cases.
314
+ // To enforce design:type validation explicitly, the @type validator can be used.
315
+ if (designTypeDec.key === constants_1.ModelKeys.TYPE && designType === "Object")
316
+ decorators.shift();
317
+ const designTypes = (Array.isArray(designType) ? designType : [designType]).map((e) => {
318
+ e = typeof e === "function" && !e.name ? e() : e;
319
+ return e.name ? e.name : e;
320
+ });
292
321
  // Handle array or Set types and enforce the presence of @list decorator
293
- if ([Array.name, Set.name].includes(designType)) {
322
+ // if ([Array.name, Set.name].includes(designType)) {}
323
+ if (designTypes.some((t) => [Array.name, Set.name].includes(t))) {
294
324
  if (!decorators.some((d) => d.key === constants_2.ValidationKeys.LIST)) {
295
325
  result[propKey] = {
296
326
  [constants_2.ValidationKeys.TYPE]: `Array or Set property '${propKey}' requires a @list decorator`,
@@ -312,7 +342,7 @@ function validate(model, async, ...propsToIgnore) {
312
342
  }
313
343
  propValue = propValue instanceof Set ? [...propValue] : propValue;
314
344
  }
315
- const propErrors = validateDecorators(model, propValue, decorators, async) || {};
345
+ const propErrors = validateDecorators(model, propKey, propValue, decorators, async, ...propsToIgnore) || {};
316
346
  // Check for nested properties.
317
347
  // To prevent unnecessary processing, "propValue" must be defined and validatable
318
348
  // let nestedErrors: Record<string, any> = {};
@@ -327,15 +357,22 @@ function validate(model, async, ...propsToIgnore) {
327
357
  console.warn("Model should be validatable but it's not.");
328
358
  }
329
359
  else {
330
- const Constr = Model_1.Model.get(designType);
360
+ const Constr = (Array.isArray(designType) ? designType : [designType])
361
+ .map((d) => {
362
+ if (typeof d === "function" && !d.name)
363
+ d = d();
364
+ return Model_1.Model.get(d.name || d);
365
+ })
366
+ .find((d) => !!d);
331
367
  // Ensure instance is of the expected model class.
332
368
  if (!Constr || !(instance instanceof Constr)) {
333
369
  propErrors[constants_2.ValidationKeys.TYPE] = !Constr
334
- ? `Unable to verify type consistency, missing model registry for ${designType} on prop ${propKey}`
370
+ ? `Unable to verify type consistency, missing model registry for ${designTypes.toString()} on prop ${propKey}`
335
371
  : `Value must be an instance of ${Constr.name}`;
372
+ delete propErrors[constants_1.ModelKeys.TYPE]; // remove duplicate type error
336
373
  }
337
374
  else {
338
- nestedErrors[propKey] = getNestedValidationErrors(instance, model, async);
375
+ nestedErrors[propKey] = getNestedValidationErrors(instance, model, async, ...propsToIgnore);
339
376
  }
340
377
  }
341
378
  }
@@ -391,4 +428,4 @@ function validate(model, async, ...propsToIgnore) {
391
428
  : undefined;
392
429
  });
393
430
  }
394
- //# 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 validateChildValue<M extends Model>(\n  childValue: any,\n  parentModel: M,\n  allowedTypes: string[],\n  async: boolean\n): string | undefined | ModelErrorDefinition {\n  let err: ModelErrorDefinition | string | undefined = undefined;\n  let atLeastOneMatched = false;\n  for (const allowedType of allowedTypes) {\n    const Constr = Model.get(allowedType) as any;\n    if (!Constr) {\n      err = new ModelErrorDefinition({\n        [ValidationKeys.TYPE]: `Unable to verify type consistency, missing model registry for ${allowedType}`,\n      });\n    }\n\n    if (childValue instanceof Constr) {\n      atLeastOneMatched = true;\n      err = getNestedValidationErrors(childValue, parentModel, async);\n      break;\n    }\n  }\n\n  if (atLeastOneMatched) return err;\n\n  return (\n    err ||\n    new ModelErrorDefinition({\n      [ValidationKeys.TYPE]: `Value must be an instance of one of the expected types: ${allowedTypes.join(\", \")}`,\n    })\n  );\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 = (decorator.props.class ||\n          decorator.props.clazz ||\n          decorator.props.customTypes) as string | string[];\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 validateChildValue(\n              childValue,\n              model,\n              [types].flat(),\n              !!async\n            );\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    const hasPropValue = propValue !== null && propValue !== undefined;\n    if (isConstr && hasPropValue) {\n      const instance = propValue as Model;\n      const isInvalidModel =\n        typeof instance !== \"object\" ||\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        const Constr = Model.get(designType) as any;\n\n        // Ensure instance is of the expected model class.\n        if (!Constr || !(instance instanceof Constr)) {\n          propErrors[ValidationKeys.TYPE] = !Constr\n            ? `Unable to verify type consistency, missing model registry for ${designType} on prop ${propKey}`\n            : `Value must be an instance of ${Constr.name}`;\n        } else {\n          nestedErrors[propKey] = getNestedValidationErrors(\n            instance,\n            model,\n            async\n          );\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"]}
431
+ //# 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  ...propsToIgnore: string[]\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(...propsToIgnore) as any;\n  cleanupTemporaryContext(nestedModel, VALIDATION_PARENT_KEY);\n  cleanupTemporaryContext(nestedModel, ASYNC_META_KEY);\n  return errs as any;\n}\n\nexport function validateChildValue<M extends Model>(\n  prop: string,\n  childValue: any,\n  parentModel: M,\n  allowedTypes: string[],\n  async: boolean,\n  ...propsToIgnore: string[]\n):\n  | string\n  | undefined\n  | ModelErrorDefinition\n  | Promise<string | undefined | ModelErrorDefinition> {\n  let err:\n    | ModelErrorDefinition\n    | string\n    | undefined\n    | Promise<string | undefined | ModelErrorDefinition> = undefined;\n  let atLeastOneMatched = false;\n  for (const allowedType of allowedTypes) {\n    const Constr = Model.get(allowedType) as any;\n    if (!Constr) {\n      err = new ModelErrorDefinition({\n        [prop]: {\n          [ValidationKeys.TYPE]: `Unable to verify type consistency, missing model registry for ${allowedType}`,\n        },\n      });\n    }\n\n    if (childValue instanceof Constr) {\n      atLeastOneMatched = true;\n      err = getNestedValidationErrors(\n        childValue,\n        parentModel,\n        async,\n        ...propsToIgnore\n      );\n      break;\n    }\n  }\n\n  if (atLeastOneMatched) return err;\n\n  return (\n    err ||\n    new ModelErrorDefinition({\n      [prop]: {\n        [ValidationKeys.TYPE]: `Value must be an instance of one of the expected types: ${allowedTypes.join(\", \")}`,\n      },\n    })\n  );\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    decorator.key === ModelKeys.TYPE\n      ? ({ types: (decoratorProps as any)[0].name } as any)\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 {string} prop - The model field name\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  prop: string,\n  value: any,\n  decorators: DecoratorMetadataAsync[],\n  async?: Async,\n  ...propsToIgnore: string[]\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        let types: string[] = (decorator.props.class ||\n          decorator.props.clazz ||\n          decorator.props.customTypes) as string[];\n        types = (Array.isArray(types) ? types : [types]).map((e: any) => {\n          e = typeof e === \"function\" && !e.name ? e() : e;\n          return (e as any).name ? (e as any).name : e;\n        }) as string[];\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 validateChildValue(\n              prop,\n              childValue,\n              model,\n              types.flat(),\n              !!async,\n              ...propsToIgnore\n            );\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    const name =\n      decorator.key === ModelKeys.TYPE ? ValidationKeys.TYPE : decorator.key;\n    if (validationErrors) (result as any)[name] = 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 priority = [ValidationKeys.TYPE, ModelKeys.TYPE];\n    const designTypeDec = priority\n      .map((key) => decorators.find((d) => d.key === key))\n      .find(Boolean);\n\n    // Ensures that only one type decorator remains.\n    if (designTypeDec?.key === ValidationKeys.TYPE) {\n      decorators.splice(\n        0,\n        decorators.length,\n        ...decorators.filter((d) => d.key !== ModelKeys.TYPE)\n      );\n    }\n\n    if (!designTypeDec) continue;\n\n    const designType =\n      designTypeDec.props.class ||\n      designTypeDec.props.clazz ||\n      designTypeDec.props.customTypes ||\n      designTypeDec.props.name;\n\n    // TS emits \"Object\" as design:type for unions (string | number) and intersections (A & B).\n    // Since this metadata is ambiguous for validation, skip design:type checks in these cases.\n    // To enforce design:type validation explicitly, the @type validator can be used.\n    if (designTypeDec.key === ModelKeys.TYPE && designType === \"Object\")\n      decorators.shift();\n\n    const designTypes = (\n      Array.isArray(designType) ? designType : [designType]\n    ).map((e: any) => {\n      e = typeof e === \"function\" && !e.name ? e() : e;\n      return (e as any).name ? (e as any).name : e;\n    }) as string[];\n\n    // Handle array or Set types and enforce the presence of @list decorator\n    // if ([Array.name, Set.name].includes(designType)) {}\n    if (designTypes.some((t) => [Array.name, Set.name].includes(t))) {\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(\n        model,\n        propKey,\n        propValue,\n        decorators,\n        async,\n        ...propsToIgnore\n      ) || {};\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    const hasPropValue = propValue !== null && propValue !== undefined;\n    if (isConstr && hasPropValue) {\n      const instance = propValue as Model;\n      const isInvalidModel =\n        typeof instance !== \"object\" ||\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        const Constr = (Array.isArray(designType) ? designType : [designType])\n          .map((d) => {\n            if (typeof d === \"function\" && !d.name) d = d();\n            return Model.get(d.name || d);\n          })\n          .find((d) => !!d) as any;\n\n        // Ensure instance is of the expected model class.\n        if (!Constr || !(instance instanceof Constr)) {\n          propErrors[ValidationKeys.TYPE] = !Constr\n            ? `Unable to verify type consistency, missing model registry for ${designTypes.toString()} on prop ${propKey}`\n            : `Value must be an instance of ${Constr.name}`;\n          delete propErrors[ModelKeys.TYPE]; // remove duplicate type error\n        } else {\n          nestedErrors[propKey] = getNestedValidationErrors(\n            instance,\n            model,\n            async,\n            ...propsToIgnore\n          );\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"]}
@@ -34,7 +34,7 @@ export declare function getValidationDecorators(model: Record<string, any>, prop
34
34
  * @function getValidatableProperties
35
35
  */
36
36
  export declare function getValidatableProperties<M extends Model>(model: M, propsToIgnore: string[]): ValidationPropertyDecoratorDefinition[];
37
- export declare function validateChildValue<M extends Model>(childValue: any, parentModel: M, allowedTypes: string[], async: boolean): string | undefined | ModelErrorDefinition;
37
+ export declare function validateChildValue<M extends Model>(prop: string, childValue: any, parentModel: M, allowedTypes: string[], async: boolean, ...propsToIgnore: string[]): string | undefined | ModelErrorDefinition | Promise<string | undefined | ModelErrorDefinition>;
38
38
  export declare function validateDecorator<M extends Model, Async extends boolean = false>(model: M, value: any, decorator: DecoratorMetadataAsync, async?: Async): ConditionalAsync<Async, string | undefined>;
39
39
  /**
40
40
  * @description
@@ -52,6 +52,7 @@ export declare function validateDecorator<M extends Model, Async extends boolean
52
52
  * @template Async - A boolean indicating whether validation should be performed asynchronously.
53
53
  *
54
54
  * @param {M} model - The model instance that the validation is associated with.
55
+ * @param {string} prop - The model field name
55
56
  * @param {any} value - The value to be validated against the provided decorators.
56
57
  * @param {DecoratorMetadataAsync[]} decorators - An array of metadata objects representing validation decorators.
57
58
  * @param {Async} [async] - Optional flag indicating whether validation should be performed asynchronously.
@@ -62,7 +63,7 @@ export declare function validateDecorator<M extends Model, Async extends boolean
62
63
  *
63
64
  * @function validateDecorators
64
65
  */
65
- export declare function validateDecorators<M extends Model, Async extends boolean = false>(model: M, value: any, decorators: DecoratorMetadataAsync[], async?: Async): ConditionalAsync<Async, Record<string, string> | undefined>;
66
+ export declare function validateDecorators<M extends Model, Async extends boolean = false>(model: M, prop: string, value: any, decorators: DecoratorMetadataAsync[], async?: Async, ...propsToIgnore: string[]): ConditionalAsync<Async, Record<string, string> | undefined>;
66
67
  /**
67
68
  * @function validate
68
69
  * @template M
@@ -116,23 +116,59 @@ class Decoration {
116
116
  decoratorFactory(key, f = constants_1.DefaultFlavour) {
117
117
  const contextDecorator = function contextDecorator(target, propertyKey, descriptor) {
118
118
  const flavour = Decoration.flavourResolver(target);
119
+ const cache = Decoration.decorators[key];
119
120
  let decorators;
120
- const extras = Decoration.decorators[key][flavour]
121
- ? Decoration.decorators[key][flavour].extras
122
- : Decoration.decorators[key][constants_1.DefaultFlavour].extras;
123
- if (Decoration.decorators[key] &&
124
- Decoration.decorators[key][flavour] &&
125
- Decoration.decorators[key][flavour].decorators) {
126
- decorators = Decoration.decorators[key][flavour].decorators;
121
+ const extras = cache[flavour]
122
+ ? cache[flavour].extras
123
+ : cache[constants_1.DefaultFlavour].extras;
124
+ const extraArgs = [
125
+ ...(cache[constants_1.DefaultFlavour].extras
126
+ ? cache[constants_1.DefaultFlavour].extras.values()
127
+ : []),
128
+ ].reduce((accum, e, i) => {
129
+ if (e.args)
130
+ accum[i] = e.args;
131
+ return accum;
132
+ }, {});
133
+ if (cache &&
134
+ cache[flavour] &&
135
+ cache[flavour].decorators &&
136
+ cache[flavour].decorators.size) {
137
+ decorators = cache[flavour].decorators;
127
138
  }
128
139
  else {
129
- decorators = Decoration.decorators[key][constants_1.DefaultFlavour].decorators;
140
+ decorators = cache[constants_1.DefaultFlavour].decorators;
130
141
  }
142
+ const decoratorArgs = [
143
+ ...cache[constants_1.DefaultFlavour].decorators.values(),
144
+ ].reduce((accum, e, i) => {
145
+ if (e.args)
146
+ accum[i] = e.args;
147
+ return accum;
148
+ }, {});
131
149
  const toApply = [
132
150
  ...(decorators ? decorators.values() : []),
133
151
  ...(extras ? extras.values() : []),
134
152
  ];
135
- toApply.forEach((d) => d(target, propertyKey, descriptor, descriptor));
153
+ return toApply.reduce((_, d, i) => {
154
+ switch (typeof d) {
155
+ case "object": {
156
+ const { decorator, args, transform } = d;
157
+ const argz = args || i < (decorators ? decorators.size : 0)
158
+ ? decoratorArgs[i]
159
+ : extraArgs[i - (decorators ? decorators.size : 0)] ||
160
+ (decorators ? decoratorArgs[i - decorators.size] : []);
161
+ const transformed = transform
162
+ ? transform(argz || [])
163
+ : argz || [];
164
+ return decorator(...transformed)(target, propertyKey, descriptor);
165
+ }
166
+ case "function":
167
+ return d(target, propertyKey, descriptor);
168
+ default:
169
+ throw new Error(`Unexpected decorator type: ${typeof d}`);
170
+ }
171
+ }, { target, propertyKey, descriptor });
136
172
  };
137
173
  Object.defineProperty(contextDecorator, "name", {
138
174
  value: [f, key].join("_decorator_for_"),
@@ -148,7 +184,7 @@ class Decoration {
148
184
  apply() {
149
185
  if (!this.key)
150
186
  throw new Error("No key provided for the decoration builder");
151
- Decoration.register(this.key, this.flavour, this.decorators, this.extras);
187
+ Decoration.register(this.key, this.flavour, this.decorators || new Set(), this.extras);
152
188
  return this.decoratorFactory(this.key, this.flavour);
153
189
  }
154
190
  /**
@@ -160,8 +196,9 @@ class Decoration {
160
196
  * @param [extras] Additional decorators
161
197
  */
162
198
  static register(key, flavour, decorators, extras) {
163
- if (!key)
199
+ if (!key) {
164
200
  throw new Error("No key provided for the decoration builder");
201
+ }
165
202
  if (!decorators)
166
203
  throw new Error("No decorators provided for the decoration builder");
167
204
  if (!flavour)
@@ -191,4 +228,4 @@ class Decoration {
191
228
  }
192
229
  }
193
230
  exports.Decoration = Decoration;
194
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Decoration.js","sourceRoot":"","sources":["../../src/utils/Decoration.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  DecorationBuilderBuild,\n  DecorationBuilderEnd,\n  DecorationBuilderMid,\n  DecorationBuilderStart,\n  FlavourResolver,\n  IDecorationBuilder,\n} from \"./types\";\nimport { DefaultFlavour } from \"./constants\";\n\n// eslint-disable-next-line @typescript-eslint/no-unused-vars\nfunction defaultFlavourResolver(target: object) {\n  return DefaultFlavour;\n}\n\n/**\n * @description A decorator management class that handles flavoured decorators\n * @summary The Decoration class provides a builder pattern for creating and managing decorators with different flavours.\n * It supports registering, extending, and applying decorators with context-aware flavour resolution.\n * The class implements a fluent interface for defining, extending, and applying decorators with different flavours,\n * allowing for framework-specific decorator implementations while maintaining a consistent API.\n * @template T Type of the decorator (ClassDecorator | PropertyDecorator | MethodDecorator)\n * @param {string} [flavour] Optional flavour parameter for the decorator context\n * @class\n * @category Model\n * @example\n * ```typescript\n * // Create a new decoration for 'component' with default flavour\n * const componentDecorator = new Decoration()\n *   .for('component')\n *   .define(customComponentDecorator);\n *\n * // Create a flavoured decoration\n * const vueComponent = new Decoration('vue')\n *   .for('component')\n *   .define(vueComponentDecorator);\n *\n * // Apply the decoration\n * @componentDecorator\n * class MyComponent {}\n * ```\n * @mermaid\n * sequenceDiagram\n *   participant C as Client\n *   participant D as Decoration\n *   participant R as FlavourResolver\n *   participant F as DecoratorFactory\n *\n *   C->>D: new Decoration(flavour)\n *   C->>D: for(key)\n *   C->>D: define(decorators)\n *   D->>D: register(key, flavour, decorators)\n *   D->>F: decoratorFactory(key, flavour)\n *   F->>R: resolve(target)\n *   R-->>F: resolved flavour\n *   F->>F: apply decorators\n *   F-->>C: decorated target\n */\nexport class Decoration implements IDecorationBuilder {\n  /**\n   * @description Static map of registered decorators\n   * @summary Stores all registered decorators organized by key and flavour\n   */\n  private static decorators: Record<\n    string,\n    Record<\n      string,\n      {\n        decorators?: Set<ClassDecorator | PropertyDecorator | MethodDecorator>;\n        extras?: Set<ClassDecorator | PropertyDecorator | MethodDecorator>;\n      }\n    >\n  > = {};\n\n  /**\n   * @description Function to resolve flavour from a target\n   * @summary Resolver function that determines the appropriate flavour for a given target\n   */\n  private static flavourResolver: FlavourResolver = defaultFlavourResolver;\n\n  /**\n   * @description Set of decorators for the current context\n   */\n  private decorators?: Set<\n    ClassDecorator | PropertyDecorator | MethodDecorator\n  >;\n\n  /**\n   * @description Set of additional decorators\n   */\n  private extras?: Set<ClassDecorator | PropertyDecorator | MethodDecorator>;\n\n  /**\n   * @description Current decorator key\n   */\n  private key?: string;\n\n  constructor(private flavour: string = DefaultFlavour) {}\n\n  /**\n   * @description Sets the key for the decoration builder\n   * @summary Initializes a new decoration chain with the specified key\n   * @param {string} key The identifier for the decorator\n   * @return {DecorationBuilderMid} Builder instance for method chaining\n   */\n  for(key: string): DecorationBuilderMid {\n    this.key = key;\n    return this;\n  }\n\n  /**\n   * @description Adds decorators to the current context\n   * @summary Internal method to add decorators with addon support\n   * @param {boolean} [addon=false] Whether the decorators are addons\n   * @param decorators Array of decorators\n   * @return {this} Current instance for chaining\n   */\n  private decorate(\n    addon: boolean = false,\n    ...decorators: (ClassDecorator | PropertyDecorator | MethodDecorator)[]\n  ): this {\n    if (!this.key)\n      throw new Error(\"key must be provided before decorators can be added\");\n    if (\n      (!decorators || !decorators.length) &&\n      !addon &&\n      this.flavour !== DefaultFlavour\n    )\n      throw new Error(\n        \"Must provide overrides or addons to override or extend decaf's decorators\"\n      );\n    if (this.flavour === DefaultFlavour && addon)\n      throw new Error(\"Default flavour cannot be extended\");\n\n    this[addon ? \"extras\" : \"decorators\"] = new Set([\n      ...(this[addon ? \"extras\" : \"decorators\"] || new Set()).values(),\n      ...decorators,\n    ]);\n\n    return this;\n  }\n\n  /**\n   * @description Defines the base decorators\n   * @summary Sets the primary decorators for the current context\n   * @param decorators Decorators to define\n   * @return Builder instance for finishing the chain\n   */\n  define(\n    ...decorators: (ClassDecorator | PropertyDecorator | MethodDecorator)[]\n  ): DecorationBuilderEnd & DecorationBuilderBuild {\n    return this.decorate(false, ...decorators);\n  }\n\n  /**\n   * @description Extends existing decorators\n   * @summary Adds additional decorators to the current context\n   * @param decorators Additional decorators\n   * @return {DecorationBuilderBuild} Builder instance for building the decorator\n   */\n  extend(\n    ...decorators: (ClassDecorator | PropertyDecorator | MethodDecorator)[]\n  ): DecorationBuilderBuild {\n    return this.decorate(true, ...decorators);\n  }\n\n  protected decoratorFactory(key: string, f: string = DefaultFlavour) {\n    const contextDecorator = function contextDecorator(\n      target: object,\n      propertyKey?: any,\n      descriptor?: TypedPropertyDescriptor<any>\n    ) {\n      const flavour = Decoration.flavourResolver(target);\n      let decorators;\n      const extras = Decoration.decorators[key][flavour]\n        ? Decoration.decorators[key][flavour].extras\n        : Decoration.decorators[key][DefaultFlavour].extras;\n      if (\n        Decoration.decorators[key] &&\n        Decoration.decorators[key][flavour] &&\n        Decoration.decorators[key][flavour].decorators\n      ) {\n        decorators = Decoration.decorators[key][flavour].decorators;\n      } else {\n        decorators = Decoration.decorators[key][DefaultFlavour].decorators;\n      }\n      const toApply = [\n        ...(decorators ? decorators.values() : []),\n        ...(extras ? extras.values() : []),\n      ];\n      toApply.forEach((d) =>\n        (d as any)(target, propertyKey, descriptor, descriptor)\n      );\n    };\n    Object.defineProperty(contextDecorator, \"name\", {\n      value: [f, key].join(\"_decorator_for_\"),\n      writable: false,\n    });\n    return contextDecorator;\n  }\n\n  /**\n   * @description Creates the final decorator function\n   * @summary Builds and returns the decorator factory function\n   * @return {function(any, any?, TypedPropertyDescriptor?): any} The generated decorator function\n   */\n  apply(): (\n    target: any,\n    propertyKey?: any,\n    descriptor?: TypedPropertyDescriptor<any>\n  ) => any {\n    if (!this.key)\n      throw new Error(\"No key provided for the decoration builder\");\n    Decoration.register(this.key, this.flavour, this.decorators, this.extras);\n    return this.decoratorFactory(this.key, this.flavour);\n  }\n\n  /**\n   * @description Registers decorators for a specific key and flavour\n   * @summary Internal method to store decorators in the static registry\n   * @param {string} key Decorator key\n   * @param {string} flavour Decorator flavour\n   * @param [decorators] Primary decorators\n   * @param [extras] Additional decorators\n   */\n  private static register(\n    key: string,\n    flavour: string,\n    decorators?: Set<ClassDecorator | PropertyDecorator | MethodDecorator>,\n    extras?: Set<ClassDecorator | PropertyDecorator | MethodDecorator>\n  ) {\n    if (!key) throw new Error(\"No key provided for the decoration builder\");\n    if (!decorators)\n      throw new Error(\"No decorators provided for the decoration builder\");\n    if (!flavour)\n      throw new Error(\"No flavour provided for the decoration builder\");\n\n    if (!Decoration.decorators[key]) Decoration.decorators[key] = {};\n    if (!Decoration.decorators[key][flavour])\n      Decoration.decorators[key][flavour] = {};\n    if (decorators) Decoration.decorators[key][flavour].decorators = decorators;\n    if (extras) Decoration.decorators[key][flavour].extras = extras;\n  }\n\n  /**\n   * @description Sets the global flavour resolver\n   * @summary Configures the function used to determine decorator flavours\n   * @param {FlavourResolver} resolver Function to resolve flavours\n   */\n  static setFlavourResolver(resolver: FlavourResolver) {\n    Decoration.flavourResolver = resolver;\n  }\n\n  static for(key: string): DecorationBuilderMid {\n    return new Decoration().for(key);\n  }\n\n  static flavouredAs(flavour: string): DecorationBuilderStart {\n    return new Decoration(flavour);\n  }\n}\n"]}
231
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Decoration.js","sourceRoot":"","sources":["../../src/utils/Decoration.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  DecorationBuilderBuild,\n  DecorationBuilderEnd,\n  DecorationBuilderMid,\n  DecorationBuilderStart,\n  FlavourResolver,\n  IDecorationBuilder,\n} from \"./types\";\nimport { DefaultFlavour } from \"./constants\";\n\n// eslint-disable-next-line @typescript-eslint/no-unused-vars\nfunction defaultFlavourResolver(target: object) {\n  return DefaultFlavour;\n}\n\nexport type DecoratorTypes =\n  | ClassDecorator\n  | PropertyDecorator\n  | MethodDecorator;\n\nexport type DecoratorFactory = (...args: any[]) => DecoratorTypes;\n\nexport type DecoratorFactoryArgs = {\n  decorator: DecoratorFactory;\n  args?: any[];\n  transform?: (args: any[]) => any[];\n};\n\nexport type DecoratorData = DecoratorTypes | DecoratorFactoryArgs;\n/**\n * @description A decorator management class that handles flavoured decorators\n * @summary The Decoration class provides a builder pattern for creating and managing decorators with different flavours.\n * It supports registering, extending, and applying decorators with context-aware flavour resolution.\n * The class implements a fluent interface for defining, extending, and applying decorators with different flavours,\n * allowing for framework-specific decorator implementations while maintaining a consistent API.\n * @template T Type of the decorator (ClassDecorator | PropertyDecorator | MethodDecorator)\n * @param {string} [flavour] Optional flavour parameter for the decorator context\n * @class\n * @category Model\n * @example\n * ```typescript\n * // Create a new decoration for 'component' with default flavour\n * const componentDecorator = new Decoration()\n *   .for('component')\n *   .define(customComponentDecorator);\n *\n * // Create a flavoured decoration\n * const vueComponent = new Decoration('vue')\n *   .for('component')\n *   .define(vueComponentDecorator);\n *\n * // Apply the decoration\n * @componentDecorator\n * class MyComponent {}\n * ```\n * @mermaid\n * sequenceDiagram\n *   participant C as Client\n *   participant D as Decoration\n *   participant R as FlavourResolver\n *   participant F as DecoratorFactory\n *\n *   C->>D: new Decoration(flavour)\n *   C->>D: for(key)\n *   C->>D: define(decorators)\n *   D->>D: register(key, flavour, decorators)\n *   D->>F: decoratorFactory(key, flavour)\n *   F->>R: resolve(target)\n *   R-->>F: resolved flavour\n *   F->>F: apply decorators\n *   F-->>C: decorated target\n */\nexport class Decoration implements IDecorationBuilder {\n  /**\n   * @description Static map of registered decorators\n   * @summary Stores all registered decorators organized by key and flavour\n   */\n  private static decorators: Record<\n    string,\n    Record<\n      string,\n      {\n        decorators?: Set<DecoratorData>;\n        extras?: Set<DecoratorData>;\n      }\n    >\n  > = {};\n\n  /**\n   * @description Function to resolve flavour from a target\n   * @summary Resolver function that determines the appropriate flavour for a given target\n   */\n  private static flavourResolver: FlavourResolver = defaultFlavourResolver;\n\n  /**\n   * @description Set of decorators for the current context\n   */\n  private decorators?: Set<DecoratorData>;\n\n  /**\n   * @description Set of additional decorators\n   */\n  private extras?: Set<DecoratorData>;\n\n  /**\n   * @description Current decorator key\n   */\n  private key?: string;\n\n  constructor(private flavour: string = DefaultFlavour) {}\n\n  /**\n   * @description Sets the key for the decoration builder\n   * @summary Initializes a new decoration chain with the specified key\n   * @param {string} key The identifier for the decorator\n   * @return {DecorationBuilderMid} Builder instance for method chaining\n   */\n  for(key: string): DecorationBuilderMid {\n    this.key = key;\n    return this;\n  }\n\n  /**\n   * @description Adds decorators to the current context\n   * @summary Internal method to add decorators with addon support\n   * @param {boolean} [addon=false] Whether the decorators are addons\n   * @param decorators Array of decorators\n   * @return {this} Current instance for chaining\n   */\n  private decorate(\n    addon: boolean = false,\n    ...decorators: DecoratorData[]\n  ): this {\n    if (!this.key)\n      throw new Error(\"key must be provided before decorators can be added\");\n    if (\n      (!decorators || !decorators.length) &&\n      !addon &&\n      this.flavour !== DefaultFlavour\n    )\n      throw new Error(\n        \"Must provide overrides or addons to override or extend decaf's decorators\"\n      );\n    if (this.flavour === DefaultFlavour && addon)\n      throw new Error(\"Default flavour cannot be extended\");\n\n    this[addon ? \"extras\" : \"decorators\"] = new Set([\n      ...(this[addon ? \"extras\" : \"decorators\"] || new Set()).values(),\n      ...decorators,\n    ]);\n\n    return this;\n  }\n\n  /**\n   * @description Defines the base decorators\n   * @summary Sets the primary decorators for the current context\n   * @param decorators Decorators to define\n   * @return Builder instance for finishing the chain\n   */\n  define(\n    ...decorators: DecoratorData[]\n  ): DecorationBuilderEnd & DecorationBuilderBuild {\n    return this.decorate(false, ...decorators);\n  }\n\n  /**\n   * @description Extends existing decorators\n   * @summary Adds additional decorators to the current context\n   * @param decorators Additional decorators\n   * @return {DecorationBuilderBuild} Builder instance for building the decorator\n   */\n  extend(...decorators: DecoratorData[]): DecorationBuilderBuild {\n    return this.decorate(true, ...decorators);\n  }\n\n  protected decoratorFactory(key: string, f: string = DefaultFlavour) {\n    const contextDecorator = function contextDecorator(\n      target: object,\n      propertyKey?: any,\n      descriptor?: TypedPropertyDescriptor<any>\n    ) {\n      const flavour = Decoration.flavourResolver(target);\n      const cache = Decoration.decorators[key];\n      let decorators;\n      const extras = cache[flavour]\n        ? cache[flavour].extras\n        : cache[DefaultFlavour].extras;\n      const extraArgs = [\n        ...((cache[DefaultFlavour] as any).extras\n          ? (cache[DefaultFlavour] as any).extras.values()\n          : []),\n      ].reduce((accum: Record<number, any>, e, i) => {\n        if (e.args) accum[i] = e.args;\n        return accum;\n      }, {});\n\n      if (\n        cache &&\n        cache[flavour] &&\n        cache[flavour].decorators &&\n        cache[flavour].decorators.size\n      ) {\n        decorators = cache[flavour].decorators;\n      } else {\n        decorators = cache[DefaultFlavour].decorators;\n      }\n\n      const decoratorArgs = [\n        ...(cache[DefaultFlavour] as any).decorators.values(),\n      ].reduce((accum: Record<number, any>, e, i) => {\n        if (e.args) accum[i] = e.args;\n        return accum;\n      }, {});\n\n      const toApply = [\n        ...(decorators ? decorators.values() : []),\n        ...(extras ? extras.values() : []),\n      ];\n\n      return toApply.reduce(\n        (_, d, i) => {\n          switch (typeof d) {\n            case \"object\": {\n              const { decorator, args, transform } = d as DecoratorFactoryArgs;\n              const argz =\n                args || i < (decorators ? decorators.size : 0)\n                  ? decoratorArgs[i]\n                  : extraArgs[i - (decorators ? decorators.size : 0)] ||\n                    (decorators ? decoratorArgs[i - decorators.size] : []);\n\n              const transformed = transform\n                ? transform(argz || [])\n                : argz || [];\n              return (decorator(...transformed) as any)(\n                target,\n                propertyKey,\n                descriptor\n              );\n            }\n            case \"function\":\n              return (d as any)(target, propertyKey, descriptor);\n            default:\n              throw new Error(`Unexpected decorator type: ${typeof d}`);\n          }\n        },\n        { target, propertyKey, descriptor }\n      );\n    };\n    Object.defineProperty(contextDecorator, \"name\", {\n      value: [f, key].join(\"_decorator_for_\"),\n      writable: false,\n    });\n    return contextDecorator;\n  }\n\n  /**\n   * @description Creates the final decorator function\n   * @summary Builds and returns the decorator factory function\n   * @return {function(any, any?, TypedPropertyDescriptor?): any} The generated decorator function\n   */\n  apply(): (\n    target: any,\n    propertyKey?: any,\n    descriptor?: TypedPropertyDescriptor<any>\n  ) => any {\n    if (!this.key)\n      throw new Error(\"No key provided for the decoration builder\");\n    Decoration.register(\n      this.key,\n      this.flavour,\n      this.decorators || new Set(),\n      this.extras\n    );\n    return this.decoratorFactory(this.key, this.flavour);\n  }\n\n  /**\n   * @description Registers decorators for a specific key and flavour\n   * @summary Internal method to store decorators in the static registry\n   * @param {string} key Decorator key\n   * @param {string} flavour Decorator flavour\n   * @param [decorators] Primary decorators\n   * @param [extras] Additional decorators\n   */\n  private static register(\n    key: string,\n    flavour: string,\n    decorators?: Set<DecoratorData>,\n    extras?: Set<DecoratorData>\n  ) {\n    if (!key) {\n      throw new Error(\"No key provided for the decoration builder\");\n    }\n    if (!decorators)\n      throw new Error(\"No decorators provided for the decoration builder\");\n    if (!flavour)\n      throw new Error(\"No flavour provided for the decoration builder\");\n\n    if (!Decoration.decorators[key]) Decoration.decorators[key] = {};\n    if (!Decoration.decorators[key][flavour])\n      Decoration.decorators[key][flavour] = {};\n    if (decorators) Decoration.decorators[key][flavour].decorators = decorators;\n    if (extras) Decoration.decorators[key][flavour].extras = extras;\n  }\n\n  /**\n   * @description Sets the global flavour resolver\n   * @summary Configures the function used to determine decorator flavours\n   * @param {FlavourResolver} resolver Function to resolve flavours\n   */\n  static setFlavourResolver(resolver: FlavourResolver) {\n    Decoration.flavourResolver = resolver;\n  }\n\n  static for(key: string): DecorationBuilderMid {\n    return new Decoration().for(key);\n  }\n\n  static flavouredAs(flavour: string): DecorationBuilderStart {\n    return new Decoration(flavour);\n  }\n}\n"]}
@@ -1,4 +1,12 @@
1
1
  import { DecorationBuilderBuild, DecorationBuilderEnd, DecorationBuilderMid, DecorationBuilderStart, FlavourResolver, IDecorationBuilder } from "./types";
2
+ export type DecoratorTypes = ClassDecorator | PropertyDecorator | MethodDecorator;
3
+ export type DecoratorFactory = (...args: any[]) => DecoratorTypes;
4
+ export type DecoratorFactoryArgs = {
5
+ decorator: DecoratorFactory;
6
+ args?: any[];
7
+ transform?: (args: any[]) => any[];
8
+ };
9
+ export type DecoratorData = DecoratorTypes | DecoratorFactoryArgs;
2
10
  /**
3
11
  * @description A decorator management class that handles flavoured decorators
4
12
  * @summary The Decoration class provides a builder pattern for creating and managing decorators with different flavours.
@@ -88,15 +96,19 @@ export declare class Decoration implements IDecorationBuilder {
88
96
  * @param decorators Decorators to define
89
97
  * @return Builder instance for finishing the chain
90
98
  */
91
- define(...decorators: (ClassDecorator | PropertyDecorator | MethodDecorator)[]): DecorationBuilderEnd & DecorationBuilderBuild;
99
+ define(...decorators: DecoratorData[]): DecorationBuilderEnd & DecorationBuilderBuild;
92
100
  /**
93
101
  * @description Extends existing decorators
94
102
  * @summary Adds additional decorators to the current context
95
103
  * @param decorators Additional decorators
96
104
  * @return {DecorationBuilderBuild} Builder instance for building the decorator
97
105
  */
98
- extend(...decorators: (ClassDecorator | PropertyDecorator | MethodDecorator)[]): DecorationBuilderBuild;
99
- protected decoratorFactory(key: string, f?: string): (target: object, propertyKey?: any, descriptor?: TypedPropertyDescriptor<any>) => void;
106
+ extend(...decorators: DecoratorData[]): DecorationBuilderBuild;
107
+ protected decoratorFactory(key: string, f?: string): (target: object, propertyKey?: any, descriptor?: TypedPropertyDescriptor<any>) => {
108
+ target: object;
109
+ propertyKey: any;
110
+ descriptor: TypedPropertyDescriptor<any> | undefined;
111
+ };
100
112
  /**
101
113
  * @description Creates the final decorator function
102
114
  * @summary Builds and returns the decorator factory function