Skip to content

Commit ede02f2

Browse files
committed
dashboard, watcher: refactor
1 parent c5c9f5d commit ede02f2

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

51 files changed

+327
-429
lines changed

cloud_functions/src/alarmMissingVaas.ts

+9-8
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,12 @@ import { CHAIN_ID_TO_NAME, ChainId, ChainName } from '@certusone/wormhole-sdk';
22
import { MissingVaasByChain, commonGetMissingVaas } from './getMissingVaas';
33
import { assertEnvironmentVariable, formatAndSendToSlack, isVAASigned } from './utils';
44
import { ObservedMessage, ReobserveInfo, SlackInfo } from './types';
5-
import { NETWORK, explorerBlock, explorerTx } from '@wormhole-foundation/wormhole-monitor-common';
5+
import {
6+
Environment,
7+
getEnvironment,
8+
explorerBlock,
9+
explorerTx,
10+
} from '@wormhole-foundation/wormhole-monitor-common';
611
import { Firestore } from 'firebase-admin/firestore';
712

813
interface EnqueuedVAAResponse {
@@ -34,10 +39,7 @@ interface GovernedVAA {
3439
// The key is the vaaKey
3540
type GovernedVAAMap = Map<string, GovernedVAA>;
3641

37-
const network: NETWORK =
38-
assertEnvironmentVariable('NETWORK').toLowerCase() === 'mainnet'
39-
? NETWORK.MAINNET
40-
: NETWORK.TESTNET;
42+
const network: Environment = getEnvironment();
4143

4244
export async function alarmMissingVaas(req: any, res: any) {
4345
res.set('Access-Control-Allow-Origin', '*');
@@ -127,7 +129,7 @@ export async function alarmMissingVaas(req: any, res: any) {
127129
txhash: msg.txHash,
128130
vaaKey: vaaKey,
129131
});
130-
if (network === NETWORK.MAINNET) {
132+
if (network === 'mainnet') {
131133
alarmSlackInfo.msg = formatMessage(msg);
132134
await formatAndSendToSlack(alarmSlackInfo);
133135
}
@@ -323,7 +325,6 @@ async function alarmOldBlockTimes(latestTimes: LatestTimeByChain): Promise<void>
323325
bannerTxt: 'Wormhole Missing VAA Alarm',
324326
msg: '',
325327
};
326-
const envNetwork: string = network === NETWORK.MAINNET ? 'mainnet' : 'testnet';
327328

328329
let alarmsToStore: AlarmedChainTime[] = [];
329330
// Read in the already alarmed chains.
@@ -351,7 +352,7 @@ async function alarmOldBlockTimes(latestTimes: LatestTimeByChain): Promise<void>
351352
const chainTime: Date = new Date(latestTime);
352353
const cName: string = CHAIN_ID_TO_NAME[chainId] as ChainName;
353354
const deltaTime: number = (now.getTime() - chainTime.getTime()) / (1000 * 60 * 60 * 24);
354-
alarmSlackInfo.msg = `*Chain:* ${cName}(${chainId})\nThe ${envNetwork} watcher is behind by ${deltaTime} days.`;
355+
alarmSlackInfo.msg = `*Chain:* ${cName}(${chainId})\nThe ${network} watcher is behind by ${deltaTime} days.`;
355356
await formatAndSendToSlack(alarmSlackInfo);
356357
alarmsToStore.push({ chain: chainId, alarmTime: now.toISOString() });
357358
}

common/src/consts.ts

+16-10
Original file line numberDiff line numberDiff line change
@@ -31,15 +31,19 @@ import {
3131
coalesceChainName,
3232
} from '@certusone/wormhole-sdk';
3333

34-
export enum NETWORK {
35-
MAINNET,
36-
TESTNET,
37-
}
34+
export type Environment = 'mainnet' | 'testnet' | 'devnet';
35+
export type Network = {
36+
env: Environment;
37+
endpoint: string;
38+
name: string;
39+
logo: string;
40+
type: 'guardian' | 'cloudfunction';
41+
};
3842

3943
export const INITIAL_DEPLOYMENT_BLOCK_BY_NETWORK_AND_CHAIN: {
40-
[key in NETWORK]: { [key in ChainName]?: string };
44+
[key in Environment]: { [key in ChainName]?: string };
4145
} = {
42-
[NETWORK.MAINNET]: {
46+
['mainnet']: {
4347
ethereum: '12959638',
4448
terra: '4810000', // not sure exactly but this should be before the first known message
4549
bsc: '9745450',
@@ -66,7 +70,7 @@ export const INITIAL_DEPLOYMENT_BLOCK_BY_NETWORK_AND_CHAIN: {
6670
sei: '238594',
6771
wormchain: '4510119', // https://bigdipper.live/wormhole/transactions/4D861F1BE86325D227FA006CA2745BBC6748AF5B5E0811DE536D02792928472A },
6872
},
69-
[NETWORK.TESTNET]: {
73+
['testnet']: {
7074
ethereum: '0',
7175
terra: '0',
7276
bsc: '0',
@@ -93,6 +97,7 @@ export const INITIAL_DEPLOYMENT_BLOCK_BY_NETWORK_AND_CHAIN: {
9397
sei: '0',
9498
wormchain: '0',
9599
},
100+
['devnet']: {},
96101
};
97102

98103
export const TOKEN_BRIDGE_EMITTERS: { [key in ChainName]?: string } = {
@@ -179,8 +184,8 @@ export type CHAIN_INFO = {
179184
explorerStem: string;
180185
};
181186

182-
export const CHAIN_INFO_MAP: { [key in NETWORK]: { [key: string]: CHAIN_INFO } } = {
183-
[NETWORK.MAINNET]: {
187+
export const CHAIN_INFO_MAP: { [key in Environment]: { [key: string]: CHAIN_INFO } } = {
188+
['mainnet']: {
184189
1: {
185190
name: 'solana',
186191
evm: false,
@@ -371,7 +376,7 @@ export const CHAIN_INFO_MAP: { [key in NETWORK]: { [key: string]: CHAIN_INFO } }
371376
explorerStem: '',
372377
},
373378
},
374-
[NETWORK.TESTNET]: {
379+
['testnet']: {
375380
// The chains not, currently, supported on testnet are commented out.
376381
1: {
377382
name: 'solana',
@@ -556,6 +561,7 @@ export const CHAIN_INFO_MAP: { [key in NETWORK]: { [key: string]: CHAIN_INFO } }
556561
// explorerStem: '',
557562
// },
558563
},
564+
['devnet']: {},
559565
};
560566

561567
export const JUMP_GUARDIAN_ADDRESS = '58cc3ae5c097b213ce3c81979e1b9f9570746aa5';

common/src/explorer.ts

+6-6
Original file line numberDiff line numberDiff line change
@@ -31,10 +31,10 @@ import {
3131
CHAIN_ID_XPLA,
3232
} from '@certusone/wormhole-sdk';
3333
import { base58 } from 'ethers/lib/utils';
34-
import { CHAIN_INFO_MAP, NETWORK } from './consts';
34+
import { CHAIN_INFO_MAP, Environment } from './consts';
3535

36-
export const explorerBlock = (network: NETWORK, chainId: ChainId, block: string) =>
37-
network === NETWORK.MAINNET
36+
export const explorerBlock = (network: Environment, chainId: ChainId, block: string) =>
37+
network === 'mainnet'
3838
? chainId === CHAIN_ID_ETH
3939
? `https://etherscan.io/block/${block}`
4040
: chainId === CHAIN_ID_BSC
@@ -120,8 +120,8 @@ export const explorerBlock = (network: NETWORK, chainId: ChainId, block: string)
120120
? `https://goerli.basescan.org/block/${block}`
121121
: '';
122122

123-
export const explorerTx = (network: NETWORK, chainId: ChainId, tx: string) =>
124-
network === NETWORK.MAINNET
123+
export const explorerTx = (network: Environment, chainId: ChainId, tx: string) =>
124+
network === 'mainnet'
125125
? chainId === CHAIN_ID_ETH
126126
? `https://etherscan.io/tx/${tx}`
127127
: chainId === CHAIN_ID_BSC
@@ -212,7 +212,7 @@ export const explorerVaa = (network: string, key: string) =>
212212
? `https://wormholescan.io/#/tx/${key}`
213213
: `https://wormholescan.io/#/tx/${key}?network=TESTNET`;
214214

215-
export const getExplorerTxHash = (network: NETWORK, chain: ChainId, txHash: string) => {
215+
export const getExplorerTxHash = (network: Environment, chain: ChainId, txHash: string) => {
216216
let explorerTxHash = '';
217217
if (isCosmWasmChain(chain)) {
218218
explorerTxHash = txHash.slice(2);

common/src/utils.ts

+10
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,5 @@
1+
import { Environment } from './consts';
2+
13
export async function sleep(timeout: number) {
24
return new Promise((resolve) => setTimeout(resolve, timeout));
35
}
@@ -13,3 +15,11 @@ export const padUint64 = (s: string): string => s.padStart(MAX_UINT_64.length, '
1315
// make a bigtable row key for the `signedVAAs` table
1416
export const makeSignedVAAsRowKey = (chain: number, emitter: string, sequence: string): string =>
1517
`${padUint16(chain.toString())}/${emitter}/${padUint64(sequence)}`;
18+
19+
export function getEnvironment(): Environment {
20+
const network: string = assertEnvironmentVariable('NETWORK').toLowerCase();
21+
if (network === 'mainnet' || network === 'testnet' || network === 'devnet') {
22+
return network;
23+
}
24+
throw new Error(`Unknown network: ${network}`);
25+
}

dashboard/src/components/Accountant.tsx

+7-25
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,6 @@ import {
77
Card,
88
InputAdornment,
99
LinearProgress,
10-
Link,
1110
TextField,
1211
Tooltip,
1312
Typography,
@@ -29,16 +28,13 @@ import useGetAccountantPendingTransfers, {
2928
} from '../hooks/useGetAccountantPendingTransfers';
3029
import chainIdToName from '../utils/chainIdToName';
3130
import { CHAIN_ICON_MAP, GUARDIAN_SET_3 } from '../utils/consts';
32-
import {
33-
CHAIN_INFO_MAP,
34-
explorerTx,
35-
getExplorerTxHash,
36-
} from '@wormhole-foundation/wormhole-monitor-common';
31+
import { CHAIN_INFO_MAP } from '@wormhole-foundation/wormhole-monitor-common';
3732
import CollapsibleSection from './CollapsibleSection';
3833
import Table from './Table';
3934
import useTokenData, { TokenDataEntry } from '../hooks/useTokenData';
4035
import numeral from 'numeral';
41-
import { GetNetworkFromEnv } from '../utils/GetNetworkFromEnv';
36+
import { useCurrentEnvironment } from '../contexts/NetworkContext';
37+
import { ExplorerTxHash } from './ExplorerTxHash';
4238

4339
type PendingTransferForAcct = PendingTransfer & { isEnqueuedInGov: boolean };
4440
type AccountWithTokenData = Account & {
@@ -143,23 +139,9 @@ const pendingTransferColumns = [
143139
}),
144140
pendingTransferColumnHelper.accessor('data.0.tx_hash', {
145141
header: () => 'Tx',
146-
cell: (info) => {
147-
const tx = '0x' + Buffer.from(info.getValue(), 'base64').toString('hex');
148-
const chain = info.row.original.key.emitter_chain;
149-
const network = GetNetworkFromEnv();
150-
const chainInfo = CHAIN_INFO_MAP[network][chain];
151-
if (!chainInfo) return tx;
152-
const txHash = getExplorerTxHash(network, chainInfo.chainId, tx);
153-
return (
154-
<Link
155-
href={explorerTx(network, chainInfo.chainId, txHash)}
156-
target="_blank"
157-
rel="noopener noreferrer"
158-
>
159-
{txHash}
160-
</Link>
161-
);
162-
},
142+
cell: (info) => (
143+
<ExplorerTxHash chain={info.row.original.key.emitter_chain} rawTxHash={info.getValue()} />
144+
),
163145
}),
164146
pendingTransferColumnHelper.accessor('data.0.signatures', {
165147
header: () => 'Signatures',
@@ -430,7 +412,7 @@ function Accountant({ governorInfo }: { governorInfo: CloudGovernorInfo }) {
430412
}, {} as { [chainId: number]: number }),
431413
[pendingTransferInfo]
432414
);
433-
const network = GetNetworkFromEnv();
415+
const network = useCurrentEnvironment();
434416
return (
435417
<CollapsibleSection
436418
defaultExpanded={false}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,38 @@
1+
import { CHAIN_INFO_MAP } from '@wormhole-foundation/wormhole-monitor-common';
2+
import { useCurrentEnvironment } from '../contexts/NetworkContext';
3+
import {
4+
CHAIN_ID_ALGORAND,
5+
CHAIN_ID_NEAR,
6+
CHAIN_ID_TERRA2,
7+
ChainId,
8+
tryHexToNativeAssetString,
9+
} from '@certusone/wormhole-sdk';
10+
import { Link } from '@mui/material';
11+
12+
export function ExplorerAssetURL({ chain, assetAddr }: { chain: number; assetAddr: string }) {
13+
const network = useCurrentEnvironment();
14+
const chainInfo = CHAIN_INFO_MAP[network][chain];
15+
if (!chainInfo) return <>{assetAddr}</>;
16+
const chainId: ChainId = chainInfo.chainId;
17+
var tokenAddress: string = '';
18+
if (chainId === CHAIN_ID_ALGORAND || chainId === CHAIN_ID_NEAR || chainId === CHAIN_ID_TERRA2) {
19+
return <>{assetAddr}</>;
20+
}
21+
try {
22+
tokenAddress = tryHexToNativeAssetString(
23+
assetAddr.slice(2),
24+
CHAIN_INFO_MAP[network][chain]?.chainId
25+
);
26+
} catch (e) {
27+
console.log(e);
28+
tokenAddress = assetAddr;
29+
}
30+
31+
const explorerString = chainInfo?.explorerStem;
32+
const url = `${explorerString}/address/${tokenAddress}`;
33+
return (
34+
<Link href={url} target="_blank" rel="noopener noreferrer">
35+
{tokenAddress}
36+
</Link>
37+
);
38+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
import {
2+
CHAIN_INFO_MAP,
3+
explorerTx,
4+
getExplorerTxHash,
5+
} from '@wormhole-foundation/wormhole-monitor-common';
6+
import { useCurrentEnvironment } from '../contexts/NetworkContext';
7+
import { Link } from '@mui/material';
8+
9+
export function ExplorerTxHash({ chain, rawTxHash }: { chain: number; rawTxHash: string }) {
10+
const network = useCurrentEnvironment();
11+
const chainInfo = CHAIN_INFO_MAP[network][chain];
12+
if (!chainInfo) return <>{rawTxHash}</>;
13+
const txHash = getExplorerTxHash(network, chainInfo.chainId, rawTxHash);
14+
return (
15+
<Link
16+
href={explorerTx(network, chainInfo.chainId, txHash)}
17+
target="_blank"
18+
rel="noopener noreferrer"
19+
>
20+
{txHash}
21+
</Link>
22+
);
23+
}

0 commit comments

Comments
 (0)