diff --git a/contracts/VoltMakerV4.sol b/contracts/VoltMakerV4.sol new file mode 100644 index 0000000..fb00bdf --- /dev/null +++ b/contracts/VoltMakerV4.sol @@ -0,0 +1,232 @@ +// SPDX-License-Identifier: MIT + +// P1 - P3: OK +pragma solidity 0.6.12; + +import "./libraries/SafeMath.sol"; +import "./libraries/SafeERC20.sol"; + +import "./uniswap/interfaces/IUniswapV2ERC20.sol"; +import "./uniswap/interfaces/IUniswapV2Pair.sol"; +import "./uniswap/interfaces/IUniswapV2Factory.sol"; + +import "./boringcrypto/BoringOwnable.sol"; + +// VoltMakerV2 is MasterChefVolt's left hand and kinda a wizard. He can cook up Joe from pretty much anything! +// This contract handles "serving up" and "buring" rewards for veVOLT holders by trading tokens collected from fees for Volt. + +// T1 - T4: OK +contract VoltMakerV4 is BoringOwnable { + using SafeMath for uint256; + using SafeERC20 for IERC20; + + /* ========== CONSTANT VARIABLES ========== */ + address public constant burn = 0x000000000000000000000000000000000000dEaD; + + IUniswapV2Factory public immutable factory; + address public immutable feeDistributor; + address private immutable volt; + address private immutable wfuse; + + mapping(address => address) internal _bridges; + + event LogBridgeSet(address indexed token, address indexed bridge); + + event LogConvert( + address indexed server, + address indexed token0, + address indexed token1, + uint256 amount0, + uint256 amount1, + uint256 amountVOLT + ); + + event FeesDistributed(uint256 amount); + + event FeesBurned(uint256 amount); + + /* ========== CONSTRUCTOR ========== */ + + constructor( + address _factory, + address _feeDistributor, + address _volt, + address _wfuse + ) public { + factory = IUniswapV2Factory(_factory); + feeDistributor = _feeDistributor; + volt = _volt; + wfuse = _wfuse; + } + + /* ========== External Functions ========== */ + + // F3: _convert is separate to save gas by only checking the 'onlyEOA' modifier once in case of convertMultiple + // F6: There is an exploit to add lots of VOLT to the feeDistributor, run convert, then remove the VOLT again. + // As the size of the VoltBar has grown, this requires large amounts of funds and isn't super profitable anymore + // The onlyEOA modifier prevents this being done with a flash loan. + function convert(address token0, address token1) external onlyOwner { + _convert(token0, token1); + } + + function convertMultiple(address[] calldata token0, address[] calldata token1) external onlyOwner { + // TODO: This can be optimized a fair bit, but this is safer and simpler for now + uint256 len = token0.length; + for (uint256 i = 0; i < len; i++) { + _convert(token0[i], token1[i]); + } + } + + /* ========== Public Functions ========== */ + + function bridgeFor(address token) public view returns (address bridge) { + bridge = _bridges[token]; + if (bridge == address(0)) { + bridge = wfuse; + } + } + + /* ========== Internal Functions ========== */ + + function _convert(address token0, address token1) internal { + // Interactions + // S1 - S4: OK + IUniswapV2Pair pair = IUniswapV2Pair(factory.getPair(token0, token1)); + require(address(pair) != address(0), "VoltMakerV2: Invalid pair"); + // balanceOf: S1 - S4: OK + // transfer: X1 - X5: OK + IERC20(address(pair)).safeTransfer(address(pair), pair.balanceOf(address(this))); + + // X1 - X5: OK + // We don't take amount0 and amount1 from here, as it won't take into account reflect tokens. + pair.burn(address(this)); + + // We get the amount0 and amount1 by their respective balance of VoltMakerV2. + uint256 amount0 = IERC20(token0).balanceOf(address(this)); + uint256 amount1 = IERC20(token1).balanceOf(address(this)); + + emit LogConvert(msg.sender, token0, token1, amount0, amount1, _convertStep(token0, token1, amount0, amount1)); + } + + function _convertStep( + address token0, + address token1, + uint256 amount0, + uint256 amount1 + ) internal returns (uint256 voltOut) { + // Interactions + if (token0 == token1) { + uint256 amount = amount0.add(amount1); + if (token0 == volt) { + _distributeFeesAndBurn(amount); + voltOut = amount; + } else if (token0 == wfuse) { + voltOut = _toVOLT(wfuse, amount); + } else { + address bridge = bridgeFor(token0); + amount = _swap(token0, bridge, amount, address(this)); + voltOut = _convertStep(bridge, bridge, amount, 0); + } + } else if (token0 == volt) { + // eg. JOE - AVAX + _distributeFeesAndBurn(amount0); + voltOut = _toVOLT(token1, amount1).add(amount0); + } else if (token1 == volt) { + // eg. USDT - JOE + _distributeFeesAndBurn(amount1); + voltOut = _toVOLT(token0, amount0).add(amount1); + } else if (token0 == wfuse) { + // eg. AVAX - USDC + voltOut = _toVOLT(wfuse, _swap(token1, wfuse, amount1, address(this)).add(amount0)); + } else if (token1 == wfuse) { + // eg. USDT - AVAX + voltOut = _toVOLT(wfuse, _swap(token0, wfuse, amount0, address(this)).add(amount1)); + } else { + // eg. MIC - USDT + address bridge0 = bridgeFor(token0); + address bridge1 = bridgeFor(token1); + if (bridge0 == token1) { + // eg. MIC - USDT - and bridgeFor(MIC) = USDT + voltOut = _convertStep(bridge0, token1, _swap(token0, bridge0, amount0, address(this)), amount1); + } else if (bridge1 == token0) { + // eg. WBTC - DSD - and bridgeFor(DSD) = WBTC + voltOut = _convertStep(token0, bridge1, amount0, _swap(token1, bridge1, amount1, address(this))); + } else { + voltOut = _convertStep( + bridge0, + bridge1, // eg. USDT - DSD - and bridgeFor(DSD) = WBTC + _swap(token0, bridge0, amount0, address(this)), + _swap(token1, bridge1, amount1, address(this)) + ); + } + } + } + + function _swap( + address fromToken, + address toToken, + uint256 amountIn, + address to + ) internal returns (uint256 realAmountOut) { + // Checks + // X1 - X5: OK + IUniswapV2Pair pair = IUniswapV2Pair(factory.getPair(fromToken, toToken)); + require(address(pair) != address(0), "VoltMakerV2: Cannot convert"); + + // Interactions + // X1 - X5: OK + (uint256 reserve0, uint256 reserve1, ) = pair.getReserves(); + + IERC20(fromToken).safeTransfer(address(pair), amountIn); + + // Added in case fromToken is a reflect token. + if (fromToken == pair.token0()) { + amountIn = IERC20(fromToken).balanceOf(address(pair)) - reserve0; + } else { + amountIn = IERC20(fromToken).balanceOf(address(pair)) - reserve1; + } + + uint256 balanceBefore = IERC20(toToken).balanceOf(to); + + uint256 amountInWithFee = amountIn.mul(997); + if (fromToken == pair.token0()) { + uint256 amountOut = amountInWithFee.mul(reserve1) / reserve0.mul(1000).add(amountInWithFee); + pair.swap(0, amountOut, to, new bytes(0)); + // TODO: Add maximum slippage? + } else { + uint256 amountOut = amountInWithFee.mul(reserve0) / reserve1.mul(1000).add(amountInWithFee); + pair.swap(amountOut, 0, to, new bytes(0)); + // TODO: Add maximum slippage? + } + + realAmountOut = IERC20(toToken).balanceOf(to) - balanceBefore; + } + + function _toVOLT(address token, uint256 amountIn) internal returns (uint256 amountOut) { + // X1 - X5: OK + amountOut = _swap(token, volt, amountIn, feeDistributor); + } + + /* ========== RESTRICTED FUNCTIONS ========== */ + + function setBridge(address token, address bridge) external onlyOwner { + // Checks + require(token != volt && token != wfuse && token != bridge, "VoltMakerV2: Invalid bridge"); + + // Effects + _bridges[token] = bridge; + emit LogBridgeSet(token, bridge); + } + + function _distributeFeesAndBurn(uint256 amountIn) internal returns () { + uint256 burnAmount = amountIn.mul(50).div(100); + uint256 feeAmount = amountIn.sub(burnAmount); + + IERC20(volt).safeTransfer(feeDistributor, feeAmount); + IERC20(volt).safeTransfer(burn, burnAmount); + + emit FeesDistributed(feeAmount); + + emit FeesBurned(burnAmount); + } +}