2017-01-11 15:18:17 +00:00
|
|
|
/* t-various.cpp
|
|
|
|
|
|
|
|
This file is part of qgpgme, the Qt API binding for gpgme
|
Change copyright from Intevation to BSI
* lang/cpp/src/gpggencardkeyinteractor.cpp,
lang/cpp/src/gpggencardkeyinteractor.h,
lang/cpp/src/gpgmepp_export.h,
lang/cpp/src/swdbresult.cpp,
lang/cpp/src/swdbresult.h,
lang/cpp/src/tofuinfo.cpp,
lang/cpp/src/tofuinfo.h,
lang/qt/src/abstractimportjob.h,
lang/qt/src/adduseridjob.h,
lang/qt/src/changeexpiryjob.h,
lang/qt/src/changeownertrustjob.h,
lang/qt/src/changepasswdjob.h,
lang/qt/src/cryptoconfig.cpp,
lang/qt/src/cryptoconfig.h,
lang/qt/src/dataprovider.cpp,
lang/qt/src/dataprovider.h,
lang/qt/src/decryptjob.h,
lang/qt/src/decryptverifyjob.h,
lang/qt/src/deletejob.h,
lang/qt/src/dn.cpp,
lang/qt/src/dn.h,
lang/qt/src/downloadjob.h,
lang/qt/src/encryptjob.h,
lang/qt/src/exportjob.h,
lang/qt/src/hierarchicalkeylistjob.h,
lang/qt/src/importfromkeyserverjob.h,
lang/qt/src/importjob.h,
lang/qt/src/job.cpp,
lang/qt/src/job.h,
lang/qt/src/keyformailboxjob.h,
lang/qt/src/keygenerationjob.h,
lang/qt/src/keylistjob.h,
lang/qt/src/listallkeysjob.h,
lang/qt/src/multideletejob.h,
lang/qt/src/protocol.h,
lang/qt/src/protocol_p.h,
lang/qt/src/qgpgme_export.h,
lang/qt/src/qgpgmeadduseridjob.cpp,
lang/qt/src/qgpgmeadduseridjob.h,
lang/qt/src/qgpgmebackend.cpp,
lang/qt/src/qgpgmebackend.h,
lang/qt/src/qgpgmechangeexpiryjob.cpp,
lang/qt/src/qgpgmechangeexpiryjob.h,
lang/qt/src/qgpgmechangeownertrustjob.cpp,
lang/qt/src/qgpgmechangeownertrustjob.h,
lang/qt/src/qgpgmechangepasswdjob.cpp,
lang/qt/src/qgpgmechangepasswdjob.h,
lang/qt/src/qgpgmedecryptjob.cpp,
lang/qt/src/qgpgmedecryptjob.h,
lang/qt/src/qgpgmedecryptverifyjob.cpp,
lang/qt/src/qgpgmedecryptverifyjob.h,
lang/qt/src/qgpgmedeletejob.cpp,
lang/qt/src/qgpgmedeletejob.h,
lang/qt/src/qgpgmedownloadjob.cpp,
lang/qt/src/qgpgmedownloadjob.h,
lang/qt/src/qgpgmeencryptjob.cpp,
lang/qt/src/qgpgmeencryptjob.h,
lang/qt/src/qgpgmeexportjob.cpp,
lang/qt/src/qgpgmeexportjob.h,
lang/qt/src/qgpgmeimportfromkeyserverjob.cpp,
lang/qt/src/qgpgmeimportfromkeyserverjob.h,
lang/qt/src/qgpgmeimportjob.cpp,
lang/qt/src/qgpgmeimportjob.h,
lang/qt/src/qgpgmekeyformailboxjob.cpp,
lang/qt/src/qgpgmekeyformailboxjob.h,
lang/qt/src/qgpgmekeygenerationjob.cpp,
lang/qt/src/qgpgmekeygenerationjob.h,
lang/qt/src/qgpgmekeylistjob.cpp,
lang/qt/src/qgpgmekeylistjob.h,
lang/qt/src/qgpgmelistallkeysjob.cpp,
lang/qt/src/qgpgmelistallkeysjob.h,
lang/qt/src/qgpgmenewcryptoconfig.cpp,
lang/qt/src/qgpgmenewcryptoconfig.h,
lang/qt/src/qgpgmerefreshkeysjob.cpp,
lang/qt/src/qgpgmerefreshkeysjob.h,
lang/qt/src/qgpgmesecretkeyexportjob.cpp,
lang/qt/src/qgpgmesecretkeyexportjob.h,
lang/qt/src/qgpgmesignencryptjob.cpp,
lang/qt/src/qgpgmesignencryptjob.h,
lang/qt/src/qgpgmesignjob.cpp,
lang/qt/src/qgpgmesignjob.h,
lang/qt/src/qgpgmesignkeyjob.cpp,
lang/qt/src/qgpgmesignkeyjob.h,
lang/qt/src/qgpgmetofupolicyjob.cpp,
lang/qt/src/qgpgmetofupolicyjob.h,
lang/qt/src/qgpgmeverifydetachedjob.cpp,
lang/qt/src/qgpgmeverifydetachedjob.h,
lang/qt/src/qgpgmeverifyopaquejob.cpp,
lang/qt/src/qgpgmeverifyopaquejob.h,
lang/qt/src/qgpgmewkspublishjob.cpp,
lang/qt/src/qgpgmewkspublishjob.h,
lang/qt/src/refreshkeysjob.h,
lang/qt/src/signencryptjob.h,
lang/qt/src/signjob.h,
lang/qt/src/signkeyjob.h,
lang/qt/src/specialjob.h,
lang/qt/src/threadedjobmixin.cpp,
lang/qt/src/threadedjobmixin.h,
lang/qt/src/tofupolicyjob.h,
lang/qt/src/verifydetachedjob.h,
lang/qt/src/verifyopaquejob.h,
lang/qt/src/wkspublishjob.h,
lang/qt/tests/run-keyformailboxjob.cpp,
lang/qt/tests/t-config.cpp,
lang/qt/tests/t-encrypt.cpp,
lang/qt/tests/t-keylist.cpp,
lang/qt/tests/t-keylocate.cpp,
lang/qt/tests/t-ownertrust.cpp,
lang/qt/tests/t-support.cpp,
lang/qt/tests/t-support.h,
lang/qt/tests/t-tofuinfo.cpp,
lang/qt/tests/t-various.cpp,
lang/qt/tests/t-verify.cpp,
lang/qt/tests/t-wkspublish.cpp,
tests/gpg/t-encrypt-mixed.c,
tests/gpg/t-thread-keylist-verify.c,
tests/gpg/t-thread-keylist.c,
tests/run-decrypt.c: Change Intevation GmbH copyright to BSI.
--
This should make it more transparent where the BSI is the actual
copyright holder as the code was mostly developed as part of a
development contract.
2017-04-25 08:24:11 +00:00
|
|
|
Copyright (c) 2017 by Bundesamt für Sicherheit in der Informationstechnik
|
|
|
|
Software engineering by Intevation GmbH
|
2017-01-11 15:18:17 +00:00
|
|
|
|
|
|
|
QGpgME 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 of the
|
|
|
|
License, or (at your option) any later version.
|
|
|
|
|
|
|
|
QGpgME 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, write to the Free Software
|
|
|
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
|
|
|
|
In addition, as a special exception, the copyright holders give
|
|
|
|
permission to link the code of this program with any edition of
|
|
|
|
the Qt library by Trolltech AS, Norway (or with modified versions
|
|
|
|
of Qt that use the same license as Qt), and distribute linked
|
|
|
|
combinations including the two. You must obey the GNU General
|
|
|
|
Public License in all respects for all of the code used other than
|
|
|
|
Qt. If you modify this file, you may extend this exception to
|
|
|
|
your version of the file, but you are not obligated to do so. If
|
|
|
|
you do not wish to do so, delete this exception statement from
|
|
|
|
your version.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <QDebug>
|
|
|
|
#include <QTest>
|
|
|
|
#include <QSignalSpy>
|
2017-02-07 09:01:58 +00:00
|
|
|
#include <QTemporaryDir>
|
2017-01-11 15:18:17 +00:00
|
|
|
#include "keylistjob.h"
|
|
|
|
#include "protocol.h"
|
|
|
|
#include "keylistresult.h"
|
|
|
|
#include "context.h"
|
|
|
|
#include "engineinfo.h"
|
2017-03-13 10:16:41 +00:00
|
|
|
#include "dn.h"
|
2017-03-22 15:41:04 +00:00
|
|
|
#include "data.h"
|
|
|
|
#include "dataprovider.h"
|
2021-06-22 16:49:45 +00:00
|
|
|
#include "signkeyjob.h"
|
2017-01-11 15:18:17 +00:00
|
|
|
|
|
|
|
#include "t-support.h"
|
|
|
|
|
|
|
|
using namespace QGpgME;
|
|
|
|
using namespace GpgME;
|
|
|
|
|
2017-03-22 15:41:04 +00:00
|
|
|
static const char aKey[] = "-----BEGIN PGP PUBLIC KEY BLOCK-----\n"
|
|
|
|
"\n"
|
|
|
|
"mDMEWG+w/hYJKwYBBAHaRw8BAQdAiq1oStvDYg8ZfFs5DgisYJo8dJxD+C/AA21O\n"
|
|
|
|
"K/aif0O0GXRvZnVfY29uZmxpY3RAZXhhbXBsZS5jb22IlgQTFggAPhYhBHoJBLaV\n"
|
|
|
|
"DamYAgoa1L5BwMOl/x88BQJYb7D+AhsDBQkDwmcABQsJCAcCBhUICQoLAgQWAgMB\n"
|
|
|
|
"Ah4BAheAAAoJEL5BwMOl/x88GvwA/0SxkbLyAcshGm2PRrPsFQsSVAfwaSYFVmS2\n"
|
|
|
|
"cMVIw1PfAQDclRH1Z4MpufK07ju4qI33o4s0UFpVRBuSxt7A4P2ZD7g4BFhvsP4S\n"
|
|
|
|
"CisGAQQBl1UBBQEBB0AmVrgaDNJ7K2BSalsRo2EkRJjHGqnp5bBB0tapnF81CQMB\n"
|
|
|
|
"CAeIeAQYFggAIBYhBHoJBLaVDamYAgoa1L5BwMOl/x88BQJYb7D+AhsMAAoJEL5B\n"
|
|
|
|
"wMOl/x88OR0BAMq4/vmJUORRTmzjHcv/DDrQB030DSq666rlckGIKTShAPoDXM9N\n"
|
|
|
|
"0gZK+YzvrinSKZXHmn0aSwmC1/hyPybJPEljBw==\n"
|
|
|
|
"=p2Oj\n"
|
|
|
|
"-----END PGP PUBLIC KEY BLOCK-----\n";
|
|
|
|
|
2017-01-11 15:18:17 +00:00
|
|
|
class TestVarious: public QGpgMETest
|
|
|
|
{
|
|
|
|
Q_OBJECT
|
|
|
|
|
|
|
|
Q_SIGNALS:
|
|
|
|
void asyncDone();
|
|
|
|
|
|
|
|
private Q_SLOTS:
|
2017-03-13 10:16:41 +00:00
|
|
|
void testDN()
|
|
|
|
{
|
|
|
|
DN dn(QStringLiteral("CN=Before\\0DAfter,OU=Test,DC=North America,DC=Fabrikam,DC=COM"));
|
|
|
|
QVERIFY(dn.dn() == QStringLiteral("CN=Before\rAfter,OU=Test,DC=North America,DC=Fabrikam,DC=COM"));
|
|
|
|
QStringList attrOrder;
|
|
|
|
attrOrder << QStringLiteral("DC") << QStringLiteral("OU") << QStringLiteral("CN");
|
|
|
|
dn.setAttributeOrder(attrOrder);
|
|
|
|
QVERIFY(dn.prettyDN() == QStringLiteral("DC=North America,DC=Fabrikam,DC=COM,OU=Test,CN=Before\rAfter"));
|
|
|
|
}
|
2017-01-11 15:18:17 +00:00
|
|
|
|
2017-03-22 15:41:04 +00:00
|
|
|
void testKeyFromFile()
|
|
|
|
{
|
|
|
|
if (GpgME::engineInfo(GpgME::GpgEngine).engineVersion() < "2.1.14") {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
QGpgME::QByteArrayDataProvider dp(aKey);
|
|
|
|
Data data(&dp);
|
|
|
|
const auto keys = data.toKeys();
|
|
|
|
QVERIFY(keys.size() == 1);
|
|
|
|
const auto key = keys[0];
|
|
|
|
QVERIFY(!key.isNull());
|
|
|
|
QVERIFY(key.primaryFingerprint() == QStringLiteral("7A0904B6950DA998020A1AD4BE41C0C3A5FF1F3C"));
|
|
|
|
}
|
|
|
|
|
2018-05-29 07:16:22 +00:00
|
|
|
void testDataRewind()
|
|
|
|
{
|
|
|
|
if (GpgME::engineInfo(GpgME::GpgEngine).engineVersion() < "2.1.14") {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
QGpgME::QByteArrayDataProvider dp(aKey);
|
|
|
|
Data data(&dp);
|
|
|
|
char buf[20];
|
|
|
|
data.read(buf, 20);
|
|
|
|
|
|
|
|
auto keys = data.toKeys();
|
|
|
|
QVERIFY(keys.size() == 0);
|
|
|
|
|
|
|
|
data.rewind();
|
|
|
|
|
|
|
|
keys = data.toKeys();
|
|
|
|
QVERIFY(keys.size() == 1);
|
|
|
|
}
|
|
|
|
|
2017-01-11 15:18:17 +00:00
|
|
|
void testQuickUid()
|
|
|
|
{
|
|
|
|
if (GpgME::engineInfo(GpgME::GpgEngine).engineVersion() < "2.1.13") {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
KeyListJob *job = openpgp()->keyListJob(false, true, true);
|
|
|
|
std::vector<GpgME::Key> keys;
|
|
|
|
GpgME::KeyListResult result = job->exec(QStringList() << QStringLiteral("alfa@example.net"),
|
|
|
|
false, keys);
|
|
|
|
delete job;
|
2017-01-11 15:20:31 +00:00
|
|
|
QVERIFY (!result.error());
|
|
|
|
QVERIFY (keys.size() == 1);
|
2017-01-11 15:18:17 +00:00
|
|
|
Key key = keys.front();
|
|
|
|
|
|
|
|
QVERIFY (key.numUserIDs() == 3);
|
|
|
|
const char uid[] = "Foo Bar (with comment) <foo@bar.baz>";
|
|
|
|
|
|
|
|
auto ctx = Context::createForProtocol(key.protocol());
|
2017-01-11 15:20:31 +00:00
|
|
|
QVERIFY (ctx);
|
2017-01-11 15:18:17 +00:00
|
|
|
TestPassphraseProvider provider;
|
|
|
|
ctx->setPassphraseProvider(&provider);
|
|
|
|
ctx->setPinentryMode(Context::PinentryLoopback);
|
|
|
|
|
|
|
|
QVERIFY(!ctx->addUid(key, uid));
|
|
|
|
delete ctx;
|
|
|
|
key.update();
|
|
|
|
|
|
|
|
QVERIFY (key.numUserIDs() == 4);
|
|
|
|
bool id_found = false;;
|
|
|
|
for (const auto &u: key.userIDs()) {
|
|
|
|
if (!strcmp (u.id(), uid)) {
|
|
|
|
QVERIFY (!u.isRevoked());
|
|
|
|
id_found = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
QVERIFY (id_found);
|
|
|
|
|
|
|
|
ctx = Context::createForProtocol(key.protocol());
|
|
|
|
QVERIFY (!ctx->revUid(key, uid));
|
|
|
|
delete ctx;
|
|
|
|
key.update();
|
|
|
|
|
|
|
|
bool id_revoked = false;;
|
|
|
|
for (const auto &u: key.userIDs()) {
|
|
|
|
if (!strcmp (u.id(), uid)) {
|
|
|
|
id_revoked = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2017-01-11 15:20:31 +00:00
|
|
|
QVERIFY(id_revoked);
|
2017-01-11 15:18:17 +00:00
|
|
|
}
|
|
|
|
|
2020-08-04 14:23:55 +00:00
|
|
|
void testSetExpire()
|
|
|
|
{
|
|
|
|
if (GpgME::engineInfo(GpgME::GpgEngine).engineVersion() < "2.1.22") {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
KeyListJob *job = openpgp()->keyListJob(false, true, true);
|
|
|
|
std::vector<GpgME::Key> keys;
|
|
|
|
GpgME::KeyListResult result = job->exec(QStringList() << QStringLiteral("alfa@example.net"),
|
|
|
|
false, keys);
|
|
|
|
delete job;
|
|
|
|
QVERIFY (!result.error());
|
|
|
|
QVERIFY (keys.size() == 1);
|
|
|
|
Key key = keys.front();
|
|
|
|
|
|
|
|
QVERIFY (key.subkey(0).expirationTime() == time_t(0));
|
|
|
|
QVERIFY (key.subkey(1).expirationTime() == time_t(0));
|
|
|
|
|
|
|
|
auto ctx = Context::createForProtocol(key.protocol());
|
|
|
|
QVERIFY (ctx);
|
|
|
|
TestPassphraseProvider provider;
|
|
|
|
ctx->setPassphraseProvider(&provider);
|
|
|
|
ctx->setPinentryMode(Context::PinentryLoopback);
|
|
|
|
|
|
|
|
// change expiration of the main key
|
|
|
|
QVERIFY(!ctx->setExpire(key, 1000));
|
|
|
|
delete ctx;
|
|
|
|
key.update();
|
|
|
|
|
|
|
|
QVERIFY (key.subkey(0).expirationTime() != time_t(0));
|
|
|
|
QVERIFY (key.subkey(1).expirationTime() == time_t(0));
|
|
|
|
time_t keyExpiration = key.subkey(0).expirationTime();
|
|
|
|
|
|
|
|
// change expiration of all subkeys
|
|
|
|
ctx = Context::createForProtocol(key.protocol());
|
|
|
|
QVERIFY(!ctx->setExpire(key, 2000, std::vector<Subkey>(), Context::SetExpireAllSubkeys));
|
|
|
|
delete ctx;
|
|
|
|
key.update();
|
|
|
|
|
|
|
|
QVERIFY (key.subkey(0).expirationTime() == keyExpiration);
|
|
|
|
QVERIFY (key.subkey(1).expirationTime() != time_t(0));
|
|
|
|
time_t subkeyExpiration = key.subkey(1).expirationTime();
|
|
|
|
|
|
|
|
// change expiration of specific subkey(s)
|
|
|
|
ctx = Context::createForProtocol(key.protocol());
|
|
|
|
std::vector<Subkey> specificSubkeys;
|
|
|
|
specificSubkeys.push_back(key.subkey(1));
|
|
|
|
QVERIFY(!ctx->setExpire(key, 3000, specificSubkeys));
|
|
|
|
delete ctx;
|
|
|
|
key.update();
|
|
|
|
|
|
|
|
QVERIFY (key.subkey(0).expirationTime() == keyExpiration);
|
|
|
|
QVERIFY (key.subkey(1).expirationTime() != subkeyExpiration);
|
2020-08-14 09:11:23 +00:00
|
|
|
|
|
|
|
// test error handling: calling setExpire() with the primary key as
|
|
|
|
// subkey should fail with "subkey <primary key fpr> not found"
|
|
|
|
ctx = Context::createForProtocol(key.protocol());
|
|
|
|
std::vector<Subkey> primaryKey;
|
|
|
|
primaryKey.push_back(key.subkey(0));
|
|
|
|
const auto err = ctx->setExpire(key, 3000, primaryKey);
|
|
|
|
QCOMPARE(err.code(), GPG_ERR_NOT_FOUND);
|
|
|
|
delete ctx;
|
2020-08-04 14:23:55 +00:00
|
|
|
}
|
|
|
|
|
2021-06-22 16:49:45 +00:00
|
|
|
void testSignKeyWithoutExpiration()
|
|
|
|
{
|
|
|
|
Error err;
|
|
|
|
|
|
|
|
if (!loopbackSupported()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto ctx = Context::create(OpenPGP);
|
|
|
|
QVERIFY(ctx);
|
|
|
|
|
|
|
|
// Get the signing key (alfa@example.net)
|
|
|
|
auto seckey = ctx->key("A0FF4590BB6122EDEF6E3C542D727CC768697734", err, true);
|
|
|
|
QVERIFY(!err);
|
|
|
|
QVERIFY(!seckey.isNull());
|
|
|
|
|
|
|
|
// Get the target key (Bob / Bravo Test)
|
|
|
|
auto target = ctx->key("D695676BDCEDCC2CDD6152BCFE180B1DA9E3B0B2", err, false);
|
|
|
|
QVERIFY(!err);
|
|
|
|
QVERIFY(!target.isNull());
|
|
|
|
QVERIFY(target.numUserIDs() > 0);
|
|
|
|
|
|
|
|
// Create the job
|
|
|
|
auto job = std::unique_ptr<SignKeyJob>{openpgp()->signKeyJob()};
|
|
|
|
QVERIFY(job);
|
|
|
|
|
|
|
|
// Hack in the passphrase provider
|
|
|
|
auto jobCtx = Job::context(job.get());
|
|
|
|
TestPassphraseProvider provider;
|
|
|
|
jobCtx->setPassphraseProvider(&provider);
|
|
|
|
jobCtx->setPinentryMode(Context::PinentryLoopback);
|
|
|
|
|
|
|
|
// Setup the job
|
|
|
|
job->setExportable(true);
|
|
|
|
job->setSigningKey(seckey);
|
|
|
|
job->setDupeOk(true);
|
|
|
|
|
|
|
|
connect(job.get(), &SignKeyJob::result,
|
|
|
|
this, [this] (const GpgME::Error &err2, const QString &, const GpgME::Error &) {
|
|
|
|
Q_EMIT asyncDone();
|
|
|
|
if (err2) {
|
|
|
|
if (err2.code() == GPG_ERR_GENERAL) {
|
|
|
|
QFAIL(qPrintable(QString("The SignKeyJob failed with '%1'.\n"
|
|
|
|
"Hint: Run with GPGMEPP_INTERACTOR_DEBUG=stderr to debug the edit interaction.").arg(err2.asString())));
|
|
|
|
} else {
|
|
|
|
QFAIL(qPrintable(QString("The SignKeyJob failed with '%1'.").arg(err2.asString())));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
job->start(target);
|
|
|
|
QSignalSpy spy{this, &TestVarious::asyncDone};
|
|
|
|
QVERIFY(spy.wait(QSIGNALSPY_TIMEOUT));
|
|
|
|
|
|
|
|
// At this point the signature should have been added.
|
|
|
|
target.update();
|
|
|
|
const auto keySignature = target.userID(0).signature(target.userID(0).numSignatures() - 1);
|
|
|
|
QVERIFY(keySignature.neverExpires());
|
|
|
|
}
|
|
|
|
|
|
|
|
void testSignKeyWithExpiration()
|
|
|
|
{
|
|
|
|
Error err;
|
|
|
|
|
|
|
|
if (!loopbackSupported()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto ctx = Context::create(OpenPGP);
|
|
|
|
QVERIFY(ctx);
|
|
|
|
|
|
|
|
// Get the signing key (alfa@example.net)
|
|
|
|
auto seckey = ctx->key("A0FF4590BB6122EDEF6E3C542D727CC768697734", err, true);
|
|
|
|
QVERIFY(!err);
|
|
|
|
QVERIFY(!seckey.isNull());
|
|
|
|
|
|
|
|
// Get the target key (Bob / Bravo Test)
|
|
|
|
auto target = ctx->key("D695676BDCEDCC2CDD6152BCFE180B1DA9E3B0B2", err, false);
|
|
|
|
QVERIFY(!err);
|
|
|
|
QVERIFY(!target.isNull());
|
|
|
|
QVERIFY(target.numUserIDs() > 0);
|
|
|
|
|
|
|
|
// Create the job
|
|
|
|
auto job = std::unique_ptr<SignKeyJob>{openpgp()->signKeyJob()};
|
|
|
|
QVERIFY(job);
|
|
|
|
|
|
|
|
// Hack in the passphrase provider
|
|
|
|
auto jobCtx = Job::context(job.get());
|
|
|
|
TestPassphraseProvider provider;
|
|
|
|
jobCtx->setPassphraseProvider(&provider);
|
|
|
|
jobCtx->setPinentryMode(Context::PinentryLoopback);
|
|
|
|
|
|
|
|
// Setup the job
|
|
|
|
job->setExportable(true);
|
|
|
|
job->setSigningKey(seckey);
|
|
|
|
job->setDupeOk(true);
|
|
|
|
job->setExpirationDate(QDate{2222, 2, 22});
|
|
|
|
|
|
|
|
connect(job.get(), &SignKeyJob::result,
|
|
|
|
this, [this] (const GpgME::Error &err2, const QString &, const GpgME::Error &) {
|
|
|
|
Q_EMIT asyncDone();
|
|
|
|
if (err2) {
|
|
|
|
if (err2.code() == GPG_ERR_GENERAL) {
|
|
|
|
QFAIL(qPrintable(QString("The SignKeyJob failed with '%1'.\n"
|
|
|
|
"Hint: Run with GPGMEPP_INTERACTOR_DEBUG=stderr to debug the edit interaction.").arg(err2.asString())));
|
|
|
|
} else {
|
|
|
|
QFAIL(qPrintable(QString("The SignKeyJob failed with '%1'.").arg(err2.asString())));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
QTest::ignoreMessage(QtWarningMsg, "Expiration of certification has been changed to QDate(\"2106-02-06\")");
|
|
|
|
|
|
|
|
job->start(target);
|
|
|
|
QSignalSpy spy{this, &TestVarious::asyncDone};
|
|
|
|
QVERIFY(spy.wait(QSIGNALSPY_TIMEOUT));
|
|
|
|
|
|
|
|
// At this point the signature should have been added.
|
|
|
|
target.update();
|
|
|
|
const auto keySignature = target.userID(0).signature(target.userID(0).numSignatures() - 1);
|
|
|
|
QVERIFY(!keySignature.neverExpires());
|
2021-07-08 09:54:06 +00:00
|
|
|
const auto expirationDate = QDateTime::fromSecsSinceEpoch(uint_least32_t(keySignature.expirationTime())).date();
|
2021-06-22 16:49:45 +00:00
|
|
|
QCOMPARE(expirationDate, QDate(2106, 2, 6)); // expiration date is capped at 2106-02-06
|
|
|
|
}
|
|
|
|
|
2017-09-04 09:25:34 +00:00
|
|
|
void testVersion()
|
|
|
|
{
|
|
|
|
QVERIFY(EngineInfo::Version("2.1.0") < EngineInfo::Version("2.1.1"));
|
|
|
|
QVERIFY(EngineInfo::Version("2.1.10") < EngineInfo::Version("2.1.11"));
|
|
|
|
QVERIFY(EngineInfo::Version("2.2.0") > EngineInfo::Version("2.1.19"));
|
|
|
|
QVERIFY(EngineInfo::Version("1.0.0") < EngineInfo::Version("2.0.0"));
|
|
|
|
QVERIFY(EngineInfo::Version("0.1.0") < EngineInfo::Version("1.0.0"));
|
|
|
|
QVERIFY(!(EngineInfo::Version("2.0.0") < EngineInfo::Version("2.0.0")));
|
2020-10-15 14:45:31 +00:00
|
|
|
QVERIFY(!(EngineInfo::Version("2.0.0") > EngineInfo::Version("2.0.0")));
|
2017-09-04 09:25:34 +00:00
|
|
|
QVERIFY(EngineInfo::Version("3.0.0") > EngineInfo::Version("2.3.20"));
|
|
|
|
QVERIFY(EngineInfo::Version("3.0.1") > EngineInfo::Version("3.0.0"));
|
|
|
|
QVERIFY(EngineInfo::Version("3.1.0") > EngineInfo::Version("3.0.20"));
|
2020-10-15 14:50:32 +00:00
|
|
|
|
|
|
|
QVERIFY(EngineInfo::Version("1.1.1") <= "2.0.0");
|
|
|
|
QVERIFY(EngineInfo::Version("1.1.1") <= "1.2.0");
|
|
|
|
QVERIFY(EngineInfo::Version("1.1.1") <= "1.1.2");
|
|
|
|
QVERIFY(EngineInfo::Version("1.1.1") <= "1.1.1");
|
|
|
|
QVERIFY(!(EngineInfo::Version("1.1.1") <= "1.1.0"));
|
|
|
|
QVERIFY(!(EngineInfo::Version("1.1.1") <= "1.0.9"));
|
|
|
|
QVERIFY(!(EngineInfo::Version("1.1.1") <= "0.9.9"));
|
|
|
|
|
|
|
|
QVERIFY(!(EngineInfo::Version("1.1.1") == "2.0.0"));
|
|
|
|
QVERIFY(!(EngineInfo::Version("1.1.1") == "1.2.0"));
|
|
|
|
QVERIFY(!(EngineInfo::Version("1.1.1") == "1.1.2"));
|
|
|
|
QVERIFY(EngineInfo::Version("1.1.1") == "1.1.1");
|
|
|
|
QVERIFY(!(EngineInfo::Version("1.1.1") == "1.1.0"));
|
|
|
|
QVERIFY(!(EngineInfo::Version("1.1.1") == "1.0.9"));
|
|
|
|
QVERIFY(!(EngineInfo::Version("1.1.1") == "0.9.9"));
|
|
|
|
|
|
|
|
QVERIFY(EngineInfo::Version("1.1.1") != "2.0.0");
|
|
|
|
QVERIFY(EngineInfo::Version("1.1.1") != "1.2.0");
|
|
|
|
QVERIFY(EngineInfo::Version("1.1.1") != "1.1.2");
|
|
|
|
QVERIFY(!(EngineInfo::Version("1.1.1") != "1.1.1"));
|
|
|
|
QVERIFY(EngineInfo::Version("1.1.1") != "1.1.0");
|
|
|
|
QVERIFY(EngineInfo::Version("1.1.1") != "1.0.9");
|
|
|
|
QVERIFY(EngineInfo::Version("1.1.1") != "0.9.9");
|
|
|
|
|
|
|
|
QVERIFY(!(EngineInfo::Version("1.1.1") >= "2.0.0"));
|
|
|
|
QVERIFY(!(EngineInfo::Version("1.1.1") >= "1.2.0"));
|
|
|
|
QVERIFY(!(EngineInfo::Version("1.1.1") >= "1.1.2"));
|
|
|
|
QVERIFY(EngineInfo::Version("1.1.1") >= "1.1.1");
|
|
|
|
QVERIFY(EngineInfo::Version("1.1.1") >= "1.1.0");
|
|
|
|
QVERIFY(EngineInfo::Version("1.1.1") >= "1.0.9");
|
|
|
|
QVERIFY(EngineInfo::Version("1.1.1") >= "0.9.9");
|
2017-09-04 09:25:34 +00:00
|
|
|
}
|
|
|
|
|
2017-01-11 15:18:17 +00:00
|
|
|
void initTestCase()
|
|
|
|
{
|
|
|
|
QGpgMETest::initTestCase();
|
|
|
|
const QString gpgHome = qgetenv("GNUPGHOME");
|
2017-01-11 15:20:31 +00:00
|
|
|
QVERIFY(copyKeyrings(gpgHome, mDir.path()));
|
2017-01-11 15:18:17 +00:00
|
|
|
qputenv("GNUPGHOME", mDir.path().toUtf8());
|
2021-06-22 16:49:45 +00:00
|
|
|
QFile conf(mDir.path() + QStringLiteral("/gpg.conf"));
|
|
|
|
QVERIFY(conf.open(QIODevice::WriteOnly));
|
|
|
|
if (GpgME::engineInfo(GpgME::GpgEngine).engineVersion() >= "2.2.18") {
|
|
|
|
conf.write("allow-weak-key-signatures");
|
|
|
|
}
|
|
|
|
conf.close();
|
2017-01-11 15:18:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
QTemporaryDir mDir;
|
|
|
|
};
|
|
|
|
|
|
|
|
QTEST_MAIN(TestVarious)
|
|
|
|
|
|
|
|
#include "t-various.moc"
|