import 'package:logging/logging.dart'; import 'package:nc_photos/account.dart'; import 'package:nc_photos/ci_string.dart'; import 'package:nc_photos/entity/exif.dart'; import 'package:nc_photos/entity/file.dart'; import 'package:nc_photos/iterable_extension.dart'; import 'package:nc_photos/type.dart'; import 'package:tuple/tuple.dart'; abstract class AlbumUpgrader { JsonObj? call(JsonObj json); } /// Upgrade v1 Album to v2 class AlbumUpgraderV1 implements AlbumUpgrader { AlbumUpgraderV1({ this.logFilePath, }); @override call(JsonObj json) { // v1 album items are corrupted in one of the updates, drop it _log.fine("[call] Upgrade v1 Album for file: $logFilePath"); final result = JsonObj.from(json); result["items"] = []; return result; } /// File path for logging only final String? logFilePath; static final _log = Logger("entity.album.upgrader.AlbumUpgraderV1"); } /// Upgrade v2 Album to v3 class AlbumUpgraderV2 implements AlbumUpgrader { AlbumUpgraderV2({ this.logFilePath, }); @override call(JsonObj json) { // move v2 items to v3 provider _log.fine("[call] Upgrade v2 Album for file: $logFilePath"); final result = JsonObj.from(json); result["provider"] = { "type": "static", "content": { "items": result["items"], } }; result.remove("items"); // add the auto cover provider result["coverProvider"] = { "type": "auto", "content": {}, }; return result; } /// File path for logging only final String? logFilePath; static final _log = Logger("entity.album.upgrader.AlbumUpgraderV2"); } /// Upgrade v3 Album to v4 class AlbumUpgraderV3 implements AlbumUpgrader { AlbumUpgraderV3({ this.logFilePath, }); @override call(JsonObj json) { // move v3 items to v4 provider _log.fine("[call] Upgrade v3 Album for file: $logFilePath"); final result = JsonObj.from(json); // add the descending time sort provider result["sortProvider"] = { "type": "time", "content": { "isAscending": false, }, }; return result; } /// File path for logging only final String? logFilePath; static final _log = Logger("entity.album.upgrader.AlbumUpgraderV3"); } /// Upgrade v4 Album to v5 class AlbumUpgraderV4 implements AlbumUpgrader { AlbumUpgraderV4({ this.logFilePath, }); @override call(JsonObj json) { _log.fine("[call] Upgrade v4 Album for file: $logFilePath"); final result = JsonObj.from(json); try { if (result["provider"]["type"] != "static") { return result; } final latestItem = (result["provider"]["content"]["items"] as List) .map((e) => e.cast()) .where((e) => e["type"] == "file") .map((e) => e["content"]["file"] as JsonObj) .map((e) { final overrideDateTime = e["overrideDateTime"] == null ? null : DateTime.parse(e["overrideDateTime"]); final String? dateTimeOriginalStr = e["metadata"]?["exif"]?["DateTimeOriginal"]; final dateTimeOriginal = dateTimeOriginalStr == null || dateTimeOriginalStr.isEmpty ? null : Exif.dateTimeFormat.parse(dateTimeOriginalStr).toUtc(); final lastModified = e["lastModified"] == null ? null : DateTime.parse(e["lastModified"]); final latestItemTime = overrideDateTime ?? dateTimeOriginal ?? lastModified; // remove metadata e.remove("metadata"); if (latestItemTime != null) { return Tuple2(latestItemTime, e); } else { return null; } }) .whereType>() .sorted((a, b) => a.item1.compareTo(b.item1)) .lastOrNull; if (latestItem != null) { // save the latest item time result["provider"]["content"]["latestItemTime"] = latestItem.item1.toIso8601String(); if (result["coverProvider"]["type"] == "auto") { // save the cover result["coverProvider"]["content"]["coverFile"] = Map.of(latestItem.item2); } } } catch (e, stackTrace) { // this upgrade is not a must, if it failed then just leave it and it'll // be upgraded the next time the album is saved _log.shout("[call] Failed while upgrade", e, stackTrace); } return result; } /// File path for logging only final String? logFilePath; static final _log = Logger("entity.album.upgrader.AlbumUpgraderV4"); } /// Upgrade v5 Album to v6 class AlbumUpgraderV5 implements AlbumUpgrader { const AlbumUpgraderV5( this.account, { this.albumFile, this.logFilePath, }); @override call(JsonObj json) { _log.fine("[call] Upgrade v5 Album for file: $logFilePath"); final result = JsonObj.from(json); try { if (result["provider"]["type"] != "static") { return result; } for (final item in (result["provider"]["content"]["items"] as List)) { final CiString addedBy; if (result.containsKey("albumFile")) { addedBy = result["albumFile"]["ownerId"] == null ? account.username : CiString(result["albumFile"]["ownerId"]); } else { addedBy = albumFile?.ownerId ?? account.username; } item["addedBy"] = addedBy.toString(); item["addedAt"] = result["lastUpdated"]; } } catch (e, stackTrace) { // this upgrade is not a must, if it failed then just leave it and it'll // be upgraded the next time the album is saved _log.shout("[call] Failed while upgrade", e, stackTrace); } return result; } final Account account; final File? albumFile; /// File path for logging only final String? logFilePath; static final _log = Logger("entity.album.upgrader.AlbumUpgraderV5"); } abstract class AlbumUpgraderFactory { const AlbumUpgraderFactory(); AlbumUpgraderV1? buildV1(); AlbumUpgraderV2? buildV2(); AlbumUpgraderV3? buildV3(); AlbumUpgraderV4? buildV4(); AlbumUpgraderV5? buildV5(); } class DefaultAlbumUpgraderFactory extends AlbumUpgraderFactory { const DefaultAlbumUpgraderFactory({ required this.account, this.albumFile, this.logFilePath, }); @override buildV1() => AlbumUpgraderV1(logFilePath: logFilePath); @override buildV2() => AlbumUpgraderV2(logFilePath: logFilePath); @override buildV3() => AlbumUpgraderV3(logFilePath: logFilePath); @override buildV4() => AlbumUpgraderV4(logFilePath: logFilePath); @override buildV5() => AlbumUpgraderV5( account, albumFile: albumFile, logFilePath: logFilePath, ); final Account account; final File? albumFile; /// File path for logging only final String? logFilePath; }