/* * Copyright (C) 2018 KeePassXC Team * Copyright (C) 2010 Felix Geyer * * 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 . */ #ifndef KEEPASSX_DATABASE_H #define KEEPASSX_DATABASE_H #include #include #include #include #include #include "config-keepassx.h" #include "core/ModifiableObject.h" #include "crypto/kdf/AesKdf.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 ModifiableObject { Q_OBJECT public: enum CompressionAlgorithm { CompressionNone = 0, CompressionGZip = 1 }; static const quint32 CompressionAlgorithmMax = CompressionGZip; enum SaveAction { Atomic, // Saves are transactional and atomic TempFile, // Write to a temporary location then move into place, may be non-atomic DirectWrite, // Directly write to the destination file (dangerous) }; Database(); explicit Database(const QString& filePath); ~Database() override; private: bool writeDatabase(QIODevice* device, QString* error = nullptr); bool backupDatabase(const QString& filePath, const QString& destinationFilePath); bool restoreDatabase(const QString& filePath, const QString& fromBackupFilePath); bool performSave(const QString& filePath, SaveAction flags, const QString& backupFilePath, QString* error); public: bool open(QSharedPointer key, QString* error = nullptr); bool open(const QString& filePath, QSharedPointer key, QString* error = nullptr); bool save(SaveAction action = Atomic, const QString& backupFilePath = QString(), QString* error = nullptr); bool saveAs(const QString& filePath, SaveAction action = Atomic, const QString& backupFilePath = QString(), QString* error = nullptr); bool extract(QByteArray&, QString* error = nullptr); bool import(const QString& xmlExportPath, QString* error = nullptr); quint32 formatVersion() const; void setFormatVersion(quint32 version); bool hasMinorVersionMismatch() const; void releaseData(); bool isInitialized() const; bool isModified() const; bool hasNonDataChanges() const; bool isSaving(); 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 deletedObjects(); const QList& 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& delObjs); const QStringList& commonUsernames() const; const QStringList& tagList() const; void removeTag(const QString& tag); QSharedPointer key() const; bool setKey(const QSharedPointer& key, bool updateChangedTime = true, bool updateTransformSalt = false, bool transformKey = true); QString keyError(); QByteArray challengeResponseKey() const; bool challengeMasterSeed(const QByteArray& masterSeed); const QUuid& cipher() const; void setCipher(const QUuid& cipher); Database::CompressionAlgorithm compressionAlgorithm() const; void setCompressionAlgorithm(Database::CompressionAlgorithm algo); QSharedPointer kdf() const; void setKdf(QSharedPointer kdf); bool changeKdf(const QSharedPointer& kdf); QByteArray transformedDatabaseKey() const; static Database* databaseByUuid(const QUuid& uuid); public slots: void markAsModified(); void markAsClean(); void updateCommonUsernames(int topN = 10); void updateTagList(); void markNonDataChange(); 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 databaseSaved(); void databaseDiscarded(); void databaseFileChanged(); void tagListUpdated(); private: struct DatabaseData { quint32 formatVersion = 0; QString filePath; QUuid cipher = KeePass2::CIPHER_AES256; CompressionAlgorithm compressionAlgorithm = CompressionGZip; QScopedPointer masterSeed; QScopedPointer transformedDatabaseKey; QScopedPointer challengeResponseKey; QSharedPointer key; QSharedPointer kdf = QSharedPointer::create(true); QVariantMap publicCustomData; DatabaseData() : masterSeed(new PasswordKey()) , transformedDatabaseKey(new PasswordKey()) , challengeResponseKey(new PasswordKey()) { kdf->randomizeSeed(); } void clear() { filePath.clear(); masterSeed.reset(); transformedDatabaseKey.reset(); challengeResponseKey.reset(); key.reset(); kdf.reset(); publicCustomData.clear(); } }; void createRecycleBin(); void startModifiedTimer(); void stopModifiedTimer(); QPointer const m_metadata; DatabaseData m_data; QPointer m_rootGroup; QList m_deletedObjects; QTimer m_modifiedTimer; QMutex m_saveMutex; QPointer m_fileWatcher; bool m_modified = false; bool m_hasNonDataChange = false; QString m_keyError; QStringList m_commonUsernames; QStringList m_tagList; QUuid m_uuid; static QHash> s_uuidMap; }; #endif // KEEPASSX_DATABASE_H