2023-10-03 15:41:54 +00:00
package testutil
import (
"fmt"
"math"
"math/rand"
"time"
2024-02-06 22:54:10 +00:00
tmproto "github.com/cometbft/cometbft/proto/tendermint/types"
2023-10-03 15:41:54 +00:00
"github.com/stretchr/testify/suite"
sdkmath "cosmossdk.io/math"
2024-05-01 03:17:24 +00:00
"github.com/0glabs/0g-chain/app"
"github.com/0glabs/0g-chain/x/community"
"github.com/0glabs/0g-chain/x/community/keeper"
"github.com/0glabs/0g-chain/x/community/types"
2023-10-03 15:41:54 +00:00
sdk "github.com/cosmos/cosmos-sdk/types"
authtypes "github.com/cosmos/cosmos-sdk/x/auth/types"
)
// StakingRewardsTestSuite tests staking rewards per second logic
type stakingRewardsTestSuite struct {
suite . Suite
App app . TestApp
Keeper keeper . Keeper
testFunc testFunc
}
func NewStakingRewardsTestSuite ( tf testFunc ) * stakingRewardsTestSuite {
suite := & stakingRewardsTestSuite { }
suite . testFunc = tf
return suite
}
// The default state used by each test
func ( suite * stakingRewardsTestSuite ) SetupTest ( ) {
app . SetSDKConfig ( )
tApp := app . NewTestApp ( )
tApp . InitializeFromGenesisStates ( )
suite . App = tApp
suite . Keeper = suite . App . GetCommunityKeeper ( )
}
func ( suite * stakingRewardsTestSuite ) TestStakingRewards ( ) {
testCases := [ ] struct {
// name of subtest
name string
// block time of first block
periodStart time . Time
// block time of last block
periodEnd time . Time
// block time n will be periodStart + rand(range_min...range_max)*(n-1) up to periodEnd
blockTimeRangeMin float64
blockTimeRangeMax float64
// rewards per second to set in state
rewardsPerSecond sdkmath . LegacyDec
// the amount of ukava to mint and transfer to the community pool
// to use to pay for rewards
communityPoolFunds sdkmath . Int
// how many total rewards are expected to be accumulated in ukava
expectedRewardsTotal sdkmath . Int
} {
// ** These take a long time to run **
//{
// name: "one year with 0.5 to 1 second block times",
// periodStart: time.Date(2023, 1, 1, 0, 0, 0, 0, time.UTC),
// periodEnd: time.Date(2024, 1, 1, 0, 0, 0, 0, time.UTC),
// blockTimeRangeMin: 0.5,
// blockTimeRangeMax: 1,
// rewardsPerSecond: sdkmath.LegacyMustNewDecFromStr("1585489.599188229325215626"),
// expectedRewardsTotal: sdkmath.NewInt(49999999999999), // 50 million KAVA per year
//},
//{
// name: "one year with 5.5 to 6.5 second blocktimes",
// periodStart: time.Date(2023, 1, 1, 0, 0, 0, 0, time.UTC),
// periodEnd: time.Date(2024, 1, 1, 0, 0, 0, 0, time.UTC),
// blockTimeRangeMin: 5.5,
// blockTimeRangeMax: 6.5,
// rewardsPerSecond: sdkmath.LegacyMustNewDecFromStr("1585489.599188229325215626"), // 50 million kava per year
// communityPoolFunds: sdkmath.NewInt(50000000000000),
// expectedRewardsTotal: sdkmath.NewInt(49999999999999), // truncation results in 1 ukava error
//},
//
//
// One Day of blocks with different block time variations
//
//
{
name : "one day with sub-second block times and 50 million KAVA per year" ,
periodStart : time . Date ( 2023 , 1 , 1 , 0 , 0 , 0 , 0 , time . UTC ) ,
periodEnd : time . Date ( 2023 , 1 , 2 , 0 , 0 , 0 , 0 , time . UTC ) ,
blockTimeRangeMin : 0.1 ,
blockTimeRangeMax : 1 ,
rewardsPerSecond : sdkmath . LegacyMustNewDecFromStr ( "1585489.599188229325215626" ) , // 50 million kava per year
communityPoolFunds : sdkmath . NewInt ( 200000000000 ) ,
expectedRewardsTotal : sdkmath . NewInt ( 136986301369 ) , // 50 million / 365 days - 1 ukava
} ,
{
name : "one day with 5.5 to 6.5 second block times and 50 million KAVA per year" ,
periodStart : time . Date ( 2023 , 1 , 1 , 0 , 0 , 0 , 0 , time . UTC ) ,
periodEnd : time . Date ( 2023 , 1 , 2 , 0 , 0 , 0 , 0 , time . UTC ) ,
blockTimeRangeMin : 5.5 ,
blockTimeRangeMax : 6.5 ,
rewardsPerSecond : sdkmath . LegacyMustNewDecFromStr ( "1585489.599188229325215626" ) , // 50 million kava per year
communityPoolFunds : sdkmath . NewInt ( 200000000000 ) ,
expectedRewardsTotal : sdkmath . NewInt ( 136986301369 ) , // 50 million / 365 days - 1 ukava
} ,
//
//
// Total time span under 1 second
//
//
{
name : "single 6.9 second time span and 25 million KAVA per year" ,
periodStart : time . Date ( 2023 , 1 , 1 , 0 , 0 , 0 , 0 , time . UTC ) ,
periodEnd : time . Date ( 2023 , 1 , 1 , 0 , 0 , 6 , 900000000 , time . UTC ) ,
blockTimeRangeMin : 10 , // forces only two blocks -- one time span
blockTimeRangeMax : 10 ,
rewardsPerSecond : sdkmath . LegacyMustNewDecFromStr ( "792744.799594114662607813" ) , // 25 million kava per year
communityPoolFunds : sdkmath . NewInt ( 10000000 ) ,
expectedRewardsTotal : sdkmath . NewInt ( 5469939 ) , // per second rate * 6.9
} ,
{
name : "multiple blocks across sub-second time span nd 10 million KAVA per year" ,
periodStart : time . Date ( 2023 , 1 , 1 , 0 , 0 , 0 , 0 , time . UTC ) ,
periodEnd : time . Date ( 2023 , 1 , 1 , 0 , 0 , 0 , 800000000 , time . UTC ) ,
blockTimeRangeMin : 0.1 , // multiple blocks in a sub-second time span
blockTimeRangeMax : 0.2 ,
rewardsPerSecond : sdkmath . LegacyMustNewDecFromStr ( "317097.919837645865043125" ) , // 10 million kava per year
communityPoolFunds : sdkmath . NewInt ( 300000 ) ,
expectedRewardsTotal : sdkmath . NewInt ( 253678 ) , // per second rate * 0.8
} ,
//
//
// Variations of community pool balance
//
//
{
name : "community pool exact funds -- should spend community to zero and not panic" ,
periodStart : time . Date ( 2023 , 1 , 1 , 0 , 0 , 0 , 0 , time . UTC ) ,
periodEnd : time . Date ( 2023 , 1 , 2 , 0 , 0 , 0 , 0 , time . UTC ) ,
blockTimeRangeMin : 5.5 ,
blockTimeRangeMax : 6.2 ,
rewardsPerSecond : sdkmath . LegacyMustNewDecFromStr ( "317097.919837645865043125" ) , // 10 million kava per year
communityPoolFunds : sdkmath . NewInt ( 27397260273 ) ,
expectedRewardsTotal : sdkmath . NewInt ( 27397260273 ) ,
} ,
{
name : "community pool under funded -- should spend community pool to down to zero and not panic" ,
periodStart : time . Date ( 2023 , 1 , 1 , 0 , 0 , 0 , 0 , time . UTC ) ,
periodEnd : time . Date ( 2023 , 1 , 2 , 0 , 0 , 0 , 0 , time . UTC ) ,
blockTimeRangeMin : 5.5 ,
blockTimeRangeMax : 6.5 ,
rewardsPerSecond : sdkmath . LegacyMustNewDecFromStr ( "1585489.599188229325215626" ) , // 25 million kava per year
communityPoolFunds : sdkmath . NewInt ( 100000000000 ) , // under funded
expectedRewardsTotal : sdkmath . NewInt ( 100000000000 ) , // rewards max is the community pool balance
} ,
{
name : "community pool no funds -- should pay zero rewards and not panic" ,
periodStart : time . Date ( 2023 , 1 , 1 , 0 , 0 , 0 , 0 , time . UTC ) ,
periodEnd : time . Date ( 2023 , 1 , 2 , 0 , 0 , 0 , 0 , time . UTC ) ,
blockTimeRangeMin : 5.5 ,
blockTimeRangeMax : 6.5 ,
rewardsPerSecond : sdkmath . LegacyMustNewDecFromStr ( "792744.799594114662607813" ) , // 25 million kava per year
communityPoolFunds : sdkmath . NewInt ( 0 ) ,
expectedRewardsTotal : sdkmath . NewInt ( 0 ) ,
} ,
//
//
// Disabled
//
//
{
name : "zero rewards per second results in zero rewards paid" ,
periodStart : time . Date ( 2023 , 1 , 1 , 0 , 0 , 0 , 0 , time . UTC ) ,
periodEnd : time . Date ( 2023 , 1 , 2 , 0 , 0 , 0 , 0 , time . UTC ) ,
blockTimeRangeMin : 5.5 ,
blockTimeRangeMax : 6.5 ,
rewardsPerSecond : sdkmath . LegacyMustNewDecFromStr ( "0.000000000000000000" ) , // 25 million kava per year
communityPoolFunds : sdkmath . NewInt ( 100000000000000 ) ,
expectedRewardsTotal : sdkmath . NewInt ( 0 ) ,
} ,
//
//
// Test underlying calculations are safe and overflow/underflow bounds are reasonable
//
//
{
name : "does not overflow with extremely large per second value and extremely large single block durations" ,
periodStart : time . Date ( 2023 , 1 , 1 , 0 , 0 , 0 , 0 , time . UTC ) ,
periodEnd : time . Date ( 2033 , 1 , 1 , 0 , 0 , 0 , 0 , time . UTC ) ,
blockTimeRangeMin : 315619200 , // a single 10 year long block in seconds (w/ 3 leap years)
blockTimeRangeMax : 315619200 , // a single 10 year long block in seconds (w/ 3 leap years)
rewardsPerSecond : sdkmath . LegacyMustNewDecFromStr ( "100000000000000000000000000.000000000000000000" ) , // 100 million kava per second in 18 decimal form
communityPoolFunds : newIntFromString ( "40000000000000000000000000000000000" ) ,
expectedRewardsTotal : newIntFromString ( "31561920000000000000000000000000000" ) , // 10 years worth of rewards (with three leap years)
} ,
{
name : "able to accumulate decimal ukava units across blocks" ,
periodStart : time . Date ( 2023 , 1 , 1 , 0 , 0 , 0 , 0 , time . UTC ) ,
periodEnd : time . Date ( 2023 , 1 , 2 , 0 , 0 , 0 , 0 , time . UTC ) ,
blockTimeRangeMin : 5.5 ,
blockTimeRangeMax : 6.5 ,
rewardsPerSecond : sdkmath . LegacyMustNewDecFromStr ( "0.100000000000000000" ) , // blocks are not long enough to accumulate a single ukava with this rate
communityPoolFunds : sdkmath . NewInt ( 10000 ) ,
expectedRewardsTotal : sdkmath . NewInt ( 8640 ) ,
} ,
{
name : "down to 1 ukava per year can be accumulated -- we are safe from underflow at reasonably small values" ,
periodStart : time . Date ( 2023 , 1 , 1 , 0 , 0 , 0 , 0 , time . UTC ) ,
periodEnd : time . Date ( 2024 , 1 , 1 , 0 , 0 , 0 , 0 , time . UTC ) ,
blockTimeRangeMin : 60 , // large block times speed up this test case
blockTimeRangeMax : 120 ,
rewardsPerSecond : sdkmath . LegacyMustNewDecFromStr ( "0.000000031709791984" ) ,
communityPoolFunds : sdkmath . NewInt ( 1 ) ,
expectedRewardsTotal : sdkmath . NewInt ( 1 ) ,
} ,
}
for _ , tc := range testCases {
suite . Run ( tc . name , func ( ) {
suite . SetupTest ( )
// keepers
keeper := suite . Keeper
accountKeeper := suite . App . GetAccountKeeper ( )
bankKeeper := suite . App . GetBankKeeper ( )
// initial context at height 1
height := int64 ( 1 )
blockTime := tc . periodStart
ctx := suite . App . NewContext ( true , tmproto . Header { Height : height , Time : blockTime } )
// ensure community pool balance matches the test expectations
poolAcc := accountKeeper . GetModuleAccount ( ctx , types . ModuleName )
// community pool balance should start at zero
suite . Require ( ) . True ( bankKeeper . GetBalance ( ctx , poolAcc . GetAddress ( ) , "ukava" ) . Amount . IsZero ( ) , "expected community pool to start with zero coins in test genesis" )
// fund withexact amount from test case
suite . App . FundAccount ( ctx , poolAcc . GetAddress ( ) , sdk . NewCoins ( sdk . NewCoin ( "ukava" , tc . communityPoolFunds ) ) )
// get starting balance of fee collector to substract later in case this is non-zero in genesis
feeCollectorAcc := accountKeeper . GetModuleAccount ( ctx , authtypes . FeeCollectorName )
initialFeeCollectorBalance := bankKeeper . GetBalance ( ctx , feeCollectorAcc . GetAddress ( ) , "ukava" ) . Amount
// set rewards per second in state
params , found := keeper . GetParams ( ctx )
suite . Require ( ) . True ( found )
params . StakingRewardsPerSecond = tc . rewardsPerSecond
keeper . SetParams ( ctx , params )
2023-10-03 22:10:22 +00:00
stakingRewardEvents := sdk . Events { }
2023-10-03 15:41:54 +00:00
for {
// run community begin blocker logic
suite . testFunc ( ctx , keeper )
2023-10-03 22:10:22 +00:00
// accumulate event rewards from events
stakingRewardEvents = append ( stakingRewardEvents , filterStakingRewardEvents ( ctx . EventManager ( ) . Events ( ) ) ... )
2023-10-03 15:41:54 +00:00
// exit loop if we are at last block
if blockTime . Equal ( tc . periodEnd ) {
break
}
// create random block duration in nanoseconds
randomBlockDurationInSeconds := tc . blockTimeRangeMin + rand . Float64 ( ) * ( tc . blockTimeRangeMax - tc . blockTimeRangeMin )
nextBlockDuration := time . Duration ( randomBlockDurationInSeconds * math . Pow10 ( 9 ) )
// move to next block by incrementing height, adding random duration, and settings new context
height ++
blockTime = blockTime . Add ( nextBlockDuration )
// set last block to exact end of period if we go past
if blockTime . After ( tc . periodEnd ) {
blockTime = tc . periodEnd
}
ctx = suite . App . NewContext ( true , tmproto . Header { Height : height , Time : blockTime } )
}
endingFeeCollectorBalance := bankKeeper . GetBalance ( ctx , feeCollectorAcc . GetAddress ( ) , "ukava" ) . Amount
feeCollectorBalanceAdded := endingFeeCollectorBalance . Sub ( initialFeeCollectorBalance )
// assert fee pool was payed the correct rewards
suite . Equal ( tc . expectedRewardsTotal . String ( ) , feeCollectorBalanceAdded . String ( ) , "expected fee collector balance to match" )
2023-10-03 22:10:22 +00:00
if tc . expectedRewardsTotal . IsZero ( ) {
suite . Equal ( 0 , len ( stakingRewardEvents ) , "expected no events to be emitted" )
} else {
// we add up all reward coin events
eventCoins := getRewardCoinsFromEvents ( stakingRewardEvents )
// assert events emitted match expected rewards
suite . Equal (
tc . expectedRewardsTotal . String ( ) ,
eventCoins . AmountOf ( "ukava" ) . String ( ) ,
"expected event coins to match" ,
)
}
2023-10-03 15:41:54 +00:00
// assert the community pool deducted the same amount
expectedCommunityPoolBalance := tc . communityPoolFunds . Sub ( tc . expectedRewardsTotal )
actualCommunityPoolBalance := bankKeeper . GetBalance ( ctx , poolAcc . GetAddress ( ) , "ukava" ) . Amount
suite . Equal ( expectedCommunityPoolBalance . String ( ) , actualCommunityPoolBalance . String ( ) , "expected community pool balance to match" )
} )
}
}
func ( suite * stakingRewardsTestSuite ) TestStakingRewardsDoNotAccumulateWhenPoolIsDrained ( ) {
app := suite . App
keeper := suite . Keeper
accountKeeper := suite . App . GetAccountKeeper ( )
bankKeeper := suite . App . GetBankKeeper ( )
// first block
blockTime := time . Now ( )
ctx := app . NewContext ( true , tmproto . Header { Height : 1 , Time : blockTime } )
poolAcc := accountKeeper . GetModuleAccount ( ctx , types . ModuleName )
feeCollectorAcc := accountKeeper . GetModuleAccount ( ctx , authtypes . FeeCollectorName )
// set state to pay staking rewards
params , _ := keeper . GetParams ( ctx )
// we set a decimal amount that ensures after 10 seconds we overspend the community pool
// with enough truncation error that we would have an ending balance of 20.000001 if it was
// carried over after the pool run out of funds
params . StakingRewardsPerSecond = sdkmath . LegacyMustNewDecFromStr ( "1000000.099999999999999999" ) // > 1 KAVA per second
keeper . SetParams ( ctx , params )
// fund community pool account
app . FundAccount ( ctx , poolAcc . GetAddress ( ) , sdk . NewCoins ( sdk . NewCoin ( "ukava" , sdkmath . NewInt ( 10000000 ) ) ) ) // 10 KAVA
initialFeeCollectorBalance := bankKeeper . GetBalance ( ctx , feeCollectorAcc . GetAddress ( ) , "ukava" ) . Amount
// run first block (no rewards hapeen on first block)
community . BeginBlocker ( ctx , keeper )
// run second block 10 seconds in future and spend all community pool rewards
blockTime = blockTime . Add ( 10 * time . Second )
ctx = app . NewContext ( true , tmproto . Header { Height : 2 , Time : blockTime } )
community . BeginBlocker ( ctx , keeper )
// run third block 10 seconds in future which no rewards will be paid
blockTime = blockTime . Add ( 10 * time . Second )
ctx = app . NewContext ( true , tmproto . Header { Height : 3 , Time : blockTime } )
community . BeginBlocker ( ctx , keeper )
// run fourth block 10 seconds in future which no rewards will be paid
blockTime = blockTime . Add ( 10 * time . Second )
ctx = app . NewContext ( true , tmproto . Header { Height : 4 , Time : blockTime } )
community . BeginBlocker ( ctx , keeper )
// refund the community pool with 100 KAVA -- plenty of funds
app . FundAccount ( ctx , poolAcc . GetAddress ( ) , sdk . NewCoins ( sdk . NewCoin ( "ukava" , sdkmath . NewInt ( 100000000 ) ) ) ) // 100 KAVA
// run fifth block 10 seconds in future which no rewards will be paid
blockTime = blockTime . Add ( 10 * time . Second )
ctx = app . NewContext ( true , tmproto . Header { Height : 5 , Time : blockTime } )
community . BeginBlocker ( ctx , keeper )
// assert that only 20 total KAVA has been distributed in rewards
// and blocks where community pool had d
rewards := bankKeeper . GetBalance ( ctx , feeCollectorAcc . GetAddress ( ) , "ukava" ) . Amount . Sub ( initialFeeCollectorBalance )
suite . Require ( ) . Equal ( sdkmath . NewInt ( 20000000 ) . String ( ) , rewards . String ( ) )
}
func ( suite * stakingRewardsTestSuite ) TestPanicsOnMissingParameters ( ) {
suite . SetupTest ( )
ctx := suite . App . NewContext ( true , tmproto . Header { Height : 1 , Time : time . Now ( ) } )
store := ctx . KVStore ( suite . App . GetKVStoreKey ( types . StoreKey ) )
store . Delete ( types . ParamsKey )
suite . PanicsWithValue ( "invalid state: module parameters not found" , func ( ) {
suite . testFunc ( ctx , suite . Keeper )
} )
}
// newIntFromString returns a new sdkmath.Int from a string
func newIntFromString ( str string ) sdkmath . Int {
num , ok := sdkmath . NewIntFromString ( str )
if ! ok {
panic ( fmt . Sprintf ( "overflow creating Int from %s" , str ) )
}
return num
}
2023-10-03 22:10:22 +00:00
func filterStakingRewardEvents ( events sdk . Events ) ( rewardEvents sdk . Events ) {
for _ , event := range events {
if event . Type == types . EventTypeStakingRewardsPaid {
rewardEvents = append ( rewardEvents , event )
}
}
return
}
func getRewardCoinsFromEvents ( events sdk . Events ) sdk . Coins {
coins := sdk . NewCoins ( )
for _ , event := range events {
if event . Type == types . EventTypeStakingRewardsPaid {
rewards , err := sdk . ParseCoinNormalized ( string ( event . Attributes [ 0 ] . Value ) )
if err != nil {
panic ( err )
}
coins = coins . Add ( rewards )
}
}
return coins
}