@rosen-bridge/abstract-extractor 2.0.1 → 2.0.3-52fc0239

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (115) hide show
  1. package/CHANGELOG.md +14 -0
  2. package/dist/ergo/AbstractErgoExtractor.d.ts.map +1 -1
  3. package/dist/ergo/AbstractErgoExtractor.js +3 -2
  4. package/dist/ergo/AbstractErgoExtractorAction.d.ts +1 -1
  5. package/dist/ergo/AbstractErgoExtractorAction.d.ts.map +1 -1
  6. package/dist/ergo/AbstractErgoExtractorAction.js +23 -9
  7. package/dist/ergo/AbstractErgoExtractorEntity.js +2 -2
  8. package/dist/ergo/initializable/AbstractInitializableAction.d.ts +1 -1
  9. package/dist/ergo/initializable/AbstractInitializableAction.d.ts.map +1 -1
  10. package/dist/ergo/initializable/AbstractInitializableAction.js +4 -2
  11. package/dist/lib/abstractExtractor.d.ts +25 -0
  12. package/dist/lib/abstractExtractor.d.ts.map +1 -0
  13. package/dist/lib/abstractExtractor.js +3 -0
  14. package/dist/lib/constants.d.ts +4 -0
  15. package/dist/lib/constants.d.ts.map +1 -0
  16. package/dist/lib/constants.js +4 -0
  17. package/dist/lib/ergo/AbstractErgoExtractor.d.ts +80 -0
  18. package/dist/lib/ergo/AbstractErgoExtractor.d.ts.map +1 -0
  19. package/dist/lib/ergo/AbstractErgoExtractor.js +143 -0
  20. package/dist/lib/ergo/AbstractErgoExtractorAction.d.ts +87 -0
  21. package/dist/lib/ergo/AbstractErgoExtractorAction.d.ts.map +1 -0
  22. package/dist/lib/ergo/AbstractErgoExtractorAction.js +222 -0
  23. package/dist/lib/ergo/AbstractErgoExtractorEntity.d.ts +11 -0
  24. package/dist/lib/ergo/AbstractErgoExtractorEntity.d.ts.map +1 -0
  25. package/dist/lib/ergo/AbstractErgoExtractorEntity.js +57 -0
  26. package/dist/lib/ergo/index.d.ts +10 -0
  27. package/dist/lib/ergo/index.d.ts.map +1 -0
  28. package/dist/lib/ergo/index.js +10 -0
  29. package/dist/lib/ergo/initializable/AbstractInitializable.d.ts +48 -0
  30. package/dist/lib/ergo/initializable/AbstractInitializable.d.ts.map +1 -0
  31. package/dist/lib/ergo/initializable/AbstractInitializable.js +163 -0
  32. package/dist/lib/ergo/initializable/AbstractInitializableAction.d.ts +14 -0
  33. package/dist/lib/ergo/initializable/AbstractInitializableAction.d.ts.map +1 -0
  34. package/dist/lib/ergo/initializable/AbstractInitializableAction.js +16 -0
  35. package/dist/lib/ergo/initializable/index.d.ts +3 -0
  36. package/dist/lib/ergo/initializable/index.d.ts.map +1 -0
  37. package/dist/lib/ergo/initializable/index.js +3 -0
  38. package/dist/lib/ergo/interfaces.d.ts +47 -0
  39. package/dist/lib/ergo/interfaces.d.ts.map +1 -0
  40. package/dist/lib/ergo/interfaces.js +8 -0
  41. package/dist/lib/ergo/network/AbstractNetwork.d.ts +26 -0
  42. package/dist/lib/ergo/network/AbstractNetwork.d.ts.map +1 -0
  43. package/dist/lib/ergo/network/AbstractNetwork.js +3 -0
  44. package/dist/lib/ergo/network/ExplorerNetwork.d.ts +74 -0
  45. package/dist/lib/ergo/network/ExplorerNetwork.d.ts.map +1 -0
  46. package/dist/lib/ergo/network/ExplorerNetwork.js +185 -0
  47. package/dist/lib/ergo/network/NodeNetwork.d.ts +60 -0
  48. package/dist/lib/ergo/network/NodeNetwork.d.ts.map +1 -0
  49. package/dist/lib/ergo/network/NodeNetwork.js +131 -0
  50. package/dist/lib/ergo/utils.d.ts +8 -0
  51. package/dist/lib/ergo/utils.d.ts.map +1 -0
  52. package/dist/lib/ergo/utils.js +16 -0
  53. package/dist/lib/index.d.ts +4 -0
  54. package/dist/lib/index.d.ts.map +1 -0
  55. package/dist/lib/index.js +4 -0
  56. package/dist/tests/AbstractErgoExtractor.mock.d.ts +11 -0
  57. package/dist/tests/AbstractErgoExtractor.mock.d.ts.map +1 -0
  58. package/dist/tests/AbstractErgoExtractor.mock.js +13 -0
  59. package/dist/tests/AbstractErgoExtractor.spec.d.ts +2 -0
  60. package/dist/tests/AbstractErgoExtractor.spec.d.ts.map +1 -0
  61. package/dist/tests/AbstractErgoExtractor.spec.js +241 -0
  62. package/dist/tests/AbstractErgoExtractorAction.mock.d.ts +15 -0
  63. package/dist/tests/AbstractErgoExtractorAction.mock.d.ts.map +1 -0
  64. package/dist/tests/AbstractErgoExtractorAction.mock.js +27 -0
  65. package/dist/tests/AbstractErgoExtractorAction.spec.d.ts +2 -0
  66. package/dist/tests/AbstractErgoExtractorAction.spec.d.ts.map +1 -0
  67. package/dist/tests/AbstractErgoExtractorAction.spec.js +222 -0
  68. package/dist/tests/initializable/AbstractInitializable.mock.d.ts +60 -0
  69. package/dist/tests/initializable/AbstractInitializable.mock.d.ts.map +1 -0
  70. package/dist/tests/initializable/AbstractInitializable.mock.js +22 -0
  71. package/dist/tests/initializable/AbstractInitializable.spec.d.ts +2 -0
  72. package/dist/tests/initializable/AbstractInitializable.spec.d.ts.map +1 -0
  73. package/dist/tests/initializable/AbstractInitializable.spec.js +300 -0
  74. package/dist/tests/initializable/AbstractInitializableAction.mock.d.ts +15 -0
  75. package/dist/tests/initializable/AbstractInitializableAction.mock.d.ts.map +1 -0
  76. package/dist/tests/initializable/AbstractInitializableAction.mock.js +27 -0
  77. package/dist/tests/initializable/AbstractInitializableAction.spec.d.ts +2 -0
  78. package/dist/tests/initializable/AbstractInitializableAction.spec.d.ts.map +1 -0
  79. package/dist/tests/initializable/AbstractInitializableAction.spec.js +55 -0
  80. package/dist/tests/initializable/testData.d.ts +94 -0
  81. package/dist/tests/initializable/testData.d.ts.map +1 -0
  82. package/dist/tests/initializable/testData.js +235 -0
  83. package/dist/tests/network/ExplorerNetwork.spec.d.ts +2 -0
  84. package/dist/tests/network/ExplorerNetwork.spec.d.ts.map +1 -0
  85. package/dist/tests/network/ExplorerNetwork.spec.js +62 -0
  86. package/dist/tests/network/NodeNetwork.spec.d.ts +2 -0
  87. package/dist/tests/network/NodeNetwork.spec.d.ts.map +1 -0
  88. package/dist/tests/network/NodeNetwork.spec.js +49 -0
  89. package/dist/tests/network/testData.d.ts +752 -0
  90. package/dist/tests/network/testData.d.ts.map +1 -0
  91. package/dist/tests/network/testData.js +1401 -0
  92. package/dist/tests/testData.d.ts +32 -0
  93. package/dist/tests/testData.d.ts.map +1 -0
  94. package/dist/tests/testData.js +121 -0
  95. package/dist/tests/testUtils.d.ts +9 -0
  96. package/dist/tests/testUtils.d.ts.map +1 -0
  97. package/dist/tests/testUtils.js +31 -0
  98. package/dist/tsconfig.tsbuildinfo +1 -0
  99. package/dist/vitest.config.d.ts +3 -0
  100. package/dist/vitest.config.d.ts.map +1 -0
  101. package/dist/vitest.config.js +17 -0
  102. package/lib/ergo/AbstractErgoExtractor.ts +3 -1
  103. package/lib/ergo/AbstractErgoExtractorAction.ts +47 -36
  104. package/lib/ergo/AbstractErgoExtractorEntity.ts +1 -1
  105. package/lib/ergo/initializable/AbstractInitializableAction.ts +4 -2
  106. package/package.json +9 -14
  107. package/tests/AbstractErgoExtractor.mock.ts +4 -2
  108. package/tests/AbstractErgoExtractorAction.mock.ts +1 -1
  109. package/tests/AbstractErgoExtractorAction.spec.ts +1 -1
  110. package/tests/initializable/AbstractInitializable.mock.ts +13 -4
  111. package/tests/initializable/AbstractInitializable.spec.ts +1 -9
  112. package/tests/initializable/AbstractInitializableAction.mock.ts +1 -1
  113. package/tests/initializable/AbstractInitializableAction.spec.ts +1 -1
  114. package/tests/testUtils.ts +1 -1
  115. package/tsconfig.build.tsbuildinfo +1 -1
@@ -0,0 +1,185 @@
1
+ import ergoExplorerClientFactory from '@rosen-clients/ergo-explorer';
2
+ import { AbstractNetwork } from './AbstractNetwork';
3
+ import { mapValues, pick } from 'lodash-es';
4
+ import { API_LIMIT } from '../../constants';
5
+ export class ExplorerNetwork extends AbstractNetwork {
6
+ api;
7
+ constructor(url) {
8
+ super();
9
+ this.api = ergoExplorerClientFactory(url);
10
+ }
11
+ /**
12
+ * return spending information of a specified box by having spendTxId
13
+ * @param boxId
14
+ * @param spendTxId
15
+ */
16
+ getSpendingInfo = async (boxId, spendTxId) => {
17
+ const tx = await this.api.v1.getApiV1TransactionsP1(spendTxId);
18
+ const spendIndex = tx.inputs?.findIndex((box) => box.boxId === boxId);
19
+ if (spendIndex == undefined)
20
+ throw Error(`Impossible behavior, the box [${boxId}] should have been spent in tx [${spendTxId}]`);
21
+ return {
22
+ hash: tx.blockId,
23
+ height: tx.inclusionHeight,
24
+ spendIndex,
25
+ };
26
+ };
27
+ /**
28
+ * convert explorer api boxes to ErgoBox interface
29
+ * @param box
30
+ * @returns ErgoBox
31
+ */
32
+ convertBox = async (box) => {
33
+ const spendInfo = box.spentTransactionId
34
+ ? await this.getSpendingInfo(box.boxId, box.spentTransactionId)
35
+ : undefined;
36
+ return {
37
+ blockId: box.blockId,
38
+ boxId: box.boxId,
39
+ creationHeight: box.creationHeight,
40
+ inclusionHeight: box.settlementHeight,
41
+ ergoTree: box.ergoTree,
42
+ index: box.index,
43
+ transactionId: box.transactionId,
44
+ value: box.value,
45
+ additionalRegisters: mapValues(box.additionalRegisters, 'serializedValue'),
46
+ assets: box.assets?.map((asset) => pick(asset, ['tokenId', 'amount'])) ?? [],
47
+ spentHeight: spendInfo?.height,
48
+ spentBlockId: spendInfo?.hash,
49
+ spentTransactionId: box.spentTransactionId,
50
+ spentIndex: spendInfo?.spendIndex,
51
+ };
52
+ };
53
+ /**
54
+ * convert explorer transaction to extractor transaction type
55
+ * @param tx
56
+ */
57
+ convertTransaction = (tx) => {
58
+ return {
59
+ id: tx.id,
60
+ inclusionHeight: tx.inclusionHeight,
61
+ blockId: tx.blockId,
62
+ dataInputs: tx.dataInputs?.map((dataInput) => ({
63
+ boxId: dataInput.boxId,
64
+ })) ?? [],
65
+ // TODO: Add input extension to explorer local/ergo/rosen-bridge/scanner/-/issues/156
66
+ inputs: tx.inputs?.map((input) => ({ boxId: input.boxId })) ?? [],
67
+ outputs: tx.outputs?.map((output) => ({
68
+ boxId: output.boxId,
69
+ transactionId: output.transactionId,
70
+ additionalRegisters: mapValues(output.additionalRegisters, 'serializedValue'),
71
+ assets: output.assets?.map((asset) => pick(asset, ['tokenId', 'amount'])) ??
72
+ [],
73
+ ergoTree: output.ergoTree,
74
+ creationHeight: output.creationHeight,
75
+ index: output.index,
76
+ value: output.value,
77
+ })) ?? [],
78
+ };
79
+ };
80
+ /**
81
+ * convert explorer block transaction to transaction type
82
+ * @param tx
83
+ */
84
+ convertBlockTransaction = (tx) => {
85
+ return {
86
+ id: tx.id,
87
+ dataInputs: tx.dataInputs?.map((dataInput) => ({
88
+ boxId: dataInput.id,
89
+ })) ?? [],
90
+ // TODO: Add input extension local/ergo/rosen-bridge/scanner/-/issues/156
91
+ inputs: tx.inputs?.map((input) => ({ boxId: input.id })) ?? [],
92
+ outputs: tx.outputs?.map((output) => ({
93
+ boxId: output.id,
94
+ transactionId: output.txId,
95
+ additionalRegisters: output.additionalRegisters,
96
+ assets: output.assets?.map((asset) => pick(asset, ['tokenId', 'amount'])) ??
97
+ [],
98
+ ergoTree: output.ergoTree,
99
+ creationHeight: output.creationHeight,
100
+ index: output.index,
101
+ value: output.value,
102
+ })) ?? [],
103
+ };
104
+ };
105
+ /**
106
+ * use explorer api to return related transactions of the specified address in the height range
107
+ * @param tokenId
108
+ * @param offset
109
+ * @param limit
110
+ * @returns related transactions
111
+ */
112
+ getAddressTransactionsWithHeight = async (address, fromHeight, toHeight) => {
113
+ const txs = await this.api.v1.getApiV1AddressesP1Transactions(address, {
114
+ fromHeight,
115
+ toHeight,
116
+ limit: API_LIMIT,
117
+ });
118
+ if (!txs.items)
119
+ throw new Error('Explorer AddressTransactions api expected to have items');
120
+ return txs.items.map((tx) => this.convertTransaction(tx));
121
+ };
122
+ /**
123
+ * use explorer api to get the block id at the specified height
124
+ * @param height
125
+ * @returns block id
126
+ */
127
+ getBlockIdAtHeight = async (height) => {
128
+ const id = await this.api.v0.getApiV0BlocksAtP1(height);
129
+ return id[0];
130
+ };
131
+ /**
132
+ * use explorer api to return all transactions in a block
133
+ * @param blockId
134
+ * @returns converted transactions
135
+ */
136
+ getBlockTxs = async (blockId) => {
137
+ const block = await this.api.v1.getApiV1BlocksP1(blockId);
138
+ if (!block.block.blockTransactions) {
139
+ throw new Error(`Expected explorer block api to include block transactions for block ${blockId}`);
140
+ }
141
+ return block.block.blockTransactions.map((tx) => this.convertBlockTransaction(tx));
142
+ };
143
+ /**
144
+ * use explorer api to return related boxes by specified address
145
+ * @param address
146
+ * @param offset
147
+ * @param limit
148
+ * @returns related boxes
149
+ */
150
+ getBoxesByAddress = async (address, offset, limit) => {
151
+ const boxes = await this.api.v1.getApiV1BoxesUnspentByaddressP1(address, {
152
+ offset: offset,
153
+ limit: limit,
154
+ sortDirection: 'desc',
155
+ });
156
+ if (!boxes.items)
157
+ throw new Error('Explorer BoxesByAddress api expected to have items');
158
+ const resultBoxes = [];
159
+ for (const box of boxes.items) {
160
+ resultBoxes.push(await this.convertBox(box));
161
+ }
162
+ return { boxes: resultBoxes, hasNextBatch: boxes.total > offset + limit };
163
+ };
164
+ /**
165
+ * use explorer api to return related boxes by specified token id
166
+ * @param tokenId
167
+ * @param offset
168
+ * @param limit
169
+ * @returns related boxes
170
+ */
171
+ getBoxesByTokenId = async (tokenId, offset, limit) => {
172
+ const boxes = await this.api.v1.getApiV1BoxesBytokenidP1(tokenId, {
173
+ offset: offset,
174
+ limit: limit,
175
+ });
176
+ if (!boxes.items)
177
+ throw new Error('Explorer BoxesByTokeId api expected to have items');
178
+ const resultBoxes = [];
179
+ for (const box of boxes.items) {
180
+ resultBoxes.push(await this.convertBox(box));
181
+ }
182
+ return { boxes: resultBoxes, hasNextBatch: boxes.total > offset + limit };
183
+ };
184
+ }
185
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ExplorerNetwork.js","sourceRoot":"","sources":["../../../../lib/ergo/network/ExplorerNetwork.ts"],"names":[],"mappings":"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","sourcesContent":["import ergoExplorerClientFactory from '@rosen-clients/ergo-explorer';\nimport { AbstractNetwork } from './AbstractNetwork';\nimport { V1 } from '@rosen-clients/ergo-explorer';\nimport { BlockInfo, Transaction } from '@rosen-bridge/scanner-interfaces';\n\nimport { ErgoBox, ExtendedTransaction } from '../interfaces';\nimport { mapValues, pick } from 'lodash-es';\nimport { API_LIMIT } from '../../constants';\n\nexport class ExplorerNetwork extends AbstractNetwork {\n  private api;\n\n  constructor(url: string) {\n    super();\n    this.api = ergoExplorerClientFactory(url);\n  }\n\n  /**\n   * return spending information of a specified box by having spendTxId\n   * @param boxId\n   * @param spendTxId\n   */\n  getSpendingInfo = async (\n    boxId: string,\n    spendTxId: string\n  ): Promise<BlockInfo & { spendIndex: number }> => {\n    const tx = await this.api.v1.getApiV1TransactionsP1(spendTxId);\n    const spendIndex = tx.inputs?.findIndex((box) => box.boxId === boxId);\n    if (spendIndex == undefined)\n      throw Error(\n        `Impossible behavior, the box [${boxId}] should have been spent in tx [${spendTxId}]`\n      );\n    return {\n      hash: tx.blockId,\n      height: tx.inclusionHeight,\n      spendIndex,\n    };\n  };\n\n  /**\n   * convert explorer api boxes to ErgoBox interface\n   * @param box\n   * @returns ErgoBox\n   */\n  private convertBox = async (box: V1.OutputInfo): Promise<ErgoBox> => {\n    const spendInfo = box.spentTransactionId\n      ? await this.getSpendingInfo(box.boxId, box.spentTransactionId)\n      : undefined;\n    return {\n      blockId: box.blockId,\n      boxId: box.boxId,\n      creationHeight: box.creationHeight,\n      inclusionHeight: box.settlementHeight,\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      spentHeight: spendInfo?.height,\n      spentBlockId: spendInfo?.hash,\n      spentTransactionId: box.spentTransactionId,\n      spentIndex: spendInfo?.spendIndex,\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) => ({ boxId: input.boxId })) ?? [],\n      outputs:\n        tx.outputs?.map((output) => ({\n          boxId: output.boxId,\n          transactionId: output.transactionId,\n          additionalRegisters: mapValues(\n            output.additionalRegisters,\n            'serializedValue'\n          ),\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   * 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  ): Promise<Array<ExtendedTransaction>> => {\n    const txs = await this.api.v1.getApiV1AddressesP1Transactions(address, {\n      fromHeight,\n      toHeight,\n      limit: API_LIMIT,\n    });\n    if (!txs.items)\n      throw new Error(\n        'Explorer AddressTransactions api expected to have items'\n      );\n    return txs.items.map((tx) => this.convertTransaction(tx));\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  /**\n   * use explorer api to return related boxes by specified address\n   * @param address\n   * @param offset\n   * @param limit\n   * @returns related boxes\n   */\n  getBoxesByAddress = async (\n    address: string,\n    offset: number,\n    limit: number\n  ): Promise<{ boxes: ErgoBox[]; hasNextBatch: boolean }> => {\n    const boxes = await this.api.v1.getApiV1BoxesUnspentByaddressP1(address, {\n      offset: offset,\n      limit: limit,\n      sortDirection: 'desc',\n    });\n    if (!boxes.items)\n      throw new Error('Explorer BoxesByAddress api expected to have items');\n    const resultBoxes: Array<ErgoBox> = [];\n    for (const box of boxes.items) {\n      resultBoxes.push(await this.convertBox(box));\n    }\n    return { boxes: resultBoxes, hasNextBatch: boxes.total > offset + limit };\n  };\n\n  /**\n   * use explorer api to return related boxes by specified token id\n   * @param tokenId\n   * @param offset\n   * @param limit\n   * @returns related boxes\n   */\n  getBoxesByTokenId = async (\n    tokenId: string,\n    offset: number,\n    limit: number\n  ): Promise<{ boxes: ErgoBox[]; hasNextBatch: boolean }> => {\n    const boxes = await this.api.v1.getApiV1BoxesBytokenidP1(tokenId, {\n      offset: offset,\n      limit: limit,\n    });\n    if (!boxes.items)\n      throw new Error('Explorer BoxesByTokeId api expected to have items');\n    const resultBoxes: Array<ErgoBox> = [];\n    for (const box of boxes.items) {\n      resultBoxes.push(await this.convertBox(box));\n    }\n    return { boxes: resultBoxes, hasNextBatch: boxes.total > offset + limit };\n  };\n}\n"]}
@@ -0,0 +1,60 @@
1
+ import { BlockInfo } from '@rosen-bridge/scanner-interfaces';
2
+ import { ErgoBox, ExtendedTransaction } from '../interfaces';
3
+ import { AbstractNetwork } from './AbstractNetwork';
4
+ export declare class NodeNetwork extends AbstractNetwork {
5
+ private api;
6
+ constructor(url: string);
7
+ /**
8
+ * convert node api boxes to ErgoBox interface
9
+ * @param box
10
+ * @returns ErgoBox
11
+ */
12
+ private convertBox;
13
+ /**
14
+ * convert Node transaction to extractor transaction type
15
+ * @param tx
16
+ */
17
+ private convertTransaction;
18
+ /**
19
+ * return spending information of a specified box by having spendTxId
20
+ * @param boxId
21
+ * @param spendTxId
22
+ */
23
+ getSpendingInfo: (boxId: string, spendTxId: string) => Promise<BlockInfo & {
24
+ spendIndex: number;
25
+ }>;
26
+ /**
27
+ * use node api to return related boxes by specified address
28
+ * @param address
29
+ * @param offset
30
+ * @param limit
31
+ * @returns related boxes
32
+ */
33
+ getBoxesByAddress: (address: string, offset: number, limit: number) => Promise<{
34
+ boxes: ErgoBox[];
35
+ hasNextBatch: boolean;
36
+ }>;
37
+ /**
38
+ * use node api to return related boxes by specified token id
39
+ * @param tokenId
40
+ * @param offset
41
+ * @param limit
42
+ * @returns related boxes
43
+ */
44
+ getBoxesByTokenId: (tokenId: string, offset: number, limit: number) => Promise<{
45
+ boxes: ErgoBox[];
46
+ hasNextBatch: boolean;
47
+ }>;
48
+ /**
49
+ * use node api to return related transactions of the specified address with limit offset
50
+ * @param tokenId
51
+ * @param offset
52
+ * @param limit
53
+ * @returns related transactions
54
+ */
55
+ getAddressTransactionsWithOffsetLimit: (address: string, offset: number, limit: number) => Promise<{
56
+ items: Array<ExtendedTransaction>;
57
+ total: number;
58
+ }>;
59
+ }
60
+ //# sourceMappingURL=NodeNetwork.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"NodeNetwork.d.ts","sourceRoot":"","sources":["../../../../lib/ergo/network/NodeNetwork.ts"],"names":[],"mappings":"AAIA,OAAO,EAAE,SAAS,EAAE,MAAM,kCAAkC,CAAC;AAE7D,OAAO,EAAE,OAAO,EAAE,mBAAmB,EAAE,MAAM,eAAe,CAAC;AAC7D,OAAO,EAAE,eAAe,EAAE,MAAM,mBAAmB,CAAC;AAEpD,qBAAa,WAAY,SAAQ,eAAe;IAC9C,OAAO,CAAC,GAAG,CAAC;gBAEA,GAAG,EAAE,MAAM;IAKvB;;;;OAIG;IACH,OAAO,CAAC,UAAU,CAqBhB;IAEF;;;OAGG;IACH,OAAO,CAAC,kBAAkB,CAqBxB;IAEF;;;;OAIG;IACH,eAAe,UACN,MAAM,aACF,MAAM;oBACkB,MAAM;OAYzC;IAEF;;;;;;OAMG;IACH,iBAAiB,YACN,MAAM,UACP,MAAM,SACP,MAAM;eACK,OAAO,EAAE;sBAAgB,OAAO;OAYlD;IAEF;;;;;;OAMG;IACH,iBAAiB,YACN,MAAM,UACP,MAAM,SACP,MAAM;eACK,OAAO,EAAE;sBAAgB,OAAO;OAWlD;IAEF;;;;;;OAMG;IACH,qCAAqC,YAC1B,MAAM,UACP,MAAM,SACP,MAAM,KACZ,QAAQ;QAAE,KAAK,EAAE,MAAM,mBAAmB,CAAC,CAAC;QAAC,KAAK,EAAE,MAAM,CAAA;KAAE,CAAC,CAa9D;CACH"}
@@ -0,0 +1,131 @@
1
+ import ergoNodeClientFactory from '@rosen-clients/ergo-node';
2
+ import { AbstractNetwork } from './AbstractNetwork';
3
+ export class NodeNetwork extends AbstractNetwork {
4
+ api;
5
+ constructor(url) {
6
+ super();
7
+ this.api = ergoNodeClientFactory(url);
8
+ }
9
+ /**
10
+ * convert node api boxes to ErgoBox interface
11
+ * @param box
12
+ * @returns ErgoBox
13
+ */
14
+ convertBox = async (box) => {
15
+ const tx = await await this.api.getTxById(box.transactionId);
16
+ const spendInfo = box.spentTransactionId
17
+ ? await this.getSpendingInfo(box.boxId, box.spentTransactionId)
18
+ : undefined;
19
+ return {
20
+ transactionId: box.transactionId || '',
21
+ index: box.index || 0,
22
+ value: box.value || 0n,
23
+ ergoTree: box.ergoTree || '',
24
+ creationHeight: box.creationHeight || 0,
25
+ inclusionHeight: tx.inclusionHeight,
26
+ assets: box.assets || [],
27
+ additionalRegisters: box.additionalRegisters,
28
+ boxId: box.boxId || '',
29
+ blockId: tx.blockId,
30
+ spentBlockId: spendInfo?.hash,
31
+ spentHeight: spendInfo?.height,
32
+ spentTransactionId: box.spentTransactionId,
33
+ spentIndex: spendInfo?.spendIndex,
34
+ };
35
+ };
36
+ /**
37
+ * convert Node transaction to extractor transaction type
38
+ * @param tx
39
+ */
40
+ convertTransaction = (tx) => {
41
+ return {
42
+ id: tx.id || '',
43
+ inclusionHeight: tx.inclusionHeight,
44
+ blockId: tx.blockId,
45
+ outputs: tx.outputs.map((output) => ({
46
+ transactionId: output.transactionId || '',
47
+ index: output.index || 0,
48
+ value: output.value || 0n,
49
+ ergoTree: output.ergoTree || '',
50
+ creationHeight: output.creationHeight || 0,
51
+ assets: output.assets || [],
52
+ additionalRegisters: output.additionalRegisters,
53
+ boxId: output.boxId || '',
54
+ })),
55
+ // TODO: Add input extension local/ergo/rosen-bridge/scanner/-/issues/156
56
+ inputs: tx.inputs.map((input) => ({ boxId: input.boxId })) ?? [],
57
+ dataInputs: tx.dataInputs,
58
+ };
59
+ };
60
+ /**
61
+ * return spending information of a specified box by having spendTxId
62
+ * @param boxId
63
+ * @param spendTxId
64
+ */
65
+ getSpendingInfo = async (boxId, spendTxId) => {
66
+ const tx = await this.api.getTxById(spendTxId);
67
+ const spendIndex = tx.inputs?.findIndex((box) => box.boxId == boxId);
68
+ if (spendIndex == undefined)
69
+ throw Error(`Impossible behavior, the box [${boxId}] should have been spent in tx [${spendTxId}]`);
70
+ return {
71
+ hash: tx.blockId,
72
+ height: tx.inclusionHeight,
73
+ spendIndex,
74
+ };
75
+ };
76
+ /**
77
+ * use node api to return related boxes by specified address
78
+ * @param address
79
+ * @param offset
80
+ * @param limit
81
+ * @returns related boxes
82
+ */
83
+ getBoxesByAddress = async (address, offset, limit) => {
84
+ const boxes = await this.api.getBoxesByAddressUnspent(address, {
85
+ offset: offset,
86
+ limit: limit,
87
+ sortDirection: 'desc',
88
+ });
89
+ if (!boxes)
90
+ throw new Error('Ergo node BoxesByAddress api expected to have items');
91
+ const ergoBoxes = await Promise.all(boxes.map(async (box) => await this.convertBox(box)));
92
+ return { boxes: ergoBoxes, hasNextBatch: boxes.length > 0 };
93
+ };
94
+ /**
95
+ * use node api to return related boxes by specified token id
96
+ * @param tokenId
97
+ * @param offset
98
+ * @param limit
99
+ * @returns related boxes
100
+ */
101
+ getBoxesByTokenId = async (tokenId, offset, limit) => {
102
+ const boxes = await this.api.getBoxesByTokenId(tokenId, {
103
+ offset: offset,
104
+ limit: limit,
105
+ });
106
+ if (!boxes.items)
107
+ throw new Error('Ergo node BoxesByTokenId api expected to have items');
108
+ const ergoBoxes = await Promise.all(boxes.items.map(async (box) => await this.convertBox(box)));
109
+ return { boxes: ergoBoxes, hasNextBatch: boxes.items.length > 0 };
110
+ };
111
+ /**
112
+ * use node api to return related transactions of the specified address with limit offset
113
+ * @param tokenId
114
+ * @param offset
115
+ * @param limit
116
+ * @returns related transactions
117
+ */
118
+ getAddressTransactionsWithOffsetLimit = async (address, offset, limit) => {
119
+ const txs = await this.api.getTxsByAddress(address, {
120
+ offset,
121
+ limit,
122
+ });
123
+ if (!txs.items)
124
+ throw new Error('Explorer AddressTransactions api expected to have items');
125
+ return {
126
+ items: txs.items.map((tx) => this.convertTransaction(tx)),
127
+ total: txs.total,
128
+ };
129
+ };
130
+ }
131
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"NodeNetwork.js","sourceRoot":"","sources":["../../../../lib/ergo/network/NodeNetwork.ts"],"names":[],"mappings":"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","sourcesContent":["import ergoNodeClientFactory, {\n  IndexedErgoBox,\n  IndexedErgoTransaction,\n} from '@rosen-clients/ergo-node';\nimport { BlockInfo } from '@rosen-bridge/scanner-interfaces';\n\nimport { ErgoBox, ExtendedTransaction } from '../interfaces';\nimport { AbstractNetwork } from './AbstractNetwork';\n\nexport class NodeNetwork extends AbstractNetwork {\n  private api;\n\n  constructor(url: string) {\n    super();\n    this.api = ergoNodeClientFactory(url);\n  }\n\n  /**\n   * convert node api boxes to ErgoBox interface\n   * @param box\n   * @returns ErgoBox\n   */\n  private convertBox = async (box: IndexedErgoBox): Promise<ErgoBox> => {\n    const tx = await await this.api.getTxById(box.transactionId!);\n    const spendInfo = box.spentTransactionId\n      ? await this.getSpendingInfo(box.boxId!, box.spentTransactionId)\n      : undefined;\n    return {\n      transactionId: box.transactionId || '',\n      index: box.index || 0,\n      value: box.value || 0n,\n      ergoTree: box.ergoTree || '',\n      creationHeight: box.creationHeight || 0,\n      inclusionHeight: tx.inclusionHeight,\n      assets: box.assets || [],\n      additionalRegisters: box.additionalRegisters,\n      boxId: box.boxId || '',\n      blockId: tx.blockId,\n      spentBlockId: spendInfo?.hash,\n      spentHeight: spendInfo?.height,\n      spentTransactionId: box.spentTransactionId,\n      spentIndex: spendInfo?.spendIndex,\n    };\n  };\n\n  /**\n   * convert Node transaction to extractor transaction type\n   * @param tx\n   */\n  private convertTransaction = (\n    tx: IndexedErgoTransaction\n  ): ExtendedTransaction => {\n    return {\n      id: tx.id || '',\n      inclusionHeight: tx.inclusionHeight,\n      blockId: tx.blockId,\n      outputs: tx.outputs.map((output) => ({\n        transactionId: output.transactionId || '',\n        index: output.index || 0,\n        value: output.value || 0n,\n        ergoTree: output.ergoTree || '',\n        creationHeight: output.creationHeight || 0,\n        assets: output.assets || [],\n        additionalRegisters: output.additionalRegisters,\n        boxId: output.boxId || '',\n      })),\n      // TODO: Add input extension local/ergo/rosen-bridge/scanner/-/issues/156\n      inputs: tx.inputs.map((input) => ({ boxId: input.boxId! })) ?? [],\n      dataInputs: tx.dataInputs,\n    };\n  };\n\n  /**\n   * return spending information of a specified box by having spendTxId\n   * @param boxId\n   * @param spendTxId\n   */\n  getSpendingInfo = async (\n    boxId: string,\n    spendTxId: string\n  ): Promise<BlockInfo & { spendIndex: number }> => {\n    const tx = await this.api.getTxById(spendTxId);\n    const spendIndex = tx.inputs?.findIndex((box) => box.boxId == boxId);\n    if (spendIndex == undefined)\n      throw Error(\n        `Impossible behavior, the box [${boxId}] should have been spent in tx [${spendTxId}]`\n      );\n    return {\n      hash: tx.blockId,\n      height: tx.inclusionHeight,\n      spendIndex,\n    };\n  };\n\n  /**\n   * use node api to return related boxes by specified address\n   * @param address\n   * @param offset\n   * @param limit\n   * @returns related boxes\n   */\n  getBoxesByAddress = async (\n    address: string,\n    offset: number,\n    limit: number\n  ): Promise<{ boxes: ErgoBox[]; hasNextBatch: boolean }> => {\n    const boxes = await this.api.getBoxesByAddressUnspent(address, {\n      offset: offset,\n      limit: limit,\n      sortDirection: 'desc',\n    });\n    if (!boxes)\n      throw new Error('Ergo node BoxesByAddress api expected to have items');\n    const ergoBoxes = await Promise.all(\n      boxes.map(async (box) => await this.convertBox(box))\n    );\n    return { boxes: ergoBoxes, hasNextBatch: boxes.length > 0 };\n  };\n\n  /**\n   * use node api to return related boxes by specified token id\n   * @param tokenId\n   * @param offset\n   * @param limit\n   * @returns related boxes\n   */\n  getBoxesByTokenId = async (\n    tokenId: string,\n    offset: number,\n    limit: number\n  ): Promise<{ boxes: ErgoBox[]; hasNextBatch: boolean }> => {\n    const boxes = await this.api.getBoxesByTokenId(tokenId, {\n      offset: offset,\n      limit: limit,\n    });\n    if (!boxes.items)\n      throw new Error('Ergo node BoxesByTokenId api expected to have items');\n    const ergoBoxes = await Promise.all(\n      boxes.items.map(async (box) => await this.convertBox(box))\n    );\n    return { boxes: ergoBoxes, hasNextBatch: boxes.items.length > 0 };\n  };\n\n  /**\n   * use node api to return related transactions of the specified address with limit offset\n   * @param tokenId\n   * @param offset\n   * @param limit\n   * @returns related transactions\n   */\n  getAddressTransactionsWithOffsetLimit = async (\n    address: string,\n    offset: number,\n    limit: number\n  ): Promise<{ items: Array<ExtendedTransaction>; total: number }> => {\n    const txs = await this.api.getTxsByAddress(address, {\n      offset,\n      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"]}
@@ -0,0 +1,8 @@
1
+ import { OutputBox } from '@rosen-bridge/scanner-interfaces';
2
+ /**
3
+ * Check box to have specified tokens
4
+ * @param box
5
+ * @return true if box has the required token and false otherwise
6
+ */
7
+ export declare const boxHasToken: (box: OutputBox, tokenIds: string[]) => boolean;
8
+ //# sourceMappingURL=utils.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"utils.d.ts","sourceRoot":"","sources":["../../../lib/ergo/utils.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,SAAS,EAAE,MAAM,kCAAkC,CAAC;AAE7D;;;;GAIG;AACH,eAAO,MAAM,WAAW,QAAS,SAAS,YAAY,MAAM,EAAE,YAM7D,CAAC"}
@@ -0,0 +1,16 @@
1
+ import { intersection } from 'lodash-es';
2
+ /**
3
+ * Check box to have specified tokens
4
+ * @param box
5
+ * @return true if box has the required token and false otherwise
6
+ */
7
+ export const boxHasToken = (box, tokenIds) => {
8
+ if (!box.assets)
9
+ return false;
10
+ const boxTokens = box.assets.map((token) => token.tokenId);
11
+ const requiredTokens = intersection(tokenIds, boxTokens);
12
+ if (requiredTokens.length == tokenIds.length)
13
+ return true;
14
+ return false;
15
+ };
16
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXRpbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9saWIvZXJnby91dGlscy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0sV0FBVyxDQUFDO0FBR3pDOzs7O0dBSUc7QUFDSCxNQUFNLENBQUMsTUFBTSxXQUFXLEdBQUcsQ0FBQyxHQUFjLEVBQUUsUUFBa0IsRUFBRSxFQUFFO0lBQ2hFLElBQUksQ0FBQyxHQUFHLENBQUMsTUFBTTtRQUFFLE9BQU8sS0FBSyxDQUFDO0lBQzlCLE1BQU0sU0FBUyxHQUFHLEdBQUcsQ0FBQyxNQUFNLENBQUMsR0FBRyxDQUFDLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxLQUFLLENBQUMsT0FBTyxDQUFDLENBQUM7SUFDM0QsTUFBTSxjQUFjLEdBQUcsWUFBWSxDQUFDLFFBQVEsRUFBRSxTQUFTLENBQUMsQ0FBQztJQUN6RCxJQUFJLGNBQWMsQ0FBQyxNQUFNLElBQUksUUFBUSxDQUFDLE1BQU07UUFBRSxPQUFPLElBQUksQ0FBQztJQUMxRCxPQUFPLEtBQUssQ0FBQztBQUNmLENBQUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGludGVyc2VjdGlvbiB9IGZyb20gJ2xvZGFzaC1lcyc7XG5pbXBvcnQgeyBPdXRwdXRCb3ggfSBmcm9tICdAcm9zZW4tYnJpZGdlL3NjYW5uZXItaW50ZXJmYWNlcyc7XG5cbi8qKlxuICogQ2hlY2sgYm94IHRvIGhhdmUgc3BlY2lmaWVkIHRva2Vuc1xuICogQHBhcmFtIGJveFxuICogQHJldHVybiB0cnVlIGlmIGJveCBoYXMgdGhlIHJlcXVpcmVkIHRva2VuIGFuZCBmYWxzZSBvdGhlcndpc2VcbiAqL1xuZXhwb3J0IGNvbnN0IGJveEhhc1Rva2VuID0gKGJveDogT3V0cHV0Qm94LCB0b2tlbklkczogc3RyaW5nW10pID0+IHtcbiAgaWYgKCFib3guYXNzZXRzKSByZXR1cm4gZmFsc2U7XG4gIGNvbnN0IGJveFRva2VucyA9IGJveC5hc3NldHMubWFwKCh0b2tlbikgPT4gdG9rZW4udG9rZW5JZCk7XG4gIGNvbnN0IHJlcXVpcmVkVG9rZW5zID0gaW50ZXJzZWN0aW9uKHRva2VuSWRzLCBib3hUb2tlbnMpO1xuICBpZiAocmVxdWlyZWRUb2tlbnMubGVuZ3RoID09IHRva2VuSWRzLmxlbmd0aCkgcmV0dXJuIHRydWU7XG4gIHJldHVybiBmYWxzZTtcbn07XG4iXX0=
@@ -0,0 +1,4 @@
1
+ export * from './ergo';
2
+ export * from './constants';
3
+ export * from './abstractExtractor';
4
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../lib/index.ts"],"names":[],"mappings":"AAAA,cAAc,QAAQ,CAAC;AACvB,cAAc,aAAa,CAAC;AAC5B,cAAc,qBAAqB,CAAC"}
@@ -0,0 +1,4 @@
1
+ export * from './ergo';
2
+ export * from './constants';
3
+ export * from './abstractExtractor';
4
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9saWIvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsY0FBYyxRQUFRLENBQUM7QUFDdkIsY0FBYyxhQUFhLENBQUM7QUFDNUIsY0FBYyxxQkFBcUIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vZXJnbyc7XG5leHBvcnQgKiBmcm9tICcuL2NvbnN0YW50cyc7XG5leHBvcnQgKiBmcm9tICcuL2Fic3RyYWN0RXh0cmFjdG9yJztcbiJdfQ==
@@ -0,0 +1,11 @@
1
+ import { V1 } from '@rosen-clients/ergo-explorer';
2
+ import { BlockInfo, OutputBox } from '@rosen-bridge/scanner-interfaces';
3
+ import { AbstractErgoExtractor, AbstractBoxData, AbstractErgoExtractorAction, AbstractErgoExtractorEntity } from '../lib';
4
+ export declare class MockedErgoExtractor extends AbstractErgoExtractor<AbstractBoxData, AbstractErgoExtractorEntity> {
5
+ actions: AbstractErgoExtractorAction<AbstractBoxData, AbstractErgoExtractorEntity>;
6
+ getId: () => string;
7
+ initializeBoxes: (initialBlock: BlockInfo) => Promise<void>;
8
+ hasData: (box: V1.OutputInfo | OutputBox) => boolean;
9
+ extractBoxData: (box: V1.OutputInfo | OutputBox) => AbstractBoxData | undefined;
10
+ }
11
+ //# sourceMappingURL=AbstractErgoExtractor.mock.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"AbstractErgoExtractor.mock.d.ts","sourceRoot":"","sources":["../../tests/AbstractErgoExtractor.mock.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,EAAE,EAAE,MAAM,8BAA8B,CAAC;AAClD,OAAO,EAAE,SAAS,EAAE,SAAS,EAAE,MAAM,kCAAkC,CAAC;AAExE,OAAO,EACL,qBAAqB,EACrB,eAAe,EACf,2BAA2B,EAC3B,2BAA2B,EAC5B,MAAM,QAAQ,CAAC;AAEhB,qBAAa,mBAAoB,SAAQ,qBAAqB,CAC5D,eAAe,EACf,2BAA2B,CAC5B;IACC,OAAO,EAAE,2BAA2B,CAClC,eAAe,EACf,2BAA2B,CAC5B,CAAC;IAEF,KAAK,eAAgB;IAErB,eAAe,EAAE,CAAC,YAAY,EAAE,SAAS,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;IAE5D,OAAO,QACA,GAAG,UAAU,GAAG,SAAS,aACrB;IAEX,cAAc,QACP,GAAG,UAAU,GAAG,SAAS,KAC7B,eAAe,GAAG,SAAS,CAE5B;CACH"}
@@ -0,0 +1,13 @@
1
+ import { AbstractErgoExtractor, } from '../lib';
2
+ export class MockedErgoExtractor extends AbstractErgoExtractor {
3
+ actions;
4
+ getId = () => 'Test';
5
+ initializeBoxes;
6
+ hasData = (box // eslint-disable-line @typescript-eslint/no-unused-vars
7
+ ) => false;
8
+ extractBoxData = (box // eslint-disable-line @typescript-eslint/no-unused-vars
9
+ ) => {
10
+ return undefined;
11
+ };
12
+ }
13
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiQWJzdHJhY3RFcmdvRXh0cmFjdG9yLm1vY2suanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi90ZXN0cy9BYnN0cmFjdEVyZ29FeHRyYWN0b3IubW9jay50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFHQSxPQUFPLEVBQ0wscUJBQXFCLEdBSXRCLE1BQU0sUUFBUSxDQUFDO0FBRWhCLE1BQU0sT0FBTyxtQkFBb0IsU0FBUSxxQkFHeEM7SUFDQyxPQUFPLENBR0w7SUFFRixLQUFLLEdBQUcsR0FBRyxFQUFFLENBQUMsTUFBTSxDQUFDO0lBRXJCLGVBQWUsQ0FBNkM7SUFFNUQsT0FBTyxHQUFHLENBQ1IsR0FBOEIsQ0FBQyx3REFBd0Q7TUFDdkYsRUFBRSxDQUFDLEtBQUssQ0FBQztJQUVYLGNBQWMsR0FBRyxDQUNmLEdBQThCLENBQUMsd0RBQXdEO01BQzFELEVBQUU7UUFDL0IsT0FBTyxTQUFTLENBQUM7SUFDbkIsQ0FBQyxDQUFDO0NBQ0giLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBWMSB9IGZyb20gJ0Byb3Nlbi1jbGllbnRzL2VyZ28tZXhwbG9yZXInO1xuaW1wb3J0IHsgQmxvY2tJbmZvLCBPdXRwdXRCb3ggfSBmcm9tICdAcm9zZW4tYnJpZGdlL3NjYW5uZXItaW50ZXJmYWNlcyc7XG5cbmltcG9ydCB7XG4gIEFic3RyYWN0RXJnb0V4dHJhY3RvcixcbiAgQWJzdHJhY3RCb3hEYXRhLFxuICBBYnN0cmFjdEVyZ29FeHRyYWN0b3JBY3Rpb24sXG4gIEFic3RyYWN0RXJnb0V4dHJhY3RvckVudGl0eSxcbn0gZnJvbSAnLi4vbGliJztcblxuZXhwb3J0IGNsYXNzIE1vY2tlZEVyZ29FeHRyYWN0b3IgZXh0ZW5kcyBBYnN0cmFjdEVyZ29FeHRyYWN0b3I8XG4gIEFic3RyYWN0Qm94RGF0YSxcbiAgQWJzdHJhY3RFcmdvRXh0cmFjdG9yRW50aXR5XG4+IHtcbiAgYWN0aW9uczogQWJzdHJhY3RFcmdvRXh0cmFjdG9yQWN0aW9uPFxuICAgIEFic3RyYWN0Qm94RGF0YSxcbiAgICBBYnN0cmFjdEVyZ29FeHRyYWN0b3JFbnRpdHlcbiAgPjtcblxuICBnZXRJZCA9ICgpID0+ICdUZXN0JztcblxuICBpbml0aWFsaXplQm94ZXM6IChpbml0aWFsQmxvY2s6IEJsb2NrSW5mbykgPT4gUHJvbWlzZTx2b2lkPjtcblxuICBoYXNEYXRhID0gKFxuICAgIGJveDogVjEuT3V0cHV0SW5mbyB8IE91dHB1dEJveCAvLyBlc2xpbnQtZGlzYWJsZS1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uby11bnVzZWQtdmFyc1xuICApID0+IGZhbHNlO1xuXG4gIGV4dHJhY3RCb3hEYXRhID0gKFxuICAgIGJveDogVjEuT3V0cHV0SW5mbyB8IE91dHB1dEJveCAvLyBlc2xpbnQtZGlzYWJsZS1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uby11bnVzZWQtdmFyc1xuICApOiBBYnN0cmFjdEJveERhdGEgfCB1bmRlZmluZWQgPT4ge1xuICAgIHJldHVybiB1bmRlZmluZWQ7XG4gIH07XG59XG4iXX0=
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=AbstractErgoExtractor.spec.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"AbstractErgoExtractor.spec.d.ts","sourceRoot":"","sources":["../../tests/AbstractErgoExtractor.spec.ts"],"names":[],"mappings":""}