@rosen-bridge/tx-pot 0.1.0

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 (49) hide show
  1. package/.eslintignore +1 -0
  2. package/README.md +36 -0
  3. package/dist/db/entities/TransactionEntity.d.ts +15 -0
  4. package/dist/db/entities/TransactionEntity.d.ts.map +1 -0
  5. package/dist/db/entities/TransactionEntity.js +81 -0
  6. package/dist/db/migrations/index.d.ts +7 -0
  7. package/dist/db/migrations/index.d.ts.map +1 -0
  8. package/dist/db/migrations/index.js +7 -0
  9. package/dist/db/migrations/postgres/1706350644686-migration.d.ts +7 -0
  10. package/dist/db/migrations/postgres/1706350644686-migration.d.ts.map +1 -0
  11. package/dist/db/migrations/postgres/1706350644686-migration.js +28 -0
  12. package/dist/db/migrations/sqlite/1706007154531-migration.d.ts +7 -0
  13. package/dist/db/migrations/sqlite/1706007154531-migration.d.ts.map +1 -0
  14. package/dist/db/migrations/sqlite/1706007154531-migration.js +28 -0
  15. package/dist/index.d.ts +6 -0
  16. package/dist/index.d.ts.map +1 -0
  17. package/dist/index.js +6 -0
  18. package/dist/network/AbstractPotChainManager.d.ts +36 -0
  19. package/dist/network/AbstractPotChainManager.d.ts.map +1 -0
  20. package/dist/network/AbstractPotChainManager.js +3 -0
  21. package/dist/transaction/TxPot.d.ts +164 -0
  22. package/dist/transaction/TxPot.d.ts.map +1 -0
  23. package/dist/transaction/TxPot.js +386 -0
  24. package/dist/transaction/types.d.ts +35 -0
  25. package/dist/transaction/types.d.ts.map +1 -0
  26. package/dist/transaction/types.js +21 -0
  27. package/dist/transaction/utils.d.ts +8 -0
  28. package/dist/transaction/utils.d.ts.map +1 -0
  29. package/dist/transaction/utils.js +56 -0
  30. package/lib/db/entities/TransactionEntity.ts +44 -0
  31. package/lib/db/migrations/index.ts +7 -0
  32. package/lib/db/migrations/postgres/1706350644686-migration.ts +31 -0
  33. package/lib/db/migrations/sqlite/1706007154531-migration.ts +31 -0
  34. package/lib/index.ts +5 -0
  35. package/lib/network/AbstractPotChainManager.ts +44 -0
  36. package/lib/transaction/TxPot.ts +519 -0
  37. package/lib/transaction/types.ts +46 -0
  38. package/lib/transaction/utils.ts +59 -0
  39. package/package.json +39 -0
  40. package/tests/.gitkeep +0 -0
  41. package/tests/db/dataSource.mock.ts +18 -0
  42. package/tests/network/TestPotChainManager.ts +23 -0
  43. package/tests/transaction/TestTxPot.ts +32 -0
  44. package/tests/transaction/TxPot.spec.ts +1517 -0
  45. package/tests/transaction/testData.ts +84 -0
  46. package/tsconfig.build.json +8 -0
  47. package/tsconfig.build.tsbuildinfo +1 -0
  48. package/tsconfig.json +9 -0
  49. package/vitest.config.ts +13 -0
@@ -0,0 +1,386 @@
1
+ import { TransactionEntity } from '../db/entities/TransactionEntity';
2
+ import { DummyLogger } from '@rosen-bridge/logger-interface';
3
+ import { SigningStatus, TransactionStatus, UnregisteredChain, } from './types';
4
+ import { txOptionToClause } from './utils';
5
+ export class TxPot {
6
+ static instance;
7
+ txRepository;
8
+ chains = new Map();
9
+ validators = new Map();
10
+ txTypeCallbacks = new Map();
11
+ logger;
12
+ constructor(dataSource, logger) {
13
+ this.txRepository = dataSource.getRepository(TransactionEntity);
14
+ this.logger = logger ?? new DummyLogger();
15
+ this.logger.debug('TxPot instantiated');
16
+ }
17
+ /**
18
+ * initiates TxPot
19
+ * @param dataSource typeorm data source
20
+ * @param logger
21
+ * @returns
22
+ */
23
+ static setup = (dataSource, logger) => {
24
+ TxPot.instance = new TxPot(dataSource, logger);
25
+ return TxPot.instance;
26
+ };
27
+ /**
28
+ * returns TxPot instance (throws error if none exists)
29
+ * @returns TxPot instance
30
+ */
31
+ static getInstance = () => {
32
+ if (!TxPot.instance)
33
+ throw Error(`TxPot instance doesn't exist`);
34
+ return TxPot.instance;
35
+ };
36
+ /**
37
+ * registers a chain to TxPot
38
+ * @param chain
39
+ * @param chainManager
40
+ */
41
+ registerChain = (chain, chainManager) => {
42
+ this.chains.set(chain, chainManager);
43
+ this.logger.debug(`A TxPot chain manager is registered for chain [${chain}]`);
44
+ };
45
+ /**
46
+ * registers a validator function
47
+ * @param chain
48
+ * @param txType
49
+ * @param validator
50
+ */
51
+ registerValidator = (chain, txType, validator) => {
52
+ let chainValidators = this.validators.get(chain);
53
+ if (!chainValidators) {
54
+ chainValidators = new Map();
55
+ this.validators.set(chain, chainValidators);
56
+ }
57
+ chainValidators.set(txType, validator);
58
+ this.logger.debug(`A tx validator function is registered for chain [${chain}] and type [${txType}]`);
59
+ };
60
+ /**
61
+ * registers a callback function
62
+ * the callback will be called when status of any transactions
63
+ * of given type changes to given status
64
+ * @param txType
65
+ * @param status
66
+ * @param callback
67
+ */
68
+ registerCallback = (txType, status, callback) => {
69
+ let typeCallbacks = this.txTypeCallbacks.get(txType);
70
+ if (!typeCallbacks) {
71
+ typeCallbacks = new Map();
72
+ this.txTypeCallbacks.set(txType, typeCallbacks);
73
+ }
74
+ typeCallbacks.set(status, callback);
75
+ this.logger.debug(`A tx status callback function is registered for type [${txType}] and status [${status}]`);
76
+ };
77
+ /**
78
+ * returns chain manager for given chain
79
+ * throws error if no manager is registered for it
80
+ * @param chain
81
+ */
82
+ getChainManager = (chain) => {
83
+ const manager = this.chains.get(chain);
84
+ if (!manager)
85
+ throw new UnregisteredChain(`No manager is registered for chain [${chain}]`);
86
+ return manager;
87
+ };
88
+ /**
89
+ * sets the tx as invalid if enough blocks is passed from last check
90
+ * @param tx
91
+ */
92
+ setTransactionAsInvalid = async (tx) => {
93
+ const manager = this.getChainManager(tx.chain);
94
+ const currentHeight = await manager.getHeight();
95
+ const requiredConfirmation = manager.getTxRequiredConfirmation(tx.txType);
96
+ if (currentHeight - tx.lastCheck >= requiredConfirmation) {
97
+ await this.setTxStatus(tx, TransactionStatus.INVALID);
98
+ this.logger.info(`Tx [${tx.txId}] is invalid`);
99
+ }
100
+ else {
101
+ this.logger.info(`Tx [${tx.txId}] seems invalid. Waiting for enough confirmation of this proposition [${currentHeight - tx.lastCheck}/${requiredConfirmation}]`);
102
+ }
103
+ };
104
+ /**
105
+ * validates a transaction
106
+ * returns true if no validator functions is set or tx is valid
107
+ * otherwise handle the tx as invalid and returns false
108
+ * @param tx
109
+ */
110
+ validateTx = async (tx) => {
111
+ const validator = this.validators.get(tx.chain)?.get(tx.txType);
112
+ if (validator === undefined) {
113
+ // tx is valid since no validator is found
114
+ this.logger.debug(`No validator function is found for chain [${tx.chain}] and type [${tx.txType}]`);
115
+ return true;
116
+ }
117
+ if (await validator(tx))
118
+ return true;
119
+ await this.setTransactionAsInvalid(tx);
120
+ return false;
121
+ };
122
+ /**
123
+ * updates the status of a tx
124
+ * @param txKey tx id and chain
125
+ * @param status new status
126
+ */
127
+ setTxStatus = async (tx, status) => {
128
+ await this.txRepository.update({
129
+ txId: tx.txId,
130
+ chain: tx.chain,
131
+ }, {
132
+ status: status,
133
+ lastStatusUpdate: this.currentTime(),
134
+ });
135
+ const callback = this.txTypeCallbacks.get(tx.txType)?.get(status);
136
+ if (callback)
137
+ callback(tx, status).catch((e) => {
138
+ this.logger.debug(`An error occurred while handling tx [${tx.txId}] status change: ${e}`);
139
+ if (e instanceof Error && e.stack)
140
+ this.logger.debug(e.stack);
141
+ });
142
+ else
143
+ this.logger.debug(`No callback function is set for type [${tx.txType}] and status [${status}]`);
144
+ };
145
+ /**
146
+ * @returns current timestamp in seconds and string format
147
+ */
148
+ currentTime = () => String(Math.round(Date.now() / 1000));
149
+ /**
150
+ * submits the signed transaction to the blockchain
151
+ * @param tx
152
+ */
153
+ processSignedTx = async (tx) => {
154
+ const manager = this.getChainManager(tx.chain);
155
+ try {
156
+ await manager.submitTransaction(tx.serializedTx);
157
+ }
158
+ catch (e) {
159
+ this.logger.warn(`Failed to submit tx [${tx.txId}] to chain [${tx.chain}]: ${e}`);
160
+ if (e instanceof Error && e.stack)
161
+ this.logger.warn(e.stack);
162
+ }
163
+ await this.setTxStatus(tx, TransactionStatus.SENT);
164
+ };
165
+ /**
166
+ * processes the sent transaction
167
+ * @param tx
168
+ */
169
+ processesSentTx = async (tx) => {
170
+ const manager = this.getChainManager(tx.chain);
171
+ const txConfirmation = await manager.getTxConfirmation(tx.txId);
172
+ const requiredConfirmation = manager.getTxRequiredConfirmation(tx.txType);
173
+ if (txConfirmation >= requiredConfirmation) {
174
+ // tx is confirmed enough
175
+ await this.setTxStatus(tx, TransactionStatus.COMPLETED);
176
+ }
177
+ else if (txConfirmation === -1) {
178
+ // tx is not mined, checking mempool...
179
+ if (await manager.isTxInMempool(tx.txId)) {
180
+ // tx is in mempool, updating last check...
181
+ const height = await manager.getHeight();
182
+ await this.updateTxLastCheck(tx.txId, tx.chain, height);
183
+ this.logger.info(`Tx [${tx.txId}] is in mempool`);
184
+ }
185
+ else {
186
+ // tx is not in mempool, checking if tx is still valid
187
+ const isValidTx = await manager.isTxValid(tx.serializedTx, SigningStatus.Signed);
188
+ const isValidToType = await this.validateTx(tx);
189
+ if (isValidTx && isValidToType) {
190
+ // tx is valid. resending...
191
+ this.logger.info(`Tx [${tx.txId}] is still valid. Resending tx...`);
192
+ await manager.submitTransaction(tx.serializedTx);
193
+ }
194
+ else {
195
+ // tx seems invalid. reset status if enough blocks past.
196
+ await this.setTransactionAsInvalid(tx);
197
+ }
198
+ }
199
+ }
200
+ else {
201
+ // tx is mined, but is not confirmed enough, updating last check...
202
+ const height = await manager.getHeight();
203
+ await this.updateTxLastCheck(tx.txId, tx.chain, height);
204
+ this.logger.info(`Tx [${tx.txId}] is in confirmation process [${txConfirmation}/${requiredConfirmation}]`);
205
+ }
206
+ };
207
+ /**
208
+ * runs all jobs of TxPot
209
+ * - process signed txs
210
+ * - process sent txs
211
+ */
212
+ update = async () => {
213
+ // process signed txs
214
+ const signedTxs = await this.getTxsByStatus(TransactionStatus.SIGNED);
215
+ for (const tx of signedTxs) {
216
+ try {
217
+ await this.processSignedTx(tx);
218
+ }
219
+ catch (e) {
220
+ this.logger.warn(`An error occurred while processing tx [${tx.txId}] with status [${TransactionStatus.SIGNED}]: ${e}`);
221
+ if (e instanceof Error && e.stack)
222
+ this.logger.warn(e.stack);
223
+ }
224
+ }
225
+ this.logger.debug(`Processed [${signedTxs.length}] txs with status [${TransactionStatus.SIGNED}]`);
226
+ // process sent txs
227
+ const sentTxs = await this.getTxsByStatus(TransactionStatus.SENT);
228
+ for (const tx of sentTxs) {
229
+ try {
230
+ await this.processesSentTx(tx);
231
+ }
232
+ catch (e) {
233
+ this.logger.warn(`An error occurred while processing tx [${tx.txId}] with status [${TransactionStatus.SENT}]: ${e}`);
234
+ if (e instanceof Error && e.stack)
235
+ this.logger.warn(e.stack);
236
+ }
237
+ }
238
+ this.logger.debug(`Processed [${sentTxs.length}] txs with status [${TransactionStatus.SENT}]`);
239
+ };
240
+ /**
241
+ * gets transactions by status
242
+ * @param status
243
+ * @param validate
244
+ * @returns
245
+ */
246
+ getTxsByStatus = async (status, validate = false) => {
247
+ const txs = await this.txRepository.find({
248
+ where: {
249
+ status: status,
250
+ },
251
+ });
252
+ if (!validate)
253
+ return txs;
254
+ // validate the transactions
255
+ const validTxs = [];
256
+ for (const tx of txs) {
257
+ if (await this.validateTx(tx))
258
+ validTxs.push(tx);
259
+ }
260
+ return validTxs;
261
+ };
262
+ /**
263
+ * inserts a new transaction into db
264
+ * @param txId
265
+ * @param chain
266
+ * @param txType
267
+ * @param requiredSign
268
+ * @param serializedTx
269
+ * @param initialStatus
270
+ * @param lastCheck
271
+ */
272
+ addTx = async (txId, chain, txType, requiredSign, serializedTx, initialStatus = TransactionStatus.APPROVED, lastCheck = 0) => {
273
+ await this.txRepository.insert({
274
+ txId: txId,
275
+ chain: chain,
276
+ txType: txType,
277
+ status: initialStatus,
278
+ requiredSign: requiredSign,
279
+ lastCheck: lastCheck,
280
+ lastStatusUpdate: this.currentTime(),
281
+ failedInSign: false,
282
+ signFailedCount: 0,
283
+ serializedTx: serializedTx,
284
+ });
285
+ };
286
+ /**
287
+ * updates the status of a tx
288
+ * @param txId
289
+ * @param chain
290
+ * @param status new status
291
+ */
292
+ setTxStatusById = async (txId, chain, status) => {
293
+ const tx = await this.txRepository.findOneOrFail({
294
+ where: { txId, chain },
295
+ });
296
+ await this.setTxStatus(tx, status);
297
+ };
298
+ /**
299
+ * updates tx info when failed in sign process
300
+ * @param txId
301
+ * @param chain
302
+ */
303
+ setTxAsSignFailed = async (txId, chain) => {
304
+ await this.txRepository.update({
305
+ txId: txId,
306
+ chain: chain,
307
+ status: TransactionStatus.IN_SIGN,
308
+ }, {
309
+ status: TransactionStatus.SIGN_FAILED,
310
+ lastStatusUpdate: this.currentTime(),
311
+ signFailedCount: () => '"signFailedCount" + 1',
312
+ failedInSign: true,
313
+ });
314
+ };
315
+ /**
316
+ * updates the tx and set status as signed
317
+ * @param txId
318
+ * @param chain
319
+ * @param serializedTx
320
+ * @param currentHeight current height of the blockchain
321
+ * @param extra
322
+ * @param extra2
323
+ */
324
+ setTxAsSigned = async (txId, chain, serializedTx, currentHeight, extra, extra2) => {
325
+ const updatedFields = {
326
+ serializedTx: serializedTx,
327
+ status: TransactionStatus.SIGNED,
328
+ lastStatusUpdate: this.currentTime(),
329
+ lastCheck: currentHeight,
330
+ };
331
+ if (extra)
332
+ updatedFields.extra = extra;
333
+ if (extra2)
334
+ updatedFields.extra2 = extra2;
335
+ await this.txRepository.update({ txId, chain }, updatedFields);
336
+ };
337
+ /**
338
+ * updates last check value of a tx
339
+ * @param txId
340
+ * @param chain
341
+ * @param currentHeight current height of the blockchain
342
+ */
343
+ updateTxLastCheck = async (txId, chain, currentHeight) => {
344
+ await this.txRepository.update({ txId, chain }, { lastCheck: currentHeight });
345
+ };
346
+ /**
347
+ * updates failedInSign field of a transaction to false
348
+ * @param txId
349
+ * @param chain
350
+ */
351
+ resetFailedInSign = async (txId, chain) => {
352
+ await this.txRepository.update({ txId, chain }, {
353
+ failedInSign: false,
354
+ });
355
+ };
356
+ /**
357
+ * updates requiredSign field of a transaction
358
+ * @param txId
359
+ * @param chain
360
+ * @param requiredSign
361
+ */
362
+ updateRequiredSign = async (txId, chain, requiredSign) => {
363
+ await this.txRepository.update({ txId, chain }, {
364
+ requiredSign: requiredSign,
365
+ });
366
+ };
367
+ /**
368
+ * gets the transaction by its id and chain
369
+ * @param txId
370
+ * @param chain
371
+ */
372
+ getTxByKey = async (txId, chain) => {
373
+ return await this.txRepository.findOne({
374
+ where: { txId, chain },
375
+ });
376
+ };
377
+ /**
378
+ * @returns the transactions with valid status
379
+ */
380
+ getTxsQuery = (options = []) => {
381
+ return this.txRepository.find({
382
+ where: options.map(txOptionToClause),
383
+ });
384
+ };
385
+ }
386
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"TxPot.js","sourceRoot":"","sources":["../../lib/transaction/TxPot.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,iBAAiB,EAAE,MAAM,kCAAkC,CAAC;AACrE,OAAO,EAAkB,WAAW,EAAE,MAAM,gCAAgC,CAAC;AAC7E,OAAO,EAEL,aAAa,EACb,iBAAiB,EAEjB,iBAAiB,GAElB,MAAM,SAAS,CAAC;AACjB,OAAO,EAAE,gBAAgB,EAAE,MAAM,SAAS,CAAC;AAG3C,MAAM,OAAO,KAAK;IACN,MAAM,CAAC,QAAQ,CAAQ;IACd,YAAY,CAAgC;IACrD,MAAM,GAAG,IAAI,GAAG,EAAmC,CAAC;IACpD,UAAU,GAAG,IAAI,GAAG,EAA0C,CAAC;IAC/D,eAAe,GAAG,IAAI,GAAG,EAGhC,CAAC;IACM,MAAM,CAAiB;IAEjC,YAAsB,UAAsB,EAAE,MAAuB;QACnE,IAAI,CAAC,YAAY,GAAG,UAAU,CAAC,aAAa,CAAC,iBAAiB,CAAC,CAAC;QAChE,IAAI,CAAC,MAAM,GAAG,MAAM,IAAI,IAAI,WAAW,EAAE,CAAC;QAC1C,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,oBAAoB,CAAC,CAAC;IAC1C,CAAC;IAED;;;;;OAKG;IACI,MAAM,CAAC,KAAK,GAAG,CACpB,UAAsB,EACtB,MAAuB,EAChB,EAAE;QACT,KAAK,CAAC,QAAQ,GAAG,IAAI,KAAK,CAAC,UAAU,EAAE,MAAM,CAAC,CAAC;QAC/C,OAAO,KAAK,CAAC,QAAQ,CAAC;IACxB,CAAC,CAAC;IAEF;;;OAGG;IACI,MAAM,CAAC,WAAW,GAAG,GAAU,EAAE;QACtC,IAAI,CAAC,KAAK,CAAC,QAAQ;YAAE,MAAM,KAAK,CAAC,8BAA8B,CAAC,CAAC;QACjE,OAAO,KAAK,CAAC,QAAQ,CAAC;IACxB,CAAC,CAAC;IAEF;;;;OAIG;IACH,aAAa,GAAG,CACd,KAAa,EACb,YAAqC,EAC/B,EAAE;QACR,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,EAAE,YAAY,CAAC,CAAC;QACrC,IAAI,CAAC,MAAM,CAAC,KAAK,CACf,kDAAkD,KAAK,GAAG,CAC3D,CAAC;IACJ,CAAC,CAAC;IAEF;;;;;OAKG;IACH,iBAAiB,GAAG,CAClB,KAAa,EACb,MAAc,EACd,SAA4B,EACtB,EAAE;QACR,IAAI,eAAe,GAAG,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;QACjD,IAAI,CAAC,eAAe,EAAE,CAAC;YACrB,eAAe,GAAG,IAAI,GAAG,EAA6B,CAAC;YACvD,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,KAAK,EAAE,eAAe,CAAC,CAAC;QAC9C,CAAC;QAED,eAAe,CAAC,GAAG,CAAC,MAAM,EAAE,SAAS,CAAC,CAAC;QACvC,IAAI,CAAC,MAAM,CAAC,KAAK,CACf,oDAAoD,KAAK,eAAe,MAAM,GAAG,CAClF,CAAC;IACJ,CAAC,CAAC;IAEF;;;;;;;OAOG;IACH,gBAAgB,GAAG,CACjB,MAAc,EACd,MAAyB,EACzB,QAA0B,EACpB,EAAE;QACR,IAAI,aAAa,GAAG,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;QACrD,IAAI,CAAC,aAAa,EAAE,CAAC;YACnB,aAAa,GAAG,IAAI,GAAG,EAAuC,CAAC;YAC/D,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,MAAM,EAAE,aAAa,CAAC,CAAC;QAClD,CAAC;QAED,aAAa,CAAC,GAAG,CAAC,MAAM,EAAE,QAAQ,CAAC,CAAC;QACpC,IAAI,CAAC,MAAM,CAAC,KAAK,CACf,yDAAyD,MAAM,iBAAiB,MAAM,GAAG,CAC1F,CAAC;IACJ,CAAC,CAAC;IAEF;;;;OAIG;IACO,eAAe,GAAG,CAAC,KAAa,EAA2B,EAAE;QACrE,MAAM,OAAO,GAAG,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;QACvC,IAAI,CAAC,OAAO;YACV,MAAM,IAAI,iBAAiB,CACzB,uCAAuC,KAAK,GAAG,CAChD,CAAC;QACJ,OAAO,OAAO,CAAC;IACjB,CAAC,CAAC;IAEF;;;OAGG;IACO,uBAAuB,GAAG,KAAK,EACvC,EAAqB,EACN,EAAE;QACjB,MAAM,OAAO,GAAG,IAAI,CAAC,eAAe,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC;QAE/C,MAAM,aAAa,GAAG,MAAM,OAAO,CAAC,SAAS,EAAE,CAAC;QAChD,MAAM,oBAAoB,GAAG,OAAO,CAAC,yBAAyB,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC;QAE1E,IAAI,aAAa,GAAG,EAAE,CAAC,SAAS,IAAI,oBAAoB,EAAE,CAAC;YACzD,MAAM,IAAI,CAAC,WAAW,CAAC,EAAE,EAAE,iBAAiB,CAAC,OAAO,CAAC,CAAC;YACtD,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,IAAI,cAAc,CAAC,CAAC;QACjD,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,MAAM,CAAC,IAAI,CACd,OACE,EAAE,CAAC,IACL,yEACE,aAAa,GAAG,EAAE,CAAC,SACrB,IAAI,oBAAoB,GAAG,CAC5B,CAAC;QACJ,CAAC;IACH,CAAC,CAAC;IAEF;;;;;OAKG;IACO,UAAU,GAAG,KAAK,EAAE,EAAqB,EAAoB,EAAE;QACvE,MAAM,SAAS,GAAG,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE,CAAC,KAAK,CAAC,EAAE,GAAG,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC;QAChE,IAAI,SAAS,KAAK,SAAS,EAAE,CAAC;YAC5B,0CAA0C;YAC1C,IAAI,CAAC,MAAM,CAAC,KAAK,CACf,6CAA6C,EAAE,CAAC,KAAK,eAAe,EAAE,CAAC,MAAM,GAAG,CACjF,CAAC;YACF,OAAO,IAAI,CAAC;QACd,CAAC;QACD,IAAI,MAAM,SAAS,CAAC,EAAE,CAAC;YAAE,OAAO,IAAI,CAAC;QAErC,MAAM,IAAI,CAAC,uBAAuB,CAAC,EAAE,CAAC,CAAC;QACvC,OAAO,KAAK,CAAC;IACf,CAAC,CAAC;IAEF;;;;OAIG;IACO,WAAW,GAAG,KAAK,EAC3B,EAAqB,EACrB,MAAyB,EACV,EAAE;QACjB,MAAM,IAAI,CAAC,YAAY,CAAC,MAAM,CAC5B;YACE,IAAI,EAAE,EAAE,CAAC,IAAI;YACb,KAAK,EAAE,EAAE,CAAC,KAAK;SAChB,EACD;YACE,MAAM,EAAE,MAAM;YACd,gBAAgB,EAAE,IAAI,CAAC,WAAW,EAAE;SACrC,CACF,CAAC;QACF,MAAM,QAAQ,GAAG,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,EAAE,CAAC,MAAM,CAAC,EAAE,GAAG,CAAC,MAAM,CAAC,CAAC;QAClE,IAAI,QAAQ;YACV,QAAQ,CAAC,EAAE,EAAE,MAAM,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,EAAE;gBAC/B,IAAI,CAAC,MAAM,CAAC,KAAK,CACf,wCAAwC,EAAE,CAAC,IAAI,oBAAoB,CAAC,EAAE,CACvE,CAAC;gBACF,IAAI,CAAC,YAAY,KAAK,IAAI,CAAC,CAAC,KAAK;oBAAE,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC;YAChE,CAAC,CAAC,CAAC;;YAEH,IAAI,CAAC,MAAM,CAAC,KAAK,CACf,yCAAyC,EAAE,CAAC,MAAM,iBAAiB,MAAM,GAAG,CAC7E,CAAC;IACN,CAAC,CAAC;IAEF;;OAEG;IACO,WAAW,GAAG,GAAG,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,CAAC,CAAC;IAEpE;;;OAGG;IACO,eAAe,GAAG,KAAK,EAAE,EAAqB,EAAiB,EAAE;QACzE,MAAM,OAAO,GAAG,IAAI,CAAC,eAAe,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC;QAC/C,IAAI,CAAC;YACH,MAAM,OAAO,CAAC,iBAAiB,CAAC,EAAE,CAAC,YAAY,CAAC,CAAC;QACnD,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,IAAI,CAAC,MAAM,CAAC,IAAI,CACd,wBAAwB,EAAE,CAAC,IAAI,eAAe,EAAE,CAAC,KAAK,MAAM,CAAC,EAAE,CAChE,CAAC;YACF,IAAI,CAAC,YAAY,KAAK,IAAI,CAAC,CAAC,KAAK;gBAAE,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC;QAC/D,CAAC;QACD,MAAM,IAAI,CAAC,WAAW,CAAC,EAAE,EAAE,iBAAiB,CAAC,IAAI,CAAC,CAAC;IACrD,CAAC,CAAC;IAEF;;;OAGG;IACO,eAAe,GAAG,KAAK,EAAE,EAAqB,EAAiB,EAAE;QACzE,MAAM,OAAO,GAAG,IAAI,CAAC,eAAe,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC;QAE/C,MAAM,cAAc,GAAG,MAAM,OAAO,CAAC,iBAAiB,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC;QAChE,MAAM,oBAAoB,GAAG,OAAO,CAAC,yBAAyB,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC;QAE1E,IAAI,cAAc,IAAI,oBAAoB,EAAE,CAAC;YAC3C,yBAAyB;YACzB,MAAM,IAAI,CAAC,WAAW,CAAC,EAAE,EAAE,iBAAiB,CAAC,SAAS,CAAC,CAAC;QAC1D,CAAC;aAAM,IAAI,cAAc,KAAK,CAAC,CAAC,EAAE,CAAC;YACjC,uCAAuC;YACvC,IAAI,MAAM,OAAO,CAAC,aAAa,CAAC,EAAE,CAAC,IAAI,CAAC,EAAE,CAAC;gBACzC,2CAA2C;gBAC3C,MAAM,MAAM,GAAG,MAAM,OAAO,CAAC,SAAS,EAAE,CAAC;gBACzC,MAAM,IAAI,CAAC,iBAAiB,CAAC,EAAE,CAAC,IAAI,EAAE,EAAE,CAAC,KAAK,EAAE,MAAM,CAAC,CAAC;gBACxD,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,IAAI,iBAAiB,CAAC,CAAC;YACpD,CAAC;iBAAM,CAAC;gBACN,sDAAsD;gBACtD,MAAM,SAAS,GAAG,MAAM,OAAO,CAAC,SAAS,CACvC,EAAE,CAAC,YAAY,EACf,aAAa,CAAC,MAAM,CACrB,CAAC;gBACF,MAAM,aAAa,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,EAAE,CAAC,CAAC;gBAEhD,IAAI,SAAS,IAAI,aAAa,EAAE,CAAC;oBAC/B,4BAA4B;oBAC5B,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,IAAI,mCAAmC,CAAC,CAAC;oBACpE,MAAM,OAAO,CAAC,iBAAiB,CAAC,EAAE,CAAC,YAAY,CAAC,CAAC;gBACnD,CAAC;qBAAM,CAAC;oBACN,wDAAwD;oBACxD,MAAM,IAAI,CAAC,uBAAuB,CAAC,EAAE,CAAC,CAAC;gBACzC,CAAC;YACH,CAAC;QACH,CAAC;aAAM,CAAC;YACN,mEAAmE;YACnE,MAAM,MAAM,GAAG,MAAM,OAAO,CAAC,SAAS,EAAE,CAAC;YACzC,MAAM,IAAI,CAAC,iBAAiB,CAAC,EAAE,CAAC,IAAI,EAAE,EAAE,CAAC,KAAK,EAAE,MAAM,CAAC,CAAC;YACxD,IAAI,CAAC,MAAM,CAAC,IAAI,CACd,OAAO,EAAE,CAAC,IAAI,iCAAiC,cAAc,IAAI,oBAAoB,GAAG,CACzF,CAAC;QACJ,CAAC;IACH,CAAC,CAAC;IAEF;;;;OAIG;IACH,MAAM,GAAG,KAAK,IAAmB,EAAE;QACjC,qBAAqB;QACrB,MAAM,SAAS,GAAG,MAAM,IAAI,CAAC,cAAc,CAAC,iBAAiB,CAAC,MAAM,CAAC,CAAC;QACtE,KAAK,MAAM,EAAE,IAAI,SAAS,EAAE,CAAC;YAC3B,IAAI,CAAC;gBACH,MAAM,IAAI,CAAC,eAAe,CAAC,EAAE,CAAC,CAAC;YACjC,CAAC;YAAC,OAAO,CAAC,EAAE,CAAC;gBACX,IAAI,CAAC,MAAM,CAAC,IAAI,CACd,0CAA0C,EAAE,CAAC,IAAI,kBAAkB,iBAAiB,CAAC,MAAM,MAAM,CAAC,EAAE,CACrG,CAAC;gBACF,IAAI,CAAC,YAAY,KAAK,IAAI,CAAC,CAAC,KAAK;oBAAE,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC;YAC/D,CAAC;QACH,CAAC;QACD,IAAI,CAAC,MAAM,CAAC,KAAK,CACf,cAAc,SAAS,CAAC,MAAM,sBAAsB,iBAAiB,CAAC,MAAM,GAAG,CAChF,CAAC;QAEF,mBAAmB;QACnB,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,cAAc,CAAC,iBAAiB,CAAC,IAAI,CAAC,CAAC;QAClE,KAAK,MAAM,EAAE,IAAI,OAAO,EAAE,CAAC;YACzB,IAAI,CAAC;gBACH,MAAM,IAAI,CAAC,eAAe,CAAC,EAAE,CAAC,CAAC;YACjC,CAAC;YAAC,OAAO,CAAC,EAAE,CAAC;gBACX,IAAI,CAAC,MAAM,CAAC,IAAI,CACd,0CAA0C,EAAE,CAAC,IAAI,kBAAkB,iBAAiB,CAAC,IAAI,MAAM,CAAC,EAAE,CACnG,CAAC;gBACF,IAAI,CAAC,YAAY,KAAK,IAAI,CAAC,CAAC,KAAK;oBAAE,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC;YAC/D,CAAC;QACH,CAAC;QACD,IAAI,CAAC,MAAM,CAAC,KAAK,CACf,cAAc,OAAO,CAAC,MAAM,sBAAsB,iBAAiB,CAAC,IAAI,GAAG,CAC5E,CAAC;IACJ,CAAC,CAAC;IAEF;;;;;OAKG;IACH,cAAc,GAAG,KAAK,EACpB,MAAyB,EACzB,QAAQ,GAAG,KAAK,EACmB,EAAE;QACrC,MAAM,GAAG,GAAG,MAAM,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC;YACvC,KAAK,EAAE;gBACL,MAAM,EAAE,MAAM;aACf;SACF,CAAC,CAAC;QACH,IAAI,CAAC,QAAQ;YAAE,OAAO,GAAG,CAAC;QAE1B,4BAA4B;QAC5B,MAAM,QAAQ,GAA6B,EAAE,CAAC;QAC9C,KAAK,MAAM,EAAE,IAAI,GAAG,EAAE,CAAC;YACrB,IAAI,MAAM,IAAI,CAAC,UAAU,CAAC,EAAE,CAAC;gBAAE,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACnD,CAAC;QACD,OAAO,QAAQ,CAAC;IAClB,CAAC,CAAC;IAEF;;;;;;;;;OASG;IACH,KAAK,GAAG,KAAK,EACX,IAAY,EACZ,KAAa,EACb,MAAc,EACd,YAAoB,EACpB,YAAoB,EACpB,aAAa,GAAG,iBAAiB,CAAC,QAAQ,EAC1C,SAAS,GAAG,CAAC,EACE,EAAE;QACjB,MAAM,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC;YAC7B,IAAI,EAAE,IAAI;YACV,KAAK,EAAE,KAAK;YACZ,MAAM,EAAE,MAAM;YACd,MAAM,EAAE,aAAa;YACrB,YAAY,EAAE,YAAY;YAC1B,SAAS,EAAE,SAAS;YACpB,gBAAgB,EAAE,IAAI,CAAC,WAAW,EAAE;YACpC,YAAY,EAAE,KAAK;YACnB,eAAe,EAAE,CAAC;YAClB,YAAY,EAAE,YAAY;SAC3B,CAAC,CAAC;IACL,CAAC,CAAC;IAEF;;;;;OAKG;IACH,eAAe,GAAG,KAAK,EACrB,IAAY,EACZ,KAAa,EACb,MAAyB,EACV,EAAE;QACjB,MAAM,EAAE,GAAG,MAAM,IAAI,CAAC,YAAY,CAAC,aAAa,CAAC;YAC/C,KAAK,EAAE,EAAE,IAAI,EAAE,KAAK,EAAE;SACvB,CAAC,CAAC;QACH,MAAM,IAAI,CAAC,WAAW,CAAC,EAAE,EAAE,MAAM,CAAC,CAAC;IACrC,CAAC,CAAC;IAEF;;;;OAIG;IACH,iBAAiB,GAAG,KAAK,EAAE,IAAY,EAAE,KAAa,EAAiB,EAAE;QACvE,MAAM,IAAI,CAAC,YAAY,CAAC,MAAM,CAC5B;YACE,IAAI,EAAE,IAAI;YACV,KAAK,EAAE,KAAK;YACZ,MAAM,EAAE,iBAAiB,CAAC,OAAO;SAClC,EACD;YACE,MAAM,EAAE,iBAAiB,CAAC,WAAW;YACrC,gBAAgB,EAAE,IAAI,CAAC,WAAW,EAAE;YACpC,eAAe,EAAE,GAAG,EAAE,CAAC,uBAAuB;YAC9C,YAAY,EAAE,IAAI;SACnB,CACF,CAAC;IACJ,CAAC,CAAC;IAEF;;;;;;;;OAQG;IACH,aAAa,GAAG,KAAK,EACnB,IAAY,EACZ,KAAa,EACb,YAAoB,EACpB,aAAqB,EACrB,KAAc,EACd,MAAe,EACA,EAAE;QACjB,MAAM,aAAa,GAA+B;YAChD,YAAY,EAAE,YAAY;YAC1B,MAAM,EAAE,iBAAiB,CAAC,MAAM;YAChC,gBAAgB,EAAE,IAAI,CAAC,WAAW,EAAE;YACpC,SAAS,EAAE,aAAa;SACzB,CAAC;QACF,IAAI,KAAK;YAAE,aAAa,CAAC,KAAK,GAAG,KAAK,CAAC;QACvC,IAAI,MAAM;YAAE,aAAa,CAAC,MAAM,GAAG,MAAM,CAAC;QAE1C,MAAM,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,EAAE,IAAI,EAAE,KAAK,EAAE,EAAE,aAAa,CAAC,CAAC;IACjE,CAAC,CAAC;IAEF;;;;;OAKG;IACH,iBAAiB,GAAG,KAAK,EACvB,IAAY,EACZ,KAAa,EACb,aAAqB,EACN,EAAE;QACjB,MAAM,IAAI,CAAC,YAAY,CAAC,MAAM,CAC5B,EAAE,IAAI,EAAE,KAAK,EAAE,EACf,EAAE,SAAS,EAAE,aAAa,EAAE,CAC7B,CAAC;IACJ,CAAC,CAAC;IAEF;;;;OAIG;IACH,iBAAiB,GAAG,KAAK,EAAE,IAAY,EAAE,KAAa,EAAiB,EAAE;QACvE,MAAM,IAAI,CAAC,YAAY,CAAC,MAAM,CAC5B,EAAE,IAAI,EAAE,KAAK,EAAE,EACf;YACE,YAAY,EAAE,KAAK;SACpB,CACF,CAAC;IACJ,CAAC,CAAC;IAEF;;;;;OAKG;IACH,kBAAkB,GAAG,KAAK,EACxB,IAAY,EACZ,KAAa,EACb,YAAoB,EACL,EAAE;QACjB,MAAM,IAAI,CAAC,YAAY,CAAC,MAAM,CAC5B,EAAE,IAAI,EAAE,KAAK,EAAE,EACf;YACE,YAAY,EAAE,YAAY;SAC3B,CACF,CAAC;IACJ,CAAC,CAAC;IAEF;;;;OAIG;IACH,UAAU,GAAG,KAAK,EAChB,IAAY,EACZ,KAAa,EACsB,EAAE;QACrC,OAAO,MAAM,IAAI,CAAC,YAAY,CAAC,OAAO,CAAC;YACrC,KAAK,EAAE,EAAE,IAAI,EAAE,KAAK,EAAE;SACvB,CAAC,CAAC;IACL,CAAC,CAAC;IAEF;;OAEG;IACH,WAAW,GAAG,CACZ,UAA4B,EAAE,EACA,EAAE;QAChC,OAAO,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC;YAC5B,KAAK,EAAE,OAAO,CAAC,GAAG,CAAC,gBAAgB,CAAC;SACrC,CAAC,CAAC;IACL,CAAC,CAAC","sourcesContent":["import { DataSource, Repository } from 'typeorm';\nimport { TransactionEntity } from '../db/entities/TransactionEntity';\nimport { AbstractLogger, DummyLogger } from '@rosen-bridge/logger-interface';\nimport {\n  CallbackFunction,\n  SigningStatus,\n  TransactionStatus,\n  TxOptions,\n  UnregisteredChain,\n  ValidatorFunction,\n} from './types';\nimport { txOptionToClause } from './utils';\nimport { AbstractPotChainManager } from '../network/AbstractPotChainManager';\n\nexport class TxPot {\n  protected static instance: TxPot;\n  protected readonly txRepository: Repository<TransactionEntity>;\n  protected chains = new Map<string, AbstractPotChainManager>();\n  protected validators = new Map<string, Map<string, ValidatorFunction>>();\n  protected txTypeCallbacks = new Map<\n    string,\n    Map<TransactionStatus, CallbackFunction>\n  >();\n  protected logger: AbstractLogger;\n\n  protected constructor(dataSource: DataSource, logger?: AbstractLogger) {\n    this.txRepository = dataSource.getRepository(TransactionEntity);\n    this.logger = logger ?? new DummyLogger();\n    this.logger.debug('TxPot instantiated');\n  }\n\n  /**\n   * initiates TxPot\n   * @param dataSource typeorm data source\n   * @param logger\n   * @returns\n   */\n  public static setup = (\n    dataSource: DataSource,\n    logger?: AbstractLogger\n  ): TxPot => {\n    TxPot.instance = new TxPot(dataSource, logger);\n    return TxPot.instance;\n  };\n\n  /**\n   * returns TxPot instance (throws error if none exists)\n   * @returns TxPot instance\n   */\n  public static getInstance = (): TxPot => {\n    if (!TxPot.instance) throw Error(`TxPot instance doesn't exist`);\n    return TxPot.instance;\n  };\n\n  /**\n   * registers a chain to TxPot\n   * @param chain\n   * @param chainManager\n   */\n  registerChain = (\n    chain: string,\n    chainManager: AbstractPotChainManager\n  ): void => {\n    this.chains.set(chain, chainManager);\n    this.logger.debug(\n      `A TxPot chain manager is registered for chain [${chain}]`\n    );\n  };\n\n  /**\n   * registers a validator function\n   * @param chain\n   * @param txType\n   * @param validator\n   */\n  registerValidator = (\n    chain: string,\n    txType: string,\n    validator: ValidatorFunction\n  ): void => {\n    let chainValidators = this.validators.get(chain);\n    if (!chainValidators) {\n      chainValidators = new Map<string, ValidatorFunction>();\n      this.validators.set(chain, chainValidators);\n    }\n\n    chainValidators.set(txType, validator);\n    this.logger.debug(\n      `A tx validator function is registered for chain [${chain}] and type [${txType}]`\n    );\n  };\n\n  /**\n   * registers a callback function\n   *  the callback will be called when status of any transactions\n   *  of given type changes to given status\n   * @param txType\n   * @param status\n   * @param callback\n   */\n  registerCallback = (\n    txType: string,\n    status: TransactionStatus,\n    callback: CallbackFunction\n  ): void => {\n    let typeCallbacks = this.txTypeCallbacks.get(txType);\n    if (!typeCallbacks) {\n      typeCallbacks = new Map<TransactionStatus, CallbackFunction>();\n      this.txTypeCallbacks.set(txType, typeCallbacks);\n    }\n\n    typeCallbacks.set(status, callback);\n    this.logger.debug(\n      `A tx status callback function is registered for type [${txType}] and status [${status}]`\n    );\n  };\n\n  /**\n   * returns chain manager for given chain\n   *  throws error if no manager is registered for it\n   * @param chain\n   */\n  protected getChainManager = (chain: string): AbstractPotChainManager => {\n    const manager = this.chains.get(chain);\n    if (!manager)\n      throw new UnregisteredChain(\n        `No manager is registered for chain [${chain}]`\n      );\n    return manager;\n  };\n\n  /**\n   * sets the tx as invalid if enough blocks is passed from last check\n   * @param tx\n   */\n  protected setTransactionAsInvalid = async (\n    tx: TransactionEntity\n  ): Promise<void> => {\n    const manager = this.getChainManager(tx.chain);\n\n    const currentHeight = await manager.getHeight();\n    const requiredConfirmation = manager.getTxRequiredConfirmation(tx.txType);\n\n    if (currentHeight - tx.lastCheck >= requiredConfirmation) {\n      await this.setTxStatus(tx, TransactionStatus.INVALID);\n      this.logger.info(`Tx [${tx.txId}] is invalid`);\n    } else {\n      this.logger.info(\n        `Tx [${\n          tx.txId\n        }] seems invalid. Waiting for enough confirmation of this proposition [${\n          currentHeight - tx.lastCheck\n        }/${requiredConfirmation}]`\n      );\n    }\n  };\n\n  /**\n   * validates a transaction\n   * returns true if no validator functions is set or tx is valid\n   * otherwise handle the tx as invalid and returns false\n   * @param tx\n   */\n  protected validateTx = async (tx: TransactionEntity): Promise<boolean> => {\n    const validator = this.validators.get(tx.chain)?.get(tx.txType);\n    if (validator === undefined) {\n      // tx is valid since no validator is found\n      this.logger.debug(\n        `No validator function is found for chain [${tx.chain}] and type [${tx.txType}]`\n      );\n      return true;\n    }\n    if (await validator(tx)) return true;\n\n    await this.setTransactionAsInvalid(tx);\n    return false;\n  };\n\n  /**\n   * updates the status of a tx\n   * @param txKey tx id and chain\n   * @param status new status\n   */\n  protected setTxStatus = async (\n    tx: TransactionEntity,\n    status: TransactionStatus\n  ): Promise<void> => {\n    await this.txRepository.update(\n      {\n        txId: tx.txId,\n        chain: tx.chain,\n      },\n      {\n        status: status,\n        lastStatusUpdate: this.currentTime(),\n      }\n    );\n    const callback = this.txTypeCallbacks.get(tx.txType)?.get(status);\n    if (callback)\n      callback(tx, status).catch((e) => {\n        this.logger.debug(\n          `An error occurred while handling tx [${tx.txId}] status change: ${e}`\n        );\n        if (e instanceof Error && e.stack) this.logger.debug(e.stack);\n      });\n    else\n      this.logger.debug(\n        `No callback function is set for type [${tx.txType}] and status [${status}]`\n      );\n  };\n\n  /**\n   * @returns current timestamp in seconds and string format\n   */\n  protected currentTime = () => String(Math.round(Date.now() / 1000));\n\n  /**\n   * submits the signed transaction to the blockchain\n   * @param tx\n   */\n  protected processSignedTx = async (tx: TransactionEntity): Promise<void> => {\n    const manager = this.getChainManager(tx.chain);\n    try {\n      await manager.submitTransaction(tx.serializedTx);\n    } catch (e) {\n      this.logger.warn(\n        `Failed to submit tx [${tx.txId}] to chain [${tx.chain}]: ${e}`\n      );\n      if (e instanceof Error && e.stack) this.logger.warn(e.stack);\n    }\n    await this.setTxStatus(tx, TransactionStatus.SENT);\n  };\n\n  /**\n   * processes the sent transaction\n   * @param tx\n   */\n  protected processesSentTx = async (tx: TransactionEntity): Promise<void> => {\n    const manager = this.getChainManager(tx.chain);\n\n    const txConfirmation = await manager.getTxConfirmation(tx.txId);\n    const requiredConfirmation = manager.getTxRequiredConfirmation(tx.txType);\n\n    if (txConfirmation >= requiredConfirmation) {\n      // tx is confirmed enough\n      await this.setTxStatus(tx, TransactionStatus.COMPLETED);\n    } else if (txConfirmation === -1) {\n      // tx is not mined, checking mempool...\n      if (await manager.isTxInMempool(tx.txId)) {\n        // tx is in mempool, updating last check...\n        const height = await manager.getHeight();\n        await this.updateTxLastCheck(tx.txId, tx.chain, height);\n        this.logger.info(`Tx [${tx.txId}] is in mempool`);\n      } else {\n        // tx is not in mempool, checking if tx is still valid\n        const isValidTx = await manager.isTxValid(\n          tx.serializedTx,\n          SigningStatus.Signed\n        );\n        const isValidToType = await this.validateTx(tx);\n\n        if (isValidTx && isValidToType) {\n          // tx is valid. resending...\n          this.logger.info(`Tx [${tx.txId}] is still valid. Resending tx...`);\n          await manager.submitTransaction(tx.serializedTx);\n        } else {\n          // tx seems invalid. reset status if enough blocks past.\n          await this.setTransactionAsInvalid(tx);\n        }\n      }\n    } else {\n      // tx is mined, but is not confirmed enough, updating last check...\n      const height = await manager.getHeight();\n      await this.updateTxLastCheck(tx.txId, tx.chain, height);\n      this.logger.info(\n        `Tx [${tx.txId}] is in confirmation process [${txConfirmation}/${requiredConfirmation}]`\n      );\n    }\n  };\n\n  /**\n   * runs all jobs of TxPot\n   * - process signed txs\n   * - process sent txs\n   */\n  update = async (): Promise<void> => {\n    // process signed txs\n    const signedTxs = await this.getTxsByStatus(TransactionStatus.SIGNED);\n    for (const tx of signedTxs) {\n      try {\n        await this.processSignedTx(tx);\n      } catch (e) {\n        this.logger.warn(\n          `An error occurred while processing tx [${tx.txId}] with status [${TransactionStatus.SIGNED}]: ${e}`\n        );\n        if (e instanceof Error && e.stack) this.logger.warn(e.stack);\n      }\n    }\n    this.logger.debug(\n      `Processed [${signedTxs.length}] txs with status [${TransactionStatus.SIGNED}]`\n    );\n\n    // process sent txs\n    const sentTxs = await this.getTxsByStatus(TransactionStatus.SENT);\n    for (const tx of sentTxs) {\n      try {\n        await this.processesSentTx(tx);\n      } catch (e) {\n        this.logger.warn(\n          `An error occurred while processing tx [${tx.txId}] with status [${TransactionStatus.SENT}]: ${e}`\n        );\n        if (e instanceof Error && e.stack) this.logger.warn(e.stack);\n      }\n    }\n    this.logger.debug(\n      `Processed [${sentTxs.length}] txs with status [${TransactionStatus.SENT}]`\n    );\n  };\n\n  /**\n   * gets transactions by status\n   * @param status\n   * @param validate\n   * @returns\n   */\n  getTxsByStatus = async (\n    status: TransactionStatus,\n    validate = false\n  ): Promise<Array<TransactionEntity>> => {\n    const txs = await this.txRepository.find({\n      where: {\n        status: status,\n      },\n    });\n    if (!validate) return txs;\n\n    // validate the transactions\n    const validTxs: Array<TransactionEntity> = [];\n    for (const tx of txs) {\n      if (await this.validateTx(tx)) validTxs.push(tx);\n    }\n    return validTxs;\n  };\n\n  /**\n   * inserts a new transaction into db\n   * @param txId\n   * @param chain\n   * @param txType\n   * @param requiredSign\n   * @param serializedTx\n   * @param initialStatus\n   * @param lastCheck\n   */\n  addTx = async (\n    txId: string,\n    chain: string,\n    txType: string,\n    requiredSign: number,\n    serializedTx: string,\n    initialStatus = TransactionStatus.APPROVED,\n    lastCheck = 0\n  ): Promise<void> => {\n    await this.txRepository.insert({\n      txId: txId,\n      chain: chain,\n      txType: txType,\n      status: initialStatus,\n      requiredSign: requiredSign,\n      lastCheck: lastCheck,\n      lastStatusUpdate: this.currentTime(),\n      failedInSign: false,\n      signFailedCount: 0,\n      serializedTx: serializedTx,\n    });\n  };\n\n  /**\n   * updates the status of a tx\n   * @param txId\n   * @param chain\n   * @param status new status\n   */\n  setTxStatusById = async (\n    txId: string,\n    chain: string,\n    status: TransactionStatus\n  ): Promise<void> => {\n    const tx = await this.txRepository.findOneOrFail({\n      where: { txId, chain },\n    });\n    await this.setTxStatus(tx, status);\n  };\n\n  /**\n   * updates tx info when failed in sign process\n   * @param txId\n   * @param chain\n   */\n  setTxAsSignFailed = async (txId: string, chain: string): Promise<void> => {\n    await this.txRepository.update(\n      {\n        txId: txId,\n        chain: chain,\n        status: TransactionStatus.IN_SIGN,\n      },\n      {\n        status: TransactionStatus.SIGN_FAILED,\n        lastStatusUpdate: this.currentTime(),\n        signFailedCount: () => '\"signFailedCount\" + 1',\n        failedInSign: true,\n      }\n    );\n  };\n\n  /**\n   * updates the tx and set status as signed\n   * @param txId\n   * @param chain\n   * @param serializedTx\n   * @param currentHeight current height of the blockchain\n   * @param extra\n   * @param extra2\n   */\n  setTxAsSigned = async (\n    txId: string,\n    chain: string,\n    serializedTx: string,\n    currentHeight: number,\n    extra?: string,\n    extra2?: string\n  ): Promise<void> => {\n    const updatedFields: Partial<TransactionEntity> = {\n      serializedTx: serializedTx,\n      status: TransactionStatus.SIGNED,\n      lastStatusUpdate: this.currentTime(),\n      lastCheck: currentHeight,\n    };\n    if (extra) updatedFields.extra = extra;\n    if (extra2) updatedFields.extra2 = extra2;\n\n    await this.txRepository.update({ txId, chain }, updatedFields);\n  };\n\n  /**\n   * updates last check value of a tx\n   * @param txId\n   * @param chain\n   * @param currentHeight current height of the blockchain\n   */\n  updateTxLastCheck = async (\n    txId: string,\n    chain: string,\n    currentHeight: number\n  ): Promise<void> => {\n    await this.txRepository.update(\n      { txId, chain },\n      { lastCheck: currentHeight }\n    );\n  };\n\n  /**\n   * updates failedInSign field of a transaction to false\n   * @param txId\n   * @param chain\n   */\n  resetFailedInSign = async (txId: string, chain: string): Promise<void> => {\n    await this.txRepository.update(\n      { txId, chain },\n      {\n        failedInSign: false,\n      }\n    );\n  };\n\n  /**\n   * updates requiredSign field of a transaction\n   * @param txId\n   * @param chain\n   * @param requiredSign\n   */\n  updateRequiredSign = async (\n    txId: string,\n    chain: string,\n    requiredSign: number\n  ): Promise<void> => {\n    await this.txRepository.update(\n      { txId, chain },\n      {\n        requiredSign: requiredSign,\n      }\n    );\n  };\n\n  /**\n   * gets the transaction by its id and chain\n   * @param txId\n   * @param chain\n   */\n  getTxByKey = async (\n    txId: string,\n    chain: string\n  ): Promise<TransactionEntity | null> => {\n    return await this.txRepository.findOne({\n      where: { txId, chain },\n    });\n  };\n\n  /**\n   * @returns the transactions with valid status\n   */\n  getTxsQuery = (\n    options: Array<TxOptions> = []\n  ): Promise<TransactionEntity[]> => {\n    return this.txRepository.find({\n      where: options.map(txOptionToClause),\n    });\n  };\n}\n"]}
@@ -0,0 +1,35 @@
1
+ import { TransactionEntity } from '../db/entities/TransactionEntity';
2
+ export type ChainRequiredConfirmations = Record<string, number>;
3
+ export type RequiredConfirmations = Record<string, ChainRequiredConfirmations>;
4
+ export type ValidatorFunction = (tx: TransactionEntity) => Promise<boolean>;
5
+ export type CallbackFunction = (tx: TransactionEntity, newStatus: TransactionStatus) => Promise<void>;
6
+ export declare enum TransactionStatus {
7
+ APPROVED = "approved",
8
+ IN_SIGN = "in-sign",
9
+ SIGN_FAILED = "sign-failed",
10
+ SIGNED = "signed",
11
+ SENT = "sent",
12
+ INVALID = "invalid",
13
+ COMPLETED = "completed"
14
+ }
15
+ export declare enum SigningStatus {
16
+ Signed = 0,
17
+ UnSigned = 1
18
+ }
19
+ export type FieldValue<T> = T | Array<T>;
20
+ export interface FieldOption<T> {
21
+ not: boolean;
22
+ value: FieldValue<T>;
23
+ }
24
+ export interface TxOptions {
25
+ txId?: FieldValue<string>;
26
+ chain?: string;
27
+ txType?: string;
28
+ status?: FieldOption<TransactionStatus>;
29
+ failedInSign?: boolean;
30
+ extra?: FieldValue<string>;
31
+ }
32
+ export declare class UnregisteredChain extends Error {
33
+ constructor(msg: string);
34
+ }
35
+ //# sourceMappingURL=types.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../lib/transaction/types.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,iBAAiB,EAAE,MAAM,kCAAkC,CAAC;AAErE,MAAM,MAAM,0BAA0B,GAAG,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;AAChE,MAAM,MAAM,qBAAqB,GAAG,MAAM,CAAC,MAAM,EAAE,0BAA0B,CAAC,CAAC;AAE/E,MAAM,MAAM,iBAAiB,GAAG,CAAC,EAAE,EAAE,iBAAiB,KAAK,OAAO,CAAC,OAAO,CAAC,CAAC;AAC5E,MAAM,MAAM,gBAAgB,GAAG,CAC7B,EAAE,EAAE,iBAAiB,EACrB,SAAS,EAAE,iBAAiB,KACzB,OAAO,CAAC,IAAI,CAAC,CAAC;AAEnB,oBAAY,iBAAiB;IAC3B,QAAQ,aAAa;IACrB,OAAO,YAAY;IACnB,WAAW,gBAAgB;IAC3B,MAAM,WAAW;IACjB,IAAI,SAAS;IACb,OAAO,YAAY;IACnB,SAAS,cAAc;CACxB;AAED,oBAAY,aAAa;IACvB,MAAM,IAAA;IACN,QAAQ,IAAA;CACT;AAED,MAAM,MAAM,UAAU,CAAC,CAAC,IAAI,CAAC,GAAG,KAAK,CAAC,CAAC,CAAC,CAAC;AACzC,MAAM,WAAW,WAAW,CAAC,CAAC;IAC5B,GAAG,EAAE,OAAO,CAAC;IACb,KAAK,EAAE,UAAU,CAAC,CAAC,CAAC,CAAC;CACtB;AAED,MAAM,WAAW,SAAS;IACxB,IAAI,CAAC,EAAE,UAAU,CAAC,MAAM,CAAC,CAAC;IAC1B,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,MAAM,CAAC,EAAE,WAAW,CAAC,iBAAiB,CAAC,CAAC;IACxC,YAAY,CAAC,EAAE,OAAO,CAAC;IACvB,KAAK,CAAC,EAAE,UAAU,CAAC,MAAM,CAAC,CAAC;CAC5B;AAED,qBAAa,iBAAkB,SAAQ,KAAK;gBAC9B,GAAG,EAAE,MAAM;CAGxB"}
@@ -0,0 +1,21 @@
1
+ export var TransactionStatus;
2
+ (function (TransactionStatus) {
3
+ TransactionStatus["APPROVED"] = "approved";
4
+ TransactionStatus["IN_SIGN"] = "in-sign";
5
+ TransactionStatus["SIGN_FAILED"] = "sign-failed";
6
+ TransactionStatus["SIGNED"] = "signed";
7
+ TransactionStatus["SENT"] = "sent";
8
+ TransactionStatus["INVALID"] = "invalid";
9
+ TransactionStatus["COMPLETED"] = "completed";
10
+ })(TransactionStatus || (TransactionStatus = {}));
11
+ export var SigningStatus;
12
+ (function (SigningStatus) {
13
+ SigningStatus[SigningStatus["Signed"] = 0] = "Signed";
14
+ SigningStatus[SigningStatus["UnSigned"] = 1] = "UnSigned";
15
+ })(SigningStatus || (SigningStatus = {}));
16
+ export class UnregisteredChain extends Error {
17
+ constructor(msg) {
18
+ super('UnregisteredChain: ' + msg);
19
+ }
20
+ }
21
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9saWIvdHJhbnNhY3Rpb24vdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBV0EsTUFBTSxDQUFOLElBQVksaUJBUVg7QUFSRCxXQUFZLGlCQUFpQjtJQUMzQiwwQ0FBcUIsQ0FBQTtJQUNyQix3Q0FBbUIsQ0FBQTtJQUNuQixnREFBMkIsQ0FBQTtJQUMzQixzQ0FBaUIsQ0FBQTtJQUNqQixrQ0FBYSxDQUFBO0lBQ2Isd0NBQW1CLENBQUE7SUFDbkIsNENBQXVCLENBQUE7QUFDekIsQ0FBQyxFQVJXLGlCQUFpQixLQUFqQixpQkFBaUIsUUFRNUI7QUFFRCxNQUFNLENBQU4sSUFBWSxhQUdYO0FBSEQsV0FBWSxhQUFhO0lBQ3ZCLHFEQUFNLENBQUE7SUFDTix5REFBUSxDQUFBO0FBQ1YsQ0FBQyxFQUhXLGFBQWEsS0FBYixhQUFhLFFBR3hCO0FBaUJELE1BQU0sT0FBTyxpQkFBa0IsU0FBUSxLQUFLO0lBQzFDLFlBQVksR0FBVztRQUNyQixLQUFLLENBQUMscUJBQXFCLEdBQUcsR0FBRyxDQUFDLENBQUM7SUFDckMsQ0FBQztDQUNGIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgVHJhbnNhY3Rpb25FbnRpdHkgfSBmcm9tICcuLi9kYi9lbnRpdGllcy9UcmFuc2FjdGlvbkVudGl0eSc7XG5cbmV4cG9ydCB0eXBlIENoYWluUmVxdWlyZWRDb25maXJtYXRpb25zID0gUmVjb3JkPHN0cmluZywgbnVtYmVyPjsgLy8gdHggdHlwZSA9PiByZXF1aXJlZCBudW1iZXJcbmV4cG9ydCB0eXBlIFJlcXVpcmVkQ29uZmlybWF0aW9ucyA9IFJlY29yZDxzdHJpbmcsIENoYWluUmVxdWlyZWRDb25maXJtYXRpb25zPjtcblxuZXhwb3J0IHR5cGUgVmFsaWRhdG9yRnVuY3Rpb24gPSAodHg6IFRyYW5zYWN0aW9uRW50aXR5KSA9PiBQcm9taXNlPGJvb2xlYW4+O1xuZXhwb3J0IHR5cGUgQ2FsbGJhY2tGdW5jdGlvbiA9IChcbiAgdHg6IFRyYW5zYWN0aW9uRW50aXR5LFxuICBuZXdTdGF0dXM6IFRyYW5zYWN0aW9uU3RhdHVzXG4pID0+IFByb21pc2U8dm9pZD47XG5cbmV4cG9ydCBlbnVtIFRyYW5zYWN0aW9uU3RhdHVzIHtcbiAgQVBQUk9WRUQgPSAnYXBwcm92ZWQnLFxuICBJTl9TSUdOID0gJ2luLXNpZ24nLFxuICBTSUdOX0ZBSUxFRCA9ICdzaWduLWZhaWxlZCcsXG4gIFNJR05FRCA9ICdzaWduZWQnLFxuICBTRU5UID0gJ3NlbnQnLFxuICBJTlZBTElEID0gJ2ludmFsaWQnLFxuICBDT01QTEVURUQgPSAnY29tcGxldGVkJyxcbn1cblxuZXhwb3J0IGVudW0gU2lnbmluZ1N0YXR1cyB7XG4gIFNpZ25lZCxcbiAgVW5TaWduZWQsXG59XG5cbmV4cG9ydCB0eXBlIEZpZWxkVmFsdWU8VD4gPSBUIHwgQXJyYXk8VD47XG5leHBvcnQgaW50ZXJmYWNlIEZpZWxkT3B0aW9uPFQ+IHtcbiAgbm90OiBib29sZWFuO1xuICB2YWx1ZTogRmllbGRWYWx1ZTxUPjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBUeE9wdGlvbnMge1xuICB0eElkPzogRmllbGRWYWx1ZTxzdHJpbmc+O1xuICBjaGFpbj86IHN0cmluZztcbiAgdHhUeXBlPzogc3RyaW5nO1xuICBzdGF0dXM/OiBGaWVsZE9wdGlvbjxUcmFuc2FjdGlvblN0YXR1cz47XG4gIGZhaWxlZEluU2lnbj86IGJvb2xlYW47XG4gIGV4dHJhPzogRmllbGRWYWx1ZTxzdHJpbmc+O1xufVxuXG5leHBvcnQgY2xhc3MgVW5yZWdpc3RlcmVkQ2hhaW4gZXh0ZW5kcyBFcnJvciB7XG4gIGNvbnN0cnVjdG9yKG1zZzogc3RyaW5nKSB7XG4gICAgc3VwZXIoJ1VucmVnaXN0ZXJlZENoYWluOiAnICsgbXNnKTtcbiAgfVxufVxuIl19
@@ -0,0 +1,8 @@
1
+ import { TxOptions } from './types';
2
+ /**
3
+ * converts options for fetching txs to typeorm clause
4
+ * @param options
5
+ * @returns
6
+ */
7
+ export declare const txOptionToClause: (options: TxOptions) => Record<string, any>;
8
+ //# sourceMappingURL=utils.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"utils.d.ts","sourceRoot":"","sources":["../../lib/transaction/utils.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,SAAS,EAAE,MAAM,SAAS,CAAC;AAEpC;;;;GAIG;AACH,eAAO,MAAM,gBAAgB,YAAa,SAAS,wBAkDlD,CAAC"}
@@ -0,0 +1,56 @@
1
+ import { In, Not } from 'typeorm';
2
+ /**
3
+ * converts options for fetching txs to typeorm clause
4
+ * @param options
5
+ * @returns
6
+ */
7
+ export const txOptionToClause = (options) => {
8
+ const clause = {};
9
+ // add txId clause
10
+ if (typeof options.txId === 'string') {
11
+ clause.txId = options.txId;
12
+ }
13
+ else if (Array.isArray(options.txId)) {
14
+ clause.txId = In(options.txId);
15
+ }
16
+ // add chain clause
17
+ if (typeof options.chain === 'string') {
18
+ clause.chain = options.chain;
19
+ }
20
+ // add txType clause
21
+ if (typeof options.txType === 'string') {
22
+ clause.txType = options.txType;
23
+ }
24
+ // add status clause
25
+ if (options.status) {
26
+ if (typeof options.status.value === 'string') {
27
+ if (options.status.not) {
28
+ clause.status = Not(options.status.value);
29
+ }
30
+ else {
31
+ clause.status = options.status.value;
32
+ }
33
+ }
34
+ else if (Array.isArray(options.status.value)) {
35
+ if (options.status.not) {
36
+ clause.status = Not(In(options.status.value));
37
+ }
38
+ else {
39
+ clause.status = In(options.status.value);
40
+ }
41
+ }
42
+ }
43
+ // add failedInSign clause
44
+ if (typeof options.failedInSign === 'boolean') {
45
+ clause.failedInSign = options.failedInSign;
46
+ }
47
+ // add extra clause
48
+ if (typeof options.extra === 'string') {
49
+ clause.extra = options.extra;
50
+ }
51
+ else if (Array.isArray(options.extra)) {
52
+ clause.extra = In(options.extra);
53
+ }
54
+ return clause;
55
+ };
56
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXRpbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9saWIvdHJhbnNhY3Rpb24vdXRpbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLEVBQUUsRUFBRSxHQUFHLEVBQUUsTUFBTSxTQUFTLENBQUM7QUFHbEM7Ozs7R0FJRztBQUNILE1BQU0sQ0FBQyxNQUFNLGdCQUFnQixHQUFHLENBQUMsT0FBa0IsRUFBRSxFQUFFO0lBQ3JELE1BQU0sTUFBTSxHQUF3QixFQUFFLENBQUM7SUFFdkMsa0JBQWtCO0lBQ2xCLElBQUksT0FBTyxPQUFPLENBQUMsSUFBSSxLQUFLLFFBQVEsRUFBRSxDQUFDO1FBQ3JDLE1BQU0sQ0FBQyxJQUFJLEdBQUcsT0FBTyxDQUFDLElBQUksQ0FBQztJQUM3QixDQUFDO1NBQU0sSUFBSSxLQUFLLENBQUMsT0FBTyxDQUFDLE9BQU8sQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDO1FBQ3ZDLE1BQU0sQ0FBQyxJQUFJLEdBQUcsRUFBRSxDQUFDLE9BQU8sQ0FBQyxJQUFJLENBQUMsQ0FBQztJQUNqQyxDQUFDO0lBRUQsbUJBQW1CO0lBQ25CLElBQUksT0FBTyxPQUFPLENBQUMsS0FBSyxLQUFLLFFBQVEsRUFBRSxDQUFDO1FBQ3RDLE1BQU0sQ0FBQyxLQUFLLEdBQUcsT0FBTyxDQUFDLEtBQUssQ0FBQztJQUMvQixDQUFDO0lBRUQsb0JBQW9CO0lBQ3BCLElBQUksT0FBTyxPQUFPLENBQUMsTUFBTSxLQUFLLFFBQVEsRUFBRSxDQUFDO1FBQ3ZDLE1BQU0sQ0FBQyxNQUFNLEdBQUcsT0FBTyxDQUFDLE1BQU0sQ0FBQztJQUNqQyxDQUFDO0lBRUQsb0JBQW9CO0lBQ3BCLElBQUksT0FBTyxDQUFDLE1BQU0sRUFBRSxDQUFDO1FBQ25CLElBQUksT0FBTyxPQUFPLENBQUMsTUFBTSxDQUFDLEtBQUssS0FBSyxRQUFRLEVBQUUsQ0FBQztZQUM3QyxJQUFJLE9BQU8sQ0FBQyxNQUFNLENBQUMsR0FBRyxFQUFFLENBQUM7Z0JBQ3ZCLE1BQU0sQ0FBQyxNQUFNLEdBQUcsR0FBRyxDQUFDLE9BQU8sQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLENBQUM7WUFDNUMsQ0FBQztpQkFBTSxDQUFDO2dCQUNOLE1BQU0sQ0FBQyxNQUFNLEdBQUcsT0FBTyxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUM7WUFDdkMsQ0FBQztRQUNILENBQUM7YUFBTSxJQUFJLEtBQUssQ0FBQyxPQUFPLENBQUMsT0FBTyxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsRUFBRSxDQUFDO1lBQy9DLElBQUksT0FBTyxDQUFDLE1BQU0sQ0FBQyxHQUFHLEVBQUUsQ0FBQztnQkFDdkIsTUFBTSxDQUFDLE1BQU0sR0FBRyxHQUFHLENBQUMsRUFBRSxDQUFDLE9BQU8sQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQztZQUNoRCxDQUFDO2lCQUFNLENBQUM7Z0JBQ04sTUFBTSxDQUFDLE1BQU0sR0FBRyxFQUFFLENBQUMsT0FBTyxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsQ0FBQztZQUMzQyxDQUFDO1FBQ0gsQ0FBQztJQUNILENBQUM7SUFFRCwwQkFBMEI7SUFDMUIsSUFBSSxPQUFPLE9BQU8sQ0FBQyxZQUFZLEtBQUssU0FBUyxFQUFFLENBQUM7UUFDOUMsTUFBTSxDQUFDLFlBQVksR0FBRyxPQUFPLENBQUMsWUFBWSxDQUFDO0lBQzdDLENBQUM7SUFFRCxtQkFBbUI7SUFDbkIsSUFBSSxPQUFPLE9BQU8sQ0FBQyxLQUFLLEtBQUssUUFBUSxFQUFFLENBQUM7UUFDdEMsTUFBTSxDQUFDLEtBQUssR0FBRyxPQUFPLENBQUMsS0FBSyxDQUFDO0lBQy9CLENBQUM7U0FBTSxJQUFJLEtBQUssQ0FBQyxPQUFPLENBQUMsT0FBTyxDQUFDLEtBQUssQ0FBQyxFQUFFLENBQUM7UUFDeEMsTUFBTSxDQUFDLEtBQUssR0FBRyxFQUFFLENBQUMsT0FBTyxDQUFDLEtBQUssQ0FBQyxDQUFDO0lBQ25DLENBQUM7SUFFRCxPQUFPLE1BQU0sQ0FBQztBQUNoQixDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBJbiwgTm90IH0gZnJvbSAndHlwZW9ybSc7XG5pbXBvcnQgeyBUeE9wdGlvbnMgfSBmcm9tICcuL3R5cGVzJztcblxuLyoqXG4gKiBjb252ZXJ0cyBvcHRpb25zIGZvciBmZXRjaGluZyB0eHMgdG8gdHlwZW9ybSBjbGF1c2VcbiAqIEBwYXJhbSBvcHRpb25zXG4gKiBAcmV0dXJuc1xuICovXG5leHBvcnQgY29uc3QgdHhPcHRpb25Ub0NsYXVzZSA9IChvcHRpb25zOiBUeE9wdGlvbnMpID0+IHtcbiAgY29uc3QgY2xhdXNlOiBSZWNvcmQ8c3RyaW5nLCBhbnk+ID0ge307XG5cbiAgLy8gYWRkIHR4SWQgY2xhdXNlXG4gIGlmICh0eXBlb2Ygb3B0aW9ucy50eElkID09PSAnc3RyaW5nJykge1xuICAgIGNsYXVzZS50eElkID0gb3B0aW9ucy50eElkO1xuICB9IGVsc2UgaWYgKEFycmF5LmlzQXJyYXkob3B0aW9ucy50eElkKSkge1xuICAgIGNsYXVzZS50eElkID0gSW4ob3B0aW9ucy50eElkKTtcbiAgfVxuXG4gIC8vIGFkZCBjaGFpbiBjbGF1c2VcbiAgaWYgKHR5cGVvZiBvcHRpb25zLmNoYWluID09PSAnc3RyaW5nJykge1xuICAgIGNsYXVzZS5jaGFpbiA9IG9wdGlvbnMuY2hhaW47XG4gIH1cblxuICAvLyBhZGQgdHhUeXBlIGNsYXVzZVxuICBpZiAodHlwZW9mIG9wdGlvbnMudHhUeXBlID09PSAnc3RyaW5nJykge1xuICAgIGNsYXVzZS50eFR5cGUgPSBvcHRpb25zLnR4VHlwZTtcbiAgfVxuXG4gIC8vIGFkZCBzdGF0dXMgY2xhdXNlXG4gIGlmIChvcHRpb25zLnN0YXR1cykge1xuICAgIGlmICh0eXBlb2Ygb3B0aW9ucy5zdGF0dXMudmFsdWUgPT09ICdzdHJpbmcnKSB7XG4gICAgICBpZiAob3B0aW9ucy5zdGF0dXMubm90KSB7XG4gICAgICAgIGNsYXVzZS5zdGF0dXMgPSBOb3Qob3B0aW9ucy5zdGF0dXMudmFsdWUpO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgY2xhdXNlLnN0YXR1cyA9IG9wdGlvbnMuc3RhdHVzLnZhbHVlO1xuICAgICAgfVxuICAgIH0gZWxzZSBpZiAoQXJyYXkuaXNBcnJheShvcHRpb25zLnN0YXR1cy52YWx1ZSkpIHtcbiAgICAgIGlmIChvcHRpb25zLnN0YXR1cy5ub3QpIHtcbiAgICAgICAgY2xhdXNlLnN0YXR1cyA9IE5vdChJbihvcHRpb25zLnN0YXR1cy52YWx1ZSkpO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgY2xhdXNlLnN0YXR1cyA9IEluKG9wdGlvbnMuc3RhdHVzLnZhbHVlKTtcbiAgICAgIH1cbiAgICB9XG4gIH1cblxuICAvLyBhZGQgZmFpbGVkSW5TaWduIGNsYXVzZVxuICBpZiAodHlwZW9mIG9wdGlvbnMuZmFpbGVkSW5TaWduID09PSAnYm9vbGVhbicpIHtcbiAgICBjbGF1c2UuZmFpbGVkSW5TaWduID0gb3B0aW9ucy5mYWlsZWRJblNpZ247XG4gIH1cblxuICAvLyBhZGQgZXh0cmEgY2xhdXNlXG4gIGlmICh0eXBlb2Ygb3B0aW9ucy5leHRyYSA9PT0gJ3N0cmluZycpIHtcbiAgICBjbGF1c2UuZXh0cmEgPSBvcHRpb25zLmV4dHJhO1xuICB9IGVsc2UgaWYgKEFycmF5LmlzQXJyYXkob3B0aW9ucy5leHRyYSkpIHtcbiAgICBjbGF1c2UuZXh0cmEgPSBJbihvcHRpb25zLmV4dHJhKTtcbiAgfVxuXG4gIHJldHVybiBjbGF1c2U7XG59O1xuIl19
@@ -0,0 +1,44 @@
1
+ import { Column, Entity, PrimaryColumn } from 'typeorm';
2
+
3
+ @Entity()
4
+ export class TransactionEntity {
5
+ @PrimaryColumn('varchar')
6
+ txId: string;
7
+
8
+ @PrimaryColumn('varchar')
9
+ chain: string;
10
+
11
+ @Column('varchar')
12
+ txType: string;
13
+
14
+ @Column('varchar')
15
+ status: string;
16
+
17
+ @Column('integer')
18
+ requiredSign: number;
19
+
20
+ @Column('integer')
21
+ lastCheck: number;
22
+
23
+ @Column('varchar')
24
+ lastStatusUpdate: string;
25
+
26
+ @Column('boolean')
27
+ failedInSign: boolean;
28
+
29
+ @Column('integer')
30
+ signFailedCount: number;
31
+
32
+ @Column('varchar')
33
+ serializedTx: string;
34
+
35
+ @Column('varchar', {
36
+ nullable: true,
37
+ })
38
+ extra?: string;
39
+
40
+ @Column('varchar', {
41
+ nullable: true,
42
+ })
43
+ extra2?: string;
44
+ }
@@ -0,0 +1,7 @@
1
+ import { Migration1706350644686 } from './postgres/1706350644686-migration';
2
+ import { Migration1706007154531 } from './sqlite/1706007154531-migration';
3
+
4
+ export const migrations = {
5
+ sqlite: [Migration1706007154531],
6
+ postgres: [Migration1706350644686],
7
+ };