mirror of
https://github.com/keepassxreboot/keepassxc.git
synced 2025-01-21 12:11:06 -05:00
f9cb2bd5df
* Mark the database as clean after fully completing the file save operation INSTEAD of when merely writing the database to a file. * Stop the modified timer when marking the database as clean, this prevents latent erroneous modified signals from being emitted. * Do not restart the modified timer after a new change is detected while it is still running.
223 lines
6.8 KiB
C++
223 lines
6.8 KiB
C++
/*
|
|
* Copyright (C) 2018 KeePassXC Team <team@keepassxc.org>
|
|
* Copyright (C) 2010 Felix Geyer <debfx@fobos.de>
|
|
*
|
|
* 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/>.
|
|
*/
|
|
|
|
#ifndef KEEPASSX_DATABASE_H
|
|
#define KEEPASSX_DATABASE_H
|
|
|
|
#include <QDateTime>
|
|
#include <QHash>
|
|
#include <QPointer>
|
|
#include <QScopedPointer>
|
|
#include <QTimer>
|
|
|
|
#include "config-keepassx.h"
|
|
#include "crypto/kdf/AesKdf.h"
|
|
#include "crypto/kdf/Kdf.h"
|
|
#include "format/KeePass2.h"
|
|
#include "keys/CompositeKey.h"
|
|
#include "keys/PasswordKey.h"
|
|
|
|
class Entry;
|
|
enum class EntryReferenceType;
|
|
class FileWatcher;
|
|
class Group;
|
|
class Metadata;
|
|
class QIODevice;
|
|
|
|
struct DeletedObject
|
|
{
|
|
QUuid uuid;
|
|
QDateTime deletionTime;
|
|
bool operator==(const DeletedObject& other) const
|
|
{
|
|
return uuid == other.uuid && deletionTime == other.deletionTime;
|
|
}
|
|
};
|
|
|
|
Q_DECLARE_TYPEINFO(DeletedObject, Q_MOVABLE_TYPE);
|
|
|
|
class Database : public QObject
|
|
{
|
|
Q_OBJECT
|
|
|
|
public:
|
|
enum CompressionAlgorithm
|
|
{
|
|
CompressionNone = 0,
|
|
CompressionGZip = 1
|
|
};
|
|
static const quint32 CompressionAlgorithmMax = CompressionGZip;
|
|
|
|
Database();
|
|
explicit Database(const QString& filePath);
|
|
~Database() override;
|
|
|
|
bool open(QSharedPointer<const CompositeKey> key, QString* error = nullptr, bool readOnly = false);
|
|
bool open(const QString& filePath,
|
|
QSharedPointer<const CompositeKey> key,
|
|
QString* error = nullptr,
|
|
bool readOnly = false);
|
|
bool save(QString* error = nullptr, bool atomic = true, bool backup = false);
|
|
bool saveAs(const QString& filePath, QString* error = nullptr, bool atomic = true, bool backup = false);
|
|
bool extract(QByteArray&, QString* error = nullptr);
|
|
bool import(const QString& xmlExportPath, QString* error = nullptr);
|
|
|
|
void releaseData();
|
|
|
|
bool isInitialized() const;
|
|
void setInitialized(bool initialized);
|
|
bool isModified() const;
|
|
void setEmitModified(bool value);
|
|
bool isReadOnly() const;
|
|
void setReadOnly(bool readOnly);
|
|
|
|
QUuid uuid() const;
|
|
QString filePath() const;
|
|
QString canonicalFilePath() const;
|
|
void setFilePath(const QString& filePath);
|
|
|
|
Metadata* metadata();
|
|
const Metadata* metadata() const;
|
|
Group* rootGroup();
|
|
const Group* rootGroup() const;
|
|
void setRootGroup(Group* group);
|
|
QVariantMap& publicCustomData();
|
|
const QVariantMap& publicCustomData() const;
|
|
void setPublicCustomData(const QVariantMap& customData);
|
|
|
|
void recycleGroup(Group* group);
|
|
void recycleEntry(Entry* entry);
|
|
void emptyRecycleBin();
|
|
QList<DeletedObject> deletedObjects();
|
|
const QList<DeletedObject>& deletedObjects() const;
|
|
void addDeletedObject(const DeletedObject& delObj);
|
|
void addDeletedObject(const QUuid& uuid);
|
|
bool containsDeletedObject(const QUuid& uuid) const;
|
|
bool containsDeletedObject(const DeletedObject& uuid) const;
|
|
void setDeletedObjects(const QList<DeletedObject>& delObjs);
|
|
|
|
QList<QString> commonUsernames();
|
|
|
|
bool hasKey() const;
|
|
QSharedPointer<const CompositeKey> key() const;
|
|
bool setKey(const QSharedPointer<const CompositeKey>& key,
|
|
bool updateChangedTime = true,
|
|
bool updateTransformSalt = false,
|
|
bool transformKey = true);
|
|
QByteArray challengeResponseKey() const;
|
|
bool challengeMasterSeed(const QByteArray& masterSeed);
|
|
bool verifyKey(const QSharedPointer<CompositeKey>& key) const;
|
|
const QUuid& cipher() const;
|
|
void setCipher(const QUuid& cipher);
|
|
Database::CompressionAlgorithm compressionAlgorithm() const;
|
|
void setCompressionAlgorithm(Database::CompressionAlgorithm algo);
|
|
|
|
QSharedPointer<Kdf> kdf() const;
|
|
void setKdf(QSharedPointer<Kdf> kdf);
|
|
bool changeKdf(const QSharedPointer<Kdf>& kdf);
|
|
QByteArray transformedMasterKey() const;
|
|
|
|
static Database* databaseByUuid(const QUuid& uuid);
|
|
|
|
public slots:
|
|
void markAsModified();
|
|
void markAsClean();
|
|
void updateCommonUsernames(int topN = 10);
|
|
|
|
signals:
|
|
void filePathChanged(const QString& oldPath, const QString& newPath);
|
|
void groupDataChanged(Group* group);
|
|
void groupAboutToAdd(Group* group, int index);
|
|
void groupAdded();
|
|
void groupAboutToRemove(Group* group);
|
|
void groupRemoved();
|
|
void groupAboutToMove(Group* group, Group* toGroup, int index);
|
|
void groupMoved();
|
|
void databaseOpened();
|
|
void databaseModified();
|
|
void databaseSaved();
|
|
void databaseDiscarded();
|
|
void databaseFileChanged();
|
|
|
|
private:
|
|
struct DatabaseData
|
|
{
|
|
QString filePath;
|
|
bool isReadOnly = false;
|
|
QUuid cipher = KeePass2::CIPHER_AES256;
|
|
CompressionAlgorithm compressionAlgorithm = CompressionGZip;
|
|
|
|
QScopedPointer<PasswordKey> masterSeed;
|
|
QScopedPointer<PasswordKey> transformedMasterKey;
|
|
QScopedPointer<PasswordKey> challengeResponseKey;
|
|
|
|
bool hasKey = false;
|
|
QSharedPointer<const CompositeKey> key;
|
|
QSharedPointer<Kdf> kdf = QSharedPointer<AesKdf>::create(true);
|
|
|
|
QVariantMap publicCustomData;
|
|
|
|
DatabaseData()
|
|
: masterSeed(new PasswordKey())
|
|
, transformedMasterKey(new PasswordKey())
|
|
, challengeResponseKey(new PasswordKey())
|
|
{
|
|
kdf->randomizeSeed();
|
|
}
|
|
|
|
void clear()
|
|
{
|
|
filePath.clear();
|
|
|
|
masterSeed.reset();
|
|
transformedMasterKey.reset();
|
|
challengeResponseKey.reset();
|
|
|
|
hasKey = false;
|
|
key.reset();
|
|
kdf.reset();
|
|
|
|
publicCustomData.clear();
|
|
}
|
|
};
|
|
|
|
void createRecycleBin();
|
|
|
|
bool writeDatabase(QIODevice* device, QString* error = nullptr);
|
|
bool backupDatabase(const QString& filePath);
|
|
bool restoreDatabase(const QString& filePath);
|
|
bool performSave(const QString& filePath, QString* error, bool atomic, bool backup);
|
|
|
|
QPointer<Metadata> const m_metadata;
|
|
DatabaseData m_data;
|
|
QPointer<Group> m_rootGroup;
|
|
QList<DeletedObject> m_deletedObjects;
|
|
QTimer m_modifiedTimer;
|
|
QPointer<FileWatcher> m_fileWatcher;
|
|
bool m_initialized = false;
|
|
bool m_modified = false;
|
|
bool m_emitModified;
|
|
|
|
QList<QString> m_commonUsernames;
|
|
|
|
QUuid m_uuid;
|
|
static QHash<QUuid, QPointer<Database>> s_uuidMap;
|
|
};
|
|
|
|
#endif // KEEPASSX_DATABASE_H
|