@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
@@ -1,10 +1,8 @@
1
- export * from './abstractErgoExtractor';
2
- export * from './abstractErgoExtractorAction';
1
+ export * from './extractors';
2
+ export * from './initializers';
3
+ export * from './database';
3
4
  export * from './interfaces';
4
- export * from './network/explorerNetwork';
5
- export * from './network/nodeNetwork';
6
- export * from './network/abstractNetwork';
7
- export * from './initializable';
5
+ export * from './networks';
6
+ export * from './initializers';
8
7
  export * from './utils';
9
- export * from './abstractErgoExtractorEntity';
10
8
  //# sourceMappingURL=index.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../lib/ergo/index.ts"],"names":[],"mappings":"AAAA,cAAc,yBAAyB,CAAC;AACxC,cAAc,+BAA+B,CAAC;AAC9C,cAAc,cAAc,CAAC;AAC7B,cAAc,2BAA2B,CAAC;AAC1C,cAAc,uBAAuB,CAAC;AACtC,cAAc,2BAA2B,CAAC;AAC1C,cAAc,iBAAiB,CAAC;AAChC,cAAc,SAAS,CAAC;AACxB,cAAc,+BAA+B,CAAC"}
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../lib/ergo/index.ts"],"names":[],"mappings":"AAAA,cAAc,cAAc,CAAC;AAC7B,cAAc,gBAAgB,CAAC;AAC/B,cAAc,YAAY,CAAC;AAC3B,cAAc,cAAc,CAAC;AAC7B,cAAc,YAAY,CAAC;AAC3B,cAAc,gBAAgB,CAAC;AAC/B,cAAc,SAAS,CAAC"}
@@ -1,10 +1,8 @@
1
- export * from './abstractErgoExtractor';
2
- export * from './abstractErgoExtractorAction';
1
+ export * from './extractors';
2
+ export * from './initializers';
3
+ export * from './database';
3
4
  export * from './interfaces';
4
- export * from './network/explorerNetwork';
5
- export * from './network/nodeNetwork';
6
- export * from './network/abstractNetwork';
7
- export * from './initializable';
5
+ export * from './networks';
6
+ export * from './initializers';
8
7
  export * from './utils';
9
- export * from './abstractErgoExtractorEntity';
10
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9saWIvZXJnby9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLHlCQUF5QixDQUFDO0FBQ3hDLGNBQWMsK0JBQStCLENBQUM7QUFDOUMsY0FBYyxjQUFjLENBQUM7QUFDN0IsY0FBYywyQkFBMkIsQ0FBQztBQUMxQyxjQUFjLHVCQUF1QixDQUFDO0FBQ3RDLGNBQWMsMkJBQTJCLENBQUM7QUFDMUMsY0FBYyxpQkFBaUIsQ0FBQztBQUNoQyxjQUFjLFNBQVMsQ0FBQztBQUN4QixjQUFjLCtCQUErQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9hYnN0cmFjdEVyZ29FeHRyYWN0b3InO1xuZXhwb3J0ICogZnJvbSAnLi9hYnN0cmFjdEVyZ29FeHRyYWN0b3JBY3Rpb24nO1xuZXhwb3J0ICogZnJvbSAnLi9pbnRlcmZhY2VzJztcbmV4cG9ydCAqIGZyb20gJy4vbmV0d29yay9leHBsb3Jlck5ldHdvcmsnO1xuZXhwb3J0ICogZnJvbSAnLi9uZXR3b3JrL25vZGVOZXR3b3JrJztcbmV4cG9ydCAqIGZyb20gJy4vbmV0d29yay9hYnN0cmFjdE5ldHdvcmsnO1xuZXhwb3J0ICogZnJvbSAnLi9pbml0aWFsaXphYmxlJztcbmV4cG9ydCAqIGZyb20gJy4vdXRpbHMnO1xuZXhwb3J0ICogZnJvbSAnLi9hYnN0cmFjdEVyZ29FeHRyYWN0b3JFbnRpdHknO1xuIl19
8
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9saWIvZXJnby9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLGNBQWMsQ0FBQztBQUM3QixjQUFjLGdCQUFnQixDQUFDO0FBQy9CLGNBQWMsWUFBWSxDQUFDO0FBQzNCLGNBQWMsY0FBYyxDQUFDO0FBQzdCLGNBQWMsWUFBWSxDQUFDO0FBQzNCLGNBQWMsZ0JBQWdCLENBQUM7QUFDL0IsY0FBYyxTQUFTLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL2V4dHJhY3RvcnMnO1xuZXhwb3J0ICogZnJvbSAnLi9pbml0aWFsaXplcnMnO1xuZXhwb3J0ICogZnJvbSAnLi9kYXRhYmFzZSc7XG5leHBvcnQgKiBmcm9tICcuL2ludGVyZmFjZXMnO1xuZXhwb3J0ICogZnJvbSAnLi9uZXR3b3Jrcyc7XG5leHBvcnQgKiBmcm9tICcuL2luaXRpYWxpemVycyc7XG5leHBvcnQgKiBmcm9tICcuL3V0aWxzJztcbiJdfQ==
@@ -0,0 +1,36 @@
1
+ import { DummyLogger } from '@rosen-bridge/abstract-logger';
2
+ import { BlockInfo, ErgoNetworkType, OutputBox, Transaction } from '@rosen-bridge/scanner-interfaces';
3
+ import { AbstractErgoBoxAction, AbstractErgoEntity } from '../database';
4
+ import { AbstractEntityData, ExtendedSpendInfo, ExtendedTransaction } from '../interfaces';
5
+ import { ErgoInitializer } from './ergoInitializer';
6
+ export declare class ErgoBoxInitializer<ExtractedData extends AbstractEntityData, ExtractorEntity extends AbstractErgoEntity> extends ErgoInitializer<ExtractedData, ExtractorEntity> {
7
+ protected extractorId: string;
8
+ protected hasBoxData: (box: OutputBox) => boolean;
9
+ protected processTransactions: (txs: Transaction[], block: BlockInfo) => Promise<boolean>;
10
+ protected actions: AbstractErgoBoxAction<ExtractedData, ExtractorEntity>;
11
+ protected logger: DummyLogger;
12
+ private spendRecordsMutex;
13
+ private spendRecords;
14
+ constructor(networkType: ErgoNetworkType, url: string, address: string, extractorId: string, hasBoxData: (box: OutputBox) => boolean, processTransactions: (txs: Transaction[], block: BlockInfo) => Promise<boolean>, actions: AbstractErgoBoxAction<ExtractedData, ExtractorEntity>, maxParallelRequests?: number, logger?: DummyLogger);
15
+ /**
16
+ * Extracts spending information of all related boxes in the transaction
17
+ * Note: override this function if the extractor needs extra spending info
18
+ * @param tx
19
+ * @returns transaction spend info
20
+ */
21
+ protected extractTxSpendInfo: (tx: ExtendedTransaction) => ExtendedSpendInfo[];
22
+ /**
23
+ * Extract and store all spending information of a transaction batch
24
+ * @param txs
25
+ */
26
+ protected storeExtraInfo: (txs: ExtendedTransaction[]) => Promise<void>;
27
+ /**
28
+ * Apply stored spend records into extractor database
29
+ * Note: As transactions are processed out of order (due to parallel
30
+ * processing), some box spend information may be invalid after the first pass
31
+ * To avoid processing everything twice, we keep all spend records in the
32
+ * first pass and reapply them at the end
33
+ */
34
+ protected applyExtraInfo: () => Promise<void>;
35
+ }
36
+ //# sourceMappingURL=ergoBoxInitializer.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ergoBoxInitializer.d.ts","sourceRoot":"","sources":["../../../lib/ergo/initializers/ergoBoxInitializer.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,WAAW,EAAE,MAAM,+BAA+B,CAAC;AAC5D,OAAO,EACL,SAAS,EACT,eAAe,EACf,SAAS,EACT,WAAW,EACZ,MAAM,kCAAkC,CAAC;AAG1C,OAAO,EAAE,qBAAqB,EAAE,kBAAkB,EAAE,MAAM,aAAa,CAAC;AACxE,OAAO,EACL,kBAAkB,EAClB,iBAAiB,EACjB,mBAAmB,EACpB,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,eAAe,EAAE,MAAM,mBAAmB,CAAC;AAEpD,qBAAa,kBAAkB,CAC7B,aAAa,SAAS,kBAAkB,EACxC,eAAe,SAAS,kBAAkB,CAC1C,SAAQ,eAAe,CAAC,aAAa,EAAE,eAAe,CAAC;IAQrD,SAAS,CAAC,WAAW,EAAE,MAAM;IAC7B,SAAS,CAAC,UAAU,EAAE,CAAC,GAAG,EAAE,SAAS,KAAK,OAAO;IACjD,SAAS,CAAC,mBAAmB,EAAE,CAC7B,GAAG,EAAE,WAAW,EAAE,EAClB,KAAK,EAAE,SAAS,KACb,OAAO,CAAC,OAAO,CAAC;IACrB,SAAS,CAAC,OAAO,EAAE,qBAAqB,CAAC,aAAa,EAAE,eAAe,CAAC;IAExE,SAAS,CAAC,MAAM;IAflB,OAAO,CAAC,iBAAiB,CAAe;IACxC,OAAO,CAAC,YAAY,CAAsB;gBAGxC,WAAW,EAAE,eAAe,EAC5B,GAAG,EAAE,MAAM,EACX,OAAO,EAAE,MAAM,EACL,WAAW,EAAE,MAAM,EACnB,UAAU,EAAE,CAAC,GAAG,EAAE,SAAS,KAAK,OAAO,EACvC,mBAAmB,EAAE,CAC7B,GAAG,EAAE,WAAW,EAAE,EAClB,KAAK,EAAE,SAAS,KACb,OAAO,CAAC,OAAO,CAAC,EACX,OAAO,EAAE,qBAAqB,CAAC,aAAa,EAAE,eAAe,CAAC,EACxE,mBAAmB,SAAwB,EACjC,MAAM,cAAoB;IAetC;;;;;OAKG;IACH,SAAS,CAAC,kBAAkB,GAC1B,IAAI,mBAAmB,KACtB,iBAAiB,EAAE,CAepB;IAEF;;;OAGG;IACH,SAAS,CAAC,cAAc,GACtB,KAAK,mBAAmB,EAAE,KACzB,OAAO,CAAC,IAAI,CAAC,CASd;IAEF;;;;;;OAMG;IACH,SAAS,CAAC,cAAc,sBAoBtB;CACH"}
@@ -0,0 +1,80 @@
1
+ import { Mutex } from 'await-semaphore';
2
+ import { groupBy, sortBy } from 'lodash-es';
3
+ import { DummyLogger } from '@rosen-bridge/abstract-logger';
4
+ import { MAX_PARALLEL_REQUESTS } from '../../constants';
5
+ import { ErgoInitializer } from './ergoInitializer';
6
+ export class ErgoBoxInitializer extends ErgoInitializer {
7
+ extractorId;
8
+ hasBoxData;
9
+ processTransactions;
10
+ actions;
11
+ logger;
12
+ spendRecordsMutex = new Mutex();
13
+ spendRecords;
14
+ constructor(networkType, url, address, extractorId, hasBoxData, processTransactions, actions, maxParallelRequests = MAX_PARALLEL_REQUESTS, logger = new DummyLogger()) {
15
+ super(networkType, url, address, extractorId, processTransactions, actions, maxParallelRequests, logger);
16
+ this.extractorId = extractorId;
17
+ this.hasBoxData = hasBoxData;
18
+ this.processTransactions = processTransactions;
19
+ this.actions = actions;
20
+ this.logger = logger;
21
+ this.spendRecords = [];
22
+ }
23
+ /**
24
+ * Extracts spending information of all related boxes in the transaction
25
+ * Note: override this function if the extractor needs extra spending info
26
+ * @param tx
27
+ * @returns transaction spend info
28
+ */
29
+ extractTxSpendInfo = (tx) => {
30
+ const txSpendInfo = [];
31
+ for (let i = 0; i < tx.inputs.length; i++) {
32
+ const box = tx.inputs[i];
33
+ if (this.hasBoxData(box)) {
34
+ txSpendInfo.push({
35
+ boxId: box.boxId,
36
+ txId: box.transactionId,
37
+ index: i,
38
+ height: tx.inclusionHeight,
39
+ block: tx.blockId,
40
+ });
41
+ }
42
+ }
43
+ return txSpendInfo;
44
+ };
45
+ /**
46
+ * Extract and store all spending information of a transaction batch
47
+ * @param txs
48
+ */
49
+ storeExtraInfo = async (txs) => {
50
+ const spendRecordsBatch = [];
51
+ for (const tx of txs) {
52
+ spendRecordsBatch.push(...this.extractTxSpendInfo(tx));
53
+ }
54
+ const release = await this.spendRecordsMutex.acquire();
55
+ this.spendRecords.push(...spendRecordsBatch);
56
+ release();
57
+ this.logger.debug(`Stored ${spendRecordsBatch.length} new spend records`);
58
+ };
59
+ /**
60
+ * Apply stored spend records into extractor database
61
+ * Note: As transactions are processed out of order (due to parallel
62
+ * processing), some box spend information may be invalid after the first pass
63
+ * To avoid processing everything twice, we keep all spend records in the
64
+ * first pass and reapply them at the end
65
+ */
66
+ applyExtraInfo = async () => {
67
+ const sortedRecords = sortBy(this.spendRecords, (record) => record.height);
68
+ const groupedRecords = groupBy(sortedRecords, (tx) => tx.block);
69
+ this.logger.debug(`Spend records grouped to ${Object.keys(groupedRecords).length} blocks`);
70
+ const release = await this.dbMutex.acquire();
71
+ for (const blockId in groupedRecords) {
72
+ const blockRecords = groupedRecords[blockId];
73
+ const block = { hash: blockId, height: blockRecords[0].height };
74
+ this.logger.debug(`Processing spend records at height ${blockRecords[0].height}`);
75
+ await this.actions.updateSpendingInfo(blockRecords, block, this.extractorId);
76
+ }
77
+ release();
78
+ };
79
+ }
80
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ergoBoxInitializer.js","sourceRoot":"","sources":["../../../lib/ergo/initializers/ergoBoxInitializer.ts"],"names":[],"mappings":"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","sourcesContent":["import { Mutex } from 'await-semaphore';\nimport { groupBy, sortBy } from 'lodash-es';\n\nimport { DummyLogger } from '@rosen-bridge/abstract-logger';\nimport {\n  BlockInfo,\n  ErgoNetworkType,\n  OutputBox,\n  Transaction,\n} from '@rosen-bridge/scanner-interfaces';\n\nimport { MAX_PARALLEL_REQUESTS } from '../../constants';\nimport { AbstractErgoBoxAction, AbstractErgoEntity } from '../database';\nimport {\n  AbstractEntityData,\n  ExtendedSpendInfo,\n  ExtendedTransaction,\n} from '../interfaces';\nimport { ErgoInitializer } from './ergoInitializer';\n\nexport class ErgoBoxInitializer<\n  ExtractedData extends AbstractEntityData,\n  ExtractorEntity extends AbstractErgoEntity,\n> extends ErgoInitializer<ExtractedData, ExtractorEntity> {\n  private spendRecordsMutex = new Mutex();\n  private spendRecords: ExtendedSpendInfo[];\n\n  constructor(\n    networkType: ErgoNetworkType,\n    url: string,\n    address: string,\n    protected extractorId: string,\n    protected hasBoxData: (box: OutputBox) => boolean,\n    protected processTransactions: (\n      txs: Transaction[],\n      block: BlockInfo,\n    ) => Promise<boolean>,\n    protected actions: AbstractErgoBoxAction<ExtractedData, ExtractorEntity>,\n    maxParallelRequests = MAX_PARALLEL_REQUESTS,\n    protected logger = new DummyLogger(),\n  ) {\n    super(\n      networkType,\n      url,\n      address,\n      extractorId,\n      processTransactions,\n      actions,\n      maxParallelRequests,\n      logger,\n    );\n    this.spendRecords = [];\n  }\n\n  /**\n   * Extracts spending information of all related boxes in the transaction\n   * Note: override this function if the extractor needs extra spending info\n   * @param tx\n   * @returns transaction spend info\n   */\n  protected extractTxSpendInfo = (\n    tx: ExtendedTransaction,\n  ): ExtendedSpendInfo[] => {\n    const txSpendInfo = [];\n    for (let i = 0; i < tx.inputs.length; i++) {\n      const box = tx.inputs[i];\n      if (this.hasBoxData(box)) {\n        txSpendInfo.push({\n          boxId: box.boxId,\n          txId: box.transactionId,\n          index: i,\n          height: tx.inclusionHeight,\n          block: tx.blockId,\n        });\n      }\n    }\n    return txSpendInfo;\n  };\n\n  /**\n   * Extract and store all spending information of a transaction batch\n   * @param txs\n   */\n  protected storeExtraInfo = async (\n    txs: ExtendedTransaction[],\n  ): Promise<void> => {\n    const spendRecordsBatch: ExtendedSpendInfo[] = [];\n    for (const tx of txs) {\n      spendRecordsBatch.push(...this.extractTxSpendInfo(tx));\n    }\n    const release = await this.spendRecordsMutex.acquire();\n    this.spendRecords.push(...spendRecordsBatch);\n    release();\n    this.logger.debug(`Stored ${spendRecordsBatch.length} new spend records`);\n  };\n\n  /**\n   * Apply stored spend records into extractor database\n   * Note: As transactions are processed out of order (due to parallel\n   * processing), some box spend information may be invalid after the first pass\n   * To avoid processing everything twice, we keep all spend records in the\n   * first pass and reapply them at the end\n   */\n  protected applyExtraInfo = async () => {\n    const sortedRecords = sortBy(this.spendRecords, (record) => record.height);\n    const groupedRecords = groupBy(sortedRecords, (tx) => tx.block);\n    this.logger.debug(\n      `Spend records grouped to ${Object.keys(groupedRecords).length} blocks`,\n    );\n    const release = await this.dbMutex.acquire();\n    for (const blockId in groupedRecords) {\n      const blockRecords = groupedRecords[blockId];\n      const block = { hash: blockId, height: blockRecords[0].height };\n      this.logger.debug(\n        `Processing spend records at height ${blockRecords[0].height}`,\n      );\n      await this.actions.updateSpendingInfo(\n        blockRecords,\n        block,\n        this.extractorId,\n      );\n    }\n    release();\n  };\n}\n"]}
@@ -0,0 +1,39 @@
1
+ import { Mutex } from 'await-semaphore';
2
+ import { DummyLogger } from '@rosen-bridge/abstract-logger';
3
+ import { BlockInfo, ErgoNetworkType, Transaction } from '@rosen-bridge/scanner-interfaces';
4
+ import { AbstractErgoAction, AbstractErgoEntity } from '../database';
5
+ import { AbstractEntityData, ExtendedTransaction } from '../interfaces';
6
+ import { ExplorerInitializationStrategy, NodeInitializationStrategy } from './strategies';
7
+ export declare class ErgoInitializer<ExtractedData extends AbstractEntityData, ExtractorEntity extends AbstractErgoEntity> {
8
+ protected extractorId: string;
9
+ protected processTransactions: (txs: Transaction[], block: BlockInfo) => Promise<boolean>;
10
+ protected actions: AbstractErgoAction<ExtractedData, ExtractorEntity>;
11
+ protected logger: DummyLogger;
12
+ protected dbMutex: Mutex;
13
+ protected initializationStrategy: ExplorerInitializationStrategy | NodeInitializationStrategy;
14
+ constructor(networkType: ErgoNetworkType, url: string, address: string, extractorId: string, processTransactions: (txs: Transaction[], block: BlockInfo) => Promise<boolean>, actions: AbstractErgoAction<ExtractedData, ExtractorEntity>, maxParallelRequests?: number, logger?: DummyLogger);
15
+ /**
16
+ * override this function to store extra information of a transaction batch
17
+ * @param txs list of transactions
18
+ */
19
+ protected storeExtraInfo(txs: ExtendedTransaction[]): Promise<void>;
20
+ /**
21
+ * override this function to apply extra information to the extractor database
22
+ */
23
+ protected applyExtraInfo: () => Promise<void>;
24
+ /**
25
+ * Process a batch of transactions
26
+ * group txs into blocks and process them using `processTransactions`
27
+ * @param txs
28
+ */
29
+ private processTransactionBatch;
30
+ /**
31
+ * remove all old data and initialize extractor database with data created
32
+ * below the initial height and finally apply the stored spend records to make
33
+ * sure all stored data is valid
34
+ * ignore initialization if this feature is off
35
+ * @param initialBlock
36
+ */
37
+ initializeData: (initialBlock: BlockInfo) => Promise<void>;
38
+ }
39
+ //# sourceMappingURL=ergoInitializer.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ergoInitializer.d.ts","sourceRoot":"","sources":["../../../lib/ergo/initializers/ergoInitializer.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,EAAE,MAAM,iBAAiB,CAAC;AAGxC,OAAO,EAAE,WAAW,EAAE,MAAM,+BAA+B,CAAC;AAC5D,OAAO,EACL,SAAS,EACT,eAAe,EACf,WAAW,EACZ,MAAM,kCAAkC,CAAC;AAG1C,OAAO,EAAE,kBAAkB,EAAE,kBAAkB,EAAE,MAAM,aAAa,CAAC;AACrE,OAAO,EAAE,kBAAkB,EAAE,mBAAmB,EAAE,MAAM,eAAe,CAAC;AACxE,OAAO,EACL,8BAA8B,EAC9B,0BAA0B,EAC3B,MAAM,cAAc,CAAC;AAEtB,qBAAa,eAAe,CAC1B,aAAa,SAAS,kBAAkB,EACxC,eAAe,SAAS,kBAAkB;IAWxC,SAAS,CAAC,WAAW,EAAE,MAAM;IAC7B,SAAS,CAAC,mBAAmB,EAAE,CAC7B,GAAG,EAAE,WAAW,EAAE,EAClB,KAAK,EAAE,SAAS,KACb,OAAO,CAAC,OAAO,CAAC;IACrB,SAAS,CAAC,OAAO,EAAE,kBAAkB,CAAC,aAAa,EAAE,eAAe,CAAC;IAErE,SAAS,CAAC,MAAM;IAhBlB,SAAS,CAAC,OAAO,QAAe;IAChC,SAAS,CAAC,sBAAsB,EAC5B,8BAA8B,GAC9B,0BAA0B,CAAC;gBAG7B,WAAW,EAAE,eAAe,EAC5B,GAAG,EAAE,MAAM,EACX,OAAO,EAAE,MAAM,EACL,WAAW,EAAE,MAAM,EACnB,mBAAmB,EAAE,CAC7B,GAAG,EAAE,WAAW,EAAE,EAClB,KAAK,EAAE,SAAS,KACb,OAAO,CAAC,OAAO,CAAC,EACX,OAAO,EAAE,kBAAkB,CAAC,aAAa,EAAE,eAAe,CAAC,EACrE,mBAAmB,SAAwB,EACjC,MAAM,cAAoB;IAsBtC;;;OAGG;IACH,SAAS,CAAC,cAAc,CACtB,GAAG,EAAE,mBAAmB,EAAE,GACzB,OAAO,CAAC,IAAI,CAAC;IAIhB;;OAEG;IACH,SAAS,CAAC,cAAc,QAAO,OAAO,CAAC,IAAI,CAAC,CAE1C;IAEF;;;;OAIG;IACH,OAAO,CAAC,uBAAuB,CA0B7B;IAEF;;;;;;OAMG;IACH,cAAc,GAAU,cAAc,SAAS,mBAQ7C;CACH"}
@@ -0,0 +1,80 @@
1
+ import { Mutex } from 'await-semaphore';
2
+ import { groupBy, sortBy } from 'lodash-es';
3
+ import { DummyLogger } from '@rosen-bridge/abstract-logger';
4
+ import { ErgoNetworkType, } from '@rosen-bridge/scanner-interfaces';
5
+ import { MAX_PARALLEL_REQUESTS } from '../../constants';
6
+ import { ExplorerInitializationStrategy, NodeInitializationStrategy, } from './strategies';
7
+ export class ErgoInitializer {
8
+ extractorId;
9
+ processTransactions;
10
+ actions;
11
+ logger;
12
+ dbMutex = new Mutex();
13
+ initializationStrategy;
14
+ constructor(networkType, url, address, extractorId, processTransactions, actions, maxParallelRequests = MAX_PARALLEL_REQUESTS, logger = new DummyLogger()) {
15
+ this.extractorId = extractorId;
16
+ this.processTransactions = processTransactions;
17
+ this.actions = actions;
18
+ this.logger = logger;
19
+ if (networkType == ErgoNetworkType.Explorer) {
20
+ this.initializationStrategy = new ExplorerInitializationStrategy(url, address, maxParallelRequests, this.processTransactions, this.processTransactionBatch, logger);
21
+ }
22
+ else if (networkType == ErgoNetworkType.Node) {
23
+ this.initializationStrategy = new NodeInitializationStrategy(url, address, maxParallelRequests, this.processTransactionBatch, logger);
24
+ }
25
+ else
26
+ throw new Error(`Network type ${networkType} is not supported`);
27
+ }
28
+ /**
29
+ * override this function to store extra information of a transaction batch
30
+ * @param txs list of transactions
31
+ */
32
+ storeExtraInfo(txs) {
33
+ return Promise.resolve();
34
+ }
35
+ /**
36
+ * override this function to apply extra information to the extractor database
37
+ */
38
+ applyExtraInfo = () => {
39
+ return Promise.resolve();
40
+ };
41
+ /**
42
+ * Process a batch of transactions
43
+ * group txs into blocks and process them using `processTransactions`
44
+ * @param txs
45
+ */
46
+ processTransactionBatch = async (txs) => {
47
+ txs = sortBy(txs, (tx) => tx.inclusionHeight);
48
+ const groupedTxs = groupBy(txs, (tx) => tx.blockId);
49
+ this.logger.debug(`The transaction batch grouped to ${Object.keys(groupedTxs).length} blocks`);
50
+ const release = await this.dbMutex.acquire();
51
+ for (const blockId in groupedTxs) {
52
+ const blockTxs = groupedTxs[blockId];
53
+ const block = { hash: blockId, height: blockTxs[0].inclusionHeight };
54
+ this.logger.debug(`Processing transactions at height ${blockTxs[0].inclusionHeight}`);
55
+ const success = await this.processTransactions(blockTxs, block);
56
+ if (!success) {
57
+ release();
58
+ throw Error(`Processing transactions failed at height ${blockTxs[0].inclusionHeight}`);
59
+ }
60
+ }
61
+ release();
62
+ this.logger.debug(`storing spend info of transaction batch`);
63
+ await this.storeExtraInfo(txs);
64
+ };
65
+ /**
66
+ * remove all old data and initialize extractor database with data created
67
+ * below the initial height and finally apply the stored spend records to make
68
+ * sure all stored data is valid
69
+ * ignore initialization if this feature is off
70
+ * @param initialBlock
71
+ */
72
+ initializeData = async (initialBlock) => {
73
+ this.logger.info(`Initialization process for ${this.extractorId} started`);
74
+ await this.actions.removeAllData(this.extractorId);
75
+ await this.initializationStrategy.initialize(initialBlock);
76
+ await this.applyExtraInfo();
77
+ this.logger.info(`Initialization completed successfully for ${this.extractorId}`);
78
+ };
79
+ }
80
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ergoInitializer.js","sourceRoot":"","sources":["../../../lib/ergo/initializers/ergoInitializer.ts"],"names":[],"mappings":"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","sourcesContent":["import { Mutex } from 'await-semaphore';\nimport { groupBy, sortBy } from 'lodash-es';\n\nimport { DummyLogger } from '@rosen-bridge/abstract-logger';\nimport {\n  BlockInfo,\n  ErgoNetworkType,\n  Transaction,\n} from '@rosen-bridge/scanner-interfaces';\n\nimport { MAX_PARALLEL_REQUESTS } from '../../constants';\nimport { AbstractErgoAction, AbstractErgoEntity } from '../database';\nimport { AbstractEntityData, ExtendedTransaction } from '../interfaces';\nimport {\n  ExplorerInitializationStrategy,\n  NodeInitializationStrategy,\n} from './strategies';\n\nexport class ErgoInitializer<\n  ExtractedData extends AbstractEntityData,\n  ExtractorEntity extends AbstractErgoEntity,\n> {\n  protected dbMutex = new Mutex();\n  protected initializationStrategy:\n    | ExplorerInitializationStrategy\n    | NodeInitializationStrategy;\n\n  constructor(\n    networkType: ErgoNetworkType,\n    url: string,\n    address: string,\n    protected extractorId: string,\n    protected processTransactions: (\n      txs: Transaction[],\n      block: BlockInfo,\n    ) => Promise<boolean>,\n    protected actions: AbstractErgoAction<ExtractedData, ExtractorEntity>,\n    maxParallelRequests = MAX_PARALLEL_REQUESTS,\n    protected logger = new DummyLogger(),\n  ) {\n    if (networkType == ErgoNetworkType.Explorer) {\n      this.initializationStrategy = new ExplorerInitializationStrategy(\n        url,\n        address,\n        maxParallelRequests,\n        this.processTransactions,\n        this.processTransactionBatch,\n        logger,\n      );\n    } else if (networkType == ErgoNetworkType.Node) {\n      this.initializationStrategy = new NodeInitializationStrategy(\n        url,\n        address,\n        maxParallelRequests,\n        this.processTransactionBatch,\n        logger,\n      );\n    } else throw new Error(`Network type ${networkType} is not supported`);\n  }\n\n  /**\n   * override this function to store extra information of a transaction batch\n   * @param txs list of transactions\n   */\n  protected storeExtraInfo(\n    txs: ExtendedTransaction[], // eslint-disable-line @typescript-eslint/no-unused-vars\n  ): Promise<void> {\n    return Promise.resolve();\n  }\n\n  /**\n   * override this function to apply extra information to the extractor database\n   */\n  protected applyExtraInfo = (): Promise<void> => {\n    return Promise.resolve();\n  };\n\n  /**\n   * Process a batch of transactions\n   * group txs into blocks and process them using `processTransactions`\n   * @param txs\n   */\n  private processTransactionBatch = async (txs: Array<ExtendedTransaction>) => {\n    txs = sortBy(txs, (tx) => tx.inclusionHeight);\n    const groupedTxs = groupBy(txs, (tx) => tx.blockId);\n    this.logger.debug(\n      `The transaction batch grouped to ${\n        Object.keys(groupedTxs).length\n      } blocks`,\n    );\n    const release = await this.dbMutex.acquire();\n    for (const blockId in groupedTxs) {\n      const blockTxs = groupedTxs[blockId];\n      const block = { hash: blockId, height: blockTxs[0].inclusionHeight };\n      this.logger.debug(\n        `Processing transactions at height ${blockTxs[0].inclusionHeight}`,\n      );\n      const success = await this.processTransactions(blockTxs, block);\n      if (!success) {\n        release();\n        throw Error(\n          `Processing transactions failed at height ${blockTxs[0].inclusionHeight}`,\n        );\n      }\n    }\n    release();\n    this.logger.debug(`storing spend info of transaction batch`);\n    await this.storeExtraInfo(txs);\n  };\n\n  /**\n   * remove all old data and initialize extractor database with data created\n   * below the initial height and finally apply the stored spend records to make\n   * sure all stored data is valid\n   * ignore initialization if this feature is off\n   * @param initialBlock\n   */\n  initializeData = async (initialBlock: BlockInfo) => {\n    this.logger.info(`Initialization process for ${this.extractorId} started`);\n    await this.actions.removeAllData(this.extractorId);\n    await this.initializationStrategy.initialize(initialBlock);\n    await this.applyExtraInfo();\n    this.logger.info(\n      `Initialization completed successfully for ${this.extractorId}`,\n    );\n  };\n}\n"]}
@@ -0,0 +1,4 @@
1
+ export * from './ergoInitializer';
2
+ export * from './ergoBoxInitializer';
3
+ export * from './strategies';
4
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../lib/ergo/initializers/index.ts"],"names":[],"mappings":"AAAA,cAAc,mBAAmB,CAAC;AAClC,cAAc,sBAAsB,CAAC;AACrC,cAAc,cAAc,CAAC"}
@@ -0,0 +1,4 @@
1
+ export * from './ergoInitializer';
2
+ export * from './ergoBoxInitializer';
3
+ export * from './strategies';
4
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9saWIvZXJnby9pbml0aWFsaXplcnMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsY0FBYyxtQkFBbUIsQ0FBQztBQUNsQyxjQUFjLHNCQUFzQixDQUFDO0FBQ3JDLGNBQWMsY0FBYyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9lcmdvSW5pdGlhbGl6ZXInO1xuZXhwb3J0ICogZnJvbSAnLi9lcmdvQm94SW5pdGlhbGl6ZXInO1xuZXhwb3J0ICogZnJvbSAnLi9zdHJhdGVnaWVzJztcbiJdfQ==
@@ -0,0 +1,3 @@
1
+ export declare const DELAY_BETWEEN_INIT_REQUESTS = 100;
2
+ export declare const INIT_WORKERS_REASSIGN_INTERVAL = 5000;
3
+ //# sourceMappingURL=constants.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"constants.d.ts","sourceRoot":"","sources":["../../../../lib/ergo/initializers/strategies/constants.ts"],"names":[],"mappings":"AAAA,eAAO,MAAM,2BAA2B,MAAM,CAAC;AAC/C,eAAO,MAAM,8BAA8B,OAAO,CAAC"}
@@ -0,0 +1,3 @@
1
+ export const DELAY_BETWEEN_INIT_REQUESTS = 100; // In milliseconds (only applies to node initialization)
2
+ export const INIT_WORKERS_REASSIGN_INTERVAL = 5000; //milliseconds (only applies to explorer initialization)
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uc3RhbnRzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vbGliL2VyZ28vaW5pdGlhbGl6ZXJzL3N0cmF0ZWdpZXMvY29uc3RhbnRzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE1BQU0sQ0FBQyxNQUFNLDJCQUEyQixHQUFHLEdBQUcsQ0FBQyxDQUFDLHdEQUF3RDtBQUN4RyxNQUFNLENBQUMsTUFBTSw4QkFBOEIsR0FBRyxJQUFJLENBQUMsQ0FBQyx3REFBd0QiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgY29uc3QgREVMQVlfQkVUV0VFTl9JTklUX1JFUVVFU1RTID0gMTAwOyAvLyBJbiBtaWxsaXNlY29uZHMgKG9ubHkgYXBwbGllcyB0byBub2RlIGluaXRpYWxpemF0aW9uKVxuZXhwb3J0IGNvbnN0IElOSVRfV09SS0VSU19SRUFTU0lHTl9JTlRFUlZBTCA9IDUwMDA7IC8vbWlsbGlzZWNvbmRzIChvbmx5IGFwcGxpZXMgdG8gZXhwbG9yZXIgaW5pdGlhbGl6YXRpb24pXG4iXX0=
@@ -0,0 +1,59 @@
1
+ import { DummyLogger } from '@rosen-bridge/abstract-logger';
2
+ import { BlockInfo, Transaction } from '@rosen-bridge/scanner-interfaces';
3
+ import { ExtendedTransaction } from '../../interfaces';
4
+ export declare class ExplorerInitializationStrategy {
5
+ private address;
6
+ private maxWorkers;
7
+ private processTransactions;
8
+ private processTransactionBatch;
9
+ private logger;
10
+ private network;
11
+ private extraLargeBlocks;
12
+ private promiseQueue;
13
+ private workerManager;
14
+ constructor(url: string, address: string, maxWorkers: number, processTransactions: (txs: Transaction[], block: BlockInfo) => Promise<boolean>, processTransactionBatch: (txs: ExtendedTransaction[]) => Promise<void>, logger?: DummyLogger);
15
+ /**
16
+ * Get height range transaction count
17
+ * retry the request to avoid failure in case of accidental network issues
18
+ * @param fromHeight
19
+ * @param toHeight
20
+ * @returns transaction count
21
+ */
22
+ private getRangeTxCount;
23
+ /**
24
+ * Get height range transactions and process them
25
+ * retry the request to avoid failure in case of accidental network issues
26
+ * @param rangeQuery
27
+ */
28
+ private processRange;
29
+ /**
30
+ * Get block id in the specified height and process the block
31
+ * add the block to extra large blocks
32
+ * @param height
33
+ */
34
+ private processBlockAtHeight;
35
+ /**
36
+ * Get block transactions and process them
37
+ * retry the request to avoid failure in case of accidental network issues
38
+ * @param block
39
+ */
40
+ private processBlock;
41
+ /**
42
+ * Start the worker on the assigned range
43
+ * - worker process the range and split it to smaller ones if it's not processable
44
+ * - a range is processable if:
45
+ * 1- has less than or equal to API_LIMIT transactions (Uses processRange)
46
+ * 2- contains a single block (uses processBlockAtHeight)
47
+ * - after processing a range, all older ranges are updated accordingly
48
+ * - to optimize the workflow, the worker selects the biggest processable
49
+ * range from top of the range list
50
+ * @param workerIndex
51
+ */
52
+ private startWorker;
53
+ /**
54
+ * Initialize extractor using Explorer network
55
+ * @param initialBlock
56
+ */
57
+ initialize: (initialBlock: BlockInfo) => Promise<void>;
58
+ }
59
+ //# sourceMappingURL=explorerInitializationStrategy.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"explorerInitializationStrategy.d.ts","sourceRoot":"","sources":["../../../../lib/ergo/initializers/strategies/explorerInitializationStrategy.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,WAAW,EAAE,MAAM,+BAA+B,CAAC;AAC5D,OAAO,EAAE,SAAS,EAAE,WAAW,EAAE,MAAM,kCAAkC,CAAC;AAG1E,OAAO,EAAE,mBAAmB,EAAc,MAAM,kBAAkB,CAAC;AAMnE,qBAAa,8BAA8B;IAQvC,OAAO,CAAC,OAAO;IACf,OAAO,CAAC,UAAU;IAClB,OAAO,CAAC,mBAAmB;IAI3B,OAAO,CAAC,uBAAuB;IAG/B,OAAO,CAAC,MAAM;IAhBhB,OAAO,CAAC,OAAO,CAAkB;IACjC,OAAO,CAAC,gBAAgB,CAAc;IACtC,OAAO,CAAC,YAAY,CAAS;IAC7B,OAAO,CAAC,aAAa,CAAgB;gBAGnC,GAAG,EAAE,MAAM,EACH,OAAO,EAAE,MAAM,EACf,UAAU,EAAE,MAAM,EAClB,mBAAmB,EAAE,CAC3B,GAAG,EAAE,WAAW,EAAE,EAClB,KAAK,EAAE,SAAS,KACb,OAAO,CAAC,OAAO,CAAC,EACb,uBAAuB,EAAE,CAC/B,GAAG,EAAE,mBAAmB,EAAE,KACvB,OAAO,CAAC,IAAI,CAAC,EACV,MAAM,cAAoB;IAYpC;;;;;;OAMG;IACH,OAAO,CAAC,eAAe,CAarB;IAEF;;;;OAIG;IACH,OAAO,CAAC,YAAY,CA2BlB;IAEF;;;;OAIG;IACH,OAAO,CAAC,oBAAoB,CAS1B;IAEF;;;;OAIG;IACH,OAAO,CAAC,YAAY,CASlB;IAEF;;;;;;;;;;OAUG;IACH,OAAO,CAAC,WAAW,CA+BjB;IAEF;;;OAGG;IACH,UAAU,GAAU,cAAc,SAAS,mBA0BzC;CACH"}
@@ -0,0 +1,141 @@
1
+ import PQueue from 'p-queue';
2
+ import { DummyLogger } from '@rosen-bridge/abstract-logger';
3
+ import { API_LIMIT } from '../../../constants';
4
+ import { ExplorerNetwork } from '../../networks/explorerNetwork';
5
+ import { requestWithRetrial } from '../../utils';
6
+ import { INIT_WORKERS_REASSIGN_INTERVAL } from './constants';
7
+ import { WorkerManager } from './workerManager';
8
+ export class ExplorerInitializationStrategy {
9
+ address;
10
+ maxWorkers;
11
+ processTransactions;
12
+ processTransactionBatch;
13
+ logger;
14
+ network;
15
+ extraLargeBlocks;
16
+ promiseQueue;
17
+ workerManager;
18
+ constructor(url, address, maxWorkers, processTransactions, processTransactionBatch, logger = new DummyLogger()) {
19
+ this.address = address;
20
+ this.maxWorkers = maxWorkers;
21
+ this.processTransactions = processTransactions;
22
+ this.processTransactionBatch = processTransactionBatch;
23
+ this.logger = logger;
24
+ this.network = new ExplorerNetwork(url);
25
+ this.extraLargeBlocks = [];
26
+ this.promiseQueue = new PQueue({ concurrency: maxWorkers });
27
+ this.workerManager = new WorkerManager(this.maxWorkers, this.getRangeTxCount, this.logger);
28
+ }
29
+ /**
30
+ * Get height range transaction count
31
+ * retry the request to avoid failure in case of accidental network issues
32
+ * @param fromHeight
33
+ * @param toHeight
34
+ * @returns transaction count
35
+ */
36
+ getRangeTxCount = async (fromHeight, toHeight) => {
37
+ return (await requestWithRetrial(() => this.network.getAddressTransactionsWithHeight(this.address, fromHeight, toHeight, 1), this.logger)).total;
38
+ };
39
+ /**
40
+ * Get height range transactions and process them
41
+ * retry the request to avoid failure in case of accidental network issues
42
+ * @param rangeQuery
43
+ */
44
+ processRange = async (rangeQuery) => {
45
+ if (rangeQuery.count == 0) {
46
+ this.logger.debug(`skipping range [${rangeQuery.start}, ${rangeQuery.end}] with 0 txs`);
47
+ return;
48
+ }
49
+ const txs = await requestWithRetrial(() => this.network.getAddressTransactionsWithHeight(this.address, rangeQuery.start, rangeQuery.end), this.logger);
50
+ if (txs.total != rangeQuery.count)
51
+ this.logger.error(`Impossible behavior: Range query count ${rangeQuery.count} differs from total ${txs.total} for range [${rangeQuery.start}, ${rangeQuery.end}]`);
52
+ this.logger.debug(`Processing started for [${rangeQuery.start}, ${rangeQuery.end}] with ${txs.total} txs`);
53
+ await this.processTransactionBatch(txs.items);
54
+ this.logger.debug(`Processing finished for [${rangeQuery.start}, ${rangeQuery.end}] with ${txs.total} txs`);
55
+ };
56
+ /**
57
+ * Get block id in the specified height and process the block
58
+ * add the block to extra large blocks
59
+ * @param height
60
+ */
61
+ processBlockAtHeight = async (height) => {
62
+ const blockId = await requestWithRetrial(() => this.network.getBlockIdAtHeight(height), this.logger);
63
+ const block = { hash: blockId, height: height };
64
+ await this.processBlock(block);
65
+ this.extraLargeBlocks.push(block);
66
+ this.logger.debug(`Added block at height ${height} to extra large blocks`);
67
+ };
68
+ /**
69
+ * Get block transactions and process them
70
+ * retry the request to avoid failure in case of accidental network issues
71
+ * @param block
72
+ */
73
+ processBlock = async (block) => {
74
+ const blockTxs = await requestWithRetrial(() => this.network.getBlockTxs(block.hash), this.logger);
75
+ this.logger.debug(`Found ${blockTxs.length} transactions at height ${block.height}`);
76
+ await this.processTransactions(blockTxs, block);
77
+ };
78
+ /**
79
+ * Start the worker on the assigned range
80
+ * - worker process the range and split it to smaller ones if it's not processable
81
+ * - a range is processable if:
82
+ * 1- has less than or equal to API_LIMIT transactions (Uses processRange)
83
+ * 2- contains a single block (uses processBlockAtHeight)
84
+ * - after processing a range, all older ranges are updated accordingly
85
+ * - to optimize the workflow, the worker selects the biggest processable
86
+ * range from top of the range list
87
+ * @param workerIndex
88
+ */
89
+ startWorker = async (workerIndex) => {
90
+ while (this.workerManager.isWorkerActive(workerIndex)) {
91
+ const lastRangeQuery = await this.workerManager.getLastRange(workerIndex, API_LIMIT);
92
+ this.logger.debug(`Worker-${workerIndex} is checking range query ${JSON.stringify(lastRangeQuery)}`);
93
+ if (lastRangeQuery.count > API_LIMIT &&
94
+ lastRangeQuery.start != lastRangeQuery.end) {
95
+ // range is not processable
96
+ await this.workerManager.limitLastRange(workerIndex);
97
+ }
98
+ else {
99
+ // range is processable
100
+ if (lastRangeQuery.count <= API_LIMIT) {
101
+ this.logger.debug(`Processing transactions in range query`);
102
+ await this.processRange(lastRangeQuery);
103
+ }
104
+ else {
105
+ this.logger.debug(`processing extra large block at height ${lastRangeQuery.start}`);
106
+ await this.processBlockAtHeight(lastRangeQuery.start);
107
+ }
108
+ await this.workerManager.popLastRangeQuery(workerIndex);
109
+ }
110
+ }
111
+ };
112
+ /**
113
+ * Initialize extractor using Explorer network
114
+ * @param initialBlock
115
+ */
116
+ initialize = async (initialBlock) => {
117
+ this.workerManager.setup(initialBlock.height);
118
+ const addWorkerJob = (i) => this.promiseQueue.add(() => this.startWorker(i));
119
+ // Initialize the workers
120
+ await Promise.all(Array.from({ length: this.maxWorkers }, async (_, i) => {
121
+ await this.workerManager.registerWorker(i);
122
+ addWorkerJob(i);
123
+ }));
124
+ // Periodically check for idle workers and reassign a new range to them
125
+ const reassignWorker = setInterval(async () => {
126
+ const newWorkers = await this.workerManager.reassignIdleWorkers();
127
+ if (newWorkers.length > 0) {
128
+ this.logger.debug(`Reassigned workers ${newWorkers}`);
129
+ newWorkers.forEach((workerIndex) => addWorkerJob(workerIndex));
130
+ }
131
+ }, INIT_WORKERS_REASSIGN_INTERVAL);
132
+ // Wait for all workers to finish their jobs
133
+ await this.promiseQueue.onIdle();
134
+ // Stop reassigning interval
135
+ clearInterval(reassignWorker);
136
+ for (const block of this.extraLargeBlocks) {
137
+ await this.processBlock(block);
138
+ }
139
+ };
140
+ }
141
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"explorerInitializationStrategy.js","sourceRoot":"","sources":["../../../../lib/ergo/initializers/strategies/explorerInitializationStrategy.ts"],"names":[],"mappings":"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","sourcesContent":["import PQueue from 'p-queue';\n\nimport { DummyLogger } from '@rosen-bridge/abstract-logger';\nimport { BlockInfo, Transaction } from '@rosen-bridge/scanner-interfaces';\n\nimport { API_LIMIT } from '../../../constants';\nimport { ExtendedTransaction, RangeQuery } from '../../interfaces';\nimport { ExplorerNetwork } from '../../networks/explorerNetwork';\nimport { requestWithRetrial } from '../../utils';\nimport { INIT_WORKERS_REASSIGN_INTERVAL } from './constants';\nimport { WorkerManager } from './workerManager';\n\nexport class ExplorerInitializationStrategy {\n  private network: ExplorerNetwork;\n  private extraLargeBlocks: BlockInfo[];\n  private promiseQueue: PQueue;\n  private workerManager: WorkerManager;\n\n  constructor(\n    url: string,\n    private address: string,\n    private maxWorkers: number,\n    private processTransactions: (\n      txs: Transaction[],\n      block: BlockInfo,\n    ) => Promise<boolean>,\n    private processTransactionBatch: (\n      txs: ExtendedTransaction[],\n    ) => Promise<void>,\n    private logger = new DummyLogger(),\n  ) {\n    this.network = new ExplorerNetwork(url);\n    this.extraLargeBlocks = [];\n    this.promiseQueue = new PQueue({ concurrency: maxWorkers });\n    this.workerManager = new WorkerManager(\n      this.maxWorkers,\n      this.getRangeTxCount,\n      this.logger,\n    );\n  }\n\n  /**\n   * Get height range transaction count\n   * retry the request to avoid failure in case of accidental network issues\n   * @param fromHeight\n   * @param toHeight\n   * @returns transaction count\n   */\n  private getRangeTxCount = async (fromHeight: number, toHeight: number) => {\n    return (\n      await requestWithRetrial(\n        () =>\n          this.network.getAddressTransactionsWithHeight(\n            this.address,\n            fromHeight,\n            toHeight,\n            1,\n          ),\n        this.logger,\n      )\n    ).total;\n  };\n\n  /**\n   * Get height range transactions and process them\n   * retry the request to avoid failure in case of accidental network issues\n   * @param rangeQuery\n   */\n  private processRange = async (rangeQuery: RangeQuery): Promise<void> => {\n    if (rangeQuery.count == 0) {\n      this.logger.debug(\n        `skipping range [${rangeQuery.start}, ${rangeQuery.end}] with 0 txs`,\n      );\n      return;\n    }\n    const txs = await requestWithRetrial(\n      () =>\n        this.network.getAddressTransactionsWithHeight(\n          this.address,\n          rangeQuery.start,\n          rangeQuery.end,\n        ),\n      this.logger,\n    );\n    if (txs.total != rangeQuery.count)\n      this.logger.error(\n        `Impossible behavior: Range query count ${rangeQuery.count} differs from total ${txs.total} for range [${rangeQuery.start}, ${rangeQuery.end}]`,\n      );\n    this.logger.debug(\n      `Processing started for [${rangeQuery.start}, ${rangeQuery.end}] with ${txs.total} txs`,\n    );\n    await this.processTransactionBatch(txs.items);\n    this.logger.debug(\n      `Processing finished for [${rangeQuery.start}, ${rangeQuery.end}] with ${txs.total} txs`,\n    );\n  };\n\n  /**\n   * Get block id in the specified height and process the block\n   * add the block to extra large blocks\n   * @param height\n   */\n  private processBlockAtHeight = async (height: number) => {\n    const blockId = await requestWithRetrial(\n      () => this.network.getBlockIdAtHeight(height),\n      this.logger,\n    );\n    const block = { hash: blockId, height: height };\n    await this.processBlock(block);\n    this.extraLargeBlocks.push(block);\n    this.logger.debug(`Added block at height ${height} to extra large blocks`);\n  };\n\n  /**\n   * Get block transactions and process them\n   * retry the request to avoid failure in case of accidental network issues\n   * @param block\n   */\n  private processBlock = async (block: BlockInfo) => {\n    const blockTxs = await requestWithRetrial(\n      () => this.network.getBlockTxs(block.hash),\n      this.logger,\n    );\n    this.logger.debug(\n      `Found ${blockTxs.length} transactions at height ${block.height}`,\n    );\n    await this.processTransactions(blockTxs, block);\n  };\n\n  /**\n   * Start the worker on the assigned range\n   * - worker process the range and split it to smaller ones if it's not processable\n   * - a range is processable if:\n   *    1- has less than or equal to API_LIMIT transactions (Uses processRange)\n   *    2- contains a single block (uses processBlockAtHeight)\n   * - after processing a range, all older ranges are updated accordingly\n   * - to optimize the workflow, the worker selects the biggest processable\n   *    range from top of the range list\n   * @param workerIndex\n   */\n  private startWorker = async (workerIndex: number) => {\n    while (this.workerManager.isWorkerActive(workerIndex)) {\n      const lastRangeQuery = await this.workerManager.getLastRange(\n        workerIndex,\n        API_LIMIT,\n      );\n      this.logger.debug(\n        `Worker-${workerIndex} is checking range query ${JSON.stringify(\n          lastRangeQuery,\n        )}`,\n      );\n      if (\n        lastRangeQuery.count > API_LIMIT &&\n        lastRangeQuery.start != lastRangeQuery.end\n      ) {\n        // range is not processable\n        await this.workerManager.limitLastRange(workerIndex);\n      } else {\n        // range is processable\n        if (lastRangeQuery.count <= API_LIMIT) {\n          this.logger.debug(`Processing transactions in range query`);\n          await this.processRange(lastRangeQuery);\n        } else {\n          this.logger.debug(\n            `processing extra large block at height ${lastRangeQuery.start}`,\n          );\n          await this.processBlockAtHeight(lastRangeQuery.start);\n        }\n        await this.workerManager.popLastRangeQuery(workerIndex);\n      }\n    }\n  };\n\n  /**\n   * Initialize extractor using Explorer network\n   * @param initialBlock\n   */\n  initialize = async (initialBlock: BlockInfo) => {\n    this.workerManager.setup(initialBlock.height);\n    const addWorkerJob = (i: number) =>\n      this.promiseQueue.add(() => this.startWorker(i));\n    // Initialize the workers\n    await Promise.all(\n      Array.from({ length: this.maxWorkers }, async (_, i) => {\n        await this.workerManager.registerWorker(i);\n        addWorkerJob(i);\n      }),\n    );\n    // Periodically check for idle workers and reassign a new range to them\n    const reassignWorker = setInterval(async () => {\n      const newWorkers = await this.workerManager.reassignIdleWorkers();\n      if (newWorkers.length > 0) {\n        this.logger.debug(`Reassigned workers ${newWorkers}`);\n        newWorkers.forEach((workerIndex) => addWorkerJob(workerIndex));\n      }\n    }, INIT_WORKERS_REASSIGN_INTERVAL);\n    // Wait for all workers to finish their jobs\n    await this.promiseQueue.onIdle();\n    // Stop reassigning interval\n    clearInterval(reassignWorker);\n    for (const block of this.extraLargeBlocks) {\n      await this.processBlock(block);\n    }\n  };\n}\n"]}
@@ -0,0 +1,4 @@
1
+ export * from './explorerInitializationStrategy';
2
+ export * from './nodeInitializationStrategy';
3
+ export * from './workerManager';
4
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../../lib/ergo/initializers/strategies/index.ts"],"names":[],"mappings":"AAAA,cAAc,kCAAkC,CAAC;AACjD,cAAc,8BAA8B,CAAC;AAC7C,cAAc,iBAAiB,CAAC"}
@@ -0,0 +1,4 @@
1
+ export * from './explorerInitializationStrategy';
2
+ export * from './nodeInitializationStrategy';
3
+ export * from './workerManager';
4
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9saWIvZXJnby9pbml0aWFsaXplcnMvc3RyYXRlZ2llcy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLGtDQUFrQyxDQUFDO0FBQ2pELGNBQWMsOEJBQThCLENBQUM7QUFDN0MsY0FBYyxpQkFBaUIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vZXhwbG9yZXJJbml0aWFsaXphdGlvblN0cmF0ZWd5JztcbmV4cG9ydCAqIGZyb20gJy4vbm9kZUluaXRpYWxpemF0aW9uU3RyYXRlZ3knO1xuZXhwb3J0ICogZnJvbSAnLi93b3JrZXJNYW5hZ2VyJztcbiJdfQ==