nc-photos/app/lib/entity/album/upgrader.dart
2022-05-26 12:01:12 +08:00

294 lines
7.9 KiB
Dart

import 'package:collection/collection.dart';
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/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"] = <String, dynamic>{
"type": "static",
"content": <String, dynamic>{
"items": result["items"],
}
};
result.remove("items");
// add the auto cover provider
result["coverProvider"] = <String, dynamic>{
"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"] = <String, dynamic>{
"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<String, dynamic>())
.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<Tuple2<DateTime, JsonObj>>()
.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");
}
/// Upgrade v6 Album to v7
class AlbumUpgraderV6 implements AlbumUpgrader {
const AlbumUpgraderV6({
this.logFilePath,
});
@override
call(JsonObj json) {
_log.fine("[call] Upgrade v6 Album for file: $logFilePath");
return json;
}
static final _log = Logger("entity.album.upgrader.AlbumUpgraderV6");
/// File path for logging only
final String? logFilePath;
}
/// Upgrade v7 Album to v8
class AlbumUpgraderV7 implements AlbumUpgrader {
const AlbumUpgraderV7({
this.logFilePath,
});
@override
call(JsonObj json) {
_log.fine("[call] Upgrade v7 Album for file: $logFilePath");
return json;
}
static final _log = Logger("entity.album.upgrader.AlbumUpgraderV7");
/// File path for logging only
final String? logFilePath;
}
abstract class AlbumUpgraderFactory {
const AlbumUpgraderFactory();
AlbumUpgraderV1? buildV1();
AlbumUpgraderV2? buildV2();
AlbumUpgraderV3? buildV3();
AlbumUpgraderV4? buildV4();
AlbumUpgraderV5? buildV5();
AlbumUpgraderV6? buildV6();
AlbumUpgraderV7? buildV7();
}
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,
);
@override
buildV6() => AlbumUpgraderV6(logFilePath: logFilePath);
@override
buildV7() => AlbumUpgraderV7(logFilePath: logFilePath);
final Account account;
final File? albumFile;
/// File path for logging only
final String? logFilePath;
}