2022-11-07 16:36:12 +00:00
|
|
|
import BigNumber from 'bignumber.js'
|
2022-11-09 09:04:06 +00:00
|
|
|
import { useCallback, useMemo } from 'react'
|
2022-11-07 16:36:12 +00:00
|
|
|
|
2022-11-09 09:04:06 +00:00
|
|
|
import useCreditManagerStore from 'stores/useCreditManagerStore'
|
2022-11-07 16:36:12 +00:00
|
|
|
import { getTokenDecimals } from 'utils/tokens'
|
2022-11-09 09:04:06 +00:00
|
|
|
|
2022-11-07 16:36:12 +00:00
|
|
|
import useCreditAccountPositions from './useCreditAccountPositions'
|
|
|
|
import useMarkets from './useMarkets'
|
2022-11-09 09:04:06 +00:00
|
|
|
import useTokenPrices from './useTokenPrices'
|
2022-11-09 12:47:36 +00:00
|
|
|
import useRedbankBalances from './useRedbankBalances'
|
2022-11-07 16:36:12 +00:00
|
|
|
|
2022-11-07 17:14:16 +00:00
|
|
|
const getApproximateHourlyInterest = (amount: string, borrowAPY: string) => {
|
|
|
|
const hourlyAPY = BigNumber(borrowAPY).div(24 * 365)
|
|
|
|
|
|
|
|
return hourlyAPY.times(amount).toNumber()
|
|
|
|
}
|
|
|
|
|
2022-11-07 16:36:12 +00:00
|
|
|
const useCalculateMaxWithdrawAmount = (denom: string, borrow: boolean) => {
|
|
|
|
const selectedAccount = useCreditManagerStore((s) => s.selectedAccount)
|
|
|
|
|
|
|
|
const { data: positionsData } = useCreditAccountPositions(selectedAccount ?? '')
|
|
|
|
const { data: marketsData } = useMarkets()
|
|
|
|
const { data: tokenPrices } = useTokenPrices()
|
2022-11-09 12:47:36 +00:00
|
|
|
const { data: redbankBalances } = useRedbankBalances()
|
2022-11-07 16:36:12 +00:00
|
|
|
|
|
|
|
const tokenDecimals = getTokenDecimals(denom)
|
|
|
|
|
|
|
|
const getTokenValue = useCallback(
|
|
|
|
(amount: string, denom: string) => {
|
|
|
|
if (!tokenPrices) return 0
|
|
|
|
|
|
|
|
return BigNumber(amount).times(tokenPrices[denom]).toNumber()
|
|
|
|
},
|
2022-11-09 09:04:06 +00:00
|
|
|
[tokenPrices],
|
2022-11-07 16:36:12 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
const tokenAmountInCreditAccount = useMemo(() => {
|
|
|
|
return positionsData?.coins.find((coin) => coin.denom === denom)?.amount ?? 0
|
|
|
|
}, [denom, positionsData])
|
|
|
|
|
|
|
|
const maxAmount = useMemo(() => {
|
2022-11-09 12:47:36 +00:00
|
|
|
if (!marketsData || !tokenPrices || !positionsData || !redbankBalances || !denom) return 0
|
2022-11-07 16:36:12 +00:00
|
|
|
|
|
|
|
const hasDebt = positionsData.debts.length > 0
|
|
|
|
|
|
|
|
const borrowTokenPrice = tokenPrices[denom]
|
|
|
|
const borrowTokenMaxLTV = Number(marketsData[denom].max_loan_to_value)
|
|
|
|
|
2022-11-07 17:14:16 +00:00
|
|
|
// approximate debt value in an hour timespan to avoid throwing on smart contract level
|
|
|
|
// due to debt interest being applied
|
2022-11-07 16:36:12 +00:00
|
|
|
const totalLiabilitiesValue = positionsData?.debts.reduce((acc, coin) => {
|
2022-11-07 17:14:16 +00:00
|
|
|
const estimatedInterestAmount = getApproximateHourlyInterest(
|
|
|
|
coin.amount,
|
2022-11-09 09:04:06 +00:00
|
|
|
marketsData[coin.denom].borrow_rate,
|
2022-11-07 17:14:16 +00:00
|
|
|
)
|
|
|
|
const tokenDebtValue = BigNumber(getTokenValue(coin.amount, coin.denom)).plus(
|
2022-11-09 09:04:06 +00:00
|
|
|
estimatedInterestAmount,
|
2022-11-07 17:14:16 +00:00
|
|
|
)
|
2022-11-07 16:36:12 +00:00
|
|
|
|
2022-11-07 17:14:16 +00:00
|
|
|
return tokenDebtValue.plus(acc).toNumber()
|
2022-11-07 16:36:12 +00:00
|
|
|
}, 0)
|
|
|
|
|
|
|
|
const positionsWeightedAverageWithoutAsset = positionsData?.coins.reduce((acc, coin) => {
|
|
|
|
if (coin.denom === denom) return acc
|
|
|
|
|
|
|
|
const tokenWeightedValue = BigNumber(getTokenValue(coin.amount, coin.denom)).times(
|
2022-11-09 09:04:06 +00:00
|
|
|
Number(marketsData[coin.denom].max_loan_to_value),
|
2022-11-07 16:36:12 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
return tokenWeightedValue.plus(acc).toNumber()
|
|
|
|
}, 0)
|
|
|
|
|
|
|
|
const isHealthyAfterFullWithdraw = !hasDebt
|
|
|
|
? true
|
|
|
|
: positionsWeightedAverageWithoutAsset / totalLiabilitiesValue > 1
|
|
|
|
|
|
|
|
let maxAmountCapacity = 0
|
|
|
|
|
|
|
|
if (isHealthyAfterFullWithdraw) {
|
|
|
|
const maxBorrow = BigNumber(positionsWeightedAverageWithoutAsset)
|
|
|
|
.minus(totalLiabilitiesValue)
|
|
|
|
.div(borrowTokenPrice)
|
|
|
|
|
|
|
|
maxAmountCapacity = maxBorrow
|
|
|
|
.plus(tokenAmountInCreditAccount)
|
|
|
|
.decimalPlaces(tokenDecimals)
|
|
|
|
.toNumber()
|
|
|
|
} else {
|
|
|
|
const requiredCollateral = BigNumber(totalLiabilitiesValue)
|
|
|
|
.minus(positionsWeightedAverageWithoutAsset)
|
|
|
|
.dividedBy(borrowTokenPrice * borrowTokenMaxLTV)
|
|
|
|
|
|
|
|
maxAmountCapacity = BigNumber(tokenAmountInCreditAccount)
|
|
|
|
.minus(requiredCollateral)
|
|
|
|
.decimalPlaces(tokenDecimals)
|
|
|
|
.toNumber()
|
2022-11-07 17:14:16 +00:00
|
|
|
|
|
|
|
// required collateral might be greater than the amount in the credit account which will result
|
|
|
|
// in a negative max amount capacity
|
|
|
|
maxAmountCapacity = maxAmountCapacity < 0 ? 0 : maxAmountCapacity
|
2022-11-07 16:36:12 +00:00
|
|
|
}
|
|
|
|
|
2022-11-09 12:47:36 +00:00
|
|
|
const marketLiquidity = redbankBalances?.[denom] ?? 0
|
2022-11-07 16:36:12 +00:00
|
|
|
|
2022-11-09 12:47:36 +00:00
|
|
|
const maxWithdrawAmount = BigNumber(maxAmountCapacity).gt(marketLiquidity)
|
|
|
|
? marketLiquidity
|
|
|
|
: maxAmountCapacity
|
|
|
|
const isMaxAmountHigherThanBalance = BigNumber(maxWithdrawAmount).gt(tokenAmountInCreditAccount)
|
|
|
|
|
|
|
|
if (!borrow && isMaxAmountHigherThanBalance) {
|
2022-11-07 16:36:12 +00:00
|
|
|
return BigNumber(tokenAmountInCreditAccount)
|
|
|
|
.div(10 ** tokenDecimals)
|
|
|
|
.toNumber()
|
|
|
|
}
|
|
|
|
|
2022-11-09 12:47:36 +00:00
|
|
|
return BigNumber(maxWithdrawAmount)
|
2022-11-07 16:36:12 +00:00
|
|
|
.div(10 ** tokenDecimals)
|
|
|
|
.decimalPlaces(tokenDecimals)
|
|
|
|
.toNumber()
|
|
|
|
}, [
|
|
|
|
borrow,
|
|
|
|
denom,
|
|
|
|
getTokenValue,
|
|
|
|
marketsData,
|
|
|
|
positionsData,
|
2022-11-09 12:47:36 +00:00
|
|
|
redbankBalances,
|
2022-11-07 16:36:12 +00:00
|
|
|
tokenAmountInCreditAccount,
|
|
|
|
tokenDecimals,
|
|
|
|
tokenPrices,
|
|
|
|
])
|
|
|
|
|
|
|
|
return maxAmount
|
|
|
|
}
|
|
|
|
|
|
|
|
export default useCalculateMaxWithdrawAmount
|