@decaf-ts/transactional-decorators 0.1.2 → 0.1.4

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 (72) hide show
  1. package/LICENSE.md +17 -152
  2. package/README.md +242 -17
  3. package/dist/transactional-decorators.cjs +2 -544
  4. package/dist/transactional-decorators.cjs.map +1 -0
  5. package/dist/transactional-decorators.js +2 -0
  6. package/dist/transactional-decorators.js.map +1 -0
  7. package/lib/Transaction.cjs +87 -39
  8. package/lib/Transaction.d.ts +86 -38
  9. package/lib/Transaction.js.map +1 -0
  10. package/lib/constants.cjs +14 -1
  11. package/lib/constants.d.ts +13 -0
  12. package/lib/constants.js.map +1 -0
  13. package/lib/decorators.cjs +35 -11
  14. package/lib/decorators.d.ts +34 -10
  15. package/lib/decorators.js.map +1 -0
  16. package/lib/esm/Transaction.d.ts +86 -38
  17. package/lib/esm/Transaction.js +90 -42
  18. package/lib/esm/Transaction.js.map +1 -0
  19. package/lib/esm/constants.d.ts +13 -0
  20. package/lib/esm/constants.js +14 -1
  21. package/lib/esm/constants.js.map +1 -0
  22. package/lib/esm/decorators.d.ts +34 -10
  23. package/lib/esm/decorators.js +37 -13
  24. package/lib/esm/decorators.js.map +1 -0
  25. package/lib/esm/index.d.ts +7 -13
  26. package/lib/esm/index.js +14 -20
  27. package/lib/esm/index.js.map +1 -0
  28. package/lib/esm/interfaces/TransactionLock.d.ts +13 -11
  29. package/lib/esm/interfaces/TransactionLock.js +1 -1
  30. package/lib/esm/interfaces/TransactionLock.js.map +1 -0
  31. package/lib/esm/interfaces/index.d.ts +5 -0
  32. package/lib/esm/interfaces/index.js +7 -2
  33. package/lib/esm/interfaces/index.js.map +1 -0
  34. package/lib/esm/locks/Lock.d.ts +13 -5
  35. package/lib/esm/locks/Lock.js +14 -6
  36. package/lib/esm/locks/Lock.js.map +1 -0
  37. package/lib/esm/locks/SyncronousLock.d.ts +3 -4
  38. package/lib/esm/locks/SyncronousLock.js +14 -2
  39. package/lib/esm/locks/SyncronousLock.js.map +1 -0
  40. package/lib/esm/locks/index.d.ts +5 -0
  41. package/lib/esm/locks/index.js +8 -3
  42. package/lib/esm/locks/index.js.map +1 -0
  43. package/lib/esm/types.d.ts +13 -3
  44. package/lib/esm/types.js +1 -1
  45. package/lib/esm/types.js.map +1 -0
  46. package/lib/esm/utils.js +1 -1
  47. package/lib/esm/utils.js.map +1 -0
  48. package/lib/index.cjs +8 -14
  49. package/lib/index.d.ts +7 -13
  50. package/lib/index.js.map +1 -0
  51. package/lib/interfaces/TransactionLock.cjs +1 -1
  52. package/lib/interfaces/TransactionLock.d.ts +13 -11
  53. package/lib/interfaces/TransactionLock.js.map +1 -0
  54. package/lib/interfaces/index.cjs +6 -1
  55. package/lib/interfaces/index.d.ts +5 -0
  56. package/lib/interfaces/index.js.map +1 -0
  57. package/lib/locks/Lock.cjs +14 -6
  58. package/lib/locks/Lock.d.ts +13 -5
  59. package/lib/locks/Lock.js.map +1 -0
  60. package/lib/locks/SyncronousLock.cjs +13 -1
  61. package/lib/locks/SyncronousLock.d.ts +3 -4
  62. package/lib/locks/SyncronousLock.js.map +1 -0
  63. package/lib/locks/index.cjs +6 -1
  64. package/lib/locks/index.d.ts +5 -0
  65. package/lib/locks/index.js.map +1 -0
  66. package/lib/types.cjs +1 -1
  67. package/lib/types.d.ts +13 -3
  68. package/lib/types.js.map +1 -0
  69. package/lib/utils.cjs +1 -1
  70. package/lib/utils.js.map +1 -0
  71. package/package.json +24 -25
  72. package/dist/transactional-decorators.esm.cjs +0 -533
@@ -1,544 +1,2 @@
1
- (function (global, factory) {
2
- typeof exports === 'object' && typeof module !== 'undefined' ? factory(exports, require('@decaf-ts/reflection'), require('@decaf-ts/db-decorators')) :
3
- typeof define === 'function' && define.amd ? define(['exports', '@decaf-ts/reflection', '@decaf-ts/db-decorators'], factory) :
4
- (global = typeof globalThis !== 'undefined' ? globalThis : global || self, factory(global["transactional-decorators"] = {}, global.reflection, global.dbDecorators));
5
- })(this, (function (exports, reflection, dbDecorators) { 'use strict';
6
-
7
- /**
8
- * @summary Simple Promise based Lock class
9
- *
10
- * @class Lock
11
- * @category Transactions
12
- */
13
- class Lock {
14
- constructor() {
15
- this.queue = [];
16
- this.locked = false;
17
- }
18
- /**
19
- * @summary executes when lock is available
20
- * @param {Function} func
21
- */
22
- async execute(func) {
23
- await this.acquire();
24
- let result;
25
- try {
26
- result = await Promise.resolve(func());
27
- }
28
- catch (e) {
29
- this.release();
30
- throw e;
31
- }
32
- this.release();
33
- return result;
34
- }
35
- /**
36
- * @summary waits to acquire the lock
37
- * @param {string} [issuer]
38
- */
39
- async acquire() {
40
- // eslint-disable-next-line @typescript-eslint/no-this-alias
41
- const self = this;
42
- if (self.locked) {
43
- return new Promise((resolve) => self.queue.push(resolve));
44
- }
45
- else {
46
- self.locked = true;
47
- return Promise.resolve();
48
- }
49
- }
50
- /**
51
- * @summary releases the lock
52
- */
53
- release() {
54
- // eslint-disable-next-line @typescript-eslint/no-this-alias
55
- const self = this;
56
- const next = self.queue.shift();
57
- if (next) {
58
- if (typeof globalThis.window === "undefined")
59
- globalThis.process.nextTick(next); // if you are on node
60
- else
61
- setTimeout(next, 0); // if you are in the browser
62
- }
63
- else {
64
- self.locked = false;
65
- }
66
- }
67
- }
68
-
69
- class SyncronousLock {
70
- constructor(counter = 1, onBegin, onEnd) {
71
- this.currentTransaction = undefined;
72
- this.lock = new Lock();
73
- this.counter = counter;
74
- this.pendingTransactions = [];
75
- this.onBegin = onBegin;
76
- this.onEnd = onEnd;
77
- }
78
- /**
79
- * @summary Submits a transaction to be processed
80
- * @param {Transaction} transaction
81
- */
82
- submit(transaction) {
83
- // eslint-disable-next-line @typescript-eslint/no-this-alias
84
- const self = this;
85
- self.lock.acquire().then(() => {
86
- if (self.currentTransaction &&
87
- self.currentTransaction.id === transaction.id) {
88
- self.lock.release();
89
- return transaction.fire();
90
- }
91
- if (self.counter > 0) {
92
- self.counter--;
93
- self.lock.release();
94
- return self.fireTransaction(transaction);
95
- }
96
- else {
97
- self.pendingTransactions.push(transaction);
98
- self.lock.release();
99
- }
100
- });
101
- }
102
- /**
103
- * @summary Executes a transaction
104
- *
105
- * @param {Transaction} transaction
106
- * @private
107
- */
108
- fireTransaction(transaction) {
109
- // eslint-disable-next-line @typescript-eslint/no-this-alias
110
- const self = this;
111
- self.lock.acquire().then(() => {
112
- self.currentTransaction = transaction;
113
- self.lock.release();
114
- if (self.onBegin)
115
- self.onBegin().then(() => {
116
- // all.call(
117
- // self,
118
- // `Firing transaction {0}. {1} remaining...`,
119
- // transaction.id,
120
- // this.pendingTransactions.length,
121
- // );
122
- transaction.fire();
123
- });
124
- else {
125
- // all.call(
126
- // self,
127
- // `Firing transaction {0}. {1} remaining...`,
128
- // transaction.id,
129
- // this.pendingTransactions.length,
130
- // );
131
- transaction.fire();
132
- }
133
- });
134
- }
135
- /**
136
- * @summary Releases The lock after the conclusion of a transaction
137
- */
138
- async release(err) {
139
- // eslint-disable-next-line @typescript-eslint/no-this-alias
140
- const self = this;
141
- return new Promise((resolve) => {
142
- self.lock.acquire().then(() => {
143
- if (!self.currentTransaction)
144
- console.warn("Trying to release an unexisting transaction. should never happen...");
145
- // debug.call(
146
- // self,
147
- // "Releasing transaction: {0}",
148
- // self.currentTransaction?.toString(true, true),
149
- // );
150
- self.currentTransaction = undefined;
151
- self.lock.release();
152
- const afterConclusionCB = () => {
153
- self.lock.acquire().then(() => {
154
- if (self.pendingTransactions.length > 0) {
155
- const transaction = self.pendingTransactions.shift();
156
- const cb = () => self.fireTransaction(transaction);
157
- //
158
- // all(
159
- // `Releasing Transaction Lock on transaction {0}`,
160
- // transaction.id,
161
- // );
162
- if (typeof globalThis.window ===
163
- "undefined")
164
- globalThis.process.nextTick(cb); // if you are on node
165
- else
166
- setTimeout(cb, 0); // if you are in the browser
167
- }
168
- else {
169
- self.counter++;
170
- }
171
- self.lock.release();
172
- resolve();
173
- });
174
- };
175
- if (self.onEnd)
176
- self.onEnd(err).then(() => afterConclusionCB());
177
- else
178
- afterConclusionCB();
179
- });
180
- });
181
- }
182
- }
183
-
184
- const TransactionalKeys = {
185
- REFLECT: "model.transactional.",
186
- TRANSACTIONAL: "transactional",
187
- };
188
-
189
- function getObjectName(obj) {
190
- if (!obj)
191
- return;
192
- if (typeof obj === "string")
193
- return obj;
194
- if (obj.constructor &&
195
- obj.constructor.name &&
196
- ["Function", "Object"].indexOf(obj.constructor.name) === -1)
197
- return obj.constructor.name;
198
- if (typeof obj === "function" && obj.name)
199
- return obj.name;
200
- return obj.toString();
201
- }
202
-
203
- /**
204
- * @summary Transaction Class
205
- *
206
- * @param {string} source
207
- * @param {string} [method]
208
- * @param {function(): void} [action]
209
- * @param {any[]} [metadata]
210
- *
211
- * @class Transaction
212
- *
213
- * @category Transactions
214
- */
215
- class Transaction {
216
- constructor(source, method, action, metadata) {
217
- this.id = Date.now();
218
- this.action = action;
219
- this.method = method;
220
- this.log = [[this.id, source, method].join(" | ")];
221
- this.source = source;
222
- this.metadata = metadata;
223
- }
224
- /**
225
- * @summary Pushes a transaction to que queue and waits its resolution
226
- *
227
- * @param {any} issuer any class. will be used as this when calling the callbackMethod
228
- * @param {Function} callbackMethod callback function containing the transaction. will be called with the issuear as this
229
- * @param {any[]} args arguments to pass to the method. Last one must be the callback
230
- */
231
- static push(issuer, callbackMethod, ...args) {
232
- const callback = args.pop();
233
- if (!callback || typeof callback !== "function")
234
- throw new Error("Missing callback");
235
- const cb = (err, ...args) => {
236
- Transaction.getLock()
237
- .release(err)
238
- .then(() => callback(err, ...args));
239
- };
240
- const transaction = new Transaction(issuer.constructor.name, callbackMethod.name ? getObjectName(callbackMethod) : "Anonymous", () => {
241
- return callbackMethod.call(transaction.bindToTransaction(issuer), ...args, cb);
242
- });
243
- Transaction.getLock().submit(transaction);
244
- }
245
- /**
246
- * @summary Sets the lock to be used
247
- * @param lock
248
- */
249
- static setLock(lock) {
250
- this.lock = lock;
251
- }
252
- /**
253
- * @summary gets the lock
254
- */
255
- static getLock() {
256
- if (!this.lock)
257
- this.lock = new SyncronousLock();
258
- return this.lock;
259
- }
260
- /**
261
- * @summary submits a transaction
262
- * @param {Transaction} transaction
263
- */
264
- static submit(transaction) {
265
- Transaction.getLock().submit(transaction);
266
- }
267
- /**
268
- * @summary releases the lock
269
- * @param {Err} err
270
- */
271
- static async release(err) {
272
- return Transaction.getLock().release(err);
273
- }
274
- /**
275
- * @summary retrieves the metadata for the transaction
276
- */
277
- getMetadata() {
278
- return this.metadata ? [...this.metadata] : undefined;
279
- }
280
- /**
281
- * @summary Binds a new operation to the current transaction
282
- * @param {Transaction} nextTransaction
283
- */
284
- bindTransaction(nextTransaction) {
285
- // all(`Binding the {0} to {1}`, nextTransaction, this);
286
- this.log.push(...nextTransaction.log);
287
- nextTransaction.bindTransaction = this.bindToTransaction.bind(this);
288
- nextTransaction.bindToTransaction = this.bindToTransaction.bind(this);
289
- this.action = nextTransaction.action;
290
- }
291
- /**
292
- * @summary Binds the Transactional Decorated Object to the transaction
293
- * @description by having all {@link transactional} decorated
294
- * methods always pass the current Transaction as an argument
295
- *
296
- * @param {any} obj
297
- * @return {any} the bound {@param obj}
298
- */
299
- bindToTransaction(obj) {
300
- const transactionalMethods = dbDecorators.getAllPropertyDecoratorsRecursive(obj, undefined, TransactionalKeys.REFLECT);
301
- if (!transactionalMethods)
302
- return obj;
303
- // eslint-disable-next-line @typescript-eslint/no-this-alias
304
- const self = this;
305
- const boundObj = reflection.Reflection.getAllProperties(obj).reduce((accum, k) => {
306
- if (Object.keys(transactionalMethods).indexOf(k) !== -1 &&
307
- transactionalMethods[k].find((o) => o.key === TransactionalKeys.TRANSACTIONAL))
308
- accum[k] = (...args) => obj[k].call(obj.__originalObj || obj, self, ...args);
309
- else if (k === "clazz" || k === "constructor")
310
- accum[k] = obj[k];
311
- else if (typeof obj[k] === "function")
312
- accum[k] = obj[k].bind(obj.__originalObj || obj);
313
- else if (typeof obj[k] === "object" && obj[k].constructor) {
314
- const decs = reflection.Reflection.getClassDecorators(TransactionalKeys.REFLECT, obj[k]);
315
- if (decs.find((e) => e.key === TransactionalKeys.TRANSACTIONAL))
316
- accum[k] = self.bindToTransaction(obj[k]);
317
- else
318
- accum[k] = obj[k];
319
- }
320
- else
321
- accum[k] = obj[k];
322
- return accum;
323
- }, {});
324
- boundObj[dbDecorators.DBKeys.ORIGINAL] = obj[dbDecorators.DBKeys.ORIGINAL] || obj;
325
- boundObj.toString = () => getObjectName(boundObj[dbDecorators.DBKeys.ORIGINAL]) +
326
- " proxy for transaction " +
327
- this.id;
328
- return boundObj;
329
- }
330
- /**
331
- * @summary Fires the Transaction
332
- */
333
- fire() {
334
- if (!this.action)
335
- throw new Error(`Missing the method`);
336
- return this.action();
337
- }
338
- /**
339
- * @summary toString override
340
- * @param {boolean} [withId] defaults to true
341
- * @param {boolean} [withLog] defaults to true
342
- */
343
- toString(withId = true, withLog = false) {
344
- return `${withId ? `[${this.id}]` : ""}[Transaction][${this.source}.${this.method}${withLog ? `]\nTransaction Log:\n${this.log.join("\n")}` : "]"}`;
345
- }
346
- /**
347
- * @summary gets the transactions reflections key
348
- * @function getRepoKey
349
- * @param {string} key
350
- * @memberOf module:db-decorators.Transactions
351
- * */
352
- static key(key) {
353
- return TransactionalKeys.REFLECT + key;
354
- }
355
- }
356
-
357
- /**
358
- * @summary Sets a class Async method as transactional
359
- *
360
- * @param {any[]} [data] option metadata available to the {@link TransactionLock}
361
- *
362
- * @function transactional
363
- *
364
- * @memberOf module:db-decorators.Decorators.transactions
365
- */
366
- function transactional(...data) {
367
- return function (target, propertyKey, descriptor) {
368
- if (!descriptor)
369
- throw new dbDecorators.InternalError("Missing descriptor. Should be impossible");
370
- reflection.metadata(Transaction.key(TransactionalKeys.TRANSACTIONAL), data)(target, propertyKey);
371
- const originalMethod = descriptor.value;
372
- const methodWrapper = function (...args) {
373
- // eslint-disable-next-line @typescript-eslint/no-this-alias
374
- const self = this;
375
- return new Promise((resolve, reject) => {
376
- const cb = (err, result) => {
377
- Transaction.release(err).then(() => {
378
- if (err)
379
- return reject(err);
380
- resolve(result);
381
- });
382
- };
383
- let transaction = args.shift();
384
- if (transaction instanceof Transaction) {
385
- const updatedTransaction = new Transaction(this.constructor.name, propertyKey, async () => {
386
- originalMethod
387
- .call(updatedTransaction.bindToTransaction(self), ...args)
388
- .then(resolve)
389
- .catch(reject);
390
- }, data.length ? data : undefined);
391
- transaction.bindTransaction(updatedTransaction);
392
- transaction.fire();
393
- }
394
- else {
395
- args.unshift(transaction);
396
- transaction = new Transaction(this.constructor.name, propertyKey, () => {
397
- originalMethod
398
- .call(transaction.bindToTransaction(self), ...args)
399
- .then((result) => cb(undefined, result))
400
- .catch(cb);
401
- }, data.length ? data : undefined);
402
- Transaction.submit(transaction);
403
- }
404
- });
405
- };
406
- Object.defineProperty(methodWrapper, "name", {
407
- value: propertyKey,
408
- });
409
- descriptor.value = methodWrapper;
410
- };
411
- }
412
- //
413
- // /**
414
- // * @summary Sets a class Async method as transactional
415
- // *
416
- // * @param {any[]} [metadata] option metadata available to the {@link TransactionLock}
417
- // *
418
- // * @function transactionalAsync
419
- // *
420
- // * @memberOf module:db-decorators.Decorators.transactions
421
- // */
422
- // export function transactionalAsync(...metadata: any[]) {
423
- // return function (
424
- // target: any,
425
- // propertyKey: string,
426
- // descriptor: PropertyDescriptor,
427
- // ) {
428
- // metadasta(getTransactionalKey(TransactionalKeys.TRANSACTIONAL))
429
- // Reflect.defineMetadata(
430
- // ,
431
- // {
432
- // type: "async",
433
- // metadata: metadata.length ? metadata : undefined,
434
- // } as TransactionalMetadata,
435
- // target,
436
- // propertyKey,
437
- // );
438
- //
439
- // const originalMethod = descriptor.value;
440
- //
441
- // const methodWrapper = function (this: any, ...args: any[]) {
442
- // const callback: Callback = args.pop();
443
- // if (!callback || typeof callback !== "function")
444
- // throw new CriticalError(`Missing Callback`);
445
- //
446
- // const cb = (err?: Err, ...args: any[]) => {
447
- // Transaction.release(err).then((_) => callback(err, ...args));
448
- // };
449
- //
450
- // const self = this;
451
- //
452
- // let transaction = args.shift();
453
- // if (transaction instanceof Transaction) {
454
- // const updatedTransaction: Transaction = new Transaction(
455
- // this.constructor.name,
456
- // propertyKey,
457
- // () => {
458
- // try {
459
- // return originalMethod.call(
460
- // updatedTransaction.bindToTransaction(self),
461
- // ...args,
462
- // callback,
463
- // );
464
- // } catch (e: any) {
465
- // return callback(e);
466
- // }
467
- // },
468
- // metadata.length ? metadata : undefined,
469
- // );
470
- //
471
- // transaction.bindTransaction(updatedTransaction);
472
- // transaction.fire();
473
- // } else {
474
- // args.unshift(transaction);
475
- // transaction = undefined;
476
- // transaction = new Transaction(
477
- // this.constructor.name,
478
- // propertyKey,
479
- // () => {
480
- // try {
481
- // return originalMethod.call(
482
- // transaction.bindToTransaction(self),
483
- // ...args,
484
- // cb,
485
- // );
486
- // } catch (e: any) {
487
- // return cb(e);
488
- // }
489
- // },
490
- // metadata.length ? metadata : undefined,
491
- // );
492
- // Transaction.submit(transaction);
493
- // }
494
- // };
495
- //
496
- // Object.defineProperty(methodWrapper, "name", {
497
- // value: propertyKey,
498
- // });
499
- // descriptor.value = methodWrapper;
500
- // };
501
- // }
502
- /**
503
- * @summary Util function to wrap super calls with the transaction when the super's method is also transactional
504
- *
505
- * @param {Function} method the super method (must be bound to the proper this), eg: super.create.bind(this)
506
- * @param {any[]} args the arguments to call the method with
507
- *
508
- * @memberOf module:db-decorators.Transaction
509
- */
510
- function transactionalSuperCall(method, ...args) {
511
- const lock = Transaction.getLock();
512
- const currentTransaction = lock.currentTransaction;
513
- return method(currentTransaction, ...args);
514
- }
515
-
516
- /**
517
- * @summary Module summary
518
- * @description Module description
519
- * @module ts-workspace
520
- */
521
- /**
522
- * @summary Namespace summary
523
- * @description Namespace description
524
- * @namespace Namespace
525
- * @memberOf module:ts-workspace
526
- */
527
- /**
528
- * @summary stores the current package version
529
- * @description this is how you should document a constant
530
- * @const VERSION
531
- * @memberOf module:ts-workspace
532
- */
533
- const VERSION = "0.1.2";
534
-
535
- exports.Lock = Lock;
536
- exports.SyncronousLock = SyncronousLock;
537
- exports.Transaction = Transaction;
538
- exports.TransactionalKeys = TransactionalKeys;
539
- exports.VERSION = VERSION;
540
- exports.transactional = transactional;
541
- exports.transactionalSuperCall = transactionalSuperCall;
542
-
543
- }));
544
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"transactional-decorators.cjs","sources":["../src/locks/Lock.ts","../src/locks/SyncronousLock.ts","../src/constants.ts","../src/utils.ts","../src/Transaction.ts","../src/decorators.ts","../src/index.ts"],"sourcesContent":["import { LockCallable } from \"../types\";\n\n/**\n * @summary Simple Promise based Lock class\n *\n * @class Lock\n * @category Transactions\n */\nexport class Lock {\n  private queue: LockCallable[] = [];\n  private locked = false;\n\n  /**\n   * @summary executes when lock is available\n   * @param {Function} func\n   */\n  async execute(func: () => any) {\n    await this.acquire();\n    let result: any;\n    try {\n      result = await Promise.resolve(func());\n    } catch (e: any) {\n      this.release();\n      throw e;\n    }\n    this.release();\n    return result;\n  }\n\n  /**\n   * @summary waits to acquire the lock\n   * @param {string} [issuer]\n   */\n  async acquire(): Promise<void> {\n    // eslint-disable-next-line @typescript-eslint/no-this-alias\n    const self = this;\n    if (self.locked) {\n      return new Promise<void>((resolve) => self.queue.push(resolve));\n    } else {\n      self.locked = true;\n      return Promise.resolve();\n    }\n  }\n\n  /**\n   * @summary releases the lock\n   */\n  release() {\n    // eslint-disable-next-line @typescript-eslint/no-this-alias\n    const self = this;\n    const next: LockCallable | undefined = self.queue.shift();\n    if (next) {\n      if (\n        typeof (globalThis as unknown as { window: any }).window === \"undefined\"\n      )\n        globalThis.process.nextTick(next); // if you are on node\n      else setTimeout(next, 0); // if you are in the browser\n    } else {\n      self.locked = false;\n    }\n  }\n}\n","/**\n * @summary Simple Synchronous Lock implementation\n * @description for transaction management\n * adapted from {@link https://www.talkinghightech.com/en/creating-a-js-lock-for-a-resource/}\n *\n * @param {number} [counter] the number of simultaneous transactions allowed. defaults to 1\n * @param {Function} [onBegin] to be called at the start of the transaction\n * @param {Function} [onEnd] to be called at the conclusion of the transaction\n *\n * @class SyncronousLock\n * @implements TransactionLock\n *\n * @category Transactions\n */ import { Transaction } from \"../Transaction\";\nimport { TransactionLock } from \"../interfaces/TransactionLock\";\nimport { Lock } from \"./Lock\";\n\nexport class SyncronousLock implements TransactionLock {\n  private counter: number;\n  private pendingTransactions: Transaction[];\n  currentTransaction?: Transaction = undefined;\n  private readonly onBegin?: () => Promise<void>;\n  private readonly onEnd?: (err?: Error) => Promise<void>;\n\n  private readonly lock = new Lock();\n\n  constructor(\n    counter: number = 1,\n    onBegin?: () => Promise<void>,\n    onEnd?: (err?: Error) => Promise<void>,\n  ) {\n    this.counter = counter;\n    this.pendingTransactions = [];\n    this.onBegin = onBegin;\n    this.onEnd = onEnd;\n  }\n\n  /**\n   * @summary Submits a transaction to be processed\n   * @param {Transaction} transaction\n   */\n  submit(transaction: Transaction): void {\n    // eslint-disable-next-line @typescript-eslint/no-this-alias\n    const self = this;\n    self.lock.acquire().then(() => {\n      if (\n        self.currentTransaction &&\n        self.currentTransaction.id === transaction.id\n      ) {\n        self.lock.release();\n        return transaction.fire();\n      }\n\n      if (self.counter > 0) {\n        self.counter--;\n        self.lock.release();\n        return self.fireTransaction(transaction);\n      } else {\n        self.pendingTransactions.push(transaction);\n        self.lock.release();\n      }\n    });\n  }\n\n  /**\n   * @summary Executes a transaction\n   *\n   * @param {Transaction} transaction\n   * @private\n   */\n  private fireTransaction(transaction: Transaction) {\n    // eslint-disable-next-line @typescript-eslint/no-this-alias\n    const self = this;\n    self.lock.acquire().then(() => {\n      self.currentTransaction = transaction;\n      self.lock.release();\n      if (self.onBegin)\n        self.onBegin().then(() => {\n          // all.call(\n          //   self,\n          //   `Firing transaction {0}. {1} remaining...`,\n          //   transaction.id,\n          //   this.pendingTransactions.length,\n          // );\n          transaction.fire();\n        });\n      else {\n        // all.call(\n        //   self,\n        //   `Firing transaction {0}. {1} remaining...`,\n        //   transaction.id,\n        //   this.pendingTransactions.length,\n        // );\n        transaction.fire();\n      }\n    });\n  }\n  /**\n   * @summary Releases The lock after the conclusion of a transaction\n   */\n  async release(err?: Error): Promise<void> {\n    // eslint-disable-next-line @typescript-eslint/no-this-alias\n    const self = this;\n    return new Promise<void>((resolve) => {\n      self.lock.acquire().then(() => {\n        if (!self.currentTransaction)\n          console.warn(\n            \"Trying to release an unexisting transaction. should never happen...\",\n          );\n        // debug.call(\n        //   self,\n        //   \"Releasing transaction: {0}\",\n        //   self.currentTransaction?.toString(true, true),\n        // );\n        self.currentTransaction = undefined;\n        self.lock.release();\n\n        const afterConclusionCB = () => {\n          self.lock.acquire().then(() => {\n            if (self.pendingTransactions.length > 0) {\n              const transaction =\n                self.pendingTransactions.shift() as Transaction;\n\n              const cb = () => self.fireTransaction(transaction);\n              //\n              // all(\n              //   `Releasing Transaction Lock on transaction {0}`,\n              //   transaction.id,\n              // );\n\n              if (\n                typeof (globalThis as unknown as { window: any }).window ===\n                \"undefined\"\n              )\n                globalThis.process.nextTick(cb); // if you are on node\n              else setTimeout(cb, 0); // if you are in the browser\n            } else {\n              self.counter++;\n            }\n            self.lock.release();\n            resolve();\n          });\n        };\n\n        if (self.onEnd) self.onEnd(err).then(() => afterConclusionCB());\n        else afterConclusionCB();\n      });\n    });\n  }\n}\n","export const TransactionalKeys: Record<string, string> = {\n  REFLECT: \"model.transactional.\",\n  TRANSACTIONAL: \"transactional\",\n};\n","export function getObjectName(obj: any): string | undefined {\n  if (!obj) return;\n  if (typeof obj === \"string\") return obj;\n  if (\n    obj.constructor &&\n    obj.constructor.name &&\n    [\"Function\", \"Object\"].indexOf(obj.constructor.name) === -1\n  )\n    return obj.constructor.name;\n  if (typeof obj === \"function\" && obj.name) return obj.name;\n  return obj.toString();\n}\n","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","import { TransactionalKeys } from \"./constants\";\nimport { metadata } from \"@decaf-ts/reflection\";\nimport { Transaction } from \"./Transaction\";\nimport { InternalError } from \"@decaf-ts/db-decorators\";\n\n/**\n * @summary Sets a class Async method as transactional\n *\n * @param {any[]}  [data] option metadata available to the {@link TransactionLock}\n *\n * @function transactional\n *\n * @memberOf module:db-decorators.Decorators.transactions\n */\nexport function transactional(...data: any[]) {\n  return function (\n    target: any,\n    propertyKey?: any,\n    descriptor?: PropertyDescriptor\n  ) {\n    if (!descriptor)\n      throw new InternalError(\"Missing descriptor. Should be impossible\");\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 = 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  return method(currentTransaction, ...args);\n}\n","export * from \"./interfaces\";\nexport * from \"./locks\";\nexport * from \"./constants\";\nexport * from \"./decorators\";\nexport * from \"./Transaction\";\nexport * from \"./types\";\n\n/**\n * @summary Module summary\n * @description Module description\n * @module ts-workspace\n */\n\n/**\n * @summary Namespace summary\n * @description Namespace description\n * @namespace Namespace\n * @memberOf module:ts-workspace\n */\n\n/**\n * @summary stores the current package version\n * @description this is how you should document a constant\n * @const VERSION\n * @memberOf module:ts-workspace\n */\nexport const VERSION = \"##VERSION##\";\n"],"names":["getAllPropertyDecoratorsRecursive","Reflection","DBKeys","InternalError","metadata"],"mappings":"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IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IACA;IAEA;;;;;;;IAOG;aACa,sBAAsB,CAAC,MAAW,EAAE,GAAG,IAAS,EAAA;IAC9D,IAAA,MAAM,IAAI,GAAG,WAAW,CAAC,OAAO,EAAE;IAClC,IAAA,MAAM,kBAAkB,GAAG,IAAI,CAAC,kBAAkB;IAClD,IAAA,OAAO,MAAM,CAAC,kBAAkB,EAAE,GAAG,IAAI,CAAC;IAC5C;;IChLA;;;;IAIG;IAEH;;;;;IAKG;IAEH;;;;;IAKG;AACI,UAAM,OAAO,GAAG;;;;;;;;;;;;;;"}
1
+ var t,e;t=this,e=function(t,e,n){"use strict";class o{constructor(){this.queue=[],this.locked=!1}async execute(t){let e;await this.acquire();try{e=await Promise.resolve(t())}catch(t){throw this.release(),t}return this.release(),e}async acquire(){const t=this;return t.locked?new Promise(e=>t.queue.push(e)):(t.locked=!0,Promise.resolve())}release(){const t=this.queue.shift();t?void 0===globalThis.window?globalThis.process.nextTick(t):setTimeout(t,0):this.locked=!1}}class i{constructor(t=1,e,n){this.currentTransaction=void 0,this.lock=new o,this.counter=t,this.pendingTransactions=[],this.onBegin=e,this.onEnd=n}submit(t){const e=this;e.lock.acquire().then(()=>e.currentTransaction&&e.currentTransaction.id===t.id?(e.lock.release(),t.fire()):e.counter>0?(e.counter--,e.lock.release(),e.fireTransaction(t)):(e.pendingTransactions.push(t),void e.lock.release()))}fireTransaction(t){const e=this;e.lock.acquire().then(()=>{e.currentTransaction=t,e.lock.release(),e.onBegin?e.onBegin().then(()=>{t.fire()}):t.fire()})}async release(t){const e=this;return new Promise(n=>{e.lock.acquire().then(()=>{e.currentTransaction,e.currentTransaction=void 0,e.lock.release();const o=()=>{e.lock.acquire().then(()=>{if(e.pendingTransactions.length>0){const t=e.pendingTransactions.shift(),n=()=>e.fireTransaction(t);void 0===globalThis.window?globalThis.process.nextTick(n):setTimeout(n,0)}else e.counter++;e.lock.release(),n()})};e.onEnd?e.onEnd(t).then(()=>o()):o()})})}}const s={REFLECT:"model.transactional.",TRANSACTIONAL:"transactional"};function r(t){if(t)return"string"==typeof t?t:t.constructor&&t.constructor.name&&-1===["Function","Object"].indexOf(t.constructor.name)?t.constructor.name:"function"==typeof t&&t.name?t.name:t.toString()}class c{constructor(t,e,n,o){this.id=Date.now(),this.action=n,this.method=e,this.log=[[this.id,t,e].join(" | ")],this.source=t,this.metadata=o}static push(t,e,...n){const o=n.pop();if(!o||"function"!=typeof o)throw Error("Missing callback");const i=(t,...e)=>{c.getLock().release(t).then(()=>o(t,...e))},s=new c(t.constructor.name,e.name?r(e):"Anonymous",()=>e.call(s.bindToTransaction(t),...n,i));c.getLock().submit(s)}static setLock(t){this.lock=t}static getLock(){return this.lock||(this.lock=new i),this.lock}static submit(t){c.getLock().submit(t)}static async release(t){return c.getLock().release(t)}getMetadata(){return this.metadata?[...this.metadata]:void 0}bindTransaction(t){this.log.push(...t.log),t.bindTransaction=this.bindToTransaction.bind(this),t.bindToTransaction=this.bindToTransaction.bind(this),this.action=t.action}bindToTransaction(t){const o=n.getAllPropertyDecoratorsRecursive(t,void 0,s.REFLECT);if(!o)return t;const i=this,c=e.Reflection.getAllProperties(t).reduce((n,r)=>(-1!==Object.keys(o).indexOf(r)&&o[r].find(t=>t.key===s.TRANSACTIONAL)?n[r]=(...e)=>t[r].call(t.__originalObj||t,i,...e):"clazz"===r||"constructor"===r?n[r]=t[r]:"function"==typeof t[r]?n[r]=t[r].bind(t.__originalObj||t):"object"==typeof t[r]&&t[r].constructor&&e.Reflection.getClassDecorators(s.REFLECT,t[r]).find(t=>t.key===s.TRANSACTIONAL)?n[r]=i.bindToTransaction(t[r]):n[r]=t[r],n),{});return c[n.DBKeys.ORIGINAL]=t[n.DBKeys.ORIGINAL]||t,c.toString=()=>r(c[n.DBKeys.ORIGINAL])+" proxy for transaction "+this.id,c}fire(){if(!this.action)throw Error("Missing the method");return this.action()}toString(t=!0,e=!1){return`${t?`[${this.id}]`:""}[Transaction][${this.source}.${this.method}${e?"]\nTransaction Log:\n"+this.log.join("\n"):"]"}`}static key(t){return s.REFLECT+t}}t.Lock=o,t.SyncronousLock=i,t.Transaction=c,t.TransactionalKeys=s,t.VERSION="##VERSION##",t.transactional=function(...t){return function(o,i,r){if(!r)throw new n.InternalError("Missing descriptor. Should be impossible");e.metadata(c.key(s.TRANSACTIONAL),t)(o,i);const a=r.value,l=function(...e){const n=this;return new Promise((o,s)=>{const r=(t,e)=>{c.release(t).then(()=>{if(t)return s(t);o(e)})};let l=e.shift();if(l instanceof c){const r=new c(this.constructor.name,i,async()=>{a.call(r.bindToTransaction(n),...e).then(o).catch(s)},t.length?t:void 0);l.bindTransaction(r),l.fire()}else e.unshift(l),l=new c(this.constructor.name,i,()=>{a.call(l.bindToTransaction(n),...e).then(t=>r(void 0,t)).catch(r)},t.length?t:void 0),c.submit(l)})};Object.defineProperty(l,"name",{value:i}),r.value=l}},t.transactionalSuperCall=(t,...e)=>t(c.getLock().currentTransaction,...e)},"object"==typeof exports&&"undefined"!=typeof module?e(exports,require("@decaf-ts/reflection"),require("@decaf-ts/db-decorators")):"function"==typeof define&&define.amd?define(["exports","@decaf-ts/reflection","@decaf-ts/db-decorators"],e):e((t="undefined"!=typeof globalThis?globalThis:t||self)["transactional-decorators"]={},t.decafTsReflection,t.decafTsDbDecorators);
2
+ //# sourceMappingURL=transactional-decorators.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"transactional-decorators.cjs","sources":["../src/locks/Lock.ts","../src/locks/SyncronousLock.ts","../src/constants.ts","../src/utils.ts","../src/Transaction.ts","../src/index.ts","../src/decorators.ts"],"sourcesContent":[null,null,null,null,null,null,null],"names":["Lock","constructor","this","queue","locked","execute","func","result","acquire","Promise","resolve","e","release","self","push","next","shift","globalThis","window","process","nextTick","setTimeout","SyncronousLock","counter","onBegin","onEnd","currentTransaction","undefined","lock","pendingTransactions","submit","transaction","then","id","fire","fireTransaction","err","afterConclusionCB","length","cb","TransactionalKeys","REFLECT","TRANSACTIONAL","getObjectName","obj","name","indexOf","toString","Transaction","source","method","action","metadata","Date","now","log","join","issuer","callbackMethod","args","callback","pop","Error","getLock","call","bindToTransaction","setLock","getMetadata","bindTransaction","nextTransaction","bind","transactionalMethods","getAllPropertyDecoratorsRecursive","boundObj","Reflection","getAllProperties","reduce","accum","k","Object","keys","find","o","key","__originalObj","getClassDecorators","DBKeys","ORIGINAL","withId","withLog","data","target","propertyKey","descriptor","InternalError","originalMethod","value","methodWrapper","reject","updatedTransaction","async","catch","unshift","defineProperty"],"mappings":"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"}