18 #include <boost/test/unit_test.hpp> 62 return transformed_params;
68 std::string strMethod = vArgs[0];
69 vArgs.erase(vArgs.begin());
89 const std::vector<std::pair<std::string, bool>> arg_names{{
"arg1",
false}, {
"arg2",
false}, {
"arg3",
false}, {
"arg4",
false}, {
"arg5",
false}};
92 BOOST_CHECK_EQUAL(TransformParams(
JSON(R
"({"arg2": 2, "arg4": 4})"), arg_names).write(), "[null,2,null,4]");
95 BOOST_CHECK_EXCEPTION(TransformParams(
JSON(R
"({"arg2": 2, "arg2": 4})"), arg_names), UniValue, 96 HasJSON(R"({"code":-8,"message":"Parameter arg2 specified multiple times"})")); 99 BOOST_CHECK_EQUAL(TransformParams(
JSON(R
"({"arg5": 5, "args": [1, 2], "arg4": 4})"), arg_names).write(), "[1,2,null,4,5]");
102 BOOST_CHECK_EXCEPTION(TransformParams(
JSON(R
"({"arg2": 2, "unknown": 6})"), arg_names), UniValue, 103 HasJSON(R"({"code":-8,"message":"Unknown named parameter unknown"})")); 106 BOOST_CHECK_EXCEPTION(TransformParams(
JSON(R
"({"args": [1,2,3], "arg4": 4, "arg2": 2})"), arg_names), UniValue, 107 HasJSON(R"({"code":-8,"message":"Parameter arg2 specified twice both as positional and named argument"})")); 110 BOOST_CHECK_EQUAL(TransformParams(
JSON(R
"({"args": [1,2,3,4,5,6,7,8,9,10]})"), arg_names).write(), "[1,2,3,4,5,6,7,8,9,10]");
111 BOOST_CHECK_EQUAL(TransformParams(
JSON(R
"([1,2,3,4,5,6,7,8,9,10])"), arg_names).write(), "[1,2,3,4,5,6,7,8,9,10]");
116 const std::vector<std::pair<std::string, bool>> arg_names{{
"arg1",
false}, {
"arg2",
false}, {
"opt1",
true}, {
"opt2",
true}, {
"options",
false}};
122 BOOST_CHECK_EQUAL(TransformParams(
JSON(R
"({"arg1": 1, "arg2": 2, "opt1": 10, "opt2": 20})"), arg_names).write(), R"([1,2,{"opt1":10,"opt2":20}])"); 125 BOOST_CHECK_EQUAL(TransformParams(
JSON(R
"({"arg1": 1, "arg2": 2, "options":{"opt1": 10, "opt2": 20}})"), arg_names).write(), R"([1,2,{"opt1":10,"opt2":20}])"); 128 BOOST_CHECK_EXCEPTION(TransformParams(
JSON(R
"({"arg1": 1, "arg2": 2, "opt1": 10, "options":{"opt1": 10}})"), arg_names), UniValue, 129 HasJSON(R"({"code":-8,"message":"Parameter options conflicts with parameter opt1"})")); 132 BOOST_CHECK_EXCEPTION(TransformParams(
JSON(R
"({"args": [1, 2, {"opt1": 10}], "opt2": 20})"), arg_names), UniValue, 133 HasJSON(R"({"code":-8,"message":"Parameter options specified twice both as positional and named argument"})")); 143 BOOST_CHECK_THROW(
CallRPC(
"getrawtransaction a3b807410df0b60fcb9736768df5823938b2f838694939ba45f3c0a1bff150ed not_int"), std::runtime_error);
155 std::string rawtx =
"0100000001a15d57094aa7a21a28cb20b59aab8fc7d1149a3bdbcddba9c622e4f5f6a99ece010000006c493046022100f93bb0e7d8db7bd46e40132d1f8242026e045f03a0efe71bbb8e3f475e970d790221009337cd7f1f929f00cc6ff01f03729b069a7c21b59b1736ddfee5db5946c5da8c0121033b9b137ee87d5a812d6f506efdd37f0affa7ffc310711c06c7f3e097c9447c52ffffffff0100e1f505000000001976a9140389035a9225b3839e2bbf32d826a1e222031fd888ac00000000";
197 std::string prevout =
198 "[{\"txid\":\"b4cc287e58f87cdae59417329f710f3ecd75a4ee1d2872b7248f50977c8493f3\"," 199 "\"vout\":1,\"scriptPubKey\":\"a914b10c9df5f7edf436c697f02f1efdba4cf399615187\"," 200 "\"redeemScript\":\"512103debedc17b3df2badbcdd86d5feb4562b86fe182e5998abd8bcd4f122c6155b1b21027e940bb73ab8732bfdf7f9216ecefca5b94d6df834e77e108f68e66f126044c052ae\"}]";
201 r =
CallRPC(std::string(
"createrawtransaction ")+prevout+
" "+
202 "{\"3HqAe9LtNBjnsfM4CyYaWTnvCaUYT7v4oZ\":11}");
203 std::string notsigned = r.
get_str();
204 std::string privkey1 =
"\"KzsXybp9jX64P5ekX1KUxRQ79Jht9uzW7LorgwE65i5rWACL6LQe\"";
205 std::string privkey2 =
"\"Kyhdf5LuKTRx4ge69ybABsiUAWjVRK4XGxAKk2FQLp2HjGMy87Z4\"";
206 r =
CallRPC(std::string(
"signrawtransactionwithkey ")+notsigned+
" [] "+prevout);
208 r =
CallRPC(std::string(
"signrawtransactionwithkey ")+notsigned+
" ["+privkey1+
","+privkey2+
"] "+prevout);
209 BOOST_CHECK(r.get_obj().find_value(
"complete").get_bool() ==
true);
214 BOOST_CHECK_NO_THROW(
CallRPC(
"createrawtransaction [{\"txid\":\"a3b807410df0b60fcb9736768df5823938b2f838694939ba45f3c0a1bff150ed\",\"vout\":0}] {\"data\":\"68656c6c6f776f726c64\"}"));
217 BOOST_CHECK_THROW(
CallRPC(
"createrawtransaction [{\"txid\":\"a3b807410df0b60fcb9736768df5823938b2f838694939ba45f3c0a1bff150ed\",\"vout\":0}] {\"somedata\":\"68656c6c6f776f726c64\"}"), std::runtime_error);
220 BOOST_CHECK_THROW(
CallRPC(
"createrawtransaction [{\"txid\":\"a3b807410df0b60fcb9736768df5823938b2f838694939ba45f3c0a1bff150ed\",\"vout\":0}] {\"data\":\"12345\"}"), std::runtime_error);
221 BOOST_CHECK_THROW(
CallRPC(
"createrawtransaction [{\"txid\":\"a3b807410df0b60fcb9736768df5823938b2f838694939ba45f3c0a1bff150ed\",\"vout\":0}] {\"data\":\"12345g\"}"), std::runtime_error);
224 BOOST_CHECK_NO_THROW(
CallRPC(
"createrawtransaction [{\"txid\":\"a3b807410df0b60fcb9736768df5823938b2f838694939ba45f3c0a1bff150ed\",\"vout\":0}] {\"data\":\"010203040506070809101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081\"}"));
344 BOOST_CHECK_NO_THROW(r = CallRPC(std::string("setban 127.0.0.0/24 add 200"))); 345 SetMockTime(now += 2s); 346 const int64_t time_remaining_expected{198}; 347 BOOST_CHECK_NO_THROW(r = CallRPC(std::string("listbanned"))); 349 o1 = ar[0].get_obj(); 350 adr = o1.find_value("address"); 351 banned_until = o1.find_value("banned_until").getInt<int64_t>(); 352 const int64_t ban_created{o1.find_value("ban_created").getInt<int64_t>()}; 353 const int64_t ban_duration{o1.find_value("ban_duration").getInt<int64_t>()}; 354 const int64_t time_remaining{o1.find_value("time_remaining").getInt<int64_t>()}; 355 BOOST_CHECK_EQUAL(adr.get_str(), "127.0.0.0/24"); 356 BOOST_CHECK_EQUAL(banned_until, time_remaining_expected + now.count()); 357 BOOST_CHECK_EQUAL(ban_duration, banned_until - ban_created); 358 BOOST_CHECK_EQUAL(time_remaining, time_remaining_expected); 360 // must throw an exception because 127.0.0.1 is in already banned subnet range 361 BOOST_CHECK_THROW(r = CallRPC(std::string("setban 127.0.0.1 add")), std::runtime_error); 363 BOOST_CHECK_NO_THROW(CallRPC(std::string("setban 127.0.0.0/24 remove"))); 364 BOOST_CHECK_NO_THROW(r = CallRPC(std::string("listbanned"))); 366 BOOST_CHECK_EQUAL(ar.size(), 0U); 368 BOOST_CHECK_NO_THROW(r = CallRPC(std::string("setban 127.0.0.0/255.255.0.0 add"))); 369 BOOST_CHECK_THROW(r = CallRPC(std::string("setban 127.0.1.1 add")), std::runtime_error); 371 BOOST_CHECK_NO_THROW(CallRPC(std::string("clearbanned"))); 372 BOOST_CHECK_NO_THROW(r = CallRPC(std::string("listbanned"))); 374 BOOST_CHECK_EQUAL(ar.size(), 0U); 377 BOOST_CHECK_THROW(r = CallRPC(std::string("setban test add")), std::runtime_error); //invalid IP 380 BOOST_CHECK_NO_THROW(r = CallRPC(std::string("setban FE80:0000:0000:0000:0202:B3FF:FE1E:8329 add"))); 381 BOOST_CHECK_NO_THROW(r = CallRPC(std::string("listbanned"))); 383 o1 = ar[0].get_obj(); 384 adr = o1.find_value("address"); 385 BOOST_CHECK_EQUAL(adr.get_str(), "fe80::202:b3ff:fe1e:8329/128"); 387 BOOST_CHECK_NO_THROW(CallRPC(std::string("clearbanned"))); 388 BOOST_CHECK_NO_THROW(r = CallRPC(std::string("setban 2001:db8::/ffff:fffc:0:0:0:0:0:0 add"))); 389 BOOST_CHECK_NO_THROW(r = CallRPC(std::string("listbanned"))); 391 o1 = ar[0].get_obj(); 392 adr = o1.find_value("address"); 393 BOOST_CHECK_EQUAL(adr.get_str(), "2001:db8::/30"); 395 BOOST_CHECK_NO_THROW(CallRPC(std::string("clearbanned"))); 396 BOOST_CHECK_NO_THROW(r = CallRPC(std::string("setban 2001:4d48:ac57:400:cacf:e9ff:fe1d:9c63/128 add"))); 397 BOOST_CHECK_NO_THROW(r = CallRPC(std::string("listbanned"))); 399 o1 = ar[0].get_obj(); 400 adr = o1.find_value("address"); 401 BOOST_CHECK_EQUAL(adr.get_str(), "2001:4d48:ac57:400:cacf:e9ff:fe1d:9c63/128"); 404 BOOST_AUTO_TEST_CASE(rpc_convert_values_generatetoaddress) 408 BOOST_CHECK_NO_THROW(result = RPCConvertValues("generatetoaddress", {"101", "mkESjLZW66TmHhiFX8MCaBjrhZ543PPh9a"})); 409 BOOST_CHECK_EQUAL(result[0].getInt<int>(), 101); 410 BOOST_CHECK_EQUAL(result[1].get_str(), "mkESjLZW66TmHhiFX8MCaBjrhZ543PPh9a"); 412 BOOST_CHECK_NO_THROW(result = RPCConvertValues("generatetoaddress", {"101", "mhMbmE2tE9xzJYCV9aNC8jKWN31vtGrguU"})); 413 BOOST_CHECK_EQUAL(result[0].getInt<int>(), 101); 414 BOOST_CHECK_EQUAL(result[1].get_str(), "mhMbmE2tE9xzJYCV9aNC8jKWN31vtGrguU"); 416 BOOST_CHECK_NO_THROW(result = RPCConvertValues("generatetoaddress", {"1", "mkESjLZW66TmHhiFX8MCaBjrhZ543PPh9a", "9"})); 417 BOOST_CHECK_EQUAL(result[0].getInt<int>(), 1); 418 BOOST_CHECK_EQUAL(result[1].get_str(), "mkESjLZW66TmHhiFX8MCaBjrhZ543PPh9a"); 419 BOOST_CHECK_EQUAL(result[2].getInt<int>(), 9); 421 BOOST_CHECK_NO_THROW(result = RPCConvertValues("generatetoaddress", {"1", "mhMbmE2tE9xzJYCV9aNC8jKWN31vtGrguU", "9"})); 422 BOOST_CHECK_EQUAL(result[0].getInt<int>(), 1); 423 BOOST_CHECK_EQUAL(result[1].get_str(), "mhMbmE2tE9xzJYCV9aNC8jKWN31vtGrguU"); 424 BOOST_CHECK_EQUAL(result[2].getInt<int>(), 9); 427 BOOST_AUTO_TEST_CASE(rpc_getblockstats_calculate_percentiles_by_weight) 429 int64_t total_weight = 200; 430 std::vector<std::pair<CAmount, int64_t>> feerates; 431 feerates.reserve(200); 432 CAmount result[NUM_GETBLOCKSTATS_PERCENTILES] = { 0 }; 434 for (int64_t i = 0; i < 100; i++) { 435 feerates.emplace_back(1 ,1); 438 for (int64_t i = 0; i < 100; i++) { 439 feerates.emplace_back(2 ,1); 442 CalculatePercentilesByWeight(result, feerates, total_weight); 443 BOOST_CHECK_EQUAL(result[0], 1); 444 BOOST_CHECK_EQUAL(result[1], 1); 445 BOOST_CHECK_EQUAL(result[2], 1); 446 BOOST_CHECK_EQUAL(result[3], 2); 447 BOOST_CHECK_EQUAL(result[4], 2); 449 // Test with more pairs, and two pairs overlapping 2 percentiles. 451 CAmount result2[NUM_GETBLOCKSTATS_PERCENTILES] = { 0 }; 454 feerates.emplace_back(1, 9); 455 feerates.emplace_back(2 , 16); //10th + 25th percentile 456 feerates.emplace_back(4 ,50); //50th + 75th percentile 457 feerates.emplace_back(5 ,10); 458 feerates.emplace_back(9 ,15); // 90th percentile 460 CalculatePercentilesByWeight(result2, feerates, total_weight); 462 BOOST_CHECK_EQUAL(result2[0], 2); 463 BOOST_CHECK_EQUAL(result2[1], 2); 464 BOOST_CHECK_EQUAL(result2[2], 4); 465 BOOST_CHECK_EQUAL(result2[3], 4); 466 BOOST_CHECK_EQUAL(result2[4], 9); 468 // Same test as above, but one of the percentile-overlapping pairs is split in 2. 470 CAmount result3[NUM_GETBLOCKSTATS_PERCENTILES] = { 0 }; 473 feerates.emplace_back(1, 9); 474 feerates.emplace_back(2 , 11); // 10th percentile 475 feerates.emplace_back(2 , 5); // 25th percentile 476 feerates.emplace_back(4 ,50); //50th + 75th percentile 477 feerates.emplace_back(5 ,10); 478 feerates.emplace_back(9 ,15); // 90th percentile 480 CalculatePercentilesByWeight(result3, feerates, total_weight); 482 BOOST_CHECK_EQUAL(result3[0], 2); 483 BOOST_CHECK_EQUAL(result3[1], 2); 484 BOOST_CHECK_EQUAL(result3[2], 4); 485 BOOST_CHECK_EQUAL(result3[3], 4); 486 BOOST_CHECK_EQUAL(result3[4], 9); 488 // Test with one transaction spanning all percentiles. 490 CAmount result4[NUM_GETBLOCKSTATS_PERCENTILES] = { 0 }; 493 feerates.emplace_back(1, 100); 494 feerates.emplace_back(2, 1); 495 feerates.emplace_back(3, 1); 496 feerates.emplace_back(3, 1); 497 feerates.emplace_back(999999, 1); 499 CalculatePercentilesByWeight(result4, feerates, total_weight); 501 for (int64_t i = 0; i < NUM_GETBLOCKSTATS_PERCENTILES; i++) { 502 BOOST_CHECK_EQUAL(result4[i], 1); 506 // Make sure errors are triggered appropriately if parameters have the same names. 507 BOOST_AUTO_TEST_CASE(check_dup_param_names) 509 enum ParamType { POSITIONAL, NAMED, NAMED_ONLY }; 510 auto make_rpc = [](std::vector<std::tuple<std::string, ParamType>> param_names) { 511 std::vector<RPCArg> params; 512 std::vector<RPCArg> options; 513 auto push_options = [&] { if (!options.empty()) params.emplace_back(strprintf("options%i", params.size()), RPCArg::Type::OBJ_NAMED_PARAMS, RPCArg::Optional::OMITTED, "", std::move(options)); }; 514 for (auto& [param_name, param_type] : param_names) { 515 if (param_type == POSITIONAL) { 517 params.emplace_back(std::move(param_name), RPCArg::Type::NUM, RPCArg::Optional::OMITTED, "description"); 519 options.emplace_back(std::move(param_name), RPCArg::Type::NUM, RPCArg::Optional::OMITTED, "description", RPCArgOptions{.also_positional = param_type == NAMED}); 523 return RPCHelpMan{"method_name", "description", params, RPCResults{}, RPCExamples{""}}; 526 // No errors if parameter names are unique. 527 make_rpc({{"p1", POSITIONAL}, {"p2", POSITIONAL}}); 528 make_rpc({{"p1", POSITIONAL}, {"p2", NAMED}}); 529 make_rpc({{"p1", POSITIONAL}, {"p2", NAMED_ONLY}}); 530 make_rpc({{"p1", NAMED}, {"p2", POSITIONAL}}); 531 make_rpc({{"p1", NAMED}, {"p2", NAMED}}); 532 make_rpc({{"p1", NAMED}, {"p2", NAMED_ONLY}}); 533 make_rpc({{"p1", NAMED_ONLY}, {"p2", POSITIONAL}}); 534 make_rpc({{"p1", NAMED_ONLY}, {"p2", NAMED}}); 535 make_rpc({{"p1", NAMED_ONLY}, {"p2", NAMED_ONLY}}); 537 // Error if parameters names are duplicates, unless one parameter is 538 // positional and the other is named and .also_positional is true. 539 BOOST_CHECK_THROW(make_rpc({{"p1", POSITIONAL}, {"p1", POSITIONAL}}), NonFatalCheckError); 540 make_rpc({{"p1", POSITIONAL}, {"p1", NAMED}}); 541 BOOST_CHECK_THROW(make_rpc({{"p1", POSITIONAL}, {"p1", NAMED_ONLY}}), NonFatalCheckError); 542 make_rpc({{"p1", NAMED}, {"p1", POSITIONAL}}); 543 BOOST_CHECK_THROW(make_rpc({{"p1", NAMED}, {"p1", NAMED}}), NonFatalCheckError); 544 BOOST_CHECK_THROW(make_rpc({{"p1", NAMED}, {"p1", NAMED_ONLY}}), NonFatalCheckError); 545 BOOST_CHECK_THROW(make_rpc({{"p1", NAMED_ONLY}, {"p1", POSITIONAL}}), NonFatalCheckError); 546 BOOST_CHECK_THROW(make_rpc({{"p1", NAMED_ONLY}, {"p1", NAMED}}), NonFatalCheckError); 547 BOOST_CHECK_THROW(make_rpc({{"p1", NAMED_ONLY}, {"p1", NAMED_ONLY}}), NonFatalCheckError); 549 // Make sure duplicate aliases are detected too. 550 BOOST_CHECK_THROW(make_rpc({{"p1", POSITIONAL}, {"p2|p1", NAMED_ONLY}}), NonFatalCheckError); 553 BOOST_AUTO_TEST_CASE(help_example) 555 // test different argument types 556 const RPCArgList& args = {{"foo", "bar"}, {"b", true}, {"n", 1}}; 557 BOOST_CHECK_EQUAL(HelpExampleCliNamed("test", args), "> bitcoin-cli -named test foo=bar b=true n=1\n"); 558 BOOST_CHECK_EQUAL(HelpExampleRpcNamed("test", args), "> curl --user myusername --data-binary '{\
"jsonrpc\": \"2.0\", \"id\": \"curltest\", \"method\": \"test\", \"params\": {\"foo\":\"bar\",\"b\":true,\"n\":1}}' -H 'content-type: application/json' http://127.0.0.1:8332/\n");
567 obj_value.
pushKV(
"foo",
"bar");
568 obj_value.
pushKV(
"b",
false);
571 BOOST_CHECK_EQUAL(
HelpExampleRpcNamed(
"test", {{
"name", obj_value}}),
"> curl --user myusername --data-binary '{\"jsonrpc\": \"2.0\", \"id\": \"curltest\", \"method\": \"test\", \"params\": {\"name\":{\"foo\":\"bar\",\"b\":false,\"n\":1}}}' -H 'content-type: application/json' http://127.0.0.1:8332/\n");
579 BOOST_CHECK_EQUAL(
HelpExampleRpcNamed(
"test", {{
"name", arr_value}}),
"> curl --user myusername --data-binary '{\"jsonrpc\": \"2.0\", \"id\": \"curltest\", \"method\": \"test\", \"params\": {\"name\":[\"bar\",false,1]}}' -H 'content-type: application/json' http://127.0.0.1:8332/\n");
595 rpc.HandleRequest(req);
600 constexpr
bool DEFAULT_BOOL =
true;
601 constexpr
auto DEFAULT_STRING =
"default";
602 constexpr uint64_t DEFAULT_UINT64_T = 3;
605 const std::vector<RPCArg> params{
621 BOOST_CHECK_EQUAL(
self.Arg<std::string>(
"req_str"), request.params[1].get_str());
622 BOOST_CHECK_EQUAL(
self.Arg<uint64_t>(
"def_uint64_t"), request.params[2].isNull() ? DEFAULT_UINT64_T : request.params[2].getInt<uint64_t>());
623 BOOST_CHECK_EQUAL(
self.Arg<std::string>(
"def_string"), request.params[3].isNull() ? DEFAULT_STRING : request.params[3].get_str());
624 BOOST_CHECK_EQUAL(
self.Arg<bool>(
"def_bool"), request.params[4].isNull() ? DEFAULT_BOOL : request.params[4].get_bool());
625 if (!request.params[5].isNull()) {
626 BOOST_CHECK_EQUAL(
self.MaybeArg<double>(
"opt_double").value(), request.params[5].get_real());
630 if (!request.params[6].isNull()) {
631 BOOST_CHECK(
self.MaybeArg<std::string>(
"opt_string"));
632 BOOST_CHECK_EQUAL(*
self.MaybeArg<std::string>(
"opt_string"), request.params[6].get_str());
634 BOOST_CHECK(!
self.MaybeArg<std::string>(
"opt_string"));
638 CheckRpc(params,
UniValue{
JSON(R
"([5, "hello", null, null, null, null, null])")}, check_positional); 639 CheckRpc(params, UniValue{JSON(R"([5, "hello", 4, "test", true, 1.23, "world"])")}, check_positional); char const * json() noexcept
Template to generate JSON data.
UniValue CallRPC(std::string args)
void push_back(UniValue val)
std::vector< std::string > SplitString(std::string_view str, char sep)
std::string HelpExampleRpcNamed(const std::string &methodname, const RPCArgList &args)
#define BOOST_CHECK_THROW(stmt, excMatch)
static UniValue ValueFromString(const std::string &str) noexcept
void appendCommand(const std::string &name, const CRPCCommand *pcmd)
Appends a CRPCCommand to the dispatch table.
static void CheckRpc(const std::vector< RPCArg > ¶ms, const UniValue &args, RPCHelpMan::RPCMethodImpl test_impl)
bool read(std::string_view raw)
UniValue TransformParams(const UniValue ¶ms, std::vector< std::pair< std::string, bool >> arg_names) const
const std::string & get_str() const
const UniValue & get_array() const
bool operator()(const UniValue &value) const
HasJSON(std::string json)
UniValue RPCConvertValues(const std::string &strMethod, const std::vector< std::string > &strParams)
Convert positional arguments to command-specific RPC representation.
UniValue execute(const JSONRPCRequest &request) const
Execute a method.
const UniValue & find_value(std::string_view key) const
void SetRPCWarmupFinished()
static CAmount AmountFromValue(const UniValue &value)
BOOST_FIXTURE_TEST_SUITE(cuckoocache_tests, BasicTestingSetup)
Test Suite for CuckooCache.
static UniValue CallRPC(BaseRequestHandler *rh, const std::string &strMethod, const std::vector< std::string > &args, const std::optional< std::string > &rpcwallet={})
BOOST_AUTO_TEST_SUITE_END()
std::function< UniValue(const RPCHelpMan &, const JSONRPCRequest &)> RPCMethodImpl
std::string write(unsigned int prettyIndent=0, unsigned int indentLevel=0) const
std::string HelpExampleCliNamed(const std::string &methodname, const RPCArgList &args)
static UniValue JSON(std::string_view json)
Optional argument for which the default value is omitted from help text for one of two reasons: ...
void setNumStr(std::string str)
#define BOOST_CHECK_EQUAL(v1, v2)
void pushKV(std::string key, UniValue val)
const UniValue & get_obj() const
BOOST_AUTO_TEST_CASE(rpc_namedparams)
UniValue ValueFromAmount(const CAmount amount)
bool RPCIsInWarmup(std::string *outStatus)
#define BOOST_CHECK_NO_THROW(stmt)
Testing setup that configures a complete environment.
#define BOOST_CHECK(expr)
static constexpr CAmount COIN
The amount of satoshis in one BTC.