mirror of
https://github.com/bitcoin/bitcoin.git
synced 2025-01-09 11:27:28 -03:00
180 lines
6.7 KiB
Python
Executable file
180 lines
6.7 KiB
Python
Executable file
#!/usr/bin/env python3
|
|
# Copyright 2014 BitPay Inc.
|
|
# Copyright 2016-2017 The Bitcoin Core developers
|
|
# Distributed under the MIT software license, see the accompanying
|
|
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
"""Test framework for bitcoin utils.
|
|
|
|
Runs automatically during `ctest --test-dir build/`.
|
|
|
|
Can also be run manually."""
|
|
|
|
import argparse
|
|
import configparser
|
|
import difflib
|
|
import json
|
|
import logging
|
|
import os
|
|
import pprint
|
|
import subprocess
|
|
import sys
|
|
|
|
def main():
|
|
config = configparser.ConfigParser()
|
|
config.optionxform = str
|
|
with open(os.path.join(os.path.dirname(__file__), "../config.ini"), encoding="utf8") as f:
|
|
config.read_file(f)
|
|
env_conf = dict(config.items('environment'))
|
|
|
|
parser = argparse.ArgumentParser(description=__doc__)
|
|
parser.add_argument('-v', '--verbose', action='store_true')
|
|
args = parser.parse_args()
|
|
verbose = args.verbose
|
|
|
|
if verbose:
|
|
level = logging.DEBUG
|
|
else:
|
|
level = logging.ERROR
|
|
formatter = '%(asctime)s - %(levelname)s - %(message)s'
|
|
# Add the format/level to the logger
|
|
logging.basicConfig(format=formatter, level=level)
|
|
|
|
bctester(os.path.join(env_conf["SRCDIR"], "test", "util", "data"), "bitcoin-util-test.json", env_conf)
|
|
|
|
def bctester(testDir, input_basename, buildenv):
|
|
""" Loads and parses the input file, runs all tests and reports results"""
|
|
input_filename = os.path.join(testDir, input_basename)
|
|
with open(input_filename, encoding="utf8") as f:
|
|
raw_data = f.read()
|
|
input_data = json.loads(raw_data)
|
|
|
|
failed_testcases = []
|
|
|
|
for testObj in input_data:
|
|
try:
|
|
bctest(testDir, testObj, buildenv)
|
|
logging.info("PASSED: " + testObj["description"])
|
|
except Exception:
|
|
logging.info("FAILED: " + testObj["description"])
|
|
failed_testcases.append(testObj["description"])
|
|
|
|
if failed_testcases:
|
|
error_message = "FAILED_TESTCASES:\n"
|
|
error_message += pprint.pformat(failed_testcases, width=400)
|
|
logging.error(error_message)
|
|
sys.exit(1)
|
|
else:
|
|
sys.exit(0)
|
|
|
|
def bctest(testDir, testObj, buildenv):
|
|
"""Runs a single test, comparing output and RC to expected output and RC.
|
|
|
|
Raises an error if input can't be read, executable fails, or output/RC
|
|
are not as expected. Error is caught by bctester() and reported.
|
|
"""
|
|
# Get the exec names and arguments
|
|
execprog = os.path.join(buildenv["BUILDDIR"], "src", testObj["exec"] + buildenv["EXEEXT"])
|
|
if testObj["exec"] == "./bitcoin-util":
|
|
execprog = os.getenv("BITCOINUTIL", default=execprog)
|
|
elif testObj["exec"] == "./bitcoin-tx":
|
|
execprog = os.getenv("BITCOINTX", default=execprog)
|
|
|
|
execargs = testObj['args']
|
|
execrun = [execprog] + execargs
|
|
|
|
# Read the input data (if there is any)
|
|
inputData = None
|
|
if "input" in testObj:
|
|
filename = os.path.join(testDir, testObj["input"])
|
|
with open(filename, encoding="utf8") as f:
|
|
inputData = f.read()
|
|
|
|
# Read the expected output data (if there is any)
|
|
outputFn = None
|
|
outputData = None
|
|
outputType = None
|
|
if "output_cmp" in testObj:
|
|
outputFn = testObj['output_cmp']
|
|
outputType = os.path.splitext(outputFn)[1][1:] # output type from file extension (determines how to compare)
|
|
try:
|
|
with open(os.path.join(testDir, outputFn), encoding="utf8") as f:
|
|
outputData = f.read()
|
|
except Exception:
|
|
logging.error("Output file " + outputFn + " cannot be opened")
|
|
raise
|
|
if not outputData:
|
|
logging.error("Output data missing for " + outputFn)
|
|
raise Exception
|
|
if not outputType:
|
|
logging.error("Output file %s does not have a file extension" % outputFn)
|
|
raise Exception
|
|
|
|
# Run the test
|
|
try:
|
|
res = subprocess.run(execrun, capture_output=True, text=True, input=inputData)
|
|
except OSError:
|
|
logging.error("OSError, Failed to execute " + execprog)
|
|
raise
|
|
|
|
if outputData:
|
|
data_mismatch, formatting_mismatch = False, False
|
|
# Parse command output and expected output
|
|
try:
|
|
a_parsed = parse_output(res.stdout, outputType)
|
|
except Exception as e:
|
|
logging.error(f"Error parsing command output as {outputType}: '{str(e)}'; res: {str(res)}")
|
|
raise
|
|
try:
|
|
b_parsed = parse_output(outputData, outputType)
|
|
except Exception as e:
|
|
logging.error('Error parsing expected output %s as %s: %s' % (outputFn, outputType, e))
|
|
raise
|
|
# Compare data
|
|
if a_parsed != b_parsed:
|
|
logging.error(f"Output data mismatch for {outputFn} (format {outputType}); res: {str(res)}")
|
|
data_mismatch = True
|
|
# Compare formatting
|
|
if res.stdout != outputData:
|
|
error_message = f"Output formatting mismatch for {outputFn}:\nres: {str(res)}\n"
|
|
error_message += "".join(difflib.context_diff(outputData.splitlines(True),
|
|
res.stdout.splitlines(True),
|
|
fromfile=outputFn,
|
|
tofile="returned"))
|
|
logging.error(error_message)
|
|
formatting_mismatch = True
|
|
|
|
assert not data_mismatch and not formatting_mismatch
|
|
|
|
# Compare the return code to the expected return code
|
|
wantRC = 0
|
|
if "return_code" in testObj:
|
|
wantRC = testObj['return_code']
|
|
if res.returncode != wantRC:
|
|
logging.error(f"Return code mismatch for {outputFn}; res: {str(res)}")
|
|
raise Exception
|
|
|
|
if "error_txt" in testObj:
|
|
want_error = testObj["error_txt"]
|
|
# Compare error text
|
|
# TODO: ideally, we'd compare the strings exactly and also assert
|
|
# That stderr is empty if no errors are expected. However, bitcoin-tx
|
|
# emits DISPLAY errors when running as a windows application on
|
|
# linux through wine. Just assert that the expected error text appears
|
|
# somewhere in stderr.
|
|
if want_error not in res.stderr:
|
|
logging.error(f"Error mismatch:\nExpected: {want_error}\nReceived: {res.stderr.rstrip()}\nres: {str(res)}")
|
|
raise Exception
|
|
|
|
def parse_output(a, fmt):
|
|
"""Parse the output according to specified format.
|
|
|
|
Raise an error if the output can't be parsed."""
|
|
if fmt == 'json': # json: compare parsed data
|
|
return json.loads(a)
|
|
elif fmt == 'hex': # hex: parse and compare binary data
|
|
return bytes.fromhex(a.strip())
|
|
else:
|
|
raise NotImplementedError("Don't know how to compare %s" % fmt)
|
|
|
|
if __name__ == '__main__':
|
|
main()
|