@rosen-bridge/fraud-extractor 0.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.
- package/README.md +39 -0
- package/dist/actions/fraudAction.d.ts +64 -0
- package/dist/actions/fraudAction.d.ts.map +1 -0
- package/dist/actions/fraudAction.js +196 -0
- package/dist/constants.d.ts +4 -0
- package/dist/constants.d.ts.map +1 -0
- package/dist/constants.js +4 -0
- package/dist/entities/fraudEntity.d.ts +16 -0
- package/dist/entities/fraudEntity.d.ts.map +1 -0
- package/dist/entities/fraudEntity.js +82 -0
- package/dist/extractor/fraudExtractor.d.ts +144 -0
- package/dist/extractor/fraudExtractor.d.ts.map +1 -0
- package/dist/extractor/fraudExtractor.js +262 -0
- package/dist/index.d.ts +4 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +4 -0
- package/dist/interfaces/types.d.ts +15 -0
- package/dist/interfaces/types.d.ts.map +1 -0
- package/dist/interfaces/types.js +2 -0
- package/dist/migrations/index.d.ts +7 -0
- package/dist/migrations/index.d.ts.map +1 -0
- package/dist/migrations/index.js +7 -0
- package/dist/migrations/postgres/1693308311652-migration.d.ts +7 -0
- package/dist/migrations/postgres/1693308311652-migration.d.ts.map +1 -0
- package/dist/migrations/postgres/1693308311652-migration.js +29 -0
- package/dist/migrations/sqlite/1693122371215-migration.d.ts +7 -0
- package/dist/migrations/sqlite/1693122371215-migration.d.ts.map +1 -0
- package/dist/migrations/sqlite/1693122371215-migration.js +28 -0
- package/package.json +52 -0
package/README.md
ADDED
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
# Fraud Extractor
|
|
2
|
+
|
|
3
|
+
### Table of Contents
|
|
4
|
+
|
|
5
|
+
- [Description](#description)
|
|
6
|
+
- [Related Projects](#related-projects)
|
|
7
|
+
- [How to Use the Fraud Extractor](#how-to-use-the-fraud-extractor)
|
|
8
|
+
- [Install](#install)
|
|
9
|
+
- [Usage](#usage)
|
|
10
|
+
- [Contributing](#contributing)
|
|
11
|
+
- [License](#license)
|
|
12
|
+
|
|
13
|
+
<a name="headers"/>
|
|
14
|
+
|
|
15
|
+
## Description
|
|
16
|
+
|
|
17
|
+
Data extractors are modules that are integrated within a scanner. They drive the required information from the scanned blocks and store them in their own database. Fraud extractor, extracts fraud boxes information from blocks and stores them in the database. Cleanup service creates frauds from timed out unspent triggers; these triggers might have incorrect data or be a duplicated version of a valid trigger. Then the fraud funds will be seized by the service to punish the faulty reports.
|
|
18
|
+
|
|
19
|
+
## Related Projects
|
|
20
|
+
|
|
21
|
+
TBD
|
|
22
|
+
|
|
23
|
+
## How to Use the Fraud Extractor
|
|
24
|
+
|
|
25
|
+
### Install
|
|
26
|
+
|
|
27
|
+
TBD
|
|
28
|
+
|
|
29
|
+
### Usage
|
|
30
|
+
|
|
31
|
+
TBD
|
|
32
|
+
|
|
33
|
+
## Contributing
|
|
34
|
+
|
|
35
|
+
TBD
|
|
36
|
+
|
|
37
|
+
## License
|
|
38
|
+
|
|
39
|
+
TBD
|
|
@@ -0,0 +1,64 @@
|
|
|
1
|
+
import { DataSource } from 'typeorm';
|
|
2
|
+
import { AbstractLogger } from '@rosen-bridge/abstract-logger';
|
|
3
|
+
import { BlockEntity } from '@rosen-bridge/scanner';
|
|
4
|
+
import { ExtractedFraud } from '../interfaces/types';
|
|
5
|
+
export declare class FraudAction {
|
|
6
|
+
private readonly datasource;
|
|
7
|
+
private readonly logger;
|
|
8
|
+
private readonly repository;
|
|
9
|
+
constructor(dataSource: DataSource, logger: AbstractLogger);
|
|
10
|
+
/**
|
|
11
|
+
* Store a list of frauds in a specific block
|
|
12
|
+
* @param frauds
|
|
13
|
+
* @param spendBoxes
|
|
14
|
+
* @param block
|
|
15
|
+
* @param extractor
|
|
16
|
+
*/
|
|
17
|
+
storeBlockFrauds: (frauds: Array<ExtractedFraud>, block: BlockEntity, extractor: string) => Promise<boolean>;
|
|
18
|
+
/**
|
|
19
|
+
* Insert a new fraud into database
|
|
20
|
+
* @param fraud
|
|
21
|
+
* @param extractor
|
|
22
|
+
*/
|
|
23
|
+
insertFraud: (fraud: ExtractedFraud, extractor: string) => Promise<import("typeorm").InsertResult>;
|
|
24
|
+
/**
|
|
25
|
+
* Update an unspent fraud information in the database
|
|
26
|
+
* @param fraud
|
|
27
|
+
* @param extractor
|
|
28
|
+
*/
|
|
29
|
+
updateFraud: (fraud: ExtractedFraud, extractor: string) => Promise<import("typeorm").UpdateResult>;
|
|
30
|
+
/**
|
|
31
|
+
* Update spendBlock and spendHeight of frauds spent on the block
|
|
32
|
+
* @param spendIds
|
|
33
|
+
* @param block
|
|
34
|
+
* @param extractor
|
|
35
|
+
*/
|
|
36
|
+
spendFrauds: (spendIds: Array<string>, block: BlockEntity, extractor: string, txId: string) => Promise<void>;
|
|
37
|
+
/**
|
|
38
|
+
* Update all frauds related to an specific invalid block
|
|
39
|
+
* if box had been spent in the block mark it as unspent,
|
|
40
|
+
* and if it was created within the block remove it from database
|
|
41
|
+
* @param block
|
|
42
|
+
* @param extractor
|
|
43
|
+
*/
|
|
44
|
+
deleteBlock: (block: string, extractor: string) => Promise<void>;
|
|
45
|
+
/**
|
|
46
|
+
* Return all stored fraud box ids
|
|
47
|
+
*/
|
|
48
|
+
getAllBoxIds: (extractor: string) => Promise<Array<string>>;
|
|
49
|
+
/**
|
|
50
|
+
* Remove an specified fraud
|
|
51
|
+
* @param boxId
|
|
52
|
+
* @param extractor
|
|
53
|
+
*/
|
|
54
|
+
removeFraud: (boxId: string, extractor: string) => Promise<import("typeorm").DeleteResult>;
|
|
55
|
+
/**
|
|
56
|
+
* Update the fraud spending information
|
|
57
|
+
* @param boxId
|
|
58
|
+
* @param extractor
|
|
59
|
+
* @param blockId
|
|
60
|
+
* @param blockHeight
|
|
61
|
+
*/
|
|
62
|
+
updateSpendBlock: (boxId: string, extractor: string, blockId: string, blockHeight: number) => Promise<import("typeorm").UpdateResult>;
|
|
63
|
+
}
|
|
64
|
+
//# sourceMappingURL=fraudAction.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"fraudAction.d.ts","sourceRoot":"","sources":["../../lib/actions/fraudAction.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAkB,MAAM,SAAS,CAAC;AACrD,OAAO,EAAE,cAAc,EAAE,MAAM,+BAA+B,CAAC;AAC/D,OAAO,EAAE,WAAW,EAAE,MAAM,uBAAuB,CAAC;AAGpD,OAAO,EAAE,cAAc,EAAE,MAAM,qBAAqB,CAAC;AAIrD,qBAAa,WAAW;IACtB,OAAO,CAAC,QAAQ,CAAC,UAAU,CAAa;IACxC,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAiB;IACxC,OAAO,CAAC,QAAQ,CAAC,UAAU,CAA0B;gBAEzC,UAAU,EAAE,UAAU,EAAE,MAAM,EAAE,cAAc;IAK1D;;;;;;OAMG;IACH,gBAAgB,WACN,MAAM,cAAc,CAAC,SACtB,WAAW,aACP,MAAM,sBAgDjB;IAEF;;;;OAIG;IACH,WAAW,UAAiB,cAAc,aAAa,MAAM,6CAY3D;IAEF;;;;OAIG;IACH,WAAW,UAAiB,cAAc,aAAa,MAAM,6CAe3D;IAEF;;;;;OAKG;IACH,WAAW,aACC,MAAM,MAAM,CAAC,SAChB,WAAW,aACP,MAAM,QACX,MAAM,KACX,QAAQ,IAAI,CAAC,CAqBd;IAEF;;;;;;OAMG;IACH,WAAW,UAAiB,MAAM,aAAa,MAAM,KAAG,QAAQ,IAAI,CAAC,CAgCnE;IAEF;;OAEG;IACH,YAAY,cAAqB,MAAM,KAAG,QAAQ,MAAM,MAAM,CAAC,CAAC,CAU9D;IAEF;;;;OAIG;IACH,WAAW,UAAiB,MAAM,aAAa,MAAM,6CAEnD;IAEF;;;;;;OAMG;IACH,gBAAgB,UACP,MAAM,aACF,MAAM,WACR,MAAM,eACF,MAAM,6CAMnB;CACH"}
|
|
@@ -0,0 +1,196 @@
|
|
|
1
|
+
import { In } from 'typeorm';
|
|
2
|
+
import { FraudEntity } from '../entities/fraudEntity';
|
|
3
|
+
import { dbIdChunkSize } from '../constants';
|
|
4
|
+
import { chunk } from 'lodash-es';
|
|
5
|
+
export class FraudAction {
|
|
6
|
+
datasource;
|
|
7
|
+
logger;
|
|
8
|
+
repository;
|
|
9
|
+
constructor(dataSource, logger) {
|
|
10
|
+
this.datasource = dataSource;
|
|
11
|
+
this.logger = logger;
|
|
12
|
+
this.repository = dataSource.getRepository(FraudEntity);
|
|
13
|
+
}
|
|
14
|
+
/**
|
|
15
|
+
* Store a list of frauds in a specific block
|
|
16
|
+
* @param frauds
|
|
17
|
+
* @param spendBoxes
|
|
18
|
+
* @param block
|
|
19
|
+
* @param extractor
|
|
20
|
+
*/
|
|
21
|
+
storeBlockFrauds = async (frauds, block, extractor) => {
|
|
22
|
+
const boxIds = frauds.map((item) => item.boxId);
|
|
23
|
+
const dbBoxes = await this.repository.findBy({
|
|
24
|
+
boxId: In(boxIds),
|
|
25
|
+
extractor: extractor,
|
|
26
|
+
});
|
|
27
|
+
let success = true;
|
|
28
|
+
const queryRunner = this.datasource.createQueryRunner();
|
|
29
|
+
await queryRunner.connect();
|
|
30
|
+
await queryRunner.startTransaction();
|
|
31
|
+
const repository = await queryRunner.manager.getRepository(FraudEntity);
|
|
32
|
+
try {
|
|
33
|
+
for (const fraud of frauds) {
|
|
34
|
+
const entity = {
|
|
35
|
+
triggerBoxId: fraud.triggerBoxId,
|
|
36
|
+
boxId: fraud.boxId,
|
|
37
|
+
wid: fraud.wid,
|
|
38
|
+
rwtCount: fraud.rwtCount,
|
|
39
|
+
creationBlock: block.hash,
|
|
40
|
+
creationHeight: block.height,
|
|
41
|
+
creationTxId: fraud.txId,
|
|
42
|
+
serialized: fraud.serialized,
|
|
43
|
+
extractor: extractor,
|
|
44
|
+
spendBlock: undefined,
|
|
45
|
+
spendHeight: undefined,
|
|
46
|
+
spendTxId: undefined,
|
|
47
|
+
};
|
|
48
|
+
const dbBox = dbBoxes.filter((item) => item.boxId === fraud.boxId);
|
|
49
|
+
if (dbBox.length > 0) {
|
|
50
|
+
this.logger.info(`Updating fraud with boxId [${fraud.boxId}]`);
|
|
51
|
+
this.logger.debug(`Updated fraud: [${JSON.stringify(entity)}]`);
|
|
52
|
+
await repository.update({ id: dbBox[0].id }, entity);
|
|
53
|
+
}
|
|
54
|
+
else {
|
|
55
|
+
this.logger.info(`Storing fraud with boxId: [${fraud.boxId}]`);
|
|
56
|
+
this.logger.debug(`Inserted fraud: [${JSON.stringify(entity)}]`);
|
|
57
|
+
await repository.insert(entity);
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
await queryRunner.commitTransaction();
|
|
61
|
+
}
|
|
62
|
+
catch (e) {
|
|
63
|
+
this.logger.error(`An error occurred during storing fraud boxes: ${e}`);
|
|
64
|
+
await queryRunner.rollbackTransaction();
|
|
65
|
+
success = false;
|
|
66
|
+
}
|
|
67
|
+
finally {
|
|
68
|
+
await queryRunner.release();
|
|
69
|
+
}
|
|
70
|
+
return success;
|
|
71
|
+
};
|
|
72
|
+
/**
|
|
73
|
+
* Insert a new fraud into database
|
|
74
|
+
* @param fraud
|
|
75
|
+
* @param extractor
|
|
76
|
+
*/
|
|
77
|
+
insertFraud = async (fraud, extractor) => {
|
|
78
|
+
return this.repository.insert({
|
|
79
|
+
boxId: fraud.boxId,
|
|
80
|
+
triggerBoxId: fraud.triggerBoxId,
|
|
81
|
+
wid: fraud.wid,
|
|
82
|
+
rwtCount: fraud.rwtCount,
|
|
83
|
+
creationBlock: fraud.blockId,
|
|
84
|
+
creationHeight: fraud.height,
|
|
85
|
+
creationTxId: fraud.txId,
|
|
86
|
+
serialized: fraud.serialized,
|
|
87
|
+
extractor: extractor,
|
|
88
|
+
});
|
|
89
|
+
};
|
|
90
|
+
/**
|
|
91
|
+
* Update an unspent fraud information in the database
|
|
92
|
+
* @param fraud
|
|
93
|
+
* @param extractor
|
|
94
|
+
*/
|
|
95
|
+
updateFraud = async (fraud, extractor) => {
|
|
96
|
+
return this.repository.update({ boxId: fraud.boxId, extractor: extractor }, {
|
|
97
|
+
triggerBoxId: fraud.triggerBoxId,
|
|
98
|
+
creationBlock: fraud.blockId,
|
|
99
|
+
creationHeight: fraud.height,
|
|
100
|
+
creationTxId: fraud.txId,
|
|
101
|
+
serialized: fraud.serialized,
|
|
102
|
+
wid: fraud.wid,
|
|
103
|
+
rwtCount: fraud.rwtCount,
|
|
104
|
+
spendBlock: null,
|
|
105
|
+
spendHeight: 0,
|
|
106
|
+
});
|
|
107
|
+
};
|
|
108
|
+
/**
|
|
109
|
+
* Update spendBlock and spendHeight of frauds spent on the block
|
|
110
|
+
* @param spendIds
|
|
111
|
+
* @param block
|
|
112
|
+
* @param extractor
|
|
113
|
+
*/
|
|
114
|
+
spendFrauds = async (spendIds, block, extractor, txId) => {
|
|
115
|
+
const spendIdChunks = chunk(spendIds, dbIdChunkSize);
|
|
116
|
+
for (const spendIdChunk of spendIdChunks) {
|
|
117
|
+
const updateResult = await this.repository.update({ boxId: In(spendIdChunk), extractor: extractor }, { spendBlock: block.hash, spendHeight: block.height, spendTxId: txId });
|
|
118
|
+
if (updateResult.affected && updateResult.affected > 0) {
|
|
119
|
+
const spentRows = await this.repository.findBy({
|
|
120
|
+
boxId: In(spendIdChunk),
|
|
121
|
+
spendBlock: block.hash,
|
|
122
|
+
spendTxId: txId,
|
|
123
|
+
});
|
|
124
|
+
for (const row of spentRows) {
|
|
125
|
+
this.logger.debug(`Spent box with boxId [${row.boxId}] at transaction [${txId}] at height ${block.height}`);
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
};
|
|
130
|
+
/**
|
|
131
|
+
* Update all frauds related to an specific invalid block
|
|
132
|
+
* if box had been spent in the block mark it as unspent,
|
|
133
|
+
* and if it was created within the block remove it from database
|
|
134
|
+
* @param block
|
|
135
|
+
* @param extractor
|
|
136
|
+
*/
|
|
137
|
+
deleteBlock = async (block, extractor) => {
|
|
138
|
+
this.logger.info(`Deleting frauds in block [${block}]`);
|
|
139
|
+
const invalidRows = await this.repository.findBy({
|
|
140
|
+
extractor: extractor,
|
|
141
|
+
creationBlock: block,
|
|
142
|
+
});
|
|
143
|
+
if (invalidRows.length > 0) {
|
|
144
|
+
await this.repository.delete({
|
|
145
|
+
extractor: extractor,
|
|
146
|
+
creationBlock: block,
|
|
147
|
+
});
|
|
148
|
+
for (const row of invalidRows) {
|
|
149
|
+
this.logger.debug(`deleted invalid fraud with boxId [${row.boxId}] at the forked block [${block}]`);
|
|
150
|
+
}
|
|
151
|
+
}
|
|
152
|
+
const updatingRows = await this.repository.findBy({
|
|
153
|
+
extractor: extractor,
|
|
154
|
+
spendBlock: block,
|
|
155
|
+
});
|
|
156
|
+
if (updatingRows.length > 0) {
|
|
157
|
+
await this.repository.update({ spendBlock: block, extractor: extractor }, { spendBlock: null, spendHeight: 0, spendTxId: null });
|
|
158
|
+
for (const row of updatingRows) {
|
|
159
|
+
this.logger.debug(`removed spending information of the fraud with boxId [${row.boxId}], spent at the forked block [${block}]`);
|
|
160
|
+
}
|
|
161
|
+
}
|
|
162
|
+
};
|
|
163
|
+
/**
|
|
164
|
+
* Return all stored fraud box ids
|
|
165
|
+
*/
|
|
166
|
+
getAllBoxIds = async (extractor) => {
|
|
167
|
+
const boxIds = await this.repository.find({
|
|
168
|
+
select: {
|
|
169
|
+
boxId: true,
|
|
170
|
+
},
|
|
171
|
+
where: {
|
|
172
|
+
extractor: extractor,
|
|
173
|
+
},
|
|
174
|
+
});
|
|
175
|
+
return boxIds.map((item) => item.boxId);
|
|
176
|
+
};
|
|
177
|
+
/**
|
|
178
|
+
* Remove an specified fraud
|
|
179
|
+
* @param boxId
|
|
180
|
+
* @param extractor
|
|
181
|
+
*/
|
|
182
|
+
removeFraud = async (boxId, extractor) => {
|
|
183
|
+
return await this.repository.delete({ boxId: boxId, extractor: extractor });
|
|
184
|
+
};
|
|
185
|
+
/**
|
|
186
|
+
* Update the fraud spending information
|
|
187
|
+
* @param boxId
|
|
188
|
+
* @param extractor
|
|
189
|
+
* @param blockId
|
|
190
|
+
* @param blockHeight
|
|
191
|
+
*/
|
|
192
|
+
updateSpendBlock = async (boxId, extractor, blockId, blockHeight) => {
|
|
193
|
+
return await this.repository.update({ boxId: boxId, extractor: extractor }, { spendBlock: blockId, spendHeight: blockHeight });
|
|
194
|
+
};
|
|
195
|
+
}
|
|
196
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fraudAction.js","sourceRoot":"","sources":["../../lib/actions/fraudAction.ts"],"names":[],"mappings":"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","sourcesContent":["import { DataSource, In, Repository } from 'typeorm';\nimport { AbstractLogger } from '@rosen-bridge/abstract-logger';\nimport { BlockEntity } from '@rosen-bridge/scanner';\n\nimport { FraudEntity } from '../entities/fraudEntity';\nimport { ExtractedFraud } from '../interfaces/types';\nimport { dbIdChunkSize } from '../constants';\nimport { chunk } from 'lodash-es';\n\nexport class FraudAction {\n  private readonly datasource: DataSource;\n  private readonly logger: AbstractLogger;\n  private readonly repository: Repository<FraudEntity>;\n\n  constructor(dataSource: DataSource, logger: AbstractLogger) {\n    this.datasource = dataSource;\n    this.logger = logger;\n    this.repository = dataSource.getRepository(FraudEntity);\n  }\n  /**\n   * Store a list of frauds in a specific block\n   * @param frauds\n   * @param spendBoxes\n   * @param block\n   * @param extractor\n   */\n  storeBlockFrauds = async (\n    frauds: Array<ExtractedFraud>,\n    block: BlockEntity,\n    extractor: string\n  ) => {\n    const boxIds = frauds.map((item) => item.boxId);\n    const dbBoxes = await this.repository.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(FraudEntity);\n    try {\n      for (const fraud of frauds) {\n        const entity = {\n          triggerBoxId: fraud.triggerBoxId,\n          boxId: fraud.boxId,\n          wid: fraud.wid,\n          rwtCount: fraud.rwtCount,\n          creationBlock: block.hash,\n          creationHeight: block.height,\n          creationTxId: fraud.txId,\n          serialized: fraud.serialized,\n          extractor: extractor,\n          spendBlock: undefined,\n          spendHeight: undefined,\n          spendTxId: undefined,\n        };\n        const dbBox = dbBoxes.filter((item) => item.boxId === fraud.boxId);\n        if (dbBox.length > 0) {\n          this.logger.info(`Updating fraud with boxId [${fraud.boxId}]`);\n          this.logger.debug(`Updated fraud: [${JSON.stringify(entity)}]`);\n          await repository.update({ id: dbBox[0].id }, entity);\n        } else {\n          this.logger.info(`Storing fraud with boxId: [${fraud.boxId}]`);\n          this.logger.debug(`Inserted fraud: [${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 storing fraud boxes: ${e}`);\n      await queryRunner.rollbackTransaction();\n      success = false;\n    } finally {\n      await queryRunner.release();\n    }\n    return success;\n  };\n\n  /**\n   * Insert a new fraud into database\n   * @param fraud\n   * @param extractor\n   */\n  insertFraud = async (fraud: ExtractedFraud, extractor: string) => {\n    return this.repository.insert({\n      boxId: fraud.boxId,\n      triggerBoxId: fraud.triggerBoxId,\n      wid: fraud.wid,\n      rwtCount: fraud.rwtCount,\n      creationBlock: fraud.blockId,\n      creationHeight: fraud.height,\n      creationTxId: fraud.txId,\n      serialized: fraud.serialized,\n      extractor: extractor,\n    });\n  };\n\n  /**\n   * Update an unspent fraud information in the database\n   * @param fraud\n   * @param extractor\n   */\n  updateFraud = async (fraud: ExtractedFraud, extractor: string) => {\n    return this.repository.update(\n      { boxId: fraud.boxId, extractor: extractor },\n      {\n        triggerBoxId: fraud.triggerBoxId,\n        creationBlock: fraud.blockId,\n        creationHeight: fraud.height,\n        creationTxId: fraud.txId,\n        serialized: fraud.serialized,\n        wid: fraud.wid,\n        rwtCount: fraud.rwtCount,\n        spendBlock: null,\n        spendHeight: 0,\n      }\n    );\n  };\n\n  /**\n   * Update spendBlock and spendHeight of frauds spent on the block\n   * @param spendIds\n   * @param block\n   * @param extractor\n   */\n  spendFrauds = async (\n    spendIds: Array<string>,\n    block: BlockEntity,\n    extractor: string,\n    txId: 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, spendTxId: txId }\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          spendTxId: txId,\n        });\n        for (const row of spentRows) {\n          this.logger.debug(\n            `Spent box with boxId [${row.boxId}] at transaction [${txId}] at height ${block.height}`\n          );\n        }\n      }\n    }\n  };\n\n  /**\n   * Update all frauds related to an specific invalid block\n   * if box had been spent in the block mark it as unspent,\n   * and if it was created within the block remove it from database\n   * @param block\n   * @param extractor\n   */\n  deleteBlock = async (block: string, extractor: string): Promise<void> => {\n    this.logger.info(`Deleting frauds in block [${block}]`);\n    const invalidRows = await this.repository.findBy({\n      extractor: extractor,\n      creationBlock: block,\n    });\n    if (invalidRows.length > 0) {\n      await this.repository.delete({\n        extractor: extractor,\n        creationBlock: block,\n      });\n      for (const row of invalidRows) {\n        this.logger.debug(\n          `deleted invalid fraud with boxId [${row.boxId}] at the forked block [${block}]`\n        );\n      }\n    }\n    const updatingRows = await this.repository.findBy({\n      extractor: extractor,\n      spendBlock: block,\n    });\n    if (updatingRows.length > 0) {\n      await this.repository.update(\n        { spendBlock: block, extractor: extractor },\n        { spendBlock: null, spendHeight: 0, spendTxId: null }\n      );\n      for (const row of updatingRows) {\n        this.logger.debug(\n          `removed spending information of the fraud with boxId [${row.boxId}], spent at the forked block [${block}]`\n        );\n      }\n    }\n  };\n\n  /**\n   * Return all stored fraud 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   * Remove an specified fraud\n   * @param boxId\n   * @param extractor\n   */\n  removeFraud = async (boxId: string, extractor: string) => {\n    return await this.repository.delete({ boxId: boxId, extractor: extractor });\n  };\n\n  /**\n   * Update the fraud 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"]}
|
|
@@ -0,0 +1 @@
|
|
|
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"}
|
|
@@ -0,0 +1,4 @@
|
|
|
1
|
+
const DefaultApiLimit = 100;
|
|
2
|
+
const dbIdChunkSize = 100;
|
|
3
|
+
export { DefaultApiLimit, dbIdChunkSize };
|
|
4
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uc3RhbnRzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vbGliL2NvbnN0YW50cy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxNQUFNLGVBQWUsR0FBRyxHQUFHLENBQUM7QUFDNUIsTUFBTSxhQUFhLEdBQUcsR0FBRyxDQUFDO0FBRTFCLE9BQU8sRUFBRSxlQUFlLEVBQUUsYUFBYSxFQUFFLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJjb25zdCBEZWZhdWx0QXBpTGltaXQgPSAxMDA7XG5jb25zdCBkYklkQ2h1bmtTaXplID0gMTAwO1xuXG5leHBvcnQgeyBEZWZhdWx0QXBpTGltaXQsIGRiSWRDaHVua1NpemUgfTtcbiJdfQ==
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
export declare class FraudEntity {
|
|
2
|
+
id: number;
|
|
3
|
+
boxId: string;
|
|
4
|
+
creationBlock: string;
|
|
5
|
+
creationHeight: number;
|
|
6
|
+
creationTxId: string;
|
|
7
|
+
serialized: string;
|
|
8
|
+
triggerBoxId: string;
|
|
9
|
+
wid: string;
|
|
10
|
+
rwtCount: string;
|
|
11
|
+
spendBlock?: string | null;
|
|
12
|
+
spendHeight?: number;
|
|
13
|
+
spendTxId?: string | null;
|
|
14
|
+
extractor: string;
|
|
15
|
+
}
|
|
16
|
+
//# sourceMappingURL=fraudEntity.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"fraudEntity.d.ts","sourceRoot":"","sources":["../../lib/entities/fraudEntity.ts"],"names":[],"mappings":"AAEA,qBACa,WAAW;IAEtB,EAAE,EAAE,MAAM,CAAC;IAGX,KAAK,EAAE,MAAM,CAAC;IAGd,aAAa,EAAE,MAAM,CAAC;IAGtB,cAAc,EAAE,MAAM,CAAC;IAGvB,YAAY,EAAE,MAAM,CAAC;IAGrB,UAAU,EAAE,MAAM,CAAC;IAGnB,YAAY,EAAE,MAAM,CAAC;IAGrB,GAAG,EAAE,MAAM,CAAC;IAGZ,QAAQ,EAAE,MAAM,CAAC;IAGjB,UAAU,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IAG3B,WAAW,CAAC,EAAE,MAAM,CAAC;IAGrB,SAAS,CAAC,EAAE,MAAM,GAAG,IAAI,CAAC;IAG1B,SAAS,EAAE,MAAM,CAAC;CACnB"}
|
|
@@ -0,0 +1,82 @@
|
|
|
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, Entity, PrimaryGeneratedColumn } from 'typeorm';
|
|
11
|
+
let FraudEntity = class FraudEntity {
|
|
12
|
+
id;
|
|
13
|
+
boxId;
|
|
14
|
+
creationBlock;
|
|
15
|
+
creationHeight;
|
|
16
|
+
creationTxId;
|
|
17
|
+
serialized;
|
|
18
|
+
triggerBoxId;
|
|
19
|
+
wid;
|
|
20
|
+
rwtCount;
|
|
21
|
+
spendBlock;
|
|
22
|
+
spendHeight;
|
|
23
|
+
spendTxId;
|
|
24
|
+
extractor;
|
|
25
|
+
};
|
|
26
|
+
__decorate([
|
|
27
|
+
PrimaryGeneratedColumn(),
|
|
28
|
+
__metadata("design:type", Number)
|
|
29
|
+
], FraudEntity.prototype, "id", void 0);
|
|
30
|
+
__decorate([
|
|
31
|
+
Column(),
|
|
32
|
+
__metadata("design:type", String)
|
|
33
|
+
], FraudEntity.prototype, "boxId", void 0);
|
|
34
|
+
__decorate([
|
|
35
|
+
Column(),
|
|
36
|
+
__metadata("design:type", String)
|
|
37
|
+
], FraudEntity.prototype, "creationBlock", void 0);
|
|
38
|
+
__decorate([
|
|
39
|
+
Column(),
|
|
40
|
+
__metadata("design:type", Number)
|
|
41
|
+
], FraudEntity.prototype, "creationHeight", void 0);
|
|
42
|
+
__decorate([
|
|
43
|
+
Column(),
|
|
44
|
+
__metadata("design:type", String)
|
|
45
|
+
], FraudEntity.prototype, "creationTxId", void 0);
|
|
46
|
+
__decorate([
|
|
47
|
+
Column(),
|
|
48
|
+
__metadata("design:type", String)
|
|
49
|
+
], FraudEntity.prototype, "serialized", void 0);
|
|
50
|
+
__decorate([
|
|
51
|
+
Column(),
|
|
52
|
+
__metadata("design:type", String)
|
|
53
|
+
], FraudEntity.prototype, "triggerBoxId", void 0);
|
|
54
|
+
__decorate([
|
|
55
|
+
Column(),
|
|
56
|
+
__metadata("design:type", String)
|
|
57
|
+
], FraudEntity.prototype, "wid", void 0);
|
|
58
|
+
__decorate([
|
|
59
|
+
Column(),
|
|
60
|
+
__metadata("design:type", String)
|
|
61
|
+
], FraudEntity.prototype, "rwtCount", void 0);
|
|
62
|
+
__decorate([
|
|
63
|
+
Column({ nullable: true, type: 'text' }),
|
|
64
|
+
__metadata("design:type", Object)
|
|
65
|
+
], FraudEntity.prototype, "spendBlock", void 0);
|
|
66
|
+
__decorate([
|
|
67
|
+
Column({ nullable: true }),
|
|
68
|
+
__metadata("design:type", Number)
|
|
69
|
+
], FraudEntity.prototype, "spendHeight", void 0);
|
|
70
|
+
__decorate([
|
|
71
|
+
Column({ nullable: true, type: 'text' }),
|
|
72
|
+
__metadata("design:type", Object)
|
|
73
|
+
], FraudEntity.prototype, "spendTxId", void 0);
|
|
74
|
+
__decorate([
|
|
75
|
+
Column(),
|
|
76
|
+
__metadata("design:type", String)
|
|
77
|
+
], FraudEntity.prototype, "extractor", void 0);
|
|
78
|
+
FraudEntity = __decorate([
|
|
79
|
+
Entity()
|
|
80
|
+
], FraudEntity);
|
|
81
|
+
export { FraudEntity };
|
|
82
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZnJhdWRFbnRpdHkuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9saWIvZW50aXRpZXMvZnJhdWRFbnRpdHkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7O0FBQUEsT0FBTyxFQUFFLE1BQU0sRUFBRSxNQUFNLEVBQUUsc0JBQXNCLEVBQUUsTUFBTSxTQUFTLENBQUM7QUFHMUQsSUFBTSxXQUFXLEdBQWpCLE1BQU0sV0FBVztJQUV0QixFQUFFLENBQVM7SUFHWCxLQUFLLENBQVM7SUFHZCxhQUFhLENBQVM7SUFHdEIsY0FBYyxDQUFTO0lBR3ZCLFlBQVksQ0FBUztJQUdyQixVQUFVLENBQVM7SUFHbkIsWUFBWSxDQUFTO0lBR3JCLEdBQUcsQ0FBUztJQUdaLFFBQVEsQ0FBUztJQUdqQixVQUFVLENBQWlCO0lBRzNCLFdBQVcsQ0FBVTtJQUdyQixTQUFTLENBQWlCO0lBRzFCLFNBQVMsQ0FBUztDQUNuQixDQUFBO0FBdENDO0lBQUMsc0JBQXNCLEVBQUU7O3VDQUNkO0FBRVg7SUFBQyxNQUFNLEVBQUU7OzBDQUNLO0FBRWQ7SUFBQyxNQUFNLEVBQUU7O2tEQUNhO0FBRXRCO0lBQUMsTUFBTSxFQUFFOzttREFDYztBQUV2QjtJQUFDLE1BQU0sRUFBRTs7aURBQ1k7QUFFckI7SUFBQyxNQUFNLEVBQUU7OytDQUNVO0FBRW5CO0lBQUMsTUFBTSxFQUFFOztpREFDWTtBQUVyQjtJQUFDLE1BQU0sRUFBRTs7d0NBQ0c7QUFFWjtJQUFDLE1BQU0sRUFBRTs7NkNBQ1E7QUFFakI7SUFBQyxNQUFNLENBQUMsRUFBRSxRQUFRLEVBQUUsSUFBSSxFQUFFLElBQUksRUFBRSxNQUFNLEVBQUUsQ0FBQzs7K0NBQ2Q7QUFFM0I7SUFBQyxNQUFNLENBQUMsRUFBRSxRQUFRLEVBQUUsSUFBSSxFQUFFLENBQUM7O2dEQUNOO0FBRXJCO0lBQUMsTUFBTSxDQUFDLEVBQUUsUUFBUSxFQUFFLElBQUksRUFBRSxJQUFJLEVBQUUsTUFBTSxFQUFFLENBQUM7OzhDQUNmO0FBRTFCO0lBQUMsTUFBTSxFQUFFOzs4Q0FDUztBQXRDUCxXQUFXO0lBRHZCLE1BQU0sRUFBRTtHQUNJLFdBQVcsQ0F1Q3ZCO1NBdkNZLFdBQVciLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb2x1bW4sIEVudGl0eSwgUHJpbWFyeUdlbmVyYXRlZENvbHVtbiB9IGZyb20gJ3R5cGVvcm0nO1xuXG5ARW50aXR5KClcbmV4cG9ydCBjbGFzcyBGcmF1ZEVudGl0eSB7XG4gIEBQcmltYXJ5R2VuZXJhdGVkQ29sdW1uKClcbiAgaWQ6IG51bWJlcjtcblxuICBAQ29sdW1uKClcbiAgYm94SWQ6IHN0cmluZztcblxuICBAQ29sdW1uKClcbiAgY3JlYXRpb25CbG9jazogc3RyaW5nO1xuXG4gIEBDb2x1bW4oKVxuICBjcmVhdGlvbkhlaWdodDogbnVtYmVyO1xuXG4gIEBDb2x1bW4oKVxuICBjcmVhdGlvblR4SWQ6IHN0cmluZztcblxuICBAQ29sdW1uKClcbiAgc2VyaWFsaXplZDogc3RyaW5nO1xuXG4gIEBDb2x1bW4oKVxuICB0cmlnZ2VyQm94SWQ6IHN0cmluZztcblxuICBAQ29sdW1uKClcbiAgd2lkOiBzdHJpbmc7XG5cbiAgQENvbHVtbigpXG4gIHJ3dENvdW50OiBzdHJpbmc7XG5cbiAgQENvbHVtbih7IG51bGxhYmxlOiB0cnVlLCB0eXBlOiAndGV4dCcgfSlcbiAgc3BlbmRCbG9jaz86IHN0cmluZyB8IG51bGw7XG5cbiAgQENvbHVtbih7IG51bGxhYmxlOiB0cnVlIH0pXG4gIHNwZW5kSGVpZ2h0PzogbnVtYmVyO1xuXG4gIEBDb2x1bW4oeyBudWxsYWJsZTogdHJ1ZSwgdHlwZTogJ3RleHQnIH0pXG4gIHNwZW5kVHhJZD86IHN0cmluZyB8IG51bGw7XG5cbiAgQENvbHVtbigpXG4gIGV4dHJhY3Rvcjogc3RyaW5nO1xufVxuIl19
|
|
@@ -0,0 +1,144 @@
|
|
|
1
|
+
import { DataSource } from 'typeorm';
|
|
2
|
+
import { AbstractExtractor } from '@rosen-bridge/scanner';
|
|
3
|
+
import { AbstractLogger } from '@rosen-bridge/abstract-logger';
|
|
4
|
+
import { BlockEntity, Transaction } from '@rosen-bridge/scanner';
|
|
5
|
+
import { V1 } from '@rosen-clients/ergo-explorer';
|
|
6
|
+
import { ExtractedFraud } from '../interfaces/types';
|
|
7
|
+
export declare class FraudExtractor implements AbstractExtractor<Transaction> {
|
|
8
|
+
private readonly logger;
|
|
9
|
+
private readonly actions;
|
|
10
|
+
private readonly id;
|
|
11
|
+
private readonly ergoTree;
|
|
12
|
+
private readonly rwt;
|
|
13
|
+
readonly api: {
|
|
14
|
+
v0: {
|
|
15
|
+
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>;
|
|
16
|
+
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>;
|
|
17
|
+
getApiV0AddressesAssetholdersP1: (p1: string, params?: import("@rosen-clients/ergo-explorer/dist/src/v0/types").GetApiV0AddressesAssetholdersP1Params | undefined) => Promise<string[]>;
|
|
18
|
+
getApiV0AddressesBalances: (params?: import("@rosen-clients/ergo-explorer/dist/src/v0/types").GetApiV0AddressesBalancesParams | undefined) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").ItemsBalanceInfo>;
|
|
19
|
+
getApiV0AssetsIssuingboxes: (params?: import("@rosen-clients/ergo-explorer/dist/src/v0/types").GetApiV0AssetsIssuingboxesParams | undefined) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").ItemsOutputInfo>;
|
|
20
|
+
getApiV0AssetsP1Issuingbox: (p1: string) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").OutputInfo[]>;
|
|
21
|
+
getApiV0Blocks: (params?: import("@rosen-clients/ergo-explorer/dist/src/v0/types").GetApiV0BlocksParams | undefined) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").ItemsBlockInfo>;
|
|
22
|
+
getApiV0BlocksP1: (p1: string) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").BlockSummary>;
|
|
23
|
+
getApiV0BlocksAtP1: (p1: number) => Promise<string[]>;
|
|
24
|
+
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[]>;
|
|
25
|
+
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[]>;
|
|
26
|
+
getApiV0TransactionsP1: (p1: string) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").TransactionSummary>;
|
|
27
|
+
getApiV0TransactionsUnconfirmedP1: (p1: string) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").UTransactionSummary>;
|
|
28
|
+
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>;
|
|
29
|
+
getApiV0TransactionsUnconfirmed: (params?: import("@rosen-clients/ergo-explorer/dist/src/v0/types").GetApiV0TransactionsUnconfirmedParams | undefined) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").ItemsUTransactionInfo>;
|
|
30
|
+
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[]>;
|
|
31
|
+
postApiV0TransactionsSend: (postApiV0TransactionsSendBody: import("@rosen-clients/ergo-explorer/dist/src/v0/types").PostApiV0TransactionsSendBody) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").TxIdResponse>;
|
|
32
|
+
getApiV0TransactionsBoxesP1: (p1: string) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").OutputInfo>;
|
|
33
|
+
getApiV0TransactionsBoxesByergotreeP1: (p1: string) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").OutputInfo[]>;
|
|
34
|
+
getApiV0TransactionsBoxesByergotreeUnspentP1: (p1: string) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").OutputInfo[]>;
|
|
35
|
+
getApiV0TransactionsBoxesByaddressP1: (p1: string) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").OutputInfo[]>;
|
|
36
|
+
getApiV0TransactionsBoxesByaddressUnspentP1: (p1: string) => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").OutputInfo[]>;
|
|
37
|
+
getApiV0Info: () => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").BlockChainInfo>;
|
|
38
|
+
getApiV0InfoSupply: () => Promise<string>;
|
|
39
|
+
getApiV0Stats: () => Promise<import("@rosen-clients/ergo-explorer/dist/src/v0/types").StatsSummary>;
|
|
40
|
+
getApiV0DocsOpenapi: () => Promise<string>;
|
|
41
|
+
};
|
|
42
|
+
v1: {
|
|
43
|
+
getApiV1TransactionsP1: (p1: string) => Promise<V1.TransactionInfo>;
|
|
44
|
+
getApiV1TransactionsByinputsscripttemplatehashP1: (p1: string, params?: V1.GetApiV1TransactionsByinputsscripttemplatehashP1Params | undefined) => Promise<V1.ItemsTransactionInfo>;
|
|
45
|
+
getApiV1TransactionsByglobalindexStream: (params: V1.GetApiV1TransactionsByglobalindexStreamParams) => Promise<V1.ListTransactionInfo>;
|
|
46
|
+
getApiV1BoxesUnspentBylastepochsStream: (params: V1.GetApiV1BoxesUnspentBylastepochsStreamParams) => Promise<V1.ListOutputInfo>;
|
|
47
|
+
getApiV1BoxesUnspentByglobalindexStream: (params: V1.GetApiV1BoxesUnspentByglobalindexStreamParams) => Promise<V1.ListOutputInfo>;
|
|
48
|
+
getApiV1BoxesUnspentStream: (params: V1.GetApiV1BoxesUnspentStreamParams) => Promise<V1.ListOutputInfo>;
|
|
49
|
+
getApiV1BoxesByergotreetemplatehashP1Stream: (p1: string, params: V1.GetApiV1BoxesByergotreetemplatehashP1StreamParams) => Promise<V1.ListOutputInfo>;
|
|
50
|
+
getApiV1BoxesUnspentByergotreetemplatehashP1Stream: (p1: string, params: V1.GetApiV1BoxesUnspentByergotreetemplatehashP1StreamParams) => Promise<V1.ListOutputInfo>;
|
|
51
|
+
getApiV1BoxesUnspentBytokenidP1: (p1: string, params?: V1.GetApiV1BoxesUnspentBytokenidP1Params | undefined) => Promise<V1.ItemsOutputInfo>;
|
|
52
|
+
getApiV1BoxesBytokenidP1: (p1: string, params?: V1.GetApiV1BoxesBytokenidP1Params | undefined) => Promise<V1.ItemsOutputInfo>;
|
|
53
|
+
getApiV1BoxesP1: (p1: string) => Promise<V1.OutputInfo>;
|
|
54
|
+
getApiV1BoxesByergotreeP1: (p1: string, params?: V1.GetApiV1BoxesByergotreeP1Params | undefined) => Promise<V1.ItemsOutputInfo>;
|
|
55
|
+
getApiV1BoxesByergotreetemplatehashP1: (p1: string, params?: V1.GetApiV1BoxesByergotreetemplatehashP1Params | undefined) => Promise<V1.ItemsOutputInfo>;
|
|
56
|
+
getApiV1BoxesUnspentByergotreeP1: (p1: string, params?: V1.GetApiV1BoxesUnspentByergotreeP1Params | undefined) => Promise<V1.ItemsOutputInfo>;
|
|
57
|
+
getApiV1BoxesUnspentByergotreetemplatehashP1: (p1: string, params?: V1.GetApiV1BoxesUnspentByergotreetemplatehashP1Params | undefined) => Promise<V1.ItemsOutputInfo>;
|
|
58
|
+
getApiV1BoxesByaddressP1: (p1: string, params?: V1.GetApiV1BoxesByaddressP1Params | undefined) => Promise<V1.ItemsOutputInfo>;
|
|
59
|
+
getApiV1BoxesUnspentByaddressP1: (p1: string, params?: V1.GetApiV1BoxesUnspentByaddressP1Params | undefined) => Promise<V1.ItemsOutputInfo>;
|
|
60
|
+
getApiV1BoxesByglobalindexStream: (params: V1.GetApiV1BoxesByglobalindexStreamParams) => Promise<V1.ListOutputInfo>;
|
|
61
|
+
postApiV1BoxesUnspentSearchUnion: (boxAssetsQuery: V1.BoxAssetsQuery, params?: V1.PostApiV1BoxesUnspentSearchUnionParams | undefined) => Promise<V1.ItemsOutputInfo>;
|
|
62
|
+
postApiV1BoxesUnspentSearch: (boxQuery: V1.BoxQuery, params?: V1.PostApiV1BoxesUnspentSearchParams | undefined) => Promise<V1.ItemsOutputInfo>;
|
|
63
|
+
postApiV1BoxesSearch: (boxQuery: V1.BoxQuery, params?: V1.PostApiV1BoxesSearchParams | undefined) => Promise<V1.ItemsOutputInfo>;
|
|
64
|
+
getApiV1Tokens: (params?: V1.GetApiV1TokensParams | undefined) => Promise<V1.ItemsTokenInfo>;
|
|
65
|
+
getApiV1TokensSearch: (params: V1.GetApiV1TokensSearchParams) => Promise<V1.ItemsTokenInfo>;
|
|
66
|
+
getApiV1TokensBysymbolP1: (p1: string) => Promise<V1.TokenInfo[]>;
|
|
67
|
+
getApiV1TokensP1: (p1: string) => Promise<V1.TokenInfo>;
|
|
68
|
+
getApiV1Assets: (params?: V1.GetApiV1AssetsParams | undefined) => Promise<V1.ItemsTokenInfo>;
|
|
69
|
+
getApiV1AssetsSearchBytokenid: (params: V1.GetApiV1AssetsSearchBytokenidParams) => Promise<V1.ItemsAssetInfo>;
|
|
70
|
+
getApiV1EpochsParams: () => Promise<V1.EpochInfo>;
|
|
71
|
+
getApiV1AddressesP1Transactions: (p1: string, params?: V1.GetApiV1AddressesP1TransactionsParams | undefined) => Promise<V1.ItemsTransactionInfo>;
|
|
72
|
+
getApiV1AddressesP1BalanceConfirmed: (p1: string, params?: V1.GetApiV1AddressesP1BalanceConfirmedParams | undefined) => Promise<V1.Balance>;
|
|
73
|
+
getApiV1AddressesP1BalanceTotal: (p1: string) => Promise<V1.TotalBalance>;
|
|
74
|
+
getApiV1Blocks: (params?: V1.GetApiV1BlocksParams | undefined) => Promise<V1.ItemsBlockInfo>;
|
|
75
|
+
getApiV1BlocksP1: (p1: string) => Promise<V1.BlockSummary>;
|
|
76
|
+
getApiV1BlocksHeaders: (params?: V1.GetApiV1BlocksHeadersParams | undefined) => Promise<V1.ItemsBlockHeader>;
|
|
77
|
+
getApiV1BlocksByglobalindexStream: (params: V1.GetApiV1BlocksByglobalindexStreamParams) => Promise<V1.ListBlockInfo>;
|
|
78
|
+
getApiV1BlocksStreamSummary: (params?: V1.GetApiV1BlocksStreamSummaryParams | undefined) => Promise<V1.ListBlockSummaryV1>;
|
|
79
|
+
postApiV1MempoolTransactionsSubmit: (postApiV1MempoolTransactionsSubmitBody: V1.PostApiV1MempoolTransactionsSubmitBody) => Promise<V1.TxIdResponse>;
|
|
80
|
+
getApiV1MempoolTransactionsByaddressP1: (p1: string, params?: V1.GetApiV1MempoolTransactionsByaddressP1Params | undefined) => Promise<V1.ItemsUTransactionInfo>;
|
|
81
|
+
getApiV1MempoolBoxesUnspent: () => Promise<V1.ListOutputInfo>;
|
|
82
|
+
getApiV1Info: () => Promise<V1.NetworkState>;
|
|
83
|
+
getApiV1Networkstate: () => Promise<V1.NetworkState>;
|
|
84
|
+
getApiV1Networkstats: () => Promise<V1.NetworkStats>;
|
|
85
|
+
};
|
|
86
|
+
};
|
|
87
|
+
constructor(dataSource: DataSource, id: string, explorerUrl: string, fraudAddress: string, rwt: string, logger?: AbstractLogger);
|
|
88
|
+
/**
|
|
89
|
+
* get Id for current extractor
|
|
90
|
+
*/
|
|
91
|
+
getId: () => string;
|
|
92
|
+
/**
|
|
93
|
+
* Extract fraud boxes in the specified block transactions
|
|
94
|
+
* @param txs
|
|
95
|
+
* @param block
|
|
96
|
+
*/
|
|
97
|
+
processTransactions: (txs: Array<Transaction>, block: BlockEntity) => Promise<boolean>;
|
|
98
|
+
/**
|
|
99
|
+
* fork one block and remove all stored information for this block
|
|
100
|
+
* @param hash: block hash
|
|
101
|
+
*/
|
|
102
|
+
forkBlock: (hash: string) => Promise<void>;
|
|
103
|
+
/**
|
|
104
|
+
* Initializes the database with older frauds
|
|
105
|
+
*/
|
|
106
|
+
initializeBoxes: (initialHeight: number) => Promise<void>;
|
|
107
|
+
/**
|
|
108
|
+
* Validate all remaining boxes in the database
|
|
109
|
+
* update the correct ones and remove the invalid ones
|
|
110
|
+
* @param unchangedStoredBoxIds
|
|
111
|
+
* @param initialHeight
|
|
112
|
+
*/
|
|
113
|
+
validateOldStoredFrauds: (unchangedStoredBoxIds: Array<string>, initialHeight: number) => Promise<void>;
|
|
114
|
+
/**
|
|
115
|
+
* Return extracted information of a fraud with its boxId
|
|
116
|
+
* @param boxId
|
|
117
|
+
*/
|
|
118
|
+
getFraudInfoWithBoxId: (boxId: string) => Promise<ExtractedFraud | undefined>;
|
|
119
|
+
/**
|
|
120
|
+
* Get unspent frauds created bellow the initial height
|
|
121
|
+
* @param initialHeight
|
|
122
|
+
* @returns
|
|
123
|
+
*/
|
|
124
|
+
getUnspentFrauds: (initialHeight: number) => Promise<Array<ExtractedFraud>>;
|
|
125
|
+
/**
|
|
126
|
+
* Returns block information of tx
|
|
127
|
+
* @param txId
|
|
128
|
+
*/
|
|
129
|
+
getTxBlock: (txId: string) => Promise<{
|
|
130
|
+
id: string;
|
|
131
|
+
height: number;
|
|
132
|
+
}>;
|
|
133
|
+
/**
|
|
134
|
+
* Returns trigger boxId from fraud transaction id
|
|
135
|
+
* @param txId
|
|
136
|
+
*/
|
|
137
|
+
getTriggerBoxId: (txId: string) => Promise<string | undefined>;
|
|
138
|
+
/**
|
|
139
|
+
* Extract needed information for storing in database from api json outputs
|
|
140
|
+
* @param boxes
|
|
141
|
+
*/
|
|
142
|
+
extractBoxData: (boxes: Array<V1.OutputInfo>) => Promise<ExtractedFraud[]>;
|
|
143
|
+
}
|
|
144
|
+
//# sourceMappingURL=fraudExtractor.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"fraudExtractor.d.ts","sourceRoot":"","sources":["../../lib/extractor/fraudExtractor.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,SAAS,CAAC;AAIrC,OAAO,EAAE,iBAAiB,EAAE,MAAM,uBAAuB,CAAC;AAC1D,OAAO,EAAE,cAAc,EAAe,MAAM,+BAA+B,CAAC;AAC5E,OAAO,EAAE,WAAW,EAAE,WAAW,EAAE,MAAM,uBAAuB,CAAC;AAEjE,OAAO,EAAE,EAAE,EAAE,MAAM,8BAA8B,CAAC;AAKlD,OAAO,EAAE,cAAc,EAAE,MAAM,qBAAqB,CAAC;AAErD,qBAAa,cAAe,YAAW,iBAAiB,CAAC,WAAW,CAAC;IACnE,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAiB;IACxC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAc;IACtC,OAAO,CAAC,QAAQ,CAAC,EAAE,CAAS;IAC5B,OAAO,CAAC,QAAQ,CAAC,QAAQ,CAAS;IAClC,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAS;IAC7B,QAAQ,CAAC,GAAG;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;MAAC;gBAGX,UAAU,EAAE,UAAU,EACtB,EAAE,EAAE,MAAM,EACV,WAAW,EAAE,MAAM,EACnB,YAAY,EAAE,MAAM,EACpB,GAAG,EAAE,MAAM,EACX,MAAM,CAAC,EAAE,cAAc;IAYzB;;OAEG;IACH,KAAK,eAAsB;IAE3B;;;;OAIG;IACH,mBAAmB,QACZ,MAAM,WAAW,CAAC,SAChB,WAAW,KACjB,QAAQ,OAAO,CAAC,CA2EjB;IAEF;;;OAGG;IACH,SAAS,SAAgB,MAAM,KAAG,QAAQ,IAAI,CAAC,CAE7C;IAEF;;OAEG;IACH,eAAe,kBAAyB,MAAM,mBAkC5C;IAEF;;;;;OAKG;IACH,uBAAuB,0BACE,MAAM,MAAM,CAAC,iBACrB,MAAM,mBA2BrB;IAEF;;;OAGG;IACH,qBAAqB,UACZ,MAAM,KACZ,QAAQ,cAAc,GAAG,SAAS,CAAC,CAQpC;IAEF;;;;OAIG;IACH,gBAAgB,kBACC,MAAM,KACpB,QAAQ,MAAM,cAAc,CAAC,CAAC,CA+B/B;IAEF;;;OAGG;IACH,UAAU,SAAgB,MAAM;;;OAM9B;IAEF;;;OAGG;IACH,eAAe,SAAgB,MAAM,iCAGnC;IAEF;;;OAGG;IACH,cAAc,UAAiB,MAAM,GAAG,UAAU,CAAC,+BAoDjD;CACH"}
|
|
@@ -0,0 +1,262 @@
|
|
|
1
|
+
import * as wasm from 'ergo-lib-wasm-nodejs';
|
|
2
|
+
import { Buffer } from 'buffer';
|
|
3
|
+
import { difference } from 'lodash-es';
|
|
4
|
+
import { DummyLogger } from '@rosen-bridge/abstract-logger';
|
|
5
|
+
import ergoExplorerClientFactory from '@rosen-clients/ergo-explorer';
|
|
6
|
+
import JsonBI from '@rosen-bridge/json-bigint';
|
|
7
|
+
import { FraudAction } from '../actions/fraudAction';
|
|
8
|
+
import { DefaultApiLimit } from '../constants';
|
|
9
|
+
export class FraudExtractor {
|
|
10
|
+
logger;
|
|
11
|
+
actions;
|
|
12
|
+
id;
|
|
13
|
+
ergoTree;
|
|
14
|
+
rwt;
|
|
15
|
+
api;
|
|
16
|
+
constructor(dataSource, id, explorerUrl, fraudAddress, rwt, logger) {
|
|
17
|
+
this.id = id;
|
|
18
|
+
this.ergoTree = wasm.Address.from_base58(fraudAddress)
|
|
19
|
+
.to_ergo_tree()
|
|
20
|
+
.to_base16_bytes();
|
|
21
|
+
this.rwt = rwt;
|
|
22
|
+
this.logger = logger ? logger : new DummyLogger();
|
|
23
|
+
this.actions = new FraudAction(dataSource, this.logger);
|
|
24
|
+
this.api = ergoExplorerClientFactory(explorerUrl);
|
|
25
|
+
}
|
|
26
|
+
/**
|
|
27
|
+
* get Id for current extractor
|
|
28
|
+
*/
|
|
29
|
+
getId = () => `${this.id}`;
|
|
30
|
+
/**
|
|
31
|
+
* Extract fraud boxes in the specified block transactions
|
|
32
|
+
* @param txs
|
|
33
|
+
* @param block
|
|
34
|
+
*/
|
|
35
|
+
processTransactions = (txs, block) => {
|
|
36
|
+
return new Promise((resolve, reject) => {
|
|
37
|
+
try {
|
|
38
|
+
const newFrauds = [];
|
|
39
|
+
const txSpendIds = [];
|
|
40
|
+
txs.forEach((transaction) => {
|
|
41
|
+
for (const output of transaction.outputs) {
|
|
42
|
+
if (output.ergoTree !== this.ergoTree ||
|
|
43
|
+
!output.assets ||
|
|
44
|
+
output.assets[0].tokenId !== this.rwt) {
|
|
45
|
+
continue;
|
|
46
|
+
}
|
|
47
|
+
const boxOutput = wasm.ErgoBox.from_json(JsonBI.stringify(output));
|
|
48
|
+
const r4 = boxOutput
|
|
49
|
+
.register_value(wasm.NonMandatoryRegisterId.R4)
|
|
50
|
+
?.to_coll_coll_byte();
|
|
51
|
+
if (!r4) {
|
|
52
|
+
this.logger.debug(`A new fraud box found without correct wid format at height ${block.height}`);
|
|
53
|
+
continue;
|
|
54
|
+
}
|
|
55
|
+
const newFraud = {
|
|
56
|
+
boxId: output.boxId,
|
|
57
|
+
txId: transaction.id,
|
|
58
|
+
triggerBoxId: transaction.inputs[0].boxId,
|
|
59
|
+
wid: Buffer.from(r4[0]).toString('hex'),
|
|
60
|
+
rwtCount: output.assets[0].amount.toString(),
|
|
61
|
+
serialized: Buffer.from(wasm.ErgoBox.from_json(JsonBI.stringify(output)).sigma_serialize_bytes()).toString('base64'),
|
|
62
|
+
};
|
|
63
|
+
newFrauds.push(newFraud);
|
|
64
|
+
this.logger.debug(`new fraud found [${newFraud}] at height ${block.height}`);
|
|
65
|
+
}
|
|
66
|
+
txSpendIds.push({
|
|
67
|
+
txId: transaction.id,
|
|
68
|
+
spendBoxes: transaction.inputs.map((box) => box.boxId),
|
|
69
|
+
});
|
|
70
|
+
});
|
|
71
|
+
this.actions
|
|
72
|
+
.storeBlockFrauds(newFrauds, block, this.getId())
|
|
73
|
+
.then(async (status) => {
|
|
74
|
+
if (status) {
|
|
75
|
+
if (newFrauds.length > 0)
|
|
76
|
+
this.logger.debug(`successfully stored new frauds at hight ${block.height}`);
|
|
77
|
+
for (const spendIds of txSpendIds)
|
|
78
|
+
await this.actions.spendFrauds(spendIds.spendBoxes, block, this.getId(), spendIds.txId);
|
|
79
|
+
}
|
|
80
|
+
resolve(status);
|
|
81
|
+
})
|
|
82
|
+
.catch((e) => {
|
|
83
|
+
this.logger.error(`An error occurred in processing frauds in block [${block.hash}]: ${e}`);
|
|
84
|
+
reject(e);
|
|
85
|
+
});
|
|
86
|
+
}
|
|
87
|
+
catch (e) {
|
|
88
|
+
reject(e);
|
|
89
|
+
}
|
|
90
|
+
});
|
|
91
|
+
};
|
|
92
|
+
/**
|
|
93
|
+
* fork one block and remove all stored information for this block
|
|
94
|
+
* @param hash: block hash
|
|
95
|
+
*/
|
|
96
|
+
forkBlock = async (hash) => {
|
|
97
|
+
await this.actions.deleteBlock(hash, this.getId());
|
|
98
|
+
};
|
|
99
|
+
/**
|
|
100
|
+
* Initializes the database with older frauds
|
|
101
|
+
*/
|
|
102
|
+
initializeBoxes = async (initialHeight) => {
|
|
103
|
+
// Getting unspent boxes
|
|
104
|
+
this.logger.debug(`Initializing fraud table. storing fraud boxes created bellow height ${initialHeight}.`);
|
|
105
|
+
const unspentFrauds = await this.getUnspentFrauds(initialHeight);
|
|
106
|
+
const unspentBoxIds = unspentFrauds.map((box) => box.boxId);
|
|
107
|
+
this.logger.debug(`Unspent fraud boxIds ${unspentBoxIds}`);
|
|
108
|
+
// Storing extracted boxes
|
|
109
|
+
let allStoredBoxIds = await this.actions.getAllBoxIds(this.getId());
|
|
110
|
+
for (const fraud of unspentFrauds) {
|
|
111
|
+
if (allStoredBoxIds.includes(fraud.boxId)) {
|
|
112
|
+
await this.actions.updateFraud(fraud, this.getId());
|
|
113
|
+
this.logger.info(`Updated the existing unspent fraud with boxId, [${fraud.boxId}]`);
|
|
114
|
+
this.logger.debug(`Updated fraud: [${JSON.stringify(fraud)}]`);
|
|
115
|
+
}
|
|
116
|
+
else {
|
|
117
|
+
await this.actions.insertFraud(fraud, this.getId());
|
|
118
|
+
this.logger.info(`Inserted new unspent fraud with boxId, [${fraud.boxId}]`);
|
|
119
|
+
this.logger.debug(`Inserted fraud: [${JSON.stringify(fraud)}]`);
|
|
120
|
+
}
|
|
121
|
+
}
|
|
122
|
+
// Remove updated box ids from existing boxes in database
|
|
123
|
+
allStoredBoxIds = difference(allStoredBoxIds, unspentBoxIds);
|
|
124
|
+
// Validating remained boxes
|
|
125
|
+
this.logger.debug(`Validating and updating stored fraud boxes with boxIds ${allStoredBoxIds}`);
|
|
126
|
+
await this.validateOldStoredFrauds(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
|
+
validateOldStoredFrauds = async (unchangedStoredBoxIds, initialHeight) => {
|
|
135
|
+
for (const boxId of unchangedStoredBoxIds) {
|
|
136
|
+
const box = await this.getFraudInfoWithBoxId(boxId);
|
|
137
|
+
if (box && box.spendBlock && box.spendHeight) {
|
|
138
|
+
if (box.spendHeight < initialHeight) {
|
|
139
|
+
this.logger.debug(`updating spending information of fraud with boxId [${box.boxId}] spent at height [${box.spendHeight}]`);
|
|
140
|
+
await this.actions.updateSpendBlock(boxId, this.getId(), box.spendBlock, box.spendHeight);
|
|
141
|
+
}
|
|
142
|
+
else {
|
|
143
|
+
this.logger.debug(`fraud with boxId [${box.boxId}] has been spent after the initialization height, updating spending information skipped.`);
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
else {
|
|
147
|
+
await this.actions.removeFraud(boxId, this.getId());
|
|
148
|
+
this.logger.info(`Removed invalid fraud [${boxId}] in initialization validation`);
|
|
149
|
+
}
|
|
150
|
+
}
|
|
151
|
+
};
|
|
152
|
+
/**
|
|
153
|
+
* Return extracted information of a fraud with its boxId
|
|
154
|
+
* @param boxId
|
|
155
|
+
*/
|
|
156
|
+
getFraudInfoWithBoxId = async (boxId) => {
|
|
157
|
+
try {
|
|
158
|
+
const box = await this.api.v1.getApiV1BoxesP1(boxId);
|
|
159
|
+
return (await this.extractBoxData([box]))[0];
|
|
160
|
+
}
|
|
161
|
+
catch {
|
|
162
|
+
this.logger.warn(`Box with id [${boxId}] does not exists`);
|
|
163
|
+
return undefined;
|
|
164
|
+
}
|
|
165
|
+
};
|
|
166
|
+
/**
|
|
167
|
+
* Get unspent frauds created bellow the initial height
|
|
168
|
+
* @param initialHeight
|
|
169
|
+
* @returns
|
|
170
|
+
*/
|
|
171
|
+
getUnspentFrauds = async (initialHeight) => {
|
|
172
|
+
let allBoxes = [];
|
|
173
|
+
let offset = 0, total = DefaultApiLimit, boxes;
|
|
174
|
+
while (offset < total) {
|
|
175
|
+
boxes = await this.api.v1.getApiV1BoxesUnspentByergotreeP1(this.ergoTree, {
|
|
176
|
+
offset: offset,
|
|
177
|
+
limit: DefaultApiLimit,
|
|
178
|
+
});
|
|
179
|
+
if (!boxes.items) {
|
|
180
|
+
throw new Error('Explorer api output items should not be undefined.');
|
|
181
|
+
}
|
|
182
|
+
allBoxes = [
|
|
183
|
+
...allBoxes,
|
|
184
|
+
...(await this.extractBoxData(boxes.items.filter((box) => box.creationHeight < initialHeight &&
|
|
185
|
+
box.assets &&
|
|
186
|
+
box.assets[0].tokenId == this.rwt))),
|
|
187
|
+
];
|
|
188
|
+
total = boxes.total;
|
|
189
|
+
offset += DefaultApiLimit;
|
|
190
|
+
}
|
|
191
|
+
return allBoxes;
|
|
192
|
+
};
|
|
193
|
+
/**
|
|
194
|
+
* Returns block information of tx
|
|
195
|
+
* @param txId
|
|
196
|
+
*/
|
|
197
|
+
getTxBlock = async (txId) => {
|
|
198
|
+
const tx = await this.api.v1.getApiV1TransactionsP1(txId);
|
|
199
|
+
return {
|
|
200
|
+
id: tx.blockId,
|
|
201
|
+
height: tx.inclusionHeight,
|
|
202
|
+
};
|
|
203
|
+
};
|
|
204
|
+
/**
|
|
205
|
+
* Returns trigger boxId from fraud transaction id
|
|
206
|
+
* @param txId
|
|
207
|
+
*/
|
|
208
|
+
getTriggerBoxId = async (txId) => {
|
|
209
|
+
const tx = await this.api.v1.getApiV1TransactionsP1(txId);
|
|
210
|
+
return tx.inputs?.[0].boxId;
|
|
211
|
+
};
|
|
212
|
+
/**
|
|
213
|
+
* Extract needed information for storing in database from api json outputs
|
|
214
|
+
* @param boxes
|
|
215
|
+
*/
|
|
216
|
+
extractBoxData = async (boxes) => {
|
|
217
|
+
const extractedFrauds = [];
|
|
218
|
+
for (const box of boxes) {
|
|
219
|
+
try {
|
|
220
|
+
this.logger.debug(`Extracting box fraud information from box with boxId [${box.boxId}]`);
|
|
221
|
+
let spendBlock, spendHeight;
|
|
222
|
+
// Extract spending information
|
|
223
|
+
if (box.spentTransactionId) {
|
|
224
|
+
const block = await this.getTxBlock(box.spentTransactionId);
|
|
225
|
+
spendBlock = block.id;
|
|
226
|
+
spendHeight = block.height;
|
|
227
|
+
}
|
|
228
|
+
// Extract WID
|
|
229
|
+
const ergoBox = wasm.ErgoBox.from_json(JsonBI.stringify(box));
|
|
230
|
+
const r4 = ergoBox
|
|
231
|
+
.register_value(wasm.NonMandatoryRegisterId.R4)
|
|
232
|
+
?.to_coll_coll_byte();
|
|
233
|
+
// Extract trigger boxId
|
|
234
|
+
const triggerBoxId = await this.getTriggerBoxId(box.transactionId);
|
|
235
|
+
if (!r4 || !triggerBoxId) {
|
|
236
|
+
this.logger.warn(`Skipping storing fraud with boxId [${box.boxId}], wid or trigger box id is undefined`);
|
|
237
|
+
continue;
|
|
238
|
+
}
|
|
239
|
+
const extractedFraud = {
|
|
240
|
+
boxId: ergoBox.box_id().to_str(),
|
|
241
|
+
triggerBoxId: triggerBoxId,
|
|
242
|
+
wid: Buffer.from(r4[0]).toString('hex'),
|
|
243
|
+
rwtCount: box.assets[0].amount.toString(),
|
|
244
|
+
serialized: Buffer.from(ergoBox.sigma_serialize_bytes()).toString('base64'),
|
|
245
|
+
blockId: box.blockId,
|
|
246
|
+
height: box.settlementHeight,
|
|
247
|
+
txId: box.transactionId,
|
|
248
|
+
spendBlock: spendBlock,
|
|
249
|
+
spendHeight: spendHeight,
|
|
250
|
+
spendTxId: box.spentTransactionId,
|
|
251
|
+
};
|
|
252
|
+
extractedFrauds.push(extractedFraud);
|
|
253
|
+
this.logger.debug(`Extracted fraud: [${extractedFraud}]`);
|
|
254
|
+
}
|
|
255
|
+
catch (e) {
|
|
256
|
+
this.logger.error(`Extracting fraud information failed for box [${box.boxId}] with error: ${e}`);
|
|
257
|
+
}
|
|
258
|
+
}
|
|
259
|
+
return extractedFrauds;
|
|
260
|
+
};
|
|
261
|
+
}
|
|
262
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fraudExtractor.js","sourceRoot":"","sources":["../../lib/extractor/fraudExtractor.ts"],"names":[],"mappings":"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","sourcesContent":["import { DataSource } from 'typeorm';\nimport * as wasm from 'ergo-lib-wasm-nodejs';\nimport { Buffer } from 'buffer';\nimport { difference } from 'lodash-es';\nimport { AbstractExtractor } from '@rosen-bridge/scanner';\nimport { AbstractLogger, DummyLogger } from '@rosen-bridge/abstract-logger';\nimport { BlockEntity, Transaction } from '@rosen-bridge/scanner';\nimport ergoExplorerClientFactory from '@rosen-clients/ergo-explorer';\nimport { V1 } from '@rosen-clients/ergo-explorer';\nimport JsonBI from '@rosen-bridge/json-bigint';\n\nimport { FraudAction } from '../actions/fraudAction';\nimport { DefaultApiLimit } from '../constants';\nimport { ExtractedFraud } from '../interfaces/types';\n\nexport class FraudExtractor implements AbstractExtractor<Transaction> {\n  private readonly logger: AbstractLogger;\n  private readonly actions: FraudAction;\n  private readonly id: string;\n  private readonly ergoTree: string;\n  private readonly rwt: string;\n  readonly api;\n\n  constructor(\n    dataSource: DataSource,\n    id: string,\n    explorerUrl: string,\n    fraudAddress: string,\n    rwt: string,\n    logger?: AbstractLogger\n  ) {\n    this.id = id;\n    this.ergoTree = wasm.Address.from_base58(fraudAddress)\n      .to_ergo_tree()\n      .to_base16_bytes();\n    this.rwt = rwt;\n    this.logger = logger ? logger : new DummyLogger();\n    this.actions = new FraudAction(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   * Extract fraud boxes in the specified block transactions\n   * @param txs\n   * @param block\n   */\n  processTransactions = (\n    txs: Array<Transaction>,\n    block: BlockEntity\n  ): Promise<boolean> => {\n    return new Promise((resolve, reject) => {\n      try {\n        const newFrauds: Array<ExtractedFraud> = [];\n        const txSpendIds: Array<{ txId: string; spendBoxes: Array<string> }> =\n          [];\n        txs.forEach((transaction) => {\n          for (const output of transaction.outputs) {\n            if (\n              output.ergoTree !== this.ergoTree ||\n              !output.assets ||\n              output.assets[0].tokenId !== this.rwt\n            ) {\n              continue;\n            }\n            const boxOutput = wasm.ErgoBox.from_json(JsonBI.stringify(output));\n            const r4 = boxOutput\n              .register_value(wasm.NonMandatoryRegisterId.R4)\n              ?.to_coll_coll_byte();\n            if (!r4) {\n              this.logger.debug(\n                `A new fraud box found without correct wid format at height ${block.height}`\n              );\n              continue;\n            }\n            const newFraud = {\n              boxId: output.boxId,\n              txId: transaction.id,\n              triggerBoxId: transaction.inputs[0].boxId,\n              wid: Buffer.from(r4[0]).toString('hex'),\n              rwtCount: output.assets[0].amount.toString(),\n              serialized: Buffer.from(\n                wasm.ErgoBox.from_json(\n                  JsonBI.stringify(output)\n                ).sigma_serialize_bytes()\n              ).toString('base64'),\n            };\n            newFrauds.push(newFraud);\n            this.logger.debug(\n              `new fraud found [${newFraud}] at height ${block.height}`\n            );\n          }\n          txSpendIds.push({\n            txId: transaction.id,\n            spendBoxes: transaction.inputs.map((box) => box.boxId),\n          });\n        });\n        this.actions\n          .storeBlockFrauds(newFrauds, block, this.getId())\n          .then(async (status) => {\n            if (status) {\n              if (newFrauds.length > 0)\n                this.logger.debug(\n                  `successfully stored new frauds at hight ${block.height}`\n                );\n              for (const spendIds of txSpendIds)\n                await this.actions.spendFrauds(\n                  spendIds.spendBoxes,\n                  block,\n                  this.getId(),\n                  spendIds.txId\n                );\n            }\n            resolve(status);\n          })\n          .catch((e) => {\n            this.logger.error(\n              `An error occurred in processing frauds in block [${block.hash}]: ${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.deleteBlock(hash, this.getId());\n  };\n\n  /**\n   * Initializes the database with older frauds\n   */\n  initializeBoxes = async (initialHeight: number) => {\n    // Getting unspent boxes\n    this.logger.debug(\n      `Initializing fraud table. storing fraud boxes created bellow height ${initialHeight}.`\n    );\n    const unspentFrauds = await this.getUnspentFrauds(initialHeight);\n    const unspentBoxIds = unspentFrauds.map((box) => box.boxId);\n    this.logger.debug(`Unspent fraud boxIds ${unspentBoxIds}`);\n\n    // Storing extracted boxes\n    let allStoredBoxIds = await this.actions.getAllBoxIds(this.getId());\n    for (const fraud of unspentFrauds) {\n      if (allStoredBoxIds.includes(fraud.boxId)) {\n        await this.actions.updateFraud(fraud, this.getId());\n        this.logger.info(\n          `Updated the existing unspent fraud with boxId, [${fraud.boxId}]`\n        );\n        this.logger.debug(`Updated fraud: [${JSON.stringify(fraud)}]`);\n      } else {\n        await this.actions.insertFraud(fraud, this.getId());\n        this.logger.info(\n          `Inserted new unspent fraud with boxId, [${fraud.boxId}]`\n        );\n        this.logger.debug(`Inserted fraud: [${JSON.stringify(fraud)}]`);\n      }\n    }\n\n    // Remove updated box ids from existing boxes in database\n    allStoredBoxIds = difference(allStoredBoxIds, unspentBoxIds);\n    // Validating remained boxes\n    this.logger.debug(\n      `Validating and updating stored fraud boxes with boxIds ${allStoredBoxIds}`\n    );\n    await this.validateOldStoredFrauds(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  validateOldStoredFrauds = async (\n    unchangedStoredBoxIds: Array<string>,\n    initialHeight: number\n  ) => {\n    for (const boxId of unchangedStoredBoxIds) {\n      const box = await this.getFraudInfoWithBoxId(boxId);\n      if (box && box.spendBlock && box.spendHeight) {\n        if (box.spendHeight < initialHeight) {\n          this.logger.debug(\n            `updating spending information of fraud with boxId [${box.boxId}] spent at height [${box.spendHeight}]`\n          );\n          await this.actions.updateSpendBlock(\n            boxId,\n            this.getId(),\n            box.spendBlock,\n            box.spendHeight\n          );\n        } else {\n          this.logger.debug(\n            `fraud with boxId [${box.boxId}] has been spent after the initialization height, updating spending information skipped.`\n          );\n        }\n      } else {\n        await this.actions.removeFraud(boxId, this.getId());\n        this.logger.info(\n          `Removed invalid fraud [${boxId}] in initialization validation`\n        );\n      }\n    }\n  };\n\n  /**\n   * Return extracted information of a fraud with its boxId\n   * @param boxId\n   */\n  getFraudInfoWithBoxId = async (\n    boxId: string\n  ): Promise<ExtractedFraud | 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 frauds created bellow the initial height\n   * @param initialHeight\n   * @returns\n   */\n  getUnspentFrauds = async (\n    initialHeight: number\n  ): Promise<Array<ExtractedFraud>> => {\n    let allBoxes: Array<ExtractedFraud> = [];\n    let offset = 0,\n      total = DefaultApiLimit,\n      boxes: V1.ItemsOutputInfo;\n    while (offset < total) {\n      boxes = await this.api.v1.getApiV1BoxesUnspentByergotreeP1(\n        this.ergoTree,\n        {\n          offset: offset,\n          limit: DefaultApiLimit,\n        }\n      );\n      if (!boxes.items) {\n        throw new Error('Explorer api output items should not be undefined.');\n      }\n      allBoxes = [\n        ...allBoxes,\n        ...(await this.extractBoxData(\n          boxes.items.filter(\n            (box) =>\n              box.creationHeight < initialHeight &&\n              box.assets &&\n              box.assets[0].tokenId == this.rwt\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   * Returns trigger boxId from fraud transaction id\n   * @param txId\n   */\n  getTriggerBoxId = async (txId: string) => {\n    const tx = await this.api.v1.getApiV1TransactionsP1(txId);\n    return tx.inputs?.[0].boxId;\n  };\n\n  /**\n   * Extract needed information for storing in database from api json outputs\n   * @param boxes\n   */\n  extractBoxData = async (boxes: Array<V1.OutputInfo>) => {\n    const extractedFrauds: Array<ExtractedFraud> = [];\n    for (const box of boxes) {\n      try {\n        this.logger.debug(\n          `Extracting box fraud information from box with boxId [${box.boxId}]`\n        );\n        let spendBlock, spendHeight;\n        // Extract spending information\n        if (box.spentTransactionId) {\n          const block = await this.getTxBlock(box.spentTransactionId);\n          spendBlock = block.id;\n          spendHeight = block.height;\n        }\n        // Extract WID\n        const ergoBox = wasm.ErgoBox.from_json(JsonBI.stringify(box));\n        const r4 = ergoBox\n          .register_value(wasm.NonMandatoryRegisterId.R4)\n          ?.to_coll_coll_byte();\n        // Extract trigger boxId\n        const triggerBoxId = await this.getTriggerBoxId(box.transactionId);\n        if (!r4 || !triggerBoxId) {\n          this.logger.warn(\n            `Skipping storing fraud with boxId [${box.boxId}], wid or trigger box id is undefined`\n          );\n          continue;\n        }\n\n        const extractedFraud = {\n          boxId: ergoBox.box_id().to_str(),\n          triggerBoxId: triggerBoxId,\n          wid: Buffer.from(r4[0]).toString('hex'),\n          rwtCount: box.assets![0].amount.toString(),\n          serialized: Buffer.from(ergoBox.sigma_serialize_bytes()).toString(\n            'base64'\n          ),\n          blockId: box.blockId,\n          height: box.settlementHeight,\n          txId: box.transactionId,\n          spendBlock: spendBlock,\n          spendHeight: spendHeight,\n          spendTxId: box.spentTransactionId,\n        };\n        extractedFrauds.push(extractedFraud);\n        this.logger.debug(`Extracted fraud: [${extractedFraud}]`);\n      } catch (e) {\n        this.logger.error(\n          `Extracting fraud information failed for box [${box.boxId}] with error: ${e}`\n        );\n      }\n    }\n    return extractedFrauds;\n  };\n}\n"]}
|
package/dist/index.d.ts
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../lib/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,cAAc,EAAE,MAAM,4BAA4B,CAAC;AAC5D,OAAO,EAAE,WAAW,EAAE,MAAM,wBAAwB,CAAC;AACrD,OAAO,EAAE,UAAU,EAAE,MAAM,oBAAoB,CAAC"}
|
package/dist/index.js
ADDED
|
@@ -0,0 +1,4 @@
|
|
|
1
|
+
export { FraudExtractor } from './extractor/fraudExtractor';
|
|
2
|
+
export { FraudEntity } from './entities/fraudEntity';
|
|
3
|
+
export { migrations } from './migrations/index';
|
|
4
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9saWIvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLDRCQUE0QixDQUFDO0FBQzVELE9BQU8sRUFBRSxXQUFXLEVBQUUsTUFBTSx3QkFBd0IsQ0FBQztBQUNyRCxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sb0JBQW9CLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgeyBGcmF1ZEV4dHJhY3RvciB9IGZyb20gJy4vZXh0cmFjdG9yL2ZyYXVkRXh0cmFjdG9yJztcbmV4cG9ydCB7IEZyYXVkRW50aXR5IH0gZnJvbSAnLi9lbnRpdGllcy9mcmF1ZEVudGl0eSc7XG5leHBvcnQgeyBtaWdyYXRpb25zIH0gZnJvbSAnLi9taWdyYXRpb25zL2luZGV4JztcbiJdfQ==
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
interface ExtractedFraud {
|
|
2
|
+
boxId: string;
|
|
3
|
+
serialized: string;
|
|
4
|
+
triggerBoxId: string;
|
|
5
|
+
wid: string;
|
|
6
|
+
rwtCount: string;
|
|
7
|
+
blockId?: string;
|
|
8
|
+
height?: number;
|
|
9
|
+
txId?: string;
|
|
10
|
+
spendBlock?: string;
|
|
11
|
+
spendHeight?: number;
|
|
12
|
+
spendTxId?: string;
|
|
13
|
+
}
|
|
14
|
+
export { ExtractedFraud };
|
|
15
|
+
//# sourceMappingURL=types.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"types.d.ts","sourceRoot":"","sources":["../../lib/interfaces/types.ts"],"names":[],"mappings":"AAAA,UAAU,cAAc;IACtB,KAAK,EAAE,MAAM,CAAC;IACd,UAAU,EAAE,MAAM,CAAC;IACnB,YAAY,EAAE,MAAM,CAAC;IACrB,GAAG,EAAE,MAAM,CAAC;IACZ,QAAQ,EAAE,MAAM,CAAC;IACjB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,SAAS,CAAC,EAAE,MAAM,CAAC;CACpB;AAED,OAAO,EAAE,cAAc,EAAE,CAAC"}
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export {};
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9saWIvaW50ZXJmYWNlcy90eXBlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiaW50ZXJmYWNlIEV4dHJhY3RlZEZyYXVkIHtcbiAgYm94SWQ6IHN0cmluZztcbiAgc2VyaWFsaXplZDogc3RyaW5nO1xuICB0cmlnZ2VyQm94SWQ6IHN0cmluZztcbiAgd2lkOiBzdHJpbmc7XG4gIHJ3dENvdW50OiBzdHJpbmc7XG4gIGJsb2NrSWQ/OiBzdHJpbmc7XG4gIGhlaWdodD86IG51bWJlcjtcbiAgdHhJZD86IHN0cmluZztcbiAgc3BlbmRCbG9jaz86IHN0cmluZztcbiAgc3BlbmRIZWlnaHQ/OiBudW1iZXI7XG4gIHNwZW5kVHhJZD86IHN0cmluZztcbn1cblxuZXhwb3J0IHsgRXh0cmFjdGVkRnJhdWQgfTtcbiJdfQ==
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
import { migration1693308311652 } from './postgres/1693308311652-migration';
|
|
2
|
+
import { migration1693122371215 } from './sqlite/1693122371215-migration';
|
|
3
|
+
export declare const migrations: {
|
|
4
|
+
sqlite: (typeof migration1693122371215)[];
|
|
5
|
+
postgres: (typeof migration1693308311652)[];
|
|
6
|
+
};
|
|
7
|
+
//# sourceMappingURL=index.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../lib/migrations/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,sBAAsB,EAAE,MAAM,oCAAoC,CAAC;AAC5E,OAAO,EAAE,sBAAsB,EAAE,MAAM,kCAAkC,CAAC;AAE1E,eAAO,MAAM,UAAU;;;CAGtB,CAAC"}
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
import { migration1693308311652 } from './postgres/1693308311652-migration';
|
|
2
|
+
import { migration1693122371215 } from './sqlite/1693122371215-migration';
|
|
3
|
+
export const migrations = {
|
|
4
|
+
sqlite: [migration1693122371215],
|
|
5
|
+
postgres: [migration1693308311652],
|
|
6
|
+
};
|
|
7
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9saWIvbWlncmF0aW9ucy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsc0JBQXNCLEVBQUUsTUFBTSxvQ0FBb0MsQ0FBQztBQUM1RSxPQUFPLEVBQUUsc0JBQXNCLEVBQUUsTUFBTSxrQ0FBa0MsQ0FBQztBQUUxRSxNQUFNLENBQUMsTUFBTSxVQUFVLEdBQUc7SUFDeEIsTUFBTSxFQUFFLENBQUMsc0JBQXNCLENBQUM7SUFDaEMsUUFBUSxFQUFFLENBQUMsc0JBQXNCLENBQUM7Q0FDbkMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IG1pZ3JhdGlvbjE2OTMzMDgzMTE2NTIgfSBmcm9tICcuL3Bvc3RncmVzLzE2OTMzMDgzMTE2NTItbWlncmF0aW9uJztcbmltcG9ydCB7IG1pZ3JhdGlvbjE2OTMxMjIzNzEyMTUgfSBmcm9tICcuL3NxbGl0ZS8xNjkzMTIyMzcxMjE1LW1pZ3JhdGlvbic7XG5cbmV4cG9ydCBjb25zdCBtaWdyYXRpb25zID0ge1xuICBzcWxpdGU6IFttaWdyYXRpb24xNjkzMTIyMzcxMjE1XSxcbiAgcG9zdGdyZXM6IFttaWdyYXRpb24xNjkzMzA4MzExNjUyXSxcbn07XG4iXX0=
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
import { MigrationInterface, QueryRunner } from 'typeorm';
|
|
2
|
+
export declare class migration1693308311652 implements MigrationInterface {
|
|
3
|
+
name: string;
|
|
4
|
+
up(queryRunner: QueryRunner): Promise<void>;
|
|
5
|
+
down(queryRunner: QueryRunner): Promise<void>;
|
|
6
|
+
}
|
|
7
|
+
//# sourceMappingURL=1693308311652-migration.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"1693308311652-migration.d.ts","sourceRoot":"","sources":["../../../lib/migrations/postgres/1693308311652-migration.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,kBAAkB,EAAE,WAAW,EAAE,MAAM,SAAS,CAAC;AAE1D,qBAAa,sBAAuB,YAAW,kBAAkB;IAC/D,IAAI,SAA4B;IAEnB,EAAE,CAAC,WAAW,EAAE,WAAW,GAAG,OAAO,CAAC,IAAI,CAAC;IAqB3C,IAAI,CAAC,WAAW,EAAE,WAAW,GAAG,OAAO,CAAC,IAAI,CAAC;CAK3D"}
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
export class migration1693308311652 {
|
|
2
|
+
name = 'migration1693308311652';
|
|
3
|
+
async up(queryRunner) {
|
|
4
|
+
await queryRunner.query(`
|
|
5
|
+
CREATE TABLE "fraud_entity" (
|
|
6
|
+
"id" SERIAL NOT NULL,
|
|
7
|
+
"boxId" character varying NOT NULL,
|
|
8
|
+
"creationBlock" character varying NOT NULL,
|
|
9
|
+
"creationHeight" integer NOT NULL,
|
|
10
|
+
"creationTxId" character varying NOT NULL,
|
|
11
|
+
"serialized" character varying NOT NULL,
|
|
12
|
+
"triggerBoxId" character varying NOT NULL,
|
|
13
|
+
"wid" character varying NOT NULL,
|
|
14
|
+
"rwtCount" character varying NOT NULL,
|
|
15
|
+
"spendBlock" text,
|
|
16
|
+
"spendHeight" integer,
|
|
17
|
+
"spendTxId" text,
|
|
18
|
+
"extractor" character varying NOT NULL,
|
|
19
|
+
CONSTRAINT "PK_2cf2cfa218dd4a5642bc3f3cf98" PRIMARY KEY ("id")
|
|
20
|
+
)
|
|
21
|
+
`);
|
|
22
|
+
}
|
|
23
|
+
async down(queryRunner) {
|
|
24
|
+
await queryRunner.query(`
|
|
25
|
+
DROP TABLE "fraud_entity"
|
|
26
|
+
`);
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiMTY5MzMwODMxMTY1Mi1taWdyYXRpb24uanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9saWIvbWlncmF0aW9ucy9wb3N0Z3Jlcy8xNjkzMzA4MzExNjUyLW1pZ3JhdGlvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFFQSxNQUFNLE9BQU8sc0JBQXNCO0lBQ2pDLElBQUksR0FBRyx3QkFBd0IsQ0FBQztJQUV6QixLQUFLLENBQUMsRUFBRSxDQUFDLFdBQXdCO1FBQ3RDLE1BQU0sV0FBVyxDQUFDLEtBQUssQ0FBQzs7Ozs7Ozs7Ozs7Ozs7Ozs7U0FpQm5CLENBQUMsQ0FBQztJQUNULENBQUM7SUFFTSxLQUFLLENBQUMsSUFBSSxDQUFDLFdBQXdCO1FBQ3hDLE1BQU0sV0FBVyxDQUFDLEtBQUssQ0FBQzs7U0FFbkIsQ0FBQyxDQUFDO0lBQ1QsQ0FBQztDQUNGIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgTWlncmF0aW9uSW50ZXJmYWNlLCBRdWVyeVJ1bm5lciB9IGZyb20gJ3R5cGVvcm0nO1xuXG5leHBvcnQgY2xhc3MgbWlncmF0aW9uMTY5MzMwODMxMTY1MiBpbXBsZW1lbnRzIE1pZ3JhdGlvbkludGVyZmFjZSB7XG4gIG5hbWUgPSAnbWlncmF0aW9uMTY5MzMwODMxMTY1Mic7XG5cbiAgcHVibGljIGFzeW5jIHVwKHF1ZXJ5UnVubmVyOiBRdWVyeVJ1bm5lcik6IFByb21pc2U8dm9pZD4ge1xuICAgIGF3YWl0IHF1ZXJ5UnVubmVyLnF1ZXJ5KGBcbiAgICAgICAgICAgIENSRUFURSBUQUJMRSBcImZyYXVkX2VudGl0eVwiIChcbiAgICAgICAgICAgICAgICBcImlkXCIgU0VSSUFMIE5PVCBOVUxMLFxuICAgICAgICAgICAgICAgIFwiYm94SWRcIiBjaGFyYWN0ZXIgdmFyeWluZyBOT1QgTlVMTCxcbiAgICAgICAgICAgICAgICBcImNyZWF0aW9uQmxvY2tcIiBjaGFyYWN0ZXIgdmFyeWluZyBOT1QgTlVMTCxcbiAgICAgICAgICAgICAgICBcImNyZWF0aW9uSGVpZ2h0XCIgaW50ZWdlciBOT1QgTlVMTCxcbiAgICAgICAgICAgICAgICBcImNyZWF0aW9uVHhJZFwiIGNoYXJhY3RlciB2YXJ5aW5nIE5PVCBOVUxMLFxuICAgICAgICAgICAgICAgIFwic2VyaWFsaXplZFwiIGNoYXJhY3RlciB2YXJ5aW5nIE5PVCBOVUxMLFxuICAgICAgICAgICAgICAgIFwidHJpZ2dlckJveElkXCIgY2hhcmFjdGVyIHZhcnlpbmcgTk9UIE5VTEwsXG4gICAgICAgICAgICAgICAgXCJ3aWRcIiBjaGFyYWN0ZXIgdmFyeWluZyBOT1QgTlVMTCxcbiAgICAgICAgICAgICAgICBcInJ3dENvdW50XCIgY2hhcmFjdGVyIHZhcnlpbmcgTk9UIE5VTEwsXG4gICAgICAgICAgICAgICAgXCJzcGVuZEJsb2NrXCIgdGV4dCxcbiAgICAgICAgICAgICAgICBcInNwZW5kSGVpZ2h0XCIgaW50ZWdlcixcbiAgICAgICAgICAgICAgICBcInNwZW5kVHhJZFwiIHRleHQsXG4gICAgICAgICAgICAgICAgXCJleHRyYWN0b3JcIiBjaGFyYWN0ZXIgdmFyeWluZyBOT1QgTlVMTCxcbiAgICAgICAgICAgICAgICBDT05TVFJBSU5UIFwiUEtfMmNmMmNmYTIxOGRkNGE1NjQyYmMzZjNjZjk4XCIgUFJJTUFSWSBLRVkgKFwiaWRcIilcbiAgICAgICAgICAgIClcbiAgICAgICAgYCk7XG4gIH1cblxuICBwdWJsaWMgYXN5bmMgZG93bihxdWVyeVJ1bm5lcjogUXVlcnlSdW5uZXIpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICBhd2FpdCBxdWVyeVJ1bm5lci5xdWVyeShgXG4gICAgICAgICAgICBEUk9QIFRBQkxFIFwiZnJhdWRfZW50aXR5XCJcbiAgICAgICAgYCk7XG4gIH1cbn1cbiJdfQ==
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
import { MigrationInterface, QueryRunner } from 'typeorm';
|
|
2
|
+
export declare class migration1693122371215 implements MigrationInterface {
|
|
3
|
+
name: string;
|
|
4
|
+
up(queryRunner: QueryRunner): Promise<void>;
|
|
5
|
+
down(queryRunner: QueryRunner): Promise<void>;
|
|
6
|
+
}
|
|
7
|
+
//# sourceMappingURL=1693122371215-migration.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"1693122371215-migration.d.ts","sourceRoot":"","sources":["../../../lib/migrations/sqlite/1693122371215-migration.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,kBAAkB,EAAE,WAAW,EAAE,MAAM,SAAS,CAAC;AAE1D,qBAAa,sBAAuB,YAAW,kBAAkB;IAC/D,IAAI,SAA4B;IAEnB,EAAE,CAAC,WAAW,EAAE,WAAW,GAAG,OAAO,CAAC,IAAI,CAAC;IAoB3C,IAAI,CAAC,WAAW,EAAE,WAAW,GAAG,OAAO,CAAC,IAAI,CAAC;CAK3D"}
|
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
export class migration1693122371215 {
|
|
2
|
+
name = 'migration1693122371215';
|
|
3
|
+
async up(queryRunner) {
|
|
4
|
+
await queryRunner.query(`
|
|
5
|
+
CREATE TABLE "fraud_entity" (
|
|
6
|
+
"id" integer PRIMARY KEY AUTOINCREMENT NOT NULL,
|
|
7
|
+
"boxId" varchar NOT NULL,
|
|
8
|
+
"creationBlock" varchar NOT NULL,
|
|
9
|
+
"creationHeight" integer NOT NULL,
|
|
10
|
+
"creationTxId" varchar NOT NULL,
|
|
11
|
+
"serialized" varchar NOT NULL,
|
|
12
|
+
"triggerBoxId" varchar NOT NULL,
|
|
13
|
+
"wid" varchar NOT NULL,
|
|
14
|
+
"rwtCount" varchar NOT NULL,
|
|
15
|
+
"spendBlock" text,
|
|
16
|
+
"spendHeight" integer,
|
|
17
|
+
"spendTxId" text,
|
|
18
|
+
"extractor" varchar NOT NULL
|
|
19
|
+
)
|
|
20
|
+
`);
|
|
21
|
+
}
|
|
22
|
+
async down(queryRunner) {
|
|
23
|
+
await queryRunner.query(`
|
|
24
|
+
DROP TABLE "fraud_entity"
|
|
25
|
+
`);
|
|
26
|
+
}
|
|
27
|
+
}
|
|
28
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiMTY5MzEyMjM3MTIxNS1taWdyYXRpb24uanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9saWIvbWlncmF0aW9ucy9zcWxpdGUvMTY5MzEyMjM3MTIxNS1taWdyYXRpb24udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBRUEsTUFBTSxPQUFPLHNCQUFzQjtJQUNqQyxJQUFJLEdBQUcsd0JBQXdCLENBQUM7SUFFekIsS0FBSyxDQUFDLEVBQUUsQ0FBQyxXQUF3QjtRQUN0QyxNQUFNLFdBQVcsQ0FBQyxLQUFLLENBQUM7Ozs7Ozs7Ozs7Ozs7Ozs7U0FnQm5CLENBQUMsQ0FBQztJQUNULENBQUM7SUFFTSxLQUFLLENBQUMsSUFBSSxDQUFDLFdBQXdCO1FBQ3hDLE1BQU0sV0FBVyxDQUFDLEtBQUssQ0FBQzs7U0FFbkIsQ0FBQyxDQUFDO0lBQ1QsQ0FBQztDQUNGIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgTWlncmF0aW9uSW50ZXJmYWNlLCBRdWVyeVJ1bm5lciB9IGZyb20gJ3R5cGVvcm0nO1xuXG5leHBvcnQgY2xhc3MgbWlncmF0aW9uMTY5MzEyMjM3MTIxNSBpbXBsZW1lbnRzIE1pZ3JhdGlvbkludGVyZmFjZSB7XG4gIG5hbWUgPSAnbWlncmF0aW9uMTY5MzEyMjM3MTIxNSc7XG5cbiAgcHVibGljIGFzeW5jIHVwKHF1ZXJ5UnVubmVyOiBRdWVyeVJ1bm5lcik6IFByb21pc2U8dm9pZD4ge1xuICAgIGF3YWl0IHF1ZXJ5UnVubmVyLnF1ZXJ5KGBcbiAgICAgICAgICAgIENSRUFURSBUQUJMRSBcImZyYXVkX2VudGl0eVwiIChcbiAgICAgICAgICAgICAgICBcImlkXCIgaW50ZWdlciBQUklNQVJZIEtFWSBBVVRPSU5DUkVNRU5UIE5PVCBOVUxMLFxuICAgICAgICAgICAgICAgIFwiYm94SWRcIiB2YXJjaGFyIE5PVCBOVUxMLFxuICAgICAgICAgICAgICAgIFwiY3JlYXRpb25CbG9ja1wiIHZhcmNoYXIgTk9UIE5VTEwsXG4gICAgICAgICAgICAgICAgXCJjcmVhdGlvbkhlaWdodFwiIGludGVnZXIgTk9UIE5VTEwsXG4gICAgICAgICAgICAgICAgXCJjcmVhdGlvblR4SWRcIiB2YXJjaGFyIE5PVCBOVUxMLFxuICAgICAgICAgICAgICAgIFwic2VyaWFsaXplZFwiIHZhcmNoYXIgTk9UIE5VTEwsXG4gICAgICAgICAgICAgICAgXCJ0cmlnZ2VyQm94SWRcIiB2YXJjaGFyIE5PVCBOVUxMLFxuICAgICAgICAgICAgICAgIFwid2lkXCIgdmFyY2hhciBOT1QgTlVMTCxcbiAgICAgICAgICAgICAgICBcInJ3dENvdW50XCIgdmFyY2hhciBOT1QgTlVMTCxcbiAgICAgICAgICAgICAgICBcInNwZW5kQmxvY2tcIiB0ZXh0LFxuICAgICAgICAgICAgICAgIFwic3BlbmRIZWlnaHRcIiBpbnRlZ2VyLFxuICAgICAgICAgICAgICAgIFwic3BlbmRUeElkXCIgdGV4dCxcbiAgICAgICAgICAgICAgICBcImV4dHJhY3RvclwiIHZhcmNoYXIgTk9UIE5VTExcbiAgICAgICAgICAgIClcbiAgICAgICAgYCk7XG4gIH1cblxuICBwdWJsaWMgYXN5bmMgZG93bihxdWVyeVJ1bm5lcjogUXVlcnlSdW5uZXIpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICBhd2FpdCBxdWVyeVJ1bm5lci5xdWVyeShgXG4gICAgICAgICAgICBEUk9QIFRBQkxFIFwiZnJhdWRfZW50aXR5XCJcbiAgICAgICAgYCk7XG4gIH1cbn1cbiJdfQ==
|
package/package.json
ADDED
|
@@ -0,0 +1,52 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "@rosen-bridge/fraud-extractor",
|
|
3
|
+
"version": "0.1.1",
|
|
4
|
+
"description": "fraud information extractor for watcher faulty reports",
|
|
5
|
+
"author": "Rosen Team",
|
|
6
|
+
"license": "GPL-3.0",
|
|
7
|
+
"main": "dist/index.js",
|
|
8
|
+
"types": "dist/index.d.ts",
|
|
9
|
+
"files": [
|
|
10
|
+
"dist"
|
|
11
|
+
],
|
|
12
|
+
"scripts": {
|
|
13
|
+
"prettify": "prettier --write . --ignore-path ../../.gitignore",
|
|
14
|
+
"lint": "eslint --fix . && npm run prettify",
|
|
15
|
+
"test": "jest --detectOpenHandles",
|
|
16
|
+
"coverage": "jest --detectOpenHandles --coverage",
|
|
17
|
+
"build": "tsc --build",
|
|
18
|
+
"type-check": "tsc --noEmit",
|
|
19
|
+
"release": "npm run test && npm run build && npm publish --access public"
|
|
20
|
+
},
|
|
21
|
+
"dependencies": {
|
|
22
|
+
"@rosen-bridge/abstract-logger": "^0.3.0",
|
|
23
|
+
"@rosen-bridge/json-bigint": "^0.1.0",
|
|
24
|
+
"@rosen-bridge/scanner": "^2.2.1",
|
|
25
|
+
"blakejs": "^1.2.1",
|
|
26
|
+
"ergo-lib-wasm-nodejs": "^0.24.1",
|
|
27
|
+
"lodash-es": "^4.17.21",
|
|
28
|
+
"reflect-metadata": "^0.1.13",
|
|
29
|
+
"sqlite3": "^5.0.9",
|
|
30
|
+
"typeorm": "^0.3.7"
|
|
31
|
+
},
|
|
32
|
+
"type": "module",
|
|
33
|
+
"devDependencies": {
|
|
34
|
+
"@babel/preset-env": "^7.19.0",
|
|
35
|
+
"@types/jest": "^28.1.6",
|
|
36
|
+
"@types/json-bigint": "^1.0.1",
|
|
37
|
+
"@types/node": "^18.0.6",
|
|
38
|
+
"@typescript-eslint/eslint-plugin": "^5.30.7",
|
|
39
|
+
"@typescript-eslint/parser": "^5.26.0",
|
|
40
|
+
"eslint": "^8.16.0",
|
|
41
|
+
"eslint-config-prettier": "^8.5.0",
|
|
42
|
+
"jest": "^28.1.3",
|
|
43
|
+
"json-bigint": "^1.0.0",
|
|
44
|
+
"prettier": "2.7.1",
|
|
45
|
+
"ts-jest": "^28.0.7",
|
|
46
|
+
"ts-node": "^10.9.1",
|
|
47
|
+
"typescript": "^4.7.4"
|
|
48
|
+
},
|
|
49
|
+
"engines": {
|
|
50
|
+
"node": ">=16.14.0"
|
|
51
|
+
}
|
|
52
|
+
}
|