From e9ccd751bcd91f186f0079bddd91babf0bbeea0d Mon Sep 17 00:00:00 2001 From: pradyuman-verma Date: Fri, 27 May 2022 23:48:07 +0530 Subject: [PATCH 1/2] fixed native token transfer bug: mainnet --- contracts/mainnet/connectors/hop/helpers.sol | 11 +++++++---- contracts/mainnet/connectors/hop/interface.sol | 2 +- contracts/mainnet/connectors/hop/main.sol | 16 +++++++++------- 3 files changed, 17 insertions(+), 12 deletions(-) diff --git a/contracts/mainnet/connectors/hop/helpers.sol b/contracts/mainnet/connectors/hop/helpers.sol index 9b8b099c..6a0c6ed9 100644 --- a/contracts/mainnet/connectors/hop/helpers.sol +++ b/contracts/mainnet/connectors/hop/helpers.sol @@ -26,13 +26,16 @@ contract Helpers is DSMath, Basic { uint256 destinationDeadline; } - function _sendToL2(BridgeParams memory params) internal { + function _sendToL2(BridgeParams memory params, bool isNative) internal { IHopRouter router = IHopRouter(params.router); - TokenInterface tokenContract = TokenInterface(params.token); - approve(tokenContract, params.router, params.amount); + uint256 nativeTokenAmt = isNative ? params.amount : 0; + if (!isNative) { + TokenInterface tokenContract = TokenInterface(params.token); + approve(tokenContract, params.router, params.amount); + } - router.sendToL2( + router.sendToL2{ value: nativeTokenAmt }( params.targetChainId, params.recipient, params.amount, diff --git a/contracts/mainnet/connectors/hop/interface.sol b/contracts/mainnet/connectors/hop/interface.sol index 09b8997a..d38c7518 100644 --- a/contracts/mainnet/connectors/hop/interface.sol +++ b/contracts/mainnet/connectors/hop/interface.sol @@ -12,5 +12,5 @@ interface IHopRouter { uint256 deadline, address relayer, uint256 relayerFee - ) external; + ) external payable; } diff --git a/contracts/mainnet/connectors/hop/main.sol b/contracts/mainnet/connectors/hop/main.sol index 32481914..3a907da4 100644 --- a/contracts/mainnet/connectors/hop/main.sol +++ b/contracts/mainnet/connectors/hop/main.sol @@ -26,24 +26,26 @@ abstract contract Resolver is Helpers { returns (string memory _eventName, bytes memory _eventParam) { params.amount = getUint(getId, params.amount); - - bool isEth = params.token == ethAddr; - params.token = params.token == ethAddr ? wethAddr : params.token; - TokenInterface tokenContract = TokenInterface(params.token); - if (isEth) { + if (params.token == wethAddr) { + convertWethToEth(true, tokenContract, params.amount); + params.token = ethAddr; + } + + bool isNative = params.token == ethAddr; + + if (isNative) { params.amount = params.amount == uint256(-1) ? address(this).balance : params.amount; - convertEthToWeth(isEth, tokenContract, params.amount); } else { params.amount = params.amount == uint256(-1) ? tokenContract.balanceOf(address(this)) : params.amount; } - _sendToL2(params); + _sendToL2(params, isNative); _eventName = "LogBridge(address,uint256,address,uint256,uint256,uint256,uint256)"; _eventParam = abi.encode( From 796c6d7ab03bff882cd135289011caa7dc3a2d87 Mon Sep 17 00:00:00 2001 From: pradyuman-verma Date: Sat, 28 May 2022 00:56:55 +0530 Subject: [PATCH 2/2] hop mainnet tests --- test/mainnet/hop/hop.test.ts | 127 +++++++++++++++++++++++++++++++++++ 1 file changed, 127 insertions(+) create mode 100644 test/mainnet/hop/hop.test.ts diff --git a/test/mainnet/hop/hop.test.ts b/test/mainnet/hop/hop.test.ts new file mode 100644 index 00000000..415d3825 --- /dev/null +++ b/test/mainnet/hop/hop.test.ts @@ -0,0 +1,127 @@ +import { expect } from "chai"; +import hre from "hardhat"; +const { waffle, ethers } = hre; +const { provider } = waffle; + +import { deployAndEnableConnector } from "../../../scripts/tests/deployAndEnableConnector"; +import { buildDSAv2 } from "../../../scripts/tests/buildDSAv2"; +import { encodeSpells } from "../../../scripts/tests/encodeSpells"; +import { getMasterSigner } from "../../../scripts/tests/getMasterSigner"; +import { addLiquidity } from "../../../scripts/tests/addLiquidity"; +import { addresses } from "../../../scripts/tests/mainnet/addresses"; +import { abis } from "../../../scripts/constant/abis"; +import { ConnectV2Hop__factory } from "../../../typechain"; +import { Signer, Contract } from "ethers"; + +describe("Hop Connector", function () { + const connectorName = "HOP-MAINNET-X"; + + let dsaWallet0: Contract; + let masterSigner: Signer; + let instaConnectorsV2: Contract; + let connector: Contract; + + const wallets = provider.getWallets(); + const [wallet0, wallet1, wallet2, wallet3] = wallets; + + const DAI_ADDR = "0x6B175474E89094C44Da98b954EedeAC495271d0F"; + const l2AmmWrapper = "0x3d4Cc8A61c7528Fd86C55cfe061a78dCBA48EDd1"; + + const token = new ethers.Contract(DAI_ADDR, abis.basic.erc20); + + before(async () => { + await hre.network.provider.request({ + method: "hardhat_reset", + params: [ + { + forking: { + // @ts-ignore + jsonRpcUrl: hre.config.networks.hardhat.forking.url, + blockNumber: 14854895 + } + } + ] + }); + + masterSigner = await getMasterSigner(); + instaConnectorsV2 = await ethers.getContractAt(abis.core.connectorsV2, addresses.core.connectorsV2); + connector = await deployAndEnableConnector({ + connectorName, + contractArtifact: ConnectV2Hop__factory, + signer: masterSigner, + connectors: instaConnectorsV2 + }); + console.log("Connector address", connector.address); + }); + + it("Should have contracts deployed.", async function () { + expect(!!instaConnectorsV2.address).to.be.true; + expect(!!connector.address).to.be.true; + expect(!!(await masterSigner.getAddress())).to.be.true; + }); + + describe("DSA wallet setup", function () { + it("Should build DSA v2", async function () { + dsaWallet0 = await buildDSAv2(wallet0.address); + expect(!!dsaWallet0.address).to.be.true; + }); + + it("Deposit ETH & DAI into DSA wallet", async function () { + await wallet0.sendTransaction({ + to: dsaWallet0.address, + value: ethers.utils.parseEther("10") + }); + expect(await ethers.provider.getBalance(dsaWallet0.address)).to.be.gte(ethers.utils.parseEther("10")); + + await addLiquidity("dai", dsaWallet0.address, ethers.utils.parseEther("10000")); + }); + }); + + describe("Main", function () { + it("should migrate DAI from L1 to L2", async function () { + const amount = ethers.utils.parseEther("10"); + const deadline = Date.now() + 604800; + const getId = "0"; + + const params: any = [DAI_ADDR, wallet0.address, l2AmmWrapper, 137, amount.toString(), "0", deadline]; + + const spells = [ + { + connector: connectorName, + method: "bridge", + args: [params, getId] + } + ]; + + const tx = await dsaWallet0.connect(wallet0).cast(...encodeSpells(spells), wallet1.address); + let receipt = await tx.wait(); + }); + + it("should migrate ETH from L1 to L2", async function () { + const amount = ethers.utils.parseEther("10"); + const deadline = Date.now() + 604800; + const getId = "0"; + + const params: any = [ + "0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE", + wallet0.address, + "0xb8901acB165ed027E32754E0FFe830802919727f", + 137, + amount.toString(), + "0", + deadline + ]; + + const spells = [ + { + connector: connectorName, + method: "bridge", + args: [params, getId] + } + ]; + + const tx = await dsaWallet0.connect(wallet0).cast(...encodeSpells(spells), wallet1.address); + let receipt = await tx.wait(); + }); + }); +});