libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
ChildCloner.hxx
1#pragma once
2
3// C++
4#include <functional>
5#include <iosfwd>
6#include <optional>
7#include <string_view>
8
9// cosmos
10#include <cosmos/error/UsageError.hxx>
11#include <cosmos/fs/FileDescriptor.hxx>
12#include <cosmos/proc/Scheduler.hxx>
13#include <cosmos/proc/SubProc.hxx>
14#include <cosmos/string.hxx>
15
16namespace cosmos {
17
19
40class COSMOS_API ChildCloner {
41public: // types
42
44 typedef std::function<void (const ChildCloner&)> Callback;
45
46public: // functions
47
49 ChildCloner() = default;
50
52
57 explicit ChildCloner(const StringViewVector args) {
58 setArgsFromView(args);
59 }
60
62 bool hasExe() const { return !m_executable.empty(); }
63
65 auto& getExe() { return m_executable; }
66 const auto& getExe() const { return m_executable; }
67
69
73 void setExe(const std::string_view exe) {
74 m_executable = exe;
75 setArgv0();
76 }
77
79
85 const auto& getArgs() const { return m_argv; }
86
88 StringVector& getArgs() { return m_argv; }
89
91
95 void setArgs(const StringVector &sv) {
96 m_argv = sv;
97 setExeFromArgv0();
98 }
99
102 m_argv.clear();
103 for (const auto s: svv)
104 m_argv.push_back(std::string{s});
105 setExeFromArgv0();
106 }
107
109
113 void clearArgs() {
114 if (m_argv.size() > 1) {
115 m_argv.erase(m_argv.begin() + 1);
116 }
117 }
118
120
124 void setCWD(const std::string_view cwd) { m_cwd = cwd; }
125
127 const auto& getCWD() const { return m_cwd; }
128
130 void setInheritCWD() { m_cwd.clear(); }
131
133
141 void setEnv(const StringVector &vars) { m_env = vars; }
142
145 void setInheritEnv() { m_env.reset(); }
146
148
154 void setStdErr(FileDescriptor fd) { m_stderr = fd; }
156 void setStdOut(FileDescriptor fd) { m_stdout = fd; }
158 void setStdIn(FileDescriptor fd) { m_stdin = fd; }
159
161
185 if (fd.raw() <= FileNum::STDERR) {
186 cosmos_throw(UsageError{"added stdio or invalid FD as extra inherit FD"});
187 };
188 m_inherit_fds.push_back(fd);
189 }
190
192
198 m_stderr.reset();
199 m_stdin.reset();
200 m_stdout.reset();
201 }
202
204
209 template <typename SCHED_SETTING>
210 void setSchedulerSettings(const SCHED_SETTING &ss) { m_sched_settings = ss; };
211
213 void setInheritSchedulerSettings() { m_sched_settings.reset(); }
214
216
229 void setPostForkCB(Callback cb) { m_post_fork_cb = cb; }
230
232 void resetPostForkCB() { m_post_fork_cb = nullptr; }
233
235
247 SubProc run();
248
249protected: // functions
250
252 void postFork();
253
255 void resetSignals();
256
258
261 void redirectFD(FileDescriptor orig, FileDescriptor redirect);
262
264 void setArgv0() {
265 if (m_argv.empty())
266 m_argv.emplace_back(m_executable);
267 else
268 m_argv[0] = m_executable;
269 }
270
271 void setExeFromArgv0() {
272 if (m_argv.empty())
273 m_executable.clear();
274 else
275 m_executable = m_argv[0];
276 }
277
278protected: // data
279
281 std::string m_executable;
285 std::string m_cwd;
287 std::optional<StringVector> m_env;
289 std::optional<SchedulerSettingsVariant> m_sched_settings;
290
298 std::vector<FileDescriptor> m_inherit_fds;
299
300 Callback m_post_fork_cb = nullptr;
301
302 friend std::ostream& operator<<(std::ostream&, const ChildCloner&);
303};
304
305} // end ns
306
308
313inline cosmos::ChildCloner& operator<<(cosmos::ChildCloner &cloner, const std::string_view arg) {
314 if (!cloner.hasExe()) {
315 cloner.setExe(arg);
316 } else {
317 cloner.getArgs().push_back(std::string{arg});
318 }
319
320 return cloner;
321}
322
324COSMOS_API std::ostream& operator<<(std::ostream&, const cosmos::ChildCloner &);
Sub process creation facility.
void clearArgs()
Clears any currently set parameters.
StringVector m_argv
Argument vector including argv0 denoting the executable name (which can be different than m_executabl...
void addInheritFD(FileDescriptor fd)
Adds a file descriptor to inherit to the child process.
const auto & getCWD() const
Returns the currently set CWD for sub process execution.
void setArgs(const StringVector &sv)
Sets the argument vector to be used including argv0.
void resetPostForkCB()
Removes a previously stored post fork callback.
void setExe(const std::string_view exe)
Sets the path to the executable and argv0.
void setStdIn(FileDescriptor fd)
ChildCloner(const StringViewVector args)
Creates an instance configured with the provided arguments.
void setStdErr(FileDescriptor fd)
Redirect the child's stderr to the given file descriptor.
void resetStdFiles()
Restore the default inheritance behaviour for stdin/stderr/stdout.
void setInheritEnv()
Clears any previously set environment variables and let's to-be-started child processes inherit the p...
ChildCloner()=default
Creates an instance with default settings.
void setArgv0()
sets argv0 from the current executable name.
void setStdOut(FileDescriptor fd)
std::function< void(const ChildCloner &)> Callback
callback function type used in setPostForkCB().
void setEnv(const StringVector &vars)
Sets explicit environment variables for the child process.
std::string m_executable
Path to the child process executable to run.
void setPostForkCB(Callback cb)
Sets a callback function to be invoked in the child process context.
std::optional< StringVector > m_env
Explicit environment child environment variables, if any.
StringVector & getArgs()
std::string m_cwd
Path to an explicit working directory, if any.
std::optional< SchedulerSettingsVariant > m_sched_settings
Scheduler policy settings, if any.
FileDescriptor m_stderr
File descriptor to use as child's stderr.
const auto & getArgs() const
Returns the currently configured argument vector.
void setSchedulerSettings(const SCHED_SETTING &ss)
Sets scheduler type and settings.
FileDescriptor m_stdout
File descriptor to use as child's stdin.
void setInheritSchedulerSettings()
clear previously set scheduler settings and inherit them from the parent instead
void setArgsFromView(const StringViewVector &svv)
void setInheritCWD()
Clear a previously configured CWD and inherit it from the parent.
FileDescriptor m_stdin
File descriptor to use as child's stdin.
auto & getExe()
Returns the currently set executable name.
std::vector< FileDescriptor > m_inherit_fds
Additional file descriptors to inherit to the child process.
void setCWD(const std::string_view cwd)
Set an explicit working directory the child process.
bool hasExe() const
Returns whether currently an executable is set.
Thin Wrapper around OS file descriptors.
FileNum raw() const
Returns the primitive file descriptor contained in the object.
Represents a child process created via ChildCloner.
Definition SubProc.hxx:30
Exception type for logical usage errors within the application.
std::vector< std::string > StringVector
A vector of std::string.
Definition string.hxx:18
std::vector< std::string_view > StringViewVector
A vector of std::string_view.
Definition string.hxx:20