2022-02-13 23:10:42 +02:00
|
|
|
import 'dart:async';
|
|
|
|
|
2022-02-03 18:06:44 +02:00
|
|
|
import 'package:dio/dio.dart';
|
|
|
|
import 'package:flutter/foundation.dart';
|
2022-02-08 07:42:35 +02:00
|
|
|
import 'package:hive_flutter/hive_flutter.dart';
|
2022-02-03 18:06:44 +02:00
|
|
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
2022-02-08 07:42:35 +02:00
|
|
|
import 'package:immich_mobile/constants/hive_box.dart';
|
|
|
|
import 'package:immich_mobile/modules/login/providers/authentication.provider.dart';
|
2022-02-03 18:06:44 +02:00
|
|
|
import 'package:immich_mobile/shared/services/server_info.service.dart';
|
|
|
|
import 'package:immich_mobile/shared/models/backup_state.model.dart';
|
|
|
|
import 'package:immich_mobile/shared/models/server_info.model.dart';
|
|
|
|
import 'package:immich_mobile/shared/services/backup.service.dart';
|
|
|
|
import 'package:photo_manager/photo_manager.dart';
|
|
|
|
|
|
|
|
class BackupNotifier extends StateNotifier<BackUpState> {
|
2022-02-13 23:10:42 +02:00
|
|
|
BackupNotifier({this.ref})
|
2022-02-03 18:06:44 +02:00
|
|
|
: super(
|
|
|
|
BackUpState(
|
|
|
|
backupProgress: BackUpProgressEnum.idle,
|
|
|
|
backingUpAssetCount: 0,
|
|
|
|
assetOnDatabase: 0,
|
|
|
|
totalAssetCount: 0,
|
|
|
|
progressInPercentage: 0,
|
|
|
|
cancelToken: CancelToken(),
|
|
|
|
serverInfo: ServerInfo(
|
|
|
|
diskAvailable: "0",
|
|
|
|
diskAvailableRaw: 0,
|
|
|
|
diskSize: "0",
|
|
|
|
diskSizeRaw: 0,
|
|
|
|
diskUsagePercentage: 0.0,
|
|
|
|
diskUse: "0",
|
|
|
|
diskUseRaw: 0,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
);
|
|
|
|
|
2022-02-13 23:10:42 +02:00
|
|
|
Ref? ref;
|
2022-02-03 18:06:44 +02:00
|
|
|
final BackupService _backupService = BackupService();
|
|
|
|
final ServerInfoService _serverInfoService = ServerInfoService();
|
2022-02-13 23:10:42 +02:00
|
|
|
final StreamController _onAssetBackupStreamCtrl = StreamController.broadcast();
|
2022-02-03 18:06:44 +02:00
|
|
|
|
|
|
|
void getBackupInfo() async {
|
|
|
|
_updateServerInfo();
|
|
|
|
|
2022-02-06 08:07:56 +02:00
|
|
|
List<AssetPathEntity> list = await PhotoManager.getAssetPathList(onlyAll: true, type: RequestType.common);
|
2022-02-13 23:10:42 +02:00
|
|
|
List<String> didBackupAsset = await _backupService.getDeviceBackupAsset();
|
2022-02-03 18:06:44 +02:00
|
|
|
|
|
|
|
if (list.isEmpty) {
|
|
|
|
debugPrint("No Asset On Device");
|
2022-02-13 23:10:42 +02:00
|
|
|
state = state.copyWith(
|
|
|
|
backupProgress: BackUpProgressEnum.idle, totalAssetCount: 0, assetOnDatabase: didBackupAsset.length);
|
2022-02-03 18:06:44 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
int totalAsset = list[0].assetCount;
|
|
|
|
|
|
|
|
state = state.copyWith(totalAssetCount: totalAsset, assetOnDatabase: didBackupAsset.length);
|
|
|
|
}
|
|
|
|
|
|
|
|
void startBackupProcess() async {
|
|
|
|
_updateServerInfo();
|
|
|
|
|
|
|
|
state = state.copyWith(backupProgress: BackUpProgressEnum.inProgress);
|
|
|
|
|
|
|
|
var authResult = await PhotoManager.requestPermissionExtend();
|
|
|
|
if (authResult.isAuth) {
|
|
|
|
await PhotoManager.clearFileCache();
|
|
|
|
// await PhotoManager.presentLimited();
|
|
|
|
// Gather assets info
|
|
|
|
List<AssetPathEntity> list =
|
2022-02-06 08:07:56 +02:00
|
|
|
await PhotoManager.getAssetPathList(hasAll: true, onlyAll: true, type: RequestType.common);
|
2022-02-03 18:06:44 +02:00
|
|
|
|
2022-02-13 23:10:42 +02:00
|
|
|
// Get device assets info from database
|
|
|
|
// Compare and find different assets that has not been backing up
|
|
|
|
// Backup those assets
|
|
|
|
List<String> backupAsset = await _backupService.getDeviceBackupAsset();
|
|
|
|
|
2022-02-03 18:06:44 +02:00
|
|
|
if (list.isEmpty) {
|
|
|
|
debugPrint("No Asset On Device - Abort Backup Process");
|
2022-02-13 23:10:42 +02:00
|
|
|
state = state.copyWith(
|
|
|
|
backupProgress: BackUpProgressEnum.idle, totalAssetCount: 0, assetOnDatabase: backupAsset.length);
|
2022-02-03 18:06:44 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
int totalAsset = list[0].assetCount;
|
|
|
|
List<AssetEntity> currentAssets = await list[0].getAssetListRange(start: 0, end: totalAsset);
|
|
|
|
|
|
|
|
state = state.copyWith(totalAssetCount: totalAsset, assetOnDatabase: backupAsset.length);
|
|
|
|
// Remove item that has already been backed up
|
|
|
|
for (var backupAssetId in backupAsset) {
|
|
|
|
currentAssets.removeWhere((e) => e.id == backupAssetId);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (currentAssets.isEmpty) {
|
|
|
|
state = state.copyWith(backupProgress: BackUpProgressEnum.idle);
|
|
|
|
}
|
|
|
|
|
|
|
|
state = state.copyWith(backingUpAssetCount: currentAssets.length);
|
|
|
|
|
|
|
|
// Perform Packup
|
|
|
|
state = state.copyWith(cancelToken: CancelToken());
|
|
|
|
_backupService.backupAsset(currentAssets, state.cancelToken, _onAssetUploaded, _onUploadProgress);
|
|
|
|
} else {
|
|
|
|
PhotoManager.openSetting();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void cancelBackup() {
|
|
|
|
state.cancelToken.cancel('Cancel Backup');
|
2022-02-08 07:42:35 +02:00
|
|
|
state = state.copyWith(backupProgress: BackUpProgressEnum.idle, progressInPercentage: 0.0);
|
2022-02-03 18:06:44 +02:00
|
|
|
}
|
|
|
|
|
2022-02-13 23:10:42 +02:00
|
|
|
void _onAssetUploaded(String deviceAssetId, String deviceId) {
|
2022-02-03 18:06:44 +02:00
|
|
|
state =
|
|
|
|
state.copyWith(backingUpAssetCount: state.backingUpAssetCount - 1, assetOnDatabase: state.assetOnDatabase + 1);
|
|
|
|
|
|
|
|
if (state.backingUpAssetCount == 0) {
|
|
|
|
state = state.copyWith(backupProgress: BackUpProgressEnum.done, progressInPercentage: 0.0);
|
|
|
|
}
|
|
|
|
|
|
|
|
_updateServerInfo();
|
|
|
|
}
|
|
|
|
|
|
|
|
void _onUploadProgress(int sent, int total) {
|
|
|
|
state = state.copyWith(progressInPercentage: (sent.toDouble() / total.toDouble() * 100));
|
|
|
|
}
|
|
|
|
|
|
|
|
void _updateServerInfo() async {
|
|
|
|
var serverInfo = await _serverInfoService.getServerInfo();
|
|
|
|
|
|
|
|
// Update server info
|
|
|
|
state = state.copyWith(
|
|
|
|
serverInfo: ServerInfo(
|
|
|
|
diskSize: serverInfo.diskSize,
|
|
|
|
diskUse: serverInfo.diskUse,
|
|
|
|
diskAvailable: serverInfo.diskAvailable,
|
|
|
|
diskSizeRaw: serverInfo.diskSizeRaw,
|
|
|
|
diskUseRaw: serverInfo.diskUseRaw,
|
|
|
|
diskAvailableRaw: serverInfo.diskAvailableRaw,
|
|
|
|
diskUsagePercentage: serverInfo.diskUsagePercentage,
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
2022-02-08 07:42:35 +02:00
|
|
|
|
|
|
|
void resumeBackup() {
|
2022-02-13 23:10:42 +02:00
|
|
|
var authState = ref?.read(authenticationProvider);
|
2022-02-08 07:42:35 +02:00
|
|
|
|
|
|
|
// Check if user is login
|
|
|
|
var accessKey = Hive.box(userInfoBox).get(accessTokenKey);
|
|
|
|
|
|
|
|
// User has been logged out return
|
2022-02-13 23:10:42 +02:00
|
|
|
if (authState != null) {
|
|
|
|
if (accessKey == null || !authState.isAuthenticated) {
|
|
|
|
debugPrint("[resumeBackup] not authenticated - abort");
|
2022-02-08 07:42:35 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-02-13 23:10:42 +02:00
|
|
|
// Check if this device is enable backup by the user
|
|
|
|
if ((authState.deviceInfo.deviceId == authState.deviceId) && authState.deviceInfo.isAutoBackup) {
|
|
|
|
// check if backup is alreayd in process - then return
|
|
|
|
if (state.backupProgress == BackUpProgressEnum.inProgress) {
|
|
|
|
debugPrint("[resumeBackup] Backup is already in progress - abort");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Run backup
|
|
|
|
debugPrint("[resumeBackup] Start back up");
|
|
|
|
startBackupProcess();
|
|
|
|
}
|
2022-02-08 07:42:35 +02:00
|
|
|
|
2022-02-13 23:10:42 +02:00
|
|
|
return;
|
|
|
|
}
|
2022-02-08 07:42:35 +02:00
|
|
|
}
|
2022-02-03 18:06:44 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
final backupProvider = StateNotifierProvider<BackupNotifier, BackUpState>((ref) {
|
2022-02-13 23:10:42 +02:00
|
|
|
return BackupNotifier(ref: ref);
|
2022-02-03 18:06:44 +02:00
|
|
|
});
|