mirror of
https://github.com/immich-app/immich.git
synced 2025-08-11 09:16:31 -04:00
* new upload button * wip * pr feedback * fix: updateAll override album selection value * feat: status box * feat: handle upload resume * re-enable websocket event * fix: update state condition and upload status * Better backup detail page --------- Co-authored-by: shenlong-tanwen <139912620+shalong-tanwen@users.noreply.github.com>
294 lines
8.2 KiB
Dart
294 lines
8.2 KiB
Dart
import 'dart:async';
|
|
import 'dart:convert';
|
|
import 'dart:io';
|
|
|
|
import 'package:background_downloader/background_downloader.dart';
|
|
import 'package:flutter/material.dart';
|
|
import 'package:immich_mobile/constants/constants.dart';
|
|
import 'package:immich_mobile/domain/models/asset/base_asset.model.dart';
|
|
import 'package:immich_mobile/infrastructure/repositories/backup.repository.dart';
|
|
import 'package:immich_mobile/infrastructure/repositories/local_asset.repository.dart';
|
|
import 'package:immich_mobile/infrastructure/repositories/storage.repository.dart';
|
|
import 'package:immich_mobile/providers/backup/drift_backup.provider.dart';
|
|
import 'package:immich_mobile/providers/infrastructure/asset.provider.dart';
|
|
import 'package:immich_mobile/providers/infrastructure/storage.provider.dart';
|
|
import 'package:immich_mobile/services/upload.service.dart';
|
|
import 'package:logging/logging.dart';
|
|
import 'package:path/path.dart' as p;
|
|
import 'package:riverpod_annotation/riverpod_annotation.dart';
|
|
|
|
final driftBackupServiceProvider = Provider<DriftBackupService>(
|
|
(ref) => DriftBackupService(
|
|
ref.watch(backupRepositoryProvider),
|
|
ref.watch(storageRepositoryProvider),
|
|
ref.watch(uploadServiceProvider),
|
|
ref.watch(localAssetRepository),
|
|
),
|
|
);
|
|
|
|
class DriftBackupService {
|
|
DriftBackupService(
|
|
this._backupRepository,
|
|
this._storageRepository,
|
|
this._uploadService,
|
|
this._localAssetRepository,
|
|
) {
|
|
_uploadService.taskStatusStream.listen(_handleTaskStatusUpdate);
|
|
}
|
|
|
|
final DriftBackupRepository _backupRepository;
|
|
final StorageRepository _storageRepository;
|
|
final DriftLocalAssetRepository _localAssetRepository;
|
|
final UploadService _uploadService;
|
|
final _log = Logger("DriftBackupService");
|
|
|
|
bool shouldCancel = false;
|
|
|
|
Future<int> getTotalCount() {
|
|
return _backupRepository.getTotalCount();
|
|
}
|
|
|
|
Future<int> getRemainderCount() {
|
|
return _backupRepository.getRemainderCount();
|
|
}
|
|
|
|
Future<int> getBackupCount() {
|
|
return _backupRepository.getBackupCount();
|
|
}
|
|
|
|
Future<void> backup(
|
|
void Function(EnqueueStatus status) onEnqueueTasks,
|
|
) async {
|
|
shouldCancel = false;
|
|
|
|
final candidates = await _backupRepository.getCandidates();
|
|
if (candidates.isEmpty) {
|
|
return;
|
|
}
|
|
|
|
const batchSize = 100;
|
|
int count = 0;
|
|
for (int i = 0; i < candidates.length; i += batchSize) {
|
|
if (shouldCancel) {
|
|
break;
|
|
}
|
|
|
|
final batch = candidates.skip(i).take(batchSize).toList();
|
|
|
|
List<UploadTask> tasks = [];
|
|
for (final asset in batch) {
|
|
final task = await _getUploadTask(asset);
|
|
if (task != null) {
|
|
tasks.add(task);
|
|
}
|
|
}
|
|
|
|
if (tasks.isNotEmpty && !shouldCancel) {
|
|
count += tasks.length;
|
|
_uploadService.enqueueTasks(tasks);
|
|
|
|
onEnqueueTasks(
|
|
EnqueueStatus(
|
|
enqueueCount: count,
|
|
totalCount: candidates.length,
|
|
),
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
void _handleTaskStatusUpdate(TaskStatusUpdate update) {
|
|
switch (update.status) {
|
|
case TaskStatus.complete:
|
|
_handleLivePhoto(update);
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
Future<void> _handleLivePhoto(TaskStatusUpdate update) async {
|
|
try {
|
|
if (update.task.metaData.isEmpty || update.task.metaData == '') {
|
|
return;
|
|
}
|
|
|
|
final metadata = UploadTaskMetadata.fromJson(update.task.metaData);
|
|
if (!metadata.isLivePhotos) {
|
|
return;
|
|
}
|
|
|
|
if (update.responseBody == null || update.responseBody!.isEmpty) {
|
|
return;
|
|
}
|
|
final response = jsonDecode(update.responseBody!);
|
|
|
|
final localAsset =
|
|
await _localAssetRepository.getById(metadata.localAssetId);
|
|
if (localAsset == null) {
|
|
return;
|
|
}
|
|
|
|
final uploadTask = await _getLivePhotoUploadTask(
|
|
localAsset,
|
|
response['id'] as String,
|
|
);
|
|
|
|
if (uploadTask == null) {
|
|
return;
|
|
}
|
|
|
|
_uploadService.enqueueTasks([uploadTask]);
|
|
} catch (error, stackTrace) {
|
|
_log.severe("Error handling live photo upload task", error, stackTrace);
|
|
debugPrint("Error handling live photo upload task: $error $stackTrace");
|
|
}
|
|
}
|
|
|
|
Future<UploadTask?> _getUploadTask(LocalAsset asset) async {
|
|
final entity = await _storageRepository.getAssetEntityForAsset(asset);
|
|
if (entity == null) {
|
|
return null;
|
|
}
|
|
|
|
File? file;
|
|
|
|
/// iOS LivePhoto has two files: a photo and a video.
|
|
/// They are uploaded separately, with video file being upload first, then returned with the assetId
|
|
/// The assetId is then used as a metadata for the photo file upload task.
|
|
///
|
|
/// We implement two separate upload groups for this, the normal one for the video file
|
|
/// and the higher priority group for the photo file because the video file is already uploaded.
|
|
///
|
|
/// The cancel operation will only cancel the video group (normal group), the photo group will not
|
|
/// be touched, as the video file is already uploaded.
|
|
|
|
if (entity.isLivePhoto) {
|
|
file = await _storageRepository.getMotionFileForAsset(asset);
|
|
} else {
|
|
file = await _storageRepository.getFileForAsset(asset.id);
|
|
}
|
|
|
|
if (file == null) {
|
|
return null;
|
|
}
|
|
|
|
final originalFileName = entity.isLivePhoto
|
|
? p.setExtension(
|
|
asset.name,
|
|
p.extension(file.path),
|
|
)
|
|
: asset.name;
|
|
|
|
String metadata = UploadTaskMetadata(
|
|
localAssetId: asset.id,
|
|
isLivePhotos: entity.isLivePhoto,
|
|
livePhotoVideoId: '',
|
|
).toJson();
|
|
|
|
return _uploadService.buildUploadTask(
|
|
file,
|
|
originalFileName: originalFileName,
|
|
deviceAssetId: asset.id,
|
|
metadata: metadata,
|
|
group: kBackupGroup,
|
|
);
|
|
}
|
|
|
|
Future<UploadTask?> _getLivePhotoUploadTask(
|
|
LocalAsset asset,
|
|
String livePhotoVideoId,
|
|
) async {
|
|
final entity = await _storageRepository.getAssetEntityForAsset(asset);
|
|
if (entity == null) {
|
|
return null;
|
|
}
|
|
|
|
final file = await _storageRepository.getFileForAsset(asset.id);
|
|
if (file == null) {
|
|
return null;
|
|
}
|
|
|
|
final fields = {
|
|
'livePhotoVideoId': livePhotoVideoId,
|
|
};
|
|
|
|
return _uploadService.buildUploadTask(
|
|
file,
|
|
originalFileName: asset.name,
|
|
deviceAssetId: asset.id,
|
|
fields: fields,
|
|
group: kBackupLivePhotoGroup,
|
|
priority: 0, // Highest priority to get upload immediately
|
|
);
|
|
}
|
|
|
|
Future<void> cancel() async {
|
|
shouldCancel = true;
|
|
await _uploadService.cancelAllForGroup(kBackupGroup);
|
|
}
|
|
}
|
|
|
|
class UploadTaskMetadata {
|
|
final String localAssetId;
|
|
final bool isLivePhotos;
|
|
final String livePhotoVideoId;
|
|
|
|
const UploadTaskMetadata({
|
|
required this.localAssetId,
|
|
required this.isLivePhotos,
|
|
required this.livePhotoVideoId,
|
|
});
|
|
|
|
UploadTaskMetadata copyWith({
|
|
String? localAssetId,
|
|
bool? isLivePhotos,
|
|
String? livePhotoVideoId,
|
|
}) {
|
|
return UploadTaskMetadata(
|
|
localAssetId: localAssetId ?? this.localAssetId,
|
|
isLivePhotos: isLivePhotos ?? this.isLivePhotos,
|
|
livePhotoVideoId: livePhotoVideoId ?? this.livePhotoVideoId,
|
|
);
|
|
}
|
|
|
|
Map<String, dynamic> toMap() {
|
|
return <String, dynamic>{
|
|
'localAssetId': localAssetId,
|
|
'isLivePhotos': isLivePhotos,
|
|
'livePhotoVideoId': livePhotoVideoId,
|
|
};
|
|
}
|
|
|
|
factory UploadTaskMetadata.fromMap(Map<String, dynamic> map) {
|
|
return UploadTaskMetadata(
|
|
localAssetId: map['localAssetId'] as String,
|
|
isLivePhotos: map['isLivePhotos'] as bool,
|
|
livePhotoVideoId: map['livePhotoVideoId'] as String,
|
|
);
|
|
}
|
|
|
|
String toJson() => json.encode(toMap());
|
|
|
|
factory UploadTaskMetadata.fromJson(String source) =>
|
|
UploadTaskMetadata.fromMap(json.decode(source) as Map<String, dynamic>);
|
|
|
|
@override
|
|
String toString() =>
|
|
'UploadTaskMetadata(localAssetId: $localAssetId, isLivePhotos: $isLivePhotos, livePhotoVideoId: $livePhotoVideoId)';
|
|
|
|
@override
|
|
bool operator ==(covariant UploadTaskMetadata other) {
|
|
if (identical(this, other)) return true;
|
|
|
|
return other.localAssetId == localAssetId &&
|
|
other.isLivePhotos == isLivePhotos &&
|
|
other.livePhotoVideoId == livePhotoVideoId;
|
|
}
|
|
|
|
@override
|
|
int get hashCode =>
|
|
localAssetId.hashCode ^ isLivePhotos.hashCode ^ livePhotoVideoId.hashCode;
|
|
}
|