2018-10-01 10:26:24 -04:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2018 KeePassXC Team <team@keepassxc.org>
|
|
|
|
*
|
|
|
|
* 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 "TestSharing.h"
|
|
|
|
#include "TestGlobal.h"
|
|
|
|
|
|
|
|
#include <QBuffer>
|
|
|
|
#include <QSignalSpy>
|
|
|
|
#include <QTemporaryFile>
|
|
|
|
#include <QXmlStreamReader>
|
|
|
|
#include <QXmlStreamWriter>
|
|
|
|
|
|
|
|
#include "config-keepassx-tests.h"
|
|
|
|
#include "core/Metadata.h"
|
|
|
|
#include "crypto/Crypto.h"
|
|
|
|
#include "crypto/Random.h"
|
|
|
|
#include "format/KeePass2Writer.h"
|
|
|
|
#include "keeshare/KeeShareSettings.h"
|
|
|
|
#include "keys/PasswordKey.h"
|
|
|
|
|
2021-04-04 08:56:00 -04:00
|
|
|
#include <botan/pkcs8.h>
|
|
|
|
#include <botan/rsa.h>
|
|
|
|
#include <botan/x509_key.h>
|
|
|
|
|
2018-10-01 10:26:24 -04:00
|
|
|
#include <format/KeePass2Reader.h>
|
|
|
|
|
|
|
|
QTEST_GUILESS_MAIN(TestSharing)
|
|
|
|
|
|
|
|
Q_DECLARE_METATYPE(KeeShareSettings::Type)
|
|
|
|
Q_DECLARE_METATYPE(KeeShareSettings::Key)
|
|
|
|
Q_DECLARE_METATYPE(KeeShareSettings::Certificate)
|
2019-01-03 12:22:41 -05:00
|
|
|
Q_DECLARE_METATYPE(KeeShareSettings::Trust)
|
2019-01-03 11:50:36 -05:00
|
|
|
Q_DECLARE_METATYPE(KeeShareSettings::ScopedCertificate)
|
|
|
|
Q_DECLARE_METATYPE(QList<KeeShareSettings::ScopedCertificate>)
|
2018-10-01 10:26:24 -04:00
|
|
|
|
|
|
|
void TestSharing::initTestCase()
|
|
|
|
{
|
|
|
|
QVERIFY(Crypto::init());
|
|
|
|
}
|
|
|
|
|
|
|
|
void TestSharing::testNullObjects()
|
|
|
|
{
|
|
|
|
const QString empty;
|
|
|
|
QXmlStreamReader reader(empty);
|
|
|
|
|
2019-01-04 06:38:34 -05:00
|
|
|
const auto nullKey = KeeShareSettings::Key();
|
2018-10-01 10:26:24 -04:00
|
|
|
QVERIFY(nullKey.isNull());
|
2019-01-04 06:28:36 -05:00
|
|
|
const auto xmlKey = KeeShareSettings::Key::deserialize(reader);
|
2018-10-01 10:26:24 -04:00
|
|
|
QVERIFY(xmlKey.isNull());
|
|
|
|
|
2019-01-04 06:38:34 -05:00
|
|
|
const auto certificate = KeeShareSettings::Certificate();
|
2018-10-01 10:26:24 -04:00
|
|
|
QVERIFY(certificate.isNull());
|
2019-01-04 06:28:36 -05:00
|
|
|
const auto xmlCertificate = KeeShareSettings::Certificate::deserialize(reader);
|
2018-10-01 10:26:24 -04:00
|
|
|
QVERIFY(xmlCertificate.isNull());
|
|
|
|
|
2019-01-04 06:38:34 -05:00
|
|
|
const auto own = KeeShareSettings::Own();
|
2018-10-01 10:26:24 -04:00
|
|
|
QVERIFY(own.isNull());
|
2019-01-04 06:28:36 -05:00
|
|
|
const auto xmlOwn = KeeShareSettings::Own::deserialize(empty);
|
2018-10-01 10:26:24 -04:00
|
|
|
QVERIFY(xmlOwn.isNull());
|
|
|
|
|
2019-01-04 06:38:34 -05:00
|
|
|
const auto active = KeeShareSettings::Active();
|
2018-10-01 10:26:24 -04:00
|
|
|
QVERIFY(active.isNull());
|
2019-01-04 06:28:36 -05:00
|
|
|
const auto xmlActive = KeeShareSettings::Active::deserialize(empty);
|
2018-10-01 10:26:24 -04:00
|
|
|
QVERIFY(xmlActive.isNull());
|
|
|
|
|
2019-01-04 06:38:34 -05:00
|
|
|
const auto foreign = KeeShareSettings::Foreign();
|
2019-01-03 11:50:36 -05:00
|
|
|
QVERIFY(foreign.certificates.isEmpty());
|
2019-01-04 06:28:36 -05:00
|
|
|
const auto xmlForeign = KeeShareSettings::Foreign::deserialize(empty);
|
2019-01-03 11:50:36 -05:00
|
|
|
QVERIFY(xmlForeign.certificates.isEmpty());
|
2018-10-01 10:26:24 -04:00
|
|
|
|
2019-01-04 06:38:34 -05:00
|
|
|
const auto reference = KeeShareSettings::Reference();
|
2018-10-01 10:26:24 -04:00
|
|
|
QVERIFY(reference.isNull());
|
2019-01-04 06:28:36 -05:00
|
|
|
const auto xmlReference = KeeShareSettings::Reference::deserialize(empty);
|
2018-10-01 10:26:24 -04:00
|
|
|
QVERIFY(xmlReference.isNull());
|
|
|
|
}
|
|
|
|
|
|
|
|
void TestSharing::testCertificateSerialization()
|
|
|
|
{
|
2019-01-03 12:22:41 -05:00
|
|
|
QFETCH(KeeShareSettings::Trust, trusted);
|
2021-04-04 08:56:00 -04:00
|
|
|
auto key = stubkey();
|
2019-01-03 11:50:36 -05:00
|
|
|
KeeShareSettings::ScopedCertificate original;
|
|
|
|
original.path = "/path";
|
2021-04-04 08:56:00 -04:00
|
|
|
original.certificate = KeeShareSettings::Certificate{key, "Some <!> &#_\"\" weird string"};
|
2019-01-03 12:22:41 -05:00
|
|
|
original.trust = trusted;
|
2018-10-01 10:26:24 -04:00
|
|
|
|
|
|
|
QString buffer;
|
|
|
|
QXmlStreamWriter writer(&buffer);
|
|
|
|
writer.writeStartDocument();
|
|
|
|
writer.writeStartElement("Certificate");
|
2019-01-03 11:50:36 -05:00
|
|
|
KeeShareSettings::ScopedCertificate::serialize(writer, original);
|
2018-10-01 10:26:24 -04:00
|
|
|
writer.writeEndElement();
|
|
|
|
writer.writeEndDocument();
|
2021-04-04 08:56:00 -04:00
|
|
|
|
2018-10-01 10:26:24 -04:00
|
|
|
QXmlStreamReader reader(buffer);
|
|
|
|
reader.readNextStartElement();
|
|
|
|
QVERIFY(reader.name() == "Certificate");
|
2019-01-03 11:50:36 -05:00
|
|
|
KeeShareSettings::ScopedCertificate restored = KeeShareSettings::ScopedCertificate::deserialize(reader);
|
2018-10-01 10:26:24 -04:00
|
|
|
|
2021-04-04 08:56:00 -04:00
|
|
|
QCOMPARE(restored.certificate.key->private_key_bits(), original.certificate.key->private_key_bits());
|
2019-01-03 11:50:36 -05:00
|
|
|
QCOMPARE(restored.certificate.signer, original.certificate.signer);
|
2019-01-03 12:22:41 -05:00
|
|
|
QCOMPARE(restored.trust, original.trust);
|
2019-01-03 11:50:36 -05:00
|
|
|
QCOMPARE(restored.path, original.path);
|
2018-10-01 10:26:24 -04:00
|
|
|
|
2021-04-04 08:56:00 -04:00
|
|
|
QCOMPARE(restored.certificate.key->public_key_bits(), key->public_key_bits());
|
2018-10-01 10:26:24 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void TestSharing::testCertificateSerialization_data()
|
|
|
|
{
|
2019-01-03 12:22:41 -05:00
|
|
|
QTest::addColumn<KeeShareSettings::Trust>("trusted");
|
|
|
|
QTest::newRow("Ask") << KeeShareSettings::Trust::Ask;
|
|
|
|
QTest::newRow("Trusted") << KeeShareSettings::Trust::Trusted;
|
|
|
|
QTest::newRow("Untrusted") << KeeShareSettings::Trust::Untrusted;
|
2018-10-01 10:26:24 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void TestSharing::testKeySerialization()
|
|
|
|
{
|
2021-04-04 08:56:00 -04:00
|
|
|
auto key = stubkey();
|
2018-10-01 10:26:24 -04:00
|
|
|
KeeShareSettings::Key original;
|
2021-04-04 08:56:00 -04:00
|
|
|
original.key = key;
|
2018-10-01 10:26:24 -04:00
|
|
|
|
|
|
|
QString buffer;
|
|
|
|
QXmlStreamWriter writer(&buffer);
|
|
|
|
writer.writeStartDocument();
|
|
|
|
writer.writeStartElement("Key");
|
|
|
|
KeeShareSettings::Key::serialize(writer, original);
|
|
|
|
writer.writeEndElement();
|
|
|
|
writer.writeEndDocument();
|
|
|
|
QXmlStreamReader reader(buffer);
|
|
|
|
reader.readNextStartElement();
|
|
|
|
QVERIFY(reader.name() == "Key");
|
|
|
|
KeeShareSettings::Key restored = KeeShareSettings::Key::deserialize(reader);
|
|
|
|
|
2021-04-04 08:56:00 -04:00
|
|
|
QCOMPARE(restored.key->private_key_bits(), original.key->private_key_bits());
|
|
|
|
QCOMPARE(restored.key->algo_name(), original.key->algo_name());
|
2018-10-01 10:26:24 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void TestSharing::testReferenceSerialization()
|
|
|
|
{
|
|
|
|
QFETCH(QString, password);
|
|
|
|
QFETCH(QString, path);
|
|
|
|
QFETCH(QUuid, uuid);
|
|
|
|
QFETCH(int, type);
|
|
|
|
KeeShareSettings::Reference original;
|
|
|
|
original.password = password;
|
|
|
|
original.path = path;
|
|
|
|
original.uuid = uuid;
|
|
|
|
original.type = static_cast<KeeShareSettings::Type>(type);
|
|
|
|
|
|
|
|
const QString serialized = KeeShareSettings::Reference::serialize(original);
|
|
|
|
const KeeShareSettings::Reference restored = KeeShareSettings::Reference::deserialize(serialized);
|
|
|
|
|
|
|
|
QCOMPARE(restored.password, original.password);
|
|
|
|
QCOMPARE(restored.path, original.path);
|
|
|
|
QCOMPARE(restored.uuid, original.uuid);
|
|
|
|
QCOMPARE(int(restored.type), int(original.type));
|
|
|
|
}
|
|
|
|
|
|
|
|
void TestSharing::testReferenceSerialization_data()
|
|
|
|
{
|
|
|
|
QTest::addColumn<QString>("password");
|
|
|
|
QTest::addColumn<QString>("path");
|
|
|
|
QTest::addColumn<QUuid>("uuid");
|
|
|
|
QTest::addColumn<int>("type");
|
2019-01-20 09:50:20 -05:00
|
|
|
QTest::newRow("1") << "Password"
|
|
|
|
<< "/some/path" << QUuid::createUuid() << int(KeeShareSettings::Inactive);
|
|
|
|
QTest::newRow("2") << ""
|
|
|
|
<< "" << QUuid() << int(KeeShareSettings::SynchronizeWith);
|
|
|
|
QTest::newRow("3") << ""
|
|
|
|
<< "/some/path" << QUuid() << int(KeeShareSettings::ExportTo);
|
2018-10-01 10:26:24 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void TestSharing::testSettingsSerialization()
|
|
|
|
{
|
|
|
|
QFETCH(bool, importing);
|
|
|
|
QFETCH(bool, exporting);
|
|
|
|
QFETCH(KeeShareSettings::Certificate, ownCertificate);
|
|
|
|
QFETCH(KeeShareSettings::Key, ownKey);
|
2019-01-03 11:50:36 -05:00
|
|
|
QFETCH(QList<KeeShareSettings::ScopedCertificate>, foreignCertificates);
|
2018-10-01 10:26:24 -04:00
|
|
|
|
|
|
|
KeeShareSettings::Own originalOwn;
|
|
|
|
KeeShareSettings::Foreign originalForeign;
|
|
|
|
KeeShareSettings::Active originalActive;
|
|
|
|
originalActive.in = importing;
|
|
|
|
originalActive.out = exporting;
|
|
|
|
originalOwn.certificate = ownCertificate;
|
|
|
|
originalOwn.key = ownKey;
|
|
|
|
originalForeign.certificates = foreignCertificates;
|
|
|
|
|
|
|
|
const QString serializedActive = KeeShareSettings::Active::serialize(originalActive);
|
|
|
|
KeeShareSettings::Active restoredActive = KeeShareSettings::Active::deserialize(serializedActive);
|
|
|
|
|
|
|
|
const QString serializedOwn = KeeShareSettings::Own::serialize(originalOwn);
|
|
|
|
KeeShareSettings::Own restoredOwn = KeeShareSettings::Own::deserialize(serializedOwn);
|
|
|
|
|
|
|
|
const QString serializedForeign = KeeShareSettings::Foreign::serialize(originalForeign);
|
|
|
|
KeeShareSettings::Foreign restoredForeign = KeeShareSettings::Foreign::deserialize(serializedForeign);
|
|
|
|
|
|
|
|
QCOMPARE(restoredActive.in, importing);
|
|
|
|
QCOMPARE(restoredActive.out, exporting);
|
2021-04-04 08:56:00 -04:00
|
|
|
if (ownCertificate.key) {
|
|
|
|
QCOMPARE(restoredOwn.certificate, ownCertificate);
|
|
|
|
}
|
|
|
|
if (ownKey.key) {
|
|
|
|
QCOMPARE(restoredOwn.key, ownKey);
|
|
|
|
}
|
2018-10-01 10:26:24 -04:00
|
|
|
QCOMPARE(restoredForeign.certificates.count(), foreignCertificates.count());
|
|
|
|
for (int i = 0; i < foreignCertificates.count(); ++i) {
|
2021-04-04 08:56:00 -04:00
|
|
|
QCOMPARE(restoredForeign.certificates[i].certificate, foreignCertificates[i].certificate);
|
2018-10-01 10:26:24 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void TestSharing::testSettingsSerialization_data()
|
|
|
|
{
|
2021-04-04 08:56:00 -04:00
|
|
|
auto sshKey0 = stubkey(0);
|
2019-01-03 11:50:36 -05:00
|
|
|
KeeShareSettings::ScopedCertificate certificate0;
|
|
|
|
certificate0.path = "/path/0";
|
2021-04-04 08:56:00 -04:00
|
|
|
certificate0.certificate = KeeShareSettings::Certificate{sshKey0, "Some <!> &#_\"\" weird string"};
|
2019-01-03 12:22:41 -05:00
|
|
|
certificate0.trust = KeeShareSettings::Trust::Trusted;
|
2018-10-01 10:26:24 -04:00
|
|
|
|
|
|
|
KeeShareSettings::Key key0;
|
2021-04-04 08:56:00 -04:00
|
|
|
key0.key = sshKey0;
|
2018-10-01 10:26:24 -04:00
|
|
|
|
2021-04-04 08:56:00 -04:00
|
|
|
auto sshKey1 = stubkey(1);
|
2019-01-03 11:50:36 -05:00
|
|
|
KeeShareSettings::ScopedCertificate certificate1;
|
|
|
|
certificate1.path = "/path/1";
|
2021-04-04 08:56:00 -04:00
|
|
|
certificate1.certificate = KeeShareSettings::Certificate{sshKey1, "Another "};
|
2019-01-03 12:22:41 -05:00
|
|
|
certificate1.trust = KeeShareSettings::Trust::Untrusted;
|
2018-10-01 10:26:24 -04:00
|
|
|
|
|
|
|
QTest::addColumn<bool>("importing");
|
|
|
|
QTest::addColumn<bool>("exporting");
|
|
|
|
QTest::addColumn<KeeShareSettings::Certificate>("ownCertificate");
|
|
|
|
QTest::addColumn<KeeShareSettings::Key>("ownKey");
|
2019-01-04 06:04:46 -05:00
|
|
|
QTest::addColumn<QList<KeeShareSettings::ScopedCertificate>>("foreignCertificates");
|
2019-01-20 09:50:20 -05:00
|
|
|
QTest::newRow("1") << false << false << KeeShareSettings::Certificate() << KeeShareSettings::Key()
|
|
|
|
<< QList<KeeShareSettings::ScopedCertificate>();
|
|
|
|
QTest::newRow("2") << true << false << KeeShareSettings::Certificate() << KeeShareSettings::Key()
|
|
|
|
<< QList<KeeShareSettings::ScopedCertificate>();
|
|
|
|
QTest::newRow("3") << true << true << KeeShareSettings::Certificate() << KeeShareSettings::Key()
|
|
|
|
<< QList<KeeShareSettings::ScopedCertificate>({certificate0, certificate1});
|
|
|
|
QTest::newRow("4") << false << true << certificate0.certificate << key0
|
|
|
|
<< QList<KeeShareSettings::ScopedCertificate>();
|
|
|
|
QTest::newRow("5") << false << false << certificate0.certificate << key0
|
|
|
|
<< QList<KeeShareSettings::ScopedCertificate>({certificate1});
|
2018-10-01 10:26:24 -04:00
|
|
|
}
|
|
|
|
|
2021-04-04 08:56:00 -04:00
|
|
|
const QSharedPointer<Botan::RSA_PrivateKey> TestSharing::stubkey(int index)
|
2018-10-01 10:26:24 -04:00
|
|
|
{
|
2021-04-04 08:56:00 -04:00
|
|
|
static QMap<int, QSharedPointer<Botan::RSA_PrivateKey>> keys;
|
2018-10-01 10:26:24 -04:00
|
|
|
if (!keys.contains(index)) {
|
2021-04-04 08:56:00 -04:00
|
|
|
keys.insert(index,
|
|
|
|
QSharedPointer<Botan::RSA_PrivateKey>(new Botan::RSA_PrivateKey(*randomGen()->getRng(), 2048)));
|
2018-10-01 10:26:24 -04:00
|
|
|
}
|
2021-04-04 08:56:00 -04:00
|
|
|
return keys[index];
|
2018-10-01 10:26:24 -04:00
|
|
|
}
|