1#include <xrpld/rpc/RPCCall.h>
2#include <xrpld/rpc/ServerHandler.h>
4#include <xrpl/basics/ByteUtilities.h>
5#include <xrpl/basics/Log.h>
6#include <xrpl/basics/StringUtilities.h>
7#include <xrpl/basics/base64.h>
8#include <xrpl/basics/contract.h>
9#include <xrpl/beast/core/LexicalCast.h>
10#include <xrpl/json/json_forwards.h>
11#include <xrpl/json/json_reader.h>
12#include <xrpl/json/to_string.h>
13#include <xrpl/net/HTTPClient.h>
14#include <xrpl/protocol/ApiVersion.h>
15#include <xrpl/protocol/ErrorCodes.h>
16#include <xrpl/protocol/PublicKey.h>
17#include <xrpl/protocol/RPCErr.h>
18#include <xrpl/protocol/SystemParameters.h>
19#include <xrpl/protocol/UintTypes.h>
20#include <xrpl/protocol/jss.h>
22#include <boost/algorithm/string/predicate.hpp>
23#include <boost/asio/streambuf.hpp>
24#include <boost/regex.hpp>
55 s <<
"POST " << (strPath.
empty() ?
"/" : strPath) <<
" HTTP/1.0\r\n"
56 <<
"User-Agent: " <<
systemName() <<
"-json-rpc/v1\r\n"
57 <<
"Host: " << strHost <<
"\r\n"
58 <<
"Content-Type: application/json\r\n"
59 <<
"Content-Length: " << strMsg.
size() <<
"\r\n"
60 <<
"Accept: application/json\r\n";
62 for (
auto const& [k, v] : mapRequestHeaders)
63 s << k <<
": " << v <<
"\r\n";
65 s <<
"\r\n" << strMsg;
81 if (strLedger ==
"current" || strLedger ==
"closed" || strLedger ==
"validated")
83 jvRequest[jss::ledger_index] = strLedger;
85 else if (strLedger.
length() == 64)
88 jvRequest[jss::ledger_hash] = strLedger;
92 jvRequest[jss::ledger_index] = beast::lexicalCast<std::uint32_t>(strLedger);
107 static boost::regex reCurIss(
"\\`([][:alnum:]<>(){}[|?!@#$%^&*]{3})(?:/(.+))?\\'");
109 boost::smatch smMatch;
111 if (boost::regex_match(strCurrencyIssuer, smMatch, reCurIss))
117 jvResult[jss::currency] = strCurrency;
122 jvResult[jss::issuer] = strIssuer;
136 if (parseBase58<xrpl::PublicKey>(type, strPk))
158 v[jss::params] = jvParams;
167 v[jss::internal_command] = jvParams[0u];
171 for (
unsigned i = 1; i < jvParams.
size(); ++i)
172 params.
append(jvParams[i]);
174 v[jss::params] = params;
182 if (jvParams.
size() == 1)
190 jvRequest[jss::public_key] = strPk;
203 unsigned int iParams = jvParams.
size();
206 jvRequest[jvParams[0u].
asString()] =
true;
217 unsigned int iParams = jvParams.
size();
219 auto const account = parseBase58<AccountID>(jvParams[0u].asString());
223 jvRequest[jss::account] =
toBase58(*account);
227 while (!bDone && iParams >= 2)
230 if (jvParams[iParams - 1].asString() == jss::binary)
232 jvRequest[jss::binary] =
true;
235 else if (jvParams[iParams - 1].asString() == jss::count)
237 jvRequest[jss::count] =
true;
240 else if (jvParams[iParams - 1].asString() == jss::descending)
242 jvRequest[jss::descending] =
true;
254 else if (2 == iParams)
264 if (uLedgerMax != -1 && uLedgerMax < uLedgerMin)
271 jvRequest[jss::ledger_index_min] = jvParams[1u].
asInt();
272 jvRequest[jss::ledger_index_max] = jvParams[2u].
asInt();
275 jvRequest[jss::limit] = jvParams[3u].
asInt();
278 jvRequest[jss::offset] = jvParams[4u].
asInt();
302 jvRequest[jss::taker_pays] = jvTakerPays;
311 jvRequest[jss::taker_gets] = jvTakerGets;
314 if (jvParams.
size() >= 3)
316 jvRequest[jss::issuer] = jvParams[2u].
asString();
322 if (jvParams.
size() >= 5)
326 int iLimit = jvParams[4u].
asInt();
329 jvRequest[jss::limit] = iLimit;
337 if (jvParams.
size() >= 6)
341 int bProof = jvParams[5u].
asInt();
343 jvRequest[jss::proof] =
true;
351 if (jvParams.
size() == 7)
352 jvRequest[jss::marker] = jvParams[6u];
363 if (!jvParams.
size())
368 jvRequest[
"can_delete"] = jvParams[0u].
asUInt();
370 jvRequest[
"can_delete"] = input;
381 if (jvParams.
size() == 2)
383 jvRequest[jss::ip] = ip;
384 jvRequest[jss::port] = jvParams[1u].
asUInt();
398 jvRequest[jss::ip] = ip;
408 jvRequest[jss::source_account] = jvParams[0u].
asString();
409 jvRequest[jss::destination_account] = jvParams[1u].
asString();
411 if (jvParams.
size() >= 3)
415 if ((jvParams.
size() >= 4) && (jvParams.
size() <= 11))
418 for (uint32_t i = 3; i < jvParams.
size(); ++i)
419 jvRequest[jss::credentials].append(jvParams[i].asString());
438 if (jvParams.
size() > 0)
439 jvRequest[jss::feature] = jvParams[0u].
asString();
441 if (jvParams.
size() > 1)
443 auto const action = jvParams[1u].
asString();
448 if (boost::iequals(action,
"reject"))
450 else if (boost::iequals(action,
"accept"))
466 jvRequest[jss::min_count] = jvParams[0u].
asUInt();
476 bool const bOffline = 4 == jvParams.
size() && jvParams[3u].
asString() ==
"offline";
478 if (3 == jvParams.
size() || bOffline)
487 jvRequest[jss::account] = jvParams[0u].
asString();
488 jvRequest[jss::secret] = jvParams[1u].
asString();
489 jvRequest[jss::tx_json] = txJSON;
492 jvRequest[jss::offline] =
true;
507 JLOG(
j_.
trace()) <<
"RPC method: " << jvParams[0u];
508 JLOG(
j_.
trace()) <<
"RPC json: " << jvParams[1u];
515 jvRequest[jss::method] = jvParams[0u];
530 for (
auto const& j : jv)
539 if (jv.
isMember(jss::jsonrpc) && jv[jss::jsonrpc] ==
"2.0" && jv.
isMember(jss::ripplerpc) &&
540 jv[jss::ripplerpc] ==
"2.0" && jv.
isMember(jss::id) && jv.
isMember(jss::method))
556 bool valid_parse = reader.
parse(jvParams[0u].asString(), jv);
564 auto const& params = jv[jss::params];
565 for (
auto i = params.begin(); i != params.end(); ++i)
566 jv1[i.key().asString()] = *i;
568 jv1[jss::jsonrpc] = jv[jss::jsonrpc];
569 jv1[jss::ripplerpc] = jv[jss::ripplerpc];
570 jv1[jss::id] = jv[jss::id];
571 jv1[jss::method] = jv[jss::method];
578 if (jv[j].isMember(jss::params))
580 auto const& params = jv[j][jss::params];
581 for (
auto i = params.begin(); i != params.end(); ++i)
582 jv1[j][i.key().asString()] = *i;
584 jv1[j][jss::jsonrpc] = jv[j][jss::jsonrpc];
585 jv1[j][jss::ripplerpc] = jv[j][jss::ripplerpc];
586 jv1[j][jss::id] = jv[j][jss::id];
587 jv1[j][jss::method] = jv[j][jss::method];
593 jv_error[jss::jsonrpc] = jv[jss::jsonrpc];
595 jv_error[jss::ripplerpc] = jv[jss::ripplerpc];
597 jv_error[jss::id] = jv[jss::id];
607 if (!jvParams.
size())
614 if (2 == jvParams.
size())
616 if (jvParams[1u].asString() ==
"full")
618 jvRequest[jss::full] =
true;
620 else if (jvParams[1u].asString() ==
"tx")
622 jvRequest[jss::transactions] =
true;
623 jvRequest[jss::expand] =
true;
638 if (strLedger.
length() == 64)
640 jvRequest[jss::ledger_hash] = strLedger;
644 jvRequest[jss::ledger_index] = beast::lexicalCast<std::uint32_t>(strLedger);
656 jvRequest[jss::index] = jvParams[0u].
asString();
673 if (jvParams.
size() == 1)
675 jvRequest[jss::severity] = jvParams[0u].
asString();
677 else if (jvParams.
size() == 2)
679 jvRequest[jss::partition] = jvParams[0u].
asString();
680 jvRequest[jss::severity] = jvParams[1u].
asString();
721 unsigned int index = 0;
723 if (jvParams.
size() == 4)
725 jvRequest[jss::passphrase] = jvParams[index];
730 jvRequest[jss::key_type] = jvParams[index];
735 jvRequest[jss::secret] = jvParams[index];
744 jvRequest[jss::channel_id] =
to_string(channelId);
748 if (!jvParams[index].isString() || !
to_uint64(jvParams[index].asString()))
750 jvRequest[jss::amount] = jvParams[index];
769 jvRequest[jss::public_key] = strPk;
776 jvRequest[jss::channel_id] = jvParams[1u].
asString();
778 if (!jvParams[2u].isString() || !
to_uint64(jvParams[2u].asString()))
780 jvRequest[jss::amount] = jvParams[2u];
782 jvRequest[jss::signature] = jvParams[3u].
asString();
791 auto const nParams = jvParams.
size();
793 for (
auto i = 0; i < nParams; ++i)
797 if (i == 1 && strParam.
empty())
803 if (parseBase58<AccountID>(strParam))
805 jvRequest[accFields[i]] = std::move(strParam);
828 unsigned int iCursor = jvParams.
size();
830 if (!parseBase58<AccountID>(strIdent))
836 jvRequest[jss::account] = strIdent;
838 if (iCursor == 2 && !
jvParseLedger(jvRequest, jvParams[1u].asString()))
849 if (!
id.parseHex(strVaultID))
853 jvRequest[jss::vault_id] = strVaultID;
855 if (jvParams.
size() > 1)
866 jvRequest[jss::public_key] = jvParams[0u].
asString();
867 if (jvParams.
size() > 1)
869 jvRequest[jss::description] = jvParams[1u].
asString();
879 jvRequest[jss::public_key] = jvParams[0u].
asString();
889 bool bLedger = 2 == jvParams.
size();
891 JLOG(
j_.
trace()) <<
"RPC json: " << jvParams[0u];
919 jvRequest[jss::tx_json] = txJSON;
923 jvRequest[jss::tx_blob] = jvParams[0u].
asString();
926 if (jvParams.
size() == 2)
928 if (!jvParams[1u].isString() || jvParams[1u].asString() !=
"binary")
930 jvRequest[jss::binary] =
true;
946 bool const bOffline = jvParams.
size() >= 3 && jvParams[2u].
asString() ==
"offline";
948 if (jvParams.
size() < 3)
950 if (jvParams.
size() < 4 && bOffline)
956 if (1 == jvParams.
size())
962 jvRequest[jss::tx_blob] = jvParams[0u].
asString();
966 else if ((jvParams.
size() >= 2 || bOffline) && reader.
parse(jvParams[1u].
asString(), txJSON))
971 jvRequest[jss::secret] = jvParams[0u].
asString();
972 jvRequest[jss::tx_json] = txJSON;
975 jvRequest[jss::offline] =
true;
978 jvRequest[jss::signature_target] = *field;
992 if (1 == jvParams.
size())
999 jvRequest[jss::tx_json] = txJSON;
1012 XRPL_ASSERT(jvParams.
size() == 2,
"xrpl::RPCParser::parseTransactionEntry : valid parameter count");
1015 if (txHash.
length() != 64)
1019 jvRequest[jss::tx_hash] = txHash;
1025 if (jvRequest.isMember(jss::ledger_index) && jvRequest[jss::ledger_index] == 0)
1037 if (jvParams.
size() == 2 || jvParams.
size() == 4)
1039 if (jvParams[1u].asString() == jss::binary)
1040 jvRequest[jss::binary] =
true;
1043 if (jvParams.
size() >= 3)
1045 auto const offset = jvParams.
size() == 3 ? 0 : 1;
1047 jvRequest[jss::min_ledger] = jvParams[1u + offset].
asString();
1048 jvRequest[jss::max_ledger] = jvParams[2u + offset].
asString();
1051 if (jvParams[0u].asString().length() == 16)
1052 jvRequest[jss::ctid] = jvParams[0u].
asString();
1054 jvRequest[jss::transaction] = jvParams[0u].
asString();
1065 jvRequest[jss::start] = jvParams[0u].
asUInt();
1081 if (jvParams.
size())
1082 jvRequest[jss::secret] = jvParams[0u].
asString();
1095 if (jvParams.
size())
1096 jvRequest[jss::passphrase] = jvParams[0u].
asString();
1108 unsigned int index = 0;
1109 unsigned int const size = jvParams.
size();
1117 if (param[0] !=
'r')
1119 if (param.
size() == 64)
1120 jvRequest[jss::ledger_hash] = param;
1122 jvRequest[jss::ledger_index] = param;
1127 param = jvParams[index++].
asString();
1130 jvRequest[jss::account] = param;
1135 while (index < size)
1136 hotWallets.
append(jvParams[index++].asString());
1148 if (jvParams.
size() == 1)
1150 jvRequest[jss::hash] = jvParams[0u].
asString();
1161 if (jvParams.
size() == 1 && jvParams[0u].
asString() ==
"counters")
1162 jvRequest[jss::counters] =
true;
1182 stream <<
"Method: '" << strMethod <<
"'";
1183 stream <<
"Params: " << jvParams;
1194 static constexpr Command commands[] = {
1268 auto const count = jvParams.
size();
1270 for (
auto const& command : commands)
1272 if (strMethod == command.name)
1274 if ((command.minParams >= 0 && count < command.minParams) ||
1275 (command.maxParams >= 0 && count > command.maxParams))
1277 JLOG(
j_.
debug()) <<
"Wrong number of parameters for " << command.name
1278 <<
" minimum=" << command.minParams <<
" maximum=" << command.maxParams
1279 <<
" actual=" << count;
1284 return (this->*(command.parse))(jvParams);
1289 if (!allowAnyCommand)
1311 request[jss::method] = strMethod;
1312 request[jss::params] = params;
1313 request[jss::id] = id;
1334 (*jvOutput) = jvInput;
1340 boost::system::error_code
const& ecResult,
1351 if (strData.
empty())
1352 Throw<std::runtime_error>(
1353 "no response from server. Please "
1354 "ensure that the rippled server is running in another "
1359 if (strData.
find(
"Unable to parse request") == 0 || strData.
find(jss::invalid_API_version.c_str()) == 0)
1360 Throw<RequestNotParsable>(strData);
1363 if (!reader.
parse(strData, jvReply))
1364 Throw<std::runtime_error>(
"couldn't parse reply from server");
1367 Throw<std::runtime_error>(
"expected reply to have result, error and id properties");
1371 jvResult[
"result"] = jvReply;
1373 (callbackFuncP)(jvResult);
1386 boost::asio::streambuf& sb,
1390 JLOG(j.
debug()) <<
"requestRPC: strPath='" << strPath <<
"'";
1408 for (
int i = 1; i != args.
size(); i++)
1409 jvRpcParams.
append(args[i]);
1413 retParams[jss::method] = args[0];
1414 retParams[jss::params] = jvRpcParams;
1416 jvRequest = rpParser.
parseCommand(args[0], jvRpcParams,
true);
1418 auto insert_api_version = [apiVersion](
Json::Value& jr) {
1419 if (jr.isObject() && !jr.isMember(jss::error) && !jr.isMember(jss::api_version))
1421 jr[jss::api_version] = apiVersion;
1426 insert_api_version(jvRequest);
1441 unsigned int apiVersion,
1457 if (jvRequest.
isMember(jss::error))
1459 jvOutput = jvRequest;
1460 jvOutput[
"rpc"] = jvRpc;
1490 jvParams.
append(jvRequest);
1494 jvParams.
append(jvRequest[i]);
1498 boost::asio::io_context isService;
1507 ? jvRequest[jss::method].
asString()
1508 : jvRequest.
isArray() ?
"batch" : args[0],
1521 jvOutput = jvOutput[
"result"];
1532 jvOutput[
"result"] = jvRpcError;
1538 jvOutput[
"rpc"] = jvRpc;
1539 jvOutput[
"request_sent"] = jvRequest;
1545 jvOutput[jss::status] =
"error";
1546 if (jvOutput.
isMember(jss::error_code))
1547 nRet =
std::stoi(jvOutput[jss::error_code].asString());
1548 else if (jvOutput[jss::error].isMember(jss::error_code))
1549 nRet =
std::stoi(jvOutput[jss::error][jss::error_code].asString());
1557 catch (RequestNotParsable& e)
1560 jvOutput[
"error_what"] = e.what();
1566 jvOutput[
"error_what"] = e.what();
1570 return {nRet, std::move(jvOutput)};
1582 std::cout << result.second.toStyledString();
1584 return result.first;
1591 boost::asio::io_context& io_context,
1605 auto j = logs.
journal(
"HTTPClient");
1610 JLOG(j.info()) << (bSSL ?
"Securely connecting to " :
"Connecting to ") << strIp <<
":" << iPort <<
std::endl;
1620 constexpr auto RPC_REPLY_MAX_BYTES =
megabytes(256);
1622 using namespace std::chrono_literals;
1623 auto constexpr RPC_WEBHOOK_TIMEOUT = 30s;
1636 std::placeholders::_1,
1637 std::placeholders::_2,
1639 RPC_REPLY_MAX_BYTES,
1640 RPC_WEBHOOK_TIMEOUT,
1644 std::placeholders::_1,
1645 std::placeholders::_2,
1646 std::placeholders::_3,
Unserialize a JSON document into a Value.
bool parse(std::string const &document, Value &root)
Read a Value from a JSON document.
const_iterator begin() const
Value & append(Value const &value)
Append value to array at the end.
UInt size() const
Number of values in array or object.
const_iterator end() const
bool isObjectOrNull() const
std::string asString() const
Returns the unquoted string value.
bool isNull() const
isNull() tests to see if this field is null.
bool isMember(char const *key) const
Return true if the object has a member named key.
A generic endpoint for log messages.
Stream trace() const
Severity stream access functions.
std::optional< beast::IP::Endpoint > rpc_ip
static void request(bool bSSL, boost::asio::io_context &io_context, std::string strSite, unsigned short const port, std::function< void(boost::asio::streambuf &sb, std::string const &strHost)> build, std::size_t responseMax, std::chrono::seconds timeout, std::function< bool(boost::system::error_code const &ecResult, int iStatus, std::string const &strData)> complete, beast::Journal &j)
Manages partitions for logging.
beast::Journal journal(std::string const &name)
Json::Value parseFeature(Json::Value const &jvParams)
RPCParser(unsigned apiVersion, beast::Journal j)
Json::Value parseServerDefinitions(Json::Value const &jvParams)
Json::Value parseChannelVerify(Json::Value const &jvParams)
static bool jvParseLedger(Json::Value &jvRequest, std::string const &strLedger)
static bool validPublicKey(std::string const &strPk, TokenType type=TokenType::AccountPublic)
Json::Value parseConnect(Json::Value const &jvParams)
Json::Value parseJson2(Json::Value const &jvParams)
Json::Value parseAsIs(Json::Value const &jvParams)
bool isValidJson2(Json::Value const &jv)
Json::Value parsePeerReservationsDel(Json::Value const &jvParams)
Json::Value parseSignSubmit(Json::Value const &jvParams)
Json::Value parseValidationCreate(Json::Value const &jvParams)
Json::Value parseJson(Json::Value const &jvParams)
Json::Value parseSignFor(Json::Value const &jvParams)
Json::Value parseVault(Json::Value const &jvParams)
static Json::Value jvParseCurrencyIssuer(std::string const &strCurrencyIssuer)
Json::Value parseLedgerId(Json::Value const &jvParams)
Json::Value parseInternal(Json::Value const &jvParams)
Json::Value parseChannelAuthorize(Json::Value const &jvParams)
Json::Value parsePeerReservationsAdd(Json::Value const &jvParams)
Json::Value parseSubmitMultiSigned(Json::Value const &jvParams)
Json::Value parseLedger(Json::Value const &jvParams)
Json::Value parseAccountRaw2(Json::Value const &jvParams, char const *const acc2Field)
Json::Value parseAccountRaw1(Json::Value const &jvParams)
Json::Value parseGatewayBalances(Json::Value const &jvParams)
Json::Value parseCommand(std::string strMethod, Json::Value jvParams, bool allowAnyCommand)
Json::Value parseTx(Json::Value const &jvParams)
Json::Value parseGetCounts(Json::Value const &jvParams)
Json::Value parseLedgerEntry(Json::Value const &jvParams)
Json::Value parseServerInfo(Json::Value const &jvParams)
Json::Value parseDepositAuthorized(Json::Value const &jvParams)
Json::Value parseAccountChannels(Json::Value const &jvParams)
Json::Value parseAccountCurrencies(Json::Value const &jvParams)
Json::Value parseLogLevel(Json::Value const &jvParams)
Json::Value parseEvented(Json::Value const &jvParams)
Json::Value parseAccountItems(Json::Value const &jvParams)
Json::Value parseCanDelete(Json::Value const &jvParams)
Json::Value parseAccountLines(Json::Value const &jvParams)
Json::Value parseFetchInfo(Json::Value const &jvParams)
Json::Value parseBookOffers(Json::Value const &jvParams)
unsigned const apiVersion_
Json::Value parseManifest(Json::Value const &jvParams)
Json::Value parseTxHistory(Json::Value const &jvParams)
Json::Value parseSimulate(Json::Value const &jvParams)
Json::Value(RPCParser::*)(Json::Value const &jvParams) parseFuncPtr
Json::Value parseRipplePathFind(Json::Value const &jvParams)
Json::Value parseAccountTransactions(Json::Value const &jvParams)
Json::Value parseWalletPropose(Json::Value const &jvParams)
Json::Value parseTransactionEntry(Json::Value const &jvParams)
constexpr bool parseHex(std::string_view sv)
Parse a hex string into a base_uint.
T find_first_not_of(T... args)
T find_last_of(T... args)
JSON (JavaScript Object Notation).
@ arrayValue
array value (ordered list)
@ objectValue
object value (collection of name/value pairs).
int fromCommandLine(Config const &config, std::vector< std::string > const &vCmd, Logs &logs)
void fromNetwork(boost::asio::io_context &io_context, std::string const &strIp, std::uint16_t const iPort, std::string const &strUsername, std::string const &strPassword, std::string const &strPath, std::string const &strMethod, Json::Value const &jvParams, bool const bSSL, bool const quiet, Logs &logs, std::function< void(Json::Value const &jvInput)> callbackFuncP, std::unordered_map< std::string, std::string > headers)
Json::Value invalid_field_error(std::string const &name)
Json::Value make_param_error(std::string const &message)
Returns a new json object that indicates invalid parameters.
static constexpr auto apiCommandLineVersion
Use hash_* containers for keys that do not need a cryptographically secure hashing algorithm.
std::optional< KeyType > keyTypeFromString(std::string const &s)
std::string JSONRPCRequest(std::string const &strMethod, Json::Value const ¶ms, Json::Value const &id)
std::string to_string(base_uint< Bits, Tag > const &a)
std::optional< std::uint64_t > to_uint64(std::string const &s)
std::string toBase58(AccountID const &v)
Convert AccountID to base58 checked string.
std::string base64_encode(std::uint8_t const *data, std::size_t len)
std::pair< int, Json::Value > rpcClient(std::vector< std::string > const &args, Config const &config, Logs &logs, unsigned int apiVersion, std::unordered_map< std::string, std::string > const &headers)
Internal invocation of RPC client.
std::optional< KeyType > publicKeyType(Slice const &slice)
Returns the type of public key.
ServerHandler::Setup setup_ServerHandler(Config const &config, std::ostream &&log)
constexpr auto megabytes(T value) noexcept
std::optional< Blob > strUnHex(std::size_t strSize, Iterator begin, Iterator end)
bool isRpcError(Json::Value jvResult)
static std::string const & systemName()
Json::Value rpcError(error_code_i iError)
std::enable_if_t< std::is_same< T, char >::value||std::is_same< T, unsigned char >::value, Slice > makeSlice(std::array< T, N > const &a)
std::string createHTTPPost(std::string const &strHost, std::string const &strPath, std::string const &strMsg, std::unordered_map< std::string, std::string > const &mapRequestHeaders)
Json::Value rpcCmdToJson(std::vector< std::string > const &args, Json::Value &retParams, unsigned int apiVersion, beast::Journal j)
@ rpcCHANNEL_AMT_MALFORMED
static void callRPCHandler(Json::Value *jvOutput, Json::Value const &jvInput)
static void onRequest(std::string const &strMethod, Json::Value const &jvParams, std::unordered_map< std::string, std::string > const &headers, std::string const &strPath, boost::asio::streambuf &sb, std::string const &strHost, beast::Journal j)
static bool onResponse(std::function< void(Json::Value const &jvInput)> callbackFuncP, boost::system::error_code const &ecResult, int iStatus, std::string const &strData, beast::Journal j)
std::string admin_password