@rosen-bridge/address-extractor 4.0.0 → 5.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,63 +1,39 @@
1
1
  import { DataSource } from 'typeorm';
2
2
  import { AbstractLogger } from '@rosen-bridge/abstract-logger';
3
- import { Block } from '@rosen-bridge/abstract-extractor';
3
+ import { Block, AbstractInitializableErgoExtractorAction, SpendInfo } from '@rosen-bridge/abstract-extractor';
4
4
  import { ExtractedBox } from '../interfaces/types';
5
- export declare class BoxEntityAction {
5
+ export declare class BoxEntityAction extends AbstractInitializableErgoExtractorAction<ExtractedBox> {
6
6
  private readonly datasource;
7
7
  readonly logger: AbstractLogger;
8
8
  private readonly repository;
9
9
  constructor(dataSource: DataSource, logger?: AbstractLogger);
10
10
  /**
11
- * insert new box into database
12
- * @param box
13
- * @param extractor
14
- */
15
- insertBox: (box: ExtractedBox, extractor: string) => Promise<import("typeorm").InsertResult>;
16
- /**
17
- * Update an unspent box information in the database
18
- * @param box
11
+ * insert all extracted box data in an atomic transaction
12
+ * @param boxes
19
13
  * @param extractor
14
+ * @return success
20
15
  */
21
- updateBox: (box: ExtractedBox, extractor: string) => Promise<import("typeorm").UpdateResult>;
16
+ insertBoxes: (boxes: Array<ExtractedBox>, extractor: string) => Promise<boolean>;
22
17
  /**
23
- * It stores list of blocks in the dataSource with block id
24
- * @param boxes
25
- * @param spendBoxes
18
+ * update spending information of stored boxes
19
+ * chunk spendInfos to prevent large database queries
20
+ * @param spendInfos
26
21
  * @param block
27
22
  * @param extractor
28
23
  */
29
- storeBox: (boxes: Array<ExtractedBox>, block: Block, extractor: string) => Promise<boolean>;
24
+ spendBoxes: (spendInfos: Array<SpendInfo>, block: Block, extractor: string) => Promise<void>;
30
25
  /**
31
- * Update spendBlock and spendHeight of boxes spent on the block
32
- * @param spendIds
33
- * @param block
34
- * @param extractor
26
+ * remove all existing data for the extractor
27
+ * @param extractorId
35
28
  */
36
- spendBoxes: (spendIds: Array<string>, block: Block, extractor: string) => Promise<void>;
29
+ removeAllData: (extractorId: string) => Promise<void>;
37
30
  /**
38
- * delete boxes in specific block from database. if box spend in this block marked as unspent
39
- * and if created in this block remove it from database
31
+ * delete extracted data from a specific block
32
+ * if a box is spend in this block mark it as unspent
33
+ * if a box is created in this block remove it from database
40
34
  * @param block
41
- * @param extractor
35
+ * @param extractorId
42
36
  */
43
37
  deleteBlockBoxes: (block: string, extractor: string) => Promise<void>;
44
- /**
45
- * Returns all stored box ids
46
- */
47
- getAllBoxIds: (extractor: string) => Promise<Array<string>>;
48
- /**
49
- * Removes specified box
50
- * @param boxId
51
- * @param extractor
52
- */
53
- removeBox: (boxId: string, extractor: string) => Promise<import("typeorm").DeleteResult>;
54
- /**
55
- * Update the box spending information
56
- * @param boxId
57
- * @param extractor
58
- * @param blockId
59
- * @param blockHeight
60
- */
61
- updateSpendBlock: (boxId: string, extractor: string, blockId: string, blockHeight: number) => Promise<import("typeorm").UpdateResult>;
62
38
  }
63
39
  //# sourceMappingURL=boxAction.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"boxAction.d.ts","sourceRoot":"","sources":["../../lib/actions/boxAction.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAkB,MAAM,SAAS,CAAC;AAErD,OAAO,EAAE,cAAc,EAAe,MAAM,+BAA+B,CAAC;AAC5E,OAAO,EAAE,KAAK,EAAE,MAAM,kCAAkC,CAAC;AAGzD,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AAGnD,qBAAa,eAAe;IAC1B,OAAO,CAAC,QAAQ,CAAC,UAAU,CAAa;IACxC,QAAQ,CAAC,MAAM,EAAE,cAAc,CAAC;IAChC,OAAO,CAAC,QAAQ,CAAC,UAAU,CAAwB;gBAEvC,UAAU,EAAE,UAAU,EAAE,MAAM,CAAC,EAAE,cAAc;IAM3D;;;;OAIG;IACH,SAAS,QAAe,YAAY,aAAa,MAAM,6CASrD;IAEF;;;;OAIG;IACH,SAAS,QAAe,YAAY,aAAa,MAAM,6CAYrD;IAEF;;;;;;OAMG;IACH,QAAQ,UACC,MAAM,YAAY,CAAC,SACnB,KAAK,aACD,MAAM,sBA6CjB;IAEF;;;;;OAKG;IACH,UAAU,aACE,MAAM,MAAM,CAAC,SAChB,KAAK,aACD,MAAM,KAChB,QAAQ,IAAI,CAAC,CAoBd;IAEF;;;;;OAKG;IACH,gBAAgB,UAAiB,MAAM,aAAa,MAAM,mBAYxD;IAEF;;OAEG;IACH,YAAY,cAAqB,MAAM,KAAG,QAAQ,MAAM,MAAM,CAAC,CAAC,CAU9D;IAEF;;;;OAIG;IACH,SAAS,UAAiB,MAAM,aAAa,MAAM,6CAEjD;IAEF;;;;;;OAMG;IACH,gBAAgB,UACP,MAAM,aACF,MAAM,WACR,MAAM,eACF,MAAM,6CAMnB;CACH"}
1
+ {"version":3,"file":"boxAction.d.ts","sourceRoot":"","sources":["../../lib/actions/boxAction.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAkB,MAAM,SAAS,CAAC;AAErD,OAAO,EAAE,cAAc,EAAe,MAAM,+BAA+B,CAAC;AAC5E,OAAO,EACL,KAAK,EACL,wCAAwC,EACxC,SAAS,EAEV,MAAM,kCAAkC,CAAC;AAG1C,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AAGnD,qBAAa,eAAgB,SAAQ,wCAAwC,CAAC,YAAY,CAAC;IACzF,OAAO,CAAC,QAAQ,CAAC,UAAU,CAAa;IACxC,QAAQ,CAAC,MAAM,EAAE,cAAc,CAAC;IAChC,OAAO,CAAC,QAAQ,CAAC,UAAU,CAAwB;gBAEvC,UAAU,EAAE,UAAU,EAAE,MAAM,CAAC,EAAE,cAAc;IAO3D;;;;;OAKG;IACH,WAAW,UAAiB,MAAM,YAAY,CAAC,aAAa,MAAM,sBAmDhE;IAEF;;;;;;OAMG;IACH,UAAU,eACI,MAAM,SAAS,CAAC,SACrB,KAAK,aACD,MAAM,KAChB,QAAQ,IAAI,CAAC,CAqBd;IAEF;;;OAGG;IACH,aAAa,gBAAuB,MAAM,mBAExC;IAEF;;;;;;OAMG;IACH,gBAAgB,UAAiB,MAAM,aAAa,MAAM,mBAYxD;CACH"}
@@ -1,60 +1,33 @@
1
1
  import { In } from 'typeorm';
2
2
  import { chunk } from 'lodash-es';
3
3
  import { DummyLogger } from '@rosen-bridge/abstract-logger';
4
+ import { AbstractInitializableErgoExtractorAction, DB_CHUNK_SIZE, } from '@rosen-bridge/abstract-extractor';
4
5
  import { BoxEntity } from '../entities/boxEntity';
5
- import { dbIdChunkSize } from '../constants';
6
- export class BoxEntityAction {
6
+ import { JsonBI } from '../utils';
7
+ export class BoxEntityAction extends AbstractInitializableErgoExtractorAction {
7
8
  datasource;
8
9
  logger;
9
10
  repository;
10
11
  constructor(dataSource, logger) {
12
+ super();
11
13
  this.datasource = dataSource;
12
14
  this.logger = logger ? logger : new DummyLogger();
13
15
  this.repository = dataSource.getRepository(BoxEntity);
14
16
  }
15
17
  /**
16
- * insert new box into database
17
- * @param box
18
- * @param extractor
19
- */
20
- insertBox = async (box, extractor) => {
21
- return this.repository.insert({
22
- address: box.address,
23
- boxId: box.boxId,
24
- createBlock: box.blockId,
25
- creationHeight: box.height,
26
- serialized: box.serialized,
27
- extractor: extractor,
28
- });
29
- };
30
- /**
31
- * Update an unspent box information in the database
32
- * @param box
33
- * @param extractor
34
- */
35
- updateBox = async (box, extractor) => {
36
- return this.repository.update({ boxId: box.boxId, extractor: extractor }, {
37
- address: box.address,
38
- createBlock: box.blockId,
39
- creationHeight: box.height,
40
- serialized: box.serialized,
41
- spendBlock: null,
42
- spendHeight: 0,
43
- });
44
- };
45
- /**
46
- * It stores list of blocks in the dataSource with block id
18
+ * insert all extracted box data in an atomic transaction
47
19
  * @param boxes
48
- * @param spendBoxes
49
- * @param block
50
20
  * @param extractor
21
+ * @return success
51
22
  */
52
- storeBox = async (boxes, block, extractor) => {
23
+ insertBoxes = async (boxes, extractor) => {
53
24
  const boxIds = boxes.map((item) => item.boxId);
54
25
  const dbBoxes = await this.datasource.getRepository(BoxEntity).findBy({
55
26
  boxId: In(boxIds),
56
27
  extractor: extractor,
57
28
  });
29
+ if (dbBoxes.length > 0)
30
+ this.logger.debug(`Found stored boxes with same boxId`, dbBoxes);
58
31
  let success = true;
59
32
  const queryRunner = this.datasource.createQueryRunner();
60
33
  await queryRunner.connect();
@@ -65,22 +38,23 @@ export class BoxEntityAction {
65
38
  const entity = {
66
39
  address: box.address,
67
40
  boxId: box.boxId,
68
- createBlock: block.hash,
69
- creationHeight: block.height,
70
- spendBlock: undefined,
41
+ createBlock: box.blockId,
42
+ creationHeight: box.height,
43
+ spendBlock: box.spendBlock,
44
+ spendHeight: box.spendHeight,
71
45
  serialized: box.serialized,
72
46
  extractor: extractor,
73
47
  };
74
48
  const dbBox = dbBoxes.filter((item) => item.boxId === box.boxId);
75
49
  if (dbBox.length > 0) {
76
50
  this.logger.info(`Updating box ${box.boxId} and extractor ${extractor}`);
77
- this.logger.debug(`Entity: ${JSON.stringify(entity)}`);
78
51
  await repository.update({ id: dbBoxes[0].id }, entity);
52
+ this.logger.debug(`Updated entity is [${JsonBI.stringify(box)}], and stored similar box is [${JsonBI.stringify(dbBox)}]`);
79
53
  }
80
54
  else {
81
55
  this.logger.info(`Storing box ${box.boxId}`);
82
- this.logger.debug(JSON.stringify(entity));
83
56
  await repository.insert(entity);
57
+ this.logger.debug(`Stored ${JsonBI.stringify(entity)}`);
84
58
  }
85
59
  }
86
60
  await queryRunner.commitTransaction();
@@ -96,18 +70,20 @@ export class BoxEntityAction {
96
70
  return success;
97
71
  };
98
72
  /**
99
- * Update spendBlock and spendHeight of boxes spent on the block
100
- * @param spendIds
73
+ * update spending information of stored boxes
74
+ * chunk spendInfos to prevent large database queries
75
+ * @param spendInfos
101
76
  * @param block
102
77
  * @param extractor
103
78
  */
104
- spendBoxes = async (spendIds, block, extractor) => {
105
- const spendIdChunks = chunk(spendIds, dbIdChunkSize);
106
- for (const spendIdChunk of spendIdChunks) {
107
- const updateResult = await this.repository.update({ boxId: In(spendIdChunk), extractor: extractor }, { spendBlock: block.hash, spendHeight: block.height });
79
+ spendBoxes = async (spendInfos, block, extractor) => {
80
+ const spendInfoChunks = chunk(spendInfos, DB_CHUNK_SIZE);
81
+ for (const spendInfoChunk of spendInfoChunks) {
82
+ const boxIds = spendInfoChunk.map((info) => info.boxId);
83
+ const updateResult = await this.repository.update({ boxId: In(boxIds), extractor: extractor }, { spendBlock: block.hash, spendHeight: block.height });
108
84
  if (updateResult.affected && updateResult.affected > 0) {
109
85
  const spentRows = await this.repository.findBy({
110
- boxId: In(spendIdChunk),
86
+ boxId: In(boxIds),
111
87
  spendBlock: block.hash,
112
88
  });
113
89
  for (const row of spentRows) {
@@ -117,10 +93,18 @@ export class BoxEntityAction {
117
93
  }
118
94
  };
119
95
  /**
120
- * delete boxes in specific block from database. if box spend in this block marked as unspent
121
- * and if created in this block remove it from database
96
+ * remove all existing data for the extractor
97
+ * @param extractorId
98
+ */
99
+ removeAllData = async (extractorId) => {
100
+ await this.repository.delete({ extractor: extractorId });
101
+ };
102
+ /**
103
+ * delete extracted data from a specific block
104
+ * if a box is spend in this block mark it as unspent
105
+ * if a box is created in this block remove it from database
122
106
  * @param block
123
- * @param extractor
107
+ * @param extractorId
124
108
  */
125
109
  deleteBlockBoxes = async (block, extractor) => {
126
110
  this.logger.info(`Deleting boxes in block ${block} and extractor ${extractor}`);
@@ -130,37 +114,5 @@ export class BoxEntityAction {
130
114
  });
131
115
  await this.repository.update({ spendBlock: block, extractor: extractor }, { spendBlock: null, spendHeight: 0 });
132
116
  };
133
- /**
134
- * Returns all stored box ids
135
- */
136
- getAllBoxIds = async (extractor) => {
137
- const boxIds = await this.repository.find({
138
- select: {
139
- boxId: true,
140
- },
141
- where: {
142
- extractor: extractor,
143
- },
144
- });
145
- return boxIds.map((item) => item.boxId);
146
- };
147
- /**
148
- * Removes specified box
149
- * @param boxId
150
- * @param extractor
151
- */
152
- removeBox = async (boxId, extractor) => {
153
- return await this.repository.delete({ boxId: boxId, extractor: extractor });
154
- };
155
- /**
156
- * Update the box spending information
157
- * @param boxId
158
- * @param extractor
159
- * @param blockId
160
- * @param blockHeight
161
- */
162
- updateSpendBlock = async (boxId, extractor, blockId, blockHeight) => {
163
- return await this.repository.update({ boxId: boxId, extractor: extractor }, { spendBlock: blockId, spendHeight: blockHeight });
164
- };
165
117
  }
166
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"boxAction.js","sourceRoot":"","sources":["../../lib/actions/boxAction.ts"],"names":[],"mappings":"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","sourcesContent":["import { DataSource, In, Repository } from 'typeorm';\nimport { chunk } from 'lodash-es';\nimport { AbstractLogger, DummyLogger } from '@rosen-bridge/abstract-logger';\nimport { Block } from '@rosen-bridge/abstract-extractor';\n\nimport { BoxEntity } from '../entities/boxEntity';\nimport { ExtractedBox } from '../interfaces/types';\nimport { dbIdChunkSize } from '../constants';\n\nexport class BoxEntityAction {\n  private readonly datasource: DataSource;\n  readonly logger: AbstractLogger;\n  private readonly repository: Repository<BoxEntity>;\n\n  constructor(dataSource: DataSource, logger?: AbstractLogger) {\n    this.datasource = dataSource;\n    this.logger = logger ? logger : new DummyLogger();\n    this.repository = dataSource.getRepository(BoxEntity);\n  }\n\n  /**\n   * insert new box into database\n   * @param box\n   * @param extractor\n   */\n  insertBox = async (box: ExtractedBox, extractor: string) => {\n    return this.repository.insert({\n      address: box.address,\n      boxId: box.boxId,\n      createBlock: box.blockId,\n      creationHeight: box.height,\n      serialized: box.serialized,\n      extractor: extractor,\n    });\n  };\n\n  /**\n   * Update an unspent box information in the database\n   * @param box\n   * @param extractor\n   */\n  updateBox = async (box: ExtractedBox, extractor: string) => {\n    return this.repository.update(\n      { boxId: box.boxId, extractor: extractor },\n      {\n        address: box.address,\n        createBlock: box.blockId,\n        creationHeight: box.height,\n        serialized: box.serialized,\n        spendBlock: null,\n        spendHeight: 0,\n      }\n    );\n  };\n\n  /**\n   * It stores list of blocks in the dataSource with block id\n   * @param boxes\n   * @param spendBoxes\n   * @param block\n   * @param extractor\n   */\n  storeBox = async (\n    boxes: Array<ExtractedBox>,\n    block: Block,\n    extractor: string\n  ) => {\n    const boxIds = boxes.map((item) => item.boxId);\n    const dbBoxes = await this.datasource.getRepository(BoxEntity).findBy({\n      boxId: In(boxIds),\n      extractor: extractor,\n    });\n    let success = true;\n    const queryRunner = this.datasource.createQueryRunner();\n    await queryRunner.connect();\n    await queryRunner.startTransaction();\n    const repository = await queryRunner.manager.getRepository(BoxEntity);\n    try {\n      for (const box of boxes) {\n        const entity = {\n          address: box.address,\n          boxId: box.boxId,\n          createBlock: block.hash,\n          creationHeight: block.height,\n          spendBlock: undefined,\n          serialized: box.serialized,\n          extractor: extractor,\n        };\n        const dbBox = dbBoxes.filter((item) => item.boxId === box.boxId);\n        if (dbBox.length > 0) {\n          this.logger.info(\n            `Updating box ${box.boxId} and extractor ${extractor}`\n          );\n          this.logger.debug(`Entity: ${JSON.stringify(entity)}`);\n          await repository.update({ id: dbBoxes[0].id }, entity);\n        } else {\n          this.logger.info(`Storing box ${box.boxId}`);\n          this.logger.debug(JSON.stringify(entity));\n          await repository.insert(entity);\n        }\n      }\n      await queryRunner.commitTransaction();\n    } catch (e) {\n      this.logger.error(`An error occurred during store boxes action: ${e}`);\n      await queryRunner.rollbackTransaction();\n      success = false;\n    } finally {\n      await queryRunner.release();\n    }\n    return success;\n  };\n\n  /**\n   * Update spendBlock and spendHeight of boxes spent on the block\n   * @param spendIds\n   * @param block\n   * @param extractor\n   */\n  spendBoxes = async (\n    spendIds: Array<string>,\n    block: Block,\n    extractor: string\n  ): Promise<void> => {\n    const spendIdChunks = chunk(spendIds, dbIdChunkSize);\n    for (const spendIdChunk of spendIdChunks) {\n      const updateResult = await this.repository.update(\n        { boxId: In(spendIdChunk), extractor: extractor },\n        { spendBlock: block.hash, spendHeight: block.height }\n      );\n\n      if (updateResult.affected && updateResult.affected > 0) {\n        const spentRows = await this.repository.findBy({\n          boxId: In(spendIdChunk),\n          spendBlock: block.hash,\n        });\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  };\n\n  /**\n   * delete boxes in specific block from database. if box spend in this block marked as unspent\n   * and if created in this block remove it from database\n   * @param block\n   * @param extractor\n   */\n  deleteBlockBoxes = async (block: string, extractor: string) => {\n    this.logger.info(\n      `Deleting boxes in block ${block} and extractor ${extractor}`\n    );\n    await this.repository.delete({\n      extractor: extractor,\n      createBlock: block,\n    });\n    await this.repository.update(\n      { spendBlock: block, extractor: extractor },\n      { spendBlock: null, spendHeight: 0 }\n    );\n  };\n\n  /**\n   *  Returns all stored box ids\n   */\n  getAllBoxIds = async (extractor: string): Promise<Array<string>> => {\n    const boxIds = await this.repository.find({\n      select: {\n        boxId: true,\n      },\n      where: {\n        extractor: extractor,\n      },\n    });\n    return boxIds.map((item: { boxId: string }) => item.boxId);\n  };\n\n  /**\n   * Removes specified box\n   * @param boxId\n   * @param extractor\n   */\n  removeBox = async (boxId: string, extractor: string) => {\n    return await this.repository.delete({ boxId: boxId, extractor: extractor });\n  };\n\n  /**\n   * Update the box spending information\n   * @param boxId\n   * @param extractor\n   * @param blockId\n   * @param blockHeight\n   */\n  updateSpendBlock = async (\n    boxId: string,\n    extractor: string,\n    blockId: string,\n    blockHeight: number\n  ) => {\n    return await this.repository.update(\n      { boxId: boxId, extractor: extractor },\n      { spendBlock: blockId, spendHeight: blockHeight }\n    );\n  };\n}\n"]}
118
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"boxAction.js","sourceRoot":"","sources":["../../lib/actions/boxAction.ts"],"names":[],"mappings":"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","sourcesContent":["import { DataSource, In, Repository } from 'typeorm';\nimport { chunk } from 'lodash-es';\nimport { AbstractLogger, DummyLogger } from '@rosen-bridge/abstract-logger';\nimport {\n  Block,\n  AbstractInitializableErgoExtractorAction,\n  SpendInfo,\n  DB_CHUNK_SIZE,\n} from '@rosen-bridge/abstract-extractor';\n\nimport { BoxEntity } from '../entities/boxEntity';\nimport { ExtractedBox } from '../interfaces/types';\nimport { JsonBI } from '../utils';\n\nexport class BoxEntityAction extends AbstractInitializableErgoExtractorAction<ExtractedBox> {\n  private readonly datasource: DataSource;\n  readonly logger: AbstractLogger;\n  private readonly repository: Repository<BoxEntity>;\n\n  constructor(dataSource: DataSource, logger?: AbstractLogger) {\n    super();\n    this.datasource = dataSource;\n    this.logger = logger ? logger : new DummyLogger();\n    this.repository = dataSource.getRepository(BoxEntity);\n  }\n\n  /**\n   * insert all extracted box data in an atomic transaction\n   * @param boxes\n   * @param extractor\n   * @return success\n   */\n  insertBoxes = async (boxes: Array<ExtractedBox>, extractor: string) => {\n    const boxIds = boxes.map((item) => item.boxId);\n    const dbBoxes = await this.datasource.getRepository(BoxEntity).findBy({\n      boxId: In(boxIds),\n      extractor: extractor,\n    });\n    if (dbBoxes.length > 0)\n      this.logger.debug(`Found stored boxes with same boxId`, dbBoxes);\n    let success = true;\n    const queryRunner = this.datasource.createQueryRunner();\n    await queryRunner.connect();\n    await queryRunner.startTransaction();\n    const repository = await queryRunner.manager.getRepository(BoxEntity);\n    try {\n      for (const box of boxes) {\n        const entity = {\n          address: box.address,\n          boxId: box.boxId,\n          createBlock: box.blockId,\n          creationHeight: box.height,\n          spendBlock: box.spendBlock,\n          spendHeight: box.spendHeight,\n          serialized: box.serialized,\n          extractor: extractor,\n        };\n        const dbBox = dbBoxes.filter((item) => item.boxId === box.boxId);\n        if (dbBox.length > 0) {\n          this.logger.info(\n            `Updating box ${box.boxId} and extractor ${extractor}`\n          );\n          await repository.update({ id: dbBoxes[0].id }, entity);\n          this.logger.debug(\n            `Updated entity is [${JsonBI.stringify(\n              box\n            )}], and stored similar box is [${JsonBI.stringify(dbBox)}]`\n          );\n        } else {\n          this.logger.info(`Storing box ${box.boxId}`);\n          await repository.insert(entity);\n          this.logger.debug(`Stored ${JsonBI.stringify(entity)}`);\n        }\n      }\n      await queryRunner.commitTransaction();\n    } catch (e) {\n      this.logger.error(`An error occurred during store boxes action: ${e}`);\n      await queryRunner.rollbackTransaction();\n      success = false;\n    } finally {\n      await queryRunner.release();\n    }\n    return success;\n  };\n\n  /**\n   * update spending information of stored boxes\n   * chunk spendInfos to prevent large database queries\n   * @param spendInfos\n   * @param block\n   * @param extractor\n   */\n  spendBoxes = async (\n    spendInfos: Array<SpendInfo>,\n    block: Block,\n    extractor: string\n  ): Promise<void> => {\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        { boxId: In(boxIds), extractor: extractor },\n        { spendBlock: block.hash, spendHeight: block.height }\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        });\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  };\n\n  /**\n   * remove all existing data for the extractor\n   * @param extractorId\n   */\n  removeAllData = async (extractorId: string) => {\n    await this.repository.delete({ extractor: extractorId });\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 extractorId\n   */\n  deleteBlockBoxes = async (block: string, extractor: string) => {\n    this.logger.info(\n      `Deleting boxes in block ${block} and extractor ${extractor}`\n    );\n    await this.repository.delete({\n      extractor: extractor,\n      createBlock: block,\n    });\n    await this.repository.update(\n      { spendBlock: block, extractor: extractor },\n      { spendBlock: null, spendHeight: 0 }\n    );\n  };\n}\n"]}
@@ -1,158 +1,51 @@
1
1
  import { DataSource } from 'typeorm';
2
2
  import * as ergoLib from 'ergo-lib-wasm-nodejs';
3
- import { InitialInfo } from '@rosen-bridge/scanner';
4
3
  import { AbstractLogger } from '@rosen-bridge/abstract-logger';
5
- import { Transaction } from '@rosen-bridge/scanner';
6
- import { ItemsOutputInfo, OutputInfo } from '@rosen-clients/ergo-explorer/dist/src/v1/types';
7
- import { AbstractExtractor, Block } from '@rosen-bridge/abstract-extractor';
4
+ import { AbstractInitializableErgoExtractor, ErgoBox, AbstractNetwork, OutputBox } from '@rosen-bridge/abstract-extractor';
8
5
  import { BoxEntityAction } from '../actions/boxAction';
9
6
  import { ExtractedBox } from '../interfaces/types';
10
- export declare class ErgoUTXOExtractor implements AbstractExtractor<Transaction> {
11
- readonly logger: AbstractLogger;
12
- private readonly dataSource;
7
+ import { ErgoNetworkType } from '../../../abstract-extractor/lib/ergo/interfaces';
8
+ export declare class ErgoUTXOExtractor extends AbstractInitializableErgoExtractor<ExtractedBox> {
13
9
  readonly actions: BoxEntityAction;
14
10
  private readonly id;
15
11
  private readonly networkType;
12
+ private readonly address?;
16
13
  private readonly ergoTree?;
17
14
  private readonly tokens;
18
- readonly api: {
19
- v0: {
20
- getApiV0AddressesP1: (p1: string, params?: import("@rosen-clients/ergo-explorer/dist/src/v0/types").GetApiV0AddressesP1Params | undefined) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").AddressInfo>;
21
- getApiV0AddressesP1Transactions: (p1: string, params?: import("@rosen-clients/ergo-explorer/dist/src/v0/types").GetApiV0AddressesP1TransactionsParams | undefined) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").ItemsTransactionInfo>;
22
- getApiV0AddressesAssetholdersP1: (p1: string, params?: import("@rosen-clients/ergo-explorer/dist/src/v0/types").GetApiV0AddressesAssetholdersP1Params | undefined) => Promise<string[]>;
23
- getApiV0AddressesBalances: (params?: import("@rosen-clients/ergo-explorer/dist/src/v0/types").GetApiV0AddressesBalancesParams | undefined) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").ItemsBalanceInfo>;
24
- getApiV0AssetsIssuingboxes: (params?: import("@rosen-clients/ergo-explorer/dist/src/v0/types").GetApiV0AssetsIssuingboxesParams | undefined) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").ItemsOutputInfo>;
25
- getApiV0AssetsP1Issuingbox: (p1: string) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").OutputInfo[]>;
26
- getApiV0Blocks: (params?: import("@rosen-clients/ergo-explorer/dist/src/v0/types").GetApiV0BlocksParams | undefined) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").ItemsBlockInfo>;
27
- getApiV0BlocksP1: (p1: string) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").BlockSummary>;
28
- getApiV0BlocksAtP1: (p1: number) => Promise<string[]>;
29
- getApiV0DexTokensP1Unspentsellorders: (p1: string, params?: import("@rosen-clients/ergo-explorer/dist/src/v0/types").GetApiV0DexTokensP1UnspentsellordersParams | undefined) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").DexSellOrderInfo[]>;
30
- getApiV0DexTokensP1Unspentbuyorders: (p1: string, params?: import("@rosen-clients/ergo-explorer/dist/src/v0/types").GetApiV0DexTokensP1UnspentbuyordersParams | undefined) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").DexBuyOrderInfo[]>;
31
- getApiV0TransactionsP1: (p1: string) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").TransactionSummary>;
32
- getApiV0TransactionsUnconfirmedP1: (p1: string) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").UTransactionSummary>;
33
- /**
34
- * gets block id and transactions corresponding to the block and saves if they are valid rosen
35
- * transactions and in case of success return true and in case of failure returns false
36
- * @param txs
37
- * @param block
38
- */
39
- getApiV0TransactionsUnconfirmedByaddressP1: (p1: string, params?: import("@rosen-clients/ergo-explorer/dist/src/v0/types").GetApiV0TransactionsUnconfirmedByaddressP1Params | undefined) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").ItemsUTransactionInfo>;
40
- getApiV0TransactionsUnconfirmed: (params?: import("@rosen-clients/ergo-explorer/dist/src/v0/types").GetApiV0TransactionsUnconfirmedParams | undefined) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").ItemsUTransactionInfo>;
41
- getApiV0TransactionsSinceP1: (p1: number, params?: import("@rosen-clients/ergo-explorer/dist/src/v0/types").GetApiV0TransactionsSinceP1Params | undefined) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").TransactionInfo[]>;
42
- postApiV0TransactionsSend: (postApiV0TransactionsSendBody: import("@rosen-clients/ergo-explorer/dist/src/v0/types").PostApiV0TransactionsSendBody) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").TxIdResponse>;
43
- getApiV0TransactionsBoxesP1: (p1: string) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").OutputInfo>;
44
- getApiV0TransactionsBoxesByergotreeP1: (p1: string) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").OutputInfo[]>;
45
- getApiV0TransactionsBoxesByergotreeUnspentP1: (p1: string) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").OutputInfo[]>;
46
- getApiV0TransactionsBoxesByaddressP1: (p1: string) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").OutputInfo[]>;
47
- getApiV0TransactionsBoxesByaddressUnspentP1: (p1: string) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").OutputInfo[]>;
48
- getApiV0Info: () => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").BlockChainInfo>;
49
- getApiV0InfoSupply: () => Promise<string>;
50
- getApiV0Stats: () => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").StatsSummary>;
51
- getApiV0DocsOpenapi: () => Promise<string>;
52
- };
53
- v1: {
54
- getApiV1TransactionsP1: (p1: string) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v1/types").TransactionInfo>;
55
- getApiV1TransactionsByinputsscripttemplatehashP1: (p1: string, params?: import("@rosen-clients/ergo-explorer/dist/src/v1/types").GetApiV1TransactionsByinputsscripttemplatehashP1Params | undefined) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v1/types").ItemsTransactionInfo>;
56
- getApiV1TransactionsByglobalindexStream: (params: import("@rosen-clients/ergo-explorer/dist/src/v1/types").GetApiV1TransactionsByglobalindexStreamParams) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v1/types").ListTransactionInfo>;
57
- getApiV1BoxesUnspentBylastepochsStream: (params: import("@rosen-clients/ergo-explorer/dist/src/v1/types").GetApiV1BoxesUnspentBylastepochsStreamParams) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v1/types").ListOutputInfo>;
58
- getApiV1BoxesUnspentByglobalindexStream: (params: import("@rosen-clients/ergo-explorer/dist/src/v1/types").GetApiV1BoxesUnspentByglobalindexStreamParams) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v1/types").ListOutputInfo>;
59
- getApiV1BoxesUnspentStream: (params: import("@rosen-clients/ergo-explorer/dist/src/v1/types").GetApiV1BoxesUnspentStreamParams) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v1/types").ListOutputInfo>;
60
- getApiV1BoxesByergotreetemplatehashP1Stream: (p1: string, params: import("@rosen-clients/ergo-explorer/dist/src/v1/types").GetApiV1BoxesByergotreetemplatehashP1StreamParams) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v1/types").ListOutputInfo>;
61
- getApiV1BoxesUnspentByergotreetemplatehashP1Stream: (p1: string, params: import("@rosen-clients/ergo-explorer/dist/src/v1/types").GetApiV1BoxesUnspentByergotreetemplatehashP1StreamParams) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v1/types").ListOutputInfo>;
62
- getApiV1BoxesUnspentBytokenidP1: (p1: string, params?: import("@rosen-clients/ergo-explorer/dist/src/v1/types").GetApiV1BoxesUnspentBytokenidP1Params | undefined) => Promise<ItemsOutputInfo>;
63
- getApiV1BoxesBytokenidP1: (p1: string, params?: import("@rosen-clients/ergo-explorer/dist/src/v1/types").GetApiV1BoxesBytokenidP1Params | undefined) => Promise<ItemsOutputInfo>;
64
- getApiV1BoxesP1: (p1: string) => Promise<OutputInfo>;
65
- getApiV1BoxesByergotreeP1: (p1: string, params?: import("@rosen-clients/ergo-explorer/dist/src/v1/types").GetApiV1BoxesByergotreeP1Params | undefined) => Promise<ItemsOutputInfo>;
66
- getApiV1BoxesByergotreetemplatehashP1: (p1: string, params?: import("@rosen-clients/ergo-explorer/dist/src/v1/types").GetApiV1BoxesByergotreetemplatehashP1Params | undefined) => Promise<ItemsOutputInfo>;
67
- getApiV1BoxesUnspentByergotreeP1: (p1: string, params?: import("@rosen-clients/ergo-explorer/dist/src/v1/types").GetApiV1BoxesUnspentByergotreeP1Params | undefined) => Promise<ItemsOutputInfo>;
68
- getApiV1BoxesUnspentByergotreetemplatehashP1: (p1: string, params?: import("@rosen-clients/ergo-explorer/dist/src/v1/types").GetApiV1BoxesUnspentByergotreetemplatehashP1Params | undefined) => Promise<ItemsOutputInfo>;
69
- getApiV1BoxesByaddressP1: (p1: string, params?: import("@rosen-clients/ergo-explorer/dist/src/v1/types").GetApiV1BoxesByaddressP1Params | undefined) => Promise<ItemsOutputInfo>;
70
- getApiV1BoxesUnspentUnconfirmedByaddressP1: (p1: string, params?: import("@rosen-clients/ergo-explorer/dist/src/v1/types").GetApiV1BoxesUnspentUnconfirmedByaddressP1Params | undefined) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v1/types").MOutputInfo[]>;
71
- getApiV1BoxesUnspentByaddressP1: (p1: string, params?: import("@rosen-clients/ergo-explorer/dist/src/v1/types").GetApiV1BoxesUnspentByaddressP1Params | undefined) => Promise<ItemsOutputInfo>;
72
- getApiV1BoxesByglobalindexStream: (params: import("@rosen-clients/ergo-explorer/dist/src/v1/types").GetApiV1BoxesByglobalindexStreamParams) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v1/types").ListOutputInfo>;
73
- postApiV1BoxesUnspentSearchUnion: (boxAssetsQuery: import("@rosen-clients/ergo-explorer/dist/src/v1/types").BoxAssetsQuery, params?: import("@rosen-clients/ergo-explorer/dist/src/v1/types").PostApiV1BoxesUnspentSearchUnionParams | undefined) => Promise<ItemsOutputInfo>;
74
- postApiV1BoxesUnspentSearch: (boxQuery: import("@rosen-clients/ergo-explorer/dist/src/v1/types").BoxQuery, params?: import("@rosen-clients/ergo-explorer/dist/src/v1/types").PostApiV1BoxesUnspentSearchParams | undefined) => Promise<ItemsOutputInfo>;
75
- postApiV1BoxesSearch: (boxQuery: import("@rosen-clients/ergo-explorer/dist/src/v1/types").BoxQuery, params?: import("@rosen-clients/ergo-explorer/dist/src/v1/types").PostApiV1BoxesSearchParams | undefined) => Promise<ItemsOutputInfo>;
76
- getApiV1Tokens: (params?: import("@rosen-clients/ergo-explorer/dist/src/v1/types").GetApiV1TokensParams | undefined) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v1/types").ItemsTokenInfo>;
77
- getApiV1TokensSearch: (params: import("@rosen-clients/ergo-explorer/dist/src/v1/types").GetApiV1TokensSearchParams) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v1/types").ItemsTokenInfo>;
78
- getApiV1TokensBysymbolP1: (p1: string) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v1/types").TokenInfo[]>;
79
- getApiV1TokensP1: (p1: string) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v1/types").TokenInfo>;
80
- getApiV1Assets: (params?: import("@rosen-clients/ergo-explorer/dist/src/v1/types").GetApiV1AssetsParams | undefined) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v1/types").ItemsTokenInfo>;
81
- getApiV1AssetsSearchBytokenid: (params: import("@rosen-clients/ergo-explorer/dist/src/v1/types").GetApiV1AssetsSearchBytokenidParams) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v1/types").ItemsAssetInfo>;
82
- getApiV1EpochsParams: () => Promise<import("@rosen-clients/ergo-explorer/dist/src/v1/types").EpochInfo>;
83
- getApiV1AddressesP1Transactions: (p1: string, params?: import("@rosen-clients/ergo-explorer/dist/src/v1/types").GetApiV1AddressesP1TransactionsParams | undefined) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v1/types").ItemsTransactionInfo>;
84
- getApiV1AddressesP1BalanceConfirmed: (p1: string, params?: import("@rosen-clients/ergo-explorer/dist/src/v1/types").GetApiV1AddressesP1BalanceConfirmedParams | undefined) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v1/types").Balance>;
85
- getApiV1AddressesP1BalanceTotal: (p1: string) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v1/types").TotalBalance>;
86
- getApiV1Blocks: (params?: import("@rosen-clients/ergo-explorer/dist/src/v1/types").GetApiV1BlocksParams | undefined) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v1/types").ItemsBlockInfo>;
87
- getApiV1BlocksP1: (p1: string) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v1/types").BlockSummary>;
88
- getApiV1BlocksHeaders: (params?: import("@rosen-clients/ergo-explorer/dist/src/v1/types").GetApiV1BlocksHeadersParams | undefined) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v1/types").ItemsBlockHeader>;
89
- getApiV1BlocksByglobalindexStream: (params: import("@rosen-clients/ergo-explorer/dist/src/v1/types").GetApiV1BlocksByglobalindexStreamParams) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v1/types").ListBlockInfo>;
90
- getApiV1BlocksStreamSummary: (params?: import("@rosen-clients/ergo-explorer/dist/src/v1/types").GetApiV1BlocksStreamSummaryParams | undefined) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v1/types").ListBlockSummaryV1>;
91
- postApiV1MempoolTransactionsSubmit: (postApiV1MempoolTransactionsSubmitBody: import("@rosen-clients/ergo-explorer/dist/src/v1/types").PostApiV1MempoolTransactionsSubmitBody) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v1/types").TxIdResponse>;
92
- getApiV1MempoolTransactionsByaddressP1: (p1: string, params?: import("@rosen-clients/ergo-explorer/dist/src/v1/types").GetApiV1MempoolTransactionsByaddressP1Params | undefined) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v1/types").ItemsUTransactionInfo>;
93
- getApiV1MempoolBoxesUnspent: () => Promise<import("@rosen-clients/ergo-explorer/dist/src/v1/types").ListOutputInfo>;
94
- getApiV1Info: () => Promise<import("@rosen-clients/ergo-explorer/dist/src/v1/types").NetworkState>;
95
- getApiV1Networkstate: () => Promise<import("@rosen-clients/ergo-explorer/dist/src/v1/types").NetworkState>;
96
- getApiV1Networkstats: () => Promise<import("@rosen-clients/ergo-explorer/dist/src/v1/types").NetworkStats>;
97
- postApiV1ErgotreeConvert: (ergoTreeConversionRequest: import("@rosen-clients/ergo-explorer/dist/src/v1/types").ErgoTreeConversionRequest) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v1/types").ErgoTreeHuman>;
98
- };
99
- };
100
- constructor(dataSource: DataSource, id: string, networkType: ergoLib.NetworkPrefix, explorerUrl: string, address?: string, tokens?: Array<string>, logger?: AbstractLogger);
15
+ network: AbstractNetwork;
16
+ constructor(dataSource: DataSource, id: string, networkType: ergoLib.NetworkPrefix, url: string, type: ErgoNetworkType, address?: string, tokens?: Array<string>, logger?: AbstractLogger, initialize?: boolean);
101
17
  /**
102
18
  * get Id for current extractor
103
19
  */
104
20
  getId: () => string;
105
21
  /**
106
- * gets block id and transactions corresponding to the block and saves if they are valid rosen
107
- * transactions and in case of success return true and in case of failure returns false
108
- * @param txs
109
- * @param block
110
- */
111
- processTransactions: (txs: Array<Transaction>, block: Block) => Promise<boolean>;
112
- /**
113
- * fork one block and remove all stored information for this block
114
- * @param hash: block hash
115
- */
116
- forkBlock: (hash: string) => Promise<void>;
117
- /**
118
- * Initializes the database with older boxes related to the address
119
- */
120
- initializeBoxes: (initialBlock: InitialInfo) => Promise<void>;
121
- /**
122
- * Validate all remaining boxes in the database
123
- * update the correct ones and remove the invalid ones
124
- * @param unchangedStoredBoxIds
125
- * @param initialHeight
126
- */
127
- validateOldStoredBoxes: (unchangedStoredBoxIds: Array<string>, initialHeight: number) => Promise<void>;
128
- /**
129
- * Return extracted information of a box
130
- * @param boxId
131
- */
132
- getBoxInfoWithBoxId: (boxId: string) => Promise<ExtractedBox | undefined>;
133
- /**
134
- * Get unspent boxes created bellow the initial height
135
- * @param initialHeight
136
- * @returns
22
+ * check proper data format in the box
23
+ * @param box
24
+ * @return true if the box has the required data and false otherwise
137
25
  */
138
- getUnspentBoxes: (initialHeight: number) => Promise<Array<ExtractedBox>>;
26
+ hasData: (box: OutputBox) => boolean;
139
27
  /**
140
28
  * Returns block information of tx
141
29
  * @param txId
142
30
  */
143
- getTxBlock: (txId: string) => Promise<{
144
- id: string;
145
- height: number;
146
- }>;
31
+ getTxBlock: (txId: string) => Promise<import("@rosen-bridge/abstract-extractor").BlockInfo>;
147
32
  /**
148
- * Extract needed information for storing in database from api json outputs
149
- * @param boxes
33
+ * extract box data to proper format (not including spending information)
34
+ * @param box
35
+ * @param blockId box inclusion block hash
36
+ * @param height box inclusion block height
37
+ * @return extracted data in proper format
150
38
  */
151
- extractBoxData: (boxes: Array<OutputInfo>) => Promise<ExtractedBox[]>;
39
+ extractBoxData: (box: OutputBox, blockId: string, height: number) => Omit<ExtractedBox, 'spendBlock' | 'spendHeight'> | undefined;
152
40
  /**
153
- * Returns true if box has the required token and false otherwise
154
- * @param box
41
+ * return init required boxes with offset limit
42
+ * @param offset
43
+ * @param limit
44
+ * @return boxes in batch
155
45
  */
156
- boxHasToken: (box: OutputInfo) => boolean;
46
+ getBoxesWithOffsetLimit: (offset: number, limit: number) => Promise<{
47
+ boxes: ErgoBox[];
48
+ hasNextBatch: boolean;
49
+ }>;
157
50
  }
158
51
  //# sourceMappingURL=ergoUtxoExtractor.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"ergoUtxoExtractor.d.ts","sourceRoot":"","sources":["../../lib/extractor/ergoUtxoExtractor.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,SAAS,CAAC;AACrC,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;AAGhD,OAAO,EAAE,WAAW,EAAE,MAAM,uBAAuB,CAAC;AACpD,OAAO,EAAE,cAAc,EAAe,MAAM,+BAA+B,CAAC;AAC5E,OAAO,EAAE,WAAW,EAAE,MAAM,uBAAuB,CAAC;AAEpD,OAAO,EACL,eAAe,EACf,UAAU,EACX,MAAM,gDAAgD,CAAC;AACxD,OAAO,EAAE,iBAAiB,EAAE,KAAK,EAAE,MAAM,kCAAkC,CAAC;AAE5E,OAAO,EAAE,eAAe,EAAE,MAAM,sBAAsB,CAAC;AAGvD,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AAEnD,qBAAa,iBAAkB,YAAW,iBAAiB,CAAC,WAAW,CAAC;IACtE,QAAQ,CAAC,MAAM,EAAE,cAAc,CAAC;IAChC,OAAO,CAAC,QAAQ,CAAC,UAAU,CAAa;IACxC,QAAQ,CAAC,OAAO,EAAE,eAAe,CAAC;IAClC,OAAO,CAAC,QAAQ,CAAC,EAAE,CAAS;IAC5B,OAAO,CAAC,QAAQ,CAAC,WAAW,CAAwB;IACpD,OAAO,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAS;IACnC,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAgB;IACvC,QAAQ,CAAC,GAAG;;;;;;;;;;;;;;;YA4BZ;;;;;eAKG;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;MAjCU;gBAGX,UAAU,EAAE,UAAU,EACtB,EAAE,EAAE,MAAM,EACV,WAAW,EAAE,OAAO,CAAC,aAAa,EAClC,WAAW,EAAE,MAAM,EACnB,OAAO,CAAC,EAAE,MAAM,EAChB,MAAM,CAAC,EAAE,KAAK,CAAC,MAAM,CAAC,EACtB,MAAM,CAAC,EAAE,cAAc;IAczB;;OAEG;IACH,KAAK,eAAsB;IAE3B;;;;;OAKG;IACH,mBAAmB,QACZ,MAAM,WAAW,CAAC,SAChB,KAAK,KACX,QAAQ,OAAO,CAAC,CAkEjB;IAEF;;;OAGG;IACH,SAAS,SAAgB,MAAM,KAAG,QAAQ,IAAI,CAAC,CAE7C;IAEF;;OAEG;IACH,eAAe,iBAAwB,WAAW,mBA0BhD;IAEF;;;;;OAKG;IACH,sBAAsB,0BACG,MAAM,MAAM,CAAC,iBACrB,MAAM,mBAmBrB;IAEF;;;OAGG;IACH,mBAAmB,UACV,MAAM,KACZ,QAAQ,YAAY,GAAG,SAAS,CAAC,CAQlC;IAEF;;;;OAIG;IACH,eAAe,kBACE,MAAM,KACpB,QAAQ,MAAM,YAAY,CAAC,CAAC,CAmC7B;IAEF;;;OAGG;IACH,UAAU,SAAgB,MAAM;;;OAM9B;IAEF;;;OAGG;IACH,cAAc,UAAiB,MAAM,UAAU,CAAC,6BA2B9C;IAEF;;;OAGG;IACH,WAAW,QAAS,UAAU,aAM5B;CACH"}
1
+ {"version":3,"file":"ergoUtxoExtractor.d.ts","sourceRoot":"","sources":["../../lib/extractor/ergoUtxoExtractor.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,SAAS,CAAC;AACrC,OAAO,KAAK,OAAO,MAAM,sBAAsB,CAAC;AAEhD,OAAO,EAAE,cAAc,EAAE,MAAM,+BAA+B,CAAC;AAC/D,OAAO,EACL,kCAAkC,EAClC,OAAO,EACP,eAAe,EAEf,SAAS,EAGV,MAAM,kCAAkC,CAAC;AAE1C,OAAO,EAAE,eAAe,EAAE,MAAM,sBAAsB,CAAC;AAEvD,OAAO,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AACnD,OAAO,EAAE,eAAe,EAAE,MAAM,iDAAiD,CAAC;AAElF,qBAAa,iBAAkB,SAAQ,kCAAkC,CAAC,YAAY,CAAC;IACrF,QAAQ,CAAC,OAAO,EAAE,eAAe,CAAC;IAClC,OAAO,CAAC,QAAQ,CAAC,EAAE,CAAS;IAC5B,OAAO,CAAC,QAAQ,CAAC,WAAW,CAAwB;IACpD,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAS;IAClC,OAAO,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAS;IACnC,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAgB;IACvC,OAAO,EAAE,eAAe,CAAC;gBAGvB,UAAU,EAAE,UAAU,EACtB,EAAE,EAAE,MAAM,EACV,WAAW,EAAE,OAAO,CAAC,aAAa,EAClC,GAAG,EAAE,MAAM,EACX,IAAI,EAAE,eAAe,EACrB,OAAO,CAAC,EAAE,MAAM,EAChB,MAAM,CAAC,EAAE,KAAK,CAAC,MAAM,CAAC,EACtB,MAAM,CAAC,EAAE,cAAc,EACvB,UAAU,UAAO;IAiBnB;;OAEG;IACH,KAAK,eAAsB;IAE3B;;;;OAIG;IACH,OAAO,QAAS,SAAS,KAAG,OAAO,CAKjC;IAEF;;;OAGG;IACH,UAAU,SAAgB,MAAM,mEAE9B;IAEF;;;;;;OAMG;IACH,cAAc,QACP,SAAS,WACL,MAAM,UACP,MAAM,KACb,KAAK,YAAY,EAAE,YAAY,GAAG,aAAa,CAAC,GAAG,SAAS,CAe7D;IAEF;;;;;OAKG;IACH,uBAAuB,WACb,MAAM,SACP,MAAM,KACZ,QAAQ;QAAE,KAAK,EAAE,OAAO,EAAE,CAAC;QAAC,YAAY,EAAE,OAAO,CAAA;KAAE,CAAC,CAQrD;CACH"}
@@ -1,243 +1,87 @@
1
1
  import * as ergoLib from 'ergo-lib-wasm-nodejs';
2
2
  import { Buffer } from 'buffer';
3
- import { intersection, difference } from 'lodash-es';
4
- import { DummyLogger } from '@rosen-bridge/abstract-logger';
5
- import ergoExplorerClientFactory from '@rosen-clients/ergo-explorer';
3
+ import { AbstractInitializableErgoExtractor, NodeNetwork, boxHasToken, ExplorerNetwork, } from '@rosen-bridge/abstract-extractor';
6
4
  import { BoxEntityAction } from '../actions/boxAction';
7
5
  import { JsonBI } from '../utils';
8
- import { DefaultApiLimit } from '../constants';
9
- export class ErgoUTXOExtractor {
10
- logger;
11
- dataSource;
6
+ import { ErgoNetworkType } from '../../../abstract-extractor/lib/ergo/interfaces';
7
+ export class ErgoUTXOExtractor extends AbstractInitializableErgoExtractor {
12
8
  actions;
13
9
  id;
14
10
  networkType;
11
+ address;
15
12
  ergoTree;
16
13
  tokens;
17
- api;
18
- constructor(dataSource, id, networkType, explorerUrl, address, tokens, logger) {
19
- this.dataSource = dataSource;
14
+ network;
15
+ constructor(dataSource, id, networkType, url, type, address, tokens, logger, initialize = true) {
16
+ super(initialize, logger);
20
17
  this.id = id;
21
18
  this.networkType = networkType;
19
+ this.address = address;
22
20
  this.ergoTree = address
23
21
  ? ergoLib.Address.from_base58(address).to_ergo_tree().to_base16_bytes()
24
22
  : undefined;
25
23
  this.tokens = tokens ? tokens : [];
26
- this.logger = logger ? logger : new DummyLogger();
27
24
  this.actions = new BoxEntityAction(dataSource, this.logger);
28
- this.api = ergoExplorerClientFactory(explorerUrl);
25
+ if (type == ErgoNetworkType.Explorer)
26
+ this.network = new ExplorerNetwork(url);
27
+ else if (type == ErgoNetworkType.Node)
28
+ this.network = new NodeNetwork(url);
29
+ else
30
+ throw Error('Network type is not supported');
29
31
  }
30
32
  /**
31
33
  * get Id for current extractor
32
34
  */
33
35
  getId = () => `${this.id}`;
34
36
  /**
35
- * gets block id and transactions corresponding to the block and saves if they are valid rosen
36
- * transactions and in case of success return true and in case of failure returns false
37
- * @param txs
38
- * @param block
39
- */
40
- processTransactions = (txs, block) => {
41
- return new Promise((resolve, reject) => {
42
- try {
43
- const boxes = [];
44
- const spendBoxes = [];
45
- txs.forEach((transaction) => {
46
- for (const output of transaction.outputs) {
47
- if (this.ergoTree && output.ergoTree !== this.ergoTree) {
48
- continue;
49
- }
50
- const filteredTokens = this.tokens.filter((token) => {
51
- const outputTokens = output.assets ? output.assets : [];
52
- for (const outputToken of outputTokens) {
53
- if (outputToken.tokenId === token) {
54
- return true;
55
- }
56
- }
57
- return false;
58
- });
59
- if (this.tokens.filter((token) => filteredTokens.indexOf(token) === -1).length > 0) {
60
- continue;
61
- }
62
- boxes.push({
63
- boxId: output.boxId,
64
- address: ergoLib.Address.recreate_from_ergo_tree(ergoLib.ErgoTree.from_base16_bytes(output.ergoTree)).to_base58(this.networkType),
65
- serialized: Buffer.from(ergoLib.ErgoBox.from_json(JsonBI.stringify(output)).sigma_serialize_bytes()).toString('base64'),
66
- });
67
- }
68
- for (const input of transaction.inputs) {
69
- spendBoxes.push(input.boxId);
70
- }
71
- });
72
- this.actions
73
- .storeBox(boxes, block, this.getId())
74
- .then(async (status) => {
75
- if (status)
76
- await this.actions
77
- .spendBoxes(spendBoxes, block, this.getId())
78
- .catch((e) => {
79
- this.logger.error(`An error occurred during spending ergo UTXOs for block at height [${block.height}] : ${e}`);
80
- reject(e);
81
- });
82
- resolve(status);
83
- })
84
- .catch((e) => {
85
- this.logger.error(`An error occurred during storing ergo UTXOs for block at height [${block.height}] : ${e}`);
86
- reject(e);
87
- });
88
- }
89
- catch (e) {
90
- reject(e);
91
- }
92
- });
93
- };
94
- /**
95
- * fork one block and remove all stored information for this block
96
- * @param hash: block hash
97
- */
98
- forkBlock = async (hash) => {
99
- await this.actions.deleteBlockBoxes(hash, this.getId());
100
- };
101
- /**
102
- * Initializes the database with older boxes related to the address
103
- */
104
- initializeBoxes = async (initialBlock) => {
105
- // Getting unspent boxes
106
- const unspentBoxes = await this.getUnspentBoxes(initialBlock.height);
107
- const unspentBoxIds = unspentBoxes.map((box) => box.boxId);
108
- // Storing extracted boxes
109
- let allStoredBoxIds = await this.actions.getAllBoxIds(this.getId());
110
- for (const box of unspentBoxes) {
111
- if (allStoredBoxIds.includes(box.boxId)) {
112
- await this.actions.updateBox(box, this.getId());
113
- this.logger.info(`Updated the existing unspent box with boxId, [${box.boxId}]`);
114
- this.logger.debug(`Updated box [${JSON.stringify(box)}]`);
115
- }
116
- else {
117
- await this.actions.insertBox(box, this.getId());
118
- this.logger.info(`Inserted new unspent box with boxId, [${box.boxId}]`);
119
- this.logger.debug(`Inserted box [${JSON.stringify(box)}]`);
120
- }
121
- }
122
- // Remove updated box ids from existing boxes in database
123
- allStoredBoxIds = difference(allStoredBoxIds, unspentBoxIds);
124
- // Validating remained boxes
125
- // TODO: Fix extractor initialization local:ergo/rosen-bridge/scanner#102
126
- // await this.validateOldStoredBoxes(allStoredBoxIds, initialHeight);
127
- };
128
- /**
129
- * Validate all remaining boxes in the database
130
- * update the correct ones and remove the invalid ones
131
- * @param unchangedStoredBoxIds
132
- * @param initialHeight
133
- */
134
- validateOldStoredBoxes = async (unchangedStoredBoxIds, initialHeight) => {
135
- for (const boxId of unchangedStoredBoxIds) {
136
- const box = await this.getBoxInfoWithBoxId(boxId);
137
- if (box && box.spendBlock && box.spendHeight) {
138
- if (box.spendHeight < initialHeight)
139
- await this.actions.updateSpendBlock(boxId, this.getId(), box.spendBlock, box.spendHeight);
140
- }
141
- else {
142
- await this.actions.removeBox(boxId, this.getId());
143
- this.logger.info(`Removed invalid box [${boxId}] in initialization validation`);
144
- }
145
- }
146
- };
147
- /**
148
- * Return extracted information of a box
149
- * @param boxId
150
- */
151
- getBoxInfoWithBoxId = async (boxId) => {
152
- try {
153
- const box = await this.api.v1.getApiV1BoxesP1(boxId);
154
- return (await this.extractBoxData([box]))[0];
155
- }
156
- catch {
157
- this.logger.warn(`Box with id [${boxId}] does not exists`);
158
- return undefined;
159
- }
160
- };
161
- /**
162
- * Get unspent boxes created bellow the initial height
163
- * @param initialHeight
164
- * @returns
37
+ * check proper data format in the box
38
+ * @param box
39
+ * @return true if the box has the required data and false otherwise
165
40
  */
166
- getUnspentBoxes = async (initialHeight) => {
167
- let allBoxes = [];
168
- let offset = 0, total = DefaultApiLimit, boxes;
169
- while (offset < total) {
170
- if (this.ergoTree) {
171
- boxes = await this.api.v1.getApiV1BoxesUnspentByergotreeP1(this.ergoTree, { offset: offset, limit: DefaultApiLimit });
172
- }
173
- else if (!this.ergoTree && this.tokens.length > 0) {
174
- boxes = await this.api.v1.getApiV1BoxesUnspentBytokenidP1(this.tokens[0], { offset: offset, limit: DefaultApiLimit });
175
- }
176
- else
177
- return [];
178
- if (!boxes.items) {
179
- this.logger.warn('Explorer api output items should not be undefined.');
180
- throw new Error('Incorrect explorer api output');
181
- }
182
- allBoxes = [
183
- ...allBoxes,
184
- ...(await this.extractBoxData(boxes.items.filter((box) => box.creationHeight < initialHeight &&
185
- (this.tokens.length == 0 || this.boxHasToken(box))))),
186
- ];
187
- total = boxes.total;
188
- offset += DefaultApiLimit;
189
- }
190
- return allBoxes;
41
+ hasData = (box) => {
42
+ return ((!this.ergoTree || box.ergoTree == this.ergoTree) &&
43
+ (this.tokens.length == 0 || boxHasToken(box, this.tokens)));
191
44
  };
192
45
  /**
193
46
  * Returns block information of tx
194
47
  * @param txId
195
48
  */
196
49
  getTxBlock = async (txId) => {
197
- const tx = await this.api.v1.getApiV1TransactionsP1(txId);
198
- return {
199
- id: tx.blockId,
200
- height: tx.inclusionHeight,
201
- };
50
+ return this.network.getTxBlock(txId);
202
51
  };
203
52
  /**
204
- * Extract needed information for storing in database from api json outputs
205
- * @param boxes
53
+ * extract box data to proper format (not including spending information)
54
+ * @param box
55
+ * @param blockId box inclusion block hash
56
+ * @param height box inclusion block height
57
+ * @return extracted data in proper format
206
58
  */
207
- extractBoxData = async (boxes) => {
208
- const extractedBoxes = [];
209
- for (const box of boxes) {
210
- let spendBlock, spendHeight;
211
- if (box.spentTransactionId) {
212
- const block = await this.getTxBlock(box.spentTransactionId);
213
- spendBlock = block.id;
214
- spendHeight = block.height;
215
- }
216
- const ergoBox = ergoLib.ErgoBox.from_json(JsonBI.stringify(box));
217
- extractedBoxes.push({
218
- boxId: ergoBox.box_id().to_str(),
219
- address: ergoLib.Address.recreate_from_ergo_tree(ergoLib.ErgoTree.from_base16_bytes(ergoBox.ergo_tree().to_base16_bytes())).to_base58(this.networkType),
220
- serialized: Buffer.from(ergoBox.sigma_serialize_bytes()).toString('base64'),
221
- blockId: box.blockId,
222
- height: box.settlementHeight,
223
- spendBlock: spendBlock,
224
- spendHeight: spendHeight,
225
- });
226
- }
227
- return extractedBoxes;
59
+ extractBoxData = (box, blockId, height) => {
60
+ const ergoBox = ergoLib.ErgoBox.from_json(JsonBI.stringify(box));
61
+ return {
62
+ boxId: ergoBox.box_id().to_str(),
63
+ address: ergoLib.Address.recreate_from_ergo_tree(ergoLib.ErgoTree.from_base16_bytes(ergoBox.ergo_tree().to_base16_bytes())).to_base58(this.networkType),
64
+ serialized: Buffer.from(ergoBox.sigma_serialize_bytes()).toString('base64'),
65
+ blockId: blockId,
66
+ height: height,
67
+ };
228
68
  };
229
69
  /**
230
- * Returns true if box has the required token and false otherwise
231
- * @param box
70
+ * return init required boxes with offset limit
71
+ * @param offset
72
+ * @param limit
73
+ * @return boxes in batch
232
74
  */
233
- boxHasToken = (box) => {
234
- if (!box.assets)
235
- return false;
236
- const boxTokens = box.assets.map((token) => token.tokenId);
237
- const requiredTokens = intersection(this.tokens, boxTokens);
238
- if (requiredTokens.length == this.tokens.length)
239
- return true;
240
- return false;
75
+ getBoxesWithOffsetLimit = async (offset, limit) => {
76
+ if (this.address) {
77
+ return this.network.getBoxesByAddress(this.address, offset, limit);
78
+ }
79
+ else if (!this.address && this.tokens.length > 0) {
80
+ return this.network.getBoxesByAddress(this.tokens[0], offset, limit);
81
+ }
82
+ else {
83
+ return { boxes: [], hasNextBatch: false };
84
+ }
241
85
  };
242
86
  }
243
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ergoUtxoExtractor.js","sourceRoot":"","sources":["../../lib/extractor/ergoUtxoExtractor.ts"],"names":[],"mappings":"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","sourcesContent":["import { DataSource } from 'typeorm';\nimport * as ergoLib from 'ergo-lib-wasm-nodejs';\nimport { Buffer } from 'buffer';\nimport { intersection, difference } from 'lodash-es';\nimport { InitialInfo } from '@rosen-bridge/scanner';\nimport { AbstractLogger, DummyLogger } from '@rosen-bridge/abstract-logger';\nimport { Transaction } from '@rosen-bridge/scanner';\nimport ergoExplorerClientFactory from '@rosen-clients/ergo-explorer';\nimport {\n  ItemsOutputInfo,\n  OutputInfo,\n} from '@rosen-clients/ergo-explorer/dist/src/v1/types';\nimport { AbstractExtractor, Block } from '@rosen-bridge/abstract-extractor';\n\nimport { BoxEntityAction } from '../actions/boxAction';\nimport { JsonBI } from '../utils';\nimport { DefaultApiLimit } from '../constants';\nimport { ExtractedBox } from '../interfaces/types';\n\nexport class ErgoUTXOExtractor implements AbstractExtractor<Transaction> {\n  readonly logger: AbstractLogger;\n  private readonly dataSource: DataSource;\n  readonly actions: BoxEntityAction;\n  private readonly id: string;\n  private readonly networkType: ergoLib.NetworkPrefix;\n  private readonly ergoTree?: string;\n  private readonly tokens: Array<string>;\n  readonly api;\n\n  constructor(\n    dataSource: DataSource,\n    id: string,\n    networkType: ergoLib.NetworkPrefix,\n    explorerUrl: string,\n    address?: string,\n    tokens?: Array<string>,\n    logger?: AbstractLogger\n  ) {\n    this.dataSource = dataSource;\n    this.id = id;\n    this.networkType = networkType;\n    this.ergoTree = address\n      ? ergoLib.Address.from_base58(address).to_ergo_tree().to_base16_bytes()\n      : undefined;\n    this.tokens = tokens ? tokens : [];\n    this.logger = logger ? logger : new DummyLogger();\n    this.actions = new BoxEntityAction(dataSource, this.logger);\n    this.api = ergoExplorerClientFactory(explorerUrl);\n  }\n\n  /**\n   * get Id for current extractor\n   */\n  getId = () => `${this.id}`;\n\n  /**\n   * gets block id and transactions corresponding to the block and saves if they are valid rosen\n   *  transactions and in case of success return true and in case of failure returns false\n   * @param txs\n   * @param block\n   */\n  processTransactions = (\n    txs: Array<Transaction>,\n    block: Block\n  ): Promise<boolean> => {\n    return new Promise((resolve, reject) => {\n      try {\n        const boxes: Array<ExtractedBox> = [];\n        const spendBoxes: Array<string> = [];\n        txs.forEach((transaction) => {\n          for (const output of transaction.outputs) {\n            if (this.ergoTree && output.ergoTree !== this.ergoTree) {\n              continue;\n            }\n            const filteredTokens = this.tokens.filter((token) => {\n              const outputTokens = output.assets ? output.assets : [];\n              for (const outputToken of outputTokens) {\n                if (outputToken.tokenId === token) {\n                  return true;\n                }\n              }\n              return false;\n            });\n            if (\n              this.tokens.filter(\n                (token) => filteredTokens.indexOf(token) === -1\n              ).length > 0\n            ) {\n              continue;\n            }\n            boxes.push({\n              boxId: output.boxId,\n              address: ergoLib.Address.recreate_from_ergo_tree(\n                ergoLib.ErgoTree.from_base16_bytes(output.ergoTree)\n              ).to_base58(this.networkType),\n              serialized: Buffer.from(\n                ergoLib.ErgoBox.from_json(\n                  JsonBI.stringify(output)\n                ).sigma_serialize_bytes()\n              ).toString('base64'),\n            });\n          }\n          for (const input of transaction.inputs) {\n            spendBoxes.push(input.boxId);\n          }\n        });\n        this.actions\n          .storeBox(boxes, block, this.getId())\n          .then(async (status) => {\n            if (status)\n              await this.actions\n                .spendBoxes(spendBoxes, block, this.getId())\n                .catch((e) => {\n                  this.logger.error(\n                    `An error occurred during spending ergo UTXOs for block at height [${block.height}] : ${e}`\n                  );\n                  reject(e);\n                });\n            resolve(status);\n          })\n          .catch((e) => {\n            this.logger.error(\n              `An error occurred during storing ergo UTXOs for block at height [${block.height}] : ${e}`\n            );\n            reject(e);\n          });\n      } catch (e) {\n        reject(e);\n      }\n    });\n  };\n\n  /**\n   * fork one block and remove all stored information for this block\n   * @param hash: block hash\n   */\n  forkBlock = async (hash: string): Promise<void> => {\n    await this.actions.deleteBlockBoxes(hash, this.getId());\n  };\n\n  /**\n   * Initializes the database with older boxes related to the address\n   */\n  initializeBoxes = async (initialBlock: InitialInfo) => {\n    // Getting unspent boxes\n    const unspentBoxes = await this.getUnspentBoxes(initialBlock.height);\n    const unspentBoxIds = unspentBoxes.map((box) => box.boxId);\n\n    // Storing extracted boxes\n    let allStoredBoxIds = await this.actions.getAllBoxIds(this.getId());\n    for (const box of unspentBoxes) {\n      if (allStoredBoxIds.includes(box.boxId)) {\n        await this.actions.updateBox(box, this.getId());\n        this.logger.info(\n          `Updated the existing unspent box with boxId, [${box.boxId}]`\n        );\n        this.logger.debug(`Updated box [${JSON.stringify(box)}]`);\n      } else {\n        await this.actions.insertBox(box, this.getId());\n        this.logger.info(`Inserted new unspent box with boxId, [${box.boxId}]`);\n        this.logger.debug(`Inserted box [${JSON.stringify(box)}]`);\n      }\n    }\n\n    // Remove updated box ids from existing boxes in database\n    allStoredBoxIds = difference(allStoredBoxIds, unspentBoxIds);\n    // Validating remained boxes\n    // TODO: Fix extractor initialization local:ergo/rosen-bridge/scanner#102\n    // await this.validateOldStoredBoxes(allStoredBoxIds, initialHeight);\n  };\n\n  /**\n   * Validate all remaining boxes in the database\n   * update the correct ones and remove the invalid ones\n   * @param unchangedStoredBoxIds\n   * @param initialHeight\n   */\n  validateOldStoredBoxes = async (\n    unchangedStoredBoxIds: Array<string>,\n    initialHeight: number\n  ) => {\n    for (const boxId of unchangedStoredBoxIds) {\n      const box = await this.getBoxInfoWithBoxId(boxId);\n      if (box && box.spendBlock && box.spendHeight) {\n        if (box.spendHeight < initialHeight)\n          await this.actions.updateSpendBlock(\n            boxId,\n            this.getId(),\n            box.spendBlock,\n            box.spendHeight\n          );\n      } else {\n        await this.actions.removeBox(boxId, this.getId());\n        this.logger.info(\n          `Removed invalid box [${boxId}] in initialization validation`\n        );\n      }\n    }\n  };\n\n  /**\n   * Return extracted information of a box\n   * @param boxId\n   */\n  getBoxInfoWithBoxId = async (\n    boxId: string\n  ): Promise<ExtractedBox | undefined> => {\n    try {\n      const box = await this.api.v1.getApiV1BoxesP1(boxId);\n      return (await this.extractBoxData([box]))[0];\n    } catch {\n      this.logger.warn(`Box with id [${boxId}] does not exists`);\n      return undefined;\n    }\n  };\n\n  /**\n   * Get unspent boxes created bellow the initial height\n   * @param initialHeight\n   * @returns\n   */\n  getUnspentBoxes = async (\n    initialHeight: number\n  ): Promise<Array<ExtractedBox>> => {\n    let allBoxes: Array<ExtractedBox> = [];\n    let offset = 0,\n      total = DefaultApiLimit,\n      boxes: ItemsOutputInfo;\n    while (offset < total) {\n      if (this.ergoTree) {\n        boxes = await this.api.v1.getApiV1BoxesUnspentByergotreeP1(\n          this.ergoTree,\n          { offset: offset, limit: DefaultApiLimit }\n        );\n      } else if (!this.ergoTree && this.tokens.length > 0) {\n        boxes = await this.api.v1.getApiV1BoxesUnspentBytokenidP1(\n          this.tokens[0],\n          { offset: offset, limit: DefaultApiLimit }\n        );\n      } else return [];\n      if (!boxes.items) {\n        this.logger.warn('Explorer api output items should not be undefined.');\n        throw new Error('Incorrect explorer api output');\n      }\n      allBoxes = [\n        ...allBoxes,\n        ...(await this.extractBoxData(\n          boxes.items.filter(\n            (box) =>\n              box.creationHeight < initialHeight &&\n              (this.tokens.length == 0 || this.boxHasToken(box))\n          )\n        )),\n      ];\n      total = boxes.total;\n      offset += DefaultApiLimit;\n    }\n    return allBoxes;\n  };\n\n  /**\n   * Returns block information of tx\n   * @param txId\n   */\n  getTxBlock = async (txId: string) => {\n    const tx = await this.api.v1.getApiV1TransactionsP1(txId);\n    return {\n      id: tx.blockId,\n      height: tx.inclusionHeight,\n    };\n  };\n\n  /**\n   * Extract needed information for storing in database from api json outputs\n   * @param boxes\n   */\n  extractBoxData = async (boxes: Array<OutputInfo>) => {\n    const extractedBoxes: Array<ExtractedBox> = [];\n    for (const box of boxes) {\n      let spendBlock, spendHeight;\n      if (box.spentTransactionId) {\n        const block = await this.getTxBlock(box.spentTransactionId);\n        spendBlock = block.id;\n        spendHeight = block.height;\n      }\n      const ergoBox = ergoLib.ErgoBox.from_json(JsonBI.stringify(box));\n      extractedBoxes.push({\n        boxId: ergoBox.box_id().to_str(),\n        address: ergoLib.Address.recreate_from_ergo_tree(\n          ergoLib.ErgoTree.from_base16_bytes(\n            ergoBox.ergo_tree().to_base16_bytes()\n          )\n        ).to_base58(this.networkType),\n        serialized: Buffer.from(ergoBox.sigma_serialize_bytes()).toString(\n          'base64'\n        ),\n        blockId: box.blockId,\n        height: box.settlementHeight,\n        spendBlock: spendBlock,\n        spendHeight: spendHeight,\n      });\n    }\n    return extractedBoxes;\n  };\n\n  /**\n   * Returns true if box has the required token and false otherwise\n   * @param box\n   */\n  boxHasToken = (box: OutputInfo) => {\n    if (!box.assets) return false;\n    const boxTokens = box.assets.map((token) => token.tokenId);\n    const requiredTokens = intersection(this.tokens, boxTokens);\n    if (requiredTokens.length == this.tokens.length) return true;\n    return false;\n  };\n}\n"]}
87
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ergoUtxoExtractor.js","sourceRoot":"","sources":["../../lib/extractor/ergoUtxoExtractor.ts"],"names":[],"mappings":"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","sourcesContent":["import { DataSource } from 'typeorm';\nimport * as ergoLib from 'ergo-lib-wasm-nodejs';\nimport { Buffer } from 'buffer';\nimport { AbstractLogger } from '@rosen-bridge/abstract-logger';\nimport {\n  AbstractInitializableErgoExtractor,\n  ErgoBox,\n  AbstractNetwork,\n  NodeNetwork,\n  OutputBox,\n  boxHasToken,\n  ExplorerNetwork,\n} from '@rosen-bridge/abstract-extractor';\n\nimport { BoxEntityAction } from '../actions/boxAction';\nimport { JsonBI } from '../utils';\nimport { ExtractedBox } from '../interfaces/types';\nimport { ErgoNetworkType } from '../../../abstract-extractor/lib/ergo/interfaces';\n\nexport class ErgoUTXOExtractor extends AbstractInitializableErgoExtractor<ExtractedBox> {\n  readonly actions: BoxEntityAction;\n  private readonly id: string;\n  private readonly networkType: ergoLib.NetworkPrefix;\n  private readonly address?: string;\n  private readonly ergoTree?: string;\n  private readonly tokens: Array<string>;\n  network: AbstractNetwork;\n\n  constructor(\n    dataSource: DataSource,\n    id: string,\n    networkType: ergoLib.NetworkPrefix,\n    url: string,\n    type: ErgoNetworkType,\n    address?: string,\n    tokens?: Array<string>,\n    logger?: AbstractLogger,\n    initialize = true\n  ) {\n    super(initialize, logger);\n    this.id = id;\n    this.networkType = networkType;\n    this.address = address;\n    this.ergoTree = address\n      ? ergoLib.Address.from_base58(address).to_ergo_tree().to_base16_bytes()\n      : undefined;\n    this.tokens = tokens ? tokens : [];\n    this.actions = new BoxEntityAction(dataSource, this.logger);\n    if (type == ErgoNetworkType.Explorer)\n      this.network = new ExplorerNetwork(url);\n    else if (type == ErgoNetworkType.Node) this.network = new NodeNetwork(url);\n    else throw Error('Network type is not supported');\n  }\n\n  /**\n   * get Id for current extractor\n   */\n  getId = () => `${this.id}`;\n\n  /**\n   * check proper data format in the box\n   * @param box\n   * @return true if the box has the required data and false otherwise\n   */\n  hasData = (box: OutputBox): boolean => {\n    return (\n      (!this.ergoTree || box.ergoTree == this.ergoTree) &&\n      (this.tokens.length == 0 || boxHasToken(box, this.tokens))\n    );\n  };\n\n  /**\n   * Returns block information of tx\n   * @param txId\n   */\n  getTxBlock = async (txId: string) => {\n    return this.network.getTxBlock(txId);\n  };\n\n  /**\n   * extract box data to proper format (not including spending information)\n   * @param box\n   * @param blockId box inclusion block hash\n   * @param height box inclusion block height\n   * @return extracted data in proper format\n   */\n  extractBoxData = (\n    box: OutputBox,\n    blockId: string,\n    height: number\n  ): Omit<ExtractedBox, 'spendBlock' | 'spendHeight'> | undefined => {\n    const ergoBox = ergoLib.ErgoBox.from_json(JsonBI.stringify(box));\n    return {\n      boxId: ergoBox.box_id().to_str(),\n      address: ergoLib.Address.recreate_from_ergo_tree(\n        ergoLib.ErgoTree.from_base16_bytes(\n          ergoBox.ergo_tree().to_base16_bytes()\n        )\n      ).to_base58(this.networkType),\n      serialized: Buffer.from(ergoBox.sigma_serialize_bytes()).toString(\n        'base64'\n      ),\n      blockId: blockId,\n      height: height,\n    };\n  };\n\n  /**\n   * return init required boxes with offset limit\n   * @param offset\n   * @param limit\n   * @return boxes in batch\n   */\n  getBoxesWithOffsetLimit = async (\n    offset: number,\n    limit: number\n  ): Promise<{ boxes: ErgoBox[]; hasNextBatch: boolean }> => {\n    if (this.address) {\n      return this.network.getBoxesByAddress(this.address, offset, limit);\n    } else if (!this.address && this.tokens.length > 0) {\n      return this.network.getBoxesByAddress(this.tokens[0], offset, limit);\n    } else {\n      return { boxes: [], hasNextBatch: false };\n    }\n  };\n}\n"]}
@@ -1,28 +1,10 @@
1
- type Asset = {
2
- tokenId: string;
3
- index: number;
4
- amount: bigint;
5
- };
6
- interface ErgoBoxJson {
7
- boxId: string;
8
- address: string;
9
- value: bigint;
10
- blockId: string;
11
- settlementHeight: number;
12
- assets: Array<Asset>;
13
- }
14
- interface Boxes {
15
- items: Array<ErgoBoxJson>;
16
- total: number;
17
- }
18
- interface ExtractedBox {
1
+ export interface ExtractedBox {
19
2
  boxId: string;
20
3
  address: string;
21
4
  serialized: string;
22
- blockId?: string;
23
- height?: number;
5
+ blockId: string;
6
+ height: number;
24
7
  spendBlock?: string;
25
8
  spendHeight?: number;
26
9
  }
27
- export { ErgoBoxJson, Boxes, ExtractedBox };
28
10
  //# sourceMappingURL=types.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../lib/interfaces/types.ts"],"names":[],"mappings":"AAAA,KAAK,KAAK,GAAG;IACX,OAAO,EAAE,MAAM,CAAC;IAChB,KAAK,EAAE,MAAM,CAAC;IACd,MAAM,EAAE,MAAM,CAAC;CAChB,CAAC;AAEF,UAAU,WAAW;IACnB,KAAK,EAAE,MAAM,CAAC;IACd,OAAO,EAAE,MAAM,CAAC;IAChB,KAAK,EAAE,MAAM,CAAC;IACd,OAAO,EAAE,MAAM,CAAC;IAChB,gBAAgB,EAAE,MAAM,CAAC;IACzB,MAAM,EAAE,KAAK,CAAC,KAAK,CAAC,CAAC;CACtB;AAED,UAAU,KAAK;IACb,KAAK,EAAE,KAAK,CAAC,WAAW,CAAC,CAAC;IAC1B,KAAK,EAAE,MAAM,CAAC;CACf;AAED,UAAU,YAAY;IACpB,KAAK,EAAE,MAAM,CAAC;IACd,OAAO,EAAE,MAAM,CAAC;IAChB,UAAU,EAAE,MAAM,CAAC;IACnB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,WAAW,CAAC,EAAE,MAAM,CAAC;CACtB;AAED,OAAO,EAAE,WAAW,EAAE,KAAK,EAAE,YAAY,EAAE,CAAC"}
1
+ {"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../lib/interfaces/types.ts"],"names":[],"mappings":"AAAA,MAAM,WAAW,YAAY;IAC3B,KAAK,EAAE,MAAM,CAAC;IACd,OAAO,EAAE,MAAM,CAAC;IAChB,UAAU,EAAE,MAAM,CAAC;IACnB,OAAO,EAAE,MAAM,CAAC;IAChB,MAAM,EAAE,MAAM,CAAC;IACf,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,WAAW,CAAC,EAAE,MAAM,CAAC;CACtB"}
@@ -1,2 +1,2 @@
1
1
  export {};
2
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9saWIvaW50ZXJmYWNlcy90eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsidHlwZSBBc3NldCA9IHtcbiAgdG9rZW5JZDogc3RyaW5nO1xuICBpbmRleDogbnVtYmVyO1xuICBhbW91bnQ6IGJpZ2ludDtcbn07XG5cbmludGVyZmFjZSBFcmdvQm94SnNvbiB7XG4gIGJveElkOiBzdHJpbmc7XG4gIGFkZHJlc3M6IHN0cmluZztcbiAgdmFsdWU6IGJpZ2ludDtcbiAgYmxvY2tJZDogc3RyaW5nO1xuICBzZXR0bGVtZW50SGVpZ2h0OiBudW1iZXI7XG4gIGFzc2V0czogQXJyYXk8QXNzZXQ+O1xufVxuXG5pbnRlcmZhY2UgQm94ZXMge1xuICBpdGVtczogQXJyYXk8RXJnb0JveEpzb24+O1xuICB0b3RhbDogbnVtYmVyO1xufVxuXG5pbnRlcmZhY2UgRXh0cmFjdGVkQm94IHtcbiAgYm94SWQ6IHN0cmluZztcbiAgYWRkcmVzczogc3RyaW5nO1xuICBzZXJpYWxpemVkOiBzdHJpbmc7XG4gIGJsb2NrSWQ/OiBzdHJpbmc7XG4gIGhlaWdodD86IG51bWJlcjtcbiAgc3BlbmRCbG9jaz86IHN0cmluZztcbiAgc3BlbmRIZWlnaHQ/OiBudW1iZXI7XG59XG5cbmV4cG9ydCB7IEVyZ29Cb3hKc29uLCBCb3hlcywgRXh0cmFjdGVkQm94IH07XG4iXX0=
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9saWIvaW50ZXJmYWNlcy90eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGludGVyZmFjZSBFeHRyYWN0ZWRCb3gge1xuICBib3hJZDogc3RyaW5nO1xuICBhZGRyZXNzOiBzdHJpbmc7XG4gIHNlcmlhbGl6ZWQ6IHN0cmluZztcbiAgYmxvY2tJZDogc3RyaW5nO1xuICBoZWlnaHQ6IG51bWJlcjtcbiAgc3BlbmRCbG9jaz86IHN0cmluZztcbiAgc3BlbmRIZWlnaHQ/OiBudW1iZXI7XG59XG4iXX0=
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@rosen-bridge/address-extractor",
3
- "version": "4.0.0",
3
+ "version": "5.0.0",
4
4
  "description": "UTXO box extractor for any address or token.",
5
5
  "author": "Rosen Team",
6
6
  "license": "GPL-3.0",
@@ -20,8 +20,7 @@
20
20
  },
21
21
  "dependencies": {
22
22
  "@rosen-bridge/abstract-logger": "^1.0.0",
23
- "@rosen-bridge/abstract-extractor": "^0.1.0",
24
- "@rosen-bridge/scanner": "^4.0.0",
23
+ "@rosen-bridge/abstract-extractor": "^0.1.1",
25
24
  "@rosen-clients/ergo-explorer": "^1.1.1",
26
25
  "blakejs": "^1.2.1",
27
26
  "ergo-lib-wasm-nodejs": "^0.24.1",
@@ -1,4 +0,0 @@
1
- declare const DefaultApiLimit = 100;
2
- declare const dbIdChunkSize = 100;
3
- export { DefaultApiLimit, dbIdChunkSize };
4
- //# sourceMappingURL=constants.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"constants.d.ts","sourceRoot":"","sources":["../lib/constants.ts"],"names":[],"mappings":"AAAA,QAAA,MAAM,eAAe,MAAM,CAAC;AAC5B,QAAA,MAAM,aAAa,MAAM,CAAC;AAE1B,OAAO,EAAE,eAAe,EAAE,aAAa,EAAE,CAAC"}
package/dist/constants.js DELETED
@@ -1,4 +0,0 @@
1
- const DefaultApiLimit = 100;
2
- const dbIdChunkSize = 100;
3
- export { DefaultApiLimit, dbIdChunkSize };
4
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uc3RhbnRzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vbGliL2NvbnN0YW50cy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxNQUFNLGVBQWUsR0FBRyxHQUFHLENBQUM7QUFDNUIsTUFBTSxhQUFhLEdBQUcsR0FBRyxDQUFDO0FBRTFCLE9BQU8sRUFBRSxlQUFlLEVBQUUsYUFBYSxFQUFFLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJjb25zdCBEZWZhdWx0QXBpTGltaXQgPSAxMDA7XG5jb25zdCBkYklkQ2h1bmtTaXplID0gMTAwO1xuXG5leHBvcnQgeyBEZWZhdWx0QXBpTGltaXQsIGRiSWRDaHVua1NpemUgfTtcbiJdfQ==