2022-11-06 17:34:54 +01:00
|
|
|
import 'package:drift/drift.dart' as sql;
|
|
|
|
import 'package:flutter/foundation.dart';
|
|
|
|
import 'package:logging/logging.dart';
|
|
|
|
import 'package:nc_photos/entity/file_util.dart' as file_util;
|
2023-02-20 15:21:35 +01:00
|
|
|
import 'package:nc_photos/entity/sqlite/database.dart' as sql;
|
2022-12-16 16:01:04 +01:00
|
|
|
import 'package:np_codegen/np_codegen.dart';
|
2023-08-25 19:31:06 +02:00
|
|
|
import 'package:np_collection/np_collection.dart';
|
2022-11-06 17:34:54 +01:00
|
|
|
import 'package:tuple/tuple.dart';
|
|
|
|
|
2022-12-16 16:01:04 +01:00
|
|
|
part 'v55.g.dart';
|
|
|
|
|
|
|
|
@npLog
|
2022-11-06 17:34:54 +01:00
|
|
|
class CompatV55 {
|
|
|
|
static Future<void> migrateDb(
|
|
|
|
sql.SqliteDb db, {
|
|
|
|
void Function(int current, int count)? onProgress,
|
|
|
|
}) {
|
|
|
|
return db.use((db) async {
|
|
|
|
final countExp = db.accountFiles.rowId.count();
|
|
|
|
final countQ = db.selectOnly(db.accountFiles)..addColumns([countExp]);
|
2023-07-10 19:30:56 +02:00
|
|
|
final count = await countQ.map((r) => r.read(countExp)!).getSingle();
|
2022-11-06 17:34:54 +01:00
|
|
|
onProgress?.call(0, count);
|
|
|
|
|
2022-11-19 14:16:56 +01:00
|
|
|
final dateTimeUpdates = <Tuple2<int, DateTime>>[];
|
|
|
|
final imageRemoves = <int>[];
|
2022-11-06 17:34:54 +01:00
|
|
|
for (var i = 0; i < count; i += 1000) {
|
|
|
|
final q = db.select(db.files).join([
|
|
|
|
sql.innerJoin(
|
|
|
|
db.accountFiles, db.accountFiles.file.equalsExp(db.files.rowId)),
|
|
|
|
sql.innerJoin(db.images,
|
|
|
|
db.images.accountFile.equalsExp(db.accountFiles.rowId)),
|
|
|
|
]);
|
|
|
|
q
|
|
|
|
..orderBy([
|
|
|
|
sql.OrderingTerm(
|
|
|
|
expression: db.accountFiles.rowId,
|
|
|
|
mode: sql.OrderingMode.asc,
|
|
|
|
),
|
|
|
|
])
|
|
|
|
..limit(1000, offset: i);
|
|
|
|
final dbFiles = await q
|
|
|
|
.map((r) => sql.CompleteFile(
|
|
|
|
r.readTable(db.files),
|
|
|
|
r.readTable(db.accountFiles),
|
|
|
|
r.readTable(db.images),
|
|
|
|
null,
|
|
|
|
null,
|
|
|
|
))
|
|
|
|
.get();
|
|
|
|
for (final f in dbFiles) {
|
|
|
|
final bestDateTime = file_util.getBestDateTime(
|
|
|
|
overrideDateTime: f.accountFile.overrideDateTime,
|
|
|
|
dateTimeOriginal: f.image?.dateTimeOriginal,
|
|
|
|
lastModified: f.file.lastModified,
|
|
|
|
);
|
|
|
|
if (f.accountFile.bestDateTime != bestDateTime) {
|
|
|
|
// need update
|
2022-11-19 14:16:56 +01:00
|
|
|
dateTimeUpdates.add(Tuple2(f.accountFile.rowId, bestDateTime));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (f.file.contentType == "image/heic" &&
|
|
|
|
f.image != null &&
|
|
|
|
f.image!.exifRaw == null) {
|
|
|
|
imageRemoves.add(f.accountFile.rowId);
|
2022-11-06 17:34:54 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
onProgress?.call(i, count);
|
|
|
|
}
|
|
|
|
|
2022-11-19 14:16:56 +01:00
|
|
|
_log.info(
|
|
|
|
"[migrateDb] ${dateTimeUpdates.length} rows require updating, ${imageRemoves.length} rows require removing");
|
2022-11-06 17:34:54 +01:00
|
|
|
if (kDebugMode) {
|
|
|
|
_log.fine(
|
2022-11-19 14:16:56 +01:00
|
|
|
"[migrateDb] dateTimeUpdates: ${dateTimeUpdates.map((e) => e.item1).toReadableString()}");
|
|
|
|
_log.fine(
|
|
|
|
"[migrateDb] imageRemoves: ${imageRemoves.map((e) => e).toReadableString()}");
|
2022-11-06 17:34:54 +01:00
|
|
|
}
|
|
|
|
await db.batch((batch) {
|
2022-11-19 14:16:56 +01:00
|
|
|
for (final pair in dateTimeUpdates) {
|
2022-11-06 17:34:54 +01:00
|
|
|
batch.update(
|
|
|
|
db.accountFiles,
|
|
|
|
sql.AccountFilesCompanion(
|
|
|
|
bestDateTime: sql.Value(pair.item2),
|
|
|
|
),
|
|
|
|
where: (sql.$AccountFilesTable table) =>
|
|
|
|
table.rowId.equals(pair.item1),
|
|
|
|
);
|
|
|
|
}
|
2022-11-19 14:16:56 +01:00
|
|
|
for (final r in imageRemoves) {
|
|
|
|
batch.deleteWhere(
|
|
|
|
db.images,
|
|
|
|
(sql.$ImagesTable table) => table.accountFile.equals(r),
|
|
|
|
);
|
|
|
|
}
|
2022-11-06 17:34:54 +01:00
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-12-20 17:49:14 +01:00
|
|
|
static final _log = _$CompatV55NpLog.log;
|
2022-11-06 17:34:54 +01:00
|
|
|
}
|