Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Yield-earning Warp Routes with ERC4626 #3076

Merged
merged 36 commits into from
Mar 13, 2024
Merged
Show file tree
Hide file tree
Changes from 21 commits
Commits
Show all changes
36 commits
Select commit Hold shift + click to select a range
b88739e
Add 4626 mocks
ltyu Dec 18, 2023
2a1c329
Add HypERC20CollateralVaultDeposit and test
ltyu Dec 18, 2023
da5da9b
Add virtual to allow inheritance
ltyu Dec 18, 2023
c901ac1
Add _transferTo logic
ltyu Dec 18, 2023
fbda4e7
Add sweep and tests
ltyu Dec 18, 2023
63bc6f9
Add mint
ltyu Dec 18, 2023
0dd7526
Add virtual to override
ltyu Dec 18, 2023
c484626
Update tests to use fuzzer
ltyu Dec 18, 2023
fcc63a8
Add more sweep test
ltyu Dec 18, 2023
8926218
Add natspec. Move into internal function
ltyu Dec 18, 2023
eb7175a
Remove console
ltyu Dec 18, 2023
af45235
Add event
ltyu Dec 18, 2023
28e3dff
Add fuzzing parameters
ltyu Dec 18, 2023
e2025e9
Merge branch 'main' into yield-warp-routes
ltyu Dec 18, 2023
b4dcc0f
forge install: solmate
ltyu Dec 18, 2023
804e855
Merge branch 'main' into yield-warp-routes
ltyu Dec 19, 2023
ef3c6ce
Update comments
ltyu Dec 19, 2023
f95c8a9
Merge branch 'main' into yield-warp-routes
ltyu Dec 19, 2023
17095e3
Merge branch 'yield-warp-routes' of https://github.com/ltyu/hyperlane…
ltyu Dec 19, 2023
0f505b8
Revert gitmodule changes
ltyu Dec 19, 2023
2d5ae90
Revert lib changes
ltyu Dec 19, 2023
9f8b263
Update to follow CEI
ltyu Jan 2, 2024
0686f78
Update solidity/contracts/token/HypERC20CollateralVaultDeposit.sol
ltyu Jan 2, 2024
50c517b
Update according to comments: withdraw directly to recipient, remove …
ltyu Jan 2, 2024
62f56e7
Update tests
ltyu Jan 2, 2024
4bc3e44
Update according to comments: move max approve to constructor
ltyu Jan 2, 2024
fb40a64
Add msg.value check into AbstractMessageIdAuthHook
ltyu Jan 10, 2024
65dceff
Merge branch 'main' into yield-warp-routes
ltyu Jan 14, 2024
d34a608
Merge branch 'main' into yield-warp-routes
ltyu Jan 17, 2024
30011be
Use bound. Move duplicate calls to helper functions
ltyu Jan 18, 2024
eed40f8
Merge branch 'main' into yield-warp-routes
ltyu Jan 18, 2024
9102f9e
Update variable names
ltyu Jan 18, 2024
ee594ae
Merge branch 'yield-warp-routes' of https://github.com/ltyu/hyperlane…
ltyu Jan 18, 2024
8b9a772
Merge branch 'main' into yield-warp-routes
ltyu Mar 13, 2024
e03eaaa
Fix test
ltyu Mar 13, 2024
e9ed058
Merge branch 'main' into yield-warp-routes
ltyu Mar 13, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 4 additions & 0 deletions solidity/contracts/test/ERC20Test.sol
Original file line number Diff line number Diff line change
Expand Up @@ -19,4 +19,8 @@ contract ERC20Test is ERC20 {
function decimals() public view override returns (uint8) {
return _decimals;
}

function mint(uint256 amount) public {
_mint(msg.sender, amount);
}
}
13 changes: 13 additions & 0 deletions solidity/contracts/test/ERC4626/ERC4626Test.sol
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
// SPDX-License-Identifier: Apache-2.0
pragma solidity >=0.8.0;
import "@openzeppelin/contracts/token/ERC20/extensions/ERC4626.sol";
import "@openzeppelin/contracts/token/ERC20/ERC20.sol";
import "@openzeppelin/contracts/interfaces/IERC20.sol";

contract ERC4626Test is ERC4626 {
constructor(
address _asset,
string memory _name,
string memory _symbol
) ERC4626(IERC20(_asset)) ERC20(_name, _symbol) {}
}
2 changes: 1 addition & 1 deletion solidity/contracts/token/HypERC20Collateral.sol
Original file line number Diff line number Diff line change
Expand Up @@ -37,7 +37,7 @@ contract HypERC20Collateral is TokenRouter {
*/
function _transferFromSender(
uint256 _amount
) internal override returns (bytes memory) {
) internal virtual override returns (bytes memory) {
wrappedToken.safeTransferFrom(msg.sender, address(this), _amount);
return bytes(""); // no metadata
}
Expand Down
98 changes: 98 additions & 0 deletions solidity/contracts/token/HypERC20CollateralVaultDeposit.sol
Original file line number Diff line number Diff line change
@@ -0,0 +1,98 @@
// SPDX-License-Identifier: Apache-2.0
pragma solidity >=0.8.0;
import "@openzeppelin/contracts/token/ERC20/extensions/ERC4626.sol";
import {HypERC20Collateral} from "./HypERC20Collateral.sol";

/**
* @title Hyperlane ERC20 Token Collateral with deposits collateral to a vault
* @author ltyu
*/
contract HypERC20CollateralVaultDeposit is HypERC20Collateral {
// Address of the ERC4626 compatible vault
ERC4626 public immutable vault;

// Internal balance of total asset deposited
uint256 public assetDeposited;
ltyu marked this conversation as resolved.
Show resolved Hide resolved

event ExcessSharesSwept(uint256 amount, uint256 assetsRedeemed);

constructor(
address _vault,
address erc20,
address _mailbox
) HypERC20Collateral(erc20, _mailbox) {
vault = ERC4626(_vault);
}
ltyu marked this conversation as resolved.
Show resolved Hide resolved

/**
* @dev Transfers `_amount` of `wrappedToken` from `msg.sender` to this contract, and deposit into vault
* @inheritdoc HypERC20Collateral
*/
function _transferFromSender(
uint256 _amount
) internal override returns (bytes memory metadata) {
metadata = super._transferFromSender(_amount);
_depositIntoVault(_amount);
}

/**
* @dev Deposits into the vault and increment assetDeposited
* @param _amount amount to deposit into vault
*/
function _depositIntoVault(uint256 _amount) internal {
wrappedToken.approve(address(vault), _amount);
ltyu marked this conversation as resolved.
Show resolved Hide resolved
vault.deposit(_amount, address(this));
assetDeposited += _amount;
ltyu marked this conversation as resolved.
Show resolved Hide resolved
}
Fixed Show fixed Hide fixed
Fixed Show fixed Hide fixed
github-advanced-security[bot] marked this conversation as resolved.
Fixed
Show resolved Hide resolved

/**
* @dev Transfers `_amount` of `wrappedToken` from this contract to `_recipient`, and withdraws from vault
* @inheritdoc HypERC20Collateral
*/
function _transferTo(
address _recipient,
uint256 _amount,
bytes calldata _metadata
Fixed Show fixed Hide fixed
) internal virtual override {
_withdrawFromVault(_amount);
super._transferTo(_recipient, _amount, _metadata);
}

/**
* @dev Withdraws from the vault and decrement assetDeposited
* @param _amount amount to withdraw from vault
*/
function _withdrawFromVault(uint256 _amount) internal {
vault.withdraw(_amount, address(this), address(this));
assetDeposited -= _amount;
ltyu marked this conversation as resolved.
Show resolved Hide resolved
ltyu marked this conversation as resolved.
Show resolved Hide resolved
}
Fixed Show fixed Hide fixed
github-advanced-security[bot] marked this conversation as resolved.
Fixed
Show resolved Hide resolved

/**
* @notice Allows the owner to redeem excess shares
*/
function sweep() external onlyOwner {
if (_excessVaultShares() > 0) {
uint256 excessShares = vault.maxRedeem(address(this)) -
vault.convertToShares(assetDeposited);
yorhodes marked this conversation as resolved.
Show resolved Hide resolved
uint256 assetsRedeemed = vault.redeem(
excessShares,
owner(),
address(this)
);
emit ExcessSharesSwept(excessShares, assetsRedeemed);
}
}
Fixed Show fixed Hide fixed

/**
* @notice Calculates excess vault shares using the converted assetDeposited and max redeemable shares
* @return excess vault shares or 0
*/
function _excessVaultShares() internal view returns (uint256) {
return
vault.maxRedeem(address(this)) >
vault.convertToShares(assetDeposited)
? vault.maxRedeem(address(this)) -
vault.convertToShares(assetDeposited)
: 0;
}
ltyu marked this conversation as resolved.
Show resolved Hide resolved
}
2 changes: 1 addition & 1 deletion solidity/test/token/HypERC20.t.sol
Original file line number Diff line number Diff line change
Expand Up @@ -153,7 +153,7 @@ abstract contract HypTokenTest is Test {
_performRemoteTransferAndGas(_msgValue, _amount, _gasOverhead);
}

function testBenchmark_overheadGasUsage() public {
function testBenchmark_overheadGasUsage() public virtual {
vm.prank(address(localMailbox));

uint256 gasBefore = gasleft();
Expand Down
237 changes: 237 additions & 0 deletions solidity/test/token/HypERC20CollateralVaultDeposit.t.sol
ltyu marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
@@ -0,0 +1,237 @@
// SPDX-License-Identifier: MIT OR Apache-2.0
pragma solidity ^0.8.13;

/*@@@@@@@ @@@@@@@@@
@@@@@@@@@ @@@@@@@@@
@@@@@@@@@ @@@@@@@@@
@@@@@@@@@ @@@@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@ HYPERLANE @@@@@@@
@@@@@@@@@@@@@@@@@@@@@@@@@
@@@@@@@@@ @@@@@@@@@
@@@@@@@@@ @@@@@@@@@
@@@@@@@@@ @@@@@@@@@
@@@@@@@@@ @@@@@@@@*/

import "forge-std/Test.sol";
import {ERC4626Test} from "../../contracts/test/ERC4626/ERC4626Test.sol";
import {TypeCasts} from "../../contracts/libs/TypeCasts.sol";
import {HypTokenTest} from "./HypERC20.t.sol";

import {HypERC20CollateralVaultDeposit} from "../../contracts/token/HypERC20CollateralVaultDeposit.sol";
import "../../contracts/test/ERC4626/ERC4626Test.sol";

contract HypERC20CollateralVaultDepositTest is HypTokenTest {
using TypeCasts for address;
uint constant DUST_AMOUNT = 1e11;
HypERC20CollateralVaultDeposit internal erc20CollateralVaultDeposit;
ERC4626Test vault;

function setUp() public override {
super.setUp();
vault = new ERC4626Test(address(primaryToken), "Regular Vault", "RV");

localToken = new HypERC20CollateralVaultDeposit(
address(vault),
address(primaryToken),
address(localMailbox)
);
erc20CollateralVaultDeposit = HypERC20CollateralVaultDeposit(
address(localToken)
);
ltyu marked this conversation as resolved.
Show resolved Hide resolved

erc20CollateralVaultDeposit.enrollRemoteRouter(
DESTINATION,
address(remoteToken).addressToBytes32()
);

remoteMailbox.setDefaultHook(address(noopHook));
remoteMailbox.setRequiredHook(address(noopHook));
primaryToken.transfer(ALICE, 1000e18);
_enrollRemoteTokenRouter();
}

function testRemoteTransfer_deposits_intoVault(
uint256 transferAmount
) public {
vm.assume(transferAmount < TOTAL_SUPPLY);

vm.startPrank(ALICE);
primaryToken.mint(transferAmount);
primaryToken.approve(address(localToken), transferAmount);
vm.stopPrank();

// Check vault shares balance before and after transfer
assertEq(vault.maxRedeem(address(erc20CollateralVaultDeposit)), 0);
assertEq(erc20CollateralVaultDeposit.assetDeposited(), 0);

_performRemoteTransfer(0, transferAmount);
assertApproxEqAbs(
vault.maxRedeem(address(erc20CollateralVaultDeposit)),
transferAmount,
1
);
assertEq(erc20CollateralVaultDeposit.assetDeposited(), transferAmount);
}

function testRemoteTransfer_withdraws_fromVault(
uint256 transferAmount
) public {
vm.assume(transferAmount < TOTAL_SUPPLY);

// Transfer to Bob
vm.startPrank(ALICE);
primaryToken.mint(transferAmount);
primaryToken.approve(address(localToken), transferAmount);
vm.stopPrank();
ltyu marked this conversation as resolved.
Show resolved Hide resolved

_performRemoteTransfer(0, transferAmount);

// Transfer back from Bob to Alice
vm.prank(BOB);
remoteToken.transferRemote(
ORIGIN,
BOB.addressToBytes32(),
transferAmount
);

// Check Alice's local token balance
uint256 prevBalance = localToken.balanceOf(ALICE);
vm.prank(address(localMailbox));
localToken.handle(
DESTINATION,
address(remoteToken).addressToBytes32(),
abi.encodePacked(ALICE.addressToBytes32(), transferAmount)
);

assertEq(localToken.balanceOf(ALICE), prevBalance + transferAmount);
assertEq(erc20CollateralVaultDeposit.assetDeposited(), 0);
}

function testRemoteTransfer_withdraws_lessShares(
uint256 rewardAmount
) public {
// @dev a rewardAmount less than the DUST_AMOUNT will round down
vm.assume(rewardAmount > DUST_AMOUNT);
vm.assume(rewardAmount < TOTAL_SUPPLY);
ltyu marked this conversation as resolved.
Show resolved Hide resolved

// Transfer to Bob
vm.prank(ALICE);
primaryToken.approve(address(localToken), TRANSFER_AMT);
_performRemoteTransfer(0, TRANSFER_AMT);

// Increase vault balance, which will reduce share redeemed for the same amount
primaryToken.mint(rewardAmount);
primaryToken.transfer(address(vault), rewardAmount);

// Transfer back from Bob to Alice
vm.prank(BOB);
remoteToken.transferRemote(
ORIGIN,
BOB.addressToBytes32(),
TRANSFER_AMT
);

// Check Alice's local token balance
uint256 prevBalance = localToken.balanceOf(ALICE);
vm.prank(address(localMailbox));
localToken.handle(
DESTINATION,
address(remoteToken).addressToBytes32(),
abi.encodePacked(ALICE.addressToBytes32(), TRANSFER_AMT)
);

assertEq(localToken.balanceOf(ALICE), prevBalance + TRANSFER_AMT);

// Has leftover shares, but no assets deposited
assertEq(erc20CollateralVaultDeposit.assetDeposited(), 0);
assertGt(vault.maxRedeem(address(erc20CollateralVaultDeposit)), 0);
}

function testRemoteTransfer_sweep_revertNonOwner(
uint256 rewardAmount
) public {
testRemoteTransfer_withdraws_lessShares(rewardAmount);
vm.startPrank(BOB);
vm.expectRevert(abi.encodePacked("Ownable: caller is not the owner"));
erc20CollateralVaultDeposit.sweep();
vm.stopPrank();
}

function testRemoteTransfer_sweep_noExcessShares(
uint256 transferAmount
) public {
testRemoteTransfer_deposits_intoVault(transferAmount);

uint256 ownerBalancePrev = primaryToken.balanceOf(
erc20CollateralVaultDeposit.owner()
);

erc20CollateralVaultDeposit.sweep();
assertEq(
primaryToken.balanceOf(erc20CollateralVaultDeposit.owner()),
ownerBalancePrev
);
}

function testRemoteTransfer_sweep_excessShares(
uint256 rewardAmount
) public {
testRemoteTransfer_withdraws_lessShares(rewardAmount);

uint256 ownerBalancePrev = primaryToken.balanceOf(
erc20CollateralVaultDeposit.owner()
);
uint256 excessAmount = vault.maxRedeem(
address(erc20CollateralVaultDeposit)
);

erc20CollateralVaultDeposit.sweep();
assertGt(
primaryToken.balanceOf(erc20CollateralVaultDeposit.owner()),
ownerBalancePrev + excessAmount
);
}

function testRemoteTransfer_sweep_excessSharesMultipleDeposit(
uint256 rewardAmount
) public {
testRemoteTransfer_withdraws_lessShares(rewardAmount);

uint256 ownerBalancePrev = primaryToken.balanceOf(
erc20CollateralVaultDeposit.owner()
);
uint256 excessAmount = vault.maxRedeem(
address(erc20CollateralVaultDeposit)
);

// Deposit again for Alice
vm.prank(ALICE);
primaryToken.approve(address(localToken), TRANSFER_AMT);
_performRemoteTransfer(0, TRANSFER_AMT);

// Sweep and check
erc20CollateralVaultDeposit.sweep();
assertGt(
primaryToken.balanceOf(erc20CollateralVaultDeposit.owner()),
ownerBalancePrev + excessAmount
);
}

function testBenchmark_overheadGasUsage() public override {
vm.prank(ALICE);
primaryToken.approve(address(localToken), TRANSFER_AMT);
_performRemoteTransfer(0, TRANSFER_AMT);

vm.prank(address(localMailbox));

uint256 gasBefore = gasleft();
localToken.handle(
DESTINATION,
address(remoteToken).addressToBytes32(),
abi.encodePacked(BOB.addressToBytes32(), TRANSFER_AMT)
);
uint256 gasAfter = gasleft();
console.log("Overhead gas usage: %d", gasBefore - gasAfter);
}
}
Loading