2016-03-19 16:58:06 -03:00
#!/usr/bin/env python3
2018-07-26 18:36:45 -04:00
# Copyright (c) 2014-2018 The Bitcoin Core developers
2014-10-22 22:48:19 -03:00
# Distributed under the MIT software license, see the accompanying
2014-07-08 12:07:23 -04:00
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
2017-01-17 20:34:40 -03:00
""" Base class for RPC testing. """
2014-07-08 12:07:23 -04:00
2018-04-19 09:38:59 -03:00
import configparser
2017-05-18 16:36:39 -04:00
from enum import Enum
2016-05-09 14:55:49 -03:00
import logging
2018-08-11 16:24:14 -04:00
import argparse
2014-07-08 12:07:23 -04:00
import os
2017-03-31 23:44:04 -03:00
import pdb
2014-07-08 12:07:23 -04:00
import shutil
2017-03-23 17:49:02 -03:00
import sys
2014-07-08 12:07:23 -04:00
import tempfile
2017-03-21 11:05:59 -03:00
import time
2014-07-08 12:07:23 -04:00
2017-06-02 13:14:14 -04:00
from . authproxy import JSONRPCException
from . import coverage
2017-06-02 14:30:36 -04:00
from . test_node import TestNode
2018-06-18 17:28:37 -04:00
from . mininode import NetworkThread
2015-10-11 02:41:19 -03:00
from . util import (
2017-03-23 17:49:02 -03:00
MAX_NODES ,
2017-06-02 13:14:14 -04:00
PortSeed ,
assert_equal ,
2017-03-23 17:49:02 -03:00
check_json_precision ,
2015-10-11 02:41:19 -03:00
connect_nodes_bi ,
2017-04-03 10:34:04 -03:00
disconnect_nodes ,
2018-03-07 15:49:44 -03:00
get_datadir_path ,
2017-03-23 17:49:02 -03:00
initialize_datadir ,
p2p_port ,
2015-10-11 02:41:19 -03:00
sync_blocks ,
sync_mempools ,
)
2014-07-08 12:07:23 -04:00
2019-02-11 13:18:36 -03:00
2017-05-18 16:36:39 -04:00
class TestStatus ( Enum ) :
PASSED = 1
FAILED = 2
SKIPPED = 3
TEST_EXIT_PASSED = 0
TEST_EXIT_FAILED = 1
TEST_EXIT_SKIPPED = 77
2018-11-07 14:07:25 -03:00
TMPDIR_PREFIX = " bitcoin_func_test_ "
2018-04-02 02:45:41 -03:00
2018-09-09 14:32:37 -03:00
class SkipTest ( Exception ) :
""" This exception is raised to skip a test """
def __init__ ( self , message ) :
self . message = message
2018-04-02 02:45:41 -03:00
class BitcoinTestMetaClass ( type ) :
""" Metaclass for BitcoinTestFramework.
Ensures that any attempt to register a subclass of ` BitcoinTestFramework `
adheres to a standard whereby the subclass overrides ` set_test_params ` and
` run_test ` but DOES NOT override either ` __init__ ` or ` main ` . If any of
those standards are violated , a ` ` TypeError ` ` is raised . """
def __new__ ( cls , clsname , bases , dct ) :
if not clsname == ' BitcoinTestFramework ' :
if not ( ' run_test ' in dct and ' set_test_params ' in dct ) :
raise TypeError ( " BitcoinTestFramework subclasses must override "
" ' run_test ' and ' set_test_params ' " )
if ' __init__ ' in dct or ' main ' in dct :
raise TypeError ( " BitcoinTestFramework subclasses may not override "
" ' __init__ ' or ' main ' " )
return super ( ) . __new__ ( cls , clsname , bases , dct )
class BitcoinTestFramework ( metaclass = BitcoinTestMetaClass ) :
2017-03-23 18:48:29 -03:00
""" Base class for a bitcoin test script.
2017-08-24 12:11:56 -03:00
Individual bitcoin test scripts should subclass this class and override the set_test_params ( ) and run_test ( ) methods .
2017-03-23 18:48:29 -03:00
2017-06-09 18:21:21 -04:00
Individual tests can also override the following methods to customize the test setup :
2017-03-23 18:48:29 -03:00
- add_options ( )
- setup_chain ( )
- setup_network ( )
2017-06-09 18:21:21 -04:00
- setup_nodes ( )
2017-03-23 18:48:29 -03:00
2017-06-09 18:21:21 -04:00
The __init__ ( ) and main ( ) methods should not be overridden .
2017-03-23 18:48:29 -03:00
This class also contains various public and private helper methods . """
2016-05-14 08:01:31 -03:00
def __init__ ( self ) :
2017-06-09 18:21:21 -04:00
""" Sets test framework defaults. Do not override this method. Instead, override the set_test_params() method """
2016-05-14 08:01:31 -03:00
self . setup_clean_chain = False
2017-06-02 13:14:14 -04:00
self . nodes = [ ]
2018-06-18 17:28:37 -04:00
self . network_thread = None
2018-12-22 13:22:55 -03:00
self . rpc_timeout = 60 # Wait for up to 60 seconds for the RPC server to respond
2017-07-11 13:14:18 -04:00
self . supports_cli = False
2018-03-06 18:48:15 -03:00
self . bind_to_localhost_only = True
2017-06-09 18:21:21 -04:00
self . set_test_params ( )
2014-07-08 12:07:23 -04:00
2017-08-24 12:11:56 -03:00
assert hasattr ( self , " num_nodes " ) , " Test must set self.num_nodes in set_test_params() "
2014-07-08 12:07:23 -04:00
def main ( self ) :
2017-06-09 18:21:21 -04:00
""" Main function. This should not be overridden by the subclass test scripts. """
2014-07-08 12:07:23 -04:00
2018-08-11 16:24:14 -04:00
parser = argparse . ArgumentParser ( usage = " %(prog)s [options] " )
parser . add_argument ( " --nocleanup " , dest = " nocleanup " , default = False , action = " store_true " ,
help = " Leave bitcoinds and test.* datadir on exit or error " )
parser . add_argument ( " --noshutdown " , dest = " noshutdown " , default = False , action = " store_true " ,
help = " Don ' t stop bitcoinds after the test execution " )
parser . add_argument ( " --cachedir " , dest = " cachedir " , default = os . path . abspath ( os . path . dirname ( os . path . realpath ( __file__ ) ) + " /../../cache " ) ,
help = " Directory for caching pregenerated datadirs (default: %(default)s ) " )
parser . add_argument ( " --tmpdir " , dest = " tmpdir " , help = " Root directory for datadirs " )
parser . add_argument ( " -l " , " --loglevel " , dest = " loglevel " , default = " INFO " ,
help = " log events at this level and higher to the console. Can be set to DEBUG, INFO, WARNING, ERROR or CRITICAL. Passing --loglevel DEBUG will output all logs to console. Note that logs at all levels are always written to the test_framework.log file in the temporary test directory. " )
parser . add_argument ( " --tracerpc " , dest = " trace_rpc " , default = False , action = " store_true " ,
help = " Print out all RPC calls as they are made " )
parser . add_argument ( " --portseed " , dest = " port_seed " , default = os . getpid ( ) , type = int ,
help = " The seed to use for assigning port numbers (default: current process id) " )
parser . add_argument ( " --coveragedir " , dest = " coveragedir " ,
help = " Write tested RPC commands into this directory " )
parser . add_argument ( " --configfile " , dest = " configfile " ,
default = os . path . abspath ( os . path . dirname ( os . path . realpath ( __file__ ) ) + " /../../config.ini " ) ,
help = " Location of the test framework config file (default: %(default)s ) " )
parser . add_argument ( " --pdbonfailure " , dest = " pdbonfailure " , default = False , action = " store_true " ,
help = " Attach a python debugger if test fails " )
parser . add_argument ( " --usecli " , dest = " usecli " , default = False , action = " store_true " ,
help = " use bitcoin-cli instead of RPC for all commands " )
2018-10-19 13:28:47 -03:00
parser . add_argument ( " --perf " , dest = " perf " , default = False , action = " store_true " ,
help = " profile running nodes with perf for the duration of the test " )
2014-07-08 12:07:23 -04:00
self . add_options ( parser )
2018-08-11 16:24:14 -04:00
self . options = parser . parse_args ( )
2014-07-08 12:07:23 -04:00
2016-05-09 14:55:49 -03:00
PortSeed . n = self . options . port_seed
2014-07-08 12:07:23 -04:00
check_json_precision ( )
2017-10-09 19:04:45 -03:00
self . options . cachedir = os . path . abspath ( self . options . cachedir )
2018-04-19 09:38:59 -03:00
config = configparser . ConfigParser ( )
config . read_file ( open ( self . options . configfile ) )
2018-09-20 21:23:42 -03:00
self . config = config
2018-04-19 09:38:59 -03:00
self . options . bitcoind = os . getenv ( " BITCOIND " , default = config [ " environment " ] [ " BUILDDIR " ] + ' /src/bitcoind ' + config [ " environment " ] [ " EXEEXT " ] )
self . options . bitcoincli = os . getenv ( " BITCOINCLI " , default = config [ " environment " ] [ " BUILDDIR " ] + ' /src/bitcoin-cli ' + config [ " environment " ] [ " EXEEXT " ] )
2018-05-30 17:09:59 -04:00
os . environ [ ' PATH ' ] = os . pathsep . join ( [
os . path . join ( config [ ' environment ' ] [ ' BUILDDIR ' ] , ' src ' ) ,
os . path . join ( config [ ' environment ' ] [ ' BUILDDIR ' ] , ' src ' , ' qt ' ) ,
os . environ [ ' PATH ' ]
] )
2018-05-08 13:56:13 -03:00
2017-02-15 13:36:46 -03:00
# Set up temp directory and start logging
2017-05-18 17:33:33 -04:00
if self . options . tmpdir :
2017-10-09 19:04:45 -03:00
self . options . tmpdir = os . path . abspath ( self . options . tmpdir )
2017-05-18 17:33:33 -04:00
os . makedirs ( self . options . tmpdir , exist_ok = False )
else :
2018-11-07 14:07:25 -03:00
self . options . tmpdir = tempfile . mkdtemp ( prefix = TMPDIR_PREFIX )
2017-02-15 13:36:46 -03:00
self . _start_logging ( )
2018-06-18 17:28:37 -04:00
self . log . debug ( ' Setting up network thread ' )
self . network_thread = NetworkThread ( )
self . network_thread . start ( )
2017-05-18 16:36:39 -04:00
success = TestStatus . FAILED
2017-02-15 13:36:46 -03:00
2014-07-08 12:07:23 -04:00
try :
2018-10-03 05:10:35 -03:00
if self . options . usecli :
if not self . supports_cli :
raise SkipTest ( " --usecli specified but test does not support using CLI " )
self . skip_if_no_cli ( )
2018-09-09 14:32:37 -03:00
self . skip_test_if_missing_module ( )
2014-10-20 09:14:04 -03:00
self . setup_chain ( )
self . setup_network ( )
self . run_test ( )
2017-05-18 16:36:39 -04:00
success = TestStatus . PASSED
2014-07-08 21:24:40 -04:00
except JSONRPCException as e :
2017-02-15 13:36:46 -03:00
self . log . exception ( " JSONRPC error " )
2017-05-18 16:36:39 -04:00
except SkipTest as e :
self . log . warning ( " Test Skipped: %s " % e . message )
success = TestStatus . SKIPPED
2014-07-08 12:07:23 -04:00
except AssertionError as e :
2017-02-15 13:36:46 -03:00
self . log . exception ( " Assertion failed " )
2016-04-13 04:24:07 -03:00
except KeyError as e :
2017-02-15 13:36:46 -03:00
self . log . exception ( " Key error " )
2014-07-08 12:07:23 -04:00
except Exception as e :
2017-02-15 13:36:46 -03:00
self . log . exception ( " Unexpected exception caught during testing " )
2016-04-27 17:29:52 -03:00
except KeyboardInterrupt as e :
2017-02-15 13:36:46 -03:00
self . log . warning ( " Exiting after keyboard interrupt " )
2014-07-08 12:07:23 -04:00
2017-03-31 23:44:04 -03:00
if success == TestStatus . FAILED and self . options . pdbonfailure :
print ( " Testcase failed. Attaching python debugger. Enter ? for help " )
pdb . set_trace ( )
2018-06-18 17:28:37 -04:00
self . log . debug ( ' Closing down network thread ' )
self . network_thread . close ( )
2015-04-23 09:19:00 -03:00
if not self . options . noshutdown :
2017-02-15 13:36:46 -03:00
self . log . info ( " Stopping nodes " )
2017-05-18 16:36:39 -04:00
if self . nodes :
self . stop_nodes ( )
2015-04-23 09:19:00 -03:00
else :
2018-04-06 11:53:35 -03:00
for node in self . nodes :
node . cleanup_on_exit = False
2017-02-15 13:36:46 -03:00
self . log . info ( " Note: bitcoinds were not stopped and may still be running " )
2015-04-20 06:50:33 -03:00
2018-10-19 13:28:47 -03:00
should_clean_up = (
not self . options . nocleanup and
not self . options . noshutdown and
success != TestStatus . FAILED and
not self . options . perf
)
if should_clean_up :
2017-12-09 20:39:06 -03:00
self . log . info ( " Cleaning up {} on exit " . format ( self . options . tmpdir ) )
cleanup_tree_on_exit = True
2018-10-19 13:28:47 -03:00
elif self . options . perf :
self . log . warning ( " Not cleaning up dir {} due to perf data " . format ( self . options . tmpdir ) )
cleanup_tree_on_exit = False
2016-05-24 11:35:06 -04:00
else :
2018-10-19 13:28:47 -03:00
self . log . warning ( " Not cleaning up dir {} " . format ( self . options . tmpdir ) )
2017-12-09 20:39:06 -03:00
cleanup_tree_on_exit = False
2017-05-18 16:36:39 -04:00
if success == TestStatus . PASSED :
2017-02-15 13:36:46 -03:00
self . log . info ( " Tests successful " )
2017-11-29 15:21:51 -03:00
exit_code = TEST_EXIT_PASSED
2017-05-18 16:36:39 -04:00
elif success == TestStatus . SKIPPED :
self . log . info ( " Test skipped " )
2017-11-29 15:21:51 -03:00
exit_code = TEST_EXIT_SKIPPED
2014-07-08 12:07:23 -04:00
else :
2017-02-15 13:36:46 -03:00
self . log . error ( " Test failed. Test logging available at %s /test_framework.log " , self . options . tmpdir )
2017-11-29 15:21:51 -03:00
self . log . error ( " Hint: Call {} ' {} ' to consolidate all logs " . format ( os . path . normpath ( os . path . dirname ( os . path . realpath ( __file__ ) ) + " /../combine_logs.py " ) , self . options . tmpdir ) )
exit_code = TEST_EXIT_FAILED
logging . shutdown ( )
2017-12-09 20:39:06 -03:00
if cleanup_tree_on_exit :
shutil . rmtree ( self . options . tmpdir )
2017-11-29 15:21:51 -03:00
sys . exit ( exit_code )
2015-04-28 13:39:47 -03:00
2017-06-09 18:21:21 -04:00
# Methods to override in subclass test scripts.
def set_test_params ( self ) :
2017-08-24 12:11:56 -03:00
""" Tests must this method to change default values for number of nodes, topology, etc """
raise NotImplementedError
2017-06-09 18:21:21 -04:00
def add_options ( self , parser ) :
""" Override this method to add command-line options to the test """
pass
2018-09-09 14:32:37 -03:00
def skip_test_if_missing_module ( self ) :
""" Override this method to skip a test if a module is not compiled """
pass
2017-06-09 18:21:21 -04:00
def setup_chain ( self ) :
""" Override this method to customize blockchain setup """
self . log . info ( " Initializing test directory " + self . options . tmpdir )
if self . setup_clean_chain :
self . _initialize_chain_clean ( )
else :
self . _initialize_chain ( )
def setup_network ( self ) :
""" Override this method to customize test network topology """
self . setup_nodes ( )
# Connect the nodes as a "chain". This allows us
# to split the network between nodes 1 and 2 to get
# two halves that can work on competing chains.
for i in range ( self . num_nodes - 1 ) :
connect_nodes_bi ( self . nodes , i , i + 1 )
self . sync_all ( )
def setup_nodes ( self ) :
""" Override this method to customize test node setup """
extra_args = None
if hasattr ( self , " extra_args " ) :
extra_args = self . extra_args
self . add_nodes ( self . num_nodes , extra_args )
self . start_nodes ( )
2018-10-16 12:21:07 -03:00
self . import_deterministic_coinbase_privkeys ( )
2019-02-11 15:56:19 -03:00
if not self . setup_clean_chain :
for n in self . nodes :
assert_equal ( n . getblockchaininfo ( ) [ " blocks " ] , 199 )
2019-02-23 14:56:54 -03:00
# To ensure that all nodes are out of IBD, the most recent block
# must have a timestamp not too old (see IsInitialBlockDownload()).
self . log . debug ( ' Generate a block with current time ' )
2019-02-11 15:56:19 -03:00
block_hash = self . nodes [ 0 ] . generate ( 1 ) [ 0 ]
block = self . nodes [ 0 ] . getblock ( blockhash = block_hash , verbosity = 0 )
for n in self . nodes :
n . submitblock ( block )
chain_info = n . getblockchaininfo ( )
assert_equal ( chain_info [ " blocks " ] , 200 )
assert_equal ( chain_info [ " initialblockdownload " ] , False )
2017-06-09 18:21:21 -04:00
2018-09-10 17:58:15 -03:00
def import_deterministic_coinbase_privkeys ( self ) :
for n in self . nodes :
try :
n . getwalletinfo ( )
except JSONRPCException as e :
assert str ( e ) . startswith ( ' Method not found ' )
continue
2018-10-16 12:21:07 -03:00
n . importprivkey ( privkey = n . get_deterministic_priv_key ( ) . key , label = ' coinbase ' )
2018-09-10 17:58:15 -03:00
2017-06-09 18:21:21 -04:00
def run_test ( self ) :
2017-08-24 12:11:56 -03:00
""" Tests must override this method to define test logic """
2017-06-09 18:21:21 -04:00
raise NotImplementedError
2017-03-23 18:48:29 -03:00
# Public helper methods. These can be accessed by the subclass test scripts.
2018-08-01 14:37:47 -04:00
def add_nodes ( self , num_nodes , extra_args = None , * , rpchost = None , binary = None ) :
2018-12-13 17:13:04 -03:00
""" Instantiate TestNode objects.
Should only be called once after the nodes have been specified in
set_test_params ( ) . """
2018-03-06 18:48:15 -03:00
if self . bind_to_localhost_only :
extra_confs = [ [ " bind=127.0.0.1 " ] ] * num_nodes
else :
extra_confs = [ [ ] ] * num_nodes
2017-06-02 14:30:36 -04:00
if extra_args is None :
2017-06-09 16:35:17 -04:00
extra_args = [ [ ] ] * num_nodes
2017-06-02 13:14:14 -04:00
if binary is None :
2018-04-19 09:38:59 -03:00
binary = [ self . options . bitcoind ] * num_nodes
2018-03-06 18:48:15 -03:00
assert_equal ( len ( extra_confs ) , num_nodes )
2017-06-09 16:35:17 -04:00
assert_equal ( len ( extra_args ) , num_nodes )
assert_equal ( len ( binary ) , num_nodes )
for i in range ( num_nodes ) :
2018-12-13 17:13:04 -03:00
self . nodes . append ( TestNode (
i ,
get_datadir_path ( self . options . tmpdir , i ) ,
rpchost = rpchost ,
2018-12-22 13:22:55 -03:00
timewait = self . rpc_timeout ,
2018-12-13 17:13:04 -03:00
bitcoind = binary [ i ] ,
bitcoin_cli = self . options . bitcoincli ,
coverage_dir = self . options . coveragedir ,
2019-02-15 08:54:29 -03:00
cwd = self . options . tmpdir ,
2018-12-13 17:13:04 -03:00
extra_conf = extra_confs [ i ] ,
extra_args = extra_args [ i ] ,
use_cli = self . options . usecli ,
2018-10-19 13:28:47 -03:00
start_perf = self . options . perf ,
2018-12-13 17:13:04 -03:00
) )
2017-06-09 16:35:17 -04:00
2018-01-18 15:15:00 -03:00
def start_node ( self , i , * args , * * kwargs ) :
2017-06-09 16:35:17 -04:00
""" Start a bitcoind """
node = self . nodes [ i ]
2018-01-18 15:15:00 -03:00
node . start ( * args , * * kwargs )
2017-06-02 14:30:36 -04:00
node . wait_for_rpc_connection ( )
2017-06-02 13:14:14 -04:00
2017-06-02 14:30:36 -04:00
if self . options . coveragedir is not None :
coverage . write_all_rpc_commands ( self . options . coveragedir , node . rpc )
2017-06-02 13:14:14 -04:00
2018-01-18 15:15:00 -03:00
def start_nodes ( self , extra_args = None , * args , * * kwargs ) :
2017-06-09 16:35:17 -04:00
""" Start multiple bitcoinds """
2017-06-02 13:14:14 -04:00
if extra_args is None :
2017-06-09 16:35:17 -04:00
extra_args = [ None ] * self . num_nodes
assert_equal ( len ( extra_args ) , self . num_nodes )
2017-06-02 13:14:14 -04:00
try :
2017-06-09 16:35:17 -04:00
for i , node in enumerate ( self . nodes ) :
2018-01-18 15:15:00 -03:00
node . start ( extra_args [ i ] , * args , * * kwargs )
2017-06-09 16:35:17 -04:00
for node in self . nodes :
2017-06-02 14:30:36 -04:00
node . wait_for_rpc_connection ( )
2017-06-02 13:14:14 -04:00
except :
# If one node failed to start, stop the others
self . stop_nodes ( )
raise
2017-06-02 14:30:36 -04:00
if self . options . coveragedir is not None :
2017-06-09 16:35:17 -04:00
for node in self . nodes :
2017-06-02 14:30:36 -04:00
coverage . write_all_rpc_commands ( self . options . coveragedir , node . rpc )
2018-11-20 14:59:07 -03:00
def stop_node ( self , i , expected_stderr = ' ' , wait = 0 ) :
2017-06-02 13:14:14 -04:00
""" Stop a bitcoind test node """
2018-11-20 14:59:07 -03:00
self . nodes [ i ] . stop_node ( expected_stderr , wait = wait )
2017-08-16 12:52:24 -03:00
self . nodes [ i ] . wait_until_stopped ( )
2017-03-23 18:48:29 -03:00
2018-11-20 14:59:07 -03:00
def stop_nodes ( self , wait = 0 ) :
2017-06-02 13:14:14 -04:00
""" Stop multiple bitcoind test nodes """
2017-06-02 14:30:36 -04:00
for node in self . nodes :
# Issue RPC to stop nodes
2018-11-20 14:59:07 -03:00
node . stop_node ( wait = wait )
2017-06-02 13:14:14 -04:00
2017-06-02 14:30:36 -04:00
for node in self . nodes :
# Wait for nodes to stop
2017-08-16 12:52:24 -03:00
node . wait_until_stopped ( )
2017-06-02 13:14:14 -04:00
2017-09-20 11:00:28 -03:00
def restart_node ( self , i , extra_args = None ) :
""" Stop and start a test node """
self . stop_node ( i )
self . start_node ( i , extra_args )
2017-06-02 13:14:14 -04:00
def wait_for_node_exit ( self , i , timeout ) :
2017-06-02 14:30:36 -04:00
self . nodes [ i ] . process . wait ( timeout )
2017-03-23 18:48:29 -03:00
def split_network ( self ) :
"""
Split the network of four nodes into nodes 0 / 1 and 2 / 3.
"""
disconnect_nodes ( self . nodes [ 1 ] , 2 )
disconnect_nodes ( self . nodes [ 2 ] , 1 )
self . sync_all ( [ self . nodes [ : 2 ] , self . nodes [ 2 : ] ] )
def join_network ( self ) :
"""
Join the ( previously split ) network halves together .
"""
connect_nodes_bi ( self . nodes , 1 , 2 )
self . sync_all ( )
def sync_all ( self , node_groups = None ) :
if not node_groups :
node_groups = [ self . nodes ]
for group in node_groups :
sync_blocks ( group )
sync_mempools ( group )
# Private helper methods. These should not be accessed by the subclass test scripts.
2017-02-15 13:36:46 -03:00
def _start_logging ( self ) :
# Add logger and logging handlers
self . log = logging . getLogger ( ' TestFramework ' )
self . log . setLevel ( logging . DEBUG )
# Create file handler to log all messages
2018-06-11 16:25:44 -04:00
fh = logging . FileHandler ( self . options . tmpdir + ' /test_framework.log ' , encoding = ' utf-8 ' )
2017-02-15 13:36:46 -03:00
fh . setLevel ( logging . DEBUG )
# Create console handler to log messages to stderr. By default this logs only error messages, but can be configured with --loglevel.
ch = logging . StreamHandler ( sys . stdout )
# User can provide log level as a number or string (eg DEBUG). loglevel was caught as a string, so try to convert it to an int
ll = int ( self . options . loglevel ) if self . options . loglevel . isdigit ( ) else self . options . loglevel . upper ( )
ch . setLevel ( ll )
# Format logs the same as bitcoind's debug.log with microprecision (so log files can be concatenated and sorted)
2018-02-28 12:46:31 -03:00
formatter = logging . Formatter ( fmt = ' %(asctime)s . %(msecs)03d 000Z %(name)s ( %(levelname)s ): %(message)s ' , datefmt = ' % Y- % m- %d T % H: % M: % S ' )
2017-03-21 11:05:59 -03:00
formatter . converter = time . gmtime
2017-02-15 13:36:46 -03:00
fh . setFormatter ( formatter )
ch . setFormatter ( formatter )
# add the handlers to the logger
self . log . addHandler ( fh )
self . log . addHandler ( ch )
if self . options . trace_rpc :
rpc_logger = logging . getLogger ( " BitcoinRPC " )
rpc_logger . setLevel ( logging . DEBUG )
rpc_handler = logging . StreamHandler ( sys . stdout )
rpc_handler . setLevel ( logging . DEBUG )
rpc_logger . addHandler ( rpc_handler )
2015-04-28 13:39:47 -03:00
2017-08-24 12:37:25 -03:00
def _initialize_chain ( self ) :
2017-03-23 17:49:02 -03:00
""" Initialize a pre-mined blockchain for use by the test.
2019-02-11 15:56:19 -03:00
Create a cache of a 199 - block - long chain ( with wallet ) for MAX_NODES
2017-03-23 17:49:02 -03:00
Afterward , create num_nodes copies from the cache . """
2017-08-24 12:37:25 -03:00
assert self . num_nodes < = MAX_NODES
2017-03-23 17:49:02 -03:00
create_cache = False
for i in range ( MAX_NODES ) :
2018-03-07 15:49:44 -03:00
if not os . path . isdir ( get_datadir_path ( self . options . cachedir , i ) ) :
2017-03-23 17:49:02 -03:00
create_cache = True
break
if create_cache :
self . log . debug ( " Creating data directories from cached datadir " )
# find and delete old cache directories if any exist
for i in range ( MAX_NODES ) :
2018-03-07 15:49:44 -03:00
if os . path . isdir ( get_datadir_path ( self . options . cachedir , i ) ) :
shutil . rmtree ( get_datadir_path ( self . options . cachedir , i ) )
2017-03-23 17:49:02 -03:00
# Create cache directories, run bitcoinds:
for i in range ( MAX_NODES ) :
2017-08-24 12:37:25 -03:00
datadir = initialize_datadir ( self . options . cachedir , i )
2018-09-10 17:58:15 -03:00
args = [ self . options . bitcoind , " -datadir= " + datadir , ' -disablewallet ' ]
2017-03-23 17:49:02 -03:00
if i > 0 :
args . append ( " -connect=127.0.0.1: " + str ( p2p_port ( 0 ) ) )
2018-12-22 13:33:48 -03:00
self . nodes . append ( TestNode (
i ,
get_datadir_path ( self . options . cachedir , i ) ,
extra_conf = [ " bind=127.0.0.1 " ] ,
extra_args = [ ] ,
rpchost = None ,
timewait = self . rpc_timeout ,
bitcoind = self . options . bitcoind ,
bitcoin_cli = self . options . bitcoincli ,
coverage_dir = None ,
2019-02-15 08:54:29 -03:00
cwd = self . options . tmpdir ,
2018-12-22 13:33:48 -03:00
) )
2017-06-02 14:30:36 -04:00
self . nodes [ i ] . args = args
2017-06-09 16:35:17 -04:00
self . start_node ( i )
2017-03-23 17:49:02 -03:00
2017-06-02 14:30:36 -04:00
# Wait for RPC connections to be ready
for node in self . nodes :
node . wait_for_rpc_connection ( )
2017-03-23 17:49:02 -03:00
2019-02-11 15:56:19 -03:00
# Create a 199-block-long chain; each of the 4 first nodes
2017-03-23 17:49:02 -03:00
# gets 25 mature blocks and 25 immature.
2019-02-11 15:56:19 -03:00
# The 4th node gets only 24 immature blocks so that the very last
# block in the cache does not age too much (have an old tip age).
# This is needed so that we are out of IBD when the test starts,
# see the tip age check in IsInitialBlockDownload().
2019-02-23 14:56:54 -03:00
for i in range ( 8 ) :
self . nodes [ 0 ] . generatetoaddress ( 25 if i != 7 else 24 , self . nodes [ i % 4 ] . get_deterministic_priv_key ( ) . address )
sync_blocks ( self . nodes )
2017-03-23 17:49:02 -03:00
2019-02-11 15:56:19 -03:00
for n in self . nodes :
assert_equal ( n . getblockchaininfo ( ) [ " blocks " ] , 199 )
2017-03-23 17:49:02 -03:00
# Shut them down, and clean up cache directories:
2017-03-23 17:56:45 -03:00
self . stop_nodes ( )
self . nodes = [ ]
2018-03-07 15:49:44 -03:00
def cache_path ( n , * paths ) :
return os . path . join ( get_datadir_path ( self . options . cachedir , n ) , " regtest " , * paths )
2017-03-23 17:49:02 -03:00
for i in range ( MAX_NODES ) :
2018-09-10 17:58:15 -03:00
os . rmdir ( cache_path ( i , ' wallets ' ) ) # Remove empty wallets dir
2018-03-07 15:49:44 -03:00
for entry in os . listdir ( cache_path ( i ) ) :
2018-09-10 17:58:15 -03:00
if entry not in [ ' chainstate ' , ' blocks ' ] :
2018-03-07 15:49:44 -03:00
os . remove ( cache_path ( i , entry ) )
2017-08-24 12:37:25 -03:00
for i in range ( self . num_nodes ) :
2018-03-07 15:49:44 -03:00
from_dir = get_datadir_path ( self . options . cachedir , i )
to_dir = get_datadir_path ( self . options . tmpdir , i )
2017-03-23 17:49:02 -03:00
shutil . copytree ( from_dir , to_dir )
2017-08-24 12:37:25 -03:00
initialize_datadir ( self . options . tmpdir , i ) # Overwrite port/rpcport in bitcoin.conf
2017-03-23 17:49:02 -03:00
2017-08-24 12:37:25 -03:00
def _initialize_chain_clean ( self ) :
2017-03-23 17:49:02 -03:00
""" Initialize empty blockchain for use by the test.
Create an empty blockchain and num_nodes wallets .
Useful if a test case wants complete control over initialization . """
2017-08-24 12:37:25 -03:00
for i in range ( self . num_nodes ) :
initialize_datadir ( self . options . tmpdir , i )
2017-03-23 17:49:02 -03:00
2018-09-09 14:32:37 -03:00
def skip_if_no_py3_zmq ( self ) :
""" Attempt to import the zmq package and skip the test if the import fails. """
try :
import zmq # noqa
except ImportError :
raise SkipTest ( " python3-zmq module not available. " )
def skip_if_no_bitcoind_zmq ( self ) :
""" Skip the running test if bitcoind has not been compiled with zmq support. """
if not self . is_zmq_compiled ( ) :
raise SkipTest ( " bitcoind has not been built with zmq enabled. " )
def skip_if_no_wallet ( self ) :
""" Skip the running test if wallet has not been compiled. """
if not self . is_wallet_compiled ( ) :
raise SkipTest ( " wallet has not been compiled. " )
def skip_if_no_cli ( self ) :
""" Skip the running test if bitcoin-cli has not been compiled. """
if not self . is_cli_compiled ( ) :
raise SkipTest ( " bitcoin-cli has not been compiled. " )
def is_cli_compiled ( self ) :
""" Checks whether bitcoin-cli was compiled. """
config = configparser . ConfigParser ( )
config . read_file ( open ( self . options . configfile ) )
2018-04-02 02:45:41 -03:00
2014-12-25 08:43:52 -03:00
return config [ " components " ] . getboolean ( " ENABLE_CLI " )
2018-07-12 11:52:02 -04:00
2018-09-09 14:32:37 -03:00
def is_wallet_compiled ( self ) :
""" Checks whether the wallet module was compiled. """
config = configparser . ConfigParser ( )
config . read_file ( open ( self . options . configfile ) )
2018-08-31 17:30:26 -03:00
2018-09-09 14:32:37 -03:00
return config [ " components " ] . getboolean ( " ENABLE_WALLET " )
2018-08-31 17:30:26 -03:00
2018-09-09 14:32:37 -03:00
def is_zmq_compiled ( self ) :
""" Checks whether the zmq module was compiled. """
config = configparser . ConfigParser ( )
config . read_file ( open ( self . options . configfile ) )
2018-07-12 11:52:02 -04:00
2018-09-09 14:32:37 -03:00
return config [ " components " ] . getboolean ( " ENABLE_ZMQ " )