2020-10-30 09:59:47 +00:00
|
|
|
package keeper
|
|
|
|
|
|
|
|
import (
|
2020-11-11 15:05:17 +00:00
|
|
|
"strings"
|
|
|
|
|
2020-10-30 09:59:47 +00:00
|
|
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
2020-11-03 09:46:08 +00:00
|
|
|
sdkerrors "github.com/cosmos/cosmos-sdk/types/errors"
|
2020-10-30 09:59:47 +00:00
|
|
|
|
|
|
|
"github.com/kava-labs/kava/x/harvest/types"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Borrow funds
|
|
|
|
func (k Keeper) Borrow(ctx sdk.Context, borrower sdk.AccAddress, coins sdk.Coins) error {
|
2020-11-12 15:50:54 +00:00
|
|
|
// Validate borrow amount within user and protocol limits
|
2020-11-09 21:52:08 +00:00
|
|
|
err := k.ValidateBorrow(ctx, borrower, coins)
|
2020-11-03 09:46:08 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-11-12 15:50:54 +00:00
|
|
|
// Sends coins from Harvest module account to user
|
2020-11-03 09:46:08 +00:00
|
|
|
err = k.supplyKeeper.SendCoinsFromModuleToAccount(ctx, types.ModuleAccountName, borrower, coins)
|
2020-10-30 09:59:47 +00:00
|
|
|
if err != nil {
|
2020-11-11 15:05:17 +00:00
|
|
|
if strings.Contains(err.Error(), "insufficient account funds") {
|
|
|
|
modAccCoins := k.supplyKeeper.GetModuleAccount(ctx, types.ModuleAccountName).GetCoins()
|
|
|
|
for _, coin := range coins {
|
|
|
|
_, isNegative := modAccCoins.SafeSub(sdk.NewCoins(coin))
|
|
|
|
if isNegative {
|
|
|
|
return sdkerrors.Wrapf(types.ErrBorrowExceedsAvailableBalance,
|
|
|
|
"the requested borrow amount of %s exceeds the total amount of %s%s available to borrow",
|
|
|
|
coin, modAccCoins.AmountOf(coin.Denom), coin.Denom,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-10-30 09:59:47 +00:00
|
|
|
}
|
|
|
|
|
2020-11-12 15:50:54 +00:00
|
|
|
// Update user's borrow in store
|
2020-10-30 09:59:47 +00:00
|
|
|
borrow, found := k.GetBorrow(ctx, borrower)
|
|
|
|
if !found {
|
|
|
|
borrow = types.NewBorrow(borrower, coins)
|
|
|
|
} else {
|
|
|
|
borrow.Amount = borrow.Amount.Add(coins...)
|
|
|
|
}
|
|
|
|
k.SetBorrow(ctx, borrow)
|
|
|
|
|
2020-11-12 15:50:54 +00:00
|
|
|
// Update total borrowed amount
|
|
|
|
k.IncrementBorrowedCoins(ctx, coins)
|
|
|
|
|
2020-10-30 09:59:47 +00:00
|
|
|
ctx.EventManager().EmitEvent(
|
|
|
|
sdk.NewEvent(
|
|
|
|
types.EventTypeHarvestBorrow,
|
|
|
|
sdk.NewAttribute(types.AttributeKeyBorrower, borrow.Borrower.String()),
|
|
|
|
sdk.NewAttribute(types.AttributeKeyBorrowCoins, coins.String()),
|
|
|
|
),
|
|
|
|
)
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
2020-11-03 09:46:08 +00:00
|
|
|
|
|
|
|
// ValidateBorrow validates a borrow request against borrower and protocol requirements
|
2020-11-09 21:52:08 +00:00
|
|
|
func (k Keeper) ValidateBorrow(ctx sdk.Context, borrower sdk.AccAddress, amount sdk.Coins) error {
|
2020-11-12 15:50:54 +00:00
|
|
|
if amount.IsZero() {
|
|
|
|
return types.ErrBorrowEmptyCoins
|
|
|
|
}
|
|
|
|
|
2020-11-09 21:52:08 +00:00
|
|
|
// Get the proposed borrow USD value
|
|
|
|
moneyMarketCache := map[string]types.MoneyMarket{}
|
|
|
|
proprosedBorrowUSDValue := sdk.ZeroDec()
|
|
|
|
for _, coin := range amount {
|
|
|
|
moneyMarket, ok := moneyMarketCache[coin.Denom]
|
|
|
|
// Fetch money market and store in local cache
|
|
|
|
if !ok {
|
|
|
|
newMoneyMarket, found := k.GetMoneyMarket(ctx, coin.Denom)
|
|
|
|
if !found {
|
|
|
|
return sdkerrors.Wrapf(types.ErrMarketNotFound, "no market found for denom %s", coin.Denom)
|
|
|
|
}
|
|
|
|
moneyMarketCache[coin.Denom] = newMoneyMarket
|
|
|
|
moneyMarket = newMoneyMarket
|
|
|
|
}
|
|
|
|
|
|
|
|
// Calculate this coin's USD value and add it borrow's total USD value
|
|
|
|
assetPriceInfo, err := k.pricefeedKeeper.GetCurrentPrice(ctx, moneyMarket.SpotMarketID)
|
|
|
|
if err != nil {
|
|
|
|
return sdkerrors.Wrapf(types.ErrPriceNotFound, "no price found for market %s", moneyMarket.SpotMarketID)
|
|
|
|
}
|
|
|
|
coinUSDValue := sdk.NewDecFromInt(coin.Amount).Quo(sdk.NewDecFromInt(moneyMarket.ConversionFactor)).Mul(assetPriceInfo.Price)
|
2020-11-12 15:50:54 +00:00
|
|
|
|
|
|
|
// Validate the requested borrow value for the asset against the money market's global borrow limit
|
|
|
|
if moneyMarket.BorrowLimit.HasMaxLimit {
|
|
|
|
var assetTotalBorrowedAmount sdk.Int
|
|
|
|
totalBorrowedCoins, found := k.GetBorrowedCoins(ctx)
|
|
|
|
if !found {
|
|
|
|
assetTotalBorrowedAmount = sdk.ZeroInt()
|
|
|
|
} else {
|
|
|
|
assetTotalBorrowedAmount = totalBorrowedCoins.AmountOf(coin.Denom)
|
|
|
|
}
|
|
|
|
newProposedAssetTotalBorrowedAmount := sdk.NewDecFromInt(assetTotalBorrowedAmount.Add(coin.Amount))
|
|
|
|
if newProposedAssetTotalBorrowedAmount.GT(moneyMarket.BorrowLimit.MaximumLimit) {
|
|
|
|
return sdkerrors.Wrapf(types.ErrGreaterThanAssetBorrowLimit,
|
|
|
|
"proposed borrow would result in %s borrowed, but the maximum global asset borrow limit is %s",
|
|
|
|
newProposedAssetTotalBorrowedAmount, moneyMarket.BorrowLimit.MaximumLimit)
|
|
|
|
}
|
|
|
|
}
|
2020-11-09 21:52:08 +00:00
|
|
|
proprosedBorrowUSDValue = proprosedBorrowUSDValue.Add(coinUSDValue)
|
2020-11-03 09:46:08 +00:00
|
|
|
}
|
|
|
|
|
2020-11-09 21:52:08 +00:00
|
|
|
// Get the total borrowable USD amount at user's existing deposits
|
2020-11-03 09:46:08 +00:00
|
|
|
deposits := k.GetDepositsByUser(ctx, borrower)
|
|
|
|
if len(deposits) == 0 {
|
|
|
|
return sdkerrors.Wrapf(types.ErrDepositsNotFound, "no deposits found for %s", borrower)
|
|
|
|
}
|
2020-11-05 17:36:49 +00:00
|
|
|
totalBorrowableAmount := sdk.ZeroDec()
|
|
|
|
for _, deposit := range deposits {
|
2020-11-09 21:52:08 +00:00
|
|
|
moneyMarket, ok := moneyMarketCache[deposit.Amount.Denom]
|
|
|
|
// Fetch money market and store in local cache
|
|
|
|
if !ok {
|
|
|
|
newMoneyMarket, found := k.GetMoneyMarket(ctx, deposit.Amount.Denom)
|
|
|
|
if !found {
|
|
|
|
return sdkerrors.Wrapf(types.ErrMarketNotFound, "no market found for denom %s", deposit.Amount.Denom)
|
|
|
|
}
|
|
|
|
moneyMarketCache[deposit.Amount.Denom] = newMoneyMarket
|
|
|
|
moneyMarket = newMoneyMarket
|
|
|
|
}
|
|
|
|
|
|
|
|
// Calculate the borrowable amount and add it to the user's total borrowable amount
|
|
|
|
assetPriceInfo, err := k.pricefeedKeeper.GetCurrentPrice(ctx, moneyMarket.SpotMarketID)
|
2020-11-05 17:36:49 +00:00
|
|
|
if err != nil {
|
2020-11-09 21:52:08 +00:00
|
|
|
sdkerrors.Wrapf(types.ErrPriceNotFound, "no price found for market %s", moneyMarket.SpotMarketID)
|
2020-11-05 17:36:49 +00:00
|
|
|
}
|
2020-11-09 21:52:08 +00:00
|
|
|
depositUSDValue := sdk.NewDecFromInt(deposit.Amount.Amount).Quo(sdk.NewDecFromInt(moneyMarket.ConversionFactor)).Mul(assetPriceInfo.Price)
|
|
|
|
borrowableAmountForDeposit := depositUSDValue.Mul(moneyMarket.BorrowLimit.LoanToValue)
|
2020-11-05 17:36:49 +00:00
|
|
|
totalBorrowableAmount = totalBorrowableAmount.Add(borrowableAmountForDeposit)
|
2020-11-03 09:46:08 +00:00
|
|
|
}
|
|
|
|
|
2020-11-09 21:52:08 +00:00
|
|
|
// Get the total USD value of user's existing borrows
|
|
|
|
existingBorrowUSDValue := sdk.ZeroDec()
|
|
|
|
existingBorrow, found := k.GetBorrow(ctx, borrower)
|
2020-11-03 09:46:08 +00:00
|
|
|
if found {
|
2020-11-09 21:52:08 +00:00
|
|
|
for _, borrowedCoin := range existingBorrow.Amount {
|
|
|
|
moneyMarket, ok := moneyMarketCache[borrowedCoin.Denom]
|
|
|
|
// Fetch money market and store in local cache
|
|
|
|
if !ok {
|
|
|
|
newMoneyMarket, found := k.GetMoneyMarket(ctx, borrowedCoin.Denom)
|
|
|
|
if !found {
|
|
|
|
return sdkerrors.Wrapf(types.ErrMarketNotFound, "no market found for denom %s", borrowedCoin.Denom)
|
|
|
|
}
|
|
|
|
moneyMarketCache[borrowedCoin.Denom] = newMoneyMarket
|
|
|
|
moneyMarket = newMoneyMarket
|
|
|
|
}
|
|
|
|
|
|
|
|
// Calculate this borrow coin's USD value and add it to the total previous borrowed USD value
|
|
|
|
assetPriceInfo, err := k.pricefeedKeeper.GetCurrentPrice(ctx, moneyMarket.SpotMarketID)
|
|
|
|
if err != nil {
|
|
|
|
return sdkerrors.Wrapf(types.ErrPriceNotFound, "no price found for market %s", moneyMarket.SpotMarketID)
|
|
|
|
}
|
|
|
|
coinUSDValue := sdk.NewDecFromInt(borrowedCoin.Amount).Quo(sdk.NewDecFromInt(moneyMarket.ConversionFactor)).Mul(assetPriceInfo.Price)
|
|
|
|
existingBorrowUSDValue = existingBorrowUSDValue.Add(coinUSDValue)
|
2020-11-03 09:46:08 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-05 17:36:49 +00:00
|
|
|
// Validate that the proposed borrow's USD value is within user's borrowable limit
|
2020-11-09 21:52:08 +00:00
|
|
|
if proprosedBorrowUSDValue.GT(totalBorrowableAmount.Sub(existingBorrowUSDValue)) {
|
2020-11-03 09:46:08 +00:00
|
|
|
return sdkerrors.Wrapf(types.ErrInsufficientLoanToValue, "requested borrow %s is greater than maximum valid borrow", amount)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
2020-11-12 15:50:54 +00:00
|
|
|
|
|
|
|
// IncrementBorrowedCoins increments the amount of borrowed coins by the newCoins parameter
|
|
|
|
func (k Keeper) IncrementBorrowedCoins(ctx sdk.Context, newCoins sdk.Coins) {
|
|
|
|
borrowedCoins, found := k.GetBorrowedCoins(ctx)
|
|
|
|
if !found {
|
|
|
|
k.SetBorrowedCoins(ctx, newCoins)
|
|
|
|
} else {
|
|
|
|
k.SetBorrowedCoins(ctx, borrowedCoins.Add(newCoins...))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// DecrementBorrowedCoins decrements the amount of borrowed coins by the coins parameter
|
|
|
|
func (k Keeper) DecrementBorrowedCoins(ctx sdk.Context, coins sdk.Coins) error {
|
|
|
|
borrowedCoins, found := k.GetBorrowedCoins(ctx)
|
|
|
|
if !found {
|
|
|
|
return sdkerrors.Wrapf(types.ErrBorrowedCoinsNotFound, "cannot repay coins if no coins are currently borrowed")
|
|
|
|
}
|
|
|
|
|
|
|
|
updatedBorrowedCoins, isAnyNegative := borrowedCoins.SafeSub(coins)
|
|
|
|
if isAnyNegative {
|
|
|
|
return types.ErrNegativeBorrowedCoins
|
|
|
|
}
|
|
|
|
|
|
|
|
k.SetBorrowedCoins(ctx, updatedBorrowedCoins)
|
|
|
|
return nil
|
|
|
|
}
|