BlueWallet/class/blue-app.ts

971 lines
35 KiB
TypeScript
Raw Normal View History

2023-04-21 17:39:12 +02:00
import AsyncStorage from '@react-native-async-storage/async-storage';
2024-04-14 11:31:08 +02:00
import createHash from 'create-hash';
2024-01-29 04:04:48 +01:00
import DefaultPreference from 'react-native-default-preference';
2024-04-14 11:31:08 +02:00
import RNFS from 'react-native-fs';
import Keychain from 'react-native-keychain';
2024-05-20 11:54:13 +02:00
import RNSecureKeyStore, { ACCESSIBLE } from 'react-native-secure-key-store';
import Realm from 'realm';
import * as encryption from '../blue_modules/encryption';
2024-04-14 11:31:08 +02:00
import presentAlert from '../components/Alert';
2024-05-20 11:54:13 +02:00
import { randomBytes } from './rng';
import { HDAezeedWallet } from './wallets/hd-aezeed-wallet';
2024-04-14 11:31:08 +02:00
import { HDLegacyBreadwalletWallet } from './wallets/hd-legacy-breadwallet-wallet';
import { HDLegacyElectrumSeedP2PKHWallet } from './wallets/hd-legacy-electrum-seed-p2pkh-wallet';
2024-05-20 11:54:13 +02:00
import { HDLegacyP2PKHWallet } from './wallets/hd-legacy-p2pkh-wallet';
import { HDSegwitBech32Wallet } from './wallets/hd-segwit-bech32-wallet';
2024-04-14 11:31:08 +02:00
import { HDSegwitElectrumSeedP2WPKHWallet } from './wallets/hd-segwit-electrum-seed-p2wpkh-wallet';
2024-05-20 11:54:13 +02:00
import { HDSegwitP2SHWallet } from './wallets/hd-segwit-p2sh-wallet';
import { LegacyWallet } from './wallets/legacy-wallet';
import { LightningCustodianWallet } from './wallets/lightning-custodian-wallet';
import { MultisigHDWallet } from './wallets/multisig-hd-wallet';
import { SegwitBech32Wallet } from './wallets/segwit-bech32-wallet';
import { SegwitP2SHWallet } from './wallets/segwit-p2sh-wallet';
2024-04-14 11:31:08 +02:00
import { SLIP39LegacyP2PKHWallet, SLIP39SegwitBech32Wallet, SLIP39SegwitP2SHWallet } from './wallets/slip39-wallets';
2024-05-20 11:54:13 +02:00
import { ExtendedTransaction, Transaction, TWallet } from './wallets/types';
import { WatchOnlyWallet } from './wallets/watch-only-wallet';
import { getLNDHub } from '../helpers/lndHub';
2024-04-14 11:31:08 +02:00
let usedBucketNum: boolean | number = false;
2023-04-21 17:39:12 +02:00
let savingInProgress = 0; // its both a flag and a counter of attempts to write to disk
2024-04-14 11:31:08 +02:00
export type TTXMetadata = {
[txid: string]: {
memo?: string;
2024-05-08 20:08:52 +02:00
};
};
export type TCounterpartyMetadata = {
/**
* our contact identifier, such as bip47 payment code
*/
[counterparty: string]: {
/**
* custom human-readable name we assign ourselves
*/
label: string;
2024-06-12 00:18:13 +02:00
/**
* some counterparties cannot be deleted because they sent a notif tx onchain, so we just mark them as hidden when user deletes
*/
hidden?: boolean;
2024-04-14 11:31:08 +02:00
};
};
type TRealmTransaction = {
internal: boolean;
index: number;
tx: string;
};
2024-04-29 15:01:46 +02:00
type TBucketStorage = {
wallets: string[]; // array of serialized wallets, not actual wallet objects
tx_metadata: TTXMetadata;
2024-05-08 20:08:52 +02:00
counterparty_metadata: TCounterpartyMetadata;
2024-04-29 15:01:46 +02:00
};
2024-04-14 11:31:08 +02:00
const isReactNative = typeof navigator !== 'undefined' && navigator?.product === 'ReactNative';
2024-04-15 22:53:44 +02:00
export class BlueApp {
2023-04-21 17:39:12 +02:00
static FLAG_ENCRYPTED = 'data_encrypted';
static LNDHUB = 'lndhub';
static DO_NOT_TRACK = 'donottrack';
static HANDOFF_STORAGE_KEY = 'HandOff';
2024-04-15 22:53:44 +02:00
private static _instance: BlueApp | null = null;
static keys2migrate = [BlueApp.HANDOFF_STORAGE_KEY, BlueApp.DO_NOT_TRACK];
2024-04-14 11:31:08 +02:00
public cachedPassword?: false | string;
public tx_metadata: TTXMetadata;
2024-05-08 20:08:52 +02:00
public counterparty_metadata: TCounterpartyMetadata;
2024-04-14 11:31:08 +02:00
public wallets: TWallet[];
2023-04-21 17:39:12 +02:00
constructor() {
this.wallets = [];
this.tx_metadata = {};
2024-05-08 20:08:52 +02:00
this.counterparty_metadata = {};
2023-04-21 17:39:12 +02:00
this.cachedPassword = false;
}
2024-04-15 22:53:44 +02:00
static getInstance(): BlueApp {
if (!BlueApp._instance) {
BlueApp._instance = new BlueApp();
}
return BlueApp._instance;
}
2023-04-21 17:39:12 +02:00
async migrateKeys() {
2024-04-14 11:31:08 +02:00
// do not migrate keys if we are not in RN env
if (!isReactNative) {
return;
}
2024-04-15 22:53:44 +02:00
for (const key of BlueApp.keys2migrate) {
2023-04-21 17:39:12 +02:00
try {
const value = await RNSecureKeyStore.get(key);
if (value) {
await AsyncStorage.setItem(key, value);
await RNSecureKeyStore.remove(key);
}
} catch (_) {}
}
}
/**
* Wrapper for storage call. Secure store works only in RN environment. AsyncStorage is
* used for cli/tests
*/
2024-04-14 11:31:08 +02:00
setItem = (key: string, value: any): Promise<any> => {
if (isReactNative) {
2023-04-21 17:39:12 +02:00
return RNSecureKeyStore.set(key, value, { accessible: ACCESSIBLE.WHEN_UNLOCKED_THIS_DEVICE_ONLY });
} else {
return AsyncStorage.setItem(key, value);
}
};
/**
* Wrapper for storage call. Secure store works only in RN environment. AsyncStorage is
* used for cli/tests
*/
2024-04-14 11:31:08 +02:00
getItem = (key: string): Promise<any> => {
if (isReactNative) {
2023-04-21 17:39:12 +02:00
return RNSecureKeyStore.get(key);
} else {
return AsyncStorage.getItem(key);
}
};
2024-04-14 11:31:08 +02:00
getItemWithFallbackToRealm = async (key: string): Promise<any | null> => {
2023-04-21 17:39:12 +02:00
let value;
try {
return await this.getItem(key);
2024-04-14 11:31:08 +02:00
} catch (error: any) {
2023-04-21 17:39:12 +02:00
console.warn('error reading', key, error.message);
console.warn('fallback to realm');
const realmKeyValue = await this.openRealmKeyValue();
const obj = realmKeyValue.objectForPrimaryKey('KeyValue', key); // search for a realm object with a primary key
value = obj?.value;
realmKeyValue.close();
if (value) {
2024-04-14 11:31:08 +02:00
// @ts-ignore value.length
2023-04-21 17:39:12 +02:00
console.warn('successfully recovered', value.length, 'bytes from realm for key', key);
return value;
}
return null;
}
};
2024-04-14 11:31:08 +02:00
storageIsEncrypted = async (): Promise<boolean> => {
2023-04-21 17:39:12 +02:00
let data;
try {
2024-04-15 22:53:44 +02:00
data = await this.getItemWithFallbackToRealm(BlueApp.FLAG_ENCRYPTED);
2024-04-14 11:31:08 +02:00
} catch (error: any) {
2024-04-15 22:53:44 +02:00
console.warn('error reading `' + BlueApp.FLAG_ENCRYPTED + '` key:', error.message);
2023-04-21 17:39:12 +02:00
return false;
}
2024-04-14 11:31:08 +02:00
return Boolean(data);
2023-04-21 17:39:12 +02:00
};
2024-04-14 11:31:08 +02:00
isPasswordInUse = async (password: string) => {
2023-04-21 17:39:12 +02:00
try {
let data = await this.getItem('data');
data = this.decryptData(data, password);
2024-04-14 11:31:08 +02:00
return Boolean(data);
2023-04-21 17:39:12 +02:00
} catch (_e) {
return false;
}
};
/**
* Iterates through all values of `data` trying to
* decrypt each one, and returns first one successfully decrypted
*/
2024-04-14 11:31:08 +02:00
decryptData(data: string, password: string): boolean | string {
2023-04-21 17:39:12 +02:00
data = JSON.parse(data);
let decrypted;
let num = 0;
for (const value of data) {
decrypted = encryption.decrypt(value, password);
if (decrypted) {
usedBucketNum = num;
return decrypted;
}
num++;
}
return false;
}
2024-04-14 11:31:08 +02:00
decryptStorage = async (password: string): Promise<boolean> => {
2023-04-21 17:39:12 +02:00
if (password === this.cachedPassword) {
this.cachedPassword = undefined;
await this.saveToDisk();
this.wallets = [];
2024-04-14 11:31:08 +02:00
this.tx_metadata = {};
2024-05-08 20:08:52 +02:00
this.counterparty_metadata = {};
2023-04-21 17:39:12 +02:00
return this.loadFromDisk();
} else {
throw new Error('Incorrect password. Please, try again.');
}
};
2024-04-14 11:31:08 +02:00
encryptStorage = async (password: string): Promise<void> => {
2023-04-21 17:39:12 +02:00
// assuming the storage is not yet encrypted
await this.saveToDisk();
let data = await this.getItem('data');
// TODO: refactor ^^^ (should not save & load to fetch data)
const encrypted = encryption.encrypt(data, password);
data = [];
data.push(encrypted); // putting in array as we might have many buckets with storages
data = JSON.stringify(data);
this.cachedPassword = password;
await this.setItem('data', data);
2024-04-15 22:53:44 +02:00
await this.setItem(BlueApp.FLAG_ENCRYPTED, '1');
2023-04-21 17:39:12 +02:00
};
/**
* Cleans up all current application data (wallets, tx metadata etc)
* Encrypts the bucket and saves it storage
*/
2024-04-14 11:31:08 +02:00
createFakeStorage = async (fakePassword: string): Promise<boolean> => {
2023-04-21 17:39:12 +02:00
usedBucketNum = false; // resetting currently used bucket so we wont overwrite it
this.wallets = [];
this.tx_metadata = {};
2024-05-08 20:08:52 +02:00
this.counterparty_metadata = {};
2023-04-21 17:39:12 +02:00
2024-04-29 15:01:46 +02:00
const data: TBucketStorage = {
2023-04-21 17:39:12 +02:00
wallets: [],
tx_metadata: {},
2024-05-08 20:08:52 +02:00
counterparty_metadata: {},
2023-04-21 17:39:12 +02:00
};
let buckets = await this.getItem('data');
buckets = JSON.parse(buckets);
buckets.push(encryption.encrypt(JSON.stringify(data), fakePassword));
this.cachedPassword = fakePassword;
const bucketsString = JSON.stringify(buckets);
await this.setItem('data', bucketsString);
return (await this.getItem('data')) === bucketsString;
};
2024-04-14 11:31:08 +02:00
hashIt = (s: string): string => {
2023-04-21 17:39:12 +02:00
return createHash('sha256').update(s).digest().toString('hex');
};
/**
* Returns instace of the Realm database, which is encrypted either by cached user's password OR default password.
* Database file is deterministically derived from encryption key.
*/
2024-04-29 15:01:46 +02:00
async getRealmForTransactions() {
2024-04-14 11:31:08 +02:00
const cacheFolderPath = RNFS.CachesDirectoryPath; // Path to cache folder
2023-04-21 17:39:12 +02:00
const password = this.hashIt(this.cachedPassword || 'fyegjitkyf[eqjnc.lf');
const buf = Buffer.from(this.hashIt(password) + this.hashIt(password), 'hex');
const encryptionKey = Int8Array.from(buf);
2024-04-14 11:31:08 +02:00
const fileName = this.hashIt(this.hashIt(password)) + '-wallettransactions.realm';
const path = `${cacheFolderPath}/${fileName}`; // Use cache folder path
2023-04-21 17:39:12 +02:00
const schema = [
{
name: 'WalletTransactions',
properties: {
walletid: { type: 'string', indexed: true },
internal: 'bool?', // true - internal, false - external
index: 'int?',
tx: 'string', // stringified json
},
},
];
2024-04-14 11:31:08 +02:00
// @ts-ignore schema doesn't match Realm's schema type
2023-04-21 17:39:12 +02:00
return Realm.open({
2024-04-14 11:31:08 +02:00
// @ts-ignore schema doesn't match Realm's schema type
2023-04-21 17:39:12 +02:00
schema,
path,
encryptionKey,
});
}
/**
2024-04-29 15:01:46 +02:00
* Returns instace of the Realm database, which is encrypted by random bytes stored in keychain.
2023-04-21 17:39:12 +02:00
* Database file is static.
*
* @returns {Promise<Realm>}
*/
2024-04-14 11:31:08 +02:00
async openRealmKeyValue(): Promise<Realm> {
const cacheFolderPath = RNFS.CachesDirectoryPath; // Path to cache folder
2023-04-21 17:39:12 +02:00
const service = 'realm_encryption_key';
let password;
const credentials = await Keychain.getGenericPassword({ service });
if (credentials) {
password = credentials.password;
} else {
const buf = await randomBytes(64);
password = buf.toString('hex');
await Keychain.setGenericPassword(service, password, { service });
}
const buf = Buffer.from(password, 'hex');
const encryptionKey = Int8Array.from(buf);
2024-04-14 11:31:08 +02:00
const path = `${cacheFolderPath}/keyvalue.realm`; // Use cache folder path
2023-04-21 17:39:12 +02:00
const schema = [
{
name: 'KeyValue',
primaryKey: 'key',
properties: {
key: { type: 'string', indexed: true },
value: 'string', // stringified json, or whatever
},
},
];
2024-04-14 11:31:08 +02:00
// @ts-ignore schema doesn't match Realm's schema type
2023-04-21 17:39:12 +02:00
return Realm.open({
2024-04-14 11:31:08 +02:00
// @ts-ignore schema doesn't match Realm's schema type
2023-04-21 17:39:12 +02:00
schema,
path,
encryptionKey,
});
}
2024-04-14 11:31:08 +02:00
saveToRealmKeyValue(realmkeyValue: Realm, key: string, value: any) {
2023-04-21 17:39:12 +02:00
realmkeyValue.write(() => {
realmkeyValue.create(
'KeyValue',
{
key,
value,
},
Realm.UpdateMode.Modified,
);
});
}
/**
* Loads from storage all wallets and
* maps them to `this.wallets`
*
* @param password If present means storage must be decrypted before usage
* @returns {Promise.<boolean>}
*/
2024-04-14 11:31:08 +02:00
async loadFromDisk(password?: string): Promise<boolean> {
2024-08-17 15:37:30 +02:00
// Wrap inside a try so if anything goes wrong it wont block loadFromDisk from continuing
2024-04-14 11:31:08 +02:00
try {
await this.moveRealmFilesToCacheDirectory();
} catch (error: any) {
console.warn('moveRealmFilesToCacheDirectory error:', error.message);
}
2024-04-29 15:01:46 +02:00
let dataRaw = await this.getItemWithFallbackToRealm('data');
2023-04-21 17:39:12 +02:00
if (password) {
2024-04-29 15:01:46 +02:00
dataRaw = this.decryptData(dataRaw, password);
if (dataRaw) {
2023-04-21 17:39:12 +02:00
// password is good, cache it
this.cachedPassword = password;
}
}
2024-04-29 15:01:46 +02:00
if (dataRaw !== null) {
2023-04-21 17:39:12 +02:00
let realm;
try {
2024-04-29 15:01:46 +02:00
realm = await this.getRealmForTransactions();
2024-04-14 11:31:08 +02:00
} catch (error: any) {
presentAlert({ message: error.message });
2023-04-21 17:39:12 +02:00
}
2024-04-29 15:01:46 +02:00
const data: TBucketStorage = JSON.parse(dataRaw);
2023-04-21 17:39:12 +02:00
if (!data.wallets) return false;
const wallets = data.wallets;
for (const key of wallets) {
2024-08-17 15:37:30 +02:00
// deciding which type is wallet and instantiating correct object
2023-04-21 17:39:12 +02:00
const tempObj = JSON.parse(key);
2024-04-14 11:31:08 +02:00
let unserializedWallet: TWallet;
2024-08-17 15:37:30 +02:00
switch (tempObj.type) {
case SegwitBech32Wallet.type:
unserializedWallet = SegwitBech32Wallet.fromJson(key) as unknown as SegwitBech32Wallet;
break;
case SegwitP2SHWallet.type:
unserializedWallet = SegwitP2SHWallet.fromJson(key) as unknown as SegwitP2SHWallet;
break;
case WatchOnlyWallet.type:
unserializedWallet = WatchOnlyWallet.fromJson(key) as unknown as WatchOnlyWallet;
unserializedWallet.init();
if (unserializedWallet.isHd() && !unserializedWallet.isXpubValid()) {
continue;
}
break;
case HDLegacyP2PKHWallet.type:
unserializedWallet = HDLegacyP2PKHWallet.fromJson(key) as unknown as HDLegacyP2PKHWallet;
break;
case HDSegwitP2SHWallet.type:
unserializedWallet = HDSegwitP2SHWallet.fromJson(key) as unknown as HDSegwitP2SHWallet;
break;
case HDSegwitBech32Wallet.type:
unserializedWallet = HDSegwitBech32Wallet.fromJson(key) as unknown as HDSegwitBech32Wallet;
break;
case HDLegacyBreadwalletWallet.type:
unserializedWallet = HDLegacyBreadwalletWallet.fromJson(key) as unknown as HDLegacyBreadwalletWallet;
break;
case HDLegacyElectrumSeedP2PKHWallet.type:
unserializedWallet = HDLegacyElectrumSeedP2PKHWallet.fromJson(key) as unknown as HDLegacyElectrumSeedP2PKHWallet;
break;
case HDSegwitElectrumSeedP2WPKHWallet.type:
unserializedWallet = HDSegwitElectrumSeedP2WPKHWallet.fromJson(key) as unknown as HDSegwitElectrumSeedP2WPKHWallet;
break;
case MultisigHDWallet.type:
unserializedWallet = MultisigHDWallet.fromJson(key) as unknown as MultisigHDWallet;
break;
case HDAezeedWallet.type:
unserializedWallet = HDAezeedWallet.fromJson(key) as unknown as HDAezeedWallet;
// migrate password to this.passphrase field
// remove this code somewhere in year 2022
if (unserializedWallet.secret.includes(':')) {
const [mnemonic, passphrase] = unserializedWallet.secret.split(':');
unserializedWallet.secret = mnemonic;
unserializedWallet.passphrase = passphrase;
}
2023-04-21 17:39:12 +02:00
2024-08-17 15:37:30 +02:00
break;
case SLIP39SegwitP2SHWallet.type:
unserializedWallet = SLIP39SegwitP2SHWallet.fromJson(key) as unknown as SLIP39SegwitP2SHWallet;
break;
case SLIP39LegacyP2PKHWallet.type:
unserializedWallet = SLIP39LegacyP2PKHWallet.fromJson(key) as unknown as SLIP39LegacyP2PKHWallet;
break;
case SLIP39SegwitBech32Wallet.type:
unserializedWallet = SLIP39SegwitBech32Wallet.fromJson(key) as unknown as SLIP39SegwitBech32Wallet;
break;
case LightningCustodianWallet.type: {
unserializedWallet = LightningCustodianWallet.fromJson(key) as unknown as LightningCustodianWallet;
let lndhub: false | any = false;
try {
lndhub = await getLNDHub();
2024-08-17 15:37:30 +02:00
} catch (error) {
console.warn(error);
2023-04-21 17:39:12 +02:00
}
2024-08-17 15:37:30 +02:00
if (unserializedWallet.baseURI) {
unserializedWallet.setBaseURI(unserializedWallet.baseURI); // not really necessary, just for the sake of readability
console.log('using saved uri for for ln wallet:', unserializedWallet.baseURI);
} else if (lndhub) {
console.log('using wallet-wide settings ', lndhub, 'for ln wallet');
unserializedWallet.setBaseURI(lndhub);
} else {
console.log('wallet does not have a baseURI. Continuing init...');
}
unserializedWallet.init();
break;
2023-04-21 17:39:12 +02:00
}
case 'lightningLdk':
// since ldk wallets are deprecated and removed, we need to handle a case when such wallet still exists in storage
unserializedWallet = new HDSegwitBech32Wallet();
unserializedWallet.setSecret(tempObj.secret.replace('ldk://', ''));
break;
2024-08-17 15:37:30 +02:00
case LegacyWallet.type:
default:
unserializedWallet = LegacyWallet.fromJson(key) as unknown as LegacyWallet;
break;
2023-04-21 17:39:12 +02:00
}
try {
if (realm) this.inflateWalletFromRealm(realm, unserializedWallet);
2024-04-14 11:31:08 +02:00
} catch (error: any) {
presentAlert({ message: error.message });
2023-04-21 17:39:12 +02:00
}
2024-08-17 15:37:30 +02:00
// done
2023-04-21 17:39:12 +02:00
const ID = unserializedWallet.getID();
if (!this.wallets.some(wallet => wallet.getID() === ID)) {
this.wallets.push(unserializedWallet);
this.tx_metadata = data.tx_metadata;
2024-05-08 20:08:52 +02:00
this.counterparty_metadata = data.counterparty_metadata;
2023-04-21 17:39:12 +02:00
}
}
if (realm) realm.close();
return true;
} else {
2024-08-17 15:37:30 +02:00
return false; // failed loading data or loading/decryptin data
2023-04-21 17:39:12 +02:00
}
}
/**
* Lookup wallet in list by it's secret and
* remove it from `this.wallets`
*
* @param wallet {AbstractWallet}
*/
2024-04-14 11:31:08 +02:00
deleteWallet = (wallet: TWallet): void => {
2023-04-21 17:39:12 +02:00
const ID = wallet.getID();
const tempWallets = [];
for (const value of this.wallets) {
if (value.getID() === ID) {
// the one we should delete
// nop
} else {
// the one we must keep
tempWallets.push(value);
}
}
this.wallets = tempWallets;
};
2024-04-14 11:31:08 +02:00
inflateWalletFromRealm(realm: Realm, walletToInflate: TWallet) {
2023-04-21 17:39:12 +02:00
const transactions = realm.objects('WalletTransactions');
2024-04-14 11:31:08 +02:00
const transactionsForWallet = transactions.filtered(`walletid = "${walletToInflate.getID()}"`) as unknown as TRealmTransaction[];
2023-04-21 17:39:12 +02:00
for (const tx of transactionsForWallet) {
if (tx.internal === false) {
2024-04-14 11:31:08 +02:00
if ('_hdWalletInstance' in walletToInflate && walletToInflate._hdWalletInstance) {
const hd = walletToInflate._hdWalletInstance;
hd._txs_by_external_index[tx.index] = hd._txs_by_external_index[tx.index] || [];
const transaction = JSON.parse(tx.tx);
hd._txs_by_external_index[tx.index].push(transaction);
2023-04-21 17:39:12 +02:00
} else {
walletToInflate._txs_by_external_index[tx.index] = walletToInflate._txs_by_external_index[tx.index] || [];
2024-04-14 11:31:08 +02:00
const transaction = JSON.parse(tx.tx);
(walletToInflate._txs_by_external_index[tx.index] as Transaction[]).push(transaction);
2023-04-21 17:39:12 +02:00
}
} else if (tx.internal === true) {
2024-04-14 11:31:08 +02:00
if ('_hdWalletInstance' in walletToInflate && walletToInflate._hdWalletInstance) {
const hd = walletToInflate._hdWalletInstance;
hd._txs_by_internal_index[tx.index] = hd._txs_by_internal_index[tx.index] || [];
const transaction = JSON.parse(tx.tx);
hd._txs_by_internal_index[tx.index].push(transaction);
2023-04-21 17:39:12 +02:00
} else {
walletToInflate._txs_by_internal_index[tx.index] = walletToInflate._txs_by_internal_index[tx.index] || [];
2024-04-14 11:31:08 +02:00
const transaction = JSON.parse(tx.tx);
(walletToInflate._txs_by_internal_index[tx.index] as Transaction[]).push(transaction);
2023-04-21 17:39:12 +02:00
}
} else {
if (!Array.isArray(walletToInflate._txs_by_external_index)) walletToInflate._txs_by_external_index = [];
walletToInflate._txs_by_external_index = walletToInflate._txs_by_external_index || [];
2024-04-14 11:31:08 +02:00
const transaction = JSON.parse(tx.tx);
(walletToInflate._txs_by_external_index as Transaction[]).push(transaction);
2023-04-21 17:39:12 +02:00
}
}
}
2024-04-14 11:31:08 +02:00
offloadWalletToRealm(realm: Realm, wallet: TWallet): void {
2023-04-21 17:39:12 +02:00
const id = wallet.getID();
2024-04-14 11:31:08 +02:00
const walletToSave = ('_hdWalletInstance' in wallet && wallet._hdWalletInstance) || wallet;
2023-04-21 17:39:12 +02:00
if (Array.isArray(walletToSave._txs_by_external_index)) {
// if this var is an array that means its a single-address wallet class, and this var is a flat array
// with transactions
realm.write(() => {
// cleanup all existing transactions for the wallet first
const walletTransactionsToDelete = realm.objects('WalletTransactions').filtered(`walletid = '${id}'`);
realm.delete(walletTransactionsToDelete);
2024-04-14 11:31:08 +02:00
// @ts-ignore walletToSave._txs_by_external_index is array
2023-04-21 17:39:12 +02:00
for (const tx of walletToSave._txs_by_external_index) {
realm.create(
'WalletTransactions',
{
walletid: id,
tx: JSON.stringify(tx),
},
Realm.UpdateMode.Modified,
);
}
});
return;
}
/// ########################################################################################################
if (walletToSave._txs_by_external_index) {
realm.write(() => {
// cleanup all existing transactions for the wallet first
const walletTransactionsToDelete = realm.objects('WalletTransactions').filtered(`walletid = '${id}'`);
realm.delete(walletTransactionsToDelete);
// insert new ones:
for (const index of Object.keys(walletToSave._txs_by_external_index)) {
2024-04-14 11:31:08 +02:00
// @ts-ignore index is number
2023-04-21 17:39:12 +02:00
const txs = walletToSave._txs_by_external_index[index];
for (const tx of txs) {
realm.create(
'WalletTransactions',
{
walletid: id,
internal: false,
index: parseInt(index, 10),
2023-04-21 17:39:12 +02:00
tx: JSON.stringify(tx),
},
Realm.UpdateMode.Modified,
);
}
}
for (const index of Object.keys(walletToSave._txs_by_internal_index)) {
2024-04-14 11:31:08 +02:00
// @ts-ignore index is number
2023-04-21 17:39:12 +02:00
const txs = walletToSave._txs_by_internal_index[index];
for (const tx of txs) {
realm.create(
'WalletTransactions',
{
walletid: id,
internal: true,
index: parseInt(index, 10),
2023-04-21 17:39:12 +02:00
tx: JSON.stringify(tx),
},
Realm.UpdateMode.Modified,
);
}
}
});
}
}
/**
* Serializes and saves to storage object data.
* If cached password is saved - finds the correct bucket
* to save to, encrypts and then saves.
*
* @returns {Promise} Result of storage save
*/
2024-04-14 11:31:08 +02:00
async saveToDisk(): Promise<void> {
2023-04-21 17:39:12 +02:00
if (savingInProgress) {
console.warn('saveToDisk is in progress');
if (++savingInProgress > 10) presentAlert({ message: 'Critical error. Last actions were not saved' }); // should never happen
2023-04-21 17:39:12 +02:00
await new Promise(resolve => setTimeout(resolve, 1000 * savingInProgress)); // sleep
return this.saveToDisk();
}
savingInProgress = 1;
try {
2024-04-29 15:01:46 +02:00
const walletsToSave: string[] = []; // serialized wallets
2023-04-21 17:39:12 +02:00
let realm;
try {
2024-04-29 15:01:46 +02:00
realm = await this.getRealmForTransactions();
2024-04-14 11:31:08 +02:00
} catch (error: any) {
presentAlert({ message: error.message });
2023-04-21 17:39:12 +02:00
}
for (const key of this.wallets) {
if (typeof key === 'boolean') continue;
key.prepareForSerialization();
2024-04-14 11:31:08 +02:00
// @ts-ignore wtf is wallet.current? Does it even exist?
2023-04-21 17:39:12 +02:00
delete key.current;
const keyCloned = Object.assign({}, key); // stripped-down version of a wallet to save to secure keystore
2024-04-14 11:31:08 +02:00
if ('_hdWalletInstance' in key) {
const k = keyCloned as any & WatchOnlyWallet;
k._hdWalletInstance = Object.assign({}, key._hdWalletInstance);
k._hdWalletInstance._txs_by_external_index = {};
k._hdWalletInstance._txs_by_internal_index = {};
}
2023-04-21 17:39:12 +02:00
if (realm) this.offloadWalletToRealm(realm, key);
// stripping down:
if (key._txs_by_external_index) {
keyCloned._txs_by_external_index = {};
keyCloned._txs_by_internal_index = {};
}
2024-04-14 11:31:08 +02:00
if ('_bip47_instance' in keyCloned) {
2023-04-21 17:39:12 +02:00
delete keyCloned._bip47_instance; // since it wont be restored into a proper class instance
}
walletsToSave.push(JSON.stringify({ ...keyCloned, type: keyCloned.type }));
}
if (realm) realm.close();
2024-04-29 15:01:46 +02:00
let data: TBucketStorage | string[] /* either a bucket, or an array of encrypted buckets */ = {
2023-04-21 17:39:12 +02:00
wallets: walletsToSave,
tx_metadata: this.tx_metadata,
2024-05-08 20:08:52 +02:00
counterparty_metadata: this.counterparty_metadata,
2023-04-21 17:39:12 +02:00
};
if (this.cachedPassword) {
// should find the correct bucket, encrypt and then save
let buckets = await this.getItemWithFallbackToRealm('data');
buckets = JSON.parse(buckets);
2024-04-29 15:01:46 +02:00
const newData: string[] = []; // serialized buckets
2023-04-21 17:39:12 +02:00
let num = 0;
for (const bucket of buckets) {
let decrypted;
// if we had `usedBucketNum` during loadFromDisk(), no point to try to decode each bucket to find the one we
// need, we just to find bucket with the same index
if (usedBucketNum !== false) {
if (num === usedBucketNum) {
decrypted = true;
}
num++;
} else {
// we dont have `usedBucketNum` for whatever reason, so lets try to decrypt each bucket after bucket
// till we find the right one
decrypted = encryption.decrypt(bucket, this.cachedPassword);
}
if (!decrypted) {
// no luck decrypting, its not our bucket
newData.push(bucket);
} else {
// decrypted ok, this is our bucket
// we serialize our object's data, encrypt it, and add it to buckets
newData.push(encryption.encrypt(JSON.stringify(data), this.cachedPassword));
}
}
2024-04-29 15:01:46 +02:00
2023-04-21 17:39:12 +02:00
data = newData;
}
await this.setItem('data', JSON.stringify(data));
2024-04-15 22:53:44 +02:00
await this.setItem(BlueApp.FLAG_ENCRYPTED, this.cachedPassword ? '1' : '');
2023-04-21 17:39:12 +02:00
// now, backing up same data in realm:
const realmkeyValue = await this.openRealmKeyValue();
this.saveToRealmKeyValue(realmkeyValue, 'data', JSON.stringify(data));
2024-04-15 22:53:44 +02:00
this.saveToRealmKeyValue(realmkeyValue, BlueApp.FLAG_ENCRYPTED, this.cachedPassword ? '1' : '');
2023-04-21 17:39:12 +02:00
realmkeyValue.close();
2024-04-14 11:31:08 +02:00
} catch (error: any) {
2023-04-21 17:39:12 +02:00
console.error('save to disk exception:', error.message);
presentAlert({ message: 'save to disk exception: ' + error.message });
2023-04-21 17:39:12 +02:00
if (error.message.includes('Realm file decryption failed')) {
console.warn('purging realm key-value database file');
this.purgeRealmKeyValueFile();
}
} finally {
savingInProgress = 0;
}
}
/**
* For each wallet, fetches balance from remote endpoint.
* Use getter for a specific wallet to get actual balance.
* Returns void.
* If index is present then fetch only from this specific wallet
*/
2024-04-14 11:31:08 +02:00
fetchWalletBalances = async (index?: number): Promise<void> => {
2023-04-21 17:39:12 +02:00
console.log('fetchWalletBalances for wallet#', typeof index === 'undefined' ? '(all)' : index);
if (index || index === 0) {
let c = 0;
for (const wallet of this.wallets) {
if (c++ === index) {
await wallet.fetchBalance();
}
}
} else {
for (const wallet of this.wallets) {
console.log('fetching balance for', wallet.getLabel());
await wallet.fetchBalance();
}
}
};
/**
* Fetches from remote endpoint all transactions for each wallet.
* Returns void.
* To access transactions - get them from each respective wallet.
* If index is present then fetch only from this specific wallet.
*
* @param index {Integer} Index of the wallet in this.wallets array,
* blank to fetch from all wallets
* @return {Promise.<void>}
*/
2024-04-14 11:31:08 +02:00
fetchWalletTransactions = async (index?: number) => {
2023-04-21 17:39:12 +02:00
console.log('fetchWalletTransactions for wallet#', typeof index === 'undefined' ? '(all)' : index);
if (index || index === 0) {
let c = 0;
for (const wallet of this.wallets) {
if (c++ === index) {
await wallet.fetchTransactions();
2024-04-14 11:31:08 +02:00
if ('fetchPendingTransactions' in wallet) {
2023-04-21 17:39:12 +02:00
await wallet.fetchPendingTransactions();
await wallet.fetchUserInvoices();
}
}
}
} else {
for (const wallet of this.wallets) {
await wallet.fetchTransactions();
2024-04-14 11:31:08 +02:00
if ('fetchPendingTransactions' in wallet) {
2023-04-21 17:39:12 +02:00
await wallet.fetchPendingTransactions();
await wallet.fetchUserInvoices();
}
}
}
};
2024-04-14 11:31:08 +02:00
fetchSenderPaymentCodes = async (index?: number) => {
2023-04-21 17:39:12 +02:00
console.log('fetchSenderPaymentCodes for wallet#', typeof index === 'undefined' ? '(all)' : index);
if (index || index === 0) {
2024-04-14 11:31:08 +02:00
const wallet = this.wallets[index];
2023-04-21 17:39:12 +02:00
try {
2024-04-14 11:31:08 +02:00
if (!(wallet.allowBIP47() && wallet.isBIP47Enabled() && 'fetchBIP47SenderPaymentCodes' in wallet)) return;
await wallet.fetchBIP47SenderPaymentCodes();
2023-04-21 17:39:12 +02:00
} catch (error) {
console.error('Failed to fetch sender payment codes for wallet', index, error);
}
} else {
for (const wallet of this.wallets) {
try {
2024-04-14 11:31:08 +02:00
if (!(wallet.allowBIP47() && wallet.isBIP47Enabled() && 'fetchBIP47SenderPaymentCodes' in wallet)) continue;
2023-04-21 17:39:12 +02:00
await wallet.fetchBIP47SenderPaymentCodes();
} catch (error) {
console.error('Failed to fetch sender payment codes for wallet', wallet.label, error);
}
}
}
};
2024-04-14 11:31:08 +02:00
getWallets = (): TWallet[] => {
2023-04-21 17:39:12 +02:00
return this.wallets;
};
/**
* Getter for all transactions in all wallets.
* But if index is provided - only for wallet with corresponding index
*
2024-05-08 20:08:52 +02:00
* @param index {number|undefined} Wallet index in this.wallets. Empty (or undef) for all wallets.
* @param limit {number} How many txs return, starting from the earliest. Default: all of them.
* @param includeWalletsWithHideTransactionsEnabled {boolean} Wallets' _hideTransactionsInWalletsList property determines wether the user wants this wallet's txs hidden from the main list view.
2023-04-21 17:39:12 +02:00
*/
2024-04-14 11:31:08 +02:00
getTransactions = (
index?: number,
limit: number = Infinity,
includeWalletsWithHideTransactionsEnabled: boolean = false,
2024-05-08 20:08:52 +02:00
): ExtendedTransaction[] => {
2023-04-21 17:39:12 +02:00
if (index || index === 0) {
2024-04-14 11:31:08 +02:00
let txs: Transaction[] = [];
2023-04-21 17:39:12 +02:00
let c = 0;
for (const wallet of this.wallets) {
if (c++ === index) {
txs = txs.concat(wallet.getTransactions());
2024-05-08 20:08:52 +02:00
const txsRet: ExtendedTransaction[] = [];
const walletID = wallet.getID();
const walletPreferredBalanceUnit = wallet.getPreferredBalanceUnit();
txs.map(tx =>
txsRet.push({
...tx,
walletID,
walletPreferredBalanceUnit,
}),
);
return txsRet;
2023-04-21 17:39:12 +02:00
}
}
}
2024-05-08 20:08:52 +02:00
const txs: ExtendedTransaction[] = [];
2023-04-21 17:39:12 +02:00
for (const wallet of this.wallets.filter(w => includeWalletsWithHideTransactionsEnabled || !w.getHideTransactionsInWalletsList())) {
2024-05-08 20:08:52 +02:00
const walletTransactions: Transaction[] = wallet.getTransactions();
2023-04-21 17:39:12 +02:00
const walletID = wallet.getID();
2024-05-08 20:08:52 +02:00
const walletPreferredBalanceUnit = wallet.getPreferredBalanceUnit();
2023-04-21 17:39:12 +02:00
for (const t of walletTransactions) {
2024-05-08 20:08:52 +02:00
txs.push({
...t,
walletID,
walletPreferredBalanceUnit,
});
2023-04-21 17:39:12 +02:00
}
}
return txs
2024-04-14 11:31:08 +02:00
.sort((a, b) => {
const bTime = new Date(b.received!).getTime();
const aTime = new Date(a.received!).getTime();
return bTime - aTime;
2023-04-21 17:39:12 +02:00
})
.slice(0, limit);
};
/**
* Getter for a sum of all balances of all wallets
*/
2024-04-14 11:31:08 +02:00
getBalance = (): number => {
2023-04-21 17:39:12 +02:00
let finalBalance = 0;
for (const wal of this.wallets) {
finalBalance += wal.getBalance();
}
return finalBalance;
};
2024-04-14 11:31:08 +02:00
isHandoffEnabled = async (): Promise<boolean> => {
2023-04-21 17:39:12 +02:00
try {
2024-04-15 22:53:44 +02:00
return !!(await AsyncStorage.getItem(BlueApp.HANDOFF_STORAGE_KEY));
2023-04-21 17:39:12 +02:00
} catch (_) {}
return false;
};
2024-04-14 11:31:08 +02:00
setIsHandoffEnabled = async (value: boolean): Promise<void> => {
2024-04-15 22:53:44 +02:00
await AsyncStorage.setItem(BlueApp.HANDOFF_STORAGE_KEY, value ? '1' : '');
2023-04-21 17:39:12 +02:00
};
2024-04-14 11:31:08 +02:00
isDoNotTrackEnabled = async (): Promise<boolean> => {
2023-04-21 17:39:12 +02:00
try {
2024-04-15 22:53:44 +02:00
const keyExists = await AsyncStorage.getItem(BlueApp.DO_NOT_TRACK);
2024-04-14 11:31:08 +02:00
if (keyExists !== null) {
const doNotTrackValue = !!keyExists;
if (doNotTrackValue) {
await DefaultPreference.setName('group.io.bluewallet.bluewallet');
2024-04-15 22:53:44 +02:00
await DefaultPreference.set(BlueApp.DO_NOT_TRACK, '1');
AsyncStorage.removeItem(BlueApp.DO_NOT_TRACK);
2024-04-14 11:31:08 +02:00
} else {
2024-04-15 22:53:44 +02:00
return Boolean(await DefaultPreference.get(BlueApp.DO_NOT_TRACK));
2024-04-14 11:31:08 +02:00
}
}
2023-04-21 17:39:12 +02:00
} catch (_) {}
2024-04-15 22:53:44 +02:00
const doNotTrackValue = await DefaultPreference.get(BlueApp.DO_NOT_TRACK);
2024-04-14 11:31:08 +02:00
return doNotTrackValue === '1' || false;
2023-04-21 17:39:12 +02:00
};
2024-04-14 11:31:08 +02:00
setDoNotTrack = async (value: boolean) => {
2024-01-29 04:04:48 +01:00
await DefaultPreference.setName('group.io.bluewallet.bluewallet');
2024-04-14 11:31:08 +02:00
if (value) {
2024-04-15 22:53:44 +02:00
await DefaultPreference.set(BlueApp.DO_NOT_TRACK, '1');
2024-04-14 11:31:08 +02:00
} else {
2024-04-15 22:53:44 +02:00
await DefaultPreference.clear(BlueApp.DO_NOT_TRACK);
2024-04-14 11:31:08 +02:00
}
2023-04-21 17:39:12 +02:00
};
/**
* Simple async sleeper function
*/
2024-04-14 11:31:08 +02:00
sleep = (ms: number): Promise<void> => {
2023-04-21 17:39:12 +02:00
return new Promise(resolve => setTimeout(resolve, ms));
};
purgeRealmKeyValueFile() {
const path = 'keyvalue.realm';
return Realm.deleteFile({
path,
});
}
2021-02-13 00:38:29 +01:00
2024-04-14 11:31:08 +02:00
async moveRealmFilesToCacheDirectory() {
const documentPath = RNFS.DocumentDirectoryPath; // Path to documentPath folder
const cachePath = RNFS.CachesDirectoryPath; // Path to cachePath folder
2018-07-02 15:51:24 +02:00
try {
2024-04-14 11:31:08 +02:00
if (!(await RNFS.exists(documentPath))) return; // If the documentPath directory does not exist, return (nothing to move)
const files = await RNFS.readDir(documentPath); // Read all files in documentPath directory
if (Array.isArray(files) && files.length === 0) return; // If there are no files, return (nothing to move)
const appRealmFiles = files.filter(
file => file.name.endsWith('.realm') || file.name.endsWith('.realm.lock') || file.name.includes('.realm.management'),
);
2018-07-02 15:51:24 +02:00
2024-04-14 11:31:08 +02:00
for (const file of appRealmFiles) {
const filePath = `${documentPath}/${file.name}`;
const newFilePath = `${cachePath}/${file.name}`;
const fileExists = await RNFS.exists(filePath); // Check if the file exists
const cacheFileExists = await RNFS.exists(newFilePath); // Check if the file already exists in the cache directory
2018-03-31 02:03:58 +02:00
2024-04-14 11:31:08 +02:00
if (fileExists) {
if (cacheFileExists) {
await RNFS.unlink(newFilePath); // Delete the file in the cache directory if it exists
console.log(`Existing file removed from cache: ${newFilePath}`);
}
await RNFS.moveFile(filePath, newFilePath); // Move the file
console.log(`Moved Realm file: ${filePath} to ${newFilePath}`);
} else {
console.log(`File does not exist: ${filePath}`);
}
}
} catch (error) {
console.error('Error moving Realm files:', error);
throw new Error(`Error moving Realm files: ${(error as Error).message}`);
}
2018-03-31 02:03:58 +02:00
}
2024-04-14 11:31:08 +02:00
}