-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathFlashBorrowerMock.sol
156 lines (143 loc) · 6.49 KB
/
FlashBorrowerMock.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
143
144
145
146
147
148
149
150
151
152
153
154
155
156
// SPDX-License-Identifier: BUSL-1.1
pragma solidity 0.8.4;
import '@openzeppelin/contracts/token/ERC20/IERC20.sol';
import "@openzeppelin/contracts/token/ERC20/IERC20.sol";
import "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol";
import "@uniswap/v3-core/contracts/interfaces/IUniswapV3Pool.sol";
import "@uniswap/v3-periphery/contracts/interfaces/ISwapRouter.sol";
import './../interfaces/IERC3156FlashLender.sol';
import './../interfaces/IERC3156FlashBorrower.sol';
import './../Vault.sol';
import './../VaultFactory.sol';
contract FlashBorrowerMock {
enum Action {NORMAL, OTHER}
IERC3156FlashLender public lender;
VaultFactory vaultFactory;
constructor(IERC3156FlashLender _lender, VaultFactory _vaultFactory) public {
lender = _lender;
vaultFactory = _vaultFactory;
}
using SafeERC20 for IERC20;
address public constant WETH_ADDRESS = 0xc778417E063141139Fce010982780140Aa0cD5Ab;
address public constant USDC_ADDRESS = 0x2791Bca1f2de4661ED88A30C99A7a9449Aa84174;
address public constant DAI_ADDRESS = 0x8f3Cf7ad23Cd3CaDbD9735AFf958023239c6A063;
address public constant UNI_V3_ROUTER_ADDRESS = 0xE592427A0AEce92De3Edee1F18E0157C05861564;
address public constant UNI_V3_FACTORY_ADDRESS = 0x1F98431c8aD98523631AE4a59f267346ea31F984;
IERC3156FlashBorrower public constant flashLoan = IERC3156FlashBorrower(0x0d4A11d5EEaaC28EC3F61d100daF4d40471f1852);
ISwapRouter public constant swapRouter = ISwapRouter(UNI_V3_ROUTER_ADDRESS);
function executeOperation() external {
// Step 1: Add liquidity to Uniswap V3
/// @notice Increases liquidity in the current range
/// @dev Pool must be initialized already to add liquidity
/// @param tokenId The id of the erc721 token
/// @param amount0 The amount to add of token0
/// @param amount1 The amount to add of token1
// Step 2: Swap tokens in Uniswap V3
uint256 amountIn = 1 ether;
uint256 amountOutMin = 1000000;
uint160 sqrtPriceLimitX96Swap = 0;
bytes memory path = abi.encodePacked(WETH_ADDRESS, USDC_ADDRESS);
IERC20(WETH_ADDRESS).safeApprove(UNI_V3_ROUTER_ADDRESS, amountIn);
swapRouter.exactInputSingle(
ISwapRouter.ExactInputSingleParams({
tokenIn: WETH_ADDRESS,
tokenOut: USDC_ADDRESS,
fee: fee,
recipient: address(this),
deadline: block.timestamp + 1 hours,
amountIn: amountIn,
amountOutMinimum: amountOutMin,
sqrtPriceLimitX96: sqrtPriceLimitX96Swap
})
);
// Step 3: Remove liquidity from Uniswap V3
uint128 liquidityToRemove = uint128(liquidity);
uint256 amount0MinRemove = 0;
uint256 amount1MinRemove = 0;
IERC20(pool.token0()).safeApprove(UNI_V3_ROUTER_ADDRESS, liquidityToRemove);
IERC20(pool.token1()).safeApprove(UNI_V3_ROUTER_ADDRESS, liquidityToRemove);
(uint256 amount0Remove, uint256 amount1Remove) = swapRouter.removeLiquidity(
WETH_ADDRESS,
USDC_ADDRESS,
liquidityToRemove,
amount0MinRemove,
amount1MinRemove,
address(this),
block.timestamp + 1 hours
);
require(amount0Remove >= amount0MinRemove, "INSUFFICIENT_AMOUNT0");
require(amount1Remove >= amount1MinRemove, "INSUFFICIENT_AMOUNT1");
// Step 4: Pay back the flashloan
IERC20(USDC_ADDRESS).safeApprove(address(flashLoan), 1000000);
flashLoan.repay(USDC_ADDRESS, 1000000);
// Step 5: Transfer any remaining tokens back to the sender
IERC20(WETH_ADDRESS).safeTransfer(msg.sender, IERC20(WETH_ADDRESS).balanceOf(address(this)));
IERC20(USDC_ADDRESS).safeTransfer(msg.sender, IERC20(USDC_ADDRESS).balanceOf(address(this)));
}
/// @dev ERC-3156 Flash loan callback
function onFlashLoan(
address token,
uint256 amount,
uint256 fee,
address _initiator,
address[] calldata _assets,
uint256[] calldata _amounts,
uint256[] calldata _premiums,
address _originator,
bytes calldata _params
) external returns (bytes32) {
require(_assets.length == 1, "INVALID_ASSETS_LENGTH");
require(_amounts.length == 1, "INVALID_AMOUNTS_LENGTH");
require(_premiums.length == 1, "INVALID_PREMIUMS_LENGTH");
require(_assets[0] == USDC_ADDRESS, "INVALID_ASSET");
require(_amounts[0] == 1000000, "INVALID_AMOUNT");
require(_premiums[0] == 0, "INVALID_PREMIUM");
require(_originator == address(this), "INVALID_ORIGINATOR");
require(_params.length == 0, "INVALID_PARAMS");
return keccak256("executeOperation()");
}
/// @dev Initiate a flash loan
//function flashBorrow(address token, uint256 amount) public {
// bytes memory data = abi.encode(Action.NORMAL);
// IERC20 stakedToken = IERC20(token);
// uint256 _allowance = stakedToken.allowance(address(this), address(lender));
// uint256 _fee = lender.flashFee(token, amount);
// uint256 _repayment = amount + _fee;
// stakedToken.approve(address(lender), _allowance + _repayment);
// lender.flashLoan(this, token, amount, data);
//}
function startFlashLoan() external {
address[] memory assets = new address[](1);
assets[0] = USDC_ADDRESS;
uint256[] memory amounts = new uint256[](1);
amounts[0] = 1000000; // 1 USDC
uint256[] memory modes = new uint256[](1);
modes[0] = 0; // no debt limit
address receiverAddress = address(this);
bytes memory params = abi.encodeWithSignature("executeOperation()");
flashLoan.flashLoan(receiverAddress, assets, amounts, modes, address(this), params, 0);
}
function increaseLiquidityCurrentRange(
uint256 tokenId,
uint256 amountAdd0,
uint256 amountAdd1
)
external
returns (
uint128 liquidity,
uint256 amount0,
uint256 amount1
)
{
INonfungiblePositionManager.IncreaseLiquidityParams memory params =
INonfungiblePositionManager.IncreaseLiquidityParams({
tokenId: tokenId,
amount0Desired: amountAdd0,
amount1Desired: amountAdd1,
amount0Min: 0,
amount1Min: 0,
deadline: block.timestamp
});
(liquidity, amount0, amount1) = nonfungiblePositionManager.increaseLiquidity(params);
}
}