mirror of
https://github.com/bitcoin/bitcoin.git
synced 2026-04-30 00:38:12 +02:00
Merge bitcoin/bitcoin#28981: Replace Boost.Process with cpp-subprocess
d5a715536ebuild: remove boost::process dependency for building external signer support (Sebastian Falbesoner)70434b1c44external_signer: replace boost::process with cpp-subprocess (Sebastian Falbesoner)cc8b9875b1Add `cpp-subprocess` header-only library (Hennadii Stepanov) Pull request description: Closes https://github.com/bitcoin/bitcoin/issues/24907. This PR is based on **theStack**'s [work](https://github.com/bitcoin/bitcoin/issues/24907#issuecomment-1466087049). The `subprocess.hpp` header has been sourced from the [upstream repo](https://github.com/arun11299/cpp-subprocess) with the only modification being the removal of convenience functions, which are not utilized in our codebase. Windows-related changes will be addressed in subsequent follow-ups. ACKs for top commit: achow101: reACKd5a715536eSjors: re-tACKd5a715536etheStack: Light re-ACKd5a715536efanquake: ACKd5a715536e- with the expectation that this code is going to be maintained as our own. Next PRs should: Tree-SHA512: d7fb6fecc3f5792496204190afb7d85b3e207b858fb1a75efe483c05260843b81b27d14b299323bb667c990e87a07197059afea3796cf218ed8b614086bd3611
This commit is contained in:
@@ -320,6 +320,7 @@ BITCOIN_CORE_H = \
|
||||
util/sock.h \
|
||||
util/spanparsing.h \
|
||||
util/string.h \
|
||||
util/subprocess.hpp \
|
||||
util/syserror.h \
|
||||
util/task_runner.h \
|
||||
util/thread.h \
|
||||
|
||||
@@ -12,39 +12,34 @@
|
||||
#include <univalue.h>
|
||||
|
||||
#ifdef ENABLE_EXTERNAL_SIGNER
|
||||
#include <boost/process.hpp>
|
||||
#include <util/subprocess.hpp>
|
||||
#endif // ENABLE_EXTERNAL_SIGNER
|
||||
|
||||
UniValue RunCommandParseJSON(const std::string& str_command, const std::string& str_std_in)
|
||||
{
|
||||
#ifdef ENABLE_EXTERNAL_SIGNER
|
||||
namespace bp = boost::process;
|
||||
namespace sp = subprocess;
|
||||
|
||||
UniValue result_json;
|
||||
bp::opstream stdin_stream;
|
||||
bp::ipstream stdout_stream;
|
||||
bp::ipstream stderr_stream;
|
||||
std::istringstream stdout_stream;
|
||||
std::istringstream stderr_stream;
|
||||
|
||||
if (str_command.empty()) return UniValue::VNULL;
|
||||
|
||||
bp::child c(
|
||||
str_command,
|
||||
bp::std_out > stdout_stream,
|
||||
bp::std_err > stderr_stream,
|
||||
bp::std_in < stdin_stream
|
||||
);
|
||||
auto c = sp::Popen(str_command, sp::input{sp::PIPE}, sp::output{sp::PIPE}, sp::error{sp::PIPE});
|
||||
if (!str_std_in.empty()) {
|
||||
stdin_stream << str_std_in << std::endl;
|
||||
c.send(str_std_in);
|
||||
}
|
||||
stdin_stream.pipe().close();
|
||||
auto [out_res, err_res] = c.communicate();
|
||||
stdout_stream.str(std::string{out_res.buf.begin(), out_res.buf.end()});
|
||||
stderr_stream.str(std::string{err_res.buf.begin(), err_res.buf.end()});
|
||||
|
||||
std::string result;
|
||||
std::string error;
|
||||
std::getline(stdout_stream, result);
|
||||
std::getline(stderr_stream, error);
|
||||
|
||||
c.wait();
|
||||
const int n_error = c.exit_code();
|
||||
const int n_error = c.retcode();
|
||||
if (n_error) throw std::runtime_error(strprintf("RunCommandParseJSON error: process(%s) returned %d: %s\n", str_command, n_error, error));
|
||||
if (!result_json.read(result)) throw std::runtime_error("Unable to parse JSON: " + result);
|
||||
|
||||
|
||||
@@ -62,12 +62,12 @@ bool ExternalSigner::Enumerate(const std::string& command, std::vector<ExternalS
|
||||
|
||||
UniValue ExternalSigner::DisplayAddress(const std::string& descriptor) const
|
||||
{
|
||||
return RunCommandParseJSON(m_command + " --fingerprint \"" + m_fingerprint + "\"" + NetworkArg() + " displayaddress --desc \"" + descriptor + "\"");
|
||||
return RunCommandParseJSON(m_command + " --fingerprint " + m_fingerprint + NetworkArg() + " displayaddress --desc " + descriptor);
|
||||
}
|
||||
|
||||
UniValue ExternalSigner::GetDescriptors(const int account)
|
||||
{
|
||||
return RunCommandParseJSON(m_command + " --fingerprint \"" + m_fingerprint + "\"" + NetworkArg() + " getdescriptors --account " + strprintf("%d", account));
|
||||
return RunCommandParseJSON(m_command + " --fingerprint " + m_fingerprint + NetworkArg() + " getdescriptors --account " + strprintf("%d", account));
|
||||
}
|
||||
|
||||
bool ExternalSigner::SignTransaction(PartiallySignedTransaction& psbtx, std::string& error)
|
||||
@@ -93,8 +93,8 @@ bool ExternalSigner::SignTransaction(PartiallySignedTransaction& psbtx, std::str
|
||||
return false;
|
||||
}
|
||||
|
||||
const std::string command = m_command + " --stdin --fingerprint \"" + m_fingerprint + "\"" + NetworkArg();
|
||||
const std::string stdinStr = "signtx \"" + EncodeBase64(ssTx.str()) + "\"";
|
||||
const std::string command = m_command + " --stdin --fingerprint " + m_fingerprint + NetworkArg();
|
||||
const std::string stdinStr = "signtx " + EncodeBase64(ssTx.str());
|
||||
|
||||
const UniValue signer_result = RunCommandParseJSON(command, stdinStr);
|
||||
|
||||
|
||||
@@ -11,7 +11,7 @@
|
||||
#include <univalue.h>
|
||||
|
||||
#ifdef ENABLE_EXTERNAL_SIGNER
|
||||
#include <boost/process.hpp>
|
||||
#include <util/subprocess.hpp>
|
||||
#endif // ENABLE_EXTERNAL_SIGNER
|
||||
|
||||
#include <boost/test/unit_test.hpp>
|
||||
@@ -34,20 +34,16 @@ BOOST_AUTO_TEST_CASE(run_command)
|
||||
BOOST_CHECK(result.isNull());
|
||||
}
|
||||
{
|
||||
const UniValue result = RunCommandParseJSON("echo \"{\"success\": true}\"");
|
||||
const UniValue result = RunCommandParseJSON("echo {\"success\": true}");
|
||||
BOOST_CHECK(result.isObject());
|
||||
const UniValue& success = result.find_value("success");
|
||||
BOOST_CHECK(!success.isNull());
|
||||
BOOST_CHECK_EQUAL(success.get_bool(), true);
|
||||
}
|
||||
{
|
||||
// An invalid command is handled by Boost
|
||||
const int expected_error{2};
|
||||
BOOST_CHECK_EXCEPTION(RunCommandParseJSON("invalid_command"), boost::process::process_error, [&](const boost::process::process_error& e) {
|
||||
BOOST_CHECK(std::string(e.what()).find("RunCommandParseJSON error:") == std::string::npos);
|
||||
BOOST_CHECK_EQUAL(e.code().value(), expected_error);
|
||||
return true;
|
||||
});
|
||||
// An invalid command is handled by cpp-subprocess
|
||||
const std::string expected{"execve failed: "};
|
||||
BOOST_CHECK_EXCEPTION(RunCommandParseJSON("invalid_command"), subprocess::CalledProcessError, HasReason(expected));
|
||||
}
|
||||
{
|
||||
// Return non-zero exit code, no output to stderr
|
||||
|
||||
1986
src/util/subprocess.hpp
Normal file
1986
src/util/subprocess.hpp
Normal file
File diff suppressed because it is too large
Load Diff
Reference in New Issue
Block a user