@rosen-bridge/abstract-extractor 2.1.0 → 2.1.1

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 (127) hide show
  1. package/CHANGELOG.md +11 -0
  2. package/package.json +14 -7
  3. package/.eslintignore +0 -1
  4. package/dist/lib/abstractExtractor.d.ts +0 -25
  5. package/dist/lib/abstractExtractor.d.ts.map +0 -1
  6. package/dist/lib/abstractExtractor.js +0 -3
  7. package/dist/lib/constants.d.ts +0 -4
  8. package/dist/lib/constants.d.ts.map +0 -1
  9. package/dist/lib/constants.js +0 -4
  10. package/dist/lib/ergo/AbstractErgoExtractor.d.ts +0 -80
  11. package/dist/lib/ergo/AbstractErgoExtractor.d.ts.map +0 -1
  12. package/dist/lib/ergo/AbstractErgoExtractor.js +0 -142
  13. package/dist/lib/ergo/AbstractErgoExtractorAction.d.ts +0 -92
  14. package/dist/lib/ergo/AbstractErgoExtractorAction.d.ts.map +0 -1
  15. package/dist/lib/ergo/AbstractErgoExtractorAction.js +0 -222
  16. package/dist/lib/ergo/AbstractErgoExtractorEntity.d.ts +0 -11
  17. package/dist/lib/ergo/AbstractErgoExtractorEntity.d.ts.map +0 -1
  18. package/dist/lib/ergo/AbstractErgoExtractorEntity.js +0 -57
  19. package/dist/lib/ergo/index.d.ts +0 -10
  20. package/dist/lib/ergo/index.d.ts.map +0 -1
  21. package/dist/lib/ergo/index.js +0 -10
  22. package/dist/lib/ergo/initializable/AbstractInitializable.d.ts +0 -48
  23. package/dist/lib/ergo/initializable/AbstractInitializable.d.ts.map +0 -1
  24. package/dist/lib/ergo/initializable/AbstractInitializable.js +0 -163
  25. package/dist/lib/ergo/initializable/AbstractInitializableAction.d.ts +0 -14
  26. package/dist/lib/ergo/initializable/AbstractInitializableAction.d.ts.map +0 -1
  27. package/dist/lib/ergo/initializable/AbstractInitializableAction.js +0 -16
  28. package/dist/lib/ergo/initializable/index.d.ts +0 -3
  29. package/dist/lib/ergo/initializable/index.d.ts.map +0 -1
  30. package/dist/lib/ergo/initializable/index.js +0 -3
  31. package/dist/lib/ergo/interfaces.d.ts +0 -47
  32. package/dist/lib/ergo/interfaces.d.ts.map +0 -1
  33. package/dist/lib/ergo/interfaces.js +0 -8
  34. package/dist/lib/ergo/network/AbstractNetwork.d.ts +0 -26
  35. package/dist/lib/ergo/network/AbstractNetwork.d.ts.map +0 -1
  36. package/dist/lib/ergo/network/AbstractNetwork.js +0 -3
  37. package/dist/lib/ergo/network/ExplorerNetwork.d.ts +0 -74
  38. package/dist/lib/ergo/network/ExplorerNetwork.d.ts.map +0 -1
  39. package/dist/lib/ergo/network/ExplorerNetwork.js +0 -185
  40. package/dist/lib/ergo/network/NodeNetwork.d.ts +0 -60
  41. package/dist/lib/ergo/network/NodeNetwork.d.ts.map +0 -1
  42. package/dist/lib/ergo/network/NodeNetwork.js +0 -131
  43. package/dist/lib/ergo/utils.d.ts +0 -8
  44. package/dist/lib/ergo/utils.d.ts.map +0 -1
  45. package/dist/lib/ergo/utils.js +0 -16
  46. package/dist/lib/index.d.ts +0 -4
  47. package/dist/lib/index.d.ts.map +0 -1
  48. package/dist/lib/index.js +0 -4
  49. package/dist/tests/AbstractErgoExtractor.mock.d.ts +0 -11
  50. package/dist/tests/AbstractErgoExtractor.mock.d.ts.map +0 -1
  51. package/dist/tests/AbstractErgoExtractor.mock.js +0 -11
  52. package/dist/tests/AbstractErgoExtractor.spec.d.ts +0 -2
  53. package/dist/tests/AbstractErgoExtractor.spec.d.ts.map +0 -1
  54. package/dist/tests/AbstractErgoExtractor.spec.js +0 -240
  55. package/dist/tests/AbstractErgoExtractorAction.mock.d.ts +0 -15
  56. package/dist/tests/AbstractErgoExtractorAction.mock.d.ts.map +0 -1
  57. package/dist/tests/AbstractErgoExtractorAction.mock.js +0 -27
  58. package/dist/tests/AbstractErgoExtractorAction.spec.d.ts +0 -2
  59. package/dist/tests/AbstractErgoExtractorAction.spec.d.ts.map +0 -1
  60. package/dist/tests/AbstractErgoExtractorAction.spec.js +0 -221
  61. package/dist/tests/initializable/AbstractInitializable.mock.d.ts +0 -61
  62. package/dist/tests/initializable/AbstractInitializable.mock.d.ts.map +0 -1
  63. package/dist/tests/initializable/AbstractInitializable.mock.js +0 -18
  64. package/dist/tests/initializable/AbstractInitializable.spec.d.ts +0 -2
  65. package/dist/tests/initializable/AbstractInitializable.spec.d.ts.map +0 -1
  66. package/dist/tests/initializable/AbstractInitializable.spec.js +0 -299
  67. package/dist/tests/initializable/AbstractInitializableAction.mock.d.ts +0 -15
  68. package/dist/tests/initializable/AbstractInitializableAction.mock.d.ts.map +0 -1
  69. package/dist/tests/initializable/AbstractInitializableAction.mock.js +0 -27
  70. package/dist/tests/initializable/AbstractInitializableAction.spec.d.ts +0 -2
  71. package/dist/tests/initializable/AbstractInitializableAction.spec.d.ts.map +0 -1
  72. package/dist/tests/initializable/AbstractInitializableAction.spec.js +0 -54
  73. package/dist/tests/initializable/testData.d.ts +0 -94
  74. package/dist/tests/initializable/testData.d.ts.map +0 -1
  75. package/dist/tests/initializable/testData.js +0 -235
  76. package/dist/tests/network/ExplorerNetwork.spec.d.ts +0 -2
  77. package/dist/tests/network/ExplorerNetwork.spec.d.ts.map +0 -1
  78. package/dist/tests/network/ExplorerNetwork.spec.js +0 -61
  79. package/dist/tests/network/NodeNetwork.spec.d.ts +0 -2
  80. package/dist/tests/network/NodeNetwork.spec.d.ts.map +0 -1
  81. package/dist/tests/network/NodeNetwork.spec.js +0 -48
  82. package/dist/tests/network/testData.d.ts +0 -752
  83. package/dist/tests/network/testData.d.ts.map +0 -1
  84. package/dist/tests/network/testData.js +0 -1401
  85. package/dist/tests/testData.d.ts +0 -32
  86. package/dist/tests/testData.d.ts.map +0 -1
  87. package/dist/tests/testData.js +0 -121
  88. package/dist/tests/testUtils.d.ts +0 -9
  89. package/dist/tests/testUtils.d.ts.map +0 -1
  90. package/dist/tests/testUtils.js +0 -31
  91. package/dist/tsconfig.tsbuildinfo +0 -1
  92. package/dist/vitest.config.d.ts +0 -3
  93. package/dist/vitest.config.d.ts.map +0 -1
  94. package/dist/vitest.config.js +0 -18
  95. package/lib/abstractExtractor.ts +0 -31
  96. package/lib/constants.ts +0 -3
  97. package/lib/ergo/AbstractErgoExtractor.ts +0 -227
  98. package/lib/ergo/AbstractErgoExtractorAction.ts +0 -319
  99. package/lib/ergo/AbstractErgoExtractorEntity.ts +0 -32
  100. package/lib/ergo/index.ts +0 -9
  101. package/lib/ergo/initializable/AbstractInitializable.ts +0 -228
  102. package/lib/ergo/initializable/AbstractInitializableAction.ts +0 -33
  103. package/lib/ergo/initializable/index.ts +0 -2
  104. package/lib/ergo/interfaces.ts +0 -51
  105. package/lib/ergo/network/AbstractNetwork.ts +0 -29
  106. package/lib/ergo/network/ExplorerNetwork.ts +0 -237
  107. package/lib/ergo/network/NodeNetwork.ts +0 -169
  108. package/lib/ergo/utils.ts +0 -15
  109. package/lib/index.ts +0 -3
  110. package/tests/AbstractErgoExtractor.mock.ts +0 -33
  111. package/tests/AbstractErgoExtractor.spec.ts +0 -284
  112. package/tests/AbstractErgoExtractorAction.mock.ts +0 -45
  113. package/tests/AbstractErgoExtractorAction.spec.ts +0 -268
  114. package/tests/initializable/AbstractInitializable.mock.ts +0 -44
  115. package/tests/initializable/AbstractInitializable.spec.ts +0 -386
  116. package/tests/initializable/AbstractInitializableAction.mock.ts +0 -45
  117. package/tests/initializable/AbstractInitializableAction.spec.ts +0 -64
  118. package/tests/initializable/testData.ts +0 -283
  119. package/tests/network/ExplorerNetwork.spec.ts +0 -73
  120. package/tests/network/NodeNetwork.spec.ts +0 -56
  121. package/tests/network/testData.ts +0 -1708
  122. package/tests/testData.ts +0 -140
  123. package/tests/testUtils.ts +0 -22
  124. package/tsconfig.build.json +0 -9
  125. package/tsconfig.build.tsbuildinfo +0 -1
  126. package/tsconfig.json +0 -9
  127. package/vitest.config.ts +0 -18
@@ -1,222 +0,0 @@
1
- import { In, Not, } from '@rosen-bridge/extended-typeorm';
2
- import { chunk, difference, pick } from 'lodash-es';
3
- import { DummyLogger } from '@rosen-bridge/abstract-logger';
4
- import JsonBigInt from '@rosen-bridge/json-bigint';
5
- import { DB_CHUNK_SIZE } from '../constants';
6
- export class AbstractErgoExtractorAction {
7
- datasource;
8
- logger;
9
- repository;
10
- repo;
11
- constructor(dataSource, repo, logger) {
12
- this.datasource = dataSource;
13
- this.logger = logger ? logger : new DummyLogger();
14
- this.repository = this.datasource.getRepository(repo);
15
- this.repo = repo;
16
- }
17
- /**
18
- * create the database entity from extracted data and block information
19
- */
20
- createEntity = (data, // eslint-disable-line @typescript-eslint/no-unused-vars
21
- block, // eslint-disable-line @typescript-eslint/no-unused-vars
22
- extractor) => {
23
- throw Error('You must implement `createEntity` or override `insertEntities` and `updateEntities`');
24
- };
25
- /**
26
- * convert the database entity back to raw data
27
- */
28
- convertEntityToData = (entities) => {
29
- throw Error('You must implement `convertEntityToData` or override `deleteBlockEntities`');
30
- };
31
- /**
32
- * insert entities extracted from a block to database
33
- * @param queryRunner
34
- * @param boxesToInsert
35
- * @param block
36
- * @param extractor
37
- */
38
- insertEntities = async (queryRunner, boxesToInsert, block, extractor) => {
39
- const repository = queryRunner.manager.getRepository(this.repo);
40
- await repository.insert(this.createEntity(boxesToInsert, block, extractor));
41
- };
42
- /**
43
- * update entities related to a box
44
- * @param queryRunner
45
- * @param updateBox
46
- * @param block
47
- * @param extractor
48
- */
49
- updateEntity = async (queryRunner, updateBox, // eslint-disable-line @typescript-eslint/no-unused-vars
50
- block, // eslint-disable-line @typescript-eslint/no-unused-vars
51
- extractor) => {
52
- const repository = queryRunner.manager.getRepository(this.repo);
53
- const box = this.createEntity([updateBox], block, extractor)[0];
54
- await repository.update({
55
- boxId: box.boxId,
56
- extractor: extractor,
57
- }, box);
58
- };
59
- /**
60
- * delete all data extracted from a block
61
- * @param queryRunner
62
- * @param extractor
63
- * @param block
64
- * @returns
65
- */
66
- deleteBlockEntities = async (queryRunner, extractor, block) => {
67
- const repository = queryRunner.manager.getRepository(this.repo);
68
- const deletedData = await repository.find({
69
- where: {
70
- extractor: extractor,
71
- block: block,
72
- },
73
- });
74
- await repository.delete({
75
- extractor: extractor,
76
- block: block,
77
- });
78
- return this.convertEntityToData(deletedData);
79
- };
80
- /**
81
- * delete all data extracted from a block
82
- * @param queryRunner
83
- * @param extractor
84
- * @param block
85
- * @returns
86
- */
87
- updateBlockEntities = async (queryRunner, // eslint-disable-line @typescript-eslint/no-unused-vars
88
- extractor, block) => {
89
- const repository = this.datasource.getRepository(this.repo);
90
- const updatedData = await repository.find({
91
- where: {
92
- extractor: extractor,
93
- spendBlock: block,
94
- block: Not(block),
95
- },
96
- });
97
- await repository.update({
98
- spendBlock: block,
99
- extractor: extractor,
100
- }, {
101
- spendBlock: null,
102
- spendHeight: 0,
103
- });
104
- return updatedData;
105
- };
106
- /**
107
- * insert all extracted box data in an atomic transaction
108
- * update the data if a box with the same id is already stored in db
109
- * @param boxes
110
- * @param block
111
- * @param extractor
112
- * @return inserted items and updated box ids
113
- * returns undefined in case of any problem
114
- */
115
- storeBoxes = async (boxes, block, extractor) => {
116
- let boxesToInsert = [], boxesToUpdate = [];
117
- const queryRunner = this.datasource.createQueryRunner();
118
- await queryRunner.connect();
119
- await queryRunner.startTransaction();
120
- try {
121
- const repository = queryRunner.manager.getRepository(this.repo);
122
- const dbBoxIds = (await repository.findBy({
123
- boxId: In(boxes.map((item) => item.boxId)),
124
- extractor: extractor,
125
- })).map((box) => box.boxId);
126
- if (dbBoxIds.length > 0)
127
- this.logger.debug(`Found stored boxes with same boxId`, dbBoxIds);
128
- boxesToUpdate = boxes.filter((box) => dbBoxIds.includes(box.boxId));
129
- boxesToInsert = difference(boxes, boxesToUpdate);
130
- if (boxesToInsert.length > 0) {
131
- this.logger.debug(`Inserting boxes`);
132
- await this.insertEntities(queryRunner, boxesToInsert, block, extractor);
133
- }
134
- if (boxesToUpdate.length > 0)
135
- this.logger.info(`Updating boxes with following Ids in the database: [${boxesToUpdate
136
- .map((col) => col.boxId)
137
- .join(', ')}]`);
138
- for (const box of boxesToUpdate) {
139
- this.logger.debug(`Updating boxes in database [${JsonBigInt.stringify(box)}]`);
140
- await this.updateEntity(queryRunner, box, block, extractor);
141
- }
142
- await queryRunner.commitTransaction();
143
- return true;
144
- }
145
- catch (e) {
146
- this.logger.error(`An error occurred during store boxes action: ${e}`);
147
- await queryRunner.rollbackTransaction();
148
- return false;
149
- }
150
- finally {
151
- await queryRunner.release();
152
- }
153
- };
154
- /**
155
- * update spending information of stored boxes
156
- * chunk spendInfos to prevent large database queries
157
- * Note: It only updates the spendHeight and spendBlock fields. If updating
158
- * anything else is required, override this implementation to include the
159
- * additional fields.
160
- * @param spendInfos
161
- * @param block
162
- * @param extractor
163
- * @returns spent box ids
164
- */
165
- spendBoxes = async (spendInfos, block, extractor) => {
166
- const spentData = [];
167
- const spendInfoChunks = chunk(spendInfos, DB_CHUNK_SIZE);
168
- for (const spendInfoChunk of spendInfoChunks) {
169
- const boxIds = spendInfoChunk.map((info) => info.boxId);
170
- const updateResult = await this.repository.update({
171
- boxId: In(boxIds),
172
- extractor: extractor,
173
- }, {
174
- spendBlock: block.hash,
175
- spendHeight: block.height,
176
- });
177
- if (updateResult.affected && updateResult.affected > 0) {
178
- const spentRows = await this.repository.findBy({
179
- boxId: In(boxIds),
180
- spendBlock: block.hash,
181
- });
182
- spentData.push(...spentRows);
183
- for (const row of spentRows) {
184
- this.logger.debug(`Spent box with boxId [${row.boxId}] at height ${block.height}`);
185
- }
186
- }
187
- }
188
- return spentData.map((data) => pick(data, 'boxId'));
189
- };
190
- /**
191
- * delete extracted data from a specific block
192
- * if a box is spend in this block mark it as unspent
193
- * if a box is created in this block remove it from database
194
- * @param block
195
- * @param extractor
196
- * @return deleted items and updated box ids
197
- */
198
- deleteBlockBoxes = async (block, extractor) => {
199
- const queryRunner = this.datasource.createQueryRunner();
200
- await queryRunner.connect();
201
- await queryRunner.startTransaction();
202
- try {
203
- this.logger.info(`Deleting boxes in block ${block} and extractor ${extractor}`);
204
- const updatedData = await this.updateBlockEntities(queryRunner, extractor, block);
205
- const deletedData = await this.deleteBlockEntities(queryRunner, extractor, block);
206
- await queryRunner.commitTransaction();
207
- return {
208
- deletedData,
209
- updatedData: updatedData.map((data) => pick(data, 'boxId')),
210
- };
211
- }
212
- catch (error) {
213
- await queryRunner.rollbackTransaction();
214
- this.logger.error(`An error occurred while deleting data extracted from block ${block}`, error);
215
- throw error;
216
- }
217
- finally {
218
- await queryRunner.release();
219
- }
220
- };
221
- }
222
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"AbstractErgoExtractorAction.js","sourceRoot":"","sources":["../../../lib/ergo/AbstractErgoExtractorAction.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  DataSource,\n  In,\n  Repository,\n  Not,\n  EntityTarget,\n  FindOptionsWhere,\n  QueryRunner,\n} from '@rosen-bridge/extended-typeorm';\nimport { chunk, difference, pick } from 'lodash-es';\nimport { AbstractLogger, DummyLogger } from '@rosen-bridge/abstract-logger';\nimport JsonBigInt from '@rosen-bridge/json-bigint';\nimport { BlockInfo } from '@rosen-bridge/scanner-interfaces';\n\nimport { AbstractBoxData, BoxInfo, SpendInfo } from './interfaces';\nimport { DB_CHUNK_SIZE } from '../constants';\nimport { AbstractErgoExtractorEntity } from './AbstractErgoExtractorEntity';\n\nexport abstract class AbstractErgoExtractorAction<\n  ExtractedData extends AbstractBoxData,\n  ExtractorEntity extends AbstractErgoExtractorEntity,\n> {\n  private readonly datasource: DataSource;\n  readonly logger: AbstractLogger;\n  protected readonly repository: Repository<ExtractorEntity>;\n  private repo: EntityTarget<ExtractorEntity>;\n\n  constructor(\n    dataSource: DataSource,\n    repo: EntityTarget<ExtractorEntity>,\n    logger?: AbstractLogger,\n  ) {\n    this.datasource = dataSource;\n    this.logger = logger ? logger : new DummyLogger();\n    this.repository = this.datasource.getRepository(repo);\n    this.repo = repo;\n  }\n\n  /**\n   * create the database entity from extracted data and block information\n   */\n  protected createEntity = (\n    data: ExtractedData[], // eslint-disable-line @typescript-eslint/no-unused-vars\n    block: BlockInfo, // eslint-disable-line @typescript-eslint/no-unused-vars\n    extractor: string, // eslint-disable-line @typescript-eslint/no-unused-vars\n  ): Array<Omit<ExtractorEntity, 'id'>> => {\n    throw Error(\n      'You must implement `createEntity` or override `insertEntities` and `updateEntities`',\n    );\n  };\n\n  /**\n   * convert the database entity back to raw data\n   */\n  protected convertEntityToData = (\n    entities: ExtractorEntity[], // eslint-disable-line @typescript-eslint/no-unused-vars\n  ): ExtractedData[] => {\n    throw Error(\n      'You must implement `convertEntityToData` or override `deleteBlockEntities`',\n    );\n  };\n\n  /**\n   * insert entities extracted from a block to database\n   * @param queryRunner\n   * @param boxesToInsert\n   * @param block\n   * @param extractor\n   */\n  protected insertEntities = async (\n    queryRunner: QueryRunner,\n    boxesToInsert: Array<ExtractedData>,\n    block: BlockInfo,\n    extractor: string,\n  ) => {\n    const repository = queryRunner.manager.getRepository(this.repo);\n    await repository.insert(\n      this.createEntity(boxesToInsert, block, extractor) as any, // eslint-disable-line @typescript-eslint/no-explicit-any\n    );\n  };\n\n  /**\n   * update entities related to a box\n   * @param queryRunner\n   * @param updateBox\n   * @param block\n   * @param extractor\n   */\n  protected updateEntity = async (\n    queryRunner: QueryRunner,\n    updateBox: ExtractedData, // eslint-disable-line @typescript-eslint/no-unused-vars\n    block: BlockInfo, // eslint-disable-line @typescript-eslint/no-unused-vars\n    extractor: string,\n  ) => {\n    const repository = queryRunner.manager.getRepository(this.repo);\n    const box = this.createEntity([updateBox], block, extractor)[0];\n    await repository.update(\n      {\n        boxId: box.boxId,\n        extractor: extractor,\n      } as FindOptionsWhere<ExtractorEntity>,\n      box as any, // eslint-disable-line @typescript-eslint/no-explicit-any\n    );\n  };\n\n  /**\n   * delete all data extracted from a block\n   * @param queryRunner\n   * @param extractor\n   * @param block\n   * @returns\n   */\n  protected deleteBlockEntities = async (\n    queryRunner: QueryRunner,\n    extractor: string,\n    block: string,\n  ): Promise<ExtractedData[]> => {\n    const repository = queryRunner.manager.getRepository(this.repo);\n    const deletedData = await repository.find({\n      where: {\n        extractor: extractor,\n        block: block,\n      } as FindOptionsWhere<ExtractorEntity>,\n    });\n    await repository.delete({\n      extractor: extractor,\n      block: block,\n    } as unknown as FindOptionsWhere<ExtractorEntity>);\n    return this.convertEntityToData(deletedData);\n  };\n\n  /**\n   * delete all data extracted from a block\n   * @param queryRunner\n   * @param extractor\n   * @param block\n   * @returns\n   */\n  protected updateBlockEntities = async (\n    queryRunner: QueryRunner, // eslint-disable-line @typescript-eslint/no-unused-vars\n    extractor: string,\n    block: string,\n  ): Promise<ExtractorEntity[]> => {\n    const repository = this.datasource.getRepository(\n      this.repo as EntityTarget<new () => ExtractorEntity>,\n    );\n    const updatedData = await repository.find({\n      where: {\n        extractor: extractor,\n        spendBlock: block,\n        block: Not(block),\n      } as unknown as FindOptionsWhere<ExtractorEntity>,\n    });\n    await repository.update(\n      {\n        spendBlock: block,\n        extractor: extractor,\n      } as unknown as FindOptionsWhere<ExtractorEntity>,\n      {\n        spendBlock: null,\n        spendHeight: 0,\n      },\n    );\n    return updatedData as unknown as ExtractorEntity[];\n  };\n\n  /**\n   * insert all extracted box data in an atomic transaction\n   * update the data if a box with the same id is already stored in db\n   * @param boxes\n   * @param block\n   * @param extractor\n   * @return inserted items and updated box ids\n   * returns undefined in case of any problem\n   */\n  storeBoxes = async (\n    boxes: Array<ExtractedData>,\n    block: BlockInfo,\n    extractor: string,\n  ): Promise<boolean> => {\n    let boxesToInsert: ExtractedData[] = [],\n      boxesToUpdate: ExtractedData[] = [];\n    const queryRunner = this.datasource.createQueryRunner();\n    await queryRunner.connect();\n    await queryRunner.startTransaction();\n    try {\n      const repository = queryRunner.manager.getRepository(this.repo);\n      const dbBoxIds = (\n        await repository.findBy({\n          boxId: In(boxes.map((item) => item.boxId)),\n          extractor: extractor,\n        } as FindOptionsWhere<ExtractorEntity>)\n      ).map((box) => box.boxId);\n      if (dbBoxIds.length > 0)\n        this.logger.debug(`Found stored boxes with same boxId`, dbBoxIds);\n\n      boxesToUpdate = boxes.filter((box) => dbBoxIds.includes(box.boxId));\n      boxesToInsert = difference(boxes, boxesToUpdate);\n\n      if (boxesToInsert.length > 0) {\n        this.logger.debug(`Inserting boxes`);\n        await this.insertEntities(queryRunner, boxesToInsert, block, extractor);\n      }\n      if (boxesToUpdate.length > 0)\n        this.logger.info(\n          `Updating boxes with following Ids in the database: [${boxesToUpdate\n            .map((col) => col.boxId)\n            .join(', ')}]`,\n        );\n      for (const box of boxesToUpdate) {\n        this.logger.debug(\n          `Updating boxes in database [${JsonBigInt.stringify(box)}]`,\n        );\n        await this.updateEntity(queryRunner, box, block, extractor);\n      }\n      await queryRunner.commitTransaction();\n      return true;\n    } catch (e) {\n      this.logger.error(`An error occurred during store boxes action: ${e}`);\n      await queryRunner.rollbackTransaction();\n      return false;\n    } finally {\n      await queryRunner.release();\n    }\n  };\n\n  /**\n   * update spending information of stored boxes\n   * chunk spendInfos to prevent large database queries\n   * Note: It only updates the spendHeight and spendBlock fields. If updating\n   * anything else is required, override this implementation to include the\n   * additional fields.\n   * @param spendInfos\n   * @param block\n   * @param extractor\n   * @returns spent box ids\n   */\n  spendBoxes = async (\n    spendInfos: Array<SpendInfo>,\n    block: BlockInfo,\n    extractor: string,\n  ): Promise<BoxInfo[]> => {\n    const spentData = [];\n    const spendInfoChunks = chunk(spendInfos, DB_CHUNK_SIZE);\n    for (const spendInfoChunk of spendInfoChunks) {\n      const boxIds = spendInfoChunk.map((info) => info.boxId);\n      const updateResult = await this.repository.update(\n        {\n          boxId: In(boxIds),\n          extractor: extractor,\n        } as FindOptionsWhere<ExtractorEntity>,\n        {\n          spendBlock: block.hash,\n          spendHeight: block.height,\n        } as any, // eslint-disable-line @typescript-eslint/no-explicit-any\n      );\n\n      if (updateResult.affected && updateResult.affected > 0) {\n        const spentRows = await this.repository.findBy({\n          boxId: In(boxIds),\n          spendBlock: block.hash,\n        } as FindOptionsWhere<ExtractorEntity>);\n        spentData.push(...spentRows);\n        for (const row of spentRows) {\n          this.logger.debug(\n            `Spent box with boxId [${row.boxId}] at height ${block.height}`,\n          );\n        }\n      }\n    }\n    return spentData.map((data) => pick(data, 'boxId'));\n  };\n\n  /**\n   * delete extracted data from a specific block\n   * if a box is spend in this block mark it as unspent\n   * if a box is created in this block remove it from database\n   * @param block\n   * @param extractor\n   * @return deleted items and updated box ids\n   */\n  deleteBlockBoxes = async (\n    block: string,\n    extractor: string,\n  ): Promise<{ deletedData: ExtractedData[]; updatedData: BoxInfo[] }> => {\n    const queryRunner = this.datasource.createQueryRunner();\n    await queryRunner.connect();\n    await queryRunner.startTransaction();\n    try {\n      this.logger.info(\n        `Deleting boxes in block ${block} and extractor ${extractor}`,\n      );\n      const updatedData = await this.updateBlockEntities(\n        queryRunner,\n        extractor,\n        block,\n      );\n      const deletedData = await this.deleteBlockEntities(\n        queryRunner,\n        extractor,\n        block,\n      );\n      await queryRunner.commitTransaction();\n      return {\n        deletedData,\n        updatedData: updatedData.map((data) => pick(data, 'boxId')),\n      };\n    } catch (error) {\n      await queryRunner.rollbackTransaction();\n      this.logger.error(\n        `An error occurred while deleting data extracted from block ${block}`,\n        error,\n      );\n      throw error;\n    } finally {\n      await queryRunner.release();\n    }\n  };\n}\n"]}
@@ -1,11 +0,0 @@
1
- export declare abstract class AbstractErgoExtractorEntity {
2
- id: number;
3
- boxId: string;
4
- block: string;
5
- height: number;
6
- spendBlock?: string | null;
7
- spendHeight?: number | null;
8
- extractor: string;
9
- serialized: string;
10
- }
11
- //# sourceMappingURL=AbstractErgoExtractorEntity.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"AbstractErgoExtractorEntity.d.ts","sourceRoot":"","sources":["../../../lib/ergo/AbstractErgoExtractorEntity.ts"],"names":[],"mappings":"AAMA,8BACsB,2BAA2B;IAE/C,EAAE,EAAE,MAAM,CAAC;IAGX,KAAK,EAAE,MAAM,CAAC;IAGd,KAAK,EAAE,MAAM,CAAC;IAGd,MAAM,EAAE,MAAM,CAAC;IAGf,UAAU,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IAG3B,WAAW,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IAG5B,SAAS,EAAE,MAAM,CAAC;IAGlB,UAAU,EAAE,MAAM,CAAC;CACpB"}
@@ -1,57 +0,0 @@
1
- var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
2
- var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
3
- if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
4
- else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
5
- return c > 3 && r && Object.defineProperty(target, key, r), r;
6
- };
7
- var __metadata = (this && this.__metadata) || function (k, v) {
8
- if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
9
- };
10
- import { Column, PrimaryGeneratedColumn, Unique, } from '@rosen-bridge/extended-typeorm';
11
- let AbstractErgoExtractorEntity = class AbstractErgoExtractorEntity {
12
- id;
13
- boxId;
14
- block;
15
- height;
16
- spendBlock;
17
- spendHeight;
18
- extractor;
19
- serialized;
20
- };
21
- __decorate([
22
- PrimaryGeneratedColumn(),
23
- __metadata("design:type", Number)
24
- ], AbstractErgoExtractorEntity.prototype, "id", void 0);
25
- __decorate([
26
- Column({ type: 'varchar' }),
27
- __metadata("design:type", String)
28
- ], AbstractErgoExtractorEntity.prototype, "boxId", void 0);
29
- __decorate([
30
- Column({ type: 'varchar' }),
31
- __metadata("design:type", String)
32
- ], AbstractErgoExtractorEntity.prototype, "block", void 0);
33
- __decorate([
34
- Column({ type: 'int' }),
35
- __metadata("design:type", Number)
36
- ], AbstractErgoExtractorEntity.prototype, "height", void 0);
37
- __decorate([
38
- Column({ nullable: true, type: 'varchar' }),
39
- __metadata("design:type", Object)
40
- ], AbstractErgoExtractorEntity.prototype, "spendBlock", void 0);
41
- __decorate([
42
- Column({ nullable: true, type: 'int' }),
43
- __metadata("design:type", Object)
44
- ], AbstractErgoExtractorEntity.prototype, "spendHeight", void 0);
45
- __decorate([
46
- Column({ type: 'varchar' }),
47
- __metadata("design:type", String)
48
- ], AbstractErgoExtractorEntity.prototype, "extractor", void 0);
49
- __decorate([
50
- Column({ type: 'varchar' }),
51
- __metadata("design:type", String)
52
- ], AbstractErgoExtractorEntity.prototype, "serialized", void 0);
53
- AbstractErgoExtractorEntity = __decorate([
54
- Unique(['boxId', 'extractor'])
55
- ], AbstractErgoExtractorEntity);
56
- export { AbstractErgoExtractorEntity };
57
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiQWJzdHJhY3RFcmdvRXh0cmFjdG9yRW50aXR5LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vbGliL2VyZ28vQWJzdHJhY3RFcmdvRXh0cmFjdG9yRW50aXR5LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7OztBQUFBLE9BQU8sRUFDTCxNQUFNLEVBQ04sc0JBQXNCLEVBQ3RCLE1BQU0sR0FDUCxNQUFNLGdDQUFnQyxDQUFDO0FBR2pDLElBQWUsMkJBQTJCLEdBQTFDLE1BQWUsMkJBQTJCO0lBRS9DLEVBQUUsQ0FBUztJQUdYLEtBQUssQ0FBUztJQUdkLEtBQUssQ0FBUztJQUdkLE1BQU0sQ0FBUztJQUdmLFVBQVUsQ0FBaUI7SUFHM0IsV0FBVyxDQUFpQjtJQUc1QixTQUFTLENBQVM7SUFHbEIsVUFBVSxDQUFTO0NBQ3BCLENBQUE7QUF0QkM7SUFEQyxzQkFBc0IsRUFBRTs7dURBQ2Q7QUFHWDtJQURDLE1BQU0sQ0FBQyxFQUFFLElBQUksRUFBRSxTQUFTLEVBQUUsQ0FBQzs7MERBQ2Q7QUFHZDtJQURDLE1BQU0sQ0FBQyxFQUFFLElBQUksRUFBRSxTQUFTLEVBQUUsQ0FBQzs7MERBQ2Q7QUFHZDtJQURDLE1BQU0sQ0FBQyxFQUFFLElBQUksRUFBRSxLQUFLLEVBQUUsQ0FBQzs7MkRBQ1Q7QUFHZjtJQURDLE1BQU0sQ0FBQyxFQUFFLFFBQVEsRUFBRSxJQUFJLEVBQUUsSUFBSSxFQUFFLFNBQVMsRUFBRSxDQUFDOzsrREFDakI7QUFHM0I7SUFEQyxNQUFNLENBQUMsRUFBRSxRQUFRLEVBQUUsSUFBSSxFQUFFLElBQUksRUFBRSxLQUFLLEVBQUUsQ0FBQzs7Z0VBQ1o7QUFHNUI7SUFEQyxNQUFNLENBQUMsRUFBRSxJQUFJLEVBQUUsU0FBUyxFQUFFLENBQUM7OzhEQUNWO0FBR2xCO0lBREMsTUFBTSxDQUFDLEVBQUUsSUFBSSxFQUFFLFNBQVMsRUFBRSxDQUFDOzsrREFDVDtBQXZCQywyQkFBMkI7SUFEaEQsTUFBTSxDQUFDLENBQUMsT0FBTyxFQUFFLFdBQVcsQ0FBQyxDQUFDO0dBQ1QsMkJBQTJCLENBd0JoRCIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7XG4gIENvbHVtbixcbiAgUHJpbWFyeUdlbmVyYXRlZENvbHVtbixcbiAgVW5pcXVlLFxufSBmcm9tICdAcm9zZW4tYnJpZGdlL2V4dGVuZGVkLXR5cGVvcm0nO1xuXG5AVW5pcXVlKFsnYm94SWQnLCAnZXh0cmFjdG9yJ10pXG5leHBvcnQgYWJzdHJhY3QgY2xhc3MgQWJzdHJhY3RFcmdvRXh0cmFjdG9yRW50aXR5IHtcbiAgQFByaW1hcnlHZW5lcmF0ZWRDb2x1bW4oKVxuICBpZDogbnVtYmVyO1xuXG4gIEBDb2x1bW4oeyB0eXBlOiAndmFyY2hhcicgfSlcbiAgYm94SWQ6IHN0cmluZztcblxuICBAQ29sdW1uKHsgdHlwZTogJ3ZhcmNoYXInIH0pXG4gIGJsb2NrOiBzdHJpbmc7XG5cbiAgQENvbHVtbih7IHR5cGU6ICdpbnQnIH0pXG4gIGhlaWdodDogbnVtYmVyO1xuXG4gIEBDb2x1bW4oeyBudWxsYWJsZTogdHJ1ZSwgdHlwZTogJ3ZhcmNoYXInIH0pXG4gIHNwZW5kQmxvY2s/OiBzdHJpbmcgfCBudWxsO1xuXG4gIEBDb2x1bW4oeyBudWxsYWJsZTogdHJ1ZSwgdHlwZTogJ2ludCcgfSlcbiAgc3BlbmRIZWlnaHQ/OiBudW1iZXIgfCBudWxsO1xuXG4gIEBDb2x1bW4oeyB0eXBlOiAndmFyY2hhcicgfSlcbiAgZXh0cmFjdG9yOiBzdHJpbmc7XG5cbiAgQENvbHVtbih7IHR5cGU6ICd2YXJjaGFyJyB9KVxuICBzZXJpYWxpemVkOiBzdHJpbmc7XG59XG4iXX0=
@@ -1,10 +0,0 @@
1
- export * from './AbstractErgoExtractor';
2
- export * from './AbstractErgoExtractorAction';
3
- export * from './interfaces';
4
- export * from './network/ExplorerNetwork';
5
- export * from './network/NodeNetwork';
6
- export * from './network/AbstractNetwork';
7
- export * from './initializable';
8
- export * from './utils';
9
- export * from './AbstractErgoExtractorEntity';
10
- //# sourceMappingURL=index.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../lib/ergo/index.ts"],"names":[],"mappings":"AAAA,cAAc,yBAAyB,CAAC;AACxC,cAAc,+BAA+B,CAAC;AAC9C,cAAc,cAAc,CAAC;AAC7B,cAAc,2BAA2B,CAAC;AAC1C,cAAc,uBAAuB,CAAC;AACtC,cAAc,2BAA2B,CAAC;AAC1C,cAAc,iBAAiB,CAAC;AAChC,cAAc,SAAS,CAAC;AACxB,cAAc,+BAA+B,CAAC"}
@@ -1,10 +0,0 @@
1
- export * from './AbstractErgoExtractor';
2
- export * from './AbstractErgoExtractorAction';
3
- export * from './interfaces';
4
- export * from './network/ExplorerNetwork';
5
- export * from './network/NodeNetwork';
6
- export * from './network/AbstractNetwork';
7
- export * from './initializable';
8
- export * from './utils';
9
- export * from './AbstractErgoExtractorEntity';
10
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9saWIvZXJnby9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLHlCQUF5QixDQUFDO0FBQ3hDLGNBQWMsK0JBQStCLENBQUM7QUFDOUMsY0FBYyxjQUFjLENBQUM7QUFDN0IsY0FBYywyQkFBMkIsQ0FBQztBQUMxQyxjQUFjLHVCQUF1QixDQUFDO0FBQ3RDLGNBQWMsMkJBQTJCLENBQUM7QUFDMUMsY0FBYyxpQkFBaUIsQ0FBQztBQUNoQyxjQUFjLFNBQVMsQ0FBQztBQUN4QixjQUFjLCtCQUErQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9BYnN0cmFjdEVyZ29FeHRyYWN0b3InO1xuZXhwb3J0ICogZnJvbSAnLi9BYnN0cmFjdEVyZ29FeHRyYWN0b3JBY3Rpb24nO1xuZXhwb3J0ICogZnJvbSAnLi9pbnRlcmZhY2VzJztcbmV4cG9ydCAqIGZyb20gJy4vbmV0d29yay9FeHBsb3Jlck5ldHdvcmsnO1xuZXhwb3J0ICogZnJvbSAnLi9uZXR3b3JrL05vZGVOZXR3b3JrJztcbmV4cG9ydCAqIGZyb20gJy4vbmV0d29yay9BYnN0cmFjdE5ldHdvcmsnO1xuZXhwb3J0ICogZnJvbSAnLi9pbml0aWFsaXphYmxlJztcbmV4cG9ydCAqIGZyb20gJy4vdXRpbHMnO1xuZXhwb3J0ICogZnJvbSAnLi9BYnN0cmFjdEVyZ29FeHRyYWN0b3JFbnRpdHknO1xuIl19
@@ -1,48 +0,0 @@
1
- import { AbstractLogger } from '@rosen-bridge/abstract-logger';
2
- import { BlockInfo, ErgoNetworkType } from '@rosen-bridge/scanner-interfaces';
3
- import { AbstractBoxData } from '../interfaces';
4
- import { AbstractErgoExtractor } from '../AbstractErgoExtractor';
5
- import { AbstractInitializableErgoExtractorAction } from './AbstractInitializableAction';
6
- import { AbstractErgoExtractorEntity } from '../AbstractErgoExtractorEntity';
7
- export declare abstract class AbstractInitializableErgoExtractor<ExtractedData extends AbstractBoxData, ExtractorEntity extends AbstractErgoExtractorEntity> extends AbstractErgoExtractor<ExtractedData, ExtractorEntity> {
8
- protected initialize: boolean;
9
- private address;
10
- protected abstract actions: AbstractInitializableErgoExtractorAction<ExtractedData, ExtractorEntity>;
11
- private network;
12
- constructor(type: ErgoNetworkType, url: string, address: string, logger?: AbstractLogger, initialize?: boolean);
13
- /**
14
- * Initialize extractor using Explorer network
15
- * @param initialBlock
16
- */
17
- private initializeWithExplorer;
18
- /**
19
- * Get the total tx count from Node network
20
- * @returns total tx count of the address
21
- */
22
- private getTotalTxCount;
23
- /**
24
- * Initialize extractor using Node network
25
- * @param initialBlock
26
- */
27
- private initializeWithNode;
28
- /**
29
- * Process a batch of transactions
30
- * group txs into blocks and process them using `processTransactions`
31
- * @param txs
32
- */
33
- private processTransactionBatch;
34
- /**
35
- * Initialize the extractor with retrial on any unexpected problem
36
- * its the common part of initialize with Node and Explorer network
37
- * @param job
38
- */
39
- private initWithRetrial;
40
- /**
41
- * initialize extractor database with data created below the initial height
42
- * ignore initialization if this feature is off
43
- * try to get data multiple times to pass accidental network problems
44
- * @param initialBlock
45
- */
46
- initializeBoxes: (initialBlock: BlockInfo) => Promise<void>;
47
- }
48
- //# sourceMappingURL=AbstractInitializable.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"AbstractInitializable.d.ts","sourceRoot":"","sources":["../../../../lib/ergo/initializable/AbstractInitializable.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,cAAc,EAAE,MAAM,+BAA+B,CAAC;AAE/D,OAAO,EAAE,SAAS,EAAE,eAAe,EAAE,MAAM,kCAAkC,CAAC;AAE9E,OAAO,EAAE,eAAe,EAAuB,MAAM,eAAe,CAAC;AAErE,OAAO,EAAE,qBAAqB,EAAE,MAAM,0BAA0B,CAAC;AACjE,OAAO,EAAE,wCAAwC,EAAE,MAAM,+BAA+B,CAAC;AAGzF,OAAO,EAAE,2BAA2B,EAAE,MAAM,gCAAgC,CAAC;AAE7E,8BAAsB,kCAAkC,CACtD,aAAa,SAAS,eAAe,EACrC,eAAe,SAAS,2BAA2B,CACnD,SAAQ,qBAAqB,CAAC,aAAa,EAAE,eAAe,CAAC;IAC7D,SAAS,CAAC,UAAU,EAAE,OAAO,CAAC;IAC9B,OAAO,CAAC,OAAO,CAAS;IACxB,SAAS,CAAC,QAAQ,CAAC,OAAO,EAAE,wCAAwC,CAClE,aAAa,EACb,eAAe,CAChB,CAAC;IAEF,OAAO,CAAC,OAAO,CAAgC;gBAG7C,IAAI,EAAE,eAAe,EACrB,GAAG,EAAE,MAAM,EACX,OAAO,EAAE,MAAM,EACf,MAAM,CAAC,EAAE,cAAc,EACvB,UAAU,UAAO;IAkBnB;;;OAGG;IACH,OAAO,CAAC,sBAAsB,CA6C5B;IAEF;;;OAGG;IACH,OAAO,CAAC,eAAe,CAKrB;IAEF;;;OAGG;IACH,OAAO,CAAC,kBAAkB,CAuCxB;IAEF;;;;OAIG;IACH,OAAO,CAAC,uBAAuB,CAoB7B;IAEF;;;;OAIG;IACH,OAAO,CAAC,eAAe,CA+BrB;IAEF;;;;;OAKG;IACH,eAAe,EAAE,CAAC,YAAY,EAAE,SAAS,KAAK,OAAO,CAAC,IAAI,CAAC,CAAC;CAC7D"}
@@ -1,163 +0,0 @@
1
- import { groupBy, sortBy } from 'lodash-es';
2
- import { ErgoNetworkType } from '@rosen-bridge/scanner-interfaces';
3
- import { API_LIMIT, RETRIAL_COUNT } from '../../constants';
4
- import { AbstractErgoExtractor } from '../AbstractErgoExtractor';
5
- import { ExplorerNetwork } from '../network/ExplorerNetwork';
6
- import { NodeNetwork } from '../network/NodeNetwork';
7
- export class AbstractInitializableErgoExtractor extends AbstractErgoExtractor {
8
- initialize;
9
- address;
10
- network;
11
- constructor(type, url, address, logger, initialize = true) {
12
- super(logger);
13
- this.address = address;
14
- this.initialize = initialize;
15
- if (type == ErgoNetworkType.Explorer) {
16
- this.network = new ExplorerNetwork(url);
17
- this.initializeBoxes = (initialBlock) => {
18
- return this.initializeWithExplorer(initialBlock);
19
- };
20
- }
21
- else if (type == ErgoNetworkType.Node) {
22
- this.network = new NodeNetwork(url);
23
- this.initializeBoxes = (initialBlock) => {
24
- return this.initializeWithNode(initialBlock);
25
- };
26
- }
27
- else
28
- throw new Error(`Network type ${type} is not supported`);
29
- }
30
- /**
31
- * Initialize extractor using Explorer network
32
- * @param initialBlock
33
- */
34
- initializeWithExplorer = async (initialBlock) => {
35
- const explorerNetwork = this.network;
36
- let fromHeight = 0, toHeight = initialBlock.height;
37
- await this.initWithRetrial(async () => {
38
- while (fromHeight < toHeight) {
39
- let txs;
40
- // eslint-disable-next-line no-constant-condition
41
- while (true) {
42
- txs = await explorerNetwork.getAddressTransactionsWithHeight(this.address, fromHeight, toHeight);
43
- this.logger.debug(`Found ${txs.length} transactions for the address from height ${fromHeight} to height ${toHeight}`);
44
- if (txs.length < API_LIMIT || fromHeight === toHeight) {
45
- break; // Exit loop if we have fewer transactions than the limit or if the range is reduced to a single height
46
- }
47
- toHeight = Math.floor((toHeight - fromHeight) / 2) + fromHeight;
48
- this.logger.debug(`Limiting the query height range to [${fromHeight}, ${toHeight}]`);
49
- }
50
- if (txs.length < API_LIMIT) {
51
- if (txs.length > 0)
52
- await this.processTransactionBatch(txs);
53
- }
54
- else {
55
- this.logger.debug(`Block at height ${fromHeight} has more than (or equal) ${API_LIMIT} relevant txs, processing all txs in the block`);
56
- const blockId = await explorerNetwork.getBlockIdAtHeight(fromHeight);
57
- const blockTxs = await explorerNetwork.getBlockTxs(blockId);
58
- this.logger.debug(`Found ${blockTxs.length} transactions at height ${fromHeight}`);
59
- await this.processTransactions(blockTxs, {
60
- hash: blockId,
61
- height: fromHeight,
62
- });
63
- }
64
- fromHeight = toHeight + 1;
65
- toHeight = initialBlock.height;
66
- }
67
- });
68
- };
69
- /**
70
- * Get the total tx count from Node network
71
- * @returns total tx count of the address
72
- */
73
- getTotalTxCount = async () => {
74
- const response = await this.network.getAddressTransactionsWithOffsetLimit(this.address, 0, 0);
75
- return response.total;
76
- };
77
- /**
78
- * Initialize extractor using Node network
79
- * @param initialBlock
80
- */
81
- initializeWithNode = async (initialBlock) => {
82
- const txCountBeforeInit = await this.getTotalTxCount();
83
- let offset = 0, total = 1, round = 1;
84
- await this.initWithRetrial(async () => {
85
- // Repeat the whole process twice to cover all spent boxes
86
- // After round 1 all boxes have been saved and processed once
87
- // After round 2 spending information of all stored boxes are updated successfully
88
- while (round <= 2) {
89
- this.logger.debug(`Starting round ${round} of initialization`);
90
- while (offset < total) {
91
- const response = await this.network.getAddressTransactionsWithOffsetLimit(this.address, offset, API_LIMIT);
92
- total = response.total;
93
- const txs = response.items.filter((tx) => tx.inclusionHeight <= initialBlock.height);
94
- this.logger.debug(`Found ${txs.length} transactions below the initial height with offset ${offset} and total number of transactions ${total}`);
95
- if (txs.length > 0)
96
- await this.processTransactionBatch(txs);
97
- offset += API_LIMIT;
98
- }
99
- round++;
100
- offset = 0; // next round initial offset
101
- }
102
- });
103
- const txCountAfterInit = await this.getTotalTxCount();
104
- if (txCountAfterInit != txCountBeforeInit) {
105
- throw Error('Total transaction count changed during initialization phase, the stored data is not valid');
106
- }
107
- };
108
- /**
109
- * Process a batch of transactions
110
- * group txs into blocks and process them using `processTransactions`
111
- * @param txs
112
- */
113
- processTransactionBatch = async (txs) => {
114
- txs = sortBy(txs, (tx) => tx.inclusionHeight);
115
- const groupedTxs = groupBy(txs, (tx) => tx.blockId);
116
- this.logger.debug(`The transaction batch grouped to ${Object.keys(groupedTxs).length} blocks`);
117
- for (const blockId in groupedTxs) {
118
- const blockTxs = groupedTxs[blockId];
119
- const block = { hash: blockId, height: blockTxs[0].inclusionHeight };
120
- this.logger.debug(`Processing transactions at height ${blockTxs[0].inclusionHeight}`);
121
- const success = await this.processTransactions(blockTxs, block);
122
- if (!success)
123
- throw Error(`Processing transactions failed at height ${blockTxs[0].inclusionHeight}`);
124
- }
125
- };
126
- /**
127
- * Initialize the extractor with retrial on any unexpected problem
128
- * its the common part of initialize with Node and Explorer network
129
- * @param job
130
- */
131
- initWithRetrial = async (job) => {
132
- let trial = 1;
133
- if (this.initialize) {
134
- this.logger.debug(`Initializing ${this.getId()} started, removing all existing data`);
135
- await this.actions.removeAllData(this.getId());
136
- while (trial <= RETRIAL_COUNT) {
137
- try {
138
- await job();
139
- break;
140
- }
141
- catch (e) {
142
- this.logger.warn(`Initialization for ${this.getId()} failed with error :${e}`);
143
- if (trial == RETRIAL_COUNT)
144
- throw Error(`Initialization for ${this.getId()} failed after ${RETRIAL_COUNT} retrial`);
145
- trial += 1;
146
- this.logger.info(`Trying again to initialize ${this.getId()} with trial step ${trial}`);
147
- }
148
- }
149
- this.logger.info(`Initialization completed successfully for ${this.getId()}`);
150
- }
151
- else {
152
- this.logger.info(`Initialization for ${this.getId()} is turned off`);
153
- }
154
- };
155
- /**
156
- * initialize extractor database with data created below the initial height
157
- * ignore initialization if this feature is off
158
- * try to get data multiple times to pass accidental network problems
159
- * @param initialBlock
160
- */
161
- initializeBoxes;
162
- }
163
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"AbstractInitializable.js","sourceRoot":"","sources":["../../../../lib/ergo/initializable/AbstractInitializable.ts"],"names":[],"mappings":"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","sourcesContent":["import { AbstractLogger } from '@rosen-bridge/abstract-logger';\nimport { groupBy, sortBy } from 'lodash-es';\nimport { BlockInfo, ErgoNetworkType } from '@rosen-bridge/scanner-interfaces';\n\nimport { AbstractBoxData, ExtendedTransaction } from '../interfaces';\nimport { API_LIMIT, RETRIAL_COUNT } from '../../constants';\nimport { AbstractErgoExtractor } from '../AbstractErgoExtractor';\nimport { AbstractInitializableErgoExtractorAction } from './AbstractInitializableAction';\nimport { ExplorerNetwork } from '../network/ExplorerNetwork';\nimport { NodeNetwork } from '../network/NodeNetwork';\nimport { AbstractErgoExtractorEntity } from '../AbstractErgoExtractorEntity';\n\nexport abstract class AbstractInitializableErgoExtractor<\n  ExtractedData extends AbstractBoxData,\n  ExtractorEntity extends AbstractErgoExtractorEntity,\n> extends AbstractErgoExtractor<ExtractedData, ExtractorEntity> {\n  protected initialize: boolean;\n  private address: string;\n  protected abstract actions: AbstractInitializableErgoExtractorAction<\n    ExtractedData,\n    ExtractorEntity\n  >;\n\n  private network: ExplorerNetwork | NodeNetwork;\n\n  constructor(\n    type: ErgoNetworkType,\n    url: string,\n    address: string,\n    logger?: AbstractLogger,\n    initialize = true,\n  ) {\n    super(logger);\n    this.address = address;\n    this.initialize = initialize;\n    if (type == ErgoNetworkType.Explorer) {\n      this.network = new ExplorerNetwork(url);\n      this.initializeBoxes = (initialBlock: BlockInfo) => {\n        return this.initializeWithExplorer(initialBlock);\n      };\n    } else if (type == ErgoNetworkType.Node) {\n      this.network = new NodeNetwork(url);\n      this.initializeBoxes = (initialBlock: BlockInfo) => {\n        return this.initializeWithNode(initialBlock);\n      };\n    } else throw new Error(`Network type ${type} is not supported`);\n  }\n\n  /**\n   * Initialize extractor using Explorer network\n   * @param initialBlock\n   */\n  private initializeWithExplorer = async (initialBlock: BlockInfo) => {\n    const explorerNetwork = this.network as ExplorerNetwork;\n    let fromHeight = 0,\n      toHeight = initialBlock.height;\n    await this.initWithRetrial(async () => {\n      while (fromHeight < toHeight) {\n        let txs: Array<ExtendedTransaction>;\n        // eslint-disable-next-line no-constant-condition\n        while (true) {\n          txs = await explorerNetwork.getAddressTransactionsWithHeight(\n            this.address,\n            fromHeight,\n            toHeight,\n          );\n          this.logger.debug(\n            `Found ${txs.length} transactions for the address from height ${fromHeight} to height ${toHeight}`,\n          );\n          if (txs.length < API_LIMIT || fromHeight === toHeight) {\n            break; // Exit loop if we have fewer transactions than the limit or if the range is reduced to a single height\n          }\n          toHeight = Math.floor((toHeight - fromHeight) / 2) + fromHeight;\n          this.logger.debug(\n            `Limiting the query height range to [${fromHeight}, ${toHeight}]`,\n          );\n        }\n        if (txs.length < API_LIMIT) {\n          if (txs.length > 0) await this.processTransactionBatch(txs);\n        } else {\n          this.logger.debug(\n            `Block at height ${fromHeight} has more than (or equal) ${API_LIMIT} relevant txs, processing all txs in the block`,\n          );\n          const blockId = await explorerNetwork.getBlockIdAtHeight(fromHeight);\n          const blockTxs = await explorerNetwork.getBlockTxs(blockId);\n          this.logger.debug(\n            `Found ${blockTxs.length} transactions at height ${fromHeight}`,\n          );\n          await this.processTransactions(blockTxs, {\n            hash: blockId,\n            height: fromHeight,\n          });\n        }\n        fromHeight = toHeight + 1;\n        toHeight = initialBlock.height;\n      }\n    });\n  };\n\n  /**\n   * Get the total tx count from Node network\n   * @returns total tx count of the address\n   */\n  private getTotalTxCount = async () => {\n    const response = await (\n      this.network as NodeNetwork\n    ).getAddressTransactionsWithOffsetLimit(this.address, 0, 0);\n    return response.total;\n  };\n\n  /**\n   * Initialize extractor using Node network\n   * @param initialBlock\n   */\n  private initializeWithNode = async (initialBlock: BlockInfo) => {\n    const txCountBeforeInit = await this.getTotalTxCount();\n    let offset = 0,\n      total = 1,\n      round = 1;\n    await this.initWithRetrial(async () => {\n      // Repeat the whole process twice to cover all spent boxes\n      // After round 1 all boxes have been saved and processed once\n      // After round 2 spending information of all stored boxes are updated successfully\n      while (round <= 2) {\n        this.logger.debug(`Starting round ${round} of initialization`);\n        while (offset < total) {\n          const response = await (\n            this.network as NodeNetwork\n          ).getAddressTransactionsWithOffsetLimit(\n            this.address,\n            offset,\n            API_LIMIT,\n          );\n          total = response.total;\n          const txs = response.items.filter(\n            (tx) => tx.inclusionHeight <= initialBlock.height,\n          );\n          this.logger.debug(\n            `Found ${txs.length} transactions below the initial height with offset ${offset} and total number of transactions ${total}`,\n          );\n          if (txs.length > 0) await this.processTransactionBatch(txs);\n          offset += API_LIMIT;\n        }\n        round++;\n        offset = 0; // next round initial offset\n      }\n    });\n    const txCountAfterInit = await this.getTotalTxCount();\n    if (txCountAfterInit != txCountBeforeInit) {\n      throw Error(\n        'Total transaction count changed during initialization phase, the stored data is not valid',\n      );\n    }\n  };\n\n  /**\n   * Process a batch of transactions\n   * group txs into blocks and process them using `processTransactions`\n   * @param txs\n   */\n  private processTransactionBatch = async (txs: Array<ExtendedTransaction>) => {\n    txs = sortBy(txs, (tx) => tx.inclusionHeight);\n    const groupedTxs = groupBy(txs, (tx) => tx.blockId);\n    this.logger.debug(\n      `The transaction batch grouped to ${\n        Object.keys(groupedTxs).length\n      } blocks`,\n    );\n    for (const blockId in groupedTxs) {\n      const blockTxs = groupedTxs[blockId];\n      const block = { hash: blockId, height: blockTxs[0].inclusionHeight };\n      this.logger.debug(\n        `Processing transactions at height ${blockTxs[0].inclusionHeight}`,\n      );\n      const success = await this.processTransactions(blockTxs, block);\n      if (!success)\n        throw Error(\n          `Processing transactions failed at height ${blockTxs[0].inclusionHeight}`,\n        );\n    }\n  };\n\n  /**\n   * Initialize the extractor with retrial on any unexpected problem\n   * its the common part of initialize with Node and Explorer network\n   * @param job\n   */\n  private initWithRetrial = async (job: () => Promise<void>) => {\n    let trial = 1;\n    if (this.initialize) {\n      this.logger.debug(\n        `Initializing ${this.getId()} started, removing all existing data`,\n      );\n      await this.actions.removeAllData(this.getId());\n      while (trial <= RETRIAL_COUNT) {\n        try {\n          await job();\n          break;\n        } catch (e) {\n          this.logger.warn(\n            `Initialization for ${this.getId()} failed with error :${e}`,\n          );\n          if (trial == RETRIAL_COUNT)\n            throw Error(\n              `Initialization for ${this.getId()} failed after ${RETRIAL_COUNT} retrial`,\n            );\n          trial += 1;\n          this.logger.info(\n            `Trying again to initialize ${this.getId()} with trial step ${trial}`,\n          );\n        }\n      }\n      this.logger.info(\n        `Initialization completed successfully for ${this.getId()}`,\n      );\n    } else {\n      this.logger.info(`Initialization for ${this.getId()} is turned off`);\n    }\n  };\n\n  /**\n   * initialize extractor database with data created below the initial height\n   * ignore initialization if this feature is off\n   * try to get data multiple times to pass accidental network problems\n   * @param initialBlock\n   */\n  initializeBoxes: (initialBlock: BlockInfo) => Promise<void>;\n}\n"]}
@@ -1,14 +0,0 @@
1
- import { DataSource, EntityTarget } from '@rosen-bridge/extended-typeorm';
2
- import { AbstractLogger } from '@rosen-bridge/abstract-logger';
3
- import { AbstractErgoExtractorAction } from '../AbstractErgoExtractorAction';
4
- import { AbstractBoxData } from '../interfaces';
5
- import { AbstractErgoExtractorEntity } from '../AbstractErgoExtractorEntity';
6
- export declare abstract class AbstractInitializableErgoExtractorAction<ExtractedData extends AbstractBoxData, ExtractorEntity extends AbstractErgoExtractorEntity> extends AbstractErgoExtractorAction<ExtractedData, ExtractorEntity> {
7
- constructor(dataSource: DataSource, repo: EntityTarget<ExtractorEntity>, logger?: AbstractLogger);
8
- /**
9
- * remove all existing data for the extractor
10
- * @param extractorId
11
- */
12
- removeAllData: (extractorId: string) => Promise<void>;
13
- }
14
- //# sourceMappingURL=AbstractInitializableAction.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"AbstractInitializableAction.d.ts","sourceRoot":"","sources":["../../../../lib/ergo/initializable/AbstractInitializableAction.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,UAAU,EACV,YAAY,EAEb,MAAM,gCAAgC,CAAC;AACxC,OAAO,EAAE,cAAc,EAAE,MAAM,+BAA+B,CAAC;AAE/D,OAAO,EAAE,2BAA2B,EAAE,MAAM,gCAAgC,CAAC;AAC7E,OAAO,EAAE,eAAe,EAAE,MAAM,eAAe,CAAC;AAChD,OAAO,EAAE,2BAA2B,EAAE,MAAM,gCAAgC,CAAC;AAE7E,8BAAsB,wCAAwC,CAC5D,aAAa,SAAS,eAAe,EACrC,eAAe,SAAS,2BAA2B,CACnD,SAAQ,2BAA2B,CAAC,aAAa,EAAE,eAAe,CAAC;gBAEjE,UAAU,EAAE,UAAU,EACtB,IAAI,EAAE,YAAY,CAAC,eAAe,CAAC,EACnC,MAAM,CAAC,EAAE,cAAc;IAKzB;;;OAGG;IACH,aAAa,GAAU,aAAa,MAAM,mBAIxC;CACH"}