mirror of
https://github.com/immich-app/immich.git
synced 2025-01-01 08:31:59 +00:00
Fixed backup stuck at unsupported format (#108)
* Added webp as supported file type, allow continue upload when an image fail * Added webp as supported file type, allow continue upload when an image fail * Solved issue with bad assets cause backup to stop
This commit is contained in:
parent
b69f6e0df7
commit
3bdcdef198
5 changed files with 52 additions and 27 deletions
|
@ -37,24 +37,29 @@ class BackupNotifier extends StateNotifier<BackUpState> {
|
||||||
Ref? ref;
|
Ref? ref;
|
||||||
final BackupService _backupService = BackupService();
|
final BackupService _backupService = BackupService();
|
||||||
final ServerInfoService _serverInfoService = ServerInfoService();
|
final ServerInfoService _serverInfoService = ServerInfoService();
|
||||||
final StreamController _onAssetBackupStreamCtrl = StreamController.broadcast();
|
final StreamController _onAssetBackupStreamCtrl =
|
||||||
|
StreamController.broadcast();
|
||||||
|
|
||||||
void getBackupInfo() async {
|
void getBackupInfo() async {
|
||||||
_updateServerInfo();
|
_updateServerInfo();
|
||||||
|
|
||||||
List<AssetPathEntity> list = await PhotoManager.getAssetPathList(onlyAll: true, type: RequestType.common);
|
List<AssetPathEntity> list = await PhotoManager.getAssetPathList(
|
||||||
|
onlyAll: true, type: RequestType.common);
|
||||||
List<String> didBackupAsset = await _backupService.getDeviceBackupAsset();
|
List<String> didBackupAsset = await _backupService.getDeviceBackupAsset();
|
||||||
|
|
||||||
if (list.isEmpty) {
|
if (list.isEmpty) {
|
||||||
debugPrint("No Asset On Device");
|
debugPrint("No Asset On Device");
|
||||||
state = state.copyWith(
|
state = state.copyWith(
|
||||||
backupProgress: BackUpProgressEnum.idle, totalAssetCount: 0, assetOnDatabase: didBackupAsset.length);
|
backupProgress: BackUpProgressEnum.idle,
|
||||||
|
totalAssetCount: 0,
|
||||||
|
assetOnDatabase: didBackupAsset.length);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
int totalAsset = list[0].assetCount;
|
int totalAsset = list[0].assetCount;
|
||||||
|
|
||||||
state = state.copyWith(totalAssetCount: totalAsset, assetOnDatabase: didBackupAsset.length);
|
state = state.copyWith(
|
||||||
|
totalAssetCount: totalAsset, assetOnDatabase: didBackupAsset.length);
|
||||||
}
|
}
|
||||||
|
|
||||||
void startBackupProcess() async {
|
void startBackupProcess() async {
|
||||||
|
@ -67,8 +72,8 @@ class BackupNotifier extends StateNotifier<BackUpState> {
|
||||||
await PhotoManager.clearFileCache();
|
await PhotoManager.clearFileCache();
|
||||||
// await PhotoManager.presentLimited();
|
// await PhotoManager.presentLimited();
|
||||||
// Gather assets info
|
// Gather assets info
|
||||||
List<AssetPathEntity> list =
|
List<AssetPathEntity> list = await PhotoManager.getAssetPathList(
|
||||||
await PhotoManager.getAssetPathList(hasAll: true, onlyAll: true, type: RequestType.common);
|
hasAll: true, onlyAll: true, type: RequestType.common);
|
||||||
|
|
||||||
// Get device assets info from database
|
// Get device assets info from database
|
||||||
// Compare and find different assets that has not been backing up
|
// Compare and find different assets that has not been backing up
|
||||||
|
@ -78,14 +83,18 @@ class BackupNotifier extends StateNotifier<BackUpState> {
|
||||||
if (list.isEmpty) {
|
if (list.isEmpty) {
|
||||||
debugPrint("No Asset On Device - Abort Backup Process");
|
debugPrint("No Asset On Device - Abort Backup Process");
|
||||||
state = state.copyWith(
|
state = state.copyWith(
|
||||||
backupProgress: BackUpProgressEnum.idle, totalAssetCount: 0, assetOnDatabase: backupAsset.length);
|
backupProgress: BackUpProgressEnum.idle,
|
||||||
|
totalAssetCount: 0,
|
||||||
|
assetOnDatabase: backupAsset.length);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
int totalAsset = list[0].assetCount;
|
int totalAsset = list[0].assetCount;
|
||||||
List<AssetEntity> currentAssets = await list[0].getAssetListRange(start: 0, end: totalAsset);
|
List<AssetEntity> currentAssets =
|
||||||
|
await list[0].getAssetListRange(start: 0, end: totalAsset);
|
||||||
|
|
||||||
state = state.copyWith(totalAssetCount: totalAsset, assetOnDatabase: backupAsset.length);
|
state = state.copyWith(
|
||||||
|
totalAssetCount: totalAsset, assetOnDatabase: backupAsset.length);
|
||||||
// Remove item that has already been backed up
|
// Remove item that has already been backed up
|
||||||
for (var backupAssetId in backupAsset) {
|
for (var backupAssetId in backupAsset) {
|
||||||
currentAssets.removeWhere((e) => e.id == backupAssetId);
|
currentAssets.removeWhere((e) => e.id == backupAssetId);
|
||||||
|
@ -97,9 +106,10 @@ class BackupNotifier extends StateNotifier<BackUpState> {
|
||||||
|
|
||||||
state = state.copyWith(backingUpAssetCount: currentAssets.length);
|
state = state.copyWith(backingUpAssetCount: currentAssets.length);
|
||||||
|
|
||||||
// Perform Packup
|
// Perform Backup
|
||||||
state = state.copyWith(cancelToken: CancelToken());
|
state = state.copyWith(cancelToken: CancelToken());
|
||||||
_backupService.backupAsset(currentAssets, state.cancelToken, _onAssetUploaded, _onUploadProgress);
|
_backupService.backupAsset(currentAssets, state.cancelToken,
|
||||||
|
_onAssetUploaded, _onUploadProgress);
|
||||||
} else {
|
} else {
|
||||||
PhotoManager.openSetting();
|
PhotoManager.openSetting();
|
||||||
}
|
}
|
||||||
|
@ -107,22 +117,26 @@ class BackupNotifier extends StateNotifier<BackUpState> {
|
||||||
|
|
||||||
void cancelBackup() {
|
void cancelBackup() {
|
||||||
state.cancelToken.cancel('Cancel Backup');
|
state.cancelToken.cancel('Cancel Backup');
|
||||||
state = state.copyWith(backupProgress: BackUpProgressEnum.idle, progressInPercentage: 0.0);
|
state = state.copyWith(
|
||||||
|
backupProgress: BackUpProgressEnum.idle, progressInPercentage: 0.0);
|
||||||
}
|
}
|
||||||
|
|
||||||
void _onAssetUploaded(String deviceAssetId, String deviceId) {
|
void _onAssetUploaded(String deviceAssetId, String deviceId) {
|
||||||
state =
|
state = state.copyWith(
|
||||||
state.copyWith(backingUpAssetCount: state.backingUpAssetCount - 1, assetOnDatabase: state.assetOnDatabase + 1);
|
backingUpAssetCount: state.backingUpAssetCount - 1,
|
||||||
|
assetOnDatabase: state.assetOnDatabase + 1);
|
||||||
|
|
||||||
if (state.backingUpAssetCount == 0) {
|
if (state.backingUpAssetCount == 0) {
|
||||||
state = state.copyWith(backupProgress: BackUpProgressEnum.done, progressInPercentage: 0.0);
|
state = state.copyWith(
|
||||||
|
backupProgress: BackUpProgressEnum.done, progressInPercentage: 0.0);
|
||||||
}
|
}
|
||||||
|
|
||||||
_updateServerInfo();
|
_updateServerInfo();
|
||||||
}
|
}
|
||||||
|
|
||||||
void _onUploadProgress(int sent, int total) {
|
void _onUploadProgress(int sent, int total) {
|
||||||
state = state.copyWith(progressInPercentage: (sent.toDouble() / total.toDouble() * 100));
|
state = state.copyWith(
|
||||||
|
progressInPercentage: (sent.toDouble() / total.toDouble() * 100));
|
||||||
}
|
}
|
||||||
|
|
||||||
void _updateServerInfo() async {
|
void _updateServerInfo() async {
|
||||||
|
@ -156,7 +170,8 @@ class BackupNotifier extends StateNotifier<BackUpState> {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check if this device is enable backup by the user
|
// Check if this device is enable backup by the user
|
||||||
if ((authState.deviceInfo.deviceId == authState.deviceId) && authState.deviceInfo.isAutoBackup) {
|
if ((authState.deviceInfo.deviceId == authState.deviceId) &&
|
||||||
|
authState.deviceInfo.isAutoBackup) {
|
||||||
// check if backup is alreayd in process - then return
|
// check if backup is alreayd in process - then return
|
||||||
if (state.backupProgress == BackUpProgressEnum.inProgress) {
|
if (state.backupProgress == BackUpProgressEnum.inProgress) {
|
||||||
debugPrint("[resumeBackup] Backup is already in progress - abort");
|
debugPrint("[resumeBackup] Backup is already in progress - abort");
|
||||||
|
@ -173,6 +188,7 @@ class BackupNotifier extends StateNotifier<BackUpState> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
final backupProvider = StateNotifierProvider<BackupNotifier, BackUpState>((ref) {
|
final backupProvider =
|
||||||
|
StateNotifierProvider<BackupNotifier, BackUpState>((ref) {
|
||||||
return BackupNotifier(ref: ref);
|
return BackupNotifier(ref: ref);
|
||||||
});
|
});
|
||||||
|
|
|
@ -112,7 +112,10 @@ class BackupService {
|
||||||
}
|
}
|
||||||
} on DioError catch (e) {
|
} on DioError catch (e) {
|
||||||
debugPrint("DioError backupAsset: ${e.response}");
|
debugPrint("DioError backupAsset: ${e.response}");
|
||||||
break;
|
if (e.type == DioErrorType.cancel || e.type == DioErrorType.other) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
continue;
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
debugPrint("ERROR backupAsset: ${e.toString()}");
|
debugPrint("ERROR backupAsset: ${e.toString()}");
|
||||||
continue;
|
continue;
|
||||||
|
|
|
@ -87,13 +87,16 @@ class BackupControllerPage extends HookConsumerWidget {
|
||||||
style: TextStyle(fontSize: 14),
|
style: TextStyle(fontSize: 14),
|
||||||
)
|
)
|
||||||
: Container(),
|
: Container(),
|
||||||
OutlinedButton(
|
Padding(
|
||||||
|
padding: const EdgeInsets.only(top: 8.0),
|
||||||
|
child: OutlinedButton(
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
isAutoBackup
|
isAutoBackup
|
||||||
? ref.watch(authenticationProvider.notifier).setAutoBackup(false)
|
? ref.watch(authenticationProvider.notifier).setAutoBackup(false)
|
||||||
: ref.watch(authenticationProvider.notifier).setAutoBackup(true);
|
: ref.watch(authenticationProvider.notifier).setAutoBackup(true);
|
||||||
},
|
},
|
||||||
child: Text("Turn $backupBtnText Backup"),
|
child: Text("Turn $backupBtnText Backup", style: const TextStyle(fontWeight: FontWeight.bold)),
|
||||||
|
),
|
||||||
)
|
)
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
|
|
|
@ -35,6 +35,9 @@ class FileHelper {
|
||||||
case 'dng':
|
case 'dng':
|
||||||
return {"type": "image", "subType": "dng"};
|
return {"type": "image", "subType": "dng"};
|
||||||
|
|
||||||
|
case 'webp':
|
||||||
|
return {"type": "image", "subType": "webp"};
|
||||||
|
|
||||||
default:
|
default:
|
||||||
return {"type": "unsupport", "subType": "unsupport"};
|
return {"type": "unsupport", "subType": "unsupport"};
|
||||||
}
|
}
|
||||||
|
|
|
@ -12,7 +12,7 @@ export const multerConfig = {
|
||||||
|
|
||||||
export const multerOption: MulterOptions = {
|
export const multerOption: MulterOptions = {
|
||||||
fileFilter: (req: Request, file: any, cb: any) => {
|
fileFilter: (req: Request, file: any, cb: any) => {
|
||||||
if (file.mimetype.match(/\/(jpg|jpeg|png|gif|mp4|x-msvideo|quicktime|heic|heif|dng)$/)) {
|
if (file.mimetype.match(/\/(jpg|jpeg|png|gif|mp4|x-msvideo|quicktime|heic|heif|dng|webp)$/)) {
|
||||||
cb(null, true);
|
cb(null, true);
|
||||||
} else {
|
} else {
|
||||||
cb(new HttpException(`Unsupported file type ${extname(file.originalname)}`, HttpStatus.BAD_REQUEST), false);
|
cb(new HttpException(`Unsupported file type ${extname(file.originalname)}`, HttpStatus.BAD_REQUEST), false);
|
||||||
|
|
Loading…
Reference in a new issue