import 'package:collection/collection.dart'; import 'package:drift/drift.dart'; import 'package:immich_mobile/constants/constants.dart'; import 'package:immich_mobile/domain/models/album/local_album.model.dart'; import 'package:immich_mobile/domain/models/asset/base_asset.model.dart'; import 'package:immich_mobile/infrastructure/entities/local_album.entity.dart'; import 'package:immich_mobile/infrastructure/entities/local_asset.entity.dart'; import 'package:immich_mobile/infrastructure/entities/local_asset.entity.drift.dart'; import 'package:immich_mobile/infrastructure/repositories/db.repository.dart'; class DriftLocalAssetRepository extends DriftDatabaseRepository { final Drift _db; const DriftLocalAssetRepository(this._db) : super(_db); SingleOrNullSelectable _assetSelectable(String id) { final query = _db.localAssetEntity.select().addColumns([_db.remoteAssetEntity.id]).join([ leftOuterJoin( _db.remoteAssetEntity, _db.localAssetEntity.checksum.equalsExp(_db.remoteAssetEntity.checksum), useColumns: false, ), ])..where(_db.localAssetEntity.id.equals(id)); return query.map((row) { final asset = row.readTable(_db.localAssetEntity).toDto(); return asset.copyWith(remoteId: row.read(_db.remoteAssetEntity.id)); }); } Future get(String id) => _assetSelectable(id).getSingleOrNull(); Future> getByChecksum(String checksum) { final query = _db.localAssetEntity.select()..where((lae) => lae.checksum.equals(checksum)); return query.map((row) => row.toDto()).get(); } Stream watch(String id) => _assetSelectable(id).watchSingleOrNull(); Future updateHashes(Map hashes) { if (hashes.isEmpty) { return Future.value(); } return _db.batch((batch) async { for (final entry in hashes.entries) { batch.update( _db.localAssetEntity, LocalAssetEntityCompanion(checksum: Value(entry.value)), where: (e) => e.id.equals(entry.key), ); } }); } Future delete(List ids) { if (ids.isEmpty) { return Future.value(); } return _db.batch((batch) { for (final id in ids) { batch.deleteWhere(_db.localAssetEntity, (e) => e.id.equals(id)); } }); } Future getById(String id) { final query = _db.localAssetEntity.select()..where((lae) => lae.id.equals(id)); return query.map((row) => row.toDto()).getSingleOrNull(); } Future getCount() { return _db.managers.localAssetEntity.count(); } Future getHashedCount() { return _db.managers.localAssetEntity.filter((e) => e.checksum.isNull().not()).count(); } Future> getSourceAlbums(String localAssetId, {BackupSelection? backupSelection}) { final query = _db.localAlbumEntity.select() ..where( (lae) => existsQuery( _db.localAlbumAssetEntity.selectOnly() ..addColumns([_db.localAlbumAssetEntity.albumId]) ..where( _db.localAlbumAssetEntity.albumId.equalsExp(lae.id) & _db.localAlbumAssetEntity.assetId.equals(localAssetId), ), ), ) ..orderBy([(lae) => OrderingTerm.asc(lae.name)]); if (backupSelection != null) { query.where((lae) => lae.backupSelection.equalsValue(backupSelection)); } return query.map((localAlbum) => localAlbum.toDto()).get(); } Future>> getAssetsFromBackupAlbums(Iterable checksums) async { if (checksums.isEmpty) { return {}; } final result = >{}; for (final slice in checksums.toSet().slices(kDriftMaxChunk)) { final rows = await (_db.select(_db.localAlbumAssetEntity).join([ innerJoin(_db.localAlbumEntity, _db.localAlbumAssetEntity.albumId.equalsExp(_db.localAlbumEntity.id)), innerJoin(_db.localAssetEntity, _db.localAlbumAssetEntity.assetId.equalsExp(_db.localAssetEntity.id)), ])..where( _db.localAlbumEntity.backupSelection.equalsValue(BackupSelection.selected) & _db.localAssetEntity.checksum.isIn(slice), )) .get(); for (final row in rows) { final albumId = row.readTable(_db.localAlbumAssetEntity).albumId; final assetData = row.readTable(_db.localAssetEntity); final asset = assetData.toDto(); (result[albumId] ??= []).add(asset); } } return result; } }