diff --git a/public/images/common/propose-recovery.svg b/public/images/common/propose-recovery.svg new file mode 100644 index 0000000000..897a20a7e5 --- /dev/null +++ b/public/images/common/propose-recovery.svg @@ -0,0 +1,29 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/src/components/common/PageLayout/index.tsx b/src/components/common/PageLayout/index.tsx index 317150384b..0df0489f86 100644 --- a/src/components/common/PageLayout/index.tsx +++ b/src/components/common/PageLayout/index.tsx @@ -11,6 +11,7 @@ import useDebounce from '@/hooks/useDebounce' import { useRouter } from 'next/router' import { TxModalContext } from '@/components/tx-flow' import BatchSidebar from '@/components/batch/BatchSidebar' +import { RecoveryModal } from '@/components/recovery/RecoveryModal' const isNoSidebarRoute = (pathname: string): boolean => { return [ @@ -60,7 +61,9 @@ const PageLayout = ({ pathname, children }: { pathname: string; children: ReactE })} >
- {children} + + {children} +
diff --git a/src/components/dashboard/RecoveryHeader/index.test.tsx b/src/components/dashboard/RecoveryHeader/index.test.tsx new file mode 100644 index 0000000000..fe89e17587 --- /dev/null +++ b/src/components/dashboard/RecoveryHeader/index.test.tsx @@ -0,0 +1,58 @@ +import { BigNumber } from 'ethers' + +import { _RecoveryHeader } from '.' +import { render } from '@/tests/test-utils' +import type { RecoveryQueueItem } from '@/store/recoverySlice' + +describe('RecoveryHeader', () => { + it('should not render a widget if the chain does not support recovery', () => { + const { container } = render( + <_RecoveryHeader + isOwner + isGuardian + queue={[{ validFrom: BigNumber.from(0) } as RecoveryQueueItem]} + supportsRecovery={false} + />, + ) + + expect(container).toBeEmptyDOMElement() + }) + + it('should render the in-progress widget if there is a queue for guardians', () => { + const { queryByText } = render( + <_RecoveryHeader + isOwner={false} + isGuardian + queue={[{ validFrom: BigNumber.from(0) } as RecoveryQueueItem]} + supportsRecovery + />, + ) + + expect(queryByText('Account recovery in progress')).toBeTruthy() + }) + + it('should render the in-progress widget if there is a queue for owners', () => { + const { queryByText } = render( + <_RecoveryHeader + isOwner + isGuardian={false} + queue={[{ validFrom: BigNumber.from(0) } as RecoveryQueueItem]} + supportsRecovery + />, + ) + + expect(queryByText('Account recovery in progress')).toBeTruthy() + }) + + it('should render the proposal widget when there is no queue for guardians', () => { + const { queryByText } = render(<_RecoveryHeader isOwner={false} isGuardian queue={[]} supportsRecovery />) + + expect(queryByText('Recover this Account')).toBeTruthy() + }) + + it('should not render the proposal widget when there is no queue for owners', () => { + const { container } = render(<_RecoveryHeader isOwner isGuardian={false} queue={[]} supportsRecovery />) + + expect(container).toBeEmptyDOMElement() + }) +}) diff --git a/src/components/dashboard/RecoveryHeader/index.tsx b/src/components/dashboard/RecoveryHeader/index.tsx new file mode 100644 index 0000000000..533b8aae2a --- /dev/null +++ b/src/components/dashboard/RecoveryHeader/index.tsx @@ -0,0 +1,57 @@ +import { Grid } from '@mui/material' +import type { ReactElement } from 'react' + +import { useRecoveryQueue } from '@/hooks/useRecoveryQueue' +import { useIsGuardian } from '@/hooks/useIsGuardian' +import madProps from '@/utils/mad-props' +import { FEATURES } from '@/utils/chains' +import { useHasFeature } from '@/hooks/useChains' +import { RecoveryProposalCard } from '@/components/recovery/RecoveryCards/RecoveryProposalCard' +import { RecoveryInProgressCard } from '@/components/recovery/RecoveryCards/RecoveryInProgressCard' +import { WidgetContainer, WidgetBody } from '../styled' +import useIsSafeOwner from '@/hooks/useIsSafeOwner' +import type { RecoveryQueueItem } from '@/store/recoverySlice' + +export function _RecoveryHeader({ + isGuardian, + supportsRecovery, + queue, +}: { + isOwner: boolean + isGuardian: boolean + supportsRecovery: boolean + queue: Array +}): ReactElement | null { + const next = queue[0] + + if (!supportsRecovery) { + return null + } + + const modal = next ? ( + + ) : isGuardian ? ( + + ) : null + + if (modal) { + return ( + + + {modal} + + + ) + } + return null +} + +// Appease TypeScript +const _useSupportedRecovery = () => useHasFeature(FEATURES.RECOVERY) + +export const RecoveryHeader = madProps(_RecoveryHeader, { + isOwner: useIsSafeOwner, + isGuardian: useIsGuardian, + supportsRecovery: _useSupportedRecovery, + queue: useRecoveryQueue, +}) diff --git a/src/components/dashboard/RecoveryInProgress/index.test.tsx b/src/components/dashboard/RecoveryInProgress/index.test.tsx deleted file mode 100644 index cc0d8ff299..0000000000 --- a/src/components/dashboard/RecoveryInProgress/index.test.tsx +++ /dev/null @@ -1,182 +0,0 @@ -import { render } from '@testing-library/react' -import { BigNumber } from 'ethers' - -import { _RecoveryInProgress } from '.' -import { useRecoveryTxState } from '@/hooks/useRecoveryTxState' -import type { RecoveryQueueItem } from '@/store/recoverySlice' - -jest.mock('@/hooks/useRecoveryTxState') - -const mockUseRecoveryTxState = useRecoveryTxState as jest.MockedFunction - -describe('RecoveryInProgress', () => { - beforeEach(() => { - jest.resetAllMocks() - }) - - it('should return null if the chain does not support recovery', () => { - mockUseRecoveryTxState.mockReturnValue({} as any) - - const result = render( - <_RecoveryInProgress - supportsRecovery={false} - timestamp={0} - queuedTxs={[{ timestamp: BigNumber.from(0) } as RecoveryQueueItem]} - />, - ) - - expect(result.container).toBeEmptyDOMElement() - }) - - it('should return null if there are no delayed transactions', () => { - mockUseRecoveryTxState.mockReturnValue({} as any) - - const result = render(<_RecoveryInProgress supportsRecovery={true} timestamp={69420} queuedTxs={[]} />) - - expect(result.container).toBeEmptyDOMElement() - }) - - it('should return null if all the delayed transactions are expired and invalid', () => { - mockUseRecoveryTxState.mockReturnValue({} as any) - - const result = render( - <_RecoveryInProgress - supportsRecovery={true} - timestamp={69420} - queuedTxs={[ - { - timestamp: BigNumber.from(0), - validFrom: BigNumber.from(69), - expiresAt: BigNumber.from(420), - } as RecoveryQueueItem, - ]} - />, - ) - - expect(result.container).toBeEmptyDOMElement() - }) - - it('should return the countdown of the next non-expired/invalid transactions if none are non-expired/valid', () => { - mockUseRecoveryTxState.mockReturnValue({ - remainingSeconds: 69 * 420 * 1337, - isExecutable: false, - isNext: true, - } as any) - - const mockBlockTimestamp = BigNumber.from(69420) - - const { queryByText } = render( - <_RecoveryInProgress - supportsRecovery={true} - timestamp={mockBlockTimestamp.toNumber()} - queuedTxs={[ - { - timestamp: mockBlockTimestamp.add(1), - validFrom: mockBlockTimestamp.add(1), // Invalid - expiresAt: mockBlockTimestamp.add(1), // Non-expired - } as RecoveryQueueItem, - { - // Older - should render this - timestamp: mockBlockTimestamp, - validFrom: mockBlockTimestamp.mul(4), // Invalid - expiresAt: null, // Non-expired - } as RecoveryQueueItem, - ]} - />, - ) - - expect(queryByText('Account recovery in progress')).toBeInTheDocument() - expect( - queryByText('The recovery process has started. This Account will be ready to recover in:'), - ).toBeInTheDocument() - ;['day', 'hr', 'min'].forEach((unit) => { - // May be pluralised - expect(queryByText(unit, { exact: false })).toBeInTheDocument() - }) - // Days - expect(queryByText('448')).toBeInTheDocument() - // Hours - expect(queryByText('10')).toBeInTheDocument() - // Mins - expect(queryByText('51')).toBeInTheDocument() - }) - - it('should return the info of the next non-expired/valid transaction', () => { - mockUseRecoveryTxState.mockReturnValue({ isExecutable: true, remainingSeconds: 0 } as any) - - const mockBlockTimestamp = BigNumber.from(69420) - - const { queryByText } = render( - <_RecoveryInProgress - supportsRecovery={true} - timestamp={mockBlockTimestamp.toNumber()} - queuedTxs={[ - { - timestamp: mockBlockTimestamp.sub(1), - validFrom: mockBlockTimestamp.sub(1), // Invalid - expiresAt: mockBlockTimestamp.sub(1), // Non-expired - } as RecoveryQueueItem, - { - // Older - should render this - timestamp: mockBlockTimestamp.sub(2), - validFrom: mockBlockTimestamp.sub(1), // Invalid - expiresAt: null, // Non-expired - } as RecoveryQueueItem, - ]} - />, - ) - - expect(queryByText('Account recovery possible')).toBeInTheDocument() - expect(queryByText('The recovery process is possible. This Account can be recovered.')).toBeInTheDocument() - ;['day', 'hr', 'min'].forEach((unit) => { - // May be pluralised - expect(queryByText(unit, { exact: false })).not.toBeInTheDocument() - }) - }) - - it('should return the intemediary info for of the queued, non-expired/valid transactions', () => { - mockUseRecoveryTxState.mockReturnValue({ - isExecutable: false, - isNext: false, - remainingSeconds: 69 * 420 * 1337, - } as any) - - const mockBlockTimestamp = BigNumber.from(69420) - - const { queryByText } = render( - <_RecoveryInProgress - supportsRecovery={true} - timestamp={mockBlockTimestamp.toNumber()} - queuedTxs={[ - { - timestamp: mockBlockTimestamp.sub(1), - validFrom: mockBlockTimestamp.sub(1), // Invalid - expiresAt: mockBlockTimestamp.sub(1), // Non-expired - } as RecoveryQueueItem, - { - // Older - should render this - timestamp: mockBlockTimestamp.sub(2), - validFrom: mockBlockTimestamp.sub(1), // Invalid - expiresAt: null, // Non-expired - } as RecoveryQueueItem, - ]} - />, - ) - - expect(queryByText('Account recovery in progress')).toBeInTheDocument() - expect( - queryByText( - 'The recovery process has started. This Account can be recovered after previous attempts are executed or skipped and the delay period has passed:', - ), - ) - ;['day', 'hr', 'min'].forEach((unit) => { - // May be pluralised - expect(queryByText(unit, { exact: false })).toBeInTheDocument() - }) - // Days - expect(queryByText('448')).toBeInTheDocument() - // Hours - expect(queryByText('10')).toBeInTheDocument() - // Mins - }) -}) diff --git a/src/components/dashboard/RecoveryInProgress/index.tsx b/src/components/dashboard/RecoveryInProgress/index.tsx deleted file mode 100644 index 90972699ec..0000000000 --- a/src/components/dashboard/RecoveryInProgress/index.tsx +++ /dev/null @@ -1,91 +0,0 @@ -import { Card, Grid, Typography } from '@mui/material' -import type { ReactElement } from 'react' - -import { useAppSelector } from '@/store' -import { useClock } from '@/hooks/useClock' -import { WidgetContainer, WidgetBody } from '../styled' -import RecoveryPending from '@/public/images/common/recovery-pending.svg' -import ExternalLink from '@/components/common/ExternalLink' -import { useHasFeature } from '@/hooks/useChains' -import { FEATURES } from '@/utils/chains' -import { selectRecoveryQueues } from '@/store/recoverySlice' -import madProps from '@/utils/mad-props' -import { Countdown } from '@/components/common/Countdown' -import { useRecoveryTxState } from '@/hooks/useRecoveryTxState' -import type { RecoveryQueueItem } from '@/store/recoverySlice' - -export function _RecoveryInProgress({ - timestamp, - supportsRecovery, - queuedTxs, -}: { - timestamp: number - supportsRecovery: boolean - queuedTxs: Array -}): ReactElement | null { - const nonExpiredTxs = queuedTxs.filter((queuedTx) => { - return queuedTx.expiresAt ? queuedTx.expiresAt.gt(timestamp) : true - }) - - if (!supportsRecovery || nonExpiredTxs.length === 0) { - return null - } - - // Conditional hook - return <_RecoveryInProgressWidget nextTx={nonExpiredTxs[0]} /> -} - -function _RecoveryInProgressWidget({ nextTx }: { nextTx: RecoveryQueueItem }): ReactElement { - const { isExecutable, isNext, remainingSeconds } = useRecoveryTxState(nextTx) - - // TODO: Migrate `isValid` components when https://github.com/safe-global/safe-wallet-web/issues/2758 is done - return ( - - - - - - - - - - - {isExecutable ? 'Account recovery possible' : 'Account recovery in progress'} - - - {isExecutable - ? 'The recovery process is possible. This Account can be recovered.' - : !isNext - ? remainingSeconds > 0 - ? 'The recovery process has started. This Account can be recovered after previous attempts are executed or skipped and the delay period has passed:' - : 'The recovery process has started. This Account can be recovered after previous attempts are executed or skipped.' - : 'The recovery process has started. This Account will be ready to recover in:'} - - - - - - Learn more - - - - - - - - ) -} - -// Appease React TypeScript warnings -const _useTimestamp = () => useClock(60_000) // Countdown does not display -const _useSupportsRecovery = () => useHasFeature(FEATURES.RECOVERY) -const _useQueuedRecoveryTxs = () => useAppSelector(selectRecoveryQueues) - -export const RecoveryInProgress = madProps(_RecoveryInProgress, { - timestamp: _useTimestamp, - supportsRecovery: _useSupportsRecovery, - queuedTxs: _useQueuedRecoveryTxs, -}) diff --git a/src/components/dashboard/index.tsx b/src/components/dashboard/index.tsx index 007d9179c7..d51d5f15cb 100644 --- a/src/components/dashboard/index.tsx +++ b/src/components/dashboard/index.tsx @@ -11,7 +11,7 @@ import { Recovery } from './Recovery' import { FEATURES } from '@/utils/chains' import { useHasFeature } from '@/hooks/useChains' import { CREATION_MODAL_QUERY_PARM } from '../new-safe/create/logic' -import { RecoveryInProgress } from './RecoveryInProgress' +import { RecoveryHeader } from './RecoveryHeader' const Dashboard = (): ReactElement => { const router = useRouter() @@ -21,7 +21,7 @@ const Dashboard = (): ReactElement => { return ( <> - + diff --git a/src/components/recovery/RecoveryCards/RecoveryInProgressCard.tsx b/src/components/recovery/RecoveryCards/RecoveryInProgressCard.tsx new file mode 100644 index 0000000000..2598a8b222 --- /dev/null +++ b/src/components/recovery/RecoveryCards/RecoveryInProgressCard.tsx @@ -0,0 +1,104 @@ +import { Button, Card, Divider, Grid, Typography } from '@mui/material' +import { useRouter } from 'next/dist/client/router' +import type { ReactElement } from 'react' + +import { useRecoveryTxState } from '@/hooks/useRecoveryTxState' +import { Countdown } from '@/components/common/Countdown' +import RecoveryPending from '@/public/images/common/recovery-pending.svg' +import ExternalLink from '@/components/common/ExternalLink' +import { AppRoutes } from '@/config/routes' +import type { RecoveryQueueItem } from '@/store/recoverySlice' + +import css from './styles.module.css' + +type Props = + | { + orientation?: 'vertical' + onClose: () => void + recovery: RecoveryQueueItem + } + | { + orientation: 'horizontal' + onClose?: never + recovery: RecoveryQueueItem + } + +export function RecoveryInProgressCard({ orientation = 'vertical', onClose, recovery }: Props): ReactElement { + const { isExecutable, remainingSeconds } = useRecoveryTxState(recovery) + const router = useRouter() + + const onClick = async () => { + await router.push({ + pathname: AppRoutes.home, + query: router.query, + }) + onClose?.() + } + + const icon = + const title = isExecutable ? 'Account recovery possible' : 'Account recovery in progress' + const desc = isExecutable + ? 'The recovery process is possible. This Account can be recovered.' + : 'The recovery process has started. This Account will be ready to recover in:' + + const link = ( + + Learn more + + ) + + if (orientation === 'horizontal') { + return ( + + + {icon} + + + + {title} + + + + {desc} + + + + + + {link} + + + ) + } + + return ( + + + + {icon} + + {link} + + + + + {title} + + + {desc} + + + + + + + + + + ) +} diff --git a/src/components/recovery/RecoveryCards/RecoveryProposalCard.tsx b/src/components/recovery/RecoveryCards/RecoveryProposalCard.tsx new file mode 100644 index 0000000000..9ac434f099 --- /dev/null +++ b/src/components/recovery/RecoveryCards/RecoveryProposalCard.tsx @@ -0,0 +1,118 @@ +import { Button, Card, Divider, Grid, Typography } from '@mui/material' +import { useContext } from 'react' +import type { ReactElement } from 'react' + +import ProposeRecovery from '@/public/images/common/propose-recovery.svg' +import ExternalLink from '@/components/common/ExternalLink' +import { RecoverAccountFlow } from '@/components/tx-flow/flows/RecoverAccount' +import useSafeInfo from '@/hooks/useSafeInfo' +import madProps from '@/utils/mad-props' +import { TxModalContext } from '@/components/tx-flow' +import type { TxModalContextType } from '@/components/tx-flow' +import type { SafeInfo } from '@safe-global/safe-gateway-typescript-sdk' + +import css from './styles.module.css' + +type Props = + | { + orientation?: 'vertical' + onClose: () => void + safe: SafeInfo + setTxFlow: TxModalContextType['setTxFlow'] + } + | { + orientation: 'horizontal' + onClose?: never + safe: SafeInfo + setTxFlow: TxModalContextType['setTxFlow'] + } + +export function _RecoveryProposalCard({ orientation = 'vertical', onClose, safe, setTxFlow }: Props): ReactElement { + const onRecover = async () => { + onClose?.() + setTxFlow() + } + + const icon = + const title = 'Recover this Account' + const desc = `The connect wallet was chosen as a trusted guardian. You can help the owner${ + safe.owners.length > 1 ? 's' : '' + } regain access by updating the owner list.` + + const link = ( + + Learn more + + ) + + const recoveryButton = ( + + ) + + if (orientation === 'horizontal') { + return ( + + + {icon} + + + + {title} + + + + {desc} + + + {link} + + + {recoveryButton} + + + ) + } + + return ( + + + + {icon} + + {link} + + + + + {title} + + + + {desc} + + + + + + + + {recoveryButton} + + + + ) +} + +// Appease TypeScript +const _useSafe = () => useSafeInfo().safe +const _useSetTxFlow = () => useContext(TxModalContext).setTxFlow + +export const RecoveryProposalCard = madProps(_RecoveryProposalCard, { + safe: _useSafe, + setTxFlow: _useSetTxFlow, +}) diff --git a/src/components/recovery/RecoveryCards/__tests__/RecoveryInProgressCard.test.tsx b/src/components/recovery/RecoveryCards/__tests__/RecoveryInProgressCard.test.tsx new file mode 100644 index 0000000000..91b0d95cbe --- /dev/null +++ b/src/components/recovery/RecoveryCards/__tests__/RecoveryInProgressCard.test.tsx @@ -0,0 +1,131 @@ +import { BigNumber } from 'ethers' +import { fireEvent, waitFor } from '@testing-library/react' + +import { render } from '@/tests/test-utils' +import { RecoveryInProgressCard } from '../RecoveryInProgressCard' +import { useRecoveryTxState } from '@/hooks/useRecoveryTxState' +import type { RecoveryQueueItem } from '@/store/recoverySlice' + +jest.mock('@/hooks/useRecoveryTxState') + +const mockUseRecoveryTxState = useRecoveryTxState as jest.MockedFunction + +describe('RecoveryInProgressCard', () => { + beforeEach(() => { + jest.clearAllMocks() + }) + + describe('vertical', () => { + it('should render executable recovery state correctly', async () => { + mockUseRecoveryTxState.mockReturnValue({ + isExecutable: true, + remainingSeconds: 0, + } as any) + + const mockClose = jest.fn() + + const { queryByText } = render( + , + ) + + ;['days', 'hrs', 'mins'].forEach((unit) => { + expect(queryByText(unit)).toBeFalsy() + }) + + expect(queryByText('Account recovery possible')).toBeTruthy() + expect(queryByText('Learn more')).toBeTruthy() + + const dashboardButton = queryByText('Go to dashboard') + expect(dashboardButton).toBeTruthy() + + fireEvent.click(dashboardButton!) + + await waitFor(() => { + expect(mockClose).toHaveBeenCalled() + }) + }) + + it('should render non-executable recovery state correctly', async () => { + mockUseRecoveryTxState.mockReturnValue({ + isExecutable: false, + remainingSeconds: 420 * 69 * 1337, + } as any) + + const mockClose = jest.fn() + + const { queryByText } = render( + , + ) + + expect(queryByText('Account recovery in progress')).toBeTruthy() + expect(queryByText('The recovery process has started. This Account will be ready to recover in:')).toBeTruthy() + ;['days', 'hrs', 'mins'].forEach((unit) => { + expect(queryByText(unit)).toBeTruthy() + }) + expect(queryByText('Learn more')).toBeTruthy() + + const dashboardButton = queryByText('Go to dashboard') + expect(dashboardButton).toBeTruthy() + + fireEvent.click(dashboardButton!) + + await waitFor(() => { + expect(mockClose).toHaveBeenCalled() + }) + }) + }) + describe('horizontal', () => { + it('should render executable recovery state correctly', () => { + mockUseRecoveryTxState.mockReturnValue({ + isExecutable: true, + remainingSeconds: 0, + } as any) + + const { queryByText } = render( + , + ) + + ;['days', 'hrs', 'mins'].forEach((unit) => { + expect(queryByText(unit)).toBeFalsy() + }) + expect(queryByText('Go to dashboard')).toBeFalsy() + + expect(queryByText('Account recovery possible')).toBeTruthy() + expect(queryByText('Learn more')).toBeTruthy() + }) + + it('should render non-executable recovery state correctly', () => { + mockUseRecoveryTxState.mockReturnValue({ + isExecutable: false, + remainingSeconds: 420 * 69 * 1337, + } as any) + + const { queryByText } = render( + , + ) + + expect(queryByText('Go to dashboard')).toBeFalsy() + + expect(queryByText('Account recovery in progress')).toBeTruthy() + expect(queryByText('The recovery process has started. This Account will be ready to recover in:')).toBeTruthy() + ;['days', 'hrs', 'mins'].forEach((unit) => { + expect(queryByText(unit)).toBeTruthy() + }) + expect(queryByText('Learn more')).toBeTruthy() + }) + }) +}) diff --git a/src/components/recovery/RecoveryCards/__tests__/RecoveryProposalCard.test.tsx b/src/components/recovery/RecoveryCards/__tests__/RecoveryProposalCard.test.tsx new file mode 100644 index 0000000000..8ca5df4bed --- /dev/null +++ b/src/components/recovery/RecoveryCards/__tests__/RecoveryProposalCard.test.tsx @@ -0,0 +1,66 @@ +import { faker } from '@faker-js/faker' +import type { SafeInfo } from '@safe-global/safe-gateway-typescript-sdk' + +import { fireEvent, render } from '@/tests/test-utils' +import { _RecoveryProposalCard } from '../RecoveryProposalCard' + +describe('RecoveryProposalCard', () => { + describe('vertical', () => { + it('should render correctly', () => { + const mockClose = jest.fn() + const mockSetTxFlow = jest.fn() + + const { queryByText } = render( + <_RecoveryProposalCard + orientation="vertical" + onClose={mockClose} + safe={{ owners: [{ value: faker.finance.ethereumAddress() }] } as SafeInfo} + setTxFlow={mockSetTxFlow} + />, + ) + + expect(queryByText('Recover this Account')).toBeTruthy() + expect( + queryByText( + 'The connect wallet was chosen as a trusted guardian. You can help the owner regain access by updating the owner list.', + ), + ).toBeTruthy() + expect(queryByText('Learn more')).toBeTruthy() + + const recoveryButton = queryByText('Start recovery') + expect(recoveryButton).toBeTruthy() + + fireEvent.click(recoveryButton!) + + expect(mockClose).toHaveBeenCalled() + expect(mockSetTxFlow).toHaveBeenCalled() + }) + }) + describe('horizontal', () => {}) + it('should render correctly', () => { + const mockSetTxFlow = jest.fn() + + const { queryByText } = render( + <_RecoveryProposalCard + orientation="horizontal" + safe={{ owners: [{ value: faker.finance.ethereumAddress() }] } as SafeInfo} + setTxFlow={mockSetTxFlow} + />, + ) + + expect(queryByText('Recover this Account')).toBeTruthy() + expect( + queryByText( + 'The connect wallet was chosen as a trusted guardian. You can help the owner regain access by updating the owner list.', + ), + ).toBeTruthy() + expect(queryByText('Learn more')).toBeTruthy() + + const recoveryButton = queryByText('Start recovery') + expect(recoveryButton).toBeTruthy() + + fireEvent.click(recoveryButton!) + + expect(mockSetTxFlow).toHaveBeenCalled() + }) +}) diff --git a/src/components/recovery/RecoveryCards/styles.module.css b/src/components/recovery/RecoveryCards/styles.module.css new file mode 100644 index 0000000000..6aa366375a --- /dev/null +++ b/src/components/recovery/RecoveryCards/styles.module.css @@ -0,0 +1,8 @@ +.card { + max-width: 576px; + padding: var(--space-4); + position: absolute; + top: 50%; + left: 50%; + transform: translate(-50%, -50%); +} diff --git a/src/components/recovery/RecoveryModal/index.test.tsx b/src/components/recovery/RecoveryModal/index.test.tsx new file mode 100644 index 0000000000..5e8c2a0e95 --- /dev/null +++ b/src/components/recovery/RecoveryModal/index.test.tsx @@ -0,0 +1,252 @@ +import { BigNumber } from 'ethers' +import { faker } from '@faker-js/faker' +import { renderHook } from '@testing-library/react' +import * as router from 'next/router' + +import { render, waitFor } from '@/tests/test-utils' +import { safeInfoBuilder } from '@/tests/builders/safe' +import { connectedWalletBuilder } from '@/tests/builders/wallet' +import * as safeInfo from '@/hooks/useSafeInfo' +import { _useDidDismissProposal } from './index' +import type { RecoveryQueueItem } from '@/store/recoverySlice' + +describe('RecoveryModal', () => { + describe('component', () => { + // eslint-disable-next-line @typescript-eslint/consistent-type-imports + let _RecoveryModal: typeof import('./index')._RecoveryModal + + beforeEach(() => { + localStorage.clear() + + // Clear cache in between tests + _RecoveryModal = require('./index')._RecoveryModal + }) + + it('should not render the modal if there is a queue but the user is not an owner or guardian', () => { + const wallet = connectedWalletBuilder().build() + const queue = [{ validFrom: BigNumber.from(0) } as RecoveryQueueItem] + + const { queryByText } = render( + <_RecoveryModal wallet={wallet} isOwner={false} isGuardian={false} queue={queue}> + Test + , + ) + + expect(queryByText('Test')).toBeTruthy() + expect(queryByText('recovery')).toBeFalsy() + }) + + it('should not render the modal if there is no queue and the user is a guardian', () => { + const wallet = connectedWalletBuilder().build() + const queue = [] as Array + + const { queryByText } = render( + <_RecoveryModal wallet={wallet} isOwner={false} isGuardian queue={queue}> + Test + , + ) + + expect(queryByText('Test')).toBeTruthy() + expect(queryByText('recovery')).toBeFalsy() + }) + + it('should not render the modal if there is no queue and the user is an owner', () => { + const wallet = connectedWalletBuilder().build() + const queue = [] as Array + + const { queryByText } = render( + <_RecoveryModal wallet={wallet} isOwner isGuardian={false} queue={queue}> + Test + , + ) + + expect(queryByText('Test')).toBeTruthy() + expect(queryByText('recovery')).toBeFalsy() + }) + + it('should render the in-progress modal when there is a queue for guardians', () => { + const wallet = connectedWalletBuilder().build() + const queue = [{ validFrom: BigNumber.from(0) } as RecoveryQueueItem] + + const { queryByText } = render( + <_RecoveryModal wallet={wallet} isOwner={false} isGuardian queue={queue}> + Test + , + ) + + expect(queryByText('Test')).toBeTruthy() + expect(queryByText('Account recovery in progress')).toBeTruthy() + }) + + it('should render the in-progress modal when there is a queue for owners', () => { + const wallet = connectedWalletBuilder().build() + const queue = [{ validFrom: BigNumber.from(0) } as RecoveryQueueItem] + + const { queryByText } = render( + <_RecoveryModal wallet={wallet} isOwner isGuardian={false} queue={queue}> + Test + , + ) + + expect(queryByText('Test')).toBeTruthy() + expect(queryByText('Account recovery in progress')).toBeTruthy() + }) + + it('should render the proposal modal when there is no queue for guardians', () => { + const wallet = connectedWalletBuilder().build() + const queue = [] as Array + + const { queryByText } = render( + <_RecoveryModal wallet={wallet} isOwner={false} isGuardian queue={queue}> + Test + , + ) + + expect(queryByText('Test')).toBeTruthy() + expect(queryByText('Recover this Account')).toBeTruthy() + }) + + it('should not render the proposal modal when there is no queue for owners', () => { + const wallet = connectedWalletBuilder().build() + const queue = [] as Array + + const { queryByText } = render( + <_RecoveryModal wallet={wallet} isOwner isGuardian={false} queue={queue}> + Test + , + ) + + expect(queryByText('Test')).toBeTruthy() + expect(queryByText('Recover this Account')).toBeFalsy() + }) + + it('should close the modal when the user navigates away', async () => { + const mockUseRouter = { + push: jest.fn(), + query: {}, + events: { + on: jest.fn(), + off: jest.fn(), + }, + } + + jest.spyOn(router, 'useRouter').mockReturnValue(mockUseRouter as any) + + const wallet = connectedWalletBuilder().build() + const queue = [{ validFrom: BigNumber.from(0), transactionHash: faker.string.hexadecimal() } as RecoveryQueueItem] + + const { queryByText } = render( + <_RecoveryModal wallet={wallet} isOwner isGuardian={false} queue={queue}> + Test + , + ) + + expect(queryByText('Test')).toBeTruthy() + expect(queryByText('Account recovery in progress')).toBeTruthy() + + // Trigger the route change + mockUseRouter.events.on.mock.calls[0][1]() + + await waitFor(() => { + expect(queryByText('Account recovery in progress')).toBeFalsy() + }) + }) + }) + + describe('hooks', () => { + beforeEach(() => { + localStorage.clear() + + const safe = safeInfoBuilder().build() + jest + .spyOn(safeInfo, 'default') + .mockReturnValue({ safe, safeAddress: safe.address.value } as ReturnType) + }) + + describe('useDidDismissProposal', () => { + it('should return false if the proposal was not dismissed before', () => { + const guardianAddress = faker.finance.ethereumAddress() + + const { result } = renderHook(() => _useDidDismissProposal()) + + expect(result.current.wasProposalDismissed(guardianAddress)).toBeFalsy() + }) + + it('should return true if the proposal was dismissed before', () => { + const guardianAddress = faker.finance.ethereumAddress() + + const { result, rerender } = renderHook(() => _useDidDismissProposal()) + + expect(result.current.wasProposalDismissed(guardianAddress)).toBeFalsy() + result.current.dismissProposal(guardianAddress) + + rerender() + + expect(result.current.wasProposalDismissed(guardianAddress)).toBeTruthy() + }) + + it('should persist dismissals between sessions', () => { + const guardianAddress = faker.finance.ethereumAddress() + + const firstRender = renderHook(() => _useDidDismissProposal()) + + expect(firstRender.result.current.wasProposalDismissed(guardianAddress)).toBeFalsy() + firstRender.result.current.dismissProposal(guardianAddress) + + firstRender.rerender() + + expect(firstRender.result.current.wasProposalDismissed(guardianAddress)).toBeTruthy() + + firstRender.unmount() + + const secondRender = renderHook(() => _useDidDismissProposal()) + expect(secondRender.result.current.wasProposalDismissed(guardianAddress)).toBeTruthy() + }) + }) + + describe('useDidDismissInProgress', () => { + // eslint-disable-next-line @typescript-eslint/consistent-type-imports + let _useDidDismissInProgress: typeof import('./index')._useDidDismissInProgress + + beforeEach(() => { + localStorage.clear() + + // Clear cache in between tests + _useDidDismissInProgress = require('./index')._useDidDismissInProgress + }) + + it('should return false if in-progress was not dismissed before', () => { + const guardianAddress = faker.finance.ethereumAddress() + + const { result } = renderHook(() => _useDidDismissInProgress()) + + expect(result.current.wasInProgressDismissed(guardianAddress)).toBeFalsy() + }) + + it('should return true if in-progress was not dismissed before', () => { + const guardianAddress = faker.finance.ethereumAddress() + + const { result } = renderHook(() => _useDidDismissInProgress()) + + expect(result.current.wasInProgressDismissed(guardianAddress)).toBeFalsy() + result.current.dismissInProgress(guardianAddress) + expect(result.current.wasInProgressDismissed(guardianAddress)).toBeTruthy() + }) + + it('should not persist dismissals between sessions', () => { + const guardianAddress = faker.finance.ethereumAddress() + + const firstRender = renderHook(() => _useDidDismissInProgress()) + + expect(firstRender.result.current.wasInProgressDismissed(guardianAddress)).toBeFalsy() + firstRender.result.current.dismissInProgress(guardianAddress) + expect(firstRender.result.current.wasInProgressDismissed(guardianAddress)).toBeTruthy() + + firstRender.unmount() + + const secondRender = renderHook(() => _useDidDismissInProgress()) + expect(secondRender.result.current.wasInProgressDismissed(guardianAddress)).toBeFalsy() + }) + }) + }) +}) diff --git a/src/components/recovery/RecoveryModal/index.tsx b/src/components/recovery/RecoveryModal/index.tsx new file mode 100644 index 0000000000..703865d947 --- /dev/null +++ b/src/components/recovery/RecoveryModal/index.tsx @@ -0,0 +1,178 @@ +import { Backdrop, Fade } from '@mui/material' +import { useCallback, useEffect, useRef, useState } from 'react' +import { useRouter } from 'next/router' +import type { ReactElement, ReactNode } from 'react' + +import { useRecoveryQueue } from '@/hooks/useRecoveryQueue' +import { RecoveryInProgressCard } from '../RecoveryCards/RecoveryInProgressCard' +import { RecoveryProposalCard } from '../RecoveryCards/RecoveryProposalCard' +import useIsSafeOwner from '@/hooks/useIsSafeOwner' +import { useIsGuardian } from '@/hooks/useIsGuardian' +import madProps from '@/utils/mad-props' +import useLocalStorage from '@/services/local-storage/useLocalStorage' +import useWallet from '@/hooks/wallets/useWallet' +import useSafeInfo from '@/hooks/useSafeInfo' +import { sameAddress } from '@/utils/addresses' +import type { RecoveryQueueItem } from '@/store/recoverySlice' + +export function _RecoveryModal({ + children, + isOwner, + isGuardian, + queue, + wallet, +}: { + children: ReactNode + isOwner: boolean + isGuardian: boolean + queue: Array + wallet: ReturnType +}): ReactElement { + const { wasProposalDismissed, dismissProposal } = _useDidDismissProposal() + const { wasInProgressDismissed, dismissInProgress } = _useDidDismissInProgress() + + const [modal, setModal] = useState(null) + const router = useRouter() + + const next = queue[0] + + // Close modal + const onClose = () => { + setModal(null) + } + + // Trigger modal + useEffect(() => { + setModal(() => { + if (next && !wasInProgressDismissed(next.transactionHash)) { + const onCloseWithDismiss = () => { + dismissInProgress(next.transactionHash) + onClose() + } + + return + } + + if (wallet?.address && !isOwner && !wasProposalDismissed(wallet.address)) { + const onCloseWithDismiss = () => { + dismissProposal(wallet.address) + onClose() + } + + return + } + + return null + }) + }, [ + dismissInProgress, + dismissProposal, + isGuardian, + isOwner, + next, + queue.length, + wallet, + wasInProgressDismissed, + wasProposalDismissed, + ]) + + // Close modal on navigation + useEffect(() => { + router.events.on('routeChangeComplete', onClose) + return () => { + router.events.off('routeChangeComplete', onClose) + } + }, [router]) + + return ( + <> + + palette.background.main }}> + {modal} + + + {children} + + ) +} + +export const RecoveryModal = madProps(_RecoveryModal, { + isOwner: useIsSafeOwner, + isGuardian: useIsGuardian, + queue: useRecoveryQueue, + wallet: useWallet, +}) + +export function _useDidDismissProposal() { + const LS_KEY = 'dismissedRecoveryProposals' + + type Guardian = string + type DismissedProposalCache = { [chainId: string]: { [safeAddress: string]: Guardian } } + + const { safe, safeAddress } = useSafeInfo() + const chainId = safe.chainId + + const [dismissedProposals, setDismissedProposals] = useLocalStorage(LS_KEY) + + // Cache dismissal of proposal modal + const dismissProposal = useCallback( + (guardianAddress: string) => { + const dismissed = dismissedProposals?.[chainId] ?? {} + + setDismissedProposals({ + ...(dismissedProposals ?? {}), + [chainId]: { + ...dismissed, + [safeAddress]: guardianAddress, + }, + }) + }, + [dismissedProposals, chainId, safeAddress, setDismissedProposals], + ) + + const wasProposalDismissed = useCallback( + (guardianAddress: string) => { + // If no proposals, is guardian and didn't ever dismiss + return sameAddress(dismissedProposals?.[chainId]?.[safeAddress], guardianAddress) + }, + [chainId, dismissedProposals, safeAddress], + ) + + return { wasProposalDismissed, dismissProposal } +} + +export function _useDidDismissInProgress() { + type TxHash = string + type DismissedInProgressCache = { [chainId: string]: { [safeAddress: string]: TxHash } } + + const { safe, safeAddress } = useSafeInfo() + const chainId = safe.chainId + + const dismissedInProgress = useRef({}) + + // Cache dismissal of in-progress modal + const dismissInProgress = useCallback( + (txHash: string) => { + const dismissed = dismissedInProgress.current?.[chainId] ?? {} + + dismissedInProgress.current = { + ...dismissedInProgress.current, + [chainId]: { + ...dismissed, + [safeAddress]: txHash, + }, + } + }, + [chainId, safeAddress], + ) + + const wasInProgressDismissed = useCallback( + (txHash: string) => { + // If proposal and did not notify during current session of Safe + return sameAddress(txHash, dismissedInProgress.current?.[chainId]?.[safeAddress]) + }, + [chainId, safeAddress], + ) + + return { wasInProgressDismissed, dismissInProgress } +} diff --git a/src/components/tx-flow/flows/RecoverAccount/RecoverAccountFlowSetup.tsx b/src/components/tx-flow/flows/RecoverAccount/RecoverAccountFlowSetup.tsx index a11d7ba3d7..b51ab5f7a8 100644 --- a/src/components/tx-flow/flows/RecoverAccount/RecoverAccountFlowSetup.tsx +++ b/src/components/tx-flow/flows/RecoverAccount/RecoverAccountFlowSetup.tsx @@ -11,6 +11,7 @@ import { Tooltip, } from '@mui/material' import { useForm, FormProvider, useFieldArray, Controller } from 'react-hook-form' +import { Fragment } from 'react' import type { ReactElement } from 'react' import TxCard from '../../common/TxCard' @@ -64,7 +65,7 @@ export function RecoverAccountFlowSetup({ {fields.map((field, index) => ( - <> + )} - + ))} diff --git a/src/components/tx-flow/flows/UpsertRecovery/UpsertRecoveryFlowReview.tsx b/src/components/tx-flow/flows/UpsertRecovery/UpsertRecoveryFlowReview.tsx index c2993f477e..5832e4cc70 100644 --- a/src/components/tx-flow/flows/UpsertRecovery/UpsertRecoveryFlowReview.tsx +++ b/src/components/tx-flow/flows/UpsertRecovery/UpsertRecoveryFlowReview.tsx @@ -63,7 +63,7 @@ export function UpsertRecoveryFlowReview({ This transaction will {moduleAddress ? 'update' : 'enable'} the Account recovery feature once executed. - + diff --git a/src/hooks/useIsGuardian.ts b/src/hooks/useIsGuardian.ts new file mode 100644 index 0000000000..eb2ae9eb4e --- /dev/null +++ b/src/hooks/useIsGuardian.ts @@ -0,0 +1,8 @@ +import { useAppSelector } from '@/store' +import { selectDelayModifierByGuardian } from '@/store/recoverySlice' +import useWallet from './wallets/useWallet' + +export function useIsGuardian() { + const wallet = useWallet() + return !!useAppSelector((state) => selectDelayModifierByGuardian(state, wallet?.address ?? '')) +} diff --git a/src/hooks/useRecoveryQueue.ts b/src/hooks/useRecoveryQueue.ts new file mode 100644 index 0000000000..69bf04458c --- /dev/null +++ b/src/hooks/useRecoveryQueue.ts @@ -0,0 +1,13 @@ +import { useAppSelector } from '@/store' +import { selectRecoveryQueues } from '@/store/recoverySlice' +import { useClock } from './useClock' +import type { RecoveryQueueItem } from '@/store/recoverySlice' + +export function useRecoveryQueue(): Array { + const queue = useAppSelector(selectRecoveryQueues) + const clock = useClock() + + return queue.filter(({ expiresAt }) => { + return expiresAt ? expiresAt.gt(clock) : true + }) +} diff --git a/src/hooks/useRecoveryTxState.ts b/src/hooks/useRecoveryTxState.ts index 97b6f103eb..4c39fe56a3 100644 --- a/src/hooks/useRecoveryTxState.ts +++ b/src/hooks/useRecoveryTxState.ts @@ -3,7 +3,6 @@ import { useAppSelector } from '@/store' import { selectDelayModifierByTxHash } from '@/store/recoverySlice' import type { RecoveryQueueItem } from '@/store/recoverySlice' -// TODO: Test export function useRecoveryTxState({ validFrom, expiresAt, transactionHash, args }: RecoveryQueueItem): { isNext: boolean isExecutable: boolean