2016-08-09 12:10:15 +00:00
|
|
|
/* t-encrypt.cpp
|
|
|
|
|
|
|
|
This file is part of qgpgme, the Qt API binding for gpgme
|
|
|
|
Copyright (c) 2016 Intevation GmbH
|
|
|
|
|
|
|
|
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.
|
|
|
|
*/
|
cpp, qt: Include config.h
lang/cpp/src/callbacks.cpp,
lang/cpp/src/configuration.cpp,
lang/cpp/src/context.cpp,
lang/cpp/src/context_glib.cpp,
lang/cpp/src/context_qt.cpp,
lang/cpp/src/context_vanilla.cpp,
lang/cpp/src/data.cpp,
lang/cpp/src/decryptionresult.cpp,
lang/cpp/src/defaultassuantransaction.cpp,
lang/cpp/src/editinteractor.cpp,
lang/cpp/src/encryptionresult.cpp,
lang/cpp/src/engineinfo.cpp,
lang/cpp/src/eventloopinteractor.cpp,
lang/cpp/src/exception.cpp,
lang/cpp/src/gpgadduserideditinteractor.cpp,
lang/cpp/src/gpgagentgetinfoassuantransaction.cpp,
lang/cpp/src/gpgsetexpirytimeeditinteractor.cpp,
lang/cpp/src/gpgsetownertrusteditinteractor.cpp,
lang/cpp/src/gpgsignkeyeditinteractor.cpp,
lang/cpp/src/importresult.cpp,
lang/cpp/src/key.cpp,
lang/cpp/src/keygenerationresult.cpp,
lang/cpp/src/keylistresult.cpp,
lang/cpp/src/scdgetinfoassuantransaction.cpp,
lang/cpp/src/signingresult.cpp,
lang/cpp/src/tofuinfo.cpp,
lang/cpp/src/trustitem.cpp,
lang/cpp/src/verificationresult.cpp,
lang/cpp/src/vfsmountresult.cpp,
lang/qt/src/dataprovider.cpp,
lang/qt/src/defaultkeygenerationjob.cpp,
lang/qt/src/gpgme_backend_debug.cpp,
lang/qt/src/job.cpp,
lang/qt/src/qgpgmeadduseridjob.cpp,
lang/qt/src/qgpgmebackend.cpp,
lang/qt/src/qgpgmechangeexpiryjob.cpp,
lang/qt/src/qgpgmechangeownertrustjob.cpp,
lang/qt/src/qgpgmechangepasswdjob.cpp,
lang/qt/src/qgpgmedecryptjob.cpp,
lang/qt/src/qgpgmedecryptverifyjob.cpp,
lang/qt/src/qgpgmedeletejob.cpp,
lang/qt/src/qgpgmedownloadjob.cpp,
lang/qt/src/qgpgmeencryptjob.cpp,
lang/qt/src/qgpgmeexportjob.cpp,
lang/qt/src/qgpgmeimportfromkeyserverjob.cpp,
lang/qt/src/qgpgmeimportjob.cpp,
lang/qt/src/qgpgmekeyformailboxjob.cpp,
lang/qt/src/qgpgmekeygenerationjob.cpp,
lang/qt/src/qgpgmekeylistjob.cpp,
lang/qt/src/qgpgmelistallkeysjob.cpp,
lang/qt/src/qgpgmenewcryptoconfig.cpp,
lang/qt/src/qgpgmerefreshkeysjob.cpp,
lang/qt/src/qgpgmesecretkeyexportjob.cpp,
lang/qt/src/qgpgmesignencryptjob.cpp,
lang/qt/src/qgpgmesignjob.cpp,
lang/qt/src/qgpgmesignkeyjob.cpp,
lang/qt/src/qgpgmetofupolicyjob.cpp,
lang/qt/src/qgpgmeverifydetachedjob.cpp,
lang/qt/src/qgpgmeverifyopaquejob.cpp,
lang/qt/src/qgpgmewkspublishjob.cpp,
lang/qt/src/threadedjobmixin.cpp,
lang/qt/tests/run-keyformailboxjob.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-tofuinfo.cpp,
lang/qt/tests/t-wkspublish.cpp: Include config.h
--
This fixes problems with mismatching definitions. Most
notably _FILE_OFFSET_BITS is now always set correctly.
2016-09-23 13:22:29 +00:00
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
2016-08-09 12:10:15 +00:00
|
|
|
#include <QDebug>
|
|
|
|
#include <QTest>
|
|
|
|
#include <QTemporaryDir>
|
2016-08-12 14:55:51 +00:00
|
|
|
#include <QSignalSpy>
|
|
|
|
#include <QBuffer>
|
2016-08-09 12:10:15 +00:00
|
|
|
#include "keylistjob.h"
|
|
|
|
#include "encryptjob.h"
|
|
|
|
#include "qgpgmeencryptjob.h"
|
|
|
|
#include "encryptionresult.h"
|
|
|
|
#include "decryptionresult.h"
|
|
|
|
#include "qgpgmedecryptjob.h"
|
|
|
|
#include "qgpgmebackend.h"
|
|
|
|
#include "keylistresult.h"
|
2016-08-12 14:55:51 +00:00
|
|
|
#include "engineinfo.h"
|
2016-08-09 12:10:15 +00:00
|
|
|
#include "t-support.h"
|
|
|
|
|
2016-08-12 14:55:51 +00:00
|
|
|
#define PROGRESS_TEST_SIZE 1 * 1024 * 1024
|
|
|
|
|
2016-08-09 12:10:15 +00:00
|
|
|
using namespace QGpgME;
|
|
|
|
using namespace GpgME;
|
|
|
|
|
2016-10-05 15:19:20 +00:00
|
|
|
static bool decryptSupported()
|
|
|
|
{
|
|
|
|
/* With GnuPG 2.0.x (at least 2.0.26 by default on jessie)
|
|
|
|
* the passphrase_cb does not work. So the test popped up
|
|
|
|
* a pinentry. So tests requiring decryption don't work. */
|
|
|
|
static auto version = GpgME::engineInfo(GpgME::GpgEngine).engineVersion();
|
|
|
|
if (version < "2.0.0") {
|
|
|
|
/* With 1.4 it just works */
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (version < "2.1.0") {
|
|
|
|
/* With 2.1 it works with loopback mode */
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-08-09 12:10:15 +00:00
|
|
|
class EncryptionTest : public QGpgMETest
|
|
|
|
{
|
|
|
|
Q_OBJECT
|
|
|
|
|
2016-08-12 14:55:51 +00:00
|
|
|
Q_SIGNALS:
|
|
|
|
void asyncDone();
|
|
|
|
|
2016-08-09 12:10:15 +00:00
|
|
|
private Q_SLOTS:
|
|
|
|
|
|
|
|
void testSimpleEncryptDecrypt()
|
|
|
|
{
|
|
|
|
auto listjob = openpgp()->keyListJob(false, false, false);
|
|
|
|
std::vector<Key> keys;
|
|
|
|
auto keylistresult = listjob->exec(QStringList() << QStringLiteral("alfa@example.net"),
|
|
|
|
false, keys);
|
2017-01-11 15:20:31 +00:00
|
|
|
QVERIFY(!keylistresult.error());
|
|
|
|
QVERIFY(keys.size() == 1);
|
2016-08-09 12:10:15 +00:00
|
|
|
delete listjob;
|
|
|
|
|
|
|
|
auto job = openpgp()->encryptJob(/*ASCII Armor */true, /* Textmode */ true);
|
2017-01-11 15:20:31 +00:00
|
|
|
QVERIFY(job);
|
2016-08-09 12:10:15 +00:00
|
|
|
QByteArray cipherText;
|
|
|
|
auto result = job->exec(keys, QStringLiteral("Hello World").toUtf8(), Context::AlwaysTrust, cipherText);
|
|
|
|
delete job;
|
2017-01-11 15:20:31 +00:00
|
|
|
QVERIFY(!result.error());
|
2016-08-09 12:10:15 +00:00
|
|
|
const auto cipherString = QString::fromUtf8(cipherText);
|
2017-01-11 15:20:31 +00:00
|
|
|
QVERIFY(cipherString.startsWith("-----BEGIN PGP MESSAGE-----"));
|
2016-08-09 12:10:15 +00:00
|
|
|
|
|
|
|
/* Now decrypt */
|
2016-10-05 15:19:20 +00:00
|
|
|
if (!decryptSupported()) {
|
|
|
|
return;
|
|
|
|
}
|
2016-08-09 12:10:15 +00:00
|
|
|
auto ctx = Context::createForProtocol(OpenPGP);
|
2016-08-10 10:05:32 +00:00
|
|
|
TestPassphraseProvider provider;
|
|
|
|
ctx->setPassphraseProvider(&provider);
|
2016-08-09 12:10:15 +00:00
|
|
|
ctx->setPinentryMode(Context::PinentryLoopback);
|
|
|
|
auto decJob = new QGpgMEDecryptJob(ctx);
|
|
|
|
QByteArray plainText;
|
|
|
|
auto decResult = decJob->exec(cipherText, plainText);
|
2017-01-11 15:20:31 +00:00
|
|
|
QVERIFY(!result.error());
|
|
|
|
QVERIFY(QString::fromUtf8(plainText) == QStringLiteral("Hello World"));
|
2016-08-09 12:10:15 +00:00
|
|
|
delete decJob;
|
|
|
|
}
|
|
|
|
|
2016-08-12 14:55:51 +00:00
|
|
|
void testProgress()
|
|
|
|
{
|
|
|
|
if (GpgME::engineInfo(GpgME::GpgEngine).engineVersion() < "2.1.15") {
|
|
|
|
// We can only test the progress with 2.1.15 as this started to
|
|
|
|
// have total progress for memory callbacks
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
auto listjob = openpgp()->keyListJob(false, false, false);
|
|
|
|
std::vector<Key> keys;
|
|
|
|
auto keylistresult = listjob->exec(QStringList() << QStringLiteral("alfa@example.net"),
|
|
|
|
false, keys);
|
2017-01-11 15:20:31 +00:00
|
|
|
QVERIFY(!keylistresult.error());
|
|
|
|
QVERIFY(keys.size() == 1);
|
2016-08-12 14:55:51 +00:00
|
|
|
delete listjob;
|
|
|
|
|
|
|
|
auto job = openpgp()->encryptJob(/*ASCII Armor */false, /* Textmode */ false);
|
2017-01-11 15:20:31 +00:00
|
|
|
QVERIFY(job);
|
2016-08-12 14:55:51 +00:00
|
|
|
QByteArray plainBa;
|
|
|
|
plainBa.fill('X', PROGRESS_TEST_SIZE);
|
|
|
|
QByteArray cipherText;
|
|
|
|
|
|
|
|
bool initSeen = false;
|
|
|
|
bool finishSeen = false;
|
2016-10-05 14:44:53 +00:00
|
|
|
connect(job, &Job::progress, this, [this, &initSeen, &finishSeen] (const QString&, int current, int total) {
|
2016-08-12 14:55:51 +00:00
|
|
|
// We only check for progress 0 and max progress as the other progress
|
|
|
|
// lines depend on the system speed and are as such unreliable to test.
|
2017-01-11 15:20:31 +00:00
|
|
|
QVERIFY(total == PROGRESS_TEST_SIZE);
|
2016-08-12 14:55:51 +00:00
|
|
|
if (current == 0) {
|
|
|
|
initSeen = true;
|
|
|
|
}
|
|
|
|
if (current == total) {
|
|
|
|
finishSeen = true;
|
|
|
|
}
|
2017-01-11 15:20:31 +00:00
|
|
|
QVERIFY(current >= 0 && current <= total);
|
2016-08-12 14:55:51 +00:00
|
|
|
});
|
2016-10-05 14:44:53 +00:00
|
|
|
connect(job, &EncryptJob::result, this, [this, &initSeen, &finishSeen] (const GpgME::EncryptionResult &,
|
|
|
|
const QByteArray &,
|
2016-08-12 14:55:51 +00:00
|
|
|
const QString,
|
|
|
|
const GpgME::Error) {
|
2017-01-11 15:20:31 +00:00
|
|
|
QVERIFY(initSeen);
|
|
|
|
QVERIFY(finishSeen);
|
2016-08-12 14:55:51 +00:00
|
|
|
Q_EMIT asyncDone();
|
|
|
|
});
|
|
|
|
|
|
|
|
auto inptr = std::shared_ptr<QIODevice>(new QBuffer(&plainBa));
|
|
|
|
inptr->open(QIODevice::ReadOnly);
|
|
|
|
auto outptr = std::shared_ptr<QIODevice>(new QBuffer(&cipherText));
|
|
|
|
outptr->open(QIODevice::WriteOnly);
|
|
|
|
|
|
|
|
job->start(keys, inptr, outptr, Context::AlwaysTrust);
|
|
|
|
QSignalSpy spy (this, SIGNAL(asyncDone()));
|
2017-01-11 15:20:31 +00:00
|
|
|
QVERIFY(spy.wait());
|
2016-08-12 14:55:51 +00:00
|
|
|
}
|
|
|
|
|
2016-08-09 12:10:15 +00:00
|
|
|
void testSymmetricEncryptDecrypt()
|
|
|
|
{
|
2016-10-05 15:19:20 +00:00
|
|
|
if (!decryptSupported()) {
|
|
|
|
return;
|
|
|
|
}
|
2016-08-09 12:10:15 +00:00
|
|
|
auto ctx = Context::createForProtocol(OpenPGP);
|
2016-08-10 10:05:32 +00:00
|
|
|
TestPassphraseProvider provider;
|
|
|
|
ctx->setPassphraseProvider(&provider);
|
2016-08-09 12:10:15 +00:00
|
|
|
ctx->setPinentryMode(Context::PinentryLoopback);
|
|
|
|
ctx->setArmor(true);
|
|
|
|
ctx->setTextMode(true);
|
|
|
|
auto job = new QGpgMEEncryptJob(ctx);
|
|
|
|
QByteArray cipherText;
|
|
|
|
auto result = job->exec(std::vector<Key>(), QStringLiteral("Hello symmetric World").toUtf8(), Context::AlwaysTrust, cipherText);
|
|
|
|
delete job;
|
2017-01-11 15:20:31 +00:00
|
|
|
QVERIFY(!result.error());
|
2016-08-09 12:10:15 +00:00
|
|
|
const auto cipherString = QString::fromUtf8(cipherText);
|
2017-01-11 15:20:31 +00:00
|
|
|
QVERIFY(cipherString.startsWith("-----BEGIN PGP MESSAGE-----"));
|
2016-08-09 12:10:15 +00:00
|
|
|
|
|
|
|
killAgent(mDir.path());
|
|
|
|
|
|
|
|
auto ctx2 = Context::createForProtocol(OpenPGP);
|
2016-08-10 10:05:32 +00:00
|
|
|
ctx2->setPassphraseProvider(&provider);
|
2016-08-09 12:10:15 +00:00
|
|
|
ctx2->setPinentryMode(Context::PinentryLoopback);
|
|
|
|
auto decJob = new QGpgMEDecryptJob(ctx2);
|
|
|
|
QByteArray plainText;
|
|
|
|
auto decResult = decJob->exec(cipherText, plainText);
|
2017-01-11 15:20:31 +00:00
|
|
|
QVERIFY(!result.error());
|
|
|
|
QVERIFY(QString::fromUtf8(plainText) == QStringLiteral("Hello symmetric World"));
|
2016-08-09 12:10:15 +00:00
|
|
|
delete decJob;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
/* Loopback and passphrase provider don't work for mixed encryption.
|
|
|
|
* So this test is disabled until gnupg(?) is fixed for this. */
|
|
|
|
void testMixedEncryptDecrypt()
|
|
|
|
{
|
2016-10-05 15:19:20 +00:00
|
|
|
if (!decryptSupported()) {
|
|
|
|
return;
|
|
|
|
}
|
2016-08-09 12:10:15 +00:00
|
|
|
auto listjob = openpgp()->keyListJob(false, false, false);
|
|
|
|
std::vector<Key> keys;
|
|
|
|
auto keylistresult = listjob->exec(QStringList() << QStringLiteral("alfa@example.net"),
|
|
|
|
false, keys);
|
2017-01-11 15:20:31 +00:00
|
|
|
QVERIFY(!keylistresult.error());
|
|
|
|
QVERIFY(keys.size() == 1);
|
2016-08-09 12:10:15 +00:00
|
|
|
delete listjob;
|
|
|
|
|
|
|
|
auto ctx = Context::createForProtocol(OpenPGP);
|
|
|
|
ctx->setPassphraseProvider(new TestPassphraseProvider);
|
|
|
|
ctx->setPinentryMode(Context::PinentryLoopback);
|
|
|
|
ctx->setArmor(true);
|
|
|
|
ctx->setTextMode(true);
|
|
|
|
auto job = new QGpgMEEncryptJob(ctx);
|
|
|
|
QByteArray cipherText;
|
|
|
|
printf("Before exec, flags: %x\n", Context::Symmetric | Context::AlwaysTrust);
|
|
|
|
auto result = job->exec(keys, QStringLiteral("Hello symmetric World").toUtf8(),
|
|
|
|
static_cast<Context::EncryptionFlags>(Context::Symmetric | Context::AlwaysTrust),
|
|
|
|
cipherText);
|
|
|
|
printf("After exec\n");
|
|
|
|
delete job;
|
2017-01-11 15:20:31 +00:00
|
|
|
QVERIFY(!result.error());
|
2016-08-09 12:10:15 +00:00
|
|
|
printf("Cipher:\n%s\n", cipherText.constData());
|
|
|
|
const auto cipherString = QString::fromUtf8(cipherText);
|
2017-01-11 15:20:31 +00:00
|
|
|
QVERIFY(cipherString.startsWith("-----BEGIN PGP MESSAGE-----"));
|
2016-08-09 12:10:15 +00:00
|
|
|
|
|
|
|
killAgent(mDir.path());
|
|
|
|
|
|
|
|
/* Now create a new homedir which with we test symetric decrypt. */
|
|
|
|
QTemporaryDir tmp;
|
|
|
|
qputenv("GNUPGHOME", tmp.path().toUtf8());
|
|
|
|
QFile agentConf(tmp.path() + QStringLiteral("/gpg-agent.conf"));
|
2017-01-11 15:20:31 +00:00
|
|
|
QVERIFY(agentConf.open(QIODevice::WriteOnly));
|
2016-08-09 12:10:15 +00:00
|
|
|
agentConf.write("allow-loopback-pinentry");
|
|
|
|
agentConf.close();
|
|
|
|
|
|
|
|
auto ctx2 = Context::createForProtocol(OpenPGP);
|
|
|
|
ctx2->setPassphraseProvider(new TestPassphraseProvider);
|
|
|
|
ctx2->setPinentryMode(Context::PinentryLoopback);
|
|
|
|
ctx2->setTextMode(true);
|
|
|
|
auto decJob = new QGpgMEDecryptJob(ctx2);
|
|
|
|
QByteArray plainText;
|
|
|
|
auto decResult = decJob->exec(cipherText, plainText);
|
2017-01-11 15:20:31 +00:00
|
|
|
QVERIFY(!decResult.error());
|
2016-08-09 12:10:15 +00:00
|
|
|
qDebug() << "Plain: " << plainText;
|
2017-01-11 15:20:31 +00:00
|
|
|
QVERIFY(QString::fromUtf8(plainText) == QStringLiteral("Hello symmetric World"));
|
2016-08-09 12:10:15 +00:00
|
|
|
delete decJob;
|
|
|
|
|
|
|
|
killAgent(tmp.path());
|
|
|
|
qputenv("GNUPGHOME", mDir.path().toUtf8());
|
|
|
|
}
|
|
|
|
|
|
|
|
public Q_SLOT:
|
|
|
|
|
|
|
|
void initTestCase()
|
|
|
|
{
|
|
|
|
QGpgMETest::initTestCase();
|
|
|
|
const QString gpgHome = qgetenv("GNUPGHOME");
|
|
|
|
qputenv("GNUPGHOME", mDir.path().toUtf8());
|
2017-01-11 15:20:31 +00:00
|
|
|
QVERIFY(mDir.isValid());
|
2016-08-09 12:10:15 +00:00
|
|
|
QFile agentConf(mDir.path() + QStringLiteral("/gpg-agent.conf"));
|
2017-01-11 15:20:31 +00:00
|
|
|
QVERIFY(agentConf.open(QIODevice::WriteOnly));
|
2016-08-09 12:10:15 +00:00
|
|
|
agentConf.write("allow-loopback-pinentry");
|
|
|
|
agentConf.close();
|
2017-01-11 15:20:31 +00:00
|
|
|
QVERIFY(copyKeyrings(gpgHome, mDir.path()));
|
2016-08-09 12:10:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
QTemporaryDir mDir;
|
|
|
|
};
|
|
|
|
|
|
|
|
QTEST_MAIN(EncryptionTest)
|
|
|
|
|
|
|
|
#include "t-encrypt.moc"
|