/* * Copyright (C) 2010 Felix Geyer * Copyright (C) 2017 KeePassXC Team * * 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 "crypto/kdf/Kdf.h" #include "keys/CompositeKey.h" class Entry; enum class EntryReferenceType; class Group; class Metadata; class QTimer; 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; struct DatabaseData { QUuid cipher; CompressionAlgorithm compressionAlgo; QByteArray transformedMasterKey; QSharedPointer kdf; QSharedPointer key; bool hasKey; QByteArray masterSeed; QByteArray challengeResponseKey; QVariantMap publicCustomData; }; Database(); ~Database() override; Group* rootGroup(); const Group* rootGroup() const; /** * Sets group as the root group and takes ownership of it. * Warning: Be careful when calling this method as it doesn't * emit any notifications so e.g. models aren't updated. * The caller is responsible for cleaning up the previous root group. */ void setRootGroup(Group* group); Metadata* metadata(); const Metadata* metadata() const; QString filePath() const; void setFilePath(const QString& filePath); Entry* resolveEntry(const QUuid& uuid); Entry* resolveEntry(const QString& text, EntryReferenceType referenceType); Group* resolveGroup(const QUuid& uuid); 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 QUuid& cipher() const; Database::CompressionAlgorithm compressionAlgo() const; QSharedPointer kdf() const; QByteArray transformedMasterKey() const; QSharedPointer key() const; QByteArray challengeResponseKey() const; bool challengeMasterSeed(const QByteArray& masterSeed); void setCipher(const QUuid& cipher); void setCompressionAlgo(Database::CompressionAlgorithm algo); void setKdf(QSharedPointer kdf); bool setKey(QSharedPointer key, bool updateChangedTime = true, bool updateTransformSalt = false); bool hasKey() const; bool verifyKey(QSharedPointer key) const; QVariantMap& publicCustomData(); const QVariantMap& publicCustomData() const; void setPublicCustomData(const QVariantMap& customData); void recycleEntry(Entry* entry); void recycleGroup(Group* group); void emptyRecycleBin(); void setEmitModified(bool value); void markAsModified(); QString saveToFile(QString filePath, bool atomic = true, bool backup = false); /** * Returns a unique id that is only valid as long as the Database exists. */ const QUuid& uuid(); bool changeKdf(QSharedPointer kdf); static Database* databaseByUuid(const QUuid& uuid); static Database* openDatabaseFile(const QString& fileName, QSharedPointer key); static Database* unlockFromStdin(QString databaseFilename, QString keyFilename = QString("")); signals: 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 nameTextChanged(); void modified(); void modifiedImmediate(); private slots: void startModifiedTimer(); private: Entry* findEntryRecursive(const QUuid& uuid, Group* group); Entry* findEntryRecursive(const QString& text, EntryReferenceType referenceType, Group* group); Group* findGroupRecursive(const QUuid& uuid, Group* group); void createRecycleBin(); QString writeDatabase(QIODevice* device); bool backupDatabase(QString filePath); Metadata* const m_metadata; Group* m_rootGroup; QList m_deletedObjects; QTimer* m_timer; DatabaseData m_data; bool m_emitModified; QString m_filePath; QUuid m_uuid; static QHash m_uuidMap; }; #endif // KEEPASSX_DATABASE_H