2023-05-26 21:21:16 +00:00
/ *
* This file is part of Stack Wallet .
*
* Copyright ( c ) 2023 Cypher Stack
* All Rights Reserved .
* The code is distributed under GPLv3 license , see LICENSE file for details .
* Generated by Cypher Stack on 2023 - 05 - 26
*
* /
2022-10-27 23:24:14 +00:00
import ' dart:async ' ;
import ' dart:convert ' ;
import ' dart:io ' ;
import ' package:bech32/bech32.dart ' ;
import ' package:bip32/bip32.dart ' as bip32 ;
import ' package:bip39/bip39.dart ' as bip39 ;
import ' package:bitcoindart/bitcoindart.dart ' ;
import ' package:bs58check/bs58check.dart ' as bs58check ;
import ' package:crypto/crypto.dart ' ;
import ' package:decimal/decimal.dart ' ;
import ' package:flutter/foundation.dart ' ;
2023-01-12 00:59:01 +00:00
import ' package:isar/isar.dart ' ;
2023-03-01 21:52:13 +00:00
import ' package:stackwallet/db/isar/main_db.dart ' ;
2022-10-27 23:24:14 +00:00
import ' package:stackwallet/electrumx_rpc/cached_electrumx.dart ' ;
import ' package:stackwallet/electrumx_rpc/electrumx.dart ' ;
2023-01-12 00:59:01 +00:00
import ' package:stackwallet/models/balance.dart ' ;
import ' package:stackwallet/models/isar/models/isar_models.dart ' as isar_models ;
2022-10-27 23:24:14 +00:00
import ' package:stackwallet/models/paymint/fee_object_model.dart ' ;
2023-03-09 19:49:39 +00:00
import ' package:stackwallet/models/signing_data.dart ' ;
2022-10-27 23:24:14 +00:00
import ' package:stackwallet/services/coins/coin_service.dart ' ;
import ' package:stackwallet/services/event_bus/events/global/node_connection_status_changed_event.dart ' ;
import ' package:stackwallet/services/event_bus/events/global/refresh_percent_changed_event.dart ' ;
import ' package:stackwallet/services/event_bus/events/global/updated_in_background_event.dart ' ;
import ' package:stackwallet/services/event_bus/events/global/wallet_sync_status_changed_event.dart ' ;
import ' package:stackwallet/services/event_bus/global_event_bus.dart ' ;
2023-03-08 22:11:46 +00:00
import ' package:stackwallet/services/mixins/coin_control_interface.dart ' ;
2023-01-20 18:16:27 +00:00
import ' package:stackwallet/services/mixins/electrum_x_parsing.dart ' ;
2023-01-12 18:46:01 +00:00
import ' package:stackwallet/services/mixins/wallet_cache.dart ' ;
import ' package:stackwallet/services/mixins/wallet_db.dart ' ;
2023-04-08 00:44:43 +00:00
import ' package:stackwallet/services/mixins/xpubable.dart ' ;
2022-10-27 23:24:14 +00:00
import ' package:stackwallet/services/node_service.dart ' ;
import ' package:stackwallet/services/transaction_notification_tracker.dart ' ;
2023-04-06 21:24:56 +00:00
import ' package:stackwallet/utilities/amount/amount.dart ' ;
2023-02-03 22:34:06 +00:00
import ' package:stackwallet/utilities/bip32_utils.dart ' ;
2022-10-27 23:24:14 +00:00
import ' package:stackwallet/utilities/constants.dart ' ;
import ' package:stackwallet/utilities/default_nodes.dart ' ;
import ' package:stackwallet/utilities/enums/coin_enum.dart ' ;
2023-01-25 18:08:48 +00:00
import ' package:stackwallet/utilities/enums/derive_path_type_enum.dart ' ;
2022-10-27 23:24:14 +00:00
import ' package:stackwallet/utilities/enums/fee_rate_type_enum.dart ' ;
import ' package:stackwallet/utilities/flutter_secure_storage_interface.dart ' ;
import ' package:stackwallet/utilities/format.dart ' ;
import ' package:stackwallet/utilities/logger.dart ' ;
import ' package:stackwallet/utilities/prefs.dart ' ;
2023-05-09 17:54:15 +00:00
import ' package:stackwallet/widgets/crypto_notifications.dart ' ;
2022-10-27 23:24:14 +00:00
import ' package:tuple/tuple.dart ' ;
import ' package:uuid/uuid.dart ' ;
const int MINIMUM_CONFIRMATIONS = 1 ;
2023-04-05 22:06:31 +00:00
final Amount DUST_LIMIT = Amount (
rawValue: BigInt . from ( 294 ) ,
fractionDigits: Coin . particl . decimals ,
) ;
final Amount DUST_LIMIT_P2PKH = Amount (
rawValue: BigInt . from ( 546 ) ,
fractionDigits: Coin . particl . decimals ,
) ;
2022-10-27 23:24:14 +00:00
const String GENESIS_HASH_MAINNET =
2022-10-28 18:03:52 +00:00
" 12a765e31ffd4059bada1e25190f6e98c99d9714d334efa41a195a7e7e04bfe2 " ;
2022-10-27 23:24:14 +00:00
const String GENESIS_HASH_TESTNET =
2022-10-28 18:03:52 +00:00
" 4966625a4b2851d9fdee139e56211a0d88575f59ed816ff5e6a63deb4e3e29a0 " ;
2022-10-27 23:24:14 +00:00
2023-02-03 22:34:06 +00:00
String constructDerivePath ( {
required DerivePathType derivePathType ,
required int networkWIF ,
int account = 0 ,
required int chain ,
required int index ,
} ) {
2022-10-27 23:24:14 +00:00
String coinType ;
2023-02-03 22:34:06 +00:00
switch ( networkWIF ) {
2022-10-28 18:03:52 +00:00
case 0xb0 : // ltc mainnet wif
coinType = " 2 " ; // ltc mainnet
2022-10-27 23:24:14 +00:00
break ;
2022-10-28 18:03:52 +00:00
case 0xef : // ltc testnet wif
coinType = " 1 " ; // ltc testnet
2022-10-27 23:24:14 +00:00
break ;
default :
2023-02-03 22:34:06 +00:00
throw Exception ( " Invalid Litecoin network wif used! " ) ;
2022-10-27 23:24:14 +00:00
}
2023-02-03 22:34:06 +00:00
int purpose ;
2022-10-27 23:24:14 +00:00
switch ( derivePathType ) {
case DerivePathType . bip44:
2023-02-03 22:34:06 +00:00
purpose = 44 ;
break ;
2022-10-27 23:24:14 +00:00
case DerivePathType . bip49:
2023-02-03 22:34:06 +00:00
purpose = 49 ;
break ;
2022-10-27 23:24:14 +00:00
case DerivePathType . bip84:
2023-02-03 22:34:06 +00:00
purpose = 84 ;
break ;
2022-10-27 23:24:14 +00:00
default :
2023-02-03 22:34:06 +00:00
throw Exception ( " DerivePathType $ derivePathType not supported " ) ;
2022-10-27 23:24:14 +00:00
}
2023-02-03 22:34:06 +00:00
return " m/ $ purpose '/ $ coinType '/ $ account '/ $ chain / $ index " ;
2022-10-27 23:24:14 +00:00
}
2023-01-20 18:16:27 +00:00
class LitecoinWallet extends CoinServiceAPI
2023-04-08 00:44:43 +00:00
with WalletCache , WalletDB , ElectrumXParsing , CoinControlInterface
implements XPubAble {
2023-02-03 22:48:16 +00:00
LitecoinWallet ( {
required String walletId ,
required String walletName ,
required Coin coin ,
required ElectrumX client ,
required CachedElectrumX cachedClient ,
required TransactionNotificationTracker tracker ,
required SecureStorageInterface secureStore ,
MainDB ? mockableOverride ,
} ) {
txTracker = tracker ;
_walletId = walletId ;
_walletName = walletName ;
_coin = coin ;
_electrumXClient = client ;
_cachedElectrumXClient = cachedClient ;
_secureStore = secureStore ;
initCache ( walletId , coin ) ;
initWalletDB ( mockableOverride: mockableOverride ) ;
2023-03-08 22:11:46 +00:00
initCoinControlInterface (
walletId: walletId ,
walletName: walletName ,
coin: coin ,
db: db ,
getChainHeight: ( ) = > chainHeight ,
refreshedBalanceCallback: ( balance ) async {
_balance = balance ;
await updateCachedBalance ( _balance ! ) ;
} ,
) ;
2023-02-03 22:48:16 +00:00
}
2022-10-27 23:24:14 +00:00
static const integrationTestFlag =
bool . fromEnvironment ( " IS_INTEGRATION_TEST " ) ;
final _prefs = Prefs . instance ;
Timer ? timer ;
2023-01-12 21:32:25 +00:00
late final Coin _coin ;
2022-10-27 23:24:14 +00:00
late final TransactionNotificationTracker txTracker ;
NetworkType get _network {
switch ( coin ) {
2022-10-28 18:03:52 +00:00
case Coin . litecoin:
return litecoin ;
case Coin . litecoinTestNet:
return litecointestnet ;
2022-10-27 23:24:14 +00:00
default :
throw Exception ( " Invalid network type! " ) ;
}
}
@ override
set isFavorite ( bool markFavorite ) {
2023-01-12 21:32:25 +00:00
_isFavorite = markFavorite ;
updateCachedIsFavorite ( markFavorite ) ;
2022-10-27 23:24:14 +00:00
}
@ override
2023-01-12 21:32:25 +00:00
bool get isFavorite = > _isFavorite ? ? = getCachedIsFavorite ( ) ;
bool ? _isFavorite ;
2022-10-27 23:24:14 +00:00
@ override
Coin get coin = > _coin ;
@ override
2023-01-16 21:04:03 +00:00
Future < List < isar_models . UTXO > > get utxos = > db . getUTXOs ( walletId ) . findAll ( ) ;
2022-10-27 23:24:14 +00:00
@ override
2023-01-12 00:59:01 +00:00
Future < List < isar_models . Transaction > > get transactions = >
2023-01-16 21:04:03 +00:00
db . getTransactions ( walletId ) . sortByTimestampDesc ( ) . findAll ( ) ;
2022-10-27 23:24:14 +00:00
@ override
2023-01-12 00:59:01 +00:00
Future < String > get currentReceivingAddress async = >
( await _currentReceivingAddress ) . value ;
2023-01-23 16:32:53 +00:00
Future < isar_models . Address > get _currentReceivingAddress async = >
( await db
. getAddresses ( walletId )
. filter ( )
. typeEqualTo ( isar_models . AddressType . p2wpkh )
. subTypeEqualTo ( isar_models . AddressSubType . receiving )
. sortByDerivationIndexDesc ( )
. findFirst ( ) ) ? ?
2023-01-25 19:49:14 +00:00
await _generateAddressForChain ( 0 , 0 , DerivePathTypeExt . primaryFor ( coin ) ) ;
2023-01-12 00:59:01 +00:00
Future < String > get currentChangeAddress async = >
( await _currentChangeAddress ) . value ;
2023-01-23 16:32:53 +00:00
Future < isar_models . Address > get _currentChangeAddress async = >
( await db
. getAddresses ( walletId )
. filter ( )
. typeEqualTo ( isar_models . AddressType . p2wpkh )
. subTypeEqualTo ( isar_models . AddressSubType . change )
. sortByDerivationIndexDesc ( )
. findFirst ( ) ) ? ?
2023-01-25 19:49:14 +00:00
await _generateAddressForChain ( 1 , 0 , DerivePathTypeExt . primaryFor ( coin ) ) ;
2022-10-27 23:24:14 +00:00
@ override
Future < void > exit ( ) async {
_hasCalledExit = true ;
timer ? . cancel ( ) ;
timer = null ;
stopNetworkAlivePinging ( ) ;
}
bool _hasCalledExit = false ;
@ override
bool get hasCalledExit = > _hasCalledExit ;
@ override
Future < FeeObject > get fees = > _feeObject ? ? = _getFees ( ) ;
Future < FeeObject > ? _feeObject ;
@ override
Future < int > get maxFee async {
final fee = ( await fees ) . fast as String ;
2023-06-17 16:42:23 +00:00
final satsFee = Decimal . parse ( fee ) *
Decimal . fromInt ( Constants . satsPerCoin ( coin ) . toInt ( ) ) ;
2022-10-27 23:24:14 +00:00
return satsFee . floor ( ) . toBigInt ( ) . toInt ( ) ;
}
@ override
Future < List < String > > get mnemonic = > _getMnemonicList ( ) ;
2023-02-03 22:34:06 +00:00
@ override
Future < String ? > get mnemonicString = >
_secureStore . read ( key: ' ${ _walletId } _mnemonic ' ) ;
@ override
Future < String ? > get mnemonicPassphrase = > _secureStore . read (
key: ' ${ _walletId } _mnemonicPassphrase ' ,
) ;
2022-10-27 23:24:14 +00:00
Future < int > get chainHeight async {
try {
final result = await _electrumXClient . getBlockHeadTip ( ) ;
2023-01-12 21:32:25 +00:00
final height = result [ " height " ] as int ;
await updateCachedChainHeight ( height ) ;
2023-01-30 17:06:28 +00:00
if ( height > storedChainHeight ) {
GlobalEventBus . instance . fire (
UpdatedInBackgroundEvent (
" Updated current chain height in $ walletId $ walletName ! " ,
walletId ,
) ,
) ;
}
2023-01-12 21:32:25 +00:00
return height ;
2022-10-27 23:24:14 +00:00
} catch ( e , s ) {
Logging . instance . log ( " Exception caught in chainHeight: $ e \n $ s " ,
level: LogLevel . Error ) ;
2023-01-12 21:32:25 +00:00
return storedChainHeight ;
2022-10-27 23:24:14 +00:00
}
}
2023-01-12 00:59:01 +00:00
@ override
2023-01-12 21:32:25 +00:00
int get storedChainHeight = > getCachedChainHeight ( ) ;
2022-10-27 23:24:14 +00:00
DerivePathType addressType ( { required String address } ) {
Uint8List ? decodeBase58 ;
Segwit ? decodeBech32 ;
try {
decodeBase58 = bs58check . decode ( address ) ;
} catch ( err ) {
// Base58check decode fail
}
if ( decodeBase58 ! = null ) {
if ( decodeBase58 [ 0 ] = = _network . pubKeyHash ) {
// P2PKH
return DerivePathType . bip44 ;
}
if ( decodeBase58 [ 0 ] = = _network . scriptHash ) {
// P2SH
return DerivePathType . bip49 ;
}
throw ArgumentError ( ' Invalid version or Network mismatch ' ) ;
} else {
try {
2022-10-28 18:03:52 +00:00
decodeBech32 = segwit . decode ( address , _network . bech32 ! ) ;
2022-10-27 23:24:14 +00:00
} catch ( err ) {
// Bech32 decode fail
}
if ( _network . bech32 ! = decodeBech32 ! . hrp ) {
throw ArgumentError ( ' Invalid prefix or Network mismatch ' ) ;
}
if ( decodeBech32 . version ! = 0 ) {
throw ArgumentError ( ' Invalid address version ' ) ;
}
// P2WPKH
return DerivePathType . bip84 ;
}
}
bool longMutex = false ;
@ override
Future < void > recoverFromMnemonic ( {
required String mnemonic ,
2023-02-03 22:34:06 +00:00
String ? mnemonicPassphrase ,
2022-10-27 23:24:14 +00:00
required int maxUnusedAddressGap ,
required int maxNumberOfIndexesToCheck ,
required int height ,
} ) async {
longMutex = true ;
final start = DateTime . now ( ) ;
try {
Logging . instance . log ( " IS_INTEGRATION_TEST: $ integrationTestFlag " ,
level: LogLevel . Info ) ;
if ( ! integrationTestFlag ) {
final features = await electrumXClient . getServerFeatures ( ) ;
Logging . instance . log ( " features: $ features " , level: LogLevel . Info ) ;
switch ( coin ) {
2022-10-28 18:03:52 +00:00
case Coin . litecoin:
2022-10-27 23:24:14 +00:00
if ( features [ ' genesis_hash ' ] ! = GENESIS_HASH_MAINNET ) {
throw Exception ( " genesis hash does not match main net! " ) ;
}
break ;
2022-10-28 18:03:52 +00:00
case Coin . litecoinTestNet:
2022-10-27 23:24:14 +00:00
if ( features [ ' genesis_hash ' ] ! = GENESIS_HASH_TESTNET ) {
throw Exception ( " genesis hash does not match test net! " ) ;
}
break ;
default :
throw Exception (
2022-10-28 18:03:52 +00:00
" Attempted to generate a LitecoinWallet using a non litecoin coin type: ${ coin . name } " ) ;
2022-10-27 23:24:14 +00:00
}
// if (_networkType == BasicNetworkType.main) {
// if (features['genesis_hash'] != GENESIS_HASH_MAINNET) {
// throw Exception("genesis hash does not match main net!");
// }
// } else if (_networkType == BasicNetworkType.test) {
// if (features['genesis_hash'] != GENESIS_HASH_TESTNET) {
// throw Exception("genesis hash does not match test net!");
// }
// }
}
// check to make sure we aren't overwriting a mnemonic
// this should never fail
2023-02-03 22:34:06 +00:00
if ( ( await mnemonicString ) ! = null | |
( await this . mnemonicPassphrase ) ! = null ) {
2022-10-27 23:24:14 +00:00
longMutex = false ;
throw Exception ( " Attempted to overwrite mnemonic on restore! " ) ;
}
await _secureStore . write (
key: ' ${ _walletId } _mnemonic ' , value: mnemonic . trim ( ) ) ;
2023-02-03 22:34:06 +00:00
await _secureStore . write (
key: ' ${ _walletId } _mnemonicPassphrase ' ,
value: mnemonicPassphrase ? ? " " ,
) ;
2022-10-27 23:24:14 +00:00
await _recoverWalletFromBIP32SeedPhrase (
mnemonic: mnemonic . trim ( ) ,
2023-02-03 22:34:06 +00:00
mnemonicPassphrase: mnemonicPassphrase ? ? " " ,
2022-10-27 23:24:14 +00:00
maxUnusedAddressGap: maxUnusedAddressGap ,
maxNumberOfIndexesToCheck: maxNumberOfIndexesToCheck ,
) ;
} catch ( e , s ) {
Logging . instance . log (
" Exception rethrown from recoverFromMnemonic(): $ e \n $ s " ,
level: LogLevel . Error ) ;
longMutex = false ;
rethrow ;
}
longMutex = false ;
final end = DateTime . now ( ) ;
Logging . instance . log (
" $ walletName recovery time: ${ end . difference ( start ) . inMilliseconds } millis " ,
level: LogLevel . Info ) ;
}
Future < Map < String , dynamic > > _checkGaps (
int maxNumberOfIndexesToCheck ,
int maxUnusedAddressGap ,
int txCountBatchSize ,
bip32 . BIP32 root ,
DerivePathType type ,
2023-01-12 00:59:01 +00:00
int chain ) async {
List < isar_models . Address > addressArray = [ ] ;
2022-10-27 23:24:14 +00:00
int returningIndex = - 1 ;
Map < String , Map < String , String > > derivations = { } ;
int gapCounter = 0 ;
for ( int index = 0 ;
index < maxNumberOfIndexesToCheck & & gapCounter < maxUnusedAddressGap ;
index + = txCountBatchSize ) {
List < String > iterationsAddressArray = [ ] ;
Logging . instance . log (
2023-01-12 00:59:01 +00:00
" index: $ index , \t GapCounter $ chain ${ type . name } : $ gapCounter " ,
2022-10-27 23:24:14 +00:00
level: LogLevel . Info ) ;
final _id = " k_ $ index " ;
Map < String , String > txCountCallArgs = { } ;
final Map < String , dynamic > receivingNodes = { } ;
for ( int j = 0 ; j < txCountBatchSize ; j + + ) {
2023-02-03 22:34:06 +00:00
final derivePath = constructDerivePath (
derivePathType: type ,
networkWIF: root . network . wif ,
chain: chain ,
index: index + j ,
2022-10-27 23:24:14 +00:00
) ;
2023-02-03 22:34:06 +00:00
final node = await Bip32Utils . getBip32NodeFromRoot ( root , derivePath ) ;
2023-01-12 00:59:01 +00:00
String addressString ;
final data = PaymentData ( pubkey: node . publicKey ) ;
isar_models . AddressType addrType ;
2022-10-27 23:24:14 +00:00
switch ( type ) {
case DerivePathType . bip44:
2023-01-12 00:59:01 +00:00
addressString = P2PKH ( data: data , network: _network ) . data . address ! ;
addrType = isar_models . AddressType . p2pkh ;
2022-10-27 23:24:14 +00:00
break ;
case DerivePathType . bip49:
2023-01-12 00:59:01 +00:00
addressString = P2SH (
2022-10-27 23:24:14 +00:00
data: PaymentData (
redeem: P2WPKH (
2023-01-12 00:59:01 +00:00
data: data ,
2022-10-28 18:03:52 +00:00
network: _network ,
overridePrefix: _network . bech32 ! )
2022-10-27 23:24:14 +00:00
. data ) ,
network: _network )
. data
. address ! ;
2023-01-12 00:59:01 +00:00
addrType = isar_models . AddressType . p2sh ;
2022-10-27 23:24:14 +00:00
break ;
case DerivePathType . bip84:
2023-01-12 00:59:01 +00:00
addressString = P2WPKH (
2022-10-27 23:24:14 +00:00
network: _network ,
2023-01-12 00:59:01 +00:00
data: data ,
2022-10-28 18:03:52 +00:00
overridePrefix: _network . bech32 ! )
2022-10-27 23:24:14 +00:00
. data
. address ! ;
2023-01-12 00:59:01 +00:00
addrType = isar_models . AddressType . p2wpkh ;
2022-10-27 23:24:14 +00:00
break ;
default :
2023-02-02 15:24:26 +00:00
throw Exception ( " DerivePathType unsupported " ) ;
2022-10-27 23:24:14 +00:00
}
2023-01-12 00:59:01 +00:00
2023-01-16 22:37:00 +00:00
final address = isar_models . Address (
walletId: walletId ,
value: addressString ,
publicKey: node . publicKey ,
type: addrType ,
derivationIndex: index + j ,
2023-02-03 23:30:32 +00:00
derivationPath: isar_models . DerivationPath ( ) . . value = derivePath ,
2023-01-16 22:37:00 +00:00
subType: chain = = 0
2023-01-12 00:59:01 +00:00
? isar_models . AddressSubType . receiving
2023-01-16 22:37:00 +00:00
: isar_models . AddressSubType . change ,
) ;
2023-01-12 00:59:01 +00:00
2022-10-27 23:24:14 +00:00
receivingNodes . addAll ( {
" ${ _id } _ $ j " : {
" node " : node ,
" address " : address ,
}
} ) ;
txCountCallArgs . addAll ( {
2023-01-12 00:59:01 +00:00
" ${ _id } _ $ j " : addressString ,
2022-10-27 23:24:14 +00:00
} ) ;
}
// get address tx counts
final counts = await _getBatchTxCount ( addresses: txCountCallArgs ) ;
// check and add appropriate addresses
for ( int k = 0 ; k < txCountBatchSize ; k + + ) {
int count = counts [ " ${ _id } _ $ k " ] ! ;
if ( count > 0 ) {
final node = receivingNodes [ " ${ _id } _ $ k " ] ;
2023-01-12 00:59:01 +00:00
final address = node [ " address " ] as isar_models . Address ;
2022-10-27 23:24:14 +00:00
// add address to array
2023-01-12 00:59:01 +00:00
addressArray . add ( address ) ;
iterationsAddressArray . add ( address . value ) ;
2022-10-27 23:24:14 +00:00
// set current index
returningIndex = index + k ;
// reset counter
gapCounter = 0 ;
// add info to derivations
2023-01-13 15:26:37 +00:00
derivations [ address . value ] = {
2022-10-27 23:24:14 +00:00
" pubKey " : Format . uint8listToString (
( node [ " node " ] as bip32 . BIP32 ) . publicKey ) ,
" wif " : ( node [ " node " ] as bip32 . BIP32 ) . toWIF ( ) ,
} ;
}
// increase counter when no tx history found
if ( count = = 0 ) {
gapCounter + + ;
}
}
// cache all the transactions while waiting for the current function to finish.
unawaited ( getTransactionCacheEarly ( iterationsAddressArray ) ) ;
}
return {
" addressArray " : addressArray ,
" index " : returningIndex ,
" derivations " : derivations
} ;
}
Future < void > getTransactionCacheEarly ( List < String > allAddresses ) async {
try {
final List < Map < String , dynamic > > allTxHashes =
await _fetchHistory ( allAddresses ) ;
for ( final txHash in allTxHashes ) {
try {
unawaited ( cachedElectrumXClient . getTransaction (
txHash: txHash [ " tx_hash " ] as String ,
verbose: true ,
coin: coin ,
) ) ;
} catch ( e ) {
continue ;
}
}
} catch ( e ) {
//
}
}
Future < void > _recoverWalletFromBIP32SeedPhrase ( {
required String mnemonic ,
2023-02-03 22:34:06 +00:00
required String mnemonicPassphrase ,
2022-10-27 23:24:14 +00:00
int maxUnusedAddressGap = 20 ,
int maxNumberOfIndexesToCheck = 1000 ,
2023-02-02 15:03:57 +00:00
bool isRescan = false ,
2022-10-27 23:24:14 +00:00
} ) async {
longMutex = true ;
Map < String , Map < String , String > > p2pkhReceiveDerivations = { } ;
Map < String , Map < String , String > > p2shReceiveDerivations = { } ;
Map < String , Map < String , String > > p2wpkhReceiveDerivations = { } ;
Map < String , Map < String , String > > p2pkhChangeDerivations = { } ;
Map < String , Map < String , String > > p2shChangeDerivations = { } ;
Map < String , Map < String , String > > p2wpkhChangeDerivations = { } ;
2023-02-03 22:34:06 +00:00
final root = await Bip32Utils . getBip32Root (
mnemonic ,
mnemonicPassphrase ,
_network ,
) ;
2022-10-27 23:24:14 +00:00
2023-01-12 00:59:01 +00:00
List < isar_models . Address > p2pkhReceiveAddressArray = [ ] ;
List < isar_models . Address > p2shReceiveAddressArray = [ ] ;
List < isar_models . Address > p2wpkhReceiveAddressArray = [ ] ;
2022-10-27 23:24:14 +00:00
int p2pkhReceiveIndex = - 1 ;
int p2shReceiveIndex = - 1 ;
int p2wpkhReceiveIndex = - 1 ;
2023-01-12 00:59:01 +00:00
List < isar_models . Address > p2pkhChangeAddressArray = [ ] ;
List < isar_models . Address > p2shChangeAddressArray = [ ] ;
List < isar_models . Address > p2wpkhChangeAddressArray = [ ] ;
2022-10-27 23:24:14 +00:00
int p2pkhChangeIndex = - 1 ;
int p2shChangeIndex = - 1 ;
int p2wpkhChangeIndex = - 1 ;
// actual size is 36 due to p2pkh, p2sh, and p2wpkh so 12x3
const txCountBatchSize = 12 ;
try {
// receiving addresses
Logging . instance
. log ( " checking receiving addresses... " , level: LogLevel . Info ) ;
final resultReceive44 = _checkGaps ( maxNumberOfIndexesToCheck ,
maxUnusedAddressGap , txCountBatchSize , root , DerivePathType . bip44 , 0 ) ;
final resultReceive49 = _checkGaps ( maxNumberOfIndexesToCheck ,
maxUnusedAddressGap , txCountBatchSize , root , DerivePathType . bip49 , 0 ) ;
final resultReceive84 = _checkGaps ( maxNumberOfIndexesToCheck ,
maxUnusedAddressGap , txCountBatchSize , root , DerivePathType . bip84 , 0 ) ;
Logging . instance
. log ( " checking change addresses... " , level: LogLevel . Info ) ;
// change addresses
final resultChange44 = _checkGaps ( maxNumberOfIndexesToCheck ,
maxUnusedAddressGap , txCountBatchSize , root , DerivePathType . bip44 , 1 ) ;
final resultChange49 = _checkGaps ( maxNumberOfIndexesToCheck ,
maxUnusedAddressGap , txCountBatchSize , root , DerivePathType . bip49 , 1 ) ;
final resultChange84 = _checkGaps ( maxNumberOfIndexesToCheck ,
maxUnusedAddressGap , txCountBatchSize , root , DerivePathType . bip84 , 1 ) ;
await Future . wait ( [
resultReceive44 ,
resultReceive49 ,
resultReceive84 ,
resultChange44 ,
resultChange49 ,
resultChange84
] ) ;
p2pkhReceiveAddressArray =
2023-01-12 00:59:01 +00:00
( await resultReceive44 ) [ ' addressArray ' ] as List < isar_models . Address > ;
2022-10-27 23:24:14 +00:00
p2pkhReceiveIndex = ( await resultReceive44 ) [ ' index ' ] as int ;
p2pkhReceiveDerivations = ( await resultReceive44 ) [ ' derivations ' ]
as Map < String , Map < String , String > > ;
p2shReceiveAddressArray =
2023-01-12 00:59:01 +00:00
( await resultReceive49 ) [ ' addressArray ' ] as List < isar_models . Address > ;
2022-10-27 23:24:14 +00:00
p2shReceiveIndex = ( await resultReceive49 ) [ ' index ' ] as int ;
p2shReceiveDerivations = ( await resultReceive49 ) [ ' derivations ' ]
as Map < String , Map < String , String > > ;
p2wpkhReceiveAddressArray =
2023-01-12 00:59:01 +00:00
( await resultReceive84 ) [ ' addressArray ' ] as List < isar_models . Address > ;
2022-10-27 23:24:14 +00:00
p2wpkhReceiveIndex = ( await resultReceive84 ) [ ' index ' ] as int ;
p2wpkhReceiveDerivations = ( await resultReceive84 ) [ ' derivations ' ]
as Map < String , Map < String , String > > ;
p2pkhChangeAddressArray =
2023-01-12 00:59:01 +00:00
( await resultChange44 ) [ ' addressArray ' ] as List < isar_models . Address > ;
2022-10-27 23:24:14 +00:00
p2pkhChangeIndex = ( await resultChange44 ) [ ' index ' ] as int ;
p2pkhChangeDerivations = ( await resultChange44 ) [ ' derivations ' ]
as Map < String , Map < String , String > > ;
p2shChangeAddressArray =
2023-01-12 00:59:01 +00:00
( await resultChange49 ) [ ' addressArray ' ] as List < isar_models . Address > ;
2022-10-27 23:24:14 +00:00
p2shChangeIndex = ( await resultChange49 ) [ ' index ' ] as int ;
p2shChangeDerivations = ( await resultChange49 ) [ ' derivations ' ]
as Map < String , Map < String , String > > ;
p2wpkhChangeAddressArray =
2023-01-12 00:59:01 +00:00
( await resultChange84 ) [ ' addressArray ' ] as List < isar_models . Address > ;
2022-10-27 23:24:14 +00:00
p2wpkhChangeIndex = ( await resultChange84 ) [ ' index ' ] as int ;
p2wpkhChangeDerivations = ( await resultChange84 ) [ ' derivations ' ]
as Map < String , Map < String , String > > ;
// save the derivations (if any)
if ( p2pkhReceiveDerivations . isNotEmpty ) {
await addDerivations (
chain: 0 ,
derivePathType: DerivePathType . bip44 ,
derivationsToAdd: p2pkhReceiveDerivations ) ;
}
if ( p2shReceiveDerivations . isNotEmpty ) {
await addDerivations (
chain: 0 ,
derivePathType: DerivePathType . bip49 ,
derivationsToAdd: p2shReceiveDerivations ) ;
}
if ( p2wpkhReceiveDerivations . isNotEmpty ) {
await addDerivations (
chain: 0 ,
derivePathType: DerivePathType . bip84 ,
derivationsToAdd: p2wpkhReceiveDerivations ) ;
}
if ( p2pkhChangeDerivations . isNotEmpty ) {
await addDerivations (
chain: 1 ,
derivePathType: DerivePathType . bip44 ,
derivationsToAdd: p2pkhChangeDerivations ) ;
}
if ( p2shChangeDerivations . isNotEmpty ) {
await addDerivations (
chain: 1 ,
derivePathType: DerivePathType . bip49 ,
derivationsToAdd: p2shChangeDerivations ) ;
}
if ( p2wpkhChangeDerivations . isNotEmpty ) {
await addDerivations (
chain: 1 ,
derivePathType: DerivePathType . bip84 ,
derivationsToAdd: p2wpkhChangeDerivations ) ;
}
// If restoring a wallet that never received any funds, then set receivingArray manually
// If we didn't do this, it'd store an empty array
if ( p2pkhReceiveIndex = = - 1 ) {
final address =
await _generateAddressForChain ( 0 , 0 , DerivePathType . bip44 ) ;
p2pkhReceiveAddressArray . add ( address ) ;
}
if ( p2shReceiveIndex = = - 1 ) {
final address =
await _generateAddressForChain ( 0 , 0 , DerivePathType . bip49 ) ;
p2shReceiveAddressArray . add ( address ) ;
}
if ( p2wpkhReceiveIndex = = - 1 ) {
final address =
await _generateAddressForChain ( 0 , 0 , DerivePathType . bip84 ) ;
p2wpkhReceiveAddressArray . add ( address ) ;
}
// If restoring a wallet that never sent any funds with change, then set changeArray
// manually. If we didn't do this, it'd store an empty array.
if ( p2pkhChangeIndex = = - 1 ) {
final address =
await _generateAddressForChain ( 1 , 0 , DerivePathType . bip44 ) ;
p2pkhChangeAddressArray . add ( address ) ;
}
if ( p2shChangeIndex = = - 1 ) {
final address =
await _generateAddressForChain ( 1 , 0 , DerivePathType . bip49 ) ;
p2shChangeAddressArray . add ( address ) ;
}
if ( p2wpkhChangeIndex = = - 1 ) {
final address =
await _generateAddressForChain ( 1 , 0 , DerivePathType . bip84 ) ;
p2wpkhChangeAddressArray . add ( address ) ;
}
2023-02-02 15:03:57 +00:00
if ( isRescan ) {
await db . updateOrPutAddresses ( [
. . . p2wpkhReceiveAddressArray ,
. . . p2wpkhChangeAddressArray ,
. . . p2pkhReceiveAddressArray ,
. . . p2pkhChangeAddressArray ,
. . . p2shReceiveAddressArray ,
. . . p2shChangeAddressArray ,
] ) ;
} else {
await db . putAddresses ( [
. . . p2wpkhReceiveAddressArray ,
. . . p2wpkhChangeAddressArray ,
. . . p2pkhReceiveAddressArray ,
. . . p2pkhChangeAddressArray ,
. . . p2shReceiveAddressArray ,
. . . p2shChangeAddressArray ,
] ) ;
}
2023-01-12 00:59:01 +00:00
await _updateUTXOs ( ) ;
2023-01-12 21:32:25 +00:00
await Future . wait ( [
updateCachedId ( walletId ) ,
updateCachedIsFavorite ( false ) ,
] ) ;
2022-10-27 23:24:14 +00:00
longMutex = false ;
} catch ( e , s ) {
Logging . instance . log (
" Exception rethrown from _recoverWalletFromBIP32SeedPhrase(): $ e \n $ s " ,
level: LogLevel . Error ) ;
longMutex = false ;
rethrow ;
}
}
Future < bool > refreshIfThereIsNewData ( ) async {
if ( longMutex ) return false ;
if ( _hasCalledExit ) return false ;
Logging . instance . log ( " refreshIfThereIsNewData " , level: LogLevel . Info ) ;
try {
bool needsRefresh = false ;
Set < String > txnsToCheck = { } ;
for ( final String txid in txTracker . pendings ) {
if ( ! txTracker . wasNotifiedConfirmed ( txid ) ) {
txnsToCheck . add ( txid ) ;
}
}
for ( String txid in txnsToCheck ) {
final txn = await electrumXClient . getTransaction ( txHash: txid ) ;
int confirmations = txn [ " confirmations " ] as int ? ? ? 0 ;
bool isUnconfirmed = confirmations < MINIMUM_CONFIRMATIONS ;
if ( ! isUnconfirmed ) {
// unconfirmedTxs = {};
needsRefresh = true ;
break ;
}
}
if ( ! needsRefresh ) {
var allOwnAddresses = await _fetchAllOwnAddresses ( ) ;
2023-01-12 00:59:01 +00:00
List < Map < String , dynamic > > allTxs = await _fetchHistory (
allOwnAddresses . map ( ( e ) = > e . value ) . toList ( growable: false ) ) ;
2022-10-27 23:24:14 +00:00
for ( Map < String , dynamic > transaction in allTxs ) {
2023-01-12 00:59:01 +00:00
final txid = transaction [ ' tx_hash ' ] as String ;
2023-01-16 21:04:03 +00:00
if ( ( await db
. getTransactions ( walletId )
2023-01-12 00:59:01 +00:00
. filter ( )
. txidMatches ( txid )
. findFirst ( ) ) = =
2022-10-27 23:24:14 +00:00
null ) {
Logging . instance . log (
" txid not found in address history already ${ transaction [ ' tx_hash ' ] } " ,
level: LogLevel . Info ) ;
needsRefresh = true ;
break ;
}
}
}
return needsRefresh ;
} catch ( e , s ) {
Logging . instance . log (
" Exception caught in refreshIfThereIsNewData: $ e \n $ s " ,
level: LogLevel . Error ) ;
rethrow ;
}
}
2023-01-12 00:59:01 +00:00
Future < void > getAllTxsToWatch ( ) async {
2022-10-27 23:24:14 +00:00
if ( _hasCalledExit ) return ;
2023-01-12 00:59:01 +00:00
List < isar_models . Transaction > unconfirmedTxnsToNotifyPending = [ ] ;
List < isar_models . Transaction > unconfirmedTxnsToNotifyConfirmed = [ ] ;
final currentChainHeight = await chainHeight ;
2023-01-16 21:04:03 +00:00
final txCount = await db . getTransactions ( walletId ) . count ( ) ;
2022-10-27 23:24:14 +00:00
2023-01-12 00:59:01 +00:00
const paginateLimit = 50 ;
for ( int i = 0 ; i < txCount ; i + = paginateLimit ) {
2023-01-16 21:04:03 +00:00
final transactions = await db
. getTransactions ( walletId )
2023-01-12 00:59:01 +00:00
. offset ( i )
. limit ( paginateLimit )
. findAll ( ) ;
for ( final tx in transactions ) {
if ( tx . isConfirmed ( currentChainHeight , MINIMUM_CONFIRMATIONS ) ) {
2022-10-27 23:24:14 +00:00
// get all transactions that were notified as pending but not as confirmed
if ( txTracker . wasNotifiedPending ( tx . txid ) & &
! txTracker . wasNotifiedConfirmed ( tx . txid ) ) {
unconfirmedTxnsToNotifyConfirmed . add ( tx ) ;
}
} else {
// get all transactions that were not notified as pending yet
if ( ! txTracker . wasNotifiedPending ( tx . txid ) ) {
unconfirmedTxnsToNotifyPending . add ( tx ) ;
}
}
}
}
// notify on unconfirmed transactions
for ( final tx in unconfirmedTxnsToNotifyPending ) {
2023-01-12 00:59:01 +00:00
final confirmations = tx . getConfirmations ( currentChainHeight ) ;
if ( tx . type = = isar_models . TransactionType . incoming ) {
2023-05-09 17:54:15 +00:00
CryptoNotificationsEventBus . instance . fire (
CryptoNotificationEvent (
title: " Incoming transaction " ,
walletId: walletId ,
date: DateTime . fromMillisecondsSinceEpoch ( tx . timestamp * 1000 ) ,
shouldWatchForUpdates: confirmations < MINIMUM_CONFIRMATIONS ,
txid: tx . txid ,
confirmations: confirmations ,
requiredConfirmations: MINIMUM_CONFIRMATIONS ,
walletName: walletName ,
coin: coin ,
) ,
) ;
2022-10-27 23:24:14 +00:00
await txTracker . addNotifiedPending ( tx . txid ) ;
2023-01-12 00:59:01 +00:00
} else if ( tx . type = = isar_models . TransactionType . outgoing ) {
2023-05-09 17:54:15 +00:00
CryptoNotificationsEventBus . instance . fire (
CryptoNotificationEvent (
title: " Sending transaction " ,
walletId: walletId ,
date: DateTime . fromMillisecondsSinceEpoch ( tx . timestamp * 1000 ) ,
shouldWatchForUpdates: confirmations < MINIMUM_CONFIRMATIONS ,
txid: tx . txid ,
confirmations: confirmations ,
requiredConfirmations: MINIMUM_CONFIRMATIONS ,
walletName: walletName ,
coin: coin ,
) ,
) ;
2022-10-27 23:24:14 +00:00
await txTracker . addNotifiedPending ( tx . txid ) ;
}
}
// notify on confirmed
for ( final tx in unconfirmedTxnsToNotifyConfirmed ) {
2023-01-12 00:59:01 +00:00
if ( tx . type = = isar_models . TransactionType . incoming ) {
2023-05-09 17:54:15 +00:00
CryptoNotificationsEventBus . instance . fire (
CryptoNotificationEvent (
title: " Incoming transaction confirmed " ,
walletId: walletId ,
date: DateTime . fromMillisecondsSinceEpoch ( tx . timestamp * 1000 ) ,
shouldWatchForUpdates: false ,
txid: tx . txid ,
requiredConfirmations: MINIMUM_CONFIRMATIONS ,
walletName: walletName ,
coin: coin ,
) ,
) ;
2022-10-27 23:24:14 +00:00
await txTracker . addNotifiedConfirmed ( tx . txid ) ;
2023-01-12 00:59:01 +00:00
} else if ( tx . type = = isar_models . TransactionType . outgoing ) {
2023-05-09 17:54:15 +00:00
CryptoNotificationsEventBus . instance . fire (
CryptoNotificationEvent (
title: " Outgoing transaction confirmed " ,
walletId: walletId ,
date: DateTime . fromMillisecondsSinceEpoch ( tx . timestamp * 1000 ) ,
shouldWatchForUpdates: false ,
txid: tx . txid ,
requiredConfirmations: MINIMUM_CONFIRMATIONS ,
walletName: walletName ,
coin: coin ,
) ,
) ;
2022-10-27 23:24:14 +00:00
await txTracker . addNotifiedConfirmed ( tx . txid ) ;
}
}
}
bool _shouldAutoSync = false ;
@ override
bool get shouldAutoSync = > _shouldAutoSync ;
@ override
set shouldAutoSync ( bool shouldAutoSync ) {
if ( _shouldAutoSync ! = shouldAutoSync ) {
_shouldAutoSync = shouldAutoSync ;
if ( ! shouldAutoSync ) {
timer ? . cancel ( ) ;
timer = null ;
stopNetworkAlivePinging ( ) ;
} else {
startNetworkAlivePinging ( ) ;
refresh ( ) ;
}
}
}
@ override
bool get isRefreshing = > refreshMutex ;
bool refreshMutex = false ;
//TODO Show percentages properly/more consistently
/// Refreshes display data for the wallet
@ override
Future < void > refresh ( ) async {
if ( refreshMutex ) {
Logging . instance . log ( " $ walletId $ walletName refreshMutex denied " ,
level: LogLevel . Info ) ;
return ;
} else {
refreshMutex = true ;
}
try {
GlobalEventBus . instance . fire (
WalletSyncStatusChangedEvent (
WalletSyncStatus . syncing ,
walletId ,
coin ,
) ,
) ;
GlobalEventBus . instance . fire ( RefreshPercentChangedEvent ( 0.0 , walletId ) ) ;
GlobalEventBus . instance . fire ( RefreshPercentChangedEvent ( 0.1 , walletId ) ) ;
final currentHeight = await chainHeight ;
const storedHeight = 1 ; //await storedChainHeight;
Logging . instance
. log ( " chain height: $ currentHeight " , level: LogLevel . Info ) ;
Logging . instance
. log ( " cached height: $ storedHeight " , level: LogLevel . Info ) ;
if ( currentHeight ! = storedHeight ) {
GlobalEventBus . instance . fire ( RefreshPercentChangedEvent ( 0.2 , walletId ) ) ;
2023-01-12 00:59:01 +00:00
await _checkChangeAddressForTransactions ( ) ;
2022-10-27 23:24:14 +00:00
GlobalEventBus . instance . fire ( RefreshPercentChangedEvent ( 0.3 , walletId ) ) ;
2023-01-12 00:59:01 +00:00
await _checkCurrentReceivingAddressesForTransactions ( ) ;
2022-10-27 23:24:14 +00:00
2023-01-12 00:59:01 +00:00
final fetchFuture = _refreshTransactions ( ) ;
final utxosRefreshFuture = _updateUTXOs ( ) ;
2022-10-27 23:24:14 +00:00
GlobalEventBus . instance
. fire ( RefreshPercentChangedEvent ( 0.50 , walletId ) ) ;
final feeObj = _getFees ( ) ;
GlobalEventBus . instance
. fire ( RefreshPercentChangedEvent ( 0.60 , walletId ) ) ;
GlobalEventBus . instance
. fire ( RefreshPercentChangedEvent ( 0.70 , walletId ) ) ;
_feeObject = Future ( ( ) = > feeObj ) ;
2023-01-12 00:59:01 +00:00
await utxosRefreshFuture ;
2022-10-27 23:24:14 +00:00
GlobalEventBus . instance
. fire ( RefreshPercentChangedEvent ( 0.80 , walletId ) ) ;
2023-01-12 00:59:01 +00:00
await fetchFuture ;
await getAllTxsToWatch ( ) ;
2022-10-27 23:24:14 +00:00
GlobalEventBus . instance
. fire ( RefreshPercentChangedEvent ( 0.90 , walletId ) ) ;
}
refreshMutex = false ;
GlobalEventBus . instance . fire ( RefreshPercentChangedEvent ( 1.0 , walletId ) ) ;
GlobalEventBus . instance . fire (
WalletSyncStatusChangedEvent (
WalletSyncStatus . synced ,
walletId ,
coin ,
) ,
) ;
if ( shouldAutoSync ) {
timer ? ? = Timer . periodic ( const Duration ( seconds: 30 ) , ( timer ) async {
Logging . instance . log (
" Periodic refresh check for $ walletId $ walletName in object instance: $ hashCode " ,
level: LogLevel . Info ) ;
// chain height check currently broken
// if ((await chainHeight) != (await storedChainHeight)) {
if ( await refreshIfThereIsNewData ( ) ) {
await refresh ( ) ;
GlobalEventBus . instance . fire ( UpdatedInBackgroundEvent (
" New data found in $ walletId $ walletName in background! " ,
walletId ) ) ;
}
// }
} ) ;
}
} catch ( error , strace ) {
refreshMutex = false ;
GlobalEventBus . instance . fire (
NodeConnectionStatusChangedEvent (
NodeConnectionStatus . disconnected ,
walletId ,
coin ,
) ,
) ;
GlobalEventBus . instance . fire (
WalletSyncStatusChangedEvent (
WalletSyncStatus . unableToSync ,
walletId ,
coin ,
) ,
) ;
Logging . instance . log (
" Caught exception in refreshWalletData(): $ error \n $ strace " ,
level: LogLevel . Error ) ;
}
}
@ override
Future < Map < String , dynamic > > prepareSend ( {
required String address ,
2023-04-05 22:06:31 +00:00
required Amount amount ,
2022-10-27 23:24:14 +00:00
Map < String , dynamic > ? args ,
} ) async {
try {
final feeRateType = args ? [ " feeRate " ] ;
2023-06-17 16:42:23 +00:00
final customSatsPerVByte = args ? [ " satsPerVByte " ] as int ? ;
2022-10-27 23:24:14 +00:00
final feeRateAmount = args ? [ " feeRateAmount " ] ;
2023-03-08 22:11:46 +00:00
final utxos = args ? [ " UTXOs " ] as Set < isar_models . UTXO > ? ;
2023-06-17 16:42:23 +00:00
if ( customSatsPerVByte ! = null ) {
// check for send all
bool isSendAll = false ;
if ( amount = = balance . spendable ) {
isSendAll = true ;
}
final bool coinControl = utxos ! = null ;
final result = await coinSelection (
satoshiAmountToSend: amount . raw . toInt ( ) ,
selectedTxFeeRate: - 1 ,
satsPerVByte: customSatsPerVByte ,
recipientAddress: address ,
isSendAll: isSendAll ,
utxos: utxos ? . toList ( ) ,
coinControl: coinControl ,
) ;
Logging . instance
. log ( " PREPARE SEND RESULT: $ result " , level: LogLevel . Info ) ;
if ( result is int ) {
switch ( result ) {
case 1 :
throw Exception ( " Insufficient balance! " ) ;
case 2 :
throw Exception ( " Insufficient funds to pay for transaction fee! " ) ;
default :
throw Exception ( " Transaction failed with error code $ result " ) ;
}
} else {
final hex = result [ " hex " ] ;
if ( hex is String ) {
final fee = result [ " fee " ] as int ;
final vSize = result [ " vSize " ] as int ;
Logging . instance . log ( " txHex: $ hex " , level: LogLevel . Info ) ;
Logging . instance . log ( " fee: $ fee " , level: LogLevel . Info ) ;
Logging . instance . log ( " vsize: $ vSize " , level: LogLevel . Info ) ;
// fee should never be less than vSize sanity check
if ( fee < vSize ) {
throw Exception (
" Error in fee calculation: Transaction fee cannot be less than vSize " ) ;
}
return result as Map < String , dynamic > ;
} else {
throw Exception ( " sent hex is not a String!!! " ) ;
}
}
} else if ( feeRateType is FeeRateType | | feeRateAmount is int ) {
2022-10-27 23:24:14 +00:00
late final int rate ;
if ( feeRateType is FeeRateType ) {
int fee = 0 ;
final feeObject = await fees ;
switch ( feeRateType ) {
case FeeRateType . fast:
fee = feeObject . fast ;
break ;
case FeeRateType . average:
fee = feeObject . medium ;
break ;
case FeeRateType . slow:
fee = feeObject . slow ;
break ;
2023-06-17 16:42:23 +00:00
default :
throw ArgumentError ( " Invalid use of custom fee " ) ;
2022-10-27 23:24:14 +00:00
}
rate = fee ;
} else {
rate = feeRateAmount as int ;
}
// check for send all
bool isSendAll = false ;
2023-04-05 22:06:31 +00:00
if ( amount = = balance . spendable ) {
2022-10-27 23:24:14 +00:00
isSendAll = true ;
}
2023-03-09 16:30:10 +00:00
final bool coinControl = utxos ! = null ;
2023-03-08 22:11:46 +00:00
final txData = await coinSelection (
2023-04-05 22:06:31 +00:00
satoshiAmountToSend: amount . raw . toInt ( ) ,
2023-03-08 22:11:46 +00:00
selectedTxFeeRate: rate ,
recipientAddress: address ,
isSendAll: isSendAll ,
utxos: utxos ? . toList ( ) ,
2023-03-09 16:30:10 +00:00
coinControl: coinControl ,
2023-03-08 22:11:46 +00:00
) ;
2022-10-27 23:24:14 +00:00
Logging . instance . log ( " prepare send: $ txData " , level: LogLevel . Info ) ;
try {
if ( txData is int ) {
switch ( txData ) {
case 1 :
throw Exception ( " Insufficient balance! " ) ;
case 2 :
throw Exception (
" Insufficient funds to pay for transaction fee! " ) ;
default :
throw Exception ( " Transaction failed with error code $ txData " ) ;
}
} else {
final hex = txData [ " hex " ] ;
if ( hex is String ) {
final fee = txData [ " fee " ] as int ;
final vSize = txData [ " vSize " ] as int ;
Logging . instance
. log ( " prepared txHex: $ hex " , level: LogLevel . Info ) ;
Logging . instance . log ( " prepared fee: $ fee " , level: LogLevel . Info ) ;
Logging . instance
. log ( " prepared vSize: $ vSize " , level: LogLevel . Info ) ;
// fee should never be less than vSize sanity check
if ( fee < vSize ) {
throw Exception (
" Error in fee calculation: Transaction fee cannot be less than vSize " ) ;
}
return txData as Map < String , dynamic > ;
} else {
throw Exception ( " prepared hex is not a String!!! " ) ;
}
}
} catch ( e , s ) {
Logging . instance . log ( " Exception rethrown from prepareSend(): $ e \n $ s " ,
level: LogLevel . Error ) ;
rethrow ;
}
} else {
throw ArgumentError ( " Invalid fee rate argument provided! " ) ;
}
} catch ( e , s ) {
Logging . instance . log ( " Exception rethrown from prepareSend(): $ e \n $ s " ,
level: LogLevel . Error ) ;
rethrow ;
}
}
@ override
Future < String > confirmSend ( { required Map < String , dynamic > txData } ) async {
try {
Logging . instance . log ( " confirmSend txData: $ txData " , level: LogLevel . Info ) ;
final hex = txData [ " hex " ] as String ;
final txHash = await _electrumXClient . broadcastTransaction ( rawTx: hex ) ;
Logging . instance . log ( " Sent txHash: $ txHash " , level: LogLevel . Info ) ;
2023-03-08 22:11:46 +00:00
final utxos = txData [ " usedUTXOs " ] as List < isar_models . UTXO > ;
// mark utxos as used
await db . putUTXOs ( utxos . map ( ( e ) = > e . copyWith ( used: true ) ) . toList ( ) ) ;
2022-10-27 23:24:14 +00:00
return txHash ;
} catch ( e , s ) {
Logging . instance . log ( " Exception rethrown from confirmSend(): $ e \n $ s " ,
level: LogLevel . Error ) ;
rethrow ;
}
}
@ override
Future < bool > testNetworkConnection ( ) async {
try {
final result = await _electrumXClient . ping ( ) ;
return result ;
} catch ( _ ) {
return false ;
}
}
Timer ? _networkAliveTimer ;
void startNetworkAlivePinging ( ) {
// call once on start right away
_periodicPingCheck ( ) ;
// then periodically check
_networkAliveTimer = Timer . periodic (
Constants . networkAliveTimerDuration ,
( _ ) async {
_periodicPingCheck ( ) ;
} ,
) ;
}
void _periodicPingCheck ( ) async {
bool hasNetwork = await testNetworkConnection ( ) ;
2023-05-29 16:04:18 +00:00
2022-10-27 23:24:14 +00:00
if ( _isConnected ! = hasNetwork ) {
NodeConnectionStatus status = hasNetwork
? NodeConnectionStatus . connected
: NodeConnectionStatus . disconnected ;
GlobalEventBus . instance
. fire ( NodeConnectionStatusChangedEvent ( status , walletId , coin ) ) ;
2023-05-29 16:04:18 +00:00
_isConnected = hasNetwork ;
if ( hasNetwork ) {
unawaited ( refresh ( ) ) ;
}
2022-10-27 23:24:14 +00:00
}
}
void stopNetworkAlivePinging ( ) {
_networkAliveTimer ? . cancel ( ) ;
_networkAliveTimer = null ;
}
bool _isConnected = false ;
@ override
bool get isConnected = > _isConnected ;
@ override
Future < void > initializeNew ( ) async {
Logging . instance
. log ( " Generating new ${ coin . prettyName } wallet. " , level: LogLevel . Info ) ;
2023-01-12 21:32:25 +00:00
if ( getCachedId ( ) ! = null ) {
2022-10-27 23:24:14 +00:00
throw Exception (
" Attempted to initialize a new wallet using an existing wallet ID! " ) ;
}
await _prefs . init ( ) ;
try {
await _generateNewWallet ( ) ;
} catch ( e , s ) {
Logging . instance . log ( " Exception rethrown from initializeNew(): $ e \n $ s " ,
level: LogLevel . Fatal ) ;
rethrow ;
}
2023-01-12 21:32:25 +00:00
2022-10-27 23:24:14 +00:00
await Future . wait ( [
2023-01-12 21:32:25 +00:00
updateCachedId ( walletId ) ,
updateCachedIsFavorite ( false ) ,
2022-10-27 23:24:14 +00:00
] ) ;
}
@ override
Future < void > initializeExisting ( ) async {
2023-02-13 18:13:30 +00:00
Logging . instance . log ( " initializeExisting() ${ coin . prettyName } wallet. " ,
2022-10-27 23:24:14 +00:00
level: LogLevel . Info ) ;
2023-01-12 21:32:25 +00:00
if ( getCachedId ( ) = = null ) {
2022-10-27 23:24:14 +00:00
throw Exception (
" Attempted to initialize an existing wallet using an unknown wallet ID! " ) ;
}
await _prefs . init ( ) ;
2023-02-13 18:13:30 +00:00
// await _checkCurrentChangeAddressesForTransactions();
// await _checkCurrentReceivingAddressesForTransactions();
2022-10-27 23:24:14 +00:00
}
2022-11-07 16:24:08 +00:00
// hack to add tx to txData before refresh completes
// required based on current app architecture where we don't properly store
// transactions locally in a good way
@ override
Future < void > updateSentCachedTxData ( Map < String , dynamic > txData ) async {
2023-02-02 22:19:14 +00:00
final transaction = isar_models . Transaction (
walletId: walletId ,
txid: txData [ " txid " ] as String ,
timestamp: DateTime . now ( ) . millisecondsSinceEpoch ~ / 1000 ,
type: isar_models . TransactionType . outgoing ,
subType: isar_models . TransactionSubType . none ,
2023-04-06 23:49:13 +00:00
// precision may be lost here hence the following amountString
amount: ( txData [ " recipientAmt " ] as Amount ) . raw . toInt ( ) ,
amountString: ( txData [ " recipientAmt " ] as Amount ) . toJsonString ( ) ,
2023-02-02 22:19:14 +00:00
fee: txData [ " fee " ] as int ,
height: null ,
isCancelled: false ,
isLelantus: false ,
otherData: null ,
slateId: null ,
2023-03-31 16:15:42 +00:00
nonce: null ,
2023-02-03 19:22:21 +00:00
inputs: [ ] ,
outputs: [ ] ,
2023-05-28 12:57:05 +00:00
numberOfMessages: null ,
2023-02-02 22:19:14 +00:00
) ;
final address = txData [ " address " ] is String
? await db . getAddress ( walletId , txData [ " address " ] as String )
: null ;
await db . addNewTransactionData (
[
2023-02-03 19:22:21 +00:00
Tuple2 ( transaction , address ) ,
2023-02-02 22:19:14 +00:00
] ,
walletId ,
) ;
2022-11-07 16:24:08 +00:00
}
2022-10-27 23:24:14 +00:00
@ override
bool validateAddress ( String address ) {
2022-10-28 18:03:52 +00:00
return Address . validateAddress ( address , _network , _network . bech32 ! ) ;
2022-10-27 23:24:14 +00:00
}
@ override
String get walletId = > _walletId ;
2023-01-12 21:32:25 +00:00
late final String _walletId ;
2022-10-27 23:24:14 +00:00
@ override
String get walletName = > _walletName ;
late String _walletName ;
// setter for updating on rename
@ override
set walletName ( String newName ) = > _walletName = newName ;
late ElectrumX _electrumXClient ;
ElectrumX get electrumXClient = > _electrumXClient ;
late CachedElectrumX _cachedElectrumXClient ;
CachedElectrumX get cachedElectrumXClient = > _cachedElectrumXClient ;
2022-11-09 23:48:43 +00:00
late SecureStorageInterface _secureStore ;
2022-10-27 23:24:14 +00:00
@ override
Future < void > updateNode ( bool shouldRefresh ) async {
2022-11-09 22:43:26 +00:00
final failovers = NodeService ( secureStorageInterface: _secureStore )
2022-10-27 23:24:14 +00:00
. failoverNodesFor ( coin: coin )
. map ( ( e ) = > ElectrumXNode (
address: e . host ,
port: e . port ,
name: e . name ,
id: e . id ,
useSSL: e . useSSL ,
) )
. toList ( ) ;
final newNode = await getCurrentNode ( ) ;
_electrumXClient = ElectrumX . from (
node: newNode ,
prefs: _prefs ,
failovers: failovers ,
) ;
2023-05-25 20:37:18 +00:00
_cachedElectrumXClient = CachedElectrumX . from (
electrumXClient: _electrumXClient ,
) ;
2022-10-27 23:24:14 +00:00
if ( shouldRefresh ) {
unawaited ( refresh ( ) ) ;
}
}
Future < List < String > > _getMnemonicList ( ) async {
2023-02-03 22:34:06 +00:00
final _mnemonicString = await mnemonicString ;
if ( _mnemonicString = = null ) {
2022-10-27 23:24:14 +00:00
return [ ] ;
}
2023-02-03 22:34:06 +00:00
final List < String > data = _mnemonicString . split ( ' ' ) ;
2022-10-27 23:24:14 +00:00
return data ;
}
Future < ElectrumXNode > getCurrentNode ( ) async {
2022-11-09 22:43:26 +00:00
final node = NodeService ( secureStorageInterface: _secureStore )
. getPrimaryNodeFor ( coin: coin ) ? ?
2022-10-27 23:24:14 +00:00
DefaultNodes . getNodeFor ( coin ) ;
return ElectrumXNode (
address: node . host ,
port: node . port ,
name: node . name ,
useSSL: node . useSSL ,
id: node . id ,
) ;
}
2023-01-12 00:59:01 +00:00
Future < List < isar_models . Address > > _fetchAllOwnAddresses ( ) async {
2023-01-16 21:04:03 +00:00
final allAddresses = await db
. getAddresses ( walletId )
2023-01-12 00:59:01 +00:00
. filter ( )
2023-01-18 23:20:23 +00:00
. not ( )
. typeEqualTo ( isar_models . AddressType . nonWallet )
. and ( )
. group ( ( q ) = > q
. subTypeEqualTo ( isar_models . AddressSubType . receiving )
. or ( )
. subTypeEqualTo ( isar_models . AddressSubType . change ) )
2023-01-12 00:59:01 +00:00
. findAll ( ) ;
// final List<String> allAddresses = [];
// final receivingAddresses = DB.instance.get<dynamic>(
// boxName: walletId, key: 'receivingAddressesP2WPKH') as List<dynamic>;
// final changeAddresses = DB.instance.get<dynamic>(
// boxName: walletId, key: 'changeAddressesP2WPKH') as List<dynamic>;
// final receivingAddressesP2PKH = DB.instance.get<dynamic>(
// boxName: walletId, key: 'receivingAddressesP2PKH') as List<dynamic>;
// final changeAddressesP2PKH =
// DB.instance.get<dynamic>(boxName: walletId, key: 'changeAddressesP2PKH')
// as List<dynamic>;
// final receivingAddressesP2SH = DB.instance.get<dynamic>(
// boxName: walletId, key: 'receivingAddressesP2SH') as List<dynamic>;
// final changeAddressesP2SH =
// DB.instance.get<dynamic>(boxName: walletId, key: 'changeAddressesP2SH')
// as List<dynamic>;
//
// for (var i = 0; i < receivingAddresses.length; i++) {
// if (!allAddresses.contains(receivingAddresses[i])) {
// allAddresses.add(receivingAddresses[i] as String);
// }
// }
// for (var i = 0; i < changeAddresses.length; i++) {
// if (!allAddresses.contains(changeAddresses[i])) {
// allAddresses.add(changeAddresses[i] as String);
// }
// }
// for (var i = 0; i < receivingAddressesP2PKH.length; i++) {
// if (!allAddresses.contains(receivingAddressesP2PKH[i])) {
// allAddresses.add(receivingAddressesP2PKH[i] as String);
// }
// }
// for (var i = 0; i < changeAddressesP2PKH.length; i++) {
// if (!allAddresses.contains(changeAddressesP2PKH[i])) {
// allAddresses.add(changeAddressesP2PKH[i] as String);
// }
// }
// for (var i = 0; i < receivingAddressesP2SH.length; i++) {
// if (!allAddresses.contains(receivingAddressesP2SH[i])) {
// allAddresses.add(receivingAddressesP2SH[i] as String);
// }
// }
// for (var i = 0; i < changeAddressesP2SH.length; i++) {
// if (!allAddresses.contains(changeAddressesP2SH[i])) {
// allAddresses.add(changeAddressesP2SH[i] as String);
// }
// }
2022-10-27 23:24:14 +00:00
return allAddresses ;
}
Future < FeeObject > _getFees ( ) async {
try {
//TODO adjust numbers for different speeds?
const int f = 1 , m = 5 , s = 20 ;
final fast = await electrumXClient . estimateFee ( blocks: f ) ;
final medium = await electrumXClient . estimateFee ( blocks: m ) ;
final slow = await electrumXClient . estimateFee ( blocks: s ) ;
final feeObject = FeeObject (
numberOfBlocksFast: f ,
numberOfBlocksAverage: m ,
numberOfBlocksSlow: s ,
2023-04-05 22:06:31 +00:00
fast: Amount . fromDecimal (
fast ,
fractionDigits: coin . decimals ,
) . raw . toInt ( ) ,
medium: Amount . fromDecimal (
medium ,
fractionDigits: coin . decimals ,
) . raw . toInt ( ) ,
slow: Amount . fromDecimal (
slow ,
fractionDigits: coin . decimals ,
) . raw . toInt ( ) ,
2022-10-27 23:24:14 +00:00
) ;
Logging . instance . log ( " fetched fees: $ feeObject " , level: LogLevel . Info ) ;
return feeObject ;
} catch ( e ) {
Logging . instance
. log ( " Exception rethrown from _getFees(): $ e " , level: LogLevel . Error ) ;
rethrow ;
}
}
Future < void > _generateNewWallet ( ) async {
2022-11-30 15:46:28 +00:00
Logging . instance
. log ( " IS_INTEGRATION_TEST: $ integrationTestFlag " , level: LogLevel . Info ) ;
if ( ! integrationTestFlag ) {
2022-11-30 15:54:46 +00:00
try {
final features = await electrumXClient
. getServerFeatures ( )
. timeout ( const Duration ( seconds: 3 ) ) ;
Logging . instance . log ( " features: $ features " , level: LogLevel . Info ) ;
switch ( coin ) {
case Coin . litecoin:
if ( features [ ' genesis_hash ' ] ! = GENESIS_HASH_MAINNET ) {
2023-01-12 00:59:01 +00:00
// print(features['genesis_hash']);
2022-11-30 15:54:46 +00:00
throw Exception ( " genesis hash does not match main net! " ) ;
}
break ;
case Coin . litecoinTestNet:
if ( features [ ' genesis_hash ' ] ! = GENESIS_HASH_TESTNET ) {
throw Exception ( " genesis hash does not match test net! " ) ;
}
break ;
default :
throw Exception (
" Attempted to generate a LitecoinWallet using a non litecoin coin type: ${ coin . name } " ) ;
}
} catch ( e , s ) {
Logging . instance . log ( " $ e /n $ s " , level: LogLevel . Info ) ;
2022-11-30 15:46:28 +00:00
}
}
2022-10-27 23:24:14 +00:00
// this should never fail
2023-02-03 22:34:06 +00:00
if ( ( await mnemonicString ) ! = null | | ( await mnemonicPassphrase ) ! = null ) {
2022-10-27 23:24:14 +00:00
throw Exception (
" Attempted to overwrite mnemonic on generate new wallet! " ) ;
}
await _secureStore . write (
key: ' ${ _walletId } _mnemonic ' ,
2023-05-19 20:05:58 +00:00
value: bip39 . generateMnemonic ( strength: 128 ) ) ;
2023-02-03 22:34:06 +00:00
await _secureStore . write (
key: ' ${ _walletId } _mnemonicPassphrase ' ,
value: " " ,
) ;
2022-10-27 23:24:14 +00:00
// Generate and add addresses to relevant arrays
2023-01-12 00:59:01 +00:00
final initialAddresses = await Future . wait ( [
2022-10-27 23:24:14 +00:00
// P2WPKH
2023-01-12 00:59:01 +00:00
_generateAddressForChain ( 0 , 0 , DerivePathType . bip84 ) ,
_generateAddressForChain ( 1 , 0 , DerivePathType . bip84 ) ,
2022-10-27 23:24:14 +00:00
// P2PKH
2023-01-12 00:59:01 +00:00
_generateAddressForChain ( 0 , 0 , DerivePathType . bip44 ) ,
_generateAddressForChain ( 1 , 0 , DerivePathType . bip44 ) ,
2022-10-27 23:24:14 +00:00
// P2SH
2023-01-12 00:59:01 +00:00
_generateAddressForChain ( 0 , 0 , DerivePathType . bip49 ) ,
_generateAddressForChain ( 1 , 0 , DerivePathType . bip49 ) ,
2022-10-27 23:24:14 +00:00
] ) ;
2023-01-16 21:04:03 +00:00
await db . putAddresses ( initialAddresses ) ;
2022-10-27 23:24:14 +00:00
Logging . instance . log ( " _generateNewWalletFinished " , level: LogLevel . Info ) ;
}
/// Generates a new internal or external chain address for the wallet using a BIP84, BIP44, or BIP49 derivation path.
/// [chain] - Use 0 for receiving (external), 1 for change (internal). Should not be any other value!
/// [index] - This can be any integer >= 0
2023-01-12 00:59:01 +00:00
Future < isar_models . Address > _generateAddressForChain (
2022-10-27 23:24:14 +00:00
int chain ,
int index ,
DerivePathType derivePathType ,
) async {
2023-02-03 22:34:06 +00:00
final _mnemonic = await mnemonicString ;
final _mnemonicPassphrase = await mnemonicPassphrase ;
2023-02-13 22:53:28 +00:00
if ( _mnemonicPassphrase = = null ) {
Logging . instance . log (
" Exception in _generateAddressForChain: mnemonic passphrase null, possible migration issue; if using internal builds, delete wallet and restore from seed, if using a release build, please file bug report " ,
level: LogLevel . Error ) ;
}
2023-02-03 22:34:06 +00:00
final derivePath = constructDerivePath (
derivePathType: derivePathType ,
networkWIF: _network . wif ,
chain: chain ,
index: index ,
) ;
final node = await Bip32Utils . getBip32Node (
_mnemonic ! ,
_mnemonicPassphrase ! ,
_network ,
derivePath ,
2022-10-27 23:24:14 +00:00
) ;
2023-02-03 22:34:06 +00:00
2022-10-27 23:24:14 +00:00
final data = PaymentData ( pubkey: node . publicKey ) ;
String address ;
2023-01-12 00:59:01 +00:00
isar_models . AddressType addrType ;
2022-10-27 23:24:14 +00:00
switch ( derivePathType ) {
case DerivePathType . bip44:
address = P2PKH ( data: data , network: _network ) . data . address ! ;
2023-01-12 00:59:01 +00:00
addrType = isar_models . AddressType . p2pkh ;
2022-10-27 23:24:14 +00:00
break ;
case DerivePathType . bip49:
address = P2SH (
data: PaymentData (
2022-10-28 18:03:52 +00:00
redeem: P2WPKH (
data: data ,
network: _network ,
overridePrefix: _network . bech32 ! )
. data ) ,
2022-10-27 23:24:14 +00:00
network: _network )
. data
. address ! ;
2023-01-12 00:59:01 +00:00
addrType = isar_models . AddressType . p2sh ;
2022-10-27 23:24:14 +00:00
break ;
case DerivePathType . bip84:
2022-10-28 18:03:52 +00:00
address = P2WPKH (
network: _network , data: data , overridePrefix: _network . bech32 ! )
. data
. address ! ;
2023-01-12 00:59:01 +00:00
addrType = isar_models . AddressType . p2wpkh ;
2022-10-27 23:24:14 +00:00
break ;
2023-02-02 15:24:26 +00:00
default :
throw Exception ( " DerivePathType unsupported " ) ;
2022-10-27 23:24:14 +00:00
}
// add generated address & info to derivations
await addDerivation (
chain: chain ,
address: address ,
pubKey: Format . uint8listToString ( node . publicKey ) ,
wif: node . toWIF ( ) ,
derivePathType: derivePathType ,
) ;
2023-01-16 22:37:00 +00:00
return isar_models . Address (
walletId: walletId ,
value: address ,
publicKey: node . publicKey ,
type: addrType ,
derivationIndex: index ,
2023-02-03 23:30:32 +00:00
derivationPath: isar_models . DerivationPath ( ) . . value = derivePath ,
2023-01-16 22:37:00 +00:00
subType: chain = = 0
2023-01-12 00:59:01 +00:00
? isar_models . AddressSubType . receiving
2023-01-16 22:37:00 +00:00
: isar_models . AddressSubType . change ,
) ;
2022-10-27 23:24:14 +00:00
}
/// Returns the latest receiving/change (external/internal) address for the wallet depending on [chain]
/// and
/// [chain] - Use 0 for receiving (external), 1 for change (internal). Should not be any other value!
Future < String > _getCurrentAddressForChain (
2023-01-12 00:59:01 +00:00
int chain ,
DerivePathType derivePathType ,
) async {
final subType = chain = = 0 // Here, we assume that chain == 1 if it isn't 0
? isar_models . AddressSubType . receiving
: isar_models . AddressSubType . change ;
isar_models . AddressType type ;
isar_models . Address ? address ;
2022-10-27 23:24:14 +00:00
switch ( derivePathType ) {
case DerivePathType . bip44:
2023-01-12 00:59:01 +00:00
type = isar_models . AddressType . p2pkh ;
2022-10-27 23:24:14 +00:00
break ;
case DerivePathType . bip49:
2023-01-12 00:59:01 +00:00
type = isar_models . AddressType . p2sh ;
2022-10-27 23:24:14 +00:00
break ;
case DerivePathType . bip84:
2023-01-12 00:59:01 +00:00
type = isar_models . AddressType . p2wpkh ;
2022-10-27 23:24:14 +00:00
break ;
2023-02-02 15:24:26 +00:00
default :
throw Exception ( " DerivePathType unsupported " ) ;
2022-10-27 23:24:14 +00:00
}
2023-01-16 21:04:03 +00:00
address = await db
. getAddresses ( walletId )
2023-01-12 00:59:01 +00:00
. filter ( )
. typeEqualTo ( type )
. subTypeEqualTo ( subType )
. sortByDerivationIndexDesc ( )
. findFirst ( ) ;
return address ! . value ;
2022-10-27 23:24:14 +00:00
}
String _buildDerivationStorageKey ( {
required int chain ,
required DerivePathType derivePathType ,
} ) {
String key ;
String chainId = chain = = 0 ? " receive " : " change " ;
switch ( derivePathType ) {
case DerivePathType . bip44:
key = " ${ walletId } _ ${ chainId } DerivationsP2PKH " ;
break ;
case DerivePathType . bip49:
key = " ${ walletId } _ ${ chainId } DerivationsP2SH " ;
break ;
case DerivePathType . bip84:
key = " ${ walletId } _ ${ chainId } DerivationsP2WPKH " ;
break ;
2023-02-02 15:24:26 +00:00
default :
throw Exception ( " DerivePathType unsupported " ) ;
2022-10-27 23:24:14 +00:00
}
return key ;
}
Future < Map < String , dynamic > > _fetchDerivations ( {
required int chain ,
required DerivePathType derivePathType ,
} ) async {
// build lookup key
final key = _buildDerivationStorageKey (
chain: chain , derivePathType: derivePathType ) ;
// fetch current derivations
final derivationsString = await _secureStore . read ( key: key ) ;
return Map < String , dynamic > . from (
jsonDecode ( derivationsString ? ? " {} " ) as Map ) ;
}
/// Add a single derivation to the local secure storage for [chain] and
/// [derivePathType] where [chain] must either be 1 for change or 0 for receive.
/// This will overwrite a previous entry where the address of the new derivation
/// matches a derivation currently stored.
Future < void > addDerivation ( {
required int chain ,
required String address ,
required String pubKey ,
required String wif ,
required DerivePathType derivePathType ,
} ) async {
// build lookup key
final key = _buildDerivationStorageKey (
chain: chain , derivePathType: derivePathType ) ;
// fetch current derivations
final derivationsString = await _secureStore . read ( key: key ) ;
final derivations =
Map < String , dynamic > . from ( jsonDecode ( derivationsString ? ? " {} " ) as Map ) ;
// add derivation
derivations [ address ] = {
" pubKey " : pubKey ,
" wif " : wif ,
} ;
// save derivations
final newReceiveDerivationsString = jsonEncode ( derivations ) ;
await _secureStore . write ( key: key , value: newReceiveDerivationsString ) ;
}
/// Add multiple derivations to the local secure storage for [chain] and
/// [derivePathType] where [chain] must either be 1 for change or 0 for receive.
/// This will overwrite any previous entries where the address of the new derivation
/// matches a derivation currently stored.
/// The [derivationsToAdd] must be in the format of:
/// {
/// addressA : {
/// "pubKey": <the pubKey string>,
/// "wif": <the wif string>,
/// },
/// addressB : {
/// "pubKey": <the pubKey string>,
/// "wif": <the wif string>,
/// },
/// }
Future < void > addDerivations ( {
required int chain ,
required DerivePathType derivePathType ,
required Map < String , dynamic > derivationsToAdd ,
} ) async {
// build lookup key
final key = _buildDerivationStorageKey (
chain: chain , derivePathType: derivePathType ) ;
// fetch current derivations
final derivationsString = await _secureStore . read ( key: key ) ;
final derivations =
Map < String , dynamic > . from ( jsonDecode ( derivationsString ? ? " {} " ) as Map ) ;
// add derivation
derivations . addAll ( derivationsToAdd ) ;
// save derivations
final newReceiveDerivationsString = jsonEncode ( derivations ) ;
await _secureStore . write ( key: key , value: newReceiveDerivationsString ) ;
}
2023-01-12 00:59:01 +00:00
Future < void > _updateUTXOs ( ) async {
final allAddresses = await _fetchAllOwnAddresses ( ) ;
2022-10-27 23:24:14 +00:00
try {
final fetchedUtxoList = < List < Map < String , dynamic > > > [ ] ;
final Map < int , Map < String , List < dynamic > > > batches = { } ;
const batchSizeMax = 100 ;
int batchNumber = 0 ;
for ( int i = 0 ; i < allAddresses . length ; i + + ) {
if ( batches [ batchNumber ] = = null ) {
batches [ batchNumber ] = { } ;
}
2023-01-12 00:59:01 +00:00
final scripthash =
_convertToScriptHash ( allAddresses [ i ] . value , _network ) ;
2022-10-27 23:24:14 +00:00
batches [ batchNumber ] ! . addAll ( {
scripthash: [ scripthash ]
} ) ;
if ( i % batchSizeMax = = batchSizeMax - 1 ) {
batchNumber + + ;
}
}
for ( int i = 0 ; i < batches . length ; i + + ) {
final response =
await _electrumXClient . getBatchUTXOs ( args: batches [ i ] ! ) ;
for ( final entry in response . entries ) {
if ( entry . value . isNotEmpty ) {
fetchedUtxoList . add ( entry . value ) ;
}
}
}
2023-01-12 00:59:01 +00:00
final List < isar_models . UTXO > outputArray = [ ] ;
2022-10-27 23:24:14 +00:00
for ( int i = 0 ; i < fetchedUtxoList . length ; i + + ) {
for ( int j = 0 ; j < fetchedUtxoList [ i ] . length ; j + + ) {
2023-03-08 22:11:46 +00:00
final jsonUTXO = fetchedUtxoList [ i ] [ j ] ;
2022-10-27 23:24:14 +00:00
final txn = await cachedElectrumXClient . getTransaction (
2023-03-08 22:11:46 +00:00
txHash: jsonUTXO [ " tx_hash " ] as String ,
2022-10-27 23:24:14 +00:00
verbose: true ,
coin: coin ,
) ;
2023-05-24 19:02:36 +00:00
bool shouldBlock = false ;
String ? blockReason ;
String ? label ;
2023-05-24 19:07:44 +00:00
final utxoAmount = jsonUTXO [ " value " ] as int ;
2023-05-24 19:02:36 +00:00
2023-05-24 19:07:44 +00:00
if ( utxoAmount < = 10000 ) {
shouldBlock = true ;
blockReason = " May contain ordinal " ;
label = " Possible ordinal " ;
2023-05-24 19:02:36 +00:00
}
2023-03-08 22:11:46 +00:00
final vout = jsonUTXO [ " tx_pos " ] as int ;
final outputs = txn [ " vout " ] as List ;
String ? utxoOwnerAddress ;
// get UTXO owner address
for ( final output in outputs ) {
if ( output [ " n " ] = = vout ) {
utxoOwnerAddress =
output [ " scriptPubKey " ] ? [ " addresses " ] ? [ 0 ] as String ? ? ?
output [ " scriptPubKey " ] ? [ " address " ] as String ? ;
}
}
2023-01-16 22:37:00 +00:00
final utxo = isar_models . UTXO (
walletId: walletId ,
txid: txn [ " txid " ] as String ,
2023-03-08 22:11:46 +00:00
vout: vout ,
2023-05-24 19:07:44 +00:00
value: utxoAmount ,
2023-05-24 19:02:36 +00:00
name: label ? ? " " ,
isBlocked: shouldBlock ,
blockedReason: blockReason ,
2023-01-16 22:37:00 +00:00
isCoinbase: txn [ " is_coinbase " ] as bool ? ? ? false ,
blockHash: txn [ " blockhash " ] as String ? ,
2023-03-08 22:11:46 +00:00
blockHeight: jsonUTXO [ " height " ] as int ? ,
2023-01-16 22:37:00 +00:00
blockTime: txn [ " blocktime " ] as int ? ,
2023-03-08 22:11:46 +00:00
address: utxoOwnerAddress ,
2023-01-16 22:37:00 +00:00
) ;
2023-01-12 00:59:01 +00:00
2022-10-27 23:24:14 +00:00
outputArray . add ( utxo ) ;
}
}
2023-05-24 19:07:44 +00:00
Logging . instance . log (
' Outputs fetched: $ outputArray ' ,
level: LogLevel . Info ,
) ;
2022-10-27 23:24:14 +00:00
2023-03-08 22:11:46 +00:00
await db . updateUTXOs ( walletId , outputArray ) ;
2022-10-27 23:24:14 +00:00
2023-01-12 00:59:01 +00:00
// finally update balance
2023-03-08 22:11:46 +00:00
await _updateBalance ( ) ;
2022-10-27 23:24:14 +00:00
} catch ( e , s ) {
2023-05-24 19:07:44 +00:00
Logging . instance . log (
" Output fetch unsuccessful: $ e \n $ s " ,
level: LogLevel . Error ,
) ;
2022-10-27 23:24:14 +00:00
}
}
2023-03-08 22:11:46 +00:00
Future < void > _updateBalance ( ) async {
await refreshBalance ( ) ;
}
2023-01-12 00:59:01 +00:00
@ override
2023-01-12 21:32:25 +00:00
Balance get balance = > _balance ? ? = getCachedBalance ( ) ;
2023-01-12 00:59:01 +00:00
Balance ? _balance ;
// /// Takes in a list of UtxoObjects and adds a name (dependent on object index within list)
// /// and checks for the txid associated with the utxo being blocked and marks it accordingly.
// /// Now also checks for output labeling.
// Future<void> _sortOutputs(List<UtxoObject> utxos) async {
// final blockedHashArray =
// DB.instance.get<dynamic>(boxName: walletId, key: 'blocked_tx_hashes')
// as List<dynamic>?;
// final List<String> lst = [];
// if (blockedHashArray != null) {
// for (var hash in blockedHashArray) {
// lst.add(hash as String);
// }
// }
// final labels =
// DB.instance.get<dynamic>(boxName: walletId, key: 'labels') as Map? ??
// {};
//
// outputsList = [];
//
// for (var i = 0; i < utxos.length; i++) {
// if (labels[utxos[i].txid] != null) {
// utxos[i].txName = labels[utxos[i].txid] as String? ?? "";
// } else {
// utxos[i].txName = 'Output #$i';
// }
//
// if (utxos[i].status.confirmed == false) {
// outputsList.add(utxos[i]);
// } else {
// if (lst.contains(utxos[i].txid)) {
// utxos[i].blocked = true;
// outputsList.add(utxos[i]);
// } else if (!lst.contains(utxos[i].txid)) {
// outputsList.add(utxos[i]);
// }
// }
// }
// }
2022-10-27 23:24:14 +00:00
Future < int > getTxCount ( { required String address } ) async {
String ? scripthash ;
try {
scripthash = _convertToScriptHash ( address , _network ) ;
final transactions =
await electrumXClient . getHistory ( scripthash: scripthash ) ;
return transactions . length ;
} catch ( e ) {
Logging . instance . log (
" Exception rethrown in _getTxCount(address: $ address , scripthash: $ scripthash ): $ e " ,
level: LogLevel . Error ) ;
rethrow ;
}
}
Future < Map < String , int > > _getBatchTxCount ( {
required Map < String , String > addresses ,
} ) async {
try {
final Map < String , List < dynamic > > args = { } ;
for ( final entry in addresses . entries ) {
args [ entry . key ] = [ _convertToScriptHash ( entry . value , _network ) ] ;
}
final response = await electrumXClient . getBatchHistory ( args: args ) ;
final Map < String , int > result = { } ;
for ( final entry in response . entries ) {
result [ entry . key ] = entry . value . length ;
}
return result ;
} catch ( e , s ) {
Logging . instance . log (
" Exception rethrown in _getBatchTxCount(address: $ addresses : $ e \n $ s " ,
level: LogLevel . Error ) ;
rethrow ;
}
}
2023-01-12 00:59:01 +00:00
Future < void > _checkReceivingAddressForTransactions ( ) async {
2022-10-27 23:24:14 +00:00
try {
2023-01-12 00:59:01 +00:00
final currentReceiving = await _currentReceivingAddress ;
final int txCount = await getTxCount ( address: currentReceiving . value ) ;
2022-10-27 23:24:14 +00:00
Logging . instance . log (
2023-01-12 00:59:01 +00:00
' Number of txs for current receiving address $ currentReceiving : $ txCount ' ,
2022-10-27 23:24:14 +00:00
level: LogLevel . Info ) ;
2023-01-23 16:32:53 +00:00
if ( txCount > = 1 | | currentReceiving . derivationIndex < 0 ) {
2022-10-27 23:24:14 +00:00
// First increment the receiving index
2023-01-12 00:59:01 +00:00
final newReceivingIndex = currentReceiving . derivationIndex + 1 ;
2022-10-27 23:24:14 +00:00
// Use new index to derive a new receiving address
final newReceivingAddress = await _generateAddressForChain (
2023-01-25 19:49:14 +00:00
0 , newReceivingIndex , DerivePathTypeExt . primaryFor ( coin ) ) ;
2022-10-27 23:24:14 +00:00
2023-01-18 22:55:59 +00:00
final existing = await db
. getAddresses ( walletId )
. filter ( )
. valueEqualTo ( newReceivingAddress . value )
. findFirst ( ) ;
if ( existing = = null ) {
// Add that new change address
await db . putAddress ( newReceivingAddress ) ;
} else {
// we need to update the address
await db . updateAddress ( existing , newReceivingAddress ) ;
}
2023-01-23 16:32:53 +00:00
// keep checking until address with no tx history is set as current
await _checkReceivingAddressForTransactions ( ) ;
2022-10-27 23:24:14 +00:00
}
} catch ( e , s ) {
Logging . instance . log (
2023-01-25 19:49:14 +00:00
" Exception rethrown from _checkReceivingAddressForTransactions( ${ DerivePathTypeExt . primaryFor ( coin ) } ): $ e \n $ s " ,
2022-10-27 23:24:14 +00:00
level: LogLevel . Error ) ;
rethrow ;
}
}
2023-01-12 00:59:01 +00:00
Future < void > _checkChangeAddressForTransactions ( ) async {
2022-10-27 23:24:14 +00:00
try {
2023-01-12 00:59:01 +00:00
final currentChange = await _currentChangeAddress ;
final int txCount = await getTxCount ( address: currentChange . value ) ;
2022-10-27 23:24:14 +00:00
Logging . instance . log (
2023-01-12 00:59:01 +00:00
' Number of txs for current change address $ currentChange : $ txCount ' ,
2022-10-27 23:24:14 +00:00
level: LogLevel . Info ) ;
2023-01-23 16:32:53 +00:00
if ( txCount > = 1 | | currentChange . derivationIndex < 0 ) {
2022-10-27 23:24:14 +00:00
// First increment the change index
2023-01-12 00:59:01 +00:00
final newChangeIndex = currentChange . derivationIndex + 1 ;
2022-10-27 23:24:14 +00:00
// Use new index to derive a new change address
2023-01-12 00:59:01 +00:00
final newChangeAddress = await _generateAddressForChain (
2023-01-25 19:49:14 +00:00
1 , newChangeIndex , DerivePathTypeExt . primaryFor ( coin ) ) ;
2022-10-27 23:24:14 +00:00
2023-01-18 22:55:59 +00:00
final existing = await db
. getAddresses ( walletId )
. filter ( )
. valueEqualTo ( newChangeAddress . value )
. findFirst ( ) ;
if ( existing = = null ) {
// Add that new change address
await db . putAddress ( newChangeAddress ) ;
} else {
// we need to update the address
await db . updateAddress ( existing , newChangeAddress ) ;
}
2023-01-23 16:32:53 +00:00
// keep checking until address with no tx history is set as current
await _checkChangeAddressForTransactions ( ) ;
2022-10-27 23:24:14 +00:00
}
} on SocketException catch ( se , s ) {
Logging . instance . log (
2023-01-25 19:49:14 +00:00
" SocketException caught in _checkReceivingAddressForTransactions( ${ DerivePathTypeExt . primaryFor ( coin ) } ): $ se \n $ s " ,
2022-10-27 23:24:14 +00:00
level: LogLevel . Error ) ;
return ;
} catch ( e , s ) {
Logging . instance . log (
2023-01-25 19:49:14 +00:00
" Exception rethrown from _checkReceivingAddressForTransactions( ${ DerivePathTypeExt . primaryFor ( coin ) } ): $ e \n $ s " ,
2022-10-27 23:24:14 +00:00
level: LogLevel . Error ) ;
rethrow ;
}
}
Future < void > _checkCurrentReceivingAddressesForTransactions ( ) async {
try {
2023-01-12 00:59:01 +00:00
// for (final type in DerivePathType.values) {
await _checkReceivingAddressForTransactions ( ) ;
// }
2022-10-27 23:24:14 +00:00
} catch ( e , s ) {
Logging . instance . log (
" Exception rethrown from _checkCurrentReceivingAddressesForTransactions(): $ e \n $ s " ,
level: LogLevel . Error ) ;
rethrow ;
}
}
/// public wrapper because dart can't test private...
Future < void > checkCurrentReceivingAddressesForTransactions ( ) async {
if ( Platform . environment [ " FLUTTER_TEST " ] = = " true " ) {
try {
return _checkCurrentReceivingAddressesForTransactions ( ) ;
} catch ( _ ) {
rethrow ;
}
}
}
Future < void > _checkCurrentChangeAddressesForTransactions ( ) async {
try {
2023-01-12 00:59:01 +00:00
// for (final type in DerivePathType.values) {
await _checkChangeAddressForTransactions ( ) ;
// }
2022-10-27 23:24:14 +00:00
} catch ( e , s ) {
Logging . instance . log (
" Exception rethrown from _checkCurrentChangeAddressesForTransactions(): $ e \n $ s " ,
level: LogLevel . Error ) ;
rethrow ;
}
}
/// public wrapper because dart can't test private...
Future < void > checkCurrentChangeAddressesForTransactions ( ) async {
if ( Platform . environment [ " FLUTTER_TEST " ] = = " true " ) {
try {
return _checkCurrentChangeAddressesForTransactions ( ) ;
} catch ( _ ) {
rethrow ;
}
}
}
/// attempts to convert a string to a valid scripthash
///
2022-10-28 18:03:52 +00:00
/// Returns the scripthash or throws an exception on invalid litecoin address
String _convertToScriptHash ( String litecoinAddress , NetworkType network ) {
2022-10-27 23:24:14 +00:00
try {
2022-10-28 18:03:52 +00:00
final output = Address . addressToOutputScript (
litecoinAddress , network , _network . bech32 ! ) ;
2022-10-27 23:24:14 +00:00
final hash = sha256 . convert ( output . toList ( growable: false ) ) . toString ( ) ;
final chars = hash . split ( " " ) ;
final reversedPairs = < String > [ ] ;
var i = chars . length - 1 ;
while ( i > 0 ) {
reversedPairs . add ( chars [ i - 1 ] ) ;
reversedPairs . add ( chars [ i ] ) ;
i - = 2 ;
}
return reversedPairs . join ( " " ) ;
} catch ( e ) {
rethrow ;
}
}
Future < List < Map < String , dynamic > > > _fetchHistory (
List < String > allAddresses ) async {
try {
List < Map < String , dynamic > > allTxHashes = [ ] ;
final Map < int , Map < String , List < dynamic > > > batches = { } ;
final Map < String , String > requestIdToAddressMap = { } ;
const batchSizeMax = 100 ;
int batchNumber = 0 ;
for ( int i = 0 ; i < allAddresses . length ; i + + ) {
if ( batches [ batchNumber ] = = null ) {
batches [ batchNumber ] = { } ;
}
final scripthash = _convertToScriptHash ( allAddresses [ i ] , _network ) ;
final id = Logger . isTestEnv ? " $ i " : const Uuid ( ) . v1 ( ) ;
requestIdToAddressMap [ id ] = allAddresses [ i ] ;
batches [ batchNumber ] ! . addAll ( {
id: [ scripthash ]
} ) ;
if ( i % batchSizeMax = = batchSizeMax - 1 ) {
batchNumber + + ;
}
}
for ( int i = 0 ; i < batches . length ; i + + ) {
final response =
await _electrumXClient . getBatchHistory ( args: batches [ i ] ! ) ;
for ( final entry in response . entries ) {
for ( int j = 0 ; j < entry . value . length ; j + + ) {
entry . value [ j ] [ " address " ] = requestIdToAddressMap [ entry . key ] ;
if ( ! allTxHashes . contains ( entry . value [ j ] ) ) {
allTxHashes . add ( entry . value [ j ] ) ;
}
}
}
}
return allTxHashes ;
} catch ( e , s ) {
Logging . instance . log ( " _fetchHistory: $ e \n $ s " , level: LogLevel . Error ) ;
rethrow ;
}
}
Future < List < Map < String , dynamic > > > fastFetch ( List < String > allTxHashes ) async {
List < Map < String , dynamic > > allTransactions = [ ] ;
const futureLimit = 30 ;
List < Future < Map < String , dynamic > > > transactionFutures = [ ] ;
int currentFutureCount = 0 ;
for ( final txHash in allTxHashes ) {
Future < Map < String , dynamic > > transactionFuture =
cachedElectrumXClient . getTransaction (
txHash: txHash ,
verbose: true ,
coin: coin ,
) ;
transactionFutures . add ( transactionFuture ) ;
currentFutureCount + + ;
if ( currentFutureCount > futureLimit ) {
currentFutureCount = 0 ;
await Future . wait ( transactionFutures ) ;
for ( final fTx in transactionFutures ) {
final tx = await fTx ;
allTransactions . add ( tx ) ;
}
}
}
if ( currentFutureCount ! = 0 ) {
currentFutureCount = 0 ;
await Future . wait ( transactionFutures ) ;
for ( final fTx in transactionFutures ) {
final tx = await fTx ;
allTransactions . add ( tx ) ;
}
}
return allTransactions ;
}
2023-01-12 00:59:01 +00:00
bool _duplicateTxCheck (
List < Map < String , dynamic > > allTransactions , String txid ) {
for ( int i = 0 ; i < allTransactions . length ; i + + ) {
if ( allTransactions [ i ] [ " txid " ] = = txid ) {
return true ;
}
}
return false ;
}
2022-10-27 23:24:14 +00:00
2023-01-12 00:59:01 +00:00
Future < void > _refreshTransactions ( ) async {
final List < isar_models . Address > allAddresses =
await _fetchAllOwnAddresses ( ) ;
2022-10-27 23:24:14 +00:00
final List < Map < String , dynamic > > allTxHashes =
2023-01-12 00:59:01 +00:00
await _fetchHistory ( allAddresses . map ( ( e ) = > e . value ) . toList ( ) ) ;
2022-10-27 23:24:14 +00:00
Set < String > hashes = { } ;
for ( var element in allTxHashes ) {
hashes . add ( element [ ' tx_hash ' ] as String ) ;
}
await fastFetch ( hashes . toList ( ) ) ;
2023-01-13 21:36:19 +00:00
2022-10-27 23:24:14 +00:00
List < Map < String , dynamic > > allTransactions = [ ] ;
2023-01-13 21:36:19 +00:00
final currentHeight = await chainHeight ;
2022-10-27 23:24:14 +00:00
for ( final txHash in allTxHashes ) {
2023-01-16 21:04:03 +00:00
final storedTx = await db
. getTransactions ( walletId )
. filter ( )
2023-01-13 21:36:19 +00:00
. txidEqualTo ( txHash [ " tx_hash " ] as String )
. findFirst ( ) ;
if ( storedTx = = null | |
! storedTx . isConfirmed ( currentHeight , MINIMUM_CONFIRMATIONS ) ) {
final tx = await cachedElectrumXClient . getTransaction (
txHash: txHash [ " tx_hash " ] as String ,
verbose: true ,
coin: coin ,
) ;
2022-10-27 23:24:14 +00:00
2023-01-13 21:36:19 +00:00
// Logging.instance.log("TRANSACTION: ${jsonEncode(tx)}");
if ( ! _duplicateTxCheck ( allTransactions , tx [ " txid " ] as String ) ) {
2023-01-16 21:04:03 +00:00
tx [ " address " ] = await db
. getAddresses ( walletId )
2023-01-13 21:44:14 +00:00
. filter ( )
. valueEqualTo ( txHash [ " address " ] as String )
. findFirst ( ) ;
2023-01-13 21:36:19 +00:00
tx [ " height " ] = txHash [ " height " ] ;
allTransactions . add ( tx ) ;
}
2022-10-27 23:24:14 +00:00
}
}
2023-01-13 21:36:19 +00:00
// prefetch/cache
2022-10-27 23:24:14 +00:00
Set < String > vHashes = { } ;
for ( final txObject in allTransactions ) {
for ( int i = 0 ; i < ( txObject [ " vin " ] as List ) . length ; i + + ) {
final input = txObject [ " vin " ] ! [ i ] as Map ;
final prevTxid = input [ " txid " ] as String ;
vHashes . add ( prevTxid ) ;
}
}
await fastFetch ( vHashes . toList ( ) ) ;
2023-02-03 19:22:21 +00:00
final List < Tuple2 < isar_models . Transaction , isar_models . Address ? > > txnsData =
[ ] ;
2023-01-13 21:36:19 +00:00
2022-10-27 23:24:14 +00:00
for ( final txObject in allTransactions ) {
2023-01-13 21:36:19 +00:00
final data = await parseTransaction (
2023-01-12 00:59:01 +00:00
txObject ,
cachedElectrumXClient ,
allAddresses ,
coin ,
MINIMUM_CONFIRMATIONS ,
2023-01-16 21:04:03 +00:00
walletId ,
2023-01-12 00:59:01 +00:00
) ;
2022-10-27 23:24:14 +00:00
2023-01-13 21:36:19 +00:00
txnsData . add ( data ) ;
2022-10-27 23:24:14 +00:00
}
2023-01-19 21:13:03 +00:00
await db . addNewTransactionData ( txnsData , walletId ) ;
2023-01-19 16:29:00 +00:00
// quick hack to notify manager to call notifyListeners if
// transactions changed
if ( txnsData . isNotEmpty ) {
GlobalEventBus . instance . fire (
UpdatedInBackgroundEvent (
" Transactions updated/added for: $ walletId $ walletName " ,
walletId ,
) ,
) ;
}
2022-10-27 23:24:14 +00:00
}
int estimateTxFee ( { required int vSize , required int feeRatePerKB } ) {
return vSize * ( feeRatePerKB / 1000 ) . ceil ( ) ;
}
/// The coinselection algorithm decides whether or not the user is eligible to make the transaction
/// with [satoshiAmountToSend] and [selectedTxFeeRate]. If so, it will call buildTrasaction() and return
/// a map containing the tx hex along with other important information. If not, then it will return
/// an integer (1 or 2)
2023-03-08 22:11:46 +00:00
dynamic coinSelection ( {
required int satoshiAmountToSend ,
required int selectedTxFeeRate ,
required String recipientAddress ,
required bool coinControl ,
required bool isSendAll ,
2023-06-17 16:42:23 +00:00
int ? satsPerVByte ,
2022-10-27 23:24:14 +00:00
int additionalOutputs = 0 ,
2023-01-12 00:59:01 +00:00
List < isar_models . UTXO > ? utxos ,
2022-10-27 23:24:14 +00:00
} ) async {
Logging . instance
. log ( " Starting coinSelection ---------- " , level: LogLevel . Info ) ;
2023-01-12 00:59:01 +00:00
final List < isar_models . UTXO > availableOutputs = utxos ? ? await this . utxos ;
final currentChainHeight = await chainHeight ;
final List < isar_models . UTXO > spendableOutputs = [ ] ;
2022-10-27 23:24:14 +00:00
int spendableSatoshiValue = 0 ;
// Build list of spendable outputs and totaling their satoshi amount
2023-03-08 22:11:46 +00:00
for ( final utxo in availableOutputs ) {
if ( utxo . isBlocked = = false & &
utxo . isConfirmed ( currentChainHeight , MINIMUM_CONFIRMATIONS ) & &
utxo . used ! = true ) {
spendableOutputs . add ( utxo ) ;
spendableSatoshiValue + = utxo . value ;
}
}
if ( coinControl ) {
if ( spendableOutputs . length < availableOutputs . length ) {
throw ArgumentError ( " Attempted to use an unavailable utxo " ) ;
2022-10-27 23:24:14 +00:00
}
}
2023-03-08 22:11:46 +00:00
// don't care about sorting if using all utxos
if ( ! coinControl ) {
// sort spendable by age (oldest first)
spendableOutputs . sort ( ( a , b ) = > b . blockTime ! . compareTo ( a . blockTime ! ) ) ;
}
2022-10-27 23:24:14 +00:00
Logging . instance . log ( " spendableOutputs.length: ${ spendableOutputs . length } " ,
level: LogLevel . Info ) ;
Logging . instance
. log ( " spendableOutputs: $ spendableOutputs " , level: LogLevel . Info ) ;
Logging . instance . log ( " spendableSatoshiValue: $ spendableSatoshiValue " ,
level: LogLevel . Info ) ;
Logging . instance
. log ( " satoshiAmountToSend: $ satoshiAmountToSend " , level: LogLevel . Info ) ;
// If the amount the user is trying to send is smaller than the amount that they have spendable,
// then return 1, which indicates that they have an insufficient balance.
if ( spendableSatoshiValue < satoshiAmountToSend ) {
return 1 ;
// If the amount the user wants to send is exactly equal to the amount they can spend, then return
// 2, which indicates that they are not leaving enough over to pay the transaction fee
} else if ( spendableSatoshiValue = = satoshiAmountToSend & & ! isSendAll ) {
return 2 ;
}
// If neither of these statements pass, we assume that the user has a spendable balance greater
// than the amount they're attempting to send. Note that this value still does not account for
// the added transaction fee, which may require an extra input and will need to be checked for
// later on.
// Possible situation right here
int satoshisBeingUsed = 0 ;
int inputsBeingConsumed = 0 ;
2023-01-12 00:59:01 +00:00
List < isar_models . UTXO > utxoObjectsToUse = [ ] ;
2022-10-27 23:24:14 +00:00
2023-03-08 22:11:46 +00:00
if ( ! coinControl ) {
for ( var i = 0 ;
satoshisBeingUsed < satoshiAmountToSend & &
i < spendableOutputs . length ;
i + + ) {
utxoObjectsToUse . add ( spendableOutputs [ i ] ) ;
satoshisBeingUsed + = spendableOutputs [ i ] . value ;
inputsBeingConsumed + = 1 ;
}
for ( int i = 0 ;
i < additionalOutputs & &
inputsBeingConsumed < spendableOutputs . length ;
i + + ) {
utxoObjectsToUse . add ( spendableOutputs [ inputsBeingConsumed ] ) ;
satoshisBeingUsed + = spendableOutputs [ inputsBeingConsumed ] . value ;
inputsBeingConsumed + = 1 ;
}
} else {
satoshisBeingUsed = spendableSatoshiValue ;
utxoObjectsToUse = spendableOutputs ;
2023-03-09 16:30:10 +00:00
inputsBeingConsumed = spendableOutputs . length ;
2022-10-27 23:24:14 +00:00
}
Logging . instance
. log ( " satoshisBeingUsed: $ satoshisBeingUsed " , level: LogLevel . Info ) ;
Logging . instance
. log ( " inputsBeingConsumed: $ inputsBeingConsumed " , level: LogLevel . Info ) ;
Logging . instance
. log ( ' utxoObjectsToUse: $ utxoObjectsToUse ' , level: LogLevel . Info ) ;
// numberOfOutputs' length must always be equal to that of recipientsArray and recipientsAmtArray
2023-03-08 22:11:46 +00:00
List < String > recipientsArray = [ recipientAddress ] ;
2022-10-27 23:24:14 +00:00
List < int > recipientsAmtArray = [ satoshiAmountToSend ] ;
// gather required signing data
final utxoSigningData = await fetchBuildTxData ( utxoObjectsToUse ) ;
if ( isSendAll ) {
Logging . instance
. log ( " Attempting to send all $ coin " , level: LogLevel . Info ) ;
final int vSizeForOneOutput = ( await buildTransaction (
utxoSigningData: utxoSigningData ,
2023-03-08 22:11:46 +00:00
recipients: [ recipientAddress ] ,
2022-10-27 23:24:14 +00:00
satoshiAmounts: [ satoshisBeingUsed - 1 ] ,
) ) [ " vSize " ] as int ;
2023-06-17 16:42:23 +00:00
int feeForOneOutput = satsPerVByte ! = null
? ( satsPerVByte * vSizeForOneOutput )
: estimateTxFee (
vSize: vSizeForOneOutput ,
feeRatePerKB: selectedTxFeeRate ,
) ;
2022-10-27 23:24:14 +00:00
2023-06-17 16:42:23 +00:00
if ( satsPerVByte = = null ) {
final int roughEstimate = roughFeeEstimate (
spendableOutputs . length ,
1 ,
selectedTxFeeRate ,
) . raw . toInt ( ) ;
if ( feeForOneOutput < roughEstimate ) {
feeForOneOutput = roughEstimate ;
}
2022-10-27 23:24:14 +00:00
}
final int amount = satoshiAmountToSend - feeForOneOutput ;
dynamic txn = await buildTransaction (
utxoSigningData: utxoSigningData ,
recipients: recipientsArray ,
satoshiAmounts: [ amount ] ,
) ;
Map < String , dynamic > transactionObject = {
" hex " : txn [ " hex " ] ,
" recipient " : recipientsArray [ 0 ] ,
2023-04-11 15:17:58 +00:00
" recipientAmt " : Amount (
rawValue: BigInt . from ( amount ) ,
fractionDigits: coin . decimals ,
) ,
2022-10-27 23:24:14 +00:00
" fee " : feeForOneOutput ,
" vSize " : txn [ " vSize " ] ,
2023-03-09 19:49:39 +00:00
" usedUTXOs " : utxoSigningData . map ( ( e ) = > e . utxo ) . toList ( ) ,
2022-10-27 23:24:14 +00:00
} ;
return transactionObject ;
}
final int vSizeForOneOutput = ( await buildTransaction (
utxoSigningData: utxoSigningData ,
2023-03-08 22:11:46 +00:00
recipients: [ recipientAddress ] ,
2022-10-27 23:24:14 +00:00
satoshiAmounts: [ satoshisBeingUsed - 1 ] ,
) ) [ " vSize " ] as int ;
final int vSizeForTwoOutPuts = ( await buildTransaction (
utxoSigningData: utxoSigningData ,
recipients: [
2023-03-08 22:11:46 +00:00
recipientAddress ,
2023-01-25 19:49:14 +00:00
await _getCurrentAddressForChain ( 1 , DerivePathTypeExt . primaryFor ( coin ) ) ,
2022-10-27 23:24:14 +00:00
] ,
satoshiAmounts: [
satoshiAmountToSend ,
satoshisBeingUsed - satoshiAmountToSend - 1
] , // dust limit is the minimum amount a change output should be
) ) [ " vSize " ] as int ;
// Assume 1 output, only for recipient and no change
2023-06-17 16:42:23 +00:00
final feeForOneOutput = satsPerVByte ! = null
? ( satsPerVByte * vSizeForOneOutput )
: estimateTxFee (
vSize: vSizeForOneOutput ,
feeRatePerKB: selectedTxFeeRate ,
) ;
2022-10-27 23:24:14 +00:00
// Assume 2 outputs, one for recipient and one for change
2023-06-17 16:42:23 +00:00
final feeForTwoOutputs = satsPerVByte ! = null
? ( satsPerVByte * vSizeForTwoOutPuts )
: estimateTxFee (
vSize: vSizeForTwoOutPuts ,
feeRatePerKB: selectedTxFeeRate ,
) ;
2022-10-27 23:24:14 +00:00
Logging . instance
. log ( " feeForTwoOutputs: $ feeForTwoOutputs " , level: LogLevel . Info ) ;
Logging . instance
. log ( " feeForOneOutput: $ feeForOneOutput " , level: LogLevel . Info ) ;
if ( satoshisBeingUsed - satoshiAmountToSend > feeForOneOutput ) {
if ( satoshisBeingUsed - satoshiAmountToSend >
2023-04-05 22:06:31 +00:00
feeForOneOutput + DUST_LIMIT . raw . toInt ( ) ) {
2022-10-27 23:24:14 +00:00
// Here, we know that theoretically, we may be able to include another output(change) but we first need to
// factor in the value of this output in satoshis.
int changeOutputSize =
satoshisBeingUsed - satoshiAmountToSend - feeForTwoOutputs ;
// We check to see if the user can pay for the new transaction with 2 outputs instead of one. If they can and
// the second output's size > DUST_LIMIT satoshis, we perform the mechanics required to properly generate and use a new
// change address.
2023-04-05 22:06:31 +00:00
if ( changeOutputSize > DUST_LIMIT . raw . toInt ( ) & &
2022-10-27 23:24:14 +00:00
satoshisBeingUsed - satoshiAmountToSend - changeOutputSize = =
feeForTwoOutputs ) {
// generate new change address if current change address has been used
2023-01-12 00:59:01 +00:00
await _checkChangeAddressForTransactions ( ) ;
2023-01-25 19:49:14 +00:00
final String newChangeAddress = await _getCurrentAddressForChain (
1 , DerivePathTypeExt . primaryFor ( coin ) ) ;
2022-10-27 23:24:14 +00:00
int feeBeingPaid =
satoshisBeingUsed - satoshiAmountToSend - changeOutputSize ;
recipientsArray . add ( newChangeAddress ) ;
recipientsAmtArray . add ( changeOutputSize ) ;
// At this point, we have the outputs we're going to use, the amounts to send along with which addresses
// we intend to send these amounts to. We have enough to send instructions to build the transaction.
Logging . instance . log ( ' 2 outputs in tx ' , level: LogLevel . Info ) ;
Logging . instance
. log ( ' Input size: $ satoshisBeingUsed ' , level: LogLevel . Info ) ;
Logging . instance . log ( ' Recipient output size: $ satoshiAmountToSend ' ,
level: LogLevel . Info ) ;
Logging . instance . log ( ' Change Output Size: $ changeOutputSize ' ,
level: LogLevel . Info ) ;
Logging . instance . log (
' Difference (fee being paid): $ feeBeingPaid sats ' ,
level: LogLevel . Info ) ;
Logging . instance
. log ( ' Estimated fee: $ feeForTwoOutputs ' , level: LogLevel . Info ) ;
dynamic txn = await buildTransaction (
utxoSigningData: utxoSigningData ,
recipients: recipientsArray ,
satoshiAmounts: recipientsAmtArray ,
) ;
// make sure minimum fee is accurate if that is being used
if ( txn [ " vSize " ] - feeBeingPaid = = 1 ) {
int changeOutputSize =
satoshisBeingUsed - satoshiAmountToSend - ( txn [ " vSize " ] as int ) ;
feeBeingPaid =
satoshisBeingUsed - satoshiAmountToSend - changeOutputSize ;
recipientsAmtArray . removeLast ( ) ;
recipientsAmtArray . add ( changeOutputSize ) ;
Logging . instance . log ( ' Adjusted Input size: $ satoshisBeingUsed ' ,
level: LogLevel . Info ) ;
Logging . instance . log (
' Adjusted Recipient output size: $ satoshiAmountToSend ' ,
level: LogLevel . Info ) ;
Logging . instance . log (
' Adjusted Change Output Size: $ changeOutputSize ' ,
level: LogLevel . Info ) ;
Logging . instance . log (
' Adjusted Difference (fee being paid): $ feeBeingPaid sats ' ,
level: LogLevel . Info ) ;
Logging . instance . log ( ' Adjusted Estimated fee: $ feeForTwoOutputs ' ,
level: LogLevel . Info ) ;
txn = await buildTransaction (
utxoSigningData: utxoSigningData ,
recipients: recipientsArray ,
satoshiAmounts: recipientsAmtArray ,
) ;
}
Map < String , dynamic > transactionObject = {
" hex " : txn [ " hex " ] ,
" recipient " : recipientsArray [ 0 ] ,
2023-04-11 15:17:58 +00:00
" recipientAmt " : Amount (
rawValue: BigInt . from ( recipientsAmtArray [ 0 ] ) ,
fractionDigits: coin . decimals ,
) ,
2022-10-27 23:24:14 +00:00
" fee " : feeBeingPaid ,
" vSize " : txn [ " vSize " ] ,
2023-03-09 19:49:39 +00:00
" usedUTXOs " : utxoSigningData . map ( ( e ) = > e . utxo ) . toList ( ) ,
2022-10-27 23:24:14 +00:00
} ;
return transactionObject ;
} else {
// Something went wrong here. It either overshot or undershot the estimated fee amount or the changeOutputSize
// is smaller than or equal to DUST_LIMIT. Revert to single output transaction.
Logging . instance . log ( ' 1 output in tx ' , level: LogLevel . Info ) ;
Logging . instance
. log ( ' Input size: $ satoshisBeingUsed ' , level: LogLevel . Info ) ;
Logging . instance . log ( ' Recipient output size: $ satoshiAmountToSend ' ,
level: LogLevel . Info ) ;
Logging . instance . log (
' Difference (fee being paid): ${ satoshisBeingUsed - satoshiAmountToSend } sats ' ,
level: LogLevel . Info ) ;
Logging . instance
. log ( ' Estimated fee: $ feeForOneOutput ' , level: LogLevel . Info ) ;
dynamic txn = await buildTransaction (
utxoSigningData: utxoSigningData ,
recipients: recipientsArray ,
satoshiAmounts: recipientsAmtArray ,
) ;
Map < String , dynamic > transactionObject = {
" hex " : txn [ " hex " ] ,
" recipient " : recipientsArray [ 0 ] ,
2023-04-11 15:17:58 +00:00
" recipientAmt " : Amount (
rawValue: BigInt . from ( recipientsAmtArray [ 0 ] ) ,
fractionDigits: coin . decimals ,
) ,
2022-10-27 23:24:14 +00:00
" fee " : satoshisBeingUsed - satoshiAmountToSend ,
" vSize " : txn [ " vSize " ] ,
2023-03-09 19:49:39 +00:00
" usedUTXOs " : utxoSigningData . map ( ( e ) = > e . utxo ) . toList ( ) ,
2022-10-27 23:24:14 +00:00
} ;
return transactionObject ;
}
} else {
// No additional outputs needed since adding one would mean that it'd be smaller than DUST_LIMIT sats
// which makes it uneconomical to add to the transaction. Here, we pass data directly to instruct
// the wallet to begin crafting the transaction that the user requested.
Logging . instance . log ( ' 1 output in tx ' , level: LogLevel . Info ) ;
Logging . instance
. log ( ' Input size: $ satoshisBeingUsed ' , level: LogLevel . Info ) ;
Logging . instance . log ( ' Recipient output size: $ satoshiAmountToSend ' ,
level: LogLevel . Info ) ;
Logging . instance . log (
' Difference (fee being paid): ${ satoshisBeingUsed - satoshiAmountToSend } sats ' ,
level: LogLevel . Info ) ;
Logging . instance
. log ( ' Estimated fee: $ feeForOneOutput ' , level: LogLevel . Info ) ;
dynamic txn = await buildTransaction (
utxoSigningData: utxoSigningData ,
recipients: recipientsArray ,
satoshiAmounts: recipientsAmtArray ,
) ;
Map < String , dynamic > transactionObject = {
" hex " : txn [ " hex " ] ,
" recipient " : recipientsArray [ 0 ] ,
2023-04-11 15:17:58 +00:00
" recipientAmt " : Amount (
rawValue: BigInt . from ( recipientsAmtArray [ 0 ] ) ,
fractionDigits: coin . decimals ,
) ,
2022-10-27 23:24:14 +00:00
" fee " : satoshisBeingUsed - satoshiAmountToSend ,
" vSize " : txn [ " vSize " ] ,
2023-03-09 19:49:39 +00:00
" usedUTXOs " : utxoSigningData . map ( ( e ) = > e . utxo ) . toList ( ) ,
2022-10-27 23:24:14 +00:00
} ;
return transactionObject ;
}
} else if ( satoshisBeingUsed - satoshiAmountToSend = = feeForOneOutput ) {
// In this scenario, no additional change output is needed since inputs - outputs equal exactly
// what we need to pay for fees. Here, we pass data directly to instruct the wallet to begin
// crafting the transaction that the user requested.
Logging . instance . log ( ' 1 output in tx ' , level: LogLevel . Info ) ;
Logging . instance
. log ( ' Input size: $ satoshisBeingUsed ' , level: LogLevel . Info ) ;
Logging . instance . log ( ' Recipient output size: $ satoshiAmountToSend ' ,
level: LogLevel . Info ) ;
Logging . instance . log (
' Fee being paid: ${ satoshisBeingUsed - satoshiAmountToSend } sats ' ,
level: LogLevel . Info ) ;
Logging . instance
. log ( ' Estimated fee: $ feeForOneOutput ' , level: LogLevel . Info ) ;
dynamic txn = await buildTransaction (
utxoSigningData: utxoSigningData ,
recipients: recipientsArray ,
satoshiAmounts: recipientsAmtArray ,
) ;
Map < String , dynamic > transactionObject = {
" hex " : txn [ " hex " ] ,
" recipient " : recipientsArray [ 0 ] ,
2023-04-11 15:17:58 +00:00
" recipientAmt " : Amount (
rawValue: BigInt . from ( recipientsAmtArray [ 0 ] ) ,
fractionDigits: coin . decimals ,
) ,
2022-10-27 23:24:14 +00:00
" fee " : feeForOneOutput ,
" vSize " : txn [ " vSize " ] ,
2023-03-09 19:49:39 +00:00
" usedUTXOs " : utxoSigningData . map ( ( e ) = > e . utxo ) . toList ( ) ,
2022-10-27 23:24:14 +00:00
} ;
return transactionObject ;
} else {
// Remember that returning 2 indicates that the user does not have a sufficient balance to
// pay for the transaction fee. Ideally, at this stage, we should check if the user has any
// additional outputs they're able to spend and then recalculate fees.
Logging . instance . log (
' Cannot pay tx fee - checking for more outputs and trying again ' ,
level: LogLevel . Warning ) ;
// try adding more outputs
if ( spendableOutputs . length > inputsBeingConsumed ) {
2023-03-08 22:11:46 +00:00
return coinSelection (
satoshiAmountToSend: satoshiAmountToSend ,
selectedTxFeeRate: selectedTxFeeRate ,
2023-06-17 16:42:23 +00:00
satsPerVByte: satsPerVByte ,
2023-03-08 22:11:46 +00:00
recipientAddress: recipientAddress ,
isSendAll: isSendAll ,
additionalOutputs: additionalOutputs + 1 ,
utxos: utxos ,
coinControl: coinControl ,
) ;
2022-10-27 23:24:14 +00:00
}
return 2 ;
}
}
2023-03-09 19:49:39 +00:00
Future < List < SigningData > > fetchBuildTxData (
2023-01-12 00:59:01 +00:00
List < isar_models . UTXO > utxosToUse ,
2022-10-27 23:24:14 +00:00
) async {
// return data
2023-03-09 19:49:39 +00:00
List < SigningData > signingData = [ ] ;
2022-10-27 23:24:14 +00:00
try {
// Populating the addresses to check
for ( var i = 0 ; i < utxosToUse . length ; i + + ) {
2023-03-09 19:49:39 +00:00
if ( utxosToUse [ i ] . address = = null ) {
final txid = utxosToUse [ i ] . txid ;
final tx = await _cachedElectrumXClient . getTransaction (
txHash: txid ,
coin: coin ,
) ;
for ( final output in tx [ " vout " ] as List ) {
final n = output [ " n " ] ;
if ( n ! = null & & n = = utxosToUse [ i ] . vout ) {
utxosToUse [ i ] = utxosToUse [ i ] . copyWith (
address: output [ " scriptPubKey " ] ? [ " addresses " ] ? [ 0 ] as String ? ? ?
output [ " scriptPubKey " ] [ " address " ] as String ,
) ;
2022-10-27 23:24:14 +00:00
}
}
}
2023-03-09 19:49:39 +00:00
final derivePathType = addressType ( address: utxosToUse [ i ] . address ! ) ;
signingData . add (
SigningData (
derivePathType: derivePathType ,
utxo: utxosToUse [ i ] ,
) ,
) ;
2022-10-27 23:24:14 +00:00
}
2023-03-09 19:49:39 +00:00
Map < DerivePathType , Map < String , dynamic > > receiveDerivations = { } ;
Map < DerivePathType , Map < String , dynamic > > changeDerivations = { } ;
for ( final sd in signingData ) {
String ? pubKey ;
String ? wif ;
// fetch receiving derivations if null
receiveDerivations [ sd . derivePathType ] ? ? = await _fetchDerivations (
2022-10-27 23:24:14 +00:00
chain: 0 ,
2023-03-09 19:49:39 +00:00
derivePathType: sd . derivePathType ,
2022-10-27 23:24:14 +00:00
) ;
2023-03-09 19:49:39 +00:00
final receiveDerivation =
receiveDerivations [ sd . derivePathType ] ! [ sd . utxo . address ! ] ;
2022-10-27 23:24:14 +00:00
2023-03-09 19:49:39 +00:00
if ( receiveDerivation ! = null ) {
pubKey = receiveDerivation [ " pubKey " ] as String ;
wif = receiveDerivation [ " wif " ] as String ;
} else {
// fetch change derivations if null
changeDerivations [ sd . derivePathType ] ? ? = await _fetchDerivations (
chain: 1 ,
derivePathType: sd . derivePathType ,
) ;
final changeDerivation =
changeDerivations [ sd . derivePathType ] ! [ sd . utxo . address ! ] ;
if ( changeDerivation ! = null ) {
pubKey = changeDerivation [ " pubKey " ] as String ;
wif = changeDerivation [ " wif " ] as String ;
2022-10-27 23:24:14 +00:00
}
}
2023-03-09 21:08:13 +00:00
if ( wif = = null | | pubKey = = null ) {
final address = await db . getAddress ( walletId , sd . utxo . address ! ) ;
if ( address ? . derivationPath ! = null ) {
final node = await Bip32Utils . getBip32Node (
( await mnemonicString ) ! ,
( await mnemonicPassphrase ) ! ,
_network ,
address ! . derivationPath ! . value ,
) ;
wif = node . toWIF ( ) ;
pubKey = Format . uint8listToString ( node . publicKey ) ;
}
}
2023-03-09 19:49:39 +00:00
if ( wif ! = null & & pubKey ! = null ) {
final PaymentData data ;
final Uint8List ? redeemScript ;
2022-10-27 23:24:14 +00:00
2023-03-09 19:49:39 +00:00
switch ( sd . derivePathType ) {
case DerivePathType . bip44:
data = P2PKH (
data: PaymentData (
pubkey: Format . stringToUint8List ( pubKey ) ,
) ,
network: _network ,
) . data ;
redeemScript = null ;
break ;
2022-10-27 23:24:14 +00:00
2023-03-09 19:49:39 +00:00
case DerivePathType . bip49:
final p2wpkh = P2WPKH (
data: PaymentData (
pubkey: Format . stringToUint8List ( pubKey ) ,
) ,
network: _network ,
overridePrefix: _network . bech32 ! ,
) . data ;
redeemScript = p2wpkh . output ;
data = P2SH (
data: PaymentData ( redeem: p2wpkh ) ,
network: _network ,
) . data ;
break ;
2022-10-27 23:24:14 +00:00
2023-03-09 19:49:39 +00:00
case DerivePathType . bip84:
data = P2WPKH (
data: PaymentData (
pubkey: Format . stringToUint8List ( pubKey ) ,
2022-10-27 23:24:14 +00:00
) ,
2023-03-09 19:49:39 +00:00
network: _network ,
overridePrefix: _network . bech32 ! ,
) . data ;
redeemScript = null ;
break ;
default :
throw Exception ( " DerivePathType unsupported " ) ;
2022-10-27 23:24:14 +00:00
}
2023-03-09 19:49:39 +00:00
final keyPair = ECPair . fromWIF (
wif ,
network: _network ,
) ;
2022-10-27 23:24:14 +00:00
2023-03-09 19:49:39 +00:00
sd . redeemScript = redeemScript ;
sd . output = data . output ;
sd . keyPair = keyPair ;
2022-10-27 23:24:14 +00:00
}
}
2023-03-09 19:49:39 +00:00
return signingData ;
2022-10-27 23:24:14 +00:00
} catch ( e , s ) {
Logging . instance
. log ( " fetchBuildTxData() threw: $ e , \n $ s " , level: LogLevel . Error ) ;
rethrow ;
}
}
/// Builds and signs a transaction
Future < Map < String , dynamic > > buildTransaction ( {
2023-03-09 19:49:39 +00:00
required List < SigningData > utxoSigningData ,
2022-10-27 23:24:14 +00:00
required List < String > recipients ,
required List < int > satoshiAmounts ,
} ) async {
Logging . instance
. log ( " Starting buildTransaction ---------- " , level: LogLevel . Info ) ;
final txb = TransactionBuilder ( network: _network ) ;
txb . setVersion ( 1 ) ;
// Add transaction inputs
2023-03-09 19:49:39 +00:00
for ( var i = 0 ; i < utxoSigningData . length ; i + + ) {
final txid = utxoSigningData [ i ] . utxo . txid ;
txb . addInput (
txid ,
utxoSigningData [ i ] . utxo . vout ,
null ,
utxoSigningData [ i ] . output ! ,
_network . bech32 ! ,
) ;
2022-10-27 23:24:14 +00:00
}
// Add transaction output
for ( var i = 0 ; i < recipients . length ; i + + ) {
2022-10-28 18:03:52 +00:00
txb . addOutput ( recipients [ i ] , satoshiAmounts [ i ] , _network . bech32 ! ) ;
2022-10-27 23:24:14 +00:00
}
try {
// Sign the transaction accordingly
2023-03-09 19:49:39 +00:00
for ( var i = 0 ; i < utxoSigningData . length ; i + + ) {
2022-10-27 23:24:14 +00:00
txb . sign (
2023-03-09 19:49:39 +00:00
vin: i ,
keyPair: utxoSigningData [ i ] . keyPair ! ,
witnessValue: utxoSigningData [ i ] . utxo . value ,
redeemScript: utxoSigningData [ i ] . redeemScript ,
overridePrefix: _network . bech32 ! ,
) ;
2022-10-27 23:24:14 +00:00
}
} catch ( e , s ) {
Logging . instance . log ( " Caught exception while signing transaction: $ e \n $ s " ,
level: LogLevel . Error ) ;
rethrow ;
}
2022-10-28 18:03:52 +00:00
final builtTx = txb . build ( _network . bech32 ! ) ;
2022-10-27 23:24:14 +00:00
final vSize = builtTx . virtualSize ( ) ;
return { " hex " : builtTx . toHex ( ) , " vSize " : vSize } ;
}
@ override
Future < void > fullRescan (
int maxUnusedAddressGap ,
int maxNumberOfIndexesToCheck ,
) async {
Logging . instance . log ( " Starting full rescan! " , level: LogLevel . Info ) ;
longMutex = true ;
GlobalEventBus . instance . fire (
WalletSyncStatusChangedEvent (
WalletSyncStatus . syncing ,
walletId ,
coin ,
) ,
) ;
// clear cache
await _cachedElectrumXClient . clearSharedTransactionCache ( coin: coin ) ;
// back up data
2023-01-12 21:32:25 +00:00
// await _rescanBackup();
2022-10-27 23:24:14 +00:00
2023-01-13 21:47:56 +00:00
// clear blockchain info
2023-01-16 21:04:03 +00:00
await db . deleteWalletBlockchainData ( walletId ) ;
2023-01-17 14:19:30 +00:00
await _deleteDerivations ( ) ;
2023-01-13 21:47:56 +00:00
2022-10-27 23:24:14 +00:00
try {
2023-02-03 22:34:06 +00:00
final _mnemonic = await mnemonicString ;
final _mnemonicPassphrase = await mnemonicPassphrase ;
2023-02-13 22:53:28 +00:00
if ( _mnemonicPassphrase = = null ) {
Logging . instance . log (
" Exception in fullRescan: mnemonic passphrase null, possible migration issue; if using internal builds, delete wallet and restore from seed, if using a release build, please file bug report " ,
level: LogLevel . Error ) ;
}
2023-02-03 22:34:06 +00:00
2022-10-27 23:24:14 +00:00
await _recoverWalletFromBIP32SeedPhrase (
2023-02-03 22:34:06 +00:00
mnemonic: _mnemonic ! ,
mnemonicPassphrase: _mnemonicPassphrase ! ,
2022-10-27 23:24:14 +00:00
maxUnusedAddressGap: maxUnusedAddressGap ,
maxNumberOfIndexesToCheck: maxNumberOfIndexesToCheck ,
2023-02-02 15:03:57 +00:00
isRescan: true ,
2022-10-27 23:24:14 +00:00
) ;
longMutex = false ;
2023-01-19 16:29:00 +00:00
await refresh ( ) ;
2022-10-27 23:24:14 +00:00
Logging . instance . log ( " Full rescan complete! " , level: LogLevel . Info ) ;
GlobalEventBus . instance . fire (
WalletSyncStatusChangedEvent (
WalletSyncStatus . synced ,
walletId ,
coin ,
) ,
) ;
} catch ( e , s ) {
GlobalEventBus . instance . fire (
WalletSyncStatusChangedEvent (
WalletSyncStatus . unableToSync ,
walletId ,
coin ,
) ,
) ;
// restore from backup
2023-01-12 21:32:25 +00:00
// await _rescanRestore();
2022-10-27 23:24:14 +00:00
longMutex = false ;
Logging . instance . log ( " Exception rethrown from fullRescan(): $ e \n $ s " ,
level: LogLevel . Error ) ;
rethrow ;
}
}
2023-01-17 14:19:30 +00:00
Future < void > _deleteDerivations ( ) async {
// P2PKH derivations
await _secureStore . delete ( key: " ${ walletId } _receiveDerivationsP2PKH " ) ;
await _secureStore . delete ( key: " ${ walletId } _changeDerivationsP2PKH " ) ;
// P2SH derivations
await _secureStore . delete ( key: " ${ walletId } _receiveDerivationsP2SH " ) ;
await _secureStore . delete ( key: " ${ walletId } _changeDerivationsP2SH " ) ;
// P2WPKH derivations
await _secureStore . delete ( key: " ${ walletId } _receiveDerivationsP2WPKH " ) ;
await _secureStore . delete ( key: " ${ walletId } _changeDerivationsP2WPKH " ) ;
}
2023-01-12 21:32:25 +00:00
// Future<void> _rescanRestore() async {
// Logging.instance.log("starting rescan restore", level: LogLevel.Info);
//
// // restore from backup
// // p2pkh
// final tempReceivingAddressesP2PKH = DB.instance
// .get<dynamic>(boxName: walletId, key: 'receivingAddressesP2PKH_BACKUP');
// final tempChangeAddressesP2PKH = DB.instance
// .get<dynamic>(boxName: walletId, key: 'changeAddressesP2PKH_BACKUP');
// final tempReceivingIndexP2PKH = DB.instance
// .get<dynamic>(boxName: walletId, key: 'receivingIndexP2PKH_BACKUP');
// final tempChangeIndexP2PKH = DB.instance
// .get<dynamic>(boxName: walletId, key: 'changeIndexP2PKH_BACKUP');
// await DB.instance.put<dynamic>(
// boxName: walletId,
// key: 'receivingAddressesP2PKH',
// value: tempReceivingAddressesP2PKH);
// await DB.instance.put<dynamic>(
// boxName: walletId,
// key: 'changeAddressesP2PKH',
// value: tempChangeAddressesP2PKH);
// await DB.instance.put<dynamic>(
// boxName: walletId,
// key: 'receivingIndexP2PKH',
// value: tempReceivingIndexP2PKH);
// await DB.instance.put<dynamic>(
// boxName: walletId,
// key: 'changeIndexP2PKH',
// value: tempChangeIndexP2PKH);
// await DB.instance.delete<dynamic>(
// key: 'receivingAddressesP2PKH_BACKUP', boxName: walletId);
// await DB.instance
// .delete<dynamic>(key: 'changeAddressesP2PKH_BACKUP', boxName: walletId);
// await DB.instance
// .delete<dynamic>(key: 'receivingIndexP2PKH_BACKUP', boxName: walletId);
// await DB.instance
// .delete<dynamic>(key: 'changeIndexP2PKH_BACKUP', boxName: walletId);
//
// // p2Sh
// final tempReceivingAddressesP2SH = DB.instance
// .get<dynamic>(boxName: walletId, key: 'receivingAddressesP2SH_BACKUP');
// final tempChangeAddressesP2SH = DB.instance
// .get<dynamic>(boxName: walletId, key: 'changeAddressesP2SH_BACKUP');
// final tempReceivingIndexP2SH = DB.instance
// .get<dynamic>(boxName: walletId, key: 'receivingIndexP2SH_BACKUP');
// final tempChangeIndexP2SH = DB.instance
// .get<dynamic>(boxName: walletId, key: 'changeIndexP2SH_BACKUP');
// await DB.instance.put<dynamic>(
// boxName: walletId,
// key: 'receivingAddressesP2SH',
// value: tempReceivingAddressesP2SH);
// await DB.instance.put<dynamic>(
// boxName: walletId,
// key: 'changeAddressesP2SH',
// value: tempChangeAddressesP2SH);
// await DB.instance.put<dynamic>(
// boxName: walletId,
// key: 'receivingIndexP2SH',
// value: tempReceivingIndexP2SH);
// await DB.instance.put<dynamic>(
// boxName: walletId, key: 'changeIndexP2SH', value: tempChangeIndexP2SH);
// await DB.instance.delete<dynamic>(
// key: 'receivingAddressesP2SH_BACKUP', boxName: walletId);
// await DB.instance
// .delete<dynamic>(key: 'changeAddressesP2SH_BACKUP', boxName: walletId);
// await DB.instance
// .delete<dynamic>(key: 'receivingIndexP2SH_BACKUP', boxName: walletId);
// await DB.instance
// .delete<dynamic>(key: 'changeIndexP2SH_BACKUP', boxName: walletId);
//
// // p2wpkh
// final tempReceivingAddressesP2WPKH = DB.instance.get<dynamic>(
// boxName: walletId, key: 'receivingAddressesP2WPKH_BACKUP');
// final tempChangeAddressesP2WPKH = DB.instance
// .get<dynamic>(boxName: walletId, key: 'changeAddressesP2WPKH_BACKUP');
// final tempReceivingIndexP2WPKH = DB.instance
// .get<dynamic>(boxName: walletId, key: 'receivingIndexP2WPKH_BACKUP');
// final tempChangeIndexP2WPKH = DB.instance
// .get<dynamic>(boxName: walletId, key: 'changeIndexP2WPKH_BACKUP');
// await DB.instance.put<dynamic>(
// boxName: walletId,
// key: 'receivingAddressesP2WPKH',
// value: tempReceivingAddressesP2WPKH);
// await DB.instance.put<dynamic>(
// boxName: walletId,
// key: 'changeAddressesP2WPKH',
// value: tempChangeAddressesP2WPKH);
// await DB.instance.put<dynamic>(
// boxName: walletId,
// key: 'receivingIndexP2WPKH',
// value: tempReceivingIndexP2WPKH);
// await DB.instance.put<dynamic>(
// boxName: walletId,
// key: 'changeIndexP2WPKH',
// value: tempChangeIndexP2WPKH);
// await DB.instance.delete<dynamic>(
// key: 'receivingAddressesP2WPKH_BACKUP', boxName: walletId);
// await DB.instance.delete<dynamic>(
// key: 'changeAddressesP2WPKH_BACKUP', boxName: walletId);
// await DB.instance
// .delete<dynamic>(key: 'receivingIndexP2WPKH_BACKUP', boxName: walletId);
// await DB.instance
// .delete<dynamic>(key: 'changeIndexP2WPKH_BACKUP', boxName: walletId);
//
// // P2PKH derivations
// final p2pkhReceiveDerivationsString = await _secureStore.read(
// key: "${walletId}_receiveDerivationsP2PKH_BACKUP");
// final p2pkhChangeDerivationsString = await _secureStore.read(
// key: "${walletId}_changeDerivationsP2PKH_BACKUP");
//
// await _secureStore.write(
// key: "${walletId}_receiveDerivationsP2PKH",
// value: p2pkhReceiveDerivationsString);
// await _secureStore.write(
// key: "${walletId}_changeDerivationsP2PKH",
// value: p2pkhChangeDerivationsString);
//
// await _secureStore.delete(
// key: "${walletId}_receiveDerivationsP2PKH_BACKUP");
// await _secureStore.delete(key: "${walletId}_changeDerivationsP2PKH_BACKUP");
//
// // P2SH derivations
// final p2shReceiveDerivationsString = await _secureStore.read(
// key: "${walletId}_receiveDerivationsP2SH_BACKUP");
// final p2shChangeDerivationsString = await _secureStore.read(
// key: "${walletId}_changeDerivationsP2SH_BACKUP");
//
// await _secureStore.write(
// key: "${walletId}_receiveDerivationsP2SH",
// value: p2shReceiveDerivationsString);
// await _secureStore.write(
// key: "${walletId}_changeDerivationsP2SH",
// value: p2shChangeDerivationsString);
//
// await _secureStore.delete(key: "${walletId}_receiveDerivationsP2SH_BACKUP");
// await _secureStore.delete(key: "${walletId}_changeDerivationsP2SH_BACKUP");
//
// // P2WPKH derivations
// final p2wpkhReceiveDerivationsString = await _secureStore.read(
// key: "${walletId}_receiveDerivationsP2WPKH_BACKUP");
// final p2wpkhChangeDerivationsString = await _secureStore.read(
// key: "${walletId}_changeDerivationsP2WPKH_BACKUP");
//
// await _secureStore.write(
// key: "${walletId}_receiveDerivationsP2WPKH",
// value: p2wpkhReceiveDerivationsString);
// await _secureStore.write(
// key: "${walletId}_changeDerivationsP2WPKH",
// value: p2wpkhChangeDerivationsString);
//
// await _secureStore.delete(
// key: "${walletId}_receiveDerivationsP2WPKH_BACKUP");
// await _secureStore.delete(
// key: "${walletId}_changeDerivationsP2WPKH_BACKUP");
//
// // UTXOs
// final utxoData = DB.instance
// .get<dynamic>(boxName: walletId, key: 'latest_utxo_model_BACKUP');
// await DB.instance.put<dynamic>(
// boxName: walletId, key: 'latest_utxo_model', value: utxoData);
// await DB.instance
// .delete<dynamic>(key: 'latest_utxo_model_BACKUP', boxName: walletId);
//
// Logging.instance.log("rescan restore complete", level: LogLevel.Info);
// }
//
// Future<void> _rescanBackup() async {
// Logging.instance.log("starting rescan backup", level: LogLevel.Info);
//
// // backup current and clear data
// // p2pkh
// final tempReceivingAddressesP2PKH = DB.instance
// .get<dynamic>(boxName: walletId, key: 'receivingAddressesP2PKH');
// await DB.instance.put<dynamic>(
// boxName: walletId,
// key: 'receivingAddressesP2PKH_BACKUP',
// value: tempReceivingAddressesP2PKH);
// await DB.instance
// .delete<dynamic>(key: 'receivingAddressesP2PKH', boxName: walletId);
//
// final tempChangeAddressesP2PKH = DB.instance
// .get<dynamic>(boxName: walletId, key: 'changeAddressesP2PKH');
// await DB.instance.put<dynamic>(
// boxName: walletId,
// key: 'changeAddressesP2PKH_BACKUP',
// value: tempChangeAddressesP2PKH);
// await DB.instance
// .delete<dynamic>(key: 'changeAddressesP2PKH', boxName: walletId);
//
// final tempReceivingIndexP2PKH =
// DB.instance.get<dynamic>(boxName: walletId, key: 'receivingIndexP2PKH');
// await DB.instance.put<dynamic>(
// boxName: walletId,
// key: 'receivingIndexP2PKH_BACKUP',
// value: tempReceivingIndexP2PKH);
// await DB.instance
// .delete<dynamic>(key: 'receivingIndexP2PKH', boxName: walletId);
//
// final tempChangeIndexP2PKH =
// DB.instance.get<dynamic>(boxName: walletId, key: 'changeIndexP2PKH');
// await DB.instance.put<dynamic>(
// boxName: walletId,
// key: 'changeIndexP2PKH_BACKUP',
// value: tempChangeIndexP2PKH);
// await DB.instance
// .delete<dynamic>(key: 'changeIndexP2PKH', boxName: walletId);
//
// // p2sh
// final tempReceivingAddressesP2SH = DB.instance
// .get<dynamic>(boxName: walletId, key: 'receivingAddressesP2SH');
// await DB.instance.put<dynamic>(
// boxName: walletId,
// key: 'receivingAddressesP2SH_BACKUP',
// value: tempReceivingAddressesP2SH);
// await DB.instance
// .delete<dynamic>(key: 'receivingAddressesP2SH', boxName: walletId);
//
// final tempChangeAddressesP2SH =
// DB.instance.get<dynamic>(boxName: walletId, key: 'changeAddressesP2SH');
// await DB.instance.put<dynamic>(
// boxName: walletId,
// key: 'changeAddressesP2SH_BACKUP',
// value: tempChangeAddressesP2SH);
// await DB.instance
// .delete<dynamic>(key: 'changeAddressesP2SH', boxName: walletId);
//
// final tempReceivingIndexP2SH =
// DB.instance.get<dynamic>(boxName: walletId, key: 'receivingIndexP2SH');
// await DB.instance.put<dynamic>(
// boxName: walletId,
// key: 'receivingIndexP2SH_BACKUP',
// value: tempReceivingIndexP2SH);
// await DB.instance
// .delete<dynamic>(key: 'receivingIndexP2SH', boxName: walletId);
//
// final tempChangeIndexP2SH =
// DB.instance.get<dynamic>(boxName: walletId, key: 'changeIndexP2SH');
// await DB.instance.put<dynamic>(
// boxName: walletId,
// key: 'changeIndexP2SH_BACKUP',
// value: tempChangeIndexP2SH);
// await DB.instance
// .delete<dynamic>(key: 'changeIndexP2SH', boxName: walletId);
//
// // p2wpkh
// final tempReceivingAddressesP2WPKH = DB.instance
// .get<dynamic>(boxName: walletId, key: 'receivingAddressesP2WPKH');
// await DB.instance.put<dynamic>(
// boxName: walletId,
// key: 'receivingAddressesP2WPKH_BACKUP',
// value: tempReceivingAddressesP2WPKH);
// await DB.instance
// .delete<dynamic>(key: 'receivingAddressesP2WPKH', boxName: walletId);
//
// final tempChangeAddressesP2WPKH = DB.instance
// .get<dynamic>(boxName: walletId, key: 'changeAddressesP2WPKH');
// await DB.instance.put<dynamic>(
// boxName: walletId,
// key: 'changeAddressesP2WPKH_BACKUP',
// value: tempChangeAddressesP2WPKH);
// await DB.instance
// .delete<dynamic>(key: 'changeAddressesP2WPKH', boxName: walletId);
//
// final tempReceivingIndexP2WPKH = DB.instance
// .get<dynamic>(boxName: walletId, key: 'receivingIndexP2WPKH');
// await DB.instance.put<dynamic>(
// boxName: walletId,
// key: 'receivingIndexP2WPKH_BACKUP',
// value: tempReceivingIndexP2WPKH);
// await DB.instance
// .delete<dynamic>(key: 'receivingIndexP2WPKH', boxName: walletId);
//
// final tempChangeIndexP2WPKH =
// DB.instance.get<dynamic>(boxName: walletId, key: 'changeIndexP2WPKH');
// await DB.instance.put<dynamic>(
// boxName: walletId,
// key: 'changeIndexP2WPKH_BACKUP',
// value: tempChangeIndexP2WPKH);
// await DB.instance
// .delete<dynamic>(key: 'changeIndexP2WPKH', boxName: walletId);
//
// // P2PKH derivations
// final p2pkhReceiveDerivationsString =
// await _secureStore.read(key: "${walletId}_receiveDerivationsP2PKH");
// final p2pkhChangeDerivationsString =
// await _secureStore.read(key: "${walletId}_changeDerivationsP2PKH");
//
// await _secureStore.write(
// key: "${walletId}_receiveDerivationsP2PKH_BACKUP",
// value: p2pkhReceiveDerivationsString);
// await _secureStore.write(
// key: "${walletId}_changeDerivationsP2PKH_BACKUP",
// value: p2pkhChangeDerivationsString);
//
// await _secureStore.delete(key: "${walletId}_receiveDerivationsP2PKH");
// await _secureStore.delete(key: "${walletId}_changeDerivationsP2PKH");
//
// // P2SH derivations
// final p2shReceiveDerivationsString =
// await _secureStore.read(key: "${walletId}_receiveDerivationsP2SH");
// final p2shChangeDerivationsString =
// await _secureStore.read(key: "${walletId}_changeDerivationsP2SH");
//
// await _secureStore.write(
// key: "${walletId}_receiveDerivationsP2SH_BACKUP",
// value: p2shReceiveDerivationsString);
// await _secureStore.write(
// key: "${walletId}_changeDerivationsP2SH_BACKUP",
// value: p2shChangeDerivationsString);
//
// await _secureStore.delete(key: "${walletId}_receiveDerivationsP2SH");
// await _secureStore.delete(key: "${walletId}_changeDerivationsP2SH");
//
// // P2WPKH derivations
// final p2wpkhReceiveDerivationsString =
// await _secureStore.read(key: "${walletId}_receiveDerivationsP2WPKH");
// final p2wpkhChangeDerivationsString =
// await _secureStore.read(key: "${walletId}_changeDerivationsP2WPKH");
//
// await _secureStore.write(
// key: "${walletId}_receiveDerivationsP2WPKH_BACKUP",
// value: p2wpkhReceiveDerivationsString);
// await _secureStore.write(
// key: "${walletId}_changeDerivationsP2WPKH_BACKUP",
// value: p2wpkhChangeDerivationsString);
//
// await _secureStore.delete(key: "${walletId}_receiveDerivationsP2WPKH");
// await _secureStore.delete(key: "${walletId}_changeDerivationsP2WPKH");
//
// // UTXOs
// final utxoData =
// DB.instance.get<dynamic>(boxName: walletId, key: 'latest_utxo_model');
// await DB.instance.put<dynamic>(
// boxName: walletId, key: 'latest_utxo_model_BACKUP', value: utxoData);
// await DB.instance
// .delete<dynamic>(key: 'latest_utxo_model', boxName: walletId);
//
// Logging.instance.log("rescan backup complete", level: LogLevel.Info);
// }
2022-10-27 23:24:14 +00:00
bool isActive = false ;
@ override
void Function ( bool ) ? get onIsActiveWalletChanged = >
( isActive ) = > this . isActive = isActive ;
@ override
2023-04-05 22:06:31 +00:00
Future < Amount > estimateFeeFor ( Amount amount , int feeRate ) async {
2023-01-12 00:59:01 +00:00
final available = balance . spendable ;
2022-10-27 23:24:14 +00:00
2023-04-05 22:06:31 +00:00
if ( available = = amount ) {
return amount - ( await sweepAllEstimate ( feeRate ) ) ;
} else if ( amount < = Amount . zero | | amount > available ) {
2022-10-27 23:24:14 +00:00
return roughFeeEstimate ( 1 , 2 , feeRate ) ;
}
2023-04-05 22:06:31 +00:00
Amount runningBalance = Amount (
rawValue: BigInt . zero ,
fractionDigits: coin . decimals ,
) ;
2022-10-27 23:24:14 +00:00
int inputCount = 0 ;
2023-01-12 00:59:01 +00:00
for ( final output in ( await utxos ) ) {
if ( ! output . isBlocked ) {
2023-04-05 22:06:31 +00:00
runningBalance = runningBalance +
Amount (
rawValue: BigInt . from ( output . value ) ,
fractionDigits: coin . decimals ,
) ;
2023-01-12 00:59:01 +00:00
inputCount + + ;
2023-04-05 22:06:31 +00:00
if ( runningBalance > amount ) {
2023-01-12 00:59:01 +00:00
break ;
}
2022-10-27 23:24:14 +00:00
}
}
final oneOutPutFee = roughFeeEstimate ( inputCount , 1 , feeRate ) ;
final twoOutPutFee = roughFeeEstimate ( inputCount , 2 , feeRate ) ;
2023-04-05 22:06:31 +00:00
if ( runningBalance - amount > oneOutPutFee ) {
if ( runningBalance - amount > oneOutPutFee + DUST_LIMIT ) {
final change = runningBalance - amount - twoOutPutFee ;
2022-10-27 23:24:14 +00:00
if ( change > DUST_LIMIT & &
2023-04-05 22:06:31 +00:00
runningBalance - amount - change = = twoOutPutFee ) {
return runningBalance - amount - change ;
2022-10-27 23:24:14 +00:00
} else {
2023-04-05 22:06:31 +00:00
return runningBalance - amount ;
2022-10-27 23:24:14 +00:00
}
} else {
2023-04-05 22:06:31 +00:00
return runningBalance - amount ;
2022-10-27 23:24:14 +00:00
}
2023-04-05 22:06:31 +00:00
} else if ( runningBalance - amount = = oneOutPutFee ) {
2022-10-27 23:24:14 +00:00
return oneOutPutFee ;
} else {
return twoOutPutFee ;
}
}
2023-04-05 22:06:31 +00:00
Amount roughFeeEstimate ( int inputCount , int outputCount , int feeRatePerKB ) {
return Amount (
rawValue: BigInt . from (
( ( 42 + ( 272 * inputCount ) + ( 128 * outputCount ) ) / 4 ) . ceil ( ) *
( feeRatePerKB / 1000 ) . ceil ( ) ) ,
fractionDigits: coin . decimals ,
) ;
2022-10-27 23:24:14 +00:00
}
2023-04-05 22:06:31 +00:00
Future < Amount > sweepAllEstimate ( int feeRate ) async {
2022-10-27 23:24:14 +00:00
int available = 0 ;
int inputCount = 0 ;
2023-01-12 00:59:01 +00:00
for ( final output in ( await utxos ) ) {
if ( ! output . isBlocked & &
output . isConfirmed ( storedChainHeight , MINIMUM_CONFIRMATIONS ) ) {
2022-10-27 23:24:14 +00:00
available + = output . value ;
inputCount + + ;
}
}
// transaction will only have 1 output minus the fee
final estimatedFee = roughFeeEstimate ( inputCount , 1 , feeRate ) ;
2023-04-05 22:06:31 +00:00
return Amount (
rawValue: BigInt . from ( available ) ,
fractionDigits: coin . decimals ,
) -
estimatedFee ;
2022-10-27 23:24:14 +00:00
}
@ override
Future < bool > generateNewAddress ( ) async {
try {
2023-01-12 00:59:01 +00:00
final currentReceiving = await _currentReceivingAddress ;
final newReceivingIndex = currentReceiving . derivationIndex + 1 ;
// Use new index to derive a new receiving address
2022-10-27 23:24:14 +00:00
final newReceivingAddress = await _generateAddressForChain (
2023-01-25 19:49:14 +00:00
0 , newReceivingIndex , DerivePathTypeExt . primaryFor ( coin ) ) ;
2023-01-12 00:59:01 +00:00
// Add that new receiving address
2023-01-16 21:04:03 +00:00
await db . putAddress ( newReceivingAddress ) ;
2022-10-27 23:24:14 +00:00
return true ;
} catch ( e , s ) {
Logging . instance . log (
" Exception rethrown from generateNewAddress(): $ e \n $ s " ,
level: LogLevel . Error ) ;
return false ;
}
}
2023-04-08 00:44:43 +00:00
@ override
Future < String > get xpub async {
final node = await Bip32Utils . getBip32Root (
( await mnemonic ) . join ( " " ) ,
await mnemonicPassphrase ? ? " " ,
_network ,
) ;
return node . neutered ( ) . toBase58 ( ) ;
}
2022-10-27 23:24:14 +00:00
}
2022-10-28 18:03:52 +00:00
final litecoin = NetworkType (
messagePrefix: ' \x19 Litecoin Signed Message: \n ' ,
bech32: ' ltc ' ,
bip32: Bip32Type ( public: 0x0488b21e , private: 0x0488ade4 ) ,
pubKeyHash: 0x30 ,
scriptHash: 0x32 ,
wif: 0xb0 ) ;
final litecointestnet = NetworkType (
messagePrefix: ' \x19 Litecoin Signed Message: \n ' ,
bech32: ' tltc ' ,
bip32: Bip32Type ( public: 0x043587cf , private: 0x04358394 ) ,
pubKeyHash: 0x6f ,
scriptHash: 0x3a ,
wif: 0xef ) ;