mirror of
https://github.com/0glabs/0g-chain.git
synced 2024-12-28 09:15:24 +00:00
103 lines
4.0 KiB
Go
103 lines
4.0 KiB
Go
|
package keeper
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
|
||
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
||
|
"github.com/kava-labs/kava/x/cdp/types"
|
||
|
)
|
||
|
|
||
|
// SeizeCollateral liquidates the collateral in the input cdp.
|
||
|
// the following operations are performed:
|
||
|
// 1. updates the fees for the input cdp,
|
||
|
// 2. sends collateral for all deposits from the cdp module to the liquidator module,
|
||
|
// 3. moves debt coins from the cdp module to the liquidator module,
|
||
|
// 4. decrements the total amount of principal outstanding for that collateral type
|
||
|
// (this is the equivalent of saying that fees are no longer accumulated by a cdp once it
|
||
|
// gets liquidated)
|
||
|
func (k Keeper) SeizeCollateral(ctx sdk.Context, cdp types.CDP) {
|
||
|
// Update fees
|
||
|
periods := sdk.NewInt(ctx.BlockTime().Unix()).Sub(sdk.NewInt(cdp.FeesUpdated.Unix()))
|
||
|
fees := k.CalculateFees(ctx, cdp.Principal.Add(cdp.AccumulatedFees), periods, cdp.Collateral[0].Denom)
|
||
|
cdp.AccumulatedFees = cdp.AccumulatedFees.Add(fees)
|
||
|
cdp.FeesUpdated = ctx.BlockTime()
|
||
|
|
||
|
// Liquidate deposits
|
||
|
deposits := k.GetDeposits(ctx, cdp.ID)
|
||
|
for _, dep := range deposits {
|
||
|
if !dep.InLiquidation {
|
||
|
dep.InLiquidation = true
|
||
|
|
||
|
ctx.EventManager().EmitEvent(
|
||
|
sdk.NewEvent(
|
||
|
types.EventTypeCdpLiquidation,
|
||
|
sdk.NewAttribute(sdk.AttributeKeyModule, types.AttributeValueCategory),
|
||
|
sdk.NewAttribute(types.AttributeKeyCdpID, fmt.Sprintf("%d", cdp.ID)),
|
||
|
sdk.NewAttribute(types.AttributeKeyDepositor, fmt.Sprintf("%s", dep.Depositor)),
|
||
|
),
|
||
|
)
|
||
|
k.DeleteDeposit(ctx, types.StatusNil, cdp.ID, dep.Depositor)
|
||
|
k.SetDeposit(ctx, dep)
|
||
|
err := k.supplyKeeper.SendCoinsFromModuleToModule(ctx, types.ModuleName, types.LiquidatorMacc, dep.Amount)
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
} else {
|
||
|
return
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Transfer debt coins from cdp module account to liquidator module account
|
||
|
debtAmt := sdk.ZeroInt()
|
||
|
for _, dc := range cdp.Principal {
|
||
|
debtAmt = debtAmt.Add(dc.Amount)
|
||
|
}
|
||
|
for _, dc := range cdp.AccumulatedFees {
|
||
|
debtAmt = debtAmt.Add(dc.Amount)
|
||
|
}
|
||
|
debtCoins := sdk.NewCoins(sdk.NewCoin(k.GetDebtDenom(ctx), debtAmt))
|
||
|
err := k.supplyKeeper.SendCoinsFromModuleToModule(ctx, types.ModuleName, types.LiquidatorMacc, debtCoins)
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
|
||
|
// Decrement total principal for this collateral type
|
||
|
for _, dc := range cdp.Principal {
|
||
|
feeAmount := cdp.AccumulatedFees.AmountOf(dc.Denom)
|
||
|
coinsToDecrement := sdk.NewCoins(dc)
|
||
|
if feeAmount.IsPositive() {
|
||
|
feeCoins := sdk.NewCoins(sdk.NewCoin(dc.Denom, feeAmount))
|
||
|
coinsToDecrement = coinsToDecrement.Add(feeCoins)
|
||
|
}
|
||
|
k.DecrementTotalPrincipal(ctx, cdp.Collateral[0].Denom, coinsToDecrement)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// HandleNewDebt compounds the accumulated fees for the input collateral and principal coins.
|
||
|
// the following operations are performed:
|
||
|
// 1. mints the fee coins in the liquidator module account,
|
||
|
// 2. mints the same amount of debt coins in the cdp module account
|
||
|
// 3. updates the total amount of principal for the input collateral type in the store,
|
||
|
func (k Keeper) HandleNewDebt(ctx sdk.Context, collateralDenom string, principalDenom string, periods sdk.Int) {
|
||
|
previousDebt := k.GetTotalPrincipal(ctx, collateralDenom, principalDenom)
|
||
|
feeCoins := sdk.NewCoins(sdk.NewCoin(principalDenom, previousDebt))
|
||
|
newFees := k.CalculateFees(ctx, feeCoins, periods, collateralDenom)
|
||
|
k.MintDebtCoins(ctx, types.ModuleName, k.GetDebtDenom(ctx), newFees)
|
||
|
k.supplyKeeper.MintCoins(ctx, types.LiquidatorMacc, newFees)
|
||
|
k.SetTotalPrincipal(ctx, collateralDenom, principalDenom, feeCoins.Add(newFees).AmountOf(principalDenom))
|
||
|
}
|
||
|
|
||
|
// LiquidateCdps seizes collateral from all CDPs below the input liquidation ratio
|
||
|
func (k Keeper) LiquidateCdps(ctx sdk.Context, marketID string, denom string, liquidationRatio sdk.Dec) {
|
||
|
price, err := k.pricefeedKeeper.GetCurrentPrice(ctx, marketID)
|
||
|
if err != nil {
|
||
|
return
|
||
|
}
|
||
|
normalizedRatio := sdk.OneDec().Quo(price.Price.Quo(liquidationRatio))
|
||
|
cdpsToLiquidate := k.GetAllCdpsByDenomAndRatio(ctx, denom, normalizedRatio)
|
||
|
for _, c := range cdpsToLiquidate {
|
||
|
k.SeizeCollateral(ctx, c)
|
||
|
}
|
||
|
return
|
||
|
}
|