diff --git a/.github/workflows/e2e-tests.yml b/.github/workflows/e2e-tests.yml index 22ee93104d..373f26b16b 100644 --- a/.github/workflows/e2e-tests.yml +++ b/.github/workflows/e2e-tests.yml @@ -33,7 +33,7 @@ jobs: secrets: inherit test-e2e: - name: "Test E2E - ${{ matrix.test.name }}${{ matrix.test.type == 'orbit' && ' with L3' || ''}}" + name: "${{ matrix.test.name }}${{ matrix.test.type == 'orbit-eth' && ' with L3' || matrix.test.type == 'orbit-custom' && ' with custom fee token' || ''}}" needs: [build, load-e2e-files] runs-on: ubuntu-latest strategy: @@ -96,15 +96,22 @@ jobs: if: inputs.test_type != 'cctp' uses: OffchainLabs/actions/run-nitro-test-node@a20a76172ce524832ac897bef2fa10a62ed81c29 with: - nitro-testnode-ref: aab133aceadec2e622f15fa438f6327e3165392d - l3-node: ${{ matrix.test.type == 'orbit' }} - no-l3-token-bridge: ${{ matrix.test.type != 'orbit' }} + nitro-testnode-ref: badbcbea9b43d46e115da4d7c9f2f57c31af8431 + l3-node: ${{ matrix.test.type != 'regular' }} + no-l3-token-bridge: ${{ matrix.test.type == 'regular' }} + args: ${{ matrix.test.type == 'orbit-custom' && '--l3-fee-token' || '' }} - name: Run e2e tests via cypress-io/github-action uses: cypress-io/github-action@0da3c06ed8217b912deea9d8ee69630baed1737e # pin@v6.7.6 with: start: yarn start - command: "yarn test:e2e${{ (matrix.test.type == 'cctp' && ':cctp') || (matrix.test.type == 'orbit' && ':orbit') || '' }} --browser chrome" + command: >- + ${{ + (matrix.test.type == 'orbit-eth') && 'yarn test:e2e:orbit --browser chrome' || + (matrix.test.type == 'orbit-custom' && 'yarn test:e2e:orbit:custom-gas-token --browser chrome') || + (matrix.test.type == 'cctp' && 'yarn test:e2e:cctp --browser chrome') || + 'yarn test:e2e --browser chrome' + }} wait-on: http://127.0.0.1:3000 wait-on-timeout: 120 spec: ./packages/arb-token-bridge-ui/tests/e2e/specs/* @@ -126,7 +133,7 @@ jobs: uses: actions/upload-artifact@v4 if: always() with: - name: e2e-artifacts-${{ github.sha }}-${{ matrix.test.name }}-${{ (matrix.test.type == 'cctp' && 'cctp') || (matrix.test.type == 'orbit' && 'l3') || 'regular'}} + name: e2e-artifacts-${{ github.sha }}-${{ matrix.test.name }}-${{ (matrix.test.type == 'cctp' && 'cctp') || (matrix.test.type == 'orbit-eth' && 'l3') || (matrix.test.type == 'orbit-custom' && 'custom-fee-token') || 'regular'}} path: | ./packages/arb-token-bridge-ui/cypress/videos ./packages/arb-token-bridge-ui/cypress/screenshots diff --git a/.github/workflows/formatSpecfiles.js b/.github/workflows/formatSpecfiles.js index fac4b2f6c2..bb6b23e1ef 100644 --- a/.github/workflows/formatSpecfiles.js +++ b/.github/workflows/formatSpecfiles.js @@ -14,7 +14,11 @@ switch (testType) { }); tests.push({ ...spec, - type: "orbit", + type: "orbit-eth", + }); + tests.push({ + ...spec, + type: "orbit-custom", }); }); break; diff --git a/README.md b/README.md index 73f9acd030..74739c0054 100644 --- a/README.md +++ b/README.md @@ -128,6 +128,12 @@ It is important for any code change to pass both unit and end-to-end tests. This ./test-node.bash --init --no-simple --tokenbridge --l3node --l3-token-bridge ``` + To run with a custom fee token also include the following flags: + + ```bash + --l3-fee-token --l3-fee-token-decimals 18 + ``` + 2. When the Nitro test-node is up and running you should see logs like `sequencer_1` and `staker-unsafe_1` in the terminal. This can take up to 10 minutes. 2. At the root of the token bridge UI: diff --git a/package.json b/package.json index c60ae31523..28b20b0721 100644 --- a/package.json +++ b/package.json @@ -17,7 +17,8 @@ "lint:fix": "yarn workspace arb-token-bridge-ui lint:fix", "test:e2e": "yarn workspace arb-token-bridge-ui env-cmd --silent --file .e2e.env yarn synpress run --configFile synpress.config.ts", "test:e2e:cctp": "yarn test:e2e --configFile synpress.cctp.config.ts", - "test:e2e:orbit": "E2E_ORBIT=true yarn test:e2e" + "test:e2e:orbit": "E2E_ORBIT=true yarn test:e2e", + "test:e2e:orbit:custom-gas-token": "E2E_ORBIT_CUSTOM_GAS_TOKEN=true yarn test:e2e" }, "resolutions": { "**/@walletconnect/ethereum-provider": "2.13.1", diff --git a/packages/arb-token-bridge-ui/src/pages/_app.tsx b/packages/arb-token-bridge-ui/src/pages/_app.tsx index 54412f8abf..6fb6a7eca0 100644 --- a/packages/arb-token-bridge-ui/src/pages/_app.tsx +++ b/packages/arb-token-bridge-ui/src/pages/_app.tsx @@ -13,7 +13,6 @@ import 'tippy.js/themes/light.css' import '@rainbow-me/rainbowkit/styles.css' -import { registerLocalNetwork } from '../util/networks' import { Layout } from '../components/common/Layout' import { siteTitle } from './_document' @@ -21,13 +20,6 @@ import '../styles/tailwind.css' import '../styles/purple.css' import { isUserRejectedError } from '../util/isUserRejectedError' -if ( - process.env.NODE_ENV !== 'production' || - process.env.NEXT_PUBLIC_IS_E2E_TEST -) { - registerLocalNetwork() -} - dayjs.extend(utc) dayjs.extend(relativeTime) dayjs.extend(timeZone) diff --git a/packages/arb-token-bridge-ui/src/pages/index.tsx b/packages/arb-token-bridge-ui/src/pages/index.tsx index 1720c741f8..325bc493f4 100644 --- a/packages/arb-token-bridge-ui/src/pages/index.tsx +++ b/packages/arb-token-bridge-ui/src/pages/index.tsx @@ -1,4 +1,4 @@ -import React, { useEffect } from 'react' +import React, { ComponentType, useEffect } from 'react' import { GetServerSidePropsContext, GetServerSidePropsResult } from 'next' import dynamic from 'next/dynamic' import { decodeString, encodeString } from 'use-query-params' @@ -7,7 +7,8 @@ import { registerCustomArbitrumNetwork } from '@arbitrum/sdk' import { Loader } from '../components/common/atoms/Loader' import { getCustomChainsFromLocalStorage, - mapCustomChainToNetworkData + mapCustomChainToNetworkData, + registerLocalNetwork } from '../util/networks' import { getOrbitChains } from '../util/orbitChainsList' import { sanitizeQueryParams } from '../hooks/useNetworks' @@ -17,17 +18,32 @@ import { } from '../hooks/useArbQueryParams' import { sanitizeExperimentalFeaturesQueryParam } from '../util' -const App = dynamic(() => import('../components/App/App'), { - ssr: false, - loading: () => ( - <> -
-
- -
- - ) -}) +const App = dynamic( + () => { + return new Promise<{ default: ComponentType }>(async resolve => { + if ( + process.env.NODE_ENV !== 'production' || + process.env.NEXT_PUBLIC_IS_E2E_TEST + ) { + await registerLocalNetwork() + } + + const AppComponent = await import('../components/App/App') + resolve(AppComponent) + }) + }, + { + ssr: false, + loading: () => ( + <> +
+
+ +
+ + ) + } +) function getDestinationWithSanitizedQueryParams( sanitized: { @@ -89,9 +105,11 @@ function addOrbitChainsToArbitrumSDK() { ) } -export function getServerSideProps({ +export async function getServerSideProps({ query -}: GetServerSidePropsContext): GetServerSidePropsResult> { +}: GetServerSidePropsContext): Promise< + GetServerSidePropsResult> +> { const sourceChainId = decodeChainQueryParam(query.sourceChain) const destinationChainId = decodeChainQueryParam(query.destinationChain) const experiments = decodeString(query.experiments) @@ -103,6 +121,12 @@ export function getServerSideProps({ } } + if ( + process.env.NODE_ENV !== 'production' || + process.env.NEXT_PUBLIC_IS_E2E_TEST + ) { + await registerLocalNetwork() + } // it's necessary to call this before sanitization to make sure all chains are registered addOrbitChainsToArbitrumSDK() diff --git a/packages/arb-token-bridge-ui/src/util/TokenUtils.ts b/packages/arb-token-bridge-ui/src/util/TokenUtils.ts index 91318f35e7..99eaf03aaa 100644 --- a/packages/arb-token-bridge-ui/src/util/TokenUtils.ts +++ b/packages/arb-token-bridge-ui/src/util/TokenUtils.ts @@ -66,6 +66,13 @@ function getErc20DataCache(params: GetErc20DataCacheParams): Erc20Data | null function getErc20DataCache( params?: GetErc20DataCacheParams ): Erc20DataCache | (Erc20Data | null) { + if ( + typeof window === 'undefined' || + typeof window.localStorage === 'undefined' + ) { + return null + } + const cache: Erc20DataCache = JSON.parse( // intentionally using || instead of ?? for it to work with an empty string localStorage.getItem(erc20DataCacheLocalStorageKey) || '{}' diff --git a/packages/arb-token-bridge-ui/src/util/networks.ts b/packages/arb-token-bridge-ui/src/util/networks.ts index 5dc8e5627c..02d26cdeb3 100644 --- a/packages/arb-token-bridge-ui/src/util/networks.ts +++ b/packages/arb-token-bridge-ui/src/util/networks.ts @@ -1,3 +1,4 @@ +import { StaticJsonRpcProvider } from '@ethersproject/providers' import { ArbitrumNetwork, getChildrenForNetwork, @@ -9,6 +10,7 @@ import { import { loadEnvironmentVariableWithFallback } from './index' import { getBridgeUiConfigForChain } from './bridgeUiConfig' import { chainIdToInfuraUrl } from './infura' +import { fetchErc20Data } from './TokenUtils' export enum ChainId { // L1 @@ -385,6 +387,42 @@ export const defaultL3Network: ArbitrumNetwork = { } } +export const defaultL3CustomGasTokenNetwork: ArbitrumNetwork = { + chainId: 333333, + parentChainId: ChainId.ArbitrumLocal, + confirmPeriodBlocks: 20, + ethBridge: { + bridge: '0xA584795e24628D9c067A6480b033C9E96281fcA3', + inbox: '0xDcA690902d3154886Ec259308258D10EA5450996', + outbox: '0xda243bD61B011024FC923164db75Dde198AC6175', + rollup: process.env.NEXT_PUBLIC_IS_E2E_TEST + ? '0x17d70d77AAEe46ACDF8b87BB2f085f36f63eC638' + : '0x7a23F33C1C384eFc11b8Cf207420c464ba2959CC', + sequencerInbox: '0x16c54EE2015CD824415c2077F4103f444E00A8cb' + }, + nativeToken: '0xE069078bA9ACCE4eeAE609d8754515Cf13dd6706', + isCustom: true, + isTestnet: true, + name: 'L3 Local', + retryableLifetimeSeconds: 604800, + tokenBridge: { + parentCustomGateway: '0xCe02eA568090ae7d5184B0a98df90f6aa69C1552', + parentErc20Gateway: '0x59156b0596689D965Ba707E160e5370AF22461a0', + parentGatewayRouter: '0x0C085152C2799834fc1603533ff6916fa1FdA302', + parentMultiCall: '0x20a3627Dcc53756E38aE3F92717DE9B23617b422', + parentProxyAdmin: '0x1A61102c26ad3f64bA715B444C93388491fd8E68', + parentWeth: '0xA1abD387192e3bb4e84D3109181F9f005aBaF5CA', + parentWethGateway: '0x59156b0596689D965Ba707E160e5370AF22461a0', + childCustomGateway: '0xD4816AeF8f85A3C1E01Cd071a81daD4fa941625f', + childErc20Gateway: '0xaa7d51aFFEeB32d99b1CB2fd6d81D7adA4a896e8', + childGatewayRouter: '0x8B6BC759226f8Fe687c8aD8Cc0DbF85E095e9297', + childMultiCall: '0x052B15c8Ff0544287AE689C4F2FC53A3905d7Db3', + childProxyAdmin: '0x36C56eC2CF3a3f53db9F01d0A5Ae84b36fb0A1e2', + childWeth: '0x0000000000000000000000000000000000000000', + childWethGateway: '0x0000000000000000000000000000000000000000' + } +} + export const localL1NetworkRpcUrl = loadEnvironmentVariableWithFallback({ env: process.env.NEXT_PUBLIC_LOCAL_ETHEREUM_RPC_URL, fallback: 'http://127.0.0.1:8545' @@ -398,14 +436,34 @@ export const localL3NetworkRpcUrl = loadEnvironmentVariableWithFallback({ fallback: 'http://127.0.0.1:3347' }) -export function registerLocalNetwork() { +export async function registerLocalNetwork() { try { rpcURLs[defaultL1Network.chainId] = localL1NetworkRpcUrl rpcURLs[defaultL2Network.chainId] = localL2NetworkRpcUrl rpcURLs[defaultL3Network.chainId] = localL3NetworkRpcUrl registerCustomArbitrumNetwork(defaultL2Network) - registerCustomArbitrumNetwork(defaultL3Network) + + let isLocalCustomNativeToken = false + + try { + const data = await fetchErc20Data({ + address: defaultL3CustomGasTokenNetwork.nativeToken!, + provider: new StaticJsonRpcProvider(localL2NetworkRpcUrl) + }) + if (data.symbol === 'TN') { + isLocalCustomNativeToken = true + } + } catch (e) { + // not the native token + isLocalCustomNativeToken = false + } + + registerCustomArbitrumNetwork( + isLocalCustomNativeToken + ? defaultL3CustomGasTokenNetwork + : defaultL3Network + ) } catch (error: any) { console.error(`Failed to register local network: ${error.message}`) } diff --git a/packages/arb-token-bridge-ui/src/util/wagmi/wagmiAdditionalNetworks.ts b/packages/arb-token-bridge-ui/src/util/wagmi/wagmiAdditionalNetworks.ts index 857f4f8bee..a241bcc182 100644 --- a/packages/arb-token-bridge-ui/src/util/wagmi/wagmiAdditionalNetworks.ts +++ b/packages/arb-token-bridge-ui/src/util/wagmi/wagmiAdditionalNetworks.ts @@ -5,7 +5,17 @@ import { ChainId, ChainWithRpcUrl, explorerUrls, rpcURLs } from '../networks' import { getBridgeUiConfigForChain } from '../bridgeUiConfig' export function chainToWagmiChain(chain: ChainWithRpcUrl): Chain { - const { nativeTokenData } = getBridgeUiConfigForChain(chain.chainId) + let { nativeTokenData } = getBridgeUiConfigForChain(chain.chainId) + + if (chain.chainId === ChainId.L3Local) { + nativeTokenData = chain.nativeToken + ? { + name: 'testnode', + symbol: 'TN', + decimals: 18 + } + : ether + } return { id: chain.chainId, diff --git a/packages/arb-token-bridge-ui/synpress.config.ts b/packages/arb-token-bridge-ui/synpress.config.ts index 7b85b3ede7..2150d747e3 100644 --- a/packages/arb-token-bridge-ui/synpress.config.ts +++ b/packages/arb-token-bridge-ui/synpress.config.ts @@ -10,8 +10,9 @@ import { formatUnits, parseUnits } from 'ethers/lib/utils' import { defineConfig } from 'cypress' import { StaticJsonRpcProvider } from '@ethersproject/providers' import synpressPlugins from '@synthetixio/synpress/plugins' +import { TestERC20__factory } from '@arbitrum/sdk/dist/lib/abi/factories/TestERC20__factory' import { TestWETH9__factory } from '@arbitrum/sdk/dist/lib/abi/factories/TestWETH9__factory' -import { Erc20Bridger } from '@arbitrum/sdk' +import { Erc20Bridger, EthBridger } from '@arbitrum/sdk' import logsPrinter from 'cypress-terminal-report/src/installLogsPrinter' import { getL2ERC20Address } from './src/util/TokenUtils' import specFiles from './tests/e2e/specfiles.json' @@ -19,7 +20,6 @@ import { contractAbi, contractByteCode } from './testErc20Token' import { checkForAssertions, generateActivityOnChains, - NetworkType, fundEth, setupCypressTasks, getCustomDestinationAddress, @@ -31,6 +31,7 @@ import { import { defaultL2Network, defaultL3Network, + defaultL3CustomGasTokenNetwork, registerLocalNetwork } from './src/util/networks' import { getCommonSynpressConfig } from './tests/e2e/getCommonSynpressConfig' @@ -39,19 +40,27 @@ const tests = process.env.TEST_FILE ? [process.env.TEST_FILE] : specFiles.map(file => file.file) -const isOrbitTest = process.env.E2E_ORBIT == 'true' +const isOrbitTest = [ + process.env.E2E_ORBIT, + process.env.E2E_ORBIT_CUSTOM_GAS_TOKEN +].includes('true') const shouldRecordVideo = process.env.CYPRESS_RECORD_VIDEO === 'true' +const l3Network = + process.env.ORBIT_CUSTOM_GAS_TOKEN === 'true' + ? defaultL3CustomGasTokenNetwork + : defaultL3Network + const l1WethGateway = isOrbitTest - ? defaultL3Network.tokenBridge!.parentWethGateway + ? l3Network.tokenBridge!.parentWethGateway : defaultL2Network.tokenBridge!.parentWethGateway -const l1WethAddress = isOrbitTest - ? defaultL3Network.tokenBridge!.parentWeth +let l1WethAddress = isOrbitTest + ? l3Network.tokenBridge!.parentWeth : defaultL2Network.tokenBridge!.parentWeth -const l2WethAddress = isOrbitTest - ? defaultL3Network.tokenBridge!.childWeth +let l2WethAddress = isOrbitTest + ? l3Network.tokenBridge!.childWeth : defaultL2Network.tokenBridge!.childWeth export default defineConfig({ @@ -59,7 +68,12 @@ export default defineConfig({ e2e: { async setupNodeEvents(on, config) { logsPrinter(on) - registerLocalNetwork() + + await registerLocalNetwork() + + const erc20Bridger = await Erc20Bridger.fromProvider(childProvider) + const ethBridger = await EthBridger.fromProvider(childProvider) + const isCustomFeeToken = isNonZeroAddress(ethBridger.nativeToken) if (!ethRpcUrl && !isOrbitTest) { throw new Error('NEXT_PUBLIC_LOCAL_ETHEREUM_RPC_URL variable missing.') @@ -98,14 +112,64 @@ export default defineConfig({ // Deploy and fund ERC20 to Parent and Child chains const l1ERC20Token = await deployERC20ToParentChain() + + // Approve custom fee token if not ETH + if (isCustomFeeToken) { + await approveCustomFeeToken({ + signer: localWallet.connect(parentProvider), + erc20ParentAddress: l1ERC20Token.address + }) + await approveCustomFeeToken({ + signer: localWallet.connect(parentProvider), + erc20ParentAddress: erc20Bridger.nativeToken! + }) + await ethBridger.approveGasToken({ + parentSigner: localWallet.connect(parentProvider) + }) + } + if (isCustomFeeToken) { + await fundUserWalletNativeCurrency() + } + await fundErc20ToParentChain(l1ERC20Token) - await fundErc20ToChildChain(l1ERC20Token) + await fundErc20ToChildChain({ + parentSigner: localWallet.connect(parentProvider), + parentErc20Address: l1ERC20Token.address, + amount: parseUnits('5', ERC20TokenDecimals), + isCustomFeeToken + }) await approveErc20(l1ERC20Token) + if (isCustomFeeToken) { + await approveCustomFeeToken({ + signer: userWallet.connect(parentProvider), + erc20ParentAddress: erc20Bridger.nativeToken! + }) + await ethBridger.approveGasToken({ + parentSigner: userWallet.connect(parentProvider) + }) + await erc20Bridger.approveGasToken({ + parentSigner: userWallet.connect(parentProvider), + erc20ParentAddress: l1WethAddress + }) + } + // Wrap ETH to test WETH transactions and approve it's usage - await fundWeth('parentChain') - await fundWeth('childChain') + await fundWethOnParentChain() await approveWeth() + if (isCustomFeeToken) { + await approveCustomFeeToken({ + signer: userWallet.connect(parentProvider), + erc20ParentAddress: l1WethAddress + }) + } + + await fundErc20ToChildChain({ + parentSigner: userWallet.connect(parentProvider), + parentErc20Address: l1WethAddress, + amount: utils.parseEther('0.1'), + isCustomFeeToken + }) // Generate activity on chains so that assertions get posted and claims can be made generateActivityOnChains({ @@ -114,7 +178,14 @@ export default defineConfig({ wallet: localWallet }) // Also keep watching assertions since they will act as a proof of activity and claims for withdrawals - checkForAssertions({ parentProvider, isOrbitTest }) + checkForAssertions({ + parentProvider, + testType: isCustomFeeToken + ? 'orbit-custom' + : process.env.E2E_ORBIT === 'true' + ? 'orbit-eth' + : 'regular' + }) // Set Cypress variables config.env.ETH_RPC_URL = isOrbitTest ? arbRpcUrl : ethRpcUrl @@ -127,6 +198,8 @@ export default defineConfig({ config.env.ERC20_TOKEN_ADDRESS_PARENT_CHAIN = l1ERC20Token.address config.env.LOCAL_WALLET_PRIVATE_KEY = localWallet.privateKey config.env.ORBIT_TEST = isOrbitTest ? '1' : '0' + config.env.NATIVE_TOKEN_SYMBOL = isCustomFeeToken ? 'TN' : 'ETH' + config.env.NATIVE_TOKEN_ADDRESS = ethBridger.nativeToken config.env.CUSTOM_DESTINATION_ADDRESS = await getCustomDestinationAddress() @@ -196,9 +269,58 @@ if (!process.env.PRIVATE_KEY_USER) { throw new Error('PRIVATE_KEY_USER variable missing.') } -const localWallet = new Wallet(process.env.PRIVATE_KEY_CUSTOM) +const localWallet = new Wallet( + process.env.E2E_ORBIT_CUSTOM_GAS_TOKEN === 'true' + ? utils.sha256(utils.toUtf8Bytes('user_fee_token_deployer')) + : process.env.PRIVATE_KEY_CUSTOM +) const userWallet = new Wallet(process.env.PRIVATE_KEY_USER) +async function approveCustomFeeToken({ + signer, + erc20ParentAddress, + amount +}: { + signer: Wallet + erc20ParentAddress: string + amount?: BigNumber +}) { + console.log('Approving custom fee token...') + const childErc20Bridger = await Erc20Bridger.fromProvider(childProvider) + + await childErc20Bridger.approveGasToken({ + parentSigner: signer, + erc20ParentAddress, + amount + }) +} + +async function fundUserWalletNativeCurrency() { + const childEthBridger = await EthBridger.fromProvider(childProvider) + + const address = await userWallet.getAddress() + + const tokenContract = TestERC20__factory.connect( + childEthBridger.nativeToken!, + localWallet.connect(parentProvider) + ) + + const userBalance = await tokenContract.balanceOf(address) + const shouldFund = userBalance.lt(utils.parseEther('0.3')) + + if (!shouldFund) { + console.log( + `User wallet has enough L3 native currency for testing, skip funding...` + ) + return + } + + console.log(`Funding native currency to user wallet on L2...`) + + const tx = await tokenContract.transfer(address, utils.parseEther('3')) + await tx.wait() +} + async function deployERC20ToParentChain() { console.log('Deploying ERC20...') const signer = localWallet.connect(parentProvider) @@ -219,6 +341,14 @@ async function deployERC20ToParentChain() { return l1TokenContract } +function isNonZeroAddress(address: string | undefined) { + return ( + typeof address === 'string' && + address !== constants.AddressZero && + utils.isAddress(address) + ) +} + async function deployERC20ToChildChain(erc20L1Address: string) { const bridger = await Erc20Bridger.fromProvider(childProvider) const deploy = await bridger.deposit({ @@ -228,6 +358,15 @@ async function deployERC20ToChildChain(erc20L1Address: string) { childProvider }) await deploy.wait() + + // store deployed weth address + if (erc20L1Address === l1WethAddress) { + l2WethAddress = await getL2ERC20Address({ + erc20L1Address: l1WethAddress, + l1Provider: parentProvider, + l2Provider: childProvider + }) + } } function getWethContract( @@ -237,14 +376,10 @@ function getWethContract( return TestWETH9__factory.connect(tokenAddress, userWallet.connect(provider)) } -async function fundWeth(networkType: NetworkType) { - console.log(`Funding WETH: ${networkType}...`) - const amount = networkType === 'parentChain' ? '0.2' : '0.1' - const address = networkType === 'parentChain' ? l1WethAddress : l2WethAddress - const provider = - networkType === 'parentChain' ? parentProvider : childProvider - const tx = await getWethContract(provider, address).deposit({ - value: utils.parseEther(amount) +async function fundWethOnParentChain() { + console.log(`Funding WETH...`) + const tx = await getWethContract(parentProvider, l1WethAddress).deposit({ + value: utils.parseEther('0.3') }) await tx.wait() } @@ -278,27 +413,39 @@ async function fundErc20ToParentChain(l1ERC20Token: Contract) { await transferTx.wait() } -async function fundErc20ToChildChain(l1ERC20Token: Contract) { - console.log('Funding ERC20 on Child Chain...') - // first deploy the ERC20 to L2 (if not, it might throw a gas error later) - await deployERC20ToChildChain(l1ERC20Token.address) +async function fundErc20ToChildChain({ + parentErc20Address, + parentSigner, + amount, + isCustomFeeToken +}: { + parentErc20Address: string + parentSigner: Wallet + amount: BigNumber + isCustomFeeToken: boolean +}) { + // deploy any token that's not WETH + // only deploy WETH for custom fee token chains because it's not deployed there + if (parentErc20Address !== l1WethAddress || isCustomFeeToken) { + // first deploy the ERC20 to L2 (if not, it might throw a gas error later) + await deployERC20ToChildChain(parentErc20Address) + } + const erc20Bridger = await Erc20Bridger.fromProvider(childProvider) - const parentSigner = localWallet.connect(parentProvider) // approve the ERC20 token for spending const approvalTx = await erc20Bridger.approveToken({ - erc20ParentAddress: l1ERC20Token.address, + erc20ParentAddress: parentErc20Address, parentSigner, amount: constants.MaxUint256 }) await approvalTx.wait() - // deposit the ERC20 token to L2 (fund the L2 account) const depositTx = await erc20Bridger.deposit({ parentSigner, childProvider, - erc20ParentAddress: l1ERC20Token.address, - amount: parseUnits('5', ERC20TokenDecimals), + erc20ParentAddress: parentErc20Address, + amount, destinationAddress: userWallet.address }) const depositRec = await depositTx.wait() diff --git a/packages/arb-token-bridge-ui/tests/e2e/specfiles.json b/packages/arb-token-bridge-ui/tests/e2e/specfiles.json index 8e740d8962..fd907011e2 100644 --- a/packages/arb-token-bridge-ui/tests/e2e/specfiles.json +++ b/packages/arb-token-bridge-ui/tests/e2e/specfiles.json @@ -5,13 +5,13 @@ "recordVideo": "false" }, { - "name": "Deposit ETH", - "file": "tests/e2e/specs/**/depositETH.cy.{js,jsx,ts,tsx}", + "name": "Deposit native token", + "file": "tests/e2e/specs/**/depositNativeToken.cy.{js,jsx,ts,tsx}", "recordVideo": "false" }, { - "name": "Withdraw ETH", - "file": "tests/e2e/specs/**/withdrawETH.cy.{js,jsx,ts,tsx}", + "name": "Withdraw native token", + "file": "tests/e2e/specs/**/withdrawNativeToken.cy.{js,jsx,ts,tsx}", "recordVideo": "false" }, { diff --git a/packages/arb-token-bridge-ui/tests/e2e/specs/approveToken.cy.ts b/packages/arb-token-bridge-ui/tests/e2e/specs/approveToken.cy.ts index ad6c3aa8c4..c96ceb903c 100644 --- a/packages/arb-token-bridge-ui/tests/e2e/specs/approveToken.cy.ts +++ b/packages/arb-token-bridge-ui/tests/e2e/specs/approveToken.cy.ts @@ -2,7 +2,7 @@ import { importTokenThroughUI, ERC20TokenName, ERC20TokenSymbol, - zeroToLessThanOneETH, + getZeroToLessThanOneToken, getL1NetworkName, getL2NetworkName } from '../../support/common' @@ -11,6 +11,10 @@ const ERC20TokenAddressL1 = Cypress.env('ERC20_TOKEN_ADDRESS_PARENT_CHAIN') describe('Approve token for deposit', () => { // log in to metamask + const zeroToLessThanOneEth = getZeroToLessThanOneToken('ETH') + const zeroToLessThanOneNativeToken = getZeroToLessThanOneToken( + Cypress.env('NATIVE_TOKEN_SYMBOL') + ) it('should approve and deposit ERC-20 token', () => { context('Approve token', () => { @@ -26,9 +30,9 @@ describe('Approve token for deposit', () => { cy.findByText('MAX').click() - cy.findGasFeeSummary(zeroToLessThanOneETH) - cy.findGasFeeForChain(getL1NetworkName(), zeroToLessThanOneETH) - cy.findGasFeeForChain(getL2NetworkName(), zeroToLessThanOneETH) + cy.findGasFeeSummary(zeroToLessThanOneEth) + cy.findGasFeeForChain(getL1NetworkName(), zeroToLessThanOneEth) + cy.findGasFeeForChain(getL2NetworkName(), zeroToLessThanOneNativeToken) cy.waitUntil(() => cy.findMoveFundsButton().should('not.be.disabled'), { errorMsg: 'move funds button is disabled (expected to be enabled)', diff --git a/packages/arb-token-bridge-ui/tests/e2e/specs/batchDeposit.cy.ts b/packages/arb-token-bridge-ui/tests/e2e/specs/batchDeposit.cy.ts index 08f768bc31..6f002c0385 100644 --- a/packages/arb-token-bridge-ui/tests/e2e/specs/batchDeposit.cy.ts +++ b/packages/arb-token-bridge-ui/tests/e2e/specs/batchDeposit.cy.ts @@ -6,7 +6,7 @@ import { getL1NetworkName, getL2NetworkConfig, getL2NetworkName, - zeroToLessThanOneETH + getZeroToLessThanOneToken } from '../../support/common' import { formatAmount } from '../../../src/util/NumberUtils' @@ -16,6 +16,11 @@ describe('Batch Deposit', () => { childNativeTokenBalance, childErc20Balance: string + const nativeTokenSymbol = Cypress.env('NATIVE_TOKEN_SYMBOL') + const zeroToLessThanOneEth = getZeroToLessThanOneToken('ETH') + const zeroToLessThanOneNativeToken = + getZeroToLessThanOneToken(nativeTokenSymbol) + beforeEach(() => { getInitialERC20Balance({ tokenAddress: Cypress.env('ERC20_TOKEN_ADDRESS_CHILD_CHAIN'), @@ -49,7 +54,7 @@ describe('Batch Deposit', () => { }) cy.findSourceChainButton(getL1NetworkName()) cy.findDestinationChainButton(getL2NetworkName()) - cy.findSelectTokenButton('ETH') + cy.findSelectTokenButton(nativeTokenSymbol) }) it('should deposit erc-20 and native currency to the same address', () => { @@ -81,7 +86,9 @@ describe('Batch Deposit', () => { }) context('native currency balance on child chain should not exist', () => { - cy.findByLabelText(`ETH balance amount on childChain`).should('not.exist') + cy.findByLabelText( + `${nativeTokenSymbol} balance amount on childChain` + ).should('not.exist') }) context('amount2 input should not exist', () => { @@ -99,7 +106,7 @@ describe('Batch Deposit', () => { }) context('native currency balance on child chain should show', () => { - cy.findByLabelText(`ETH balance amount on childChain`) + cy.findByLabelText(`${nativeTokenSymbol} balance amount on childChain`) .should('be.visible') .contains(childNativeTokenBalance) }) @@ -111,14 +118,14 @@ describe('Batch Deposit', () => { context('should show gas estimations and summary', () => { cy.typeAmount(ERC20AmountToSend) cy.typeAmount2(nativeCurrencyAmountToSend) - cy.findGasFeeSummary(zeroToLessThanOneETH) - cy.findGasFeeForChain(getL1NetworkName(), zeroToLessThanOneETH) - cy.findGasFeeForChain(getL2NetworkName(), zeroToLessThanOneETH) + cy.findGasFeeSummary(zeroToLessThanOneEth) + cy.findGasFeeForChain(getL1NetworkName(), zeroToLessThanOneEth) + cy.findGasFeeForChain(getL2NetworkName(), zeroToLessThanOneNativeToken) }) const txData = { symbol: ERC20TokenSymbol, - symbol2: 'ETH', + symbol2: nativeTokenSymbol, amount: ERC20AmountToSend, amount2: nativeCurrencyAmountToSend } @@ -154,7 +161,7 @@ describe('Batch Deposit', () => { .invoke('text') .then(parseFloat) .should('be.gt', Number(parentErc20Balance)) - cy.findByLabelText(`ETH balance amount on childChain`) + cy.findByLabelText(`${nativeTokenSymbol} balance amount on childChain`) .invoke('text') .then(parseFloat) .should( @@ -217,14 +224,14 @@ describe('Batch Deposit', () => { context('should show gas estimations and summary', () => { cy.typeAmount(ERC20AmountToSend) cy.typeAmount2(nativeCurrencyAmountToSend) - cy.findGasFeeSummary(zeroToLessThanOneETH) - cy.findGasFeeForChain(getL1NetworkName(), zeroToLessThanOneETH) - cy.findGasFeeForChain(getL2NetworkName(), zeroToLessThanOneETH) + cy.findGasFeeSummary(zeroToLessThanOneEth) + cy.findGasFeeForChain(getL1NetworkName(), zeroToLessThanOneEth) + cy.findGasFeeForChain(getL2NetworkName(), zeroToLessThanOneNativeToken) }) const txData = { symbol: ERC20TokenSymbol, - symbol2: 'ETH', + symbol2: nativeTokenSymbol, amount: ERC20AmountToSend, amount2: nativeCurrencyAmountToSend } diff --git a/packages/arb-token-bridge-ui/tests/e2e/specs/depositCctp.cy.ts b/packages/arb-token-bridge-ui/tests/e2e/specs/depositCctp.cy.ts index f1d39bbf2d..f655a787ce 100644 --- a/packages/arb-token-bridge-ui/tests/e2e/specs/depositCctp.cy.ts +++ b/packages/arb-token-bridge-ui/tests/e2e/specs/depositCctp.cy.ts @@ -2,9 +2,8 @@ * When user wants to bridge USDC through CCTP from L1 to L2 */ -import { zeroToLessThanOneETH } from '../../support/common' +import { getZeroToLessThanOneToken } from '../../support/common' import { CommonAddress } from '../../../src/util/CommonAddressUtils' -import { formatAmount } from 'packages/arb-token-bridge-ui/src/util/NumberUtils' // common function for this cctp deposit const confirmAndApproveCctpDeposit = () => { @@ -65,6 +64,7 @@ const confirmAndApproveCctpDeposit = () => { describe('Deposit USDC through CCTP', () => { // Happy Path const USDCAmountToSend = 0.0001 + const zeroToLessThanOneETH = getZeroToLessThanOneToken('ETH') beforeEach(() => { cy.login({ networkType: 'parentChain', networkName: 'sepolia' }) diff --git a/packages/arb-token-bridge-ui/tests/e2e/specs/depositERC20.cy.ts b/packages/arb-token-bridge-ui/tests/e2e/specs/depositERC20.cy.ts index 2568bfbef9..1cd9ac5c13 100644 --- a/packages/arb-token-bridge-ui/tests/e2e/specs/depositERC20.cy.ts +++ b/packages/arb-token-bridge-ui/tests/e2e/specs/depositERC20.cy.ts @@ -6,7 +6,7 @@ import { formatAmount } from '../../../src/util/NumberUtils' import { getInitialERC20Balance, getL1NetworkConfig, - zeroToLessThanOneETH, + getZeroToLessThanOneToken, moreThanZeroBalance, getL1NetworkName, getL2NetworkName, @@ -33,6 +33,10 @@ describe('Deposit Token', () => { const isOrbitTest = Cypress.env('ORBIT_TEST') == '1' const depositTime = isOrbitTest ? 'Less than a minute' : '9 minutes' + const nativeTokenSymbol = Cypress.env('NATIVE_TOKEN_SYMBOL') + const zeroToLessThanOneEth = getZeroToLessThanOneToken('ETH') + const zeroToLessThanOneNativeToken = + getZeroToLessThanOneToken(nativeTokenSymbol) // Happy Path Object.keys(depositTestCases).forEach(tokenType => { @@ -50,11 +54,11 @@ describe('Deposit Token', () => { }).then(val => (l1ERC20bal = formatAmount(val))) }) - it('should show L1 and L2 chains, and ETH correctly', () => { + it('should show L1 and L2 chains, and native token correctly', () => { cy.login({ networkType: 'parentChain' }) cy.findSourceChainButton(getL1NetworkName()) cy.findDestinationChainButton(getL2NetworkName()) - cy.findSelectTokenButton('ETH') + cy.findSelectTokenButton(nativeTokenSymbol) }) it(`should deposit ${tokenType} successfully to the same address`, () => { @@ -76,9 +80,12 @@ describe('Deposit Token', () => { context('should show gas estimations', () => { cy.typeAmount(ERC20AmountToSend) - cy.findGasFeeSummary(zeroToLessThanOneETH) - cy.findGasFeeForChain(getL1NetworkName(), zeroToLessThanOneETH) - cy.findGasFeeForChain(getL2NetworkName(), zeroToLessThanOneETH) + cy.findGasFeeSummary(zeroToLessThanOneEth) + cy.findGasFeeForChain(getL1NetworkName(), zeroToLessThanOneEth) + cy.findGasFeeForChain( + getL2NetworkName(), + zeroToLessThanOneNativeToken + ) }) context('should deposit successfully', () => { @@ -111,9 +118,12 @@ describe('Deposit Token', () => { context('should show summary', () => { cy.typeAmount(ERC20AmountToSend) - cy.findGasFeeSummary(zeroToLessThanOneETH) - cy.findGasFeeForChain(getL1NetworkName(), zeroToLessThanOneETH) - cy.findGasFeeForChain(getL2NetworkName(), zeroToLessThanOneETH) + cy.findGasFeeSummary(zeroToLessThanOneEth) + cy.findGasFeeForChain(getL1NetworkName(), zeroToLessThanOneEth) + cy.findGasFeeForChain( + getL2NetworkName(), + zeroToLessThanOneNativeToken + ) }) context('should fill custom destination address successfully', () => { diff --git a/packages/arb-token-bridge-ui/tests/e2e/specs/depositETH.cy.ts b/packages/arb-token-bridge-ui/tests/e2e/specs/depositNativeToken.cy.ts similarity index 73% rename from packages/arb-token-bridge-ui/tests/e2e/specs/depositETH.cy.ts rename to packages/arb-token-bridge-ui/tests/e2e/specs/depositNativeToken.cy.ts index 5de9ab52a4..7e674337b4 100644 --- a/packages/arb-token-bridge-ui/tests/e2e/specs/depositETH.cy.ts +++ b/packages/arb-token-bridge-ui/tests/e2e/specs/depositNativeToken.cy.ts @@ -1,15 +1,19 @@ /** - * When user wants to bridge ETH from L1 to L2 + * When user wants to bridge native token from L1 to L2 */ import { getL1NetworkName, getL2NetworkName, - zeroToLessThanOneETH + getZeroToLessThanOneToken } from '../../support/common' -describe('Deposit ETH', () => { +describe('Deposit native token', () => { const ETHAmountToDeposit = 0.0001 + const nativeTokenSymbol = Cypress.env('NATIVE_TOKEN_SYMBOL') + const zeroToLessThanOneEth = getZeroToLessThanOneToken('ETH') + const zeroToLessThanOneNativeToken = + getZeroToLessThanOneToken(nativeTokenSymbol) const isOrbitTest = Cypress.env('ORBIT_TEST') == '1' const depositTime = isOrbitTest ? 'Less than a minute' : '9 minutes' @@ -24,15 +28,15 @@ describe('Deposit ETH', () => { it('should show gas estimations and bridge successfully', () => { cy.login({ networkType: 'parentChain' }) cy.typeAmount(ETHAmountToDeposit) - cy.findGasFeeSummary(zeroToLessThanOneETH) - cy.findGasFeeForChain(getL1NetworkName(), zeroToLessThanOneETH) - cy.findGasFeeForChain(getL2NetworkName(), zeroToLessThanOneETH) + cy.findGasFeeSummary(zeroToLessThanOneEth) + cy.findGasFeeForChain(getL1NetworkName(), zeroToLessThanOneEth) + cy.findGasFeeForChain(getL2NetworkName(), zeroToLessThanOneNativeToken) cy.findMoveFundsButton().click() cy.confirmMetamaskTransaction() cy.findTransactionInTransactionHistory({ duration: depositTime, amount: ETHAmountToDeposit, - symbol: 'ETH' + symbol: nativeTokenSymbol }) cy.closeTransactionHistoryPanel() cy.findAmountInput().should('have.value', '') diff --git a/packages/arb-token-bridge-ui/tests/e2e/specs/importToken.cy.ts b/packages/arb-token-bridge-ui/tests/e2e/specs/importToken.cy.ts index 3946254dbf..86c712c350 100644 --- a/packages/arb-token-bridge-ui/tests/e2e/specs/importToken.cy.ts +++ b/packages/arb-token-bridge-ui/tests/e2e/specs/importToken.cy.ts @@ -15,6 +15,7 @@ const ERC20TokenAddressL2: string = Cypress.env( ) describe('Import token', () => { + const nativeTokenSymbol = Cypress.env('NATIVE_TOKEN_SYMBOL') // we use mainnet to test token lists context('User import token through UI', () => { @@ -136,7 +137,7 @@ describe('Import token', () => { const addressWithoutLastChar = ERC20TokenAddressL1.slice(0, -1) // Remove the last character cy.login({ networkType: 'parentChain' }) - cy.findSelectTokenButton('ETH').click() + cy.findSelectTokenButton(nativeTokenSymbol).click() // open the Select Token popup cy.findByPlaceholderText(/Search by token name/i) @@ -246,6 +247,8 @@ describe('Import token', () => { visitAfterSomeDelay('/', { qs: { + sourceChain: 'arbitrum-localhost', + destinationChain: 'l3-localhost', token: invalidTokenAddress } }) @@ -261,7 +264,7 @@ describe('Import token', () => { .trigger('click', { force: true }) - cy.findSelectTokenButton('ETH') + cy.findSelectTokenButton(nativeTokenSymbol) // Modal is closed cy.findByRole('button', { name: 'Dialog Cancel' }).should('not.exist') diff --git a/packages/arb-token-bridge-ui/tests/e2e/specs/login.cy.ts b/packages/arb-token-bridge-ui/tests/e2e/specs/login.cy.ts index 0b6286f7e3..5c627e2344 100644 --- a/packages/arb-token-bridge-ui/tests/e2e/specs/login.cy.ts +++ b/packages/arb-token-bridge-ui/tests/e2e/specs/login.cy.ts @@ -4,7 +4,9 @@ import { formatAmount } from '../../../src/util/NumberUtils' import { + getInitialERC20Balance, getInitialETHBalance, + getL1NetworkConfig, getL1NetworkName, getL2NetworkName } from './../../support/common' @@ -13,12 +15,24 @@ describe('Login Account', () => { let l1ETHbal let l2ETHbal + const nativeTokenSymbol = Cypress.env('NATIVE_TOKEN_SYMBOL') + const isCustomFeeToken = nativeTokenSymbol !== 'ETH' + before(() => { - getInitialETHBalance(Cypress.env('ETH_RPC_URL')).then( - val => (l1ETHbal = formatAmount(val)) - ) + if (isCustomFeeToken) { + getInitialERC20Balance({ + tokenAddress: Cypress.env('NATIVE_TOKEN_ADDRESS'), + multiCallerAddress: getL1NetworkConfig().multiCall, + address: Cypress.env('ADDRESS'), + rpcURL: Cypress.env('ETH_RPC_URL') + }).then(val => (l1ETHbal = formatAmount(val))) + } else { + getInitialETHBalance(Cypress.env('ETH_RPC_URL')).then( + val => (l1ETHbal = formatAmount(val)) + ) + } getInitialETHBalance(Cypress.env('ARB_RPC_URL')).then( - val => (l2ETHbal = formatAmount(val, { symbol: 'ETH' })) + val => (l2ETHbal = formatAmount(val)) ) }) @@ -33,16 +47,13 @@ describe('Login Account', () => { it('should connect wallet using MetaMask and display L1 and L2 balances', () => { cy.login({ networkType: 'parentChain' }) - // Balance: is in a different element so we check for siblings - cy.findByText(l1ETHbal) + cy.findByLabelText(`${nativeTokenSymbol} balance amount on parentChain`) .should('be.visible') - .siblings() - .contains('Balance: ') - // Balance: is in a different element so we check for siblings - cy.findByText(l2ETHbal) + .contains(l1ETHbal) + cy.findByLabelText(`${nativeTokenSymbol} balance amount on childChain`) .should('be.visible') - .siblings() - .contains('Balance: ') + .contains(l2ETHbal) + cy.findSourceChainButton(getL1NetworkName()) cy.findDestinationChainButton(getL2NetworkName()) }) diff --git a/packages/arb-token-bridge-ui/tests/e2e/specs/readClassicDeposits.cy.ts b/packages/arb-token-bridge-ui/tests/e2e/specs/readClassicDeposits.cy.ts index 6725952d49..727778240c 100644 --- a/packages/arb-token-bridge-ui/tests/e2e/specs/readClassicDeposits.cy.ts +++ b/packages/arb-token-bridge-ui/tests/e2e/specs/readClassicDeposits.cy.ts @@ -22,7 +22,7 @@ function mockClassicDepositTransaction( childChainId: 42161, status: 'success', isClassic: true, - assetName: 'ETH', + assetName: Cypress.env('NATIVE_TOKEN_SYMBOL'), assetType: AssetType.ETH, sender: Cypress.env('ADDRESS'), l1NetworkID: '1', @@ -44,8 +44,8 @@ describe('Read classic deposit messages', () => { window.localStorage.clear() }) - context('User has classic ETH deposit transaction', () => { - it('can read successful ETH deposit', () => { + context('User has classic native token deposit transaction', () => { + it('can read successful native token deposit', () => { // log in to metamask cy.login({ networkType: 'parentChain', diff --git a/packages/arb-token-bridge-ui/tests/e2e/specs/withdrawERC20.cy.ts b/packages/arb-token-bridge-ui/tests/e2e/specs/withdrawERC20.cy.ts index 54233ff35b..bbc0ff696f 100644 --- a/packages/arb-token-bridge-ui/tests/e2e/specs/withdrawERC20.cy.ts +++ b/packages/arb-token-bridge-ui/tests/e2e/specs/withdrawERC20.cy.ts @@ -9,7 +9,7 @@ import { getL2NetworkConfig, getL1NetworkName, getL2NetworkName, - zeroToLessThanOneETH, + getZeroToLessThanOneToken, ERC20TokenSymbol } from '../../support/common' @@ -27,6 +27,9 @@ const withdrawalTestCases = { } describe('Withdraw ERC20 Token', () => { + const nativeTokenSymbol = Cypress.env('NATIVE_TOKEN_SYMBOL') + const zeroToLessThanOneNativeToken = + getZeroToLessThanOneToken(nativeTokenSymbol) let ERC20AmountToSend = Number((Math.random() * 0.001).toFixed(5)) // randomize the amount to be sure that previous transactions are not checked in e2e // when all of our tests need to run in a logged-in state // we have to make sure we preserve a healthy LocalStorage state @@ -70,7 +73,7 @@ describe('Withdraw ERC20 Token', () => { cy.findSourceChainButton(getL2NetworkName()) cy.findDestinationChainButton(getL1NetworkName()) cy.findMoveFundsButton().should('be.disabled') - cy.findSelectTokenButton('ETH') + cy.findSelectTokenButton(nativeTokenSymbol) }) it(`should withdraw ${tokenType} to the same address successfully`, () => { @@ -86,8 +89,11 @@ describe('Withdraw ERC20 Token', () => { context('should show summary', () => { cy.typeAmount(ERC20AmountToSend) - cy.findGasFeeSummary(zeroToLessThanOneETH) - cy.findGasFeeForChain(getL2NetworkName(), zeroToLessThanOneETH) + cy.findGasFeeSummary(zeroToLessThanOneNativeToken) + cy.findGasFeeForChain( + getL2NetworkName(), + zeroToLessThanOneNativeToken + ) cy.findGasFeeForChain( new RegExp( `You'll have to pay ${getL1NetworkName()} gas fee upon claiming.`, @@ -196,8 +202,11 @@ describe('Withdraw ERC20 Token', () => { context('should show summary', () => { cy.typeAmount(ERC20AmountToSend) - cy.findGasFeeSummary(zeroToLessThanOneETH) - cy.findGasFeeForChain(getL2NetworkName(), zeroToLessThanOneETH) + cy.findGasFeeSummary(zeroToLessThanOneNativeToken) + cy.findGasFeeForChain( + getL2NetworkName(), + zeroToLessThanOneNativeToken + ) cy.findGasFeeForChain( new RegExp( `You'll have to pay ${getL1NetworkName()} gas fee upon claiming.`, diff --git a/packages/arb-token-bridge-ui/tests/e2e/specs/withdrawETH.cy.ts b/packages/arb-token-bridge-ui/tests/e2e/specs/withdrawETH.cy.ts deleted file mode 100644 index 1f45382c16..0000000000 --- a/packages/arb-token-bridge-ui/tests/e2e/specs/withdrawETH.cy.ts +++ /dev/null @@ -1,148 +0,0 @@ -/** - * When user wants to bridge ETH from L2 to L1 - */ - -import { - getInitialETHBalance, - getL1NetworkName, - getL2NetworkName, - zeroToLessThanOneETH -} from '../../support/common' -import { formatAmount } from '../../../src/util/NumberUtils' - -describe('Withdraw ETH', () => { - let ETHToWithdraw = Number((Math.random() * 0.001).toFixed(5)) // randomize the amount to be sure that previous transactions are not checked in e2e - let l1EthBal: string - - beforeEach(() => { - getInitialETHBalance( - Cypress.env('ETH_RPC_URL'), - Cypress.env('ADDRESS') - ).then( - val => - (l1EthBal = formatAmount(val, { - symbol: 'ETH' - })) - ) - }) - - // Happy Path - context('user has some ETH and is on L2', () => { - it('should show form fields correctly', () => { - cy.login({ networkType: 'childChain' }) - cy.findSourceChainButton(getL2NetworkName()) - cy.findDestinationChainButton(getL1NetworkName()) - cy.findMoveFundsButton().should('be.disabled') - }) - - context("bridge amount is lower than user's L2 ETH balance value", () => { - it('should show gas estimations', () => { - cy.login({ networkType: 'childChain' }) - cy.typeAmount(ETHToWithdraw) - cy.findGasFeeSummary(zeroToLessThanOneETH) - cy.findGasFeeForChain(getL2NetworkName(), zeroToLessThanOneETH) - cy.findGasFeeForChain( - new RegExp( - `You'll have to pay ${getL1NetworkName()} gas fee upon claiming.`, - 'i' - ) - ) - }) - - it('should show withdrawal confirmation and withdraw', () => { - ETHToWithdraw = Number((Math.random() * 0.001).toFixed(5)) // generate a new withdrawal amount for each test-run attempt so that findAllByText doesn't stall coz of prev transactions - cy.login({ networkType: 'childChain' }) - cy.typeAmount(ETHToWithdraw) - cy.findMoveFundsButton().click() - cy.findByText(/Arbitrum’s bridge/i).should('be.visible') - - // the Continue withdrawal button should be disabled at first - cy.findByRole('button', { - name: /Continue/i - }).should('be.disabled') - - cy.findByRole('switch', { - name: /before I can claim my funds/i - }) - .should('be.visible') - .click() - - cy.findByRole('switch', { - name: /after claiming my funds/i - }) - .should('be.visible') - .click() - // the Continue withdrawal button should not be disabled now - cy.findByRole('button', { - name: /Continue/i - }) - .should('be.enabled') - .click() - - cy.confirmMetamaskTransaction() - - cy.findTransactionInTransactionHistory({ - duration: 'an hour', - amount: ETHToWithdraw, - symbol: 'ETH' - }) - - context('transfer panel amount should be reset', () => { - cy.closeTransactionHistoryPanel() - cy.findAmountInput().should('have.value', '') - cy.findMoveFundsButton().should('be.disabled') - }) - }) - - it('should claim funds', { defaultCommandTimeout: 200_000 }, () => { - // increase the timeout for this test as claim button can take ~(20 blocks *10 blocks/sec) to activate - cy.login({ networkType: 'parentChain' }) // login to L1 to claim the funds (otherwise would need to change network after clicking on claim) - - cy.findByLabelText('Open Transaction History') - .should('be.visible') - .click() - - cy.findClaimButton( - formatAmount(ETHToWithdraw, { - symbol: 'ETH' - }) - ).click() - - cy.confirmMetamaskTransaction() - - cy.findByLabelText('show settled transactions') - .should('be.visible') - .click() - - cy.findByText( - `${formatAmount(ETHToWithdraw, { - symbol: 'ETH' - })}` - ).should('be.visible') - - cy.closeTransactionHistoryPanel() - - // the balance on the destination chain should not be the same as before - cy.findByLabelText('ETH balance amount on parentChain') - .should('be.visible') - .its('text') - .should('not.eq', l1EthBal) - }) - }) - - // TODO => test for bridge amount higher than user's L2 ETH balance - }) - - // TODO - will have both cases: - // 1. Arbitrum network is not added to metamask yet (add + switch) - // 2. Arbitrum network already configured in metamask (only switch) - context('user has some ETH and is on L1', () => {}) - // TODO - context('user has some ETH and is on wrong chain', () => {}) - // TODO - context('user has 0 ETH and is on L1', () => {}) - // TODO - context('user has 0 ETH and is on L2', () => {}) - // TODO - context('user has 0 ETH and is on wrong chain', () => {}) -}) diff --git a/packages/arb-token-bridge-ui/tests/e2e/specs/withdrawNativeToken.cy.ts b/packages/arb-token-bridge-ui/tests/e2e/specs/withdrawNativeToken.cy.ts new file mode 100644 index 0000000000..bee9061c24 --- /dev/null +++ b/packages/arb-token-bridge-ui/tests/e2e/specs/withdrawNativeToken.cy.ts @@ -0,0 +1,159 @@ +/** + * When user wants to bridge native token from L2 to L1 + */ + +import { + getInitialETHBalance, + getL1NetworkName, + getL2NetworkName, + getZeroToLessThanOneToken +} from '../../support/common' +import { formatAmount } from '../../../src/util/NumberUtils' + +describe('Withdraw native token', () => { + const nativeTokenSymbol = Cypress.env('NATIVE_TOKEN_SYMBOL') + const zeroToLessThanOneNativeToken = + getZeroToLessThanOneToken(nativeTokenSymbol) + let ETHToWithdraw = Number((Math.random() * 0.001).toFixed(5)) // randomize the amount to be sure that previous transactions are not checked in e2e + let l1EthBal: string + + beforeEach(() => { + getInitialETHBalance( + Cypress.env('ETH_RPC_URL'), + Cypress.env('ADDRESS') + ).then( + val => + (l1EthBal = formatAmount(val, { + symbol: nativeTokenSymbol + })) + ) + }) + + // Happy Path + context('user has some native token and is on L2', () => { + it('should show form fields correctly', () => { + cy.login({ networkType: 'childChain' }) + cy.findSourceChainButton(getL2NetworkName()) + cy.findDestinationChainButton(getL1NetworkName()) + cy.findMoveFundsButton().should('be.disabled') + }) + + context( + "bridge amount is lower than user's L2 native token balance value", + () => { + it('should show gas estimations', () => { + cy.login({ networkType: 'childChain' }) + cy.typeAmount(ETHToWithdraw) + cy.findGasFeeSummary(zeroToLessThanOneNativeToken) + cy.findGasFeeForChain( + getL2NetworkName(), + zeroToLessThanOneNativeToken + ) + cy.findGasFeeForChain( + new RegExp( + `You'll have to pay ${getL1NetworkName()} gas fee upon claiming.`, + 'i' + ) + ) + }) + + it('should show withdrawal confirmation and withdraw', () => { + ETHToWithdraw = Number((Math.random() * 0.001).toFixed(5)) // generate a new withdrawal amount for each test-run attempt so that findAllByText doesn't stall coz of prev transactions + cy.login({ networkType: 'childChain' }) + cy.typeAmount(ETHToWithdraw) + cy.findMoveFundsButton().click() + cy.findByText(/Arbitrum’s bridge/i).should('be.visible') + + // the Continue withdrawal button should be disabled at first + cy.findByRole('button', { + name: /Continue/i + }).should('be.disabled') + + cy.findByRole('switch', { + name: /before I can claim my funds/i + }) + .should('be.visible') + .click() + + cy.findByRole('switch', { + name: /after claiming my funds/i + }) + .should('be.visible') + .click() + // the Continue withdrawal button should not be disabled now + cy.findByRole('button', { + name: /Continue/i + }) + .should('be.enabled') + .click() + + cy.confirmMetamaskTransaction() + + cy.findTransactionInTransactionHistory({ + duration: 'an hour', + amount: ETHToWithdraw, + symbol: nativeTokenSymbol + }) + + context('transfer panel amount should be reset', () => { + cy.closeTransactionHistoryPanel() + cy.findAmountInput().should('have.value', '') + cy.findMoveFundsButton().should('be.disabled') + }) + }) + + it('should claim funds', { defaultCommandTimeout: 200_000 }, () => { + // increase the timeout for this test as claim button can take ~(20 blocks *10 blocks/sec) to activate + cy.login({ networkType: 'parentChain' }) // login to L1 to claim the funds (otherwise would need to change network after clicking on claim) + + cy.findByLabelText('Open Transaction History') + .should('be.visible') + .click() + + cy.findClaimButton( + formatAmount(ETHToWithdraw, { + symbol: nativeTokenSymbol + }) + ).click() + + cy.confirmMetamaskTransaction() + + cy.findByLabelText('show settled transactions') + .should('be.visible') + .click() + + cy.findByText( + `${formatAmount(ETHToWithdraw, { + symbol: nativeTokenSymbol + })}` + ).should('be.visible') + + cy.closeTransactionHistoryPanel() + + // the balance on the destination chain should not be the same as before + cy.findByLabelText( + `${nativeTokenSymbol} balance amount on parentChain` + ) + .should('be.visible') + .its('text') + .should('not.eq', l1EthBal) + }) + } + ) + + // TODO => test for bridge amount higher than user's L2 ETH balance + }) + + // TODO - will have both cases: + // 1. Arbitrum network is not added to metamask yet (add + switch) + // 2. Arbitrum network already configured in metamask (only switch) + context('user has some ETH and is on L1', () => {}) + // TODO + context('user has some ETH and is on wrong chain', () => {}) + // TODO + context('user has 0 ETH and is on L1', () => {}) + // TODO + context('user has 0 ETH and is on L2', () => {}) + // TODO + context('user has 0 ETH and is on wrong chain', () => {}) +}) diff --git a/packages/arb-token-bridge-ui/tests/support/commands.ts b/packages/arb-token-bridge-ui/tests/support/commands.ts index 767668fb92..a3ae2a0dea 100644 --- a/packages/arb-token-bridge-ui/tests/support/commands.ts +++ b/packages/arb-token-bridge-ui/tests/support/commands.ts @@ -140,8 +140,8 @@ export const searchAndSelectToken = ({ tokenName: string tokenAddress: string }) => { - // Click on the ETH dropdown (Select token button) - cy.findSelectTokenButton('ETH').click() + // Click on the native token dropdown (Select token button) + cy.findSelectTokenButton(Cypress.env('NATIVE_TOKEN_SYMBOL') ?? 'ETH').click() // open the Select Token popup cy.findByPlaceholderText(/Search by token name/i) diff --git a/packages/arb-token-bridge-ui/tests/support/common.ts b/packages/arb-token-bridge-ui/tests/support/common.ts index 575bba6e4d..76664a8e91 100644 --- a/packages/arb-token-bridge-ui/tests/support/common.ts +++ b/packages/arb-token-bridge-ui/tests/support/common.ts @@ -6,7 +6,11 @@ import { Provider, StaticJsonRpcProvider } from '@ethersproject/providers' import { BigNumber, Signer, Wallet, ethers, utils } from 'ethers' import { MultiCaller } from '@arbitrum/sdk' import { MULTICALL_TESTNET_ADDRESS } from '../../src/constants' -import { defaultL2Network, defaultL3Network } from '../../src/util/networks' +import { + defaultL2Network, + defaultL3Network, + defaultL3CustomGasTokenNetwork +} from '../../src/util/networks' import { getChainIdFromProvider } from '../../src/token-bridge-sdk/utils' export type NetworkType = 'parentChain' | 'childChain' @@ -54,15 +58,21 @@ export const getL1NetworkConfig = (): NetworkConfig => { export const getL2NetworkConfig = (): NetworkConfig => { const isOrbitTest = Cypress.env('ORBIT_TEST') == '1' + const nativeTokenSymbol = Cypress.env('NATIVE_TOKEN_SYMBOL') ?? 'ETH' + const isCustomFeeToken = nativeTokenSymbol !== 'ETH' + + const l3Network = isCustomFeeToken + ? defaultL3CustomGasTokenNetwork + : defaultL3Network return { networkName: isOrbitTest ? 'l3-localhost' : 'arbitrum-localhost', rpcUrl: Cypress.env('ARB_RPC_URL'), chainId: isOrbitTest ? 333333 : 412346, - symbol: 'ETH', + symbol: nativeTokenSymbol, isTestnet: true, multiCall: isOrbitTest - ? defaultL3Network.tokenBridge!.childMultiCall + ? l3Network.tokenBridge!.childMultiCall : defaultL2Network.tokenBridge!.childMultiCall } } @@ -92,11 +102,14 @@ export const getL2TestnetNetworkConfig = (): NetworkConfig => { export const ERC20TokenName = 'Test Arbitrum Token' export const ERC20TokenSymbol = 'TESTARB' export const ERC20TokenDecimals = 18 -export const invalidTokenAddress = '0x0000000000000000000000000000000000000000' +export const invalidTokenAddress = utils.computeAddress(utils.randomBytes(32)) -export const zeroToLessThanOneETH = /0(\.\d+)*( ETH)/ export const moreThanZeroBalance = /0(\.\d+)/ +export function getZeroToLessThanOneToken(symbol: string) { + return new RegExp(`0(\\.\\d+)*( ${symbol})`) +} + export const importTokenThroughUI = (address: string) => { // Click on the ETH dropdown (Select token button) cy.findSelectTokenButton('ETH').click() @@ -229,19 +242,28 @@ export async function generateActivityOnChains({ export async function checkForAssertions({ parentProvider, - isOrbitTest + testType }: { parentProvider: Provider - isOrbitTest: boolean + testType: 'regular' | 'orbit-eth' | 'orbit-custom' }) { const abi = [ 'function latestConfirmed() public view returns (uint64)', 'function latestNodeCreated() public view returns (uint64)' ] - const rollupAddress = isOrbitTest - ? defaultL3Network.ethBridge.rollup - : defaultL2Network.ethBridge.rollup + let rollupAddress: string + + switch (testType) { + case 'orbit-eth': + rollupAddress = defaultL3Network.ethBridge.rollup + break + case 'orbit-custom': + rollupAddress = defaultL3CustomGasTokenNetwork.ethBridge.rollup + break + default: + rollupAddress = defaultL2Network.ethBridge.rollup + } const rollupContract = new ethers.Contract(rollupAddress, abi, parentProvider)