mirror of
https://github.com/Instadapp/dsa-connectors.git
synced 2024-07-29 22:37:00 +00:00
161 lines
5.0 KiB
Solidity
161 lines
5.0 KiB
Solidity
//SPDX-License-Identifier: MIT
|
|
pragma solidity ^0.7.0;
|
|
pragma experimental ABIEncoderV2;
|
|
|
|
import { protocolHelpers } from "../helpers.sol";
|
|
|
|
import {
|
|
ComptrollerInterface,
|
|
CTokenInterface,
|
|
CompoundMappingInterface,
|
|
CETHInterface
|
|
} from "../interfaces.sol";
|
|
|
|
import { TokenInterface } from "../../../common/interfaces.sol";
|
|
|
|
|
|
|
|
contract CompoundHelpers is protocolHelpers {
|
|
|
|
struct CompoundBorrowData {
|
|
uint length;
|
|
uint fee;
|
|
Protocol target;
|
|
CTokenInterface[] ctokens;
|
|
TokenInterface[] tokens;
|
|
uint[] amts;
|
|
uint[] rateModes;
|
|
}
|
|
|
|
function _compEnterMarkets(uint length, CTokenInterface[] memory ctokens) internal {
|
|
ComptrollerInterface troller = ComptrollerInterface(getComptrollerAddress);
|
|
address[] memory _cTokens = new address[](length);
|
|
|
|
for (uint i = 0; i < length; i++) {
|
|
_cTokens[i] = address(ctokens[i]);
|
|
}
|
|
troller.enterMarkets(_cTokens);
|
|
}
|
|
|
|
function _compBorrowOne(
|
|
uint fee,
|
|
CTokenInterface ctoken,
|
|
TokenInterface token,
|
|
uint amt,
|
|
Protocol target,
|
|
uint rateMode
|
|
) internal returns (uint) {
|
|
if (amt > 0) {
|
|
address _token = address(token) == wethAddr ? ethAddr : address(token);
|
|
|
|
if (amt == uint(-1)) {
|
|
amt = getMaxBorrow(target, address(token), ctoken, rateMode);
|
|
}
|
|
|
|
(uint feeAmt, uint _amt) = calculateFee(amt, fee, true);
|
|
|
|
require(ctoken.borrow(_amt) == 0, "borrow-failed-collateral?");
|
|
transferFees(_token, feeAmt);
|
|
}
|
|
return amt;
|
|
}
|
|
|
|
function _compBorrow(
|
|
CompoundBorrowData memory data
|
|
) internal returns (uint[] memory) {
|
|
uint[] memory finalAmts = new uint[](data.length);
|
|
for (uint i = 0; i < data.length; i++) {
|
|
finalAmts[i] = _compBorrowOne(
|
|
data.fee,
|
|
data.ctokens[i],
|
|
data.tokens[i],
|
|
data.amts[i],
|
|
data.target,
|
|
data.rateModes[i]
|
|
);
|
|
}
|
|
return finalAmts;
|
|
}
|
|
|
|
function _compDepositOne(uint fee, CTokenInterface ctoken, TokenInterface token, uint amt) internal {
|
|
if (amt > 0) {
|
|
address _token = address(token) == wethAddr ? ethAddr : address(token);
|
|
|
|
(uint feeAmt, uint _amt) = calculateFee(amt, fee, false);
|
|
|
|
if (_token != ethAddr) {
|
|
approve(token, address(ctoken), _amt);
|
|
require(ctoken.mint(_amt) == 0, "deposit-failed");
|
|
} else {
|
|
CETHInterface(address(ctoken)).mint{value:_amt}();
|
|
}
|
|
transferFees(_token, feeAmt);
|
|
}
|
|
}
|
|
|
|
function _compDeposit(
|
|
uint length,
|
|
uint fee,
|
|
CTokenInterface[] memory ctokens,
|
|
TokenInterface[] memory tokens,
|
|
uint[] memory amts
|
|
) internal {
|
|
for (uint i = 0; i < length; i++) {
|
|
_compDepositOne(fee, ctokens[i], tokens[i], amts[i]);
|
|
}
|
|
}
|
|
|
|
function _compWithdrawOne(CTokenInterface ctoken, TokenInterface token, uint amt) internal returns (uint) {
|
|
if (amt > 0) {
|
|
if (amt == uint(-1)) {
|
|
bool isEth = address(token) == wethAddr;
|
|
uint initalBal = isEth ? address(this).balance : token.balanceOf(address(this));
|
|
require(ctoken.redeem(ctoken.balanceOf(address(this))) == 0, "withdraw-failed");
|
|
uint finalBal = isEth ? address(this).balance : token.balanceOf(address(this));
|
|
amt = sub(finalBal, initalBal);
|
|
} else {
|
|
require(ctoken.redeemUnderlying(amt) == 0, "withdraw-failed");
|
|
}
|
|
}
|
|
return amt;
|
|
}
|
|
|
|
function _compWithdraw(
|
|
uint length,
|
|
CTokenInterface[] memory ctokens,
|
|
TokenInterface[] memory tokens,
|
|
uint[] memory amts
|
|
) internal returns(uint[] memory) {
|
|
uint[] memory finalAmts = new uint[](length);
|
|
for (uint i = 0; i < length; i++) {
|
|
finalAmts[i] = _compWithdrawOne(ctokens[i], tokens[i], amts[i]);
|
|
}
|
|
return finalAmts;
|
|
}
|
|
|
|
function _compPaybackOne(CTokenInterface ctoken, TokenInterface token, uint amt) internal returns (uint) {
|
|
if (amt > 0) {
|
|
if (amt == uint(-1)) {
|
|
amt = ctoken.borrowBalanceCurrent(address(this));
|
|
}
|
|
if (address(token) != wethAddr) {
|
|
approve(token, address(ctoken), amt);
|
|
require(ctoken.repayBorrow(amt) == 0, "repay-failed.");
|
|
} else {
|
|
CETHInterface(address(ctoken)).repayBorrow{value:amt}();
|
|
}
|
|
}
|
|
return amt;
|
|
}
|
|
|
|
function _compPayback(
|
|
uint length,
|
|
CTokenInterface[] memory ctokens,
|
|
TokenInterface[] memory tokens,
|
|
uint[] memory amts
|
|
) internal {
|
|
for (uint i = 0; i < length; i++) {
|
|
_compPaybackOne(ctokens[i], tokens[i], amts[i]);
|
|
}
|
|
}
|
|
} |