Bitcoin Core 28.0.0
P2P Digital Currency
Loading...
Searching...
No Matches
util.cpp
Go to the documentation of this file.
1// Copyright (c) 2017-2022 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 <config/bitcoin-config.h> // IWYU pragma: keep
6
7#include <clientversion.h>
8#include <common/args.h>
9#include <common/messages.h>
10#include <common/types.h>
11#include <consensus/amount.h>
12#include <core_io.h>
13#include <key_io.h>
14#include <node/types.h>
15#include <outputtype.h>
16#include <rpc/util.h>
17#include <script/descriptor.h>
18#include <script/interpreter.h>
20#include <script/solver.h>
21#include <tinyformat.h>
22#include <univalue.h>
23#include <util/check.h>
24#include <util/result.h>
25#include <util/strencodings.h>
26#include <util/string.h>
27#include <util/translation.h>
28
29#include <algorithm>
30#include <iterator>
31#include <string_view>
32#include <tuple>
33#include <utility>
34
39using util::Join;
42
43const std::string UNIX_EPOCH_TIME = "UNIX epoch time";
44const std::string EXAMPLE_ADDRESS[2] = {"bc1q09vm5lfy0j5reeulh4x5752q25uqqvz34hufdl", "bc1q02ad21edsxd23d32dfgqqsz4vv4nmtfzuklhy3"};
45
46std::string GetAllOutputTypes()
47{
48 std::vector<std::string> ret;
49 using U = std::underlying_type<TxoutType>::type;
50 for (U i = (U)TxoutType::NONSTANDARD; i <= (U)TxoutType::WITNESS_UNKNOWN; ++i) {
51 ret.emplace_back(GetTxnOutputType(static_cast<TxoutType>(i)));
52 }
53 return Join(ret, ", ");
54}
55
57 const std::map<std::string, UniValueType>& typesExpected,
58 bool fAllowNull,
59 bool fStrict)
60{
61 for (const auto& t : typesExpected) {
62 const UniValue& v = o.find_value(t.first);
63 if (!fAllowNull && v.isNull())
64 throw JSONRPCError(RPC_TYPE_ERROR, strprintf("Missing %s", t.first));
65
66 if (!(t.second.typeAny || v.type() == t.second.type || (fAllowNull && v.isNull())))
67 throw JSONRPCError(RPC_TYPE_ERROR, strprintf("JSON value of type %s for field %s is not of expected type %s", uvTypeName(v.type()), t.first, uvTypeName(t.second.type)));
68 }
69
70 if (fStrict)
71 {
72 for (const std::string& k : o.getKeys())
73 {
74 if (typesExpected.count(k) == 0)
75 {
76 std::string err = strprintf("Unexpected key %s", k);
77 throw JSONRPCError(RPC_TYPE_ERROR, err);
78 }
79 }
80 }
81}
82
83CAmount AmountFromValue(const UniValue& value, int decimals)
84{
85 if (!value.isNum() && !value.isStr())
86 throw JSONRPCError(RPC_TYPE_ERROR, "Amount is not a number or string");
87 CAmount amount;
88 if (!ParseFixedPoint(value.getValStr(), decimals, &amount))
89 throw JSONRPCError(RPC_TYPE_ERROR, "Invalid amount");
90 if (!MoneyRange(amount))
91 throw JSONRPCError(RPC_TYPE_ERROR, "Amount out of range");
92 return amount;
93}
94
96{
97 CAmount val{AmountFromValue(json)};
98 if (val >= COIN) throw JSONRPCError(RPC_INVALID_PARAMETER, "Fee rates larger than or equal to 1BTC/kvB are not accepted");
99 return CFeeRate{val};
100}
101
102uint256 ParseHashV(const UniValue& v, std::string_view name)
103{
104 const std::string& strHex(v.get_str());
105 if (64 != strHex.length())
106 throw JSONRPCError(RPC_INVALID_PARAMETER, strprintf("%s must be of length %d (not %d, for '%s')", name, 64, strHex.length(), strHex));
107 if (!IsHex(strHex)) // Note: IsHex("") is false
108 throw JSONRPCError(RPC_INVALID_PARAMETER, strprintf("%s must be hexadecimal string (not '%s')", name, strHex));
109 return uint256S(strHex);
110}
111uint256 ParseHashO(const UniValue& o, std::string_view strKey)
112{
113 return ParseHashV(o.find_value(strKey), strKey);
114}
115std::vector<unsigned char> ParseHexV(const UniValue& v, std::string_view name)
116{
117 std::string strHex;
118 if (v.isStr())
119 strHex = v.get_str();
120 if (!IsHex(strHex))
121 throw JSONRPCError(RPC_INVALID_PARAMETER, strprintf("%s must be hexadecimal string (not '%s')", name, strHex));
122 return ParseHex(strHex);
123}
124std::vector<unsigned char> ParseHexO(const UniValue& o, std::string_view strKey)
125{
126 return ParseHexV(o.find_value(strKey), strKey);
127}
128
129namespace {
130
136std::string ShellQuote(const std::string& s)
137{
138 std::string result;
139 result.reserve(s.size() * 2);
140 for (const char ch: s) {
141 if (ch == '\'') {
142 result += "'\''";
143 } else {
144 result += ch;
145 }
146 }
147 return "'" + result + "'";
148}
149
155std::string ShellQuoteIfNeeded(const std::string& s)
156{
157 for (const char ch: s) {
158 if (ch == ' ' || ch == '\'' || ch == '"') {
159 return ShellQuote(s);
160 }
161 }
162
163 return s;
164}
165
166}
167
168std::string HelpExampleCli(const std::string& methodname, const std::string& args)
169{
170 return "> bitcoin-cli " + methodname + " " + args + "\n";
171}
172
173std::string HelpExampleCliNamed(const std::string& methodname, const RPCArgList& args)
174{
175 std::string result = "> bitcoin-cli -named " + methodname;
176 for (const auto& argpair: args) {
177 const auto& value = argpair.second.isStr()
178 ? argpair.second.get_str()
179 : argpair.second.write();
180 result += " " + argpair.first + "=" + ShellQuoteIfNeeded(value);
181 }
182 result += "\n";
183 return result;
184}
185
186std::string HelpExampleRpc(const std::string& methodname, const std::string& args)
187{
188 return "> curl --user myusername --data-binary '{\"jsonrpc\": \"2.0\", \"id\": \"curltest\", "
189 "\"method\": \"" + methodname + "\", \"params\": [" + args + "]}' -H 'content-type: application/json' http://127.0.0.1:8332/\n";
190}
191
192std::string HelpExampleRpcNamed(const std::string& methodname, const RPCArgList& args)
193{
194 UniValue params(UniValue::VOBJ);
195 for (const auto& param: args) {
196 params.pushKV(param.first, param.second);
197 }
198
199 return "> curl --user myusername --data-binary '{\"jsonrpc\": \"2.0\", \"id\": \"curltest\", "
200 "\"method\": \"" + methodname + "\", \"params\": " + params.write() + "}' -H 'content-type: application/json' http://127.0.0.1:8332/\n";
201}
202
203// Converts a hex string to a public key if possible
204CPubKey HexToPubKey(const std::string& hex_in)
205{
206 if (!IsHex(hex_in)) {
207 throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, "Pubkey \"" + hex_in + "\" must be a hex string");
208 }
209 if (hex_in.length() != 66 && hex_in.length() != 130) {
210 throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, "Pubkey \"" + hex_in + "\" must have a length of either 33 or 65 bytes");
211 }
212 CPubKey vchPubKey(ParseHex(hex_in));
213 if (!vchPubKey.IsFullyValid()) {
214 throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, "Pubkey \"" + hex_in + "\" must be cryptographically valid.");
215 }
216 return vchPubKey;
217}
218
219// Retrieves a public key for an address from the given FillableSigningProvider
220CPubKey AddrToPubKey(const FillableSigningProvider& keystore, const std::string& addr_in)
221{
222 CTxDestination dest = DecodeDestination(addr_in);
223 if (!IsValidDestination(dest)) {
224 throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, "Invalid address: " + addr_in);
225 }
226 CKeyID key = GetKeyForDestination(keystore, dest);
227 if (key.IsNull()) {
228 throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, strprintf("'%s' does not refer to a key", addr_in));
229 }
230 CPubKey vchPubKey;
231 if (!keystore.GetPubKey(key, vchPubKey)) {
232 throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, strprintf("no full public key for address %s", addr_in));
233 }
234 if (!vchPubKey.IsFullyValid()) {
235 throw JSONRPCError(RPC_INTERNAL_ERROR, "Wallet contains an invalid public key");
236 }
237 return vchPubKey;
238}
239
240// Creates a multisig address from a given list of public keys, number of signatures required, and the address type
241CTxDestination AddAndGetMultisigDestination(const int required, const std::vector<CPubKey>& pubkeys, OutputType type, FlatSigningProvider& keystore, CScript& script_out)
242{
243 // Gather public keys
244 if (required < 1) {
245 throw JSONRPCError(RPC_INVALID_PARAMETER, "a multisignature address must require at least one key to redeem");
246 }
247 if ((int)pubkeys.size() < required) {
248 throw JSONRPCError(RPC_INVALID_PARAMETER, strprintf("not enough keys supplied (got %u keys, but need at least %d to redeem)", pubkeys.size(), required));
249 }
250 if (pubkeys.size() > MAX_PUBKEYS_PER_MULTISIG) {
251 throw JSONRPCError(RPC_INVALID_PARAMETER, strprintf("Number of keys involved in the multisignature address creation > %d\nReduce the number", MAX_PUBKEYS_PER_MULTISIG));
252 }
253
254 script_out = GetScriptForMultisig(required, pubkeys);
255
256 // Check if any keys are uncompressed. If so, the type is legacy
257 for (const CPubKey& pk : pubkeys) {
258 if (!pk.IsCompressed()) {
259 type = OutputType::LEGACY;
260 break;
261 }
262 }
263
264 if (type == OutputType::LEGACY && script_out.size() > MAX_SCRIPT_ELEMENT_SIZE) {
265 throw JSONRPCError(RPC_INVALID_PARAMETER, (strprintf("redeemScript exceeds size limit: %d > %d", script_out.size(), MAX_SCRIPT_ELEMENT_SIZE)));
266 }
267
268 // Make the address
269 CTxDestination dest = AddAndGetDestinationForScript(keystore, script_out, type);
270
271 return dest;
272}
273
275{
276public:
277 explicit DescribeAddressVisitor() = default;
278
280 {
281 return UniValue(UniValue::VOBJ);
282 }
283
285 {
286 return UniValue(UniValue::VOBJ);
287 }
288
289 UniValue operator()(const PKHash& keyID) const
290 {
292 obj.pushKV("isscript", false);
293 obj.pushKV("iswitness", false);
294 return obj;
295 }
296
297 UniValue operator()(const ScriptHash& scriptID) const
298 {
300 obj.pushKV("isscript", true);
301 obj.pushKV("iswitness", false);
302 return obj;
303 }
304
306 {
308 obj.pushKV("isscript", false);
309 obj.pushKV("iswitness", true);
310 obj.pushKV("witness_version", 0);
311 obj.pushKV("witness_program", HexStr(id));
312 return obj;
313 }
314
316 {
318 obj.pushKV("isscript", true);
319 obj.pushKV("iswitness", true);
320 obj.pushKV("witness_version", 0);
321 obj.pushKV("witness_program", HexStr(id));
322 return obj;
323 }
324
326 {
328 obj.pushKV("isscript", true);
329 obj.pushKV("iswitness", true);
330 obj.pushKV("witness_version", 1);
331 obj.pushKV("witness_program", HexStr(tap));
332 return obj;
333 }
334
335 UniValue operator()(const PayToAnchor& anchor) const
336 {
338 obj.pushKV("isscript", true);
339 obj.pushKV("iswitness", true);
340 return obj;
341 }
342
344 {
346 obj.pushKV("iswitness", true);
347 obj.pushKV("witness_version", id.GetWitnessVersion());
348 obj.pushKV("witness_program", HexStr(id.GetWitnessProgram()));
349 return obj;
350 }
351};
352
354{
355 return std::visit(DescribeAddressVisitor(), dest);
356}
357
363int ParseSighashString(const UniValue& sighash)
364{
365 if (sighash.isNull()) {
366 return SIGHASH_DEFAULT;
367 }
368 const auto result{SighashFromStr(sighash.get_str())};
369 if (!result) {
371 }
372 return result.value();
373}
374
375unsigned int ParseConfirmTarget(const UniValue& value, unsigned int max_target)
376{
377 const int target{value.getInt<int>()};
378 const unsigned int unsigned_target{static_cast<unsigned int>(target)};
379 if (target < 1 || unsigned_target > max_target) {
380 throw JSONRPCError(RPC_INVALID_PARAMETER, strprintf("Invalid conf_target, must be between %u and %u", 1, max_target));
381 }
382 return unsigned_target;
383}
384
386{
387 switch (err) {
392 default: break;
393 }
395}
396
408
410{
411 return JSONRPCError(RPCErrorFromPSBTError(err), PSBTErrorString(err).original);
412}
413
414UniValue JSONRPCTransactionError(TransactionError terr, const std::string& err_string)
415{
416 if (err_string.length() > 0) {
417 return JSONRPCError(RPCErrorFromTransactionError(terr), err_string);
418 } else {
419 return JSONRPCError(RPCErrorFromTransactionError(terr), TransactionErrorString(terr).original);
420 }
421}
422
427struct Section {
428 Section(const std::string& left, const std::string& right)
429 : m_left{left}, m_right{right} {}
430 std::string m_left;
431 const std::string m_right;
432};
433
438struct Sections {
439 std::vector<Section> m_sections;
440 size_t m_max_pad{0};
441
442 void PushSection(const Section& s)
443 {
444 m_max_pad = std::max(m_max_pad, s.m_left.size());
445 m_sections.push_back(s);
446 }
447
451 // NOLINTNEXTLINE(misc-no-recursion)
452 void Push(const RPCArg& arg, const size_t current_indent = 5, const OuterType outer_type = OuterType::NONE)
453 {
454 const auto indent = std::string(current_indent, ' ');
455 const auto indent_next = std::string(current_indent + 2, ' ');
456 const bool push_name{outer_type == OuterType::OBJ}; // Dictionary keys must have a name
457 const bool is_top_level_arg{outer_type == OuterType::NONE}; // True on the first recursion
458
459 switch (arg.m_type) {
467 if (is_top_level_arg) return; // Nothing more to do for non-recursive types on first recursion
468 auto left = indent;
469 if (arg.m_opts.type_str.size() != 0 && push_name) {
470 left += "\"" + arg.GetName() + "\": " + arg.m_opts.type_str.at(0);
471 } else {
472 left += push_name ? arg.ToStringObj(/*oneline=*/false) : arg.ToString(/*oneline=*/false);
473 }
474 left += ",";
475 PushSection({left, arg.ToDescriptionString(/*is_named_arg=*/push_name)});
476 break;
477 }
480 const auto right = is_top_level_arg ? "" : arg.ToDescriptionString(/*is_named_arg=*/push_name);
481 PushSection({indent + (push_name ? "\"" + arg.GetName() + "\": " : "") + "{", right});
482 for (const auto& arg_inner : arg.m_inner) {
483 Push(arg_inner, current_indent + 2, OuterType::OBJ);
484 }
485 if (arg.m_type != RPCArg::Type::OBJ) {
486 PushSection({indent_next + "...", ""});
487 }
488 PushSection({indent + "}" + (is_top_level_arg ? "" : ","), ""});
489 break;
490 }
491 case RPCArg::Type::ARR: {
492 auto left = indent;
493 left += push_name ? "\"" + arg.GetName() + "\": " : "";
494 left += "[";
495 const auto right = is_top_level_arg ? "" : arg.ToDescriptionString(/*is_named_arg=*/push_name);
496 PushSection({left, right});
497 for (const auto& arg_inner : arg.m_inner) {
498 Push(arg_inner, current_indent + 2, OuterType::ARR);
499 }
500 PushSection({indent_next + "...", ""});
501 PushSection({indent + "]" + (is_top_level_arg ? "" : ","), ""});
502 break;
503 }
504 } // no default case, so the compiler can warn about missing cases
505 }
506
510 std::string ToString() const
511 {
512 std::string ret;
513 const size_t pad = m_max_pad + 4;
514 for (const auto& s : m_sections) {
515 // The left part of a section is assumed to be a single line, usually it is the name of the JSON struct or a
516 // brace like {, }, [, or ]
517 CHECK_NONFATAL(s.m_left.find('\n') == std::string::npos);
518 if (s.m_right.empty()) {
519 ret += s.m_left;
520 ret += "\n";
521 continue;
522 }
523
524 std::string left = s.m_left;
525 left.resize(pad, ' ');
526 ret += left;
527
528 // Properly pad after newlines
529 std::string right;
530 size_t begin = 0;
531 size_t new_line_pos = s.m_right.find_first_of('\n');
532 while (true) {
533 right += s.m_right.substr(begin, new_line_pos - begin);
534 if (new_line_pos == std::string::npos) {
535 break; //No new line
536 }
537 right += "\n" + std::string(pad, ' ');
538 begin = s.m_right.find_first_not_of(' ', new_line_pos + 1);
539 if (begin == std::string::npos) {
540 break; // Empty line
541 }
542 new_line_pos = s.m_right.find_first_of('\n', begin + 1);
543 }
544 ret += right;
545 ret += "\n";
546 }
547 return ret;
548 }
549};
550
551RPCHelpMan::RPCHelpMan(std::string name, std::string description, std::vector<RPCArg> args, RPCResults results, RPCExamples examples)
552 : RPCHelpMan{std::move(name), std::move(description), std::move(args), std::move(results), std::move(examples), nullptr} {}
553
554RPCHelpMan::RPCHelpMan(std::string name, std::string description, std::vector<RPCArg> args, RPCResults results, RPCExamples examples, RPCMethodImpl fun)
555 : m_name{std::move(name)},
556 m_fun{std::move(fun)},
557 m_description{std::move(description)},
558 m_args{std::move(args)},
559 m_results{std::move(results)},
560 m_examples{std::move(examples)}
561{
562 // Map of parameter names and types just used to check whether the names are
563 // unique. Parameter names always need to be unique, with the exception that
564 // there can be pairs of POSITIONAL and NAMED parameters with the same name.
565 enum ParamType { POSITIONAL = 1, NAMED = 2, NAMED_ONLY = 4 };
566 std::map<std::string, int> param_names;
567
568 for (const auto& arg : m_args) {
569 std::vector<std::string> names = SplitString(arg.m_names, '|');
570 // Should have unique named arguments
571 for (const std::string& name : names) {
572 auto& param_type = param_names[name];
573 CHECK_NONFATAL(!(param_type & POSITIONAL));
574 CHECK_NONFATAL(!(param_type & NAMED_ONLY));
575 param_type |= POSITIONAL;
576 }
577 if (arg.m_type == RPCArg::Type::OBJ_NAMED_PARAMS) {
578 for (const auto& inner : arg.m_inner) {
579 std::vector<std::string> inner_names = SplitString(inner.m_names, '|');
580 for (const std::string& inner_name : inner_names) {
581 auto& param_type = param_names[inner_name];
582 CHECK_NONFATAL(!(param_type & POSITIONAL) || inner.m_opts.also_positional);
583 CHECK_NONFATAL(!(param_type & NAMED));
584 CHECK_NONFATAL(!(param_type & NAMED_ONLY));
585 param_type |= inner.m_opts.also_positional ? NAMED : NAMED_ONLY;
586 }
587 }
588 }
589 // Default value type should match argument type only when defined
590 if (arg.m_fallback.index() == 2) {
591 const RPCArg::Type type = arg.m_type;
592 switch (std::get<RPCArg::Default>(arg.m_fallback).getType()) {
593 case UniValue::VOBJ:
595 break;
596 case UniValue::VARR:
598 break;
599 case UniValue::VSTR:
601 break;
602 case UniValue::VNUM:
604 break;
605 case UniValue::VBOOL:
607 break;
608 case UniValue::VNULL:
609 // Null values are accepted in all arguments
610 break;
611 default:
613 break;
614 }
615 }
616 }
617}
618
620{
621 std::string result;
622 for (const auto& r : m_results) {
623 if (r.m_type == RPCResult::Type::ANY) continue; // for testing only
624 if (r.m_cond.empty()) {
625 result += "\nResult:\n";
626 } else {
627 result += "\nResult (" + r.m_cond + "):\n";
628 }
629 Sections sections;
630 r.ToSections(sections);
631 result += sections.ToString();
632 }
633 return result;
634}
635
637{
638 return m_examples.empty() ? m_examples : "\nExamples:\n" + m_examples;
639}
640
642{
643 if (request.mode == JSONRPCRequest::GET_ARGS) {
644 return GetArgMap();
645 }
646 /*
647 * Check if the given request is valid according to this command or if
648 * the user is asking for help information, and throw help when appropriate.
649 */
650 if (request.mode == JSONRPCRequest::GET_HELP || !IsValidNumArgs(request.params.size())) {
651 throw std::runtime_error(ToString());
652 }
653 UniValue arg_mismatch{UniValue::VOBJ};
654 for (size_t i{0}; i < m_args.size(); ++i) {
655 const auto& arg{m_args.at(i)};
656 UniValue match{arg.MatchesType(request.params[i])};
657 if (!match.isTrue()) {
658 arg_mismatch.pushKV(strprintf("Position %s (%s)", i + 1, arg.m_names), std::move(match));
659 }
660 }
661 if (!arg_mismatch.empty()) {
662 throw JSONRPCError(RPC_TYPE_ERROR, strprintf("Wrong type passed:\n%s", arg_mismatch.write(4)));
663 }
664 CHECK_NONFATAL(m_req == nullptr);
665 m_req = &request;
666 UniValue ret = m_fun(*this, request);
667 m_req = nullptr;
668 if (gArgs.GetBoolArg("-rpcdoccheck", DEFAULT_RPC_DOC_CHECK)) {
669 UniValue mismatch{UniValue::VARR};
670 for (const auto& res : m_results.m_results) {
671 UniValue match{res.MatchesType(ret)};
672 if (match.isTrue()) {
673 mismatch.setNull();
674 break;
675 }
676 mismatch.push_back(std::move(match));
677 }
678 if (!mismatch.isNull()) {
679 std::string explain{
680 mismatch.empty() ? "no possible results defined" :
681 mismatch.size() == 1 ? mismatch[0].write(4) :
682 mismatch.write(4)};
683 throw std::runtime_error{
684 strprintf("Internal bug detected: RPC call \"%s\" returned incorrect type:\n%s\n%s %s\nPlease report this issue here: %s\n",
685 m_name, explain,
688 }
689 }
690 return ret;
691}
692
693using CheckFn = void(const RPCArg&);
694static const UniValue* DetailMaybeArg(CheckFn* check, const std::vector<RPCArg>& params, const JSONRPCRequest* req, size_t i)
695{
696 CHECK_NONFATAL(i < params.size());
697 const UniValue& arg{CHECK_NONFATAL(req)->params[i]};
698 const RPCArg& param{params.at(i)};
699 if (check) check(param);
700
701 if (!arg.isNull()) return &arg;
702 if (!std::holds_alternative<RPCArg::Default>(param.m_fallback)) return nullptr;
703 return &std::get<RPCArg::Default>(param.m_fallback);
704}
705
706static void CheckRequiredOrDefault(const RPCArg& param)
707{
708 // Must use `Arg<Type>(key)` to get the argument or its default value.
709 const bool required{
710 std::holds_alternative<RPCArg::Optional>(param.m_fallback) && RPCArg::Optional::NO == std::get<RPCArg::Optional>(param.m_fallback),
711 };
712 CHECK_NONFATAL(required || std::holds_alternative<RPCArg::Default>(param.m_fallback));
713}
714
715#define TMPL_INST(check_param, ret_type, return_code) \
716 template <> \
717 ret_type RPCHelpMan::ArgValue<ret_type>(size_t i) const \
718 { \
719 const UniValue* maybe_arg{ \
720 DetailMaybeArg(check_param, m_args, m_req, i), \
721 }; \
722 return return_code \
723 } \
724 void force_semicolon(ret_type)
725
726// Optional arg (without default). Can also be called on required args, if needed.
727TMPL_INST(nullptr, const UniValue*, maybe_arg;);
728TMPL_INST(nullptr, std::optional<double>, maybe_arg ? std::optional{maybe_arg->get_real()} : std::nullopt;);
729TMPL_INST(nullptr, std::optional<bool>, maybe_arg ? std::optional{maybe_arg->get_bool()} : std::nullopt;);
730TMPL_INST(nullptr, const std::string*, maybe_arg ? &maybe_arg->get_str() : nullptr;);
731
732// Required arg or optional arg with default value.
734TMPL_INST(CheckRequiredOrDefault, bool, CHECK_NONFATAL(maybe_arg)->get_bool(););
735TMPL_INST(CheckRequiredOrDefault, int, CHECK_NONFATAL(maybe_arg)->getInt<int>(););
736TMPL_INST(CheckRequiredOrDefault, uint64_t, CHECK_NONFATAL(maybe_arg)->getInt<uint64_t>(););
737TMPL_INST(CheckRequiredOrDefault, const std::string&, CHECK_NONFATAL(maybe_arg)->get_str(););
738
739bool RPCHelpMan::IsValidNumArgs(size_t num_args) const
740{
741 size_t num_required_args = 0;
742 for (size_t n = m_args.size(); n > 0; --n) {
743 if (!m_args.at(n - 1).IsOptional()) {
744 num_required_args = n;
745 break;
746 }
747 }
748 return num_required_args <= num_args && num_args <= m_args.size();
749}
750
751std::vector<std::pair<std::string, bool>> RPCHelpMan::GetArgNames() const
752{
753 std::vector<std::pair<std::string, bool>> ret;
754 ret.reserve(m_args.size());
755 for (const auto& arg : m_args) {
756 if (arg.m_type == RPCArg::Type::OBJ_NAMED_PARAMS) {
757 for (const auto& inner : arg.m_inner) {
758 ret.emplace_back(inner.m_names, /*named_only=*/true);
759 }
760 }
761 ret.emplace_back(arg.m_names, /*named_only=*/false);
762 }
763 return ret;
764}
765
766size_t RPCHelpMan::GetParamIndex(std::string_view key) const
767{
768 auto it{std::find_if(
769 m_args.begin(), m_args.end(), [&key](const auto& arg) { return arg.GetName() == key;}
770 )};
771
772 CHECK_NONFATAL(it != m_args.end()); // TODO: ideally this is checked at compile time
773 return std::distance(m_args.begin(), it);
774}
775
776std::string RPCHelpMan::ToString() const
777{
778 std::string ret;
779
780 // Oneline summary
781 ret += m_name;
782 bool was_optional{false};
783 for (const auto& arg : m_args) {
784 if (arg.m_opts.hidden) break; // Any arg that follows is also hidden
785 const bool optional = arg.IsOptional();
786 ret += " ";
787 if (optional) {
788 if (!was_optional) ret += "( ";
789 was_optional = true;
790 } else {
791 if (was_optional) ret += ") ";
792 was_optional = false;
793 }
794 ret += arg.ToString(/*oneline=*/true);
795 }
796 if (was_optional) ret += " )";
797
798 // Description
799 ret += "\n\n" + TrimString(m_description) + "\n";
800
801 // Arguments
802 Sections sections;
803 Sections named_only_sections;
804 for (size_t i{0}; i < m_args.size(); ++i) {
805 const auto& arg = m_args.at(i);
806 if (arg.m_opts.hidden) break; // Any arg that follows is also hidden
807
808 // Push named argument name and description
809 sections.m_sections.emplace_back(util::ToString(i + 1) + ". " + arg.GetFirstName(), arg.ToDescriptionString(/*is_named_arg=*/true));
810 sections.m_max_pad = std::max(sections.m_max_pad, sections.m_sections.back().m_left.size());
811
812 // Recursively push nested args
813 sections.Push(arg);
814
815 // Push named-only argument sections
816 if (arg.m_type == RPCArg::Type::OBJ_NAMED_PARAMS) {
817 for (const auto& arg_inner : arg.m_inner) {
818 named_only_sections.PushSection({arg_inner.GetFirstName(), arg_inner.ToDescriptionString(/*is_named_arg=*/true)});
819 named_only_sections.Push(arg_inner);
820 }
821 }
822 }
823
824 if (!sections.m_sections.empty()) ret += "\nArguments:\n";
825 ret += sections.ToString();
826 if (!named_only_sections.m_sections.empty()) ret += "\nNamed Arguments:\n";
827 ret += named_only_sections.ToString();
828
829 // Result
831
832 // Examples
834
835 return ret;
836}
837
839{
841
842 auto push_back_arg_info = [&arr](const std::string& rpc_name, int pos, const std::string& arg_name, const RPCArg::Type& type) {
844 map.push_back(rpc_name);
845 map.push_back(pos);
846 map.push_back(arg_name);
847 map.push_back(type == RPCArg::Type::STR ||
848 type == RPCArg::Type::STR_HEX);
849 arr.push_back(std::move(map));
850 };
851
852 for (int i{0}; i < int(m_args.size()); ++i) {
853 const auto& arg = m_args.at(i);
854 std::vector<std::string> arg_names = SplitString(arg.m_names, '|');
855 for (const auto& arg_name : arg_names) {
856 push_back_arg_info(m_name, i, arg_name, arg.m_type);
857 if (arg.m_type == RPCArg::Type::OBJ_NAMED_PARAMS) {
858 for (const auto& inner : arg.m_inner) {
859 std::vector<std::string> inner_names = SplitString(inner.m_names, '|');
860 for (const std::string& inner_name : inner_names) {
861 push_back_arg_info(m_name, i, inner_name, inner.m_type);
862 }
863 }
864 }
865 }
866 }
867 return arr;
868}
869
870static std::optional<UniValue::VType> ExpectedType(RPCArg::Type type)
871{
872 using Type = RPCArg::Type;
873 switch (type) {
874 case Type::STR_HEX:
875 case Type::STR: {
876 return UniValue::VSTR;
877 }
878 case Type::NUM: {
879 return UniValue::VNUM;
880 }
881 case Type::AMOUNT: {
882 // VNUM or VSTR, checked inside AmountFromValue()
883 return std::nullopt;
884 }
885 case Type::RANGE: {
886 // VNUM or VARR, checked inside ParseRange()
887 return std::nullopt;
888 }
889 case Type::BOOL: {
890 return UniValue::VBOOL;
891 }
892 case Type::OBJ:
893 case Type::OBJ_NAMED_PARAMS:
894 case Type::OBJ_USER_KEYS: {
895 return UniValue::VOBJ;
896 }
897 case Type::ARR: {
898 return UniValue::VARR;
899 }
900 } // no default case, so the compiler can warn about missing cases
902}
903
905{
906 if (m_opts.skip_type_check) return true;
907 if (IsOptional() && request.isNull()) return true;
908 const auto exp_type{ExpectedType(m_type)};
909 if (!exp_type) return true; // nothing to check
910
911 if (*exp_type != request.getType()) {
912 return strprintf("JSON value of type %s is not of expected type %s", uvTypeName(request.getType()), uvTypeName(*exp_type));
913 }
914 return true;
915}
916
917std::string RPCArg::GetFirstName() const
918{
919 return m_names.substr(0, m_names.find('|'));
920}
921
922std::string RPCArg::GetName() const
923{
924 CHECK_NONFATAL(std::string::npos == m_names.find('|'));
925 return m_names;
926}
927
929{
930 if (m_fallback.index() != 0) {
931 return true;
932 } else {
933 return RPCArg::Optional::NO != std::get<RPCArg::Optional>(m_fallback);
934 }
935}
936
937std::string RPCArg::ToDescriptionString(bool is_named_arg) const
938{
939 std::string ret;
940 ret += "(";
941 if (m_opts.type_str.size() != 0) {
942 ret += m_opts.type_str.at(1);
943 } else {
944 switch (m_type) {
945 case Type::STR_HEX:
946 case Type::STR: {
947 ret += "string";
948 break;
949 }
950 case Type::NUM: {
951 ret += "numeric";
952 break;
953 }
954 case Type::AMOUNT: {
955 ret += "numeric or string";
956 break;
957 }
958 case Type::RANGE: {
959 ret += "numeric or array";
960 break;
961 }
962 case Type::BOOL: {
963 ret += "boolean";
964 break;
965 }
966 case Type::OBJ:
968 case Type::OBJ_USER_KEYS: {
969 ret += "json object";
970 break;
971 }
972 case Type::ARR: {
973 ret += "json array";
974 break;
975 }
976 } // no default case, so the compiler can warn about missing cases
977 }
978 if (m_fallback.index() == 1) {
979 ret += ", optional, default=" + std::get<RPCArg::DefaultHint>(m_fallback);
980 } else if (m_fallback.index() == 2) {
981 ret += ", optional, default=" + std::get<RPCArg::Default>(m_fallback).write();
982 } else {
983 switch (std::get<RPCArg::Optional>(m_fallback)) {
985 if (is_named_arg) ret += ", optional"; // Default value is "null" in dicts. Otherwise,
986 // nothing to do. Element is treated as if not present and has no default value
987 break;
988 }
990 ret += ", required";
991 break;
992 }
993 } // no default case, so the compiler can warn about missing cases
994 }
995 ret += ")";
996 if (m_type == Type::OBJ_NAMED_PARAMS) ret += " Options object that can be used to pass named arguments, listed below.";
997 ret += m_description.empty() ? "" : " " + m_description;
998 return ret;
999}
1000
1001// NOLINTNEXTLINE(misc-no-recursion)
1002void RPCResult::ToSections(Sections& sections, const OuterType outer_type, const int current_indent) const
1003{
1004 // Indentation
1005 const std::string indent(current_indent, ' ');
1006 const std::string indent_next(current_indent + 2, ' ');
1007
1008 // Elements in a JSON structure (dictionary or array) are separated by a comma
1009 const std::string maybe_separator{outer_type != OuterType::NONE ? "," : ""};
1010
1011 // The key name if recursed into a dictionary
1012 const std::string maybe_key{
1013 outer_type == OuterType::OBJ ?
1014 "\"" + this->m_key_name + "\" : " :
1015 ""};
1016
1017 // Format description with type
1018 const auto Description = [&](const std::string& type) {
1019 return "(" + type + (this->m_optional ? ", optional" : "") + ")" +
1020 (this->m_description.empty() ? "" : " " + this->m_description);
1021 };
1022
1023 switch (m_type) {
1024 case Type::ELISION: {
1025 // If the inner result is empty, use three dots for elision
1026 sections.PushSection({indent + "..." + maybe_separator, m_description});
1027 return;
1028 }
1029 case Type::ANY: {
1030 NONFATAL_UNREACHABLE(); // Only for testing
1031 }
1032 case Type::NONE: {
1033 sections.PushSection({indent + "null" + maybe_separator, Description("json null")});
1034 return;
1035 }
1036 case Type::STR: {
1037 sections.PushSection({indent + maybe_key + "\"str\"" + maybe_separator, Description("string")});
1038 return;
1039 }
1040 case Type::STR_AMOUNT: {
1041 sections.PushSection({indent + maybe_key + "n" + maybe_separator, Description("numeric")});
1042 return;
1043 }
1044 case Type::STR_HEX: {
1045 sections.PushSection({indent + maybe_key + "\"hex\"" + maybe_separator, Description("string")});
1046 return;
1047 }
1048 case Type::NUM: {
1049 sections.PushSection({indent + maybe_key + "n" + maybe_separator, Description("numeric")});
1050 return;
1051 }
1052 case Type::NUM_TIME: {
1053 sections.PushSection({indent + maybe_key + "xxx" + maybe_separator, Description("numeric")});
1054 return;
1055 }
1056 case Type::BOOL: {
1057 sections.PushSection({indent + maybe_key + "true|false" + maybe_separator, Description("boolean")});
1058 return;
1059 }
1060 case Type::ARR_FIXED:
1061 case Type::ARR: {
1062 sections.PushSection({indent + maybe_key + "[", Description("json array")});
1063 for (const auto& i : m_inner) {
1064 i.ToSections(sections, OuterType::ARR, current_indent + 2);
1065 }
1066 CHECK_NONFATAL(!m_inner.empty());
1067 if (m_type == Type::ARR && m_inner.back().m_type != Type::ELISION) {
1068 sections.PushSection({indent_next + "...", ""});
1069 } else {
1070 // Remove final comma, which would be invalid JSON
1071 sections.m_sections.back().m_left.pop_back();
1072 }
1073 sections.PushSection({indent + "]" + maybe_separator, ""});
1074 return;
1075 }
1076 case Type::OBJ_DYN:
1077 case Type::OBJ: {
1078 if (m_inner.empty()) {
1079 sections.PushSection({indent + maybe_key + "{}", Description("empty JSON object")});
1080 return;
1081 }
1082 sections.PushSection({indent + maybe_key + "{", Description("json object")});
1083 for (const auto& i : m_inner) {
1084 i.ToSections(sections, OuterType::OBJ, current_indent + 2);
1085 }
1086 if (m_type == Type::OBJ_DYN && m_inner.back().m_type != Type::ELISION) {
1087 // If the dictionary keys are dynamic, use three dots for continuation
1088 sections.PushSection({indent_next + "...", ""});
1089 } else {
1090 // Remove final comma, which would be invalid JSON
1091 sections.m_sections.back().m_left.pop_back();
1092 }
1093 sections.PushSection({indent + "}" + maybe_separator, ""});
1094 return;
1095 }
1096 } // no default case, so the compiler can warn about missing cases
1098}
1099
1100static std::optional<UniValue::VType> ExpectedType(RPCResult::Type type)
1101{
1102 using Type = RPCResult::Type;
1103 switch (type) {
1104 case Type::ELISION:
1105 case Type::ANY: {
1106 return std::nullopt;
1107 }
1108 case Type::NONE: {
1109 return UniValue::VNULL;
1110 }
1111 case Type::STR:
1112 case Type::STR_HEX: {
1113 return UniValue::VSTR;
1114 }
1115 case Type::NUM:
1116 case Type::STR_AMOUNT:
1117 case Type::NUM_TIME: {
1118 return UniValue::VNUM;
1119 }
1120 case Type::BOOL: {
1121 return UniValue::VBOOL;
1122 }
1123 case Type::ARR_FIXED:
1124 case Type::ARR: {
1125 return UniValue::VARR;
1126 }
1127 case Type::OBJ_DYN:
1128 case Type::OBJ: {
1129 return UniValue::VOBJ;
1130 }
1131 } // no default case, so the compiler can warn about missing cases
1133}
1134
1135// NOLINTNEXTLINE(misc-no-recursion)
1137{
1138 if (m_skip_type_check) {
1139 return true;
1140 }
1141
1142 const auto exp_type = ExpectedType(m_type);
1143 if (!exp_type) return true; // can be any type, so nothing to check
1144
1145 if (*exp_type != result.getType()) {
1146 return strprintf("returned type is %s, but declared as %s in doc", uvTypeName(result.getType()), uvTypeName(*exp_type));
1147 }
1148
1149 if (UniValue::VARR == result.getType()) {
1150 UniValue errors(UniValue::VOBJ);
1151 for (size_t i{0}; i < result.get_array().size(); ++i) {
1152 // If there are more results than documented, reuse the last doc_inner.
1153 const RPCResult& doc_inner{m_inner.at(std::min(m_inner.size() - 1, i))};
1154 UniValue match{doc_inner.MatchesType(result.get_array()[i])};
1155 if (!match.isTrue()) errors.pushKV(strprintf("%d", i), std::move(match));
1156 }
1157 if (errors.empty()) return true; // empty result array is valid
1158 return errors;
1159 }
1160
1161 if (UniValue::VOBJ == result.getType()) {
1162 if (!m_inner.empty() && m_inner.at(0).m_type == Type::ELISION) return true;
1163 UniValue errors(UniValue::VOBJ);
1164 if (m_type == Type::OBJ_DYN) {
1165 const RPCResult& doc_inner{m_inner.at(0)}; // Assume all types are the same, randomly pick the first
1166 for (size_t i{0}; i < result.get_obj().size(); ++i) {
1167 UniValue match{doc_inner.MatchesType(result.get_obj()[i])};
1168 if (!match.isTrue()) errors.pushKV(result.getKeys()[i], std::move(match));
1169 }
1170 if (errors.empty()) return true; // empty result obj is valid
1171 return errors;
1172 }
1173 std::set<std::string> doc_keys;
1174 for (const auto& doc_entry : m_inner) {
1175 doc_keys.insert(doc_entry.m_key_name);
1176 }
1177 std::map<std::string, UniValue> result_obj;
1178 result.getObjMap(result_obj);
1179 for (const auto& result_entry : result_obj) {
1180 if (doc_keys.find(result_entry.first) == doc_keys.end()) {
1181 errors.pushKV(result_entry.first, "key returned that was not in doc");
1182 }
1183 }
1184
1185 for (const auto& doc_entry : m_inner) {
1186 const auto result_it{result_obj.find(doc_entry.m_key_name)};
1187 if (result_it == result_obj.end()) {
1188 if (!doc_entry.m_optional) {
1189 errors.pushKV(doc_entry.m_key_name, "key missing, despite not being optional in doc");
1190 }
1191 continue;
1192 }
1193 UniValue match{doc_entry.MatchesType(result_it->second)};
1194 if (!match.isTrue()) errors.pushKV(doc_entry.m_key_name, std::move(match));
1195 }
1196 if (errors.empty()) return true;
1197 return errors;
1198 }
1199
1200 return true;
1201}
1202
1204{
1205 if (m_type == Type::OBJ) {
1206 // May or may not be empty
1207 return;
1208 }
1209 // Everything else must either be empty or not
1210 const bool inner_needed{m_type == Type::ARR || m_type == Type::ARR_FIXED || m_type == Type::OBJ_DYN};
1211 CHECK_NONFATAL(inner_needed != m_inner.empty());
1212}
1213
1214// NOLINTNEXTLINE(misc-no-recursion)
1215std::string RPCArg::ToStringObj(const bool oneline) const
1216{
1217 std::string res;
1218 res += "\"";
1219 res += GetFirstName();
1220 if (oneline) {
1221 res += "\":";
1222 } else {
1223 res += "\": ";
1224 }
1225 switch (m_type) {
1226 case Type::STR:
1227 return res + "\"str\"";
1228 case Type::STR_HEX:
1229 return res + "\"hex\"";
1230 case Type::NUM:
1231 return res + "n";
1232 case Type::RANGE:
1233 return res + "n or [n,n]";
1234 case Type::AMOUNT:
1235 return res + "amount";
1236 case Type::BOOL:
1237 return res + "bool";
1238 case Type::ARR:
1239 res += "[";
1240 for (const auto& i : m_inner) {
1241 res += i.ToString(oneline) + ",";
1242 }
1243 return res + "...]";
1244 case Type::OBJ:
1247 // Currently unused, so avoid writing dead code
1249 } // no default case, so the compiler can warn about missing cases
1251}
1252
1253// NOLINTNEXTLINE(misc-no-recursion)
1254std::string RPCArg::ToString(const bool oneline) const
1255{
1256 if (oneline && !m_opts.oneline_description.empty()) {
1257 if (m_opts.oneline_description[0] == '\"' && m_type != Type::STR_HEX && m_type != Type::STR && gArgs.GetBoolArg("-rpcdoccheck", DEFAULT_RPC_DOC_CHECK)) {
1258 throw std::runtime_error{
1259 STR_INTERNAL_BUG(strprintf("non-string RPC arg \"%s\" quotes oneline_description:\n%s",
1261 )};
1262 }
1264 }
1265
1266 switch (m_type) {
1267 case Type::STR_HEX:
1268 case Type::STR: {
1269 return "\"" + GetFirstName() + "\"";
1270 }
1271 case Type::NUM:
1272 case Type::RANGE:
1273 case Type::AMOUNT:
1274 case Type::BOOL: {
1275 return GetFirstName();
1276 }
1277 case Type::OBJ:
1279 case Type::OBJ_USER_KEYS: {
1280 // NOLINTNEXTLINE(misc-no-recursion)
1281 const std::string res = Join(m_inner, ",", [&](const RPCArg& i) { return i.ToStringObj(oneline); });
1282 if (m_type == Type::OBJ) {
1283 return "{" + res + "}";
1284 } else {
1285 return "{" + res + ",...}";
1286 }
1287 }
1288 case Type::ARR: {
1289 std::string res;
1290 for (const auto& i : m_inner) {
1291 res += i.ToString(oneline) + ",";
1292 }
1293 return "[" + res + "...]";
1294 }
1295 } // no default case, so the compiler can warn about missing cases
1297}
1298
1299static std::pair<int64_t, int64_t> ParseRange(const UniValue& value)
1300{
1301 if (value.isNum()) {
1302 return {0, value.getInt<int64_t>()};
1303 }
1304 if (value.isArray() && value.size() == 2 && value[0].isNum() && value[1].isNum()) {
1305 int64_t low = value[0].getInt<int64_t>();
1306 int64_t high = value[1].getInt<int64_t>();
1307 if (low > high) throw JSONRPCError(RPC_INVALID_PARAMETER, "Range specified as [begin,end] must not have begin after end");
1308 return {low, high};
1309 }
1310 throw JSONRPCError(RPC_INVALID_PARAMETER, "Range must be specified as end or as [begin,end]");
1311}
1312
1313std::pair<int64_t, int64_t> ParseDescriptorRange(const UniValue& value)
1314{
1315 int64_t low, high;
1316 std::tie(low, high) = ParseRange(value);
1317 if (low < 0) {
1318 throw JSONRPCError(RPC_INVALID_PARAMETER, "Range should be greater or equal than 0");
1319 }
1320 if ((high >> 31) != 0) {
1321 throw JSONRPCError(RPC_INVALID_PARAMETER, "End of range is too high");
1322 }
1323 if (high >= low + 1000000) {
1324 throw JSONRPCError(RPC_INVALID_PARAMETER, "Range is too large");
1325 }
1326 return {low, high};
1327}
1328
1329std::vector<CScript> EvalDescriptorStringOrObject(const UniValue& scanobject, FlatSigningProvider& provider, const bool expand_priv)
1330{
1331 std::string desc_str;
1332 std::pair<int64_t, int64_t> range = {0, 1000};
1333 if (scanobject.isStr()) {
1334 desc_str = scanobject.get_str();
1335 } else if (scanobject.isObject()) {
1336 const UniValue& desc_uni{scanobject.find_value("desc")};
1337 if (desc_uni.isNull()) throw JSONRPCError(RPC_INVALID_PARAMETER, "Descriptor needs to be provided in scan object");
1338 desc_str = desc_uni.get_str();
1339 const UniValue& range_uni{scanobject.find_value("range")};
1340 if (!range_uni.isNull()) {
1341 range = ParseDescriptorRange(range_uni);
1342 }
1343 } else {
1344 throw JSONRPCError(RPC_INVALID_PARAMETER, "Scan object needs to be either a string or an object");
1345 }
1346
1347 std::string error;
1348 auto desc = Parse(desc_str, provider, error);
1349 if (!desc) {
1351 }
1352 if (!desc->IsRange()) {
1353 range.first = 0;
1354 range.second = 0;
1355 }
1356 std::vector<CScript> ret;
1357 for (int i = range.first; i <= range.second; ++i) {
1358 std::vector<CScript> scripts;
1359 if (!desc->Expand(i, provider, scripts, provider)) {
1360 throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, strprintf("Cannot derive script without private keys: '%s'", desc_str));
1361 }
1362 if (expand_priv) {
1363 desc->ExpandPrivate(/*pos=*/i, provider, /*out=*/provider);
1364 }
1365 std::move(scripts.begin(), scripts.end(), std::back_inserter(ret));
1366 }
1367 return ret;
1368}
1369
1371[[nodiscard]] static UniValue BilingualStringsToUniValue(const std::vector<bilingual_str>& bilingual_strings)
1372{
1373 CHECK_NONFATAL(!bilingual_strings.empty());
1374 UniValue result{UniValue::VARR};
1375 for (const auto& s : bilingual_strings) {
1376 result.push_back(s.original);
1377 }
1378 return result;
1379}
1380
1381void PushWarnings(const UniValue& warnings, UniValue& obj)
1382{
1383 if (warnings.empty()) return;
1384 obj.pushKV("warnings", warnings);
1385}
1386
1387void PushWarnings(const std::vector<bilingual_str>& warnings, UniValue& obj)
1388{
1389 if (warnings.empty()) return;
1390 obj.pushKV("warnings", BilingualStringsToUniValue(warnings));
1391}
bool IsValidDestination(const CTxDestination &dest)
Check whether a CTxDestination corresponds to one with an address.
std::variant< CNoDestination, PubKeyDestination, PKHash, ScriptHash, WitnessV0ScriptHash, WitnessV0KeyHash, WitnessV1Taproot, PayToAnchor, WitnessUnknown > CTxDestination
A txout script categorized into standard templates.
bool MoneyRange(const CAmount &nValue)
Definition amount.h:27
int64_t CAmount
Amount in satoshis (Can be negative)
Definition amount.h:12
static constexpr CAmount COIN
The amount of satoshis in one BTC.
Definition amount.h:15
ArgsManager gArgs
Definition args.cpp:41
int ret
#define PACKAGE_NAME
#define PACKAGE_BUGREPORT
ArgsManager & args
Definition bitcoind.cpp:270
#define CHECK_NONFATAL(condition)
Identity function.
Definition check.h:73
#define NONFATAL_UNREACHABLE()
NONFATAL_UNREACHABLE() is a macro that is used to mark unreachable code.
Definition check.h:94
#define STR_INTERNAL_BUG(msg)
Definition check.h:60
bool GetBoolArg(const std::string &strArg, bool fDefault) const
Return boolean argument or default value.
Definition args.cpp:506
Fee rate in satoshis per kilovirtualbyte: CAmount / kvB.
Definition feerate.h:33
A reference to a CKey: the Hash160 of its serialized public key.
Definition pubkey.h:24
An encapsulated public key.
Definition pubkey.h:34
bool IsFullyValid() const
fully validate whether this is a valid public key (more expensive than IsValid())
Definition pubkey.cpp:316
Serialized script, used inside transaction inputs and outputs.
Definition script.h:414
UniValue operator()(const WitnessUnknown &id) const
Definition util.cpp:343
UniValue operator()(const WitnessV0KeyHash &id) const
Definition util.cpp:305
UniValue operator()(const WitnessV0ScriptHash &id) const
Definition util.cpp:315
DescribeAddressVisitor()=default
UniValue operator()(const CNoDestination &dest) const
Definition util.cpp:279
UniValue operator()(const PubKeyDestination &dest) const
Definition util.cpp:284
UniValue operator()(const WitnessV1Taproot &tap) const
Definition util.cpp:325
UniValue operator()(const ScriptHash &scriptID) const
Definition util.cpp:297
UniValue operator()(const PKHash &keyID) const
Definition util.cpp:289
UniValue operator()(const PayToAnchor &anchor) const
Definition util.cpp:335
Fillable signing provider that keeps keys in an address->secret map.
virtual bool GetPubKey(const CKeyID &address, CPubKey &vchPubKeyOut) const override
UniValue params
Definition request.h:40
enum JSONRPCRequest::Mode mode
std::function< UniValue(const RPCHelpMan &, const JSONRPCRequest &)> RPCMethodImpl
Definition util.h:408
const RPCExamples m_examples
Definition util.h:489
size_t GetParamIndex(std::string_view key) const
Return positional index of a parameter using its name as key.
Definition util.cpp:766
RPCHelpMan(std::string name, std::string description, std::vector< RPCArg > args, RPCResults results, RPCExamples examples)
Definition util.cpp:551
const std::string m_description
Definition util.h:486
bool IsValidNumArgs(size_t num_args) const
If the supplied number of args is neither too small nor too high.
Definition util.cpp:739
const RPCMethodImpl m_fun
Definition util.h:485
const std::string m_name
Definition util.h:482
const RPCResults m_results
Definition util.h:488
const std::vector< RPCArg > m_args
Definition util.h:487
std::string ToString() const
Definition util.cpp:776
UniValue GetArgMap() const
Return the named args that need to be converted from string to another JSON type.
Definition util.cpp:838
std::vector< std::pair< std::string, bool > > GetArgNames() const
Return list of arguments and whether they are named-only.
Definition util.cpp:751
const JSONRPCRequest * m_req
Definition util.h:490
UniValue HandleRequest(const JSONRPCRequest &request) const
Definition util.cpp:641
void push_back(UniValue val)
Definition univalue.cpp:104
const std::string & get_str() const
bool isArray() const
Definition univalue.h:85
const UniValue & find_value(std::string_view key) const
Definition univalue.cpp:233
enum VType getType() const
Definition univalue.h:67
std::string write(unsigned int prettyIndent=0, unsigned int indentLevel=0) const
bool isNull() const
Definition univalue.h:79
const std::string & getValStr() const
Definition univalue.h:68
const UniValue & get_obj() const
size_t size() const
Definition univalue.h:71
enum VType type() const
Definition univalue.h:126
const std::vector< std::string > & getKeys() const
bool empty() const
Definition univalue.h:69
bool isStr() const
Definition univalue.h:83
Int getInt() const
Definition univalue.h:138
const UniValue & get_array() const
bool isNum() const
Definition univalue.h:84
void pushKV(std::string key, UniValue val)
Definition univalue.cpp:126
void getObjMap(std::map< std::string, UniValue > &kv) const
Definition univalue.cpp:146
bool isObject() const
Definition univalue.h:86
constexpr bool IsNull() const
Definition uint256.h:46
size_type size() const
Definition prevector.h:296
256-bit opaque blob.
Definition uint256.h:178
static UniValue Parse(std::string_view raw)
Parse string to UniValue or throw runtime_error if string contains invalid JSON.
Definition client.cpp:321
std::string FormatFullVersion()
is a home for simple enum and struct type definitions that can be used internally by functions in the...
util::Result< int > SighashFromStr(const std::string &sighash)
std::string HexStr(const Span< const uint8_t > s)
Convert a span of bytes to a lower-case hexadecimal string.
Definition hex_base.cpp:29
@ SIGHASH_DEFAULT
Taproot only; implied when sighash byte is missing, and equivalent to SIGHASH_ALL.
Definition interpreter.h:35
CTxDestination DecodeDestination(const std::string &str, std::string &error_msg, std::vector< int > *error_locations)
Definition key_io.cpp:296
is a home for simple string functions returning descriptive messages that are used in RPC and GUI int...
PSBTError
Definition types.h:17
bilingual_str PSBTErrorString(PSBTError err)
Definition messages.cpp:106
bilingual_str TransactionErrorString(const TransactionError err)
Definition messages.cpp:124
TransactionError
Definition types.h:19
std::vector< std::string > SplitString(std::string_view str, char sep)
Definition string.h:59
bilingual_str ErrorString(const Result< T > &result)
Definition result.h:93
std::string ToString(const T &t)
Locale-independent version of std::to_string.
Definition string.h:156
std::string TrimString(std::string_view str, std::string_view pattern=" \f\n\r\t\v")
Definition string.h:79
auto Join(const C &container, const S &separator, UnaryOp unary_op)
Join all container items.
Definition string.h:115
is a home for public enum and struct type definitions that are used by internally by node code,...
CTxDestination AddAndGetDestinationForScript(FlatSigningProvider &keystore, const CScript &script, OutputType type)
Get a destination of the requested type (if possible) to the specified script.
OutputType
Definition outputtype.h:17
UniValue JSONRPCError(int code, const std::string &message)
Definition request.cpp:70
const char * name
Definition rest.cpp:49
RPCErrorCode
Bitcoin RPC error codes.
Definition protocol.h:25
@ RPC_VERIFY_ALREADY_IN_UTXO_SET
Transaction already in utxo set.
Definition protocol.h:49
@ RPC_TYPE_ERROR
Unexpected type was passed as parameter.
Definition protocol.h:41
@ RPC_TRANSACTION_REJECTED
Definition protocol.h:55
@ RPC_TRANSACTION_ERROR
Aliases for backward compatibility.
Definition protocol.h:54
@ RPC_INVALID_PARAMETER
Invalid, missing or duplicate parameter.
Definition protocol.h:44
@ RPC_INTERNAL_ERROR
Definition protocol.h:36
@ RPC_DESERIALIZATION_ERROR
Error parsing or validating structure in raw format.
Definition protocol.h:46
@ RPC_INVALID_ADDRESS_OR_KEY
Invalid address or key.
Definition protocol.h:42
static const UniValue * DetailMaybeArg(CheckFn *check, const std::vector< RPCArg > &params, const JSONRPCRequest *req, size_t i)
Definition util.cpp:694
std::vector< CScript > EvalDescriptorStringOrObject(const UniValue &scanobject, FlatSigningProvider &provider, const bool expand_priv)
Evaluate a descriptor given as a string, or as a {"desc":...,"range":...} object, with default range ...
Definition util.cpp:1329
void(const RPCArg &) CheckFn
Definition util.cpp:693
std::pair< int64_t, int64_t > ParseDescriptorRange(const UniValue &value)
Parse a JSON range specified as int64, or [int64, int64].
Definition util.cpp:1313
std::string HelpExampleCli(const std::string &methodname, const std::string &args)
Definition util.cpp:168
std::string HelpExampleRpcNamed(const std::string &methodname, const RPCArgList &args)
Definition util.cpp:192
CAmount AmountFromValue(const UniValue &value, int decimals)
Validate and return a CAmount from a UniValue number or string.
Definition util.cpp:83
RPCErrorCode RPCErrorFromPSBTError(PSBTError err)
Definition util.cpp:385
std::vector< unsigned char > ParseHexV(const UniValue &v, std::string_view name)
Definition util.cpp:115
static UniValue BilingualStringsToUniValue(const std::vector< bilingual_str > &bilingual_strings)
Convert a vector of bilingual strings to a UniValue::VARR containing their original untranslated valu...
Definition util.cpp:1371
void PushWarnings(const UniValue &warnings, UniValue &obj)
Push warning messages to an RPC "warnings" field as a JSON array of strings.
Definition util.cpp:1381
int ParseSighashString(const UniValue &sighash)
Returns a sighash value corresponding to the passed in argument.
Definition util.cpp:363
UniValue JSONRPCTransactionError(TransactionError terr, const std::string &err_string)
Definition util.cpp:414
#define TMPL_INST(check_param, ret_type, return_code)
Definition util.cpp:715
std::vector< unsigned char > ParseHexO(const UniValue &o, std::string_view strKey)
Definition util.cpp:124
UniValue JSONRPCPSBTError(PSBTError err)
Definition util.cpp:409
RPCErrorCode RPCErrorFromTransactionError(TransactionError terr)
Definition util.cpp:397
CFeeRate ParseFeeRate(const UniValue &json)
Parse a json number or string, denoting BTC/kvB, into a CFeeRate (sat/kvB).
Definition util.cpp:95
static std::optional< UniValue::VType > ExpectedType(RPCArg::Type type)
Definition util.cpp:870
std::string HelpExampleRpc(const std::string &methodname, const std::string &args)
Definition util.cpp:186
const std::string UNIX_EPOCH_TIME
String used to describe UNIX epoch time in documentation, factored out to a constant for consistency.
Definition util.cpp:43
std::string GetAllOutputTypes()
Gets all existing output types formatted for RPC help sections.
Definition util.cpp:46
CPubKey HexToPubKey(const std::string &hex_in)
Definition util.cpp:204
const std::string EXAMPLE_ADDRESS[2]
Example bech32 addresses for the RPCExamples help documentation.
Definition util.cpp:44
CTxDestination AddAndGetMultisigDestination(const int required, const std::vector< CPubKey > &pubkeys, OutputType type, FlatSigningProvider &keystore, CScript &script_out)
Definition util.cpp:241
static std::pair< int64_t, int64_t > ParseRange(const UniValue &value)
Definition util.cpp:1299
uint256 ParseHashO(const UniValue &o, std::string_view strKey)
Definition util.cpp:111
unsigned int ParseConfirmTarget(const UniValue &value, unsigned int max_target)
Parse a confirm target option and raise an RPC error if it is invalid.
Definition util.cpp:375
void RPCTypeCheckObj(const UniValue &o, const std::map< std::string, UniValueType > &typesExpected, bool fAllowNull, bool fStrict)
Definition util.cpp:56
std::string HelpExampleCliNamed(const std::string &methodname, const RPCArgList &args)
Definition util.cpp:173
uint256 ParseHashV(const UniValue &v, std::string_view name)
Utilities: convert hex-encoded Values (throws error if not hex).
Definition util.cpp:102
static void CheckRequiredOrDefault(const RPCArg &param)
Definition util.cpp:706
CPubKey AddrToPubKey(const FillableSigningProvider &keystore, const std::string &addr_in)
Definition util.cpp:220
UniValue DescribeAddress(const CTxDestination &dest)
Definition util.cpp:353
static constexpr bool DEFAULT_RPC_DOC_CHECK
Definition util.h:46
OuterType
Serializing JSON objects depends on the outer type.
Definition util.h:149
std::vector< std::pair< std::string, UniValue > > RPCArgList
Definition util.h:117
static const unsigned int MAX_SCRIPT_ELEMENT_SIZE
Definition script.h:27
static const int MAX_PUBKEYS_PER_MULTISIG
Definition script.h:33
CKeyID GetKeyForDestination(const SigningProvider &store, const CTxDestination &dest)
Return the CKeyID of the key involved in a script (if there is a unique one).
CScript GetScriptForMultisig(int nRequired, const std::vector< CPubKey > &keys)
Generate a multisig script.
Definition solver.cpp:218
std::string GetTxnOutputType(TxoutType t)
Get the name of a TxoutType as a string.
Definition solver.cpp:18
TxoutType
Definition solver.h:22
@ WITNESS_UNKNOWN
Only for Witness versions not already defined above.
std::vector< Byte > ParseHex(std::string_view hex_str)
Like TryParseHex, but returns an empty vector on invalid input.
Definition util.h:172
Type
Definition util.h:173
@ RANGE
Special type that is a NUM or [NUM,NUM].
@ OBJ_USER_KEYS
Special type where the user must set the keys e.g. to define multiple addresses; as opposed to e....
@ STR_HEX
Special type that is a STR with only hex chars.
@ AMOUNT
Special type representing a floating point amount (can be either NUM or STR)
@ OBJ_NAMED_PARAMS
Special type that behaves almost exactly like OBJ, defining an options object with a list of pre-defi...
const std::vector< RPCArg > m_inner
Only used for arrays or dicts.
Definition util.h:213
const RPCArgOptions m_opts
Definition util.h:216
const std::string m_names
The name of the arg (can be empty for inner args, can contain multiple aliases separated by | for nam...
Definition util.h:211
const Fallback m_fallback
Definition util.h:214
std::string ToString(bool oneline) const
Return the type string of the argument.
Definition util.cpp:1254
UniValue MatchesType(const UniValue &request) const
Check whether the request JSON type matches.
Definition util.cpp:904
const std::string m_description
Definition util.h:215
bool IsOptional() const
Definition util.cpp:928
std::string ToDescriptionString(bool is_named_arg) const
Return the description string, including the argument type and whether the argument is required.
Definition util.cpp:937
const Type m_type
Definition util.h:212
std::string GetName() const
Return the name, throws when there are aliases.
Definition util.cpp:922
std::string GetFirstName() const
Return the first of all aliases.
Definition util.cpp:917
std::string ToStringObj(bool oneline) const
Return the type string of the argument when it is in an object (dict).
Definition util.cpp:1215
@ OMITTED
Optional argument for which the default value is omitted from help text for one of two reasons:
@ NO
Required arg.
std::vector< std::string > type_str
Should be empty unless it is supposed to override the auto-generated type strings....
Definition util.h:158
std::string oneline_description
Should be empty unless it is supposed to override the auto-generated summary line.
Definition util.h:157
bool skip_type_check
Definition util.h:156
std::string ToDescriptionString() const
Definition util.cpp:636
const std::string m_examples
Definition util.h:395
const std::string m_description
Definition util.h:304
void ToSections(Sections &sections, OuterType outer_type=OuterType::NONE, const int current_indent=0) const
Append the sections of the result.
Definition util.cpp:1002
@ ELISION
Special type to denote elision (...)
@ NUM_TIME
Special numeric to denote unix epoch time.
@ ANY
Special type to disable type checks (for testing only)
@ ARR_FIXED
Special array that has a fixed number of entries.
@ OBJ_DYN
Special dictionary with keys that are not literals.
@ STR_HEX
Special string with only hex chars.
@ STR_AMOUNT
Special string to represent a floating point amount.
const std::vector< RPCResult > m_inner
Only used for arrays or dicts.
Definition util.h:301
UniValue MatchesType(const UniValue &result) const
Check whether the result JSON type matches.
Definition util.cpp:1136
void CheckInnerDoc() const
Definition util.cpp:1203
const bool m_optional
Definition util.h:302
const std::string m_key_name
Only used for dicts.
Definition util.h:300
const Type m_type
Definition util.h:299
const bool m_skip_type_check
Definition util.h:303
std::string ToDescriptionString() const
Return the description string.
Definition util.cpp:619
const std::vector< RPCResult > m_results
Definition util.h:376
A pair of strings that can be aligned (through padding) with other Sections later on.
Definition util.cpp:427
std::string m_left
Definition util.cpp:430
Section(const std::string &left, const std::string &right)
Definition util.cpp:428
const std::string m_right
Definition util.cpp:431
Keeps track of RPCArgs by transforming them into sections for the purpose of serializing everything t...
Definition util.cpp:438
void PushSection(const Section &s)
Definition util.cpp:442
std::vector< Section > m_sections
Definition util.cpp:439
void Push(const RPCArg &arg, const size_t current_indent=5, const OuterType outer_type=OuterType::NONE)
Recursive helper to translate an RPCArg into sections.
Definition util.cpp:452
size_t m_max_pad
Definition util.cpp:440
std::string ToString() const
Concatenate all sections with proper padding.
Definition util.cpp:510
CTxDestination subtype to encode any future Witness version.
Definition addresstype.h:96
#define strprintf
Format arguments and return the string or write to given std::ostream (see tinyformat::format doc for...
uint256 uint256S(std::string_view str)
Definition uint256.h:192
const char * uvTypeName(UniValue::VType t)
Definition univalue.cpp:218
bool ParseFixedPoint(std::string_view val, int decimals, int64_t *amount_out)
Parse number as fixed point according to JSON number syntax.
bool IsHex(std::string_view str)