TestKdbx4.cpp 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452
  1. /*
  2. * Copyright (C) 2018 KeePassXC Team <team@keepassxc.org>
  3. *
  4. * This program is free software: you can redistribute it and/or modify
  5. * it under the terms of the GNU General Public License as published by
  6. * the Free Software Foundation, either version 2 or (at your option)
  7. * version 3 of the License.
  8. *
  9. * This program is distributed in the hope that it will be useful,
  10. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. * GNU General Public License for more details.
  13. *
  14. * You should have received a copy of the GNU General Public License
  15. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  16. */
  17. #include "TestKdbx4.h"
  18. #include "TestGlobal.h"
  19. #include "config-keepassx-tests.h"
  20. #include "core/Metadata.h"
  21. #include "format/KdbxXmlReader.h"
  22. #include "format/KdbxXmlWriter.h"
  23. #include "format/KeePass2.h"
  24. #include "format/KeePass2Reader.h"
  25. #include "format/KeePass2Writer.h"
  26. #include "keys/FileKey.h"
  27. #include "keys/PasswordKey.h"
  28. #include "mock/MockChallengeResponseKey.h"
  29. int main(int argc, char* argv[])
  30. {
  31. QCoreApplication app(argc, argv);
  32. QCoreApplication::setAttribute(Qt::AA_Use96Dpi, true);
  33. QTEST_SET_MAIN_SOURCE_PATH
  34. TestKdbx4Argon2 argon2Test;
  35. TestKdbx4AesKdf aesKdfTest;
  36. return QTest::qExec(&argon2Test, argc, argv) | QTest::qExec(&aesKdfTest, argc, argv);
  37. }
  38. void TestKdbx4Argon2::initTestCaseImpl()
  39. {
  40. m_xmlDb->changeKdf(fastKdf(KeePass2::uuidToKdf(KeePass2::KDF_ARGON2D)));
  41. m_kdbxSourceDb->changeKdf(fastKdf(KeePass2::uuidToKdf(KeePass2::KDF_ARGON2D)));
  42. }
  43. QSharedPointer<Database>
  44. TestKdbx4Argon2::readXml(const QString& path, bool strictMode, bool& hasError, QString& errorString)
  45. {
  46. KdbxXmlReader reader(KeePass2::FILE_VERSION_4);
  47. reader.setStrictMode(strictMode);
  48. auto db = reader.readDatabase(path);
  49. hasError = reader.hasError();
  50. errorString = reader.errorString();
  51. return db;
  52. }
  53. QSharedPointer<Database> TestKdbx4Argon2::readXml(QBuffer* buf, bool strictMode, bool& hasError, QString& errorString)
  54. {
  55. KdbxXmlReader reader(KeePass2::FILE_VERSION_4);
  56. reader.setStrictMode(strictMode);
  57. auto db = reader.readDatabase(buf);
  58. hasError = reader.hasError();
  59. errorString = reader.errorString();
  60. return db;
  61. }
  62. void TestKdbx4Argon2::writeXml(QBuffer* buf, Database* db, bool& hasError, QString& errorString)
  63. {
  64. KdbxXmlWriter writer(KeePass2::FILE_VERSION_4);
  65. writer.writeDatabase(buf, db);
  66. hasError = writer.hasError();
  67. errorString = writer.errorString();
  68. }
  69. void TestKdbx4Argon2::readKdbx(QIODevice* device,
  70. QSharedPointer<const CompositeKey> key,
  71. QSharedPointer<Database> db,
  72. bool& hasError,
  73. QString& errorString)
  74. {
  75. KeePass2Reader reader;
  76. reader.readDatabase(device, key, db.data());
  77. hasError = reader.hasError();
  78. if (hasError) {
  79. errorString = reader.errorString();
  80. }
  81. QCOMPARE(reader.version(), KeePass2::FILE_VERSION_4);
  82. }
  83. void TestKdbx4Argon2::readKdbx(const QString& path,
  84. QSharedPointer<const CompositeKey> key,
  85. QSharedPointer<Database> db,
  86. bool& hasError,
  87. QString& errorString)
  88. {
  89. KeePass2Reader reader;
  90. reader.readDatabase(path, key, db.data());
  91. hasError = reader.hasError();
  92. if (hasError) {
  93. errorString = reader.errorString();
  94. }
  95. QCOMPARE(reader.version(), KeePass2::FILE_VERSION_4);
  96. }
  97. void TestKdbx4Argon2::writeKdbx(QIODevice* device, Database* db, bool& hasError, QString& errorString)
  98. {
  99. if (db->kdf()->uuid() == KeePass2::KDF_AES_KDBX3) {
  100. db->changeKdf(fastKdf(KeePass2::uuidToKdf(KeePass2::KDF_ARGON2D)));
  101. }
  102. KeePass2Writer writer;
  103. hasError = writer.writeDatabase(device, db);
  104. hasError = writer.hasError();
  105. if (hasError) {
  106. errorString = writer.errorString();
  107. }
  108. QCOMPARE(writer.version(), KeePass2::FILE_VERSION_4);
  109. }
  110. Q_DECLARE_METATYPE(QUuid)
  111. void TestKdbx4Argon2::testFormat400()
  112. {
  113. QString filename = QString(KEEPASSX_TEST_DATA_DIR).append("/Format400.kdbx");
  114. auto key = QSharedPointer<CompositeKey>::create();
  115. key->addKey(QSharedPointer<PasswordKey>::create("t"));
  116. KeePass2Reader reader;
  117. auto db = QSharedPointer<Database>::create();
  118. reader.readDatabase(filename, key, db.data());
  119. QCOMPARE(reader.version(), KeePass2::FILE_VERSION_4);
  120. QVERIFY(db.data());
  121. QVERIFY(!reader.hasError());
  122. QCOMPARE(db->rootGroup()->name(), QString("Format400"));
  123. QCOMPARE(db->metadata()->name(), QString("Format400"));
  124. QCOMPARE(db->rootGroup()->entries().size(), 1);
  125. auto entry = db->rootGroup()->entries().at(0);
  126. QCOMPARE(entry->title(), QString("Format400"));
  127. QCOMPARE(entry->username(), QString("Format400"));
  128. QCOMPARE(entry->attributes()->keys().size(), 6);
  129. QCOMPARE(entry->attributes()->value("Format400"), QString("Format400"));
  130. QCOMPARE(entry->attachments()->keys().size(), 1);
  131. QCOMPARE(entry->attachments()->value("Format400"), QByteArray("Format400\n"));
  132. }
  133. void TestKdbx4Argon2::testFormat400Upgrade()
  134. {
  135. QFETCH(QUuid, kdfUuid);
  136. QFETCH(QUuid, cipherUuid);
  137. QFETCH(bool, addCustomData);
  138. QFETCH(quint32, expectedVersion);
  139. QScopedPointer<Database> sourceDb(new Database());
  140. sourceDb->changeKdf(fastKdf(sourceDb->kdf()));
  141. sourceDb->metadata()->setName("Wubba lubba dub dub");
  142. QCOMPARE(sourceDb->kdf()->uuid(), KeePass2::KDF_AES_KDBX3); // default is legacy AES-KDF
  143. auto key = QSharedPointer<CompositeKey>::create();
  144. key->addKey(QSharedPointer<PasswordKey>::create("I am in great pain, please help me!"));
  145. sourceDb->setKey(key, true, true);
  146. QBuffer buffer;
  147. buffer.open(QBuffer::ReadWrite);
  148. // upgrade to KDBX 4 by changing KDF and Cipher
  149. sourceDb->changeKdf(fastKdf(KeePass2::uuidToKdf(kdfUuid)));
  150. sourceDb->setCipher(cipherUuid);
  151. // CustomData in meta should not cause any version change
  152. sourceDb->metadata()->customData()->set("CustomPublicData", "Hey look, I turned myself into a pickle!");
  153. if (addCustomData) {
  154. // this, however, should
  155. sourceDb->rootGroup()->customData()->set("CustomGroupData",
  156. "I just killed my family! I don't care who they were!");
  157. }
  158. KeePass2Writer writer;
  159. writer.writeDatabase(&buffer, sourceDb.data());
  160. if (writer.hasError()) {
  161. QFAIL(qPrintable(QString("Error while writing database: %1").arg(writer.errorString())));
  162. }
  163. // read buffer back
  164. buffer.seek(0);
  165. KeePass2Reader reader;
  166. auto targetDb = QSharedPointer<Database>::create();
  167. reader.readDatabase(&buffer, key, targetDb.data());
  168. if (reader.hasError()) {
  169. QFAIL(qPrintable(QString("Error while reading database: %1").arg(reader.errorString())));
  170. }
  171. QVERIFY(targetDb->rootGroup());
  172. QCOMPARE(targetDb->metadata()->name(), sourceDb->metadata()->name());
  173. QCOMPARE(reader.version(), expectedVersion);
  174. QCOMPARE(targetDb->cipher(), cipherUuid);
  175. QCOMPARE(targetDb->metadata()->customData()->value("CustomPublicData"),
  176. sourceDb->metadata()->customData()->value("CustomPublicData"));
  177. QCOMPARE(targetDb->rootGroup()->customData()->value("CustomGroupData"),
  178. sourceDb->rootGroup()->customData()->value("CustomGroupData"));
  179. }
  180. // clang-format off
  181. void TestKdbx4Argon2::testFormat400Upgrade_data()
  182. {
  183. QTest::addColumn<QUuid>("kdfUuid");
  184. QTest::addColumn<QUuid>("cipherUuid");
  185. QTest::addColumn<bool>("addCustomData");
  186. QTest::addColumn<quint32>("expectedVersion");
  187. auto constexpr kdbx3 = KeePass2::FILE_VERSION_3_1 & KeePass2::FILE_VERSION_CRITICAL_MASK;
  188. auto constexpr kdbx4 = KeePass2::FILE_VERSION_4 & KeePass2::FILE_VERSION_CRITICAL_MASK;
  189. QTest::newRow("Argon2d + AES") << KeePass2::KDF_ARGON2D << KeePass2::CIPHER_AES256 << false << kdbx4;
  190. QTest::newRow("Argon2id + AES") << KeePass2::KDF_ARGON2ID << KeePass2::CIPHER_AES256 << false << kdbx4;
  191. QTest::newRow("AES-KDF + AES") << KeePass2::KDF_AES_KDBX4 << KeePass2::CIPHER_AES256 << false << kdbx4;
  192. QTest::newRow("AES-KDF (legacy) + AES") << KeePass2::KDF_AES_KDBX3 << KeePass2::CIPHER_AES256 << false << kdbx3;
  193. QTest::newRow("Argon2d + AES + CustomData") << KeePass2::KDF_ARGON2D << KeePass2::CIPHER_AES256 << true << kdbx4;
  194. QTest::newRow("Argon2id + AES + CustomData") << KeePass2::KDF_ARGON2ID << KeePass2::CIPHER_AES256 << true << kdbx4;
  195. QTest::newRow("AES-KDF + AES + CustomData") << KeePass2::KDF_AES_KDBX4 << KeePass2::CIPHER_AES256 << true << kdbx4;
  196. QTest::newRow("AES-KDF (legacy) + AES + CustomData") << KeePass2::KDF_AES_KDBX3 << KeePass2::CIPHER_AES256 << true << kdbx4;
  197. QTest::newRow("Argon2d + ChaCha20") << KeePass2::KDF_ARGON2D << KeePass2::CIPHER_CHACHA20 << false << kdbx4;
  198. QTest::newRow("Argon2id + ChaCha20") << KeePass2::KDF_ARGON2ID << KeePass2::CIPHER_CHACHA20 << false << kdbx4;
  199. QTest::newRow("AES-KDF + ChaCha20") << KeePass2::KDF_AES_KDBX4 << KeePass2::CIPHER_CHACHA20 << false << kdbx4;
  200. QTest::newRow("AES-KDF (legacy) + ChaCha20") << KeePass2::KDF_AES_KDBX3 << KeePass2::CIPHER_CHACHA20 << false << kdbx3;
  201. QTest::newRow("Argon2d + ChaCha20 + CustomData") << KeePass2::KDF_ARGON2D << KeePass2::CIPHER_CHACHA20 << true << kdbx4;
  202. QTest::newRow("Argon2id + ChaCha20 + CustomData") << KeePass2::KDF_ARGON2ID << KeePass2::CIPHER_CHACHA20 << true << kdbx4;
  203. QTest::newRow("AES-KDF + ChaCha20 + CustomData") << KeePass2::KDF_AES_KDBX4 << KeePass2::CIPHER_CHACHA20 << true << kdbx4;
  204. QTest::newRow("AES-KDF (legacy) + ChaCha20 + CustomData") << KeePass2::KDF_AES_KDBX3 << KeePass2::CIPHER_CHACHA20 << true << kdbx4;
  205. QTest::newRow("Argon2d + Twofish") << KeePass2::KDF_ARGON2D << KeePass2::CIPHER_TWOFISH << false << kdbx4;
  206. QTest::newRow("Argon2id + Twofish") << KeePass2::KDF_ARGON2ID << KeePass2::CIPHER_TWOFISH << false << kdbx4;
  207. QTest::newRow("AES-KDF + Twofish") << KeePass2::KDF_AES_KDBX4 << KeePass2::CIPHER_TWOFISH << false << kdbx4;
  208. QTest::newRow("AES-KDF (legacy) + Twofish") << KeePass2::KDF_AES_KDBX3 << KeePass2::CIPHER_TWOFISH << false << kdbx3;
  209. QTest::newRow("Argon2d + Twofish + CustomData") << KeePass2::KDF_ARGON2D << KeePass2::CIPHER_TWOFISH << true << kdbx4;
  210. QTest::newRow("Argon2id + Twofish + CustomData") << KeePass2::KDF_ARGON2ID << KeePass2::CIPHER_TWOFISH << true << kdbx4;
  211. QTest::newRow("AES-KDF + Twofish + CustomData") << KeePass2::KDF_AES_KDBX4 << KeePass2::CIPHER_TWOFISH << true << kdbx4;
  212. QTest::newRow("AES-KDF (legacy) + Twofish + CustomData") << KeePass2::KDF_AES_KDBX3 << KeePass2::CIPHER_TWOFISH << true << kdbx4;
  213. }
  214. // clang-format on
  215. void TestKdbx4Argon2::testUpgradeMasterKeyIntegrity()
  216. {
  217. QFETCH(QString, upgradeAction);
  218. QFETCH(quint32, expectedVersion);
  219. // prepare composite key
  220. auto passwordKey = QSharedPointer<PasswordKey>::create("turXpGMQiUE6CkPvWacydAKsnp4cxz");
  221. QByteArray fileKeyBytes("Ma6hHov98FbPeyAL22XhcgmpJk8xjQ");
  222. QBuffer fileKeyBuffer(&fileKeyBytes);
  223. fileKeyBuffer.open(QBuffer::ReadOnly);
  224. auto fileKey = QSharedPointer<FileKey>::create();
  225. fileKey->load(&fileKeyBuffer);
  226. auto crKey = QSharedPointer<MockChallengeResponseKey>::create(QByteArray("azdJnbVCFE76vV6t9RJ2DS6xvSS93k"));
  227. auto compositeKey = QSharedPointer<CompositeKey>::create();
  228. compositeKey->addKey(passwordKey);
  229. compositeKey->addKey(fileKey);
  230. compositeKey->addChallengeResponseKey(crKey);
  231. QScopedPointer<Database> db(new Database());
  232. db->changeKdf(fastKdf(db->kdf()));
  233. QCOMPARE(db->kdf()->uuid(), KeePass2::KDF_AES_KDBX3); // default is legacy AES-KDF
  234. db->setKey(compositeKey);
  235. // upgrade the database by a specific method
  236. if (upgradeAction == "none") {
  237. // do nothing
  238. } else if (upgradeAction == "meta-customdata") {
  239. db->metadata()->customData()->set("abc", "def");
  240. } else if (upgradeAction == "kdf-aes-kdbx3") {
  241. db->changeKdf(fastKdf(KeePass2::uuidToKdf(KeePass2::KDF_AES_KDBX3)));
  242. } else if (upgradeAction == "kdf-argon2") {
  243. db->changeKdf(fastKdf(KeePass2::uuidToKdf(KeePass2::KDF_ARGON2D)));
  244. } else if (upgradeAction == "kdf-aes-kdbx4") {
  245. db->changeKdf(fastKdf(KeePass2::uuidToKdf(KeePass2::KDF_AES_KDBX4)));
  246. } else if (upgradeAction == "public-customdata") {
  247. db->publicCustomData().insert("abc", "def");
  248. } else if (upgradeAction == "rootgroup-customdata") {
  249. db->rootGroup()->customData()->set("abc", "def");
  250. } else if (upgradeAction == "group-customdata") {
  251. auto group = new Group();
  252. group->setParent(db->rootGroup());
  253. group->setUuid(QUuid::createUuid());
  254. group->customData()->set("abc", "def");
  255. } else if (upgradeAction == "rootentry-customdata") {
  256. auto entry = new Entry();
  257. entry->setGroup(db->rootGroup());
  258. entry->setUuid(QUuid::createUuid());
  259. entry->customData()->set("abc", "def");
  260. } else if (upgradeAction == "entry-customdata") {
  261. auto group = new Group();
  262. group->setParent(db->rootGroup());
  263. group->setUuid(QUuid::createUuid());
  264. auto entry = new Entry();
  265. entry->setGroup(group);
  266. entry->setUuid(QUuid::createUuid());
  267. entry->customData()->set("abc", "def");
  268. } else {
  269. QFAIL(qPrintable(QString("Unknown action: %s").arg(upgradeAction)));
  270. }
  271. QBuffer buffer;
  272. buffer.open(QBuffer::ReadWrite);
  273. KeePass2Writer writer;
  274. QVERIFY(writer.writeDatabase(&buffer, db.data()));
  275. // paranoid check that we cannot decrypt the database without a key
  276. buffer.seek(0);
  277. KeePass2Reader reader;
  278. auto db2 = QSharedPointer<Database>::create();
  279. reader.readDatabase(&buffer, QSharedPointer<CompositeKey>::create(), db2.data());
  280. QVERIFY(reader.hasError());
  281. // check that we can read back the database with the original composite key,
  282. // i.e., no components have been lost on the way
  283. buffer.seek(0);
  284. db2 = QSharedPointer<Database>::create();
  285. reader.readDatabase(&buffer, compositeKey, db2.data());
  286. if (reader.hasError()) {
  287. QFAIL(qPrintable(reader.errorString()));
  288. }
  289. QCOMPARE(reader.version(), expectedVersion & KeePass2::FILE_VERSION_CRITICAL_MASK);
  290. if (expectedVersion != KeePass2::FILE_VERSION_3) {
  291. QVERIFY(db2->kdf()->uuid() != KeePass2::KDF_AES_KDBX3);
  292. }
  293. }
  294. void TestKdbx4Argon2::testUpgradeMasterKeyIntegrity_data()
  295. {
  296. QTest::addColumn<QString>("upgradeAction");
  297. QTest::addColumn<quint32>("expectedVersion");
  298. QTest::newRow("Upgrade: none") << QString("none") << KeePass2::FILE_VERSION_3;
  299. QTest::newRow("Upgrade: none (meta-customdata)") << QString("meta-customdata") << KeePass2::FILE_VERSION_3;
  300. QTest::newRow("Upgrade: none (explicit kdf-aes-kdbx3)") << QString("kdf-aes-kdbx3") << KeePass2::FILE_VERSION_3;
  301. QTest::newRow("Upgrade (explicit): kdf-argon2") << QString("kdf-argon2") << KeePass2::FILE_VERSION_4;
  302. QTest::newRow("Upgrade (explicit): kdf-aes-kdbx4") << QString("kdf-aes-kdbx4") << KeePass2::FILE_VERSION_4;
  303. QTest::newRow("Upgrade (implicit): public-customdata") << QString("public-customdata") << KeePass2::FILE_VERSION_4;
  304. QTest::newRow("Upgrade (implicit): rootgroup-customdata")
  305. << QString("rootgroup-customdata") << KeePass2::FILE_VERSION_4;
  306. QTest::newRow("Upgrade (implicit): group-customdata") << QString("group-customdata") << KeePass2::FILE_VERSION_4;
  307. QTest::newRow("Upgrade (implicit): rootentry-customdata")
  308. << QString("rootentry-customdata") << KeePass2::FILE_VERSION_4;
  309. QTest::newRow("Upgrade (implicit): entry-customdata") << QString("entry-customdata") << KeePass2::FILE_VERSION_4;
  310. }
  311. void TestKdbx4Argon2::testCustomData()
  312. {
  313. Database db;
  314. // test public custom data
  315. QVariantMap publicCustomData;
  316. publicCustomData.insert("CD1", 123);
  317. publicCustomData.insert("CD2", true);
  318. publicCustomData.insert("CD3", "abcäöü");
  319. db.setPublicCustomData(publicCustomData);
  320. publicCustomData.insert("CD4", QByteArray::fromHex("ababa123ff"));
  321. db.publicCustomData().insert("CD4", publicCustomData.value("CD4"));
  322. QCOMPARE(db.publicCustomData(), publicCustomData);
  323. const QString customDataKey1 = "CD1";
  324. const QString customDataKey2 = "CD2";
  325. const QString customData1 = "abcäöü";
  326. const QString customData2 = "Hello World";
  327. // test custom database data
  328. db.metadata()->customData()->set(customDataKey1, customData1);
  329. db.metadata()->customData()->set(customDataKey2, customData2);
  330. auto lastModified = db.metadata()->customData()->value(CustomData::LastModified);
  331. const int dataSize = customDataKey1.toUtf8().size() + customDataKey1.toUtf8().size() + customData1.toUtf8().size()
  332. + customData2.toUtf8().size() + lastModified.toUtf8().size()
  333. + CustomData::LastModified.toUtf8().size();
  334. QCOMPARE(db.metadata()->customData()->size(), 3);
  335. QCOMPARE(db.metadata()->customData()->dataSize(), dataSize);
  336. // test custom root group data
  337. Group* root = db.rootGroup();
  338. root->customData()->set(customDataKey1, customData1);
  339. root->customData()->set(customDataKey2, customData2);
  340. QCOMPARE(root->customData()->size(), 3);
  341. QCOMPARE(root->customData()->dataSize(), dataSize);
  342. // test copied custom group data
  343. auto* group = new Group();
  344. group->setParent(root);
  345. group->setUuid(QUuid::createUuid());
  346. group->customData()->copyDataFrom(root->customData());
  347. QCOMPARE(*group->customData(), *root->customData());
  348. // test copied custom entry data
  349. auto* entry = new Entry();
  350. entry->setGroup(group);
  351. entry->setUuid(QUuid::createUuid());
  352. entry->customData()->copyDataFrom(group->customData());
  353. QCOMPARE(*entry->customData(), *root->customData());
  354. // test custom data deletion
  355. entry->customData()->set("additional item", "foobar");
  356. QCOMPARE(entry->customData()->size(), 4);
  357. entry->customData()->remove("additional item");
  358. QCOMPARE(entry->customData()->size(), 3);
  359. QCOMPARE(entry->customData()->dataSize(), dataSize);
  360. // test custom data on cloned groups
  361. QScopedPointer<Group> clonedGroup(group->clone());
  362. QCOMPARE(*clonedGroup->customData(), *group->customData());
  363. // test custom data on cloned entries
  364. QScopedPointer<Entry> clonedEntry(entry->clone(Entry::CloneNoFlags));
  365. QCOMPARE(*clonedEntry->customData(), *entry->customData());
  366. QBuffer buffer;
  367. buffer.open(QBuffer::ReadWrite);
  368. KeePass2Writer writer;
  369. writer.writeDatabase(&buffer, &db);
  370. // read buffer back
  371. buffer.seek(0);
  372. KeePass2Reader reader;
  373. auto newDb = QSharedPointer<Database>::create();
  374. reader.readDatabase(&buffer, QSharedPointer<CompositeKey>::create(), newDb.data());
  375. // test all custom data are read back successfully from KDBX
  376. QCOMPARE(newDb->publicCustomData(), publicCustomData);
  377. QCOMPARE(newDb->metadata()->customData()->value(customDataKey1), customData1);
  378. QCOMPARE(newDb->metadata()->customData()->value(customDataKey2), customData2);
  379. QCOMPARE(newDb->rootGroup()->customData()->value(customDataKey1), customData1);
  380. QCOMPARE(newDb->rootGroup()->customData()->value(customDataKey2), customData2);
  381. auto* newGroup = newDb->rootGroup()->children()[0];
  382. QCOMPARE(newGroup->customData()->value(customDataKey1), customData1);
  383. QCOMPARE(newGroup->customData()->value(customDataKey2), customData2);
  384. auto* newEntry = newDb->rootGroup()->children()[0]->entries()[0];
  385. QCOMPARE(newEntry->customData()->value(customDataKey1), customData1);
  386. QCOMPARE(newEntry->customData()->value(customDataKey2), customData2);
  387. }
  388. void TestKdbx4AesKdf::initTestCaseImpl()
  389. {
  390. m_xmlDb->changeKdf(fastKdf(KeePass2::uuidToKdf(KeePass2::KDF_AES_KDBX4)));
  391. m_kdbxSourceDb->changeKdf(fastKdf(KeePass2::uuidToKdf(KeePass2::KDF_AES_KDBX4)));
  392. }