2024-05-31 17:52:29 +02:00
|
|
|
import React, { createContext, useCallback, useEffect, useMemo, useRef, useState } from 'react';
|
2024-05-20 11:54:13 +02:00
|
|
|
import { InteractionManager } from 'react-native';
|
2024-05-31 17:52:29 +02:00
|
|
|
import A from '../../blue_modules/analytics';
|
|
|
|
import Notifications from '../../blue_modules/notifications';
|
|
|
|
import { BlueApp as BlueAppClass, LegacyWallet, TCounterpartyMetadata, TTXMetadata, WatchOnlyWallet } from '../../class';
|
|
|
|
import type { TWallet } from '../../class/wallets/types';
|
|
|
|
import presentAlert from '../../components/Alert';
|
|
|
|
import loc from '../../loc';
|
|
|
|
import * as BlueElectrum from '../../blue_modules/BlueElectrum';
|
|
|
|
import triggerHapticFeedback, { HapticFeedbackTypes } from '../../blue_modules/hapticFeedback';
|
|
|
|
import { startAndDecrypt } from '../../blue_modules/start-and-decrypt';
|
2024-03-15 21:05:15 +01:00
|
|
|
|
2024-04-15 22:53:44 +02:00
|
|
|
const BlueApp = BlueAppClass.getInstance();
|
|
|
|
|
2024-03-15 21:05:15 +01:00
|
|
|
// hashmap of timestamps we _started_ refetching some wallet
|
|
|
|
const _lastTimeTriedToRefetchWallet: { [walletID: string]: number } = {};
|
|
|
|
|
2024-05-31 17:54:22 +02:00
|
|
|
interface StorageContextType {
|
2024-03-15 21:05:15 +01:00
|
|
|
wallets: TWallet[];
|
|
|
|
setWalletsWithNewOrder: (wallets: TWallet[]) => void;
|
2024-05-13 19:17:19 +02:00
|
|
|
txMetadata: TTXMetadata;
|
|
|
|
counterpartyMetadata: TCounterpartyMetadata;
|
2024-03-15 21:05:15 +01:00
|
|
|
saveToDisk: (force?: boolean) => Promise<void>;
|
|
|
|
selectedWalletID: string | undefined;
|
|
|
|
setSelectedWalletID: (walletID: string | undefined) => void;
|
|
|
|
addWallet: (wallet: TWallet) => void;
|
|
|
|
deleteWallet: (wallet: TWallet) => void;
|
|
|
|
currentSharedCosigner: string;
|
|
|
|
setSharedCosigner: (cosigner: string) => void;
|
|
|
|
addAndSaveWallet: (wallet: TWallet) => Promise<void>;
|
|
|
|
fetchAndSaveWalletTransactions: (walletID: string) => Promise<void>;
|
|
|
|
walletsInitialized: boolean;
|
|
|
|
setWalletsInitialized: (initialized: boolean) => void;
|
|
|
|
refreshAllWalletTransactions: (lastSnappedTo?: number, showUpdateStatusIndicator?: boolean) => Promise<void>;
|
|
|
|
resetWallets: () => void;
|
|
|
|
walletTransactionUpdateStatus: WalletTransactionsStatus | string;
|
|
|
|
setWalletTransactionUpdateStatus: (status: WalletTransactionsStatus | string) => void;
|
|
|
|
isElectrumDisabled: boolean;
|
|
|
|
setIsElectrumDisabled: (value: boolean) => void;
|
|
|
|
reloadTransactionsMenuActionFunction: () => void;
|
|
|
|
setReloadTransactionsMenuActionFunction: (func: () => void) => void;
|
|
|
|
getTransactions: typeof BlueApp.getTransactions;
|
|
|
|
fetchWalletBalances: typeof BlueApp.fetchWalletBalances;
|
|
|
|
fetchWalletTransactions: typeof BlueApp.fetchWalletTransactions;
|
|
|
|
getBalance: typeof BlueApp.getBalance;
|
|
|
|
isStorageEncrypted: typeof BlueApp.storageIsEncrypted;
|
|
|
|
startAndDecrypt: typeof startAndDecrypt;
|
|
|
|
encryptStorage: typeof BlueApp.encryptStorage;
|
|
|
|
sleep: typeof BlueApp.sleep;
|
|
|
|
createFakeStorage: typeof BlueApp.createFakeStorage;
|
|
|
|
decryptStorage: typeof BlueApp.decryptStorage;
|
|
|
|
isPasswordInUse: typeof BlueApp.isPasswordInUse;
|
|
|
|
cachedPassword: typeof BlueApp.cachedPassword;
|
|
|
|
getItem: typeof BlueApp.getItem;
|
|
|
|
setItem: typeof BlueApp.setItem;
|
|
|
|
}
|
|
|
|
|
|
|
|
export enum WalletTransactionsStatus {
|
|
|
|
NONE = 'NONE',
|
|
|
|
ALL = 'ALL',
|
|
|
|
}
|
|
|
|
// @ts-ignore defaut value does not match the type
|
2024-05-31 17:54:22 +02:00
|
|
|
export const StorageContext = createContext<StorageContextType>(undefined);
|
|
|
|
export const StorageProvider = ({ children }: { children: React.ReactNode }) => {
|
2024-05-27 00:58:43 +02:00
|
|
|
const txMetadata = useRef<TTXMetadata>(BlueApp.tx_metadata);
|
|
|
|
const counterpartyMetadata = useRef<TCounterpartyMetadata>(BlueApp.counterparty_metadata || {}); // init
|
2024-05-12 23:14:52 +02:00
|
|
|
const getTransactions = BlueApp.getTransactions;
|
|
|
|
const fetchWalletBalances = BlueApp.fetchWalletBalances;
|
|
|
|
const fetchWalletTransactions = BlueApp.fetchWalletTransactions;
|
|
|
|
const getBalance = BlueApp.getBalance;
|
|
|
|
const isStorageEncrypted = BlueApp.storageIsEncrypted;
|
|
|
|
const encryptStorage = BlueApp.encryptStorage;
|
|
|
|
const sleep = BlueApp.sleep;
|
|
|
|
const createFakeStorage = BlueApp.createFakeStorage;
|
|
|
|
const decryptStorage = BlueApp.decryptStorage;
|
|
|
|
const isPasswordInUse = BlueApp.isPasswordInUse;
|
|
|
|
const cachedPassword = BlueApp.cachedPassword;
|
|
|
|
|
|
|
|
const getItem = BlueApp.getItem;
|
|
|
|
const setItem = BlueApp.setItem;
|
|
|
|
|
2024-03-15 21:05:15 +01:00
|
|
|
const [wallets, setWallets] = useState<TWallet[]>([]);
|
|
|
|
const [selectedWalletID, setSelectedWalletID] = useState<undefined | string>();
|
|
|
|
const [walletTransactionUpdateStatus, setWalletTransactionUpdateStatus] = useState<WalletTransactionsStatus | string>(
|
|
|
|
WalletTransactionsStatus.NONE,
|
|
|
|
);
|
|
|
|
const [walletsInitialized, setWalletsInitialized] = useState<boolean>(false);
|
|
|
|
const [isElectrumDisabled, setIsElectrumDisabled] = useState<boolean>(true);
|
|
|
|
const [currentSharedCosigner, setCurrentSharedCosigner] = useState<string>('');
|
|
|
|
const [reloadTransactionsMenuActionFunction, setReloadTransactionsMenuActionFunction] = useState<() => void>(() => {});
|
2021-08-24 07:00:57 +02:00
|
|
|
|
|
|
|
useEffect(() => {
|
2024-05-05 01:38:39 +02:00
|
|
|
BlueElectrum.isDisabled().then(setIsElectrumDisabled);
|
2024-03-10 23:37:02 +01:00
|
|
|
if (walletsInitialized) {
|
2024-05-27 00:10:38 +02:00
|
|
|
txMetadata.current = BlueApp.tx_metadata;
|
|
|
|
counterpartyMetadata.current = BlueApp.counterparty_metadata;
|
2024-05-27 00:58:43 +02:00
|
|
|
setWallets(BlueApp.getWallets());
|
2024-03-10 23:37:02 +01:00
|
|
|
BlueElectrum.connectMain();
|
|
|
|
}
|
|
|
|
}, [walletsInitialized]);
|
|
|
|
|
2024-05-12 23:14:52 +02:00
|
|
|
const saveToDisk = useCallback(async (force: boolean = false) => {
|
2024-05-05 01:38:39 +02:00
|
|
|
InteractionManager.runAfterInteractions(async () => {
|
|
|
|
if (BlueApp.getWallets().length === 0 && !force) {
|
|
|
|
console.log('not saving empty wallets array');
|
|
|
|
return;
|
|
|
|
}
|
2024-05-12 23:14:52 +02:00
|
|
|
BlueApp.tx_metadata = txMetadata.current;
|
|
|
|
BlueApp.counterparty_metadata = counterpartyMetadata.current;
|
2024-05-05 01:38:39 +02:00
|
|
|
await BlueApp.saveToDisk();
|
|
|
|
setWallets([...BlueApp.getWallets()]);
|
2024-05-12 23:14:52 +02:00
|
|
|
txMetadata.current = BlueApp.tx_metadata;
|
|
|
|
counterpartyMetadata.current = BlueApp.counterparty_metadata;
|
2024-05-05 01:38:39 +02:00
|
|
|
});
|
2024-05-12 23:14:52 +02:00
|
|
|
}, []);
|
2020-10-24 19:20:59 +02:00
|
|
|
|
|
|
|
const resetWallets = () => {
|
|
|
|
setWallets(BlueApp.getWallets());
|
|
|
|
};
|
|
|
|
|
2024-05-12 23:14:52 +02:00
|
|
|
const setWalletsWithNewOrder = useCallback(
|
|
|
|
(wlts: TWallet[]) => {
|
|
|
|
BlueApp.wallets = wlts;
|
|
|
|
saveToDisk();
|
|
|
|
},
|
|
|
|
[saveToDisk],
|
|
|
|
);
|
2020-10-24 19:20:59 +02:00
|
|
|
|
2024-05-12 23:14:52 +02:00
|
|
|
const refreshAllWalletTransactions = useCallback(
|
|
|
|
async (lastSnappedTo?: number, showUpdateStatusIndicator: boolean = true) => {
|
2024-08-25 20:50:04 +02:00
|
|
|
const TIMEOUT_DURATION = 30000;
|
|
|
|
|
|
|
|
const timeoutPromise = new Promise<never>((_resolve, reject) =>
|
|
|
|
setTimeout(() => {
|
|
|
|
reject(new Error('refreshAllWalletTransactions: Timeout reached'));
|
|
|
|
}, TIMEOUT_DURATION),
|
|
|
|
);
|
|
|
|
|
|
|
|
const mainLogicPromise = new Promise<void>((resolve, reject) => {
|
2024-05-12 23:14:52 +02:00
|
|
|
try {
|
2024-08-25 20:50:04 +02:00
|
|
|
InteractionManager.runAfterInteractions(async () => {
|
|
|
|
let noErr = true;
|
|
|
|
try {
|
|
|
|
await BlueElectrum.waitTillConnected();
|
|
|
|
if (showUpdateStatusIndicator) {
|
|
|
|
setWalletTransactionUpdateStatus(WalletTransactionsStatus.ALL);
|
|
|
|
}
|
|
|
|
const paymentCodesStart = Date.now();
|
|
|
|
await BlueApp.fetchSenderPaymentCodes(lastSnappedTo);
|
|
|
|
const paymentCodesEnd = Date.now();
|
|
|
|
console.log('fetch payment codes took', (paymentCodesEnd - paymentCodesStart) / 1000, 'sec');
|
|
|
|
const balanceStart = +new Date();
|
|
|
|
await fetchWalletBalances(lastSnappedTo);
|
|
|
|
const balanceEnd = +new Date();
|
|
|
|
console.log('fetch balance took', (balanceEnd - balanceStart) / 1000, 'sec');
|
|
|
|
const start = +new Date();
|
|
|
|
await fetchWalletTransactions(lastSnappedTo);
|
|
|
|
const end = +new Date();
|
|
|
|
console.log('fetch tx took', (end - start) / 1000, 'sec');
|
|
|
|
} catch (err) {
|
|
|
|
noErr = false;
|
|
|
|
console.warn(err);
|
|
|
|
reject(err);
|
|
|
|
} finally {
|
|
|
|
setWalletTransactionUpdateStatus(WalletTransactionsStatus.NONE);
|
|
|
|
}
|
|
|
|
if (noErr) await saveToDisk(); // caching
|
|
|
|
resolve();
|
|
|
|
});
|
2024-05-12 23:14:52 +02:00
|
|
|
} catch (err) {
|
2024-08-25 20:50:04 +02:00
|
|
|
reject(err);
|
2024-05-12 23:14:52 +02:00
|
|
|
} finally {
|
|
|
|
setWalletTransactionUpdateStatus(WalletTransactionsStatus.NONE);
|
|
|
|
}
|
|
|
|
});
|
2024-08-25 20:50:04 +02:00
|
|
|
|
|
|
|
try {
|
|
|
|
await Promise.race([mainLogicPromise, timeoutPromise]);
|
|
|
|
} catch (err) {
|
|
|
|
console.error('Error in refreshAllWalletTransactions:', err);
|
|
|
|
} finally {
|
|
|
|
setWalletTransactionUpdateStatus(WalletTransactionsStatus.NONE);
|
|
|
|
}
|
2024-05-12 23:14:52 +02:00
|
|
|
},
|
|
|
|
[fetchWalletBalances, fetchWalletTransactions, saveToDisk],
|
|
|
|
);
|
|
|
|
const fetchAndSaveWalletTransactions = useCallback(
|
|
|
|
async (walletID: string) => {
|
|
|
|
InteractionManager.runAfterInteractions(async () => {
|
|
|
|
const index = wallets.findIndex(wallet => wallet.getID() === walletID);
|
|
|
|
let noErr = true;
|
|
|
|
try {
|
|
|
|
// 5sec debounce:
|
|
|
|
if (+new Date() - _lastTimeTriedToRefetchWallet[walletID] < 5000) {
|
|
|
|
console.log('re-fetch wallet happens too fast; NOP');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
_lastTimeTriedToRefetchWallet[walletID] = +new Date();
|
2021-01-01 20:15:40 +01:00
|
|
|
|
2024-05-12 23:14:52 +02:00
|
|
|
await BlueElectrum.waitTillConnected();
|
|
|
|
setWalletTransactionUpdateStatus(walletID);
|
|
|
|
const balanceStart = +new Date();
|
|
|
|
await fetchWalletBalances(index);
|
|
|
|
const balanceEnd = +new Date();
|
|
|
|
console.log('fetch balance took', (balanceEnd - balanceStart) / 1000, 'sec');
|
|
|
|
const start = +new Date();
|
|
|
|
await fetchWalletTransactions(index);
|
|
|
|
const end = +new Date();
|
|
|
|
console.log('fetch tx took', (end - start) / 1000, 'sec');
|
|
|
|
} catch (err) {
|
|
|
|
noErr = false;
|
|
|
|
console.warn(err);
|
|
|
|
} finally {
|
|
|
|
setWalletTransactionUpdateStatus(WalletTransactionsStatus.NONE);
|
|
|
|
}
|
|
|
|
if (noErr) await saveToDisk(); // caching
|
|
|
|
});
|
|
|
|
},
|
|
|
|
[fetchWalletBalances, fetchWalletTransactions, saveToDisk, wallets],
|
|
|
|
);
|
2020-10-24 19:20:59 +02:00
|
|
|
|
2024-05-12 23:14:52 +02:00
|
|
|
const addWallet = useCallback((wallet: TWallet) => {
|
2020-10-24 19:20:59 +02:00
|
|
|
BlueApp.wallets.push(wallet);
|
|
|
|
setWallets([...BlueApp.getWallets()]);
|
2024-05-12 23:14:52 +02:00
|
|
|
}, []);
|
2020-10-24 19:20:59 +02:00
|
|
|
|
2024-05-12 23:14:52 +02:00
|
|
|
const deleteWallet = useCallback((wallet: TWallet) => {
|
2020-10-24 19:20:59 +02:00
|
|
|
BlueApp.deleteWallet(wallet);
|
|
|
|
setWallets([...BlueApp.getWallets()]);
|
2024-05-12 23:14:52 +02:00
|
|
|
}, []);
|
2024-04-18 03:05:48 +02:00
|
|
|
|
2024-05-12 23:14:52 +02:00
|
|
|
const addAndSaveWallet = useCallback(
|
|
|
|
async (w: TWallet) => {
|
|
|
|
if (wallets.some(i => i.getID() === w.getID())) {
|
|
|
|
triggerHapticFeedback(HapticFeedbackTypes.NotificationError);
|
|
|
|
presentAlert({ message: 'This wallet has been previously imported.' });
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const emptyWalletLabel = new LegacyWallet().getLabel();
|
|
|
|
triggerHapticFeedback(HapticFeedbackTypes.NotificationSuccess);
|
|
|
|
if (w.getLabel() === emptyWalletLabel) w.setLabel(loc.wallets.import_imported + ' ' + w.typeReadable);
|
|
|
|
w.setUserHasSavedExport(true);
|
|
|
|
addWallet(w);
|
|
|
|
await saveToDisk();
|
|
|
|
A(A.ENUM.CREATED_WALLET);
|
|
|
|
presentAlert({
|
|
|
|
hapticFeedback: HapticFeedbackTypes.ImpactHeavy,
|
|
|
|
message: w.type === WatchOnlyWallet.type ? loc.wallets.import_success_watchonly : loc.wallets.import_success,
|
|
|
|
});
|
|
|
|
// @ts-ignore need to type notifications first
|
|
|
|
Notifications.majorTomToGroundControl(w.getAllExternalAddresses(), [], []);
|
|
|
|
// start balance fetching at the background
|
|
|
|
await w.fetchBalance();
|
|
|
|
setWallets([...BlueApp.getWallets()]);
|
|
|
|
},
|
|
|
|
[addWallet, saveToDisk, wallets],
|
|
|
|
);
|
2020-10-24 19:20:59 +02:00
|
|
|
|
2024-05-31 17:54:22 +02:00
|
|
|
const value: StorageContextType = useMemo(
|
2024-05-12 23:14:52 +02:00
|
|
|
() => ({
|
|
|
|
wallets,
|
|
|
|
setWalletsWithNewOrder,
|
2024-05-13 19:17:19 +02:00
|
|
|
txMetadata: txMetadata.current,
|
|
|
|
counterpartyMetadata: counterpartyMetadata.current,
|
2024-05-12 23:14:52 +02:00
|
|
|
saveToDisk,
|
|
|
|
getTransactions,
|
|
|
|
selectedWalletID,
|
|
|
|
setSelectedWalletID,
|
|
|
|
addWallet,
|
|
|
|
deleteWallet,
|
|
|
|
currentSharedCosigner,
|
|
|
|
setSharedCosigner: setCurrentSharedCosigner,
|
|
|
|
addAndSaveWallet,
|
|
|
|
setItem,
|
|
|
|
getItem,
|
|
|
|
fetchWalletBalances,
|
|
|
|
fetchWalletTransactions,
|
|
|
|
fetchAndSaveWalletTransactions,
|
|
|
|
isStorageEncrypted,
|
|
|
|
encryptStorage,
|
|
|
|
startAndDecrypt,
|
|
|
|
cachedPassword,
|
|
|
|
getBalance,
|
|
|
|
walletsInitialized,
|
|
|
|
setWalletsInitialized,
|
|
|
|
refreshAllWalletTransactions,
|
|
|
|
sleep,
|
|
|
|
createFakeStorage,
|
|
|
|
resetWallets,
|
|
|
|
decryptStorage,
|
|
|
|
isPasswordInUse,
|
|
|
|
walletTransactionUpdateStatus,
|
|
|
|
setWalletTransactionUpdateStatus,
|
|
|
|
isElectrumDisabled,
|
|
|
|
setIsElectrumDisabled,
|
|
|
|
reloadTransactionsMenuActionFunction,
|
|
|
|
setReloadTransactionsMenuActionFunction,
|
|
|
|
}),
|
|
|
|
[
|
|
|
|
wallets,
|
|
|
|
setWalletsWithNewOrder,
|
|
|
|
saveToDisk,
|
|
|
|
getTransactions,
|
|
|
|
selectedWalletID,
|
|
|
|
addWallet,
|
|
|
|
deleteWallet,
|
|
|
|
currentSharedCosigner,
|
|
|
|
addAndSaveWallet,
|
|
|
|
setItem,
|
|
|
|
getItem,
|
|
|
|
fetchWalletBalances,
|
|
|
|
fetchWalletTransactions,
|
|
|
|
fetchAndSaveWalletTransactions,
|
|
|
|
isStorageEncrypted,
|
|
|
|
encryptStorage,
|
|
|
|
cachedPassword,
|
|
|
|
getBalance,
|
|
|
|
walletsInitialized,
|
|
|
|
refreshAllWalletTransactions,
|
|
|
|
sleep,
|
|
|
|
createFakeStorage,
|
|
|
|
decryptStorage,
|
|
|
|
isPasswordInUse,
|
|
|
|
walletTransactionUpdateStatus,
|
|
|
|
isElectrumDisabled,
|
|
|
|
reloadTransactionsMenuActionFunction,
|
|
|
|
],
|
|
|
|
);
|
2024-03-15 21:05:15 +01:00
|
|
|
|
2024-05-31 17:54:22 +02:00
|
|
|
return <StorageContext.Provider value={value}>{children}</StorageContext.Provider>;
|
2020-10-24 19:20:59 +02:00
|
|
|
};
|