2013-11-20 14:18:57 +01:00
|
|
|
// Copyright (c) 2010 Satoshi Nakamoto
|
2015-12-13 14:51:43 +01:00
|
|
|
// Copyright (c) 2009-2015 The Bitcoin Core developers
|
2016-12-20 14:26:45 +01:00
|
|
|
// Copyright (c) 2014-2017 The Dash Core developers
|
2014-11-20 03:19:29 +01:00
|
|
|
// Distributed under the MIT software license, see the accompanying
|
2013-11-20 14:18:57 +01:00
|
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
|
2017-07-03 15:13:34 +02:00
|
|
|
#include "rpc/client.h"
|
|
|
|
#include "rpc/protocol.h"
|
2013-11-20 14:18:57 +01:00
|
|
|
#include "util.h"
|
|
|
|
|
2014-09-14 12:43:56 +02:00
|
|
|
#include <set>
|
2013-11-20 14:18:57 +01:00
|
|
|
#include <stdint.h>
|
|
|
|
|
2015-05-10 14:49:18 +02:00
|
|
|
#include <boost/algorithm/string/case_conv.hpp> // for to_lower()
|
2015-09-04 16:11:34 +02:00
|
|
|
#include <univalue.h>
|
2015-05-10 14:49:18 +02:00
|
|
|
|
2014-06-27 04:12:36 +02:00
|
|
|
class CRPCConvertParam
|
2013-11-20 14:18:57 +01:00
|
|
|
{
|
2014-06-27 04:12:36 +02:00
|
|
|
public:
|
2016-04-05 17:49:42 +02:00
|
|
|
std::string methodName; //!< method whose params want conversion
|
|
|
|
int paramIdx; //!< 0-based idx of param to convert
|
2017-01-10 13:52:49 +01:00
|
|
|
std::string paramName; //!< parameter name
|
2014-06-27 04:12:36 +02:00
|
|
|
};
|
2017-01-10 13:52:49 +01:00
|
|
|
/**
|
|
|
|
* Specifiy a (method, idx, name) here if the argument is a non-string RPC
|
|
|
|
* argument and needs to be converted from JSON.
|
|
|
|
*
|
|
|
|
* @note Parameter indexes start from 0.
|
|
|
|
*/
|
2014-06-27 04:12:36 +02:00
|
|
|
static const CRPCConvertParam vRPCConvertParams[] =
|
|
|
|
{
|
2017-01-10 13:52:49 +01:00
|
|
|
{ "setmocktime", 0, "timestamp" },
|
|
|
|
{ "generate", 0, "nblocks" },
|
|
|
|
{ "generate", 1, "maxtries" },
|
|
|
|
{ "generatetoaddress", 0, "nblocks" },
|
|
|
|
{ "generatetoaddress", 2, "maxtries" },
|
|
|
|
{ "getnetworkhashps", 0, "nblocks" },
|
|
|
|
{ "getnetworkhashps", 1, "height" },
|
|
|
|
{ "sendtoaddress", 1, "amount" },
|
|
|
|
{ "sendtoaddress", 4, "subtractfeefromamount" },
|
|
|
|
{ "sendtoaddress", 5, "use_is" },
|
|
|
|
{ "sendtoaddress", 6, "use_ps" },
|
|
|
|
{ "instantsendtoaddress", 1, "address" },
|
|
|
|
{ "instantsendtoaddress", 4, "comment_to" },
|
|
|
|
{ "settxfee", 0, "amount" },
|
|
|
|
{ "getreceivedbyaddress", 1, "minconf" },
|
|
|
|
{ "getreceivedbyaddress", 2, "addlockconf" },
|
|
|
|
{ "getreceivedbyaccount", 1, "minconf" },
|
|
|
|
{ "getreceivedbyaccount", 2, "addlockconf" },
|
2018-03-08 13:18:06 +01:00
|
|
|
{ "listaddressbalances", 0, "minamount" },
|
2017-01-10 13:52:49 +01:00
|
|
|
{ "listreceivedbyaddress", 0, "minconf" },
|
|
|
|
{ "listreceivedbyaddress", 1, "addlockconf" },
|
|
|
|
{ "listreceivedbyaddress", 2, "include_empty" },
|
|
|
|
{ "listreceivedbyaddress", 3, "include_watchonly" },
|
|
|
|
{ "listreceivedbyaccount", 0, "minconf" },
|
|
|
|
{ "listreceivedbyaccount", 1, "addlockconf" },
|
|
|
|
{ "listreceivedbyaccount", 2, "include_empty" },
|
|
|
|
{ "listreceivedbyaccount", 3, "include_watchonly" },
|
|
|
|
{ "getbalance", 1, "minconf" },
|
|
|
|
{ "getbalance", 2, "addlockconf" },
|
|
|
|
{ "getbalance", 3, "include_watchonly" },
|
|
|
|
{ "getchaintips", 0, "count" },
|
|
|
|
{ "getchaintips", 1, "branchlen" },
|
|
|
|
{ "getblockhash", 0, "height" },
|
|
|
|
{ "getsuperblockbudget", 0, "index" },
|
|
|
|
{ "waitforblockheight", 0, "height" },
|
|
|
|
{ "waitforblockheight", 1, "timeout" },
|
|
|
|
{ "waitforblock", 1, "timeout" },
|
|
|
|
{ "waitfornewblock", 0, "timeout" },
|
|
|
|
{ "move", 2, "amount" },
|
|
|
|
{ "move", 3, "minconf" },
|
|
|
|
{ "sendfrom", 2, "amount" },
|
|
|
|
{ "sendfrom", 3, "minconf" },
|
|
|
|
{ "sendfrom", 4, "addlockconf" },
|
|
|
|
{ "listtransactions", 1, "count" },
|
|
|
|
{ "listtransactions", 2, "skip" },
|
|
|
|
{ "listtransactions", 3, "include_watchonly" },
|
|
|
|
{ "listaccounts", 0, "minconf" },
|
|
|
|
{ "listaccounts", 1, "addlockconf" },
|
|
|
|
{ "listaccounts", 2, "include_watchonly" },
|
|
|
|
{ "walletpassphrase", 1, "timeout" },
|
|
|
|
{ "walletpassphrase", 2, "mixingonly" },
|
|
|
|
{ "getblocktemplate", 0, "template_request" },
|
|
|
|
{ "listsinceblock", 1, "target_confirmations" },
|
|
|
|
{ "listsinceblock", 2, "include_watchonly" },
|
|
|
|
{ "sendmany", 1, "amounts" },
|
|
|
|
{ "sendmany", 2, "minconf" },
|
|
|
|
{ "sendmany", 3, "addlockconf" },
|
|
|
|
{ "sendmany", 5, "subtractfeefromamount" },
|
|
|
|
{ "sendmany", 6, "use_is" },
|
|
|
|
{ "sendmany", 7, "use_ps" },
|
|
|
|
{ "addmultisigaddress", 0, "nrequired" },
|
|
|
|
{ "addmultisigaddress", 1, "keys" },
|
|
|
|
{ "createmultisig", 0, "nrequired" },
|
|
|
|
{ "createmultisig", 1, "keys" },
|
|
|
|
{ "listunspent", 0, "minconf" },
|
|
|
|
{ "listunspent", 1, "maxconf" },
|
|
|
|
{ "listunspent", 2, "addresses" },
|
2018-03-01 14:18:46 +01:00
|
|
|
{ "listunspent", 3, "include_unsafe" },
|
2017-01-10 13:52:49 +01:00
|
|
|
{ "getblock", 1, "verbose" },
|
|
|
|
{ "getblockheader", 1, "verbose" },
|
|
|
|
{ "getblockheaders", 1, "count" },
|
|
|
|
{ "getblockheaders", 2, "verbose" },
|
|
|
|
{ "gettransaction", 1, "include_watchonly" },
|
|
|
|
{ "getrawtransaction", 1, "verbose" },
|
2017-03-29 10:44:35 +02:00
|
|
|
{ "createrawtransaction", 0, "inputs" },
|
2017-01-10 13:52:49 +01:00
|
|
|
{ "createrawtransaction", 1, "outputs" },
|
|
|
|
{ "createrawtransaction", 2, "locktime" },
|
|
|
|
{ "signrawtransaction", 1, "prevtxs" },
|
|
|
|
{ "signrawtransaction", 2, "privkeys" },
|
|
|
|
{ "sendrawtransaction", 1, "allowhighfees" },
|
2018-06-12 13:33:23 +02:00
|
|
|
{ "sendrawtransaction", 2, "instantsend" },
|
|
|
|
{ "sendrawtransaction", 3, "bypasslimits" },
|
2017-01-10 13:52:49 +01:00
|
|
|
{ "fundrawtransaction", 1, "options" },
|
|
|
|
{ "gettxout", 1, "n" },
|
|
|
|
{ "gettxout", 2, "include_mempool" },
|
|
|
|
{ "gettxoutproof", 0, "txids" },
|
|
|
|
{ "lockunspent", 0, "unlock" },
|
|
|
|
{ "lockunspent", 1, "transactions" },
|
|
|
|
{ "importprivkey", 2, "rescan" },
|
|
|
|
{ "importelectrumwallet", 1, "index" },
|
|
|
|
{ "importaddress", 2, "rescan" },
|
|
|
|
{ "importaddress", 3, "p2sh" },
|
|
|
|
{ "importpubkey", 2, "rescan" },
|
|
|
|
{ "importmulti", 0, "requests" },
|
|
|
|
{ "importmulti", 1, "options" },
|
|
|
|
{ "verifychain", 0, "checklevel" },
|
|
|
|
{ "verifychain", 1, "nblocks" },
|
2017-01-11 14:16:11 +01:00
|
|
|
{ "pruneblockchain", 0, "height" },
|
2017-01-10 13:52:49 +01:00
|
|
|
{ "keypoolrefill", 0, "newsize" },
|
|
|
|
{ "getrawmempool", 0, "verbose" },
|
|
|
|
{ "estimatefee", 0, "nblocks" },
|
|
|
|
{ "estimatepriority", 0, "nblocks" },
|
|
|
|
{ "estimatesmartfee", 0, "nblocks" },
|
|
|
|
{ "estimatesmartpriority", 0, "nblocks" },
|
|
|
|
{ "prioritisetransaction", 1, "priority_delta" },
|
|
|
|
{ "prioritisetransaction", 2, "fee_delta" },
|
|
|
|
{ "setban", 2, "bantime" },
|
|
|
|
{ "setban", 3, "absolute" },
|
2018-01-22 08:05:44 +01:00
|
|
|
{ "setbip69enabled", 0, "enabled" },
|
2017-01-10 13:52:49 +01:00
|
|
|
{ "setnetworkactive", 0, "state" },
|
2018-08-21 16:07:54 +02:00
|
|
|
{ "setprivatesendrounds", 0, "rounds" },
|
|
|
|
{ "setprivatesendamount", 0, "amount" },
|
2017-01-10 13:52:49 +01:00
|
|
|
{ "getmempoolancestors", 1, "verbose" },
|
|
|
|
{ "getmempooldescendants", 1, "verbose" },
|
2018-03-01 14:18:46 +01:00
|
|
|
{ "spork", 1, "value" },
|
2017-01-10 13:52:49 +01:00
|
|
|
{ "voteraw", 1, "tx_index" },
|
|
|
|
{ "voteraw", 5, "time" },
|
|
|
|
{ "getblockhashes", 0, "high"},
|
|
|
|
{ "getblockhashes", 1, "low" },
|
|
|
|
{ "getspentinfo", 0, "json" },
|
|
|
|
{ "getaddresstxids", 0, "addresses" },
|
|
|
|
{ "getaddressbalance", 0, "addresses" },
|
|
|
|
{ "getaddressdeltas", 0, "addresses" },
|
|
|
|
{ "getaddressutxos", 0, "addresses" },
|
|
|
|
{ "getaddressmempool", 0, "addresses" },
|
|
|
|
// Echo with conversion (For testing only)
|
|
|
|
{ "echojson", 0, "arg0" },
|
|
|
|
{ "echojson", 1, "arg1" },
|
|
|
|
{ "echojson", 2, "arg2" },
|
|
|
|
{ "echojson", 3, "arg3" },
|
|
|
|
{ "echojson", 4, "arg4" },
|
|
|
|
{ "echojson", 5, "arg5" },
|
|
|
|
{ "echojson", 6, "arg6" },
|
|
|
|
{ "echojson", 7, "arg7" },
|
|
|
|
{ "echojson", 8, "arg8" },
|
|
|
|
{ "echojson", 9, "arg9" },
|
2014-06-27 04:12:36 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
class CRPCConvertTable
|
|
|
|
{
|
|
|
|
private:
|
2017-01-10 13:52:49 +01:00
|
|
|
std::set<std::pair<std::string, int>> members;
|
|
|
|
std::set<std::pair<std::string, std::string>> membersByName;
|
2013-11-20 14:18:57 +01:00
|
|
|
|
2014-06-27 04:12:36 +02:00
|
|
|
public:
|
|
|
|
CRPCConvertTable();
|
2013-11-20 14:18:57 +01:00
|
|
|
|
2014-06-27 04:12:36 +02:00
|
|
|
bool convert(const std::string& method, int idx) {
|
|
|
|
return (members.count(std::make_pair(method, idx)) > 0);
|
2013-11-20 14:18:57 +01:00
|
|
|
}
|
2017-01-10 13:52:49 +01:00
|
|
|
bool convert(const std::string& method, const std::string& name) {
|
|
|
|
return (membersByName.count(std::make_pair(method, name)) > 0);
|
|
|
|
}
|
2014-06-27 04:12:36 +02:00
|
|
|
};
|
2013-11-20 14:18:57 +01:00
|
|
|
|
2014-06-27 04:12:36 +02:00
|
|
|
CRPCConvertTable::CRPCConvertTable()
|
2013-11-20 14:18:57 +01:00
|
|
|
{
|
2014-06-27 04:12:36 +02:00
|
|
|
const unsigned int n_elem =
|
|
|
|
(sizeof(vRPCConvertParams) / sizeof(vRPCConvertParams[0]));
|
|
|
|
|
|
|
|
for (unsigned int i = 0; i < n_elem; i++) {
|
|
|
|
members.insert(std::make_pair(vRPCConvertParams[i].methodName,
|
|
|
|
vRPCConvertParams[i].paramIdx));
|
2017-01-10 13:52:49 +01:00
|
|
|
membersByName.insert(std::make_pair(vRPCConvertParams[i].methodName,
|
|
|
|
vRPCConvertParams[i].paramName));
|
2013-11-20 14:18:57 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-27 04:12:36 +02:00
|
|
|
static CRPCConvertTable rpcCvtTable;
|
|
|
|
|
2015-06-04 10:31:22 +02:00
|
|
|
/** Non-RFC4627 JSON parser, accepts internal values (such as numbers, true, false, null)
|
|
|
|
* as well as objects and arrays.
|
|
|
|
*/
|
|
|
|
UniValue ParseNonRFCJSONValue(const std::string& strVal)
|
|
|
|
{
|
|
|
|
UniValue jVal;
|
|
|
|
if (!jVal.read(std::string("[")+strVal+std::string("]")) ||
|
|
|
|
!jVal.isArray() || jVal.size()!=1)
|
2017-03-09 08:14:27 +01:00
|
|
|
throw std::runtime_error(std::string("Error parsing JSON:")+strVal);
|
2015-06-04 10:31:22 +02:00
|
|
|
return jVal[0];
|
|
|
|
}
|
|
|
|
|
2015-05-13 21:29:19 +02:00
|
|
|
UniValue RPCConvertValues(const std::string &strMethod, const std::vector<std::string> &strParams)
|
2013-11-20 14:18:57 +01:00
|
|
|
{
|
2015-05-10 13:35:44 +02:00
|
|
|
UniValue params(UniValue::VARR);
|
2013-11-20 14:18:57 +01:00
|
|
|
|
2014-06-27 04:12:36 +02:00
|
|
|
for (unsigned int idx = 0; idx < strParams.size(); idx++) {
|
|
|
|
const std::string& strVal = strParams[idx];
|
2013-11-20 14:18:57 +01:00
|
|
|
|
2014-06-27 04:12:36 +02:00
|
|
|
if (!rpcCvtTable.convert(strMethod, idx)) {
|
2015-06-04 10:31:22 +02:00
|
|
|
// insert string value directly
|
2014-06-27 04:12:36 +02:00
|
|
|
params.push_back(strVal);
|
2015-06-04 10:31:22 +02:00
|
|
|
} else {
|
|
|
|
// parse string as JSON, insert bool/number/object/etc. value
|
|
|
|
params.push_back(ParseNonRFCJSONValue(strVal));
|
2013-11-20 14:18:57 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return params;
|
2013-11-27 15:41:12 +01:00
|
|
|
}
|
2017-01-10 13:52:49 +01:00
|
|
|
|
|
|
|
UniValue RPCConvertNamedValues(const std::string &strMethod, const std::vector<std::string> &strParams)
|
|
|
|
{
|
|
|
|
UniValue params(UniValue::VOBJ);
|
|
|
|
|
|
|
|
for (const std::string &s: strParams) {
|
|
|
|
size_t pos = s.find("=");
|
|
|
|
if (pos == std::string::npos) {
|
|
|
|
throw(std::runtime_error("No '=' in named argument '"+s+"', this needs to be present for every argument (even if it is empty)"));
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string name = s.substr(0, pos);
|
|
|
|
std::string value = s.substr(pos+1);
|
|
|
|
|
|
|
|
if (!rpcCvtTable.convert(strMethod, name)) {
|
|
|
|
// insert string value directly
|
|
|
|
params.pushKV(name, value);
|
|
|
|
} else {
|
|
|
|
// parse string as JSON, insert bool/number/object/etc. value
|
|
|
|
params.pushKV(name, ParseNonRFCJSONValue(value));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return params;
|
|
|
|
}
|