import 'dart:convert'; import 'dart:math'; import 'package:equatable/equatable.dart'; import 'package:idb_sqflite/idb_sqflite.dart'; import 'package:kiwi/kiwi.dart'; import 'package:logging/logging.dart'; import 'package:nc_photos/account.dart'; import 'package:nc_photos/app_db.dart'; import 'package:nc_photos/ci_string.dart'; import 'package:nc_photos/di_container.dart'; import 'package:nc_photos/entity/album/cover_provider.dart'; import 'package:nc_photos/entity/album/item.dart'; import 'package:nc_photos/entity/album/provider.dart'; import 'package:nc_photos/entity/album/sort_provider.dart'; import 'package:nc_photos/entity/album/upgrader.dart'; import 'package:nc_photos/entity/file.dart'; import 'package:nc_photos/entity/file/data_source.dart'; import 'package:nc_photos/exception.dart'; import 'package:nc_photos/int_util.dart' as int_util; import 'package:nc_photos/iterable_extension.dart'; import 'package:nc_photos/object_extension.dart'; import 'package:nc_photos/or_null.dart'; import 'package:nc_photos/remote_storage_util.dart' as remote_storage_util; import 'package:nc_photos/type.dart'; import 'package:nc_photos/use_case/get_file_binary.dart'; import 'package:nc_photos/use_case/ls_single_file.dart'; import 'package:nc_photos/use_case/put_file_binary.dart'; import 'package:quiver/iterables.dart'; import 'package:tuple/tuple.dart'; /// Immutable object that represents an album class Album with EquatableMixin { /// Create a new album /// /// If [lastUpdated] is null, the current time will be used. /// /// [savedVersion] should be null when creating a new album, such that it'll /// be filled with the current version number automatically. You should only /// pass this argument when reading album from storage Album({ DateTime? lastUpdated, required this.name, required this.provider, required this.coverProvider, required this.sortProvider, this.shares, this.albumFile, int? savedVersion, }) : lastUpdated = (lastUpdated ?? DateTime.now()).toUtc(), savedVersion = savedVersion ?? version; static Album? fromJson( JsonObj json, { required AlbumUpgraderFactory? upgraderFactory, }) { final jsonVersion = json["version"]; JsonObj? result = json; if (jsonVersion < 2) { result = upgraderFactory?.buildV1()?.call(result); if (result == null) { _log.info("[fromJson] Version $jsonVersion not compatible"); return null; } } if (jsonVersion < 3) { result = upgraderFactory?.buildV2()?.call(result); if (result == null) { _log.info("[fromJson] Version $jsonVersion not compatible"); return null; } } if (jsonVersion < 4) { result = upgraderFactory?.buildV3()?.call(result); if (result == null) { _log.info("[fromJson] Version $jsonVersion not compatible"); return null; } } if (jsonVersion < 5) { result = upgraderFactory?.buildV4()?.call(result); if (result == null) { _log.info("[fromJson] Version $jsonVersion not compatible"); return null; } } if (jsonVersion < 6) { result = upgraderFactory?.buildV5()?.call(result); if (result == null) { _log.info("[fromJson] Version $jsonVersion not compatible"); return null; } } if (jsonVersion < 7) { result = upgraderFactory?.buildV6()?.call(result); if (result == null) { _log.info("[fromJson] Version $jsonVersion not compatible"); return null; } } if (jsonVersion < 8) { result = upgraderFactory?.buildV7()?.call(result); if (result == null) { _log.info("[fromJson] Version $jsonVersion not compatible"); return null; } } if (jsonVersion > version) { _log.warning( "[fromJson] Reading album with newer version: $jsonVersion > $version"); } return Album( lastUpdated: result["lastUpdated"] == null ? null : DateTime.parse(result["lastUpdated"]), name: result["name"], provider: AlbumProvider.fromJson(result["provider"].cast()), coverProvider: AlbumCoverProvider.fromJson( result["coverProvider"].cast()), sortProvider: AlbumSortProvider.fromJson( result["sortProvider"].cast()), shares: (result["shares"] as List?) ?.map((e) => AlbumShare.fromJson(e.cast())) .toList(), albumFile: result["albumFile"] == null ? null : File.fromJson(result["albumFile"].cast()), savedVersion: result["version"], ); } @override toString({bool isDeep = false}) { return "$runtimeType {" "lastUpdated: $lastUpdated, " "name: $name, " "provider: ${provider.toString(isDeep: isDeep)}, " "coverProvider: $coverProvider, " "sortProvider: $sortProvider, " "shares: ${shares?.toReadableString()}, " "albumFile: $albumFile, " "}"; } /// Return a copy with specified field modified /// /// [lastUpdated] is handled differently where if not set, the current time /// will be used. In order to keep [lastUpdated], you must explicitly assign /// it with value from this or a null value Album copyWith({ OrNull? lastUpdated, String? name, AlbumProvider? provider, AlbumCoverProvider? coverProvider, AlbumSortProvider? sortProvider, OrNull>? shares, OrNull? albumFile, }) { return Album( lastUpdated: lastUpdated == null ? null : (lastUpdated.obj ?? this.lastUpdated), name: name ?? this.name, provider: provider ?? this.provider, coverProvider: coverProvider ?? this.coverProvider, sortProvider: sortProvider ?? this.sortProvider, shares: shares == null ? this.shares?.run((obj) => List.of(obj)) : shares.obj, albumFile: albumFile == null ? this.albumFile : albumFile.obj, savedVersion: savedVersion, ); } JsonObj toRemoteJson() { return { "version": version, "lastUpdated": lastUpdated.toIso8601String(), "name": name, "provider": provider.toJson(), "coverProvider": coverProvider.toJson(), "sortProvider": sortProvider.toJson(), if (shares != null) "shares": shares!.map((e) => e.toJson()).toList(), // ignore albumFile }; } JsonObj toAppDbJson() { return { "version": version, "lastUpdated": lastUpdated.toIso8601String(), "name": name, "provider": provider.toJson(), "coverProvider": coverProvider.toJson(), "sortProvider": sortProvider.toJson(), if (shares != null) "shares": shares!.map((e) => e.toJson()).toList(), if (albumFile != null) "albumFile": albumFile!.toJson(), }; } @override get props => [ lastUpdated, name, provider, coverProvider, sortProvider, shares, albumFile, savedVersion, ]; final DateTime lastUpdated; final String name; final AlbumProvider provider; final AlbumCoverProvider coverProvider; final AlbumSortProvider sortProvider; final List? shares; /// How is this album stored on server /// /// This field is typically only meaningful when returned by [AlbumRepo.get] final File? albumFile; /// The original version of this class when saved /// /// This field only exists in runtime and are not persisted final int savedVersion; /// versioning of this class, use to upgrade old persisted album static const version = 8; } class AlbumShare with EquatableMixin { AlbumShare({ required this.userId, this.displayName, DateTime? sharedAt, }) : sharedAt = (sharedAt ?? DateTime.now()).toUtc(); factory AlbumShare.fromJson(JsonObj json) { return AlbumShare( userId: CiString(json["userId"]), displayName: json["displayName"], sharedAt: DateTime.parse(json["sharedAt"]), ); } JsonObj toJson() { return { "userId": userId.toString(), if (displayName != null) "displayName": displayName, "sharedAt": sharedAt.toIso8601String(), }; } /// Return a copy with specified field modified /// /// [sharedAt] is handled differently where if not set, the current time will /// be used. In order to keep [sharedAt], you must explicitly assign it with /// value from this or a null value AlbumShare copyWith({ CiString? userId, OrNull? displayName, OrNull? sharedAt, }) { return AlbumShare( userId: userId ?? this.userId, displayName: displayName == null ? this.displayName : displayName.obj, sharedAt: sharedAt == null ? null : (sharedAt.obj ?? this.sharedAt), ); } @override toString() { return "$runtimeType {" "userId: $userId, " "displayName: $displayName, " "sharedAt: $sharedAt, " "}"; } @override get props => [ userId, sharedAt, ]; /// User ID or username, case insensitive final CiString userId; final String? displayName; final DateTime sharedAt; } class AlbumRepo { AlbumRepo(this.dataSrc); /// See [AlbumDataSource.get] Future get(Account account, File albumFile) => dataSrc.get(account, albumFile); /// See [AlbumDataSource.create] Future create(Account account, Album album) => dataSrc.create(account, album); /// See [AlbumDataSource.update] Future update(Account account, Album album) => dataSrc.update(account, album); /// See [AlbumDataSource.cleanUp] Future cleanUp( Account account, String rootDir, List albumFiles) => dataSrc.cleanUp(account, rootDir, albumFiles); final AlbumDataSource dataSrc; } abstract class AlbumDataSource { /// Return the album defined by [albumFile] Future get(Account account, File albumFile); // Create a new album Future create(Account account, Album album); /// Update an album Future update(Account account, Album album); /// Clean up cached albums /// /// Remove dangling albums in cache not listed in [albumFiles] and located /// inside [rootDir]. Do nothing if this data source does not cache previous /// results Future cleanUp(Account account, String rootDir, List albumFiles); } class AlbumRemoteDataSource implements AlbumDataSource { @override get(Account account, File albumFile) async { _log.info("[get] ${albumFile.path}"); const fileRepo = FileRepo(FileWebdavDataSource()); final data = await GetFileBinary(fileRepo)(account, albumFile); try { return Album.fromJson( jsonDecode(utf8.decode(data)), upgraderFactory: DefaultAlbumUpgraderFactory( account: account, albumFile: albumFile, logFilePath: albumFile.path, ), )! .copyWith( lastUpdated: OrNull(null), albumFile: OrNull(albumFile), ); } catch (e, stacktrace) { dynamic d = data; try { d = utf8.decode(data); } catch (_) {} _log.severe("[get] Invalid json data: $d", e, stacktrace); throw const FormatException("Invalid album format"); } } @override create(Account account, Album album) async { _log.info("[create]"); final fileName = _makeAlbumFileName(); final filePath = "${remote_storage_util.getRemoteAlbumsDir(account)}/$fileName"; const fileRepo = FileRepo(FileWebdavDataSource()); await PutFileBinary(fileRepo)(account, filePath, const Utf8Encoder().convert(jsonEncode(album.toRemoteJson())), shouldCreateMissingDir: true); // query album file final newFile = await LsSingleFile(KiwiContainer().resolve())( account, filePath); return album.copyWith(albumFile: OrNull(newFile)); } @override update(Account account, Album album) async { _log.info("[update] ${album.albumFile!.path}"); const fileRepo = FileRepo(FileWebdavDataSource()); await PutFileBinary(fileRepo)(account, album.albumFile!.path, const Utf8Encoder().convert(jsonEncode(album.toRemoteJson()))); } @override cleanUp(Account account, String rootDir, List albumFiles) async {} String _makeAlbumFileName() { // just make up something final timestamp = DateTime.now().millisecondsSinceEpoch; final random = Random().nextInt(0xFFFFFF); return "${timestamp.toRadixString(16)}-${random.toRadixString(16).padLeft(6, '0')}.nc_album.json"; } static final _log = Logger("entity.album.AlbumRemoteDataSource"); } class AlbumAppDbDataSource implements AlbumDataSource { const AlbumAppDbDataSource(this.appDb); @override get(Account account, File albumFile) { _log.info("[get] ${albumFile.path}"); return appDb.use( (db) => db.transaction(AppDb.albumStoreName, idbModeReadOnly), (transaction) async { final store = transaction.objectStore(AppDb.albumStoreName); final index = store.index(AppDbAlbumEntry.indexName); final path = AppDbAlbumEntry.toPathFromFile(account, albumFile); final range = KeyRange.bound([path, 0], [path, int_util.int32Max]); final List results = await index.getAll(range); if (results.isNotEmpty == true) { final entries = results.map((e) => AppDbAlbumEntry.fromJson(e.cast(), account)); if (entries.length > 1) { final items = entries.map((e) { _log.info("[get] ${e.path}[${e.index}]"); return AlbumStaticProvider.of(e.album).items; }).reduce((value, element) => value + element); return entries.first.album.copyWith( lastUpdated: OrNull(null), provider: AlbumStaticProvider.of(entries.first.album).copyWith( items: items, ), ); } else { return entries.first.album; } } else { throw CacheNotFoundException("No entry: $path"); } }, ); } @override create(Account account, Album album) async { _log.info("[create]"); throw UnimplementedError(); } @override update(Account account, Album album) { _log.info("[update] ${album.albumFile!.path}"); return appDb.use( (db) => db.transaction(AppDb.albumStoreName, idbModeReadWrite), (transaction) async { final store = transaction.objectStore(AppDb.albumStoreName); await _cacheAlbum(store, account, album); }, ); } @override cleanUp(Account account, String rootDir, List albumFiles) async {} final AppDb appDb; static final _log = Logger("entity.album.AlbumAppDbDataSource"); } class AlbumCachedDataSource implements AlbumDataSource { AlbumCachedDataSource(this.appDb) : _appDbSrc = AlbumAppDbDataSource(appDb); @override get(Account account, File albumFile) async { try { final cache = await _appDbSrc.get(account, albumFile); if (cache.albumFile!.etag?.isNotEmpty == true && cache.albumFile!.etag == albumFile.etag) { // cache is good _log.fine( "[get] etag matched for ${AppDbAlbumEntry.toPathFromFile(account, albumFile)}"); return cache; } _log.info( "[get] Remote content updated for ${AppDbAlbumEntry.toPathFromFile(account, albumFile)}"); } on CacheNotFoundException catch (_) { // normal when there's no cache } catch (e, stacktrace) { _log.shout("[get] Cache failure", e, stacktrace); } // no cache final remote = await _remoteSrc.get(account, albumFile); await _cacheResult(account, remote); return remote; } @override update(Account account, Album album) async { await _remoteSrc.update(account, album); await _appDbSrc.update(account, album); } @override create(Account account, Album album) => _remoteSrc.create(account, album); @override cleanUp(Account account, String rootDir, List albumFiles) async { appDb.use( (db) => db.transaction(AppDb.albumStoreName, idbModeReadWrite), (transaction) async { final store = transaction.objectStore(AppDb.albumStoreName); final index = store.index(AppDbAlbumEntry.indexName); final rootPath = AppDbAlbumEntry.toPath(account, rootDir); final range = KeyRange.bound( ["$rootPath/", 0], ["$rootPath/\uffff", int_util.int32Max]); final danglingKeys = await index // get all albums for this account .openKeyCursor(range: range, autoAdvance: true) .map((cursor) => Tuple2((cursor.key as List)[0], cursor.primaryKey)) // and pick the dangling ones .where((pair) => !albumFiles.any((f) => pair.item1 == AppDbAlbumEntry.toPathFromFile(account, f))) // map to primary keys .map((pair) => pair.item2) .toList(); for (final k in danglingKeys) { _log.fine("[cleanUp] Removing albumStore entry: $k"); try { await store.delete(k); } catch (e, stackTrace) { _log.shout( "[cleanUp] Failed removing albumStore entry", e, stackTrace); } } }, ); } Future _cacheResult(Account account, Album result) { return appDb.use( (db) => db.transaction(AppDb.albumStoreName, idbModeReadWrite), (transaction) async { final store = transaction.objectStore(AppDb.albumStoreName); await _cacheAlbum(store, account, result); }, ); } final AppDb appDb; final _remoteSrc = AlbumRemoteDataSource(); final AlbumAppDbDataSource _appDbSrc; static final _log = Logger("entity.album.AlbumCachedDataSource"); } Future _cacheAlbum( ObjectStore store, Account account, Album album) async { final index = store.index(AppDbAlbumEntry.indexName); final path = AppDbAlbumEntry.toPathFromFile(account, album.albumFile!); final range = KeyRange.bound([path, 0], [path, int_util.int32Max]); // count number of entries for this album final count = await index.count(range); // cut large album into smaller pieces, needed to workaround Android DB // limitation final entries = []; if (album.provider is AlbumStaticProvider) { var albumItemLists = partition( AlbumStaticProvider.of(album).items, AppDbAlbumEntry.maxDataSize) .toList(); if (albumItemLists.isEmpty) { albumItemLists = [[]]; } entries.addAll(albumItemLists.withIndex().map((pair) => AppDbAlbumEntry( path, pair.item1, album.copyWith( lastUpdated: OrNull(null), provider: AlbumStaticProvider.of(album).copyWith( items: pair.item2, ), )))); } else { entries.add(AppDbAlbumEntry(path, 0, album)); } for (final e in entries) { _log.info("[_cacheAlbum] Caching ${e.path}[${e.index}]"); await store.put(e.toJson(), AppDbAlbumEntry.toPrimaryKey(account, e.album.albumFile!, e.index)); } if (count > entries.length) { // index is 0-based final rmRange = KeyRange.bound([path, entries.length], [path, int_util.int32Max]); final rmKeys = await index .openKeyCursor(range: rmRange, autoAdvance: true) .map((cursor) => cursor.primaryKey) .toList(); for (final k in rmKeys) { _log.fine("[_cacheAlbum] Removing albumStore entry: $k"); try { await store.delete(k); } catch (e, stackTrace) { _log.shout( "[_cacheAlbum] Failed removing albumStore entry", e, stackTrace); } } } } final _log = Logger("entity.album");