dsa-connectors/contracts/mainnet/connectors/1inch/1inch-v3/main.sol

129 lines
4.7 KiB
Solidity
Raw Normal View History

2021-02-07 16:40:28 +00:00
pragma solidity ^0.7.0;
pragma experimental ABIEncoderV2;
/**
2021-12-22 19:59:20 +00:00
* @title 1InchV3.
* @dev On-chain DEX Aggregator.
*/
// import files from common directory
2021-12-24 12:48:06 +00:00
import { TokenInterface , MemoryInterface } from "../../../common/interfaces.sol";
import { Stores } from "../../../common/stores.sol";
2021-03-21 09:13:05 +00:00
import { OneInchInterace, OneInchData } from "./interface.sol";
import { Helpers } from "./helpers.sol";
import { Events } from "./events.sol";
2021-03-21 09:13:05 +00:00
abstract contract OneInchResolver is Helpers, Events {
2020-11-29 14:26:28 +00:00
/**
* @dev 1inch swap uses `.call()`. This function restrict it to call only swap/trade functionality
* @param callData - calldata to extract the first 4 bytes for checking function signature
*/
function checkOneInchSig(bytes memory callData) internal pure returns(bool isOk) {
bytes memory _data = callData;
bytes4 sig;
// solium-disable-next-line security/no-inline-assembly
assembly {
sig := mload(add(_data, 32))
}
2021-05-26 01:33:45 +00:00
isOk = sig == oneInchSwapSig || sig == oneInchUnoswapSig;
}
2020-11-29 14:26:28 +00:00
/**
* @dev 1inch API swap handler
* @param oneInchData - contains data returned from 1inch API. Struct defined in interfaces.sol
* @param ethAmt - Eth to swap for .value()
*/
function oneInchSwap(
OneInchData memory oneInchData,
uint ethAmt
) internal returns (uint buyAmt) {
TokenInterface buyToken = oneInchData.buyToken;
(uint _buyDec, uint _sellDec) = getTokensDec(buyToken, oneInchData.sellToken);
uint _sellAmt18 = convertTo18(_sellDec, oneInchData._sellAmt);
uint _slippageAmt = convert18ToDec(_buyDec, wmul(oneInchData.unitAmt, _sellAmt18));
uint initalBal = getTokenBal(buyToken);
// solium-disable-next-line security/no-call-value
2021-03-21 09:13:05 +00:00
(bool success, ) = oneInchAddr.call{value: ethAmt}(oneInchData.callData);
if (!success) revert("1Inch-swap-failed");
uint finalBal = getTokenBal(buyToken);
buyAmt = sub(finalBal, initalBal);
require(_slippageAmt <= buyAmt, "Too much slippage");
}
}
2021-03-21 09:13:05 +00:00
abstract contract OneInchResolverHelpers is OneInchResolver {
2020-11-29 14:26:28 +00:00
/**
* @dev Gets the swapping data from 1inch's API.
* @param oneInchData Struct with multiple swap data defined in interfaces.sol
* @param setId Set token amount at this ID in `InstaMemory` Contract.
*/
2021-03-21 09:13:05 +00:00
function _sell(
OneInchData memory oneInchData,
uint setId
2021-03-15 17:22:47 +00:00
) internal returns (OneInchData memory) {
TokenInterface _sellAddr = oneInchData.sellToken;
uint ethAmt;
2021-02-07 17:33:48 +00:00
if (address(_sellAddr) == ethAddr) {
ethAmt = oneInchData._sellAmt;
} else {
approve(TokenInterface(_sellAddr), oneInchAddr, oneInchData._sellAmt);
}
require(checkOneInchSig(oneInchData.callData), "Not-swap-function");
oneInchData._buyAmt = oneInchSwap(oneInchData, ethAmt);
setUint(setId, oneInchData._buyAmt);
2021-03-15 17:22:47 +00:00
return oneInchData;
// emitLogSellThree(oneInchData, setId);
}
}
2021-03-21 09:13:05 +00:00
abstract contract OneInch is OneInchResolverHelpers {
/**
2021-03-23 10:32:30 +00:00
* @dev Sell ETH/ERC20_Token using 1Inch.
* @notice Swap tokens from exchanges like kyber, 0x etc, with calculation done off-chain.
* @param buyAddr The address of the token to buy.(For ETH: 0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE)
* @param sellAddr The address of the token to sell.(For ETH: 0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE)
* @param sellAmt The amount of the token to sell.
* @param unitAmt The amount of buyAmt/sellAmt with slippage.
* @param callData Data from 1inch API. You can generate calldata for calling 1inch route for exchange: <a href="https://api.1inch.exchange/swagger/ethereum/#/Swap/SwapFactoryCommonController_getSwap" target="_blank">here </a>
2021-03-23 10:32:30 +00:00
* @param setId ID stores the amount of token brought.
*/
2021-03-21 09:13:05 +00:00
function sell(
address buyAddr,
address sellAddr,
uint sellAmt,
uint unitAmt,
bytes calldata callData,
uint setId
2021-03-15 17:22:47 +00:00
) external payable returns (string memory _eventName, bytes memory _eventParam) {
OneInchData memory oneInchData = OneInchData({
buyToken: TokenInterface(buyAddr),
sellToken: TokenInterface(sellAddr),
unitAmt: unitAmt,
callData: callData,
_sellAmt: sellAmt,
_buyAmt: 0
});
2021-03-21 09:13:05 +00:00
oneInchData = _sell(oneInchData, setId);
2021-03-15 17:22:47 +00:00
2021-03-21 09:13:05 +00:00
_eventName = "LogSell(address,address,uint256,uint256,uint256,uint256)";
2021-03-15 17:22:47 +00:00
_eventParam = abi.encode(buyAddr, sellAddr, oneInchData._buyAmt, oneInchData._sellAmt, 0, setId);
}
}
2021-12-22 19:59:20 +00:00
contract ConnectV2OneInchV3 is OneInch {
2021-06-25 17:19:14 +00:00
string public name = "1Inch-v1.2";
}