From 3d6855ff1db3fe86ad23894abc7ab34e10e481c3 Mon Sep 17 00:00:00 2001 From: Antonio Morrone Date: Mon, 16 Dec 2024 15:37:03 +0000 Subject: [PATCH] TOK-541: add ABI (#472) * feat(ABI): annual backers incentives * refactor: pr comments --------- Co-authored-by: Francisco Tobar --- .../collective-rewards/metrics/Metrics.tsx | 14 +- .../metrics/TotalActiveBuildersMetrics.tsx | 3 +- .../metrics/TotalAllocationsMetrics.tsx | 5 +- .../components/ABIMetrics/ABIMetrics.tsx | 43 +++++++ .../components/ABIMetrics/hooks/useGetABI.ts | 120 ++++++++++++++++++ .../metrics/components/ABIMetrics/index.ts | 1 + .../metrics/hooks/useGetTotalAllocation.ts | 12 ++ .../collective-rewards/rewards/MyRewards.tsx | 10 +- .../rewards/backers/ClaimableRewards.tsx | 1 - .../backers/hooks/useGetBackerRewards.ts | 4 +- .../rewards/builders/EstimatedRewards.tsx | 11 +- .../builders/hooks/useGetBuildersRewards.ts | 27 ++-- src/app/collective-rewards/types.ts | 2 + .../utils/isBuilderOperational.ts | 10 ++ src/components/Popover/Popover.tsx | 5 +- 15 files changed, 230 insertions(+), 38 deletions(-) create mode 100644 src/app/collective-rewards/metrics/components/ABIMetrics/ABIMetrics.tsx create mode 100644 src/app/collective-rewards/metrics/components/ABIMetrics/hooks/useGetABI.ts create mode 100644 src/app/collective-rewards/metrics/components/ABIMetrics/index.ts create mode 100644 src/app/collective-rewards/metrics/hooks/useGetTotalAllocation.ts diff --git a/src/app/collective-rewards/metrics/Metrics.tsx b/src/app/collective-rewards/metrics/Metrics.tsx index 6e5e7505..b00ebcd0 100644 --- a/src/app/collective-rewards/metrics/Metrics.tsx +++ b/src/app/collective-rewards/metrics/Metrics.tsx @@ -1,4 +1,4 @@ -import { useGetGaugesArrayByType, withBuilderButton } from '@/app/collective-rewards/user' +import { useGetGaugesArray, withBuilderButton } from '@/app/collective-rewards/user' import { HeaderTitle } from '@/components/Typography' import { TotalAllocationsMetrics, @@ -11,12 +11,13 @@ import { getAddress } from 'viem' import { tokenContracts } from '@/lib/contracts' import { getCoinbaseAddress } from '@/app/collective-rewards/utils' import { PricesContextProvider } from '@/shared/context/PricesContext' +import { ABIMetrics } from './components/ABIMetrics' const HeaderWithBuilderButton = withBuilderButton(HeaderTitle) export const Metrics = () => { - const { data: activeGauges } = useGetGaugesArrayByType('active') - const gauges = activeGauges ?? [] + const { data: allGauges } = useGetGaugesArray() + const gauges = allGauges ?? [] const tokens = { rif: { @@ -35,14 +36,17 @@ export const Metrics = () => {
-
+
-
+
+
+ +
diff --git a/src/app/collective-rewards/metrics/TotalActiveBuildersMetrics.tsx b/src/app/collective-rewards/metrics/TotalActiveBuildersMetrics.tsx index 893603ef..db168f43 100644 --- a/src/app/collective-rewards/metrics/TotalActiveBuildersMetrics.tsx +++ b/src/app/collective-rewards/metrics/TotalActiveBuildersMetrics.tsx @@ -2,14 +2,13 @@ import { useGetBuildersByState } from '@/app/collective-rewards/user' import { MetricsCard, MetricsCardTitle, TokenMetricsCardRow } from '@/app/collective-rewards/rewards' import { withSpinner } from '@/components/LoadingSpinner/withLoadingSpinner' import { useHandleErrors } from '@/app/collective-rewards/utils' -import { Builder } from '../types' export const TotalActiveBuildersMetrics = () => { const { data: activatedBuilders, isLoading, error, - } = useGetBuildersByState>({ + } = useGetBuildersByState({ activated: true, communityApproved: true, kycApproved: true, diff --git a/src/app/collective-rewards/metrics/TotalAllocationsMetrics.tsx b/src/app/collective-rewards/metrics/TotalAllocationsMetrics.tsx index 51f7c447..a9bdc0ae 100644 --- a/src/app/collective-rewards/metrics/TotalAllocationsMetrics.tsx +++ b/src/app/collective-rewards/metrics/TotalAllocationsMetrics.tsx @@ -11,6 +11,7 @@ import { } from '@/app/collective-rewards/rewards' import { withSpinner } from '@/components/LoadingSpinner/withLoadingSpinner' import { useHandleErrors } from '@/app/collective-rewards/utils' +import { useGetTotalAllocation } from './hooks/useGetTotalAllocation' type TotalAllocationsProps = { gauges: Address[] @@ -24,12 +25,10 @@ export const TotalAllocationsMetrics: FC = ({ currency = 'USD', }) => { const { prices } = usePricesContext() - const { data, isLoading, error } = useGaugesGetFunction(gauges, 'totalAllocation') + const { data: totalAllocations, isLoading, error } = useGetTotalAllocation(gauges) useHandleErrors({ error, title: 'Error loading total allocations' }) const price = prices[symbol]?.price ?? 0 - - const totalAllocations = Object.values(data).reduce((acc, allocation) => acc + allocation, 0n) const { amount, fiatAmount } = formatMetrics(totalAllocations, price, symbol, currency) return ( diff --git a/src/app/collective-rewards/metrics/components/ABIMetrics/ABIMetrics.tsx b/src/app/collective-rewards/metrics/components/ABIMetrics/ABIMetrics.tsx new file mode 100644 index 00000000..f2e67ccd --- /dev/null +++ b/src/app/collective-rewards/metrics/components/ABIMetrics/ABIMetrics.tsx @@ -0,0 +1,43 @@ +import { MetricsCard, MetricsCardTitle, TokenMetricsCardRow } from '@/app/collective-rewards/rewards' +import { withSpinner } from '@/components/LoadingSpinner/withLoadingSpinner' +import { useGetABI } from './hooks/useGetABI' + +export const ABIMetrics = () => { + const { data: abiPct, isLoading } = useGetABI() + return ( + <> + + + The Annual Backers Incentives (%) represents an estimate of the annualized percentage of + rewards that backers could receive based on their backing allocations. +
+
+ The calculation follows the formula: (1 + Rewards per stRIF per Cycle / RIF price)^26 - 1. +
+
+ This estimation is dynamic and may vary based on total rewards and user activity. This data is + for informational purposes only.{' '} + + ), + popoverProps: { + size: 'medium', + position: 'left-bottom', + }, + }} + /> + {withSpinner( + TokenMetricsCardRow, + 'min-h-0 grow-0', + )({ + amount: `${abiPct.toFixed(0)}%`, + isLoading, + })} +
+ + ) +} diff --git a/src/app/collective-rewards/metrics/components/ABIMetrics/hooks/useGetABI.ts b/src/app/collective-rewards/metrics/components/ABIMetrics/hooks/useGetABI.ts new file mode 100644 index 00000000..6cbc48db --- /dev/null +++ b/src/app/collective-rewards/metrics/components/ABIMetrics/hooks/useGetABI.ts @@ -0,0 +1,120 @@ +import { + useGetBackersRewardPercentage, + useGetRewardsCoinbase, + useGetRewardsERC20, +} from '@/app/collective-rewards/rewards' +import { usePricesContext } from '@/shared/context/PricesContext' +import { useGetBuildersByState } from '@/app/collective-rewards/user' +import { RequiredBuilder } from '@/app/collective-rewards/types' +import { useGaugesGetFunction } from '@/app/collective-rewards/shared' +import { useCycleContext } from '@/app/collective-rewards/metrics' +import { formatEther, parseUnits } from 'viem' +import { useMemo } from 'react' + +export const useGetABI = () => { + const { + data: builders, + isLoading: buildersLoading, + error: buildersError, + } = useGetBuildersByState({ + activated: true, + communityApproved: true, + kycApproved: true, + revoked: false, + }) + + const { data: rifRewards, isLoading: rifRewardsLoading, error: rifRewardsError } = useGetRewardsERC20() + const { + data: rbtcRewards, + isLoading: rbtcRewardsLoading, + error: rbtcRewardsError, + } = useGetRewardsCoinbase() + + const { + data: { cycleNext }, + isLoading: cycleLoading, + error: cycleError, + } = useCycleContext() + + const gauges = builders.map(({ gauge }) => gauge) + const { + data: totalAllocation, + isLoading: totalAllocationLoading, + error: totalAllocationError, + } = useGaugesGetFunction(gauges, 'totalAllocation') + + const buildersAddress = builders.map(({ address }) => address) + const { + data: backersRewardsPct, + isLoading: backersRewardsPctLoading, + error: backersRewardsPctError, + } = useGetBackersRewardPercentage(buildersAddress, cycleNext.toSeconds()) + + const { prices } = usePricesContext() + + const abi = useMemo(() => { + const sumTotalAllocation = Object.values(totalAllocation).reduce((acc, value) => acc + (value ?? 0n), 0n) + + if (!sumTotalAllocation) { + return 0 + } + + const rifPrice = prices.RIF?.price ?? 0 + const rbtcPrice = prices.RBTC?.price ?? 0 + const rifAmount = Number(formatEther(rifRewards ?? 0n)) + const rbtcAmount = Number(formatEther(rbtcRewards ?? 0n)) + const cyclePayout = rifAmount * rifPrice + rbtcAmount * rbtcPrice + + if (!rifPrice) { + return 0 + } + + const topFiveBuilders = builders + .reduce>((acc, builder) => { + const allocation = totalAllocation[builder.gauge] + const rewardPct = backersRewardsPct[builder.address] + if (allocation && rewardPct) { + acc.push({ allocation, current: rewardPct.current }) + } + return acc + }, []) + .sort((a, b) => (a.allocation > b.allocation ? -1 : 1)) + .slice(0, 5) + + // We use the multiplication with the current backer rewards % to avoid losing precision + // Thats why we don't need to multiply by 100 + const weightedAverageBuilderRewardsPct = topFiveBuilders.reduce( + (acc, { allocation, current }) => acc + (allocation * current) / sumTotalAllocation, + 0n, + ) + const weightedAverageBuilderRewardsPctInEther = Number(formatEther(weightedAverageBuilderRewardsPct)) + + const totalAllocationInEther = Number(formatEther(sumTotalAllocation)) + const rewardsPerStRIFPerCycle = + cyclePayout * (weightedAverageBuilderRewardsPctInEther / totalAllocationInEther) + + return (Math.pow(1 + rewardsPerStRIFPerCycle / rifPrice, 26) - 1) * 100 + }, [backersRewardsPct, builders, prices, rbtcRewards, rifRewards, totalAllocation]) + + const isLoading = + buildersLoading || + rifRewardsLoading || + rbtcRewardsLoading || + cycleLoading || + totalAllocationLoading || + backersRewardsPctLoading + + const error = + buildersError ?? + rifRewardsError ?? + rbtcRewardsError ?? + totalAllocationError ?? + cycleError ?? + backersRewardsPctError + + return { + data: abi, + isLoading, + error, + } +} diff --git a/src/app/collective-rewards/metrics/components/ABIMetrics/index.ts b/src/app/collective-rewards/metrics/components/ABIMetrics/index.ts new file mode 100644 index 00000000..17d2c5bf --- /dev/null +++ b/src/app/collective-rewards/metrics/components/ABIMetrics/index.ts @@ -0,0 +1 @@ +export * from './ABIMetrics' diff --git a/src/app/collective-rewards/metrics/hooks/useGetTotalAllocation.ts b/src/app/collective-rewards/metrics/hooks/useGetTotalAllocation.ts new file mode 100644 index 00000000..b6dcc9ab --- /dev/null +++ b/src/app/collective-rewards/metrics/hooks/useGetTotalAllocation.ts @@ -0,0 +1,12 @@ +import { useGaugesGetFunction } from '../../shared' + +export const useGetTotalAllocation = (gauges: any[]) => { + const { data, isLoading, error } = useGaugesGetFunction(gauges, 'totalAllocation') + + const totalAllocations = Object.values(data).reduce((acc, allocation) => acc + allocation, 0n) + return { + data: totalAllocations, + isLoading, + error, + } +} diff --git a/src/app/collective-rewards/rewards/MyRewards.tsx b/src/app/collective-rewards/rewards/MyRewards.tsx index 0166f444..d352a82f 100644 --- a/src/app/collective-rewards/rewards/MyRewards.tsx +++ b/src/app/collective-rewards/rewards/MyRewards.tsx @@ -13,9 +13,9 @@ import { tokenContracts } from '@/lib/contracts' import { FC } from 'react' import { Address, getAddress, zeroAddress } from 'viem' import { useRouter } from 'next/navigation' -import { Builder } from '../types' import { useCanManageAllocations } from '@/app/collective-rewards/allocations/hooks' import { CRWhitepaperLink } from '@/app/collective-rewards/shared' +import { RequiredBuilder } from '@/app/collective-rewards/types' const SubText = () => { return ( @@ -28,11 +28,9 @@ const SubText = () => { export const Rewards: FC<{ builder: Address }> = ({ builder }) => { const router = useRouter() - const { data: activatedBuilders, error: activatedBuildersError } = useGetBuildersByState>( - { - activated: true, - }, - ) + const { data: activatedBuilders, error: activatedBuildersError } = useGetBuildersByState({ + activated: true, + }) const activatedGauges = activatedBuilders?.map(({ gauge }) => gauge) ?? [] const { data: gauge, error: gaugeError } = useGetBuilderToGauge(builder) const canManageAllocations = useCanManageAllocations() diff --git a/src/app/collective-rewards/rewards/backers/ClaimableRewards.tsx b/src/app/collective-rewards/rewards/backers/ClaimableRewards.tsx index a7efe14f..3188ea9f 100644 --- a/src/app/collective-rewards/rewards/backers/ClaimableRewards.tsx +++ b/src/app/collective-rewards/rewards/backers/ClaimableRewards.tsx @@ -22,7 +22,6 @@ type TokenRewardsMetricsProps = { } const TokenRewardsMetrics: FC = ({ - gauges, token: { address, symbol }, currency = 'USD', }) => { diff --git a/src/app/collective-rewards/rewards/backers/hooks/useGetBackerRewards.ts b/src/app/collective-rewards/rewards/backers/hooks/useGetBackerRewards.ts index 086a0237..054df723 100644 --- a/src/app/collective-rewards/rewards/backers/hooks/useGetBackerRewards.ts +++ b/src/app/collective-rewards/rewards/backers/hooks/useGetBackerRewards.ts @@ -12,7 +12,7 @@ import { useGaugesGetFunction } from '@/app/collective-rewards/shared' import { Address } from 'viem' import { usePricesContext } from '@/shared/context/PricesContext' import { useGetBuildersByState } from '@/app/collective-rewards//user' -import { Builder, BuilderStateFlags } from '@/app/collective-rewards/types' +import { BuilderStateFlags, RequiredBuilder } from '@/app/collective-rewards/types' import { useMemo } from 'react' export type BackerRewards = { @@ -48,7 +48,7 @@ export const useGetBackerRewards = ( data: builders, isLoading: buildersLoading, error: buildersError, - } = useGetBuildersByState>() + } = useGetBuildersByState() const buildersAddress = builders.map(({ address }) => address) const { data: backersRewardsPct, diff --git a/src/app/collective-rewards/rewards/builders/EstimatedRewards.tsx b/src/app/collective-rewards/rewards/builders/EstimatedRewards.tsx index 0392a49d..8c92e165 100644 --- a/src/app/collective-rewards/rewards/builders/EstimatedRewards.tsx +++ b/src/app/collective-rewards/rewards/builders/EstimatedRewards.tsx @@ -10,12 +10,13 @@ import { BuilderRewardDetails, useGetBackerRewardPercentage, } from '@/app/collective-rewards/rewards' -import { useHandleErrors } from '@/app/collective-rewards/utils' +import { isBuilderRewardable, useHandleErrors } from '@/app/collective-rewards/utils' import { usePricesContext } from '@/shared/context/PricesContext' import { FC, useEffect, useState } from 'react' import { Address, parseUnits } from 'viem' import { withSpinner } from '@/components/LoadingSpinner/withLoadingSpinner' import { useCycleContext } from '@/app/collective-rewards/metrics/context/CycleContext' +import { useBuilderContext } from '@/app/collective-rewards/user' type TokenRewardsProps = { builder: Address @@ -65,6 +66,10 @@ const TokenRewards: FC = ({ builder, gauge, token: { id, symb const rewardPercentageToApply = backerRewardsPct.current + const { getBuilderByAddress } = useBuilderContext() + const claimingBuilder = getBuilderByAddress(builder) + const isRewarded = isBuilderRewardable(claimingBuilder?.stateFlags) + const error = rewardsError ?? totalPotentialRewardsError ?? rewardSharesError ?? backerRewardsPctError ?? cycleError useHandleErrors({ error, title: 'Error loading estimated rewards' }) @@ -72,7 +77,9 @@ const TokenRewards: FC = ({ builder, gauge, token: { id, symb const { prices } = usePricesContext() const rewardsAmount = - rewardShares && totalPotentialRewards ? (rewards * rewardShares) / totalPotentialRewards : 0n + isRewarded && rewardShares && totalPotentialRewards + ? (rewards * rewardShares) / totalPotentialRewards + : 0n // The complement of the reward percentage is applied to the estimated rewards since are from the builder's perspective const weiPerEther = parseUnits('1', 18) const estimatedRewards = (rewardsAmount * (weiPerEther - rewardPercentageToApply)) / weiPerEther diff --git a/src/app/collective-rewards/rewards/builders/hooks/useGetBuildersRewards.ts b/src/app/collective-rewards/rewards/builders/hooks/useGetBuildersRewards.ts index 154cb27b..3a205e87 100644 --- a/src/app/collective-rewards/rewards/builders/hooks/useGetBuildersRewards.ts +++ b/src/app/collective-rewards/rewards/builders/hooks/useGetBuildersRewards.ts @@ -14,13 +14,12 @@ import { } from '@/app/collective-rewards/rewards' import { usePricesContext } from '@/shared/context/PricesContext' import { useGaugesGetFunction } from '@/app/collective-rewards/shared' -import { Builder, BuilderStateFlags } from '@/app/collective-rewards/types' +import { BuilderStateFlags, RequiredBuilder } from '@/app/collective-rewards/types' import { useGetBuildersByState } from '@/app/collective-rewards/user' import { Address, parseUnits } from 'viem' import { Allocations, AllocationsContext } from '@/app/collective-rewards/allocations/context' import { useContext, useMemo } from 'react' - -type RequiredBuilder = Required +import { isBuilderRewardable } from '@/app/collective-rewards//utils' const isBuilderShown = ( { stateFlags: { kycApproved, revoked, communityApproved, paused }, address }: RequiredBuilder, @@ -66,11 +65,6 @@ export const useGetBuildersRewards = ({ rif, rbtc }: { [token: string]: Token }, error: totalAllocationError, } = useGaugesGetFunction(gauges, 'totalAllocation') - const sumTotalAllocation = Object.values(totalAllocation ?? {}).reduce( - (acc, value) => acc + (value ?? 0n), - 0n, - ) - const { data: rewardShares, isLoading: rewardSharesLoading, @@ -155,20 +149,24 @@ export const useGetBuildersRewards = ({ rif, rbtc }: { [token: string]: Token }, const weiPerEther = parseUnits('1', 18) + const isRewarded = isBuilderRewardable(stateFlags) + // calculate rif estimated rewards const rewardRif = rewardsERC20 ?? 0n - const rewardsAmountRif = totalPotentialRewards - ? (rewardRif * builderRewardShares) / totalPotentialRewards - : 0n + const rewardsAmountRif = + isRewarded && totalPotentialRewards ? (rewardRif * builderRewardShares) / totalPotentialRewards : 0n const estimatedRifAmount = (rewardsAmountRif * rewardPercentageToApply) / weiPerEther // calculate rbtc estimated rewards const rewardRbtc = rewardsCoinbase ?? 0n - const rewardsAmountRbtc = totalPotentialRewards - ? (rewardRbtc * builderRewardShares) / totalPotentialRewards - : 0n + const rewardsAmountRbtc = + isRewarded && totalPotentialRewards ? (rewardRbtc * builderRewardShares) / totalPotentialRewards : 0n const estimatedRbtcAmount = (rewardsAmountRbtc * rewardPercentageToApply) / weiPerEther + const sumTotalAllocation = Object.values(totalAllocation).reduce( + (acc, value) => acc + (value ?? 0n), + 0n, + ) const totalAllocationPercentage = sumTotalAllocation ? (totalAllocation[gauge] * 100n) / sumTotalAllocation : 0n @@ -233,7 +231,6 @@ export const useGetBuildersRewards = ({ rif, rbtc }: { [token: string]: Token }, rewardShares, totalPotentialRewards, backersRewardsPct, - sumTotalAllocation, rifBuildersRewardsAmount, rbtcBuildersRewardsAmount, rifPrice, diff --git a/src/app/collective-rewards/types.ts b/src/app/collective-rewards/types.ts index a436433e..4da3290d 100644 --- a/src/app/collective-rewards/types.ts +++ b/src/app/collective-rewards/types.ts @@ -42,3 +42,5 @@ export type ProposalByBuilder = Record export type ProposalsToState = Record export type BuilderState = 'active' | 'inProgress' + +export type RequiredBuilder = Required diff --git a/src/app/collective-rewards/utils/isBuilderOperational.ts b/src/app/collective-rewards/utils/isBuilderOperational.ts index cce14d60..f62f05df 100644 --- a/src/app/collective-rewards/utils/isBuilderOperational.ts +++ b/src/app/collective-rewards/utils/isBuilderOperational.ts @@ -28,6 +28,16 @@ export const isBuilderActive = (stateFlags?: BuilderStateFlags) => { ) } +export const isBuilderRewardable = (stateFlags?: BuilderStateFlags) => { + return !!( + stateFlags && + stateFlags.activated && + stateFlags.communityApproved && + stateFlags.kycApproved && + !stateFlags.revoked + ) +} + const inactiveStates = ['Deactivated', 'Paused', 'Revoked'] as const export type InactiveState = (typeof inactiveStates)[number] export const getBuilderInactiveState = (state: BuilderStateFlags): InactiveState => { diff --git a/src/components/Popover/Popover.tsx b/src/components/Popover/Popover.tsx index c866755f..3546e6b4 100644 --- a/src/components/Popover/Popover.tsx +++ b/src/components/Popover/Popover.tsx @@ -8,7 +8,7 @@ export interface PopoverProps extends Omit, 'chil disabled?: boolean trigger?: 'click' | 'hover' background?: 'dark' | 'light' - position?: 'top' | 'bottom' | 'right' | 'left' + position?: 'top' | 'bottom' | 'right' | 'left' | 'left-bottom' size?: 'small' | 'medium' hasCaret?: boolean } @@ -73,6 +73,7 @@ export const Popover = ({ position === 'bottom' && 'top-full', position === 'right' && 'left-full bottom-full', position === 'left' && 'right-full bottom-full', + position === 'left-bottom' && 'right-full top-full', size === 'small' && 'w-36', size === 'medium' && 'w-96', )} @@ -92,7 +93,7 @@ export const Popover = ({ ) } -const PopoverCaret = ({ position }: { position: 'top' | 'bottom' | 'right' | 'left' }) => ( +const PopoverCaret = ({ position }: { position: 'top' | 'bottom' | 'right' | 'left' | 'left-bottom' }) => ( <> {position === 'top' && (