Page Menu
Home
GnuPG
Search
Configure Global Search
Log In
Files
F22947785
No One
Temporary
Actions
View File
Edit File
Delete File
View Transforms
Subscribe
Mute Notifications
Award Token
Size
21 KB
Subscribers
None
View Options
diff --git a/src/utils/output.cpp b/src/utils/output.cpp
index d00b7622e..dba3e2ae0 100644
--- a/src/utils/output.cpp
+++ b/src/utils/output.cpp
@@ -1,762 +1,762 @@
/* -*- mode: c++; c-basic-offset:4 -*-
utils/output.cpp
This file is part of Kleopatra, the KDE keymanager
SPDX-FileCopyrightText: 2007 Klarälvdalens Datakonsult AB
SPDX-License-Identifier: GPL-2.0-or-later
*/
#include <config-kleopatra.h>
#include "output.h"
#include "input_p.h"
#include "detail_p.h"
#include "kleo_assert.h"
#include "kdpipeiodevice.h"
#include "log.h"
#include "cached.h"
#include <Libkleo/KleoException>
#include <KLocalizedString>
#include <KMessageBox>
#include "kleopatra_debug.h"
#include <QFileInfo>
#include <QTemporaryFile>
#include <QString>
#include <QClipboard>
#include <QApplication>
#include <QBuffer>
#include <QPointer>
#include <QWidget>
#include <QDir>
#include <QProcess>
#include <QTimer>
#ifdef Q_OS_WIN
# include <windows.h>
#endif
#include <errno.h>
using namespace Kleo;
using namespace Kleo::_detail;
static const int PROCESS_MAX_RUNTIME_TIMEOUT = -1; // no timeout
static const int PROCESS_TERMINATE_TIMEOUT = 5 * 1000; // 5s
class OverwritePolicy::Private
{
public:
Private(QWidget *p, OverwritePolicy::Policy pol) : policy(pol), widget(p) {}
OverwritePolicy::Policy policy;
QWidget *widget;
};
OverwritePolicy::OverwritePolicy(QWidget *parent, Policy initialPolicy) : d(new Private(parent, initialPolicy))
{
}
OverwritePolicy::~OverwritePolicy() {}
OverwritePolicy::Policy OverwritePolicy::policy() const
{
return d->policy;
}
void OverwritePolicy::setPolicy(Policy policy)
{
d->policy = policy;
}
QWidget *OverwritePolicy::parentWidget() const
{
return d->widget;
}
namespace
{
class TemporaryFile : public QTemporaryFile
{
public:
explicit TemporaryFile() : QTemporaryFile() {}
explicit TemporaryFile(const QString &templateName) : QTemporaryFile(templateName) {}
explicit TemporaryFile(QObject *parent) : QTemporaryFile(parent) {}
explicit TemporaryFile(const QString &templateName, QObject *parent) : QTemporaryFile(templateName, parent) {}
void close() override {
if (isOpen())
{
m_oldFileName = fileName();
}
QTemporaryFile::close();
}
bool openNonInheritable()
{
if (!QTemporaryFile::open()) {
return false;
}
#if defined(Q_OS_WIN)
//QTemporaryFile (tested with 4.3.3) creates the file handle as inheritable.
//The handle is then inherited by gpgsm, which prevents deletion of the temp file
//in FileOutput::doFinalize()
//There are no inheritable handles under wince
return SetHandleInformation((HANDLE)_get_osfhandle(handle()), HANDLE_FLAG_INHERIT, 0);
#endif
return true;
}
QString oldFileName() const
{
return m_oldFileName;
}
private:
QString m_oldFileName;
};
template <typename T_IODevice>
struct inhibit_close : T_IODevice {
explicit inhibit_close() : T_IODevice() {}
template <typename T1>
explicit inhibit_close(T1 &t1) : T_IODevice(t1) {}
/* reimp */ void close() override {}
void reallyClose()
{
T_IODevice::close();
}
};
template <typename T_IODevice>
struct redirect_close : T_IODevice {
explicit redirect_close() : T_IODevice(), m_closed(false) {}
template <typename T1>
explicit redirect_close(T1 &t1) : T_IODevice(t1), m_closed(false) {}
/* reimp */ void close() override
{
this->closeWriteChannel();
m_closed = true;
}
bool isClosed() const
{
return m_closed;
}
private:
bool m_closed;
};
class FileOutput;
class OutputInput : public InputImplBase
{
public:
OutputInput(const std::shared_ptr<FileOutput> &output);
unsigned int classification() const override
{
return 0;
}
void outputFinalized()
{
if (!m_ioDevice->open(QIODevice::ReadOnly)) {
qCCritical(KLEOPATRA_LOG) << "Failed to open file for reading";
}
}
std::shared_ptr<QIODevice> ioDevice() const override
{
return m_ioDevice;
}
unsigned long long size() const override
{
return 0;
}
private:
std::shared_ptr<FileOutput> m_output;
mutable std::shared_ptr<QIODevice> m_ioDevice = nullptr;
};
class OutputImplBase : public Output
{
public:
OutputImplBase()
: Output(),
m_defaultLabel(),
m_customLabel(),
m_errorString(),
m_isFinalized(false),
m_isFinalizing(false),
m_cancelPending(false),
m_canceled(false),
m_binaryOpt(false)
{
}
QString label() const override
{
return m_customLabel.isEmpty() ? m_defaultLabel : m_customLabel;
}
void setLabel(const QString &label) override {
m_customLabel = label;
}
void setDefaultLabel(const QString &l)
{
m_defaultLabel = l;
}
void setBinaryOpt(bool value) override {
m_binaryOpt = value;
}
bool binaryOpt() const override
{
return m_binaryOpt;
}
QString errorString() const override
{
if (m_errorString.dirty()) {
m_errorString = doErrorString();
}
return m_errorString;
}
bool isFinalized() const override
{
return m_isFinalized;
}
void finalize() override {
qCDebug(KLEOPATRA_LOG) << this;
if (m_isFinalized || m_isFinalizing)
{
return;
}
m_isFinalizing = true;
try {
doFinalize();
} catch (...)
{
m_isFinalizing = false;
throw;
}
m_isFinalizing = false;
m_isFinalized = true;
if (m_cancelPending)
{
cancel();
}
}
void cancel() override {
qCDebug(KLEOPATRA_LOG) << this;
if (m_isFinalizing)
{
m_cancelPending = true;
} else if (!m_canceled)
{
m_isFinalizing = true;
try {
doCancel();
} catch (...) {}
m_isFinalizing = false;
m_isFinalized = false;
m_canceled = true;
}
}
private:
virtual QString doErrorString() const
{
if (std::shared_ptr<QIODevice> io = ioDevice()) {
return io->errorString();
} else {
return i18n("No output device");
}
}
virtual void doFinalize() = 0;
virtual void doCancel() = 0;
private:
QString m_defaultLabel;
QString m_customLabel;
mutable cached<QString> m_errorString;
bool m_isFinalized : 1;
bool m_isFinalizing : 1;
bool m_cancelPending : 1;
bool m_canceled : 1;
bool m_binaryOpt : 1;
};
class PipeOutput : public OutputImplBase
{
public:
explicit PipeOutput(assuan_fd_t fd);
std::shared_ptr<QIODevice> ioDevice() const override
{
return m_io;
}
void doFinalize() override {
m_io->reallyClose();
}
void doCancel() override {
doFinalize();
}
private:
std::shared_ptr< inhibit_close<KDPipeIODevice> > m_io;
};
class ProcessStdInOutput : public OutputImplBase
{
public:
explicit ProcessStdInOutput(const QString &cmd, const QStringList &args, const QDir &wd);
std::shared_ptr<QIODevice> ioDevice() const override
{
return m_proc;
}
void doFinalize() override {
/*
Make sure the data is written in the output here. If this
is not done the output will be written in small chunks
trough the eventloop causing an unnecessary delay before
the process has even a chance to work and finish.
This delay is mainly noticeable on Windows where it can
take ~30 seconds to write out a 10MB file in the 512 byte
chunks gpgme serves. */
qCDebug(KLEOPATRA_LOG) << "Waiting for " << m_proc->bytesToWrite()
<< " Bytes to be written";
while (m_proc->waitForBytesWritten(PROCESS_MAX_RUNTIME_TIMEOUT));
if (!m_proc->isClosed())
{
m_proc->close();
}
m_proc->waitForFinished(PROCESS_MAX_RUNTIME_TIMEOUT);
}
bool failed() const override
{
if (!m_proc) {
return false;
}
return !(m_proc->exitStatus() == QProcess::NormalExit && m_proc->exitCode() == 0);
}
void doCancel() override {
m_proc->terminate();
QTimer::singleShot(PROCESS_TERMINATE_TIMEOUT, m_proc.get(), &QProcess::kill);
}
QString label() const override;
private:
QString doErrorString() const override;
private:
const QString m_command;
const QStringList m_arguments;
const std::shared_ptr< redirect_close<QProcess> > m_proc;
};
class FileOutput : public OutputImplBase
{
public:
explicit FileOutput(const QString &fileName, const std::shared_ptr<OverwritePolicy> &policy);
~FileOutput()
{
qCDebug(KLEOPATRA_LOG) << this;
}
QString label() const override
{
return QFileInfo(m_fileName).fileName();
}
std::shared_ptr<QIODevice> ioDevice() const override
{
return m_tmpFile;
}
void doFinalize() override;
void doCancel() override {
qCDebug(KLEOPATRA_LOG) << this;
}
QString fileName() const
{
return m_fileName;
}
void attachInput(const std::shared_ptr<OutputInput> &input)
{
m_attachedInput = std::weak_ptr<OutputInput>(input);
}
private:
bool obtainOverwritePermission();
private:
const QString m_fileName;
std::shared_ptr< TemporaryFile > m_tmpFile;
const std::shared_ptr<OverwritePolicy> m_policy;
std::weak_ptr<OutputInput> m_attachedInput;
};
#ifndef QT_NO_CLIPBOARD
class ClipboardOutput : public OutputImplBase
{
public:
explicit ClipboardOutput(QClipboard::Mode mode);
QString label() const override;
std::shared_ptr<QIODevice> ioDevice() const override
{
return m_buffer;
}
void doFinalize() override;
void doCancel() override {}
private:
QString doErrorString() const override
{
return QString();
}
private:
const QClipboard::Mode m_mode;
std::shared_ptr<QBuffer> m_buffer;
};
#endif // QT_NO_CLIPBOARD
class ByteArrayOutput: public OutputImplBase
{
public:
explicit ByteArrayOutput(QByteArray *data):
m_buffer(std::shared_ptr<QBuffer>(new QBuffer(data)))
{
if (!m_buffer->open(QIODevice::WriteOnly))
throw Exception(gpg_error(GPG_ERR_EIO),
QStringLiteral("Could not open bytearray for writing?!"));
}
QString label() const override
{
return m_label;
}
void setLabel(const QString &label) override
{
m_label = label;
}
std::shared_ptr<QIODevice> ioDevice() const override
{
return m_buffer;
}
void doFinalize() override
{
m_buffer->close();
}
void doCancel() override
{
m_buffer->close();
}
private:
QString doErrorString() const override
{
return QString();
}
private:
QString m_label;
std::shared_ptr<QBuffer> m_buffer;
};
}
std::shared_ptr<Output> Output::createFromPipeDevice(assuan_fd_t fd, const QString &label)
{
std::shared_ptr<PipeOutput> po(new PipeOutput(fd));
po->setDefaultLabel(label);
return po;
}
PipeOutput::PipeOutput(assuan_fd_t fd)
: OutputImplBase(),
m_io(new inhibit_close<KDPipeIODevice>)
{
errno = 0;
if (!m_io->open(fd, QIODevice::WriteOnly))
throw Exception(errno ? gpg_error_from_errno(errno) : gpg_error(GPG_ERR_EIO),
i18n("Could not open FD %1 for writing",
assuanFD2int(fd)));
}
std::shared_ptr<Output> Output::createFromFile(const QString &fileName, bool forceOverwrite)
{
return createFromFile(fileName, std::shared_ptr<OverwritePolicy>(new OverwritePolicy(nullptr, forceOverwrite ? OverwritePolicy::Allow : OverwritePolicy::Deny)));
}
std::shared_ptr<Output> Output::createFromFile(const QString &fileName, const std::shared_ptr<OverwritePolicy> &policy)
{
std::shared_ptr<FileOutput> fo(new FileOutput(fileName, policy));
qCDebug(KLEOPATRA_LOG) << fo.get();
return fo;
}
FileOutput::FileOutput(const QString &fileName, const std::shared_ptr<OverwritePolicy> &policy)
: OutputImplBase(),
m_fileName(fileName),
m_tmpFile(new TemporaryFile(fileName)),
m_policy(policy)
{
Q_ASSERT(m_policy);
errno = 0;
if (!m_tmpFile->openNonInheritable())
throw Exception(errno ? gpg_error_from_errno(errno) : gpg_error(GPG_ERR_EIO),
i18n("Could not create temporary file for output \"%1\"", fileName));
}
bool FileOutput::obtainOverwritePermission()
{
if (m_policy->policy() != OverwritePolicy::Ask) {
return m_policy->policy() == OverwritePolicy::Allow;
}
const int sel = KMessageBox::questionYesNoCancel(m_policy->parentWidget(), i18n("The file <b>%1</b> already exists.\n"
"Overwrite?", m_fileName),
i18n("Overwrite Existing File?"),
KStandardGuiItem::overwrite(),
KGuiItem(i18n("Overwrite All")),
KStandardGuiItem::cancel());
if (sel == KMessageBox::No) { //Overwrite All
m_policy->setPolicy(OverwritePolicy::Allow);
}
return sel == KMessageBox::Yes || sel == KMessageBox::No;
}
void FileOutput::doFinalize()
{
qCDebug(KLEOPATRA_LOG) << this;
struct Remover {
QString file;
~Remover()
{
if (QFile::exists(file)) {
QFile::remove(file);
}
}
} remover;
kleo_assert(m_tmpFile);
if (m_tmpFile->isOpen()) {
m_tmpFile->close();
}
QString tmpFileName = remover.file = m_tmpFile->oldFileName();
m_tmpFile->setAutoRemove(false);
QPointer<QObject> guard = m_tmpFile.get();
m_tmpFile.reset(); // really close the file - needed on Windows for renaming :/
kleo_assert(!guard); // if this triggers, we need to audit for holder of std::shared_ptr<QIODevice>s.
const QFileInfo fi(tmpFileName);
bool qtbug83365_workaround = false;
if (!fi.exists()) {
/* QT Bug 83365 since qt 5.13 causes the filename of temporary files
* in UNC path name directories (unmounted samba shares) to start with
* UNC/ instead of the // that Qt can actually handle for things like
* rename and remove. So if we can't find our temporary file we try
* to workaround that bug. */
qCDebug(KLEOPATRA_LOG) << "failure to find " << tmpFileName;
if (tmpFileName.startsWith(QStringLiteral("UNC"))) {
tmpFileName.replace(QRegExp(QStringLiteral("^UNC")), QStringLiteral("/"));
qtbug83365_workaround = true;
}
const QFileInfo fi2(tmpFileName);
if (!fi2.exists()) {
throw Exception(gpg_error(GPG_ERR_EIO),
- QStringLiteral("Could not find temporary file \"%1\".").arg(m_fileName));
+ QStringLiteral("Could not find temporary file \"%1\".").arg(tmpFileName));
}
}
qCDebug(KLEOPATRA_LOG) << this << " renaming " << tmpFileName << "->" << m_fileName;
if (QFile::rename(tmpFileName, m_fileName)) {
qCDebug(KLEOPATRA_LOG) << this << "succeeded";
if (!m_attachedInput.expired()) {
m_attachedInput.lock()->outputFinalized();
}
return;
}
qCDebug(KLEOPATRA_LOG) << this << "failed";
if (!obtainOverwritePermission())
throw Exception(gpg_error(GPG_ERR_CANCELED),
i18n("Overwriting declined"));
qCDebug(KLEOPATRA_LOG) << this << "going to overwrite" << m_fileName;
if (!QFile::remove(m_fileName))
throw Exception(errno ? gpg_error_from_errno(errno) : gpg_error(GPG_ERR_EIO),
i18n("Could not remove file \"%1\" for overwriting.", m_fileName));
qCDebug(KLEOPATRA_LOG) << this << "succeeded, renaming " << tmpFileName << "->" << m_fileName;
if (QFile::rename(tmpFileName, m_fileName)) {
qCDebug(KLEOPATRA_LOG) << this << "succeeded";
if (!m_attachedInput.expired()) {
m_attachedInput.lock()->outputFinalized();
}
if (qtbug83365_workaround) {
QFile::remove(tmpFileName);
}
return;
}
qCDebug(KLEOPATRA_LOG) << this << "failed";
throw Exception(errno ? gpg_error_from_errno(errno) : gpg_error(GPG_ERR_EIO),
i18n("Could not rename file \"%1\" to \"%2\"",
tmpFileName, m_fileName));
}
std::shared_ptr<Output> Output::createFromProcessStdIn(const QString &command)
{
return std::shared_ptr<Output>(new ProcessStdInOutput(command, QStringList(), QDir::current()));
}
std::shared_ptr<Output> Output::createFromProcessStdIn(const QString &command, const QStringList &args)
{
return std::shared_ptr<Output>(new ProcessStdInOutput(command, args, QDir::current()));
}
std::shared_ptr<Output> Output::createFromProcessStdIn(const QString &command, const QStringList &args, const QDir &wd)
{
return std::shared_ptr<Output>(new ProcessStdInOutput(command, args, wd));
}
ProcessStdInOutput::ProcessStdInOutput(const QString &cmd, const QStringList &args, const QDir &wd)
: OutputImplBase(),
m_command(cmd),
m_arguments(args),
m_proc(new redirect_close<QProcess>)
{
qCDebug(KLEOPATRA_LOG) << "cd" << wd.absolutePath() << '\n' << cmd << args;
if (cmd.isEmpty())
throw Exception(gpg_error(GPG_ERR_INV_ARG),
i18n("Command not specified"));
m_proc->setWorkingDirectory(wd.absolutePath());
m_proc->start(cmd, args);
m_proc->setReadChannel(QProcess::StandardError);
if (!m_proc->waitForStarted())
throw Exception(gpg_error(GPG_ERR_EIO),
i18n("Could not start %1 process: %2", cmd, m_proc->errorString()));
}
QString ProcessStdInOutput::label() const
{
if (!m_proc) {
return OutputImplBase::label();
}
// output max. 3 arguments
const QString cmdline = (QStringList(m_command) + m_arguments.mid(0, 3)).join(QLatin1Char(' '));
if (m_arguments.size() > 3) {
return i18nc("e.g. \"Input to tar xf - file1 ...\"", "Input to %1 ...", cmdline);
} else {
return i18nc("e.g. \"Input to tar xf - file\"", "Input to %1", cmdline);
}
}
QString ProcessStdInOutput::doErrorString() const
{
kleo_assert(m_proc);
if (m_proc->exitStatus() == QProcess::NormalExit && m_proc->exitCode() == 0) {
return QString();
}
if (m_proc->error() == QProcess::UnknownError)
return i18n("Error while running %1: %2", m_command,
QString::fromLocal8Bit(m_proc->readAllStandardError().trimmed().constData()));
else {
return i18n("Failed to execute %1: %2", m_command, m_proc->errorString());
}
}
#ifndef QT_NO_CLIPBOARD
std::shared_ptr<Output> Output::createFromClipboard()
{
return std::shared_ptr<Output>(new ClipboardOutput(QClipboard::Clipboard));
}
ClipboardOutput::ClipboardOutput(QClipboard::Mode mode)
: OutputImplBase(),
m_mode(mode),
m_buffer(new QBuffer)
{
errno = 0;
if (!m_buffer->open(QIODevice::WriteOnly))
throw Exception(errno ? gpg_error_from_errno(errno) : gpg_error(GPG_ERR_EIO),
i18n("Could not write to clipboard"));
}
QString ClipboardOutput::label() const
{
switch (m_mode) {
case QClipboard::Clipboard:
return i18n("Clipboard");
case QClipboard::FindBuffer:
return i18n("Find buffer");
case QClipboard::Selection:
return i18n("Selection");
}
return QString();
}
void ClipboardOutput::doFinalize()
{
if (m_buffer->isOpen()) {
m_buffer->close();
}
if (QClipboard *const cb = QApplication::clipboard()) {
cb->setText(QString::fromUtf8(m_buffer->data()));
} else
throw Exception(gpg_error(GPG_ERR_EIO),
i18n("Could not find clipboard"));
}
#endif // QT_NO_CLIPBOARD
Output::~Output() {}
OutputInput::OutputInput(const std::shared_ptr<FileOutput> &output)
: m_output(output)
, m_ioDevice(new QFile(output->fileName()))
{
}
std::shared_ptr<Input> Input::createFromOutput(const std::shared_ptr<Output> &output)
{
if (auto fo = std::dynamic_pointer_cast<FileOutput>(output)) {
auto input = std::shared_ptr<OutputInput>(new OutputInput(fo));
fo->attachInput(input);
return input;
} else {
return {};
}
}
std::shared_ptr<Output> Output::createFromByteArray(QByteArray *data, const QString &label)
{
auto ret = std::shared_ptr<ByteArrayOutput>(new ByteArrayOutput(data));
ret->setLabel(label);
return ret;
}
File Metadata
Details
Attached
Mime Type
text/x-diff
Expires
Sat, May 10, 8:33 AM (1 d, 11 h)
Storage Engine
local-disk
Storage Format
Raw Data
Storage Handle
b9/f5/324f2c9740c2da19a5f09d2ebf63
Attached To
rKLEOPATRA Kleopatra
Event Timeline
Log In to Comment