diff --git a/src/components/ManageAccountsDrawer/components/ImportAccounts.tsx b/src/components/ManageAccountsDrawer/components/ImportAccounts.tsx index c2d1d342cb3..ae32cd518e7 100644 --- a/src/components/ManageAccountsDrawer/components/ImportAccounts.tsx +++ b/src/components/ManageAccountsDrawer/components/ImportAccounts.tsx @@ -305,6 +305,11 @@ export const ImportAccounts = ({ chainId, onClose }: ImportAccountsProps) => { }, []) const handleDone = useCallback(async () => { + if (!walletDeviceId) { + console.error('Missing walletDeviceId') + return + } + if (isDemoWallet) { walletDispatch({ type: WalletActions.SET_WALLET_MODAL, payload: true }) accountManagementPopover.close() diff --git a/src/components/MultiHopTrade/hooks/useReceiveAddress.tsx b/src/components/MultiHopTrade/hooks/useReceiveAddress.tsx index f8825d8d403..00e6f4624a8 100644 --- a/src/components/MultiHopTrade/hooks/useReceiveAddress.tsx +++ b/src/components/MultiHopTrade/hooks/useReceiveAddress.tsx @@ -62,41 +62,42 @@ export const useReceiveAddress = ({ sellAccountId, buyAccountId, ], - queryFn: isInitializing - ? skipToken - : async () => { - // Already partially covered in isInitializing, but TypeScript lyfe mang. - if (!buyAsset || !wallet || !buyAccountId || !buyAccountMetadata) { - return undefined - } + queryFn: + !isInitializing && buyAsset && wallet && buyAccountId && buyAccountMetadata && deviceId + ? async () => { + // Already partially covered in isInitializing, but TypeScript lyfe mang. + if (!buyAsset || !wallet || !buyAccountId || !buyAccountMetadata || !deviceId) { + return undefined + } - const buyAssetChainId = buyAsset.chainId - const buyAssetAccountChainId = buyAccountId - ? fromAccountId(buyAccountId).chainId - : undefined + const buyAssetChainId = buyAsset.chainId + const buyAssetAccountChainId = buyAccountId + ? fromAccountId(buyAccountId).chainId + : undefined - /** - * do NOT remove - * super dangerous - don't use the wrong bip44 params to generate receive addresses - */ - if (buyAssetChainId !== buyAssetAccountChainId) { - return undefined - } + /** + * do NOT remove + * super dangerous - don't use the wrong bip44 params to generate receive addresses + */ + if (buyAssetChainId !== buyAssetAccountChainId) { + return undefined + } - if (isUtxoAccountId(buyAccountId) && !buyAccountMetadata?.accountType) - throw new Error(`Missing accountType for UTXO account ${buyAccountId}`) + if (isUtxoAccountId(buyAccountId) && !buyAccountMetadata?.accountType) + throw new Error(`Missing accountType for UTXO account ${buyAccountId}`) - const shouldFetchUnchainedAddress = Boolean(wallet && isLedger(wallet)) - const walletReceiveAddress = await getReceiveAddress({ - asset: buyAsset, - wallet, - accountMetadata: buyAccountMetadata, - deviceId, - pubKey: shouldFetchUnchainedAddress ? fromAccountId(buyAccountId).account : undefined, - }) + const shouldFetchUnchainedAddress = Boolean(wallet && isLedger(wallet)) + const walletReceiveAddress = await getReceiveAddress({ + asset: buyAsset, + wallet, + accountMetadata: buyAccountMetadata, + deviceId, + pubKey: shouldFetchUnchainedAddress ? fromAccountId(buyAccountId).account : undefined, + }) - return walletReceiveAddress - }, + return walletReceiveAddress + } + : skipToken, staleTime: Infinity, }) diff --git a/src/context/WalletProvider/KeepKey/components/Passphrase.tsx b/src/context/WalletProvider/KeepKey/components/Passphrase.tsx index 76559c9d7f6..67b77ed3b6f 100644 --- a/src/context/WalletProvider/KeepKey/components/Passphrase.tsx +++ b/src/context/WalletProvider/KeepKey/components/Passphrase.tsx @@ -22,7 +22,7 @@ export const KeepKeyPassphrase = () => { state: { deviceId, keyring }, dispatch, } = useWallet() - const wallet = keyring.get(deviceId) + const wallet = keyring.get(deviceId ?? '') const walletId = useAppSelector(selectWalletId) const appDispatch = useAppDispatch() diff --git a/src/context/WalletProvider/KeepKey/components/Pin.tsx b/src/context/WalletProvider/KeepKey/components/Pin.tsx index f599ba70e7c..ec95503e9b5 100644 --- a/src/context/WalletProvider/KeepKey/components/Pin.tsx +++ b/src/context/WalletProvider/KeepKey/components/Pin.tsx @@ -2,7 +2,7 @@ import type { ButtonProps, SimpleGridProps } from '@chakra-ui/react' import { Alert, AlertDescription, AlertIcon, Button, Input, SimpleGrid } from '@chakra-ui/react' import type { Event } from '@shapeshiftoss/hdwallet-core' import type { KeyboardEvent } from 'react' -import { useCallback, useEffect, useRef, useState } from 'react' +import { useCallback, useEffect, useMemo, useRef, useState } from 'react' import { CircleIcon } from 'components/Icons/Circle' import { Text } from 'components/Text' import { WalletActions } from 'context/WalletProvider/actions' @@ -38,7 +38,11 @@ export const KeepKeyPin = ({ }, dispatch, } = useWallet() - const wallet = keyring.get(deviceId) + + const wallet = useMemo(() => { + if (!deviceId) return null + return keyring.get(deviceId) + }, [deviceId, keyring]) const pinFieldRef = useRef(null) @@ -138,10 +142,10 @@ export const KeepKeyPin = ({ } } - keyring.on(['KeepKey', deviceId, String(MessageType.FAILURE)], handleError) + deviceId && keyring.on(['KeepKey', deviceId, String(MessageType.FAILURE)], handleError) return () => { - keyring.off(['KeepKey', deviceId, String(MessageType.FAILURE)], handleError) + deviceId && keyring.off(['KeepKey', deviceId, String(MessageType.FAILURE)], handleError) } }, [deviceId, keyring]) diff --git a/src/context/WalletProvider/Ledger/components/Chains.tsx b/src/context/WalletProvider/Ledger/components/Chains.tsx index fb4d50c6d85..a23814ae673 100644 --- a/src/context/WalletProvider/Ledger/components/Chains.tsx +++ b/src/context/WalletProvider/Ledger/components/Chains.tsx @@ -52,7 +52,9 @@ export const LedgerChains = () => { const handleConnectClick = useCallback( async (chainId: ChainId) => { - if (!walletState?.wallet) { + const { wallet, deviceId } = walletState + + if (!wallet || !deviceId) { console.error('No wallet found') return } @@ -67,7 +69,7 @@ export const LedgerChains = () => { ]({ accountNumber: 0, chainIds, - wallet: walletState.wallet, + wallet, isSnapInstalled: false, }) @@ -102,7 +104,7 @@ export const LedgerChains = () => { const accountMetadata = accountMetadataByAccountId[accountId] const payload = { accountMetadataByAccountId: { [accountId]: accountMetadata }, - walletId: walletState.deviceId, + walletId: deviceId, } dispatch(portfolio.actions.upsertAccountMetadata(payload)) @@ -120,7 +122,7 @@ export const LedgerChains = () => { setLoadingChains(prevLoading => ({ ...prevLoading, [chainId]: false })) } }, - [dispatch, walletState.deviceId, walletState.wallet], + [dispatch, walletState], ) const chainsRows = useMemo( diff --git a/src/context/WalletProvider/NativeWallet/components/EnterPassword.tsx b/src/context/WalletProvider/NativeWallet/components/EnterPassword.tsx index 9a48401dbbf..5bd48c4c05d 100644 --- a/src/context/WalletProvider/NativeWallet/components/EnterPassword.tsx +++ b/src/context/WalletProvider/NativeWallet/components/EnterPassword.tsx @@ -43,7 +43,7 @@ export const EnterPassword = () => { const translate = useTranslate() const { state, dispatch, disconnect } = useWallet() const localWallet = useLocalWallet() - const { deviceId, keyring } = state + const { nativeWalletPendingDeviceId: deviceId, keyring } = state const [showPw, setShowPw] = useState(false) @@ -58,6 +58,7 @@ export const EnterPassword = () => { const onSubmit = useCallback( async (values: FieldValues) => { try { + if (!deviceId) return const wallet = keyring.get(deviceId) const Vault = await import('@shapeshiftoss/hdwallet-native-vault').then(m => m.Vault) const vault = await Vault.open(deviceId, values.password) @@ -83,6 +84,7 @@ export const EnterPassword = () => { type: WalletActions.SET_IS_CONNECTED, payload: true, }) + dispatch({ type: WalletActions.RESET_NATIVE_PENDING_DEVICE_ID }) dispatch({ type: WalletActions.SET_LOCAL_WALLET_LOADING, payload: false }) dispatch({ type: WalletActions.SET_WALLET_MODAL, payload: false }) } catch (e) { diff --git a/src/context/WalletProvider/NativeWallet/components/NativeLoad.tsx b/src/context/WalletProvider/NativeWallet/components/NativeLoad.tsx index b4bad70d145..33f928d76b7 100644 --- a/src/context/WalletProvider/NativeWallet/components/NativeLoad.tsx +++ b/src/context/WalletProvider/NativeWallet/components/NativeLoad.tsx @@ -81,6 +81,13 @@ export const NativeLoad = ({ history }: RouteComponentProps) => { if (adapter) { const { name, icon } = NativeConfig try { + // Set a pending device ID so the event handler doesn't redirect the user to password input + // for the previous wallet + dispatch({ + type: WalletActions.SET_NATIVE_PENDING_DEVICE_ID, + payload: deviceId, + }) + const wallet = await adapter.pairDevice(deviceId) if (!(await wallet?.isInitialized())) { // This will trigger the password modal and the modal will set the wallet on state @@ -103,6 +110,7 @@ export const NativeLoad = ({ history }: RouteComponentProps) => { type: WalletActions.SET_IS_CONNECTED, payload: true, }) + dispatch({ type: WalletActions.RESET_NATIVE_PENDING_DEVICE_ID }) // The wallet is already initialized so we can close the modal dispatch({ type: WalletActions.SET_WALLET_MODAL, payload: false }) } diff --git a/src/context/WalletProvider/NativeWallet/hooks/useNativeEventHandler.ts b/src/context/WalletProvider/NativeWallet/hooks/useNativeEventHandler.ts index 0d2b822c41f..70b1f3c2fb7 100644 --- a/src/context/WalletProvider/NativeWallet/hooks/useNativeEventHandler.ts +++ b/src/context/WalletProvider/NativeWallet/hooks/useNativeEventHandler.ts @@ -6,6 +6,7 @@ import type { ActionTypes } from 'context/WalletProvider/actions' import { WalletActions } from 'context/WalletProvider/actions' import type { InitialState } from 'context/WalletProvider/WalletProvider' import { isMobile as isMobileApp } from 'lib/globals' +import { assertUnreachable } from 'lib/utils' export const useNativeEventHandler = (state: InitialState, dispatch: Dispatch) => { const { keyring, modal, modalType } = state @@ -13,13 +14,20 @@ export const useNativeEventHandler = (state: InitialState, dispatch: Dispatch { const handleEvent = (e: [deviceId: string, message: Event]) => { const deviceId = e[0] - switch (e[1].message_type) { + const messageType = e[1].message_type as NativeEvents + switch (messageType) { case NativeEvents.MNEMONIC_REQUIRED: if (!deviceId) break + + // Don't show password input for previous wallet when switching + if (deviceId !== state.nativeWalletPendingDeviceId) { + break + } + // If we're on the native mobile app we don't need to handle the MNEMONIC_REQUIRED event as we use the device's native authentication instead // Reacting to this event will incorrectly open the native password modal after authentication completes when on the mobile app if (isMobileApp) break - dispatch({ type: WalletActions.NATIVE_PASSWORD_OPEN, payload: { modal: true, deviceId } }) + dispatch({ type: WalletActions.NATIVE_PASSWORD_OPEN, payload: { modal: true } }) break case NativeEvents.READY: @@ -28,8 +36,7 @@ export const useNativeEventHandler = (state: InitialState, dispatch: Dispatch { // This effect should never run for wallets other than WalletConnectV2 since we explicitly tap into @walletconnect/ethereum-provider provider diff --git a/src/context/WalletProvider/WalletProvider.tsx b/src/context/WalletProvider/WalletProvider.tsx index 7b5923da73f..666ac67c75b 100644 --- a/src/context/WalletProvider/WalletProvider.tsx +++ b/src/context/WalletProvider/WalletProvider.tsx @@ -25,7 +25,7 @@ import { selectWalletRdns, selectWalletType, } from 'state/slices/localWalletSlice/selectors' -import { portfolio } from 'state/slices/portfolioSlice/portfolioSlice' +import { portfolio as portfolioSlice } from 'state/slices/portfolioSlice/portfolioSlice' import { store } from 'state/store' import type { ActionTypes } from './actions' @@ -93,11 +93,12 @@ export type InitialState = { isLocked: boolean modal: boolean isLoadingLocalWallet: boolean - deviceId: string + deviceId: string | null showBackButton: boolean keepKeyPinRequestType: PinMatrixRequestType | null deviceState: DeviceState disconnectOnCloseModal: boolean + nativeWalletPendingDeviceId: string | null } & ( | { modalType: KeyManager | null @@ -124,11 +125,12 @@ const initialState: InitialState = { isLocked: false, modal: false, isLoadingLocalWallet: false, - deviceId: '', + deviceId: null, showBackButton: true, keepKeyPinRequestType: null, deviceState: initialDeviceState, disconnectOnCloseModal: false, + nativeWalletPendingDeviceId: null, } const reducer = (state: InitialState, action: ActionTypes): InitialState => { @@ -140,6 +142,7 @@ const reducer = (state: InitialState, action: ActionTypes): InitialState => { if (currentConnectedType === 'walletconnectv2') { state.wallet?.disconnect?.() store.dispatch(localWalletSlice.actions.clearLocalWallet()) + store.dispatch(portfolioSlice.actions.setWalletMeta(undefined)) } const { deviceId, name, wallet, icon, meta, isDemoWallet, connectedType } = action.payload // set wallet metadata in redux store @@ -147,7 +150,7 @@ const reducer = (state: InitialState, action: ActionTypes): InitialState => { walletId: deviceId, walletName: name, } - store.dispatch(portfolio.actions.setWalletMeta(walletMeta)) + store.dispatch(portfolioSlice.actions.setWalletMeta(walletMeta)) return { ...state, deviceId, @@ -221,7 +224,8 @@ const reducer = (state: InitialState, action: ActionTypes): InitialState => { modal: action.payload.modal, modalType: KeyManager.Native, showBackButton: !state.isLoadingLocalWallet, - deviceId: action.payload.deviceId, + deviceId: null, + walletInfo: null, initialRoute: NativeWalletRoutes.EnterPassword, } case WalletActions.OPEN_KEEPKEY_PIN: { @@ -290,9 +294,10 @@ const reducer = (state: InitialState, action: ActionTypes): InitialState => { case WalletActions.SET_LOCAL_WALLET_LOADING: return { ...state, isLoadingLocalWallet: action.payload } case WalletActions.RESET_STATE: - const resetProperties = omit(initialState, ['keyring', 'adapters', 'modal', 'deviceId']) + const resetProperties = omit(initialState, ['keyring', 'adapters', 'modal']) // reset wallet meta in redux store - store.dispatch(portfolio.actions.setWalletMeta(undefined)) + store.dispatch(localWalletSlice.actions.clearLocalWallet()) + store.dispatch(portfolioSlice.actions.setWalletMeta(undefined)) return { ...state, ...resetProperties } // TODO: Remove this once we update SET_DEVICE_STATE to allow explicitly setting falsey values case WalletActions.RESET_LAST_DEVICE_INTERACTION_STATE: { @@ -320,6 +325,21 @@ const reducer = (state: InitialState, action: ActionTypes): InitialState => { modalType: KeyManager.KeepKey, initialRoute: KeepKeyRoutes.Disconnect, } + case WalletActions.SET_NATIVE_PENDING_DEVICE_ID: + store.dispatch(localWalletSlice.actions.clearLocalWallet()) + store.dispatch(portfolioSlice.actions.setWalletMeta(undefined)) + return { + ...state, + isConnected: false, + deviceId: null, + walletInfo: null, + nativeWalletPendingDeviceId: action.payload, + } + case WalletActions.RESET_NATIVE_PENDING_DEVICE_ID: + return { + ...state, + nativeWalletPendingDeviceId: null, + } default: return state } @@ -334,6 +354,7 @@ const getInitialState = () => { */ return { ...initialState, + nativeWalletPendingDeviceId: localWalletDeviceId, isLoadingLocalWallet: true, } } @@ -413,7 +434,6 @@ export const WalletProvider = ({ children }: { children: React.ReactNode }): JSX */ state.wallet?.disconnect?.() dispatch({ type: WalletActions.RESET_STATE }) - store.dispatch(localWalletSlice.actions.clearLocalWallet()) }, [state.wallet]) const load = useCallback(() => { diff --git a/src/context/WalletProvider/WalletViewsSwitch.tsx b/src/context/WalletProvider/WalletViewsSwitch.tsx index b500dac2fd0..65a2566231a 100644 --- a/src/context/WalletProvider/WalletViewsSwitch.tsx +++ b/src/context/WalletProvider/WalletViewsSwitch.tsx @@ -15,8 +15,6 @@ import { Route, Switch, useHistory, useLocation, useRouteMatch } from 'react-rou import { SlideTransition } from 'components/SlideTransition' import { WalletActions } from 'context/WalletProvider/actions' import { useWallet } from 'hooks/useWallet/useWallet' -import { localWalletSlice } from 'state/slices/localWalletSlice/localWalletSlice' -import { store } from 'state/store' import { SUPPORTED_WALLETS } from './config' import { KeyManager } from './KeyManager' @@ -63,14 +61,11 @@ export const WalletViewsSwitch = () => { if (disposition === 'initializing' || disposition === 'recovering') { await wallet?.cancel() disconnect() - store.dispatch(localWalletSlice.actions.clearLocalWallet()) dispatch({ type: WalletActions.OPEN_KEEPKEY_DISCONNECT }) } else { history.replace(INITIAL_WALLET_MODAL_ROUTE) if (disconnectOnCloseModal) { disconnect() - dispatch({ type: WalletActions.RESET_STATE }) - store.dispatch(localWalletSlice.actions.clearLocalWallet()) } else { dispatch({ type: WalletActions.SET_WALLET_MODAL, payload: false }) } diff --git a/src/context/WalletProvider/actions.ts b/src/context/WalletProvider/actions.ts index ca7e5acb850..54466c7e366 100644 --- a/src/context/WalletProvider/actions.ts +++ b/src/context/WalletProvider/actions.ts @@ -28,6 +28,8 @@ export enum WalletActions { OPEN_KEEPKEY_RECOVERY = 'OPEN_KEEPKEY_RECOVERY', OPEN_KEEPKEY_CHARACTER_REQUEST = 'OPEN_KEEPKEY_CHARACTER_REQUEST', DOWNLOAD_UPDATER = 'DOWNLOAD_UPDATER', + SET_NATIVE_PENDING_DEVICE_ID = 'SET_NATIVE_PENDING_DEVICE_ID', + RESET_NATIVE_PENDING_DEVICE_ID = 'RESET_NATIVE_PENDING_DEVICE_ID', } export type ActionTypes = @@ -62,7 +64,6 @@ export type ActionTypes = type: WalletActions.NATIVE_PASSWORD_OPEN payload: { modal: boolean - deviceId: string } } | { @@ -107,3 +108,8 @@ export type ActionTypes = } } | { type: WalletActions.OPEN_KEEPKEY_DISCONNECT } + | { + type: WalletActions.SET_NATIVE_PENDING_DEVICE_ID + payload: string + } + | { type: WalletActions.RESET_NATIVE_PENDING_DEVICE_ID } diff --git a/src/pages/Accounts/AddAccountModal.tsx b/src/pages/Accounts/AddAccountModal.tsx index 3b00b600744..5bcc970067d 100644 --- a/src/pages/Accounts/AddAccountModal.tsx +++ b/src/pages/Accounts/AddAccountModal.tsx @@ -98,6 +98,7 @@ export const AddAccountModal = () => { if (!wallet) return if (!selectedChainId) return if (!nextAccountNumber) return + if (!walletDeviceId) return ;(async () => { const accountNumber = nextAccountNumber const chainIds = [selectedChainId]