| 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 |
16×
8×
8×
5×
9×
19×
15×
15×
15×
15×
7×
7×
13×
15×
15×
| // SPDX-License-Identifier: GPL-2.0-or-later
pragma solidity ^0.8.0;
import "../Interfaces.sol";
contract FlashLoanAdaptorTest is IERC3156FlashBorrower {
event BorrowResult(address token, uint balance, uint fee, uint borrowIndex, address sender, address initiator);
function setMaxAllowance(address token, address to) public returns (bool success) {
(success,) = token.call(abi.encodeWithSelector(IERC20.approve.selector, to, type(uint).max));
}
function testFlashBorrow(address lender, address[] calldata receivers, address[] calldata tokens, uint[] calldata amounts) external {
bytes memory data = abi.encode(receivers, tokens, amounts, 0);
_borrow(lender, receivers[0], tokens[0], amounts[0], data);
for (uint i = 0; i < receivers.length; ++i) {
for (uint j = 0; j < tokens.length; ++j) {
assert(IERC20(tokens[j]).balanceOf(receivers[i]) == 0);
}
}
}
function onFlashLoan(address initiator, address token, uint256, uint256 fee, bytes calldata data) override external returns(bytes32) {
(address[] memory receivers, address[] memory tokens, uint[] memory amounts, uint borrowIndex) =
abi.decode(data, (address[], address[], uint[], uint));
setMaxAllowance(token, msg.sender);
_emitBorrowResult(token, fee, borrowIndex, initiator);
if(tokens.length > 0 && borrowIndex < tokens.length - 1) {
uint nextBorrowIndex = borrowIndex + 1;
_borrow(
msg.sender,
receivers[nextBorrowIndex],
tokens[nextBorrowIndex],
amounts[nextBorrowIndex],
abi.encode(receivers, tokens, amounts, nextBorrowIndex)
);
}
return keccak256("ERC3156FlashBorrower.onFlashLoan");
}
function _borrow(address lender, address receiver, address token, uint amount, bytes memory data) internal {
IERC3156FlashLender(lender).flashLoan(IERC3156FlashBorrower(receiver), token, amount, data);
}
function _emitBorrowResult(address token, uint fee, uint borrowIndex, address initiator) internal {
emit BorrowResult(
token,
IERC20(token).balanceOf(address(this)),
fee,
borrowIndex,
msg.sender,
initiator
);
}
}
|