forked from quickshell/quickshell
225 lines
8.4 KiB
C++
225 lines
8.4 KiB
C++
#pragma once
|
|
|
|
#include <qcontainerfwd.h>
|
|
#include <qobject.h>
|
|
#include <qprocess.h>
|
|
#include <qqmlintegration.h>
|
|
#include <qtclasshelpermacros.h>
|
|
#include <qtmetamacros.h>
|
|
#include <qtypes.h>
|
|
#include <qvariant.h>
|
|
|
|
#include "datastream.hpp"
|
|
|
|
///! Child process.
|
|
/// #### Example
|
|
/// ```qml
|
|
/// Process {
|
|
/// running: true
|
|
/// command: [ "some-command", "arg" ]
|
|
/// stdout: SplitParser {
|
|
/// onRead: data => console.log(`line read: ${data}`)
|
|
/// }
|
|
/// }
|
|
/// ```
|
|
class Process: public QObject {
|
|
Q_OBJECT;
|
|
// clang-format off
|
|
/// If the process is currently running. Defaults to false.
|
|
///
|
|
/// Setting this property to true will start the process if command has at least
|
|
/// one element.
|
|
/// Setting it to false will send SIGTERM. To immediately kill the process,
|
|
/// use @@signal() with SIGKILL. The process will be killed when
|
|
/// quickshell dies.
|
|
///
|
|
/// If you want to run the process in a loop, use the onRunningChanged signal handler
|
|
/// to restart the process.
|
|
/// ```qml
|
|
/// Process {
|
|
/// running: true
|
|
/// onRunningChanged: if (!running) running = true
|
|
/// }
|
|
/// ```
|
|
Q_PROPERTY(bool running READ isRunning WRITE setRunning NOTIFY runningChanged);
|
|
/// The process ID of the running process or `null` if @@running is false.
|
|
Q_PROPERTY(QVariant processId READ processId NOTIFY processIdChanged);
|
|
/// The command to execute. Each argument is its own string, which means you don't have
|
|
/// to deal with quoting anything.
|
|
///
|
|
/// If the process is already running changing this property will affect the next
|
|
/// started process. If the property has been changed after starting a process it will
|
|
/// return the new value, not the one for the currently running process.
|
|
///
|
|
/// > [!WARNING] This does not run command in a shell. All arguments to the command
|
|
/// > must be in separate values in the list, e.g. `["echo", "hello"]`
|
|
/// > and not `["echo hello"]`.
|
|
/// >
|
|
/// > Additionally, shell scripts must be run by your shell,
|
|
/// > e.g. `["sh", "script.sh"]` instead of `["script.sh"]` unless the script
|
|
/// > has a shebang.
|
|
///
|
|
/// > [!INFO] You can use `["sh", "-c", <your command>]` to execute your command with
|
|
/// > the system shell.
|
|
Q_PROPERTY(QList<QString> command READ command WRITE setCommand NOTIFY commandChanged);
|
|
/// The working directory of the process. Defaults to [quickshell's working directory].
|
|
///
|
|
/// If the process is already running changing this property will affect the next
|
|
/// started process. If the property has been changed after starting a process it will
|
|
/// return the new value, not the one for the currently running process.
|
|
///
|
|
/// [quickshell's working directory]: ../../quickshell/quickshell#prop.workingDirectory
|
|
Q_PROPERTY(QString workingDirectory READ workingDirectory WRITE setWorkingDirectory NOTIFY workingDirectoryChanged);
|
|
/// Environment of the executed process.
|
|
///
|
|
/// This is a javascript object (json). Environment variables can be added by setting
|
|
/// them to a string and removed by setting them to null (except when @@clearEnvironment is true,
|
|
/// in which case this behavior is inverted, see @@clearEnvironment for details).
|
|
///
|
|
///
|
|
/// ```qml
|
|
/// environment: ({
|
|
/// ADDED: "value",
|
|
/// REMOVED: null,
|
|
/// "i'm different": "value",
|
|
/// })
|
|
/// ```
|
|
///
|
|
/// > [!INFO] You need to wrap the returned object in () otherwise it won't parse due to javascript ambiguity.
|
|
///
|
|
/// If the process is already running changing this property will affect the next
|
|
/// started process. If the property has been changed after starting a process it will
|
|
/// return the new value, not the one for the currently running process.
|
|
Q_PROPERTY(QMap<QString, QVariant> environment READ environment WRITE setEnvironment NOTIFY environmentChanged);
|
|
/// If the process's environment should be cleared prior to applying @@environment.
|
|
/// Defaults to false.
|
|
///
|
|
/// If true, all environment variables will be removed before the @@environment
|
|
/// object is applied, meaning the variables listed will be the only ones visible to the process.
|
|
/// This changes the behavior of `null` to pass in the system value of the variable if present instead
|
|
/// of removing it.
|
|
///
|
|
/// ```qml
|
|
/// clearEnvironment: true
|
|
/// environment: ({
|
|
/// ADDED: "value",
|
|
/// PASSED_FROM_SYSTEM: null,
|
|
/// })
|
|
/// ```
|
|
///
|
|
/// If the process is already running changing this property will affect the next
|
|
/// started process. If the property has been changed after starting a process it will
|
|
/// return the new value, not the one for the currently running process.
|
|
Q_PROPERTY(bool clearEnvironment READ environmentCleared WRITE setEnvironmentCleared NOTIFY environmentClearChanged);
|
|
/// The parser for stdout. If the parser is null the process's stdout channel will be closed
|
|
/// and no further data will be read, even if a new parser is attached.
|
|
Q_PROPERTY(DataStreamParser* stdout READ stdoutParser WRITE setStdoutParser NOTIFY stdoutParserChanged);
|
|
/// The parser for stderr. If the parser is null the process's stdout channel will be closed
|
|
/// and no further data will be read, even if a new parser is attached.
|
|
Q_PROPERTY(DataStreamParser* stderr READ stderrParser WRITE setStderrParser NOTIFY stderrParserChanged);
|
|
/// If stdin is enabled. Defaults to false. If this property is false the process's stdin channel
|
|
/// will be closed and @@write() will do nothing, even if set back to true.
|
|
Q_PROPERTY(bool stdinEnabled READ stdinEnabled WRITE setStdinEnabled NOTIFY stdinEnabledChanged);
|
|
/// If the process should be killed when the Process object is destroyed or quickshell exits.
|
|
/// Defaults to true.
|
|
///
|
|
/// This property may be changed while the process is running and will affect it.
|
|
///
|
|
/// > [!WARNING] If set to false the process will still be killed if the quickshell config reloads.
|
|
/// > It will not be killed if quickshell exits normally or crashes.
|
|
Q_PROPERTY(bool manageLifetime READ isLifetimeManaged WRITE setLifetimeManaged NOTIFY lifetimeManagedChanged);
|
|
// clang-format on
|
|
QML_ELEMENT;
|
|
|
|
public:
|
|
explicit Process(QObject* parent = nullptr);
|
|
~Process() override;
|
|
Q_DISABLE_COPY_MOVE(Process);
|
|
|
|
/// Sends a signal to the process if @@running is true, otherwise does nothing.
|
|
Q_INVOKABLE void signal(qint32 signal);
|
|
|
|
/// Writes to the process's stdin. Does nothing if @@running is false.
|
|
Q_INVOKABLE void write(const QString& data);
|
|
|
|
[[nodiscard]] bool isRunning() const;
|
|
void setRunning(bool running);
|
|
|
|
[[nodiscard]] QVariant processId() const;
|
|
|
|
[[nodiscard]] QList<QString> command() const;
|
|
void setCommand(QList<QString> command);
|
|
|
|
[[nodiscard]] QString workingDirectory() const;
|
|
void setWorkingDirectory(const QString& workingDirectory);
|
|
|
|
[[nodiscard]] QMap<QString, QVariant> environment() const;
|
|
void setEnvironment(QMap<QString, QVariant> environment);
|
|
|
|
[[nodiscard]] bool environmentCleared() const;
|
|
void setEnvironmentCleared(bool cleared);
|
|
|
|
[[nodiscard]] DataStreamParser* stdoutParser() const;
|
|
void setStdoutParser(DataStreamParser* parser);
|
|
|
|
[[nodiscard]] DataStreamParser* stderrParser() const;
|
|
void setStderrParser(DataStreamParser* parser);
|
|
|
|
[[nodiscard]] bool stdinEnabled() const;
|
|
void setStdinEnabled(bool enabled);
|
|
|
|
[[nodiscard]] bool isLifetimeManaged() const;
|
|
void setLifetimeManaged(bool managed);
|
|
|
|
signals:
|
|
void started();
|
|
void exited(qint32 exitCode, QProcess::ExitStatus exitStatus);
|
|
|
|
void runningChanged();
|
|
void processIdChanged();
|
|
void commandChanged();
|
|
void workingDirectoryChanged();
|
|
void environmentChanged();
|
|
void environmentClearChanged();
|
|
void stdoutParserChanged();
|
|
void stderrParserChanged();
|
|
void stdinEnabledChanged();
|
|
void lifetimeManagedChanged();
|
|
|
|
private slots:
|
|
void onStarted();
|
|
void onFinished(qint32 exitCode, QProcess::ExitStatus exitStatus);
|
|
void onErrorOccurred(QProcess::ProcessError error);
|
|
void onStdoutReadyRead();
|
|
void onStderrReadyRead();
|
|
void onStdoutParserDestroyed();
|
|
void onStderrParserDestroyed();
|
|
void onGlobalWorkingDirectoryChanged();
|
|
|
|
private:
|
|
void startProcessIfReady();
|
|
|
|
QProcess* process = nullptr;
|
|
QList<QString> mCommand;
|
|
QString mWorkingDirectory;
|
|
QMap<QString, QVariant> mEnvironment;
|
|
DataStreamParser* mStdoutParser = nullptr;
|
|
DataStreamParser* mStderrParser = nullptr;
|
|
QByteArray stdoutBuffer;
|
|
QByteArray stderrBuffer;
|
|
|
|
bool targetRunning = false;
|
|
bool mStdinEnabled = false;
|
|
bool mClearEnvironment = false;
|
|
bool mLifetimeManaged = true;
|
|
};
|
|
|
|
class DisownedProcessContext: public QObject {
|
|
Q_OBJECT;
|
|
|
|
void reparent(QProcess* process);
|
|
friend class Process;
|
|
|
|
public:
|
|
static void destroyInstance();
|
|
};
|