/* Copyright (C) 2023 The gtkmm Development Team * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library. If not, see . */ _CONFIGINCLUDE(giommconfig.h) #include #include #include #include #include #include #include #include // std::pair _DEFS(giomm,gio) _PINCLUDE(glibmm/private/object_p.h) namespace Gio { /** Child processes. * * %Gio::Subprocess allows the creation of and interaction with child * processes. * * Processes can be communicated with using standard GIO-style APIs * (Gio::InputStream, Gio::OutputStream). There are GIO-style APIs to wait for * process termination (cancellable and with an asynchronous variant). * * There is an API to force a process to terminate, as well as a * race-free API for sending UNIX signals to a subprocess. * * One major advantage that GIO brings over the core GLib library is * comprehensive API for asynchronous I/O, such as * Gio::OutputStream::splice_async(). This makes %Gio::Subprocess * significantly more powerful and flexible than equivalent APIs in * some other languages such as the `subprocess.py` * included with Python. For example, using %Gio::Subprocess one could * create two child processes, reading standard output from the first, * processing it, and writing to the input stream of the second, all * without blocking the main loop. * * A powerful communicate() API is provided similar to the * `%communicate()` method of `subprocess.py`. This enables very easy * interaction with a subprocess that has been opened with pipes. * * %Gio::Subprocess defaults to tight control over the file descriptors open * in the child process, avoiding dangling-fd issues that are caused by * a simple fork()/exec(). The only open file descriptors in the * spawned process are ones that were explicitly specified by the * %Gio::Subprocess API (unless Gio::Subprocess::Flags::INHERIT_FDS was * specified). * * %Gio::Subprocess will quickly reap all child processes as they exit, * avoiding "zombie processes" remaining around for long periods of * time. wait() can be used to wait for this to happen, * but it will happen even without the call being explicitly made. * * As a matter of principle, %Gio::Subprocess has no API that accepts * shell-style space-separated strings. It will, however, match the * typical shell behaviour of searching the PATH for executables that do * not contain a directory separator in their name. By default, the `PATH` * of the current process is used. You can specify * Gio::Subprocess::Flags::SEARCH_PATH_FROM_ENVP to use the `PATH` of the * launcher environment instead. * * %Gio::Subprocess attempts to have a very simple API for most uses (ie: * spawning a subprocess with arguments and support for most typical * kinds of input and output redirection). See create(). The * Gio::SubprocessLauncher API is provided for more complicated cases * (advanced types of redirection, environment variable manipulation, * change of working directory, child setup functions, etc). * * A typical use of %Gio::Subprocess will involve calling create(), followed by * wait_async() or wait(). After the process exits, the status can be * checked using functions such as get_if_exited() (which are similar to * the familiar WIFEXITED-style POSIX macros). * * @see Gio::SubprocessLauncher * @newin{2,78} */ class GIOMM_API Subprocess : public Glib::Object, public Initable { _CLASS_GOBJECT(Subprocess, GSubprocess, G_SUBPROCESS, Glib::Object, GObject, , , GIOMM_API) _IMPLEMENTS_INTERFACE(Initable) public: _WRAP_ENUM(Flags, GSubprocessFlags, newin "2,78", decl_prefix GIOMM_API) protected: // Handwritten to ignore the final GError** parameter in the g_subprocess_newv() function. // It can throw, due to its call to Initable::init(). explicit Subprocess(const std::vector& argv, Flags flags = Flags::NONE); _IGNORE(g_subprocess_new, g_subprocess_newv) public: /** Create a new process with the given flags and argument list. * * @newin{2,78} * * @param argv Commandline arguments for the subprocess. * @param flags Flags that define the behaviour of the subprocess. * @return A newly created Subprocess. On error, an exception is thrown. * * @throws Glib::Error */ _WRAP_CREATE(const std::vector& argv, Flags flags = Flags::NONE) _WRAP_METHOD(Glib::RefPtr get_stdin_pipe(), g_subprocess_get_stdin_pipe, refreturn, newin "2,78") _WRAP_METHOD(Glib::RefPtr get_stdin_pipe() const, g_subprocess_get_stdin_pipe, refreturn, constversion, newin "2,78") _WRAP_METHOD(Glib::RefPtr get_stdout_pipe(), g_subprocess_get_stdout_pipe, refreturn, newin "2,78") _WRAP_METHOD(Glib::RefPtr get_stdout_pipe() const, g_subprocess_get_stdout_pipe, refreturn, constversion, newin "2,78") _WRAP_METHOD(Glib::RefPtr get_stderr_pipe(), g_subprocess_get_stderr_pipe, refreturn, newin "2,78") _WRAP_METHOD(Glib::RefPtr get_stderr_pipe() const, g_subprocess_get_stderr_pipe, refreturn, constversion, newin "2,78") _WRAP_METHOD(Glib::ustring get_identifier() const, g_subprocess_get_identifier, newin "2,78") _WRAP_METHOD(void send_signal(int signal_num), g_subprocess_send_signal, ifdef G_OS_UNIX, newin "2,78") _WRAP_METHOD(void force_exit(), g_subprocess_force_exit, newin "2,78") _WRAP_METHOD(void wait(const Glib::RefPtr& cancellable = {}) const, g_subprocess_wait, errthrow, newin "2,78") _WRAP_METHOD(void wait_async(const SlotAsyncReady& slot{callback}, const Glib::RefPtr& cancellable{.} = {}) const, g_subprocess_wait_async, slot_name slot, slot_callback giomm_SignalProxy_async_callback, newin "2,78") _WRAP_METHOD(void wait_finish(const Glib::RefPtr& result) const, g_subprocess_wait_finish, errthrow, newin "2,78") _WRAP_METHOD(void wait_check(const Glib::RefPtr& cancellable = {}) const, g_subprocess_wait_check, errthrow, newin "2,78") _WRAP_METHOD(void wait_check_async(const SlotAsyncReady& slot{callback}, const Glib::RefPtr& cancellable{.} = {}) const, g_subprocess_wait_check_async, slot_name slot, slot_callback giomm_SignalProxy_async_callback, newin "2,78") _WRAP_METHOD(void wait_check_finish(const Glib::RefPtr& result) const, g_subprocess_wait_check_finish, errthrow, newin "2,78") _WRAP_METHOD(int get_status() const, g_subprocess_get_status, newin "2,78") _WRAP_METHOD(bool get_successful() const, g_subprocess_get_successful, newin "2,78") _WRAP_METHOD(bool get_if_exited() const, g_subprocess_get_if_exited, newin "2,78") _WRAP_METHOD(int get_exit_status() const, g_subprocess_get_exit_status, newin "2,78") _WRAP_METHOD(bool get_if_signaled() const, g_subprocess_get_if_signaled, newin "2,78") _WRAP_METHOD(int get_term_sig() const, g_subprocess_get_term_sig, newin "2,78") /** Communicate with the subprocess until it terminates, and all input * and output has been completed. * * If @a stdin_buf is given, the subprocess must have been created with * Gio::Subprocess::Flags::STDIN_PIPE. The given data is fed to the * stdin of the subprocess and the pipe is closed (ie: EOF). * * At the same time (as not to cause blocking when dealing with large * amounts of data), if Gio::Subprocess::Flags::STDOUT_PIPE or * Gio::Subprocess::Flags::STDERR_PIPE were used, reads from those * streams. The data that was read is returned in @a stdout_buf and/or * the @a stderr_buf. * * If the subprocess was created with Gio::Subprocess::Flags::STDOUT_PIPE, * @a stdout_buf will contain the data read from stdout. Otherwise, for * subprocesses not created with Gio::Subprocess::Flags::STDOUT_PIPE, * @a stdout_buf will be set to an empty RefPtr. Similar provisions apply to * @a stderr_buf and Gio::Subprocess::Flags::STDERR_PIPE. * * If you desire the stdout and stderr data to be interleaved, create * the subprocess with Gio::Subprocess::Flags::STDOUT_PIPE and * Gio::Subprocess::Flags::STDERR_MERGE. The merged result will be returned * in @a stdout_buf, and @a stderr_buf will be set to an empty RefPtr. * * In case of any error (including cancellation), an exception will be thrown. * * After a normal return (no exception thrown), the subprocess has exited and the * exit status inspection APIs (eg: get_if_exited(), get_exit_status()) may be used. * * You should not attempt to use any of the subprocess pipes after * starting this function, since they may be left in strange states, * even if the operation was cancelled. You should especially not * attempt to interact with the pipes while the operation is in progress * (either from another thread or if using the asynchronous version). * * @newin{2,78} * * @param stdin_buf Data to send to the stdin of the subprocess, or an empty RefPtr. * @param cancellable A Cancellable. * @return {stdout_buf, stderr_buf} stdout data and stderr data. * Can be empty RefPtrs, if there are no data. * * @throws Glib::Error */ std::pair, Glib::RefPtr> communicate(const Glib::RefPtr& stdin_buf, const Glib::RefPtr& cancellable = {}); _IGNORE(g_subprocess_communicate) _WRAP_METHOD(void communicate_async(const Glib::RefPtr& stdin_buf{.}, const SlotAsyncReady& slot{callback}, const Glib::RefPtr& cancellable{.} = {}), g_subprocess_communicate_async, slot_name slot, slot_callback giomm_SignalProxy_async_callback, newin "2,78") /** Complete an invocation of communicate_async(). * * @newin{2,78} * * @param result Result. * @return {stdout_buf, stderr_buf} stdout data and stderr data. * Can be empty RefPtrs, if there are no data. * * @throws Glib::Error */ std::pair, Glib::RefPtr> communicate_finish(const Glib::RefPtr& result); _IGNORE(g_subprocess_communicate_finish) /** Like communicate(), but validates the output of the * process as UTF-8, and returns it as a regular Glib::ustring. * * On error, an exception is thrown. * * @newin{2,78} * * @param stdin_buf Data to send to the stdin of the subprocess, or an empty string. * @param cancellable A Cancellable. * @return {stdout_buf, stderr_buf} stdout data and stderr data. * * @throws Glib::Error */ std::pair communicate_utf8(const Glib::ustring& stdin_buf, const Glib::RefPtr& cancellable = {}); _IGNORE(g_subprocess_communicate_utf8) _WRAP_METHOD(void communicate_utf8_async(const Glib::ustring& stdin_buf{. NULL}, const SlotAsyncReady& slot{callback}, const Glib::RefPtr& cancellable{.} = {}), g_subprocess_communicate_utf8_async, slot_name slot, slot_callback giomm_SignalProxy_async_callback, newin "2,78") /** Complete an invocation of communicate_utf8_async(). * * @newin{2,78} * * @param result Result. * @return {stdout_buf, stderr_buf} stdout data and stderr data. * * @throws Glib::Error */ std::pair communicate_utf8_finish(const Glib::RefPtr& result); _IGNORE(g_subprocess_communicate_utf8_finish) // _IGNORE_PROPERTY("flags", "argv") // write-only, construct-only }; } // namespace Gio