Generate random 128-byte stream instead of legacy XML format when creating key files

Add API documentation for FileKey class
Resolves #1325
This commit is contained in:
Janek Bevendorff 2017-12-26 21:55:08 +01:00
parent ac6a9e7696
commit 901bf62213
2 changed files with 123 additions and 56 deletions

View File

@ -1,4 +1,5 @@
/* /*
* Copyright (C) 2017 KeePassXC Team <team@keepassxc.org>
* Copyright (C) 2011 Felix Geyer <debfx@fobos.de> * Copyright (C) 2011 Felix Geyer <debfx@fobos.de>
* *
* This program is free software: you can redistribute it and/or modify * This program is free software: you can redistribute it and/or modify
@ -18,16 +19,30 @@
#include "FileKey.h" #include "FileKey.h"
#include <QFile> #include <QFile>
#include <QXmlStreamReader>
#include "core/Tools.h" #include "core/Tools.h"
#include "crypto/CryptoHash.h" #include "crypto/CryptoHash.h"
#include "crypto/Random.h" #include "crypto/Random.h"
FileKey::FileKey() /**
{ * Read key file from device while trying to detect its file format.
} *
* If no legacy key file format was detected, the SHA-256 hash of the
* key file will be used, allowing usage of arbitrary files as key files.
* In case of a detected legacy key file format, the raw byte contents
* will be extracted from the file.
*
* Supported legacy formats are:
* - KeePass 2 XML key file
* - Fixed 32 byte binary
* - Fixed 32 byte ASCII hex-encoded binary
*
* Usage of legacy formats is discouraged and support for them may be
* removed in a future version.
*
* @param device input device
* @return true if key file was loaded successfully
*/
bool FileKey::load(QIODevice* device) bool FileKey::load(QIODevice* device)
{ {
// we may need to read the file multiple times // we may need to read the file multiple times
@ -39,8 +54,7 @@ bool FileKey::load(QIODevice* device)
return false; return false;
} }
// try different key file formats // try different legacy key file formats
if (!device->reset()) { if (!device->reset()) {
return false; return false;
} }
@ -65,13 +79,31 @@ bool FileKey::load(QIODevice* device)
if (!device->reset()) { if (!device->reset()) {
return false; return false;
} }
if (loadHashed(device)) {
return true;
}
return false; // if no legacy format was detected, generate SHA-256 hash of key file
return loadHashed(device);
} }
/**
* Load key file from path while trying to detect its file format.
*
* If no legacy key file format was detected, the SHA-256 hash of the
* key file will be used, allowing usage of arbitrary files as key files.
* In case of a detected legacy key file format, the raw byte contents
* will be extracted from the file.
*
* Supported legacy formats are:
* - KeePass 2 XML key file
* - Fixed 32 byte binary
* - Fixed 32 byte ASCII hex-encoded binary
*
* Usage of legacy formats is discouraged and support for them may be
* removed in a future version.
*
* @param fileName input file name
* @param errorMsg error message if loading failed
* @return true if key file was loaded successfully
*/
bool FileKey::load(const QString& fileName, QString* errorMsg) bool FileKey::load(const QString& fileName, QString* errorMsg)
{ {
QFile file(fileName); QFile file(fileName);
@ -95,41 +127,42 @@ bool FileKey::load(const QString& fileName, QString* errorMsg)
return result; return result;
} }
/**
* @return key data as bytes
*/
QByteArray FileKey::rawKey() const QByteArray FileKey::rawKey() const
{ {
return m_key; return m_key;
} }
/**
* @return cloned \link FileKey instance
*/
FileKey* FileKey::clone() const FileKey* FileKey::clone() const
{ {
return new FileKey(*this); return new FileKey(*this);
} }
void FileKey::create(QIODevice* device) /**
* Generate a new key file from random bytes.
*
* @param device output device
* @param number of random bytes to generate
*/
void FileKey::create(QIODevice* device, int size)
{ {
QXmlStreamWriter xmlWriter(device); device->write(randomGen()->randomArray(size));
xmlWriter.writeStartDocument("1.0");
xmlWriter.writeStartElement("KeyFile");
xmlWriter.writeStartElement("Meta");
xmlWriter.writeTextElement("Version", "1.00");
xmlWriter.writeEndElement();
xmlWriter.writeStartElement("Key");
QByteArray data = randomGen()->randomArray(32);
xmlWriter.writeTextElement("Data", QString::fromLatin1(data.toBase64()));
xmlWriter.writeEndElement();
xmlWriter.writeEndDocument();
} }
bool FileKey::create(const QString& fileName, QString* errorMsg) /**
* Create a new key file from random bytes.
*
* @param fileName output file name
* @param errorMsg error message if generation failed
* @param number of random bytes to generate
* @return true on successful creation
*/
bool FileKey::create(const QString& fileName, QString* errorMsg, int size)
{ {
QFile file(fileName); QFile file(fileName);
if (!file.open(QFile::WriteOnly)) { if (!file.open(QFile::WriteOnly)) {
@ -138,8 +171,7 @@ bool FileKey::create(const QString& fileName, QString* errorMsg)
} }
return false; return false;
} }
create(&file); create(&file, size);
file.close(); file.close();
if (file.error()) { if (file.error()) {
@ -149,11 +181,17 @@ bool FileKey::create(const QString& fileName, QString* errorMsg)
return false; return false;
} }
else {
return true; return true;
}
} }
/**
* Load key file in legacy KeePass 2 XML format.
*
* @param device input device
* @return true on success
* @deprecated
*/
bool FileKey::loadXml(QIODevice* device) bool FileKey::loadXml(QIODevice* device)
{ {
QXmlStreamReader xmlReader(device); QXmlStreamReader xmlReader(device);
@ -162,8 +200,7 @@ bool FileKey::loadXml(QIODevice* device)
if (xmlReader.name() != "KeyFile") { if (xmlReader.name() != "KeyFile") {
return false; return false;
} }
} } else {
else {
return false; return false;
} }
@ -173,8 +210,7 @@ bool FileKey::loadXml(QIODevice* device)
while (!xmlReader.error() && xmlReader.readNextStartElement()) { while (!xmlReader.error() && xmlReader.readNextStartElement()) {
if (xmlReader.name() == "Meta") { if (xmlReader.name() == "Meta") {
correctMeta = loadXmlMeta(xmlReader); correctMeta = loadXmlMeta(xmlReader);
} } else if (xmlReader.name() == "Key") {
else if (xmlReader.name() == "Key") {
data = loadXmlKey(xmlReader); data = loadXmlKey(xmlReader);
} }
} }
@ -183,18 +219,23 @@ bool FileKey::loadXml(QIODevice* device)
m_key = data; m_key = data;
return true; return true;
} }
else {
return false; return false;
}
} }
/**
* Load meta data from legacy KeePass 2 XML key file.
*
* @param xmlReader input XML reader
* @return true on success
* @deprecated
*/
bool FileKey::loadXmlMeta(QXmlStreamReader& xmlReader) bool FileKey::loadXmlMeta(QXmlStreamReader& xmlReader)
{ {
bool correctVersion = false; bool correctVersion = false;
while (!xmlReader.error() && xmlReader.readNextStartElement()) { while (!xmlReader.error() && xmlReader.readNextStartElement()) {
if (xmlReader.name() == "Version") { if (xmlReader.name() == "Version") {
// TODO: error message about incompatible key file version
if (xmlReader.readElementText() == "1.00") { if (xmlReader.readElementText() == "1.00") {
correctVersion = true; correctVersion = true;
} }
@ -204,13 +245,19 @@ bool FileKey::loadXmlMeta(QXmlStreamReader& xmlReader)
return correctVersion; return correctVersion;
} }
/**
* Load base64 key data from legacy KeePass 2 XML key file.
*
* @param xmlReader input XML reader
* @return true on success
* @deprecated
*/
QByteArray FileKey::loadXmlKey(QXmlStreamReader& xmlReader) QByteArray FileKey::loadXmlKey(QXmlStreamReader& xmlReader)
{ {
QByteArray data; QByteArray data;
while (!xmlReader.error() && xmlReader.readNextStartElement()) { while (!xmlReader.error() && xmlReader.readNextStartElement()) {
if (xmlReader.name() == "Data") { if (xmlReader.name() == "Data") {
// TODO: do we need to enforce a specific data.size()?
QByteArray rawData = xmlReader.readElementText().toLatin1(); QByteArray rawData = xmlReader.readElementText().toLatin1();
if (Tools::isBase64(rawData)) { if (Tools::isBase64(rawData)) {
data = QByteArray::fromBase64(rawData); data = QByteArray::fromBase64(rawData);
@ -221,6 +268,13 @@ QByteArray FileKey::loadXmlKey(QXmlStreamReader& xmlReader)
return data; return data;
} }
/**
* Load fixed 32-bit binary key file.
*
* @param device input device
* @return true on success
* @deprecated
*/
bool FileKey::loadBinary(QIODevice* device) bool FileKey::loadBinary(QIODevice* device)
{ {
if (device->size() != 32) { if (device->size() != 32) {
@ -230,13 +284,19 @@ bool FileKey::loadBinary(QIODevice* device)
QByteArray data; QByteArray data;
if (!Tools::readAllFromDevice(device, data) || data.size() != 32) { if (!Tools::readAllFromDevice(device, data) || data.size() != 32) {
return false; return false;
} } else {
else {
m_key = data; m_key = data;
return true; return true;
} }
} }
/**
* Load hex-encoded representation of fixed 32-bit binary key file.
*
* @param device input device
* @return true on success
* @deprecated
*/
bool FileKey::loadHex(QIODevice* device) bool FileKey::loadHex(QIODevice* device)
{ {
if (device->size() != 64) { if (device->size() != 64) {
@ -262,6 +322,12 @@ bool FileKey::loadHex(QIODevice* device)
return true; return true;
} }
/**
* Generate SHA-256 hash of arbitrary text or binary key file.
*
* @param device input device
* @return true on success
*/
bool FileKey::loadHashed(QIODevice* device) bool FileKey::loadHashed(QIODevice* device)
{ {
CryptoHash cryptoHash(CryptoHash::Sha256); CryptoHash cryptoHash(CryptoHash::Sha256);
@ -272,7 +338,8 @@ bool FileKey::loadHashed(QIODevice* device)
return false; return false;
} }
cryptoHash.addData(buffer); cryptoHash.addData(buffer);
} while (!buffer.isEmpty()); }
while (!buffer.isEmpty());
m_key = cryptoHash.result(); m_key = cryptoHash.result();

View File

@ -1,4 +1,5 @@
/* /*
* Copyright (C) 2017 KeePassXC Team <team@keepassxc.org>
* Copyright (C) 2011 Felix Geyer <debfx@fobos.de> * Copyright (C) 2011 Felix Geyer <debfx@fobos.de>
* *
* This program is free software: you can redistribute it and/or modify * This program is free software: you can redistribute it and/or modify
@ -24,16 +25,15 @@
class QIODevice; class QIODevice;
class FileKey : public Key class FileKey: public Key
{ {
public: public:
FileKey();
bool load(QIODevice* device); bool load(QIODevice* device);
bool load(const QString& fileName, QString* errorMsg = nullptr); bool load(const QString& fileName, QString* errorMsg = nullptr);
QByteArray rawKey() const; QByteArray rawKey() const override;
FileKey* clone() const; FileKey* clone() const override;
static void create(QIODevice* device); static void create(QIODevice* device, int size = 128);
static bool create(const QString& fileName, QString* errorMsg = nullptr); static bool create(const QString& fileName, QString* errorMsg = nullptr, int size = 128);
private: private:
bool loadXml(QIODevice* device); bool loadXml(QIODevice* device);