mirror of
https://github.com/0glabs/0g-chain.git
synced 2025-01-15 01:35:21 +00:00
44467569af
* fix exhibits 19,20,28,32 ineffectual if statements * Apply suggestions from code review * Apply suggestions from code review * Update x/cdp/keeper/cdp.go Co-authored-by: Federico Kunze <31522760+fedekunze@users.noreply.github.com>
554 lines
20 KiB
Go
554 lines
20 KiB
Go
package keeper
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"sort"
|
|
|
|
"github.com/cosmos/cosmos-sdk/store/prefix"
|
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
|
sdkerrors "github.com/cosmos/cosmos-sdk/types/errors"
|
|
|
|
"github.com/kava-labs/kava/x/cdp/types"
|
|
)
|
|
|
|
// BaseDigitFactor is 10**18, used during coin calculations
|
|
const BaseDigitFactor = 1000000000000000000
|
|
|
|
// AddCdp adds a cdp for a specific owner and collateral type
|
|
func (k Keeper) AddCdp(ctx sdk.Context, owner sdk.AccAddress, collateral sdk.Coin, principal sdk.Coin) error {
|
|
// validation
|
|
err := k.ValidateCollateral(ctx, collateral)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
_, found := k.GetCdpByOwnerAndDenom(ctx, owner, collateral.Denom)
|
|
if found {
|
|
return sdkerrors.Wrapf(types.ErrCdpAlreadyExists, "owner %s, denom %s", owner, collateral.Denom)
|
|
}
|
|
err = k.ValidatePrincipalAdd(ctx, principal)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = k.ValidateDebtLimit(ctx, collateral.Denom, principal)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
err = k.ValidateCollateralizationRatio(ctx, collateral, principal, sdk.NewCoin(principal.Denom, sdk.ZeroInt()))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// send coins from the owners account to the cdp module
|
|
id := k.GetNextCdpID(ctx)
|
|
cdp := types.NewCDP(id, owner, collateral, principal, ctx.BlockHeader().Time)
|
|
deposit := types.NewDeposit(cdp.ID, owner, collateral)
|
|
err = k.supplyKeeper.SendCoinsFromAccountToModule(ctx, owner, types.ModuleName, sdk.NewCoins(collateral))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// mint the principal and send to the owners account
|
|
err = k.supplyKeeper.MintCoins(ctx, types.ModuleName, sdk.NewCoins(principal))
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
err = k.supplyKeeper.SendCoinsFromModuleToAccount(ctx, types.ModuleName, owner, sdk.NewCoins(principal))
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
// mint the corresponding amount of debt coins
|
|
err = k.MintDebtCoins(ctx, types.ModuleName, k.GetDebtDenom(ctx), principal)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
// update total principal for input collateral type
|
|
k.IncrementTotalPrincipal(ctx, collateral.Denom, principal)
|
|
|
|
// set the cdp, deposit, and indexes in the store
|
|
collateralToDebtRatio := k.CalculateCollateralToDebtRatio(ctx, collateral, principal)
|
|
err = k.SetCdpAndCollateralRatioIndex(ctx, cdp, collateralToDebtRatio)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
k.IndexCdpByOwner(ctx, cdp)
|
|
k.SetDeposit(ctx, deposit)
|
|
k.SetNextCdpID(ctx, id+1)
|
|
|
|
// emit events for cdp creation, deposit, and draw
|
|
ctx.EventManager().EmitEvent(
|
|
sdk.NewEvent(
|
|
types.EventTypeCreateCdp,
|
|
sdk.NewAttribute(types.AttributeKeyCdpID, fmt.Sprintf("%d", cdp.ID)),
|
|
),
|
|
)
|
|
ctx.EventManager().EmitEvent(
|
|
sdk.NewEvent(
|
|
types.EventTypeCdpDeposit,
|
|
sdk.NewAttribute(sdk.AttributeKeyAmount, collateral.String()),
|
|
sdk.NewAttribute(types.AttributeKeyCdpID, fmt.Sprintf("%d", cdp.ID)),
|
|
),
|
|
)
|
|
ctx.EventManager().EmitEvent(
|
|
sdk.NewEvent(
|
|
types.EventTypeCdpDraw,
|
|
sdk.NewAttribute(sdk.AttributeKeyAmount, principal.String()),
|
|
sdk.NewAttribute(types.AttributeKeyCdpID, fmt.Sprintf("%d", cdp.ID)),
|
|
),
|
|
)
|
|
|
|
return nil
|
|
}
|
|
|
|
// SetCdpAndCollateralRatioIndex sets the cdp and collateral ratio index in the store
|
|
func (k Keeper) SetCdpAndCollateralRatioIndex(ctx sdk.Context, cdp types.CDP, ratio sdk.Dec) error {
|
|
err := k.SetCDP(ctx, cdp)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
k.IndexCdpByCollateralRatio(ctx, cdp.Collateral.Denom, cdp.ID, ratio)
|
|
return nil
|
|
}
|
|
|
|
// MintDebtCoins mints debt coins in the cdp module account
|
|
func (k Keeper) MintDebtCoins(ctx sdk.Context, moduleAccount string, denom string, principalCoins sdk.Coin) error {
|
|
debtCoins := sdk.NewCoins(sdk.NewCoin(denom, principalCoins.Amount))
|
|
return k.supplyKeeper.MintCoins(ctx, moduleAccount, debtCoins)
|
|
}
|
|
|
|
// BurnDebtCoins burns debt coins from the cdp module account
|
|
func (k Keeper) BurnDebtCoins(ctx sdk.Context, moduleAccount string, denom string, paymentCoins sdk.Coin) error {
|
|
debtCoins := sdk.NewCoins(sdk.NewCoin(denom, paymentCoins.Amount))
|
|
return k.supplyKeeper.BurnCoins(ctx, moduleAccount, debtCoins)
|
|
}
|
|
|
|
// GetCdpID returns the id of the cdp corresponding to a specific owner and collateral denom
|
|
func (k Keeper) GetCdpID(ctx sdk.Context, owner sdk.AccAddress, denom string) (uint64, bool) {
|
|
|
|
cdpIDs, found := k.GetCdpIdsByOwner(ctx, owner)
|
|
if !found {
|
|
return 0, false
|
|
}
|
|
for _, id := range cdpIDs {
|
|
_, found = k.GetCDP(ctx, denom, id)
|
|
if found {
|
|
return id, true
|
|
}
|
|
}
|
|
return 0, false
|
|
|
|
}
|
|
|
|
// GetCdpIdsByOwner returns all the ids of cdps corresponding to a particular owner
|
|
func (k Keeper) GetCdpIdsByOwner(ctx sdk.Context, owner sdk.AccAddress) ([]uint64, bool) {
|
|
store := prefix.NewStore(ctx.KVStore(k.key), types.CdpIDKeyPrefix)
|
|
bz := store.Get(owner)
|
|
if bz == nil {
|
|
return []uint64{}, false
|
|
}
|
|
var cdpIDs []uint64
|
|
k.cdc.MustUnmarshalBinaryLengthPrefixed(bz, &cdpIDs)
|
|
return cdpIDs, true
|
|
}
|
|
|
|
// GetCdpByOwnerAndDenom queries cdps owned by owner and returns the cdp with matching denom
|
|
func (k Keeper) GetCdpByOwnerAndDenom(ctx sdk.Context, owner sdk.AccAddress, denom string) (types.CDP, bool) {
|
|
cdpIDs, found := k.GetCdpIdsByOwner(ctx, owner)
|
|
if !found {
|
|
return types.CDP{}, false
|
|
}
|
|
for _, id := range cdpIDs {
|
|
cdp, found := k.GetCDP(ctx, denom, id)
|
|
if found {
|
|
return cdp, true
|
|
}
|
|
}
|
|
return types.CDP{}, false
|
|
}
|
|
|
|
// GetCDP returns the cdp associated with a particular collateral denom and id
|
|
func (k Keeper) GetCDP(ctx sdk.Context, collateralDenom string, cdpID uint64) (types.CDP, bool) {
|
|
// get store
|
|
store := prefix.NewStore(ctx.KVStore(k.key), types.CdpKeyPrefix)
|
|
db, found := k.GetDenomPrefix(ctx, collateralDenom)
|
|
if !found {
|
|
return types.CDP{}, false
|
|
}
|
|
// get CDP
|
|
bz := store.Get(types.CdpKey(db, cdpID))
|
|
// unmarshal
|
|
if bz == nil {
|
|
return types.CDP{}, false
|
|
}
|
|
var cdp types.CDP
|
|
k.cdc.MustUnmarshalBinaryLengthPrefixed(bz, &cdp)
|
|
return cdp, true
|
|
}
|
|
|
|
// SetCDP sets a cdp in the store
|
|
func (k Keeper) SetCDP(ctx sdk.Context, cdp types.CDP) error {
|
|
store := prefix.NewStore(ctx.KVStore(k.key), types.CdpKeyPrefix)
|
|
db, found := k.GetDenomPrefix(ctx, cdp.Collateral.Denom)
|
|
if !found {
|
|
return sdkerrors.Wrapf(types.ErrDenomPrefixNotFound, "%s", cdp.Collateral.Denom)
|
|
}
|
|
bz := k.cdc.MustMarshalBinaryLengthPrefixed(cdp)
|
|
store.Set(types.CdpKey(db, cdp.ID), bz)
|
|
return nil
|
|
}
|
|
|
|
// DeleteCDP deletes a cdp from the store
|
|
func (k Keeper) DeleteCDP(ctx sdk.Context, cdp types.CDP) error {
|
|
store := prefix.NewStore(ctx.KVStore(k.key), types.CdpKeyPrefix)
|
|
db, found := k.GetDenomPrefix(ctx, cdp.Collateral.Denom)
|
|
if !found {
|
|
return sdkerrors.Wrapf(types.ErrDenomPrefixNotFound, "%s", cdp.Collateral.Denom)
|
|
}
|
|
store.Delete(types.CdpKey(db, cdp.ID))
|
|
return nil
|
|
|
|
}
|
|
|
|
// GetAllCdps returns all cdps from the store
|
|
func (k Keeper) GetAllCdps(ctx sdk.Context) (cdps types.CDPs) {
|
|
k.IterateAllCdps(ctx, func(cdp types.CDP) bool {
|
|
cdps = append(cdps, cdp)
|
|
return false
|
|
})
|
|
return
|
|
}
|
|
|
|
// GetAllCdpsByDenom returns all cdps of a particular collateral type from the store
|
|
func (k Keeper) GetAllCdpsByDenom(ctx sdk.Context, denom string) (cdps types.CDPs) {
|
|
k.IterateCdpsByDenom(ctx, denom, func(cdp types.CDP) bool {
|
|
cdps = append(cdps, cdp)
|
|
return false
|
|
})
|
|
return
|
|
}
|
|
|
|
// GetAllCdpsByDenomAndRatio returns all cdps of a particular collateral type and below a certain collateralization ratio
|
|
func (k Keeper) GetAllCdpsByDenomAndRatio(ctx sdk.Context, denom string, targetRatio sdk.Dec) (cdps types.CDPs) {
|
|
k.IterateCdpsByCollateralRatio(ctx, denom, targetRatio, func(cdp types.CDP) bool {
|
|
cdps = append(cdps, cdp)
|
|
return false
|
|
})
|
|
return
|
|
}
|
|
|
|
// SetNextCdpID sets the highest cdp id in the store
|
|
func (k Keeper) SetNextCdpID(ctx sdk.Context, id uint64) {
|
|
store := prefix.NewStore(ctx.KVStore(k.key), types.CdpIDKey)
|
|
store.Set([]byte{}, types.GetCdpIDBytes(id))
|
|
}
|
|
|
|
// GetNextCdpID returns the highest cdp id from the store
|
|
func (k Keeper) GetNextCdpID(ctx sdk.Context) (id uint64) {
|
|
store := prefix.NewStore(ctx.KVStore(k.key), types.CdpIDKey)
|
|
bz := store.Get([]byte{})
|
|
if bz == nil {
|
|
panic("starting cdp id not set in genesis")
|
|
}
|
|
id = types.GetCdpIDFromBytes(bz)
|
|
return
|
|
}
|
|
|
|
// IndexCdpByOwner sets the cdp id in the store, indexed by the owner
|
|
func (k Keeper) IndexCdpByOwner(ctx sdk.Context, cdp types.CDP) {
|
|
store := prefix.NewStore(ctx.KVStore(k.key), types.CdpIDKeyPrefix)
|
|
cdpIDs, found := k.GetCdpIdsByOwner(ctx, cdp.Owner)
|
|
|
|
if !found {
|
|
idBytes := k.cdc.MustMarshalBinaryLengthPrefixed([]uint64{cdp.ID})
|
|
store.Set(cdp.Owner, idBytes)
|
|
return
|
|
}
|
|
cdpIDs = append(cdpIDs, cdp.ID)
|
|
sort.Slice(cdpIDs, func(i, j int) bool { return cdpIDs[i] < cdpIDs[j] })
|
|
store.Set(cdp.Owner, k.cdc.MustMarshalBinaryLengthPrefixed(cdpIDs))
|
|
}
|
|
|
|
// RemoveCdpOwnerIndex deletes the cdp id from the store's index of cdps by owner
|
|
func (k Keeper) RemoveCdpOwnerIndex(ctx sdk.Context, cdp types.CDP) {
|
|
store := prefix.NewStore(ctx.KVStore(k.key), types.CdpIDKeyPrefix)
|
|
cdpIDs, found := k.GetCdpIdsByOwner(ctx, cdp.Owner)
|
|
if !found {
|
|
return
|
|
}
|
|
updatedCdpIds := []uint64{}
|
|
for _, id := range cdpIDs {
|
|
if id != cdp.ID {
|
|
updatedCdpIds = append(updatedCdpIds, id)
|
|
}
|
|
}
|
|
if len(updatedCdpIds) == 0 {
|
|
store.Delete(cdp.Owner)
|
|
return
|
|
}
|
|
store.Set(cdp.Owner, k.cdc.MustMarshalBinaryLengthPrefixed(updatedCdpIds))
|
|
}
|
|
|
|
// IndexCdpByCollateralRatio sets the cdp id in the store, indexed by the collateral type and collateral to debt ratio
|
|
func (k Keeper) IndexCdpByCollateralRatio(ctx sdk.Context, denom string, id uint64, collateralRatio sdk.Dec) {
|
|
store := prefix.NewStore(ctx.KVStore(k.key), types.CollateralRatioIndexPrefix)
|
|
db, found := k.GetDenomPrefix(ctx, denom)
|
|
if !found {
|
|
panic(fmt.Sprintf("denom %s prefix not found", denom))
|
|
}
|
|
store.Set(types.CollateralRatioKey(db, id, collateralRatio), types.GetCdpIDBytes(id))
|
|
}
|
|
|
|
// RemoveCdpCollateralRatioIndex deletes the cdp id from the store's index of cdps by collateral type and collateral to debt ratio
|
|
func (k Keeper) RemoveCdpCollateralRatioIndex(ctx sdk.Context, denom string, id uint64, collateralRatio sdk.Dec) {
|
|
store := prefix.NewStore(ctx.KVStore(k.key), types.CollateralRatioIndexPrefix)
|
|
db, found := k.GetDenomPrefix(ctx, denom)
|
|
if !found {
|
|
panic(fmt.Sprintf("denom %s prefix not found", denom))
|
|
}
|
|
store.Delete(types.CollateralRatioKey(db, id, collateralRatio))
|
|
}
|
|
|
|
// GetDebtDenom returns the denom of debt in the system
|
|
func (k Keeper) GetDebtDenom(ctx sdk.Context) (denom string) {
|
|
store := prefix.NewStore(ctx.KVStore(k.key), types.DebtDenomKey)
|
|
bz := store.Get([]byte{})
|
|
k.cdc.MustUnmarshalBinaryLengthPrefixed(bz, &denom)
|
|
return
|
|
}
|
|
|
|
// GetGovDenom returns the denom of debt in the system
|
|
func (k Keeper) GetGovDenom(ctx sdk.Context) (denom string) {
|
|
store := prefix.NewStore(ctx.KVStore(k.key), types.GovDenomKey)
|
|
bz := store.Get([]byte{})
|
|
k.cdc.MustUnmarshalBinaryLengthPrefixed(bz, &denom)
|
|
return
|
|
}
|
|
|
|
// SetDebtDenom set the denom of debt in the system
|
|
func (k Keeper) SetDebtDenom(ctx sdk.Context, denom string) {
|
|
if denom == "" {
|
|
panic("debt denom not set in genesis")
|
|
}
|
|
store := prefix.NewStore(ctx.KVStore(k.key), types.DebtDenomKey)
|
|
store.Set([]byte{}, k.cdc.MustMarshalBinaryLengthPrefixed(denom))
|
|
}
|
|
|
|
// SetGovDenom set the denom of the governance token in the system
|
|
func (k Keeper) SetGovDenom(ctx sdk.Context, denom string) {
|
|
if denom == "" {
|
|
panic("gov denom not set in genesis")
|
|
}
|
|
store := prefix.NewStore(ctx.KVStore(k.key), types.GovDenomKey)
|
|
store.Set([]byte{}, k.cdc.MustMarshalBinaryLengthPrefixed(denom))
|
|
}
|
|
|
|
// ValidateCollateral validates that a collateral is valid for use in cdps
|
|
func (k Keeper) ValidateCollateral(ctx sdk.Context, collateral sdk.Coin) error {
|
|
cp, found := k.GetCollateral(ctx, collateral.Denom)
|
|
if !found {
|
|
return sdkerrors.Wrap(types.ErrCollateralNotSupported, collateral.Denom)
|
|
}
|
|
ok := k.GetMarketStatus(ctx, cp.SpotMarketID)
|
|
if !ok {
|
|
return sdkerrors.Wrap(types.ErrPricefeedDown, collateral.Denom)
|
|
}
|
|
ok = k.GetMarketStatus(ctx, cp.LiquidationMarketID)
|
|
if !ok {
|
|
return sdkerrors.Wrap(types.ErrPricefeedDown, collateral.Denom)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// ValidatePrincipalAdd validates that an asset is valid for use as debt when creating a new cdp
|
|
func (k Keeper) ValidatePrincipalAdd(ctx sdk.Context, principal sdk.Coin) error {
|
|
dp, found := k.GetDebtParam(ctx, principal.Denom)
|
|
if !found {
|
|
return sdkerrors.Wrap(types.ErrDebtNotSupported, principal.Denom)
|
|
}
|
|
if principal.Amount.LT(dp.DebtFloor) {
|
|
return sdkerrors.Wrapf(types.ErrBelowDebtFloor, "proposed %s < minimum %s", principal, dp.DebtFloor)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// ValidatePrincipalDraw validates that an asset is valid for use as debt when drawing debt off an existing cdp
|
|
func (k Keeper) ValidatePrincipalDraw(ctx sdk.Context, principal sdk.Coin, expectedDenom string) error {
|
|
if principal.Denom != expectedDenom {
|
|
return sdkerrors.Wrapf(types.ErrInvalidDebtRequest, "proposed %s, expected %s", principal.Denom, expectedDenom)
|
|
}
|
|
_, found := k.GetDebtParam(ctx, principal.Denom)
|
|
if !found {
|
|
return sdkerrors.Wrap(types.ErrDebtNotSupported, principal.Denom)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// ValidateDebtLimit validates that the input debt amount does not exceed the global debt limit or the debt limit for that collateral
|
|
func (k Keeper) ValidateDebtLimit(ctx sdk.Context, collateralDenom string, principal sdk.Coin) error {
|
|
cp, found := k.GetCollateral(ctx, collateralDenom)
|
|
if !found {
|
|
return sdkerrors.Wrap(types.ErrCollateralNotSupported, collateralDenom)
|
|
}
|
|
totalPrincipal := k.GetTotalPrincipal(ctx, collateralDenom, principal.Denom).Add(principal.Amount)
|
|
collateralLimit := cp.DebtLimit.Amount
|
|
if totalPrincipal.GT(collateralLimit) {
|
|
return sdkerrors.Wrapf(types.ErrExceedsDebtLimit, "debt increase %s > collateral debt limit %s", sdk.NewCoins(sdk.NewCoin(principal.Denom, totalPrincipal)), sdk.NewCoins(sdk.NewCoin(principal.Denom, collateralLimit)))
|
|
}
|
|
globalLimit := k.GetParams(ctx).GlobalDebtLimit.Amount
|
|
if totalPrincipal.GT(globalLimit) {
|
|
return sdkerrors.Wrapf(types.ErrExceedsDebtLimit, "debt increase %s > global debt limit %s", sdk.NewCoin(principal.Denom, totalPrincipal), sdk.NewCoin(principal.Denom, globalLimit))
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// ValidateCollateralizationRatio validate that adding the input principal doesn't put the cdp below the liquidation ratio
|
|
func (k Keeper) ValidateCollateralizationRatio(ctx sdk.Context, collateral sdk.Coin, principal sdk.Coin, fees sdk.Coin) error {
|
|
//
|
|
collateralizationRatio, err := k.CalculateCollateralizationRatio(ctx, collateral, principal, fees, spot)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
liquidationRatio := k.getLiquidationRatio(ctx, collateral.Denom)
|
|
if collateralizationRatio.LT(liquidationRatio) {
|
|
return sdkerrors.Wrapf(types.ErrInvalidCollateralRatio, "collateral %s, collateral ratio %s, liquidation ratio %s", collateral.Denom, collateralizationRatio, liquidationRatio)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// CalculateCollateralToDebtRatio returns the collateral to debt ratio of the input collateral and debt amounts
|
|
func (k Keeper) CalculateCollateralToDebtRatio(ctx sdk.Context, collateral sdk.Coin, debt sdk.Coin) sdk.Dec {
|
|
debtTotal := k.convertDebtToBaseUnits(ctx, debt)
|
|
|
|
if debtTotal.IsZero() || debtTotal.GTE(types.MaxSortableDec) {
|
|
return types.MaxSortableDec.Sub(sdk.SmallestDec())
|
|
}
|
|
|
|
collateralBaseUnits := k.convertCollateralToBaseUnits(ctx, collateral)
|
|
return collateralBaseUnits.Quo(debtTotal)
|
|
}
|
|
|
|
// LoadAugmentedCDP creates a new augmented CDP from an existing CDP
|
|
func (k Keeper) LoadAugmentedCDP(ctx sdk.Context, cdp types.CDP) types.AugmentedCDP {
|
|
// calculate collateralization ratio
|
|
collateralizationRatio, err := k.CalculateCollateralizationRatio(ctx, cdp.Collateral, cdp.Principal, cdp.AccumulatedFees, liquidation)
|
|
if err != nil {
|
|
return types.AugmentedCDP{CDP: cdp}
|
|
}
|
|
// convert collateral value to debt coin
|
|
totalDebt := cdp.Principal.Amount.Add(cdp.AccumulatedFees.Amount)
|
|
collateralValueInDebtDenom := sdk.NewDecFromInt(totalDebt).Mul(collateralizationRatio)
|
|
collateralValueInDebt := sdk.NewCoin(cdp.Principal.Denom, collateralValueInDebtDenom.RoundInt())
|
|
// create new augmuented cdp
|
|
augmentedCDP := types.NewAugmentedCDP(cdp, collateralValueInDebt, collateralizationRatio)
|
|
return augmentedCDP
|
|
}
|
|
|
|
// CalculateCollateralizationRatio returns the collateralization ratio of the input collateral to the input debt plus fees
|
|
func (k Keeper) CalculateCollateralizationRatio(ctx sdk.Context, collateral sdk.Coin, principal sdk.Coin, fees sdk.Coin, pfType pricefeedType) (sdk.Dec, error) {
|
|
if collateral.IsZero() {
|
|
return sdk.ZeroDec(), nil
|
|
}
|
|
var marketID string
|
|
switch pfType {
|
|
case spot:
|
|
marketID = k.getSpotMarketID(ctx, collateral.Denom)
|
|
case liquidation:
|
|
marketID = k.getliquidationMarketID(ctx, collateral.Denom)
|
|
default:
|
|
return sdk.Dec{}, pfType.IsValid()
|
|
}
|
|
|
|
price, err := k.pricefeedKeeper.GetCurrentPrice(ctx, marketID)
|
|
if err != nil {
|
|
return sdk.Dec{}, err
|
|
}
|
|
collateralBaseUnits := k.convertCollateralToBaseUnits(ctx, collateral)
|
|
collateralValue := collateralBaseUnits.Mul(price.Price)
|
|
|
|
prinicpalBaseUnits := k.convertDebtToBaseUnits(ctx, principal)
|
|
principalTotal := prinicpalBaseUnits
|
|
feeBaseUnits := k.convertDebtToBaseUnits(ctx, fees)
|
|
principalTotal = principalTotal.Add(feeBaseUnits)
|
|
|
|
collateralRatio := collateralValue.Quo(principalTotal)
|
|
return collateralRatio, nil
|
|
}
|
|
|
|
// CalculateCollateralizationRatioFromAbsoluteRatio takes a coin's denom and an absolute ratio and returns the respective collateralization ratio
|
|
func (k Keeper) CalculateCollateralizationRatioFromAbsoluteRatio(ctx sdk.Context, collateralDenom string, absoluteRatio sdk.Dec, pfType pricefeedType) (sdk.Dec, error) {
|
|
// get price of collateral
|
|
var marketID string
|
|
switch pfType {
|
|
case spot:
|
|
marketID = k.getSpotMarketID(ctx, collateralDenom)
|
|
case liquidation:
|
|
marketID = k.getliquidationMarketID(ctx, collateralDenom)
|
|
default:
|
|
return sdk.Dec{}, pfType.IsValid()
|
|
}
|
|
|
|
price, err := k.pricefeedKeeper.GetCurrentPrice(ctx, marketID)
|
|
if err != nil {
|
|
return sdk.Dec{}, err
|
|
}
|
|
// convert absolute ratio to collateralization ratio
|
|
respectiveCollateralRatio := absoluteRatio.Quo(price.Price)
|
|
return respectiveCollateralRatio, nil
|
|
}
|
|
|
|
// SetMarketStatus sets the status of the input market, true means the market is up and running, false means it is down
|
|
func (k Keeper) SetMarketStatus(ctx sdk.Context, marketID string, up bool) {
|
|
store := prefix.NewStore(ctx.KVStore(k.key), types.PricefeedStatusKeyPrefix)
|
|
store.Set([]byte(marketID), k.cdc.MustMarshalBinaryBare(up))
|
|
return
|
|
}
|
|
|
|
// GetMarketStatus returns true if the market has a price, otherwise false
|
|
func (k Keeper) GetMarketStatus(ctx sdk.Context, marketID string) (up bool) {
|
|
store := prefix.NewStore(ctx.KVStore(k.key), types.PricefeedStatusKeyPrefix)
|
|
bz := store.Get([]byte(marketID))
|
|
k.cdc.MustUnmarshalBinaryBare(bz, &up)
|
|
return up
|
|
}
|
|
|
|
// UpdatePricefeedStatus determines if the price of an asset is available and updates the global status of the market
|
|
func (k Keeper) UpdatePricefeedStatus(ctx sdk.Context, marketID string) (ok bool) {
|
|
_, err := k.pricefeedKeeper.GetCurrentPrice(ctx, marketID)
|
|
if err != nil {
|
|
k.SetMarketStatus(ctx, marketID, false)
|
|
return false
|
|
}
|
|
k.SetMarketStatus(ctx, marketID, true)
|
|
return true
|
|
}
|
|
|
|
// converts the input collateral to base units (ie multiplies the input by 10^(-ConversionFactor))
|
|
func (k Keeper) convertCollateralToBaseUnits(ctx sdk.Context, collateral sdk.Coin) (baseUnits sdk.Dec) {
|
|
cp, _ := k.GetCollateral(ctx, collateral.Denom)
|
|
return sdk.NewDecFromInt(collateral.Amount).Mul(sdk.NewDecFromIntWithPrec(sdk.OneInt(), cp.ConversionFactor.Int64()))
|
|
}
|
|
|
|
// converts the input debt to base units (ie multiplies the input by 10^(-ConversionFactor))
|
|
func (k Keeper) convertDebtToBaseUnits(ctx sdk.Context, debt sdk.Coin) (baseUnits sdk.Dec) {
|
|
dp, _ := k.GetDebtParam(ctx, debt.Denom)
|
|
return sdk.NewDecFromInt(debt.Amount).Mul(sdk.NewDecFromIntWithPrec(sdk.OneInt(), dp.ConversionFactor.Int64()))
|
|
}
|
|
|
|
type pricefeedType string
|
|
|
|
const (
|
|
spot pricefeedType = "spot"
|
|
liquidation = "liquidation"
|
|
)
|
|
|
|
func (pft pricefeedType) IsValid() error {
|
|
switch pft {
|
|
case spot, liquidation:
|
|
return nil
|
|
}
|
|
return errors.New(fmt.Sprintf("invalid pricefeed type: %s", pft))
|
|
}
|