mirror of
https://github.com/immich-app/immich.git
synced 2025-01-04 02:46:47 +01:00
15c04d3056
* refactor(mobile): DB repository for asset, backup, sync service * review feedback * fix bug found by Alex --------- Co-authored-by: Alex <alex.tran1502@gmail.com>
216 lines
7.5 KiB
Dart
216 lines
7.5 KiB
Dart
import 'dart:async';
|
|
import 'dart:typed_data';
|
|
|
|
import 'package:collection/collection.dart';
|
|
import 'package:flutter/foundation.dart';
|
|
import 'package:flutter/services.dart';
|
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
|
import 'package:immich_mobile/entities/asset.entity.dart';
|
|
import 'package:immich_mobile/entities/exif_info.entity.dart';
|
|
import 'package:immich_mobile/entities/store.entity.dart';
|
|
import 'package:immich_mobile/interfaces/asset.interface.dart';
|
|
import 'package:immich_mobile/interfaces/exif_info.interface.dart';
|
|
import 'package:immich_mobile/interfaces/file_media.interface.dart';
|
|
import 'package:immich_mobile/repositories/asset.repository.dart';
|
|
import 'package:immich_mobile/repositories/exif_info.repository.dart';
|
|
import 'package:immich_mobile/repositories/file_media.repository.dart';
|
|
import 'package:immich_mobile/services/api.service.dart';
|
|
import 'package:immich_mobile/utils/diff.dart';
|
|
|
|
/// Finds duplicates originating from missing EXIF information
|
|
class BackupVerificationService {
|
|
final IFileMediaRepository _fileMediaRepository;
|
|
final IAssetRepository _assetRepository;
|
|
final IExifInfoRepository _exifInfoRepository;
|
|
|
|
BackupVerificationService(
|
|
this._fileMediaRepository,
|
|
this._assetRepository,
|
|
this._exifInfoRepository,
|
|
);
|
|
|
|
/// Returns at most [limit] assets that were backed up without exif
|
|
Future<List<Asset>> findWronglyBackedUpAssets({int limit = 100}) async {
|
|
final owner = Store.get(StoreKey.currentUser).isarId;
|
|
final List<Asset> onlyLocal = await _assetRepository.getAll(
|
|
ownerId: owner,
|
|
state: AssetState.local,
|
|
limit: limit,
|
|
);
|
|
final List<Asset> remoteMatches = await _assetRepository.getMatches(
|
|
assets: onlyLocal,
|
|
ownerId: owner,
|
|
state: AssetState.remote,
|
|
limit: limit,
|
|
);
|
|
final List<Asset> localMatches = await _assetRepository.getMatches(
|
|
assets: remoteMatches,
|
|
ownerId: owner,
|
|
state: AssetState.local,
|
|
limit: limit,
|
|
);
|
|
|
|
final List<Asset> deleteCandidates = [], originals = [];
|
|
|
|
await diffSortedLists(
|
|
remoteMatches,
|
|
localMatches,
|
|
compare: (a, b) => a.fileName.compareTo(b.fileName),
|
|
both: (a, b) async {
|
|
a.exifInfo = await _exifInfoRepository.get(a.id);
|
|
deleteCandidates.add(a);
|
|
originals.add(b);
|
|
return false;
|
|
},
|
|
onlyFirst: (a) {},
|
|
onlySecond: (b) {},
|
|
);
|
|
final isolateToken = ServicesBinding.rootIsolateToken!;
|
|
final List<Asset> toDelete;
|
|
if (deleteCandidates.length > 10) {
|
|
// performs 2 checks in parallel for a nice speedup
|
|
final half = deleteCandidates.length ~/ 2;
|
|
final lower = compute(
|
|
_computeSaveToDelete,
|
|
(
|
|
deleteCandidates: deleteCandidates.slice(0, half),
|
|
originals: originals.slice(0, half),
|
|
auth: Store.get(StoreKey.accessToken),
|
|
endpoint: Store.get(StoreKey.serverEndpoint),
|
|
rootIsolateToken: isolateToken,
|
|
fileMediaRepository: _fileMediaRepository,
|
|
),
|
|
);
|
|
final upper = compute(
|
|
_computeSaveToDelete,
|
|
(
|
|
deleteCandidates: deleteCandidates.slice(half),
|
|
originals: originals.slice(half),
|
|
auth: Store.get(StoreKey.accessToken),
|
|
endpoint: Store.get(StoreKey.serverEndpoint),
|
|
rootIsolateToken: isolateToken,
|
|
fileMediaRepository: _fileMediaRepository,
|
|
),
|
|
);
|
|
toDelete = await lower + await upper;
|
|
} else {
|
|
toDelete = await compute(
|
|
_computeSaveToDelete,
|
|
(
|
|
deleteCandidates: deleteCandidates,
|
|
originals: originals,
|
|
auth: Store.get(StoreKey.accessToken),
|
|
endpoint: Store.get(StoreKey.serverEndpoint),
|
|
rootIsolateToken: isolateToken,
|
|
fileMediaRepository: _fileMediaRepository,
|
|
),
|
|
);
|
|
}
|
|
return toDelete;
|
|
}
|
|
|
|
static Future<List<Asset>> _computeSaveToDelete(
|
|
({
|
|
List<Asset> deleteCandidates,
|
|
List<Asset> originals,
|
|
String auth,
|
|
String endpoint,
|
|
RootIsolateToken rootIsolateToken,
|
|
IFileMediaRepository fileMediaRepository,
|
|
}) tuple,
|
|
) async {
|
|
assert(tuple.deleteCandidates.length == tuple.originals.length);
|
|
final List<Asset> result = [];
|
|
BackgroundIsolateBinaryMessenger.ensureInitialized(tuple.rootIsolateToken);
|
|
await tuple.fileMediaRepository.enableBackgroundAccess();
|
|
final ApiService apiService = ApiService();
|
|
apiService.setEndpoint(tuple.endpoint);
|
|
apiService.setAccessToken(tuple.auth);
|
|
for (int i = 0; i < tuple.deleteCandidates.length; i++) {
|
|
if (await _compareAssets(
|
|
tuple.deleteCandidates[i],
|
|
tuple.originals[i],
|
|
apiService,
|
|
)) {
|
|
result.add(tuple.deleteCandidates[i]);
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
static Future<bool> _compareAssets(
|
|
Asset remote,
|
|
Asset local,
|
|
ApiService apiService,
|
|
) async {
|
|
if (remote.checksum == local.checksum) return false;
|
|
ExifInfo? exif = remote.exifInfo;
|
|
if (exif != null && exif.lat != null) return false;
|
|
if (exif == null || exif.fileSize == null) {
|
|
final dto = await apiService.assetsApi.getAssetInfo(remote.remoteId!);
|
|
if (dto != null && dto.exifInfo != null) {
|
|
exif = ExifInfo.fromDto(dto.exifInfo!);
|
|
}
|
|
}
|
|
final file = await local.local!.originFile;
|
|
if (exif != null && file != null && exif.fileSize != null) {
|
|
final origSize = await file.length();
|
|
if (exif.fileSize! == origSize || exif.fileSize! != origSize) {
|
|
final latLng = await local.local!.latlngAsync();
|
|
|
|
if (exif.lat == null &&
|
|
latLng.latitude != null &&
|
|
(remote.fileCreatedAt.isAtSameMomentAs(local.fileCreatedAt) ||
|
|
remote.fileModifiedAt.isAtSameMomentAs(local.fileModifiedAt) ||
|
|
_sameExceptTimeZone(
|
|
remote.fileCreatedAt,
|
|
local.fileCreatedAt,
|
|
))) {
|
|
if (remote.type == AssetType.video) {
|
|
// it's very unlikely that a video of same length, filesize, name
|
|
// and date is wrong match. Cannot easily compare videos anyway
|
|
return true;
|
|
}
|
|
|
|
// for images: make sure they are pixel-wise identical
|
|
// (skip first few KBs containing metadata)
|
|
final Uint64List localImage =
|
|
_fakeDecodeImg(local, await file.readAsBytes());
|
|
final res = await apiService.assetsApi
|
|
.downloadAssetWithHttpInfo(remote.remoteId!);
|
|
final Uint64List remoteImage = _fakeDecodeImg(remote, res.bodyBytes);
|
|
|
|
final eq = const ListEquality().equals(remoteImage, localImage);
|
|
return eq;
|
|
}
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
static Uint64List _fakeDecodeImg(Asset asset, Uint8List bytes) {
|
|
const headerLength = 131072; // assume header is at most 128 KB
|
|
final start = bytes.length < headerLength * 2
|
|
? (bytes.length ~/ (4 * 8)) * 8
|
|
: headerLength;
|
|
return bytes.buffer.asUint64List(start);
|
|
}
|
|
|
|
static bool _sameExceptTimeZone(DateTime a, DateTime b) {
|
|
final ms = a.isAfter(b)
|
|
? a.millisecondsSinceEpoch - b.millisecondsSinceEpoch
|
|
: b.millisecondsSinceEpoch - a.microsecondsSinceEpoch;
|
|
final x = ms / (1000 * 60 * 30);
|
|
final y = ms ~/ (1000 * 60 * 30);
|
|
return y.toDouble() == x && y < 24;
|
|
}
|
|
}
|
|
|
|
final backupVerificationServiceProvider = Provider(
|
|
(ref) => BackupVerificationService(
|
|
ref.watch(fileMediaRepositoryProvider),
|
|
ref.watch(assetRepositoryProvider),
|
|
ref.watch(exifInfoRepositoryProvider),
|
|
),
|
|
);
|