from concurrent import futures from utils import NodeFactory, BitcoinD import logging import os import pytest import re import shutil import sys import tempfile with open('config.vars') as configfile: config = dict([(line.rstrip().split('=', 1)) for line in configfile]) VALGRIND = os.getenv("VALGRIND", config['VALGRIND']) == "1" DEVELOPER = os.getenv("DEVELOPER", config['DEVELOPER']) == "1" TEST_DEBUG = os.getenv("TEST_DEBUG", "0") == "1" if TEST_DEBUG: logging.basicConfig(level=logging.DEBUG, stream=sys.stdout) # A dict in which we count how often a particular test has run so far. Used to # give each attempt its own numbered directory, and avoid clashes. __attempts = {} @pytest.fixture(scope="session") def test_base_dir(): directory = tempfile.mkdtemp(prefix='ltests-') print("Running tests in {}".format(directory)) yield directory if os.listdir(directory) == []: shutil.rmtree(directory) @pytest.fixture def directory(request, test_base_dir, test_name): """Return a per-test specific directory. This makes a unique test-directory even if a test is rerun multiple times. """ global __attempts # Auto set value if it isn't in the dict yet __attempts[test_name] = __attempts.get(test_name, 0) + 1 directory = os.path.join(test_base_dir, "{}_{}".format(test_name, __attempts[test_name])) request.node.has_errors = False yield directory # This uses the status set in conftest.pytest_runtest_makereport to # determine whether we succeeded or failed. Outcome can be None if the # failure occurs during the setup phase, hence the use to getattr instead # of accessing it directly. outcome = getattr(request.node, 'rep_call', None).outcome failed = not outcome or request.node.has_errors or outcome != 'passed' if not failed: shutil.rmtree(directory) else: logging.debug("Test execution failed, leaving the test directory {} intact.".format(directory)) @pytest.fixture def test_name(request): yield request.function.__name__ @pytest.fixture def bitcoind(directory): bitcoind = BitcoinD(bitcoin_dir=directory) try: bitcoind.start() except Exception: bitcoind.stop() raise info = bitcoind.rpc.getnetworkinfo() if info['version'] < 160000: bitcoind.rpc.stop() raise ValueError("bitcoind is too old. At least version 16000 (v0.16.0)" " is needed, current version is {}".format(info['version'])) info = bitcoind.rpc.getblockchaininfo() # Make sure we have some spendable funds if info['blocks'] < 101: bitcoind.generate_block(101 - info['blocks']) elif bitcoind.rpc.getwalletinfo()['balance'] < 1: logging.debug("Insufficient balance, generating 1 block") bitcoind.generate_block(1) yield bitcoind try: bitcoind.stop() except Exception: bitcoind.proc.kill() bitcoind.proc.wait() @pytest.fixture def node_factory(request, directory, test_name, bitcoind, executor): nf = NodeFactory(test_name, bitcoind, executor, directory=directory) yield nf err_count = 0 ok, errs = nf.killall([not n.may_fail for n in nf.nodes]) def check_errors(request, err_count, msg): """Just a simple helper to format a message, set flags on request and then raise """ if err_count: request.node.has_errors = True raise ValueError(msg.format(err_count)) if VALGRIND: for node in nf.nodes: err_count += printValgrindErrors(node) check_errors(request, err_count, "{} nodes reported valgrind errors") for node in nf.nodes: err_count += printCrashLog(node) check_errors(request, err_count, "{} nodes had crash.log files") for node in nf.nodes: err_count += checkReconnect(node) check_errors(request, err_count, "{} nodes had unexpected reconnections") for node in [n for n in nf.nodes if not n.allow_bad_gossip]: err_count += checkBadGossip(node) check_errors(request, err_count, "{} nodes had bad gossip messages") for node in nf.nodes: err_count += checkBadReestablish(node) check_errors(request, err_count, "{} nodes had bad reestablish") for node in nf.nodes: err_count += checkBadHSMRequest(node) if err_count: raise ValueError("{} nodes had bad hsm requests".format(err_count)) for node in nf.nodes: err_count += checkMemleak(node) if err_count: raise ValueError("{} nodes had memleak messages \n{}".format(err_count, '\n'.join(errs))) for node in [n for n in nf.nodes if not n.allow_broken_log]: err_count += checkBroken(node) check_errors(request, err_count, "{} nodes had BROKEN messages") if not ok: request.node.has_errors = True raise Exception("At least one lightning node exited with unexpected non-zero return code\n Recorded errors: {}".format('\n'.join(errs))) def getValgrindErrors(node): for error_file in os.listdir(node.daemon.lightning_dir): if not re.fullmatch(r"valgrind-errors.\d+", error_file): continue with open(os.path.join(node.daemon.lightning_dir, error_file), 'r') as f: errors = f.read().strip() if errors: return errors, error_file return None, None def printValgrindErrors(node): errors, fname = getValgrindErrors(node) if errors: print("-" * 31, "Valgrind errors", "-" * 32) print("Valgrind error file:", fname) print(errors) print("-" * 80) return 1 if errors else 0 def getCrashLog(node): if node.may_fail: return None, None try: crashlog = os.path.join(node.daemon.lightning_dir, 'crash.log') with open(crashlog, 'r') as f: return f.readlines(), crashlog except Exception: return None, None def printCrashLog(node): errors, fname = getCrashLog(node) if errors: print("-" * 10, "{} (last 50 lines)".format(fname), "-" * 10) print("".join(errors[-50:])) print("-" * 80) return 1 if errors else 0 def checkReconnect(node): # Without DEVELOPER, we can't suppress reconnection. if node.may_reconnect or not DEVELOPER: return 0 if node.daemon.is_in_log('Peer has reconnected'): return 1 return 0 def checkBadGossip(node): # We can get bad gossip order from inside error msgs. if node.daemon.is_in_log('Bad gossip order from (?!error)'): # This can happen if a node sees a node_announce after a channel # is deleted, however. if node.daemon.is_in_log('Deleting channel'): return 0 return 1 # Other 'Bad' messages shouldn't happen. if node.daemon.is_in_log(r'gossipd.*Bad (?!gossip order from error)'): return 1 return 0 def checkBroken(node): # We can get bad gossip order from inside error msgs. if node.daemon.is_in_log(r'\*\*BROKEN\*\*'): return 1 return 0 def checkBadReestablish(node): if node.daemon.is_in_log('Bad reestablish'): return 1 return 0 def checkBadHSMRequest(node): if node.daemon.is_in_log('bad hsm request'): return 1 return 0 def checkMemleak(node): if node.daemon.is_in_log('MEMLEAK:'): return 1 return 0 @pytest.fixture def executor(): ex = futures.ThreadPoolExecutor(max_workers=20) yield ex ex.shutdown(wait=False)