2020-04-16 13:14:08 -04:00
// Copyright (c) 2011-2020 The Bitcoin Core developers
2014-12-13 01:09:33 -03:00
// Distributed under the MIT software license, see the accompanying
2013-11-04 12:20:43 -03:00
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
2015-12-09 07:53:12 -03:00
# if defined(HAVE_CONFIG_H)
2017-11-09 21:57:53 -03:00
# include <config/bitcoin-config.h>
2015-12-09 07:53:12 -03:00
# endif
2017-11-09 21:57:53 -03:00
# include <qt/rpcconsole.h>
2017-08-15 12:31:26 -03:00
# include <qt/forms/ui_debugwindow.h>
2012-04-09 16:07:25 -03:00
2017-11-09 21:57:53 -03:00
# include <qt/bantablemodel.h>
# include <qt/clientmodel.h>
# include <qt/platformstyle.h>
2016-09-09 17:55:59 -03:00
# include <qt/walletmodel.h>
2017-11-09 21:57:53 -03:00
# include <chainparams.h>
2018-04-07 04:42:02 -03:00
# include <interfaces/node.h>
2017-11-09 21:57:53 -03:00
# include <netbase.h>
# include <rpc/server.h>
# include <rpc/client.h>
2018-10-29 05:13:07 -03:00
# include <util/strencodings.h>
2018-10-22 19:51:11 -03:00
# include <util/system.h>
2013-04-13 02:13:08 -03:00
2015-09-04 11:11:34 -03:00
# include <univalue.h>
2014-08-20 15:15:16 -04:00
2014-06-05 01:00:16 -04:00
# ifdef ENABLE_WALLET
2014-06-04 16:00:59 -04:00
# include <db_cxx.h>
2017-07-18 14:19:47 -04:00
# include <wallet/wallet.h>
2014-06-05 01:00:16 -04:00
# endif
2014-06-04 16:00:59 -04:00
2012-04-09 16:07:25 -03:00
# include <QKeyEvent>
2015-06-01 10:32:25 -03:00
# include <QMenu>
2016-11-16 09:32:15 -03:00
# include <QMessageBox>
2013-04-13 02:13:08 -03:00
# include <QScrollBar>
2019-08-23 13:30:46 -04:00
# include <QScreen>
2016-01-22 14:00:36 -03:00
# include <QSettings>
2013-04-13 02:13:08 -03:00
# include <QTime>
2015-08-28 11:46:20 -03:00
# include <QTimer>
2016-02-27 00:57:12 -03:00
# include <QStringList>
2013-04-13 02:13:08 -03:00
2012-08-31 03:43:40 -04:00
// TODO: add a scrollback limit, as there is currently none
2012-04-09 16:07:25 -03:00
// TODO: make it possible to filter out categories (esp debug messages when implemented)
// TODO: receive errors and debug messages through ClientModel
const int CONSOLE_HISTORY = 50 ;
2013-08-22 12:09:32 -04:00
const int INITIAL_TRAFFIC_GRAPH_MINS = 30 ;
2016-01-22 14:00:36 -03:00
const QSize FONT_RANGE ( 4 , 40 ) ;
const char fontSizeSettingsKey [ ] = " consoleFontSize " ;
2013-08-22 12:09:32 -04:00
2012-05-12 06:30:07 -04:00
const struct {
const char * url ;
const char * source ;
} ICON_MAPPING [ ] = {
{ " cmd-request " , " :/icons/tx_input " } ,
{ " cmd-reply " , " :/icons/tx_output " } ,
{ " cmd-error " , " :/icons/tx_output " } ,
{ " misc " , " :/icons/tx_inout " } ,
2017-08-07 01:36:37 -04:00
{ nullptr , nullptr }
2012-05-12 06:30:07 -04:00
} ;
2016-10-04 01:17:27 -03:00
namespace {
2015-03-13 17:51:27 -03:00
// don't add private key handling cmd's to the history
2016-10-04 01:17:27 -03:00
const QStringList historyFilter = QStringList ( )
2015-03-13 17:51:27 -03:00
< < " importprivkey "
2016-11-22 20:37:23 -03:00
< < " importmulti "
2017-09-12 18:01:12 -03:00
< < " sethdseed "
2016-10-04 11:07:50 -03:00
< < " signmessagewithprivkey "
2017-06-12 15:23:02 -04:00
< < " signrawtransactionwithkey "
2015-03-13 17:51:27 -03:00
< < " walletpassphrase "
< < " walletpassphrasechange "
< < " encryptwallet " ;
2016-10-04 01:17:27 -03:00
}
2012-04-09 16:07:25 -03:00
/* Object for executing console RPC commands in a separate thread.
*/
2013-01-23 17:51:02 -03:00
class RPCExecutor : public QObject
2012-04-09 16:07:25 -03:00
{
Q_OBJECT
2017-04-17 17:38:51 -03:00
public :
2018-07-26 11:15:32 -04:00
explicit RPCExecutor ( interfaces : : Node & node ) : m_node ( node ) { }
2013-01-23 17:51:02 -03:00
2015-07-14 08:59:05 -03:00
public Q_SLOTS :
2019-01-04 09:17:53 -03:00
void request ( const QString & command , const WalletModel * wallet_model ) ;
2013-01-23 17:51:02 -03:00
2015-07-14 08:59:05 -03:00
Q_SIGNALS :
2012-04-09 16:07:25 -03:00
void reply ( int category , const QString & command ) ;
2017-04-17 17:38:51 -03:00
private :
2018-04-07 04:42:02 -03:00
interfaces : : Node & m_node ;
2012-04-09 16:07:25 -03:00
} ;
2015-08-28 11:46:20 -03:00
/** Class for handling RPC timers
* ( used for e . g . re - locking the wallet after a timeout )
*/
class QtRPCTimerBase : public QObject , public RPCTimerBase
{
Q_OBJECT
public :
2018-09-13 14:36:41 -03:00
QtRPCTimerBase ( std : : function < void ( ) > & _func , int64_t millis ) :
2016-09-09 08:43:29 -03:00
func ( _func )
2015-08-28 11:46:20 -03:00
{
timer . setSingleShot ( true ) ;
2018-06-24 11:18:22 -04:00
connect ( & timer , & QTimer : : timeout , [ this ] { func ( ) ; } ) ;
2015-08-28 11:46:20 -03:00
timer . start ( millis ) ;
}
~ QtRPCTimerBase ( ) { }
private :
QTimer timer ;
2018-09-13 14:36:41 -03:00
std : : function < void ( ) > func ;
2015-08-28 11:46:20 -03:00
} ;
class QtRPCTimerInterface : public RPCTimerInterface
{
public :
~ QtRPCTimerInterface ( ) { }
const char * Name ( ) { return " Qt " ; }
2018-09-13 14:36:41 -03:00
RPCTimerBase * NewTimer ( std : : function < void ( ) > & func , int64_t millis )
2015-08-28 11:46:20 -03:00
{
return new QtRPCTimerBase ( func , millis ) ;
}
} ;
2017-08-15 12:31:26 -03:00
# include <qt/rpcconsole.moc>
2012-04-09 16:07:25 -03:00
2012-08-30 15:42:18 -04:00
/**
2016-11-16 07:56:32 -03:00
* Split shell command line into a list of arguments and optionally execute the command ( s ) .
2016-08-20 06:19:35 -03:00
* Aims to emulate \ c bash and friends .
2012-08-30 15:42:18 -04:00
*
2016-11-16 07:56:32 -03:00
* - Command nesting is possible with parenthesis ; for example : validateaddress ( getnewaddress ( ) )
2016-08-20 06:19:35 -03:00
* - Arguments are delimited with whitespace or comma
2012-08-30 15:42:18 -04:00
* - Extra whitespace at the beginning and end and between arguments will be ignored
2012-08-30 15:42:18 -04:00
* - Text can be " double " or ' single ' quoted
* - The backslash \ c \ is used as escape character
2012-08-30 15:42:18 -04:00
* - Outside quotes , any character can be escaped
2012-08-30 15:42:18 -04:00
* - Within double quotes , only escape \ c " and backslashes before a \ c " or another backslash
* - Within single quotes , no escaping is possible and no special interpretation takes place
2012-08-30 15:42:18 -04:00
*
2017-04-17 17:38:51 -03:00
* @ param [ in ] node optional node to execute command on
2018-08-08 15:12:14 -04:00
* @ param [ out ] strResult stringified result from the executed command ( chain )
2012-08-30 15:42:18 -04:00
* @ param [ in ] strCommand Command line to split
2016-11-16 07:56:32 -03:00
* @ param [ in ] fExecute set true if you want the command to be executed
2016-11-16 08:36:21 -03:00
* @ param [ out ] pstrFilteredOut Command line , filtered to remove any sensitive data
2012-08-30 15:42:18 -04:00
*/
2016-08-20 06:19:35 -03:00
2019-01-04 09:17:53 -03:00
bool RPCConsole : : RPCParseCommandLine ( interfaces : : Node * node , std : : string & strResult , const std : : string & strCommand , const bool fExecute , std : : string * const pstrFilteredOut , const WalletModel * wallet_model )
2012-04-09 16:07:25 -03:00
{
2016-08-20 06:19:35 -03:00
std : : vector < std : : vector < std : : string > > stack ;
stack . push_back ( std : : vector < std : : string > ( ) ) ;
2012-08-30 15:42:18 -04:00
enum CmdParseState
{
STATE_EATING_SPACES ,
2016-12-12 11:38:22 -03:00
STATE_EATING_SPACES_IN_ARG ,
2016-12-14 04:59:09 -03:00
STATE_EATING_SPACES_IN_BRACKETS ,
2012-08-30 15:42:18 -04:00
STATE_ARGUMENT ,
STATE_SINGLEQUOTED ,
STATE_DOUBLEQUOTED ,
STATE_ESCAPE_OUTER ,
2016-08-20 06:19:35 -03:00
STATE_ESCAPE_DOUBLEQUOTED ,
STATE_COMMAND_EXECUTED ,
STATE_COMMAND_EXECUTED_INNER
2012-08-30 15:42:18 -04:00
} state = STATE_EATING_SPACES ;
std : : string curarg ;
2016-08-20 06:19:35 -03:00
UniValue lastResult ;
2016-11-16 08:36:21 -03:00
unsigned nDepthInsideSensitive = 0 ;
2016-11-16 09:32:15 -03:00
size_t filter_begin_pos = 0 , chpos ;
2016-11-16 08:36:21 -03:00
std : : vector < std : : pair < size_t , size_t > > filter_ranges ;
2016-08-20 06:19:35 -03:00
2017-01-09 15:31:19 -03:00
auto add_to_current_stack = [ & ] ( const std : : string & strArg ) {
if ( stack . back ( ) . empty ( ) & & ( ! nDepthInsideSensitive ) & & historyFilter . contains ( QString : : fromStdString ( strArg ) , Qt : : CaseInsensitive ) ) {
2016-11-16 09:32:15 -03:00
nDepthInsideSensitive = 1 ;
filter_begin_pos = chpos ;
}
2017-03-22 21:34:27 -03:00
// Make sure stack is not empty before adding something
if ( stack . empty ( ) ) {
stack . push_back ( std : : vector < std : : string > ( ) ) ;
}
2017-01-09 15:31:19 -03:00
stack . back ( ) . push_back ( strArg ) ;
2016-11-16 09:32:15 -03:00
} ;
auto close_out_params = [ & ] ( ) {
if ( nDepthInsideSensitive ) {
if ( ! - - nDepthInsideSensitive ) {
assert ( filter_begin_pos ) ;
filter_ranges . push_back ( std : : make_pair ( filter_begin_pos , chpos ) ) ;
filter_begin_pos = 0 ;
}
}
stack . pop_back ( ) ;
} ;
2016-08-20 06:19:35 -03:00
std : : string strCommandTerminated = strCommand ;
if ( strCommandTerminated . back ( ) ! = ' \n ' )
strCommandTerminated + = " \n " ;
2016-11-16 09:32:15 -03:00
for ( chpos = 0 ; chpos < strCommandTerminated . size ( ) ; + + chpos )
2012-08-30 15:42:18 -04:00
{
2016-11-16 08:36:21 -03:00
char ch = strCommandTerminated [ chpos ] ;
2012-08-30 15:42:18 -04:00
switch ( state )
2012-05-12 12:14:29 -04:00
{
2016-08-20 06:19:35 -03:00
case STATE_COMMAND_EXECUTED_INNER :
case STATE_COMMAND_EXECUTED :
2012-08-30 15:42:18 -04:00
{
2016-08-20 06:19:35 -03:00
bool breakParsing = true ;
switch ( ch )
2012-08-30 15:42:18 -04:00
{
2016-08-20 06:19:35 -03:00
case ' [ ' : curarg . clear ( ) ; state = STATE_COMMAND_EXECUTED_INNER ; break ;
default :
if ( state = = STATE_COMMAND_EXECUTED_INNER )
{
if ( ch ! = ' ] ' )
{
// append char to the current argument (which is also used for the query command)
curarg + = ch ;
break ;
}
2016-11-16 07:56:32 -03:00
if ( curarg . size ( ) & & fExecute )
2016-08-20 06:19:35 -03:00
{
// if we have a value query, query arrays with index and objects with a string key
UniValue subelement ;
if ( lastResult . isArray ( ) )
{
for ( char argch : curarg )
2018-10-29 05:13:07 -03:00
if ( ! IsDigit ( argch ) )
2016-08-20 06:19:35 -03:00
throw std : : runtime_error ( " Invalid result query " ) ;
subelement = lastResult [ atoi ( curarg . c_str ( ) ) ] ;
}
else if ( lastResult . isObject ( ) )
subelement = find_value ( lastResult , curarg ) ;
else
throw std : : runtime_error ( " Invalid result query " ) ; //no array or object: abort
lastResult = subelement ;
}
state = STATE_COMMAND_EXECUTED ;
break ;
}
// don't break parsing when the char is required for the next argument
breakParsing = false ;
// pop the stack and return the result to the current command arguments
2016-11-16 09:32:15 -03:00
close_out_params ( ) ;
2016-08-20 06:19:35 -03:00
// don't stringify the json in case of a string to avoid doublequotes
if ( lastResult . isStr ( ) )
curarg = lastResult . get_str ( ) ;
else
curarg = lastResult . write ( 2 ) ;
// if we have a non empty result, use it as stack argument otherwise as general result
if ( curarg . size ( ) )
{
if ( stack . size ( ) )
2016-11-16 09:32:15 -03:00
add_to_current_stack ( curarg ) ;
2016-08-20 06:19:35 -03:00
else
strResult = curarg ;
}
curarg . clear ( ) ;
// assume eating space state
state = STATE_EATING_SPACES ;
2012-08-30 15:42:18 -04:00
}
2016-08-20 06:19:35 -03:00
if ( breakParsing )
break ;
}
case STATE_ARGUMENT : // In or after argument
2016-12-12 11:38:22 -03:00
case STATE_EATING_SPACES_IN_ARG :
2016-12-14 04:59:09 -03:00
case STATE_EATING_SPACES_IN_BRACKETS :
2016-08-20 06:19:35 -03:00
case STATE_EATING_SPACES : // Handle runs of whitespace
switch ( ch )
{
case ' " ' : state = STATE_DOUBLEQUOTED ; break ;
case ' \' ' : state = STATE_SINGLEQUOTED ; break ;
case ' \\ ' : state = STATE_ESCAPE_OUTER ; break ;
case ' ( ' : case ' ) ' : case ' \n ' :
2016-12-14 04:59:09 -03:00
if ( state = = STATE_EATING_SPACES_IN_ARG )
throw std : : runtime_error ( " Invalid Syntax " ) ;
2016-08-20 06:19:35 -03:00
if ( state = = STATE_ARGUMENT )
{
if ( ch = = ' ( ' & & stack . size ( ) & & stack . back ( ) . size ( ) > 0 )
2016-11-16 08:36:21 -03:00
{
if ( nDepthInsideSensitive ) {
+ + nDepthInsideSensitive ;
}
2016-08-20 06:19:35 -03:00
stack . push_back ( std : : vector < std : : string > ( ) ) ;
2016-11-16 08:36:21 -03:00
}
2016-12-12 11:38:22 -03:00
// don't allow commands after executed commands on baselevel
if ( ! stack . size ( ) )
throw std : : runtime_error ( " Invalid Syntax " ) ;
2016-11-16 09:32:15 -03:00
add_to_current_stack ( curarg ) ;
2016-08-20 06:19:35 -03:00
curarg . clear ( ) ;
2016-12-14 04:59:09 -03:00
state = STATE_EATING_SPACES_IN_BRACKETS ;
2016-08-20 06:19:35 -03:00
}
if ( ( ch = = ' ) ' | | ch = = ' \n ' ) & & stack . size ( ) > 0 )
{
2016-11-16 07:56:32 -03:00
if ( fExecute ) {
// Convert argument list to JSON objects in method-dependent way,
// and pass it along with the method name to the dispatcher.
2017-04-17 17:38:51 -03:00
UniValue params = RPCConvertValues ( stack . back ( ) [ 0 ] , std : : vector < std : : string > ( stack . back ( ) . begin ( ) + 1 , stack . back ( ) . end ( ) ) ) ;
std : : string method = stack . back ( ) [ 0 ] ;
std : : string uri ;
2017-07-18 14:19:47 -04:00
# ifdef ENABLE_WALLET
2019-01-04 09:17:53 -03:00
if ( wallet_model ) {
QByteArray encodedName = QUrl : : toPercentEncoding ( wallet_model - > getWalletName ( ) ) ;
2017-04-17 17:38:51 -03:00
uri = " /wallet/ " + std : : string ( encodedName . constData ( ) , encodedName . length ( ) ) ;
2017-07-18 14:19:47 -04:00
}
# endif
2017-04-17 17:38:51 -03:00
assert ( node ) ;
lastResult = node - > executeRpc ( method , params , uri ) ;
2016-11-16 07:56:32 -03:00
}
2016-08-20 06:19:35 -03:00
state = STATE_COMMAND_EXECUTED ;
curarg . clear ( ) ;
}
break ;
case ' ' : case ' , ' : case ' \t ' :
2016-12-14 04:59:09 -03:00
if ( state = = STATE_EATING_SPACES_IN_ARG & & curarg . empty ( ) & & ch = = ' , ' )
throw std : : runtime_error ( " Invalid Syntax " ) ;
else if ( state = = STATE_ARGUMENT ) // Space ends argument
2016-08-20 06:19:35 -03:00
{
2016-11-16 09:32:15 -03:00
add_to_current_stack ( curarg ) ;
2016-08-20 06:19:35 -03:00
curarg . clear ( ) ;
}
2016-12-14 04:59:09 -03:00
if ( ( state = = STATE_EATING_SPACES_IN_BRACKETS | | state = = STATE_ARGUMENT ) & & ch = = ' , ' )
{
state = STATE_EATING_SPACES_IN_ARG ;
break ;
}
2016-08-20 06:19:35 -03:00
state = STATE_EATING_SPACES ;
break ;
default : curarg + = ch ; state = STATE_ARGUMENT ;
2012-08-30 15:42:18 -04:00
}
2016-08-20 06:19:35 -03:00
break ;
case STATE_SINGLEQUOTED : // Single-quoted string
switch ( ch )
2012-08-30 15:42:18 -04:00
{
2016-08-20 06:19:35 -03:00
case ' \' ' : state = STATE_ARGUMENT ; break ;
default : curarg + = ch ;
2012-08-30 15:42:18 -04:00
}
2016-08-20 06:19:35 -03:00
break ;
case STATE_DOUBLEQUOTED : // Double-quoted string
switch ( ch )
2012-08-30 15:42:18 -04:00
{
2016-08-20 06:19:35 -03:00
case ' " ' : state = STATE_ARGUMENT ; break ;
case ' \\ ' : state = STATE_ESCAPE_DOUBLEQUOTED ; break ;
default : curarg + = ch ;
2012-08-30 15:42:18 -04:00
}
2016-08-20 06:19:35 -03:00
break ;
case STATE_ESCAPE_OUTER : // '\' outside quotes
curarg + = ch ; state = STATE_ARGUMENT ;
break ;
case STATE_ESCAPE_DOUBLEQUOTED : // '\' in double-quoted text
if ( ch ! = ' " ' & & ch ! = ' \\ ' ) curarg + = ' \\ ' ; // keep '\' for everything but the quote and '\' itself
curarg + = ch ; state = STATE_DOUBLEQUOTED ;
break ;
2012-05-12 12:14:29 -04:00
}
}
2016-11-16 08:36:21 -03:00
if ( pstrFilteredOut ) {
2016-11-16 09:32:15 -03:00
if ( STATE_COMMAND_EXECUTED = = state ) {
assert ( ! stack . empty ( ) ) ;
close_out_params ( ) ;
}
2016-11-16 08:36:21 -03:00
* pstrFilteredOut = strCommand ;
for ( auto i = filter_ranges . rbegin ( ) ; i ! = filter_ranges . rend ( ) ; + + i ) {
2016-11-16 09:32:15 -03:00
pstrFilteredOut - > replace ( i - > first , i - > second - i - > first , " (…) " ) ;
2016-11-16 08:36:21 -03:00
}
}
2012-08-30 15:42:18 -04:00
switch ( state ) // final state
2012-04-09 16:07:25 -03:00
{
2016-08-20 06:19:35 -03:00
case STATE_COMMAND_EXECUTED :
if ( lastResult . isStr ( ) )
strResult = lastResult . get_str ( ) ;
else
strResult = lastResult . write ( 2 ) ;
case STATE_ARGUMENT :
case STATE_EATING_SPACES :
return true ;
default : // ERROR to end in one of the other states
return false ;
2012-04-09 16:07:25 -03:00
}
2012-08-30 15:42:18 -04:00
}
2012-04-09 16:07:25 -03:00
2019-01-04 09:17:53 -03:00
void RPCExecutor : : request ( const QString & command , const WalletModel * wallet_model )
2012-08-30 15:42:18 -04:00
{
2012-08-31 11:40:13 -04:00
try
{
2016-08-20 06:19:35 -03:00
std : : string result ;
std : : string executableCommand = command . toStdString ( ) + " \n " ;
2017-11-18 01:43:17 -03:00
// Catch the console-only-help command before RPC call is executed and reply with help text as-if a RPC reply.
2019-01-16 08:50:06 -03:00
if ( executableCommand = = " help-console \n " ) {
2017-11-18 01:43:17 -03:00
Q_EMIT reply ( RPCConsole : : CMD_REPLY , QString ( ( " \n "
" This console accepts RPC commands using the standard syntax. \n "
" example: getblockhash 0 \n \n "
2019-01-16 08:50:06 -03:00
" This console can also accept RPC commands using the parenthesized syntax. \n "
2017-11-18 01:43:17 -03:00
" example: getblockhash(0) \n \n "
" Commands may be nested when specified with the parenthesized syntax. \n "
" example: getblock(getblockhash(0) 1) \n \n "
" A space or a comma can be used to delimit arguments for either syntax. \n "
" example: getblockhash 0 \n "
" getblockhash,0 \n \n "
2019-01-16 08:50:06 -03:00
" Named results can be queried with a non-quoted key string in brackets using the parenthesized syntax. \n "
" example: getblock(getblockhash(0) 1)[tx] \n \n "
2017-11-18 01:43:17 -03:00
2019-01-16 08:50:06 -03:00
" Results without keys can be queried with an integer in brackets using the parenthesized syntax. \n "
" example: getblock(getblockhash(0),1)[tx][0] \n \n " ) ) ) ;
2017-11-18 01:43:17 -03:00
return ;
}
2019-01-04 09:17:53 -03:00
if ( ! RPCConsole : : RPCExecuteCommandLine ( m_node , result , executableCommand , nullptr , wallet_model ) ) {
2016-08-20 06:19:35 -03:00
Q_EMIT reply ( RPCConsole : : CMD_ERROR , QString ( " Parse error: unbalanced ' or \" " ) ) ;
return ;
}
2017-11-18 01:43:17 -03:00
2016-08-20 06:19:35 -03:00
Q_EMIT reply ( RPCConsole : : CMD_REPLY , QString : : fromStdString ( result ) ) ;
2012-04-09 16:07:25 -03:00
}
2014-08-20 15:15:16 -04:00
catch ( UniValue & objError )
2012-04-09 16:07:25 -03:00
{
2012-08-31 11:40:13 -04:00
try // Nice formatting for standard-format error
{
int code = find_value ( objError , " code " ) . get_int ( ) ;
std : : string message = find_value ( objError , " message " ) . get_str ( ) ;
2015-07-14 08:59:05 -03:00
Q_EMIT reply ( RPCConsole : : CMD_ERROR , QString : : fromStdString ( message ) + " (code " + QString : : number ( code ) + " ) " ) ;
2012-08-31 11:40:13 -04:00
}
2014-12-07 09:29:06 -03:00
catch ( const std : : runtime_error & ) // raised when converting to invalid type, i.e. missing code or message
2012-08-30 15:42:18 -04:00
{ // Show raw JSON object
2015-07-14 08:59:05 -03:00
Q_EMIT reply ( RPCConsole : : CMD_ERROR , QString : : fromStdString ( objError . write ( ) ) ) ;
2012-08-31 11:40:13 -04:00
}
2012-04-09 16:07:25 -03:00
}
2014-12-07 09:29:06 -03:00
catch ( const std : : exception & e )
2012-04-09 16:07:25 -03:00
{
2015-07-14 08:59:05 -03:00
Q_EMIT reply ( RPCConsole : : CMD_ERROR , QString ( " Error: " ) + QString : : fromStdString ( e . what ( ) ) ) ;
2012-04-09 16:07:25 -03:00
}
}
2018-04-07 04:42:02 -03:00
RPCConsole : : RPCConsole ( interfaces : : Node & node , const PlatformStyle * _platformStyle , QWidget * parent ) :
2014-11-10 12:41:57 -03:00
QWidget ( parent ) ,
2017-04-17 17:38:51 -03:00
m_node ( node ) ,
2012-04-09 16:07:25 -03:00
ui ( new Ui : : RPCConsole ) ,
2018-04-11 06:56:44 -03:00
platformStyle ( _platformStyle )
2012-04-09 16:07:25 -03:00
{
ui - > setupUi ( this ) ;
2017-09-15 01:38:42 -03:00
QSettings settings ;
if ( ! restoreGeometry ( settings . value ( " RPCConsoleWindowGeometry " ) . toByteArray ( ) ) ) {
// Restore failed (perhaps missing setting), center the window
2019-08-23 13:30:46 -04:00
move ( QGuiApplication : : primaryScreen ( ) - > availableGeometry ( ) . center ( ) - frameGeometry ( ) . center ( ) ) ;
2017-09-15 01:38:42 -03:00
}
2012-04-09 16:07:25 -03:00
2018-10-05 18:05:51 -03:00
QChar nonbreaking_hyphen ( 8209 ) ;
ui - > dataDir - > setToolTip ( ui - > dataDir - > toolTip ( ) . arg ( QString ( nonbreaking_hyphen ) + " datadir " ) ) ;
ui - > blocksDir - > setToolTip ( ui - > blocksDir - > toolTip ( ) . arg ( QString ( nonbreaking_hyphen ) + " blocksdir " ) ) ;
2019-06-26 10:28:13 -04:00
ui - > openDebugLogfileButton - > setToolTip ( ui - > openDebugLogfileButton - > toolTip ( ) . arg ( PACKAGE_NAME ) ) ;
2015-12-09 07:53:12 -03:00
2015-07-28 10:20:14 -03:00
if ( platformStyle - > getImagesOnButtons ( ) ) {
ui - > openDebugLogfileButton - > setIcon ( platformStyle - > SingleColorIcon ( " :/icons/export " ) ) ;
}
ui - > clearButton - > setIcon ( platformStyle - > SingleColorIcon ( " :/icons/remove " ) ) ;
2016-01-22 14:00:36 -03:00
ui - > fontBiggerButton - > setIcon ( platformStyle - > SingleColorIcon ( " :/icons/fontbigger " ) ) ;
ui - > fontSmallerButton - > setIcon ( platformStyle - > SingleColorIcon ( " :/icons/fontsmaller " ) ) ;
2012-05-20 09:49:17 -04:00
2012-04-09 16:07:25 -03:00
// Install event filter for up and down arrow
ui - > lineEdit - > installEventFilter ( this ) ;
2012-09-09 15:07:22 -03:00
ui - > messagesWidget - > installEventFilter ( this ) ;
2012-04-09 16:07:25 -03:00
2018-06-24 11:18:22 -04:00
connect ( ui - > clearButton , & QPushButton : : clicked , this , & RPCConsole : : clear ) ;
connect ( ui - > fontBiggerButton , & QPushButton : : clicked , this , & RPCConsole : : fontBigger ) ;
connect ( ui - > fontSmallerButton , & QPushButton : : clicked , this , & RPCConsole : : fontSmaller ) ;
connect ( ui - > btnClearTrafficGraph , & QPushButton : : clicked , ui - > trafficGraph , & TrafficGraphWidget : : clear ) ;
2012-04-09 16:07:25 -03:00
2018-03-18 05:35:51 -03:00
// disable the wallet selector by default
ui - > WalletSelector - > setVisible ( false ) ;
ui - > WalletSelectorLabel - > setVisible ( false ) ;
2014-06-04 16:00:59 -04:00
// set library version labels
# ifdef ENABLE_WALLET
2018-07-31 14:02:34 -04:00
ui - > berkeleyDBVersion - > setText ( DbEnv : : version ( nullptr , nullptr , nullptr ) ) ;
2014-06-04 16:00:59 -04:00
# else
ui - > label_berkeleyDBVersion - > hide ( ) ;
ui - > berkeleyDBVersion - > hide ( ) ;
# endif
2015-08-28 11:46:20 -03:00
// Register RPC timer interface
rpcTimerInterface = new QtRPCTimerInterface ( ) ;
2016-01-08 07:03:52 -03:00
// avoid accidentally overwriting an existing, non QTThread
// based timer interface
2017-04-17 17:38:51 -03:00
m_node . rpcSetTimerInterfaceIfUnset ( rpcTimerInterface ) ;
2012-06-14 13:18:30 -04:00
2013-08-22 12:09:32 -04:00
setTrafficGraphRange ( INITIAL_TRAFFIC_GRAPH_MINS ) ;
2014-06-23 02:06:52 -04:00
2014-05-23 13:09:59 -04:00
ui - > detailWidget - > hide ( ) ;
2014-06-04 06:06:18 -04:00
ui - > peerHeading - > setText ( tr ( " Select a peer to view detailed information. " ) ) ;
2012-04-09 16:07:25 -03:00
2016-01-22 14:00:36 -03:00
consoleFontSize = settings . value ( fontSizeSettingsKey , QFontInfo ( QFont ( ) ) . pointSize ( ) ) . toInt ( ) ;
2012-04-09 16:07:25 -03:00
clear ( ) ;
2019-04-07 15:33:35 -04:00
GUIUtil : : handleCloseWindowShortcut ( this ) ;
2012-04-09 16:07:25 -03:00
}
RPCConsole : : ~ RPCConsole ( )
{
2017-09-15 01:38:42 -03:00
QSettings settings ;
settings . setValue ( " RPCConsoleWindowGeometry " , saveGeometry ( ) ) ;
2017-04-17 17:38:51 -03:00
m_node . rpcUnsetTimerInterface ( rpcTimerInterface ) ;
2015-08-28 11:46:20 -03:00
delete rpcTimerInterface ;
2012-04-09 16:07:25 -03:00
delete ui ;
}
bool RPCConsole : : eventFilter ( QObject * obj , QEvent * event )
{
2012-09-09 15:07:22 -03:00
if ( event - > type ( ) = = QEvent : : KeyPress ) // Special key handling
2012-04-09 16:07:25 -03:00
{
2012-09-09 15:07:22 -03:00
QKeyEvent * keyevt = static_cast < QKeyEvent * > ( event ) ;
int key = keyevt - > key ( ) ;
Qt : : KeyboardModifiers mod = keyevt - > modifiers ( ) ;
switch ( key )
2012-04-09 16:07:25 -03:00
{
2012-09-09 15:07:22 -03:00
case Qt : : Key_Up : if ( obj = = ui - > lineEdit ) { browseHistory ( - 1 ) ; return true ; } break ;
case Qt : : Key_Down : if ( obj = = ui - > lineEdit ) { browseHistory ( 1 ) ; return true ; } break ;
case Qt : : Key_PageUp : /* pass paging keys to messages widget */
case Qt : : Key_PageDown :
if ( obj = = ui - > lineEdit )
2012-04-09 16:07:25 -03:00
{
2012-09-09 15:07:22 -03:00
QApplication : : postEvent ( ui - > messagesWidget , new QKeyEvent ( * keyevt ) ) ;
return true ;
}
break ;
2016-05-30 22:30:35 -04:00
case Qt : : Key_Return :
case Qt : : Key_Enter :
// forward these events to lineEdit
if ( obj = = autoCompleter - > popup ( ) ) {
QApplication : : postEvent ( ui - > lineEdit , new QKeyEvent ( * keyevt ) ) ;
2018-10-09 17:53:15 -03:00
autoCompleter - > popup ( ) - > hide ( ) ;
2016-05-30 22:30:35 -04:00
return true ;
}
break ;
2012-09-09 15:07:22 -03:00
default :
// Typing in messages widget brings focus to line edit, and redirects key there
// Exclude most combinations and keys that emit no text, except paste shortcuts
if ( obj = = ui - > messagesWidget & & (
( ! mod & & ! keyevt - > text ( ) . isEmpty ( ) & & key ! = Qt : : Key_Tab ) | |
( ( mod & Qt : : ControlModifier ) & & key = = Qt : : Key_V ) | |
( ( mod & Qt : : ShiftModifier ) & & key = = Qt : : Key_Insert ) ) )
{
ui - > lineEdit - > setFocus ( ) ;
QApplication : : postEvent ( ui - > lineEdit , new QKeyEvent ( * keyevt ) ) ;
return true ;
2012-04-09 16:07:25 -03:00
}
}
}
2014-11-10 12:41:57 -03:00
return QWidget : : eventFilter ( obj , event ) ;
2012-04-09 16:07:25 -03:00
}
void RPCConsole : : setClientModel ( ClientModel * model )
{
2013-08-22 12:09:32 -04:00
clientModel = model ;
2018-12-05 17:51:49 -03:00
bool wallet_enabled { false } ;
# ifdef ENABLE_WALLET
wallet_enabled = WalletModel : : isWalletEnabled ( ) ;
# endif // ENABLE_WALLET
if ( model & & ! wallet_enabled ) {
// Show warning, for example if this is a prerelease version
connect ( model , & ClientModel : : alertsChanged , this , & RPCConsole : : updateAlerts ) ;
updateAlerts ( model - > getStatusBarWarnings ( ) ) ;
}
2013-08-22 12:09:32 -04:00
ui - > trafficGraph - > setClientModel ( model ) ;
2015-06-26 05:23:51 -03:00
if ( model & & clientModel - > getPeerTableModel ( ) & & clientModel - > getBanTableModel ( ) ) {
2013-06-03 08:10:14 -04:00
// Keep up to date with client
setNumConnections ( model - > getNumConnections ( ) ) ;
2018-06-24 11:18:22 -04:00
connect ( model , & ClientModel : : numConnectionsChanged , this , & RPCConsole : : setNumConnections ) ;
2013-06-03 08:10:14 -04:00
2018-04-07 04:42:02 -03:00
interfaces : : Node & node = clientModel - > node ( ) ;
2017-04-17 16:37:36 -03:00
setNumBlocks ( node . getNumBlocks ( ) , QDateTime : : fromTime_t ( node . getLastBlockTime ( ) ) , node . getVerificationProgress ( ) , false ) ;
2018-06-24 11:18:22 -04:00
connect ( model , & ClientModel : : numBlocksChanged , this , & RPCConsole : : setNumBlocks ) ;
2012-04-09 16:07:25 -03:00
2013-03-25 23:07:06 -03:00
updateNetworkState ( ) ;
2018-06-24 11:18:22 -04:00
connect ( model , & ClientModel : : networkActiveChanged , this , & RPCConsole : : setNetworkActive ) ;
2013-03-25 23:07:06 -03:00
2017-04-17 16:37:36 -03:00
updateTrafficStats ( node . getTotalBytesRecv ( ) , node . getTotalBytesSent ( ) ) ;
2018-06-24 11:18:22 -04:00
connect ( model , & ClientModel : : bytesChanged , this , & RPCConsole : : updateTrafficStats ) ;
2013-08-22 12:09:32 -04:00
2018-06-24 11:18:22 -04:00
connect ( model , & ClientModel : : mempoolSizeChanged , this , & RPCConsole : : setMempoolSize ) ;
2015-11-09 07:45:07 -03:00
2014-05-23 13:09:59 -04:00
// set up peer table
ui - > peerWidget - > setModel ( model - > getPeerTableModel ( ) ) ;
ui - > peerWidget - > verticalHeader ( ) - > hide ( ) ;
ui - > peerWidget - > setEditTriggers ( QAbstractItemView : : NoEditTriggers ) ;
ui - > peerWidget - > setSelectionBehavior ( QAbstractItemView : : SelectRows ) ;
2016-10-03 20:40:40 -03:00
ui - > peerWidget - > setSelectionMode ( QAbstractItemView : : ExtendedSelection ) ;
2015-06-01 10:32:25 -03:00
ui - > peerWidget - > setContextMenuPolicy ( Qt : : CustomContextMenu ) ;
2014-05-23 13:09:59 -04:00
ui - > peerWidget - > setColumnWidth ( PeerTableModel : : Address , ADDRESS_COLUMN_WIDTH ) ;
2014-06-04 06:06:18 -04:00
ui - > peerWidget - > setColumnWidth ( PeerTableModel : : Subversion , SUBVERSION_COLUMN_WIDTH ) ;
ui - > peerWidget - > setColumnWidth ( PeerTableModel : : Ping , PING_COLUMN_WIDTH ) ;
2015-06-20 16:48:10 -03:00
ui - > peerWidget - > horizontalHeader ( ) - > setStretchLastSection ( true ) ;
2015-06-20 15:27:03 -03:00
2015-06-26 09:55:52 -03:00
// create peer table context menu actions
2016-10-03 20:40:40 -03:00
QAction * disconnectAction = new QAction ( tr ( " &Disconnect " ) , this ) ;
QAction * banAction1h = new QAction ( tr ( " Ban for " ) + " " + tr ( " 1 &hour " ) , this ) ;
QAction * banAction24h = new QAction ( tr ( " Ban for " ) + " " + tr ( " 1 &day " ) , this ) ;
QAction * banAction7d = new QAction ( tr ( " Ban for " ) + " " + tr ( " 1 &week " ) , this ) ;
QAction * banAction365d = new QAction ( tr ( " Ban for " ) + " " + tr ( " 1 &year " ) , this ) ;
2015-06-26 09:55:52 -03:00
// create peer table context menu
2016-11-18 11:47:20 -03:00
peersTableContextMenu = new QMenu ( this ) ;
2015-06-20 15:55:21 -03:00
peersTableContextMenu - > addAction ( disconnectAction ) ;
peersTableContextMenu - > addAction ( banAction1h ) ;
peersTableContextMenu - > addAction ( banAction24h ) ;
peersTableContextMenu - > addAction ( banAction7d ) ;
peersTableContextMenu - > addAction ( banAction365d ) ;
2015-06-01 10:32:25 -03:00
2019-08-24 03:21:04 -04:00
connect ( banAction1h , & QAction : : triggered , [ this ] { banSelectedNode ( 60 * 60 ) ; } ) ;
connect ( banAction24h , & QAction : : triggered , [ this ] { banSelectedNode ( 60 * 60 * 24 ) ; } ) ;
connect ( banAction7d , & QAction : : triggered , [ this ] { banSelectedNode ( 60 * 60 * 24 * 7 ) ; } ) ;
connect ( banAction365d , & QAction : : triggered , [ this ] { banSelectedNode ( 60 * 60 * 24 * 365 ) ; } ) ;
2015-06-19 08:24:34 -03:00
2015-06-26 09:55:52 -03:00
// peer table context menu signals
2018-06-24 11:18:22 -04:00
connect ( ui - > peerWidget , & QTableView : : customContextMenuRequested , this , & RPCConsole : : showPeersTableContextMenu ) ;
connect ( disconnectAction , & QAction : : triggered , this , & RPCConsole : : disconnectSelectedNode ) ;
2015-06-26 09:55:52 -03:00
// peer table signal handling - update peer details when selecting new node
2018-06-24 11:18:22 -04:00
connect ( ui - > peerWidget - > selectionModel ( ) , & QItemSelectionModel : : selectionChanged , this , & RPCConsole : : peerSelected ) ;
2015-06-26 09:55:52 -03:00
// peer table signal handling - update peer details when new nodes are added to the model
2018-06-24 11:18:22 -04:00
connect ( model - > getPeerTableModel ( ) , & PeerTableModel : : layoutChanged , this , & RPCConsole : : peerLayoutChanged ) ;
2016-11-08 12:41:23 -03:00
// peer table signal handling - cache selected node ids
2018-06-24 11:18:22 -04:00
connect ( model - > getPeerTableModel ( ) , & PeerTableModel : : layoutAboutToBeChanged , this , & RPCConsole : : peerLayoutAboutToChange ) ;
2017-06-12 15:23:02 -04:00
2015-06-20 15:55:21 -03:00
// set up ban table
ui - > banlistWidget - > setModel ( model - > getBanTableModel ( ) ) ;
ui - > banlistWidget - > verticalHeader ( ) - > hide ( ) ;
ui - > banlistWidget - > setEditTriggers ( QAbstractItemView : : NoEditTriggers ) ;
ui - > banlistWidget - > setSelectionBehavior ( QAbstractItemView : : SelectRows ) ;
ui - > banlistWidget - > setSelectionMode ( QAbstractItemView : : SingleSelection ) ;
ui - > banlistWidget - > setContextMenuPolicy ( Qt : : CustomContextMenu ) ;
2015-06-26 09:55:52 -03:00
ui - > banlistWidget - > setColumnWidth ( BanTableModel : : Address , BANSUBNET_COLUMN_WIDTH ) ;
ui - > banlistWidget - > setColumnWidth ( BanTableModel : : Bantime , BANTIME_COLUMN_WIDTH ) ;
2015-06-20 16:48:10 -03:00
ui - > banlistWidget - > horizontalHeader ( ) - > setStretchLastSection ( true ) ;
2015-06-20 15:55:21 -03:00
2015-06-26 05:23:51 -03:00
// create ban table context menu action
2016-10-03 20:40:40 -03:00
QAction * unbanAction = new QAction ( tr ( " &Unban " ) , this ) ;
2015-06-26 05:23:51 -03:00
// create ban table context menu
2016-11-18 11:47:20 -03:00
banTableContextMenu = new QMenu ( this ) ;
2015-06-20 15:55:21 -03:00
banTableContextMenu - > addAction ( unbanAction ) ;
2015-06-26 05:23:51 -03:00
// ban table context menu signals
2018-06-24 11:18:22 -04:00
connect ( ui - > banlistWidget , & QTableView : : customContextMenuRequested , this , & RPCConsole : : showBanTableContextMenu ) ;
connect ( unbanAction , & QAction : : triggered , this , & RPCConsole : : unbanSelectedNode ) ;
2015-06-20 15:55:21 -03:00
2015-06-26 05:23:51 -03:00
// ban table signal handling - clear peer details when clicking a peer in the ban table
2018-06-24 11:18:22 -04:00
connect ( ui - > banlistWidget , & QTableView : : clicked , this , & RPCConsole : : clearSelectedNode ) ;
2015-06-26 05:23:51 -03:00
// ban table signal handling - ensure ban table is shown or hidden (if empty)
2018-06-24 11:18:22 -04:00
connect ( model - > getBanTableModel ( ) , & BanTableModel : : layoutChanged , this , & RPCConsole : : showOrHideBanTableIfRequired ) ;
2015-06-26 05:23:51 -03:00
showOrHideBanTableIfRequired ( ) ;
2012-04-09 16:07:25 -03:00
// Provide initial values
ui - > clientVersion - > setText ( model - > formatFullVersion ( ) ) ;
2015-08-06 10:40:50 -03:00
ui - > clientUserAgent - > setText ( model - > formatSubVersion ( ) ) ;
2016-03-22 04:40:10 -03:00
ui - > dataDir - > setText ( model - > dataDir ( ) ) ;
2018-10-02 17:12:17 -03:00
ui - > blocksDir - > setText ( model - > blocksDir ( ) ) ;
2012-05-21 17:05:54 -04:00
ui - > startupTime - > setText ( model - > formatClientStartupTime ( ) ) ;
2014-06-11 06:23:49 -04:00
ui - > networkName - > setText ( QString : : fromStdString ( Params ( ) . NetworkIDString ( ) ) ) ;
2016-02-27 00:57:12 -03:00
//Setup autocomplete and attach it
QStringList wordList ;
2017-04-17 17:38:51 -03:00
std : : vector < std : : string > commandList = m_node . listRpcCommands ( ) ;
2016-02-27 00:57:12 -03:00
for ( size_t i = 0 ; i < commandList . size ( ) ; + + i )
{
wordList < < commandList [ i ] . c_str ( ) ;
2017-01-09 22:53:00 -03:00
wordList < < ( " help " + commandList [ i ] ) . c_str ( ) ;
2016-02-27 00:57:12 -03:00
}
2017-11-18 01:43:17 -03:00
wordList < < " help-console " ;
2017-01-09 22:53:00 -03:00
wordList . sort ( ) ;
2016-02-27 00:57:12 -03:00
autoCompleter = new QCompleter ( wordList , this ) ;
2017-01-09 22:53:00 -03:00
autoCompleter - > setModelSorting ( QCompleter : : CaseSensitivelySortedModel ) ;
2019-05-29 17:53:25 -04:00
// ui->lineEdit is initially disabled because running commands is only
// possible from now on.
ui - > lineEdit - > setEnabled ( true ) ;
2016-02-27 00:57:12 -03:00
ui - > lineEdit - > setCompleter ( autoCompleter ) ;
2016-05-30 22:30:35 -04:00
autoCompleter - > popup ( ) - > installEventFilter ( this ) ;
2016-11-18 12:35:14 -03:00
// Start thread to execute RPC commands.
startExecutor ( ) ;
}
if ( ! model ) {
// Client model is being set to 0, this means shutdown() is about to be called.
2018-10-21 09:10:16 -03:00
thread . quit ( ) ;
2016-11-18 12:35:14 -03:00
thread . wait ( ) ;
2012-04-09 16:07:25 -03:00
}
}
2016-09-09 17:55:59 -03:00
# ifdef ENABLE_WALLET
2017-10-12 04:22:48 -03:00
void RPCConsole : : addWallet ( WalletModel * const walletModel )
2016-09-09 17:55:59 -03:00
{
2019-01-04 09:17:53 -03:00
// use name for text and wallet model for internal data object (to allow to move to a wallet id later)
ui - > WalletSelector - > addItem ( walletModel - > getDisplayName ( ) , QVariant : : fromValue ( walletModel ) ) ;
2016-09-09 17:55:59 -03:00
if ( ui - > WalletSelector - > count ( ) = = 2 & & ! isVisible ( ) ) {
// First wallet added, set to default so long as the window isn't presently visible (and potentially in use)
ui - > WalletSelector - > setCurrentIndex ( 1 ) ;
}
2018-03-18 05:35:51 -03:00
if ( ui - > WalletSelector - > count ( ) > 2 ) {
ui - > WalletSelector - > setVisible ( true ) ;
ui - > WalletSelectorLabel - > setVisible ( true ) ;
}
2016-09-09 17:55:59 -03:00
}
2018-06-05 06:17:28 -04:00
void RPCConsole : : removeWallet ( WalletModel * const walletModel )
{
2019-01-04 09:17:53 -03:00
ui - > WalletSelector - > removeItem ( ui - > WalletSelector - > findData ( QVariant : : fromValue ( walletModel ) ) ) ;
2018-06-05 06:17:28 -04:00
if ( ui - > WalletSelector - > count ( ) = = 2 ) {
ui - > WalletSelector - > setVisible ( false ) ;
ui - > WalletSelectorLabel - > setVisible ( false ) ;
}
}
2016-09-09 17:55:59 -03:00
# endif
2012-05-12 06:30:07 -04:00
static QString categoryClass ( int category )
2012-04-09 16:07:25 -03:00
{
switch ( category )
{
2012-05-12 06:30:07 -04:00
case RPCConsole : : CMD_REQUEST : return " cmd-request " ; break ;
case RPCConsole : : CMD_REPLY : return " cmd-reply " ; break ;
case RPCConsole : : CMD_ERROR : return " cmd-error " ; break ;
default : return " misc " ;
2012-04-09 16:07:25 -03:00
}
}
2016-01-22 14:00:36 -03:00
void RPCConsole : : fontBigger ( )
{
setFontSize ( consoleFontSize + 1 ) ;
}
void RPCConsole : : fontSmaller ( )
{
setFontSize ( consoleFontSize - 1 ) ;
}
void RPCConsole : : setFontSize ( int newSize )
{
QSettings settings ;
2017-06-19 18:57:31 -04:00
//don't allow an insane font size
2016-01-22 14:00:36 -03:00
if ( newSize < FONT_RANGE . width ( ) | | newSize > FONT_RANGE . height ( ) )
return ;
// temp. store the console content
QString str = ui - > messagesWidget - > toHtml ( ) ;
// replace font tags size in current content
str . replace ( QString ( " font-size:%1pt " ) . arg ( consoleFontSize ) , QString ( " font-size:%1pt " ) . arg ( newSize ) ) ;
// store the new font size
consoleFontSize = newSize ;
settings . setValue ( fontSizeSettingsKey , consoleFontSize ) ;
// clear console (reset icon sizes, default stylesheet) and re-add the content
2016-01-22 20:05:14 -03:00
float oldPosFactor = 1.0 / ui - > messagesWidget - > verticalScrollBar ( ) - > maximum ( ) * ui - > messagesWidget - > verticalScrollBar ( ) - > value ( ) ;
2016-03-11 13:57:10 -03:00
clear ( false ) ;
2016-01-22 14:00:36 -03:00
ui - > messagesWidget - > setHtml ( str ) ;
2016-01-22 20:05:14 -03:00
ui - > messagesWidget - > verticalScrollBar ( ) - > setValue ( oldPosFactor * ui - > messagesWidget - > verticalScrollBar ( ) - > maximum ( ) ) ;
2016-01-22 14:00:36 -03:00
}
2016-03-11 13:57:10 -03:00
void RPCConsole : : clear ( bool clearHistory )
2012-04-09 16:07:25 -03:00
{
ui - > messagesWidget - > clear ( ) ;
2016-03-11 13:57:10 -03:00
if ( clearHistory )
{
history . clear ( ) ;
historyPtr = 0 ;
}
2012-04-09 16:07:25 -03:00
ui - > lineEdit - > clear ( ) ;
ui - > lineEdit - > setFocus ( ) ;
2012-05-12 06:30:07 -04:00
// Add smoothly scaled icon images.
// (when using width/height on an img, Qt uses nearest instead of linear interpolation)
for ( int i = 0 ; ICON_MAPPING [ i ] . url ; + + i )
{
ui - > messagesWidget - > document ( ) - > addResource (
QTextDocument : : ImageResource ,
QUrl ( ICON_MAPPING [ i ] . url ) ,
2016-01-22 14:00:36 -03:00
platformStyle - > SingleColorImage ( ICON_MAPPING [ i ] . source ) . scaled ( QSize ( consoleFontSize * 2 , consoleFontSize * 2 ) , Qt : : IgnoreAspectRatio , Qt : : SmoothTransformation ) ) ;
2012-05-12 06:30:07 -04:00
}
// Set default style sheet
2015-10-22 08:33:58 -03:00
QFontInfo fixedFontInfo ( GUIUtil : : fixedPitchFont ( ) ) ;
2012-05-12 06:30:07 -04:00
ui - > messagesWidget - > document ( ) - > setDefaultStyleSheet (
2015-10-22 08:33:58 -03:00
QString (
2012-05-12 06:30:07 -04:00
" table { } "
2016-01-22 14:00:36 -03:00
" td.time { color: #808080; font-size: %2; padding-top: 3px; } "
2015-10-22 08:33:58 -03:00
" td.message { font-family: %1; font-size: %2; white-space:pre-wrap; } "
2012-05-12 06:30:07 -04:00
" td.cmd-request { color: #006060; } "
" td.cmd-error { color: red; } "
2016-12-12 11:57:45 -03:00
" .secwarning { color: red; } "
2012-05-12 06:30:07 -04:00
" b { color: #006060; } "
2016-01-22 14:00:36 -03:00
) . arg ( fixedFontInfo . family ( ) , QString ( " %1pt " ) . arg ( consoleFontSize ) )
2015-10-22 08:33:58 -03:00
) ;
2012-05-12 06:30:07 -04:00
2017-05-08 16:39:34 -03:00
# ifdef Q_OS_MAC
QString clsKey = " (⌘)-L " ;
# else
QString clsKey = " Ctrl-L " ;
# endif
2017-06-12 15:23:02 -04:00
2019-06-26 10:28:13 -04:00
message ( CMD_REPLY , ( tr ( " Welcome to the %1 RPC console. " ) . arg ( PACKAGE_NAME ) + " <br> " +
2017-05-08 16:39:34 -03:00
tr ( " Use up and down arrows to navigate history, and %1 to clear screen. " ) . arg ( " <b> " + clsKey + " </b> " ) + " <br> " +
2017-11-18 01:43:17 -03:00
tr ( " Type %1 for an overview of available commands. " ) . arg ( " <b>help</b> " ) + " <br> " +
tr ( " For more information on using this console type %1. " ) . arg ( " <b>help-console</b> " ) +
" <br><span class= \" secwarning \" ><br> " +
2017-06-19 18:57:31 -04:00
tr ( " WARNING: Scammers have been active, telling users to type commands here, stealing their wallet contents. Do not use this console without fully understanding the ramifications of a command. " ) +
2017-11-18 01:43:17 -03:00
" </span> " ) ,
2016-12-12 11:57:45 -03:00
true ) ;
2012-04-09 16:07:25 -03:00
}
2014-11-10 12:41:57 -03:00
void RPCConsole : : keyPressEvent ( QKeyEvent * event )
2014-03-18 10:51:28 -03:00
{
2014-11-10 12:41:57 -03:00
if ( windowType ( ) ! = Qt : : Widget & & event - > key ( ) = = Qt : : Key_Escape )
{
close ( ) ;
}
2014-03-18 10:51:28 -03:00
}
2012-05-12 06:30:07 -04:00
void RPCConsole : : message ( int category , const QString & message , bool html )
2012-04-09 16:07:25 -03:00
{
QTime time = QTime : : currentTime ( ) ;
2012-05-12 06:30:07 -04:00
QString timeString = time . toString ( ) ;
QString out ;
out + = " <table><tr><td class= \" time \" width= \" 65 \" > " + timeString + " </td> " ;
out + = " <td class= \" icon \" width= \" 32 \" ><img src= \" " + categoryClass ( category ) + " \" ></td> " ;
out + = " <td class= \" message " + categoryClass ( category ) + " \" valign= \" middle \" > " ;
if ( html )
out + = message ;
else
2015-10-22 08:33:58 -03:00
out + = GUIUtil : : HtmlEscape ( message , false ) ;
2012-05-12 06:30:07 -04:00
out + = " </td></tr></table> " ;
ui - > messagesWidget - > append ( out ) ;
2012-04-09 16:07:25 -03:00
}
2013-03-25 23:07:06 -03:00
void RPCConsole : : updateNetworkState ( )
{
QString connections = QString : : number ( clientModel - > getNumConnections ( ) ) + " ( " ;
connections + = tr ( " In: " ) + " " + QString : : number ( clientModel - > getNumConnections ( CONNECTIONS_IN ) ) + " / " ;
connections + = tr ( " Out: " ) + " " + QString : : number ( clientModel - > getNumConnections ( CONNECTIONS_OUT ) ) + " ) " ;
2017-04-17 16:37:36 -03:00
if ( ! clientModel - > node ( ) . getNetworkActive ( ) ) {
2013-03-25 23:07:06 -03:00
connections + = " ( " + tr ( " Network activity disabled " ) + " ) " ;
}
ui - > numberOfConnections - > setText ( connections ) ;
}
2012-04-09 16:07:25 -03:00
void RPCConsole : : setNumConnections ( int count )
{
2014-02-16 15:48:27 -03:00
if ( ! clientModel )
return ;
2013-03-25 23:07:06 -03:00
updateNetworkState ( ) ;
}
2014-02-16 15:48:27 -03:00
2013-03-25 23:07:06 -03:00
void RPCConsole : : setNetworkActive ( bool networkActive )
{
updateNetworkState ( ) ;
2012-04-09 16:07:25 -03:00
}
2016-04-28 11:18:45 -03:00
void RPCConsole : : setNumBlocks ( int count , const QDateTime & blockDate , double nVerificationProgress , bool headers )
2012-04-09 16:07:25 -03:00
{
2016-04-28 11:18:45 -03:00
if ( ! headers ) {
ui - > numberOfBlocks - > setText ( QString : : number ( count ) ) ;
ui - > lastBlockTime - > setText ( blockDate . toString ( ) ) ;
}
2012-04-09 16:07:25 -03:00
}
2015-11-09 07:45:07 -03:00
void RPCConsole : : setMempoolSize ( long numberOfTxs , size_t dynUsage )
{
ui - > mempoolNumberTxs - > setText ( QString : : number ( numberOfTxs ) ) ;
if ( dynUsage < 1000000 )
ui - > mempoolSize - > setText ( QString : : number ( dynUsage / 1000.0 , ' f ' , 2 ) + " KB " ) ;
else
ui - > mempoolSize - > setText ( QString : : number ( dynUsage / 1000000.0 , ' f ' , 2 ) + " MB " ) ;
}
2012-04-09 16:07:25 -03:00
void RPCConsole : : on_lineEdit_returnPressed ( )
{
QString cmd = ui - > lineEdit - > text ( ) ;
if ( ! cmd . isEmpty ( ) )
{
2016-11-16 09:32:15 -03:00
std : : string strFilteredCmd ;
try {
std : : string dummy ;
2017-04-17 17:38:51 -03:00
if ( ! RPCParseCommandLine ( nullptr , dummy , cmd . toStdString ( ) , false , & strFilteredCmd ) ) {
2016-11-16 09:32:15 -03:00
// Failed to parse command, so we cannot even filter it for the history
throw std : : runtime_error ( " Invalid command line " ) ;
}
} catch ( const std : : exception & e ) {
QMessageBox : : critical ( this , " Error " , QString ( " Error: " ) + QString : : fromStdString ( e . what ( ) ) ) ;
return ;
}
ui - > lineEdit - > clear ( ) ;
2015-03-13 17:51:27 -03:00
cmdBeforeBrowsing = QString ( ) ;
2016-09-09 17:55:59 -03:00
# ifdef ENABLE_WALLET
2019-01-07 13:46:58 -03:00
WalletModel * wallet_model = ui - > WalletSelector - > currentData ( ) . value < WalletModel * > ( ) ;
2017-12-14 00:29:55 -03:00
2019-01-04 09:17:53 -03:00
if ( m_last_wallet_model ! = wallet_model ) {
if ( wallet_model ) {
message ( CMD_REQUEST , tr ( " Executing command using \" %1 \" wallet " ) . arg ( wallet_model - > getWalletName ( ) ) ) ;
2017-12-14 00:29:55 -03:00
} else {
2019-01-04 09:17:53 -03:00
message ( CMD_REQUEST , tr ( " Executing command without any wallet " ) ) ;
2017-12-14 00:29:55 -03:00
}
2019-01-04 09:17:53 -03:00
m_last_wallet_model = wallet_model ;
2017-12-14 00:29:55 -03:00
}
2016-09-09 17:55:59 -03:00
# endif
2017-03-27 05:19:55 -03:00
message ( CMD_REQUEST , QString : : fromStdString ( strFilteredCmd ) ) ;
2019-01-04 09:17:53 -03:00
Q_EMIT cmdRequest ( cmd , m_last_wallet_model ) ;
2015-03-13 17:51:27 -03:00
2016-11-16 09:32:15 -03:00
cmd = QString : : fromStdString ( strFilteredCmd ) ;
2016-10-04 12:43:10 -03:00
// Remove command, if already in history
history . removeOne ( cmd ) ;
// Append command to history
history . append ( cmd ) ;
// Enforce maximum history size
while ( history . size ( ) > CONSOLE_HISTORY )
history . removeFirst ( ) ;
// Set pointer to end of history
historyPtr = history . size ( ) ;
2015-03-13 17:51:27 -03:00
2012-05-14 12:17:12 -04:00
// Scroll console view to end
scrollToEnd ( ) ;
2012-04-09 16:07:25 -03:00
}
}
void RPCConsole : : browseHistory ( int offset )
{
2015-03-13 17:51:27 -03:00
// store current text when start browsing through the history
if ( historyPtr = = history . size ( ) ) {
cmdBeforeBrowsing = ui - > lineEdit - > text ( ) ;
}
2012-04-09 16:07:25 -03:00
historyPtr + = offset ;
if ( historyPtr < 0 )
historyPtr = 0 ;
if ( historyPtr > history . size ( ) )
historyPtr = history . size ( ) ;
QString cmd ;
if ( historyPtr < history . size ( ) )
cmd = history . at ( historyPtr ) ;
2015-03-13 17:51:27 -03:00
else if ( ! cmdBeforeBrowsing . isNull ( ) ) {
cmd = cmdBeforeBrowsing ;
}
2012-04-09 16:07:25 -03:00
ui - > lineEdit - > setText ( cmd ) ;
}
void RPCConsole : : startExecutor ( )
{
2017-04-17 17:38:51 -03:00
RPCExecutor * executor = new RPCExecutor ( m_node ) ;
2016-11-18 12:35:14 -03:00
executor - > moveToThread ( & thread ) ;
2012-04-09 16:07:25 -03:00
// Replies from executor object must go to this object
2018-06-24 11:18:22 -04:00
connect ( executor , & RPCExecutor : : reply , this , static_cast < void ( RPCConsole : : * ) ( int , const QString & ) > ( & RPCConsole : : message ) ) ;
2012-04-09 16:07:25 -03:00
// Requests from this object must go to executor
2018-06-24 11:18:22 -04:00
connect ( this , & RPCConsole : : cmdRequest , executor , & RPCExecutor : : request ) ;
2013-04-02 06:24:10 -03:00
2018-10-21 09:10:16 -03:00
// Make sure executor object is deleted in its own thread
connect ( & thread , & QThread : : finished , executor , & RPCExecutor : : deleteLater ) ;
2012-04-09 16:07:25 -03:00
// Default implementation of QThread::run() simply spins up an event loop in the thread,
// which is what we want.
2016-11-18 12:35:14 -03:00
thread . start ( ) ;
2012-04-09 16:07:25 -03:00
}
2012-05-09 11:12:05 -04:00
void RPCConsole : : on_tabWidget_currentChanged ( int index )
{
2019-01-09 17:30:25 -03:00
if ( ui - > tabWidget - > widget ( index ) = = ui - > tab_console ) {
2012-05-09 11:12:05 -04:00
ui - > lineEdit - > setFocus ( ) ;
2019-01-09 17:30:25 -03:00
}
2012-05-09 11:12:05 -04:00
}
2012-05-09 16:07:00 -04:00
void RPCConsole : : on_openDebugLogfileButton_clicked ( )
{
GUIUtil : : openDebugLogfile ( ) ;
}
2012-05-14 12:17:12 -04:00
void RPCConsole : : scrollToEnd ( )
{
QScrollBar * scrollbar = ui - > messagesWidget - > verticalScrollBar ( ) ;
scrollbar - > setValue ( scrollbar - > maximum ( ) ) ;
}
2012-05-20 09:49:17 -04:00
2013-08-22 12:09:32 -04:00
void RPCConsole : : on_sldGraphRange_valueChanged ( int value )
{
const int multiplier = 5 ; // each position on the slider represents 5 min
int mins = value * multiplier ;
setTrafficGraphRange ( mins ) ;
}
void RPCConsole : : setTrafficGraphRange ( int mins )
{
ui - > trafficGraph - > setGraphRangeMins ( mins ) ;
2014-05-23 13:09:59 -04:00
ui - > lblGraphRange - > setText ( GUIUtil : : formatDurationStr ( mins * 60 ) ) ;
2013-08-22 12:09:32 -04:00
}
void RPCConsole : : updateTrafficStats ( quint64 totalBytesIn , quint64 totalBytesOut )
{
2017-10-14 20:06:21 -03:00
ui - > lblBytesIn - > setText ( GUIUtil : : formatBytes ( totalBytesIn ) ) ;
ui - > lblBytesOut - > setText ( GUIUtil : : formatBytes ( totalBytesOut ) ) ;
2013-08-22 12:09:32 -04:00
}
2014-05-23 13:09:59 -04:00
void RPCConsole : : peerSelected ( const QItemSelection & selected , const QItemSelection & deselected )
{
2014-06-03 08:42:20 -04:00
Q_UNUSED ( deselected ) ;
2015-06-26 09:55:52 -03:00
if ( ! clientModel | | ! clientModel - > getPeerTableModel ( ) | | selected . indexes ( ) . isEmpty ( ) )
2014-05-23 13:09:59 -04:00
return ;
const CNodeCombinedStats * stats = clientModel - > getPeerTableModel ( ) - > getNodeStats ( selected . indexes ( ) . first ( ) . row ( ) ) ;
if ( stats )
updateNodeDetail ( stats ) ;
}
2016-11-08 12:41:23 -03:00
void RPCConsole : : peerLayoutAboutToChange ( )
{
QModelIndexList selected = ui - > peerWidget - > selectionModel ( ) - > selectedIndexes ( ) ;
cachedNodeids . clear ( ) ;
for ( int i = 0 ; i < selected . size ( ) ; i + + )
{
const CNodeCombinedStats * stats = clientModel - > getPeerTableModel ( ) - > getNodeStats ( selected . at ( i ) . row ( ) ) ;
cachedNodeids . append ( stats - > nodeStats . nodeid ) ;
}
}
2014-05-23 13:09:59 -04:00
void RPCConsole : : peerLayoutChanged ( )
{
2015-06-26 09:55:52 -03:00
if ( ! clientModel | | ! clientModel - > getPeerTableModel ( ) )
2014-06-04 06:06:18 -04:00
return ;
2017-08-07 01:36:37 -04:00
const CNodeCombinedStats * stats = nullptr ;
2014-06-04 06:06:18 -04:00
bool fUnselect = false ;
bool fReselect = false ;
2014-05-23 13:09:59 -04:00
2016-11-08 12:41:23 -03:00
if ( cachedNodeids . empty ( ) ) // no node selected yet
2014-05-23 13:09:59 -04:00
return ;
// find the currently selected row
2015-06-06 05:38:15 -03:00
int selectedRow = - 1 ;
2014-05-23 13:09:59 -04:00
QModelIndexList selectedModelIndex = ui - > peerWidget - > selectionModel ( ) - > selectedIndexes ( ) ;
2015-06-06 05:38:15 -03:00
if ( ! selectedModelIndex . isEmpty ( ) ) {
2014-05-23 13:09:59 -04:00
selectedRow = selectedModelIndex . first ( ) . row ( ) ;
2015-06-06 05:38:15 -03:00
}
2014-05-23 13:09:59 -04:00
// check if our detail node has a row in the table (it may not necessarily
// be at selectedRow since its position can change after a layout change)
2016-11-08 12:41:23 -03:00
int detailNodeRow = clientModel - > getPeerTableModel ( ) - > getRowByNodeId ( cachedNodeids . first ( ) ) ;
2014-05-23 13:09:59 -04:00
if ( detailNodeRow < 0 )
{
2015-08-08 20:17:27 -03:00
// detail node disappeared from table (node disconnected)
2014-05-23 13:09:59 -04:00
fUnselect = true ;
}
else
{
if ( detailNodeRow ! = selectedRow )
{
// detail node moved position
fUnselect = true ;
fReselect = true ;
}
// get fresh stats on the detail node.
stats = clientModel - > getPeerTableModel ( ) - > getNodeStats ( detailNodeRow ) ;
}
2015-06-06 05:38:15 -03:00
if ( fUnselect & & selectedRow > = 0 ) {
clearSelectedNode ( ) ;
2014-05-23 13:09:59 -04:00
}
if ( fReselect )
{
2016-11-08 12:41:23 -03:00
for ( int i = 0 ; i < cachedNodeids . size ( ) ; i + + )
{
ui - > peerWidget - > selectRow ( clientModel - > getPeerTableModel ( ) - > getRowByNodeId ( cachedNodeids . at ( i ) ) ) ;
}
2014-05-23 13:09:59 -04:00
}
if ( stats )
updateNodeDetail ( stats ) ;
}
2014-06-04 06:06:18 -04:00
void RPCConsole : : updateNodeDetail ( const CNodeCombinedStats * stats )
2014-05-23 13:09:59 -04:00
{
// update the detail ui with latest node information
2015-06-01 04:09:51 -03:00
QString peerAddrDetails ( QString : : fromStdString ( stats - > nodeStats . addrName ) + " " ) ;
peerAddrDetails + = tr ( " (node id: %1) " ) . arg ( QString : : number ( stats - > nodeStats . nodeid ) ) ;
2014-06-04 06:06:18 -04:00
if ( ! stats - > nodeStats . addrLocal . empty ( ) )
peerAddrDetails + = " <br /> " + tr ( " via %1 " ) . arg ( QString : : fromStdString ( stats - > nodeStats . addrLocal ) ) ;
ui - > peerHeading - > setText ( peerAddrDetails ) ;
ui - > peerServices - > setText ( GUIUtil : : formatServicesStr ( stats - > nodeStats . nServices ) ) ;
2017-01-19 15:01:18 -03:00
ui - > peerLastSend - > setText ( stats - > nodeStats . nLastSend ? GUIUtil : : formatDurationStr ( GetSystemTimeInSeconds ( ) - stats - > nodeStats . nLastSend ) : tr ( " never " ) ) ;
ui - > peerLastRecv - > setText ( stats - > nodeStats . nLastRecv ? GUIUtil : : formatDurationStr ( GetSystemTimeInSeconds ( ) - stats - > nodeStats . nLastRecv ) : tr ( " never " ) ) ;
2017-10-14 20:06:21 -03:00
ui - > peerBytesSent - > setText ( GUIUtil : : formatBytes ( stats - > nodeStats . nSendBytes ) ) ;
ui - > peerBytesRecv - > setText ( GUIUtil : : formatBytes ( stats - > nodeStats . nRecvBytes ) ) ;
2017-01-19 15:01:18 -03:00
ui - > peerConnTime - > setText ( GUIUtil : : formatDurationStr ( GetSystemTimeInSeconds ( ) - stats - > nodeStats . nTimeConnected ) ) ;
2020-03-03 10:35:01 -03:00
ui - > peerPingTime - > setText ( GUIUtil : : formatPingTime ( stats - > nodeStats . m_ping_usec ) ) ;
2020-03-03 10:42:50 -03:00
ui - > peerPingWait - > setText ( GUIUtil : : formatPingTime ( stats - > nodeStats . m_ping_wait_usec ) ) ;
2020-03-03 10:40:29 -03:00
ui - > peerMinPing - > setText ( GUIUtil : : formatPingTime ( stats - > nodeStats . m_min_ping_usec ) ) ;
2014-12-15 07:07:55 -03:00
ui - > timeoffset - > setText ( GUIUtil : : formatTimeOffset ( stats - > nodeStats . nTimeOffset ) ) ;
2020-03-22 08:27:46 -03:00
ui - > peerVersion - > setText ( QString : : number ( stats - > nodeStats . nVersion ) ) ;
2014-06-04 06:06:18 -04:00
ui - > peerSubversion - > setText ( QString : : fromStdString ( stats - > nodeStats . cleanSubVer ) ) ;
ui - > peerDirection - > setText ( stats - > nodeStats . fInbound ? tr ( " Inbound " ) : tr ( " Outbound " ) ) ;
2020-03-22 08:27:46 -03:00
ui - > peerHeight - > setText ( QString : : number ( stats - > nodeStats . nStartingHeight ) ) ;
2019-06-20 22:42:04 -04:00
ui - > peerWhitelisted - > setText ( stats - > nodeStats . m_legacyWhitelisted ? tr ( " Yes " ) : tr ( " No " ) ) ;
2020-03-22 08:27:46 -03:00
ui - > peerMappedAS - > setText ( stats - > nodeStats . m_mapped_as ! = 0 ? QString : : number ( stats - > nodeStats . m_mapped_as ) : tr ( " N/A " ) ) ;
2014-06-04 06:06:18 -04:00
// This check fails for example if the lock was busy and
// nodeStateStats couldn't be fetched.
if ( stats - > fNodeStateStatsAvailable ) {
// Ban score is init to 0
ui - > peerBanScore - > setText ( QString ( " %1 " ) . arg ( stats - > nodeStateStats . nMisbehavior ) ) ;
// Sync height is init to -1
if ( stats - > nodeStateStats . nSyncHeight > - 1 )
ui - > peerSyncHeight - > setText ( QString ( " %1 " ) . arg ( stats - > nodeStateStats . nSyncHeight ) ) ;
else
ui - > peerSyncHeight - > setText ( tr ( " Unknown " ) ) ;
2015-06-01 04:09:51 -03:00
// Common height is init to -1
if ( stats - > nodeStateStats . nCommonHeight > - 1 )
ui - > peerCommonHeight - > setText ( QString ( " %1 " ) . arg ( stats - > nodeStateStats . nCommonHeight ) ) ;
else
ui - > peerCommonHeight - > setText ( tr ( " Unknown " ) ) ;
2014-06-04 06:06:18 -04:00
}
2014-07-13 00:27:29 -04:00
2014-06-04 06:06:18 -04:00
ui - > detailWidget - > show ( ) ;
2014-07-13 00:27:29 -04:00
}
2014-05-23 13:09:59 -04:00
void RPCConsole : : resizeEvent ( QResizeEvent * event )
{
QWidget : : resizeEvent ( event ) ;
}
void RPCConsole : : showEvent ( QShowEvent * event )
{
QWidget : : showEvent ( event ) ;
2015-06-26 09:55:52 -03:00
if ( ! clientModel | | ! clientModel - > getPeerTableModel ( ) )
2014-06-04 06:06:18 -04:00
return ;
// start PeerTableModel auto refresh
clientModel - > getPeerTableModel ( ) - > startAutoRefresh ( ) ;
2014-05-23 13:09:59 -04:00
}
void RPCConsole : : hideEvent ( QHideEvent * event )
{
QWidget : : hideEvent ( event ) ;
2015-06-26 09:55:52 -03:00
if ( ! clientModel | | ! clientModel - > getPeerTableModel ( ) )
2014-07-13 00:27:29 -04:00
return ;
2014-05-23 13:09:59 -04:00
// stop PeerTableModel auto refresh
clientModel - > getPeerTableModel ( ) - > stopAutoRefresh ( ) ;
}
2015-06-01 10:32:25 -03:00
2015-06-20 15:55:21 -03:00
void RPCConsole : : showPeersTableContextMenu ( const QPoint & point )
2015-06-01 10:32:25 -03:00
{
QModelIndex index = ui - > peerWidget - > indexAt ( point ) ;
if ( index . isValid ( ) )
2015-06-20 15:55:21 -03:00
peersTableContextMenu - > exec ( QCursor : : pos ( ) ) ;
}
void RPCConsole : : showBanTableContextMenu ( const QPoint & point )
{
QModelIndex index = ui - > banlistWidget - > indexAt ( point ) ;
if ( index . isValid ( ) )
banTableContextMenu - > exec ( QCursor : : pos ( ) ) ;
2015-06-01 10:32:25 -03:00
}
void RPCConsole : : disconnectSelectedNode ( )
{
2016-10-03 20:40:40 -03:00
// Get selected peer addresses
2016-12-02 10:48:33 -03:00
QList < QModelIndex > nodes = GUIUtil : : getEntryData ( ui - > peerWidget , PeerTableModel : : NetNodeId ) ;
2016-10-03 20:40:40 -03:00
for ( int i = 0 ; i < nodes . count ( ) ; i + + )
{
// Get currently selected peer address
2017-04-10 16:00:23 -03:00
NodeId id = nodes . at ( i ) . data ( ) . toLongLong ( ) ;
2016-10-03 20:40:40 -03:00
// Find the node, disconnect it and clear the selected node
2017-12-05 17:57:12 -03:00
if ( m_node . disconnectById ( id ) )
2016-10-03 20:40:40 -03:00
clearSelectedNode ( ) ;
}
2015-06-01 10:32:25 -03:00
}
2015-06-06 05:38:15 -03:00
2015-06-19 08:24:34 -03:00
void RPCConsole : : banSelectedNode ( int bantime )
{
2017-04-17 17:38:51 -03:00
if ( ! clientModel )
2015-06-26 05:23:51 -03:00
return ;
2017-06-12 15:23:02 -04:00
2016-10-03 20:40:40 -03:00
// Get selected peer addresses
2016-12-02 10:48:33 -03:00
QList < QModelIndex > nodes = GUIUtil : : getEntryData ( ui - > peerWidget , PeerTableModel : : NetNodeId ) ;
2016-10-03 20:40:40 -03:00
for ( int i = 0 ; i < nodes . count ( ) ; i + + )
{
// Get currently selected peer address
2017-04-10 16:00:23 -03:00
NodeId id = nodes . at ( i ) . data ( ) . toLongLong ( ) ;
2016-10-03 20:40:40 -03:00
2017-10-04 19:25:34 -03:00
// Get currently selected peer address
int detailNodeRow = clientModel - > getPeerTableModel ( ) - > getRowByNodeId ( id ) ;
if ( detailNodeRow < 0 ) return ;
// Find possible nodes, ban it and clear the selected node
const CNodeCombinedStats * stats = clientModel - > getPeerTableModel ( ) - > getNodeStats ( detailNodeRow ) ;
if ( stats ) {
2017-04-17 17:38:51 -03:00
m_node . ban ( stats - > nodeStats . addr , BanReasonManuallyAdded , bantime ) ;
2017-12-05 17:57:12 -03:00
m_node . disconnectByAddress ( stats - > nodeStats . addr ) ;
2017-10-04 19:25:34 -03:00
}
2016-10-04 20:27:11 -03:00
}
2016-10-03 20:40:40 -03:00
clearSelectedNode ( ) ;
clientModel - > getBanTableModel ( ) - > refresh ( ) ;
2015-06-19 08:24:34 -03:00
}
2015-06-20 15:55:21 -03:00
void RPCConsole : : unbanSelectedNode ( )
{
2015-06-26 05:23:51 -03:00
if ( ! clientModel )
return ;
2016-10-03 20:40:40 -03:00
// Get selected ban addresses
2016-12-02 10:48:33 -03:00
QList < QModelIndex > nodes = GUIUtil : : getEntryData ( ui - > banlistWidget , BanTableModel : : Address ) ;
2016-10-03 20:40:40 -03:00
for ( int i = 0 ; i < nodes . count ( ) ; i + + )
2015-06-20 15:55:21 -03:00
{
2016-10-03 20:40:40 -03:00
// Get currently selected ban address
2016-12-02 10:48:33 -03:00
QString strNode = nodes . at ( i ) . data ( ) . toString ( ) ;
2016-10-03 20:40:40 -03:00
CSubNet possibleSubnet ;
2019-12-11 13:39:29 -03:00
LookupSubNet ( strNode . toStdString ( ) , possibleSubnet ) ;
2017-04-17 17:38:51 -03:00
if ( possibleSubnet . IsValid ( ) & & m_node . unban ( possibleSubnet ) )
2016-10-03 20:40:40 -03:00
{
clientModel - > getBanTableModel ( ) - > refresh ( ) ;
}
2015-06-20 15:55:21 -03:00
}
}
2015-06-06 05:38:15 -03:00
void RPCConsole : : clearSelectedNode ( )
{
ui - > peerWidget - > selectionModel ( ) - > clearSelection ( ) ;
2016-11-08 12:41:23 -03:00
cachedNodeids . clear ( ) ;
2015-06-06 05:38:15 -03:00
ui - > detailWidget - > hide ( ) ;
ui - > peerHeading - > setText ( tr ( " Select a peer to view detailed information. " ) ) ;
}
2015-06-20 16:48:10 -03:00
void RPCConsole : : showOrHideBanTableIfRequired ( )
{
2015-06-21 05:44:48 -03:00
if ( ! clientModel )
return ;
2015-06-23 16:10:42 -03:00
2015-06-20 16:48:10 -03:00
bool visible = clientModel - > getBanTableModel ( ) - > shouldShow ( ) ;
ui - > banlistWidget - > setVisible ( visible ) ;
ui - > banHeading - > setVisible ( visible ) ;
2015-10-22 08:33:58 -03:00
}
2015-11-12 08:59:26 -03:00
void RPCConsole : : setTabFocus ( enum TabTypes tabType )
{
2019-10-10 21:50:49 -03:00
ui - > tabWidget - > setCurrentIndex ( int ( tabType ) ) ;
2015-11-12 08:59:26 -03:00
}
2018-12-11 10:13:48 -03:00
QString RPCConsole : : tabTitle ( TabTypes tab_type ) const
{
2019-10-10 21:50:49 -03:00
return ui - > tabWidget - > tabText ( int ( tab_type ) ) ;
2018-12-11 10:13:48 -03:00
}
2018-12-05 17:51:49 -03:00
2019-04-05 12:20:57 -03:00
QKeySequence RPCConsole : : tabShortcut ( TabTypes tab_type ) const
{
switch ( tab_type ) {
2019-10-10 21:50:49 -03:00
case TabTypes : : INFO : return QKeySequence ( Qt : : CTRL + Qt : : Key_I ) ;
case TabTypes : : CONSOLE : return QKeySequence ( Qt : : CTRL + Qt : : Key_T ) ;
case TabTypes : : GRAPH : return QKeySequence ( Qt : : CTRL + Qt : : Key_N ) ;
case TabTypes : : PEERS : return QKeySequence ( Qt : : CTRL + Qt : : Key_P ) ;
} // no default case, so the compiler can warn about missing cases
assert ( false ) ;
2019-04-05 12:20:57 -03:00
}
2018-12-05 17:51:49 -03:00
void RPCConsole : : updateAlerts ( const QString & warnings )
{
this - > ui - > label_alerts - > setVisible ( ! warnings . isEmpty ( ) ) ;
this - > ui - > label_alerts - > setText ( warnings ) ;
}