2021-01-01 11:02:49 -03:00
// Copyright (c) 2017-2021 The Bitcoin Core developers
2017-09-29 01:21:28 -03:00
// Distributed under the MIT software license, see the accompanying
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
2017-09-19 22:12:25 -03:00
# include <key_io.h>
2019-05-15 00:21:11 -04:00
# include <outputtype.h>
2017-09-29 01:21:28 -03:00
# include <rpc/util.h>
2019-02-16 20:24:15 -03:00
# include <script/descriptor.h>
2019-08-20 16:58:50 -04:00
# include <script/signingprovider.h>
2017-09-29 01:21:28 -03:00
# include <tinyformat.h>
2018-10-22 19:51:11 -03:00
# include <util/strencodings.h>
2019-08-20 16:58:50 -04:00
# include <util/string.h>
2020-06-05 08:26:16 -04:00
# include <util/translation.h>
2017-09-29 01:21:28 -03:00
2019-04-04 04:39:04 -03:00
# include <tuple>
2020-04-12 17:09:23 -04:00
# include <boost/algorithm/string/classification.hpp>
# include <boost/algorithm/string/split.hpp>
2019-11-26 15:25:56 -03:00
const std : : string UNIX_EPOCH_TIME = " UNIX epoch time " ;
2020-02-25 13:12:04 -03:00
const std : : string EXAMPLE_ADDRESS [ 2 ] = { " bc1q09vm5lfy0j5reeulh4x5752q25uqqvz34hufdl " , " bc1q02ad21edsxd23d32dfgqqsz4vv4nmtfzuklhy3 " } ;
2019-11-26 15:25:56 -03:00
2019-04-02 17:42:51 -03:00
void RPCTypeCheck ( const UniValue & params ,
const std : : list < UniValueType > & typesExpected ,
bool fAllowNull )
{
unsigned int i = 0 ;
for ( const UniValueType & t : typesExpected ) {
if ( params . size ( ) < = i )
break ;
const UniValue & v = params [ i ] ;
if ( ! ( fAllowNull & & v . isNull ( ) ) ) {
RPCTypeCheckArgument ( v , t ) ;
}
i + + ;
}
}
void RPCTypeCheckArgument ( const UniValue & value , const UniValueType & typeExpected )
{
if ( ! typeExpected . typeAny & & value . type ( ) ! = typeExpected . type ) {
throw JSONRPCError ( RPC_TYPE_ERROR , strprintf ( " Expected type %s, got %s " , uvTypeName ( typeExpected . type ) , uvTypeName ( value . type ( ) ) ) ) ;
}
}
void RPCTypeCheckObj ( const UniValue & o ,
const std : : map < std : : string , UniValueType > & typesExpected ,
bool fAllowNull ,
bool fStrict )
{
for ( const auto & t : typesExpected ) {
const UniValue & v = find_value ( o , t . first ) ;
if ( ! fAllowNull & & v . isNull ( ) )
throw JSONRPCError ( RPC_TYPE_ERROR , strprintf ( " Missing %s " , t . first ) ) ;
if ( ! ( t . second . typeAny | | v . type ( ) = = t . second . type | | ( fAllowNull & & v . isNull ( ) ) ) ) {
std : : string err = strprintf ( " Expected type %s for %s, got %s " ,
uvTypeName ( t . second . type ) , t . first , uvTypeName ( v . type ( ) ) ) ;
throw JSONRPCError ( RPC_TYPE_ERROR , err ) ;
}
}
if ( fStrict )
{
for ( const std : : string & k : o . getKeys ( ) )
{
if ( typesExpected . count ( k ) = = 0 )
{
std : : string err = strprintf ( " Unexpected key %s " , k ) ;
throw JSONRPCError ( RPC_TYPE_ERROR , err ) ;
}
}
}
}
CAmount AmountFromValue ( const UniValue & value )
{
if ( ! value . isNum ( ) & & ! value . isStr ( ) )
throw JSONRPCError ( RPC_TYPE_ERROR , " Amount is not a number or string " ) ;
CAmount amount ;
if ( ! ParseFixedPoint ( value . getValStr ( ) , 8 , & amount ) )
throw JSONRPCError ( RPC_TYPE_ERROR , " Invalid amount " ) ;
if ( ! MoneyRange ( amount ) )
throw JSONRPCError ( RPC_TYPE_ERROR , " Amount out of range " ) ;
return amount ;
}
uint256 ParseHashV ( const UniValue & v , std : : string strName )
{
std : : string strHex ( v . get_str ( ) ) ;
if ( 64 ! = strHex . length ( ) )
throw JSONRPCError ( RPC_INVALID_PARAMETER , strprintf ( " %s must be of length %d (not %d, for '%s') " , strName, 64, strHex.length(), strHex)) ;
if ( ! IsHex ( strHex ) ) // Note: IsHex("") is false
throw JSONRPCError ( RPC_INVALID_PARAMETER , strName + " must be hexadecimal string (not ' " + strHex + " ') " ) ;
return uint256S ( strHex ) ;
}
uint256 ParseHashO ( const UniValue & o , std : : string strKey )
{
return ParseHashV ( find_value ( o , strKey ) , strKey ) ;
}
std : : vector < unsigned char > ParseHexV ( const UniValue & v , std : : string strName )
{
std : : string strHex ;
if ( v . isStr ( ) )
strHex = v . get_str ( ) ;
if ( ! IsHex ( strHex ) )
throw JSONRPCError ( RPC_INVALID_PARAMETER , strName + " must be hexadecimal string (not ' " + strHex + " ') " ) ;
return ParseHex ( strHex ) ;
}
std : : vector < unsigned char > ParseHexO ( const UniValue & o , std : : string strKey )
{
return ParseHexV ( find_value ( o , strKey ) , strKey ) ;
}
2021-02-26 11:03:50 -03:00
namespace {
/**
* Quote an argument for shell .
*
* @ note This is intended for help , not for security - sensitive purposes .
*/
std : : string ShellQuote ( const std : : string & s )
{
std : : string result ;
result . reserve ( s . size ( ) * 2 ) ;
for ( const char ch : s ) {
if ( ch = = ' \' ' ) {
result + = " ' \' ' " ;
} else {
result + = ch ;
}
}
return " ' " + result + " ' " ;
}
/**
* Shell - quotes the argument if it needs quoting , else returns it literally , to save typing .
*
* @ note This is intended for help , not for security - sensitive purposes .
*/
std : : string ShellQuoteIfNeeded ( const std : : string & s )
{
for ( const char ch : s ) {
if ( ch = = ' ' | | ch = = ' \' ' | | ch = = ' " ' ) {
return ShellQuote ( s ) ;
}
}
return s ;
}
}
2019-04-02 17:42:51 -03:00
std : : string HelpExampleCli ( const std : : string & methodname , const std : : string & args )
{
return " > bitcoin-cli " + methodname + " " + args + " \n " ;
}
2021-02-26 11:03:50 -03:00
std : : string HelpExampleCliNamed ( const std : : string & methodname , const RPCArgList & args )
{
std : : string result = " > bitcoin-cli -named " + methodname ;
for ( const auto & argpair : args ) {
const auto & value = argpair . second . isStr ( )
? argpair . second . get_str ( )
: argpair . second . write ( ) ;
result + = " " + argpair . first + " = " + ShellQuoteIfNeeded ( value ) ;
}
result + = " \n " ;
return result ;
}
2019-04-02 17:42:51 -03:00
std : : string HelpExampleRpc ( const std : : string & methodname , const std : : string & args )
{
2020-03-11 12:32:06 -03:00
return " > curl --user myusername --data-binary '{ \" jsonrpc \" : \" 1.0 \" , \" id \" : \" curltest \" , "
" \" method \" : \" " + methodname + " \" , \" params \" : [ " + args + " ]}' -H 'content-type: text/plain;' http://127.0.0.1:8332/ \n " ;
2019-04-02 17:42:51 -03:00
}
2021-03-03 05:32:14 -03:00
std : : string HelpExampleRpcNamed ( const std : : string & methodname , const RPCArgList & args )
{
UniValue params ( UniValue : : VOBJ ) ;
for ( const auto & param : args ) {
params . pushKV ( param . first , param . second ) ;
}
return " > curl --user myusername --data-binary '{ \" jsonrpc \" : \" 1.0 \" , \" id \" : \" curltest \" , "
" \" method \" : \" " + methodname + " \" , \" params \" : " + params . write ( ) + " }' -H 'content-type: text/plain;' http://127.0.0.1:8332/ \n " ;
}
2017-09-29 01:21:28 -03:00
// Converts a hex string to a public key if possible
CPubKey HexToPubKey ( const std : : string & hex_in )
{
if ( ! IsHex ( hex_in ) ) {
throw JSONRPCError ( RPC_INVALID_ADDRESS_OR_KEY , " Invalid public key: " + hex_in ) ;
}
CPubKey vchPubKey ( ParseHex ( hex_in ) ) ;
if ( ! vchPubKey . IsFullyValid ( ) ) {
throw JSONRPCError ( RPC_INVALID_ADDRESS_OR_KEY , " Invalid public key: " + hex_in ) ;
}
return vchPubKey ;
}
2019-06-06 10:33:23 -04:00
// Retrieves a public key for an address from the given FillableSigningProvider
2019-11-24 18:53:42 -03:00
CPubKey AddrToPubKey ( const FillableSigningProvider & keystore , const std : : string & addr_in )
2017-09-29 01:21:28 -03:00
{
CTxDestination dest = DecodeDestination ( addr_in ) ;
if ( ! IsValidDestination ( dest ) ) {
throw JSONRPCError ( RPC_INVALID_ADDRESS_OR_KEY , " Invalid address: " + addr_in ) ;
}
2019-11-24 18:53:42 -03:00
CKeyID key = GetKeyForDestination ( keystore , dest ) ;
2017-09-29 01:21:28 -03:00
if ( key . IsNull ( ) ) {
throw JSONRPCError ( RPC_INVALID_ADDRESS_OR_KEY , strprintf ( " %s does not refer to a key " , addr_in ) ) ;
}
CPubKey vchPubKey ;
2019-11-24 18:53:42 -03:00
if ( ! keystore . GetPubKey ( key , vchPubKey ) ) {
2017-09-29 01:21:28 -03:00
throw JSONRPCError ( RPC_INVALID_ADDRESS_OR_KEY , strprintf ( " no full public key for address %s " , addr_in ) ) ;
}
if ( ! vchPubKey . IsFullyValid ( ) ) {
throw JSONRPCError ( RPC_INTERNAL_ERROR , " Wallet contains an invalid public key " ) ;
}
return vchPubKey ;
}
2019-05-15 00:21:11 -04:00
// Creates a multisig address from a given list of public keys, number of signatures required, and the address type
2019-06-06 10:33:23 -04:00
CTxDestination AddAndGetMultisigDestination ( const int required , const std : : vector < CPubKey > & pubkeys , OutputType type , FillableSigningProvider & keystore , CScript & script_out )
2017-09-29 01:21:28 -03:00
{
// Gather public keys
if ( required < 1 ) {
throw JSONRPCError ( RPC_INVALID_PARAMETER , " a multisignature address must require at least one key to redeem " ) ;
}
if ( ( int ) pubkeys . size ( ) < required ) {
throw JSONRPCError ( RPC_INVALID_PARAMETER , strprintf ( " not enough keys supplied (got %u keys, but need at least %d to redeem) " , pubkeys.size(), required)) ;
}
if ( pubkeys . size ( ) > 16 ) {
throw JSONRPCError ( RPC_INVALID_PARAMETER , " Number of keys involved in the multisignature address creation > 16 \n Reduce the number " ) ;
}
2019-05-15 00:21:11 -04:00
script_out = GetScriptForMultisig ( required , pubkeys ) ;
2017-09-29 01:21:28 -03:00
2019-05-15 00:21:11 -04:00
if ( script_out . size ( ) > MAX_SCRIPT_ELEMENT_SIZE ) {
throw JSONRPCError ( RPC_INVALID_PARAMETER , ( strprintf ( " redeemScript exceeds size limit: %d > %d " , script_out . size ( ) , MAX_SCRIPT_ELEMENT_SIZE ) ) ) ;
2017-09-29 01:21:28 -03:00
}
2019-05-15 00:21:11 -04:00
// Check if any keys are uncompressed. If so, the type is legacy
for ( const CPubKey & pk : pubkeys ) {
if ( ! pk . IsCompressed ( ) ) {
type = OutputType : : LEGACY ;
break ;
}
}
// Make the address
CTxDestination dest = AddAndGetDestinationForScript ( keystore , script_out , type ) ;
return dest ;
2017-09-29 01:21:28 -03:00
}
2017-12-04 14:49:20 -03:00
2021-01-04 07:20:02 -03:00
class DescribeAddressVisitor
2017-12-04 14:49:20 -03:00
{
public :
explicit DescribeAddressVisitor ( ) { }
UniValue operator ( ) ( const CNoDestination & dest ) const
{
return UniValue ( UniValue : : VOBJ ) ;
}
2019-02-19 19:00:45 -03:00
UniValue operator ( ) ( const PKHash & keyID ) const
2017-12-04 14:49:20 -03:00
{
UniValue obj ( UniValue : : VOBJ ) ;
obj . pushKV ( " isscript " , false ) ;
obj . pushKV ( " iswitness " , false ) ;
return obj ;
}
2019-02-19 19:00:45 -03:00
UniValue operator ( ) ( const ScriptHash & scriptID ) const
2017-12-04 14:49:20 -03:00
{
UniValue obj ( UniValue : : VOBJ ) ;
obj . pushKV ( " isscript " , true ) ;
obj . pushKV ( " iswitness " , false ) ;
return obj ;
}
UniValue operator ( ) ( const WitnessV0KeyHash & id ) const
{
UniValue obj ( UniValue : : VOBJ ) ;
obj . pushKV ( " isscript " , false ) ;
obj . pushKV ( " iswitness " , true ) ;
obj . pushKV ( " witness_version " , 0 ) ;
2020-06-24 09:48:26 -04:00
obj . pushKV ( " witness_program " , HexStr ( id ) ) ;
2017-12-04 14:49:20 -03:00
return obj ;
}
UniValue operator ( ) ( const WitnessV0ScriptHash & id ) const
{
UniValue obj ( UniValue : : VOBJ ) ;
obj . pushKV ( " isscript " , true ) ;
obj . pushKV ( " iswitness " , true ) ;
obj . pushKV ( " witness_version " , 0 ) ;
2020-06-24 09:48:26 -04:00
obj . pushKV ( " witness_program " , HexStr ( id ) ) ;
2017-12-04 14:49:20 -03:00
return obj ;
}
UniValue operator ( ) ( const WitnessUnknown & id ) const
{
UniValue obj ( UniValue : : VOBJ ) ;
obj . pushKV ( " iswitness " , true ) ;
obj . pushKV ( " witness_version " , ( int ) id . version ) ;
2020-06-24 11:26:47 -04:00
obj . pushKV ( " witness_program " , HexStr ( Span < const unsigned char > ( id . program , id . length ) ) ) ;
2017-12-04 14:49:20 -03:00
return obj ;
}
} ;
UniValue DescribeAddress ( const CTxDestination & dest )
{
2021-01-04 07:20:02 -03:00
return std : : visit ( DescribeAddressVisitor ( ) , dest ) ;
2017-12-04 14:49:20 -03:00
}
2018-10-23 16:22:28 -03:00
2017-07-28 21:40:29 -04:00
unsigned int ParseConfirmTarget ( const UniValue & value , unsigned int max_target )
2019-02-08 17:29:45 -03:00
{
2020-10-19 06:35:10 -03:00
const int target { value . get_int ( ) } ;
const unsigned int unsigned_target { static_cast < unsigned int > ( target ) } ;
if ( target < 1 | | unsigned_target > max_target ) {
throw JSONRPCError ( RPC_INVALID_PARAMETER , strprintf ( " Invalid conf_target, must be between %u and %u " , 1 , max_target ) ) ;
2019-02-08 17:29:45 -03:00
}
2020-10-19 06:35:10 -03:00
return unsigned_target ;
2019-02-08 17:29:45 -03:00
}
2019-02-10 01:51:33 -03:00
RPCErrorCode RPCErrorFromTransactionError ( TransactionError terr )
{
switch ( terr ) {
case TransactionError : : MEMPOOL_REJECTED :
return RPC_TRANSACTION_REJECTED ;
case TransactionError : : ALREADY_IN_CHAIN :
return RPC_TRANSACTION_ALREADY_IN_CHAIN ;
case TransactionError : : P2P_DISABLED :
return RPC_CLIENT_P2P_DISABLED ;
case TransactionError : : INVALID_PSBT :
2019-01-09 08:08:32 -03:00
case TransactionError : : PSBT_MISMATCH :
return RPC_INVALID_PARAMETER ;
2019-02-10 01:51:33 -03:00
case TransactionError : : SIGHASH_MISMATCH :
return RPC_DESERIALIZATION_ERROR ;
default : break ;
}
return RPC_TRANSACTION_ERROR ;
}
UniValue JSONRPCTransactionError ( TransactionError terr , const std : : string & err_string )
{
if ( err_string . length ( ) > 0 ) {
return JSONRPCError ( RPCErrorFromTransactionError ( terr ) , err_string ) ;
} else {
2020-06-05 08:26:16 -04:00
return JSONRPCError ( RPCErrorFromTransactionError ( terr ) , TransactionErrorString ( terr ) . original ) ;
2019-02-10 01:51:33 -03:00
}
}
2019-04-04 11:43:22 -03:00
/**
* A pair of strings that can be aligned ( through padding ) with other Sections
* later on
*/
2018-11-23 13:21:38 -03:00
struct Section {
Section ( const std : : string & left , const std : : string & right )
: m_left { left } , m_right { right } { }
2020-03-26 17:28:38 -03:00
std : : string m_left ;
2018-11-23 13:21:38 -03:00
const std : : string m_right ;
} ;
2019-04-04 11:43:22 -03:00
/**
* Keeps track of RPCArgs by transforming them into sections for the purpose
* of serializing everything to a single string
*/
2018-11-23 13:21:38 -03:00
struct Sections {
std : : vector < Section > m_sections ;
size_t m_max_pad { 0 } ;
void PushSection ( const Section & s )
{
m_max_pad = std : : max ( m_max_pad , s . m_left . size ( ) ) ;
m_sections . push_back ( s ) ;
}
2019-04-04 11:43:22 -03:00
/**
* Recursive helper to translate an RPCArg into sections
*/
2019-12-25 17:26:59 -03:00
void Push ( const RPCArg & arg , const size_t current_indent = 5 , const OuterType outer_type = OuterType : : NONE )
2018-11-23 13:21:38 -03:00
{
const auto indent = std : : string ( current_indent , ' ' ) ;
const auto indent_next = std : : string ( current_indent + 2 , ' ' ) ;
2019-04-04 12:04:14 -03:00
const bool push_name { outer_type = = OuterType : : OBJ } ; // Dictionary keys must have a name
2018-11-23 13:21:38 -03:00
switch ( arg . m_type ) {
case RPCArg : : Type : : STR_HEX :
case RPCArg : : Type : : STR :
case RPCArg : : Type : : NUM :
case RPCArg : : Type : : AMOUNT :
2019-02-27 18:45:47 -03:00
case RPCArg : : Type : : RANGE :
2018-11-23 13:21:38 -03:00
case RPCArg : : Type : : BOOL : {
2019-12-25 17:26:59 -03:00
if ( outer_type = = OuterType : : NONE ) return ; // Nothing more to do for non-recursive types on first recursion
2018-11-23 13:21:38 -03:00
auto left = indent ;
2019-04-04 12:04:14 -03:00
if ( arg . m_type_str . size ( ) ! = 0 & & push_name ) {
2020-04-12 17:09:23 -04:00
left + = " \" " + arg . GetName ( ) + " \" : " + arg . m_type_str . at ( 0 ) ;
2018-11-23 13:21:38 -03:00
} else {
2019-04-04 12:04:14 -03:00
left + = push_name ? arg . ToStringObj ( /* oneline */ false ) : arg . ToString ( /* oneline */ false ) ;
2018-11-23 13:21:38 -03:00
}
left + = " , " ;
2018-12-10 18:56:51 -03:00
PushSection ( { left , arg . ToDescriptionString ( ) } ) ;
2018-11-23 13:21:38 -03:00
break ;
}
case RPCArg : : Type : : OBJ :
case RPCArg : : Type : : OBJ_USER_KEYS : {
2019-12-25 17:26:59 -03:00
const auto right = outer_type = = OuterType : : NONE ? " " : arg . ToDescriptionString ( ) ;
2020-04-12 17:09:23 -04:00
PushSection ( { indent + ( push_name ? " \" " + arg . GetName ( ) + " \" : " : " " ) + " { " , right } ) ;
2018-11-23 13:21:38 -03:00
for ( const auto & arg_inner : arg . m_inner ) {
Push ( arg_inner , current_indent + 2 , OuterType : : OBJ ) ;
}
if ( arg . m_type ! = RPCArg : : Type : : OBJ ) {
PushSection ( { indent_next + " ... " , " " } ) ;
}
2019-12-25 17:26:59 -03:00
PushSection ( { indent + " } " + ( outer_type ! = OuterType : : NONE ? " , " : " " ) , " " } ) ;
2018-11-23 13:21:38 -03:00
break ;
}
case RPCArg : : Type : : ARR : {
auto left = indent ;
2020-04-12 17:09:23 -04:00
left + = push_name ? " \" " + arg . GetName ( ) + " \" : " : " " ;
2018-11-23 13:21:38 -03:00
left + = " [ " ;
2019-12-25 17:26:59 -03:00
const auto right = outer_type = = OuterType : : NONE ? " " : arg . ToDescriptionString ( ) ;
2018-11-23 13:21:38 -03:00
PushSection ( { left , right } ) ;
for ( const auto & arg_inner : arg . m_inner ) {
Push ( arg_inner , current_indent + 2 , OuterType : : ARR ) ;
}
PushSection ( { indent_next + " ... " , " " } ) ;
2019-12-25 17:26:59 -03:00
PushSection ( { indent + " ] " + ( outer_type ! = OuterType : : NONE ? " , " : " " ) , " " } ) ;
2018-11-23 13:21:38 -03:00
break ;
}
2020-07-03 10:10:26 -04:00
} // no default case, so the compiler can warn about missing cases
2018-11-23 13:21:38 -03:00
}
2019-04-04 11:43:22 -03:00
/**
* Concatenate all sections with proper padding
*/
2018-11-23 13:21:38 -03:00
std : : string ToString ( ) const
{
std : : string ret ;
const size_t pad = m_max_pad + 4 ;
for ( const auto & s : m_sections ) {
2020-04-17 13:28:29 -04:00
// The left part of a section is assumed to be a single line, usually it is the name of the JSON struct or a
// brace like {, }, [, or ]
CHECK_NONFATAL ( s . m_left . find ( ' \n ' ) = = std : : string : : npos ) ;
2018-11-23 13:21:38 -03:00
if ( s . m_right . empty ( ) ) {
ret + = s . m_left ;
ret + = " \n " ;
continue ;
}
std : : string left = s . m_left ;
left . resize ( pad , ' ' ) ;
ret + = left ;
// Properly pad after newlines
std : : string right ;
size_t begin = 0 ;
size_t new_line_pos = s . m_right . find_first_of ( ' \n ' ) ;
while ( true ) {
right + = s . m_right . substr ( begin , new_line_pos - begin ) ;
if ( new_line_pos = = std : : string : : npos ) {
break ; //No new line
}
right + = " \n " + std : : string ( pad , ' ' ) ;
begin = s . m_right . find_first_not_of ( ' ' , new_line_pos + 1 ) ;
if ( begin = = std : : string : : npos ) {
break ; // Empty line
}
new_line_pos = s . m_right . find_first_of ( ' \n ' , begin + 1 ) ;
}
ret + = right ;
ret + = " \n " ;
}
return ret ;
}
} ;
2018-12-21 14:29:36 -03:00
RPCHelpMan : : RPCHelpMan ( std : : string name , std : : string description , std : : vector < RPCArg > args , RPCResults results , RPCExamples examples )
2020-06-26 12:16:22 -04:00
: RPCHelpMan { std : : move ( name ) , std : : move ( description ) , std : : move ( args ) , std : : move ( results ) , std : : move ( examples ) , nullptr } { }
RPCHelpMan : : RPCHelpMan ( std : : string name , std : : string description , std : : vector < RPCArg > args , RPCResults results , RPCExamples examples , RPCMethodImpl fun )
2018-12-21 14:29:36 -03:00
: m_name { std : : move ( name ) } ,
2020-06-26 12:16:22 -04:00
m_fun { std : : move ( fun ) } ,
2018-12-21 14:29:36 -03:00
m_description { std : : move ( description ) } ,
m_args { std : : move ( args ) } ,
m_results { std : : move ( results ) } ,
m_examples { std : : move ( examples ) }
2018-12-06 12:37:39 -03:00
{
std : : set < std : : string > named_args ;
for ( const auto & arg : m_args ) {
2020-04-12 17:09:23 -04:00
std : : vector < std : : string > names ;
boost : : split ( names , arg . m_names , boost : : is_any_of ( " | " ) ) ;
2018-12-06 12:37:39 -03:00
// Should have unique named arguments
2020-04-12 17:09:23 -04:00
for ( const std : : string & name : names ) {
CHECK_NONFATAL ( named_args . insert ( name ) . second ) ;
}
2018-12-06 12:37:39 -03:00
}
}
2018-12-21 14:29:36 -03:00
std : : string RPCResults : : ToDescriptionString ( ) const
{
std : : string result ;
for ( const auto & r : m_results ) {
2020-11-23 06:41:27 -03:00
if ( r . m_type = = RPCResult : : Type : : ANY ) continue ; // for testing only
2018-12-21 14:29:36 -03:00
if ( r . m_cond . empty ( ) ) {
result + = " \n Result: \n " ;
} else {
result + = " \n Result ( " + r . m_cond + " ): \n " ;
}
2020-01-09 14:00:57 -03:00
Sections sections ;
r . ToSections ( sections ) ;
result + = sections . ToString ( ) ;
2018-12-21 14:29:36 -03:00
}
return result ;
}
std : : string RPCExamples : : ToDescriptionString ( ) const
{
return m_examples . empty ( ) ? m_examples : " \n Examples: \n " + m_examples ;
}
2020-11-23 06:41:27 -03:00
UniValue RPCHelpMan : : HandleRequest ( const JSONRPCRequest & request ) const
2021-01-29 20:09:46 -03:00
{
2021-01-29 20:15:48 -03:00
if ( request . mode = = JSONRPCRequest : : GET_ARGS ) {
return GetArgMap ( ) ;
}
2021-01-29 20:09:46 -03:00
/*
* Check if the given request is valid according to this command or if
* the user is asking for help information , and throw help when appropriate .
*/
if ( request . mode = = JSONRPCRequest : : GET_HELP | | ! IsValidNumArgs ( request . params . size ( ) ) ) {
throw std : : runtime_error ( ToString ( ) ) ;
}
2020-11-23 06:41:27 -03:00
const UniValue ret = m_fun ( * this , request ) ;
CHECK_NONFATAL ( std : : any_of ( m_results . m_results . begin ( ) , m_results . m_results . end ( ) , [ ret ] ( const RPCResult & res ) { return res . MatchesType ( ret ) ; } ) ) ;
return ret ;
2021-01-29 20:09:46 -03:00
}
2019-02-12 22:16:29 -03:00
bool RPCHelpMan : : IsValidNumArgs ( size_t num_args ) const
{
size_t num_required_args = 0 ;
for ( size_t n = m_args . size ( ) ; n > 0 ; - - n ) {
if ( ! m_args . at ( n - 1 ) . IsOptional ( ) ) {
num_required_args = n ;
break ;
}
}
return num_required_args < = num_args & & num_args < = m_args . size ( ) ;
}
2020-06-26 12:16:22 -04:00
std : : vector < std : : string > RPCHelpMan : : GetArgNames ( ) const
{
std : : vector < std : : string > ret ;
for ( const auto & arg : m_args ) {
ret . emplace_back ( arg . m_names ) ;
}
return ret ;
}
2018-10-23 16:22:28 -03:00
std : : string RPCHelpMan : : ToString ( ) const
{
std : : string ret ;
2018-11-23 13:21:38 -03:00
// Oneline summary
2018-10-23 16:22:28 -03:00
ret + = m_name ;
2018-12-05 13:43:55 -03:00
bool was_optional { false } ;
2018-10-23 16:22:28 -03:00
for ( const auto & arg : m_args ) {
2020-07-15 13:45:01 -04:00
if ( arg . m_hidden ) break ; // Any arg that follows is also hidden
2019-02-12 22:16:29 -03:00
const bool optional = arg . IsOptional ( ) ;
2018-10-23 16:22:28 -03:00
ret + = " " ;
2018-12-10 18:56:51 -03:00
if ( optional ) {
2018-12-05 13:43:55 -03:00
if ( ! was_optional ) ret + = " ( " ;
was_optional = true ;
2018-10-23 16:22:28 -03:00
} else {
2018-12-05 13:43:55 -03:00
if ( was_optional ) ret + = " ) " ;
was_optional = false ;
2018-10-23 16:22:28 -03:00
}
2018-11-23 13:21:38 -03:00
ret + = arg . ToString ( /* oneline */ true ) ;
2018-10-23 16:22:28 -03:00
}
2018-12-05 13:43:55 -03:00
if ( was_optional ) ret + = " ) " ;
2018-10-23 16:22:28 -03:00
ret + = " \n " ;
2018-11-23 13:21:38 -03:00
// Description
2018-10-20 09:19:44 -03:00
ret + = m_description ;
2018-11-23 13:21:38 -03:00
// Arguments
Sections sections ;
for ( size_t i { 0 } ; i < m_args . size ( ) ; + + i ) {
const auto & arg = m_args . at ( i ) ;
2020-07-15 13:45:01 -04:00
if ( arg . m_hidden ) break ; // Any arg that follows is also hidden
2018-11-23 13:21:38 -03:00
if ( i = = 0 ) ret + = " \n Arguments: \n " ;
// Push named argument name and description
2020-04-12 17:09:23 -04:00
sections . m_sections . emplace_back ( : : ToString ( i + 1 ) + " . " + arg . GetFirstName ( ) , arg . ToDescriptionString ( ) ) ;
2018-11-23 13:21:38 -03:00
sections . m_max_pad = std : : max ( sections . m_max_pad , sections . m_sections . back ( ) . m_left . size ( ) ) ;
// Recursively push nested args
sections . Push ( arg ) ;
}
ret + = sections . ToString ( ) ;
2018-12-21 14:29:36 -03:00
// Result
ret + = m_results . ToDescriptionString ( ) ;
// Examples
ret + = m_examples . ToDescriptionString ( ) ;
2018-11-23 13:21:38 -03:00
return ret ;
}
2021-01-29 20:15:48 -03:00
UniValue RPCHelpMan : : GetArgMap ( ) const
2021-01-12 02:28:13 -03:00
{
2021-01-29 20:15:48 -03:00
UniValue arr { UniValue : : VARR } ;
2021-01-12 02:28:13 -03:00
for ( int i { 0 } ; i < int ( m_args . size ( ) ) ; + + i ) {
const auto & arg = m_args . at ( i ) ;
std : : vector < std : : string > arg_names ;
boost : : split ( arg_names , arg . m_names , boost : : is_any_of ( " | " ) ) ;
for ( const auto & arg_name : arg_names ) {
UniValue map { UniValue : : VARR } ;
map . push_back ( m_name ) ;
map . push_back ( i ) ;
map . push_back ( arg_name ) ;
map . push_back ( arg . m_type = = RPCArg : : Type : : STR | |
arg . m_type = = RPCArg : : Type : : STR_HEX ) ;
arr . push_back ( map ) ;
}
}
2021-01-29 20:15:48 -03:00
return arr ;
2021-01-12 02:28:13 -03:00
}
2020-04-12 17:09:23 -04:00
std : : string RPCArg : : GetFirstName ( ) const
{
return m_names . substr ( 0 , m_names . find ( " | " ) ) ;
}
std : : string RPCArg : : GetName ( ) const
{
CHECK_NONFATAL ( std : : string : : npos = = m_names . find ( " | " ) ) ;
return m_names ;
}
2019-02-12 22:16:29 -03:00
bool RPCArg : : IsOptional ( ) const
{
2021-04-14 10:01:00 -04:00
if ( m_fallback . index ( ) ! = 0 ) {
2019-02-12 22:16:29 -03:00
return true ;
} else {
2021-01-01 11:02:49 -03:00
return RPCArg : : Optional : : NO ! = std : : get < RPCArg : : Optional > ( m_fallback ) ;
2019-02-12 22:16:29 -03:00
}
}
2018-12-10 18:56:51 -03:00
std : : string RPCArg : : ToDescriptionString ( ) const
2018-11-23 13:21:38 -03:00
{
std : : string ret ;
ret + = " ( " ;
if ( m_type_str . size ( ) ! = 0 ) {
ret + = m_type_str . at ( 1 ) ;
} else {
switch ( m_type ) {
case Type : : STR_HEX :
case Type : : STR : {
ret + = " string " ;
break ;
}
case Type : : NUM : {
ret + = " numeric " ;
break ;
}
case Type : : AMOUNT : {
ret + = " numeric or string " ;
break ;
}
2019-02-27 18:45:47 -03:00
case Type : : RANGE : {
ret + = " numeric or array " ;
break ;
}
2018-11-23 13:21:38 -03:00
case Type : : BOOL : {
ret + = " boolean " ;
break ;
}
case Type : : OBJ :
case Type : : OBJ_USER_KEYS : {
ret + = " json object " ;
break ;
}
case Type : : ARR : {
ret + = " json array " ;
break ;
}
2020-07-03 10:10:26 -04:00
} // no default case, so the compiler can warn about missing cases
2018-11-23 13:21:38 -03:00
}
2021-01-01 11:02:49 -03:00
if ( m_fallback . index ( ) = = 1 ) {
2021-04-14 10:01:00 -04:00
ret + = " , optional, default= " + std : : get < RPCArg : : DefaultHint > ( m_fallback ) ;
} else if ( m_fallback . index ( ) = = 2 ) {
ret + = " , optional, default= " + std : : get < RPCArg : : Default > ( m_fallback ) . write ( ) ;
2018-12-10 18:56:51 -03:00
} else {
2021-01-01 11:02:49 -03:00
switch ( std : : get < RPCArg : : Optional > ( m_fallback ) ) {
2018-12-10 18:56:51 -03:00
case RPCArg : : Optional : : OMITTED : {
// nothing to do. Element is treated as if not present and has no default value
break ;
}
case RPCArg : : Optional : : OMITTED_NAMED_ARG : {
ret + = " , optional " ; // Default value is "null"
break ;
}
case RPCArg : : Optional : : NO : {
ret + = " , required " ;
break ;
}
2020-07-03 10:10:26 -04:00
} // no default case, so the compiler can warn about missing cases
2018-11-23 13:21:38 -03:00
}
ret + = " ) " ;
ret + = m_description . empty ( ) ? " " : " " + m_description ;
2018-10-23 16:22:28 -03:00
return ret ;
}
2020-01-09 14:00:57 -03:00
void RPCResult : : ToSections ( Sections & sections , const OuterType outer_type , const int current_indent ) const
{
// Indentation
const std : : string indent ( current_indent , ' ' ) ;
const std : : string indent_next ( current_indent + 2 , ' ' ) ;
// Elements in a JSON structure (dictionary or array) are separated by a comma
const std : : string maybe_separator { outer_type ! = OuterType : : NONE ? " , " : " " } ;
// The key name if recursed into an dictionary
const std : : string maybe_key {
outer_type = = OuterType : : OBJ ?
" \" " + this - > m_key_name + " \" : " :
" " } ;
// Format description with type
const auto Description = [ & ] ( const std : : string & type ) {
return " ( " + type + ( this - > m_optional ? " , optional " : " " ) + " ) " +
( this - > m_description . empty ( ) ? " " : " " + this - > m_description ) ;
} ;
switch ( m_type ) {
case Type : : ELISION : {
// If the inner result is empty, use three dots for elision
2020-04-02 12:12:32 -03:00
sections . PushSection ( { indent + " ... " + maybe_separator , m_description } ) ;
2020-01-09 14:00:57 -03:00
return ;
}
2020-11-23 06:41:27 -03:00
case Type : : ANY : {
CHECK_NONFATAL ( false ) ; // Only for testing
}
2020-01-09 14:00:57 -03:00
case Type : : NONE : {
2020-04-02 12:28:05 -03:00
sections . PushSection ( { indent + " null " + maybe_separator , Description ( " json null " ) } ) ;
2020-01-09 14:00:57 -03:00
return ;
}
case Type : : STR : {
sections . PushSection ( { indent + maybe_key + " \" str \" " + maybe_separator , Description ( " string " ) } ) ;
return ;
}
case Type : : STR_AMOUNT : {
sections . PushSection ( { indent + maybe_key + " n " + maybe_separator , Description ( " numeric " ) } ) ;
return ;
}
case Type : : STR_HEX : {
sections . PushSection ( { indent + maybe_key + " \" hex \" " + maybe_separator , Description ( " string " ) } ) ;
return ;
}
case Type : : NUM : {
sections . PushSection ( { indent + maybe_key + " n " + maybe_separator , Description ( " numeric " ) } ) ;
return ;
}
case Type : : NUM_TIME : {
sections . PushSection ( { indent + maybe_key + " xxx " + maybe_separator , Description ( " numeric " ) } ) ;
return ;
}
case Type : : BOOL : {
sections . PushSection ( { indent + maybe_key + " true|false " + maybe_separator , Description ( " boolean " ) } ) ;
return ;
}
case Type : : ARR_FIXED :
case Type : : ARR : {
sections . PushSection ( { indent + maybe_key + " [ " , Description ( " json array " ) } ) ;
for ( const auto & i : m_inner ) {
i . ToSections ( sections , OuterType : : ARR , current_indent + 2 ) ;
}
2020-04-02 12:22:55 -03:00
CHECK_NONFATAL ( ! m_inner . empty ( ) ) ;
if ( m_type = = Type : : ARR & & m_inner . back ( ) . m_type ! = Type : : ELISION ) {
2020-01-09 14:00:57 -03:00
sections . PushSection ( { indent_next + " ... " , " " } ) ;
2020-03-26 17:28:38 -03:00
} else {
// Remove final comma, which would be invalid JSON
sections . m_sections . back ( ) . m_left . pop_back ( ) ;
2020-01-09 14:00:57 -03:00
}
sections . PushSection ( { indent + " ] " + maybe_separator , " " } ) ;
return ;
}
case Type : : OBJ_DYN :
case Type : : OBJ : {
sections . PushSection ( { indent + maybe_key + " { " , Description ( " json object " ) } ) ;
for ( const auto & i : m_inner ) {
i . ToSections ( sections , OuterType : : OBJ , current_indent + 2 ) ;
}
2020-04-02 12:22:55 -03:00
CHECK_NONFATAL ( ! m_inner . empty ( ) ) ;
if ( m_type = = Type : : OBJ_DYN & & m_inner . back ( ) . m_type ! = Type : : ELISION ) {
2020-01-09 14:00:57 -03:00
// If the dictionary keys are dynamic, use three dots for continuation
sections . PushSection ( { indent_next + " ... " , " " } ) ;
2020-03-26 17:28:38 -03:00
} else {
// Remove final comma, which would be invalid JSON
sections . m_sections . back ( ) . m_left . pop_back ( ) ;
2020-01-09 14:00:57 -03:00
}
sections . PushSection ( { indent + " } " + maybe_separator , " " } ) ;
return ;
}
2020-07-03 10:10:26 -04:00
} // no default case, so the compiler can warn about missing cases
2020-01-09 14:00:57 -03:00
CHECK_NONFATAL ( false ) ;
}
2020-11-23 06:41:27 -03:00
bool RPCResult : : MatchesType ( const UniValue & result ) const
{
switch ( m_type ) {
case Type : : ELISION : {
return false ;
}
case Type : : ANY : {
return true ;
}
case Type : : NONE : {
return UniValue : : VNULL = = result . getType ( ) ;
}
case Type : : STR :
case Type : : STR_HEX : {
return UniValue : : VSTR = = result . getType ( ) ;
}
case Type : : NUM :
case Type : : STR_AMOUNT :
case Type : : NUM_TIME : {
return UniValue : : VNUM = = result . getType ( ) ;
}
case Type : : BOOL : {
return UniValue : : VBOOL = = result . getType ( ) ;
}
case Type : : ARR_FIXED :
case Type : : ARR : {
return UniValue : : VARR = = result . getType ( ) ;
}
case Type : : OBJ_DYN :
case Type : : OBJ : {
return UniValue : : VOBJ = = result . getType ( ) ;
}
} // no default case, so the compiler can warn about missing cases
CHECK_NONFATAL ( false ) ;
}
2018-12-04 15:30:06 -03:00
std : : string RPCArg : : ToStringObj ( const bool oneline ) const
2018-10-23 16:22:28 -03:00
{
2018-12-04 15:30:06 -03:00
std : : string res ;
res + = " \" " ;
2020-04-12 17:09:23 -04:00
res + = GetFirstName ( ) ;
2018-12-04 15:30:06 -03:00
if ( oneline ) {
res + = " \" : " ;
} else {
res + = " \" : " ;
}
2018-10-23 16:22:28 -03:00
switch ( m_type ) {
case Type : : STR :
return res + " \" str \" " ;
case Type : : STR_HEX :
return res + " \" hex \" " ;
case Type : : NUM :
return res + " n " ;
2019-02-27 18:45:47 -03:00
case Type : : RANGE :
return res + " n or [n,n] " ;
2018-10-23 16:22:28 -03:00
case Type : : AMOUNT :
return res + " amount " ;
case Type : : BOOL :
return res + " bool " ;
case Type : : ARR :
res + = " [ " ;
for ( const auto & i : m_inner ) {
2018-12-04 15:30:06 -03:00
res + = i . ToString ( oneline ) + " , " ;
2018-10-23 16:22:28 -03:00
}
return res + " ...] " ;
case Type : : OBJ :
case Type : : OBJ_USER_KEYS :
// Currently unused, so avoid writing dead code
2019-10-29 17:26:01 -03:00
CHECK_NONFATAL ( false ) ;
2020-07-03 10:10:26 -04:00
} // no default case, so the compiler can warn about missing cases
2019-10-29 17:26:01 -03:00
CHECK_NONFATAL ( false ) ;
2018-10-23 16:22:28 -03:00
}
2018-11-23 13:21:38 -03:00
std : : string RPCArg : : ToString ( const bool oneline ) const
2018-10-23 16:22:28 -03:00
{
2018-11-23 13:21:38 -03:00
if ( oneline & & ! m_oneline_description . empty ( ) ) return m_oneline_description ;
2018-10-20 09:19:44 -03:00
2018-10-23 16:22:28 -03:00
switch ( m_type ) {
case Type : : STR_HEX :
case Type : : STR : {
2020-04-12 17:09:23 -04:00
return " \" " + GetFirstName ( ) + " \" " ;
2018-10-23 16:22:28 -03:00
}
case Type : : NUM :
2019-02-27 18:45:47 -03:00
case Type : : RANGE :
2018-10-23 16:22:28 -03:00
case Type : : AMOUNT :
case Type : : BOOL : {
2020-04-12 17:09:23 -04:00
return GetFirstName ( ) ;
2018-10-23 16:22:28 -03:00
}
case Type : : OBJ :
case Type : : OBJ_USER_KEYS : {
2019-08-20 16:58:50 -04:00
const std : : string res = Join ( m_inner , " , " , [ & ] ( const RPCArg & i ) { return i . ToStringObj ( oneline ) ; } ) ;
2018-10-23 16:22:28 -03:00
if ( m_type = = Type : : OBJ ) {
return " { " + res + " } " ;
} else {
return " { " + res + " ,...} " ;
}
}
case Type : : ARR : {
std : : string res ;
for ( const auto & i : m_inner ) {
2018-11-23 13:21:38 -03:00
res + = i . ToString ( oneline ) + " , " ;
2018-10-23 16:22:28 -03:00
}
return " [ " + res + " ...] " ;
}
2020-07-03 10:10:26 -04:00
} // no default case, so the compiler can warn about missing cases
2019-10-29 17:26:01 -03:00
CHECK_NONFATAL ( false ) ;
2018-10-23 16:22:28 -03:00
}
2019-02-27 18:41:41 -03:00
2019-04-04 04:39:04 -03:00
static std : : pair < int64_t , int64_t > ParseRange ( const UniValue & value )
2019-02-27 18:41:41 -03:00
{
if ( value . isNum ( ) ) {
return { 0 , value . get_int64 ( ) } ;
}
if ( value . isArray ( ) & & value . size ( ) = = 2 & & value [ 0 ] . isNum ( ) & & value [ 1 ] . isNum ( ) ) {
int64_t low = value [ 0 ] . get_int64 ( ) ;
int64_t high = value [ 1 ] . get_int64 ( ) ;
if ( low > high ) throw JSONRPCError ( RPC_INVALID_PARAMETER , " Range specified as [begin,end] must not have begin after end " ) ;
return { low , high } ;
}
throw JSONRPCError ( RPC_INVALID_PARAMETER , " Range must be specified as end or as [begin,end] " ) ;
}
2019-04-04 04:39:04 -03:00
std : : pair < int64_t , int64_t > ParseDescriptorRange ( const UniValue & value )
{
int64_t low , high ;
std : : tie ( low , high ) = ParseRange ( value ) ;
if ( low < 0 ) {
throw JSONRPCError ( RPC_INVALID_PARAMETER , " Range should be greater or equal than 0 " ) ;
}
if ( ( high > > 31 ) ! = 0 ) {
throw JSONRPCError ( RPC_INVALID_PARAMETER , " End of range is too high " ) ;
}
if ( high > = low + 1000000 ) {
throw JSONRPCError ( RPC_INVALID_PARAMETER , " Range is too large " ) ;
}
return { low , high } ;
}
2019-02-16 20:24:15 -03:00
std : : vector < CScript > EvalDescriptorStringOrObject ( const UniValue & scanobject , FlatSigningProvider & provider )
{
std : : string desc_str ;
std : : pair < int64_t , int64_t > range = { 0 , 1000 } ;
if ( scanobject . isStr ( ) ) {
desc_str = scanobject . get_str ( ) ;
} else if ( scanobject . isObject ( ) ) {
UniValue desc_uni = find_value ( scanobject , " desc " ) ;
if ( desc_uni . isNull ( ) ) throw JSONRPCError ( RPC_INVALID_PARAMETER , " Descriptor needs to be provided in scan object " ) ;
desc_str = desc_uni . get_str ( ) ;
UniValue range_uni = find_value ( scanobject , " range " ) ;
if ( ! range_uni . isNull ( ) ) {
range = ParseDescriptorRange ( range_uni ) ;
}
} else {
throw JSONRPCError ( RPC_INVALID_PARAMETER , " Scan object needs to be either a string or an object " ) ;
}
2019-08-02 18:04:02 -04:00
std : : string error ;
auto desc = Parse ( desc_str , provider , error ) ;
2019-02-16 20:24:15 -03:00
if ( ! desc ) {
2019-08-02 19:19:53 -04:00
throw JSONRPCError ( RPC_INVALID_ADDRESS_OR_KEY , error ) ;
2019-02-16 20:24:15 -03:00
}
if ( ! desc - > IsRange ( ) ) {
range . first = 0 ;
range . second = 0 ;
}
std : : vector < CScript > ret ;
for ( int i = range . first ; i < = range . second ; + + i ) {
std : : vector < CScript > scripts ;
if ( ! desc - > Expand ( i , provider , scripts , provider ) ) {
throw JSONRPCError ( RPC_INVALID_ADDRESS_OR_KEY , strprintf ( " Cannot derive script without private keys: '%s' " , desc_str ) ) ;
}
std : : move ( scripts . begin ( ) , scripts . end ( ) , std : : back_inserter ( ret ) ) ;
}
return ret ;
}
2019-03-01 13:06:27 -03:00
UniValue GetServicesNames ( ServiceFlags services )
{
UniValue servicesNames ( UniValue : : VARR ) ;
2020-05-29 12:52:59 -04:00
for ( const auto & flag : serviceFlagsToStr ( services ) ) {
servicesNames . push_back ( flag ) ;
2020-02-16 22:53:13 -03:00
}
2019-03-01 13:06:27 -03:00
return servicesNames ;
}