Bitcoin Core Fuzz Coverage Report

Coverage Report

Created: 2026-03-24 13:57

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/root/bitcoin/src/external_signer.cpp
Line
Count
Source
1
// Copyright (c) 2018-present The Bitcoin Core developers
2
// Distributed under the MIT software license, see the accompanying
3
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
4
5
#include <external_signer.h>
6
7
#include <chainparams.h>
8
#include <common/run_command.h>
9
#include <core_io.h>
10
#include <psbt.h>
11
#include <util/strencodings.h>
12
#include <util/subprocess.h>
13
14
#include <algorithm>
15
#include <stdexcept>
16
#include <string>
17
#include <vector>
18
19
ExternalSigner::ExternalSigner(std::vector<std::string> command, std::string chain, std::string fingerprint, std::string name)
20
0
    : m_command{std::move(command)}, m_chain{std::move(chain)}, m_fingerprint{std::move(fingerprint)}, m_name{std::move(name)} {}
21
22
std::vector<std::string> ExternalSigner::NetworkArg() const
23
0
{
24
0
    return {"--chain", m_chain};
25
0
}
26
27
bool ExternalSigner::Enumerate(const std::string& command, std::vector<ExternalSigner>& signers, const std::string& chain)
28
0
{
29
    // Call <command> enumerate
30
0
    std::vector<std::string> cmd_args = Cat(subprocess::util::split(command), {"enumerate"});
31
32
0
    const UniValue result = RunCommandParseJSON(cmd_args, "");
33
0
    if (!result.isArray()) {
34
0
        throw std::runtime_error(strprintf("'%s' received invalid response, expected array of signers", command));
Line
Count
Source
1172
0
#define strprintf tfm::format
35
0
    }
36
0
    for (const UniValue& signer : result.getValues()) {
37
        // Check for error
38
0
        const UniValue& error = signer.find_value("error");
39
0
        if (!error.isNull()) {
40
0
            if (!error.isStr()) {
41
0
                throw std::runtime_error(strprintf("'%s' error", command));
Line
Count
Source
1172
0
#define strprintf tfm::format
42
0
            }
43
0
            throw std::runtime_error(strprintf("'%s' error: %s", command, error.getValStr()));
Line
Count
Source
1172
0
#define strprintf tfm::format
44
0
        }
45
        // Check if fingerprint is present
46
0
        const UniValue& fingerprint = signer.find_value("fingerprint");
47
0
        if (fingerprint.isNull()) {
48
0
            throw std::runtime_error(strprintf("'%s' received invalid response, missing signer fingerprint", command));
Line
Count
Source
1172
0
#define strprintf tfm::format
49
0
        }
50
0
        const std::string& fingerprintStr{fingerprint.get_str()};
51
        // Skip duplicate signer
52
0
        bool duplicate = false;
53
0
        for (const ExternalSigner& signer : signers) {
54
0
            if (signer.m_fingerprint.compare(fingerprintStr) == 0) duplicate = true;
55
0
        }
56
0
        if (duplicate) break;
57
0
        std::string name;
58
0
        const UniValue& model_field = signer.find_value("model");
59
0
        if (model_field.isStr() && model_field.getValStr() != "") {
60
0
            name += model_field.getValStr();
61
0
        }
62
0
        signers.emplace_back(subprocess::util::split(command), chain, fingerprintStr, name);
63
0
    }
64
0
    return true;
65
0
}
66
67
UniValue ExternalSigner::DisplayAddress(const std::string& descriptor) const
68
0
{
69
0
    return RunCommandParseJSON(Cat(m_command, Cat(Cat({"--fingerprint", m_fingerprint}, NetworkArg()), {"displayaddress", "--desc", descriptor})), "");
70
0
}
71
72
UniValue ExternalSigner::GetDescriptors(const int account)
73
0
{
74
0
    return RunCommandParseJSON(Cat(m_command, Cat(Cat({"--fingerprint", m_fingerprint}, NetworkArg()), {"getdescriptors", "--account", strprintf("%d", account)})), "");
Line
Count
Source
1172
0
#define strprintf tfm::format
75
0
}
76
77
bool ExternalSigner::SignTransaction(PartiallySignedTransaction& psbtx, std::string& error)
78
0
{
79
    // Serialize the PSBT
80
0
    DataStream ssTx{};
81
0
    ssTx << psbtx;
82
    // parse ExternalSigner master fingerprint
83
0
    std::vector<unsigned char> parsed_m_fingerprint = ParseHex(m_fingerprint);
84
    // Check if signer fingerprint matches any input master key fingerprint
85
0
    auto matches_signer_fingerprint = [&](const PSBTInput& input) {
86
0
        for (const auto& entry : input.hd_keypaths) {
87
0
            if (std::ranges::equal(parsed_m_fingerprint, entry.second.fingerprint)) return true;
88
0
        }
89
0
        for (const auto& entry : input.m_tap_bip32_paths) {
90
0
            if (std::ranges::equal(parsed_m_fingerprint, entry.second.second.fingerprint)) return true;
91
0
        }
92
0
        return false;
93
0
    };
94
95
0
    if (!std::any_of(psbtx.inputs.begin(), psbtx.inputs.end(), matches_signer_fingerprint)) {
96
0
        error = "Signer fingerprint " + m_fingerprint + " does not match any of the inputs:\n" + EncodeBase64(ssTx.str());
97
0
        return false;
98
0
    }
99
100
0
    const std::vector<std::string> command = Cat(m_command, Cat({"--stdin", "--fingerprint", m_fingerprint}, NetworkArg()));
101
0
    const std::string stdinStr = "signtx " + EncodeBase64(ssTx.str());
102
103
0
    const UniValue signer_result = RunCommandParseJSON(command, stdinStr);
104
105
0
    if (signer_result.find_value("error").isStr()) {
106
0
        error = signer_result.find_value("error").get_str();
107
0
        return false;
108
0
    }
109
110
0
    if (!signer_result.find_value("psbt").isStr()) {
111
0
        error = "Unexpected result from signer";
112
0
        return false;
113
0
    }
114
115
0
    PartiallySignedTransaction signer_psbtx;
116
0
    std::string signer_psbt_error;
117
0
    if (!DecodeBase64PSBT(signer_psbtx, signer_result.find_value("psbt").get_str(), signer_psbt_error)) {
118
0
        error = strprintf("TX decode failed %s", signer_psbt_error);
Line
Count
Source
1172
0
#define strprintf tfm::format
119
0
        return false;
120
0
    }
121
122
0
    psbtx = signer_psbtx;
123
124
0
    return true;
125
0
}