mirror of
https://github.com/keepassxreboot/keepassxc.git
synced 2025-05-11 19:15:18 -04:00
Refactor database readers/writers and XML handling
* Refactor Kdbx*Reader * Refactor KdbxWriter * Refactor KdbxXmlReader * Refactor KdbxXmlWriter
This commit is contained in:
parent
72a1c65d00
commit
a6ddc22fb8
29 changed files with 1313 additions and 2917 deletions
|
@ -15,29 +15,21 @@
|
|||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#include <QtGlobal>
|
||||
#include <QString>
|
||||
#include <QFile>
|
||||
|
||||
#include "core/Endian.h"
|
||||
#include "keys/CompositeKey.h"
|
||||
#include "format/KeePass2Reader.h"
|
||||
#include "format/KeePass1.h"
|
||||
#include "format/KeePass2.h"
|
||||
#include "format/Kdbx3Reader.h"
|
||||
#include "format/Kdbx4Reader.h"
|
||||
|
||||
BaseKeePass2Reader::BaseKeePass2Reader()
|
||||
: m_error(false)
|
||||
, m_saveXml(false)
|
||||
, m_irsAlgo(KeePass2::ProtectedStreamAlgo::InvalidProtectedStreamAlgo)
|
||||
{
|
||||
m_errorStr.clear();
|
||||
m_xmlData.clear();
|
||||
m_protectedStreamKey.clear();
|
||||
}
|
||||
#include <QFile>
|
||||
|
||||
Database* BaseKeePass2Reader::readDatabase(const QString& filename, const CompositeKey& key)
|
||||
/**
|
||||
* Read database from file and detect correct file format.
|
||||
*
|
||||
* @param filename input file
|
||||
* @param key database encryption composite key
|
||||
* @return pointer to the read database, nullptr on failure
|
||||
*/
|
||||
Database* KeePass2Reader::readDatabase(const QString& filename, const CompositeKey& key)
|
||||
{
|
||||
QFile file(filename);
|
||||
if (!file.open(QFile::ReadOnly)) {
|
||||
|
@ -55,79 +47,45 @@ Database* BaseKeePass2Reader::readDatabase(const QString& filename, const Compos
|
|||
return db.take();
|
||||
}
|
||||
|
||||
bool BaseKeePass2Reader::hasError()
|
||||
{
|
||||
return m_error;
|
||||
}
|
||||
|
||||
QString BaseKeePass2Reader::errorString()
|
||||
{
|
||||
return m_errorStr;
|
||||
}
|
||||
|
||||
void BaseKeePass2Reader::setSaveXml(bool save)
|
||||
{
|
||||
m_saveXml = save;
|
||||
}
|
||||
|
||||
QByteArray BaseKeePass2Reader::xmlData()
|
||||
{
|
||||
return m_xmlData;
|
||||
}
|
||||
|
||||
QByteArray BaseKeePass2Reader::streamKey()
|
||||
{
|
||||
return m_protectedStreamKey;
|
||||
}
|
||||
|
||||
KeePass2::ProtectedStreamAlgo BaseKeePass2Reader::protectedStreamAlgo() const {
|
||||
return m_irsAlgo;
|
||||
}
|
||||
|
||||
|
||||
void BaseKeePass2Reader::raiseError(const QString& errorMessage)
|
||||
{
|
||||
m_error = true;
|
||||
m_errorStr = errorMessage;
|
||||
}
|
||||
|
||||
/**
|
||||
* Read database from device and detect correct file format.
|
||||
*
|
||||
* @param device input device
|
||||
* @param key database encryption composite key
|
||||
* @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)
|
||||
{
|
||||
m_error = false;
|
||||
m_errorStr.clear();
|
||||
|
||||
bool ok;
|
||||
quint32 signature1, signature2;
|
||||
bool ok = KdbxReader::readMagicNumbers(device, signature1, signature2, m_version);
|
||||
|
||||
quint32 signature1 = Endian::readSizedInt<quint32>(device, KeePass2::BYTEORDER, &ok);
|
||||
if (!ok || signature1 != KeePass2::SIGNATURE_1) {
|
||||
// mask out minor version
|
||||
m_version &= KeePass2::FILE_VERSION_CRITICAL_MASK;
|
||||
|
||||
if (!ok || signature1 != KeePass2::SIGNATURE_1 || signature2 != KeePass2::SIGNATURE_2) {
|
||||
raiseError(tr("Not a KeePass database."));
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
quint32 signature2 = Endian::readSizedInt<quint32>(device, KeePass2::BYTEORDER, &ok);
|
||||
if (ok && signature2 == KeePass1::SIGNATURE_2) {
|
||||
if (signature2 == KeePass1::SIGNATURE_2) {
|
||||
raiseError(tr("The selected file is an old KeePass 1 database (.kdb).\n\n"
|
||||
"You can import it by clicking on Database > 'Import KeePass 1 database...'.\n"
|
||||
"This is a one-way migration. You won't be able to open the imported "
|
||||
"database with the old KeePassX 0.4 version."));
|
||||
return nullptr;
|
||||
}
|
||||
else if (!ok || signature2 != KeePass2::SIGNATURE_2) {
|
||||
raiseError(tr("Not a KeePass database."));
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
m_version = Endian::readSizedInt<quint32>(device, KeePass2::BYTEORDER, &ok)
|
||||
& KeePass2::FILE_VERSION_CRITICAL_MASK;
|
||||
quint32 maxVersion = KeePass2::FILE_VERSION_4 & KeePass2::FILE_VERSION_CRITICAL_MASK;
|
||||
if (!ok || (m_version < KeePass2::FILE_VERSION_MIN) || (m_version > maxVersion)) {
|
||||
if (m_version < KeePass2::FILE_VERSION_MIN || m_version > maxVersion) {
|
||||
raiseError(tr("Unsupported KeePass 2 database version."));
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
device->seek(0);
|
||||
|
||||
// Determine KDBX3 vs KDBX4
|
||||
// determine file format (KDBX 2/3 or 4)
|
||||
if (m_version < KeePass2::FILE_VERSION_4) {
|
||||
m_reader.reset(new Kdbx3Reader());
|
||||
} else {
|
||||
|
@ -138,37 +96,49 @@ Database* KeePass2Reader::readDatabase(QIODevice* device, const CompositeKey& ke
|
|||
return m_reader->readDatabase(device, key, keepDatabase);
|
||||
}
|
||||
|
||||
bool KeePass2Reader::hasError()
|
||||
bool KeePass2Reader::hasError() const
|
||||
{
|
||||
return m_error || (!m_reader.isNull() && m_reader->hasError());
|
||||
}
|
||||
|
||||
QString KeePass2Reader::errorString()
|
||||
QString KeePass2Reader::errorString() const
|
||||
{
|
||||
return !m_reader.isNull() ? m_reader->errorString() : m_errorStr;
|
||||
}
|
||||
|
||||
QByteArray KeePass2Reader::xmlData()
|
||||
bool KeePass2Reader::saveXml() const
|
||||
{
|
||||
return !m_reader.isNull() ? m_reader->xmlData() : m_xmlData;
|
||||
return m_saveXml;
|
||||
}
|
||||
|
||||
QByteArray KeePass2Reader::streamKey()
|
||||
void KeePass2Reader::setSaveXml(bool save)
|
||||
{
|
||||
return !m_reader.isNull() ? m_reader->streamKey() : m_protectedStreamKey;
|
||||
}
|
||||
|
||||
KeePass2::ProtectedStreamAlgo KeePass2Reader::protectedStreamAlgo() const
|
||||
{
|
||||
return !m_reader.isNull() ? m_reader->protectedStreamAlgo() : m_irsAlgo;
|
||||
m_saveXml = save;
|
||||
}
|
||||
|
||||
/**
|
||||
* @return detected KDBX version
|
||||
*/
|
||||
quint32 KeePass2Reader::version() const
|
||||
{
|
||||
return m_version;
|
||||
}
|
||||
|
||||
QSharedPointer<BaseKeePass2Reader> KeePass2Reader::reader()
|
||||
/**
|
||||
* @return KDBX reader used for reading the input file
|
||||
*/
|
||||
QSharedPointer<KdbxReader> KeePass2Reader::reader() const
|
||||
{
|
||||
return m_reader;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Raise an error. Use in case of an unexpected read error.
|
||||
*
|
||||
* @param errorMessage error message
|
||||
*/
|
||||
void KeePass2Reader::raiseError(const QString& errorMessage)
|
||||
{
|
||||
m_error = true;
|
||||
m_errorStr = errorMessage;
|
||||
}
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue