import 'dart:io'; import 'package:cw_bitcoin/bitcoin_mnemonic.dart'; import 'package:cw_bitcoin/bitcoin_mnemonic_is_incorrect_exception.dart'; import 'package:cw_bitcoin/bitcoin_wallet_creation_credentials.dart'; import 'package:cw_bitcoin/electrum.dart'; import 'package:cw_bitcoin/script_hash.dart'; import 'package:cw_core/node.dart'; import 'package:cw_core/unspent_coins_info.dart'; import 'package:cw_core/wallet_base.dart'; import 'package:cw_core/wallet_credentials.dart'; import 'package:cw_core/wallet_service.dart'; import 'package:cw_bitcoin/bitcoin_wallet.dart'; import 'package:cw_core/pathForWallet.dart'; import 'package:cw_core/wallet_info.dart'; import 'package:cw_core/wallet_type.dart'; import 'package:hive/hive.dart'; import 'package:collection/collection.dart'; import 'package:mobx/mobx.dart'; import 'package:bitcoin_flutter/bitcoin_flutter.dart' as bitcoin; import 'package:cw_bitcoin/bitcoin_derivations.dart'; class BitcoinWalletService extends WalletService { BitcoinWalletService(this.walletInfoSource, this.unspentCoinsInfoSource); final Box walletInfoSource; final Box unspentCoinsInfoSource; @override WalletType getType() => WalletType.bitcoin; @override Future create(BitcoinNewWalletCredentials credentials) async { // default derivation type/path for bitcoin wallets: // TODO: figure out what the default derivation type is // credentials.walletInfo!.derivationType = DerivationType.bip39; credentials.walletInfo!.derivationPath = "m/0'/1"; final wallet = await BitcoinWalletBase.create( mnemonic: await generateMnemonic(), password: credentials.password!, walletInfo: credentials.walletInfo!, unspentCoinsInfo: unspentCoinsInfoSource); await wallet.save(); await wallet.init(); return wallet; } @override Future isWalletExit(String name) async => File(await pathForWallet(name: name, type: getType())).existsSync(); @override Future openWallet(String name, String password) async { final walletInfo = walletInfoSource.values .firstWhereOrNull((info) => info.id == WalletBase.idFor(name, getType()))!; final wallet = await BitcoinWalletBase.open( password: password, name: name, walletInfo: walletInfo, unspentCoinsInfo: unspentCoinsInfoSource); await wallet.init(); return wallet; } @override Future remove(String wallet) async { File(await pathForWalletDir(name: wallet, type: getType())).delete(recursive: true); final walletInfo = walletInfoSource.values .firstWhereOrNull((info) => info.id == WalletBase.idFor(wallet, getType()))!; await walletInfoSource.delete(walletInfo.key); } @override Future rename(String currentName, String password, String newName) async { final currentWalletInfo = walletInfoSource.values .firstWhereOrNull((info) => info.id == WalletBase.idFor(currentName, getType()))!; final currentWallet = await BitcoinWalletBase.open( password: password, name: currentName, walletInfo: currentWalletInfo, unspentCoinsInfo: unspentCoinsInfoSource); await currentWallet.renameWalletFiles(newName); final newWalletInfo = currentWalletInfo; newWalletInfo.id = WalletBase.idFor(newName, getType()); newWalletInfo.name = newName; await walletInfoSource.put(currentWalletInfo.key, newWalletInfo); } @override Future restoreFromKeys(BitcoinRestoreWalletFromWIFCredentials credentials) async => throw UnimplementedError(); @override Future restoreFromSeed(BitcoinRestoreWalletFromSeedCredentials credentials) async { if (!validateMnemonic(credentials.mnemonic)) { throw BitcoinMnemonicIsIncorrectException(); } final wallet = await BitcoinWalletBase.create( password: credentials.password!, mnemonic: credentials.mnemonic, walletInfo: credentials.walletInfo!, unspentCoinsInfo: unspentCoinsInfoSource); await wallet.save(); await wallet.init(); return wallet; } static Future> compareDerivationMethods( {required mnemonic, required Node node}) async { return [DerivationType.bip39]; } static Future> getDerivationsFromMnemonic( {required String mnemonic, required Node node}) async { var list = []; final electrumClient = ElectrumClient(); await electrumClient.connectToUri(node.uri); print("@@@@@@@@@@@@@@"); for (DerivationType dType in bitcoin_derivations.keys) { if (dType == DerivationType.bip39) { for (DerivationInfo dInfo in bitcoin_derivations[dType]!) { try { print("${dInfo.derivationType.toString()} : ${dInfo.derivationPath}"); var wallet = bitcoin.HDWallet.fromSeed(await mnemonicToSeedBytes(mnemonic), network: bitcoin.bitcoin) .derivePath("m/0'/1"); // get addresses: final sh = scriptHash(wallet.address!, networkType: bitcoin.bitcoin); final balance = await electrumClient.getBalance(sh); final history = await electrumClient.getHistory(sh); print("history:"); print(history); print(history.length); dInfo.balance = balance.entries.first.value.toString(); dInfo.address = wallet.address ?? ""; dInfo.height = history.length; list.add(dInfo); } catch (e) { print(e); } } } } // default derivation path: var wallet = bitcoin.HDWallet.fromSeed(await mnemonicToSeedBytes(mnemonic), network: bitcoin.bitcoin) .derivePath("m/0'/1"); // get addresses: final sh = scriptHash(wallet.address!, networkType: bitcoin.bitcoin); final balance = await electrumClient.getBalance(sh); wallet.derive(0); print(wallet.address); print(balance.entries); print("@@@@@@@@@@@@@"); // final wallet = await BitcoinWalletBase.create( // password: "password", // mnemonic: mnemonic, // walletInfo: WalletInfo( // "id", // "test", // WalletType.bitcoin, // false, // 0, // 0, // "dirPath", // "path", // "", // null, // "yatLastUsedAddressRaw", // false, // DerivationType.bip39, // "derivationPath", // ), // unspentCoinsInfo: unspentCoinsInfoSource); list.add(DerivationInfo( derivationType: DerivationType.bip39, balance: "0.00000", address: "address", height: 0, )); return []; } static Future getInfoFromSeed({required String seed, required Node node}) async { throw UnimplementedError(); } }