@decaf-ts/transactional-decorators 0.0.4 → 0.0.5
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +4 -4
- package/dist/esm/transactional-decorators.bundle.min.esm.js +1 -1
- package/dist/transactional-decorators.bundle.min.js +1 -1
- package/lib/Transaction.cjs +152 -3
- package/lib/Transaction.d.ts +7 -0
- package/lib/constants.cjs +9 -1
- package/lib/decorators.cjs +166 -1
- package/lib/decorators.d.ts +0 -7
- package/lib/esm/Transaction.d.ts +7 -0
- package/lib/esm/Transaction.js +148 -3
- package/lib/esm/constants.js +6 -1
- package/lib/esm/decorators.d.ts +0 -7
- package/lib/esm/decorators.js +162 -1
- package/lib/esm/index.d.ts +1 -1
- package/lib/esm/index.js +26 -1
- package/lib/esm/interfaces/TransactionLock.js +3 -0
- package/lib/esm/interfaces/index.js +3 -1
- package/lib/esm/locks/Lock.js +63 -1
- package/lib/esm/locks/SyncronousLock.js +117 -1
- package/lib/esm/locks/index.js +4 -1
- package/lib/esm/types.js +3 -0
- package/lib/esm/utils.js +15 -1
- package/lib/index.cjs +43 -1
- package/lib/index.d.ts +1 -1
- package/lib/interfaces/TransactionLock.cjs +4 -1
- package/lib/interfaces/index.cjs +19 -1
- package/lib/locks/Lock.cjs +67 -1
- package/lib/locks/SyncronousLock.cjs +121 -1
- package/lib/locks/index.cjs +20 -1
- package/lib/types.cjs +4 -1
- package/lib/utils.cjs +18 -1
- package/package.json +21 -18
package/lib/Transaction.cjs
CHANGED
|
@@ -1,3 +1,152 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.Transaction = void 0;
|
|
4
|
+
const reflection_1 = require("@decaf-ts/reflection");
|
|
5
|
+
const SyncronousLock_1 = require("./locks/SyncronousLock.cjs");
|
|
6
|
+
const db_decorators_1 = require("@decaf-ts/db-decorators");
|
|
7
|
+
const utils_1 = require("./utils.cjs");
|
|
8
|
+
const constants_1 = require("./constants.cjs");
|
|
9
|
+
class Transaction {
|
|
10
|
+
constructor(source, method, action, metadata) {
|
|
11
|
+
this.id = Date.now();
|
|
12
|
+
this.action = action;
|
|
13
|
+
this.method = method;
|
|
14
|
+
this.log = [[this.id, source, method].join(" | ")];
|
|
15
|
+
this.source = source;
|
|
16
|
+
this.metadata = metadata;
|
|
17
|
+
}
|
|
18
|
+
/**
|
|
19
|
+
* @summary Pushes a transaction to que queue and waits its resolution
|
|
20
|
+
*
|
|
21
|
+
* @param {any} issuer any class. will be used as this when calling the callbackMethod
|
|
22
|
+
* @param {Function} callbackMethod callback function containing the transaction. will be called with the issuear as this
|
|
23
|
+
* @param {any[]} args arguments to pass to the method. Last one must be the callback
|
|
24
|
+
*/
|
|
25
|
+
static push(issuer, callbackMethod, ...args) {
|
|
26
|
+
const callback = args.pop();
|
|
27
|
+
if (!callback || typeof callback !== "function")
|
|
28
|
+
throw new Error("Missing callback");
|
|
29
|
+
const cb = (err, ...args) => {
|
|
30
|
+
this.getLock()
|
|
31
|
+
.release(err)
|
|
32
|
+
.then(() => callback(err, ...args));
|
|
33
|
+
};
|
|
34
|
+
const transaction = new Transaction(issuer.constructor.name, callbackMethod.name ? (0, utils_1.getObjectName)(callbackMethod) : "Anonymous", () => {
|
|
35
|
+
return callbackMethod.call(transaction.bindToTransaction(issuer), ...args, cb);
|
|
36
|
+
});
|
|
37
|
+
this.getLock().submit(transaction);
|
|
38
|
+
}
|
|
39
|
+
/**
|
|
40
|
+
* @summary Sets the lock to be used
|
|
41
|
+
* @param lock
|
|
42
|
+
*/
|
|
43
|
+
static setLock(lock) {
|
|
44
|
+
this.lock = lock;
|
|
45
|
+
}
|
|
46
|
+
/**
|
|
47
|
+
* @summary gets the lock
|
|
48
|
+
*/
|
|
49
|
+
static getLock() {
|
|
50
|
+
if (!this.lock)
|
|
51
|
+
this.lock = new SyncronousLock_1.SyncronousLock();
|
|
52
|
+
return this.lock;
|
|
53
|
+
}
|
|
54
|
+
/**
|
|
55
|
+
* @summary submits a transaction
|
|
56
|
+
* @param {Transaction} transaction
|
|
57
|
+
*/
|
|
58
|
+
static submit(transaction) {
|
|
59
|
+
this.getLock().submit(transaction);
|
|
60
|
+
}
|
|
61
|
+
/**
|
|
62
|
+
* @summary releases the lock
|
|
63
|
+
* @param {Err} err
|
|
64
|
+
*/
|
|
65
|
+
static async release(err) {
|
|
66
|
+
return this.getLock().release(err);
|
|
67
|
+
}
|
|
68
|
+
/**
|
|
69
|
+
* @summary retrieves the metadata for the transaction
|
|
70
|
+
*/
|
|
71
|
+
getMetadata() {
|
|
72
|
+
return this.metadata ? [...this.metadata] : undefined;
|
|
73
|
+
}
|
|
74
|
+
/**
|
|
75
|
+
* @summary Binds a new operation to the current transaction
|
|
76
|
+
* @param {Transaction} nextTransaction
|
|
77
|
+
*/
|
|
78
|
+
bindTransaction(nextTransaction) {
|
|
79
|
+
// all(`Binding the {0} to {1}`, nextTransaction, this);
|
|
80
|
+
this.log.push(...nextTransaction.log);
|
|
81
|
+
nextTransaction.bindTransaction = this.bindToTransaction.bind(this);
|
|
82
|
+
nextTransaction.bindToTransaction = this.bindToTransaction.bind(this);
|
|
83
|
+
this.action = nextTransaction.action;
|
|
84
|
+
}
|
|
85
|
+
/**
|
|
86
|
+
* @summary Binds the Transactional Decorated Object to the transaction
|
|
87
|
+
* @description by having all {@link transactional} decorated
|
|
88
|
+
* methods always pass the current Transaction as an argument
|
|
89
|
+
*
|
|
90
|
+
* @param {any} obj
|
|
91
|
+
* @return {any} the bound {@param obj}
|
|
92
|
+
*/
|
|
93
|
+
bindToTransaction(obj) {
|
|
94
|
+
const transactionalMethods = (0, db_decorators_1.getAllPropertyDecoratorsRecursive)(obj, undefined, constants_1.TransactionalKeys.REFLECT);
|
|
95
|
+
if (!transactionalMethods)
|
|
96
|
+
return obj;
|
|
97
|
+
// eslint-disable-next-line @typescript-eslint/no-this-alias
|
|
98
|
+
const self = this;
|
|
99
|
+
const boundObj = (0, reflection_1.getAllProperties)(obj).reduce((accum, k) => {
|
|
100
|
+
if (Object.keys(transactionalMethods).indexOf(k) !== -1 &&
|
|
101
|
+
transactionalMethods[k].find((o) => o.key === constants_1.TransactionalKeys.TRANSACTIONAL))
|
|
102
|
+
accum[k] = (...args) => obj[k].call(obj.__originalObj || obj, self, ...args);
|
|
103
|
+
else if (k === "clazz" || k === "constructor")
|
|
104
|
+
accum[k] = obj[k];
|
|
105
|
+
else if (typeof obj[k] === "function")
|
|
106
|
+
accum[k] = obj[k].bind(obj.__originalObj || obj);
|
|
107
|
+
else if (typeof obj[k] === "object" && obj[k].constructor) {
|
|
108
|
+
const decs = (0, reflection_1.getClassDecorators)(constants_1.TransactionalKeys.REFLECT, obj[k]);
|
|
109
|
+
if (decs.find((e) => e.key === constants_1.TransactionalKeys.TRANSACTIONAL))
|
|
110
|
+
accum[k] = self.bindToTransaction(obj[k]);
|
|
111
|
+
else
|
|
112
|
+
accum[k] = obj[k];
|
|
113
|
+
}
|
|
114
|
+
else
|
|
115
|
+
accum[k] = obj[k];
|
|
116
|
+
return accum;
|
|
117
|
+
}, {});
|
|
118
|
+
boundObj[db_decorators_1.DBKeys.ORIGINAL] = obj[db_decorators_1.DBKeys.ORIGINAL] || obj;
|
|
119
|
+
boundObj.toString = () => (0, utils_1.getObjectName)(boundObj[db_decorators_1.DBKeys.ORIGINAL]) +
|
|
120
|
+
" proxy for transaction " +
|
|
121
|
+
this.id;
|
|
122
|
+
return boundObj;
|
|
123
|
+
}
|
|
124
|
+
/**
|
|
125
|
+
* @summary Fires the Transaction
|
|
126
|
+
*/
|
|
127
|
+
fire() {
|
|
128
|
+
if (!this.action)
|
|
129
|
+
throw new Error(`Missing the method`);
|
|
130
|
+
return this.action();
|
|
131
|
+
}
|
|
132
|
+
/**
|
|
133
|
+
* @summary toString override
|
|
134
|
+
* @param {boolean} [withId] defaults to true
|
|
135
|
+
* @param {boolean} [withLog] defaults to true
|
|
136
|
+
*/
|
|
137
|
+
toString(withId = true, withLog = false) {
|
|
138
|
+
return `${withId ? `[${this.id}]` : ""}[Transaction][${this.source}.${this.method}${withLog ? `]\nTransaction Log:\n${this.log.join("\n")}` : "]"}`;
|
|
139
|
+
}
|
|
140
|
+
/**
|
|
141
|
+
* @summary gets the transactions reflections key
|
|
142
|
+
* @function getRepoKey
|
|
143
|
+
* @param {string} key
|
|
144
|
+
* @memberOf module:db-decorators.Transactions
|
|
145
|
+
* */
|
|
146
|
+
static key(key) {
|
|
147
|
+
return constants_1.TransactionalKeys.REFLECT + key;
|
|
148
|
+
}
|
|
149
|
+
}
|
|
150
|
+
exports.Transaction = Transaction;
|
|
151
|
+
|
|
152
|
+
//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["src/Transaction.ts"],"names":[],"mappings":"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","file":"Transaction.js","sourcesContent":["/**\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 */\nimport { TransactionLock } from \"./interfaces/TransactionLock\";\nimport { getAllProperties, getClassDecorators } 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\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      this.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    this.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    this.getLock().submit(transaction);\n  }\n\n  /**\n   * @summary releases the lock\n   * @param {Err} err\n   */\n  static async release(err?: Error) {\n    return this.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 = getAllProperties(obj).reduce((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 = getClassDecorators(TransactionalKeys.REFLECT, obj[k]);\n        if (decs.find((e) => 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    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"]}
|
package/lib/Transaction.d.ts
CHANGED
|
@@ -76,4 +76,11 @@ export declare class Transaction {
|
|
|
76
76
|
* @param {boolean} [withLog] defaults to true
|
|
77
77
|
*/
|
|
78
78
|
toString(withId?: boolean, withLog?: boolean): string;
|
|
79
|
+
/**
|
|
80
|
+
* @summary gets the transactions reflections key
|
|
81
|
+
* @function getRepoKey
|
|
82
|
+
* @param {string} key
|
|
83
|
+
* @memberOf module:db-decorators.Transactions
|
|
84
|
+
* */
|
|
85
|
+
static key(key: string): string;
|
|
79
86
|
}
|
package/lib/constants.cjs
CHANGED
|
@@ -1 +1,9 @@
|
|
|
1
|
-
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.TransactionalKeys = void 0;
|
|
4
|
+
exports.TransactionalKeys = {
|
|
5
|
+
REFLECT: "model.transactional.",
|
|
6
|
+
TRANSACTIONAL: "transactional",
|
|
7
|
+
};
|
|
8
|
+
|
|
9
|
+
//# sourceMappingURL=data:application/json;charset=utf8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbInNyYy9jb25zdGFudHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQWEsUUFBQSxpQkFBaUIsR0FBMkI7SUFDdkQsT0FBTyxFQUFFLHNCQUFzQjtJQUMvQixhQUFhLEVBQUUsZUFBZTtDQUMvQixDQUFDIiwiZmlsZSI6ImNvbnN0YW50cy5qcyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBjb25zdCBUcmFuc2FjdGlvbmFsS2V5czogUmVjb3JkPHN0cmluZywgc3RyaW5nPiA9IHtcbiAgUkVGTEVDVDogXCJtb2RlbC50cmFuc2FjdGlvbmFsLlwiLFxuICBUUkFOU0FDVElPTkFMOiBcInRyYW5zYWN0aW9uYWxcIixcbn07XG4iXX0=
|
package/lib/decorators.cjs
CHANGED
|
@@ -1 +1,166 @@
|
|
|
1
|
-
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.transactional = transactional;
|
|
4
|
+
exports.transactionalSuperCall = transactionalSuperCall;
|
|
5
|
+
const constants_1 = require("./constants.cjs");
|
|
6
|
+
const reflection_1 = require("@decaf-ts/reflection");
|
|
7
|
+
const Transaction_1 = require("./Transaction.cjs");
|
|
8
|
+
/**
|
|
9
|
+
* @summary Sets a class Async (promise based) method as transactional
|
|
10
|
+
*
|
|
11
|
+
* @param {any[]} [data] option metadata available to the {@link TransactionLock}
|
|
12
|
+
*
|
|
13
|
+
* @function transactionalPromise
|
|
14
|
+
*
|
|
15
|
+
* @memberOf module:db-decorators.Decorators.transactions
|
|
16
|
+
*/
|
|
17
|
+
function transactional(...data) {
|
|
18
|
+
return function (target, propertyKey, descriptor) {
|
|
19
|
+
(0, reflection_1.metadata)(Transaction_1.Transaction.key(constants_1.TransactionalKeys.TRANSACTIONAL), data)(target, propertyKey);
|
|
20
|
+
const originalMethod = descriptor.value;
|
|
21
|
+
const methodWrapper = function (...args) {
|
|
22
|
+
// eslint-disable-next-line @typescript-eslint/no-this-alias
|
|
23
|
+
const self = this;
|
|
24
|
+
return new Promise((resolve, reject) => {
|
|
25
|
+
const cb = (err, result) => {
|
|
26
|
+
Transaction_1.Transaction.release(err).then(() => {
|
|
27
|
+
if (err)
|
|
28
|
+
return reject(err);
|
|
29
|
+
resolve(result);
|
|
30
|
+
});
|
|
31
|
+
};
|
|
32
|
+
let transaction = args.shift();
|
|
33
|
+
if (transaction instanceof Transaction_1.Transaction) {
|
|
34
|
+
const updatedTransaction = new Transaction_1.Transaction(this.constructor.name, propertyKey, async () => {
|
|
35
|
+
originalMethod
|
|
36
|
+
.call(updatedTransaction.bindToTransaction(self), ...args)
|
|
37
|
+
.then(resolve)
|
|
38
|
+
.catch(reject);
|
|
39
|
+
}, data.length ? data : undefined);
|
|
40
|
+
transaction.bindTransaction(updatedTransaction);
|
|
41
|
+
transaction.fire();
|
|
42
|
+
}
|
|
43
|
+
else {
|
|
44
|
+
args.unshift(transaction);
|
|
45
|
+
transaction = undefined;
|
|
46
|
+
transaction = new Transaction_1.Transaction(this.constructor.name, propertyKey, () => {
|
|
47
|
+
originalMethod
|
|
48
|
+
.call(transaction.bindToTransaction(self), ...args)
|
|
49
|
+
.then((result) => cb(undefined, result))
|
|
50
|
+
.catch(cb);
|
|
51
|
+
}, data.length ? data : undefined);
|
|
52
|
+
Transaction_1.Transaction.submit(transaction);
|
|
53
|
+
}
|
|
54
|
+
});
|
|
55
|
+
};
|
|
56
|
+
Object.defineProperty(methodWrapper, "name", {
|
|
57
|
+
value: propertyKey,
|
|
58
|
+
});
|
|
59
|
+
descriptor.value = methodWrapper;
|
|
60
|
+
};
|
|
61
|
+
}
|
|
62
|
+
//
|
|
63
|
+
// /**
|
|
64
|
+
// * @summary Sets a class Async method as transactional
|
|
65
|
+
// *
|
|
66
|
+
// * @param {any[]} [metadata] option metadata available to the {@link TransactionLock}
|
|
67
|
+
// *
|
|
68
|
+
// * @function transactionalAsync
|
|
69
|
+
// *
|
|
70
|
+
// * @memberOf module:db-decorators.Decorators.transactions
|
|
71
|
+
// */
|
|
72
|
+
// export function transactionalAsync(...metadata: any[]) {
|
|
73
|
+
// return function (
|
|
74
|
+
// target: any,
|
|
75
|
+
// propertyKey: string,
|
|
76
|
+
// descriptor: PropertyDescriptor,
|
|
77
|
+
// ) {
|
|
78
|
+
// metadasta(getTransactionalKey(TransactionalKeys.TRANSACTIONAL))
|
|
79
|
+
// Reflect.defineMetadata(
|
|
80
|
+
// ,
|
|
81
|
+
// {
|
|
82
|
+
// type: "async",
|
|
83
|
+
// metadata: metadata.length ? metadata : undefined,
|
|
84
|
+
// } as TransactionalMetadata,
|
|
85
|
+
// target,
|
|
86
|
+
// propertyKey,
|
|
87
|
+
// );
|
|
88
|
+
//
|
|
89
|
+
// const originalMethod = descriptor.value;
|
|
90
|
+
//
|
|
91
|
+
// const methodWrapper = function (this: any, ...args: any[]) {
|
|
92
|
+
// const callback: Callback = args.pop();
|
|
93
|
+
// if (!callback || typeof callback !== "function")
|
|
94
|
+
// throw new CriticalError(`Missing Callback`);
|
|
95
|
+
//
|
|
96
|
+
// const cb = (err?: Err, ...args: any[]) => {
|
|
97
|
+
// Transaction.release(err).then((_) => callback(err, ...args));
|
|
98
|
+
// };
|
|
99
|
+
//
|
|
100
|
+
// const self = this;
|
|
101
|
+
//
|
|
102
|
+
// let transaction = args.shift();
|
|
103
|
+
// if (transaction instanceof Transaction) {
|
|
104
|
+
// const updatedTransaction: Transaction = new Transaction(
|
|
105
|
+
// this.constructor.name,
|
|
106
|
+
// propertyKey,
|
|
107
|
+
// () => {
|
|
108
|
+
// try {
|
|
109
|
+
// return originalMethod.call(
|
|
110
|
+
// updatedTransaction.bindToTransaction(self),
|
|
111
|
+
// ...args,
|
|
112
|
+
// callback,
|
|
113
|
+
// );
|
|
114
|
+
// } catch (e: any) {
|
|
115
|
+
// return callback(e);
|
|
116
|
+
// }
|
|
117
|
+
// },
|
|
118
|
+
// metadata.length ? metadata : undefined,
|
|
119
|
+
// );
|
|
120
|
+
//
|
|
121
|
+
// transaction.bindTransaction(updatedTransaction);
|
|
122
|
+
// transaction.fire();
|
|
123
|
+
// } else {
|
|
124
|
+
// args.unshift(transaction);
|
|
125
|
+
// transaction = undefined;
|
|
126
|
+
// transaction = new Transaction(
|
|
127
|
+
// this.constructor.name,
|
|
128
|
+
// propertyKey,
|
|
129
|
+
// () => {
|
|
130
|
+
// try {
|
|
131
|
+
// return originalMethod.call(
|
|
132
|
+
// transaction.bindToTransaction(self),
|
|
133
|
+
// ...args,
|
|
134
|
+
// cb,
|
|
135
|
+
// );
|
|
136
|
+
// } catch (e: any) {
|
|
137
|
+
// return cb(e);
|
|
138
|
+
// }
|
|
139
|
+
// },
|
|
140
|
+
// metadata.length ? metadata : undefined,
|
|
141
|
+
// );
|
|
142
|
+
// Transaction.submit(transaction);
|
|
143
|
+
// }
|
|
144
|
+
// };
|
|
145
|
+
//
|
|
146
|
+
// Object.defineProperty(methodWrapper, "name", {
|
|
147
|
+
// value: propertyKey,
|
|
148
|
+
// });
|
|
149
|
+
// descriptor.value = methodWrapper;
|
|
150
|
+
// };
|
|
151
|
+
// }
|
|
152
|
+
/**
|
|
153
|
+
* @summary Util function to wrap super calls with the transaction when the super's method is also transactional
|
|
154
|
+
*
|
|
155
|
+
* @param {Function} method the super method (must be bound to the proper this), eg: super.create.bind(this)
|
|
156
|
+
* @param {any[]} args the arguments to call the method with
|
|
157
|
+
*
|
|
158
|
+
* @memberOf module:db-decorators.Transaction
|
|
159
|
+
*/
|
|
160
|
+
function transactionalSuperCall(method, ...args) {
|
|
161
|
+
const lock = Transaction_1.Transaction.getLock();
|
|
162
|
+
const currentTransaction = lock.currentTransaction;
|
|
163
|
+
method(currentTransaction, ...args);
|
|
164
|
+
}
|
|
165
|
+
|
|
166
|
+
//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["src/decorators.ts"],"names":[],"mappings":";;AAaA,sCAgEC;AAoGD,wDAIC;AArLD,2CAAgD;AAChD,qDAAgD;AAChD,+CAA4C;AAE5C;;;;;;;;GAQG;AACH,SAAgB,aAAa,CAAC,GAAG,IAAW;IAC1C,OAAO,UACL,MAAW,EACX,WAAmB,EACnB,UAA8B;QAE9B,IAAA,qBAAQ,EAAC,yBAAW,CAAC,GAAG,CAAC,6BAAiB,CAAC,aAAa,CAAC,EAAE,IAAI,CAAC,CAC9D,MAAM,EACN,WAAW,CACZ,CAAC;QAEF,MAAM,cAAc,GAAG,UAAU,CAAC,KAAK,CAAC;QAExC,MAAM,aAAa,GAAG,UAAqB,GAAG,IAAW;YACvD,4DAA4D;YAC5D,MAAM,IAAI,GAAG,IAAI,CAAC;YAClB,OAAO,IAAI,OAAO,CAAM,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;gBAC1C,MAAM,EAAE,GAAG,CAAC,GAAW,EAAE,MAAY,EAAE,EAAE;oBACvC,yBAAW,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,GAAG,EAAE;wBACjC,IAAI,GAAG;4BAAE,OAAO,MAAM,CAAC,GAAG,CAAC,CAAC;wBAC5B,OAAO,CAAC,MAAM,CAAC,CAAC;oBAClB,CAAC,CAAC,CAAC;gBACL,CAAC,CAAC;gBAEF,IAAI,WAAW,GAAG,IAAI,CAAC,KAAK,EAAE,CAAC;gBAC/B,IAAI,WAAW,YAAY,yBAAW,EAAE,CAAC;oBACvC,MAAM,kBAAkB,GAAgB,IAAI,yBAAW,CACrD,IAAI,CAAC,WAAW,CAAC,IAAI,EACrB,WAAW,EACX,KAAK,IAAI,EAAE;wBACT,cAAc;6BACX,IAAI,CAAC,kBAAkB,CAAC,iBAAiB,CAAC,IAAI,CAAC,EAAE,GAAG,IAAI,CAAC;6BACzD,IAAI,CAAC,OAAO,CAAC;6BACb,KAAK,CAAC,MAAM,CAAC,CAAC;oBACnB,CAAC,EACD,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,SAAS,CAC/B,CAAC;oBAEF,WAAW,CAAC,eAAe,CAAC,kBAAkB,CAAC,CAAC;oBAChD,WAAW,CAAC,IAAI,EAAE,CAAC;gBACrB,CAAC;qBAAM,CAAC;oBACN,IAAI,CAAC,OAAO,CAAC,WAAW,CAAC,CAAC;oBAC1B,WAAW,GAAG,SAAS,CAAC;oBACxB,WAAW,GAAG,IAAI,yBAAW,CAC3B,IAAI,CAAC,WAAW,CAAC,IAAI,EACrB,WAAW,EACX,GAAG,EAAE;wBACH,cAAc;6BACX,IAAI,CAAC,WAAW,CAAC,iBAAiB,CAAC,IAAI,CAAC,EAAE,GAAG,IAAI,CAAC;6BAClD,IAAI,CAAC,CAAC,MAAW,EAAE,EAAE,CAAC,EAAE,CAAC,SAAS,EAAE,MAAM,CAAC,CAAC;6BAC5C,KAAK,CAAC,EAAE,CAAC,CAAC;oBACf,CAAC,EACD,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,SAAS,CAC/B,CAAC;oBACF,yBAAW,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC;gBAClC,CAAC;YACH,CAAC,CAAC,CAAC;QACL,CAAC,CAAC;QAEF,MAAM,CAAC,cAAc,CAAC,aAAa,EAAE,MAAM,EAAE;YAC3C,KAAK,EAAE,WAAW;SACnB,CAAC,CAAC;QACH,UAAU,CAAC,KAAK,GAAG,aAAa,CAAC;IACnC,CAAC,CAAC;AACJ,CAAC;AACD,EAAE;AACF,MAAM;AACN,yDAAyD;AACzD,KAAK;AACL,yFAAyF;AACzF,KAAK;AACL,kCAAkC;AAClC,KAAK;AACL,4DAA4D;AAC5D,MAAM;AACN,2DAA2D;AAC3D,sBAAsB;AACtB,mBAAmB;AACnB,2BAA2B;AAC3B,sCAAsC;AACtC,QAAQ;AACR,sEAAsE;AACtE,8BAA8B;AAC9B,UAAU;AACV,UAAU;AACV,yBAAyB;AACzB,4DAA4D;AAC5D,oCAAoC;AACpC,gBAAgB;AAChB,qBAAqB;AACrB,SAAS;AACT,EAAE;AACF,+CAA+C;AAC/C,EAAE;AACF,mEAAmE;AACnE,+CAA+C;AAC/C,yDAAyD;AACzD,uDAAuD;AACvD,EAAE;AACF,oDAAoD;AACpD,wEAAwE;AACxE,WAAW;AACX,EAAE;AACF,2BAA2B;AAC3B,EAAE;AACF,wCAAwC;AACxC,kDAAkD;AAClD,mEAAmE;AACnE,mCAAmC;AACnC,yBAAyB;AACzB,oBAAoB;AACpB,oBAAoB;AACpB,4CAA4C;AAC5C,8DAA8D;AAC9D,2BAA2B;AAC3B,4BAA4B;AAC5B,mBAAmB;AACnB,iCAAiC;AACjC,oCAAoC;AACpC,gBAAgB;AAChB,eAAe;AACf,oDAAoD;AACpD,aAAa;AACb,EAAE;AACF,2DAA2D;AAC3D,8BAA8B;AAC9B,iBAAiB;AACjB,qCAAqC;AACrC,mCAAmC;AACnC,yCAAyC;AACzC,mCAAmC;AACnC,yBAAyB;AACzB,oBAAoB;AACpB,oBAAoB;AACpB,4CAA4C;AAC5C,uDAAuD;AACvD,2BAA2B;AAC3B,sBAAsB;AACtB,mBAAmB;AACnB,iCAAiC;AACjC,8BAA8B;AAC9B,gBAAgB;AAChB,eAAe;AACf,oDAAoD;AACpD,aAAa;AACb,2CAA2C;AAC3C,UAAU;AACV,SAAS;AACT,EAAE;AACF,qDAAqD;AACrD,4BAA4B;AAC5B,UAAU;AACV,wCAAwC;AACxC,OAAO;AACP,IAAI;AAEJ;;;;;;;GAOG;AACH,SAAgB,sBAAsB,CAAC,MAAW,EAAE,GAAG,IAAS;IAC9D,MAAM,IAAI,GAAG,yBAAW,CAAC,OAAO,EAAE,CAAC;IACnC,MAAM,kBAAkB,GAAG,IAAI,CAAC,kBAAkB,CAAC;IACnD,MAAM,CAAC,kBAAkB,EAAE,GAAG,IAAI,CAAC,CAAC;AACtC,CAAC","file":"decorators.js","sourcesContent":["import { TransactionalKeys } from \"./constants\";\nimport { metadata } from \"@decaf-ts/reflection\";\nimport { Transaction } from \"./Transaction\";\n\n/**\n * @summary Sets a class Async (promise based) method as transactional\n *\n * @param {any[]}  [data] option metadata available to the {@link TransactionLock}\n *\n * @function transactionalPromise\n *\n * @memberOf module:db-decorators.Decorators.transactions\n */\nexport function transactional(...data: any[]) {\n  return function (\n    target: any,\n    propertyKey: string,\n    descriptor: PropertyDescriptor\n  ) {\n    metadata(Transaction.key(TransactionalKeys.TRANSACTIONAL), data)(\n      target,\n      propertyKey\n    );\n\n    const originalMethod = descriptor.value;\n\n    const methodWrapper = function (this: any, ...args: any[]): Promise<any> {\n      // eslint-disable-next-line @typescript-eslint/no-this-alias\n      const self = this;\n      return new Promise<any>((resolve, reject) => {\n        const cb = (err?: Error, result?: any) => {\n          Transaction.release(err).then(() => {\n            if (err) return reject(err);\n            resolve(result);\n          });\n        };\n\n        let transaction = args.shift();\n        if (transaction instanceof Transaction) {\n          const updatedTransaction: Transaction = new Transaction(\n            this.constructor.name,\n            propertyKey,\n            async () => {\n              originalMethod\n                .call(updatedTransaction.bindToTransaction(self), ...args)\n                .then(resolve)\n                .catch(reject);\n            },\n            data.length ? data : undefined\n          );\n\n          transaction.bindTransaction(updatedTransaction);\n          transaction.fire();\n        } else {\n          args.unshift(transaction);\n          transaction = undefined;\n          transaction = new Transaction(\n            this.constructor.name,\n            propertyKey,\n            () => {\n              originalMethod\n                .call(transaction.bindToTransaction(self), ...args)\n                .then((result: any) => cb(undefined, result))\n                .catch(cb);\n            },\n            data.length ? data : undefined\n          );\n          Transaction.submit(transaction);\n        }\n      });\n    };\n\n    Object.defineProperty(methodWrapper, \"name\", {\n      value: propertyKey,\n    });\n    descriptor.value = methodWrapper;\n  };\n}\n//\n// /**\n//  * @summary Sets a class Async method as transactional\n//  *\n//  * @param {any[]}  [metadata] option metadata available to the {@link TransactionLock}\n//  *\n//  * @function transactionalAsync\n//  *\n//  * @memberOf module:db-decorators.Decorators.transactions\n//  */\n// export function transactionalAsync(...metadata: any[]) {\n//   return function (\n//     target: any,\n//     propertyKey: string,\n//     descriptor: PropertyDescriptor,\n//   ) {\n//     metadasta(getTransactionalKey(TransactionalKeys.TRANSACTIONAL))\n//     Reflect.defineMetadata(\n//       ,\n//       {\n//         type: \"async\",\n//         metadata: metadata.length ? metadata : undefined,\n//       } as TransactionalMetadata,\n//       target,\n//       propertyKey,\n//     );\n//\n//     const originalMethod = descriptor.value;\n//\n//     const methodWrapper = function (this: any, ...args: any[]) {\n//       const callback: Callback = args.pop();\n//       if (!callback || typeof callback !== \"function\")\n//         throw new CriticalError(`Missing Callback`);\n//\n//       const cb = (err?: Err, ...args: any[]) => {\n//         Transaction.release(err).then((_) => callback(err, ...args));\n//       };\n//\n//       const self = this;\n//\n//       let transaction = args.shift();\n//       if (transaction instanceof Transaction) {\n//         const updatedTransaction: Transaction = new Transaction(\n//           this.constructor.name,\n//           propertyKey,\n//           () => {\n//             try {\n//               return originalMethod.call(\n//                 updatedTransaction.bindToTransaction(self),\n//                 ...args,\n//                 callback,\n//               );\n//             } catch (e: any) {\n//               return callback(e);\n//             }\n//           },\n//           metadata.length ? metadata : undefined,\n//         );\n//\n//         transaction.bindTransaction(updatedTransaction);\n//         transaction.fire();\n//       } else {\n//         args.unshift(transaction);\n//         transaction = undefined;\n//         transaction = new Transaction(\n//           this.constructor.name,\n//           propertyKey,\n//           () => {\n//             try {\n//               return originalMethod.call(\n//                 transaction.bindToTransaction(self),\n//                 ...args,\n//                 cb,\n//               );\n//             } catch (e: any) {\n//               return cb(e);\n//             }\n//           },\n//           metadata.length ? metadata : undefined,\n//         );\n//         Transaction.submit(transaction);\n//       }\n//     };\n//\n//     Object.defineProperty(methodWrapper, \"name\", {\n//       value: propertyKey,\n//     });\n//     descriptor.value = methodWrapper;\n//   };\n// }\n\n/**\n * @summary Util function to wrap super calls with the transaction when the super's method is also transactional\n *\n * @param {Function} method the super method (must be bound to the proper this), eg: super.create.bind(this)\n * @param {any[]} args the arguments to call the method with\n *\n * @memberOf module:db-decorators.Transaction\n */\nexport function transactionalSuperCall(method: any, ...args: any) {\n  const lock = Transaction.getLock();\n  const currentTransaction = lock.currentTransaction;\n  method(currentTransaction, ...args);\n}\n"]}
|
package/lib/decorators.d.ts
CHANGED
|
@@ -1,10 +1,3 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @summary gets the transactions reflections key
|
|
3
|
-
* @function getRepoKey
|
|
4
|
-
* @param {string} key
|
|
5
|
-
* @memberOf module:db-decorators.Transactions
|
|
6
|
-
* */
|
|
7
|
-
export declare const getTransactionalKey: (key: string) => string;
|
|
8
1
|
/**
|
|
9
2
|
* @summary Sets a class Async (promise based) method as transactional
|
|
10
3
|
*
|
package/lib/esm/Transaction.d.ts
CHANGED
|
@@ -76,4 +76,11 @@ export declare class Transaction {
|
|
|
76
76
|
* @param {boolean} [withLog] defaults to true
|
|
77
77
|
*/
|
|
78
78
|
toString(withId?: boolean, withLog?: boolean): string;
|
|
79
|
+
/**
|
|
80
|
+
* @summary gets the transactions reflections key
|
|
81
|
+
* @function getRepoKey
|
|
82
|
+
* @param {string} key
|
|
83
|
+
* @memberOf module:db-decorators.Transactions
|
|
84
|
+
* */
|
|
85
|
+
static key(key: string): string;
|
|
79
86
|
}
|
package/lib/esm/Transaction.js
CHANGED
|
@@ -1,3 +1,148 @@
|
|
|
1
|
-
import{getAllProperties,getClassDecorators}from"@decaf-ts/reflection";
|
|
2
|
-
|
|
3
|
-
|
|
1
|
+
import { getAllProperties, getClassDecorators } from "@decaf-ts/reflection";
|
|
2
|
+
import { SyncronousLock } from "./locks/SyncronousLock";
|
|
3
|
+
import { DBKeys, getAllPropertyDecoratorsRecursive, } from "@decaf-ts/db-decorators";
|
|
4
|
+
import { getObjectName } from "./utils";
|
|
5
|
+
import { TransactionalKeys } from "./constants";
|
|
6
|
+
export class Transaction {
|
|
7
|
+
constructor(source, method, action, metadata) {
|
|
8
|
+
this.id = Date.now();
|
|
9
|
+
this.action = action;
|
|
10
|
+
this.method = method;
|
|
11
|
+
this.log = [[this.id, source, method].join(" | ")];
|
|
12
|
+
this.source = source;
|
|
13
|
+
this.metadata = metadata;
|
|
14
|
+
}
|
|
15
|
+
/**
|
|
16
|
+
* @summary Pushes a transaction to que queue and waits its resolution
|
|
17
|
+
*
|
|
18
|
+
* @param {any} issuer any class. will be used as this when calling the callbackMethod
|
|
19
|
+
* @param {Function} callbackMethod callback function containing the transaction. will be called with the issuear as this
|
|
20
|
+
* @param {any[]} args arguments to pass to the method. Last one must be the callback
|
|
21
|
+
*/
|
|
22
|
+
static push(issuer, callbackMethod, ...args) {
|
|
23
|
+
const callback = args.pop();
|
|
24
|
+
if (!callback || typeof callback !== "function")
|
|
25
|
+
throw new Error("Missing callback");
|
|
26
|
+
const cb = (err, ...args) => {
|
|
27
|
+
this.getLock()
|
|
28
|
+
.release(err)
|
|
29
|
+
.then(() => callback(err, ...args));
|
|
30
|
+
};
|
|
31
|
+
const transaction = new Transaction(issuer.constructor.name, callbackMethod.name ? getObjectName(callbackMethod) : "Anonymous", () => {
|
|
32
|
+
return callbackMethod.call(transaction.bindToTransaction(issuer), ...args, cb);
|
|
33
|
+
});
|
|
34
|
+
this.getLock().submit(transaction);
|
|
35
|
+
}
|
|
36
|
+
/**
|
|
37
|
+
* @summary Sets the lock to be used
|
|
38
|
+
* @param lock
|
|
39
|
+
*/
|
|
40
|
+
static setLock(lock) {
|
|
41
|
+
this.lock = lock;
|
|
42
|
+
}
|
|
43
|
+
/**
|
|
44
|
+
* @summary gets the lock
|
|
45
|
+
*/
|
|
46
|
+
static getLock() {
|
|
47
|
+
if (!this.lock)
|
|
48
|
+
this.lock = new SyncronousLock();
|
|
49
|
+
return this.lock;
|
|
50
|
+
}
|
|
51
|
+
/**
|
|
52
|
+
* @summary submits a transaction
|
|
53
|
+
* @param {Transaction} transaction
|
|
54
|
+
*/
|
|
55
|
+
static submit(transaction) {
|
|
56
|
+
this.getLock().submit(transaction);
|
|
57
|
+
}
|
|
58
|
+
/**
|
|
59
|
+
* @summary releases the lock
|
|
60
|
+
* @param {Err} err
|
|
61
|
+
*/
|
|
62
|
+
static async release(err) {
|
|
63
|
+
return this.getLock().release(err);
|
|
64
|
+
}
|
|
65
|
+
/**
|
|
66
|
+
* @summary retrieves the metadata for the transaction
|
|
67
|
+
*/
|
|
68
|
+
getMetadata() {
|
|
69
|
+
return this.metadata ? [...this.metadata] : undefined;
|
|
70
|
+
}
|
|
71
|
+
/**
|
|
72
|
+
* @summary Binds a new operation to the current transaction
|
|
73
|
+
* @param {Transaction} nextTransaction
|
|
74
|
+
*/
|
|
75
|
+
bindTransaction(nextTransaction) {
|
|
76
|
+
// all(`Binding the {0} to {1}`, nextTransaction, this);
|
|
77
|
+
this.log.push(...nextTransaction.log);
|
|
78
|
+
nextTransaction.bindTransaction = this.bindToTransaction.bind(this);
|
|
79
|
+
nextTransaction.bindToTransaction = this.bindToTransaction.bind(this);
|
|
80
|
+
this.action = nextTransaction.action;
|
|
81
|
+
}
|
|
82
|
+
/**
|
|
83
|
+
* @summary Binds the Transactional Decorated Object to the transaction
|
|
84
|
+
* @description by having all {@link transactional} decorated
|
|
85
|
+
* methods always pass the current Transaction as an argument
|
|
86
|
+
*
|
|
87
|
+
* @param {any} obj
|
|
88
|
+
* @return {any} the bound {@param obj}
|
|
89
|
+
*/
|
|
90
|
+
bindToTransaction(obj) {
|
|
91
|
+
const transactionalMethods = getAllPropertyDecoratorsRecursive(obj, undefined, TransactionalKeys.REFLECT);
|
|
92
|
+
if (!transactionalMethods)
|
|
93
|
+
return obj;
|
|
94
|
+
// eslint-disable-next-line @typescript-eslint/no-this-alias
|
|
95
|
+
const self = this;
|
|
96
|
+
const boundObj = getAllProperties(obj).reduce((accum, k) => {
|
|
97
|
+
if (Object.keys(transactionalMethods).indexOf(k) !== -1 &&
|
|
98
|
+
transactionalMethods[k].find((o) => o.key === TransactionalKeys.TRANSACTIONAL))
|
|
99
|
+
accum[k] = (...args) => obj[k].call(obj.__originalObj || obj, self, ...args);
|
|
100
|
+
else if (k === "clazz" || k === "constructor")
|
|
101
|
+
accum[k] = obj[k];
|
|
102
|
+
else if (typeof obj[k] === "function")
|
|
103
|
+
accum[k] = obj[k].bind(obj.__originalObj || obj);
|
|
104
|
+
else if (typeof obj[k] === "object" && obj[k].constructor) {
|
|
105
|
+
const decs = getClassDecorators(TransactionalKeys.REFLECT, obj[k]);
|
|
106
|
+
if (decs.find((e) => e.key === TransactionalKeys.TRANSACTIONAL))
|
|
107
|
+
accum[k] = self.bindToTransaction(obj[k]);
|
|
108
|
+
else
|
|
109
|
+
accum[k] = obj[k];
|
|
110
|
+
}
|
|
111
|
+
else
|
|
112
|
+
accum[k] = obj[k];
|
|
113
|
+
return accum;
|
|
114
|
+
}, {});
|
|
115
|
+
boundObj[DBKeys.ORIGINAL] = obj[DBKeys.ORIGINAL] || obj;
|
|
116
|
+
boundObj.toString = () => getObjectName(boundObj[DBKeys.ORIGINAL]) +
|
|
117
|
+
" proxy for transaction " +
|
|
118
|
+
this.id;
|
|
119
|
+
return boundObj;
|
|
120
|
+
}
|
|
121
|
+
/**
|
|
122
|
+
* @summary Fires the Transaction
|
|
123
|
+
*/
|
|
124
|
+
fire() {
|
|
125
|
+
if (!this.action)
|
|
126
|
+
throw new Error(`Missing the method`);
|
|
127
|
+
return this.action();
|
|
128
|
+
}
|
|
129
|
+
/**
|
|
130
|
+
* @summary toString override
|
|
131
|
+
* @param {boolean} [withId] defaults to true
|
|
132
|
+
* @param {boolean} [withLog] defaults to true
|
|
133
|
+
*/
|
|
134
|
+
toString(withId = true, withLog = false) {
|
|
135
|
+
return `${withId ? `[${this.id}]` : ""}[Transaction][${this.source}.${this.method}${withLog ? `]\nTransaction Log:\n${this.log.join("\n")}` : "]"}`;
|
|
136
|
+
}
|
|
137
|
+
/**
|
|
138
|
+
* @summary gets the transactions reflections key
|
|
139
|
+
* @function getRepoKey
|
|
140
|
+
* @param {string} key
|
|
141
|
+
* @memberOf module:db-decorators.Transactions
|
|
142
|
+
* */
|
|
143
|
+
static key(key) {
|
|
144
|
+
return TransactionalKeys.REFLECT + key;
|
|
145
|
+
}
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["src/Transaction.ts"],"names":[],"mappings":"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","file":"Transaction.js","sourcesContent":["/**\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 */\nimport { TransactionLock } from \"./interfaces/TransactionLock\";\nimport { getAllProperties, getClassDecorators } 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\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      this.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    this.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    this.getLock().submit(transaction);\n  }\n\n  /**\n   * @summary releases the lock\n   * @param {Err} err\n   */\n  static async release(err?: Error) {\n    return this.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 = getAllProperties(obj).reduce((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 = getClassDecorators(TransactionalKeys.REFLECT, obj[k]);\n        if (decs.find((e) => 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    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"]}
|
package/lib/esm/constants.js
CHANGED
|
@@ -1 +1,6 @@
|
|
|
1
|
-
|
|
1
|
+
export const TransactionalKeys = {
|
|
2
|
+
REFLECT: "model.transactional.",
|
|
3
|
+
TRANSACTIONAL: "transactional",
|
|
4
|
+
};
|
|
5
|
+
|
|
6
|
+
//# sourceMappingURL=data:application/json;charset=utf8;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbInNyYy9jb25zdGFudHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsTUFBTSxDQUFDLE1BQU0saUJBQWlCLEdBQTJCO0lBQ3ZELE9BQU8sRUFBRSxzQkFBc0I7SUFDL0IsYUFBYSxFQUFFLGVBQWU7Q0FDL0IsQ0FBQyIsImZpbGUiOiJjb25zdGFudHMuanMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgY29uc3QgVHJhbnNhY3Rpb25hbEtleXM6IFJlY29yZDxzdHJpbmcsIHN0cmluZz4gPSB7XG4gIFJFRkxFQ1Q6IFwibW9kZWwudHJhbnNhY3Rpb25hbC5cIixcbiAgVFJBTlNBQ1RJT05BTDogXCJ0cmFuc2FjdGlvbmFsXCIsXG59O1xuIl19
|
package/lib/esm/decorators.d.ts
CHANGED
|
@@ -1,10 +1,3 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* @summary gets the transactions reflections key
|
|
3
|
-
* @function getRepoKey
|
|
4
|
-
* @param {string} key
|
|
5
|
-
* @memberOf module:db-decorators.Transactions
|
|
6
|
-
* */
|
|
7
|
-
export declare const getTransactionalKey: (key: string) => string;
|
|
8
1
|
/**
|
|
9
2
|
* @summary Sets a class Async (promise based) method as transactional
|
|
10
3
|
*
|