@rosen-bridge/cardano-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/CHANGELOG.md +9 -0
- package/README.md +38 -0
- package/dist/constants.d.ts +5 -0
- package/dist/constants.d.ts.map +1 -0
- package/dist/constants.js +5 -0
- package/dist/index.d.ts +10 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +10 -0
- package/dist/interfaces/BlockFrost.d.ts +9 -0
- package/dist/interfaces/BlockFrost.d.ts.map +1 -0
- package/dist/interfaces/BlockFrost.js +6 -0
- package/dist/interfaces/Koios.d.ts +25 -0
- package/dist/interfaces/Koios.d.ts.map +1 -0
- package/dist/interfaces/Koios.js +2 -0
- package/dist/interfaces/index.d.ts +19 -0
- package/dist/interfaces/index.d.ts.map +1 -0
- package/dist/interfaces/index.js +2 -0
- package/dist/network/blockfrost.d.ts +26 -0
- package/dist/network/blockfrost.d.ts.map +1 -0
- package/dist/network/blockfrost.js +76 -0
- package/dist/network/koios.d.ts +26 -0
- package/dist/network/koios.d.ts.map +1 -0
- package/dist/network/koios.js +89 -0
- package/dist/scanner/blockfrost.d.ts +7 -0
- package/dist/scanner/blockfrost.d.ts.map +1 -0
- package/dist/scanner/blockfrost.js +8 -0
- package/dist/scanner/koios.d.ts +7 -0
- package/dist/scanner/koios.d.ts.map +1 -0
- package/dist/scanner/koios.js +8 -0
- package/dist/scanner/ogmios.d.ts +51 -0
- package/dist/scanner/ogmios.d.ts.map +1 -0
- package/dist/scanner/ogmios.js +172 -0
- package/package.json +47 -0
package/CHANGELOG.md
ADDED
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
# @rosen-bridge/cardano-scanner
|
|
2
|
+
|
|
3
|
+
---
|
|
4
|
+
|
|
5
|
+
## 0.1.0
|
|
6
|
+
|
|
7
|
+
- Added optional suffix support for Cardano Ogmios scanner
|
|
8
|
+
- This package was **extracted** from `@rosen-bridge/scanner`.
|
|
9
|
+
You can follow the previous history in the old package’s changelog. The last changelog update is available [here](https://github.com/rosen-bridge/scanner/blob/d4a5539b01c523b101104470b03ff7023a10b70b/packages/scanner/CHANGELOG.md).
|
package/README.md
ADDED
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
# @rosen-bridge/cardano-scanner
|
|
2
|
+
|
|
3
|
+
## Table of contents
|
|
4
|
+
|
|
5
|
+
- [Introduction](#introduction)
|
|
6
|
+
- [Installation](#installation)
|
|
7
|
+
|
|
8
|
+
## Introduction
|
|
9
|
+
|
|
10
|
+
A Cardano blockchain scanner based on blackfrost , koios and ogmios API.
|
|
11
|
+
|
|
12
|
+
## Installation
|
|
13
|
+
|
|
14
|
+
npm:
|
|
15
|
+
|
|
16
|
+
```sh
|
|
17
|
+
npm i @rosen-bridge/cardano-scanner
|
|
18
|
+
```
|
|
19
|
+
|
|
20
|
+
yarn:
|
|
21
|
+
|
|
22
|
+
```sh
|
|
23
|
+
yarn add @rosen-bridge/cardano-scanner
|
|
24
|
+
```
|
|
25
|
+
|
|
26
|
+
### Usage
|
|
27
|
+
|
|
28
|
+
Cardano Scanner:
|
|
29
|
+
|
|
30
|
+
```javascript
|
|
31
|
+
const cardanoScannerConfig = {
|
|
32
|
+
koiosUrl: <koios_url>,
|
|
33
|
+
timeout: <api_timeout>,
|
|
34
|
+
initialHeight: <cardano_initial_height>,
|
|
35
|
+
dataSource: dataSource,
|
|
36
|
+
}
|
|
37
|
+
cardanoScanner = new CardanoKoiosScanner(cardanoScannerConfig)
|
|
38
|
+
```
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"constants.d.ts","sourceRoot":"","sources":["../lib/constants.ts"],"names":[],"mappings":"AAAA,eAAO,MAAM,kBAAkB,aAAa,CAAC;AAC7C,eAAO,MAAM,sBAAsB,QAAiB,CAAC;AACrD,eAAO,MAAM,0BAA0B,QAAY,CAAC;AACpD,eAAO,MAAM,yBAAyB,QAAW,CAAC"}
|
|
@@ -0,0 +1,5 @@
|
|
|
1
|
+
export const SLOT_SHELLY_NUMBER = 1591566291;
|
|
2
|
+
export const RECONNECTION_MAX_DELAY = 60 * 60 * 1000; // One hour
|
|
3
|
+
export const RECONNECTION_INITIAL_DELAY = 30 * 1000; // 30 seconds
|
|
4
|
+
export const RECONNECTION_MAX_ATTEMPTS = Infinity;
|
|
5
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uc3RhbnRzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vbGliL2NvbnN0YW50cy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxNQUFNLENBQUMsTUFBTSxrQkFBa0IsR0FBRyxVQUFVLENBQUM7QUFDN0MsTUFBTSxDQUFDLE1BQU0sc0JBQXNCLEdBQUcsRUFBRSxHQUFHLEVBQUUsR0FBRyxJQUFJLENBQUMsQ0FBQyxXQUFXO0FBQ2pFLE1BQU0sQ0FBQyxNQUFNLDBCQUEwQixHQUFHLEVBQUUsR0FBRyxJQUFJLENBQUMsQ0FBQyxhQUFhO0FBQ2xFLE1BQU0sQ0FBQyxNQUFNLHlCQUF5QixHQUFHLFFBQVEsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBjb25zdCBTTE9UX1NIRUxMWV9OVU1CRVIgPSAxNTkxNTY2MjkxO1xuZXhwb3J0IGNvbnN0IFJFQ09OTkVDVElPTl9NQVhfREVMQVkgPSA2MCAqIDYwICogMTAwMDsgLy8gT25lIGhvdXJcbmV4cG9ydCBjb25zdCBSRUNPTk5FQ1RJT05fSU5JVElBTF9ERUxBWSA9IDMwICogMTAwMDsgLy8gMzAgc2Vjb25kc1xuZXhwb3J0IGNvbnN0IFJFQ09OTkVDVElPTl9NQVhfQVRURU1QVFMgPSBJbmZpbml0eTtcbiJdfQ==
|
package/dist/index.d.ts
ADDED
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
export * from './scanner/blockfrost';
|
|
2
|
+
export * from './interfaces/BlockFrost';
|
|
3
|
+
export * from './interfaces/index';
|
|
4
|
+
export * from './interfaces/Koios';
|
|
5
|
+
export * from './scanner/koios';
|
|
6
|
+
export * from './scanner/ogmios';
|
|
7
|
+
export * from './constants';
|
|
8
|
+
export * from './network/blockfrost';
|
|
9
|
+
export * from './network/koios';
|
|
10
|
+
//# sourceMappingURL=index.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../lib/index.ts"],"names":[],"mappings":"AAAA,cAAc,sBAAsB,CAAC;AACrC,cAAc,yBAAyB,CAAC;AACxC,cAAc,oBAAoB,CAAC;AACnC,cAAc,oBAAoB,CAAC;AACnC,cAAc,iBAAiB,CAAC;AAChC,cAAc,kBAAkB,CAAC;AACjC,cAAc,aAAa,CAAC;AAC5B,cAAc,sBAAsB,CAAC;AACrC,cAAc,iBAAiB,CAAC"}
|
package/dist/index.js
ADDED
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
export * from './scanner/blockfrost';
|
|
2
|
+
export * from './interfaces/BlockFrost';
|
|
3
|
+
export * from './interfaces/index';
|
|
4
|
+
export * from './interfaces/Koios';
|
|
5
|
+
export * from './scanner/koios';
|
|
6
|
+
export * from './scanner/ogmios';
|
|
7
|
+
export * from './constants';
|
|
8
|
+
export * from './network/blockfrost';
|
|
9
|
+
export * from './network/koios';
|
|
10
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9saWIvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsY0FBYyxzQkFBc0IsQ0FBQztBQUNyQyxjQUFjLHlCQUF5QixDQUFDO0FBQ3hDLGNBQWMsb0JBQW9CLENBQUM7QUFDbkMsY0FBYyxvQkFBb0IsQ0FBQztBQUNuQyxjQUFjLGlCQUFpQixDQUFDO0FBQ2hDLGNBQWMsa0JBQWtCLENBQUM7QUFDakMsY0FBYyxhQUFhLENBQUM7QUFDNUIsY0FBYyxzQkFBc0IsQ0FBQztBQUNyQyxjQUFjLGlCQUFpQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9zY2FubmVyL2Jsb2NrZnJvc3QnO1xuZXhwb3J0ICogZnJvbSAnLi9pbnRlcmZhY2VzL0Jsb2NrRnJvc3QnO1xuZXhwb3J0ICogZnJvbSAnLi9pbnRlcmZhY2VzL2luZGV4JztcbmV4cG9ydCAqIGZyb20gJy4vaW50ZXJmYWNlcy9Lb2lvcyc7XG5leHBvcnQgKiBmcm9tICcuL3NjYW5uZXIva29pb3MnO1xuZXhwb3J0ICogZnJvbSAnLi9zY2FubmVyL29nbWlvcyc7XG5leHBvcnQgKiBmcm9tICcuL2NvbnN0YW50cyc7XG5leHBvcnQgKiBmcm9tICcuL25ldHdvcmsvYmxvY2tmcm9zdCc7XG5leHBvcnQgKiBmcm9tICcuL25ldHdvcmsva29pb3MnO1xuIl19
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
import { components } from '@blockfrost/openapi';
|
|
2
|
+
export declare class BlockFrostNullValueError extends Error {
|
|
3
|
+
constructor(msg: string);
|
|
4
|
+
}
|
|
5
|
+
export interface BlockFrostTransaction {
|
|
6
|
+
utxos: components['schemas']['tx_content_utxo'];
|
|
7
|
+
metadata: components['schemas']['tx_content_metadata'];
|
|
8
|
+
}
|
|
9
|
+
//# sourceMappingURL=BlockFrost.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"BlockFrost.d.ts","sourceRoot":"","sources":["../../lib/interfaces/BlockFrost.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,qBAAqB,CAAC;AAEjD,qBAAa,wBAAyB,SAAQ,KAAK;gBACrC,GAAG,EAAE,MAAM;CAGxB;AAED,MAAM,WAAW,qBAAqB;IACpC,KAAK,EAAE,UAAU,CAAC,SAAS,CAAC,CAAC,iBAAiB,CAAC,CAAC;IAChD,QAAQ,EAAE,UAAU,CAAC,SAAS,CAAC,CAAC,qBAAqB,CAAC,CAAC;CACxD"}
|
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
export class BlockFrostNullValueError extends Error {
|
|
2
|
+
constructor(msg) {
|
|
3
|
+
super('BlockFrostNullValueError: ' + msg);
|
|
4
|
+
}
|
|
5
|
+
}
|
|
6
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiQmxvY2tGcm9zdC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL2xpYi9pbnRlcmZhY2VzL0Jsb2NrRnJvc3QudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBRUEsTUFBTSxPQUFPLHdCQUF5QixTQUFRLEtBQUs7SUFDakQsWUFBWSxHQUFXO1FBQ3JCLEtBQUssQ0FBQyw0QkFBNEIsR0FBRyxHQUFHLENBQUMsQ0FBQztJQUM1QyxDQUFDO0NBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBjb21wb25lbnRzIH0gZnJvbSAnQGJsb2NrZnJvc3Qvb3BlbmFwaSc7XG5cbmV4cG9ydCBjbGFzcyBCbG9ja0Zyb3N0TnVsbFZhbHVlRXJyb3IgZXh0ZW5kcyBFcnJvciB7XG4gIGNvbnN0cnVjdG9yKG1zZzogc3RyaW5nKSB7XG4gICAgc3VwZXIoJ0Jsb2NrRnJvc3ROdWxsVmFsdWVFcnJvcjogJyArIG1zZyk7XG4gIH1cbn1cblxuZXhwb3J0IGludGVyZmFjZSBCbG9ja0Zyb3N0VHJhbnNhY3Rpb24ge1xuICB1dHhvczogY29tcG9uZW50c1snc2NoZW1hcyddWyd0eF9jb250ZW50X3V0eG8nXTtcbiAgbWV0YWRhdGE6IGNvbXBvbmVudHNbJ3NjaGVtYXMnXVsndHhfY29udGVudF9tZXRhZGF0YSddO1xufVxuIl19
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
import { TransactionJSON } from '@emurgo/cardano-serialization-lib-nodejs';
|
|
2
|
+
interface KoiosBlock {
|
|
3
|
+
hash: string;
|
|
4
|
+
block_height: number;
|
|
5
|
+
block_time: number;
|
|
6
|
+
tx_count: number;
|
|
7
|
+
}
|
|
8
|
+
interface KoiosBlockInfo {
|
|
9
|
+
hash: string;
|
|
10
|
+
block_height: number;
|
|
11
|
+
parent_hash: string;
|
|
12
|
+
child_hash?: string;
|
|
13
|
+
}
|
|
14
|
+
interface KoiosCborTx {
|
|
15
|
+
tx_hash: string;
|
|
16
|
+
block_hash: string;
|
|
17
|
+
block_height: number;
|
|
18
|
+
epoch_no: number;
|
|
19
|
+
absolute_slot: number;
|
|
20
|
+
tx_timestamp: number;
|
|
21
|
+
cbor: string;
|
|
22
|
+
}
|
|
23
|
+
type KoiosTransaction = KoiosCborTx & TransactionJSON;
|
|
24
|
+
export { KoiosBlock, KoiosBlockInfo, KoiosTransaction, KoiosCborTx };
|
|
25
|
+
//# sourceMappingURL=Koios.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"Koios.d.ts","sourceRoot":"","sources":["../../lib/interfaces/Koios.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,eAAe,EAAE,MAAM,0CAA0C,CAAC;AAE3E,UAAU,UAAU;IAClB,IAAI,EAAE,MAAM,CAAC;IACb,YAAY,EAAE,MAAM,CAAC;IACrB,UAAU,EAAE,MAAM,CAAC;IACnB,QAAQ,EAAE,MAAM,CAAC;CAClB;AAED,UAAU,cAAc;IACtB,IAAI,EAAE,MAAM,CAAC;IACb,YAAY,EAAE,MAAM,CAAC;IACrB,WAAW,EAAE,MAAM,CAAC;IACpB,UAAU,CAAC,EAAE,MAAM,CAAC;CACrB;AAED,UAAU,WAAW;IACnB,OAAO,EAAE,MAAM,CAAC;IAChB,UAAU,EAAE,MAAM,CAAC;IACnB,YAAY,EAAE,MAAM,CAAC;IACrB,QAAQ,EAAE,MAAM,CAAC;IACjB,aAAa,EAAE,MAAM,CAAC;IACtB,YAAY,EAAE,MAAM,CAAC;IACrB,IAAI,EAAE,MAAM,CAAC;CACd;AAED,KAAK,gBAAgB,GAAG,WAAW,GAAG,eAAe,CAAC;AAEtD,OAAO,EAAE,UAAU,EAAE,cAAc,EAAE,gBAAgB,EAAE,WAAW,EAAE,CAAC"}
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export {};
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiS29pb3MuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9saWIvaW50ZXJmYWNlcy9Lb2lvcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgVHJhbnNhY3Rpb25KU09OIH0gZnJvbSAnQGVtdXJnby9jYXJkYW5vLXNlcmlhbGl6YXRpb24tbGliLW5vZGVqcyc7XG5cbmludGVyZmFjZSBLb2lvc0Jsb2NrIHtcbiAgaGFzaDogc3RyaW5nO1xuICBibG9ja19oZWlnaHQ6IG51bWJlcjtcbiAgYmxvY2tfdGltZTogbnVtYmVyO1xuICB0eF9jb3VudDogbnVtYmVyO1xufVxuXG5pbnRlcmZhY2UgS29pb3NCbG9ja0luZm8ge1xuICBoYXNoOiBzdHJpbmc7XG4gIGJsb2NrX2hlaWdodDogbnVtYmVyO1xuICBwYXJlbnRfaGFzaDogc3RyaW5nO1xuICBjaGlsZF9oYXNoPzogc3RyaW5nO1xufVxuXG5pbnRlcmZhY2UgS29pb3NDYm9yVHgge1xuICB0eF9oYXNoOiBzdHJpbmc7XG4gIGJsb2NrX2hhc2g6IHN0cmluZztcbiAgYmxvY2tfaGVpZ2h0OiBudW1iZXI7XG4gIGVwb2NoX25vOiBudW1iZXI7XG4gIGFic29sdXRlX3Nsb3Q6IG51bWJlcjtcbiAgdHhfdGltZXN0YW1wOiBudW1iZXI7XG4gIGNib3I6IHN0cmluZztcbn1cblxudHlwZSBLb2lvc1RyYW5zYWN0aW9uID0gS29pb3NDYm9yVHggJiBUcmFuc2FjdGlvbkpTT047XG5cbmV4cG9ydCB7IEtvaW9zQmxvY2ssIEtvaW9zQmxvY2tJbmZvLCBLb2lvc1RyYW5zYWN0aW9uLCBLb2lvc0Nib3JUeCB9O1xuIl19
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
import { DataSource } from '@rosen-bridge/extended-typeorm';
|
|
2
|
+
interface OgmiosReconnectionConfig {
|
|
3
|
+
initialDelay?: number;
|
|
4
|
+
maxDelay?: number;
|
|
5
|
+
maxAttempts?: number;
|
|
6
|
+
}
|
|
7
|
+
interface CardanoOgmiosConfig {
|
|
8
|
+
nodeHostOrIp: string;
|
|
9
|
+
nodePort: number;
|
|
10
|
+
initialSlot: number;
|
|
11
|
+
initialHash: string;
|
|
12
|
+
maxTryBlock?: number;
|
|
13
|
+
dataSource: DataSource;
|
|
14
|
+
useTls?: boolean;
|
|
15
|
+
reconnectionConfig?: OgmiosReconnectionConfig;
|
|
16
|
+
suffix?: string;
|
|
17
|
+
}
|
|
18
|
+
export { OgmiosReconnectionConfig, CardanoOgmiosConfig };
|
|
19
|
+
//# sourceMappingURL=index.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../lib/interfaces/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,gCAAgC,CAAC;AAE5D,UAAU,wBAAwB;IAChC,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,WAAW,CAAC,EAAE,MAAM,CAAC;CACtB;AAED,UAAU,mBAAmB;IAC3B,YAAY,EAAE,MAAM,CAAC;IACrB,QAAQ,EAAE,MAAM,CAAC;IACjB,WAAW,EAAE,MAAM,CAAC;IACpB,WAAW,EAAE,MAAM,CAAC;IACpB,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,UAAU,EAAE,UAAU,CAAC;IACvB,MAAM,CAAC,EAAE,OAAO,CAAC;IACjB,kBAAkB,CAAC,EAAE,wBAAwB,CAAC;IAC9C,MAAM,CAAC,EAAE,MAAM,CAAC;CACjB;AAED,OAAO,EAAE,wBAAwB,EAAE,mBAAmB,EAAE,CAAC"}
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
export {};
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9saWIvaW50ZXJmYWNlcy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgRGF0YVNvdXJjZSB9IGZyb20gJ0Byb3Nlbi1icmlkZ2UvZXh0ZW5kZWQtdHlwZW9ybSc7XG5cbmludGVyZmFjZSBPZ21pb3NSZWNvbm5lY3Rpb25Db25maWcge1xuICBpbml0aWFsRGVsYXk/OiBudW1iZXI7XG4gIG1heERlbGF5PzogbnVtYmVyO1xuICBtYXhBdHRlbXB0cz86IG51bWJlcjtcbn1cblxuaW50ZXJmYWNlIENhcmRhbm9PZ21pb3NDb25maWcge1xuICBub2RlSG9zdE9ySXA6IHN0cmluZztcbiAgbm9kZVBvcnQ6IG51bWJlcjtcbiAgaW5pdGlhbFNsb3Q6IG51bWJlcjtcbiAgaW5pdGlhbEhhc2g6IHN0cmluZztcbiAgbWF4VHJ5QmxvY2s/OiBudW1iZXI7XG4gIGRhdGFTb3VyY2U6IERhdGFTb3VyY2U7XG4gIHVzZVRscz86IGJvb2xlYW47XG4gIHJlY29ubmVjdGlvbkNvbmZpZz86IE9nbWlvc1JlY29ubmVjdGlvbkNvbmZpZztcbiAgc3VmZml4Pzogc3RyaW5nO1xufVxuXG5leHBvcnQgeyBPZ21pb3NSZWNvbm5lY3Rpb25Db25maWcsIENhcmRhbm9PZ21pb3NDb25maWcgfTtcbiJdfQ==
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
import { AbstractNetworkConnector, Block } from '@rosen-bridge/scanner-interfaces';
|
|
2
|
+
import { BlockFrostTransaction } from '../interfaces/BlockFrost';
|
|
3
|
+
export declare class BlockFrostNetwork extends AbstractNetworkConnector<BlockFrostTransaction> {
|
|
4
|
+
private client;
|
|
5
|
+
constructor(projectId: string, url?: string);
|
|
6
|
+
/**
|
|
7
|
+
* get block header from height
|
|
8
|
+
* @param height
|
|
9
|
+
*/
|
|
10
|
+
getBlockAtHeight: (height: number) => Promise<Block>;
|
|
11
|
+
/**
|
|
12
|
+
* get current height for blockchain
|
|
13
|
+
*/
|
|
14
|
+
getCurrentHeight: () => Promise<number>;
|
|
15
|
+
/**
|
|
16
|
+
* Try getting transactions information.
|
|
17
|
+
* @param txHashes
|
|
18
|
+
*/
|
|
19
|
+
getTxInformation: (txId: string) => Promise<BlockFrostTransaction>;
|
|
20
|
+
/**
|
|
21
|
+
* fetch list of transaction for a specific block
|
|
22
|
+
* @param blockId
|
|
23
|
+
*/
|
|
24
|
+
getBlockTxs: (blockId: string) => Promise<Array<BlockFrostTransaction>>;
|
|
25
|
+
}
|
|
26
|
+
//# sourceMappingURL=blockfrost.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"blockfrost.d.ts","sourceRoot":"","sources":["../../lib/network/blockfrost.ts"],"names":[],"mappings":"AACA,OAAO,EACL,wBAAwB,EACxB,KAAK,EACN,MAAM,kCAAkC,CAAC;AAC1C,OAAO,EAEL,qBAAqB,EACtB,MAAM,0BAA0B,CAAC;AAElC,qBAAa,iBAAkB,SAAQ,wBAAwB,CAAC,qBAAqB,CAAC;IACpF,OAAO,CAAC,MAAM,CAAgB;gBAElB,SAAS,EAAE,MAAM,EAAE,GAAG,CAAC,EAAE,MAAM;IAS3C;;;OAGG;IACH,gBAAgB,GAAU,QAAQ,MAAM,KAAG,OAAO,CAAC,KAAK,CAAC,CAkBvD;IAEF;;OAEG;IACH,gBAAgB,QAAa,OAAO,CAAC,MAAM,CAAC,CAW1C;IAEF;;;OAGG;IACH,gBAAgB,GAAU,MAAM,MAAM,KAAG,OAAO,CAAC,qBAAqB,CAAC,CAQrE;IAEF;;;OAGG;IACH,WAAW,GACT,SAAS,MAAM,KACd,OAAO,CAAC,KAAK,CAAC,qBAAqB,CAAC,CAAC,CAStC;CACH"}
|
|
@@ -0,0 +1,76 @@
|
|
|
1
|
+
import { BlockFrostAPI } from '@blockfrost/blockfrost-js';
|
|
2
|
+
import { AbstractNetworkConnector, } from '@rosen-bridge/scanner-interfaces';
|
|
3
|
+
import { BlockFrostNullValueError, } from '../interfaces/BlockFrost';
|
|
4
|
+
export class BlockFrostNetwork extends AbstractNetworkConnector {
|
|
5
|
+
client;
|
|
6
|
+
constructor(projectId, url) {
|
|
7
|
+
super();
|
|
8
|
+
this.client = new BlockFrostAPI({
|
|
9
|
+
projectId: projectId,
|
|
10
|
+
customBackend: url,
|
|
11
|
+
network: 'mainnet',
|
|
12
|
+
});
|
|
13
|
+
}
|
|
14
|
+
/**
|
|
15
|
+
* get block header from height
|
|
16
|
+
* @param height
|
|
17
|
+
*/
|
|
18
|
+
getBlockAtHeight = async (height) => {
|
|
19
|
+
return this.client
|
|
20
|
+
.blocks(height)
|
|
21
|
+
.then((res) => {
|
|
22
|
+
if (!res.height || !res.previous_block)
|
|
23
|
+
throw new BlockFrostNullValueError(`Block height or previous_block is null`);
|
|
24
|
+
return {
|
|
25
|
+
hash: res.hash,
|
|
26
|
+
height: res.height,
|
|
27
|
+
parentHash: res.previous_block,
|
|
28
|
+
timestamp: res.time,
|
|
29
|
+
};
|
|
30
|
+
})
|
|
31
|
+
.catch((exp) => {
|
|
32
|
+
throw exp;
|
|
33
|
+
});
|
|
34
|
+
};
|
|
35
|
+
/**
|
|
36
|
+
* get current height for blockchain
|
|
37
|
+
*/
|
|
38
|
+
getCurrentHeight = async () => {
|
|
39
|
+
return this.client
|
|
40
|
+
.blocksLatest()
|
|
41
|
+
.then((block) => {
|
|
42
|
+
const height = block.height;
|
|
43
|
+
if (height)
|
|
44
|
+
return height;
|
|
45
|
+
throw new BlockFrostNullValueError('Height of last block is null');
|
|
46
|
+
})
|
|
47
|
+
.catch((exp) => {
|
|
48
|
+
throw exp;
|
|
49
|
+
});
|
|
50
|
+
};
|
|
51
|
+
/**
|
|
52
|
+
* Try getting transactions information.
|
|
53
|
+
* @param txHashes
|
|
54
|
+
*/
|
|
55
|
+
getTxInformation = async (txId) => {
|
|
56
|
+
const txUtxos = await this.client.txsUtxos(txId);
|
|
57
|
+
const txMetadata = await this.client.txsMetadata(txId);
|
|
58
|
+
return {
|
|
59
|
+
utxos: txUtxos,
|
|
60
|
+
metadata: txMetadata,
|
|
61
|
+
};
|
|
62
|
+
};
|
|
63
|
+
/**
|
|
64
|
+
* fetch list of transaction for a specific block
|
|
65
|
+
* @param blockId
|
|
66
|
+
*/
|
|
67
|
+
getBlockTxs = async (blockId) => {
|
|
68
|
+
const blockTxIds = await this.client.blocksTxsAll(blockId);
|
|
69
|
+
const blockTxs = [];
|
|
70
|
+
for (const txId of blockTxIds) {
|
|
71
|
+
blockTxs.push(await this.getTxInformation(txId));
|
|
72
|
+
}
|
|
73
|
+
return blockTxs;
|
|
74
|
+
};
|
|
75
|
+
}
|
|
76
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmxvY2tmcm9zdC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL2xpYi9uZXR3b3JrL2Jsb2NrZnJvc3QudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLGFBQWEsRUFBRSxNQUFNLDJCQUEyQixDQUFDO0FBQzFELE9BQU8sRUFDTCx3QkFBd0IsR0FFekIsTUFBTSxrQ0FBa0MsQ0FBQztBQUMxQyxPQUFPLEVBQ0wsd0JBQXdCLEdBRXpCLE1BQU0sMEJBQTBCLENBQUM7QUFFbEMsTUFBTSxPQUFPLGlCQUFrQixTQUFRLHdCQUErQztJQUM1RSxNQUFNLENBQWdCO0lBRTlCLFlBQVksU0FBaUIsRUFBRSxHQUFZO1FBQ3pDLEtBQUssRUFBRSxDQUFDO1FBQ1IsSUFBSSxDQUFDLE1BQU0sR0FBRyxJQUFJLGFBQWEsQ0FBQztZQUM5QixTQUFTLEVBQUUsU0FBUztZQUNwQixhQUFhLEVBQUUsR0FBRztZQUNsQixPQUFPLEVBQUUsU0FBUztTQUNuQixDQUFDLENBQUM7SUFDTCxDQUFDO0lBRUQ7OztPQUdHO0lBQ0gsZ0JBQWdCLEdBQUcsS0FBSyxFQUFFLE1BQWMsRUFBa0IsRUFBRTtRQUMxRCxPQUFPLElBQUksQ0FBQyxNQUFNO2FBQ2YsTUFBTSxDQUFDLE1BQU0sQ0FBQzthQUNkLElBQUksQ0FBQyxDQUFDLEdBQUcsRUFBRSxFQUFFO1lBQ1osSUFBSSxDQUFDLEdBQUcsQ0FBQyxNQUFNLElBQUksQ0FBQyxHQUFHLENBQUMsY0FBYztnQkFDcEMsTUFBTSxJQUFJLHdCQUF3QixDQUNoQyx3Q0FBd0MsQ0FDekMsQ0FBQztZQUNKLE9BQU87Z0JBQ0wsSUFBSSxFQUFFLEdBQUcsQ0FBQyxJQUFJO2dCQUNkLE1BQU0sRUFBRSxHQUFHLENBQUMsTUFBTTtnQkFDbEIsVUFBVSxFQUFFLEdBQUcsQ0FBQyxjQUFjO2dCQUM5QixTQUFTLEVBQUUsR0FBRyxDQUFDLElBQUk7YUFDcEIsQ0FBQztRQUNKLENBQUMsQ0FBQzthQUNELEtBQUssQ0FBQyxDQUFDLEdBQUcsRUFBRSxFQUFFO1lBQ2IsTUFBTSxHQUFHLENBQUM7UUFDWixDQUFDLENBQUMsQ0FBQztJQUNQLENBQUMsQ0FBQztJQUVGOztPQUVHO0lBQ0gsZ0JBQWdCLEdBQUcsS0FBSyxJQUFxQixFQUFFO1FBQzdDLE9BQU8sSUFBSSxDQUFDLE1BQU07YUFDZixZQUFZLEVBQUU7YUFDZCxJQUFJLENBQUMsQ0FBQyxLQUFLLEVBQUUsRUFBRTtZQUNkLE1BQU0sTUFBTSxHQUFHLEtBQUssQ0FBQyxNQUFNLENBQUM7WUFDNUIsSUFBSSxNQUFNO2dCQUFFLE9BQU8sTUFBTSxDQUFDO1lBQzFCLE1BQU0sSUFBSSx3QkFBd0IsQ0FBQyw4QkFBOEIsQ0FBQyxDQUFDO1FBQ3JFLENBQUMsQ0FBQzthQUNELEtBQUssQ0FBQyxDQUFDLEdBQUcsRUFBRSxFQUFFO1lBQ2IsTUFBTSxHQUFHLENBQUM7UUFDWixDQUFDLENBQUMsQ0FBQztJQUNQLENBQUMsQ0FBQztJQUVGOzs7T0FHRztJQUNILGdCQUFnQixHQUFHLEtBQUssRUFBRSxJQUFZLEVBQWtDLEVBQUU7UUFDeEUsTUFBTSxPQUFPLEdBQUcsTUFBTSxJQUFJLENBQUMsTUFBTSxDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUNqRCxNQUFNLFVBQVUsR0FBRyxNQUFNLElBQUksQ0FBQyxNQUFNLENBQUMsV0FBVyxDQUFDLElBQUksQ0FBQyxDQUFDO1FBRXZELE9BQU87WUFDTCxLQUFLLEVBQUUsT0FBTztZQUNkLFFBQVEsRUFBRSxVQUFVO1NBQ3JCLENBQUM7SUFDSixDQUFDLENBQUM7SUFFRjs7O09BR0c7SUFDSCxXQUFXLEdBQUcsS0FBSyxFQUNqQixPQUFlLEVBQ3dCLEVBQUU7UUFDekMsTUFBTSxVQUFVLEdBQUcsTUFBTSxJQUFJLENBQUMsTUFBTSxDQUFDLFlBQVksQ0FBQyxPQUFPLENBQUMsQ0FBQztRQUUzRCxNQUFNLFFBQVEsR0FBRyxFQUFFLENBQUM7UUFDcEIsS0FBSyxNQUFNLElBQUksSUFBSSxVQUFVLEVBQUUsQ0FBQztZQUM5QixRQUFRLENBQUMsSUFBSSxDQUFDLE1BQU0sSUFBSSxDQUFDLGdCQUFnQixDQUFDLElBQUksQ0FBQyxDQUFDLENBQUM7UUFDbkQsQ0FBQztRQUVELE9BQU8sUUFBUSxDQUFDO0lBQ2xCLENBQUMsQ0FBQztDQUNIIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQmxvY2tGcm9zdEFQSSB9IGZyb20gJ0BibG9ja2Zyb3N0L2Jsb2NrZnJvc3QtanMnO1xuaW1wb3J0IHtcbiAgQWJzdHJhY3ROZXR3b3JrQ29ubmVjdG9yLFxuICBCbG9jayxcbn0gZnJvbSAnQHJvc2VuLWJyaWRnZS9zY2FubmVyLWludGVyZmFjZXMnO1xuaW1wb3J0IHtcbiAgQmxvY2tGcm9zdE51bGxWYWx1ZUVycm9yLFxuICBCbG9ja0Zyb3N0VHJhbnNhY3Rpb24sXG59IGZyb20gJy4uL2ludGVyZmFjZXMvQmxvY2tGcm9zdCc7XG5cbmV4cG9ydCBjbGFzcyBCbG9ja0Zyb3N0TmV0d29yayBleHRlbmRzIEFic3RyYWN0TmV0d29ya0Nvbm5lY3RvcjxCbG9ja0Zyb3N0VHJhbnNhY3Rpb24+IHtcbiAgcHJpdmF0ZSBjbGllbnQ6IEJsb2NrRnJvc3RBUEk7XG5cbiAgY29uc3RydWN0b3IocHJvamVjdElkOiBzdHJpbmcsIHVybD86IHN0cmluZykge1xuICAgIHN1cGVyKCk7XG4gICAgdGhpcy5jbGllbnQgPSBuZXcgQmxvY2tGcm9zdEFQSSh7XG4gICAgICBwcm9qZWN0SWQ6IHByb2plY3RJZCxcbiAgICAgIGN1c3RvbUJhY2tlbmQ6IHVybCxcbiAgICAgIG5ldHdvcms6ICdtYWlubmV0JyxcbiAgICB9KTtcbiAgfVxuXG4gIC8qKlxuICAgKiBnZXQgYmxvY2sgaGVhZGVyIGZyb20gaGVpZ2h0XG4gICAqIEBwYXJhbSBoZWlnaHRcbiAgICovXG4gIGdldEJsb2NrQXRIZWlnaHQgPSBhc3luYyAoaGVpZ2h0OiBudW1iZXIpOiBQcm9taXNlPEJsb2NrPiA9PiB7XG4gICAgcmV0dXJuIHRoaXMuY2xpZW50XG4gICAgICAuYmxvY2tzKGhlaWdodClcbiAgICAgIC50aGVuKChyZXMpID0+IHtcbiAgICAgICAgaWYgKCFyZXMuaGVpZ2h0IHx8ICFyZXMucHJldmlvdXNfYmxvY2spXG4gICAgICAgICAgdGhyb3cgbmV3IEJsb2NrRnJvc3ROdWxsVmFsdWVFcnJvcihcbiAgICAgICAgICAgIGBCbG9jayBoZWlnaHQgb3IgcHJldmlvdXNfYmxvY2sgaXMgbnVsbGBcbiAgICAgICAgICApO1xuICAgICAgICByZXR1cm4ge1xuICAgICAgICAgIGhhc2g6IHJlcy5oYXNoLFxuICAgICAgICAgIGhlaWdodDogcmVzLmhlaWdodCxcbiAgICAgICAgICBwYXJlbnRIYXNoOiByZXMucHJldmlvdXNfYmxvY2ssXG4gICAgICAgICAgdGltZXN0YW1wOiByZXMudGltZSxcbiAgICAgICAgfTtcbiAgICAgIH0pXG4gICAgICAuY2F0Y2goKGV4cCkgPT4ge1xuICAgICAgICB0aHJvdyBleHA7XG4gICAgICB9KTtcbiAgfTtcblxuICAvKipcbiAgICogZ2V0IGN1cnJlbnQgaGVpZ2h0IGZvciBibG9ja2NoYWluXG4gICAqL1xuICBnZXRDdXJyZW50SGVpZ2h0ID0gYXN5bmMgKCk6IFByb21pc2U8bnVtYmVyPiA9PiB7XG4gICAgcmV0dXJuIHRoaXMuY2xpZW50XG4gICAgICAuYmxvY2tzTGF0ZXN0KClcbiAgICAgIC50aGVuKChibG9jaykgPT4ge1xuICAgICAgICBjb25zdCBoZWlnaHQgPSBibG9jay5oZWlnaHQ7XG4gICAgICAgIGlmIChoZWlnaHQpIHJldHVybiBoZWlnaHQ7XG4gICAgICAgIHRocm93IG5ldyBCbG9ja0Zyb3N0TnVsbFZhbHVlRXJyb3IoJ0hlaWdodCBvZiBsYXN0IGJsb2NrIGlzIG51bGwnKTtcbiAgICAgIH0pXG4gICAgICAuY2F0Y2goKGV4cCkgPT4ge1xuICAgICAgICB0aHJvdyBleHA7XG4gICAgICB9KTtcbiAgfTtcblxuICAvKipcbiAgICogVHJ5IGdldHRpbmcgdHJhbnNhY3Rpb25zIGluZm9ybWF0aW9uLlxuICAgKiBAcGFyYW0gdHhIYXNoZXNcbiAgICovXG4gIGdldFR4SW5mb3JtYXRpb24gPSBhc3luYyAodHhJZDogc3RyaW5nKTogUHJvbWlzZTxCbG9ja0Zyb3N0VHJhbnNhY3Rpb24+ID0+IHtcbiAgICBjb25zdCB0eFV0eG9zID0gYXdhaXQgdGhpcy5jbGllbnQudHhzVXR4b3ModHhJZCk7XG4gICAgY29uc3QgdHhNZXRhZGF0YSA9IGF3YWl0IHRoaXMuY2xpZW50LnR4c01ldGFkYXRhKHR4SWQpO1xuXG4gICAgcmV0dXJuIHtcbiAgICAgIHV0eG9zOiB0eFV0eG9zLFxuICAgICAgbWV0YWRhdGE6IHR4TWV0YWRhdGEsXG4gICAgfTtcbiAgfTtcblxuICAvKipcbiAgICogZmV0Y2ggbGlzdCBvZiB0cmFuc2FjdGlvbiBmb3IgYSBzcGVjaWZpYyBibG9ja1xuICAgKiBAcGFyYW0gYmxvY2tJZFxuICAgKi9cbiAgZ2V0QmxvY2tUeHMgPSBhc3luYyAoXG4gICAgYmxvY2tJZDogc3RyaW5nXG4gICk6IFByb21pc2U8QXJyYXk8QmxvY2tGcm9zdFRyYW5zYWN0aW9uPj4gPT4ge1xuICAgIGNvbnN0IGJsb2NrVHhJZHMgPSBhd2FpdCB0aGlzLmNsaWVudC5ibG9ja3NUeHNBbGwoYmxvY2tJZCk7XG5cbiAgICBjb25zdCBibG9ja1R4cyA9IFtdO1xuICAgIGZvciAoY29uc3QgdHhJZCBvZiBibG9ja1R4SWRzKSB7XG4gICAgICBibG9ja1R4cy5wdXNoKGF3YWl0IHRoaXMuZ2V0VHhJbmZvcm1hdGlvbih0eElkKSk7XG4gICAgfVxuXG4gICAgcmV0dXJuIGJsb2NrVHhzO1xuICB9O1xufVxuIl19
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
import { AbstractNetworkConnector, Block } from '@rosen-bridge/scanner-interfaces';
|
|
2
|
+
import { KoiosTransaction } from '../interfaces/Koios';
|
|
3
|
+
export declare class KoiosNetwork extends AbstractNetworkConnector<KoiosTransaction> {
|
|
4
|
+
private readonly url;
|
|
5
|
+
private readonly timeout;
|
|
6
|
+
private koios;
|
|
7
|
+
constructor(url: string, timeout: number, authToken?: string);
|
|
8
|
+
/**
|
|
9
|
+
* Returns block at height
|
|
10
|
+
* @param height
|
|
11
|
+
* @returns Block
|
|
12
|
+
*/
|
|
13
|
+
getBlockAtHeight: (height: number) => Promise<Block>;
|
|
14
|
+
/**
|
|
15
|
+
* Returns current network height
|
|
16
|
+
* @returns current height
|
|
17
|
+
*/
|
|
18
|
+
getCurrentHeight: () => Promise<number>;
|
|
19
|
+
/**
|
|
20
|
+
* Return transactions in a block with specified hash
|
|
21
|
+
* @param blockHash
|
|
22
|
+
* @returns array of KoiosTransaction
|
|
23
|
+
*/
|
|
24
|
+
getBlockTxs: (blockHash: string) => Promise<Array<KoiosTransaction>>;
|
|
25
|
+
}
|
|
26
|
+
//# sourceMappingURL=koios.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"koios.d.ts","sourceRoot":"","sources":["../../lib/network/koios.ts"],"names":[],"mappings":"AAEA,OAAO,EACL,wBAAwB,EACxB,KAAK,EACN,MAAM,kCAAkC,CAAC;AAC1C,OAAO,EAA2B,gBAAgB,EAAE,MAAM,qBAAqB,CAAC;AAGhF,qBAAa,YAAa,SAAQ,wBAAwB,CAAC,gBAAgB,CAAC;IAC1E,OAAO,CAAC,QAAQ,CAAC,GAAG,CAAS;IAC7B,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAS;IACjC,OAAO,CAAC,KAAK,CAAQ;gBAET,GAAG,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,EAAE,SAAS,CAAC,EAAE,MAAM;IAgB5D;;;;OAIG;IACH,gBAAgB,GAAI,QAAQ,MAAM,KAAG,OAAO,CAAC,KAAK,CAAC,CAuBjD;IAEF;;;OAGG;IACH,gBAAgB,QAAO,OAAO,CAAC,MAAM,CAAC,CASpC;IAEF;;;;OAIG;IACH,WAAW,GAAI,WAAW,MAAM,KAAG,OAAO,CAAC,KAAK,CAAC,gBAAgB,CAAC,CAAC,CAmBjE;CACH"}
|
|
@@ -0,0 +1,89 @@
|
|
|
1
|
+
import axios from '@rosen-bridge/rate-limited-axios';
|
|
2
|
+
import { Transaction } from '@emurgo/cardano-serialization-lib-nodejs';
|
|
3
|
+
import { AbstractNetworkConnector, } from '@rosen-bridge/scanner-interfaces';
|
|
4
|
+
import JsonBigint from '@rosen-bridge/json-bigint';
|
|
5
|
+
export class KoiosNetwork extends AbstractNetworkConnector {
|
|
6
|
+
url;
|
|
7
|
+
timeout;
|
|
8
|
+
koios;
|
|
9
|
+
constructor(url, timeout, authToken) {
|
|
10
|
+
super();
|
|
11
|
+
this.url = url;
|
|
12
|
+
this.timeout = timeout;
|
|
13
|
+
this.koios = axios.create({
|
|
14
|
+
baseURL: this.url,
|
|
15
|
+
timeout: this.timeout,
|
|
16
|
+
headers: { 'Content-Type': 'application/json' },
|
|
17
|
+
});
|
|
18
|
+
if (authToken) {
|
|
19
|
+
this.koios.defaults.headers.common['Authorization'] = `Bearer ${authToken}`;
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
/**
|
|
23
|
+
* Returns block at height
|
|
24
|
+
* @param height
|
|
25
|
+
* @returns Block
|
|
26
|
+
*/
|
|
27
|
+
getBlockAtHeight = (height) => {
|
|
28
|
+
return this.koios
|
|
29
|
+
.get('/blocks', {
|
|
30
|
+
params: {
|
|
31
|
+
block_height: `lte.${height}`,
|
|
32
|
+
limit: 2,
|
|
33
|
+
select: 'hash,block_height,block_time,tx_count',
|
|
34
|
+
},
|
|
35
|
+
})
|
|
36
|
+
.then((res) => {
|
|
37
|
+
const block = res.data[0];
|
|
38
|
+
const parentBlock = res.data[1];
|
|
39
|
+
return {
|
|
40
|
+
hash: block.hash,
|
|
41
|
+
height: block.block_height,
|
|
42
|
+
parentHash: parentBlock.hash,
|
|
43
|
+
timestamp: block.block_time,
|
|
44
|
+
txCount: block.tx_count,
|
|
45
|
+
};
|
|
46
|
+
})
|
|
47
|
+
.catch((exp) => {
|
|
48
|
+
throw exp;
|
|
49
|
+
});
|
|
50
|
+
};
|
|
51
|
+
/**
|
|
52
|
+
* Returns current network height
|
|
53
|
+
* @returns current height
|
|
54
|
+
*/
|
|
55
|
+
getCurrentHeight = () => {
|
|
56
|
+
return this.koios
|
|
57
|
+
.get('/blocks', {
|
|
58
|
+
params: { offset: 0, limit: 1, select: 'hash,block_height' },
|
|
59
|
+
})
|
|
60
|
+
.then((res) => res.data[0].block_height)
|
|
61
|
+
.catch((exp) => {
|
|
62
|
+
throw exp;
|
|
63
|
+
});
|
|
64
|
+
};
|
|
65
|
+
/**
|
|
66
|
+
* Return transactions in a block with specified hash
|
|
67
|
+
* @param blockHash
|
|
68
|
+
* @returns array of KoiosTransaction
|
|
69
|
+
*/
|
|
70
|
+
getBlockTxs = (blockHash) => {
|
|
71
|
+
return this.koios
|
|
72
|
+
.post('/block_tx_cbor', {
|
|
73
|
+
_block_hashes: [blockHash],
|
|
74
|
+
})
|
|
75
|
+
.then((res) => {
|
|
76
|
+
return res.data.map((tx) => {
|
|
77
|
+
const serializedTx = Transaction.from_bytes(new Uint8Array(Buffer.from(tx.cbor, 'hex')));
|
|
78
|
+
return {
|
|
79
|
+
...tx,
|
|
80
|
+
...JsonBigint.parse(serializedTx.to_json()),
|
|
81
|
+
};
|
|
82
|
+
});
|
|
83
|
+
})
|
|
84
|
+
.catch((exp) => {
|
|
85
|
+
throw exp;
|
|
86
|
+
});
|
|
87
|
+
};
|
|
88
|
+
}
|
|
89
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoia29pb3MuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9saWIvbmV0d29yay9rb2lvcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEtBQWdCLE1BQU0sa0NBQWtDLENBQUM7QUFDaEUsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLDBDQUEwQyxDQUFDO0FBQ3ZFLE9BQU8sRUFDTCx3QkFBd0IsR0FFekIsTUFBTSxrQ0FBa0MsQ0FBQztBQUUxQyxPQUFPLFVBQVUsTUFBTSwyQkFBMkIsQ0FBQztBQUVuRCxNQUFNLE9BQU8sWUFBYSxTQUFRLHdCQUEwQztJQUN6RCxHQUFHLENBQVM7SUFDWixPQUFPLENBQVM7SUFDekIsS0FBSyxDQUFRO0lBRXJCLFlBQVksR0FBVyxFQUFFLE9BQWUsRUFBRSxTQUFrQjtRQUMxRCxLQUFLLEVBQUUsQ0FBQztRQUNSLElBQUksQ0FBQyxHQUFHLEdBQUcsR0FBRyxDQUFDO1FBQ2YsSUFBSSxDQUFDLE9BQU8sR0FBRyxPQUFPLENBQUM7UUFDdkIsSUFBSSxDQUFDLEtBQUssR0FBRyxLQUFLLENBQUMsTUFBTSxDQUFDO1lBQ3hCLE9BQU8sRUFBRSxJQUFJLENBQUMsR0FBRztZQUNqQixPQUFPLEVBQUUsSUFBSSxDQUFDLE9BQU87WUFDckIsT0FBTyxFQUFFLEVBQUUsY0FBYyxFQUFFLGtCQUFrQixFQUFFO1NBQ2hELENBQUMsQ0FBQztRQUNILElBQUksU0FBUyxFQUFFLENBQUM7WUFDZCxJQUFJLENBQUMsS0FBSyxDQUFDLFFBQVEsQ0FBQyxPQUFPLENBQUMsTUFBTSxDQUNoQyxlQUFlLENBQ2hCLEdBQUcsVUFBVSxTQUFTLEVBQUUsQ0FBQztRQUM1QixDQUFDO0lBQ0gsQ0FBQztJQUVEOzs7O09BSUc7SUFDSCxnQkFBZ0IsR0FBRyxDQUFDLE1BQWMsRUFBa0IsRUFBRTtRQUNwRCxPQUFPLElBQUksQ0FBQyxLQUFLO2FBQ2QsR0FBRyxDQUFvQixTQUFTLEVBQUU7WUFDakMsTUFBTSxFQUFFO2dCQUNOLFlBQVksRUFBRSxPQUFPLE1BQU0sRUFBRTtnQkFDN0IsS0FBSyxFQUFFLENBQUM7Z0JBQ1IsTUFBTSxFQUFFLHVDQUF1QzthQUNoRDtTQUNGLENBQUM7YUFDRCxJQUFJLENBQUMsQ0FBQyxHQUFHLEVBQUUsRUFBRTtZQUNaLE1BQU0sS0FBSyxHQUFHLEdBQUcsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLENBQUM7WUFDMUIsTUFBTSxXQUFXLEdBQUcsR0FBRyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQztZQUNoQyxPQUFPO2dCQUNMLElBQUksRUFBRSxLQUFLLENBQUMsSUFBSTtnQkFDaEIsTUFBTSxFQUFFLEtBQUssQ0FBQyxZQUFZO2dCQUMxQixVQUFVLEVBQUUsV0FBVyxDQUFDLElBQUk7Z0JBQzVCLFNBQVMsRUFBRSxLQUFLLENBQUMsVUFBVTtnQkFDM0IsT0FBTyxFQUFFLEtBQUssQ0FBQyxRQUFRO2FBQ3hCLENBQUM7UUFDSixDQUFDLENBQUM7YUFDRCxLQUFLLENBQUMsQ0FBQyxHQUFHLEVBQUUsRUFBRTtZQUNiLE1BQU0sR0FBRyxDQUFDO1FBQ1osQ0FBQyxDQUFDLENBQUM7SUFDUCxDQUFDLENBQUM7SUFFRjs7O09BR0c7SUFDSCxnQkFBZ0IsR0FBRyxHQUFvQixFQUFFO1FBQ3ZDLE9BQU8sSUFBSSxDQUFDLEtBQUs7YUFDZCxHQUFHLENBQW9CLFNBQVMsRUFBRTtZQUNqQyxNQUFNLEVBQUUsRUFBRSxNQUFNLEVBQUUsQ0FBQyxFQUFFLEtBQUssRUFBRSxDQUFDLEVBQUUsTUFBTSxFQUFFLG1CQUFtQixFQUFFO1NBQzdELENBQUM7YUFDRCxJQUFJLENBQUMsQ0FBQyxHQUFHLEVBQUUsRUFBRSxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLENBQUMsWUFBWSxDQUFDO2FBQ3ZDLEtBQUssQ0FBQyxDQUFDLEdBQUcsRUFBRSxFQUFFO1lBQ2IsTUFBTSxHQUFHLENBQUM7UUFDWixDQUFDLENBQUMsQ0FBQztJQUNQLENBQUMsQ0FBQztJQUVGOzs7O09BSUc7SUFDSCxXQUFXLEdBQUcsQ0FBQyxTQUFpQixFQUFvQyxFQUFFO1FBQ3BFLE9BQU8sSUFBSSxDQUFDLEtBQUs7YUFDZCxJQUFJLENBQXFCLGdCQUFnQixFQUFFO1lBQzFDLGFBQWEsRUFBRSxDQUFDLFNBQVMsQ0FBQztTQUMzQixDQUFDO2FBQ0QsSUFBSSxDQUFDLENBQUMsR0FBRyxFQUFFLEVBQUU7WUFDWixPQUFPLEdBQUcsQ0FBQyxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUMsRUFBRSxFQUFFLEVBQUU7Z0JBQ3pCLE1BQU0sWUFBWSxHQUFHLFdBQVcsQ0FBQyxVQUFVLENBQ3pDLElBQUksVUFBVSxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDLElBQUksRUFBRSxLQUFLLENBQUMsQ0FBQyxDQUM1QyxDQUFDO2dCQUNGLE9BQU87b0JBQ0wsR0FBRyxFQUFFO29CQUNMLEdBQUcsVUFBVSxDQUFDLEtBQUssQ0FBQyxZQUFZLENBQUMsT0FBTyxFQUFFLENBQUM7aUJBQzVDLENBQUM7WUFDSixDQUFDLENBQUMsQ0FBQztRQUNMLENBQUMsQ0FBQzthQUNELEtBQUssQ0FBQyxDQUFDLEdBQUcsRUFBRSxFQUFFO1lBQ2IsTUFBTSxHQUFHLENBQUM7UUFDWixDQUFDLENBQUMsQ0FBQztJQUNQLENBQUMsQ0FBQztDQUNIIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IGF4aW9zLCB7IEF4aW9zIH0gZnJvbSAnQHJvc2VuLWJyaWRnZS9yYXRlLWxpbWl0ZWQtYXhpb3MnO1xuaW1wb3J0IHsgVHJhbnNhY3Rpb24gfSBmcm9tICdAZW11cmdvL2NhcmRhbm8tc2VyaWFsaXphdGlvbi1saWItbm9kZWpzJztcbmltcG9ydCB7XG4gIEFic3RyYWN0TmV0d29ya0Nvbm5lY3RvcixcbiAgQmxvY2ssXG59IGZyb20gJ0Byb3Nlbi1icmlkZ2Uvc2Nhbm5lci1pbnRlcmZhY2VzJztcbmltcG9ydCB7IEtvaW9zQmxvY2ssIEtvaW9zQ2JvclR4LCBLb2lvc1RyYW5zYWN0aW9uIH0gZnJvbSAnLi4vaW50ZXJmYWNlcy9Lb2lvcyc7XG5pbXBvcnQgSnNvbkJpZ2ludCBmcm9tICdAcm9zZW4tYnJpZGdlL2pzb24tYmlnaW50JztcblxuZXhwb3J0IGNsYXNzIEtvaW9zTmV0d29yayBleHRlbmRzIEFic3RyYWN0TmV0d29ya0Nvbm5lY3RvcjxLb2lvc1RyYW5zYWN0aW9uPiB7XG4gIHByaXZhdGUgcmVhZG9ubHkgdXJsOiBzdHJpbmc7XG4gIHByaXZhdGUgcmVhZG9ubHkgdGltZW91dDogbnVtYmVyO1xuICBwcml2YXRlIGtvaW9zOiBBeGlvcztcblxuICBjb25zdHJ1Y3Rvcih1cmw6IHN0cmluZywgdGltZW91dDogbnVtYmVyLCBhdXRoVG9rZW4/OiBzdHJpbmcpIHtcbiAgICBzdXBlcigpO1xuICAgIHRoaXMudXJsID0gdXJsO1xuICAgIHRoaXMudGltZW91dCA9IHRpbWVvdXQ7XG4gICAgdGhpcy5rb2lvcyA9IGF4aW9zLmNyZWF0ZSh7XG4gICAgICBiYXNlVVJMOiB0aGlzLnVybCxcbiAgICAgIHRpbWVvdXQ6IHRoaXMudGltZW91dCxcbiAgICAgIGhlYWRlcnM6IHsgJ0NvbnRlbnQtVHlwZSc6ICdhcHBsaWNhdGlvbi9qc29uJyB9LFxuICAgIH0pO1xuICAgIGlmIChhdXRoVG9rZW4pIHtcbiAgICAgIHRoaXMua29pb3MuZGVmYXVsdHMuaGVhZGVycy5jb21tb25bXG4gICAgICAgICdBdXRob3JpemF0aW9uJ1xuICAgICAgXSA9IGBCZWFyZXIgJHthdXRoVG9rZW59YDtcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogUmV0dXJucyBibG9jayBhdCBoZWlnaHRcbiAgICogQHBhcmFtIGhlaWdodFxuICAgKiBAcmV0dXJucyBCbG9ja1xuICAgKi9cbiAgZ2V0QmxvY2tBdEhlaWdodCA9IChoZWlnaHQ6IG51bWJlcik6IFByb21pc2U8QmxvY2s+ID0+IHtcbiAgICByZXR1cm4gdGhpcy5rb2lvc1xuICAgICAgLmdldDxBcnJheTxLb2lvc0Jsb2NrPj4oJy9ibG9ja3MnLCB7XG4gICAgICAgIHBhcmFtczoge1xuICAgICAgICAgIGJsb2NrX2hlaWdodDogYGx0ZS4ke2hlaWdodH1gLFxuICAgICAgICAgIGxpbWl0OiAyLFxuICAgICAgICAgIHNlbGVjdDogJ2hhc2gsYmxvY2tfaGVpZ2h0LGJsb2NrX3RpbWUsdHhfY291bnQnLFxuICAgICAgICB9LFxuICAgICAgfSlcbiAgICAgIC50aGVuKChyZXMpID0+IHtcbiAgICAgICAgY29uc3QgYmxvY2sgPSByZXMuZGF0YVswXTtcbiAgICAgICAgY29uc3QgcGFyZW50QmxvY2sgPSByZXMuZGF0YVsxXTtcbiAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICBoYXNoOiBibG9jay5oYXNoLFxuICAgICAgICAgIGhlaWdodDogYmxvY2suYmxvY2tfaGVpZ2h0LFxuICAgICAgICAgIHBhcmVudEhhc2g6IHBhcmVudEJsb2NrLmhhc2gsXG4gICAgICAgICAgdGltZXN0YW1wOiBibG9jay5ibG9ja190aW1lLFxuICAgICAgICAgIHR4Q291bnQ6IGJsb2NrLnR4X2NvdW50LFxuICAgICAgICB9O1xuICAgICAgfSlcbiAgICAgIC5jYXRjaCgoZXhwKSA9PiB7XG4gICAgICAgIHRocm93IGV4cDtcbiAgICAgIH0pO1xuICB9O1xuXG4gIC8qKlxuICAgKiBSZXR1cm5zIGN1cnJlbnQgbmV0d29yayBoZWlnaHRcbiAgICogQHJldHVybnMgY3VycmVudCBoZWlnaHRcbiAgICovXG4gIGdldEN1cnJlbnRIZWlnaHQgPSAoKTogUHJvbWlzZTxudW1iZXI+ID0+IHtcbiAgICByZXR1cm4gdGhpcy5rb2lvc1xuICAgICAgLmdldDxBcnJheTxLb2lvc0Jsb2NrPj4oJy9ibG9ja3MnLCB7XG4gICAgICAgIHBhcmFtczogeyBvZmZzZXQ6IDAsIGxpbWl0OiAxLCBzZWxlY3Q6ICdoYXNoLGJsb2NrX2hlaWdodCcgfSxcbiAgICAgIH0pXG4gICAgICAudGhlbigocmVzKSA9PiByZXMuZGF0YVswXS5ibG9ja19oZWlnaHQpXG4gICAgICAuY2F0Y2goKGV4cCkgPT4ge1xuICAgICAgICB0aHJvdyBleHA7XG4gICAgICB9KTtcbiAgfTtcblxuICAvKipcbiAgICogUmV0dXJuIHRyYW5zYWN0aW9ucyBpbiBhIGJsb2NrIHdpdGggc3BlY2lmaWVkIGhhc2hcbiAgICogQHBhcmFtIGJsb2NrSGFzaFxuICAgKiBAcmV0dXJucyBhcnJheSBvZiBLb2lvc1RyYW5zYWN0aW9uXG4gICAqL1xuICBnZXRCbG9ja1R4cyA9IChibG9ja0hhc2g6IHN0cmluZyk6IFByb21pc2U8QXJyYXk8S29pb3NUcmFuc2FjdGlvbj4+ID0+IHtcbiAgICByZXR1cm4gdGhpcy5rb2lvc1xuICAgICAgLnBvc3Q8QXJyYXk8S29pb3NDYm9yVHg+PignL2Jsb2NrX3R4X2Nib3InLCB7XG4gICAgICAgIF9ibG9ja19oYXNoZXM6IFtibG9ja0hhc2hdLFxuICAgICAgfSlcbiAgICAgIC50aGVuKChyZXMpID0+IHtcbiAgICAgICAgcmV0dXJuIHJlcy5kYXRhLm1hcCgodHgpID0+IHtcbiAgICAgICAgICBjb25zdCBzZXJpYWxpemVkVHggPSBUcmFuc2FjdGlvbi5mcm9tX2J5dGVzKFxuICAgICAgICAgICAgbmV3IFVpbnQ4QXJyYXkoQnVmZmVyLmZyb20odHguY2JvciwgJ2hleCcpKVxuICAgICAgICAgICk7XG4gICAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICAgIC4uLnR4LFxuICAgICAgICAgICAgLi4uSnNvbkJpZ2ludC5wYXJzZShzZXJpYWxpemVkVHgudG9fanNvbigpKSxcbiAgICAgICAgICB9O1xuICAgICAgICB9KTtcbiAgICAgIH0pXG4gICAgICAuY2F0Y2goKGV4cCkgPT4ge1xuICAgICAgICB0aHJvdyBleHA7XG4gICAgICB9KTtcbiAgfTtcbn1cbiJdfQ==
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
import { BlockFrostTransaction } from '../interfaces/BlockFrost';
|
|
2
|
+
import { GeneralScanner, ScannerConfig } from '@rosen-bridge/abstract-scanner';
|
|
3
|
+
declare class CardanoBlockFrostScanner extends GeneralScanner<BlockFrostTransaction> {
|
|
4
|
+
constructor(config: ScannerConfig<BlockFrostTransaction>);
|
|
5
|
+
}
|
|
6
|
+
export { CardanoBlockFrostScanner };
|
|
7
|
+
//# sourceMappingURL=blockfrost.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"blockfrost.d.ts","sourceRoot":"","sources":["../../lib/scanner/blockfrost.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,qBAAqB,EAAE,MAAM,0BAA0B,CAAC;AACjE,OAAO,EAAE,cAAc,EAAE,aAAa,EAAE,MAAM,gCAAgC,CAAC;AAE/E,cAAM,wBAAyB,SAAQ,cAAc,CAAC,qBAAqB,CAAC;gBAC9D,MAAM,EAAE,aAAa,CAAC,qBAAqB,CAAC;CAWzD;AACD,OAAO,EAAE,wBAAwB,EAAE,CAAC"}
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
import { GeneralScanner } from '@rosen-bridge/abstract-scanner';
|
|
2
|
+
class CardanoBlockFrostScanner extends GeneralScanner {
|
|
3
|
+
constructor(config) {
|
|
4
|
+
super('cardano', config.dataSource, config.initialHeight, config.network, config.blockRetrieveGap, config.logger, config.suffix);
|
|
5
|
+
}
|
|
6
|
+
}
|
|
7
|
+
export { CardanoBlockFrostScanner };
|
|
8
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmxvY2tmcm9zdC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL2xpYi9zY2FubmVyL2Jsb2NrZnJvc3QudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQ0EsT0FBTyxFQUFFLGNBQWMsRUFBaUIsTUFBTSxnQ0FBZ0MsQ0FBQztBQUUvRSxNQUFNLHdCQUF5QixTQUFRLGNBQXFDO0lBQzFFLFlBQVksTUFBNEM7UUFDdEQsS0FBSyxDQUNILFNBQVMsRUFDVCxNQUFNLENBQUMsVUFBVSxFQUNqQixNQUFNLENBQUMsYUFBYSxFQUNwQixNQUFNLENBQUMsT0FBTyxFQUNkLE1BQU0sQ0FBQyxnQkFBZ0IsRUFDdkIsTUFBTSxDQUFDLE1BQU0sRUFDYixNQUFNLENBQUMsTUFBTSxDQUNkLENBQUM7SUFDSixDQUFDO0NBQ0Y7QUFDRCxPQUFPLEVBQUUsd0JBQXdCLEVBQUUsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEJsb2NrRnJvc3RUcmFuc2FjdGlvbiB9IGZyb20gJy4uL2ludGVyZmFjZXMvQmxvY2tGcm9zdCc7XG5pbXBvcnQgeyBHZW5lcmFsU2Nhbm5lciwgU2Nhbm5lckNvbmZpZyB9IGZyb20gJ0Byb3Nlbi1icmlkZ2UvYWJzdHJhY3Qtc2Nhbm5lcic7XG5cbmNsYXNzIENhcmRhbm9CbG9ja0Zyb3N0U2Nhbm5lciBleHRlbmRzIEdlbmVyYWxTY2FubmVyPEJsb2NrRnJvc3RUcmFuc2FjdGlvbj4ge1xuICBjb25zdHJ1Y3Rvcihjb25maWc6IFNjYW5uZXJDb25maWc8QmxvY2tGcm9zdFRyYW5zYWN0aW9uPikge1xuICAgIHN1cGVyKFxuICAgICAgJ2NhcmRhbm8nLFxuICAgICAgY29uZmlnLmRhdGFTb3VyY2UsXG4gICAgICBjb25maWcuaW5pdGlhbEhlaWdodCxcbiAgICAgIGNvbmZpZy5uZXR3b3JrLFxuICAgICAgY29uZmlnLmJsb2NrUmV0cmlldmVHYXAsXG4gICAgICBjb25maWcubG9nZ2VyLFxuICAgICAgY29uZmlnLnN1ZmZpeFxuICAgICk7XG4gIH1cbn1cbmV4cG9ydCB7IENhcmRhbm9CbG9ja0Zyb3N0U2Nhbm5lciB9O1xuIl19
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
import { KoiosTransaction } from '../interfaces/Koios';
|
|
2
|
+
import { GeneralScanner, ScannerConfig } from '@rosen-bridge/abstract-scanner';
|
|
3
|
+
declare class CardanoKoiosScanner extends GeneralScanner<KoiosTransaction> {
|
|
4
|
+
constructor(config: ScannerConfig<KoiosTransaction>);
|
|
5
|
+
}
|
|
6
|
+
export { CardanoKoiosScanner };
|
|
7
|
+
//# sourceMappingURL=koios.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"koios.d.ts","sourceRoot":"","sources":["../../lib/scanner/koios.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,gBAAgB,EAAE,MAAM,qBAAqB,CAAC;AACvD,OAAO,EAAE,cAAc,EAAE,aAAa,EAAE,MAAM,gCAAgC,CAAC;AAE/E,cAAM,mBAAoB,SAAQ,cAAc,CAAC,gBAAgB,CAAC;gBACpD,MAAM,EAAE,aAAa,CAAC,gBAAgB,CAAC;CAWpD;AAED,OAAO,EAAE,mBAAmB,EAAE,CAAC"}
|
|
@@ -0,0 +1,8 @@
|
|
|
1
|
+
import { GeneralScanner } from '@rosen-bridge/abstract-scanner';
|
|
2
|
+
class CardanoKoiosScanner extends GeneralScanner {
|
|
3
|
+
constructor(config) {
|
|
4
|
+
super('cardano', config.dataSource, config.initialHeight, config.network, config.blockRetrieveGap, config.logger, config.suffix);
|
|
5
|
+
}
|
|
6
|
+
}
|
|
7
|
+
export { CardanoKoiosScanner };
|
|
8
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoia29pb3MuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9saWIvc2Nhbm5lci9rb2lvcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFDQSxPQUFPLEVBQUUsY0FBYyxFQUFpQixNQUFNLGdDQUFnQyxDQUFDO0FBRS9FLE1BQU0sbUJBQW9CLFNBQVEsY0FBZ0M7SUFDaEUsWUFBWSxNQUF1QztRQUNqRCxLQUFLLENBQ0gsU0FBUyxFQUNULE1BQU0sQ0FBQyxVQUFVLEVBQ2pCLE1BQU0sQ0FBQyxhQUFhLEVBQ3BCLE1BQU0sQ0FBQyxPQUFPLEVBQ2QsTUFBTSxDQUFDLGdCQUFnQixFQUN2QixNQUFNLENBQUMsTUFBTSxFQUNiLE1BQU0sQ0FBQyxNQUFNLENBQ2QsQ0FBQztJQUNKLENBQUM7Q0FDRjtBQUVELE9BQU8sRUFBRSxtQkFBbUIsRUFBRSxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgS29pb3NUcmFuc2FjdGlvbiB9IGZyb20gJy4uL2ludGVyZmFjZXMvS29pb3MnO1xuaW1wb3J0IHsgR2VuZXJhbFNjYW5uZXIsIFNjYW5uZXJDb25maWcgfSBmcm9tICdAcm9zZW4tYnJpZGdlL2Fic3RyYWN0LXNjYW5uZXInO1xuXG5jbGFzcyBDYXJkYW5vS29pb3NTY2FubmVyIGV4dGVuZHMgR2VuZXJhbFNjYW5uZXI8S29pb3NUcmFuc2FjdGlvbj4ge1xuICBjb25zdHJ1Y3Rvcihjb25maWc6IFNjYW5uZXJDb25maWc8S29pb3NUcmFuc2FjdGlvbj4pIHtcbiAgICBzdXBlcihcbiAgICAgICdjYXJkYW5vJyxcbiAgICAgIGNvbmZpZy5kYXRhU291cmNlLFxuICAgICAgY29uZmlnLmluaXRpYWxIZWlnaHQsXG4gICAgICBjb25maWcubmV0d29yayxcbiAgICAgIGNvbmZpZy5ibG9ja1JldHJpZXZlR2FwLFxuICAgICAgY29uZmlnLmxvZ2dlcixcbiAgICAgIGNvbmZpZy5zdWZmaXhcbiAgICApO1xuICB9XG59XG5cbmV4cG9ydCB7IENhcmRhbm9Lb2lvc1NjYW5uZXIgfTtcbiJdfQ==
|
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
import { Point, Transaction } from '@cardano-ogmios/schema';
|
|
2
|
+
import { ChainSynchronizationClient } from '@cardano-ogmios/client/dist/ChainSynchronization';
|
|
3
|
+
import { WebSocketScanner } from '@rosen-bridge/abstract-scanner';
|
|
4
|
+
import { CardanoOgmiosConfig } from '../interfaces';
|
|
5
|
+
import { AbstractLogger } from '@rosen-bridge/abstract-logger';
|
|
6
|
+
declare class CardanoOgmiosScanner extends WebSocketScanner<Transaction> {
|
|
7
|
+
client: ChainSynchronizationClient;
|
|
8
|
+
initPoint: Point;
|
|
9
|
+
host: string;
|
|
10
|
+
port: number;
|
|
11
|
+
useTls: boolean;
|
|
12
|
+
private connected;
|
|
13
|
+
private stopped;
|
|
14
|
+
private reconnectionConfig;
|
|
15
|
+
constructor(config: CardanoOgmiosConfig, logger?: AbstractLogger);
|
|
16
|
+
/**
|
|
17
|
+
* process block when a block forked library call this function
|
|
18
|
+
* @param response: forked block
|
|
19
|
+
* @param requestNext: a function to tell node to send next block
|
|
20
|
+
*/
|
|
21
|
+
private rollBackward;
|
|
22
|
+
/**
|
|
23
|
+
* Step forward when new block arrived ogmios-client call this function
|
|
24
|
+
* @param response: new block
|
|
25
|
+
* @param requestNext: tell library that this block proceeds. pass next block when available
|
|
26
|
+
*/
|
|
27
|
+
private rollForward;
|
|
28
|
+
/**
|
|
29
|
+
* find intersect between stored blocks and blockchain.
|
|
30
|
+
* @param context: blockchain context
|
|
31
|
+
*/
|
|
32
|
+
private findIntersection;
|
|
33
|
+
/**
|
|
34
|
+
* Handles ogmios connection closure
|
|
35
|
+
*/
|
|
36
|
+
private connectionCloseHandler;
|
|
37
|
+
/**
|
|
38
|
+
* start scanner. first find fork point. fork all blocks and request updates from node.
|
|
39
|
+
*/
|
|
40
|
+
start: () => Promise<void>;
|
|
41
|
+
/**
|
|
42
|
+
* stop ws connection to node.
|
|
43
|
+
*/
|
|
44
|
+
stop: () => Promise<void>;
|
|
45
|
+
/**
|
|
46
|
+
* @returns The connection status of ogmios scanner
|
|
47
|
+
*/
|
|
48
|
+
getConnectionStatus: () => boolean;
|
|
49
|
+
}
|
|
50
|
+
export { CardanoOgmiosScanner };
|
|
51
|
+
//# sourceMappingURL=ogmios.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ogmios.d.ts","sourceRoot":"","sources":["../../lib/scanner/ogmios.ts"],"names":[],"mappings":"AAAA,OAAO,EAGL,KAAK,EAGL,WAAW,EACZ,MAAM,wBAAwB,CAAC;AAMhC,OAAO,EACL,0BAA0B,EAE3B,MAAM,kDAAkD,CAAC;AAC1D,OAAO,EAEL,gBAAgB,EACjB,MAAM,gCAAgC,CAAC;AACxC,OAAO,EAAE,mBAAmB,EAA4B,MAAM,eAAe,CAAC;AAC9E,OAAO,EAAE,cAAc,EAAE,MAAM,+BAA+B,CAAC;AAmB/D,cAAM,oBAAqB,SAAQ,gBAAgB,CAAC,WAAW,CAAC;IAC9D,MAAM,EAAE,0BAA0B,CAAC;IACnC,SAAS,EAAE,KAAK,CAAC;IACjB,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,EAAE,MAAM,CAAC;IACb,MAAM,EAAE,OAAO,CAAC;IAChB,OAAO,CAAC,SAAS,CAAU;IAC3B,OAAO,CAAC,OAAO,CAAU;IACzB,OAAO,CAAC,kBAAkB,CAA2B;gBAEzC,MAAM,EAAE,mBAAmB,EAAE,MAAM,CAAC,EAAE,cAAc;IAoBhE;;;;OAIG;IACH,OAAO,CAAC,YAAY,CAsBlB;IAEF;;;;OAIG;IACH,OAAO,CAAC,WAAW,CAuBjB;IAEF;;;OAGG;IACH,OAAO,CAAC,gBAAgB,CAkCtB;IAEF;;OAEG;IACH,OAAO,CAAC,sBAAsB,CA6B5B;IAEF;;OAEG;IACH,KAAK,QAAa,OAAO,CAAC,IAAI,CAAC,CAwB7B;IAEF;;OAEG;IACH,IAAI,QAAa,OAAO,CAAC,IAAI,CAAC,CAG5B;IAEF;;OAEG;IACH,mBAAmB,gBAAwB;CAC5C;AAED,OAAO,EAAE,oBAAoB,EAAE,CAAC"}
|
|
@@ -0,0 +1,172 @@
|
|
|
1
|
+
import { createInteractionContext, createChainSynchronizationClient, } from '@cardano-ogmios/client';
|
|
2
|
+
import { findIntersection, } from '@cardano-ogmios/client/dist/ChainSynchronization';
|
|
3
|
+
import { BlockDbAction, WebSocketScanner, } from '@rosen-bridge/abstract-scanner';
|
|
4
|
+
import { RECONNECTION_INITIAL_DELAY, RECONNECTION_MAX_DELAY, SLOT_SHELLY_NUMBER, RECONNECTION_MAX_ATTEMPTS, } from '../constants';
|
|
5
|
+
import { ExponentialBackoff, handleWhen, retry } from 'cockatiel';
|
|
6
|
+
class CardanoOgmiosScanner extends WebSocketScanner {
|
|
7
|
+
client;
|
|
8
|
+
initPoint;
|
|
9
|
+
host;
|
|
10
|
+
port;
|
|
11
|
+
useTls;
|
|
12
|
+
connected;
|
|
13
|
+
stopped;
|
|
14
|
+
reconnectionConfig;
|
|
15
|
+
constructor(config, logger) {
|
|
16
|
+
super('cardano', logger, config.maxTryBlock, config.suffix);
|
|
17
|
+
this.action = new BlockDbAction(config.dataSource, this.name(), logger);
|
|
18
|
+
this.host = config.nodeHostOrIp;
|
|
19
|
+
this.port = config.nodePort;
|
|
20
|
+
this.useTls = config.useTls ?? false;
|
|
21
|
+
this.connected = false;
|
|
22
|
+
this.initPoint = {
|
|
23
|
+
id: config.initialHash,
|
|
24
|
+
slot: config.initialSlot,
|
|
25
|
+
};
|
|
26
|
+
this.reconnectionConfig = {
|
|
27
|
+
initialDelay: config.reconnectionConfig?.initialDelay ?? RECONNECTION_INITIAL_DELAY,
|
|
28
|
+
maxDelay: config.reconnectionConfig?.maxDelay ?? RECONNECTION_MAX_DELAY,
|
|
29
|
+
maxAttempts: config.reconnectionConfig?.maxAttempts ?? RECONNECTION_MAX_ATTEMPTS,
|
|
30
|
+
};
|
|
31
|
+
}
|
|
32
|
+
/**
|
|
33
|
+
* process block when a block forked library call this function
|
|
34
|
+
* @param response: forked block
|
|
35
|
+
* @param requestNext: a function to tell node to send next block
|
|
36
|
+
*/
|
|
37
|
+
rollBackward = async (response, requestNext) => {
|
|
38
|
+
const hash = response.point.id;
|
|
39
|
+
const savedBlock = await this.action.getBlockWithHash(hash);
|
|
40
|
+
this.logger.debug(`Rolling backward to height ${savedBlock?.height} in scanner ${this.name()}`);
|
|
41
|
+
if (savedBlock) {
|
|
42
|
+
const block = {
|
|
43
|
+
hash: savedBlock.hash,
|
|
44
|
+
height: savedBlock.height,
|
|
45
|
+
parentHash: savedBlock.parentHash,
|
|
46
|
+
extra: savedBlock.extra,
|
|
47
|
+
timestamp: savedBlock.timestamp,
|
|
48
|
+
};
|
|
49
|
+
await this.stepBackward(block);
|
|
50
|
+
}
|
|
51
|
+
requestNext();
|
|
52
|
+
};
|
|
53
|
+
/**
|
|
54
|
+
* Step forward when new block arrived ogmios-client call this function
|
|
55
|
+
* @param response: new block
|
|
56
|
+
* @param requestNext: tell library that this block proceeds. pass next block when available
|
|
57
|
+
*/
|
|
58
|
+
rollForward = async (response, requestNext) => {
|
|
59
|
+
if (response.block.type === 'praos') {
|
|
60
|
+
const praosBlock = response.block;
|
|
61
|
+
this.logger.debug(`Queueing new block at height ${praosBlock.height} in scanner ${this.name()}`);
|
|
62
|
+
const block = {
|
|
63
|
+
hash: praosBlock.id,
|
|
64
|
+
height: praosBlock.height,
|
|
65
|
+
parentHash: praosBlock.ancestor,
|
|
66
|
+
extra: `${praosBlock.slot}`,
|
|
67
|
+
// Caution: In case of a hard fork and change in slot duration, this must change!
|
|
68
|
+
timestamp: praosBlock.slot + SLOT_SHELLY_NUMBER,
|
|
69
|
+
};
|
|
70
|
+
if (praosBlock.transactions)
|
|
71
|
+
await this.stepForward(block, praosBlock.transactions);
|
|
72
|
+
}
|
|
73
|
+
requestNext();
|
|
74
|
+
};
|
|
75
|
+
/**
|
|
76
|
+
* find intersect between stored blocks and blockchain.
|
|
77
|
+
* @param context: blockchain context
|
|
78
|
+
*/
|
|
79
|
+
findIntersection = async (context) => {
|
|
80
|
+
let count = 1;
|
|
81
|
+
let skip = 0;
|
|
82
|
+
while (count !== 0) {
|
|
83
|
+
try {
|
|
84
|
+
const release = await this.mutex.acquire();
|
|
85
|
+
const blocks = await this.action.getLastSavedBlocks(skip, count);
|
|
86
|
+
release();
|
|
87
|
+
if (blocks.length === 0)
|
|
88
|
+
count = 0;
|
|
89
|
+
const points = blocks.length > 0
|
|
90
|
+
? blocks.map((item) => ({
|
|
91
|
+
slot: item.extra ? parseInt(item.extra) : 0,
|
|
92
|
+
id: item.hash,
|
|
93
|
+
}))
|
|
94
|
+
: [this.initPoint];
|
|
95
|
+
const intersect = await findIntersection(context, points);
|
|
96
|
+
const intersectPoint = intersect.intersection;
|
|
97
|
+
let height = 0;
|
|
98
|
+
if (blocks.length) {
|
|
99
|
+
const foundedBlock = blocks.find((item) => (item.hash = intersectPoint.id));
|
|
100
|
+
if (foundedBlock) {
|
|
101
|
+
height = foundedBlock.height;
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
return { point: intersect.intersection, height: height };
|
|
105
|
+
}
|
|
106
|
+
catch {
|
|
107
|
+
skip += count;
|
|
108
|
+
count *= 2;
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
return undefined;
|
|
112
|
+
};
|
|
113
|
+
/**
|
|
114
|
+
* Handles ogmios connection closure
|
|
115
|
+
*/
|
|
116
|
+
connectionCloseHandler = async () => {
|
|
117
|
+
this.connected = false;
|
|
118
|
+
this.logger.warn('Ogmios connection closed');
|
|
119
|
+
if (this.stopped)
|
|
120
|
+
return;
|
|
121
|
+
const retryPolicy = retry(handleWhen((error) => {
|
|
122
|
+
this.logger.warn(`An error occurred while reconnecting to ogmios client: ${error}`);
|
|
123
|
+
return true;
|
|
124
|
+
}), {
|
|
125
|
+
maxAttempts: this.reconnectionConfig.maxAttempts,
|
|
126
|
+
backoff: new ExponentialBackoff({
|
|
127
|
+
maxDelay: this.reconnectionConfig.maxDelay,
|
|
128
|
+
initialDelay: this.reconnectionConfig.initialDelay,
|
|
129
|
+
}),
|
|
130
|
+
});
|
|
131
|
+
let trial = 0;
|
|
132
|
+
retryPolicy.execute(async () => {
|
|
133
|
+
trial++;
|
|
134
|
+
this.logger.debug(`Retrying to connect to ogmios client in trial step #${trial}`);
|
|
135
|
+
return await this.start();
|
|
136
|
+
});
|
|
137
|
+
};
|
|
138
|
+
/**
|
|
139
|
+
* start scanner. first find fork point. fork all blocks and request updates from node.
|
|
140
|
+
*/
|
|
141
|
+
start = async () => {
|
|
142
|
+
this.stopped = false;
|
|
143
|
+
const context = await createInteractionContext((err) => this.logger.error(`${err}`), this.connectionCloseHandler, { connection: { port: this.port, host: this.host, tls: this.useTls } });
|
|
144
|
+
const intersect = await this.findIntersection(context);
|
|
145
|
+
if (intersect) {
|
|
146
|
+
// find intersect then start from that point
|
|
147
|
+
this.client = await createChainSynchronizationClient(context, {
|
|
148
|
+
rollBackward: this.rollBackward,
|
|
149
|
+
rollForward: this.rollForward,
|
|
150
|
+
}, { sequential: true });
|
|
151
|
+
this.connected = true;
|
|
152
|
+
await this.forkBlock(intersect.height + 1);
|
|
153
|
+
await this.client.resume([intersect.point], 2);
|
|
154
|
+
}
|
|
155
|
+
else {
|
|
156
|
+
throw Error('Can not start scanner. initial block is invalid');
|
|
157
|
+
}
|
|
158
|
+
};
|
|
159
|
+
/**
|
|
160
|
+
* stop ws connection to node.
|
|
161
|
+
*/
|
|
162
|
+
stop = async () => {
|
|
163
|
+
this.stopped = true;
|
|
164
|
+
await this.client.shutdown();
|
|
165
|
+
};
|
|
166
|
+
/**
|
|
167
|
+
* @returns The connection status of ogmios scanner
|
|
168
|
+
*/
|
|
169
|
+
getConnectionStatus = () => this.connected;
|
|
170
|
+
}
|
|
171
|
+
export { CardanoOgmiosScanner };
|
|
172
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ogmios.js","sourceRoot":"","sources":["../../lib/scanner/ogmios.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Block,\n  BlockPraos,\n  Point,\n  PointOrOrigin,\n  TipOrOrigin,\n  Transaction,\n} from '@cardano-ogmios/schema';\nimport {\n  createInteractionContext,\n  createChainSynchronizationClient,\n  InteractionContext,\n} from '@cardano-ogmios/client';\nimport {\n  ChainSynchronizationClient,\n  findIntersection,\n} from '@cardano-ogmios/client/dist/ChainSynchronization';\nimport {\n  BlockDbAction,\n  WebSocketScanner,\n} from '@rosen-bridge/abstract-scanner';\nimport { CardanoOgmiosConfig, OgmiosReconnectionConfig } from '../interfaces';\nimport { AbstractLogger } from '@rosen-bridge/abstract-logger';\nimport {\n  RECONNECTION_INITIAL_DELAY,\n  RECONNECTION_MAX_DELAY,\n  SLOT_SHELLY_NUMBER,\n  RECONNECTION_MAX_ATTEMPTS,\n} from '../constants';\nimport { ExponentialBackoff, handleWhen, retry } from 'cockatiel';\n\ninterface BackwardResponse {\n  point: PointOrOrigin;\n  tip: TipOrOrigin;\n}\n\ninterface ForwardResponse {\n  block: Block;\n  tip: TipOrOrigin;\n}\n\nclass CardanoOgmiosScanner extends WebSocketScanner<Transaction> {\n  client: ChainSynchronizationClient;\n  initPoint: Point;\n  host: string;\n  port: number;\n  useTls: boolean;\n  private connected: boolean;\n  private stopped: boolean;\n  private reconnectionConfig: OgmiosReconnectionConfig;\n\n  constructor(config: CardanoOgmiosConfig, logger?: AbstractLogger) {\n    super('cardano', logger, config.maxTryBlock, config.suffix);\n    this.action = new BlockDbAction(config.dataSource, this.name(), logger);\n    this.host = config.nodeHostOrIp;\n    this.port = config.nodePort;\n    this.useTls = config.useTls ?? false;\n    this.connected = false;\n    this.initPoint = {\n      id: config.initialHash,\n      slot: config.initialSlot,\n    };\n    this.reconnectionConfig = {\n      initialDelay:\n        config.reconnectionConfig?.initialDelay ?? RECONNECTION_INITIAL_DELAY,\n      maxDelay: config.reconnectionConfig?.maxDelay ?? RECONNECTION_MAX_DELAY,\n      maxAttempts:\n        config.reconnectionConfig?.maxAttempts ?? RECONNECTION_MAX_ATTEMPTS,\n    };\n  }\n\n  /**\n   * process block when a block forked library call this function\n   * @param response: forked block\n   * @param requestNext: a function to tell node to send next block\n   */\n  private rollBackward = async (\n    response: BackwardResponse,\n    requestNext: () => void\n  ) => {\n    const hash = (response.point as Point).id;\n    const savedBlock = await this.action.getBlockWithHash(hash);\n    this.logger.debug(\n      `Rolling backward to height ${\n        savedBlock?.height\n      } in scanner ${this.name()}`\n    );\n    if (savedBlock) {\n      const block = {\n        hash: savedBlock.hash,\n        height: savedBlock.height,\n        parentHash: savedBlock.parentHash,\n        extra: savedBlock.extra,\n        timestamp: savedBlock.timestamp,\n      };\n      await this.stepBackward(block);\n    }\n    requestNext();\n  };\n\n  /**\n   * Step forward when new block arrived ogmios-client call this function\n   * @param response: new block\n   * @param requestNext: tell library that this block proceeds. pass next block when available\n   */\n  private rollForward = async (\n    response: ForwardResponse,\n    requestNext: () => void\n  ) => {\n    if (response.block.type === 'praos') {\n      const praosBlock = response.block as BlockPraos;\n      this.logger.debug(\n        `Queueing new block at height ${\n          praosBlock.height\n        } in scanner ${this.name()}`\n      );\n      const block = {\n        hash: praosBlock.id,\n        height: praosBlock.height,\n        parentHash: praosBlock.ancestor,\n        extra: `${praosBlock.slot}`,\n        // Caution: In case of a hard fork and change in slot duration, this must change!\n        timestamp: praosBlock.slot + SLOT_SHELLY_NUMBER,\n      };\n      if (praosBlock.transactions)\n        await this.stepForward(block, praosBlock.transactions);\n    }\n    requestNext();\n  };\n\n  /**\n   * find intersect between stored blocks and blockchain.\n   * @param context: blockchain context\n   */\n  private findIntersection = async (context: InteractionContext) => {\n    let count = 1;\n    let skip = 0;\n    while (count !== 0) {\n      try {\n        const release = await this.mutex.acquire();\n        const blocks = await this.action.getLastSavedBlocks(skip, count);\n        release();\n        if (blocks.length === 0) count = 0;\n        const points =\n          blocks.length > 0\n            ? blocks.map((item) => ({\n                slot: item.extra ? parseInt(item.extra) : 0,\n                id: item.hash,\n              }))\n            : [this.initPoint];\n        const intersect = await findIntersection(context, points);\n        const intersectPoint = intersect.intersection as Point;\n        let height = 0;\n        if (blocks.length) {\n          const foundedBlock = blocks.find(\n            (item) => (item.hash = intersectPoint.id)\n          );\n          if (foundedBlock) {\n            height = foundedBlock.height;\n          }\n        }\n        return { point: intersect.intersection as Point, height: height };\n      } catch {\n        skip += count;\n        count *= 2;\n      }\n    }\n    return undefined;\n  };\n\n  /**\n   * Handles ogmios connection closure\n   */\n  private connectionCloseHandler = async () => {\n    this.connected = false;\n    this.logger.warn('Ogmios connection closed');\n    if (this.stopped) return;\n\n    const retryPolicy = retry(\n      handleWhen((error: Error) => {\n        this.logger.warn(\n          `An error occurred while reconnecting to ogmios client: ${error}`\n        );\n        return true;\n      }),\n      {\n        maxAttempts: this.reconnectionConfig.maxAttempts,\n        backoff: new ExponentialBackoff({\n          maxDelay: this.reconnectionConfig.maxDelay,\n          initialDelay: this.reconnectionConfig.initialDelay,\n        }),\n      }\n    );\n\n    let trial = 0;\n    retryPolicy.execute(async () => {\n      trial++;\n      this.logger.debug(\n        `Retrying to connect to ogmios client in trial step #${trial}`\n      );\n      return await this.start();\n    });\n  };\n\n  /**\n   * start scanner. first find fork point. fork all blocks and request updates from node.\n   */\n  start = async (): Promise<void> => {\n    this.stopped = false;\n    const context: InteractionContext = await createInteractionContext(\n      (err) => this.logger.error(`${err}`),\n      this.connectionCloseHandler,\n      { connection: { port: this.port, host: this.host, tls: this.useTls } }\n    );\n    const intersect = await this.findIntersection(context);\n    if (intersect) {\n      // find intersect then start from that point\n      this.client = await createChainSynchronizationClient(\n        context,\n        {\n          rollBackward: this.rollBackward,\n          rollForward: this.rollForward,\n        },\n        { sequential: true }\n      );\n      this.connected = true;\n      await this.forkBlock(intersect.height + 1);\n      await this.client.resume([intersect.point], 2);\n    } else {\n      throw Error('Can not start scanner. initial block is invalid');\n    }\n  };\n\n  /**\n   * stop ws connection to node.\n   */\n  stop = async (): Promise<void> => {\n    this.stopped = true;\n    await this.client.shutdown();\n  };\n\n  /**\n   * @returns The connection status of ogmios scanner\n   */\n  getConnectionStatus = () => this.connected;\n}\n\nexport { CardanoOgmiosScanner };\n"]}
|
package/package.json
ADDED
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "@rosen-bridge/cardano-scanner",
|
|
3
|
+
"version": "0.1.0-52fc0239",
|
|
4
|
+
"description": "A Cardano chain scanner.",
|
|
5
|
+
"repository": {
|
|
6
|
+
"type": "git",
|
|
7
|
+
"url": "git+https://github.com/rosen-bridge/scanner.git"
|
|
8
|
+
},
|
|
9
|
+
"license": "MIT",
|
|
10
|
+
"author": "Rosen Team",
|
|
11
|
+
"type": "module",
|
|
12
|
+
"main": "dist/index.js",
|
|
13
|
+
"types": "dist/index.d.ts",
|
|
14
|
+
"files": [
|
|
15
|
+
"dist",
|
|
16
|
+
"CHANGELOG.md"
|
|
17
|
+
],
|
|
18
|
+
"scripts": {
|
|
19
|
+
"build": "tsc --build tsconfig.build.json",
|
|
20
|
+
"lint": "eslint --fix . && npm run prettify",
|
|
21
|
+
"prettify": "prettier --write . --ignore-path ./.gitignore",
|
|
22
|
+
"release": "npm run test -- --run && npm run build && npm publish --access public",
|
|
23
|
+
"type-check": "tsc --noEmit"
|
|
24
|
+
},
|
|
25
|
+
"devDependencies": {
|
|
26
|
+
"@types/node": "^22.18.0",
|
|
27
|
+
"tsx": "^4.19.4",
|
|
28
|
+
"typescript": "^5.3.3",
|
|
29
|
+
"@rosen-bridge/extended-typeorm": "^0.2.1"
|
|
30
|
+
},
|
|
31
|
+
"engines": {
|
|
32
|
+
"node": ">=22.18.0"
|
|
33
|
+
},
|
|
34
|
+
"dependencies": {
|
|
35
|
+
"@rosen-bridge/abstract-scanner": "^0.1.0-52fc0239",
|
|
36
|
+
"@rosen-bridge/scanner-interfaces": "^0.1.1-52fc0239",
|
|
37
|
+
"@rosen-bridge/rate-limited-axios": "^0.2.1",
|
|
38
|
+
"@rosen-bridge/abstract-logger": "^2.0.1",
|
|
39
|
+
"@blockfrost/blockfrost-js": "^6.0.0",
|
|
40
|
+
"cockatiel": "^3.2.1",
|
|
41
|
+
"@blockfrost/openapi": "0.1.78",
|
|
42
|
+
"@emurgo/cardano-serialization-lib-nodejs": "^13.2.1",
|
|
43
|
+
"@cardano-ogmios/client": "^6.6.1",
|
|
44
|
+
"@cardano-ogmios/schema": "^6.6.1",
|
|
45
|
+
"@rosen-bridge/json-bigint": "^0.1.0"
|
|
46
|
+
}
|
|
47
|
+
}
|