import 'dart:io' show Directory, File, Platform; import 'package:cake_wallet/bitcoin/bitcoin.dart'; import 'package:cake_wallet/entities/exchange_api_mode.dart'; import 'package:cw_core/pathForWallet.dart'; import 'package:cake_wallet/entities/secret_store_key.dart'; import 'package:flutter_secure_storage/flutter_secure_storage.dart'; import 'package:hive/hive.dart'; import 'package:path_provider/path_provider.dart'; import 'package:shared_preferences/shared_preferences.dart'; import 'package:cake_wallet/entities/preferences_key.dart'; import 'package:cw_core/wallet_type.dart'; import 'package:cw_core/node.dart'; import 'package:cake_wallet/entities/balance_display_mode.dart'; import 'package:cake_wallet/entities/fiat_currency.dart'; import 'package:cake_wallet/entities/node_list.dart'; import 'package:cake_wallet/monero/monero.dart'; import 'package:cake_wallet/entities/contact.dart'; import 'package:cake_wallet/entities/fs_migration.dart'; import 'package:cw_core/wallet_info.dart'; import 'package:cake_wallet/exchange/trade.dart'; import 'package:encrypt/encrypt.dart' as encrypt; import 'package:collection/collection.dart'; const newCakeWalletMoneroUri = 'xmr-node.cakewallet.com:18081'; const cakeWalletBitcoinElectrumUri = 'electrum.cakewallet.com:50002'; const cakeWalletLitecoinElectrumUri = 'ltc-electrum.cakewallet.com:50002'; const havenDefaultNodeUri = 'nodes.havenprotocol.org:443'; const ethereumDefaultNodeUri = 'ethereum.publicnode.com'; const nanoDefaultNodeUri = 'rpc.nano.to'; const nanoDefaultPowNodeUri = 'rpc.nano.to'; Future defaultSettingsMigration( {required int version, required SharedPreferences sharedPreferences, required FlutterSecureStorage secureStorage, required Box nodes, required Box powNodes, required Box walletInfoSource, required Box tradeSource, required Box contactSource}) async { if (Platform.isIOS) { await ios_migrate_v1(walletInfoSource, tradeSource, contactSource); } // check current nodes for nullability regardless of the version await checkCurrentNodes(nodes, powNodes, sharedPreferences); final isNewInstall = sharedPreferences.getInt(PreferencesKey.currentDefaultSettingsMigrationVersion) == null; await _validateWalletInfoBoxData(walletInfoSource); await sharedPreferences.setBool(PreferencesKey.isNewInstall, isNewInstall); final currentVersion = sharedPreferences.getInt(PreferencesKey.currentDefaultSettingsMigrationVersion) ?? 0; if (currentVersion >= version) { return; } final migrationVersionsLength = version - currentVersion; final migrationVersions = List.generate(migrationVersionsLength, (i) => currentVersion + (i + 1)); await Future.forEach(migrationVersions, (int version) async { try { switch (version) { case 1: await sharedPreferences.setString( PreferencesKey.currentFiatCurrencyKey, FiatCurrency.usd.toString()); await sharedPreferences.setInt(PreferencesKey.currentTransactionPriorityKeyLegacy, monero!.getDefaultTransactionPriority().raw); await sharedPreferences.setInt( PreferencesKey.currentBalanceDisplayModeKey, BalanceDisplayMode.availableBalance.raw); await sharedPreferences.setBool('save_recipient_address', true); await resetToDefault(nodes); await changeMoneroCurrentNodeToDefault( sharedPreferences: sharedPreferences, nodes: nodes); await changeBitcoinCurrentElectrumServerToDefault( sharedPreferences: sharedPreferences, nodes: nodes); await changeLitecoinCurrentElectrumServerToDefault( sharedPreferences: sharedPreferences, nodes: nodes); await changeHavenCurrentNodeToDefault(sharedPreferences: sharedPreferences, nodes: nodes); break; case 2: await replaceNodesMigration(nodes: nodes); await replaceDefaultNode(sharedPreferences: sharedPreferences, nodes: nodes); break; case 3: await updateNodeTypes(nodes: nodes); await addBitcoinElectrumServerList(nodes: nodes); break; case 4: await changeBitcoinCurrentElectrumServerToDefault( sharedPreferences: sharedPreferences, nodes: nodes); break; case 5: await addAddressesForMoneroWallets(walletInfoSource); break; case 6: await updateDisplayModes(sharedPreferences); break; case 9: await generateBackupPassword(secureStorage); break; case 10: await changeTransactionPriorityAndFeeRateKeys(sharedPreferences); break; case 11: await changeDefaultMoneroNode(nodes, sharedPreferences); break; case 12: await checkCurrentNodes(nodes, powNodes, sharedPreferences); break; case 13: await resetBitcoinElectrumServer(nodes, sharedPreferences); break; case 15: await addLitecoinElectrumServerList(nodes: nodes); await changeLitecoinCurrentElectrumServerToDefault( sharedPreferences: sharedPreferences, nodes: nodes); await checkCurrentNodes(nodes, powNodes, sharedPreferences); break; case 16: await addHavenNodeList(nodes: nodes); await changeHavenCurrentNodeToDefault(sharedPreferences: sharedPreferences, nodes: nodes); await checkCurrentNodes(nodes, powNodes, sharedPreferences); break; case 17: await changeDefaultHavenNode(nodes); break; case 18: await addOnionNode(nodes); break; case 19: await validateBitcoinSavedTransactionPriority(sharedPreferences); break; case 20: await migrateExchangeStatus(sharedPreferences); break; case 21: await addEthereumNodeList(nodes: nodes); await changeEthereumCurrentNodeToDefault( sharedPreferences: sharedPreferences, nodes: nodes); break; case 22: await addNanoNodeList(nodes: nodes); await addNanoPowNodeList(nodes: powNodes); await changeNanoCurrentNodeToDefault(sharedPreferences: sharedPreferences, nodes: nodes); await changeNanoCurrentPowNodeToDefault( sharedPreferences: sharedPreferences, nodes: powNodes); break; case 23: // TODO: await updateBtcAndNanoWalletInfos() break; default: break; } await sharedPreferences.setInt( PreferencesKey.currentDefaultSettingsMigrationVersion, version); } catch (e) { print('Migration error: ${e.toString()}'); } }); await sharedPreferences.setInt(PreferencesKey.currentDefaultSettingsMigrationVersion, version); } Future _validateWalletInfoBoxData(Box walletInfoSource) async { final root = await getApplicationDocumentsDirectory(); for (var type in WalletType.values) { if (type == WalletType.none) { continue; } String prefix = walletTypeToString(type).toLowerCase(); Directory walletsDir = Directory('${root.path}/wallets/$prefix/'); if (!walletsDir.existsSync()) { continue; } List walletNames = walletsDir.listSync().map((e) => e.path.split("/").last).toList(); for (var name in walletNames) { final dir = Directory(await pathForWalletDir(name: name, type: type)); final walletFiles = dir.listSync(); final hasCacheFile = walletFiles.any((element) => element.path.contains("$name/$name")); if (!hasCacheFile) { continue; } if (type == WalletType.monero || type == WalletType.haven) { final hasKeysFile = walletFiles.any((element) => element.path.contains(".keys")); if (!hasKeysFile) { continue; } } final id = prefix + '_' + name; final exist = walletInfoSource.values.any((el) => el.id == id); if (exist) { continue; } final walletInfo = WalletInfo.external( id: id, type: type, name: name, isRecovery: true, restoreHeight: 0, date: DateTime.now(), dirPath: dir.path, path: '${dir.path}/$name', address: '', showIntroCakePayCard: false, ); walletInfoSource.add(walletInfo); } } } Future validateBitcoinSavedTransactionPriority(SharedPreferences sharedPreferences) async { if (bitcoin == null) { return; } final int? savedBitcoinPriority = sharedPreferences.getInt(PreferencesKey.bitcoinTransactionPriority); if (!bitcoin!.getTransactionPriorities().any((element) => element.raw == savedBitcoinPriority)) { await sharedPreferences.setInt(PreferencesKey.bitcoinTransactionPriority, bitcoin!.getMediumTransactionPriority().serialize()); } } Future addOnionNode(Box nodes) async { final onionNodeUri = "cakexmrl7bonq7ovjka5kuwuyd3f7qnkz6z6s6dmsy3uckwra7bvggyd.onion:18081"; // check if the user has this node before (added it manually) if (nodes.values.firstWhereOrNull((element) => element.uriRaw == onionNodeUri) == null) { await nodes.add(Node(uri: onionNodeUri, type: WalletType.monero)); } } Future replaceNodesMigration({required Box nodes}) async { final replaceNodes = { 'eu-node.cakewallet.io:18081': Node(uri: 'xmr-node-eu.cakewallet.com:18081', type: WalletType.monero), 'node.cakewallet.io:18081': Node(uri: 'xmr-node-usa-east.cakewallet.com:18081', type: WalletType.monero), 'node.xmr.ru:13666': Node(uri: 'node.monero.net:18081', type: WalletType.monero) }; nodes.values.forEach((Node node) async { final nodeToReplace = replaceNodes[node.uri]; if (nodeToReplace != null) { node.uriRaw = nodeToReplace.uriRaw; node.login = nodeToReplace.login; node.password = nodeToReplace.password; await node.save(); } }); } Future changeMoneroCurrentNodeToDefault( {required SharedPreferences sharedPreferences, required Box nodes}) async { final node = getMoneroDefaultNode(nodes: nodes); final nodeId = node.key as int? ?? 0; // 0 - England await sharedPreferences.setInt(PreferencesKey.currentNodeIdKey, nodeId); } Node? getBitcoinDefaultElectrumServer({required Box nodes}) { return nodes.values .firstWhereOrNull((Node node) => node.uriRaw == cakeWalletBitcoinElectrumUri) ?? nodes.values.firstWhereOrNull((node) => node.type == WalletType.bitcoin); } Node? getLitecoinDefaultElectrumServer({required Box nodes}) { return nodes.values .firstWhereOrNull((Node node) => node.uriRaw == cakeWalletLitecoinElectrumUri) ?? nodes.values.firstWhereOrNull((node) => node.type == WalletType.litecoin); } Node? getHavenDefaultNode({required Box nodes}) { return nodes.values.firstWhereOrNull((Node node) => node.uriRaw == havenDefaultNodeUri) ?? nodes.values.firstWhereOrNull((node) => node.type == WalletType.haven); } Node? getEthereumDefaultNode({required Box nodes}) { return nodes.values.firstWhereOrNull((Node node) => node.uriRaw == ethereumDefaultNodeUri) ?? nodes.values.firstWhereOrNull((node) => node.type == WalletType.ethereum); } Node? getNanoDefaultNode({required Box nodes}) { return nodes.values.firstWhereOrNull((Node node) => node.uriRaw == nanoDefaultNodeUri) ?? nodes.values.firstWhereOrNull((node) => node.type == WalletType.nano); } Node? getNanoDefaultPowNode({required Box nodes}) { return nodes.values.firstWhereOrNull((Node node) => node.uriRaw == nanoDefaultPowNodeUri) ?? nodes.values.firstWhereOrNull((node) => (node.type == WalletType.nano)); } Node getMoneroDefaultNode({required Box nodes}) { final timeZone = DateTime.now().timeZoneOffset.inHours; var nodeUri = ''; if (timeZone >= 1) { // Eurasia nodeUri = 'xmr-node-eu.cakewallet.com:18081'; } else if (timeZone <= -4) { // America nodeUri = 'xmr-node-usa-east.cakewallet.com:18081'; } try { return nodes.values.firstWhere((Node node) => node.uriRaw == nodeUri); } catch (_) { return nodes.values.first; } } Future changeBitcoinCurrentElectrumServerToDefault( {required SharedPreferences sharedPreferences, required Box nodes}) async { final server = getBitcoinDefaultElectrumServer(nodes: nodes); final serverId = server?.key as int? ?? 0; await sharedPreferences.setInt(PreferencesKey.currentBitcoinElectrumSererIdKey, serverId); } Future changeLitecoinCurrentElectrumServerToDefault( {required SharedPreferences sharedPreferences, required Box nodes}) async { final server = getLitecoinDefaultElectrumServer(nodes: nodes); final serverId = server?.key as int? ?? 0; await sharedPreferences.setInt(PreferencesKey.currentLitecoinElectrumSererIdKey, serverId); } Future changeHavenCurrentNodeToDefault( {required SharedPreferences sharedPreferences, required Box nodes}) async { final node = getHavenDefaultNode(nodes: nodes); final nodeId = node?.key as int? ?? 0; await sharedPreferences.setInt(PreferencesKey.currentHavenNodeIdKey, nodeId); } Future replaceDefaultNode( {required SharedPreferences sharedPreferences, required Box nodes}) async { const nodesForReplace = [ 'xmr-node-uk.cakewallet.com:18081', 'eu-node.cakewallet.io:18081', 'node.cakewallet.io:18081' ]; final currentNodeId = sharedPreferences.getInt(PreferencesKey.currentNodeIdKey); final currentNode = nodes.values.firstWhereOrNull((Node node) => node.key == currentNodeId); final needToReplace = currentNode == null ? true : nodesForReplace.contains(currentNode.uriRaw); if (!needToReplace) { return; } await changeMoneroCurrentNodeToDefault(sharedPreferences: sharedPreferences, nodes: nodes); } Future updateNodeTypes({required Box nodes}) async { nodes.values.forEach((node) async { if (node.type == null) { node.type = WalletType.monero; await node.save(); } }); } Future addBitcoinElectrumServerList({required Box nodes}) async { final serverList = await loadBitcoinElectrumServerList(); for (var node in serverList) { if (nodes.values.firstWhereOrNull((element) => element.uriRaw == node.uriRaw) == null) { await nodes.add(node); } } } Future addLitecoinElectrumServerList({required Box nodes}) async { final serverList = await loadLitecoinElectrumServerList(); for (var node in serverList) { if (nodes.values.firstWhereOrNull((element) => element.uriRaw == node.uriRaw) == null) { await nodes.add(node); } } } Future addHavenNodeList({required Box nodes}) async { final nodeList = await loadDefaultHavenNodes(); for (var node in nodeList) { if (nodes.values.firstWhereOrNull((element) => element.uriRaw == node.uriRaw) == null) { await nodes.add(node); } } } Future addAddressesForMoneroWallets(Box walletInfoSource) async { final moneroWalletsInfo = walletInfoSource.values.where((info) => info.type == WalletType.monero); moneroWalletsInfo.forEach((info) async { try { final walletPath = await pathForWallet(name: info.name, type: WalletType.monero); final addressFilePath = '$walletPath.address.txt'; final addressFile = File(addressFilePath); if (!addressFile.existsSync()) { return; } final addressText = await addressFile.readAsString(); info.address = addressText; await info.save(); } catch (e) { print(e.toString()); } }); } Future updateDisplayModes(SharedPreferences sharedPreferences) async { final currentBalanceDisplayMode = sharedPreferences.getInt(PreferencesKey.currentBalanceDisplayModeKey) ?? -1; final balanceDisplayMode = currentBalanceDisplayMode < 2 ? 3 : 2; await sharedPreferences.setInt(PreferencesKey.currentBalanceDisplayModeKey, balanceDisplayMode); } Future generateBackupPassword(FlutterSecureStorage secureStorage) async { final key = generateStoreKeyFor(key: SecretStoreKey.backupPassword); if ((await secureStorage.read(key: key))?.isNotEmpty ?? false) { return; } final password = encrypt.Key.fromSecureRandom(32).base16; await secureStorage.write(key: key, value: password); } Future changeTransactionPriorityAndFeeRateKeys(SharedPreferences sharedPreferences) async { final legacyTransactionPriority = sharedPreferences.getInt(PreferencesKey.currentTransactionPriorityKeyLegacy)!; await sharedPreferences.setInt( PreferencesKey.moneroTransactionPriority, legacyTransactionPriority); await sharedPreferences.setInt(PreferencesKey.bitcoinTransactionPriority, bitcoin!.getMediumTransactionPriority().serialize()); } Future changeDefaultMoneroNode( Box nodeSource, SharedPreferences sharedPreferences) async { const cakeWalletMoneroNodeUriPattern = '.cakewallet.com'; final currentMoneroNodeId = sharedPreferences.getInt(PreferencesKey.currentNodeIdKey); final currentMoneroNode = nodeSource.values.firstWhere((node) => node.key == currentMoneroNodeId); final needToReplaceCurrentMoneroNode = currentMoneroNode.uri.toString().contains(cakeWalletMoneroNodeUriPattern); nodeSource.values.forEach((node) async { if (node.type == WalletType.monero && node.uri.toString().contains(cakeWalletMoneroNodeUriPattern)) { await node.delete(); } }); final newCakeWalletNode = Node(uri: newCakeWalletMoneroUri, type: WalletType.monero); await nodeSource.add(newCakeWalletNode); if (needToReplaceCurrentMoneroNode) { await sharedPreferences.setInt(PreferencesKey.currentNodeIdKey, newCakeWalletNode.key as int); } } Future checkCurrentNodes( Box nodeSource, Box powNodeSource, SharedPreferences sharedPreferences) async { final currentMoneroNodeId = sharedPreferences.getInt(PreferencesKey.currentNodeIdKey); final currentBitcoinElectrumSeverId = sharedPreferences.getInt(PreferencesKey.currentBitcoinElectrumSererIdKey); final currentLitecoinElectrumSeverId = sharedPreferences.getInt(PreferencesKey.currentLitecoinElectrumSererIdKey); final currentHavenNodeId = sharedPreferences.getInt(PreferencesKey.currentHavenNodeIdKey); final currentEthereumNodeId = sharedPreferences.getInt(PreferencesKey.currentEthereumNodeIdKey); final currentNanoNodeId = sharedPreferences.getInt(PreferencesKey.currentNanoNodeIdKey); final currentNanoPowNodeId = sharedPreferences.getInt(PreferencesKey.currentNanoPowNodeIdKey); final currentMoneroNode = nodeSource.values.firstWhereOrNull((node) => node.key == currentMoneroNodeId); final currentBitcoinElectrumServer = nodeSource.values.firstWhereOrNull((node) => node.key == currentBitcoinElectrumSeverId); final currentLitecoinElectrumServer = nodeSource.values.firstWhereOrNull((node) => node.key == currentLitecoinElectrumSeverId); final currentHavenNodeServer = nodeSource.values.firstWhereOrNull((node) => node.key == currentHavenNodeId); final currentEthereumNodeServer = nodeSource.values.firstWhereOrNull((node) => node.key == currentEthereumNodeId); final currentNanoNodeServer = nodeSource.values.firstWhereOrNull((node) => node.key == currentNanoNodeId); final currentNanoPowNodeServer = powNodeSource.values.firstWhereOrNull((node) => node.key == currentNanoPowNodeId); if (currentMoneroNode == null) { final newCakeWalletNode = Node(uri: newCakeWalletMoneroUri, type: WalletType.monero); await nodeSource.add(newCakeWalletNode); await sharedPreferences.setInt(PreferencesKey.currentNodeIdKey, newCakeWalletNode.key as int); } if (currentBitcoinElectrumServer == null) { final cakeWalletElectrum = Node(uri: cakeWalletBitcoinElectrumUri, type: WalletType.bitcoin); await nodeSource.add(cakeWalletElectrum); await sharedPreferences.setInt( PreferencesKey.currentBitcoinElectrumSererIdKey, cakeWalletElectrum.key as int); } if (currentLitecoinElectrumServer == null) { final cakeWalletElectrum = Node(uri: cakeWalletLitecoinElectrumUri, type: WalletType.litecoin); await nodeSource.add(cakeWalletElectrum); await sharedPreferences.setInt( PreferencesKey.currentLitecoinElectrumSererIdKey, cakeWalletElectrum.key as int); } if (currentHavenNodeServer == null) { final node = Node(uri: havenDefaultNodeUri, type: WalletType.haven); await nodeSource.add(node); await sharedPreferences.setInt(PreferencesKey.currentHavenNodeIdKey, node.key as int); } if (currentEthereumNodeServer == null) { final node = Node(uri: ethereumDefaultNodeUri, type: WalletType.ethereum); await nodeSource.add(node); await sharedPreferences.setInt(PreferencesKey.currentEthereumNodeIdKey, node.key as int); } if (currentNanoNodeServer == null) { final node = Node(uri: nanoDefaultNodeUri, useSSL: true, type: WalletType.nano); await nodeSource.add(node); await sharedPreferences.setInt(PreferencesKey.currentNanoNodeIdKey, node.key as int); } if (currentNanoPowNodeServer == null) { Node? node = powNodeSource.values .firstWhereOrNull((node) => node.uri.toString() == nanoDefaultPowNodeUri); if (node == null) { node = Node(uri: nanoDefaultPowNodeUri, useSSL: true, type: WalletType.nano); await powNodeSource.add(node); } await sharedPreferences.setInt(PreferencesKey.currentNanoPowNodeIdKey, node.key as int); } } Future resetBitcoinElectrumServer( Box nodeSource, SharedPreferences sharedPreferences) async { final currentElectrumSeverId = sharedPreferences.getInt(PreferencesKey.currentBitcoinElectrumSererIdKey); final oldElectrumServer = nodeSource.values .firstWhereOrNull((node) => node.uri.toString().contains('electrumx.cakewallet.com')); var cakeWalletNode = nodeSource.values .firstWhereOrNull((node) => node.uriRaw.toString() == cakeWalletBitcoinElectrumUri); if (cakeWalletNode == null) { cakeWalletNode = Node(uri: cakeWalletBitcoinElectrumUri, type: WalletType.bitcoin); await nodeSource.add(cakeWalletNode); } if (currentElectrumSeverId == oldElectrumServer?.key) { await sharedPreferences.setInt( PreferencesKey.currentBitcoinElectrumSererIdKey, cakeWalletNode.key as int); } await oldElectrumServer?.delete(); } Future changeDefaultHavenNode(Box nodeSource) async { const previousHavenDefaultNodeUri = 'vault.havenprotocol.org:443'; final havenNodes = nodeSource.values.where((node) => node.uriRaw == previousHavenDefaultNodeUri); havenNodes.forEach((node) async { node.uriRaw = havenDefaultNodeUri; await node.save(); }); } Future migrateExchangeStatus(SharedPreferences sharedPreferences) async { final isExchangeDisabled = sharedPreferences.getBool(PreferencesKey.disableExchangeKey); if (isExchangeDisabled == null) { return; } await sharedPreferences.setInt(PreferencesKey.exchangeStatusKey, isExchangeDisabled ? ExchangeApiMode.disabled.raw : ExchangeApiMode.enabled.raw); await sharedPreferences.remove(PreferencesKey.disableExchangeKey); } Future addEthereumNodeList({required Box nodes}) async { final nodeList = await loadDefaultEthereumNodes(); for (var node in nodeList) { if (nodes.values.firstWhereOrNull((element) => element.uriRaw == node.uriRaw) == null) { await nodes.add(node); } } } Future changeEthereumCurrentNodeToDefault( {required SharedPreferences sharedPreferences, required Box nodes}) async { final node = getEthereumDefaultNode(nodes: nodes); final nodeId = node?.key as int? ?? 0; await sharedPreferences.setInt(PreferencesKey.currentEthereumNodeIdKey, nodeId); } Future addNanoNodeList({required Box nodes}) async { final nodeList = await loadDefaultNanoNodes(); for (var node in nodeList) { if (nodes.values.firstWhereOrNull((element) => element.uriRaw == node.uriRaw) == null) { await nodes.add(node); } } } Future addNanoPowNodeList({required Box nodes}) async { final nodeList = await loadDefaultNanoPowNodes(); for (var node in nodeList) { if (nodes.values.firstWhereOrNull((element) => element.uriRaw == node.uriRaw) == null) { await nodes.add(node); } } } Future changeNanoCurrentNodeToDefault( {required SharedPreferences sharedPreferences, required Box nodes}) async { final node = getNanoDefaultNode(nodes: nodes); final nodeId = node?.key as int? ?? 0; await sharedPreferences.setInt(PreferencesKey.currentNanoNodeIdKey, nodeId); } Future changeNanoCurrentPowNodeToDefault( {required SharedPreferences sharedPreferences, required Box nodes}) async { final node = getNanoDefaultPowNode(nodes: nodes); final nodeId = node?.key as int? ?? 0; await sharedPreferences.setInt(PreferencesKey.currentNanoPowNodeIdKey, nodeId); }