mirror of
https://github.com/0glabs/0g-chain.git
synced 2024-12-26 00:05:18 +00:00
Incentives tests (#429)
* USDX Incentives tests (#429) Co-authored-by: Denali Marsh <denali@kava.io> Co-authored-by: John Maheswaran <jmaheswaran@users.noreply.github.com> Co-authored-by: John Maheswaran <john@kava.io>
This commit is contained in:
parent
1ef9bd331b
commit
e9a73b80ce
5
run_a_bunch_of_sims.sh
Executable file
5
run_a_bunch_of_sims.sh
Executable file
@ -0,0 +1,5 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
for i in {1..10}; do
|
||||||
|
go test ./app -run TestAppStateDeterminism -Enabled -Commit -NumBlocks=100 -BlockSize=200 -Seed ${i} -v -timeout 24h
|
||||||
|
done
|
74
x/incentive/handler_test.go
Normal file
74
x/incentive/handler_test.go
Normal file
@ -0,0 +1,74 @@
|
|||||||
|
package incentive_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||||
|
"github.com/kava-labs/kava/app"
|
||||||
|
"github.com/kava-labs/kava/x/incentive"
|
||||||
|
"github.com/kava-labs/kava/x/kavadist"
|
||||||
|
"github.com/stretchr/testify/suite"
|
||||||
|
abci "github.com/tendermint/tendermint/abci/types"
|
||||||
|
tmtime "github.com/tendermint/tendermint/types/time"
|
||||||
|
)
|
||||||
|
|
||||||
|
func cs(coins ...sdk.Coin) sdk.Coins { return sdk.NewCoins(coins...) }
|
||||||
|
func c(denom string, amount int64) sdk.Coin { return sdk.NewInt64Coin(denom, amount) }
|
||||||
|
|
||||||
|
type HandlerTestSuite struct {
|
||||||
|
suite.Suite
|
||||||
|
|
||||||
|
ctx sdk.Context
|
||||||
|
app app.TestApp
|
||||||
|
handler sdk.Handler
|
||||||
|
keeper incentive.Keeper
|
||||||
|
addrs []sdk.AccAddress
|
||||||
|
}
|
||||||
|
|
||||||
|
func (suite *HandlerTestSuite) SetupTest() {
|
||||||
|
tApp := app.NewTestApp()
|
||||||
|
ctx := tApp.NewContext(true, abci.Header{Height: 1, Time: tmtime.Now()})
|
||||||
|
keeper := tApp.GetIncentiveKeeper()
|
||||||
|
|
||||||
|
// Set up genesis state and initialize
|
||||||
|
_, addrs := app.GeneratePrivKeyAddressPairs(3)
|
||||||
|
coins := []sdk.Coins{}
|
||||||
|
for j := 0; j < 3; j++ {
|
||||||
|
coins = append(coins, cs(c("bnb", 10000000000), c("ukava", 10000000000)))
|
||||||
|
}
|
||||||
|
authGS := app.NewAuthGenState(addrs, coins)
|
||||||
|
tApp.InitializeFromGenesisStates(authGS)
|
||||||
|
|
||||||
|
suite.addrs = addrs
|
||||||
|
suite.handler = incentive.NewHandler(keeper)
|
||||||
|
suite.keeper = keeper
|
||||||
|
suite.app = tApp
|
||||||
|
suite.ctx = ctx
|
||||||
|
}
|
||||||
|
|
||||||
|
func (suite *HandlerTestSuite) addClaim() {
|
||||||
|
supplyKeeper := suite.app.GetSupplyKeeper()
|
||||||
|
macc := supplyKeeper.GetModuleAccount(suite.ctx, kavadist.ModuleName)
|
||||||
|
err := supplyKeeper.MintCoins(suite.ctx, macc.GetName(), cs(c("ukava", 1000000)))
|
||||||
|
suite.Require().NoError(err)
|
||||||
|
cp := incentive.NewClaimPeriod("bnb", 1, suite.ctx.BlockTime().Add(time.Hour*168), time.Hour*8766)
|
||||||
|
suite.NotPanics(func() {
|
||||||
|
suite.keeper.SetClaimPeriod(suite.ctx, cp)
|
||||||
|
})
|
||||||
|
c1 := incentive.NewClaim(suite.addrs[0], c("ukava", 1000000), "bnb", 1)
|
||||||
|
suite.NotPanics(func() {
|
||||||
|
suite.keeper.SetClaim(suite.ctx, c1)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (suite *HandlerTestSuite) TestMsgClaimReward() {
|
||||||
|
suite.addClaim()
|
||||||
|
msg := incentive.NewMsgClaimReward(suite.addrs[0], "bnb")
|
||||||
|
res, err := suite.handler(suite.ctx, msg)
|
||||||
|
suite.NoError(err)
|
||||||
|
suite.Require().NotNil(res)
|
||||||
|
}
|
||||||
|
func TestHandlerTestSuite(t *testing.T) {
|
||||||
|
suite.Run(t, new(HandlerTestSuite))
|
||||||
|
}
|
174
x/incentive/keeper/keeper_test.go
Normal file
174
x/incentive/keeper/keeper_test.go
Normal file
@ -0,0 +1,174 @@
|
|||||||
|
package keeper_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||||
|
authexported "github.com/cosmos/cosmos-sdk/x/auth/exported"
|
||||||
|
supplyexported "github.com/cosmos/cosmos-sdk/x/supply/exported"
|
||||||
|
"github.com/kava-labs/kava/app"
|
||||||
|
"github.com/kava-labs/kava/x/incentive/keeper"
|
||||||
|
"github.com/kava-labs/kava/x/incentive/types"
|
||||||
|
"github.com/stretchr/testify/suite"
|
||||||
|
abci "github.com/tendermint/tendermint/abci/types"
|
||||||
|
tmtime "github.com/tendermint/tendermint/types/time"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Test suite used for all keeper tests
|
||||||
|
type KeeperTestSuite struct {
|
||||||
|
suite.Suite
|
||||||
|
|
||||||
|
keeper keeper.Keeper
|
||||||
|
app app.TestApp
|
||||||
|
ctx sdk.Context
|
||||||
|
addrs []sdk.AccAddress
|
||||||
|
}
|
||||||
|
|
||||||
|
// The default state used by each test
|
||||||
|
func (suite *KeeperTestSuite) SetupTest() {
|
||||||
|
tApp := app.NewTestApp()
|
||||||
|
ctx := tApp.NewContext(true, abci.Header{Height: 1, Time: tmtime.Now()})
|
||||||
|
tApp.InitializeFromGenesisStates()
|
||||||
|
_, addrs := app.GeneratePrivKeyAddressPairs(1)
|
||||||
|
keeper := tApp.GetIncentiveKeeper()
|
||||||
|
suite.app = tApp
|
||||||
|
suite.ctx = ctx
|
||||||
|
suite.keeper = keeper
|
||||||
|
suite.addrs = addrs
|
||||||
|
}
|
||||||
|
|
||||||
|
func (suite *KeeperTestSuite) getAccount(addr sdk.AccAddress) authexported.Account {
|
||||||
|
ak := suite.app.GetAccountKeeper()
|
||||||
|
return ak.GetAccount(suite.ctx, addr)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (suite *KeeperTestSuite) getModuleAccount(name string) supplyexported.ModuleAccountI {
|
||||||
|
sk := suite.app.GetSupplyKeeper()
|
||||||
|
return sk.GetModuleAccount(suite.ctx, name)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (suite *KeeperTestSuite) TestGetSetDeleteRewardPeriod() {
|
||||||
|
rp := types.NewRewardPeriod("bnb", suite.ctx.BlockTime(), suite.ctx.BlockTime().Add(time.Hour*168), c("ukava", 100000000), suite.ctx.BlockTime().Add(time.Hour*168*2), time.Hour*8766)
|
||||||
|
_, found := suite.keeper.GetRewardPeriod(suite.ctx, "bnb")
|
||||||
|
suite.False(found)
|
||||||
|
suite.NotPanics(func() {
|
||||||
|
suite.keeper.SetRewardPeriod(suite.ctx, rp)
|
||||||
|
})
|
||||||
|
testRP, found := suite.keeper.GetRewardPeriod(suite.ctx, "bnb")
|
||||||
|
suite.True(found)
|
||||||
|
suite.Equal(rp, testRP)
|
||||||
|
suite.NotPanics(func() {
|
||||||
|
suite.keeper.DeleteRewardPeriod(suite.ctx, "bnb")
|
||||||
|
})
|
||||||
|
_, found = suite.keeper.GetRewardPeriod(suite.ctx, "bnb")
|
||||||
|
suite.False(found)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (suite *KeeperTestSuite) TestGetSetDeleteClaimPeriod() {
|
||||||
|
cp := types.NewClaimPeriod("bnb", 1, suite.ctx.BlockTime().Add(time.Hour*168), time.Hour*8766)
|
||||||
|
_, found := suite.keeper.GetClaimPeriod(suite.ctx, 1, "bnb")
|
||||||
|
suite.False(found)
|
||||||
|
suite.NotPanics(func() {
|
||||||
|
suite.keeper.SetClaimPeriod(suite.ctx, cp)
|
||||||
|
})
|
||||||
|
testCP, found := suite.keeper.GetClaimPeriod(suite.ctx, 1, "bnb")
|
||||||
|
suite.True(found)
|
||||||
|
suite.Equal(cp, testCP)
|
||||||
|
suite.NotPanics(func() {
|
||||||
|
suite.keeper.DeleteClaimPeriod(suite.ctx, 1, "bnb")
|
||||||
|
})
|
||||||
|
_, found = suite.keeper.GetClaimPeriod(suite.ctx, 1, "bnb")
|
||||||
|
suite.False(found)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (suite *KeeperTestSuite) TestGetSetClaimPeriodID() {
|
||||||
|
suite.Panics(func() {
|
||||||
|
suite.keeper.GetNextClaimPeriodID(suite.ctx, "bnb")
|
||||||
|
})
|
||||||
|
suite.NotPanics(func() {
|
||||||
|
suite.keeper.SetNextClaimPeriodID(suite.ctx, "bnb", 1)
|
||||||
|
})
|
||||||
|
testID := suite.keeper.GetNextClaimPeriodID(suite.ctx, "bnb")
|
||||||
|
suite.Equal(uint64(1), testID)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (suite *KeeperTestSuite) TestGetSetDeleteClaim() {
|
||||||
|
c := types.NewClaim(suite.addrs[0], c("ukava", 1000000), "bnb", 1)
|
||||||
|
_, found := suite.keeper.GetClaim(suite.ctx, suite.addrs[0], "bnb", 1)
|
||||||
|
suite.False(found)
|
||||||
|
suite.NotPanics(func() {
|
||||||
|
suite.keeper.SetClaim(suite.ctx, c)
|
||||||
|
})
|
||||||
|
testC, found := suite.keeper.GetClaim(suite.ctx, suite.addrs[0], "bnb", 1)
|
||||||
|
suite.True(found)
|
||||||
|
suite.Equal(c, testC)
|
||||||
|
suite.NotPanics(func() {
|
||||||
|
suite.keeper.DeleteClaim(suite.ctx, suite.addrs[0], "bnb", 1)
|
||||||
|
})
|
||||||
|
_, found = suite.keeper.GetClaim(suite.ctx, suite.addrs[0], "bnb", 1)
|
||||||
|
suite.False(found)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (suite *KeeperTestSuite) TestIterateMethods() {
|
||||||
|
suite.addObjectsToStore() // adds 2 objects of each type to the store
|
||||||
|
|
||||||
|
var rewardPeriods types.RewardPeriods
|
||||||
|
suite.keeper.IterateRewardPeriods(suite.ctx, func(rp types.RewardPeriod) (stop bool) {
|
||||||
|
rewardPeriods = append(rewardPeriods, rp)
|
||||||
|
return false
|
||||||
|
})
|
||||||
|
suite.Equal(2, len(rewardPeriods))
|
||||||
|
|
||||||
|
var claimPeriods types.ClaimPeriods
|
||||||
|
suite.keeper.IterateClaimPeriods(suite.ctx, func(cp types.ClaimPeriod) (stop bool) {
|
||||||
|
claimPeriods = append(claimPeriods, cp)
|
||||||
|
return false
|
||||||
|
})
|
||||||
|
suite.Equal(2, len(claimPeriods))
|
||||||
|
|
||||||
|
var claims types.Claims
|
||||||
|
suite.keeper.IterateClaims(suite.ctx, func(c types.Claim) (stop bool) {
|
||||||
|
claims = append(claims, c)
|
||||||
|
return false
|
||||||
|
})
|
||||||
|
suite.Equal(2, len(claims))
|
||||||
|
|
||||||
|
var genIDs types.GenesisClaimPeriodIDs
|
||||||
|
suite.keeper.IterateClaimPeriodIDKeysAndValues(suite.ctx, func(denom string, id uint64) (stop bool) {
|
||||||
|
genID := types.GenesisClaimPeriodID{Denom: denom, ID: id}
|
||||||
|
genIDs = append(genIDs, genID)
|
||||||
|
return false
|
||||||
|
})
|
||||||
|
suite.Equal(2, len(genIDs))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (suite *KeeperTestSuite) addObjectsToStore() {
|
||||||
|
rp1 := types.NewRewardPeriod("bnb", suite.ctx.BlockTime(), suite.ctx.BlockTime().Add(time.Hour*168), c("ukava", 100000000), suite.ctx.BlockTime().Add(time.Hour*168*2), time.Hour*8766)
|
||||||
|
rp2 := types.NewRewardPeriod("xrp", suite.ctx.BlockTime(), suite.ctx.BlockTime().Add(time.Hour*168), c("ukava", 100000000), suite.ctx.BlockTime().Add(time.Hour*168*2), time.Hour*8766)
|
||||||
|
suite.keeper.SetRewardPeriod(suite.ctx, rp1)
|
||||||
|
suite.keeper.SetRewardPeriod(suite.ctx, rp2)
|
||||||
|
|
||||||
|
cp1 := types.NewClaimPeriod("bnb", 1, suite.ctx.BlockTime().Add(time.Hour*168), time.Hour*8766)
|
||||||
|
cp2 := types.NewClaimPeriod("xrp", 1, suite.ctx.BlockTime().Add(time.Hour*168), time.Hour*8766)
|
||||||
|
suite.keeper.SetClaimPeriod(suite.ctx, cp1)
|
||||||
|
suite.keeper.SetClaimPeriod(suite.ctx, cp2)
|
||||||
|
|
||||||
|
suite.keeper.SetNextClaimPeriodID(suite.ctx, "bnb", 1)
|
||||||
|
suite.keeper.SetNextClaimPeriodID(suite.ctx, "xrp", 1)
|
||||||
|
|
||||||
|
c1 := types.NewClaim(suite.addrs[0], c("ukava", 1000000), "bnb", 1)
|
||||||
|
c2 := types.NewClaim(suite.addrs[0], c("ukava", 1000000), "xrp", 1)
|
||||||
|
suite.keeper.SetClaim(suite.ctx, c1)
|
||||||
|
suite.keeper.SetClaim(suite.ctx, c2)
|
||||||
|
|
||||||
|
params := types.NewParams(
|
||||||
|
true, types.Rewards{types.NewReward(true, "bnb", c("ukava", 1000000000), time.Hour*7*24, time.Hour*24*365, time.Hour*7*24)},
|
||||||
|
)
|
||||||
|
suite.keeper.SetParams(suite.ctx, params)
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestKeeperTestSuite(t *testing.T) {
|
||||||
|
suite.Run(t, new(KeeperTestSuite))
|
||||||
|
}
|
357
x/incentive/keeper/payout_test.go
Normal file
357
x/incentive/keeper/payout_test.go
Normal file
@ -0,0 +1,357 @@
|
|||||||
|
package keeper_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||||
|
"github.com/cosmos/cosmos-sdk/x/auth"
|
||||||
|
"github.com/cosmos/cosmos-sdk/x/auth/vesting"
|
||||||
|
"github.com/kava-labs/kava/app"
|
||||||
|
"github.com/kava-labs/kava/x/cdp"
|
||||||
|
"github.com/kava-labs/kava/x/incentive/types"
|
||||||
|
"github.com/kava-labs/kava/x/kavadist"
|
||||||
|
validatorvesting "github.com/kava-labs/kava/x/validator-vesting"
|
||||||
|
abci "github.com/tendermint/tendermint/abci/types"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (suite *KeeperTestSuite) setupChain() {
|
||||||
|
// creates a new app state with 4 funded addresses and 1 module account
|
||||||
|
tApp := app.NewTestApp()
|
||||||
|
ctx := tApp.NewContext(true, abci.Header{Height: 1, Time: time.Unix(100, 0)})
|
||||||
|
_, addrs := app.GeneratePrivKeyAddressPairs(4)
|
||||||
|
authGS := app.NewAuthGenState(
|
||||||
|
addrs,
|
||||||
|
[]sdk.Coins{
|
||||||
|
cs(c("ukava", 400)),
|
||||||
|
cs(c("ukava", 400)),
|
||||||
|
cs(c("ukava", 400)),
|
||||||
|
cs(c("ukava", 400)),
|
||||||
|
})
|
||||||
|
tApp.InitializeFromGenesisStates(
|
||||||
|
authGS,
|
||||||
|
)
|
||||||
|
supplyKeeper := tApp.GetSupplyKeeper()
|
||||||
|
macc := supplyKeeper.GetModuleAccount(ctx, kavadist.ModuleName)
|
||||||
|
err := supplyKeeper.MintCoins(ctx, macc.GetName(), cs(c("ukava", 500)))
|
||||||
|
suite.Require().NoError(err)
|
||||||
|
|
||||||
|
// sets addrs[0] to be a periodic vesting account
|
||||||
|
ak := tApp.GetAccountKeeper()
|
||||||
|
acc := ak.GetAccount(ctx, addrs[0])
|
||||||
|
bacc := auth.NewBaseAccount(acc.GetAddress(), acc.GetCoins(), acc.GetPubKey(), acc.GetAccountNumber(), acc.GetSequence())
|
||||||
|
periods := vesting.Periods{
|
||||||
|
vesting.Period{Length: int64(1), Amount: cs(c("ukava", 100))},
|
||||||
|
vesting.Period{Length: int64(2), Amount: cs(c("ukava", 100))},
|
||||||
|
vesting.Period{Length: int64(8), Amount: cs(c("ukava", 100))},
|
||||||
|
vesting.Period{Length: int64(5), Amount: cs(c("ukava", 100))},
|
||||||
|
}
|
||||||
|
bva, err2 := vesting.NewBaseVestingAccount(bacc, cs(c("ukava", 400)), ctx.BlockTime().Unix()+16)
|
||||||
|
suite.Require().NoError(err2)
|
||||||
|
pva := vesting.NewPeriodicVestingAccountRaw(bva, ctx.BlockTime().Unix(), periods)
|
||||||
|
ak.SetAccount(ctx, pva)
|
||||||
|
|
||||||
|
// sets addrs[2] to be a validator vesting account
|
||||||
|
acc = ak.GetAccount(ctx, addrs[2])
|
||||||
|
bacc = auth.NewBaseAccount(acc.GetAddress(), acc.GetCoins(), acc.GetPubKey(), acc.GetAccountNumber(), acc.GetSequence())
|
||||||
|
bva, err2 = vesting.NewBaseVestingAccount(bacc, cs(c("ukava", 400)), ctx.BlockTime().Unix()+16)
|
||||||
|
suite.Require().NoError(err2)
|
||||||
|
vva := validatorvesting.NewValidatorVestingAccountRaw(bva, ctx.BlockTime().Unix(), periods, sdk.ConsAddress{}, nil, 90)
|
||||||
|
ak.SetAccount(ctx, vva)
|
||||||
|
suite.app = tApp
|
||||||
|
suite.keeper = tApp.GetIncentiveKeeper()
|
||||||
|
suite.ctx = ctx
|
||||||
|
suite.addrs = addrs
|
||||||
|
}
|
||||||
|
|
||||||
|
func (suite *KeeperTestSuite) setupExpiredClaims() {
|
||||||
|
// creates a new app state with 4 funded addresses
|
||||||
|
tApp := app.NewTestApp()
|
||||||
|
ctx := tApp.NewContext(true, abci.Header{Height: 1, Time: time.Unix(100, 0)})
|
||||||
|
_, addrs := app.GeneratePrivKeyAddressPairs(4)
|
||||||
|
authGS := app.NewAuthGenState(
|
||||||
|
addrs,
|
||||||
|
[]sdk.Coins{
|
||||||
|
cs(c("ukava", 400)),
|
||||||
|
cs(c("ukava", 400)),
|
||||||
|
cs(c("ukava", 400)),
|
||||||
|
cs(c("ukava", 400)),
|
||||||
|
})
|
||||||
|
tApp.InitializeFromGenesisStates(
|
||||||
|
authGS,
|
||||||
|
)
|
||||||
|
|
||||||
|
// creates two claim periods, one expired, and one that expires in the future
|
||||||
|
cp1 := types.NewClaimPeriod("bnb", 1, time.Unix(90, 0), time.Hour*8766)
|
||||||
|
cp2 := types.NewClaimPeriod("xrp", 1, time.Unix(110, 0), time.Hour*8766)
|
||||||
|
suite.keeper = tApp.GetIncentiveKeeper()
|
||||||
|
suite.keeper.SetClaimPeriod(ctx, cp1)
|
||||||
|
suite.keeper.SetClaimPeriod(ctx, cp2)
|
||||||
|
// creates one claim for the non-expired claim period and one claim for the expired claim period
|
||||||
|
c1 := types.NewClaim(addrs[0], c("ukava", 1000000), "bnb", 1)
|
||||||
|
c2 := types.NewClaim(addrs[0], c("ukava", 1000000), "xrp", 1)
|
||||||
|
suite.keeper.SetClaim(ctx, c1)
|
||||||
|
suite.keeper.SetClaim(ctx, c2)
|
||||||
|
suite.app = tApp
|
||||||
|
suite.ctx = ctx
|
||||||
|
suite.addrs = addrs
|
||||||
|
}
|
||||||
|
|
||||||
|
func (suite *KeeperTestSuite) TestSendCoinsToPeriodicVestingAccount() {
|
||||||
|
suite.setupChain()
|
||||||
|
|
||||||
|
type args struct {
|
||||||
|
coins sdk.Coins
|
||||||
|
length int64
|
||||||
|
}
|
||||||
|
|
||||||
|
type errArgs struct {
|
||||||
|
expectErr bool
|
||||||
|
errType error
|
||||||
|
}
|
||||||
|
|
||||||
|
type vestingAccountTest struct {
|
||||||
|
name string
|
||||||
|
blockTime time.Time
|
||||||
|
args args
|
||||||
|
errArgs errArgs
|
||||||
|
expectedPeriods vesting.Periods
|
||||||
|
expectedOriginalVesting sdk.Coins
|
||||||
|
expectedCoins sdk.Coins
|
||||||
|
expectedStartTime int64
|
||||||
|
expectedEndTime int64
|
||||||
|
}
|
||||||
|
|
||||||
|
type vestingAccountTests []vestingAccountTest
|
||||||
|
|
||||||
|
testCases := vestingAccountTests{
|
||||||
|
vestingAccountTest{
|
||||||
|
name: "insert period into an existing vesting schedule",
|
||||||
|
blockTime: time.Unix(100, 0),
|
||||||
|
args: args{coins: cs(c("ukava", 100)), length: 5},
|
||||||
|
errArgs: errArgs{expectErr: false, errType: nil},
|
||||||
|
expectedPeriods: vesting.Periods{
|
||||||
|
vesting.Period{Length: int64(1), Amount: cs(c("ukava", 100))},
|
||||||
|
vesting.Period{Length: int64(2), Amount: cs(c("ukava", 100))},
|
||||||
|
vesting.Period{Length: int64(2), Amount: cs(c("ukava", 100))},
|
||||||
|
vesting.Period{Length: int64(6), Amount: cs(c("ukava", 100))},
|
||||||
|
vesting.Period{Length: int64(5), Amount: cs(c("ukava", 100))},
|
||||||
|
},
|
||||||
|
expectedOriginalVesting: cs(c("ukava", 500)),
|
||||||
|
expectedCoins: cs(c("ukava", 500)),
|
||||||
|
expectedStartTime: int64(100),
|
||||||
|
expectedEndTime: int64(116),
|
||||||
|
},
|
||||||
|
vestingAccountTest{
|
||||||
|
name: "append period to the end of an existing vesting schedule",
|
||||||
|
blockTime: time.Unix(100, 0),
|
||||||
|
args: args{coins: cs(c("ukava", 100)), length: 17},
|
||||||
|
errArgs: errArgs{expectErr: false, errType: nil},
|
||||||
|
expectedPeriods: vesting.Periods{
|
||||||
|
vesting.Period{Length: int64(1), Amount: cs(c("ukava", 100))},
|
||||||
|
vesting.Period{Length: int64(2), Amount: cs(c("ukava", 100))},
|
||||||
|
vesting.Period{Length: int64(2), Amount: cs(c("ukava", 100))},
|
||||||
|
vesting.Period{Length: int64(6), Amount: cs(c("ukava", 100))},
|
||||||
|
vesting.Period{Length: int64(5), Amount: cs(c("ukava", 100))},
|
||||||
|
vesting.Period{Length: int64(1), Amount: cs(c("ukava", 100))},
|
||||||
|
},
|
||||||
|
expectedOriginalVesting: cs(c("ukava", 600)),
|
||||||
|
expectedCoins: cs(c("ukava", 600)),
|
||||||
|
expectedStartTime: int64(100),
|
||||||
|
expectedEndTime: int64(117),
|
||||||
|
},
|
||||||
|
vestingAccountTest{
|
||||||
|
name: "append period to the end of a completed vesting schedule",
|
||||||
|
blockTime: time.Unix(120, 0),
|
||||||
|
args: args{coins: cs(c("ukava", 100)), length: 5},
|
||||||
|
errArgs: errArgs{expectErr: false, errType: nil},
|
||||||
|
expectedPeriods: vesting.Periods{
|
||||||
|
vesting.Period{Length: int64(1), Amount: cs(c("ukava", 100))},
|
||||||
|
vesting.Period{Length: int64(2), Amount: cs(c("ukava", 100))},
|
||||||
|
vesting.Period{Length: int64(2), Amount: cs(c("ukava", 100))},
|
||||||
|
vesting.Period{Length: int64(6), Amount: cs(c("ukava", 100))},
|
||||||
|
vesting.Period{Length: int64(5), Amount: cs(c("ukava", 100))},
|
||||||
|
vesting.Period{Length: int64(1), Amount: cs(c("ukava", 100))},
|
||||||
|
vesting.Period{Length: int64(8), Amount: cs(c("ukava", 100))},
|
||||||
|
},
|
||||||
|
expectedOriginalVesting: cs(c("ukava", 700)),
|
||||||
|
expectedCoins: cs(c("ukava", 700)),
|
||||||
|
expectedStartTime: int64(100),
|
||||||
|
expectedEndTime: int64(125),
|
||||||
|
},
|
||||||
|
vestingAccountTest{
|
||||||
|
name: "prepend period to to an upcoming vesting schedule",
|
||||||
|
blockTime: time.Unix(90, 0),
|
||||||
|
args: args{coins: cs(c("ukava", 100)), length: 5},
|
||||||
|
errArgs: errArgs{expectErr: false, errType: nil},
|
||||||
|
expectedPeriods: vesting.Periods{
|
||||||
|
vesting.Period{Length: int64(5), Amount: cs(c("ukava", 100))},
|
||||||
|
vesting.Period{Length: int64(6), Amount: cs(c("ukava", 100))},
|
||||||
|
vesting.Period{Length: int64(2), Amount: cs(c("ukava", 100))},
|
||||||
|
vesting.Period{Length: int64(2), Amount: cs(c("ukava", 100))},
|
||||||
|
vesting.Period{Length: int64(6), Amount: cs(c("ukava", 100))},
|
||||||
|
vesting.Period{Length: int64(5), Amount: cs(c("ukava", 100))},
|
||||||
|
vesting.Period{Length: int64(1), Amount: cs(c("ukava", 100))},
|
||||||
|
vesting.Period{Length: int64(8), Amount: cs(c("ukava", 100))},
|
||||||
|
},
|
||||||
|
expectedOriginalVesting: cs(c("ukava", 800)),
|
||||||
|
expectedCoins: cs(c("ukava", 800)),
|
||||||
|
expectedStartTime: int64(90),
|
||||||
|
expectedEndTime: int64(125),
|
||||||
|
},
|
||||||
|
vestingAccountTest{
|
||||||
|
name: "add period that coincides with an existing end time",
|
||||||
|
blockTime: time.Unix(90, 0),
|
||||||
|
args: args{coins: cs(c("ukava", 100)), length: 11},
|
||||||
|
errArgs: errArgs{expectErr: false, errType: nil},
|
||||||
|
expectedPeriods: vesting.Periods{
|
||||||
|
vesting.Period{Length: int64(5), Amount: cs(c("ukava", 100))},
|
||||||
|
vesting.Period{Length: int64(6), Amount: cs(c("ukava", 200))},
|
||||||
|
vesting.Period{Length: int64(2), Amount: cs(c("ukava", 100))},
|
||||||
|
vesting.Period{Length: int64(2), Amount: cs(c("ukava", 100))},
|
||||||
|
vesting.Period{Length: int64(6), Amount: cs(c("ukava", 100))},
|
||||||
|
vesting.Period{Length: int64(5), Amount: cs(c("ukava", 100))},
|
||||||
|
vesting.Period{Length: int64(1), Amount: cs(c("ukava", 100))},
|
||||||
|
vesting.Period{Length: int64(8), Amount: cs(c("ukava", 100))},
|
||||||
|
},
|
||||||
|
expectedOriginalVesting: cs(c("ukava", 900)),
|
||||||
|
expectedCoins: cs(c("ukava", 900)),
|
||||||
|
expectedStartTime: int64(90),
|
||||||
|
expectedEndTime: int64(125),
|
||||||
|
},
|
||||||
|
vestingAccountTest{
|
||||||
|
name: "insufficient module account balance",
|
||||||
|
blockTime: time.Unix(90, 0),
|
||||||
|
args: args{coins: cs(c("ukava", 1000)), length: 11},
|
||||||
|
errArgs: errArgs{expectErr: true, errType: types.ErrInsufficientModAccountBalance},
|
||||||
|
expectedPeriods: vesting.Periods{},
|
||||||
|
expectedOriginalVesting: sdk.Coins{},
|
||||||
|
expectedCoins: sdk.Coins{},
|
||||||
|
expectedStartTime: int64(0),
|
||||||
|
expectedEndTime: int64(0),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tc := range testCases {
|
||||||
|
suite.Run(tc.name, func() {
|
||||||
|
suite.ctx = suite.ctx.WithBlockTime(tc.blockTime)
|
||||||
|
err := suite.keeper.SendTimeLockedCoinsToAccount(suite.ctx, kavadist.ModuleName, suite.addrs[0], tc.args.coins, tc.args.length)
|
||||||
|
if tc.errArgs.expectErr {
|
||||||
|
suite.Require().True(errors.Is(err, tc.errArgs.errType))
|
||||||
|
} else {
|
||||||
|
suite.Require().NoError(err)
|
||||||
|
acc := suite.getAccount(suite.addrs[0])
|
||||||
|
vacc, ok := acc.(*vesting.PeriodicVestingAccount)
|
||||||
|
suite.True(ok)
|
||||||
|
suite.Equal(tc.expectedPeriods, vacc.VestingPeriods)
|
||||||
|
suite.Equal(tc.expectedOriginalVesting, vacc.OriginalVesting)
|
||||||
|
suite.Equal(tc.expectedCoins, vacc.Coins)
|
||||||
|
suite.Equal(tc.expectedStartTime, vacc.StartTime)
|
||||||
|
suite.Equal(tc.expectedEndTime, vacc.EndTime)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (suite *KeeperTestSuite) TestSendCoinsToBaseAccount() {
|
||||||
|
suite.setupChain()
|
||||||
|
// send coins to base account
|
||||||
|
err := suite.keeper.SendTimeLockedCoinsToAccount(suite.ctx, kavadist.ModuleName, suite.addrs[1], cs(c("ukava", 100)), 5)
|
||||||
|
suite.Require().NoError(err)
|
||||||
|
acc := suite.getAccount(suite.addrs[1])
|
||||||
|
vacc, ok := acc.(*vesting.PeriodicVestingAccount)
|
||||||
|
suite.True(ok)
|
||||||
|
expectedPeriods := vesting.Periods{
|
||||||
|
vesting.Period{Length: int64(5), Amount: cs(c("ukava", 100))},
|
||||||
|
}
|
||||||
|
suite.Equal(expectedPeriods, vacc.VestingPeriods)
|
||||||
|
suite.Equal(cs(c("ukava", 100)), vacc.OriginalVesting)
|
||||||
|
suite.Equal(cs(c("ukava", 500)), vacc.Coins)
|
||||||
|
suite.Equal(int64(105), vacc.EndTime)
|
||||||
|
suite.Equal(int64(100), vacc.StartTime)
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
func (suite *KeeperTestSuite) TestSendCoinsToInvalidAccount() {
|
||||||
|
suite.setupChain()
|
||||||
|
err := suite.keeper.SendTimeLockedCoinsToAccount(suite.ctx, kavadist.ModuleName, suite.addrs[2], cs(c("ukava", 100)), 5)
|
||||||
|
suite.Require().True(errors.Is(err, types.ErrInvalidAccountType))
|
||||||
|
macc := suite.getModuleAccount(cdp.ModuleName)
|
||||||
|
err = suite.keeper.SendTimeLockedCoinsToAccount(suite.ctx, kavadist.ModuleName, macc.GetAddress(), cs(c("ukava", 100)), 5)
|
||||||
|
suite.Require().True(errors.Is(err, types.ErrInvalidAccountType))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (suite *KeeperTestSuite) TestPayoutClaim() {
|
||||||
|
suite.setupChain() // adds 3 accounts - 1 periodic vesting account, 1 base account, and 1 validator vesting account
|
||||||
|
|
||||||
|
// add 2 claims that correspond to an existing claim period and one claim that has no corresponding claim period
|
||||||
|
cp1 := types.NewClaimPeriod("bnb", 1, suite.ctx.BlockTime().Add(time.Hour*168), time.Hour*8766)
|
||||||
|
suite.keeper.SetClaimPeriod(suite.ctx, cp1)
|
||||||
|
// valid claim for addrs[0]
|
||||||
|
c1 := types.NewClaim(suite.addrs[0], c("ukava", 100), "bnb", 1)
|
||||||
|
// invalid claim for addrs[0]
|
||||||
|
c2 := types.NewClaim(suite.addrs[0], c("ukava", 100), "xrp", 1)
|
||||||
|
// valid claim for addrs[1]
|
||||||
|
c3 := types.NewClaim(suite.addrs[1], c("ukava", 100), "bnb", 1)
|
||||||
|
suite.keeper.SetClaim(suite.ctx, c1)
|
||||||
|
suite.keeper.SetClaim(suite.ctx, c2)
|
||||||
|
suite.keeper.SetClaim(suite.ctx, c3)
|
||||||
|
|
||||||
|
// existing claim with corresponding claim period successfully claimed by existing periodic vesting account
|
||||||
|
err := suite.keeper.PayoutClaim(suite.ctx, suite.addrs[0], "bnb", 1)
|
||||||
|
suite.Require().NoError(err)
|
||||||
|
acc := suite.getAccount(suite.addrs[0])
|
||||||
|
// account is a periodic vesting account
|
||||||
|
vacc, ok := acc.(*vesting.PeriodicVestingAccount)
|
||||||
|
suite.True(ok)
|
||||||
|
// vesting balance is correct
|
||||||
|
suite.Equal(cs(c("ukava", 500)), vacc.OriginalVesting)
|
||||||
|
|
||||||
|
// existing claim with corresponding claim period successfully claimed by base account
|
||||||
|
err = suite.keeper.PayoutClaim(suite.ctx, suite.addrs[1], "bnb", 1)
|
||||||
|
suite.Require().NoError(err)
|
||||||
|
acc = suite.getAccount(suite.addrs[1])
|
||||||
|
// account has become a periodic vesting account
|
||||||
|
vacc, ok = acc.(*vesting.PeriodicVestingAccount)
|
||||||
|
suite.True(ok)
|
||||||
|
// vesting balance is correct
|
||||||
|
suite.Equal(cs(c("ukava", 100)), vacc.OriginalVesting)
|
||||||
|
|
||||||
|
// addrs[3] has no claims
|
||||||
|
err = suite.keeper.PayoutClaim(suite.ctx, suite.addrs[3], "bnb", 1)
|
||||||
|
suite.Require().True(errors.Is(err, types.ErrClaimNotFound))
|
||||||
|
// addrs[0] has an xrp claim, but there is not corresponding claim period
|
||||||
|
err = suite.keeper.PayoutClaim(suite.ctx, suite.addrs[0], "xrp", 1)
|
||||||
|
suite.Require().True(errors.Is(err, types.ErrClaimPeriodNotFound))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (suite *KeeperTestSuite) TestDeleteExpiredClaimPeriods() {
|
||||||
|
suite.setupExpiredClaims() // creates new app state with one non-expired claim period (xrp) and one expired claim period (bnb) as well as a claim that corresponds to each claim period
|
||||||
|
|
||||||
|
// both claim periods are present
|
||||||
|
_, found := suite.keeper.GetClaimPeriod(suite.ctx, 1, "bnb")
|
||||||
|
suite.True(found)
|
||||||
|
_, found = suite.keeper.GetClaimPeriod(suite.ctx, 1, "xrp")
|
||||||
|
suite.True(found)
|
||||||
|
// both claims are present
|
||||||
|
_, found = suite.keeper.GetClaim(suite.ctx, suite.addrs[0], "bnb", 1)
|
||||||
|
suite.True(found)
|
||||||
|
_, found = suite.keeper.GetClaim(suite.ctx, suite.addrs[0], "xrp", 1)
|
||||||
|
suite.True(found)
|
||||||
|
|
||||||
|
// expired claim period and associated claims should get deleted
|
||||||
|
suite.NotPanics(func() {
|
||||||
|
suite.keeper.DeleteExpiredClaimsAndClaimPeriods(suite.ctx)
|
||||||
|
})
|
||||||
|
// expired claim period and claim are not found
|
||||||
|
_, found = suite.keeper.GetClaimPeriod(suite.ctx, 1, "bnb")
|
||||||
|
suite.False(found)
|
||||||
|
_, found = suite.keeper.GetClaim(suite.ctx, suite.addrs[0], "bnb", 1)
|
||||||
|
suite.False(found)
|
||||||
|
// non-expired claim period and claim are found
|
||||||
|
_, found = suite.keeper.GetClaimPeriod(suite.ctx, 1, "xrp")
|
||||||
|
suite.True(found)
|
||||||
|
_, found = suite.keeper.GetClaim(suite.ctx, suite.addrs[0], "xrp", 1)
|
||||||
|
suite.True(found)
|
||||||
|
|
||||||
|
}
|
32
x/incentive/keeper/querier_test.go
Normal file
32
x/incentive/keeper/querier_test.go
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
package keeper_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/kava-labs/kava/x/incentive/keeper"
|
||||||
|
"github.com/kava-labs/kava/x/incentive/types"
|
||||||
|
abci "github.com/tendermint/tendermint/abci/types"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (suite *KeeperTestSuite) TestQuerier() {
|
||||||
|
suite.addObjectsToStore()
|
||||||
|
querier := keeper.NewQuerier(suite.keeper)
|
||||||
|
bz, err := querier(suite.ctx, []string{types.QueryGetParams}, abci.RequestQuery{})
|
||||||
|
suite.Require().NoError(err)
|
||||||
|
suite.NotNil(bz)
|
||||||
|
|
||||||
|
var p types.Params
|
||||||
|
suite.Nil(types.ModuleCdc.UnmarshalJSON(bz, &p))
|
||||||
|
|
||||||
|
claimQueryParams := types.NewQueryClaimsParams(suite.addrs[0], "bnb")
|
||||||
|
query := abci.RequestQuery{
|
||||||
|
Path: strings.Join([]string{"custom", types.QuerierRoute, types.QueryGetClaims}, "/"),
|
||||||
|
Data: types.ModuleCdc.MustMarshalJSON(claimQueryParams),
|
||||||
|
}
|
||||||
|
bz, err = querier(suite.ctx, []string{types.QueryGetClaims}, query)
|
||||||
|
|
||||||
|
var claims types.Claims
|
||||||
|
suite.Nil(types.ModuleCdc.UnmarshalJSON(bz, &claims))
|
||||||
|
suite.Equal(1, len(claims))
|
||||||
|
suite.Equal(types.Claims{types.NewClaim(suite.addrs[0], c("ukava", 1000000), "bnb", 1)}, claims)
|
||||||
|
}
|
258
x/incentive/keeper/rewards_test.go
Normal file
258
x/incentive/keeper/rewards_test.go
Normal file
@ -0,0 +1,258 @@
|
|||||||
|
package keeper_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"time"
|
||||||
|
|
||||||
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||||
|
"github.com/kava-labs/kava/app"
|
||||||
|
"github.com/kava-labs/kava/x/cdp"
|
||||||
|
"github.com/kava-labs/kava/x/incentive/types"
|
||||||
|
"github.com/kava-labs/kava/x/pricefeed"
|
||||||
|
abci "github.com/tendermint/tendermint/abci/types"
|
||||||
|
tmtime "github.com/tendermint/tendermint/types/time"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (suite *KeeperTestSuite) TestExpireRewardPeriod() {
|
||||||
|
rp := types.NewRewardPeriod("bnb", suite.ctx.BlockTime(), suite.ctx.BlockTime().Add(time.Hour*168), c("ukava", 100000000), suite.ctx.BlockTime().Add(time.Hour*168*2), time.Hour*8766)
|
||||||
|
suite.keeper.SetRewardPeriod(suite.ctx, rp)
|
||||||
|
suite.keeper.SetNextClaimPeriodID(suite.ctx, "bnb", 1)
|
||||||
|
suite.NotPanics(func() {
|
||||||
|
suite.keeper.HandleRewardPeriodExpiry(suite.ctx, rp)
|
||||||
|
})
|
||||||
|
_, found := suite.keeper.GetClaimPeriod(suite.ctx, 1, "bnb")
|
||||||
|
suite.True(found)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (suite *KeeperTestSuite) TestAddToClaim() {
|
||||||
|
rp := types.NewRewardPeriod("bnb", suite.ctx.BlockTime(), suite.ctx.BlockTime().Add(time.Hour*168), c("ukava", 100000000), suite.ctx.BlockTime().Add(time.Hour*168*2), time.Hour*8766)
|
||||||
|
suite.keeper.SetRewardPeriod(suite.ctx, rp)
|
||||||
|
suite.keeper.SetNextClaimPeriodID(suite.ctx, "bnb", 1)
|
||||||
|
suite.keeper.HandleRewardPeriodExpiry(suite.ctx, rp)
|
||||||
|
c1 := types.NewClaim(suite.addrs[0], c("ukava", 1000000), "bnb", 1)
|
||||||
|
suite.keeper.SetClaim(suite.ctx, c1)
|
||||||
|
suite.NotPanics(func() {
|
||||||
|
suite.keeper.AddToClaim(suite.ctx, suite.addrs[0], "bnb", 1, c("ukava", 1000000))
|
||||||
|
})
|
||||||
|
testC, _ := suite.keeper.GetClaim(suite.ctx, suite.addrs[0], "bnb", 1)
|
||||||
|
suite.Equal(c("ukava", 2000000), testC.Reward)
|
||||||
|
|
||||||
|
suite.NotPanics(func() {
|
||||||
|
suite.keeper.AddToClaim(suite.ctx, suite.addrs[0], "xpr", 1, c("ukava", 1000000))
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (suite *KeeperTestSuite) TestCreateRewardPeriod() {
|
||||||
|
reward := types.NewReward(true, "bnb", c("ukava", 1000000000), time.Hour*7*24, time.Hour*24*365, time.Hour*7*24)
|
||||||
|
suite.NotPanics(func() {
|
||||||
|
suite.keeper.CreateNewRewardPeriod(suite.ctx, reward)
|
||||||
|
})
|
||||||
|
_, found := suite.keeper.GetRewardPeriod(suite.ctx, "bnb")
|
||||||
|
suite.True(found)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (suite *KeeperTestSuite) TestCreateAndDeleteRewardsPeriods() {
|
||||||
|
reward1 := types.NewReward(true, "bnb", c("ukava", 1000000000), time.Hour*7*24, time.Hour*24*365, time.Hour*7*24)
|
||||||
|
reward2 := types.NewReward(false, "xrp", c("ukava", 1000000000), time.Hour*7*24, time.Hour*24*365, time.Hour*7*24)
|
||||||
|
reward3 := types.NewReward(false, "btc", c("ukava", 1000000000), time.Hour*7*24, time.Hour*24*365, time.Hour*7*24)
|
||||||
|
// add a reward period to the store for a non-active reward
|
||||||
|
suite.NotPanics(func() {
|
||||||
|
suite.keeper.CreateNewRewardPeriod(suite.ctx, reward3)
|
||||||
|
})
|
||||||
|
params := types.NewParams(true, types.Rewards{reward1, reward2, reward3})
|
||||||
|
suite.keeper.SetParams(suite.ctx, params)
|
||||||
|
|
||||||
|
suite.NotPanics(func() {
|
||||||
|
suite.keeper.CreateAndDeleteRewardPeriods(suite.ctx)
|
||||||
|
})
|
||||||
|
testCases := []struct {
|
||||||
|
name string
|
||||||
|
arg string
|
||||||
|
expectFound bool
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
"active reward period",
|
||||||
|
"bnb",
|
||||||
|
true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"attempt to add inactive reward period",
|
||||||
|
"xrp",
|
||||||
|
false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"remove inactive reward period",
|
||||||
|
"btc",
|
||||||
|
false,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
for _, tc := range testCases {
|
||||||
|
suite.Run(tc.name, func() {
|
||||||
|
_, found := suite.keeper.GetRewardPeriod(suite.ctx, tc.arg)
|
||||||
|
if tc.expectFound {
|
||||||
|
suite.True(found)
|
||||||
|
} else {
|
||||||
|
suite.False(found)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (suite *KeeperTestSuite) TestApplyRewardsToCdps() {
|
||||||
|
suite.setupCdpChain() // creates a test app with 3 BNB cdps and usdx incentives for bnb - each reward period is one week
|
||||||
|
|
||||||
|
// move the context forward by 100 periods
|
||||||
|
suite.ctx = suite.ctx.WithBlockTime(suite.ctx.BlockTime().Add(time.Second * 100))
|
||||||
|
// apply rewards to BNB cdps
|
||||||
|
suite.NotPanics(func() {
|
||||||
|
suite.keeper.ApplyRewardsToCdps(suite.ctx)
|
||||||
|
})
|
||||||
|
// each cdp should have a claim
|
||||||
|
claims := types.Claims{}
|
||||||
|
suite.keeper.IterateClaims(suite.ctx, func(c types.Claim) (stop bool) {
|
||||||
|
claims = append(claims, c)
|
||||||
|
return false
|
||||||
|
})
|
||||||
|
suite.Equal(3, len(claims))
|
||||||
|
// there should be no associated claim period, because the reward period has not ended yet
|
||||||
|
_, found := suite.keeper.GetClaimPeriod(suite.ctx, 1, "bnb")
|
||||||
|
suite.False(found)
|
||||||
|
|
||||||
|
// move ctx to the reward period expiry and check that the claim period has been created and the next claim period id has increased
|
||||||
|
suite.ctx = suite.ctx.WithBlockTime(suite.ctx.BlockTime().Add(time.Hour * 24 * 7))
|
||||||
|
|
||||||
|
suite.NotPanics(func() {
|
||||||
|
// apply rewards to cdps
|
||||||
|
suite.keeper.ApplyRewardsToCdps(suite.ctx)
|
||||||
|
// delete the old reward period amd create a new one
|
||||||
|
suite.keeper.CreateAndDeleteRewardPeriods(suite.ctx)
|
||||||
|
})
|
||||||
|
_, found = suite.keeper.GetClaimPeriod(suite.ctx, 1, "bnb")
|
||||||
|
suite.True(found)
|
||||||
|
testID := suite.keeper.GetNextClaimPeriodID(suite.ctx, "bnb")
|
||||||
|
suite.Equal(uint64(2), testID)
|
||||||
|
|
||||||
|
// move the context forward by 100 periods
|
||||||
|
suite.ctx = suite.ctx.WithBlockTime(suite.ctx.BlockTime().Add(time.Second * 100))
|
||||||
|
// run the begin blocker functions
|
||||||
|
suite.NotPanics(func() {
|
||||||
|
suite.keeper.DeleteExpiredClaimsAndClaimPeriods(suite.ctx)
|
||||||
|
suite.keeper.ApplyRewardsToCdps(suite.ctx)
|
||||||
|
suite.keeper.CreateAndDeleteRewardPeriods(suite.ctx)
|
||||||
|
})
|
||||||
|
// each cdp should now have two claims
|
||||||
|
claims = types.Claims{}
|
||||||
|
suite.keeper.IterateClaims(suite.ctx, func(c types.Claim) (stop bool) {
|
||||||
|
claims = append(claims, c)
|
||||||
|
return false
|
||||||
|
})
|
||||||
|
suite.Equal(6, len(claims))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (suite *KeeperTestSuite) setupCdpChain() {
|
||||||
|
// creates a new test app with bnb as the only asset the pricefeed and cdp modules
|
||||||
|
// funds three addresses and creates 3 cdps, funded with 100 BNB, 1000 BNB, and 10000 BNB
|
||||||
|
// each CDP draws 10, 100, and 1000 USDX respectively
|
||||||
|
// adds usdx incentives for bnb - 1000 KAVA per week with a 1 year time lock
|
||||||
|
|
||||||
|
tApp := app.NewTestApp()
|
||||||
|
ctx := tApp.NewContext(true, abci.Header{Height: 1, Time: tmtime.Now()})
|
||||||
|
// need pricefeed and cdp gen state with one collateral
|
||||||
|
pricefeedGS := pricefeed.GenesisState{
|
||||||
|
Params: pricefeed.Params{
|
||||||
|
Markets: []pricefeed.Market{
|
||||||
|
pricefeed.Market{MarketID: "bnb:usd", BaseAsset: "bnb", QuoteAsset: "usd", Oracles: []sdk.AccAddress{}, Active: true},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
PostedPrices: []pricefeed.PostedPrice{
|
||||||
|
pricefeed.PostedPrice{
|
||||||
|
MarketID: "bnb:usd",
|
||||||
|
OracleAddress: sdk.AccAddress{},
|
||||||
|
Price: d("12.29"),
|
||||||
|
Expiry: time.Now().Add(100000 * time.Hour),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
// need incentive params for one collateral
|
||||||
|
cdpGS := cdp.GenesisState{
|
||||||
|
Params: cdp.Params{
|
||||||
|
GlobalDebtLimit: sdk.NewCoins(sdk.NewInt64Coin("usdx", 1000000000000)),
|
||||||
|
SurplusAuctionThreshold: cdp.DefaultSurplusThreshold,
|
||||||
|
DebtAuctionThreshold: cdp.DefaultDebtThreshold,
|
||||||
|
SavingsDistributionFrequency: cdp.DefaultSavingsDistributionFrequency,
|
||||||
|
CollateralParams: cdp.CollateralParams{
|
||||||
|
{
|
||||||
|
Denom: "bnb",
|
||||||
|
LiquidationRatio: sdk.MustNewDecFromStr("2.0"),
|
||||||
|
DebtLimit: sdk.NewCoins(sdk.NewInt64Coin("usdx", 1000000000000)),
|
||||||
|
StabilityFee: sdk.MustNewDecFromStr("1.000000001547125958"), // %5 apr
|
||||||
|
LiquidationPenalty: d("0.05"),
|
||||||
|
AuctionSize: i(10000000000),
|
||||||
|
Prefix: 0x20,
|
||||||
|
MarketID: "bnb:usd",
|
||||||
|
ConversionFactor: i(8),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
DebtParams: cdp.DebtParams{
|
||||||
|
{
|
||||||
|
Denom: "usdx",
|
||||||
|
ReferenceAsset: "usd",
|
||||||
|
ConversionFactor: i(6),
|
||||||
|
DebtFloor: i(10000000),
|
||||||
|
SavingsRate: d("0.95"),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
StartingCdpID: cdp.DefaultCdpStartingID,
|
||||||
|
DebtDenom: cdp.DefaultDebtDenom,
|
||||||
|
GovDenom: cdp.DefaultGovDenom,
|
||||||
|
CDPs: cdp.CDPs{},
|
||||||
|
PreviousDistributionTime: cdp.DefaultPreviousDistributionTime,
|
||||||
|
}
|
||||||
|
incentiveGS := types.NewGenesisState(
|
||||||
|
types.NewParams(
|
||||||
|
true, types.Rewards{types.NewReward(true, "bnb", c("ukava", 1000000000), time.Hour*7*24, time.Hour*24*365, time.Hour*7*24)},
|
||||||
|
),
|
||||||
|
types.DefaultPreviousBlockTime,
|
||||||
|
types.RewardPeriods{types.NewRewardPeriod("bnb", ctx.BlockTime(), ctx.BlockTime().Add(time.Hour*7*24), c("ukava", 1000), ctx.BlockTime().Add(time.Hour*7*24*2), time.Hour*365*24)},
|
||||||
|
types.ClaimPeriods{},
|
||||||
|
types.Claims{},
|
||||||
|
types.GenesisClaimPeriodIDs{})
|
||||||
|
pricefeedAppGs := app.GenesisState{pricefeed.ModuleName: pricefeed.ModuleCdc.MustMarshalJSON(pricefeedGS)}
|
||||||
|
cdpAppGs := app.GenesisState{cdp.ModuleName: cdp.ModuleCdc.MustMarshalJSON(cdpGS)}
|
||||||
|
incentiveAppGs := app.GenesisState{types.ModuleName: types.ModuleCdc.MustMarshalJSON(incentiveGS)}
|
||||||
|
_, addrs := app.GeneratePrivKeyAddressPairs(3)
|
||||||
|
authGS := app.NewAuthGenState(
|
||||||
|
addrs[0:3],
|
||||||
|
[]sdk.Coins{
|
||||||
|
cs(c("bnb", 10000000000)),
|
||||||
|
cs(c("bnb", 100000000000)),
|
||||||
|
cs(c("bnb", 1000000000000)),
|
||||||
|
})
|
||||||
|
tApp.InitializeFromGenesisStates(
|
||||||
|
authGS,
|
||||||
|
pricefeedAppGs,
|
||||||
|
incentiveAppGs,
|
||||||
|
cdpAppGs,
|
||||||
|
)
|
||||||
|
suite.app = tApp
|
||||||
|
suite.keeper = tApp.GetIncentiveKeeper()
|
||||||
|
suite.ctx = ctx
|
||||||
|
// create 3 cdps
|
||||||
|
cdpKeeper := tApp.GetCDPKeeper()
|
||||||
|
err := cdpKeeper.AddCdp(suite.ctx, addrs[0], cs(c("bnb", 10000000000)), cs(c("usdx", 10000000)))
|
||||||
|
suite.Require().NoError(err)
|
||||||
|
err = cdpKeeper.AddCdp(suite.ctx, addrs[1], cs(c("bnb", 100000000000)), cs(c("usdx", 100000000)))
|
||||||
|
suite.Require().NoError(err)
|
||||||
|
err = cdpKeeper.AddCdp(suite.ctx, addrs[2], cs(c("bnb", 1000000000000)), cs(c("usdx", 1000000000)))
|
||||||
|
suite.Require().NoError(err)
|
||||||
|
// total usd is 1110
|
||||||
|
|
||||||
|
// set the previous block time
|
||||||
|
suite.keeper.SetPreviousBlockTime(suite.ctx, suite.ctx.BlockTime())
|
||||||
|
}
|
||||||
|
|
||||||
|
// Avoid cluttering test cases with long function names
|
||||||
|
func i(in int64) sdk.Int { return sdk.NewInt(in) }
|
||||||
|
func d(str string) sdk.Dec { return sdk.MustNewDecFromStr(str) }
|
||||||
|
func c(denom string, amount int64) sdk.Coin { return sdk.NewInt64Coin(denom, amount) }
|
||||||
|
func cs(coins ...sdk.Coin) sdk.Coins { return sdk.NewCoins(coins...) }
|
@ -47,9 +47,9 @@ func (suite *MsgTestSuite) TestMsgValidation() {
|
|||||||
msg := types.NewMsgClaimReward(t.from, t.denom)
|
msg := types.NewMsgClaimReward(t.from, t.denom)
|
||||||
err := msg.ValidateBasic()
|
err := msg.ValidateBasic()
|
||||||
if t.expectPass {
|
if t.expectPass {
|
||||||
suite.NoError(err)
|
suite.Require().NoError(err)
|
||||||
} else {
|
} else {
|
||||||
suite.Error(err)
|
suite.Require().Error(err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -112,7 +112,7 @@ func validateRewardsParam(i interface{}) error {
|
|||||||
return fmt.Errorf("reward timelock must be non-negative, is %s for %s", reward.TimeLock.String(), reward.Denom)
|
return fmt.Errorf("reward timelock must be non-negative, is %s for %s", reward.TimeLock.String(), reward.Denom)
|
||||||
}
|
}
|
||||||
if int(reward.ClaimDuration.Seconds()) <= 0 {
|
if int(reward.ClaimDuration.Seconds()) <= 0 {
|
||||||
return fmt.Errorf("reward timelock must be positive, is %s for %s", reward.ClaimDuration.String(), reward.Denom)
|
return fmt.Errorf("claim duration must be positive, is %s for %s", reward.ClaimDuration.String(), reward.Denom)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
package types_test
|
package types_test
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"strings"
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
@ -10,8 +11,14 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
type paramTest struct {
|
type paramTest struct {
|
||||||
params types.Params
|
name string
|
||||||
|
params types.Params
|
||||||
|
errResult errResult
|
||||||
|
}
|
||||||
|
|
||||||
|
type errResult struct {
|
||||||
expectPass bool
|
expectPass bool
|
||||||
|
contains string
|
||||||
}
|
}
|
||||||
|
|
||||||
type ParamTestSuite struct {
|
type ParamTestSuite struct {
|
||||||
@ -21,129 +28,189 @@ type ParamTestSuite struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (suite *ParamTestSuite) SetupTest() {
|
func (suite *ParamTestSuite) SetupTest() {
|
||||||
p1 := types.Params{
|
|
||||||
Active: true,
|
|
||||||
Rewards: types.Rewards{
|
|
||||||
types.Reward{
|
|
||||||
Active: true,
|
|
||||||
Denom: "bnb",
|
|
||||||
AvailableRewards: sdk.NewCoin("ukava", sdk.NewInt(10000000000)),
|
|
||||||
Duration: time.Hour * 24 * 7,
|
|
||||||
TimeLock: time.Hour * 8766,
|
|
||||||
ClaimDuration: time.Hour * 24 * 14,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
p2 := types.Params{
|
|
||||||
Active: true,
|
|
||||||
Rewards: types.Rewards{
|
|
||||||
types.Reward{
|
|
||||||
Active: true,
|
|
||||||
Denom: "bnb",
|
|
||||||
AvailableRewards: sdk.NewCoin("ukava", sdk.NewInt(10000000000)),
|
|
||||||
Duration: time.Hour * 24 * 7,
|
|
||||||
TimeLock: time.Hour * 8766,
|
|
||||||
ClaimDuration: time.Hour * 24 * 14,
|
|
||||||
},
|
|
||||||
types.Reward{
|
|
||||||
Active: true,
|
|
||||||
Denom: "bnb",
|
|
||||||
AvailableRewards: sdk.NewCoin("ukava", sdk.NewInt(10000000000)),
|
|
||||||
Duration: time.Hour * 24 * 7,
|
|
||||||
TimeLock: time.Hour * 8766,
|
|
||||||
ClaimDuration: time.Hour * 24 * 14,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
p3 := types.Params{
|
|
||||||
Active: true,
|
|
||||||
Rewards: types.Rewards{
|
|
||||||
types.Reward{
|
|
||||||
Active: true,
|
|
||||||
Denom: "bnb",
|
|
||||||
AvailableRewards: sdk.NewCoin("ukava", sdk.NewInt(10000000000)),
|
|
||||||
Duration: time.Hour * -24 * 7,
|
|
||||||
TimeLock: time.Hour * 8766,
|
|
||||||
ClaimDuration: time.Hour * 24 * 14,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
p4 := types.Params{
|
|
||||||
Active: true,
|
|
||||||
Rewards: types.Rewards{
|
|
||||||
types.Reward{
|
|
||||||
Active: true,
|
|
||||||
Denom: "bnb",
|
|
||||||
AvailableRewards: sdk.NewCoin("ukava", sdk.NewInt(10000000000)),
|
|
||||||
Duration: time.Hour * 24 * 7,
|
|
||||||
TimeLock: time.Hour * -8766,
|
|
||||||
ClaimDuration: time.Hour * 24 * 14,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
p5 := types.Params{
|
|
||||||
Active: true,
|
|
||||||
Rewards: types.Rewards{
|
|
||||||
types.Reward{
|
|
||||||
Active: true,
|
|
||||||
Denom: "bnb",
|
|
||||||
AvailableRewards: sdk.NewCoin("ukava", sdk.NewInt(10000000000)),
|
|
||||||
Duration: time.Hour * 24 * 7,
|
|
||||||
TimeLock: time.Hour * 8766,
|
|
||||||
ClaimDuration: time.Hour * 0,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
p6 := types.Params{
|
|
||||||
Active: true,
|
|
||||||
Rewards: types.Rewards{
|
|
||||||
types.Reward{
|
|
||||||
Active: true,
|
|
||||||
Denom: "bnb",
|
|
||||||
AvailableRewards: sdk.NewCoin("ukava", sdk.NewInt(0)),
|
|
||||||
Duration: time.Hour * 24 * 7,
|
|
||||||
TimeLock: time.Hour * 8766,
|
|
||||||
ClaimDuration: time.Hour * 0,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
suite.tests = []paramTest{
|
suite.tests = []paramTest{
|
||||||
paramTest{
|
paramTest{
|
||||||
params: p1,
|
name: "valid - active",
|
||||||
expectPass: true,
|
params: types.Params{
|
||||||
|
Active: true,
|
||||||
|
Rewards: types.Rewards{
|
||||||
|
types.Reward{
|
||||||
|
Active: true,
|
||||||
|
Denom: "bnb",
|
||||||
|
AvailableRewards: sdk.NewCoin("ukava", sdk.NewInt(10000000000)),
|
||||||
|
Duration: time.Hour * 24 * 7,
|
||||||
|
TimeLock: time.Hour * 8766,
|
||||||
|
ClaimDuration: time.Hour * 24 * 14,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
errResult: errResult{
|
||||||
|
expectPass: true,
|
||||||
|
contains: "",
|
||||||
|
},
|
||||||
},
|
},
|
||||||
paramTest{
|
paramTest{
|
||||||
params: p2,
|
name: "valid - inactive",
|
||||||
expectPass: false,
|
params: types.Params{
|
||||||
|
Active: false,
|
||||||
|
Rewards: types.Rewards{
|
||||||
|
types.Reward{
|
||||||
|
Active: true,
|
||||||
|
Denom: "bnb",
|
||||||
|
AvailableRewards: sdk.NewCoin("ukava", sdk.NewInt(10000000000)),
|
||||||
|
Duration: time.Hour * 24 * 7,
|
||||||
|
TimeLock: time.Hour * 8766,
|
||||||
|
ClaimDuration: time.Hour * 24 * 14,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
errResult: errResult{
|
||||||
|
expectPass: true,
|
||||||
|
contains: "",
|
||||||
|
},
|
||||||
},
|
},
|
||||||
paramTest{
|
paramTest{
|
||||||
params: p3,
|
name: "duplicate reward",
|
||||||
expectPass: false,
|
params: types.Params{
|
||||||
|
Active: true,
|
||||||
|
Rewards: types.Rewards{
|
||||||
|
types.Reward{
|
||||||
|
Active: true,
|
||||||
|
Denom: "bnb",
|
||||||
|
AvailableRewards: sdk.NewCoin("ukava", sdk.NewInt(10000000000)),
|
||||||
|
Duration: time.Hour * 24 * 7,
|
||||||
|
TimeLock: time.Hour * 8766,
|
||||||
|
ClaimDuration: time.Hour * 24 * 14,
|
||||||
|
},
|
||||||
|
types.Reward{
|
||||||
|
Active: true,
|
||||||
|
Denom: "bnb",
|
||||||
|
AvailableRewards: sdk.NewCoin("ukava", sdk.NewInt(10000000000)),
|
||||||
|
Duration: time.Hour * 24 * 7,
|
||||||
|
TimeLock: time.Hour * 8766,
|
||||||
|
ClaimDuration: time.Hour * 24 * 14,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
errResult: errResult{
|
||||||
|
expectPass: false,
|
||||||
|
contains: "cannot have duplicate reward denoms",
|
||||||
|
},
|
||||||
},
|
},
|
||||||
paramTest{
|
paramTest{
|
||||||
params: p4,
|
name: "negative reward duration",
|
||||||
expectPass: false,
|
params: types.Params{
|
||||||
|
Active: true,
|
||||||
|
Rewards: types.Rewards{
|
||||||
|
types.Reward{
|
||||||
|
Active: true,
|
||||||
|
Denom: "bnb",
|
||||||
|
AvailableRewards: sdk.NewCoin("ukava", sdk.NewInt(10000000000)),
|
||||||
|
Duration: time.Hour * -24 * 7,
|
||||||
|
TimeLock: time.Hour * 8766,
|
||||||
|
ClaimDuration: time.Hour * 24 * 14,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
errResult: errResult{
|
||||||
|
expectPass: false,
|
||||||
|
contains: "reward duration must be positive",
|
||||||
|
},
|
||||||
},
|
},
|
||||||
paramTest{
|
paramTest{
|
||||||
params: p5,
|
name: "negative time lock",
|
||||||
expectPass: false,
|
params: types.Params{
|
||||||
|
Active: true,
|
||||||
|
Rewards: types.Rewards{
|
||||||
|
types.Reward{
|
||||||
|
Active: true,
|
||||||
|
Denom: "bnb",
|
||||||
|
AvailableRewards: sdk.NewCoin("ukava", sdk.NewInt(10000000000)),
|
||||||
|
Duration: time.Hour * 24 * 7,
|
||||||
|
TimeLock: time.Hour * -8766,
|
||||||
|
ClaimDuration: time.Hour * 24 * 14,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
errResult: errResult{
|
||||||
|
expectPass: false,
|
||||||
|
contains: "reward timelock must be non-negative",
|
||||||
|
},
|
||||||
},
|
},
|
||||||
paramTest{
|
paramTest{
|
||||||
params: p6,
|
name: "zero claim duration",
|
||||||
expectPass: false,
|
params: types.Params{
|
||||||
|
Active: true,
|
||||||
|
Rewards: types.Rewards{
|
||||||
|
types.Reward{
|
||||||
|
Active: true,
|
||||||
|
Denom: "bnb",
|
||||||
|
AvailableRewards: sdk.NewCoin("ukava", sdk.NewInt(10000000000)),
|
||||||
|
Duration: time.Hour * 24 * 7,
|
||||||
|
TimeLock: time.Hour * 8766,
|
||||||
|
ClaimDuration: time.Hour * 0,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
errResult: errResult{
|
||||||
|
expectPass: false,
|
||||||
|
contains: "claim duration must be positive",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
paramTest{
|
||||||
|
name: "zero reward",
|
||||||
|
params: types.Params{
|
||||||
|
Active: true,
|
||||||
|
Rewards: types.Rewards{
|
||||||
|
types.Reward{
|
||||||
|
Active: true,
|
||||||
|
Denom: "bnb",
|
||||||
|
AvailableRewards: sdk.NewCoin("ukava", sdk.NewInt(0)),
|
||||||
|
Duration: time.Hour * 24 * 7,
|
||||||
|
TimeLock: time.Hour * 8766,
|
||||||
|
ClaimDuration: time.Hour * 24 * 14,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
errResult: errResult{
|
||||||
|
expectPass: false,
|
||||||
|
contains: "reward amount must be positive",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
paramTest{
|
||||||
|
name: "empty reward denom",
|
||||||
|
params: types.Params{
|
||||||
|
Active: true,
|
||||||
|
Rewards: types.Rewards{
|
||||||
|
types.Reward{
|
||||||
|
Active: true,
|
||||||
|
Denom: "",
|
||||||
|
AvailableRewards: sdk.NewCoin("ukava", sdk.NewInt(0)),
|
||||||
|
Duration: time.Hour * 24 * 7,
|
||||||
|
TimeLock: time.Hour * 8766,
|
||||||
|
ClaimDuration: time.Hour * 24 * 14,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
errResult: errResult{
|
||||||
|
expectPass: false,
|
||||||
|
contains: "cannot have empty reward denom",
|
||||||
|
},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (suite *ParamTestSuite) TestParamValidation() {
|
func (suite *ParamTestSuite) TestParamValidation() {
|
||||||
for _, t := range suite.tests {
|
for _, t := range suite.tests {
|
||||||
err := t.params.Validate()
|
suite.Run(t.name, func() {
|
||||||
if t.expectPass {
|
err := t.params.Validate()
|
||||||
suite.NoError(err)
|
if t.errResult.expectPass {
|
||||||
} else {
|
suite.Require().NoError(err)
|
||||||
suite.Error(err)
|
} else {
|
||||||
}
|
suite.Require().Error(err)
|
||||||
|
suite.Require().True(strings.Contains(err.Error(), t.errResult.contains))
|
||||||
|
}
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user