Branch data Line data Source code
1 : : // Copyright (c) 2017-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 : : #ifndef BITCOIN_RPC_UTIL_H
6 : : #define BITCOIN_RPC_UTIL_H
7 : :
8 : : #include <addresstype.h>
9 : : #include <consensus/amount.h>
10 : : #include <node/transaction.h>
11 : : #include <outputtype.h>
12 : : #include <pubkey.h>
13 : : #include <rpc/protocol.h>
14 : : #include <rpc/request.h>
15 : : #include <script/script.h>
16 : : #include <script/sign.h>
17 : : #include <uint256.h>
18 : : #include <univalue.h>
19 : : #include <util/check.h>
20 : :
21 : : #include <cstddef>
22 : : #include <cstdint>
23 : : #include <functional>
24 : : #include <initializer_list>
25 : : #include <map>
26 : : #include <optional>
27 : : #include <string>
28 : : #include <string_view>
29 : : #include <type_traits>
30 : : #include <utility>
31 : : #include <variant>
32 : : #include <vector>
33 : :
34 : : class JSONRPCRequest;
35 : : enum ServiceFlags : uint64_t;
36 : : enum class OutputType;
37 : : struct FlatSigningProvider;
38 : : struct bilingual_str;
39 : : namespace common {
40 : : enum class PSBTError;
41 : : } // namespace common
42 : : namespace node {
43 : : enum class TransactionError;
44 : : } // namespace node
45 : :
46 : : static constexpr bool DEFAULT_RPC_DOC_CHECK{
47 : : #ifdef RPC_DOC_CHECK
48 : : true
49 : : #else
50 : : false
51 : : #endif
52 : : };
53 : :
54 : : /**
55 : : * String used to describe UNIX epoch time in documentation, factored out to a
56 : : * constant for consistency.
57 : : */
58 : : extern const std::string UNIX_EPOCH_TIME;
59 : :
60 : : /**
61 : : * Example bech32 addresses for the RPCExamples help documentation. They are intentionally
62 : : * invalid to prevent accidental transactions by users.
63 : : */
64 : : extern const std::string EXAMPLE_ADDRESS[2];
65 : :
66 : : class FillableSigningProvider;
67 : : class CScript;
68 : : struct Sections;
69 : :
70 : : struct HelpResult : std::runtime_error {
71 [ + - ]: 1030 : explicit HelpResult(const std::string& msg) : std::runtime_error{msg} {}
72 : : };
73 : :
74 : : /**
75 : : * Gets all existing output types formatted for RPC help sections.
76 : : *
77 : : * @return Comma separated string representing output type names.
78 : : */
79 : : std::string GetAllOutputTypes();
80 : :
81 : : /** Wrapper for UniValue::VType, which includes typeAny:
82 : : * Used to denote don't care type. */
83 : : struct UniValueType {
84 [ + - + - : 1865 : UniValueType(UniValue::VType _type) : typeAny(false), type(_type) {}
+ - + - ]
[ + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- ]
[ + - + - ]
[ + - + -
+ - + - +
- ]
85 [ + - + - : 798 : UniValueType() : typeAny(true) {}
+ - ][ + -
+ - + - +
- ]
86 : : bool typeAny;
87 : : UniValue::VType type;
88 : : };
89 : :
90 : : /*
91 : : Check for expected keys/value types in an Object.
92 : : */
93 : : void RPCTypeCheckObj(const UniValue& o,
94 : : const std::map<std::string, UniValueType>& typesExpected,
95 : : bool fAllowNull = false,
96 : : bool fStrict = false);
97 : :
98 : : /**
99 : : * Utilities: convert hex-encoded Values
100 : : * (throws error if not hex).
101 : : */
102 : : uint256 ParseHashV(const UniValue& v, std::string_view name);
103 : : uint256 ParseHashO(const UniValue& o, std::string_view strKey);
104 : : std::vector<unsigned char> ParseHexV(const UniValue& v, std::string_view name);
105 : : std::vector<unsigned char> ParseHexO(const UniValue& o, std::string_view strKey);
106 : :
107 : : /**
108 : : * Parses verbosity from provided UniValue.
109 : : *
110 : : * @param[in] arg The verbosity argument as an int (0, 1, 2,...) or bool if allow_bool is set to true
111 : : * @param[in] default_verbosity The value to return if verbosity argument is null
112 : : * @param[in] allow_bool If true, allows arg to be a bool and parses it
113 : : * @returns An integer describing the verbosity level (e.g. 0, 1, 2, etc.)
114 : : * @throws JSONRPCError if allow_bool is false but arg provided is boolean
115 : : */
116 : : int ParseVerbosity(const UniValue& arg, int default_verbosity, bool allow_bool);
117 : :
118 : : /**
119 : : * Validate and return a CAmount from a UniValue number or string.
120 : : *
121 : : * @param[in] value UniValue number or string to parse.
122 : : * @param[in] decimals Number of significant digits (default: 8).
123 : : * @returns a CAmount if the various checks pass.
124 : : */
125 : : CAmount AmountFromValue(const UniValue& value, int decimals = 8);
126 : : /**
127 : : * Parse a json number or string, denoting BTC/kvB, into a CFeeRate (sat/kvB).
128 : : * Reject negative values or rates larger than 1BTC/kvB.
129 : : */
130 : : CFeeRate ParseFeeRate(const UniValue& json);
131 : :
132 : : using RPCArgList = std::vector<std::pair<std::string, UniValue>>;
133 : : std::string HelpExampleCli(const std::string& methodname, const std::string& args);
134 : : std::string HelpExampleCliNamed(const std::string& methodname, const RPCArgList& args);
135 : : std::string HelpExampleRpc(const std::string& methodname, const std::string& args);
136 : : std::string HelpExampleRpcNamed(const std::string& methodname, const RPCArgList& args);
137 : :
138 : : CPubKey HexToPubKey(const std::string& hex_in);
139 : : CTxDestination AddAndGetMultisigDestination(const int required, const std::vector<CPubKey>& pubkeys, OutputType type, FlatSigningProvider& keystore, CScript& script_out);
140 : :
141 : : UniValue DescribeAddress(const CTxDestination& dest);
142 : :
143 : : /** Parse a sighash string representation and raise an RPC error if it is invalid. */
144 : : std::optional<int> ParseSighashString(const UniValue& sighash);
145 : :
146 : : //! Parse a confirm target option and raise an RPC error if it is invalid.
147 : : unsigned int ParseConfirmTarget(const UniValue& value, unsigned int max_target);
148 : :
149 : : RPCErrorCode RPCErrorFromTransactionError(node::TransactionError terr);
150 : : UniValue JSONRPCPSBTError(common::PSBTError err);
151 : : UniValue JSONRPCTransactionError(node::TransactionError terr, const std::string& err_string = "");
152 : :
153 : : //! Parse a JSON range specified as int64, or [int64, int64]
154 : : std::pair<int64_t, int64_t> ParseDescriptorRange(const UniValue& value);
155 : :
156 : : /** Evaluate a descriptor given as a string, or as a {"desc":...,"range":...} object, with default range of 1000. */
157 : : std::vector<CScript> EvalDescriptorStringOrObject(const UniValue& scanobject, FlatSigningProvider& provider, const bool expand_priv = false);
158 : :
159 : : /**
160 : : * Serializing JSON objects depends on the outer type. Only arrays and
161 : : * dictionaries can be nested in json. The top-level outer type is "NONE".
162 : : */
163 : : enum class OuterType {
164 : : ARR,
165 : : OBJ,
166 : : NONE, // Only set on first recursion
167 : : };
168 : :
169 : 1207906 : struct RPCArgOptions {
170 : : bool skip_type_check{false};
171 : : std::string oneline_description{}; //!< Should be empty unless it is supposed to override the auto-generated summary line
172 : : std::vector<std::string> type_str{}; //!< Should be empty unless it is supposed to override the auto-generated type strings. Vector length is either 0 or 2, m_opts.type_str.at(0) will override the type of the value in a key-value pair, m_opts.type_str.at(1) will override the type in the argument description.
173 : : bool hidden{false}; //!< For testing only
174 : : bool also_positional{false}; //!< If set allows a named-parameter field in an OBJ_NAMED_PARAM options object
175 : : //!< to have the same name as a top-level parameter. By default the RPC
176 : : //!< framework disallows this, because if an RPC request passes the value by
177 : : //!< name, it is assigned to top-level parameter position, not to the options
178 : : //!< position, defeating the purpose of using OBJ_NAMED_PARAMS instead OBJ for
179 : : //!< that option. But sometimes it makes sense to allow less-commonly used
180 : : //!< options to be passed by name only, and more commonly used options to be
181 : : //!< passed by name or position, so the RPC framework allows this as long as
182 : : //!< methods set the also_positional flag and read values from both positions.
183 : : };
184 : :
185 : : // NOLINTNEXTLINE(misc-no-recursion)
186 : : struct RPCArg {
187 : : enum class Type {
188 : : OBJ,
189 : : ARR,
190 : : STR,
191 : : NUM,
192 : : BOOL,
193 : : OBJ_NAMED_PARAMS, //!< Special type that behaves almost exactly like
194 : : //!< OBJ, defining an options object with a list of
195 : : //!< pre-defined keys. The only difference between OBJ
196 : : //!< and OBJ_NAMED_PARAMS is that OBJ_NAMED_PARMS
197 : : //!< also allows the keys to be passed as top-level
198 : : //!< named parameters, as a more convenient way to pass
199 : : //!< options to the RPC method without nesting them.
200 : : OBJ_USER_KEYS, //!< Special type where the user must set the keys e.g. to define multiple addresses; as opposed to e.g. an options object where the keys are predefined
201 : : AMOUNT, //!< Special type representing a floating point amount (can be either NUM or STR)
202 : : STR_HEX, //!< Special type that is a STR with only hex chars
203 : : RANGE, //!< Special type that is a NUM or [NUM,NUM]
204 : : };
205 : :
206 : : enum class Optional {
207 : : /** Required arg */
208 : : NO,
209 : : /**
210 : : * Optional argument for which the default value is omitted from
211 : : * help text for one of two reasons:
212 : : * - It's a named argument and has a default value of `null`.
213 : : * - Its default value is implicitly clear. That is, elements in an
214 : : * array may not exist by default.
215 : : * When possible, the default value should be specified.
216 : : */
217 : : OMITTED,
218 : : };
219 : : /** Hint for default value */
220 : : using DefaultHint = std::string;
221 : : /** Default constant value */
222 : : using Default = UniValue;
223 : : using Fallback = std::variant<Optional, DefaultHint, Default>;
224 : :
225 : : const std::string m_names; //!< The name of the arg (can be empty for inner args, can contain multiple aliases separated by | for named request arguments)
226 : : const Type m_type;
227 : : const std::vector<RPCArg> m_inner; //!< Only used for arrays or dicts
228 : : const Fallback m_fallback;
229 : : const std::string m_description;
230 : : const RPCArgOptions m_opts;
231 : :
232 : 1061291 : RPCArg(
233 : : std::string name,
234 : : Type type,
235 : : Fallback fallback,
236 : : std::string description,
237 : : RPCArgOptions opts = {})
238 : 1061291 : : m_names{std::move(name)},
239 : 1061291 : m_type{std::move(type)},
240 : 1061291 : m_fallback{std::move(fallback)},
241 : 1061291 : m_description{std::move(description)},
242 : 2122582 : m_opts{std::move(opts)}
243 : : {
244 [ + - - + : 1061291 : CHECK_NONFATAL(type != Type::ARR && type != Type::OBJ && type != Type::OBJ_NAMED_PARAMS && type != Type::OBJ_USER_KEYS);
+ - ]
245 : 1061291 : }
246 : :
247 : 154066 : RPCArg(
248 : : std::string name,
249 : : Type type,
250 : : Fallback fallback,
251 : : std::string description,
252 : : std::vector<RPCArg> inner,
253 : : RPCArgOptions opts = {})
254 : 154066 : : m_names{std::move(name)},
255 : 154066 : m_type{std::move(type)},
256 : 154066 : m_inner{std::move(inner)},
257 : 154066 : m_fallback{std::move(fallback)},
258 : 154066 : m_description{std::move(description)},
259 : 154066 : m_opts{std::move(opts)}
260 : : {
261 [ + + - + : 308132 : CHECK_NONFATAL(type == Type::ARR || type == Type::OBJ || type == Type::OBJ_NAMED_PARAMS || type == Type::OBJ_USER_KEYS);
- ]
262 : 154066 : }
263 : :
264 : : bool IsOptional() const;
265 : :
266 : : /**
267 : : * Check whether the request JSON type matches.
268 : : * Returns true if type matches, or object describing error(s) if not.
269 : : */
270 : : UniValue MatchesType(const UniValue& request) const;
271 : :
272 : : /** Return the first of all aliases */
273 : : std::string GetFirstName() const;
274 : :
275 : : /** Return the name, throws when there are aliases */
276 : : std::string GetName() const;
277 : :
278 : : /**
279 : : * Return the type string of the argument.
280 : : * Set oneline to allow it to be overridden by a custom oneline type string (m_opts.oneline_description).
281 : : */
282 : : std::string ToString(bool oneline) const;
283 : : /**
284 : : * Return the type string of the argument when it is in an object (dict).
285 : : * Set oneline to get the oneline representation (less whitespace)
286 : : */
287 : : std::string ToStringObj(bool oneline) const;
288 : : /**
289 : : * Return the description string, including the argument type and whether
290 : : * the argument is required.
291 : : */
292 : : std::string ToDescriptionString(bool is_named_arg) const;
293 : : };
294 : :
295 : : // NOLINTNEXTLINE(misc-no-recursion)
296 : : struct RPCResult {
297 : : enum class Type {
298 : : OBJ,
299 : : ARR,
300 : : STR,
301 : : NUM,
302 : : BOOL,
303 : : NONE,
304 : : ANY, //!< Special type to disable type checks (for testing only)
305 : : STR_AMOUNT, //!< Special string to represent a floating point amount
306 : : STR_HEX, //!< Special string with only hex chars
307 : : OBJ_DYN, //!< Special dictionary with keys that are not literals
308 : : ARR_FIXED, //!< Special array that has a fixed number of entries
309 : : NUM_TIME, //!< Special numeric to denote unix epoch time
310 : : ELISION, //!< Special type to denote elision (...)
311 : : };
312 : :
313 : : const Type m_type;
314 : : const std::string m_key_name; //!< Only used for dicts
315 : : const std::vector<RPCResult> m_inner; //!< Only used for arrays or dicts
316 : : const bool m_optional;
317 : : const bool m_skip_type_check;
318 : : const std::string m_description;
319 : : const std::string m_cond;
320 : :
321 : 212143 : RPCResult(
322 : : std::string cond,
323 : : Type type,
324 : : std::string m_key_name,
325 : : bool optional,
326 : : std::string description,
327 : : std::vector<RPCResult> inner = {})
328 : 212143 : : m_type{std::move(type)},
329 : 212143 : m_key_name{std::move(m_key_name)},
330 : 212143 : m_inner{std::move(inner)},
331 : 212143 : m_optional{optional},
332 : 212143 : m_skip_type_check{false},
333 : 212143 : m_description{std::move(description)},
334 : 212143 : m_cond{std::move(cond)}
335 : : {
336 [ + - ]: 212143 : CHECK_NONFATAL(!m_cond.empty());
337 [ + - ]: 212143 : CheckInnerDoc();
338 : 212143 : }
339 : :
340 : 212143 : RPCResult(
341 : : std::string cond,
342 : : Type type,
343 : : std::string m_key_name,
344 : : std::string description,
345 : : std::vector<RPCResult> inner = {})
346 [ + - ]: 212143 : : RPCResult{std::move(cond), type, std::move(m_key_name), /*optional=*/false, std::move(description), std::move(inner)} {}
347 : :
348 : 4306334 : RPCResult(
349 : : Type type,
350 : : std::string m_key_name,
351 : : bool optional,
352 : : std::string description,
353 : : std::vector<RPCResult> inner = {},
354 : : bool skip_type_check = false)
355 : 4306334 : : m_type{std::move(type)},
356 : 4306334 : m_key_name{std::move(m_key_name)},
357 : 4306334 : m_inner{std::move(inner)},
358 : 4306334 : m_optional{optional},
359 : 4306334 : m_skip_type_check{skip_type_check},
360 : 4306334 : m_description{std::move(description)},
361 [ + - ]: 4306334 : m_cond{}
362 : : {
363 [ + - ]: 4306334 : CheckInnerDoc();
364 : 4306334 : }
365 : :
366 : 3541112 : RPCResult(
367 : : Type type,
368 : : std::string m_key_name,
369 : : std::string description,
370 : : std::vector<RPCResult> inner = {},
371 : : bool skip_type_check = false)
372 [ + - ]: 3541112 : : RPCResult{type, std::move(m_key_name), /*optional=*/false, std::move(description), std::move(inner), skip_type_check} {}
373 : :
374 : : /** Append the sections of the result. */
375 : : void ToSections(Sections& sections, OuterType outer_type = OuterType::NONE, const int current_indent = 0) const;
376 : : /** Return the type string of the result when it is in an object (dict). */
377 : : std::string ToStringObj() const;
378 : : /** Return the description string, including the result type. */
379 : : std::string ToDescriptionString() const;
380 : : /** Check whether the result JSON type matches.
381 : : * Returns true if type matches, or object describing error(s) if not.
382 : : */
383 : : UniValue MatchesType(const UniValue& result) const;
384 : :
385 : : private:
386 : : void CheckInnerDoc() const;
387 : : };
388 : :
389 [ + - + - ]: 974706 : struct RPCResults {
390 : : const std::vector<RPCResult> m_results;
391 : :
392 : 397204 : RPCResults(RPCResult result)
393 [ + - + + : 1191612 : : m_results{{result}}
- - ]
394 : : {
395 : 794408 : }
396 : :
397 : 90126 : RPCResults(std::initializer_list<RPCResult> results)
398 [ + - + - ]: 90126 : : m_results{results}
[ + - + -
+ - + - ]
[ + - + -
+ - + - #
# ][ + - +
- + - + -
+ - ][ + -
+ - + - #
# ]
399 : : {
400 : 90126 : }
401 : :
402 : : /**
403 : : * Return the description string.
404 : : */
405 : : std::string ToDescriptionString() const;
406 : : };
407 : :
408 [ + - + - ]: 974706 : struct RPCExamples {
409 : : const std::string m_examples;
410 : 487330 : explicit RPCExamples(
411 : : std::string examples)
412 : 487330 : : m_examples(std::move(examples))
413 : : {
414 : : }
415 : : std::string ToDescriptionString() const;
416 : : };
417 : :
418 : : class RPCHelpMan
419 : : {
420 : : public:
421 : : RPCHelpMan(std::string name, std::string description, std::vector<RPCArg> args, RPCResults results, RPCExamples examples);
422 : : using RPCMethodImpl = std::function<UniValue(const RPCHelpMan&, const JSONRPCRequest&)>;
423 : : RPCHelpMan(std::string name, std::string description, std::vector<RPCArg> args, RPCResults results, RPCExamples examples, RPCMethodImpl fun);
424 : :
425 : : UniValue HandleRequest(const JSONRPCRequest& request) const;
426 : : /**
427 : : * @brief Helper to get a required or default-valued request argument.
428 : : *
429 : : * Use this function when the argument is required or when it has a default value. If the
430 : : * argument is optional and may not be provided, use MaybeArg instead.
431 : : *
432 : : * This function only works during m_fun(), i.e., it should only be used in
433 : : * RPC method implementations. It internally checks whether the user-passed
434 : : * argument isNull() and parses (from JSON) and returns the user-passed argument,
435 : : * or the default value derived from the RPCArg documentation.
436 : : *
437 : : * The instantiation of this helper for type R must match the corresponding RPCArg::Type.
438 : : *
439 : : * @return The value of the RPC argument (or the default value) cast to type R.
440 : : *
441 : : * @see MaybeArg for handling optional arguments without default values.
442 : : */
443 : : template <typename R>
444 : 29246 : auto Arg(std::string_view key) const
445 : : {
446 : 29246 : auto i{GetParamIndex(key)};
447 : : // Return argument (required or with default value).
448 : : if constexpr (std::is_integral_v<R> || std::is_floating_point_v<R>) {
449 : : // Return numbers by value.
450 : 2861 : return ArgValue<R>(i);
451 : : } else {
452 : : // Return everything else by reference.
453 : 26385 : return ArgValue<const R&>(i);
454 : : }
455 : : }
456 : : /**
457 : : * @brief Helper to get an optional request argument.
458 : : *
459 : : * Use this function when the argument is optional and does not have a default value. If the
460 : : * argument is required or has a default value, use Arg instead.
461 : : *
462 : : * This function only works during m_fun(), i.e., it should only be used in
463 : : * RPC method implementations. It internally checks whether the user-passed
464 : : * argument isNull() and parses (from JSON) and returns the user-passed argument,
465 : : * or a falsy value if no argument was passed.
466 : : *
467 : : * The instantiation of this helper for type R must match the corresponding RPCArg::Type.
468 : : *
469 : : * @return For integral and floating-point types, a std::optional<R> is returned.
470 : : * For other types, a R* pointer to the argument is returned. If the
471 : : * argument is not provided, std::nullopt or a null pointer is returned.
472 : : *
473 : : * @see Arg for handling arguments that are required or have a default value.
474 : : */
475 : : template <typename R>
476 : 1954 : auto MaybeArg(std::string_view key) const
477 : : {
478 : 1954 : auto i{GetParamIndex(key)};
479 : : // Return optional argument (without default).
480 : : if constexpr (std::is_integral_v<R> || std::is_floating_point_v<R>) {
481 : : // Return numbers by value, wrapped in optional.
482 : 1454 : return ArgValue<std::optional<R>>(i);
483 : : } else {
484 : : // Return other types by pointer.
485 : 500 : return ArgValue<const R*>(i);
486 : : }
487 : : }
488 : : std::string ToString() const;
489 : : /** Return the named args that need to be converted from string to another JSON type */
490 : : UniValue GetArgMap() const;
491 : : /** If the supplied number of args is neither too small nor too high */
492 : : bool IsValidNumArgs(size_t num_args) const;
493 : : //! Return list of arguments and whether they are named-only.
494 : : std::vector<std::pair<std::string, bool>> GetArgNames() const;
495 : :
496 : : const std::string m_name;
497 : :
498 : : private:
499 : : const RPCMethodImpl m_fun;
500 : : const std::string m_description;
501 : : const std::vector<RPCArg> m_args;
502 : : const RPCResults m_results;
503 : : const RPCExamples m_examples;
504 : : mutable const JSONRPCRequest* m_req{nullptr}; // A pointer to the request for the duration of m_fun()
505 : : template <typename R>
506 : : R ArgValue(size_t i) const;
507 : : //! Return positional index of a parameter using its name as key.
508 : : size_t GetParamIndex(std::string_view key) const;
509 : : };
510 : :
511 : : /**
512 : : * Push warning messages to an RPC "warnings" field as a JSON array of strings.
513 : : *
514 : : * @param[in] warnings Warning messages to push.
515 : : * @param[out] obj UniValue object to push the warnings array object to.
516 : : */
517 : : void PushWarnings(const UniValue& warnings, UniValue& obj);
518 : : void PushWarnings(const std::vector<bilingual_str>& warnings, UniValue& obj);
519 : :
520 : : std::vector<RPCResult> ScriptPubKeyDoc();
521 : :
522 : : /***
523 : : * Get the target for a given block index.
524 : : *
525 : : * @param[in] blockindex the block
526 : : * @param[in] pow_limit PoW limit (consensus parameter)
527 : : *
528 : : * @return the target
529 : : */
530 : : uint256 GetTarget(const CBlockIndex& blockindex, const uint256 pow_limit);
531 : :
532 : : #endif // BITCOIN_RPC_UTIL_H
|