@decaf-ts/transactional-decorators 0.1.1 → 0.1.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 (44) hide show
  1. package/LICENSE.md +21 -157
  2. package/dist/transactional-decorators.cjs +166 -67
  3. package/dist/transactional-decorators.esm.cjs +166 -67
  4. package/lib/Transaction.cjs +87 -39
  5. package/lib/Transaction.d.ts +86 -38
  6. package/lib/constants.cjs +14 -1
  7. package/lib/constants.d.ts +13 -0
  8. package/lib/decorators.cjs +35 -11
  9. package/lib/decorators.d.ts +34 -10
  10. package/lib/esm/Transaction.d.ts +86 -38
  11. package/lib/esm/Transaction.js +90 -42
  12. package/lib/esm/constants.d.ts +13 -0
  13. package/lib/esm/constants.js +14 -1
  14. package/lib/esm/decorators.d.ts +34 -10
  15. package/lib/esm/decorators.js +37 -13
  16. package/lib/esm/index.d.ts +7 -13
  17. package/lib/esm/index.js +14 -20
  18. package/lib/esm/interfaces/TransactionLock.d.ts +13 -11
  19. package/lib/esm/interfaces/TransactionLock.js +1 -1
  20. package/lib/esm/interfaces/index.d.ts +5 -0
  21. package/lib/esm/interfaces/index.js +7 -2
  22. package/lib/esm/locks/Lock.d.ts +13 -5
  23. package/lib/esm/locks/Lock.js +14 -6
  24. package/lib/esm/locks/SyncronousLock.d.ts +3 -4
  25. package/lib/esm/locks/SyncronousLock.js +14 -2
  26. package/lib/esm/locks/index.d.ts +5 -0
  27. package/lib/esm/locks/index.js +8 -3
  28. package/lib/esm/types.d.ts +13 -3
  29. package/lib/esm/types.js +1 -1
  30. package/lib/index.cjs +8 -14
  31. package/lib/index.d.ts +7 -13
  32. package/lib/interfaces/TransactionLock.cjs +1 -1
  33. package/lib/interfaces/TransactionLock.d.ts +13 -11
  34. package/lib/interfaces/index.cjs +6 -1
  35. package/lib/interfaces/index.d.ts +5 -0
  36. package/lib/locks/Lock.cjs +14 -6
  37. package/lib/locks/Lock.d.ts +13 -5
  38. package/lib/locks/SyncronousLock.cjs +13 -1
  39. package/lib/locks/SyncronousLock.d.ts +3 -4
  40. package/lib/locks/index.cjs +6 -1
  41. package/lib/locks/index.d.ts +5 -0
  42. package/lib/types.cjs +1 -1
  43. package/lib/types.d.ts +13 -3
  44. package/package.json +2 -2
@@ -1,19 +1,43 @@
1
1
  /**
2
- * @summary Sets a class Async method as transactional
2
+ * @description Method decorator that enables transactional behavior
3
+ * @summary Sets a class async method as transactional, wrapping it in a transaction context that can be managed by the transaction system. This decorator handles transaction creation, binding, and error handling.
4
+ * @param {any[]} [data] - Optional metadata available to the {@link TransactionLock} implementation
5
+ * @return {Function} A decorator function that wraps the original method with transactional behavior
6
+ * @function transactional
7
+ * @category Method Decorators
8
+ * @mermaid
9
+ * sequenceDiagram
10
+ * participant C as Client Code
11
+ * participant D as Decorator
12
+ * participant T as Transaction
13
+ * participant O as Original Method
3
14
  *
4
- * @param {any[]} [data] option metadata available to the {@link TransactionLock}
15
+ * C->>D: Call decorated method
16
+ * D->>D: Check if transaction exists in args
5
17
  *
6
- * @function transactional
18
+ * alt Transaction exists in args
19
+ * D->>T: Create updated transaction
20
+ * T->>T: Bind to original transaction
21
+ * T->>T: Fire transaction
22
+ * else No transaction
23
+ * D->>T: Create new transaction
24
+ * T->>T: Submit transaction
25
+ * end
7
26
  *
8
- * @memberOf module:db-decorators.Decorators.transactions
27
+ * T->>O: Execute original method
28
+ * O-->>T: Return result/error
29
+ * T->>T: Release transaction
30
+ * T-->>C: Return result/error
31
+ * @category Decorators
9
32
  */
10
33
  export declare function transactional(...data: any[]): (target: any, propertyKey?: any, descriptor?: PropertyDescriptor) => void;
11
34
  /**
12
- * @summary Util function to wrap super calls with the transaction when the super's method is also transactional
13
- *
14
- * @param {Function} method the super method (must be bound to the proper this), eg: super.create.bind(this)
15
- * @param {any[]} args the arguments to call the method with
16
- *
17
- * @memberOf module:db-decorators.Transaction
35
+ * @description Utility for handling super calls in transactional methods
36
+ * @summary Wraps super method calls with the current transaction context when the super's method is also transactional, ensuring transaction continuity through the inheritance chain
37
+ * @param {Function} method - The super method (must be bound to the proper this), e.g., super.create.bind(this)
38
+ * @param {any[]} args - The arguments to call the method with
39
+ * @return {any} The result of the super method call
40
+ * @function transactionalSuperCall
41
+ * @memberOf module:transactions
18
42
  */
19
43
  export declare function transactionalSuperCall(method: any, ...args: any): any;
@@ -1,16 +1,48 @@
1
1
  import { TransactionLock } from "./interfaces/TransactionLock";
2
2
  import { Callback } from "./types";
3
3
  /**
4
- * @summary Transaction Class
5
- *
6
- * @param {string} source
7
- * @param {string} [method]
8
- * @param {function(): void} [action]
9
- * @param {any[]} [metadata]
10
- *
4
+ * @description Core transaction management class
5
+ * @summary Manages transaction lifecycle, including creation, execution, and cleanup. Provides mechanisms for binding transactions to objects and methods, ensuring proper transaction context propagation.
6
+ * @param {string} source - The source/origin of the transaction (typically a class name)
7
+ * @param {string} [method] - The method name associated with the transaction
8
+ * @param {function(): any} [action] - The function to execute within the transaction
9
+ * @param {any[]} [metadata] - Additional metadata to associate with the transaction
11
10
  * @class Transaction
11
+ * @example
12
+ * // Creating and submitting a transaction
13
+ * const transaction = new Transaction(
14
+ * 'UserService',
15
+ * 'createUser',
16
+ * async () => {
17
+ * // Transaction logic here
18
+ * await db.insert('users', { name: 'John' });
19
+ * }
20
+ * );
21
+ * Transaction.submit(transaction);
12
22
  *
13
- * @category Transactions
23
+ * // Using the transactional decorator
24
+ * class UserService {
25
+ * @transactional()
26
+ * async createUser(data) {
27
+ * // Method will be executed within a transaction
28
+ * return await db.insert('users', data);
29
+ * }
30
+ * }
31
+ * @mermaid
32
+ * sequenceDiagram
33
+ * participant C as Client Code
34
+ * participant T as Transaction
35
+ * participant L as TransactionLock
36
+ * participant O as Original Method
37
+ *
38
+ * C->>T: new Transaction(source, method, action)
39
+ * C->>T: Transaction.submit(transaction)
40
+ * T->>L: submit(transaction)
41
+ * L->>T: fire()
42
+ * T->>O: Execute action()
43
+ * O-->>T: Return result/error
44
+ * T->>L: release(error?)
45
+ * L-->>C: Return result/error
14
46
  */
15
47
  export declare class Transaction {
16
48
  readonly id: number;
@@ -22,65 +54,81 @@ export declare class Transaction {
22
54
  private static lock;
23
55
  constructor(source: string, method?: string, action?: () => any, metadata?: any[]);
24
56
  /**
25
- * @summary Pushes a transaction to que queue and waits its resolution
26
- *
27
- * @param {any} issuer any class. will be used as this when calling the callbackMethod
28
- * @param {Function} callbackMethod callback function containing the transaction. will be called with the issuear as this
29
- * @param {any[]} args arguments to pass to the method. Last one must be the callback
57
+ * @description Queues a transaction for execution
58
+ * @summary Pushes a transaction to the queue and waits for its resolution. Creates a new transaction with the provided issuer and callback method, then submits it to the transaction lock.
59
+ * @param {any} issuer - Any class instance that will be used as 'this' when calling the callbackMethod
60
+ * @param {Function} callbackMethod - Callback function containing the transaction logic, will be called with the issuer as 'this'
61
+ * @param {any[]} args - Arguments to pass to the method. Last one must be the callback function
62
+ * @return {void}
30
63
  */
31
64
  static push(issuer: any, callbackMethod: (...argzz: (any | Callback)[]) => void, ...args: (any | Callback)[]): void;
32
65
  /**
33
- * @summary Sets the lock to be used
34
- * @param lock
66
+ * @description Configures the transaction lock implementation
67
+ * @summary Sets the lock implementation to be used for transaction management, allowing customization of the transaction behavior
68
+ * @param {TransactionLock} lock - The lock implementation to use for managing transactions
69
+ * @return {void}
35
70
  */
36
71
  static setLock(lock: TransactionLock): void;
37
72
  /**
38
- * @summary gets the lock
73
+ * @description Retrieves the current transaction lock
74
+ * @summary Gets the current transaction lock instance, creating a default SyncronousLock if none exists
75
+ * @return {TransactionLock} The current transaction lock implementation
39
76
  */
40
77
  static getLock(): TransactionLock;
41
78
  /**
42
- * @summary submits a transaction
43
- * @param {Transaction} transaction
79
+ * @description Submits a transaction for processing
80
+ * @summary Submits a transaction to the current transaction lock for processing and execution
81
+ * @param {Transaction} transaction - The transaction to submit for processing
82
+ * @return {void}
44
83
  */
45
84
  static submit(transaction: Transaction): void;
46
85
  /**
47
- * @summary releases the lock
48
- * @param {Err} err
86
+ * @description Releases the transaction lock
87
+ * @summary Releases the current transaction lock, optionally with an error, allowing the next transaction to proceed
88
+ * @param {Error} [err] - Optional error that occurred during transaction execution
89
+ * @return {Promise<void>} A promise that resolves when the lock has been released
49
90
  */
50
91
  static release(err?: Error): Promise<void>;
51
92
  /**
52
- * @summary retrieves the metadata for the transaction
93
+ * @description Retrieves transaction metadata
94
+ * @summary Returns a copy of the metadata associated with this transaction, ensuring the original metadata remains unmodified
95
+ * @return {any[] | undefined} A copy of the transaction metadata or undefined if no metadata exists
53
96
  */
54
97
  getMetadata(): any[] | undefined;
55
98
  /**
56
- * @summary Binds a new operation to the current transaction
57
- * @param {Transaction} nextTransaction
99
+ * @description Links a new transaction to the current one
100
+ * @summary Binds a new transaction operation to the current transaction, transferring logs and binding methods to maintain transaction context
101
+ * @param {Transaction} nextTransaction - The new transaction to bind to the current one
102
+ * @return {void}
58
103
  */
59
104
  bindTransaction(nextTransaction: Transaction): void;
60
105
  /**
61
- * @summary Binds the Transactional Decorated Object to the transaction
62
- * @description by having all {@link transactional} decorated
63
- * methods always pass the current Transaction as an argument
64
- *
65
- * @param {any} obj
66
- * @return {any} the bound {@param obj}
106
+ * @description Binds an object to the current transaction context
107
+ * @summary Binds a transactional decorated object to the transaction by ensuring all transactional methods automatically receive the current transaction as their first argument
108
+ * @param {any} obj - The object to bind to the transaction
109
+ * @return {any} The bound object with transaction-aware method wrappers
67
110
  */
68
111
  bindToTransaction(obj: any): any;
69
112
  /**
70
- * @summary Fires the Transaction
113
+ * @description Executes the transaction action
114
+ * @summary Fires the transaction by executing its associated action function, throwing an error if no action is defined
115
+ * @return {any} The result of the transaction action
71
116
  */
72
117
  fire(): any;
73
118
  /**
74
- * @summary toString override
75
- * @param {boolean} [withId] defaults to true
76
- * @param {boolean} [withLog] defaults to true
119
+ * @description Provides a string representation of the transaction
120
+ * @summary Overrides the default toString method to provide a formatted string representation of the transaction, optionally including the transaction ID and log
121
+ * @param {boolean} [withId=true] - Whether to include the transaction ID in the output
122
+ * @param {boolean} [withLog=false] - Whether to include the transaction log in the output
123
+ * @return {string} A string representation of the transaction
77
124
  */
78
125
  toString(withId?: boolean, withLog?: boolean): string;
79
126
  /**
80
- * @summary gets the transactions reflections key
81
- * @function getRepoKey
82
- * @param {string} key
83
- * @memberOf module:db-decorators.Transactions
84
- * */
127
+ * @description Generates a reflection metadata key for transactions
128
+ * @summary Creates a prefixed reflection key for transaction-related metadata, ensuring proper namespacing
129
+ * @param {string} key - The base key to prefix with the transaction reflection namespace
130
+ * @return {string} The complete reflection key for transaction metadata
131
+ * @function key
132
+ */
85
133
  static key(key: string): string;
86
134
  }
@@ -1,19 +1,51 @@
1
1
  import { Reflection } from "@decaf-ts/reflection";
2
- import { SyncronousLock } from "./locks/SyncronousLock";
2
+ import { SyncronousLock } from "./locks/SyncronousLock.js";
3
3
  import { DBKeys, getAllPropertyDecoratorsRecursive, } from "@decaf-ts/db-decorators";
4
- import { getObjectName } from "./utils";
5
- import { TransactionalKeys } from "./constants";
4
+ import { getObjectName } from "./utils.js";
5
+ import { TransactionalKeys } from "./constants.js";
6
6
  /**
7
- * @summary Transaction Class
8
- *
9
- * @param {string} source
10
- * @param {string} [method]
11
- * @param {function(): void} [action]
12
- * @param {any[]} [metadata]
13
- *
7
+ * @description Core transaction management class
8
+ * @summary Manages transaction lifecycle, including creation, execution, and cleanup. Provides mechanisms for binding transactions to objects and methods, ensuring proper transaction context propagation.
9
+ * @param {string} source - The source/origin of the transaction (typically a class name)
10
+ * @param {string} [method] - The method name associated with the transaction
11
+ * @param {function(): any} [action] - The function to execute within the transaction
12
+ * @param {any[]} [metadata] - Additional metadata to associate with the transaction
14
13
  * @class Transaction
14
+ * @example
15
+ * // Creating and submitting a transaction
16
+ * const transaction = new Transaction(
17
+ * 'UserService',
18
+ * 'createUser',
19
+ * async () => {
20
+ * // Transaction logic here
21
+ * await db.insert('users', { name: 'John' });
22
+ * }
23
+ * );
24
+ * Transaction.submit(transaction);
15
25
  *
16
- * @category Transactions
26
+ * // Using the transactional decorator
27
+ * class UserService {
28
+ * @transactional()
29
+ * async createUser(data) {
30
+ * // Method will be executed within a transaction
31
+ * return await db.insert('users', data);
32
+ * }
33
+ * }
34
+ * @mermaid
35
+ * sequenceDiagram
36
+ * participant C as Client Code
37
+ * participant T as Transaction
38
+ * participant L as TransactionLock
39
+ * participant O as Original Method
40
+ *
41
+ * C->>T: new Transaction(source, method, action)
42
+ * C->>T: Transaction.submit(transaction)
43
+ * T->>L: submit(transaction)
44
+ * L->>T: fire()
45
+ * T->>O: Execute action()
46
+ * O-->>T: Return result/error
47
+ * T->>L: release(error?)
48
+ * L-->>C: Return result/error
17
49
  */
18
50
  export class Transaction {
19
51
  constructor(source, method, action, metadata) {
@@ -25,11 +57,12 @@ export class Transaction {
25
57
  this.metadata = metadata;
26
58
  }
27
59
  /**
28
- * @summary Pushes a transaction to que queue and waits its resolution
29
- *
30
- * @param {any} issuer any class. will be used as this when calling the callbackMethod
31
- * @param {Function} callbackMethod callback function containing the transaction. will be called with the issuear as this
32
- * @param {any[]} args arguments to pass to the method. Last one must be the callback
60
+ * @description Queues a transaction for execution
61
+ * @summary Pushes a transaction to the queue and waits for its resolution. Creates a new transaction with the provided issuer and callback method, then submits it to the transaction lock.
62
+ * @param {any} issuer - Any class instance that will be used as 'this' when calling the callbackMethod
63
+ * @param {Function} callbackMethod - Callback function containing the transaction logic, will be called with the issuer as 'this'
64
+ * @param {any[]} args - Arguments to pass to the method. Last one must be the callback function
65
+ * @return {void}
33
66
  */
34
67
  static push(issuer, callbackMethod, ...args) {
35
68
  const callback = args.pop();
@@ -46,14 +79,18 @@ export class Transaction {
46
79
  Transaction.getLock().submit(transaction);
47
80
  }
48
81
  /**
49
- * @summary Sets the lock to be used
50
- * @param lock
82
+ * @description Configures the transaction lock implementation
83
+ * @summary Sets the lock implementation to be used for transaction management, allowing customization of the transaction behavior
84
+ * @param {TransactionLock} lock - The lock implementation to use for managing transactions
85
+ * @return {void}
51
86
  */
52
87
  static setLock(lock) {
53
88
  this.lock = lock;
54
89
  }
55
90
  /**
56
- * @summary gets the lock
91
+ * @description Retrieves the current transaction lock
92
+ * @summary Gets the current transaction lock instance, creating a default SyncronousLock if none exists
93
+ * @return {TransactionLock} The current transaction lock implementation
57
94
  */
58
95
  static getLock() {
59
96
  if (!this.lock)
@@ -61,28 +98,36 @@ export class Transaction {
61
98
  return this.lock;
62
99
  }
63
100
  /**
64
- * @summary submits a transaction
65
- * @param {Transaction} transaction
101
+ * @description Submits a transaction for processing
102
+ * @summary Submits a transaction to the current transaction lock for processing and execution
103
+ * @param {Transaction} transaction - The transaction to submit for processing
104
+ * @return {void}
66
105
  */
67
106
  static submit(transaction) {
68
107
  Transaction.getLock().submit(transaction);
69
108
  }
70
109
  /**
71
- * @summary releases the lock
72
- * @param {Err} err
110
+ * @description Releases the transaction lock
111
+ * @summary Releases the current transaction lock, optionally with an error, allowing the next transaction to proceed
112
+ * @param {Error} [err] - Optional error that occurred during transaction execution
113
+ * @return {Promise<void>} A promise that resolves when the lock has been released
73
114
  */
74
115
  static async release(err) {
75
116
  return Transaction.getLock().release(err);
76
117
  }
77
118
  /**
78
- * @summary retrieves the metadata for the transaction
119
+ * @description Retrieves transaction metadata
120
+ * @summary Returns a copy of the metadata associated with this transaction, ensuring the original metadata remains unmodified
121
+ * @return {any[] | undefined} A copy of the transaction metadata or undefined if no metadata exists
79
122
  */
80
123
  getMetadata() {
81
124
  return this.metadata ? [...this.metadata] : undefined;
82
125
  }
83
126
  /**
84
- * @summary Binds a new operation to the current transaction
85
- * @param {Transaction} nextTransaction
127
+ * @description Links a new transaction to the current one
128
+ * @summary Binds a new transaction operation to the current transaction, transferring logs and binding methods to maintain transaction context
129
+ * @param {Transaction} nextTransaction - The new transaction to bind to the current one
130
+ * @return {void}
86
131
  */
87
132
  bindTransaction(nextTransaction) {
88
133
  // all(`Binding the {0} to {1}`, nextTransaction, this);
@@ -92,12 +137,10 @@ export class Transaction {
92
137
  this.action = nextTransaction.action;
93
138
  }
94
139
  /**
95
- * @summary Binds the Transactional Decorated Object to the transaction
96
- * @description by having all {@link transactional} decorated
97
- * methods always pass the current Transaction as an argument
98
- *
99
- * @param {any} obj
100
- * @return {any} the bound {@param obj}
140
+ * @description Binds an object to the current transaction context
141
+ * @summary Binds a transactional decorated object to the transaction by ensuring all transactional methods automatically receive the current transaction as their first argument
142
+ * @param {any} obj - The object to bind to the transaction
143
+ * @return {any} The bound object with transaction-aware method wrappers
101
144
  */
102
145
  bindToTransaction(obj) {
103
146
  const transactionalMethods = getAllPropertyDecoratorsRecursive(obj, undefined, TransactionalKeys.REFLECT);
@@ -131,7 +174,9 @@ export class Transaction {
131
174
  return boundObj;
132
175
  }
133
176
  /**
134
- * @summary Fires the Transaction
177
+ * @description Executes the transaction action
178
+ * @summary Fires the transaction by executing its associated action function, throwing an error if no action is defined
179
+ * @return {any} The result of the transaction action
135
180
  */
136
181
  fire() {
137
182
  if (!this.action)
@@ -139,21 +184,24 @@ export class Transaction {
139
184
  return this.action();
140
185
  }
141
186
  /**
142
- * @summary toString override
143
- * @param {boolean} [withId] defaults to true
144
- * @param {boolean} [withLog] defaults to true
187
+ * @description Provides a string representation of the transaction
188
+ * @summary Overrides the default toString method to provide a formatted string representation of the transaction, optionally including the transaction ID and log
189
+ * @param {boolean} [withId=true] - Whether to include the transaction ID in the output
190
+ * @param {boolean} [withLog=false] - Whether to include the transaction log in the output
191
+ * @return {string} A string representation of the transaction
145
192
  */
146
193
  toString(withId = true, withLog = false) {
147
194
  return `${withId ? `[${this.id}]` : ""}[Transaction][${this.source}.${this.method}${withLog ? `]\nTransaction Log:\n${this.log.join("\n")}` : "]"}`;
148
195
  }
149
196
  /**
150
- * @summary gets the transactions reflections key
151
- * @function getRepoKey
152
- * @param {string} key
153
- * @memberOf module:db-decorators.Transactions
154
- * */
197
+ * @description Generates a reflection metadata key for transactions
198
+ * @summary Creates a prefixed reflection key for transaction-related metadata, ensuring proper namespacing
199
+ * @param {string} key - The base key to prefix with the transaction reflection namespace
200
+ * @return {string} The complete reflection key for transaction metadata
201
+ * @function key
202
+ */
155
203
  static key(key) {
156
204
  return TransactionalKeys.REFLECT + key;
157
205
  }
158
206
  }
159
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Transaction.js","sourceRoot":"","sources":["../../src/Transaction.ts"],"names":[],"mappings":"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","sourcesContent":["import { TransactionLock } from \"./interfaces/TransactionLock\";\nimport { Reflection } from \"@decaf-ts/reflection\";\nimport { Callback } from \"./types\";\nimport { SyncronousLock } from \"./locks/SyncronousLock\";\nimport {\n  DBKeys,\n  getAllPropertyDecoratorsRecursive,\n} from \"@decaf-ts/db-decorators\";\nimport { getObjectName } from \"./utils\";\nimport { TransactionalKeys } from \"./constants\";\n\n/**\n * @summary Transaction Class\n *\n * @param {string} source\n * @param {string} [method]\n * @param {function(): void} [action]\n * @param {any[]} [metadata]\n *\n * @class Transaction\n *\n * @category Transactions\n */\nexport class Transaction {\n  readonly id: number;\n  protected action?: () => any;\n  readonly method?: string;\n  readonly source?: string;\n  readonly log: string[];\n  private readonly metadata?: any[];\n\n  private static lock: TransactionLock;\n\n  constructor(\n    source: string,\n    method?: string,\n    action?: () => any,\n    metadata?: any[]\n  ) {\n    this.id = Date.now();\n    this.action = action;\n    this.method = method;\n    this.log = [[this.id, source, method].join(\" | \")];\n    this.source = source;\n    this.metadata = metadata;\n  }\n\n  /**\n   * @summary Pushes a transaction to que queue and waits its resolution\n   *\n   * @param {any} issuer any class. will be used as this when calling the callbackMethod\n   * @param {Function} callbackMethod callback function containing the transaction. will be called with the issuear as this\n   * @param {any[]} args arguments to pass to the method. Last one must be the callback\n   */\n  static push(\n    issuer: any,\n    callbackMethod: (...argzz: (any | Callback)[]) => void,\n    ...args: (any | Callback)[]\n  ) {\n    const callback: Callback = args.pop();\n    if (!callback || typeof callback !== \"function\")\n      throw new Error(\"Missing callback\");\n    const cb = (err?: Error, ...args: any[]) => {\n      Transaction.getLock()\n        .release(err)\n        .then(() => callback(err, ...args));\n    };\n    const transaction: Transaction = new Transaction(\n      issuer.constructor.name,\n      callbackMethod.name ? getObjectName(callbackMethod) : \"Anonymous\",\n      () => {\n        return callbackMethod.call(\n          transaction.bindToTransaction(issuer),\n          ...args,\n          cb\n        );\n      }\n    );\n    Transaction.getLock().submit(transaction);\n  }\n\n  /**\n   * @summary Sets the lock to be used\n   * @param lock\n   */\n  static setLock(lock: TransactionLock) {\n    this.lock = lock;\n  }\n\n  /**\n   * @summary gets the lock\n   */\n  static getLock(): TransactionLock {\n    if (!this.lock) this.lock = new SyncronousLock();\n    return this.lock;\n  }\n\n  /**\n   * @summary submits a transaction\n   * @param {Transaction} transaction\n   */\n  static submit(transaction: Transaction) {\n    Transaction.getLock().submit(transaction);\n  }\n\n  /**\n   * @summary releases the lock\n   * @param {Err} err\n   */\n  static async release(err?: Error) {\n    return Transaction.getLock().release(err);\n  }\n\n  /**\n   * @summary retrieves the metadata for the transaction\n   */\n  getMetadata() {\n    return this.metadata ? [...this.metadata] : undefined;\n  }\n\n  /**\n   * @summary Binds a new operation to the current transaction\n   * @param {Transaction} nextTransaction\n   */\n  bindTransaction(nextTransaction: Transaction) {\n    // all(`Binding the {0} to {1}`, nextTransaction, this);\n    this.log.push(...nextTransaction.log);\n    nextTransaction.bindTransaction = this.bindToTransaction.bind(this);\n    nextTransaction.bindToTransaction = this.bindToTransaction.bind(this);\n    this.action = nextTransaction.action;\n  }\n\n  /**\n   * @summary Binds the Transactional Decorated Object to the transaction\n   * @description by having all {@link transactional} decorated\n   * methods always pass the current Transaction as an argument\n   *\n   * @param {any} obj\n   * @return {any} the bound {@param obj}\n   */\n  bindToTransaction(obj: any): any {\n    const transactionalMethods = getAllPropertyDecoratorsRecursive(\n      obj,\n      undefined,\n      TransactionalKeys.REFLECT\n    );\n    if (!transactionalMethods) return obj;\n    // eslint-disable-next-line @typescript-eslint/no-this-alias\n    const self = this;\n\n    const boundObj = Reflection.getAllProperties(obj).reduce(\n      (accum: any, k: string) => {\n        if (\n          Object.keys(transactionalMethods).indexOf(k) !== -1 &&\n          transactionalMethods[k].find(\n            (o) => o.key === TransactionalKeys.TRANSACTIONAL\n          )\n        )\n          accum[k] = (...args: any[]) =>\n            obj[k].call(obj.__originalObj || obj, self, ...args);\n        else if (k === \"clazz\" || k === \"constructor\") accum[k] = obj[k];\n        else if (typeof obj[k] === \"function\")\n          accum[k] = obj[k].bind(obj.__originalObj || obj);\n        else if (typeof obj[k] === \"object\" && obj[k].constructor) {\n          const decs = Reflection.getClassDecorators(\n            TransactionalKeys.REFLECT,\n            obj[k]\n          );\n          if (decs.find((e: any) => e.key === TransactionalKeys.TRANSACTIONAL))\n            accum[k] = self.bindToTransaction(obj[k]);\n          else accum[k] = obj[k];\n        } else accum[k] = obj[k];\n\n        return accum;\n      },\n      {}\n    );\n\n    boundObj[DBKeys.ORIGINAL] = obj[DBKeys.ORIGINAL] || obj;\n    boundObj.toString = () =>\n      getObjectName(boundObj[DBKeys.ORIGINAL]) +\n      \" proxy for transaction \" +\n      this.id;\n\n    return boundObj;\n  }\n\n  /**\n   * @summary Fires the Transaction\n   */\n  fire() {\n    if (!this.action) throw new Error(`Missing the method`);\n    return this.action();\n  }\n\n  /**\n   * @summary toString override\n   * @param {boolean} [withId] defaults to true\n   * @param {boolean} [withLog] defaults to true\n   */\n  toString(withId = true, withLog = false) {\n    return `${withId ? `[${this.id}]` : \"\"}[Transaction][${this.source}.${this.method}${\n      withLog ? `]\\nTransaction Log:\\n${this.log.join(\"\\n\")}` : \"]\"\n    }`;\n  }\n\n  /**\n   * @summary gets the transactions reflections key\n   * @function getRepoKey\n   * @param {string} key\n   * @memberOf module:db-decorators.Transactions\n   * */\n  static key(key: string) {\n    return TransactionalKeys.REFLECT + key;\n  }\n}\n"]}
207
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Transaction.js","sourceRoot":"","sources":["../../src/Transaction.ts"],"names":[],"mappings":"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","sourcesContent":["import { TransactionLock } from \"./interfaces/TransactionLock\";\nimport { Reflection } from \"@decaf-ts/reflection\";\nimport { Callback } from \"./types\";\nimport { SyncronousLock } from \"./locks/SyncronousLock\";\nimport {\n  DBKeys,\n  getAllPropertyDecoratorsRecursive,\n} from \"@decaf-ts/db-decorators\";\nimport { getObjectName } from \"./utils\";\nimport { TransactionalKeys } from \"./constants\";\n\n/**\n * @description Core transaction management class\n * @summary Manages transaction lifecycle, including creation, execution, and cleanup. Provides mechanisms for binding transactions to objects and methods, ensuring proper transaction context propagation.\n * @param {string} source - The source/origin of the transaction (typically a class name)\n * @param {string} [method] - The method name associated with the transaction\n * @param {function(): any} [action] - The function to execute within the transaction\n * @param {any[]} [metadata] - Additional metadata to associate with the transaction\n * @class Transaction\n * @example\n * // Creating and submitting a transaction\n * const transaction = new Transaction(\n *   'UserService',\n *   'createUser',\n *   async () => {\n *     // Transaction logic here\n *     await db.insert('users', { name: 'John' });\n *   }\n * );\n * Transaction.submit(transaction);\n * \n * // Using the transactional decorator\n * class UserService {\n *   @transactional()\n *   async createUser(data) {\n *     // Method will be executed within a transaction\n *     return await db.insert('users', data);\n *   }\n * }\n * @mermaid\n * sequenceDiagram\n *   participant C as Client Code\n *   participant T as Transaction\n *   participant L as TransactionLock\n *   participant O as Original Method\n *   \n *   C->>T: new Transaction(source, method, action)\n *   C->>T: Transaction.submit(transaction)\n *   T->>L: submit(transaction)\n *   L->>T: fire()\n *   T->>O: Execute action()\n *   O-->>T: Return result/error\n *   T->>L: release(error?)\n *   L-->>C: Return result/error\n */\nexport class Transaction {\n  readonly id: number;\n  protected action?: () => any;\n  readonly method?: string;\n  readonly source?: string;\n  readonly log: string[];\n  private readonly metadata?: any[];\n\n  private static lock: TransactionLock;\n\n  constructor(\n    source: string,\n    method?: string,\n    action?: () => any,\n    metadata?: any[]\n  ) {\n    this.id = Date.now();\n    this.action = action;\n    this.method = method;\n    this.log = [[this.id, source, method].join(\" | \")];\n    this.source = source;\n    this.metadata = metadata;\n  }\n\n  /**\n   * @description Queues a transaction for execution\n   * @summary Pushes a transaction to the queue and waits for its resolution. Creates a new transaction with the provided issuer and callback method, then submits it to the transaction lock.\n   * @param {any} issuer - Any class instance that will be used as 'this' when calling the callbackMethod\n   * @param {Function} callbackMethod - Callback function containing the transaction logic, will be called with the issuer as 'this'\n   * @param {any[]} args - Arguments to pass to the method. Last one must be the callback function\n   * @return {void}\n   */\n  static push(\n    issuer: any,\n    callbackMethod: (...argzz: (any | Callback)[]) => void,\n    ...args: (any | Callback)[]\n  ) {\n    const callback: Callback = args.pop();\n    if (!callback || typeof callback !== \"function\")\n      throw new Error(\"Missing callback\");\n    const cb = (err?: Error, ...args: any[]) => {\n      Transaction.getLock()\n        .release(err)\n        .then(() => callback(err, ...args));\n    };\n    const transaction: Transaction = new Transaction(\n      issuer.constructor.name,\n      callbackMethod.name ? getObjectName(callbackMethod) : \"Anonymous\",\n      () => {\n        return callbackMethod.call(\n          transaction.bindToTransaction(issuer),\n          ...args,\n          cb\n        );\n      }\n    );\n    Transaction.getLock().submit(transaction);\n  }\n\n  /**\n   * @description Configures the transaction lock implementation\n   * @summary Sets the lock implementation to be used for transaction management, allowing customization of the transaction behavior\n   * @param {TransactionLock} lock - The lock implementation to use for managing transactions\n   * @return {void}\n   */\n  static setLock(lock: TransactionLock) {\n    this.lock = lock;\n  }\n\n  /**\n   * @description Retrieves the current transaction lock\n   * @summary Gets the current transaction lock instance, creating a default SyncronousLock if none exists\n   * @return {TransactionLock} The current transaction lock implementation\n   */\n  static getLock(): TransactionLock {\n    if (!this.lock) this.lock = new SyncronousLock();\n    return this.lock;\n  }\n\n  /**\n   * @description Submits a transaction for processing\n   * @summary Submits a transaction to the current transaction lock for processing and execution\n   * @param {Transaction} transaction - The transaction to submit for processing\n   * @return {void}\n   */\n  static submit(transaction: Transaction) {\n    Transaction.getLock().submit(transaction);\n  }\n\n  /**\n   * @description Releases the transaction lock\n   * @summary Releases the current transaction lock, optionally with an error, allowing the next transaction to proceed\n   * @param {Error} [err] - Optional error that occurred during transaction execution\n   * @return {Promise<void>} A promise that resolves when the lock has been released\n   */\n  static async release(err?: Error) {\n    return Transaction.getLock().release(err);\n  }\n\n  /**\n   * @description Retrieves transaction metadata\n   * @summary Returns a copy of the metadata associated with this transaction, ensuring the original metadata remains unmodified\n   * @return {any[] | undefined} A copy of the transaction metadata or undefined if no metadata exists\n   */\n  getMetadata() {\n    return this.metadata ? [...this.metadata] : undefined;\n  }\n\n  /**\n   * @description Links a new transaction to the current one\n   * @summary Binds a new transaction operation to the current transaction, transferring logs and binding methods to maintain transaction context\n   * @param {Transaction} nextTransaction - The new transaction to bind to the current one\n   * @return {void}\n   */\n  bindTransaction(nextTransaction: Transaction) {\n    // all(`Binding the {0} to {1}`, nextTransaction, this);\n    this.log.push(...nextTransaction.log);\n    nextTransaction.bindTransaction = this.bindToTransaction.bind(this);\n    nextTransaction.bindToTransaction = this.bindToTransaction.bind(this);\n    this.action = nextTransaction.action;\n  }\n\n  /**\n   * @description Binds an object to the current transaction context\n   * @summary Binds a transactional decorated object to the transaction by ensuring all transactional methods automatically receive the current transaction as their first argument\n   * @param {any} obj - The object to bind to the transaction\n   * @return {any} The bound object with transaction-aware method wrappers\n   */\n  bindToTransaction(obj: any): any {\n    const transactionalMethods = getAllPropertyDecoratorsRecursive(\n      obj,\n      undefined,\n      TransactionalKeys.REFLECT\n    );\n    if (!transactionalMethods) return obj;\n    // eslint-disable-next-line @typescript-eslint/no-this-alias\n    const self = this;\n\n    const boundObj = Reflection.getAllProperties(obj).reduce(\n      (accum: any, k: string) => {\n        if (\n          Object.keys(transactionalMethods).indexOf(k) !== -1 &&\n          transactionalMethods[k].find(\n            (o) => o.key === TransactionalKeys.TRANSACTIONAL\n          )\n        )\n          accum[k] = (...args: any[]) =>\n            obj[k].call(obj.__originalObj || obj, self, ...args);\n        else if (k === \"clazz\" || k === \"constructor\") accum[k] = obj[k];\n        else if (typeof obj[k] === \"function\")\n          accum[k] = obj[k].bind(obj.__originalObj || obj);\n        else if (typeof obj[k] === \"object\" && obj[k].constructor) {\n          const decs = Reflection.getClassDecorators(\n            TransactionalKeys.REFLECT,\n            obj[k]\n          );\n          if (decs.find((e: any) => e.key === TransactionalKeys.TRANSACTIONAL))\n            accum[k] = self.bindToTransaction(obj[k]);\n          else accum[k] = obj[k];\n        } else accum[k] = obj[k];\n\n        return accum;\n      },\n      {}\n    );\n\n    boundObj[DBKeys.ORIGINAL] = obj[DBKeys.ORIGINAL] || obj;\n    boundObj.toString = () =>\n      getObjectName(boundObj[DBKeys.ORIGINAL]) +\n      \" proxy for transaction \" +\n      this.id;\n\n    return boundObj;\n  }\n\n  /**\n   * @description Executes the transaction action\n   * @summary Fires the transaction by executing its associated action function, throwing an error if no action is defined\n   * @return {any} The result of the transaction action\n   */\n  fire() {\n    if (!this.action) throw new Error(`Missing the method`);\n    return this.action();\n  }\n\n  /**\n   * @description Provides a string representation of the transaction\n   * @summary Overrides the default toString method to provide a formatted string representation of the transaction, optionally including the transaction ID and log\n   * @param {boolean} [withId=true] - Whether to include the transaction ID in the output\n   * @param {boolean} [withLog=false] - Whether to include the transaction log in the output\n   * @return {string} A string representation of the transaction\n   */\n  toString(withId = true, withLog = false) {\n    return `${withId ? `[${this.id}]` : \"\"}[Transaction][${this.source}.${this.method}${\n      withLog ? `]\\nTransaction Log:\\n${this.log.join(\"\\n\")}` : \"]\"\n    }`;\n  }\n\n  /**\n   * @description Generates a reflection metadata key for transactions\n   * @summary Creates a prefixed reflection key for transaction-related metadata, ensuring proper namespacing\n   * @param {string} key - The base key to prefix with the transaction reflection namespace\n   * @return {string} The complete reflection key for transaction metadata\n   * @function key\n   */\n  static key(key: string) {\n    return TransactionalKeys.REFLECT + key;\n  }\n}\n"]}
@@ -1 +1,14 @@
1
+ /**
2
+ * @typedef {Object} TransactionalKeysType
3
+ * @property {string} REFLECT - Key used for reflection metadata related to transactional models
4
+ * @property {string} TRANSACTIONAL - Key used to identify transactional properties
5
+ * @memberOf module:transactions
6
+ */
7
+ /**
8
+ * @description Keys used for transactional operations
9
+ * @summary Constant object containing string keys used throughout the transactional system for reflection and identification
10
+ * @type {TransactionalKeysType}
11
+ * @const TransactionalKeys
12
+ * @memberOf module:transactions
13
+ */
1
14
  export declare const TransactionalKeys: Record<string, string>;
@@ -1,5 +1,18 @@
1
+ /**
2
+ * @typedef {Object} TransactionalKeysType
3
+ * @property {string} REFLECT - Key used for reflection metadata related to transactional models
4
+ * @property {string} TRANSACTIONAL - Key used to identify transactional properties
5
+ * @memberOf module:transactions
6
+ */
7
+ /**
8
+ * @description Keys used for transactional operations
9
+ * @summary Constant object containing string keys used throughout the transactional system for reflection and identification
10
+ * @type {TransactionalKeysType}
11
+ * @const TransactionalKeys
12
+ * @memberOf module:transactions
13
+ */
1
14
  export const TransactionalKeys = {
2
15
  REFLECT: "model.transactional.",
3
16
  TRANSACTIONAL: "transactional",
4
17
  };
5
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uc3RhbnRzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2NvbnN0YW50cy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxNQUFNLENBQUMsTUFBTSxpQkFBaUIsR0FBMkI7SUFDdkQsT0FBTyxFQUFFLHNCQUFzQjtJQUMvQixhQUFhLEVBQUUsZUFBZTtDQUMvQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGNvbnN0IFRyYW5zYWN0aW9uYWxLZXlzOiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+ID0ge1xuICBSRUZMRUNUOiBcIm1vZGVsLnRyYW5zYWN0aW9uYWwuXCIsXG4gIFRSQU5TQUNUSU9OQUw6IFwidHJhbnNhY3Rpb25hbFwiLFxufTtcbiJdfQ==
18
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uc3RhbnRzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2NvbnN0YW50cy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7Ozs7R0FLRztBQUVIOzs7Ozs7R0FNRztBQUNILE1BQU0sQ0FBQyxNQUFNLGlCQUFpQixHQUEyQjtJQUN2RCxPQUFPLEVBQUUsc0JBQXNCO0lBQy9CLGFBQWEsRUFBRSxlQUFlO0NBQy9CLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEB0eXBlZGVmIHtPYmplY3R9IFRyYW5zYWN0aW9uYWxLZXlzVHlwZVxuICogQHByb3BlcnR5IHtzdHJpbmd9IFJFRkxFQ1QgLSBLZXkgdXNlZCBmb3IgcmVmbGVjdGlvbiBtZXRhZGF0YSByZWxhdGVkIHRvIHRyYW5zYWN0aW9uYWwgbW9kZWxzXG4gKiBAcHJvcGVydHkge3N0cmluZ30gVFJBTlNBQ1RJT05BTCAtIEtleSB1c2VkIHRvIGlkZW50aWZ5IHRyYW5zYWN0aW9uYWwgcHJvcGVydGllc1xuICogQG1lbWJlck9mIG1vZHVsZTp0cmFuc2FjdGlvbnNcbiAqL1xuXG4vKipcbiAqIEBkZXNjcmlwdGlvbiBLZXlzIHVzZWQgZm9yIHRyYW5zYWN0aW9uYWwgb3BlcmF0aW9uc1xuICogQHN1bW1hcnkgQ29uc3RhbnQgb2JqZWN0IGNvbnRhaW5pbmcgc3RyaW5nIGtleXMgdXNlZCB0aHJvdWdob3V0IHRoZSB0cmFuc2FjdGlvbmFsIHN5c3RlbSBmb3IgcmVmbGVjdGlvbiBhbmQgaWRlbnRpZmljYXRpb25cbiAqIEB0eXBlIHtUcmFuc2FjdGlvbmFsS2V5c1R5cGV9XG4gKiBAY29uc3QgVHJhbnNhY3Rpb25hbEtleXNcbiAqIEBtZW1iZXJPZiBtb2R1bGU6dHJhbnNhY3Rpb25zXG4gKi9cbmV4cG9ydCBjb25zdCBUcmFuc2FjdGlvbmFsS2V5czogUmVjb3JkPHN0cmluZywgc3RyaW5nPiA9IHtcbiAgUkVGTEVDVDogXCJtb2RlbC50cmFuc2FjdGlvbmFsLlwiLFxuICBUUkFOU0FDVElPTkFMOiBcInRyYW5zYWN0aW9uYWxcIixcbn07XG4iXX0=
@@ -1,19 +1,43 @@
1
1
  /**
2
- * @summary Sets a class Async method as transactional
2
+ * @description Method decorator that enables transactional behavior
3
+ * @summary Sets a class async method as transactional, wrapping it in a transaction context that can be managed by the transaction system. This decorator handles transaction creation, binding, and error handling.
4
+ * @param {any[]} [data] - Optional metadata available to the {@link TransactionLock} implementation
5
+ * @return {Function} A decorator function that wraps the original method with transactional behavior
6
+ * @function transactional
7
+ * @category Method Decorators
8
+ * @mermaid
9
+ * sequenceDiagram
10
+ * participant C as Client Code
11
+ * participant D as Decorator
12
+ * participant T as Transaction
13
+ * participant O as Original Method
3
14
  *
4
- * @param {any[]} [data] option metadata available to the {@link TransactionLock}
15
+ * C->>D: Call decorated method
16
+ * D->>D: Check if transaction exists in args
5
17
  *
6
- * @function transactional
18
+ * alt Transaction exists in args
19
+ * D->>T: Create updated transaction
20
+ * T->>T: Bind to original transaction
21
+ * T->>T: Fire transaction
22
+ * else No transaction
23
+ * D->>T: Create new transaction
24
+ * T->>T: Submit transaction
25
+ * end
7
26
  *
8
- * @memberOf module:db-decorators.Decorators.transactions
27
+ * T->>O: Execute original method
28
+ * O-->>T: Return result/error
29
+ * T->>T: Release transaction
30
+ * T-->>C: Return result/error
31
+ * @category Decorators
9
32
  */
10
33
  export declare function transactional(...data: any[]): (target: any, propertyKey?: any, descriptor?: PropertyDescriptor) => void;
11
34
  /**
12
- * @summary Util function to wrap super calls with the transaction when the super's method is also transactional
13
- *
14
- * @param {Function} method the super method (must be bound to the proper this), eg: super.create.bind(this)
15
- * @param {any[]} args the arguments to call the method with
16
- *
17
- * @memberOf module:db-decorators.Transaction
35
+ * @description Utility for handling super calls in transactional methods
36
+ * @summary Wraps super method calls with the current transaction context when the super's method is also transactional, ensuring transaction continuity through the inheritance chain
37
+ * @param {Function} method - The super method (must be bound to the proper this), e.g., super.create.bind(this)
38
+ * @param {any[]} args - The arguments to call the method with
39
+ * @return {any} The result of the super method call
40
+ * @function transactionalSuperCall
41
+ * @memberOf module:transactions
18
42
  */
19
43
  export declare function transactionalSuperCall(method: any, ...args: any): any;