@decaf-ts/db-decorators 0.6.6 → 0.6.8

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,20 +1,28 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.ConflictError = exports.NotFoundError = exports.SerializationError = exports.InternalError = exports.ValidationError = exports.BaseError = void 0;
3
+ exports.ConflictError = exports.NotFoundError = exports.SerializationError = exports.InternalError = exports.ValidationError = exports.BadRequestError = exports.BaseError = void 0;
4
4
  /**
5
5
  * @description Base error class for the repository module
6
- * @summary Abstract base error class that all other error types extend from. Provides common error handling functionality.
6
+ * @summary Abstract base error class that all other error types extend from. Provides common error handling functionality and standardized HTTP code mapping.
7
7
  * @param {string} name - The name of the error
8
- * @param {string|Error} msg - The error message or Error object
8
+ * @param {string|Error} msg - The error message or Error object to wrap
9
9
  * @param {number} code - The HTTP status code associated with this error
10
+ * @return {void}
10
11
  * @class BaseError
11
12
  * @example
12
13
  * // This is an abstract class and should not be instantiated directly
13
14
  * // Instead, use one of the concrete error classes:
14
15
  * throw new ValidationError('Invalid data provided');
16
+ * @mermaid
17
+ * sequenceDiagram
18
+ * participant C as Caller
19
+ * participant E as BaseError
20
+ * C->>E: new BaseError(name,msg,code)
21
+ * E-->>C: Error instance with message and code
22
+ * @category Errors
15
23
  */
16
24
  class BaseError extends Error {
17
- constructor(name, msg, code = 500) {
25
+ constructor(name, msg, code) {
18
26
  if (msg instanceof BaseError)
19
27
  return msg;
20
28
  const message = `[${name}] ${msg instanceof Error ? msg.message : msg}`;
@@ -26,7 +34,21 @@ class BaseError extends Error {
26
34
  }
27
35
  exports.BaseError = BaseError;
28
36
  /**
29
- * @description Error thrown when validation fails
37
+ * @description Error (40) thrown when a bad request is received
38
+ * @summary Represents a failure in the input data, typically thrown when a client sends invalid or incomplete data
39
+ * @param {string|Error} msg - The error message or Error object
40
+ * @return {BadRequestError} A new BadRequestError instance
41
+ * @class BadRequestError
42
+ * @category Errors
43
+ */
44
+ class BadRequestError extends BaseError {
45
+ constructor(msg, name = BadRequestError.name, code = 400) {
46
+ super(name, msg, code);
47
+ }
48
+ }
49
+ exports.BadRequestError = BadRequestError;
50
+ /**
51
+ * @description Error (422) thrown when validation fails
30
52
  * @summary Represents a failure in the Model details, typically thrown when data validation fails
31
53
  * @param {string|Error} msg - The error message or Error object
32
54
  * @return {ValidationError} A new ValidationError instance
@@ -36,15 +58,16 @@ exports.BaseError = BaseError;
36
58
  * if (!isValid(data)) {
37
59
  * throw new ValidationError('Invalid data format');
38
60
  * }
61
+ * @category Errors
39
62
  */
40
- class ValidationError extends BaseError {
63
+ class ValidationError extends BadRequestError {
41
64
  constructor(msg) {
42
- super(ValidationError.name, msg, 422);
65
+ super(msg, ValidationError.name, 422);
43
66
  }
44
67
  }
45
68
  exports.ValidationError = ValidationError;
46
69
  /**
47
- * @description Error thrown for internal system failures
70
+ * @description Error (500) thrown for internal system failures
48
71
  * @summary Represents an internal failure (should mean an error in code) with HTTP 500 status code
49
72
  * @param {string|Error} msg - The error message or Error object
50
73
  * @return {InternalError} A new InternalError instance
@@ -56,15 +79,16 @@ exports.ValidationError = ValidationError;
56
79
  * } catch (error) {
57
80
  * throw new InternalError('Unexpected internal error occurred');
58
81
  * }
82
+ * @category Errors
59
83
  */
60
84
  class InternalError extends BaseError {
61
- constructor(msg) {
62
- super(InternalError.name, msg, 500);
85
+ constructor(msg, name = InternalError.name, code = 500) {
86
+ super(name, msg, code);
63
87
  }
64
88
  }
65
89
  exports.InternalError = InternalError;
66
90
  /**
67
- * @description Error thrown when serialization or deserialization fails
91
+ * @description Error (500) thrown when serialization or deserialization fails
68
92
  * @summary Represents a failure in the Model de/serialization, typically when converting between data formats
69
93
  * @param {string|Error} msg - The error message or Error object
70
94
  * @return {SerializationError} A new SerializationError instance
@@ -76,10 +100,11 @@ exports.InternalError = InternalError;
76
100
  * } catch (error) {
77
101
  * throw new SerializationError('Failed to parse JSON data');
78
102
  * }
103
+ * @category Errors
79
104
  */
80
- class SerializationError extends BaseError {
105
+ class SerializationError extends InternalError {
81
106
  constructor(msg) {
82
- super(SerializationError.name, msg, 422);
107
+ super(msg, SerializationError.name, 500);
83
108
  }
84
109
  }
85
110
  exports.SerializationError = SerializationError;
@@ -95,10 +120,11 @@ exports.SerializationError = SerializationError;
95
120
  * if (!user) {
96
121
  * throw new NotFoundError(`User with ID ${id} not found`);
97
122
  * }
123
+ * @category Errors
98
124
  */
99
- class NotFoundError extends BaseError {
125
+ class NotFoundError extends BadRequestError {
100
126
  constructor(msg) {
101
- super(NotFoundError.name, msg, 404);
127
+ super(msg, NotFoundError.name, 404);
102
128
  }
103
129
  }
104
130
  exports.NotFoundError = NotFoundError;
@@ -114,11 +140,12 @@ exports.NotFoundError = NotFoundError;
114
140
  * if (existingUser) {
115
141
  * throw new ConflictError(`User with email ${email} already exists`);
116
142
  * }
143
+ * @category Errors
117
144
  */
118
- class ConflictError extends BaseError {
145
+ class ConflictError extends BadRequestError {
119
146
  constructor(msg) {
120
- super(ConflictError.name, msg, 409);
147
+ super(msg, ConflictError.name, 409);
121
148
  }
122
149
  }
123
150
  exports.ConflictError = ConflictError;
124
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXJyb3JzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3JlcG9zaXRvcnkvZXJyb3JzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUFBOzs7Ozs7Ozs7OztHQVdHO0FBQ0gsTUFBc0IsU0FBVSxTQUFRLEtBQUs7SUFFM0MsWUFBc0IsSUFBWSxFQUFFLEdBQW1CLEVBQUUsT0FBZSxHQUFHO1FBQ3pFLElBQUksR0FBRyxZQUFZLFNBQVM7WUFBRSxPQUFPLEdBQUcsQ0FBQztRQUN6QyxNQUFNLE9BQU8sR0FBRyxJQUFJLElBQUksS0FBSyxHQUFHLFlBQVksS0FBSyxDQUFDLENBQUMsQ0FBQyxHQUFHLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQyxHQUFHLEVBQUUsQ0FBQztRQUN4RSxLQUFLLENBQUMsT0FBTyxDQUFDLENBQUM7UUFDZixJQUFJLENBQUMsSUFBSSxHQUFHLElBQUksQ0FBQztRQUNqQixJQUFJLEdBQUcsWUFBWSxLQUFLO1lBQUUsSUFBSSxDQUFDLEtBQUssR0FBRyxHQUFHLENBQUMsS0FBSyxDQUFDO0lBQ25ELENBQUM7Q0FDRjtBQVRELDhCQVNDO0FBRUQ7Ozs7Ozs7Ozs7O0dBV0c7QUFDSCxNQUFhLGVBQWdCLFNBQVEsU0FBUztJQUM1QyxZQUFZLEdBQW1CO1FBQzdCLEtBQUssQ0FBQyxlQUFlLENBQUMsSUFBSSxFQUFFLEdBQUcsRUFBRSxHQUFHLENBQUMsQ0FBQztJQUN4QyxDQUFDO0NBQ0Y7QUFKRCwwQ0FJQztBQUNEOzs7Ozs7Ozs7Ozs7O0dBYUc7QUFDSCxNQUFhLGFBQWMsU0FBUSxTQUFTO0lBQzFDLFlBQVksR0FBbUI7UUFDN0IsS0FBSyxDQUFDLGFBQWEsQ0FBQyxJQUFJLEVBQUUsR0FBRyxFQUFFLEdBQUcsQ0FBQyxDQUFDO0lBQ3RDLENBQUM7Q0FDRjtBQUpELHNDQUlDO0FBQ0Q7Ozs7Ozs7Ozs7Ozs7R0FhRztBQUNILE1BQWEsa0JBQW1CLFNBQVEsU0FBUztJQUMvQyxZQUFZLEdBQW1CO1FBQzdCLEtBQUssQ0FBQyxrQkFBa0IsQ0FBQyxJQUFJLEVBQUUsR0FBRyxFQUFFLEdBQUcsQ0FBQyxDQUFDO0lBQzNDLENBQUM7Q0FDRjtBQUpELGdEQUlDO0FBRUQ7Ozs7Ozs7Ozs7OztHQVlHO0FBQ0gsTUFBYSxhQUFjLFNBQVEsU0FBUztJQUMxQyxZQUFZLEdBQW1CO1FBQzdCLEtBQUssQ0FBQyxhQUFhLENBQUMsSUFBSSxFQUFFLEdBQUcsRUFBRSxHQUFHLENBQUMsQ0FBQztJQUN0QyxDQUFDO0NBQ0Y7QUFKRCxzQ0FJQztBQUNEOzs7Ozs7Ozs7Ozs7R0FZRztBQUNILE1BQWEsYUFBYyxTQUFRLFNBQVM7SUFDMUMsWUFBWSxHQUFtQjtRQUM3QixLQUFLLENBQUMsYUFBYSxDQUFDLElBQUksRUFBRSxHQUFHLEVBQUUsR0FBRyxDQUFDLENBQUM7SUFDdEMsQ0FBQztDQUNGO0FBSkQsc0NBSUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEBkZXNjcmlwdGlvbiBCYXNlIGVycm9yIGNsYXNzIGZvciB0aGUgcmVwb3NpdG9yeSBtb2R1bGVcbiAqIEBzdW1tYXJ5IEFic3RyYWN0IGJhc2UgZXJyb3IgY2xhc3MgdGhhdCBhbGwgb3RoZXIgZXJyb3IgdHlwZXMgZXh0ZW5kIGZyb20uIFByb3ZpZGVzIGNvbW1vbiBlcnJvciBoYW5kbGluZyBmdW5jdGlvbmFsaXR5LlxuICogQHBhcmFtIHtzdHJpbmd9IG5hbWUgLSBUaGUgbmFtZSBvZiB0aGUgZXJyb3JcbiAqIEBwYXJhbSB7c3RyaW5nfEVycm9yfSBtc2cgLSBUaGUgZXJyb3IgbWVzc2FnZSBvciBFcnJvciBvYmplY3RcbiAqIEBwYXJhbSB7bnVtYmVyfSBjb2RlIC0gVGhlIEhUVFAgc3RhdHVzIGNvZGUgYXNzb2NpYXRlZCB3aXRoIHRoaXMgZXJyb3JcbiAqIEBjbGFzcyBCYXNlRXJyb3JcbiAqIEBleGFtcGxlXG4gKiAvLyBUaGlzIGlzIGFuIGFic3RyYWN0IGNsYXNzIGFuZCBzaG91bGQgbm90IGJlIGluc3RhbnRpYXRlZCBkaXJlY3RseVxuICogLy8gSW5zdGVhZCwgdXNlIG9uZSBvZiB0aGUgY29uY3JldGUgZXJyb3IgY2xhc3NlczpcbiAqIHRocm93IG5ldyBWYWxpZGF0aW9uRXJyb3IoJ0ludmFsaWQgZGF0YSBwcm92aWRlZCcpO1xuICovXG5leHBvcnQgYWJzdHJhY3QgY2xhc3MgQmFzZUVycm9yIGV4dGVuZHMgRXJyb3Ige1xuICByZWFkb25seSBjb2RlITogbnVtYmVyO1xuICBwcm90ZWN0ZWQgY29uc3RydWN0b3IobmFtZTogc3RyaW5nLCBtc2c6IHN0cmluZyB8IEVycm9yLCBjb2RlOiBudW1iZXIgPSA1MDApIHtcbiAgICBpZiAobXNnIGluc3RhbmNlb2YgQmFzZUVycm9yKSByZXR1cm4gbXNnO1xuICAgIGNvbnN0IG1lc3NhZ2UgPSBgWyR7bmFtZX1dICR7bXNnIGluc3RhbmNlb2YgRXJyb3IgPyBtc2cubWVzc2FnZSA6IG1zZ31gO1xuICAgIHN1cGVyKG1lc3NhZ2UpO1xuICAgIHRoaXMuY29kZSA9IGNvZGU7XG4gICAgaWYgKG1zZyBpbnN0YW5jZW9mIEVycm9yKSB0aGlzLnN0YWNrID0gbXNnLnN0YWNrO1xuICB9XG59XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uIEVycm9yIHRocm93biB3aGVuIHZhbGlkYXRpb24gZmFpbHNcbiAqIEBzdW1tYXJ5IFJlcHJlc2VudHMgYSBmYWlsdXJlIGluIHRoZSBNb2RlbCBkZXRhaWxzLCB0eXBpY2FsbHkgdGhyb3duIHdoZW4gZGF0YSB2YWxpZGF0aW9uIGZhaWxzXG4gKiBAcGFyYW0ge3N0cmluZ3xFcnJvcn0gbXNnIC0gVGhlIGVycm9yIG1lc3NhZ2Ugb3IgRXJyb3Igb2JqZWN0XG4gKiBAcmV0dXJuIHtWYWxpZGF0aW9uRXJyb3J9IEEgbmV3IFZhbGlkYXRpb25FcnJvciBpbnN0YW5jZVxuICogQGNsYXNzIFZhbGlkYXRpb25FcnJvclxuICogQGV4YW1wbGVcbiAqIC8vIFRocm93IGEgdmFsaWRhdGlvbiBlcnJvciB3aGVuIGRhdGEgaXMgaW52YWxpZFxuICogaWYgKCFpc1ZhbGlkKGRhdGEpKSB7XG4gKiAgIHRocm93IG5ldyBWYWxpZGF0aW9uRXJyb3IoJ0ludmFsaWQgZGF0YSBmb3JtYXQnKTtcbiAqIH1cbiAqL1xuZXhwb3J0IGNsYXNzIFZhbGlkYXRpb25FcnJvciBleHRlbmRzIEJhc2VFcnJvciB7XG4gIGNvbnN0cnVjdG9yKG1zZzogc3RyaW5nIHwgRXJyb3IpIHtcbiAgICBzdXBlcihWYWxpZGF0aW9uRXJyb3IubmFtZSwgbXNnLCA0MjIpO1xuICB9XG59XG4vKipcbiAqIEBkZXNjcmlwdGlvbiBFcnJvciB0aHJvd24gZm9yIGludGVybmFsIHN5c3RlbSBmYWlsdXJlc1xuICogQHN1bW1hcnkgUmVwcmVzZW50cyBhbiBpbnRlcm5hbCBmYWlsdXJlIChzaG91bGQgbWVhbiBhbiBlcnJvciBpbiBjb2RlKSB3aXRoIEhUVFAgNTAwIHN0YXR1cyBjb2RlXG4gKiBAcGFyYW0ge3N0cmluZ3xFcnJvcn0gbXNnIC0gVGhlIGVycm9yIG1lc3NhZ2Ugb3IgRXJyb3Igb2JqZWN0XG4gKiBAcmV0dXJuIHtJbnRlcm5hbEVycm9yfSBBIG5ldyBJbnRlcm5hbEVycm9yIGluc3RhbmNlXG4gKiBAY2xhc3MgSW50ZXJuYWxFcnJvclxuICogQGV4YW1wbGVcbiAqIC8vIFRocm93IGFuIGludGVybmFsIGVycm9yIHdoZW4gYW4gdW5leHBlY3RlZCBjb25kaXRpb24gb2NjdXJzXG4gKiB0cnkge1xuICogICAvLyBTb21lIG9wZXJhdGlvblxuICogfSBjYXRjaCAoZXJyb3IpIHtcbiAqICAgdGhyb3cgbmV3IEludGVybmFsRXJyb3IoJ1VuZXhwZWN0ZWQgaW50ZXJuYWwgZXJyb3Igb2NjdXJyZWQnKTtcbiAqIH1cbiAqL1xuZXhwb3J0IGNsYXNzIEludGVybmFsRXJyb3IgZXh0ZW5kcyBCYXNlRXJyb3Ige1xuICBjb25zdHJ1Y3Rvcihtc2c6IHN0cmluZyB8IEVycm9yKSB7XG4gICAgc3VwZXIoSW50ZXJuYWxFcnJvci5uYW1lLCBtc2csIDUwMCk7XG4gIH1cbn1cbi8qKlxuICogQGRlc2NyaXB0aW9uIEVycm9yIHRocm93biB3aGVuIHNlcmlhbGl6YXRpb24gb3IgZGVzZXJpYWxpemF0aW9uIGZhaWxzXG4gKiBAc3VtbWFyeSBSZXByZXNlbnRzIGEgZmFpbHVyZSBpbiB0aGUgTW9kZWwgZGUvc2VyaWFsaXphdGlvbiwgdHlwaWNhbGx5IHdoZW4gY29udmVydGluZyBiZXR3ZWVuIGRhdGEgZm9ybWF0c1xuICogQHBhcmFtIHtzdHJpbmd8RXJyb3J9IG1zZyAtIFRoZSBlcnJvciBtZXNzYWdlIG9yIEVycm9yIG9iamVjdFxuICogQHJldHVybiB7U2VyaWFsaXphdGlvbkVycm9yfSBBIG5ldyBTZXJpYWxpemF0aW9uRXJyb3IgaW5zdGFuY2VcbiAqIEBjbGFzcyBTZXJpYWxpemF0aW9uRXJyb3JcbiAqIEBleGFtcGxlXG4gKiAvLyBUaHJvdyBhIHNlcmlhbGl6YXRpb24gZXJyb3Igd2hlbiBKU09OIHBhcnNpbmcgZmFpbHNcbiAqIHRyeSB7XG4gKiAgIGNvbnN0IGRhdGEgPSBKU09OLnBhcnNlKGludmFsaWRKc29uKTtcbiAqIH0gY2F0Y2ggKGVycm9yKSB7XG4gKiAgIHRocm93IG5ldyBTZXJpYWxpemF0aW9uRXJyb3IoJ0ZhaWxlZCB0byBwYXJzZSBKU09OIGRhdGEnKTtcbiAqIH1cbiAqL1xuZXhwb3J0IGNsYXNzIFNlcmlhbGl6YXRpb25FcnJvciBleHRlbmRzIEJhc2VFcnJvciB7XG4gIGNvbnN0cnVjdG9yKG1zZzogc3RyaW5nIHwgRXJyb3IpIHtcbiAgICBzdXBlcihTZXJpYWxpemF0aW9uRXJyb3IubmFtZSwgbXNnLCA0MjIpO1xuICB9XG59XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uIEVycm9yIHRocm93biB3aGVuIGEgcmVxdWVzdGVkIHJlc291cmNlIGlzIG5vdCBmb3VuZFxuICogQHN1bW1hcnkgUmVwcmVzZW50cyBhIGZhaWx1cmUgaW4gZmluZGluZyBhIG1vZGVsLCByZXN1bHRpbmcgaW4gYSA0MDQgSFRUUCBzdGF0dXMgY29kZVxuICogQHBhcmFtIHtzdHJpbmd8RXJyb3J9IG1zZyAtIFRoZSBlcnJvciBtZXNzYWdlIG9yIEVycm9yIG9iamVjdFxuICogQHJldHVybiB7Tm90Rm91bmRFcnJvcn0gQSBuZXcgTm90Rm91bmRFcnJvciBpbnN0YW5jZVxuICogQGNsYXNzIE5vdEZvdW5kRXJyb3JcbiAqIEBleGFtcGxlXG4gKiAvLyBUaHJvdyBhIG5vdCBmb3VuZCBlcnJvciB3aGVuIGEgcmVjb3JkIGRvZXNuJ3QgZXhpc3RcbiAqIGNvbnN0IHVzZXIgPSBhd2FpdCByZXBvc2l0b3J5LmZpbmRCeUlkKGlkKTtcbiAqIGlmICghdXNlcikge1xuICogICB0aHJvdyBuZXcgTm90Rm91bmRFcnJvcihgVXNlciB3aXRoIElEICR7aWR9IG5vdCBmb3VuZGApO1xuICogfVxuICovXG5leHBvcnQgY2xhc3MgTm90Rm91bmRFcnJvciBleHRlbmRzIEJhc2VFcnJvciB7XG4gIGNvbnN0cnVjdG9yKG1zZzogc3RyaW5nIHwgRXJyb3IpIHtcbiAgICBzdXBlcihOb3RGb3VuZEVycm9yLm5hbWUsIG1zZywgNDA0KTtcbiAgfVxufVxuLyoqXG4gKiBAZGVzY3JpcHRpb24gRXJyb3IgdGhyb3duIHdoZW4gYSBjb25mbGljdCBvY2N1cnMgaW4gdGhlIHN0b3JhZ2VcbiAqIEBzdW1tYXJ5IFJlcHJlc2VudHMgYSBjb25mbGljdCBpbiB0aGUgc3RvcmFnZSwgdHlwaWNhbGx5IHdoZW4gdHJ5aW5nIHRvIGNyZWF0ZSBhIGR1cGxpY2F0ZSByZXNvdXJjZVxuICogQHBhcmFtIHtzdHJpbmd8RXJyb3J9IG1zZyAtIFRoZSBlcnJvciBtZXNzYWdlIG9yIEVycm9yIG9iamVjdFxuICogQHJldHVybiB7Q29uZmxpY3RFcnJvcn0gQSBuZXcgQ29uZmxpY3RFcnJvciBpbnN0YW5jZVxuICogQGNsYXNzIENvbmZsaWN0RXJyb3JcbiAqIEBleGFtcGxlXG4gKiAvLyBUaHJvdyBhIGNvbmZsaWN0IGVycm9yIHdoZW4gdHJ5aW5nIHRvIGNyZWF0ZSBhIGR1cGxpY2F0ZSByZWNvcmRcbiAqIGNvbnN0IGV4aXN0aW5nVXNlciA9IGF3YWl0IHJlcG9zaXRvcnkuZmluZEJ5RW1haWwoZW1haWwpO1xuICogaWYgKGV4aXN0aW5nVXNlcikge1xuICogICB0aHJvdyBuZXcgQ29uZmxpY3RFcnJvcihgVXNlciB3aXRoIGVtYWlsICR7ZW1haWx9IGFscmVhZHkgZXhpc3RzYCk7XG4gKiB9XG4gKi9cbmV4cG9ydCBjbGFzcyBDb25mbGljdEVycm9yIGV4dGVuZHMgQmFzZUVycm9yIHtcbiAgY29uc3RydWN0b3IobXNnOiBzdHJpbmcgfCBFcnJvcikge1xuICAgIHN1cGVyKENvbmZsaWN0RXJyb3IubmFtZSwgbXNnLCA0MDkpO1xuICB9XG59XG4iXX0=
151
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"errors.js","sourceRoot":"","sources":["../../src/repository/errors.ts"],"names":[],"mappings":";;;AAAA;;;;;;;;;;;;;;;;;;;GAmBG;AACH,MAAsB,SAAU,SAAQ,KAAK;IAE3C,YAAsB,IAAY,EAAE,GAAmB,EAAE,IAAY;QACnE,IAAI,GAAG,YAAY,SAAS;YAAE,OAAO,GAAG,CAAC;QACzC,MAAM,OAAO,GAAG,IAAI,IAAI,KAAK,GAAG,YAAY,KAAK,CAAC,CAAC,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAC,GAAG,EAAE,CAAC;QACxE,KAAK,CAAC,OAAO,CAAC,CAAC;QACf,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC;QACjB,IAAI,GAAG,YAAY,KAAK;YAAE,IAAI,CAAC,KAAK,GAAG,GAAG,CAAC,KAAK,CAAC;IACnD,CAAC;CACF;AATD,8BASC;AAED;;;;;;;GAOG;AACH,MAAa,eAAgB,SAAQ,SAAS;IAC5C,YAAY,GAAmB,EAAE,IAAI,GAAG,eAAe,CAAC,IAAI,EAAE,IAAI,GAAG,GAAG;QACtE,KAAK,CAAC,IAAI,EAAE,GAAG,EAAE,IAAI,CAAC,CAAC;IACzB,CAAC;CACF;AAJD,0CAIC;AAED;;;;;;;;;;;;GAYG;AACH,MAAa,eAAgB,SAAQ,eAAe;IAClD,YAAY,GAAmB;QAC7B,KAAK,CAAC,GAAG,EAAE,eAAe,CAAC,IAAI,EAAE,GAAG,CAAC,CAAC;IACxC,CAAC;CACF;AAJD,0CAIC;AACD;;;;;;;;;;;;;;GAcG;AACH,MAAa,aAAc,SAAQ,SAAS;IAC1C,YAAY,GAAmB,EAAE,IAAI,GAAG,aAAa,CAAC,IAAI,EAAE,IAAI,GAAG,GAAG;QACpE,KAAK,CAAC,IAAI,EAAE,GAAG,EAAE,IAAI,CAAC,CAAC;IACzB,CAAC;CACF;AAJD,sCAIC;AACD;;;;;;;;;;;;;;GAcG;AACH,MAAa,kBAAmB,SAAQ,aAAa;IACnD,YAAY,GAAmB;QAC7B,KAAK,CAAC,GAAG,EAAE,kBAAkB,CAAC,IAAI,EAAE,GAAG,CAAC,CAAC;IAC3C,CAAC;CACF;AAJD,gDAIC;AAED;;;;;;;;;;;;;GAaG;AACH,MAAa,aAAc,SAAQ,eAAe;IAChD,YAAY,GAAmB;QAC7B,KAAK,CAAC,GAAG,EAAE,aAAa,CAAC,IAAI,EAAE,GAAG,CAAC,CAAC;IACtC,CAAC;CACF;AAJD,sCAIC;AACD;;;;;;;;;;;;;GAaG;AACH,MAAa,aAAc,SAAQ,eAAe;IAChD,YAAY,GAAmB;QAC7B,KAAK,CAAC,GAAG,EAAE,aAAa,CAAC,IAAI,EAAE,GAAG,CAAC,CAAC;IACtC,CAAC;CACF;AAJD,sCAIC","sourcesContent":["/**\n * @description Base error class for the repository module\n * @summary Abstract base error class that all other error types extend from. Provides common error handling functionality and standardized HTTP code mapping.\n * @param {string} name - The name of the error\n * @param {string|Error} msg - The error message or Error object to wrap\n * @param {number} code - The HTTP status code associated with this error\n * @return {void}\n * @class BaseError\n * @example\n * // This is an abstract class and should not be instantiated directly\n * // Instead, use one of the concrete error classes:\n * throw new ValidationError('Invalid data provided');\n * @mermaid\n * sequenceDiagram\n *   participant C as Caller\n *   participant E as BaseError\n *   C->>E: new BaseError(name,msg,code)\n *   E-->>C: Error instance with message and code\n * @category Errors\n */\nexport abstract class BaseError extends Error {\n  readonly code!: number;\n  protected constructor(name: string, msg: string | Error, code: number) {\n    if (msg instanceof BaseError) return msg;\n    const message = `[${name}] ${msg instanceof Error ? msg.message : msg}`;\n    super(message);\n    this.code = code;\n    if (msg instanceof Error) this.stack = msg.stack;\n  }\n}\n\n/**\n * @description Error (40) thrown when a bad request is received\n * @summary Represents a failure in the input data, typically thrown when a client sends invalid or incomplete data\n * @param {string|Error} msg - The error message or Error object\n * @return {BadRequestError} A new BadRequestError instance\n * @class BadRequestError\n * @category Errors\n */\nexport class BadRequestError extends BaseError {\n  constructor(msg: string | Error, name = BadRequestError.name, code = 400) {\n    super(name, msg, code);\n  }\n}\n\n/**\n * @description Error (422) thrown when validation fails\n * @summary Represents a failure in the Model details, typically thrown when data validation fails\n * @param {string|Error} msg - The error message or Error object\n * @return {ValidationError} A new ValidationError instance\n * @class ValidationError\n * @example\n * // Throw a validation error when data is invalid\n * if (!isValid(data)) {\n *   throw new ValidationError('Invalid data format');\n * }\n * @category Errors\n */\nexport class ValidationError extends BadRequestError {\n  constructor(msg: string | Error) {\n    super(msg, ValidationError.name, 422);\n  }\n}\n/**\n * @description Error (500) thrown for internal system failures\n * @summary Represents an internal failure (should mean an error in code) with HTTP 500 status code\n * @param {string|Error} msg - The error message or Error object\n * @return {InternalError} A new InternalError instance\n * @class InternalError\n * @example\n * // Throw an internal error when an unexpected condition occurs\n * try {\n *   // Some operation\n * } catch (error) {\n *   throw new InternalError('Unexpected internal error occurred');\n * }\n * @category Errors\n */\nexport class InternalError extends BaseError {\n  constructor(msg: string | Error, name = InternalError.name, code = 500) {\n    super(name, msg, code);\n  }\n}\n/**\n * @description Error (500) thrown when serialization or deserialization fails\n * @summary Represents a failure in the Model de/serialization, typically when converting between data formats\n * @param {string|Error} msg - The error message or Error object\n * @return {SerializationError} A new SerializationError instance\n * @class SerializationError\n * @example\n * // Throw a serialization error when JSON parsing fails\n * try {\n *   const data = JSON.parse(invalidJson);\n * } catch (error) {\n *   throw new SerializationError('Failed to parse JSON data');\n * }\n * @category Errors\n */\nexport class SerializationError extends InternalError {\n  constructor(msg: string | Error) {\n    super(msg, SerializationError.name, 500);\n  }\n}\n\n/**\n * @description Error thrown when a requested resource is not found\n * @summary Represents a failure in finding a model, resulting in a 404 HTTP status code\n * @param {string|Error} msg - The error message or Error object\n * @return {NotFoundError} A new NotFoundError instance\n * @class NotFoundError\n * @example\n * // Throw a not found error when a record doesn't exist\n * const user = await repository.findById(id);\n * if (!user) {\n *   throw new NotFoundError(`User with ID ${id} not found`);\n * }\n * @category Errors\n */\nexport class NotFoundError extends BadRequestError {\n  constructor(msg: string | Error) {\n    super(msg, NotFoundError.name, 404);\n  }\n}\n/**\n * @description Error thrown when a conflict occurs in the storage\n * @summary Represents a conflict in the storage, typically when trying to create a duplicate resource\n * @param {string|Error} msg - The error message or Error object\n * @return {ConflictError} A new ConflictError instance\n * @class ConflictError\n * @example\n * // Throw a conflict error when trying to create a duplicate record\n * const existingUser = await repository.findByEmail(email);\n * if (existingUser) {\n *   throw new ConflictError(`User with email ${email} already exists`);\n * }\n * @category Errors\n */\nexport class ConflictError extends BadRequestError {\n  constructor(msg: string | Error) {\n    super(msg, ConflictError.name, 409);\n  }\n}\n"]}
@@ -1,21 +1,40 @@
1
1
  /**
2
2
  * @description Base error class for the repository module
3
- * @summary Abstract base error class that all other error types extend from. Provides common error handling functionality.
3
+ * @summary Abstract base error class that all other error types extend from. Provides common error handling functionality and standardized HTTP code mapping.
4
4
  * @param {string} name - The name of the error
5
- * @param {string|Error} msg - The error message or Error object
5
+ * @param {string|Error} msg - The error message or Error object to wrap
6
6
  * @param {number} code - The HTTP status code associated with this error
7
+ * @return {void}
7
8
  * @class BaseError
8
9
  * @example
9
10
  * // This is an abstract class and should not be instantiated directly
10
11
  * // Instead, use one of the concrete error classes:
11
12
  * throw new ValidationError('Invalid data provided');
13
+ * @mermaid
14
+ * sequenceDiagram
15
+ * participant C as Caller
16
+ * participant E as BaseError
17
+ * C->>E: new BaseError(name,msg,code)
18
+ * E-->>C: Error instance with message and code
19
+ * @category Errors
12
20
  */
13
21
  export declare abstract class BaseError extends Error {
14
22
  readonly code: number;
15
- protected constructor(name: string, msg: string | Error, code?: number);
23
+ protected constructor(name: string, msg: string | Error, code: number);
24
+ }
25
+ /**
26
+ * @description Error (40) thrown when a bad request is received
27
+ * @summary Represents a failure in the input data, typically thrown when a client sends invalid or incomplete data
28
+ * @param {string|Error} msg - The error message or Error object
29
+ * @return {BadRequestError} A new BadRequestError instance
30
+ * @class BadRequestError
31
+ * @category Errors
32
+ */
33
+ export declare class BadRequestError extends BaseError {
34
+ constructor(msg: string | Error, name?: string, code?: number);
16
35
  }
17
36
  /**
18
- * @description Error thrown when validation fails
37
+ * @description Error (422) thrown when validation fails
19
38
  * @summary Represents a failure in the Model details, typically thrown when data validation fails
20
39
  * @param {string|Error} msg - The error message or Error object
21
40
  * @return {ValidationError} A new ValidationError instance
@@ -25,12 +44,13 @@ export declare abstract class BaseError extends Error {
25
44
  * if (!isValid(data)) {
26
45
  * throw new ValidationError('Invalid data format');
27
46
  * }
47
+ * @category Errors
28
48
  */
29
- export declare class ValidationError extends BaseError {
49
+ export declare class ValidationError extends BadRequestError {
30
50
  constructor(msg: string | Error);
31
51
  }
32
52
  /**
33
- * @description Error thrown for internal system failures
53
+ * @description Error (500) thrown for internal system failures
34
54
  * @summary Represents an internal failure (should mean an error in code) with HTTP 500 status code
35
55
  * @param {string|Error} msg - The error message or Error object
36
56
  * @return {InternalError} A new InternalError instance
@@ -42,12 +62,13 @@ export declare class ValidationError extends BaseError {
42
62
  * } catch (error) {
43
63
  * throw new InternalError('Unexpected internal error occurred');
44
64
  * }
65
+ * @category Errors
45
66
  */
46
67
  export declare class InternalError extends BaseError {
47
- constructor(msg: string | Error);
68
+ constructor(msg: string | Error, name?: string, code?: number);
48
69
  }
49
70
  /**
50
- * @description Error thrown when serialization or deserialization fails
71
+ * @description Error (500) thrown when serialization or deserialization fails
51
72
  * @summary Represents a failure in the Model de/serialization, typically when converting between data formats
52
73
  * @param {string|Error} msg - The error message or Error object
53
74
  * @return {SerializationError} A new SerializationError instance
@@ -59,8 +80,9 @@ export declare class InternalError extends BaseError {
59
80
  * } catch (error) {
60
81
  * throw new SerializationError('Failed to parse JSON data');
61
82
  * }
83
+ * @category Errors
62
84
  */
63
- export declare class SerializationError extends BaseError {
85
+ export declare class SerializationError extends InternalError {
64
86
  constructor(msg: string | Error);
65
87
  }
66
88
  /**
@@ -75,8 +97,9 @@ export declare class SerializationError extends BaseError {
75
97
  * if (!user) {
76
98
  * throw new NotFoundError(`User with ID ${id} not found`);
77
99
  * }
100
+ * @category Errors
78
101
  */
79
- export declare class NotFoundError extends BaseError {
102
+ export declare class NotFoundError extends BadRequestError {
80
103
  constructor(msg: string | Error);
81
104
  }
82
105
  /**
@@ -91,7 +114,8 @@ export declare class NotFoundError extends BaseError {
91
114
  * if (existingUser) {
92
115
  * throw new ConflictError(`User with email ${email} already exists`);
93
116
  * }
117
+ * @category Errors
94
118
  */
95
- export declare class ConflictError extends BaseError {
119
+ export declare class ConflictError extends BadRequestError {
96
120
  constructor(msg: string | Error);
97
121
  }
@@ -45,7 +45,6 @@ function readonly(message = constants_2.DEFAULT_ERROR_MESSAGES.READONLY.INVALID)
45
45
  * @param {M} model - The model instance being updated
46
46
  * @return {Promise<void>} A promise that resolves when the timestamp has been set
47
47
  * @function timestampHandler
48
- * @memberOf module:db-decorators
49
48
  */
50
49
  async function timestampHandler(context, data, key, model) {
51
50
  model[key] = context.timestamp;
@@ -138,7 +137,6 @@ function timestamp(operation = constants_3.DBOperations.CREATE_UPDATE, format =
138
137
  * @param {M} model - The model instance being processed
139
138
  * @return {Promise<void>} A promise that resolves when the property has been serialized
140
139
  * @function serializeOnCreateUpdate
141
- * @memberOf module:db-decorators
142
140
  */
143
141
  async function serializeOnCreateUpdate(context, data, key, model) {
144
142
  if (!model[key])
@@ -165,7 +163,6 @@ async function serializeOnCreateUpdate(context, data, key, model) {
165
163
  * @param {M} model - The model instance being processed
166
164
  * @return {Promise<void>} A promise that resolves when the property has been deserialized
167
165
  * @function serializeAfterAll
168
- * @memberOf module:db-decorators
169
166
  */
170
167
  async function serializeAfterAll(context, data, key, model) {
171
168
  if (!model[key])
@@ -209,4 +206,4 @@ async function serializeAfterAll(context, data, key, model) {
209
206
  function serialize() {
210
207
  return (0, reflection_1.apply)((0, decorators_1.onCreateUpdate)(serializeOnCreateUpdate), (0, decorators_1.after)(constants_3.DBOperations.ALL, serializeAfterAll), (0, decorator_validation_1.type)([String.name, Object.name]), (0, reflection_1.metadata)(repository_1.Repository.key(constants_1.DBKeys.SERIALIZE), {}));
211
208
  }
212
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"decorators.js","sourceRoot":"","sources":["../../src/validation/decorators.ts"],"names":[],"mappings":"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","sourcesContent":["import \"./validation\";\nimport {\n  date,\n  Decoration,\n  Model,\n  prop,\n  propMetadata,\n  required,\n  type,\n  Validation,\n} from \"@decaf-ts/decorator-validation\";\nimport { DBKeys, DEFAULT_TIMESTAMP_FORMAT } from \"../model/constants\";\nimport { DEFAULT_ERROR_MESSAGES } from \"./constants\";\nimport { DBOperations, OperationKeys } from \"../operations/constants\";\nimport { after, on, onCreateUpdate } from \"../operations/decorators\";\nimport { IRepository } from \"../interfaces/IRepository\";\nimport { SerializationError } from \"../repository/errors\";\nimport { apply, metadata } from \"@decaf-ts/reflection\";\nimport { Repository } from \"../repository\";\nimport { Context } from \"../repository/Context\";\nimport { RepositoryFlags } from \"../repository/types\";\n\n/**\n * @description Prevents a property from being modified after initial creation.\n * @summary Marks the property as readonly, causing validation errors if attempts are made to modify it during updates.\n * @param {string} [message] - The error message to display when validation fails. Defaults to {@link DEFAULT_ERROR_MESSAGES.READONLY.INVALID}\n * @return {PropertyDecorator} A decorator function that can be applied to class properties\n * @function readonly\n * @category Property Decorators\n */\nexport function readonly(\n  message: string = DEFAULT_ERROR_MESSAGES.READONLY.INVALID\n) {\n  const key = Validation.updateKey(DBKeys.READONLY);\n  return Decoration.for(key)\n    .define(\n      propMetadata(key, {\n        message: message,\n      })\n    )\n    .apply();\n}\n\n/**\n * @description Handler function that sets a timestamp property to the current timestamp.\n * @summary Updates a model property with the current timestamp from the repository context.\n * @template M - The model type extending Model\n * @template R - The repository type extending IRepository\n * @template V - The data type for the operation\n * @template F - The repository flags type\n * @template C - The context type\n * @param {C} context - The repository context containing the current timestamp\n * @param {V} data - The data being processed\n * @param key - The property key to update\n * @param {M} model - The model instance being updated\n * @return {Promise<void>} A promise that resolves when the timestamp has been set\n * @function timestampHandler\n * @memberOf module:db-decorators\n */\nexport async function timestampHandler<\n  M extends Model,\n  R extends IRepository<M, F, C>,\n  V,\n  F extends RepositoryFlags = RepositoryFlags,\n  C extends Context<F> = Context<F>,\n>(this: R, context: C, data: V, key: keyof M, model: M): Promise<void> {\n  (model as any)[key] = context.timestamp;\n}\n\n/**\n * @description Automatically manages timestamp properties for tracking creation and update times.\n * @summary Marks the property as a timestamp, making it required and ensuring it's a valid date. The property will be automatically updated with the current timestamp during specified operations.\n *\n * Date Format:\n *\n * <pre>\n *      Using similar formatting as Moment.js, Class DateTimeFormatter (Java), and Class SimpleDateFormat (Java),\n *      I implemented a comprehensive solution formatDate(date, patternStr) where the code is easy to read and modify.\n *      You can display date, time, AM/PM, etc.\n *\n *      Date and Time Patterns\n *      yy = 2-digit year; yyyy = full year\n *      M = digit month; MM = 2-digit month; MMM = short month name; MMMM = full month name\n *      EEEE = full weekday name; EEE = short weekday name\n *      d = digit day; dd = 2-digit day\n *      h = hours am/pm; hh = 2-digit hours am/pm; H = hours; HH = 2-digit hours\n *      m = minutes; mm = 2-digit minutes; aaa = AM/PM\n *      s = seconds; ss = 2-digit seconds\n *      S = miliseconds\n * </pre>\n *\n * @param {OperationKeys[]} operation - The operations to act on. Defaults to {@link DBOperations.CREATE_UPDATE}\n * @param {string} [format] - The timestamp format. Defaults to {@link DEFAULT_TIMESTAMP_FORMAT}\n * @return {PropertyDecorator} A decorator function that can be applied to class properties\n * @function timestamp\n * @category Property Decorators\n * @mermaid\n * sequenceDiagram\n *   participant C as Client\n *   participant M as Model\n *   participant T as TimestampDecorator\n *   participant V as Validator\n *\n *   C->>M: Create/Update model\n *   M->>T: Process timestamp property\n *   T->>M: Apply required validation\n *   T->>M: Apply date format validation\n *\n *   alt Update operation\n *     T->>V: Register timestamp validator\n *     V->>M: Validate timestamp is newer\n *   end\n *\n *   T->>M: Set current timestamp\n *   M->>C: Return updated model\n */\nexport function timestamp(\n  operation: OperationKeys[] = DBOperations.CREATE_UPDATE as unknown as OperationKeys[],\n  format: string = DEFAULT_TIMESTAMP_FORMAT\n) {\n  const key = Validation.updateKey(DBKeys.TIMESTAMP);\n\n  function ts(operation: OperationKeys[], format: string) {\n    const decorators: any[] = [\n      date(format, DEFAULT_ERROR_MESSAGES.TIMESTAMP.DATE),\n      required(DEFAULT_ERROR_MESSAGES.TIMESTAMP.REQUIRED),\n      propMetadata(Validation.key(DBKeys.TIMESTAMP), {\n        operation: operation,\n        format: format,\n      }),\n      on(operation, timestampHandler),\n    ];\n    if (operation.indexOf(OperationKeys.UPDATE) !== -1)\n      decorators.push(\n        propMetadata(key, {\n          message: DEFAULT_ERROR_MESSAGES.TIMESTAMP.INVALID,\n        })\n      );\n    else decorators.push(readonly());\n    return apply(...decorators);\n  }\n  return Decoration.for(key)\n    .define({\n      decorator: ts,\n      args: [operation, format],\n    })\n    .apply();\n}\n\n/**\n * @description Handler function that serializes a property to JSON string during create and update operations.\n * @summary Converts a complex object property to a JSON string before storing it in the database.\n * @template M - The model type extending Model\n * @template R - The repository type extending IRepository\n * @template V - The data type for the operation\n * @template F - The repository flags type\n * @template C - The context type\n * @param {C} context - The repository context\n * @param {V} data - The data being processed\n * @param key - The property key to serialize\n * @param {M} model - The model instance being processed\n * @return {Promise<void>} A promise that resolves when the property has been serialized\n * @function serializeOnCreateUpdate\n * @memberOf module:db-decorators\n */\nexport async function serializeOnCreateUpdate<\n  M extends Model,\n  R extends IRepository<M, F, C>,\n  V,\n  F extends RepositoryFlags = RepositoryFlags,\n  C extends Context<F> = Context<F>,\n>(this: R, context: C, data: V, key: keyof M, model: M): Promise<void> {\n  if (!model[key]) return;\n  try {\n    model[key] = JSON.stringify(model[key]) as M[keyof M];\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  } catch (e: unknown) {\n    throw new SerializationError(\n      `Failed to serialize ${key.toString()} property of model ${model.constructor.name}: e`\n    );\n  }\n}\n\n/**\n * @description Handler function that deserializes a property from JSON string after database operations.\n * @summary Converts a JSON string property back to its original complex object form after retrieving it from the database.\n * @template M - The model type extending Model\n * @template R - The repository type extending IRepository\n * @template V - The data type for the operation\n * @template F - The repository flags type\n * @template C - The context type\n * @param {C} context - The repository context\n * @param {V} data - The data being processed\n * @param key - The property key to deserialize\n * @param {M} model - The model instance being processed\n * @return {Promise<void>} A promise that resolves when the property has been deserialized\n * @function serializeAfterAll\n * @memberOf module:db-decorators\n */\nexport async function serializeAfterAll<\n  M extends Model,\n  R extends IRepository<M, F, C>,\n  V,\n  F extends RepositoryFlags = RepositoryFlags,\n  C extends Context<F> = Context<F>,\n>(this: R, context: C, data: V, key: keyof M, model: M): Promise<void> {\n  if (!model[key]) return;\n  if (typeof model[key] !== \"string\") return;\n\n  try {\n    model[key] = JSON.parse(model[key]);\n  } catch (e: unknown) {\n    throw new SerializationError(\n      `Failed to deserialize ${key.toString()} property of model ${model.constructor.name}: ${e}`\n    );\n  }\n}\n\n/**\n * @description Enables automatic JSON serialization and deserialization for complex object properties.\n * @summary Decorator that automatically converts complex objects to JSON strings before storing in the database and back to objects when retrieving them.\n * @return {PropertyDecorator} A decorator function that can be applied to class properties\n * @function serialize\n * @category Property Decorators\n * @mermaid\n * sequenceDiagram\n *   participant C as Client\n *   participant M as Model\n *   participant S as SerializeDecorator\n *   participant DB as Database\n *\n *   Note over C,DB: Create/Update Flow\n *   C->>M: Set complex object property\n *   M->>S: Process property (create/update)\n *   S->>M: Convert to JSON string\n *   M->>DB: Store serialized data\n *\n *   Note over C,DB: Retrieval Flow\n *   C->>M: Request model\n *   M->>DB: Fetch data\n *   DB->>M: Return with serialized property\n *   M->>S: Process property (after all ops)\n *   S->>M: Parse JSON back to object\n *   M->>C: Return model with deserialized property\n */\nexport function serialize() {\n  return apply(\n    onCreateUpdate(serializeOnCreateUpdate),\n    after(DBOperations.ALL, serializeAfterAll),\n    type([String.name, Object.name]),\n    metadata(Repository.key(DBKeys.SERIALIZE), {})\n  );\n}\n"]}
209
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"decorators.js","sourceRoot":"","sources":["../../src/validation/decorators.ts"],"names":[],"mappings":"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","sourcesContent":["import \"./validation\";\nimport {\n  date,\n  Decoration,\n  Model,\n  propMetadata,\n  required,\n  type,\n  Validation,\n} from \"@decaf-ts/decorator-validation\";\nimport { DBKeys, DEFAULT_TIMESTAMP_FORMAT } from \"../model/constants\";\nimport { DEFAULT_ERROR_MESSAGES } from \"./constants\";\nimport { DBOperations, OperationKeys } from \"../operations/constants\";\nimport { after, on, onCreateUpdate } from \"../operations/decorators\";\nimport { IRepository } from \"../interfaces/IRepository\";\nimport { SerializationError } from \"../repository/errors\";\nimport { apply, metadata } from \"@decaf-ts/reflection\";\nimport { Repository } from \"../repository\";\nimport { Context } from \"../repository/Context\";\nimport { RepositoryFlags } from \"../repository/types\";\n\n/**\n * @description Prevents a property from being modified after initial creation.\n * @summary Marks the property as readonly, causing validation errors if attempts are made to modify it during updates.\n * @param {string} [message] - The error message to display when validation fails. Defaults to {@link DEFAULT_ERROR_MESSAGES.READONLY.INVALID}\n * @return {PropertyDecorator} A decorator function that can be applied to class properties\n * @function readonly\n * @category Property Decorators\n */\nexport function readonly(\n  message: string = DEFAULT_ERROR_MESSAGES.READONLY.INVALID\n) {\n  const key = Validation.updateKey(DBKeys.READONLY);\n  return Decoration.for(key)\n    .define(\n      propMetadata(key, {\n        message: message,\n      })\n    )\n    .apply();\n}\n\n/**\n * @description Handler function that sets a timestamp property to the current timestamp.\n * @summary Updates a model property with the current timestamp from the repository context.\n * @template M - The model type extending Model\n * @template R - The repository type extending IRepository\n * @template V - The data type for the operation\n * @template F - The repository flags type\n * @template C - The context type\n * @param {C} context - The repository context containing the current timestamp\n * @param {V} data - The data being processed\n * @param key - The property key to update\n * @param {M} model - The model instance being updated\n * @return {Promise<void>} A promise that resolves when the timestamp has been set\n * @function timestampHandler\n */\nexport async function timestampHandler<\n  M extends Model,\n  R extends IRepository<M, F, C>,\n  V,\n  F extends RepositoryFlags = RepositoryFlags,\n  C extends Context<F> = Context<F>,\n>(this: R, context: C, data: V, key: keyof M, model: M): Promise<void> {\n  (model as any)[key] = context.timestamp;\n}\n\n/**\n * @description Automatically manages timestamp properties for tracking creation and update times.\n * @summary Marks the property as a timestamp, making it required and ensuring it's a valid date. The property will be automatically updated with the current timestamp during specified operations.\n *\n * Date Format:\n *\n * <pre>\n *      Using similar formatting as Moment.js, Class DateTimeFormatter (Java), and Class SimpleDateFormat (Java),\n *      I implemented a comprehensive solution formatDate(date, patternStr) where the code is easy to read and modify.\n *      You can display date, time, AM/PM, etc.\n *\n *      Date and Time Patterns\n *      yy = 2-digit year; yyyy = full year\n *      M = digit month; MM = 2-digit month; MMM = short month name; MMMM = full month name\n *      EEEE = full weekday name; EEE = short weekday name\n *      d = digit day; dd = 2-digit day\n *      h = hours am/pm; hh = 2-digit hours am/pm; H = hours; HH = 2-digit hours\n *      m = minutes; mm = 2-digit minutes; aaa = AM/PM\n *      s = seconds; ss = 2-digit seconds\n *      S = miliseconds\n * </pre>\n *\n * @param {OperationKeys[]} operation - The operations to act on. Defaults to {@link DBOperations.CREATE_UPDATE}\n * @param {string} [format] - The timestamp format. Defaults to {@link DEFAULT_TIMESTAMP_FORMAT}\n * @return {PropertyDecorator} A decorator function that can be applied to class properties\n * @function timestamp\n * @category Property Decorators\n * @mermaid\n * sequenceDiagram\n *   participant C as Client\n *   participant M as Model\n *   participant T as TimestampDecorator\n *   participant V as Validator\n *\n *   C->>M: Create/Update model\n *   M->>T: Process timestamp property\n *   T->>M: Apply required validation\n *   T->>M: Apply date format validation\n *\n *   alt Update operation\n *     T->>V: Register timestamp validator\n *     V->>M: Validate timestamp is newer\n *   end\n *\n *   T->>M: Set current timestamp\n *   M->>C: Return updated model\n */\nexport function timestamp(\n  operation: OperationKeys[] = DBOperations.CREATE_UPDATE as unknown as OperationKeys[],\n  format: string = DEFAULT_TIMESTAMP_FORMAT\n) {\n  const key = Validation.updateKey(DBKeys.TIMESTAMP);\n\n  function ts(operation: OperationKeys[], format: string) {\n    const decorators: any[] = [\n      date(format, DEFAULT_ERROR_MESSAGES.TIMESTAMP.DATE),\n      required(DEFAULT_ERROR_MESSAGES.TIMESTAMP.REQUIRED),\n      propMetadata(Validation.key(DBKeys.TIMESTAMP), {\n        operation: operation,\n        format: format,\n      }),\n      on(operation, timestampHandler),\n    ];\n    if (operation.indexOf(OperationKeys.UPDATE) !== -1)\n      decorators.push(\n        propMetadata(key, {\n          message: DEFAULT_ERROR_MESSAGES.TIMESTAMP.INVALID,\n        })\n      );\n    else decorators.push(readonly());\n    return apply(...decorators);\n  }\n  return Decoration.for(key)\n    .define({\n      decorator: ts,\n      args: [operation, format],\n    })\n    .apply();\n}\n\n/**\n * @description Handler function that serializes a property to JSON string during create and update operations.\n * @summary Converts a complex object property to a JSON string before storing it in the database.\n * @template M - The model type extending Model\n * @template R - The repository type extending IRepository\n * @template V - The data type for the operation\n * @template F - The repository flags type\n * @template C - The context type\n * @param {C} context - The repository context\n * @param {V} data - The data being processed\n * @param key - The property key to serialize\n * @param {M} model - The model instance being processed\n * @return {Promise<void>} A promise that resolves when the property has been serialized\n * @function serializeOnCreateUpdate\n */\nexport async function serializeOnCreateUpdate<\n  M extends Model,\n  R extends IRepository<M, F, C>,\n  V,\n  F extends RepositoryFlags = RepositoryFlags,\n  C extends Context<F> = Context<F>,\n>(this: R, context: C, data: V, key: keyof M, model: M): Promise<void> {\n  if (!model[key]) return;\n  try {\n    model[key] = JSON.stringify(model[key]) as M[keyof M];\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n  } catch (e: unknown) {\n    throw new SerializationError(\n      `Failed to serialize ${key.toString()} property of model ${model.constructor.name}: e`\n    );\n  }\n}\n\n/**\n * @description Handler function that deserializes a property from JSON string after database operations.\n * @summary Converts a JSON string property back to its original complex object form after retrieving it from the database.\n * @template M - The model type extending Model\n * @template R - The repository type extending IRepository\n * @template V - The data type for the operation\n * @template F - The repository flags type\n * @template C - The context type\n * @param {C} context - The repository context\n * @param {V} data - The data being processed\n * @param key - The property key to deserialize\n * @param {M} model - The model instance being processed\n * @return {Promise<void>} A promise that resolves when the property has been deserialized\n * @function serializeAfterAll\n */\nexport async function serializeAfterAll<\n  M extends Model,\n  R extends IRepository<M, F, C>,\n  V,\n  F extends RepositoryFlags = RepositoryFlags,\n  C extends Context<F> = Context<F>,\n>(this: R, context: C, data: V, key: keyof M, model: M): Promise<void> {\n  if (!model[key]) return;\n  if (typeof model[key] !== \"string\") return;\n\n  try {\n    model[key] = JSON.parse(model[key]);\n  } catch (e: unknown) {\n    throw new SerializationError(\n      `Failed to deserialize ${key.toString()} property of model ${model.constructor.name}: ${e}`\n    );\n  }\n}\n\n/**\n * @description Enables automatic JSON serialization and deserialization for complex object properties.\n * @summary Decorator that automatically converts complex objects to JSON strings before storing in the database and back to objects when retrieving them.\n * @return {PropertyDecorator} A decorator function that can be applied to class properties\n * @function serialize\n * @category Property Decorators\n * @mermaid\n * sequenceDiagram\n *   participant C as Client\n *   participant M as Model\n *   participant S as SerializeDecorator\n *   participant DB as Database\n *\n *   Note over C,DB: Create/Update Flow\n *   C->>M: Set complex object property\n *   M->>S: Process property (create/update)\n *   S->>M: Convert to JSON string\n *   M->>DB: Store serialized data\n *\n *   Note over C,DB: Retrieval Flow\n *   C->>M: Request model\n *   M->>DB: Fetch data\n *   DB->>M: Return with serialized property\n *   M->>S: Process property (after all ops)\n *   S->>M: Parse JSON back to object\n *   M->>C: Return model with deserialized property\n */\nexport function serialize() {\n  return apply(\n    onCreateUpdate(serializeOnCreateUpdate),\n    after(DBOperations.ALL, serializeAfterAll),\n    type([String.name, Object.name]),\n    metadata(Repository.key(DBKeys.SERIALIZE), {})\n  );\n}\n"]}
@@ -27,7 +27,6 @@ export declare function readonly(message?: string): (target: any, propertyKey?:
27
27
  * @param {M} model - The model instance being updated
28
28
  * @return {Promise<void>} A promise that resolves when the timestamp has been set
29
29
  * @function timestampHandler
30
- * @memberOf module:db-decorators
31
30
  */
32
31
  export declare function timestampHandler<M extends Model, R extends IRepository<M, F, C>, V, F extends RepositoryFlags = RepositoryFlags, C extends Context<F> = Context<F>>(this: R, context: C, data: V, key: keyof M, model: M): Promise<void>;
33
32
  /**
@@ -92,7 +91,6 @@ export declare function timestamp(operation?: OperationKeys[], format?: string):
92
91
  * @param {M} model - The model instance being processed
93
92
  * @return {Promise<void>} A promise that resolves when the property has been serialized
94
93
  * @function serializeOnCreateUpdate
95
- * @memberOf module:db-decorators
96
94
  */
97
95
  export declare function serializeOnCreateUpdate<M extends Model, R extends IRepository<M, F, C>, V, F extends RepositoryFlags = RepositoryFlags, C extends Context<F> = Context<F>>(this: R, context: C, data: V, key: keyof M, model: M): Promise<void>;
98
96
  /**
@@ -109,7 +107,6 @@ export declare function serializeOnCreateUpdate<M extends Model, R extends IRepo
109
107
  * @param {M} model - The model instance being processed
110
108
  * @return {Promise<void>} A promise that resolves when the property has been deserialized
111
109
  * @function serializeAfterAll
112
- * @memberOf module:db-decorators
113
110
  */
114
111
  export declare function serializeAfterAll<M extends Model, R extends IRepository<M, F, C>, V, F extends RepositoryFlags = RepositoryFlags, C extends Context<F> = Context<F>>(this: R, context: C, data: V, key: keyof M, model: M): Promise<void>;
115
112
  /**
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@decaf-ts/db-decorators",
3
- "version": "0.6.6",
3
+ "version": "0.6.8",
4
4
  "description": "Agnostic database decorators and repository",
5
5
  "type": "module",
6
6
  "exports": {
@@ -26,6 +26,7 @@
26
26
  "coverage": "rimraf ./workdocs/reports/data/*.json && npm run test:all -- --coverage --config=./workdocs/reports/jest.coverage.config.ts",
27
27
  "lint": "eslint .",
28
28
  "lint-fix": "eslint --fix .",
29
+ "prepare-pr": "npm run lint-fix && npm run build:prod && npm run coverage && npm run docs",
29
30
  "prepare-release": "npm run lint-fix && npm run build:prod && npm run coverage && npm run docs",
30
31
  "release": "./bin/tag-release.sh",
31
32
  "clean-publish": "npx clean-publish",