import 'package:decimal/decimal.dart'; import 'package:flutter_native_splash/cli_commands.dart'; import 'package:isar/isar.dart'; import 'package:stackwallet/exceptions/main_db/main_db_exception.dart'; import 'package:stackwallet/models/isar/models/isar_models.dart'; import 'package:stackwallet/utilities/amount/amount.dart'; import 'package:stackwallet/utilities/enums/coin_enum.dart'; import 'package:stackwallet/utilities/stack_file_system.dart'; import 'package:tuple/tuple.dart'; part '../queries/queries.dart'; class MainDB { MainDB._(); static MainDB? _instance; static MainDB get instance => _instance ??= MainDB._(); Isar? _isar; Isar get isar => _isar!; Future initMainDB({Isar? mock}) async { if (mock != null) { _isar = mock; return true; } if (_isar != null && isar.isOpen) return false; _isar = await Isar.open( [ TransactionSchema, TransactionNoteSchema, UTXOSchema, AddressSchema, AddressLabelSchema, EthContractSchema, ], directory: (await StackFileSystem.applicationIsarDirectory()).path, // inspector: kDebugMode, inspector: false, name: "wallet_data", maxSizeMiB: 512, ); return true; } // addresses QueryBuilder getAddresses( String walletId) => isar.addresses.where().walletIdEqualTo(walletId); Future putAddress(Address address) async { try { return await isar.writeTxn(() async { return await isar.addresses.put(address); }); } catch (e) { throw MainDBException("failed putAddress: $address", e); } } Future> putAddresses(List
addresses) async { try { return await isar.writeTxn(() async { return await isar.addresses.putAll(addresses); }); } catch (e) { throw MainDBException("failed putAddresses: $addresses", e); } } Future> updateOrPutAddresses(List
addresses) async { try { List ids = []; await isar.writeTxn(() async { for (final address in addresses) { final storedAddress = await isar.addresses .getByValueWalletId(address.value, address.walletId); int id; if (storedAddress == null) { id = await isar.addresses.put(address); } else { address.id = storedAddress.id; await storedAddress.transactions.load(); final txns = storedAddress.transactions.toList(); await isar.addresses.delete(storedAddress.id); id = await isar.addresses.put(address); address.transactions.addAll(txns); await address.transactions.save(); } ids.add(id); } }); return ids; } catch (e) { throw MainDBException("failed updateOrPutAddresses: $addresses", e); } } Future getAddress(String walletId, String address) async { return isar.addresses.getByValueWalletId(address, walletId); } Future updateAddress(Address oldAddress, Address newAddress) async { try { return await isar.writeTxn(() async { newAddress.id = oldAddress.id; await oldAddress.transactions.load(); final txns = oldAddress.transactions.toList(); await isar.addresses.delete(oldAddress.id); final id = await isar.addresses.put(newAddress); newAddress.transactions.addAll(txns); await newAddress.transactions.save(); return id; }); } catch (e) { throw MainDBException( "failed updateAddress: from=$oldAddress to=$newAddress", e); } } // transactions QueryBuilder getTransactions( String walletId) => isar.transactions.where().walletIdEqualTo(walletId); Future putTransaction(Transaction transaction) async { try { return await isar.writeTxn(() async { return await isar.transactions.put(transaction); }); } catch (e) { throw MainDBException("failed putTransaction: $transaction", e); } } Future> putTransactions(List transactions) async { try { return await isar.writeTxn(() async { return await isar.transactions.putAll(transactions); }); } catch (e) { throw MainDBException("failed putTransactions: $transactions", e); } } Future getTransaction(String walletId, String txid) async { return isar.transactions.getByTxidWalletId(txid, walletId); } Stream watchTransaction({ required Id id, bool fireImmediately = false, }) { return isar.transactions.watchObject(id, fireImmediately: fireImmediately); } // utxos QueryBuilder getUTXOs(String walletId) => isar.utxos.where().walletIdEqualTo(walletId); Future putUTXO(UTXO utxo) => isar.writeTxn(() async { await isar.utxos.put(utxo); }); Future putUTXOs(List utxos) => isar.writeTxn(() async { await isar.utxos.putAll(utxos); }); Future updateUTXOs(String walletId, List utxos) async { await isar.writeTxn(() async { final set = utxos.toSet(); for (final utxo in utxos) { // check if utxo exists in db and update accordingly final storedUtxo = await isar.utxos .where() .txidWalletIdVoutEqualTo(utxo.txid, utxo.walletId, utxo.vout) .findFirst(); if (storedUtxo != null) { // update set.remove(utxo); set.add( storedUtxo.copyWith( value: utxo.value, address: utxo.address, blockTime: utxo.blockTime, blockHeight: utxo.blockHeight, blockHash: utxo.blockHash, ), ); } } await isar.utxos.where().walletIdEqualTo(walletId).deleteAll(); await isar.utxos.putAll(set.toList()); }); } Stream watchUTXO({ required Id id, bool fireImmediately = false, }) { return isar.utxos.watchObject(id, fireImmediately: fireImmediately); } // transaction notes QueryBuilder getTransactionNotes(String walletId) => isar.transactionNotes.where().walletIdEqualTo(walletId); Future putTransactionNote(TransactionNote transactionNote) => isar.writeTxn(() async { await isar.transactionNotes.put(transactionNote); }); Future putTransactionNotes(List transactionNotes) => isar.writeTxn(() async { await isar.transactionNotes.putAll(transactionNotes); }); Future getTransactionNote( String walletId, String txid) async { return isar.transactionNotes.getByTxidWalletId( txid, walletId, ); } Stream watchTransactionNote({ required Id id, bool fireImmediately = false, }) { return isar.transactionNotes .watchObject(id, fireImmediately: fireImmediately); } // address labels QueryBuilder getAddressLabels( String walletId) => isar.addressLabels.where().walletIdEqualTo(walletId); Future putAddressLabel(AddressLabel addressLabel) => isar.writeTxn(() async { return await isar.addressLabels.put(addressLabel); }); int putAddressLabelSync(AddressLabel addressLabel) => isar.writeTxnSync(() { return isar.addressLabels.putSync(addressLabel); }); Future putAddressLabels(List addressLabels) => isar.writeTxn(() async { await isar.addressLabels.putAll(addressLabels); }); Future getAddressLabel( String walletId, String addressString) async { return isar.addressLabels.getByAddressStringWalletId( addressString, walletId, ); } AddressLabel? getAddressLabelSync(String walletId, String addressString) { return isar.addressLabels.getByAddressStringWalletIdSync( addressString, walletId, ); } Stream watchAddressLabel({ required Id id, bool fireImmediately = false, }) { return isar.addressLabels.watchObject(id, fireImmediately: fireImmediately); } Future updateAddressLabel(AddressLabel addressLabel) async { try { return await isar.writeTxn(() async { return await isar.addressLabels.put(addressLabel); }); } catch (e) { throw MainDBException("failed updateAddressLabel", e); } } // Future deleteWalletBlockchainData(String walletId) async { final transactionCount = await getTransactions(walletId).count(); final addressCount = await getAddresses(walletId).count(); final utxoCount = await getUTXOs(walletId).count(); await isar.writeTxn(() async { const paginateLimit = 50; // transactions for (int i = 0; i < transactionCount; i += paginateLimit) { final txnIds = await getTransactions(walletId) .offset(i) .limit(paginateLimit) .idProperty() .findAll(); await isar.transactions.deleteAll(txnIds); } // addresses for (int i = 0; i < addressCount; i += paginateLimit) { final addressIds = await getAddresses(walletId) .offset(i) .limit(paginateLimit) .idProperty() .findAll(); await isar.addresses.deleteAll(addressIds); } // utxos for (int i = 0; i < utxoCount; i += paginateLimit) { final utxoIds = await getUTXOs(walletId) .offset(i) .limit(paginateLimit) .idProperty() .findAll(); await isar.utxos.deleteAll(utxoIds); } }); } Future deleteAddressLabels(String walletId) async { final addressLabelCount = await getAddressLabels(walletId).count(); await isar.writeTxn(() async { const paginateLimit = 50; for (int i = 0; i < addressLabelCount; i += paginateLimit) { final labelIds = await getAddressLabels(walletId) .offset(i) .limit(paginateLimit) .idProperty() .findAll(); await isar.addressLabels.deleteAll(labelIds); } }); } Future deleteTransactionNotes(String walletId) async { final noteCount = await getTransactionNotes(walletId).count(); await isar.writeTxn(() async { const paginateLimit = 50; for (int i = 0; i < noteCount; i += paginateLimit) { final labelIds = await getTransactionNotes(walletId) .offset(i) .limit(paginateLimit) .idProperty() .findAll(); await isar.transactionNotes.deleteAll(labelIds); } }); } Future addNewTransactionData( List> transactionsData, String walletId, ) async { try { await isar.writeTxn(() async { for (final data in transactionsData) { final tx = data.item1; final potentiallyUnconfirmedTx = await getTransaction( walletId, tx.txid, ); if (potentiallyUnconfirmedTx != null) { // update use id to replace tx tx.id = potentiallyUnconfirmedTx.id; await isar.transactions.delete(potentiallyUnconfirmedTx.id); } // save transaction await isar.transactions.put(tx); if (data.item2 != null) { final address = await getAddress(walletId, data.item2!.value); // check if address exists in db and add if it does not if (address == null) { await isar.addresses.put(data.item2!); } // link and save address tx.address.value = address ?? data.item2!; await tx.address.save(); } } }); } catch (e) { throw MainDBException("failed addNewTransactionData", e); } } // ========== Ethereum ======================================================= // eth contracts QueryBuilder getEthContracts() => isar.ethContracts.where(); Future getEthContract(String contractAddress) => isar.ethContracts.where().addressEqualTo(contractAddress).findFirst(); EthContract? getEthContractSync(String contractAddress) => isar.ethContracts.where().addressEqualTo(contractAddress).findFirstSync(); Future putEthContract(EthContract contract) => isar.writeTxn(() async { return await isar.ethContracts.put(contract); }); Future putEthContracts(List contracts) => isar.writeTxn(() async { await isar.ethContracts.putAll(contracts); }); }