Branch data Line data Source code
1 : : // Copyright (c) 2012-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 <core_io.h>
6 : : #include <interfaces/chain.h>
7 : : #include <node/context.h>
8 : : #include <rpc/blockchain.h>
9 : : #include <rpc/client.h>
10 : : #include <rpc/server.h>
11 : : #include <rpc/util.h>
12 : : #include <test/util/setup_common.h>
13 : : #include <univalue.h>
14 : : #include <util/time.h>
15 : :
16 : : #include <any>
17 : :
18 : : #include <boost/test/unit_test.hpp>
19 : :
20 : : using util::SplitString;
21 : :
22 : 14 : static UniValue JSON(std::string_view json)
23 : : {
24 [ + - ]: 14 : UniValue value;
25 [ + - + - : 28 : BOOST_CHECK(value.read(json));
+ - ]
26 : 14 : return value;
27 : 0 : }
28 : :
29 : 0 : class HasJSON
30 : : {
31 : : public:
32 : 6 : explicit HasJSON(std::string json) : m_json(std::move(json)) {}
33 : 6 : bool operator()(const UniValue& value) const
34 : : {
35 : 6 : std::string json{value.write()};
36 [ + - + - ]: 6 : BOOST_CHECK_EQUAL(json, m_json);
37 : 6 : return json == m_json;
38 : 6 : };
39 : :
40 : : private:
41 : : const std::string m_json;
42 : : };
43 : :
44 : 28 : class RPCTestingSetup : public TestingSetup
45 : : {
46 : : public:
47 : : UniValue TransformParams(const UniValue& params, std::vector<std::pair<std::string, bool>> arg_names) const;
48 : : UniValue CallRPC(std::string args);
49 : : };
50 : :
51 : 12 : UniValue RPCTestingSetup::TransformParams(const UniValue& params, std::vector<std::pair<std::string, bool>> arg_names) const
52 : : {
53 [ + - ]: 12 : UniValue transformed_params;
54 [ + - ]: 12 : CRPCTable table;
55 [ + - + - : 19 : CRPCCommand command{"category", "method", [&](const JSONRPCRequest& request, UniValue&, bool) -> bool { transformed_params = request.params; return true; }, arg_names, /*unique_id=*/0};
+ - ]
56 [ + - + - ]: 12 : table.appendCommand("method", &command);
57 : 12 : JSONRPCRequest request;
58 [ + - ]: 12 : request.strMethod = "method";
59 [ + - ]: 12 : request.params = params;
60 [ + - + + : 12 : if (RPCIsInWarmup(nullptr)) SetRPCWarmupFinished();
+ - ]
61 [ + + ]: 12 : table.execute(request);
62 : 14 : return transformed_params;
63 : 22 : }
64 : :
65 : 60 : UniValue RPCTestingSetup::CallRPC(std::string args)
66 : : {
67 : 60 : std::vector<std::string> vArgs{SplitString(args, ' ')};
68 [ + - ]: 60 : std::string strMethod = vArgs[0];
69 : 60 : vArgs.erase(vArgs.begin());
70 : 60 : JSONRPCRequest request;
71 : 60 : request.context = &m_node;
72 [ + - ]: 60 : request.strMethod = strMethod;
73 [ + + ]: 60 : request.params = RPCConvertValues(strMethod, vArgs);
74 [ + - - + : 55 : if (RPCIsInWarmup(nullptr)) SetRPCWarmupFinished();
- - ]
75 : 55 : try {
76 [ + + ]: 55 : UniValue result = tableRPC.execute(request);
77 : 36 : return result;
78 : : }
79 [ - + ]: 19 : catch (const UniValue& objError) {
80 [ + - + - : 19 : throw std::runtime_error(objError.find_value("message").get_str());
+ - ]
81 : 19 : }
82 : 84 : }
83 : :
84 : :
85 : : BOOST_FIXTURE_TEST_SUITE(rpc_tests, RPCTestingSetup)
86 : :
87 [ + - + - : 7 : BOOST_AUTO_TEST_CASE(rpc_namedparams)
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- ]
88 : : {
89 [ + + - - ]: 6 : const std::vector<std::pair<std::string, bool>> arg_names{{"arg1", false}, {"arg2", false}, {"arg3", false}, {"arg4", false}, {"arg5", false}};
90 : :
91 : : // Make sure named arguments are transformed into positional arguments in correct places separated by nulls
92 [ + - + - : 1 : BOOST_CHECK_EQUAL(TransformParams(JSON(R"({"arg2": 2, "arg4": 4})"), arg_names).write(), "[null,2,null,4]");
+ - + - +
- + - ]
93 : :
94 : : // Make sure named argument specified multiple times raises an exception
95 [ + - + - : 5 : BOOST_CHECK_EXCEPTION(TransformParams(JSON(R"({"arg2": 2, "arg2": 4})"), arg_names), UniValue,
+ - - + -
- - - - +
+ - + - +
- + - ]
96 : : HasJSON(R"({"code":-8,"message":"Parameter arg2 specified multiple times"})"));
97 : :
98 : : // Make sure named and positional arguments can be combined.
99 [ + - + - : 1 : BOOST_CHECK_EQUAL(TransformParams(JSON(R"({"arg5": 5, "args": [1, 2], "arg4": 4})"), arg_names).write(), "[1,2,null,4,5]");
+ - + - +
- + - ]
100 : :
101 : : // Make sure a unknown named argument raises an exception
102 [ + - + - : 5 : BOOST_CHECK_EXCEPTION(TransformParams(JSON(R"({"arg2": 2, "unknown": 6})"), arg_names), UniValue,
+ - - + -
- - - - +
+ - + - +
- + - ]
103 : : HasJSON(R"({"code":-8,"message":"Unknown named parameter unknown"})"));
104 : :
105 : : // Make sure an overlap between a named argument and positional argument raises an exception
106 [ + - + - : 5 : 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"})"));
108 : :
109 : : // Make sure extra positional arguments can be passed through to the method implementation, as long as they don't overlap with named arguments.
110 [ + - + - : 1 : 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 [ + - + - : 1 : 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]");
+ - + - +
- + - ]
112 [ + - + - : 2 : }
+ - + - +
- - + -
- ]
113 : :
114 [ + - + - : 7 : BOOST_AUTO_TEST_CASE(rpc_namedonlyparams)
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- ]
115 : : {
116 [ + + - - ]: 6 : const std::vector<std::pair<std::string, bool>> arg_names{{"arg1", false}, {"arg2", false}, {"opt1", true}, {"opt2", true}, {"options", false}};
117 : :
118 : : // Make sure optional parameters are really optional.
119 [ + - + - : 1 : BOOST_CHECK_EQUAL(TransformParams(JSON(R"({"arg1": 1, "arg2": 2})"), arg_names).write(), "[1,2]");
+ - + - +
- + - ]
120 : :
121 : : // Make sure named-only parameters are passed as options.
122 [ + - + - : 1 : BOOST_CHECK_EQUAL(TransformParams(JSON(R"({"arg1": 1, "arg2": 2, "opt1": 10, "opt2": 20})"), arg_names).write(), R"([1,2,{"opt1":10,"opt2":20}])");
+ - + - +
- + - ]
123 : :
124 : : // Make sure options can be passed directly.
125 [ + - + - : 1 : 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}])");
+ - + - +
- + - ]
126 : :
127 : : // Make sure options and named parameters conflict.
128 [ + - + - : 5 : 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"})"));
130 : :
131 : : // Make sure options object specified through args array conflicts.
132 [ + - + - : 5 : 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"})"));
134 [ + - + - : 2 : }
+ - + - +
- - + -
- ]
135 : :
136 [ + - + - : 7 : BOOST_AUTO_TEST_CASE(rpc_rawparams)
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- ]
137 : : {
138 : : // Test raw transaction API argument handling
139 [ + - ]: 1 : UniValue r;
140 : :
141 [ + - + - : 3 : BOOST_CHECK_THROW(CallRPC("getrawtransaction"), std::runtime_error);
- + - - -
- - + + -
+ - ]
142 [ + - + - : 3 : BOOST_CHECK_THROW(CallRPC("getrawtransaction not_hex"), std::runtime_error);
- + - - -
- - + + -
+ - ]
143 [ + - + - : 3 : BOOST_CHECK_THROW(CallRPC("getrawtransaction a3b807410df0b60fcb9736768df5823938b2f838694939ba45f3c0a1bff150ed not_int"), std::runtime_error);
- + - - -
- - + + -
+ - ]
144 : :
145 [ + - + - : 3 : BOOST_CHECK_THROW(CallRPC("createrawtransaction"), std::runtime_error);
- + - - -
- - + + -
+ - ]
146 [ + - + - : 3 : BOOST_CHECK_THROW(CallRPC("createrawtransaction null null"), std::runtime_error);
- + - - -
- - + + -
+ - ]
147 [ + - + - : 3 : BOOST_CHECK_THROW(CallRPC("createrawtransaction not_array"), std::runtime_error);
- + - - -
- - + + -
+ - ]
148 [ + - + - : 3 : BOOST_CHECK_THROW(CallRPC("createrawtransaction {} {}"), std::runtime_error);
- + - - -
- - + + -
+ - ]
149 [ + - + - : 2 : BOOST_CHECK_NO_THROW(CallRPC("createrawtransaction [] {}"));
+ - + - -
- - - -
- ]
150 [ + - + - : 3 : BOOST_CHECK_THROW(CallRPC("createrawtransaction [] {} extra"), std::runtime_error);
- + - - -
- - + + -
+ - ]
151 : :
152 [ + - + - : 3 : BOOST_CHECK_THROW(CallRPC("decoderawtransaction"), std::runtime_error);
- + - - -
- - + + -
+ - ]
153 [ + - + - : 3 : BOOST_CHECK_THROW(CallRPC("decoderawtransaction null"), std::runtime_error);
- + - - -
- - + + -
+ - ]
154 [ + - + - : 3 : BOOST_CHECK_THROW(CallRPC("decoderawtransaction DEADBEEF"), std::runtime_error);
- + - - -
- - + + -
+ - ]
155 [ + - ]: 1 : std::string rawtx = "0100000001a15d57094aa7a21a28cb20b59aab8fc7d1149a3bdbcddba9c622e4f5f6a99ece010000006c493046022100f93bb0e7d8db7bd46e40132d1f8242026e045f03a0efe71bbb8e3f475e970d790221009337cd7f1f929f00cc6ff01f03729b069a7c21b59b1736ddfee5db5946c5da8c0121033b9b137ee87d5a812d6f506efdd37f0affa7ffc310711c06c7f3e097c9447c52ffffffff0100e1f505000000001976a9140389035a9225b3839e2bbf32d826a1e222031fd888ac00000000";
156 [ + - + - : 2 : BOOST_CHECK_NO_THROW(r = CallRPC(std::string("decoderawtransaction ")+rawtx));
+ - + - +
- - - - -
- - ]
157 [ + - + - : 1 : BOOST_CHECK_EQUAL(r.get_obj().find_value("size").getInt<int>(), 193);
+ - + - +
- ]
158 [ + - + - : 1 : BOOST_CHECK_EQUAL(r.get_obj().find_value("version").getInt<int>(), 1);
+ - + - +
- ]
159 [ + - + - : 1 : BOOST_CHECK_EQUAL(r.get_obj().find_value("locktime").getInt<int>(), 0);
+ - + - +
- ]
160 [ + - + - : 6 : BOOST_CHECK_THROW(CallRPC(std::string("decoderawtransaction ")+rawtx+" extra"), std::runtime_error);
+ - - + -
- - - - +
+ - + - ]
161 [ + - + - : 4 : BOOST_CHECK_NO_THROW(r = CallRPC(std::string("decoderawtransaction ")+rawtx+" false"));
+ - + - +
- + - - -
- - - - ]
162 [ + - + - : 6 : BOOST_CHECK_THROW(r = CallRPC(std::string("decoderawtransaction ")+rawtx+" false extra"), std::runtime_error);
+ - - + -
- - - - +
+ - + - ]
163 : :
164 : : // Only check failure cases for sendrawtransaction, there's no network to send to...
165 [ + - + - : 3 : BOOST_CHECK_THROW(CallRPC("sendrawtransaction"), std::runtime_error);
- + - - -
- - + + -
+ - ]
166 [ + - + - : 3 : BOOST_CHECK_THROW(CallRPC("sendrawtransaction null"), std::runtime_error);
- + - - -
- - + + -
+ - ]
167 [ + - + - : 3 : BOOST_CHECK_THROW(CallRPC("sendrawtransaction DEADBEEF"), std::runtime_error);
- + - - -
- - + + -
+ - ]
168 [ + - + - : 6 : BOOST_CHECK_THROW(CallRPC(std::string("sendrawtransaction ")+rawtx+" extra"), std::runtime_error);
+ - - + -
- - - - +
+ - + - ]
169 : 1 : }
170 : :
171 [ + - + - : 7 : BOOST_AUTO_TEST_CASE(rpc_togglenetwork)
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- ]
172 : : {
173 [ + - ]: 1 : UniValue r;
174 : :
175 [ + - + - ]: 1 : r = CallRPC("getnetworkinfo");
176 [ + - + - : 1 : bool netState = r.get_obj().find_value("networkactive").get_bool();
+ - ]
177 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(netState, true);
178 : :
179 [ + - + - : 2 : BOOST_CHECK_NO_THROW(CallRPC("setnetworkactive false"));
+ - + - -
- - - -
- ]
180 [ + - + - ]: 1 : r = CallRPC("getnetworkinfo");
181 [ + - + - : 1 : int numConnection = r.get_obj().find_value("connections").getInt<int>();
+ - ]
182 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(numConnection, 0);
183 : :
184 [ + - + - : 1 : netState = r.get_obj().find_value("networkactive").get_bool();
+ - ]
185 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(netState, false);
186 : :
187 [ + - + - : 2 : BOOST_CHECK_NO_THROW(CallRPC("setnetworkactive true"));
+ - + - -
- - - -
- ]
188 [ + - + - ]: 1 : r = CallRPC("getnetworkinfo");
189 [ + - + - : 1 : netState = r.get_obj().find_value("networkactive").get_bool();
+ - ]
190 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(netState, true);
191 : 1 : }
192 : :
193 [ + - + - : 7 : BOOST_AUTO_TEST_CASE(rpc_rawsign)
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- ]
194 : : {
195 [ + - ]: 1 : UniValue r;
196 : : // input is a 1-of-2 multisig (so is output):
197 : 1 : std::string prevout =
198 : : "[{\"txid\":\"b4cc287e58f87cdae59417329f710f3ecd75a4ee1d2872b7248f50977c8493f3\","
199 : : "\"vout\":1,\"scriptPubKey\":\"a914b10c9df5f7edf436c697f02f1efdba4cf399615187\","
200 [ + - ]: 1 : "\"redeemScript\":\"512103debedc17b3df2badbcdd86d5feb4562b86fe182e5998abd8bcd4f122c6155b1b21027e940bb73ab8732bfdf7f9216ecefca5b94d6df834e77e108f68e66f126044c052ae\"}]";
201 [ + - + - : 3 : r = CallRPC(std::string("createrawtransaction ")+prevout+" "+
+ - ]
202 [ + - ]: 1 : "{\"3HqAe9LtNBjnsfM4CyYaWTnvCaUYT7v4oZ\":11}");
203 [ + - + - ]: 1 : std::string notsigned = r.get_str();
204 [ + - ]: 1 : std::string privkey1 = "\"KzsXybp9jX64P5ekX1KUxRQ79Jht9uzW7LorgwE65i5rWACL6LQe\"";
205 [ + - ]: 1 : std::string privkey2 = "\"Kyhdf5LuKTRx4ge69ybABsiUAWjVRK4XGxAKk2FQLp2HjGMy87Z4\"";
206 [ + - + - : 2 : r = CallRPC(std::string("signrawtransactionwithkey ")+notsigned+" [] "+prevout);
+ - + - ]
207 [ + - + - : 2 : BOOST_CHECK(r.get_obj().find_value("complete").get_bool() == false);
+ - + - +
- + - ]
208 [ + - + - : 4 : r = CallRPC(std::string("signrawtransactionwithkey ")+notsigned+" ["+privkey1+","+privkey2+"] "+prevout);
+ - + - +
- + - ]
209 [ + - + - : 2 : BOOST_CHECK(r.get_obj().find_value("complete").get_bool() == true);
+ - + - +
- ]
210 : 1 : }
211 : :
212 [ + - + - : 7 : BOOST_AUTO_TEST_CASE(rpc_createraw_op_return)
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- ]
213 : : {
214 [ + - + - : 2 : BOOST_CHECK_NO_THROW(CallRPC("createrawtransaction [{\"txid\":\"a3b807410df0b60fcb9736768df5823938b2f838694939ba45f3c0a1bff150ed\",\"vout\":0}] {\"data\":\"68656c6c6f776f726c64\"}"));
+ - + - -
- - - ]
215 : :
216 : : // Key not "data" (bad address)
217 [ + - + - : 3 : BOOST_CHECK_THROW(CallRPC("createrawtransaction [{\"txid\":\"a3b807410df0b60fcb9736768df5823938b2f838694939ba45f3c0a1bff150ed\",\"vout\":0}] {\"somedata\":\"68656c6c6f776f726c64\"}"), std::runtime_error);
- + - - -
- - + + -
+ - ]
218 : :
219 : : // Bad hex encoding of data output
220 [ + - + - : 3 : BOOST_CHECK_THROW(CallRPC("createrawtransaction [{\"txid\":\"a3b807410df0b60fcb9736768df5823938b2f838694939ba45f3c0a1bff150ed\",\"vout\":0}] {\"data\":\"12345\"}"), std::runtime_error);
- + - - -
- - + + -
+ - ]
221 [ + - + - : 3 : BOOST_CHECK_THROW(CallRPC("createrawtransaction [{\"txid\":\"a3b807410df0b60fcb9736768df5823938b2f838694939ba45f3c0a1bff150ed\",\"vout\":0}] {\"data\":\"12345g\"}"), std::runtime_error);
- + - - -
- - + + -
+ - ]
222 : :
223 : : // Data 81 bytes long
224 [ + - + - : 2 : BOOST_CHECK_NO_THROW(CallRPC("createrawtransaction [{\"txid\":\"a3b807410df0b60fcb9736768df5823938b2f838694939ba45f3c0a1bff150ed\",\"vout\":0}] {\"data\":\"010203040506070809101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081\"}"));
+ - + - -
- - - ]
225 : 1 : }
226 : :
227 [ + - + - : 7 : BOOST_AUTO_TEST_CASE(rpc_format_monetary_values)
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- ]
228 : : {
229 [ + - + - : 2 : BOOST_CHECK(ValueFromAmount(0LL).write() == "0.00000000");
+ - ]
230 [ + - + - : 2 : BOOST_CHECK(ValueFromAmount(1LL).write() == "0.00000001");
+ - ]
231 [ + - + - : 2 : BOOST_CHECK(ValueFromAmount(17622195LL).write() == "0.17622195");
+ - ]
232 [ + - + - : 2 : BOOST_CHECK(ValueFromAmount(50000000LL).write() == "0.50000000");
+ - ]
233 [ + - + - : 2 : BOOST_CHECK(ValueFromAmount(89898989LL).write() == "0.89898989");
+ - ]
234 [ + - + - : 2 : BOOST_CHECK(ValueFromAmount(100000000LL).write() == "1.00000000");
+ - ]
235 [ + - + - : 2 : BOOST_CHECK(ValueFromAmount(2099999999999990LL).write() == "20999999.99999990");
+ - ]
236 [ + - + - : 2 : BOOST_CHECK(ValueFromAmount(2099999999999999LL).write() == "20999999.99999999");
+ - ]
237 : :
238 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(ValueFromAmount(0).write(), "0.00000000");
239 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(ValueFromAmount((COIN/10000)*123456789).write(), "12345.67890000");
240 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(ValueFromAmount(-COIN).write(), "-1.00000000");
241 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(ValueFromAmount(-COIN/10).write(), "-0.10000000");
242 : :
243 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(ValueFromAmount(COIN*100000000).write(), "100000000.00000000");
244 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(ValueFromAmount(COIN*10000000).write(), "10000000.00000000");
245 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(ValueFromAmount(COIN*1000000).write(), "1000000.00000000");
246 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(ValueFromAmount(COIN*100000).write(), "100000.00000000");
247 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(ValueFromAmount(COIN*10000).write(), "10000.00000000");
248 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(ValueFromAmount(COIN*1000).write(), "1000.00000000");
249 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(ValueFromAmount(COIN*100).write(), "100.00000000");
250 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(ValueFromAmount(COIN*10).write(), "10.00000000");
251 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(ValueFromAmount(COIN).write(), "1.00000000");
252 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(ValueFromAmount(COIN/10).write(), "0.10000000");
253 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(ValueFromAmount(COIN/100).write(), "0.01000000");
254 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(ValueFromAmount(COIN/1000).write(), "0.00100000");
255 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(ValueFromAmount(COIN/10000).write(), "0.00010000");
256 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(ValueFromAmount(COIN/100000).write(), "0.00001000");
257 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(ValueFromAmount(COIN/1000000).write(), "0.00000100");
258 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(ValueFromAmount(COIN/10000000).write(), "0.00000010");
259 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(ValueFromAmount(COIN/100000000).write(), "0.00000001");
260 : :
261 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(ValueFromAmount(std::numeric_limits<CAmount>::max()).write(), "92233720368.54775807");
262 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(ValueFromAmount(std::numeric_limits<CAmount>::max() - 1).write(), "92233720368.54775806");
263 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(ValueFromAmount(std::numeric_limits<CAmount>::max() - 2).write(), "92233720368.54775805");
264 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(ValueFromAmount(std::numeric_limits<CAmount>::max() - 3).write(), "92233720368.54775804");
265 : : // ...
266 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(ValueFromAmount(std::numeric_limits<CAmount>::min() + 3).write(), "-92233720368.54775805");
267 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(ValueFromAmount(std::numeric_limits<CAmount>::min() + 2).write(), "-92233720368.54775806");
268 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(ValueFromAmount(std::numeric_limits<CAmount>::min() + 1).write(), "-92233720368.54775807");
269 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(ValueFromAmount(std::numeric_limits<CAmount>::min()).write(), "-92233720368.54775808");
270 : 1 : }
271 : :
272 : 27 : static UniValue ValueFromString(const std::string& str) noexcept
273 : : {
274 : 27 : UniValue value;
275 : 27 : value.setNumStr(str);
276 : 27 : return value;
277 : : }
278 : :
279 [ + - + - : 7 : BOOST_AUTO_TEST_CASE(rpc_parse_monetary_values)
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- ]
280 : : {
281 [ + - + - : 3 : BOOST_CHECK_THROW(AmountFromValue(ValueFromString("-0.00000001")), UniValue);
- + - - -
- - + + -
+ - ]
282 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(AmountFromValue(ValueFromString("0")), 0LL);
283 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(AmountFromValue(ValueFromString("0.00000000")), 0LL);
284 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(AmountFromValue(ValueFromString("0.00000001")), 1LL);
285 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(AmountFromValue(ValueFromString("0.17622195")), 17622195LL);
286 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(AmountFromValue(ValueFromString("0.5")), 50000000LL);
287 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(AmountFromValue(ValueFromString("0.50000000")), 50000000LL);
288 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(AmountFromValue(ValueFromString("0.89898989")), 89898989LL);
289 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(AmountFromValue(ValueFromString("1.00000000")), 100000000LL);
290 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(AmountFromValue(ValueFromString("20999999.9999999")), 2099999999999990LL);
291 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(AmountFromValue(ValueFromString("20999999.99999999")), 2099999999999999LL);
292 : :
293 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(AmountFromValue(ValueFromString("1e-8")), COIN/100000000);
294 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(AmountFromValue(ValueFromString("0.1e-7")), COIN/100000000);
295 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(AmountFromValue(ValueFromString("0.01e-6")), COIN/100000000);
296 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(AmountFromValue(ValueFromString("0.00000000000000000000000000000000000001e+30")), 1);
297 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(AmountFromValue(ValueFromString("0.0000000000000000000000000000000000000000000000000000000000000000000000000001e+68")), COIN/100000000);
298 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(AmountFromValue(ValueFromString("10000000000000000000000000000000000000000000000000000000000000000e-64")), COIN);
299 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(AmountFromValue(ValueFromString("0.000000000000000000000000000000000000000000000000000000000000000100000000000000000000000000000000000000000000000000000e64")), COIN);
300 : :
301 [ + - + - : 3 : BOOST_CHECK_THROW(AmountFromValue(ValueFromString("1e-9")), UniValue); //should fail
- + - - -
- - + + -
+ - ]
302 [ + - + - : 3 : BOOST_CHECK_THROW(AmountFromValue(ValueFromString("0.000000019")), UniValue); //should fail
- + - - -
- - + + -
+ - ]
303 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(AmountFromValue(ValueFromString("0.00000001000000")), 1LL); //should pass, cut trailing 0
304 [ + - + - : 3 : BOOST_CHECK_THROW(AmountFromValue(ValueFromString("19e-9")), UniValue); //should fail
- + - - -
- - + + -
+ - ]
305 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(AmountFromValue(ValueFromString("0.19e-6")), 19); //should pass, leading 0 is present
306 [ + - + - : 4 : BOOST_CHECK_EXCEPTION(AmountFromValue(".19e-6"), UniValue, HasJSON(R"({"code":-3,"message":"Invalid amount"})")); //should fail, no leading 0
- + - - -
- - + + -
+ - + - +
- ]
307 : :
308 [ + - + - : 3 : BOOST_CHECK_THROW(AmountFromValue(ValueFromString("92233720368.54775808")), UniValue); //overflow error
- + - - -
- - + + -
+ - ]
309 [ + - + - : 3 : BOOST_CHECK_THROW(AmountFromValue(ValueFromString("1e+11")), UniValue); //overflow error
- + - - -
- - + + -
+ - ]
310 [ + - + - : 3 : BOOST_CHECK_THROW(AmountFromValue(ValueFromString("1e11")), UniValue); //overflow error signless
- + - - -
- - + + -
+ - ]
311 [ + - + - : 3 : BOOST_CHECK_THROW(AmountFromValue(ValueFromString("93e+9")), UniValue); //overflow error
- + - - -
- - + + -
+ - ]
312 : 1 : }
313 : :
314 [ + - + - : 7 : BOOST_AUTO_TEST_CASE(rpc_ban)
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- ]
315 : : {
316 [ + - + - : 2 : BOOST_CHECK_NO_THROW(CallRPC(std::string("clearbanned")));
+ - + - -
- - - ]
317 : :
318 [ + - ]: 1 : UniValue r;
319 [ + - + - : 2 : BOOST_CHECK_NO_THROW(r = CallRPC(std::string("setban 127.0.0.0 add")));
+ - + - -
- - - -
- ]
320 [ + - + - : 3 : BOOST_CHECK_THROW(r = CallRPC(std::string("setban 127.0.0.0:8334")), std::runtime_error); //portnumber for setban not allowed
- + - - -
- - + + -
+ - ]
321 [ + - + - : 2 : BOOST_CHECK_NO_THROW(r = CallRPC(std::string("listbanned")));
+ - + - -
- - - -
- ]
322 [ + - + - ]: 1 : UniValue ar = r.get_array();
323 [ + - + - : 1 : UniValue o1 = ar[0].get_obj();
+ - ]
324 [ + - + - ]: 1 : UniValue adr = o1.find_value("address");
325 [ + - + - : 1 : BOOST_CHECK_EQUAL(adr.get_str(), "127.0.0.0/32");
+ - ]
326 [ + - + - : 2 : BOOST_CHECK_NO_THROW(CallRPC(std::string("setban 127.0.0.0 remove")));
+ - + - -
- - - -
- ]
327 [ + - + - : 2 : BOOST_CHECK_NO_THROW(r = CallRPC(std::string("listbanned")));
+ - + - -
- - - -
- ]
328 [ + - + - ]: 1 : ar = r.get_array();
329 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(ar.size(), 0U);
330 : :
331 [ + - + - : 2 : BOOST_CHECK_NO_THROW(r = CallRPC(std::string("setban 127.0.0.0/24 add 9907731200 true")));
+ - + - -
- - - -
- ]
332 [ + - + - : 2 : BOOST_CHECK_NO_THROW(r = CallRPC(std::string("listbanned")));
+ - + - -
- - - -
- ]
333 [ + - + - ]: 1 : ar = r.get_array();
334 [ + - + - : 1 : o1 = ar[0].get_obj();
+ - ]
335 [ + - + - ]: 1 : adr = o1.find_value("address");
336 [ + - + - ]: 1 : int64_t banned_until{o1.find_value("banned_until").getInt<int64_t>()};
337 [ + - + - : 1 : BOOST_CHECK_EQUAL(adr.get_str(), "127.0.0.0/24");
+ - ]
338 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(banned_until, 9907731200); // absolute time check
339 : :
340 [ + - + - : 2 : BOOST_CHECK_NO_THROW(CallRPC(std::string("clearbanned")));
+ - + - -
- - - -
- ]
341 : :
342 : 1 : auto now = 10'000s;
343 [ + - ]: 1 : SetMockTime(now);
344 [ + - + - : 2 : BOOST_CHECK_NO_THROW(r = CallRPC(std::string("setban 127.0.0.0/24 add 200")));
+ - + - -
- - - -
- ]
345 [ + - ]: 1 : SetMockTime(now += 2s);
346 : 1 : const int64_t time_remaining_expected{198};
347 [ + - + - : 2 : BOOST_CHECK_NO_THROW(r = CallRPC(std::string("listbanned")));
+ - + - -
- - - -
- ]
348 [ + - + - ]: 1 : ar = r.get_array();
349 [ + - + - : 1 : o1 = ar[0].get_obj();
+ - ]
350 [ + - + - ]: 1 : adr = o1.find_value("address");
351 [ + - + - ]: 1 : banned_until = o1.find_value("banned_until").getInt<int64_t>();
352 [ + - + - ]: 1 : const int64_t ban_created{o1.find_value("ban_created").getInt<int64_t>()};
353 [ + - + - ]: 1 : const int64_t ban_duration{o1.find_value("ban_duration").getInt<int64_t>()};
354 [ + - + - ]: 1 : const int64_t time_remaining{o1.find_value("time_remaining").getInt<int64_t>()};
355 [ + - + - : 1 : BOOST_CHECK_EQUAL(adr.get_str(), "127.0.0.0/24");
+ - ]
356 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(banned_until, time_remaining_expected + now.count());
357 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(ban_duration, banned_until - ban_created);
358 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(time_remaining, time_remaining_expected);
359 : :
360 : : // must throw an exception because 127.0.0.1 is in already banned subnet range
361 [ + - + - : 3 : BOOST_CHECK_THROW(r = CallRPC(std::string("setban 127.0.0.1 add")), std::runtime_error);
- + - - -
- - + + -
+ - ]
362 : :
363 [ + - + - : 2 : BOOST_CHECK_NO_THROW(CallRPC(std::string("setban 127.0.0.0/24 remove")));
+ - + - -
- - - -
- ]
364 [ + - + - : 2 : BOOST_CHECK_NO_THROW(r = CallRPC(std::string("listbanned")));
+ - + - -
- - - -
- ]
365 [ + - + - ]: 1 : ar = r.get_array();
366 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(ar.size(), 0U);
367 : :
368 [ + - + - : 2 : BOOST_CHECK_NO_THROW(r = CallRPC(std::string("setban 127.0.0.0/255.255.0.0 add")));
+ - + - -
- - - -
- ]
369 [ + - + - : 3 : BOOST_CHECK_THROW(r = CallRPC(std::string("setban 127.0.1.1 add")), std::runtime_error);
- + - - -
- - + + -
+ - ]
370 : :
371 [ + - + - : 2 : BOOST_CHECK_NO_THROW(CallRPC(std::string("clearbanned")));
+ - + - -
- - - -
- ]
372 [ + - + - : 2 : BOOST_CHECK_NO_THROW(r = CallRPC(std::string("listbanned")));
+ - + - -
- - - -
- ]
373 [ + - + - ]: 1 : ar = r.get_array();
374 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(ar.size(), 0U);
375 : :
376 : :
377 [ + - + - : 3 : BOOST_CHECK_THROW(r = CallRPC(std::string("setban test add")), std::runtime_error); //invalid IP
- + - - -
- - + + -
+ - ]
378 : :
379 : : //IPv6 tests
380 [ + - + - : 2 : BOOST_CHECK_NO_THROW(r = CallRPC(std::string("setban FE80:0000:0000:0000:0202:B3FF:FE1E:8329 add")));
+ - + - -
- - - -
- ]
381 [ + - + - : 2 : BOOST_CHECK_NO_THROW(r = CallRPC(std::string("listbanned")));
+ - + - -
- - - -
- ]
382 [ + - + - ]: 1 : ar = r.get_array();
383 [ + - + - : 1 : o1 = ar[0].get_obj();
+ - ]
384 [ + - + - ]: 1 : adr = o1.find_value("address");
385 [ + - + - : 1 : BOOST_CHECK_EQUAL(adr.get_str(), "fe80::202:b3ff:fe1e:8329/128");
+ - ]
386 : :
387 [ + - + - : 2 : BOOST_CHECK_NO_THROW(CallRPC(std::string("clearbanned")));
+ - + - -
- - - -
- ]
388 [ + - + - : 2 : BOOST_CHECK_NO_THROW(r = CallRPC(std::string("setban 2001:db8::/ffff:fffc:0:0:0:0:0:0 add")));
+ - + - -
- - - -
- ]
389 [ + - + - : 2 : BOOST_CHECK_NO_THROW(r = CallRPC(std::string("listbanned")));
+ - + - -
- - - -
- ]
390 [ + - + - ]: 1 : ar = r.get_array();
391 [ + - + - : 1 : o1 = ar[0].get_obj();
+ - ]
392 [ + - + - ]: 1 : adr = o1.find_value("address");
393 [ + - + - : 1 : BOOST_CHECK_EQUAL(adr.get_str(), "2001:db8::/30");
+ - ]
394 : :
395 [ + - + - : 2 : BOOST_CHECK_NO_THROW(CallRPC(std::string("clearbanned")));
+ - + - -
- - - -
- ]
396 [ + - + - : 2 : BOOST_CHECK_NO_THROW(r = CallRPC(std::string("setban 2001:4d48:ac57:400:cacf:e9ff:fe1d:9c63/128 add")));
+ - + - -
- - - -
- ]
397 [ + - + - : 2 : BOOST_CHECK_NO_THROW(r = CallRPC(std::string("listbanned")));
+ - + - -
- - - -
- ]
398 [ + - + - ]: 1 : ar = r.get_array();
399 [ + - + - : 1 : o1 = ar[0].get_obj();
+ - ]
400 [ + - + - ]: 1 : adr = o1.find_value("address");
401 [ + - + - : 1 : BOOST_CHECK_EQUAL(adr.get_str(), "2001:4d48:ac57:400:cacf:e9ff:fe1d:9c63/128");
+ - ]
402 : 1 : }
403 : :
404 [ + - + - : 7 : BOOST_AUTO_TEST_CASE(rpc_convert_values_generatetoaddress)
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- ]
405 : : {
406 [ + - ]: 1 : UniValue result;
407 : :
408 [ + - + - : 2 : BOOST_CHECK_NO_THROW(result = RPCConvertValues("generatetoaddress", {"101", "mkESjLZW66TmHhiFX8MCaBjrhZ543PPh9a"}));
+ - + - +
- - - - -
- - ]
409 [ + - + - : 1 : BOOST_CHECK_EQUAL(result[0].getInt<int>(), 101);
+ - + - ]
410 [ + - + - : 1 : BOOST_CHECK_EQUAL(result[1].get_str(), "mkESjLZW66TmHhiFX8MCaBjrhZ543PPh9a");
+ - + - ]
411 : :
412 [ + - + - : 2 : BOOST_CHECK_NO_THROW(result = RPCConvertValues("generatetoaddress", {"101", "mhMbmE2tE9xzJYCV9aNC8jKWN31vtGrguU"}));
+ - + - +
- - - - -
- - ]
413 [ + - + - : 1 : BOOST_CHECK_EQUAL(result[0].getInt<int>(), 101);
+ - + - ]
414 [ + - + - : 1 : BOOST_CHECK_EQUAL(result[1].get_str(), "mhMbmE2tE9xzJYCV9aNC8jKWN31vtGrguU");
+ - + - ]
415 : :
416 [ + - + - : 2 : BOOST_CHECK_NO_THROW(result = RPCConvertValues("generatetoaddress", {"1", "mkESjLZW66TmHhiFX8MCaBjrhZ543PPh9a", "9"}));
+ - + - +
- - - - -
- - ]
417 [ + - + - : 1 : BOOST_CHECK_EQUAL(result[0].getInt<int>(), 1);
+ - + - ]
418 [ + - + - : 1 : BOOST_CHECK_EQUAL(result[1].get_str(), "mkESjLZW66TmHhiFX8MCaBjrhZ543PPh9a");
+ - + - ]
419 [ + - + - : 1 : BOOST_CHECK_EQUAL(result[2].getInt<int>(), 9);
+ - + - ]
420 : :
421 [ + - + - : 2 : BOOST_CHECK_NO_THROW(result = RPCConvertValues("generatetoaddress", {"1", "mhMbmE2tE9xzJYCV9aNC8jKWN31vtGrguU", "9"}));
+ - + - +
- - - - -
- - ]
422 [ + - + - : 1 : BOOST_CHECK_EQUAL(result[0].getInt<int>(), 1);
+ - + - ]
423 [ + - + - : 1 : BOOST_CHECK_EQUAL(result[1].get_str(), "mhMbmE2tE9xzJYCV9aNC8jKWN31vtGrguU");
+ - + - ]
424 [ + - + - : 1 : BOOST_CHECK_EQUAL(result[2].getInt<int>(), 9);
+ - + - ]
425 : 1 : }
426 : :
427 [ + - + - : 7 : BOOST_AUTO_TEST_CASE(rpc_getblockstats_calculate_percentiles_by_weight)
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- ]
428 : : {
429 : 1 : int64_t total_weight = 200;
430 : 1 : std::vector<std::pair<CAmount, int64_t>> feerates;
431 : 1 : CAmount result[NUM_GETBLOCKSTATS_PERCENTILES] = { 0 };
432 : :
433 [ + + ]: 101 : for (int64_t i = 0; i < 100; i++) {
434 [ + - ]: 100 : feerates.emplace_back(1 ,1);
435 : : }
436 : :
437 [ + + ]: 101 : for (int64_t i = 0; i < 100; i++) {
438 [ + - ]: 100 : feerates.emplace_back(2 ,1);
439 : : }
440 : :
441 [ + - ]: 1 : CalculatePercentilesByWeight(result, feerates, total_weight);
442 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(result[0], 1);
443 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(result[1], 1);
444 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(result[2], 1);
445 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(result[3], 2);
446 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(result[4], 2);
447 : :
448 : : // Test with more pairs, and two pairs overlapping 2 percentiles.
449 : 1 : total_weight = 100;
450 : 1 : CAmount result2[NUM_GETBLOCKSTATS_PERCENTILES] = { 0 };
451 [ + - ]: 1 : feerates.clear();
452 : :
453 [ + - ]: 1 : feerates.emplace_back(1, 9);
454 [ + - ]: 1 : feerates.emplace_back(2 , 16); //10th + 25th percentile
455 [ + - ]: 1 : feerates.emplace_back(4 ,50); //50th + 75th percentile
456 [ + - ]: 1 : feerates.emplace_back(5 ,10);
457 [ + - ]: 1 : feerates.emplace_back(9 ,15); // 90th percentile
458 : :
459 [ + - ]: 1 : CalculatePercentilesByWeight(result2, feerates, total_weight);
460 : :
461 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(result2[0], 2);
462 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(result2[1], 2);
463 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(result2[2], 4);
464 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(result2[3], 4);
465 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(result2[4], 9);
466 : :
467 : : // Same test as above, but one of the percentile-overlapping pairs is split in 2.
468 : 1 : total_weight = 100;
469 : 1 : CAmount result3[NUM_GETBLOCKSTATS_PERCENTILES] = { 0 };
470 [ + - ]: 1 : feerates.clear();
471 : :
472 [ + - ]: 1 : feerates.emplace_back(1, 9);
473 [ + - ]: 1 : feerates.emplace_back(2 , 11); // 10th percentile
474 [ + - ]: 1 : feerates.emplace_back(2 , 5); // 25th percentile
475 [ + - ]: 1 : feerates.emplace_back(4 ,50); //50th + 75th percentile
476 [ + - ]: 1 : feerates.emplace_back(5 ,10);
477 [ + - ]: 1 : feerates.emplace_back(9 ,15); // 90th percentile
478 : :
479 [ + - ]: 1 : CalculatePercentilesByWeight(result3, feerates, total_weight);
480 : :
481 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(result3[0], 2);
482 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(result3[1], 2);
483 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(result3[2], 4);
484 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(result3[3], 4);
485 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(result3[4], 9);
486 : :
487 : : // Test with one transaction spanning all percentiles.
488 : 1 : total_weight = 104;
489 : 1 : CAmount result4[NUM_GETBLOCKSTATS_PERCENTILES] = { 0 };
490 [ + - ]: 1 : feerates.clear();
491 : :
492 [ + - ]: 1 : feerates.emplace_back(1, 100);
493 [ + - ]: 1 : feerates.emplace_back(2, 1);
494 [ + - ]: 1 : feerates.emplace_back(3, 1);
495 [ + - ]: 1 : feerates.emplace_back(3, 1);
496 [ + - ]: 1 : feerates.emplace_back(999999, 1);
497 : :
498 [ + - ]: 1 : CalculatePercentilesByWeight(result4, feerates, total_weight);
499 : :
500 [ + + ]: 6 : for (int64_t i = 0; i < NUM_GETBLOCKSTATS_PERCENTILES; i++) {
501 [ + - + - ]: 5 : BOOST_CHECK_EQUAL(result4[i], 1);
502 : : }
503 : 1 : }
504 : :
505 : : // Make sure errors are triggered appropriately if parameters have the same names.
506 [ + - + - : 7 : BOOST_AUTO_TEST_CASE(check_dup_param_names)
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- ]
507 : : {
508 : 1 : enum ParamType { POSITIONAL, NAMED, NAMED_ONLY };
509 : 20 : auto make_rpc = [](std::vector<std::tuple<std::string, ParamType>> param_names) {
510 : 19 : std::vector<RPCArg> params;
511 : 19 : std::vector<RPCArg> options;
512 [ + + + - ]: 49 : 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)); };
513 [ + + + + ]: 57 : for (auto& [param_name, param_type] : param_names) {
514 [ + + ]: 38 : if (param_type == POSITIONAL) {
515 [ + - ]: 13 : push_options();
516 [ + - ]: 13 : params.emplace_back(std::move(param_name), RPCArg::Type::NUM, RPCArg::Optional::OMITTED, "description");
517 : : } else {
518 [ + - ]: 50 : options.emplace_back(std::move(param_name), RPCArg::Type::NUM, RPCArg::Optional::OMITTED, "description", RPCArgOptions{.also_positional = param_type == NAMED});
519 : : }
520 : : }
521 [ + - ]: 19 : push_options();
522 [ + - + - : 119 : return RPCHelpMan{"method_name", "description", params, RPCResults{}, RPCExamples{""}};
+ - + - +
- + + ]
523 : 27 : };
524 : :
525 : : // No errors if parameter names are unique.
526 [ + - + - : 4 : make_rpc({{"p1", POSITIONAL}, {"p2", POSITIONAL}});
+ + - - ]
527 [ + - + - : 4 : make_rpc({{"p1", POSITIONAL}, {"p2", NAMED}});
+ + - - ]
528 [ + - + - : 4 : make_rpc({{"p1", POSITIONAL}, {"p2", NAMED_ONLY}});
+ + - - ]
529 [ + - + - : 4 : make_rpc({{"p1", NAMED}, {"p2", POSITIONAL}});
+ + - - ]
530 [ + - + - : 4 : make_rpc({{"p1", NAMED}, {"p2", NAMED}});
+ + - - ]
531 [ + - + - : 4 : make_rpc({{"p1", NAMED}, {"p2", NAMED_ONLY}});
+ + - - ]
532 [ + - + - : 4 : make_rpc({{"p1", NAMED_ONLY}, {"p2", POSITIONAL}});
+ + - - ]
533 [ + - + - : 4 : make_rpc({{"p1", NAMED_ONLY}, {"p2", NAMED}});
+ + - - ]
534 [ + - + - : 4 : make_rpc({{"p1", NAMED_ONLY}, {"p2", NAMED_ONLY}});
+ + - - ]
535 : :
536 : : // Error if parameters names are duplicates, unless one parameter is
537 : : // positional and the other is named and .also_positional is true.
538 [ + - + - : 7 : BOOST_CHECK_THROW(make_rpc({{"p1", POSITIONAL}, {"p1", POSITIONAL}}), NonFatalCheckError);
- + - - -
- - - + +
- + + - +
- ]
539 [ + - + - : 4 : make_rpc({{"p1", POSITIONAL}, {"p1", NAMED}});
+ + - - ]
540 [ + - + - : 7 : BOOST_CHECK_THROW(make_rpc({{"p1", POSITIONAL}, {"p1", NAMED_ONLY}}), NonFatalCheckError);
- + - - -
- - - + +
- + + - +
- ]
541 [ + - + - : 4 : make_rpc({{"p1", NAMED}, {"p1", POSITIONAL}});
+ + - - ]
542 [ + - + - : 7 : BOOST_CHECK_THROW(make_rpc({{"p1", NAMED}, {"p1", NAMED}}), NonFatalCheckError);
- + - - -
- - - + +
- + + - +
- ]
543 [ + - + - : 7 : BOOST_CHECK_THROW(make_rpc({{"p1", NAMED}, {"p1", NAMED_ONLY}}), NonFatalCheckError);
- + - - -
- - - + +
- + + - +
- ]
544 [ + - + - : 7 : BOOST_CHECK_THROW(make_rpc({{"p1", NAMED_ONLY}, {"p1", POSITIONAL}}), NonFatalCheckError);
- + - - -
- - - + +
- + + - +
- ]
545 [ + - + - : 7 : BOOST_CHECK_THROW(make_rpc({{"p1", NAMED_ONLY}, {"p1", NAMED}}), NonFatalCheckError);
- + - - -
- - - + +
- + + - +
- ]
546 [ + - + - : 7 : BOOST_CHECK_THROW(make_rpc({{"p1", NAMED_ONLY}, {"p1", NAMED_ONLY}}), NonFatalCheckError);
- + - - -
- - - + +
- + + - +
- ]
547 : :
548 : : // Make sure duplicate aliases are detected too.
549 [ + - + - : 7 : BOOST_CHECK_THROW(make_rpc({{"p1", POSITIONAL}, {"p2|p1", NAMED_ONLY}}), NonFatalCheckError);
- + - - -
- - - + +
- + + - +
- ]
550 [ + - + - : 47 : }
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - - - -
- - - - -
- - - - -
- - - - -
- + - - -
+ - - - +
- + - + -
+ - + -
+ ]
551 : :
552 [ + - + - : 7 : BOOST_AUTO_TEST_CASE(help_example)
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- ]
553 : : {
554 : : // test different argument types
555 [ + - + + : 5 : const RPCArgList& args = {{"foo", "bar"}, {"b", true}, {"n", 1}};
- - ]
556 [ + - + - : 1 : BOOST_CHECK_EQUAL(HelpExampleCliNamed("test", args), "> bitcoin-cli -named test foo=bar b=true n=1\n");
+ - + - ]
557 [ + - + - : 1 : 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");
+ - + - ]
558 : :
559 : : // test shell escape
560 [ + - + - : 3 : BOOST_CHECK_EQUAL(HelpExampleCliNamed("test", {{"foo", "b'ar"}}), "> bitcoin-cli -named test foo='b'''ar'\n");
+ - + - +
- + + -
- ]
561 [ + - + - : 3 : BOOST_CHECK_EQUAL(HelpExampleCliNamed("test", {{"foo", "b\"ar"}}), "> bitcoin-cli -named test foo='b\"ar'\n");
+ - + - +
- + + -
- ]
562 [ + - + - : 3 : BOOST_CHECK_EQUAL(HelpExampleCliNamed("test", {{"foo", "b ar"}}), "> bitcoin-cli -named test foo='b ar'\n");
+ - + - +
- + + -
- ]
563 : :
564 : : // test object params
565 : 1 : UniValue obj_value(UniValue::VOBJ);
566 [ + - + - : 2 : obj_value.pushKV("foo", "bar");
+ - ]
567 [ + - + - : 2 : obj_value.pushKV("b", false);
+ - ]
568 [ + - + - : 2 : obj_value.pushKV("n", 1);
+ - ]
569 [ + - + - : 3 : BOOST_CHECK_EQUAL(HelpExampleCliNamed("test", {{"name", obj_value}}), "> bitcoin-cli -named test name='{\"foo\":\"bar\",\"b\":false,\"n\":1}'\n");
+ - + - +
- + + -
- ]
570 [ + - + - : 3 : 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");
+ - + - +
- + + -
- ]
571 : :
572 : : // test array params
573 : 1 : UniValue arr_value(UniValue::VARR);
574 [ + - + - ]: 1 : arr_value.push_back("bar");
575 [ + - + - ]: 1 : arr_value.push_back(false);
576 [ + - + - ]: 1 : arr_value.push_back(1);
577 [ + - + - : 3 : BOOST_CHECK_EQUAL(HelpExampleCliNamed("test", {{"name", arr_value}}), "> bitcoin-cli -named test name='[\"bar\",false,1]'\n");
+ - + - +
- + + -
- ]
578 [ + - + - : 3 : 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");
+ - + - +
- + + -
- ]
579 : :
580 : : // test types don't matter for shell
581 [ + - + - : 6 : BOOST_CHECK_EQUAL(HelpExampleCliNamed("foo", {{"arg", true}}), HelpExampleCliNamed("foo", {{"arg", "true"}}));
+ - + - +
- + - + -
+ - + + +
+ - - -
- ]
582 : :
583 : : // test types matter for Rpc
584 [ + - + - : 6 : BOOST_CHECK_NE(HelpExampleRpcNamed("foo", {{"arg", true}}), HelpExampleRpcNamed("foo", {{"arg", "true"}}));
+ - + - +
- + - + -
+ - + + +
+ - - -
- ]
585 [ + - + - : 13 : }
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - -
- ]
586 : :
587 : 2 : static void CheckRpc(const std::vector<RPCArg>& params, const UniValue& args, RPCHelpMan::RPCMethodImpl test_impl)
588 : : {
589 [ + - + - ]: 2 : auto null_result{RPCResult{RPCResult::Type::NONE, "", "None"}};
590 [ + - + - : 6 : const RPCHelpMan rpc{"dummy", "dummy description", params, null_result, RPCExamples{""}, test_impl};
+ - + - +
- + - + -
+ - ]
591 : 2 : JSONRPCRequest req;
592 [ + - ]: 2 : req.params = args;
593 : :
594 [ + - ]: 2 : rpc.HandleRequest(req);
595 : 2 : }
596 : :
597 [ + - + - : 7 : BOOST_AUTO_TEST_CASE(rpc_arg_helper)
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- ]
598 : : {
599 : 1 : constexpr bool DEFAULT_BOOL = true;
600 : 1 : constexpr auto DEFAULT_STRING = "default";
601 : 1 : constexpr uint64_t DEFAULT_UINT64_T = 3;
602 : :
603 : : //! Parameters with which the RPCHelpMan is instantiated
604 : 1 : const std::vector<RPCArg> params{
605 : : // Required arg
606 [ + - ]: 1 : {"req_int", RPCArg::Type::NUM, RPCArg::Optional::NO, ""},
607 [ + - ]: 1 : {"req_str", RPCArg::Type::STR, RPCArg::Optional::NO, ""},
608 : : // Default arg
609 [ + - ]: 2 : {"def_uint64_t", RPCArg::Type::NUM, RPCArg::Default{DEFAULT_UINT64_T}, ""},
610 [ + - ]: 2 : {"def_string", RPCArg::Type::STR, RPCArg::Default{DEFAULT_STRING}, ""},
611 [ + - ]: 2 : {"def_bool", RPCArg::Type::BOOL, RPCArg::Default{DEFAULT_BOOL}, ""},
612 : : // Optional arg without default
613 [ + - ]: 1 : {"opt_double", RPCArg::Type::NUM, RPCArg::Optional::OMITTED, ""},
614 [ + - ]: 1 : {"opt_string", RPCArg::Type::STR, RPCArg::Optional::OMITTED, ""}
615 [ + - + - : 29 : };
+ - + - +
- + - + -
+ - + - +
- + - + -
+ - + - +
- + + -
- ]
616 : :
617 : : //! Check that `self.Arg` returns the same value as the `request.params` accessors
618 [ + - ]: 3 : RPCHelpMan::RPCMethodImpl check_positional = [&](const RPCHelpMan& self, const JSONRPCRequest& request) -> UniValue {
619 [ + - ]: 2 : BOOST_CHECK_EQUAL(self.Arg<int>("req_int"), request.params[0].getInt<int>());
620 [ + - ]: 2 : BOOST_CHECK_EQUAL(self.Arg<std::string>("req_str"), request.params[1].get_str());
621 [ + + + - ]: 2 : BOOST_CHECK_EQUAL(self.Arg<uint64_t>("def_uint64_t"), request.params[2].isNull() ? DEFAULT_UINT64_T : request.params[2].getInt<uint64_t>());
622 [ + + + - : 2 : BOOST_CHECK_EQUAL(self.Arg<std::string>("def_string"), request.params[3].isNull() ? DEFAULT_STRING : request.params[3].get_str());
+ - ]
623 [ + + + - ]: 2 : BOOST_CHECK_EQUAL(self.Arg<bool>("def_bool"), request.params[4].isNull() ? DEFAULT_BOOL : request.params[4].get_bool());
624 [ + + ]: 2 : if (!request.params[5].isNull()) {
625 [ + - + - ]: 1 : BOOST_CHECK_EQUAL(self.MaybeArg<double>("opt_double").value(), request.params[5].get_real());
626 : : } else {
627 [ + - + - ]: 2 : BOOST_CHECK(!self.MaybeArg<double>("opt_double"));
628 : : }
629 [ + + ]: 2 : if (!request.params[6].isNull()) {
630 [ + - + - ]: 2 : BOOST_CHECK(self.MaybeArg<std::string>("opt_string"));
631 [ + - ]: 1 : BOOST_CHECK_EQUAL(*self.MaybeArg<std::string>("opt_string"), request.params[6].get_str());
632 : : } else {
633 [ + - + - ]: 2 : BOOST_CHECK(!self.MaybeArg<std::string>("opt_string"));
634 : : }
635 : 2 : return UniValue{};
636 : 1 : };
637 [ + - + - : 1 : CheckRpc(params, UniValue{JSON(R"([5, "hello", null, null, null, null, null])")}, check_positional);
+ - ]
638 [ + - + - : 1 : CheckRpc(params, UniValue{JSON(R"([5, "hello", 4, "test", true, 1.23, "world"])")}, check_positional);
+ - ]
639 [ + - + - : 16 : }
+ - + - +
- + - + -
+ - - - ]
640 : :
641 : : BOOST_AUTO_TEST_SUITE_END()
|