blob: 1d45a3f3bb22c175b7865468ecdcfe88ffe001e1 [file] [log] [blame]
/*
* Copyright (C) 2021 CESNET, https://photonics.cesnet.cz/
*
* Written by Václav Kubernát <kubernat@cesnet.cz>
*
*/
#include <boost/algorithm/string/join.hpp>
#include <boost/process.hpp>
#include <boost/process/extend.hpp>
#include "exec.h"
#include "log.h"
#include "system_vars.h"
std::string velia::utils::execAndWait(
velia::Log logger,
const std::string& absolutePath,
std::initializer_list<std::string> args,
std::string_view std_in,
const std::set<ExecOptions> opts)
{
namespace bp = boost::process;
bp::pipe stdinPipe;
bp::ipstream stdoutStream;
bp::ipstream stderrStream;
auto onExecSetup = [opts] (const auto&) {
if (opts.count(ExecOptions::DropRoot) == 1) {
if (getuid() == 0) {
if (setgid(NOBODY_GID) == -1) {
perror("couldn't drop root privileges");
exit(1);
}
if (setuid(NOBODY_UID) == -1) {
perror("couldn't drop root privileges");
exit(1);
}
}
}
};
logger->trace("exec: {} {}", absolutePath, boost::algorithm::join(args, " "));
bp::child c(
absolutePath,
boost::process::args=std::move(args),
bp::std_in < stdinPipe, bp::std_out > stdoutStream, bp::std_err > stderrStream,
bp::extend::on_exec_setup=onExecSetup);
stdinPipe.write(std_in.data(), std_in.size());
stdinPipe.close();
c.wait();
logger->trace("{} exited", absolutePath);
if (c.exit_code()) {
std::istreambuf_iterator<char> begin(stderrStream), end;
std::string stderrOutput(begin, end);
logger->critical("{} ended with a non-zero exit code. stderr: {}", absolutePath, stderrOutput);
throw std::runtime_error(absolutePath + " returned non-zero exit code " + std::to_string(c.exit_code()));
}
std::istreambuf_iterator<char> begin(stdoutStream), end;
return {begin, end};
}