forked from qt-creator/qt-creator
Task-number: QTCREATORBUG-16628 Change-Id: I037acc37d8e3df88050a3853734c7f4c99e52b6b Reviewed-by: Tim Jenssen <tim.jenssen@qt.io> Reviewed-by: Orgad Shaneh <orgads@gmail.com>
173 lines
5.8 KiB
C++
173 lines
5.8 KiB
C++
/****************************************************************************
|
|
**
|
|
** Copyright (C) 2016 The Qt Company Ltd.
|
|
** Contact: https://www.qt.io/licensing/
|
|
**
|
|
** This file is part of Qt Creator.
|
|
**
|
|
** Commercial License Usage
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
** accordance with the commercial license agreement provided with the
|
|
** Software or, alternatively, in accordance with the terms contained in
|
|
** a written agreement between you and The Qt Company. For licensing terms
|
|
** and conditions see https://www.qt.io/terms-conditions. For further
|
|
** information use the contact form at https://www.qt.io/contact-us.
|
|
**
|
|
** GNU General Public License Usage
|
|
** Alternatively, this file may be used under the terms of the GNU
|
|
** General Public License version 3 as published by the Free Software
|
|
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
|
|
** included in the packaging of this file. Please review the following
|
|
** information to ensure the GNU General Public License requirements will
|
|
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
|
|
**
|
|
****************************************************************************/
|
|
|
|
#pragma once
|
|
|
|
#include "utils_global.h"
|
|
|
|
#include <QProcess>
|
|
#include <QSharedPointer>
|
|
#include <QTextCodec>
|
|
|
|
#include <functional>
|
|
|
|
QT_FORWARD_DECLARE_CLASS(QDebug)
|
|
|
|
namespace Utils {
|
|
|
|
class SynchronousProcessPrivate;
|
|
|
|
/* Result of SynchronousProcess execution */
|
|
class QTCREATOR_UTILS_EXPORT SynchronousProcessResponse
|
|
{
|
|
public:
|
|
enum Result {
|
|
// Finished with return code 0
|
|
Finished,
|
|
// Finished with return code != 0
|
|
FinishedError,
|
|
// Process terminated abnormally (kill)
|
|
TerminatedAbnormally,
|
|
// Executable could not be started
|
|
StartFailed,
|
|
// Hang, no output after time out
|
|
Hang };
|
|
|
|
void clear();
|
|
|
|
// Helper to format an exit message.
|
|
QString exitMessage(const QString &binary, int timeoutS) const;
|
|
|
|
QByteArray allRawOutput() const;
|
|
QString allOutput() const;
|
|
|
|
QString stdOut() const;
|
|
QString stdErr() const;
|
|
|
|
Result result = StartFailed;
|
|
int exitCode = -1;
|
|
|
|
QByteArray rawStdOut;
|
|
QByteArray rawStdErr;
|
|
QTextCodec *codec = QTextCodec::codecForLocale();
|
|
};
|
|
|
|
QTCREATOR_UTILS_EXPORT QDebug operator<<(QDebug str, const SynchronousProcessResponse &);
|
|
|
|
using ExitCodeInterpreter = std::function<SynchronousProcessResponse::Result(int /*exitCode*/)>;
|
|
QTCREATOR_UTILS_EXPORT SynchronousProcessResponse::Result defaultExitCodeInterpreter(int code);
|
|
|
|
class QTCREATOR_UTILS_EXPORT SynchronousProcess : public QObject
|
|
{
|
|
Q_OBJECT
|
|
public:
|
|
enum Flags {
|
|
// Unix: Do not give the child process a terminal for input prompting.
|
|
UnixTerminalDisabled = 0x1
|
|
};
|
|
|
|
SynchronousProcess();
|
|
virtual ~SynchronousProcess();
|
|
|
|
/* Timeout for hanging processes (triggers after no more output
|
|
* occurs on stderr/stdout). */
|
|
void setTimeoutS(int timeoutS);
|
|
int timeoutS() const;
|
|
|
|
void setCodec(QTextCodec *c);
|
|
QTextCodec *codec() const;
|
|
|
|
QProcess::ProcessChannelMode processChannelMode () const;
|
|
void setProcessChannelMode(QProcess::ProcessChannelMode m);
|
|
|
|
bool stdOutBufferedSignalsEnabled() const;
|
|
void setStdOutBufferedSignalsEnabled(bool);
|
|
|
|
bool stdErrBufferedSignalsEnabled() const;
|
|
void setStdErrBufferedSignalsEnabled(bool);
|
|
|
|
bool timeOutMessageBoxEnabled() const;
|
|
void setTimeOutMessageBoxEnabled(bool);
|
|
|
|
QStringList environment() const;
|
|
void setEnvironment(const QStringList &);
|
|
|
|
void setProcessEnvironment(const QProcessEnvironment &environment);
|
|
QProcessEnvironment processEnvironment() const;
|
|
|
|
void setWorkingDirectory(const QString &workingDirectory);
|
|
QString workingDirectory() const;
|
|
|
|
unsigned flags() const;
|
|
void setFlags(unsigned);
|
|
|
|
void setExitCodeInterpreter(const ExitCodeInterpreter &interpreter);
|
|
ExitCodeInterpreter exitCodeInterpreter() const;
|
|
|
|
// Starts an nested event loop and runs the binary with the arguments
|
|
SynchronousProcessResponse run(const QString &binary, const QStringList &args);
|
|
// Starts the binary with the arguments blocking the UI fully
|
|
SynchronousProcessResponse runBlocking(const QString &binary, const QStringList &args);
|
|
|
|
// Create a (derived) processes with flags applied.
|
|
static QSharedPointer<QProcess> createProcess(unsigned flags);
|
|
|
|
// Static helper for running a process synchronously in the foreground with timeout
|
|
// detection similar SynchronousProcess' handling (taking effect after no more output
|
|
// occurs on stderr/stdout as opposed to waitForFinished()). Returns false if a timeout
|
|
// occurs. Checking of the process' exit state/code still has to be done.
|
|
static bool readDataFromProcess(QProcess &p, int timeoutS,
|
|
QByteArray *rawStdOut = 0, QByteArray *rawStdErr = 0,
|
|
bool timeOutMessageBox = false);
|
|
// Stop a process by first calling terminate() (allowing for signal handling) and
|
|
// then kill().
|
|
static bool stopProcess(QProcess &p);
|
|
|
|
// Helpers to find binaries. Do not use it for other path variables
|
|
// and file types.
|
|
static QString locateBinary(const QString &binary);
|
|
static QString locateBinary(const QString &path, const QString &binary);
|
|
|
|
static QString normalizeNewlines(const QString &text);
|
|
|
|
signals:
|
|
void stdOutBuffered(const QString &lines, bool firstTime);
|
|
void stdErrBuffered(const QString &lines, bool firstTime);
|
|
|
|
public slots:
|
|
bool terminate();
|
|
|
|
private:
|
|
void slotTimeout();
|
|
void finished(int exitCode, QProcess::ExitStatus e);
|
|
void error(QProcess::ProcessError);
|
|
void processStdOut(bool emitSignals);
|
|
void processStdErr(bool emitSignals);
|
|
|
|
SynchronousProcessPrivate *d;
|
|
};
|
|
|
|
} // namespace Utils
|