mirror of
				https://github.com/immich-app/immich.git
				synced 2025-10-26 08:12:33 -04:00 
			
		
		
		
	* refactor(mobile): DB repository for asset, backup, sync service * review feedback * fix bug found by Alex --------- Co-authored-by: Alex <alex.tran1502@gmail.com>
		
			
				
	
	
		
			778 lines
		
	
	
		
			26 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
			
		
		
	
	
			778 lines
		
	
	
		
			26 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
| import 'dart:io';
 | |
| 
 | |
| import 'package:cancellation_token_http/http.dart';
 | |
| import 'package:collection/collection.dart';
 | |
| import 'package:flutter/foundation.dart';
 | |
| import 'package:flutter/widgets.dart';
 | |
| import 'package:hooks_riverpod/hooks_riverpod.dart';
 | |
| import 'package:immich_mobile/entities/album.entity.dart';
 | |
| import 'package:immich_mobile/interfaces/album_media.interface.dart';
 | |
| import 'package:immich_mobile/interfaces/backup.interface.dart';
 | |
| import 'package:immich_mobile/interfaces/file_media.interface.dart';
 | |
| import 'package:immich_mobile/models/backup/available_album.model.dart';
 | |
| import 'package:immich_mobile/entities/backup_album.entity.dart';
 | |
| import 'package:immich_mobile/models/backup/backup_candidate.model.dart';
 | |
| import 'package:immich_mobile/models/backup/backup_state.model.dart';
 | |
| import 'package:immich_mobile/models/backup/current_upload_asset.model.dart';
 | |
| import 'package:immich_mobile/models/backup/error_upload_asset.model.dart';
 | |
| import 'package:immich_mobile/models/backup/success_upload_asset.model.dart';
 | |
| import 'package:immich_mobile/providers/backup/error_backup_list.provider.dart';
 | |
| import 'package:immich_mobile/repositories/album_media.repository.dart';
 | |
| import 'package:immich_mobile/repositories/backup.repository.dart';
 | |
| import 'package:immich_mobile/repositories/file_media.repository.dart';
 | |
| import 'package:immich_mobile/services/background.service.dart';
 | |
| import 'package:immich_mobile/services/backup.service.dart';
 | |
| import 'package:immich_mobile/models/authentication/authentication_state.model.dart';
 | |
| import 'package:immich_mobile/providers/authentication.provider.dart';
 | |
| import 'package:immich_mobile/providers/gallery_permission.provider.dart';
 | |
| import 'package:immich_mobile/models/server_info/server_disk_info.model.dart';
 | |
| import 'package:immich_mobile/entities/store.entity.dart';
 | |
| import 'package:immich_mobile/providers/app_life_cycle.provider.dart';
 | |
| import 'package:immich_mobile/providers/db.provider.dart';
 | |
| import 'package:immich_mobile/services/server_info.service.dart';
 | |
| import 'package:immich_mobile/utils/backup_progress.dart';
 | |
| import 'package:immich_mobile/utils/diff.dart';
 | |
| import 'package:isar/isar.dart';
 | |
| import 'package:logging/logging.dart';
 | |
| import 'package:permission_handler/permission_handler.dart';
 | |
| import 'package:photo_manager/photo_manager.dart' show PMProgressHandler;
 | |
| 
 | |
| class BackupNotifier extends StateNotifier<BackUpState> {
 | |
|   BackupNotifier(
 | |
|     this._backupService,
 | |
|     this._serverInfoService,
 | |
|     this._authState,
 | |
|     this._backgroundService,
 | |
|     this._galleryPermissionNotifier,
 | |
|     this._db,
 | |
|     this._albumMediaRepository,
 | |
|     this._fileMediaRepository,
 | |
|     this._backupRepository,
 | |
|     this.ref,
 | |
|   ) : super(
 | |
|           BackUpState(
 | |
|             backupProgress: BackUpProgressEnum.idle,
 | |
|             allAssetsInDatabase: const [],
 | |
|             progressInPercentage: 0,
 | |
|             progressInFileSize: "0 B / 0 B",
 | |
|             progressInFileSpeed: 0,
 | |
|             progressInFileSpeeds: const [],
 | |
|             progressInFileSpeedUpdateTime: DateTime.now(),
 | |
|             progressInFileSpeedUpdateSentBytes: 0,
 | |
|             cancelToken: CancellationToken(),
 | |
|             autoBackup: Store.get(StoreKey.autoBackup, false),
 | |
|             backgroundBackup: Store.get(StoreKey.backgroundBackup, false),
 | |
|             backupRequireWifi: Store.get(StoreKey.backupRequireWifi, true),
 | |
|             backupRequireCharging:
 | |
|                 Store.get(StoreKey.backupRequireCharging, false),
 | |
|             backupTriggerDelay: Store.get(StoreKey.backupTriggerDelay, 5000),
 | |
|             serverInfo: const ServerDiskInfo(
 | |
|               diskAvailable: "0",
 | |
|               diskSize: "0",
 | |
|               diskUse: "0",
 | |
|               diskUsagePercentage: 0,
 | |
|             ),
 | |
|             availableAlbums: const [],
 | |
|             selectedBackupAlbums: const {},
 | |
|             excludedBackupAlbums: const {},
 | |
|             allUniqueAssets: const {},
 | |
|             selectedAlbumsBackupAssetsIds: const {},
 | |
|             currentUploadAsset: CurrentUploadAsset(
 | |
|               id: '...',
 | |
|               fileCreatedAt: DateTime.parse('2020-10-04'),
 | |
|               fileName: '...',
 | |
|               fileType: '...',
 | |
|               fileSize: 0,
 | |
|               iCloudAsset: false,
 | |
|             ),
 | |
|             iCloudDownloadProgress: 0.0,
 | |
|           ),
 | |
|         );
 | |
| 
 | |
|   final log = Logger('BackupNotifier');
 | |
|   final BackupService _backupService;
 | |
|   final ServerInfoService _serverInfoService;
 | |
|   final AuthenticationState _authState;
 | |
|   final BackgroundService _backgroundService;
 | |
|   final GalleryPermissionNotifier _galleryPermissionNotifier;
 | |
|   final Isar _db;
 | |
|   final IAlbumMediaRepository _albumMediaRepository;
 | |
|   final IFileMediaRepository _fileMediaRepository;
 | |
|   final IBackupRepository _backupRepository;
 | |
|   final Ref ref;
 | |
| 
 | |
|   ///
 | |
|   /// UI INTERACTION
 | |
|   ///
 | |
|   /// Album selection
 | |
|   /// Due to the overlapping assets across multiple albums on the device
 | |
|   /// We have method to include and exclude albums
 | |
|   /// The total unique assets will be used for backing mechanism
 | |
|   ///
 | |
|   void addAlbumForBackup(AvailableAlbum album) {
 | |
|     if (state.excludedBackupAlbums.contains(album)) {
 | |
|       removeExcludedAlbumForBackup(album);
 | |
|     }
 | |
| 
 | |
|     state = state
 | |
|         .copyWith(selectedBackupAlbums: {...state.selectedBackupAlbums, album});
 | |
|   }
 | |
| 
 | |
|   void addExcludedAlbumForBackup(AvailableAlbum album) {
 | |
|     if (state.selectedBackupAlbums.contains(album)) {
 | |
|       removeAlbumForBackup(album);
 | |
|     }
 | |
|     state = state
 | |
|         .copyWith(excludedBackupAlbums: {...state.excludedBackupAlbums, album});
 | |
|   }
 | |
| 
 | |
|   void removeAlbumForBackup(AvailableAlbum album) {
 | |
|     Set<AvailableAlbum> currentSelectedAlbums = state.selectedBackupAlbums;
 | |
| 
 | |
|     currentSelectedAlbums.removeWhere((a) => a == album);
 | |
| 
 | |
|     state = state.copyWith(selectedBackupAlbums: currentSelectedAlbums);
 | |
|   }
 | |
| 
 | |
|   void removeExcludedAlbumForBackup(AvailableAlbum album) {
 | |
|     Set<AvailableAlbum> currentExcludedAlbums = state.excludedBackupAlbums;
 | |
| 
 | |
|     currentExcludedAlbums.removeWhere((a) => a == album);
 | |
| 
 | |
|     state = state.copyWith(excludedBackupAlbums: currentExcludedAlbums);
 | |
|   }
 | |
| 
 | |
|   Future<void> backupAlbumSelectionDone() {
 | |
|     if (state.selectedBackupAlbums.isEmpty) {
 | |
|       // disable any backup
 | |
|       cancelBackup();
 | |
|       setAutoBackup(false);
 | |
|       configureBackgroundBackup(
 | |
|         enabled: false,
 | |
|         onError: (msg) {},
 | |
|         onBatteryInfo: () {},
 | |
|       );
 | |
|     }
 | |
|     return _updateBackupAssetCount();
 | |
|   }
 | |
| 
 | |
|   void setAutoBackup(bool enabled) {
 | |
|     Store.put(StoreKey.autoBackup, enabled);
 | |
|     state = state.copyWith(autoBackup: enabled);
 | |
|   }
 | |
| 
 | |
|   void configureBackgroundBackup({
 | |
|     bool? enabled,
 | |
|     bool? requireWifi,
 | |
|     bool? requireCharging,
 | |
|     int? triggerDelay,
 | |
|     required void Function(String msg) onError,
 | |
|     required void Function() onBatteryInfo,
 | |
|   }) async {
 | |
|     assert(
 | |
|       enabled != null ||
 | |
|           requireWifi != null ||
 | |
|           requireCharging != null ||
 | |
|           triggerDelay != null,
 | |
|     );
 | |
|     final bool wasEnabled = state.backgroundBackup;
 | |
|     final bool wasWifi = state.backupRequireWifi;
 | |
|     final bool wasCharging = state.backupRequireCharging;
 | |
|     final int oldTriggerDelay = state.backupTriggerDelay;
 | |
|     state = state.copyWith(
 | |
|       backgroundBackup: enabled,
 | |
|       backupRequireWifi: requireWifi,
 | |
|       backupRequireCharging: requireCharging,
 | |
|       backupTriggerDelay: triggerDelay,
 | |
|     );
 | |
| 
 | |
|     if (state.backgroundBackup) {
 | |
|       bool success = true;
 | |
|       if (!wasEnabled) {
 | |
|         if (!await _backgroundService.isIgnoringBatteryOptimizations()) {
 | |
|           onBatteryInfo();
 | |
|         }
 | |
|         success &= await _backgroundService.enableService(immediate: true);
 | |
|       }
 | |
|       success &= success &&
 | |
|           await _backgroundService.configureService(
 | |
|             requireUnmetered: state.backupRequireWifi,
 | |
|             requireCharging: state.backupRequireCharging,
 | |
|             triggerUpdateDelay: state.backupTriggerDelay,
 | |
|             triggerMaxDelay: state.backupTriggerDelay * 10,
 | |
|           );
 | |
|       if (success) {
 | |
|         await Store.put(StoreKey.backupRequireWifi, state.backupRequireWifi);
 | |
|         await Store.put(
 | |
|           StoreKey.backupRequireCharging,
 | |
|           state.backupRequireCharging,
 | |
|         );
 | |
|         await Store.put(StoreKey.backupTriggerDelay, state.backupTriggerDelay);
 | |
|         await Store.put(StoreKey.backgroundBackup, state.backgroundBackup);
 | |
|       } else {
 | |
|         state = state.copyWith(
 | |
|           backgroundBackup: wasEnabled,
 | |
|           backupRequireWifi: wasWifi,
 | |
|           backupRequireCharging: wasCharging,
 | |
|           backupTriggerDelay: oldTriggerDelay,
 | |
|         );
 | |
|         onError("backup_controller_page_background_configure_error");
 | |
|       }
 | |
|     } else {
 | |
|       final bool success = await _backgroundService.disableService();
 | |
|       if (!success) {
 | |
|         state = state.copyWith(backgroundBackup: wasEnabled);
 | |
|         onError("backup_controller_page_background_configure_error");
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   ///
 | |
|   /// Get all album on the device
 | |
|   /// Get all selected and excluded album from the user's persistent storage
 | |
|   /// If this is the first time performing backup - set the default selected album to be
 | |
|   /// the one that has all assets (`Recent` on Android, `Recents` on iOS)
 | |
|   ///
 | |
|   Future<void> _getBackupAlbumsInfo() async {
 | |
|     Stopwatch stopwatch = Stopwatch()..start();
 | |
|     // Get all albums on the device
 | |
|     List<AvailableAlbum> availableAlbums = [];
 | |
|     List<Album> albums = await _albumMediaRepository.getAll();
 | |
| 
 | |
|     // Map of id -> album for quick album lookup later on.
 | |
|     Map<String, Album> albumMap = {};
 | |
| 
 | |
|     log.info('Found ${albums.length} local albums');
 | |
| 
 | |
|     for (Album album in albums) {
 | |
|       AvailableAlbum availableAlbum = AvailableAlbum(
 | |
|         album: album,
 | |
|         assetCount: await ref
 | |
|             .read(albumMediaRepositoryProvider)
 | |
|             .getAssetCount(album.localId!),
 | |
|       );
 | |
| 
 | |
|       availableAlbums.add(availableAlbum);
 | |
| 
 | |
|       albumMap[album.localId!] = album;
 | |
|     }
 | |
|     state = state.copyWith(availableAlbums: availableAlbums);
 | |
| 
 | |
|     final List<BackupAlbum> excludedBackupAlbums =
 | |
|         await _backupRepository.getAllBySelection(BackupSelection.exclude);
 | |
|     final List<BackupAlbum> selectedBackupAlbums =
 | |
|         await _backupRepository.getAllBySelection(BackupSelection.select);
 | |
| 
 | |
|     final Set<AvailableAlbum> selectedAlbums = {};
 | |
|     for (final BackupAlbum ba in selectedBackupAlbums) {
 | |
|       final albumAsset = albumMap[ba.id];
 | |
| 
 | |
|       if (albumAsset != null) {
 | |
|         selectedAlbums.add(
 | |
|           AvailableAlbum(
 | |
|             album: albumAsset,
 | |
|             assetCount:
 | |
|                 await _albumMediaRepository.getAssetCount(albumAsset.localId!),
 | |
|             lastBackup: ba.lastBackup,
 | |
|           ),
 | |
|         );
 | |
|       } else {
 | |
|         log.severe('Selected album not found');
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     final Set<AvailableAlbum> excludedAlbums = {};
 | |
|     for (final BackupAlbum ba in excludedBackupAlbums) {
 | |
|       final albumAsset = albumMap[ba.id];
 | |
| 
 | |
|       if (albumAsset != null) {
 | |
|         excludedAlbums.add(
 | |
|           AvailableAlbum(
 | |
|             album: albumAsset,
 | |
|             assetCount: await ref
 | |
|                 .read(albumMediaRepositoryProvider)
 | |
|                 .getAssetCount(albumAsset.localId!),
 | |
|             lastBackup: ba.lastBackup,
 | |
|           ),
 | |
|         );
 | |
|       } else {
 | |
|         log.severe('Excluded album not found');
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     state = state.copyWith(
 | |
|       selectedBackupAlbums: selectedAlbums,
 | |
|       excludedBackupAlbums: excludedAlbums,
 | |
|     );
 | |
| 
 | |
|     log.info(
 | |
|       "_getBackupAlbumsInfo: Found ${availableAlbums.length} available albums",
 | |
|     );
 | |
|     debugPrint("_getBackupAlbumsInfo takes ${stopwatch.elapsedMilliseconds}ms");
 | |
|   }
 | |
| 
 | |
|   ///
 | |
|   /// From all the selected and albums assets
 | |
|   /// Find the assets that are not overlapping between the two sets
 | |
|   /// Those assets are unique and are used as the total assets
 | |
|   ///
 | |
|   Future<void> _updateBackupAssetCount() async {
 | |
|     // Save to persistent storage
 | |
|     await _updatePersistentAlbumsSelection();
 | |
| 
 | |
|     final duplicatedAssetIds = await _backupService.getDuplicatedAssetIds();
 | |
|     final Set<BackupCandidate> assetsFromSelectedAlbums = {};
 | |
|     final Set<BackupCandidate> assetsFromExcludedAlbums = {};
 | |
| 
 | |
|     for (final album in state.selectedBackupAlbums) {
 | |
|       final assetCount = await ref
 | |
|           .read(albumMediaRepositoryProvider)
 | |
|           .getAssetCount(album.album.localId!);
 | |
| 
 | |
|       if (assetCount == 0) {
 | |
|         continue;
 | |
|       }
 | |
| 
 | |
|       final assets = await ref
 | |
|           .read(albumMediaRepositoryProvider)
 | |
|           .getAssets(album.album.localId!);
 | |
| 
 | |
|       // Add album's name to the asset info
 | |
|       for (final asset in assets) {
 | |
|         List<String> albumNames = [album.name];
 | |
| 
 | |
|         final existingAsset = assetsFromSelectedAlbums.firstWhereOrNull(
 | |
|           (a) => a.asset.localId == asset.localId,
 | |
|         );
 | |
| 
 | |
|         if (existingAsset != null) {
 | |
|           albumNames.addAll(existingAsset.albumNames);
 | |
|           assetsFromSelectedAlbums.remove(existingAsset);
 | |
|         }
 | |
| 
 | |
|         assetsFromSelectedAlbums.add(
 | |
|           BackupCandidate(
 | |
|             asset: asset,
 | |
|             albumNames: albumNames,
 | |
|           ),
 | |
|         );
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     for (final album in state.excludedBackupAlbums) {
 | |
|       final assetCount = await ref
 | |
|           .read(albumMediaRepositoryProvider)
 | |
|           .getAssetCount(album.album.localId!);
 | |
| 
 | |
|       if (assetCount == 0) {
 | |
|         continue;
 | |
|       }
 | |
| 
 | |
|       final assets = await ref
 | |
|           .read(albumMediaRepositoryProvider)
 | |
|           .getAssets(album.album.localId!);
 | |
| 
 | |
|       for (final asset in assets) {
 | |
|         assetsFromExcludedAlbums.add(
 | |
|           BackupCandidate(asset: asset, albumNames: [album.name]),
 | |
|         );
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     final Set<BackupCandidate> allUniqueAssets =
 | |
|         assetsFromSelectedAlbums.difference(assetsFromExcludedAlbums);
 | |
| 
 | |
|     final allAssetsInDatabase = await _backupService.getDeviceBackupAsset();
 | |
| 
 | |
|     if (allAssetsInDatabase == null) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     // Find asset that were backup from selected albums
 | |
|     final Set<String> selectedAlbumsBackupAssets =
 | |
|         Set.from(allUniqueAssets.map((e) => e.asset.localId));
 | |
| 
 | |
|     selectedAlbumsBackupAssets
 | |
|         .removeWhere((assetId) => !allAssetsInDatabase.contains(assetId));
 | |
| 
 | |
|     // Remove duplicated asset from all unique assets
 | |
|     allUniqueAssets.removeWhere(
 | |
|       (candidate) => duplicatedAssetIds.contains(candidate.asset.localId),
 | |
|     );
 | |
| 
 | |
|     if (allUniqueAssets.isEmpty) {
 | |
|       log.info("No assets are selected for back up");
 | |
|       state = state.copyWith(
 | |
|         backupProgress: BackUpProgressEnum.idle,
 | |
|         allAssetsInDatabase: allAssetsInDatabase,
 | |
|         allUniqueAssets: {},
 | |
|         selectedAlbumsBackupAssetsIds: selectedAlbumsBackupAssets,
 | |
|       );
 | |
|     } else {
 | |
|       state = state.copyWith(
 | |
|         allAssetsInDatabase: allAssetsInDatabase,
 | |
|         allUniqueAssets: allUniqueAssets,
 | |
|         selectedAlbumsBackupAssetsIds: selectedAlbumsBackupAssets,
 | |
|       );
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   /// Get all necessary information for calculating the available albums,
 | |
|   /// which albums are selected or excluded
 | |
|   /// and then update the UI according to those information
 | |
|   Future<void> getBackupInfo() async {
 | |
|     final isEnabled = await _backgroundService.isBackgroundBackupEnabled();
 | |
| 
 | |
|     state = state.copyWith(backgroundBackup: isEnabled);
 | |
|     if (isEnabled != Store.get(StoreKey.backgroundBackup, !isEnabled)) {
 | |
|       Store.put(StoreKey.backgroundBackup, isEnabled);
 | |
|     }
 | |
| 
 | |
|     if (state.backupProgress != BackUpProgressEnum.inBackground) {
 | |
|       await _getBackupAlbumsInfo();
 | |
|       await updateDiskInfo();
 | |
|       await _updateBackupAssetCount();
 | |
|     } else {
 | |
|       log.warning("cannot get backup info - background backup is in progress!");
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   /// Save user selection of selected albums and excluded albums to database
 | |
|   Future<void> _updatePersistentAlbumsSelection() {
 | |
|     final epoch = DateTime.fromMillisecondsSinceEpoch(0, isUtc: true);
 | |
|     final selected = state.selectedBackupAlbums.map(
 | |
|       (e) => BackupAlbum(e.id, e.lastBackup ?? epoch, BackupSelection.select),
 | |
|     );
 | |
|     final excluded = state.excludedBackupAlbums.map(
 | |
|       (e) => BackupAlbum(e.id, e.lastBackup ?? epoch, BackupSelection.exclude),
 | |
|     );
 | |
|     final backupAlbums = selected.followedBy(excluded).toList();
 | |
|     backupAlbums.sortBy((e) => e.id);
 | |
|     return _db.writeTxn(() async {
 | |
|       final dbAlbums = await _db.backupAlbums.where().sortById().findAll();
 | |
|       final List<int> toDelete = [];
 | |
|       final List<BackupAlbum> toUpsert = [];
 | |
|       // stores the most recent `lastBackup` per album but always keeps the `selection` the user just made
 | |
|       diffSortedListsSync(
 | |
|         dbAlbums,
 | |
|         backupAlbums,
 | |
|         compare: (BackupAlbum a, BackupAlbum b) => a.id.compareTo(b.id),
 | |
|         both: (BackupAlbum a, BackupAlbum b) {
 | |
|           b.lastBackup =
 | |
|               a.lastBackup.isAfter(b.lastBackup) ? a.lastBackup : b.lastBackup;
 | |
|           toUpsert.add(b);
 | |
|           return true;
 | |
|         },
 | |
|         onlyFirst: (BackupAlbum a) => toDelete.add(a.isarId),
 | |
|         onlySecond: (BackupAlbum b) => toUpsert.add(b),
 | |
|       );
 | |
|       await _db.backupAlbums.deleteAll(toDelete);
 | |
|       await _db.backupAlbums.putAll(toUpsert);
 | |
|     });
 | |
|   }
 | |
| 
 | |
|   /// Invoke backup process
 | |
|   Future<void> startBackupProcess() async {
 | |
|     debugPrint("Start backup process");
 | |
|     assert(state.backupProgress == BackUpProgressEnum.idle);
 | |
|     state = state.copyWith(backupProgress: BackUpProgressEnum.inProgress);
 | |
| 
 | |
|     await getBackupInfo();
 | |
| 
 | |
|     final hasPermission = _galleryPermissionNotifier.hasPermission;
 | |
|     if (hasPermission) {
 | |
|       await _fileMediaRepository.clearFileCache();
 | |
| 
 | |
|       if (state.allUniqueAssets.isEmpty) {
 | |
|         log.info("No Asset On Device - Abort Backup Process");
 | |
|         state = state.copyWith(backupProgress: BackUpProgressEnum.idle);
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       Set<BackupCandidate> assetsWillBeBackup = Set.from(state.allUniqueAssets);
 | |
|       // Remove item that has already been backed up
 | |
|       for (final assetId in state.allAssetsInDatabase) {
 | |
|         assetsWillBeBackup.removeWhere((e) => e.asset.localId == assetId);
 | |
|       }
 | |
| 
 | |
|       if (assetsWillBeBackup.isEmpty) {
 | |
|         state = state.copyWith(backupProgress: BackUpProgressEnum.idle);
 | |
|       }
 | |
| 
 | |
|       // Perform Backup
 | |
|       state = state.copyWith(cancelToken: CancellationToken());
 | |
| 
 | |
|       final pmProgressHandler = Platform.isIOS ? PMProgressHandler() : null;
 | |
| 
 | |
|       pmProgressHandler?.stream.listen((event) {
 | |
|         final double progress = event.progress;
 | |
|         state = state.copyWith(iCloudDownloadProgress: progress);
 | |
|       });
 | |
| 
 | |
|       await _backupService.backupAsset(
 | |
|         assetsWillBeBackup,
 | |
|         state.cancelToken,
 | |
|         pmProgressHandler: pmProgressHandler,
 | |
|         onSuccess: _onAssetUploaded,
 | |
|         onProgress: _onUploadProgress,
 | |
|         onCurrentAsset: _onSetCurrentBackupAsset,
 | |
|         onError: _onBackupError,
 | |
|       );
 | |
|       await notifyBackgroundServiceCanRun();
 | |
|     } else {
 | |
|       openAppSettings();
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   void setAvailableAlbums(availableAlbums) {
 | |
|     state = state.copyWith(
 | |
|       availableAlbums: availableAlbums,
 | |
|     );
 | |
|   }
 | |
| 
 | |
|   void _onBackupError(ErrorUploadAsset errorAssetInfo) {
 | |
|     ref.watch(errorBackupListProvider.notifier).add(errorAssetInfo);
 | |
|   }
 | |
| 
 | |
|   void _onSetCurrentBackupAsset(CurrentUploadAsset currentUploadAsset) {
 | |
|     state = state.copyWith(currentUploadAsset: currentUploadAsset);
 | |
|   }
 | |
| 
 | |
|   void cancelBackup() {
 | |
|     if (state.backupProgress != BackUpProgressEnum.inProgress) {
 | |
|       notifyBackgroundServiceCanRun();
 | |
|     }
 | |
|     state.cancelToken.cancel();
 | |
|     state = state.copyWith(
 | |
|       backupProgress: BackUpProgressEnum.idle,
 | |
|       progressInPercentage: 0.0,
 | |
|       progressInFileSize: "0 B / 0 B",
 | |
|       progressInFileSpeed: 0,
 | |
|       progressInFileSpeedUpdateTime: DateTime.now(),
 | |
|       progressInFileSpeedUpdateSentBytes: 0,
 | |
|     );
 | |
|   }
 | |
| 
 | |
|   void _onAssetUploaded(SuccessUploadAsset result) async {
 | |
|     if (result.isDuplicate) {
 | |
|       state = state.copyWith(
 | |
|         allUniqueAssets: state.allUniqueAssets
 | |
|             .where(
 | |
|               (candidate) =>
 | |
|                   candidate.asset.localId != result.candidate.asset.localId,
 | |
|             )
 | |
|             .toSet(),
 | |
|       );
 | |
|     } else {
 | |
|       state = state.copyWith(
 | |
|         selectedAlbumsBackupAssetsIds: {
 | |
|           ...state.selectedAlbumsBackupAssetsIds,
 | |
|           result.candidate.asset.localId!,
 | |
|         },
 | |
|         allAssetsInDatabase: [
 | |
|           ...state.allAssetsInDatabase,
 | |
|           result.candidate.asset.localId!,
 | |
|         ],
 | |
|       );
 | |
|     }
 | |
| 
 | |
|     if (state.allUniqueAssets.length -
 | |
|             state.selectedAlbumsBackupAssetsIds.length ==
 | |
|         0) {
 | |
|       final latestAssetBackup = state.allUniqueAssets
 | |
|           .map((candidate) => candidate.asset.fileModifiedAt)
 | |
|           .reduce(
 | |
|             (v, e) => e.isAfter(v) ? e : v,
 | |
|           );
 | |
|       state = state.copyWith(
 | |
|         selectedBackupAlbums: state.selectedBackupAlbums
 | |
|             .map((e) => e.copyWith(lastBackup: latestAssetBackup))
 | |
|             .toSet(),
 | |
|         excludedBackupAlbums: state.excludedBackupAlbums
 | |
|             .map((e) => e.copyWith(lastBackup: latestAssetBackup))
 | |
|             .toSet(),
 | |
|         backupProgress: BackUpProgressEnum.done,
 | |
|         progressInPercentage: 0.0,
 | |
|         progressInFileSize: "0 B / 0 B",
 | |
|         progressInFileSpeed: 0,
 | |
|         progressInFileSpeedUpdateTime: DateTime.now(),
 | |
|         progressInFileSpeedUpdateSentBytes: 0,
 | |
|       );
 | |
|       _updatePersistentAlbumsSelection();
 | |
|     }
 | |
| 
 | |
|     updateDiskInfo();
 | |
|   }
 | |
| 
 | |
|   void _onUploadProgress(int sent, int total) {
 | |
|     double lastUploadSpeed = state.progressInFileSpeed;
 | |
|     List<double> lastUploadSpeeds = state.progressInFileSpeeds.toList();
 | |
|     DateTime lastUpdateTime = state.progressInFileSpeedUpdateTime;
 | |
|     int lastSentBytes = state.progressInFileSpeedUpdateSentBytes;
 | |
| 
 | |
|     final now = DateTime.now();
 | |
|     final duration = now.difference(lastUpdateTime);
 | |
| 
 | |
|     // Keep the upload speed average span limited, to keep it somewhat relevant
 | |
|     if (lastUploadSpeeds.length > 10) {
 | |
|       lastUploadSpeeds.removeAt(0);
 | |
|     }
 | |
| 
 | |
|     if (duration.inSeconds > 0) {
 | |
|       lastUploadSpeeds.add(
 | |
|         ((sent - lastSentBytes) / duration.inSeconds).abs().roundToDouble(),
 | |
|       );
 | |
| 
 | |
|       lastUploadSpeed = lastUploadSpeeds.average.abs().roundToDouble();
 | |
|       lastUpdateTime = now;
 | |
|       lastSentBytes = sent;
 | |
|     }
 | |
| 
 | |
|     state = state.copyWith(
 | |
|       progressInPercentage: (sent.toDouble() / total.toDouble() * 100),
 | |
|       progressInFileSize: humanReadableFileBytesProgress(sent, total),
 | |
|       progressInFileSpeed: lastUploadSpeed,
 | |
|       progressInFileSpeeds: lastUploadSpeeds,
 | |
|       progressInFileSpeedUpdateTime: lastUpdateTime,
 | |
|       progressInFileSpeedUpdateSentBytes: lastSentBytes,
 | |
|     );
 | |
|   }
 | |
| 
 | |
|   Future<void> updateDiskInfo() async {
 | |
|     final diskInfo = await _serverInfoService.getDiskInfo();
 | |
| 
 | |
|     // Update server info
 | |
|     if (diskInfo != null) {
 | |
|       state = state.copyWith(
 | |
|         serverInfo: diskInfo,
 | |
|       );
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   Future<void> _resumeBackup() async {
 | |
|     // Check if user is login
 | |
|     final accessKey = Store.tryGet(StoreKey.accessToken);
 | |
| 
 | |
|     // User has been logged out return
 | |
|     if (accessKey == null || !_authState.isAuthenticated) {
 | |
|       log.info("[_resumeBackup] not authenticated - abort");
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     // Check if this device is enable backup by the user
 | |
|     if (state.autoBackup) {
 | |
|       // check if backup is already in process - then return
 | |
|       if (state.backupProgress == BackUpProgressEnum.inProgress) {
 | |
|         log.info("[_resumeBackup] Auto Backup is already in progress - abort");
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       if (state.backupProgress == BackUpProgressEnum.inBackground) {
 | |
|         log.info("[_resumeBackup] Background backup is running - abort");
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       if (state.backupProgress == BackUpProgressEnum.manualInProgress) {
 | |
|         log.info("[_resumeBackup] Manual upload is running - abort");
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       // Run backup
 | |
|       log.info("[_resumeBackup] Start back up");
 | |
|       await startBackupProcess();
 | |
|     }
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   Future<void> resumeBackup() async {
 | |
|     final List<BackupAlbum> selectedBackupAlbums = await _db.backupAlbums
 | |
|         .filter()
 | |
|         .selectionEqualTo(BackupSelection.select)
 | |
|         .findAll();
 | |
|     final List<BackupAlbum> excludedBackupAlbums = await _db.backupAlbums
 | |
|         .filter()
 | |
|         .selectionEqualTo(BackupSelection.exclude)
 | |
|         .findAll();
 | |
|     Set<AvailableAlbum> selectedAlbums = state.selectedBackupAlbums;
 | |
|     Set<AvailableAlbum> excludedAlbums = state.excludedBackupAlbums;
 | |
|     if (selectedAlbums.isNotEmpty) {
 | |
|       selectedAlbums = _updateAlbumsBackupTime(
 | |
|         selectedAlbums,
 | |
|         selectedBackupAlbums,
 | |
|       );
 | |
|     }
 | |
| 
 | |
|     if (excludedAlbums.isNotEmpty) {
 | |
|       excludedAlbums = _updateAlbumsBackupTime(
 | |
|         excludedAlbums,
 | |
|         excludedBackupAlbums,
 | |
|       );
 | |
|     }
 | |
|     final BackUpProgressEnum previous = state.backupProgress;
 | |
|     state = state.copyWith(
 | |
|       backupProgress: BackUpProgressEnum.inBackground,
 | |
|       selectedBackupAlbums: selectedAlbums,
 | |
|       excludedBackupAlbums: excludedAlbums,
 | |
|     );
 | |
|     // assumes the background service is currently running
 | |
|     // if true, waits until it has stopped to start the backup
 | |
|     final bool hasLock = await _backgroundService.acquireLock();
 | |
|     if (hasLock) {
 | |
|       state = state.copyWith(backupProgress: previous);
 | |
|     }
 | |
|     return _resumeBackup();
 | |
|   }
 | |
| 
 | |
|   Set<AvailableAlbum> _updateAlbumsBackupTime(
 | |
|     Set<AvailableAlbum> albums,
 | |
|     List<BackupAlbum> backupAlbums,
 | |
|   ) {
 | |
|     Set<AvailableAlbum> result = {};
 | |
|     for (BackupAlbum ba in backupAlbums) {
 | |
|       try {
 | |
|         AvailableAlbum a = albums.firstWhere((e) => e.id == ba.id);
 | |
|         result.add(a.copyWith(lastBackup: ba.lastBackup));
 | |
|       } on StateError {
 | |
|         log.severe(
 | |
|           "[_updateAlbumBackupTime] failed to find album in state",
 | |
|           "State Error",
 | |
|           StackTrace.current,
 | |
|         );
 | |
|       }
 | |
|     }
 | |
|     return result;
 | |
|   }
 | |
| 
 | |
|   Future<void> notifyBackgroundServiceCanRun() async {
 | |
|     const allowedStates = [
 | |
|       AppLifeCycleEnum.inactive,
 | |
|       AppLifeCycleEnum.paused,
 | |
|       AppLifeCycleEnum.detached,
 | |
|     ];
 | |
|     if (allowedStates.contains(ref.read(appStateProvider.notifier).state)) {
 | |
|       _backgroundService.releaseLock();
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   BackUpProgressEnum get backupProgress => state.backupProgress;
 | |
|   void updateBackupProgress(BackUpProgressEnum backupProgress) {
 | |
|     state = state.copyWith(backupProgress: backupProgress);
 | |
|   }
 | |
| }
 | |
| 
 | |
| final backupProvider =
 | |
|     StateNotifierProvider<BackupNotifier, BackUpState>((ref) {
 | |
|   return BackupNotifier(
 | |
|     ref.watch(backupServiceProvider),
 | |
|     ref.watch(serverInfoServiceProvider),
 | |
|     ref.watch(authenticationProvider),
 | |
|     ref.watch(backgroundServiceProvider),
 | |
|     ref.watch(galleryPermissionNotifier.notifier),
 | |
|     ref.watch(dbProvider),
 | |
|     ref.watch(albumMediaRepositoryProvider),
 | |
|     ref.watch(fileMediaRepositoryProvider),
 | |
|     ref.watch(backupRepositoryProvider),
 | |
|     ref,
 | |
|   );
 | |
| });
 |