mirror of
https://github.com/0glabs/0g-chain.git
synced 2025-01-18 19:15:19 +00:00
c63ecf908a
* Add 'InterestFactor' to CDP type (#734) * update cdp type to include interest factor * fix build * Add cdp accumulator methods (#735) * remame fees to interest * add accumulate interest method * add basic test * add note * address review comments * update tests * Add sync cdp interest method (#737) * remame fees to interest * add accumulate interest method * add basic test * add note * address review comments * update tests * remove old fee functions * add method to synchronize cdp interest * add multi-cdp tests * add test with many blocks * add test for interest getter * address review comments * calculate time difference then convert to seconds * fix: update collateral index when syncing interest * fix: differentiate between case when apy is zero and all fees are being rounded to zero * fix: round time difference properly * update cdp genesis state and migrations (#738) * remame fees to interest * add accumulate interest method * add basic test * add note * address review comments * update tests * remove old fee functions * add method to synchronize cdp interest * add multi-cdp tests * add test with many blocks * add test for interest getter * update cdp genesis state and migrations * address review comments * calculate time difference then convert to seconds * fix: update collateral index when syncing interest * fix: differentiate between case when apy is zero and all fees are being rounded to zero * fix: simplify add/remove/update collateral index * update genesis state to include total principal amounts * update migration * Delete kava-4-cdp-state-block-500000.json * Add cdp liquidations by external keeper (#750) * feat: split liquidations between external keepers and automated begin blocker * address review comments * USDX incentive accumulators (#752) * feat: split liquidations between external keepers and automated begin blocker * wip: refactor usdx minting incentives to use accumulators/hooks * wip: refactor usdx minting claim object * feat: use accumulators/hooks for usdx minting rewards * fix: get tests passing * fix: don't create claim objects unless that cdp type is eligable for rewards * add begin blocker * update client * cleanup comments/tests * update querier * address review comments * fix: check for division by zero * address review comments * run hook before interest is synced * Remove savings rate (#764) * remove savings rate * remove savings rate from debt param * update migrations * address review comments * Add usdx incentives calculation test (#765) * add usdx incentive calculation test * update reward calculation * add allowable error to test criteria * Update x/incentive/keeper/rewards_test.go Co-authored-by: Kevin Davis <karzak@users.noreply.github.com> * fix: remove old fields from test genesis state Co-authored-by: Ruaridh <rhuairahrighairidh@users.noreply.github.com> Co-authored-by: Ruaridh <rhuairahrighairidh@users.noreply.github.com>
764 lines
26 KiB
Go
764 lines
26 KiB
Go
package keeper_test
|
|
|
|
import (
|
|
"testing"
|
|
"time"
|
|
|
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
|
|
|
"github.com/stretchr/testify/suite"
|
|
|
|
abci "github.com/tendermint/tendermint/abci/types"
|
|
tmtime "github.com/tendermint/tendermint/types/time"
|
|
|
|
"github.com/kava-labs/kava/app"
|
|
"github.com/kava-labs/kava/x/cdp/keeper"
|
|
"github.com/kava-labs/kava/x/cdp/types"
|
|
)
|
|
|
|
type InterestTestSuite struct {
|
|
suite.Suite
|
|
|
|
keeper keeper.Keeper
|
|
app app.TestApp
|
|
ctx sdk.Context
|
|
}
|
|
|
|
func (suite *InterestTestSuite) SetupTest() {
|
|
tApp := app.NewTestApp()
|
|
ctx := tApp.NewContext(true, abci.Header{Height: 1, Time: tmtime.Now()})
|
|
tApp.InitializeFromGenesisStates(
|
|
NewPricefeedGenStateMulti(),
|
|
NewCDPGenStateMulti(),
|
|
)
|
|
keeper := tApp.GetCDPKeeper()
|
|
suite.app = tApp
|
|
suite.ctx = ctx
|
|
suite.keeper = keeper
|
|
}
|
|
|
|
// createCdps is a helper function to create two CDPs each with zero fees
|
|
func (suite *InterestTestSuite) createCdps() {
|
|
// create 2 accounts in the state and give them some coins
|
|
// create two private key pair addresses
|
|
_, addrs := app.GeneratePrivKeyAddressPairs(2)
|
|
ak := suite.app.GetAccountKeeper()
|
|
// setup the first account
|
|
acc := ak.NewAccountWithAddress(suite.ctx, addrs[0])
|
|
acc.SetCoins(cs(c("xrp", 200000000), c("btc", 500000000)))
|
|
|
|
ak.SetAccount(suite.ctx, acc)
|
|
// now setup the second account
|
|
acc2 := ak.NewAccountWithAddress(suite.ctx, addrs[1])
|
|
acc2.SetCoins(cs(c("xrp", 200000000), c("btc", 500000000)))
|
|
ak.SetAccount(suite.ctx, acc2)
|
|
|
|
// now create two cdps with the addresses we just created
|
|
// use the created account to create a cdp that SHOULD have fees updated
|
|
// to get a ratio between 100 - 110% of liquidation ratio we can use 200xrp ($50) and 24 usdx (208% collateralization with liquidation ratio of 200%)
|
|
// create CDP for the first address
|
|
err := suite.keeper.AddCdp(suite.ctx, addrs[0], c("xrp", 200000000), c("usdx", 24000000), "xrp-a")
|
|
suite.NoError(err) // check that no error was thrown
|
|
|
|
// use the other account to create a cdp that SHOULD NOT have fees updated - 500% collateralization
|
|
// create CDP for the second address
|
|
err = suite.keeper.AddCdp(suite.ctx, addrs[1], c("xrp", 200000000), c("usdx", 10000000), "xrp-a")
|
|
suite.NoError(err) // check that no error was thrown
|
|
|
|
}
|
|
|
|
func (suite *InterestTestSuite) TestCalculateInterestFactor() {
|
|
type args struct {
|
|
perSecondInterestRate sdk.Dec
|
|
timeElapsed sdk.Int
|
|
expectedValue sdk.Dec
|
|
}
|
|
|
|
type test struct {
|
|
name string
|
|
args args
|
|
}
|
|
|
|
oneYearInSeconds := int64(31536000)
|
|
|
|
testCases := []test{
|
|
{
|
|
"1 year",
|
|
args{
|
|
perSecondInterestRate: sdk.MustNewDecFromStr("1.000000005555"),
|
|
timeElapsed: sdk.NewInt(oneYearInSeconds),
|
|
expectedValue: sdk.MustNewDecFromStr("1.191463614477847370"),
|
|
},
|
|
},
|
|
{
|
|
"10 year",
|
|
args{
|
|
perSecondInterestRate: sdk.MustNewDecFromStr("1.000000005555"),
|
|
timeElapsed: sdk.NewInt(oneYearInSeconds * 10),
|
|
expectedValue: sdk.MustNewDecFromStr("5.765113233897391189"),
|
|
},
|
|
},
|
|
{
|
|
"1 month",
|
|
args{
|
|
perSecondInterestRate: sdk.MustNewDecFromStr("1.000000005555"),
|
|
timeElapsed: sdk.NewInt(oneYearInSeconds / 12),
|
|
expectedValue: sdk.MustNewDecFromStr("1.014705619075717373"),
|
|
},
|
|
},
|
|
{
|
|
"1 day",
|
|
args{
|
|
perSecondInterestRate: sdk.MustNewDecFromStr("1.000000005555"),
|
|
timeElapsed: sdk.NewInt(oneYearInSeconds / 365),
|
|
expectedValue: sdk.MustNewDecFromStr("1.000480067194057924"),
|
|
},
|
|
},
|
|
{
|
|
"1 year: low interest rate",
|
|
args{
|
|
perSecondInterestRate: sdk.MustNewDecFromStr("1.000000000555"),
|
|
timeElapsed: sdk.NewInt(oneYearInSeconds),
|
|
expectedValue: sdk.MustNewDecFromStr("1.017656545925063632"),
|
|
},
|
|
},
|
|
{
|
|
"1 year, lower interest rate",
|
|
args{
|
|
perSecondInterestRate: sdk.MustNewDecFromStr("1.000000000055"),
|
|
timeElapsed: sdk.NewInt(oneYearInSeconds),
|
|
expectedValue: sdk.MustNewDecFromStr("1.001735985079841390"),
|
|
},
|
|
},
|
|
{
|
|
"1 year, lowest interest rate",
|
|
args{
|
|
perSecondInterestRate: sdk.MustNewDecFromStr("1.000000000005"),
|
|
timeElapsed: sdk.NewInt(oneYearInSeconds),
|
|
expectedValue: sdk.MustNewDecFromStr("1.000157692432076670"),
|
|
},
|
|
},
|
|
{
|
|
"1 year: high interest rate",
|
|
args{
|
|
perSecondInterestRate: sdk.MustNewDecFromStr("1.000000055555"),
|
|
timeElapsed: sdk.NewInt(oneYearInSeconds),
|
|
expectedValue: sdk.MustNewDecFromStr("5.766022095987868825"),
|
|
},
|
|
},
|
|
{
|
|
"1 year: higher interest rate",
|
|
args{
|
|
perSecondInterestRate: sdk.MustNewDecFromStr("1.000000555555"),
|
|
timeElapsed: sdk.NewInt(oneYearInSeconds),
|
|
expectedValue: sdk.MustNewDecFromStr("40628388.864535408465693310"),
|
|
},
|
|
},
|
|
// If we raise the per second interest rate too much we'll cause an integer overflow.
|
|
// For example, perSecondInterestRate: '1.000005555555' will cause a panic.
|
|
{
|
|
"1 year: highest interest rate",
|
|
args{
|
|
perSecondInterestRate: sdk.MustNewDecFromStr("1.000001555555"),
|
|
timeElapsed: sdk.NewInt(oneYearInSeconds),
|
|
expectedValue: sdk.MustNewDecFromStr("2017093013158200407564.613502861572552603"),
|
|
},
|
|
},
|
|
}
|
|
|
|
for _, tc := range testCases {
|
|
suite.Run(tc.name, func() {
|
|
interestFactor := keeper.CalculateInterestFactor(tc.args.perSecondInterestRate, tc.args.timeElapsed)
|
|
suite.Require().Equal(tc.args.expectedValue, interestFactor)
|
|
})
|
|
}
|
|
}
|
|
|
|
func (suite *InterestTestSuite) TestAccumulateInterest() {
|
|
|
|
type args struct {
|
|
ctype string
|
|
initialTime time.Time
|
|
totalPrincipal sdk.Int
|
|
timeElapsed int
|
|
expectedTotalPrincipal sdk.Int
|
|
expectedLastAccrualTime time.Time
|
|
}
|
|
|
|
type test struct {
|
|
name string
|
|
args args
|
|
}
|
|
oneYearInSeconds := 31536000
|
|
|
|
testCases := []test{
|
|
{
|
|
"1 year",
|
|
args{
|
|
ctype: "bnb-a",
|
|
initialTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC),
|
|
totalPrincipal: sdk.NewInt(100000000000000),
|
|
timeElapsed: oneYearInSeconds,
|
|
expectedTotalPrincipal: sdk.NewInt(105000000000012),
|
|
expectedLastAccrualTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC).Add(time.Duration(int(time.Second) * oneYearInSeconds)),
|
|
},
|
|
},
|
|
{
|
|
"1 year - zero principal",
|
|
args{
|
|
ctype: "bnb-a",
|
|
initialTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC),
|
|
totalPrincipal: sdk.ZeroInt(),
|
|
timeElapsed: oneYearInSeconds,
|
|
expectedTotalPrincipal: sdk.ZeroInt(),
|
|
expectedLastAccrualTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC).Add(time.Duration(int(time.Second) * oneYearInSeconds)),
|
|
},
|
|
},
|
|
{
|
|
"1 month",
|
|
args{
|
|
ctype: "bnb-a",
|
|
initialTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC),
|
|
totalPrincipal: sdk.NewInt(100000000000000),
|
|
timeElapsed: 86400 * 30,
|
|
expectedTotalPrincipal: sdk.NewInt(100401820189198),
|
|
expectedLastAccrualTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC).Add(time.Duration(int(time.Second) * 86400 * 30)),
|
|
},
|
|
},
|
|
{
|
|
"1 month - interest rounds to zero",
|
|
args{
|
|
ctype: "bnb-a",
|
|
initialTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC),
|
|
totalPrincipal: sdk.NewInt(10),
|
|
timeElapsed: 86400 * 30,
|
|
expectedTotalPrincipal: sdk.NewInt(10),
|
|
expectedLastAccrualTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC),
|
|
},
|
|
},
|
|
{
|
|
"7 seconds",
|
|
args{
|
|
ctype: "bnb-a",
|
|
initialTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC),
|
|
totalPrincipal: sdk.NewInt(100000000000000),
|
|
timeElapsed: 7,
|
|
expectedTotalPrincipal: sdk.NewInt(100000001082988),
|
|
expectedLastAccrualTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC).Add(time.Duration(int(time.Second) * 7)),
|
|
},
|
|
},
|
|
{
|
|
"7 seconds - interest rounds to zero",
|
|
args{
|
|
ctype: "bnb-a",
|
|
initialTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC),
|
|
totalPrincipal: sdk.NewInt(30000000),
|
|
timeElapsed: 7,
|
|
expectedTotalPrincipal: sdk.NewInt(30000000),
|
|
expectedLastAccrualTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC),
|
|
},
|
|
},
|
|
{
|
|
"7 seconds - zero interest",
|
|
args{
|
|
ctype: "busd-a",
|
|
initialTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC),
|
|
totalPrincipal: sdk.NewInt(100000000000000),
|
|
timeElapsed: 7,
|
|
expectedTotalPrincipal: sdk.NewInt(100000000000000),
|
|
expectedLastAccrualTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC).Add(time.Duration(int(time.Second) * 7)),
|
|
},
|
|
},
|
|
}
|
|
for _, tc := range testCases {
|
|
suite.Run(tc.name, func() {
|
|
suite.ctx = suite.ctx.WithBlockTime(tc.args.initialTime)
|
|
suite.keeper.SetTotalPrincipal(suite.ctx, tc.args.ctype, types.DefaultStableDenom, tc.args.totalPrincipal)
|
|
suite.keeper.SetPreviousAccrualTime(suite.ctx, tc.args.ctype, suite.ctx.BlockTime())
|
|
suite.keeper.SetInterestFactor(suite.ctx, tc.args.ctype, sdk.OneDec())
|
|
|
|
updatedBlockTime := suite.ctx.BlockTime().Add(time.Duration(int(time.Second) * tc.args.timeElapsed))
|
|
suite.ctx = suite.ctx.WithBlockTime(updatedBlockTime)
|
|
err := suite.keeper.AccumulateInterest(suite.ctx, tc.args.ctype)
|
|
suite.Require().NoError(err)
|
|
|
|
actualTotalPrincipal := suite.keeper.GetTotalPrincipal(suite.ctx, tc.args.ctype, types.DefaultStableDenom)
|
|
suite.Require().Equal(tc.args.expectedTotalPrincipal, actualTotalPrincipal)
|
|
actualAccrualTime, _ := suite.keeper.GetPreviousAccrualTime(suite.ctx, tc.args.ctype)
|
|
suite.Require().Equal(tc.args.expectedLastAccrualTime, actualAccrualTime)
|
|
})
|
|
}
|
|
}
|
|
|
|
// TestSynchronizeInterest tests the functionality of synchronizing the accumulated interest for CDPs
|
|
func (suite *InterestTestSuite) TestSynchronizeInterest() {
|
|
type args struct {
|
|
ctype string
|
|
initialTime time.Time
|
|
initialCollateral sdk.Coin
|
|
initialPrincipal sdk.Coin
|
|
timeElapsed int
|
|
expectedFees sdk.Coin
|
|
expectedFeesUpdatedTime time.Time
|
|
}
|
|
|
|
type test struct {
|
|
name string
|
|
args args
|
|
}
|
|
|
|
oneYearInSeconds := 31536000
|
|
testCases := []test{
|
|
{
|
|
"1 year",
|
|
args{
|
|
ctype: "bnb-a",
|
|
initialTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC),
|
|
initialCollateral: c("bnb", 1000000000000),
|
|
initialPrincipal: c("usdx", 100000000000),
|
|
timeElapsed: oneYearInSeconds,
|
|
expectedFees: c("usdx", 5000000000),
|
|
expectedFeesUpdatedTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC).Add(time.Duration(int(time.Second) * oneYearInSeconds)),
|
|
},
|
|
},
|
|
{
|
|
"1 month",
|
|
args{
|
|
ctype: "bnb-a",
|
|
initialTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC),
|
|
initialCollateral: c("bnb", 1000000000000),
|
|
initialPrincipal: c("usdx", 100000000000),
|
|
timeElapsed: 86400 * 30,
|
|
expectedFees: c("usdx", 401820189),
|
|
expectedFeesUpdatedTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC).Add(time.Duration(int(time.Second) * 86400 * 30)),
|
|
},
|
|
},
|
|
{
|
|
"7 seconds",
|
|
args{
|
|
ctype: "bnb-a",
|
|
initialTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC),
|
|
initialCollateral: c("bnb", 1000000000000),
|
|
initialPrincipal: c("usdx", 100000000000),
|
|
timeElapsed: 7,
|
|
expectedFees: c("usdx", 1083),
|
|
expectedFeesUpdatedTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC).Add(time.Duration(int(time.Second) * 7)),
|
|
},
|
|
},
|
|
{
|
|
"7 seconds - zero apy",
|
|
args{
|
|
ctype: "busd-a",
|
|
initialTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC),
|
|
initialCollateral: c("busd", 10000000000000),
|
|
initialPrincipal: c("usdx", 10000000000),
|
|
timeElapsed: 7,
|
|
expectedFees: c("usdx", 0),
|
|
expectedFeesUpdatedTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC).Add(time.Duration(int(time.Second) * 7)),
|
|
},
|
|
},
|
|
{
|
|
"7 seconds - fees round to zero",
|
|
args{
|
|
ctype: "bnb-a",
|
|
initialTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC),
|
|
initialCollateral: c("bnb", 1000000000),
|
|
initialPrincipal: c("usdx", 10000000),
|
|
timeElapsed: 7,
|
|
expectedFees: c("usdx", 0),
|
|
expectedFeesUpdatedTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC),
|
|
},
|
|
},
|
|
}
|
|
|
|
for _, tc := range testCases {
|
|
suite.Run(tc.name, func() {
|
|
suite.SetupTest()
|
|
suite.ctx = suite.ctx.WithBlockTime(tc.args.initialTime)
|
|
|
|
// setup account state
|
|
_, addrs := app.GeneratePrivKeyAddressPairs(1)
|
|
ak := suite.app.GetAccountKeeper()
|
|
// setup the first account
|
|
acc := ak.NewAccountWithAddress(suite.ctx, addrs[0])
|
|
ak.SetAccount(suite.ctx, acc)
|
|
sk := suite.app.GetSupplyKeeper()
|
|
err := sk.MintCoins(suite.ctx, types.ModuleName, cs(tc.args.initialCollateral))
|
|
suite.Require().NoError(err)
|
|
err = sk.SendCoinsFromModuleToAccount(suite.ctx, types.ModuleName, addrs[0], cs(tc.args.initialCollateral))
|
|
suite.Require().NoError(err)
|
|
|
|
// setup pricefeed
|
|
pk := suite.app.GetPriceFeedKeeper()
|
|
pk.SetPrice(suite.ctx, sdk.AccAddress{}, "bnb:usd", d("17.25"), tc.args.expectedFeesUpdatedTime.Add(time.Second))
|
|
pk.SetPrice(suite.ctx, sdk.AccAddress{}, "busd:usd", d("1"), tc.args.expectedFeesUpdatedTime.Add(time.Second))
|
|
|
|
// setup cdp state
|
|
suite.keeper.SetPreviousAccrualTime(suite.ctx, tc.args.ctype, suite.ctx.BlockTime())
|
|
suite.keeper.SetInterestFactor(suite.ctx, tc.args.ctype, sdk.OneDec())
|
|
err = suite.keeper.AddCdp(suite.ctx, addrs[0], tc.args.initialCollateral, tc.args.initialPrincipal, tc.args.ctype)
|
|
suite.Require().NoError(err)
|
|
|
|
updatedBlockTime := suite.ctx.BlockTime().Add(time.Duration(int(time.Second) * tc.args.timeElapsed))
|
|
suite.ctx = suite.ctx.WithBlockTime(updatedBlockTime)
|
|
err = suite.keeper.AccumulateInterest(suite.ctx, tc.args.ctype)
|
|
suite.Require().NoError(err)
|
|
|
|
cdp, found := suite.keeper.GetCDP(suite.ctx, tc.args.ctype, 1)
|
|
suite.Require().True(found)
|
|
|
|
cdp = suite.keeper.SynchronizeInterest(suite.ctx, cdp)
|
|
|
|
suite.Require().Equal(tc.args.expectedFees, cdp.AccumulatedFees)
|
|
suite.Require().Equal(tc.args.expectedFeesUpdatedTime, cdp.FeesUpdated)
|
|
|
|
})
|
|
}
|
|
}
|
|
|
|
func (suite *InterestTestSuite) TestMultipleCDPInterest() {
|
|
type args struct {
|
|
ctype string
|
|
initialTime time.Time
|
|
blockInterval int
|
|
numberOfBlocks int
|
|
initialCDPCollateral sdk.Coin
|
|
initialCDPPrincipal sdk.Coin
|
|
numberOfCdps int
|
|
expectedFeesPerCDP sdk.Coin
|
|
expectedTotalPrincipalPerCDP sdk.Coin
|
|
expectedFeesUpdatedTime time.Time
|
|
expectedTotalPrincipal sdk.Int
|
|
expectedDebtBalance sdk.Int
|
|
expectedStableBalance sdk.Int
|
|
expectedSumOfCDPPrincipal sdk.Int
|
|
}
|
|
|
|
type test struct {
|
|
name string
|
|
args args
|
|
}
|
|
|
|
testCases := []test{
|
|
{
|
|
"1 block",
|
|
args{
|
|
ctype: "bnb-a",
|
|
initialTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC),
|
|
blockInterval: 7,
|
|
numberOfBlocks: 1,
|
|
initialCDPCollateral: c("bnb", 10000000000),
|
|
initialCDPPrincipal: c("usdx", 500000000),
|
|
numberOfCdps: 100,
|
|
expectedFeesPerCDP: c("usdx", 5),
|
|
expectedTotalPrincipalPerCDP: c("usdx", 500000005),
|
|
expectedFeesUpdatedTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC).Add(time.Duration(int(time.Second) * 7)),
|
|
expectedTotalPrincipal: i(50000000541),
|
|
expectedDebtBalance: i(50000000541),
|
|
expectedStableBalance: i(50000000541),
|
|
expectedSumOfCDPPrincipal: i(50000000500),
|
|
},
|
|
},
|
|
{
|
|
"100 blocks",
|
|
args{
|
|
ctype: "bnb-a",
|
|
initialTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC),
|
|
blockInterval: 7,
|
|
numberOfBlocks: 100,
|
|
initialCDPCollateral: c("bnb", 10000000000),
|
|
initialCDPPrincipal: c("usdx", 500000000),
|
|
numberOfCdps: 100,
|
|
expectedFeesPerCDP: c("usdx", 541),
|
|
expectedTotalPrincipalPerCDP: c("usdx", 500000541),
|
|
expectedFeesUpdatedTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC).Add(time.Duration(int(time.Second) * 7 * 100)),
|
|
expectedTotalPrincipal: i(50000054100),
|
|
expectedDebtBalance: i(50000054100),
|
|
expectedStableBalance: i(50000054100),
|
|
expectedSumOfCDPPrincipal: i(50000054100),
|
|
},
|
|
},
|
|
{
|
|
"10000 blocks",
|
|
args{
|
|
ctype: "bnb-a",
|
|
initialTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC),
|
|
blockInterval: 7,
|
|
numberOfBlocks: 10000,
|
|
initialCDPCollateral: c("bnb", 10000000000),
|
|
initialCDPPrincipal: c("usdx", 500000000),
|
|
numberOfCdps: 100,
|
|
expectedFeesPerCDP: c("usdx", 54152),
|
|
expectedTotalPrincipalPerCDP: c("usdx", 500054152),
|
|
expectedFeesUpdatedTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC).Add(time.Duration(int(time.Second) * 7 * 10000)),
|
|
expectedTotalPrincipal: i(50005418990),
|
|
expectedDebtBalance: i(50005418990),
|
|
expectedStableBalance: i(50005418990),
|
|
expectedSumOfCDPPrincipal: i(50005415200),
|
|
},
|
|
},
|
|
}
|
|
for _, tc := range testCases {
|
|
suite.Run(tc.name, func() {
|
|
suite.SetupTest()
|
|
suite.ctx = suite.ctx.WithBlockTime(tc.args.initialTime)
|
|
|
|
// setup pricefeed
|
|
pk := suite.app.GetPriceFeedKeeper()
|
|
pk.SetPrice(suite.ctx, sdk.AccAddress{}, "bnb:usd", d("17.25"), tc.args.expectedFeesUpdatedTime.Add(time.Second))
|
|
|
|
// setup cdp state
|
|
suite.keeper.SetPreviousAccrualTime(suite.ctx, tc.args.ctype, suite.ctx.BlockTime())
|
|
suite.keeper.SetInterestFactor(suite.ctx, tc.args.ctype, sdk.OneDec())
|
|
|
|
// setup account state
|
|
_, addrs := app.GeneratePrivKeyAddressPairs(tc.args.numberOfCdps)
|
|
for j := 0; j < tc.args.numberOfCdps; j++ {
|
|
ak := suite.app.GetAccountKeeper()
|
|
// setup the first account
|
|
acc := ak.NewAccountWithAddress(suite.ctx, addrs[j])
|
|
ak.SetAccount(suite.ctx, acc)
|
|
sk := suite.app.GetSupplyKeeper()
|
|
err := sk.MintCoins(suite.ctx, types.ModuleName, cs(tc.args.initialCDPCollateral))
|
|
suite.Require().NoError(err)
|
|
err = sk.SendCoinsFromModuleToAccount(suite.ctx, types.ModuleName, addrs[j], cs(tc.args.initialCDPCollateral))
|
|
suite.Require().NoError(err)
|
|
err = suite.keeper.AddCdp(suite.ctx, addrs[j], tc.args.initialCDPCollateral, tc.args.initialCDPPrincipal, tc.args.ctype)
|
|
suite.Require().NoError(err)
|
|
}
|
|
|
|
// run a number of blocks where CDPs are not synchronized
|
|
for j := 0; j < tc.args.numberOfBlocks; j++ {
|
|
updatedBlockTime := suite.ctx.BlockTime().Add(time.Duration(int(time.Second) * tc.args.blockInterval))
|
|
suite.ctx = suite.ctx.WithBlockTime(updatedBlockTime)
|
|
err := suite.keeper.AccumulateInterest(suite.ctx, tc.args.ctype)
|
|
suite.Require().NoError(err)
|
|
}
|
|
|
|
sk := suite.app.GetSupplyKeeper()
|
|
supplyTotal := sk.GetSupply(suite.ctx).GetTotal()
|
|
debtSupply := supplyTotal.AmountOf(types.DefaultDebtDenom)
|
|
usdxSupply := supplyTotal.AmountOf(types.DefaultStableDenom)
|
|
totalPrincipal := suite.keeper.GetTotalPrincipal(suite.ctx, tc.args.ctype, types.DefaultStableDenom)
|
|
|
|
suite.Require().Equal(tc.args.expectedDebtBalance, debtSupply)
|
|
suite.Require().Equal(tc.args.expectedStableBalance, usdxSupply)
|
|
suite.Require().Equal(tc.args.expectedTotalPrincipal, totalPrincipal)
|
|
|
|
sumOfCDPPrincipal := sdk.ZeroInt()
|
|
|
|
for j := 0; j < tc.args.numberOfCdps; j++ {
|
|
cdp, found := suite.keeper.GetCDP(suite.ctx, tc.args.ctype, uint64(j+1))
|
|
suite.Require().True(found)
|
|
cdp = suite.keeper.SynchronizeInterest(suite.ctx, cdp)
|
|
suite.Require().Equal(tc.args.expectedFeesPerCDP, cdp.AccumulatedFees)
|
|
suite.Require().Equal(tc.args.expectedTotalPrincipalPerCDP, cdp.GetTotalPrincipal())
|
|
suite.Require().Equal(tc.args.expectedFeesUpdatedTime, cdp.FeesUpdated)
|
|
sumOfCDPPrincipal = sumOfCDPPrincipal.Add(cdp.GetTotalPrincipal().Amount)
|
|
}
|
|
|
|
suite.Require().Equal(tc.args.expectedSumOfCDPPrincipal, sumOfCDPPrincipal)
|
|
|
|
})
|
|
}
|
|
}
|
|
|
|
// TestSynchronizeInterest tests the functionality of synchronizing the accumulated interest for CDPs
|
|
func (suite *InterestTestSuite) TestCalculateCDPInterest() {
|
|
type args struct {
|
|
ctype string
|
|
initialTime time.Time
|
|
initialCollateral sdk.Coin
|
|
initialPrincipal sdk.Coin
|
|
timeElapsed int
|
|
expectedFees sdk.Coin
|
|
}
|
|
|
|
type test struct {
|
|
name string
|
|
args args
|
|
}
|
|
|
|
oneYearInSeconds := 31536000
|
|
testCases := []test{
|
|
{
|
|
"1 year",
|
|
args{
|
|
ctype: "bnb-a",
|
|
initialTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC),
|
|
initialCollateral: c("bnb", 1000000000000),
|
|
initialPrincipal: c("usdx", 100000000000),
|
|
timeElapsed: oneYearInSeconds,
|
|
expectedFees: c("usdx", 5000000000),
|
|
},
|
|
},
|
|
{
|
|
"1 month",
|
|
args{
|
|
ctype: "bnb-a",
|
|
initialTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC),
|
|
initialCollateral: c("bnb", 1000000000000),
|
|
initialPrincipal: c("usdx", 100000000000),
|
|
timeElapsed: 86400 * 30,
|
|
expectedFees: c("usdx", 401820189),
|
|
},
|
|
},
|
|
{
|
|
"7 seconds",
|
|
args{
|
|
ctype: "bnb-a",
|
|
initialTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC),
|
|
initialCollateral: c("bnb", 1000000000000),
|
|
initialPrincipal: c("usdx", 100000000000),
|
|
timeElapsed: 7,
|
|
expectedFees: c("usdx", 1083),
|
|
},
|
|
},
|
|
{
|
|
"7 seconds - fees round to zero",
|
|
args{
|
|
ctype: "bnb-a",
|
|
initialTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC),
|
|
initialCollateral: c("bnb", 1000000000),
|
|
initialPrincipal: c("usdx", 10000000),
|
|
timeElapsed: 7,
|
|
expectedFees: c("usdx", 0),
|
|
},
|
|
},
|
|
}
|
|
|
|
for _, tc := range testCases {
|
|
suite.Run(tc.name, func() {
|
|
suite.SetupTest()
|
|
suite.ctx = suite.ctx.WithBlockTime(tc.args.initialTime)
|
|
|
|
// setup account state
|
|
_, addrs := app.GeneratePrivKeyAddressPairs(1)
|
|
ak := suite.app.GetAccountKeeper()
|
|
// setup the first account
|
|
acc := ak.NewAccountWithAddress(suite.ctx, addrs[0])
|
|
ak.SetAccount(suite.ctx, acc)
|
|
sk := suite.app.GetSupplyKeeper()
|
|
err := sk.MintCoins(suite.ctx, types.ModuleName, cs(tc.args.initialCollateral))
|
|
suite.Require().NoError(err)
|
|
err = sk.SendCoinsFromModuleToAccount(suite.ctx, types.ModuleName, addrs[0], cs(tc.args.initialCollateral))
|
|
suite.Require().NoError(err)
|
|
|
|
// setup pricefeed
|
|
pk := suite.app.GetPriceFeedKeeper()
|
|
pk.SetPrice(suite.ctx, sdk.AccAddress{}, "bnb:usd", d("17.25"), tc.args.initialTime.Add(time.Duration(int(time.Second)*tc.args.timeElapsed)))
|
|
|
|
// setup cdp state
|
|
suite.keeper.SetPreviousAccrualTime(suite.ctx, tc.args.ctype, suite.ctx.BlockTime())
|
|
suite.keeper.SetInterestFactor(suite.ctx, tc.args.ctype, sdk.OneDec())
|
|
err = suite.keeper.AddCdp(suite.ctx, addrs[0], tc.args.initialCollateral, tc.args.initialPrincipal, tc.args.ctype)
|
|
suite.Require().NoError(err)
|
|
|
|
updatedBlockTime := suite.ctx.BlockTime().Add(time.Duration(int(time.Second) * tc.args.timeElapsed))
|
|
suite.ctx = suite.ctx.WithBlockTime(updatedBlockTime)
|
|
err = suite.keeper.AccumulateInterest(suite.ctx, tc.args.ctype)
|
|
suite.Require().NoError(err)
|
|
|
|
cdp, found := suite.keeper.GetCDP(suite.ctx, tc.args.ctype, 1)
|
|
suite.Require().True(found)
|
|
|
|
newInterest := suite.keeper.CalculateNewInterest(suite.ctx, cdp)
|
|
|
|
suite.Require().Equal(tc.args.expectedFees, newInterest)
|
|
|
|
})
|
|
}
|
|
}
|
|
|
|
func (suite *InterestTestSuite) TestSyncInterestForRiskyCDPs() {
|
|
type args struct {
|
|
ctype string
|
|
numberCdps int
|
|
slice int
|
|
initialCollateral sdk.Coin
|
|
minPrincipal sdk.Coin
|
|
principalIncrement sdk.Coin
|
|
initialTime time.Time
|
|
timeElapsed int
|
|
expectedCDPs int
|
|
}
|
|
|
|
type test struct {
|
|
name string
|
|
args args
|
|
}
|
|
|
|
oneYearInSeconds := 31536000
|
|
testCases := []test{
|
|
|
|
{
|
|
"1 year",
|
|
args{
|
|
ctype: "bnb-a",
|
|
numberCdps: 20,
|
|
slice: 10,
|
|
initialCollateral: c("bnb", 100000000000),
|
|
minPrincipal: c("usdx", 100000000),
|
|
principalIncrement: c("usdx", 10000000),
|
|
initialTime: time.Date(2020, 12, 15, 14, 0, 0, 0, time.UTC),
|
|
timeElapsed: oneYearInSeconds,
|
|
expectedCDPs: 10,
|
|
},
|
|
},
|
|
}
|
|
|
|
for _, tc := range testCases {
|
|
suite.Run(tc.name, func() {
|
|
suite.SetupTest()
|
|
suite.ctx = suite.ctx.WithBlockTime(tc.args.initialTime)
|
|
// setup account state
|
|
_, addrs := app.GeneratePrivKeyAddressPairs(tc.args.numberCdps)
|
|
ak := suite.app.GetAccountKeeper()
|
|
sk := suite.app.GetSupplyKeeper()
|
|
for _, addr := range addrs {
|
|
acc := ak.NewAccountWithAddress(suite.ctx, addr)
|
|
ak.SetAccount(suite.ctx, acc)
|
|
err := sk.MintCoins(suite.ctx, types.ModuleName, cs(tc.args.initialCollateral))
|
|
suite.Require().NoError(err)
|
|
err = sk.SendCoinsFromModuleToAccount(suite.ctx, types.ModuleName, addr, cs(tc.args.initialCollateral))
|
|
suite.Require().NoError(err)
|
|
}
|
|
// setup pricefeed
|
|
pk := suite.app.GetPriceFeedKeeper()
|
|
pk.SetPrice(suite.ctx, sdk.AccAddress{}, "bnb:usd", d("20.0"), tc.args.initialTime.Add(time.Duration(int(time.Second)*tc.args.timeElapsed)))
|
|
|
|
// setup cdp state
|
|
suite.keeper.SetPreviousAccrualTime(suite.ctx, tc.args.ctype, suite.ctx.BlockTime())
|
|
suite.keeper.SetInterestFactor(suite.ctx, tc.args.ctype, sdk.OneDec())
|
|
for j, addr := range addrs {
|
|
initialPrincipal := tc.args.minPrincipal.Add(c("usdx", int64(j)*tc.args.principalIncrement.Amount.Int64()))
|
|
err := suite.keeper.AddCdp(suite.ctx, addr, tc.args.initialCollateral, initialPrincipal, tc.args.ctype)
|
|
suite.Require().NoError(err)
|
|
}
|
|
|
|
updatedBlockTime := suite.ctx.BlockTime().Add(time.Duration(int(time.Second) * tc.args.timeElapsed))
|
|
suite.ctx = suite.ctx.WithBlockTime(updatedBlockTime)
|
|
err := suite.keeper.AccumulateInterest(suite.ctx, tc.args.ctype)
|
|
suite.Require().NoError(err)
|
|
|
|
err = suite.keeper.SynchronizeInterestForRiskyCDPs(suite.ctx, i(int64(tc.args.slice)), sdk.MaxSortableDec, tc.args.ctype)
|
|
suite.Require().NoError(err)
|
|
|
|
cdpsUpdatedCount := 0
|
|
|
|
for _, addr := range addrs {
|
|
cdp, found := suite.keeper.GetCdpByOwnerAndCollateralType(suite.ctx, addr, tc.args.ctype)
|
|
suite.Require().True(found)
|
|
if cdp.FeesUpdated.Equal(suite.ctx.BlockTime()) {
|
|
cdpsUpdatedCount += 1
|
|
}
|
|
}
|
|
suite.Require().Equal(tc.args.expectedCDPs, cdpsUpdatedCount)
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestInterestTestSuite(t *testing.T) {
|
|
suite.Run(t, new(InterestTestSuite))
|
|
}
|