2017-06-02 14:30:36 -04:00
#!/usr/bin/env python3
# Copyright (c) 2017 The Bitcoin Core developers
# Distributed under the MIT software license, see the accompanying
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
""" Class for bitcoind node under test """
2017-07-11 13:01:44 -04:00
import decimal
2017-06-02 14:30:36 -04:00
import errno
import http . client
2017-07-11 13:01:44 -04:00
import json
2017-06-02 14:30:36 -04:00
import logging
import os
2017-12-20 18:38:40 -05:00
import re
2017-06-02 14:30:36 -04:00
import subprocess
import time
2017-03-27 09:42:17 -04:00
from . authproxy import JSONRPCException
2017-06-02 14:30:36 -04:00
from . util import (
assert_equal ,
2018-04-09 14:07:47 -04:00
delete_cookie_file ,
2017-06-02 14:30:36 -04:00
get_rpc_proxy ,
rpc_url ,
2017-08-16 08:52:24 -07:00
wait_until ,
2017-03-27 09:42:17 -04:00
p2p_port ,
2017-06-02 14:30:36 -04:00
)
2017-12-20 18:38:40 -05:00
# For Python 3.4 compatibility
JSONDecodeError = getattr ( json , " JSONDecodeError " , ValueError )
2019-02-09 14:52:22 -08:00
BITCOIND_PROC_WAIT_TIMEOUT = 180
2017-08-16 08:52:24 -07:00
2017-06-02 14:30:36 -04:00
class TestNode ( ) :
""" A class for representing a bitcoind node under test.
This class contains :
- state about the node ( whether it ' s running, etc)
- a Python subprocess . Popen object representing the running process
- an RPC connection to the node
2017-03-27 09:42:17 -04:00
- one or more P2P connections to the node
2017-06-02 14:30:36 -04:00
2017-03-27 09:42:17 -04:00
To make things easier for the test writer , any unrecognised messages will
be dispatched to the RPC connection . """
2017-06-02 14:30:36 -04:00
2017-07-11 13:14:18 -04:00
def __init__ ( self , i , dirname , extra_args , rpchost , timewait , binary , stderr , mocktime , coverage_dir , use_cli = False ) :
2017-06-02 14:30:36 -04:00
self . index = i
self . datadir = os . path . join ( dirname , " node " + str ( i ) )
self . rpchost = rpchost
2017-08-16 15:46:48 -04:00
if timewait :
self . rpc_timeout = timewait
else :
2019-02-09 14:52:22 -08:00
# Wait for up to 180 seconds for the RPC server to respond
self . rpc_timeout = 180
2017-06-02 14:30:36 -04:00
if binary is None :
2019-01-28 17:21:28 -08:00
self . binary = os . getenv ( " LITECOIND " , " kevacoind " )
2017-06-02 14:30:36 -04:00
else :
self . binary = binary
self . stderr = stderr
self . coverage_dir = coverage_dir
# Most callers will just need to add extra args to the standard list below. For those callers that need more flexibity, they can just set the args property directly.
self . extra_args = extra_args
self . args = [ self . binary , " -datadir= " + self . datadir , " -server " , " -keypool=1 " , " -discover=0 " , " -rest " , " -logtimemicros " , " -debug " , " -debugexclude=libevent " , " -debugexclude=leveldb " , " -mocktime= " + str ( mocktime ) , " -uacomment=testnode %d " % i ]
2019-01-28 17:21:28 -08:00
self . cli = TestNodeCLI ( os . getenv ( " LITECOINCLI " , " kevacoin-cli " ) , self . datadir )
2017-07-11 13:14:18 -04:00
self . use_cli = use_cli
2017-07-11 13:01:44 -04:00
2017-06-02 14:30:36 -04:00
self . running = False
self . process = None
self . rpc_connected = False
self . rpc = None
self . url = None
self . log = logging . getLogger ( ' TestFramework.node %d ' % i )
2018-04-06 10:53:35 -04:00
self . cleanup_on_exit = True # Whether to kill the node when this object goes away
2017-06-02 14:30:36 -04:00
2017-03-27 09:42:17 -04:00
self . p2ps = [ ]
2018-04-06 10:53:35 -04:00
def __del__ ( self ) :
# Ensure that we don't leave any bitcoind processes lying around after
# the test ends
if self . process and self . cleanup_on_exit :
# Should only happen on test failure
# Avoid using logger, as that may have already been shutdown when
# this destructor is called.
print ( " Cleaning up leftover process " )
self . process . kill ( )
2017-10-20 09:27:55 -04:00
def __getattr__ ( self , name ) :
2017-07-11 13:14:18 -04:00
""" Dispatches any unrecognised messages to the RPC connection or a CLI instance. """
if self . use_cli :
return getattr ( self . cli , name )
else :
assert self . rpc_connected and self . rpc is not None , " Error: no RPC connection "
return getattr ( self . rpc , name )
2017-06-02 14:30:36 -04:00
2018-01-18 13:15:00 -05:00
def start ( self , extra_args = None , stderr = None , * args , * * kwargs ) :
2017-06-02 14:30:36 -04:00
""" Start the node. """
2017-06-09 16:35:17 -04:00
if extra_args is None :
extra_args = self . extra_args
if stderr is None :
stderr = self . stderr
2018-04-09 14:07:47 -04:00
# Delete any existing cookie file -- if such a file exists (eg due to
# unclean shutdown), it will get overwritten anyway by bitcoind, and
# potentially interfere with our attempt to authenticate
delete_cookie_file ( self . datadir )
2018-01-18 13:15:00 -05:00
self . process = subprocess . Popen ( self . args + extra_args , stderr = stderr , * args , * * kwargs )
2017-06-02 14:30:36 -04:00
self . running = True
2019-01-28 17:21:28 -08:00
self . log . debug ( " kevacoind started, waiting for RPC to come up " )
2017-06-02 14:30:36 -04:00
def wait_for_rpc_connection ( self ) :
""" Sets up an RPC connection to the bitcoind process. Returns False if unable to connect. """
2017-08-16 15:46:48 -04:00
# Poll at a rate of four times per second
poll_per_s = 4
for _ in range ( poll_per_s * self . rpc_timeout ) :
2019-01-28 17:21:28 -08:00
assert self . process . poll ( ) is None , " kevacoind exited with status %i during initialization " % self . process . returncode
2017-06-02 14:30:36 -04:00
try :
2017-08-23 15:49:01 -04:00
self . rpc = get_rpc_proxy ( rpc_url ( self . datadir , self . index , self . rpchost ) , self . index , timeout = self . rpc_timeout , coveragedir = self . coverage_dir )
2017-06-02 14:30:36 -04:00
self . rpc . getblockcount ( )
# If the call to getblockcount() succeeds then the RPC connection is up
self . rpc_connected = True
self . url = self . rpc . url
self . log . debug ( " RPC successfully started " )
return
except IOError as e :
if e . errno != errno . ECONNREFUSED : # Port not yet open?
raise # unknown IO error
except JSONRPCException as e : # Initialization phase
if e . error [ ' code ' ] != - 28 : # RPC in warmup?
raise # unknown JSON RPC exception
except ValueError as e : # cookie file not found and no rpcuser or rpcassword. bitcoind still starting
if " No RPC credentials " not in str ( e ) :
raise
2017-08-18 22:09:58 +02:00
time . sleep ( 1.0 / poll_per_s )
2019-01-28 17:21:28 -08:00
raise AssertionError ( " Unable to connect to kevacoind " )
2017-06-02 14:30:36 -04:00
def get_wallet_rpc ( self , wallet_name ) :
2017-07-11 13:14:18 -04:00
if self . use_cli :
return self . cli ( " -rpcwallet= {} " . format ( wallet_name ) )
else :
assert self . rpc_connected
assert self . rpc
wallet_path = " wallet/ %s " % wallet_name
return self . rpc / wallet_path
2017-06-02 14:30:36 -04:00
def stop_node ( self ) :
""" Stop the node. """
if not self . running :
return
self . log . debug ( " Stopping node " )
try :
self . stop ( )
except http . client . CannotSendRequest :
self . log . exception ( " Unable to stop node. " )
2017-03-27 09:42:17 -04:00
del self . p2ps [ : ]
2017-06-02 14:30:36 -04:00
def is_node_stopped ( self ) :
""" Checks whether the node has stopped.
Returns True if the node has stopped . False otherwise .
This method is responsible for freeing resources ( self . process ) . """
if not self . running :
return True
return_code = self . process . poll ( )
2017-08-16 08:52:24 -07:00
if return_code is None :
return False
# process has stopped. Assert that it didn't return an error code.
assert_equal ( return_code , 0 )
self . running = False
self . process = None
self . rpc_connected = False
self . rpc = None
self . log . debug ( " Node stopped " )
return True
def wait_until_stopped ( self , timeout = BITCOIND_PROC_WAIT_TIMEOUT ) :
wait_until ( self . is_node_stopped , timeout = timeout )
2017-06-02 14:30:36 -04:00
def node_encrypt_wallet ( self , passphrase ) :
""" " Encrypts the wallet.
This causes bitcoind to shutdown , so this method takes
care of cleaning up resources . """
self . encryptwallet ( passphrase )
2017-08-16 08:52:24 -07:00
self . wait_until_stopped ( )
2017-07-11 13:01:44 -04:00
2017-11-17 15:01:24 -05:00
def add_p2p_connection ( self , p2p_conn , * args , * * kwargs ) :
2017-03-27 09:42:17 -04:00
""" Add a p2p connection to the node.
This method adds the p2p connection to the self . p2ps list and also
returns the connection to the caller . """
if ' dstport ' not in kwargs :
kwargs [ ' dstport ' ] = p2p_port ( self . index )
if ' dstaddr ' not in kwargs :
kwargs [ ' dstaddr ' ] = ' 127.0.0.1 '
2017-11-17 15:01:24 -05:00
p2p_conn . peer_connect ( * args , * * kwargs )
2017-03-27 09:42:17 -04:00
self . p2ps . append ( p2p_conn )
return p2p_conn
@property
def p2p ( self ) :
""" Return the first p2p connection
Convenience property - most tests only use a single p2p connection to each
node , so this saves having to write node . p2ps [ 0 ] many times . """
assert self . p2ps , " No p2p connection "
return self . p2ps [ 0 ]
2017-11-08 16:28:17 -05:00
def disconnect_p2ps ( self ) :
""" Close all p2p connections to the node. """
for p in self . p2ps :
2017-11-17 15:01:24 -05:00
p . peer_disconnect ( )
del self . p2ps [ : ]
2017-11-08 16:28:17 -05:00
2017-12-21 04:54:43 -05:00
class TestNodeCLIAttr :
def __init__ ( self , cli , command ) :
self . cli = cli
self . command = command
def __call__ ( self , * args , * * kwargs ) :
return self . cli . send_cli ( self . command , * args , * * kwargs )
def get_request ( self , * args , * * kwargs ) :
return lambda : self ( * args , * * kwargs )
2017-03-27 09:42:17 -04:00
2017-07-11 13:01:44 -04:00
class TestNodeCLI ( ) :
""" Interface to bitcoin-cli for an individual node """
def __init__ ( self , binary , datadir ) :
2018-01-23 13:58:53 -05:00
self . options = [ ]
2017-07-11 13:01:44 -04:00
self . binary = binary
self . datadir = datadir
2017-09-06 17:07:21 +01:00
self . input = None
2017-07-11 13:14:18 -04:00
self . log = logging . getLogger ( ' TestFramework.bitcoincli ' )
2017-09-06 17:07:21 +01:00
2018-01-23 13:58:53 -05:00
def __call__ ( self , * options , input = None ) :
# TestNodeCLI is callable with bitcoin-cli command-line options
2017-12-20 18:41:12 -05:00
cli = TestNodeCLI ( self . binary , self . datadir )
2018-01-23 13:58:53 -05:00
cli . options = [ str ( o ) for o in options ]
2017-12-20 18:41:12 -05:00
cli . input = input
return cli
2017-07-11 13:01:44 -04:00
def __getattr__ ( self , command ) :
2017-12-21 04:54:43 -05:00
return TestNodeCLIAttr ( self , command )
def batch ( self , requests ) :
results = [ ]
for request in requests :
try :
results . append ( dict ( result = request ( ) ) )
except JSONRPCException as e :
results . append ( dict ( error = e ) )
return results
2017-07-11 13:01:44 -04:00
2018-01-23 14:00:34 -05:00
def send_cli ( self , command = None , * args , * * kwargs ) :
2017-07-11 13:01:44 -04:00
""" Run bitcoin-cli command. Deserializes returned string as python object. """
pos_args = [ str ( arg ) for arg in args ]
named_args = [ str ( key ) + " = " + str ( value ) for ( key , value ) in kwargs . items ( ) ]
assert not ( pos_args and named_args ) , " Cannot use positional arguments and named arguments in the same bitcoin-cli call "
2018-01-23 13:58:53 -05:00
p_args = [ self . binary , " -datadir= " + self . datadir ] + self . options
2017-07-11 13:01:44 -04:00
if named_args :
p_args + = [ " -named " ]
2018-01-23 14:00:34 -05:00
if command is not None :
p_args + = [ command ]
p_args + = pos_args + named_args
2019-01-28 17:21:28 -08:00
self . log . debug ( " Running kevacoin-cli command: %s " % command )
2017-09-06 16:36:13 +01:00
process = subprocess . Popen ( p_args , stdin = subprocess . PIPE , stdout = subprocess . PIPE , stderr = subprocess . PIPE , universal_newlines = True )
cli_stdout , cli_stderr = process . communicate ( input = self . input )
returncode = process . poll ( )
if returncode :
2017-12-20 18:38:40 -05:00
match = re . match ( r ' error code: ([-0-9]+) \ nerror message: \ n(.*) ' , cli_stderr )
if match :
code , message = match . groups ( )
raise JSONRPCException ( dict ( code = int ( code ) , message = message ) )
2017-09-06 16:36:13 +01:00
# Ignore cli_stdout, raise with cli_stderr
raise subprocess . CalledProcessError ( returncode , self . binary , output = cli_stderr )
2017-12-20 18:38:40 -05:00
try :
return json . loads ( cli_stdout , parse_float = decimal . Decimal )
except JSONDecodeError :
return cli_stdout . rstrip ( " \n " )