import 'dart:async'; import 'dart:io'; import 'dart:ui'; import 'package:battery_plus/battery_plus.dart'; import 'package:cake_wallet/bitcoin/bitcoin.dart'; import 'package:cake_wallet/core/wallet_loading_service.dart'; import 'package:cake_wallet/store/settings_store.dart'; import 'package:cake_wallet/utils/device_info.dart'; import 'package:cake_wallet/utils/feature_flag.dart'; import 'package:cake_wallet/view_model/settings/sync_mode.dart'; import 'package:cake_wallet/view_model/wallet_list/wallet_list_item.dart'; import 'package:cake_wallet/view_model/wallet_list/wallet_list_view_model.dart'; import 'package:connectivity_plus/connectivity_plus.dart'; import 'package:cw_bitcoin/electrum_wallet.dart'; import 'package:cw_core/node.dart'; import 'package:cw_core/sync_status.dart'; import 'package:cw_core/utils/print_verbose.dart'; import 'package:cw_core/wallet_base.dart'; import 'package:cw_core/wallet_type.dart'; import 'package:flutter/widgets.dart'; import 'package:flutter_background_service/flutter_background_service.dart'; import 'package:flutter_local_notifications/flutter_local_notifications.dart'; import 'package:cake_wallet/main.dart'; import 'package:cake_wallet/di.dart'; const initialNotificationTitle = "Cake Background Sync"; const standbyMessage = "On standby - app is in the foreground"; const readyMessage = "Ready to sync - waiting until the app has been in the background for a while"; const startMessage = "Starting sync - app is in the background"; const allWalletsSyncedMessage = "All wallets synced - waiting for next queue refresh"; const notificationId = 888; const notificationChannelId = "cake_service"; const notificationChannelName = "CAKE BACKGROUND SERVICE"; const notificationChannelDescription = "Cake Wallet Background Service"; const DELAY_SECONDS_BEFORE_SYNC_START = 15; const spNodeNotificationMessage = "Currently configured Bitcoin node does not support Silent Payments. skipping wallet"; const SYNC_THRESHOLD = 0.98; Duration REFRESH_QUEUE_DURATION = Duration(hours: 1); bool syncOnBattery = false; bool syncOnData = false; void setMainNotification( FlutterLocalNotificationsPlugin flutterLocalNotificationsPlugin, { required String title, required String content, }) async { flutterLocalNotificationsPlugin.show( notificationId, title, content, const NotificationDetails( android: AndroidNotificationDetails( notificationChannelId, notificationChannelName, icon: "ic_bg_service_small", ongoing: true, silent: true, ), ), ); } void setNotificationStandby(FlutterLocalNotificationsPlugin flutterLocalNotificationsPlugin) async { flutterLocalNotificationsPlugin.cancelAll(); setMainNotification( flutterLocalNotificationsPlugin, title: initialNotificationTitle, content: standbyMessage, ); } void setNotificationReady(FlutterLocalNotificationsPlugin flutterLocalNotificationsPlugin) async { flutterLocalNotificationsPlugin.cancelAll(); setMainNotification( flutterLocalNotificationsPlugin, title: initialNotificationTitle, content: readyMessage, ); } void setNotificationStarting( FlutterLocalNotificationsPlugin flutterLocalNotificationsPlugin) async { flutterLocalNotificationsPlugin.cancelAll(); setMainNotification( flutterLocalNotificationsPlugin, title: initialNotificationTitle, content: startMessage, ); } void setNotificationWalletsSynced( FlutterLocalNotificationsPlugin flutterLocalNotificationsPlugin) async { flutterLocalNotificationsPlugin.cancelAll(); setMainNotification( flutterLocalNotificationsPlugin, title: initialNotificationTitle, content: allWalletsSyncedMessage, ); } void setWalletNotification(FlutterLocalNotificationsPlugin flutterLocalNotificationsPlugin, {required String title, required String content, required int walletNum}) async { flutterLocalNotificationsPlugin.show( notificationId + walletNum, title, content, NotificationDetails( android: AndroidNotificationDetails( "${notificationChannelId}_$walletNum", "${notificationChannelName}_$walletNum", icon: "ic_bg_service_small", ongoing: true, silent: true, ), ), ); } @pragma("vm:entry-point") Future onStart(ServiceInstance service) async { printV("BACKGROUND SERVICE STARTED"); bool bgSyncStarted = false; Timer? _syncTimer; Timer? _stuckSyncTimer; Timer? _queueTimer; List syncingWallets = []; List standbyWallets = []; Timer? _bgTimer; int fgCount = 0; // commented because the behavior appears to be bugged: // DartPluginRegistrant.ensureInitialized(); final FlutterLocalNotificationsPlugin flutterLocalNotificationsPlugin = FlutterLocalNotificationsPlugin(); service.on("stopService").listen((event) async { printV("STOPPING BACKGROUND SERVICE"); _syncTimer?.cancel(); _stuckSyncTimer?.cancel(); _queueTimer?.cancel(); try { // stop all syncing wallets: for (int i = 0; i < syncingWallets.length; i++) { final wallet = syncingWallets[i]; await wallet.stopSync(isBackgroundSync: true); await wallet.close(); } // stop all standby wallets (just in case): for (int i = 0; i < standbyWallets.length; i++) { final wallet = standbyWallets[i]; await wallet.stopSync(isBackgroundSync: true); await wallet.close(); } } catch (e) { printV("error stopping sync: $e"); } // stop the service itself: service.invoke("serviceState", {"state": "NOT_RUNNING"}); await service.stopSelf(); }); service.on("status").listen((event) async { printV(event); }); void setForeground() { bgSyncStarted = false; _syncTimer?.cancel(); setNotificationStandby(flutterLocalNotificationsPlugin); } service.on("setForeground").listen((event) async { setForeground(); service.invoke("serviceState", {"state": "FOREGROUND"}); }); void setReady() { setNotificationReady(flutterLocalNotificationsPlugin); } service.on("setReady").listen((event) async { setReady(); service.invoke("serviceState", {"state": "READY"}); }); service.on("foregroundPing").listen((event) async { fgCount = 0; }); // we have entered the background, start the sync: void setBackground() async { if (bgSyncStarted) { return; } bgSyncStarted = true; await Future.delayed(const Duration(seconds: DELAY_SECONDS_BEFORE_SYNC_START)); printV("STARTING SYNC FROM BG"); setNotificationStarting(flutterLocalNotificationsPlugin); try { await initializeAppConfigs(loadWallet: false); } catch (_) { // these errors still show up in logs which doesn't really make sense to me } printV("INITIALIZED APP CONFIGS"); // final currentWallet = getIt.get().wallet; // // don't start syncing immediately: // await currentWallet?.stopSync(); final walletLoadingService = getIt.get(); final settingsStore = getIt.get(); final walletListViewModel = getIt.get(); // get all Monero / Wownero wallets and add them final List moneroWallets = walletListViewModel.wallets .where((element) => [WalletType.monero, WalletType.wownero].contains(element.type)) .toList(); printV("LOADING MONERO WALLETS"); for (int i = 0; i < moneroWallets.length; i++) { final wallet = await walletLoadingService.load(moneroWallets[i].type, moneroWallets[i].name); // stop regular sync process if it's been started // await wallet.stopSync(isBackgroundSync: false); syncingWallets.add(wallet); } printV("MONERO WALLETS LOADED"); // get all litecoin wallets and add them: final List litecoinWallets = walletListViewModel.wallets .where((element) => element.type == WalletType.litecoin) .toList(); // we only need to sync the first litecoin wallet since they share the same collection of blocks if (litecoinWallets.isNotEmpty) { try { final firstWallet = litecoinWallets.first; final wallet = await walletLoadingService.load(firstWallet.type, firstWallet.name); await wallet.stopSync(); if (bitcoin!.getMwebEnabled(wallet)) { syncingWallets.add(wallet); } } catch (e) { // couldn't connect to mwebd (most likely) printV("error syncing litecoin wallet: $e"); } } // get all bitcoin wallets and add them: final List bitcoinWallets = walletListViewModel.wallets.where((element) => element.type == WalletType.bitcoin).toList(); for (int i = 0; i < bitcoinWallets.length; i++) { try { final wallet = await walletLoadingService.load(bitcoinWallets[i].type, bitcoinWallets[i].name); var node = settingsStore.getCurrentNode(WalletType.bitcoin); await wallet.connectToNode(node: node); bool nodeSupportsSP = await (wallet as ElectrumWallet).getNodeSupportsSilentPayments(); if (!nodeSupportsSP) { // printV("Configured node does not support silent payments, skipping wallet"); // setWalletNotification( // flutterLocalNotificationsPlugin, // title: initialNotificationTitle, // content: spNodeNotificationMessage, // walletNum: syncingWallets.length + 1, // ); // spSupported = false; // continue; node = Node(uri: "electrs.cakewallet.com:50001"); await wallet.connectToNode(node: node); } await wallet.stopSync(); syncingWallets.add(wallet); } catch (e) { printV("error syncing bitcoin wallet_$i: $e"); } } printV("STARTING SYNC TIMER"); int syncedTicks = 0; _syncTimer?.cancel(); _syncTimer = Timer.periodic(const Duration(milliseconds: 2000), (timer) async { for (int i = 0; i < syncingWallets.length; i++) { final wallet = syncingWallets[i]; final syncStatus = wallet.syncStatus; final progress = wallet.syncStatus.progress(); final progressPercent = (progress * 100).toStringAsPrecision(5) + "%"; bool shouldSync = i == 0; String title = "${walletTypeToCryptoCurrency(wallet.type).title} - ${wallet.name}"; late String content; if (shouldSync) { if (syncStatus is NotConnectedSyncStatus) { printV("${wallet.name} NOT CONNECTED"); final node = settingsStore.getCurrentNode(wallet.type); await wallet.connectToNode(node: node); wallet.startSync(isBackgroundSync: true); printV("STARTED SYNC"); } if (progress > 0.999 || syncStatus is SyncedSyncStatus) { syncedTicks++; if (syncedTicks > 5) { syncedTicks = 0; printV("WALLET $i SYNCED"); try { await wallet.stopSync(isBackgroundSync: true); } catch (e) { printV("error stopping sync: $e"); } // pop the first wallet from the list standbyWallets.add(syncingWallets.removeAt(i)); flutterLocalNotificationsPlugin.cancelAll(); // if all wallets are synced, show a one time notification saying so: if (syncingWallets.isEmpty) { setNotificationWalletsSynced(flutterLocalNotificationsPlugin); } continue; } } else { syncedTicks = 0; } if (syncStatus is SyncingSyncStatus) { final blocksLeft = syncStatus.blocksLeft; content = "$blocksLeft Blocks Left"; } else if (syncStatus is SyncedSyncStatus) { content = "Synced"; } else if (syncStatus is SyncedTipSyncStatus) { final tip = syncStatus.tip; content = "Scanned Tip: $tip"; } else if (syncStatus is NotConnectedSyncStatus) { content = "Still Not Connected"; } else if (syncStatus is AttemptingSyncStatus) { content = "Attempting Sync"; } else if (syncStatus is StartingScanSyncStatus) { content = "Starting Scan"; } else if (syncStatus is SyncronizingSyncStatus) { content = "Syncronizing"; } else if (syncStatus is FailedSyncStatus) { content = "Failed Sync"; } else if (syncStatus is ConnectingSyncStatus) { content = "Connecting"; } else { // throw Exception("sync type not covered"); content = "Unknown Sync Status ${syncStatus.runtimeType}"; } if (syncedTicks > 0) { content += " - Finishing up..."; } } else { if (syncStatus is! NotConnectedSyncStatus) { wallet.stopSync(isBackgroundSync: true); } if (progress < SYNC_THRESHOLD) { content = "$progressPercent - Waiting in sync queue"; } else { content = "$progressPercent - This shouldn't happen, wallet is > SYNC_THRESHOLD"; } } // content += " - ${DateFormat("hh:mm:ss").format(DateTime.now())}"; if (i == 0) { setWalletNotification( flutterLocalNotificationsPlugin, title: title, content: content, walletNum: i, ); } } // for (int i = 0; i < standbyWallets.length; i++) { // int notificationIndex = syncingWallets.length + i + 1; // final wallet = standbyWallets[i]; // final title = "${walletTypeToCryptoCurrency(wallet.type).title} - ${wallet.name}"; // String content = "Synced - on standby until next queue refresh"; // setWalletNotification( // flutterLocalNotificationsPlugin, // title: title, // content: content, // walletNum: notificationIndex, // ); // } }); _queueTimer?.cancel(); // add a timer that checks all wallets and adds them to the queue if they are less than SYNC_THRESHOLD synced: _queueTimer = Timer.periodic(REFRESH_QUEUE_DURATION, (timer) async { final batteryState = await Battery().batteryState; bool onBattery = batteryState == BatteryState.connectedNotCharging || batteryState == BatteryState.discharging; ConnectivityResult connectivityResult = await Connectivity().checkConnectivity(); bool onData = connectivityResult == ConnectivityResult.mobile; if (onBattery && !syncOnBattery) { return; } if (onData && !syncOnData) { return; } // don't refresh the queue until we've finished syncing all wallets: if (syncingWallets.isNotEmpty) { return; } for (int i = 0; i < standbyWallets.length; i++) { final wallet = standbyWallets[i]; final syncStatus = wallet.syncStatus; // connect to the node if we haven't already: if (syncStatus is NotConnectedSyncStatus) { final node = settingsStore.getCurrentNode(wallet.type); await wallet.connectToNode(node: node); await wallet.startSync(isBackgroundSync: true); } // wait a while before checking progress: await Future.delayed(const Duration(seconds: 20)); if (syncStatus.progress() < SYNC_THRESHOLD) { syncingWallets.add(standbyWallets.removeAt(i)); } } }); // setup a watch dog to restart the wallet sync process if it appears to get stuck: List lastFewProgresses = []; List stuckWallets = []; _stuckSyncTimer?.cancel(); _stuckSyncTimer = Timer.periodic(const Duration(seconds: 10), (timer) async { if (syncingWallets.isEmpty) return; final wallet = syncingWallets.first; final syncStatus = wallet.syncStatus; if (syncStatus is! SyncingSyncStatus) return; lastFewProgresses.add(syncStatus.progress()); if (lastFewProgresses.length < 10) return; // limit list size to 10: while (lastFewProgresses.length > 10) { lastFewProgresses.removeAt(0); } // if the progress is the same over the last 100 seconds, restart the sync: if (lastFewProgresses.every((p) => p == lastFewProgresses.first)) { printV("syncing appears to be stuck, restarting..."); try { stuckWallets.add(wallet.name); await wallet.stopSync(isBackgroundSync: true); } catch (e) { printV("error restarting sync: $e"); } // if this wallet has been stuck more than twice, don't restart it, instead, add it to the standby list and try again on next queue refresh: // check if stuckWallets contains wallet.name more than 2 times: if (stuckWallets.where((name) => name == wallet.name).length > 2) { printV("wallet ${wallet.name} has been stuck more than 2 times, adding to standby list"); standbyWallets.add(syncingWallets.removeAt(0)); stuckWallets = []; return; } wallet.startSync(isBackgroundSync: true); } }); } service.on("setBackground").listen((event) async { setBackground(); }); // this is a backup timer to trigger in case the user fully closes the app, so that we still // start the background sync process: // annoyingly foreground code still runs in the background on android for some time, so we still use the original method // to detect if we are in the background since it's much faster _bgTimer = Timer.periodic(const Duration(seconds: 1), (timer) async { fgCount++; // we haven't been pinged in a while, so we are likely in the background: if (fgCount == 4) { setReady(); return; } if (fgCount > 10) { fgCount = 0; setBackground(); service.invoke("serviceState", {"state": "BACKGROUND"}); _bgTimer?.cancel(); } }); } @pragma('vm:entry-point') Future onIosBackground(ServiceInstance service) async { WidgetsFlutterBinding.ensureInitialized(); DartPluginRegistrant.ensureInitialized(); return true; } Future initializeService(FlutterBackgroundService bgService, bool useNotifications) async { if (useNotifications) { FlutterLocalNotificationsPlugin flutterLocalNotificationsPlugin = FlutterLocalNotificationsPlugin(); if (Platform.isIOS || Platform.isAndroid) { await flutterLocalNotificationsPlugin.initialize( const InitializationSettings( iOS: DarwinInitializationSettings(), android: AndroidInitializationSettings('ic_bg_service_small'), ), ); } for (int i = 0; i < 10; i++) { AndroidNotificationChannel channel = AndroidNotificationChannel( "${notificationChannelId}_$i", "${notificationChannelName}_$i", description: notificationChannelDescription, importance: Importance.min, playSound: false, showBadge: false, enableVibration: false, enableLights: false, ); await flutterLocalNotificationsPlugin .resolvePlatformSpecificImplementation() ?.createNotificationChannel(channel); } setNotificationStandby(flutterLocalNotificationsPlugin); } // notify the service that we are in the foreground: bgService.invoke("setForeground"); try { bool isServiceRunning = await bgService.isRunning(); if (isServiceRunning) { printV("Service is ALREADY running!"); return; } } catch (_) {} printV("INITIALIZING SERVICE"); await bgService.configure( androidConfiguration: AndroidConfiguration( onStart: onStart, autoStart: true, isForegroundMode: true, notificationChannelId: notificationChannelId, initialNotificationTitle: initialNotificationTitle, initialNotificationContent: standbyMessage, foregroundServiceNotificationId: notificationId, foregroundServiceTypes: [AndroidForegroundType.dataSync], ), iosConfiguration: IosConfiguration( autoStart: true, onForeground: onStart, onBackground: onIosBackground, ), ); } class BackgroundTasks { FlutterBackgroundService bgService = FlutterBackgroundService(); FlutterLocalNotificationsPlugin flutterLocalNotificationsPlugin = FlutterLocalNotificationsPlugin(); Timer? _pingTimer; String serviceState = "NOT_RUNNING"; void serviceBackground() { bgService.invoke("setBackground"); _pingTimer?.cancel(); } void foregroundPing() { bgService.invoke("foregroundPing"); } Future serviceForeground() async { if (serviceState == "FOREGROUND") { return; } final settingsStore = getIt.get(); bool showNotifications = settingsStore.showSyncNotification; bgService.invoke("stopService"); await Future.delayed(const Duration(seconds: 2)); initializeService(bgService, showNotifications); } void serviceReady() { final settingsStore = getIt.get(); bool showNotifications = settingsStore.showSyncNotification; if (showNotifications) { bgService.invoke("setReady"); } } void registerBackgroundService() async { printV("REGISTER BACKGROUND SERVICE"); try { final settingsStore = getIt.get(); final walletListViewModel = getIt.get(); bool hasMonero = walletListViewModel.wallets.any((element) => element.type == WalletType.monero); bool hasLitecoin = walletListViewModel.wallets.any((element) => element.type == WalletType.litecoin); bool hasBitcoin = walletListViewModel.wallets.any((element) => element.type == WalletType.bitcoin); if (!settingsStore.silentPaymentsAlwaysScan) { hasBitcoin = false; } /// if its not android nor ios, or the user has no monero wallets; exit if (!DeviceInfo.instance.isMobile || (!hasMonero && !hasLitecoin && !hasBitcoin)) { return; } final SyncMode syncMode = settingsStore.currentSyncMode; final bool useNotifications = settingsStore.showSyncNotification; final bool syncEnabled = settingsStore.backgroundSyncEnabled; syncOnBattery = settingsStore.backgroundSyncOnBattery; syncOnData = settingsStore.backgroundSyncOnData; if (useNotifications) { flutterLocalNotificationsPlugin .resolvePlatformSpecificImplementation() ?.requestNotificationsPermission(); } bgService.invoke("stopService"); if (!syncEnabled || !FeatureFlag.isBackgroundSyncEnabled) { return; } REFRESH_QUEUE_DURATION = syncMode.frequency; _pingTimer?.cancel(); _pingTimer = Timer.periodic(const Duration(seconds: 1), (timer) async { getIt.get().foregroundPing(); }); bgService.on("serviceState").listen((event) { serviceState = event?["state"] as String; }); await initializeService(bgService, useNotifications); } catch (error, stackTrace) { printV(error); printV(stackTrace); } } }