import { ReactFragment } from 'react'; import log from 'loglevel'; import { captureException } from '@sentry/browser'; import { capitalize, isEqual } from 'lodash'; import { ThunkAction } from 'redux-thunk'; import { Action, AnyAction } from 'redux'; import { Hex, Json } from '@metamask/utils'; import { AssetsContractController, BalanceMap, Nft, Token, } from '@metamask/assets-controllers'; import { PayloadAction } from '@reduxjs/toolkit'; import { GasFeeController } from '@metamask/gas-fee-controller'; import { PermissionsRequest } from '@metamask/permission-controller'; import { NonEmptyArray } from '@metamask/controller-utils'; import { getMethodDataAsync } from '../helpers/utils/transactions.util'; import switchDirection from '../../shared/lib/switch-direction'; import { ENVIRONMENT_TYPE_NOTIFICATION, ORIGIN_METAMASK, POLLING_TOKEN_ENVIRONMENT_TYPES, MESSAGE_TYPE, } from '../../shared/constants/app'; import { hasUnconfirmedTransactions } from '../helpers/utils/confirm-tx.util'; import { getEnvironmentType, addHexPrefix } from '../../app/scripts/lib/util'; import { getMetaMaskAccounts, getPermittedAccountsForCurrentTab, getSelectedAddress, ///: BEGIN:ONLY_INCLUDE_IN(flask) getNotifications, ///: END:ONLY_INCLUDE_IN } from '../selectors'; import { computeEstimatedGasLimit, initializeSendState, resetSendState, // NOTE: Until the send duck is typescript that this is importing a typedef // that does not have an explicit export statement. lets see if it breaks the // compiler DraftTransaction, } from '../ducks/send'; import { switchedToUnconnectedAccount } from '../ducks/alerts/unconnected-account'; import { getUnconnectedAccountAlertEnabledness } from '../ducks/metamask/metamask'; import { toChecksumHexAddress } from '../../shared/modules/hexstring-utils'; import { HardwareDeviceNames, LedgerTransportTypes, LEDGER_USB_VENDOR_ID, } from '../../shared/constants/hardware-wallets'; import { MetaMetricsEventCategory, MetaMetricsEventFragment, MetaMetricsEventOptions, MetaMetricsEventPayload, MetaMetricsPageObject, MetaMetricsPageOptions, MetaMetricsPagePayload, MetaMetricsReferrerObject, } from '../../shared/constants/metametrics'; import { parseSmartTransactionsError } from '../pages/swaps/swaps.util'; import { isEqualCaseInsensitive } from '../../shared/modules/string-utils'; ///: BEGIN:ONLY_INCLUDE_IN(flask) import { NOTIFICATIONS_EXPIRATION_DELAY } from '../helpers/constants/notifications'; ///: END:ONLY_INCLUDE_IN import { fetchLocale, loadRelativeTimeFormatLocaleData, } from '../helpers/utils/i18n-helper'; import { decimalToHex } from '../../shared/modules/conversion.utils'; import { TxGasFees, PriorityLevels } from '../../shared/constants/gas'; import { TransactionMetaMetricsEvent, TransactionType, } from '../../shared/constants/transaction'; import { NetworkType, RPCDefinition } from '../../shared/constants/network'; import { EtherDenomination } from '../../shared/constants/common'; import { isErrorWithMessage, logErrorWithMessage, } from '../../shared/modules/error'; import { TransactionMeta } from '../../app/scripts/controllers/incoming-transactions'; import { TxParams } from '../../app/scripts/controllers/transactions/tx-state-manager'; import { CustomGasSettings } from '../../app/scripts/controllers/transactions'; import { ThemeType } from '../../shared/constants/preferences'; import * as actionConstants from './actionConstants'; ///: BEGIN:ONLY_INCLUDE_IN(mmi) import { checkForUnapprovedTypedMessages, updateCustodyState, } from './institutional/institution-actions'; ///: END:ONLY_INCLUDE_IN import { generateActionId, callBackgroundMethod, submitRequestToBackground, } from './action-queue'; import { MetaMaskReduxDispatch, MetaMaskReduxState, TemporaryMessageDataType, } from './store'; export function goHome() { return { type: actionConstants.GO_HOME, }; } // async actions export function tryUnlockMetamask( password: string, ): ThunkAction { return (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); dispatch(unlockInProgress()); log.debug(`background.submitPassword`); return new Promise((resolve, reject) => { callBackgroundMethod('submitPassword', [password], (error) => { if (error) { reject(error); return; } resolve(); }); }) .then(() => { dispatch(unlockSucceeded()); return forceUpdateMetamaskState(dispatch); }) .then(() => { dispatch(hideLoadingIndication()); }) .catch((err) => { dispatch(unlockFailed(err.message)); dispatch(hideLoadingIndication()); return Promise.reject(err); }); }; } /** * Adds a new account where all data is encrypted using the given password and * where all addresses are generated from a given seed phrase. * * @param password - The password. * @param seedPhrase - The seed phrase. * @returns The updated state of the keyring controller. */ export function createNewVaultAndRestore( password: string, seedPhrase: string, ): ThunkAction { return (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); log.debug(`background.createNewVaultAndRestore`); // Encode the secret recovery phrase as an array of integers so that it is // serialized as JSON properly. const encodedSeedPhrase = Array.from( Buffer.from(seedPhrase, 'utf8').values(), ); // TODO: Add types for vault let vault: any; return new Promise((resolve, reject) => { callBackgroundMethod( 'createNewVaultAndRestore', [password, encodedSeedPhrase], (err, _vault) => { if (err) { reject(err); return; } vault = _vault; resolve(); }, ); }) .then(() => dispatch(unMarkPasswordForgotten())) .then(() => { dispatch(showAccountsPage()); dispatch(hideLoadingIndication()); return vault; }) .catch((err) => { dispatch(displayWarning(err.message)); dispatch(hideLoadingIndication()); return Promise.reject(err); }); }; } export function createNewVaultAndGetSeedPhrase( password: string, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); try { await createNewVault(password); const seedPhrase = await verifySeedPhrase(); return seedPhrase; } catch (error) { dispatch(displayWarning(error)); if (isErrorWithMessage(error)) { throw new Error(error.message); } else { throw error; } } finally { dispatch(hideLoadingIndication()); } }; } export function unlockAndGetSeedPhrase( password: string, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); try { await submitPassword(password); const seedPhrase = await verifySeedPhrase(); await forceUpdateMetamaskState(dispatch); return seedPhrase; } catch (error) { dispatch(displayWarning(error)); if (isErrorWithMessage(error)) { throw new Error(error.message); } else { throw error; } } finally { dispatch(hideLoadingIndication()); } }; } export function submitPassword(password: string): Promise { return new Promise((resolve, reject) => { callBackgroundMethod('submitPassword', [password], (error) => { if (error) { reject(error); return; } resolve(); }); }); } export function createNewVault(password: string): Promise { return new Promise((resolve, reject) => { callBackgroundMethod('createNewVaultAndKeychain', [password], (error) => { if (error) { reject(error); return; } resolve(true); }); }); } export function verifyPassword(password: string): Promise { return new Promise((resolve, reject) => { callBackgroundMethod('verifyPassword', [password], (error) => { if (error) { reject(error); return; } resolve(true); }); }); } export async function verifySeedPhrase() { const encodedSeedPhrase = await submitRequestToBackground( 'verifySeedPhrase', ); return Buffer.from(encodedSeedPhrase).toString('utf8'); } export function requestRevealSeedWords( password: string, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); log.debug(`background.verifyPassword`); try { await verifyPassword(password); const seedPhrase = await verifySeedPhrase(); return seedPhrase; } finally { dispatch(hideLoadingIndication()); } }; } export function tryReverseResolveAddress( address: string, ): ThunkAction { return () => { return new Promise((resolve) => { callBackgroundMethod('tryReverseResolveAddress', [address], (err) => { if (err) { logErrorWithMessage(err); } resolve(); }); }); }; } export function resetAccount(): ThunkAction< Promise, MetaMaskReduxState, unknown, AnyAction > { return (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); return new Promise((resolve, reject) => { callBackgroundMethod('resetAccount', [], (err, account) => { dispatch(hideLoadingIndication()); if (err) { if (isErrorWithMessage(err)) { dispatch(displayWarning(err.message)); } reject(err); return; } log.info(`Transaction history reset for ${account}`); dispatch(showAccountsPage()); resolve(account as string); }); }); }; } export function removeAccount( address: string, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); try { await new Promise((resolve, reject) => { callBackgroundMethod('removeAccount', [address], (error, account) => { if (error) { reject(error); return; } resolve(account); }); }); await forceUpdateMetamaskState(dispatch); } catch (error) { dispatch(displayWarning(error)); throw error; } finally { dispatch(hideLoadingIndication()); } log.info(`Account removed: ${address}`); dispatch(showAccountsPage()); }; } export function importNewAccount( strategy: string, args: any[], loadingMessage: ReactFragment, ): ThunkAction< Promise, MetaMaskReduxState, unknown, AnyAction > { return async (dispatch: MetaMaskReduxDispatch) => { let newState; dispatch(showLoadingIndication(loadingMessage)); try { log.debug(`background.importAccountWithStrategy`); await submitRequestToBackground('importAccountWithStrategy', [ strategy, args, ]); log.debug(`background.getState`); newState = await submitRequestToBackground< MetaMaskReduxState['metamask'] >('getState'); } finally { dispatch(hideLoadingIndication()); } dispatch(updateMetamaskState(newState)); return newState; }; } export function addNewAccount(): ThunkAction< void, MetaMaskReduxState, unknown, AnyAction > { log.debug(`background.addNewAccount`); return async (dispatch, getState) => { const oldIdentities = getState().metamask.identities; dispatch(showLoadingIndication()); let newIdentities; try { const { identities } = await submitRequestToBackground('addNewAccount', [ Object.keys(oldIdentities).length, ]); newIdentities = identities; } catch (error) { dispatch(displayWarning(error)); throw error; } finally { dispatch(hideLoadingIndication()); } const newAccountAddress = Object.keys(newIdentities).find( (address) => !oldIdentities[address], ); await forceUpdateMetamaskState(dispatch); return newAccountAddress; }; } export function checkHardwareStatus( deviceName: HardwareDeviceNames, hdPath: string, ): ThunkAction, MetaMaskReduxState, unknown, AnyAction> { log.debug(`background.checkHardwareStatus`, deviceName, hdPath); return async (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); let unlocked = false; try { unlocked = await submitRequestToBackground( 'checkHardwareStatus', [deviceName, hdPath], ); } catch (error) { logErrorWithMessage(error); dispatch(displayWarning(error)); throw error; } finally { dispatch(hideLoadingIndication()); } await forceUpdateMetamaskState(dispatch); return unlocked; }; } export function forgetDevice( deviceName: HardwareDeviceNames, ): ThunkAction { log.debug(`background.forgetDevice`, deviceName); return async (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); try { await submitRequestToBackground('forgetDevice', [deviceName]); } catch (error) { logErrorWithMessage(error); dispatch(displayWarning(error)); throw error; } finally { dispatch(hideLoadingIndication()); } await forceUpdateMetamaskState(dispatch); }; } // TODO: Define an Account Type for the return type of this method and anywhere // else dealing with accounts. export function connectHardware( deviceName: HardwareDeviceNames, page: string, hdPath: string, t: (key: string) => string, ): ThunkAction< Promise<{ address: string }[]>, MetaMaskReduxState, unknown, AnyAction > { log.debug(`background.connectHardware`, deviceName, page, hdPath); return async (dispatch, getState) => { const { ledgerTransportType } = getState().metamask; dispatch( showLoadingIndication(`Looking for your ${capitalize(deviceName)}...`), ); let accounts: { address: string }[]; try { if (deviceName === HardwareDeviceNames.ledger) { await submitRequestToBackground('establishLedgerTransportPreference'); } if ( deviceName === HardwareDeviceNames.ledger && ledgerTransportType === LedgerTransportTypes.webhid ) { const connectedDevices = await window.navigator.hid.requestDevice({ // The types for web hid were provided by @types/w3c-web-hid and may // not be fully formed or correct, because LEDGER_USB_VENDOR_ID is a // string and this integration with Navigator.hid works before // TypeScript. As a note, on the next declaration we convert the // LEDGER_USB_VENDOR_ID to a number for a different API so.... // TODO: Get David Walsh's opinion here filters: [{ vendorId: LEDGER_USB_VENDOR_ID as unknown as number }], }); const userApprovedWebHidConnection = connectedDevices.some( (device) => device.vendorId === Number(LEDGER_USB_VENDOR_ID), ); if (!userApprovedWebHidConnection) { throw new Error(t('ledgerWebHIDNotConnectedErrorMessage')); } } accounts = await submitRequestToBackground<{ address: string }[]>( 'connectHardware', [deviceName, page, hdPath], ); } catch (error) { logErrorWithMessage(error); if ( deviceName === HardwareDeviceNames.ledger && ledgerTransportType === LedgerTransportTypes.webhid && isErrorWithMessage(error) && error.message.match('Failed to open the device') ) { dispatch(displayWarning(t('ledgerDeviceOpenFailureMessage'))); throw new Error(t('ledgerDeviceOpenFailureMessage')); } else { if (deviceName !== HardwareDeviceNames.qr) { dispatch(displayWarning(error)); } throw error; } } finally { dispatch(hideLoadingIndication()); } await forceUpdateMetamaskState(dispatch); return accounts; }; } export function unlockHardwareWalletAccounts( indexes: string[], deviceName: HardwareDeviceNames, hdPath: string, hdPathDescription: string, ): ThunkAction, MetaMaskReduxState, unknown, AnyAction> { log.debug( `background.unlockHardwareWalletAccount`, indexes, deviceName, hdPath, hdPathDescription, ); return async (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); for (const index of indexes) { try { await submitRequestToBackground('unlockHardwareWalletAccount', [ index, deviceName, hdPath, hdPathDescription, ]); } catch (err) { logErrorWithMessage(err); dispatch(displayWarning(err)); dispatch(hideLoadingIndication()); throw err; } } dispatch(hideLoadingIndication()); return undefined; }; } export function showQrScanner(): ThunkAction< void, MetaMaskReduxState, unknown, AnyAction > { return (dispatch: MetaMaskReduxDispatch) => { dispatch( showModal({ name: 'QR_SCANNER', }), ); }; } export function setCurrentCurrency( currencyCode: string, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); log.debug(`background.setCurrentCurrency`); try { await submitRequestToBackground('setCurrentCurrency', [currencyCode]); await forceUpdateMetamaskState(dispatch); } catch (error) { logErrorWithMessage(error); dispatch(displayWarning(error)); return; } finally { dispatch(hideLoadingIndication()); } }; } export function signMsg( msgData: TemporaryMessageDataType['msgParams'], ): ThunkAction< Promise, MetaMaskReduxState, unknown, AnyAction > { log.debug('action - signMsg'); return async (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); log.debug(`actions calling background.signMessage`); let newState; try { newState = await submitRequestToBackground< MetaMaskReduxState['metamask'] >('signMessage', [msgData]); } catch (error) { logErrorWithMessage(error); dispatch(displayWarning(error)); throw error; } finally { dispatch(hideLoadingIndication()); } dispatch(updateMetamaskState(newState)); dispatch(completedTx(msgData.metamaskId)); dispatch(closeCurrentNotificationWindow()); return msgData; }; } export function signPersonalMsg( msgData: TemporaryMessageDataType['msgParams'], ): ThunkAction< Promise, MetaMaskReduxState, unknown, AnyAction > { log.debug('action - signPersonalMsg'); return async (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); log.debug(`actions calling background.signPersonalMessage`); let newState; try { newState = await submitRequestToBackground< MetaMaskReduxState['metamask'] >('signPersonalMessage', [msgData]); } catch (error) { logErrorWithMessage(error); dispatch(displayWarning(error)); throw error; } finally { dispatch(hideLoadingIndication()); } dispatch(updateMetamaskState(newState)); ///: BEGIN:ONLY_INCLUDE_IN(mmi) if (newState.unapprovedTypedMessages) { return checkForUnapprovedTypedMessages(msgData, newState); } ///: END:ONLY_INCLUDE_IN dispatch(completedTx(msgData.metamaskId)); dispatch(closeCurrentNotificationWindow()); return msgData; }; } export function decryptMsgInline( decryptedMsgData: TemporaryMessageDataType['msgParams'], ): ThunkAction< Promise, MetaMaskReduxState, unknown, AnyAction > { log.debug('action - decryptMsgInline'); return async (dispatch: MetaMaskReduxDispatch) => { log.debug(`actions calling background.decryptMessageInline`); let newState; try { newState = await submitRequestToBackground< MetaMaskReduxState['metamask'] >('decryptMessageInline', [decryptedMsgData]); } catch (error) { logErrorWithMessage(error); dispatch(displayWarning(error)); throw error; } dispatch(updateMetamaskState(newState)); return newState.unapprovedDecryptMsgs[decryptedMsgData.metamaskId]; }; } export function decryptMsg( decryptedMsgData: TemporaryMessageDataType['msgParams'], ): ThunkAction< Promise, MetaMaskReduxState, unknown, AnyAction > { log.debug('action - decryptMsg'); return async (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); log.debug(`actions calling background.decryptMessage`); let newState: MetaMaskReduxState['metamask']; try { newState = await submitRequestToBackground< MetaMaskReduxState['metamask'] >('decryptMessage', [decryptedMsgData]); } catch (error) { logErrorWithMessage(error); dispatch(displayWarning(error)); throw error; } finally { dispatch(hideLoadingIndication()); } dispatch(updateMetamaskState(newState)); dispatch(completedTx(decryptedMsgData.metamaskId)); dispatch(closeCurrentNotificationWindow()); return decryptedMsgData; }; } export function encryptionPublicKeyMsg( msgData: TemporaryMessageDataType['msgParams'], ): ThunkAction< Promise, MetaMaskReduxState, unknown, AnyAction > { log.debug('action - encryptionPublicKeyMsg'); return async (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); log.debug(`actions calling background.encryptionPublicKey`); let newState: MetaMaskReduxState['metamask']; try { newState = await submitRequestToBackground< MetaMaskReduxState['metamask'] >('encryptionPublicKey', [msgData]); } catch (error) { logErrorWithMessage(error); dispatch(displayWarning(error)); throw error; } finally { dispatch(hideLoadingIndication()); } dispatch(updateMetamaskState(newState)); dispatch(completedTx(msgData.metamaskId)); dispatch(closeCurrentNotificationWindow()); return msgData; }; } export function signTypedMsg( msgData: TemporaryMessageDataType['msgParams'], ): ThunkAction< Promise, MetaMaskReduxState, unknown, AnyAction > { log.debug('action - signTypedMsg'); return async (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); log.debug(`actions calling background.signTypedMessage`); let newState: MetaMaskReduxState['metamask']; try { newState = await submitRequestToBackground< MetaMaskReduxState['metamask'] >('signTypedMessage', [msgData]); } catch (error) { logErrorWithMessage(error); dispatch(displayWarning(error)); throw error; } finally { dispatch(hideLoadingIndication()); } dispatch(updateMetamaskState(newState)); ///: BEGIN:ONLY_INCLUDE_IN(mmi) if (newState.unapprovedTypedMessages) { return checkForUnapprovedTypedMessages(msgData, newState); } ///: END:ONLY_INCLUDE_IN dispatch(completedTx(msgData.metamaskId)); dispatch(closeCurrentNotificationWindow()); return msgData; }; } export function updateCustomNonce(value: string) { return { type: actionConstants.UPDATE_CUSTOM_NONCE, value, }; } const updateMetamaskStateFromBackground = (): Promise< MetaMaskReduxState['metamask'] > => { log.debug(`background.getState`); return new Promise((resolve, reject) => { callBackgroundMethod( 'getState', [], (error, newState) => { if (error) { reject(error); return; } resolve(newState as MetaMaskReduxState['metamask']); }, ); }); }; /** * TODO: update previousGasParams to use typed gas params object * TODO: codeword: NOT_A_THUNK @brad-decker * * @param txId - MetaMask internal transaction id * @param previousGasParams - Object of gas params to set as previous */ export function updatePreviousGasParams( txId: number, previousGasParams: Record, ): ThunkAction< Promise, MetaMaskReduxState, unknown, AnyAction > { return async () => { let updatedTransaction: TransactionMeta; try { updatedTransaction = await submitRequestToBackground( 'updatePreviousGasParams', [txId, previousGasParams], ); } catch (error) { logErrorWithMessage(error); throw error; } return updatedTransaction; }; } // TODO: codeword: NOT_A_THUNK @brad-decker export function updateSwapApprovalTransaction( txId: number, txSwapApproval: TransactionMeta, ): ThunkAction< Promise, MetaMaskReduxState, unknown, AnyAction > { return async () => { let updatedTransaction: TransactionMeta; try { updatedTransaction = await submitRequestToBackground( 'updateSwapApprovalTransaction', [txId, txSwapApproval], ); } catch (error) { logErrorWithMessage(error); throw error; } return updatedTransaction; }; } export function updateEditableParams( txId: number, editableParams: Partial, ): ThunkAction< Promise, MetaMaskReduxState, unknown, AnyAction > { return async (dispatch: MetaMaskReduxDispatch) => { let updatedTransaction: TransactionMeta; try { updatedTransaction = await submitRequestToBackground( 'updateEditableParams', [txId, editableParams], ); } catch (error) { logErrorWithMessage(error); throw error; } await forceUpdateMetamaskState(dispatch); return updatedTransaction; }; } /** * Appends new send flow history to a transaction * TODO: codeword: NOT_A_THUNK @brad-decker * * @param txId - the id of the transaction to update * @param currentSendFlowHistoryLength - sendFlowHistory entries currently * @param sendFlowHistory - the new send flow history to append to the * transaction * @returns */ export function updateTransactionSendFlowHistory( txId: number, currentSendFlowHistoryLength: number, sendFlowHistory: DraftTransaction['history'], ): ThunkAction< Promise, MetaMaskReduxState, unknown, AnyAction > { return async () => { let updatedTransaction: TransactionMeta; try { updatedTransaction = await submitRequestToBackground( 'updateTransactionSendFlowHistory', [txId, currentSendFlowHistoryLength, sendFlowHistory], ); } catch (error) { logErrorWithMessage(error); throw error; } return updatedTransaction; }; } export async function backupUserData(): Promise<{ filename: string; data: string; }> { let backedupData; try { backedupData = await submitRequestToBackground<{ filename: string; data: string; }>('backupUserData'); } catch (error) { logErrorWithMessage(error); throw error; } return backedupData; } export async function restoreUserData(jsonString: Json): Promise { try { await submitRequestToBackground('restoreUserData', [jsonString]); } catch (error) { logErrorWithMessage(error); throw error; } return true; } // TODO: codeword: NOT_A_THUNK @brad-decker export function updateTransactionGasFees( txId: number, txGasFees: Partial, ): ThunkAction< Promise, MetaMaskReduxState, unknown, AnyAction > { return async () => { let updatedTransaction: TransactionMeta; try { updatedTransaction = await submitRequestToBackground( 'updateTransactionGasFees', [txId, txGasFees], ); } catch (error) { logErrorWithMessage(error); throw error; } return updatedTransaction; }; } // TODO: codeword: NOT_A_THUNK @brad-decker export function updateSwapTransaction( txId: number, txSwap: TransactionMeta, ): ThunkAction< Promise, MetaMaskReduxState, unknown, AnyAction > { return async () => { let updatedTransaction: TransactionMeta; try { updatedTransaction = await submitRequestToBackground( 'updateSwapTransaction', [txId, txSwap], ); } catch (error) { logErrorWithMessage(error); throw error; } return updatedTransaction; }; } export function updateTransaction( txMeta: TransactionMeta, dontShowLoadingIndicator: boolean, ): ThunkAction< Promise, MetaMaskReduxState, unknown, AnyAction > { return async (dispatch: MetaMaskReduxDispatch) => { !dontShowLoadingIndicator && dispatch(showLoadingIndication()); try { await submitRequestToBackground('updateTransaction', [txMeta]); } catch (error) { dispatch(updateTransactionParams(txMeta.id, txMeta.txParams)); dispatch(hideLoadingIndication()); dispatch(goHome()); logErrorWithMessage(error); throw error; } try { dispatch(updateTransactionParams(txMeta.id, txMeta.txParams)); const newState = await updateMetamaskStateFromBackground(); dispatch(updateMetamaskState(newState)); dispatch(showConfTxPage({ id: txMeta.id })); return txMeta; } finally { dispatch(hideLoadingIndication()); } }; } /** * Action to create a new transaction in the controller and route to the * confirmation page. Returns the newly created txMeta in case additional logic * should be applied to the transaction after creation. * * @param method * @param txParams - The transaction parameters * @param type - The type of the transaction being added. * @param sendFlowHistory - The history of the send flow at time of creation. * @returns */ export function addUnapprovedTransactionAndRouteToConfirmationPage( method: string, txParams: TxParams, type: TransactionType, sendFlowHistory: DraftTransaction['history'], ): ThunkAction< Promise, MetaMaskReduxState, unknown, AnyAction > { return async (dispatch: MetaMaskReduxDispatch) => { const actionId = generateActionId(); try { log.debug('background.addUnapprovedTransaction'); const txMeta = await submitRequestToBackground( 'addUnapprovedTransaction', [method, txParams, ORIGIN_METAMASK, type, sendFlowHistory, actionId], actionId, ); dispatch(showConfTxPage()); return txMeta; } catch (error) { dispatch(hideLoadingIndication()); dispatch(displayWarning(error)); } return null; }; } /** * Wrapper around the promisifedBackground to create a new unapproved * transaction in the background and return the newly created txMeta. * This method does not show errors or route to a confirmation page and is * used primarily for swaps functionality. * * @param method * @param txParams - the transaction parameters * @param type - The type of the transaction being added. * @returns */ export async function addUnapprovedTransaction( method: string, txParams: TxParams, type: TransactionType, ): Promise { log.debug('background.addUnapprovedTransaction'); const actionId = generateActionId(); const txMeta = await submitRequestToBackground( 'addUnapprovedTransaction', [method, txParams, ORIGIN_METAMASK, type, undefined, actionId], actionId, ); return txMeta; } export function updateAndApproveTx( txMeta: TransactionMeta, dontShowLoadingIndicator: boolean, ): ThunkAction< Promise, MetaMaskReduxState, unknown, AnyAction > { return (dispatch: MetaMaskReduxDispatch) => { !dontShowLoadingIndicator && dispatch(showLoadingIndication()); return new Promise((resolve, reject) => { const actionId = generateActionId(); callBackgroundMethod( 'updateAndApproveTransaction', [txMeta, actionId], (err) => { dispatch(updateTransactionParams(txMeta.id, txMeta.txParams)); dispatch(resetSendState()); if (err) { dispatch(goHome()); logErrorWithMessage(err); reject(err); return; } resolve(txMeta); }, ); }) .then(() => updateMetamaskStateFromBackground()) .then((newState) => dispatch(updateMetamaskState(newState))) .then(() => { dispatch(resetSendState()); dispatch(completedTx(txMeta.id)); dispatch(hideLoadingIndication()); dispatch(updateCustomNonce('')); dispatch(closeCurrentNotificationWindow()); return txMeta; }) .catch((err) => { dispatch(hideLoadingIndication()); return Promise.reject(err); }); }; } export async function getTransactions( filters: { filterToCurrentNetwork?: boolean; searchCriteria?: Partial & Partial; } = {}, ): Promise { return await submitRequestToBackground('getTransactions', [ filters, ]); } export function completedTx( txId: number, ): ThunkAction { return (dispatch: MetaMaskReduxDispatch) => { dispatch({ type: actionConstants.COMPLETED_TX, value: { id: txId, }, }); }; } export function updateTransactionParams(txId: number, txParams: TxParams) { return { type: actionConstants.UPDATE_TRANSACTION_PARAMS, id: txId, value: txParams, }; } ///: BEGIN:ONLY_INCLUDE_IN(flask) export function disableSnap( snapId: string, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { await submitRequestToBackground('disableSnap', [snapId]); await forceUpdateMetamaskState(dispatch); }; } export function enableSnap( snapId: string, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { await submitRequestToBackground('enableSnap', [snapId]); await forceUpdateMetamaskState(dispatch); }; } export function removeSnap( snapId: string, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { await submitRequestToBackground('removeSnap', [snapId]); await forceUpdateMetamaskState(dispatch); }; } export async function removeSnapError(msgData: string): Promise { return submitRequestToBackground('removeSnapError', [msgData]); } export async function handleSnapRequest(args: { snapId: string; origin: string; handler: string; request: { jsonrpc: '2.0'; method: string; params: Record; }; }): Promise { return submitRequestToBackground('handleSnapRequest', [args]); } export function dismissNotifications( ids: string[], ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { await submitRequestToBackground('dismissNotifications', [ids]); await forceUpdateMetamaskState(dispatch); }; } export function deleteExpiredNotifications(): ThunkAction< void, MetaMaskReduxState, unknown, AnyAction > { return async (dispatch, getState) => { const state = getState(); const notifications = getNotifications(state); const notificationIdsToDelete = notifications .filter((notification) => { const expirationTime = new Date( Date.now() - NOTIFICATIONS_EXPIRATION_DELAY, ); return Boolean( notification.readDate && new Date(notification.readDate) < expirationTime, ); }) .map(({ id }) => id); if (notificationIdsToDelete.length) { await submitRequestToBackground('dismissNotifications', [ notificationIdsToDelete, ]); await forceUpdateMetamaskState(dispatch); } }; } export function markNotificationsAsRead( ids: string[], ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { await submitRequestToBackground('markNotificationsAsRead', [ids]); await forceUpdateMetamaskState(dispatch); }; } export function setDesktopEnabled(desktopEnabled: boolean) { return async () => { try { await submitRequestToBackground('setDesktopEnabled', [desktopEnabled]); } catch (error) { log.error(error); } }; } export async function generateDesktopOtp() { return await submitRequestToBackground('generateDesktopOtp'); } export async function testDesktopConnection() { return await submitRequestToBackground('testDesktopConnection'); } export async function disableDesktop() { return await submitRequestToBackground('disableDesktop'); } ///: END:ONLY_INCLUDE_IN export function cancelMsg( msgData: TemporaryMessageDataType, ): ThunkAction< Promise, MetaMaskReduxState, unknown, AnyAction > { return async (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); let newState; try { newState = await submitRequestToBackground< MetaMaskReduxState['metamask'] >('cancelMessage', [msgData.id]); } finally { dispatch(hideLoadingIndication()); } dispatch(updateMetamaskState(newState)); dispatch(completedTx(msgData.id)); dispatch(closeCurrentNotificationWindow()); return msgData; }; } /** * Cancels all of the given messages * * @param msgDataList - a list of msg data objects * @returns */ export function cancelMsgs( msgDataList: TemporaryMessageDataType[], ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); try { const msgIds = msgDataList.map(({ id }) => id); const cancellations = msgDataList.map( ({ id, type }) => new Promise((resolve, reject) => { switch (type) { case MESSAGE_TYPE.ETH_SIGN_TYPED_DATA: callBackgroundMethod('cancelTypedMessage', [id], (err) => { if (err) { reject(err); return; } resolve(); }); return; case MESSAGE_TYPE.PERSONAL_SIGN: callBackgroundMethod('cancelPersonalMessage', [id], (err) => { if (err) { reject(err); return; } resolve(); }); return; case MESSAGE_TYPE.ETH_DECRYPT: callBackgroundMethod('cancelDecryptMessage', [id], (err) => { if (err) { reject(err); return; } resolve(); }); return; case MESSAGE_TYPE.ETH_GET_ENCRYPTION_PUBLIC_KEY: callBackgroundMethod( 'cancelEncryptionPublicKey', [id], (err) => { if (err) { reject(err); return; } resolve(); }, ); return; case MESSAGE_TYPE.ETH_SIGN: callBackgroundMethod('cancelMessage', [id], (err) => { if (err) { reject(err); return; } resolve(); }); return; default: reject( new Error( `MetaMask Message Signature: Unknown message type: ${id}`, ), ); } }), ); await Promise.all(cancellations); const newState = await updateMetamaskStateFromBackground(); dispatch(updateMetamaskState(newState)); msgIds.forEach((id) => { dispatch(completedTx(id)); }); } catch (err) { logErrorWithMessage(err); } finally { if (getEnvironmentType() === ENVIRONMENT_TYPE_NOTIFICATION) { closeNotificationPopup(); } else { dispatch(hideLoadingIndication()); } } }; } export function cancelPersonalMsg( msgData: TemporaryMessageDataType, ): ThunkAction< Promise, MetaMaskReduxState, unknown, AnyAction > { return async (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); let newState; try { newState = await submitRequestToBackground< MetaMaskReduxState['metamask'] >('cancelPersonalMessage', [msgData.id]); } finally { dispatch(hideLoadingIndication()); } dispatch(updateMetamaskState(newState)); dispatch(completedTx(msgData.id)); dispatch(closeCurrentNotificationWindow()); return msgData; }; } export function cancelDecryptMsg( msgData: TemporaryMessageDataType, ): ThunkAction< Promise, MetaMaskReduxState, unknown, AnyAction > { return async (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); let newState; try { newState = await submitRequestToBackground< MetaMaskReduxState['metamask'] >('cancelDecryptMessage', [msgData.id]); } finally { dispatch(hideLoadingIndication()); } dispatch(updateMetamaskState(newState)); dispatch(completedTx(msgData.id)); dispatch(closeCurrentNotificationWindow()); return msgData; }; } export function cancelEncryptionPublicKeyMsg( msgData: TemporaryMessageDataType, ): ThunkAction< Promise, MetaMaskReduxState, unknown, AnyAction > { return async (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); let newState; try { newState = await submitRequestToBackground< MetaMaskReduxState['metamask'] >('cancelEncryptionPublicKey', [msgData.id]); } finally { dispatch(hideLoadingIndication()); } dispatch(updateMetamaskState(newState)); dispatch(completedTx(msgData.id)); dispatch(closeCurrentNotificationWindow()); return msgData; }; } export function cancelTypedMsg( msgData: TemporaryMessageDataType, ): ThunkAction< Promise, MetaMaskReduxState, unknown, AnyAction > { return async (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); let newState; try { newState = await submitRequestToBackground< MetaMaskReduxState['metamask'] >('cancelTypedMessage', [msgData.id]); } finally { dispatch(hideLoadingIndication()); } dispatch(updateMetamaskState(newState)); dispatch(completedTx(msgData.id)); dispatch(closeCurrentNotificationWindow()); return msgData; }; } export function cancelTx( txMeta: TransactionMeta, _showLoadingIndication = true, ): ThunkAction< Promise, MetaMaskReduxState, unknown, AnyAction > { return (dispatch: MetaMaskReduxDispatch) => { _showLoadingIndication && dispatch(showLoadingIndication()); return new Promise((resolve, reject) => { const actionId = generateActionId(); callBackgroundMethod( 'cancelTransaction', [txMeta.id, actionId], (error) => { if (error) { reject(error); return; } resolve(); }, ); }) .then(() => updateMetamaskStateFromBackground()) .then((newState) => dispatch(updateMetamaskState(newState))) .then(() => { dispatch(resetSendState()); dispatch(completedTx(txMeta.id)); dispatch(hideLoadingIndication()); dispatch(closeCurrentNotificationWindow()); return txMeta; }) .catch((error) => { dispatch(hideLoadingIndication()); throw error; }); }; } /** * Cancels all of the given transactions * * @param txMetaList * @returns */ export function cancelTxs( txMetaList: TransactionMeta[], ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); try { const txIds = txMetaList.map(({ id }) => id); const cancellations = txIds.map( (id) => new Promise((resolve, reject) => { const actionId = generateActionId(); callBackgroundMethod('cancelTransaction', [id, actionId], (err) => { if (err) { reject(err); return; } resolve(); }); }), ); await Promise.all(cancellations); const newState = await updateMetamaskStateFromBackground(); dispatch(updateMetamaskState(newState)); dispatch(resetSendState()); txIds.forEach((id) => { dispatch(completedTx(id)); }); } finally { if (getEnvironmentType() === ENVIRONMENT_TYPE_NOTIFICATION) { closeNotificationPopup(); } else { dispatch(hideLoadingIndication()); } } }; } export function markPasswordForgotten(): ThunkAction< void, MetaMaskReduxState, unknown, AnyAction > { return async (dispatch: MetaMaskReduxDispatch) => { try { await new Promise((resolve, reject) => { callBackgroundMethod('markPasswordForgotten', [], (error) => { if (error) { reject(error); return; } resolve(); }); }); } finally { // TODO: handle errors dispatch(hideLoadingIndication()); await forceUpdateMetamaskState(dispatch); } }; } export function unMarkPasswordForgotten(): ThunkAction< void, MetaMaskReduxState, unknown, AnyAction > { return (dispatch: MetaMaskReduxDispatch) => { return new Promise((resolve) => { callBackgroundMethod('unMarkPasswordForgotten', [], () => { resolve(); }); }).then(() => forceUpdateMetamaskState(dispatch)); }; } export function closeWelcomeScreen() { return { type: actionConstants.CLOSE_WELCOME_SCREEN, }; } // // unlock screen // export function unlockInProgress() { return { type: actionConstants.UNLOCK_IN_PROGRESS, }; } export function unlockFailed(message?: string) { return { type: actionConstants.UNLOCK_FAILED, value: message, }; } export function unlockSucceeded(message?: string) { return { type: actionConstants.UNLOCK_SUCCEEDED, value: message, }; } export function updateMetamaskState( newState: MetaMaskReduxState['metamask'], ): ThunkAction { return (dispatch, getState) => { const { metamask: currentState } = getState(); const { currentLocale, selectedAddress, provider } = currentState; const { currentLocale: newLocale, selectedAddress: newSelectedAddress, provider: newProvider, } = newState; if (currentLocale && newLocale && currentLocale !== newLocale) { dispatch(updateCurrentLocale(newLocale)); } if (selectedAddress !== newSelectedAddress) { dispatch({ type: actionConstants.SELECTED_ADDRESS_CHANGED }); } const newAddressBook = newState.addressBook?.[newProvider?.chainId] ?? {}; const oldAddressBook = currentState.addressBook?.[provider?.chainId] ?? {}; const newAccounts: { [address: string]: Record } = getMetaMaskAccounts({ metamask: newState }); const oldAccounts: { [address: string]: Record } = getMetaMaskAccounts({ metamask: currentState }); const newSelectedAccount = newAccounts[newSelectedAddress]; const oldSelectedAccount = newAccounts[selectedAddress]; // dispatch an ACCOUNT_CHANGED for any account whose balance or other // properties changed in this update Object.entries(oldAccounts).forEach(([address, oldAccount]) => { if (!isEqual(oldAccount, newAccounts[address])) { dispatch({ type: actionConstants.ACCOUNT_CHANGED, payload: { account: newAccounts[address] }, }); } }); // Also emit an event for the selected account changing, either due to a // property update or if the entire account changes. if (isEqual(oldSelectedAccount, newSelectedAccount) === false) { dispatch({ type: actionConstants.SELECTED_ACCOUNT_CHANGED, payload: { account: newSelectedAccount }, }); } // We need to keep track of changing address book entries if (isEqual(oldAddressBook, newAddressBook) === false) { dispatch({ type: actionConstants.ADDRESS_BOOK_UPDATED, payload: { addressBook: newAddressBook }, }); } // track when gasFeeEstimates change if ( isEqual(currentState.gasFeeEstimates, newState.gasFeeEstimates) === false ) { dispatch({ type: actionConstants.GAS_FEE_ESTIMATES_UPDATED, payload: { gasFeeEstimates: newState.gasFeeEstimates, gasEstimateType: newState.gasEstimateType, }, }); } dispatch({ type: actionConstants.UPDATE_METAMASK_STATE, value: newState, }); if (provider.chainId !== newProvider.chainId) { dispatch({ type: actionConstants.CHAIN_CHANGED, payload: newProvider.chainId, }); // We dispatch this action to ensure that the send state stays up to date // after the chain changes. This async thunk will fail gracefully in the // event that we are not yet on the send flow with a draftTransaction in // progress. dispatch(initializeSendState({ chainHasChanged: true })); } ///: BEGIN:ONLY_INCLUDE_IN(mmi) updateCustodyState(dispatch, newState, getState()); ///: END:ONLY_INCLUDE_IN }; } const backgroundSetLocked = (): Promise => { return new Promise((resolve, reject) => { callBackgroundMethod('setLocked', [], (error) => { if (error) { reject(error); return; } resolve(); }); }); }; export function lockMetamask(): ThunkAction< void, MetaMaskReduxState, unknown, AnyAction > { log.debug(`background.setLocked`); return (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); return backgroundSetLocked() .then(() => updateMetamaskStateFromBackground()) .catch((error) => { dispatch(displayWarning(error.message)); return Promise.reject(error); }) .then((newState) => { dispatch(updateMetamaskState(newState)); dispatch(hideLoadingIndication()); dispatch({ type: actionConstants.LOCK_METAMASK }); }) .catch(() => { dispatch(hideLoadingIndication()); dispatch({ type: actionConstants.LOCK_METAMASK }); }); }; } async function _setSelectedAddress(address: string): Promise { log.debug(`background.setSelectedAddress`); await submitRequestToBackground('setSelectedAddress', [address]); } export function setSelectedAddress( address: string, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); log.debug(`background.setSelectedAddress`); try { await _setSelectedAddress(address); } catch (error) { dispatch(displayWarning(error)); return; } finally { dispatch(hideLoadingIndication()); } }; } export function setSelectedAccount( address: string, ): ThunkAction { return async (dispatch, getState) => { dispatch(showLoadingIndication()); log.debug(`background.setSelectedAddress`); const state = getState(); const unconnectedAccountAccountAlertIsEnabled = getUnconnectedAccountAlertEnabledness(state); const activeTabOrigin = state.activeTab.origin; const selectedAddress = getSelectedAddress(state); const permittedAccountsForCurrentTab = getPermittedAccountsForCurrentTab(state); const currentTabIsConnectedToPreviousAddress = Boolean(activeTabOrigin) && permittedAccountsForCurrentTab.includes(selectedAddress); const currentTabIsConnectedToNextAddress = Boolean(activeTabOrigin) && permittedAccountsForCurrentTab.includes(address); const switchingToUnconnectedAddress = currentTabIsConnectedToPreviousAddress && !currentTabIsConnectedToNextAddress; try { await _setSelectedAddress(address); await forceUpdateMetamaskState(dispatch); } catch (error) { dispatch(displayWarning(error)); return; } finally { dispatch(hideLoadingIndication()); } if ( unconnectedAccountAccountAlertIsEnabled && switchingToUnconnectedAddress ) { dispatch(switchedToUnconnectedAccount()); await setUnconnectedAccountAlertShown(activeTabOrigin); } }; } export function addPermittedAccount( origin: string, address: string, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { await new Promise((resolve, reject) => { callBackgroundMethod( 'addPermittedAccount', [origin, address], (error) => { if (error) { reject(error); return; } resolve(); }, ); }); await forceUpdateMetamaskState(dispatch); }; } export function removePermittedAccount( origin: string, address: string, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { await new Promise((resolve, reject) => { callBackgroundMethod( 'removePermittedAccount', [origin, address], (error) => { if (error) { reject(error); return; } resolve(); }, ); }); await forceUpdateMetamaskState(dispatch); }; } export function showAccountsPage() { return { type: actionConstants.SHOW_ACCOUNTS_PAGE, }; } export function showConfTxPage({ id }: Partial = {}) { return { type: actionConstants.SHOW_CONF_TX_PAGE, id, }; } export function addToken( address?: string, symbol?: string, decimals?: number, image?: string, dontShowLoadingIndicator?: boolean, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { if (!address) { throw new Error('MetaMask - Cannot add token without address'); } if (!dontShowLoadingIndicator) { dispatch(showLoadingIndication()); } try { await submitRequestToBackground('addToken', [ address, symbol, decimals, image, ]); } catch (error) { logErrorWithMessage(error); dispatch(displayWarning(error)); } finally { await forceUpdateMetamaskState(dispatch); dispatch(hideLoadingIndication()); } }; } /** * To add the tokens user selected to state * * @param tokensToImport */ export function addImportedTokens( tokensToImport: Token[], ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { try { await submitRequestToBackground('addImportedTokens', [tokensToImport]); } catch (error) { logErrorWithMessage(error); } finally { await forceUpdateMetamaskState(dispatch); } }; } /** * To add ignored token addresses to state * * @param options * @param options.tokensToIgnore * @param options.dontShowLoadingIndicator */ export function ignoreTokens({ tokensToIgnore, dontShowLoadingIndicator = false, }: { tokensToIgnore: string[]; dontShowLoadingIndicator: boolean; }): ThunkAction { const _tokensToIgnore = Array.isArray(tokensToIgnore) ? tokensToIgnore : [tokensToIgnore]; return async (dispatch: MetaMaskReduxDispatch) => { if (!dontShowLoadingIndicator) { dispatch(showLoadingIndication()); } try { await submitRequestToBackground('ignoreTokens', [_tokensToIgnore]); } catch (error) { logErrorWithMessage(error); dispatch(displayWarning(error)); } finally { await forceUpdateMetamaskState(dispatch); dispatch(hideLoadingIndication()); } }; } /** * To fetch the ERC20 tokens with non-zero balance in a single call * * @param tokens */ export async function getBalancesInSingleCall( tokens: string[], ): Promise { return await submitRequestToBackground('getBalancesInSingleCall', [tokens]); } export function addNft( address: string, tokenID: string, dontShowLoadingIndicator: boolean, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { if (!address) { throw new Error('MetaMask - Cannot add NFT without address'); } if (!tokenID) { throw new Error('MetaMask - Cannot add NFT without tokenID'); } if (!dontShowLoadingIndicator) { dispatch(showLoadingIndication()); } try { await submitRequestToBackground('addNft', [address, tokenID]); } catch (error) { logErrorWithMessage(error); dispatch(displayWarning(error)); } finally { await forceUpdateMetamaskState(dispatch); dispatch(hideLoadingIndication()); } }; } export function addNftVerifyOwnership( address: string, tokenID: string, dontShowLoadingIndicator: boolean, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { if (!address) { throw new Error('MetaMask - Cannot add NFT without address'); } if (!tokenID) { throw new Error('MetaMask - Cannot add NFT without tokenID'); } if (!dontShowLoadingIndicator) { dispatch(showLoadingIndication()); } try { await submitRequestToBackground('addNftVerifyOwnership', [ address, tokenID, ]); } catch (error) { if ( isErrorWithMessage(error) && (error.message.includes('This NFT is not owned by the user') || error.message.includes('Unable to verify ownership')) ) { throw error; } else { logErrorWithMessage(error); dispatch(displayWarning(error)); } } finally { await forceUpdateMetamaskState(dispatch); dispatch(hideLoadingIndication()); } }; } export function removeAndIgnoreNft( address: string, tokenID: string, dontShowLoadingIndicator: boolean, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { if (!address) { throw new Error('MetaMask - Cannot ignore NFT without address'); } if (!tokenID) { throw new Error('MetaMask - Cannot ignore NFT without tokenID'); } if (!dontShowLoadingIndicator) { dispatch(showLoadingIndication()); } try { await submitRequestToBackground('removeAndIgnoreNft', [address, tokenID]); } catch (error) { logErrorWithMessage(error); dispatch(displayWarning(error)); } finally { await forceUpdateMetamaskState(dispatch); dispatch(hideLoadingIndication()); } }; } export function removeNft( address: string, tokenID: string, dontShowLoadingIndicator: boolean, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { if (!address) { throw new Error('MetaMask - Cannot remove NFT without address'); } if (!tokenID) { throw new Error('MetaMask - Cannot remove NFT without tokenID'); } if (!dontShowLoadingIndicator) { dispatch(showLoadingIndication()); } try { await submitRequestToBackground('removeNft', [address, tokenID]); } catch (error) { logErrorWithMessage(error); dispatch(displayWarning(error)); } finally { await forceUpdateMetamaskState(dispatch); dispatch(hideLoadingIndication()); } }; } export async function checkAndUpdateAllNftsOwnershipStatus() { await submitRequestToBackground('checkAndUpdateAllNftsOwnershipStatus'); } export async function isNftOwner( ownerAddress: string, nftAddress: string, nftId: string, ): Promise { return await submitRequestToBackground('isNftOwner', [ ownerAddress, nftAddress, nftId, ]); } export async function checkAndUpdateSingleNftOwnershipStatus(nft: Nft) { await submitRequestToBackground('checkAndUpdateSingleNftOwnershipStatus', [ nft, false, ]); } // When we upgrade to TypeScript 4.5 this is part of the language. It will get // the underlying type of a Promise generic type. So Awaited> is // void. type Awaited = T extends PromiseLike ? U : T; export async function getTokenStandardAndDetails( address: string, userAddress: string, tokenId: string, ): Promise< Awaited< ReturnType > & { balance?: string } > { return await submitRequestToBackground('getTokenStandardAndDetails', [ address, userAddress, tokenId, ]); } export function addTokens( tokens: Token[] | { [address: string]: Token }, ): ThunkAction { return (dispatch: MetaMaskReduxDispatch) => { if (Array.isArray(tokens)) { return Promise.all( tokens.map(({ address, symbol, decimals }) => dispatch(addToken(address, symbol, decimals)), ), ); } return Promise.all( Object.entries(tokens).map(([_, { address, symbol, decimals }]) => dispatch(addToken(address, symbol, decimals)), ), ); }; } export function rejectWatchAsset( suggestedAssetID: string, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); try { await submitRequestToBackground('rejectWatchAsset', [suggestedAssetID]); await forceUpdateMetamaskState(dispatch); } catch (error) { logErrorWithMessage(error); dispatch(displayWarning(error)); return; } finally { dispatch(hideLoadingIndication()); } dispatch(closeCurrentNotificationWindow()); }; } export function acceptWatchAsset( suggestedAssetID: string, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); try { await submitRequestToBackground('acceptWatchAsset', [suggestedAssetID]); await forceUpdateMetamaskState(dispatch); } catch (error) { logErrorWithMessage(error); dispatch(displayWarning(error)); return; } finally { dispatch(hideLoadingIndication()); } dispatch(closeCurrentNotificationWindow()); }; } export function clearPendingTokens(): Action { return { type: actionConstants.CLEAR_PENDING_TOKENS, }; } export function createCancelTransaction( txId: number, customGasSettings: CustomGasSettings, options: { estimatedBaseFee?: string } = {}, ): ThunkAction { log.debug('background.cancelTransaction'); let newTxId: number; return (dispatch: MetaMaskReduxDispatch) => { const actionId = generateActionId(); return new Promise((resolve, reject) => { callBackgroundMethod( 'createCancelTransaction', [txId, customGasSettings, { ...options, actionId }], (err, newState) => { if (err) { dispatch(displayWarning(err)); reject(err); return; } if (newState) { const { currentNetworkTxList } = newState; const { id } = currentNetworkTxList[currentNetworkTxList.length - 1]; newTxId = id; resolve(newState); } }, actionId, ); }) .then((newState) => dispatch(updateMetamaskState(newState))) .then(() => newTxId); }; } export function createSpeedUpTransaction( txId: string, customGasSettings: CustomGasSettings, options: { estimatedBaseFee?: string } = {}, ): ThunkAction { log.debug('background.createSpeedUpTransaction'); let newTx: TransactionMeta; return (dispatch: MetaMaskReduxDispatch) => { const actionId = generateActionId(); return new Promise((resolve, reject) => { callBackgroundMethod( 'createSpeedUpTransaction', [txId, customGasSettings, { ...options, actionId }], (err, newState) => { if (err) { dispatch(displayWarning(err)); reject(err); return; } if (newState) { const { currentNetworkTxList } = newState; newTx = currentNetworkTxList[currentNetworkTxList.length - 1]; resolve(newState); } }, actionId, ); }) .then((newState) => dispatch(updateMetamaskState(newState))) .then(() => newTx); }; } export function createRetryTransaction( txId: string, customGasSettings: CustomGasSettings, ): ThunkAction { let newTx: TransactionMeta; return (dispatch: MetaMaskReduxDispatch) => { return new Promise((resolve, reject) => { const actionId = generateActionId(); callBackgroundMethod( 'createSpeedUpTransaction', [txId, customGasSettings, { actionId }], (err, newState) => { if (err) { dispatch(displayWarning(err)); reject(err); return; } if (newState) { const { currentNetworkTxList } = newState; newTx = currentNetworkTxList[currentNetworkTxList.length - 1]; resolve(newState); } }, ); }) .then((newState) => dispatch(updateMetamaskState(newState))) .then(() => newTx); }; } // // config // export function setProviderType( type: NetworkType, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { log.debug(`background.setProviderType`, type); try { await submitRequestToBackground('setProviderType', [type]); } catch (error) { logErrorWithMessage(error); dispatch(displayWarning('Had a problem changing networks!')); } }; } export function upsertNetworkConfiguration( { rpcUrl, chainId, nickname, rpcPrefs, ticker = EtherDenomination.ETH, }: { rpcUrl: string; chainId: string; nickname: string; rpcPrefs: RPCDefinition['rpcPrefs']; ticker: string; }, { setActive, source, }: { setActive: boolean; source: string; }, ): ThunkAction { return async (dispatch) => { log.debug( `background.upsertNetworkConfiguration: ${rpcUrl} ${chainId} ${ticker} ${nickname}`, ); let networkConfigurationId; try { networkConfigurationId = await submitRequestToBackground( 'upsertNetworkConfiguration', [ { rpcUrl, chainId, ticker, nickname: nickname || rpcUrl, rpcPrefs }, { setActive, source, referrer: ORIGIN_METAMASK }, ], ); } catch (error) { log.error(error); dispatch(displayWarning('Had a problem adding network!')); } return networkConfigurationId; }; } export function editAndSetNetworkConfiguration( { networkConfigurationId, rpcUrl, chainId, nickname, rpcPrefs, ticker = EtherDenomination.ETH, }: { networkConfigurationId: string; rpcUrl: string; chainId: string; nickname: string; rpcPrefs: RPCDefinition['rpcPrefs']; ticker: string; }, { source }: { source: string }, ): ThunkAction { return async (dispatch) => { log.debug( `background.removeNetworkConfiguration: ${networkConfigurationId}`, ); try { await submitRequestToBackground('removeNetworkConfiguration', [ networkConfigurationId, ]); } catch (error) { logErrorWithMessage(error); dispatch(displayWarning('Had a problem removing network!')); return; } try { await submitRequestToBackground('upsertNetworkConfiguration', [ { rpcUrl, chainId, ticker, nickname: nickname || rpcUrl, rpcPrefs, }, { setActive: true, referrer: ORIGIN_METAMASK, source }, ]); } catch (error) { logErrorWithMessage(error); dispatch(displayWarning('Had a problem changing networks!')); } }; } export function setActiveNetwork( networkConfigurationId: string, ): ThunkAction { return async (dispatch) => { log.debug(`background.setActiveNetwork: ${networkConfigurationId}`); try { await submitRequestToBackground('setActiveNetwork', [ networkConfigurationId, ]); } catch (error) { logErrorWithMessage(error); dispatch(displayWarning('Had a problem changing networks!')); } }; } export function rollbackToPreviousProvider(): ThunkAction< void, MetaMaskReduxState, unknown, AnyAction > { return async (dispatch: MetaMaskReduxDispatch) => { try { await submitRequestToBackground('rollbackToPreviousProvider'); } catch (error) { logErrorWithMessage(error); dispatch(displayWarning('Had a problem changing networks!')); } }; } export function removeNetworkConfiguration( networkConfigurationId: string, ): ThunkAction, MetaMaskReduxState, unknown, AnyAction> { return (dispatch) => { log.debug( `background.removeNetworkConfiguration: ${networkConfigurationId}`, ); return new Promise((resolve, reject) => { callBackgroundMethod( 'removeNetworkConfiguration', [networkConfigurationId], (err) => { if (err) { logErrorWithMessage(err); dispatch(displayWarning('Had a problem removing network!')); reject(err); return; } resolve(); }, ); }); }; } // Calls the addressBookController to add a new address. export function addToAddressBook( recipient: string, nickname = '', memo = '', ): ThunkAction { log.debug(`background.addToAddressBook`); return async (dispatch, getState) => { const { chainId } = getState().metamask.provider; let set; try { set = await submitRequestToBackground('setAddressBook', [ toChecksumHexAddress(recipient), nickname, chainId, memo, ]); } catch (error) { logErrorWithMessage(error); dispatch(displayWarning('Address book failed to update')); throw error; } if (!set) { dispatch(displayWarning('Address book failed to update')); } }; } /** * @description Calls the addressBookController to remove an existing address. * @param chainId * @param addressToRemove - Address of the entry to remove from the address book */ export function removeFromAddressBook( chainId: string, addressToRemove: string, ): ThunkAction { log.debug(`background.removeFromAddressBook`); return async () => { await submitRequestToBackground('removeFromAddressBook', [ chainId, toChecksumHexAddress(addressToRemove), ]); }; } export function showNetworkDropdown(): Action { return { type: actionConstants.NETWORK_DROPDOWN_OPEN, }; } export function hideNetworkDropdown() { return { type: actionConstants.NETWORK_DROPDOWN_CLOSE, }; } type ModalPayload = { name: string } & Record; export function showModal(payload: ModalPayload): PayloadAction { return { type: actionConstants.MODAL_OPEN, payload, }; } export function hideModal(): Action { return { type: actionConstants.MODAL_CLOSE, }; } export function closeCurrentNotificationWindow(): ThunkAction< void, MetaMaskReduxState, unknown, AnyAction > { return (_, getState) => { if ( getEnvironmentType() === ENVIRONMENT_TYPE_NOTIFICATION && !hasUnconfirmedTransactions(getState()) ) { closeNotificationPopup(); } }; } export function showAlert(msg: string): PayloadAction { return { type: actionConstants.ALERT_OPEN, payload: msg, }; } export function hideAlert(): Action { return { type: actionConstants.ALERT_CLOSE, }; } /** * TODO: this should be moved somewhere else when it makese sense to do so */ interface NftDropDownState { [address: string]: { [chainId: string]: { [nftAddress: string]: boolean; }; }; } export function updateNftDropDownState( value: NftDropDownState, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { await submitRequestToBackground('updateNftDropDownState', [value]); await forceUpdateMetamaskState(dispatch); }; } interface QrCodeData { // Address when a Ethereum Address has been detected type?: 'address' | string; // contains an address key when Ethereum Address detected values?: { address?: string } & Json; } /** * This action will receive two types of values via qrCodeData * an object with the following structure {type, values} * or null (used to clear the previous value) * * @param qrCodeData */ export function qrCodeDetected( qrCodeData: QrCodeData, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { await dispatch({ type: actionConstants.QR_CODE_DETECTED, value: qrCodeData, }); // If on the send page, the send slice will listen for the QR_CODE_DETECTED // action and update its state. Address changes need to recompute gasLimit // so we fire this method so that the send page gasLimit can be recomputed dispatch(computeEstimatedGasLimit()); }; } export function showLoadingIndication( message?: string | ReactFragment, ): PayloadAction { return { type: actionConstants.SHOW_LOADING, payload: message, }; } export function setHardwareWalletDefaultHdPath({ device, path, }: { device: HardwareDeviceNames; path: string; }): PayloadAction<{ device: HardwareDeviceNames; path: string }> { return { type: actionConstants.SET_HARDWARE_WALLET_DEFAULT_HD_PATH, payload: { device, path }, }; } export function hideLoadingIndication(): Action { return { type: actionConstants.HIDE_LOADING, }; } export function displayWarning(payload: unknown): PayloadAction { if (isErrorWithMessage(payload)) { return { type: actionConstants.DISPLAY_WARNING, payload: payload.message, }; } else if (typeof payload === 'string') { return { type: actionConstants.DISPLAY_WARNING, payload, }; } return { type: actionConstants.DISPLAY_WARNING, payload: `${payload}`, }; } export function hideWarning() { return { type: actionConstants.HIDE_WARNING, }; } export function exportAccount( password: string, address: string, ): ThunkAction { return function (dispatch) { dispatch(showLoadingIndication()); log.debug(`background.verifyPassword`); return new Promise((resolve, reject) => { callBackgroundMethod('verifyPassword', [password], function (err) { if (err) { log.error('Error in verifying password.'); dispatch(hideLoadingIndication()); dispatch(displayWarning('Incorrect Password.')); reject(err); return; } log.debug(`background.exportAccount`); callBackgroundMethod( 'exportAccount', [address, password], function (err2, result) { dispatch(hideLoadingIndication()); if (err2) { logErrorWithMessage(err2); dispatch(displayWarning('Had a problem exporting the account.')); reject(err2); return; } dispatch(showPrivateKey(result as string)); resolve(result as string); }, ); }); }); }; } export function exportAccounts( password: string, addresses: string[], ): ThunkAction, MetaMaskReduxState, unknown, AnyAction> { return function (dispatch) { log.debug(`background.verifyPassword`); return new Promise((resolve, reject) => { callBackgroundMethod('verifyPassword', [password], function (err) { if (err) { log.error('Error in submitting password.'); reject(err); return; } log.debug(`background.exportAccounts`); const accountPromises = addresses.map( (address) => new Promise((resolve2, reject2) => callBackgroundMethod( 'exportAccount', [address, password], function (err2, result) { if (err2) { logErrorWithMessage(err2); dispatch( displayWarning('Had a problem exporting the account.'), ); reject2(err2); return; } resolve2(result as string); }, ), ), ); resolve(Promise.all(accountPromises)); }); }); }; } export function showPrivateKey(key: string): PayloadAction { return { type: actionConstants.SHOW_PRIVATE_KEY, payload: key, }; } export function setAccountLabel( account: string, label: string, ): ThunkAction, MetaMaskReduxState, unknown, AnyAction> { return (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); log.debug(`background.setAccountLabel`); return new Promise((resolve, reject) => { callBackgroundMethod('setAccountLabel', [account, label], (err) => { dispatch(hideLoadingIndication()); if (err) { dispatch(displayWarning(err)); reject(err); return; } dispatch({ type: actionConstants.SET_ACCOUNT_LABEL, value: { account, label }, }); resolve(account); }); }); }; } export function clearAccountDetails(): Action { return { type: actionConstants.CLEAR_ACCOUNT_DETAILS, }; } export function showSendTokenPage(): Action { return { type: actionConstants.SHOW_SEND_TOKEN_PAGE, }; } // TODO: Lift to shared folder when it makes sense interface TemporaryFeatureFlagDef { [feature: string]: boolean; } interface TemporaryPreferenceFlagDef { [preference: string]: boolean; } export function setFeatureFlag( feature: string, activated: boolean, notificationType: string, ): ThunkAction< Promise, MetaMaskReduxState, unknown, AnyAction > { return (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); return new Promise((resolve, reject) => { callBackgroundMethod( 'setFeatureFlag', [feature, activated], (err, updatedFeatureFlags) => { dispatch(hideLoadingIndication()); if (err) { dispatch(displayWarning(err)); reject(err); return; } notificationType && dispatch(showModal({ name: notificationType })); resolve(updatedFeatureFlags as TemporaryFeatureFlagDef); }, ); }); }; } export function setPreference( preference: string, value: boolean | string, ): ThunkAction< Promise, MetaMaskReduxState, unknown, AnyAction > { return (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); return new Promise((resolve, reject) => { callBackgroundMethod( 'setPreference', [preference, value], (err, updatedPreferences) => { dispatch(hideLoadingIndication()); if (err) { dispatch(displayWarning(err)); reject(err); return; } resolve(updatedPreferences as TemporaryPreferenceFlagDef); }, ); }); }; } export function setDefaultHomeActiveTabName( value: string, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { await submitRequestToBackground('setDefaultHomeActiveTabName', [value]); await forceUpdateMetamaskState(dispatch); }; } export function setUseNativeCurrencyAsPrimaryCurrencyPreference( value: boolean, ) { return setPreference('useNativeCurrencyAsPrimaryCurrency', value); } export function setHideZeroBalanceTokens(value: boolean) { return setPreference('hideZeroBalanceTokens', value); } export function setShowFiatConversionOnTestnetsPreference(value: boolean) { return setPreference('showFiatInTestnets', value); } export function setShowTestNetworks(value: boolean) { return setPreference('showTestNetworks', value); } export function setAutoLockTimeLimit(value: boolean) { return setPreference('autoLockTimeLimit', value); } export function setCompletedOnboarding(): ThunkAction< void, MetaMaskReduxState, unknown, AnyAction > { return async (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); try { await submitRequestToBackground('completeOnboarding'); dispatch(completeOnboarding()); } catch (err) { dispatch(displayWarning(err)); throw err; } finally { dispatch(hideLoadingIndication()); } }; } export function completeOnboarding() { return { type: actionConstants.COMPLETE_ONBOARDING, }; } export function setMouseUserState( isMouseUser: boolean, ): PayloadAction { return { type: actionConstants.SET_MOUSE_USER_STATE, payload: isMouseUser, }; } export async function forceUpdateMetamaskState( dispatch: MetaMaskReduxDispatch, ) { log.debug(`background.getState`); let newState; try { newState = await submitRequestToBackground( 'getState', ); } catch (error) { dispatch(displayWarning(error)); throw error; } dispatch(updateMetamaskState(newState)); return newState; } export function toggleAccountMenu() { return { type: actionConstants.TOGGLE_ACCOUNT_MENU, }; } export function toggleNetworkMenu() { return { type: actionConstants.TOGGLE_NETWORK_MENU, }; } export function setParticipateInMetaMetrics( participationPreference: boolean, ): ThunkAction< Promise<[boolean, string]>, MetaMaskReduxState, unknown, AnyAction > { return (dispatch: MetaMaskReduxDispatch) => { log.debug(`background.setParticipateInMetaMetrics`); return new Promise((resolve, reject) => { callBackgroundMethod( 'setParticipateInMetaMetrics', [participationPreference], (err, metaMetricsId) => { log.debug(err); if (err) { dispatch(displayWarning(err)); reject(err); return; } dispatch({ type: actionConstants.SET_PARTICIPATE_IN_METAMETRICS, value: participationPreference, }); resolve([participationPreference, metaMetricsId as string]); }, ); }); }; } export function setUseBlockie( val: boolean, ): ThunkAction { return (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); log.debug(`background.setUseBlockie`); callBackgroundMethod('setUseBlockie', [val], (err) => { dispatch(hideLoadingIndication()); if (err) { dispatch(displayWarning(err)); } }); }; } export function setUseNonceField( val: boolean, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); log.debug(`background.setUseNonceField`); try { await submitRequestToBackground('setUseNonceField', [val]); } catch (error) { dispatch(displayWarning(error)); } dispatch(hideLoadingIndication()); }; } export function setUsePhishDetect( val: boolean, ): ThunkAction { return (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); log.debug(`background.setUsePhishDetect`); callBackgroundMethod('setUsePhishDetect', [val], (err) => { dispatch(hideLoadingIndication()); if (err) { dispatch(displayWarning(err)); } }); }; } export function setUseMultiAccountBalanceChecker( val: boolean, ): ThunkAction { return (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); log.debug(`background.setUseMultiAccountBalanceChecker`); callBackgroundMethod('setUseMultiAccountBalanceChecker', [val], (err) => { dispatch(hideLoadingIndication()); if (err) { dispatch(displayWarning(err)); } }); }; } export function setUseTokenDetection( val: boolean, ): ThunkAction { return (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); log.debug(`background.setUseTokenDetection`); callBackgroundMethod('setUseTokenDetection', [val], (err) => { dispatch(hideLoadingIndication()); if (err) { dispatch(displayWarning(err)); } }); }; } export function setUseNftDetection( val: boolean, ): ThunkAction { return (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); log.debug(`background.setUseNftDetection`); callBackgroundMethod('setUseNftDetection', [val], (err) => { dispatch(hideLoadingIndication()); if (err) { dispatch(displayWarning(err)); } }); }; } export function setUseCurrencyRateCheck( val: boolean, ): ThunkAction { return (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); log.debug(`background.setUseCurrencyRateCheck`); callBackgroundMethod('setUseCurrencyRateCheck', [val], (err) => { dispatch(hideLoadingIndication()); if (err) { dispatch(displayWarning(err)); } }); }; } export function setOpenSeaEnabled( val: boolean, ): ThunkAction { return (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); log.debug(`background.setOpenSeaEnabled`); callBackgroundMethod('setOpenSeaEnabled', [val], (err) => { dispatch(hideLoadingIndication()); if (err) { dispatch(displayWarning(err)); } }); }; } export function detectNfts(): ThunkAction< void, MetaMaskReduxState, unknown, AnyAction > { return async (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); log.debug(`background.detectNfts`); await submitRequestToBackground('detectNfts'); dispatch(hideLoadingIndication()); await forceUpdateMetamaskState(dispatch); }; } export function setAdvancedGasFee( val: { maxBaseFee?: Hex; priorityFee?: Hex } | null, ): ThunkAction { return (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); log.debug(`background.setAdvancedGasFee`); callBackgroundMethod('setAdvancedGasFee', [val], (err) => { dispatch(hideLoadingIndication()); if (err) { dispatch(displayWarning(err)); } }); }; } export function setTheme( val: ThemeType, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); log.debug(`background.setTheme`); try { await submitRequestToBackground('setTheme', [val]); } finally { dispatch(hideLoadingIndication()); } }; } export function setIpfsGateway( val: string, ): ThunkAction { return (dispatch: MetaMaskReduxDispatch) => { log.debug(`background.setIpfsGateway`); callBackgroundMethod('setIpfsGateway', [val], (err) => { if (err) { dispatch(displayWarning(err)); } }); }; } export function updateCurrentLocale( key: string, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); try { await loadRelativeTimeFormatLocaleData(key); const localeMessages = await fetchLocale(key); const textDirection = await submitRequestToBackground< 'rtl' | 'ltr' | 'auto' >('setCurrentLocale', [key]); await switchDirection(textDirection); dispatch(setCurrentLocale(key, localeMessages)); } catch (error) { dispatch(displayWarning(error)); return; } finally { dispatch(hideLoadingIndication()); } }; } export function setCurrentLocale( locale: string, messages: { [translationKey: string]: { message: string; description?: string }; }, ): PayloadAction<{ locale: string; messages: { [translationKey: string]: { message: string; description?: string }; }; }> { return { type: actionConstants.SET_CURRENT_LOCALE, payload: { locale, messages, }, }; } export function setPendingTokens(pendingTokens: { customToken?: Token; selectedTokens?: { [address: string]: Token & { isCustom?: boolean; unlisted?: boolean }; }; tokenAddressList: string[]; }) { const { customToken, selectedTokens = {}, tokenAddressList = [], } = pendingTokens; const tokens = customToken?.address && customToken?.symbol && Boolean(customToken?.decimals >= 0 && customToken?.decimals <= 36) ? { ...selectedTokens, [customToken.address]: { ...customToken, isCustom: true, }, } : selectedTokens; Object.keys(tokens).forEach((tokenAddress) => { tokens[tokenAddress].unlisted = !tokenAddressList.find((addr) => isEqualCaseInsensitive(addr, tokenAddress), ); }); return { type: actionConstants.SET_PENDING_TOKENS, payload: tokens, }; } // Swaps export function setSwapsLiveness( swapsLiveness: boolean, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { await submitRequestToBackground('setSwapsLiveness', [swapsLiveness]); await forceUpdateMetamaskState(dispatch); }; } export function setSwapsFeatureFlags( featureFlags: TemporaryFeatureFlagDef, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { await submitRequestToBackground('setSwapsFeatureFlags', [featureFlags]); await forceUpdateMetamaskState(dispatch); }; } export function fetchAndSetQuotes( fetchParams: { slippage: string; sourceToken: string; destinationToken: string; value: string; fromAddress: string; destinationTokenAddedForSwap: string; balanceError: string; sourceDecimals: number; }, fetchParamsMetaData: { sourceTokenInfo: Token; destinationTokenInfo: Token; accountBalance: string; chainId: string; }, ): ThunkAction< Promise< [ { destinationAmount: string; decimals: number; aggregator: string }, string, ] >, MetaMaskReduxState, unknown, AnyAction > { return async (dispatch: MetaMaskReduxDispatch) => { const [quotes, selectedAggId] = await submitRequestToBackground( 'fetchAndSetQuotes', [fetchParams, fetchParamsMetaData], ); await forceUpdateMetamaskState(dispatch); return [quotes, selectedAggId]; }; } export function setSelectedQuoteAggId( aggId: string, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { await submitRequestToBackground('setSelectedQuoteAggId', [aggId]); await forceUpdateMetamaskState(dispatch); }; } export function setSwapsTokens( tokens: Token[], ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { await submitRequestToBackground('setSwapsTokens', [tokens]); await forceUpdateMetamaskState(dispatch); }; } export function clearSwapsQuotes(): ThunkAction< void, MetaMaskReduxState, unknown, AnyAction > { return async (dispatch: MetaMaskReduxDispatch) => { await submitRequestToBackground('clearSwapsQuotes'); await forceUpdateMetamaskState(dispatch); }; } export function resetBackgroundSwapsState(): ThunkAction< void, MetaMaskReduxState, unknown, AnyAction > { return async (dispatch: MetaMaskReduxDispatch) => { await submitRequestToBackground('resetSwapsState'); await forceUpdateMetamaskState(dispatch); }; } export function setCustomApproveTxData( data: string, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { await submitRequestToBackground('setCustomApproveTxData', [data]); await forceUpdateMetamaskState(dispatch); }; } export function setSwapsTxGasPrice( gasPrice: string, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { await submitRequestToBackground('setSwapsTxGasPrice', [gasPrice]); await forceUpdateMetamaskState(dispatch); }; } export function setSwapsTxGasLimit( gasLimit: string, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { await submitRequestToBackground('setSwapsTxGasLimit', [gasLimit, true]); await forceUpdateMetamaskState(dispatch); }; } export function updateCustomSwapsEIP1559GasParams({ gasLimit, maxFeePerGas, maxPriorityFeePerGas, }: { gasLimit: string; maxFeePerGas: string; maxPriorityFeePerGas: string; }): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { await Promise.all([ submitRequestToBackground('setSwapsTxGasLimit', [gasLimit]), submitRequestToBackground('setSwapsTxMaxFeePerGas', [maxFeePerGas]), submitRequestToBackground('setSwapsTxMaxFeePriorityPerGas', [ maxPriorityFeePerGas, ]), ]); await forceUpdateMetamaskState(dispatch); }; } // Note that the type widening happening below will resolve when we switch gas // constants to TypeScript, at which point we'll get better type safety. // TODO: Remove this comment when gas constants is typescript export function updateSwapsUserFeeLevel( swapsCustomUserFeeLevel: PriorityLevels, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { await submitRequestToBackground('setSwapsUserFeeLevel', [ swapsCustomUserFeeLevel, ]); await forceUpdateMetamaskState(dispatch); }; } export function setSwapsQuotesPollingLimitEnabled( quotesPollingLimitEnabled: boolean, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { await submitRequestToBackground('setSwapsQuotesPollingLimitEnabled', [ quotesPollingLimitEnabled, ]); await forceUpdateMetamaskState(dispatch); }; } export function setTradeTxId( tradeTxId: string, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { await submitRequestToBackground('setTradeTxId', [tradeTxId]); await forceUpdateMetamaskState(dispatch); }; } export function setApproveTxId( approveTxId: string, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { await submitRequestToBackground('setApproveTxId', [approveTxId]); await forceUpdateMetamaskState(dispatch); }; } export function safeRefetchQuotes(): ThunkAction< void, MetaMaskReduxState, unknown, AnyAction > { return async (dispatch: MetaMaskReduxDispatch) => { await submitRequestToBackground('safeRefetchQuotes'); await forceUpdateMetamaskState(dispatch); }; } export function stopPollingForQuotes(): ThunkAction< void, MetaMaskReduxState, unknown, AnyAction > { return async (dispatch: MetaMaskReduxDispatch) => { await submitRequestToBackground('stopPollingForQuotes'); await forceUpdateMetamaskState(dispatch); }; } export function setBackgroundSwapRouteState( routeState: '' | 'loading' | 'awaiting' | 'smartTransactionStatus', ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { await submitRequestToBackground('setBackgroundSwapRouteState', [ routeState, ]); await forceUpdateMetamaskState(dispatch); }; } export function resetSwapsPostFetchState(): ThunkAction< void, MetaMaskReduxState, unknown, AnyAction > { return async (dispatch: MetaMaskReduxDispatch) => { await submitRequestToBackground('resetPostFetchState'); await forceUpdateMetamaskState(dispatch); }; } export function setSwapsErrorKey( errorKey: string, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { await submitRequestToBackground('setSwapsErrorKey', [errorKey]); await forceUpdateMetamaskState(dispatch); }; } export function setInitialGasEstimate( initialAggId: string, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { await submitRequestToBackground('setInitialGasEstimate', [initialAggId]); await forceUpdateMetamaskState(dispatch); }; } // Permissions export function requestAccountsPermissionWithId( origin: string, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { const id = await submitRequestToBackground( 'requestAccountsPermissionWithId', [origin], ); await forceUpdateMetamaskState(dispatch); return id; }; } /** * Approves the permissions request. * * @param request - The permissions request to approve. */ export function approvePermissionsRequest( request: PermissionsRequest, ): ThunkAction { return (dispatch: MetaMaskReduxDispatch) => { callBackgroundMethod('approvePermissionsRequest', [request], (err) => { if (err) { dispatch(displayWarning(err)); } forceUpdateMetamaskState(dispatch); }); }; } /** * Rejects the permissions request with the given ID. * * @param requestId - The id of the request to be rejected */ export function rejectPermissionsRequest( requestId: string, ): ThunkAction { return (dispatch: MetaMaskReduxDispatch) => { return new Promise((resolve, reject) => { callBackgroundMethod('rejectPermissionsRequest', [requestId], (err) => { if (err) { dispatch(displayWarning(err)); reject(err); return; } forceUpdateMetamaskState(dispatch).then(resolve).catch(reject); }); }); }; } /** * Clears the given permissions for the given origin. * * @param subjects */ export function removePermissionsFor( subjects: Record>, ): ThunkAction { return (dispatch: MetaMaskReduxDispatch) => { callBackgroundMethod('removePermissionsFor', [subjects], (err) => { if (err) { dispatch(displayWarning(err)); } }); }; } ///: BEGIN:ONLY_INCLUDE_IN(flask) /** * Updates the caveat value for the specified origin, permission and caveat type. * * @param origin * @param target * @param caveatType * @param caveatValue */ export function updateCaveat( origin: string, target: string, caveatType: string, caveatValue: Record, ): ThunkAction { return (dispatch) => { callBackgroundMethod( 'updateCaveat', [origin, target, caveatType, caveatValue], (err) => { if (err) { dispatch(displayWarning(err)); } }, ); }; } ///: END:ONLY_INCLUDE_IN // Pending Approvals /** * Resolves a pending approval and closes the current notification window if no * further approvals are pending after the background state updates. * * @param id - The pending approval id * @param [value] - The value required to confirm a pending approval */ export function resolvePendingApproval( id: string, value: unknown, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { await submitRequestToBackground('resolvePendingApproval', [id, value]); // Before closing the current window, check if any additional confirmations // are added as a result of this confirmation being accepted const { pendingApprovals } = await forceUpdateMetamaskState(dispatch); if (Object.values(pendingApprovals).length === 0) { dispatch(closeCurrentNotificationWindow()); } }; } /** * Rejects a pending approval and closes the current notification window if no * further approvals are pending after the background state updates. * * @param id - The pending approval id * @param [error] - The error to throw when rejecting the approval */ export function rejectPendingApproval( id: string, error: unknown, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { await submitRequestToBackground('rejectPendingApproval', [id, error]); // Before closing the current window, check if any additional confirmations // are added as a result of this confirmation being rejected const { pendingApprovals } = await forceUpdateMetamaskState(dispatch); if (Object.values(pendingApprovals).length === 0) { dispatch(closeCurrentNotificationWindow()); } }; } export function setFirstTimeFlowType( type: 'create' | 'import', ): ThunkAction { return (dispatch: MetaMaskReduxDispatch) => { log.debug(`background.setFirstTimeFlowType`); callBackgroundMethod('setFirstTimeFlowType', [type], (err) => { if (err) { dispatch(displayWarning(err)); } }); dispatch({ type: actionConstants.SET_FIRST_TIME_FLOW_TYPE, value: type, }); }; } export function setSelectedNetworkConfigurationId( networkConfigurationId: string, ): PayloadAction { return { type: actionConstants.SET_SELECTED_NETWORK_CONFIGURATION_ID, payload: networkConfigurationId, }; } export function setNewNetworkAdded({ networkConfigurationId, nickname, }: { networkConfigurationId: string; nickname: string; }): PayloadAction { return { type: actionConstants.SET_NEW_NETWORK_ADDED, payload: { networkConfigurationId, nickname }, }; } export function setNewNftAddedMessage( newNftAddedMessage: string, ): PayloadAction { return { type: actionConstants.SET_NEW_NFT_ADDED_MESSAGE, payload: newNftAddedMessage, }; } export function setRemoveNftMessage( removeNftMessage: string, ): PayloadAction { return { type: actionConstants.SET_REMOVE_NFT_MESSAGE, payload: removeNftMessage, }; } export function setNewTokensImported( newTokensImported: string, ): PayloadAction { return { type: actionConstants.SET_NEW_TOKENS_IMPORTED, payload: newTokensImported, }; } export function setLastActiveTime(): ThunkAction< void, MetaMaskReduxState, unknown, AnyAction > { return (dispatch: MetaMaskReduxDispatch) => { callBackgroundMethod('setLastActiveTime', [], (err) => { if (err) { dispatch(displayWarning(err)); } }); }; } export function setDismissSeedBackUpReminder( value: boolean, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); await submitRequestToBackground('setDismissSeedBackUpReminder', [value]); dispatch(hideLoadingIndication()); }; } export function setDisabledRpcMethodPreference( methodName: string, value: number, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); await submitRequestToBackground('setDisabledRpcMethodPreference', [ methodName, value, ]); dispatch(hideLoadingIndication()); }; } export function getRpcMethodPreferences(): ThunkAction< void, MetaMaskReduxState, unknown, AnyAction > { return async (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); await submitRequestToBackground('getRpcMethodPreferences', []); dispatch(hideLoadingIndication()); }; } export function setConnectedStatusPopoverHasBeenShown(): ThunkAction< void, MetaMaskReduxState, unknown, AnyAction > { return () => { callBackgroundMethod('setConnectedStatusPopoverHasBeenShown', [], (err) => { if (isErrorWithMessage(err)) { throw new Error(err.message); } }); }; } export function setRecoveryPhraseReminderHasBeenShown() { return () => { callBackgroundMethod('setRecoveryPhraseReminderHasBeenShown', [], (err) => { if (isErrorWithMessage(err)) { throw new Error(err.message); } }); }; } export function setRecoveryPhraseReminderLastShown( lastShown: number, ): ThunkAction { return () => { callBackgroundMethod( 'setRecoveryPhraseReminderLastShown', [lastShown], (err) => { if (isErrorWithMessage(err)) { throw new Error(err.message); } }, ); }; } export function setTermsOfUseLastAgreed(lastAgreed: number) { return async () => { await submitRequestToBackground('setTermsOfUseLastAgreed', [lastAgreed]); }; } export function setOutdatedBrowserWarningLastShown(lastShown: number) { return async () => { await submitRequestToBackground('setOutdatedBrowserWarningLastShown', [ lastShown, ]); }; } export function getContractMethodData( data = '', ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch, getState) => { const prefixedData = addHexPrefix(data); const fourBytePrefix = prefixedData.slice(0, 10); if (fourBytePrefix.length < 10) { return {}; } const { knownMethodData } = getState().metamask; if ( knownMethodData?.[fourBytePrefix] && Object.keys(knownMethodData[fourBytePrefix]).length !== 0 ) { return knownMethodData[fourBytePrefix]; } log.debug(`loadingMethodData`); const { name, params } = (await getMethodDataAsync(fourBytePrefix)) as { name: string; params: unknown; }; callBackgroundMethod( 'addKnownMethodData', [fourBytePrefix, { name, params }], (err) => { if (err) { dispatch(displayWarning(err)); } }, ); return { name, params }; }; } export function setSeedPhraseBackedUp( seedPhraseBackupState: boolean, ): ThunkAction { return (dispatch: MetaMaskReduxDispatch) => { log.debug(`background.setSeedPhraseBackedUp`); return new Promise((resolve, reject) => { callBackgroundMethod( 'setSeedPhraseBackedUp', [seedPhraseBackupState], (err) => { if (err) { dispatch(displayWarning(err)); reject(err); return; } forceUpdateMetamaskState(dispatch).then(resolve).catch(reject); }, ); }); }; } export function setNextNonce(nextNonce: string): PayloadAction { return { type: actionConstants.SET_NEXT_NONCE, payload: nextNonce, }; } /** * This function initiates the nonceLock in the background for the given * address, and returns the next nonce to use. It then calls setNextNonce which * sets the nonce in state on the nextNonce key. NOTE: The nextNonce key is * actually ephemeral application state. It does not appear to be part of the * background state. * * TODO: move this to a different slice, MetaMask slice will eventually be * deprecated because it should not contain any ephemeral/app state but just * background state. In addition we should key nextNonce by address to prevent * accidental usage of a stale nonce as the call to getNextNonce only works for * the currently selected address. * * @returns */ export function getNextNonce(): ThunkAction< Promise, MetaMaskReduxState, unknown, AnyAction > { return async (dispatch, getState) => { const address = getState().metamask.selectedAddress; let nextNonce; try { nextNonce = await submitRequestToBackground('getNextNonce', [ address, ]); } catch (error) { dispatch(displayWarning(error)); throw error; } dispatch(setNextNonce(nextNonce)); return nextNonce; }; } export function setRequestAccountTabIds(requestAccountTabIds: { [origin: string]: string; }): PayloadAction<{ [origin: string]: string; }> { return { type: actionConstants.SET_REQUEST_ACCOUNT_TABS, payload: requestAccountTabIds, }; } export function getRequestAccountTabIds(): ThunkAction< void, MetaMaskReduxState, unknown, AnyAction > { return async (dispatch: MetaMaskReduxDispatch) => { const requestAccountTabIds = await submitRequestToBackground<{ [origin: string]: string; }>('getRequestAccountTabIds'); dispatch(setRequestAccountTabIds(requestAccountTabIds)); }; } export function setOpenMetamaskTabsIDs(openMetaMaskTabIDs: { [tabId: string]: boolean; }): PayloadAction<{ [tabId: string]: boolean }> { return { type: actionConstants.SET_OPEN_METAMASK_TAB_IDS, payload: openMetaMaskTabIDs, }; } export function getOpenMetamaskTabsIds(): ThunkAction< void, MetaMaskReduxState, unknown, AnyAction > { return async (dispatch: MetaMaskReduxDispatch) => { const openMetaMaskTabIDs = await submitRequestToBackground<{ [tabId: string]: boolean; }>('getOpenMetamaskTabsIds'); dispatch(setOpenMetamaskTabsIDs(openMetaMaskTabIDs)); }; } export function setLedgerTransportPreference( value: LedgerTransportTypes, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { dispatch(showLoadingIndication()); await submitRequestToBackground('setLedgerTransportPreference', [value]); dispatch(hideLoadingIndication()); }; } export async function attemptLedgerTransportCreation() { return await submitRequestToBackground('attemptLedgerTransportCreation'); } /** * This method deduplicates error reports to sentry by maintaining a state * object 'singleExceptions' in the app slice. The only place this state object * is accessed from is within this method, to check if it has already seen and * therefore tracked this error. This is to avoid overloading sentry with lots * of duplicate errors. * * @param error * @returns */ export function captureSingleException( error: string, ): ThunkAction { return async (dispatch, getState) => { const { singleExceptions } = getState().appState; if (!(error in singleExceptions)) { dispatch({ type: actionConstants.CAPTURE_SINGLE_EXCEPTION, value: error, }); captureException(Error(error)); } }; } // Wrappers around promisifedBackground /** * The "actions" below are not actions nor action creators. They cannot use * dispatch nor should they be dispatched when used. Instead they can be * called directly. These wrappers will be moved into their location at some * point in the future. */ export function estimateGas(params: TxParams): Promise { return submitRequestToBackground('estimateGas', [params]); } export async function updateTokenType( tokenAddress: string, ): Promise { try { return await submitRequestToBackground('updateTokenType', [tokenAddress]); } catch (error) { logErrorWithMessage(error); } return undefined; } /** * initiates polling for gas fee estimates. * * @returns a unique identify of the polling request that can be used * to remove that request from consideration of whether polling needs to * continue. */ export function getGasFeeEstimatesAndStartPolling(): Promise { return submitRequestToBackground('getGasFeeEstimatesAndStartPolling'); } /** * Informs the GasFeeController that a specific token is no longer requiring * gas fee estimates. If all tokens unsubscribe the controller stops polling. * * @param pollToken - Poll token received from calling * `getGasFeeEstimatesAndStartPolling`. */ export function disconnectGasFeeEstimatePoller(pollToken: string) { return submitRequestToBackground('disconnectGasFeeEstimatePoller', [ pollToken, ]); } export async function addPollingTokenToAppState(pollingToken: string) { return submitRequestToBackground('addPollingTokenToAppState', [ pollingToken, POLLING_TOKEN_ENVIRONMENT_TYPES[getEnvironmentType()], ]); } export async function removePollingTokenFromAppState(pollingToken: string) { return submitRequestToBackground('removePollingTokenFromAppState', [ pollingToken, POLLING_TOKEN_ENVIRONMENT_TYPES[getEnvironmentType()], ]); } export function getGasFeeTimeEstimate( maxPriorityFeePerGas: string, maxFeePerGas: string, ): Promise> { return submitRequestToBackground('getGasFeeTimeEstimate', [ maxPriorityFeePerGas, maxFeePerGas, ]); } export async function closeNotificationPopup() { await submitRequestToBackground('markNotificationPopupAsAutomaticallyClosed'); global.platform.closeCurrentWindow(); } /** * @param payload - details of the event to track * @param options - options for routing/handling of event * @returns */ export function trackMetaMetricsEvent( payload: MetaMetricsEventPayload, options?: MetaMetricsEventOptions, ) { return submitRequestToBackground('trackMetaMetricsEvent', [ { ...payload, actionId: generateActionId() }, options, ]); } export function createEventFragment( options: MetaMetricsEventFragment, ): Promise { const actionId = generateActionId(); return submitRequestToBackground('createEventFragment', [ { ...options, actionId }, ]); } export function createTransactionEventFragment( transactionId: string, event: TransactionMetaMetricsEvent, ): Promise { const actionId = generateActionId(); return submitRequestToBackground('createTransactionEventFragment', [ transactionId, event, actionId, ]); } export function updateEventFragment( id: string, payload: MetaMetricsEventFragment, ) { return submitRequestToBackground('updateEventFragment', [id, payload]); } export function finalizeEventFragment( id: string, options?: { abandoned?: boolean; page?: MetaMetricsPageObject; referrer?: MetaMetricsReferrerObject; }, ) { return submitRequestToBackground('finalizeEventFragment', [id, options]); } /** * @param payload - details of the page viewed * @param options - options for handling the page view */ export function trackMetaMetricsPage( payload: MetaMetricsPagePayload, options: MetaMetricsPageOptions, ) { return submitRequestToBackground('trackMetaMetricsPage', [ { ...payload, actionId: generateActionId() }, options, ]); } export function updateViewedNotifications(notificationIdViewedStatusMap: { [notificationId: string]: boolean; }) { return submitRequestToBackground('updateViewedNotifications', [ notificationIdViewedStatusMap, ]); } export async function setAlertEnabledness( alertId: string, enabledness: boolean, ) { await submitRequestToBackground('setAlertEnabledness', [ alertId, enabledness, ]); } export async function setUnconnectedAccountAlertShown(origin: string) { await submitRequestToBackground('setUnconnectedAccountAlertShown', [origin]); } export async function setWeb3ShimUsageAlertDismissed(origin: string) { await submitRequestToBackground('setWeb3ShimUsageAlertDismissed', [origin]); } // Smart Transactions Controller export async function setSmartTransactionsOptInStatus( optInState: boolean, prevOptInState: boolean, ) { trackMetaMetricsEvent({ actionId: generateActionId(), event: 'STX OptIn', category: MetaMetricsEventCategory.Swaps, sensitiveProperties: { stx_enabled: true, current_stx_enabled: true, stx_user_opt_in: optInState, stx_prev_user_opt_in: prevOptInState, }, }); await submitRequestToBackground('setSmartTransactionsOptInStatus', [ optInState, ]); } export function clearSmartTransactionFees() { submitRequestToBackground('clearSmartTransactionFees'); } export function fetchSmartTransactionFees( unsignedTransaction: Partial & { chainId: string }, approveTxParams: TxParams, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { if (approveTxParams) { approveTxParams.value = '0x0'; } try { const smartTransactionFees = await await submitRequestToBackground( 'fetchSmartTransactionFees', [unsignedTransaction, approveTxParams], ); dispatch({ type: actionConstants.SET_SMART_TRANSACTIONS_ERROR, payload: null, }); return smartTransactionFees; } catch (err) { logErrorWithMessage(err); if (isErrorWithMessage(err) && err.message.startsWith('Fetch error:')) { const errorObj = parseSmartTransactionsError(err.message); dispatch({ type: actionConstants.SET_SMART_TRANSACTIONS_ERROR, payload: errorObj, }); } throw err; } }; } interface TemporarySmartTransactionGasFees { maxFeePerGas: string; maxPriorityFeePerGas: string; gas: string; value: string; } const createSignedTransactions = async ( unsignedTransaction: Partial & { chainId: string }, fees: TemporarySmartTransactionGasFees[], areCancelTransactions?: boolean, ): Promise => { const unsignedTransactionsWithFees = fees.map((fee) => { const unsignedTransactionWithFees = { ...unsignedTransaction, maxFeePerGas: decimalToHex(fee.maxFeePerGas), maxPriorityFeePerGas: decimalToHex(fee.maxPriorityFeePerGas), gas: areCancelTransactions ? decimalToHex(21000) // It has to be 21000 for cancel transactions, otherwise the API would reject it. : unsignedTransaction.gas, value: unsignedTransaction.value, }; if (areCancelTransactions) { unsignedTransactionWithFees.to = unsignedTransactionWithFees.from; unsignedTransactionWithFees.data = '0x'; } return unsignedTransactionWithFees; }); const signedTransactions = await submitRequestToBackground( 'approveTransactionsWithSameNonce', [unsignedTransactionsWithFees], ); return signedTransactions; }; export function signAndSendSmartTransaction({ unsignedTransaction, smartTransactionFees, }: { unsignedTransaction: Partial & { chainId: string }; smartTransactionFees: { fees: TemporarySmartTransactionGasFees[]; cancelFees: TemporarySmartTransactionGasFees[]; }; }): ThunkAction, MetaMaskReduxState, unknown, AnyAction> { return async (dispatch: MetaMaskReduxDispatch) => { const signedTransactions = await createSignedTransactions( unsignedTransaction, smartTransactionFees.fees, ); const signedCanceledTransactions = await createSignedTransactions( unsignedTransaction, smartTransactionFees.cancelFees, true, ); try { const response = await submitRequestToBackground<{ uuid: string }>( 'submitSignedTransactions', [ { signedTransactions, signedCanceledTransactions, txParams: unsignedTransaction, }, ], ); // Returns e.g.: { uuid: 'dP23W7c2kt4FK9TmXOkz1UM2F20' } return response.uuid; } catch (err) { logErrorWithMessage(err); if (isErrorWithMessage(err) && err.message.startsWith('Fetch error:')) { const errorObj = parseSmartTransactionsError(err.message); dispatch({ type: actionConstants.SET_SMART_TRANSACTIONS_ERROR, payload: errorObj, }); } throw err; } }; } export function updateSmartTransaction( uuid: string, txMeta: TransactionMeta, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { try { await submitRequestToBackground('updateSmartTransaction', [ { uuid, ...txMeta, }, ]); } catch (err) { logErrorWithMessage(err); if (isErrorWithMessage(err) && err.message.startsWith('Fetch error:')) { const errorObj = parseSmartTransactionsError(err.message); dispatch({ type: actionConstants.SET_SMART_TRANSACTIONS_ERROR, payload: errorObj, }); } throw err; } }; } export function setSmartTransactionsRefreshInterval( refreshInterval: number, ): ThunkAction { return async () => { try { await submitRequestToBackground('setStatusRefreshInterval', [ refreshInterval, ]); } catch (err) { logErrorWithMessage(err); } }; } export function cancelSmartTransaction( uuid: string, ): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { try { await submitRequestToBackground('cancelSmartTransaction', [uuid]); } catch (err) { logErrorWithMessage(err); if (isErrorWithMessage(err) && err.message.startsWith('Fetch error:')) { const errorObj = parseSmartTransactionsError(err.message); dispatch({ type: actionConstants.SET_SMART_TRANSACTIONS_ERROR, payload: errorObj, }); } throw err; } }; } // TODO: codeword NOT_A_THUNK @brad-decker export function fetchSmartTransactionsLiveness() { return async () => { try { await submitRequestToBackground('fetchSmartTransactionsLiveness'); } catch (err) { logErrorWithMessage(err); } }; } export function dismissSmartTransactionsErrorMessage(): Action { return { type: actionConstants.DISMISS_SMART_TRANSACTIONS_ERROR_MESSAGE, }; } // DetectTokenController export async function detectNewTokens() { return submitRequestToBackground('detectNewTokens'); } // App state export function hideTestNetMessage() { return submitRequestToBackground('setShowTestnetMessageInDropdown', [false]); } export function hideBetaHeader() { return submitRequestToBackground('setShowBetaHeader', [false]); } export function hideProductTour() { return submitRequestToBackground('setShowProductTour', [false]); } // TODO: codeword NOT_A_THUNK @brad-decker export function setTransactionSecurityCheckEnabled( transactionSecurityCheckEnabled: boolean, ): ThunkAction { return async () => { try { await submitRequestToBackground('setTransactionSecurityCheckEnabled', [ transactionSecurityCheckEnabled, ]); } catch (error) { logErrorWithMessage(error); } }; } export function setFirstTimeUsedNetwork(chainId: string) { return submitRequestToBackground('setFirstTimeUsedNetwork', [chainId]); } // QR Hardware Wallets export async function submitQRHardwareCryptoHDKey(cbor: Hex) { await submitRequestToBackground('submitQRHardwareCryptoHDKey', [cbor]); } export async function submitQRHardwareCryptoAccount(cbor: Hex) { await submitRequestToBackground('submitQRHardwareCryptoAccount', [cbor]); } export function cancelSyncQRHardware(): ThunkAction< void, MetaMaskReduxState, unknown, AnyAction > { return async (dispatch: MetaMaskReduxDispatch) => { dispatch(hideLoadingIndication()); await submitRequestToBackground('cancelSyncQRHardware'); }; } export async function submitQRHardwareSignature(requestId: string, cbor: Hex) { await submitRequestToBackground('submitQRHardwareSignature', [ requestId, cbor, ]); } export function cancelQRHardwareSignRequest(): ThunkAction< void, MetaMaskReduxState, unknown, AnyAction > { return async (dispatch: MetaMaskReduxDispatch) => { dispatch(hideLoadingIndication()); await submitRequestToBackground('cancelQRHardwareSignRequest'); }; } export function requestUserApproval({ origin, type, requestData, }: { origin: string; type: string; requestData: object; }): ThunkAction { return async (dispatch: MetaMaskReduxDispatch) => { try { await submitRequestToBackground('requestUserApproval', [ { origin, type, requestData, }, ]); } catch (error) { logErrorWithMessage(error); dispatch(displayWarning('Had trouble requesting user approval')); } }; }