mirror of
https://github.com/Instadapp/assembly.git
synced 2024-07-29 22:37:06 +00:00
308 lines
8.6 KiB
TypeScript
308 lines
8.6 KiB
TypeScript
import BigNumber from "bignumber.js";
|
|
import {
|
|
defineStrategy,
|
|
defineStrategyComponent,
|
|
StrategyComponentType,
|
|
StrategyProtocol
|
|
} from "../../helpers";
|
|
|
|
export default defineStrategy({
|
|
protocol: StrategyProtocol.COMPOUND,
|
|
name: "Payback & Withdraw",
|
|
description: "Payback debt & withdraw collateral in a single txn.",
|
|
author: "Instadapp Team",
|
|
|
|
submitText: "Payback & Withdraw",
|
|
|
|
details: `<p class="text-center">This strategy executes:</p>
|
|
<ul>
|
|
<li>Payback debt</li>
|
|
<li>Withdraw collateral</li>
|
|
</ul>`,
|
|
|
|
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, tokenIdMapping }) => {
|
|
if (!input.token) {
|
|
return "Collateral token is required";
|
|
}
|
|
|
|
if (!input.value) {
|
|
return "Collateral amount is required";
|
|
}
|
|
|
|
const { tokenToId } = tokenIdMapping;
|
|
|
|
if (position) {
|
|
const collateralBalance = toBN(
|
|
position.data.find(
|
|
pos => pos.cTokenId === tokenToId.compound[input.token.key]
|
|
)?.supply || "0"
|
|
);
|
|
|
|
if (collateralBalance.lt(input.value)) {
|
|
const collateralBalanceFormatted = collateralBalance.toFixed(
|
|
2,
|
|
BigNumber.ROUND_FLOOR
|
|
);
|
|
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, tokenIdMapping }) => {
|
|
if (
|
|
toBN(components[0].value).isZero() &&
|
|
toBN(components[1].value).isZero()
|
|
) {
|
|
return;
|
|
}
|
|
|
|
if (!position) {
|
|
return;
|
|
}
|
|
|
|
const newPositionData = changedPositionData(position, components, tokenIdMapping.tokenToId);
|
|
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, tokenIdMapping }) => {
|
|
if (!position) {
|
|
return;
|
|
}
|
|
|
|
const newPositionData = changedPositionData(position, components, tokenIdMapping.tokenToId);
|
|
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.totalMaxBorrowLimitInEth)
|
|
.times(position.ethPriceInUsd),
|
|
"0"
|
|
).toFixed();
|
|
}
|
|
|
|
component.value = `${formatting.formatUsdMax(
|
|
liquidationPrice,
|
|
position.ethPriceInUsd
|
|
)} / ${formatting.formatUsd(position.ethPriceInUsd)}`;
|
|
}
|
|
})
|
|
],
|
|
|
|
validate: async ({ position, components: inputs, toBN, tokenIdMapping }) => {
|
|
if (toBN(inputs[0].value).isZero() && toBN(inputs[1].value).isZero()) {
|
|
return;
|
|
}
|
|
const { tokenToId } = tokenIdMapping;
|
|
|
|
const newPositionData = position.data.map(position => {
|
|
const changedPosition = { ...position };
|
|
if (tokenToId.compound[inputs[0].token.key] === position.cTokenId) {
|
|
changedPosition.borrow = BigNumber.max(
|
|
toBN(position.borrow).minus(inputs[0].value),
|
|
"0"
|
|
).toFixed();
|
|
}
|
|
|
|
if (tokenToId.compound[inputs[1].token.key] === position.cTokenId) {
|
|
changedPosition.supply = BigNumber.max(
|
|
toBN(position.supply).minus(inputs[1].value),
|
|
"0"
|
|
).toFixed();
|
|
}
|
|
|
|
return changedPosition;
|
|
});
|
|
|
|
const stats = newPositionData.reduce(
|
|
(stats, { key, supply, borrow, priceInEth, factor }) => {
|
|
if (key === "eth") {
|
|
stats.ethSupplied = supply;
|
|
}
|
|
|
|
stats.totalSupplyInEth = toBN(supply)
|
|
.times(priceInEth)
|
|
.plus(stats.totalSupplyInEth)
|
|
.toFixed();
|
|
stats.totalBorrowInEth = toBN(borrow)
|
|
.times(priceInEth)
|
|
.plus(stats.totalBorrowInEth)
|
|
.toFixed();
|
|
|
|
stats.totalMaxBorrowLimitInEth = toBN(priceInEth)
|
|
.times(factor)
|
|
.times(supply)
|
|
.plus(stats.totalMaxBorrowLimitInEth)
|
|
.toFixed();
|
|
|
|
return stats;
|
|
},
|
|
{
|
|
totalSupplyInEth: "0",
|
|
totalBorrowInEth: "0",
|
|
totalMaxBorrowLimitInEth: "0",
|
|
ethSupplied: '0',
|
|
}
|
|
);
|
|
|
|
let liquidation = "0";
|
|
|
|
if (!toBN(stats.totalSupplyInEth).isZero()) {
|
|
liquidation = BigNumber.max(
|
|
toBN(stats.totalMaxBorrowLimitInEth).div(stats.totalSupplyInEth),
|
|
"0"
|
|
).toFixed();
|
|
}
|
|
|
|
const status = BigNumber.max(
|
|
toBN(stats.totalBorrowInEth).div(stats.totalSupplyInEth),
|
|
"0"
|
|
);
|
|
|
|
if (status.gt(toBN(liquidation).minus("0.0001"))) {
|
|
return "Position will liquidate.";
|
|
}
|
|
},
|
|
|
|
spells: async ({ components: inputs, convertTokenAmountToWei, tokenIdMapping }) => {
|
|
const { tokenToId } = tokenIdMapping;
|
|
|
|
const debtTokenId = tokenToId.compound[inputs[0].token.key];
|
|
const collateralTokenId = tokenToId.compound[inputs[1].token.key];
|
|
|
|
return [
|
|
{
|
|
connector: "compound",
|
|
method: "payback",
|
|
args: [
|
|
debtTokenId,
|
|
convertTokenAmountToWei(inputs[0].value, inputs[0].token.decimals),
|
|
0,
|
|
0
|
|
]
|
|
},
|
|
{
|
|
connector: "compound",
|
|
method: "withdraw",
|
|
args: [
|
|
collateralTokenId,
|
|
convertTokenAmountToWei(inputs[1].value, inputs[1].token.decimals),
|
|
0,
|
|
0
|
|
]
|
|
}
|
|
];
|
|
}
|
|
});
|
|
|
|
const changedPositionData = (position, inputs, tokenToId) => {
|
|
return position.data.map(position => {
|
|
const changedPosition = { ...position };
|
|
if (tokenToId.compound[inputs[0].token.key] === position.cTokenId) {
|
|
changedPosition.borrow = BigNumber.max(
|
|
new BigNumber(position.borrow).minus(inputs[0].value),
|
|
"0"
|
|
).toFixed();
|
|
}
|
|
|
|
if (tokenToId.compound[inputs[1].token.key] === position.cTokenId) {
|
|
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, priceInEth, factor }) => {
|
|
if (key === "eth") {
|
|
stats.ethSupplied = supply;
|
|
}
|
|
|
|
stats.totalSupplyInEth = new BigNumber(supply)
|
|
.times(priceInEth)
|
|
.plus(stats.totalSupplyInEth)
|
|
.toFixed();
|
|
stats.totalBorrowInEth = new BigNumber(borrow)
|
|
.times(priceInEth)
|
|
.plus(stats.totalBorrowInEth)
|
|
.toFixed();
|
|
|
|
stats.totalMaxBorrowLimitInEth = new BigNumber(priceInEth)
|
|
.times(factor)
|
|
.times(supply)
|
|
.plus(stats.totalMaxBorrowLimitInEth)
|
|
.toFixed();
|
|
|
|
return stats;
|
|
},
|
|
{
|
|
totalSupplyInEth: "0",
|
|
totalBorrowInEth: "0",
|
|
totalMaxBorrowLimitInEth: "0",
|
|
ethSupplied: "0"
|
|
}
|
|
);
|
|
};
|