2016-03-19 20:58:06 +01:00
#!/usr/bin/env python3
# Copyright (c) 2014-2016 The Bitcoin Core developers
2015-08-26 12:05:36 +02:00
# Distributed under the MIT software license, see the accompanying
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
2017-03-09 15:53:26 +01:00
""" Run regression test suite.
2015-10-11 07:41:19 +02:00
This module calls down into individual test cases via subprocess . It will
2017-01-31 20:32:49 +01:00
forward all unrecognized arguments onto the individual test scripts .
2015-10-11 07:41:19 +02:00
2017-03-09 15:44:57 +01:00
Functional tests are disabled on Windows by default . Use - - force to run them anyway .
2015-10-11 07:41:19 +02:00
For a description of arguments recognized by test scripts , see
2017-03-09 15:53:26 +01:00
` test / functional / test_framework / test_framework . py : BitcoinTestFramework . main ` .
2015-10-11 07:41:19 +02:00
"""
2015-08-26 12:05:36 +02:00
2017-01-31 19:15:40 +01:00
import argparse
2017-01-30 23:57:27 +01:00
import configparser
2017-05-18 23:33:33 +02:00
import datetime
2015-08-26 12:05:36 +02:00
import os
2015-11-30 14:53:07 +01:00
import time
2015-10-11 07:41:19 +02:00
import shutil
2017-06-08 15:33:52 +02:00
import signal
2015-08-26 12:05:36 +02:00
import sys
import subprocess
2015-10-11 07:41:19 +02:00
import tempfile
2015-08-26 12:05:36 +02:00
import re
2017-02-16 20:00:35 +01:00
import logging
2015-10-11 07:41:19 +02:00
2017-07-25 21:39:39 +02:00
# Formatting. Default colors to empty strings.
BOLD , BLUE , RED , GREY = ( " " , " " ) , ( " " , " " ) , ( " " , " " ) , ( " " , " " )
2017-04-17 19:46:20 +02:00
try :
# Make sure python thinks it can write unicode to its stdout
" \u2713 " . encode ( " utf_8 " ) . decode ( sys . stdout . encoding )
TICK = " ✓ "
CROSS = " ✖ "
CIRCLE = " ○ "
except UnicodeDecodeError :
TICK = " P "
CROSS = " x "
CIRCLE = " o "
2017-04-05 21:19:26 +02:00
if os . name == ' posix ' :
# primitive formatting on supported
# terminal via ANSI escape sequences:
BOLD = ( ' \033 [0m ' , ' \033 [1m ' )
2017-04-20 20:56:37 +02:00
BLUE = ( ' \033 [0m ' , ' \033 [0;34m ' )
2017-07-25 21:39:39 +02:00
RED = ( ' \033 [0m ' , ' \033 [0;31m ' )
GREY = ( ' \033 [0m ' , ' \033 [1;30m ' )
2017-04-05 21:19:26 +02:00
2017-03-22 15:26:02 +01:00
TEST_EXIT_PASSED = 0
TEST_EXIT_SKIPPED = 77
2017-01-31 19:15:40 +01:00
BASE_SCRIPTS = [
2017-02-17 20:22:56 +01:00
# Scripts that are run by the travis build process.
# Longest test should go first, to favor running tests in parallel
2016-12-03 21:46:33 +01:00
' wallet-hd.py ' ,
2016-04-29 12:51:15 +02:00
' walletbackup.py ' ,
2016-12-03 21:46:33 +01:00
# vv Tests less than 5m vv
' p2p-fullblocktest.py ' ,
' fundrawtransaction.py ' ,
' p2p-compactblocks.py ' ,
' segwit.py ' ,
# vv Tests less than 2m vv
2015-08-26 12:05:36 +02:00
' wallet.py ' ,
2016-08-04 04:28:39 +02:00
' wallet-accounts.py ' ,
2016-12-03 21:46:33 +01:00
' p2p-segwit.py ' ,
2016-07-28 14:59:13 +02:00
' wallet-dump.py ' ,
2015-08-26 12:05:36 +02:00
' listtransactions.py ' ,
2016-12-03 21:46:33 +01:00
# vv Tests less than 60s vv
' sendheaders.py ' ,
' zapwallettxes.py ' ,
' importmulti.py ' ,
' mempool_limit.py ' ,
' merkle_blocks.py ' ,
2015-11-14 21:44:35 +01:00
' receivedby.py ' ,
2016-12-03 21:46:33 +01:00
' abandonconflict.py ' ,
' bip68-112-113-p2p.py ' ,
' rawtransactions.py ' ,
' reindex.py ' ,
# vv Tests less than 30s vv
2017-05-03 16:51:51 +02:00
' keypool-topup.py ' ,
2017-06-07 22:14:20 +02:00
' zmq_test.py ' ,
2017-07-11 19:02:01 +02:00
' bitcoin_cli.py ' ,
2015-08-26 12:05:36 +02:00
' mempool_resurrect_test.py ' ,
' txn_doublespend.py --mineblock ' ,
' txn_clone.py ' ,
' getchaintips.py ' ,
' rest.py ' ,
' mempool_spendcoinbase.py ' ,
2015-08-27 03:15:04 +02:00
' mempool_reorg.py ' ,
2017-03-09 23:14:55 +01:00
' mempool_persist.py ' ,
2017-06-15 15:05:32 +02:00
' multiwallet.py ' ,
2015-08-26 12:05:36 +02:00
' httpbasics.py ' ,
2015-11-11 16:49:32 +01:00
' multi_rpc.py ' ,
2015-08-26 12:05:36 +02:00
' proxy_test.py ' ,
' signrawtransactions.py ' ,
2017-04-19 17:14:53 +02:00
' disconnect_ban.py ' ,
2015-08-26 12:05:36 +02:00
' decodescript.py ' ,
2015-10-08 10:22:50 +02:00
' blockchain.py ' ,
2017-08-23 16:58:59 +02:00
' deprecated_rpc.py ' ,
2015-11-09 08:40:46 +01:00
' disablewallet.py ' ,
2017-03-25 15:29:56 +01:00
' net.py ' ,
2015-11-30 14:53:07 +01:00
' keypool.py ' ,
2016-06-30 14:49:59 +02:00
' p2p-mempool.py ' ,
2015-11-19 02:55:52 +01:00
' prioritise_transaction.py ' ,
2015-12-08 17:10:41 +01:00
' invalidblockrequest.py ' ,
' invalidtxrequest.py ' ,
2016-03-15 17:09:16 +01:00
' p2p-versionbits-warning.py ' ,
2016-08-26 23:05:26 +02:00
' preciousblock.py ' ,
2016-02-19 01:31:12 +01:00
' importprunedfunds.py ' ,
2016-04-29 12:51:15 +02:00
' signmessages.py ' ,
2016-08-31 13:38:23 +02:00
' nulldummy.py ' ,
2016-12-12 16:18:11 +01:00
' import-rescan.py ' ,
2017-04-11 22:47:10 +02:00
' mining.py ' ,
2016-12-09 19:45:27 +01:00
' bumpfee.py ' ,
2016-09-25 21:09:13 +02:00
' rpcnamedargs.py ' ,
2017-01-11 12:02:25 +01:00
' listsinceblock.py ' ,
2017-02-08 07:17:58 +01:00
' p2p-leaktests.py ' ,
2017-06-07 21:32:45 +02:00
' wallet-encryption.py ' ,
2017-06-28 22:56:37 +02:00
' bipdersig-p2p.py ' ,
2017-06-28 20:52:24 +02:00
' bip65-cltv-p2p.py ' ,
2017-05-14 20:18:26 +02:00
' uptime.py ' ,
2017-08-07 16:49:41 +02:00
' resendwallettransactions.py ' ,
2017-05-08 15:59:00 +02:00
' minchainwork.py ' ,
2015-08-26 12:05:36 +02:00
]
2016-04-09 22:17:52 +02:00
2017-01-31 19:15:40 +01:00
EXTENDED_SCRIPTS = [
2017-02-06 15:07:14 +01:00
# These tests are not run by the travis build process.
# Longest test should go first, to favor running tests in parallel
2016-12-03 21:46:33 +01:00
' pruning.py ' ,
# vv Tests less than 20m vv
' smartfees.py ' ,
# vv Tests less than 5m vv
' maxuploadtarget.py ' ,
' mempool_packages.py ' ,
2017-06-15 23:08:48 +02:00
' dbcrash.py ' ,
2016-12-03 21:46:33 +01:00
# vv Tests less than 2m vv
' bip68-sequence.py ' ,
' getblocktemplate_longpoll.py ' ,
2017-02-07 23:43:36 +01:00
' p2p-timeouts.py ' ,
2016-12-03 21:46:33 +01:00
# vv Tests less than 60s vv
2016-03-16 06:30:04 +01:00
' bip9-softforks.py ' ,
2016-12-03 21:46:33 +01:00
' p2p-feefilter.py ' ,
' rpcbind_test.py ' ,
# vv Tests less than 30s vv
2017-02-24 00:00:33 +01:00
' assumevalid.py ' ,
2017-06-15 21:29:53 +02:00
' example_test.py ' ,
2015-08-26 12:05:36 +02:00
' txn_doublespend.py ' ,
' txn_clone.py --mineblock ' ,
' forknotify.py ' ,
' invalidateblock.py ' ,
' p2p-acceptblock.py ' ,
2015-10-30 19:55:32 +01:00
' replace-by-fee.py ' ,
2015-08-26 12:05:36 +02:00
]
2017-04-18 00:20:35 +02:00
# Place EXTENDED_SCRIPTS first since it has the 3 longest running tests
2017-03-24 23:36:55 +01:00
ALL_SCRIPTS = EXTENDED_SCRIPTS + BASE_SCRIPTS
2016-04-29 12:51:15 +02:00
2017-03-27 17:33:00 +02:00
NON_SCRIPTS = [
# These are python files that live in the functional tests directory, but are not test scripts.
" combine_logs.py " ,
" create_cache.py " ,
" test_runner.py " ,
]
2017-02-06 15:07:14 +01:00
def main ( ) :
# Parse arguments and pass through unrecognised args
parser = argparse . ArgumentParser ( add_help = False ,
2017-03-09 15:53:26 +01:00
usage = ' %(prog)s [test_runner.py options] [script options] [scripts] ' ,
2017-02-06 15:07:14 +01:00
description = __doc__ ,
epilog = '''
Help text and arguments for individual test script : ''' ,
formatter_class = argparse . RawTextHelpFormatter )
parser . add_argument ( ' --coverage ' , action = ' store_true ' , help = ' generate a basic coverage report for the RPC interface ' )
2017-08-16 00:24:39 +02:00
parser . add_argument ( ' --exclude ' , ' -x ' , help = ' specify a comma-separated-list of scripts to exclude. ' )
2017-02-06 15:07:14 +01:00
parser . add_argument ( ' --extended ' , action = ' store_true ' , help = ' run the extended test suite in addition to the basic tests ' )
2017-02-17 20:22:56 +01:00
parser . add_argument ( ' --force ' , ' -f ' , action = ' store_true ' , help = ' run tests even on platforms where they are disabled by default (e.g. windows). ' )
2017-02-06 15:07:14 +01:00
parser . add_argument ( ' --help ' , ' -h ' , ' -? ' , action = ' store_true ' , help = ' print help text and exit ' )
parser . add_argument ( ' --jobs ' , ' -j ' , type = int , default = 4 , help = ' how many test scripts to run in parallel. Default=4. ' )
2017-04-12 17:01:31 +02:00
parser . add_argument ( ' --keepcache ' , ' -k ' , action = ' store_true ' , help = ' the default behavior is to flush the cache directory on startup. --keepcache retains the cache from the previous testrun. ' )
2017-02-16 20:00:35 +01:00
parser . add_argument ( ' --quiet ' , ' -q ' , action = ' store_true ' , help = ' only print results summary and failure logs ' )
2017-05-18 23:33:33 +02:00
parser . add_argument ( ' --tmpdirprefix ' , ' -t ' , default = tempfile . gettempdir ( ) , help = " Root directory for datadirs " )
2017-02-17 20:22:56 +01:00
args , unknown_args = parser . parse_known_args ( )
2017-02-06 15:07:14 +01:00
2017-05-09 16:39:37 +02:00
# args to be passed on always start with two dashes; tests are the remaining unknown args
tests = [ arg for arg in unknown_args if arg [ : 2 ] != " -- " ]
2017-02-06 15:07:14 +01:00
passon_args = [ arg for arg in unknown_args if arg [ : 2 ] == " -- " ]
# Read config generated by configure.
config = configparser . ConfigParser ( )
2017-03-21 19:47:20 +01:00
configfile = os . path . abspath ( os . path . dirname ( __file__ ) ) + " /../config.ini "
2017-05-01 21:12:49 +02:00
config . read_file ( open ( configfile ) )
passon_args . append ( " --configfile= %s " % configfile )
2017-02-06 15:07:14 +01:00
2017-02-16 20:00:35 +01:00
# Set up logging
logging_level = logging . INFO if args . quiet else logging . DEBUG
logging . basicConfig ( format = ' %(message)s ' , level = logging_level )
2017-05-18 23:33:33 +02:00
# Create base test directory
tmpdir = " %s /bitcoin_test_runner_ %s " % ( args . tmpdirprefix , datetime . datetime . now ( ) . strftime ( " % Y % m %d _ % H % M % S " ) )
os . makedirs ( tmpdir )
logging . debug ( " Temporary test directory at %s " % tmpdir )
2017-02-17 20:22:56 +01:00
enable_wallet = config [ " components " ] . getboolean ( " ENABLE_WALLET " )
enable_utils = config [ " components " ] . getboolean ( " ENABLE_UTILS " )
enable_bitcoind = config [ " components " ] . getboolean ( " ENABLE_BITCOIND " )
2017-02-06 15:07:14 +01:00
2017-02-17 20:22:56 +01:00
if config [ " environment " ] [ " EXEEXT " ] == " .exe " and not args . force :
2017-02-06 15:07:14 +01:00
# https://github.com/bitcoin/bitcoin/commit/d52802551752140cf41f0d9a225a43e84404d3e9
# https://github.com/bitcoin/bitcoin/pull/5677#issuecomment-136646964
2017-02-17 20:22:56 +01:00
print ( " Tests currently disabled on Windows by default. Use --force option to enable " )
2017-02-06 15:07:14 +01:00
sys . exit ( 0 )
2016-04-29 12:51:15 +02:00
2017-02-06 15:07:14 +01:00
if not ( enable_wallet and enable_utils and enable_bitcoind ) :
2017-03-09 15:44:57 +01:00
print ( " No functional tests to run. Wallet, utils, and bitcoind must all be enabled " )
2017-02-06 15:07:14 +01:00
print ( " Rerun `configure` with -enable-wallet, -with-utils and -with-daemon and rerun make " )
sys . exit ( 0 )
2017-01-31 19:15:40 +01:00
# Build list of tests
2017-02-17 20:22:56 +01:00
if tests :
2017-01-31 19:15:40 +01:00
# Individual tests have been specified. Run specified tests that exist
# in the ALL_SCRIPTS list. Accept the name with or without .py extension.
2017-05-09 16:39:37 +02:00
tests = [ re . sub ( " \ .py$ " , " " , t ) + " .py " for t in tests ]
test_list = [ ]
for t in tests :
if t in ALL_SCRIPTS :
test_list . append ( t )
else :
print ( " {} WARNING! {} Test ' {} ' not found in full test list. " . format ( BOLD [ 1 ] , BOLD [ 0 ] , t ) )
2016-04-27 22:29:52 +02:00
else :
2017-03-24 23:36:55 +01:00
# No individual tests have been specified.
# Run all base tests, and optionally run extended tests.
2017-01-31 19:15:40 +01:00
test_list = BASE_SCRIPTS
if args . extended :
2017-04-18 00:20:35 +02:00
# place the EXTENDED_SCRIPTS first since the three longest ones
# are there and the list is shorter
test_list = EXTENDED_SCRIPTS + test_list
2017-01-31 19:15:40 +01:00
2017-02-15 15:59:19 +01:00
# Remove the test cases that the user has explicitly asked to exclude.
if args . exclude :
2017-05-09 16:39:37 +02:00
tests_excl = [ re . sub ( " \ .py$ " , " " , t ) + " .py " for t in args . exclude . split ( ' , ' ) ]
for exclude_test in tests_excl :
if exclude_test in test_list :
test_list . remove ( exclude_test )
else :
print ( " {} WARNING! {} Test ' {} ' not found in current test list. " . format ( BOLD [ 1 ] , BOLD [ 0 ] , exclude_test ) )
2017-02-15 15:59:19 +01:00
if not test_list :
print ( " No valid test scripts specified. Check that your test is in one "
2017-03-09 15:53:26 +01:00
" of the test lists in test_runner.py, or run test_runner.py with no arguments to run all tests " )
2017-02-15 15:59:19 +01:00
sys . exit ( 0 )
2017-01-31 19:15:40 +01:00
if args . help :
2017-03-23 18:54:18 +01:00
# Print help for test_runner.py, then print help of the first script (with args removed) and exit.
2017-01-31 20:32:49 +01:00
parser . print_help ( )
2017-03-23 18:54:18 +01:00
subprocess . check_call ( [ ( config [ " environment " ] [ " SRCDIR " ] + ' /test/functional/ ' + test_list [ 0 ] . split ( ) [ 0 ] ) ] + [ ' -h ' ] )
2016-04-27 22:29:52 +02:00
sys . exit ( 0 )
2017-03-27 17:33:00 +02:00
check_script_list ( config [ " environment " ] [ " SRCDIR " ] )
2017-11-30 11:11:18 +01:00
check_script_prefixes ( )
2017-03-27 17:33:00 +02:00
2017-04-12 17:01:31 +02:00
if not args . keepcache :
shutil . rmtree ( " %s /test/cache " % config [ " environment " ] [ " BUILDDIR " ] , ignore_errors = True )
2017-05-18 23:33:33 +02:00
run_tests ( test_list , config [ " environment " ] [ " SRCDIR " ] , config [ " environment " ] [ " BUILDDIR " ] , config [ " environment " ] [ " EXEEXT " ] , tmpdir , args . jobs , args . coverage , passon_args )
2017-02-06 15:07:14 +01:00
2017-05-18 23:33:33 +02:00
def run_tests ( test_list , src_dir , build_dir , exeext , tmpdir , jobs = 1 , enable_coverage = False , args = [ ] ) :
2017-04-12 16:41:13 +02:00
# Warn if bitcoind is already running (unix only)
try :
if subprocess . check_output ( [ " pidof " , " bitcoind " ] ) is not None :
print ( " %s WARNING! %s There is already a bitcoind process running on this system. Tests may fail unexpectedly due to resource contention! " % ( BOLD [ 1 ] , BOLD [ 0 ] ) )
except ( OSError , subprocess . SubprocessError ) :
pass
# Warn if there is a cache directory
cache_dir = " %s /test/cache " % build_dir
if os . path . isdir ( cache_dir ) :
print ( " %s WARNING! %s There is a cache directory here: %s . If tests fail unexpectedly, try deleting the cache directory. " % ( BOLD [ 1 ] , BOLD [ 0 ] , cache_dir ) )
2017-02-06 15:07:14 +01:00
#Set env vars
if " BITCOIND " not in os . environ :
os . environ [ " BITCOIND " ] = build_dir + ' /src/bitcoind ' + exeext
2017-07-11 19:01:44 +02:00
os . environ [ " BITCOINCLI " ] = build_dir + ' /src/bitcoin-cli ' + exeext
2017-02-06 15:07:14 +01:00
2017-03-09 15:44:57 +01:00
tests_dir = src_dir + ' /test/functional/ '
2015-10-11 07:41:19 +02:00
2017-02-22 10:50:44 +01:00
flags = [ " --srcdir= {} /src " . format ( build_dir ) ] + args
2017-04-12 16:41:13 +02:00
flags . append ( " --cachedir= %s " % cache_dir )
2017-02-06 15:07:14 +01:00
if enable_coverage :
2015-10-11 07:41:19 +02:00
coverage = RPCCoverage ( )
2016-04-29 12:51:15 +02:00
flags . append ( coverage . flag )
2017-02-16 20:00:35 +01:00
logging . debug ( " Initializing coverage directory at %s " % coverage . dir )
2017-02-06 15:07:14 +01:00
else :
coverage = None
2016-04-29 12:51:15 +02:00
2017-01-31 20:32:49 +01:00
if len ( test_list ) > 1 and jobs > 1 :
2016-04-29 12:51:15 +02:00
# Populate cache
2017-05-18 23:33:33 +02:00
subprocess . check_output ( [ tests_dir + ' create_cache.py ' ] + flags + [ " --tmpdir= %s /cache " % tmpdir ] )
2016-04-09 22:17:52 +02:00
#Run Tests
2017-05-18 23:33:33 +02:00
job_queue = TestHandler ( jobs , tests_dir , tmpdir , test_list , flags )
2017-04-05 21:19:26 +02:00
time0 = time . time ( )
test_results = [ ]
2017-02-06 15:07:14 +01:00
max_len_name = len ( max ( test_list , key = len ) )
2017-04-05 21:19:26 +02:00
2016-04-29 12:51:15 +02:00
for _ in range ( len ( test_list ) ) :
2017-04-05 21:19:26 +02:00
test_result , stdout , stderr = job_queue . get_next ( )
test_results . append ( test_result )
2017-07-25 21:39:39 +02:00
if test_result . status == " Passed " :
2017-04-05 21:19:26 +02:00
logging . debug ( " \n %s %s %s passed, Duration: %s s " % ( BOLD [ 1 ] , test_result . name , BOLD [ 0 ] , test_result . time ) )
2017-07-25 21:39:39 +02:00
elif test_result . status == " Skipped " :
2017-04-05 21:19:26 +02:00
logging . debug ( " \n %s %s %s skipped " % ( BOLD [ 1 ] , test_result . name , BOLD [ 0 ] ) )
2017-02-16 20:00:35 +01:00
else :
2017-04-05 21:19:26 +02:00
print ( " \n %s %s %s failed, Duration: %s s \n " % ( BOLD [ 1 ] , test_result . name , BOLD [ 0 ] , test_result . time ) )
2017-02-16 20:00:35 +01:00
print ( BOLD [ 1 ] + ' stdout: \n ' + BOLD [ 0 ] + stdout + ' \n ' )
print ( BOLD [ 1 ] + ' stderr: \n ' + BOLD [ 0 ] + stderr + ' \n ' )
2017-02-06 15:07:14 +01:00
2017-04-05 21:19:26 +02:00
print_results ( test_results , max_len_name , ( int ( time . time ( ) - time0 ) ) )
2016-04-09 22:17:52 +02:00
if coverage :
coverage . report_rpc_coverage ( )
2017-02-16 20:00:35 +01:00
logging . debug ( " Cleaning up coverage data " )
2016-04-09 22:17:52 +02:00
coverage . cleanup ( )
2015-10-11 07:41:19 +02:00
2017-05-18 23:33:33 +02:00
# Clear up the temp directory if all subdirectories are gone
if not os . listdir ( tmpdir ) :
os . rmdir ( tmpdir )
2017-04-11 19:40:54 +02:00
all_passed = all ( map ( lambda test_result : test_result . was_successful , test_results ) )
2017-04-05 21:19:26 +02:00
2016-04-29 12:51:15 +02:00
sys . exit ( not all_passed )
2017-04-05 21:19:26 +02:00
def print_results ( test_results , max_len_name , runtime ) :
2017-07-25 21:39:39 +02:00
results = " \n " + BOLD [ 1 ] + " %s | %s | %s \n \n " % ( " TEST " . ljust ( max_len_name ) , " STATUS " , " DURATION " ) + BOLD [ 0 ]
2017-04-05 21:19:26 +02:00
test_results . sort ( key = lambda result : result . name . lower ( ) )
all_passed = True
time_sum = 0
for test_result in test_results :
2017-04-11 19:40:54 +02:00
all_passed = all_passed and test_result . was_successful
2017-04-05 21:19:26 +02:00
time_sum + = test_result . time
test_result . padding = max_len_name
results + = str ( test_result )
2017-04-07 15:15:29 +02:00
status = TICK + " Passed " if all_passed else CROSS + " Failed "
2017-07-25 21:39:39 +02:00
results + = BOLD [ 1 ] + " \n %s | %s | %s s (accumulated) \n " % ( " ALL " . ljust ( max_len_name ) , status . ljust ( 9 ) , time_sum ) + BOLD [ 0 ]
2017-04-05 21:19:26 +02:00
results + = " Runtime: %s s \n " % ( runtime )
print ( results )
2017-03-09 15:44:57 +01:00
class TestHandler :
2016-04-29 12:51:15 +02:00
"""
2017-07-01 06:00:51 +02:00
Trigger the test scripts passed in via the list .
2016-04-29 12:51:15 +02:00
"""
2017-05-18 23:33:33 +02:00
def __init__ ( self , num_tests_parallel , tests_dir , tmpdir , test_list = None , flags = None ) :
2016-04-29 12:51:15 +02:00
assert ( num_tests_parallel > = 1 )
self . num_jobs = num_tests_parallel
2017-02-06 15:07:14 +01:00
self . tests_dir = tests_dir
2017-05-18 23:33:33 +02:00
self . tmpdir = tmpdir
2016-04-29 12:51:15 +02:00
self . test_list = test_list
self . flags = flags
self . num_running = 0
2016-11-07 22:33:22 +01:00
# In case there is a graveyard of zombie bitcoinds, we can apply a
# pseudorandom offset to hopefully jump over them.
# (625 is PORT_RANGE/MAX_NODES)
self . portseed_offset = int ( time . time ( ) * 1000 ) % 625
2016-04-29 12:51:15 +02:00
self . jobs = [ ]
def get_next ( self ) :
while self . num_running < self . num_jobs and self . test_list :
# Add tests
self . num_running + = 1
t = self . test_list . pop ( 0 )
2017-05-18 23:33:33 +02:00
portseed = len ( self . test_list ) + self . portseed_offset
portseed_arg = [ " --portseed= {} " . format ( portseed ) ]
2016-09-17 11:47:51 +02:00
log_stdout = tempfile . SpooledTemporaryFile ( max_size = 2 * * 16 )
log_stderr = tempfile . SpooledTemporaryFile ( max_size = 2 * * 16 )
2017-03-23 18:54:18 +01:00
test_argv = t . split ( )
2017-05-18 23:33:33 +02:00
tmpdir = [ " --tmpdir= %s / %s _ %s " % ( self . tmpdir , re . sub ( " .py$ " , " " , test_argv [ 0 ] ) , portseed ) ]
2016-04-29 12:51:15 +02:00
self . jobs . append ( ( t ,
time . time ( ) ,
2017-05-18 23:33:33 +02:00
subprocess . Popen ( [ self . tests_dir + test_argv [ 0 ] ] + test_argv [ 1 : ] + self . flags + portseed_arg + tmpdir ,
2016-04-29 12:51:15 +02:00
universal_newlines = True ,
2016-09-17 11:47:51 +02:00
stdout = log_stdout ,
stderr = log_stderr ) ,
log_stdout ,
log_stderr ) )
2016-04-29 12:51:15 +02:00
if not self . jobs :
2016-05-09 21:29:18 +02:00
raise IndexError ( ' pop from empty list ' )
2016-04-29 12:51:15 +02:00
while True :
# Return first proc that finishes
time . sleep ( .5 )
for j in self . jobs :
2016-09-17 11:47:51 +02:00
( name , time0 , proc , log_out , log_err ) = j
2017-06-08 15:33:52 +02:00
if os . getenv ( ' TRAVIS ' ) == ' true ' and int ( time . time ( ) - time0 ) > 20 * 60 :
# In travis, timeout individual tests after 20 minutes (to stop tests hanging and not
# providing useful output.
proc . send_signal ( signal . SIGINT )
2016-04-29 12:51:15 +02:00
if proc . poll ( ) is not None :
2016-09-17 11:47:51 +02:00
log_out . seek ( 0 ) , log_err . seek ( 0 )
[ stdout , stderr ] = [ l . read ( ) . decode ( ' utf-8 ' ) for l in ( log_out , log_err ) ]
log_out . close ( ) , log_err . close ( )
2017-03-22 15:26:02 +01:00
if proc . returncode == TEST_EXIT_PASSED and stderr == " " :
2017-07-25 21:39:39 +02:00
status = " Passed "
2017-03-22 15:26:02 +01:00
elif proc . returncode == TEST_EXIT_SKIPPED :
2017-07-25 21:39:39 +02:00
status = " Skipped "
2017-03-22 15:26:02 +01:00
else :
2017-07-25 21:39:39 +02:00
status = " Failed "
2016-04-29 12:51:15 +02:00
self . num_running - = 1
self . jobs . remove ( j )
2017-04-05 21:19:26 +02:00
return TestResult ( name , status , int ( time . time ( ) - time0 ) ) , stdout , stderr
2016-04-29 12:51:15 +02:00
print ( ' . ' , end = ' ' , flush = True )
2017-04-05 21:19:26 +02:00
class TestResult ( ) :
def __init__ ( self , name , status , time ) :
self . name = name
self . status = status
self . time = time
self . padding = 0
def __repr__ ( self ) :
2017-07-25 21:39:39 +02:00
if self . status == " Passed " :
2017-04-07 15:15:29 +02:00
color = BLUE
glyph = TICK
2017-07-25 21:39:39 +02:00
elif self . status == " Failed " :
2017-04-20 20:56:37 +02:00
color = RED
glyph = CROSS
2017-07-25 21:39:39 +02:00
elif self . status == " Skipped " :
color = GREY
glyph = CIRCLE
2017-04-06 19:31:47 +02:00
2017-07-25 21:39:39 +02:00
return color [ 1 ] + " %s | %s %s | %s s \n " % ( self . name . ljust ( self . padding ) , glyph , self . status . ljust ( 7 ) , self . time ) + color [ 0 ]
2017-04-05 21:19:26 +02:00
2017-04-11 19:40:54 +02:00
@property
def was_successful ( self ) :
return self . status != " Failed "
2017-04-05 21:19:26 +02:00
2017-11-30 11:11:18 +01:00
def check_script_prefixes ( ) :
""" Check that no more than `EXPECTED_VIOLATION_COUNT` of the
test scripts don ' t start with one of the allowed name prefixes. " " "
EXPECTED_VIOLATION_COUNT = 77
# LEEWAY is provided as a transition measure, so that pull-requests
# that introduce new tests that don't conform with the naming
# convention don't immediately cause the tests to fail.
LEEWAY = 10
good_prefixes_re = re . compile ( " (example|feature|interface|mempool|mining|p2p|rpc|wallet)_ " )
bad_script_names = [ script for script in ALL_SCRIPTS if good_prefixes_re . match ( script ) is None ]
if len ( bad_script_names ) < EXPECTED_VIOLATION_COUNT :
print ( " {} HURRAY! {} Number of functional tests violating naming convention reduced! " . format ( BOLD [ 1 ] , BOLD [ 0 ] ) )
print ( " Consider reducing EXPECTED_VIOLATION_COUNT from %d to %d " % ( EXPECTED_VIOLATION_COUNT , len ( bad_script_names ) ) )
elif len ( bad_script_names ) > EXPECTED_VIOLATION_COUNT :
print ( " INFO: %d tests not meeting naming conventions (expected %d ): " % ( len ( bad_script_names ) , EXPECTED_VIOLATION_COUNT ) )
print ( " %s " % ( " \n " . join ( sorted ( bad_script_names ) ) ) )
assert len ( bad_script_names ) < = EXPECTED_VIOLATION_COUNT + LEEWAY , " Too many tests not following naming convention! ( %d found, expected: <= %d ) " % ( len ( bad_script_names ) , EXPECTED_VIOLATION_COUNT )
2017-03-27 17:33:00 +02:00
def check_script_list ( src_dir ) :
""" Check scripts directory.
Check that there are no scripts in the functional tests directory which are
not being run by pull - tester . py . """
script_dir = src_dir + ' /test/functional/ '
python_files = set ( [ t for t in os . listdir ( script_dir ) if t [ - 3 : ] == " .py " ] )
missed_tests = list ( python_files - set ( map ( lambda x : x . split ( ) [ 0 ] , ALL_SCRIPTS + NON_SCRIPTS ) ) )
if len ( missed_tests ) != 0 :
2017-04-12 16:41:13 +02:00
print ( " %s WARNING! %s The following scripts are not being run: %s . Check the test lists in test_runner.py. " % ( BOLD [ 1 ] , BOLD [ 0 ] , str ( missed_tests ) ) )
if os . getenv ( ' TRAVIS ' ) == ' true ' :
# On travis this warning is an error to prevent merging incomplete commits into master
sys . exit ( 1 )
2015-10-11 07:41:19 +02:00
class RPCCoverage ( object ) :
"""
2017-03-09 15:53:26 +01:00
Coverage reporting utilities for test_runner .
2015-10-11 07:41:19 +02:00
Coverage calculation works by having each test script subprocess write
coverage files into a particular directory . These files contain the RPC
commands invoked during testing , as well as a complete listing of RPC
commands per ` bitcoin - cli help ` ( ` rpc_interface . txt ` ) .
After all tests complete , the commands run are combined and diff ' d against
the complete list to calculate uncovered RPC commands .
2017-03-09 15:44:57 +01:00
See also : test / functional / test_framework / coverage . py
2015-10-11 07:41:19 +02:00
"""
def __init__ ( self ) :
self . dir = tempfile . mkdtemp ( prefix = " coverage " )
2016-04-29 12:51:15 +02:00
self . flag = ' --coveragedir= %s ' % self . dir
2015-10-11 07:41:19 +02:00
def report_rpc_coverage ( self ) :
"""
Print out RPC commands that were unexercised by tests .
"""
uncovered = self . _get_uncovered_rpc_commands ( )
if uncovered :
print ( " Uncovered RPC commands: " )
print ( " " . join ( ( " - %s \n " % i ) for i in sorted ( uncovered ) ) )
else :
print ( " All RPC commands covered. " )
def cleanup ( self ) :
return shutil . rmtree ( self . dir )
def _get_uncovered_rpc_commands ( self ) :
"""
Return a set of currently untested RPC commands .
"""
2017-03-09 15:44:57 +01:00
# This is shared from `test/functional/test-framework/coverage.py`
2017-02-06 15:07:14 +01:00
reference_filename = ' rpc_interface.txt '
coverage_file_prefix = ' coverage. '
2015-10-11 07:41:19 +02:00
2017-02-06 15:07:14 +01:00
coverage_ref_filename = os . path . join ( self . dir , reference_filename )
2015-10-11 07:41:19 +02:00
coverage_filenames = set ( )
all_cmds = set ( )
covered_cmds = set ( )
if not os . path . isfile ( coverage_ref_filename ) :
raise RuntimeError ( " No coverage reference found " )
with open ( coverage_ref_filename , ' r ' ) as f :
all_cmds . update ( [ i . strip ( ) for i in f . readlines ( ) ] )
for root , dirs , files in os . walk ( self . dir ) :
for filename in files :
2017-02-06 15:07:14 +01:00
if filename . startswith ( coverage_file_prefix ) :
2015-10-11 07:41:19 +02:00
coverage_filenames . add ( os . path . join ( root , filename ) )
for filename in coverage_filenames :
with open ( filename , ' r ' ) as f :
covered_cmds . update ( [ i . strip ( ) for i in f . readlines ( ) ] )
return all_cmds - covered_cmds
if __name__ == ' __main__ ' :
2017-02-06 15:07:14 +01:00
main ( )