diff --git a/src/persistence/dbupgrader.cpp b/src/persistence/dbupgrader.cpp index 848da07d3..b7fbe9bd2 100644 --- a/src/persistence/dbupgrader.cpp +++ b/src/persistence/dbupgrader.cpp @@ -26,7 +26,7 @@ #include namespace { -constexpr int SCHEMA_VERSION = 9; +constexpr int SCHEMA_VERSION = 10; struct BadEntry { @@ -262,7 +262,8 @@ bool DbUpgrader::dbSchemaUpgrade(std::shared_ptr& db) using DbSchemaUpgradeFn = bool (*)(RawDatabase&); std::vector upgradeFns = {dbSchema0to1, dbSchema1to2, dbSchema2to3, dbSchema3to4, dbSchema4to5, dbSchema5to6, - dbSchema6to7, dbSchema7to8, dbSchema8to9}; + dbSchema6to7, dbSchema7to8, dbSchema8to9, + dbSchema9to10}; assert(databaseSchemaVersion < static_cast(upgradeFns.size())); assert(upgradeFns.size() == SCHEMA_VERSION); @@ -610,3 +611,18 @@ bool DbUpgrader::dbSchema8to9(RawDatabase& db) upgradeQueries += RawDatabase::Query(QStringLiteral("PRAGMA user_version = 9;")); return db.execNow(upgradeQueries); } + +bool DbUpgrader::dbSchema9to10(RawDatabase& db) +{ + // not technically a schema update, but still a database version update based on healing invalid user data. + // We inserted some empty resume_file_id's due to #6553. Heal the under-length entries. + // The resume file ID isn't actually used for loaded files at this time, so we can heal it to an arbitrary + // value of full length. + constexpr int resumeFileIdLengthNow = 32; + QByteArray dummyResumeId(resumeFileIdLengthNow, 0); + QVector upgradeQueries; + upgradeQueries += RawDatabase::Query(QStringLiteral("UPDATE file_transfers SET file_restart_id = ? WHERE LENGTH(file_restart_id) != 32;"), + {dummyResumeId}); + upgradeQueries += RawDatabase::Query(QStringLiteral("PRAGMA user_version = 10;")); + return db.execNow(upgradeQueries); +} diff --git a/src/persistence/dbupgrader.h b/src/persistence/dbupgrader.h index f4af0e6da..b6fdb709f 100644 --- a/src/persistence/dbupgrader.h +++ b/src/persistence/dbupgrader.h @@ -37,4 +37,5 @@ namespace DbUpgrader bool dbSchema6to7(RawDatabase& db); bool dbSchema7to8(RawDatabase& db); bool dbSchema8to9(RawDatabase& db); + bool dbSchema9to10(RawDatabase& db); } diff --git a/test/persistence/dbschema_test.cpp b/test/persistence/dbschema_test.cpp index d22b924fc..26cb3d884 100644 --- a/test/persistence/dbschema_test.cpp +++ b/test/persistence/dbschema_test.cpp @@ -19,6 +19,7 @@ #include "src/persistence/db/rawdatabase.h" #include "src/persistence/dbupgrader.h" +#include "src/core/toxfile.h" #include #include @@ -26,6 +27,47 @@ #include #include +namespace { +bool insertFileId(RawDatabase& db, int row, bool valid) +{ + QByteArray validResumeId(32, 1); + QByteArray invalidResumeId; + + QByteArray resumeId; + if (valid) { + resumeId = validResumeId; + } else { + resumeId = invalidResumeId; + } + + QVector upgradeQueries; + upgradeQueries += RawDatabase::Query( + QString("INSERT INTO file_transfers " + " (id, message_type, sender_alias, " + " file_restart_id, file_name, file_path, " + " file_hash, file_size, direction, file_state) " + "VALUES ( " + " %1, " + " 'F', " + " 1, " + " ?, " + " %2, " + " %3, " + " %4, " + " 1, " + " 1, " + " %5 " + ");") + .arg(row) + .arg("\"fooname\"") + .arg("\"foo/path\"") + .arg("\"foohash\"") + .arg(ToxFile::CANCELED) + , {resumeId}); + return db.execNow(upgradeQueries); +} +} // namespace + struct SqliteMasterEntry { QString name; QString sql; @@ -54,6 +96,7 @@ private slots: void test6to7(); // test7to8 omitted, version only upgrade, versions are not verified in this // test8to9 omitted, data corruption correction upgrade with no schema change + void test9to10(); // test suite void cleanupTestCase() const; @@ -66,7 +109,7 @@ private: const QString testFileList[] = {"testCreation.db", "testIsNewDbTrue.db", "testIsNewDbFalse.db", "test0to1.db", "test1to2.db", "test2to3.db", "test3to4.db", "test4to5.db", "test5to6.db", - "test6to7.db"}; + "test6to7.db", "test9to10.db"}; // db schemas can be select with "SELECT name, sql FROM sqlite_master;" on the database. @@ -162,6 +205,8 @@ const std::vector schema7{ "FOREIGN KEY (id, message_type) REFERENCES history(id, message_type))"}, {"chat_id_idx", "CREATE INDEX chat_id_idx on history (chat_id)"}}; +const std::vector schema9 = schema7; +const std::vector schema10 = schema9; void TestDbSchema::initTestCase() { @@ -434,5 +479,34 @@ void TestDbSchema::test6to7() verifyDb(db, schema7); } +void TestDbSchema::test9to10() +{ + auto db = std::shared_ptr{new RawDatabase{"test9to10.db", {}, {}}}; + createSchemaAtVersion(db, schema9); + + QVERIFY(insertFileId(*db, 1, true)); + QVERIFY(insertFileId(*db, 2, true)); + QVERIFY(insertFileId(*db, 3, false)); + QVERIFY(insertFileId(*db, 4, true)); + QVERIFY(insertFileId(*db, 5, false)); + QVERIFY(DbUpgrader::dbSchema9to10(*db)); + int numHealed = 0; + int numUnchanged = 0; + QVERIFY(db->execNow(RawDatabase::Query("SELECT file_restart_id from file_transfers;", + [&](const QVector& row) { + auto resumeId = row[0].toByteArray(); + if (resumeId == QByteArray(32, 0)) { + ++numHealed; + } else if (resumeId == QByteArray(32, 1)) { + ++numUnchanged; + } else { + QFAIL("Invalid file_restart_id"); + } + }))); + QVERIFY(numHealed == 2); + QVERIFY(numUnchanged == 3); + verifyDb(db, schema10); +} + QTEST_GUILESS_MAIN(TestDbSchema) #include "dbschema_test.moc"