import React from 'react'; import configureMockStore from 'redux-mock-store'; import { act, fireEvent } from '@testing-library/react'; import thunk from 'redux-thunk'; import { NetworkType } from '@metamask/controller-utils'; import { NetworkStatus } from '@metamask/network-controller'; import { renderWithProvider } from '../../../test/lib/render-helpers'; import { KeyringType } from '../../../shared/constants/keyring'; import TokenAllowance from './token-allowance'; const testTokenAddress = '0xC011a73ee8576Fb46F5E1c5751cA3B9Fe0af2a6F'; const state = { appState: { customTokenAmount: '1', }, metamask: { accounts: { '0x0dcd5d886577d5081b0c52e242ef29e70be3e7bc': { address: '0x0dcd5d886577d5081b0c52e242ef29e70be3e7bc', balance: '0x0', }, }, gasEstimateType: 'none', selectedAddress: '0x0dcd5d886577d5081b0c52e242ef29e70be3e7bc', identities: { '0x0dcd5d886577d5081b0c52e242ef29e70be3e7bc': { address: '0x0dcd5d886577d5081b0c52e242ef29e70be3e7bc', name: 'Account 1', }, '0xc42edfcc21ed14dda456aa0756c153f7985d8813': { address: '0xc42edfcc21ed14dda456aa0756c153f7985d8813', name: 'Account 2', }, }, cachedBalances: {}, addressBook: [ { address: '0xc42edfcc21ed14dda456aa0756c153f7985d8813', chainId: '0x5', isEns: false, memo: '', name: 'Address Book Account 1', }, ], providerConfig: { type: 'mainnet', nickname: '', }, selectedNetworkClientId: NetworkType.mainnet, networksMetadata: { [NetworkType.mainnet]: { EIPS: { 1559: true }, status: NetworkStatus.Available, }, }, preferences: { showFiatInTestnets: true, }, knownMethodData: {}, tokens: [ { address: testTokenAddress, symbol: 'SNX', decimals: 18, image: 'testImage', isERC721: false, }, { address: '0xaD6D458402F60fD3Bd25163575031ACDce07538U', symbol: 'DAU', decimals: 18, image: null, isERC721: false, }, ], transactions: [], keyringTypes: [], keyrings: [ { type: KeyringType.hdKeyTree, accounts: ['0x0dcd5d886577d5081b0c52e242ef29e70be3e7bc'], }, ], nextNonce: 1, customNonceValue: '', }, history: { mostRecentOverviewPage: '/', }, confirmTransaction: { txData: {}, }, send: { draftTransactions: {}, }, }; const mockShowModal = jest.fn(); jest.mock('../../store/actions', () => ({ disconnectGasFeeEstimatePoller: jest.fn(), getGasFeeTimeEstimate: jest.fn().mockImplementation(() => Promise.resolve()), getGasFeeEstimatesAndStartPolling: jest .fn() .mockImplementation(() => Promise.resolve()), addPollingTokenToAppState: jest.fn(), removePollingTokenFromAppState: jest.fn(), updateTransactionGasFees: () => ({ type: 'UPDATE_TRANSACTION_PARAMS' }), updatePreviousGasParams: () => ({ type: 'UPDATE_TRANSACTION_PARAMS' }), createTransactionEventFragment: jest.fn(), getNextNonce: () => jest.fn(), showModal: () => mockShowModal, updateCustomNonce: () => ({ type: 'UPDATE_TRANSACTION_PARAMS' }), estimateGas: jest.fn().mockImplementation(() => Promise.resolve()), })); jest.mock('../../contexts/gasFee', () => ({ useGasFeeContext: () => ({ maxPriorityFeePerGas: '0.1', maxFeePerGas: '0.1', updateTransaction: jest.fn(), }), })); jest.mock('react-router-dom', () => { const original = jest.requireActual('react-router-dom'); return { ...original, useHistory: () => ({ push: jest.fn(), }), useParams: () => ({ address: testTokenAddress, }), }; }); describe('TokenAllowancePage', () => { const props = { origin: 'https://metamask.github.io', siteImage: 'https://metamask.github.io/test-dapp/metamask-fox.svg', useNonceField: false, currentCurrency: 'usd', nativeCurrency: 'GoerliETH', ethTransactionTotal: '0.0012', fiatTransactionTotal: '1.6', hexTransactionTotal: '0x44364c5bb0000', isMultiLayerFeeNetwork: false, supportsEIP1559: true, userAddress: '0x0dcd5d886577d5081b0c52e242ef29e70be3e7bc', tokenAddress: '0x55797717b9947b31306f4aac7ad1365c6e3923bd', data: '0x095ea7b30000000000000000000000009bc5baf874d2da8d216ae9f137804184ee5afef40000000000000000000000000000000000000000000000000000000000011170', isSetApproveForAll: false, setApproveForAllArg: false, decimals: '4', dappProposedTokenAmount: '7', currentTokenBalance: '10', toAddress: '0x9bc5baf874d2da8d216ae9f137804184ee5afef4', tokenSymbol: 'TST', showCustomizeGasModal: jest.fn(), warning: '', txData: { id: 3049568294499567, time: 1664449552289, status: 'unapproved', metamaskNetworkId: '3', originalGasEstimate: '0xea60', userEditedGasLimit: false, chainId: '0x3', loadingDefaults: false, dappSuggestedGasFees: { gasPrice: '0x4a817c800', gas: '0xea60', }, sendFlowHistory: [], txParams: { from: '0x0dcd5d886577d5081b0c52e242ef29e70be3e7bc', to: '0x55797717b9947b31306f4aac7ad1365c6e3923bd', value: '0x0', data: '0x095ea7b30000000000000000000000009bc5baf874d2da8d216ae9f137804184ee5afef40000000000000000000000000000000000000000000000000000000000011170', gas: '0xea60', gasPrice: '0x4a817c800', maxFeePerGas: '0x4a817c800', }, origin: 'https://metamask.github.io', type: 'approve', userFeeLevel: 'custom', defaultGasEstimates: { estimateType: 'custom', gas: '0xea60', maxFeePerGas: '0x4a817c800', maxPriorityFeePerGas: '0x4a817c800', gasPrice: '0x4a817c800', }, }, }; let store; beforeEach(() => { store = configureMockStore([thunk])(state); }); it('should match snapshot', () => { const { container } = renderWithProvider( , store, ); expect(container).toMatchSnapshot(); }); it('should render title "Spending cap request for your" in token allowance page', () => { const { getByText } = renderWithProvider( , store, ); expect(getByText('Spending cap request for your')).toBeInTheDocument(); }); it('should render reject button', () => { const { getByTestId } = renderWithProvider( , store, ); const onCloseBtn = getByTestId('page-container-footer-cancel'); expect(onCloseBtn).toBeInTheDocument(); }); it('should not render customize nonce modal if useNonceField is set to false', () => { const { queryByText } = renderWithProvider( , store, ); expect(queryByText('Nonce')).not.toBeInTheDocument(); expect(queryByText('1')).not.toBeInTheDocument(); expect(mockShowModal).not.toHaveBeenCalledTimes(1); }); it('should render customize nonce modal if useNonceField is set to true', () => { props.useNonceField = true; props.nextNonce = 1; const { queryByText, getByText } = renderWithProvider( , store, ); const editButton = getByText('Edit'); expect(queryByText('Nonce')).toBeInTheDocument(); expect(queryByText('1')).toBeInTheDocument(); fireEvent.click(editButton); expect(mockShowModal).toHaveBeenCalledTimes(1); }); it('should render nextNonce value when custom nonce value is a empty string', () => { props.useNonceField = true; props.customNonceValue = ''; const { queryByText, getByText } = renderWithProvider( , store, ); const editButton = getByText('Edit'); expect(queryByText('Nonce')).toBeInTheDocument(); expect(queryByText('1')).toBeInTheDocument(); fireEvent.click(editButton); expect(mockShowModal).toHaveBeenCalledTimes(2); }); it('should render edited custom nonce value', () => { props.useNonceField = true; state.metamask.customNonceValue = '3'; const { queryByText, getByText } = renderWithProvider( , store, ); const editButton = getByText('Edit'); expect(queryByText('Nonce')).toBeInTheDocument(); expect(queryByText('3')).toBeInTheDocument(); fireEvent.click(editButton); expect(mockShowModal).toHaveBeenCalledTimes(3); }); it('should render customize nonce warning if custom nonce value is higher than nextNonce value', () => { props.useNonceField = true; props.nextNonce = 2; props.customNonceValue = '3'; props.warning = 'Nonce is higher than suggested nonce of 2'; const { getByText } = renderWithProvider( , store, ); expect( getByText('Nonce is higher than suggested nonce of 2'), ).toBeInTheDocument(); }); it('should not render customize nonce warning if custom nonce value is lower than nextNonce value', () => { props.useNonceField = true; props.nextNonce = 2; props.customNonceValue = '1'; props.warning = ''; const { container } = renderWithProvider( , store, ); const customizeNonceWarning = container.querySelector( '.token-allowance-container__custom-nonce-warning', ); expect(customizeNonceWarning).not.toBeInTheDocument(); }); it('should render customize nonce modal when next button is clicked and if useNonceField is set to true', () => { props.useNonceField = true; state.metamask.customNonceValue = '2'; const { getByText, getAllByText, queryByText } = renderWithProvider( , store, ); const nextButton = getByText('Next'); fireEvent.click(nextButton); const editButton = getAllByText('Edit'); expect(queryByText('Nonce')).toBeInTheDocument(); expect(queryByText('2')).toBeInTheDocument(); fireEvent.click(editButton[1]); expect(mockShowModal).toHaveBeenCalledTimes(4); }); it('should render customize nonce modal when next button is clicked, than back button is clicked, than return to previous page and if useNonceField is set to true', () => { props.useNonceField = true; state.metamask.customNonceValue = '2'; const { getByText, queryByText } = renderWithProvider( , store, ); const nextButton = getByText('Next'); fireEvent.click(nextButton); const backButton = getByText('< Back'); fireEvent.click(backButton); const editButton = getByText('Edit'); expect(queryByText('Nonce')).toBeInTheDocument(); expect(queryByText('2')).toBeInTheDocument(); fireEvent.click(editButton); expect(mockShowModal).toHaveBeenCalledTimes(5); }); it('should click View details and show function type', () => { const { getByText } = renderWithProvider( , store, ); const viewDetailsButton = getByText('View details'); fireEvent.click(viewDetailsButton); expect(getByText('Function: Approve')).toBeInTheDocument(); }); it('should load the page with dappProposedAmount prefilled and "Use site suggestion" should not be displayed', () => { const { queryByText, getByTestId } = renderWithProvider( , store, ); act(() => { const useSiteSuggestion = queryByText('Use site suggestion'); expect(useSiteSuggestion).not.toBeInTheDocument(); }); const input = getByTestId('custom-spending-cap-input'); expect(input.value).toBe('7'); }); it('should click Use site suggestion and set input value to default', () => { const { getByText, getByTestId } = renderWithProvider( , store, ); const textField = getByTestId('custom-spending-cap-input'); expect(textField.value).toBe('7'); fireEvent.change(textField, { target: { value: '1' } }); expect(textField.value).toBe('1'); act(() => { const useSiteSuggestion = getByText('Use site suggestion'); expect(useSiteSuggestion).toBeInTheDocument(); fireEvent.click(useSiteSuggestion); }); expect(textField.value).toBe('7'); }); it('should call back button when button is clicked and return to previous page', () => { const { getByText } = renderWithProvider( , store, ); const nextButton = getByText('Next'); fireEvent.click(nextButton); expect(getByText('Site requested spending cap')).toBeInTheDocument(); const backButton = getByText('< Back'); fireEvent.click(backButton); expect(getByText('Spending cap request for your')).toBeInTheDocument(); }); it('should click Verify third-party details and show popup Third-party details, then close popup', () => { const { getByText } = renderWithProvider( , store, ); const verifyThirdPartyDetails = getByText('Verify third-party details'); fireEvent.click(verifyThirdPartyDetails); expect(getByText('Third-party details')).toBeInTheDocument(); const gotIt = getByText('Got it'); fireEvent.click(gotIt); expect(gotIt).not.toBeInTheDocument(); }); it('should show ledger info text if the sending address is ledger', () => { const { queryByText, getByText } = renderWithProvider( , store, ); expect(queryByText('Prior to clicking confirm:')).toBeNull(); const nextButton = getByText('Next'); fireEvent.click(nextButton); expect(queryByText('Prior to clicking confirm:')).toBeInTheDocument(); }); it('should not show ledger info text if the sending address is not ledger', () => { const { queryByText, getByText } = renderWithProvider( , store, ); expect(queryByText('Prior to clicking confirm:')).toBeNull(); const nextButton = getByText('Next'); fireEvent.click(nextButton); expect(queryByText('Prior to clicking confirm:')).toBeNull(); }); it('should render security provider response if transaction is malicious', () => { const securityProviderResponse = { flagAsDangerous: 1, reason: 'This has been flagged as potentially suspicious. If you sign, you could lose access to all of your NFTs and any funds or other assets in your wallet.', reason_header: 'Warning', }; const { getByText } = renderWithProvider( , store, ); expect(getByText(securityProviderResponse.reason)).toBeInTheDocument(); }); it('should render from account name in header', () => { const { getByText } = renderWithProvider( , store, ); expect(getByText('Account 1')).toBeInTheDocument(); }); it('should account name from transaction even if currently selected account is different', () => { const newState = { ...state, metamask: { ...state.metamask, selectedAddress: '0xc42edfcc21ed14dda456aa0756c153f7985d8813', }, }; const newStore = configureMockStore([thunk])(newState); const { queryByText } = renderWithProvider( , newStore, ); expect(queryByText('Account 1')).toBeInTheDocument(); expect(queryByText('Account 2')).not.toBeInTheDocument(); }); it('should display security alert if present', () => { const { getByText } = renderWithProvider( , store, ); expect(getByText('This is a deceptive request')).toBeInTheDocument(); }); });