@rosen-bridge/abstract-scanner 0.1.0-52fc0239
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +49 -0
- package/dist/entities/blockEntity.d.ts +17 -0
- package/dist/entities/blockEntity.d.ts.map +1 -0
- package/dist/entities/blockEntity.js +78 -0
- package/dist/entities/extractorStatusEntity.d.ts +7 -0
- package/dist/entities/extractorStatusEntity.d.ts.map +1 -0
- package/dist/entities/extractorStatusEntity.js +37 -0
- package/dist/index.d.ts +11 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +10 -0
- package/dist/migrations/index.d.ts +7 -0
- package/dist/migrations/index.d.ts.map +1 -0
- package/dist/migrations/index.js +31 -0
- package/dist/migrations/postgres/1688545690867-migration.d.ts +7 -0
- package/dist/migrations/postgres/1688545690867-migration.d.ts.map +1 -0
- package/dist/migrations/postgres/1688545690867-migration.js +30 -0
- package/dist/migrations/postgres/1713803486477-migration.d.ts +7 -0
- package/dist/migrations/postgres/1713803486477-migration.d.ts.map +1 -0
- package/dist/migrations/postgres/1713803486477-migration.js +20 -0
- package/dist/migrations/postgres/1718789786123-migration.d.ts +7 -0
- package/dist/migrations/postgres/1718789786123-migration.d.ts.map +1 -0
- package/dist/migrations/postgres/1718789786123-migration.js +12 -0
- package/dist/migrations/postgres/1722697954558-migration.d.ts +7 -0
- package/dist/migrations/postgres/1722697954558-migration.d.ts.map +1 -0
- package/dist/migrations/postgres/1722697954558-migration.js +52 -0
- package/dist/migrations/postgres/1746701087567-migration.d.ts +7 -0
- package/dist/migrations/postgres/1746701087567-migration.d.ts.map +1 -0
- package/dist/migrations/postgres/1746701087567-migration.js +32 -0
- package/dist/migrations/postgres/1747657653564-migration.d.ts +7 -0
- package/dist/migrations/postgres/1747657653564-migration.d.ts.map +1 -0
- package/dist/migrations/postgres/1747657653564-migration.js +134 -0
- package/dist/migrations/sqlite/1688555497475-migration.d.ts +7 -0
- package/dist/migrations/sqlite/1688555497475-migration.d.ts.map +1 -0
- package/dist/migrations/sqlite/1688555497475-migration.js +29 -0
- package/dist/migrations/sqlite/1713786682123-migration.d.ts +7 -0
- package/dist/migrations/sqlite/1713786682123-migration.d.ts.map +1 -0
- package/dist/migrations/sqlite/1713786682123-migration.js +20 -0
- package/dist/migrations/sqlite/1718789744123-migration.d.ts +7 -0
- package/dist/migrations/sqlite/1718789744123-migration.d.ts.map +1 -0
- package/dist/migrations/sqlite/1718789744123-migration.js +12 -0
- package/dist/migrations/sqlite/1722697111974-migration.d.ts +7 -0
- package/dist/migrations/sqlite/1722697111974-migration.d.ts.map +1 -0
- package/dist/migrations/sqlite/1722697111974-migration.js +112 -0
- package/dist/migrations/sqlite/1746701087234-migration.d.ts +7 -0
- package/dist/migrations/sqlite/1746701087234-migration.d.ts.map +1 -0
- package/dist/migrations/sqlite/1746701087234-migration.js +32 -0
- package/dist/migrations/sqlite/1747655941239-migration.d.ts +7 -0
- package/dist/migrations/sqlite/1747655941239-migration.d.ts.map +1 -0
- package/dist/migrations/sqlite/1747655941239-migration.js +134 -0
- package/dist/scanner/abstract/generalScanner.d.ts +63 -0
- package/dist/scanner/abstract/generalScanner.d.ts.map +1 -0
- package/dist/scanner/abstract/generalScanner.js +173 -0
- package/dist/scanner/abstract/scanner.d.ts +48 -0
- package/dist/scanner/abstract/scanner.d.ts.map +1 -0
- package/dist/scanner/abstract/scanner.js +154 -0
- package/dist/scanner/abstract/webSocketScanner.d.ts +29 -0
- package/dist/scanner/abstract/webSocketScanner.d.ts.map +1 -0
- package/dist/scanner/abstract/webSocketScanner.js +89 -0
- package/dist/scanner/action.d.ts +83 -0
- package/dist/scanner/action.d.ts.map +1 -0
- package/dist/scanner/action.js +251 -0
- package/dist/scanner/interfaces.d.ts +12 -0
- package/dist/scanner/interfaces.d.ts.map +1 -0
- package/dist/scanner/interfaces.js +2 -0
- package/dist/scanner/network/ConnectorSelectionStrategies.d.ts +27 -0
- package/dist/scanner/network/ConnectorSelectionStrategies.d.ts.map +1 -0
- package/dist/scanner/network/ConnectorSelectionStrategies.js +22 -0
- package/dist/scanner/network/NetworkConnectorManager.d.ts +64 -0
- package/dist/scanner/network/NetworkConnectorManager.d.ts.map +1 -0
- package/dist/scanner/network/NetworkConnectorManager.js +125 -0
- package/package.json +54 -0
|
@@ -0,0 +1,83 @@
|
|
|
1
|
+
import { BlockEntity } from '../entities/blockEntity';
|
|
2
|
+
import { DataSource, DeleteResult, Repository } from '@rosen-bridge/extended-typeorm';
|
|
3
|
+
import { Block } from '@rosen-bridge/scanner-interfaces';
|
|
4
|
+
import { AbstractLogger } from '@rosen-bridge/abstract-logger';
|
|
5
|
+
import { ExtractorStatusEntity } from '../entities/extractorStatusEntity';
|
|
6
|
+
export declare class BlockDbAction {
|
|
7
|
+
readonly blockRepository: Repository<BlockEntity>;
|
|
8
|
+
readonly extractorStatusRepository: Repository<ExtractorStatusEntity>;
|
|
9
|
+
readonly dataSource: DataSource;
|
|
10
|
+
readonly scannerName: string;
|
|
11
|
+
readonly logger: AbstractLogger;
|
|
12
|
+
constructor(dataSource: DataSource, scannerName: string, logger?: AbstractLogger);
|
|
13
|
+
readonly name: () => string;
|
|
14
|
+
/**
|
|
15
|
+
* get last saved block
|
|
16
|
+
* @return Promise<BlockEntity or undefined>
|
|
17
|
+
*/
|
|
18
|
+
getLastSavedBlock: () => Promise<BlockEntity | undefined>;
|
|
19
|
+
/**
|
|
20
|
+
* get last n stored blocks in database
|
|
21
|
+
* @param skip
|
|
22
|
+
* @param count
|
|
23
|
+
*/
|
|
24
|
+
getLastSavedBlocks: (skip: number, count: number) => Promise<Array<BlockEntity>>;
|
|
25
|
+
/**
|
|
26
|
+
* get first saved block
|
|
27
|
+
* @return Promise<BlockEntity or undefined>
|
|
28
|
+
*/
|
|
29
|
+
getFirstSavedBlock: () => Promise<BlockEntity | undefined>;
|
|
30
|
+
/**
|
|
31
|
+
* get block hash and height
|
|
32
|
+
* @param height
|
|
33
|
+
* @param status
|
|
34
|
+
* @return Promise<BlockEntity|undefined>
|
|
35
|
+
*/
|
|
36
|
+
getBlockAtHeight: (height: number, status?: string) => Promise<BlockEntity | undefined>;
|
|
37
|
+
/**
|
|
38
|
+
* get block with entered blockhash
|
|
39
|
+
* @param hash
|
|
40
|
+
* @param status
|
|
41
|
+
*/
|
|
42
|
+
getBlockWithHash: (hash: string, status?: string) => Promise<BlockEntity | null>;
|
|
43
|
+
/**
|
|
44
|
+
* it deletes every block that more than or equal height
|
|
45
|
+
* @param height
|
|
46
|
+
* @return Promise<DeleteResult>
|
|
47
|
+
*/
|
|
48
|
+
removeBlocksFromHeight: (height: number) => Promise<DeleteResult>;
|
|
49
|
+
/**
|
|
50
|
+
* store a block into database.
|
|
51
|
+
* @param block
|
|
52
|
+
*/
|
|
53
|
+
saveBlock: (block: Block) => Promise<BlockEntity | boolean>;
|
|
54
|
+
/**
|
|
55
|
+
* Update status of a block to proceed
|
|
56
|
+
* @param height: height of expected block
|
|
57
|
+
*/
|
|
58
|
+
updateBlockStatus: (height: number, blockHash: string, extractorIds: string[]) => Promise<boolean>;
|
|
59
|
+
/**
|
|
60
|
+
* Update status of a block to processing in case of fork
|
|
61
|
+
* @param height: height of expected block
|
|
62
|
+
*/
|
|
63
|
+
revertBlockStatus: (height: number, parentHash: string, extractorIds: string[]) => Promise<boolean>;
|
|
64
|
+
/**
|
|
65
|
+
* Insert a newly initialized extractor or update an existing extractor status
|
|
66
|
+
* @param extractorId
|
|
67
|
+
* @param height
|
|
68
|
+
* @returns
|
|
69
|
+
*/
|
|
70
|
+
updateOrInsertExtractorStatus: (extractorId: string, height: number, blockHash: string) => Promise<{
|
|
71
|
+
scannerId: string;
|
|
72
|
+
extractorId: string;
|
|
73
|
+
updateHeight: number;
|
|
74
|
+
updateBlockHash: string;
|
|
75
|
+
} & ExtractorStatusEntity>;
|
|
76
|
+
/**
|
|
77
|
+
* Return extractors` status specified by id
|
|
78
|
+
* @param extractorIds
|
|
79
|
+
* @returns extractors` status
|
|
80
|
+
*/
|
|
81
|
+
getExtractorsStatus: (extractorIds: string[]) => Promise<ExtractorStatusEntity[]>;
|
|
82
|
+
}
|
|
83
|
+
//# sourceMappingURL=action.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"action.d.ts","sourceRoot":"","sources":["../../lib/scanner/action.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,WAAW,EAAuB,MAAM,yBAAyB,CAAC;AAC3E,OAAO,EACL,UAAU,EACV,YAAY,EAEZ,UAAU,EAEX,MAAM,gCAAgC,CAAC;AACxC,OAAO,EAAE,KAAK,EAAE,MAAM,kCAAkC,CAAC;AACzD,OAAO,EAAE,cAAc,EAAe,MAAM,+BAA+B,CAAC;AAC5E,OAAO,EAAE,qBAAqB,EAAE,MAAM,mCAAmC,CAAC;AAE1E,qBAAa,aAAa;IACxB,QAAQ,CAAC,eAAe,EAAE,UAAU,CAAC,WAAW,CAAC,CAAC;IAClD,QAAQ,CAAC,yBAAyB,EAAE,UAAU,CAAC,qBAAqB,CAAC,CAAC;IACtE,QAAQ,CAAC,UAAU,EAAE,UAAU,CAAC;IAChC,QAAQ,CAAC,WAAW,EAAE,MAAM,CAAC;IAC7B,QAAQ,CAAC,MAAM,EAAE,cAAc,CAAC;gBAG9B,UAAU,EAAE,UAAU,EACtB,WAAW,EAAE,MAAM,EACnB,MAAM,CAAC,EAAE,cAAc;IAWzB,QAAQ,CAAC,IAAI,eAA0B;IAEvC;;;OAGG;IACH,iBAAiB,QAAa,QAAQ,WAAW,GAAG,SAAS,CAAC,CAW5D;IAEF;;;;OAIG;IACH,kBAAkB,SACV,MAAM,SACL,MAAM,KACZ,QAAQ,MAAM,WAAW,CAAC,CAAC,CAO5B;IAEF;;;OAGG;IACH,kBAAkB,QAAa,QAAQ,WAAW,GAAG,SAAS,CAAC,CAW7D;IAEF;;;;;OAKG;IACH,gBAAgB,WACN,MAAM,WACN,MAAM,KACb,QAAQ,WAAW,GAAG,SAAS,CAAC,CAWjC;IAEF;;;;OAIG;IACH,gBAAgB,SAAgB,MAAM,WAAU,MAAM,iCAMpD;IAEF;;;;OAIG;IACH,sBAAsB,WAAkB,MAAM,KAAG,QAAQ,YAAY,CAAC,CAMpE;IAEF;;;OAGG;IACH,SAAS,UAAiB,KAAK,KAAG,QAAQ,WAAW,GAAG,OAAO,CAAC,CA0C9D;IAEF;;;OAGG;IACH,iBAAiB,WACP,MAAM,aACH,MAAM,gBACH,MAAM,EAAE,KACrB,QAAQ,OAAO,CAAC,CAqCjB;IAEF;;;OAGG;IACH,iBAAiB,WACP,MAAM,cACF,MAAM,gBACJ,MAAM,EAAE,KACrB,QAAQ,OAAO,CAAC,CAsCjB;IAEF;;;;;OAKG;IACH,6BAA6B,gBACd,MAAM,UACX,MAAM,aACH,MAAM;;;;;+BAgBjB;IAEF;;;;OAIG;IACH,mBAAmB,iBACH,MAAM,EAAE,KACrB,QAAQ,qBAAqB,EAAE,CAAC,CAKjC;CACH"}
|
|
@@ -0,0 +1,251 @@
|
|
|
1
|
+
import { BlockEntity, PROCEED, PROCESSING } from '../entities/blockEntity';
|
|
2
|
+
import { MoreThanOrEqual, In, } from '@rosen-bridge/extended-typeorm';
|
|
3
|
+
import { DummyLogger } from '@rosen-bridge/abstract-logger';
|
|
4
|
+
import { ExtractorStatusEntity } from '../entities/extractorStatusEntity';
|
|
5
|
+
export class BlockDbAction {
|
|
6
|
+
blockRepository;
|
|
7
|
+
extractorStatusRepository;
|
|
8
|
+
dataSource;
|
|
9
|
+
scannerName;
|
|
10
|
+
logger;
|
|
11
|
+
constructor(dataSource, scannerName, logger) {
|
|
12
|
+
this.blockRepository = dataSource.getRepository(BlockEntity);
|
|
13
|
+
this.extractorStatusRepository = dataSource.getRepository(ExtractorStatusEntity);
|
|
14
|
+
this.dataSource = dataSource;
|
|
15
|
+
this.scannerName = scannerName;
|
|
16
|
+
this.logger = logger ? logger : new DummyLogger();
|
|
17
|
+
}
|
|
18
|
+
name = () => this.scannerName;
|
|
19
|
+
/**
|
|
20
|
+
* get last saved block
|
|
21
|
+
* @return Promise<BlockEntity or undefined>
|
|
22
|
+
*/
|
|
23
|
+
getLastSavedBlock = async () => {
|
|
24
|
+
const lastBlock = await this.blockRepository.find({
|
|
25
|
+
where: { status: PROCEED, scanner: this.name() },
|
|
26
|
+
order: { height: 'DESC' },
|
|
27
|
+
take: 1,
|
|
28
|
+
});
|
|
29
|
+
if (lastBlock.length !== 0) {
|
|
30
|
+
return lastBlock[0];
|
|
31
|
+
}
|
|
32
|
+
else {
|
|
33
|
+
return undefined;
|
|
34
|
+
}
|
|
35
|
+
};
|
|
36
|
+
/**
|
|
37
|
+
* get last n stored blocks in database
|
|
38
|
+
* @param skip
|
|
39
|
+
* @param count
|
|
40
|
+
*/
|
|
41
|
+
getLastSavedBlocks = async (skip, count) => {
|
|
42
|
+
return await this.blockRepository.find({
|
|
43
|
+
where: { status: PROCEED, scanner: this.name() },
|
|
44
|
+
order: { height: 'DESC' },
|
|
45
|
+
skip: skip,
|
|
46
|
+
take: count,
|
|
47
|
+
});
|
|
48
|
+
};
|
|
49
|
+
/**
|
|
50
|
+
* get first saved block
|
|
51
|
+
* @return Promise<BlockEntity or undefined>
|
|
52
|
+
*/
|
|
53
|
+
getFirstSavedBlock = async () => {
|
|
54
|
+
const lastBlock = await this.blockRepository.find({
|
|
55
|
+
where: { status: PROCEED, scanner: this.name() },
|
|
56
|
+
order: { height: 'ASC' },
|
|
57
|
+
take: 1,
|
|
58
|
+
});
|
|
59
|
+
if (lastBlock.length !== 0) {
|
|
60
|
+
return lastBlock[0];
|
|
61
|
+
}
|
|
62
|
+
else {
|
|
63
|
+
return undefined;
|
|
64
|
+
}
|
|
65
|
+
};
|
|
66
|
+
/**
|
|
67
|
+
* get block hash and height
|
|
68
|
+
* @param height
|
|
69
|
+
* @param status
|
|
70
|
+
* @return Promise<BlockEntity|undefined>
|
|
71
|
+
*/
|
|
72
|
+
getBlockAtHeight = async (height, status = PROCEED) => {
|
|
73
|
+
const block = await this.blockRepository.findOneBy({
|
|
74
|
+
status: status,
|
|
75
|
+
height: height,
|
|
76
|
+
scanner: this.name(),
|
|
77
|
+
});
|
|
78
|
+
if (block !== null) {
|
|
79
|
+
return block;
|
|
80
|
+
}
|
|
81
|
+
else {
|
|
82
|
+
return undefined;
|
|
83
|
+
}
|
|
84
|
+
};
|
|
85
|
+
/**
|
|
86
|
+
* get block with entered blockhash
|
|
87
|
+
* @param hash
|
|
88
|
+
* @param status
|
|
89
|
+
*/
|
|
90
|
+
getBlockWithHash = async (hash, status = PROCEED) => {
|
|
91
|
+
return this.blockRepository.findOneBy({
|
|
92
|
+
hash: hash,
|
|
93
|
+
status: status,
|
|
94
|
+
scanner: this.name(),
|
|
95
|
+
});
|
|
96
|
+
};
|
|
97
|
+
/**
|
|
98
|
+
* it deletes every block that more than or equal height
|
|
99
|
+
* @param height
|
|
100
|
+
* @return Promise<DeleteResult>
|
|
101
|
+
*/
|
|
102
|
+
removeBlocksFromHeight = async (height) => {
|
|
103
|
+
this.logger.debug(`Removing blocks from height ${height}`);
|
|
104
|
+
return await this.blockRepository.delete({
|
|
105
|
+
height: MoreThanOrEqual(height),
|
|
106
|
+
scanner: this.name(),
|
|
107
|
+
});
|
|
108
|
+
};
|
|
109
|
+
/**
|
|
110
|
+
* store a block into database.
|
|
111
|
+
* @param block
|
|
112
|
+
*/
|
|
113
|
+
saveBlock = async (block) => {
|
|
114
|
+
try {
|
|
115
|
+
const instance = await this.blockRepository.findOneBy({
|
|
116
|
+
height: block.height,
|
|
117
|
+
scanner: this.name(),
|
|
118
|
+
});
|
|
119
|
+
const date = new Date(block.timestamp * 1000);
|
|
120
|
+
const blockInfo = {
|
|
121
|
+
height: block.height,
|
|
122
|
+
hash: block.hash,
|
|
123
|
+
parentHash: block.parentHash,
|
|
124
|
+
status: PROCESSING,
|
|
125
|
+
scanner: this.name(),
|
|
126
|
+
extra: block.extra,
|
|
127
|
+
timestamp: block.timestamp,
|
|
128
|
+
year: date.getFullYear(),
|
|
129
|
+
month: date.getMonth() + 1,
|
|
130
|
+
day: date.getDate(),
|
|
131
|
+
};
|
|
132
|
+
if (!instance) {
|
|
133
|
+
this.logger.debug(`Inserting block with info: ${JSON.stringify(blockInfo)}`);
|
|
134
|
+
await this.blockRepository.insert(blockInfo);
|
|
135
|
+
}
|
|
136
|
+
else {
|
|
137
|
+
this.logger.debug(`Updating block with info: ${JSON.stringify(blockInfo)}`);
|
|
138
|
+
await this.blockRepository.update({ height: block.height, scanner: this.name() }, blockInfo);
|
|
139
|
+
}
|
|
140
|
+
const res = await this.blockRepository.findOneBy({
|
|
141
|
+
height: block.height,
|
|
142
|
+
scanner: this.name(),
|
|
143
|
+
});
|
|
144
|
+
return res ? res : false;
|
|
145
|
+
}
|
|
146
|
+
catch (exp) {
|
|
147
|
+
this.logger.error(`An error occurred during save new block: ${exp}`);
|
|
148
|
+
return false;
|
|
149
|
+
}
|
|
150
|
+
};
|
|
151
|
+
/**
|
|
152
|
+
* Update status of a block to proceed
|
|
153
|
+
* @param height: height of expected block
|
|
154
|
+
*/
|
|
155
|
+
updateBlockStatus = async (height, blockHash, extractorIds) => {
|
|
156
|
+
let success = true;
|
|
157
|
+
this.logger.debug(`Block at height ${height} has been proceed in scanner ${this.name()}, updating status`);
|
|
158
|
+
const runner = this.dataSource.createQueryRunner();
|
|
159
|
+
await runner.connect();
|
|
160
|
+
await runner.startTransaction();
|
|
161
|
+
try {
|
|
162
|
+
await runner.manager.getRepository(BlockEntity).update({
|
|
163
|
+
height: height,
|
|
164
|
+
status: PROCESSING,
|
|
165
|
+
scanner: this.name(),
|
|
166
|
+
}, {
|
|
167
|
+
status: PROCEED,
|
|
168
|
+
});
|
|
169
|
+
this.logger.debug('Updating extractors status', {
|
|
170
|
+
extractorIds,
|
|
171
|
+
});
|
|
172
|
+
await runner.manager
|
|
173
|
+
.getRepository(ExtractorStatusEntity)
|
|
174
|
+
.update({ extractorId: In(extractorIds), scannerId: this.name() }, { updateHeight: height, updateBlockHash: blockHash });
|
|
175
|
+
await runner.commitTransaction();
|
|
176
|
+
}
|
|
177
|
+
catch (e) {
|
|
178
|
+
await runner.rollbackTransaction();
|
|
179
|
+
success = false;
|
|
180
|
+
}
|
|
181
|
+
finally {
|
|
182
|
+
await runner.release();
|
|
183
|
+
}
|
|
184
|
+
return success;
|
|
185
|
+
};
|
|
186
|
+
/**
|
|
187
|
+
* Update status of a block to processing in case of fork
|
|
188
|
+
* @param height: height of expected block
|
|
189
|
+
*/
|
|
190
|
+
revertBlockStatus = async (height, parentHash, extractorIds) => {
|
|
191
|
+
let success = true;
|
|
192
|
+
const runner = this.dataSource.createQueryRunner();
|
|
193
|
+
await runner.connect();
|
|
194
|
+
await runner.startTransaction();
|
|
195
|
+
try {
|
|
196
|
+
this.logger.debug(`Reverting block status at height ${height}`);
|
|
197
|
+
await runner.manager.getRepository(BlockEntity).update({
|
|
198
|
+
height: height,
|
|
199
|
+
status: PROCEED,
|
|
200
|
+
scanner: this.name(),
|
|
201
|
+
}, {
|
|
202
|
+
status: PROCESSING,
|
|
203
|
+
});
|
|
204
|
+
this.logger.debug(`Reverting extractors update height to previous block at height ${height - 1}`);
|
|
205
|
+
await runner.manager
|
|
206
|
+
.getRepository(ExtractorStatusEntity)
|
|
207
|
+
.update({ extractorId: In(extractorIds), scannerId: this.name() }, { updateHeight: height - 1, updateBlockHash: parentHash });
|
|
208
|
+
await runner.commitTransaction();
|
|
209
|
+
}
|
|
210
|
+
catch (e) {
|
|
211
|
+
this.logger.warn(`An Error occurred while reverting block status: ${e}`);
|
|
212
|
+
await runner.rollbackTransaction();
|
|
213
|
+
success = false;
|
|
214
|
+
}
|
|
215
|
+
finally {
|
|
216
|
+
await runner.release();
|
|
217
|
+
}
|
|
218
|
+
return success;
|
|
219
|
+
};
|
|
220
|
+
/**
|
|
221
|
+
* Insert a newly initialized extractor or update an existing extractor status
|
|
222
|
+
* @param extractorId
|
|
223
|
+
* @param height
|
|
224
|
+
* @returns
|
|
225
|
+
*/
|
|
226
|
+
updateOrInsertExtractorStatus = async (extractorId, height, blockHash) => {
|
|
227
|
+
this.logger.debug('Inserting new extractor status or Updating existing resynced extractor status', {
|
|
228
|
+
extractorId,
|
|
229
|
+
height,
|
|
230
|
+
blockHash,
|
|
231
|
+
});
|
|
232
|
+
return await this.extractorStatusRepository.save({
|
|
233
|
+
scannerId: this.name(),
|
|
234
|
+
extractorId,
|
|
235
|
+
updateHeight: height,
|
|
236
|
+
updateBlockHash: blockHash,
|
|
237
|
+
});
|
|
238
|
+
};
|
|
239
|
+
/**
|
|
240
|
+
* Return extractors` status specified by id
|
|
241
|
+
* @param extractorIds
|
|
242
|
+
* @returns extractors` status
|
|
243
|
+
*/
|
|
244
|
+
getExtractorsStatus = async (extractorIds) => {
|
|
245
|
+
return await this.extractorStatusRepository.findBy({
|
|
246
|
+
scannerId: this.name(),
|
|
247
|
+
extractorId: In(extractorIds),
|
|
248
|
+
});
|
|
249
|
+
};
|
|
250
|
+
}
|
|
251
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"action.js","sourceRoot":"","sources":["../../lib/scanner/action.ts"],"names":[],"mappings":"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","sourcesContent":["import { BlockEntity, PROCEED, PROCESSING } from '../entities/blockEntity';\nimport {\n  DataSource,\n  DeleteResult,\n  MoreThanOrEqual,\n  Repository,\n  In,\n} from '@rosen-bridge/extended-typeorm';\nimport { Block } from '@rosen-bridge/scanner-interfaces';\nimport { AbstractLogger, DummyLogger } from '@rosen-bridge/abstract-logger';\nimport { ExtractorStatusEntity } from '../entities/extractorStatusEntity';\n\nexport class BlockDbAction {\n  readonly blockRepository: Repository<BlockEntity>;\n  readonly extractorStatusRepository: Repository<ExtractorStatusEntity>;\n  readonly dataSource: DataSource;\n  readonly scannerName: string;\n  readonly logger: AbstractLogger;\n\n  constructor(\n    dataSource: DataSource,\n    scannerName: string,\n    logger?: AbstractLogger\n  ) {\n    this.blockRepository = dataSource.getRepository(BlockEntity);\n    this.extractorStatusRepository = dataSource.getRepository(\n      ExtractorStatusEntity\n    );\n    this.dataSource = dataSource;\n    this.scannerName = scannerName;\n    this.logger = logger ? logger : new DummyLogger();\n  }\n\n  readonly name = () => this.scannerName;\n\n  /**\n   * get last saved block\n   * @return Promise<BlockEntity or undefined>\n   */\n  getLastSavedBlock = async (): Promise<BlockEntity | undefined> => {\n    const lastBlock = await this.blockRepository.find({\n      where: { status: PROCEED, scanner: this.name() },\n      order: { height: 'DESC' },\n      take: 1,\n    });\n    if (lastBlock.length !== 0) {\n      return lastBlock[0];\n    } else {\n      return undefined;\n    }\n  };\n\n  /**\n   * get last n stored blocks in database\n   * @param skip\n   * @param count\n   */\n  getLastSavedBlocks = async (\n    skip: number,\n    count: number\n  ): Promise<Array<BlockEntity>> => {\n    return await this.blockRepository.find({\n      where: { status: PROCEED, scanner: this.name() },\n      order: { height: 'DESC' },\n      skip: skip,\n      take: count,\n    });\n  };\n\n  /**\n   * get first saved block\n   * @return Promise<BlockEntity or undefined>\n   */\n  getFirstSavedBlock = async (): Promise<BlockEntity | undefined> => {\n    const lastBlock = await this.blockRepository.find({\n      where: { status: PROCEED, scanner: this.name() },\n      order: { height: 'ASC' },\n      take: 1,\n    });\n    if (lastBlock.length !== 0) {\n      return lastBlock[0];\n    } else {\n      return undefined;\n    }\n  };\n\n  /**\n   * get block hash and height\n   * @param height\n   * @param status\n   * @return Promise<BlockEntity|undefined>\n   */\n  getBlockAtHeight = async (\n    height: number,\n    status: string = PROCEED\n  ): Promise<BlockEntity | undefined> => {\n    const block = await this.blockRepository.findOneBy({\n      status: status,\n      height: height,\n      scanner: this.name(),\n    });\n    if (block !== null) {\n      return block;\n    } else {\n      return undefined;\n    }\n  };\n\n  /**\n   * get block with entered blockhash\n   * @param hash\n   * @param status\n   */\n  getBlockWithHash = async (hash: string, status: string = PROCEED) => {\n    return this.blockRepository.findOneBy({\n      hash: hash,\n      status: status,\n      scanner: this.name(),\n    });\n  };\n\n  /**\n   * it deletes every block that more than or equal height\n   * @param height\n   * @return Promise<DeleteResult>\n   */\n  removeBlocksFromHeight = async (height: number): Promise<DeleteResult> => {\n    this.logger.debug(`Removing blocks from height ${height}`);\n    return await this.blockRepository.delete({\n      height: MoreThanOrEqual(height),\n      scanner: this.name(),\n    });\n  };\n\n  /**\n   * store a block into database.\n   * @param block\n   */\n  saveBlock = async (block: Block): Promise<BlockEntity | boolean> => {\n    try {\n      const instance = await this.blockRepository.findOneBy({\n        height: block.height,\n        scanner: this.name(),\n      });\n      const date = new Date(block.timestamp * 1000);\n      const blockInfo = {\n        height: block.height,\n        hash: block.hash,\n        parentHash: block.parentHash,\n        status: PROCESSING,\n        scanner: this.name(),\n        extra: block.extra,\n        timestamp: block.timestamp,\n        year: date.getFullYear(),\n        month: date.getMonth() + 1,\n        day: date.getDate(),\n      };\n      if (!instance) {\n        this.logger.debug(\n          `Inserting block with info: ${JSON.stringify(blockInfo)}`\n        );\n        await this.blockRepository.insert(blockInfo);\n      } else {\n        this.logger.debug(\n          `Updating block with info: ${JSON.stringify(blockInfo)}`\n        );\n        await this.blockRepository.update(\n          { height: block.height, scanner: this.name() },\n          blockInfo\n        );\n      }\n      const res = await this.blockRepository.findOneBy({\n        height: block.height,\n        scanner: this.name(),\n      });\n      return res ? res : false;\n    } catch (exp) {\n      this.logger.error(`An error occurred during save new block: ${exp}`);\n      return false;\n    }\n  };\n\n  /**\n   * Update status of a block to proceed\n   * @param height: height of expected block\n   */\n  updateBlockStatus = async (\n    height: number,\n    blockHash: string,\n    extractorIds: string[]\n  ): Promise<boolean> => {\n    let success = true;\n    this.logger.debug(\n      `Block at height ${height} has been proceed in scanner ${this.name()}, updating status`\n    );\n    const runner = this.dataSource.createQueryRunner();\n    await runner.connect();\n    await runner.startTransaction();\n    try {\n      await runner.manager.getRepository(BlockEntity).update(\n        {\n          height: height,\n          status: PROCESSING,\n          scanner: this.name(),\n        },\n        {\n          status: PROCEED,\n        }\n      );\n\n      this.logger.debug('Updating extractors status', {\n        extractorIds,\n      });\n      await runner.manager\n        .getRepository(ExtractorStatusEntity)\n        .update(\n          { extractorId: In(extractorIds), scannerId: this.name() },\n          { updateHeight: height, updateBlockHash: blockHash }\n        );\n      await runner.commitTransaction();\n    } catch (e) {\n      await runner.rollbackTransaction();\n      success = false;\n    } finally {\n      await runner.release();\n    }\n    return success;\n  };\n\n  /**\n   * Update status of a block to processing in case of fork\n   * @param height: height of expected block\n   */\n  revertBlockStatus = async (\n    height: number,\n    parentHash: string,\n    extractorIds: string[]\n  ): Promise<boolean> => {\n    let success = true;\n    const runner = this.dataSource.createQueryRunner();\n    await runner.connect();\n    await runner.startTransaction();\n    try {\n      this.logger.debug(`Reverting block status at height ${height}`);\n      await runner.manager.getRepository(BlockEntity).update(\n        {\n          height: height,\n          status: PROCEED,\n          scanner: this.name(),\n        },\n        {\n          status: PROCESSING,\n        }\n      );\n\n      this.logger.debug(\n        `Reverting extractors update height to previous block at height ${\n          height - 1\n        }`\n      );\n      await runner.manager\n        .getRepository(ExtractorStatusEntity)\n        .update(\n          { extractorId: In(extractorIds), scannerId: this.name() },\n          { updateHeight: height - 1, updateBlockHash: parentHash }\n        );\n      await runner.commitTransaction();\n    } catch (e) {\n      this.logger.warn(`An Error occurred while reverting block status: ${e}`);\n      await runner.rollbackTransaction();\n      success = false;\n    } finally {\n      await runner.release();\n    }\n    return success;\n  };\n\n  /**\n   * Insert a newly initialized extractor or update an existing extractor status\n   * @param extractorId\n   * @param height\n   * @returns\n   */\n  updateOrInsertExtractorStatus = async (\n    extractorId: string,\n    height: number,\n    blockHash: string\n  ) => {\n    this.logger.debug(\n      'Inserting new extractor status or Updating existing resynced extractor status',\n      {\n        extractorId,\n        height,\n        blockHash,\n      }\n    );\n    return await this.extractorStatusRepository.save({\n      scannerId: this.name(),\n      extractorId,\n      updateHeight: height,\n      updateBlockHash: blockHash,\n    });\n  };\n\n  /**\n   * Return extractors` status specified by id\n   * @param extractorIds\n   * @returns extractors` status\n   */\n  getExtractorsStatus = async (\n    extractorIds: string[]\n  ): Promise<ExtractorStatusEntity[]> => {\n    return await this.extractorStatusRepository.findBy({\n      scannerId: this.name(),\n      extractorId: In(extractorIds),\n    });\n  };\n}\n"]}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
import { AbstractLogger } from '@rosen-bridge/abstract-logger';
|
|
2
|
+
import { DataSource } from '@rosen-bridge/extended-typeorm';
|
|
3
|
+
import { AbstractNetworkConnector } from '@rosen-bridge/scanner-interfaces';
|
|
4
|
+
export interface ScannerConfig<TransactionType> {
|
|
5
|
+
dataSource: DataSource;
|
|
6
|
+
initialHeight: number;
|
|
7
|
+
network: AbstractNetworkConnector<TransactionType>;
|
|
8
|
+
blockRetrieveGap?: number;
|
|
9
|
+
suffix?: string;
|
|
10
|
+
logger?: AbstractLogger;
|
|
11
|
+
}
|
|
12
|
+
//# sourceMappingURL=interfaces.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"interfaces.d.ts","sourceRoot":"","sources":["../../lib/scanner/interfaces.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,cAAc,EAAE,MAAM,+BAA+B,CAAC;AAC/D,OAAO,EAAE,UAAU,EAAE,MAAM,gCAAgC,CAAC;AAC5D,OAAO,EAAE,wBAAwB,EAAE,MAAM,kCAAkC,CAAC;AAE5E,MAAM,WAAW,aAAa,CAAC,eAAe;IAC5C,UAAU,EAAE,UAAU,CAAC;IACvB,aAAa,EAAE,MAAM,CAAC;IACtB,OAAO,EAAE,wBAAwB,CAAC,eAAe,CAAC,CAAC;IACnD,gBAAgB,CAAC,EAAE,MAAM,CAAC;IAC1B,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,MAAM,CAAC,EAAE,cAAc,CAAC;CACzB"}
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export {};
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW50ZXJmYWNlcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL2xpYi9zY2FubmVyL2ludGVyZmFjZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEFic3RyYWN0TG9nZ2VyIH0gZnJvbSAnQHJvc2VuLWJyaWRnZS9hYnN0cmFjdC1sb2dnZXInO1xuaW1wb3J0IHsgRGF0YVNvdXJjZSB9IGZyb20gJ0Byb3Nlbi1icmlkZ2UvZXh0ZW5kZWQtdHlwZW9ybSc7XG5pbXBvcnQgeyBBYnN0cmFjdE5ldHdvcmtDb25uZWN0b3IgfSBmcm9tICdAcm9zZW4tYnJpZGdlL3NjYW5uZXItaW50ZXJmYWNlcyc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgU2Nhbm5lckNvbmZpZzxUcmFuc2FjdGlvblR5cGU+IHtcbiAgZGF0YVNvdXJjZTogRGF0YVNvdXJjZTtcbiAgaW5pdGlhbEhlaWdodDogbnVtYmVyO1xuICBuZXR3b3JrOiBBYnN0cmFjdE5ldHdvcmtDb25uZWN0b3I8VHJhbnNhY3Rpb25UeXBlPjtcbiAgYmxvY2tSZXRyaWV2ZUdhcD86IG51bWJlcjtcbiAgc3VmZml4Pzogc3RyaW5nO1xuICBsb2dnZXI/OiBBYnN0cmFjdExvZ2dlcjtcbn1cbiJdfQ==
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
import { AbstractNetworkConnector } from '@rosen-bridge/scanner-interfaces';
|
|
2
|
+
/**
|
|
3
|
+
* Interface for connector selection strategies
|
|
4
|
+
*/
|
|
5
|
+
export interface ConnectorSelectionStrategy<TransactionType> {
|
|
6
|
+
/**
|
|
7
|
+
* Select the next connector to use
|
|
8
|
+
* @param connectors Array of available connectors
|
|
9
|
+
* @param currentIndex Current connector index
|
|
10
|
+
* @param lastError Error from the last operation, if any
|
|
11
|
+
* @returns The index of the next connector to use
|
|
12
|
+
*/
|
|
13
|
+
selectNextConnector(connectors: Array<AbstractNetworkConnector<TransactionType>>, currentIndex: number, lastError?: Error): number;
|
|
14
|
+
}
|
|
15
|
+
/**
|
|
16
|
+
* Failover strategy - switches to next connector only on failure
|
|
17
|
+
*/
|
|
18
|
+
export declare class FailoverStrategy<TransactionType> implements ConnectorSelectionStrategy<TransactionType> {
|
|
19
|
+
selectNextConnector(connectors: Array<AbstractNetworkConnector<TransactionType>>, currentIndex: number, lastError?: Error): number;
|
|
20
|
+
}
|
|
21
|
+
/**
|
|
22
|
+
* Round-robin strategy - switches to next connector after each operation
|
|
23
|
+
*/
|
|
24
|
+
export declare class RoundRobinStrategy<TransactionType> implements ConnectorSelectionStrategy<TransactionType> {
|
|
25
|
+
selectNextConnector(connectors: Array<AbstractNetworkConnector<TransactionType>>, currentIndex: number): number;
|
|
26
|
+
}
|
|
27
|
+
//# sourceMappingURL=ConnectorSelectionStrategies.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ConnectorSelectionStrategies.d.ts","sourceRoot":"","sources":["../../../lib/scanner/network/ConnectorSelectionStrategies.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,wBAAwB,EAAE,MAAM,kCAAkC,CAAC;AAE5E;;GAEG;AACH,MAAM,WAAW,0BAA0B,CAAC,eAAe;IACzD;;;;;;OAMG;IACH,mBAAmB,CACjB,UAAU,EAAE,KAAK,CAAC,wBAAwB,CAAC,eAAe,CAAC,CAAC,EAC5D,YAAY,EAAE,MAAM,EACpB,SAAS,CAAC,EAAE,KAAK,GAChB,MAAM,CAAC;CACX;AAED;;GAEG;AACH,qBAAa,gBAAgB,CAAC,eAAe,CAC3C,YAAW,0BAA0B,CAAC,eAAe,CAAC;IAEtD,mBAAmB,CACjB,UAAU,EAAE,KAAK,CAAC,wBAAwB,CAAC,eAAe,CAAC,CAAC,EAC5D,YAAY,EAAE,MAAM,EACpB,SAAS,CAAC,EAAE,KAAK,GAChB,MAAM;CAQV;AAED;;GAEG;AACH,qBAAa,kBAAkB,CAAC,eAAe,CAC7C,YAAW,0BAA0B,CAAC,eAAe,CAAC;IAEtD,mBAAmB,CACjB,UAAU,EAAE,KAAK,CAAC,wBAAwB,CAAC,eAAe,CAAC,CAAC,EAC5D,YAAY,EAAE,MAAM,GACnB,MAAM;CAGV"}
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Failover strategy - switches to next connector only on failure
|
|
3
|
+
*/
|
|
4
|
+
export class FailoverStrategy {
|
|
5
|
+
selectNextConnector(connectors, currentIndex, lastError) {
|
|
6
|
+
// If there was an error, try the next connector
|
|
7
|
+
if (lastError) {
|
|
8
|
+
return (currentIndex + 1) % connectors.length;
|
|
9
|
+
}
|
|
10
|
+
// Otherwise stay with current connector
|
|
11
|
+
return currentIndex;
|
|
12
|
+
}
|
|
13
|
+
}
|
|
14
|
+
/**
|
|
15
|
+
* Round-robin strategy - switches to next connector after each operation
|
|
16
|
+
*/
|
|
17
|
+
export class RoundRobinStrategy {
|
|
18
|
+
selectNextConnector(connectors, currentIndex) {
|
|
19
|
+
return (currentIndex + 1) % connectors.length;
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiQ29ubmVjdG9yU2VsZWN0aW9uU3RyYXRlZ2llcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL2xpYi9zY2FubmVyL25ldHdvcmsvQ29ubmVjdG9yU2VsZWN0aW9uU3RyYXRlZ2llcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFvQkE7O0dBRUc7QUFDSCxNQUFNLE9BQU8sZ0JBQWdCO0lBRzNCLG1CQUFtQixDQUNqQixVQUE0RCxFQUM1RCxZQUFvQixFQUNwQixTQUFpQjtRQUVqQixnREFBZ0Q7UUFDaEQsSUFBSSxTQUFTLEVBQUU7WUFDYixPQUFPLENBQUMsWUFBWSxHQUFHLENBQUMsQ0FBQyxHQUFHLFVBQVUsQ0FBQyxNQUFNLENBQUM7U0FDL0M7UUFDRCx3Q0FBd0M7UUFDeEMsT0FBTyxZQUFZLENBQUM7SUFDdEIsQ0FBQztDQUNGO0FBRUQ7O0dBRUc7QUFDSCxNQUFNLE9BQU8sa0JBQWtCO0lBRzdCLG1CQUFtQixDQUNqQixVQUE0RCxFQUM1RCxZQUFvQjtRQUVwQixPQUFPLENBQUMsWUFBWSxHQUFHLENBQUMsQ0FBQyxHQUFHLFVBQVUsQ0FBQyxNQUFNLENBQUM7SUFDaEQsQ0FBQztDQUNGIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQWJzdHJhY3ROZXR3b3JrQ29ubmVjdG9yIH0gZnJvbSAnQHJvc2VuLWJyaWRnZS9zY2FubmVyLWludGVyZmFjZXMnO1xuXG4vKipcbiAqIEludGVyZmFjZSBmb3IgY29ubmVjdG9yIHNlbGVjdGlvbiBzdHJhdGVnaWVzXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgQ29ubmVjdG9yU2VsZWN0aW9uU3RyYXRlZ3k8VHJhbnNhY3Rpb25UeXBlPiB7XG4gIC8qKlxuICAgKiBTZWxlY3QgdGhlIG5leHQgY29ubmVjdG9yIHRvIHVzZVxuICAgKiBAcGFyYW0gY29ubmVjdG9ycyBBcnJheSBvZiBhdmFpbGFibGUgY29ubmVjdG9yc1xuICAgKiBAcGFyYW0gY3VycmVudEluZGV4IEN1cnJlbnQgY29ubmVjdG9yIGluZGV4XG4gICAqIEBwYXJhbSBsYXN0RXJyb3IgRXJyb3IgZnJvbSB0aGUgbGFzdCBvcGVyYXRpb24sIGlmIGFueVxuICAgKiBAcmV0dXJucyBUaGUgaW5kZXggb2YgdGhlIG5leHQgY29ubmVjdG9yIHRvIHVzZVxuICAgKi9cbiAgc2VsZWN0TmV4dENvbm5lY3RvcihcbiAgICBjb25uZWN0b3JzOiBBcnJheTxBYnN0cmFjdE5ldHdvcmtDb25uZWN0b3I8VHJhbnNhY3Rpb25UeXBlPj4sXG4gICAgY3VycmVudEluZGV4OiBudW1iZXIsXG4gICAgbGFzdEVycm9yPzogRXJyb3JcbiAgKTogbnVtYmVyO1xufVxuXG4vKipcbiAqIEZhaWxvdmVyIHN0cmF0ZWd5IC0gc3dpdGNoZXMgdG8gbmV4dCBjb25uZWN0b3Igb25seSBvbiBmYWlsdXJlXG4gKi9cbmV4cG9ydCBjbGFzcyBGYWlsb3ZlclN0cmF0ZWd5PFRyYW5zYWN0aW9uVHlwZT5cbiAgaW1wbGVtZW50cyBDb25uZWN0b3JTZWxlY3Rpb25TdHJhdGVneTxUcmFuc2FjdGlvblR5cGU+XG57XG4gIHNlbGVjdE5leHRDb25uZWN0b3IoXG4gICAgY29ubmVjdG9yczogQXJyYXk8QWJzdHJhY3ROZXR3b3JrQ29ubmVjdG9yPFRyYW5zYWN0aW9uVHlwZT4+LFxuICAgIGN1cnJlbnRJbmRleDogbnVtYmVyLFxuICAgIGxhc3RFcnJvcj86IEVycm9yXG4gICk6IG51bWJlciB7XG4gICAgLy8gSWYgdGhlcmUgd2FzIGFuIGVycm9yLCB0cnkgdGhlIG5leHQgY29ubmVjdG9yXG4gICAgaWYgKGxhc3RFcnJvcikge1xuICAgICAgcmV0dXJuIChjdXJyZW50SW5kZXggKyAxKSAlIGNvbm5lY3RvcnMubGVuZ3RoO1xuICAgIH1cbiAgICAvLyBPdGhlcndpc2Ugc3RheSB3aXRoIGN1cnJlbnQgY29ubmVjdG9yXG4gICAgcmV0dXJuIGN1cnJlbnRJbmRleDtcbiAgfVxufVxuXG4vKipcbiAqIFJvdW5kLXJvYmluIHN0cmF0ZWd5IC0gc3dpdGNoZXMgdG8gbmV4dCBjb25uZWN0b3IgYWZ0ZXIgZWFjaCBvcGVyYXRpb25cbiAqL1xuZXhwb3J0IGNsYXNzIFJvdW5kUm9iaW5TdHJhdGVneTxUcmFuc2FjdGlvblR5cGU+XG4gIGltcGxlbWVudHMgQ29ubmVjdG9yU2VsZWN0aW9uU3RyYXRlZ3k8VHJhbnNhY3Rpb25UeXBlPlxue1xuICBzZWxlY3ROZXh0Q29ubmVjdG9yKFxuICAgIGNvbm5lY3RvcnM6IEFycmF5PEFic3RyYWN0TmV0d29ya0Nvbm5lY3RvcjxUcmFuc2FjdGlvblR5cGU+PixcbiAgICBjdXJyZW50SW5kZXg6IG51bWJlclxuICApOiBudW1iZXIge1xuICAgIHJldHVybiAoY3VycmVudEluZGV4ICsgMSkgJSBjb25uZWN0b3JzLmxlbmd0aDtcbiAgfVxufVxuIl19
|
|
@@ -0,0 +1,64 @@
|
|
|
1
|
+
import { AbstractNetworkConnector, Block } from '@rosen-bridge/scanner-interfaces';
|
|
2
|
+
import { ConnectorSelectionStrategy } from './ConnectorSelectionStrategies';
|
|
3
|
+
import { AbstractLogger } from '@rosen-bridge/abstract-logger';
|
|
4
|
+
/**
|
|
5
|
+
* Network connector manager that can handle multiple connectors
|
|
6
|
+
* for a single network with pluggable selection strategy
|
|
7
|
+
*/
|
|
8
|
+
export declare class NetworkConnectorManager<TransactionType> implements AbstractNetworkConnector<TransactionType> {
|
|
9
|
+
private strategy;
|
|
10
|
+
private logger;
|
|
11
|
+
private connectors;
|
|
12
|
+
private currentConnectorIndex;
|
|
13
|
+
constructor(strategy?: ConnectorSelectionStrategy<TransactionType>, logger?: AbstractLogger);
|
|
14
|
+
/**
|
|
15
|
+
* Add a new connector to the manager
|
|
16
|
+
* @param connector The network connector instance
|
|
17
|
+
*/
|
|
18
|
+
addConnector(connector: AbstractNetworkConnector<TransactionType>): void;
|
|
19
|
+
/**
|
|
20
|
+
* Get the current active connector
|
|
21
|
+
* @returns The current network connector instance
|
|
22
|
+
* @throws Error if no connectors are available
|
|
23
|
+
*/
|
|
24
|
+
getCurrentConnector(): AbstractNetworkConnector<TransactionType>;
|
|
25
|
+
/**
|
|
26
|
+
* Execute an operation using the configured strategy
|
|
27
|
+
* @param operation The operation to execute
|
|
28
|
+
* @returns The result of the operation
|
|
29
|
+
*/
|
|
30
|
+
private executeWithStrategy;
|
|
31
|
+
/**
|
|
32
|
+
* Set the strategy to use for connector selection
|
|
33
|
+
* @param strategy The strategy to use
|
|
34
|
+
*/
|
|
35
|
+
setStrategy(strategy: ConnectorSelectionStrategy<TransactionType>): void;
|
|
36
|
+
/**
|
|
37
|
+
* Get a block at a specific height
|
|
38
|
+
* @param height The height of the block to get
|
|
39
|
+
* @returns The block at the specified height
|
|
40
|
+
*/
|
|
41
|
+
getBlockAtHeight: (height: number) => Promise<Block>;
|
|
42
|
+
/**
|
|
43
|
+
* Get the current height of the blockchain
|
|
44
|
+
* @returns The current height
|
|
45
|
+
*/
|
|
46
|
+
getCurrentHeight: () => Promise<number>;
|
|
47
|
+
/**
|
|
48
|
+
* Get all transactions in a block
|
|
49
|
+
* @param blockHash The hash of the block to get transactions from
|
|
50
|
+
* @returns Array of transactions in the block
|
|
51
|
+
*/
|
|
52
|
+
getBlockTxs: (blockHash: string) => Promise<Array<TransactionType>>;
|
|
53
|
+
/**
|
|
54
|
+
* Get the number of available connectors
|
|
55
|
+
* @returns The number of connectors
|
|
56
|
+
*/
|
|
57
|
+
getConnectorCount(): number;
|
|
58
|
+
/**
|
|
59
|
+
* Remove a connector at the specified index
|
|
60
|
+
* @param index The index of the connector to remove
|
|
61
|
+
*/
|
|
62
|
+
removeConnector(index: number): void;
|
|
63
|
+
}
|
|
64
|
+
//# sourceMappingURL=NetworkConnectorManager.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"NetworkConnectorManager.d.ts","sourceRoot":"","sources":["../../../lib/scanner/network/NetworkConnectorManager.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,wBAAwB,EACxB,KAAK,EACN,MAAM,kCAAkC,CAAC;AAC1C,OAAO,EACL,0BAA0B,EAE3B,MAAM,gCAAgC,CAAC;AACxC,OAAO,EAAE,cAAc,EAAe,MAAM,+BAA+B,CAAC;AAE5E;;;GAGG;AACH,qBAAa,uBAAuB,CAAC,eAAe,CAClD,YAAW,wBAAwB,CAAC,eAAe,CAAC;IAMlD,OAAO,CAAC,QAAQ;IAChB,OAAO,CAAC,MAAM;IALhB,OAAO,CAAC,UAAU,CAAmD;IACrE,OAAO,CAAC,qBAAqB,CAAS;gBAG5B,QAAQ,GAAE,0BAA0B,CAAC,eAAe,CAA2C,EAC/F,MAAM,GAAE,cAAkC;IAQpD;;;OAGG;IACI,YAAY,CACjB,SAAS,EAAE,wBAAwB,CAAC,eAAe,CAAC,GACnD,IAAI;IAOP;;;;OAIG;IACH,mBAAmB,IAAI,wBAAwB,CAAC,eAAe,CAAC;IAQhE;;;;OAIG;YACW,mBAAmB;IA8CjC;;;OAGG;IACI,WAAW,CAChB,QAAQ,EAAE,0BAA0B,CAAC,eAAe,CAAC,GACpD,IAAI;IAKP;;;;OAIG;IACH,gBAAgB,WAAkB,MAAM,KAAG,QAAQ,KAAK,CAAC,CAIvD;IAEF;;;OAGG;IACH,gBAAgB,QAAa,QAAQ,MAAM,CAAC,CAI1C;IAEF;;;;OAIG;IACH,WAAW,cAAqB,MAAM,KAAG,QAAQ,MAAM,eAAe,CAAC,CAAC,CAItE;IAEF;;;OAGG;IACI,iBAAiB,IAAI,MAAM;IAMlC;;;OAGG;IACI,eAAe,CAAC,KAAK,EAAE,MAAM,GAAG,IAAI;CAU5C"}
|