import BigNumber from "bignumber.js"; import tokens from "~/constant/tokens"; import { defineStrategy, defineStrategyComponent, StrategyComponentType, StrategyProtocol } from "../../helpers"; export default defineStrategy({ protocol: StrategyProtocol.AAVE_V2, name: "Payback & Withdraw", description: "Payback debt & withdraw collateral in a single txn.", author: "Instadapp Team", submitText: "Payback & Withdraw", details: `

This strategy executes:

`, components: [ defineStrategyComponent({ type: StrategyComponentType.INPUT_WITH_TOKEN, name: "Debt", placeholder: ({ component: input }) => input.token ? `${input.token.symbol} to Payback` : "", validate: ({ component: input, toBN, dsaBalances }) => { if (!input.token) { return "Debt token is required"; } const balance = toBN(dsaBalances[input.token.address]?.balance); if (toBN(balance).lt(input.value)) { return "You don't have enough balance to payback."; } }, defaults: ({ getTokenByKey }) => ({ token: getTokenByKey?.("dai") }) }), defineStrategyComponent({ type: StrategyComponentType.INPUT_WITH_TOKEN, name: "Collateral", placeholder: ({ component: input }) => input.token ? `${input.token.symbol} to Withdraw` : "", validate: ({ component: input, position, toBN }) => { if (!input.token) { return "Collateral token is required"; } if (!input.value) { return "Collateral amount is required"; } if (position) { const collateralPosition = position.data.find( item => item.key === input.token.key ); if (collateralPosition) { const collateralBalance = toBN(collateralPosition.supply); if (collateralBalance.lt(input.value)) { const collateralBalanceFormatted = collateralBalance.toFixed(2); return `Your amount exceeds your maximum limit of ${collateralBalanceFormatted} ${input.token.symbol}`; } } } }, defaults: ({ getTokenByKey }) => ({ token: getTokenByKey?.("eth") }) }), defineStrategyComponent({ type: StrategyComponentType.HEADING, name: "Projected Debt Position" }), defineStrategyComponent({ type: StrategyComponentType.STATUS, name: "Status", update: ({ position, component, components, toBN }) => { if ( toBN(components[0].value).isZero() && toBN(components[1].value).isZero() ) { return; } if (!position) { return; } const newPositionData = changedPositionData(position, components); const stats = calculateStats(newPositionData); component.liquidation = BigNumber.max( toBN(stats.totalMaxLiquidationLimitInEth).div(stats.totalSupplyInEth), "0" ).toFixed(); component.status = BigNumber.max( toBN(stats.totalBorrowInEth).div(stats.totalSupplyInEth), "0" ).toFixed(); } }), defineStrategyComponent({ type: StrategyComponentType.VALUE, name: "LIQUIDATION PRICE (IN ETH)", value: "-", update: ({ position, component, components, toBN, formatting }) => { if (!position) { return; } const newPositionData = changedPositionData(position, components); const initialStats = calculateStats(position.data); const newStats = calculateStats(newPositionData); const stats = toBN(components[0].value).isZero() && toBN(components[1].value).isZero() ? initialStats : newStats; let liquidationPrice = "0"; if (!toBN(stats.ethSupplied).isZero()) { liquidationPrice = BigNumber.max( toBN(stats.totalBorrowInEth) .div(stats.totalMaxLiquidationLimitInEth) .times(position.ethPriceInUsd), "0" ).toFixed(); } component.value = `${formatting.formatUsdMax( liquidationPrice, position.ethPriceInUsd )} / ${formatting.formatUsd(position.ethPriceInUsd)}`; } }) ], validate: async ({ position, components: inputs, toBN }) => { if (toBN(inputs[0].value).isZero() && toBN(inputs[1].value).isZero()) { return; } const newPositionData = changedPositionData(position, inputs); const stats = calculateStats(newPositionData); let maxLiquidation = "0"; if (!toBN(stats.totalSupplyInEth).isZero()) { maxLiquidation = BigNumber.max( toBN(stats.totalMaxLiquidationLimitInEth).div(stats.totalSupplyInEth), "0" ).toFixed(); } const status = BigNumber.max( toBN(stats.totalBorrowInEth).div(stats.totalSupplyInEth), "0" ); if (status.gt(toBN(maxLiquidation).minus("0.0001"))) { return "Position will liquidate."; } }, spells: async ({ components: inputs, convertTokenAmountToWei }) => { return [ { connector: "aave_v2", method: "payback", args: [ inputs[0].token.address, convertTokenAmountToWei(inputs[0].value, inputs[0].token.decimals), 2, 0, 0 ] }, { connector: "aave_v2", method: "withdraw", args: [ inputs[1].token.address, convertTokenAmountToWei(inputs[1].value, inputs[1].token.decimals), 0, 0 ] } ]; } }); const changedPositionData = (position, inputs) => { return position.data.map(position => { const changedPosition = { ...position }; if (inputs[0].token.key === position.key) { changedPosition.borrow = BigNumber.max( new BigNumber(position.borrow).minus(inputs[0].value), "0" ).toFixed(); } if (inputs[1].token.key === position.key) { changedPosition.supply = BigNumber.max( new BigNumber(position.supply).minus(inputs[1].value), "0" ).toFixed(); } return changedPosition; }); }; const calculateStats = positionData => { return positionData.reduce( ( stats, { key, supply, borrow, borrowStable, priceInEth, factor, liquidation } ) => { if (key === "eth") { stats.ethSupplied = supply; } const borrowTotal = new BigNumber(borrow).plus(borrowStable); stats.totalSupplyInEth = new BigNumber(supply) .times(priceInEth) .plus(stats.totalSupplyInEth) .toFixed(); stats.totalBorrowInEth = new BigNumber(borrowTotal) .times(priceInEth) .plus(stats.totalBorrowInEth) .toFixed(); stats.totalMaxBorrowLimitInEth = new BigNumber(priceInEth) .times(factor) .times(supply) .plus(stats.totalMaxBorrowLimitInEth) .toFixed(); stats.totalMaxLiquidationLimitInEth = new BigNumber(priceInEth) .times(liquidation) .times(supply) .plus(stats.totalMaxLiquidationLimitInEth) .toFixed(); return stats; }, { totalSupplyInEth: "0", totalBorrowInEth: "0", totalMaxBorrowLimitInEth: "0", totalMaxLiquidationLimitInEth: "0" } ); };