2025-09-03 19:21:15 +03:00
|
|
|
import 'package:immich_mobile/domain/models/asset/base_asset.model.dart';
|
2025-08-06 11:54:18 +03:00
|
|
|
import 'package:immich_mobile/infrastructure/repositories/local_asset.repository.dart';
|
|
|
|
|
import 'package:immich_mobile/infrastructure/repositories/remote_asset.repository.dart';
|
|
|
|
|
import 'package:immich_mobile/infrastructure/repositories/storage.repository.dart';
|
|
|
|
|
import 'package:immich_mobile/repositories/local_files_manager.repository.dart';
|
|
|
|
|
import 'package:immich_mobile/services/app_settings.service.dart';
|
|
|
|
|
import 'package:logging/logging.dart';
|
|
|
|
|
import 'package:platform/platform.dart';
|
|
|
|
|
|
2025-08-28 12:43:38 +03:00
|
|
|
typedef TrashSyncItem = ({String checksum, DateTime? deletedAt});
|
|
|
|
|
|
2025-08-08 14:24:36 +03:00
|
|
|
class TrashSyncService {
|
2025-08-06 11:54:18 +03:00
|
|
|
final AppSettingsService _appSettingsService;
|
|
|
|
|
final RemoteAssetRepository _remoteAssetRepository;
|
|
|
|
|
final DriftLocalAssetRepository _localAssetRepository;
|
|
|
|
|
final LocalFilesManagerRepository _localFilesManager;
|
|
|
|
|
final StorageRepository _storageRepository;
|
|
|
|
|
final Platform _platform;
|
2025-08-06 18:20:51 +03:00
|
|
|
final Logger _logger = Logger('TrashService');
|
2025-08-06 11:54:18 +03:00
|
|
|
|
2025-08-08 14:24:36 +03:00
|
|
|
TrashSyncService({
|
2025-08-06 11:54:18 +03:00
|
|
|
required AppSettingsService appSettingsService,
|
|
|
|
|
required RemoteAssetRepository remoteAssetRepository,
|
|
|
|
|
required DriftLocalAssetRepository localAssetRepository,
|
|
|
|
|
required LocalFilesManagerRepository localFilesManager,
|
|
|
|
|
required StorageRepository storageRepository,
|
|
|
|
|
}) : _appSettingsService = appSettingsService,
|
|
|
|
|
_remoteAssetRepository = remoteAssetRepository,
|
|
|
|
|
_localAssetRepository = localAssetRepository,
|
|
|
|
|
_localFilesManager = localFilesManager,
|
|
|
|
|
_storageRepository = storageRepository,
|
2025-08-06 18:20:51 +03:00
|
|
|
_platform = const LocalPlatform();
|
2025-08-06 11:54:18 +03:00
|
|
|
|
2025-08-28 12:43:38 +03:00
|
|
|
Future<void> handleRemoteChanges(Iterable<TrashSyncItem> syncItems) async {
|
2025-08-06 18:20:51 +03:00
|
|
|
if (!_platform.isAndroid || !_appSettingsService.getSetting<bool>(AppSettingsEnum.manageLocalMediaAndroid)) {
|
|
|
|
|
return Future.value();
|
|
|
|
|
}
|
2025-09-05 17:41:30 +03:00
|
|
|
final trashedAssetsChecksums = <String>{};
|
|
|
|
|
final modifiedAssetsChecksums = <String>{};
|
2025-08-28 12:43:38 +03:00
|
|
|
for (var syncItem in syncItems) {
|
|
|
|
|
if (syncItem.deletedAt != null) {
|
|
|
|
|
trashedAssetsChecksums.add(syncItem.checksum);
|
|
|
|
|
} else {
|
|
|
|
|
modifiedAssetsChecksums.add(syncItem.checksum);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
await _applyRemoteTrashToLocal(trashedAssetsChecksums);
|
|
|
|
|
await _applyRemoteRestoreToLocal(modifiedAssetsChecksums);
|
2025-08-06 18:20:51 +03:00
|
|
|
}
|
2025-08-06 11:54:18 +03:00
|
|
|
|
2025-08-28 12:43:38 +03:00
|
|
|
Future<void> _applyRemoteTrashToLocal(Iterable<String> trashedAssetsChecksums) async {
|
|
|
|
|
if (trashedAssetsChecksums.isEmpty) {
|
|
|
|
|
return Future.value();
|
|
|
|
|
} else {
|
2025-09-05 17:41:30 +03:00
|
|
|
final candidatesToTrash = await _localAssetRepository.getByChecksums(trashedAssetsChecksums);
|
|
|
|
|
if (candidatesToTrash.isNotEmpty) {
|
|
|
|
|
final groupedByChecksum = <String, LocalAsset>{};
|
|
|
|
|
for (final localAsset in candidatesToTrash) {
|
|
|
|
|
groupedByChecksum[localAsset.checksum!] = localAsset;
|
|
|
|
|
}
|
2025-08-06 18:20:51 +03:00
|
|
|
final mediaUrls = await Future.wait(
|
2025-09-05 17:41:30 +03:00
|
|
|
groupedByChecksum.values.map(
|
2025-08-06 18:20:51 +03:00
|
|
|
(localAsset) => _storageRepository.getAssetEntityForAsset(localAsset).then((e) => e?.getMediaUrl()),
|
|
|
|
|
),
|
|
|
|
|
);
|
|
|
|
|
_logger.info("Moving to trash ${mediaUrls.join(", ")} assets");
|
|
|
|
|
await _localFilesManager.moveToTrash(mediaUrls.nonNulls.toList());
|
2025-09-05 17:41:30 +03:00
|
|
|
await _localAssetRepository.delete(candidatesToTrash.map((asset) => asset.id));
|
2025-08-06 11:54:18 +03:00
|
|
|
}
|
2025-08-06 18:20:51 +03:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-28 12:43:38 +03:00
|
|
|
Future<void> _applyRemoteRestoreToLocal(Iterable<String> modifiedAssetsChecksums) async {
|
|
|
|
|
if (modifiedAssetsChecksums.isEmpty) {
|
|
|
|
|
return Future.value();
|
|
|
|
|
} else {
|
2025-08-06 18:20:51 +03:00
|
|
|
final remoteAssetsToRestore = await _remoteAssetRepository.getByChecksums(
|
|
|
|
|
modifiedAssetsChecksums,
|
|
|
|
|
isTrashed: true,
|
|
|
|
|
);
|
|
|
|
|
if (remoteAssetsToRestore.isNotEmpty) {
|
|
|
|
|
_logger.info("Restoring from trash ${remoteAssetsToRestore.map((e) => e.name).join(", ")} assets");
|
2025-09-03 19:21:15 +03:00
|
|
|
for (RemoteAsset asset in remoteAssetsToRestore) {
|
|
|
|
|
await _localFilesManager.restoreFromTrash(asset.name, asset.type.index);
|
|
|
|
|
}
|
2025-08-06 11:54:18 +03:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|