@rosen-bridge/abstract-extractor 2.1.2 → 3.0.0-8f3c7016

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 (104) hide show
  1. package/CHANGELOG.md +9 -0
  2. package/dist/abstractExtractor.d.ts +24 -12
  3. package/dist/abstractExtractor.d.ts.map +1 -1
  4. package/dist/abstractExtractor.js +1 -1
  5. package/dist/constants.d.ts +1 -0
  6. package/dist/constants.d.ts.map +1 -1
  7. package/dist/constants.js +2 -1
  8. package/dist/ergo/database/actions/abstractErgoAction.d.ts +83 -0
  9. package/dist/ergo/database/actions/abstractErgoAction.d.ts.map +1 -0
  10. package/dist/ergo/database/actions/abstractErgoAction.js +167 -0
  11. package/dist/ergo/database/actions/abstractErgoBoxAction.d.ts +31 -0
  12. package/dist/ergo/database/actions/abstractErgoBoxAction.d.ts.map +1 -0
  13. package/dist/ergo/database/actions/abstractErgoBoxAction.js +70 -0
  14. package/dist/ergo/database/entities/abstractErgoBoxEntity.d.ts +18 -0
  15. package/dist/ergo/database/entities/abstractErgoBoxEntity.d.ts.map +1 -0
  16. package/dist/ergo/database/entities/abstractErgoBoxEntity.js +36 -0
  17. package/dist/ergo/database/entities/abstractErgoEntity.d.ts +26 -0
  18. package/dist/ergo/database/entities/abstractErgoEntity.d.ts.map +1 -0
  19. package/dist/ergo/database/entities/abstractErgoEntity.js +64 -0
  20. package/dist/ergo/database/index.d.ts +5 -0
  21. package/dist/ergo/database/index.d.ts.map +1 -0
  22. package/dist/ergo/database/index.js +5 -0
  23. package/dist/ergo/extractors/abstractErgoBoxExtractor.d.ts +67 -0
  24. package/dist/ergo/extractors/abstractErgoBoxExtractor.d.ts.map +1 -0
  25. package/dist/ergo/extractors/abstractErgoBoxExtractor.js +106 -0
  26. package/dist/ergo/extractors/abstractErgoExtractor.d.ts +53 -0
  27. package/dist/ergo/extractors/abstractErgoExtractor.d.ts.map +1 -0
  28. package/dist/ergo/extractors/abstractErgoExtractor.js +92 -0
  29. package/dist/ergo/extractors/abstractErgoTxExtractor.d.ts +56 -0
  30. package/dist/ergo/extractors/abstractErgoTxExtractor.d.ts.map +1 -0
  31. package/dist/ergo/extractors/abstractErgoTxExtractor.js +81 -0
  32. package/dist/ergo/extractors/index.d.ts +4 -0
  33. package/dist/ergo/extractors/index.d.ts.map +1 -0
  34. package/dist/ergo/extractors/index.js +4 -0
  35. package/dist/ergo/index.d.ts +5 -7
  36. package/dist/ergo/index.d.ts.map +1 -1
  37. package/dist/ergo/index.js +6 -8
  38. package/dist/ergo/initializers/ergoBoxInitializer.d.ts +36 -0
  39. package/dist/ergo/initializers/ergoBoxInitializer.d.ts.map +1 -0
  40. package/dist/ergo/initializers/ergoBoxInitializer.js +80 -0
  41. package/dist/ergo/initializers/ergoInitializer.d.ts +39 -0
  42. package/dist/ergo/initializers/ergoInitializer.d.ts.map +1 -0
  43. package/dist/ergo/initializers/ergoInitializer.js +80 -0
  44. package/dist/ergo/initializers/index.d.ts +4 -0
  45. package/dist/ergo/initializers/index.d.ts.map +1 -0
  46. package/dist/ergo/initializers/index.js +4 -0
  47. package/dist/ergo/initializers/strategies/constants.d.ts +3 -0
  48. package/dist/ergo/initializers/strategies/constants.d.ts.map +1 -0
  49. package/dist/ergo/initializers/strategies/constants.js +3 -0
  50. package/dist/ergo/initializers/strategies/explorerInitializationStrategy.d.ts +59 -0
  51. package/dist/ergo/initializers/strategies/explorerInitializationStrategy.d.ts.map +1 -0
  52. package/dist/ergo/initializers/strategies/explorerInitializationStrategy.js +141 -0
  53. package/dist/ergo/initializers/strategies/index.d.ts +4 -0
  54. package/dist/ergo/initializers/strategies/index.d.ts.map +1 -0
  55. package/dist/ergo/initializers/strategies/index.js +4 -0
  56. package/dist/ergo/initializers/strategies/nodeInitializationStrategy.d.ts +29 -0
  57. package/dist/ergo/initializers/strategies/nodeInitializationStrategy.d.ts.map +1 -0
  58. package/dist/ergo/initializers/strategies/nodeInitializationStrategy.js +66 -0
  59. package/dist/ergo/initializers/strategies/workerManager.d.ts +79 -0
  60. package/dist/ergo/initializers/strategies/workerManager.d.ts.map +1 -0
  61. package/dist/ergo/initializers/strategies/workerManager.js +183 -0
  62. package/dist/ergo/interfaces.d.ts +31 -17
  63. package/dist/ergo/interfaces.d.ts.map +1 -1
  64. package/dist/ergo/interfaces.js +1 -1
  65. package/dist/ergo/networks/explorerNetwork.d.ts +52 -0
  66. package/dist/ergo/networks/explorerNetwork.d.ts.map +1 -0
  67. package/dist/ergo/networks/explorerNetwork.js +127 -0
  68. package/dist/ergo/networks/index.d.ts +3 -0
  69. package/dist/ergo/networks/index.d.ts.map +1 -0
  70. package/dist/ergo/networks/index.js +3 -0
  71. package/dist/ergo/networks/nodeNetwork.d.ts +28 -0
  72. package/dist/ergo/networks/nodeNetwork.d.ts.map +1 -0
  73. package/dist/ergo/networks/nodeNetwork.js +59 -0
  74. package/dist/ergo/utils.d.ts +15 -0
  75. package/dist/ergo/utils.d.ts.map +1 -1
  76. package/dist/ergo/utils.js +34 -1
  77. package/package.json +3 -1
  78. package/dist/ergo/abstractErgoExtractor.d.ts +0 -80
  79. package/dist/ergo/abstractErgoExtractor.d.ts.map +0 -1
  80. package/dist/ergo/abstractErgoExtractor.js +0 -142
  81. package/dist/ergo/abstractErgoExtractorAction.d.ts +0 -89
  82. package/dist/ergo/abstractErgoExtractorAction.d.ts.map +0 -1
  83. package/dist/ergo/abstractErgoExtractorAction.js +0 -219
  84. package/dist/ergo/abstractErgoExtractorEntity.d.ts +0 -11
  85. package/dist/ergo/abstractErgoExtractorEntity.d.ts.map +0 -1
  86. package/dist/ergo/abstractErgoExtractorEntity.js +0 -57
  87. package/dist/ergo/initializable/abstractInitializable.d.ts +0 -48
  88. package/dist/ergo/initializable/abstractInitializable.d.ts.map +0 -1
  89. package/dist/ergo/initializable/abstractInitializable.js +0 -162
  90. package/dist/ergo/initializable/abstractInitializableAction.d.ts +0 -14
  91. package/dist/ergo/initializable/abstractInitializableAction.d.ts.map +0 -1
  92. package/dist/ergo/initializable/abstractInitializableAction.js +0 -16
  93. package/dist/ergo/initializable/index.d.ts +0 -3
  94. package/dist/ergo/initializable/index.d.ts.map +0 -1
  95. package/dist/ergo/initializable/index.js +0 -3
  96. package/dist/ergo/network/abstractNetwork.d.ts +0 -26
  97. package/dist/ergo/network/abstractNetwork.d.ts.map +0 -1
  98. package/dist/ergo/network/abstractNetwork.js +0 -3
  99. package/dist/ergo/network/explorerNetwork.d.ts +0 -74
  100. package/dist/ergo/network/explorerNetwork.d.ts.map +0 -1
  101. package/dist/ergo/network/explorerNetwork.js +0 -185
  102. package/dist/ergo/network/nodeNetwork.d.ts +0 -60
  103. package/dist/ergo/network/nodeNetwork.d.ts.map +0 -1
  104. package/dist/ergo/network/nodeNetwork.js +0 -131
@@ -0,0 +1,29 @@
1
+ import { DummyLogger } from '@rosen-bridge/abstract-logger';
2
+ import { BlockInfo } from '@rosen-bridge/scanner-interfaces';
3
+ import { ExtendedTransaction } from '../../interfaces';
4
+ export declare class NodeInitializationStrategy {
5
+ private address;
6
+ private maxParallelRequests;
7
+ private processTransactionBatch;
8
+ private logger;
9
+ private network;
10
+ constructor(url: string, address: string, maxParallelRequests: number, processTransactionBatch: (txs: ExtendedTransaction[]) => Promise<void>, logger?: DummyLogger);
11
+ /**
12
+ * Get the total tx count from Node network
13
+ * @returns total tx count of the address
14
+ */
15
+ private getTotalTxCount;
16
+ /**
17
+ * Processing node transactions with offset and limit
18
+ * @param offset
19
+ * @param limit
20
+ * @param initialHeight
21
+ */
22
+ private processWithOffsetLimit;
23
+ /**
24
+ * Initialize extractor using Node network
25
+ * @param initialBlock
26
+ */
27
+ initialize: (initialBlock: BlockInfo) => Promise<void>;
28
+ }
29
+ //# sourceMappingURL=nodeInitializationStrategy.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"nodeInitializationStrategy.d.ts","sourceRoot":"","sources":["../../../../lib/ergo/initializers/strategies/nodeInitializationStrategy.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,WAAW,EAAE,MAAM,+BAA+B,CAAC;AAC5D,OAAO,EAAE,SAAS,EAAE,MAAM,kCAAkC,CAAC;AAG7D,OAAO,EAAE,mBAAmB,EAAE,MAAM,kBAAkB,CAAC;AAKvD,qBAAa,0BAA0B;IAKnC,OAAO,CAAC,OAAO;IACf,OAAO,CAAC,mBAAmB;IAC3B,OAAO,CAAC,uBAAuB;IAG/B,OAAO,CAAC,MAAM;IAThB,OAAO,CAAC,OAAO,CAAc;gBAG3B,GAAG,EAAE,MAAM,EACH,OAAO,EAAE,MAAM,EACf,mBAAmB,EAAE,MAAM,EAC3B,uBAAuB,EAAE,CAC/B,GAAG,EAAE,mBAAmB,EAAE,KACvB,OAAO,CAAC,IAAI,CAAC,EACV,MAAM,cAAoB;IAKpC;;;OAGG;IACH,OAAO,CAAC,eAAe,CAOrB;IAEF;;;;;OAKG;IACH,OAAO,CAAC,sBAAsB,CAuB5B;IAEF;;;OAGG;IACH,UAAU,GAAU,cAAc,SAAS,mBAmBzC;CACH"}
@@ -0,0 +1,66 @@
1
+ import PQueue from 'p-queue';
2
+ import { DummyLogger } from '@rosen-bridge/abstract-logger';
3
+ import { API_LIMIT } from '../../../constants';
4
+ import { NodeNetwork } from '../../networks/nodeNetwork';
5
+ import { delay, requestWithRetrial } from '../../utils';
6
+ import { DELAY_BETWEEN_INIT_REQUESTS } from './constants';
7
+ export class NodeInitializationStrategy {
8
+ address;
9
+ maxParallelRequests;
10
+ processTransactionBatch;
11
+ logger;
12
+ network;
13
+ constructor(url, address, maxParallelRequests, processTransactionBatch, logger = new DummyLogger()) {
14
+ this.address = address;
15
+ this.maxParallelRequests = maxParallelRequests;
16
+ this.processTransactionBatch = processTransactionBatch;
17
+ this.logger = logger;
18
+ this.network = new NodeNetwork(url);
19
+ }
20
+ /**
21
+ * Get the total tx count from Node network
22
+ * @returns total tx count of the address
23
+ */
24
+ getTotalTxCount = async () => {
25
+ const response = await this.network.getAddressTransactionsWithOffsetLimit(this.address, 0, 0);
26
+ return response.total;
27
+ };
28
+ /**
29
+ * Processing node transactions with offset and limit
30
+ * @param offset
31
+ * @param limit
32
+ * @param initialHeight
33
+ */
34
+ processWithOffsetLimit = async (offset, limit, initialHeight) => {
35
+ this.logger.debug(`Requesting node getTxsByAddress with offset ${offset}`);
36
+ const response = await requestWithRetrial(() => this.network.getAddressTransactionsWithOffsetLimit(this.address, offset, limit), this.logger);
37
+ const txs = response.items.filter((tx) => tx.inclusionHeight <= initialHeight);
38
+ this.logger.debug(`Found ${txs.length} new transactions below the initial height with offset ${offset}, total is ${response.total}`);
39
+ if (txs.length > 0)
40
+ await this.processTransactionBatch(txs);
41
+ this.logger.debug(`Processing completed for request with offset ${offset}`);
42
+ };
43
+ /**
44
+ * Initialize extractor using Node network
45
+ * @param initialBlock
46
+ */
47
+ initialize = async (initialBlock) => {
48
+ try {
49
+ let offset = 0;
50
+ const total = await this.getTotalTxCount();
51
+ const promiseQueue = new PQueue({
52
+ concurrency: this.maxParallelRequests,
53
+ });
54
+ while (offset < total) {
55
+ await delay(DELAY_BETWEEN_INIT_REQUESTS);
56
+ ((offset) => promiseQueue.add(() => this.processWithOffsetLimit(offset, API_LIMIT, initialBlock.height)))(offset);
57
+ offset += API_LIMIT;
58
+ }
59
+ await promiseQueue.onIdle();
60
+ }
61
+ catch (error) {
62
+ throw new Error(`Error initializing extractor by node: ${error}`);
63
+ }
64
+ };
65
+ }
66
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibm9kZUluaXRpYWxpemF0aW9uU3RyYXRlZ3kuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9saWIvZXJnby9pbml0aWFsaXplcnMvc3RyYXRlZ2llcy9ub2RlSW5pdGlhbGl6YXRpb25TdHJhdGVneS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLE1BQU0sTUFBTSxTQUFTLENBQUM7QUFFN0IsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLCtCQUErQixDQUFDO0FBRzVELE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxvQkFBb0IsQ0FBQztBQUUvQyxPQUFPLEVBQUUsV0FBVyxFQUFFLE1BQU0sNEJBQTRCLENBQUM7QUFDekQsT0FBTyxFQUFFLEtBQUssRUFBRSxrQkFBa0IsRUFBRSxNQUFNLGFBQWEsQ0FBQztBQUN4RCxPQUFPLEVBQUUsMkJBQTJCLEVBQUUsTUFBTSxhQUFhLENBQUM7QUFFMUQsTUFBTSxPQUFPLDBCQUEwQjtJQUszQjtJQUNBO0lBQ0E7SUFHQTtJQVRGLE9BQU8sQ0FBYztJQUU3QixZQUNFLEdBQVcsRUFDSCxPQUFlLEVBQ2YsbUJBQTJCLEVBQzNCLHVCQUVVLEVBQ1YsU0FBUyxJQUFJLFdBQVcsRUFBRTtRQUwxQixZQUFPLEdBQVAsT0FBTyxDQUFRO1FBQ2Ysd0JBQW1CLEdBQW5CLG1CQUFtQixDQUFRO1FBQzNCLDRCQUF1QixHQUF2Qix1QkFBdUIsQ0FFYjtRQUNWLFdBQU0sR0FBTixNQUFNLENBQW9CO1FBRWxDLElBQUksQ0FBQyxPQUFPLEdBQUcsSUFBSSxXQUFXLENBQUMsR0FBRyxDQUFDLENBQUM7SUFDdEMsQ0FBQztJQUVEOzs7T0FHRztJQUNLLGVBQWUsR0FBRyxLQUFLLElBQUksRUFBRTtRQUNuQyxNQUFNLFFBQVEsR0FBRyxNQUFNLElBQUksQ0FBQyxPQUFPLENBQUMscUNBQXFDLENBQ3ZFLElBQUksQ0FBQyxPQUFPLEVBQ1osQ0FBQyxFQUNELENBQUMsQ0FDRixDQUFDO1FBQ0YsT0FBTyxRQUFRLENBQUMsS0FBSyxDQUFDO0lBQ3hCLENBQUMsQ0FBQztJQUVGOzs7OztPQUtHO0lBQ0ssc0JBQXNCLEdBQUcsS0FBSyxFQUNwQyxNQUFjLEVBQ2QsS0FBYSxFQUNiLGFBQXFCLEVBQ3JCLEVBQUU7UUFDRixJQUFJLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQywrQ0FBK0MsTUFBTSxFQUFFLENBQUMsQ0FBQztRQUMzRSxNQUFNLFFBQVEsR0FBRyxNQUFNLGtCQUFrQixDQUN2QyxHQUFHLEVBQUUsQ0FDSCxJQUFJLENBQUMsT0FBTyxDQUFDLHFDQUFxQyxDQUNoRCxJQUFJLENBQUMsT0FBTyxFQUNaLE1BQU0sRUFDTixLQUFLLENBQ04sRUFDSCxJQUFJLENBQUMsTUFBTSxDQUNaLENBQUM7UUFDRixNQUFNLEdBQUcsR0FBRyxRQUFRLENBQUMsS0FBSyxDQUFDLE1BQU0sQ0FDL0IsQ0FBQyxFQUFFLEVBQUUsRUFBRSxDQUFDLEVBQUUsQ0FBQyxlQUFlLElBQUksYUFBYSxDQUM1QyxDQUFDO1FBQ0YsSUFBSSxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQ2YsU0FBUyxHQUFHLENBQUMsTUFBTSwwREFBMEQsTUFBTSxjQUFjLFFBQVEsQ0FBQyxLQUFLLEVBQUUsQ0FDbEgsQ0FBQztRQUNGLElBQUksR0FBRyxDQUFDLE1BQU0sR0FBRyxDQUFDO1lBQUUsTUFBTSxJQUFJLENBQUMsdUJBQXVCLENBQUMsR0FBRyxDQUFDLENBQUM7UUFDNUQsSUFBSSxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsZ0RBQWdELE1BQU0sRUFBRSxDQUFDLENBQUM7SUFDOUUsQ0FBQyxDQUFDO0lBRUY7OztPQUdHO0lBQ0gsVUFBVSxHQUFHLEtBQUssRUFBRSxZQUF1QixFQUFFLEVBQUU7UUFDN0MsSUFBSSxDQUFDO1lBQ0gsSUFBSSxNQUFNLEdBQUcsQ0FBQyxDQUFDO1lBQ2YsTUFBTSxLQUFLLEdBQUcsTUFBTSxJQUFJLENBQUMsZUFBZSxFQUFFLENBQUM7WUFDM0MsTUFBTSxZQUFZLEdBQUcsSUFBSSxNQUFNLENBQUM7Z0JBQzlCLFdBQVcsRUFBRSxJQUFJLENBQUMsbUJBQW1CO2FBQ3RDLENBQUMsQ0FBQztZQUNILE9BQU8sTUFBTSxHQUFHLEtBQUssRUFBRSxDQUFDO2dCQUN0QixNQUFNLEtBQUssQ0FBQywyQkFBMkIsQ0FBQyxDQUFDO2dCQUN6QyxDQUFDLENBQUMsTUFBYyxFQUFFLEVBQUUsQ0FDbEIsWUFBWSxDQUFDLEdBQUcsQ0FBQyxHQUFHLEVBQUUsQ0FDcEIsSUFBSSxDQUFDLHNCQUFzQixDQUFDLE1BQU0sRUFBRSxTQUFTLEVBQUUsWUFBWSxDQUFDLE1BQU0sQ0FBQyxDQUNwRSxDQUFDLENBQUMsTUFBTSxDQUFDLENBQUM7Z0JBQ2IsTUFBTSxJQUFJLFNBQVMsQ0FBQztZQUN0QixDQUFDO1lBQ0QsTUFBTSxZQUFZLENBQUMsTUFBTSxFQUFFLENBQUM7UUFDOUIsQ0FBQztRQUFDLE9BQU8sS0FBSyxFQUFFLENBQUM7WUFDZixNQUFNLElBQUksS0FBSyxDQUFDLHlDQUF5QyxLQUFLLEVBQUUsQ0FBQyxDQUFDO1FBQ3BFLENBQUM7SUFDSCxDQUFDLENBQUM7Q0FDSCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBQUXVldWUgZnJvbSAncC1xdWV1ZSc7XG5cbmltcG9ydCB7IER1bW15TG9nZ2VyIH0gZnJvbSAnQHJvc2VuLWJyaWRnZS9hYnN0cmFjdC1sb2dnZXInO1xuaW1wb3J0IHsgQmxvY2tJbmZvIH0gZnJvbSAnQHJvc2VuLWJyaWRnZS9zY2FubmVyLWludGVyZmFjZXMnO1xuXG5pbXBvcnQgeyBBUElfTElNSVQgfSBmcm9tICcuLi8uLi8uLi9jb25zdGFudHMnO1xuaW1wb3J0IHsgRXh0ZW5kZWRUcmFuc2FjdGlvbiB9IGZyb20gJy4uLy4uL2ludGVyZmFjZXMnO1xuaW1wb3J0IHsgTm9kZU5ldHdvcmsgfSBmcm9tICcuLi8uLi9uZXR3b3Jrcy9ub2RlTmV0d29yayc7XG5pbXBvcnQgeyBkZWxheSwgcmVxdWVzdFdpdGhSZXRyaWFsIH0gZnJvbSAnLi4vLi4vdXRpbHMnO1xuaW1wb3J0IHsgREVMQVlfQkVUV0VFTl9JTklUX1JFUVVFU1RTIH0gZnJvbSAnLi9jb25zdGFudHMnO1xuXG5leHBvcnQgY2xhc3MgTm9kZUluaXRpYWxpemF0aW9uU3RyYXRlZ3kge1xuICBwcml2YXRlIG5ldHdvcms6IE5vZGVOZXR3b3JrO1xuXG4gIGNvbnN0cnVjdG9yKFxuICAgIHVybDogc3RyaW5nLFxuICAgIHByaXZhdGUgYWRkcmVzczogc3RyaW5nLFxuICAgIHByaXZhdGUgbWF4UGFyYWxsZWxSZXF1ZXN0czogbnVtYmVyLFxuICAgIHByaXZhdGUgcHJvY2Vzc1RyYW5zYWN0aW9uQmF0Y2g6IChcbiAgICAgIHR4czogRXh0ZW5kZWRUcmFuc2FjdGlvbltdLFxuICAgICkgPT4gUHJvbWlzZTx2b2lkPixcbiAgICBwcml2YXRlIGxvZ2dlciA9IG5ldyBEdW1teUxvZ2dlcigpLFxuICApIHtcbiAgICB0aGlzLm5ldHdvcmsgPSBuZXcgTm9kZU5ldHdvcmsodXJsKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBHZXQgdGhlIHRvdGFsIHR4IGNvdW50IGZyb20gTm9kZSBuZXR3b3JrXG4gICAqIEByZXR1cm5zIHRvdGFsIHR4IGNvdW50IG9mIHRoZSBhZGRyZXNzXG4gICAqL1xuICBwcml2YXRlIGdldFRvdGFsVHhDb3VudCA9IGFzeW5jICgpID0+IHtcbiAgICBjb25zdCByZXNwb25zZSA9IGF3YWl0IHRoaXMubmV0d29yay5nZXRBZGRyZXNzVHJhbnNhY3Rpb25zV2l0aE9mZnNldExpbWl0KFxuICAgICAgdGhpcy5hZGRyZXNzLFxuICAgICAgMCxcbiAgICAgIDAsXG4gICAgKTtcbiAgICByZXR1cm4gcmVzcG9uc2UudG90YWw7XG4gIH07XG5cbiAgLyoqXG4gICAqIFByb2Nlc3Npbmcgbm9kZSB0cmFuc2FjdGlvbnMgd2l0aCBvZmZzZXQgYW5kIGxpbWl0XG4gICAqIEBwYXJhbSBvZmZzZXRcbiAgICogQHBhcmFtIGxpbWl0XG4gICAqIEBwYXJhbSBpbml0aWFsSGVpZ2h0XG4gICAqL1xuICBwcml2YXRlIHByb2Nlc3NXaXRoT2Zmc2V0TGltaXQgPSBhc3luYyAoXG4gICAgb2Zmc2V0OiBudW1iZXIsXG4gICAgbGltaXQ6IG51bWJlcixcbiAgICBpbml0aWFsSGVpZ2h0OiBudW1iZXIsXG4gICkgPT4ge1xuICAgIHRoaXMubG9nZ2VyLmRlYnVnKGBSZXF1ZXN0aW5nIG5vZGUgZ2V0VHhzQnlBZGRyZXNzIHdpdGggb2Zmc2V0ICR7b2Zmc2V0fWApO1xuICAgIGNvbnN0IHJlc3BvbnNlID0gYXdhaXQgcmVxdWVzdFdpdGhSZXRyaWFsKFxuICAgICAgKCkgPT5cbiAgICAgICAgdGhpcy5uZXR3b3JrLmdldEFkZHJlc3NUcmFuc2FjdGlvbnNXaXRoT2Zmc2V0TGltaXQoXG4gICAgICAgICAgdGhpcy5hZGRyZXNzLFxuICAgICAgICAgIG9mZnNldCxcbiAgICAgICAgICBsaW1pdCxcbiAgICAgICAgKSxcbiAgICAgIHRoaXMubG9nZ2VyLFxuICAgICk7XG4gICAgY29uc3QgdHhzID0gcmVzcG9uc2UuaXRlbXMuZmlsdGVyKFxuICAgICAgKHR4KSA9PiB0eC5pbmNsdXNpb25IZWlnaHQgPD0gaW5pdGlhbEhlaWdodCxcbiAgICApO1xuICAgIHRoaXMubG9nZ2VyLmRlYnVnKFxuICAgICAgYEZvdW5kICR7dHhzLmxlbmd0aH0gbmV3IHRyYW5zYWN0aW9ucyBiZWxvdyB0aGUgaW5pdGlhbCBoZWlnaHQgd2l0aCBvZmZzZXQgJHtvZmZzZXR9LCB0b3RhbCBpcyAke3Jlc3BvbnNlLnRvdGFsfWAsXG4gICAgKTtcbiAgICBpZiAodHhzLmxlbmd0aCA+IDApIGF3YWl0IHRoaXMucHJvY2Vzc1RyYW5zYWN0aW9uQmF0Y2godHhzKTtcbiAgICB0aGlzLmxvZ2dlci5kZWJ1ZyhgUHJvY2Vzc2luZyBjb21wbGV0ZWQgZm9yIHJlcXVlc3Qgd2l0aCBvZmZzZXQgJHtvZmZzZXR9YCk7XG4gIH07XG5cbiAgLyoqXG4gICAqIEluaXRpYWxpemUgZXh0cmFjdG9yIHVzaW5nIE5vZGUgbmV0d29ya1xuICAgKiBAcGFyYW0gaW5pdGlhbEJsb2NrXG4gICAqL1xuICBpbml0aWFsaXplID0gYXN5bmMgKGluaXRpYWxCbG9jazogQmxvY2tJbmZvKSA9PiB7XG4gICAgdHJ5IHtcbiAgICAgIGxldCBvZmZzZXQgPSAwO1xuICAgICAgY29uc3QgdG90YWwgPSBhd2FpdCB0aGlzLmdldFRvdGFsVHhDb3VudCgpO1xuICAgICAgY29uc3QgcHJvbWlzZVF1ZXVlID0gbmV3IFBRdWV1ZSh7XG4gICAgICAgIGNvbmN1cnJlbmN5OiB0aGlzLm1heFBhcmFsbGVsUmVxdWVzdHMsXG4gICAgICB9KTtcbiAgICAgIHdoaWxlIChvZmZzZXQgPCB0b3RhbCkge1xuICAgICAgICBhd2FpdCBkZWxheShERUxBWV9CRVRXRUVOX0lOSVRfUkVRVUVTVFMpO1xuICAgICAgICAoKG9mZnNldDogbnVtYmVyKSA9PlxuICAgICAgICAgIHByb21pc2VRdWV1ZS5hZGQoKCkgPT5cbiAgICAgICAgICAgIHRoaXMucHJvY2Vzc1dpdGhPZmZzZXRMaW1pdChvZmZzZXQsIEFQSV9MSU1JVCwgaW5pdGlhbEJsb2NrLmhlaWdodCksXG4gICAgICAgICAgKSkob2Zmc2V0KTtcbiAgICAgICAgb2Zmc2V0ICs9IEFQSV9MSU1JVDtcbiAgICAgIH1cbiAgICAgIGF3YWl0IHByb21pc2VRdWV1ZS5vbklkbGUoKTtcbiAgICB9IGNhdGNoIChlcnJvcikge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKGBFcnJvciBpbml0aWFsaXppbmcgZXh0cmFjdG9yIGJ5IG5vZGU6ICR7ZXJyb3J9YCk7XG4gICAgfVxuICB9O1xufVxuIl19
@@ -0,0 +1,79 @@
1
+ import { DummyLogger } from '@rosen-bridge/abstract-logger';
2
+ import { RangeQuery } from '../../interfaces';
3
+ export declare class WorkerManager {
4
+ private workerCount;
5
+ private getRangeTxCount;
6
+ private logger;
7
+ private workersRangeList;
8
+ private mutex;
9
+ private initialSegmentSize;
10
+ private maxHeight?;
11
+ constructor(workerCount: number, getRangeTxCount: (start: number, end: number) => Promise<number>, logger?: DummyLogger);
12
+ /**
13
+ * Setup the worker manager to start working
14
+ * @param maxHeight
15
+ */
16
+ setup: (maxHeight: number) => void;
17
+ /**
18
+ * Checks if the worker's search range is completed
19
+ * @param workerIndex
20
+ * @returns true if worker has an incomplete range to work on
21
+ */
22
+ isWorkerActive: (workerIndex: number) => boolean;
23
+ /**
24
+ * Register the worker for the first time
25
+ * splits the whole range equally between workers
26
+ * @param workerIndex
27
+ */
28
+ registerWorker: (workerIndex: number) => Promise<void>;
29
+ /**
30
+ * Add a new range to the workers range list
31
+ * new range is a subset of all existing ranges
32
+ * @param workerIndex
33
+ * @param start
34
+ * @param end
35
+ */
36
+ private addWorkerRange;
37
+ /**
38
+ * Limit the last range by adding a new range query to the workers range list
39
+ * the new range the first half of the last existing range
40
+ * @param workerIndex
41
+ */
42
+ limitLastRange: (workerIndex: number) => Promise<void>;
43
+ /**
44
+ * Pop the last range query after being processed
45
+ * also update all older ranges and deduct the processed numbers
46
+ * @param workerIndex
47
+ */
48
+ popLastRangeQuery: (workerIndex: number) => Promise<void>;
49
+ /**
50
+ * Find the biggest incomplete range list containing at least 2 ranges
51
+ * the range list must have at least two range so that it wont become empty
52
+ * after splitting the range between two workers
53
+ * @returns biggest incomplete range list index
54
+ */
55
+ private getBiggestIncompleteRangeIndex;
56
+ /**
57
+ * Check all workers and reassign incomplete ranges to the idle ones
58
+ * do nothing if all workers are active
59
+ * do nothing if there is no incomplete range list
60
+ * split the biggest range list between the current worker and the idle worker
61
+ * @returns
62
+ */
63
+ reassignIdleWorkers: () => Promise<number[]>;
64
+ /**
65
+ * Get the last (biggest) processable range from the worker's range list
66
+ *
67
+ * After the last range is processed, the older ones are updated and usually
68
+ * become smaller. Some of them might shrink below a certain threshold, which
69
+ * means they can now be processed too.
70
+ * This function finds the biggest range that is small enough to be processed,
71
+ * and removes all of its child ranges from the list.
72
+ * Returns the last range if none of them is processable.
73
+ * @param workerIndex
74
+ * @param rangeThreshold
75
+ * @returns range query
76
+ */
77
+ getLastRange: (workerIndex: number, rangeThreshold: number) => Promise<RangeQuery>;
78
+ }
79
+ //# sourceMappingURL=workerManager.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"workerManager.d.ts","sourceRoot":"","sources":["../../../../lib/ergo/initializers/strategies/workerManager.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,WAAW,EAAE,MAAM,+BAA+B,CAAC;AAE5D,OAAO,EAAa,UAAU,EAAE,MAAM,kBAAkB,CAAC;AAEzD,qBAAa,aAAa;IAOtB,OAAO,CAAC,WAAW;IACnB,OAAO,CAAC,eAAe;IACvB,OAAO,CAAC,MAAM;IARhB,OAAO,CAAC,gBAAgB,CAAc;IACtC,OAAO,CAAC,KAAK,CAAe;IAC5B,OAAO,CAAC,kBAAkB,CAAS;IACnC,OAAO,CAAC,SAAS,CAAC,CAAS;gBAGjB,WAAW,EAAE,MAAM,EACnB,eAAe,EAAE,CAAC,KAAK,EAAE,MAAM,EAAE,GAAG,EAAE,MAAM,KAAK,OAAO,CAAC,MAAM,CAAC,EAChE,MAAM,cAAoB;IAGpC;;;OAGG;IACH,KAAK,GAAI,WAAW,MAAM,UAIxB;IAEF;;;;OAIG;IACH,cAAc,GAAI,aAAa,MAAM,KAAG,OAAO,CAE7C;IAEF;;;;OAIG;IACH,cAAc,GAAU,aAAa,MAAM,mBAYzC;IAEF;;;;;;OAMG;IACH,OAAO,CAAC,cAAc,CAapB;IAEF;;;;OAIG;IACH,cAAc,GAAU,aAAa,MAAM,mBAazC;IAEF;;;;OAIG;IACH,iBAAiB,GAAU,aAAa,MAAM,mBAU5C;IAEF;;;;;OAKG;IACH,OAAO,CAAC,8BAA8B,CAWpC;IAEF;;;;;;OAMG;IACH,mBAAmB,QAAa,OAAO,CAAC,MAAM,EAAE,CAAC,CA+C/C;IAEF;;;;;;;;;;;;OAYG;IACH,YAAY,GACV,aAAa,MAAM,EACnB,gBAAgB,MAAM,KACrB,OAAO,CAAC,UAAU,CAAC,CAkBpB;CACH"}
@@ -0,0 +1,183 @@
1
+ import { Mutex } from 'await-semaphore';
2
+ import { DummyLogger } from '@rosen-bridge/abstract-logger';
3
+ export class WorkerManager {
4
+ workerCount;
5
+ getRangeTxCount;
6
+ logger;
7
+ workersRangeList;
8
+ mutex = new Mutex();
9
+ initialSegmentSize;
10
+ maxHeight;
11
+ constructor(workerCount, getRangeTxCount, logger = new DummyLogger()) {
12
+ this.workerCount = workerCount;
13
+ this.getRangeTxCount = getRangeTxCount;
14
+ this.logger = logger;
15
+ }
16
+ /**
17
+ * Setup the worker manager to start working
18
+ * @param maxHeight
19
+ */
20
+ setup = (maxHeight) => {
21
+ this.maxHeight = maxHeight;
22
+ this.workersRangeList = Array.from({ length: this.workerCount }, () => []);
23
+ this.initialSegmentSize = Math.ceil(this.maxHeight / this.workerCount);
24
+ };
25
+ /**
26
+ * Checks if the worker's search range is completed
27
+ * @param workerIndex
28
+ * @returns true if worker has an incomplete range to work on
29
+ */
30
+ isWorkerActive = (workerIndex) => {
31
+ return this.workersRangeList[workerIndex].length > 0;
32
+ };
33
+ /**
34
+ * Register the worker for the first time
35
+ * splits the whole range equally between workers
36
+ * @param workerIndex
37
+ */
38
+ registerWorker = async (workerIndex) => {
39
+ if (!this.maxHeight)
40
+ throw Error('Please call setup before registering workers');
41
+ const start = workerIndex * this.initialSegmentSize;
42
+ const end = Math.min((workerIndex + 1) * this.initialSegmentSize - 1, this.maxHeight);
43
+ await this.addWorkerRange(workerIndex, start, end);
44
+ this.logger.debug(`Worker-${workerIndex} is initialized with range [${start}, ${end}]`);
45
+ };
46
+ /**
47
+ * Add a new range to the workers range list
48
+ * new range is a subset of all existing ranges
49
+ * @param workerIndex
50
+ * @param start
51
+ * @param end
52
+ */
53
+ addWorkerRange = async (workerIndex, start, end) => {
54
+ const count = await this.getRangeTxCount(start, end);
55
+ const release = await this.mutex.acquire();
56
+ this.workersRangeList[workerIndex].push({
57
+ start,
58
+ end,
59
+ count,
60
+ });
61
+ release();
62
+ };
63
+ /**
64
+ * Limit the last range by adding a new range query to the workers range list
65
+ * the new range the first half of the last existing range
66
+ * @param workerIndex
67
+ */
68
+ limitLastRange = async (workerIndex) => {
69
+ const lastRangeQuery = this.workersRangeList[workerIndex].at(-1);
70
+ if (!lastRangeQuery)
71
+ throw Error('Impossible case, worker does not have any range query');
72
+ const newQueryEnd = Math.floor((lastRangeQuery.end - lastRangeQuery.start) / 2) +
73
+ lastRangeQuery.start;
74
+ this.logger.debug(`Limiting the range by adding a new range query [${(lastRangeQuery.start, newQueryEnd)}]`);
75
+ await this.addWorkerRange(workerIndex, lastRangeQuery.start, newQueryEnd);
76
+ };
77
+ /**
78
+ * Pop the last range query after being processed
79
+ * also update all older ranges and deduct the processed numbers
80
+ * @param workerIndex
81
+ */
82
+ popLastRangeQuery = async (workerIndex) => {
83
+ const release = await this.mutex.acquire();
84
+ const lastRangeQuery = this.workersRangeList[workerIndex].pop();
85
+ if (!lastRangeQuery)
86
+ throw Error('Impossible case, worker does not have any range query');
87
+ this.workersRangeList[workerIndex].map((rangeQuery) => {
88
+ rangeQuery.count -= lastRangeQuery.count;
89
+ rangeQuery.start = lastRangeQuery.end + 1;
90
+ });
91
+ release();
92
+ };
93
+ /**
94
+ * Find the biggest incomplete range list containing at least 2 ranges
95
+ * the range list must have at least two range so that it wont become empty
96
+ * after splitting the range between two workers
97
+ * @returns biggest incomplete range list index
98
+ */
99
+ getBiggestIncompleteRangeIndex = () => {
100
+ let biggestRange = 0, biggestRangeIndex = undefined;
101
+ for (let i = 0; i < this.workersRangeList.length; i++) {
102
+ const rangeList = this.workersRangeList[i];
103
+ if (rangeList.length > 1 && rangeList[0].count > biggestRange) {
104
+ biggestRange = rangeList[0].count;
105
+ biggestRangeIndex = i;
106
+ }
107
+ }
108
+ return biggestRangeIndex;
109
+ };
110
+ /**
111
+ * Check all workers and reassign incomplete ranges to the idle ones
112
+ * do nothing if all workers are active
113
+ * do nothing if there is no incomplete range list
114
+ * split the biggest range list between the current worker and the idle worker
115
+ * @returns
116
+ */
117
+ reassignIdleWorkers = async () => {
118
+ this.logger.debug('Checking idle workers to reassign');
119
+ const release = await this.mutex.acquire();
120
+ const idleWorkerIndexes = this.workersRangeList
121
+ .map((_, index) => (this.isWorkerActive(index) ? -1 : index))
122
+ .filter((index) => index !== -1);
123
+ if (idleWorkerIndexes.length == 0) {
124
+ this.logger.debug(`There is no idle worker to reassign`);
125
+ release();
126
+ return [];
127
+ }
128
+ this.logger.debug(`Found idle workers ${idleWorkerIndexes}`);
129
+ this.logger.debug(`Workers range lists before splitting ${JSON.stringify(this.workersRangeList)}`);
130
+ const newWorkers = [];
131
+ for (const idleWorkerIndex of idleWorkerIndexes) {
132
+ this.logger.debug(`reassigning job to worker-${idleWorkerIndex}`);
133
+ const biggestIncompleteRangeIndex = this.getBiggestIncompleteRangeIndex();
134
+ if (biggestIncompleteRangeIndex == undefined) {
135
+ this.logger.debug(`there is no incomplete range to reassign`);
136
+ break;
137
+ }
138
+ this.logger.debug(`biggest incomplete range is ${JSON.stringify(this.workersRangeList[biggestIncompleteRangeIndex])}`);
139
+ const removedRangeQuery = this.workersRangeList[biggestIncompleteRangeIndex].shift();
140
+ const head = this.workersRangeList[biggestIncompleteRangeIndex][0];
141
+ this.workersRangeList[idleWorkerIndex] = [
142
+ {
143
+ start: head.end + 1,
144
+ end: removedRangeQuery.end,
145
+ count: removedRangeQuery.count - head.count,
146
+ },
147
+ ];
148
+ newWorkers.push(idleWorkerIndex);
149
+ }
150
+ release();
151
+ this.logger.debug(`Range lists after splitting ${JSON.stringify(this.workersRangeList)}`);
152
+ return newWorkers;
153
+ };
154
+ /**
155
+ * Get the last (biggest) processable range from the worker's range list
156
+ *
157
+ * After the last range is processed, the older ones are updated and usually
158
+ * become smaller. Some of them might shrink below a certain threshold, which
159
+ * means they can now be processed too.
160
+ * This function finds the biggest range that is small enough to be processed,
161
+ * and removes all of its child ranges from the list.
162
+ * Returns the last range if none of them is processable.
163
+ * @param workerIndex
164
+ * @param rangeThreshold
165
+ * @returns range query
166
+ */
167
+ getLastRange = async (workerIndex, rangeThreshold) => {
168
+ const rangeList = this.workersRangeList[workerIndex];
169
+ this.logger.debug(`Search range of worker-${workerIndex} is ${JSON.stringify(rangeList)}`);
170
+ let lastRangeQuery = rangeList.at(-1);
171
+ // avoid race condition when updating the shared `rangeList` variable
172
+ const release = await this.mutex.acquire();
173
+ // jump over small ranges to select the biggest processable range
174
+ while (rangeList.length > 1 &&
175
+ (rangeList.at(-2).count <= rangeThreshold || lastRangeQuery.count == 0)) {
176
+ rangeList.pop();
177
+ lastRangeQuery = rangeList.at(-1);
178
+ }
179
+ release();
180
+ return lastRangeQuery;
181
+ };
182
+ }
183
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"workerManager.js","sourceRoot":"","sources":["../../../../lib/ergo/initializers/strategies/workerManager.ts"],"names":[],"mappings":"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","sourcesContent":["import { Mutex } from 'await-semaphore';\n\nimport { DummyLogger } from '@rosen-bridge/abstract-logger';\n\nimport { RangeList, RangeQuery } from '../../interfaces';\n\nexport class WorkerManager {\n  private workersRangeList: RangeList[];\n  private mutex = new Mutex();\n  private initialSegmentSize: number;\n  private maxHeight?: number;\n\n  constructor(\n    private workerCount: number,\n    private getRangeTxCount: (start: number, end: number) => Promise<number>,\n    private logger = new DummyLogger(),\n  ) {}\n\n  /**\n   * Setup the worker manager to start working\n   * @param maxHeight\n   */\n  setup = (maxHeight: number) => {\n    this.maxHeight = maxHeight;\n    this.workersRangeList = Array.from({ length: this.workerCount }, () => []);\n    this.initialSegmentSize = Math.ceil(this.maxHeight / this.workerCount);\n  };\n\n  /**\n   * Checks if the worker's search range is completed\n   * @param workerIndex\n   * @returns true if worker has an incomplete range to work on\n   */\n  isWorkerActive = (workerIndex: number): boolean => {\n    return this.workersRangeList[workerIndex].length > 0;\n  };\n\n  /**\n   * Register the worker for the first time\n   * splits the whole range equally between workers\n   * @param workerIndex\n   */\n  registerWorker = async (workerIndex: number) => {\n    if (!this.maxHeight)\n      throw Error('Please call setup before registering workers');\n    const start = workerIndex * this.initialSegmentSize;\n    const end = Math.min(\n      (workerIndex + 1) * this.initialSegmentSize - 1,\n      this.maxHeight,\n    );\n    await this.addWorkerRange(workerIndex, start, end);\n    this.logger.debug(\n      `Worker-${workerIndex} is initialized with range [${start}, ${end}]`,\n    );\n  };\n\n  /**\n   * Add a new range to the workers range list\n   * new range is a subset of all existing ranges\n   * @param workerIndex\n   * @param start\n   * @param end\n   */\n  private addWorkerRange = async (\n    workerIndex: number,\n    start: number,\n    end: number,\n  ) => {\n    const count = await this.getRangeTxCount(start, end);\n    const release = await this.mutex.acquire();\n    this.workersRangeList[workerIndex].push({\n      start,\n      end,\n      count,\n    });\n    release();\n  };\n\n  /**\n   * Limit the last range by adding a new range query to the workers range list\n   * the new range the first half of the last existing range\n   * @param workerIndex\n   */\n  limitLastRange = async (workerIndex: number) => {\n    const lastRangeQuery = this.workersRangeList[workerIndex].at(-1);\n    if (!lastRangeQuery)\n      throw Error('Impossible case, worker does not have any range query');\n    const newQueryEnd =\n      Math.floor((lastRangeQuery.end - lastRangeQuery.start) / 2) +\n      lastRangeQuery.start;\n    this.logger.debug(\n      `Limiting the range by adding a new range query [${\n        (lastRangeQuery.start, newQueryEnd)\n      }]`,\n    );\n    await this.addWorkerRange(workerIndex, lastRangeQuery.start, newQueryEnd);\n  };\n\n  /**\n   * Pop the last range query after being processed\n   * also update all older ranges and deduct the processed numbers\n   * @param workerIndex\n   */\n  popLastRangeQuery = async (workerIndex: number) => {\n    const release = await this.mutex.acquire();\n    const lastRangeQuery = this.workersRangeList[workerIndex].pop();\n    if (!lastRangeQuery)\n      throw Error('Impossible case, worker does not have any range query');\n    this.workersRangeList[workerIndex].map((rangeQuery) => {\n      rangeQuery.count -= lastRangeQuery.count;\n      rangeQuery.start = lastRangeQuery.end + 1;\n    });\n    release();\n  };\n\n  /**\n   * Find the biggest incomplete range list containing at least 2 ranges\n   * the range list must have at least two range so that it wont become empty\n   * after splitting the range between two workers\n   * @returns biggest incomplete range list index\n   */\n  private getBiggestIncompleteRangeIndex = (): number | undefined => {\n    let biggestRange = 0,\n      biggestRangeIndex = undefined;\n    for (let i = 0; i < this.workersRangeList.length; i++) {\n      const rangeList = this.workersRangeList[i];\n      if (rangeList.length > 1 && rangeList[0].count > biggestRange) {\n        biggestRange = rangeList[0].count;\n        biggestRangeIndex = i;\n      }\n    }\n    return biggestRangeIndex;\n  };\n\n  /**\n   * Check all workers and reassign incomplete ranges to the idle ones\n   * do nothing if all workers are active\n   * do nothing if there is no incomplete range list\n   * split the biggest range list between the current worker and the idle worker\n   * @returns\n   */\n  reassignIdleWorkers = async (): Promise<number[]> => {\n    this.logger.debug('Checking idle workers to reassign');\n    const release = await this.mutex.acquire();\n    const idleWorkerIndexes = this.workersRangeList\n      .map((_, index) => (this.isWorkerActive(index) ? -1 : index))\n      .filter((index) => index !== -1);\n    if (idleWorkerIndexes.length == 0) {\n      this.logger.debug(`There is no idle worker to reassign`);\n      release();\n      return [];\n    }\n    this.logger.debug(`Found idle workers ${idleWorkerIndexes}`);\n    this.logger.debug(\n      `Workers range lists before splitting ${JSON.stringify(\n        this.workersRangeList,\n      )}`,\n    );\n    const newWorkers: number[] = [];\n    for (const idleWorkerIndex of idleWorkerIndexes) {\n      this.logger.debug(`reassigning job to worker-${idleWorkerIndex}`);\n      const biggestIncompleteRangeIndex = this.getBiggestIncompleteRangeIndex();\n      if (biggestIncompleteRangeIndex == undefined) {\n        this.logger.debug(`there is no incomplete range to reassign`);\n        break;\n      }\n      this.logger.debug(\n        `biggest incomplete range is ${JSON.stringify(\n          this.workersRangeList[biggestIncompleteRangeIndex],\n        )}`,\n      );\n      const removedRangeQuery =\n        this.workersRangeList[biggestIncompleteRangeIndex].shift()!;\n      const head = this.workersRangeList[biggestIncompleteRangeIndex][0];\n      this.workersRangeList[idleWorkerIndex] = [\n        {\n          start: head.end + 1,\n          end: removedRangeQuery.end,\n          count: removedRangeQuery.count - head.count,\n        },\n      ];\n      newWorkers.push(idleWorkerIndex);\n    }\n    release();\n    this.logger.debug(\n      `Range lists after splitting ${JSON.stringify(this.workersRangeList)}`,\n    );\n    return newWorkers;\n  };\n\n  /**\n   * Get the last (biggest) processable range from the worker's range list\n   *\n   * After the last range is processed, the older ones are updated and usually\n   * become smaller. Some of them might shrink below a certain threshold, which\n   * means they can now be processed too.\n   * This function finds the biggest range that is small enough to be processed,\n   * and removes all of its child ranges from the list.\n   * Returns the last range if none of them is processable.\n   * @param workerIndex\n   * @param rangeThreshold\n   * @returns range query\n   */\n  getLastRange = async (\n    workerIndex: number,\n    rangeThreshold: number,\n  ): Promise<RangeQuery> => {\n    const rangeList = this.workersRangeList[workerIndex];\n    this.logger.debug(\n      `Search range of worker-${workerIndex} is ${JSON.stringify(rangeList)}`,\n    );\n    let lastRangeQuery = rangeList.at(-1)!;\n    // avoid race condition when updating the shared `rangeList` variable\n    const release = await this.mutex.acquire();\n    // jump over small ranges to select the biggest processable range\n    while (\n      rangeList.length > 1 &&\n      (rangeList.at(-2)!.count <= rangeThreshold || lastRangeQuery.count == 0)\n    ) {\n      rangeList.pop()!;\n      lastRangeQuery = rangeList.at(-1)!;\n    }\n    release();\n    return lastRangeQuery;\n  };\n}\n"]}
@@ -1,15 +1,8 @@
1
- import { OutputBox, Transaction } from '@rosen-bridge/scanner-interfaces';
2
- export interface ErgoBox extends OutputBox {
3
- blockId: string;
4
- inclusionHeight: number;
5
- spentBlockId?: string;
6
- spentHeight?: number;
7
- spentTransactionId?: string;
8
- spentIndex?: number;
9
- }
1
+ import { ErgoNetworkType, OutputBox, Transaction } from '@rosen-bridge/scanner-interfaces';
10
2
  export interface ExtendedTransaction extends Transaction {
11
3
  inclusionHeight: number;
12
4
  blockId: string;
5
+ inputs: OutputBox[];
13
6
  }
14
7
  export interface SpendInfo {
15
8
  boxId: string;
@@ -19,8 +12,12 @@ export interface SpendInfo {
19
12
  [key: string]: string;
20
13
  };
21
14
  }
22
- export interface AbstractBoxData {
23
- boxId: string;
15
+ export interface ExtendedSpendInfo extends SpendInfo {
16
+ height: number;
17
+ block: string;
18
+ }
19
+ export interface AbstractEntityData {
20
+ identifier: string;
24
21
  serialized: string;
25
22
  }
26
23
  export declare enum CallbackType {
@@ -29,19 +26,36 @@ export declare enum CallbackType {
29
26
  Spend = "spend",
30
27
  Delete = "delete"
31
28
  }
32
- export interface BoxInfo {
33
- boxId: string;
29
+ export interface EntityInfo {
30
+ identifier: string;
34
31
  }
35
- export type CallbackDataMap<ExtractedData extends AbstractBoxData> = {
36
- [CallbackType.Update]: BoxInfo[];
32
+ export type CallbackDataMap<ExtractedData extends AbstractEntityData> = {
33
+ [CallbackType.Update]: EntityInfo[];
37
34
  [CallbackType.Insert]: ExtractedData[];
38
35
  [CallbackType.Delete]: ExtractedData[];
39
- [CallbackType.Spend]: BoxInfo[];
36
+ [CallbackType.Spend]: EntityInfo[];
40
37
  };
41
- export type CallbackMap<ExtractedData extends AbstractBoxData> = {
38
+ export type CallbackMap<ExtractedData extends AbstractEntityData> = {
42
39
  [K in CallbackType]: (data: CallbackDataMap<ExtractedData>[K]) => void;
43
40
  };
44
41
  export type TxExtra = {
45
42
  [key: string]: string;
46
43
  };
44
+ export interface RangeQuery {
45
+ start: number;
46
+ end: number;
47
+ count: number;
48
+ }
49
+ /**
50
+ * The range list is an array where each new range is a child of the previous
51
+ * one. So, any range in the list supersedes all the ranges that come after it.
52
+ * */
53
+ export type RangeList = RangeQuery[];
54
+ export interface InitializeOptions {
55
+ active: boolean;
56
+ type: ErgoNetworkType;
57
+ url: string;
58
+ address: string;
59
+ maxParallelRequests?: number;
60
+ }
47
61
  //# sourceMappingURL=interfaces.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"interfaces.d.ts","sourceRoot":"","sources":["../../lib/ergo/interfaces.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,WAAW,EAAE,MAAM,kCAAkC,CAAC;AAE1E,MAAM,WAAW,OAAQ,SAAQ,SAAS;IACxC,OAAO,EAAE,MAAM,CAAC;IAChB,eAAe,EAAE,MAAM,CAAC;IACxB,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,kBAAkB,CAAC,EAAE,MAAM,CAAC;IAC5B,UAAU,CAAC,EAAE,MAAM,CAAC;CACrB;AAED,MAAM,WAAW,mBAAoB,SAAQ,WAAW;IACtD,eAAe,EAAE,MAAM,CAAC;IACxB,OAAO,EAAE,MAAM,CAAC;CACjB;AAED,MAAM,WAAW,SAAS;IACxB,KAAK,EAAE,MAAM,CAAC;IACd,IAAI,EAAE,MAAM,CAAC;IACb,KAAK,EAAE,MAAM,CAAC;IACd,MAAM,CAAC,EAAE;QAAE,CAAC,GAAG,EAAE,MAAM,GAAG,MAAM,CAAA;KAAE,CAAC;CACpC;AAED,MAAM,WAAW,eAAe;IAC9B,KAAK,EAAE,MAAM,CAAC;IACd,UAAU,EAAE,MAAM,CAAC;CACpB;AAED,oBAAY,YAAY;IACtB,MAAM,WAAW;IACjB,MAAM,WAAW;IACjB,KAAK,UAAU;IACf,MAAM,WAAW;CAClB;AAED,MAAM,WAAW,OAAO;IACtB,KAAK,EAAE,MAAM,CAAC;CACf;AAED,MAAM,MAAM,eAAe,CAAC,aAAa,SAAS,eAAe,IAAI;IACnE,CAAC,YAAY,CAAC,MAAM,CAAC,EAAE,OAAO,EAAE,CAAC;IACjC,CAAC,YAAY,CAAC,MAAM,CAAC,EAAE,aAAa,EAAE,CAAC;IACvC,CAAC,YAAY,CAAC,MAAM,CAAC,EAAE,aAAa,EAAE,CAAC;IACvC,CAAC,YAAY,CAAC,KAAK,CAAC,EAAE,OAAO,EAAE,CAAC;CACjC,CAAC;AAEF,MAAM,MAAM,WAAW,CAAC,aAAa,SAAS,eAAe,IAAI;KAC9D,CAAC,IAAI,YAAY,GAAG,CAAC,IAAI,EAAE,eAAe,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,KAAK,IAAI;CACvE,CAAC;AAEF,MAAM,MAAM,OAAO,GAAG;IAAE,CAAC,GAAG,EAAE,MAAM,GAAG,MAAM,CAAA;CAAE,CAAC"}
1
+ {"version":3,"file":"interfaces.d.ts","sourceRoot":"","sources":["../../lib/ergo/interfaces.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,eAAe,EACf,SAAS,EACT,WAAW,EACZ,MAAM,kCAAkC,CAAC;AAE1C,MAAM,WAAW,mBAAoB,SAAQ,WAAW;IACtD,eAAe,EAAE,MAAM,CAAC;IACxB,OAAO,EAAE,MAAM,CAAC;IAChB,MAAM,EAAE,SAAS,EAAE,CAAC;CACrB;AAED,MAAM,WAAW,SAAS;IACxB,KAAK,EAAE,MAAM,CAAC;IACd,IAAI,EAAE,MAAM,CAAC;IACb,KAAK,EAAE,MAAM,CAAC;IACd,MAAM,CAAC,EAAE;QAAE,CAAC,GAAG,EAAE,MAAM,GAAG,MAAM,CAAA;KAAE,CAAC;CACpC;AAED,MAAM,WAAW,iBAAkB,SAAQ,SAAS;IAClD,MAAM,EAAE,MAAM,CAAC;IACf,KAAK,EAAE,MAAM,CAAC;CACf;AACD,MAAM,WAAW,kBAAkB;IACjC,UAAU,EAAE,MAAM,CAAC;IACnB,UAAU,EAAE,MAAM,CAAC;CACpB;AAED,oBAAY,YAAY;IACtB,MAAM,WAAW;IACjB,MAAM,WAAW;IACjB,KAAK,UAAU;IACf,MAAM,WAAW;CAClB;AAED,MAAM,WAAW,UAAU;IACzB,UAAU,EAAE,MAAM,CAAC;CACpB;AAED,MAAM,MAAM,eAAe,CAAC,aAAa,SAAS,kBAAkB,IAAI;IACtE,CAAC,YAAY,CAAC,MAAM,CAAC,EAAE,UAAU,EAAE,CAAC;IACpC,CAAC,YAAY,CAAC,MAAM,CAAC,EAAE,aAAa,EAAE,CAAC;IACvC,CAAC,YAAY,CAAC,MAAM,CAAC,EAAE,aAAa,EAAE,CAAC;IACvC,CAAC,YAAY,CAAC,KAAK,CAAC,EAAE,UAAU,EAAE,CAAC;CACpC,CAAC;AAEF,MAAM,MAAM,WAAW,CAAC,aAAa,SAAS,kBAAkB,IAAI;KACjE,CAAC,IAAI,YAAY,GAAG,CAAC,IAAI,EAAE,eAAe,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,KAAK,IAAI;CACvE,CAAC;AAEF,MAAM,MAAM,OAAO,GAAG;IAAE,CAAC,GAAG,EAAE,MAAM,GAAG,MAAM,CAAA;CAAE,CAAC;AAEhD,MAAM,WAAW,UAAU;IACzB,KAAK,EAAE,MAAM,CAAC;IACd,GAAG,EAAE,MAAM,CAAC;IACZ,KAAK,EAAE,MAAM,CAAC;CACf;AAED;;;KAGK;AACL,MAAM,MAAM,SAAS,GAAG,UAAU,EAAE,CAAC;AAErC,MAAM,WAAW,iBAAiB;IAChC,MAAM,EAAE,OAAO,CAAC;IAChB,IAAI,EAAE,eAAe,CAAC;IACtB,GAAG,EAAE,MAAM,CAAC;IACZ,OAAO,EAAE,MAAM,CAAC;IAChB,mBAAmB,CAAC,EAAE,MAAM,CAAC;CAC9B"}
@@ -5,4 +5,4 @@ export var CallbackType;
5
5
  CallbackType["Spend"] = "spend";
6
6
  CallbackType["Delete"] = "delete";
7
7
  })(CallbackType || (CallbackType = {}));
8
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW50ZXJmYWNlcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL2xpYi9lcmdvL2ludGVyZmFjZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBNEJBLE1BQU0sQ0FBTixJQUFZLFlBS1g7QUFMRCxXQUFZLFlBQVk7SUFDdEIsaUNBQWlCLENBQUE7SUFDakIsaUNBQWlCLENBQUE7SUFDakIsK0JBQWUsQ0FBQTtJQUNmLGlDQUFpQixDQUFBO0FBQ25CLENBQUMsRUFMVyxZQUFZLEtBQVosWUFBWSxRQUt2QiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IE91dHB1dEJveCwgVHJhbnNhY3Rpb24gfSBmcm9tICdAcm9zZW4tYnJpZGdlL3NjYW5uZXItaW50ZXJmYWNlcyc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgRXJnb0JveCBleHRlbmRzIE91dHB1dEJveCB7XG4gIGJsb2NrSWQ6IHN0cmluZztcbiAgaW5jbHVzaW9uSGVpZ2h0OiBudW1iZXI7XG4gIHNwZW50QmxvY2tJZD86IHN0cmluZztcbiAgc3BlbnRIZWlnaHQ/OiBudW1iZXI7XG4gIHNwZW50VHJhbnNhY3Rpb25JZD86IHN0cmluZztcbiAgc3BlbnRJbmRleD86IG51bWJlcjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBFeHRlbmRlZFRyYW5zYWN0aW9uIGV4dGVuZHMgVHJhbnNhY3Rpb24ge1xuICBpbmNsdXNpb25IZWlnaHQ6IG51bWJlcjtcbiAgYmxvY2tJZDogc3RyaW5nO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFNwZW5kSW5mbyB7XG4gIGJveElkOiBzdHJpbmc7XG4gIHR4SWQ6IHN0cmluZztcbiAgaW5kZXg6IG51bWJlcjtcbiAgZXh0cmFzPzogeyBba2V5OiBzdHJpbmddOiBzdHJpbmcgfTtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBBYnN0cmFjdEJveERhdGEge1xuICBib3hJZDogc3RyaW5nO1xuICBzZXJpYWxpemVkOiBzdHJpbmc7XG59XG5cbmV4cG9ydCBlbnVtIENhbGxiYWNrVHlwZSB7XG4gIEluc2VydCA9ICdpbnNlcnQnLFxuICBVcGRhdGUgPSAndXBkYXRlJyxcbiAgU3BlbmQgPSAnc3BlbmQnLFxuICBEZWxldGUgPSAnZGVsZXRlJyxcbn1cblxuZXhwb3J0IGludGVyZmFjZSBCb3hJbmZvIHtcbiAgYm94SWQ6IHN0cmluZztcbn1cblxuZXhwb3J0IHR5cGUgQ2FsbGJhY2tEYXRhTWFwPEV4dHJhY3RlZERhdGEgZXh0ZW5kcyBBYnN0cmFjdEJveERhdGE+ID0ge1xuICBbQ2FsbGJhY2tUeXBlLlVwZGF0ZV06IEJveEluZm9bXTtcbiAgW0NhbGxiYWNrVHlwZS5JbnNlcnRdOiBFeHRyYWN0ZWREYXRhW107XG4gIFtDYWxsYmFja1R5cGUuRGVsZXRlXTogRXh0cmFjdGVkRGF0YVtdO1xuICBbQ2FsbGJhY2tUeXBlLlNwZW5kXTogQm94SW5mb1tdO1xufTtcblxuZXhwb3J0IHR5cGUgQ2FsbGJhY2tNYXA8RXh0cmFjdGVkRGF0YSBleHRlbmRzIEFic3RyYWN0Qm94RGF0YT4gPSB7XG4gIFtLIGluIENhbGxiYWNrVHlwZV06IChkYXRhOiBDYWxsYmFja0RhdGFNYXA8RXh0cmFjdGVkRGF0YT5bS10pID0+IHZvaWQ7XG59O1xuXG5leHBvcnQgdHlwZSBUeEV4dHJhID0geyBba2V5OiBzdHJpbmddOiBzdHJpbmcgfTtcbiJdfQ==
8
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW50ZXJmYWNlcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL2xpYi9lcmdvL2ludGVyZmFjZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBNEJBLE1BQU0sQ0FBTixJQUFZLFlBS1g7QUFMRCxXQUFZLFlBQVk7SUFDdEIsaUNBQWlCLENBQUE7SUFDakIsaUNBQWlCLENBQUE7SUFDakIsK0JBQWUsQ0FBQTtJQUNmLGlDQUFpQixDQUFBO0FBQ25CLENBQUMsRUFMVyxZQUFZLEtBQVosWUFBWSxRQUt2QiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7XG4gIEVyZ29OZXR3b3JrVHlwZSxcbiAgT3V0cHV0Qm94LFxuICBUcmFuc2FjdGlvbixcbn0gZnJvbSAnQHJvc2VuLWJyaWRnZS9zY2FubmVyLWludGVyZmFjZXMnO1xuXG5leHBvcnQgaW50ZXJmYWNlIEV4dGVuZGVkVHJhbnNhY3Rpb24gZXh0ZW5kcyBUcmFuc2FjdGlvbiB7XG4gIGluY2x1c2lvbkhlaWdodDogbnVtYmVyO1xuICBibG9ja0lkOiBzdHJpbmc7XG4gIGlucHV0czogT3V0cHV0Qm94W107XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgU3BlbmRJbmZvIHtcbiAgYm94SWQ6IHN0cmluZztcbiAgdHhJZDogc3RyaW5nO1xuICBpbmRleDogbnVtYmVyO1xuICBleHRyYXM/OiB7IFtrZXk6IHN0cmluZ106IHN0cmluZyB9O1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIEV4dGVuZGVkU3BlbmRJbmZvIGV4dGVuZHMgU3BlbmRJbmZvIHtcbiAgaGVpZ2h0OiBudW1iZXI7XG4gIGJsb2NrOiBzdHJpbmc7XG59XG5leHBvcnQgaW50ZXJmYWNlIEFic3RyYWN0RW50aXR5RGF0YSB7XG4gIGlkZW50aWZpZXI6IHN0cmluZztcbiAgc2VyaWFsaXplZDogc3RyaW5nO1xufVxuXG5leHBvcnQgZW51bSBDYWxsYmFja1R5cGUge1xuICBJbnNlcnQgPSAnaW5zZXJ0JyxcbiAgVXBkYXRlID0gJ3VwZGF0ZScsXG4gIFNwZW5kID0gJ3NwZW5kJyxcbiAgRGVsZXRlID0gJ2RlbGV0ZScsXG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgRW50aXR5SW5mbyB7XG4gIGlkZW50aWZpZXI6IHN0cmluZztcbn1cblxuZXhwb3J0IHR5cGUgQ2FsbGJhY2tEYXRhTWFwPEV4dHJhY3RlZERhdGEgZXh0ZW5kcyBBYnN0cmFjdEVudGl0eURhdGE+ID0ge1xuICBbQ2FsbGJhY2tUeXBlLlVwZGF0ZV06IEVudGl0eUluZm9bXTtcbiAgW0NhbGxiYWNrVHlwZS5JbnNlcnRdOiBFeHRyYWN0ZWREYXRhW107XG4gIFtDYWxsYmFja1R5cGUuRGVsZXRlXTogRXh0cmFjdGVkRGF0YVtdO1xuICBbQ2FsbGJhY2tUeXBlLlNwZW5kXTogRW50aXR5SW5mb1tdO1xufTtcblxuZXhwb3J0IHR5cGUgQ2FsbGJhY2tNYXA8RXh0cmFjdGVkRGF0YSBleHRlbmRzIEFic3RyYWN0RW50aXR5RGF0YT4gPSB7XG4gIFtLIGluIENhbGxiYWNrVHlwZV06IChkYXRhOiBDYWxsYmFja0RhdGFNYXA8RXh0cmFjdGVkRGF0YT5bS10pID0+IHZvaWQ7XG59O1xuXG5leHBvcnQgdHlwZSBUeEV4dHJhID0geyBba2V5OiBzdHJpbmddOiBzdHJpbmcgfTtcblxuZXhwb3J0IGludGVyZmFjZSBSYW5nZVF1ZXJ5IHtcbiAgc3RhcnQ6IG51bWJlcjtcbiAgZW5kOiBudW1iZXI7XG4gIGNvdW50OiBudW1iZXI7XG59XG5cbi8qKlxuICogVGhlIHJhbmdlIGxpc3QgaXMgYW4gYXJyYXkgd2hlcmUgZWFjaCBuZXcgcmFuZ2UgaXMgYSBjaGlsZCBvZiB0aGUgcHJldmlvdXNcbiAqIG9uZS4gU28sIGFueSByYW5nZSBpbiB0aGUgbGlzdCBzdXBlcnNlZGVzIGFsbCB0aGUgcmFuZ2VzIHRoYXQgY29tZSBhZnRlciBpdC5cbiAqICovXG5leHBvcnQgdHlwZSBSYW5nZUxpc3QgPSBSYW5nZVF1ZXJ5W107XG5cbmV4cG9ydCBpbnRlcmZhY2UgSW5pdGlhbGl6ZU9wdGlvbnMge1xuICBhY3RpdmU6IGJvb2xlYW47XG4gIHR5cGU6IEVyZ29OZXR3b3JrVHlwZTtcbiAgdXJsOiBzdHJpbmc7XG4gIGFkZHJlc3M6IHN0cmluZztcbiAgbWF4UGFyYWxsZWxSZXF1ZXN0cz86IG51bWJlcjtcbn1cbiJdfQ==
@@ -0,0 +1,52 @@
1
+ import { Transaction } from '@rosen-bridge/scanner-interfaces';
2
+ import { ExtendedTransaction } from '../interfaces';
3
+ export declare class ExplorerNetwork {
4
+ private api;
5
+ constructor(url: string);
6
+ /**
7
+ * convert explorer api output boxes to OutputBox interface
8
+ * @param box
9
+ * @returns OutputBox
10
+ */
11
+ private convertOutputBox;
12
+ /**
13
+ * convert explorer api input boxes to OutputBox interface
14
+ * @param box
15
+ * @returns OutputBox
16
+ */
17
+ private convertInputBox;
18
+ /**
19
+ * convert explorer transaction to extractor transaction type
20
+ * @param tx
21
+ */
22
+ private convertTransaction;
23
+ /**
24
+ * convert explorer block transaction to transaction type
25
+ * @param tx
26
+ */
27
+ private convertBlockTransaction;
28
+ /**
29
+ * use explorer api to return related transactions of the specified address in the height range
30
+ * @param tokenId
31
+ * @param offset
32
+ * @param limit
33
+ * @returns related transactions
34
+ */
35
+ getAddressTransactionsWithHeight: (address: string, fromHeight: number, toHeight: number, limit?: number) => Promise<{
36
+ items: Array<ExtendedTransaction>;
37
+ total: number;
38
+ }>;
39
+ /**
40
+ * use explorer api to get the block id at the specified height
41
+ * @param height
42
+ * @returns block id
43
+ */
44
+ getBlockIdAtHeight: (height: number) => Promise<string>;
45
+ /**
46
+ * use explorer api to return all transactions in a block
47
+ * @param blockId
48
+ * @returns converted transactions
49
+ */
50
+ getBlockTxs: (blockId: string) => Promise<Array<Transaction>>;
51
+ }
52
+ //# sourceMappingURL=explorerNetwork.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"explorerNetwork.d.ts","sourceRoot":"","sources":["../../../lib/ergo/networks/explorerNetwork.ts"],"names":[],"mappings":"AAEA,OAAO,EAAa,WAAW,EAAE,MAAM,kCAAkC,CAAC;AAK1E,OAAO,EAAE,mBAAmB,EAAE,MAAM,eAAe,CAAC;AAEpD,qBAAa,eAAe;IAC1B,OAAO,CAAC,GAAG,CAAC;gBAEA,GAAG,EAAE,MAAM;IAIvB;;;;OAIG;IACH,OAAO,CAAC,gBAAgB,CAetB;IAEF;;;;OAIG;IACH,OAAO,CAAC,eAAe,CAerB;IAEF;;;OAGG;IACH,OAAO,CAAC,kBAAkB,CAexB;IAEF;;;OAGG;IACH,OAAO,CAAC,uBAAuB,CAuB7B;IAEF;;;;;;OAMG;IACH,gCAAgC,GAC9B,SAAS,MAAM,EACf,YAAY,MAAM,EAClB,UAAU,MAAM,EAChB,cAAiB,KAChB,OAAO,CAAC;QAAE,KAAK,EAAE,KAAK,CAAC,mBAAmB,CAAC,CAAC;QAAC,KAAK,EAAE,MAAM,CAAA;KAAE,CAAC,CAc9D;IAEF;;;;OAIG;IACH,kBAAkB,GAAU,QAAQ,MAAM,KAAG,OAAO,CAAC,MAAM,CAAC,CAG1D;IAEF;;;;OAIG;IACH,WAAW,GAAU,SAAS,MAAM,KAAG,OAAO,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC,CAUhE;CACH"}