mirror of
https://gitlab.com/nkming2/nc-photos.git
synced 2025-01-22 08:46:18 +01:00
Fix moved files are removed from DB in some case
This commit is contained in:
parent
8c85f98293
commit
5e06136b8a
2 changed files with 131 additions and 11 deletions
|
@ -129,14 +129,12 @@ class FileSqliteCacheUpdater {
|
||||||
throw StateError("Row ID for dir is null");
|
throw StateError("Row ID for dir is null");
|
||||||
}
|
}
|
||||||
|
|
||||||
final dirChildRowIdQuery = db.selectOnly(db.dirFiles)
|
final dirFileQuery = db.select(db.dirFiles)
|
||||||
..addColumns([db.dirFiles.child])
|
..where((t) => t.dir.equals(_dirRowId))
|
||||||
..where(db.dirFiles.dir.equals(_dirRowId))
|
..orderBy([(t) => sql.OrderingTerm.asc(t.rowId)]);
|
||||||
..orderBy([sql.OrderingTerm.asc(db.dirFiles.rowId)]);
|
final dirFiles = await dirFileQuery.get();
|
||||||
final dirChildRowIds =
|
final diff = list_util.diff(dirFiles.map((e) => e.child),
|
||||||
await dirChildRowIdQuery.map((r) => r.read(db.dirFiles.child)!).get();
|
_childRowIds.sorted(Comparable.compare));
|
||||||
final diff = list_util.diff(
|
|
||||||
dirChildRowIds, _childRowIds.sorted(Comparable.compare));
|
|
||||||
if (diff.item1.isNotEmpty) {
|
if (diff.item1.isNotEmpty) {
|
||||||
await db.batch((batch) {
|
await db.batch((batch) {
|
||||||
// insert new children
|
// insert new children
|
||||||
|
@ -145,9 +143,34 @@ class FileSqliteCacheUpdater {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
if (diff.item2.isNotEmpty) {
|
if (diff.item2.isNotEmpty) {
|
||||||
// delete obsolete children
|
// remove entries from the DirFiles table first
|
||||||
await _removeSqliteFiles(db, dbAccount, diff.item2);
|
await diff.item2.withPartitionNoReturn((sublist) async {
|
||||||
await db.cleanUpDanglingFiles();
|
final deleteQuery = db.delete(db.dirFiles)
|
||||||
|
..where((t) => t.child.isIn(sublist))
|
||||||
|
..where((t) =>
|
||||||
|
t.dir.equals(_dirRowId) | t.dir.equalsExp(db.dirFiles.child));
|
||||||
|
await deleteQuery.go();
|
||||||
|
}, sql.maxByFileIdsSize);
|
||||||
|
|
||||||
|
// select files having another dir parent under this account (i.e.,
|
||||||
|
// moved files)
|
||||||
|
final moved = await diff.item2.withPartition((sublist) async {
|
||||||
|
final query = db.selectOnly(db.dirFiles).join([
|
||||||
|
sql.innerJoin(db.accountFiles,
|
||||||
|
db.accountFiles.file.equalsExp(db.dirFiles.dir)),
|
||||||
|
]);
|
||||||
|
query
|
||||||
|
..addColumns([db.dirFiles.child])
|
||||||
|
..where(db.accountFiles.account.equals(dbAccount.rowId))
|
||||||
|
..where(db.dirFiles.child.isIn(sublist));
|
||||||
|
return query.map((r) => r.read(db.dirFiles.child)!).get();
|
||||||
|
}, sql.maxByFileIdsSize);
|
||||||
|
final removed = diff.item2.where((e) => !moved.contains(e)).toList();
|
||||||
|
if (removed.isNotEmpty) {
|
||||||
|
// delete obsolete children
|
||||||
|
await _removeSqliteFiles(db, dbAccount, removed);
|
||||||
|
await db.cleanUpDanglingFiles();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
|
@ -33,6 +33,9 @@ void main() {
|
||||||
test("too many files", _updaterTooManyFiles,
|
test("too many files", _updaterTooManyFiles,
|
||||||
timeout: const Timeout(Duration(minutes: 2)),
|
timeout: const Timeout(Duration(minutes: 2)),
|
||||||
skip: "too slow on gitlab");
|
skip: "too slow on gitlab");
|
||||||
|
test("moved file (to dir in front of the from dir)",
|
||||||
|
_updaterMovedFileToFront);
|
||||||
|
test("moved file (to dir behind the from dir)", _updaterMovedFileToBehind);
|
||||||
});
|
});
|
||||||
test("FileSqliteCacheEmptier", _emptier);
|
test("FileSqliteCacheEmptier", _emptier);
|
||||||
}
|
}
|
||||||
|
@ -541,6 +544,100 @@ Future<void> _updaterTooManyFiles() async {
|
||||||
// expect here
|
// expect here
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Moved a file from test2 to test1, where test2 is sorted behind test1
|
||||||
|
///
|
||||||
|
/// Expect: file moved
|
||||||
|
Future<void> _updaterMovedFileToFront() async {
|
||||||
|
final account = util.buildAccount();
|
||||||
|
final files = (util.FilesBuilder()
|
||||||
|
..addDir("admin")
|
||||||
|
..addDir("admin/test1")
|
||||||
|
..addDir("admin/test2")
|
||||||
|
..addJpeg("admin/test2/test1.jpg"))
|
||||||
|
.build();
|
||||||
|
final c = DiContainer(
|
||||||
|
sqliteDb: util.buildTestDb(),
|
||||||
|
);
|
||||||
|
addTearDown(() => c.sqliteDb.close());
|
||||||
|
await c.sqliteDb.transaction(() async {
|
||||||
|
await c.sqliteDb.insertAccountOf(account);
|
||||||
|
await util.insertFiles(c.sqliteDb, account, files);
|
||||||
|
await util.insertDirRelation(
|
||||||
|
c.sqliteDb, account, files[0], files.slice(1, 3));
|
||||||
|
await util.insertDirRelation(c.sqliteDb, account, files[1], []);
|
||||||
|
await util.insertDirRelation(c.sqliteDb, account, files[2], [files[3]]);
|
||||||
|
});
|
||||||
|
|
||||||
|
final movedFile = files[3].copyWith(
|
||||||
|
path: "remote.php/dav/files/admin/test1/test1.jpg",
|
||||||
|
);
|
||||||
|
await FileSqliteCacheUpdater(c)(
|
||||||
|
account,
|
||||||
|
files[1],
|
||||||
|
remote: [files[1], movedFile],
|
||||||
|
);
|
||||||
|
await FileSqliteCacheUpdater(c)(
|
||||||
|
account,
|
||||||
|
files[2],
|
||||||
|
remote: [files[2]],
|
||||||
|
);
|
||||||
|
expect(
|
||||||
|
await util.listSqliteDbFiles(c.sqliteDb),
|
||||||
|
{...files.slice(0, 3), movedFile},
|
||||||
|
);
|
||||||
|
final dirResult = await util.listSqliteDbDirs(c.sqliteDb);
|
||||||
|
expect(dirResult[files[0]], {...files.slice(0, 3)});
|
||||||
|
expect(dirResult[files[1]], {files[1], movedFile});
|
||||||
|
expect(dirResult[files[2]], {files[2]});
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Moved a file from test1 to test2, where test1 is sorted in front of test2
|
||||||
|
///
|
||||||
|
/// Expect: file moved
|
||||||
|
Future<void> _updaterMovedFileToBehind() async {
|
||||||
|
final account = util.buildAccount();
|
||||||
|
final files = (util.FilesBuilder()
|
||||||
|
..addDir("admin")
|
||||||
|
..addDir("admin/test1")
|
||||||
|
..addDir("admin/test2")
|
||||||
|
..addJpeg("admin/test1/test1.jpg"))
|
||||||
|
.build();
|
||||||
|
final c = DiContainer(
|
||||||
|
sqliteDb: util.buildTestDb(),
|
||||||
|
);
|
||||||
|
addTearDown(() => c.sqliteDb.close());
|
||||||
|
await c.sqliteDb.transaction(() async {
|
||||||
|
await c.sqliteDb.insertAccountOf(account);
|
||||||
|
await util.insertFiles(c.sqliteDb, account, files);
|
||||||
|
await util.insertDirRelation(
|
||||||
|
c.sqliteDb, account, files[0], files.slice(1, 3));
|
||||||
|
await util.insertDirRelation(c.sqliteDb, account, files[1], [files[3]]);
|
||||||
|
await util.insertDirRelation(c.sqliteDb, account, files[2], []);
|
||||||
|
});
|
||||||
|
|
||||||
|
final movedFile = files[3].copyWith(
|
||||||
|
path: "remote.php/dav/files/admin/test2/test1.jpg",
|
||||||
|
);
|
||||||
|
await FileSqliteCacheUpdater(c)(
|
||||||
|
account,
|
||||||
|
files[1],
|
||||||
|
remote: [files[1]],
|
||||||
|
);
|
||||||
|
await FileSqliteCacheUpdater(c)(
|
||||||
|
account,
|
||||||
|
files[2],
|
||||||
|
remote: [files[2], movedFile],
|
||||||
|
);
|
||||||
|
expect(
|
||||||
|
await util.listSqliteDbFiles(c.sqliteDb),
|
||||||
|
{...files.slice(0, 3), movedFile},
|
||||||
|
);
|
||||||
|
final dirResult = await util.listSqliteDbDirs(c.sqliteDb);
|
||||||
|
expect(dirResult[files[0]], {...files.slice(0, 3)});
|
||||||
|
expect(dirResult[files[1]], {files[1]});
|
||||||
|
expect(dirResult[files[2]], {files[2], movedFile});
|
||||||
|
}
|
||||||
|
|
||||||
/// Empty dir in cache
|
/// Empty dir in cache
|
||||||
///
|
///
|
||||||
/// Expect: dir removed from DirFiles table;
|
/// Expect: dir removed from DirFiles table;
|
||||||
|
|
Loading…
Reference in a new issue