123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271 |
- /*
- * Copyright (C) 2017 Vladimir Svyatski <v.unreal@gmail.com>
- * Copyright (C) 2017 KeePassXC Team <team@keepassxc.org>
- *
- * This program is free software: you can redistribute it and/or modify
- * it under the terms of the GNU General Public License as published by
- * the Free Software Foundation, either version 2 or (at your option)
- * version 3 of the License.
- *
- * This program is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License
- * along with this program. If not, see <http://www.gnu.org/licenses/>.
- */
- #include "TestDatabase.h"
- #include <QRegularExpression>
- #include <QSignalSpy>
- #include <QTest>
- #include "config-keepassx-tests.h"
- #include "core/Group.h"
- #include "core/Metadata.h"
- #include "core/Tools.h"
- #include "crypto/Crypto.h"
- #include "format/KeePass2Writer.h"
- #include "util/TemporaryFile.h"
- #ifdef Q_OS_WIN
- #include <QFileInfo>
- #include <Windows.h>
- #endif
- QTEST_GUILESS_MAIN(TestDatabase)
- static QString dbFileName = QStringLiteral(KEEPASSX_TEST_DATA_DIR).append("/NewDatabase.kdbx");
- void TestDatabase::initTestCase()
- {
- QVERIFY(Crypto::init());
- }
- void TestDatabase::testOpen()
- {
- auto db = QSharedPointer<Database>::create();
- QVERIFY(!db->isInitialized());
- QVERIFY(!db->isModified());
- auto key = QSharedPointer<CompositeKey>::create();
- key->addKey(QSharedPointer<PasswordKey>::create("a"));
- bool ok = db->open(dbFileName, key);
- QVERIFY(ok);
- QVERIFY(db->isInitialized());
- QVERIFY(!db->isModified());
- db->metadata()->setName("test");
- QVERIFY(db->isModified());
- }
- void TestDatabase::testSave()
- {
- TemporaryFile tempFile;
- QVERIFY(tempFile.copyFromFile(dbFileName));
- auto db = QSharedPointer<Database>::create();
- auto key = QSharedPointer<CompositeKey>::create();
- key->addKey(QSharedPointer<PasswordKey>::create("a"));
- QString error;
- bool ok = db->open(tempFile.fileName(), key, &error);
- QVERIFY(ok);
- // Test safe saves
- db->metadata()->setName("test");
- QVERIFY(db->isModified());
- QVERIFY2(db->save(Database::Atomic, {}, &error), error.toLatin1());
- QVERIFY(!db->isModified());
- // Test temp-file saves
- db->metadata()->setName("test2");
- QVERIFY2(db->save(Database::TempFile, QString(), &error), error.toLatin1());
- QVERIFY(!db->isModified());
- // Test direct-write saves
- db->metadata()->setName("test3");
- QVERIFY2(db->save(Database::DirectWrite, QString(), &error), error.toLatin1());
- QVERIFY(!db->isModified());
- // Test save backups
- TemporaryFile backupFile;
- auto backupFilePath = backupFile.fileName();
- db->metadata()->setName("test4");
- QVERIFY2(db->save(Database::Atomic, backupFilePath, &error), error.toLatin1());
- QVERIFY(!db->isModified());
- QVERIFY(QFile::exists(backupFilePath));
- QFile::remove(backupFilePath);
- QVERIFY(!QFile::exists(backupFilePath));
- }
- void TestDatabase::testSaveAs()
- {
- TemporaryFile tempFile;
- QVERIFY(tempFile.copyFromFile(dbFileName));
- auto db = QSharedPointer<Database>::create();
- auto key = QSharedPointer<CompositeKey>::create();
- key->addKey(QSharedPointer<PasswordKey>::create("a"));
- QString error;
- QVERIFY(db->open(tempFile.fileName(), key, &error));
- // Happy path case when try to save as new DB.
- QSignalSpy spyFilePathChanged(db.data(), SIGNAL(filePathChanged(const QString&, const QString&)));
- QString newDbFileName = QStringLiteral(KEEPASSX_TEST_DATA_DIR).append("/SaveAsNewDatabase.kdbx");
- QVERIFY2(db->saveAs(newDbFileName, Database::Atomic, QString(), &error), error.toLatin1());
- QVERIFY(!db->isModified());
- QCOMPARE(spyFilePathChanged.count(), 1);
- QVERIFY(QFile::exists(newDbFileName));
- #ifdef Q_OS_WIN
- QVERIFY(!QFileInfo(newDbFileName).isHidden());
- SetFileAttributes(newDbFileName.toStdString().c_str(), FILE_ATTRIBUTE_HIDDEN);
- QVERIFY2(db->saveAs(newDbFileName, Database::Atomic, QString(), &error), error.toLatin1());
- QVERIFY(QFileInfo(newDbFileName).isHidden());
- #endif
- QFile::remove(newDbFileName);
- QVERIFY(!QFile::exists(newDbFileName));
- // Negative case when try to save not initialized DB.
- db->releaseData();
- QVERIFY2(!db->saveAs(newDbFileName, Database::Atomic, QString(), &error), error.toLatin1());
- QCOMPARE(error, QString("Could not save, database has not been initialized!"));
- }
- void TestDatabase::testSignals()
- {
- TemporaryFile tempFile;
- QVERIFY(tempFile.copyFromFile(dbFileName));
- auto db = QSharedPointer<Database>::create();
- auto key = QSharedPointer<CompositeKey>::create();
- key->addKey(QSharedPointer<PasswordKey>::create("a"));
- QSignalSpy spyFilePathChanged(db.data(), SIGNAL(filePathChanged(const QString&, const QString&)));
- QString error;
- bool ok = db->open(tempFile.fileName(), key, &error);
- QVERIFY(ok);
- QCOMPARE(spyFilePathChanged.count(), 1);
- QSignalSpy spyModified(db.data(), SIGNAL(modified()));
- db->metadata()->setName("test1");
- QTRY_COMPARE(spyModified.count(), 1);
- QSignalSpy spySaved(db.data(), SIGNAL(databaseSaved()));
- QVERIFY(db->save(Database::Atomic, {}, &error));
- QCOMPARE(spySaved.count(), 1);
- // Short delay to allow file system settling to reduce test failures
- Tools::wait(100);
- QSignalSpy spyFileChanged(db.data(), SIGNAL(databaseFileChanged()));
- QVERIFY(tempFile.copyFromFile(dbFileName));
- QTRY_COMPARE(spyFileChanged.count(), 1);
- QTRY_VERIFY(!db->isModified());
- db->metadata()->setName("test2");
- QTRY_VERIFY(db->isModified());
- QSignalSpy spyDiscarded(db.data(), SIGNAL(databaseDiscarded()));
- QVERIFY(db->open(tempFile.fileName(), key, &error));
- QCOMPARE(spyDiscarded.count(), 1);
- }
- void TestDatabase::testEmptyRecycleBinOnDisabled()
- {
- QString filename = QString(KEEPASSX_TEST_DATA_DIR).append("/RecycleBinDisabled.kdbx");
- auto key = QSharedPointer<CompositeKey>::create();
- key->addKey(QSharedPointer<PasswordKey>::create("123"));
- auto db = QSharedPointer<Database>::create();
- QVERIFY(db->open(filename, key, nullptr));
- QSignalSpy spyModified(db.data(), SIGNAL(modified()));
- db->emptyRecycleBin();
- // The database must be unmodified in this test after emptying the recycle bin.
- QTRY_COMPARE(spyModified.count(), 0);
- }
- void TestDatabase::testEmptyRecycleBinOnNotCreated()
- {
- QString filename = QString(KEEPASSX_TEST_DATA_DIR).append("/RecycleBinNotYetCreated.kdbx");
- auto key = QSharedPointer<CompositeKey>::create();
- key->addKey(QSharedPointer<PasswordKey>::create("123"));
- auto db = QSharedPointer<Database>::create();
- QVERIFY(db->open(filename, key, nullptr));
- QSignalSpy spyModified(db.data(), SIGNAL(modified()));
- db->emptyRecycleBin();
- // The database must be unmodified in this test after emptying the recycle bin.
- QTRY_COMPARE(spyModified.count(), 0);
- }
- void TestDatabase::testEmptyRecycleBinOnEmpty()
- {
- QString filename = QString(KEEPASSX_TEST_DATA_DIR).append("/RecycleBinEmpty.kdbx");
- auto key = QSharedPointer<CompositeKey>::create();
- key->addKey(QSharedPointer<PasswordKey>::create("123"));
- auto db = QSharedPointer<Database>::create();
- QVERIFY(db->open(filename, key, nullptr));
- QSignalSpy spyModified(db.data(), SIGNAL(modified()));
- db->emptyRecycleBin();
- // The database must be unmodified in this test after emptying the recycle bin.
- QTRY_COMPARE(spyModified.count(), 0);
- }
- void TestDatabase::testEmptyRecycleBinWithHierarchicalData()
- {
- QString filename = QString(KEEPASSX_TEST_DATA_DIR).append("/RecycleBinWithData.kdbx");
- auto key = QSharedPointer<CompositeKey>::create();
- key->addKey(QSharedPointer<PasswordKey>::create("123"));
- auto db = QSharedPointer<Database>::create();
- QVERIFY(db->open(filename, key, nullptr));
- QFile originalFile(filename);
- qint64 initialSize = originalFile.size();
- db->emptyRecycleBin();
- QVERIFY(db->metadata()->recycleBin());
- QVERIFY(db->metadata()->recycleBin()->entries().empty());
- QVERIFY(db->metadata()->recycleBin()->children().empty());
- QTemporaryFile afterCleanup;
- afterCleanup.open();
- KeePass2Writer writer;
- writer.writeDatabase(&afterCleanup, db.data());
- QVERIFY(afterCleanup.size() < initialSize);
- }
- void TestDatabase::testCustomIcons()
- {
- Database db;
- QUuid uuid1 = QUuid::createUuid();
- QByteArray icon1("icon 1");
- Q_ASSERT(!icon1.isNull());
- db.metadata()->addCustomIcon(uuid1, icon1);
- Metadata::CustomIconData iconData = db.metadata()->customIcon(uuid1);
- QCOMPARE(iconData.data, icon1);
- QVERIFY(iconData.name.isNull());
- QVERIFY(iconData.lastModified.isNull());
- QUuid uuid2 = QUuid::createUuid();
- QByteArray icon2("icon 2");
- QDateTime date = QDateTime::currentDateTimeUtc();
- db.metadata()->addCustomIcon(uuid2, icon2, "Test", date);
- iconData = db.metadata()->customIcon(uuid2);
- QCOMPARE(iconData.data, icon2);
- QCOMPARE(iconData.name, QString("Test"));
- QCOMPARE(iconData.lastModified, date);
- }
|