cake_wallet/lib/entities/background_tasks.dart

480 lines
16 KiB
Dart
Raw Normal View History

2024-07-25 16:38:12 +00:00
import 'dart:async';
import 'dart:io';
2024-08-29 18:29:37 +00:00
import 'dart:ui';
2024-07-25 16:38:12 +00:00
import 'package:cake_wallet/bitcoin/bitcoin.dart';
import 'package:cake_wallet/core/wallet_loading_service.dart';
import 'package:cake_wallet/entities/preferences_key.dart';
2024-08-29 18:29:37 +00:00
import 'package:cake_wallet/store/app_store.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';
2024-07-25 17:12:36 +00:00
import 'package:cw_bitcoin/electrum_wallet.dart';
import 'package:cw_core/wallet_base.dart';
import 'package:cw_core/wallet_type.dart';
import 'package:flutter/foundation.dart';
2024-08-29 18:29:37 +00:00
import 'package:flutter/widgets.dart';
2024-08-28 18:54:47 +00:00
import 'package:flutter_background_service/flutter_background_service.dart';
2024-08-29 18:29:37 +00:00
import 'package:flutter_local_notifications/flutter_local_notifications.dart';
import 'package:shared_preferences/shared_preferences.dart';
import 'package:workmanager/workmanager.dart';
import 'package:cake_wallet/main.dart';
import 'package:cake_wallet/di.dart';
2024-07-27 19:18:15 +00:00
import 'package:http/http.dart' as http;
const moneroSyncTaskKey = "com.fotolockr.cakewallet.monero_sync_task";
2024-07-24 22:21:52 +00:00
const mwebSyncTaskKey = "com.fotolockr.cakewallet.mweb_sync_task";
@pragma('vm:entry-point')
void callbackDispatcher() {
2024-08-29 18:29:37 +00:00
// Workmanager().executeTask((task, inputData) async {
// try {
// switch (task) {
// case mwebSyncTaskKey:
// /// The work manager runs on a separate isolate from the main flutter isolate.
// /// thus we initialize app configs first; hive, getIt, etc...
// await initializeAppConfigs();
// final List<WalletListItem> ltcWallets = getIt
// .get<WalletListViewModel>()
// .wallets
// .where((element) => [WalletType.litecoin].contains(element.type))
// .toList();
// if (ltcWallets.isEmpty) {
// return Future.error("No ltc wallets found");
// }
// final walletLoadingService = getIt.get<WalletLoadingService>();
// var wallet =
// await walletLoadingService.load(ltcWallets.first.type, ltcWallets.first.name);
// print("STARTING SYNC FROM BG!!");
// final url = Uri.parse("https://webhook.site/a81e49d8-f5bd-4e57-8b1d-5d2c80c43f2a");
// final response = await http.get(url);
// if (response.statusCode == 200) {
// print("Background task starting: ${response.body}");
// } else {
// print("Failed to post webhook.site");
// }
// // await wallet.startSync();
// // RpcClient _stub = bitcoin!.getMwebStub();
// double syncStatus = 0.0;
// Timer? _syncTimer;
// // dynamic _stub = await bitcoin!.getMwebStub(wallet);
// _syncTimer = Timer.periodic(const Duration(milliseconds: 1500), (timer) async {
// // if (syncStatus is FailedSyncStatus) return;
// // TODO: use the proxy layer:
// // final height = await (wallet as ElectrumWallet).electrumClient.getCurrentBlockChainTip() ?? 0;
// final height = 2726590;
// // final height = 0;
// dynamic resp = await bitcoin!.getStatusRequest(wallet);
// int blockHeaderHeight = resp.blockHeaderHeight as int;
// int mwebHeaderHeight = resp.mwebHeaderHeight as int;
// int mwebUtxosHeight = resp.mwebUtxosHeight as int;
// print("blockHeaderHeight: $blockHeaderHeight");
// print("mwebHeaderHeight: $mwebHeaderHeight");
// print("mwebUtxosHeight: $mwebUtxosHeight");
// if (blockHeaderHeight < height) {
// syncStatus = blockHeaderHeight / height;
// } else if (mwebHeaderHeight < height) {
// syncStatus = mwebHeaderHeight / height;
// } else if (mwebUtxosHeight < height) {
// syncStatus = 0.999;
// } else {
// syncStatus = 1;
// }
// print("Sync status ${syncStatus}");
// });
// for (int i = 0;; i++) {
// await Future<void>.delayed(const Duration(seconds: 1));
// if (syncStatus == 1) {
// print("sync done!");
// break;
// }
// if (i > 600) {
// return Future.error("Synchronization Timed out");
// }
// }
// _syncTimer?.cancel();
// break;
// case moneroSyncTaskKey:
// /// The work manager runs on a separate isolate from the main flutter isolate.
// /// thus we initialize app configs first; hive, getIt, etc...
// await initializeAppConfigs();
// final walletLoadingService = getIt.get<WalletLoadingService>();
// final typeRaw = getIt.get<SharedPreferences>().getInt(PreferencesKey.currentWalletType);
// WalletBase? wallet;
// if (inputData!['sync_all'] as bool) {
// /// get all Monero wallets of the user and sync them
// final List<WalletListItem> moneroWallets = getIt
// .get<WalletListViewModel>()
// .wallets
// .where((element) => [WalletType.monero, WalletType.wownero].contains(element.type))
// .toList();
// for (int i = 0; i < moneroWallets.length; i++) {
// wallet =
// await walletLoadingService.load(moneroWallets[i].type, moneroWallets[i].name);
// final node = getIt.get<SettingsStore>().getCurrentNode(moneroWallets[i].type);
// await wallet.connectToNode(node: node);
// await wallet.startSync();
// }
// } else {
// /// if the user chose to sync only active wallet
// /// if the current wallet is monero; sync it only
// if (typeRaw == WalletType.monero.index || typeRaw == WalletType.wownero.index) {
// final name =
// getIt.get<SharedPreferences>().getString(PreferencesKey.currentWalletName);
// wallet = await walletLoadingService.load(WalletType.values[typeRaw!], name!);
// final node = getIt.get<SettingsStore>().getCurrentNode(WalletType.values[typeRaw]);
// await wallet.connectToNode(node: node);
// await wallet.startSync();
// }
// }
// if (wallet?.syncStatus.progress() == null) {
// return Future.error("No Monero/Wownero wallet found");
// }
// for (int i = 0;; i++) {
// await Future<void>.delayed(const Duration(seconds: 1));
// if (wallet?.syncStatus.progress() == 1.0) {
// break;
// }
// if (i > 600) {
// return Future.error("Synchronization Timed out");
// }
// }
// break;
// }
// return Future.value(true);
// } catch (error, stackTrace) {
// print(error);
// print(stackTrace);
// return Future.error(error);
// }
// });
}
2024-08-29 18:29:37 +00:00
@pragma('vm:entry-point')
Future<void> onStart(ServiceInstance service) async {
print("BACKGROUND SERVICE STARTED!");
2024-08-29 21:53:10 +00:00
// commented because the behavior appears to be bugged:
// DartPluginRegistrant.ensureInitialized();
2024-08-29 18:29:37 +00:00
final FlutterLocalNotificationsPlugin flutterLocalNotificationsPlugin =
FlutterLocalNotificationsPlugin();
2024-08-29 21:53:10 +00:00
// this will be used as notification channel id
// this will be used for notification id, So you can update your custom notification with this id.
2024-08-29 18:29:37 +00:00
const notificationChannelId = 'my_foreground';
2024-08-29 21:53:10 +00:00
const notificationChannelName = 'MY FOREGROUND SERVICE';
2024-08-29 18:29:37 +00:00
const notificationId = 888;
service.on('stopService').listen((event) {
service.stopSelf();
});
service.on('status').listen((event) async {
print(event);
});
bool bgSyncStarted = false;
2024-08-29 21:53:10 +00:00
service.on('foreground').listen((event) async {
bgSyncStarted = false;
});
2024-08-29 18:29:37 +00:00
service.on('startBgSync').listen((event) async {
if (bgSyncStarted) {
return;
}
2024-08-29 18:29:37 +00:00
bgSyncStarted = true;
print("STARTING SYNC FROM BG!!");
await initializeAppConfigs();
print("initialized app configs");
Timer.periodic(const Duration(milliseconds: 1500), (timer) {
final wallet = getIt.get<AppStore>().wallet;
2024-08-29 21:53:10 +00:00
final syncProgress = ((wallet?.syncStatus.progress() ?? 0) * 100).toStringAsPrecision(3);
2024-08-29 18:29:37 +00:00
flutterLocalNotificationsPlugin.show(
notificationId,
2024-08-29 21:53:10 +00:00
"${syncProgress}% Synced",
'Mweb background sync - ${DateTime.now()}',
2024-08-29 18:29:37 +00:00
const NotificationDetails(
android: AndroidNotificationDetails(
notificationChannelId,
2024-08-29 21:53:10 +00:00
notificationChannelName,
2024-08-29 18:29:37 +00:00
icon: 'ic_bg_service_small',
ongoing: true,
),
),
);
});
2024-08-29 21:53:10 +00:00
// print("stopping in ten seconds!");
// Timer(const Duration(seconds: 10), () {
// // stop the service after 10 seconds
// print("stopping now!");
// service.stopSelf();
// });
});
2024-08-29 18:29:37 +00:00
// final List<WalletListItem> ltcWallets = getIt
// .get<WalletListViewModel>()
// .wallets
// .where((element) => [WalletType.litecoin].contains(element.type))
// .toList();
// if (ltcWallets.isEmpty) {
// return Future.error("No ltc wallets found");
// }
// final walletLoadingService = getIt.get<WalletLoadingService>();
// var wallet = await walletLoadingService.load(ltcWallets.first.type, ltcWallets.first.name);
// var wallet = getIt.get<AppStore>().wallet;
// if (wallet?.type != WalletType.litecoin) {
// return;
// }
// print("STARTING SYNC FROM BG!!");
// final url = Uri.parse("https://webhook.site/a81e49d8-f5bd-4e57-8b1d-5d2c80c43f2a");
// final response = await http.get(url);
// if (response.statusCode == 200) {
// print("Background task starting: ${response.body}");
// } else {
// print("Failed to post webhook.site");
// }
// Timer.periodic(Duration(milliseconds: 5000), (timer) async {
// print(wallet!.syncStatus);
// });
// await wallet.startSync();
// RpcClient _stub = bitcoin!.getMwebStub();
// double syncStatus = 0.0;
// Timer? _syncTimer;
// dynamic _stub = await bitcoin!.getMwebStub(wallet);
// _syncTimer = Timer.periodic(const Duration(milliseconds: 1500), (timer) async {
// // if (syncStatus is FailedSyncStatus) return;
// // TODO: use the proxy layer:
// // final height = await (wallet as ElectrumWallet).electrumClient.getCurrentBlockChainTip() ?? 0;
// final height = 2726590;
// // final height = 0;
// dynamic resp = await bitcoin!.getStatusRequest(wallet);
// int blockHeaderHeight = resp.blockHeaderHeight as int;
// int mwebHeaderHeight = resp.mwebHeaderHeight as int;
// int mwebUtxosHeight = resp.mwebUtxosHeight as int;
// print("blockHeaderHeight: $blockHeaderHeight");
// print("mwebHeaderHeight: $mwebHeaderHeight");
// print("mwebUtxosHeight: $mwebUtxosHeight");
// if (blockHeaderHeight < height) {
// syncStatus = blockHeaderHeight / height;
// } else if (mwebHeaderHeight < height) {
// syncStatus = mwebHeaderHeight / height;
// } else if (mwebUtxosHeight < height) {
// syncStatus = 0.999;
// } else {
// syncStatus = 1;
// }
// print("Sync status ${syncStatus}");
// flutterLocalNotificationsPlugin.show(
// notificationId,
// '${syncStatus}',
// 'Awesome ${DateTime.now()}',
// const NotificationDetails(
// android: AndroidNotificationDetails(
// notificationChannelId,
// 'MY FOREGROUND SERVICE',
// icon: 'ic_bg_service_small',
// ongoing: true,
// ),
// ),
// );
// });
}
2024-08-29 18:29:37 +00:00
@pragma('vm:entry-point')
Future<bool> onIosBackground(ServiceInstance service) async {
WidgetsFlutterBinding.ensureInitialized();
DartPluginRegistrant.ensureInitialized();
return true;
}
2024-08-28 18:54:47 +00:00
2024-08-29 18:29:37 +00:00
Future<void> initializeService(FlutterBackgroundService bgService) async {
const AndroidNotificationChannel channel = AndroidNotificationChannel(
'my_foreground', // id
'MY FOREGROUND SERVICE', // title
description: 'This channel is used for important notifications.', // description
importance: Importance.low, // importance must be at low or higher level
);
final FlutterLocalNotificationsPlugin flutterLocalNotificationsPlugin =
FlutterLocalNotificationsPlugin();
if (Platform.isIOS || Platform.isAndroid) {
await flutterLocalNotificationsPlugin.initialize(
const InitializationSettings(
iOS: DarwinInitializationSettings(),
android: AndroidInitializationSettings('ic_bg_service_small'),
),
);
}
await flutterLocalNotificationsPlugin
.resolvePlatformSpecificImplementation<AndroidFlutterLocalNotificationsPlugin>()
?.createNotificationChannel(channel);
flutterLocalNotificationsPlugin
.resolvePlatformSpecificImplementation<AndroidFlutterLocalNotificationsPlugin>()
?.requestNotificationsPermission();
2024-08-29 21:53:10 +00:00
const initialNotificationTitle = 'Cake Background Sync';
const initialNotificationContent = 'On standby - app is in the foreground';
const notificationId = 888;
const notificationChannelId = 'my_foreground';
const notificationChannelName = 'MY FOREGROUND SERVICE';
flutterLocalNotificationsPlugin.show(
notificationId,
initialNotificationTitle,
initialNotificationContent,
const NotificationDetails(
android: AndroidNotificationDetails(
notificationChannelId,
notificationChannelName,
icon: 'ic_bg_service_small',
ongoing: true,
),
),
);
2024-08-29 21:58:07 +00:00
try {
bool isServiceRunning = await bgService.isRunning();
if (isServiceRunning) {
print("Service is ALREADY running!");
return;
}
} catch (_) {}
2024-08-29 18:29:37 +00:00
await bgService.configure(
2024-08-28 18:54:47 +00:00
androidConfiguration: AndroidConfiguration(
onStart: onStart,
2024-08-29 18:29:37 +00:00
autoStart: true,
isForegroundMode: true,
2024-08-29 21:53:10 +00:00
notificationChannelId: notificationChannelId,
initialNotificationTitle: initialNotificationTitle,
initialNotificationContent: initialNotificationContent,
foregroundServiceNotificationId: notificationId,
2024-08-29 18:29:37 +00:00
foregroundServiceTypes: [AndroidForegroundType.dataSync],
),
iosConfiguration: IosConfiguration(
autoStart: true,
onForeground: onStart,
onBackground: onIosBackground,
2024-08-28 18:54:47 +00:00
),
);
2024-08-29 18:29:37 +00:00
2024-08-29 21:53:10 +00:00
bgService.invoke("foreground");
2024-08-29 18:29:37 +00:00
Timer.periodic(const Duration(milliseconds: 1000), (timer) {
2024-08-29 21:53:10 +00:00
var wallet = getIt.get<AppStore>().wallet;
2024-08-29 18:29:37 +00:00
if (wallet?.syncStatus.toString() == "stopped") {
bgService.invoke("startBgSync");
2024-08-29 21:53:10 +00:00
timer.cancel();
2024-08-29 18:29:37 +00:00
}
});
2024-08-28 18:54:47 +00:00
}
class BackgroundTasks {
2024-08-29 18:29:37 +00:00
FlutterBackgroundService bgService = FlutterBackgroundService();
void registerSyncTask({bool changeExisting = false}) async {
try {
bool hasMonero = getIt
.get<WalletListViewModel>()
.wallets
.any((element) => element.type == WalletType.monero);
2024-07-24 22:21:52 +00:00
bool hasLitecoin = getIt
.get<WalletListViewModel>()
.wallets
.any((element) => element.type == WalletType.litecoin);
/// if its not android nor ios, or the user has no monero wallets; exit
2024-07-24 22:21:52 +00:00
if (!DeviceInfo.instance.isMobile || (!hasMonero && !hasLitecoin)) {
return;
}
final settingsStore = getIt.get<SettingsStore>();
final SyncMode syncMode = settingsStore.currentSyncMode;
final bool syncAll = settingsStore.currentSyncAll;
if (syncMode.type == SyncType.disabled || !FeatureFlag.isBackgroundSyncEnabled) {
2024-08-29 18:29:37 +00:00
bgService.invoke('stopService');
return;
}
2024-08-29 21:53:10 +00:00
// try {
// bool isServiceRunning = await bgService.isRunning();
// if (isServiceRunning) {
// return;
// // print("Service is ALREADY running!");
// // bgService.invoke('stopService');
// }
// } catch (_) {}
2024-08-29 18:29:37 +00:00
2024-08-29 21:53:10 +00:00
await initializeService(bgService);
} catch (error, stackTrace) {
print(error);
print(stackTrace);
}
}
}