diff --git a/.github/workflows/test.yaml b/.github/workflows/test.yaml index 02876ffad..fe6dbfbb0 100644 --- a/.github/workflows/test.yaml +++ b/.github/workflows/test.yaml @@ -54,12 +54,31 @@ jobs: Write-Output "::set-output name=SECRET_FILE_FIRO::$secretFileFiro"; Write-Output "::set-output name=SECRET_FILE_FIRO_HASH::$($secretFileFiroHash.Hash)"; Write-Output "Secret file $secretFileFiro has hash $($secretFileFiroHash.Hash)"; + + $secretFileBitcoinCash = Join-Path -Path $env:GITHUB_WORKSPACE -ChildPath "test/services/coins/bitcoincash/bitcoincash_wallet_test_parameters.dart"; + $encodedBytes = [System.Convert]::FromBase64String($env:BITCOINCASH_TEST); + Set-Content $secretFileBitcoinCash -Value $encodedBytes -AsByteStream; + $secretFileBitcoinCashHash = Get-FileHash $secretFileBitcoinCash; + Write-Output "::set-output name=SECRET_FILE_BITCOINCASH::$secretFileBitcoinCash"; + Write-Output "::set-output name=SECRET_FILE_BITCOINCASH_HASH::$($secretFileBitcoinCashHash.Hash)"; + Write-Output "Secret file $secretFileBitcoinCash has hash $($secretFileBitcoinCashHash.Hash)"; + + $secretFileNamecoin = Join-Path -Path $env:GITHUB_WORKSPACE -ChildPath "test/services/coins/namecoin/namecoin_wallet_test_parameters.dart"; + $encodedBytes = [System.Convert]::FromBase64String($env:NAMECOIN_TEST); + Set-Content $secretFileNamecoin -Value $encodedBytes -AsByteStream; + $secretFileNamecoinHash = Get-FileHash $secretFileNamecoin; + Write-Output "::set-output name=SECRET_FILE_NAMECOIN::$secretFileNamecoin"; + Write-Output "::set-output name=SECRET_FILE_NAMECOIN_HASH::$($secretFileNamecoinHash.Hash)"; + Write-Output "Secret file $secretFileNamecoin has hash $($secretFileNamecoinHash.Hash)"; + shell: pwsh env: CHANGE_NOW: ${{ secrets.CHANGE_NOW }} BITCOIN_TEST: ${{ secrets.BITCOIN_TEST }} DOGECOIN_TEST: ${{ secrets.DOGECOIN_TEST }} FIRO_TEST: ${{ secrets.FIRO_TEST }} + BITCOINCASH_TEST: ${{ secrets.BITCOINCASH_TEST }} + NAMECOIN_TEST: ${{ secrets.NAMECOIN_TEST }} # - name: Analyze # run: flutter analyze - name: Test @@ -76,6 +95,8 @@ jobs: Remove-Item -Path $env:BITCOIN_TEST; Remove-Item -Path $env:DOGECOIN_TEST; Remove-Item -Path $env:FIRO_TEST; + Remove-Item -Path $env:BITCOINCASH_TEST; + Remove-Item -Path $env:NAMECOIN_TEST; shell: pwsh if: always() env: @@ -83,3 +104,5 @@ jobs: BITCOIN_TEST: ${{ steps.secret-file1.outputs.SECRET_FILE_BITCOIN }} DOGECOIN_TEST: ${{ steps.secret-file1.outputs.SECRET_FILE_DOGECOIN }} FIRO_TEST: ${{ steps.secret-file1.outputs.SECRET_FILE_FIRO }} + BITCOINCASH_TEST: ${{ steps.secret-file1.outputs.SECRET_FILE_BITCOINCASH }} + NAMECOIN_TEST: ${{ steps.secret-file1.outputs.SECRET_FILE_NAMECOIN }} diff --git a/assets/svg/coin_icons/Wownero.svg b/assets/svg/coin_icons/Wownero.svg new file mode 100644 index 000000000..f7a90e94c --- /dev/null +++ b/assets/svg/coin_icons/Wownero.svg @@ -0,0 +1,7 @@ + + + + + + + diff --git a/crypto_plugins/flutter_libmonero b/crypto_plugins/flutter_libmonero index 4bffa40cb..8e3afd002 160000 --- a/crypto_plugins/flutter_libmonero +++ b/crypto_plugins/flutter_libmonero @@ -1 +1 @@ -Subproject commit 4bffa40cb60ad3d98cf0ea5b5d819f3f4895dcd6 +Subproject commit 8e3afd002968d21a3de788569356587a70818022 diff --git a/lib/main.dart b/lib/main.dart index be1d11d0c..0be26c39e 100644 --- a/lib/main.dart +++ b/lib/main.dart @@ -8,6 +8,7 @@ import 'package:cw_core/wallet_type.dart'; import 'package:flutter/material.dart'; import 'package:flutter/services.dart'; import 'package:flutter_libmonero/monero/monero.dart'; +import 'package:flutter_libmonero/wownero/wownero.dart'; import 'package:flutter_riverpod/flutter_riverpod.dart'; import 'package:google_fonts/google_fonts.dart'; import 'package:hive_flutter/hive_flutter.dart'; @@ -84,7 +85,6 @@ void main() async { appDirectory = (await getLibraryDirectory()); } // FlutterNativeSplash.preserve(widgetsBinding: widgetsBinding); - await Hive.initFlutter(appDirectory.path); if (!(Logging.isArmLinux || Logging.isTestEnv)) { final isar = await Isar.open( [LogSchema], @@ -128,11 +128,14 @@ void main() async { Hive.registerAdapter(NodeAdapter()); - Hive.registerAdapter(WalletInfoAdapter()); + if (!Hive.isAdapterRegistered(WalletInfoAdapter().typeId)) { + Hive.registerAdapter(WalletInfoAdapter()); + } Hive.registerAdapter(WalletTypeAdapter()); Hive.registerAdapter(UnspentCoinsInfoAdapter()); + await Hive.initFlutter(appDirectory.path); await Hive.openBox(DB.boxNameDBInfo); int dbVersion = DB.instance.get( @@ -143,6 +146,7 @@ void main() async { } monero.onStartup(); + wownero.onStartup(); await Hive.openBox(DB.boxNameTheme); diff --git a/lib/pages/add_wallet_views/new_wallet_recovery_phrase_warning_view/new_wallet_recovery_phrase_warning_view.dart b/lib/pages/add_wallet_views/new_wallet_recovery_phrase_warning_view/new_wallet_recovery_phrase_warning_view.dart index 9aa9dc341..83dc43933 100644 --- a/lib/pages/add_wallet_views/new_wallet_recovery_phrase_warning_view/new_wallet_recovery_phrase_warning_view.dart +++ b/lib/pages/add_wallet_views/new_wallet_recovery_phrase_warning_view/new_wallet_recovery_phrase_warning_view.dart @@ -59,7 +59,9 @@ class _NewWalletRecoveryPhraseWarningViewState final _numberOfPhraseWords = coin == Coin.monero ? Constants.seedPhraseWordCountMonero - : Constants.seedPhraseWordCountBip39; + : coin == Coin.wownero + ? 14 + : Constants.seedPhraseWordCountBip39; return MasterScaffold( isDesktop: isDesktop, diff --git a/lib/pages/add_wallet_views/restore_wallet_view/restore_wallet_view.dart b/lib/pages/add_wallet_views/restore_wallet_view/restore_wallet_view.dart index 3aaee6607..68f116a95 100644 --- a/lib/pages/add_wallet_views/restore_wallet_view/restore_wallet_view.dart +++ b/lib/pages/add_wallet_views/restore_wallet_view/restore_wallet_view.dart @@ -200,7 +200,7 @@ class _RestoreWalletViewState extends ConsumerState { // TODO: do actual check to make sure it is a valid mnemonic for monero if (bip39.validateMnemonic(mnemonic) == false && - !(widget.coin == Coin.monero)) { + !(widget.coin == Coin.monero || widget.coin == Coin.wownero)) { unawaited(showFloatingFlushBar( type: FlushBarType.warning, message: "Invalid seed phrase!", diff --git a/lib/pages/exchange_view/send_from_view.dart b/lib/pages/exchange_view/send_from_view.dart index e677ab5c0..586ffc0da 100644 --- a/lib/pages/exchange_view/send_from_view.dart +++ b/lib/pages/exchange_view/send_from_view.dart @@ -85,7 +85,7 @@ class _SendFromViewState extends ConsumerState { height: 8, ), Text( - "You need to send ${amount.toStringAsFixed(coin == Coin.monero ? 12 : 8)} ${coin.ticker}", + "You need to send ${amount.toStringAsFixed(coin == Coin.monero ? Constants.satsPerCoinMonero : coin == Coin.wownero ? Constants.satsPerCoinWownero : Constants.satsPerCoin)} ${coin.ticker}", style: STextStyles.itemSubtitle(context), ), const SizedBox( @@ -307,7 +307,11 @@ class _SendFromCardState extends ConsumerState { "${Format.localizedStringAsFixed( value: snapshot.data!, locale: locale, - decimalPlaces: coin == Coin.monero ? 12 : 8, + decimalPlaces: coin == Coin.monero + ? Constants.satsPerCoinMonero + : coin == Coin.wownero + ? Constants.satsPerCoinWownero + : Constants.satsPerCoin, )} ${coin.ticker}", style: STextStyles.itemSubtitle(context), ); diff --git a/lib/pages/settings_views/global_settings_view/manage_nodes_views/add_edit_node_view.dart b/lib/pages/settings_views/global_settings_view/manage_nodes_views/add_edit_node_view.dart index 2f8eca393..143b1e84d 100644 --- a/lib/pages/settings_views/global_settings_view/manage_nodes_views/add_edit_node_view.dart +++ b/lib/pages/settings_views/global_settings_view/manage_nodes_views/add_edit_node_view.dart @@ -90,6 +90,7 @@ class _AddEditNodeViewState extends ConsumerState { break; case Coin.monero: + case Coin.wownero: try { final uri = Uri.parse(formData.host!); if (uri.scheme.startsWith("http")) { @@ -384,6 +385,7 @@ class _AddEditNodeViewState extends ConsumerState { // strip unused path String address = formData.host!; if (coin == Coin.monero || + coin == Coin.wownero || coin == Coin.epicCash) { if (address.startsWith("http")) { final uri = Uri.parse(address); @@ -539,6 +541,7 @@ class _NodeFormState extends ConsumerState { case Coin.epicCash: case Coin.monero: + case Coin.wownero: return true; } } @@ -699,7 +702,9 @@ class _NodeFormState extends ConsumerState { focusNode: _hostFocusNode, style: STextStyles.field(context), decoration: standardInputDecoration( - (widget.coin != Coin.monero && widget.coin != Coin.epicCash) + (widget.coin != Coin.monero && + widget.coin != Coin.wownero && + widget.coin != Coin.epicCash) ? "IP address" : "Url", _hostFocusNode, @@ -880,7 +885,9 @@ class _NodeFormState extends ConsumerState { const SizedBox( height: 8, ), - if (widget.coin != Coin.monero && widget.coin != Coin.epicCash) + if (widget.coin != Coin.monero && + widget.coin != Coin.wownero && + widget.coin != Coin.epicCash) Row( children: [ GestureDetector( @@ -931,11 +938,15 @@ class _NodeFormState extends ConsumerState { ), ], ), - if (widget.coin != Coin.monero && widget.coin != Coin.epicCash) + if (widget.coin != Coin.monero && + widget.coin != Coin.wownero && + widget.coin != Coin.epicCash) const SizedBox( height: 8, ), - if (widget.coin != Coin.monero && widget.coin != Coin.epicCash) + if (widget.coin != Coin.monero && + widget.coin != Coin.wownero && + widget.coin != Coin.epicCash) Row( children: [ GestureDetector( diff --git a/lib/pages/settings_views/global_settings_view/manage_nodes_views/node_details_view.dart b/lib/pages/settings_views/global_settings_view/manage_nodes_views/node_details_view.dart index 340bd859b..8baddb700 100644 --- a/lib/pages/settings_views/global_settings_view/manage_nodes_views/node_details_view.dart +++ b/lib/pages/settings_views/global_settings_view/manage_nodes_views/node_details_view.dart @@ -81,6 +81,7 @@ class _NodeDetailsViewState extends ConsumerState { break; case Coin.monero: + case Coin.wownero: try { final uri = Uri.parse(node!.host); if (uri.scheme.startsWith("http")) { diff --git a/lib/pages/settings_views/wallet_settings_view/wallet_network_settings_view/wallet_network_settings_view.dart b/lib/pages/settings_views/wallet_settings_view/wallet_network_settings_view/wallet_network_settings_view.dart index 892cd13d8..c4045dc9b 100644 --- a/lib/pages/settings_views/wallet_settings_view/wallet_network_settings_view/wallet_network_settings_view.dart +++ b/lib/pages/settings_views/wallet_settings_view/wallet_network_settings_view/wallet_network_settings_view.dart @@ -12,6 +12,7 @@ import 'package:stackwallet/pages/settings_views/wallet_settings_view/wallet_net import 'package:stackwallet/providers/providers.dart'; import 'package:stackwallet/services/coins/epiccash/epiccash_wallet.dart'; import 'package:stackwallet/services/coins/monero/monero_wallet.dart'; +import 'package:stackwallet/services/coins/wownero/wownero_wallet.dart'; import 'package:stackwallet/services/event_bus/events/global/blocks_remaining_event.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'; @@ -205,7 +206,7 @@ class _WalletNetworkSettingsViewState .getManager(widget.walletId) .coin; - if (coin == Coin.monero || coin == Coin.epicCash) { + if (coin == Coin.monero || coin == Coin.wownero || coin == Coin.epicCash) { _blocksRemainingSubscription = eventBus.on().listen( (event) async { if (event.walletId == widget.walletId) { @@ -271,6 +272,15 @@ class _WalletNetworkSettingsViewState if (_percent < highestPercent) { _percent = highestPercent.clamp(0.0, 1.0); } + } else if (coin == Coin.wownero) { + double highestPercent = (ref + .read(walletsChangeNotifierProvider) + .getManager(widget.walletId) + .wallet as WowneroWallet) + .highestPercentCached; + if (_percent < highestPercent) { + _percent = highestPercent.clamp(0.0, 1.0); + } } else if (coin == Coin.epicCash) { double highestPercent = (ref .read(walletsChangeNotifierProvider) @@ -545,6 +555,7 @@ class _WalletNetworkSettingsViewState ), ), if (coin == Coin.monero || + coin == Coin.wownero || coin == Coin.epicCash) Text( " (Blocks to go: ${_blocksRemaining == -1 ? "?" : _blocksRemaining})", diff --git a/lib/pages/wallet_view/transaction_views/transaction_details_view.dart b/lib/pages/wallet_view/transaction_views/transaction_details_view.dart index 391b5caec..fdb8bf1a9 100644 --- a/lib/pages/wallet_view/transaction_views/transaction_details_view.dart +++ b/lib/pages/wallet_view/transaction_views/transaction_details_view.dart @@ -241,7 +241,9 @@ class _TransactionDetailsViewState "$amountPrefix${Format.localizedStringAsFixed( value: coin == Coin.monero ? (amount / 10000.toDecimal()).toDecimal() - : amount, + : coin == Coin.wownero + ? (amount / 1000.toDecimal()).toDecimal() + : amount, locale: ref.watch( localeServiceChangeNotifierProvider .select((value) => value.locale), @@ -254,7 +256,7 @@ class _TransactionDetailsViewState height: 2, ), SelectableText( - "${Format.localizedStringAsFixed(value: (coin == Coin.monero ? (amount / 10000.toDecimal()).toDecimal() : amount) * ref.watch(priceAnd24hChangeNotifierProvider.select((value) => value.getPrice(coin).item1)), locale: ref.watch( + "${Format.localizedStringAsFixed(value: (coin == Coin.monero ? (amount / 10000.toDecimal()).toDecimal() : coin == Coin.wownero ? (amount / 1000.toDecimal()).toDecimal() : amount) * ref.watch(priceAnd24hChangeNotifierProvider.select((value) => value.getPrice(coin).item1)), locale: ref.watch( localeServiceChangeNotifierProvider .select((value) => value.locale), ), decimalPlaces: 2)} ${ref.watch( @@ -298,14 +300,14 @@ class _TransactionDetailsViewState ], ), ), - if (!(coin == Coin.monero && + if (!((coin == Coin.monero || coin == Coin.wownero) && _transaction.txType.toLowerCase() == "sent") && !((coin == Coin.firo || coin == Coin.firoTestNet) && _transaction.subType == "mint")) const SizedBox( height: 12, ), - if (!(coin == Coin.monero && + if (!((coin == Coin.monero || coin == Coin.wownero) && _transaction.txType.toLowerCase() == "sent") && !((coin == Coin.firo || coin == Coin.firoTestNet) && _transaction.subType == "mint")) @@ -464,7 +466,10 @@ class _TransactionDetailsViewState ? Format.localizedStringAsFixed( value: coin == Coin.monero ? (fee / 10000.toDecimal()).toDecimal() - : fee, + : coin == Coin.wownero + ? (fee / 1000.toDecimal()) + .toDecimal() + : fee, locale: ref.watch( localeServiceChangeNotifierProvider .select((value) => value.locale)), @@ -473,7 +478,9 @@ class _TransactionDetailsViewState : Format.localizedStringAsFixed( value: coin == Coin.monero ? (fee / 10000.toDecimal()).toDecimal() - : fee, + : coin == Coin.wownero + ? (fee / 1000.toDecimal()).toDecimal() + : fee, locale: ref.watch( localeServiceChangeNotifierProvider .select((value) => value.locale)), diff --git a/lib/pages/wallet_view/transaction_views/transaction_search_filter_view.dart b/lib/pages/wallet_view/transaction_views/transaction_search_filter_view.dart index 728706702..8175597f6 100644 --- a/lib/pages/wallet_view/transaction_views/transaction_search_filter_view.dart +++ b/lib/pages/wallet_view/transaction_views/transaction_search_filter_view.dart @@ -755,6 +755,11 @@ class _TransactionSearchViewState .floor() .toBigInt() .toInt(); + } else if (widget.coin == Coin.wownero) { + amount = (amountDecimal * Decimal.fromInt(Constants.satsPerCoinWownero)) + .floor() + .toBigInt() + .toInt(); } else { amount = (amountDecimal * Decimal.fromInt(Constants.satsPerCoin)) .floor() diff --git a/lib/services/coins/coin_service.dart b/lib/services/coins/coin_service.dart index 7a0dec2e1..089b20a33 100644 --- a/lib/services/coins/coin_service.dart +++ b/lib/services/coins/coin_service.dart @@ -9,6 +9,7 @@ import 'package:stackwallet/services/coins/dogecoin/dogecoin_wallet.dart'; import 'package:stackwallet/services/coins/epiccash/epiccash_wallet.dart'; import 'package:stackwallet/services/coins/firo/firo_wallet.dart'; import 'package:stackwallet/services/coins/monero/monero_wallet.dart'; +import 'package:stackwallet/services/coins/wownero/wownero_wallet.dart'; import 'package:stackwallet/services/coins/namecoin/namecoin_wallet.dart'; import 'package:stackwallet/services/transaction_notification_tracker.dart'; import 'package:stackwallet/utilities/enums/coin_enum.dart'; @@ -145,6 +146,14 @@ abstract class CoinServiceAPI { // tracker: tracker, ); + case Coin.wownero: + return WowneroWallet( + walletId: walletId, + walletName: walletName, + coin: coin, + // tracker: tracker, + ); + case Coin.namecoin: return NamecoinWallet( walletId: walletId, diff --git a/lib/services/coins/wownero/wownero_wallet.dart b/lib/services/coins/wownero/wownero_wallet.dart new file mode 100644 index 000000000..7114269ec --- /dev/null +++ b/lib/services/coins/wownero/wownero_wallet.dart @@ -0,0 +1,1560 @@ +import 'dart:async'; +import 'dart:io'; + +import 'package:cw_core/monero_transaction_priority.dart'; +import 'package:cw_core/node.dart'; +import 'package:cw_core/pending_transaction.dart'; +import 'package:cw_core/sync_status.dart'; +import 'package:cw_core/transaction_direction.dart'; +import 'package:cw_core/wallet_base.dart'; +import 'package:cw_core/wallet_credentials.dart'; +import 'package:cw_core/wallet_info.dart'; +import 'package:cw_core/wallet_service.dart'; +import 'package:cw_core/wallet_type.dart'; +import 'package:cw_wownero/api/exceptions/creation_transaction_exception.dart'; +import 'package:cw_wownero/api/wallet.dart'; +import 'package:cw_wownero/pending_wownero_transaction.dart'; +import 'package:cw_wownero/wownero_amount_format.dart'; +import 'package:cw_wownero/wownero_wallet.dart'; +import 'package:dart_numerics/dart_numerics.dart'; +import 'package:decimal/decimal.dart'; +import 'package:flutter/foundation.dart'; +import 'package:flutter_libmonero/core/key_service.dart'; +import 'package:flutter_libmonero/core/wallet_creation_service.dart'; +import 'package:flutter_libmonero/view_model/send/output.dart' + as wownero_output; +import 'package:flutter_libmonero/wownero/wownero.dart'; +import 'package:flutter_secure_storage/flutter_secure_storage.dart'; +import 'package:http/http.dart'; +import 'package:mutex/mutex.dart'; +import 'package:path_provider/path_provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; +import 'package:stackwallet/hive/db.dart'; +import 'package:stackwallet/models/node_model.dart'; +import 'package:stackwallet/models/paymint/fee_object_model.dart'; +import 'package:stackwallet/models/paymint/transactions_model.dart'; +import 'package:stackwallet/models/paymint/utxo_model.dart'; +import 'package:stackwallet/services/coins/coin_service.dart'; +import 'package:stackwallet/services/event_bus/events/global/blocks_remaining_event.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'; +import 'package:stackwallet/services/node_service.dart'; +import 'package:stackwallet/services/price.dart'; +import 'package:stackwallet/utilities/constants.dart'; +import 'package:stackwallet/utilities/default_nodes.dart'; +import 'package:stackwallet/utilities/enums/coin_enum.dart'; +import 'package:stackwallet/utilities/enums/fee_rate_type_enum.dart'; +import 'package:stackwallet/utilities/flutter_secure_storage_interface.dart'; +import 'package:stackwallet/utilities/logger.dart'; +import 'package:stackwallet/utilities/prefs.dart'; + +const int MINIMUM_CONFIRMATIONS = 10; + +//https://github.com/wownero-project/wownero/blob/8361d60aef6e17908658128284899e3a11d808d4/src/cryptonote_config.h#L162 +const String GENESIS_HASH_MAINNET = + "013c01ff0001ffffffffffff03029b2e4c0281c0b02e7c53291a94d1d0cbff8883f8024f5142ee494ffbbd08807121017767aafcde9be00dcfd098715ebcf7f410daebc582fda69d24a28e9d0bc890d1"; +const String GENESIS_HASH_TESTNET = + "013c01ff0001ffffffffffff03029b2e4c0281c0b02e7c53291a94d1d0cbff8883f8024f5142ee494ffbbd08807121017767aafcde9be00dcfd098715ebcf7f410daebc582fda69d24a28e9d0bc890d1"; + +class WowneroWallet extends CoinServiceAPI { + static const integrationTestFlag = + bool.fromEnvironment("IS_INTEGRATION_TEST"); + final _prefs = Prefs.instance; + + Timer? timer; + Timer? wowneroAutosaveTimer; + late Coin _coin; + + late FlutterSecureStorageInterface _secureStore; + + late PriceAPI _priceAPI; + + Future getCurrentNode() async { + return NodeService().getPrimaryNodeFor(coin: coin) ?? + DefaultNodes.getNodeFor(coin); + } + + WowneroWallet( + {required String walletId, + required String walletName, + required Coin coin, + PriceAPI? priceAPI, + FlutterSecureStorageInterface? secureStore}) { + _walletId = walletId; + _walletName = walletName; + _coin = coin; + + _priceAPI = priceAPI ?? PriceAPI(Client()); + _secureStore = + secureStore ?? const SecureStorageWrapper(FlutterSecureStorage()); + } + + bool _shouldAutoSync = false; + + @override + bool get shouldAutoSync => _shouldAutoSync; + + @override + set shouldAutoSync(bool shouldAutoSync) { + if (_shouldAutoSync != shouldAutoSync) { + _shouldAutoSync = shouldAutoSync; + if (!shouldAutoSync) { + timer?.cancel(); + wowneroAutosaveTimer?.cancel(); + timer = null; + wowneroAutosaveTimer = null; + stopNetworkAlivePinging(); + } else { + startNetworkAlivePinging(); + // Walletbase needs to be open for this to work + refresh(); + } + } + } + + @override + Future updateNode(bool shouldRefresh) async { + final node = await getCurrentNode(); + + final host = Uri.parse(node.host).host; + await walletBase?.connectToNode( + node: Node(uri: "$host:${node.port}", type: WalletType.wownero)); + + // TODO: is this sync call needed? Do we need to notify ui here? + await walletBase?.startSync(); + + if (shouldRefresh) { + await refresh(); + } + } + + Future> _getMnemonicList() async { + final mnemonicString = + await _secureStore.read(key: '${_walletId}_mnemonic'); + if (mnemonicString == null) { + return []; + } + final List data = mnemonicString.split(' '); + return data; + } + + @override + Future> get mnemonic => _getMnemonicList(); + + Future get currentNodeHeight async { + try { + if (walletBase!.syncStatus! is SyncedSyncStatus && + walletBase!.syncStatus!.progress() == 1.0) { + return await walletBase!.getNodeHeight(); + } + } catch (e, s) {} + int _height = -1; + try { + _height = (walletBase!.syncStatus as SyncingSyncStatus).height; + } catch (e, s) { + Logging.instance.log("$e $s", level: LogLevel.Warning); + } + + int blocksRemaining = -1; + + try { + blocksRemaining = + (walletBase!.syncStatus as SyncingSyncStatus).blocksLeft; + } catch (e, s) { + Logging.instance.log("$e $s", level: LogLevel.Warning); + } + int currentHeight = _height + blocksRemaining; + if (_height == -1 || blocksRemaining == -1) { + currentHeight = int64MaxValue; + } + final cachedHeight = DB.instance + .get(boxName: walletId, key: "storedNodeHeight") as int? ?? + 0; + + if (currentHeight > cachedHeight && currentHeight != int64MaxValue) { + await DB.instance.put( + boxName: walletId, key: "storedNodeHeight", value: currentHeight); + return currentHeight; + } else { + return cachedHeight; + } + } + + Future get currentSyncingHeight async { + //TODO return the tip of the wownero blockchain + try { + if (walletBase!.syncStatus! is SyncedSyncStatus && + walletBase!.syncStatus!.progress() == 1.0) { + Logging.instance + .log("currentSyncingHeight lol", level: LogLevel.Warning); + return getSyncingHeight(); + } + } catch (e, s) {} + int syncingHeight = -1; + try { + syncingHeight = (walletBase!.syncStatus as SyncingSyncStatus).height; + } catch (e, s) { + Logging.instance.log("$e $s", level: LogLevel.Warning); + } + final cachedHeight = + DB.instance.get(boxName: walletId, key: "storedSyncingHeight") + as int? ?? + 0; + + if (syncingHeight > cachedHeight) { + await DB.instance.put( + boxName: walletId, key: "storedSyncingHeight", value: syncingHeight); + return syncingHeight; + } else { + return cachedHeight; + } + } + + Future updateStoredChainHeight({required int newHeight}) async { + await DB.instance.put( + boxName: walletId, key: "storedChainHeight", value: newHeight); + } + + int get storedChainHeight { + return DB.instance.get(boxName: walletId, key: "storedChainHeight") + as int? ?? + 0; + } + + /// Increases the index for either the internal or external chain, depending on [chain]. + /// [chain] - Use 0 for receiving (external), 1 for change (internal). Should not be any other value! + Future _incrementAddressIndexForChain(int chain) async { + // Here we assume chain == 1 if it isn't 0 + String indexKey = chain == 0 ? "receivingIndex" : "changeIndex"; + + final newIndex = + (DB.instance.get(boxName: walletId, key: indexKey)) + 1; + await DB.instance + .put(boxName: walletId, key: indexKey, value: newIndex); + } + + Future _checkCurrentReceivingAddressesForTransactions() async { + try { + await _checkReceivingAddressForTransactions(); + } catch (e, s) { + Logging.instance.log( + "Exception rethrown from _checkCurrentReceivingAddressesForTransactions(): $e\n$s", + level: LogLevel.Error); + rethrow; + } + } + + Future _checkReceivingAddressForTransactions() async { + try { + int highestIndex = -1; + for (var element + in walletBase!.transactionHistory!.transactions!.entries) { + if (element.value.direction == TransactionDirection.incoming) { + int curAddressIndex = + element.value.additionalInfo!['addressIndex'] as int; + if (curAddressIndex > highestIndex) { + highestIndex = curAddressIndex; + } + } + } + + // Check the new receiving index + String indexKey = "receivingIndex"; + final curIndex = + DB.instance.get(boxName: walletId, key: indexKey) as int; + if (highestIndex >= curIndex) { + // First increment the receiving index + await _incrementAddressIndexForChain(0); + final newReceivingIndex = + DB.instance.get(boxName: walletId, key: indexKey) as int; + + // Use new index to derive a new receiving address + final newReceivingAddress = + await _generateAddressForChain(0, newReceivingIndex); + + // Add that new receiving address to the array of receiving addresses + await _addToAddressesArrayForChain(newReceivingAddress, 0); + + // Set the new receiving address that the service + + _currentReceivingAddress = Future(() => newReceivingAddress); + } + } on SocketException catch (se, s) { + Logging.instance.log( + "SocketException caught in _checkReceivingAddressForTransactions(): $se\n$s", + level: LogLevel.Error); + return; + } catch (e, s) { + Logging.instance.log( + "Exception rethrown from _checkReceivingAddressForTransactions(): $e\n$s", + level: LogLevel.Error); + rethrow; + } + } + + @override + bool get isRefreshing => refreshMutex; + + bool refreshMutex = false; + + Timer? syncPercentTimer; + + Mutex syncHeightMutex = Mutex(); + Future stopSyncPercentTimer() async { + syncPercentTimer?.cancel(); + syncPercentTimer = null; + } + + Future startSyncPercentTimer() async { + if (syncPercentTimer != null) { + return; + } + syncPercentTimer?.cancel(); + GlobalEventBus.instance + .fire(RefreshPercentChangedEvent(highestPercentCached, walletId)); + syncPercentTimer = Timer.periodic(const Duration(seconds: 30), (_) async { + if (syncHeightMutex.isLocked) { + return; + } + await syncHeightMutex.protect(() async { + // int restoreheight = walletBase!.walletInfo.restoreHeight ?? 0; + int _height = await currentSyncingHeight; + int _currentHeight = await currentNodeHeight; + double progress = 0; + try { + progress = walletBase!.syncStatus!.progress(); + } catch (e, s) { + Logging.instance.log("$e $s", level: LogLevel.Warning); + } + + final int blocksRemaining = _currentHeight - _height; + + GlobalEventBus.instance + .fire(BlocksRemainingEvent(blocksRemaining, walletId)); + + if (progress == 1 && _currentHeight > 0 && _height > 0) { + await stopSyncPercentTimer(); + GlobalEventBus.instance.fire( + WalletSyncStatusChangedEvent( + WalletSyncStatus.synced, + walletId, + coin, + ), + ); + return; + } + + // for some reason this can be 0 which screws up the percent calculation + // int64MaxValue is NOT the best value to use here + if (_currentHeight < 1) { + _currentHeight = int64MaxValue; + } + + if (_height < 1) { + _height = 1; + } + + double restorePercent = progress; + double highestPercent = highestPercentCached; + + Logging.instance.log( + "currentSyncingHeight: $_height, nodeHeight: $_currentHeight, restorePercent: $restorePercent, highestPercentCached: $highestPercentCached", + level: LogLevel.Info); + + if (restorePercent > 0 && restorePercent <= 1) { + // if (restorePercent > highestPercent) { + highestPercent = restorePercent; + highestPercentCached = restorePercent; + // } + } + + GlobalEventBus.instance + .fire(RefreshPercentChangedEvent(highestPercent, walletId)); + }); + }); + } + + double get highestPercentCached => + DB.instance.get(boxName: walletId, key: "highestPercentCached") + as double? ?? + 0; + set highestPercentCached(double value) => DB.instance.put( + boxName: walletId, + key: "highestPercentCached", + value: value, + ); + + /// Refreshes display data for the wallet + @override + Future refresh() async { + if (refreshMutex) { + Logging.instance.log("$walletId $walletName refreshMutex denied", + level: LogLevel.Info); + return; + } else { + refreshMutex = true; + } + + if (walletBase == null) { + throw Exception("Tried to call refresh() in wownero without walletBase!"); + } + + try { + await startSyncPercentTimer(); + GlobalEventBus.instance.fire( + WalletSyncStatusChangedEvent( + WalletSyncStatus.syncing, + walletId, + coin, + ), + ); + + final int _currentSyncingHeight = await currentSyncingHeight; + final int storedHeight = storedChainHeight; + int _currentNodeHeight = await currentNodeHeight; + + double progress = 0; + try { + progress = (walletBase!.syncStatus!).progress(); + } catch (e, s) { + Logging.instance.log("$e $s", level: LogLevel.Warning); + } + await _fetchTransactionData(); + + bool stillSyncing = false; + Logging.instance.log( + "storedHeight: $storedHeight, _currentSyncingHeight: $_currentSyncingHeight, _currentNodeHeight: $_currentNodeHeight, progress: $progress, issynced: ${await walletBase!.isConnected()}", + level: LogLevel.Info); + + if (progress < 1.0) { + stillSyncing = true; + } + + if (_currentSyncingHeight > storedHeight) { + // 0 is returned from wownero as I assume an error????? + if (_currentSyncingHeight > 0) { + // 0 failed to fetch current height??? + await updateStoredChainHeight(newHeight: _currentSyncingHeight); + } + } + + await _checkCurrentReceivingAddressesForTransactions(); + String indexKey = "receivingIndex"; + final curIndex = + DB.instance.get(boxName: walletId, key: indexKey) as int; + // Use new index to derive a new receiving address + try { + final newReceivingAddress = await _generateAddressForChain(0, curIndex); + _currentReceivingAddress = Future(() => newReceivingAddress); + } catch (e, s) { + Logging.instance.log( + "Failed to call _generateAddressForChain(0, $curIndex): $e\n$s", + level: LogLevel.Error); + } + final newTxData = await _fetchTransactionData(); + _transactionData = Future(() => newTxData); + + if (isActive || shouldAutoSync) { + timer ??= Timer.periodic(const Duration(seconds: 60), (timer) async { + debugPrint("run timer"); + //TODO: check for new data and refresh if needed. if wownero even needs this + // 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)); + // } + // } + }); + wowneroAutosaveTimer ??= + Timer.periodic(const Duration(seconds: 93), (timer) async { + debugPrint("run wownero timer"); + if (isActive) { + await walletBase?.save(); + GlobalEventBus.instance.fire(UpdatedInBackgroundEvent( + "New data found in $walletId $walletName in background!", + walletId)); + } + }); + } + + if (stillSyncing) { + debugPrint("still syncing"); + GlobalEventBus.instance.fire( + WalletSyncStatusChangedEvent( + WalletSyncStatus.syncing, + walletId, + coin, + ), + ); + refreshMutex = false; + return; + } + await stopSyncPercentTimer(); + GlobalEventBus.instance.fire( + WalletSyncStatusChangedEvent( + WalletSyncStatus.synced, + walletId, + coin, + ), + ); + refreshMutex = false; + } catch (error, strace) { + refreshMutex = false; + await stopSyncPercentTimer(); + 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 + // TODO: implement allOwnAddresses + Future> get allOwnAddresses { + return Future(() => []); + } + + @override + Future get balanceMinusMaxFee async => + (await availableBalance) - + (Decimal.fromInt((await maxFee)) / Decimal.fromInt(Constants.satsPerCoin)) + .toDecimal(); + + @override + Future get currentReceivingAddress => + _currentReceivingAddress ??= _getCurrentAddressForChain(0); + + @override + Future exit() async { + await stopSyncPercentTimer(); + _hasCalledExit = true; + isActive = false; + await walletBase?.save(prioritySave: true); + walletBase?.close(); + wowneroAutosaveTimer?.cancel(); + wowneroAutosaveTimer = null; + timer?.cancel(); + timer = null; + stopNetworkAlivePinging(); + } + + bool _hasCalledExit = false; + + @override + bool get hasCalledExit => _hasCalledExit; + + Future? _currentReceivingAddress; + + Future _getFees() async { + return FeeObject( + numberOfBlocksFast: 10, + numberOfBlocksAverage: 10, + numberOfBlocksSlow: 10, + fast: 4, + medium: 2, + slow: 0); + } + + @override + Future get fees => _feeObject ??= _getFees(); + Future? _feeObject; + + @override + // TODO: implement fullRescan + Future fullRescan( + int maxUnusedAddressGap, + int maxNumberOfIndexesToCheck, + ) async { + var restoreHeight = walletBase?.walletInfo.restoreHeight; + await walletBase?.rescan(height: restoreHeight); + GlobalEventBus.instance.fire( + WalletSyncStatusChangedEvent( + WalletSyncStatus.syncing, + walletId, + coin, + ), + ); + return; + } + + Future _generateAddressForChain(int chain, int index) async { + // + String address = walletBase!.getTransactionAddress(chain, index); + + return address; + } + + /// Adds [address] to the relevant chain's address array, which is determined by [chain]. + /// [address] - Expects a standard native segwit address + /// [chain] - Use 0 for receiving (external), 1 for change (internal). Should not be any other value! + Future _addToAddressesArrayForChain(String address, int chain) async { + String chainArray = ''; + if (chain == 0) { + chainArray = 'receivingAddresses'; + } else { + chainArray = 'changeAddresses'; + } + + final addressArray = + DB.instance.get(boxName: walletId, key: chainArray); + if (addressArray == null) { + Logging.instance.log( + 'Attempting to add the following to $chainArray array for chain $chain:${[ + address + ]}', + level: LogLevel.Info); + await DB.instance + .put(boxName: walletId, key: chainArray, value: [address]); + } else { + // Make a deep copy of the existing list + final List newArray = []; + addressArray + .forEach((dynamic _address) => newArray.add(_address as String)); + newArray.add(address); // Add the address passed into the method + await DB.instance + .put(boxName: walletId, key: chainArray, value: newArray); + } + } + + /// 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 _getCurrentAddressForChain(int chain) async { + // Here, we assume that chain == 1 if it isn't 0 + String arrayKey = chain == 0 ? "receivingAddresses" : "changeAddresses"; + final internalChainArray = (DB.instance + .get(boxName: walletId, key: arrayKey)) as List; + return internalChainArray.last as String; + } + + //TODO: take in the default language when creating wallet. + Future _generateNewWallet() async { + Logging.instance + .log("IS_INTEGRATION_TEST: $integrationTestFlag", level: LogLevel.Info); + // TODO: ping wownero server and make sure the genesis hash matches + // if (!integrationTestFlag) { + // final features = await electrumXClient.getServerFeatures(); + // Logging.instance.log("features: $features"); + // 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!"); + // } + // } + // } + + // this should never fail + if ((await _secureStore.read(key: '${_walletId}_mnemonic')) != null) { + throw Exception( + "Attempted to overwrite mnemonic on generate new wallet!"); + } + + storage = const FlutterSecureStorage(); + // TODO: Wallet Service may need to be switched to Wownero + walletService = + wownero.createWowneroWalletService(DB.instance.moneroWalletInfoBox); + prefs = await SharedPreferences.getInstance(); + keysStorage = KeyService(storage!); + WalletInfo walletInfo; + WalletCredentials credentials; + try { + String name = _walletId; + final dirPath = + await pathForWalletDir(name: name, type: WalletType.wownero); + final path = await pathForWallet(name: name, type: WalletType.wownero); + credentials = wownero.createWowneroNewWalletCredentials( + name: name, + language: "English", + ); + + walletInfo = WalletInfo.external( + id: WalletBase.idFor(name, WalletType.wownero), + name: name, + type: WalletType.wownero, + isRecovery: false, + restoreHeight: credentials.height ?? 0, + date: DateTime.now(), + path: path, + dirPath: dirPath, + // TODO: find out what to put for address + address: ''); + credentials.walletInfo = walletInfo; + + _walletCreationService = WalletCreationService( + secureStorage: storage, + sharedPreferences: prefs, + walletService: walletService, + keyService: keysStorage, + ); + _walletCreationService?.changeWalletType(); + // To restore from a seed + final wallet = await _walletCreationService?.create(credentials); + + // subtract a couple days to ensure we have a buffer for SWB + final bufferedCreateHeight = + getSeedHeightSync(wallet?.seed.trim() as String); + + await DB.instance.put( + boxName: walletId, key: "restoreHeight", value: bufferedCreateHeight); + walletInfo.restoreHeight = bufferedCreateHeight; + + await _secureStore.write( + key: '${_walletId}_mnemonic', value: wallet?.seed.trim()); + walletInfo.address = wallet?.walletAddresses.address; + await DB.instance + .add(boxName: WalletInfo.boxName, value: walletInfo); + walletBase?.close(); + walletBase = wallet as WowneroWalletBase; + } catch (e, s) { + debugPrint(e.toString()); + debugPrint(s.toString()); + } + final node = await getCurrentNode(); + final host = Uri.parse(node.host).host; + await walletBase?.connectToNode( + node: Node(uri: "$host:${node.port}", type: WalletType.wownero)); + await walletBase?.startSync(); + await DB.instance + .put(boxName: walletId, key: "id", value: _walletId); + + // Set relevant indexes + await DB.instance + .put(boxName: walletId, key: "receivingIndex", value: 0); + await DB.instance + .put(boxName: walletId, key: "changeIndex", value: 0); + await DB.instance.put( + boxName: walletId, + key: 'blocked_tx_hashes', + value: ["0xdefault"], + ); // A list of transaction hashes to represent frozen utxos in wallet + // initialize address book entries + await DB.instance.put( + boxName: walletId, + key: 'addressBookEntries', + value: {}); + await DB.instance + .put(boxName: walletId, key: "isFavorite", value: false); + + // Generate and add addresses to relevant arrays + final initialReceivingAddress = await _generateAddressForChain(0, 0); + // final initialChangeAddress = await _generateAddressForChain(1, 0); + + await _addToAddressesArrayForChain(initialReceivingAddress, 0); + // await _addToAddressesArrayForChain(initialChangeAddress, 1); + + await DB.instance.put( + boxName: walletId, + key: 'receivingAddresses', + value: [initialReceivingAddress]); + await DB.instance + .put(boxName: walletId, key: "receivingIndex", value: 0); + + _currentReceivingAddress = Future(() => initialReceivingAddress); + + Logging.instance.log("_generateNewWalletFinished", level: LogLevel.Info); + } + + @override + // TODO: implement initializeWallet + Future initializeNew() async { + await _prefs.init(); + // TODO: ping actual wownero network + // try { + // final hasNetwork = await _electrumXClient.ping(); + // if (!hasNetwork) { + // return false; + // } + // } catch (e, s) { + // Logging.instance.log("Caught in initializeWallet(): $e\n$s"); + // return false; + // } + storage = const FlutterSecureStorage(); + walletService = + wownero.createWowneroWalletService(DB.instance.moneroWalletInfoBox); + prefs = await SharedPreferences.getInstance(); + keysStorage = KeyService(storage!); + + await _generateNewWallet(); + // var password; + // try { + // password = + // await keysStorage?.getWalletPassword(walletName: this._walletId); + // } catch (e, s) { + // Logging.instance.log("$e $s"); + // Logging.instance.log("Generating new ${coin.ticker} wallet."); + // // Triggers for new users automatically. Generates new wallet + // await _generateNewWallet(wallet); + // await wallet.put("id", this._walletId); + // return true; + // } + // walletBase = (await walletService?.openWallet(this._walletId, password)) + // as WowneroWalletBase; + // Logging.instance.log("Opening existing ${coin.ticker} wallet."); + // // Wallet already exists, triggers for a returning user + // final currentAddress = awaicurrentHeightt _getCurrentAddressForChain(0); + // this._currentReceivingAddress = Future(() => currentAddress); + // + // await walletBase?.connectToNode( + // node: Node( + // uri: "xmr-node.cakewallet.com:18081", type: WalletType.wownero)); + // walletBase?.startSync(); + + return true; + } + + @override + Future initializeExisting() async { + Logging.instance.log( + "Opening existing ${coin.prettyName} wallet $walletName...", + level: LogLevel.Info); + + if ((DB.instance.get(boxName: walletId, key: "id")) == null) { + debugPrint("Exception was thrown"); + throw Exception( + "Attempted to initialize an existing wallet using an unknown wallet ID!"); + } + + storage = const FlutterSecureStorage(); + walletService = + wownero.createWowneroWalletService(DB.instance.moneroWalletInfoBox); + prefs = await SharedPreferences.getInstance(); + keysStorage = KeyService(storage!); + + await _prefs.init(); + final data = + DB.instance.get(boxName: walletId, key: "latest_tx_model") + as TransactionData?; + if (data != null) { + _transactionData = Future(() => data); + } + + String? password; + try { + password = await keysStorage?.getWalletPassword(walletName: _walletId); + } catch (e, s) { + debugPrint("Exception was thrown $e $s"); + throw Exception("Password not found $e, $s"); + } + walletBase = (await walletService?.openWallet(_walletId, password!)) + as WowneroWalletBase; + debugPrint("walletBase $walletBase"); + Logging.instance.log( + "Opened existing ${coin.prettyName} wallet $walletName", + level: LogLevel.Info); + // Wallet already exists, triggers for a returning user + + String indexKey = "receivingIndex"; + final curIndex = + await DB.instance.get(boxName: walletId, key: indexKey) as int; + // Use new index to derive a new receiving address + final newReceivingAddress = await _generateAddressForChain(0, curIndex); + Logging.instance.log("xmr address in init existing: $newReceivingAddress", + level: LogLevel.Info); + _currentReceivingAddress = Future(() => newReceivingAddress); + } + + @override + Future get maxFee async { + var bal = await availableBalance; + var fee = walletBase!.calculateEstimatedFee( + wownero.getDefaultTransactionPriority(), bal.toBigInt().toInt()) ~/ + 10000; + + return fee; + } + + @override + // TODO: implement pendingBalance + Future get pendingBalance => throw UnimplementedError(); + + bool longMutex = false; + + // TODO: are these needed? + FlutterSecureStorage? storage; + WalletService? walletService; + SharedPreferences? prefs; + KeyService? keysStorage; + WowneroWalletBase? walletBase; + WalletCreationService? _walletCreationService; + + String toStringForinfo(WalletInfo info) { + return "id: ${info.id} name: ${info.name} type: ${info.type} recovery: ${info.isRecovery}" + " restoreheight: ${info.restoreHeight} timestamp: ${info.timestamp} dirPath: ${info.dirPath} " + "path: ${info.path} address: ${info.address} addresses: ${info.addresses}"; + } + + Future pathForWalletDir({ + required String name, + required WalletType type, + }) async { + Directory root = (await getApplicationDocumentsDirectory()); + if (Platform.isIOS) { + root = (await getLibraryDirectory()); + } + final prefix = walletTypeToString(type).toLowerCase(); + final walletsDir = Directory('${root.path}/wallets'); + final walletDire = Directory('${walletsDir.path}/$prefix/$name'); + + if (!walletDire.existsSync()) { + walletDire.createSync(recursive: true); + } + + return walletDire.path; + } + + Future pathForWallet({ + required String name, + required WalletType type, + }) async => + await pathForWalletDir(name: name, type: type) + .then((path) => '$path/$name'); + + // TODO: take in a dynamic height + @override + Future recoverFromMnemonic({ + required String mnemonic, + required int maxUnusedAddressGap, + required int maxNumberOfIndexesToCheck, + required int height, + }) async { + await _prefs.init(); + longMutex = true; + final start = DateTime.now(); + try { + // Logging.instance.log("IS_INTEGRATION_TEST: $integrationTestFlag"); + // if (!integrationTestFlag) { + // final features = await electrumXClient.getServerFeatures(); + // Logging.instance.log("features: $features"); + // 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 + if ((await _secureStore.read(key: '${_walletId}_mnemonic')) != null) { + longMutex = false; + throw Exception("Attempted to overwrite mnemonic on restore!"); + } + await _secureStore.write( + key: '${_walletId}_mnemonic', value: mnemonic.trim()); + + height = getSeedHeightSync(mnemonic.trim()); + + await DB.instance + .put(boxName: walletId, key: "restoreHeight", value: height); + + storage = const FlutterSecureStorage(); + walletService = + wownero.createWowneroWalletService(DB.instance.moneroWalletInfoBox); + prefs = await SharedPreferences.getInstance(); + keysStorage = KeyService(storage!); + WalletInfo walletInfo; + WalletCredentials credentials; + String name = _walletId; + final dirPath = + await pathForWalletDir(name: name, type: WalletType.wownero); + final path = await pathForWallet(name: name, type: WalletType.wownero); + credentials = wownero.createWowneroRestoreWalletFromSeedCredentials( + name: name, + height: height, + mnemonic: mnemonic.trim(), + ); + try { + walletInfo = WalletInfo.external( + id: WalletBase.idFor(name, WalletType.wownero), + name: name, + type: WalletType.wownero, + isRecovery: false, + restoreHeight: credentials.height ?? 0, + date: DateTime.now(), + path: path, + dirPath: dirPath, + // TODO: find out what to put for address + address: ''); + credentials.walletInfo = walletInfo; + + _walletCreationService = WalletCreationService( + secureStorage: storage, + sharedPreferences: prefs, + walletService: walletService, + keyService: keysStorage, + ); + _walletCreationService!.changeWalletType(); + // To restore from a seed + final wallet = + await _walletCreationService!.restoreFromSeed(credentials); + walletInfo.address = wallet.walletAddresses.address; + await DB.instance + .add(boxName: WalletInfo.boxName, value: walletInfo); + walletBase?.close(); + walletBase = wallet as WowneroWalletBase; + await DB.instance.put( + boxName: walletId, + key: 'receivingAddresses', + value: [walletInfo.address!]); + await DB.instance + .put(boxName: walletId, key: "receivingIndex", value: 0); + await DB.instance + .put(boxName: walletId, key: "id", value: _walletId); + await DB.instance + .put(boxName: walletId, key: "changeIndex", value: 0); + await DB.instance.put( + boxName: walletId, + key: 'blocked_tx_hashes', + value: ["0xdefault"], + ); // A list of transaction hashes to represent frozen utxos in wallet + // initialize address book entries + await DB.instance.put( + boxName: walletId, + key: 'addressBookEntries', + value: {}); + await DB.instance + .put(boxName: walletId, key: "isFavorite", value: false); + } catch (e, s) { + debugPrint(e.toString()); + debugPrint(s.toString()); + } + final node = await getCurrentNode(); + final host = Uri.parse(node.host).host; + await walletBase?.connectToNode( + node: Node(uri: "$host:${node.port}", type: WalletType.wownero)); + await walletBase?.rescan(height: credentials.height); + } 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); + } + + @override + Future send({ + required String toAddress, + required int amount, + Map args = const {}, + }) async { + try { + final txData = await prepareSend( + address: toAddress, satoshiAmount: amount, args: args); + final txHash = await confirmSend(txData: txData); + return txHash; + } catch (e, s) { + Logging.instance + .log("Exception rethrown from send(): $e\n$s", level: LogLevel.Error); + rethrow; + } + } + + @override + Future testNetworkConnection() async { + return await walletBase?.isConnected() ?? 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(); + _isConnected = hasNetwork; + if (_isConnected != hasNetwork) { + NodeConnectionStatus status = hasNetwork + ? NodeConnectionStatus.connected + : NodeConnectionStatus.disconnected; + GlobalEventBus.instance + .fire(NodeConnectionStatusChangedEvent(status, walletId, coin)); + } + } + + void stopNetworkAlivePinging() { + _networkAliveTimer?.cancel(); + _networkAliveTimer = null; + } + + bool _isConnected = false; + + @override + bool get isConnected => _isConnected; + + @override + Future get totalBalance async { + var transactions = walletBase?.transactionHistory!.transactions; + int transactionBalance = 0; + for (var tx in transactions!.entries) { + if (tx.value.direction == TransactionDirection.incoming) { + transactionBalance += tx.value.amount!; + } else { + transactionBalance += -tx.value.amount! - tx.value.fee!; + } + } + + // TODO: grab total balance + var bal = 0; + for (var element in walletBase!.balance!.entries) { + bal = bal + element.value.fullBalance; + } + debugPrint("balances: $transactionBalance $bal"); + if (isActive) { + String am = wowneroAmountToString(amount: bal); + + return Decimal.parse(am); + } else { + String am = wowneroAmountToString(amount: transactionBalance); + + return Decimal.parse(am); + } + } + + @override + // TODO: implement onIsActiveWalletChanged + void Function(bool)? get onIsActiveWalletChanged => (isActive) async { + await walletBase?.save(); + walletBase?.close(); + wowneroAutosaveTimer?.cancel(); + wowneroAutosaveTimer = null; + timer?.cancel(); + timer = null; + await stopSyncPercentTimer(); + if (isActive) { + String? password; + try { + password = + await keysStorage?.getWalletPassword(walletName: _walletId); + } catch (e, s) { + debugPrint("Exception was thrown $e $s"); + throw Exception("Password not found $e, $s"); + } + walletBase = (await walletService?.openWallet(_walletId, password!)) + as WowneroWalletBase?; + if (!(await walletBase!.isConnected())) { + final node = await getCurrentNode(); + final host = Uri.parse(node.host).host; + await walletBase?.connectToNode( + node: + Node(uri: "$host:${node.port}", type: WalletType.wownero)); + await walletBase?.startSync(); + } + await refresh(); + } + this.isActive = isActive; + }; + + bool isActive = false; + + @override + Future get transactionData => + _transactionData ??= _fetchTransactionData(); + Future? _transactionData; + + Future _fetchTransactionData() async { + final transactions = walletBase?.transactionHistory!.transactions; + + final cachedTransactions = + DB.instance.get(boxName: walletId, key: 'latest_tx_model') + as TransactionData?; + int latestTxnBlockHeight = + DB.instance.get(boxName: walletId, key: "storedTxnDataHeight") + as int? ?? + 0; + + final txidsList = DB.instance + .get(boxName: walletId, key: "cachedTxids") as List? ?? + []; + + final Set cachedTxids = Set.from(txidsList); + + // TODO: filter to skip cached + confirmed txn processing in next step + // final unconfirmedCachedTransactions = + // cachedTransactions?.getAllTransactions() ?? {}; + // unconfirmedCachedTransactions + // .removeWhere((key, value) => value.confirmedStatus); + // + // if (cachedTransactions != null) { + // for (final tx in allTxHashes.toList(growable: false)) { + // final txHeight = tx["height"] as int; + // if (txHeight > 0 && + // txHeight < latestTxnBlockHeight - MINIMUM_CONFIRMATIONS) { + // if (unconfirmedCachedTransactions[tx["tx_hash"] as String] == null) { + // allTxHashes.remove(tx); + // } + // } + // } + // } + + // sort thing stuff + // change to get Wownero price + final priceData = + await _priceAPI.getPricesAnd24hChange(baseCurrency: _prefs.currency); + Decimal currentPrice = priceData[coin]?.item1 ?? Decimal.zero; + final List> midSortedArray = []; + + if (transactions != null) { + for (var tx in transactions.entries) { + cachedTxids.add(tx.value.id); + Logging.instance.log( + "${tx.value.accountIndex} ${tx.value.addressIndex} ${tx.value.amount} ${tx.value.date} " + "${tx.value.direction} ${tx.value.fee} ${tx.value.height} ${tx.value.id} ${tx.value.isPending} ${tx.value.key} " + "${tx.value.recipientAddress}, ${tx.value.additionalInfo} con:${tx.value.confirmations}" + " ${tx.value.keyIndex}", + level: LogLevel.Info); + String am = wowneroAmountToString(amount: tx.value.amount!); + final worthNow = (currentPrice * Decimal.parse(am)).toStringAsFixed(2); + Map midSortedTx = {}; + // // create final tx map + midSortedTx["txid"] = tx.value.id; + midSortedTx["confirmed_status"] = !tx.value.isPending && + tx.value.confirmations != null && + tx.value.confirmations! >= MINIMUM_CONFIRMATIONS; + midSortedTx["confirmations"] = tx.value.confirmations ?? 0; + midSortedTx["timestamp"] = + (tx.value.date.millisecondsSinceEpoch ~/ 1000); + midSortedTx["txType"] = + tx.value.direction == TransactionDirection.incoming + ? "Received" + : "Sent"; + midSortedTx["amount"] = tx.value.amount; + midSortedTx["worthNow"] = worthNow; + midSortedTx["worthAtBlockTimestamp"] = worthNow; + midSortedTx["fees"] = tx.value.fee; + // TODO: shouldn't wownero have an address I can grab + if (tx.value.direction == TransactionDirection.incoming) { + final addressInfo = tx.value.additionalInfo; + + midSortedTx["address"] = walletBase?.getTransactionAddress( + addressInfo!['accountIndex'] as int, + addressInfo['addressIndex'] as int, + ); + } else { + midSortedTx["address"] = ""; + } + + final int txHeight = tx.value.height ?? 0; + midSortedTx["height"] = txHeight; + if (txHeight >= latestTxnBlockHeight) { + latestTxnBlockHeight = txHeight; + } + + midSortedTx["aliens"] = []; + midSortedTx["inputSize"] = 0; + midSortedTx["outputSize"] = 0; + midSortedTx["inputs"] = []; + midSortedTx["outputs"] = []; + midSortedArray.add(midSortedTx); + } + } + + // sort by date ---- + midSortedArray + .sort((a, b) => (b["timestamp"] as int) - (a["timestamp"] as int)); + Logging.instance.log(midSortedArray, level: LogLevel.Info); + + // buildDateTimeChunks + final Map result = {"dateTimeChunks": []}; + final dateArray = []; + + for (int i = 0; i < midSortedArray.length; i++) { + final txObject = midSortedArray[i]; + final date = extractDateFromTimestamp(txObject["timestamp"] as int); + final txTimeArray = [txObject["timestamp"], date]; + + if (dateArray.contains(txTimeArray[1])) { + result["dateTimeChunks"].forEach((dynamic chunk) { + if (extractDateFromTimestamp(chunk["timestamp"] as int) == + txTimeArray[1]) { + if (chunk["transactions"] == null) { + chunk["transactions"] = >[]; + } + chunk["transactions"].add(txObject); + } + }); + } else { + dateArray.add(txTimeArray[1]); + final chunk = { + "timestamp": txTimeArray[0], + "transactions": [txObject], + }; + result["dateTimeChunks"].add(chunk); + } + } + + final transactionsMap = cachedTransactions?.getAllTransactions() ?? {}; + transactionsMap + .addAll(TransactionData.fromJson(result).getAllTransactions()); + + final txModel = TransactionData.fromMap(transactionsMap); + + await DB.instance.put( + boxName: walletId, + key: 'storedTxnDataHeight', + value: latestTxnBlockHeight); + await DB.instance.put( + boxName: walletId, key: 'latest_tx_model', value: txModel); + await DB.instance.put( + boxName: walletId, + key: 'cachedTxids', + value: cachedTxids.toList(growable: false)); + + return txModel; + } + + @override + // TODO: implement unspentOutputs + Future> get unspentOutputs => throw UnimplementedError(); + + @override + // TODO: implement validateAddress + bool validateAddress(String address) { + bool valid = RegExp("[a-zA-Z0-9]{95}").hasMatch(address) || + RegExp("[a-zA-Z0-9]{106}").hasMatch(address); + return valid; + } + + @override + String get walletId => _walletId; + late String _walletId; + + @override + String get walletName => _walletName; + late String _walletName; + + // setter for updating on rename + @override + set walletName(String newName) => _walletName = newName; + + @override + set isFavorite(bool markFavorite) { + DB.instance.put( + boxName: walletId, key: "isFavorite", value: markFavorite); + } + + @override + bool get isFavorite { + try { + return DB.instance.get(boxName: walletId, key: "isFavorite") + as bool; + } catch (e, s) { + Logging.instance + .log("isFavorite fetch failed: $e\n$s", level: LogLevel.Error); + rethrow; + } + } + + @override + // TODO: implement availableBalance + Future get availableBalance async { + var bal = 0; + for (var element in walletBase!.balance!.entries) { + bal = bal + element.value.unlockedBalance; + } + String am = wowneroAmountToString(amount: bal); + + return Decimal.parse(am); + } + + @override + Coin get coin => _coin; + + @override + Future confirmSend({required Map txData}) async { + try { + Logging.instance.log("confirmSend txData: $txData", level: LogLevel.Info); + final pendingWowneroTransaction = + txData['pendingWowneroTransaction'] as PendingWowneroTransaction; + try { + await pendingWowneroTransaction.commit(); + Logging.instance.log( + "transaction ${pendingWowneroTransaction.id} has been sent", + level: LogLevel.Info); + return pendingWowneroTransaction.id; + } catch (e, s) { + Logging.instance.log("$walletName wownero confirmSend: $e\n$s", + level: LogLevel.Error); + rethrow; + } + } catch (e, s) { + Logging.instance.log("Exception rethrown from confirmSend(): $e\n$s", + level: LogLevel.Info); + rethrow; + } + } + + // TODO: fix the double free memory crash error. + @override + Future> prepareSend( + {required String address, + required int satoshiAmount, + Map? args}) async { + int amount = satoshiAmount; + String toAddress = address; + try { + final feeRate = args?["feeRate"]; + if (feeRate is FeeRateType) { + MoneroTransactionPriority feePriority = MoneroTransactionPriority.slow; + switch (feeRate) { + case FeeRateType.fast: + feePriority = MoneroTransactionPriority.fastest; + break; + case FeeRateType.average: + feePriority = MoneroTransactionPriority.medium; + break; + case FeeRateType.slow: + feePriority = MoneroTransactionPriority.slow; + break; + } + + Future? awaitPendingTransaction; + try { + Logging.instance + .log("$toAddress $amount $args", level: LogLevel.Info); + String amountToSend = wowneroAmountToString(amount: amount * 1000); + Logging.instance.log("$amount $amountToSend", level: LogLevel.Info); + + wownero_output.Output output = wownero_output.Output(walletBase!); + output.address = toAddress; + output.setCryptoAmount(amountToSend); + + List outputs = [output]; + Object tmp = wownero.createWowneroTransactionCreationCredentials( + outputs: outputs, priority: feePriority); + + awaitPendingTransaction = walletBase!.createTransaction(tmp); + } catch (e, s) { + Logging.instance.log("Exception rethrown from prepareSend(): $e\n$s", + level: LogLevel.Warning); + } + + PendingWowneroTransaction pendingWowneroTransaction = + await (awaitPendingTransaction!) as PendingWowneroTransaction; + int realfee = (Decimal.parse(pendingWowneroTransaction.feeFormatted) * + 100000000.toDecimal()) + .toBigInt() + .toInt(); + debugPrint("fee? $realfee"); + Map txData = { + "pendingWowneroTransaction": pendingWowneroTransaction, + "fee": realfee, + "addresss": toAddress, + "recipientAmt": satoshiAmount, + }; + + Logging.instance.log("prepare send: $txData", level: LogLevel.Info); + return txData; + } else { + throw ArgumentError("Invalid fee rate argument provided!"); + } + } catch (e, s) { + Logging.instance.log("Exception rethrown from prepare send(): $e\n$s", + level: LogLevel.Info); + + if (e.toString().contains("Incorrect unlocked balance")) { + throw Exception("Insufficient balance!"); + } else if (e is CreationTransactionException) { + throw Exception("Insufficient funds to pay for transaction fee!"); + } else { + throw Exception("Transaction failed with error code $e"); + } + } + } + + @override + Future estimateFeeFor(int satoshiAmount, int feeRate) async { + MoneroTransactionPriority? priority; + switch (feeRate) { + case 1: + priority = MoneroTransactionPriority.regular; + break; + case 2: + priority = MoneroTransactionPriority.medium; + break; + case 3: + priority = MoneroTransactionPriority.fast; + break; + case 4: + priority = MoneroTransactionPriority.fastest; + break; + case 0: + default: + priority = MoneroTransactionPriority.slow; + break; + } + final fee = + (walletBase?.calculateEstimatedFee(priority, satoshiAmount) ?? 0) ~/ + 10000; + return fee; + } + + @override + Future generateNewAddress() async { + try { + const String indexKey = "receivingIndex"; + // First increment the receiving index + await _incrementAddressIndexForChain(0); + final newReceivingIndex = + DB.instance.get(boxName: walletId, key: indexKey) as int; + + // Use new index to derive a new receiving address + final newReceivingAddress = + await _generateAddressForChain(0, newReceivingIndex); + + // Add that new receiving address to the array of receiving addresses + await _addToAddressesArrayForChain(newReceivingAddress, 0); + + // Set the new receiving address that the service + + _currentReceivingAddress = Future(() => newReceivingAddress); + + return true; + } catch (e, s) { + Logging.instance.log( + "Exception rethrown from generateNewAddress(): $e\n$s", + level: LogLevel.Error); + return false; + } + } +} diff --git a/lib/services/price.dart b/lib/services/price.dart index c79be324f..d157580ad 100644 --- a/lib/services/price.dart +++ b/lib/services/price.dart @@ -79,7 +79,7 @@ class PriceAPI { Map> result = {}; try { final uri = Uri.parse( - "https://api.coingecko.com/api/v3/coins/markets?vs_currency=${baseCurrency.toLowerCase()}&ids=monero,bitcoin,epic-cash,zcoin,dogecoin,bitcoin-cash,namecoin&order=market_cap_desc&per_page=10&page=1&sparkline=false"); + "https://api.coingecko.com/api/v3/coins/markets?vs_currency=${baseCurrency.toLowerCase()}&ids=monero,bitcoin,epic-cash,zcoin,dogecoin,bitcoin-cash,namecoin,wownero&order=market_cap_desc&per_page=10&page=1&sparkline=false"); // final uri = Uri.parse( // "https://api.coingecko.com/api/v3/coins/markets?vs_currency=${baseCurrency.toLowerCase()}&ids=monero%2Cbitcoin%2Cepic-cash%2Czcoin%2Cdogecoin&order=market_cap_desc&per_page=10&page=1&sparkline=false"); diff --git a/lib/services/wallets.dart b/lib/services/wallets.dart index 034db9308..db3011d4e 100644 --- a/lib/services/wallets.dart +++ b/lib/services/wallets.dart @@ -223,7 +223,7 @@ class Wallets extends ChangeNotifier { final shouldSetAutoSync = shouldAutoSyncAll || walletIdsToEnableAutoSync.contains(manager.walletId); - if (manager.coin == Coin.monero) { + if (manager.coin == Coin.monero || manager.coin == Coin.wownero) { walletsToInitLinearly.add(Tuple2(manager, shouldSetAutoSync)); } else { walletInitFutures.add(manager.initializeExisting().then((value) { @@ -312,7 +312,7 @@ class Wallets extends ChangeNotifier { final shouldSetAutoSync = shouldAutoSyncAll || walletIdsToEnableAutoSync.contains(manager.walletId); - if (manager.coin == Coin.monero) { + if (manager.coin == Coin.monero || manager.coin == Coin.wownero) { walletsToInitLinearly.add(Tuple2(manager, shouldSetAutoSync)); } else { walletInitFutures.add(manager.initializeExisting().then((value) { diff --git a/lib/services/wallets_service.dart b/lib/services/wallets_service.dart index 0700950a4..b30f9e9e5 100644 --- a/lib/services/wallets_service.dart +++ b/lib/services/wallets_service.dart @@ -2,6 +2,7 @@ import 'dart:convert'; import 'package:flutter/material.dart'; import 'package:flutter_libmonero/monero/monero.dart'; +import 'package:flutter_libmonero/wownero/wownero.dart'; import 'package:flutter_secure_storage/flutter_secure_storage.dart'; import 'package:stackwallet/hive/db.dart'; import 'package:stackwallet/services/coins/epiccash/epiccash_wallet.dart'; @@ -367,8 +368,13 @@ class WalletsService extends ChangeNotifier { await DB.instance.delete( boxName: DB.boxNameAllWalletsData, key: "${walletId}_mnemonicHasBeenVerified"); - - if (coinFromPrettyName(shell['coin'] as String) == Coin.monero) { + if (coinFromPrettyName(shell['coin'] as String) == Coin.wownero) { + final wowService = + wownero.createWowneroWalletService(DB.instance.moneroWalletInfoBox); + await wowService.remove(walletId); + Logging.instance + .log("monero wallet: $walletId deleted", level: LogLevel.Info); + } else if (coinFromPrettyName(shell['coin'] as String) == Coin.monero) { final xmrService = monero.createMoneroWalletService(DB.instance.moneroWalletInfoBox); await xmrService.remove(walletId); diff --git a/lib/utilities/address_utils.dart b/lib/utilities/address_utils.dart index 5dd805941..d73c5d3ce 100644 --- a/lib/utilities/address_utils.dart +++ b/lib/utilities/address_utils.dart @@ -53,6 +53,9 @@ class AddressUtils { case Coin.monero: return RegExp("[a-zA-Z0-9]{95}").hasMatch(address) || RegExp("[a-zA-Z0-9]{106}").hasMatch(address); + case Coin.wownero: + return RegExp("[a-zA-Z0-9]{95}").hasMatch(address) || + RegExp("[a-zA-Z0-9]{106}").hasMatch(address); case Coin.namecoin: return Address.validateAddress(address, namecoin, namecoin.bech32!); case Coin.bitcoinTestNet: diff --git a/lib/utilities/assets.dart b/lib/utilities/assets.dart index 002bf452c..94558e623 100644 --- a/lib/utilities/assets.dart +++ b/lib/utilities/assets.dart @@ -128,6 +128,7 @@ class _SVG { String get epicCash => "assets/svg/coin_icons/EpicCash.svg"; String get firo => "assets/svg/coin_icons/Firo.svg"; String get monero => "assets/svg/coin_icons/Monero.svg"; + String get wownero => "assets/svg/coin_icons/Wownero.svg"; String get namecoin => "assets/svg/coin_icons/Namecoin.svg"; String get chevronRight => "assets/svg/chevron-right.svg"; @@ -156,6 +157,8 @@ class _SVG { return firo; case Coin.monero: return monero; + case Coin.wownero: + return wownero; case Coin.namecoin: return namecoin; case Coin.bitcoinTestNet: @@ -177,6 +180,7 @@ class _PNG { String get splash => "assets/images/splash.png"; String get monero => "assets/images/monero.png"; + String get wownero => "assets/images/wownero.png"; String get firo => "assets/images/firo.png"; String get dogecoin => "assets/images/doge.png"; String get bitcoin => "assets/images/bitcoin.png"; @@ -203,6 +207,8 @@ class _PNG { return firo; case Coin.monero: return monero; + case Coin.wownero: + return wownero; case Coin.namecoin: return namecoin; } diff --git a/lib/utilities/block_explorers.dart b/lib/utilities/block_explorers.dart index bf619d88f..f89f270be 100644 --- a/lib/utilities/block_explorers.dart +++ b/lib/utilities/block_explorers.dart @@ -18,6 +18,8 @@ Uri getBlockExplorerTransactionUrlFor({ throw UnimplementedError("missing block explorer for epic cash"); case Coin.monero: return Uri.parse("https://xmrchain.net/tx/$txid"); + case Coin.wownero: + return Uri.parse("https://explore.wownero.com/search?value=$txid"); case Coin.firo: return Uri.parse("https://explorer.firo.org/tx/$txid"); case Coin.firoTestNet: diff --git a/lib/utilities/constants.dart b/lib/utilities/constants.dart index e561a1f95..f5139477c 100644 --- a/lib/utilities/constants.dart +++ b/lib/utilities/constants.dart @@ -18,6 +18,7 @@ abstract class Constants { //TODO: correct for monero? static const int satsPerCoinMonero = 1000000000000; + static const int satsPerCoinWownero = 100000000000; static const int satsPerCoin = 100000000; static const int decimalPlaces = 8; @@ -54,6 +55,9 @@ abstract class Constants { case Coin.monero: values.addAll([25]); break; + case Coin.wownero: + values.addAll([14]); + break; } return values; } @@ -83,6 +87,9 @@ abstract class Constants { case Coin.monero: return 120; + case Coin.wownero: + return 120; + case Coin.namecoin: return 600; } diff --git a/lib/utilities/default_nodes.dart b/lib/utilities/default_nodes.dart index fe40f8094..2ffd6f307 100644 --- a/lib/utilities/default_nodes.dart +++ b/lib/utilities/default_nodes.dart @@ -15,6 +15,7 @@ abstract class DefaultNodes { epicCash, bitcoincash, namecoin, + wownero, bitcoinTestnet, bitcoincashTestnet, dogecoinTestnet, @@ -83,6 +84,20 @@ abstract class DefaultNodes { isDown: false, ); + // TODO: eventually enable ssl and set scheme to https + // currently get certificate failure + static NodeModel get wownero => NodeModel( + host: "http://eu-west-2.wow.xmr.pm", + port: 34568, + name: defaultName, + id: _nodeId(Coin.wownero), + useSSL: false, + enabled: true, + coinName: Coin.wownero.name, + isFailover: true, + isDown: false, + ); + static NodeModel get epicCash => NodeModel( host: "http://epiccash.stackwallet.com", port: 3413, @@ -175,6 +190,9 @@ abstract class DefaultNodes { case Coin.monero: return monero; + case Coin.wownero: + return wownero; + case Coin.namecoin: return namecoin; diff --git a/lib/utilities/enums/coin_enum.dart b/lib/utilities/enums/coin_enum.dart index e0bc52433..78a04e220 100644 --- a/lib/utilities/enums/coin_enum.dart +++ b/lib/utilities/enums/coin_enum.dart @@ -9,6 +9,7 @@ import 'package:stackwallet/services/coins/firo/firo_wallet.dart' as firo; import 'package:stackwallet/services/coins/monero/monero_wallet.dart' as xmr; import 'package:stackwallet/services/coins/namecoin/namecoin_wallet.dart' as nmc; +import 'package:stackwallet/services/coins/wownero/wownero_wallet.dart' as wow; enum Coin { bitcoin, @@ -17,6 +18,7 @@ enum Coin { epicCash, firo, monero, + wownero, namecoin, /// @@ -47,6 +49,8 @@ extension CoinExt on Coin { return "Firo"; case Coin.monero: return "Monero"; + case Coin.wownero: + return "Wownero"; case Coin.namecoin: return "Namecoin"; case Coin.bitcoinTestNet: @@ -74,6 +78,8 @@ extension CoinExt on Coin { return "FIRO"; case Coin.monero: return "XMR"; + case Coin.wownero: + return "WOW"; case Coin.namecoin: return "NMC"; case Coin.bitcoinTestNet: @@ -102,6 +108,8 @@ extension CoinExt on Coin { return "firo"; case Coin.monero: return "monero"; + case Coin.wownero: + return "wownero"; case Coin.namecoin: return "namecoin"; case Coin.bitcoinTestNet: @@ -130,6 +138,7 @@ extension CoinExt on Coin { case Coin.epicCash: case Coin.monero: + case Coin.wownero: return false; } } @@ -157,6 +166,10 @@ extension CoinExt on Coin { case Coin.monero: return xmr.MINIMUM_CONFIRMATIONS; + + case Coin.wownero: + return wow.MINIMUM_CONFIRMATIONS; + case Coin.namecoin: return nmc.MINIMUM_CONFIRMATIONS; } @@ -204,6 +217,10 @@ Coin coinFromPrettyName(String name) { case "tDogecoin": case "dogecoinTestNet": return Coin.dogecoinTestNet; + case "Wownero": + case "tWownero": + case "wownero": + return Coin.wownero; default: throw ArgumentError.value( name, "name", "No Coin enum value with that prettyName"); @@ -234,6 +251,8 @@ Coin coinFromTickerCaseInsensitive(String ticker) { return Coin.firoTestNet; case "tdoge": return Coin.dogecoinTestNet; + case "wow": + return Coin.wownero; default: throw ArgumentError.value( ticker, "name", "No Coin enum value with that ticker"); diff --git a/lib/utilities/theme/color_theme.dart b/lib/utilities/theme/color_theme.dart index 9ab9aa293..361d922dc 100644 --- a/lib/utilities/theme/color_theme.dart +++ b/lib/utilities/theme/color_theme.dart @@ -209,8 +209,8 @@ class CoinThemeColor { return monero; case Coin.namecoin: return namecoin; - // case Coin.wownero: - // return wownero; + case Coin.wownero: + return wownero; } } } diff --git a/lib/utilities/theme/stack_colors.dart b/lib/utilities/theme/stack_colors.dart index 852bc7d6b..c6ee28892 100644 --- a/lib/utilities/theme/stack_colors.dart +++ b/lib/utilities/theme/stack_colors.dart @@ -1422,8 +1422,8 @@ class StackColors extends ThemeExtension { return _coin.monero; case Coin.namecoin: return _coin.namecoin; - // case Coin.wownero: - // return wownero; + case Coin.wownero: + return _coin.wownero; } } diff --git a/lib/widgets/node_options_sheet.dart b/lib/widgets/node_options_sheet.dart index 02570b76c..58a5bf30a 100644 --- a/lib/widgets/node_options_sheet.dart +++ b/lib/widgets/node_options_sheet.dart @@ -85,6 +85,7 @@ class NodeOptionsSheet extends ConsumerWidget { break; case Coin.monero: + case Coin.wownero: try { final uri = Uri.parse(node.host); if (uri.scheme.startsWith("http")) { diff --git a/lib/widgets/transaction_card.dart b/lib/widgets/transaction_card.dart index 4a8da2357..b2feaca7a 100644 --- a/lib/widgets/transaction_card.dart +++ b/lib/widgets/transaction_card.dart @@ -172,7 +172,9 @@ class _TransactionCardState extends ConsumerState { builder: (_) { final amount = coin == Coin.monero ? (_transaction.amount ~/ 10000) - : _transaction.amount; + : coin == Coin.wownero + ? (_transaction.amount ~/ 1000) + : _transaction.amount; return Text( "${Format.satoshiAmountToPrettyString(amount, locale)} ${coin.ticker}", style: @@ -212,6 +214,8 @@ class _TransactionCardState extends ConsumerState { int value = _transaction.amount; if (coin == Coin.monero) { value = (value ~/ 10000); + } else if (coin == Coin.wownero) { + value = (value ~/ 1000); } return Text( diff --git a/pubspec.lock b/pubspec.lock index 71145aef4..2cc67f611 100644 --- a/pubspec.lock +++ b/pubspec.lock @@ -317,6 +317,13 @@ packages: relative: true source: path version: "0.0.1" + cw_wownero: + dependency: "direct main" + description: + path: "crypto_plugins/flutter_libmonero/cw_wownero" + relative: true + source: path + version: "0.0.1" dart_numerics: dependency: "direct main" description: diff --git a/pubspec.yaml b/pubspec.yaml index f87c6587d..dbe44c852 100644 --- a/pubspec.yaml +++ b/pubspec.yaml @@ -11,7 +11,7 @@ description: Stack Wallet # In iOS, build-name is used as CFBundleShortVersionString while build-number used as CFBundleVersion. # Read more about iOS versioning at # https://developer.apple.com/library/archive/documentation/General/Reference/InfoPlistKeyReference/Articles/CoreFoundationKeys.html -version: 1.4.50+65 +version: 1.4.52+68 environment: sdk: ">=2.17.0 <3.0.0" @@ -31,6 +31,9 @@ dependencies: cw_monero: path: ./crypto_plugins/flutter_libmonero/cw_monero + cw_wownero: + path: ./crypto_plugins/flutter_libmonero/cw_wownero + cw_core: path: ./crypto_plugins/flutter_libmonero/cw_core @@ -189,6 +192,7 @@ flutter: - assets/svg/clipboard.svg - assets/images/stack.png - assets/images/monero.png + - assets/images/wownero.png - assets/images/firo.png - assets/images/doge.png - assets/images/bitcoin.png @@ -292,6 +296,7 @@ flutter: - assets/svg/coin_icons/EpicCash.svg - assets/svg/coin_icons/Firo.svg - assets/svg/coin_icons/Monero.svg + - assets/svg/coin_icons/Wownero.svg - assets/svg/coin_icons/Namecoin.svg # lottie animations - assets/lottie/test.json diff --git a/test/models/transactions_model_test.dart b/test/models/transactions_model_test.dart index 31df03a5e..1e19f769a 100644 --- a/test/models/transactions_model_test.dart +++ b/test/models/transactions_model_test.dart @@ -1,7 +1,152 @@ import 'package:flutter_test/flutter_test.dart'; import 'package:stackwallet/models/models.dart'; +import '../services/coins/firo/sample_data/transaction_data_samples.dart'; + void main() { + group("TransactionData", () { + test("TransactionData from Json", () { + final txChunk = TransactionChunk.fromJson({ + "timestamp": 993260735, + "transactions": [ + { + "txid": "txid", + "confirmed_status": true, + "timestamp": 1876352482, + "txType": "txType", + "amount": 10, + "worthNow": "1", + "worthAtBlockTimestamp": "1", + "fees": 1, + "inputSize": 1, + "outputSize": 1, + "inputs": [], + "outputs": [], + "address": "address", + "height": 1, + "confirmations": 1, + "aliens": [], + "subType": "mint", + "isCancelled": false, + "slateId": "slateId", + "otherData": "otherData", + } + ] + }); + final txdata = + TransactionData.fromJson({"dateTimeChunks": [], "txChunks": []}); + txdata.findTransaction("txid"); + txdata.getAllTransactions(); + }); + }); + + group("Timestamp", () { + test("Timestamp is now", () { + final date = extractDateFromTimestamp(0); + }); + + test("Timestamp is null", () { + final date = extractDateFromTimestamp(null); + }); + + test("Timestamp is a random date", () { + final date = extractDateFromTimestamp(1876352482); + }); + }); + + group("Transaction", () { + test("Transaction from Json", () { + final tx = Transaction.fromJson({ + "txid": "txid", + "confirmed_status": true, + "timestamp": 1876352482, + "txType": "txType", + "amount": 10, + "worthNow": "1", + "worthAtBlockTimestamp": "1", + "fees": 1, + "inputSize": 1, + "outputSize": 1, + "inputs": [], + "outputs": [], + "address": "address", + "height": 1, + "confirmations": 1, + "aliens": [], + "subType": "mint", + "isCancelled": false, + "slateId": "slateId", + "otherData": "otherData", + }); + }); + + test("Transaction from Lelantus Json", () { + final tx = Transaction.fromLelantusJson({ + "txid": "txid", + "confirmed_status": true, + "timestamp": 1876352482, + "txType": "txType", + "amount": 10, + "worthNow": "1", + "worthAtBlockTimestamp": "1", + "fees": 1, + "inputSize": 1, + "outputSize": 1, + "inputs": [], + "outputs": [], + "address": "address", + "height": 1, + "confirmations": 1, + "aliens": [], + "subType": "mint", + "isCancelled": false, + "slateId": "slateId", + "otherData": "otherData", + }); + }); + + test("TransactionChunk", () { + final transactionchunk = TransactionChunk.fromJson({ + "timestamp": 45920, + "transactions": [], + }); + expect( + transactionchunk.toString(), "timestamp: 45920 transactions: [\n]"); + }); + + test("TransactionChunk with a transaction", () { + final txChunk = TransactionChunk.fromJson({ + "timestamp": 993260735, + "transactions": [ + { + "txid": "txid", + "confirmed_status": true, + "timestamp": 1876352482, + "txType": "txType", + "amount": 10, + "worthNow": "1", + "worthAtBlockTimestamp": "1", + "fees": 1, + "inputSize": 1, + "outputSize": 1, + "inputs": [], + "outputs": [], + "address": "address", + "height": 1, + "confirmations": 1, + "aliens": [], + "subType": "mint", + "isCancelled": false, + "slateId": "slateId", + "otherData": "otherData", + } + ] + }); + expect(txChunk.toString(), + "timestamp: 993260735 transactions: [\n {txid: txid, type: txType, subType: mint, value: 10, fee: 1, height: 1, confirm: true, confirmations: 1, address: address, timestamp: 1876352482, worthNow: 1, inputs: [], slateid: slateId } \n]"); + }); + }); + group("Transaction isMinting", () { test("Transaction isMinting unconfirmed mint", () { final tx = Transaction( @@ -94,4 +239,57 @@ void main() { expect(tx1 == tx2, false); expect(tx2.toString(), tx1.toString()); }); + + group("Input", () { + test("Input.toString", () { + final input = Input( + txid: "txid", + vout: 1, + prevout: null, + scriptsig: "scriptsig", + scriptsigAsm: "scriptsigAsm", + witness: [], + isCoinbase: false, + sequence: 1, + innerRedeemscriptAsm: "innerRedeemscriptAsm", + ); //Input + + expect(input.toString(), "{txid: txid}"); + }); + + test("Input.toString", () { + final input = Input.fromJson({ + "txid": "txid", + "vout": 1, + "prevout": null, + "scriptSig": {"hex": "somehexString", "asm": "someasmthing"}, + "scriptsigAsm": "scriptsigAsm", + "witness": [], + "isCoinbase": false, + "sequence": 1, + "innerRedeemscriptAsm": "innerRedeemscriptAsm", + }); //Input + + expect(input.toString(), "{txid: txid}"); + }); + }); + + group("Output", () { + test("Output.toString", () { + final output = Output.fromJson({ + "scriptPubKey": { + "hex": "somehexSting", + "asm": "someasmthing", + "type": "sometype", + "addresses": "someaddresses", + }, + "scriptpubkeyAsm": "scriptpubkeyAsm", + "scriptpubkeyType": "scriptpubkeyType", + "scriptpubkeyAddress": "address", + "value": 2, + }); //Input + + expect(output.toString(), "Instance of \'Output\'"); + }); + }); } diff --git a/test/price_test.dart b/test/price_test.dart index ac1110df5..b806bfd61 100644 --- a/test/price_test.dart +++ b/test/price_test.dart @@ -23,7 +23,7 @@ void main() { when(client.get( Uri.parse( - "https://api.coingecko.com/api/v3/coins/markets?vs_currency=btc&ids=monero,bitcoin,epic-cash,zcoin,dogecoin,bitcoin-cash,namecoin&order=market_cap_desc&per_page=10&page=1&sparkline=false"), + "https://api.coingecko.com/api/v3/coins/markets?vs_currency=btc&ids=monero,bitcoin,epic-cash,zcoin,dogecoin,bitcoin-cash,namecoin,wownero&order=market_cap_desc&per_page=10&page=1&sparkline=false"), headers: { 'Content-Type': 'application/json' })).thenAnswer((_) async => Response( @@ -36,10 +36,10 @@ void main() { final price = await priceAPI.getPricesAnd24hChange(baseCurrency: "btc"); expect(price.toString(), - '{Coin.bitcoin: [1, 0.0], Coin.bitcoincash: [0, 0.0], Coin.dogecoin: [0.00000315, -2.68533], Coin.epicCash: [0.00002803, 7.27524], Coin.firo: [0.0001096, -0.89304], Coin.monero: [0.00717236, -0.77656], Coin.namecoin: [0, 0.0], Coin.bitcoinTestNet: [0, 0.0], Coin.bitcoincashTestnet: [0, 0.0], Coin.dogecoinTestNet: [0, 0.0], Coin.firoTestNet: [0, 0.0]}'); + '{Coin.bitcoin: [1, 0.0], Coin.bitcoincash: [0, 0.0], Coin.dogecoin: [0.00000315, -2.68533], Coin.epicCash: [0.00002803, 7.27524], Coin.firo: [0.0001096, -0.89304], Coin.monero: [0.00717236, -0.77656], Coin.wownero: [0, 0.0], Coin.namecoin: [0, 0.0], Coin.bitcoinTestNet: [0, 0.0], Coin.bitcoincashTestnet: [0, 0.0], Coin.dogecoinTestNet: [0, 0.0], Coin.firoTestNet: [0, 0.0]}'); verify(client.get( Uri.parse( - "https://api.coingecko.com/api/v3/coins/markets?vs_currency=btc&ids=monero,bitcoin,epic-cash,zcoin,dogecoin,bitcoin-cash,namecoin&order=market_cap_desc&per_page=10&page=1&sparkline=false"), + "https://api.coingecko.com/api/v3/coins/markets?vs_currency=btc&ids=monero,bitcoin,epic-cash,zcoin,dogecoin,bitcoin-cash,namecoin,wownero&order=market_cap_desc&per_page=10&page=1&sparkline=false"), headers: {'Content-Type': 'application/json'})).called(1); verifyNoMoreInteractions(client); @@ -50,7 +50,7 @@ void main() { when(client.get( Uri.parse( - "https://api.coingecko.com/api/v3/coins/markets?vs_currency=btc&ids=monero,bitcoin,epic-cash,zcoin,dogecoin,bitcoin-cash,namecoin&order=market_cap_desc&per_page=10&page=1&sparkline=false"), + "https://api.coingecko.com/api/v3/coins/markets?vs_currency=btc&ids=monero,bitcoin,epic-cash,zcoin,dogecoin,bitcoin-cash,namecoin,wownero&order=market_cap_desc&per_page=10&page=1&sparkline=false"), headers: { 'Content-Type': 'application/json' })).thenAnswer((_) async => Response( @@ -68,12 +68,12 @@ void main() { await priceAPI.getPricesAnd24hChange(baseCurrency: "btc"); expect(cachedPrice.toString(), - '{Coin.bitcoin: [1, 0.0], Coin.bitcoincash: [0, 0.0], Coin.dogecoin: [0.00000315, -2.68533], Coin.epicCash: [0.00002803, 7.27524], Coin.firo: [0.0001096, -0.89304], Coin.monero: [0.00717236, -0.77656], Coin.namecoin: [0, 0.0], Coin.bitcoinTestNet: [0, 0.0], Coin.bitcoincashTestnet: [0, 0.0], Coin.dogecoinTestNet: [0, 0.0], Coin.firoTestNet: [0, 0.0]}'); + '{Coin.bitcoin: [1, 0.0], Coin.bitcoincash: [0, 0.0], Coin.dogecoin: [0.00000315, -2.68533], Coin.epicCash: [0.00002803, 7.27524], Coin.firo: [0.0001096, -0.89304], Coin.monero: [0.00717236, -0.77656], Coin.wownero: [0, 0.0], Coin.namecoin: [0, 0.0], Coin.bitcoinTestNet: [0, 0.0], Coin.bitcoincashTestnet: [0, 0.0], Coin.dogecoinTestNet: [0, 0.0], Coin.firoTestNet: [0, 0.0]}'); // verify only called once during filling of cache verify(client.get( Uri.parse( - "https://api.coingecko.com/api/v3/coins/markets?vs_currency=btc&ids=monero,bitcoin,epic-cash,zcoin,dogecoin,bitcoin-cash,namecoin&order=market_cap_desc&per_page=10&page=1&sparkline=false"), + "https://api.coingecko.com/api/v3/coins/markets?vs_currency=btc&ids=monero,bitcoin,epic-cash,zcoin,dogecoin,bitcoin-cash,namecoin,wownero&order=market_cap_desc&per_page=10&page=1&sparkline=false"), headers: {'Content-Type': 'application/json'})).called(1); verifyNoMoreInteractions(client); @@ -84,7 +84,7 @@ void main() { when(client.get( Uri.parse( - "https://api.coingecko.com/api/v3/coins/markets?vs_currency=btc&ids=monero,bitcoin,epic-cash,zcoin,dogecoin,bitcoin-cash,namecoin&order=market_cap_desc&per_page=10&page=1&sparkline=false"), + "https://api.coingecko.com/api/v3/coins/markets?vs_currency=btc&ids=monero,bitcoin,epic-cash,zcoin,dogecoin,bitcoin-cash,namecoin,wownero&order=market_cap_desc&per_page=10&page=1&sparkline=false"), headers: { 'Content-Type': 'application/json' })).thenAnswer((_) async => Response( @@ -97,7 +97,7 @@ void main() { final price = await priceAPI.getPricesAnd24hChange(baseCurrency: "btc"); expect(price.toString(), - '{Coin.bitcoin: [0, 0.0], Coin.bitcoincash: [0, 0.0], Coin.dogecoin: [0, 0.0], Coin.epicCash: [0, 0.0], Coin.firo: [0, 0.0], Coin.monero: [0, 0.0], Coin.namecoin: [0, 0.0], Coin.bitcoinTestNet: [0, 0.0], Coin.bitcoincashTestnet: [0, 0.0], Coin.dogecoinTestNet: [0, 0.0], Coin.firoTestNet: [0, 0.0]}'); + '{Coin.bitcoin: [0, 0.0], Coin.bitcoincash: [0, 0.0], Coin.dogecoin: [0, 0.0], Coin.epicCash: [0, 0.0], Coin.firo: [0, 0.0], Coin.monero: [0, 0.0], Coin.wownero: [0, 0.0], Coin.namecoin: [0, 0.0], Coin.bitcoinTestNet: [0, 0.0], Coin.bitcoincashTestnet: [0, 0.0], Coin.dogecoinTestNet: [0, 0.0], Coin.firoTestNet: [0, 0.0]}'); }); test("no internet available", () async { @@ -105,7 +105,7 @@ void main() { when(client.get( Uri.parse( - "https://api.coingecko.com/api/v3/coins/markets?vs_currency=btc&ids=monero,bitcoin,epic-cash,zcoin,dogecoin,bitcoin-cash,namecoin&order=market_cap_desc&per_page=10&page=1&sparkline=false"), + "https://api.coingecko.com/api/v3/coins/markets?vs_currency=btc&ids=monero,bitcoin,epic-cash,zcoin,dogecoin,bitcoin-cash,namecoin,wownero&order=market_cap_desc&per_page=10&page=1&sparkline=false"), headers: { 'Content-Type': 'application/json' })).thenThrow(const SocketException( @@ -117,7 +117,7 @@ void main() { final price = await priceAPI.getPricesAnd24hChange(baseCurrency: "btc"); expect(price.toString(), - '{Coin.bitcoin: [0, 0.0], Coin.bitcoincash: [0, 0.0], Coin.dogecoin: [0, 0.0], Coin.epicCash: [0, 0.0], Coin.firo: [0, 0.0], Coin.monero: [0, 0.0], Coin.namecoin: [0, 0.0], Coin.bitcoinTestNet: [0, 0.0], Coin.bitcoincashTestnet: [0, 0.0], Coin.dogecoinTestNet: [0, 0.0], Coin.firoTestNet: [0, 0.0]}'); + '{Coin.bitcoin: [0, 0.0], Coin.bitcoincash: [0, 0.0], Coin.dogecoin: [0, 0.0], Coin.epicCash: [0, 0.0], Coin.firo: [0, 0.0], Coin.monero: [0, 0.0], Coin.wownero: [0, 0.0], Coin.namecoin: [0, 0.0], Coin.bitcoinTestNet: [0, 0.0], Coin.bitcoincashTestnet: [0, 0.0], Coin.dogecoinTestNet: [0, 0.0], Coin.firoTestNet: [0, 0.0]}'); }); tearDown(() async {