import 'dart:async'; import 'dart:developer'; import 'dart:io'; import 'dart:isolate'; import 'dart:ui' show IsolateNameServer, PluginUtilities; import 'package:cancellation_token_http/http.dart'; import 'package:collection/collection.dart'; import 'package:easy_localization/easy_localization.dart'; import 'package:flutter/services.dart'; import 'package:flutter/widgets.dart'; import 'package:hive_flutter/hive_flutter.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:immich_mobile/constants/hive_box.dart'; import 'package:immich_mobile/modules/backup/background_service/localization.dart'; import 'package:immich_mobile/modules/backup/models/current_upload_asset.model.dart'; import 'package:immich_mobile/modules/backup/models/error_upload_asset.model.dart'; import 'package:immich_mobile/modules/backup/models/hive_backup_albums.model.dart'; import 'package:immich_mobile/modules/backup/services/backup.service.dart'; import 'package:immich_mobile/modules/login/models/hive_saved_login_info.model.dart'; import 'package:immich_mobile/modules/settings/services/app_settings.service.dart'; import 'package:immich_mobile/shared/services/api.service.dart'; import 'package:photo_manager/photo_manager.dart'; final backgroundServiceProvider = Provider( (ref) => BackgroundService(), ); /// Background backup service class BackgroundService { static const String _portNameLock = "immichLock"; BackgroundService(); static const MethodChannel _foregroundChannel = MethodChannel('immich/foregroundChannel'); static const MethodChannel _backgroundChannel = MethodChannel('immich/backgroundChannel'); bool _isForegroundInitialized = false; bool _isBackgroundInitialized = false; CancellationToken? _cancellationToken; bool _canceledBySystem = false; int _wantsLockTime = 0; bool _hasLock = false; SendPort? _waitingIsolate; ReceivePort? _rp; bool _errorGracePeriodExceeded = true; bool get isForegroundInitialized { return _isForegroundInitialized; } bool get isBackgroundInitialized { return _isBackgroundInitialized; } Future _initialize() async { final callback = PluginUtilities.getCallbackHandle(_nativeEntry)!; var result = await _foregroundChannel .invokeMethod('initialize', [callback.toRawHandle()]); _isForegroundInitialized = true; return result; } /// Ensures that the background service is enqueued if enabled in settings Future resumeServiceIfEnabled() async { return await isBackgroundBackupEnabled() && await startService(keepExisting: true); } /// Enqueues the background service Future startService({ bool immediate = false, bool keepExisting = false, bool requireUnmetered = true, bool requireCharging = false, }) async { if (!Platform.isAndroid) { return true; } try { if (!_isForegroundInitialized) { await _initialize(); } final String title = "backup_background_service_default_notification".tr(); final bool ok = await _foregroundChannel.invokeMethod( 'start', [immediate, keepExisting, requireUnmetered, requireCharging, title], ); return ok; } catch (error) { return false; } } /// Cancels the background service (if currently running) and removes it from work queue Future stopService() async { if (!Platform.isAndroid) { return true; } try { if (!_isForegroundInitialized) { await _initialize(); } final ok = await _foregroundChannel.invokeMethod('stop'); return ok; } catch (error) { return false; } } /// Returns `true` if the background service is enabled Future isBackgroundBackupEnabled() async { if (!Platform.isAndroid) { return false; } try { if (!_isForegroundInitialized) { await _initialize(); } return await _foregroundChannel.invokeMethod("isEnabled"); } catch (error) { return false; } } /// Opens an activity to let the user disable battery optimizations for Immich Future disableBatteryOptimizations() async { if (!Platform.isAndroid) { return true; } try { if (!_isForegroundInitialized) { await _initialize(); } final String message = "backup_background_service_disable_battery_optimizations".tr(); return await _foregroundChannel.invokeMethod( 'disableBatteryOptimizations', message, ); } catch (error) { return false; } } /// Updates the notification shown by the background service Future _updateNotification({ required String title, String? content, }) async { if (!Platform.isAndroid) { return true; } try { if (_isBackgroundInitialized) { return await _backgroundChannel .invokeMethod('updateNotification', [title, content]); } } catch (error) { debugPrint("[_updateNotification] failed to communicate with plugin"); } return Future.value(false); } /// Shows a new priority notification Future _showErrorNotification({ required String title, String? content, String? individualTag, }) async { if (!Platform.isAndroid) { return true; } try { if (_isBackgroundInitialized && _errorGracePeriodExceeded) { return await _backgroundChannel .invokeMethod('showError', [title, content, individualTag]); } } catch (error) { debugPrint("[_showErrorNotification] failed to communicate with plugin"); } return false; } Future _clearErrorNotifications() async { if (!Platform.isAndroid) { return true; } try { if (_isBackgroundInitialized) { return await _backgroundChannel.invokeMethod('clearErrorNotifications'); } } catch (error) { debugPrint( "[_clearErrorNotifications] failed to communicate with plugin"); } return false; } /// await to ensure this thread (foreground or background) has exclusive access Future acquireLock() async { if (!Platform.isAndroid) { return true; } final int lockTime = Timeline.now; _wantsLockTime = lockTime; final ReceivePort rp = ReceivePort(_portNameLock); _rp = rp; final SendPort sp = rp.sendPort; while (!IsolateNameServer.registerPortWithName(sp, _portNameLock)) { try { await _checkLockReleasedWithHeartbeat(lockTime); } catch (error) { return false; } if (_wantsLockTime != lockTime) { return false; } } _hasLock = true; rp.listen(_heartbeatListener); return true; } Future _checkLockReleasedWithHeartbeat(final int lockTime) async { SendPort? other = IsolateNameServer.lookupPortByName(_portNameLock); if (other != null) { final ReceivePort tempRp = ReceivePort(); final SendPort tempSp = tempRp.sendPort; final bs = tempRp.asBroadcastStream(); while (_wantsLockTime == lockTime) { other.send(tempSp); final dynamic answer = await bs.first .timeout(const Duration(seconds: 5), onTimeout: () => null); if (_wantsLockTime != lockTime) { break; } if (answer == null) { // other isolate failed to answer, assuming it exited without releasing the lock if (other == IsolateNameServer.lookupPortByName(_portNameLock)) { IsolateNameServer.removePortNameMapping(_portNameLock); } break; } else if (answer == true) { // other isolate released the lock break; } else if (answer == false) { // other isolate is still active } final dynamic isFinished = await bs.first .timeout(const Duration(seconds: 5), onTimeout: () => false); if (isFinished == true) { break; } } tempRp.close(); } } void _heartbeatListener(dynamic msg) { if (msg is SendPort) { _waitingIsolate = msg; msg.send(false); } } /// releases the exclusive access lock void releaseLock() { if (!Platform.isAndroid) { return; } _wantsLockTime = 0; if (_hasLock) { IsolateNameServer.removePortNameMapping(_portNameLock); _waitingIsolate?.send(true); _waitingIsolate = null; _hasLock = false; } _rp?.close(); _rp = null; } void _setupBackgroundCallHandler() { _backgroundChannel.setMethodCallHandler(_callHandler); _isBackgroundInitialized = true; _backgroundChannel.invokeMethod('initialized'); } Future _callHandler(MethodCall call) async { switch (call.method) { case "onAssetsChanged": final Future translationsLoaded = loadTranslations(); try { final bool hasAccess = await acquireLock(); if (!hasAccess) { debugPrint("[_callHandler] could acquire lock, exiting"); return false; } await translationsLoaded; final bool ok = await _onAssetsChanged(); if (ok) { Hive.box(backgroundBackupInfoBox).delete(backupFailedSince); } else if (Hive.box(backgroundBackupInfoBox).get(backupFailedSince) == null) { Hive.box(backgroundBackupInfoBox) .put(backupFailedSince, DateTime.now()); } return ok; } catch (error) { debugPrint(error.toString()); return false; } finally { await Hive.close(); releaseLock(); } case "systemStop": _canceledBySystem = true; _cancellationToken?.cancel(); return true; default: debugPrint("Unknown method ${call.method}"); return false; } } Future _onAssetsChanged() async { await Hive.initFlutter(); Hive.registerAdapter(HiveSavedLoginInfoAdapter()); Hive.registerAdapter(HiveBackupAlbumsAdapter()); await Hive.openBox(userInfoBox); await Hive.openBox(hiveLoginInfoBox); await Hive.openBox(userSettingInfoBox); await Hive.openBox(backgroundBackupInfoBox); ApiService apiService = ApiService(); apiService.setEndpoint(Hive.box(userInfoBox).get(serverEndpointKey)); apiService.setAccessToken(Hive.box(userInfoBox).get(accessTokenKey)); BackupService backupService = BackupService(apiService); final Box box = await Hive.openBox(hiveBackupInfoBox); final HiveBackupAlbums? backupAlbumInfo = box.get(backupInfoKey); if (backupAlbumInfo == null) { _clearErrorNotifications(); return true; } await PhotoManager.setIgnorePermissionCheck(true); _errorGracePeriodExceeded = _isErrorGracePeriodExceeded(); if (_canceledBySystem) { return false; } List toUpload = await backupService.buildUploadCandidates(backupAlbumInfo); try { toUpload = await backupService.removeAlreadyUploadedAssets(toUpload); } catch (e) { _showErrorNotification( title: "backup_background_service_error_title".tr(), content: "backup_background_service_connection_failed_message".tr(), ); return false; } if (_canceledBySystem) { return false; } if (toUpload.isEmpty) { _clearErrorNotifications(); return true; } _cancellationToken = CancellationToken(); final bool ok = await backupService.backupAsset( toUpload, _cancellationToken!, _onAssetUploaded, _onProgress, _onSetCurrentBackupAsset, _onBackupError, ); if (ok) { _clearErrorNotifications(); await box.put( backupInfoKey, backupAlbumInfo, ); } else { _showErrorNotification( title: "backup_background_service_error_title".tr(), content: "backup_background_service_backup_failed_message".tr(), ); } return ok; } void _onAssetUploaded(String deviceAssetId, String deviceId) { debugPrint("Uploaded $deviceAssetId from $deviceId"); } void _onProgress(int sent, int total) {} void _onBackupError(ErrorUploadAsset errorAssetInfo) { _showErrorNotification( title: "Upload failed", content: "backup_background_service_upload_failure_notification" .tr(args: [errorAssetInfo.fileName]), individualTag: errorAssetInfo.id, ); } void _onSetCurrentBackupAsset(CurrentUploadAsset currentUploadAsset) { _updateNotification( title: "backup_background_service_in_progress_notification".tr(), content: "backup_background_service_current_upload_notification" .tr(args: [currentUploadAsset.fileName]), ); } bool _isErrorGracePeriodExceeded() { final int value = AppSettingsService() .getSetting(AppSettingsEnum.uploadErrorNotificationGracePeriod); if (value == 0) { return true; } else if (value == 5) { return false; } final DateTime? failedSince = Hive.box(backgroundBackupInfoBox).get(backupFailedSince); if (failedSince == null) { return false; } final Duration duration = DateTime.now().difference(failedSince); if (value == 1) { return duration > const Duration(minutes: 30); } else if (value == 2) { return duration > const Duration(hours: 2); } else if (value == 3) { return duration > const Duration(hours: 8); } else if (value == 4) { return duration > const Duration(hours: 24); } assert(false, "Invalid value"); return true; } } /// entry point called by Kotlin/Java code; needs to be a top-level function void _nativeEntry() { WidgetsFlutterBinding.ensureInitialized(); BackgroundService backgroundService = BackgroundService(); backgroundService._setupBackgroundCallHandler(); }