mirror of
https://github.com/0glabs/0g-chain.git
synced 2024-12-26 00:05:18 +00:00
Hard: fix liquidation engine (#771)
* initial * liquidation debugging * max lot == macc coin balance * add print statements * add test for pricefeed liquidation scenarios * skip zero lot * add insolvency liquidation test scenario * remove debugging statements * fix tests after rebase Co-authored-by: karzak <kjydavis3@gmail.com>
This commit is contained in:
parent
b5e02fde35
commit
58494fe357
@ -19,16 +19,18 @@ import (
|
|||||||
aucKeeper "github.com/kava-labs/kava/x/auction/keeper"
|
aucKeeper "github.com/kava-labs/kava/x/auction/keeper"
|
||||||
"github.com/kava-labs/kava/x/hard/keeper"
|
"github.com/kava-labs/kava/x/hard/keeper"
|
||||||
"github.com/kava-labs/kava/x/hard/types"
|
"github.com/kava-labs/kava/x/hard/types"
|
||||||
|
pfKeeper "github.com/kava-labs/kava/x/pricefeed/keeper"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Test suite used for all keeper tests
|
// Test suite used for all keeper tests
|
||||||
type KeeperTestSuite struct {
|
type KeeperTestSuite struct {
|
||||||
suite.Suite
|
suite.Suite
|
||||||
keeper keeper.Keeper
|
keeper keeper.Keeper
|
||||||
auctionKeeper aucKeeper.Keeper
|
auctionKeeper aucKeeper.Keeper
|
||||||
app app.TestApp
|
pricefeedKeeper pfKeeper.Keeper
|
||||||
ctx sdk.Context
|
app app.TestApp
|
||||||
addrs []sdk.AccAddress
|
ctx sdk.Context
|
||||||
|
addrs []sdk.AccAddress
|
||||||
}
|
}
|
||||||
|
|
||||||
// The default state used by each test
|
// The default state used by each test
|
||||||
|
@ -101,7 +101,7 @@ func (k Keeper) SeizeDeposits(ctx sdk.Context, keeper sdk.AccAddress, deposit ty
|
|||||||
amount := depCoin.Amount
|
amount := depCoin.Amount
|
||||||
mm, _ := k.GetMoneyMarket(ctx, denom)
|
mm, _ := k.GetMoneyMarket(ctx, denom)
|
||||||
|
|
||||||
// No rewards for anyone if liquidated by LTV index
|
// No keeper rewards if liquidated by LTV index
|
||||||
if !keeper.Equals(sdk.AccAddress(types.LiquidatorAccount)) {
|
if !keeper.Equals(sdk.AccAddress(types.LiquidatorAccount)) {
|
||||||
keeperReward := mm.KeeperRewardPercentage.MulInt(amount).TruncateInt()
|
keeperReward := mm.KeeperRewardPercentage.MulInt(amount).TruncateInt()
|
||||||
if keeperReward.GT(sdk.ZeroInt()) {
|
if keeperReward.GT(sdk.ZeroInt()) {
|
||||||
@ -158,6 +158,9 @@ func (k Keeper) StartAuctions(ctx sdk.Context, borrower sdk.AccAddress, borrows,
|
|||||||
weights := []sdk.Int{sdk.NewInt(100)}
|
weights := []sdk.Int{sdk.NewInt(100)}
|
||||||
debt := sdk.NewCoin("debt", sdk.ZeroInt())
|
debt := sdk.NewCoin("debt", sdk.ZeroInt())
|
||||||
|
|
||||||
|
macc := k.supplyKeeper.GetModuleAccount(ctx, types.ModuleAccountName)
|
||||||
|
maccCoins := macc.SpendableCoins(ctx.BlockTime())
|
||||||
|
|
||||||
for _, bKey := range bKeys {
|
for _, bKey := range bKeys {
|
||||||
bValue := borrowCoinValues.Get(bKey)
|
bValue := borrowCoinValues.Get(bKey)
|
||||||
maxLotSize := bValue.Quo(ltv)
|
maxLotSize := bValue.Quo(ltv)
|
||||||
@ -168,10 +171,21 @@ func (k Keeper) StartAuctions(ctx sdk.Context, borrower sdk.AccAddress, borrows,
|
|||||||
break // exit out of the loop if we have cleared the full amount
|
break // exit out of the loop if we have cleared the full amount
|
||||||
}
|
}
|
||||||
|
|
||||||
if dValue.GTE(maxLotSize) { // We can start an auction for the whole borrow amount
|
if dValue.GTE(maxLotSize) { // We can start an auction for the whole borrow amount]
|
||||||
bid := sdk.NewCoin(bKey, borrows.AmountOf(bKey))
|
bid := sdk.NewCoin(bKey, borrows.AmountOf(bKey))
|
||||||
|
|
||||||
lotSize := maxLotSize.MulInt(liqMap[dKey].conversionFactor).Quo(liqMap[dKey].price)
|
lotSize := maxLotSize.MulInt(liqMap[dKey].conversionFactor).Quo(liqMap[dKey].price)
|
||||||
|
if lotSize.TruncateInt().Equal(sdk.ZeroInt()) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
lot := sdk.NewCoin(dKey, lotSize.TruncateInt())
|
lot := sdk.NewCoin(dKey, lotSize.TruncateInt())
|
||||||
|
|
||||||
|
insufficientLotFunds := false
|
||||||
|
if lot.Amount.GT(maccCoins.AmountOf(dKey)) {
|
||||||
|
insufficientLotFunds = true
|
||||||
|
lot = sdk.NewCoin(lot.Denom, maccCoins.AmountOf(dKey))
|
||||||
|
}
|
||||||
|
|
||||||
// Sanity check that we can deliver coins to the liquidator account
|
// Sanity check that we can deliver coins to the liquidator account
|
||||||
if deposits.AmountOf(dKey).LT(lot.Amount) {
|
if deposits.AmountOf(dKey).LT(lot.Amount) {
|
||||||
return types.ErrInsufficientCoins
|
return types.ErrInsufficientCoins
|
||||||
@ -192,7 +206,11 @@ func (k Keeper) StartAuctions(ctx sdk.Context, borrower sdk.AccAddress, borrows,
|
|||||||
depositCoinValues.Decrement(dKey, maxLotSize)
|
depositCoinValues.Decrement(dKey, maxLotSize)
|
||||||
// Update deposits, borrows
|
// Update deposits, borrows
|
||||||
borrows = borrows.Sub(sdk.NewCoins(bid))
|
borrows = borrows.Sub(sdk.NewCoins(bid))
|
||||||
deposits = deposits.Sub(sdk.NewCoins(lot))
|
if insufficientLotFunds {
|
||||||
|
deposits = deposits.Sub(sdk.NewCoins(sdk.NewCoin(dKey, deposits.AmountOf(dKey))))
|
||||||
|
} else {
|
||||||
|
deposits = deposits.Sub(sdk.NewCoins(lot))
|
||||||
|
}
|
||||||
// Update max lot size
|
// Update max lot size
|
||||||
maxLotSize = sdk.ZeroDec()
|
maxLotSize = sdk.ZeroDec()
|
||||||
} else { // We can only start an auction for the partial borrow amount
|
} else { // We can only start an auction for the partial borrow amount
|
||||||
@ -205,6 +223,12 @@ func (k Keeper) StartAuctions(ctx sdk.Context, borrower sdk.AccAddress, borrows,
|
|||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
|
insufficientLotFunds := false
|
||||||
|
if lot.Amount.GT(maccCoins.AmountOf(dKey)) {
|
||||||
|
insufficientLotFunds = true
|
||||||
|
lot = sdk.NewCoin(lot.Denom, maccCoins.AmountOf(dKey))
|
||||||
|
}
|
||||||
|
|
||||||
// Sanity check that we can deliver coins to the liquidator account
|
// Sanity check that we can deliver coins to the liquidator account
|
||||||
if deposits.AmountOf(dKey).LT(lot.Amount) {
|
if deposits.AmountOf(dKey).LT(lot.Amount) {
|
||||||
return types.ErrInsufficientCoins
|
return types.ErrInsufficientCoins
|
||||||
@ -223,9 +247,14 @@ func (k Keeper) StartAuctions(ctx sdk.Context, borrower sdk.AccAddress, borrows,
|
|||||||
// Update variables to account for partial auction
|
// Update variables to account for partial auction
|
||||||
borrowCoinValues.Decrement(bKey, maxBid)
|
borrowCoinValues.Decrement(bKey, maxBid)
|
||||||
depositCoinValues.SetZero(dKey)
|
depositCoinValues.SetZero(dKey)
|
||||||
// Update deposits, borrows
|
|
||||||
borrows = borrows.Sub(sdk.NewCoins(bid))
|
borrows = borrows.Sub(sdk.NewCoins(bid))
|
||||||
deposits = deposits.Sub(sdk.NewCoins(lot))
|
if insufficientLotFunds {
|
||||||
|
deposits = deposits.Sub(sdk.NewCoins(sdk.NewCoin(dKey, deposits.AmountOf(dKey))))
|
||||||
|
} else {
|
||||||
|
deposits = deposits.Sub(sdk.NewCoins(lot))
|
||||||
|
}
|
||||||
|
|
||||||
// Update max lot size
|
// Update max lot size
|
||||||
maxLotSize = borrowCoinValues.Get(bKey).Quo(ltv)
|
maxLotSize = borrowCoinValues.Get(bKey).Quo(ltv)
|
||||||
}
|
}
|
||||||
|
@ -21,7 +21,7 @@ func (suite *KeeperTestSuite) TestIndexLiquidation() {
|
|||||||
borrower sdk.AccAddress
|
borrower sdk.AccAddress
|
||||||
initialModuleCoins sdk.Coins
|
initialModuleCoins sdk.Coins
|
||||||
initialBorrowerCoins sdk.Coins
|
initialBorrowerCoins sdk.Coins
|
||||||
depositCoins []sdk.Coin
|
depositCoins sdk.Coins
|
||||||
borrowCoins sdk.Coins
|
borrowCoins sdk.Coins
|
||||||
beginBlockerTime int64
|
beginBlockerTime int64
|
||||||
ltvIndexCount int
|
ltvIndexCount int
|
||||||
@ -60,7 +60,7 @@ func (suite *KeeperTestSuite) TestIndexLiquidation() {
|
|||||||
borrower: borrower,
|
borrower: borrower,
|
||||||
initialModuleCoins: sdk.NewCoins(sdk.NewCoin("ukava", sdk.NewInt(100*KAVA_CF))),
|
initialModuleCoins: sdk.NewCoins(sdk.NewCoin("ukava", sdk.NewInt(100*KAVA_CF))),
|
||||||
initialBorrowerCoins: sdk.NewCoins(sdk.NewCoin("ukava", sdk.NewInt(100*KAVA_CF))),
|
initialBorrowerCoins: sdk.NewCoins(sdk.NewCoin("ukava", sdk.NewInt(100*KAVA_CF))),
|
||||||
depositCoins: []sdk.Coin{sdk.NewCoin("ukava", sdk.NewInt(10*KAVA_CF))},
|
depositCoins: sdk.NewCoins(sdk.NewCoin("ukava", sdk.NewInt(10*KAVA_CF))),
|
||||||
borrowCoins: sdk.NewCoins(sdk.NewCoin("ukava", sdk.NewInt(8*KAVA_CF))),
|
borrowCoins: sdk.NewCoins(sdk.NewCoin("ukava", sdk.NewInt(8*KAVA_CF))),
|
||||||
beginBlockerTime: oneMonthInSeconds,
|
beginBlockerTime: oneMonthInSeconds,
|
||||||
ltvIndexCount: int(10),
|
ltvIndexCount: int(10),
|
||||||
@ -88,13 +88,160 @@ func (suite *KeeperTestSuite) TestIndexLiquidation() {
|
|||||||
contains: "",
|
contains: "",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"valid: HARD module account starts with no coins",
|
||||||
|
args{
|
||||||
|
borrower: borrower,
|
||||||
|
initialModuleCoins: sdk.Coins{},
|
||||||
|
initialBorrowerCoins: sdk.NewCoins(sdk.NewCoin("ukava", sdk.NewInt(100*KAVA_CF))),
|
||||||
|
depositCoins: sdk.NewCoins(sdk.NewCoin("ukava", sdk.NewInt(10*KAVA_CF))),
|
||||||
|
borrowCoins: sdk.NewCoins(sdk.NewCoin("ukava", sdk.NewInt(8*KAVA_CF))),
|
||||||
|
beginBlockerTime: oneMonthInSeconds,
|
||||||
|
ltvIndexCount: int(10),
|
||||||
|
expectedBorrowerCoins: sdk.NewCoins(sdk.NewCoin("ukava", sdk.NewInt(98*KAVA_CF))), // initial - deposit + borrow + liquidation leftovers
|
||||||
|
expectedAuctions: auctypes.Auctions{
|
||||||
|
auctypes.CollateralAuction{
|
||||||
|
BaseAuction: auctypes.BaseAuction{
|
||||||
|
ID: 1,
|
||||||
|
Initiator: "hard_liquidator",
|
||||||
|
Lot: sdk.NewInt64Coin("ukava", 2000000),
|
||||||
|
Bidder: nil,
|
||||||
|
Bid: sdk.NewInt64Coin("ukava", 0),
|
||||||
|
HasReceivedBids: false,
|
||||||
|
EndTime: endTime,
|
||||||
|
MaxEndTime: endTime,
|
||||||
|
},
|
||||||
|
CorrespondingDebt: sdk.NewInt64Coin("debt", 0),
|
||||||
|
MaxBid: sdk.NewInt64Coin("ukava", 8050763),
|
||||||
|
LotReturns: lotReturns,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
errArgs{
|
||||||
|
expectLiquidate: true,
|
||||||
|
contains: "",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"valid: LTV index liquidates borrow with multiple coin types",
|
||||||
|
args{
|
||||||
|
borrower: borrower,
|
||||||
|
initialModuleCoins: sdk.NewCoins(sdk.NewCoin("ukava", sdk.NewInt(100*KAVA_CF))),
|
||||||
|
initialBorrowerCoins: sdk.NewCoins(sdk.NewCoin("ukava", sdk.NewInt(100*KAVA_CF)), sdk.NewCoin("bnb", sdk.NewInt(100*BNB_CF))),
|
||||||
|
depositCoins: sdk.NewCoins(sdk.NewCoin("ukava", sdk.NewInt(10*KAVA_CF)), sdk.NewCoin("bnb", sdk.NewInt(10*BNB_CF))),
|
||||||
|
borrowCoins: sdk.NewCoins(sdk.NewCoin("ukava", sdk.NewInt(8*KAVA_CF)), sdk.NewCoin("bnb", sdk.NewInt(8*BNB_CF))),
|
||||||
|
beginBlockerTime: oneMonthInSeconds,
|
||||||
|
ltvIndexCount: int(10),
|
||||||
|
expectedBorrowerCoins: sdk.NewCoins(sdk.NewCoin("ukava", sdk.NewInt(98.000001*KAVA_CF)), sdk.NewCoin("bnb", sdk.NewInt(98*BNB_CF))), // initial - deposit + borrow + liquidation leftovers
|
||||||
|
expectedAuctions: auctypes.Auctions{
|
||||||
|
auctypes.CollateralAuction{
|
||||||
|
BaseAuction: auctypes.BaseAuction{
|
||||||
|
ID: 1,
|
||||||
|
Initiator: "hard_liquidator",
|
||||||
|
Lot: sdk.NewInt64Coin("bnb", 200000000),
|
||||||
|
Bidder: nil,
|
||||||
|
Bid: sdk.NewInt64Coin("bnb", 0),
|
||||||
|
HasReceivedBids: false,
|
||||||
|
EndTime: endTime,
|
||||||
|
MaxEndTime: endTime,
|
||||||
|
},
|
||||||
|
CorrespondingDebt: sdk.NewInt64Coin("debt", 0),
|
||||||
|
MaxBid: sdk.NewInt64Coin("bnb", 804948248),
|
||||||
|
LotReturns: lotReturns,
|
||||||
|
},
|
||||||
|
auctypes.CollateralAuction{
|
||||||
|
BaseAuction: auctypes.BaseAuction{
|
||||||
|
ID: 2,
|
||||||
|
Initiator: "hard_liquidator",
|
||||||
|
Lot: sdk.NewInt64Coin("ukava", 8004),
|
||||||
|
Bidder: nil,
|
||||||
|
Bid: sdk.NewInt64Coin("bnb", 0),
|
||||||
|
HasReceivedBids: false,
|
||||||
|
EndTime: endTime,
|
||||||
|
MaxEndTime: endTime,
|
||||||
|
},
|
||||||
|
CorrespondingDebt: sdk.NewInt64Coin("debt", 0),
|
||||||
|
MaxBid: sdk.NewInt64Coin("bnb", 128242),
|
||||||
|
LotReturns: lotReturns,
|
||||||
|
},
|
||||||
|
auctypes.CollateralAuction{
|
||||||
|
BaseAuction: auctypes.BaseAuction{
|
||||||
|
ID: 3,
|
||||||
|
Initiator: "hard_liquidator",
|
||||||
|
Lot: sdk.NewInt64Coin("ukava", 9992406),
|
||||||
|
Bidder: nil,
|
||||||
|
Bid: sdk.NewInt64Coin("ukava", 0),
|
||||||
|
HasReceivedBids: false,
|
||||||
|
EndTime: endTime,
|
||||||
|
MaxEndTime: endTime,
|
||||||
|
},
|
||||||
|
CorrespondingDebt: sdk.NewInt64Coin("debt", 0),
|
||||||
|
MaxBid: sdk.NewInt64Coin("ukava", 8004766),
|
||||||
|
LotReturns: lotReturns,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
errArgs{
|
||||||
|
expectLiquidate: true,
|
||||||
|
contains: "",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"valid: HARD module account starts with no coins, LTV index liquidates borrow with multiple coin types",
|
||||||
|
args{
|
||||||
|
borrower: borrower,
|
||||||
|
initialModuleCoins: sdk.Coins{},
|
||||||
|
initialBorrowerCoins: sdk.NewCoins(sdk.NewCoin("ukava", sdk.NewInt(100*KAVA_CF)), sdk.NewCoin("bnb", sdk.NewInt(100*BNB_CF))),
|
||||||
|
depositCoins: sdk.NewCoins(sdk.NewCoin("ukava", sdk.NewInt(10*KAVA_CF)), sdk.NewCoin("bnb", sdk.NewInt(10*BNB_CF))),
|
||||||
|
borrowCoins: sdk.NewCoins(sdk.NewCoin("ukava", sdk.NewInt(8*KAVA_CF)), sdk.NewCoin("bnb", sdk.NewInt(8*BNB_CF))),
|
||||||
|
beginBlockerTime: oneMonthInSeconds,
|
||||||
|
ltvIndexCount: int(10),
|
||||||
|
expectedBorrowerCoins: sdk.NewCoins(sdk.NewCoin("ukava", sdk.NewInt(98*KAVA_CF)), sdk.NewCoin("bnb", sdk.NewInt(98*BNB_CF))), // initial - deposit + borrow + liquidation leftovers
|
||||||
|
expectedAuctions: auctypes.Auctions{
|
||||||
|
auctypes.CollateralAuction{
|
||||||
|
BaseAuction: auctypes.BaseAuction{
|
||||||
|
ID: 1,
|
||||||
|
Initiator: "hard_liquidator",
|
||||||
|
Lot: sdk.NewInt64Coin("bnb", 200000000),
|
||||||
|
Bidder: nil,
|
||||||
|
Bid: sdk.NewInt64Coin("bnb", 0),
|
||||||
|
HasReceivedBids: false,
|
||||||
|
EndTime: endTime,
|
||||||
|
MaxEndTime: endTime,
|
||||||
|
},
|
||||||
|
CorrespondingDebt: sdk.NewInt64Coin("debt", 0),
|
||||||
|
MaxBid: sdk.NewInt64Coin("bnb", 805076483),
|
||||||
|
LotReturns: lotReturns,
|
||||||
|
},
|
||||||
|
auctypes.CollateralAuction{
|
||||||
|
BaseAuction: auctypes.BaseAuction{
|
||||||
|
ID: 2,
|
||||||
|
Initiator: "hard_liquidator",
|
||||||
|
Lot: sdk.NewInt64Coin("ukava", 2000000),
|
||||||
|
Bidder: nil,
|
||||||
|
Bid: sdk.NewInt64Coin("ukava", 0),
|
||||||
|
HasReceivedBids: false,
|
||||||
|
EndTime: endTime,
|
||||||
|
MaxEndTime: endTime,
|
||||||
|
},
|
||||||
|
CorrespondingDebt: sdk.NewInt64Coin("debt", 0),
|
||||||
|
MaxBid: sdk.NewInt64Coin("ukava", 8050764),
|
||||||
|
LotReturns: lotReturns,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
errArgs{
|
||||||
|
expectLiquidate: true,
|
||||||
|
contains: "",
|
||||||
|
},
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"invalid: borrow not over limit, LTV index does not liquidate",
|
"invalid: borrow not over limit, LTV index does not liquidate",
|
||||||
args{
|
args{
|
||||||
borrower: borrower,
|
borrower: borrower,
|
||||||
initialModuleCoins: sdk.NewCoins(sdk.NewCoin("ukava", sdk.NewInt(100*KAVA_CF))),
|
initialModuleCoins: sdk.NewCoins(sdk.NewCoin("ukava", sdk.NewInt(100*KAVA_CF))),
|
||||||
initialBorrowerCoins: sdk.NewCoins(sdk.NewCoin("ukava", sdk.NewInt(100*KAVA_CF))),
|
initialBorrowerCoins: sdk.NewCoins(sdk.NewCoin("ukava", sdk.NewInt(100*KAVA_CF))),
|
||||||
depositCoins: []sdk.Coin{sdk.NewCoin("ukava", sdk.NewInt(10*KAVA_CF))},
|
depositCoins: sdk.NewCoins(sdk.NewCoin("ukava", sdk.NewInt(10*KAVA_CF))),
|
||||||
borrowCoins: sdk.NewCoins(sdk.NewCoin("ukava", sdk.NewInt(7*KAVA_CF))),
|
borrowCoins: sdk.NewCoins(sdk.NewCoin("ukava", sdk.NewInt(7*KAVA_CF))),
|
||||||
beginBlockerTime: oneMonthInSeconds,
|
beginBlockerTime: oneMonthInSeconds,
|
||||||
ltvIndexCount: int(10),
|
ltvIndexCount: int(10),
|
||||||
@ -319,6 +466,329 @@ func (suite *KeeperTestSuite) TestIndexLiquidation() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (suite *KeeperTestSuite) TestPricefeedLiquidation() {
|
||||||
|
type step struct {
|
||||||
|
action string
|
||||||
|
moveTimeForward int64 // Seconds to increase blocktime before taking action
|
||||||
|
sender sdk.AccAddress
|
||||||
|
coins sdk.Coins
|
||||||
|
}
|
||||||
|
|
||||||
|
type args struct {
|
||||||
|
users []sdk.AccAddress
|
||||||
|
initialUserCoins []sdk.Coins
|
||||||
|
steps []step
|
||||||
|
liquidateUser sdk.AccAddress
|
||||||
|
oracle sdk.AccAddress
|
||||||
|
pricefeedMarket string
|
||||||
|
pricefeedPrice sdk.Dec
|
||||||
|
expectedUserCoins sdk.Coins // additional coins (if any) the borrower address should have after successfully liquidating position
|
||||||
|
expectedAuctions auctypes.Auctions // the auctions we should expect to find have been started
|
||||||
|
}
|
||||||
|
|
||||||
|
type errArgs struct {
|
||||||
|
expectLiquidate bool
|
||||||
|
contains string
|
||||||
|
}
|
||||||
|
|
||||||
|
type liqTest struct {
|
||||||
|
name string
|
||||||
|
args args
|
||||||
|
errArgs errArgs
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set up test constants
|
||||||
|
model := types.NewInterestRateModel(sdk.MustNewDecFromStr("0"), sdk.MustNewDecFromStr("0.1"), sdk.MustNewDecFromStr("0.8"), sdk.MustNewDecFromStr("0.5"))
|
||||||
|
reserveFactor := sdk.MustNewDecFromStr("0.05")
|
||||||
|
oracle := sdk.AccAddress(crypto.AddressHash([]byte("oracleaddr")))
|
||||||
|
|
||||||
|
userOne := sdk.AccAddress(crypto.AddressHash([]byte("useraddrone")))
|
||||||
|
userTwo := sdk.AccAddress(crypto.AddressHash([]byte("useraddrtwo")))
|
||||||
|
userThree := sdk.AccAddress(crypto.AddressHash([]byte("useraddrthree")))
|
||||||
|
users := []sdk.AccAddress{userOne, userTwo, userThree}
|
||||||
|
|
||||||
|
// All users start with 100 KAVA, 100 USDX, 100 HARD
|
||||||
|
initialCoins := sdk.NewCoins(sdk.NewCoin("ukava", sdk.NewInt(100*KAVA_CF)), sdk.NewCoin("usdx", sdk.NewInt(100*KAVA_CF)), sdk.NewCoin("hard", sdk.NewInt(100*KAVA_CF)))
|
||||||
|
initialUserCoins := []sdk.Coins{initialCoins, initialCoins, initialCoins}
|
||||||
|
|
||||||
|
// Set up auction constants
|
||||||
|
layout := "2006-01-02T15:04:05.000Z"
|
||||||
|
endTimeStr := "9000-01-01T00:00:00.000Z"
|
||||||
|
endTime, _ := time.Parse(layout, endTimeStr)
|
||||||
|
lotReturns, _ := auctypes.NewWeightedAddresses([]sdk.AccAddress{userOne}, []sdk.Int{sdk.NewInt(100)})
|
||||||
|
|
||||||
|
testCases := []liqTest{
|
||||||
|
{
|
||||||
|
"scenario 1: solvent",
|
||||||
|
args{
|
||||||
|
users: users,
|
||||||
|
initialUserCoins: initialUserCoins,
|
||||||
|
steps: []step{
|
||||||
|
{ // User one deposits 10 KAVA
|
||||||
|
action: "deposit",
|
||||||
|
moveTimeForward: 0,
|
||||||
|
sender: userOne,
|
||||||
|
coins: sdk.NewCoins(sdk.NewCoin("ukava", sdk.NewInt(10*KAVA_CF))),
|
||||||
|
},
|
||||||
|
{ // User two deposits 10 USDX
|
||||||
|
action: "deposit",
|
||||||
|
moveTimeForward: 0,
|
||||||
|
sender: userTwo,
|
||||||
|
coins: sdk.NewCoins(sdk.NewCoin("usdx", sdk.NewInt(10*KAVA_CF))),
|
||||||
|
},
|
||||||
|
{ // User one borrows 8 USDX
|
||||||
|
action: "borrow",
|
||||||
|
moveTimeForward: 0,
|
||||||
|
sender: userOne,
|
||||||
|
coins: sdk.NewCoins(sdk.NewCoin("usdx", sdk.NewInt(8*KAVA_CF))),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
liquidateUser: userOne,
|
||||||
|
oracle: oracle,
|
||||||
|
pricefeedMarket: "kava:usd",
|
||||||
|
pricefeedPrice: sdk.MustNewDecFromStr("0.99"),
|
||||||
|
expectedUserCoins: sdk.NewCoins(sdk.NewCoin("ukava", sdk.NewInt(90.000001*KAVA_CF)), sdk.NewCoin("usdx", sdk.NewInt(108*KAVA_CF)), sdk.NewCoin("hard", sdk.NewInt(100*KAVA_CF))), // initial - deposit + borrow + liquidation leftovers
|
||||||
|
expectedAuctions: auctypes.Auctions{
|
||||||
|
auctypes.CollateralAuction{
|
||||||
|
BaseAuction: auctypes.BaseAuction{ // Expect: lot = 10 KAVA, max bid = 8 USDX
|
||||||
|
ID: 1,
|
||||||
|
Initiator: "hard_liquidator",
|
||||||
|
Lot: sdk.NewInt64Coin("ukava", 9999999),
|
||||||
|
Bidder: nil,
|
||||||
|
Bid: sdk.NewInt64Coin("usdx", 0),
|
||||||
|
HasReceivedBids: false,
|
||||||
|
EndTime: endTime,
|
||||||
|
MaxEndTime: endTime,
|
||||||
|
},
|
||||||
|
CorrespondingDebt: sdk.NewInt64Coin("debt", 0),
|
||||||
|
MaxBid: sdk.NewInt64Coin("usdx", 8*KAVA_CF),
|
||||||
|
LotReturns: lotReturns,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
errArgs{
|
||||||
|
expectLiquidate: true,
|
||||||
|
contains: "",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"scenario 2: insolvent",
|
||||||
|
args{
|
||||||
|
users: users,
|
||||||
|
initialUserCoins: initialUserCoins,
|
||||||
|
steps: []step{
|
||||||
|
{ // User one deposits 10 KAVA
|
||||||
|
action: "deposit",
|
||||||
|
moveTimeForward: 0,
|
||||||
|
sender: userOne,
|
||||||
|
coins: sdk.NewCoins(sdk.NewCoin("ukava", sdk.NewInt(10*KAVA_CF))),
|
||||||
|
},
|
||||||
|
{ // User two deposits 10 USDX
|
||||||
|
action: "deposit",
|
||||||
|
moveTimeForward: 0,
|
||||||
|
sender: userTwo,
|
||||||
|
coins: sdk.NewCoins(sdk.NewCoin("usdx", sdk.NewInt(10*KAVA_CF))),
|
||||||
|
},
|
||||||
|
{ // User one borrows 8 USDX
|
||||||
|
action: "borrow",
|
||||||
|
moveTimeForward: 0,
|
||||||
|
sender: userOne,
|
||||||
|
coins: sdk.NewCoins(sdk.NewCoin("usdx", sdk.NewInt(8*KAVA_CF))),
|
||||||
|
},
|
||||||
|
{ // User three deposits 20 HARD
|
||||||
|
action: "deposit",
|
||||||
|
moveTimeForward: 0,
|
||||||
|
sender: userThree,
|
||||||
|
coins: sdk.NewCoins(sdk.NewCoin("hard", sdk.NewInt(20*KAVA_CF))),
|
||||||
|
},
|
||||||
|
{ // User three borrows 8 KAVA
|
||||||
|
action: "borrow",
|
||||||
|
moveTimeForward: 0,
|
||||||
|
sender: userThree,
|
||||||
|
coins: sdk.NewCoins(sdk.NewCoin("ukava", sdk.NewInt(8*KAVA_CF))),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
liquidateUser: userOne,
|
||||||
|
oracle: oracle,
|
||||||
|
pricefeedMarket: "kava:usd",
|
||||||
|
pricefeedPrice: sdk.MustNewDecFromStr("0.99"),
|
||||||
|
expectedUserCoins: sdk.NewCoins(sdk.NewCoin("ukava", sdk.NewInt(90.000000*KAVA_CF)), sdk.NewCoin("usdx", sdk.NewInt(108*KAVA_CF)), sdk.NewCoin("hard", sdk.NewInt(100*KAVA_CF))), // initial - deposit + borrow + liquidation leftovers
|
||||||
|
expectedAuctions: auctypes.Auctions{ // Expect: lot = 2 KAVA, max bid = 8 USDX
|
||||||
|
auctypes.CollateralAuction{
|
||||||
|
BaseAuction: auctypes.BaseAuction{
|
||||||
|
ID: 1,
|
||||||
|
Initiator: "hard_liquidator",
|
||||||
|
Lot: sdk.NewInt64Coin("ukava", 2*KAVA_CF),
|
||||||
|
Bidder: nil,
|
||||||
|
Bid: sdk.NewInt64Coin("usdx", 0),
|
||||||
|
HasReceivedBids: false,
|
||||||
|
EndTime: endTime,
|
||||||
|
MaxEndTime: endTime,
|
||||||
|
},
|
||||||
|
CorrespondingDebt: sdk.NewInt64Coin("debt", 0),
|
||||||
|
MaxBid: sdk.NewInt64Coin("usdx", 8*KAVA_CF),
|
||||||
|
LotReturns: lotReturns,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
errArgs{
|
||||||
|
expectLiquidate: true,
|
||||||
|
contains: "",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, tc := range testCases {
|
||||||
|
suite.Run(tc.name, func() {
|
||||||
|
// Initialize test app and set context
|
||||||
|
tApp := app.NewTestApp()
|
||||||
|
ctx := tApp.NewContext(true, abci.Header{Height: 1, Time: tmtime.Now()})
|
||||||
|
|
||||||
|
// Auth module genesis state
|
||||||
|
authGS := app.NewAuthGenState(users, initialUserCoins)
|
||||||
|
|
||||||
|
// Hard module genesis state
|
||||||
|
hardGS := types.NewGenesisState(types.NewParams(
|
||||||
|
types.MoneyMarkets{
|
||||||
|
types.NewMoneyMarket("usdx",
|
||||||
|
types.NewBorrowLimit(false, sdk.NewDec(100000000*KAVA_CF), sdk.MustNewDecFromStr("0.8")), // Borrow Limit
|
||||||
|
"usdx:usd", // Market ID
|
||||||
|
sdk.NewInt(KAVA_CF), // Conversion Factor
|
||||||
|
sdk.NewInt(100000*KAVA_CF), // Auction Size
|
||||||
|
model, // Interest Rate Model
|
||||||
|
reserveFactor, // Reserve Factor
|
||||||
|
sdk.MustNewDecFromStr("0.05")), // Keeper Reward Percent
|
||||||
|
types.NewMoneyMarket("ukava",
|
||||||
|
types.NewBorrowLimit(false, sdk.NewDec(100000000*KAVA_CF), sdk.MustNewDecFromStr("0.8")), // Borrow Limit
|
||||||
|
"kava:usd", // Market ID
|
||||||
|
sdk.NewInt(KAVA_CF), // Conversion Factor
|
||||||
|
sdk.NewInt(100000*KAVA_CF), // Auction Size
|
||||||
|
model, // Interest Rate Model
|
||||||
|
reserveFactor, // Reserve Factor
|
||||||
|
sdk.MustNewDecFromStr("0.05")), // Keeper Reward Percent
|
||||||
|
types.NewMoneyMarket("hard",
|
||||||
|
types.NewBorrowLimit(false, sdk.NewDec(100000000*KAVA_CF), sdk.MustNewDecFromStr("0.8")), // Borrow Limit
|
||||||
|
"hard:usd", // Market ID
|
||||||
|
sdk.NewInt(KAVA_CF), // Conversion Factor
|
||||||
|
sdk.NewInt(100000*KAVA_CF), // Auction Size
|
||||||
|
model, // Interest Rate Model
|
||||||
|
reserveFactor, // Reserve Factor
|
||||||
|
sdk.MustNewDecFromStr("0.05")), // Keeper Reward Percent
|
||||||
|
},
|
||||||
|
10, // LTV counter
|
||||||
|
), types.DefaultAccumulationTimes, types.DefaultDeposits, types.DefaultBorrows, types.DefaultTotalSupplied, types.DefaultTotalBorrowed, types.DefaultTotalReserves)
|
||||||
|
|
||||||
|
// Pricefeed module genesis state
|
||||||
|
pricefeedGS := pricefeed.GenesisState{
|
||||||
|
Params: pricefeed.Params{
|
||||||
|
Markets: []pricefeed.Market{
|
||||||
|
{MarketID: "usdx:usd", BaseAsset: "usdx", QuoteAsset: "usd", Oracles: []sdk.AccAddress{oracle}, Active: true},
|
||||||
|
{MarketID: "kava:usd", BaseAsset: "kava", QuoteAsset: "usd", Oracles: []sdk.AccAddress{oracle}, Active: true},
|
||||||
|
{MarketID: "hard:usd", BaseAsset: "hard", QuoteAsset: "usd", Oracles: []sdk.AccAddress{oracle}, Active: true},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
PostedPrices: []pricefeed.PostedPrice{
|
||||||
|
{
|
||||||
|
MarketID: "usdx:usd",
|
||||||
|
OracleAddress: oracle,
|
||||||
|
Price: sdk.MustNewDecFromStr("1.00"),
|
||||||
|
Expiry: time.Now().Add(100 * time.Hour),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
MarketID: "kava:usd",
|
||||||
|
OracleAddress: oracle,
|
||||||
|
Price: sdk.MustNewDecFromStr("2.00"),
|
||||||
|
Expiry: time.Now().Add(100 * time.Hour),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
MarketID: "hard:usd",
|
||||||
|
OracleAddress: oracle,
|
||||||
|
Price: sdk.MustNewDecFromStr("1.00"),
|
||||||
|
Expiry: time.Now().Add(100 * time.Hour),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
// Initialize test application
|
||||||
|
tApp.InitializeFromGenesisStates(authGS,
|
||||||
|
app.GenesisState{pricefeed.ModuleName: pricefeed.ModuleCdc.MustMarshalJSON(pricefeedGS)},
|
||||||
|
app.GenesisState{types.ModuleName: types.ModuleCdc.MustMarshalJSON(hardGS)})
|
||||||
|
// Load keepers
|
||||||
|
auctionKeeper := tApp.GetAuctionKeeper()
|
||||||
|
pricefeedKeeper := tApp.GetPriceFeedKeeper()
|
||||||
|
keeper := tApp.GetHardKeeper()
|
||||||
|
// Set test suite globals
|
||||||
|
suite.app = tApp
|
||||||
|
suite.ctx = ctx
|
||||||
|
suite.keeper = keeper
|
||||||
|
suite.auctionKeeper = auctionKeeper
|
||||||
|
suite.pricefeedKeeper = pricefeedKeeper
|
||||||
|
|
||||||
|
var err error
|
||||||
|
|
||||||
|
// Run begin blocker to set up state
|
||||||
|
hard.BeginBlocker(suite.ctx, suite.keeper)
|
||||||
|
|
||||||
|
currCtx := suite.ctx
|
||||||
|
for _, step := range tc.args.steps {
|
||||||
|
currCtx = currCtx.WithBlockTime(time.Unix(currCtx.BlockTime().Unix()+(step.moveTimeForward), 0))
|
||||||
|
switch step.action {
|
||||||
|
case "deposit":
|
||||||
|
err = suite.keeper.Deposit(currCtx, step.sender, step.coins)
|
||||||
|
suite.Require().NoError(err)
|
||||||
|
case "borrow":
|
||||||
|
err = suite.keeper.Borrow(currCtx, step.sender, step.coins)
|
||||||
|
suite.Require().NoError(err)
|
||||||
|
default:
|
||||||
|
suite.Fail("each define action for each step")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update asset's price in price feed
|
||||||
|
expiryTime := currCtx.BlockTime().Add(100 * time.Hour)
|
||||||
|
_, err = suite.pricefeedKeeper.SetPrice(currCtx, tc.args.oracle, tc.args.pricefeedMarket, tc.args.pricefeedPrice, expiryTime)
|
||||||
|
suite.Require().NoError(err)
|
||||||
|
err = suite.pricefeedKeeper.SetCurrentPrices(currCtx, tc.args.pricefeedMarket)
|
||||||
|
suite.Require().NoError(err)
|
||||||
|
priceInfo, err := suite.pricefeedKeeper.GetCurrentPrice(ctx, tc.args.pricefeedMarket)
|
||||||
|
suite.Require().NoError(err)
|
||||||
|
suite.Require().Equal(tc.args.pricefeedPrice, priceInfo.Price)
|
||||||
|
|
||||||
|
// Liquidate the borrow by running begin blocker
|
||||||
|
hard.BeginBlocker(currCtx, suite.keeper)
|
||||||
|
|
||||||
|
if tc.errArgs.expectLiquidate {
|
||||||
|
// Check borrow does not exist after liquidation
|
||||||
|
_, foundBorrowAfter := suite.keeper.GetBorrow(currCtx, tc.args.liquidateUser)
|
||||||
|
suite.Require().False(foundBorrowAfter)
|
||||||
|
// Check deposits do not exist after liquidation
|
||||||
|
_, foundDepositAfter := suite.keeper.GetDeposit(currCtx, tc.args.liquidateUser)
|
||||||
|
suite.Require().False(foundDepositAfter)
|
||||||
|
|
||||||
|
// Check that borrower's balance contains the expected coins
|
||||||
|
accBorrower := suite.getAccountAtCtx(tc.args.liquidateUser, currCtx)
|
||||||
|
suite.Require().Equal(tc.args.expectedUserCoins, accBorrower.GetCoins())
|
||||||
|
|
||||||
|
// Check that the expected auctions have been created
|
||||||
|
auctions := suite.auctionKeeper.GetAllAuctions(currCtx)
|
||||||
|
suite.Require().True(len(auctions) > 0)
|
||||||
|
suite.Require().Equal(tc.args.expectedAuctions, auctions)
|
||||||
|
} else {
|
||||||
|
// Check that the user's borrow exists
|
||||||
|
_, foundBorrowAfter := suite.keeper.GetBorrow(currCtx, tc.args.liquidateUser)
|
||||||
|
suite.Require().True(foundBorrowAfter)
|
||||||
|
// Check that the user's deposits exist
|
||||||
|
_, foundDepositAfter := suite.keeper.GetDeposit(currCtx, tc.args.liquidateUser)
|
||||||
|
suite.Require().True(foundDepositAfter)
|
||||||
|
|
||||||
|
// Check that no auctions have been created
|
||||||
|
auctions := suite.auctionKeeper.GetAllAuctions(currCtx)
|
||||||
|
suite.Require().True(len(auctions) == 0)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func (suite *KeeperTestSuite) TestFullIndexLiquidation() {
|
func (suite *KeeperTestSuite) TestFullIndexLiquidation() {
|
||||||
type args struct {
|
type args struct {
|
||||||
borrower sdk.AccAddress
|
borrower sdk.AccAddress
|
||||||
|
Loading…
Reference in New Issue
Block a user