Blame lang/qt/src/qgpgmerefreshkeysjob.cpp

Packit d7e8d0
/*
Packit d7e8d0
    qgpgmerefreshkeysjob.cpp
Packit d7e8d0
Packit d7e8d0
    This file is part of qgpgme, the Qt API binding for gpgme
Packit d7e8d0
    Copyright (c) 2004 Klarävdalens Datakonsult AB
Packit d7e8d0
    Copyright (c) 2016 by Bundesamt für Sicherheit in der Informationstechnik
Packit d7e8d0
    Software engineering by Intevation GmbH
Packit d7e8d0
Packit d7e8d0
    QGpgME is free software; you can redistribute it and/or
Packit d7e8d0
    modify it under the terms of the GNU General Public License as
Packit d7e8d0
    published by the Free Software Foundation; either version 2 of the
Packit d7e8d0
    License, or (at your option) any later version.
Packit d7e8d0
Packit d7e8d0
    QGpgME is distributed in the hope that it will be useful,
Packit d7e8d0
    but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit d7e8d0
    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit d7e8d0
    General Public License for more details.
Packit d7e8d0
Packit d7e8d0
    You should have received a copy of the GNU General Public License along
Packit d7e8d0
    with this program; if not, write to the Free Software Foundation, Inc.,
Packit d7e8d0
    51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
Packit d7e8d0
Packit d7e8d0
    In addition, as a special exception, the copyright holders give
Packit d7e8d0
    permission to link the code of this program with any edition of
Packit d7e8d0
    the Qt library by Trolltech AS, Norway (or with modified versions
Packit d7e8d0
    of Qt that use the same license as Qt), and distribute linked
Packit d7e8d0
    combinations including the two.  You must obey the GNU General
Packit d7e8d0
    Public License in all respects for all of the code used other than
Packit d7e8d0
    Qt.  If you modify this file, you may extend this exception to
Packit d7e8d0
    your version of the file, but you are not obligated to do so.  If
Packit d7e8d0
    you do not wish to do so, delete this exception statement from
Packit d7e8d0
    your version.
Packit d7e8d0
*/
Packit d7e8d0
Packit d7e8d0
#define MAX_CMD_LENGTH 32768
Packit d7e8d0
Packit d7e8d0
#ifdef HAVE_CONFIG_H
Packit d7e8d0
 #include "config.h"
Packit d7e8d0
#endif
Packit d7e8d0
Packit d7e8d0
#include "qgpgmerefreshkeysjob.h"
Packit d7e8d0
Packit d7e8d0
#include <QDebug>
Packit d7e8d0
#include "gpgme_backend_debug.h"
Packit d7e8d0
Packit d7e8d0
#include "context.h"
Packit d7e8d0
Packit d7e8d0
#include <QByteArray>
Packit d7e8d0
#include <QStringList>
Packit d7e8d0
Packit d7e8d0
#include <gpg-error.h>
Packit d7e8d0
Packit d7e8d0
#include <assert.h>
Packit d7e8d0
Packit d7e8d0
QGpgME::QGpgMERefreshKeysJob::QGpgMERefreshKeysJob()
Packit d7e8d0
    : RefreshKeysJob(0),
Packit d7e8d0
      mProcess(0),
Packit d7e8d0
      mError(0)
Packit d7e8d0
{
Packit d7e8d0
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
QGpgME::QGpgMERefreshKeysJob::~QGpgMERefreshKeysJob()
Packit d7e8d0
{
Packit d7e8d0
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
GpgME::Error QGpgME::QGpgMERefreshKeysJob::start(const QStringList &patterns)
Packit d7e8d0
{
Packit d7e8d0
    assert(mPatternsToDo.empty());
Packit d7e8d0
Packit d7e8d0
    mPatternsToDo = patterns;
Packit d7e8d0
    if (mPatternsToDo.empty()) {
Packit d7e8d0
        mPatternsToDo.push_back(QStringLiteral(" "));    // empty list means all -> mae
Packit d7e8d0
    }
Packit d7e8d0
    // sure to fail the first
Packit d7e8d0
    // startAProcess() guard clause
Packit d7e8d0
Packit d7e8d0
    return startAProcess();
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
#if MAX_CMD_LENGTH < 65 + 128
Packit d7e8d0
#error MAX_CMD_LENGTH is too low
Packit d7e8d0
#endif
Packit d7e8d0
Packit d7e8d0
GpgME::Error QGpgME::QGpgMERefreshKeysJob::startAProcess()
Packit d7e8d0
{
Packit d7e8d0
    if (mPatternsToDo.empty()) {
Packit d7e8d0
        return GpgME::Error();
Packit d7e8d0
    }
Packit d7e8d0
    // create and start gpgsm process:
Packit d7e8d0
    mProcess = new QProcess(this);
Packit d7e8d0
    mProcess->setObjectName(QStringLiteral("gpgsm -k --with-validation --force-crl-refresh --enable-crl-checks"));
Packit d7e8d0
Packit d7e8d0
    // FIXME: obbtain the path to gpgsm from gpgme, so we use the same instance.
Packit d7e8d0
    mProcess->setProgram(QStringLiteral("gpgsm"));
Packit d7e8d0
    QStringList arguments;
Packit d7e8d0
    arguments << QStringLiteral("-k")
Packit d7e8d0
              << QStringLiteral("--with-validation")
Packit d7e8d0
              << QStringLiteral("--force-crl-refresh")
Packit d7e8d0
              << QStringLiteral("--enable-crl-checks");
Packit d7e8d0
    unsigned int commandLineLength = MAX_CMD_LENGTH;
Packit d7e8d0
    commandLineLength -=
Packit d7e8d0
        strlen("gpgsm") + 1 + strlen("-k") + 1 +
Packit d7e8d0
        strlen("--with-validation") + 1 + strlen("--force-crl-refresh") + 1 +
Packit d7e8d0
        strlen("--enable-crl-checks") + 1;
Packit d7e8d0
    while (!mPatternsToDo.empty()) {
Packit d7e8d0
        const QByteArray pat = mPatternsToDo.front().toUtf8().trimmed();
Packit d7e8d0
        const unsigned int patLength = pat.length();
Packit d7e8d0
        if (patLength >= commandLineLength) {
Packit d7e8d0
            break;
Packit d7e8d0
        }
Packit d7e8d0
        mPatternsToDo.pop_front();
Packit d7e8d0
        if (pat.isEmpty()) {
Packit d7e8d0
            continue;
Packit d7e8d0
        }
Packit d7e8d0
        arguments << QLatin1String(pat);
Packit d7e8d0
        commandLineLength -= patLength + 1;
Packit d7e8d0
    }
Packit d7e8d0
Packit d7e8d0
    mProcess->setArguments(arguments);
Packit d7e8d0
Packit d7e8d0
    connect(mProcess, SIGNAL(finished(int,QProcess::ExitStatus)),
Packit d7e8d0
            SLOT(slotProcessExited(int,QProcess::ExitStatus)));
Packit d7e8d0
    connect(mProcess, SIGNAL(readyReadStandardOutput()),
Packit d7e8d0
            SLOT(slotStdout()));
Packit d7e8d0
    connect(mProcess, &QProcess::readyReadStandardError,
Packit d7e8d0
            this, &QGpgMERefreshKeysJob::slotStderr);
Packit d7e8d0
Packit d7e8d0
    mProcess->start();
Packit d7e8d0
    if (!mProcess->waitForStarted()) {
Packit d7e8d0
        mError = GpgME::Error::fromCode(GPG_ERR_ENOENT, GPG_ERR_SOURCE_GPGSM);   // what else?
Packit d7e8d0
        deleteLater();
Packit d7e8d0
        return mError;
Packit d7e8d0
    } else {
Packit d7e8d0
        return GpgME::Error();
Packit d7e8d0
    }
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
void QGpgME::QGpgMERefreshKeysJob::slotCancel()
Packit d7e8d0
{
Packit d7e8d0
    if (mProcess) {
Packit d7e8d0
        mProcess->kill();
Packit d7e8d0
    }
Packit d7e8d0
    mProcess = 0;
Packit d7e8d0
    mError = GpgME::Error::fromCode(GPG_ERR_CANCELED, GPG_ERR_SOURCE_GPGSM);
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
void QGpgME::QGpgMERefreshKeysJob::slotStatus(QProcess *proc, const QString &type, const QStringList &args)
Packit d7e8d0
{
Packit d7e8d0
    if (proc != mProcess) {
Packit d7e8d0
        return;
Packit d7e8d0
    }
Packit d7e8d0
    QStringList::const_iterator it = args.begin();
Packit d7e8d0
    bool ok = false;
Packit d7e8d0
Packit d7e8d0
    if (type == QLatin1String("ERROR")) {
Packit d7e8d0
Packit d7e8d0
        if (args.size() < 2) {
Packit d7e8d0
            qCDebug(GPGPME_BACKEND_LOG) << "not recognising ERROR with < 2 args!";
Packit d7e8d0
            return;
Packit d7e8d0
        }
Packit d7e8d0
        const int source = (*++it).toInt(&ok;;
Packit d7e8d0
        if (!ok) {
Packit d7e8d0
            qCDebug(GPGPME_BACKEND_LOG) << "expected number for first ERROR arg, got something else";
Packit d7e8d0
            return;
Packit d7e8d0
        }
Packit d7e8d0
        ok = false;
Packit d7e8d0
        const int code = (*++it).toInt(&ok;;
Packit d7e8d0
        if (!ok) {
Packit d7e8d0
            qCDebug(GPGPME_BACKEND_LOG) << "expected number for second ERROR arg, got something else";
Packit d7e8d0
            return;
Packit d7e8d0
        }
Packit d7e8d0
        mError = GpgME::Error::fromCode(code, source);
Packit d7e8d0
Packit d7e8d0
    } else if (type == QLatin1String("PROGRESS")) {
Packit d7e8d0
Packit d7e8d0
        if (args.size() < 4) {
Packit d7e8d0
            qCDebug(GPGPME_BACKEND_LOG) << "not recognising PROGRESS with < 4 args!";
Packit d7e8d0
            return;
Packit d7e8d0
        }
Packit d7e8d0
        const QString what = *++it;
Packit d7e8d0
        ok = false;
Packit d7e8d0
        (*++it).toInt(&ok;;
Packit d7e8d0
        if (!ok) {
Packit d7e8d0
            qCDebug(GPGPME_BACKEND_LOG) << "expected number for \"type\", got something else";
Packit d7e8d0
            return;
Packit d7e8d0
        }
Packit d7e8d0
        ok = false;
Packit d7e8d0
        const int cur = (*++it).toInt(&ok;;
Packit d7e8d0
        if (!ok) {
Packit d7e8d0
            qCDebug(GPGPME_BACKEND_LOG) << "expected number for \"cur\", got something else";
Packit d7e8d0
            return;
Packit d7e8d0
        }
Packit d7e8d0
        ok = false;
Packit d7e8d0
        const int total = (*++it).toInt(&ok;;
Packit d7e8d0
        if (!ok) {
Packit d7e8d0
            qCDebug(GPGPME_BACKEND_LOG) << "expected number for \"total\", got something else";
Packit d7e8d0
            return;
Packit d7e8d0
        }
Packit d7e8d0
        // TODO port
Packit d7e8d0
        Q_EMIT progress(QString(), cur, total);
Packit d7e8d0
Packit d7e8d0
    }
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
void QGpgME::QGpgMERefreshKeysJob::slotStderr()
Packit d7e8d0
{
Packit d7e8d0
    // implement? or not?
Packit d7e8d0
}
Packit d7e8d0
Packit d7e8d0
void QGpgME::QGpgMERefreshKeysJob::slotProcessExited(int exitCode, QProcess::ExitStatus exitStatus)
Packit d7e8d0
{
Packit d7e8d0
    if (!mError && !mPatternsToDo.empty()) {
Packit d7e8d0
        if (const GpgME::Error err = startAProcess()) {
Packit d7e8d0
            mError = err;
Packit d7e8d0
        } else {
Packit d7e8d0
            return;
Packit d7e8d0
        }
Packit d7e8d0
    }
Packit d7e8d0
Packit d7e8d0
    Q_EMIT done();
Packit d7e8d0
    if (!mError &&
Packit d7e8d0
            (exitStatus != QProcess::NormalExit || exitCode != 0)) {
Packit d7e8d0
        mError = GpgME::Error::fromCode(GPG_ERR_GENERAL, GPG_ERR_SOURCE_GPGSM);
Packit d7e8d0
    }
Packit d7e8d0
    Q_EMIT result(mError);
Packit d7e8d0
    deleteLater();
Packit d7e8d0
}
Packit d7e8d0
#include "qgpgmerefreshkeysjob.moc"