mirror of
https://github.com/immich-app/immich
synced 2025-11-14 17:36:12 +00:00
feature(mobile, beta, Android): handle remote asset trash/restore events and rescan media
- Handle move to trash and restore from trash for remote assets on Android - Trigger MediaScannerConnection to rescan affected media files
This commit is contained in:
parent
f972b8d514
commit
cf920ea438
9 changed files with 140 additions and 6 deletions
|
|
@ -5,6 +5,7 @@ import android.content.ContentResolver
|
||||||
import android.content.ContentUris
|
import android.content.ContentUris
|
||||||
import android.content.Context
|
import android.content.Context
|
||||||
import android.content.Intent
|
import android.content.Intent
|
||||||
|
import android.media.MediaScannerConnection
|
||||||
import android.net.Uri
|
import android.net.Uri
|
||||||
import android.os.Build
|
import android.os.Build
|
||||||
import android.os.Bundle
|
import android.os.Bundle
|
||||||
|
|
@ -37,6 +38,7 @@ class BackgroundServicePlugin : FlutterPlugin, MethodChannel.MethodCallHandler,
|
||||||
private val permissionRequestCode = 1001
|
private val permissionRequestCode = 1001
|
||||||
private val trashRequestCode = 1002
|
private val trashRequestCode = 1002
|
||||||
private var activityBinding: ActivityPluginBinding? = null
|
private var activityBinding: ActivityPluginBinding? = null
|
||||||
|
private var lastToggledUris: List<Uri>? = null
|
||||||
|
|
||||||
override fun onAttachedToEngine(binding: FlutterPlugin.FlutterPluginBinding) {
|
override fun onAttachedToEngine(binding: FlutterPlugin.FlutterPluginBinding) {
|
||||||
onAttachedToEngine(binding.applicationContext, binding.binaryMessenger)
|
onAttachedToEngine(binding.applicationContext, binding.binaryMessenger)
|
||||||
|
|
@ -231,7 +233,7 @@ class BackgroundServicePlugin : FlutterPlugin, MethodChannel.MethodCallHandler,
|
||||||
result.error("TrashError", "Activity or ContentResolver not available", null)
|
result.error("TrashError", "Activity or ContentResolver not available", null)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
lastToggledUris = contentUris
|
||||||
try {
|
try {
|
||||||
val pendingIntent = MediaStore.createTrashRequest(contentResolver, contentUris, isTrashed)
|
val pendingIntent = MediaStore.createTrashRequest(contentResolver, contentUris, isTrashed)
|
||||||
pendingResult = result // Store for onActivityResult
|
pendingResult = result // Store for onActivityResult
|
||||||
|
|
@ -309,6 +311,35 @@ class BackgroundServicePlugin : FlutterPlugin, MethodChannel.MethodCallHandler,
|
||||||
|
|
||||||
if (requestCode == trashRequestCode) {
|
if (requestCode == trashRequestCode) {
|
||||||
val approved = resultCode == Activity.RESULT_OK
|
val approved = resultCode == Activity.RESULT_OK
|
||||||
|
if (approved) {
|
||||||
|
lastToggledUris?.forEach { uri ->
|
||||||
|
val projection = arrayOf(MediaStore.MediaColumns.DATA)
|
||||||
|
try {
|
||||||
|
val cursor = context?.contentResolver?.query(uri, projection, null, null, null)
|
||||||
|
if (cursor == null) {
|
||||||
|
Log.w(TAG, "Cursor is null for URI: $uri")
|
||||||
|
return@forEach
|
||||||
|
}
|
||||||
|
|
||||||
|
cursor.use {
|
||||||
|
if (it.moveToFirst()) {
|
||||||
|
val path = it.getStringOrNull(it.getColumnIndex(MediaStore.MediaColumns.DATA))
|
||||||
|
if (!path.isNullOrBlank()) {
|
||||||
|
Log.i(TAG, "Scanning updated file: $path")
|
||||||
|
MediaScannerConnection.scanFile(context, arrayOf(path), null, null)
|
||||||
|
} else {
|
||||||
|
Log.w(TAG, "Path is null or blank for URI: $uri")
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
Log.w(TAG, "Cursor is empty for URI: $uri")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (e: Exception) {
|
||||||
|
Log.e(TAG, "Error during rescan for URI: $uri", e)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
lastToggledUris = null
|
||||||
pendingResult?.success(approved)
|
pendingResult?.success(approved)
|
||||||
pendingResult = null
|
pendingResult = null
|
||||||
return true
|
return true
|
||||||
|
|
|
||||||
|
|
@ -3,19 +3,36 @@ 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/local_asset.repository.dart';
|
||||||
import 'package:immich_mobile/infrastructure/repositories/remote_asset.repository.dart';
|
import 'package:immich_mobile/infrastructure/repositories/remote_asset.repository.dart';
|
||||||
import 'package:immich_mobile/infrastructure/utils/exif.converter.dart';
|
import 'package:immich_mobile/infrastructure/utils/exif.converter.dart';
|
||||||
|
import 'package:immich_mobile/services/app_settings.service.dart';
|
||||||
|
import 'package:logging/logging.dart';
|
||||||
import 'package:platform/platform.dart';
|
import 'package:platform/platform.dart';
|
||||||
|
|
||||||
|
import '../../infrastructure/repositories/storage.repository.dart';
|
||||||
|
import '../../repositories/local_files_manager.repository.dart';
|
||||||
|
|
||||||
class AssetService {
|
class AssetService {
|
||||||
|
final AppSettingsService _appSettingsService;
|
||||||
final RemoteAssetRepository _remoteAssetRepository;
|
final RemoteAssetRepository _remoteAssetRepository;
|
||||||
final DriftLocalAssetRepository _localAssetRepository;
|
final DriftLocalAssetRepository _localAssetRepository;
|
||||||
|
final LocalFilesManagerRepository _localFilesManager;
|
||||||
|
final StorageRepository _storageRepository;
|
||||||
final Platform _platform;
|
final Platform _platform;
|
||||||
|
final Logger _logger;
|
||||||
|
|
||||||
const AssetService({
|
const AssetService({
|
||||||
|
required AppSettingsService appSettingsService,
|
||||||
required RemoteAssetRepository remoteAssetRepository,
|
required RemoteAssetRepository remoteAssetRepository,
|
||||||
required DriftLocalAssetRepository localAssetRepository,
|
required DriftLocalAssetRepository localAssetRepository,
|
||||||
}) : _remoteAssetRepository = remoteAssetRepository,
|
required LocalFilesManagerRepository localFilesManager,
|
||||||
|
required StorageRepository storageRepository,
|
||||||
|
required Logger logger,
|
||||||
|
}) : _appSettingsService = appSettingsService,
|
||||||
|
_remoteAssetRepository = remoteAssetRepository,
|
||||||
_localAssetRepository = localAssetRepository,
|
_localAssetRepository = localAssetRepository,
|
||||||
_platform = const LocalPlatform();
|
_localFilesManager = localFilesManager,
|
||||||
|
_storageRepository = storageRepository,
|
||||||
|
_platform = const LocalPlatform(),
|
||||||
|
_logger = logger;
|
||||||
|
|
||||||
Stream<BaseAsset?> watchAsset(BaseAsset asset) {
|
Stream<BaseAsset?> watchAsset(BaseAsset asset) {
|
||||||
final id = asset is LocalAsset ? asset.id : (asset as RemoteAsset).id;
|
final id = asset is LocalAsset ? asset.id : (asset as RemoteAsset).id;
|
||||||
|
|
@ -84,4 +101,34 @@ class AssetService {
|
||||||
Future<int> getLocalHashedCount() {
|
Future<int> getLocalHashedCount() {
|
||||||
return _localAssetRepository.getHashedCount();
|
return _localAssetRepository.getHashedCount();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Future<void> handleRemoteTrashChanges(Iterable<({String checksum, DateTime? deletedAt})> syncData) async {
|
||||||
|
if (_platform.isAndroid && _appSettingsService.getSetting<bool>(AppSettingsEnum.manageLocalMediaAndroid)) {}
|
||||||
|
final trashedItems = syncData.where((item) => item.deletedAt != null);
|
||||||
|
|
||||||
|
if (trashedItems.isNotEmpty) {
|
||||||
|
final trashedAssetsChecksums = trashedItems.map((syncAsset) => syncAsset.checksum);
|
||||||
|
final localAssetsToTrash = await _localAssetRepository.getAssetsByChecksums(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 = syncData.where((e) => e.deletedAt == null);
|
||||||
|
if (modifiedItems.isNotEmpty) {
|
||||||
|
final modifiedChecksums = modifiedItems.map((syncAsset) => syncAsset.checksum);
|
||||||
|
final remoteAssetsToRestore = await _remoteAssetRepository.getAssetsByChecksums(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);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -1,6 +1,7 @@
|
||||||
import 'dart:async';
|
import 'dart:async';
|
||||||
|
|
||||||
import 'package:immich_mobile/domain/models/sync_event.model.dart';
|
import 'package:immich_mobile/domain/models/sync_event.model.dart';
|
||||||
|
import 'package:immich_mobile/domain/services/asset.service.dart';
|
||||||
import 'package:immich_mobile/infrastructure/repositories/sync_api.repository.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/infrastructure/repositories/sync_stream.repository.dart';
|
||||||
import 'package:immich_mobile/presentation/pages/dev/dev_logger.dart';
|
import 'package:immich_mobile/presentation/pages/dev/dev_logger.dart';
|
||||||
|
|
@ -12,14 +13,17 @@ class SyncStreamService {
|
||||||
|
|
||||||
final SyncApiRepository _syncApiRepository;
|
final SyncApiRepository _syncApiRepository;
|
||||||
final SyncStreamRepository _syncStreamRepository;
|
final SyncStreamRepository _syncStreamRepository;
|
||||||
|
final AssetService _assetService;
|
||||||
final bool Function()? _cancelChecker;
|
final bool Function()? _cancelChecker;
|
||||||
|
|
||||||
SyncStreamService({
|
SyncStreamService({
|
||||||
required SyncApiRepository syncApiRepository,
|
required SyncApiRepository syncApiRepository,
|
||||||
required SyncStreamRepository syncStreamRepository,
|
required SyncStreamRepository syncStreamRepository,
|
||||||
|
required AssetService assetService,
|
||||||
bool Function()? cancelChecker,
|
bool Function()? cancelChecker,
|
||||||
}) : _syncApiRepository = syncApiRepository,
|
}) : _syncApiRepository = syncApiRepository,
|
||||||
_syncStreamRepository = syncStreamRepository,
|
_syncStreamRepository = syncStreamRepository,
|
||||||
|
_assetService = assetService,
|
||||||
_cancelChecker = cancelChecker;
|
_cancelChecker = cancelChecker;
|
||||||
|
|
||||||
bool get isCancelled => _cancelChecker?.call() ?? false;
|
bool get isCancelled => _cancelChecker?.call() ?? false;
|
||||||
|
|
@ -114,7 +118,11 @@ class SyncStreamService {
|
||||||
case SyncEntityType.partnerDeleteV1:
|
case SyncEntityType.partnerDeleteV1:
|
||||||
return _syncStreamRepository.deletePartnerV1(data.cast());
|
return _syncStreamRepository.deletePartnerV1(data.cast());
|
||||||
case SyncEntityType.assetV1:
|
case SyncEntityType.assetV1:
|
||||||
return _syncStreamRepository.updateAssetsV1(data.cast());
|
final remoteSyncAssets = data.cast<SyncAssetV1>();
|
||||||
|
await _assetService.handleRemoteTrashChanges(
|
||||||
|
remoteSyncAssets.map((e) => (checksum: e.checksum, deletedAt: e.deletedAt)),
|
||||||
|
);
|
||||||
|
return _syncStreamRepository.updateAssetsV1(remoteSyncAssets);
|
||||||
case SyncEntityType.assetDeleteV1:
|
case SyncEntityType.assetDeleteV1:
|
||||||
return _syncStreamRepository.deleteAssetsV1(data.cast());
|
return _syncStreamRepository.deleteAssetsV1(data.cast());
|
||||||
case SyncEntityType.assetExifV1:
|
case SyncEntityType.assetExifV1:
|
||||||
|
|
|
||||||
|
|
@ -65,4 +65,10 @@ class DriftLocalAssetRepository extends DriftDatabaseRepository {
|
||||||
Future<int> getHashedCount() {
|
Future<int> getHashedCount() {
|
||||||
return _db.managers.localAssetEntity.filter((e) => e.checksum.isNull().not()).count();
|
return _db.managers.localAssetEntity.filter((e) => e.checksum.isNull().not()).count();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Future<List<LocalAsset>> getAssetsByChecksums(Iterable<String> checksums) {
|
||||||
|
if (checksums.isEmpty) return Future.value([]);
|
||||||
|
final query = _db.localAssetEntity.select()..where((lae) => lae.checksum.isIn(checksums));
|
||||||
|
return query.map((row) => row.toDto()).get();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -12,6 +12,7 @@ import 'package:maplibre_gl/maplibre_gl.dart';
|
||||||
|
|
||||||
class RemoteAssetRepository extends DriftDatabaseRepository {
|
class RemoteAssetRepository extends DriftDatabaseRepository {
|
||||||
final Drift _db;
|
final Drift _db;
|
||||||
|
|
||||||
const RemoteAssetRepository(this._db) : super(_db);
|
const RemoteAssetRepository(this._db) : super(_db);
|
||||||
|
|
||||||
/// For testing purposes
|
/// For testing purposes
|
||||||
|
|
@ -252,4 +253,23 @@ class RemoteAssetRepository extends DriftDatabaseRepository {
|
||||||
Future<int> getCount() {
|
Future<int> getCount() {
|
||||||
return _db.managers.remoteAssetEntity.count();
|
return _db.managers.remoteAssetEntity.count();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Future<List<RemoteAsset>> getAssetsByChecksums(Iterable<String> checksums, {bool? isTrashed}) {
|
||||||
|
if (checksums.isEmpty) return Future.value([]);
|
||||||
|
final conditions = <Expression<bool>>[
|
||||||
|
_db.remoteAssetEntity.checksum.isIn(checksums),
|
||||||
|
];
|
||||||
|
|
||||||
|
if (isTrashed != null) {
|
||||||
|
if (isTrashed) {
|
||||||
|
conditions.add(_db.remoteAssetEntity.deletedAt.isNotNull());
|
||||||
|
} else {
|
||||||
|
conditions.add(_db.remoteAssetEntity.deletedAt.isNull());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
final query = _db.remoteAssetEntity.select()..where((rae) => conditions.reduce((a, b) => a & b));
|
||||||
|
return query.map((row) => row.toDto()).get();
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -2,7 +2,12 @@ import 'package:hooks_riverpod/hooks_riverpod.dart';
|
||||||
import 'package:immich_mobile/domain/services/asset.service.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/local_asset.repository.dart';
|
||||||
import 'package:immich_mobile/infrastructure/repositories/remote_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/db.provider.dart';
|
||||||
|
import 'package:immich_mobile/providers/infrastructure/storage.provider.dart';
|
||||||
|
import 'package:logging/logging.dart';
|
||||||
|
|
||||||
|
import '../../repositories/local_files_manager.repository.dart';
|
||||||
|
|
||||||
final localAssetRepository = Provider<DriftLocalAssetRepository>(
|
final localAssetRepository = Provider<DriftLocalAssetRepository>(
|
||||||
(ref) => DriftLocalAssetRepository(ref.watch(driftProvider)),
|
(ref) => DriftLocalAssetRepository(ref.watch(driftProvider)),
|
||||||
|
|
@ -14,14 +19,22 @@ final remoteAssetRepositoryProvider = Provider<RemoteAssetRepository>(
|
||||||
|
|
||||||
final assetServiceProvider = Provider(
|
final assetServiceProvider = Provider(
|
||||||
(ref) => AssetService(
|
(ref) => AssetService(
|
||||||
|
appSettingsService: ref.watch(appSettingsServiceProvider),
|
||||||
remoteAssetRepository: ref.watch(remoteAssetRepositoryProvider),
|
remoteAssetRepository: ref.watch(remoteAssetRepositoryProvider),
|
||||||
localAssetRepository: ref.watch(localAssetRepository),
|
localAssetRepository: ref.watch(localAssetRepository),
|
||||||
|
localFilesManager: ref.watch(localFilesManagerRepositoryProvider),
|
||||||
|
storageRepository: ref.watch(storageRepositoryProvider),
|
||||||
|
logger: Logger('AssetService'),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|
||||||
final placesProvider = FutureProvider<List<(String, String)>>(
|
final placesProvider = FutureProvider<List<(String, String)>>(
|
||||||
(ref) => AssetService(
|
(ref) => AssetService(
|
||||||
|
appSettingsService: ref.watch(appSettingsServiceProvider),
|
||||||
remoteAssetRepository: ref.watch(remoteAssetRepositoryProvider),
|
remoteAssetRepository: ref.watch(remoteAssetRepositoryProvider),
|
||||||
localAssetRepository: ref.watch(localAssetRepository),
|
localAssetRepository: ref.watch(localAssetRepository),
|
||||||
|
localFilesManager: ref.watch(localFilesManagerRepositoryProvider),
|
||||||
|
storageRepository: ref.watch(storageRepositoryProvider),
|
||||||
|
logger: Logger('AssetService'),
|
||||||
).getPlaces(),
|
).getPlaces(),
|
||||||
);
|
);
|
||||||
|
|
|
||||||
|
|
@ -16,6 +16,7 @@ final syncStreamServiceProvider = Provider(
|
||||||
(ref) => SyncStreamService(
|
(ref) => SyncStreamService(
|
||||||
syncApiRepository: ref.watch(syncApiRepositoryProvider),
|
syncApiRepository: ref.watch(syncApiRepositoryProvider),
|
||||||
syncStreamRepository: ref.watch(syncStreamRepositoryProvider),
|
syncStreamRepository: ref.watch(syncStreamRepositoryProvider),
|
||||||
|
assetService: ref.watch(assetServiceProvider),
|
||||||
cancelChecker: ref.watch(cancellationProvider),
|
cancelChecker: ref.watch(cancellationProvider),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|
|
||||||
|
|
@ -1,3 +1,4 @@
|
||||||
|
import 'package:immich_mobile/domain/services/asset.service.dart';
|
||||||
import 'package:immich_mobile/domain/services/store.service.dart';
|
import 'package:immich_mobile/domain/services/store.service.dart';
|
||||||
import 'package:immich_mobile/domain/services/user.service.dart';
|
import 'package:immich_mobile/domain/services/user.service.dart';
|
||||||
import 'package:immich_mobile/domain/utils/background_sync.dart';
|
import 'package:immich_mobile/domain/utils/background_sync.dart';
|
||||||
|
|
@ -17,3 +18,5 @@ class MockNativeSyncApi extends Mock implements NativeSyncApi {}
|
||||||
class MockAppSettingsService extends Mock implements AppSettingsService {}
|
class MockAppSettingsService extends Mock implements AppSettingsService {}
|
||||||
|
|
||||||
class MockUploadService extends Mock implements UploadService {}
|
class MockUploadService extends Mock implements UploadService {}
|
||||||
|
|
||||||
|
class MockAssetService extends Mock implements AssetService {}
|
||||||
|
|
|
||||||
|
|
@ -2,6 +2,7 @@ import 'dart:async';
|
||||||
|
|
||||||
import 'package:flutter_test/flutter_test.dart';
|
import 'package:flutter_test/flutter_test.dart';
|
||||||
import 'package:immich_mobile/domain/models/sync_event.model.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/sync_stream.service.dart';
|
||||||
import 'package:immich_mobile/infrastructure/repositories/sync_api.repository.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/infrastructure/repositories/sync_stream.repository.dart';
|
||||||
|
|
@ -9,6 +10,7 @@ import 'package:mocktail/mocktail.dart';
|
||||||
|
|
||||||
import '../../fixtures/sync_stream.stub.dart';
|
import '../../fixtures/sync_stream.stub.dart';
|
||||||
import '../../infrastructure/repository.mock.dart';
|
import '../../infrastructure/repository.mock.dart';
|
||||||
|
import '../service.mock.dart';
|
||||||
|
|
||||||
class _AbortCallbackWrapper {
|
class _AbortCallbackWrapper {
|
||||||
const _AbortCallbackWrapper();
|
const _AbortCallbackWrapper();
|
||||||
|
|
@ -30,6 +32,7 @@ void main() {
|
||||||
late SyncStreamService sut;
|
late SyncStreamService sut;
|
||||||
late SyncStreamRepository mockSyncStreamRepo;
|
late SyncStreamRepository mockSyncStreamRepo;
|
||||||
late SyncApiRepository mockSyncApiRepo;
|
late SyncApiRepository mockSyncApiRepo;
|
||||||
|
late AssetService mockAssetService;
|
||||||
late Function(List<SyncEvent>, Function()) handleEventsCallback;
|
late Function(List<SyncEvent>, Function()) handleEventsCallback;
|
||||||
late _MockAbortCallbackWrapper mockAbortCallbackWrapper;
|
late _MockAbortCallbackWrapper mockAbortCallbackWrapper;
|
||||||
|
|
||||||
|
|
@ -39,7 +42,7 @@ void main() {
|
||||||
mockSyncStreamRepo = MockSyncStreamRepository();
|
mockSyncStreamRepo = MockSyncStreamRepository();
|
||||||
mockSyncApiRepo = MockSyncApiRepository();
|
mockSyncApiRepo = MockSyncApiRepository();
|
||||||
mockAbortCallbackWrapper = _MockAbortCallbackWrapper();
|
mockAbortCallbackWrapper = _MockAbortCallbackWrapper();
|
||||||
|
mockAssetService = MockAssetService();
|
||||||
when(() => mockAbortCallbackWrapper()).thenReturn(false);
|
when(() => mockAbortCallbackWrapper()).thenReturn(false);
|
||||||
|
|
||||||
when(() => mockSyncApiRepo.streamChanges(any())).thenAnswer((invocation) async {
|
when(() => mockSyncApiRepo.streamChanges(any())).thenAnswer((invocation) async {
|
||||||
|
|
@ -81,7 +84,7 @@ void main() {
|
||||||
when(() => mockSyncStreamRepo.updateAssetFacesV1(any())).thenAnswer(successHandler);
|
when(() => mockSyncStreamRepo.updateAssetFacesV1(any())).thenAnswer(successHandler);
|
||||||
when(() => mockSyncStreamRepo.deleteAssetFacesV1(any())).thenAnswer(successHandler);
|
when(() => mockSyncStreamRepo.deleteAssetFacesV1(any())).thenAnswer(successHandler);
|
||||||
|
|
||||||
sut = SyncStreamService(syncApiRepository: mockSyncApiRepo, syncStreamRepository: mockSyncStreamRepo);
|
sut = SyncStreamService(syncApiRepository: mockSyncApiRepo, syncStreamRepository: mockSyncStreamRepo, assetService: mockAssetService);
|
||||||
});
|
});
|
||||||
|
|
||||||
Future<void> simulateEvents(List<SyncEvent> events) async {
|
Future<void> simulateEvents(List<SyncEvent> events) async {
|
||||||
|
|
@ -146,6 +149,7 @@ void main() {
|
||||||
sut = SyncStreamService(
|
sut = SyncStreamService(
|
||||||
syncApiRepository: mockSyncApiRepo,
|
syncApiRepository: mockSyncApiRepo,
|
||||||
syncStreamRepository: mockSyncStreamRepo,
|
syncStreamRepository: mockSyncStreamRepo,
|
||||||
|
assetService: mockAssetService,
|
||||||
cancelChecker: cancellationChecker.call,
|
cancelChecker: cancellationChecker.call,
|
||||||
);
|
);
|
||||||
await sut.sync();
|
await sut.sync();
|
||||||
|
|
@ -181,6 +185,7 @@ void main() {
|
||||||
sut = SyncStreamService(
|
sut = SyncStreamService(
|
||||||
syncApiRepository: mockSyncApiRepo,
|
syncApiRepository: mockSyncApiRepo,
|
||||||
syncStreamRepository: mockSyncStreamRepo,
|
syncStreamRepository: mockSyncStreamRepo,
|
||||||
|
assetService: mockAssetService,
|
||||||
cancelChecker: cancellationChecker.call,
|
cancelChecker: cancellationChecker.call,
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue