mirror of
https://github.com/keepassxreboot/keepassxc.git
synced 2025-01-23 13:11:12 -05:00
Handle CustomData element.
This should make support for reading and writing KeePass 2 XML files complete (closes #1).
This commit is contained in:
parent
8df8f69e10
commit
3bf0564436
@ -19,7 +19,6 @@ configure_file( config-keepassx.h.cmake ${CMAKE_CURRENT_BINARY_DIR}/config-keepa
|
|||||||
|
|
||||||
set(keepassx_SOURCES
|
set(keepassx_SOURCES
|
||||||
core/Database.cpp
|
core/Database.cpp
|
||||||
core/DbAttribute.cpp
|
|
||||||
core/Entry.cpp
|
core/Entry.cpp
|
||||||
core/Group.cpp
|
core/Group.cpp
|
||||||
core/Metadata.cpp
|
core/Metadata.cpp
|
||||||
|
@ -20,8 +20,6 @@
|
|||||||
|
|
||||||
#include "Group.h"
|
#include "Group.h"
|
||||||
|
|
||||||
#include "DbAttribute.h"
|
|
||||||
|
|
||||||
#include <QtCore/QHash>
|
#include <QtCore/QHash>
|
||||||
#include <QtGui/QImage>
|
#include <QtGui/QImage>
|
||||||
|
|
||||||
@ -64,7 +62,6 @@ private:
|
|||||||
Metadata* m_metadata;
|
Metadata* m_metadata;
|
||||||
Group* m_rootGroup;
|
Group* m_rootGroup;
|
||||||
QHash<Uuid, QImage> m_customIcons;
|
QHash<Uuid, QImage> m_customIcons;
|
||||||
DbAttribute m_unhandledAttirbute;
|
|
||||||
QList<DeletedObject> m_deletedObjects;
|
QList<DeletedObject> m_deletedObjects;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -1,22 +0,0 @@
|
|||||||
/*
|
|
||||||
* 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/>.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include "DbAttribute.h"
|
|
||||||
|
|
||||||
DbAttribute::DbAttribute()
|
|
||||||
{
|
|
||||||
}
|
|
@ -1,33 +0,0 @@
|
|||||||
/*
|
|
||||||
* 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_DBATTRIBUTE_H
|
|
||||||
#define KEEPASSX_DBATTRIBUTE_H
|
|
||||||
|
|
||||||
#include <QtCore/QHash>
|
|
||||||
|
|
||||||
class DbAttribute
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
DbAttribute();
|
|
||||||
|
|
||||||
private:
|
|
||||||
QHash<QString, QString> properties;
|
|
||||||
QHash<QString, DbAttribute> children;
|
|
||||||
};
|
|
||||||
|
|
||||||
#endif // KEEPASSX_DBATTRIBUTE_H
|
|
@ -210,9 +210,31 @@ void Parser::parseCustomData()
|
|||||||
{
|
{
|
||||||
Q_ASSERT(m_xml.isStartElement() && m_xml.name() == "CustomData");
|
Q_ASSERT(m_xml.isStartElement() && m_xml.name() == "CustomData");
|
||||||
|
|
||||||
// TODO implement
|
|
||||||
while (!m_xml.error() && m_xml.readNextStartElement()) {
|
while (!m_xml.error() && m_xml.readNextStartElement()) {
|
||||||
skipCurrentElement();
|
if (m_xml.name() == "Item") {
|
||||||
|
parseCustomDataItem();
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
skipCurrentElement();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Parser::parseCustomDataItem()
|
||||||
|
{
|
||||||
|
Q_ASSERT(m_xml.isStartElement() && m_xml.name() == "Item");
|
||||||
|
|
||||||
|
QString key;
|
||||||
|
while (!m_xml.error() && m_xml.readNextStartElement()) {
|
||||||
|
if (m_xml.name() == "Key") {
|
||||||
|
key = readString();
|
||||||
|
}
|
||||||
|
else if (m_xml.name() == "Value") {
|
||||||
|
m_meta->addCustomField(key, readString());
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
skipCurrentElement();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -46,6 +46,7 @@ private:
|
|||||||
void parseCustomIcons();
|
void parseCustomIcons();
|
||||||
void parseIcon();
|
void parseIcon();
|
||||||
void parseCustomData();
|
void parseCustomData();
|
||||||
|
void parseCustomDataItem();
|
||||||
void parseRoot();
|
void parseRoot();
|
||||||
Group* parseGroup();
|
Group* parseGroup();
|
||||||
void parseDeletedObjects();
|
void parseDeletedObjects();
|
||||||
|
@ -100,8 +100,9 @@ void Writer::writeCustomIcons()
|
|||||||
{
|
{
|
||||||
m_xml.writeStartElement("CustomIcons");
|
m_xml.writeStartElement("CustomIcons");
|
||||||
|
|
||||||
Q_FOREACH (const Uuid& uuid, m_meta->customIcons().keys()) {
|
QHash<Uuid, QImage> customIcons = m_meta->customIcons();
|
||||||
writeIcon(uuid, m_meta->customIcons().value(uuid));
|
Q_FOREACH (const Uuid& uuid, customIcons.keys()) {
|
||||||
|
writeIcon(uuid, customIcons.value(uuid));
|
||||||
}
|
}
|
||||||
|
|
||||||
m_xml.writeEndElement();
|
m_xml.writeEndElement();
|
||||||
@ -127,7 +128,20 @@ void Writer::writeCustomData()
|
|||||||
{
|
{
|
||||||
m_xml.writeStartElement("CustomData");
|
m_xml.writeStartElement("CustomData");
|
||||||
|
|
||||||
// TODO implement
|
QHash<QString, QString> customFields = m_meta->customFields();
|
||||||
|
Q_FOREACH (const QString& key, customFields.keys()) {
|
||||||
|
writeCustomDataItem(key, customFields.value(key));
|
||||||
|
}
|
||||||
|
|
||||||
|
m_xml.writeEndElement();
|
||||||
|
}
|
||||||
|
|
||||||
|
void Writer::writeCustomDataItem(const QString& key, const QString& value)
|
||||||
|
{
|
||||||
|
m_xml.writeStartElement("Item");
|
||||||
|
|
||||||
|
writeString("Key", key);
|
||||||
|
writeString("Value", value);
|
||||||
|
|
||||||
m_xml.writeEndElement();
|
m_xml.writeEndElement();
|
||||||
}
|
}
|
||||||
|
@ -46,6 +46,7 @@ private:
|
|||||||
void writeCustomIcons();
|
void writeCustomIcons();
|
||||||
void writeIcon(const Uuid& uuid, const QImage& image);
|
void writeIcon(const Uuid& uuid, const QImage& image);
|
||||||
void writeCustomData();
|
void writeCustomData();
|
||||||
|
void writeCustomDataItem(const QString& key, const QString& value);
|
||||||
void writeRoot();
|
void writeRoot();
|
||||||
void writeGroup(const Group* group);
|
void writeGroup(const Group* group);
|
||||||
void writeTimes(const TimeInfo& ti);
|
void writeTimes(const TimeInfo& ti);
|
||||||
|
@ -30,7 +30,16 @@
|
|||||||
<EntryTemplatesGroupChanged>2010-08-08T17:24:19Z</EntryTemplatesGroupChanged>
|
<EntryTemplatesGroupChanged>2010-08-08T17:24:19Z</EntryTemplatesGroupChanged>
|
||||||
<LastSelectedGroup>lmU+9n0aeESKZvcEze+bRg==</LastSelectedGroup>
|
<LastSelectedGroup>lmU+9n0aeESKZvcEze+bRg==</LastSelectedGroup>
|
||||||
<LastTopVisibleGroup>lmU+9n0aeESKZvcEze+bRg==</LastTopVisibleGroup>
|
<LastTopVisibleGroup>lmU+9n0aeESKZvcEze+bRg==</LastTopVisibleGroup>
|
||||||
<CustomData />
|
<CustomData>
|
||||||
|
<Item>
|
||||||
|
<Key>A Sample Test Key</Key>
|
||||||
|
<Value>valu</Value>
|
||||||
|
</Item>
|
||||||
|
<Item>
|
||||||
|
<Key>custom key</Key>
|
||||||
|
<Value>blub</Value>
|
||||||
|
</Item>
|
||||||
|
</CustomData>
|
||||||
</Meta>
|
</Meta>
|
||||||
<Root>
|
<Root>
|
||||||
<Group>
|
<Group>
|
||||||
|
@ -40,7 +40,8 @@ class TestParser : public QObject
|
|||||||
private Q_SLOTS:
|
private Q_SLOTS:
|
||||||
void initTestCase();
|
void initTestCase();
|
||||||
void testMetadata();
|
void testMetadata();
|
||||||
void testCustomIcon();
|
void testCustomIcons();
|
||||||
|
void testCustomData();
|
||||||
void testGroupRoot();
|
void testGroupRoot();
|
||||||
void testGroup1();
|
void testGroup1();
|
||||||
void testGroup2();
|
void testGroup2();
|
||||||
@ -97,7 +98,7 @@ void TestParser::testMetadata()
|
|||||||
QVERIFY(m_db->metadata()->lastTopVisibleGroup() == m_db->metadata()->lastSelectedGroup());
|
QVERIFY(m_db->metadata()->lastTopVisibleGroup() == m_db->metadata()->lastSelectedGroup());
|
||||||
}
|
}
|
||||||
|
|
||||||
void TestParser::testCustomIcon()
|
void TestParser::testCustomIcons()
|
||||||
{
|
{
|
||||||
QCOMPARE(m_db->metadata()->customIcons().size(), 1);
|
QCOMPARE(m_db->metadata()->customIcons().size(), 1);
|
||||||
Uuid uuid = Uuid::fromBase64("++vyI+daLk6omox4a6kQGA==");
|
Uuid uuid = Uuid::fromBase64("++vyI+daLk6omox4a6kQGA==");
|
||||||
@ -115,6 +116,15 @@ void TestParser::testCustomIcon()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void TestParser::testCustomData()
|
||||||
|
{
|
||||||
|
QHash<QString, QString> customFields = m_db->metadata()->customFields();
|
||||||
|
|
||||||
|
QCOMPARE(customFields.size(), 2);
|
||||||
|
QCOMPARE(customFields.value("A Sample Test Key"), QString("valu"));
|
||||||
|
QCOMPARE(customFields.value("custom key"), QString("blub"));
|
||||||
|
}
|
||||||
|
|
||||||
void TestParser::testGroupRoot()
|
void TestParser::testGroupRoot()
|
||||||
{
|
{
|
||||||
const Group* group = m_db->rootGroup();
|
const Group* group = m_db->rootGroup();
|
||||||
|
Loading…
Reference in New Issue
Block a user