Skip to content
This repository has been archived by the owner on Nov 22, 2021. It is now read-only.

Commit

Permalink
feat: add Strategy[StrategyCurveUSTVoterProxy] (#87)
Browse files Browse the repository at this point in the history
* feat: add Strategy[StrategyCurveUSTVoterProxy]

* fix: add license[StrategyCurveUSTVoterProxy]

* chore: refactor interfaces and style

Co-authored-by: dmolina79 <dmolina79@users.noreply.github.com>
  • Loading branch information
orbxball and dmolina79 committed Jan 29, 2021
1 parent 4a3bfd4 commit c054321
Showing 1 changed file with 224 additions and 0 deletions.
224 changes: 224 additions & 0 deletions contracts/strategies/StrategyCurveUSTVoterProxy.sol
Original file line number Diff line number Diff line change
@@ -0,0 +1,224 @@
// SPDX-License-Identifier: AGPL-3.0

pragma solidity ^0.5.17;

import "@openzeppelinV2/contracts/token/ERC20/IERC20.sol";
import "@openzeppelinV2/contracts/math/SafeMath.sol";
import "@openzeppelinV2/contracts/utils/Address.sol";
import "@openzeppelinV2/contracts/token/ERC20/SafeERC20.sol";

import "../../interfaces/yearn/IController.sol";
import "../../interfaces/curve/Gauge.sol";
import "../../interfaces/curve/Mintr.sol";
import "../../interfaces/uniswap/Uni.sol";
import "../../interfaces/curve/Curve.sol";
import "../../interfaces/yearn/IVoterProxy.sol";

contract StrategyCurveUSTVoterProxy {
using SafeERC20 for IERC20;
using Address for address;
using SafeMath for uint256;

address public constant want = address(0x94e131324b6054c0D789b190b2dAC504e4361b53);
address public constant crv = address(0xD533a949740bb3306d119CC777fa900bA034cd52);

address public constant curve = address(0xB0a0716841F2Fc03fbA72A891B8Bb13584F52F2d);
address public constant gauge = address(0x3B7020743Bc2A4ca9EaF9D0722d42E20d6935855);
address public constant voter = address(0xF147b8125d2ef93FB6965Db97D6746952a133934);

address public constant uniswap = address(0x7a250d5630B4cF539739dF2C5dAcb4c659F2488D);
address public constant sushiswap = address(0xd9e1cE17f2641f24aE83637ab66a2cca9C378B9F);
address public constant dai = address(0x6B175474E89094C44Da98b954EedeAC495271d0F);
address public constant weth = address(0xC02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc2); // used for crv <> weth <> dai route

uint256 public keepCRV = 1000;
uint256 public treasuryFee = 500;
uint256 public strategistReward = 50;
uint256 public withdrawalFee = 0;
uint256 public constant FEE_DENOMINATOR = 10000;

address public proxy;
address public dex;

address public governance;
address public controller;
address public strategist;
address public keeper;

uint256 public earned; // lifetime strategy earnings denominated in `want` token

event Harvested(uint256 wantEarned, uint256 lifetimeEarned);

constructor(address _controller) public {
governance = msg.sender;
strategist = msg.sender;
keeper = msg.sender;
controller = _controller;
// standardize constructor
proxy = address(0xC17ADf949f524213a540609c386035D7D685B16F);
dex = sushiswap;
}

function getName() external pure returns (string memory) {
return "StrategyCurveUSTVoterProxy";
}

function setStrategist(address _strategist) external {
require(msg.sender == strategist || msg.sender == governance, "!authorized");
strategist = _strategist;
}

function setKeeper(address _keeper) external {
require(msg.sender == strategist || msg.sender == governance, "!authorized");
keeper = _keeper;
}

function setKeepCRV(uint256 _keepCRV) external {
require(msg.sender == governance, "!governance");
keepCRV = _keepCRV;
}

function setWithdrawalFee(uint256 _withdrawalFee) external {
require(msg.sender == governance, "!governance");
withdrawalFee = _withdrawalFee;
}

function setTreasuryFee(uint256 _treasuryFee) external {
require(msg.sender == governance, "!governance");
treasuryFee = _treasuryFee;
}

function setStrategistReward(uint256 _strategistReward) external {
require(msg.sender == governance, "!governance");
strategistReward = _strategistReward;
}

function setProxy(address _proxy) external {
require(msg.sender == governance, "!governance");
proxy = _proxy;
}

function switchDex(bool isUniswap) external {
require(msg.sender == strategist || msg.sender == governance, "!authorized");
if (isUniswap) {
dex = uniswap;
} else {
dex = sushiswap;
}
}

function deposit() public {
uint256 _want = IERC20(want).balanceOf(address(this));
if (_want > 0) {
IERC20(want).safeTransfer(proxy, _want);
IVoterProxy(proxy).deposit(gauge, want);
}
}

// Controller only function for creating additional rewards from dust
function withdraw(IERC20 _asset) external returns (uint256 balance) {
require(msg.sender == controller, "!controller");
require(want != address(_asset), "want");
require(crv != address(_asset), "crv");
require(dai != address(_asset), "dai");
balance = _asset.balanceOf(address(this));
_asset.safeTransfer(controller, balance);
}

// Withdraw partial funds, normally used with a vault withdrawal
function withdraw(uint256 _amount) external {
require(msg.sender == controller, "!controller");
uint256 _balance = IERC20(want).balanceOf(address(this));
if (_balance < _amount) {
_amount = _withdrawSome(_amount.sub(_balance));
_amount = _amount.add(_balance);
}

uint256 _fee = _amount.mul(withdrawalFee).div(FEE_DENOMINATOR);

IERC20(want).safeTransfer(IController(controller).rewards(), _fee);
address _vault = IController(controller).vaults(address(want));
require(_vault != address(0), "!vault"); // additional protection so we don't burn the funds
IERC20(want).safeTransfer(_vault, _amount.sub(_fee));
}

function _withdrawSome(uint256 _amount) internal returns (uint256) {
return IVoterProxy(proxy).withdraw(gauge, want, _amount);
}

// Withdraw all funds, normally used when migrating strategies
function withdrawAll() external returns (uint256 balance) {
require(msg.sender == controller, "!controller");
_withdrawAll();

balance = IERC20(want).balanceOf(address(this));

address _vault = IController(controller).vaults(address(want));
require(_vault != address(0), "!vault"); // additional protection so we don't burn the funds
IERC20(want).safeTransfer(_vault, balance);
}

function _withdrawAll() internal {
IVoterProxy(proxy).withdrawAll(gauge, want);
}

function harvest() public {
require(msg.sender == keeper || msg.sender == strategist || msg.sender == governance, "!keepers");
IVoterProxy(proxy).harvest(gauge);
uint256 _crv = IERC20(crv).balanceOf(address(this));
if (_crv > 0) {
uint256 _keepCRV = _crv.mul(keepCRV).div(FEE_DENOMINATOR);
IERC20(crv).safeTransfer(voter, _keepCRV);
_crv = _crv.sub(_keepCRV);

IERC20(crv).safeApprove(dex, 0);
IERC20(crv).safeApprove(dex, _crv);

address[] memory path = new address[](3);
path[0] = crv;
path[1] = weth;
path[2] = dai;

Uni(dex).swapExactTokensForTokens(_crv, uint256(0), path, address(this), now.add(1800));
}
uint256 _dai = IERC20(dai).balanceOf(address(this));
if (_dai > 0) {
IERC20(dai).safeApprove(curve, 0);
IERC20(dai).safeApprove(curve, _dai);
ICurveFi(curve).add_liquidity([0, _dai, 0, 0], 0);
}
uint256 _want = IERC20(want).balanceOf(address(this));
if (_want > 0) {
uint256 _fee = _want.mul(treasuryFee).div(FEE_DENOMINATOR);
uint256 _reward = _want.mul(strategistReward).div(FEE_DENOMINATOR);
IERC20(want).safeTransfer(IController(controller).rewards(), _fee);
IERC20(want).safeTransfer(strategist, _reward);
deposit();
}
IVoterProxy(proxy).lock();
earned = earned.add(_want);
emit Harvested(_want, earned);
}

function balanceOfWant() public view returns (uint256) {
return IERC20(want).balanceOf(address(this));
}

function balanceOfPool() public view returns (uint256) {
return IVoterProxy(proxy).balanceOf(gauge);
}

function balanceOf() public view returns (uint256) {
return balanceOfWant().add(balanceOfPool());
}

function setGovernance(address _governance) external {
require(msg.sender == governance, "!governance");
governance = _governance;
}

function setController(address _controller) external {
require(msg.sender == governance, "!governance");
controller = _controller;
}
}

0 comments on commit c054321

Please sign in to comment.