@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
|
@@ -0,0 +1,127 @@
|
|
|
1
|
+
import { mapValues, pick } from 'lodash-es';
|
|
2
|
+
import ergoExplorerClientFactory from '@rosen-clients/ergo-explorer';
|
|
3
|
+
import { API_LIMIT } from '../../constants';
|
|
4
|
+
export class ExplorerNetwork {
|
|
5
|
+
api;
|
|
6
|
+
constructor(url) {
|
|
7
|
+
this.api = ergoExplorerClientFactory(url);
|
|
8
|
+
}
|
|
9
|
+
/**
|
|
10
|
+
* convert explorer api output boxes to OutputBox interface
|
|
11
|
+
* @param box
|
|
12
|
+
* @returns OutputBox
|
|
13
|
+
*/
|
|
14
|
+
convertOutputBox = (box) => {
|
|
15
|
+
return {
|
|
16
|
+
boxId: box.boxId,
|
|
17
|
+
creationHeight: box.creationHeight,
|
|
18
|
+
ergoTree: box.ergoTree,
|
|
19
|
+
index: box.index,
|
|
20
|
+
transactionId: box.transactionId,
|
|
21
|
+
value: box.value,
|
|
22
|
+
additionalRegisters: mapValues(box.additionalRegisters, 'serializedValue'),
|
|
23
|
+
assets: box.assets?.map((asset) => pick(asset, ['tokenId', 'amount'])) ?? [],
|
|
24
|
+
};
|
|
25
|
+
};
|
|
26
|
+
/**
|
|
27
|
+
* convert explorer api input boxes to OutputBox interface
|
|
28
|
+
* @param box
|
|
29
|
+
* @returns OutputBox
|
|
30
|
+
*/
|
|
31
|
+
convertInputBox = (box) => {
|
|
32
|
+
return {
|
|
33
|
+
boxId: box.boxId,
|
|
34
|
+
creationHeight: box.outputCreatedAt,
|
|
35
|
+
ergoTree: box.ergoTree,
|
|
36
|
+
index: box.outputIndex,
|
|
37
|
+
transactionId: box.outputTransactionId,
|
|
38
|
+
value: box.value,
|
|
39
|
+
additionalRegisters: mapValues(box.additionalRegisters, 'serializedValue'),
|
|
40
|
+
assets: box.assets?.map((asset) => pick(asset, ['tokenId', 'amount'])) ?? [],
|
|
41
|
+
};
|
|
42
|
+
};
|
|
43
|
+
/**
|
|
44
|
+
* convert explorer transaction to extractor transaction type
|
|
45
|
+
* @param tx
|
|
46
|
+
*/
|
|
47
|
+
convertTransaction = (tx) => {
|
|
48
|
+
return {
|
|
49
|
+
id: tx.id,
|
|
50
|
+
inclusionHeight: tx.inclusionHeight,
|
|
51
|
+
blockId: tx.blockId,
|
|
52
|
+
dataInputs: tx.dataInputs?.map((dataInput) => ({
|
|
53
|
+
boxId: dataInput.boxId,
|
|
54
|
+
})) ?? [],
|
|
55
|
+
// TODO: Add input extension to explorer local/ergo/rosen-bridge/scanner/-/issues/156
|
|
56
|
+
inputs: tx.inputs?.map((input) => this.convertInputBox(input)) ?? [],
|
|
57
|
+
outputs: tx.outputs?.map((output) => this.convertOutputBox(output)) ?? [],
|
|
58
|
+
};
|
|
59
|
+
};
|
|
60
|
+
/**
|
|
61
|
+
* convert explorer block transaction to transaction type
|
|
62
|
+
* @param tx
|
|
63
|
+
*/
|
|
64
|
+
convertBlockTransaction = (tx) => {
|
|
65
|
+
return {
|
|
66
|
+
id: tx.id,
|
|
67
|
+
dataInputs: tx.dataInputs?.map((dataInput) => ({
|
|
68
|
+
boxId: dataInput.id,
|
|
69
|
+
})) ?? [],
|
|
70
|
+
// TODO: Add input extension local/ergo/rosen-bridge/scanner/-/issues/156
|
|
71
|
+
inputs: tx.inputs?.map((input) => ({ boxId: input.id })) ?? [],
|
|
72
|
+
outputs: tx.outputs?.map((output) => ({
|
|
73
|
+
boxId: output.id,
|
|
74
|
+
transactionId: output.txId,
|
|
75
|
+
additionalRegisters: output.additionalRegisters,
|
|
76
|
+
assets: output.assets?.map((asset) => pick(asset, ['tokenId', 'amount'])) ??
|
|
77
|
+
[],
|
|
78
|
+
ergoTree: output.ergoTree,
|
|
79
|
+
creationHeight: output.creationHeight,
|
|
80
|
+
index: output.index,
|
|
81
|
+
value: output.value,
|
|
82
|
+
})) ?? [],
|
|
83
|
+
};
|
|
84
|
+
};
|
|
85
|
+
/**
|
|
86
|
+
* use explorer api to return related transactions of the specified address in the height range
|
|
87
|
+
* @param tokenId
|
|
88
|
+
* @param offset
|
|
89
|
+
* @param limit
|
|
90
|
+
* @returns related transactions
|
|
91
|
+
*/
|
|
92
|
+
getAddressTransactionsWithHeight = async (address, fromHeight, toHeight, limit = API_LIMIT) => {
|
|
93
|
+
const txs = await this.api.v1.getApiV1AddressesP1Transactions(address, {
|
|
94
|
+
fromHeight,
|
|
95
|
+
toHeight,
|
|
96
|
+
limit: limit,
|
|
97
|
+
});
|
|
98
|
+
if (!txs.items)
|
|
99
|
+
throw new Error('Explorer AddressTransactions api expected to have items');
|
|
100
|
+
return {
|
|
101
|
+
items: txs.items.map((tx) => this.convertTransaction(tx)),
|
|
102
|
+
total: txs.total,
|
|
103
|
+
};
|
|
104
|
+
};
|
|
105
|
+
/**
|
|
106
|
+
* use explorer api to get the block id at the specified height
|
|
107
|
+
* @param height
|
|
108
|
+
* @returns block id
|
|
109
|
+
*/
|
|
110
|
+
getBlockIdAtHeight = async (height) => {
|
|
111
|
+
const id = await this.api.v0.getApiV0BlocksAtP1(height);
|
|
112
|
+
return id[0];
|
|
113
|
+
};
|
|
114
|
+
/**
|
|
115
|
+
* use explorer api to return all transactions in a block
|
|
116
|
+
* @param blockId
|
|
117
|
+
* @returns converted transactions
|
|
118
|
+
*/
|
|
119
|
+
getBlockTxs = async (blockId) => {
|
|
120
|
+
const block = await this.api.v1.getApiV1BlocksP1(blockId);
|
|
121
|
+
if (!block.block.blockTransactions) {
|
|
122
|
+
throw new Error(`Expected explorer block api to include block transactions for block ${blockId}`);
|
|
123
|
+
}
|
|
124
|
+
return block.block.blockTransactions.map((tx) => this.convertBlockTransaction(tx));
|
|
125
|
+
};
|
|
126
|
+
}
|
|
127
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"explorerNetwork.js","sourceRoot":"","sources":["../../../lib/ergo/networks/explorerNetwork.ts"],"names":[],"mappings":"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","sourcesContent":["import { mapValues, pick } from 'lodash-es';\n\nimport { OutputBox, Transaction } from '@rosen-bridge/scanner-interfaces';\nimport ergoExplorerClientFactory from '@rosen-clients/ergo-explorer';\nimport { V1 } from '@rosen-clients/ergo-explorer';\n\nimport { API_LIMIT } from '../../constants';\nimport { ExtendedTransaction } from '../interfaces';\n\nexport class ExplorerNetwork {\n  private api;\n\n  constructor(url: string) {\n    this.api = ergoExplorerClientFactory(url);\n  }\n\n  /**\n   * convert explorer api output boxes to OutputBox interface\n   * @param box\n   * @returns OutputBox\n   */\n  private convertOutputBox = (box: V1.OutputInfo): OutputBox => {\n    return {\n      boxId: box.boxId,\n      creationHeight: box.creationHeight,\n      ergoTree: box.ergoTree,\n      index: box.index,\n      transactionId: box.transactionId,\n      value: box.value,\n      additionalRegisters: mapValues(\n        box.additionalRegisters,\n        'serializedValue',\n      ),\n      assets:\n        box.assets?.map((asset) => pick(asset, ['tokenId', 'amount'])) ?? [],\n    };\n  };\n\n  /**\n   * convert explorer api input boxes to OutputBox interface\n   * @param box\n   * @returns OutputBox\n   */\n  private convertInputBox = (box: V1.InputInfo): OutputBox => {\n    return {\n      boxId: box.boxId,\n      creationHeight: box.outputCreatedAt,\n      ergoTree: box.ergoTree,\n      index: box.outputIndex,\n      transactionId: box.outputTransactionId,\n      value: box.value,\n      additionalRegisters: mapValues(\n        box.additionalRegisters,\n        'serializedValue',\n      ),\n      assets:\n        box.assets?.map((asset) => pick(asset, ['tokenId', 'amount'])) ?? [],\n    };\n  };\n\n  /**\n   * convert explorer transaction to extractor transaction type\n   * @param tx\n   */\n  private convertTransaction = (\n    tx: V1.TransactionInfo,\n  ): ExtendedTransaction => {\n    return {\n      id: tx.id,\n      inclusionHeight: tx.inclusionHeight,\n      blockId: tx.blockId,\n      dataInputs:\n        tx.dataInputs?.map((dataInput) => ({\n          boxId: dataInput.boxId,\n        })) ?? [],\n      // TODO: Add input extension to explorer local/ergo/rosen-bridge/scanner/-/issues/156\n      inputs: tx.inputs?.map((input) => this.convertInputBox(input)) ?? [],\n      outputs: tx.outputs?.map((output) => this.convertOutputBox(output)) ?? [],\n    };\n  };\n\n  /**\n   * convert explorer block transaction to transaction type\n   * @param tx\n   */\n  private convertBlockTransaction = (tx: V1.TransactionInfo1): Transaction => {\n    return {\n      id: tx.id,\n      dataInputs:\n        tx.dataInputs?.map((dataInput) => ({\n          boxId: dataInput.id,\n        })) ?? [],\n      // TODO: Add input extension local/ergo/rosen-bridge/scanner/-/issues/156\n      inputs: tx.inputs?.map((input) => ({ boxId: input.id })) ?? [],\n      outputs:\n        tx.outputs?.map((output) => ({\n          boxId: output.id,\n          transactionId: output.txId,\n          additionalRegisters: output.additionalRegisters,\n          assets:\n            output.assets?.map((asset) => pick(asset, ['tokenId', 'amount'])) ??\n            [],\n          ergoTree: output.ergoTree,\n          creationHeight: output.creationHeight,\n          index: output.index,\n          value: output.value,\n        })) ?? [],\n    };\n  };\n\n  /**\n   * use explorer api to return related transactions of the specified address in the height range\n   * @param tokenId\n   * @param offset\n   * @param limit\n   * @returns related transactions\n   */\n  getAddressTransactionsWithHeight = async (\n    address: string,\n    fromHeight: number,\n    toHeight: number,\n    limit = API_LIMIT,\n  ): Promise<{ items: Array<ExtendedTransaction>; total: number }> => {\n    const txs = await this.api.v1.getApiV1AddressesP1Transactions(address, {\n      fromHeight,\n      toHeight,\n      limit: limit,\n    });\n    if (!txs.items)\n      throw new Error(\n        'Explorer AddressTransactions api expected to have items',\n      );\n    return {\n      items: txs.items.map((tx) => this.convertTransaction(tx)),\n      total: txs.total,\n    };\n  };\n\n  /**\n   * use explorer api to get the block id at the specified height\n   * @param height\n   * @returns block id\n   */\n  getBlockIdAtHeight = async (height: number): Promise<string> => {\n    const id = await this.api.v0.getApiV0BlocksAtP1(height);\n    return id[0];\n  };\n\n  /**\n   * use explorer api to return all transactions in a block\n   * @param blockId\n   * @returns converted transactions\n   */\n  getBlockTxs = async (blockId: string): Promise<Array<Transaction>> => {\n    const block = await this.api.v1.getApiV1BlocksP1(blockId);\n    if (!block.block.blockTransactions) {\n      throw new Error(\n        `Expected explorer block api to include block transactions for block ${blockId}`,\n      );\n    }\n    return block.block.blockTransactions.map((tx) =>\n      this.convertBlockTransaction(tx),\n    );\n  };\n}\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../lib/ergo/networks/index.ts"],"names":[],"mappings":"AAAA,cAAc,mBAAmB,CAAC;AAClC,cAAc,eAAe,CAAC"}
|
|
@@ -0,0 +1,3 @@
|
|
|
1
|
+
export * from './explorerNetwork';
|
|
2
|
+
export * from './nodeNetwork';
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9saWIvZXJnby9uZXR3b3Jrcy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLG1CQUFtQixDQUFDO0FBQ2xDLGNBQWMsZUFBZSxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9leHBsb3Jlck5ldHdvcmsnO1xuZXhwb3J0ICogZnJvbSAnLi9ub2RlTmV0d29yayc7XG4iXX0=
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
import { ExtendedTransaction } from '../interfaces';
|
|
2
|
+
export declare class NodeNetwork {
|
|
3
|
+
private api;
|
|
4
|
+
constructor(url: string);
|
|
5
|
+
/**
|
|
6
|
+
* convert node api boxes to OutputBox interface
|
|
7
|
+
* @param box
|
|
8
|
+
* @returns ErgoBox
|
|
9
|
+
*/
|
|
10
|
+
private convertBox;
|
|
11
|
+
/**
|
|
12
|
+
* convert Node transaction to extractor transaction type
|
|
13
|
+
* @param tx
|
|
14
|
+
*/
|
|
15
|
+
private convertTransaction;
|
|
16
|
+
/**
|
|
17
|
+
* use node api to return related transactions of the specified address with limit offset
|
|
18
|
+
* @param address
|
|
19
|
+
* @param offset
|
|
20
|
+
* @param limit
|
|
21
|
+
* @returns related transactions
|
|
22
|
+
*/
|
|
23
|
+
getAddressTransactionsWithOffsetLimit: (address: string, offset: number, limit: number) => Promise<{
|
|
24
|
+
items: Array<ExtendedTransaction>;
|
|
25
|
+
total: number;
|
|
26
|
+
}>;
|
|
27
|
+
}
|
|
28
|
+
//# sourceMappingURL=nodeNetwork.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"nodeNetwork.d.ts","sourceRoot":"","sources":["../../../lib/ergo/networks/nodeNetwork.ts"],"names":[],"mappings":"AAMA,OAAO,EAAE,mBAAmB,EAAE,MAAM,eAAe,CAAC;AAEpD,qBAAa,WAAW;IACtB,OAAO,CAAC,GAAG,CAAC;gBAEA,GAAG,EAAE,MAAM;IAIvB;;;;OAIG;IACH,OAAO,CAAC,UAAU,CAWhB;IAEF;;;OAGG;IACH,OAAO,CAAC,kBAAkB,CAYxB;IAEF;;;;;;OAMG;IACH,qCAAqC,GACnC,SAAS,MAAM,EACf,QAAQ,MAAM,EACd,OAAO,MAAM,KACZ,OAAO,CAAC;QAAE,KAAK,EAAE,KAAK,CAAC,mBAAmB,CAAC,CAAC;QAAC,KAAK,EAAE,MAAM,CAAA;KAAE,CAAC,CAa9D;CACH"}
|
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
import ergoNodeClientFactory from '@rosen-clients/ergo-node';
|
|
2
|
+
export class NodeNetwork {
|
|
3
|
+
api;
|
|
4
|
+
constructor(url) {
|
|
5
|
+
this.api = ergoNodeClientFactory(url);
|
|
6
|
+
}
|
|
7
|
+
/**
|
|
8
|
+
* convert node api boxes to OutputBox interface
|
|
9
|
+
* @param box
|
|
10
|
+
* @returns ErgoBox
|
|
11
|
+
*/
|
|
12
|
+
convertBox = (box) => {
|
|
13
|
+
return {
|
|
14
|
+
transactionId: box.transactionId || '',
|
|
15
|
+
index: box.index || 0,
|
|
16
|
+
value: box.value || 0n,
|
|
17
|
+
ergoTree: box.ergoTree || '',
|
|
18
|
+
creationHeight: box.creationHeight || 0,
|
|
19
|
+
assets: box.assets || [],
|
|
20
|
+
additionalRegisters: box.additionalRegisters,
|
|
21
|
+
boxId: box.boxId || '',
|
|
22
|
+
};
|
|
23
|
+
};
|
|
24
|
+
/**
|
|
25
|
+
* convert Node transaction to extractor transaction type
|
|
26
|
+
* @param tx
|
|
27
|
+
*/
|
|
28
|
+
convertTransaction = (tx) => {
|
|
29
|
+
return {
|
|
30
|
+
id: tx.id || '',
|
|
31
|
+
inclusionHeight: tx.inclusionHeight,
|
|
32
|
+
blockId: tx.blockId,
|
|
33
|
+
outputs: tx.outputs.map((output) => this.convertBox(output)),
|
|
34
|
+
// TODO: Add input extension local/ergo/rosen-bridge/scanner/-/issues/156
|
|
35
|
+
inputs: tx.inputs.map((input) => this.convertBox(input)),
|
|
36
|
+
dataInputs: tx.dataInputs,
|
|
37
|
+
};
|
|
38
|
+
};
|
|
39
|
+
/**
|
|
40
|
+
* use node api to return related transactions of the specified address with limit offset
|
|
41
|
+
* @param address
|
|
42
|
+
* @param offset
|
|
43
|
+
* @param limit
|
|
44
|
+
* @returns related transactions
|
|
45
|
+
*/
|
|
46
|
+
getAddressTransactionsWithOffsetLimit = async (address, offset, limit) => {
|
|
47
|
+
const txs = await this.api.getTxsByAddress(address, {
|
|
48
|
+
offset,
|
|
49
|
+
limit,
|
|
50
|
+
});
|
|
51
|
+
if (!txs.items)
|
|
52
|
+
throw new Error('Explorer AddressTransactions api expected to have items');
|
|
53
|
+
return {
|
|
54
|
+
items: txs.items.map((tx) => this.convertTransaction(tx)),
|
|
55
|
+
total: txs.total,
|
|
56
|
+
};
|
|
57
|
+
};
|
|
58
|
+
}
|
|
59
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibm9kZU5ldHdvcmsuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9saWIvZXJnby9uZXR3b3Jrcy9ub2RlTmV0d29yay50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFDQSxPQUFPLHFCQUdOLE1BQU0sMEJBQTBCLENBQUM7QUFJbEMsTUFBTSxPQUFPLFdBQVc7SUFDZCxHQUFHLENBQUM7SUFFWixZQUFZLEdBQVc7UUFDckIsSUFBSSxDQUFDLEdBQUcsR0FBRyxxQkFBcUIsQ0FBQyxHQUFHLENBQUMsQ0FBQztJQUN4QyxDQUFDO0lBRUQ7Ozs7T0FJRztJQUNLLFVBQVUsR0FBRyxDQUFDLEdBQW1CLEVBQWEsRUFBRTtRQUN0RCxPQUFPO1lBQ0wsYUFBYSxFQUFFLEdBQUcsQ0FBQyxhQUFhLElBQUksRUFBRTtZQUN0QyxLQUFLLEVBQUUsR0FBRyxDQUFDLEtBQUssSUFBSSxDQUFDO1lBQ3JCLEtBQUssRUFBRSxHQUFHLENBQUMsS0FBSyxJQUFJLEVBQUU7WUFDdEIsUUFBUSxFQUFFLEdBQUcsQ0FBQyxRQUFRLElBQUksRUFBRTtZQUM1QixjQUFjLEVBQUUsR0FBRyxDQUFDLGNBQWMsSUFBSSxDQUFDO1lBQ3ZDLE1BQU0sRUFBRSxHQUFHLENBQUMsTUFBTSxJQUFJLEVBQUU7WUFDeEIsbUJBQW1CLEVBQUUsR0FBRyxDQUFDLG1CQUFtQjtZQUM1QyxLQUFLLEVBQUUsR0FBRyxDQUFDLEtBQUssSUFBSSxFQUFFO1NBQ3ZCLENBQUM7SUFDSixDQUFDLENBQUM7SUFFRjs7O09BR0c7SUFDSyxrQkFBa0IsR0FBRyxDQUMzQixFQUEwQixFQUNMLEVBQUU7UUFDdkIsT0FBTztZQUNMLEVBQUUsRUFBRSxFQUFFLENBQUMsRUFBRSxJQUFJLEVBQUU7WUFDZixlQUFlLEVBQUUsRUFBRSxDQUFDLGVBQWU7WUFDbkMsT0FBTyxFQUFFLEVBQUUsQ0FBQyxPQUFPO1lBQ25CLE9BQU8sRUFBRSxFQUFFLENBQUMsT0FBTyxDQUFDLEdBQUcsQ0FBQyxDQUFDLE1BQU0sRUFBRSxFQUFFLENBQUMsSUFBSSxDQUFDLFVBQVUsQ0FBQyxNQUFNLENBQUMsQ0FBQztZQUM1RCx5RUFBeUU7WUFDekUsTUFBTSxFQUFFLEVBQUUsQ0FBQyxNQUFNLENBQUMsR0FBRyxDQUFDLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxJQUFJLENBQUMsVUFBVSxDQUFDLEtBQUssQ0FBQyxDQUFDO1lBQ3hELFVBQVUsRUFBRSxFQUFFLENBQUMsVUFBVTtTQUMxQixDQUFDO0lBQ0osQ0FBQyxDQUFDO0lBRUY7Ozs7OztPQU1HO0lBQ0gscUNBQXFDLEdBQUcsS0FBSyxFQUMzQyxPQUFlLEVBQ2YsTUFBYyxFQUNkLEtBQWEsRUFDa0QsRUFBRTtRQUNqRSxNQUFNLEdBQUcsR0FBRyxNQUFNLElBQUksQ0FBQyxHQUFHLENBQUMsZUFBZSxDQUFDLE9BQU8sRUFBRTtZQUNsRCxNQUFNO1lBQ04sS0FBSztTQUNOLENBQUMsQ0FBQztRQUNILElBQUksQ0FBQyxHQUFHLENBQUMsS0FBSztZQUNaLE1BQU0sSUFBSSxLQUFLLENBQ2IseURBQXlELENBQzFELENBQUM7UUFDSixPQUFPO1lBQ0wsS0FBSyxFQUFFLEdBQUcsQ0FBQyxLQUFLLENBQUMsR0FBRyxDQUFDLENBQUMsRUFBRSxFQUFFLEVBQUUsQ0FBQyxJQUFJLENBQUMsa0JBQWtCLENBQUMsRUFBRSxDQUFDLENBQUM7WUFDekQsS0FBSyxFQUFFLEdBQUcsQ0FBQyxLQUFNO1NBQ2xCLENBQUM7SUFDSixDQUFDLENBQUM7Q0FDSCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IE91dHB1dEJveCB9IGZyb20gJ0Byb3Nlbi1icmlkZ2Uvc2Nhbm5lci1pbnRlcmZhY2VzJztcbmltcG9ydCBlcmdvTm9kZUNsaWVudEZhY3RvcnksIHtcbiAgSW5kZXhlZEVyZ29Cb3gsXG4gIEluZGV4ZWRFcmdvVHJhbnNhY3Rpb24sXG59IGZyb20gJ0Byb3Nlbi1jbGllbnRzL2VyZ28tbm9kZSc7XG5cbmltcG9ydCB7IEV4dGVuZGVkVHJhbnNhY3Rpb24gfSBmcm9tICcuLi9pbnRlcmZhY2VzJztcblxuZXhwb3J0IGNsYXNzIE5vZGVOZXR3b3JrIHtcbiAgcHJpdmF0ZSBhcGk7XG5cbiAgY29uc3RydWN0b3IodXJsOiBzdHJpbmcpIHtcbiAgICB0aGlzLmFwaSA9IGVyZ29Ob2RlQ2xpZW50RmFjdG9yeSh1cmwpO1xuICB9XG5cbiAgLyoqXG4gICAqIGNvbnZlcnQgbm9kZSBhcGkgYm94ZXMgdG8gT3V0cHV0Qm94IGludGVyZmFjZVxuICAgKiBAcGFyYW0gYm94XG4gICAqIEByZXR1cm5zIEVyZ29Cb3hcbiAgICovXG4gIHByaXZhdGUgY29udmVydEJveCA9IChib3g6IEluZGV4ZWRFcmdvQm94KTogT3V0cHV0Qm94ID0+IHtcbiAgICByZXR1cm4ge1xuICAgICAgdHJhbnNhY3Rpb25JZDogYm94LnRyYW5zYWN0aW9uSWQgfHwgJycsXG4gICAgICBpbmRleDogYm94LmluZGV4IHx8IDAsXG4gICAgICB2YWx1ZTogYm94LnZhbHVlIHx8IDBuLFxuICAgICAgZXJnb1RyZWU6IGJveC5lcmdvVHJlZSB8fCAnJyxcbiAgICAgIGNyZWF0aW9uSGVpZ2h0OiBib3guY3JlYXRpb25IZWlnaHQgfHwgMCxcbiAgICAgIGFzc2V0czogYm94LmFzc2V0cyB8fCBbXSxcbiAgICAgIGFkZGl0aW9uYWxSZWdpc3RlcnM6IGJveC5hZGRpdGlvbmFsUmVnaXN0ZXJzLFxuICAgICAgYm94SWQ6IGJveC5ib3hJZCB8fCAnJyxcbiAgICB9O1xuICB9O1xuXG4gIC8qKlxuICAgKiBjb252ZXJ0IE5vZGUgdHJhbnNhY3Rpb24gdG8gZXh0cmFjdG9yIHRyYW5zYWN0aW9uIHR5cGVcbiAgICogQHBhcmFtIHR4XG4gICAqL1xuICBwcml2YXRlIGNvbnZlcnRUcmFuc2FjdGlvbiA9IChcbiAgICB0eDogSW5kZXhlZEVyZ29UcmFuc2FjdGlvbixcbiAgKTogRXh0ZW5kZWRUcmFuc2FjdGlvbiA9PiB7XG4gICAgcmV0dXJuIHtcbiAgICAgIGlkOiB0eC5pZCB8fCAnJyxcbiAgICAgIGluY2x1c2lvbkhlaWdodDogdHguaW5jbHVzaW9uSGVpZ2h0LFxuICAgICAgYmxvY2tJZDogdHguYmxvY2tJZCxcbiAgICAgIG91dHB1dHM6IHR4Lm91dHB1dHMubWFwKChvdXRwdXQpID0+IHRoaXMuY29udmVydEJveChvdXRwdXQpKSxcbiAgICAgIC8vIFRPRE86IEFkZCBpbnB1dCBleHRlbnNpb24gbG9jYWwvZXJnby9yb3Nlbi1icmlkZ2Uvc2Nhbm5lci8tL2lzc3Vlcy8xNTZcbiAgICAgIGlucHV0czogdHguaW5wdXRzLm1hcCgoaW5wdXQpID0+IHRoaXMuY29udmVydEJveChpbnB1dCkpLFxuICAgICAgZGF0YUlucHV0czogdHguZGF0YUlucHV0cyxcbiAgICB9O1xuICB9O1xuXG4gIC8qKlxuICAgKiB1c2Ugbm9kZSBhcGkgdG8gcmV0dXJuIHJlbGF0ZWQgdHJhbnNhY3Rpb25zIG9mIHRoZSBzcGVjaWZpZWQgYWRkcmVzcyB3aXRoIGxpbWl0IG9mZnNldFxuICAgKiBAcGFyYW0gYWRkcmVzc1xuICAgKiBAcGFyYW0gb2Zmc2V0XG4gICAqIEBwYXJhbSBsaW1pdFxuICAgKiBAcmV0dXJucyByZWxhdGVkIHRyYW5zYWN0aW9uc1xuICAgKi9cbiAgZ2V0QWRkcmVzc1RyYW5zYWN0aW9uc1dpdGhPZmZzZXRMaW1pdCA9IGFzeW5jIChcbiAgICBhZGRyZXNzOiBzdHJpbmcsXG4gICAgb2Zmc2V0OiBudW1iZXIsXG4gICAgbGltaXQ6IG51bWJlcixcbiAgKTogUHJvbWlzZTx7IGl0ZW1zOiBBcnJheTxFeHRlbmRlZFRyYW5zYWN0aW9uPjsgdG90YWw6IG51bWJlciB9PiA9PiB7XG4gICAgY29uc3QgdHhzID0gYXdhaXQgdGhpcy5hcGkuZ2V0VHhzQnlBZGRyZXNzKGFkZHJlc3MsIHtcbiAgICAgIG9mZnNldCxcbiAgICAgIGxpbWl0LFxuICAgIH0pO1xuICAgIGlmICghdHhzLml0ZW1zKVxuICAgICAgdGhyb3cgbmV3IEVycm9yKFxuICAgICAgICAnRXhwbG9yZXIgQWRkcmVzc1RyYW5zYWN0aW9ucyBhcGkgZXhwZWN0ZWQgdG8gaGF2ZSBpdGVtcycsXG4gICAgICApO1xuICAgIHJldHVybiB7XG4gICAgICBpdGVtczogdHhzLml0ZW1zLm1hcCgodHgpID0+IHRoaXMuY29udmVydFRyYW5zYWN0aW9uKHR4KSksXG4gICAgICB0b3RhbDogdHhzLnRvdGFsISxcbiAgICB9O1xuICB9O1xufVxuIl19
|
package/dist/ergo/utils.d.ts
CHANGED
|
@@ -1,3 +1,4 @@
|
|
|
1
|
+
import { DummyLogger } from '@rosen-bridge/abstract-logger';
|
|
1
2
|
import { OutputBox } from '@rosen-bridge/scanner-interfaces';
|
|
2
3
|
/**
|
|
3
4
|
* Check box to have specified tokens
|
|
@@ -5,4 +6,18 @@ import { OutputBox } from '@rosen-bridge/scanner-interfaces';
|
|
|
5
6
|
* @return true if box has the required token and false otherwise
|
|
6
7
|
*/
|
|
7
8
|
export declare const boxHasToken: (box: OutputBox, tokenIds: string[]) => boolean;
|
|
9
|
+
/**
|
|
10
|
+
* Create delay in procedures based on the specified time in milliseconds
|
|
11
|
+
* @param time
|
|
12
|
+
*/
|
|
13
|
+
export declare const delay: (time: number) => Promise<unknown>;
|
|
14
|
+
/**
|
|
15
|
+
* Retry the request in case of failure
|
|
16
|
+
* Throw error if request fails after RETRIAL_COUNT retrials
|
|
17
|
+
* Wait for 1 second between each trial
|
|
18
|
+
* @param request
|
|
19
|
+
* @param logger
|
|
20
|
+
* @returns
|
|
21
|
+
*/
|
|
22
|
+
export declare const requestWithRetrial: <returnT>(request: () => Promise<returnT>, logger?: DummyLogger) => Promise<returnT>;
|
|
8
23
|
//# sourceMappingURL=utils.d.ts.map
|
package/dist/ergo/utils.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"utils.d.ts","sourceRoot":"","sources":["../../lib/ergo/utils.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,SAAS,EAAE,MAAM,kCAAkC,CAAC;
|
|
1
|
+
{"version":3,"file":"utils.d.ts","sourceRoot":"","sources":["../../lib/ergo/utils.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,WAAW,EAAE,MAAM,+BAA+B,CAAC;AAC5D,OAAO,EAAE,SAAS,EAAE,MAAM,kCAAkC,CAAC;AAI7D;;;;GAIG;AACH,eAAO,MAAM,WAAW,GAAI,KAAK,SAAS,EAAE,UAAU,MAAM,EAAE,YAM7D,CAAC;AAEF;;;GAGG;AACH,eAAO,MAAM,KAAK,GAAU,MAAM,MAAM,qBACa,CAAC;AAEtD;;;;;;;GAOG;AACH,eAAO,MAAM,kBAAkB,GAAU,OAAO,EAC9C,SAAS,MAAM,OAAO,CAAC,OAAO,CAAC,EAC/B,oBAA0B,KACzB,OAAO,CAAC,OAAO,CAkBjB,CAAC"}
|
package/dist/ergo/utils.js
CHANGED
|
@@ -1,4 +1,6 @@
|
|
|
1
1
|
import { intersection } from 'lodash-es';
|
|
2
|
+
import { DummyLogger } from '@rosen-bridge/abstract-logger';
|
|
3
|
+
import { RETRIAL_COUNT } from '../constants';
|
|
2
4
|
/**
|
|
3
5
|
* Check box to have specified tokens
|
|
4
6
|
* @param box
|
|
@@ -13,4 +15,35 @@ export const boxHasToken = (box, tokenIds) => {
|
|
|
13
15
|
return true;
|
|
14
16
|
return false;
|
|
15
17
|
};
|
|
16
|
-
|
|
18
|
+
/**
|
|
19
|
+
* Create delay in procedures based on the specified time in milliseconds
|
|
20
|
+
* @param time
|
|
21
|
+
*/
|
|
22
|
+
export const delay = async (time) => new Promise((resolve) => setTimeout(resolve, time));
|
|
23
|
+
/**
|
|
24
|
+
* Retry the request in case of failure
|
|
25
|
+
* Throw error if request fails after RETRIAL_COUNT retrials
|
|
26
|
+
* Wait for 1 second between each trial
|
|
27
|
+
* @param request
|
|
28
|
+
* @param logger
|
|
29
|
+
* @returns
|
|
30
|
+
*/
|
|
31
|
+
export const requestWithRetrial = async (request, logger = new DummyLogger()) => {
|
|
32
|
+
let trial = 0;
|
|
33
|
+
let lastErrorMessage;
|
|
34
|
+
while (trial < RETRIAL_COUNT) {
|
|
35
|
+
try {
|
|
36
|
+
const result = await request();
|
|
37
|
+
return result;
|
|
38
|
+
}
|
|
39
|
+
catch (e) {
|
|
40
|
+
lastErrorMessage = e instanceof Error ? e.message : String(e);
|
|
41
|
+
trial++;
|
|
42
|
+
logger.warn(`request failed with error ${lastErrorMessage}`);
|
|
43
|
+
logger.debug(`Retrying the request with retrial step ${trial}`);
|
|
44
|
+
await delay(1000);
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
throw new Error(`request failed after ${trial} retrials with error: ${lastErrorMessage}`);
|
|
48
|
+
};
|
|
49
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXRpbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9saWIvZXJnby91dGlscy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0sV0FBVyxDQUFDO0FBRXpDLE9BQU8sRUFBRSxXQUFXLEVBQUUsTUFBTSwrQkFBK0IsQ0FBQztBQUc1RCxPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0sY0FBYyxDQUFDO0FBRTdDOzs7O0dBSUc7QUFDSCxNQUFNLENBQUMsTUFBTSxXQUFXLEdBQUcsQ0FBQyxHQUFjLEVBQUUsUUFBa0IsRUFBRSxFQUFFO0lBQ2hFLElBQUksQ0FBQyxHQUFHLENBQUMsTUFBTTtRQUFFLE9BQU8sS0FBSyxDQUFDO0lBQzlCLE1BQU0sU0FBUyxHQUFHLEdBQUcsQ0FBQyxNQUFNLENBQUMsR0FBRyxDQUFDLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxLQUFLLENBQUMsT0FBTyxDQUFDLENBQUM7SUFDM0QsTUFBTSxjQUFjLEdBQUcsWUFBWSxDQUFDLFFBQVEsRUFBRSxTQUFTLENBQUMsQ0FBQztJQUN6RCxJQUFJLGNBQWMsQ0FBQyxNQUFNLElBQUksUUFBUSxDQUFDLE1BQU07UUFBRSxPQUFPLElBQUksQ0FBQztJQUMxRCxPQUFPLEtBQUssQ0FBQztBQUNmLENBQUMsQ0FBQztBQUVGOzs7R0FHRztBQUNILE1BQU0sQ0FBQyxNQUFNLEtBQUssR0FBRyxLQUFLLEVBQUUsSUFBWSxFQUFFLEVBQUUsQ0FDMUMsSUFBSSxPQUFPLENBQUMsQ0FBQyxPQUFPLEVBQUUsRUFBRSxDQUFDLFVBQVUsQ0FBQyxPQUFPLEVBQUUsSUFBSSxDQUFDLENBQUMsQ0FBQztBQUV0RDs7Ozs7OztHQU9HO0FBQ0gsTUFBTSxDQUFDLE1BQU0sa0JBQWtCLEdBQUcsS0FBSyxFQUNyQyxPQUErQixFQUMvQixNQUFNLEdBQUcsSUFBSSxXQUFXLEVBQUUsRUFDUixFQUFFO0lBQ3BCLElBQUksS0FBSyxHQUFHLENBQUMsQ0FBQztJQUNkLElBQUksZ0JBQW9DLENBQUM7SUFDekMsT0FBTyxLQUFLLEdBQUcsYUFBYSxFQUFFLENBQUM7UUFDN0IsSUFBSSxDQUFDO1lBQ0gsTUFBTSxNQUFNLEdBQUcsTUFBTSxPQUFPLEVBQUUsQ0FBQztZQUMvQixPQUFPLE1BQU0sQ0FBQztRQUNoQixDQUFDO1FBQUMsT0FBTyxDQUFDLEVBQUUsQ0FBQztZQUNYLGdCQUFnQixHQUFHLENBQUMsWUFBWSxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsQ0FBQztZQUM5RCxLQUFLLEVBQUUsQ0FBQztZQUNSLE1BQU0sQ0FBQyxJQUFJLENBQUMsNkJBQTZCLGdCQUFnQixFQUFFLENBQUMsQ0FBQztZQUM3RCxNQUFNLENBQUMsS0FBSyxDQUFDLDBDQUEwQyxLQUFLLEVBQUUsQ0FBQyxDQUFDO1lBQ2hFLE1BQU0sS0FBSyxDQUFDLElBQUksQ0FBQyxDQUFDO1FBQ3BCLENBQUM7SUFDSCxDQUFDO0lBQ0QsTUFBTSxJQUFJLEtBQUssQ0FDYix3QkFBd0IsS0FBSyx5QkFBeUIsZ0JBQWdCLEVBQUUsQ0FDekUsQ0FBQztBQUNKLENBQUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGludGVyc2VjdGlvbiB9IGZyb20gJ2xvZGFzaC1lcyc7XG5cbmltcG9ydCB7IER1bW15TG9nZ2VyIH0gZnJvbSAnQHJvc2VuLWJyaWRnZS9hYnN0cmFjdC1sb2dnZXInO1xuaW1wb3J0IHsgT3V0cHV0Qm94IH0gZnJvbSAnQHJvc2VuLWJyaWRnZS9zY2FubmVyLWludGVyZmFjZXMnO1xuXG5pbXBvcnQgeyBSRVRSSUFMX0NPVU5UIH0gZnJvbSAnLi4vY29uc3RhbnRzJztcblxuLyoqXG4gKiBDaGVjayBib3ggdG8gaGF2ZSBzcGVjaWZpZWQgdG9rZW5zXG4gKiBAcGFyYW0gYm94XG4gKiBAcmV0dXJuIHRydWUgaWYgYm94IGhhcyB0aGUgcmVxdWlyZWQgdG9rZW4gYW5kIGZhbHNlIG90aGVyd2lzZVxuICovXG5leHBvcnQgY29uc3QgYm94SGFzVG9rZW4gPSAoYm94OiBPdXRwdXRCb3gsIHRva2VuSWRzOiBzdHJpbmdbXSkgPT4ge1xuICBpZiAoIWJveC5hc3NldHMpIHJldHVybiBmYWxzZTtcbiAgY29uc3QgYm94VG9rZW5zID0gYm94LmFzc2V0cy5tYXAoKHRva2VuKSA9PiB0b2tlbi50b2tlbklkKTtcbiAgY29uc3QgcmVxdWlyZWRUb2tlbnMgPSBpbnRlcnNlY3Rpb24odG9rZW5JZHMsIGJveFRva2Vucyk7XG4gIGlmIChyZXF1aXJlZFRva2Vucy5sZW5ndGggPT0gdG9rZW5JZHMubGVuZ3RoKSByZXR1cm4gdHJ1ZTtcbiAgcmV0dXJuIGZhbHNlO1xufTtcblxuLyoqXG4gKiBDcmVhdGUgZGVsYXkgaW4gcHJvY2VkdXJlcyBiYXNlZCBvbiB0aGUgc3BlY2lmaWVkIHRpbWUgaW4gbWlsbGlzZWNvbmRzXG4gKiBAcGFyYW0gdGltZVxuICovXG5leHBvcnQgY29uc3QgZGVsYXkgPSBhc3luYyAodGltZTogbnVtYmVyKSA9PlxuICBuZXcgUHJvbWlzZSgocmVzb2x2ZSkgPT4gc2V0VGltZW91dChyZXNvbHZlLCB0aW1lKSk7XG5cbi8qKlxuICogUmV0cnkgdGhlIHJlcXVlc3QgaW4gY2FzZSBvZiBmYWlsdXJlXG4gKiBUaHJvdyBlcnJvciBpZiByZXF1ZXN0IGZhaWxzIGFmdGVyIFJFVFJJQUxfQ09VTlQgcmV0cmlhbHNcbiAqIFdhaXQgZm9yIDEgc2Vjb25kIGJldHdlZW4gZWFjaCB0cmlhbFxuICogQHBhcmFtIHJlcXVlc3RcbiAqIEBwYXJhbSBsb2dnZXJcbiAqIEByZXR1cm5zXG4gKi9cbmV4cG9ydCBjb25zdCByZXF1ZXN0V2l0aFJldHJpYWwgPSBhc3luYyA8cmV0dXJuVD4oXG4gIHJlcXVlc3Q6ICgpID0+IFByb21pc2U8cmV0dXJuVD4sXG4gIGxvZ2dlciA9IG5ldyBEdW1teUxvZ2dlcigpLFxuKTogUHJvbWlzZTxyZXR1cm5UPiA9PiB7XG4gIGxldCB0cmlhbCA9IDA7XG4gIGxldCBsYXN0RXJyb3JNZXNzYWdlOiBzdHJpbmcgfCB1bmRlZmluZWQ7XG4gIHdoaWxlICh0cmlhbCA8IFJFVFJJQUxfQ09VTlQpIHtcbiAgICB0cnkge1xuICAgICAgY29uc3QgcmVzdWx0ID0gYXdhaXQgcmVxdWVzdCgpO1xuICAgICAgcmV0dXJuIHJlc3VsdDtcbiAgICB9IGNhdGNoIChlKSB7XG4gICAgICBsYXN0RXJyb3JNZXNzYWdlID0gZSBpbnN0YW5jZW9mIEVycm9yID8gZS5tZXNzYWdlIDogU3RyaW5nKGUpO1xuICAgICAgdHJpYWwrKztcbiAgICAgIGxvZ2dlci53YXJuKGByZXF1ZXN0IGZhaWxlZCB3aXRoIGVycm9yICR7bGFzdEVycm9yTWVzc2FnZX1gKTtcbiAgICAgIGxvZ2dlci5kZWJ1ZyhgUmV0cnlpbmcgdGhlIHJlcXVlc3Qgd2l0aCByZXRyaWFsIHN0ZXAgJHt0cmlhbH1gKTtcbiAgICAgIGF3YWl0IGRlbGF5KDEwMDApO1xuICAgIH1cbiAgfVxuICB0aHJvdyBuZXcgRXJyb3IoXG4gICAgYHJlcXVlc3QgZmFpbGVkIGFmdGVyICR7dHJpYWx9IHJldHJpYWxzIHdpdGggZXJyb3I6ICR7bGFzdEVycm9yTWVzc2FnZX1gLFxuICApO1xufTtcbiJdfQ==
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@rosen-bridge/abstract-extractor",
|
|
3
|
-
"version": "
|
|
3
|
+
"version": "3.0.0-8f3c7016",
|
|
4
4
|
"description": "Rosen Bridge extractor interfaces to work with scanner",
|
|
5
5
|
"repository": {
|
|
6
6
|
"type": "git",
|
|
@@ -34,7 +34,9 @@
|
|
|
34
34
|
"@rosen-bridge/scanner-interfaces": "^0.2.1",
|
|
35
35
|
"@rosen-clients/ergo-explorer": "^2.1.0",
|
|
36
36
|
"@rosen-clients/ergo-node": "^3.1.0",
|
|
37
|
+
"await-semaphore": "^0.1.3",
|
|
37
38
|
"lodash-es": "^4.17.21",
|
|
39
|
+
"p-queue": "^9.0.0",
|
|
38
40
|
"uuid": "^9.0.0"
|
|
39
41
|
},
|
|
40
42
|
"devDependencies": {
|
|
@@ -1,80 +0,0 @@
|
|
|
1
|
-
import { AbstractLogger, DummyLogger } from '@rosen-bridge/abstract-logger';
|
|
2
|
-
import { Transaction, OutputBox, InputExtension, BlockInfo } from '@rosen-bridge/scanner-interfaces';
|
|
3
|
-
import { AbstractExtractor } from '../abstractExtractor';
|
|
4
|
-
import { AbstractErgoExtractorAction } from './abstractErgoExtractorAction';
|
|
5
|
-
import { AbstractErgoExtractorEntity } from './abstractErgoExtractorEntity';
|
|
6
|
-
import { AbstractBoxData, CallbackType, CallbackMap, CallbackDataMap, TxExtra } from './interfaces';
|
|
7
|
-
export declare abstract class AbstractErgoExtractor<ExtractedData extends AbstractBoxData, ExtractorEntity extends AbstractErgoExtractorEntity> extends AbstractExtractor<Transaction> {
|
|
8
|
-
protected abstract actions: AbstractErgoExtractorAction<ExtractedData, ExtractorEntity>;
|
|
9
|
-
protected logger: AbstractLogger;
|
|
10
|
-
protected callbacks: {
|
|
11
|
-
[K in CallbackType]: Map<string, CallbackMap<ExtractedData>[K]>;
|
|
12
|
-
};
|
|
13
|
-
private callbackMutex;
|
|
14
|
-
constructor(logger?: DummyLogger);
|
|
15
|
-
/**
|
|
16
|
-
* hook a new callback on a callback type
|
|
17
|
-
* @param type
|
|
18
|
-
* @param id
|
|
19
|
-
* @param callback
|
|
20
|
-
* @returns callback registered id
|
|
21
|
-
*/
|
|
22
|
-
hook: <T extends CallbackType>(type: T, callback: CallbackMap<ExtractedData>[T]) => Promise<string>;
|
|
23
|
-
/**
|
|
24
|
-
* unhook a callback on a type
|
|
25
|
-
* returns false if there is no registered callback with the id
|
|
26
|
-
* @param type
|
|
27
|
-
* @param id
|
|
28
|
-
* @returns success status
|
|
29
|
-
*/
|
|
30
|
-
unhook: (type: CallbackType, id: string) => Promise<boolean>;
|
|
31
|
-
/**
|
|
32
|
-
* trigger all callbacks registered on a specific type with the provided data
|
|
33
|
-
* @param type
|
|
34
|
-
* @param data
|
|
35
|
-
*/
|
|
36
|
-
protected triggerCallbacks<T extends CallbackType>(type: T, data: CallbackDataMap<ExtractedData>[T]): void;
|
|
37
|
-
/**
|
|
38
|
-
* extract box data to proper format (not including spending information)
|
|
39
|
-
* @param box
|
|
40
|
-
* @param inputExtensions all input box extensions in transaction
|
|
41
|
-
* @return extracted data in proper format
|
|
42
|
-
*/
|
|
43
|
-
abstract extractBoxData: (box: OutputBox, inputExtensions: InputExtension[], txExtra?: TxExtra) => ExtractedData | undefined;
|
|
44
|
-
/**
|
|
45
|
-
* check proper data format in the box
|
|
46
|
-
* @param box
|
|
47
|
-
* @return true if the box has the required data and false otherwise
|
|
48
|
-
*/
|
|
49
|
-
abstract hasData: (box: OutputBox) => boolean;
|
|
50
|
-
/**
|
|
51
|
-
* create spend info array for the transaction
|
|
52
|
-
* @param tx
|
|
53
|
-
* @returns spend info array of the transaction
|
|
54
|
-
*/
|
|
55
|
-
getTransactionSpendInfo: (tx: Transaction) => {
|
|
56
|
-
txId: string;
|
|
57
|
-
boxId: string;
|
|
58
|
-
index: number;
|
|
59
|
-
}[];
|
|
60
|
-
/**
|
|
61
|
-
* extract transaction extra information
|
|
62
|
-
* override this function if there is extra needed information
|
|
63
|
-
* @param tx
|
|
64
|
-
* @returns
|
|
65
|
-
*/
|
|
66
|
-
getTransactionExtraData: (tx: Transaction) => TxExtra;
|
|
67
|
-
/**
|
|
68
|
-
* process a list of transactions in a block and store required information
|
|
69
|
-
* @param txs list of transactions in the block
|
|
70
|
-
* @param block
|
|
71
|
-
* @return true if the process is completed successfully and false otherwise
|
|
72
|
-
*/
|
|
73
|
-
processTransactions: (txs: Transaction[], block: BlockInfo) => Promise<boolean>;
|
|
74
|
-
/**
|
|
75
|
-
* fork one block and remove all stored information for this block
|
|
76
|
-
* @param hash block hash
|
|
77
|
-
*/
|
|
78
|
-
forkBlock: (hash: string) => Promise<void>;
|
|
79
|
-
}
|
|
80
|
-
//# sourceMappingURL=abstractErgoExtractor.d.ts.map
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"abstractErgoExtractor.d.ts","sourceRoot":"","sources":["../../lib/ergo/abstractErgoExtractor.ts"],"names":[],"mappings":"AAGA,OAAO,EAAE,cAAc,EAAE,WAAW,EAAE,MAAM,+BAA+B,CAAC;AAE5E,OAAO,EACL,WAAW,EACX,SAAS,EACT,cAAc,EACd,SAAS,EACV,MAAM,kCAAkC,CAAC;AAE1C,OAAO,EAAE,iBAAiB,EAAE,MAAM,sBAAsB,CAAC;AACzD,OAAO,EAAE,2BAA2B,EAAE,MAAM,+BAA+B,CAAC;AAC5E,OAAO,EAAE,2BAA2B,EAAE,MAAM,+BAA+B,CAAC;AAC5E,OAAO,EACL,eAAe,EAEf,YAAY,EACZ,WAAW,EACX,eAAe,EACf,OAAO,EACR,MAAM,cAAc,CAAC;AAEtB,8BAAsB,qBAAqB,CACzC,aAAa,SAAS,eAAe,EACrC,eAAe,SAAS,2BAA2B,CACnD,SAAQ,iBAAiB,CAAC,WAAW,CAAC;IACtC,SAAS,CAAC,QAAQ,CAAC,OAAO,EAAE,2BAA2B,CACrD,aAAa,EACb,eAAe,CAChB,CAAC;IACF,SAAS,CAAC,MAAM,EAAE,cAAc,CAAC;IACjC,SAAS,CAAC,SAAS,EAAE;SAClB,CAAC,IAAI,YAAY,GAAG,GAAG,CAAC,MAAM,EAAE,WAAW,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,CAAC;KAChE,CAKC;IACF,OAAO,CAAC,aAAa,CAAe;gBAExB,MAAM,cAAoB;IAKtC;;;;;;OAMG;IACH,IAAI,GAAU,CAAC,SAAS,YAAY,EAClC,MAAM,CAAC,EACP,UAAU,WAAW,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,KACtC,OAAO,CAAC,MAAM,CAAC,CAOhB;IAEF;;;;;;OAMG;IACH,MAAM,GAAU,MAAM,YAAY,EAAE,IAAI,MAAM,KAAG,OAAO,CAAC,OAAO,CAAC,CAY/D;IAEF;;;;OAIG;IACH,SAAS,CAAC,gBAAgB,CAAC,CAAC,SAAS,YAAY,EAC/C,IAAI,EAAE,CAAC,EACP,IAAI,EAAE,eAAe,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,GACtC,IAAI;IAOP;;;;;OAKG;IACH,QAAQ,CAAC,cAAc,EAAE,CACvB,GAAG,EAAE,SAAS,EACd,eAAe,EAAE,cAAc,EAAE,EACjC,OAAO,CAAC,EAAE,OAAO,KACd,aAAa,GAAG,SAAS,CAAC;IAE/B;;;;OAIG;IACH,QAAQ,CAAC,OAAO,EAAE,CAAC,GAAG,EAAE,SAAS,KAAK,OAAO,CAAC;IAE9C;;;;OAIG;IACH,uBAAuB,GAAI,IAAI,WAAW;;;;QAQxC;IAEF;;;;;OAKG;IACH,uBAAuB,GACrB,IAAI,WAAW,KACd,OAAO,CAER;IAEF;;;;;OAKG;IACH,mBAAmB,GACjB,KAAK,WAAW,EAAE,EAClB,OAAO,SAAS,KACf,OAAO,CAAC,OAAO,CAAC,CAwDjB;IAEF;;;OAGG;IACH,SAAS,GAAU,MAAM,MAAM,KAAG,OAAO,CAAC,IAAI,CAAC,CAM7C;CACH"}
|
|
@@ -1,142 +0,0 @@
|
|
|
1
|
-
import { Mutex } from 'await-semaphore';
|
|
2
|
-
import { v4 as uuidv4 } from 'uuid';
|
|
3
|
-
import { DummyLogger } from '@rosen-bridge/abstract-logger';
|
|
4
|
-
import JsonBigInt from '@rosen-bridge/json-bigint';
|
|
5
|
-
import { AbstractExtractor } from '../abstractExtractor';
|
|
6
|
-
import { CallbackType, } from './interfaces';
|
|
7
|
-
export class AbstractErgoExtractor extends AbstractExtractor {
|
|
8
|
-
logger;
|
|
9
|
-
callbacks = {
|
|
10
|
-
[CallbackType.Update]: new Map(),
|
|
11
|
-
[CallbackType.Insert]: new Map(),
|
|
12
|
-
[CallbackType.Delete]: new Map(),
|
|
13
|
-
[CallbackType.Spend]: new Map(),
|
|
14
|
-
};
|
|
15
|
-
callbackMutex = new Mutex();
|
|
16
|
-
constructor(logger = new DummyLogger()) {
|
|
17
|
-
super();
|
|
18
|
-
this.logger = logger;
|
|
19
|
-
}
|
|
20
|
-
/**
|
|
21
|
-
* hook a new callback on a callback type
|
|
22
|
-
* @param type
|
|
23
|
-
* @param id
|
|
24
|
-
* @param callback
|
|
25
|
-
* @returns callback registered id
|
|
26
|
-
*/
|
|
27
|
-
hook = async (type, callback) => {
|
|
28
|
-
const release = await this.callbackMutex.acquire();
|
|
29
|
-
const callbackMap = this.callbacks[type];
|
|
30
|
-
const id = uuidv4();
|
|
31
|
-
callbackMap.set(id, callback);
|
|
32
|
-
release();
|
|
33
|
-
return id;
|
|
34
|
-
};
|
|
35
|
-
/**
|
|
36
|
-
* unhook a callback on a type
|
|
37
|
-
* returns false if there is no registered callback with the id
|
|
38
|
-
* @param type
|
|
39
|
-
* @param id
|
|
40
|
-
* @returns success status
|
|
41
|
-
*/
|
|
42
|
-
unhook = async (type, id) => {
|
|
43
|
-
const release = await this.callbackMutex.acquire();
|
|
44
|
-
const callbackMap = this.callbacks[type];
|
|
45
|
-
if (!callbackMap.has(id)) {
|
|
46
|
-
this.logger.warn(`Callback with Id [${id}] is not registered for type [${type}].`);
|
|
47
|
-
return false;
|
|
48
|
-
}
|
|
49
|
-
callbackMap.delete(id);
|
|
50
|
-
release();
|
|
51
|
-
return true;
|
|
52
|
-
};
|
|
53
|
-
/**
|
|
54
|
-
* trigger all callbacks registered on a specific type with the provided data
|
|
55
|
-
* @param type
|
|
56
|
-
* @param data
|
|
57
|
-
*/
|
|
58
|
-
triggerCallbacks(type, data) {
|
|
59
|
-
const callbackMap = this.callbacks[type];
|
|
60
|
-
callbackMap.forEach((callback) => {
|
|
61
|
-
callback(data);
|
|
62
|
-
});
|
|
63
|
-
}
|
|
64
|
-
/**
|
|
65
|
-
* create spend info array for the transaction
|
|
66
|
-
* @param tx
|
|
67
|
-
* @returns spend info array of the transaction
|
|
68
|
-
*/
|
|
69
|
-
getTransactionSpendInfo = (tx) => {
|
|
70
|
-
let boxIndex = 1;
|
|
71
|
-
const spendInfoArray = [];
|
|
72
|
-
for (const input of tx.inputs) {
|
|
73
|
-
spendInfoArray.push({ txId: tx.id, boxId: input.boxId, index: boxIndex });
|
|
74
|
-
boxIndex += 1;
|
|
75
|
-
}
|
|
76
|
-
return spendInfoArray;
|
|
77
|
-
};
|
|
78
|
-
/**
|
|
79
|
-
* extract transaction extra information
|
|
80
|
-
* override this function if there is extra needed information
|
|
81
|
-
* @param tx
|
|
82
|
-
* @returns
|
|
83
|
-
*/
|
|
84
|
-
getTransactionExtraData = (tx) => {
|
|
85
|
-
return {};
|
|
86
|
-
};
|
|
87
|
-
/**
|
|
88
|
-
* process a list of transactions in a block and store required information
|
|
89
|
-
* @param txs list of transactions in the block
|
|
90
|
-
* @param block
|
|
91
|
-
* @return true if the process is completed successfully and false otherwise
|
|
92
|
-
*/
|
|
93
|
-
processTransactions = async (txs, block) => {
|
|
94
|
-
try {
|
|
95
|
-
const boxes = [];
|
|
96
|
-
const spentInfos = [];
|
|
97
|
-
for (const tx of txs) {
|
|
98
|
-
const inputExtensions = tx.inputs.map((input) => input.extension || {});
|
|
99
|
-
for (const output of tx.outputs) {
|
|
100
|
-
if (!this.hasData(output)) {
|
|
101
|
-
continue;
|
|
102
|
-
}
|
|
103
|
-
this.logger.debug(`Trying to extract data from box ${output.boxId}`);
|
|
104
|
-
const extractedData = this.extractBoxData(output, inputExtensions, this.getTransactionExtraData(tx));
|
|
105
|
-
if (extractedData) {
|
|
106
|
-
this.logger.debug(`Extracted data ${JsonBigInt.stringify(extractedData)} from box ${output.boxId}`);
|
|
107
|
-
boxes.push(extractedData);
|
|
108
|
-
}
|
|
109
|
-
}
|
|
110
|
-
spentInfos.push(...this.getTransactionSpendInfo(tx));
|
|
111
|
-
}
|
|
112
|
-
if (boxes.length > 0) {
|
|
113
|
-
if (!(await this.actions.storeBoxes(boxes, block, this.getId()))) {
|
|
114
|
-
this.logger.warn(`Data insertion failed for ${this.getId()} at the block ${block.height}`);
|
|
115
|
-
return false;
|
|
116
|
-
}
|
|
117
|
-
this.triggerCallbacks(CallbackType.Insert, boxes);
|
|
118
|
-
}
|
|
119
|
-
const spentData = await this.actions.spendBoxes(spentInfos, block, this.getId());
|
|
120
|
-
if (spentData.length > 0) {
|
|
121
|
-
this.triggerCallbacks(CallbackType.Spend, spentData);
|
|
122
|
-
}
|
|
123
|
-
}
|
|
124
|
-
catch (e) {
|
|
125
|
-
this.logger.error(`Processing transactions failed for ${this.getId()} at the block ${block.height} with error: ${e}`);
|
|
126
|
-
return false;
|
|
127
|
-
}
|
|
128
|
-
return true;
|
|
129
|
-
};
|
|
130
|
-
/**
|
|
131
|
-
* fork one block and remove all stored information for this block
|
|
132
|
-
* @param hash block hash
|
|
133
|
-
*/
|
|
134
|
-
forkBlock = async (hash) => {
|
|
135
|
-
const result = await this.actions.deleteBlockBoxes(hash, this.getId());
|
|
136
|
-
if (result.deletedData.length > 0)
|
|
137
|
-
this.triggerCallbacks(CallbackType.Delete, result.deletedData);
|
|
138
|
-
if (result.updatedData.length > 0)
|
|
139
|
-
this.triggerCallbacks(CallbackType.Update, result.updatedData);
|
|
140
|
-
};
|
|
141
|
-
}
|
|
142
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"abstractErgoExtractor.js","sourceRoot":"","sources":["../../lib/ergo/abstractErgoExtractor.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,EAAE,MAAM,iBAAiB,CAAC;AACxC,OAAO,EAAE,EAAE,IAAI,MAAM,EAAE,MAAM,MAAM,CAAC;AAEpC,OAAO,EAAkB,WAAW,EAAE,MAAM,+BAA+B,CAAC;AAC5E,OAAO,UAAU,MAAM,2BAA2B,CAAC;AAQnD,OAAO,EAAE,iBAAiB,EAAE,MAAM,sBAAsB,CAAC;AAGzD,OAAO,EAGL,YAAY,GAIb,MAAM,cAAc,CAAC;AAEtB,MAAM,OAAgB,qBAGpB,SAAQ,iBAA8B;IAK5B,MAAM,CAAiB;IACvB,SAAS,GAEf;QACF,CAAC,YAAY,CAAC,MAAM,CAAC,EAAE,IAAI,GAAG,EAAE;QAChC,CAAC,YAAY,CAAC,MAAM,CAAC,EAAE,IAAI,GAAG,EAAE;QAChC,CAAC,YAAY,CAAC,MAAM,CAAC,EAAE,IAAI,GAAG,EAAE;QAChC,CAAC,YAAY,CAAC,KAAK,CAAC,EAAE,IAAI,GAAG,EAAE;KAChC,CAAC;IACM,aAAa,GAAG,IAAI,KAAK,EAAE,CAAC;IAEpC,YAAY,MAAM,GAAG,IAAI,WAAW,EAAE;QACpC,KAAK,EAAE,CAAC;QACR,IAAI,CAAC,MAAM,GAAG,MAAM,CAAC;IACvB,CAAC;IAED;;;;;;OAMG;IACH,IAAI,GAAG,KAAK,EACV,IAAO,EACP,QAAuC,EACtB,EAAE;QACnB,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,aAAa,CAAC,OAAO,EAAE,CAAC;QACnD,MAAM,WAAW,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;QACzC,MAAM,EAAE,GAAG,MAAM,EAAE,CAAC;QACpB,WAAW,CAAC,GAAG,CAAC,EAAE,EAAE,QAAQ,CAAC,CAAC;QAC9B,OAAO,EAAE,CAAC;QACV,OAAO,EAAE,CAAC;IACZ,CAAC,CAAC;IAEF;;;;;;OAMG;IACH,MAAM,GAAG,KAAK,EAAE,IAAkB,EAAE,EAAU,EAAoB,EAAE;QAClE,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,aAAa,CAAC,OAAO,EAAE,CAAC;QACnD,MAAM,WAAW,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;QACzC,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,CAAC;YACzB,IAAI,CAAC,MAAM,CAAC,IAAI,CACd,qBAAqB,EAAE,iCAAiC,IAAI,IAAI,CACjE,CAAC;YACF,OAAO,KAAK,CAAC;QACf,CAAC;QACD,WAAW,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;QACvB,OAAO,EAAE,CAAC;QACV,OAAO,IAAI,CAAC;IACd,CAAC,CAAC;IAEF;;;;OAIG;IACO,gBAAgB,CACxB,IAAO,EACP,IAAuC;QAEvC,MAAM,WAAW,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;QACzC,WAAW,CAAC,OAAO,CAAC,CAAC,QAAQ,EAAE,EAAE;YAC/B,QAAQ,CAAC,IAAI,CAAC,CAAC;QACjB,CAAC,CAAC,CAAC;IACL,CAAC;IAqBD;;;;OAIG;IACH,uBAAuB,GAAG,CAAC,EAAe,EAAE,EAAE;QAC5C,IAAI,QAAQ,GAAG,CAAC,CAAC;QACjB,MAAM,cAAc,GAAG,EAAE,CAAC;QAC1B,KAAK,MAAM,KAAK,IAAI,EAAE,CAAC,MAAM,EAAE,CAAC;YAC9B,cAAc,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,EAAE,CAAC,EAAE,EAAE,KAAK,EAAE,KAAK,CAAC,KAAK,EAAE,KAAK,EAAE,QAAQ,EAAE,CAAC,CAAC;YAC1E,QAAQ,IAAI,CAAC,CAAC;QAChB,CAAC;QACD,OAAO,cAAc,CAAC;IACxB,CAAC,CAAC;IAEF;;;;;OAKG;IACH,uBAAuB,GAAG,CACxB,EAAe,EACN,EAAE;QACX,OAAO,EAAE,CAAC;IACZ,CAAC,CAAC;IAEF;;;;;OAKG;IACH,mBAAmB,GAAG,KAAK,EACzB,GAAkB,EAClB,KAAgB,EACE,EAAE;QACpB,IAAI,CAAC;YACH,MAAM,KAAK,GAAyB,EAAE,CAAC;YACvC,MAAM,UAAU,GAAqB,EAAE,CAAC;YACxC,KAAK,MAAM,EAAE,IAAI,GAAG,EAAE,CAAC;gBACrB,MAAM,eAAe,GAAG,EAAE,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,SAAS,IAAI,EAAE,CAAC,CAAC;gBACxE,KAAK,MAAM,MAAM,IAAI,EAAE,CAAC,OAAO,EAAE,CAAC;oBAChC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC;wBAC1B,SAAS;oBACX,CAAC;oBACD,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,mCAAmC,MAAM,CAAC,KAAK,EAAE,CAAC,CAAC;oBACrE,MAAM,aAAa,GAAG,IAAI,CAAC,cAAc,CACvC,MAAM,EACN,eAAe,EACf,IAAI,CAAC,uBAAuB,CAAC,EAAE,CAAC,CACjC,CAAC;oBACF,IAAI,aAAa,EAAE,CAAC;wBAClB,IAAI,CAAC,MAAM,CAAC,KAAK,CACf,kBAAkB,UAAU,CAAC,SAAS,CAAC,aAAa,CAAC,aACnD,MAAM,CAAC,KACT,EAAE,CACH,CAAC;wBACF,KAAK,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;oBAC5B,CAAC;gBACH,CAAC;gBACD,UAAU,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,uBAAuB,CAAC,EAAE,CAAC,CAAC,CAAC;YACvD,CAAC;YAED,IAAI,KAAK,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBACrB,IAAI,CAAC,CAAC,MAAM,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,KAAK,EAAE,KAAK,EAAE,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC,EAAE,CAAC;oBACjE,IAAI,CAAC,MAAM,CAAC,IAAI,CACd,6BAA6B,IAAI,CAAC,KAAK,EAAE,iBACvC,KAAK,CAAC,MACR,EAAE,CACH,CAAC;oBACF,OAAO,KAAK,CAAC;gBACf,CAAC;gBACD,IAAI,CAAC,gBAAgB,CAAC,YAAY,CAAC,MAAM,EAAE,KAAK,CAAC,CAAC;YACpD,CAAC;YACD,MAAM,SAAS,GAAG,MAAM,IAAI,CAAC,OAAO,CAAC,UAAU,CAC7C,UAAU,EACV,KAAK,EACL,IAAI,CAAC,KAAK,EAAE,CACb,CAAC;YACF,IAAI,SAAS,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBACzB,IAAI,CAAC,gBAAgB,CAAC,YAAY,CAAC,KAAK,EAAE,SAAS,CAAC,CAAC;YACvD,CAAC;QACH,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,IAAI,CAAC,MAAM,CAAC,KAAK,CACf,sCAAsC,IAAI,CAAC,KAAK,EAAE,iBAChD,KAAK,CAAC,MACR,gBAAgB,CAAC,EAAE,CACpB,CAAC;YACF,OAAO,KAAK,CAAC;QACf,CAAC;QACD,OAAO,IAAI,CAAC;IACd,CAAC,CAAC;IAEF;;;OAGG;IACH,SAAS,GAAG,KAAK,EAAE,IAAY,EAAiB,EAAE;QAChD,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,OAAO,CAAC,gBAAgB,CAAC,IAAI,EAAE,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC;QACvE,IAAI,MAAM,CAAC,WAAW,CAAC,MAAM,GAAG,CAAC;YAC/B,IAAI,CAAC,gBAAgB,CAAC,YAAY,CAAC,MAAM,EAAE,MAAM,CAAC,WAAW,CAAC,CAAC;QACjE,IAAI,MAAM,CAAC,WAAW,CAAC,MAAM,GAAG,CAAC;YAC/B,IAAI,CAAC,gBAAgB,CAAC,YAAY,CAAC,MAAM,EAAE,MAAM,CAAC,WAAW,CAAC,CAAC;IACnE,CAAC,CAAC;CACH","sourcesContent":["import { Mutex } from 'await-semaphore';\nimport { v4 as uuidv4 } from 'uuid';\n\nimport { AbstractLogger, DummyLogger } from '@rosen-bridge/abstract-logger';\nimport JsonBigInt from '@rosen-bridge/json-bigint';\nimport {\n  Transaction,\n  OutputBox,\n  InputExtension,\n  BlockInfo,\n} from '@rosen-bridge/scanner-interfaces';\n\nimport { AbstractExtractor } from '../abstractExtractor';\nimport { AbstractErgoExtractorAction } from './abstractErgoExtractorAction';\nimport { AbstractErgoExtractorEntity } from './abstractErgoExtractorEntity';\nimport {\n  AbstractBoxData,\n  SpendInfo,\n  CallbackType,\n  CallbackMap,\n  CallbackDataMap,\n  TxExtra,\n} from './interfaces';\n\nexport abstract class AbstractErgoExtractor<\n  ExtractedData extends AbstractBoxData,\n  ExtractorEntity extends AbstractErgoExtractorEntity,\n> extends AbstractExtractor<Transaction> {\n  protected abstract actions: AbstractErgoExtractorAction<\n    ExtractedData,\n    ExtractorEntity\n  >;\n  protected logger: AbstractLogger;\n  protected callbacks: {\n    [K in CallbackType]: Map<string, CallbackMap<ExtractedData>[K]>;\n  } = {\n    [CallbackType.Update]: new Map(),\n    [CallbackType.Insert]: new Map(),\n    [CallbackType.Delete]: new Map(),\n    [CallbackType.Spend]: new Map(),\n  };\n  private callbackMutex = new Mutex();\n\n  constructor(logger = new DummyLogger()) {\n    super();\n    this.logger = logger;\n  }\n\n  /**\n   * hook a new callback on a callback type\n   * @param type\n   * @param id\n   * @param callback\n   * @returns callback registered id\n   */\n  hook = async <T extends CallbackType>(\n    type: T,\n    callback: CallbackMap<ExtractedData>[T],\n  ): Promise<string> => {\n    const release = await this.callbackMutex.acquire();\n    const callbackMap = this.callbacks[type];\n    const id = uuidv4();\n    callbackMap.set(id, callback);\n    release();\n    return id;\n  };\n\n  /**\n   * unhook a callback on a type\n   * returns false if there is no registered callback with the id\n   * @param type\n   * @param id\n   * @returns success status\n   */\n  unhook = async (type: CallbackType, id: string): Promise<boolean> => {\n    const release = await this.callbackMutex.acquire();\n    const callbackMap = this.callbacks[type];\n    if (!callbackMap.has(id)) {\n      this.logger.warn(\n        `Callback with Id [${id}] is not registered for type [${type}].`,\n      );\n      return false;\n    }\n    callbackMap.delete(id);\n    release();\n    return true;\n  };\n\n  /**\n   * trigger all callbacks registered on a specific type with the provided data\n   * @param type\n   * @param data\n   */\n  protected triggerCallbacks<T extends CallbackType>(\n    type: T,\n    data: CallbackDataMap<ExtractedData>[T],\n  ): void {\n    const callbackMap = this.callbacks[type];\n    callbackMap.forEach((callback) => {\n      callback(data);\n    });\n  }\n\n  /**\n   * extract box data to proper format (not including spending information)\n   * @param box\n   * @param inputExtensions all input box extensions in transaction\n   * @return extracted data in proper format\n   */\n  abstract extractBoxData: (\n    box: OutputBox,\n    inputExtensions: InputExtension[],\n    txExtra?: TxExtra,\n  ) => ExtractedData | undefined;\n\n  /**\n   * check proper data format in the box\n   * @param box\n   * @return true if the box has the required data and false otherwise\n   */\n  abstract hasData: (box: OutputBox) => boolean;\n\n  /**\n   * create spend info array for the transaction\n   * @param tx\n   * @returns spend info array of the transaction\n   */\n  getTransactionSpendInfo = (tx: Transaction) => {\n    let boxIndex = 1;\n    const spendInfoArray = [];\n    for (const input of tx.inputs) {\n      spendInfoArray.push({ txId: tx.id, boxId: input.boxId, index: boxIndex });\n      boxIndex += 1;\n    }\n    return spendInfoArray;\n  };\n\n  /**\n   * extract transaction extra information\n   * override this function if there is extra needed information\n   * @param tx\n   * @returns\n   */\n  getTransactionExtraData = (\n    tx: Transaction, // eslint-disable-line @typescript-eslint/no-unused-vars\n  ): TxExtra => {\n    return {};\n  };\n\n  /**\n   * process a list of transactions in a block and store required information\n   * @param txs list of transactions in the block\n   * @param block\n   * @return true if the process is completed successfully and false otherwise\n   */\n  processTransactions = async (\n    txs: Transaction[],\n    block: BlockInfo,\n  ): Promise<boolean> => {\n    try {\n      const boxes: Array<ExtractedData> = [];\n      const spentInfos: Array<SpendInfo> = [];\n      for (const tx of txs) {\n        const inputExtensions = tx.inputs.map((input) => input.extension || {});\n        for (const output of tx.outputs) {\n          if (!this.hasData(output)) {\n            continue;\n          }\n          this.logger.debug(`Trying to extract data from box ${output.boxId}`);\n          const extractedData = this.extractBoxData(\n            output,\n            inputExtensions,\n            this.getTransactionExtraData(tx),\n          );\n          if (extractedData) {\n            this.logger.debug(\n              `Extracted data ${JsonBigInt.stringify(extractedData)} from box ${\n                output.boxId\n              }`,\n            );\n            boxes.push(extractedData);\n          }\n        }\n        spentInfos.push(...this.getTransactionSpendInfo(tx));\n      }\n\n      if (boxes.length > 0) {\n        if (!(await this.actions.storeBoxes(boxes, block, this.getId()))) {\n          this.logger.warn(\n            `Data insertion failed for ${this.getId()} at the block ${\n              block.height\n            }`,\n          );\n          return false;\n        }\n        this.triggerCallbacks(CallbackType.Insert, boxes);\n      }\n      const spentData = await this.actions.spendBoxes(\n        spentInfos,\n        block,\n        this.getId(),\n      );\n      if (spentData.length > 0) {\n        this.triggerCallbacks(CallbackType.Spend, spentData);\n      }\n    } catch (e) {\n      this.logger.error(\n        `Processing transactions failed for ${this.getId()} at the block ${\n          block.height\n        } with error: ${e}`,\n      );\n      return false;\n    }\n    return true;\n  };\n\n  /**\n   * fork one block and remove all stored information for this block\n   * @param hash block hash\n   */\n  forkBlock = async (hash: string): Promise<void> => {\n    const result = await this.actions.deleteBlockBoxes(hash, this.getId());\n    if (result.deletedData.length > 0)\n      this.triggerCallbacks(CallbackType.Delete, result.deletedData);\n    if (result.updatedData.length > 0)\n      this.triggerCallbacks(CallbackType.Update, result.updatedData);\n  };\n}\n"]}
|