@rosen-bridge/abstract-extractor 0.3.0 → 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +20 -0
- package/dist/ergo/AbstractErgoExtractor.d.ts +31 -8
- package/dist/ergo/AbstractErgoExtractor.d.ts.map +1 -1
- package/dist/ergo/AbstractErgoExtractor.js +73 -8
- package/dist/ergo/AbstractErgoExtractorAction.d.ts +38 -10
- package/dist/ergo/AbstractErgoExtractorAction.d.ts.map +1 -1
- package/dist/ergo/AbstractErgoExtractorAction.js +134 -1
- package/dist/ergo/AbstractErgoExtractorEntity.d.ts +11 -0
- package/dist/ergo/AbstractErgoExtractorEntity.d.ts.map +1 -0
- package/dist/ergo/AbstractErgoExtractorEntity.js +57 -0
- package/dist/ergo/index.d.ts +1 -0
- package/dist/ergo/index.d.ts.map +1 -1
- package/dist/ergo/index.js +2 -1
- package/dist/ergo/initializable/AbstractInitializable.d.ts +4 -3
- package/dist/ergo/initializable/AbstractInitializable.d.ts.map +1 -1
- package/dist/ergo/initializable/AbstractInitializable.js +9 -5
- package/dist/ergo/initializable/AbstractInitializableAction.d.ts +7 -2
- package/dist/ergo/initializable/AbstractInitializableAction.d.ts.map +1 -1
- package/dist/ergo/initializable/AbstractInitializableAction.js +11 -1
- package/dist/ergo/interfaces.d.ts +21 -7
- package/dist/ergo/interfaces.d.ts.map +1 -1
- package/dist/ergo/interfaces.js +8 -1
- package/lib/ergo/AbstractErgoExtractor.ts +107 -23
- package/lib/ergo/AbstractErgoExtractorAction.ts +187 -18
- package/lib/ergo/AbstractErgoExtractorEntity.ts +28 -0
- package/lib/ergo/index.ts +1 -0
- package/lib/ergo/initializable/AbstractInitializable.ts +22 -9
- package/lib/ergo/initializable/AbstractInitializableAction.ts +19 -3
- package/lib/ergo/interfaces.ts +25 -7
- package/package.json +6 -2
- package/tests/{AbstractExtractor.mock.ts → AbstractErgoExtractor.mock.ts} +12 -7
- package/tests/AbstractErgoExtractor.spec.ts +281 -0
- package/tests/AbstractErgoExtractorAction.mock.ts +45 -0
- package/tests/AbstractErgoExtractorAction.spec.ts +269 -0
- package/tests/initializable/AbstractInitializable.mock.ts +15 -8
- package/tests/initializable/AbstractInitializable.spec.ts +37 -5
- package/tests/initializable/AbstractInitializableAction.mock.ts +45 -0
- package/tests/initializable/AbstractInitializableAction.spec.ts +65 -0
- package/tests/testData.ts +38 -2
- package/tests/testUtils.ts +22 -0
- package/tsconfig.build.tsbuildinfo +1 -1
- package/dist/ergo/initializable/InitializableByAddress.d.ts +0 -19
- package/dist/ergo/initializable/InitializableByAddress.d.ts.map +0 -1
- package/dist/ergo/initializable/InitializableByAddress.js +0 -30
- package/dist/ergo/initializable/InitializableByToken.d.ts +0 -19
- package/dist/ergo/initializable/InitializableByToken.d.ts.map +0 -1
- package/dist/ergo/initializable/InitializableByToken.js +0 -30
- package/dist/tsconfig.tsbuildinfo +0 -1
- package/tests/AbstractExtractor.spec.ts +0 -106
|
@@ -1,9 +1,9 @@
|
|
|
1
|
+
import { groupBy, sortBy } from 'lodash-es';
|
|
1
2
|
import { ErgoNetworkType, } from '../interfaces';
|
|
2
3
|
import { API_LIMIT, RETRIAL_COUNT } from '../../constants';
|
|
3
4
|
import { AbstractErgoExtractor } from '../AbstractErgoExtractor';
|
|
4
5
|
import { ExplorerNetwork } from '../network/ExplorerNetwork';
|
|
5
6
|
import { NodeNetwork } from '../network/NodeNetwork';
|
|
6
|
-
import { groupBy, sortBy } from 'lodash-es';
|
|
7
7
|
export class AbstractInitializableErgoExtractor extends AbstractErgoExtractor {
|
|
8
8
|
initialize;
|
|
9
9
|
address;
|
|
@@ -80,13 +80,13 @@ export class AbstractInitializableErgoExtractor extends AbstractErgoExtractor {
|
|
|
80
80
|
*/
|
|
81
81
|
initializeWithNode = async (initialBlock) => {
|
|
82
82
|
const txCountBeforeInit = await this.getTotalTxCount();
|
|
83
|
+
let offset = 0, total = 1, round = 1;
|
|
83
84
|
await this.initWithRetrial(async () => {
|
|
84
85
|
// Repeat the whole process twice to cover all spent boxes
|
|
85
86
|
// After round 1 all boxes have been saved and processed once
|
|
86
87
|
// After round 2 spending information of all stored boxes are updated successfully
|
|
87
|
-
|
|
88
|
+
while (round <= 2) {
|
|
88
89
|
this.logger.debug(`Starting round ${round} of initialization`);
|
|
89
|
-
let offset = 0, total = 1;
|
|
90
90
|
while (offset < total) {
|
|
91
91
|
const response = await this.network.getAddressTransactionsWithOffsetLimit(this.address, offset, API_LIMIT);
|
|
92
92
|
total = response.total;
|
|
@@ -96,6 +96,8 @@ export class AbstractInitializableErgoExtractor extends AbstractErgoExtractor {
|
|
|
96
96
|
await this.processTransactionBatch(txs);
|
|
97
97
|
offset += API_LIMIT;
|
|
98
98
|
}
|
|
99
|
+
round++;
|
|
100
|
+
offset = 0; // next round initial offset
|
|
99
101
|
}
|
|
100
102
|
});
|
|
101
103
|
const txCountAfterInit = await this.getTotalTxCount();
|
|
@@ -116,7 +118,9 @@ export class AbstractInitializableErgoExtractor extends AbstractErgoExtractor {
|
|
|
116
118
|
const blockTxs = groupedTxs[blockId];
|
|
117
119
|
const block = { hash: blockId, height: blockTxs[0].inclusionHeight };
|
|
118
120
|
this.logger.debug(`Processing transactions at height ${blockTxs[0].inclusionHeight}`);
|
|
119
|
-
await this.processTransactions(blockTxs, block);
|
|
121
|
+
const success = await this.processTransactions(blockTxs, block);
|
|
122
|
+
if (!success)
|
|
123
|
+
throw Error(`Processing transactions failed at height ${blockTxs[0].inclusionHeight}`);
|
|
120
124
|
}
|
|
121
125
|
};
|
|
122
126
|
/**
|
|
@@ -156,4 +160,4 @@ export class AbstractInitializableErgoExtractor extends AbstractErgoExtractor {
|
|
|
156
160
|
*/
|
|
157
161
|
initializeBoxes;
|
|
158
162
|
}
|
|
159
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"AbstractInitializable.js","sourceRoot":"","sources":["../../../lib/ergo/initializable/AbstractInitializable.ts"],"names":[],"mappings":"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","sourcesContent":["import { AbstractLogger } from '@rosen-bridge/abstract-logger';\nimport {\n  ErgoExtractedData,\n  ErgoNetworkType,\n  ExtendedTransaction,\n} from '../interfaces';\nimport { API_LIMIT, RETRIAL_COUNT } from '../../constants';\nimport { AbstractErgoExtractor } from '../AbstractErgoExtractor';\nimport { AbstractInitializableErgoExtractorAction } from './AbstractInitializableAction';\nimport { BlockInfo } from '../../interfaces';\nimport { ExplorerNetwork } from '../network/ExplorerNetwork';\nimport { NodeNetwork } from '../network/NodeNetwork';\nimport { groupBy, sortBy } from 'lodash-es';\n\nexport abstract class AbstractInitializableErgoExtractor<\n  ExtractedData extends ErgoExtractedData\n> extends AbstractErgoExtractor<ExtractedData> {\n  protected initialize: boolean;\n  private address: string;\n  protected abstract actions: AbstractInitializableErgoExtractorAction<ExtractedData>;\n\n  private network: ExplorerNetwork | NodeNetwork;\n\n  constructor(\n    type: ErgoNetworkType,\n    url: string,\n    address: string,\n    logger?: AbstractLogger,\n    initialize = true\n  ) {\n    super(logger);\n    this.address = address;\n    this.initialize = initialize;\n    if (type == ErgoNetworkType.Explorer) {\n      this.network = new ExplorerNetwork(url);\n      this.initializeBoxes = (initialBlock: BlockInfo) => {\n        return this.initializeWithExplorer(initialBlock);\n      };\n    } else if (type == ErgoNetworkType.Node) {\n      this.network = new NodeNetwork(url);\n      this.initializeBoxes = (initialBlock: BlockInfo) => {\n        return this.initializeWithNode(initialBlock);\n      };\n    } else throw new Error(`Network type ${type} is not supported`);\n  }\n\n  /**\n   * Initialize extractor using Explorer network\n   * @param initialBlock\n   */\n  private initializeWithExplorer = async (initialBlock: BlockInfo) => {\n    const explorerNetwork = this.network as ExplorerNetwork;\n    let fromHeight = 0,\n      toHeight = initialBlock.height;\n    await this.initWithRetrial(async () => {\n      while (fromHeight < toHeight) {\n        let txs: Array<ExtendedTransaction>;\n        // eslint-disable-next-line no-constant-condition\n        while (true) {\n          txs = await explorerNetwork.getAddressTransactionsWithHeight(\n            this.address,\n            fromHeight,\n            toHeight\n          );\n          this.logger.debug(\n            `Found ${txs.length} transactions for the address from height ${fromHeight} to height ${toHeight}`\n          );\n          if (txs.length < API_LIMIT || fromHeight === toHeight) {\n            break; // Exit loop if we have fewer transactions than the limit or if the range is reduced to a single height\n          }\n          toHeight = Math.floor((toHeight - fromHeight) / 2) + fromHeight;\n          this.logger.debug(\n            `Limiting the query height range to [${fromHeight}, ${toHeight}]`\n          );\n        }\n        if (txs.length < API_LIMIT) {\n          if (txs.length > 0) await this.processTransactionBatch(txs);\n        } else {\n          this.logger.debug(\n            `Block at height ${fromHeight} has more than (or equal) ${API_LIMIT} relevant txs, processing all txs in the block`\n          );\n          const blockId = await explorerNetwork.getBlockIdAtHeight(fromHeight);\n          const blockTxs = await explorerNetwork.getBlockTxs(blockId);\n          this.logger.debug(\n            `Found ${blockTxs.length} transactions at height ${fromHeight}`\n          );\n          await this.processTransactions(blockTxs, {\n            hash: blockId,\n            height: fromHeight,\n          });\n        }\n        fromHeight = toHeight + 1;\n        toHeight = initialBlock.height;\n      }\n    });\n  };\n\n  /**\n   * Get the total tx count from Node network\n   * @returns total tx count of the address\n   */\n  private getTotalTxCount = async () => {\n    const response = await (\n      this.network as NodeNetwork\n    ).getAddressTransactionsWithOffsetLimit(this.address, 0, 0);\n    return response.total;\n  };\n\n  /**\n   * Initialize extractor using Node network\n   * @param initialBlock\n   */\n  private initializeWithNode = async (initialBlock: BlockInfo) => {\n    const txCountBeforeInit = await this.getTotalTxCount();\n    await this.initWithRetrial(async () => {\n      // Repeat the whole process twice to cover all spent boxes\n      // After round 1 all boxes have been saved and processed once\n      // After round 2 spending information of all stored boxes are updated successfully\n      for (let round = 0; round <= 1; round++) {\n        this.logger.debug(`Starting round ${round} of initialization`);\n        let offset = 0,\n          total = 1;\n        while (offset < total) {\n          const response = await (\n            this.network as NodeNetwork\n          ).getAddressTransactionsWithOffsetLimit(\n            this.address,\n            offset,\n            API_LIMIT\n          );\n          total = response.total;\n          const txs = response.items.filter(\n            (tx) => tx.inclusionHeight <= initialBlock.height\n          );\n          this.logger.debug(\n            `Found ${txs.length} transactions below the initial height with offset ${offset} and total number of transactions ${total}`\n          );\n          if (txs.length > 0) await this.processTransactionBatch(txs);\n          offset += API_LIMIT;\n        }\n      }\n    });\n    const txCountAfterInit = await this.getTotalTxCount();\n    if (txCountAfterInit != txCountBeforeInit) {\n      throw Error(\n        'Total transaction count changed during initialization phase, the stored data is not valid'\n      );\n    }\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    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      await this.processTransactions(blockTxs, block);\n    }\n  };\n\n  /**\n   * Initialize the extractor with retrial on any unexpected problem\n   * its the common part of initialize with Node and Explorer network\n   * @param job\n   */\n  private initWithRetrial = async (job: () => Promise<void>) => {\n    let trial = 1;\n    if (this.initialize) {\n      this.logger.debug(\n        `Initializing ${this.getId()} started, removing all existing data`\n      );\n      await this.actions.removeAllData(this.getId());\n      while (trial <= RETRIAL_COUNT) {\n        try {\n          await job();\n          break;\n        } catch (e) {\n          this.logger.warn(\n            `Initialization for ${this.getId()} failed with error :${e}`\n          );\n          if (trial == RETRIAL_COUNT)\n            throw Error(\n              `Initialization for ${this.getId()} failed after ${RETRIAL_COUNT} retrial`\n            );\n          trial += 1;\n          this.logger.info(\n            `Trying again to initialize ${this.getId()} with trial step ${trial}`\n          );\n        }\n      }\n      this.logger.info(\n        `Initialization completed successfully for ${this.getId()}`\n      );\n    } else {\n      this.logger.info(`Initialization for ${this.getId()} is turned off`);\n    }\n  };\n\n  /**\n   * initialize extractor database with data created below the initial height\n   * ignore initialization if this feature is off\n   * try to get data multiple times to pass accidental network problems\n   * @param initialBlock\n   */\n  initializeBoxes: (initialBlock: BlockInfo) => Promise<void>;\n}\n"]}
|
|
163
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"AbstractInitializable.js","sourceRoot":"","sources":["../../../lib/ergo/initializable/AbstractInitializable.ts"],"names":[],"mappings":"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","sourcesContent":["import { AbstractLogger } from '@rosen-bridge/abstract-logger';\nimport { groupBy, sortBy } from 'lodash-es';\n\nimport {\n  AbstractBoxData,\n  ErgoNetworkType,\n  ExtendedTransaction,\n} from '../interfaces';\nimport { API_LIMIT, RETRIAL_COUNT } from '../../constants';\nimport { AbstractErgoExtractor } from '../AbstractErgoExtractor';\nimport { AbstractInitializableErgoExtractorAction } from './AbstractInitializableAction';\nimport { BlockInfo } from '../../interfaces';\nimport { ExplorerNetwork } from '../network/ExplorerNetwork';\nimport { NodeNetwork } from '../network/NodeNetwork';\nimport { AbstractErgoExtractorEntity } from '../AbstractErgoExtractorEntity';\n\nexport abstract class AbstractInitializableErgoExtractor<\n  ExtractedData extends AbstractBoxData,\n  ExtractorEntity extends AbstractErgoExtractorEntity\n> extends AbstractErgoExtractor<ExtractedData, ExtractorEntity> {\n  protected initialize: boolean;\n  private address: string;\n  protected abstract actions: AbstractInitializableErgoExtractorAction<\n    ExtractedData,\n    ExtractorEntity\n  >;\n\n  private network: ExplorerNetwork | NodeNetwork;\n\n  constructor(\n    type: ErgoNetworkType,\n    url: string,\n    address: string,\n    logger?: AbstractLogger,\n    initialize = true\n  ) {\n    super(logger);\n    this.address = address;\n    this.initialize = initialize;\n    if (type == ErgoNetworkType.Explorer) {\n      this.network = new ExplorerNetwork(url);\n      this.initializeBoxes = (initialBlock: BlockInfo) => {\n        return this.initializeWithExplorer(initialBlock);\n      };\n    } else if (type == ErgoNetworkType.Node) {\n      this.network = new NodeNetwork(url);\n      this.initializeBoxes = (initialBlock: BlockInfo) => {\n        return this.initializeWithNode(initialBlock);\n      };\n    } else throw new Error(`Network type ${type} is not supported`);\n  }\n\n  /**\n   * Initialize extractor using Explorer network\n   * @param initialBlock\n   */\n  private initializeWithExplorer = async (initialBlock: BlockInfo) => {\n    const explorerNetwork = this.network as ExplorerNetwork;\n    let fromHeight = 0,\n      toHeight = initialBlock.height;\n    await this.initWithRetrial(async () => {\n      while (fromHeight < toHeight) {\n        let txs: Array<ExtendedTransaction>;\n        // eslint-disable-next-line no-constant-condition\n        while (true) {\n          txs = await explorerNetwork.getAddressTransactionsWithHeight(\n            this.address,\n            fromHeight,\n            toHeight\n          );\n          this.logger.debug(\n            `Found ${txs.length} transactions for the address from height ${fromHeight} to height ${toHeight}`\n          );\n          if (txs.length < API_LIMIT || fromHeight === toHeight) {\n            break; // Exit loop if we have fewer transactions than the limit or if the range is reduced to a single height\n          }\n          toHeight = Math.floor((toHeight - fromHeight) / 2) + fromHeight;\n          this.logger.debug(\n            `Limiting the query height range to [${fromHeight}, ${toHeight}]`\n          );\n        }\n        if (txs.length < API_LIMIT) {\n          if (txs.length > 0) await this.processTransactionBatch(txs);\n        } else {\n          this.logger.debug(\n            `Block at height ${fromHeight} has more than (or equal) ${API_LIMIT} relevant txs, processing all txs in the block`\n          );\n          const blockId = await explorerNetwork.getBlockIdAtHeight(fromHeight);\n          const blockTxs = await explorerNetwork.getBlockTxs(blockId);\n          this.logger.debug(\n            `Found ${blockTxs.length} transactions at height ${fromHeight}`\n          );\n          await this.processTransactions(blockTxs, {\n            hash: blockId,\n            height: fromHeight,\n          });\n        }\n        fromHeight = toHeight + 1;\n        toHeight = initialBlock.height;\n      }\n    });\n  };\n\n  /**\n   * Get the total tx count from Node network\n   * @returns total tx count of the address\n   */\n  private getTotalTxCount = async () => {\n    const response = await (\n      this.network as NodeNetwork\n    ).getAddressTransactionsWithOffsetLimit(this.address, 0, 0);\n    return response.total;\n  };\n\n  /**\n   * Initialize extractor using Node network\n   * @param initialBlock\n   */\n  private initializeWithNode = async (initialBlock: BlockInfo) => {\n    const txCountBeforeInit = await this.getTotalTxCount();\n    let offset = 0,\n      total = 1,\n      round = 1;\n    await this.initWithRetrial(async () => {\n      // Repeat the whole process twice to cover all spent boxes\n      // After round 1 all boxes have been saved and processed once\n      // After round 2 spending information of all stored boxes are updated successfully\n      while (round <= 2) {\n        this.logger.debug(`Starting round ${round} of initialization`);\n        while (offset < total) {\n          const response = await (\n            this.network as NodeNetwork\n          ).getAddressTransactionsWithOffsetLimit(\n            this.address,\n            offset,\n            API_LIMIT\n          );\n          total = response.total;\n          const txs = response.items.filter(\n            (tx) => tx.inclusionHeight <= initialBlock.height\n          );\n          this.logger.debug(\n            `Found ${txs.length} transactions below the initial height with offset ${offset} and total number of transactions ${total}`\n          );\n          if (txs.length > 0) await this.processTransactionBatch(txs);\n          offset += API_LIMIT;\n        }\n        round++;\n        offset = 0; // next round initial offset\n      }\n    });\n    const txCountAfterInit = await this.getTotalTxCount();\n    if (txCountAfterInit != txCountBeforeInit) {\n      throw Error(\n        'Total transaction count changed during initialization phase, the stored data is not valid'\n      );\n    }\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    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        throw Error(\n          `Processing transactions failed at height ${blockTxs[0].inclusionHeight}`\n        );\n    }\n  };\n\n  /**\n   * Initialize the extractor with retrial on any unexpected problem\n   * its the common part of initialize with Node and Explorer network\n   * @param job\n   */\n  private initWithRetrial = async (job: () => Promise<void>) => {\n    let trial = 1;\n    if (this.initialize) {\n      this.logger.debug(\n        `Initializing ${this.getId()} started, removing all existing data`\n      );\n      await this.actions.removeAllData(this.getId());\n      while (trial <= RETRIAL_COUNT) {\n        try {\n          await job();\n          break;\n        } catch (e) {\n          this.logger.warn(\n            `Initialization for ${this.getId()} failed with error :${e}`\n          );\n          if (trial == RETRIAL_COUNT)\n            throw Error(\n              `Initialization for ${this.getId()} failed after ${RETRIAL_COUNT} retrial`\n            );\n          trial += 1;\n          this.logger.info(\n            `Trying again to initialize ${this.getId()} with trial step ${trial}`\n          );\n        }\n      }\n      this.logger.info(\n        `Initialization completed successfully for ${this.getId()}`\n      );\n    } else {\n      this.logger.info(`Initialization for ${this.getId()} is turned off`);\n    }\n  };\n\n  /**\n   * initialize extractor database with data created below the initial height\n   * ignore initialization if this feature is off\n   * try to get data multiple times to pass accidental network problems\n   * @param initialBlock\n   */\n  initializeBoxes: (initialBlock: BlockInfo) => Promise<void>;\n}\n"]}
|
|
@@ -1,9 +1,14 @@
|
|
|
1
|
+
import { DataSource, EntityTarget } from 'typeorm';
|
|
2
|
+
import { AbstractLogger } from '@rosen-bridge/abstract-logger';
|
|
1
3
|
import { AbstractErgoExtractorAction } from '../AbstractErgoExtractorAction';
|
|
2
|
-
|
|
4
|
+
import { AbstractBoxData } from '../interfaces';
|
|
5
|
+
import { AbstractErgoExtractorEntity } from '../AbstractErgoExtractorEntity';
|
|
6
|
+
export declare abstract class AbstractInitializableErgoExtractorAction<ExtractedData extends AbstractBoxData, ExtractorEntity extends AbstractErgoExtractorEntity> extends AbstractErgoExtractorAction<ExtractedData, ExtractorEntity> {
|
|
7
|
+
constructor(dataSource: DataSource, repo: EntityTarget<ExtractorEntity>, logger?: AbstractLogger);
|
|
3
8
|
/**
|
|
4
9
|
* remove all existing data for the extractor
|
|
5
10
|
* @param extractorId
|
|
6
11
|
*/
|
|
7
|
-
|
|
12
|
+
removeAllData: (extractorId: string) => Promise<void>;
|
|
8
13
|
}
|
|
9
14
|
//# sourceMappingURL=AbstractInitializableAction.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"AbstractInitializableAction.d.ts","sourceRoot":"","sources":["../../../lib/ergo/initializable/AbstractInitializableAction.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,2BAA2B,EAAE,MAAM,gCAAgC,CAAC;AAE7E,8BAAsB,wCAAwC,CAC5D,aAAa,
|
|
1
|
+
{"version":3,"file":"AbstractInitializableAction.d.ts","sourceRoot":"","sources":["../../../lib/ergo/initializable/AbstractInitializableAction.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,YAAY,EAAE,MAAM,SAAS,CAAC;AACnD,OAAO,EAAE,cAAc,EAAE,MAAM,+BAA+B,CAAC;AAE/D,OAAO,EAAE,2BAA2B,EAAE,MAAM,gCAAgC,CAAC;AAC7E,OAAO,EAAE,eAAe,EAAE,MAAM,eAAe,CAAC;AAChD,OAAO,EAAE,2BAA2B,EAAE,MAAM,gCAAgC,CAAC;AAE7E,8BAAsB,wCAAwC,CAC5D,aAAa,SAAS,eAAe,EACrC,eAAe,SAAS,2BAA2B,CACnD,SAAQ,2BAA2B,CAAC,aAAa,EAAE,eAAe,CAAC;gBAEjE,UAAU,EAAE,UAAU,EACtB,IAAI,EAAE,YAAY,CAAC,eAAe,CAAC,EACnC,MAAM,CAAC,EAAE,cAAc;IAKzB;;;OAGG;IACH,aAAa,gBAAuB,MAAM,mBAExC;CACH"}
|
|
@@ -1,4 +1,14 @@
|
|
|
1
1
|
import { AbstractErgoExtractorAction } from '../AbstractErgoExtractorAction';
|
|
2
2
|
export class AbstractInitializableErgoExtractorAction extends AbstractErgoExtractorAction {
|
|
3
|
+
constructor(dataSource, repo, logger) {
|
|
4
|
+
super(dataSource, repo, logger);
|
|
5
|
+
}
|
|
6
|
+
/**
|
|
7
|
+
* remove all existing data for the extractor
|
|
8
|
+
* @param extractorId
|
|
9
|
+
*/
|
|
10
|
+
removeAllData = async (extractorId) => {
|
|
11
|
+
await this.repository.delete({ extractor: extractorId });
|
|
12
|
+
};
|
|
3
13
|
}
|
|
4
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
14
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiQWJzdHJhY3RJbml0aWFsaXphYmxlQWN0aW9uLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vbGliL2VyZ28vaW5pdGlhbGl6YWJsZS9BYnN0cmFjdEluaXRpYWxpemFibGVBY3Rpb24udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBR0EsT0FBTyxFQUFFLDJCQUEyQixFQUFFLE1BQU0sZ0NBQWdDLENBQUM7QUFJN0UsTUFBTSxPQUFnQix3Q0FHcEIsU0FBUSwyQkFBMkQ7SUFDbkUsWUFDRSxVQUFzQixFQUN0QixJQUFtQyxFQUNuQyxNQUF1QjtRQUV2QixLQUFLLENBQUMsVUFBVSxFQUFFLElBQUksRUFBRSxNQUFNLENBQUMsQ0FBQztJQUNsQyxDQUFDO0lBRUQ7OztPQUdHO0lBQ0gsYUFBYSxHQUFHLEtBQUssRUFBRSxXQUFtQixFQUFFLEVBQUU7UUFDNUMsTUFBTSxJQUFJLENBQUMsVUFBVSxDQUFDLE1BQU0sQ0FBQyxFQUFFLFNBQVMsRUFBRSxXQUFXLEVBQVMsQ0FBQyxDQUFDO0lBQ2xFLENBQUMsQ0FBQztDQUNIIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgRGF0YVNvdXJjZSwgRW50aXR5VGFyZ2V0IH0gZnJvbSAndHlwZW9ybSc7XG5pbXBvcnQgeyBBYnN0cmFjdExvZ2dlciB9IGZyb20gJ0Byb3Nlbi1icmlkZ2UvYWJzdHJhY3QtbG9nZ2VyJztcblxuaW1wb3J0IHsgQWJzdHJhY3RFcmdvRXh0cmFjdG9yQWN0aW9uIH0gZnJvbSAnLi4vQWJzdHJhY3RFcmdvRXh0cmFjdG9yQWN0aW9uJztcbmltcG9ydCB7IEFic3RyYWN0Qm94RGF0YSB9IGZyb20gJy4uL2ludGVyZmFjZXMnO1xuaW1wb3J0IHsgQWJzdHJhY3RFcmdvRXh0cmFjdG9yRW50aXR5IH0gZnJvbSAnLi4vQWJzdHJhY3RFcmdvRXh0cmFjdG9yRW50aXR5JztcblxuZXhwb3J0IGFic3RyYWN0IGNsYXNzIEFic3RyYWN0SW5pdGlhbGl6YWJsZUVyZ29FeHRyYWN0b3JBY3Rpb248XG4gIEV4dHJhY3RlZERhdGEgZXh0ZW5kcyBBYnN0cmFjdEJveERhdGEsXG4gIEV4dHJhY3RvckVudGl0eSBleHRlbmRzIEFic3RyYWN0RXJnb0V4dHJhY3RvckVudGl0eVxuPiBleHRlbmRzIEFic3RyYWN0RXJnb0V4dHJhY3RvckFjdGlvbjxFeHRyYWN0ZWREYXRhLCBFeHRyYWN0b3JFbnRpdHk+IHtcbiAgY29uc3RydWN0b3IoXG4gICAgZGF0YVNvdXJjZTogRGF0YVNvdXJjZSxcbiAgICByZXBvOiBFbnRpdHlUYXJnZXQ8RXh0cmFjdG9yRW50aXR5PixcbiAgICBsb2dnZXI/OiBBYnN0cmFjdExvZ2dlclxuICApIHtcbiAgICBzdXBlcihkYXRhU291cmNlLCByZXBvLCBsb2dnZXIpO1xuICB9XG5cbiAgLyoqXG4gICAqIHJlbW92ZSBhbGwgZXhpc3RpbmcgZGF0YSBmb3IgdGhlIGV4dHJhY3RvclxuICAgKiBAcGFyYW0gZXh0cmFjdG9ySWRcbiAgICovXG4gIHJlbW92ZUFsbERhdGEgPSBhc3luYyAoZXh0cmFjdG9ySWQ6IHN0cmluZykgPT4ge1xuICAgIGF3YWl0IHRoaXMucmVwb3NpdG9yeS5kZWxldGUoeyBleHRyYWN0b3I6IGV4dHJhY3RvcklkIH0gYXMgYW55KTtcbiAgfTtcbn1cbiJdfQ==
|
|
@@ -53,14 +53,28 @@ export interface SpendInfo {
|
|
|
53
53
|
boxId: string;
|
|
54
54
|
txId: string;
|
|
55
55
|
index: number;
|
|
56
|
+
extras?: string[];
|
|
56
57
|
}
|
|
57
|
-
export interface
|
|
58
|
+
export interface AbstractBoxData {
|
|
59
|
+
boxId: string;
|
|
60
|
+
serialized: string;
|
|
61
|
+
}
|
|
62
|
+
export declare enum CallbackType {
|
|
63
|
+
Insert = "insert",
|
|
64
|
+
Update = "update",
|
|
65
|
+
Spend = "spend",
|
|
66
|
+
Delete = "delete"
|
|
67
|
+
}
|
|
68
|
+
export interface BoxInfo {
|
|
58
69
|
boxId: string;
|
|
59
|
-
height: number;
|
|
60
|
-
blockId: string;
|
|
61
|
-
spendBlock?: string;
|
|
62
|
-
spendHeight?: number;
|
|
63
|
-
spendTxId?: string;
|
|
64
|
-
spendIndex?: number;
|
|
65
70
|
}
|
|
71
|
+
export type CallbackDataMap<ExtractedData extends AbstractBoxData> = {
|
|
72
|
+
[CallbackType.Update]: BoxInfo[];
|
|
73
|
+
[CallbackType.Insert]: ExtractedData[];
|
|
74
|
+
[CallbackType.Delete]: ExtractedData[];
|
|
75
|
+
[CallbackType.Spend]: BoxInfo[];
|
|
76
|
+
};
|
|
77
|
+
export type CallbackMap<ExtractedData extends AbstractBoxData> = {
|
|
78
|
+
[K in CallbackType]: (data: CallbackDataMap<ExtractedData>[K]) => void;
|
|
79
|
+
};
|
|
66
80
|
//# sourceMappingURL=interfaces.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"interfaces.d.ts","sourceRoot":"","sources":["../../lib/ergo/interfaces.ts"],"names":[],"mappings":"AAAA,oBAAY,eAAe;IACzB,QAAQ,aAAa;IACrB,IAAI,SAAS;CACd;AAED,MAAM,MAAM,QAAQ,GAAG;IACrB,KAAK,EAAE,MAAM,CAAC;CACf,CAAC;AAEF,MAAM,MAAM,SAAS,GAAG;IACtB,KAAK,EAAE,MAAM,CAAC;CACf,CAAC;AAEF,MAAM,MAAM,KAAK,GAAG;IAClB,OAAO,EAAE,MAAM,CAAC;IAChB,MAAM,EAAE,MAAM,CAAC;CAChB,CAAC;AAEF,MAAM,MAAM,mBAAmB,GAAG;IAChC,EAAE,CAAC,EAAE,MAAM,CAAC;IACZ,EAAE,CAAC,EAAE,MAAM,CAAC;IACZ,EAAE,CAAC,EAAE,MAAM,CAAC;IACZ,EAAE,CAAC,EAAE,MAAM,CAAC;IACZ,EAAE,CAAC,EAAE,MAAM,CAAC;IACZ,EAAE,CAAC,EAAE,MAAM,CAAC;CACb,CAAC;AAEF,MAAM,MAAM,SAAS,GAAG;IACtB,KAAK,EAAE,MAAM,CAAC;IACd,KAAK,EAAE,MAAM,CAAC;IACd,QAAQ,EAAE,MAAM,CAAC;IACjB,cAAc,EAAE,MAAM,CAAC;IACvB,MAAM,CAAC,EAAE,KAAK,CAAC,KAAK,CAAC,CAAC;IACtB,mBAAmB,CAAC,EAAE,mBAAmB,CAAC;IAC1C,aAAa,EAAE,MAAM,CAAC;IACtB,KAAK,EAAE,MAAM,GAAG,MAAM,CAAC;CACxB,CAAC;AAEF,MAAM,WAAW,OAAQ,SAAQ,SAAS;IACxC,OAAO,EAAE,MAAM,CAAC;IAChB,eAAe,EAAE,MAAM,CAAC;IACxB,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,kBAAkB,CAAC,EAAE,MAAM,CAAC;IAC5B,UAAU,CAAC,EAAE,MAAM,CAAC;CACrB;AAED,MAAM,MAAM,WAAW,GAAG;IACxB,EAAE,EAAE,MAAM,CAAC;IACX,MAAM,EAAE,KAAK,CAAC,QAAQ,CAAC,CAAC;IACxB,UAAU,EAAE,KAAK,CAAC,SAAS,CAAC,CAAC;IAC7B,OAAO,EAAE,KAAK,CAAC,SAAS,CAAC,CAAC;IAC1B,IAAI,CAAC,EAAE,MAAM,CAAC;CACf,CAAC;AAEF,MAAM,WAAW,mBAAoB,SAAQ,WAAW;IACtD,eAAe,EAAE,MAAM,CAAC;IACxB,OAAO,EAAE,MAAM,CAAC;CACjB;AAED,MAAM,WAAW,SAAS;IACxB,KAAK,EAAE,MAAM,CAAC;IACd,IAAI,EAAE,MAAM,CAAC;IACb,KAAK,EAAE,MAAM,CAAC;
|
|
1
|
+
{"version":3,"file":"interfaces.d.ts","sourceRoot":"","sources":["../../lib/ergo/interfaces.ts"],"names":[],"mappings":"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"}
|
package/dist/ergo/interfaces.js
CHANGED
|
@@ -3,4 +3,11 @@ export var ErgoNetworkType;
|
|
|
3
3
|
ErgoNetworkType["Explorer"] = "explorer";
|
|
4
4
|
ErgoNetworkType["Node"] = "node";
|
|
5
5
|
})(ErgoNetworkType || (ErgoNetworkType = {}));
|
|
6
|
-
|
|
6
|
+
export var CallbackType;
|
|
7
|
+
(function (CallbackType) {
|
|
8
|
+
CallbackType["Insert"] = "insert";
|
|
9
|
+
CallbackType["Update"] = "update";
|
|
10
|
+
CallbackType["Spend"] = "spend";
|
|
11
|
+
CallbackType["Delete"] = "delete";
|
|
12
|
+
})(CallbackType || (CallbackType = {}));
|
|
13
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW50ZXJmYWNlcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL2xpYi9lcmdvL2ludGVyZmFjZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsTUFBTSxDQUFOLElBQVksZUFHWDtBQUhELFdBQVksZUFBZTtJQUN6Qix3Q0FBcUIsQ0FBQTtJQUNyQixnQ0FBYSxDQUFBO0FBQ2YsQ0FBQyxFQUhXLGVBQWUsS0FBZixlQUFlLFFBRzFCO0FBcUVELE1BQU0sQ0FBTixJQUFZLFlBS1g7QUFMRCxXQUFZLFlBQVk7SUFDdEIsaUNBQWlCLENBQUE7SUFDakIsaUNBQWlCLENBQUE7SUFDakIsK0JBQWUsQ0FBQTtJQUNmLGlDQUFpQixDQUFBO0FBQ25CLENBQUMsRUFMVyxZQUFZLEtBQVosWUFBWSxRQUt2QiIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBlbnVtIEVyZ29OZXR3b3JrVHlwZSB7XG4gIEV4cGxvcmVyID0gJ2V4cGxvcmVyJyxcbiAgTm9kZSA9ICdub2RlJyxcbn1cblxuZXhwb3J0IHR5cGUgSW5wdXRCb3ggPSB7XG4gIGJveElkOiBzdHJpbmc7XG59O1xuXG5leHBvcnQgdHlwZSBEYXRhSW5wdXQgPSB7XG4gIGJveElkOiBzdHJpbmc7XG59O1xuXG5leHBvcnQgdHlwZSBBc3NldCA9IHtcbiAgdG9rZW5JZDogc3RyaW5nO1xuICBhbW91bnQ6IGJpZ2ludDtcbn07XG5cbmV4cG9ydCB0eXBlIEFkZGl0aW9uYWxSZWdpc3RlcnMgPSB7XG4gIFI0Pzogc3RyaW5nO1xuICBSNT86IHN0cmluZztcbiAgUjY/OiBzdHJpbmc7XG4gIFI3Pzogc3RyaW5nO1xuICBSOD86IHN0cmluZztcbiAgUjk/OiBzdHJpbmc7XG59O1xuXG5leHBvcnQgdHlwZSBPdXRwdXRCb3ggPSB7XG4gIGJveElkOiBzdHJpbmc7XG4gIHZhbHVlOiBiaWdpbnQ7XG4gIGVyZ29UcmVlOiBzdHJpbmc7XG4gIGNyZWF0aW9uSGVpZ2h0OiBudW1iZXI7XG4gIGFzc2V0cz86IEFycmF5PEFzc2V0PjtcbiAgYWRkaXRpb25hbFJlZ2lzdGVycz86IEFkZGl0aW9uYWxSZWdpc3RlcnM7XG4gIHRyYW5zYWN0aW9uSWQ6IHN0cmluZztcbiAgaW5kZXg6IGJpZ2ludCB8IG51bWJlcjtcbn07XG5cbmV4cG9ydCBpbnRlcmZhY2UgRXJnb0JveCBleHRlbmRzIE91dHB1dEJveCB7XG4gIGJsb2NrSWQ6IHN0cmluZztcbiAgaW5jbHVzaW9uSGVpZ2h0OiBudW1iZXI7XG4gIHNwZW50QmxvY2tJZD86IHN0cmluZztcbiAgc3BlbnRIZWlnaHQ/OiBudW1iZXI7XG4gIHNwZW50VHJhbnNhY3Rpb25JZD86IHN0cmluZztcbiAgc3BlbnRJbmRleD86IG51bWJlcjtcbn1cblxuZXhwb3J0IHR5cGUgVHJhbnNhY3Rpb24gPSB7XG4gIGlkOiBzdHJpbmc7XG4gIGlucHV0czogQXJyYXk8SW5wdXRCb3g+O1xuICBkYXRhSW5wdXRzOiBBcnJheTxEYXRhSW5wdXQ+O1xuICBvdXRwdXRzOiBBcnJheTxPdXRwdXRCb3g+O1xuICBzaXplPzogYmlnaW50O1xufTtcblxuZXhwb3J0IGludGVyZmFjZSBFeHRlbmRlZFRyYW5zYWN0aW9uIGV4dGVuZHMgVHJhbnNhY3Rpb24ge1xuICBpbmNsdXNpb25IZWlnaHQ6IG51bWJlcjtcbiAgYmxvY2tJZDogc3RyaW5nO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIFNwZW5kSW5mbyB7XG4gIGJveElkOiBzdHJpbmc7XG4gIHR4SWQ6IHN0cmluZztcbiAgaW5kZXg6IG51bWJlcjtcbiAgZXh0cmFzPzogc3RyaW5nW107XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQWJzdHJhY3RCb3hEYXRhIHtcbiAgYm94SWQ6IHN0cmluZztcbiAgc2VyaWFsaXplZDogc3RyaW5nO1xufVxuXG5leHBvcnQgZW51bSBDYWxsYmFja1R5cGUge1xuICBJbnNlcnQgPSAnaW5zZXJ0JyxcbiAgVXBkYXRlID0gJ3VwZGF0ZScsXG4gIFNwZW5kID0gJ3NwZW5kJyxcbiAgRGVsZXRlID0gJ2RlbGV0ZScsXG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQm94SW5mbyB7XG4gIGJveElkOiBzdHJpbmc7XG59XG5cbmV4cG9ydCB0eXBlIENhbGxiYWNrRGF0YU1hcDxFeHRyYWN0ZWREYXRhIGV4dGVuZHMgQWJzdHJhY3RCb3hEYXRhPiA9IHtcbiAgW0NhbGxiYWNrVHlwZS5VcGRhdGVdOiBCb3hJbmZvW107XG4gIFtDYWxsYmFja1R5cGUuSW5zZXJ0XTogRXh0cmFjdGVkRGF0YVtdO1xuICBbQ2FsbGJhY2tUeXBlLkRlbGV0ZV06IEV4dHJhY3RlZERhdGFbXTtcbiAgW0NhbGxiYWNrVHlwZS5TcGVuZF06IEJveEluZm9bXTtcbn07XG5cbmV4cG9ydCB0eXBlIENhbGxiYWNrTWFwPEV4dHJhY3RlZERhdGEgZXh0ZW5kcyBBYnN0cmFjdEJveERhdGE+ID0ge1xuICBbSyBpbiBDYWxsYmFja1R5cGVdOiAoZGF0YTogQ2FsbGJhY2tEYXRhTWFwPEV4dHJhY3RlZERhdGE+W0tdKSA9PiB2b2lkO1xufTtcbiJdfQ==
|
|
@@ -1,6 +1,7 @@
|
|
|
1
|
-
import { DataSource } from 'typeorm';
|
|
2
1
|
import { AbstractLogger, DummyLogger } from '@rosen-bridge/abstract-logger';
|
|
3
2
|
import JsonBigInt from '@rosen-bridge/json-bigint';
|
|
3
|
+
import { Mutex } from 'await-semaphore';
|
|
4
|
+
import { v4 as uuidv4 } from 'uuid';
|
|
4
5
|
|
|
5
6
|
import { AbstractExtractor } from '../AbstractExtractor';
|
|
6
7
|
import { AbstractErgoExtractorAction } from './AbstractErgoExtractorAction';
|
|
@@ -8,34 +9,99 @@ import { BlockInfo } from '../interfaces';
|
|
|
8
9
|
import {
|
|
9
10
|
Transaction,
|
|
10
11
|
OutputBox,
|
|
11
|
-
|
|
12
|
+
AbstractBoxData,
|
|
12
13
|
SpendInfo,
|
|
14
|
+
CallbackType,
|
|
15
|
+
CallbackMap,
|
|
16
|
+
CallbackDataMap,
|
|
13
17
|
} from './interfaces';
|
|
18
|
+
import { AbstractErgoExtractorEntity } from './AbstractErgoExtractorEntity';
|
|
14
19
|
|
|
15
20
|
export abstract class AbstractErgoExtractor<
|
|
16
|
-
ExtractedData extends
|
|
21
|
+
ExtractedData extends AbstractBoxData,
|
|
22
|
+
ExtractorEntity extends AbstractErgoExtractorEntity
|
|
17
23
|
> extends AbstractExtractor<Transaction> {
|
|
18
|
-
protected
|
|
19
|
-
|
|
24
|
+
protected abstract actions: AbstractErgoExtractorAction<
|
|
25
|
+
ExtractedData,
|
|
26
|
+
ExtractorEntity
|
|
27
|
+
>;
|
|
20
28
|
protected logger: AbstractLogger;
|
|
29
|
+
protected callbacks: {
|
|
30
|
+
[K in CallbackType]: Map<string, CallbackMap<ExtractedData>[K]>;
|
|
31
|
+
} = {
|
|
32
|
+
[CallbackType.Update]: new Map(),
|
|
33
|
+
[CallbackType.Insert]: new Map(),
|
|
34
|
+
[CallbackType.Delete]: new Map(),
|
|
35
|
+
[CallbackType.Spend]: new Map(),
|
|
36
|
+
};
|
|
37
|
+
private callbackMutex = new Mutex();
|
|
21
38
|
|
|
22
39
|
constructor(logger = new DummyLogger()) {
|
|
23
40
|
super();
|
|
24
41
|
this.logger = logger;
|
|
25
42
|
}
|
|
26
43
|
|
|
44
|
+
/**
|
|
45
|
+
* hook a new callback on a callback type
|
|
46
|
+
* @param type
|
|
47
|
+
* @param id
|
|
48
|
+
* @param callback
|
|
49
|
+
* @returns callback registered id
|
|
50
|
+
*/
|
|
51
|
+
hook = async <T extends CallbackType>(
|
|
52
|
+
type: T,
|
|
53
|
+
callback: CallbackMap<ExtractedData>[T]
|
|
54
|
+
): Promise<string> => {
|
|
55
|
+
const release = await this.callbackMutex.acquire();
|
|
56
|
+
const callbackMap = this.callbacks[type];
|
|
57
|
+
const id = uuidv4();
|
|
58
|
+
callbackMap.set(id, callback);
|
|
59
|
+
release();
|
|
60
|
+
return id;
|
|
61
|
+
};
|
|
62
|
+
|
|
63
|
+
/**
|
|
64
|
+
* unhook a callback on a type
|
|
65
|
+
* returns false if there is no registered callback with the id
|
|
66
|
+
* @param type
|
|
67
|
+
* @param id
|
|
68
|
+
* @returns success status
|
|
69
|
+
*/
|
|
70
|
+
unhook = async (type: CallbackType, id: string): Promise<boolean> => {
|
|
71
|
+
const release = await this.callbackMutex.acquire();
|
|
72
|
+
const callbackMap = this.callbacks[type];
|
|
73
|
+
if (!callbackMap.has(id)) {
|
|
74
|
+
this.logger.warn(
|
|
75
|
+
`Callback with Id [${id}] is not registered for type [${type}].`
|
|
76
|
+
);
|
|
77
|
+
return false;
|
|
78
|
+
}
|
|
79
|
+
callbackMap.delete(id);
|
|
80
|
+
release();
|
|
81
|
+
return true;
|
|
82
|
+
};
|
|
83
|
+
|
|
84
|
+
/**
|
|
85
|
+
* trigger all callbacks registered on a specific type with the provided data
|
|
86
|
+
* @param type
|
|
87
|
+
* @param data
|
|
88
|
+
*/
|
|
89
|
+
protected triggerCallbacks<T extends CallbackType>(
|
|
90
|
+
type: T,
|
|
91
|
+
data: CallbackDataMap<ExtractedData>[T]
|
|
92
|
+
): void {
|
|
93
|
+
const callbackMap = this.callbacks[type];
|
|
94
|
+
callbackMap.forEach((callback) => {
|
|
95
|
+
callback(data);
|
|
96
|
+
});
|
|
97
|
+
}
|
|
98
|
+
|
|
27
99
|
/**
|
|
28
100
|
* extract box data to proper format (not including spending information)
|
|
29
101
|
* @param box
|
|
30
|
-
* @param blockId box inclusion block hash
|
|
31
|
-
* @param height box inclusion block height
|
|
32
102
|
* @return extracted data in proper format
|
|
33
103
|
*/
|
|
34
|
-
abstract extractBoxData: (
|
|
35
|
-
box: OutputBox,
|
|
36
|
-
blockId: string,
|
|
37
|
-
height: number
|
|
38
|
-
) => Omit<ExtractedData, 'spendBlock' | 'spendHeight'> | undefined;
|
|
104
|
+
abstract extractBoxData: (box: OutputBox) => ExtractedData | undefined;
|
|
39
105
|
|
|
40
106
|
/**
|
|
41
107
|
* check proper data format in the box
|
|
@@ -63,18 +129,14 @@ export abstract class AbstractErgoExtractor<
|
|
|
63
129
|
continue;
|
|
64
130
|
}
|
|
65
131
|
this.logger.debug(`Trying to extract data from box ${output.boxId}`);
|
|
66
|
-
const extractedData = this.extractBoxData(
|
|
67
|
-
output,
|
|
68
|
-
block.hash,
|
|
69
|
-
block.height
|
|
70
|
-
);
|
|
132
|
+
const extractedData = this.extractBoxData(output);
|
|
71
133
|
if (extractedData) {
|
|
72
134
|
this.logger.debug(
|
|
73
135
|
`Extracted data ${JsonBigInt.stringify(extractedData)} from box ${
|
|
74
136
|
output.boxId
|
|
75
137
|
}`
|
|
76
138
|
);
|
|
77
|
-
boxes.push(extractedData
|
|
139
|
+
boxes.push(extractedData);
|
|
78
140
|
}
|
|
79
141
|
}
|
|
80
142
|
let boxIndex = 1;
|
|
@@ -84,15 +146,33 @@ export abstract class AbstractErgoExtractor<
|
|
|
84
146
|
}
|
|
85
147
|
}
|
|
86
148
|
|
|
87
|
-
if (boxes.length > 0)
|
|
88
|
-
|
|
149
|
+
if (boxes.length > 0) {
|
|
150
|
+
if (!(await this.actions.storeBoxes(boxes, block, this.getId()))) {
|
|
151
|
+
this.logger.warn(
|
|
152
|
+
`Data insertion failed for ${this.getId()} at the block ${
|
|
153
|
+
block.height
|
|
154
|
+
}`
|
|
155
|
+
);
|
|
156
|
+
return false;
|
|
157
|
+
}
|
|
158
|
+
this.triggerCallbacks(CallbackType.Insert, boxes);
|
|
159
|
+
}
|
|
160
|
+
const spentData = await this.actions.spendBoxes(
|
|
161
|
+
spentInfos,
|
|
162
|
+
block,
|
|
163
|
+
this.getId()
|
|
164
|
+
);
|
|
165
|
+
if (spentData.length > 0) {
|
|
166
|
+
this.triggerCallbacks(CallbackType.Spend, spentData);
|
|
167
|
+
}
|
|
89
168
|
} catch (e) {
|
|
90
169
|
this.logger.error(
|
|
91
|
-
`
|
|
170
|
+
`Processing transactions failed for ${this.getId()} at the block ${
|
|
171
|
+
block.height
|
|
172
|
+
} with error: ${e}`
|
|
92
173
|
);
|
|
93
174
|
return false;
|
|
94
175
|
}
|
|
95
|
-
|
|
96
176
|
return true;
|
|
97
177
|
};
|
|
98
178
|
|
|
@@ -101,6 +181,10 @@ export abstract class AbstractErgoExtractor<
|
|
|
101
181
|
* @param hash block hash
|
|
102
182
|
*/
|
|
103
183
|
forkBlock = async (hash: string): Promise<void> => {
|
|
104
|
-
await this.actions.deleteBlockBoxes(hash, this.getId());
|
|
184
|
+
const result = await this.actions.deleteBlockBoxes(hash, this.getId());
|
|
185
|
+
if (result.deletedData.length > 0)
|
|
186
|
+
this.triggerCallbacks(CallbackType.Delete, result.deletedData);
|
|
187
|
+
if (result.updatedData.length > 0)
|
|
188
|
+
this.triggerCallbacks(CallbackType.Update, result.updatedData);
|
|
105
189
|
};
|
|
106
190
|
}
|