@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.
- package/CHANGELOG.md +9 -0
- package/dist/abstractExtractor.d.ts +24 -12
- package/dist/abstractExtractor.d.ts.map +1 -1
- package/dist/abstractExtractor.js +1 -1
- package/dist/constants.d.ts +1 -0
- package/dist/constants.d.ts.map +1 -1
- package/dist/constants.js +2 -1
- package/dist/ergo/database/actions/abstractErgoAction.d.ts +83 -0
- package/dist/ergo/database/actions/abstractErgoAction.d.ts.map +1 -0
- package/dist/ergo/database/actions/abstractErgoAction.js +167 -0
- package/dist/ergo/database/actions/abstractErgoBoxAction.d.ts +31 -0
- package/dist/ergo/database/actions/abstractErgoBoxAction.d.ts.map +1 -0
- package/dist/ergo/database/actions/abstractErgoBoxAction.js +70 -0
- package/dist/ergo/database/entities/abstractErgoBoxEntity.d.ts +18 -0
- package/dist/ergo/database/entities/abstractErgoBoxEntity.d.ts.map +1 -0
- package/dist/ergo/database/entities/abstractErgoBoxEntity.js +36 -0
- package/dist/ergo/database/entities/abstractErgoEntity.d.ts +26 -0
- package/dist/ergo/database/entities/abstractErgoEntity.d.ts.map +1 -0
- package/dist/ergo/database/entities/abstractErgoEntity.js +64 -0
- package/dist/ergo/database/index.d.ts +5 -0
- package/dist/ergo/database/index.d.ts.map +1 -0
- package/dist/ergo/database/index.js +5 -0
- package/dist/ergo/extractors/abstractErgoBoxExtractor.d.ts +67 -0
- package/dist/ergo/extractors/abstractErgoBoxExtractor.d.ts.map +1 -0
- package/dist/ergo/extractors/abstractErgoBoxExtractor.js +106 -0
- package/dist/ergo/extractors/abstractErgoExtractor.d.ts +53 -0
- package/dist/ergo/extractors/abstractErgoExtractor.d.ts.map +1 -0
- package/dist/ergo/extractors/abstractErgoExtractor.js +92 -0
- package/dist/ergo/extractors/abstractErgoTxExtractor.d.ts +56 -0
- package/dist/ergo/extractors/abstractErgoTxExtractor.d.ts.map +1 -0
- package/dist/ergo/extractors/abstractErgoTxExtractor.js +81 -0
- package/dist/ergo/extractors/index.d.ts +4 -0
- package/dist/ergo/extractors/index.d.ts.map +1 -0
- package/dist/ergo/extractors/index.js +4 -0
- package/dist/ergo/index.d.ts +5 -7
- package/dist/ergo/index.d.ts.map +1 -1
- package/dist/ergo/index.js +6 -8
- package/dist/ergo/initializers/ergoBoxInitializer.d.ts +36 -0
- package/dist/ergo/initializers/ergoBoxInitializer.d.ts.map +1 -0
- package/dist/ergo/initializers/ergoBoxInitializer.js +80 -0
- package/dist/ergo/initializers/ergoInitializer.d.ts +39 -0
- package/dist/ergo/initializers/ergoInitializer.d.ts.map +1 -0
- package/dist/ergo/initializers/ergoInitializer.js +80 -0
- package/dist/ergo/initializers/index.d.ts +4 -0
- package/dist/ergo/initializers/index.d.ts.map +1 -0
- package/dist/ergo/initializers/index.js +4 -0
- package/dist/ergo/initializers/strategies/constants.d.ts +3 -0
- package/dist/ergo/initializers/strategies/constants.d.ts.map +1 -0
- package/dist/ergo/initializers/strategies/constants.js +3 -0
- package/dist/ergo/initializers/strategies/explorerInitializationStrategy.d.ts +59 -0
- package/dist/ergo/initializers/strategies/explorerInitializationStrategy.d.ts.map +1 -0
- package/dist/ergo/initializers/strategies/explorerInitializationStrategy.js +141 -0
- package/dist/ergo/initializers/strategies/index.d.ts +4 -0
- package/dist/ergo/initializers/strategies/index.d.ts.map +1 -0
- package/dist/ergo/initializers/strategies/index.js +4 -0
- package/dist/ergo/initializers/strategies/nodeInitializationStrategy.d.ts +29 -0
- package/dist/ergo/initializers/strategies/nodeInitializationStrategy.d.ts.map +1 -0
- package/dist/ergo/initializers/strategies/nodeInitializationStrategy.js +66 -0
- package/dist/ergo/initializers/strategies/workerManager.d.ts +79 -0
- package/dist/ergo/initializers/strategies/workerManager.d.ts.map +1 -0
- package/dist/ergo/initializers/strategies/workerManager.js +183 -0
- package/dist/ergo/interfaces.d.ts +31 -17
- package/dist/ergo/interfaces.d.ts.map +1 -1
- package/dist/ergo/interfaces.js +1 -1
- package/dist/ergo/networks/explorerNetwork.d.ts +52 -0
- package/dist/ergo/networks/explorerNetwork.d.ts.map +1 -0
- package/dist/ergo/networks/explorerNetwork.js +127 -0
- package/dist/ergo/networks/index.d.ts +3 -0
- package/dist/ergo/networks/index.d.ts.map +1 -0
- package/dist/ergo/networks/index.js +3 -0
- package/dist/ergo/networks/nodeNetwork.d.ts +28 -0
- package/dist/ergo/networks/nodeNetwork.d.ts.map +1 -0
- package/dist/ergo/networks/nodeNetwork.js +59 -0
- package/dist/ergo/utils.d.ts +15 -0
- package/dist/ergo/utils.d.ts.map +1 -1
- package/dist/ergo/utils.js +34 -1
- package/package.json +3 -1
- package/dist/ergo/abstractErgoExtractor.d.ts +0 -80
- package/dist/ergo/abstractErgoExtractor.d.ts.map +0 -1
- package/dist/ergo/abstractErgoExtractor.js +0 -142
- package/dist/ergo/abstractErgoExtractorAction.d.ts +0 -89
- package/dist/ergo/abstractErgoExtractorAction.d.ts.map +0 -1
- package/dist/ergo/abstractErgoExtractorAction.js +0 -219
- package/dist/ergo/abstractErgoExtractorEntity.d.ts +0 -11
- package/dist/ergo/abstractErgoExtractorEntity.d.ts.map +0 -1
- package/dist/ergo/abstractErgoExtractorEntity.js +0 -57
- package/dist/ergo/initializable/abstractInitializable.d.ts +0 -48
- package/dist/ergo/initializable/abstractInitializable.d.ts.map +0 -1
- package/dist/ergo/initializable/abstractInitializable.js +0 -162
- package/dist/ergo/initializable/abstractInitializableAction.d.ts +0 -14
- package/dist/ergo/initializable/abstractInitializableAction.d.ts.map +0 -1
- package/dist/ergo/initializable/abstractInitializableAction.js +0 -16
- package/dist/ergo/initializable/index.d.ts +0 -3
- package/dist/ergo/initializable/index.d.ts.map +0 -1
- package/dist/ergo/initializable/index.js +0 -3
- package/dist/ergo/network/abstractNetwork.d.ts +0 -26
- package/dist/ergo/network/abstractNetwork.d.ts.map +0 -1
- package/dist/ergo/network/abstractNetwork.js +0 -3
- package/dist/ergo/network/explorerNetwork.d.ts +0 -74
- package/dist/ergo/network/explorerNetwork.d.ts.map +0 -1
- package/dist/ergo/network/explorerNetwork.js +0 -185
- package/dist/ergo/network/nodeNetwork.d.ts +0 -60
- package/dist/ergo/network/nodeNetwork.d.ts.map +0 -1
- package/dist/ergo/network/nodeNetwork.js +0 -131
package/dist/ergo/index.d.ts
CHANGED
|
@@ -1,10 +1,8 @@
|
|
|
1
|
-
export * from './
|
|
2
|
-
export * from './
|
|
1
|
+
export * from './extractors';
|
|
2
|
+
export * from './initializers';
|
|
3
|
+
export * from './database';
|
|
3
4
|
export * from './interfaces';
|
|
4
|
-
export * from './
|
|
5
|
-
export * from './
|
|
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
|
package/dist/ergo/index.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../lib/ergo/index.ts"],"names":[],"mappings":"AAAA,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"}
|
package/dist/ergo/index.js
CHANGED
|
@@ -1,10 +1,8 @@
|
|
|
1
|
-
export * from './
|
|
2
|
-
export * from './
|
|
1
|
+
export * from './extractors';
|
|
2
|
+
export * from './initializers';
|
|
3
|
+
export * from './database';
|
|
3
4
|
export * from './interfaces';
|
|
4
|
-
export * from './
|
|
5
|
-
export * from './
|
|
6
|
-
export * from './network/abstractNetwork';
|
|
7
|
-
export * from './initializable';
|
|
5
|
+
export * from './networks';
|
|
6
|
+
export * from './initializers';
|
|
8
7
|
export * from './utils';
|
|
9
|
-
|
|
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 @@
|
|
|
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 @@
|
|
|
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 @@
|
|
|
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==
|