From 078fdba97cf70dc0fead71b3058ed966415120c2 Mon Sep 17 00:00:00 2001 From: Otto Sabart Date: Tue, 9 Aug 2022 09:00:00 +0200 Subject: [PATCH] plugins/bcli: load RPC password from stdin instead of an argument --- plugins/bcli.c | 25 ++++++++++++++++++++----- 1 file changed, 20 insertions(+), 5 deletions(-) diff --git a/plugins/bcli.c b/plugins/bcli.c index ea98050ffb9c..fa0ca0d1d460 100644 --- a/plugins/bcli.c +++ b/plugins/bcli.c @@ -121,8 +121,10 @@ static const char **gather_argsv(const tal_t *ctx, const char *cmd, va_list ap) if (bitcoind->rpcuser) add_arg(&args, tal_fmt(args, "-rpcuser=%s", bitcoind->rpcuser)); if (bitcoind->rpcpass) - add_arg(&args, - tal_fmt(args, "-rpcpassword=%s", bitcoind->rpcpass)); + // Always pipe the rpcpassword via stdin. Do not pass it using an + // `-rpcpassword` argument - secrets in arguments can leak when listing + // system processes. + add_arg(&args, "-stdinrpcpass"); add_arg(&args, cmd); while ((arg = va_arg(ap, char *)) != NULL) @@ -290,6 +292,7 @@ static void next_bcli(enum bitcoind_prio prio) { struct bitcoin_cli *bcli; struct io_conn *conn; + int in; if (bitcoind->num_requests[prio] >= BITCOIND_MAX_PARALLEL) return; @@ -298,8 +301,14 @@ static void next_bcli(enum bitcoind_prio prio) if (!bcli) return; - bcli->pid = pipecmdarr(NULL, &bcli->fd, &bcli->fd, + bcli->pid = pipecmdarr(&in, &bcli->fd, &bcli->fd, cast_const2(char **, bcli->args)); + + if (bitcoind->rpcpass) + write_all(in, bitcoind->rpcpass, strlen(bitcoind->rpcpass)); + + close(in); + if (bcli->pid < 0) plugin_err(bcli->cmd->plugin, "%s exec failed: %s", bcli->args[0], strerror(errno)); @@ -854,7 +863,7 @@ static void parse_getnetworkinfo_result(struct plugin *p, const char *buf) static void wait_and_check_bitcoind(struct plugin *p) { - int from, status, ret; + int in, from, status, ret; pid_t child; const char **cmd = gather_args(bitcoind, "getnetworkinfo", NULL); bool printed = false; @@ -863,7 +872,13 @@ static void wait_and_check_bitcoind(struct plugin *p) for (;;) { tal_free(output); - child = pipecmdarr(NULL, &from, &from, cast_const2(char **,cmd)); + child = pipecmdarr(&in, &from, &from, cast_const2(char **, cmd)); + + if (bitcoind->rpcpass) + write_all(in, bitcoind->rpcpass, strlen(bitcoind->rpcpass)); + + close(in); + if (child < 0) { if (errno == ENOENT) bitcoind_failure(p, "bitcoin-cli not found. Is bitcoin-cli "