import 'dart:async'; import 'dart:convert'; import 'dart:io'; import 'package:collection/collection.dart'; import 'package:drift/drift.dart'; import 'package:flutter/foundation.dart'; import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:immich_mobile/domain/models/album/local_album.model.dart'; import 'package:immich_mobile/domain/models/store.model.dart'; import 'package:immich_mobile/entities/album.entity.dart'; import 'package:immich_mobile/entities/android_device_asset.entity.dart'; import 'package:immich_mobile/entities/asset.entity.dart'; import 'package:immich_mobile/entities/backup_album.entity.dart' as isar_backup_album; import 'package:immich_mobile/entities/etag.entity.dart'; import 'package:immich_mobile/entities/ios_device_asset.entity.dart'; import 'package:immich_mobile/entities/store.entity.dart'; import 'package:immich_mobile/infrastructure/entities/device_asset.entity.dart'; import 'package:immich_mobile/infrastructure/entities/exif.entity.dart'; import 'package:immich_mobile/infrastructure/entities/local_album.entity.drift.dart'; import 'package:immich_mobile/infrastructure/entities/local_asset.entity.drift.dart'; import 'package:immich_mobile/infrastructure/entities/store.entity.dart'; import 'package:immich_mobile/infrastructure/entities/store.entity.drift.dart'; import 'package:immich_mobile/infrastructure/entities/user.entity.dart'; import 'package:immich_mobile/infrastructure/repositories/db.repository.dart'; import 'package:immich_mobile/providers/app_settings.provider.dart'; import 'package:immich_mobile/infrastructure/repositories/local_album.repository.dart'; import 'package:immich_mobile/platform/native_sync_api.g.dart'; import 'package:immich_mobile/providers/background_sync.provider.dart'; import 'package:immich_mobile/providers/backup/backup.provider.dart'; import 'package:immich_mobile/services/app_settings.service.dart'; import 'package:immich_mobile/utils/diff.dart'; import 'package:isar/isar.dart'; import 'package:logging/logging.dart'; // ignore: import_rule_photo_manager import 'package:photo_manager/photo_manager.dart'; const int targetVersion = 15; Future migrateDatabaseIfNeeded(Isar db, Drift drift) async { final hasVersion = Store.tryGet(StoreKey.version) != null; final int version = Store.get(StoreKey.version, targetVersion); if (version < 9) { await Store.put(StoreKey.version, targetVersion); final value = await db.storeValues.get(StoreKey.currentUser.id); if (value != null) { final id = value.intValue; if (id != null) { await db.writeTxn(() async { final user = await db.users.get(id); await db.storeValues.put(StoreValue(StoreKey.currentUser.id, strValue: user?.id)); }); } } } if (version < 10) { await Store.put(StoreKey.version, targetVersion); await _migrateDeviceAsset(db); } if (version < 13) { await Store.put(StoreKey.photoManagerCustomFilter, true); } // This means that the SQLite DB is just created and has no version if (version < 14 || !hasVersion) { await migrateStoreToSqlite(db, drift); await Store.populateCache(); } if (version < 15) { try { await _updateCloudId(drift); } catch (error) { Logger("Migration").warning("Error occurred while updating cloud ID: $error"); // do not update version when error occurs so this is retried the next time return; } } if (targetVersion >= 12) { await Store.put(StoreKey.version, targetVersion); return; } final shouldTruncate = version < 8 || version < targetVersion; if (shouldTruncate) { await _migrateTo(db, targetVersion); } } Future _migrateTo(Isar db, int version) async { await Store.delete(StoreKey.assetETag); await db.writeTxn(() async { await db.assets.clear(); await db.exifInfos.clear(); await db.albums.clear(); await db.eTags.clear(); await db.users.clear(); }); await Store.put(StoreKey.version, version); } Future _migrateDeviceAsset(Isar db) async { final ids = Platform.isAndroid ? (await db.androidDeviceAssets.where().findAll()) .map((a) => _DeviceAsset(assetId: a.id.toString(), hash: a.hash)) .toList() : (await db.iOSDeviceAssets.where().findAll()).map((i) => _DeviceAsset(assetId: i.id, hash: i.hash)).toList(); final PermissionState ps = await PhotoManager.requestPermissionExtend(); if (!ps.hasAccess) { if (kDebugMode) { debugPrint("[MIGRATION] Photo library permission not granted. Skipping device asset migration."); } return; } List<_DeviceAsset> localAssets = []; final List paths = await PhotoManager.getAssetPathList(onlyAll: true); if (paths.isEmpty) { localAssets = (await db.assets.where().anyOf(ids, (query, id) => query.localIdEqualTo(id.assetId)).findAll()) .map((a) => _DeviceAsset(assetId: a.localId!, dateTime: a.fileModifiedAt)) .toList(); } else { final AssetPathEntity albumWithAll = paths.first; final int assetCount = await albumWithAll.assetCountAsync; final List allDeviceAssets = await albumWithAll.getAssetListRange(start: 0, end: assetCount); localAssets = allDeviceAssets.map((a) => _DeviceAsset(assetId: a.id, dateTime: a.modifiedDateTime)).toList(); } debugPrint("[MIGRATION] Device Asset Ids length - ${ids.length}"); debugPrint("[MIGRATION] Local Asset Ids length - ${localAssets.length}"); ids.sort((a, b) => a.assetId.compareTo(b.assetId)); localAssets.sort((a, b) => a.assetId.compareTo(b.assetId)); final List toAdd = []; await diffSortedLists( ids, localAssets, compare: (a, b) => a.assetId.compareTo(b.assetId), both: (deviceAsset, asset) { toAdd.add( DeviceAssetEntity(assetId: deviceAsset.assetId, hash: deviceAsset.hash!, modifiedTime: asset.dateTime!), ); return false; }, onlyFirst: (deviceAsset) { if (kDebugMode) { debugPrint('[MIGRATION] Local asset not found in DeviceAsset: ${deviceAsset.assetId}'); } }, onlySecond: (asset) { if (kDebugMode) { debugPrint('[MIGRATION] Local asset not found in DeviceAsset: ${asset.assetId}'); } }, ); if (kDebugMode) { debugPrint("[MIGRATION] Total number of device assets migrated - ${toAdd.length}"); } await db.writeTxn(() async { await db.deviceAssetEntitys.putAll(toAdd); }); } Future migrateDeviceAssetToSqlite(Isar db, Drift drift) async { try { final isarDeviceAssets = await db.deviceAssetEntitys.where().findAll(); await drift.batch((batch) { for (final deviceAsset in isarDeviceAssets) { batch.update( drift.localAssetEntity, LocalAssetEntityCompanion(checksum: Value(base64.encode(deviceAsset.hash))), where: (t) => t.id.equals(deviceAsset.assetId), ); } }); } catch (error) { debugPrint("[MIGRATION] Error while migrating device assets to SQLite: $error"); } } Future _updateCloudId(Drift drift) async { // Android do not have a concept of cloud IDs if (Platform.isAndroid) { return; } final query = drift.localAssetEntity.selectOnly() ..addColumns([drift.localAssetEntity.id]) ..where(drift.localAssetEntity.cloudId.isNull()); final ids = await query.map((row) => row.read(drift.localAssetEntity.id)!).get(); final cloudMapping = await NativeSyncApi().getCloudIdForAssetIds(ids); await DriftLocalAlbumRepository(drift).updateCloudMapping(cloudMapping); } Future migrateBackupAlbumsToSqlite(Isar db, Drift drift) async { try { final isarBackupAlbums = await db.backupAlbums.where().findAll(); // Recents is a virtual album on Android, and we don't have it with the new sync // If recents is selected previously, select all albums during migration except the excluded ones if (Platform.isAndroid) { final recentAlbum = isarBackupAlbums.firstWhereOrNull((album) => album.id == 'isAll'); if (recentAlbum != null) { await drift.localAlbumEntity.update().write( const LocalAlbumEntityCompanion(backupSelection: Value(BackupSelection.selected)), ); final excluded = isarBackupAlbums .where((album) => album.selection == isar_backup_album.BackupSelection.exclude) .map((album) => album.id) .toList(); await drift.batch((batch) async { for (final id in excluded) { batch.update( drift.localAlbumEntity, const LocalAlbumEntityCompanion(backupSelection: Value(BackupSelection.excluded)), where: (t) => t.id.equals(id), ); } }); return; } } await drift.batch((batch) { for (final album in isarBackupAlbums) { batch.update( drift.localAlbumEntity, LocalAlbumEntityCompanion( backupSelection: Value(switch (album.selection) { isar_backup_album.BackupSelection.none => BackupSelection.none, isar_backup_album.BackupSelection.select => BackupSelection.selected, isar_backup_album.BackupSelection.exclude => BackupSelection.excluded, }), ), where: (t) => t.id.equals(album.id), ); } }); } catch (error) { debugPrint("[MIGRATION] Error while migrating backup albums to SQLite: $error"); } } Future migrateStoreToSqlite(Isar db, Drift drift) async { try { final isarStoreValues = await db.storeValues.where().findAll(); await drift.batch((batch) { for (final storeValue in isarStoreValues) { final companion = StoreEntityCompanion( id: Value(storeValue.id), stringValue: Value(storeValue.strValue), intValue: Value(storeValue.intValue), ); batch.insert(drift.storeEntity, companion, onConflict: DoUpdate((_) => companion)); } }); } catch (error) { debugPrint("[MIGRATION] Error while migrating store values to SQLite: $error"); } } Future migrateStoreToIsar(Isar db, Drift drift) async { try { final driftStoreValues = await drift.storeEntity .select() .map((entity) => StoreValue(entity.id, intValue: entity.intValue, strValue: entity.stringValue)) .get(); await db.writeTxn(() async { await db.storeValues.putAll(driftStoreValues); }); } catch (error) { debugPrint("[MIGRATION] Error while migrating store values to Isar: $error"); } } class _DeviceAsset { final String assetId; final List? hash; final DateTime? dateTime; const _DeviceAsset({required this.assetId, this.hash, this.dateTime}); } Future> runNewSync(WidgetRef ref, {bool full = false}) { ref.read(backupProvider.notifier).cancelBackup(); final backgroundManager = ref.read(backgroundSyncProvider); final isAlbumLinkedSyncEnable = ref.read(appSettingsServiceProvider).getSetting(AppSettingsEnum.syncAlbums); return Future.wait([ backgroundManager.syncLocal(full: full).then((_) { Logger("runNewSync").fine("Hashing assets after syncLocal"); return backgroundManager.hashAssets(); }), backgroundManager.syncRemote().then((_) { if (isAlbumLinkedSyncEnable) { return backgroundManager.syncLinkedAlbum(); } }), ]); }