@decaf-ts/db-decorators 0.6.1 → 0.6.3

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 (141) hide show
  1. package/LICENSE.md +21 -157
  2. package/README.md +571 -10
  3. package/dist/db-decorators.cjs +1599 -426
  4. package/dist/db-decorators.esm.cjs +1597 -428
  5. package/lib/esm/identity/decorators.d.ts +7 -0
  6. package/lib/esm/identity/decorators.js +11 -4
  7. package/lib/esm/identity/index.js +3 -3
  8. package/lib/esm/identity/utils.d.ts +36 -23
  9. package/lib/esm/identity/utils.js +38 -25
  10. package/lib/esm/index.d.ts +12 -27
  11. package/lib/esm/index.js +13 -28
  12. package/lib/esm/interfaces/BulkCrudOperator.d.ts +39 -0
  13. package/lib/esm/interfaces/BulkCrudOperator.js +1 -1
  14. package/lib/esm/interfaces/Contextual.d.ts +17 -0
  15. package/lib/esm/interfaces/Contextual.js +1 -1
  16. package/lib/esm/interfaces/CrudOperator.d.ts +26 -23
  17. package/lib/esm/interfaces/CrudOperator.js +1 -1
  18. package/lib/esm/interfaces/IRepository.d.ts +10 -2
  19. package/lib/esm/interfaces/IRepository.js +1 -1
  20. package/lib/esm/interfaces/index.js +5 -5
  21. package/lib/esm/model/constants.d.ts +11 -13
  22. package/lib/esm/model/constants.js +12 -14
  23. package/lib/esm/model/decorators.d.ts +112 -23
  24. package/lib/esm/model/decorators.js +119 -29
  25. package/lib/esm/model/index.d.ts +1 -0
  26. package/lib/esm/model/index.js +7 -6
  27. package/lib/esm/model/model.d.ts +2 -141
  28. package/lib/esm/model/model.js +2 -13
  29. package/lib/esm/model/overrides.d.ts +1 -0
  30. package/lib/esm/model/overrides.js +34 -0
  31. package/lib/esm/model/utils.d.ts +39 -0
  32. package/lib/esm/model/utils.js +42 -3
  33. package/lib/esm/model/validation.d.ts +51 -8
  34. package/lib/esm/model/validation.js +246 -107
  35. package/lib/esm/operations/Operations.d.ts +65 -3
  36. package/lib/esm/operations/Operations.js +68 -6
  37. package/lib/esm/operations/OperationsRegistry.d.ts +44 -16
  38. package/lib/esm/operations/OperationsRegistry.js +46 -18
  39. package/lib/esm/operations/constants.d.ts +27 -8
  40. package/lib/esm/operations/constants.js +16 -9
  41. package/lib/esm/operations/decorators.d.ts +140 -134
  42. package/lib/esm/operations/decorators.js +152 -137
  43. package/lib/esm/operations/index.js +6 -6
  44. package/lib/esm/operations/types.d.ts +10 -0
  45. package/lib/esm/operations/types.js +1 -1
  46. package/lib/esm/repository/BaseRepository.d.ts +322 -0
  47. package/lib/esm/repository/BaseRepository.js +297 -7
  48. package/lib/esm/repository/Context.d.ts +153 -2
  49. package/lib/esm/repository/Context.js +154 -6
  50. package/lib/esm/repository/Repository.d.ts +89 -0
  51. package/lib/esm/repository/Repository.js +102 -15
  52. package/lib/esm/repository/constants.d.ts +7 -0
  53. package/lib/esm/repository/constants.js +8 -1
  54. package/lib/esm/repository/errors.d.ts +61 -34
  55. package/lib/esm/repository/errors.js +62 -35
  56. package/lib/esm/repository/index.js +9 -9
  57. package/lib/esm/repository/types.d.ts +25 -0
  58. package/lib/esm/repository/types.js +1 -1
  59. package/lib/esm/repository/utils.d.ts +11 -0
  60. package/lib/esm/repository/utils.js +4 -4
  61. package/lib/esm/repository/wrappers.d.ts +2 -2
  62. package/lib/esm/repository/wrappers.js +5 -5
  63. package/lib/esm/validation/constants.d.ts +20 -5
  64. package/lib/esm/validation/constants.js +22 -7
  65. package/lib/esm/validation/decorators.d.ts +101 -19
  66. package/lib/esm/validation/decorators.js +109 -27
  67. package/lib/esm/validation/index.js +5 -5
  68. package/lib/esm/validation/validation.js +10 -2
  69. package/lib/esm/validation/validators/ReadOnlyValidator.d.ts +32 -8
  70. package/lib/esm/validation/validators/ReadOnlyValidator.js +34 -10
  71. package/lib/esm/validation/validators/TimestampValidator.d.ts +37 -3
  72. package/lib/esm/validation/validators/TimestampValidator.js +39 -5
  73. package/lib/esm/validation/validators/UpdateValidator.d.ts +28 -11
  74. package/lib/esm/validation/validators/UpdateValidator.js +23 -8
  75. package/lib/esm/validation/validators/index.js +4 -4
  76. package/lib/identity/decorators.cjs +8 -1
  77. package/lib/identity/decorators.d.ts +7 -0
  78. package/lib/identity/utils.cjs +35 -22
  79. package/lib/identity/utils.d.ts +36 -23
  80. package/lib/index.cjs +14 -28
  81. package/lib/index.d.ts +12 -27
  82. package/lib/interfaces/BulkCrudOperator.cjs +1 -1
  83. package/lib/interfaces/BulkCrudOperator.d.ts +39 -0
  84. package/lib/interfaces/Contextual.cjs +1 -1
  85. package/lib/interfaces/Contextual.d.ts +17 -0
  86. package/lib/interfaces/CrudOperator.cjs +1 -1
  87. package/lib/interfaces/CrudOperator.d.ts +26 -23
  88. package/lib/interfaces/IRepository.cjs +1 -1
  89. package/lib/interfaces/IRepository.d.ts +10 -2
  90. package/lib/model/constants.cjs +12 -14
  91. package/lib/model/constants.d.ts +11 -13
  92. package/lib/model/decorators.cjs +114 -24
  93. package/lib/model/decorators.d.ts +112 -23
  94. package/lib/model/index.cjs +2 -1
  95. package/lib/model/index.d.ts +1 -0
  96. package/lib/model/model.cjs +1 -13
  97. package/lib/model/model.d.ts +2 -141
  98. package/lib/model/overrides.cjs +36 -0
  99. package/lib/model/overrides.d.ts +1 -0
  100. package/lib/model/utils.cjs +40 -1
  101. package/lib/model/utils.d.ts +39 -0
  102. package/lib/model/validation.cjs +246 -104
  103. package/lib/model/validation.d.ts +51 -8
  104. package/lib/operations/Operations.cjs +66 -4
  105. package/lib/operations/Operations.d.ts +65 -3
  106. package/lib/operations/OperationsRegistry.cjs +45 -17
  107. package/lib/operations/OperationsRegistry.d.ts +44 -16
  108. package/lib/operations/constants.cjs +16 -9
  109. package/lib/operations/constants.d.ts +27 -8
  110. package/lib/operations/decorators.cjs +150 -135
  111. package/lib/operations/decorators.d.ts +140 -134
  112. package/lib/operations/types.cjs +1 -1
  113. package/lib/operations/types.d.ts +10 -0
  114. package/lib/repository/BaseRepository.cjs +291 -1
  115. package/lib/repository/BaseRepository.d.ts +322 -0
  116. package/lib/repository/Context.cjs +153 -5
  117. package/lib/repository/Context.d.ts +153 -2
  118. package/lib/repository/Repository.cjs +96 -9
  119. package/lib/repository/Repository.d.ts +89 -0
  120. package/lib/repository/constants.cjs +8 -1
  121. package/lib/repository/constants.d.ts +7 -0
  122. package/lib/repository/errors.cjs +62 -35
  123. package/lib/repository/errors.d.ts +61 -34
  124. package/lib/repository/types.cjs +1 -1
  125. package/lib/repository/types.d.ts +25 -0
  126. package/lib/repository/utils.cjs +1 -1
  127. package/lib/repository/utils.d.ts +11 -0
  128. package/lib/repository/wrappers.cjs +3 -3
  129. package/lib/repository/wrappers.d.ts +2 -2
  130. package/lib/validation/constants.cjs +21 -6
  131. package/lib/validation/constants.d.ts +20 -5
  132. package/lib/validation/decorators.cjs +102 -20
  133. package/lib/validation/decorators.d.ts +101 -19
  134. package/lib/validation/validation.cjs +9 -1
  135. package/lib/validation/validators/ReadOnlyValidator.cjs +33 -9
  136. package/lib/validation/validators/ReadOnlyValidator.d.ts +32 -8
  137. package/lib/validation/validators/TimestampValidator.cjs +38 -4
  138. package/lib/validation/validators/TimestampValidator.d.ts +37 -3
  139. package/lib/validation/validators/UpdateValidator.cjs +23 -8
  140. package/lib/validation/validators/UpdateValidator.d.ts +28 -11
  141. package/package.json +2 -2
@@ -1,125 +1,264 @@
1
- import { ModelErrorDefinition, ModelKeys, ReservedModels, sf, Validation, ValidationKeys, } from "@decaf-ts/decorator-validation";
1
+ import { getValidationDecorators, Model, ModelErrorDefinition, ModelKeys, toConditionalPromise, Validation, ValidationKeys, } from "@decaf-ts/decorator-validation";
2
2
  import { Reflection } from "@decaf-ts/reflection";
3
- import { UpdateValidationKeys } from "../validation";
4
- import { findModelId } from "../identity";
3
+ import { UpdateValidationKeys } from "./../validation/index.js";
4
+ import { findModelId } from "./../identity/index.js";
5
5
  /**
6
- * @summary Validates the update of a model
6
+ * @description
7
+ * Retrieves validation decorator definitions from a model for update operations, including
8
+ * support for special handling of list decorators.
7
9
  *
8
- * @param {T} oldModel
9
- * @param {T} newModel
10
- * @param {string[]} [exceptions]
10
+ * @summary
11
+ * Iterates over the model's own enumerable properties and filters out those specified in the
12
+ * `propsToIgnore` array. For each remaining property, retrieves validation decorators specific
13
+ * to update operations using the `UpdateValidationKeys.REFLECT` key. Additionally, it explicitly
14
+ * checks for and appends any `LIST` type decorators to ensure proper validation of collection types.
11
15
  *
12
- * @function validateCompare
13
- * @return {ModelErrorDefinition | undefined}
16
+ * @template M - A generic parameter extending the `Model` class, representing the model type being inspected.
17
+ *
18
+ * @param {M} model - The model instance whose properties are being inspected for update-related validations.
19
+ * @param {string[]} propsToIgnore - A list of property names to exclude from the validation decorator retrieval process.
20
+ *
21
+ * @return {ValidationPropertyDecoratorDefinition[]} An array of validation decorator definitions, including both
22
+ * update-specific and list-type decorators, excluding those for ignored properties.
14
23
  *
15
- * @memberOf module:db-decorators.Model
24
+ * @function getValidatableUpdateProps
16
25
  */
17
- export function validateCompare(oldModel, newModel, ...exceptions) {
26
+ export function getValidatableUpdateProps(model, propsToIgnore) {
18
27
  const decoratedProperties = [];
19
- for (const prop in newModel)
20
- if (Object.prototype.hasOwnProperty.call(newModel, prop) &&
21
- exceptions.indexOf(prop) === -1)
22
- decoratedProperties.push(Reflection.getPropertyDecorators(UpdateValidationKeys.REFLECT, newModel, prop));
23
- let result = undefined;
24
- for (const decoratedProperty of decoratedProperties) {
25
- const { prop, decorators } = decoratedProperty;
26
- decorators.shift(); // remove the design:type decorator, since the type will already be checked
27
- if (!decorators || !decorators.length)
28
+ for (const prop in model) {
29
+ if (Object.prototype.hasOwnProperty.call(model, prop) &&
30
+ !propsToIgnore.includes(prop)) {
31
+ const validationPropertyDefinition = getValidationDecorators(model, prop, UpdateValidationKeys.REFLECT);
32
+ const listDecorator = getValidationDecorators(model, prop).decorators.find(({ key }) => key === ValidationKeys.LIST);
33
+ if (listDecorator)
34
+ validationPropertyDefinition.decorators.push(listDecorator);
35
+ decoratedProperties.push(validationPropertyDefinition);
36
+ }
37
+ }
38
+ return decoratedProperties;
39
+ }
40
+ export function validateDecorator(newModel, oldModel, prop, decorator, async) {
41
+ const validator = Validation.get(decorator.key);
42
+ if (!validator) {
43
+ throw new Error(`Missing validator for ${decorator.key}`);
44
+ }
45
+ // Skip validators that aren't UpdateValidators
46
+ if (!validator.updateHasErrors)
47
+ return toConditionalPromise(undefined, async);
48
+ // skip async decorators if validateDecorators is called synchronously (async = false)
49
+ if (!async && decorator.props.async)
50
+ return toConditionalPromise(undefined, async);
51
+ const decoratorProps = Object.values(decorator.props) || {};
52
+ // const context = PathProxyEngine.create(obj, {
53
+ // ignoreUndefined: true,
54
+ // ignoreNull: true,
55
+ // });
56
+ const maybeError = validator.updateHasErrors(newModel[prop], oldModel[prop], ...decoratorProps);
57
+ return toConditionalPromise(maybeError, async);
58
+ }
59
+ export function validateDecorators(newModel, oldModel, prop, decorators, async) {
60
+ const result = {};
61
+ for (const decorator of decorators) {
62
+ // skip async decorators if validateDecorators is called synchronously (async = false)
63
+ if (!async && decorator.props.async)
28
64
  continue;
29
- let errs = undefined;
30
- for (const decorator of decorators) {
31
- const validator = Validation.get(decorator.key);
32
- if (!validator) {
33
- console.error(`Could not find Matching validator for ${decorator.key} for property ${String(decoratedProperty.prop)}`);
34
- continue;
35
- }
36
- const err = validator.updateHasErrors(newModel[prop.toString()], oldModel[prop.toString()], ...Object.values(decorator.props));
37
- if (err) {
38
- errs = errs || {};
39
- errs[decorator.key] = err;
65
+ let validationErrors = validateDecorator(newModel, oldModel, prop, decorator, async);
66
+ /*
67
+ If the decorator is a list, each element must be checked.
68
+ When 'async' is true, the 'err' will always be a pending promise initially,
69
+ so the '!err' check will evaluate to false (even if the promise later resolves with no errors)
70
+ */
71
+ if (decorator.key === ValidationKeys.LIST && (!validationErrors || async)) {
72
+ const newPropValue = newModel[prop];
73
+ const oldPropValue = oldModel[prop];
74
+ const newValues = newPropValue instanceof Set ? [...newPropValue] : newPropValue;
75
+ const oldValues = oldPropValue instanceof Set ? [...oldPropValue] : oldPropValue;
76
+ if (newValues && newValues.length > 0) {
77
+ const types = decorator.props.class ||
78
+ decorator.props.clazz ||
79
+ decorator.props.customTypes;
80
+ const allowedTypes = [types].flat().map((t) => String(t).toLowerCase());
81
+ const errs = newValues.map((childValue) => {
82
+ // find by id so the list elements order doesn't matter
83
+ const id = findModelId(childValue, true);
84
+ if (!id)
85
+ return "Failed to find model id";
86
+ const oldModel = oldValues.find((el) => id === findModelId(el, true));
87
+ if (Model.isModel(childValue)) {
88
+ return childValue.hasErrors(oldModel);
89
+ }
90
+ return allowedTypes.includes(typeof childValue)
91
+ ? undefined
92
+ : "Value has no validatable type";
93
+ });
94
+ if (async) {
95
+ validationErrors = Promise.all(errs).then((result) => {
96
+ const allEmpty = result.every((r) => !r);
97
+ return allEmpty ? undefined : result;
98
+ });
99
+ }
100
+ else {
101
+ const allEmpty = errs.every((r) => !r);
102
+ validationErrors = errs.length > 0 && !allEmpty ? errs : undefined;
103
+ }
40
104
  }
41
105
  }
42
- if (errs) {
43
- result = result || {};
44
- result[decoratedProperty.prop.toString()] = errs;
45
- }
106
+ if (validationErrors)
107
+ result[decorator.key] = validationErrors;
46
108
  }
47
- // tests nested classes
48
- for (const prop of Object.keys(newModel).filter((k) => {
49
- if (exceptions.includes(k))
50
- return false;
51
- return !result || !result[k];
52
- })) {
53
- let err;
54
- // if a nested Model
55
- const allDecorators = Reflection.getPropertyDecorators(ValidationKeys.REFLECT, newModel, prop).decorators;
56
- const decorators = Reflection.getPropertyDecorators(ValidationKeys.REFLECT, newModel, prop).decorators.filter((d) => [ModelKeys.TYPE, ValidationKeys.TYPE].indexOf(d.key) !== -1);
57
- if (!decorators || !decorators.length)
109
+ if (!async)
110
+ return Object.keys(result).length > 0 ? result : undefined;
111
+ const keys = Object.keys(result);
112
+ const promises = Object.values(result);
113
+ return Promise.all(promises).then((resolvedValues) => {
114
+ const res = {};
115
+ for (let i = 0; i < resolvedValues.length; i++) {
116
+ const val = resolvedValues[i];
117
+ if (val !== undefined) {
118
+ res[keys[i]] = val;
119
+ }
120
+ }
121
+ return Object.keys(res).length > 0 ? res : undefined;
122
+ });
123
+ }
124
+ /**
125
+ * @description Validates changes between two model versions
126
+ * @summary Compares an old and new model version to validate update operations
127
+ * @template M - Type extending Model
128
+ * @param {M} oldModel - The original model version
129
+ * @param {M} newModel - The updated model version
130
+ * @param {boolean} async - A flag indicating whether validation should be asynchronous.
131
+ * @param {...string[]} exceptions - Properties to exclude from validation
132
+ * @return {ModelErrorDefinition|undefined} Error definition if validation fails, undefined otherwise
133
+ * @function validateCompare
134
+ * @memberOf module:db-decorators
135
+ * @mermaid
136
+ * sequenceDiagram
137
+ * participant Caller
138
+ * participant validateCompare
139
+ * participant Reflection
140
+ * participant Validation
141
+ *
142
+ * Caller->>validateCompare: oldModel, newModel, exceptions
143
+ * validateCompare->>Reflection: get decorated properties
144
+ * Reflection-->>validateCompare: property decorators
145
+ * loop For each decorated property
146
+ * validateCompare->>Validation: get validator
147
+ * Validation-->>validateCompare: validator
148
+ * validateCompare->>validateCompare: validate property update
149
+ * end
150
+ * loop For nested models
151
+ * validateCompare->>validateCompare: validate nested models
152
+ * end
153
+ * validateCompare-->>Caller: validation errors or undefined
154
+ */
155
+ export function validateCompare(oldModel, newModel, async, ...exceptions) {
156
+ const decoratedProperties = getValidatableUpdateProps(newModel, exceptions);
157
+ const result = {};
158
+ const nestedErrors = {};
159
+ for (const { prop, decorators } of decoratedProperties) {
160
+ const propKey = String(prop);
161
+ let propValue = newModel[prop];
162
+ if (!decorators?.length)
58
163
  continue;
59
- const dec = decorators.pop();
60
- const clazz = dec.props.name
61
- ? [dec.props.name]
62
- : Array.isArray(dec.props.customTypes)
63
- ? dec.props.customTypes
64
- : [dec.props.customTypes];
65
- const reserved = Object.values(ReservedModels).map((v) => v.toLowerCase());
66
- for (const c of clazz) {
67
- if (reserved.indexOf(c.toLowerCase()) === -1) {
68
- switch (c) {
69
- case Array.name:
70
- case Set.name:
71
- if (allDecorators.length) {
72
- const listDec = allDecorators.find((d) => d.key === ValidationKeys.LIST);
73
- if (listDec) {
74
- let currentList, oldList;
75
- switch (c) {
76
- case Array.name:
77
- currentList = newModel[prop];
78
- oldList = oldModel[prop];
79
- break;
80
- case Set.name:
81
- currentList = newModel[prop].values();
82
- oldList = oldModel[prop].values();
83
- break;
84
- default:
85
- throw new Error(`Invalid attribute type ${c}`);
86
- }
87
- err = currentList
88
- .map((v) => {
89
- const id = findModelId(v, true);
90
- if (!id)
91
- return "Failed to find model id";
92
- const oldModel = oldList.find((el) => id === findModelId(el, true));
93
- if (!oldModel)
94
- return; // nothing to compare with
95
- return v.hasErrors(oldModel);
96
- })
97
- .filter((e) => !!e);
98
- if (!err?.length) {
99
- // if the result is an empty list...
100
- err = undefined;
101
- }
102
- }
103
- }
104
- break;
105
- default:
106
- try {
107
- if (newModel[prop] &&
108
- oldModel[prop])
109
- err = newModel[prop].hasErrors(oldModel[prop]);
110
- // eslint-disable-next-line @typescript-eslint/no-unused-vars
111
- }
112
- catch (e) {
113
- console.warn(sf("Model should be validatable but its not"));
114
- }
164
+ // Get the default type validator
165
+ const designTypeDec = decorators.find((d) => [ModelKeys.TYPE, ValidationKeys.TYPE].includes(d.key));
166
+ if (!designTypeDec)
167
+ continue;
168
+ const designType = designTypeDec.props.name;
169
+ // Handle array or Set types and enforce the presence of @list decorator
170
+ if ([Array.name, Set.name].includes(designType)) {
171
+ const { decorators } = Reflection.getPropertyDecorators(ValidationKeys.REFLECT, newModel, propKey);
172
+ if (!decorators.some((d) => d.key === ValidationKeys.LIST)) {
173
+ result[propKey] = {
174
+ [ValidationKeys.TYPE]: `Array or Set property '${propKey}' requires a @list decorator`,
175
+ };
176
+ continue;
177
+ }
178
+ if (propValue &&
179
+ !(Array.isArray(propValue) || propValue instanceof Set)) {
180
+ result[propKey] = {
181
+ [ValidationKeys.TYPE]: `Property '${String(prop)}' must be either an array or a Set`,
182
+ };
183
+ continue;
184
+ }
185
+ // Remove design:type decorator, since @list decorator already ensures type
186
+ for (let i = decorators.length - 1; i >= 0; i--) {
187
+ if (decorators[i].key === ModelKeys.TYPE) {
188
+ decorators.splice(i, 1);
115
189
  }
116
190
  }
117
- if (err) {
118
- result = result || {};
119
- result[prop] = err;
191
+ propValue = propValue instanceof Set ? [...propValue] : propValue;
192
+ }
193
+ const propErrors = validateDecorators(newModel, oldModel, propKey, decorators, async) || {};
194
+ // Check for nested properties.
195
+ // To prevent unnecessary processing, "propValue" must be defined and validatable
196
+ const isConstr = Model.isPropertyModel(newModel, propKey);
197
+ // if propValue !== undefined, null
198
+ if (propValue && isConstr) {
199
+ const instance = propValue;
200
+ const isInvalidModel = typeof instance !== "object" ||
201
+ !instance.hasErrors ||
202
+ typeof instance.hasErrors !== "function";
203
+ if (isInvalidModel) {
204
+ // propErrors[ValidationKeys.TYPE] =
205
+ // "Model should be validatable but it's not.";
206
+ console.warn("Model should be validatable but it's not.");
120
207
  }
208
+ else {
209
+ nestedErrors[propKey] = instance.hasErrors(oldModel[prop]);
210
+ }
211
+ }
212
+ // Add to the result if we have any errors
213
+ // Async mode returns a Promise that resolves to undefined when no errors exist
214
+ if (Object.keys(propErrors).length > 0 || async)
215
+ result[propKey] = propErrors;
216
+ // Then merge any nested errors
217
+ if (!async) {
218
+ Object.entries(nestedErrors[propKey] || {}).forEach(([key, error]) => {
219
+ if (error !== undefined) {
220
+ result[`${propKey}.${key}`] = error;
221
+ }
222
+ });
121
223
  }
122
224
  }
123
- return result ? new ModelErrorDefinition(result) : undefined;
225
+ // Synchronous return
226
+ if (!async) {
227
+ return (Object.keys(result).length > 0
228
+ ? new ModelErrorDefinition(result)
229
+ : undefined);
230
+ }
231
+ const merged = result; // TODO: apply filtering
232
+ const keys = Object.keys(merged);
233
+ const promises = Object.values(merged);
234
+ return Promise.allSettled(promises).then(async (results) => {
235
+ const result = {};
236
+ for (const [parentProp, nestedErrPromise] of Object.entries(nestedErrors)) {
237
+ const nestedPropDecErrors = (await nestedErrPromise);
238
+ if (nestedPropDecErrors)
239
+ Object.entries(nestedPropDecErrors).forEach(([nestedProp, nestedPropDecError]) => {
240
+ if (nestedPropDecError !== undefined) {
241
+ const nestedKey = [parentProp, nestedProp].join(".");
242
+ result[nestedKey] = nestedPropDecError;
243
+ }
244
+ });
245
+ }
246
+ for (let i = 0; i < results.length; i++) {
247
+ const key = keys[i];
248
+ const res = results[i];
249
+ if (res.status === "fulfilled" && res.value !== undefined) {
250
+ result[key] = res.value;
251
+ }
252
+ else if (res.status === "rejected") {
253
+ result[key] =
254
+ res.reason instanceof Error
255
+ ? res.reason.message
256
+ : String(res.reason || "Validation failed");
257
+ }
258
+ }
259
+ return Object.keys(result).length > 0
260
+ ? new ModelErrorDefinition(result)
261
+ : undefined;
262
+ });
124
263
  }
125
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"validation.js","sourceRoot":"","sources":["../../../src/model/validation.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Model,\n  ModelErrorDefinition,\n  ModelErrors,\n  ModelKeys,\n  ReservedModels,\n  sf,\n  Validatable,\n  Validation,\n  ValidationKeys,\n  ValidationPropertyDecoratorDefinition,\n} from \"@decaf-ts/decorator-validation\";\nimport { DecoratorMetadata, Reflection } from \"@decaf-ts/reflection\";\nimport { UpdateValidationKeys, UpdateValidator } from \"../validation\";\nimport { findModelId } from \"../identity\";\n\n/**\n * @summary Validates the update of a model\n *\n * @param {T} oldModel\n * @param {T} newModel\n * @param {string[]} [exceptions]\n *\n * @function validateCompare\n * @return {ModelErrorDefinition | undefined}\n *\n * @memberOf module:db-decorators.Model\n */\nexport function validateCompare<M extends Model>(\n  oldModel: M,\n  newModel: M,\n  ...exceptions: string[]\n): ModelErrorDefinition | undefined {\n  const decoratedProperties: ValidationPropertyDecoratorDefinition[] = [];\n  for (const prop in newModel)\n    if (\n      Object.prototype.hasOwnProperty.call(newModel, prop) &&\n      exceptions.indexOf(prop) === -1\n    )\n      decoratedProperties.push(\n        Reflection.getPropertyDecorators(\n          UpdateValidationKeys.REFLECT,\n          newModel,\n          prop\n        ) as ValidationPropertyDecoratorDefinition\n      );\n\n  let result: ModelErrors | undefined = undefined;\n\n  for (const decoratedProperty of decoratedProperties) {\n    const { prop, decorators } = decoratedProperty;\n\n    decorators.shift(); // remove the design:type decorator, since the type will already be checked\n\n    if (!decorators || !decorators.length) continue;\n    let errs: Record<string, string | undefined> | undefined = undefined;\n\n    for (const decorator of decorators) {\n      const validator: UpdateValidator = Validation.get(\n        decorator.key\n      ) as UpdateValidator;\n      if (!validator) {\n        console.error(\n          `Could not find Matching validator for ${decorator.key} for property ${String(decoratedProperty.prop)}`\n        );\n        continue;\n      }\n\n      const err: string | undefined = validator.updateHasErrors(\n        (newModel as any)[prop.toString()],\n        (oldModel as any)[prop.toString()],\n        ...Object.values(decorator.props)\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  // tests nested classes\n  for (const prop of Object.keys(newModel).filter((k) => {\n    if (exceptions.includes(k)) return false;\n    return !result || !result[k];\n  })) {\n    let err: string | undefined;\n    // if a nested Model\n    const allDecorators = Reflection.getPropertyDecorators(\n      ValidationKeys.REFLECT,\n      newModel,\n      prop\n    ).decorators;\n    const decorators = Reflection.getPropertyDecorators(\n      ValidationKeys.REFLECT,\n      newModel,\n      prop\n    ).decorators.filter(\n      (d) => [ModelKeys.TYPE, ValidationKeys.TYPE].indexOf(d.key as any) !== -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        switch (c) {\n          case Array.name:\n          case Set.name:\n            if (allDecorators.length) {\n              const listDec = allDecorators.find(\n                (d) => d.key === ValidationKeys.LIST\n              );\n              if (listDec) {\n                let currentList, oldList;\n\n                switch (c) {\n                  case Array.name:\n                    currentList = (newModel as Record<string, any>)[prop];\n                    oldList = (oldModel as Record<string, any>)[prop];\n                    break;\n                  case Set.name:\n                    currentList = (newModel as Record<string, any>)[\n                      prop\n                    ].values();\n                    oldList = (oldModel as Record<string, any>)[prop].values();\n                    break;\n                  default:\n                    throw new Error(`Invalid attribute type ${c}`);\n                }\n\n                err = currentList\n                  .map((v: Validatable) => {\n                    const id = findModelId(v as any, true);\n                    if (!id) return \"Failed to find model id\";\n                    const oldModel = oldList.find(\n                      (el: any) => id === findModelId(el, true)\n                    );\n\n                    if (!oldModel) return; // nothing to compare with\n                    return v.hasErrors(oldModel);\n                  })\n                  .filter((e: any) => !!e) as any;\n\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 (\n                (newModel as Record<string, any>)[prop] &&\n                (oldModel as Record<string, any>)[prop]\n              )\n                err = (newModel as Record<string, any>)[prop].hasErrors(\n                  (oldModel as Record<string, any>)[prop]\n                );\n              // eslint-disable-next-line @typescript-eslint/no-unused-vars\n            } catch (e: any) {\n              console.warn(sf(\"Model should be validatable but its not\"));\n            }\n        }\n      }\n      if (err) {\n        result = result || {};\n        result[prop] = err as any;\n      }\n    }\n  }\n  return result ? new ModelErrorDefinition(result) : undefined;\n}\n"]}
264
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"validation.js","sourceRoot":"","sources":["../../../src/model/validation.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  ConditionalAsync,\n  DecoratorMetadataAsync,\n  getValidationDecorators,\n  Model,\n  ModelConditionalAsync,\n  ModelErrorDefinition,\n  ModelErrors,\n  ModelKeys,\n  toConditionalPromise,\n  Validation,\n  ValidationKeys,\n  ValidationPropertyDecoratorDefinition,\n} from \"@decaf-ts/decorator-validation\";\nimport { Reflection } from \"@decaf-ts/reflection\";\nimport { UpdateValidationKeys, UpdateValidator } from \"../validation\";\nimport { findModelId } from \"../identity\";\n\n/**\n * @description\n * Retrieves validation decorator definitions from a model for update operations, including\n * support for special handling of list decorators.\n *\n * @summary\n * Iterates over the model's own enumerable properties and filters out those specified in the\n * `propsToIgnore` array. For each remaining property, retrieves validation decorators specific\n * to update operations using the `UpdateValidationKeys.REFLECT` key. Additionally, it explicitly\n * checks for and appends any `LIST` type decorators to ensure proper validation of collection types.\n *\n * @template M - A generic parameter extending the `Model` class, representing the model type being inspected.\n *\n * @param {M} model - The model instance whose properties are being inspected for update-related validations.\n * @param {string[]} propsToIgnore - A list of property names to exclude from the validation decorator retrieval process.\n *\n * @return {ValidationPropertyDecoratorDefinition[]} An array of validation decorator definitions, including both\n * update-specific and list-type decorators, excluding those for ignored properties.\n *\n * @function getValidatableUpdateProps\n */\nexport function getValidatableUpdateProps<M extends Model>(\n  model: M,\n  propsToIgnore: string[]\n): ValidationPropertyDecoratorDefinition[] {\n  const decoratedProperties: ValidationPropertyDecoratorDefinition[] = [];\n  for (const prop in model) {\n    if (\n      Object.prototype.hasOwnProperty.call(model, prop) &&\n      !propsToIgnore.includes(prop)\n    ) {\n      const validationPropertyDefinition = getValidationDecorators(\n        model,\n        prop,\n        UpdateValidationKeys.REFLECT\n      );\n\n      const listDecorator = getValidationDecorators(\n        model,\n        prop\n      ).decorators.find(({ key }) => key === ValidationKeys.LIST);\n\n      if (listDecorator)\n        validationPropertyDefinition.decorators.push(listDecorator);\n\n      decoratedProperties.push(validationPropertyDefinition);\n    }\n  }\n\n  return decoratedProperties;\n}\n\nexport function validateDecorator<\n  M extends Model,\n  Async extends boolean = false,\n>(\n  newModel: M,\n  oldModel: M,\n  prop: string,\n  decorator: DecoratorMetadataAsync,\n  async?: Async\n): ConditionalAsync<Async, string | undefined> {\n  const validator: UpdateValidator = Validation.get(\n    decorator.key\n  ) as UpdateValidator;\n\n  if (!validator) {\n    throw new Error(`Missing validator for ${decorator.key}`);\n  }\n\n  // Skip validators that aren't UpdateValidators\n  if (!validator.updateHasErrors) return toConditionalPromise(undefined, async);\n\n  // skip async decorators if validateDecorators is called synchronously (async = false)\n  if (!async && decorator.props.async)\n    return toConditionalPromise(undefined, async);\n\n  const decoratorProps = Object.values(decorator.props) || {};\n\n  // const context = PathProxyEngine.create(obj, {\n  //   ignoreUndefined: true,\n  //   ignoreNull: true,\n  // });\n\n  const maybeError = validator.updateHasErrors(\n    (newModel as any)[prop],\n    (oldModel as any)[prop],\n    ...decoratorProps\n  );\n\n  return toConditionalPromise(maybeError, async);\n}\n\nexport function validateDecorators<\n  M extends Model,\n  Async extends boolean = false,\n>(\n  newModel: M,\n  oldModel: M,\n  prop: string,\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(\n      newModel,\n      oldModel,\n      prop,\n      decorator,\n      async\n    );\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 newPropValue = (newModel as any)[prop];\n      const oldPropValue = (oldModel as any)[prop];\n\n      const newValues =\n        newPropValue instanceof Set ? [...newPropValue] : newPropValue;\n      const oldValues =\n        oldPropValue instanceof Set ? [...oldPropValue] : oldPropValue;\n\n      if (newValues && newValues.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 errs = newValues.map((childValue: any) => {\n          // find by id so the list elements order doesn't matter\n          const id = findModelId(childValue as any, true);\n          if (!id) return \"Failed to find model id\";\n\n          const oldModel = oldValues.find(\n            (el: any) => id === findModelId(el, true)\n          );\n\n          if (Model.isModel(childValue)) {\n            return childValue.hasErrors(oldModel);\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 ? (result as any) : undefined;\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 * @description Validates changes between two model versions\n * @summary Compares an old and new model version to validate update operations\n * @template M - Type extending Model\n * @param {M} oldModel - The original model version\n * @param {M} newModel - The updated model version\n * @param {boolean} async - A flag indicating whether validation should be asynchronous.\n * @param {...string[]} exceptions - Properties to exclude from validation\n * @return {ModelErrorDefinition|undefined} Error definition if validation fails, undefined otherwise\n * @function validateCompare\n * @memberOf module:db-decorators\n * @mermaid\n * sequenceDiagram\n *   participant Caller\n *   participant validateCompare\n *   participant Reflection\n *   participant Validation\n *\n *   Caller->>validateCompare: oldModel, newModel, exceptions\n *   validateCompare->>Reflection: get decorated properties\n *   Reflection-->>validateCompare: property decorators\n *   loop For each decorated property\n *     validateCompare->>Validation: get validator\n *     Validation-->>validateCompare: validator\n *     validateCompare->>validateCompare: validate property update\n *   end\n *   loop For nested models\n *     validateCompare->>validateCompare: validate nested models\n *   end\n *   validateCompare-->>Caller: validation errors or undefined\n */\nexport function validateCompare<M extends Model<any>>(\n  oldModel: M,\n  newModel: M,\n  async: boolean,\n  ...exceptions: string[]\n): ModelConditionalAsync<M> {\n  const decoratedProperties: ValidationPropertyDecoratorDefinition[] =\n    getValidatableUpdateProps(newModel, exceptions);\n\n  const result: Record<string, any> = {};\n\n  const nestedErrors: Record<string, any> = {};\n  for (const { prop, decorators } of decoratedProperties) {\n    const propKey = String(prop);\n    let propValue = (newModel as any)[prop];\n\n    if (!decorators?.length) continue;\n\n    // Get the default type validator\n    const designTypeDec = decorators.find((d) =>\n      [ModelKeys.TYPE, ValidationKeys.TYPE].includes(d.key as any)\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      const { decorators } = Reflection.getPropertyDecorators(\n        ValidationKeys.REFLECT,\n        newModel,\n        propKey\n      ) as unknown as ValidationPropertyDecoratorDefinition;\n\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(newModel, oldModel, propKey, decorators, async) || {};\n\n    // Check for nested properties.\n    // To prevent unnecessary processing, \"propValue\" must be defined and validatable\n    const isConstr = Model.isPropertyModel(newModel, 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] =\n        //   \"Model should be validatable but it's not.\";\n        console.warn(\"Model should be validatable but it's not.\");\n      } else {\n        nestedErrors[propKey] = instance.hasErrors((oldModel as any)[prop]);\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"]}
@@ -5,18 +5,80 @@ import { IRepository } from "../interfaces";
5
5
  import { Context } from "../repository";
6
6
  import { RepositoryFlags } from "../repository/types";
7
7
  /**
8
- * @summary Static class holding common Operation Functionality
9
- *
8
+ * @description Static utility class for database operation management
9
+ * @summary Provides functionality for registering, retrieving, and managing database operation handlers
10
10
  * @class Operations
11
+ * @template M - Model type
12
+ * @template R - Repository type
13
+ * @template V - Metadata type
14
+ * @template F - Repository flags
15
+ * @template C - Context type
16
+ * @example
17
+ * // Register a handler for a create operation
18
+ * Operations.register(myHandler, OperationKeys.CREATE, targetModel, 'propertyName');
19
+ *
20
+ * // Get handlers for a specific operation
21
+ * const handlers = Operations.get(targetModel.constructor.name, 'propertyName', 'onCreate');
11
22
  *
12
- * @category Operations
23
+ * @mermaid
24
+ * classDiagram
25
+ * class Operations {
26
+ * -registry: OperationsRegistry
27
+ * +getHandlerName(handler)
28
+ * +key(str)
29
+ * +get(targetName, propKey, operation)
30
+ * -getOpRegistry()
31
+ * +register(handler, operation, target, propKey)
32
+ * }
33
+ * Operations --> OperationsRegistry : uses
13
34
  */
14
35
  export declare class Operations {
15
36
  private static registry;
16
37
  private constructor();
38
+ /**
39
+ * @description Gets a unique name for an operation handler
40
+ * @summary Returns the name of the handler function or generates a hash if name is not available
41
+ * @param {OperationHandler<any, any, any, any, any>} handler - The handler function to get the name for
42
+ * @return {string} The name of the handler or a generated hash
43
+ */
17
44
  static getHandlerName(handler: OperationHandler<any, any, any, any, any>): any;
45
+ /**
46
+ * @description Generates a reflection metadata key
47
+ * @summary Creates a fully qualified metadata key by prefixing with the reflection namespace
48
+ * @param {string} str - The operation key string to prefix
49
+ * @return {string} The fully qualified metadata key
50
+ */
18
51
  static key(str: string): string;
52
+ /**
53
+ * @description Retrieves operation handlers for a specific target and operation
54
+ * @summary Gets registered handlers from the operations registry for a given target, property, and operation
55
+ * @template M - Model type extending Model
56
+ * @template R - Repository type extending IRepository
57
+ * @template V - Metadata type, defaults to object
58
+ * @template F - Repository flags extending RepositoryFlags
59
+ * @template C - Context type extending Context<F>
60
+ * @param {string | Record<string, any>} targetName - The target class name or object
61
+ * @param {string} propKey - The property key to get handlers for
62
+ * @param {string} operation - The operation key to get handlers for
63
+ * @return {any} The registered handlers for the specified target, property, and operation
64
+ */
19
65
  static get<M extends Model, R extends IRepository<M, F, C>, V = object, F extends RepositoryFlags = RepositoryFlags, C extends Context<F> = Context<F>>(targetName: string | Record<string, any>, propKey: string, operation: string): OperationHandler<M, R, V, F, C>[] | undefined;
66
+ /**
67
+ * @description Gets or initializes the operations registry
68
+ * @summary Returns the existing registry or creates a new one if it doesn't exist
69
+ * @return {OperationsRegistry} The operations registry instance
70
+ * @private
71
+ */
20
72
  private static getOpRegistry;
73
+ /**
74
+ * @description Registers an operation handler for a specific target and operation
75
+ * @summary Adds a handler to the operations registry for a given target, property, and operation
76
+ * @template V - Model type extending Model
77
+ * @param {OperationHandler<V, any, any>} handler - The handler function to register
78
+ * @param {OperationKeys} operation - The operation key to register the handler for
79
+ * @param {V} target - The target model instance
80
+ * @param {string | symbol} propKey - The property key to register the handler for
81
+ * @return {void}
82
+ */
21
83
  static register<V extends Model>(handler: OperationHandler<V, any, any>, operation: OperationKeys, target: V, propKey: string | symbol): void;
22
84
  }