web: initial commit

This commit is contained in:
Evan Gray 2022-09-17 02:30:18 +00:00 committed by Evan Gray
parent a7e406fd17
commit cb7b6b4a00
56 changed files with 51597 additions and 0 deletions

24
web/.gitignore vendored Normal file
View File

@ -0,0 +1,24 @@
# See https://help.github.com/articles/ignoring-files/ for more about ignoring files.
# dependencies
/node_modules
/.pnp
.pnp.js
# testing
/coverage
# production
/build
# misc
.DS_Store
.env
.env.local
.env.development.local
.env.test.local
.env.production.local
npm-debug.log*
yarn-debug.log*
yarn-error.log*

67
web/config-overrides.js Normal file
View File

@ -0,0 +1,67 @@
const { ProvidePlugin } = require("webpack");
module.exports = function override(config, env) {
return {
...config,
module: {
...config.module,
rules: [
...config.module.rules,
{
test: /\.js$/,
enforce: "pre",
use: ["source-map-loader"],
},
{
test: /\.wasm$/,
type: "webassembly/async",
},
],
},
plugins: [
...config.plugins,
new ProvidePlugin({
Buffer: ["buffer", "Buffer"],
process: "process/browser",
}),
],
resolve: {
...config.resolve,
fallback: {
assert: "assert",
buffer: "buffer",
console: "console-browserify",
constants: "constants-browserify",
crypto: "crypto-browserify",
domain: "domain-browser",
events: "events",
fs: false,
http: "stream-http",
https: "https-browserify",
os: "os-browserify/browser",
path: "path-browserify",
punycode: "punycode",
process: "process/browser",
querystring: "querystring-es3",
stream: "stream-browserify",
_stream_duplex: "readable-stream/duplex",
_stream_passthrough: "readable-stream/passthrough",
_stream_readable: "readable-stream/readable",
_stream_transform: "readable-stream/transform",
_stream_writable: "readable-stream/writable",
string_decoder: "string_decoder",
sys: "util",
timers: "timers-browserify",
tty: "tty-browserify",
url: "url",
util: "util",
vm: "vm-browserify",
zlib: "browserify-zlib",
},
},
experiments: {
asyncWebAssembly: true,
},
ignoreWarnings: [/Failed to parse source map/],
};
};

48585
web/package-lock.json generated Normal file

File diff suppressed because it is too large Load Diff

81
web/package.json Normal file
View File

@ -0,0 +1,81 @@
{
"name": "wormhole-explorer",
"version": "0.1.0",
"private": true,
"license": "Apache-2.0",
"proxy": "http://localhost:4000",
"dependencies": {
"@certusone/wormhole-sdk": "^0.6.0",
"@certusone/wormhole-sdk-proto-web": "^0.0.4",
"@emotion/react": "^11.10.0",
"@emotion/styled": "^11.10.0",
"@metaplex/js": "^4.12.0",
"@mui/icons-material": "^5.8.4",
"@mui/material": "^5.9.3",
"@tanstack/react-table": "^8.5.11",
"@terra-money/terra.js": "^3.1.3",
"@testing-library/jest-dom": "^5.16.5",
"@testing-library/react": "^13.3.0",
"@testing-library/user-event": "^13.5.0",
"@types/jest": "^28.1.6",
"@types/node": "^18.6.4",
"@types/react": "^18.0.15",
"@types/react-dom": "^18.0.6",
"axios": "^0.27.2",
"coingecko-api": "^1.0.10",
"dotenv": "^16.0.2",
"ethers": "^5.6.9",
"long": "^5.2.0",
"numeral": "^2.0.6",
"react": "^18.2.0",
"react-dom": "^18.2.0",
"react-scripts": "5.0.1",
"typescript": "^4.7.4",
"web-vitals": "^2.1.4",
"web3": "^1.7.5"
},
"scripts": {
"start": "react-app-rewired start",
"build": "react-app-rewired build",
"test": "react-app-rewired test",
"eject": "react-app-rewired eject"
},
"eslintConfig": {
"extends": [
"react-app",
"react-app/jest"
]
},
"browserslist": {
"production": [
">0.2%",
"not dead",
"not op_mini all"
],
"development": [
"last 1 chrome version",
"last 1 firefox version",
"last 1 safari version"
]
},
"devDependencies": {
"@types/numeral": "^2.0.2",
"browserify-zlib": "^0.2.0",
"buffer": "^6.0.3",
"console-browserify": "^1.2.0",
"constants-browserify": "^1.0.0",
"crypto-browserify": "^3.12.0",
"https-browserify": "^1.0.0",
"os-browserify": "^0.3.0",
"path-browserify": "^1.0.1",
"prettier": "^2.3.2",
"process": "^0.11.10",
"react-app-rewired": "^2.2.1",
"readable-stream": "^3.6.0",
"stream-browserify": "^3.0.0",
"timers-browserify": "^2.0.12",
"tty-browserify": "^0.0.1",
"util": "^0.12.4",
"vm-browserify": "^1.1.2"
}
}

Binary file not shown.

After

Width:  |  Height:  |  Size: 18 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 51 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 17 KiB

View File

@ -0,0 +1,9 @@
<?xml version="1.0" encoding="utf-8"?>
<browserconfig>
<msapplication>
<tile>
<square150x150logo src="https://certusone.github.io/wormhole-dashboard/mstile-150x150.png"/>
<TileColor>#da532c</TileColor>
</tile>
</msapplication>
</browserconfig>

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.0 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.7 KiB

BIN
web/public/favicon.ico Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 12 KiB

43
web/public/index.html Normal file
View File

@ -0,0 +1,43 @@
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8" />
<meta name="viewport" content="width=device-width, initial-scale=1" />
<meta name="description" content="A dashboard for monitoring Wormhole" />
<meta name="theme-color" content="#ffffff" />
<link
rel="apple-touch-icon"
sizes="180x180"
href="%PUBLIC_URL%/apple-touch-icon.png"
/>
<link
rel="icon"
type="image/png"
sizes="32x32"
href="%PUBLIC_URL%/favicon-32x32.png"
/>
<link
rel="icon"
type="image/png"
sizes="16x16"
href="%PUBLIC_URL%/favicon-16x16.png"
/>
<link rel="manifest" href="%PUBLIC_URL%/site.webmanifest" />
<link
rel="mask-icon"
href="%PUBLIC_URL%/safari-pinned-tab.svg"
color="#5bbad5"
/>
<link rel="shortcut icon" href="%PUBLIC_URL%/favicon.ico" />
<meta name="msapplication-TileColor" content="#da532c" />
<meta
name="msapplication-config"
content="%PUBLIC_URL%/browserconfig.xml"
/>
<title>Wormhole Dashboard</title>
</head>
<body>
<noscript>You need to enable JavaScript to run this app.</noscript>
<div id="root"></div>
</body>
</html>

Binary file not shown.

After

Width:  |  Height:  |  Size: 13 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 13 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 14 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 28 KiB

BIN
web/public/mstile-70x70.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 8.9 KiB

3
web/public/robots.txt Normal file
View File

@ -0,0 +1,3 @@
# https://www.robotstxt.org/robotstxt.html
User-agent: *
Disallow:

View File

@ -0,0 +1,64 @@
<?xml version="1.0" standalone="no"?>
<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 20010904//EN"
"http://www.w3.org/TR/2001/REC-SVG-20010904/DTD/svg10.dtd">
<svg version="1.0" xmlns="http://www.w3.org/2000/svg"
width="700.000000pt" height="700.000000pt" viewBox="0 0 700.000000 700.000000"
preserveAspectRatio="xMidYMid meet">
<metadata>
Created by potrace 1.14, written by Peter Selinger 2001-2017
</metadata>
<g transform="translate(0.000000,700.000000) scale(0.100000,-0.100000)"
fill="#000000" stroke="none">
<path d="M2 5343 c0 -912 3 -1642 5 -1623 20 210 27 270 44 370 118 697 437
1318 944 1839 536 551 1184 896 1915 1020 100 17 160 24 370 44 19 2 -711 5
-1622 5 l-1658 2 2 -1657z"/>
<path d="M3720 6993 c19 -1 84 -8 145 -13 566 -55 1147 -271 1620 -603 171
-120 294 -225 476 -406 225 -226 376 -411 518 -638 l51 -83 -293 0 -292 0
-105 106 c-207 208 -435 372 -705 504 -305 150 -565 226 -905 266 -175 20
-515 15 -680 -11 -591 -91 -1101 -346 -1511 -755 -220 -218 -371 -428 -508
-700 -167 -335 -255 -656 -281 -1036 l-12 -159 -86 -87 c-48 -49 -91 -88 -96
-88 -16 0 1 432 24 582 94 624 376 1182 825 1634 434 437 985 718 1615 823
697 117 1426 -39 2015 -431 167 -111 270 -194 425 -343 81 -78 111 -103 85
-70 -395 503 -908 870 -1502 1075 -351 122 -666 174 -1043 173 -438 0 -808
-69 -1200 -225 -738 -293 -1341 -866 -1705 -1620 -320 -665 -397 -1395 -223
-2126 l32 -133 -109 -109 c-60 -61 -113 -110 -116 -110 -9 0 -63 189 -94 325
-30 131 -62 342 -74 480 -5 60 -9 -563 -10 -1557 l-1 -1658 1658 2 c911 0
1641 3 1622 5 -210 20 -270 27 -370 44 -526 89 -1040 308 -1466 624 -357 265
-698 632 -922 991 l-50 81 61 7 c43 4 87 18 142 45 44 22 80 39 81 38 242
-371 457 -612 764 -855 452 -356 1022 -591 1610 -664 139 -17 602 -17 740 0
588 74 1114 285 1576 633 130 98 144 120 32 50 -340 -214 -709 -349 -1128
-412 -118 -18 -191 -22 -395 -23 -334 -1 -533 25 -822 108 -252 73 -515 190
-725 323 -249 158 -533 413 -718 643 -152 191 -301 433 -387 631 l-35 82 68
68 c38 38 71 69 74 69 3 0 27 -52 54 -115 134 -316 335 -606 602 -865 674
-655 1637 -906 2559 -669 347 90 717 276 1005 506 178 142 395 376 522 563
l50 75 -43 -50 c-410 -472 -933 -743 -1558 -805 -643 -64 -1337 180 -1810 635
-362 348 -592 795 -677 1314 l-17 103 27 26 26 26 53 -52 c51 -50 55 -58 66
-125 106 -611 460 -1139 987 -1472 616 -390 1415 -430 2079 -105 621 303 1053
874 1175 1553 11 61 20 123 20 138 0 25 4 28 60 37 163 29 284 140 325 297 12
47 15 122 15 364 0 167 2 302 4 300 7 -7 46 -278 55 -384 5 -60 9 563 10 1558
l1 1657 -1657 -2 c-912 0 -1642 -3 -1623 -5z m967 -1308 c349 -46 695 -176
993 -374 l85 -56 -124 -3 -125 -3 -160 80 c-242 121 -476 192 -730 222 -146
17 -473 7 -611 -20 -230 -44 -414 -106 -610 -206 -446 -229 -812 -627 -1001
-1090 -18 -44 -37 -81 -43 -83 -5 -2 -22 3 -38 12 -16 8 -45 17 -65 21 l-37 7
15 42 c37 96 157 329 221 426 102 154 181 250 317 385 355 353 797 569 1306
640 135 18 462 18 607 0z m1941 -646 c62 -35 105 -96 121 -170 14 -66 15
-1226 1 -1300 -21 -109 -97 -180 -201 -187 -78 -6 -136 16 -187 72 -22 24 -45
61 -51 82 -7 25 -11 176 -11 404 l0 365 -1167 -1167 c-911 -911 -1178 -1172
-1213 -1187 -55 -25 -152 -27 -206 -5 -29 11 -245 221 -802 777 l-762 762
-758 -757 c-512 -511 -771 -763 -799 -777 -56 -28 -149 -28 -202 -2 -96 49
-148 159 -121 258 12 45 63 99 869 906 661 662 867 863 905 882 64 32 151 34
216 4 35 -15 218 -193 800 -774 l755 -755 1075 1075 1075 1075 -364 0 c-257 0
-376 4 -405 12 -54 16 -128 90 -144 144 -33 111 13 225 110 273 43 21 49 21
728 21 l685 0 53 -31z m-1710 14 c-2 -5 -10 -25 -18 -45 l-14 -38 -71 0 c-431
0 -900 -215 -1189 -545 -251 -288 -405 -685 -406 -1050 l0 -70 -46 47 -45 47
6 68 c33 368 135 643 345 928 127 172 254 289 435 402 219 137 487 227 746
252 112 11 262 13 257 4z m1600 -1858 c7 -6 -19 -159 -45 -260 -79 -317 -276
-646 -504 -843 -455 -394 -1055 -525 -1632 -357 -91 26 -257 93 -257 104 0 3
16 19 35 34 l35 28 88 -34 c498 -197 1040 -138 1489 160 370 246 628 656 694
1101 l13 84 40 -6 c22 -4 42 -9 44 -11z"/>
<path d="M6990 3268 c0 -70 -16 -205 -41 -356 -87 -522 -308 -1043 -622 -1465
-230 -310 -538 -611 -842 -824 -473 -332 -1054 -548 -1620 -603 -61 -5 -126
-12 -145 -13 -19 -2 711 -5 1623 -5 l1657 -2 0 1660 c0 913 -2 1660 -5 1660
-3 0 -5 -24 -5 -52z"/>
</g>
</svg>

After

Width:  |  Height:  |  Size: 4.1 KiB

View File

@ -0,0 +1,19 @@
{
"name": "",
"short_name": "",
"icons": [
{
"src": "https://certusone.github.io/wormhole-dashboard/android-chrome-192x192.png",
"sizes": "192x192",
"type": "image/png"
},
{
"src": "https://certusone.github.io/wormhole-dashboard/android-chrome-512x512.png",
"sizes": "512x512",
"type": "image/png"
}
],
"theme_color": "#ffffff",
"background_color": "#ffffff",
"display": "standalone"
}

8
web/src/App.test.js Normal file
View File

@ -0,0 +1,8 @@
import { render, screen } from '@testing-library/react';
import App from './App';
test('renders learn react link', () => {
render(<App />);
const linkElement = screen.getByText(/learn react/i);
expect(linkElement).toBeInTheDocument();
});

40
web/src/App.tsx Normal file
View File

@ -0,0 +1,40 @@
import { GitHub } from "@mui/icons-material";
import {
AppBar,
Box,
CssBaseline,
IconButton,
Toolbar,
Typography,
} from "@mui/material";
import CustomThemeProvider from "./components/CustomThemeProvider";
import Main from "./components/Main";
import NetworkSelector from "./components/NetworkSelector";
import Settings from "./components/Settings";
import { NetworkContextProvider } from "./contexts/NetworkContext";
import { SettingsContextProvider } from "./contexts/SettingsContext";
import WormholeStatsIcon from "./icons/WormholeStatsIcon";
function App() {
return (
<SettingsContextProvider>
<CustomThemeProvider>
<CssBaseline />
<NetworkContextProvider>
<AppBar position="static">
<Toolbar variant="dense">
<Box pr={1} display="flex" alignItems="center">
<WormholeStatsIcon />
</Box>
<Typography variant="h6">Explorer</Typography>
<Box flexGrow={1} />
</Toolbar>
</AppBar>
<Main />
</NetworkContextProvider>
</CustomThemeProvider>
</SettingsContextProvider>
);
}
export default App;

View File

@ -0,0 +1,265 @@
import { GetLastHeartbeatsResponse_Entry } from "@certusone/wormhole-sdk-proto-web/lib/cjs/publicrpc/v1/publicrpc";
import {
ErrorOutline,
InfoOutlined,
WarningAmberOutlined,
} from "@mui/icons-material";
import {
Alert,
AlertColor,
Box,
Link,
List,
ListItem,
ListItemIcon,
ListItemText,
Tooltip,
Typography,
} from "@mui/material";
import { useMemo } from "react";
import { ChainIdToHeartbeats } from "../hooks/useChainHeartbeats";
import useLatestRelease from "../hooks/useLatestRelease";
import chainIdToName from "../utils/chainIdToName";
import CollapsibleSection from "./CollapsibleSection";
export const BEHIND_DIFF = 1000;
type AlertEntry = {
severity: AlertColor;
text: string;
};
const alertSeverityOrder: AlertColor[] = [
"error",
"warning",
"success",
"info",
];
function chainDownAlerts(
heartbeats: GetLastHeartbeatsResponse_Entry[],
chainIdsToHeartbeats: ChainIdToHeartbeats
): AlertEntry[] {
const downChains: { [chainId: string]: string[] } = {};
Object.entries(chainIdsToHeartbeats).forEach(([chainId, chainHeartbeats]) => {
// Search for known guardians without heartbeats
const missingGuardians = heartbeats.filter(
(guardianHeartbeat) =>
chainHeartbeats.findIndex(
(chainHeartbeat) =>
chainHeartbeat.guardian === guardianHeartbeat.p2pNodeAddr
) === -1
);
missingGuardians.forEach((guardianHeartbeat) => {
if (!downChains[chainId]) {
downChains[chainId] = [];
}
downChains[chainId].push(guardianHeartbeat.rawHeartbeat?.nodeName || "");
});
// Search for guardians with heartbeats but who are not picking up a height
// Could be disconnected or erroring post initial checks
// Track highest height to check for lagging guardians
let highest = BigInt(0);
chainHeartbeats.forEach((chainHeartbeat) => {
const height = BigInt(chainHeartbeat.network.height);
if (height > highest) {
highest = height;
}
if (chainHeartbeat.network.height === "0") {
if (!downChains[chainId]) {
downChains[chainId] = [];
}
downChains[chainId].push(chainHeartbeat.name);
}
});
// Search for guardians which are lagging significantly behind
chainHeartbeats.forEach((chainHeartbeat) => {
if (chainHeartbeat.network.height !== "0") {
const height = BigInt(chainHeartbeat.network.height);
const diff = highest - height;
if (diff > BEHIND_DIFF) {
if (!downChains[chainId]) {
downChains[chainId] = [];
}
downChains[chainId].push(chainHeartbeat.name);
}
}
});
});
return Object.entries(downChains).map(([chainId, names]) => ({
severity: names.length >= 7 ? "error" : "warning",
text: `${names.length} guardian${names.length > 1 ? "s" : ""} [${names.join(
", "
)}] ${names.length > 1 ? "are" : "is"} down on ${chainIdToName(
Number(chainId)
)} (${chainId})!`,
}));
}
const releaseChecker = (
release: string | null,
heartbeats: GetLastHeartbeatsResponse_Entry[]
): AlertEntry[] =>
release === null
? []
: heartbeats
.filter((heartbeat) => heartbeat.rawHeartbeat?.version !== release)
.map((heartbeat) => ({
severity: "info",
text: `${heartbeat.rawHeartbeat?.nodeName} is not running the latest release (${heartbeat.rawHeartbeat?.version} !== ${release})`,
}));
function Alerts({
heartbeats,
chainIdsToHeartbeats,
}: {
heartbeats: GetLastHeartbeatsResponse_Entry[];
chainIdsToHeartbeats: ChainIdToHeartbeats;
}) {
const latestRelease = useLatestRelease();
const alerts = useMemo(() => {
const alerts: AlertEntry[] = [
...chainDownAlerts(heartbeats, chainIdsToHeartbeats),
...releaseChecker(latestRelease, heartbeats),
];
return alerts.sort((a, b) =>
alertSeverityOrder.indexOf(a.severity) <
alertSeverityOrder.indexOf(b.severity)
? -1
: alertSeverityOrder.indexOf(a.severity) >
alertSeverityOrder.indexOf(b.severity)
? 1
: 0
);
}, [latestRelease, heartbeats, chainIdsToHeartbeats]);
const numErrors = useMemo(
() => alerts.filter((alert) => alert.severity === "error").length,
[alerts]
);
const numInfos = useMemo(
() => alerts.filter((alert) => alert.severity === "info").length,
[alerts]
);
const numSuccess = useMemo(
() => alerts.filter((alert) => alert.severity === "success").length,
[alerts]
);
const numWarnings = useMemo(
() => alerts.filter((alert) => alert.severity === "warning").length,
[alerts]
);
return (
<CollapsibleSection
header={
<Box
sx={{
display: "flex",
alignItems: "center",
paddingRight: 1,
}}
>
<Tooltip
title={
<>
<Typography variant="body1">
This section shows alerts for the following conditions:
</Typography>
<List dense>
<ListItem>
<ListItemIcon>
<ErrorOutline color="error" />
</ListItemIcon>
<ListItemText
primary="Chains with a quorum of guardians down"
secondary={`A guardian is considered down if it is
reporting a height of 0, more than ${BEHIND_DIFF} behind the highest height, or missing from the list of
heartbeats`}
/>
</ListItem>
<ListItem>
<ListItemIcon>
<WarningAmberOutlined color="warning" />
</ListItemIcon>
<ListItemText
primary="Chains with one or more guardians down"
secondary={`A guardian is considered down if it is
reporting a height of 0, more than ${BEHIND_DIFF} behind the highest height, or missing from the list of
heartbeats`}
/>
</ListItem>
<ListItem>
<ListItemIcon>
<InfoOutlined color="info" />
</ListItemIcon>
<ListItemText
primary="Guardians not running the latest release"
secondary={
<>
The guardian version is compared to the latest release
from{" "}
<Link
href="https://github.com/wormhole-foundation/wormhole/releases"
target="_blank"
rel="noopener noreferrer"
>
https://github.com/wormhole-foundation/wormhole/releases
</Link>
</>
}
/>
</ListItem>
</List>
</>
}
componentsProps={{ tooltip: { sx: { maxWidth: "100%" } } }}
>
<Box>
Alerts
<InfoOutlined sx={{ fontSize: ".8em", ml: 0.5 }} />
</Box>
</Tooltip>
<Box flexGrow={1} />
{numInfos > 0 ? (
<>
<InfoOutlined color="info" sx={{ ml: 2 }} />
<Typography variant="h6" component="strong" sx={{ ml: 0.5 }}>
{numInfos}
</Typography>
</>
) : null}
{numSuccess > 0 ? (
<>
<InfoOutlined color="success" sx={{ ml: 2 }} />
<Typography variant="h6" component="strong" sx={{ ml: 0.5 }}>
{numSuccess}
</Typography>
</>
) : null}
{numWarnings > 0 ? (
<>
<WarningAmberOutlined color="warning" sx={{ ml: 2 }} />
<Typography variant="h6" component="strong" sx={{ ml: 0.5 }}>
{numWarnings}
</Typography>
</>
) : null}
{numErrors > 0 ? (
<>
<ErrorOutline color="error" sx={{ ml: 2 }} />
<Typography variant="h6" component="strong" sx={{ ml: 0.5 }}>
{numErrors}
</Typography>
</>
) : null}
</Box>
}
>
{alerts.map((alert) => (
<Alert key={alert.text} severity={alert.severity}>
{alert.text}
</Alert>
))}
</CollapsibleSection>
);
}
export default Alerts;

View File

@ -0,0 +1,111 @@
import { Box, Card, Grid, Typography } from "@mui/material";
import {
createColumnHelper,
getCoreRowModel,
getSortedRowModel,
SortingState,
useReactTable,
} from "@tanstack/react-table";
import { useCallback, useMemo, useState } from "react";
import {
ChainIdToHeartbeats,
HeartbeatInfo,
} from "../hooks/useChainHeartbeats";
import chainIdToName from "../utils/chainIdToName";
import { BEHIND_DIFF } from "./Alerts";
import Table from "./Table";
const columnHelper = createColumnHelper<HeartbeatInfo>();
const columns = [
columnHelper.accessor("name", {
header: () => "Guardian",
cell: (info) => (
<Typography variant="body2" noWrap>
{info.getValue()}
</Typography>
),
sortingFn: `text`,
}),
columnHelper.accessor("network.height", {
header: () => "Height",
}),
columnHelper.accessor("network.contractAddress", {
header: () => "Contract",
}),
];
function Chain({
chainId,
heartbeats,
}: {
chainId: string;
heartbeats: HeartbeatInfo[];
}) {
const [sorting, setSorting] = useState<SortingState>([]);
const table = useReactTable({
columns,
data: heartbeats,
state: {
sorting,
},
getRowId: (heartbeat) => heartbeat.guardian,
getCoreRowModel: getCoreRowModel(),
getSortedRowModel: getSortedRowModel(),
onSortingChange: setSorting,
});
const highest = useMemo(() => {
let highest = BigInt(0);
heartbeats.forEach((heartbeat) => {
const height = BigInt(heartbeat.network.height);
if (height > highest) {
highest = height;
}
});
return highest;
}, [heartbeats]);
const conditionalRowStyle = useCallback(
(heartbeat: HeartbeatInfo) =>
heartbeat.network.height === "0" ||
highest - BigInt(heartbeat.network.height) > BEHIND_DIFF
? { backgroundColor: "rgba(100,0,0,.2)" }
: {},
[highest]
);
return (
<Grid key={chainId} item xs={12} lg={6}>
<Card>
<Box p={2}>
<Typography variant="h5" gutterBottom>
{chainIdToName(Number(chainId))} ({chainId})
</Typography>
<Typography>Guardians Listed: {heartbeats.length}</Typography>
</Box>
<Table<HeartbeatInfo>
table={table}
conditionalRowStyle={conditionalRowStyle}
/>
</Card>
</Grid>
);
}
function Chains({
chainIdsToHeartbeats,
}: {
chainIdsToHeartbeats: ChainIdToHeartbeats;
}) {
return (
<Grid container spacing={2}>
{Object.keys(chainIdsToHeartbeats).map((chainId) => (
<Chain
key={chainId}
chainId={chainId}
heartbeats={chainIdsToHeartbeats[Number(chainId)]}
/>
))}
</Grid>
);
}
export default Chains;

View File

@ -0,0 +1,44 @@
import { ExpandMore } from "@mui/icons-material";
import {
Accordion,
AccordionDetails,
AccordionSummary,
Typography,
} from "@mui/material";
import { ReactNode } from "react";
function CollapsibleSection({
header,
children,
}: {
header: ReactNode;
children: ReactNode;
}) {
return (
<Accordion
defaultExpanded
disableGutters
sx={{
background: "transparent",
my: 0.5,
"&.Mui-expanded:first-of-type": {
marginTop: 0.5,
},
"&:not(:last-child)": {
borderBottom: 0,
},
"&:before": {
display: "none",
},
}}
>
<AccordionSummary expandIcon={<ExpandMore />}>
<Typography variant="h5" sx={{ width: "100%" }}>
{header}
</Typography>
</AccordionSummary>
<AccordionDetails>{children}</AccordionDetails>
</Accordion>
);
}
export default CollapsibleSection;

View File

@ -0,0 +1,110 @@
import {
Box,
createTheme,
responsiveFontSizes,
ThemeProvider,
} from "@mui/material";
import { grey } from "@mui/material/colors";
import { ReactNode, useCallback, useEffect, useMemo, useState } from "react";
import { useSettingsContext } from "../contexts/SettingsContext";
const mediaQueryList =
window.matchMedia && window.matchMedia("(prefers-color-scheme: dark)");
function CustomThemeProvider({ children }: { children: ReactNode }) {
const {
settings: { theme: themePreference, backgroundOpacity, backgroundUrl },
} = useSettingsContext();
const [userPrefersDark, setUserPrefersDark] = useState<boolean>(
mediaQueryList && mediaQueryList.matches ? true : false
);
const handleUserPreferenceChange = useCallback(
(event: MediaQueryListEvent) => {
setUserPrefersDark(event.matches ? true : false);
},
[]
);
useEffect(() => {
if (themePreference === "auto") {
mediaQueryList.addEventListener("change", handleUserPreferenceChange);
return () => {
mediaQueryList.removeEventListener(
"change",
handleUserPreferenceChange
);
};
}
}, [themePreference, handleUserPreferenceChange]);
const mode = "light";
// themePreference === "dark" ||
// (themePreference === "auto" && userPrefersDark)
// ? "dark"
// : "light";
const theme = useMemo(
() =>
responsiveFontSizes(
createTheme({
palette: {
mode,
},
components: {
MuiCssBaseline: {
styleOverrides: {
body: {
overflowY: "scroll",
},
"*": {
scrollbarWidth: "thin",
scrollbarColor:
// mode === "dark"
// ? `${grey[700]} ${grey[900]}`
// :
`${grey[400]} rgb(255,255,255)`,
},
"*::-webkit-scrollbar": {
width: "8px",
height: "8px",
backgroundColor:
// mode === "dark" ? grey[900] :
"rgb(255,255,255)",
},
"*::-webkit-scrollbar-thumb": {
// mode === "dark" ? grey[700] :
backgroundColor: grey[400],
borderRadius: "4px",
},
"*::-webkit-scrollbar-corner": {
// this hides an annoying white box which appears when both scrollbars are present
backgroundColor: "transparent",
},
},
},
},
})
),
[mode]
);
return (
<ThemeProvider theme={theme}>
{children}
{backgroundUrl && (
<Box
sx={{
backgroundImage: `url(${backgroundUrl})`,
backgroundPosition: "center",
backgroundSize: "cover",
opacity: backgroundOpacity || 0.1,
position: "fixed",
top: 0,
left: 0,
right: 0,
bottom: 0,
zIndex: -1,
}}
/>
)}
</ThemeProvider>
);
}
export default CustomThemeProvider;

View File

@ -0,0 +1,55 @@
import { ChainId, getSignedVAA } from "@certusone/wormhole-sdk";
import { GovernorGetEnqueuedVAAsResponse_Entry } from "@certusone/wormhole-sdk-proto-web/lib/cjs/publicrpc/v1/publicrpc";
import { useEffect, useState } from "react";
import { useNetworkContext } from "../contexts/NetworkContext";
const VAA_CHECK_TIMEOUT = 60000;
function EnqueuedVAAChecker({
vaa: { emitterAddress, emitterChain, sequence },
}: {
vaa: GovernorGetEnqueuedVAAsResponse_Entry;
}) {
const {
currentNetwork: { endpoint },
} = useNetworkContext();
const [vaaHasQuorum, setVaaHasQuorum] = useState<boolean | null>(null);
useEffect(() => {
let cancelled = false;
(async () => {
while (!cancelled) {
setVaaHasQuorum(null);
let result = false;
try {
const response = await getSignedVAA(
endpoint,
emitterChain as ChainId,
emitterAddress,
sequence
);
if (!!response.vaaBytes) result = true;
} catch (e) {}
if (!cancelled) {
setVaaHasQuorum(result);
if (result) {
cancelled = true;
return;
}
await new Promise((resolve) =>
setTimeout(resolve, VAA_CHECK_TIMEOUT)
);
}
}
})();
return () => {
cancelled = true;
};
}, [endpoint, emitterChain, emitterAddress, sequence]);
return (
<span role="img">
{vaaHasQuorum === null ? "⏳" : vaaHasQuorum ? "✅" : "❌"}
</span>
);
}
export default EnqueuedVAAChecker;

View File

@ -0,0 +1,320 @@
import {
GovernorGetAvailableNotionalByChainResponse_Entry,
GovernorGetEnqueuedVAAsResponse_Entry,
GovernorGetTokenListResponse_Entry,
} from "@certusone/wormhole-sdk-proto-web/lib/cjs/publicrpc/v1/publicrpc";
import { ExpandMore } from "@mui/icons-material";
import {
Accordion,
AccordionDetails,
AccordionSummary,
Box,
Card,
LinearProgress,
Link,
Tooltip,
Typography,
} from "@mui/material";
import {
createColumnHelper,
getCoreRowModel,
getSortedRowModel,
SortingState,
useReactTable,
} from "@tanstack/react-table";
import { useMemo, useState } from "react";
import useGovernorInfo from "../hooks/useGovernorInfo";
import chainIdToName from "../utils/chainIdToName";
import Table from "./Table";
import numeral from "numeral";
import EnqueuedVAAChecker from "./EnqueuedVAAChecker";
import { CHAIN_INFO_MAP } from "../utils/consts";
import {
ChainId,
CHAIN_ID_ALGORAND,
CHAIN_ID_NEAR,
CHAIN_ID_TERRA2,
isEVMChain,
tryHexToNativeAssetString,
tryHexToNativeString,
} from "@certusone/wormhole-sdk";
import useSymbolInfo from "../hooks/useSymbolInfo";
const calculatePercent = (
notional: GovernorGetAvailableNotionalByChainResponse_Entry
): number => {
try {
return (
((Number(notional.notionalLimit) -
Number(notional.remainingAvailableNotional)) /
Number(notional.notionalLimit)) *
100
);
} catch (e) {
return 0;
}
};
const notionalColumnHelper =
createColumnHelper<GovernorGetAvailableNotionalByChainResponse_Entry>();
const notionalColumns = [
notionalColumnHelper.accessor("chainId", {
header: () => "Chain",
cell: (info) => `${chainIdToName(info.getValue())} (${info.getValue()})`,
sortingFn: `text`,
}),
notionalColumnHelper.accessor("notionalLimit", {
header: () => <Box order="1">Limit</Box>,
cell: (info) => (
<Box textAlign="right">${numeral(info.getValue()).format("0,0")}</Box>
),
}),
notionalColumnHelper.accessor("bigTransactionSize", {
header: () => <Box order="1">Big Transaction</Box>,
cell: (info) => (
<Box textAlign="right">${numeral(info.getValue()).format("0,0")}</Box>
),
}),
notionalColumnHelper.accessor("remainingAvailableNotional", {
header: () => <Box order="1">Remaining</Box>,
cell: (info) => (
<Box textAlign="right">${numeral(info.getValue()).format("0,0")}</Box>
),
}),
notionalColumnHelper.accessor(calculatePercent, {
id: "progress",
header: () => "Progress",
cell: (info) => (
<Tooltip title={info.getValue()} arrow>
<LinearProgress
variant="determinate"
value={info.getValue()}
color={
info.getValue() > 80
? "error"
: info.getValue() > 50
? "warning"
: "success"
}
/>
</Tooltip>
),
}),
];
const enqueuedColumnHelper =
createColumnHelper<GovernorGetEnqueuedVAAsResponse_Entry>();
const enqueuedColumns = [
enqueuedColumnHelper.accessor("emitterChain", {
header: () => "Chain",
cell: (info) => `${chainIdToName(info.getValue())} (${info.getValue()})`,
sortingFn: `text`,
}),
enqueuedColumnHelper.accessor("emitterAddress", {
header: () => "Emitter",
}),
enqueuedColumnHelper.accessor("sequence", {
header: () => "Sequence",
cell: (info) => (
<Link
href={`https://wormhole-v2-mainnet-api.certus.one/v1/signed_vaa/${info.row.original.emitterChain}/${info.row.original.emitterAddress}/${info.row.original.sequence}`}
target="_blank"
rel="noopener noreferrer"
>
{info.getValue()}
</Link>
),
}),
enqueuedColumnHelper.display({
id: "hasQuorum",
header: () => "Has Quorum?",
cell: (info) => <EnqueuedVAAChecker vaa={info.row.original} />,
}),
enqueuedColumnHelper.accessor("txHash", {
header: () => "Transaction Hash",
cell: (info) => {
const chain = info.row.original.emitterChain;
const chainInfo = CHAIN_INFO_MAP[chain];
var txHash: string = "";
if (!isEVMChain(chainInfo.chainId)) {
txHash = tryHexToNativeString(
info.getValue().slice(2),
CHAIN_INFO_MAP[chain].chainId
);
} else {
txHash = info.getValue();
}
const explorerString = chainInfo.explorerStem;
const url = `${explorerString}/tx/${txHash}`;
return (
<Link href={url} target="_blank" rel="noopener noreferrer">
{txHash}
</Link>
);
},
}),
enqueuedColumnHelper.accessor("releaseTime", {
header: () => "Release Time",
cell: (info) => new Date(info.getValue() * 1000).toLocaleString(),
}),
enqueuedColumnHelper.accessor("notionalValue", {
header: () => <Box order="1">Notional Value</Box>,
cell: (info) => (
<Box textAlign="right">${numeral(info.getValue()).format("0,0")}</Box>
),
}),
];
interface GovernorGetTokenListResponse_Entry_ext
extends GovernorGetTokenListResponse_Entry {
symbol: string;
}
const tokenColumnHelper =
createColumnHelper<GovernorGetTokenListResponse_Entry_ext>();
const tokenColumns = [
tokenColumnHelper.accessor("originChainId", {
header: () => "Chain",
cell: (info) => `${chainIdToName(info.getValue())} (${info.getValue()})`,
sortingFn: `text`,
}),
tokenColumnHelper.accessor("originAddress", {
header: () => "Token",
cell: (info) => {
const chain = info.row.original.originChainId;
const chainInfo = CHAIN_INFO_MAP[chain];
const chainId: ChainId = chainInfo.chainId;
var tokenAddress: string = "";
if (
chainId === CHAIN_ID_ALGORAND ||
chainId === CHAIN_ID_NEAR ||
chainId === CHAIN_ID_TERRA2
) {
return info.getValue();
}
try {
tokenAddress = tryHexToNativeAssetString(
info.getValue().slice(2),
CHAIN_INFO_MAP[chain]?.chainId
);
} catch (e) {
console.log(e);
tokenAddress = info.getValue();
}
const explorerString = chainInfo?.explorerStem;
const url = `${explorerString}/address/${tokenAddress}`;
return (
<Link href={url} target="_blank" rel="noopener noreferrer">
{tokenAddress}
</Link>
);
},
}),
tokenColumnHelper.display({
id: "Symbol",
header: () => "Symbol",
cell: (info) => `${info.row.original?.symbol}`,
}),
tokenColumnHelper.accessor("price", {
header: () => <Box order="1">Price</Box>,
cell: (info) => (
<Box textAlign="right">
${numeral(info.getValue()).format("0,0.0000")}
</Box>
),
}),
];
function Governor() {
const governorInfo = useGovernorInfo();
const tokenSymbols = useSymbolInfo(governorInfo.tokens);
// TODO: governorInfo.tokens triggers updates to displayTokens, not tokenSymbols
// Should fix this
const displayTokens = useMemo(
() =>
governorInfo.tokens.map((tk) => ({
...tk,
symbol:
tokenSymbols.get([tk.originChainId, tk.originAddress].join("_"))
?.symbol || "",
})),
[governorInfo.tokens, tokenSymbols]
);
const [notionalSorting, setNotionalSorting] = useState<SortingState>([]);
const notionalTable = useReactTable({
columns: notionalColumns,
data: governorInfo.notionals,
state: {
sorting: notionalSorting,
},
getRowId: (notional) => notional.chainId.toString(),
getCoreRowModel: getCoreRowModel(),
getSortedRowModel: getSortedRowModel(),
onSortingChange: setNotionalSorting,
});
const [enqueuedSorting, setEnqueuedSorting] = useState<SortingState>([]);
const enqueuedTable = useReactTable({
columns: enqueuedColumns,
data: governorInfo.enqueued,
state: {
sorting: enqueuedSorting,
},
getRowId: (vaa) => JSON.stringify(vaa),
getCoreRowModel: getCoreRowModel(),
getSortedRowModel: getSortedRowModel(),
onSortingChange: setEnqueuedSorting,
});
const [tokenSorting, setTokenSorting] = useState<SortingState>([]);
const tokenTable = useReactTable({
columns: tokenColumns,
data: displayTokens,
state: {
sorting: tokenSorting,
},
getRowId: (token) => `${token.originChainId}_${token.originAddress}`,
getCoreRowModel: getCoreRowModel(),
getSortedRowModel: getSortedRowModel(),
onSortingChange: setTokenSorting,
});
return (
<>
<Box mb={2}>
<Card>
<Table<GovernorGetAvailableNotionalByChainResponse_Entry>
table={notionalTable}
/>
</Card>
</Box>
<Box my={2}>
<Card>
<Table<GovernorGetEnqueuedVAAsResponse_Entry> table={enqueuedTable} />
{governorInfo.enqueued.length === 0 ? (
<Typography variant="body2" sx={{ py: 1, textAlign: "center" }}>
No enqueued VAAs
</Typography>
) : null}
</Card>
</Box>
<Box mt={2}>
<Card>
<Accordion>
<AccordionSummary expandIcon={<ExpandMore />}>
<Typography>Tokens ({governorInfo.tokens.length})</Typography>
</AccordionSummary>
<AccordionDetails>
<Table<GovernorGetTokenListResponse_Entry_ext>
table={tokenTable}
/>
</AccordionDetails>
</Accordion>
</Card>
</Box>
</>
);
}
export default Governor;

View File

@ -0,0 +1,69 @@
import { Card } from "@mui/material";
import {
createColumnHelper,
getCoreRowModel,
getSortedRowModel,
SortingState,
useReactTable,
} from "@tanstack/react-table";
import { useState } from "react";
import { HeartbeatResponse } from "../hooks/useHeartbeats";
import longToDate from "../utils/longToDate";
import Table from "./Table";
const columnHelper = createColumnHelper<HeartbeatResponse>();
const columns = [
columnHelper.accessor("nodename", {
header: () => "Guardian",
sortingFn: `text`,
}),
columnHelper.accessor("version", {
header: () => "Version",
}),
columnHelper.accessor("features", {
header: () => "Features",
cell: (info) => {
const value = info.getValue();
return value && value.length > 0 ? value.join(", ") : "none";
},
}),
columnHelper.accessor("counter", {
header: () => "Counter",
}),
columnHelper.accessor("boottimestamp", {
header: () => "Boot",
cell: (info) =>
info.getValue() ? longToDate(info.getValue()).toLocaleString() : null,
}),
columnHelper.accessor("timestamp", {
header: () => "Timestamp",
cell: (info) =>
info.getValue() ? longToDate(info.getValue()).toLocaleString() : null,
}),
columnHelper.accessor("guardianaddr", {
header: () => "Address",
}),
];
function Guardians({ heartbeats }: { heartbeats: HeartbeatResponse[] }) {
const [sorting, setSorting] = useState<SortingState>([]);
const table = useReactTable({
columns,
data: heartbeats,
state: {
sorting,
},
getRowId: (heartbeat) => heartbeat.guardianaddr,
getCoreRowModel: getCoreRowModel(),
getSortedRowModel: getSortedRowModel(),
onSortingChange: setSorting,
});
return (
<Card>
<Table<HeartbeatResponse> table={table} />
</Card>
);
}
export default Guardians;

View File

@ -0,0 +1,107 @@
import { Card } from "@mui/material";
import {
createColumnHelper,
getCoreRowModel,
getExpandedRowModel,
getSortedRowModel,
Row,
SortingState,
useReactTable,
} from "@tanstack/react-table";
import { ReactNode, useState } from "react";
import useLatestVAAs, { VAAsResponse } from "../hooks/useLatestVAAs";
import Table from "./Table";
import { _parseVAAAlgorand } from "@certusone/wormhole-sdk/lib/esm/algorand/Algorand";
import { BigNumber } from "ethers";
import { ChainId, tryHexToNativeString } from "@certusone/wormhole-sdk";
const columnHelper = createColumnHelper<VAAsResponse>();
const columns = [
columnHelper.display({
id: "_expand",
cell: ({ row }) =>
row.getCanExpand() ? (
<button
{...{
onClick: row.getToggleExpandedHandler(),
style: { cursor: "pointer" },
}}
>
{row.getIsExpanded() ? "👇" : "👉"}
</button>
) : null,
}),
columnHelper.accessor("_id", {
id: "chain",
header: () => "Chain",
cell: (info) => info.getValue().split("/")[0],
}),
columnHelper.accessor("_id", {
id: "emitter",
header: () => "Emitter",
cell: (info) => info.getValue().split("/")[1],
}),
columnHelper.accessor("_id", {
id: "sequence",
header: () => "Sequence",
cell: (info) => info.getValue().split("/")[2],
}),
];
function VAADetails({ row }: { row: Row<VAAsResponse> }): ReactNode {
const parsedVaa = _parseVAAAlgorand(
new Uint8Array(Buffer.from(row.original.vaa, "base64"))
);
let token = parsedVaa.Contract;
// FromChain is a misnomer - actually OriginChain
if (parsedVaa.Contract && parsedVaa.FromChain)
try {
token = tryHexToNativeString(
parsedVaa.Contract,
parsedVaa.FromChain as ChainId
);
} catch (e) {}
return (
<>
Version: {parsedVaa.version}
<br />
Timestamp: {new Date(parsedVaa.timestamp * 1000).toLocaleString()}
<br />
Consistency: {parsedVaa.consistency}
<br />
Nonce: {parsedVaa.nonce}
<br />
Origin: {parsedVaa.FromChain}
<br />
Token: {token}
<br />
Amount: {BigNumber.from(parsedVaa.Amount).toString()}
<br />
</>
);
}
function LatestVAAs() {
const vaas = useLatestVAAs();
const [sorting, setSorting] = useState<SortingState>([]);
const table = useReactTable({
columns,
data: vaas,
state: {
sorting,
},
getRowId: (vaa) => vaa._id,
getRowCanExpand: () => true,
getCoreRowModel: getCoreRowModel(),
getExpandedRowModel: getExpandedRowModel(),
getSortedRowModel: getSortedRowModel(),
onSortingChange: setSorting,
});
return (
<Card>
<Table<VAAsResponse> table={table} renderSubComponent={VAADetails} />
</Card>
);
}
export default LatestVAAs;

View File

@ -0,0 +1,14 @@
import useHeartbeats from "../hooks/useHeartbeats";
import Guardians from "./Guardians";
import LatestVAAs from "./LatestVAAs";
function Main() {
const heartbeats = useHeartbeats();
return (
<>
<Guardians heartbeats={heartbeats} />
<LatestVAAs />
</>
);
}
export default Main;

View File

@ -0,0 +1,68 @@
import { MenuItem, Select, SelectChangeEvent, useTheme } from "@mui/material";
import { useCallback } from "react";
import { networkOptions, useNetworkContext } from "../contexts/NetworkContext";
function NetworkSelector() {
const theme = useTheme();
const { currentNetwork, setCurrentNetwork } = useNetworkContext();
const handleChange = useCallback(
(e: SelectChangeEvent) => {
setCurrentNetwork(networkOptions[Number(e.target.value)]);
},
[setCurrentNetwork]
);
return (
<Select
onChange={handleChange}
value={(networkOptions.indexOf(currentNetwork) || 0).toString()}
margin="dense"
size="small"
sx={{
minWidth: 130,
// theme fixes
"& img": { filter: "invert(0)!important" },
"& .MuiOutlinedInput-notchedOutline": {
borderColor:
theme.palette.mode === "light" ? "rgba(255,255,255,.6)" : null,
},
"&:hover .MuiOutlinedInput-notchedOutline": {
borderColor:
theme.palette.mode === "light" ? "rgba(255,255,255,.8)" : null,
},
"&.Mui-focused .MuiOutlinedInput-notchedOutline": {
borderColor:
theme.palette.mode === "light" ? "rgba(255,255,255,.8)" : null,
},
"& .MuiSvgIcon-root": {
fill: theme.palette.mode === "light" ? "white" : null,
},
}}
SelectDisplayProps={{
style: { paddingTop: 4, paddingBottom: 4 },
}}
>
{networkOptions.map((network, idx) => (
<MenuItem key={network.endpoint} value={idx}>
{network.logo !== "" ? (
<img
src={network.logo}
alt={network.name}
style={{
height: 20,
maxHeight: 20,
verticalAlign: "middle",
// theme fixes
...(theme.palette.mode === "light"
? { filter: "invert(1)" }
: {}),
}}
/>
) : (
network.name
)}
</MenuItem>
))}
</Select>
);
}
export default NetworkSelector;

View File

@ -0,0 +1,109 @@
import {
ContrastOutlined,
DarkModeOutlined,
LightModeOutlined,
SettingsOutlined,
} from "@mui/icons-material";
import {
Box,
Dialog,
IconButton,
Slider,
TextField,
ToggleButton,
ToggleButtonGroup,
Typography,
} from "@mui/material";
import { useCallback, useState } from "react";
import { Theme, useSettingsContext } from "../contexts/SettingsContext";
function SettingsContent() {
const {
settings,
updateBackgroundOpacity,
updateBackgroundUrl,
updateTheme,
} = useSettingsContext();
const handleThemeChange = useCallback(
(event: any, newTheme: Theme) => {
updateTheme(newTheme);
},
[updateTheme]
);
const handleBackgroundOpacityChange = useCallback(
(event: any) => {
updateBackgroundOpacity(event.target.value);
},
[updateBackgroundOpacity]
);
const handleBackgroundUrlChange = useCallback(
(event: any) => {
updateBackgroundUrl(event.target.value);
},
[updateBackgroundUrl]
);
return (
<>
<Box mt={2} mx={2} textAlign="center">
<ToggleButtonGroup
value={settings.theme}
exclusive
onChange={handleThemeChange}
>
<ToggleButton value="light">
<LightModeOutlined />
</ToggleButton>
<ToggleButton value="dark">
<DarkModeOutlined />
</ToggleButton>
<ToggleButton value="auto">
<ContrastOutlined />
</ToggleButton>
</ToggleButtonGroup>
</Box>
<Box m={2}>
<TextField
value={settings.backgroundUrl || ""}
onChange={handleBackgroundUrlChange}
label="Background URL"
margin="dense"
fullWidth
/>
</Box>
<Box m={2}>
<Typography variant="body2">Background Opacity</Typography>
<Box pr={2} pt={2}>
<Slider
min={0.05}
max={1}
step={0.05}
value={settings.backgroundOpacity || 0.1}
onChange={handleBackgroundOpacityChange}
/>
</Box>
</Box>
</>
);
}
function Settings() {
const [open, setOpen] = useState(false);
const handleOpen = useCallback(() => {
setOpen(true);
}, []);
const handleClose = useCallback(() => {
setOpen(false);
}, []);
return (
<>
<IconButton color="inherit" onClick={handleOpen}>
<SettingsOutlined />
</IconButton>
<Dialog open={open} onClose={handleClose} maxWidth="xs" fullWidth>
<SettingsContent />
</Dialog>
</>
);
}
export default Settings;

View File

@ -0,0 +1,104 @@
import { ArrowDownward, ArrowUpward } from "@mui/icons-material";
import {
Box,
SxProps,
Table as MuiTable,
TableBody,
TableCell,
TableContainer,
TableHead,
TableRow,
Theme,
useTheme,
} from "@mui/material";
import { grey } from "@mui/material/colors";
import { flexRender, Row, Table as TanTable } from "@tanstack/react-table";
import { Fragment, ReactNode } from "react";
function Table<T>({
table,
conditionalRowStyle,
renderSubComponent,
}: {
table: TanTable<T>;
conditionalRowStyle?: (a: T) => SxProps<Theme> | undefined;
renderSubComponent?: ({ row }: { row: Row<T> }) => ReactNode;
}) {
const theme = useTheme();
return (
<TableContainer>
<MuiTable size="small">
<TableHead>
{table.getHeaderGroups().map((headerGroup) => (
<TableRow key={headerGroup.id}>
{headerGroup.headers.map((header) => (
<TableCell
key={header.id}
sx={
header.column.getCanSort()
? {
cursor: "pointer",
userSelect: "select-none",
":hover": {
background:
theme.palette.mode === "dark"
? grey[800]
: grey[100],
},
}
: {}
}
onClick={header.column.getToggleSortingHandler()}
>
<Box display="flex" alignContent="center">
{header.isPlaceholder
? null
: flexRender(
header.column.columnDef.header,
header.getContext()
)}
<Box flexGrow={1} />
<Box display="flex" alignItems="center">
{{
asc: <ArrowUpward fontSize="small" sx={{ ml: 0.5 }} />,
desc: (
<ArrowDownward fontSize="small" sx={{ ml: 0.5 }} />
),
}[header.column.getIsSorted() as string] ?? null}
</Box>
</Box>
</TableCell>
))}
</TableRow>
))}
</TableHead>
<TableBody>
{table.getRowModel().rows.map((row) => (
<Fragment key={row.id}>
<TableRow
sx={
conditionalRowStyle ? conditionalRowStyle(row.original) : {}
}
>
{row.getVisibleCells().map((cell) => (
<TableCell key={cell.id}>
{flexRender(cell.column.columnDef.cell, cell.getContext())}
</TableCell>
))}
</TableRow>
{renderSubComponent && row.getIsExpanded() && (
<TableRow>
{/* 2nd row is a custom 1 cell row */}
<TableCell colSpan={row.getVisibleCells().length}>
{renderSubComponent({ row })}
</TableCell>
</TableRow>
)}
</Fragment>
))}
</TableBody>
</MuiTable>
</TableContainer>
);
}
export default Table;

File diff suppressed because one or more lines are too long

View File

@ -0,0 +1,111 @@
import React, {
ReactNode,
useCallback,
useContext,
useEffect,
useMemo,
useState,
} from "react";
const STORAGE_KEY = "settings";
export type Theme = "light" | "dark" | "auto";
type Settings = {
backgroundUrl?: string;
backgroundOpacity?: number;
defaultEndpoint?: string;
theme: Theme;
};
type SettingsContextValue = {
settings: Settings;
updateBackgroundOpacity(value: number): void;
updateBackgroundUrl(value: string): void;
updateDefaultEndpoint(value: string): void;
updateTheme(value: Theme): void;
};
const isTheme = (arg: any): arg is Theme => {
return arg && (arg === "light" || arg === "dark" || arg === "auto");
};
const isSettings = (arg: any): arg is Settings => {
return arg && arg.theme && isTheme(arg.theme);
};
let localStorageSettings: Settings | null = null;
try {
const value = localStorage.getItem(STORAGE_KEY);
if (value) {
const parsedValue = JSON.parse(value);
if (isSettings(parsedValue)) {
localStorageSettings = parsedValue;
}
}
} catch (e) {}
const initialSettings: Settings = localStorageSettings || { theme: "auto" };
const saveSettings = (settings: Settings) => {
try {
localStorage.setItem(STORAGE_KEY, JSON.stringify(settings));
} catch (e) {}
};
const SettingsContext = React.createContext<SettingsContextValue>({
settings: initialSettings,
updateBackgroundOpacity: (value: number) => {},
updateBackgroundUrl: (value: string) => {},
updateDefaultEndpoint: (value: string) => {},
updateTheme: (value: Theme) => {},
});
export const SettingsContextProvider = ({
children,
}: {
children: ReactNode;
}) => {
const [settings, setSettings] = useState<Settings>(initialSettings);
const updateBackgroundOpacity = useCallback((value: number) => {
setSettings((settings) => ({ ...settings, backgroundOpacity: value }));
}, []);
const updateBackgroundUrl = useCallback((value: string) => {
setSettings((settings) => ({ ...settings, backgroundUrl: value }));
}, []);
const updateDefaultEndpoint = useCallback((value: string) => {
setSettings((settings) => ({ ...settings, defaultEndpoint: value }));
}, []);
const updateTheme = useCallback((value: Theme) => {
setSettings((settings) => ({ ...settings, theme: value }));
}, []);
// sync settings to state
useEffect(() => {
saveSettings(settings);
}, [settings]);
const value = useMemo(
() => ({
settings,
updateBackgroundOpacity,
updateBackgroundUrl,
updateDefaultEndpoint,
updateTheme,
}),
[
settings,
updateBackgroundOpacity,
updateBackgroundUrl,
updateDefaultEndpoint,
updateTheme,
]
);
return (
<SettingsContext.Provider value={value}>
{children}
</SettingsContext.Provider>
);
};
export const useSettingsContext = () => {
return useContext(SettingsContext);
};

View File

@ -0,0 +1,30 @@
import { Heartbeat_Network } from "@certusone/wormhole-sdk-proto-web/lib/cjs/gossip/v1/gossip";
import { GetLastHeartbeatsResponse_Entry } from "@certusone/wormhole-sdk-proto-web/lib/cjs/publicrpc/v1/publicrpc";
export type HeartbeatInfo = {
guardian: string;
name: string;
network: Heartbeat_Network;
};
export type ChainIdToHeartbeats = {
[chainId: number]: HeartbeatInfo[];
};
function useChainHeartbeats(heartbeats: GetLastHeartbeatsResponse_Entry[]) {
const chainIdsToHeartbeats: ChainIdToHeartbeats = {};
heartbeats.forEach((heartbeat) => {
heartbeat.rawHeartbeat?.networks.forEach((network) => {
if (!chainIdsToHeartbeats[network.id]) {
chainIdsToHeartbeats[network.id] = [];
}
chainIdsToHeartbeats[network.id].push({
guardian: heartbeat.p2pNodeAddr,
name: heartbeat.rawHeartbeat?.nodeName || "",
network,
});
});
});
return chainIdsToHeartbeats;
}
export default useChainHeartbeats;

View File

@ -0,0 +1,84 @@
import {
GovernorGetAvailableNotionalByChainResponse_Entry,
GovernorGetEnqueuedVAAsResponse_Entry,
GovernorGetTokenListResponse_Entry,
} from "@certusone/wormhole-sdk-proto-web/lib/cjs/publicrpc/v1/publicrpc";
import { useEffect, useState } from "react";
import { useNetworkContext } from "../contexts/NetworkContext";
import { getGovernorAvailableNotionalByChain } from "../utils/getGovernorAvailableNotionalByChain";
import { getGovernorEnqueuedVAAs } from "../utils/getGovernorEnqueuedVAAs";
import { getGovernorTokenList } from "../utils/getGovernorTokenList";
type GovernorInfo = {
notionals: GovernorGetAvailableNotionalByChainResponse_Entry[];
tokens: GovernorGetTokenListResponse_Entry[];
enqueued: GovernorGetEnqueuedVAAsResponse_Entry[];
};
const createEmptyInfo = (): GovernorInfo => ({
notionals: [],
tokens: [],
enqueued: [],
});
const TIMEOUT = 10 * 1000;
function useGovernorInfo(): GovernorInfo {
const { currentNetwork } = useNetworkContext();
const [governorInfo, setGovernorInfo] = useState<GovernorInfo>(
createEmptyInfo()
);
useEffect(() => {
setGovernorInfo(createEmptyInfo());
}, [currentNetwork]);
useEffect(() => {
let cancelled = false;
(async () => {
while (!cancelled) {
const response = await getGovernorAvailableNotionalByChain(
currentNetwork
);
if (!cancelled) {
setGovernorInfo((info) => ({ ...info, notionals: response.entries }));
await new Promise((resolve) => setTimeout(resolve, TIMEOUT));
}
}
})();
return () => {
cancelled = true;
};
}, [currentNetwork]);
useEffect(() => {
let cancelled = false;
(async () => {
// TODO: only update GovernorInfo with changes to token list, but that will cause displaySymbols to break
while (!cancelled) {
const response = await getGovernorTokenList(currentNetwork);
if (!cancelled) {
setGovernorInfo((info) => ({ ...info, tokens: response.entries }));
await new Promise((resolve) => setTimeout(resolve, TIMEOUT));
}
}
})();
return () => {
cancelled = true;
};
}, [currentNetwork]);
useEffect(() => {
let cancelled = false;
(async () => {
while (!cancelled) {
const response = await getGovernorEnqueuedVAAs(currentNetwork);
if (!cancelled) {
setGovernorInfo((info) => ({ ...info, enqueued: response.entries }));
await new Promise((resolve) => setTimeout(resolve, TIMEOUT));
}
}
})();
return () => {
cancelled = true;
};
}, [currentNetwork]);
return governorInfo;
}
export default useGovernorInfo;

View File

@ -0,0 +1,54 @@
import axios from "axios";
import { useEffect, useState } from "react";
import { useNetworkContext } from "../contexts/NetworkContext";
import { NumberLong } from "../utils/longToDate";
export type HeartbeatResponse = {
boottimestamp: NumberLong;
counter: number;
createdAt: string;
features: string[] | null;
guardianaddr: string;
networks: {
contractaddress: string;
errorcount: number;
height: number;
id: number;
}[];
nodename: string;
timestamp: NumberLong;
updatedAt: string;
version: string;
_id: string;
};
function useHeartbeats(): HeartbeatResponse[] {
const { currentNetwork } = useNetworkContext();
const [heartbeats, setHeartbeats] = useState<HeartbeatResponse[]>([]);
useEffect(() => {
setHeartbeats([]);
}, [currentNetwork]);
useEffect(() => {
let cancelled = false;
(async () => {
while (!cancelled) {
const response = await axios.get<HeartbeatResponse[]>(
"http://localhost:3000/api/heartbeats"
);
if (!cancelled) {
setHeartbeats(
response.data.sort(
(a, b) => a.nodename.localeCompare(b.nodename || "") || -1
)
);
await new Promise((resolve) => setTimeout(resolve, 1000));
}
}
})();
return () => {
cancelled = true;
};
}, [currentNetwork]);
return heartbeats;
}
export default useHeartbeats;

View File

@ -0,0 +1,26 @@
import axios from "axios";
import { useEffect, useState } from "react";
// https://docs.github.com/en/rest/releases/releases#get-the-latest-release
function useLatestRelease(): string | null {
const [latestRelease, setLatestRelease] = useState<string | null>(null);
useEffect(() => {
let cancelled = false;
(async () => {
while (!cancelled) {
const response = await axios.get(
"https://api.github.com/repos/wormhole-foundation/wormhole/releases/latest"
);
if (!cancelled) {
setLatestRelease(response.data?.tag_name || null);
await new Promise((resolve) => setTimeout(resolve, 60000));
}
}
})();
return () => {
cancelled = true;
};
}, []);
return latestRelease;
}
export default useLatestRelease;

View File

@ -0,0 +1,37 @@
import axios from "axios";
import { useEffect, useState } from "react";
import { useNetworkContext } from "../contexts/NetworkContext";
export type VAAsResponse = {
createdAt: string;
updatedAt: string;
vaa: string;
_id: string;
};
function useLatestVAAs(): VAAsResponse[] {
const { currentNetwork } = useNetworkContext();
const [vaas, setVAAs] = useState<VAAsResponse[]>([]);
useEffect(() => {
setVAAs([]);
}, [currentNetwork]);
useEffect(() => {
let cancelled = false;
(async () => {
while (!cancelled) {
const response = await axios.get<VAAsResponse[]>(
"http://localhost:3000/api/vaas/2"
);
if (!cancelled) {
setVAAs(response.data);
await new Promise((resolve) => setTimeout(resolve, 1000));
}
}
})();
return () => {
cancelled = true;
};
}, [currentNetwork]);
return vaas;
}
export default useLatestVAAs;

View File

@ -0,0 +1,418 @@
import {
ChainId,
CHAIN_ID_SOLANA,
CHAIN_ID_TERRA,
isEVMChain,
isHexNativeTerra,
tryHexToNativeAssetString,
tryNativeToHexString,
} from "@certusone/wormhole-sdk";
import { GovernorGetTokenListResponse_Entry } from "@certusone/wormhole-sdk-proto-web/lib/cjs/publicrpc/v1/publicrpc";
import { LCDClient } from "@terra-money/terra.js";
import axios from "axios";
import { ethers } from "ethers";
import { useEffect, useMemo } from "react";
import { CHAIN_INFO_MAP } from "../utils/consts";
require("dotenv").config();
function newProvider(
url: string,
batch: boolean = false
): ethers.providers.JsonRpcProvider | ethers.providers.JsonRpcBatchProvider {
// only support http(s), not ws(s) as the websocket constructor can blow up the entire process
// it uses a nasty setTimeout(()=>{},0) so we are unable to cleanly catch its errors
if (url.startsWith("http")) {
if (batch) {
return new ethers.providers.JsonRpcBatchProvider(url);
}
return new ethers.providers.JsonRpcProvider(url);
}
throw new Error("url does not start with http/https!");
}
const ERC20_BASIC_ABI = [
"function name() view returns (string name)",
"function symbol() view returns (string symbol)",
"function decimals() view returns (uint8 decimals)",
];
//for evm chains
async function getTokenContract(
address: string,
provider:
| ethers.providers.JsonRpcProvider
| ethers.providers.JsonRpcBatchProvider
) {
const contract = new ethers.Contract(address, ERC20_BASIC_ABI, provider);
return contract;
}
type TokenInfo = {
address: string;
name: string;
decimals: number;
symbol: string;
};
async function getEvmTokenMetaData(
chain: ChainId,
governorTokenList: GovernorGetTokenListResponse_Entry[]
) {
const chainInfo = CHAIN_INFO_MAP[chain];
var formattedAddressMap: { [key: string]: string } = {};
var formattedTokens: string[] = [];
try {
const tokenListByChain = governorTokenList
.filter(
(g) =>
g.originChainId === chainInfo.chainId &&
!TOKEN_CACHE.get([chain, g.originAddress].join("_"))
)
.map((tk) => tk.originAddress);
tokenListByChain.forEach((tk) => {
const tk_ = tryHexToNativeAssetString(tk.slice(2), chainInfo.chainId);
formattedTokens.push(tk_);
formattedAddressMap[tk_] = tk;
});
let provider = newProvider(
chainInfo.endpointUrl,
true
) as ethers.providers.JsonRpcBatchProvider;
const tokenContracts = await Promise.all(
formattedTokens.map((tokenAddress) =>
getTokenContract(tokenAddress, provider)
)
);
const tokenInfos = await Promise.all(
tokenContracts.map((tokenContract) =>
Promise.all([
tokenContract.address.toLowerCase(),
tokenContract.name(),
tokenContract.decimals(),
// tokenContract.balanceOf(bridgeAddress),
tokenContract.symbol(),
])
)
);
tokenInfos.forEach((tk) => {
TOKEN_CACHE.set([chain, formattedAddressMap[tk[0]]].join("_"), {
address: tk[0],
name: tk[1],
decimals: tk[2],
symbol: tk[3],
});
});
} catch (e) {
console.log(e);
console.log(chain, chainInfo);
}
return;
}
async function loadTokenListSolana(
url = "https://token-list.solana.com/solana.tokenlist.json"
) {
const response: any = await axios.get(url).catch(function (error) {
if (error.response) {
console.log("could not load token list", error.response.status);
}
});
if (response["status"] === 200) {
const data = response["data"];
const token_list = data.tokens;
return token_list;
} else {
console.log("bad response for token list");
return [];
}
}
async function getSolanaTokenMetaData(
chain: ChainId,
governorTokenList: GovernorGetTokenListResponse_Entry[]
) {
const chainInfo = CHAIN_INFO_MAP[chain];
var formattedAddressMap: { [key: string]: string } = {};
var formattedTokens: string[] = [];
try {
const tokenListByChain = governorTokenList
.filter(
(g) =>
g.originChainId === chainInfo.chainId &&
!TOKEN_CACHE.get([chain, g.originAddress].join("_"))
)
.map((tk) => tk.originAddress);
tokenListByChain.forEach((tk) => {
const tk_ = tryHexToNativeAssetString(tk.slice(2), chainInfo.chainId);
formattedTokens.push(tk_);
formattedAddressMap[tk_] = tk;
});
var metaDataArr: any[] = [];
try {
metaDataArr = await loadTokenListSolana();
} catch (e) {
console.log(e);
}
var tokenContracts: TokenInfo[] = [];
formattedTokens.forEach((token) => {
const metaData = metaDataArr.filter((x) => x.address === token);
if (metaData.length > 0) {
tokenContracts.push(metaData[0]);
}
});
tokenContracts.forEach((tokenContract) => {
TOKEN_CACHE.set(
[chain, formattedAddressMap[tokenContract.address]].join("_"),
{
address: tokenContract.address,
name: tokenContract.name,
decimals: tokenContract.decimals,
symbol: tokenContract.symbol,
}
);
});
} catch (e) {
console.log(e);
console.log(chain, chainInfo);
}
return;
}
type TerraMetadata = {
address: string;
symbol?: string;
logo?: string;
name?: string;
decimals?: number;
};
const fetchSingleTerraMetadata = async (address: string, lcd: LCDClient) =>
lcd.wasm
.contractQuery(address, {
token_info: {},
})
.then(
({ symbol, name, decimals }: any) =>
({ address: address, symbol, name, decimals } as TerraMetadata)
);
async function getSingleTerraMetaData(
originAddress: string,
originChain: ChainId
) {
const TERRA_HOST = {
URL:
originChain === CHAIN_ID_TERRA
? "https://columbus-fcd.terra.dev"
: "https://phoenix-fcd.terra.dev",
chainID: originChain === CHAIN_ID_TERRA ? "columbus-5" : "phoenix-1",
name: "mainnet",
};
const lcd = new LCDClient(TERRA_HOST);
if (isHexNativeTerra(tryNativeToHexString(originAddress, originChain))) {
if (originAddress === "uusd") {
return {
address: originAddress,
name: "UST Classic",
symbol: "USTC",
decimals: 6,
};
} else if (originAddress === "uluna") {
return {
address: originAddress,
name: "Luna Classic",
symbol: "LUNC",
decimals: 6,
};
} else {
return {
address: originAddress,
name: "",
symbol: "",
decimals: 8,
};
}
} else {
return await fetchSingleTerraMetadata(originAddress, lcd);
}
}
async function getTerraTokenMetaData(
chain: ChainId,
governorTokenList: GovernorGetTokenListResponse_Entry[]
) {
const chainInfo = CHAIN_INFO_MAP[chain];
var formattedAddressMap: { [key: string]: string } = {};
var formattedTokens: string[] = [];
try {
const tokenListByChain = governorTokenList
.filter(
(g) =>
g.originChainId === chainInfo.chainId &&
!TOKEN_CACHE.get([chain, g.originAddress].join("_"))
)
.map((tk) => tk.originAddress);
tokenListByChain.forEach((tk) => {
const tk_ = tryHexToNativeAssetString(tk.slice(2), chainInfo.chainId);
formattedTokens.push(tk_);
formattedAddressMap[tk_] = tk;
});
var tokenContracts: any[] = [];
for (let i = 0; i < formattedTokens.length; i++) {
const token = formattedTokens[i];
const metaData = await getSingleTerraMetaData(token, chain);
tokenContracts.push(metaData);
}
tokenContracts.forEach((tokenContract) => {
TOKEN_CACHE.set(
[chain, formattedAddressMap[tokenContract.address]].join("_"),
{
address: tokenContract.address,
name: tokenContract.name,
decimals: tokenContract.decimals,
symbol: tokenContract.symbol,
}
);
});
} catch (e) {
console.log(e);
console.log(chain, chainInfo);
}
return;
}
const MISC_TOKEN_META_DATA: {
[key: string]: {
[key: string]: { name: string; symbol: string; decimals: number };
};
} = {
"8": {
"0x0000000000000000000000000000000000000000000000000000000000000000": {
name: "ALGO",
symbol: "ALGO",
decimals: 6,
},
"0x000000000000000000000000000000000000000000000000000000000004c5c1": {
name: "USDT",
symbol: "USDT",
decimals: 6,
},
"0x0000000000000000000000000000000000000000000000000000000001e1ab70": {
name: "USDC",
symbol: "USDC",
decimals: 6,
},
},
"15": {
"0x0000000000000000000000000000000000000000000000000000000000000000": {
name: "NEAR",
symbol: "NEAR",
decimals: 24,
},
},
"18": {
"0x01fa6c6fbc36d8c245b0a852a43eb5d644e8b4c477b27bfab9537c10945939da": {
name: "LUNA",
symbol: "LUNA",
decimals: 6,
},
},
};
async function getMiscTokenMetaData(
chain: ChainId,
governorTokenList: GovernorGetTokenListResponse_Entry[]
) {
const chainInfo = CHAIN_INFO_MAP[chain];
const tokenMetaDataByChain = MISC_TOKEN_META_DATA[chain.toString()];
try {
const tokenListByChain = governorTokenList
.filter(
(g) =>
g.originChainId === chainInfo.chainId &&
!TOKEN_CACHE.get([chain, g.originAddress].join("_"))
)
.map((tk) => tk.originAddress);
tokenListByChain.forEach((tk) => {
const metaData = tokenMetaDataByChain[tk];
TOKEN_CACHE.set([chain, tk].join("_"), {
address: tk,
name: metaData?.name,
decimals: metaData?.decimals,
symbol: metaData?.symbol,
});
});
} catch (e) {
console.log(e);
console.log(chain, chainInfo);
}
return;
}
const TOKEN_CACHE = new Map<string, TokenInfo>();
async function getTokenMetaData(
governorTokenList: GovernorGetTokenListResponse_Entry[]
) {
const chains = Object.keys(CHAIN_INFO_MAP);
for (let i = 0; i < chains.length; i++) {
const chain = chains[i];
const chainInfo = CHAIN_INFO_MAP[chain];
const chainId = chainInfo.chainId;
try {
//grab token info
if (isEVMChain(chainId)) {
await getEvmTokenMetaData(chainId, governorTokenList);
} else if (chainId === CHAIN_ID_SOLANA) {
await getSolanaTokenMetaData(chainId, governorTokenList);
} else if (chainId === CHAIN_ID_TERRA) {
await getTerraTokenMetaData(chainId, governorTokenList);
} else {
// currently no support for ALGORAND, NEAR, TERRA2
console.log(`the chain=${chain} is not supported`);
await getMiscTokenMetaData(chainId, governorTokenList);
}
await new Promise((resolve) => setTimeout(resolve, 6000));
} catch (e) {
console.log(e);
console.log(chain, chainInfo);
continue;
}
}
await new Promise((resolve) => setTimeout(resolve, 3000000));
return;
}
const TIMEOUT = 60 * 1000;
function useSymbolInfo(tokens: GovernorGetTokenListResponse_Entry[]) {
// TODO: GovernorInfo gets fetched repeatedly, but we don't need to refresh the list
// So using string'd version of token list as a hack around when to update the token list
const memoizedTokens = useMemo(() => JSON.stringify(tokens), [tokens]);
useEffect(() => {
const tokens = JSON.parse(memoizedTokens);
(async () => {
// TODO: use a state setter to update instead of relying on TOKEN_CACHE.
await getTokenMetaData(tokens);
await new Promise((resolve) => setTimeout(resolve, TIMEOUT));
})();
}, [memoizedTokens]);
return TOKEN_CACHE;
}
export default useSymbolInfo;

View File

@ -0,0 +1,48 @@
import { SvgIcon } from "@mui/material";
// paths taken from wormhole-symbol.inline.svg
function WormholeStatsIcon() {
return (
<SvgIcon viewBox="0 0 40 40">
<path
fill="#ffffff"
d="M20,40c-5.3,0-10.4-2.1-14.1-5.9C2.1,30.4,0,25.3,0,20C0,14.7,2.1,9.6,5.9,5.9C9.6,2.1,14.7,0,20,0
c5.3,0,10.4,2.1,14.1,5.9C37.9,9.6,40,14.7,40,20c0,5.3-2.1,10.4-5.9,14.1C30.4,37.9,25.3,40,20,40z M20,1.5c-4.9,0-9.6,1.9-13,5.4
s-5.4,8.1-5.4,13c0,4.9,2,9.6,5.4,13c3.5,3.5,8.2,5.4,13,5.4s9.6-1.9,13-5.4c3.5-3.5,5.4-8.1,5.4-13c0-4.9-2-9.6-5.4-13
C29.6,3.5,24.9,1.5,20,1.5L20,1.5z"
/>
<path
fill="#ffffff"
d="M22.5,36.8c-4.4,0-8.6-1.7-11.7-4.9C7.7,28.8,6,24.6,6,20.2c0-4.4,1.8-8.6,4.9-11.7c3.1-3.1,7.3-4.8,11.7-4.9
c4.4,0,8.6,1.7,11.7,4.9c3.1,3.1,4.9,7.3,4.9,11.7c0,4.4-1.8,8.6-4.9,11.7C31.1,35.1,26.9,36.8,22.5,36.8z M22.5,4.9
c-4.1,0-8,1.6-10.9,4.5c-2.9,2.9-4.5,6.8-4.5,10.9c0,4.1,1.6,8,4.5,10.9c2.9,2.9,6.8,4.5,10.9,4.5c4.1,0,8-1.6,10.9-4.5
c2.9-2.9,4.5-6.8,4.5-10.9c0-4.1-1.6-8-4.5-10.9C30.5,6.5,26.6,4.9,22.5,4.9L22.5,4.9z"
/>
<path
fill="#ffffff"
d="M25.1,33.6c-3.5,0-6.8-1.4-9.3-3.8c-2.5-2.5-3.8-5.8-3.9-9.3c0-3.5,1.4-6.8,3.9-9.3c2.5-2.5,5.8-3.8,9.3-3.8
c3.5,0,6.8,1.4,9.3,3.8c2.5,2.5,3.8,5.8,3.9,9.3c0,3.5-1.4,6.8-3.9,9.3C31.9,32.3,28.5,33.6,25.1,33.6L25.1,33.6z M25.1,8.2
c-3.3,0-6.4,1.3-8.7,3.6c-2.3,2.3-3.6,5.4-3.6,8.7c0,3.3,1.3,6.4,3.6,8.7c2.3,2.3,5.4,3.6,8.7,3.6c3.3,0,6.4-1.3,8.7-3.6
c2.3-2.3,3.6-5.4,3.6-8.7c0-3.3-1.3-6.4-3.6-8.7C31.4,9.5,28.3,8.2,25.1,8.2L25.1,8.2z"
/>
<path
fill="#ffffff"
d="M27.6,30.5c-2.6,0-5-1-6.9-2.8c-1.8-1.8-2.8-4.3-2.8-6.9c0-2.6,1-5,2.8-6.9c1.8-1.8,4.3-2.8,6.9-2.8
c2.6,0,5,1,6.9,2.8s2.8,4.3,2.8,6.9c0,2.6-1,5-2.8,6.9C32.6,29.5,30.2,30.5,27.6,30.5L27.6,30.5z M27.6,11.6c-2.4,0-4.8,1-6.5,2.7
c-1.7,1.7-2.7,4.1-2.7,6.5c0,2.4,1,4.8,2.7,6.5c1.7,1.7,4.1,2.7,6.5,2.7c2.4,0,4.8-1,6.5-2.7c1.7-1.7,2.7-4.1,2.7-6.5
c0-2.4-1-4.8-2.7-6.5C32.4,12.6,30,11.6,27.6,11.6z"
/>
<path
fill="#ffffff"
stroke="#231f20"
strokeMiterlimit="10"
d="M37.2,10.5h-7.1c-1,0-1.8,0.8-1.8,1.8s0.8,1.8,1.8,1.8h2.8l-7.6,7.6l-1,1l-0.2,0.2l-2.3,2.3l-8.2-8.2
c-0.3-0.3-0.8-0.5-1.3-0.5c-0.5,0-0.9,0.2-1.3,0.5l-9.5,9.5C1.2,26.8,1,27.2,1,27.7c0,1,0.8,1.8,1.8,1.8c0.5,0,0.9-0.2,1.3-0.5
l8.2-8.2l8.2,8.2c0.3,0.3,0.8,0.5,1.3,0.5s0.9-0.2,1.3-0.5l3.5-3.5l0,0l8.9-8.9v2.8c0,1,0.8,1.8,1.8,1.8s1.8-0.8,1.8-1.8v-7.1
C39,11.3,38.2,10.5,37.2,10.5z"
/>
</SvgIcon>
);
}
export default WormholeStatsIcon;

View File

@ -0,0 +1,31 @@
<?xml version="1.0" encoding="utf-8"?>
<!-- Generator: Adobe Illustrator 26.0.3, SVG Export Plug-In . SVG Version: 6.00 Build 0) -->
<svg version="1.1" id="Layer_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px"
viewBox="0 0 40 40" style="enable-background:new 0 0 40 40;" xml:space="preserve">
<style type="text/css">
.st0{fill:#231F20;}
.st1{fill:#FFFFFF;}
.st2{fill:#FFFFFF;stroke:#231F20;stroke-miterlimit:10;}
</style>
<rect class="st0" width="40" height="40"/>
<path class="st1" d="M20,40c-5.3,0-10.4-2.1-14.1-5.9C2.1,30.4,0,25.3,0,20C0,14.7,2.1,9.6,5.9,5.9C9.6,2.1,14.7,0,20,0
c5.3,0,10.4,2.1,14.1,5.9C37.9,9.6,40,14.7,40,20c0,5.3-2.1,10.4-5.9,14.1C30.4,37.9,25.3,40,20,40z M20,1.5c-4.9,0-9.6,1.9-13,5.4
s-5.4,8.1-5.4,13c0,4.9,2,9.6,5.4,13c3.5,3.5,8.2,5.4,13,5.4s9.6-1.9,13-5.4c3.5-3.5,5.4-8.1,5.4-13c0-4.9-2-9.6-5.4-13
C29.6,3.5,24.9,1.5,20,1.5L20,1.5z"/>
<path class="st1" d="M22.5,36.8c-4.4,0-8.6-1.7-11.7-4.9C7.7,28.8,6,24.6,6,20.2c0-4.4,1.8-8.6,4.9-11.7c3.1-3.1,7.3-4.8,11.7-4.9
c4.4,0,8.6,1.7,11.7,4.9c3.1,3.1,4.9,7.3,4.9,11.7c0,4.4-1.8,8.6-4.9,11.7C31.1,35.1,26.9,36.8,22.5,36.8z M22.5,4.9
c-4.1,0-8,1.6-10.9,4.5c-2.9,2.9-4.5,6.8-4.5,10.9c0,4.1,1.6,8,4.5,10.9c2.9,2.9,6.8,4.5,10.9,4.5c4.1,0,8-1.6,10.9-4.5
c2.9-2.9,4.5-6.8,4.5-10.9c0-4.1-1.6-8-4.5-10.9C30.5,6.5,26.6,4.9,22.5,4.9L22.5,4.9z"/>
<path class="st1" d="M25.1,33.6c-3.5,0-6.8-1.4-9.3-3.8c-2.5-2.5-3.8-5.8-3.9-9.3c0-3.5,1.4-6.8,3.9-9.3c2.5-2.5,5.8-3.8,9.3-3.8
c3.5,0,6.8,1.4,9.3,3.8c2.5,2.5,3.8,5.8,3.9,9.3c0,3.5-1.4,6.8-3.9,9.3C31.9,32.3,28.5,33.6,25.1,33.6L25.1,33.6z M25.1,8.2
c-3.3,0-6.4,1.3-8.7,3.6c-2.3,2.3-3.6,5.4-3.6,8.7c0,3.3,1.3,6.4,3.6,8.7c2.3,2.3,5.4,3.6,8.7,3.6c3.3,0,6.4-1.3,8.7-3.6
c2.3-2.3,3.6-5.4,3.6-8.7c0-3.3-1.3-6.4-3.6-8.7C31.4,9.5,28.3,8.2,25.1,8.2L25.1,8.2z"/>
<path class="st1" d="M27.6,30.5c-2.6,0-5-1-6.9-2.8c-1.8-1.8-2.8-4.3-2.8-6.9c0-2.6,1-5,2.8-6.9c1.8-1.8,4.3-2.8,6.9-2.8
c2.6,0,5,1,6.9,2.8s2.8,4.3,2.8,6.9c0,2.6-1,5-2.8,6.9C32.6,29.5,30.2,30.5,27.6,30.5L27.6,30.5z M27.6,11.6c-2.4,0-4.8,1-6.5,2.7
c-1.7,1.7-2.7,4.1-2.7,6.5c0,2.4,1,4.8,2.7,6.5c1.7,1.7,4.1,2.7,6.5,2.7c2.4,0,4.8-1,6.5-2.7c1.7-1.7,2.7-4.1,2.7-6.5
c0-2.4-1-4.8-2.7-6.5C32.4,12.6,30,11.6,27.6,11.6z"/>
<path class="st2" d="M37.2,10.5h-7.1c-1,0-1.8,0.8-1.8,1.8s0.8,1.8,1.8,1.8h2.8l-7.6,7.6l-1,1l-0.2,0.2l-2.3,2.3l-8.2-8.2
c-0.3-0.3-0.8-0.5-1.3-0.5c-0.5,0-0.9,0.2-1.3,0.5l-9.5,9.5C1.2,26.8,1,27.2,1,27.7c0,1,0.8,1.8,1.8,1.8c0.5,0,0.9-0.2,1.3-0.5
l8.2-8.2l8.2,8.2c0.3,0.3,0.8,0.5,1.3,0.5s0.9-0.2,1.3-0.5l3.5-3.5l0,0l8.9-8.9v2.8c0,1,0.8,1.8,1.8,1.8s1.8-0.8,1.8-1.8v-7.1
C39,11.3,38.2,10.5,37.2,10.5z"/>
</svg>

After

Width:  |  Height:  |  Size: 2.6 KiB

16
web/src/index.tsx Normal file
View File

@ -0,0 +1,16 @@
import React from "react";
import ReactDOM from "react-dom/client";
import App from "./App";
import reportWebVitals from "./reportWebVitals";
const root = ReactDOM.createRoot(document.getElementById("root") as any);
root.render(
<React.StrictMode>
<App />
</React.StrictMode>
);
// If you want to start measuring performance in your app, pass a function
// to log results (for example: reportWebVitals(console.log))
// or send to an analytics endpoint. Learn more: https://bit.ly/CRA-vitals
reportWebVitals();

1
web/src/logo.svg Normal file
View File

@ -0,0 +1 @@
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 841.9 595.3"><g fill="#61DAFB"><path d="M666.3 296.5c0-32.5-40.7-63.3-103.1-82.4 14.4-63.6 8-114.2-20.2-130.4-6.5-3.8-14.1-5.6-22.4-5.6v22.3c4.6 0 8.3.9 11.4 2.6 13.6 7.8 19.5 37.5 14.9 75.7-1.1 9.4-2.9 19.3-5.1 29.4-19.6-4.8-41-8.5-63.5-10.9-13.5-18.5-27.5-35.3-41.6-50 32.6-30.3 63.2-46.9 84-46.9V78c-27.5 0-63.5 19.6-99.9 53.6-36.4-33.8-72.4-53.2-99.9-53.2v22.3c20.7 0 51.4 16.5 84 46.6-14 14.7-28 31.4-41.3 49.9-22.6 2.4-44 6.1-63.6 11-2.3-10-4-19.7-5.2-29-4.7-38.2 1.1-67.9 14.6-75.8 3-1.8 6.9-2.6 11.5-2.6V78.5c-8.4 0-16 1.8-22.6 5.6-28.1 16.2-34.4 66.7-19.9 130.1-62.2 19.2-102.7 49.9-102.7 82.3 0 32.5 40.7 63.3 103.1 82.4-14.4 63.6-8 114.2 20.2 130.4 6.5 3.8 14.1 5.6 22.5 5.6 27.5 0 63.5-19.6 99.9-53.6 36.4 33.8 72.4 53.2 99.9 53.2 8.4 0 16-1.8 22.6-5.6 28.1-16.2 34.4-66.7 19.9-130.1 62-19.1 102.5-49.9 102.5-82.3zm-130.2-66.7c-3.7 12.9-8.3 26.2-13.5 39.5-4.1-8-8.4-16-13.1-24-4.6-8-9.5-15.8-14.4-23.4 14.2 2.1 27.9 4.7 41 7.9zm-45.8 106.5c-7.8 13.5-15.8 26.3-24.1 38.2-14.9 1.3-30 2-45.2 2-15.1 0-30.2-.7-45-1.9-8.3-11.9-16.4-24.6-24.2-38-7.6-13.1-14.5-26.4-20.8-39.8 6.2-13.4 13.2-26.8 20.7-39.9 7.8-13.5 15.8-26.3 24.1-38.2 14.9-1.3 30-2 45.2-2 15.1 0 30.2.7 45 1.9 8.3 11.9 16.4 24.6 24.2 38 7.6 13.1 14.5 26.4 20.8 39.8-6.3 13.4-13.2 26.8-20.7 39.9zm32.3-13c5.4 13.4 10 26.8 13.8 39.8-13.1 3.2-26.9 5.9-41.2 8 4.9-7.7 9.8-15.6 14.4-23.7 4.6-8 8.9-16.1 13-24.1zM421.2 430c-9.3-9.6-18.6-20.3-27.8-32 9 .4 18.2.7 27.5.7 9.4 0 18.7-.2 27.8-.7-9 11.7-18.3 22.4-27.5 32zm-74.4-58.9c-14.2-2.1-27.9-4.7-41-7.9 3.7-12.9 8.3-26.2 13.5-39.5 4.1 8 8.4 16 13.1 24 4.7 8 9.5 15.8 14.4 23.4zM420.7 163c9.3 9.6 18.6 20.3 27.8 32-9-.4-18.2-.7-27.5-.7-9.4 0-18.7.2-27.8.7 9-11.7 18.3-22.4 27.5-32zm-74 58.9c-4.9 7.7-9.8 15.6-14.4 23.7-4.6 8-8.9 16-13 24-5.4-13.4-10-26.8-13.8-39.8 13.1-3.1 26.9-5.8 41.2-7.9zm-90.5 125.2c-35.4-15.1-58.3-34.9-58.3-50.6 0-15.7 22.9-35.6 58.3-50.6 8.6-3.7 18-7 27.7-10.1 5.7 19.6 13.2 40 22.5 60.9-9.2 20.8-16.6 41.1-22.2 60.6-9.9-3.1-19.3-6.5-28-10.2zM310 490c-13.6-7.8-19.5-37.5-14.9-75.7 1.1-9.4 2.9-19.3 5.1-29.4 19.6 4.8 41 8.5 63.5 10.9 13.5 18.5 27.5 35.3 41.6 50-32.6 30.3-63.2 46.9-84 46.9-4.5-.1-8.3-1-11.3-2.7zm237.2-76.2c4.7 38.2-1.1 67.9-14.6 75.8-3 1.8-6.9 2.6-11.5 2.6-20.7 0-51.4-16.5-84-46.6 14-14.7 28-31.4 41.3-49.9 22.6-2.4 44-6.1 63.6-11 2.3 10.1 4.1 19.8 5.2 29.1zm38.5-66.7c-8.6 3.7-18 7-27.7 10.1-5.7-19.6-13.2-40-22.5-60.9 9.2-20.8 16.6-41.1 22.2-60.6 9.9 3.1 19.3 6.5 28.1 10.2 35.4 15.1 58.3 34.9 58.3 50.6-.1 15.7-23 35.6-58.4 50.6zM320.8 78.4z"/><circle cx="420.9" cy="296.5" r="45.7"/><path d="M520.5 78.1z"/></g></svg>

After

Width:  |  Height:  |  Size: 2.6 KiB

View File

@ -0,0 +1,13 @@
const reportWebVitals = onPerfEntry => {
if (onPerfEntry && onPerfEntry instanceof Function) {
import('web-vitals').then(({ getCLS, getFID, getFCP, getLCP, getTTFB }) => {
getCLS(onPerfEntry);
getFID(onPerfEntry);
getFCP(onPerfEntry);
getLCP(onPerfEntry);
getTTFB(onPerfEntry);
});
}
};
export default reportWebVitals;

5
web/src/setupTests.js Normal file
View File

@ -0,0 +1,5 @@
// jest-dom adds custom jest matchers for asserting on DOM nodes.
// allows you to do things like:
// expect(element).toHaveTextContent(/react/i)
// learn more: https://github.com/testing-library/jest-dom
import '@testing-library/jest-dom';

View File

@ -0,0 +1,8 @@
import { CHAINS } from "@certusone/wormhole-sdk";
const chainIdToNameMap = Object.fromEntries(
Object.entries(CHAINS).map(([key, value]) => [value, key])
);
const chainIdToName = (chainId: number) =>
chainIdToNameMap[chainId] || "Unknown";
export default chainIdToName;

215
web/src/utils/consts.ts Normal file
View File

@ -0,0 +1,215 @@
import {
CHAIN_ID_SOLANA,
CHAIN_ID_ETH,
CHAIN_ID_TERRA,
CHAIN_ID_BSC,
CHAIN_ID_POLYGON,
CHAIN_ID_AVAX,
CHAIN_ID_OASIS,
CHAIN_ID_ALGORAND,
CHAIN_ID_AURORA,
CHAIN_ID_FANTOM,
CHAIN_ID_KARURA,
CHAIN_ID_ACALA,
CHAIN_ID_KLAYTN,
CHAIN_ID_CELO,
CHAIN_ID_TERRA2,
ChainId,
CHAIN_ID_NEAR,
} from "@certusone/wormhole-sdk";
require("dotenv").config();
export type CHAIN_INFO = {
name: string;
evm: boolean;
chainId: ChainId;
endpointUrl: any;
platform: string;
covalentChain: number;
explorerStem: string;
apiKey: string;
urlStem: string;
};
export const CHAIN_INFO_MAP: { [key: string]: CHAIN_INFO } = {
1: {
name: "solana",
evm: false,
chainId: CHAIN_ID_SOLANA,
urlStem: `https://public-api.solscan.io`,
endpointUrl:
process.env.REACT_APP_SOLANA_RPC || "https://api.mainnet-beta.solana.com",
apiKey: "",
platform: "solana",
covalentChain: 1399811149,
explorerStem: `https://solscan.io`,
},
2: {
name: "eth",
evm: true,
chainId: CHAIN_ID_ETH,
endpointUrl: process.env.REACT_APP_ETH_RPC || "https://rpc.ankr.com/eth",
apiKey: "",
urlStem: `https://api.etherscan.io`,
platform: "ethereum",
covalentChain: 1,
explorerStem: `https://etherscan.io`,
},
3: {
name: "terra",
evm: false,
chainId: CHAIN_ID_TERRA,
endpointUrl: "",
apiKey: "",
urlStem: "https://columbus-fcd.terra.dev",
platform: "terra",
covalentChain: 3,
explorerStem: `https://finder.terra.money/classic`,
},
4: {
name: "bsc",
evm: true,
chainId: CHAIN_ID_BSC,
endpointUrl:
process.env.REACT_APP_BSC_RPC || "https://bsc-dataseed2.defibit.io",
apiKey: "",
urlStem: `https://api.bscscan.com`,
platform: "binance-smart-chain",
covalentChain: 56,
explorerStem: `https://bscscan.com`,
},
5: {
name: "polygon",
evm: true,
chainId: CHAIN_ID_POLYGON,
endpointUrl: process.env.REACT_APP_POLYGON_RPC || "https://polygon-rpc.com",
apiKey: "",
urlStem: `https://api.polygonscan.com`,
platform: "polygon-pos", //coingecko?,
covalentChain: 137,
explorerStem: `https://polygonscan.com`,
},
6: {
name: "avalanche",
evm: true,
chainId: CHAIN_ID_AVAX,
endpointUrl:
process.env.REACT_APP_AVAX_RPC || "https://api.avax.network/ext/bc/C/rpc",
apiKey: "",
urlStem: `https://api.snowtrace.io`,
platform: "avalanche", //coingecko?
covalentChain: 43114,
explorerStem: `https://snowtrace.io`,
},
7: {
name: "oasis",
evm: true,
chainId: CHAIN_ID_OASIS,
endpointUrl: "https://emerald.oasis.dev",
apiKey: "",
urlStem: `https://explorer.emerald.oasis.dev`,
platform: "oasis", //coingecko?
covalentChain: 0,
explorerStem: `https://explorer.emerald.oasis.dev`,
},
8: {
name: "algorand",
evm: false,
chainId: CHAIN_ID_ALGORAND,
endpointUrl: "https://node.algoexplorerapi.io",
apiKey: "",
urlStem: `https://algoexplorer.io`,
platform: "algorand", //coingecko?
covalentChain: 0,
explorerStem: `https://algoexplorer.io`,
},
9: {
name: "aurora",
evm: true,
chainId: CHAIN_ID_AURORA,
endpointUrl: "https://mainnet.aurora.dev",
apiKey: "",
urlStem: `https://api.aurorascan.dev`, //?module=account&action=txlist&address={addressHash}
covalentChain: 1313161554,
platform: "aurora", //coingecko?
explorerStem: `https://aurorascan.dev`,
},
10: {
name: "fantom",
evm: true,
chainId: CHAIN_ID_FANTOM,
endpointUrl: "https://rpc.ftm.tools",
apiKey: "",
urlStem: `https://api.FtmScan.com`,
platform: "fantom", //coingecko?
covalentChain: 250,
explorerStem: `https://ftmscan.com`,
},
11: {
name: "karura",
evm: true,
chainId: CHAIN_ID_KARURA,
endpointUrl: "https://eth-rpc-karura.aca-api.network",
apiKey: "",
urlStem: `https://blockscout.karura.network`,
platform: "karura", //coingecko?
covalentChain: 0,
explorerStem: `https://blockscout.karura.network`,
},
12: {
name: "acala",
evm: true,
chainId: CHAIN_ID_ACALA,
endpointUrl: "https://eth-rpc-acala.aca-api.network",
apiKey: "",
urlStem: `https://blockscout.acala.network`,
platform: "acala", //coingecko?
covalentChain: 0,
explorerStem: `https://blockscout.acala.network`,
},
13: {
name: "klaytn",
evm: true,
chainId: CHAIN_ID_KLAYTN,
endpointUrl: "https://klaytn-mainnet-rpc.allthatnode.com:8551",
apiKey: "",
urlStem: "https://api-cypress-v2.scope.klaytn.com/v2" || "",
platform: "klay-token", //coingecko?
covalentChain: 8217,
explorerStem: `https://scope.klaytn.com`,
},
14: {
name: "celo",
evm: true,
chainId: CHAIN_ID_CELO,
endpointUrl: "https://forno.celo.org",
apiKey: "",
urlStem: `https://explorer.celo.org`,
platform: "celo", //coingecko?
covalentChain: 0,
explorerStem: `https://explorer.celo.org`,
},
15: {
name: "near",
evm: false,
chainId: CHAIN_ID_NEAR,
endpointUrl: "",
apiKey: "",
urlStem: `https://explorer.near.org`,
platform: "near", //coingecko?
covalentChain: 0,
explorerStem: `https://explorer.near.org`,
},
18: {
name: "terra2",
evm: false,
chainId: CHAIN_ID_TERRA2,
endpointUrl: "",
apiKey: "",
urlStem: "https://phoenix-fcd.terra.dev",
platform: "terra",
covalentChain: 3,
explorerStem: `https://finder.terra.money/mainnet`,
},
};

View File

@ -0,0 +1,9 @@
import { publicrpc } from "@certusone/wormhole-sdk-proto-web";
import { Network } from "../contexts/NetworkContext";
const { GrpcWebImpl, PublicRPCServiceClientImpl } = publicrpc;
export async function getGovernorAvailableNotionalByChain(network: Network) {
const rpc = new GrpcWebImpl(network.endpoint, {});
const api = new PublicRPCServiceClientImpl(rpc);
return await api.GovernorGetAvailableNotionalByChain({});
}

View File

@ -0,0 +1,9 @@
import { publicrpc } from "@certusone/wormhole-sdk-proto-web";
import { Network } from "../contexts/NetworkContext";
const { GrpcWebImpl, PublicRPCServiceClientImpl } = publicrpc;
export async function getGovernorEnqueuedVAAs(network: Network) {
const rpc = new GrpcWebImpl(network.endpoint, {});
const api = new PublicRPCServiceClientImpl(rpc);
return await api.GovernorGetEnqueuedVAAs({});
}

View File

@ -0,0 +1,9 @@
import { publicrpc } from "@certusone/wormhole-sdk-proto-web";
import { Network } from "../contexts/NetworkContext";
const { GrpcWebImpl, PublicRPCServiceClientImpl } = publicrpc;
export async function getGovernorTokenList(network: Network) {
const rpc = new GrpcWebImpl(network.endpoint, {});
const api = new PublicRPCServiceClientImpl(rpc);
return await api.GovernorGetTokenList({});
}

View File

@ -0,0 +1,9 @@
import { publicrpc } from "@certusone/wormhole-sdk-proto-web";
import { Network } from "../contexts/NetworkContext";
const { GrpcWebImpl, PublicRPCServiceClientImpl } = publicrpc;
export async function getLastHeartbeats(network: Network) {
const rpc = new GrpcWebImpl(network.endpoint, {});
const api = new PublicRPCServiceClientImpl(rpc);
return await api.GetLastHeartbeats({});
}

View File

@ -0,0 +1,14 @@
import Long from "long";
export type NumberLong = {
low: number;
high: number;
unsigned: boolean;
};
function longToDate(l: NumberLong): Date {
const value = new Long(l.low, l.high, l.unsigned);
return new Date(value.div(1000000).toNumber());
}
export default longToDate;

21
web/tsconfig.json Normal file
View File

@ -0,0 +1,21 @@
{
"compilerOptions": {
"target": "es5",
"lib": ["dom", "dom.iterable", "esnext"],
"allowJs": true,
"skipLibCheck": true,
"esModuleInterop": true,
"allowSyntheticDefaultImports": true,
"strict": true,
"forceConsistentCasingInFileNames": true,
"noFallthroughCasesInSwitch": true,
"module": "esnext",
"moduleResolution": "node",
"resolveJsonModule": true,
"isolatedModules": true,
"downlevelIteration": true,
"noEmit": true,
"jsx": "react-jsx"
},
"include": ["src"]
}