mirror of
https://github.com/keepassxreboot/keepassxc.git
synced 2025-07-25 15:55:38 -04:00
Add a new database settings wizard
This patch implements a new database wizard to guide users through the process of setting up a new database and choosing sane encryption settings. It also reimplements the master key settings to be more user-friendly. Users can now add, change, or remove individual composite key components instead of having to set all components at once. This avoids confusion about a password being reset if the user only wants to add a key file. With these changes comes a major refactor of how database composite keys and key components are handled. Copying of keys is prohibited and each key exists only once in memory and is referenced via shared pointers. GUI components for changing individual keys are encapsulated into separate classes to be more reusable. The password edit and generator widgets have also been refactored to be more reusable.
This commit is contained in:
parent
e4ded388b4
commit
e443cde452
116 changed files with 5054 additions and 1692 deletions
|
@ -31,7 +31,7 @@
|
|||
|
||||
Database* Kdbx3Reader::readDatabaseImpl(QIODevice* device,
|
||||
const QByteArray& headerData,
|
||||
const CompositeKey& key,
|
||||
QSharedPointer<const CompositeKey> key,
|
||||
bool keepDatabase)
|
||||
{
|
||||
Q_ASSERT(m_kdbxVersion <= KeePass2::FILE_VERSION_3_1);
|
||||
|
|
|
@ -31,7 +31,7 @@ class Kdbx3Reader : public KdbxReader
|
|||
public:
|
||||
Database* readDatabaseImpl(QIODevice* device,
|
||||
const QByteArray& headerData,
|
||||
const CompositeKey& key,
|
||||
QSharedPointer<const CompositeKey> key,
|
||||
bool keepDatabase) override;
|
||||
|
||||
protected:
|
||||
|
|
|
@ -30,7 +30,7 @@
|
|||
|
||||
Database* Kdbx4Reader::readDatabaseImpl(QIODevice* device,
|
||||
const QByteArray& headerData,
|
||||
const CompositeKey& key,
|
||||
QSharedPointer<const CompositeKey> key,
|
||||
bool keepDatabase)
|
||||
{
|
||||
Q_ASSERT(m_kdbxVersion == KeePass2::FILE_VERSION_4);
|
||||
|
|
|
@ -32,7 +32,7 @@ class Kdbx4Reader : public KdbxReader
|
|||
public:
|
||||
Database* readDatabaseImpl(QIODevice* device,
|
||||
const QByteArray& headerData,
|
||||
const CompositeKey& key,
|
||||
QSharedPointer<const CompositeKey> key,
|
||||
bool keepDatabase) override;
|
||||
QHash<QByteArray, QString> binaryPoolInverse() const;
|
||||
QHash<QString, QByteArray> binaryPool() const;
|
||||
|
|
|
@ -57,7 +57,7 @@ bool KdbxReader::readMagicNumbers(QIODevice* device, quint32& sig1, quint32& sig
|
|||
* @param keepDatabase keep database in case of read failure
|
||||
* @return pointer to the read database, nullptr on failure
|
||||
*/
|
||||
Database* KdbxReader::readDatabase(QIODevice* device, const CompositeKey& key, bool keepDatabase)
|
||||
Database* KdbxReader::readDatabase(QIODevice* device, QSharedPointer<const CompositeKey> key, bool keepDatabase)
|
||||
{
|
||||
device->seek(0);
|
||||
|
||||
|
|
|
@ -40,7 +40,7 @@ public:
|
|||
virtual ~KdbxReader() = default;
|
||||
|
||||
static bool readMagicNumbers(QIODevice* device, quint32& sig1, quint32& sig2, quint32& version);
|
||||
Database* readDatabase(QIODevice* device, const CompositeKey& key, bool keepDatabase = false);
|
||||
Database* readDatabase(QIODevice* device, QSharedPointer<const CompositeKey> key, bool keepDatabase = false);
|
||||
|
||||
bool hasError() const;
|
||||
QString errorString() const;
|
||||
|
@ -62,7 +62,10 @@ protected:
|
|||
* @return pointer to the read database, nullptr on failure
|
||||
*/
|
||||
virtual Database*
|
||||
readDatabaseImpl(QIODevice* device, const QByteArray& headerData, const CompositeKey& key, bool keepDatabase) = 0;
|
||||
readDatabaseImpl(QIODevice* device,
|
||||
const QByteArray& headerData,
|
||||
QSharedPointer<const CompositeKey> key,
|
||||
bool keepDatabase) = 0;
|
||||
|
||||
/**
|
||||
* Read next header field from stream.
|
||||
|
|
|
@ -63,7 +63,7 @@ Database* KeePass1Reader::readDatabase(QIODevice* device, const QString& passwor
|
|||
m_errorStr.clear();
|
||||
|
||||
QByteArray keyfileData;
|
||||
FileKey newFileKey;
|
||||
auto newFileKey = QSharedPointer<FileKey>::create();
|
||||
|
||||
if (keyfileDevice) {
|
||||
keyfileData = readKeyfile(keyfileDevice);
|
||||
|
@ -77,7 +77,7 @@ Database* KeePass1Reader::readDatabase(QIODevice* device, const QString& passwor
|
|||
return nullptr;
|
||||
}
|
||||
|
||||
if (!newFileKey.load(keyfileDevice)) {
|
||||
if (!newFileKey->load(keyfileDevice)) {
|
||||
raiseError(tr("Unable to read keyfile.").append("\n").append(keyfileDevice->errorString()));
|
||||
return nullptr;
|
||||
}
|
||||
|
@ -233,12 +233,12 @@ Database* KeePass1Reader::readDatabase(QIODevice* device, const QString& passwor
|
|||
entry->setUpdateTimeinfo(true);
|
||||
}
|
||||
|
||||
CompositeKey key;
|
||||
auto key = QSharedPointer<CompositeKey>::create();
|
||||
if (!password.isEmpty()) {
|
||||
key.addKey(PasswordKey(password));
|
||||
key->addKey(QSharedPointer<PasswordKey>::create(password));
|
||||
}
|
||||
if (keyfileDevice) {
|
||||
key.addKey(newFileKey);
|
||||
key->addKey(newFileKey);
|
||||
}
|
||||
|
||||
if (!db->setKey(key)) {
|
||||
|
|
|
@ -23,15 +23,15 @@
|
|||
|
||||
#define UUID_LENGTH 16
|
||||
|
||||
const QUuid KeePass2::CIPHER_AES = QUuid::fromRfc4122(QByteArray::fromHex("31c1f2e6bf714350be5805216afc5aff"));
|
||||
const QUuid KeePass2::CIPHER_TWOFISH = QUuid::fromRfc4122(QByteArray::fromHex("ad68f29f576f4bb9a36ad47af965346c"));
|
||||
const QUuid KeePass2::CIPHER_CHACHA20 = QUuid::fromRfc4122(QByteArray::fromHex("D6038A2B8B6F4CB5A524339A31DBB59A"));
|
||||
const QUuid KeePass2::CIPHER_AES = QUuid("31c1f2e6-bf71-4350-be58-05216afc5aff");
|
||||
const QUuid KeePass2::CIPHER_TWOFISH = QUuid("ad68f29f-576f-4bb9-a36a-d47af965346c");
|
||||
const QUuid KeePass2::CIPHER_CHACHA20 = QUuid("d6038a2b-8b6f-4cb5-a524-339a31dbb59a");
|
||||
|
||||
const QUuid KeePass2::KDF_AES_KDBX3 = QUuid::fromRfc4122(QByteArray::fromHex("C9D9F39A628A4460BF740D08C18A4FEA"));
|
||||
const QUuid KeePass2::KDF_AES_KDBX4 = QUuid::fromRfc4122(QByteArray::fromHex("7C02BB8279A74AC0927D114A00648238"));
|
||||
const QUuid KeePass2::KDF_ARGON2 = QUuid::fromRfc4122(QByteArray::fromHex("EF636DDF8C29444B91F7A9A403E30A0C"));
|
||||
const QUuid KeePass2::KDF_AES_KDBX3 = QUuid("c9d9f39a-628a-4460-bf74-0d08c18a4fea");
|
||||
const QUuid KeePass2::KDF_AES_KDBX4 = QUuid("7c02bb82-79a7-4ac0-927d-114a00648238");
|
||||
const QUuid KeePass2::KDF_ARGON2 = QUuid("ef636ddf-8c29-444b-91f7-a9a403e30a0c");
|
||||
|
||||
const QByteArray KeePass2::INNER_STREAM_SALSA20_IV("\xE8\x30\x09\x4B\x97\x20\x5D\x2A");
|
||||
const QByteArray KeePass2::INNER_STREAM_SALSA20_IV("\xe8\x30\x09\x4b\x97\x20\x5d\x2a");
|
||||
|
||||
const QString KeePass2::KDFPARAM_UUID("$UUID");
|
||||
// AES parameters
|
||||
|
|
|
@ -29,7 +29,7 @@
|
|||
* @param key database encryption composite key
|
||||
* @return pointer to the read database, nullptr on failure
|
||||
*/
|
||||
Database* KeePass2Reader::readDatabase(const QString& filename, const CompositeKey& key)
|
||||
Database* KeePass2Reader::readDatabase(const QString& filename, QSharedPointer<const CompositeKey> key)
|
||||
{
|
||||
QFile file(filename);
|
||||
if (!file.open(QFile::ReadOnly)) {
|
||||
|
@ -55,7 +55,7 @@ Database* KeePass2Reader::readDatabase(const QString& filename, const CompositeK
|
|||
* @param keepDatabase keep database in case of read failure
|
||||
* @return pointer to the read database, nullptr on failure
|
||||
*/
|
||||
Database* KeePass2Reader::readDatabase(QIODevice* device, const CompositeKey& key, bool keepDatabase)
|
||||
Database* KeePass2Reader::readDatabase(QIODevice* device, QSharedPointer<const CompositeKey> key, bool keepDatabase)
|
||||
{
|
||||
m_error = false;
|
||||
m_errorStr.clear();
|
||||
|
|
|
@ -35,8 +35,8 @@ class KeePass2Reader
|
|||
Q_DECLARE_TR_FUNCTIONS(KdbxReader)
|
||||
|
||||
public:
|
||||
Database* readDatabase(const QString& filename, const CompositeKey& key);
|
||||
Database* readDatabase(QIODevice* device, const CompositeKey& key, bool keepDatabase = false);
|
||||
Database* readDatabase(const QString& filename, QSharedPointer<const CompositeKey> key);
|
||||
Database* readDatabase(QIODevice* device, QSharedPointer<const CompositeKey> key, bool keepDatabase = false);
|
||||
|
||||
bool hasError() const;
|
||||
QString errorString() const;
|
||||
|
|
|
@ -27,7 +27,7 @@
|
|||
#include "format/KeePass2RandomStream.h"
|
||||
#include "format/KeePass2Reader.h"
|
||||
|
||||
KeePass2Repair::RepairOutcome KeePass2Repair::repairDatabase(QIODevice* device, const CompositeKey& key)
|
||||
KeePass2Repair::RepairOutcome KeePass2Repair::repairDatabase(QIODevice* device, QSharedPointer<const CompositeKey> key)
|
||||
{
|
||||
m_errorStr.clear();
|
||||
|
||||
|
|
|
@ -40,7 +40,7 @@ public:
|
|||
};
|
||||
using RepairOutcome = QPair<RepairResult, Database*>;
|
||||
|
||||
RepairOutcome repairDatabase(QIODevice* device, const CompositeKey& key);
|
||||
RepairOutcome repairDatabase(QIODevice* device, QSharedPointer<const CompositeKey> key);
|
||||
QString errorString() const;
|
||||
|
||||
private:
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue