fix: active stake is an epoch off, use getVoteAccounts to sum active stake (#16465)

This commit is contained in:
Josh 2021-04-10 12:23:56 -07:00 committed by GitHub
parent ac5462e7a2
commit c4646b2bc3
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
1 changed files with 22 additions and 35 deletions

View File

@ -12,11 +12,8 @@ import { ClusterStatus, useCluster } from "providers/cluster";
import { TpsCard } from "components/TpsCard"; import { TpsCard } from "components/TpsCard";
import { displayTimestampWithoutDate, displayTimestampUtc } from "utils/date"; import { displayTimestampWithoutDate, displayTimestampUtc } from "utils/date";
import { Status, useFetchSupply, useSupply } from "providers/supply"; import { Status, useFetchSupply, useSupply } from "providers/supply";
import { PublicKey } from "@solana/web3.js";
import { ErrorCard } from "components/common/ErrorCard"; import { ErrorCard } from "components/common/ErrorCard";
import { LoadingCard } from "components/common/LoadingCard"; import { LoadingCard } from "components/common/LoadingCard";
import { useAccountInfo, useFetchAccountInfo } from "providers/accounts";
import { FetchStatus } from "providers/cache";
import { useVoteAccounts } from "providers/accounts/vote-accounts"; import { useVoteAccounts } from "providers/accounts/vote-accounts";
// @ts-ignore // @ts-ignore
import * as CoinGecko from "coingecko-api"; import * as CoinGecko from "coingecko-api";
@ -29,7 +26,6 @@ enum CoingeckoStatus {
const CoinGeckoClient = new CoinGecko(); const CoinGeckoClient = new CoinGecko();
const CLUSTER_STATS_TIMEOUT = 5000; const CLUSTER_STATS_TIMEOUT = 5000;
const STAKE_HISTORY_ACCOUNT = "SysvarStakeHistory1111111111111111111111111";
const PRICE_REFRESH = 10000; const PRICE_REFRESH = 10000;
export function ClusterStatsPage() { export function ClusterStatsPage() {
@ -55,14 +51,11 @@ function StakingComponent() {
const { status } = useCluster(); const { status } = useCluster();
const supply = useSupply(); const supply = useSupply();
const fetchSupply = useFetchSupply(); const fetchSupply = useFetchSupply();
const fetchAccount = useFetchAccountInfo();
const stakeInfo = useAccountInfo(STAKE_HISTORY_ACCOUNT);
const coinInfo = useCoinGecko("solana"); const coinInfo = useCoinGecko("solana");
const { fetchVoteAccounts, voteAccounts } = useVoteAccounts(); const { fetchVoteAccounts, voteAccounts } = useVoteAccounts();
function fetchData() { function fetchData() {
fetchSupply(); fetchSupply();
fetchAccount(new PublicKey(STAKE_HISTORY_ACCOUNT));
fetchVoteAccounts(); fetchVoteAccounts();
} }
@ -72,7 +65,7 @@ function StakingComponent() {
} }
}, [status]); // eslint-disable-line react-hooks/exhaustive-deps }, [status]); // eslint-disable-line react-hooks/exhaustive-deps
const deliquentStake = React.useMemo(() => { const delinquentStake = React.useMemo(() => {
if (voteAccounts) { if (voteAccounts) {
return voteAccounts.delinquent.reduce( return voteAccounts.delinquent.reduce(
(prev, current) => prev + current.activatedStake, (prev, current) => prev + current.activatedStake,
@ -81,46 +74,38 @@ function StakingComponent() {
} }
}, [voteAccounts]); }, [voteAccounts]);
let stakeHistory = stakeInfo?.data?.details?.data?.parsed.info; const activeStake = React.useMemo(() => {
if (voteAccounts && delinquentStake) {
return (
voteAccounts.current.reduce(
(prev, current) => prev + current.activatedStake,
0
) + delinquentStake
);
}
}, [voteAccounts, delinquentStake]);
if (supply === Status.Disconnected) { if (supply === Status.Disconnected) {
// we'll return here to prevent flicker // we'll return here to prevent flicker
return null; return null;
} }
if ( if (supply === Status.Idle || supply === Status.Connecting || !coinInfo) {
supply === Status.Idle ||
supply === Status.Connecting ||
!stakeInfo ||
!stakeHistory ||
!coinInfo
) {
return <LoadingCard />; return <LoadingCard />;
} else if (typeof supply === "string") { } else if (typeof supply === "string") {
return <ErrorCard text={supply} retry={fetchData} />; return <ErrorCard text={supply} retry={fetchData} />;
} else if (
stakeInfo.status === FetchStatus.FetchFailed ||
(stakeInfo.status === FetchStatus.Fetched &&
(!stakeHistory.length || stakeHistory.length < 1))
) {
return (
<ErrorCard text={"Failed to fetch active stake"} retry={fetchData} />
);
} }
stakeHistory = stakeHistory[0].stakeHistory;
const circulatingPercentage = ( const circulatingPercentage = (
(supply.circulating / supply.total) * (supply.circulating / supply.total) *
100 100
).toFixed(1); ).toFixed(1);
let delinquentStakePercentage; let delinquentStakePercentage;
if (deliquentStake) { if (delinquentStake && activeStake) {
delinquentStakePercentage = ( delinquentStakePercentage = ((delinquentStake / activeStake) * 100).toFixed(
(deliquentStake / stakeHistory.effective) * 1
100 );
).toFixed(1);
} }
let solanaInfo; let solanaInfo;
@ -145,10 +130,12 @@ function StakingComponent() {
<hr className="hidden-sm-up" /> <hr className="hidden-sm-up" />
<div className="p-2 flex-fill"> <div className="p-2 flex-fill">
<h4>Active Stake</h4> <h4>Active Stake</h4>
<h1> {activeStake && (
<em>{displayLamports(stakeHistory.effective)}</em> /{" "} <h1>
<small>{displayLamports(supply.total)}</small> <em>{displayLamports(activeStake)}</em> /{" "}
</h1> <small>{displayLamports(supply.total)}</small>
</h1>
)}
{delinquentStakePercentage && ( {delinquentStakePercentage && (
<h5> <h5>
Delinquent stake: <em>{delinquentStakePercentage}%</em> Delinquent stake: <em>{delinquentStakePercentage}%</em>