/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 | } |