mirror of
https://github.com/immich-app/immich
synced 2025-11-07 17:27:20 +00:00
refactor code (use separate TrashService)
This commit is contained in:
parent
a89a35beed
commit
31bfadb585
8 changed files with 102 additions and 80 deletions
|
|
@ -2,36 +2,20 @@ import 'package:immich_mobile/domain/models/asset/base_asset.model.dart';
|
|||
import 'package:immich_mobile/domain/models/exif.model.dart';
|
||||
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/infrastructure/utils/exif.converter.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';
|
||||
|
||||
class AssetService {
|
||||
final AppSettingsService _appSettingsService;
|
||||
final RemoteAssetRepository _remoteAssetRepository;
|
||||
final DriftLocalAssetRepository _localAssetRepository;
|
||||
final LocalFilesManagerRepository _localFilesManager;
|
||||
final StorageRepository _storageRepository;
|
||||
final Platform _platform;
|
||||
final Logger _logger;
|
||||
|
||||
const AssetService({
|
||||
required AppSettingsService appSettingsService,
|
||||
required RemoteAssetRepository remoteAssetRepository,
|
||||
required DriftLocalAssetRepository localAssetRepository,
|
||||
required LocalFilesManagerRepository localFilesManager,
|
||||
required StorageRepository storageRepository,
|
||||
required Logger logger,
|
||||
}) : _appSettingsService = appSettingsService,
|
||||
_remoteAssetRepository = remoteAssetRepository,
|
||||
}) : _remoteAssetRepository = remoteAssetRepository,
|
||||
_localAssetRepository = localAssetRepository,
|
||||
_localFilesManager = localFilesManager,
|
||||
_storageRepository = storageRepository,
|
||||
_platform = const LocalPlatform(),
|
||||
_logger = logger;
|
||||
_platform = const LocalPlatform();
|
||||
|
||||
Stream<BaseAsset?> watchAsset(BaseAsset asset) {
|
||||
final id = asset is LocalAsset ? asset.id : (asset as RemoteAsset).id;
|
||||
|
|
@ -100,40 +84,4 @@ class AssetService {
|
|||
Future<int> getLocalHashedCount() {
|
||||
return _localAssetRepository.getHashedCount();
|
||||
}
|
||||
|
||||
Future<void> handleRemoteTrashChanges(Iterable<({String checksum, DateTime? deletedAt})> syncItems) async {
|
||||
if (_platform.isAndroid && _appSettingsService.getSetting<bool>(AppSettingsEnum.manageLocalMediaAndroid)) {
|
||||
final trashedItems = syncItems.where((item) => item.deletedAt != null);
|
||||
|
||||
if (trashedItems.isNotEmpty) {
|
||||
final trashedAssetsChecksums = trashedItems.map((syncItem) => syncItem.checksum);
|
||||
final localAssetsToTrash = await _localAssetRepository.getByChecksums(trashedAssetsChecksums);
|
||||
if (localAssetsToTrash.isNotEmpty) {
|
||||
final mediaUrls = await Future.wait(
|
||||
localAssetsToTrash.map(
|
||||
(localAsset) => _storageRepository.getAssetEntityForAsset(localAsset).then((e) => e?.getMediaUrl()),
|
||||
),
|
||||
);
|
||||
_logger.fine("Moving to trash ${mediaUrls.join(", ")} assets");
|
||||
await _localFilesManager.moveToTrash(mediaUrls.nonNulls.toList());
|
||||
}
|
||||
}
|
||||
final modifiedItems = syncItems.where((e) => e.deletedAt == null);
|
||||
if (modifiedItems.isNotEmpty) {
|
||||
final modifiedChecksums = modifiedItems.map((syncItem) => syncItem.checksum);
|
||||
final remoteAssetsToRestore = await _remoteAssetRepository.getByChecksums(
|
||||
modifiedChecksums,
|
||||
isTrashed: true,
|
||||
);
|
||||
if (remoteAssetsToRestore.isNotEmpty) {
|
||||
_logger.fine("Restoring from trash ${remoteAssetsToRestore.map((e) => e.name).join(", ")} assets");
|
||||
for (final remoteAsset in remoteAssetsToRestore) {
|
||||
await _localFilesManager.restoreFromTrash(remoteAsset.name, remoteAsset.type.index);
|
||||
}
|
||||
}
|
||||
}
|
||||
} else {
|
||||
return Future.value();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
|||
|
|
@ -1,7 +1,7 @@
|
|||
import 'dart:async';
|
||||
|
||||
import 'package:immich_mobile/domain/models/sync_event.model.dart';
|
||||
import 'package:immich_mobile/domain/services/asset.service.dart';
|
||||
import 'package:immich_mobile/domain/services/trash.service.dart';
|
||||
import 'package:immich_mobile/infrastructure/repositories/sync_api.repository.dart';
|
||||
import 'package:immich_mobile/infrastructure/repositories/sync_stream.repository.dart';
|
||||
import 'package:immich_mobile/presentation/pages/dev/dev_logger.dart';
|
||||
|
|
@ -13,17 +13,17 @@ class SyncStreamService {
|
|||
|
||||
final SyncApiRepository _syncApiRepository;
|
||||
final SyncStreamRepository _syncStreamRepository;
|
||||
final AssetService _assetService;
|
||||
final TrashService _trashService;
|
||||
final bool Function()? _cancelChecker;
|
||||
|
||||
SyncStreamService({
|
||||
required SyncApiRepository syncApiRepository,
|
||||
required SyncStreamRepository syncStreamRepository,
|
||||
required AssetService assetService,
|
||||
required TrashService trashService,
|
||||
bool Function()? cancelChecker,
|
||||
}) : _syncApiRepository = syncApiRepository,
|
||||
_syncStreamRepository = syncStreamRepository,
|
||||
_assetService = assetService,
|
||||
_trashService = trashService,
|
||||
_cancelChecker = cancelChecker;
|
||||
|
||||
bool get isCancelled => _cancelChecker?.call() ?? false;
|
||||
|
|
@ -119,7 +119,7 @@ class SyncStreamService {
|
|||
return _syncStreamRepository.deletePartnerV1(data.cast());
|
||||
case SyncEntityType.assetV1:
|
||||
final remoteSyncAssets = data.cast<SyncAssetV1>();
|
||||
await _assetService.handleRemoteTrashChanges(
|
||||
await _trashService.handleRemoteChanges(
|
||||
remoteSyncAssets.map((e) => (checksum: e.checksum, deletedAt: e.deletedAt)),
|
||||
);
|
||||
return _syncStreamRepository.updateAssetsV1(remoteSyncAssets);
|
||||
|
|
|
|||
65
mobile/lib/domain/services/trash.service.dart
Normal file
65
mobile/lib/domain/services/trash.service.dart
Normal file
|
|
@ -0,0 +1,65 @@
|
|||
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';
|
||||
|
||||
class TrashService {
|
||||
final AppSettingsService _appSettingsService;
|
||||
final RemoteAssetRepository _remoteAssetRepository;
|
||||
final DriftLocalAssetRepository _localAssetRepository;
|
||||
final LocalFilesManagerRepository _localFilesManager;
|
||||
final StorageRepository _storageRepository;
|
||||
final Platform _platform;
|
||||
final Logger _logger;
|
||||
|
||||
const TrashService({
|
||||
required AppSettingsService appSettingsService,
|
||||
required RemoteAssetRepository remoteAssetRepository,
|
||||
required DriftLocalAssetRepository localAssetRepository,
|
||||
required LocalFilesManagerRepository localFilesManager,
|
||||
required StorageRepository storageRepository,
|
||||
required Logger logger,
|
||||
}) : _appSettingsService = appSettingsService,
|
||||
_remoteAssetRepository = remoteAssetRepository,
|
||||
_localAssetRepository = localAssetRepository,
|
||||
_localFilesManager = localFilesManager,
|
||||
_storageRepository = storageRepository,
|
||||
_platform = const LocalPlatform(),
|
||||
_logger = logger;
|
||||
|
||||
Future<void> handleRemoteChanges(Iterable<({String checksum, DateTime? deletedAt})> syncItems) async {
|
||||
if (_platform.isAndroid && _appSettingsService.getSetting<bool>(AppSettingsEnum.manageLocalMediaAndroid)) {
|
||||
final trashedItems = syncItems.where((item) => item.deletedAt != null);
|
||||
|
||||
if (trashedItems.isNotEmpty) {
|
||||
final trashedAssetsChecksums = trashedItems.map((syncItem) => syncItem.checksum);
|
||||
final localAssetsToTrash = await _localAssetRepository.getByChecksums(trashedAssetsChecksums);
|
||||
if (localAssetsToTrash.isNotEmpty) {
|
||||
final mediaUrls = await Future.wait(
|
||||
localAssetsToTrash.map(
|
||||
(localAsset) => _storageRepository.getAssetEntityForAsset(localAsset).then((e) => e?.getMediaUrl()),
|
||||
),
|
||||
);
|
||||
_logger.info("Moving to trash ${mediaUrls.join(", ")} assets");
|
||||
await _localFilesManager.moveToTrash(mediaUrls.nonNulls.toList());
|
||||
}
|
||||
}
|
||||
final modifiedItems = syncItems.where((e) => e.deletedAt == null);
|
||||
if (modifiedItems.isNotEmpty) {
|
||||
final modifiedChecksums = modifiedItems.map((syncItem) => syncItem.checksum);
|
||||
final remoteAssetsToRestore = await _remoteAssetRepository.getByChecksums(modifiedChecksums, isTrashed: true);
|
||||
if (remoteAssetsToRestore.isNotEmpty) {
|
||||
_logger.info("Restoring from trash ${remoteAssetsToRestore.map((e) => e.name).join(", ")} assets");
|
||||
for (final remoteAsset in remoteAssetsToRestore) {
|
||||
await _localFilesManager.restoreFromTrash(remoteAsset.name, remoteAsset.type.index);
|
||||
}
|
||||
}
|
||||
}
|
||||
} else {
|
||||
return Future.value();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
@ -2,11 +2,7 @@ import 'package:hooks_riverpod/hooks_riverpod.dart';
|
|||
import 'package:immich_mobile/domain/services/asset.service.dart';
|
||||
import 'package:immich_mobile/infrastructure/repositories/local_asset.repository.dart';
|
||||
import 'package:immich_mobile/infrastructure/repositories/remote_asset.repository.dart';
|
||||
import 'package:immich_mobile/providers/app_settings.provider.dart';
|
||||
import 'package:immich_mobile/providers/infrastructure/db.provider.dart';
|
||||
import 'package:immich_mobile/providers/infrastructure/storage.provider.dart';
|
||||
import 'package:immich_mobile/repositories/local_files_manager.repository.dart';
|
||||
import 'package:logging/logging.dart';
|
||||
|
||||
final localAssetRepository = Provider<DriftLocalAssetRepository>(
|
||||
(ref) => DriftLocalAssetRepository(ref.watch(driftProvider)),
|
||||
|
|
@ -18,22 +14,14 @@ final remoteAssetRepositoryProvider = Provider<RemoteAssetRepository>(
|
|||
|
||||
final assetServiceProvider = Provider(
|
||||
(ref) => AssetService(
|
||||
appSettingsService: ref.watch(appSettingsServiceProvider),
|
||||
remoteAssetRepository: ref.watch(remoteAssetRepositoryProvider),
|
||||
localAssetRepository: ref.watch(localAssetRepository),
|
||||
localFilesManager: ref.watch(localFilesManagerRepositoryProvider),
|
||||
storageRepository: ref.watch(storageRepositoryProvider),
|
||||
logger: Logger('AssetService'),
|
||||
),
|
||||
);
|
||||
|
||||
final placesProvider = FutureProvider<List<(String, String)>>(
|
||||
(ref) => AssetService(
|
||||
appSettingsService: ref.watch(appSettingsServiceProvider),
|
||||
remoteAssetRepository: ref.watch(remoteAssetRepositoryProvider),
|
||||
localAssetRepository: ref.watch(localAssetRepository),
|
||||
localFilesManager: ref.watch(localFilesManagerRepositoryProvider),
|
||||
storageRepository: ref.watch(storageRepositoryProvider),
|
||||
logger: Logger('AssetService'),
|
||||
).getPlaces(),
|
||||
);
|
||||
|
|
|
|||
|
|
@ -11,12 +11,13 @@ import 'package:immich_mobile/providers/infrastructure/cancel.provider.dart';
|
|||
import 'package:immich_mobile/providers/infrastructure/db.provider.dart';
|
||||
import 'package:immich_mobile/providers/infrastructure/platform.provider.dart';
|
||||
import 'package:immich_mobile/providers/infrastructure/storage.provider.dart';
|
||||
import 'package:immich_mobile/providers/infrastructure/trash.provider.dart';
|
||||
|
||||
final syncStreamServiceProvider = Provider(
|
||||
(ref) => SyncStreamService(
|
||||
syncApiRepository: ref.watch(syncApiRepositoryProvider),
|
||||
syncStreamRepository: ref.watch(syncStreamRepositoryProvider),
|
||||
assetService: ref.watch(assetServiceProvider),
|
||||
trashService: ref.watch(trashServiceProvider),
|
||||
cancelChecker: ref.watch(cancellationProvider),
|
||||
),
|
||||
);
|
||||
|
|
|
|||
20
mobile/lib/providers/infrastructure/trash.provider.dart
Normal file
20
mobile/lib/providers/infrastructure/trash.provider.dart
Normal file
|
|
@ -0,0 +1,20 @@
|
|||
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
||||
import 'package:immich_mobile/domain/services/trash.service.dart';
|
||||
import 'package:immich_mobile/providers/app_settings.provider.dart';
|
||||
import 'package:immich_mobile/providers/infrastructure/storage.provider.dart';
|
||||
import 'package:immich_mobile/repositories/local_files_manager.repository.dart';
|
||||
import 'package:logging/logging.dart';
|
||||
|
||||
import 'asset.provider.dart';
|
||||
|
||||
final trashServiceProvider = Provider(
|
||||
(ref) => TrashService(
|
||||
appSettingsService: ref.watch(appSettingsServiceProvider),
|
||||
remoteAssetRepository: ref.watch(remoteAssetRepositoryProvider),
|
||||
localAssetRepository: ref.watch(localAssetRepository),
|
||||
localFilesManager: ref.watch(localFilesManagerRepositoryProvider),
|
||||
storageRepository: ref.watch(storageRepositoryProvider),
|
||||
logger: Logger('TrashService'),
|
||||
),
|
||||
);
|
||||
|
||||
|
|
@ -1,5 +1,5 @@
|
|||
import 'package:immich_mobile/domain/services/asset.service.dart';
|
||||
import 'package:immich_mobile/domain/services/store.service.dart';
|
||||
import 'package:immich_mobile/domain/services/trash.service.dart';
|
||||
import 'package:immich_mobile/domain/services/user.service.dart';
|
||||
import 'package:immich_mobile/domain/utils/background_sync.dart';
|
||||
import 'package:immich_mobile/platform/native_sync_api.g.dart';
|
||||
|
|
@ -19,4 +19,4 @@ class MockAppSettingsService extends Mock implements AppSettingsService {}
|
|||
|
||||
class MockUploadService extends Mock implements UploadService {}
|
||||
|
||||
class MockAssetService extends Mock implements AssetService {}
|
||||
class MockTrashService extends Mock implements TrashService {}
|
||||
|
|
|
|||
|
|
@ -2,8 +2,8 @@ import 'dart:async';
|
|||
|
||||
import 'package:flutter_test/flutter_test.dart';
|
||||
import 'package:immich_mobile/domain/models/sync_event.model.dart';
|
||||
import 'package:immich_mobile/domain/services/asset.service.dart';
|
||||
import 'package:immich_mobile/domain/services/sync_stream.service.dart';
|
||||
import 'package:immich_mobile/domain/services/trash.service.dart';
|
||||
import 'package:immich_mobile/infrastructure/repositories/sync_api.repository.dart';
|
||||
import 'package:immich_mobile/infrastructure/repositories/sync_stream.repository.dart';
|
||||
import 'package:mocktail/mocktail.dart';
|
||||
|
|
@ -32,7 +32,7 @@ void main() {
|
|||
late SyncStreamService sut;
|
||||
late SyncStreamRepository mockSyncStreamRepo;
|
||||
late SyncApiRepository mockSyncApiRepo;
|
||||
late AssetService mockAssetService;
|
||||
late TrashService mockTrashService;
|
||||
late Function(List<SyncEvent>, Function()) handleEventsCallback;
|
||||
late _MockAbortCallbackWrapper mockAbortCallbackWrapper;
|
||||
|
||||
|
|
@ -42,7 +42,7 @@ void main() {
|
|||
mockSyncStreamRepo = MockSyncStreamRepository();
|
||||
mockSyncApiRepo = MockSyncApiRepository();
|
||||
mockAbortCallbackWrapper = _MockAbortCallbackWrapper();
|
||||
mockAssetService = MockAssetService();
|
||||
mockTrashService = MockTrashService();
|
||||
when(() => mockAbortCallbackWrapper()).thenReturn(false);
|
||||
|
||||
when(() => mockSyncApiRepo.streamChanges(any())).thenAnswer((invocation) async {
|
||||
|
|
@ -87,7 +87,7 @@ void main() {
|
|||
sut = SyncStreamService(
|
||||
syncApiRepository: mockSyncApiRepo,
|
||||
syncStreamRepository: mockSyncStreamRepo,
|
||||
assetService: mockAssetService,
|
||||
trashService: mockTrashService,
|
||||
);
|
||||
});
|
||||
|
||||
|
|
@ -153,7 +153,7 @@ void main() {
|
|||
sut = SyncStreamService(
|
||||
syncApiRepository: mockSyncApiRepo,
|
||||
syncStreamRepository: mockSyncStreamRepo,
|
||||
assetService: mockAssetService,
|
||||
trashService: mockTrashService,
|
||||
cancelChecker: cancellationChecker.call,
|
||||
);
|
||||
await sut.sync();
|
||||
|
|
@ -189,7 +189,7 @@ void main() {
|
|||
sut = SyncStreamService(
|
||||
syncApiRepository: mockSyncApiRepo,
|
||||
syncStreamRepository: mockSyncStreamRepo,
|
||||
assetService: mockAssetService,
|
||||
trashService: mockTrashService,
|
||||
cancelChecker: cancellationChecker.call,
|
||||
);
|
||||
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue