2016-03-19 20:58:06 +01:00
#!/usr/bin/env python3
2022-12-24 23:49:50 +00:00
# Copyright (c) 2014-2022 The Bitcoin Core developers
2014-10-23 09:48:19 +08: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 18:34:40 -05:00
""" Base class for RPC testing. """
2014-07-08 12:07:23 -04:00
2018-04-19 08:38:59 -04:00
import configparser
2017-05-18 16:36:39 -04:00
from enum import Enum
2018-08-11 22:24:14 +02:00
import argparse
2020-04-30 09:36:54 -04:00
import logging
2014-07-08 12:07:23 -04:00
import os
2022-02-03 11:25:30 +01:00
import platform
2017-03-31 22:44:04 -04:00
import pdb
2019-05-06 13:17:04 -04:00
import random
2020-04-30 09:36:54 -04:00
import re
2014-07-08 12:07:23 -04:00
import shutil
2019-12-04 10:28:09 -05:00
import subprocess
2017-03-23 16:49:02 -04:00
import sys
2014-07-08 12:07:23 -04:00
import tempfile
2017-03-21 10:05:59 -04:00
import time
2014-07-08 12:07:23 -04:00
2021-01-17 18:35:26 +01:00
from typing import List
2021-10-27 11:18:57 +02:00
from . address import create_deterministic_address_bcrt1_p2tr_op_true
2017-06-02 13:14:14 -04:00
from . authproxy import JSONRPCException
from . import coverage
2020-07-19 14:47:05 +07:00
from . p2p import NetworkThread
2020-08-17 10:45:44 +01:00
from . test_node import TestNode
2015-10-10 22:41:19 -07:00
from . util import (
2017-03-23 16:49:02 -04:00
MAX_NODES ,
2017-06-02 13:14:14 -04:00
PortSeed ,
assert_equal ,
2017-03-23 16:49:02 -04:00
check_json_precision ,
2018-03-07 13:49:44 -05:00
get_datadir_path ,
2017-03-23 16:49:02 -04:00
initialize_datadir ,
2020-09-17 00:24:03 -07:00
p2p_port ,
2020-08-27 08:55:20 +02:00
wait_until_helper ,
2015-10-10 22:41:19 -07:00
)
2014-07-08 12:07:23 -04:00
2019-02-11 11:18:36 -05: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 12:07:25 -05:00
TMPDIR_PREFIX = " bitcoin_func_test_ "
2018-04-01 22:45:41 -07:00
2018-09-09 13:32:37 -04:00
class SkipTest ( Exception ) :
""" This exception is raised to skip a test """
def __init__ ( self , message ) :
self . message = message
2018-04-01 22:45:41 -07: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 17:48:29 -04:00
""" Base class for a bitcoin test script.
2017-08-24 11:11:56 -04:00
Individual bitcoin test scripts should subclass this class and override the set_test_params ( ) and run_test ( ) methods .
2017-03-23 17:48:29 -04: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 17:48:29 -04:00
- add_options ( )
- setup_chain ( )
- setup_network ( )
2017-06-09 18:21:21 -04:00
- setup_nodes ( )
2017-03-23 17:48:29 -04:00
2017-06-09 18:21:21 -04:00
The __init__ ( ) and main ( ) methods should not be overridden .
2017-03-23 17:48:29 -04:00
This class also contains various public and private helper methods . """
2016-05-14 13:01:31 +02: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 """
2021-01-17 18:35:26 +01:00
self . chain : str = ' regtest '
self . setup_clean_chain : bool = False
self . nodes : List [ TestNode ] = [ ]
2022-12-01 12:14:10 +01:00
self . extra_args = None
2018-06-18 17:28:37 -04:00
self . network_thread = None
2018-12-22 17:22:55 +01:00
self . rpc_timeout = 60 # Wait for up to 60 seconds for the RPC server to respond
2019-12-06 13:22:04 +00:00
self . supports_cli = True
2018-03-06 16:48:15 -05:00
self . bind_to_localhost_only = True
2020-09-28 20:24:06 -04:00
self . parse_args ( )
2021-11-15 13:44:25 +01:00
self . disable_syscall_sandbox = self . options . nosandbox or self . options . valgrind
2020-10-07 14:50:05 -04:00
self . default_wallet_name = " default_wallet " if self . options . descriptors else " "
2020-09-28 20:24:06 -04:00
self . wallet_data_filename = " wallet.dat "
2020-09-28 20:24:06 -04:00
# Optional list of wallet names that can be set in set_test_params to
# create and import keys to. If unset, default is len(nodes) *
# [default_wallet_name]. If wallet names are None, wallet creation is
# skipped. If list is truncated, wallet creation is skipped and keys
# are not imported.
self . wallet_names = None
2020-10-29 13:51:41 -04:00
# By default the wallet is not required. Set to true by skip_if_no_wallet().
# When False, we ignore wallet_names regardless of what it is.
2022-11-10 10:12:39 +01:00
self . _requires_wallet = False
2021-07-26 17:36:01 +02:00
# Disable ThreadOpenConnections by default, so that adding entries to
# addrman will not result in automatic connections to them.
self . disable_autoconnect = True
2017-06-09 18:21:21 -04:00
self . set_test_params ( )
2020-10-19 08:24:27 -04:00
assert self . wallet_names is None or len ( self . wallet_names ) < = self . num_nodes
2020-05-18 09:45:55 +05:30
self . rpc_timeout = int ( self . rpc_timeout * self . options . timeout_factor ) # optionally, increase timeout by a factor
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
2019-10-26 16:00:02 +02:00
assert hasattr ( self , " num_nodes " ) , " Test must set self.num_nodes in set_test_params() "
2019-10-26 16:34:42 +02:00
try :
self . setup ( )
self . run_test ( )
except JSONRPCException :
self . log . exception ( " JSONRPC error " )
self . success = TestStatus . FAILED
except SkipTest as e :
self . log . warning ( " Test Skipped: %s " % e . message )
self . success = TestStatus . SKIPPED
except AssertionError :
self . log . exception ( " Assertion failed " )
self . success = TestStatus . FAILED
except KeyError :
self . log . exception ( " Key error " )
self . success = TestStatus . FAILED
2019-12-04 10:28:09 -05:00
except subprocess . CalledProcessError as e :
self . log . exception ( " Called Process failed with ' {} ' " . format ( e . output ) )
self . success = TestStatus . FAILED
2019-10-26 16:34:42 +02:00
except Exception :
self . log . exception ( " Unexpected exception caught during testing " )
self . success = TestStatus . FAILED
except KeyboardInterrupt :
self . log . warning ( " Exiting after keyboard interrupt " )
self . success = TestStatus . FAILED
finally :
exit_code = self . shutdown ( )
sys . exit ( exit_code )
def parse_args ( self ) :
2020-05-19 07:12:22 -04:00
previous_releases_path = os . getenv ( " PREVIOUS_RELEASES_DIR " ) or os . getcwd ( ) + " /releases "
2018-08-11 22:24:14 +02: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 " )
2021-10-01 13:53:59 +00:00
parser . add_argument ( " --nosandbox " , dest = " nosandbox " , default = False , action = " store_true " ,
help = " Don ' t use the syscall sandbox " )
2018-08-11 22:24:14 +02:00
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) " )
2020-05-19 07:12:22 -04:00
parser . add_argument ( " --previous-releases " , dest = " prev_releases " , action = " store_true " ,
2020-05-19 11:08:51 -04:00
default = os . path . isdir ( previous_releases_path ) and bool ( os . listdir ( previous_releases_path ) ) ,
2020-05-19 07:12:22 -04:00
help = " Force test of previous releases (default: %(default)s ) " )
2018-08-11 22:24:14 +02:00
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 12:28:47 -04:00
parser . add_argument ( " --perf " , dest = " perf " , default = False , action = " store_true " ,
help = " profile running nodes with perf for the duration of the test " )
2019-11-29 15:11:18 +00:00
parser . add_argument ( " --valgrind " , dest = " valgrind " , default = False , action = " store_true " ,
2021-11-15 13:44:25 +01:00
help = " run nodes under the valgrind memory error detector: expect at least a ~10x slowdown. valgrind 3.14 or later required. Forces --nosandbox. " )
2019-05-06 13:17:04 -04:00
parser . add_argument ( " --randomseed " , type = int ,
help = " set a random seed for deterministically reproducing a previous test run " )
2023-03-07 17:06:28 +01:00
parser . add_argument ( " --timeout-factor " , dest = " timeout_factor " , type = float , help = " adjust test timeouts by a factor. Setting it to 0 disables all timeouts " )
2020-04-27 15:38:20 -04:00
2014-07-08 12:07:23 -04:00
self . add_options ( parser )
2021-06-09 15:31:48 +02:00
# Running TestShell in a Jupyter notebook causes an additional -f argument
# To keep TestShell from failing with an "unrecognized argument" error, we add a dummy "-f" argument
# source: https://stackoverflow.com/questions/48796169/how-to-fix-ipykernel-launcher-py-error-unrecognized-arguments-in-jupyter/56349168#56349168
parser . add_argument ( " -f " , " --fff " , help = " a dummy argument to fool ipython " , default = " 1 " )
2018-08-11 22:24:14 +02:00
self . options = parser . parse_args ( )
2023-03-07 17:06:28 +01:00
if self . options . timeout_factor == 0 :
self . options . timeout_factor = 99999
self . options . timeout_factor = self . options . timeout_factor or ( 4 if self . options . valgrind else 1 )
2020-05-19 07:12:22 -04:00
self . options . previous_releases_path = previous_releases_path
2014-07-08 12:07:23 -04:00
2020-10-29 17:17:10 -04:00
config = configparser . ConfigParser ( )
config . read_file ( open ( self . options . configfile ) )
self . config = config
2022-11-09 12:53:13 +01:00
if " descriptors " not in self . options :
# Wallet is not required by the test at all and the value of self.options.descriptors won't matter.
# It still needs to exist and be None in order for tests to work however.
# So set it to None to force -disablewallet, because the wallet is not needed.
self . options . descriptors = None
elif self . options . descriptors is None :
# Some wallet is either required or optionally used by the test.
2022-12-05 09:54:38 +01:00
# Prefer SQLite unless it isn't available
if self . is_sqlite_compiled ( ) :
2020-10-29 17:17:10 -04:00
self . options . descriptors = True
2022-12-05 09:54:38 +01:00
elif self . is_bdb_compiled ( ) :
self . options . descriptors = False
2020-10-29 17:17:10 -04:00
else :
# If neither are compiled, tests requiring a wallet will be skipped and the value of self.options.descriptors won't matter
# It still needs to exist and be None in order for tests to work however.
2022-11-09 12:53:13 +01:00
# So set it to None, which will also set -disablewallet.
2020-10-29 17:17:10 -04:00
self . options . descriptors = None
2022-06-08 20:57:48 +02:00
PortSeed . n = self . options . port_seed
2019-10-26 16:34:42 +02:00
def setup ( self ) :
""" Call this method to start up the test framework object with options set. """
2014-07-08 12:07:23 -04:00
check_json_precision ( )
2017-10-10 00:04:45 +02:00
self . options . cachedir = os . path . abspath ( self . options . cachedir )
2020-10-29 17:17:10 -04:00
config = self . config
2020-05-11 22:31:02 +02:00
fname_bitcoind = os . path . join (
config [ " environment " ] [ " BUILDDIR " ] ,
" src " ,
2020-05-19 07:12:22 -04:00
" bitcoind " + config [ " environment " ] [ " EXEEXT " ] ,
2020-05-11 22:31:02 +02:00
)
fname_bitcoincli = os . path . join (
config [ " environment " ] [ " BUILDDIR " ] ,
" src " ,
2020-05-19 07:12:22 -04:00
" bitcoin-cli " + config [ " environment " ] [ " EXEEXT " ] ,
2020-05-11 22:31:02 +02:00
)
2022-04-11 21:30:34 +02:00
fname_bitcoinutil = os . path . join (
config [ " environment " ] [ " BUILDDIR " ] ,
" src " ,
" bitcoin-util " + config [ " environment " ] [ " EXEEXT " ] ,
)
2020-05-11 22:31:02 +02:00
self . options . bitcoind = os . getenv ( " BITCOIND " , default = fname_bitcoind )
self . options . bitcoincli = os . getenv ( " BITCOINCLI " , default = fname_bitcoincli )
2022-04-11 21:30:34 +02:00
self . options . bitcoinutil = os . getenv ( " BITCOINUTIL " , default = fname_bitcoinutil )
2018-04-19 08:38:59 -04:00
2018-05-30 17:09:59 -04:00
os . environ [ ' PATH ' ] = os . pathsep . join ( [
os . path . join ( config [ ' environment ' ] [ ' BUILDDIR ' ] , ' src ' ) ,
2020-04-30 09:36:54 -04:00
os . path . join ( config [ ' environment ' ] [ ' BUILDDIR ' ] , ' src ' , ' qt ' ) , os . environ [ ' PATH ' ]
2018-05-30 17:09:59 -04:00
] )
2018-05-08 12:56:13 -04:00
2017-02-15 11:36:46 -05:00
# Set up temp directory and start logging
2017-05-18 17:33:33 -04:00
if self . options . tmpdir :
2017-10-10 00:04:45 +02: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 12:07:25 -05:00
self . options . tmpdir = tempfile . mkdtemp ( prefix = TMPDIR_PREFIX )
2017-02-15 11:36:46 -05:00
self . _start_logging ( )
2019-05-06 13:17:04 -04:00
# Seed the PRNG. Note that test runs are reproducible if and only if
# a single thread accesses the PRNG. For more information, see
# https://docs.python.org/3/library/random.html#notes-on-reproducibility.
# The network thread shouldn't access random. If we need to change the
# network thread to access randomness, it should instantiate its own
# random.Random object.
seed = self . options . randomseed
if seed is None :
seed = random . randrange ( sys . maxsize )
else :
2023-02-21 12:01:13 -05:00
self . log . info ( " User supplied random seed {} " . format ( seed ) )
2019-05-06 13:17:04 -04:00
random . seed ( seed )
2023-02-21 12:01:13 -05:00
self . log . info ( " PRNG seed is: {} " . format ( seed ) )
2019-05-06 13:17:04 -04:00
2018-06-18 17:28:37 -04:00
self . log . debug ( ' Setting up network thread ' )
self . network_thread = NetworkThread ( )
self . network_thread . start ( )
2019-10-26 16:34:42 +02: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 ( )
self . skip_test_if_missing_module ( )
self . setup_chain ( )
self . setup_network ( )
2017-02-15 11:36:46 -05:00
2019-10-26 16:34:42 +02:00
self . success = TestStatus . PASSED
def shutdown ( self ) :
""" Call this method to shut down the test framework object. """
2014-07-08 12:07:23 -04:00
2019-10-26 16:34:42 +02:00
if self . success == TestStatus . FAILED and self . options . pdbonfailure :
2017-03-31 22:44:04 -04:00
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 14:19:00 +02:00
if not self . options . noshutdown :
2017-02-15 11:36:46 -05:00
self . log . info ( " Stopping nodes " )
2017-05-18 16:36:39 -04:00
if self . nodes :
self . stop_nodes ( )
2015-04-23 14:19:00 +02:00
else :
2018-04-06 10:53:35 -04:00
for node in self . nodes :
node . cleanup_on_exit = False
2017-02-15 11:36:46 -05:00
self . log . info ( " Note: bitcoinds were not stopped and may still be running " )
2015-04-20 11:50:33 +02:00
2018-10-19 12:28:47 -04:00
should_clean_up = (
not self . options . nocleanup and
not self . options . noshutdown and
2019-10-26 16:34:42 +02:00
self . success != TestStatus . FAILED and
2018-10-19 12:28:47 -04:00
not self . options . perf
)
if should_clean_up :
2017-12-09 18:39:06 -05:00
self . log . info ( " Cleaning up {} on exit " . format ( self . options . tmpdir ) )
cleanup_tree_on_exit = True
2018-10-19 12:28:47 -04: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 12:28:47 -04:00
self . log . warning ( " Not cleaning up dir {} " . format ( self . options . tmpdir ) )
2017-12-09 18:39:06 -05:00
cleanup_tree_on_exit = False
2017-05-18 16:36:39 -04:00
2019-10-26 16:34:42 +02:00
if self . success == TestStatus . PASSED :
2017-02-15 11:36:46 -05:00
self . log . info ( " Tests successful " )
2017-11-29 13:21:51 -05:00
exit_code = TEST_EXIT_PASSED
2019-10-26 16:34:42 +02:00
elif self . success == TestStatus . SKIPPED :
2017-05-18 16:36:39 -04:00
self . log . info ( " Test skipped " )
2017-11-29 13:21:51 -05:00
exit_code = TEST_EXIT_SKIPPED
2014-07-08 12:07:23 -04:00
else :
2017-02-15 11:36:46 -05:00
self . log . error ( " Test failed. Test logging available at %s /test_framework.log " , self . options . tmpdir )
2020-05-29 15:32:56 -04:00
self . log . error ( " " )
2017-11-29 13:21:51 -05: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 ) )
2020-05-29 15:32:56 -04:00
self . log . error ( " " )
self . log . error ( " If this failure happened unexpectedly or intermittently, please file a bug and provide a link or upload of the combined log. " )
self . log . error ( self . config [ ' environment ' ] [ ' PACKAGE_BUGREPORT ' ] )
self . log . error ( " " )
2017-11-29 13:21:51 -05:00
exit_code = TEST_EXIT_FAILED
2019-10-28 14:50:01 +01:00
# Logging.shutdown will not remove stream- and filehandlers, so we must
# do it explicitly. Handlers are removed so the next test run can apply
# different log handler settings.
# See: https://docs.python.org/3/library/logging.html#logging.shutdown
for h in list ( self . log . handlers ) :
h . flush ( )
h . close ( )
self . log . removeHandler ( h )
rpc_logger = logging . getLogger ( " BitcoinRPC " )
for h in list ( rpc_logger . handlers ) :
h . flush ( )
rpc_logger . removeHandler ( h )
2017-12-09 18:39:06 -05:00
if cleanup_tree_on_exit :
shutil . rmtree ( self . options . tmpdir )
2019-10-26 16:03:21 +02:00
self . nodes . clear ( )
2019-10-26 16:34:42 +02:00
return exit_code
2015-04-28 12:39:47 -04:00
2017-06-09 18:21:21 -04:00
# Methods to override in subclass test scripts.
def set_test_params ( self ) :
2020-12-24 10:54:56 -08:00
""" Tests must override this method to change default values for number of nodes, topology, etc """
2017-08-24 11:11:56 -04:00
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 13:32:37 -04: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.
2019-09-16 09:22:49 -04:00
#
# Topology looks like this:
# node0 <-- node1 <-- node2 <-- node3
#
# If all nodes are in IBD (clean chain from genesis), node0 is assumed to be the source of blocks (miner). To
# ensure block propagation, all nodes will establish outgoing connections toward node0.
# See fPreferredDownload in net_processing.
#
# If further outbound connections are needed, they can be added at the beginning of the test with e.g.
2020-06-07 09:54:56 -04:00
# self.connect_nodes(1, 2)
2017-06-09 18:21:21 -04:00
for i in range ( self . num_nodes - 1 ) :
2020-06-07 09:54:56 -04:00
self . connect_nodes ( i + 1 , i )
2017-06-09 18:21:21 -04:00
self . sync_all ( )
def setup_nodes ( self ) :
""" Override this method to customize test node setup """
2022-12-01 16:55:43 +01:00
self . add_nodes ( self . num_nodes , self . extra_args )
2017-06-09 18:21:21 -04:00
self . start_nodes ( )
2022-11-10 10:12:39 +01:00
if self . _requires_wallet :
2020-09-28 20:24:06 -04:00
self . import_deterministic_coinbase_privkeys ( )
2019-02-11 13:56:19 -05:00
if not self . setup_clean_chain :
for n in self . nodes :
assert_equal ( n . getblockchaininfo ( ) [ " blocks " ] , 199 )
2019-02-23 12:56:54 -05: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 ' )
2020-11-10 18:02:31 +01:00
block_hash = self . generate ( self . nodes [ 0 ] , 1 , sync_fun = self . no_op ) [ 0 ]
2019-02-11 13:56:19 -05:00
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 16:58:15 -04:00
def import_deterministic_coinbase_privkeys ( self ) :
2020-10-19 08:24:27 -04:00
for i in range ( self . num_nodes ) :
2021-10-20 00:30:28 -03:00
self . init_wallet ( node = i )
2020-10-19 08:24:27 -04:00
2021-10-20 00:30:28 -03:00
def init_wallet ( self , * , node ) :
wallet_name = self . default_wallet_name if self . wallet_names is None else self . wallet_names [ node ] if node < len ( self . wallet_names ) else False
2020-10-19 08:24:27 -04:00
if wallet_name is not False :
2021-10-20 00:30:28 -03:00
n = self . nodes [ node ]
2020-09-28 20:24:06 -04:00
if wallet_name is not None :
n . createwallet ( wallet_name = wallet_name , descriptors = self . options . descriptors , load_on_startup = True )
2021-11-26 17:10:55 +01:00
n . importprivkey ( privkey = n . get_deterministic_priv_key ( ) . key , label = ' coinbase ' , rescan = True )
2018-09-10 16:58:15 -04:00
2017-06-09 18:21:21 -04:00
def run_test ( self ) :
2017-08-24 11:11:56 -04:00
""" Tests must override this method to define test logic """
2017-06-09 18:21:21 -04:00
raise NotImplementedError
2017-03-23 17:48:29 -04:00
# Public helper methods. These can be accessed by the subclass test scripts.
2022-11-09 12:53:13 +01:00
def add_wallet_options ( self , parser , * , descriptors = True , legacy = True ) :
2022-11-15 10:05:02 +01:00
kwargs = { }
if descriptors + legacy == 1 :
# If only one type can be chosen, set it as default
kwargs [ " default " ] = descriptors
2022-11-29 14:26:46 +01:00
group = parser . add_mutually_exclusive_group (
# If only one type is allowed, require it to be set in test_runner.py
required = os . getenv ( " REQUIRE_WALLET_TYPE_SET " ) == " 1 " and " default " in kwargs )
2022-11-09 12:53:13 +01:00
if descriptors :
2022-11-15 10:05:02 +01:00
group . add_argument ( " --descriptors " , action = ' store_const ' , const = True , * * kwargs ,
2022-11-09 12:53:13 +01:00
help = " Run test using a descriptor wallet " , dest = ' descriptors ' )
if legacy :
2022-11-15 10:05:02 +01:00
group . add_argument ( " --legacy-wallet " , action = ' store_const ' , const = False , * * kwargs ,
2022-11-09 12:53:13 +01:00
help = " Run test using legacy wallets " , dest = ' descriptors ' )
2020-02-25 20:05:48 +01:00
def add_nodes ( self , num_nodes : int , extra_args = None , * , rpchost = None , binary = None , binary_cli = None , versions = None ) :
2018-12-13 15:13:04 -05:00
""" Instantiate TestNode objects.
Should only be called once after the nodes have been specified in
set_test_params ( ) . """
2020-04-30 09:36:54 -04:00
def get_bin_from_version ( version , bin_name , bin_default ) :
if not version :
return bin_default
2021-09-22 13:10:40 +02:00
if version > 219999 :
# Starting at client version 220000 the first two digits represent
# the major version, e.g. v22.0 instead of v0.22.0.
version * = 100
2020-04-30 09:36:54 -04:00
return os . path . join (
self . options . previous_releases_path ,
re . sub (
2021-09-22 13:10:40 +02:00
r ' \ .0$ ' if version < = 219999 else r ' ( \ .0) { 1,2}$ ' ,
' ' , # Remove trailing dot for point releases, after 22.0 also remove double trailing dot.
2020-04-30 09:36:54 -04:00
' v {} . {} . {} . {} ' . format (
( version % 100000000 ) / / 1000000 ,
( version % 1000000 ) / / 10000 ,
( version % 10000 ) / / 100 ,
( version % 100 ) / / 1 ,
) ,
) ,
' bin ' ,
bin_name ,
)
2018-03-06 16:48:15 -05: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
2019-01-05 20:20:42 +01:00
if versions is None :
versions = [ None ] * num_nodes
2021-10-01 13:53:59 +00:00
if self . is_syscall_sandbox_compiled ( ) and not self . disable_syscall_sandbox :
for i in range ( len ( extra_args ) ) :
2021-11-15 11:41:30 +01:00
# The -sandbox argument is not present in the v22.0 release.
if versions [ i ] is None or versions [ i ] > = 229900 :
2021-10-01 13:53:59 +00:00
extra_args [ i ] = extra_args [ i ] + [ " -sandbox=log-and-abort " ]
2017-06-02 13:14:14 -04:00
if binary is None :
2020-04-30 09:36:54 -04:00
binary = [ get_bin_from_version ( v , ' bitcoind ' , self . options . bitcoind ) for v in versions ]
2019-01-05 20:20:42 +01:00
if binary_cli is None :
2020-04-30 09:36:54 -04:00
binary_cli = [ get_bin_from_version ( v , ' bitcoin-cli ' , self . options . bitcoincli ) for v in versions ]
2018-03-06 16:48:15 -05:00
assert_equal ( len ( extra_confs ) , num_nodes )
2017-06-09 16:35:17 -04:00
assert_equal ( len ( extra_args ) , num_nodes )
2019-01-05 20:20:42 +01:00
assert_equal ( len ( versions ) , num_nodes )
2017-06-09 16:35:17 -04:00
assert_equal ( len ( binary ) , num_nodes )
2019-01-05 20:20:42 +01:00
assert_equal ( len ( binary_cli ) , num_nodes )
2017-06-09 16:35:17 -04:00
for i in range ( num_nodes ) :
2020-06-16 07:39:18 -04:00
test_node_i = TestNode (
2018-12-13 15:13:04 -05:00
i ,
get_datadir_path ( self . options . tmpdir , i ) ,
2019-07-31 14:11:32 -04:00
chain = self . chain ,
2018-12-13 15:13:04 -05:00
rpchost = rpchost ,
2018-12-22 17:22:55 +01:00
timewait = self . rpc_timeout ,
2020-05-18 09:45:55 +05:30
timeout_factor = self . options . timeout_factor ,
2018-12-13 15:13:04 -05:00
bitcoind = binary [ i ] ,
2019-01-05 20:20:42 +01:00
bitcoin_cli = binary_cli [ i ] ,
version = versions [ i ] ,
2018-12-13 15:13:04 -05:00
coverage_dir = self . options . coveragedir ,
2019-02-15 12:54:29 +01:00
cwd = self . options . tmpdir ,
2018-12-13 15:13:04 -05:00
extra_conf = extra_confs [ i ] ,
extra_args = extra_args [ i ] ,
use_cli = self . options . usecli ,
2018-10-19 12:28:47 -04:00
start_perf = self . options . perf ,
2019-11-29 15:11:18 +00:00
use_valgrind = self . options . valgrind ,
2019-07-16 15:33:35 -04:00
descriptors = self . options . descriptors ,
2020-06-16 07:39:18 -04:00
)
self . nodes . append ( test_node_i )
if not test_node_i . version_is_at_least ( 170000 ) :
# adjust conf for pre 17
2023-01-23 17:35:04 +01:00
test_node_i . replace_in_config ( [ ( ' [regtest] ' , ' ' ) ] )
2017-06-09 16:35:17 -04:00
2018-01-18 13:15:00 -05: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 13:15:00 -05: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 13:15:00 -05: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 13:15:00 -05: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 ( )
2022-08-18 20:23:15 +02:00
except Exception :
2017-06-02 13:14:14 -04:00
# 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 17:59:07 +00: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 17:59:07 +00:00
self . nodes [ i ] . stop_node ( expected_stderr , wait = wait )
2017-03-23 17:48:29 -04:00
2018-11-20 17:59:07 +00: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
2020-12-17 13:39:03 +01:00
node . stop_node ( wait = wait , wait_until_stopped = False )
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 08:52:24 -07:00
node . wait_until_stopped ( )
2017-06-02 13:14:14 -04:00
2017-09-20 15:00:28 +01: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 17:48:29 -04:00
2020-06-07 09:54:56 -04:00
def connect_nodes ( self , a , b ) :
2021-08-27 10:24:37 +02:00
from_connection = self . nodes [ a ]
to_connection = self . nodes [ b ]
2022-06-22 09:14:36 +02:00
from_num_peers = 1 + len ( from_connection . getpeerinfo ( ) )
to_num_peers = 1 + len ( to_connection . getpeerinfo ( ) )
2021-08-27 10:24:37 +02:00
ip_port = " 127.0.0.1: " + str ( p2p_port ( b ) )
from_connection . addnode ( ip_port , " onetry " )
# poll until version handshake complete to avoid race conditions
# with transaction relaying
# See comments in net_processing:
# * Must have a version message before anything else
# * Must have a verack message before anything else
2022-06-22 09:14:36 +02:00
self . wait_until ( lambda : sum ( peer [ ' version ' ] != 0 for peer in from_connection . getpeerinfo ( ) ) == from_num_peers )
self . wait_until ( lambda : sum ( peer [ ' version ' ] != 0 for peer in to_connection . getpeerinfo ( ) ) == to_num_peers )
self . wait_until ( lambda : sum ( peer [ ' bytesrecv_per_msg ' ] . pop ( ' verack ' , 0 ) == 24 for peer in from_connection . getpeerinfo ( ) ) == from_num_peers )
self . wait_until ( lambda : sum ( peer [ ' bytesrecv_per_msg ' ] . pop ( ' verack ' , 0 ) == 24 for peer in to_connection . getpeerinfo ( ) ) == to_num_peers )
2023-01-09 15:09:09 +01:00
# The message bytes are counted before processing the message, so make
# sure it was fully processed by waiting for a ping.
self . wait_until ( lambda : sum ( peer [ " bytesrecv_per_msg " ] . pop ( " pong " , 0 ) > = 32 for peer in from_connection . getpeerinfo ( ) ) == from_num_peers )
self . wait_until ( lambda : sum ( peer [ " bytesrecv_per_msg " ] . pop ( " pong " , 0 ) > = 32 for peer in to_connection . getpeerinfo ( ) ) == to_num_peers )
2020-06-07 09:54:56 -04:00
def disconnect_nodes ( self , a , b ) :
2022-09-20 16:00:43 +02:00
def disconnect_nodes_helper ( node_a , node_b ) :
def get_peer_ids ( from_connection , node_num ) :
2020-09-17 00:24:03 -07:00
result = [ ]
for peer in from_connection . getpeerinfo ( ) :
if " testnode {} " . format ( node_num ) in peer [ ' subver ' ] :
result . append ( peer [ ' id ' ] )
return result
2022-09-20 16:00:43 +02:00
peer_ids = get_peer_ids ( node_a , node_b . index )
2020-09-17 00:24:03 -07:00
if not peer_ids :
self . log . warning ( " disconnect_nodes: {} and {} were not connected " . format (
2022-09-20 16:00:43 +02:00
node_a . index ,
node_b . index ,
2020-09-17 00:24:03 -07:00
) )
return
for peer_id in peer_ids :
try :
2022-09-20 16:00:43 +02:00
node_a . disconnectnode ( nodeid = peer_id )
2020-09-17 00:24:03 -07:00
except JSONRPCException as e :
# If this node is disconnected between calculating the peer id
# and issuing the disconnect, don't worry about it.
# This avoids a race condition if we're mass-disconnecting peers.
if e . error [ ' code ' ] != - 29 : # RPC_CLIENT_NODE_NOT_CONNECTED
raise
# wait to disconnect
2022-09-20 16:00:43 +02:00
self . wait_until ( lambda : not get_peer_ids ( node_a , node_b . index ) , timeout = 5 )
self . wait_until ( lambda : not get_peer_ids ( node_b , node_a . index ) , timeout = 5 )
2020-09-17 00:24:03 -07:00
2022-09-20 16:00:43 +02:00
disconnect_nodes_helper ( self . nodes [ a ] , self . nodes [ b ] )
2020-06-07 09:54:56 -04:00
2017-03-23 17:48:29 -04:00
def split_network ( self ) :
"""
Split the network of four nodes into nodes 0 / 1 and 2 / 3.
"""
2020-06-07 09:54:56 -04:00
self . disconnect_nodes ( 1 , 2 )
2019-04-09 11:46:05 -04:00
self . sync_all ( self . nodes [ : 2 ] )
self . sync_all ( self . nodes [ 2 : ] )
2017-03-23 17:48:29 -04:00
def join_network ( self ) :
"""
Join the ( previously split ) network halves together .
"""
2020-06-07 09:54:56 -04:00
self . connect_nodes ( 1 , 2 )
2017-03-23 17:48:29 -04:00
self . sync_all ( )
2020-11-10 18:02:31 +01:00
def no_op ( self ) :
pass
def generate ( self , generator , * args , sync_fun = None , * * kwargs ) :
2021-07-27 13:59:55 +02:00
blocks = generator . generate ( * args , invalid_call = False , * * kwargs )
2020-11-10 18:02:31 +01:00
sync_fun ( ) if sync_fun else self . sync_all ( )
2021-07-27 13:26:11 +02:00
return blocks
2020-11-10 18:02:31 +01:00
def generateblock ( self , generator , * args , sync_fun = None , * * kwargs ) :
2021-07-27 13:59:55 +02:00
blocks = generator . generateblock ( * args , invalid_call = False , * * kwargs )
2020-11-10 18:02:31 +01:00
sync_fun ( ) if sync_fun else self . sync_all ( )
2021-07-27 13:26:11 +02:00
return blocks
2020-11-10 18:02:31 +01:00
def generatetoaddress ( self , generator , * args , sync_fun = None , * * kwargs ) :
2021-07-27 13:59:55 +02:00
blocks = generator . generatetoaddress ( * args , invalid_call = False , * * kwargs )
2020-11-10 18:02:31 +01:00
sync_fun ( ) if sync_fun else self . sync_all ( )
2021-07-27 13:26:11 +02:00
return blocks
2020-11-10 18:02:31 +01:00
def generatetodescriptor ( self , generator , * args , sync_fun = None , * * kwargs ) :
2021-07-27 13:59:55 +02:00
blocks = generator . generatetodescriptor ( * args , invalid_call = False , * * kwargs )
2020-11-10 18:02:31 +01:00
sync_fun ( ) if sync_fun else self . sync_all ( )
2021-07-27 13:26:11 +02:00
return blocks
2020-06-09 16:41:08 -04:00
def sync_blocks ( self , nodes = None , wait = 1 , timeout = 60 ) :
"""
Wait until everybody has the same tip .
sync_blocks needs to be called with an rpc_connections set that has least
one node already synced to the latest , stable tip , otherwise there ' s a
chance it might return before all nodes are stably synced .
"""
rpc_connections = nodes or self . nodes
timeout = int ( timeout * self . options . timeout_factor )
stop_time = time . time ( ) + timeout
while time . time ( ) < = stop_time :
best_hash = [ x . getbestblockhash ( ) for x in rpc_connections ]
if best_hash . count ( best_hash [ 0 ] ) == len ( rpc_connections ) :
return
# Check that each peer has at least one connection
assert ( all ( [ len ( x . getpeerinfo ( ) ) for x in rpc_connections ] ) )
time . sleep ( wait )
raise AssertionError ( " Block sync timed out after {} s: {} " . format (
timeout ,
" " . join ( " \n {!r} " . format ( b ) for b in best_hash ) ,
) )
def sync_mempools ( self , nodes = None , wait = 1 , timeout = 60 , flush_scheduler = True ) :
"""
Wait until everybody has the same transactions in their memory
pools
"""
rpc_connections = nodes or self . nodes
timeout = int ( timeout * self . options . timeout_factor )
stop_time = time . time ( ) + timeout
while time . time ( ) < = stop_time :
pool = [ set ( r . getrawmempool ( ) ) for r in rpc_connections ]
if pool . count ( pool [ 0 ] ) == len ( rpc_connections ) :
if flush_scheduler :
for r in rpc_connections :
r . syncwithvalidationinterfacequeue ( )
return
# Check that each peer has at least one connection
assert ( all ( [ len ( x . getpeerinfo ( ) ) for x in rpc_connections ] ) )
time . sleep ( wait )
raise AssertionError ( " Mempool sync timed out after {} s: {} " . format (
timeout ,
" " . join ( " \n {!r} " . format ( m ) for m in pool ) ,
) )
def sync_all ( self , nodes = None ) :
self . sync_blocks ( nodes )
self . sync_mempools ( nodes )
2017-03-23 17:48:29 -04:00
2020-08-27 11:52:52 +02:00
def wait_until ( self , test_function , timeout = 60 ) :
return wait_until_helper ( test_function , timeout = timeout , timeout_factor = self . options . timeout_factor )
2020-07-19 08:39:15 +02:00
2017-03-23 17:48:29 -04:00
# Private helper methods. These should not be accessed by the subclass test scripts.
2017-02-15 11:36:46 -05: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 11:36:46 -05: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 16:46:31 +01: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 10:05:59 -04:00
formatter . converter = time . gmtime
2017-02-15 11:36:46 -05: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 12:39:47 -04:00
2017-08-24 11:37:25 -04:00
def _initialize_chain ( self ) :
2017-03-23 16:49:02 -04:00
""" Initialize a pre-mined blockchain for use by the test.
2019-05-17 13:07:07 -04:00
Create a cache of a 199 - block - long chain
2017-03-23 16:49:02 -04:00
Afterward , create num_nodes copies from the cache . """
2019-05-17 13:07:07 -04:00
CACHE_NODE_ID = 0 # Use node 0 to create the cache for all other nodes
cache_node_dir = get_datadir_path ( self . options . cachedir , CACHE_NODE_ID )
2017-08-24 11:37:25 -04:00
assert self . num_nodes < = MAX_NODES
2019-05-17 13:07:07 -04:00
if not os . path . isdir ( cache_node_dir ) :
self . log . debug ( " Creating cache directory {} " . format ( cache_node_dir ) )
2021-07-26 17:36:01 +02:00
initialize_datadir ( self . options . cachedir , CACHE_NODE_ID , self . chain , self . disable_autoconnect )
2019-05-17 13:07:07 -04:00
self . nodes . append (
TestNode (
CACHE_NODE_ID ,
cache_node_dir ,
2019-07-31 14:11:32 -04:00
chain = self . chain ,
2018-12-22 17:33:48 +01:00
extra_conf = [ " bind=127.0.0.1 " ] ,
2019-05-17 13:07:07 -04:00
extra_args = [ ' -disablewallet ' ] ,
2018-12-22 17:33:48 +01:00
rpchost = None ,
timewait = self . rpc_timeout ,
2020-05-18 09:45:55 +05:30
timeout_factor = self . options . timeout_factor ,
2018-12-22 17:33:48 +01:00
bitcoind = self . options . bitcoind ,
bitcoin_cli = self . options . bitcoincli ,
coverage_dir = None ,
2019-02-15 12:54:29 +01:00
cwd = self . options . tmpdir ,
2019-07-16 15:33:35 -04:00
descriptors = self . options . descriptors ,
2018-12-22 17:33:48 +01:00
) )
2019-05-17 13:07:07 -04:00
self . start_node ( CACHE_NODE_ID )
2020-04-14 21:06:34 -04:00
cache_node = self . nodes [ CACHE_NODE_ID ]
2017-03-23 16:49:02 -04:00
2017-06-02 14:30:36 -04:00
# Wait for RPC connections to be ready
2020-04-14 21:06:34 -04:00
cache_node . wait_for_rpc_connection ( )
# Set a time in the past, so that blocks don't end up in the future
cache_node . setmocktime ( cache_node . getblockheader ( cache_node . getbestblockhash ( ) ) [ ' time ' ] )
2017-03-23 16:49:02 -04:00
2021-02-16 17:02:41 +01:00
# Create a 199-block-long chain; each of the 3 first nodes
2017-03-23 16:49:02 -04:00
# gets 25 mature blocks and 25 immature.
2021-02-16 17:02:41 +01:00
# The 4th address gets 25 mature and only 24 immature blocks so that the very last
2019-02-11 13:56:19 -05:00
# 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().
2021-10-27 11:18:57 +02:00
gen_addresses = [ k . address for k in TestNode . PRIV_KEYS ] [ : 3 ] + [ create_deterministic_address_bcrt1_p2tr_op_true ( ) [ 0 ] ]
2021-04-23 09:52:07 +02:00
assert_equal ( len ( gen_addresses ) , 4 )
2019-02-23 12:56:54 -05:00
for i in range ( 8 ) :
2021-07-27 20:04:51 +02:00
self . generatetoaddress (
cache_node ,
2019-05-17 13:07:07 -04:00
nblocks = 25 if i != 7 else 24 ,
2021-04-23 09:52:07 +02:00
address = gen_addresses [ i % len ( gen_addresses ) ] ,
2019-05-17 13:07:07 -04:00
)
2017-03-23 16:49:02 -04:00
2020-04-14 21:06:34 -04:00
assert_equal ( cache_node . getblockchaininfo ( ) [ " blocks " ] , 199 )
2019-02-11 13:56:19 -05:00
2019-05-17 13:07:07 -04:00
# Shut it down, and clean up cache directories:
2017-03-23 16:56:45 -04:00
self . stop_nodes ( )
self . nodes = [ ]
2018-03-07 13:49:44 -05:00
2019-05-17 13:07:07 -04:00
def cache_path ( * paths ) :
2019-07-31 14:11:32 -04:00
return os . path . join ( cache_node_dir , self . chain , * paths )
2018-03-07 13:49:44 -05:00
2019-05-17 13:07:07 -04:00
os . rmdir ( cache_path ( ' wallets ' ) ) # Remove empty wallets dir
for entry in os . listdir ( cache_path ( ) ) :
2019-08-26 15:49:57 -04:00
if entry not in [ ' chainstate ' , ' blocks ' , ' indexes ' ] : # Only indexes, chainstate and blocks folders
2019-05-17 13:07:07 -04:00
os . remove ( cache_path ( entry ) )
2017-08-24 11:37:25 -04:00
for i in range ( self . num_nodes ) :
2019-05-17 13:07:07 -04:00
self . log . debug ( " Copy cache directory {} to node {} " . format ( cache_node_dir , i ) )
2018-03-07 13:49:44 -05:00
to_dir = get_datadir_path ( self . options . tmpdir , i )
2019-05-17 13:07:07 -04:00
shutil . copytree ( cache_node_dir , to_dir )
2021-07-26 17:36:01 +02:00
initialize_datadir ( self . options . tmpdir , i , self . chain , self . disable_autoconnect ) # Overwrite port/rpcport in bitcoin.conf
2017-03-23 16:49:02 -04:00
2017-08-24 11:37:25 -04:00
def _initialize_chain_clean ( self ) :
2017-03-23 16:49:02 -04: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 11:37:25 -04:00
for i in range ( self . num_nodes ) :
2021-07-26 17:36:01 +02:00
initialize_datadir ( self . options . tmpdir , i , self . chain , self . disable_autoconnect )
2017-03-23 16:49:02 -04:00
2018-09-09 13:32:37 -04: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. " )
2023-01-12 13:26:05 +00:00
def skip_if_no_py_sqlite3 ( self ) :
""" Attempt to import the sqlite3 package and skip the test if the import fails. """
try :
import sqlite3 # noqa
except ImportError :
raise SkipTest ( " sqlite3 module not available. " )
2022-02-03 11:25:30 +01:00
def skip_if_no_python_bcc ( self ) :
""" Attempt to import the bcc package and skip the tests if the import fails. """
try :
import bcc # type: ignore[import] # noqa: F401
except ImportError :
raise SkipTest ( " bcc python module not available " )
def skip_if_no_bitcoind_tracepoints ( self ) :
""" Skip the running test if bitcoind has not been compiled with USDT tracepoint support. """
if not self . is_usdt_compiled ( ) :
raise SkipTest ( " bitcoind has not been built with USDT tracepoints enabled. " )
def skip_if_no_bpf_permissions ( self ) :
""" Skip the running test if we don ' t have permissions to do BPF syscalls and load BPF maps. """
# check for 'root' permissions
if os . geteuid ( ) != 0 :
raise SkipTest ( " no permissions to use BPF (please review the tests carefully before running them with higher privileges) " )
def skip_if_platform_not_linux ( self ) :
""" Skip the running test if we are not on a Linux platform """
if platform . system ( ) != " Linux " :
raise SkipTest ( " not on a Linux system " )
2023-02-06 11:08:57 +00:00
def skip_if_platform_not_posix ( self ) :
""" Skip the running test if we are not on a POSIX platform """
if os . name != ' posix ' :
raise SkipTest ( " not on a POSIX system " )
2018-09-09 13:32:37 -04:00
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. """
2022-11-10 10:12:39 +01:00
self . _requires_wallet = True
2018-09-09 13:32:37 -04:00
if not self . is_wallet_compiled ( ) :
raise SkipTest ( " wallet has not been compiled. " )
2020-10-29 12:28:07 -04:00
if self . options . descriptors :
self . skip_if_no_sqlite ( )
2020-10-29 19:41:05 -04:00
else :
self . skip_if_no_bdb ( )
2020-10-29 12:28:07 -04:00
def skip_if_no_sqlite ( self ) :
""" Skip the running test if sqlite has not been compiled. """
if not self . is_sqlite_compiled ( ) :
raise SkipTest ( " sqlite has not been compiled. " )
2018-09-09 13:32:37 -04:00
2020-11-23 11:23:31 +01:00
def skip_if_no_bdb ( self ) :
""" Skip the running test if BDB has not been compiled. """
if not self . is_bdb_compiled ( ) :
raise SkipTest ( " BDB has not been compiled. " )
2019-11-18 16:48:52 -05:00
def skip_if_no_wallet_tool ( self ) :
""" Skip the running test if bitcoin-wallet has not been compiled. """
if not self . is_wallet_tool_compiled ( ) :
raise SkipTest ( " bitcoin-wallet has not been compiled " )
2022-04-11 21:39:13 +02:00
def skip_if_no_bitcoin_util ( self ) :
""" Skip the running test if bitcoin-util has not been compiled. """
if not self . is_bitcoin_util_compiled ( ) :
raise SkipTest ( " bitcoin-util has not been compiled " )
2018-09-09 13:32:37 -04:00
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. " )
2020-04-30 09:36:54 -04:00
def skip_if_no_previous_releases ( self ) :
""" Skip the running test if previous releases are not available. """
if not self . has_previous_releases ( ) :
raise SkipTest ( " previous releases not available or disabled " )
def has_previous_releases ( self ) :
""" Checks whether previous releases are present and enabled. """
if not os . path . isdir ( self . options . previous_releases_path ) :
2020-05-19 07:12:22 -04:00
if self . options . prev_releases :
raise AssertionError ( " Force test of previous releases but releases missing: {} " . format (
2020-04-30 09:36:54 -04:00
self . options . previous_releases_path ) )
2020-05-19 07:12:22 -04:00
return self . options . prev_releases
2020-04-30 09:36:54 -04:00
2019-02-13 18:59:16 +01:00
def skip_if_no_external_signer ( self ) :
""" Skip the running test if external signer support has not been compiled. """
if not self . is_external_signer_compiled ( ) :
raise SkipTest ( " external signer support has not been compiled. " )
2018-09-09 13:32:37 -04:00
def is_cli_compiled ( self ) :
""" Checks whether bitcoin-cli was compiled. """
2019-04-25 09:12:36 +00:00
return self . config [ " components " ] . getboolean ( " ENABLE_CLI " )
2018-07-12 11:52:02 -04:00
2019-02-13 18:59:16 +01:00
def is_external_signer_compiled ( self ) :
""" Checks whether external signer support was compiled. """
return self . config [ " components " ] . getboolean ( " ENABLE_EXTERNAL_SIGNER " )
2018-09-09 13:32:37 -04:00
def is_wallet_compiled ( self ) :
""" Checks whether the wallet module was compiled. """
2019-04-25 09:12:36 +00:00
return self . config [ " components " ] . getboolean ( " ENABLE_WALLET " )
2018-08-31 22:30:26 +02:00
2021-12-08 19:57:43 +01:00
def is_specified_wallet_compiled ( self ) :
""" Checks whether wallet support for the specified type
( legacy or descriptor wallet ) was compiled . """
if self . options . descriptors :
return self . is_sqlite_compiled ( )
else :
return self . is_bdb_compiled ( )
2019-11-18 16:48:52 -05:00
def is_wallet_tool_compiled ( self ) :
""" Checks whether bitcoin-wallet was compiled. """
return self . config [ " components " ] . getboolean ( " ENABLE_WALLET_TOOL " )
2022-04-11 21:39:13 +02:00
def is_bitcoin_util_compiled ( self ) :
""" Checks whether bitcoin-util was compiled. """
return self . config [ " components " ] . getboolean ( " ENABLE_BITCOIN_UTIL " )
2018-09-09 13:32:37 -04:00
def is_zmq_compiled ( self ) :
""" Checks whether the zmq module was compiled. """
2019-04-25 09:12:36 +00:00
return self . config [ " components " ] . getboolean ( " ENABLE_ZMQ " )
2020-10-29 12:28:07 -04:00
2022-02-03 11:25:30 +01:00
def is_usdt_compiled ( self ) :
""" Checks whether the USDT tracepoints were compiled. """
return self . config [ " components " ] . getboolean ( " ENABLE_USDT_TRACEPOINTS " )
2020-10-29 12:28:07 -04:00
def is_sqlite_compiled ( self ) :
2020-11-23 11:23:31 +01:00
""" Checks whether the wallet module was compiled with Sqlite support. """
2020-10-29 12:28:07 -04:00
return self . config [ " components " ] . getboolean ( " USE_SQLITE " )
2020-11-23 11:23:31 +01:00
def is_bdb_compiled ( self ) :
""" Checks whether the wallet module was compiled with BDB support. """
return self . config [ " components " ] . getboolean ( " USE_BDB " )
2021-10-01 13:53:59 +00:00
def is_syscall_sandbox_compiled ( self ) :
""" Checks whether the syscall sandbox was compiled. """
return self . config [ " components " ] . getboolean ( " ENABLE_SYSCALL_SANDBOX " )