1
0
Fork 0
mirror of https://github.com/immich-app/immich.git synced 2025-01-19 18:26:46 +01:00

chore(mobile): use Record instead of custom pair+triple (#2483)

This commit is contained in:
Fynn Petersen-Frey 2023-05-21 03:41:34 +02:00 committed by GitHub
parent a089d9891d
commit dc7b0f75bb
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
6 changed files with 58 additions and 80 deletions

View file

@ -364,7 +364,7 @@ class BackupControllerPage extends HookConsumerWidget {
.read(backgroundServiceProvider) .read(backgroundServiceProvider)
.getIOSBackgroundAppRefreshEnabled(), .getIOSBackgroundAppRefreshEnabled(),
builder: (context, snapshot) { builder: (context, snapshot) {
final enabled = snapshot.data as bool?; final enabled = snapshot.data;
// If it's not enabled, show them some kind of alert that says // If it's not enabled, show them some kind of alert that says
// background refresh is not enabled // background refresh is not enabled
if (enabled != null && !enabled) {} if (enabled != null && !enabled) {}

View file

@ -10,7 +10,6 @@ import 'package:immich_mobile/shared/providers/db.provider.dart';
import 'package:immich_mobile/shared/services/api.service.dart'; import 'package:immich_mobile/shared/services/api.service.dart';
import 'package:immich_mobile/shared/services/sync.service.dart'; import 'package:immich_mobile/shared/services/sync.service.dart';
import 'package:immich_mobile/utils/openapi_extensions.dart'; import 'package:immich_mobile/utils/openapi_extensions.dart';
import 'package:immich_mobile/utils/tuple.dart';
import 'package:isar/isar.dart'; import 'package:isar/isar.dart';
import 'package:logging/logging.dart'; import 'package:logging/logging.dart';
import 'package:openapi/api.dart'; import 'package:openapi/api.dart';
@ -60,15 +59,14 @@ class AssetService {
}) async { }) async {
try { try {
final etag = hasCache ? Store.tryGet(StoreKey.assetETag) : null; final etag = hasCache ? Store.tryGet(StoreKey.assetETag) : null;
final Pair<List<AssetResponseDto>, String?>? remote = final (List<AssetResponseDto>? assets, String? newETag) =
await _apiService.assetApi.getAllAssetsWithETag(eTag: etag); await _apiService.assetApi.getAllAssetsWithETag(eTag: etag);
if (remote == null) { if (assets == null) {
return null; return null;
} else if (newETag != etag) {
Store.put(StoreKey.assetETag, newETag);
} }
if (remote.second != null && remote.second != etag) { return assets;
Store.put(StoreKey.assetETag, remote.second);
}
return remote.first;
} catch (e, stack) { } catch (e, stack) {
log.severe('Error while getting remote assets', e, stack); log.severe('Error while getting remote assets', e, stack);
return null; return null;

View file

@ -11,7 +11,6 @@ import 'package:immich_mobile/shared/providers/db.provider.dart';
import 'package:immich_mobile/utils/async_mutex.dart'; import 'package:immich_mobile/utils/async_mutex.dart';
import 'package:immich_mobile/utils/builtin_extensions.dart'; import 'package:immich_mobile/utils/builtin_extensions.dart';
import 'package:immich_mobile/utils/diff.dart'; import 'package:immich_mobile/utils/diff.dart';
import 'package:immich_mobile/utils/tuple.dart';
import 'package:isar/isar.dart'; import 'package:isar/isar.dart';
import 'package:logging/logging.dart'; import 'package:logging/logging.dart';
import 'package:openapi/api.dart'; import 'package:openapi/api.dart';
@ -94,7 +93,7 @@ class SyncService {
deleteCandidates.sort(Asset.compareById); deleteCandidates.sort(Asset.compareById);
existing.sort(Asset.compareById); existing.sort(Asset.compareById);
return _diffAssets(existing, deleteCandidates, compare: Asset.compareById) return _diffAssets(existing, deleteCandidates, compare: Asset.compareById)
.third .$3
.map((e) => e.id) .map((e) => e.id)
.toList(); .toList();
} }
@ -165,14 +164,14 @@ class SyncService {
.thenByFileModifiedAt() .thenByFileModifiedAt()
.findAll(); .findAll();
remote.sort(Asset.compareByOwnerDeviceLocalIdModified); remote.sort(Asset.compareByOwnerDeviceLocalIdModified);
final diff = _diffAssets(remote, inDb, remote: true); final (toAdd, toUpdate, toRemove) = _diffAssets(remote, inDb, remote: true);
if (diff.first.isEmpty && diff.second.isEmpty && diff.third.isEmpty) { if (toAdd.isEmpty && toUpdate.isEmpty && toRemove.isEmpty) {
return false; return false;
} }
final idsToDelete = diff.third.map((e) => e.id).toList(); final idsToDelete = toRemove.map((e) => e.id).toList();
try { try {
await _db.writeTxn(() => _db.assets.deleteAll(idsToDelete)); await _db.writeTxn(() => _db.assets.deleteAll(idsToDelete));
await upsertAssetsWithExif(diff.first + diff.second); await upsertAssetsWithExif(toAdd + toUpdate);
} on IsarError catch (e) { } on IsarError catch (e) {
_log.severe("Failed to sync remote assets to db: $e"); _log.severe("Failed to sync remote assets to db: $e");
} }
@ -252,8 +251,7 @@ class SyncService {
.findAll(); .findAll();
final List<Asset> assetsOnRemote = dto.getAssets(); final List<Asset> assetsOnRemote = dto.getAssets();
assetsOnRemote.sort(Asset.compareByOwnerDeviceLocalIdModified); assetsOnRemote.sort(Asset.compareByOwnerDeviceLocalIdModified);
final d = _diffAssets(assetsOnRemote, assetsInDb); final (toAdd, toUpdate, toUnlink) = _diffAssets(assetsOnRemote, assetsInDb);
final List<Asset> toAdd = d.first, toUpdate = d.second, toUnlink = d.third;
// update shared users // update shared users
final List<User> sharedUsers = album.sharedUsers.toList(growable: false); final List<User> sharedUsers = album.sharedUsers.toList(growable: false);
@ -271,9 +269,9 @@ class SyncService {
); );
// for shared album: put missing album assets into local DB // for shared album: put missing album assets into local DB
final resultPair = await _linkWithExistingFromDb(toAdd); final (existingInDb, updated) = await _linkWithExistingFromDb(toAdd);
await upsertAssetsWithExif(resultPair.second); await upsertAssetsWithExif(updated);
final assetsToLink = resultPair.first + resultPair.second; final assetsToLink = existingInDb + updated;
final usersToLink = (await _db.users.getAllById(userIdsToAdd)).cast<User>(); final usersToLink = (await _db.users.getAllById(userIdsToAdd)).cast<User>();
album.name = dto.albumName; album.name = dto.albumName;
@ -327,9 +325,10 @@ class SyncService {
if (dto.assetCount == dto.assets.length) { if (dto.assetCount == dto.assets.length) {
// in case an album contains assets not yet present in local DB: // in case an album contains assets not yet present in local DB:
// put missing album assets into local DB // put missing album assets into local DB
final result = await _linkWithExistingFromDb(dto.getAssets()); final (existingInDb, updated) =
existing.addAll(result.first); await _linkWithExistingFromDb(dto.getAssets());
await upsertAssetsWithExif(result.second); existing.addAll(existingInDb);
await upsertAssetsWithExif(updated);
final Album a = await Album.remote(dto); final Album a = await Album.remote(dto);
await _db.writeTxn(() => _db.albums.store(a)); await _db.writeTxn(() => _db.albums.store(a));
@ -393,18 +392,19 @@ class SyncService {
_log.fine( _log.fine(
"Syncing all local albums almost done. Collected ${deleteCandidates.length} asset candidates to delete", "Syncing all local albums almost done. Collected ${deleteCandidates.length} asset candidates to delete",
); );
final pair = _handleAssetRemoval(deleteCandidates, existing, remote: false); final (toDelete, toUpdate) =
_handleAssetRemoval(deleteCandidates, existing, remote: false);
_log.fine( _log.fine(
"${pair.first.length} assets to delete, ${pair.second.length} to update", "${toDelete.length} assets to delete, ${toUpdate.length} to update",
); );
if (pair.first.isNotEmpty || pair.second.isNotEmpty) { if (toDelete.isNotEmpty || toUpdate.isNotEmpty) {
await _db.writeTxn(() async { await _db.writeTxn(() async {
await _db.assets.deleteAll(pair.first); await _db.assets.deleteAll(toDelete);
await _db.exifInfos.deleteAll(pair.first); await _db.exifInfos.deleteAll(toDelete);
await _db.assets.putAll(pair.second); await _db.assets.putAll(toUpdate);
}); });
_log.info( _log.info(
"Removed ${pair.first.length} and updated ${pair.second.length} local assets from DB", "Removed ${toDelete.length} and updated ${toUpdate.length} local assets from DB",
); );
} }
return anyChanges; return anyChanges;
@ -441,8 +441,8 @@ class SyncService {
final List<Asset> onDevice = final List<Asset> onDevice =
await ape.getAssets(excludedAssets: excludedAssets); await ape.getAssets(excludedAssets: excludedAssets);
onDevice.sort(Asset.compareByLocalId); onDevice.sort(Asset.compareByLocalId);
final d = _diffAssets(onDevice, inDb, compare: Asset.compareByLocalId); final (toAdd, toUpdate, toDelete) =
final List<Asset> toAdd = d.first, toUpdate = d.second, toDelete = d.third; _diffAssets(onDevice, inDb, compare: Asset.compareByLocalId);
if (toAdd.isEmpty && if (toAdd.isEmpty &&
toUpdate.isEmpty && toUpdate.isEmpty &&
toDelete.isEmpty && toDelete.isEmpty &&
@ -458,12 +458,12 @@ class SyncService {
_log.fine( _log.fine(
"Syncing local album ${ape.name}. ${toAdd.length} assets to add, ${toUpdate.length} to update, ${toDelete.length} to delete", "Syncing local album ${ape.name}. ${toAdd.length} assets to add, ${toUpdate.length} to update, ${toDelete.length} to delete",
); );
final result = await _linkWithExistingFromDb(toAdd); final (existingInDb, updated) = await _linkWithExistingFromDb(toAdd);
_log.fine( _log.fine(
"Linking assets to add with existing from db. ${result.first.length} existing, ${result.second.length} to update", "Linking assets to add with existing from db. ${existingInDb.length} existing, ${updated.length} to update",
); );
deleteCandidates.addAll(toDelete); deleteCandidates.addAll(toDelete);
existing.addAll(result.first); existing.addAll(existingInDb);
album.name = ape.name; album.name = ape.name;
album.modifiedAt = ape.lastModified ?? DateTime.now(); album.modifiedAt = ape.lastModified ?? DateTime.now();
if (album.thumbnail.value != null && if (album.thumbnail.value != null &&
@ -472,10 +472,10 @@ class SyncService {
} }
try { try {
await _db.writeTxn(() async { await _db.writeTxn(() async {
await _db.assets.putAll(result.second); await _db.assets.putAll(updated);
await _db.assets.putAll(toUpdate); await _db.assets.putAll(toUpdate);
await album.assets await album.assets
.update(link: result.first + result.second, unlink: toDelete); .update(link: existingInDb + updated, unlink: toDelete);
await _db.albums.put(album); await _db.albums.put(album);
album.thumbnail.value ??= await album.assets.filter().findFirst(); album.thumbnail.value ??= await album.assets.filter().findFirst();
await album.thumbnail.save(); await album.thumbnail.save();
@ -510,11 +510,11 @@ class SyncService {
return false; return false;
} }
album.modifiedAt = ape.lastModified ?? DateTime.now(); album.modifiedAt = ape.lastModified ?? DateTime.now();
final result = await _linkWithExistingFromDb(newAssets); final (existingInDb, updated) = await _linkWithExistingFromDb(newAssets);
try { try {
await _db.writeTxn(() async { await _db.writeTxn(() async {
await _db.assets.putAll(result.second); await _db.assets.putAll(updated);
await album.assets.update(link: result.first + result.second); await album.assets.update(link: existingInDb + updated);
await _db.albums.put(album); await _db.albums.put(album);
}); });
_log.info("Fast synced local album ${ape.name} to DB"); _log.info("Fast synced local album ${ape.name} to DB");
@ -536,15 +536,15 @@ class SyncService {
_log.info("Syncing a new local album to DB: ${ape.name}"); _log.info("Syncing a new local album to DB: ${ape.name}");
final Album a = Album.local(ape); final Album a = Album.local(ape);
final assets = await ape.getAssets(excludedAssets: excludedAssets); final assets = await ape.getAssets(excludedAssets: excludedAssets);
final result = await _linkWithExistingFromDb(assets); final (existingInDb, updated) = await _linkWithExistingFromDb(assets);
_log.info( _log.info(
"${result.first.length} assets already existed in DB, to upsert ${result.second.length}", "${existingInDb.length} assets already existed in DB, to upsert ${updated.length}",
); );
await upsertAssetsWithExif(result.second); await upsertAssetsWithExif(updated);
existing.addAll(result.first); existing.addAll(existingInDb);
a.assets.addAll(result.first); a.assets.addAll(existingInDb);
a.assets.addAll(result.second); a.assets.addAll(updated);
final thumb = result.first.firstOrNull ?? result.second.firstOrNull; final thumb = existingInDb.firstOrNull ?? updated.firstOrNull;
a.thumbnail.value = thumb; a.thumbnail.value = thumb;
try { try {
await _db.writeTxn(() => _db.albums.store(a)); await _db.writeTxn(() => _db.albums.store(a));
@ -555,11 +555,11 @@ class SyncService {
} }
/// Returns a tuple (existing, updated) /// Returns a tuple (existing, updated)
Future<Pair<List<Asset>, List<Asset>>> _linkWithExistingFromDb( Future<(List<Asset> existing, List<Asset> updated)> _linkWithExistingFromDb(
List<Asset> assets, List<Asset> assets,
) async { ) async {
if (assets.isEmpty) { if (assets.isEmpty) {
return const Pair([], []); return ([].cast<Asset>(), [].cast<Asset>());
} }
final List<Asset> inDb = await _db.assets final List<Asset> inDb = await _db.assets
.where() .where()
@ -596,7 +596,7 @@ class SyncService {
), ),
onlySecond: (Asset b) => toUpsert.add(b), onlySecond: (Asset b) => toUpsert.add(b),
); );
return Pair(existing, toUpsert); return (existing, toUpsert);
} }
/// Inserts or updates the assets in the database with their ExifInfo (if any) /// Inserts or updates the assets in the database with their ExifInfo (if any)
@ -623,7 +623,7 @@ class SyncService {
} }
/// Returns a triple(toAdd, toUpdate, toRemove) /// Returns a triple(toAdd, toUpdate, toRemove)
Triple<List<Asset>, List<Asset>, List<Asset>> _diffAssets( (List<Asset> toAdd, List<Asset> toUpdate, List<Asset> toRemove) _diffAssets(
List<Asset> assets, List<Asset> assets,
List<Asset> inDb, { List<Asset> inDb, {
bool? remote, bool? remote,
@ -660,30 +660,30 @@ Triple<List<Asset>, List<Asset>, List<Asset>> _diffAssets(
}, },
onlySecond: (Asset b) => toAdd.add(b), onlySecond: (Asset b) => toAdd.add(b),
); );
return Triple(toAdd, toUpdate, toRemove); return (toAdd, toUpdate, toRemove);
} }
/// returns a tuple (toDelete toUpdate) when assets are to be deleted /// returns a tuple (toDelete toUpdate) when assets are to be deleted
Pair<List<int>, List<Asset>> _handleAssetRemoval( (List<int> toDelete, List<Asset> toUpdate) _handleAssetRemoval(
List<Asset> deleteCandidates, List<Asset> deleteCandidates,
List<Asset> existing, { List<Asset> existing, {
bool? remote, bool? remote,
}) { }) {
if (deleteCandidates.isEmpty) { if (deleteCandidates.isEmpty) {
return const Pair([], []); return const ([], []);
} }
deleteCandidates.sort(Asset.compareById); deleteCandidates.sort(Asset.compareById);
deleteCandidates.uniqueConsecutive((a) => a.id); deleteCandidates.uniqueConsecutive((a) => a.id);
existing.sort(Asset.compareById); existing.sort(Asset.compareById);
existing.uniqueConsecutive((a) => a.id); existing.uniqueConsecutive((a) => a.id);
final triple = _diffAssets( final (tooAdd, toUpdate, toRemove) = _diffAssets(
existing, existing,
deleteCandidates, deleteCandidates,
compare: Asset.compareById, compare: Asset.compareById,
remote: remote, remote: remote,
); );
assert(triple.first.isEmpty, "toAdd should be empty in _handleAssetRemoval"); assert(tooAdd.isEmpty, "toAdd should be empty in _handleAssetRemoval");
return Pair(triple.third.map((e) => e.id).toList(), triple.second); return (toRemove.map((e) => e.id).toList(), toUpdate);
} }
/// returns `true` if the albums differ on the surface /// returns `true` if the albums differ on the surface

View file

@ -4,8 +4,6 @@ import 'dart:io';
import 'package:http/http.dart'; import 'package:http/http.dart';
import 'package:openapi/api.dart'; import 'package:openapi/api.dart';
import 'tuple.dart';
/// Extension methods to retrieve ETag together with the API call /// Extension methods to retrieve ETag together with the API call
extension WithETag on AssetApi { extension WithETag on AssetApi {
/// Get all AssetEntity belong to the user /// Get all AssetEntity belong to the user
@ -14,7 +12,7 @@ extension WithETag on AssetApi {
/// ///
/// * [String] eTag: /// * [String] eTag:
/// ETag of data already cached on the client /// ETag of data already cached on the client
Future<Pair<List<AssetResponseDto>, String?>?> getAllAssetsWithETag({ Future<(List<AssetResponseDto>? assets, String? eTag)> getAllAssetsWithETag({
String? eTag, String? eTag,
}) async { }) async {
final response = await getAllAssetsWithHttpInfo( final response = await getAllAssetsWithHttpInfo(
@ -36,9 +34,9 @@ extension WithETag on AssetApi {
) as List) ) as List)
.cast<AssetResponseDto>() .cast<AssetResponseDto>()
.toList(); .toList();
return Pair(data, etag); return (data, etag);
} }
return null; return (null, null);
} }
} }

View file

@ -1,18 +0,0 @@
/// An immutable pair or 2-tuple
/// TODO replace with Record once Dart 2.19 is available
class Pair<T1, T2> {
final T1 first;
final T2 second;
const Pair(this.first, this.second);
}
/// An immutable triple or 3-tuple
/// TODO replace with Record once Dart 2.19 is available
class Triple<T1, T2, T3> {
final T1 first;
final T2 second;
final T3 third;
const Triple(this.first, this.second, this.third);
}

View file

@ -6,7 +6,7 @@ version: 1.56.2+79
isar_version: &isar_version 3.0.5 isar_version: &isar_version 3.0.5
environment: environment:
sdk: ">=2.17.0 <3.0.0" sdk: ">=3.0.0-0 <4.0.0"
dependencies: dependencies:
flutter: flutter: