From be181de610d15f18efa977e183d449ae3f871c54 Mon Sep 17 00:00:00 2001 From: Luiz Gomes <8636507+LuizAsFight@users.noreply.github.com> Date: Thu, 12 Dec 2024 02:16:29 -0300 Subject: [PATCH] fix: wallet reset issue (#1700) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit - Closes FE-1106 Implement mechanism for backup accounts and recover in case of disaster in db (all data lost) - synchronize db with chrome storage - implemented recoverWallet method - in case of disaster recover it - simplified accountMachine - create new database and table (outside of react) to identify if only dexiedb database is breaking - report to sentry when there's a recover happening - remove `shouldRecoverWelcomeFromError` logic --------- Co-authored-by: Hélcio Franco --- .changeset/grumpy-gorillas-drive.md | 5 + .changeset/kind-pumas-tie.md | 5 + .changeset/ninety-pianos-bake.md | 5 + .changeset/strong-tips-pretend.md | 5 + .github/workflows/pr-tests-e2e-crx-lock.yml | 7 - .github/workflows/pr-tests-e2e.yml | 7 - packages/app/jest.setup.ts | 29 +++ packages/app/load.envs.cts | 1 - packages/app/playwright.config.ts | 3 - packages/app/playwright.crx-lock.config.ts | 7 +- packages/app/playwright/crx/crx.test.ts | 3 +- packages/app/playwright/crx/lock.test.ts | 36 ++-- .../BalanceAssets/BalanceAssets.tsx | 5 +- .../components/FuelAddress/FuelAddress.tsx | 1 + packages/app/src/systems/Account/events.tsx | 7 +- .../src/systems/Account/hooks/useAccounts.tsx | 4 +- .../Account/machines/accountsMachine.tsx | 187 +++++++++++------- .../Account/machines/addAccountMachine.tsx | 2 +- .../Account/machines/editAccountMachine.tsx | 2 +- .../Account/machines/importAccountMachine.tsx | 2 +- .../src/systems/Account/services/account.ts | 122 +++++++++++- .../Account/utils/getTestNoDexieDbData.ts | 23 +++ .../app/src/systems/Account/utils/storage.tsx | 8 + .../systems/Asset/machines/assetsMachine.tsx | 2 +- .../CRX/background/actions/onInstall.ts | 11 +- .../CRX/background/services/DatabaseEvents.ts | 29 +++ .../CRX/background/services/VaultService.ts | 3 +- .../CRX/scripts/executeContentScript.ts | 5 +- .../systems/Core/services/chromeStorage.ts | 87 ++++++++ .../app/src/systems/Core/services/core.ts | 2 + .../app/src/systems/Core/services/index.ts | 1 + .../app/src/systems/Core/utils/database.ts | 66 +++++++ .../systems/Core/utils/databaseVersioning.ts | 3 + .../machines/selectNetworkRequestMachine.tsx | 2 +- .../app/src/systems/Home/pages/Home/Home.tsx | 2 +- .../Network/machines/networksMachine.ts | 2 +- .../systems/SignUp/machines/signUpMachine.ts | 2 +- .../src/systems/Vault/services/VaultServer.ts | 4 - 38 files changed, 544 insertions(+), 153 deletions(-) create mode 100644 .changeset/grumpy-gorillas-drive.md create mode 100644 .changeset/kind-pumas-tie.md create mode 100644 .changeset/ninety-pianos-bake.md create mode 100644 .changeset/strong-tips-pretend.md create mode 100644 packages/app/src/systems/Account/utils/getTestNoDexieDbData.ts create mode 100644 packages/app/src/systems/Core/services/chromeStorage.ts diff --git a/.changeset/grumpy-gorillas-drive.md b/.changeset/grumpy-gorillas-drive.md new file mode 100644 index 0000000000..0b44275e19 --- /dev/null +++ b/.changeset/grumpy-gorillas-drive.md @@ -0,0 +1,5 @@ +--- +"fuels-wallet": minor +--- + +feat: implemented recoverWallet method and in case of disaster recover it diff --git a/.changeset/kind-pumas-tie.md b/.changeset/kind-pumas-tie.md new file mode 100644 index 0000000000..0bd7bf92f6 --- /dev/null +++ b/.changeset/kind-pumas-tie.md @@ -0,0 +1,5 @@ +--- +"fuels-wallet": minor +--- + +feat: synchronize db with chrome storage diff --git a/.changeset/ninety-pianos-bake.md b/.changeset/ninety-pianos-bake.md new file mode 100644 index 0000000000..e5276c8886 --- /dev/null +++ b/.changeset/ninety-pianos-bake.md @@ -0,0 +1,5 @@ +--- +"fuels-wallet": minor +--- + +feat: create new database and table (outside of react) to identify if only dexiedb database is breaking diff --git a/.changeset/strong-tips-pretend.md b/.changeset/strong-tips-pretend.md new file mode 100644 index 0000000000..9dfb501ada --- /dev/null +++ b/.changeset/strong-tips-pretend.md @@ -0,0 +1,5 @@ +--- +"fuels-wallet": minor +--- + +chore: report to sentry when there's a recover happening diff --git a/.github/workflows/pr-tests-e2e-crx-lock.yml b/.github/workflows/pr-tests-e2e-crx-lock.yml index 5094bbe2e8..9db52bcda2 100644 --- a/.github/workflows/pr-tests-e2e-crx-lock.yml +++ b/.github/workflows/pr-tests-e2e-crx-lock.yml @@ -31,13 +31,6 @@ jobs: - name: Generate .env run: cp packages/app/.env.example packages/app/.env - - name: Build Application - run: pnpm build:app - env: - ## increase node.js m memory limit for building - ## with sourcemaps - NODE_OPTIONS: "--max-old-space-size=4096" - - uses: ./.github/actions/setup-playwright - name: Run E2E Tests diff --git a/.github/workflows/pr-tests-e2e.yml b/.github/workflows/pr-tests-e2e.yml index b1a64c22ec..0a601c2056 100644 --- a/.github/workflows/pr-tests-e2e.yml +++ b/.github/workflows/pr-tests-e2e.yml @@ -31,13 +31,6 @@ jobs: - name: Generate .env run: cp packages/app/.env.example packages/app/.env - - name: Build Application - run: pnpm build:app - env: - ## increase node.js m memory limit for building - ## with sourcemaps - NODE_OPTIONS: "--max-old-space-size=4096" - - uses: ./.github/actions/setup-playwright - name: Run E2E Tests diff --git a/packages/app/jest.setup.ts b/packages/app/jest.setup.ts index 7de8e6fc52..947361ed3f 100644 --- a/packages/app/jest.setup.ts +++ b/packages/app/jest.setup.ts @@ -30,6 +30,35 @@ jest.mock('react-dom/test-utils', () => { }; }); +// Replace chromeStorage +jest.mock('./src/systems/Core/services/chromeStorage', () => { + return { + chromeStorage: { + accounts: { + get: jest.fn(), + getAll: jest.fn(), + set: jest.fn(), + remove: jest.fn(), + clear: jest.fn(), + }, + networks: { + get: jest.fn(), + getAll: jest.fn(), + set: jest.fn(), + remove: jest.fn(), + clear: jest.fn(), + }, + vaults: { + get: jest.fn(), + getAll: jest.fn(), + set: jest.fn(), + remove: jest.fn(), + clear: jest.fn(), + }, + }, + }; +}); + console.warn = jest.fn(); const noop = () => {}; diff --git a/packages/app/load.envs.cts b/packages/app/load.envs.cts index 8a54d6e412..08d15bd80e 100644 --- a/packages/app/load.envs.cts +++ b/packages/app/load.envs.cts @@ -1,5 +1,4 @@ import { readFileSync } from 'node:fs'; -import path from 'node:path'; import { resolve } from 'node:path'; import { config } from 'dotenv'; diff --git a/packages/app/playwright.config.ts b/packages/app/playwright.config.ts index 977d4b4518..24932d420e 100644 --- a/packages/app/playwright.config.ts +++ b/packages/app/playwright.config.ts @@ -1,11 +1,8 @@ -import { join } from 'node:path'; -// biome-ignore lint/style/useNodejsImportProtocol: import { type PlaywrightTestConfig, defineConfig, devices, } from '@playwright/test'; -import './load.envs'; import './load.envs.cts'; const PORT = process.env.PORT; diff --git a/packages/app/playwright.crx-lock.config.ts b/packages/app/playwright.crx-lock.config.ts index 6e9f5c7109..e7eb453675 100644 --- a/packages/app/playwright.crx-lock.config.ts +++ b/packages/app/playwright.crx-lock.config.ts @@ -1,13 +1,10 @@ -// biome-ignore lint/style/useNodejsImportProtocol: -import { join } from 'path'; import { defineConfig, devices } from '@playwright/test'; import { playwrightConfig } from './playwright.config'; - -const __dirname = new URL('.', import.meta.url).pathname; +import './load.envs.cts'; export default defineConfig({ ...playwrightConfig, - testMatch: join(__dirname, './playwright/crx/lock.test.ts'), + testMatch: 'playwright/crx/lock.test.ts', testIgnore: undefined, projects: [ { diff --git a/packages/app/playwright/crx/crx.test.ts b/packages/app/playwright/crx/crx.test.ts index f69eafb238..e6a1735ccb 100644 --- a/packages/app/playwright/crx/crx.test.ts +++ b/packages/app/playwright/crx/crx.test.ts @@ -1,5 +1,5 @@ import type { NetworkData, Account as WalletAccount } from '@fuel-wallet/types'; -import { type Locator, type Page, expect } from '@playwright/test'; +import { type Locator, expect } from '@playwright/test'; import { delay, @@ -10,7 +10,6 @@ import { hasText, reload, seedWallet, - visit, waitAriaLabel, } from '../commons'; import { diff --git a/packages/app/playwright/crx/lock.test.ts b/packages/app/playwright/crx/lock.test.ts index fe442764c5..ba9c2d4e90 100644 --- a/packages/app/playwright/crx/lock.test.ts +++ b/packages/app/playwright/crx/lock.test.ts @@ -18,6 +18,18 @@ import { test } from './utils'; test.setTimeout(360_000); test.describe('Lock FuelWallet after inactivity', () => { + test('If user opens popup it should force open a sign-up page', async ({ + context, + extensionId, + }) => { + const popupPage = await context.newPage(); + await popupPage.goto(`chrome-extension://${extensionId}/popup.html`); + const page = await context.waitForEvent('page', { + predicate: (page) => page.url().includes('sign-up'), + }); + expect(page.url()).toContain('sign-up'); + }); + test('should lock the wallet after 1 minute of inactivity (config in .env file)', async ({ context, baseURL, @@ -50,19 +62,7 @@ test.describe('Lock FuelWallet after inactivity', () => { await test.step('Create wallet', async () => { const pages = context.pages(); - let page = pages.find((page) => page.url().includes('sign-up')); - - if (!page) { - page = await context.waitForEvent('page', { - predicate: (page) => page.url().includes('sign-up'), - timeout: 10000, // Adjust timeout as needed - }); - } - - if (!page) { - throw new Error('Sign-up page did not open'); - } - + const [page] = pages.filter((page) => page.url().includes('sign-up')); await reload(page); await getElementByText(page, /Create new wallet/i).click(); @@ -97,6 +97,7 @@ test.describe('Lock FuelWallet after inactivity', () => { /** Account created */ await hasText(page, /Wallet created successfully/i, 0, 15000); + await page.close(); }); @@ -111,10 +112,17 @@ test.describe('Lock FuelWallet after inactivity', () => { await getByAriaLabel(popupPage, 'Accounts').click(); await popupPage.waitForTimeout(65_000); await hasText(popupPage, /Assets/i); + + const pages = context.pages(); + const walletPages = pages.filter((page) => { + return page.url().includes('chrome-extension://'); + }); + for (const page of walletPages) { + await page.close(); + } }); await test.step('Resume auto-lock timer after closing wallet', async () => { - await popupPage.close(); const page = await context.newPage(); await page.waitForTimeout(65_000); await page.goto(`chrome-extension://${extensionId}/popup.html`); diff --git a/packages/app/src/systems/Account/components/BalanceAssets/BalanceAssets.tsx b/packages/app/src/systems/Account/components/BalanceAssets/BalanceAssets.tsx index 8250942b15..fb54db5634 100644 --- a/packages/app/src/systems/Account/components/BalanceAssets/BalanceAssets.tsx +++ b/packages/app/src/systems/Account/components/BalanceAssets/BalanceAssets.tsx @@ -29,7 +29,7 @@ export const BalanceAssets = ({ [balances] ); - if (isLoading) return ; + if (isLoading || !balances) return ; const isEmpty = !balances || !balances.length; if (isEmpty) return ; const balancesToShow = balances.filter( @@ -40,6 +40,7 @@ export const BalanceAssets = ({ function toggle() { setShowUnknown((s) => !s); } + return ( {balancesToShow.map((balance) => { @@ -49,7 +50,7 @@ export const BalanceAssets = ({ return ( { const account = useMemo(() => { + if (!address) return ''; if (isContract) return Address.fromDynamicInput(address).toB256(); return Address.fromDynamicInput(address).toString(); }, [isContract, address]); diff --git a/packages/app/src/systems/Account/events.tsx b/packages/app/src/systems/Account/events.tsx index 50868a1bdc..528bd7cb04 100644 --- a/packages/app/src/systems/Account/events.tsx +++ b/packages/app/src/systems/Account/events.tsx @@ -4,11 +4,8 @@ import { Services } from '~/store'; export function accountEvents(store: Store) { return { - reloadBalance() { - store.send(Services.accounts, { type: 'RELOAD_BALANCE' }); - }, - updateAccounts() { - store.send(Services.accounts, { type: 'REFRESH_ACCOUNTS' }); + refreshAccounts(input?: { skipLoading?: boolean }) { + store.send(Services.accounts, { type: 'REFRESH_ACCOUNTS', input }); }, setCurrentAccount(account: Account) { store.send(Services.accounts, { diff --git a/packages/app/src/systems/Account/hooks/useAccounts.tsx b/packages/app/src/systems/Account/hooks/useAccounts.tsx index 031c75aafa..e2ee510c00 100644 --- a/packages/app/src/systems/Account/hooks/useAccounts.tsx +++ b/packages/app/src/systems/Account/hooks/useAccounts.tsx @@ -45,9 +45,7 @@ const selectors = { }; const listenerAccountFetcher = () => { - store.send(Services.accounts, { - type: 'REFRESH_ACCOUNT', - }); + store.refreshAccounts({ skipLoading: true }); }; export function useAccounts() { diff --git a/packages/app/src/systems/Account/machines/accountsMachine.tsx b/packages/app/src/systems/Account/machines/accountsMachine.tsx index ed1bd2adc1..975bda9511 100644 --- a/packages/app/src/systems/Account/machines/accountsMachine.tsx +++ b/packages/app/src/systems/Account/machines/accountsMachine.tsx @@ -11,13 +11,17 @@ import type { AccountInputs } from '../services/account'; type MachineContext = { accounts?: Account[]; + needsRecovery?: boolean; account?: AccountWithBalance; error?: unknown; }; type MachineServices = { fetchAccounts: { - data: Account[]; + data: { + accounts: Account[]; + needsRecovery: boolean; + }; }; fetchAccount: { data: AccountWithBalance; @@ -28,9 +32,7 @@ type MachineServices = { }; export type AccountsMachineEvents = - | { type: 'REFRESH_ACCOUNT'; input?: null } - | { type: 'REFRESH_ACCOUNTS'; input?: null } - | { type: 'RELOAD_BALANCE'; input?: null } + | { type: 'REFRESH_ACCOUNTS'; input?: { skipLoading?: boolean } } | { type: 'SET_CURRENT_ACCOUNT'; input: AccountInputs['setCurrentAccount'] } // biome-ignore lint/suspicious/noConfusingVoidType: | { type: 'LOGOUT'; input?: void } @@ -39,26 +41,69 @@ export type AccountsMachineEvents = input: AccountInputs['updateAccount']; }; -const fetchAccount = { - invoke: { - src: 'fetchAccount', - onDone: [ - { - target: 'idle', - actions: ['assignAccount'], - cond: 'hasAccount', - }, - { - target: 'idle', - actions: ['assignAccount'], +const fetchingAccountsState = +{ + initial: 'fetchingAccounts', + states: { + fetchingAccounts: { + invoke: { + src: 'fetchAccounts', + onDone: [ + { + target: 'recoveringWallet', + actions: ['assignAccounts', 'setIsLogged'], + cond: 'hasAccountsOrNeedsRecovery', + }, + { + target: 'fetchingAccount', + actions: ['assignAccounts'], + }, + ], + onError: [ + { + actions: 'assignError', + target: '#(machine).failed', + }, + ], + }, + }, + recoveringWallet: { + invoke: { + src: 'recoverWallet', + onDone: [ + { + actions: 'assignError', + target: '#(machine).failed', + cond: FetchMachine.hasError, + }, + { + target: 'fetchingAccount', + }, + ], }, - ], - onError: [ - { - actions: 'assignError', - target: 'failed', + }, + fetchingAccount: { + invoke: { + src: 'fetchAccount', + onDone: [ + { + cond: FetchMachine.hasError, + actions: 'assignError', + target: '#(machine).failed', + }, + { + target: '#(machine).idle', + actions: ['assignAccount'], + }, + ], + onError: [ + { + actions: 'assignError', + target: '#(machine).failed', + }, + ], }, - ], + }, }, }; @@ -89,44 +134,17 @@ export const accountsMachine = createMachine( * Update accounts every 5 seconds */ TIMEOUT: { - target: 'refreshAccount', + target: 'refreshingAccounts', cond: 'isLoggedIn', }, }, }, fetchingAccounts: { tags: ['loading'], - invoke: { - src: 'fetchAccounts', - onDone: [ - { - target: 'fetchingAccount', - actions: ['assignAccounts', 'setIsLogged'], - cond: 'hasAccounts', - }, - { - target: 'idle', - actions: ['assignAccounts'], - }, - ], - onError: [ - { - actions: 'assignError', - target: 'failed', - }, - ], - }, - }, - fetchingAccount: { - tags: ['loading'], - ...fetchAccount, + ...fetchingAccountsState, }, - refreshAccount: { - ...fetchAccount, - }, - reloadingBalance: { - tags: ['loading'], - ...fetchAccount, + refreshingAccounts: { + ...fetchingAccountsState, }, settingCurrentAccount: { invoke: { @@ -177,16 +195,15 @@ export const accountsMachine = createMachine( LOGOUT: { target: 'loggingout', }, - REFRESH_ACCOUNTS: { - target: 'fetchingAccounts', - }, - REFRESH_ACCOUNT: { - target: 'refreshAccount', - }, - RELOAD_BALANCE: { - target: 'reloadingBalance', - actions: ['notifyUpdateAccounts'], - }, + REFRESH_ACCOUNTS: [ + { + cond: 'shouldSkipLoading', + target: 'refreshingAccounts', + }, + { + target: 'fetchingAccounts', + } + ], }, }, { @@ -196,7 +213,8 @@ export const accountsMachine = createMachine( }, actions: { assignAccounts: assign({ - accounts: (_, ev) => ev.data, + accounts: (_, ev) => ev.data.accounts, + needsRecovery: (_, ev) => ev.data.needsRecovery, }), assignAccount: assign({ account: (_, ev) => ev.data, @@ -212,17 +230,31 @@ export const accountsMachine = createMachine( Storage.setItem(IS_LOGGED_KEY, true); }, notifyUpdateAccounts: () => { - store.updateAccounts(); + store.refreshAccounts(); }, redirectToHome: () => { store.closeOverlay(); - }, + } }, services: { - fetchAccounts: FetchMachine.create({ + fetchAccounts: FetchMachine.create< + never, + { accounts: Account[]; needsRecovery: boolean } + >({ showError: true, async fetch() { - return AccountService.getAccounts(); + const accounts = await AccountService.getAccounts(); + const { needsRecovery } = await AccountService.fetchRecoveryState(); + + return { + accounts, + needsRecovery, + }; + }, + }), + recoverWallet: FetchMachine.create({ + async fetch() { + await AccountService.recoverWallet(); }, }), fetchAccount: FetchMachine.create({ @@ -235,10 +267,9 @@ export const accountsMachine = createMachine( accountToFetch = await AccountService.getCurrentAccount(); } if (!accountToFetch) return undefined; + const selectedNetwork = await NetworkService.getSelectedNetwork(); - if (!selectedNetwork) { - throw new Error('No selected network'); - } + if (!selectedNetwork) return undefined; const providerUrl = selectedNetwork.url; const accountWithBalance = await AccountService.fetchBalance({ @@ -277,11 +308,17 @@ export const accountsMachine = createMachine( isLoggedIn: () => { return !!Storage.getItem(IS_LOGGED_KEY); }, - hasAccount: (ctx, ev) => { - return Boolean(ev?.data || ctx?.account); + hasAccountsOrNeedsRecovery: (ctx, ev) => { + const hasAccounts = Boolean( + (ev.data.accounts || ctx?.accounts || []).length + ); + const needsRecovery = Boolean( + ev.data.needsRecovery || ctx?.needsRecovery + ); + return hasAccounts || needsRecovery; }, - hasAccounts: (ctx, ev) => { - return Boolean((ev.data || ctx?.accounts || []).length); + shouldSkipLoading: (_, ev) => { + return !!ev.input?.skipLoading; }, }, } diff --git a/packages/app/src/systems/Account/machines/addAccountMachine.tsx b/packages/app/src/systems/Account/machines/addAccountMachine.tsx index 401688f504..f94ff881b5 100644 --- a/packages/app/src/systems/Account/machines/addAccountMachine.tsx +++ b/packages/app/src/systems/Account/machines/addAccountMachine.tsx @@ -74,7 +74,7 @@ export const addAccountMachine = createMachine( { actions: { notifyUpdateAccounts: () => { - store.updateAccounts(); + store.refreshAccounts(); }, redirectToHome() { store.closeOverlay(); diff --git a/packages/app/src/systems/Account/machines/editAccountMachine.tsx b/packages/app/src/systems/Account/machines/editAccountMachine.tsx index 3bfc66ee49..8e8a15f117 100644 --- a/packages/app/src/systems/Account/machines/editAccountMachine.tsx +++ b/packages/app/src/systems/Account/machines/editAccountMachine.tsx @@ -97,7 +97,7 @@ export const editAccountMachine = createMachine( account: (_, ev) => ev.data, }), notifyUpdateAccounts: () => { - store.updateAccounts(); + store.refreshAccounts({ skipLoading: true }); }, redirectToList() { store.openAccountList(); diff --git a/packages/app/src/systems/Account/machines/importAccountMachine.tsx b/packages/app/src/systems/Account/machines/importAccountMachine.tsx index 2bb1b758d3..1c3431dcb8 100644 --- a/packages/app/src/systems/Account/machines/importAccountMachine.tsx +++ b/packages/app/src/systems/Account/machines/importAccountMachine.tsx @@ -68,7 +68,7 @@ export const importAccountMachine = createMachine( { actions: { notifyUpdateAccounts: () => { - store.updateAccounts(); + store.refreshAccounts(); }, redirectToHome() { store.closeOverlay(); diff --git a/packages/app/src/systems/Account/services/account.ts b/packages/app/src/systems/Account/services/account.ts index 8a86ea2e74..0130623f6f 100644 --- a/packages/app/src/systems/Account/services/account.ts +++ b/packages/app/src/systems/Account/services/account.ts @@ -5,13 +5,14 @@ import type { AccountWithBalance, CoinAsset, } from '@fuel-wallet/types'; +import * as Sentry from '@sentry/react'; import { Address, type Provider, bn } from 'fuels'; import { AssetsCache } from '~/systems/Asset/cache/AssetsCache'; -import { AssetService } from '~/systems/Asset/services'; -import { getFuelAssetByAssetId } from '~/systems/Asset/utils'; +import { chromeStorage } from '~/systems/Core/services/chromeStorage'; import type { Maybe } from '~/systems/Core/types'; import { db } from '~/systems/Core/utils/database'; import { getUniqueString } from '~/systems/Core/utils/string'; +import { getTestNoDexieDbData } from '../utils/getTestNoDexieDbData'; export type AccountInputs = { addAccount: { @@ -101,6 +102,7 @@ export class AccountService { } const { account, providerUrl } = input; + try { const provider = await createProvider(providerUrl!); const balances = await getBalances(provider, account.publicKey); @@ -194,8 +196,7 @@ export class AccountService { }); } - static setCurrentAccountToDefault() { - console.log('recovering default'); + static async setCurrentAccountToDefault() { return db.transaction('rw', db.accounts, async () => { const [firstAccount] = await db.accounts.toArray(); if (firstAccount) { @@ -206,6 +207,119 @@ export class AccountService { }); } + static async fetchRecoveryState() { + const [ + backupAccounts, + allAccounts, + backupVaults, + allVaults, + backupNetworks, + allNetworks, + ] = await Promise.all([ + chromeStorage.accounts.getAll(), + db.accounts.toArray(), + chromeStorage.vaults.getAll(), + db.vaults.toArray(), + chromeStorage.networks.getAll(), + db.networks.toArray(), + ]); + + // if there is no accounts, means the user lost it. try recovering it + const needsAccRecovery = + allAccounts?.length === 0 && backupAccounts?.length > 0; + const needsVaultRecovery = + allVaults?.length === 0 && backupVaults?.length > 0; + const needsNetworkRecovery = + allNetworks?.length === 0 && backupNetworks?.length > 0; + const needsRecovery = + needsAccRecovery || needsVaultRecovery || needsNetworkRecovery; + + return { + backupAccounts, + backupVaults, + backupNetworks, + needsRecovery, + needsAccRecovery, + needsVaultRecovery, + needsNetworkRecovery, + }; + } + + static async recoverWallet() { + const { + backupAccounts, + backupVaults, + backupNetworks, + needsRecovery, + needsAccRecovery, + needsVaultRecovery, + needsNetworkRecovery, + } = await AccountService.fetchRecoveryState(); + + if (needsRecovery) { + // biome-ignore lint/suspicious/noExplicitAny: + const dataToLog: any = { + backupAccounts: JSON.stringify(backupAccounts), + backupNetworks: JSON.stringify(backupNetworks), + }; + + (async () => { + try { + // try getting data from indexedDB (outside of dexie) to check if it's also corrupted + const testNoDexieDbData = await getTestNoDexieDbData(); + dataToLog.testNoDexieDbData = testNoDexieDbData; + } catch (_) {} + + Sentry.captureException( + 'Disaster on DB. Start recovering accounts / vaults / networks', + { + extra: dataToLog, + tags: { manual: true }, + } + ); + })(); + + await db.transaction( + 'rw', + db.accounts, + db.vaults, + db.networks, + async () => { + if (needsAccRecovery) { + let isCurrentFlag = true; + console.log('recovering accounts', backupAccounts); + for (const account of backupAccounts) { + // in case of recovery, the first account will be the current + if (account.key && account.data.address) { + await db.accounts.add({ + ...account.data, + isCurrent: isCurrentFlag, + }); + isCurrentFlag = false; + } + } + } + if (needsVaultRecovery) { + console.log('recovering vaults', backupVaults); + for (const vault of backupVaults) { + if (vault.key && vault.data) { + await db.vaults.add(vault.data); + } + } + } + if (needsNetworkRecovery) { + console.log('recovering networks', backupNetworks); + for (const network of backupNetworks) { + if (network.key && network.data.id) { + await db.networks.add(network.data); + } + } + } + } + ); + } + } + static setCurrentAccount(input: AccountInputs['setCurrentAccount']) { return db.transaction('rw', db.accounts, async () => { await db.accounts diff --git a/packages/app/src/systems/Account/utils/getTestNoDexieDbData.ts b/packages/app/src/systems/Account/utils/getTestNoDexieDbData.ts new file mode 100644 index 0000000000..c84747aaf8 --- /dev/null +++ b/packages/app/src/systems/Account/utils/getTestNoDexieDbData.ts @@ -0,0 +1,23 @@ +export const getTestNoDexieDbData = () => + new Promise((resolve, reject) => { + async function getData() { + const request = await window.indexedDB.open('TestDatabase'); + request.onsuccess = (event: Event) => { + const db = (event.target as IDBRequest).result as IDBDatabase; + const tx = db.transaction('myTable', 'readonly'); + const store = tx.objectStore('myTable'); + + // Get all the data from the object store + const request = store.getAll(); + request.onsuccess = (event: Event) => { + const data = (event.target as IDBRequest).result; + resolve(data); + }; + }; + } + getData(); + + setTimeout(() => { + reject(new Error('Timeout')); + }, 10000); + }); diff --git a/packages/app/src/systems/Account/utils/storage.tsx b/packages/app/src/systems/Account/utils/storage.tsx index 06c12378be..7c418c900c 100644 --- a/packages/app/src/systems/Account/utils/storage.tsx +++ b/packages/app/src/systems/Account/utils/storage.tsx @@ -1,8 +1,12 @@ import type { StorageAbstract } from 'fuels'; +import { chromeStorage } from '~/systems/Core/services/chromeStorage'; import { db } from '~/systems/Core/utils/database'; export class IndexedDBStorage implements StorageAbstract { async getItem(key: string) { + const vault = await chromeStorage.vaults.get({ key }); + if (vault?.data?.data) return vault?.data?.data; + return db.transaction('r', db.vaults, async () => { const vault = await db.vaults.get({ key }); return vault?.data; @@ -10,18 +14,22 @@ export class IndexedDBStorage implements StorageAbstract { } async setItem(key: string, data: string) { + await chromeStorage.vaults.set({ key, data: { key, data } }); await db.transaction('rw', db.vaults, db.accounts, async () => { await db.vaults.put({ key, data }); }); } async removeItem(key: string) { + await chromeStorage.vaults.remove({ key }); await db.transaction('rw', db.vaults, db.accounts, async () => { await db.vaults.where({ key }).delete(); }); } async clear() { + await chromeStorage.vaults.clear(); + await chromeStorage.accounts.clear(); await db.transaction('rw', db.vaults, db.accounts, async () => { await db.vaults.clear(); await db.accounts.clear(); diff --git a/packages/app/src/systems/Asset/machines/assetsMachine.tsx b/packages/app/src/systems/Asset/machines/assetsMachine.tsx index 2a21694925..6c459f2010 100644 --- a/packages/app/src/systems/Asset/machines/assetsMachine.tsx +++ b/packages/app/src/systems/Asset/machines/assetsMachine.tsx @@ -179,7 +179,7 @@ export const assetsMachine = createMachine( toast.success('Asset added successfully'); }, notifyUpdateAccounts: () => { - store.updateAccounts(); + store.refreshAccounts({ skipLoading: true }); }, }, services: { diff --git a/packages/app/src/systems/CRX/background/actions/onInstall.ts b/packages/app/src/systems/CRX/background/actions/onInstall.ts index f6560abefa..fb22b742a1 100644 --- a/packages/app/src/systems/CRX/background/actions/onInstall.ts +++ b/packages/app/src/systems/CRX/background/actions/onInstall.ts @@ -5,16 +5,7 @@ import { executeContentScript } from '../../scripts/executeContentScript'; executeContentScript(); chrome.runtime.onInstalled.addListener(async (object) => { - const { shouldRecoverWelcomeFromError } = await chrome.storage.local.get( - 'shouldRecoverWelcomeFromError' - ); - - chrome.storage.local.remove('shouldRecoverWelcomeFromError'); - - if ( - shouldRecoverWelcomeFromError || - object.reason === chrome.runtime.OnInstalledReason.INSTALL - ) { + if (object.reason === chrome.runtime.OnInstalledReason.INSTALL) { chrome.tabs.create({ url: welcomeLink() }); } }); diff --git a/packages/app/src/systems/CRX/background/services/DatabaseEvents.ts b/packages/app/src/systems/CRX/background/services/DatabaseEvents.ts index 347f9e2a4b..e126d33bb2 100644 --- a/packages/app/src/systems/CRX/background/services/DatabaseEvents.ts +++ b/packages/app/src/systems/CRX/background/services/DatabaseEvents.ts @@ -17,6 +17,7 @@ import { } from 'fuels'; import type { CommunicationProtocol } from './CommunicationProtocol'; import { DatabaseObservable } from './DatabaseObservable'; +import { chromeStorage } from '~/systems/Core/services/chromeStorage'; export class DatabaseEvents { readonly databaseObservable: DatabaseObservable< @@ -74,6 +75,34 @@ export class DatabaseEvents { } ); + // -- START Events for sync db with chrome storage + this.databaseObservable.on<'accounts:create', Account>( + 'accounts:create', + async (event) => { + const currentAccount = event.obj; + if (currentAccount) { + await chromeStorage.accounts.set({ + key: currentAccount.address, + data: currentAccount, + }); + } + } + ); + this.databaseObservable.on<'accounts:update', Account>( + 'accounts:update', + async (event) => { + const currentAccount = event.obj; + + if (currentAccount) { + await chromeStorage.accounts.set({ + key: currentAccount.address, + data: currentAccount, + }); + } + } + ); + // -- END Events for sync db with chrome storage + this.databaseObservable.on<'accounts:update', Account>( 'accounts:update', async (updateEvent) => { diff --git a/packages/app/src/systems/CRX/background/services/VaultService.ts b/packages/app/src/systems/CRX/background/services/VaultService.ts index e757118e2d..e8530c5d24 100644 --- a/packages/app/src/systems/CRX/background/services/VaultService.ts +++ b/packages/app/src/systems/CRX/background/services/VaultService.ts @@ -135,8 +135,7 @@ export class VaultService extends VaultServer { if (eventType === 'DB_EVENT' && payload.event === 'restarted') { if (!integrity) { - chrome.storage.local.set({ shouldRecoverWelcomeFromError: true }); - return this.resetAndReload(); + return this.reload(); } } diff --git a/packages/app/src/systems/CRX/scripts/executeContentScript.ts b/packages/app/src/systems/CRX/scripts/executeContentScript.ts index 292c29b194..8f4dc262f7 100644 --- a/packages/app/src/systems/CRX/scripts/executeContentScript.ts +++ b/packages/app/src/systems/CRX/scripts/executeContentScript.ts @@ -34,7 +34,10 @@ export async function executeContentScript() { } function injectContentScript(tabId: number) { - const env = process.env?.NODE_ENV; + let env: string | undefined = undefined; + if (typeof process !== 'undefined') { + env = process?.env?.NODE_ENV; + } chrome.scripting .executeScript({ target: { tabId: tabId, allFrames: true }, diff --git a/packages/app/src/systems/Core/services/chromeStorage.ts b/packages/app/src/systems/Core/services/chromeStorage.ts new file mode 100644 index 0000000000..c2eab0b9e4 --- /dev/null +++ b/packages/app/src/systems/Core/services/chromeStorage.ts @@ -0,0 +1,87 @@ +import type { Account, NetworkData, Vault } from '@fuel-wallet/types'; + +interface ChromeStorageRow { + key: string; + data: T; +} + +class ChromeStorageTable { + constructor(private readonly tableName: string) { + this.tableName = tableName; + } + + async get({ key }: { key: string }) { + const rowsMap = (await chrome?.storage?.local?.get(this.tableName)) || {}; + const rows = rowsMap[this.tableName] || []; + + let foundIndex = -1; + for (let i = 0; i < rows.length; i++) { + if (rows[i].key === key) { + foundIndex = i; + break; + } + } + const found = rows[foundIndex]?.data; + + return { + data: found, + key, + index: foundIndex, + rows, + }; + } + + async getAll(): Promise[]> { + const rowsMap = (await chrome?.storage?.local?.get(this.tableName)) || {}; + const rows: ChromeStorageRow[] = rowsMap[this.tableName] || []; + return rows; + } + + async set({ key, data }: ChromeStorageRow) { + const { index, rows } = await this.get({ key }); + + // update + if (index !== -1) { + rows[index] = { + key, + data, + }; + } else { + // create + rows.push({ + key, + data, + }); + } + + await chrome?.storage?.local?.set({ + [this.tableName]: rows, + }); + } + + async remove({ key }: { key: string }) { + const { index, rows } = await this.get({ key }); + + if (index !== -1) { + rows.splice(index, 1); + await chrome?.storage?.local?.set({ + [this.tableName]: rows, + }); + } + } + + async clear() { + await chrome?.storage?.local?.set({ + [this.tableName]: [], + }); + } +} + +export const chromeStorage = { + accounts: new ChromeStorageTable('accounts'), + networks: new ChromeStorageTable('networks'), + vaults: new ChromeStorageTable('vaults'), + clear: () => { + chrome?.storage?.local?.clear(); + } +}; diff --git a/packages/app/src/systems/Core/services/core.ts b/packages/app/src/systems/Core/services/core.ts index ad52adcf55..f6bbbd650c 100644 --- a/packages/app/src/systems/Core/services/core.ts +++ b/packages/app/src/systems/Core/services/core.ts @@ -3,12 +3,14 @@ import { VaultService } from '~/systems/Vault'; import { delay } from '../utils'; import { db } from '../utils/database'; import { Storage } from '../utils/storage'; +import { chromeStorage } from './chromeStorage'; // biome-ignore lint/complexity/noStaticOnlyClass: export class CoreService { static async clear() { toast.success('Your wallet will be reset'); await delay(1500); + await chromeStorage.clear(); await VaultService.clear(); await db.clear(); await Storage.clear(); diff --git a/packages/app/src/systems/Core/services/index.ts b/packages/app/src/systems/Core/services/index.ts index 4b0e041376..57a466133f 100644 --- a/packages/app/src/systems/Core/services/index.ts +++ b/packages/app/src/systems/Core/services/index.ts @@ -1 +1,2 @@ export * from './core'; +export * from './chromeStorage'; diff --git a/packages/app/src/systems/Core/utils/database.ts b/packages/app/src/systems/Core/utils/database.ts index 489297ae2d..ac23629546 100644 --- a/packages/app/src/systems/Core/utils/database.ts +++ b/packages/app/src/systems/Core/utils/database.ts @@ -12,6 +12,7 @@ import Dexie, { type DbEvents, type PromiseExtended, type Table } from 'dexie'; import 'dexie-observable'; import type { AssetFuel } from 'fuels'; import type { TransactionCursor } from '~/systems/Transaction'; +import { chromeStorage } from '../services/chromeStorage'; import { applyDbVersioning } from './databaseVersioning'; type FailureEvents = Extract; @@ -43,10 +44,75 @@ export class FuelDB extends Dexie { this.on('close', () => this.restart('close')); } + async createParallelDb() { + // add table outside of dexie to test if it will be corrupted also with dexie FuelDB + if (typeof window !== "undefined") { + const request = await window.indexedDB.open('TestDatabase', 2); + request.onupgradeneeded = (event) => { + const db = (event.target as IDBRequest)?.result; + db.createObjectStore('myTable', { keyPath: 'id' }); + }; + request.onsuccess = (event: Event) => { + const db = (event.target as IDBRequest).result as IDBDatabase; + const tx = db.transaction('myTable', 'readwrite'); + const store = tx.objectStore('myTable'); + + const countRequest = store.count(); + countRequest.onsuccess = () => { + if (countRequest.result === 0) { + store.add({ id: 1, name: 'John' }); + } + }; + }; + } + } + + async syncDbToChromeStorage() { + const accounts = await this.accounts.toArray(); + const vaults = await this.vaults.toArray(); + const networks = await this.networks.toArray(); + + // @TODO: this is a temporary solution to avoid the storage accounts of being wrong and + // users losing funds in case of no backup + // if has account, save to chrome storage + if (accounts.length) { + for (const account of accounts) { + await chromeStorage.accounts.set({ + key: account.address, + data: account, + }); + } + } + if (vaults.length) { + for (const vault of vaults) { + await chromeStorage.vaults.set({ + key: vault.key, + data: vault, + }); + } + } + if (networks.length) { + for (const network of networks) { + await chromeStorage.networks.set({ + key: network.id || '', + data: network, + }); + } + } + } + open(): PromiseExtended { try { return super.open().then((res) => { this.restartAttempts = 0; + try { + (() => this.createParallelDb())(); + } catch(_){} + + try { + (() => this.syncDbToChromeStorage())(); + } catch(_){} + return res; }); } catch (err) { diff --git a/packages/app/src/systems/Core/utils/databaseVersioning.ts b/packages/app/src/systems/Core/utils/databaseVersioning.ts index 8477ff9385..b4f642bd5e 100644 --- a/packages/app/src/systems/Core/utils/databaseVersioning.ts +++ b/packages/app/src/systems/Core/utils/databaseVersioning.ts @@ -232,6 +232,9 @@ export const applyDbVersioning = (db: Dexie) => { abis: '&contractId', errors: '&id', }); + + // DB VERSION 28 + // add fetchedAt column to indexedAssets table db.version(28).stores({ vaults: 'key', accounts: '&address, &name', diff --git a/packages/app/src/systems/DApp/machines/selectNetworkRequestMachine.tsx b/packages/app/src/systems/DApp/machines/selectNetworkRequestMachine.tsx index 1efa46f40c..79fec3f634 100644 --- a/packages/app/src/systems/DApp/machines/selectNetworkRequestMachine.tsx +++ b/packages/app/src/systems/DApp/machines/selectNetworkRequestMachine.tsx @@ -115,7 +115,7 @@ export const selectNetworkRequestMachine = createMachine( store.refreshNetworks(); }, notifyUpdateAccounts: () => { - store.updateAccounts(); + store.refreshAccounts(); }, }, services: { diff --git a/packages/app/src/systems/Home/pages/Home/Home.tsx b/packages/app/src/systems/Home/pages/Home/Home.tsx index 93e60fba11..bc6c0e0b68 100644 --- a/packages/app/src/systems/Home/pages/Home/Home.tsx +++ b/packages/app/src/systems/Home/pages/Home/Home.tsx @@ -43,7 +43,7 @@ export function Home() { diff --git a/packages/app/src/systems/Network/machines/networksMachine.ts b/packages/app/src/systems/Network/machines/networksMachine.ts index 62f91cca79..d357069227 100644 --- a/packages/app/src/systems/Network/machines/networksMachine.ts +++ b/packages/app/src/systems/Network/machines/networksMachine.ts @@ -319,7 +319,7 @@ export const networksMachine = createMachine( }, }), notifyUpdateAccounts: () => { - store.updateAccounts(); + store.refreshAccounts(); }, assignChainInfo: assign({ chainInfoToAdd: (_, ev) => { diff --git a/packages/app/src/systems/SignUp/machines/signUpMachine.ts b/packages/app/src/systems/SignUp/machines/signUpMachine.ts index 5651629fa4..5e5846b63d 100644 --- a/packages/app/src/systems/SignUp/machines/signUpMachine.ts +++ b/packages/app/src/systems/SignUp/machines/signUpMachine.ts @@ -228,7 +228,7 @@ export const signUpMachine = createMachine( // External actions sendAccountCreated: () => { Storage.setItem(IS_LOGGED_KEY, true); - store.updateAccounts(); + store.refreshAccounts(); }, redirectToWalletCreated() {}, }, diff --git a/packages/app/src/systems/Vault/services/VaultServer.ts b/packages/app/src/systems/Vault/services/VaultServer.ts index 176c33e51f..2ada8b89c6 100644 --- a/packages/app/src/systems/Vault/services/VaultServer.ts +++ b/packages/app/src/systems/Vault/services/VaultServer.ts @@ -209,10 +209,6 @@ export class VaultServer extends EventEmitter { async reload() { chrome.runtime.reload(); } - - async resetAndReload() { - return this.reload(); - } } export type VaultMethods = {