-
Notifications
You must be signed in to change notification settings - Fork 69
/
Copy pathruncli.ts
131 lines (102 loc) · 3.46 KB
/
runcli.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
// Copyright 2018-2025 @polkadot/json-serve authors & contributors
// SPDX-License-Identifier: Apache-2.0
import '@polkadot/api-augment';
import type { DeriveStakingElected } from '@polkadot/api-derive/types';
import type { Balance, BlockNumber, Header } from '@polkadot/types/interfaces';
import Koa from 'koa';
import koaRoute from 'koa-route';
import yargs from 'yargs';
import { hideBin } from 'yargs/helpers';
import { ApiPromise, WsProvider } from '@polkadot/api';
import { BN } from '@polkadot/util';
interface ArgV {
port: number;
ws: string
}
const { port, ws } = yargs(hideBin(process.argv))
.options({
port: {
default: 9099,
description: 'The HTTP port to listen on',
required: true,
type: 'number'
},
ws: {
description: 'The endpoint to connect to, e.g. wss://kusama-rpc.polkadot.io',
required: true,
type: 'string'
}
})
.argv as ArgV;
let currentBlockNumber: BlockNumber | undefined;
let currentTimestamp: Date = new Date();
let tokenDecimals = 12;
let totalBonded = new BN(0);
let totalInsurance = new BN(0);
function balanceFormat (balance: BN): string {
if (balance.gtn(0)) {
const strValue = balance.toString().padStart(tokenDecimals + 1, '0');
const postfix = strValue.slice(-1 * tokenDecimals);
const prefix = strValue.slice(0, strValue.length - tokenDecimals);
return `${prefix}.${postfix}`;
}
return '0.0';
}
function percentageFormat (top: BN, bottom: BN): string {
if (top.gtn(0) && bottom.gtn(0)) {
return (top.muln(10000).div(bottom).toNumber() / 100).toFixed(2);
}
return '0.00';
}
function onElectedInfo (info: DeriveStakingElected): void {
totalBonded = info.info.reduce((totalBonded, { exposurePaged }): BN => {
return totalBonded.add((exposurePaged.isSome && exposurePaged?.unwrap().pageTotal.unwrap()) || new BN(0));
}, new BN(0));
}
function onNewHead (header: Header): void {
if (currentBlockNumber && header.number.eq(currentBlockNumber.toBn())) {
return;
}
currentBlockNumber = header.number.unwrap();
currentTimestamp = new Date();
console.log(`#${currentBlockNumber.toString()} received at ${currentTimestamp.toString()}`);
}
function onTotalInsurance (_totalInsurance: Balance): void {
totalInsurance = _totalInsurance;
}
function jsonApi (ctx: Koa.Context): void {
switch (ctx.query['q']) {
case 'bondedpercentage':
ctx.body = percentageFormat(totalBonded, totalInsurance);
break;
case 'totalbonded':
ctx.body = balanceFormat(totalBonded);
break;
case 'totalcoins':
ctx.body = balanceFormat(totalInsurance);
break;
default:
ctx.body = "Error: Invalid query, expected '?q=<type>', one of 'bondedpercentage', 'totalbonded', 'totalcoins'";
break;
}
}
async function main (): Promise<void> {
const app = new Koa();
app.use(koaRoute.all('/', jsonApi));
app.listen(port);
const provider = new WsProvider(ws);
const api = await ApiPromise.create({ provider });
const chainProperties = await api.rpc.system.properties();
tokenDecimals = chainProperties.tokenDecimals.unwrapOr([new BN(12)])[0].toNumber();
await api.rpc.chain.subscribeNewHeads(onNewHead);
await api.query.balances.totalIssuance(onTotalInsurance);
await api.derive.staking.electedInfo(undefined, undefined, onElectedInfo);
}
process.on('unhandledRejection', (error): void => {
console.error(error);
process.exit(1);
});
main().catch((error): void => {
console.error(error);
process.exit(1);
});