2023-12-10 16:56:39 +01:00
|
|
|
import 'dart:io';
|
|
|
|
|
2022-06-18 14:36:58 +02:00
|
|
|
import 'package:cancellation_token_http/http.dart';
|
2023-03-18 15:55:11 +01:00
|
|
|
import 'package:collection/collection.dart';
|
2022-11-30 17:58:07 +01:00
|
|
|
import 'package:flutter/widgets.dart';
|
2022-05-06 14:22:23 +02:00
|
|
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
2024-05-01 04:36:40 +02:00
|
|
|
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_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';
|
2024-05-02 22:59:14 +02:00
|
|
|
import 'package:immich_mobile/providers/backup/error_backup_list.provider.dart';
|
|
|
|
import 'package:immich_mobile/services/background.service.dart';
|
|
|
|
import 'package:immich_mobile/services/backup.service.dart';
|
2024-05-01 04:36:40 +02:00
|
|
|
import 'package:immich_mobile/models/authentication/authentication_state.model.dart';
|
2024-05-02 22:59:14 +02:00
|
|
|
import 'package:immich_mobile/providers/authentication.provider.dart';
|
|
|
|
import 'package:immich_mobile/providers/gallery_permission.provider.dart';
|
2024-05-01 04:36:40 +02:00
|
|
|
import 'package:immich_mobile/models/server_info/server_disk_info.model.dart';
|
|
|
|
import 'package:immich_mobile/entities/store.entity.dart';
|
2024-05-02 22:59:14 +02:00
|
|
|
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';
|
feat(mobile): Adds file upload progress stats (#7760)
* feat(mobile): Adds file upload progress stats: current upload file size uploaded, current file size and formatted bytes per second upload speed. Closes #7379
* chore(mobile): Fix stan issues
* chore(mobile): Remove non-'en-US' translations, as I saw on another PR review (just looking around) that localisation is done via Localizely and this was the instruction (to only provide the en-US localisation).
* fix(mobile): Provide boundary checks to ensure overflow issues are accounted for on erroneous upload speed calculation, sometimes the numbers received back from the upload handler can be a bit wild.
* fix(mobile): Some heuristic bug fixing. Whilst thinking what could trigger overflow issues or 'zero' readouts, left over values from the previous file may do that. So adding the last upload sent bytes to the values to be reset may help! The time isn't necessary, as the period/cycle is inconsequential in this circumstance, well it should be anyway.
* fix(mobile): Actually, in combination to the last commit, some more heuristic bug fixing. I was thinking it would be advantageous not to reset the update time, as it would trigger a quicker first upload speed calculation. However, I realised that could also cause the calculation to be incorrect on the first cycle as the period wouldn't align. Not really sure if it would be a big deal, but I'm taking wild guesses in the dark here. Again, some purely heuristic debugging as I can't re-produce the underlying issue. This is mainly just ensuring that the state is fully reset and is a known state at the beginning of each file as a common strategy to reduce issues.
* refactor(mobile): Move the UI for the file progress to underneath the progress bar, it makes more sense there than in the file information table which contains only static information pertaining to the file itself. Switching to a monospace font to keep the UI from jumping around as the numbers change.
* refactor(mobile): In order to have the UI always present an 'active' upload speed (as per the discussion on PR #7760), this stores the 'upload speeds' (capped at the latest 10) in a list and calculates the current upload speed as the average over them. This way the UI can always display a 'constant' upload speed during uploading, instead of starting a fresh when each file starts uploading. Limiting it to the 10 latest keeps the average somewhat recent and ensures some level of sensible memory allocation.
2024-03-14 21:15:22 +01:00
|
|
|
import 'package:immich_mobile/utils/backup_progress.dart';
|
2023-03-18 15:55:11 +01:00
|
|
|
import 'package:immich_mobile/utils/diff.dart';
|
|
|
|
import 'package:isar/isar.dart';
|
2022-11-27 21:34:19 +01:00
|
|
|
import 'package:logging/logging.dart';
|
2023-02-28 17:22:18 +01:00
|
|
|
import 'package:permission_handler/permission_handler.dart';
|
2022-05-06 14:22:23 +02:00
|
|
|
import 'package:photo_manager/photo_manager.dart';
|
|
|
|
|
|
|
|
class BackupNotifier extends StateNotifier<BackUpState> {
|
2022-07-06 23:12:55 +02:00
|
|
|
BackupNotifier(
|
|
|
|
this._backupService,
|
|
|
|
this._serverInfoService,
|
|
|
|
this._authState,
|
2022-08-18 16:41:59 +02:00
|
|
|
this._backgroundService,
|
2023-02-28 17:22:18 +01:00
|
|
|
this._galleryPermissionNotifier,
|
2023-03-18 15:55:11 +01:00
|
|
|
this._db,
|
2022-07-06 23:12:55 +02:00
|
|
|
this.ref,
|
|
|
|
) : super(
|
2022-05-06 14:22:23 +02:00
|
|
|
BackUpState(
|
|
|
|
backupProgress: BackUpProgressEnum.idle,
|
2022-06-25 22:12:47 +02:00
|
|
|
allAssetsInDatabase: const [],
|
2022-05-06 14:22:23 +02:00
|
|
|
progressInPercentage: 0,
|
feat(mobile): Adds file upload progress stats (#7760)
* feat(mobile): Adds file upload progress stats: current upload file size uploaded, current file size and formatted bytes per second upload speed. Closes #7379
* chore(mobile): Fix stan issues
* chore(mobile): Remove non-'en-US' translations, as I saw on another PR review (just looking around) that localisation is done via Localizely and this was the instruction (to only provide the en-US localisation).
* fix(mobile): Provide boundary checks to ensure overflow issues are accounted for on erroneous upload speed calculation, sometimes the numbers received back from the upload handler can be a bit wild.
* fix(mobile): Some heuristic bug fixing. Whilst thinking what could trigger overflow issues or 'zero' readouts, left over values from the previous file may do that. So adding the last upload sent bytes to the values to be reset may help! The time isn't necessary, as the period/cycle is inconsequential in this circumstance, well it should be anyway.
* fix(mobile): Actually, in combination to the last commit, some more heuristic bug fixing. I was thinking it would be advantageous not to reset the update time, as it would trigger a quicker first upload speed calculation. However, I realised that could also cause the calculation to be incorrect on the first cycle as the period wouldn't align. Not really sure if it would be a big deal, but I'm taking wild guesses in the dark here. Again, some purely heuristic debugging as I can't re-produce the underlying issue. This is mainly just ensuring that the state is fully reset and is a known state at the beginning of each file as a common strategy to reduce issues.
* refactor(mobile): Move the UI for the file progress to underneath the progress bar, it makes more sense there than in the file information table which contains only static information pertaining to the file itself. Switching to a monospace font to keep the UI from jumping around as the numbers change.
* refactor(mobile): In order to have the UI always present an 'active' upload speed (as per the discussion on PR #7760), this stores the 'upload speeds' (capped at the latest 10) in a list and calculates the current upload speed as the average over them. This way the UI can always display a 'constant' upload speed during uploading, instead of starting a fresh when each file starts uploading. Limiting it to the 10 latest keeps the average somewhat recent and ensures some level of sensible memory allocation.
2024-03-14 21:15:22 +01:00
|
|
|
progressInFileSize: "0 B / 0 B",
|
|
|
|
progressInFileSpeed: 0,
|
|
|
|
progressInFileSpeeds: const [],
|
|
|
|
progressInFileSpeedUpdateTime: DateTime.now(),
|
|
|
|
progressInFileSpeedUpdateSentBytes: 0,
|
2022-06-18 14:36:58 +02:00
|
|
|
cancelToken: CancellationToken(),
|
2023-03-23 16:25:58 +01:00
|
|
|
autoBackup: Store.get(StoreKey.autoBackup, false),
|
2023-11-05 17:07:57 +01:00
|
|
|
backgroundBackup: Store.get(StoreKey.backgroundBackup, false),
|
2023-03-23 02:36:44 +01:00
|
|
|
backupRequireWifi: Store.get(StoreKey.backupRequireWifi, true),
|
|
|
|
backupRequireCharging:
|
|
|
|
Store.get(StoreKey.backupRequireCharging, false),
|
|
|
|
backupTriggerDelay: Store.get(StoreKey.backupTriggerDelay, 5000),
|
2023-10-22 22:15:34 +02:00
|
|
|
serverInfo: const ServerDiskInfo(
|
2022-05-06 14:22:23 +02:00
|
|
|
diskAvailable: "0",
|
|
|
|
diskSize: "0",
|
|
|
|
diskUse: "0",
|
2023-10-22 22:15:34 +02:00
|
|
|
diskUsagePercentage: 0,
|
2022-05-06 14:22:23 +02:00
|
|
|
),
|
|
|
|
availableAlbums: const [],
|
|
|
|
selectedBackupAlbums: const {},
|
2024-06-29 18:30:18 +02:00
|
|
|
excludedBackupAlbums: const {},
|
|
|
|
allUniqueAssets: const {},
|
2022-05-06 14:22:23 +02:00
|
|
|
selectedAlbumsBackupAssetsIds: const {},
|
2022-07-06 23:12:55 +02:00
|
|
|
currentUploadAsset: CurrentUploadAsset(
|
|
|
|
id: '...',
|
2023-02-19 17:44:53 +01:00
|
|
|
fileCreatedAt: DateTime.parse('2020-10-04'),
|
2022-07-06 23:12:55 +02:00
|
|
|
fileName: '...',
|
|
|
|
fileType: '...',
|
feat(mobile): Adds file upload progress stats (#7760)
* feat(mobile): Adds file upload progress stats: current upload file size uploaded, current file size and formatted bytes per second upload speed. Closes #7379
* chore(mobile): Fix stan issues
* chore(mobile): Remove non-'en-US' translations, as I saw on another PR review (just looking around) that localisation is done via Localizely and this was the instruction (to only provide the en-US localisation).
* fix(mobile): Provide boundary checks to ensure overflow issues are accounted for on erroneous upload speed calculation, sometimes the numbers received back from the upload handler can be a bit wild.
* fix(mobile): Some heuristic bug fixing. Whilst thinking what could trigger overflow issues or 'zero' readouts, left over values from the previous file may do that. So adding the last upload sent bytes to the values to be reset may help! The time isn't necessary, as the period/cycle is inconsequential in this circumstance, well it should be anyway.
* fix(mobile): Actually, in combination to the last commit, some more heuristic bug fixing. I was thinking it would be advantageous not to reset the update time, as it would trigger a quicker first upload speed calculation. However, I realised that could also cause the calculation to be incorrect on the first cycle as the period wouldn't align. Not really sure if it would be a big deal, but I'm taking wild guesses in the dark here. Again, some purely heuristic debugging as I can't re-produce the underlying issue. This is mainly just ensuring that the state is fully reset and is a known state at the beginning of each file as a common strategy to reduce issues.
* refactor(mobile): Move the UI for the file progress to underneath the progress bar, it makes more sense there than in the file information table which contains only static information pertaining to the file itself. Switching to a monospace font to keep the UI from jumping around as the numbers change.
* refactor(mobile): In order to have the UI always present an 'active' upload speed (as per the discussion on PR #7760), this stores the 'upload speeds' (capped at the latest 10) in a list and calculates the current upload speed as the average over them. This way the UI can always display a 'constant' upload speed during uploading, instead of starting a fresh when each file starts uploading. Limiting it to the 10 latest keeps the average somewhat recent and ensures some level of sensible memory allocation.
2024-03-14 21:15:22 +01:00
|
|
|
fileSize: 0,
|
2023-12-07 16:53:15 +01:00
|
|
|
iCloudAsset: false,
|
2022-07-06 23:12:55 +02:00
|
|
|
),
|
2023-12-07 16:53:15 +01:00
|
|
|
iCloudDownloadProgress: 0.0,
|
2022-05-06 14:22:23 +02:00
|
|
|
),
|
2023-02-21 04:43:39 +01:00
|
|
|
);
|
2022-05-06 14:22:23 +02:00
|
|
|
|
2022-11-27 21:34:19 +01:00
|
|
|
final log = Logger('BackupNotifier');
|
2022-06-25 20:46:51 +02:00
|
|
|
final BackupService _backupService;
|
|
|
|
final ServerInfoService _serverInfoService;
|
|
|
|
final AuthenticationState _authState;
|
2022-08-18 16:41:59 +02:00
|
|
|
final BackgroundService _backgroundService;
|
2023-02-28 17:22:18 +01:00
|
|
|
final GalleryPermissionNotifier _galleryPermissionNotifier;
|
2023-03-18 15:55:11 +01:00
|
|
|
final Isar _db;
|
2022-07-06 23:12:55 +02:00
|
|
|
final Ref ref;
|
2022-05-06 14:22:23 +02:00
|
|
|
|
|
|
|
///
|
|
|
|
/// 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
|
|
|
|
///
|
2022-08-18 16:41:59 +02:00
|
|
|
void addAlbumForBackup(AvailableAlbum album) {
|
2024-06-29 18:30:18 +02:00
|
|
|
if (state.excludedBackupAlbums.contains(album)) {
|
|
|
|
removeExcludedAlbumForBackup(album);
|
|
|
|
}
|
|
|
|
|
2022-06-22 07:23:35 +02:00
|
|
|
state = state
|
|
|
|
.copyWith(selectedBackupAlbums: {...state.selectedBackupAlbums, album});
|
2022-05-06 14:22:23 +02:00
|
|
|
}
|
|
|
|
|
2024-06-29 18:30:18 +02:00
|
|
|
void addExcludedAlbumForBackup(AvailableAlbum album) {
|
|
|
|
if (state.selectedBackupAlbums.contains(album)) {
|
|
|
|
removeAlbumForBackup(album);
|
|
|
|
}
|
|
|
|
state = state
|
|
|
|
.copyWith(excludedBackupAlbums: {...state.excludedBackupAlbums, album});
|
|
|
|
}
|
|
|
|
|
2022-08-18 16:41:59 +02:00
|
|
|
void removeAlbumForBackup(AvailableAlbum album) {
|
|
|
|
Set<AvailableAlbum> currentSelectedAlbums = state.selectedBackupAlbums;
|
2022-05-06 14:22:23 +02:00
|
|
|
|
|
|
|
currentSelectedAlbums.removeWhere((a) => a == album);
|
|
|
|
|
|
|
|
state = state.copyWith(selectedBackupAlbums: currentSelectedAlbums);
|
|
|
|
}
|
|
|
|
|
2024-06-29 18:30:18 +02:00
|
|
|
void removeExcludedAlbumForBackup(AvailableAlbum album) {
|
|
|
|
Set<AvailableAlbum> currentExcludedAlbums = state.excludedBackupAlbums;
|
|
|
|
|
|
|
|
currentExcludedAlbums.removeWhere((a) => a == album);
|
|
|
|
|
|
|
|
state = state.copyWith(excludedBackupAlbums: currentExcludedAlbums);
|
|
|
|
}
|
|
|
|
|
2023-11-14 21:30:27 +01:00
|
|
|
Future<void> backupAlbumSelectionDone() {
|
|
|
|
if (state.selectedBackupAlbums.isEmpty) {
|
|
|
|
// disable any backup
|
|
|
|
cancelBackup();
|
|
|
|
setAutoBackup(false);
|
|
|
|
configureBackgroundBackup(
|
|
|
|
enabled: false,
|
|
|
|
onError: (msg) {},
|
|
|
|
onBatteryInfo: () {},
|
|
|
|
);
|
|
|
|
}
|
|
|
|
return _updateBackupAssetCount();
|
2022-05-06 14:22:23 +02:00
|
|
|
}
|
|
|
|
|
2023-03-23 16:25:58 +01:00
|
|
|
void setAutoBackup(bool enabled) {
|
|
|
|
Store.put(StoreKey.autoBackup, enabled);
|
|
|
|
state = state.copyWith(autoBackup: enabled);
|
|
|
|
}
|
|
|
|
|
2022-08-18 16:41:59 +02:00
|
|
|
void configureBackgroundBackup({
|
|
|
|
bool? enabled,
|
|
|
|
bool? requireWifi,
|
|
|
|
bool? requireCharging,
|
2022-12-08 16:51:36 +01:00
|
|
|
int? triggerDelay,
|
2022-08-18 16:41:59 +02:00
|
|
|
required void Function(String msg) onError,
|
2022-08-31 15:08:40 +02:00
|
|
|
required void Function() onBatteryInfo,
|
2022-08-18 16:41:59 +02:00
|
|
|
}) async {
|
2022-12-08 16:51:36 +01:00
|
|
|
assert(
|
|
|
|
enabled != null ||
|
|
|
|
requireWifi != null ||
|
|
|
|
requireCharging != null ||
|
|
|
|
triggerDelay != null,
|
|
|
|
);
|
2023-02-21 04:43:39 +01:00
|
|
|
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,
|
|
|
|
);
|
2022-08-18 16:41:59 +02:00
|
|
|
|
2023-02-21 04:43:39 +01:00
|
|
|
if (state.backgroundBackup) {
|
|
|
|
bool success = true;
|
|
|
|
if (!wasEnabled) {
|
|
|
|
if (!await _backgroundService.isIgnoringBatteryOptimizations()) {
|
|
|
|
onBatteryInfo();
|
2022-08-18 16:41:59 +02:00
|
|
|
}
|
2023-02-21 04:43:39 +01:00
|
|
|
success &= await _backgroundService.enableService(immediate: true);
|
|
|
|
}
|
|
|
|
success &= success &&
|
|
|
|
await _backgroundService.configureService(
|
|
|
|
requireUnmetered: state.backupRequireWifi,
|
|
|
|
requireCharging: state.backupRequireCharging,
|
|
|
|
triggerUpdateDelay: state.backupTriggerDelay,
|
|
|
|
triggerMaxDelay: state.backupTriggerDelay * 10,
|
2022-08-18 16:41:59 +02:00
|
|
|
);
|
2023-02-21 04:43:39 +01:00
|
|
|
if (success) {
|
2023-03-23 02:36:44 +01:00
|
|
|
await Store.put(StoreKey.backupRequireWifi, state.backupRequireWifi);
|
|
|
|
await Store.put(
|
|
|
|
StoreKey.backupRequireCharging,
|
|
|
|
state.backupRequireCharging,
|
|
|
|
);
|
|
|
|
await Store.put(StoreKey.backupTriggerDelay, state.backupTriggerDelay);
|
2023-11-05 17:07:57 +01:00
|
|
|
await Store.put(StoreKey.backgroundBackup, state.backgroundBackup);
|
2022-08-18 16:41:59 +02:00
|
|
|
} else {
|
2023-02-21 04:43:39 +01:00
|
|
|
state = state.copyWith(
|
|
|
|
backgroundBackup: wasEnabled,
|
|
|
|
backupRequireWifi: wasWifi,
|
|
|
|
backupRequireCharging: wasCharging,
|
|
|
|
backupTriggerDelay: oldTriggerDelay,
|
|
|
|
);
|
|
|
|
onError("backup_controller_page_background_configure_error");
|
2022-08-18 16:41:59 +02:00
|
|
|
}
|
2023-02-21 04:43:39 +01:00
|
|
|
} else {
|
|
|
|
final bool success = await _backgroundService.disableService();
|
|
|
|
if (!success) {
|
|
|
|
state = state.copyWith(backgroundBackup: wasEnabled);
|
|
|
|
onError("backup_controller_page_background_configure_error");
|
|
|
|
}
|
|
|
|
}
|
2022-08-18 16:41:59 +02:00
|
|
|
}
|
|
|
|
|
2022-05-06 14:22:23 +02:00
|
|
|
///
|
|
|
|
/// 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
|
2022-11-30 17:58:07 +01:00
|
|
|
/// the one that has all assets (`Recent` on Android, `Recents` on iOS)
|
2022-05-06 14:22:23 +02:00
|
|
|
///
|
2022-06-25 22:12:47 +02:00
|
|
|
Future<void> _getBackupAlbumsInfo() async {
|
2022-11-30 17:58:07 +01:00
|
|
|
Stopwatch stopwatch = Stopwatch()..start();
|
2022-05-06 14:22:23 +02:00
|
|
|
// Get all albums on the device
|
|
|
|
List<AvailableAlbum> availableAlbums = [];
|
2022-06-22 07:23:35 +02:00
|
|
|
List<AssetPathEntity> albums = await PhotoManager.getAssetPathList(
|
2022-07-13 14:23:48 +02:00
|
|
|
hasAll: true,
|
|
|
|
type: RequestType.common,
|
|
|
|
);
|
2022-05-06 14:22:23 +02:00
|
|
|
|
2023-07-17 05:08:58 +02:00
|
|
|
// Map of id -> album for quick album lookup later on.
|
|
|
|
Map<String, AssetPathEntity> albumMap = {};
|
|
|
|
|
2022-11-30 17:58:07 +01:00
|
|
|
log.info('Found ${albums.length} local albums');
|
|
|
|
|
2022-05-06 14:22:23 +02:00
|
|
|
for (AssetPathEntity album in albums) {
|
|
|
|
AvailableAlbum availableAlbum = AvailableAlbum(albumEntity: album);
|
|
|
|
|
2024-03-19 14:40:14 +01:00
|
|
|
availableAlbums.add(availableAlbum);
|
2023-09-25 10:09:09 +02:00
|
|
|
|
2024-03-19 14:40:14 +01:00
|
|
|
albumMap[album.id] = album;
|
2022-05-06 14:22:23 +02:00
|
|
|
}
|
|
|
|
state = state.copyWith(availableAlbums: availableAlbums);
|
|
|
|
|
2024-06-29 18:30:18 +02:00
|
|
|
final List<BackupAlbum> excludedBackupAlbums =
|
|
|
|
await _backupService.excludedAlbumsQuery().findAll();
|
2023-03-18 15:55:11 +01:00
|
|
|
final List<BackupAlbum> selectedBackupAlbums =
|
|
|
|
await _backupService.selectedAlbumsQuery().findAll();
|
2022-05-06 14:22:23 +02:00
|
|
|
|
|
|
|
// Generate AssetPathEntity from id to add to local state
|
2023-07-17 05:08:58 +02:00
|
|
|
final Set<AvailableAlbum> selectedAlbums = {};
|
|
|
|
for (final BackupAlbum ba in selectedBackupAlbums) {
|
|
|
|
final albumAsset = albumMap[ba.id];
|
|
|
|
|
|
|
|
if (albumAsset != null) {
|
2022-08-18 16:41:59 +02:00
|
|
|
selectedAlbums.add(
|
2023-03-18 15:55:11 +01:00
|
|
|
AvailableAlbum(albumEntity: albumAsset, lastBackup: ba.lastBackup),
|
2022-07-13 14:23:48 +02:00
|
|
|
);
|
2023-07-17 05:08:58 +02:00
|
|
|
} else {
|
|
|
|
log.severe('Selected album not found');
|
2022-05-06 14:22:23 +02:00
|
|
|
}
|
2023-07-17 05:08:58 +02:00
|
|
|
}
|
|
|
|
|
2024-06-29 18:30:18 +02:00
|
|
|
final Set<AvailableAlbum> excludedAlbums = {};
|
|
|
|
for (final BackupAlbum ba in excludedBackupAlbums) {
|
|
|
|
final albumAsset = albumMap[ba.id];
|
|
|
|
|
|
|
|
if (albumAsset != null) {
|
|
|
|
excludedAlbums.add(
|
|
|
|
AvailableAlbum(albumEntity: albumAsset, lastBackup: ba.lastBackup),
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
log.severe('Excluded album not found');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-07-17 05:08:58 +02:00
|
|
|
state = state.copyWith(
|
|
|
|
selectedBackupAlbums: selectedAlbums,
|
2024-06-29 18:30:18 +02:00
|
|
|
excludedBackupAlbums: excludedAlbums,
|
2023-07-17 05:08:58 +02:00
|
|
|
);
|
|
|
|
|
2023-12-13 04:06:04 +01:00
|
|
|
log.info(
|
|
|
|
"_getBackupAlbumsInfo: Found ${availableAlbums.length} available albums",
|
|
|
|
);
|
2022-11-30 17:58:07 +01:00
|
|
|
debugPrint("_getBackupAlbumsInfo takes ${stopwatch.elapsedMilliseconds}ms");
|
2022-05-06 14:22:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
///
|
|
|
|
/// 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
|
|
|
|
///
|
2022-06-25 22:12:47 +02:00
|
|
|
Future<void> _updateBackupAssetCount() async {
|
2023-03-18 15:55:11 +01:00
|
|
|
final duplicatedAssetIds = await _backupService.getDuplicatedAssetIds();
|
2024-06-29 18:30:18 +02:00
|
|
|
final Set<AssetEntity> assetsFromSelectedAlbums = {};
|
|
|
|
final Set<AssetEntity> assetsFromExcludedAlbums = {};
|
2022-05-06 14:22:23 +02:00
|
|
|
|
2023-03-18 15:55:11 +01:00
|
|
|
for (final album in state.selectedBackupAlbums) {
|
2024-05-07 06:00:52 +02:00
|
|
|
final assetCount = await album.albumEntity.assetCountAsync;
|
|
|
|
|
|
|
|
if (assetCount == 0) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2023-03-18 15:55:11 +01:00
|
|
|
final assets = await album.albumEntity.getAssetListRange(
|
2022-09-16 23:46:23 +02:00
|
|
|
start: 0,
|
2024-05-07 06:00:52 +02:00
|
|
|
end: assetCount,
|
2022-09-16 23:46:23 +02:00
|
|
|
);
|
2024-06-29 18:30:18 +02:00
|
|
|
assetsFromSelectedAlbums.addAll(assets);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (final album in state.excludedBackupAlbums) {
|
|
|
|
final assetCount = await album.albumEntity.assetCountAsync;
|
|
|
|
|
|
|
|
if (assetCount == 0) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
final assets = await album.albumEntity.getAssetListRange(
|
|
|
|
start: 0,
|
|
|
|
end: assetCount,
|
|
|
|
);
|
|
|
|
assetsFromExcludedAlbums.addAll(assets);
|
2022-05-06 14:22:23 +02:00
|
|
|
}
|
|
|
|
|
2024-06-29 18:30:18 +02:00
|
|
|
final Set<AssetEntity> allUniqueAssets =
|
|
|
|
assetsFromSelectedAlbums.difference(assetsFromExcludedAlbums);
|
2023-03-18 15:55:11 +01:00
|
|
|
final allAssetsInDatabase = await _backupService.getDeviceBackupAsset();
|
2022-07-13 14:23:48 +02:00
|
|
|
|
|
|
|
if (allAssetsInDatabase == null) {
|
|
|
|
return;
|
|
|
|
}
|
2022-05-06 14:22:23 +02:00
|
|
|
|
|
|
|
// Find asset that were backup from selected albums
|
2023-03-18 15:55:11 +01:00
|
|
|
final Set<String> selectedAlbumsBackupAssets =
|
2024-06-29 18:30:18 +02:00
|
|
|
Set.from(allUniqueAssets.map((e) => e.id));
|
2022-10-25 16:51:03 +02:00
|
|
|
|
2022-06-22 07:23:35 +02:00
|
|
|
selectedAlbumsBackupAssets
|
2022-06-25 22:12:47 +02:00
|
|
|
.removeWhere((assetId) => !allAssetsInDatabase.contains(assetId));
|
2022-05-06 14:22:23 +02:00
|
|
|
|
2022-10-25 16:51:03 +02:00
|
|
|
// Remove duplicated asset from all unique assets
|
2024-06-29 18:30:18 +02:00
|
|
|
allUniqueAssets.removeWhere(
|
2022-10-25 16:51:03 +02:00
|
|
|
(asset) => duplicatedAssetIds.contains(asset.id),
|
|
|
|
);
|
|
|
|
|
2024-06-29 18:30:18 +02:00
|
|
|
if (allUniqueAssets.isEmpty) {
|
2023-12-12 15:58:13 +01:00
|
|
|
log.info("No assets are selected for back up");
|
2022-05-06 14:22:23 +02:00
|
|
|
state = state.copyWith(
|
|
|
|
backupProgress: BackUpProgressEnum.idle,
|
2022-06-25 22:12:47 +02:00
|
|
|
allAssetsInDatabase: allAssetsInDatabase,
|
2024-06-29 18:30:18 +02:00
|
|
|
allUniqueAssets: {},
|
2022-05-06 14:22:23 +02:00
|
|
|
selectedAlbumsBackupAssetsIds: selectedAlbumsBackupAssets,
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
state = state.copyWith(
|
2022-06-25 22:12:47 +02:00
|
|
|
allAssetsInDatabase: allAssetsInDatabase,
|
2024-06-29 18:30:18 +02:00
|
|
|
allUniqueAssets: allUniqueAssets,
|
2022-05-06 14:22:23 +02:00
|
|
|
selectedAlbumsBackupAssetsIds: selectedAlbumsBackupAssets,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Save to persistent storage
|
2023-03-18 15:55:11 +01:00
|
|
|
await _updatePersistentAlbumsSelection();
|
2022-05-06 14:22:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Get all necessary information for calculating the available albums,
|
|
|
|
/// which albums are selected or excluded
|
|
|
|
/// and then update the UI according to those information
|
2022-06-25 22:12:47 +02:00
|
|
|
Future<void> getBackupInfo() async {
|
2023-03-18 15:55:11 +01:00
|
|
|
final isEnabled = await _backgroundService.isBackgroundBackupEnabled();
|
2022-11-30 17:58:07 +01:00
|
|
|
|
2022-12-01 16:20:53 +01:00
|
|
|
state = state.copyWith(backgroundBackup: isEnabled);
|
2023-11-05 17:07:57 +01:00
|
|
|
if (isEnabled != Store.get(StoreKey.backgroundBackup, !isEnabled)) {
|
|
|
|
Store.put(StoreKey.backgroundBackup, isEnabled);
|
|
|
|
}
|
2022-11-30 17:58:07 +01:00
|
|
|
|
2022-12-01 16:20:53 +01:00
|
|
|
if (state.backupProgress != BackUpProgressEnum.inBackground) {
|
|
|
|
await _getBackupAlbumsInfo();
|
2024-05-22 11:25:55 +02:00
|
|
|
await updateDiskInfo();
|
2022-12-01 16:20:53 +01:00
|
|
|
await _updateBackupAssetCount();
|
2023-12-04 17:21:05 +01:00
|
|
|
} else {
|
|
|
|
log.warning("cannot get backup info - background backup is in progress!");
|
2022-08-18 16:41:59 +02:00
|
|
|
}
|
2022-05-06 14:22:23 +02:00
|
|
|
}
|
|
|
|
|
2023-03-18 15:55:11 +01:00
|
|
|
/// Save user selection of selected albums and excluded albums to database
|
|
|
|
Future<void> _updatePersistentAlbumsSelection() {
|
2022-08-18 16:41:59 +02:00
|
|
|
final epoch = DateTime.fromMillisecondsSinceEpoch(0, isUtc: true);
|
2023-03-18 15:55:11 +01:00
|
|
|
final selected = state.selectedBackupAlbums.map(
|
|
|
|
(e) => BackupAlbum(e.id, e.lastBackup ?? epoch, BackupSelection.select),
|
2022-05-06 14:22:23 +02:00
|
|
|
);
|
2024-06-29 18:30:18 +02:00
|
|
|
final excluded = state.excludedBackupAlbums.map(
|
|
|
|
(e) => BackupAlbum(e.id, e.lastBackup ?? epoch, BackupSelection.exclude),
|
|
|
|
);
|
|
|
|
final backupAlbums = selected.followedBy(excluded).toList();
|
2023-03-18 15:55:11 +01:00
|
|
|
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);
|
|
|
|
});
|
2022-05-06 14:22:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Invoke backup process
|
2022-08-18 16:41:59 +02:00
|
|
|
Future<void> startBackupProcess() async {
|
2022-12-01 16:20:53 +01:00
|
|
|
debugPrint("Start backup process");
|
2022-08-18 16:41:59 +02:00
|
|
|
assert(state.backupProgress == BackUpProgressEnum.idle);
|
2022-05-06 14:22:23 +02:00
|
|
|
state = state.copyWith(backupProgress: BackUpProgressEnum.inProgress);
|
|
|
|
|
2022-06-25 22:12:47 +02:00
|
|
|
await getBackupInfo();
|
|
|
|
|
2023-02-28 17:22:18 +01:00
|
|
|
final hasPermission = _galleryPermissionNotifier.hasPermission;
|
|
|
|
if (hasPermission) {
|
2022-05-06 14:22:23 +02:00
|
|
|
await PhotoManager.clearFileCache();
|
|
|
|
|
2024-06-29 18:30:18 +02:00
|
|
|
if (state.allUniqueAssets.isEmpty) {
|
2022-11-27 21:34:19 +01:00
|
|
|
log.info("No Asset On Device - Abort Backup Process");
|
2022-05-06 14:22:23 +02:00
|
|
|
state = state.copyWith(backupProgress: BackUpProgressEnum.idle);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-06-29 18:30:18 +02:00
|
|
|
Set<AssetEntity> assetsWillBeBackup = Set.from(state.allUniqueAssets);
|
2022-05-06 14:22:23 +02:00
|
|
|
// Remove item that has already been backed up
|
2023-03-18 15:55:11 +01:00
|
|
|
for (final assetId in state.allAssetsInDatabase) {
|
2022-05-06 14:22:23 +02:00
|
|
|
assetsWillBeBackup.removeWhere((e) => e.id == assetId);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (assetsWillBeBackup.isEmpty) {
|
|
|
|
state = state.copyWith(backupProgress: BackUpProgressEnum.idle);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Perform Backup
|
2022-06-18 14:36:58 +02:00
|
|
|
state = state.copyWith(cancelToken: CancellationToken());
|
2023-12-07 16:53:15 +01:00
|
|
|
|
2023-12-10 16:56:39 +01:00
|
|
|
final pmProgressHandler = Platform.isIOS ? PMProgressHandler() : null;
|
2023-12-07 16:53:15 +01:00
|
|
|
|
2023-12-10 16:56:39 +01:00
|
|
|
pmProgressHandler?.stream.listen((event) {
|
2023-12-07 16:53:15 +01:00
|
|
|
final double progress = event.progress;
|
|
|
|
state = state.copyWith(iCloudDownloadProgress: progress);
|
|
|
|
});
|
|
|
|
|
2022-08-18 16:41:59 +02:00
|
|
|
await _backupService.backupAsset(
|
2022-07-06 23:12:55 +02:00
|
|
|
assetsWillBeBackup,
|
|
|
|
state.cancelToken,
|
2023-12-07 16:53:15 +01:00
|
|
|
pmProgressHandler,
|
2022-07-06 23:12:55 +02:00
|
|
|
_onAssetUploaded,
|
|
|
|
_onUploadProgress,
|
|
|
|
_onSetCurrentBackupAsset,
|
|
|
|
_onBackupError,
|
|
|
|
);
|
2023-08-06 04:40:50 +02:00
|
|
|
await notifyBackgroundServiceCanRun();
|
2022-05-06 14:22:23 +02:00
|
|
|
} else {
|
2023-02-28 17:22:18 +01:00
|
|
|
openAppSettings();
|
2022-05-06 14:22:23 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-21 04:43:39 +01:00
|
|
|
void setAvailableAlbums(availableAlbums) {
|
|
|
|
state = state.copyWith(
|
|
|
|
availableAlbums: availableAlbums,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2022-07-06 23:12:55 +02:00
|
|
|
void _onBackupError(ErrorUploadAsset errorAssetInfo) {
|
|
|
|
ref.watch(errorBackupListProvider.notifier).add(errorAssetInfo);
|
|
|
|
}
|
|
|
|
|
|
|
|
void _onSetCurrentBackupAsset(CurrentUploadAsset currentUploadAsset) {
|
|
|
|
state = state.copyWith(currentUploadAsset: currentUploadAsset);
|
|
|
|
}
|
|
|
|
|
2022-05-06 14:22:23 +02:00
|
|
|
void cancelBackup() {
|
2022-08-18 16:41:59 +02:00
|
|
|
if (state.backupProgress != BackUpProgressEnum.inProgress) {
|
2023-08-06 04:40:50 +02:00
|
|
|
notifyBackgroundServiceCanRun();
|
2022-08-18 16:41:59 +02:00
|
|
|
}
|
2022-06-18 14:36:58 +02:00
|
|
|
state.cancelToken.cancel();
|
2022-06-22 07:23:35 +02:00
|
|
|
state = state.copyWith(
|
2022-07-13 14:23:48 +02:00
|
|
|
backupProgress: BackUpProgressEnum.idle,
|
|
|
|
progressInPercentage: 0.0,
|
feat(mobile): Adds file upload progress stats (#7760)
* feat(mobile): Adds file upload progress stats: current upload file size uploaded, current file size and formatted bytes per second upload speed. Closes #7379
* chore(mobile): Fix stan issues
* chore(mobile): Remove non-'en-US' translations, as I saw on another PR review (just looking around) that localisation is done via Localizely and this was the instruction (to only provide the en-US localisation).
* fix(mobile): Provide boundary checks to ensure overflow issues are accounted for on erroneous upload speed calculation, sometimes the numbers received back from the upload handler can be a bit wild.
* fix(mobile): Some heuristic bug fixing. Whilst thinking what could trigger overflow issues or 'zero' readouts, left over values from the previous file may do that. So adding the last upload sent bytes to the values to be reset may help! The time isn't necessary, as the period/cycle is inconsequential in this circumstance, well it should be anyway.
* fix(mobile): Actually, in combination to the last commit, some more heuristic bug fixing. I was thinking it would be advantageous not to reset the update time, as it would trigger a quicker first upload speed calculation. However, I realised that could also cause the calculation to be incorrect on the first cycle as the period wouldn't align. Not really sure if it would be a big deal, but I'm taking wild guesses in the dark here. Again, some purely heuristic debugging as I can't re-produce the underlying issue. This is mainly just ensuring that the state is fully reset and is a known state at the beginning of each file as a common strategy to reduce issues.
* refactor(mobile): Move the UI for the file progress to underneath the progress bar, it makes more sense there than in the file information table which contains only static information pertaining to the file itself. Switching to a monospace font to keep the UI from jumping around as the numbers change.
* refactor(mobile): In order to have the UI always present an 'active' upload speed (as per the discussion on PR #7760), this stores the 'upload speeds' (capped at the latest 10) in a list and calculates the current upload speed as the average over them. This way the UI can always display a 'constant' upload speed during uploading, instead of starting a fresh when each file starts uploading. Limiting it to the 10 latest keeps the average somewhat recent and ensures some level of sensible memory allocation.
2024-03-14 21:15:22 +01:00
|
|
|
progressInFileSize: "0 B / 0 B",
|
|
|
|
progressInFileSpeed: 0,
|
|
|
|
progressInFileSpeedUpdateTime: DateTime.now(),
|
|
|
|
progressInFileSpeedUpdateSentBytes: 0,
|
2022-07-13 14:23:48 +02:00
|
|
|
);
|
2022-05-06 14:22:23 +02:00
|
|
|
}
|
|
|
|
|
2022-10-25 16:51:03 +02:00
|
|
|
void _onAssetUploaded(
|
|
|
|
String deviceAssetId,
|
|
|
|
String deviceId,
|
|
|
|
bool isDuplicated,
|
|
|
|
) {
|
|
|
|
if (isDuplicated) {
|
|
|
|
state = state.copyWith(
|
2024-06-29 18:30:18 +02:00
|
|
|
allUniqueAssets: state.allUniqueAssets
|
2022-10-25 16:51:03 +02:00
|
|
|
.where((asset) => asset.id != deviceAssetId)
|
|
|
|
.toSet(),
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
state = state.copyWith(
|
|
|
|
selectedAlbumsBackupAssetsIds: {
|
|
|
|
...state.selectedAlbumsBackupAssetsIds,
|
2023-08-19 00:52:40 +02:00
|
|
|
deviceAssetId,
|
2022-10-25 16:51:03 +02:00
|
|
|
},
|
|
|
|
allAssetsInDatabase: [...state.allAssetsInDatabase, deviceAssetId],
|
|
|
|
);
|
|
|
|
}
|
2022-06-22 07:23:35 +02:00
|
|
|
|
2024-06-29 18:30:18 +02:00
|
|
|
if (state.allUniqueAssets.length -
|
2022-06-22 07:23:35 +02:00
|
|
|
state.selectedAlbumsBackupAssetsIds.length ==
|
|
|
|
0) {
|
2022-08-18 16:41:59 +02:00
|
|
|
final latestAssetBackup =
|
2024-06-29 18:30:18 +02:00
|
|
|
state.allUniqueAssets.map((e) => e.modifiedDateTime).reduce(
|
2022-08-18 16:41:59 +02:00
|
|
|
(v, e) => e.isAfter(v) ? e : v,
|
|
|
|
);
|
2022-06-22 07:23:35 +02:00
|
|
|
state = state.copyWith(
|
2022-08-18 16:41:59 +02:00
|
|
|
selectedBackupAlbums: state.selectedBackupAlbums
|
|
|
|
.map((e) => e.copyWith(lastBackup: latestAssetBackup))
|
|
|
|
.toSet(),
|
2024-06-29 18:30:18 +02:00
|
|
|
excludedBackupAlbums: state.excludedBackupAlbums
|
|
|
|
.map((e) => e.copyWith(lastBackup: latestAssetBackup))
|
|
|
|
.toSet(),
|
2022-07-13 14:23:48 +02:00
|
|
|
backupProgress: BackUpProgressEnum.done,
|
|
|
|
progressInPercentage: 0.0,
|
feat(mobile): Adds file upload progress stats (#7760)
* feat(mobile): Adds file upload progress stats: current upload file size uploaded, current file size and formatted bytes per second upload speed. Closes #7379
* chore(mobile): Fix stan issues
* chore(mobile): Remove non-'en-US' translations, as I saw on another PR review (just looking around) that localisation is done via Localizely and this was the instruction (to only provide the en-US localisation).
* fix(mobile): Provide boundary checks to ensure overflow issues are accounted for on erroneous upload speed calculation, sometimes the numbers received back from the upload handler can be a bit wild.
* fix(mobile): Some heuristic bug fixing. Whilst thinking what could trigger overflow issues or 'zero' readouts, left over values from the previous file may do that. So adding the last upload sent bytes to the values to be reset may help! The time isn't necessary, as the period/cycle is inconsequential in this circumstance, well it should be anyway.
* fix(mobile): Actually, in combination to the last commit, some more heuristic bug fixing. I was thinking it would be advantageous not to reset the update time, as it would trigger a quicker first upload speed calculation. However, I realised that could also cause the calculation to be incorrect on the first cycle as the period wouldn't align. Not really sure if it would be a big deal, but I'm taking wild guesses in the dark here. Again, some purely heuristic debugging as I can't re-produce the underlying issue. This is mainly just ensuring that the state is fully reset and is a known state at the beginning of each file as a common strategy to reduce issues.
* refactor(mobile): Move the UI for the file progress to underneath the progress bar, it makes more sense there than in the file information table which contains only static information pertaining to the file itself. Switching to a monospace font to keep the UI from jumping around as the numbers change.
* refactor(mobile): In order to have the UI always present an 'active' upload speed (as per the discussion on PR #7760), this stores the 'upload speeds' (capped at the latest 10) in a list and calculates the current upload speed as the average over them. This way the UI can always display a 'constant' upload speed during uploading, instead of starting a fresh when each file starts uploading. Limiting it to the 10 latest keeps the average somewhat recent and ensures some level of sensible memory allocation.
2024-03-14 21:15:22 +01:00
|
|
|
progressInFileSize: "0 B / 0 B",
|
|
|
|
progressInFileSpeed: 0,
|
|
|
|
progressInFileSpeedUpdateTime: DateTime.now(),
|
|
|
|
progressInFileSpeedUpdateSentBytes: 0,
|
2022-07-13 14:23:48 +02:00
|
|
|
);
|
2022-08-18 16:41:59 +02:00
|
|
|
_updatePersistentAlbumsSelection();
|
2022-05-06 14:22:23 +02:00
|
|
|
}
|
|
|
|
|
2024-05-22 11:25:55 +02:00
|
|
|
updateDiskInfo();
|
2022-05-06 14:22:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void _onUploadProgress(int sent, int total) {
|
feat(mobile): Adds file upload progress stats (#7760)
* feat(mobile): Adds file upload progress stats: current upload file size uploaded, current file size and formatted bytes per second upload speed. Closes #7379
* chore(mobile): Fix stan issues
* chore(mobile): Remove non-'en-US' translations, as I saw on another PR review (just looking around) that localisation is done via Localizely and this was the instruction (to only provide the en-US localisation).
* fix(mobile): Provide boundary checks to ensure overflow issues are accounted for on erroneous upload speed calculation, sometimes the numbers received back from the upload handler can be a bit wild.
* fix(mobile): Some heuristic bug fixing. Whilst thinking what could trigger overflow issues or 'zero' readouts, left over values from the previous file may do that. So adding the last upload sent bytes to the values to be reset may help! The time isn't necessary, as the period/cycle is inconsequential in this circumstance, well it should be anyway.
* fix(mobile): Actually, in combination to the last commit, some more heuristic bug fixing. I was thinking it would be advantageous not to reset the update time, as it would trigger a quicker first upload speed calculation. However, I realised that could also cause the calculation to be incorrect on the first cycle as the period wouldn't align. Not really sure if it would be a big deal, but I'm taking wild guesses in the dark here. Again, some purely heuristic debugging as I can't re-produce the underlying issue. This is mainly just ensuring that the state is fully reset and is a known state at the beginning of each file as a common strategy to reduce issues.
* refactor(mobile): Move the UI for the file progress to underneath the progress bar, it makes more sense there than in the file information table which contains only static information pertaining to the file itself. Switching to a monospace font to keep the UI from jumping around as the numbers change.
* refactor(mobile): In order to have the UI always present an 'active' upload speed (as per the discussion on PR #7760), this stores the 'upload speeds' (capped at the latest 10) in a list and calculates the current upload speed as the average over them. This way the UI can always display a 'constant' upload speed during uploading, instead of starting a fresh when each file starts uploading. Limiting it to the 10 latest keeps the average somewhat recent and ensures some level of sensible memory allocation.
2024-03-14 21:15:22 +01:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2022-06-22 07:23:35 +02:00
|
|
|
state = state.copyWith(
|
2022-07-13 14:23:48 +02:00
|
|
|
progressInPercentage: (sent.toDouble() / total.toDouble() * 100),
|
feat(mobile): Adds file upload progress stats (#7760)
* feat(mobile): Adds file upload progress stats: current upload file size uploaded, current file size and formatted bytes per second upload speed. Closes #7379
* chore(mobile): Fix stan issues
* chore(mobile): Remove non-'en-US' translations, as I saw on another PR review (just looking around) that localisation is done via Localizely and this was the instruction (to only provide the en-US localisation).
* fix(mobile): Provide boundary checks to ensure overflow issues are accounted for on erroneous upload speed calculation, sometimes the numbers received back from the upload handler can be a bit wild.
* fix(mobile): Some heuristic bug fixing. Whilst thinking what could trigger overflow issues or 'zero' readouts, left over values from the previous file may do that. So adding the last upload sent bytes to the values to be reset may help! The time isn't necessary, as the period/cycle is inconsequential in this circumstance, well it should be anyway.
* fix(mobile): Actually, in combination to the last commit, some more heuristic bug fixing. I was thinking it would be advantageous not to reset the update time, as it would trigger a quicker first upload speed calculation. However, I realised that could also cause the calculation to be incorrect on the first cycle as the period wouldn't align. Not really sure if it would be a big deal, but I'm taking wild guesses in the dark here. Again, some purely heuristic debugging as I can't re-produce the underlying issue. This is mainly just ensuring that the state is fully reset and is a known state at the beginning of each file as a common strategy to reduce issues.
* refactor(mobile): Move the UI for the file progress to underneath the progress bar, it makes more sense there than in the file information table which contains only static information pertaining to the file itself. Switching to a monospace font to keep the UI from jumping around as the numbers change.
* refactor(mobile): In order to have the UI always present an 'active' upload speed (as per the discussion on PR #7760), this stores the 'upload speeds' (capped at the latest 10) in a list and calculates the current upload speed as the average over them. This way the UI can always display a 'constant' upload speed during uploading, instead of starting a fresh when each file starts uploading. Limiting it to the 10 latest keeps the average somewhat recent and ensures some level of sensible memory allocation.
2024-03-14 21:15:22 +01:00
|
|
|
progressInFileSize: humanReadableFileBytesProgress(sent, total),
|
|
|
|
progressInFileSpeed: lastUploadSpeed,
|
|
|
|
progressInFileSpeeds: lastUploadSpeeds,
|
|
|
|
progressInFileSpeedUpdateTime: lastUpdateTime,
|
|
|
|
progressInFileSpeedUpdateSentBytes: lastSentBytes,
|
2022-07-13 14:23:48 +02:00
|
|
|
);
|
2022-05-06 14:22:23 +02:00
|
|
|
}
|
|
|
|
|
2024-05-22 11:25:55 +02:00
|
|
|
Future<void> updateDiskInfo() async {
|
|
|
|
final diskInfo = await _serverInfoService.getDiskInfo();
|
2022-05-06 14:22:23 +02:00
|
|
|
|
|
|
|
// Update server info
|
2024-05-22 11:25:55 +02:00
|
|
|
if (diskInfo != null) {
|
2022-07-13 14:23:48 +02:00
|
|
|
state = state.copyWith(
|
2024-05-22 11:25:55 +02:00
|
|
|
serverInfo: diskInfo,
|
2022-07-13 14:23:48 +02:00
|
|
|
);
|
|
|
|
}
|
2022-05-06 14:22:23 +02:00
|
|
|
}
|
|
|
|
|
2022-08-18 16:41:59 +02:00
|
|
|
Future<void> _resumeBackup() async {
|
2022-05-06 14:22:23 +02:00
|
|
|
// Check if user is login
|
2023-03-23 02:36:44 +01:00
|
|
|
final accessKey = Store.tryGet(StoreKey.accessToken);
|
2022-05-06 14:22:23 +02:00
|
|
|
|
|
|
|
// User has been logged out return
|
2022-06-25 20:46:51 +02:00
|
|
|
if (accessKey == null || !_authState.isAuthenticated) {
|
2022-11-27 21:34:19 +01:00
|
|
|
log.info("[_resumeBackup] not authenticated - abort");
|
2022-06-25 20:46:51 +02:00
|
|
|
return;
|
|
|
|
}
|
2022-05-06 14:22:23 +02:00
|
|
|
|
2022-06-25 20:46:51 +02:00
|
|
|
// Check if this device is enable backup by the user
|
2023-03-23 16:25:58 +01:00
|
|
|
if (state.autoBackup) {
|
2023-08-06 04:40:50 +02:00
|
|
|
// check if backup is already in process - then return
|
2022-06-25 20:46:51 +02:00
|
|
|
if (state.backupProgress == BackUpProgressEnum.inProgress) {
|
2023-08-06 04:40:50 +02:00
|
|
|
log.info("[_resumeBackup] Auto Backup is already in progress - abort");
|
2022-06-25 20:46:51 +02:00
|
|
|
return;
|
2022-05-06 14:22:23 +02:00
|
|
|
}
|
|
|
|
|
2022-08-18 16:41:59 +02:00
|
|
|
if (state.backupProgress == BackUpProgressEnum.inBackground) {
|
2022-11-27 21:34:19 +01:00
|
|
|
log.info("[_resumeBackup] Background backup is running - abort");
|
2022-08-18 16:41:59 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-08-06 04:40:50 +02:00
|
|
|
if (state.backupProgress == BackUpProgressEnum.manualInProgress) {
|
|
|
|
log.info("[_resumeBackup] Manual upload is running - abort");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-06-25 20:46:51 +02:00
|
|
|
// Run backup
|
2022-11-27 21:34:19 +01:00
|
|
|
log.info("[_resumeBackup] Start back up");
|
2022-08-18 16:41:59 +02:00
|
|
|
await startBackupProcess();
|
2022-05-06 14:22:23 +02:00
|
|
|
}
|
2022-06-25 20:46:51 +02:00
|
|
|
return;
|
2022-05-06 14:22:23 +02:00
|
|
|
}
|
2022-08-18 16:41:59 +02:00
|
|
|
|
|
|
|
Future<void> resumeBackup() async {
|
2023-03-18 15:55:11 +01:00
|
|
|
final List<BackupAlbum> selectedBackupAlbums = await _db.backupAlbums
|
|
|
|
.filter()
|
|
|
|
.selectionEqualTo(BackupSelection.select)
|
|
|
|
.findAll();
|
2024-06-29 18:30:18 +02:00
|
|
|
final List<BackupAlbum> excludedBackupAlbums = await _db.backupAlbums
|
|
|
|
.filter()
|
|
|
|
.selectionEqualTo(BackupSelection.exclude)
|
|
|
|
.findAll();
|
2023-02-20 06:59:50 +01:00
|
|
|
Set<AvailableAlbum> selectedAlbums = state.selectedBackupAlbums;
|
2024-06-29 18:30:18 +02:00
|
|
|
Set<AvailableAlbum> excludedAlbums = state.excludedBackupAlbums;
|
2023-03-18 15:55:11 +01:00
|
|
|
if (selectedAlbums.isNotEmpty) {
|
|
|
|
selectedAlbums = _updateAlbumsBackupTime(
|
|
|
|
selectedAlbums,
|
|
|
|
selectedBackupAlbums,
|
|
|
|
);
|
|
|
|
}
|
2023-02-21 16:25:31 +01:00
|
|
|
|
2024-06-29 18:30:18 +02:00
|
|
|
if (excludedAlbums.isNotEmpty) {
|
|
|
|
excludedAlbums = _updateAlbumsBackupTime(
|
|
|
|
excludedAlbums,
|
|
|
|
excludedBackupAlbums,
|
|
|
|
);
|
|
|
|
}
|
2023-03-18 15:55:11 +01:00
|
|
|
final BackUpProgressEnum previous = state.backupProgress;
|
2023-02-20 06:59:50 +01:00
|
|
|
state = state.copyWith(
|
2023-03-18 15:55:11 +01:00
|
|
|
backupProgress: BackUpProgressEnum.inBackground,
|
2023-02-20 06:59:50 +01:00
|
|
|
selectedBackupAlbums: selectedAlbums,
|
2024-06-29 18:30:18 +02:00
|
|
|
excludedBackupAlbums: excludedAlbums,
|
2023-02-20 06:59:50 +01:00
|
|
|
);
|
2023-03-18 15:55:11 +01:00
|
|
|
// 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);
|
|
|
|
}
|
2022-08-18 16:41:59 +02:00
|
|
|
return _resumeBackup();
|
|
|
|
}
|
|
|
|
|
|
|
|
Set<AvailableAlbum> _updateAlbumsBackupTime(
|
|
|
|
Set<AvailableAlbum> albums,
|
2023-03-18 15:55:11 +01:00
|
|
|
List<BackupAlbum> backupAlbums,
|
2022-08-18 16:41:59 +02:00
|
|
|
) {
|
|
|
|
Set<AvailableAlbum> result = {};
|
2023-03-18 15:55:11 +01:00
|
|
|
for (BackupAlbum ba in backupAlbums) {
|
2022-08-18 16:41:59 +02:00
|
|
|
try {
|
2023-03-18 15:55:11 +01:00
|
|
|
AvailableAlbum a = albums.firstWhere((e) => e.id == ba.id);
|
|
|
|
result.add(a.copyWith(lastBackup: ba.lastBackup));
|
2022-08-18 16:41:59 +02:00
|
|
|
} on StateError {
|
2022-11-27 21:34:19 +01:00
|
|
|
log.severe(
|
|
|
|
"[_updateAlbumBackupTime] failed to find album in state",
|
|
|
|
"State Error",
|
|
|
|
StackTrace.current,
|
|
|
|
);
|
2022-08-18 16:41:59 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2023-08-06 04:40:50 +02:00
|
|
|
Future<void> notifyBackgroundServiceCanRun() async {
|
2022-08-18 16:41:59 +02:00
|
|
|
const allowedStates = [
|
2024-05-02 22:59:14 +02:00
|
|
|
AppLifeCycleEnum.inactive,
|
|
|
|
AppLifeCycleEnum.paused,
|
|
|
|
AppLifeCycleEnum.detached,
|
2022-08-18 16:41:59 +02:00
|
|
|
];
|
2023-02-20 06:59:50 +01:00
|
|
|
if (allowedStates.contains(ref.read(appStateProvider.notifier).state)) {
|
2022-08-18 16:41:59 +02:00
|
|
|
_backgroundService.releaseLock();
|
|
|
|
}
|
|
|
|
}
|
2023-08-06 04:40:50 +02:00
|
|
|
|
|
|
|
BackUpProgressEnum get backupProgress => state.backupProgress;
|
|
|
|
void updateBackupProgress(BackUpProgressEnum backupProgress) {
|
|
|
|
state = state.copyWith(backupProgress: backupProgress);
|
|
|
|
}
|
2022-05-06 14:22:23 +02:00
|
|
|
}
|
|
|
|
|
2022-06-22 07:23:35 +02:00
|
|
|
final backupProvider =
|
|
|
|
StateNotifierProvider<BackupNotifier, BackUpState>((ref) {
|
2022-06-25 20:46:51 +02:00
|
|
|
return BackupNotifier(
|
|
|
|
ref.watch(backupServiceProvider),
|
|
|
|
ref.watch(serverInfoServiceProvider),
|
|
|
|
ref.watch(authenticationProvider),
|
2022-08-18 16:41:59 +02:00
|
|
|
ref.watch(backgroundServiceProvider),
|
2023-02-28 17:22:18 +01:00
|
|
|
ref.watch(galleryPermissionNotifier.notifier),
|
2023-03-18 15:55:11 +01:00
|
|
|
ref.watch(dbProvider),
|
2022-07-06 23:12:55 +02:00
|
|
|
ref,
|
2022-06-25 20:46:51 +02:00
|
|
|
);
|
2022-05-06 14:22:23 +02:00
|
|
|
});
|