import 'dart:async'; import 'dart:convert'; import 'dart:io'; import 'package:bitbox/bitbox.dart'; import 'package:bitcoin_base/bitcoin_base.dart'; import 'package:breez_sdk/breez_sdk.dart'; import 'package:breez_sdk/bridge_generated.dart'; import 'package:cw_bitcoin/bitcoin_mnemonic.dart'; import 'package:cw_bitcoin/electrum_wallet_snapshot.dart'; import 'package:cw_core/crypto_currency.dart'; import 'package:cw_core/node.dart'; import 'package:cw_core/pathForWallet.dart'; import 'package:cw_core/pending_transaction.dart'; import 'package:cw_core/sync_status.dart'; import 'package:cw_core/transaction_direction.dart'; import 'package:cw_core/unspent_coins_info.dart'; import 'package:cw_lightning/lightning_balance.dart'; import 'package:cw_lightning/lightning_transaction_info.dart'; import 'package:hive/hive.dart'; import 'package:mobx/mobx.dart'; import 'package:flutter/foundation.dart'; import 'package:bitcoin_flutter/bitcoin_flutter.dart' as bitcoin; import 'package:cw_core/wallet_info.dart'; import 'package:cw_bitcoin/bitcoin_address_record.dart'; import 'package:cw_bitcoin/bitcoin_wallet_addresses.dart'; import 'package:path_provider/path_provider.dart'; import 'package:cw_lightning/.secrets.g.dart' as secrets; import 'package:cw_bitcoin/electrum_wallet.dart'; part 'lightning_wallet.g.dart'; class LightningWallet = LightningWalletBase with _$LightningWallet; abstract class LightningWalletBase extends ElectrumWallet with Store { bool _isTransactionUpdating; @override @observable SyncStatus syncStatus; LightningWalletBase({ required String mnemonic, required String password, required WalletInfo walletInfo, required Box unspentCoinsInfo, required Uint8List seedBytes, String? addressPageType, List? initialAddresses, LightningBalance? initialBalance, Map? initialRegularAddressIndex, Map? initialChangeAddressIndex, }) : _isTransactionUpdating = false, syncStatus = NotConnectedSyncStatus(), _balance = ObservableMap(), mnemonic = mnemonic, super( password: password, walletInfo: walletInfo, unspentCoinsInfo: unspentCoinsInfo, networkType: bitcoin.bitcoin, initialAddresses: initialAddresses, initialBalance: initialBalance, seedBytes: seedBytes, currency: CryptoCurrency.btcln, ) { _balance[CryptoCurrency.btcln] = initialBalance ?? LightningBalance(confirmed: 0, unconfirmed: 0, frozen: 0); String derivationPath = walletInfo.derivationInfo!.derivationPath!; String sideDerivationPath = derivationPath.substring(0, derivationPath.length - 1) + "1"; final hd = bitcoin.HDWallet.fromSeed(seedBytes, network: networkType); walletAddresses = BitcoinWalletAddresses( walletInfo, initialAddresses: initialAddresses, initialRegularAddressIndex: initialRegularAddressIndex, initialChangeAddressIndex: initialChangeAddressIndex, mainHd: hd.derivePath(derivationPath), sideHd: hd.derivePath(sideDerivationPath), network: network, ); // initialize breez: try { setupBreez(seedBytes); } catch (e) { print("Error initializing Breez: $e"); } autorun((_) { this.walletAddresses.isEnabledAutoGenerateSubaddress = this.isEnabledAutoGenerateSubaddress; }); } late final ObservableMap _balance; StreamSubscription>? _paymentsSub; StreamSubscription? _nodeStateSub; @override @computed ObservableMap get balance => _balance; static Future create( {required String mnemonic, required String password, required WalletInfo walletInfo, required Box unspentCoinsInfo, String? addressPageType, List? initialAddresses, LightningBalance? initialBalance, Map? initialRegularAddressIndex, Map? initialChangeAddressIndex}) async { return LightningWallet( mnemonic: mnemonic, password: password, walletInfo: walletInfo, unspentCoinsInfo: unspentCoinsInfo, initialAddresses: initialAddresses, initialBalance: initialBalance, seedBytes: await Mnemonic.toSeed(mnemonic), initialRegularAddressIndex: initialRegularAddressIndex, initialChangeAddressIndex: initialChangeAddressIndex, addressPageType: addressPageType, ); } static Future open({ required String name, required WalletInfo walletInfo, required Box unspentCoinsInfo, required String password, }) async { final snp = await ElectrumWalletSnapshot.load(name, walletInfo.type, password, BitcoinNetwork.mainnet); return LightningWallet( mnemonic: snp.mnemonic!, password: password, walletInfo: walletInfo, unspentCoinsInfo: unspentCoinsInfo, initialAddresses: snp.addresses, initialBalance: LightningBalance( confirmed: snp.balance.confirmed, unconfirmed: snp.balance.unconfirmed, frozen: snp.balance.frozen, ), seedBytes: await mnemonicToSeedBytes(snp.mnemonic!), initialRegularAddressIndex: snp.regularAddressIndex, initialChangeAddressIndex: snp.changeAddressIndex, addressPageType: snp.addressPageType, ); } Future _handleNodeState(NodeState? nodeState) async { if (nodeState == null) return; _balance[CryptoCurrency.btcln] = LightningBalance( confirmed: nodeState.maxPayableMsat ~/ 1000, unconfirmed: nodeState.maxReceivableMsat ~/ 1000, frozen: 0, ); } Future _handlePayments(List payments) async { _isTransactionUpdating = true; final txs = convertToTxInfo(payments); transactionHistory.addMany(txs); _isTransactionUpdating = false; } @override Future renameWalletFiles(String newWalletName) async { await stopBreez(true); await super.renameWalletFiles(newWalletName); await setupBreez(await mnemonicToSeedBytes(mnemonic)); } Future setupBreez(Uint8List seedBytes) async { final sdk = await BreezSDK(); try { if (!(await sdk.isInitialized())) { sdk.initialize(); } } catch (e) { print("Error initializing Breez: $e"); return; } Uint8List deviceKey = base64.decode(secrets.greenlightKey); Uint8List deviceCert = base64.decode(secrets.greenlightCert); GreenlightCredentials greenlightCredentials = GreenlightCredentials( deviceKey: deviceKey, deviceCert: deviceCert, ); NodeConfig breezNodeConfig = NodeConfig.greenlight( config: GreenlightNodeConfig( partnerCredentials: greenlightCredentials, inviteCode: null, ), ); Config breezConfig = await sdk.defaultConfig( envType: EnvironmentType.Production, apiKey: secrets.breezApiKey, nodeConfig: breezNodeConfig, ); String workingDir = await pathForWalletDir(name: walletInfo.name, type: type); workingDir = "$workingDir/breez"; new Directory(workingDir).createSync(recursive: true); breezConfig = breezConfig.copyWith(workingDir: workingDir); // disconnect if already connected try { if (await sdk.isInitialized()) { await sdk.disconnect(); } } catch (e, s) { print("ERROR disconnecting from Breez: $e\n$s"); } try { await sdk.connect( req: ConnectRequest( config: breezConfig, seed: seedBytes, ), ); } catch (e, s) { print("Error connecting to Breez: $e\n$s"); } _nodeStateSub?.cancel(); _nodeStateSub = sdk.nodeStateStream.listen((event) { _handleNodeState(event); }); await _handleNodeState(await sdk.nodeInfo()); _paymentsSub?.cancel(); _paymentsSub = sdk.paymentsStream.listen((List payments) { _handlePayments(payments); }); await _handlePayments(await sdk.listPayments(req: ListPaymentsRequest())); // TODO: get actual nds service url: // if (Platform.isAndroid || Platform.isIOS) { // String platform = Platform.isAndroid ? "android" : "ios"; // String token = "TODO"; // await sdk.registerWebhook( // webhookUrl: "https://your-nds-service.com/notify?platform=$platform&token=$token", // ); // } print("initialized breez: ${(await sdk.isInitialized())}"); } Future stopBreez(bool disconnect) async { if (disconnect) { final sdk = await BreezSDK(); if (await sdk.isInitialized()) { await sdk.disconnect(); } } await _nodeStateSub?.cancel(); await _paymentsSub?.cancel(); } @action @override Future startSync() async { try { syncStatus = AttemptingSyncStatus(); await updateTransactions(); syncStatus = SyncedSyncStatus(); } catch (e) { print(e); syncStatus = FailedSyncStatus(); rethrow; } } @override Future changePassword(String password) { throw UnimplementedError("changePassword"); } @action @override Future connectToNode({required Node node}) async { try { syncStatus = ConnectingSyncStatus(); await updateTransactions(); syncStatus = ConnectedSyncStatus(); } catch (e) { print(e); syncStatus = FailedSyncStatus(); } } @override Future createTransaction(Object credentials) async { throw UnimplementedError("createTransaction"); } Future updateTransactions() async { try { if (_isTransactionUpdating) { return false; } _isTransactionUpdating = true; final transactions = await fetchTransactions(); transactionHistory.addMany(transactions); await transactionHistory.save(); _isTransactionUpdating = false; return true; } catch (_) { _isTransactionUpdating = false; return false; } } Map convertToTxInfo(List payments) { Map transactions = {}; for (Payment tx in payments) { if (tx.paymentType == PaymentType.ClosedChannel) { continue; } bool isSend = tx.paymentType == PaymentType.Sent; transactions[tx.id] = LightningTransactionInfo( isPending: false, id: tx.id, amount: tx.amountMsat ~/ 1000, fee: tx.feeMsat ~/ 1000, date: DateTime.fromMillisecondsSinceEpoch(tx.paymentTime * 1000), direction: isSend ? TransactionDirection.outgoing : TransactionDirection.incoming, ); } return transactions; } @override Future> fetchTransactions() async { final sdk = await BreezSDK(); final payments = await sdk.listPayments(req: ListPaymentsRequest()); final transactions = convertToTxInfo(payments); return transactions; } @override Future rescan({ required int height, int? chainTip, ScanData? scanData, bool? doSingleScan, }) async { updateTransactions(); } Future init() async { await walletAddresses.init(); await transactionHistory.init(); await save(); } String toJSON() => json.encode({ 'mnemonic': mnemonic, 'account_index': walletAddresses.currentReceiveAddressIndexByType, 'change_address_index': walletAddresses.currentChangeAddressIndexByType, 'addresses': walletAddresses.allAddresses.map((addr) => addr.toJSON()).toList(), 'address_page_type': walletInfo.addressPageType == null ? SegwitAddresType.p2wpkh.toString() : walletInfo.addressPageType.toString(), 'balance': balance[currency]?.toJSON(), 'network_type': network == BitcoinNetwork.testnet ? 'testnet' : 'mainnet', }); Future updateBalance() async { // balance is updated automatically } @override String mnemonic; @override String get seed => mnemonic; Future makePath() async => pathForWallet(name: walletInfo.name, type: walletInfo.type); @override Future close({bool? switchingToSameWalletType}) async { try { await electrumClient.close(); } catch (_) {} try { bool shouldDisconnect = switchingToSameWalletType == null || !switchingToSameWalletType; await stopBreez(shouldDisconnect); } catch (e, s) { print("Error stopping breez: $e\n$s"); } } }