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[StrategyCurveCompoundVoterProxy] (#63)
Browse files Browse the repository at this point in the history
https://etherscan.io/address/0x530da5aeF3c8f9CCbc75C97C182D6ee2284B643F#code

Signed-off-by: Andre Cronje <andre.cronje@yearn.finance>
  • Loading branch information
andrecronje committed Nov 9, 2020
1 parent f83429c commit b21d961
Showing 1 changed file with 202 additions and 0 deletions.
202 changes: 202 additions & 0 deletions contracts/strategies/StrategyCurveCompoundVoterProxy.sol
Original file line number Diff line number Diff line change
@@ -0,0 +1,202 @@
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/IToken.sol";
import "../../interfaces/yearn/IVoterProxy.sol";

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

address public constant want = address(0x845838DF265Dcd2c412A1Dc9e959c7d08537f8a2);
address public constant crv = address(0xD533a949740bb3306d119CC777fa900bA034cd52);
address public constant uni = address(0x7a250d5630B4cF539739dF2C5dAcb4c659F2488D);
address public constant weth = address(0xC02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc2); // used for crv <> weth <> dai route

address public constant dai = address(0x6B175474E89094C44Da98b954EedeAC495271d0F);
address public constant curve = address(0xA2B47E3D5c44877cca798226B7B8118F9BFb7A56);

address public constant gauge = address(0x7ca5b0a2910B33e9759DC7dDB0413949071D7575);
address public constant voter = address(0xF147b8125d2ef93FB6965Db97D6746952a133934);

uint256 public keepCRV = 1500;
uint256 public performanceFee = 450;
uint256 public strategistReward = 50;
uint256 public withdrawalFee = 0;
uint256 public constant FEE_DENOMINATOR = 10000;

address public proxy;

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

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;
controller = _controller;
}

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

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

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 setPerformanceFee(uint256 _performanceFee) external {
require(msg.sender == governance, "!governance");
performanceFee = _performanceFee;
}

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 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 == strategist || msg.sender == governance, "!authorized");
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(uni, 0);
IERC20(crv).safeApprove(uni, _crv);

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

Uni(uni).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([_dai, 0, 0], 0);
}
uint256 _want = IERC20(want).balanceOf(address(this));
if (_want > 0) {
uint256 _fee = _want.mul(performanceFee).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 b21d961

Please sign in to comment.