all files / contracts/test/ FlashLoanAdaptorTest.sol

100% Statements 15/15
100% Branches 2/2
100% Functions 5/5
100% Lines 15/15
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×           19×           15×     15×   15×   15×                 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
        );
    }
}