-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathCrvDepositorWrapper.sol
142 lines (123 loc) · 4.73 KB
/
CrvDepositorWrapper.sol
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
// SPDX-License-Identifier: MIT
pragma solidity 0.8.11;
import { IERC20 } from "@openzeppelin/contracts-0.8/token/ERC20/IERC20.sol";
import { SafeERC20 } from "@openzeppelin/contracts-0.8/token/ERC20/utils/SafeERC20.sol";
import { IVault, IPriceOracle, ICrvDepositorWrapper, IAsset } from "./Interfaces.sol";
import { IVault } from "./Interfaces.sol";
interface ICrvDepositor {
function depositFor(
address to,
uint256 _amount,
bool _lock,
address _stakeAddress
) external;
}
/**
* @title BalInvestor
* @notice Deposits $BAL into a BAL/WETH BPT. Hooks into TWAP to determine minOut.
* @dev Abstract contract for depositing BAL -> balBPT -> auraBAL via crvDepositor
*/
abstract contract BalInvestor {
using SafeERC20 for IERC20;
IVault public immutable BALANCER_VAULT;
address public immutable BAL;
address public immutable WETH;
address public immutable BALANCER_POOL_TOKEN;
bytes32 public immutable BAL_ETH_POOL_ID;
constructor(
IVault _balancerVault,
address _bal,
address _weth,
bytes32 _balETHPoolId
) {
(
address poolAddress, /* */
) = _balancerVault.getPool(_balETHPoolId);
require(poolAddress != address(0), "!poolAddress");
BALANCER_VAULT = _balancerVault;
BAL = _bal;
WETH = _weth;
BALANCER_POOL_TOKEN = poolAddress;
BAL_ETH_POOL_ID = _balETHPoolId;
}
function _setApprovals() internal {
IERC20(WETH).safeApprove(address(BALANCER_VAULT), type(uint256).max);
IERC20(BAL).safeApprove(address(BALANCER_VAULT), type(uint256).max);
}
function _getBptPrice() internal view returns (uint256) {
IPriceOracle.OracleAverageQuery[] memory queries = new IPriceOracle.OracleAverageQuery[](1);
queries[0].variable = IPriceOracle.Variable.BPT_PRICE;
queries[0].secs = 3600; // last hour
queries[0].ago = 0; // now
// Gets the balancer time weighted average price denominated in BAL
return IPriceOracle(BALANCER_POOL_TOKEN).getTimeWeightedAverage(queries)[0];
}
function _getMinOut(uint256 amount, uint256 minOutBps) internal view returns (uint256) {
// Gets the balancer time weighted average price denominated in BAL
// e.g. if 1 BAL == 0.4 BPT, bptOraclePrice == 2.5
uint256 bptOraclePrice = _getBptPrice();
// e.g. minOut = (((100e18 * 1e18) / 2.5e18) * 9980) / 10000;
// e.g. minout = 39.92e18
uint256 minOut = (((amount * 1e18) / bptOraclePrice) * minOutBps) / 10000;
return minOut;
}
function _investBalToPool(uint256 amount, uint256 minOut) internal {
IERC20(BAL).safeTransferFrom(msg.sender, address(this), amount);
IAsset[] memory assets = new IAsset[](2);
assets[0] = IAsset(BAL);
assets[1] = IAsset(WETH);
uint256[] memory maxAmountsIn = new uint256[](2);
maxAmountsIn[0] = amount;
maxAmountsIn[1] = 0;
BALANCER_VAULT.joinPool(
BAL_ETH_POOL_ID,
address(this),
address(this),
IVault.JoinPoolRequest(
assets,
maxAmountsIn,
abi.encode(IVault.JoinKind.EXACT_TOKENS_IN_FOR_BPT_OUT, maxAmountsIn, minOut),
false // Don't use internal balances
)
);
}
}
/**
* @title CrvDepositorWrapper
* @notice Converts BAL -> balBPT and then wraps to auraBAL via the crvDepositor
*/
contract CrvDepositorWrapper is ICrvDepositorWrapper, BalInvestor {
address public immutable crvDeposit;
constructor(
address _crvDeposit,
IVault _balancerVault,
address _bal,
address _weth,
bytes32 _balETHPoolId
) BalInvestor(_balancerVault, _bal, _weth, _balETHPoolId) {
crvDeposit = _crvDeposit;
}
function setApprovals() external {
_setApprovals();
require(IERC20(BALANCER_POOL_TOKEN).approve(crvDeposit, type(uint256).max), "!approval");
}
/**
* @dev Gets minimum output based on BPT oracle price
* @param _amount Units of BAL to deposit
* @param _outputBps Multiplier where 100% == 10000, 99.5% == 9950 and 98% == 9800
* @return minOut Units of BPT to expect as output
*/
function getMinOut(uint256 _amount, uint256 _outputBps) external view returns (uint256) {
return _getMinOut(_amount, _outputBps);
}
function deposit(
uint256 _amount,
uint256 _minOut,
bool _lock,
address _stakeAddress
) external {
_investBalToPool(_amount, _minOut);
uint256 bptBalance = IERC20(BALANCER_POOL_TOKEN).balanceOf(address(this));
ICrvDepositor(crvDeposit).depositFor(msg.sender, bptBalance, _lock, _stakeAddress);
}
}