2016-12-31 11:01:21 -07:00
// Copyright (c) 2009-2016 The Bitcoin Core developers
2014-11-20 10:19:29 +08:00
// Distributed under the MIT software license, see the accompanying
2012-06-28 23:18:38 -04:00
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
2013-11-18 01:25:17 +01:00
2016-01-15 11:55:17 +11:00
# include "rpc/server.h"
2013-11-18 01:25:17 +01:00
2015-07-05 14:17:46 +02:00
# include "chainparams.h"
2014-10-28 21:33:23 -04:00
# include "clientversion.h"
2017-08-07 14:38:39 +02:00
# include "core_io.h"
2016-12-01 16:06:41 -08:00
# include "validation.h"
2013-04-13 00:13:08 -05:00
# include "net.h"
2016-12-01 15:45:50 -08:00
# include "net_processing.h"
2013-04-13 00:13:08 -05:00
# include "netbase.h"
2017-01-19 21:18:46 -05:00
# include "policy/policy.h"
2013-04-13 00:13:08 -05:00
# include "protocol.h"
# include "sync.h"
2014-06-19 15:08:37 +02:00
# include "timedata.h"
2015-06-20 20:27:28 +02:00
# include "ui_interface.h"
2013-04-13 00:13:08 -05:00
# include "util.h"
2015-07-05 14:17:46 +02:00
# include "utilstrencodings.h"
2014-09-14 21:54:32 -04:00
# include "version.h"
2017-07-18 15:11:18 +02:00
# include "warnings.h"
2014-09-14 12:43:56 +02:00
2015-09-04 16:11:34 +02:00
# include <univalue.h>
2012-06-28 23:18:38 -04:00
2016-09-22 09:46:41 +02:00
UniValue getconnectioncount ( const JSONRPCRequest & request )
2012-06-28 23:18:38 -04:00
{
2016-09-22 09:46:41 +02:00
if ( request . fHelp | | request . params . size ( ) ! = 0 )
2017-01-04 13:22:19 +09:00
throw std : : runtime_error (
2012-06-28 23:18:38 -04:00
" getconnectioncount \n "
2013-10-29 22:29:44 +11:00
" \n Returns the number of connections to other nodes. \n "
2015-10-22 17:00:36 +02:00
" \n Result: \n "
2013-10-29 22:29:44 +11:00
" n (numeric) The connection count \n "
" \n Examples: \n "
+ HelpExampleCli ( " getconnectioncount " , " " )
+ HelpExampleRpc ( " getconnectioncount " , " " )
) ;
2012-06-28 23:18:38 -04:00
2016-04-16 18:30:03 -04:00
if ( ! g_connman )
throw JSONRPCError ( RPC_CLIENT_P2P_DISABLED , " Error: Peer-to-peer functionality missing or disabled " ) ;
2014-10-19 04:46:17 -04:00
2016-04-16 18:30:03 -04:00
return ( int ) g_connman - > GetNodeCount ( CConnman : : CONNECTIONS_ALL ) ;
2012-06-28 23:18:38 -04:00
}
2016-09-22 09:46:41 +02:00
UniValue ping ( const JSONRPCRequest & request )
2013-08-22 04:34:33 -07:00
{
2016-09-22 09:46:41 +02:00
if ( request . fHelp | | request . params . size ( ) ! = 0 )
2017-01-04 13:22:19 +09:00
throw std : : runtime_error (
2013-08-22 04:34:33 -07:00
" ping \n "
2013-10-29 22:29:44 +11:00
" \n Requests that a ping be sent to all other nodes, to measure ping time. \n "
2013-08-22 04:34:33 -07:00
" Results provided in getpeerinfo, pingtime and pingwait fields are decimal seconds. \n "
2014-01-30 14:13:30 +01:00
" Ping command is handled in queue with all other commands, so it measures processing backlog, not just network ping. \n "
2013-10-29 22:29:44 +11:00
" \n Examples: \n "
+ HelpExampleCli ( " ping " , " " )
+ HelpExampleRpc ( " ping " , " " )
) ;
2013-10-23 16:19:49 +02:00
2016-04-16 19:13:12 -04:00
if ( ! g_connman )
throw JSONRPCError ( RPC_CLIENT_P2P_DISABLED , " Error: Peer-to-peer functionality missing or disabled " ) ;
2013-08-22 04:34:33 -07:00
2016-04-16 19:13:12 -04:00
// Request that each node send a ping during next message processing pass
g_connman - > ForEachNode ( [ ] ( CNode * pnode ) {
pnode - > fPingQueued = true ;
} ) ;
2014-08-20 15:15:16 -04:00
return NullUniValue ;
2013-08-22 04:34:33 -07:00
}
2016-09-22 09:46:41 +02:00
UniValue getpeerinfo ( const JSONRPCRequest & request )
2012-06-29 17:24:53 -04:00
{
2016-09-22 09:46:41 +02:00
if ( request . fHelp | | request . params . size ( ) ! = 0 )
2017-01-04 13:22:19 +09:00
throw std : : runtime_error (
2012-06-29 17:24:53 -04:00
" getpeerinfo \n "
2013-10-29 22:29:44 +11:00
" \n Returns data about each connected network node as a json array of objects. \n "
2015-10-22 17:00:36 +02:00
" \n Result: \n "
2013-10-29 22:29:44 +11:00
" [ \n "
" { \n "
2014-07-15 07:13:10 -07:00
" \" id \" : n, (numeric) Peer index \n "
2017-06-20 01:57:31 +03:00
" \" addr \" : \" host:port \" , (string) The IP address and port of the peer \n "
2017-05-30 11:59:42 +02:00
" \" addrbind \" : \" ip:port \" , (string) Bind address of the connection to the peer \n "
" \" addrlocal \" : \" ip:port \" , (string) Local address as reported by the peer \n "
2014-06-06 03:15:55 -04:00
" \" services \" : \" xxxxxxxxxxxxxxxx \" , (string) The services offered \n "
2015-11-20 18:51:44 -05:00
" \" relaytxes \" :true|false, (boolean) Whether peer has asked us to relay transactions to it \n "
2013-10-29 22:29:44 +11:00
" \" lastsend \" : ttt, (numeric) The time in seconds since epoch (Jan 1 1970 GMT) of the last send \n "
" \" lastrecv \" : ttt, (numeric) The time in seconds since epoch (Jan 1 1970 GMT) of the last receive \n "
" \" bytessent \" : n, (numeric) The total bytes sent \n "
" \" bytesrecv \" : n, (numeric) The total bytes received \n "
" \" conntime \" : ttt, (numeric) The connection time in seconds since epoch (Jan 1 1970 GMT) \n "
2014-12-15 11:06:15 +01:00
" \" timeoffset \" : ttt, (numeric) The time offset in seconds \n "
2016-04-08 18:52:59 +02:00
" \" pingtime \" : n, (numeric) ping time (if available) \n "
" \" minping \" : n, (numeric) minimum observed ping time (if any at all) \n "
" \" pingwait \" : n, (numeric) ping wait (if non-zero) \n "
2013-10-29 22:29:44 +11:00
" \" version \" : v, (numeric) The peer version, such as 7001 \n "
" \" subver \" : \" /Satoshi:0.8.5/ \" , (string) The string version \n "
" \" inbound \" : true|false, (boolean) Inbound (true) or Outbound (false) \n "
2017-10-05 11:49:16 -04:00
" \" addnode \" : true|false, (boolean) Whether connection was due to addnode/-connect or if it was an automatic/inbound connection \n "
2013-10-29 22:29:44 +11:00
" \" startingheight \" : n, (numeric) The starting height (block) of the peer \n "
2014-09-16 09:00:36 +02:00
" \" banscore \" : n, (numeric) The ban score \n "
2014-07-12 00:03:10 +02:00
" \" synced_headers \" : n, (numeric) The last header we have in common with this peer \n "
" \" synced_blocks \" : n, (numeric) The last block we have in common with this peer \n "
" \" inflight \" : [ \n "
" n, (numeric) The heights of blocks we're currently asking from this peer \n "
" ... \n "
2017-01-11 14:29:46 +01:00
" ], \n "
2017-01-04 13:22:19 +09:00
" \" whitelisted \" : true|false, (boolean) Whether the peer is whitelisted \n "
2015-08-25 16:30:31 +02:00
" \" bytessent_per_msg \" : { \n "
2017-01-11 14:29:46 +01:00
" \" addr \" : n, (numeric) The total bytes sent aggregated by message type \n "
2015-08-25 16:30:31 +02:00
" ... \n "
2017-01-11 14:29:46 +01:00
" }, \n "
2015-08-25 16:30:31 +02:00
" \" bytesrecv_per_msg \" : { \n "
2017-01-11 14:29:46 +01:00
" \" addr \" : n, (numeric) The total bytes received aggregated by message type \n "
2015-08-25 16:30:31 +02:00
" ... \n "
" } \n "
2013-10-29 22:29:44 +11:00
" } \n "
" ,... \n "
2014-07-17 15:09:03 +02:00
" ] \n "
2013-10-29 22:29:44 +11:00
" \n Examples: \n "
+ HelpExampleCli ( " getpeerinfo " , " " )
+ HelpExampleRpc ( " getpeerinfo " , " " )
) ;
2012-06-29 17:24:53 -04:00
2016-04-16 18:30:03 -04:00
if ( ! g_connman )
throw JSONRPCError ( RPC_CLIENT_P2P_DISABLED , " Error: Peer-to-peer functionality missing or disabled " ) ;
2014-10-19 04:46:17 -04:00
2017-01-04 13:22:19 +09:00
std : : vector < CNodeStats > vstats ;
2016-04-16 18:30:03 -04:00
g_connman - > GetNodeStats ( vstats ) ;
2012-06-29 17:24:53 -04:00
2015-05-10 14:48:35 +02:00
UniValue ret ( UniValue : : VARR ) ;
2012-06-29 17:24:53 -04:00
2017-06-02 03:18:57 +02:00
for ( const CNodeStats & stats : vstats ) {
2015-05-10 14:48:35 +02:00
UniValue obj ( UniValue : : VOBJ ) ;
2013-11-18 01:25:17 +01:00
CNodeStateStats statestats ;
bool fStateStats = GetNodeStateStats ( stats . nodeid , statestats ) ;
2014-07-15 07:13:10 -07:00
obj . push_back ( Pair ( " id " , stats . nodeid ) ) ;
2012-06-29 17:24:53 -04:00
obj . push_back ( Pair ( " addr " , stats . addrName ) ) ;
2013-08-21 22:50:19 -07:00
if ( ! ( stats . addrLocal . empty ( ) ) )
obj . push_back ( Pair ( " addrlocal " , stats . addrLocal ) ) ;
2017-05-30 11:59:42 +02:00
if ( stats . addrBind . IsValid ( ) )
obj . push_back ( Pair ( " addrbind " , stats . addrBind . ToString ( ) ) ) ;
2014-06-06 03:15:55 -04:00
obj . push_back ( Pair ( " services " , strprintf ( " %016x " , stats . nServices ) ) ) ;
2015-11-20 18:51:44 -05:00
obj . push_back ( Pair ( " relaytxes " , stats . fRelayTxes ) ) ;
2014-05-06 14:58:43 +02:00
obj . push_back ( Pair ( " lastsend " , stats . nLastSend ) ) ;
obj . push_back ( Pair ( " lastrecv " , stats . nLastRecv ) ) ;
obj . push_back ( Pair ( " bytessent " , stats . nSendBytes ) ) ;
obj . push_back ( Pair ( " bytesrecv " , stats . nRecvBytes ) ) ;
obj . push_back ( Pair ( " conntime " , stats . nTimeConnected ) ) ;
2014-12-15 11:06:15 +01:00
obj . push_back ( Pair ( " timeoffset " , stats . nTimeOffset ) ) ;
2016-04-08 18:52:59 +02:00
if ( stats . dPingTime > 0.0 )
obj . push_back ( Pair ( " pingtime " , stats . dPingTime ) ) ;
2017-09-11 15:43:49 -04:00
if ( stats . dMinPing < static_cast < double > ( std : : numeric_limits < int64_t > : : max ( ) ) / 1e6 )
2016-10-14 21:11:38 +07:00
obj . push_back ( Pair ( " minping " , stats . dMinPing ) ) ;
2013-08-22 04:34:33 -07:00
if ( stats . dPingWait > 0.0 )
obj . push_back ( Pair ( " pingwait " , stats . dPingWait ) ) ;
2012-06-29 17:24:53 -04:00
obj . push_back ( Pair ( " version " , stats . nVersion ) ) ;
2013-11-26 12:52:21 +01:00
// Use the sanitized form of subver here, to avoid tricksy remote peers from
2017-01-18 16:15:37 +01:00
// corrupting or modifying the JSON output by putting special characters in
2013-11-26 12:52:21 +01:00
// their ver message.
obj . push_back ( Pair ( " subver " , stats . cleanSubVer ) ) ;
2012-06-29 17:24:53 -04:00
obj . push_back ( Pair ( " inbound " , stats . fInbound ) ) ;
2017-10-05 11:49:16 -04:00
obj . push_back ( Pair ( " addnode " , stats . m_manual_connection ) ) ;
2012-06-30 23:40:26 -04:00
obj . push_back ( Pair ( " startingheight " , stats . nStartingHeight ) ) ;
2013-11-18 01:25:17 +01:00
if ( fStateStats ) {
obj . push_back ( Pair ( " banscore " , statestats . nMisbehavior ) ) ;
2014-07-12 00:03:10 +02:00
obj . push_back ( Pair ( " synced_headers " , statestats . nSyncHeight ) ) ;
obj . push_back ( Pair ( " synced_blocks " , statestats . nCommonHeight ) ) ;
2015-05-10 14:48:35 +02:00
UniValue heights ( UniValue : : VARR ) ;
2017-06-02 03:18:57 +02:00
for ( int height : statestats . vHeightInFlight ) {
2014-07-12 00:03:10 +02:00
heights . push_back ( height ) ;
}
obj . push_back ( Pair ( " inflight " , heights ) ) ;
2013-11-18 01:25:17 +01:00
}
2014-06-21 13:34:36 +02:00
obj . push_back ( Pair ( " whitelisted " , stats . fWhitelisted ) ) ;
2012-06-29 17:24:53 -04:00
2015-08-25 16:30:31 +02:00
UniValue sendPerMsgCmd ( UniValue : : VOBJ ) ;
2017-06-02 03:18:57 +02:00
for ( const mapMsgCmdSize : : value_type & i : stats . mapSendBytesPerMsgCmd ) {
2015-08-25 16:30:31 +02:00
if ( i . second > 0 )
sendPerMsgCmd . push_back ( Pair ( i . first , i . second ) ) ;
}
obj . push_back ( Pair ( " bytessent_per_msg " , sendPerMsgCmd ) ) ;
UniValue recvPerMsgCmd ( UniValue : : VOBJ ) ;
2017-06-02 03:18:57 +02:00
for ( const mapMsgCmdSize : : value_type & i : stats . mapRecvBytesPerMsgCmd ) {
2015-08-25 16:30:31 +02:00
if ( i . second > 0 )
recvPerMsgCmd . push_back ( Pair ( i . first , i . second ) ) ;
}
obj . push_back ( Pair ( " bytesrecv_per_msg " , recvPerMsgCmd ) ) ;
2012-06-29 17:24:53 -04:00
ret . push_back ( obj ) ;
}
2012-09-18 15:07:58 -04:00
2012-06-29 17:24:53 -04:00
return ret ;
}
2012-06-28 23:18:38 -04:00
2016-09-22 09:46:41 +02:00
UniValue addnode ( const JSONRPCRequest & request )
2013-01-23 11:45:00 -05:00
{
2017-01-04 13:22:19 +09:00
std : : string strCommand ;
2017-08-14 19:44:02 -04:00
if ( ! request . params [ 1 ] . isNull ( ) )
2016-09-22 09:46:41 +02:00
strCommand = request . params [ 1 ] . get_str ( ) ;
if ( request . fHelp | | request . params . size ( ) ! = 2 | |
2013-01-23 11:45:00 -05:00
( strCommand ! = " onetry " & & strCommand ! = " add " & & strCommand ! = " remove " ) )
2017-01-04 13:22:19 +09:00
throw std : : runtime_error (
2013-10-29 22:29:44 +11:00
" addnode \" node \" \" add|remove|onetry \" \n "
2017-06-20 01:57:31 +03:00
" \n Attempts to add or remove a node from the addnode list. \n "
2013-10-29 22:29:44 +11:00
" Or try a connection to a node once. \n "
2017-10-06 17:27:04 -04:00
" Nodes added using addnode (or -connect) are protected from DoS disconnection and are not required to be \n "
" full nodes/support SegWit as other outbound peers are (though such peers will not be synced from). \n "
2013-10-29 22:29:44 +11:00
" \n Arguments: \n "
" 1. \" node \" (string, required) The node (see getpeerinfo for nodes) \n "
" 2. \" command \" (string, required) 'add' to add a node to the list, 'remove' to remove a node from the list, 'onetry' to try a connection to the node once \n "
" \n Examples: \n "
+ HelpExampleCli ( " addnode " , " \" 192.168.0.6:8333 \" \" onetry \" " )
+ HelpExampleRpc ( " addnode " , " \" 192.168.0.6:8333 \" , \" onetry \" " )
) ;
2013-01-23 11:45:00 -05:00
2016-04-16 15:59:10 -04:00
if ( ! g_connman )
throw JSONRPCError ( RPC_CLIENT_P2P_DISABLED , " Error: Peer-to-peer functionality missing or disabled " ) ;
2017-01-04 13:22:19 +09:00
std : : string strNode = request . params [ 0 ] . get_str ( ) ;
2013-01-23 11:45:00 -05:00
if ( strCommand = = " onetry " )
{
CAddress addr ;
2017-10-04 17:59:30 -04:00
g_connman - > OpenNetworkConnection ( addr , false , nullptr , strNode . c_str ( ) , false , false , true ) ;
2014-08-20 15:15:16 -04:00
return NullUniValue ;
2013-01-23 11:45:00 -05:00
}
if ( strCommand = = " add " )
{
2016-04-16 18:12:58 -04:00
if ( ! g_connman - > AddNode ( strNode ) )
2013-10-23 16:19:49 +02:00
throw JSONRPCError ( RPC_CLIENT_NODE_ALREADY_ADDED , " Error: Node already added " ) ;
2013-01-23 11:45:00 -05:00
}
else if ( strCommand = = " remove " )
{
2016-04-16 18:12:58 -04:00
if ( ! g_connman - > RemoveAddedNode ( strNode ) )
2013-10-23 16:19:49 +02:00
throw JSONRPCError ( RPC_CLIENT_NODE_NOT_ADDED , " Error: Node has not been added. " ) ;
2013-01-23 11:45:00 -05:00
}
2014-08-20 15:15:16 -04:00
return NullUniValue ;
2013-01-23 11:45:00 -05:00
}
2016-09-22 09:46:41 +02:00
UniValue disconnectnode ( const JSONRPCRequest & request )
2015-06-11 20:20:54 -07:00
{
2017-04-03 10:03:00 -04:00
if ( request . fHelp | | request . params . size ( ) = = 0 | | request . params . size ( ) > = 3 )
2017-01-04 13:22:19 +09:00
throw std : : runtime_error (
2017-04-03 10:03:00 -04:00
" disconnectnode \" [address] \" [nodeid] \n "
" \n Immediately disconnects from the specified peer node. \n "
" \n Strictly one out of 'address' and 'nodeid' can be provided to identify the node. \n "
" \n To disconnect by nodeid, either set 'address' to the empty string, or call using the named 'nodeid' argument only. \n "
2015-06-11 20:20:54 -07:00
" \n Arguments: \n "
2017-04-03 10:03:00 -04:00
" 1. \" address \" (string, optional) The IP address/port of the node \n "
" 2. \" nodeid \" (number, optional) The node ID (see getpeerinfo for node IDs) \n "
2015-06-11 20:20:54 -07:00
" \n Examples: \n "
+ HelpExampleCli ( " disconnectnode " , " \" 192.168.0.6:8333 \" " )
2017-04-03 10:03:00 -04:00
+ HelpExampleCli ( " disconnectnode " , " \" \" 1 " )
2015-06-11 20:20:54 -07:00
+ HelpExampleRpc ( " disconnectnode " , " \" 192.168.0.6:8333 \" " )
2017-04-03 10:03:00 -04:00
+ HelpExampleRpc ( " disconnectnode " , " \" \" , 1 " )
2015-06-11 20:20:54 -07:00
) ;
2016-04-16 18:30:03 -04:00
if ( ! g_connman )
throw JSONRPCError ( RPC_CLIENT_P2P_DISABLED , " Error: Peer-to-peer functionality missing or disabled " ) ;
2015-06-11 20:20:54 -07:00
2017-04-03 10:03:00 -04:00
bool success ;
const UniValue & address_arg = request . params [ 0 ] ;
2017-08-14 19:38:18 -04:00
const UniValue & id_arg = request . params [ 1 ] ;
2017-04-03 10:03:00 -04:00
if ( ! address_arg . isNull ( ) & & id_arg . isNull ( ) ) {
/* handle disconnect-by-address */
success = g_connman - > DisconnectNode ( address_arg . get_str ( ) ) ;
} else if ( ! id_arg . isNull ( ) & & ( address_arg . isNull ( ) | | ( address_arg . isStr ( ) & & address_arg . get_str ( ) . empty ( ) ) ) ) {
/* handle disconnect-by-id */
NodeId nodeid = ( NodeId ) id_arg . get_int64 ( ) ;
success = g_connman - > DisconnectNode ( nodeid ) ;
} else {
throw JSONRPCError ( RPC_INVALID_PARAMS , " Only one of address and nodeid should be provided. " ) ;
}
if ( ! success ) {
2016-04-16 18:30:03 -04:00
throw JSONRPCError ( RPC_CLIENT_NODE_NOT_CONNECTED , " Node not found in connected nodes " ) ;
2017-04-03 10:03:00 -04:00
}
2015-06-11 20:20:54 -07:00
return NullUniValue ;
}
2016-09-22 09:46:41 +02:00
UniValue getaddednodeinfo ( const JSONRPCRequest & request )
2013-01-23 11:48:17 -05:00
{
2016-09-22 09:46:41 +02:00
if ( request . fHelp | | request . params . size ( ) > 1 )
2017-01-04 13:22:19 +09:00
throw std : : runtime_error (
2016-06-26 20:32:37 +02:00
" getaddednodeinfo ( \" node \" ) \n "
2013-10-29 22:29:44 +11:00
" \n Returns information about the given added node, or all added nodes \n "
2013-01-23 11:48:17 -05:00
" (note that onetry addnodes are not listed here) \n "
2013-10-29 22:29:44 +11:00
" \n Arguments: \n "
2016-06-26 20:32:37 +02:00
" 1. \" node \" (string, optional) If provided, return information about this specific node, otherwise all nodes are returned. \n "
2013-10-29 22:29:44 +11:00
" \n Result: \n "
" [ \n "
" { \n "
2017-06-20 01:57:31 +03:00
" \" addednode \" : \" 192.168.0.201 \" , (string) The node IP address or name (as provided to addnode) \n "
2013-10-29 22:29:44 +11:00
" \" connected \" : true|false, (boolean) If connected \n "
2016-05-28 15:32:30 +02:00
" \" addresses \" : [ (list of objects) Only when connected = true \n "
2013-10-29 22:29:44 +11:00
" { \n "
2016-05-28 15:32:30 +02:00
" \" address \" : \" 192.168.0.201:8333 \" , (string) The bitcoin server IP and port we're connected to \n "
2013-10-29 22:29:44 +11:00
" \" connected \" : \" outbound \" (string) connection, inbound or outbound \n "
" } \n "
" ] \n "
" } \n "
" ,... \n "
" ] \n "
" \n Examples: \n "
2017-06-08 23:25:53 +02:00
+ HelpExampleCli ( " getaddednodeinfo " , " \" 192.168.0.201 \" " )
+ HelpExampleRpc ( " getaddednodeinfo " , " \" 192.168.0.201 \" " )
2013-10-29 22:29:44 +11:00
) ;
2013-01-23 11:48:17 -05:00
2016-04-16 18:12:58 -04:00
if ( ! g_connman )
throw JSONRPCError ( RPC_CLIENT_P2P_DISABLED , " Error: Peer-to-peer functionality missing or disabled " ) ;
std : : vector < AddedNodeInfo > vInfo = g_connman - > GetAddedNodeInfo ( ) ;
2013-01-23 11:48:17 -05:00
2017-08-14 19:38:18 -04:00
if ( ! request . params [ 0 ] . isNull ( ) ) {
2016-05-28 15:32:30 +02:00
bool found = false ;
for ( const AddedNodeInfo & info : vInfo ) {
2016-09-22 09:46:41 +02:00
if ( info . strAddedNode = = request . params [ 0 ] . get_str ( ) ) {
2016-05-28 15:32:30 +02:00
vInfo . assign ( 1 , info ) ;
found = true ;
2013-01-23 11:48:17 -05:00
break ;
}
2015-05-31 15:36:44 +02:00
}
2016-05-28 15:32:30 +02:00
if ( ! found ) {
2013-10-23 16:19:49 +02:00
throw JSONRPCError ( RPC_CLIENT_NODE_NOT_ADDED , " Error: Node has not been added. " ) ;
2014-01-27 10:31:22 +01:00
}
2013-01-23 11:48:17 -05:00
}
2016-05-28 15:32:30 +02:00
UniValue ret ( UniValue : : VARR ) ;
2013-01-23 11:48:17 -05:00
2016-05-28 15:32:30 +02:00
for ( const AddedNodeInfo & info : vInfo ) {
2015-05-10 14:48:35 +02:00
UniValue obj ( UniValue : : VOBJ ) ;
2016-05-28 15:32:30 +02:00
obj . push_back ( Pair ( " addednode " , info . strAddedNode ) ) ;
obj . push_back ( Pair ( " connected " , info . fConnected ) ) ;
2015-05-10 14:48:35 +02:00
UniValue addresses ( UniValue : : VARR ) ;
2016-05-28 15:32:30 +02:00
if ( info . fConnected ) {
UniValue address ( UniValue : : VOBJ ) ;
address . push_back ( Pair ( " address " , info . resolvedAddress . ToString ( ) ) ) ;
address . push_back ( Pair ( " connected " , info . fInbound ? " inbound " : " outbound " ) ) ;
addresses . push_back ( address ) ;
2013-01-23 11:48:17 -05:00
}
obj . push_back ( Pair ( " addresses " , addresses ) ) ;
ret . push_back ( obj ) ;
}
return ret ;
}
2016-09-22 09:46:41 +02:00
UniValue getnettotals ( const JSONRPCRequest & request )
2013-08-23 02:09:32 +10:00
{
2016-09-22 09:46:41 +02:00
if ( request . fHelp | | request . params . size ( ) > 0 )
2017-01-04 13:22:19 +09:00
throw std : : runtime_error (
2013-08-23 02:09:32 +10:00
" getnettotals \n "
2013-10-29 22:29:44 +11:00
" \n Returns information about network traffic, including bytes in, bytes out, \n "
" and current time. \n "
" \n Result: \n "
" { \n "
" \" totalbytesrecv \" : n, (numeric) Total bytes received \n "
2013-12-05 08:46:50 +01:00
" \" totalbytessent \" : n, (numeric) Total bytes sent \n "
2016-11-10 18:18:52 +09:00
" \" timemillis \" : t, (numeric) Current UNIX time in milliseconds \n "
2015-11-12 13:14:54 +01:00
" \" uploadtarget \" : \n "
" { \n "
" \" timeframe \" : n, (numeric) Length of the measuring timeframe in seconds \n "
" \" target \" : n, (numeric) Target in bytes \n "
" \" target_reached \" : true|false, (boolean) True if target is reached \n "
" \" serve_historical_blocks \" : true|false, (boolean) True if serving historical blocks \n "
" \" bytes_left_in_cycle \" : t, (numeric) Bytes left in current time cycle \n "
" \" time_left_in_cycle \" : t (numeric) Seconds left in current time cycle \n "
" } \n "
2013-10-29 22:29:44 +11:00
" } \n "
" \n Examples: \n "
+ HelpExampleCli ( " getnettotals " , " " )
+ HelpExampleRpc ( " getnettotals " , " " )
) ;
2016-04-18 21:44:42 -04:00
if ( ! g_connman )
throw JSONRPCError ( RPC_CLIENT_P2P_DISABLED , " Error: Peer-to-peer functionality missing or disabled " ) ;
2013-08-23 02:09:32 +10:00
2015-05-10 14:48:35 +02:00
UniValue obj ( UniValue : : VOBJ ) ;
2016-04-18 21:44:42 -04:00
obj . push_back ( Pair ( " totalbytesrecv " , g_connman - > GetTotalBytesRecv ( ) ) ) ;
obj . push_back ( Pair ( " totalbytessent " , g_connman - > GetTotalBytesSent ( ) ) ) ;
2014-05-06 14:58:43 +02:00
obj . push_back ( Pair ( " timemillis " , GetTimeMillis ( ) ) ) ;
2015-09-02 17:03:27 +02:00
UniValue outboundLimit ( UniValue : : VOBJ ) ;
2016-04-18 21:44:42 -04:00
outboundLimit . push_back ( Pair ( " timeframe " , g_connman - > GetMaxOutboundTimeframe ( ) ) ) ;
outboundLimit . push_back ( Pair ( " target " , g_connman - > GetMaxOutboundTarget ( ) ) ) ;
outboundLimit . push_back ( Pair ( " target_reached " , g_connman - > OutboundTargetReached ( false ) ) ) ;
outboundLimit . push_back ( Pair ( " serve_historical_blocks " , ! g_connman - > OutboundTargetReached ( true ) ) ) ;
outboundLimit . push_back ( Pair ( " bytes_left_in_cycle " , g_connman - > GetOutboundTargetBytesLeft ( ) ) ) ;
outboundLimit . push_back ( Pair ( " time_left_in_cycle " , g_connman - > GetMaxOutboundTimeLeftInCycle ( ) ) ) ;
2015-09-02 17:03:27 +02:00
obj . push_back ( Pair ( " uploadtarget " , outboundLimit ) ) ;
2013-08-23 02:09:32 +10:00
return obj ;
}
2014-05-05 13:22:28 +02:00
2015-05-13 21:29:19 +02:00
static UniValue GetNetworksInfo ( )
2014-07-30 15:35:14 +02:00
{
2015-05-10 14:48:35 +02:00
UniValue networks ( UniValue : : VARR ) ;
2014-07-30 15:35:14 +02:00
for ( int n = 0 ; n < NET_MAX ; + + n )
{
enum Network network = static_cast < enum Network > ( n ) ;
2017-05-23 20:04:38 -04:00
if ( network = = NET_UNROUTABLE | | network = = NET_INTERNAL )
2014-07-30 15:35:14 +02:00
continue ;
proxyType proxy ;
2015-05-10 14:48:35 +02:00
UniValue obj ( UniValue : : VOBJ ) ;
2014-07-30 15:35:14 +02:00
GetProxy ( network , proxy ) ;
obj . push_back ( Pair ( " name " , GetNetworkName ( network ) ) ) ;
obj . push_back ( Pair ( " limited " , IsLimited ( network ) ) ) ;
obj . push_back ( Pair ( " reachable " , IsReachable ( network ) ) ) ;
2017-01-04 13:22:19 +09:00
obj . push_back ( Pair ( " proxy " , proxy . IsValid ( ) ? proxy . proxy . ToStringIPPort ( ) : std : : string ( ) ) ) ;
2015-03-16 16:30:49 +01:00
obj . push_back ( Pair ( " proxy_randomize_credentials " , proxy . randomize_credentials ) ) ;
2014-07-30 15:35:14 +02:00
networks . push_back ( obj ) ;
}
return networks ;
}
2016-09-22 09:46:41 +02:00
UniValue getnetworkinfo ( const JSONRPCRequest & request )
2014-05-05 13:22:28 +02:00
{
2016-09-22 09:46:41 +02:00
if ( request . fHelp | | request . params . size ( ) ! = 0 )
2017-01-04 13:22:19 +09:00
throw std : : runtime_error (
2014-05-05 13:22:28 +02:00
" getnetworkinfo \n "
" Returns an object containing various state info regarding P2P networking. \n "
" \n Result: \n "
" { \n "
2014-09-25 11:56:05 +02:00
" \" version \" : xxxxx, (numeric) the server version \n "
" \" subversion \" : \" /Satoshi:x.x.x/ \" , (string) the server subversion string \n "
" \" protocolversion \" : xxxxx, (numeric) the protocol version \n "
" \" localservices \" : \" xxxxxxxxxxxxxxxx \" , (string) the services we offer to the network \n "
2016-05-12 14:09:43 +02:00
" \" localrelay \" : true|false, (bool) true if transaction relay is requested from peers \n "
2014-09-25 11:56:05 +02:00
" \" timeoffset \" : xxxxx, (numeric) the time offset \n "
" \" connections \" : xxxxx, (numeric) the number of connections \n "
2016-10-23 05:35:52 +00:00
" \" networkactive \" : true|false, (bool) whether p2p networking is enabled \n "
2014-09-25 11:56:05 +02:00
" \" networks \" : [ (array) information per network \n "
" { \n "
" \" name \" : \" xxx \" , (string) network (ipv4, ipv6 or onion) \n "
" \" limited \" : true|false, (boolean) is the network limited using -onlynet? \n "
" \" reachable \" : true|false, (boolean) is the network reachable? \n "
" \" proxy \" : \" host:port \" (string) the proxy that is used for this network, or empty if none \n "
2017-01-11 14:29:46 +01:00
" \" proxy_randomize_credentials \" : true|false, (string) Whether randomized credentials are used \n "
2014-09-25 11:56:05 +02:00
" } \n "
" ,... \n "
2014-07-30 15:35:14 +02:00
" ], \n "
2017-01-19 22:07:56 -05:00
" \" relayfee \" : x.xxxxxxxx, (numeric) minimum relay fee for transactions in " + CURRENCY_UNIT + " /kB \n "
2017-01-19 21:18:46 -05:00
" \" incrementalfee \" : x.xxxxxxxx, (numeric) minimum fee increment for mempool limiting or BIP 125 replacement in " + CURRENCY_UNIT + " /kB \n "
2014-09-25 11:56:05 +02:00
" \" localaddresses \" : [ (array) list of local addresses \n "
" { \n "
" \" address \" : \" xxxx \" , (string) network address \n "
" \" port \" : xxx, (numeric) network port \n "
" \" score \" : xxx (numeric) relative score \n "
" } \n "
" ,... \n "
2014-05-05 13:22:28 +02:00
" ] \n "
2017-07-17 15:55:14 -07:00
" \" warnings \" : \" ... \" (string) any network and blockchain warnings \n "
2014-05-05 13:22:28 +02:00
" } \n "
" \n Examples: \n "
+ HelpExampleCli ( " getnetworkinfo " , " " )
+ HelpExampleRpc ( " getnetworkinfo " , " " )
) ;
2014-10-19 04:46:17 -04:00
LOCK ( cs_main ) ;
2015-05-10 14:48:35 +02:00
UniValue obj ( UniValue : : VOBJ ) ;
2014-09-25 11:56:05 +02:00
obj . push_back ( Pair ( " version " , CLIENT_VERSION ) ) ;
2015-07-31 18:05:42 +02:00
obj . push_back ( Pair ( " subversion " , strSubVersion ) ) ;
2014-09-25 11:56:05 +02:00
obj . push_back ( Pair ( " protocolversion " , PROTOCOL_VERSION ) ) ;
2016-04-19 00:04:58 -04:00
if ( g_connman )
obj . push_back ( Pair ( " localservices " , strprintf ( " %016x " , g_connman - > GetLocalServices ( ) ) ) ) ;
2016-05-12 14:09:43 +02:00
obj . push_back ( Pair ( " localrelay " , fRelayTxes ) ) ;
2014-05-06 14:58:43 +02:00
obj . push_back ( Pair ( " timeoffset " , GetTimeOffset ( ) ) ) ;
2014-11-19 13:33:34 +01:00
if ( g_connman ) {
2016-10-23 05:35:52 +00:00
obj . push_back ( Pair ( " networkactive " , g_connman - > GetNetworkActive ( ) ) ) ;
2016-04-16 18:30:03 -04:00
obj . push_back ( Pair ( " connections " , ( int ) g_connman - > GetNodeCount ( CConnman : : CONNECTIONS_ALL ) ) ) ;
2014-11-19 13:33:34 +01:00
}
2014-07-30 15:35:14 +02:00
obj . push_back ( Pair ( " networks " , GetNetworksInfo ( ) ) ) ;
2014-07-03 14:25:32 -04:00
obj . push_back ( Pair ( " relayfee " , ValueFromAmount ( : : minRelayTxFee . GetFeePerK ( ) ) ) ) ;
2017-01-19 21:18:46 -05:00
obj . push_back ( Pair ( " incrementalfee " , ValueFromAmount ( : : incrementalRelayFee . GetFeePerK ( ) ) ) ) ;
2015-05-10 14:48:35 +02:00
UniValue localAddresses ( UniValue : : VARR ) ;
2014-05-05 13:22:28 +02:00
{
LOCK ( cs_mapLocalHost ) ;
2017-06-02 03:28:42 +02:00
for ( const std : : pair < CNetAddr , LocalServiceInfo > & item : mapLocalHost )
2014-05-05 13:22:28 +02:00
{
2015-05-10 14:48:35 +02:00
UniValue rec ( UniValue : : VOBJ ) ;
2014-05-05 13:22:28 +02:00
rec . push_back ( Pair ( " address " , item . first . ToString ( ) ) ) ;
rec . push_back ( Pair ( " port " , item . second . nPort ) ) ;
rec . push_back ( Pair ( " score " , item . second . nScore ) ) ;
localAddresses . push_back ( rec ) ;
}
}
obj . push_back ( Pair ( " localaddresses " , localAddresses ) ) ;
2015-06-11 16:12:34 -04:00
obj . push_back ( Pair ( " warnings " , GetWarnings ( " statusbar " ) ) ) ;
2014-05-05 13:22:28 +02:00
return obj ;
}
2015-05-19 10:07:46 +02:00
2016-09-22 09:46:41 +02:00
UniValue setban ( const JSONRPCRequest & request )
2015-05-19 10:07:46 +02:00
{
2017-01-04 13:22:19 +09:00
std : : string strCommand ;
2017-08-14 19:44:02 -04:00
if ( ! request . params [ 1 ] . isNull ( ) )
2016-09-22 09:46:41 +02:00
strCommand = request . params [ 1 ] . get_str ( ) ;
if ( request . fHelp | | request . params . size ( ) < 2 | |
2015-05-19 10:07:46 +02:00
( strCommand ! = " add " & & strCommand ! = " remove " ) )
2017-01-04 13:22:19 +09:00
throw std : : runtime_error (
2016-11-21 13:55:35 +01:00
" setban \" subnet \" \" add|remove \" (bantime) (absolute) \n "
2017-06-20 01:57:31 +03:00
" \n Attempts to add or remove an IP/Subnet from the banned list. \n "
2015-05-19 10:07:46 +02:00
" \n Arguments: \n "
2017-06-20 01:57:31 +03:00
" 1. \" subnet \" (string, required) The IP/Subnet (see getpeerinfo for nodes IP) with an optional netmask (default is /32 = single IP) \n "
" 2. \" command \" (string, required) 'add' to add an IP/Subnet to the list, 'remove' to remove an IP/Subnet from the list \n "
" 3. \" bantime \" (numeric, optional) time in seconds how long (or until when if [absolute] is set) the IP is banned (0 or empty means using the default time of 24h which can also be overwritten by the -bantime startup argument) \n "
" 4. \" absolute \" (boolean, optional) If set, the bantime must be an absolute timestamp in seconds since epoch (Jan 1 1970 GMT) \n "
2015-05-19 10:07:46 +02:00
" \n Examples: \n "
+ HelpExampleCli ( " setban " , " \" 192.168.0.6 \" \" add \" 86400 " )
2015-05-25 20:03:51 +02:00
+ HelpExampleCli ( " setban " , " \" 192.168.0.0/24 \" \" add \" " )
2016-08-14 15:35:27 +03:00
+ HelpExampleRpc ( " setban " , " \" 192.168.0.6 \" , \" add \" , 86400 " )
2015-05-19 10:07:46 +02:00
) ;
2016-04-16 17:43:11 -04:00
if ( ! g_connman )
throw JSONRPCError ( RPC_CLIENT_P2P_DISABLED , " Error: Peer-to-peer functionality missing or disabled " ) ;
2015-05-19 10:07:46 +02:00
2015-05-25 20:03:51 +02:00
CSubNet subNet ;
CNetAddr netAddr ;
bool isSubnet = false ;
2017-01-04 13:22:19 +09:00
if ( request . params [ 0 ] . get_str ( ) . find ( " / " ) ! = std : : string : : npos )
2015-05-25 20:03:51 +02:00
isSubnet = true ;
2016-05-31 13:05:52 -04:00
if ( ! isSubnet ) {
CNetAddr resolved ;
2016-09-22 09:46:41 +02:00
LookupHost ( request . params [ 0 ] . get_str ( ) . c_str ( ) , resolved , false ) ;
2016-05-31 13:05:52 -04:00
netAddr = resolved ;
}
2015-05-25 20:03:51 +02:00
else
2016-09-22 09:46:41 +02:00
LookupSubNet ( request . params [ 0 ] . get_str ( ) . c_str ( ) , subNet ) ;
2015-05-25 20:03:51 +02:00
if ( ! ( isSubnet ? subNet . IsValid ( ) : netAddr . IsValid ( ) ) )
2017-02-07 12:57:37 -05:00
throw JSONRPCError ( RPC_CLIENT_INVALID_IP_OR_SUBNET , " Error: Invalid IP/Subnet " ) ;
2015-05-19 10:07:46 +02:00
if ( strCommand = = " add " )
{
2016-04-16 17:43:11 -04:00
if ( isSubnet ? g_connman - > IsBanned ( subNet ) : g_connman - > IsBanned ( netAddr ) )
2015-05-25 20:03:51 +02:00
throw JSONRPCError ( RPC_CLIENT_NODE_ALREADY_ADDED , " Error: IP/Subnet already banned " ) ;
2015-05-19 10:07:46 +02:00
int64_t banTime = 0 ; //use standard bantime if not specified
2017-08-14 19:38:18 -04:00
if ( ! request . params [ 2 ] . isNull ( ) )
2016-09-22 09:46:41 +02:00
banTime = request . params [ 2 ] . get_int64 ( ) ;
2015-05-19 10:07:46 +02:00
2015-06-12 18:31:47 +02:00
bool absolute = false ;
2017-08-14 19:38:18 -04:00
if ( request . params [ 3 ] . isTrue ( ) )
2015-06-12 18:31:47 +02:00
absolute = true ;
2016-04-16 17:43:11 -04:00
isSubnet ? g_connman - > Ban ( subNet , BanReasonManuallyAdded , banTime , absolute ) : g_connman - > Ban ( netAddr , BanReasonManuallyAdded , banTime , absolute ) ;
2015-05-19 10:07:46 +02:00
}
else if ( strCommand = = " remove " )
{
2016-04-16 17:43:11 -04:00
if ( ! ( isSubnet ? g_connman - > Unban ( subNet ) : g_connman - > Unban ( netAddr ) ) )
2017-02-07 12:57:37 -05:00
throw JSONRPCError ( RPC_CLIENT_INVALID_IP_OR_SUBNET , " Error: Unban failed. Requested address/subnet was not previously banned. " ) ;
2015-05-19 10:07:46 +02:00
}
2015-06-12 18:31:47 +02:00
return NullUniValue ;
2015-05-19 10:07:46 +02:00
}
2016-09-22 09:46:41 +02:00
UniValue listbanned ( const JSONRPCRequest & request )
2015-05-19 10:07:46 +02:00
{
2016-09-22 09:46:41 +02:00
if ( request . fHelp | | request . params . size ( ) ! = 0 )
2017-01-04 13:22:19 +09:00
throw std : : runtime_error (
2015-05-19 10:07:46 +02:00
" listbanned \n "
2015-05-25 20:03:51 +02:00
" \n List all banned IPs/Subnets. \n "
2015-05-19 10:07:46 +02:00
" \n Examples: \n "
+ HelpExampleCli ( " listbanned " , " " )
+ HelpExampleRpc ( " listbanned " , " " )
) ;
2016-04-16 17:43:11 -04:00
if ( ! g_connman )
throw JSONRPCError ( RPC_CLIENT_P2P_DISABLED , " Error: Peer-to-peer functionality missing or disabled " ) ;
2015-06-26 21:38:33 +02:00
banmap_t banMap ;
2016-04-16 17:43:11 -04:00
g_connman - > GetBanned ( banMap ) ;
2015-05-19 10:07:46 +02:00
2015-06-12 18:31:47 +02:00
UniValue bannedAddresses ( UniValue : : VARR ) ;
2015-06-26 21:38:33 +02:00
for ( banmap_t : : iterator it = banMap . begin ( ) ; it ! = banMap . end ( ) ; it + + )
2015-05-19 10:07:46 +02:00
{
2015-06-26 21:38:33 +02:00
CBanEntry banEntry = ( * it ) . second ;
2015-06-12 18:31:47 +02:00
UniValue rec ( UniValue : : VOBJ ) ;
2015-05-19 10:07:46 +02:00
rec . push_back ( Pair ( " address " , ( * it ) . first . ToString ( ) ) ) ;
2015-06-26 21:38:33 +02:00
rec . push_back ( Pair ( " banned_until " , banEntry . nBanUntil ) ) ;
rec . push_back ( Pair ( " ban_created " , banEntry . nCreateTime ) ) ;
rec . push_back ( Pair ( " ban_reason " , banEntry . banReasonToString ( ) ) ) ;
2015-05-19 10:07:46 +02:00
bannedAddresses . push_back ( rec ) ;
}
return bannedAddresses ;
}
2016-09-22 09:46:41 +02:00
UniValue clearbanned ( const JSONRPCRequest & request )
2015-05-19 10:07:46 +02:00
{
2016-09-22 09:46:41 +02:00
if ( request . fHelp | | request . params . size ( ) ! = 0 )
2017-01-04 13:22:19 +09:00
throw std : : runtime_error (
2015-05-19 10:07:46 +02:00
" clearbanned \n "
" \n Clear all banned IPs. \n "
" \n Examples: \n "
+ HelpExampleCli ( " clearbanned " , " " )
+ HelpExampleRpc ( " clearbanned " , " " )
) ;
2016-04-16 17:43:11 -04:00
if ( ! g_connman )
throw JSONRPCError ( RPC_CLIENT_P2P_DISABLED , " Error: Peer-to-peer functionality missing or disabled " ) ;
2015-05-19 10:07:46 +02:00
2016-04-16 17:43:11 -04:00
g_connman - > ClearBanned ( ) ;
2015-05-19 10:07:46 +02:00
2015-06-12 18:31:47 +02:00
return NullUniValue ;
2015-05-19 10:07:46 +02:00
}
2016-03-29 19:43:02 +02:00
2014-11-19 13:33:34 +01:00
UniValue setnetworkactive ( const JSONRPCRequest & request )
2013-03-26 02:38:24 +01:00
{
2014-11-19 13:33:34 +01:00
if ( request . fHelp | | request . params . size ( ) ! = 1 ) {
2017-01-04 13:22:19 +09:00
throw std : : runtime_error (
2016-10-23 05:35:52 +00:00
" setnetworkactive true|false \n "
2016-11-21 13:55:35 +01:00
" \n Disable/enable all p2p network activity. \n "
" \n Arguments: \n "
" 1. \" state \" (boolean, required) true to enable networking, false to disable \n "
2013-03-26 02:38:24 +01:00
) ;
}
if ( ! g_connman ) {
throw JSONRPCError ( RPC_CLIENT_P2P_DISABLED , " Error: Peer-to-peer functionality missing or disabled " ) ;
}
2014-11-19 13:33:34 +01:00
g_connman - > SetNetworkActive ( request . params [ 0 ] . get_bool ( ) ) ;
2013-03-26 02:38:24 +01:00
return g_connman - > GetNetworkActive ( ) ;
}
2016-03-29 19:43:02 +02:00
static const CRPCCommand commands [ ] =
2017-06-08 17:38:23 -07:00
{ // category name actor (function) argNames
2016-03-29 19:43:02 +02:00
// --------------------- ------------------------ ----------------------- ----------
2017-06-08 17:38:23 -07:00
{ " network " , " getconnectioncount " , & getconnectioncount , { } } ,
{ " network " , " ping " , & ping , { } } ,
{ " network " , " getpeerinfo " , & getpeerinfo , { } } ,
{ " network " , " addnode " , & addnode , { " node " , " command " } } ,
{ " network " , " disconnectnode " , & disconnectnode , { " address " , " nodeid " } } ,
{ " network " , " getaddednodeinfo " , & getaddednodeinfo , { " node " } } ,
{ " network " , " getnettotals " , & getnettotals , { } } ,
{ " network " , " getnetworkinfo " , & getnetworkinfo , { } } ,
{ " network " , " setban " , & setban , { " subnet " , " command " , " bantime " , " absolute " } } ,
{ " network " , " listbanned " , & listbanned , { } } ,
{ " network " , " clearbanned " , & clearbanned , { } } ,
{ " network " , " setnetworkactive " , & setnetworkactive , { " state " } } ,
2016-03-29 19:43:02 +02:00
} ;
2016-06-07 18:42:42 +02:00
void RegisterNetRPCCommands ( CRPCTable & t )
2016-03-29 19:43:02 +02:00
{
for ( unsigned int vcidx = 0 ; vcidx < ARRAYLEN ( commands ) ; vcidx + + )
2016-06-07 18:42:42 +02:00
t . appendCommand ( commands [ vcidx ] . name , & commands [ vcidx ] ) ;
2016-03-29 19:43:02 +02:00
}