kde-playground/kdepim/kleopatra/crypto/task.cpp
2015-04-14 21:49:29 +00:00

242 lines
6.8 KiB
C++

/* -*- mode: c++; c-basic-offset:4 -*-
crypto/task.cpp
This file is part of Kleopatra, the KDE keymanager
Copyright (c) 2007 Klarälvdalens Datakonsult AB
Kleopatra 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.
Kleopatra 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.
*/
#include <config-kleopatra.h>
#include "task.h"
#include "task_p.h"
#include <kleo/exception.h>
#include <utils/gnupg-helper.h>
#include <utils/auditlog.h>
#include <gpgme++/exception.h>
#include <gpg-error.h>
#include <KIcon>
#include <KIconLoader>
#include <KLocalizedString>
#include <QString>
#include <boost/bind.hpp>
using namespace Kleo;
using namespace Kleo::Crypto;
using namespace boost;
using namespace GpgME;
namespace {
class ErrorResult : public Task::Result {
public:
ErrorResult( int code, const QString & details )
: Task::Result(), m_code( code ), m_details( details ) {}
/* reimp */ QString overview() const { return makeOverview( m_details ); }
/* reimp */ QString details() const { return QString(); }
/* reimp */ int errorCode() const { return m_code; }
/* reimp */ QString errorString() const { return m_details; }
/* reimp */ VisualCode code() const { return NeutralError; }
/* reimp */ AuditLog auditLog() const { return AuditLog(); }
private:
int m_code;
QString m_details;
};
}
class Task::Private {
friend class ::Kleo::Crypto::Task;
Task * const q;
public:
explicit Private( Task * qq );
private:
QString m_progressLabel;
double m_processedPercent;
bool m_asciiArmor;
int m_id;
};
namespace {
static int nextTaskId = 0;
}
Task::Private::Private( Task * qq )
: q( qq ), m_progressLabel(), m_processedPercent( 0.0 ), m_asciiArmor( false ), m_id( nextTaskId++ )
{
}
Task::Task( QObject * p )
: QObject( p ), d( new Private( this ) )
{
}
Task::~Task() {}
void Task::setAsciiArmor( bool armor )
{
d->m_asciiArmor = armor;
}
bool Task::asciiArmor() const
{
return d->m_asciiArmor;
}
shared_ptr<Task> Task::makeErrorTask( int code, const QString & details, const QString & label ) {
const shared_ptr<SimpleTask> t( new SimpleTask( label ) );
t->setResult( t->makeErrorResult( code, details ) );
return t;
}
int Task::id() const
{
return d->m_id;
}
unsigned long long Task::processedSize() const
{
return qRound( d->m_processedPercent * totalSize() );
}
unsigned long long Task::totalSize() const
{
return inputSize();
}
QString Task::tag() const {
return QString();
}
QString Task::progressLabel() const
{
return d->m_progressLabel;
}
void Task::setProgress( const QString & label, int processed, int total )
{
const double percent = total > 0 ? static_cast<double>( processed ) / total : 0.0;
d->m_processedPercent = percent;
d->m_progressLabel = label;
emit progress( label, processedSize(), totalSize() );
}
void Task::start() {
try {
doStart();
} catch ( const Kleo::Exception & e ) {
QMetaObject::invokeMethod( this, "emitError", Qt::QueuedConnection, Q_ARG( int, e.error().encodedError() ), Q_ARG( QString, e.message() ) );
} catch ( const GpgME::Exception & e ) {
QMetaObject::invokeMethod( this, "emitError", Qt::QueuedConnection, Q_ARG( int, e.error().encodedError() ), Q_ARG( QString, QString::fromLocal8Bit( e.what() ) ) );
} catch ( const std::exception & e ) {
QMetaObject::invokeMethod( this, "emitError", Qt::QueuedConnection, Q_ARG( int, makeGnuPGError( GPG_ERR_UNEXPECTED ) ), Q_ARG( QString, QString::fromLocal8Bit( e.what() ) ) );
} catch ( ... ) {
QMetaObject::invokeMethod( this, "emitError", Qt::QueuedConnection, Q_ARG( int, makeGnuPGError( GPG_ERR_UNEXPECTED ) ), Q_ARG( QString, i18n( "Unknown exception in Task::start()") ) );
}
emit started();
}
void Task::emitError( int errCode, const QString& details ) {
emitResult( makeErrorResult( errCode, details ) );
}
void Task::emitResult( const shared_ptr<const Task::Result> & r )
{
d->m_processedPercent = 1.0;
emit progress( progressLabel(), processedSize(), totalSize() );
emit result( r );
}
shared_ptr<Task::Result> Task::makeErrorResult( int errCode, const QString& details )
{
return shared_ptr<Task::Result>( new ErrorResult( errCode, details ) );
}
static QString makeNonce() {
// ### make better
return QString::number( qrand(), 16 );
}
class Task::Result::Private {
public:
Private() {}
};
Task::Result::Result() : m_nonce( makeNonce() ), d( new Private() ) {}
Task::Result::~Result() {}
QString Task::Result::formatKeyLink( const char * fpr, const QString & content ) const {
return QLatin1String("<a href=\"key:") + m_nonce + QLatin1Char(':') + QLatin1String(fpr) + QLatin1String("\">") + content + QLatin1String("</a>");
}
bool Task::Result::hasError() const
{
return errorCode() != 0;
}
static QString image( const char* img ) {
// ### escape?
return KIconLoader::global()->iconPath( QLatin1String(img), KIconLoader::Small );
}
QString Task::Result::makeOverview( const QString& msg )
{
return QLatin1String("<b>") + msg + QLatin1String("</b>");
}
QString Task::Result::iconPath( VisualCode code )
{
switch ( code ) {
case Danger:
return image( "dialog-error" );
case AllGood:
return image( "dialog-ok" );
case Warning:
return image( "dialog-warning" );
case NeutralError:
case NeutralSuccess:
default:
return QString();
}
}
QString Task::Result::icon() const { return iconPath( code() ); }
#include "moc_task_p.cpp"