mango-v4-ui/components/StatusBar.tsx

256 lines
7.7 KiB
TypeScript
Raw Normal View History

2023-08-06 22:57:02 -07:00
import { useTranslation } from 'react-i18next'
2023-10-06 02:13:07 -07:00
import Tps, { StatusDot } from './Tps'
2023-08-06 22:57:02 -07:00
import DiscordIcon from './icons/DiscordIcon'
import { TwitterIcon } from './icons/TwitterIcon'
import { DocumentTextIcon } from '@heroicons/react/20/solid'
2023-10-08 19:05:52 -07:00
import { useEffect, useMemo, useState } from 'react'
2023-08-07 04:07:34 -07:00
import { IDL } from '@blockworks-foundation/mango-v4'
2023-08-09 05:33:42 -07:00
import RpcPing from './RpcPing'
import Tooltip from './shared/Tooltip'
2023-09-12 20:39:35 -07:00
import { useRouter } from 'next/router'
2023-10-06 02:13:07 -07:00
import useOffchainServicesHealth from 'hooks/useOffchainServicesHealth'
2023-10-08 19:05:52 -07:00
import mangoStore from '@store/mangoStore'
import { Connection } from '@solana/web3.js'
import { sumBy } from 'lodash'
import useInterval from './shared/useInterval'
2023-08-06 22:57:02 -07:00
const DEFAULT_LATEST_COMMIT = { sha: '', url: '' }
2023-10-08 19:05:52 -07:00
export const tpsAlertThreshold = 1300
export const tpsWarningThreshold = 1000
export const rpcAlertThreshold = 250
export const rpcWarningThreshold = 500
2023-08-06 22:57:02 -07:00
const getLatestCommit = async () => {
try {
const response = await fetch(
2023-08-07 04:07:34 -07:00
`https://api.github.com/repos/blockworks-foundation/mango-v4-ui/commits`,
2023-08-06 22:57:02 -07:00
)
const data = await response.json()
if (data && data.length) {
const { sha, html_url } = data[0]
return {
sha: sha.slice(0, 7),
url: html_url,
}
}
return DEFAULT_LATEST_COMMIT
} catch (error) {
console.error('Error fetching latest commit:', error)
return DEFAULT_LATEST_COMMIT
}
}
2023-10-08 19:05:52 -07:00
const getRecentPerformance = async (
connection: Connection,
setTps: (x: number) => void,
) => {
try {
const samples = 2
const response = await connection.getRecentPerformanceSamples(samples)
const totalSecs = sumBy(response, 'samplePeriodSecs')
const totalTransactions = sumBy(response, 'numTransactions')
const tps = totalTransactions / totalSecs
setTps(tps)
} catch {
console.warn('Unable to fetch TPS')
}
}
const getPingTime = async (
connection: Connection,
setRpcPing: (x: number) => void,
) => {
const startTime = Date.now()
try {
await connection.getSlot()
const endTime = Date.now()
const pingTime = endTime - startTime
setRpcPing(pingTime)
} catch (error) {
console.error('Error pinging the RPC:', error)
return null
}
}
const getOverallStatus = (
tps: number,
rpcPing: number,
offchainHealth: number,
) => {
if (tps < tpsWarningThreshold) {
return 'severly-degraded'
} else if (
tps < tpsAlertThreshold ||
rpcPing > rpcWarningThreshold ||
offchainHealth !== 200
) {
return 'degraded'
} else return 'operational'
}
2023-08-08 22:24:09 -07:00
const StatusBar = ({ collapsed }: { collapsed: boolean }) => {
2023-08-06 22:57:02 -07:00
const { t } = useTranslation('common')
const [latestCommit, setLatestCommit] = useState(DEFAULT_LATEST_COMMIT)
2023-09-12 20:39:35 -07:00
const router = useRouter()
2023-10-06 02:13:07 -07:00
const { offchainHealth, isLoading: loadingOffchainHealth } =
useOffchainServicesHealth()
2023-10-08 19:05:52 -07:00
const connection = mangoStore((s) => s.connection)
const [tps, setTps] = useState(0)
const [rpcPing, setRpcPing] = useState(0)
useEffect(() => {
getPingTime(connection, setRpcPing)
}, [])
useInterval(() => {
getPingTime(connection, setRpcPing)
}, 30 * 1000)
2023-10-06 02:13:07 -07:00
2023-10-08 19:05:52 -07:00
useEffect(() => {
getRecentPerformance(connection, setTps)
}, [])
useInterval(() => {
getRecentPerformance(connection, setTps)
}, 60 * 1000)
2023-08-06 22:57:02 -07:00
useEffect(() => {
const { sha } = latestCommit
if (!sha) {
getLatestCommit().then((commit) => setLatestCommit(commit))
}
}, [latestCommit])
2023-10-08 19:05:52 -07:00
const platformHealth = useMemo(() => {
return getOverallStatus(tps, rpcPing, offchainHealth)
}, [tps, rpcPing, offchainHealth])
const dotColor =
platformHealth === 'severly-degraded'
? 'bg-th-error'
: platformHealth === 'degraded'
? 'bg-th-warning'
: 'bg-th-success'
2023-08-06 22:57:02 -07:00
return (
2023-08-08 22:24:09 -07:00
<div
className={`fixed hidden ${
2023-08-08 22:24:09 -07:00
collapsed ? 'w-[calc(100vw-64px)]' : 'w-[calc(100vw-200px)]'
} bottom-0 z-10 bg-th-input-bkg px-4 py-1 md:grid md:grid-cols-3 md:px-6`}
2023-08-08 22:24:09 -07:00
>
2023-10-08 19:05:52 -07:00
<div className="col-span-1 flex items-center">
<Tooltip
content={
<div className="space-y-2">
<div>
<p className="mb-0.5">{t('solana-tps')}</p>
<Tps tps={tps} />
</div>
<div>
<p className="mb-0.5">{t('rpc-ping')}</p>
<RpcPing rpcPing={rpcPing} />
</div>
{!loadingOffchainHealth ? (
<div className="flex items-center">
<StatusDot
status={offchainHealth}
alert={201}
warning={301}
/>
<span className="text-xs text-th-fgd-2">
{t('offchain-services')}
</span>
</div>
) : null}
</div>
}
placement="top-start"
>
<div className="flex items-center">
<div className="relative mr-1 h-3 w-3">
<div
className={`absolute left-0.5 top-0.5 h-2 w-2 rounded-full ${dotColor}`}
/>
<div
className={`absolute h-3 w-3 rounded-full opacity-40 ${dotColor}`}
/>
2023-10-06 02:13:07 -07:00
</div>
2023-10-08 19:05:52 -07:00
<p className="tooltip-underline text-xs leading-none">
{t(platformHealth)}
</p>
</div>
</Tooltip>
2023-08-06 22:57:02 -07:00
</div>
2023-08-09 05:33:42 -07:00
<div className="col-span-1 flex items-center justify-center">
<Tooltip content={t('program-version')}>
2023-08-09 22:41:11 -07:00
<a
className="text-xs text-th-fgd-3 focus:outline-none md:hover:text-th-fgd-2"
2023-08-09 22:41:11 -07:00
href={`https://github.com/blockworks-foundation/mango-v4/releases`}
rel="noreferrer noopener"
target="_blank"
>
<span>v{IDL.version}</span>
2023-08-09 22:41:11 -07:00
</a>
2023-08-09 05:33:42 -07:00
</Tooltip>
2023-08-06 22:57:02 -07:00
{latestCommit.sha && latestCommit.url ? (
2023-08-09 05:33:42 -07:00
<Tooltip content={t('latest-ui-commit')}>
2023-08-07 04:07:34 -07:00
<span className="mx-1.5 text-th-fgd-4">|</span>
<a
className="text-xs text-th-fgd-3 focus:outline-none md:hover:text-th-fgd-2"
2023-08-07 04:07:34 -07:00
href={latestCommit.url}
rel="noreferrer noopener"
target="_blank"
>
{latestCommit.sha}
</a>
2023-08-09 05:33:42 -07:00
</Tooltip>
2023-08-06 22:57:02 -07:00
) : null}
</div>
<div className="col-span-1 flex items-center justify-end space-x-4 text-xs">
2023-09-12 20:39:35 -07:00
{router.asPath.includes('/trade') ? (
<a
className="flex items-center text-th-fgd-3 focus:outline-none md:hover:text-th-fgd-2"
href="https://www.tradingview.com/"
rel="noreferrer noopener"
target="_blank"
>
2023-09-12 22:19:10 -07:00
<span>Powered by TradingView</span>
2023-09-12 20:39:35 -07:00
</a>
) : null}
2023-08-06 22:57:02 -07:00
<a
className="flex items-center text-th-fgd-3 focus:outline-none md:hover:text-th-fgd-2"
2023-08-06 22:57:02 -07:00
href="https://docs.mango.markets"
rel="noreferrer noopener"
target="_blank"
>
<DocumentTextIcon className="mr-1 h-3 w-3" />
2023-08-06 22:57:02 -07:00
<span>{t('docs')}</span>
</a>
<a
className="flex items-center text-th-fgd-3 focus:outline-none md:hover:text-th-fgd-2"
2023-08-06 22:57:02 -07:00
href="https://discord.gg/2uwjsBc5yw"
rel="noreferrer noopener"
target="_blank"
>
<DiscordIcon className="mr-1 h-3 w-3" />
2023-08-06 22:57:02 -07:00
<span>{t('discord')}</span>
</a>
<a
className="flex items-center text-th-fgd-3 focus:outline-none md:hover:text-th-fgd-2"
2023-08-06 22:57:02 -07:00
href="https://twitter.com/mangomarkets"
rel="noreferrer noopener"
target="_blank"
>
<TwitterIcon className="mr-1 h-3 w-3" />
2023-08-06 22:57:02 -07:00
<span>{t('twitter')}</span>
</a>
</div>
</div>
)
}
export default StatusBar