* fix: explorer environment * fix: stats env * fix: format * fix: remove logging
This commit is contained in:
parent
ad185651fb
commit
760a24b500
@ -3,7 +3,6 @@ NX_TENDERMINT_URL=http://localhost:26617
|
|||||||
NX_TENDERMINT_WEBSOCKET_URL=wss://localhost:26617/websocket
|
NX_TENDERMINT_WEBSOCKET_URL=wss://localhost:26617/websocket
|
||||||
NX_VEGA_URL=http://localhost:3028/query
|
NX_VEGA_URL=http://localhost:3028/query
|
||||||
NX_VEGA_ENV=CUSTOM
|
NX_VEGA_ENV=CUSTOM
|
||||||
NX_VEGA_REST=http://localhost:3029
|
|
||||||
|
|
||||||
NX_CHAIN_EXPLORER_URL=https://explorer.vega.trading/.netlify/functions/chain-explorer-api
|
NX_CHAIN_EXPLORER_URL=https://explorer.vega.trading/.netlify/functions/chain-explorer-api
|
||||||
NX_TENDERMINT_URL=https://n01.stagnet3.vega.xyz/tm
|
NX_TENDERMINT_URL=https://n01.stagnet3.vega.xyz/tm
|
||||||
@ -12,7 +11,6 @@ NX_VEGA_CONFIG_URL=https://static.vega.xyz/assets/stagnet3-network.json
|
|||||||
NX_VEGA_URL=https://n01.stagnet3.vega.xyz/query
|
NX_VEGA_URL=https://n01.stagnet3.vega.xyz/query
|
||||||
NX_VEGA_NETWORKS='{"TESTNET":"https://explorer.fairground.wtf","MAINNET":"https://explorer.vega.xyz"}'
|
NX_VEGA_NETWORKS='{"TESTNET":"https://explorer.fairground.wtf","MAINNET":"https://explorer.vega.xyz"}'
|
||||||
NX_VEGA_ENV=STAGNET3
|
NX_VEGA_ENV=STAGNET3
|
||||||
NX_VEGA_REST=https://n01.stagnet3.vega.xyz/datanode/rest
|
|
||||||
NX_GITHUB_FEEDBACK_URL=https://github.com/vegaprotocol/feedback/discussions
|
NX_GITHUB_FEEDBACK_URL=https://github.com/vegaprotocol/feedback/discussions
|
||||||
|
|
||||||
# App flags
|
# App flags
|
||||||
|
@ -5,7 +5,6 @@ NX_TENDERMINT_WEBSOCKET_URL=wss://localhost:26617/websocket
|
|||||||
NX_VEGA_URL=http://localhost:3028/query
|
NX_VEGA_URL=http://localhost:3028/query
|
||||||
NX_VEGA_NETWORKS='{"TESTNET":"https://explorer.fairground.wtf","MAINNET":"https://explorer.vega.xyz"}'
|
NX_VEGA_NETWORKS='{"TESTNET":"https://explorer.fairground.wtf","MAINNET":"https://explorer.vega.xyz"}'
|
||||||
NX_VEGA_ENV=CUSTOM
|
NX_VEGA_ENV=CUSTOM
|
||||||
NX_VEGA_REST=http://localhost:3029
|
|
||||||
NX_GITHUB_FEEDBACK_URL=https://github.com/vegaprotocol/feedback/discussions
|
NX_GITHUB_FEEDBACK_URL=https://github.com/vegaprotocol/feedback/discussions
|
||||||
|
|
||||||
# App flags
|
# App flags
|
||||||
|
@ -6,5 +6,4 @@ NX_VEGA_CONFIG_URL=https://static.vega.xyz/assets/devnet-network.json
|
|||||||
NX_VEGA_URL=https://n04.d.vega.xyz/query
|
NX_VEGA_URL=https://n04.d.vega.xyz/query
|
||||||
NX_VEGA_NETWORKS='{"TESTNET":"https://explorer.fairground.wtf","MAINNET":"https://explorer.vega.xyz"}'
|
NX_VEGA_NETWORKS='{"TESTNET":"https://explorer.fairground.wtf","MAINNET":"https://explorer.vega.xyz"}'
|
||||||
NX_VEGA_ENV=DEVNET
|
NX_VEGA_ENV=DEVNET
|
||||||
NX_VEGA_REST=https://n04.d.vega.xyz/datanode/rest
|
|
||||||
NX_GITHUB_FEEDBACK_URL=https://github.com/vegaprotocol/feedback/discussions
|
NX_GITHUB_FEEDBACK_URL=https://github.com/vegaprotocol/feedback/discussions
|
||||||
|
@ -6,5 +6,4 @@ NX_VEGA_CONFIG_URL=https://static.vega.xyz/assets/mainnet-network.json
|
|||||||
NX_VEGA_URL=https://api.token.vega.xyz/query
|
NX_VEGA_URL=https://api.token.vega.xyz/query
|
||||||
NX_VEGA_NETWORKS='{"TESTNET":"https://explorer.fairground.wtf","MAINNET":"https://explorer.vega.xyz"}'
|
NX_VEGA_NETWORKS='{"TESTNET":"https://explorer.fairground.wtf","MAINNET":"https://explorer.vega.xyz"}'
|
||||||
NX_VEGA_ENV=MAINNET
|
NX_VEGA_ENV=MAINNET
|
||||||
NX_VEGA_REST=https://api.token.vega.xyz/
|
|
||||||
NX_GITHUB_FEEDBACK_URL=https://github.com/vegaprotocol/feedback/discussions
|
NX_GITHUB_FEEDBACK_URL=https://github.com/vegaprotocol/feedback/discussions
|
||||||
|
@ -6,5 +6,4 @@ NX_VEGA_CONFIG_URL=https://static.vega.xyz/assets/stagnet1-network.json
|
|||||||
NX_VEGA_URL=https://n03.s.vega.xyz/query
|
NX_VEGA_URL=https://n03.s.vega.xyz/query
|
||||||
NX_VEGA_NETWORKS='{"TESTNET":"https://explorer.fairground.wtf","MAINNET":"https://explorer.vega.xyz"}'
|
NX_VEGA_NETWORKS='{"TESTNET":"https://explorer.fairground.wtf","MAINNET":"https://explorer.vega.xyz"}'
|
||||||
NX_VEGA_ENV=STAGNET
|
NX_VEGA_ENV=STAGNET
|
||||||
NX_VEGA_REST=https://n03.s.vega.xyz/datanode/rest
|
|
||||||
NX_GITHUB_FEEDBACK_URL=https://github.com/vegaprotocol/feedback/discussions
|
NX_GITHUB_FEEDBACK_URL=https://github.com/vegaprotocol/feedback/discussions
|
||||||
|
@ -6,5 +6,4 @@ NX_VEGA_CONFIG_URL=https://static.vega.xyz/assets/stagnet3-network.json
|
|||||||
NX_VEGA_URL=https://n01.stagnet3.vega.xyz/query
|
NX_VEGA_URL=https://n01.stagnet3.vega.xyz/query
|
||||||
NX_VEGA_NETWORKS='{"TESTNET":"https://explorer.fairground.wtf","MAINNET":"https://explorer.vega.xyz"}'
|
NX_VEGA_NETWORKS='{"TESTNET":"https://explorer.fairground.wtf","MAINNET":"https://explorer.vega.xyz"}'
|
||||||
NX_VEGA_ENV=STAGNET3
|
NX_VEGA_ENV=STAGNET3
|
||||||
NX_VEGA_REST=https://n01.stagnet3.vega.xyz/datanode/rest
|
|
||||||
NX_GITHUB_FEEDBACK_URL=https://github.com/vegaprotocol/feedback/discussions
|
NX_GITHUB_FEEDBACK_URL=https://github.com/vegaprotocol/feedback/discussions
|
||||||
|
@ -6,5 +6,4 @@ NX_VEGA_CONFIG_URL=https://static.vega.xyz/assets/testnet-network.json
|
|||||||
NX_VEGA_URL=https://api.n11.testnet.vega.xyz/graphql
|
NX_VEGA_URL=https://api.n11.testnet.vega.xyz/graphql
|
||||||
NX_VEGA_NETWORKS='{"TESTNET":"https://explorer.fairground.wtf","MAINNET":"https://explorer.vega.xyz"}'
|
NX_VEGA_NETWORKS='{"TESTNET":"https://explorer.fairground.wtf","MAINNET":"https://explorer.vega.xyz"}'
|
||||||
NX_VEGA_ENV=TESTNET
|
NX_VEGA_ENV=TESTNET
|
||||||
NX_VEGA_REST=https://api.n11.testnet.vega.xyz
|
|
||||||
NX_GITHUB_FEEDBACK_URL=https://github.com/vegaprotocol/feedback/discussions
|
NX_GITHUB_FEEDBACK_URL=https://github.com/vegaprotocol/feedback/discussions
|
||||||
|
@ -4,5 +4,4 @@ NX_TENDERMINT_URL=http://localhost:26607/
|
|||||||
NX_TENDERMINT_WEBSOCKET_URL=wss://localhost:26607/websocket
|
NX_TENDERMINT_WEBSOCKET_URL=wss://localhost:26607/websocket
|
||||||
NX_VEGA_URL=http://localhost:3003/query
|
NX_VEGA_URL=http://localhost:3003/query
|
||||||
NX_VEGA_ENV=CUSTOM
|
NX_VEGA_ENV=CUSTOM
|
||||||
NX_VEGA_REST=http://localhost:3029/rest
|
|
||||||
NX_GITHUB_FEEDBACK_URL=https://github.com/vegaprotocol/feedback/discussions
|
NX_GITHUB_FEEDBACK_URL=https://github.com/vegaprotocol/feedback/discussions
|
||||||
|
@ -47,13 +47,12 @@ yarn nx run explorer:serve --env={env} # e.g. stagnet1
|
|||||||
There are a few different configuration options offered for this app:
|
There are a few different configuration options offered for this app:
|
||||||
|
|
||||||
| **Flag** | **Purpose** |
|
| **Flag** | **Purpose** |
|
||||||
| -------------------------------- | ---------------------------------------------------------------------------------------------------- |
|
| -------------------------------- | ---------------------------------------------------------------------------------------------------- | --- |
|
||||||
| `NX_CHAIN_EXPLORER_URL` | The URL of the chain explorer service for decoding transactions |
|
| `NX_CHAIN_EXPLORER_URL` | The URL of the chain explorer service for decoding transactions |
|
||||||
| `NX_TENDERMINT_URL` | The Tendermint REST URL for the Vega consesus engine |
|
| `NX_TENDERMINT_URL` | The Tendermint REST URL for the Vega consesus engine |
|
||||||
| `NX_TENDERMINT_WEBSOCKET_URL` | The Tendermint Websocket URL for the Vega consensus engine |
|
| `NX_TENDERMINT_WEBSOCKET_URL` | The Tendermint Websocket URL for the Vega consensus engine |
|
||||||
| `NX_VEGA_URL` | The GraphQl query endpoint of a [Vega data node](https://github.com/vegaprotocol/networks#data-node) |
|
| `NX_VEGA_URL` | The GraphQl query endpoint of a [Vega data node](https://github.com/vegaprotocol/networks#data-node) |
|
||||||
| `NX_VEGA_ENV` | The name of the currently connected vega environment |
|
| `NX_VEGA_ENV` | The name of the currently connected vega environment | |
|
||||||
| `NX_VEGA_REST` | The REST URL for the Vega Data Node |
|
|
||||||
| `NX_EXPLORER_ASSETS` | Enable the assets page for the explorer |
|
| `NX_EXPLORER_ASSETS` | Enable the assets page for the explorer |
|
||||||
| `NX_EXPLORER_GENESIS` | Enable the genesis page for the explorer |
|
| `NX_EXPLORER_GENESIS` | Enable the genesis page for the explorer |
|
||||||
| `NX_EXPLORER_GOVERNANCE` | Enable the governance page for the explorer |
|
| `NX_EXPLORER_GOVERNANCE` | Enable the governance page for the explorer |
|
||||||
|
@ -1,15 +1,23 @@
|
|||||||
import { useState, useEffect } from 'react';
|
import { useState, useEffect } from 'react';
|
||||||
|
import * as Sentry from '@sentry/react';
|
||||||
|
import { BrowserTracing } from '@sentry/tracing';
|
||||||
import { useLocation } from 'react-router-dom';
|
import { useLocation } from 'react-router-dom';
|
||||||
import { ThemeContext, useThemeSwitcher } from '@vegaprotocol/react-helpers';
|
import { ThemeContext, useThemeSwitcher } from '@vegaprotocol/react-helpers';
|
||||||
import { EnvironmentProvider, NetworkLoader } from '@vegaprotocol/environment';
|
import {
|
||||||
|
EnvironmentProvider,
|
||||||
|
NetworkLoader,
|
||||||
|
useEnvironment,
|
||||||
|
} from '@vegaprotocol/environment';
|
||||||
import { NetworkInfo } from '@vegaprotocol/network-info';
|
import { NetworkInfo } from '@vegaprotocol/network-info';
|
||||||
import { createClient } from './lib/apollo-client';
|
import { createClient } from './lib/apollo-client';
|
||||||
import { Nav } from './components/nav';
|
import { Nav } from './components/nav';
|
||||||
import { Header } from './components/header';
|
import { Header } from './components/header';
|
||||||
import { Main } from './components/main';
|
import { Main } from './components/main';
|
||||||
import { TendermintWebsocketProvider } from './contexts/websocket/tendermint-websocket-provider';
|
import { TendermintWebsocketProvider } from './contexts/websocket/tendermint-websocket-provider';
|
||||||
|
import { ENV } from './config/env';
|
||||||
|
|
||||||
function App() {
|
function App() {
|
||||||
|
const { VEGA_ENV } = useEnvironment();
|
||||||
const [theme, toggleTheme] = useThemeSwitcher();
|
const [theme, toggleTheme] = useThemeSwitcher();
|
||||||
const [menuOpen, setMenuOpen] = useState(false);
|
const [menuOpen, setMenuOpen] = useState(false);
|
||||||
|
|
||||||
@ -19,34 +27,49 @@ function App() {
|
|||||||
setMenuOpen(false);
|
setMenuOpen(false);
|
||||||
}, [location]);
|
}, [location]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
Sentry.init({
|
||||||
|
dsn: ENV.dsn,
|
||||||
|
integrations: [new BrowserTracing()],
|
||||||
|
tracesSampleRate: 1,
|
||||||
|
environment: VEGA_ENV,
|
||||||
|
});
|
||||||
|
}, [VEGA_ENV]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<EnvironmentProvider>
|
<ThemeContext.Provider value={theme}>
|
||||||
<ThemeContext.Provider value={theme}>
|
<TendermintWebsocketProvider>
|
||||||
<TendermintWebsocketProvider>
|
<NetworkLoader createClient={createClient}>
|
||||||
<NetworkLoader createClient={createClient}>
|
<div
|
||||||
<div
|
className={`${
|
||||||
className={`${
|
menuOpen && 'h-[100vh] overflow-hidden'
|
||||||
menuOpen && 'h-[100vh] overflow-hidden'
|
} antialiased m-0 bg-white dark:bg-black text-black dark:text-white`}
|
||||||
} antialiased m-0 bg-white dark:bg-black text-black dark:text-white`}
|
>
|
||||||
>
|
<div className="min-h-[100vh] max-w-[1300px] grid grid-rows-[repeat(2,_auto)_1fr] grid-cols-[1fr] md:grid-rows-[auto_minmax(700px,_1fr)] md:grid-cols-[300px_1fr] border-black dark:border-white lg:border-l-1 lg:border-r-1 mx-auto">
|
||||||
<div className="min-h-[100vh] max-w-[1300px] grid grid-rows-[repeat(2,_auto)_1fr] grid-cols-[1fr] md:grid-rows-[auto_minmax(700px,_1fr)] md:grid-cols-[300px_1fr] border-black dark:border-white lg:border-l-1 lg:border-r-1 mx-auto">
|
<Header
|
||||||
<Header
|
toggleTheme={toggleTheme}
|
||||||
toggleTheme={toggleTheme}
|
menuOpen={menuOpen}
|
||||||
menuOpen={menuOpen}
|
setMenuOpen={setMenuOpen}
|
||||||
setMenuOpen={setMenuOpen}
|
/>
|
||||||
/>
|
<Nav menuOpen={menuOpen} />
|
||||||
<Nav menuOpen={menuOpen} />
|
<Main />
|
||||||
<Main />
|
<footer className="grid grid-rows-2 grid-cols-[1fr_auto] md:flex md:col-span-2 p-16 gap-12 border-t-1">
|
||||||
<footer className="grid grid-rows-2 grid-cols-[1fr_auto] md:flex md:col-span-2 p-16 gap-12 border-t-1">
|
<NetworkInfo />
|
||||||
<NetworkInfo />
|
</footer>
|
||||||
</footer>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
</NetworkLoader>
|
</div>
|
||||||
</TendermintWebsocketProvider>
|
</NetworkLoader>
|
||||||
</ThemeContext.Provider>
|
</TendermintWebsocketProvider>
|
||||||
</EnvironmentProvider>
|
</ThemeContext.Provider>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
export default App;
|
const Wrapper = () => {
|
||||||
|
return (
|
||||||
|
<EnvironmentProvider>
|
||||||
|
<App />
|
||||||
|
</EnvironmentProvider>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default Wrapper;
|
||||||
|
@ -11,13 +11,10 @@ export const ENV = {
|
|||||||
// Data sources
|
// Data sources
|
||||||
// Environment
|
// Environment
|
||||||
dsn: windowOrDefault('NX_EXPLORER_SENTRY_DSN'),
|
dsn: windowOrDefault('NX_EXPLORER_SENTRY_DSN'),
|
||||||
envName: windowOrDefault('NX_VEGA_ENV'),
|
|
||||||
dataSources: {
|
dataSources: {
|
||||||
chainExplorerUrl: windowOrDefault('NX_CHAIN_EXPLORER_URL'),
|
chainExplorerUrl: windowOrDefault('NX_CHAIN_EXPLORER_URL'),
|
||||||
tendermintUrl: windowOrDefault('NX_TENDERMINT_URL'),
|
tendermintUrl: windowOrDefault('NX_TENDERMINT_URL'),
|
||||||
tendermintWebsocketUrl: windowOrDefault('NX_TENDERMINT_WEBSOCKET_URL'),
|
tendermintWebsocketUrl: windowOrDefault('NX_TENDERMINT_WEBSOCKET_URL'),
|
||||||
dataNodeUrl: windowOrDefault('NX_VEGA_URL'),
|
|
||||||
restEndpoint: windowOrDefault('NX_VEGA_REST'),
|
|
||||||
},
|
},
|
||||||
flags: {
|
flags: {
|
||||||
assets: truthy.includes(windowOrDefault('NX_EXPLORER_ASSETS')),
|
assets: truthy.includes(windowOrDefault('NX_EXPLORER_ASSETS')),
|
||||||
|
@ -1,21 +1,10 @@
|
|||||||
import { DATA_SOURCES } from '../../config';
|
|
||||||
import { StatsManager } from '@vegaprotocol/network-stats';
|
import { StatsManager } from '@vegaprotocol/network-stats';
|
||||||
import { ENV } from '../../config/env';
|
|
||||||
|
|
||||||
const envName = ENV.envName;
|
|
||||||
const restEndpoint = DATA_SOURCES.restEndpoint;
|
|
||||||
const statsEndpoint = `${restEndpoint}/statistics`;
|
|
||||||
const nodesEndpoint = `${restEndpoint}/nodes-data`;
|
|
||||||
|
|
||||||
const Home = () => {
|
const Home = () => {
|
||||||
|
const classnames = 'mt-12 grid grid-cols-1 lg:grid-cols-2 lg:gap-16';
|
||||||
return (
|
return (
|
||||||
<section>
|
<section>
|
||||||
<StatsManager
|
<StatsManager className={classnames} />
|
||||||
envName={envName}
|
|
||||||
statsEndpoint={statsEndpoint}
|
|
||||||
nodesEndpoint={nodesEndpoint}
|
|
||||||
className="mt-12 grid grid-cols-1 lg:grid-cols-2 lg:gap-16"
|
|
||||||
/>
|
|
||||||
</section>
|
</section>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
@ -1,24 +1,10 @@
|
|||||||
import * as Sentry from '@sentry/react';
|
|
||||||
import { BrowserTracing } from '@sentry/tracing';
|
|
||||||
import { createRoot } from 'react-dom/client';
|
import { createRoot } from 'react-dom/client';
|
||||||
import { BrowserRouter } from 'react-router-dom';
|
import { BrowserRouter } from 'react-router-dom';
|
||||||
import './styles.css';
|
import './styles.css';
|
||||||
|
|
||||||
import App from './app/app';
|
import App from './app/app';
|
||||||
import { ENV } from './app/config/env';
|
|
||||||
import { StrictMode } from 'react';
|
import { StrictMode } from 'react';
|
||||||
|
|
||||||
const { dsn } = ENV;
|
|
||||||
|
|
||||||
/* istanbul ignore next */
|
|
||||||
if (dsn) {
|
|
||||||
Sentry.init({
|
|
||||||
dsn,
|
|
||||||
integrations: [new BrowserTracing()],
|
|
||||||
tracesSampleRate: 1,
|
|
||||||
environment: ENV.envName,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
const rootElement = document.getElementById('root');
|
const rootElement = document.getElementById('root');
|
||||||
const root = rootElement && createRoot(rootElement);
|
const root = rootElement && createRoot(rootElement);
|
||||||
|
|
||||||
|
@ -1,3 +1,3 @@
|
|||||||
# App configuration variables
|
# App configuration variables
|
||||||
NX_VEGA_ENV=MAINNET
|
NX_VEGA_ENV=MAINNET
|
||||||
NX_VEGA_REST=https://api.token.vega.xyz/
|
NX_VEGA_URL=https://api.token.vega.xyz/query
|
||||||
|
@ -1,3 +1,3 @@
|
|||||||
# App configuration variables
|
# App configuration variables
|
||||||
NX_VEGA_ENV=DEVNET
|
NX_VEGA_ENV=DEVNET
|
||||||
NX_VEGA_REST=https://n04.d.vega.xyz/datanode/rest
|
NX_VEGA_URL=https://n04.d.vega.xyz/query
|
||||||
|
@ -1,3 +1,3 @@
|
|||||||
# App configuration variables. No overrides for default urls
|
# App configuration variables. No overrides for default urls
|
||||||
NX_VEGA_ENV=MAINNET
|
NX_VEGA_ENV=MAINNET
|
||||||
NX_VEGA_REST=https://api.token.vega.xyz/
|
NX_VEGA_URL=https://api.token.vega.xyz/query
|
||||||
|
@ -1,3 +1,3 @@
|
|||||||
# App configuration variables
|
# App configuration variables
|
||||||
NX_VEGA_ENV=STAGNET1;
|
NX_VEGA_ENV=STAGNET1;
|
||||||
NX_VEGA_REST=https://n03.s.vega.xyz/datanode/rest;
|
NX_VEGA_URL=https://n03.s.vega.xyz/query;
|
||||||
|
@ -1,3 +1,3 @@
|
|||||||
# App configuration variables
|
# App configuration variables
|
||||||
NX_VEGA_REST=https://n01.stagnet3.vega.xyz/datanode/rest;
|
NX_VEGA_URL=https://n01.stagnet3.vega.xyz/query;
|
||||||
NX_VEGA_ENV=STAGNET3;
|
NX_VEGA_ENV=STAGNET3;
|
||||||
|
@ -1,3 +1,3 @@
|
|||||||
# App configuration variables
|
# App configuration variables
|
||||||
NX_VEGA_ENV=TESTNET
|
NX_VEGA_ENV=TESTNET
|
||||||
NX_VEGA_REST=https://api.n11.testnet.vega.xyz
|
NX_VEGA_URL=https://api.n11.testnet.vega.xyz/graphql
|
||||||
|
@ -1,33 +1,33 @@
|
|||||||
import React from 'react';
|
import { EnvironmentProvider, NetworkLoader } from '@vegaprotocol/environment';
|
||||||
import { DATA_SOURCES } from './config';
|
|
||||||
import { Header } from './components/header';
|
import { Header } from './components/header';
|
||||||
import { StatsManager } from '@vegaprotocol/network-stats';
|
import { StatsManager } from '@vegaprotocol/network-stats';
|
||||||
import { ThemeContext } from '@vegaprotocol/react-helpers';
|
import { ThemeContext } from '@vegaprotocol/react-helpers';
|
||||||
import { useThemeSwitcher } from '@vegaprotocol/react-helpers';
|
import { useThemeSwitcher } from '@vegaprotocol/react-helpers';
|
||||||
|
import { createClient } from './lib/apollo-client';
|
||||||
const envName = DATA_SOURCES.envName;
|
|
||||||
const restEndpoint = DATA_SOURCES.restEndpoint;
|
|
||||||
const statsEndpoint = `${restEndpoint}/statistics`;
|
|
||||||
const nodesEndpoint = `${restEndpoint}/nodes-data`;
|
|
||||||
|
|
||||||
function App() {
|
function App() {
|
||||||
const [theme, toggleTheme] = useThemeSwitcher();
|
const [theme, toggleTheme] = useThemeSwitcher();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<ThemeContext.Provider value={theme}>
|
<ThemeContext.Provider value={theme}>
|
||||||
<div className="w-screen min-h-screen grid pb-24 bg-white text-black-95 dark:bg-black dark:text-white-80">
|
<NetworkLoader createClient={createClient}>
|
||||||
<div className="layout-grid w-screen justify-self-center">
|
<div className="w-screen min-h-screen grid pb-24 bg-white text-black-95 dark:bg-black dark:text-white-80">
|
||||||
<Header toggleTheme={toggleTheme} />
|
<div className="layout-grid w-screen justify-self-center">
|
||||||
<StatsManager
|
<Header toggleTheme={toggleTheme} />
|
||||||
envName={envName}
|
<StatsManager className="max-w-3xl px-24" />
|
||||||
statsEndpoint={statsEndpoint}
|
</div>
|
||||||
nodesEndpoint={nodesEndpoint}
|
|
||||||
className="max-w-3xl px-24"
|
|
||||||
/>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</NetworkLoader>
|
||||||
</ThemeContext.Provider>
|
</ThemeContext.Provider>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
export default App;
|
const Wrapper = () => {
|
||||||
|
return (
|
||||||
|
<EnvironmentProvider>
|
||||||
|
<App />
|
||||||
|
</EnvironmentProvider>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default Wrapper;
|
||||||
|
@ -1,4 +0,0 @@
|
|||||||
export const DATA_SOURCES = {
|
|
||||||
envName: process.env['NX_VEGA_ENV'] as string,
|
|
||||||
restEndpoint: process.env['NX_VEGA_REST'] as string,
|
|
||||||
};
|
|
52
apps/stats/src/lib/apollo-client.tsx
Normal file
52
apps/stats/src/lib/apollo-client.tsx
Normal file
@ -0,0 +1,52 @@
|
|||||||
|
import { ApolloClient, from, HttpLink, InMemoryCache } from '@apollo/client';
|
||||||
|
import { onError } from '@apollo/client/link/error';
|
||||||
|
import { RetryLink } from '@apollo/client/link/retry';
|
||||||
|
|
||||||
|
export function createClient(base?: string) {
|
||||||
|
if (!base) {
|
||||||
|
throw new Error('Base must be passed into createClient!');
|
||||||
|
}
|
||||||
|
const urlHTTP = new URL(base);
|
||||||
|
const urlWS = new URL(base);
|
||||||
|
// Replace http with ws, preserving if its a secure connection eg. https => wss
|
||||||
|
urlWS.protocol = urlWS.protocol.replace('http', 'ws');
|
||||||
|
|
||||||
|
const cache = new InMemoryCache({
|
||||||
|
typePolicies: {
|
||||||
|
Query: {},
|
||||||
|
Account: {
|
||||||
|
keyFields: false,
|
||||||
|
fields: {
|
||||||
|
balanceFormatted: {},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
Node: {
|
||||||
|
keyFields: false,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const retryLink = new RetryLink({
|
||||||
|
delay: {
|
||||||
|
initial: 300,
|
||||||
|
max: 10000,
|
||||||
|
jitter: true,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const httpLink = new HttpLink({
|
||||||
|
uri: urlHTTP.href,
|
||||||
|
credentials: 'same-origin',
|
||||||
|
});
|
||||||
|
|
||||||
|
const errorLink = onError(({ graphQLErrors, networkError }) => {
|
||||||
|
console.log(graphQLErrors);
|
||||||
|
console.log(networkError);
|
||||||
|
});
|
||||||
|
|
||||||
|
return new ApolloClient({
|
||||||
|
connectToDevTools: process.env['NODE_ENV'] === 'development',
|
||||||
|
link: from([errorLink, retryLink, httpLink]),
|
||||||
|
cache,
|
||||||
|
});
|
||||||
|
}
|
@ -18,7 +18,7 @@ const compileHosts = (hosts: string[], envUrl?: string) => {
|
|||||||
|
|
||||||
const getCacheKey = (env: Networks) => `${LOCAL_STORAGE_NETWORK_KEY}-${env}`;
|
const getCacheKey = (env: Networks) => `${LOCAL_STORAGE_NETWORK_KEY}-${env}`;
|
||||||
|
|
||||||
const getCachedConfig = (env: Networks) => {
|
const getCachedConfig = (env: Networks, envUrl?: string) => {
|
||||||
const cacheKey = getCacheKey(env);
|
const cacheKey = getCacheKey(env);
|
||||||
const value = LocalStorage.getItem(cacheKey);
|
const value = LocalStorage.getItem(cacheKey);
|
||||||
|
|
||||||
@ -31,7 +31,10 @@ const getCachedConfig = (env: Networks) => {
|
|||||||
throw new Error('Invalid configuration found in the storage.');
|
throw new Error('Invalid configuration found in the storage.');
|
||||||
}
|
}
|
||||||
|
|
||||||
return config;
|
return {
|
||||||
|
...config,
|
||||||
|
hosts: compileHosts(config.hosts, envUrl),
|
||||||
|
};
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
LocalStorage.removeItem(cacheKey);
|
LocalStorage.removeItem(cacheKey);
|
||||||
console.warn(
|
console.warn(
|
||||||
@ -49,7 +52,7 @@ export const useConfig = (
|
|||||||
) => {
|
) => {
|
||||||
const [loading, setLoading] = useState(false);
|
const [loading, setLoading] = useState(false);
|
||||||
const [config, setConfig] = useState<Configuration | undefined>(
|
const [config, setConfig] = useState<Configuration | undefined>(
|
||||||
getCachedConfig(environment.VEGA_ENV)
|
getCachedConfig(environment.VEGA_ENV, environment.VEGA_URL)
|
||||||
);
|
);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
|
@ -110,11 +110,10 @@ export const getErrorByType = (
|
|||||||
};
|
};
|
||||||
|
|
||||||
export const getErrorType = (env: Networks, data?: NodeData) => {
|
export const getErrorType = (env: Networks, data?: NodeData) => {
|
||||||
if (data && !getIsNodeLoading(data) && data.initialized) {
|
if (data && data.initialized) {
|
||||||
if (getIsInvalidUrl(data.url)) {
|
if (getIsInvalidUrl(data.url)) {
|
||||||
return ErrorType.INVALID_URL;
|
return ErrorType.INVALID_URL;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (
|
if (
|
||||||
data.chain.hasError ||
|
data.chain.hasError ||
|
||||||
data.responseTime.hasError ||
|
data.responseTime.hasError ||
|
||||||
@ -123,7 +122,7 @@ export const getErrorType = (env: Networks, data?: NodeData) => {
|
|||||||
return ErrorType.CONNECTION_ERROR;
|
return ErrorType.CONNECTION_ERROR;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (getHasInvalidChain(env, data.chain.value)) {
|
if (!data.chain.isLoading && getHasInvalidChain(env, data.chain.value)) {
|
||||||
return ErrorType.INVALID_NETWORK;
|
return ErrorType.INVALID_NETWORK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"extends": ["plugin:@nrwl/nx/react", "../../.eslintrc.json"],
|
"extends": ["plugin:@nrwl/nx/react", "../../.eslintrc.json"],
|
||||||
"ignorePatterns": ["!**/*"],
|
"ignorePatterns": ["!**/*", "__generated__"],
|
||||||
"overrides": [
|
"overrides": [
|
||||||
{
|
{
|
||||||
"files": ["*.ts", "*.tsx", "*.js", "*.jsx"],
|
"files": ["*.ts", "*.tsx", "*.js", "*.jsx"],
|
||||||
|
95
libs/network-stats/src/components/stats-manager/__generated__/NetworkStats.ts
generated
Normal file
95
libs/network-stats/src/components/stats-manager/__generated__/NetworkStats.ts
generated
Normal file
@ -0,0 +1,95 @@
|
|||||||
|
/* tslint:disable */
|
||||||
|
/* eslint-disable */
|
||||||
|
// @generated
|
||||||
|
// This file was automatically generated and should not be edited.
|
||||||
|
|
||||||
|
// ====================================================
|
||||||
|
// GraphQL query operation: NetworkStats
|
||||||
|
// ====================================================
|
||||||
|
|
||||||
|
export interface NetworkStats_nodeData {
|
||||||
|
__typename: "NodeData";
|
||||||
|
/**
|
||||||
|
* Total staked amount across all nodes
|
||||||
|
*/
|
||||||
|
stakedTotal: string;
|
||||||
|
/**
|
||||||
|
* Total number of nodes
|
||||||
|
*/
|
||||||
|
totalNodes: number;
|
||||||
|
/**
|
||||||
|
* Number of inactive nodes
|
||||||
|
*/
|
||||||
|
inactiveNodes: number;
|
||||||
|
/**
|
||||||
|
* Number of nodes validating
|
||||||
|
*/
|
||||||
|
validatingNodes: number;
|
||||||
|
/**
|
||||||
|
* Total uptime for all epochs across all nodes. Or specify a number of epochs
|
||||||
|
*/
|
||||||
|
uptime: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface NetworkStats_statistics {
|
||||||
|
__typename: "Statistics";
|
||||||
|
/**
|
||||||
|
* Status of the Vega application connection with the chain
|
||||||
|
*/
|
||||||
|
status: string;
|
||||||
|
/**
|
||||||
|
* Current block number
|
||||||
|
*/
|
||||||
|
blockHeight: string;
|
||||||
|
/**
|
||||||
|
* Duration of the last block, in nanoseconds
|
||||||
|
*/
|
||||||
|
blockDuration: string;
|
||||||
|
/**
|
||||||
|
* Number of items in the backlog
|
||||||
|
*/
|
||||||
|
backlogLength: string;
|
||||||
|
/**
|
||||||
|
* Number of transaction processed per block
|
||||||
|
*/
|
||||||
|
txPerBlock: string;
|
||||||
|
/**
|
||||||
|
* Number of the trades per seconds
|
||||||
|
*/
|
||||||
|
tradesPerSecond: string;
|
||||||
|
/**
|
||||||
|
* Number of orders per seconds
|
||||||
|
*/
|
||||||
|
ordersPerSecond: string;
|
||||||
|
/**
|
||||||
|
* Average number of orders added per blocks
|
||||||
|
*/
|
||||||
|
averageOrdersPerBlock: string;
|
||||||
|
/**
|
||||||
|
* RFC3339Nano current time of the chain (decided through consensus)
|
||||||
|
*/
|
||||||
|
vegaTime: string;
|
||||||
|
/**
|
||||||
|
* Version of the Vega node (semver)
|
||||||
|
*/
|
||||||
|
appVersion: string;
|
||||||
|
/**
|
||||||
|
* Version of the chain (semver)
|
||||||
|
*/
|
||||||
|
chainVersion: string;
|
||||||
|
/**
|
||||||
|
* Current chain ID
|
||||||
|
*/
|
||||||
|
chainId: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface NetworkStats {
|
||||||
|
/**
|
||||||
|
* returns information about nodes
|
||||||
|
*/
|
||||||
|
nodeData: NetworkStats_nodeData | null;
|
||||||
|
/**
|
||||||
|
* get statistics about the Vega node
|
||||||
|
*/
|
||||||
|
statistics: NetworkStats_statistics;
|
||||||
|
}
|
@ -1,5 +1,7 @@
|
|||||||
import { useEffect, useState } from 'react';
|
import { useEffect } from 'react';
|
||||||
import classnames from 'classnames';
|
import classnames from 'classnames';
|
||||||
|
import { useQuery, gql } from '@apollo/client';
|
||||||
|
import { useEnvironment } from '@vegaprotocol/environment';
|
||||||
import { statsFields } from '../../config/stats-fields';
|
import { statsFields } from '../../config/stats-fields';
|
||||||
import type {
|
import type {
|
||||||
Stats as IStats,
|
Stats as IStats,
|
||||||
@ -9,75 +11,75 @@ import { Table } from '../table';
|
|||||||
import { TableRow } from '../table-row';
|
import { TableRow } from '../table-row';
|
||||||
import { PromotedStats } from '../promoted-stats';
|
import { PromotedStats } from '../promoted-stats';
|
||||||
import { PromotedStatsItem } from '../promoted-stats-item';
|
import { PromotedStatsItem } from '../promoted-stats-item';
|
||||||
|
import type { NetworkStats } from './__generated__/NetworkStats';
|
||||||
|
|
||||||
interface StatsManagerProps {
|
interface StatsManagerProps {
|
||||||
envName: string;
|
|
||||||
statsEndpoint: string;
|
|
||||||
nodesEndpoint: string;
|
|
||||||
className?: string;
|
className?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const StatsManager = ({
|
const STATS_QUERY = gql`
|
||||||
envName,
|
query NetworkStats {
|
||||||
statsEndpoint,
|
nodeData {
|
||||||
nodesEndpoint,
|
stakedTotal
|
||||||
className,
|
totalNodes
|
||||||
}: StatsManagerProps) => {
|
inactiveNodes
|
||||||
const [data, setData] = useState<IStructuredStats | null>(null);
|
validatingNodes
|
||||||
const [error, setError] = useState<Error | null>(null);
|
uptime
|
||||||
|
}
|
||||||
|
statistics {
|
||||||
|
status
|
||||||
|
blockHeight
|
||||||
|
blockDuration
|
||||||
|
backlogLength
|
||||||
|
txPerBlock
|
||||||
|
tradesPerSecond
|
||||||
|
ordersPerSecond
|
||||||
|
averageOrdersPerBlock
|
||||||
|
vegaTime
|
||||||
|
appVersion
|
||||||
|
chainVersion
|
||||||
|
chainId
|
||||||
|
}
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
|
||||||
|
const compileData = (data?: NetworkStats) => {
|
||||||
|
const { nodeData, statistics } = data || {};
|
||||||
|
const returned = { ...nodeData, ...statistics };
|
||||||
|
|
||||||
|
// Loop through the stats fields config, grabbing values from the fetched
|
||||||
|
// data and building a set of promoted and standard table entries.
|
||||||
|
return Object.entries(statsFields).reduce(
|
||||||
|
(acc, [key, value]) => {
|
||||||
|
const statKey = key as keyof IStats;
|
||||||
|
const statData = returned[statKey];
|
||||||
|
|
||||||
|
value.forEach((x) => {
|
||||||
|
const stat = {
|
||||||
|
...x,
|
||||||
|
value: statData,
|
||||||
|
};
|
||||||
|
|
||||||
|
stat.promoted ? acc.promoted.push(stat) : acc.table.push(stat);
|
||||||
|
});
|
||||||
|
|
||||||
|
return acc;
|
||||||
|
},
|
||||||
|
{ promoted: [], table: [] } as IStructuredStats
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export const StatsManager = ({ className }: StatsManagerProps) => {
|
||||||
|
const { VEGA_ENV } = useEnvironment();
|
||||||
|
const { data, error, startPolling, stopPolling } =
|
||||||
|
useQuery<NetworkStats>(STATS_QUERY);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
async function getStats() {
|
startPolling(1000);
|
||||||
try {
|
return () => stopPolling();
|
||||||
const [res1, res2] = await Promise.all([
|
});
|
||||||
fetch(statsEndpoint),
|
|
||||||
fetch(nodesEndpoint),
|
|
||||||
]);
|
|
||||||
const [{ statistics }, { nodeData }] = await Promise.all([
|
|
||||||
res1.json(),
|
|
||||||
res2.json(),
|
|
||||||
]);
|
|
||||||
const returned = { ...nodeData, ...statistics };
|
|
||||||
|
|
||||||
if (!statistics || !nodeData) {
|
const displayData = compileData(data);
|
||||||
throw new Error('Failed to get data from endpoints');
|
|
||||||
}
|
|
||||||
|
|
||||||
// Loop through the stats fields config, grabbing values from the fetched
|
|
||||||
// data and building a set of promoted and standard table entries.
|
|
||||||
const structured = Object.entries(statsFields).reduce(
|
|
||||||
(acc, [key, value]) => {
|
|
||||||
const statKey = key as keyof IStats;
|
|
||||||
const statData = returned[statKey];
|
|
||||||
|
|
||||||
value.forEach((x) => {
|
|
||||||
const stat = {
|
|
||||||
...x,
|
|
||||||
value: statData,
|
|
||||||
};
|
|
||||||
|
|
||||||
stat.promoted ? acc.promoted.push(stat) : acc.table.push(stat);
|
|
||||||
});
|
|
||||||
|
|
||||||
return acc;
|
|
||||||
},
|
|
||||||
{ promoted: [], table: [] } as IStructuredStats
|
|
||||||
);
|
|
||||||
|
|
||||||
setData(structured);
|
|
||||||
setError(null);
|
|
||||||
} catch (e) {
|
|
||||||
setData(null);
|
|
||||||
setError(e as Error);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const interval = setInterval(getStats, 1000);
|
|
||||||
|
|
||||||
return () => {
|
|
||||||
clearInterval(interval);
|
|
||||||
};
|
|
||||||
}, [nodesEndpoint, statsEndpoint]);
|
|
||||||
|
|
||||||
const classes = classnames(
|
const classes = classnames(
|
||||||
className,
|
className,
|
||||||
@ -90,12 +92,13 @@ export const StatsManager = ({
|
|||||||
data-testid="stats-environment"
|
data-testid="stats-environment"
|
||||||
className="font-alpha uppercase text-h3 pb-16 col-span-full"
|
className="font-alpha uppercase text-h3 pb-16 col-span-full"
|
||||||
>
|
>
|
||||||
{(error && `/ ${error}`) || (data ? `/ ${envName}` : '/ Connecting...')}
|
{(error && `/ ${error}`) ||
|
||||||
|
(data ? `/ ${VEGA_ENV}` : '/ Connecting...')}
|
||||||
</h3>
|
</h3>
|
||||||
|
|
||||||
{data?.promoted ? (
|
{displayData?.promoted ? (
|
||||||
<PromotedStats>
|
<PromotedStats>
|
||||||
{data.promoted.map((stat, i) => {
|
{displayData.promoted.map((stat, i) => {
|
||||||
return (
|
return (
|
||||||
<PromotedStatsItem
|
<PromotedStatsItem
|
||||||
title={stat.title}
|
title={stat.title}
|
||||||
@ -111,8 +114,8 @@ export const StatsManager = ({
|
|||||||
) : null}
|
) : null}
|
||||||
|
|
||||||
<Table>
|
<Table>
|
||||||
{data?.table
|
{displayData?.table
|
||||||
? data.table.map((stat, i) => {
|
? displayData.table.map((stat, i) => {
|
||||||
return (
|
return (
|
||||||
<TableRow
|
<TableRow
|
||||||
title={stat.title}
|
title={stat.title}
|
||||||
|
@ -1,10 +1,10 @@
|
|||||||
import { addDecimalsFormatNumber, t } from '@vegaprotocol/react-helpers';
|
import { addDecimalsFormatNumber, t } from '@vegaprotocol/react-helpers';
|
||||||
import type { Stats as IStats, StatFields as IStatFields } from './types';
|
import type { Stats, StatFields } from './types';
|
||||||
|
|
||||||
// Stats fields config. Keys will correspond to graphql queries when used, and values
|
// Stats fields config. Keys will correspond to graphql queries when used, and values
|
||||||
// contain the associated data and methods we need to render. A single query
|
// contain the associated data and methods we need to render. A single query
|
||||||
// can be rendered in multiple ways (see 'upTime').
|
// can be rendered in multiple ways (see 'upTime').
|
||||||
export const statsFields: { [key in keyof IStats]: IStatFields[] } = {
|
export const statsFields: { [key in keyof Stats]: StatFields[] } = {
|
||||||
status: [
|
status: [
|
||||||
{
|
{
|
||||||
title: t('Status'),
|
title: t('Status'),
|
||||||
@ -17,7 +17,7 @@ export const statsFields: { [key in keyof IStats]: IStatFields[] } = {
|
|||||||
if (i === -1) {
|
if (i === -1) {
|
||||||
return status;
|
return status;
|
||||||
} else {
|
} else {
|
||||||
return status.substr(i + 1);
|
return status.substring(i + 1);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
goodThreshold: (status: string) =>
|
goodThreshold: (status: string) =>
|
||||||
|
@ -1,22 +1,10 @@
|
|||||||
export interface Stats {
|
import type {
|
||||||
blockHeight: string;
|
NetworkStats_nodeData,
|
||||||
totalNodes: string;
|
NetworkStats_statistics,
|
||||||
validatingNodes: string;
|
} from '../components/stats-manager/__generated__/NetworkStats';
|
||||||
inactiveNodes: string;
|
|
||||||
stakedTotal: string;
|
export type Stats = Omit<NetworkStats_nodeData, '__typename'> &
|
||||||
backlogLength: string;
|
Omit<NetworkStats_statistics, '__typename'>;
|
||||||
tradesPerSecond: string;
|
|
||||||
averageOrdersPerBlock: string;
|
|
||||||
ordersPerSecond: string;
|
|
||||||
txPerBlock: string;
|
|
||||||
blockDuration: string;
|
|
||||||
status: string;
|
|
||||||
vegaTime: string;
|
|
||||||
appVersion: string;
|
|
||||||
chainVersion: string;
|
|
||||||
uptime: string;
|
|
||||||
chainId: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
// eslint-disable-next-line
|
// eslint-disable-next-line
|
||||||
export type value = any;
|
export type value = any;
|
||||||
|
Loading…
Reference in New Issue
Block a user