2010-08-07 09:10:44 -04:00
|
|
|
/*
|
|
|
|
* 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/>.
|
|
|
|
*/
|
|
|
|
|
2010-08-31 08:39:35 -04:00
|
|
|
#include "KeePass2XmlReader.h"
|
2010-08-07 09:10:44 -04:00
|
|
|
|
|
|
|
#include <QtCore/QFile>
|
|
|
|
|
2010-08-31 08:39:35 -04:00
|
|
|
#include "core/Database.h"
|
2010-09-22 18:21:36 -04:00
|
|
|
#include "core/DatabaseIcons.h"
|
2010-08-31 08:39:35 -04:00
|
|
|
#include "core/Metadata.h"
|
2010-08-12 15:38:59 -04:00
|
|
|
|
2010-08-31 10:18:45 -04:00
|
|
|
KeePass2XmlReader::KeePass2XmlReader()
|
|
|
|
: m_db(0)
|
|
|
|
, m_meta(0)
|
2010-08-07 09:10:44 -04:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2010-09-25 06:41:00 -04:00
|
|
|
void KeePass2XmlReader::readDatabase(QIODevice* device, Database* db)
|
2010-08-07 09:10:44 -04:00
|
|
|
{
|
2010-08-31 10:18:45 -04:00
|
|
|
m_xml.setDevice(device);
|
|
|
|
|
2010-09-25 06:41:00 -04:00
|
|
|
m_db = db;
|
2010-08-31 10:18:45 -04:00
|
|
|
m_meta = m_db->metadata();
|
2010-08-07 09:10:44 -04:00
|
|
|
|
2010-08-12 15:38:59 -04:00
|
|
|
m_tmpParent = new Group();
|
2010-08-13 12:08:06 -04:00
|
|
|
m_tmpParent->setParent(m_db);
|
2010-08-07 09:10:44 -04:00
|
|
|
|
2010-08-12 15:38:59 -04:00
|
|
|
if (!m_xml.error() && m_xml.readNextStartElement()) {
|
2010-08-07 09:10:44 -04:00
|
|
|
if (m_xml.name() == "KeePassFile") {
|
|
|
|
parseKeePassFile();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-13 12:08:06 -04:00
|
|
|
if (!m_xml.error() && !m_tmpParent->children().isEmpty()) {
|
2010-08-12 15:38:59 -04:00
|
|
|
raiseError();
|
|
|
|
}
|
|
|
|
|
2010-08-13 12:08:06 -04:00
|
|
|
delete m_tmpParent;
|
2010-09-25 06:41:00 -04:00
|
|
|
}
|
2010-08-13 12:08:06 -04:00
|
|
|
|
2010-09-25 06:41:00 -04:00
|
|
|
Database* KeePass2XmlReader::readDatabase(QIODevice* device)
|
|
|
|
{
|
|
|
|
Database* db = new Database();
|
|
|
|
readDatabase(device, db);
|
|
|
|
return db;
|
2010-08-31 10:18:45 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
Database* KeePass2XmlReader::readDatabase(const QString& filename)
|
|
|
|
{
|
|
|
|
QFile file(filename);
|
2010-09-13 17:24:36 -04:00
|
|
|
file.open(QIODevice::ReadOnly);
|
2010-08-31 10:18:45 -04:00
|
|
|
return readDatabase(&file);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool KeePass2XmlReader::error()
|
|
|
|
{
|
|
|
|
return m_xml.hasError();
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
|
2010-08-31 10:18:45 -04:00
|
|
|
QString KeePass2XmlReader::errorString()
|
2010-08-13 12:08:06 -04:00
|
|
|
{
|
|
|
|
return QString("%1\nLine %2, column %3")
|
|
|
|
.arg(m_xml.errorString())
|
|
|
|
.arg(m_xml.lineNumber())
|
|
|
|
.arg(m_xml.columnNumber());
|
|
|
|
}
|
|
|
|
|
2010-08-31 08:39:35 -04:00
|
|
|
void KeePass2XmlReader::parseKeePassFile()
|
2010-08-07 09:10:44 -04:00
|
|
|
{
|
|
|
|
Q_ASSERT(m_xml.isStartElement() && m_xml.name() == "KeePassFile");
|
|
|
|
|
2010-08-12 15:38:59 -04:00
|
|
|
while (!m_xml.error() && m_xml.readNextStartElement()) {
|
2010-08-07 09:10:44 -04:00
|
|
|
if (m_xml.name() == "Meta") {
|
|
|
|
parseMeta();
|
|
|
|
}
|
|
|
|
else if (m_xml.name() == "Root") {
|
|
|
|
parseRoot();
|
|
|
|
}
|
|
|
|
else {
|
2010-08-13 12:08:06 -04:00
|
|
|
skipCurrentElement();
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-31 08:39:35 -04:00
|
|
|
void KeePass2XmlReader::parseMeta()
|
2010-08-07 09:10:44 -04:00
|
|
|
{
|
|
|
|
Q_ASSERT(m_xml.isStartElement() && m_xml.name() == "Meta");
|
|
|
|
|
2010-08-12 15:38:59 -04:00
|
|
|
while (!m_xml.error() && m_xml.readNextStartElement()) {
|
2010-08-07 09:10:44 -04:00
|
|
|
if (m_xml.name() == "Generator") {
|
2010-08-12 15:38:59 -04:00
|
|
|
m_meta->setGenerator(readString());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "DatabaseName") {
|
2010-08-12 15:38:59 -04:00
|
|
|
m_meta->setName(readString());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "DatabaseNameChanged") {
|
2010-08-12 15:38:59 -04:00
|
|
|
m_meta->setNameChanged(readDateTime());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "DatabaseDescription") {
|
2010-08-12 15:38:59 -04:00
|
|
|
m_meta->setDescription(readString());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "DatabaseDescriptionChanged") {
|
2010-08-12 15:38:59 -04:00
|
|
|
m_meta->setDescriptionChanged(readDateTime());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "DefaultUserName") {
|
2010-08-12 15:38:59 -04:00
|
|
|
m_meta->setDefaultUserName(readString());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "DefaultUserNameChanged") {
|
2010-08-12 15:38:59 -04:00
|
|
|
m_meta->setDefaultUserNameChanged(readDateTime());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "MaintenanceHistoryDays") {
|
2010-08-12 15:38:59 -04:00
|
|
|
m_meta->setMaintenanceHistoryDays(readNumber());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
2010-09-25 06:41:00 -04:00
|
|
|
else if (m_xml.name() == "MasterKeyChanged") {
|
|
|
|
m_meta->setMasterKeyChanged(readDateTime());
|
|
|
|
}
|
|
|
|
else if (m_xml.name() == "MasterKeyChangeRec") {
|
|
|
|
m_meta->setMasterKeyChangeRec(readNumber());
|
|
|
|
}
|
|
|
|
else if (m_xml.name() == "MasterKeyChangeForce") {
|
|
|
|
m_meta->setMasterKeyChangeForce(readNumber());
|
|
|
|
}
|
2010-08-07 09:10:44 -04:00
|
|
|
else if (m_xml.name() == "MemoryProtection") {
|
|
|
|
parseMemoryProtection();
|
|
|
|
}
|
|
|
|
else if (m_xml.name() == "CustomIcons") {
|
|
|
|
parseCustomIcons();
|
|
|
|
}
|
|
|
|
else if (m_xml.name() == "RecycleBinEnabled") {
|
2010-08-12 15:38:59 -04:00
|
|
|
m_meta->setRecycleBinEnabled(readBool());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "RecycleBinUUID") {
|
2010-08-13 12:08:06 -04:00
|
|
|
m_meta->setRecycleBin(getGroup(readUuid()));
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "RecycleBinChanged") {
|
2010-08-12 15:38:59 -04:00
|
|
|
m_meta->setRecycleBinChanged(readDateTime());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "EntryTemplatesGroup") {
|
2010-08-13 12:08:06 -04:00
|
|
|
m_meta->setEntryTemplatesGroup(getGroup(readUuid()));
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "EntryTemplatesGroupChanged") {
|
2010-08-12 15:38:59 -04:00
|
|
|
m_meta->setEntryTemplatesGroupChanged(readDateTime());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "LastSelectedGroup") {
|
2010-08-13 12:08:06 -04:00
|
|
|
m_meta->setLastSelectedGroup(getGroup(readUuid()));
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "LastTopVisibleGroup") {
|
2010-08-13 12:08:06 -04:00
|
|
|
m_meta->setLastTopVisibleGroup(getGroup(readUuid()));
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "CustomData") {
|
|
|
|
parseCustomData();
|
|
|
|
}
|
|
|
|
else {
|
2010-08-13 12:08:06 -04:00
|
|
|
skipCurrentElement();
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-31 08:39:35 -04:00
|
|
|
void KeePass2XmlReader::parseMemoryProtection()
|
2010-08-07 09:10:44 -04:00
|
|
|
{
|
|
|
|
Q_ASSERT(m_xml.isStartElement() && m_xml.name() == "MemoryProtection");
|
|
|
|
|
2010-08-12 15:38:59 -04:00
|
|
|
while (!m_xml.error() && m_xml.readNextStartElement()) {
|
2010-08-07 09:10:44 -04:00
|
|
|
if (m_xml.name() == "ProtectTitle") {
|
2010-08-12 15:38:59 -04:00
|
|
|
m_meta->setProtectTitle(readBool());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "ProtectUserName") {
|
2010-08-12 15:38:59 -04:00
|
|
|
m_meta->setProtectUsername(readBool());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "ProtectPassword") {
|
2010-08-12 15:38:59 -04:00
|
|
|
m_meta->setProtectPassword(readBool());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "ProtectURL") {
|
2010-08-12 15:38:59 -04:00
|
|
|
m_meta->setProtectUrl(readBool());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "ProtectNotes") {
|
2010-08-12 15:38:59 -04:00
|
|
|
m_meta->setProtectNotes(readBool());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "AutoEnableVisualHiding") {
|
2010-08-12 15:38:59 -04:00
|
|
|
m_meta->setAutoEnableVisualHiding(readBool());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else {
|
2010-08-13 12:08:06 -04:00
|
|
|
skipCurrentElement();
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-31 08:39:35 -04:00
|
|
|
void KeePass2XmlReader::parseCustomIcons()
|
2010-08-07 09:10:44 -04:00
|
|
|
{
|
|
|
|
Q_ASSERT(m_xml.isStartElement() && m_xml.name() == "CustomIcons");
|
|
|
|
|
2010-08-12 15:38:59 -04:00
|
|
|
while (!m_xml.error() && m_xml.readNextStartElement()) {
|
2010-08-07 09:10:44 -04:00
|
|
|
if (m_xml.name() == "Icon") {
|
|
|
|
parseIcon();
|
|
|
|
}
|
|
|
|
else {
|
2010-08-13 12:08:06 -04:00
|
|
|
skipCurrentElement();
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-31 08:39:35 -04:00
|
|
|
void KeePass2XmlReader::parseIcon()
|
2010-08-07 09:10:44 -04:00
|
|
|
{
|
|
|
|
Q_ASSERT(m_xml.isStartElement() && m_xml.name() == "Icon");
|
|
|
|
|
2010-08-12 15:38:59 -04:00
|
|
|
Uuid uuid;
|
|
|
|
while (!m_xml.error() && m_xml.readNextStartElement()) {
|
2010-08-07 09:10:44 -04:00
|
|
|
if (m_xml.name() == "UUID") {
|
2010-08-12 15:38:59 -04:00
|
|
|
uuid = readUuid();
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "Data") {
|
2010-09-19 15:22:24 -04:00
|
|
|
QPixmap pixmap;
|
|
|
|
pixmap.loadFromData(readBinary());
|
|
|
|
m_meta->addCustomIcon(uuid, QIcon(pixmap));
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else {
|
2010-08-13 12:08:06 -04:00
|
|
|
skipCurrentElement();
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-31 08:39:35 -04:00
|
|
|
void KeePass2XmlReader::parseCustomData()
|
2010-08-07 09:10:44 -04:00
|
|
|
{
|
|
|
|
Q_ASSERT(m_xml.isStartElement() && m_xml.name() == "CustomData");
|
|
|
|
|
2010-08-13 12:08:06 -04:00
|
|
|
while (!m_xml.error() && m_xml.readNextStartElement()) {
|
2010-08-25 18:31:07 -04:00
|
|
|
if (m_xml.name() == "Item") {
|
|
|
|
parseCustomDataItem();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
skipCurrentElement();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-31 08:39:35 -04:00
|
|
|
void KeePass2XmlReader::parseCustomDataItem()
|
2010-08-25 18:31:07 -04:00
|
|
|
{
|
|
|
|
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();
|
|
|
|
}
|
2010-08-13 12:08:06 -04:00
|
|
|
}
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
|
2010-08-31 08:39:35 -04:00
|
|
|
void KeePass2XmlReader::parseRoot()
|
2010-08-07 09:10:44 -04:00
|
|
|
{
|
|
|
|
Q_ASSERT(m_xml.isStartElement() && m_xml.name() == "Root");
|
|
|
|
|
2010-08-12 15:38:59 -04:00
|
|
|
while (!m_xml.error() && m_xml.readNextStartElement()) {
|
2010-08-07 09:10:44 -04:00
|
|
|
if (m_xml.name() == "Group") {
|
2010-08-12 15:38:59 -04:00
|
|
|
Group* rootGroup = parseGroup();
|
2010-08-12 15:43:57 -04:00
|
|
|
if (rootGroup) {
|
2010-08-12 15:38:59 -04:00
|
|
|
rootGroup->setParent(m_db);
|
|
|
|
}
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "DeletedObjects") {
|
2010-08-25 07:52:59 -04:00
|
|
|
parseDeletedObjects();
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else {
|
2010-08-13 12:08:06 -04:00
|
|
|
skipCurrentElement();
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-31 08:39:35 -04:00
|
|
|
Group* KeePass2XmlReader::parseGroup()
|
2010-08-07 09:10:44 -04:00
|
|
|
{
|
|
|
|
Q_ASSERT(m_xml.isStartElement() && m_xml.name() == "Group");
|
|
|
|
|
2010-08-12 15:38:59 -04:00
|
|
|
Group* group = 0;
|
|
|
|
while (!m_xml.error() && m_xml.readNextStartElement()) {
|
2010-08-07 09:10:44 -04:00
|
|
|
if (m_xml.name() == "UUID") {
|
2010-08-12 15:38:59 -04:00
|
|
|
Uuid uuid = readUuid();
|
|
|
|
if (uuid.isNull()) {
|
|
|
|
raiseError();
|
|
|
|
}
|
|
|
|
else {
|
2010-08-13 12:08:06 -04:00
|
|
|
group = getGroup(uuid);
|
2010-08-12 15:38:59 -04:00
|
|
|
}
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "Name") {
|
2010-08-12 15:38:59 -04:00
|
|
|
group->setName(readString());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "Notes") {
|
2010-08-12 15:38:59 -04:00
|
|
|
group->setNotes(readString());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "IconID") {
|
2010-08-12 15:38:59 -04:00
|
|
|
int iconId = readNumber();
|
2010-08-14 06:24:35 -04:00
|
|
|
if (iconId < 0) {
|
|
|
|
raiseError();
|
|
|
|
}
|
2010-08-25 13:26:01 -04:00
|
|
|
else {
|
2010-09-22 18:21:36 -04:00
|
|
|
if (iconId >= DatabaseIcons::iconCount()) {
|
|
|
|
qWarning("KeePass2XmlReader::parseGroup: icon id \"%d\" not supported", iconId);
|
|
|
|
}
|
2010-08-12 15:38:59 -04:00
|
|
|
group->setIcon(iconId);
|
2010-08-14 06:24:35 -04:00
|
|
|
}
|
2010-08-12 15:38:59 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "CustomIconUUID") {
|
|
|
|
Uuid uuid = readUuid();
|
|
|
|
if (!uuid.isNull()) {
|
|
|
|
group->setIcon(uuid);
|
|
|
|
}
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "Times") {
|
2010-08-12 15:38:59 -04:00
|
|
|
group->setTimeInfo(parseTimes());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "IsExpanded") {
|
2010-08-12 15:38:59 -04:00
|
|
|
group->setExpanded(readBool());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "DefaultAutoTypeSequence") {
|
2010-08-12 15:38:59 -04:00
|
|
|
group->setDefaultAutoTypeSequence(readString());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "EnableAutoType") {
|
2010-08-19 08:03:54 -04:00
|
|
|
QString str = readString();
|
|
|
|
|
|
|
|
if (str.compare("null", Qt::CaseInsensitive) == 0) {
|
|
|
|
group->setAutoTypeEnabled(-1);
|
|
|
|
}
|
|
|
|
else if (str.compare("true", Qt::CaseInsensitive) == 0) {
|
|
|
|
group->setAutoTypeEnabled(1);
|
|
|
|
}
|
|
|
|
else if (str.compare("false", Qt::CaseInsensitive) == 0) {
|
|
|
|
group->setAutoTypeEnabled(0);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
raiseError();
|
|
|
|
}
|
|
|
|
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "EnableSearching") {
|
2010-08-19 08:03:54 -04:00
|
|
|
QString str = readString();
|
|
|
|
|
|
|
|
if (str.compare("null", Qt::CaseInsensitive) == 0) {
|
|
|
|
group->setSearchingEnabled(-1);
|
|
|
|
}
|
|
|
|
else if (str.compare("true", Qt::CaseInsensitive) == 0) {
|
|
|
|
group->setSearchingEnabled(1);
|
|
|
|
}
|
|
|
|
else if (str.compare("false", Qt::CaseInsensitive) == 0) {
|
|
|
|
group->setSearchingEnabled(0);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
raiseError();
|
|
|
|
}
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "LastTopVisibleEntry") {
|
2010-08-13 12:08:06 -04:00
|
|
|
group->setLastTopVisibleEntry(getEntry(readUuid()));
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "Group") {
|
2010-08-12 15:38:59 -04:00
|
|
|
Group* newGroup = parseGroup();
|
|
|
|
if (newGroup) {
|
|
|
|
newGroup->setParent(group);
|
|
|
|
}
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "Entry") {
|
2010-08-25 07:52:59 -04:00
|
|
|
Entry* newEntry = parseEntry(false);
|
2010-08-12 15:38:59 -04:00
|
|
|
if (newEntry) {
|
|
|
|
newEntry->setGroup(group);
|
|
|
|
}
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else {
|
2010-08-13 12:08:06 -04:00
|
|
|
skipCurrentElement();
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
}
|
2010-08-12 15:38:59 -04:00
|
|
|
|
|
|
|
return group;
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
|
2010-08-31 08:39:35 -04:00
|
|
|
void KeePass2XmlReader::parseDeletedObjects()
|
2010-08-25 07:52:59 -04:00
|
|
|
{
|
|
|
|
Q_ASSERT(m_xml.isStartElement() && m_xml.name() == "DeletedObjects");
|
|
|
|
|
|
|
|
while (!m_xml.error() && m_xml.readNextStartElement()) {
|
|
|
|
if (m_xml.name() == "DeletedObject") {
|
|
|
|
parseDeletedObject();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
skipCurrentElement();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-31 08:39:35 -04:00
|
|
|
void KeePass2XmlReader::parseDeletedObject()
|
2010-08-25 07:52:59 -04:00
|
|
|
{
|
|
|
|
Q_ASSERT(m_xml.isStartElement() && m_xml.name() == "DeletedObject");
|
|
|
|
|
|
|
|
DeletedObject delObj;
|
|
|
|
|
|
|
|
while (!m_xml.error() && m_xml.readNextStartElement()) {
|
|
|
|
if (m_xml.name() == "UUID") {
|
|
|
|
Uuid uuid = readUuid();
|
|
|
|
if (uuid.isNull()) {
|
|
|
|
raiseError();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
delObj.uuid = uuid;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (m_xml.name() == "DeletionTime") {
|
|
|
|
delObj.deletionTime = readDateTime();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
skipCurrentElement();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
m_db->addDeletedObject(delObj);
|
|
|
|
}
|
|
|
|
|
2010-08-31 08:39:35 -04:00
|
|
|
Entry* KeePass2XmlReader::parseEntry(bool history)
|
2010-08-07 09:10:44 -04:00
|
|
|
{
|
|
|
|
Q_ASSERT(m_xml.isStartElement() && m_xml.name() == "Entry");
|
|
|
|
|
2010-08-12 15:38:59 -04:00
|
|
|
Entry* entry = 0;
|
|
|
|
while (!m_xml.error() && m_xml.readNextStartElement()) {
|
2010-08-07 09:10:44 -04:00
|
|
|
if (m_xml.name() == "UUID") {
|
2010-08-12 15:38:59 -04:00
|
|
|
Uuid uuid = readUuid();
|
|
|
|
if (uuid.isNull()) {
|
|
|
|
raiseError();
|
|
|
|
}
|
|
|
|
else {
|
2010-08-25 07:52:59 -04:00
|
|
|
if (history) {
|
|
|
|
entry = new Entry();
|
2010-08-25 15:13:50 -04:00
|
|
|
entry->setUuid(uuid);
|
2010-08-25 07:52:59 -04:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
entry = getEntry(uuid);
|
|
|
|
}
|
|
|
|
}
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "IconID") {
|
2010-08-12 15:38:59 -04:00
|
|
|
int iconId = readNumber();
|
2010-08-14 06:24:35 -04:00
|
|
|
if (iconId < 0) {
|
|
|
|
raiseError();
|
|
|
|
}
|
2010-08-25 13:26:01 -04:00
|
|
|
else {
|
2010-08-12 15:38:59 -04:00
|
|
|
entry->setIcon(iconId);
|
2010-08-14 06:24:35 -04:00
|
|
|
}
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "CustomIconUUID") {
|
2010-08-12 15:38:59 -04:00
|
|
|
Uuid uuid = readUuid();
|
2010-09-19 15:22:24 -04:00
|
|
|
if (!uuid.isNull()) {
|
2010-08-12 15:38:59 -04:00
|
|
|
entry->setIcon(uuid);
|
2010-09-19 15:22:24 -04:00
|
|
|
}
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "ForegroundColor") {
|
2010-08-12 15:38:59 -04:00
|
|
|
entry->setForegroundColor(readColor());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "BackgroundColor") {
|
2010-08-12 15:38:59 -04:00
|
|
|
entry->setBackgroundColor(readColor());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "OverrideURL") {
|
2010-08-12 15:38:59 -04:00
|
|
|
entry->setOverrideUrl(readString());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
2010-09-25 06:41:00 -04:00
|
|
|
else if (m_xml.name() == "Tags") {
|
|
|
|
entry->setTags(readString());
|
|
|
|
}
|
2010-08-07 09:10:44 -04:00
|
|
|
else if (m_xml.name() == "Times") {
|
2010-08-12 15:38:59 -04:00
|
|
|
entry->setTimeInfo(parseTimes());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "String") {
|
2010-08-12 15:38:59 -04:00
|
|
|
parseEntryString(entry);
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "Binary") {
|
2010-08-12 15:38:59 -04:00
|
|
|
parseEntryBinary(entry);
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "AutoType") {
|
2010-08-12 15:38:59 -04:00
|
|
|
parseAutoType(entry);
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "History") {
|
2010-08-25 07:52:59 -04:00
|
|
|
if (history) {
|
|
|
|
raiseError();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
parseEntryHistory(entry);
|
|
|
|
}
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else {
|
2010-08-13 12:08:06 -04:00
|
|
|
skipCurrentElement();
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
}
|
2010-08-13 12:08:06 -04:00
|
|
|
|
|
|
|
return entry;
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
|
2010-08-31 08:39:35 -04:00
|
|
|
void KeePass2XmlReader::parseEntryString(Entry *entry)
|
2010-08-07 09:10:44 -04:00
|
|
|
{
|
|
|
|
Q_ASSERT(m_xml.isStartElement() && m_xml.name() == "String");
|
|
|
|
|
2010-08-12 15:38:59 -04:00
|
|
|
QString key;
|
|
|
|
while (!m_xml.error() && m_xml.readNextStartElement()) {
|
2010-08-07 09:10:44 -04:00
|
|
|
if (m_xml.name() == "Key") {
|
2010-08-12 15:38:59 -04:00
|
|
|
key = readString();
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "Value") {
|
2010-08-12 15:38:59 -04:00
|
|
|
entry->addAttribute(key, readString());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else {
|
2010-08-13 12:08:06 -04:00
|
|
|
skipCurrentElement();
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-31 08:39:35 -04:00
|
|
|
void KeePass2XmlReader::parseEntryBinary(Entry *entry)
|
2010-08-07 09:10:44 -04:00
|
|
|
{
|
|
|
|
Q_ASSERT(m_xml.isStartElement() && m_xml.name() == "Binary");
|
|
|
|
|
2010-08-12 15:38:59 -04:00
|
|
|
QString key;
|
|
|
|
while (!m_xml.error() && m_xml.readNextStartElement()) {
|
2010-08-07 09:10:44 -04:00
|
|
|
if (m_xml.name() == "Key") {
|
2010-08-12 15:38:59 -04:00
|
|
|
key = readString();
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "Value") {
|
2010-08-12 15:38:59 -04:00
|
|
|
entry->addAttachment(key, readBinary());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else {
|
2010-08-13 12:08:06 -04:00
|
|
|
skipCurrentElement();
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-31 08:39:35 -04:00
|
|
|
void KeePass2XmlReader::parseAutoType(Entry* entry)
|
2010-08-07 09:10:44 -04:00
|
|
|
{
|
|
|
|
Q_ASSERT(m_xml.isStartElement() && m_xml.name() == "AutoType");
|
|
|
|
|
2010-08-12 15:38:59 -04:00
|
|
|
while (!m_xml.error() && m_xml.readNextStartElement()) {
|
2010-08-07 09:10:44 -04:00
|
|
|
if (m_xml.name() == "Enabled") {
|
2010-08-12 15:38:59 -04:00
|
|
|
entry->setAutoTypeEnabled(readBool());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "DataTransferObfuscation") {
|
2010-08-12 15:38:59 -04:00
|
|
|
entry->setAutoTypeObfuscation(readNumber());
|
|
|
|
}
|
|
|
|
else if (m_xml.name() == "DefaultSequence") {
|
|
|
|
entry->setDefaultAutoTypeSequence(readString());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "Association") {
|
2010-08-12 15:38:59 -04:00
|
|
|
parseAutoTypeAssoc(entry);
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else {
|
2010-08-13 12:08:06 -04:00
|
|
|
skipCurrentElement();
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-31 08:39:35 -04:00
|
|
|
void KeePass2XmlReader::parseAutoTypeAssoc(Entry *entry)
|
2010-08-07 09:10:44 -04:00
|
|
|
{
|
|
|
|
Q_ASSERT(m_xml.isStartElement() && m_xml.name() == "Association");
|
|
|
|
|
2010-08-12 15:38:59 -04:00
|
|
|
AutoTypeAssociation assoc;
|
|
|
|
while (!m_xml.error() && m_xml.readNextStartElement()) {
|
2010-08-07 09:10:44 -04:00
|
|
|
if (m_xml.name() == "Window") {
|
2010-08-12 15:38:59 -04:00
|
|
|
assoc.window = readString();
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "KeystrokeSequence") {
|
2010-08-12 15:38:59 -04:00
|
|
|
assoc.sequence = readString();
|
|
|
|
entry->addAutoTypeAssociation(assoc);
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else {
|
2010-08-13 12:08:06 -04:00
|
|
|
skipCurrentElement();
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-31 08:39:35 -04:00
|
|
|
void KeePass2XmlReader::parseEntryHistory(Entry* entry)
|
2010-08-07 09:10:44 -04:00
|
|
|
{
|
|
|
|
Q_ASSERT(m_xml.isStartElement() && m_xml.name() == "History");
|
|
|
|
|
2010-08-12 15:38:59 -04:00
|
|
|
while (!m_xml.error() && m_xml.readNextStartElement()) {
|
2010-08-07 09:10:44 -04:00
|
|
|
if (m_xml.name() == "Entry") {
|
2010-08-25 07:52:59 -04:00
|
|
|
Entry* historyItem = parseEntry(true);
|
|
|
|
entry->addHistoryItem(historyItem);
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else {
|
2010-08-13 12:08:06 -04:00
|
|
|
skipCurrentElement();
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-31 08:39:35 -04:00
|
|
|
TimeInfo KeePass2XmlReader::parseTimes()
|
2010-08-07 09:10:44 -04:00
|
|
|
{
|
|
|
|
Q_ASSERT(m_xml.isStartElement() && m_xml.name() == "Times");
|
|
|
|
|
2010-08-12 15:38:59 -04:00
|
|
|
TimeInfo timeInfo;
|
|
|
|
while (!m_xml.error() && m_xml.readNextStartElement()) {
|
2010-08-07 09:10:44 -04:00
|
|
|
if (m_xml.name() == "LastModificationTime") {
|
2010-08-12 15:38:59 -04:00
|
|
|
timeInfo.setLastModificationTime(readDateTime());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "CreationTime") {
|
2010-08-12 15:38:59 -04:00
|
|
|
timeInfo.setCreationTime(readDateTime());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "LastAccessTime") {
|
2010-08-12 15:38:59 -04:00
|
|
|
timeInfo.setLastAccessTime(readDateTime());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "ExpiryTime") {
|
2010-08-12 15:38:59 -04:00
|
|
|
timeInfo.setExpiryTime(readDateTime());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "Expires") {
|
2010-08-12 15:38:59 -04:00
|
|
|
timeInfo.setExpires(readBool());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "UsageCount") {
|
2010-08-12 15:38:59 -04:00
|
|
|
timeInfo.setUsageCount(readNumber());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else if (m_xml.name() == "LocationChanged") {
|
2010-08-12 15:38:59 -04:00
|
|
|
timeInfo.setLocationChanged(readDateTime());
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
else {
|
2010-08-13 12:08:06 -04:00
|
|
|
skipCurrentElement();
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
}
|
2010-08-12 15:38:59 -04:00
|
|
|
|
|
|
|
return timeInfo;
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
|
2010-08-31 08:39:35 -04:00
|
|
|
QString KeePass2XmlReader::readString()
|
2010-08-07 09:10:44 -04:00
|
|
|
{
|
|
|
|
return m_xml.readElementText();
|
|
|
|
}
|
|
|
|
|
2010-08-31 08:39:35 -04:00
|
|
|
bool KeePass2XmlReader::readBool()
|
2010-08-07 09:10:44 -04:00
|
|
|
{
|
|
|
|
QString str = readString();
|
|
|
|
|
2010-08-19 08:03:54 -04:00
|
|
|
if (str.compare("True", Qt::CaseInsensitive) == 0) {
|
2010-08-07 09:10:44 -04:00
|
|
|
return true;
|
|
|
|
}
|
2010-08-19 08:03:54 -04:00
|
|
|
else if (str.compare("False", Qt::CaseInsensitive) == 0) {
|
2010-08-07 09:10:44 -04:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
else {
|
2010-08-12 15:38:59 -04:00
|
|
|
raiseError();
|
2010-08-13 12:08:06 -04:00
|
|
|
return false;
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-31 08:39:35 -04:00
|
|
|
QDateTime KeePass2XmlReader::readDateTime()
|
2010-08-07 09:10:44 -04:00
|
|
|
{
|
|
|
|
QString str = readString();
|
|
|
|
QDateTime dt = QDateTime::fromString(str, Qt::ISODate);
|
|
|
|
|
|
|
|
if (!dt.isValid()) {
|
2010-08-12 15:38:59 -04:00
|
|
|
raiseError();
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
return dt;
|
|
|
|
}
|
|
|
|
|
2010-08-31 08:39:35 -04:00
|
|
|
QColor KeePass2XmlReader::readColor()
|
2010-08-07 09:10:44 -04:00
|
|
|
{
|
2010-08-12 15:38:59 -04:00
|
|
|
QString colorStr = readString();
|
2010-08-13 12:08:06 -04:00
|
|
|
|
|
|
|
if (colorStr.isEmpty()) {
|
|
|
|
return QColor();
|
|
|
|
}
|
|
|
|
|
2010-08-12 15:38:59 -04:00
|
|
|
if (colorStr.length() != 7 || colorStr[0] != '#') {
|
|
|
|
raiseError();
|
|
|
|
return QColor();
|
|
|
|
}
|
|
|
|
|
|
|
|
QColor color;
|
|
|
|
for (int i=0; i<= 2; i++) {
|
|
|
|
QString rgbPartStr = colorStr.mid(1 + 2*i, 2);
|
|
|
|
bool ok;
|
2010-08-13 12:08:06 -04:00
|
|
|
int rgbPart = rgbPartStr.toInt(&ok, 16);
|
2010-08-12 15:38:59 -04:00
|
|
|
if (!ok || rgbPart > 255) {
|
|
|
|
raiseError();
|
|
|
|
return QColor();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (i == 0) {
|
|
|
|
color.setRed(rgbPart);
|
|
|
|
}
|
|
|
|
else if (i == 1) {
|
|
|
|
color.setGreen(rgbPart);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
color.setBlue(rgbPart);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return color;
|
|
|
|
}
|
|
|
|
|
2010-08-31 08:39:35 -04:00
|
|
|
int KeePass2XmlReader::readNumber()
|
2010-08-12 15:38:59 -04:00
|
|
|
{
|
|
|
|
bool ok;
|
|
|
|
int result = readString().toInt(&ok);
|
|
|
|
if (!ok) {
|
|
|
|
raiseError();
|
|
|
|
}
|
|
|
|
return result;
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
|
2010-08-31 08:39:35 -04:00
|
|
|
Uuid KeePass2XmlReader::readUuid()
|
2010-08-07 09:10:44 -04:00
|
|
|
{
|
2010-08-12 15:38:59 -04:00
|
|
|
QByteArray uuidBin = readBinary();
|
2010-09-05 05:46:36 -04:00
|
|
|
if (uuidBin.length() != Uuid::LENGTH) {
|
2010-08-12 15:38:59 -04:00
|
|
|
raiseError();
|
|
|
|
return Uuid();
|
|
|
|
}
|
|
|
|
else {
|
2010-08-13 12:08:06 -04:00
|
|
|
return Uuid(uuidBin);
|
2010-08-12 15:38:59 -04:00
|
|
|
}
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
|
|
|
|
2010-08-31 08:39:35 -04:00
|
|
|
QByteArray KeePass2XmlReader::readBinary()
|
2010-08-07 09:10:44 -04:00
|
|
|
{
|
2010-08-12 15:38:59 -04:00
|
|
|
return QByteArray::fromBase64(readString().toAscii());
|
|
|
|
}
|
|
|
|
|
2010-08-31 08:39:35 -04:00
|
|
|
Group* KeePass2XmlReader::getGroup(const Uuid& uuid)
|
2010-08-12 15:38:59 -04:00
|
|
|
{
|
2010-08-13 12:08:06 -04:00
|
|
|
if (uuid.isNull()) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-08-12 15:38:59 -04:00
|
|
|
Q_FOREACH (Group* group, m_groups) {
|
|
|
|
if (group->uuid() == uuid) {
|
|
|
|
return group;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Group* group = new Group();
|
|
|
|
group->setUuid(uuid);
|
|
|
|
group->setParent(m_tmpParent);
|
|
|
|
m_groups << group;
|
|
|
|
return group;
|
|
|
|
}
|
|
|
|
|
2010-08-31 08:39:35 -04:00
|
|
|
Entry* KeePass2XmlReader::getEntry(const Uuid& uuid)
|
2010-08-12 15:38:59 -04:00
|
|
|
{
|
2010-08-13 12:08:06 -04:00
|
|
|
if (uuid.isNull()) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-08-12 15:38:59 -04:00
|
|
|
Q_FOREACH (Entry* entry, m_entries) {
|
|
|
|
if (entry->uuid() == uuid) {
|
|
|
|
return entry;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Entry* entry = new Entry();
|
|
|
|
entry->setUuid(uuid);
|
|
|
|
entry->setGroup(m_tmpParent);
|
|
|
|
m_entries << entry;
|
|
|
|
return entry;
|
|
|
|
}
|
|
|
|
|
2010-08-31 08:39:35 -04:00
|
|
|
void KeePass2XmlReader::raiseError()
|
2010-08-12 15:38:59 -04:00
|
|
|
{
|
|
|
|
m_xml.raiseError(tr("Invalid database file"));
|
2010-08-07 09:10:44 -04:00
|
|
|
}
|
2010-08-13 12:08:06 -04:00
|
|
|
|
2010-08-31 08:39:35 -04:00
|
|
|
void KeePass2XmlReader::skipCurrentElement()
|
2010-08-13 12:08:06 -04:00
|
|
|
{
|
2010-09-22 18:21:36 -04:00
|
|
|
qWarning("KeePass2XmlReader::skipCurrentElement: skip element \"%s\"", qPrintable(m_xml.name().toString()));
|
2010-08-13 12:08:06 -04:00
|
|
|
m_xml.skipCurrentElement();
|
|
|
|
}
|