Clean up bctest.py and bitcoin-util-test.py
- remove newlines - change tabs for spaces, to align with convention in other py files - add comments - add 'Bitcoin Core Developers' copyright notice
This commit is contained in:
parent
6a1343f73b
commit
2b175d4b01
2 changed files with 101 additions and 81 deletions
|
@ -1,4 +1,5 @@
|
||||||
# Copyright 2014 BitPay, Inc.
|
# Copyright 2014 BitPay, Inc.
|
||||||
|
# Copyright 2016 The Bitcoin Core developers
|
||||||
# Distributed under the MIT software license, see the accompanying
|
# Distributed under the MIT software license, see the accompanying
|
||||||
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
||||||
from __future__ import division,print_function,unicode_literals
|
from __future__ import division,print_function,unicode_literals
|
||||||
|
@ -11,94 +12,110 @@ import difflib
|
||||||
import logging
|
import logging
|
||||||
|
|
||||||
def parse_output(a, fmt):
|
def parse_output(a, fmt):
|
||||||
if fmt == 'json': # json: compare parsed data
|
"""Parse the output according to specified format.
|
||||||
return json.loads(a)
|
|
||||||
elif fmt == 'hex': # hex: parse and compare binary data
|
Raise an error if the output can't be parsed."""
|
||||||
return binascii.a2b_hex(a.strip())
|
if fmt == 'json': # json: compare parsed data
|
||||||
else:
|
return json.loads(a)
|
||||||
raise NotImplementedError("Don't know how to compare %s" % fmt)
|
elif fmt == 'hex': # hex: parse and compare binary data
|
||||||
|
return binascii.a2b_hex(a.strip())
|
||||||
|
else:
|
||||||
|
raise NotImplementedError("Don't know how to compare %s" % fmt)
|
||||||
|
|
||||||
def bctest(testDir, testObj, exeext):
|
def bctest(testDir, testObj, exeext):
|
||||||
|
"""Runs a single test, comparing output and RC to expected output and RC.
|
||||||
|
|
||||||
execprog = testObj['exec'] + exeext
|
Raises an error if input can't be read, executable fails, or output/RC
|
||||||
execargs = testObj['args']
|
are not as expected. Error is caught by bctester() and reported.
|
||||||
execrun = [execprog] + execargs
|
"""
|
||||||
stdinCfg = None
|
# Get the exec names and arguments
|
||||||
inputData = None
|
execprog = testObj['exec'] + exeext
|
||||||
if "input" in testObj:
|
execargs = testObj['args']
|
||||||
filename = testDir + "/" + testObj['input']
|
execrun = [execprog] + execargs
|
||||||
inputData = open(filename).read()
|
|
||||||
stdinCfg = subprocess.PIPE
|
|
||||||
|
|
||||||
outputFn = None
|
# Read the input data (if there is any)
|
||||||
outputData = None
|
stdinCfg = None
|
||||||
if "output_cmp" in testObj:
|
inputData = None
|
||||||
outputFn = testObj['output_cmp']
|
if "input" in testObj:
|
||||||
outputType = os.path.splitext(outputFn)[1][1:] # output type from file extension (determines how to compare)
|
filename = testDir + "/" + testObj['input']
|
||||||
try:
|
inputData = open(filename).read()
|
||||||
outputData = open(testDir + "/" + outputFn).read()
|
stdinCfg = subprocess.PIPE
|
||||||
except:
|
|
||||||
logging.error("Output file " + outputFn + " can not be opened")
|
|
||||||
raise
|
|
||||||
if not outputData:
|
|
||||||
logging.error("Output data missing for " + outputFn)
|
|
||||||
raise Exception
|
|
||||||
|
|
||||||
proc = subprocess.Popen(execrun, stdin=stdinCfg, stdout=subprocess.PIPE, stderr=subprocess.PIPE,universal_newlines=True)
|
# Read the expected output data (if there is any)
|
||||||
try:
|
outputFn = None
|
||||||
outs = proc.communicate(input=inputData)
|
outputData = None
|
||||||
except OSError:
|
if "output_cmp" in testObj:
|
||||||
logging.error("OSError, Failed to execute " + execprog)
|
outputFn = testObj['output_cmp']
|
||||||
raise
|
outputType = os.path.splitext(outputFn)[1][1:] # output type from file extension (determines how to compare)
|
||||||
|
try:
|
||||||
|
outputData = open(testDir + "/" + outputFn).read()
|
||||||
|
except:
|
||||||
|
logging.error("Output file " + outputFn + " can not be opened")
|
||||||
|
raise
|
||||||
|
if not outputData:
|
||||||
|
logging.error("Output data missing for " + outputFn)
|
||||||
|
raise Exception
|
||||||
|
|
||||||
if outputData:
|
# Run the test
|
||||||
try:
|
proc = subprocess.Popen(execrun, stdin=stdinCfg, stdout=subprocess.PIPE, stderr=subprocess.PIPE,universal_newlines=True)
|
||||||
a_parsed = parse_output(outs[0], outputType)
|
try:
|
||||||
except Exception as e:
|
outs = proc.communicate(input=inputData)
|
||||||
logging.error('Error parsing command output as %s: %s' % (outputType,e))
|
except OSError:
|
||||||
raise
|
logging.error("OSError, Failed to execute " + execprog)
|
||||||
try:
|
raise
|
||||||
b_parsed = parse_output(outputData, outputType)
|
|
||||||
except Exception as e:
|
|
||||||
logging.error('Error parsing expected output %s as %s: %s' % (outputFn,outputType,e))
|
|
||||||
raise
|
|
||||||
if a_parsed != b_parsed:
|
|
||||||
logging.error("Output data mismatch for " + outputFn + " (format " + outputType + ")")
|
|
||||||
raise Exception
|
|
||||||
if outs[0] != outputData:
|
|
||||||
error_message = "Output formatting mismatch for " + outputFn + ":\n"
|
|
||||||
error_message += "".join(difflib.context_diff(outputData.splitlines(True),
|
|
||||||
outs[0].splitlines(True),
|
|
||||||
fromfile=outputFn,
|
|
||||||
tofile="returned"))
|
|
||||||
logging.error(error_message)
|
|
||||||
raise Exception
|
|
||||||
|
|
||||||
wantRC = 0
|
if outputData:
|
||||||
if "return_code" in testObj:
|
# Parse command output and expected output
|
||||||
wantRC = testObj['return_code']
|
try:
|
||||||
if proc.returncode != wantRC:
|
a_parsed = parse_output(outs[0], outputType)
|
||||||
logging.error("Return code mismatch for " + outputFn)
|
except Exception as e:
|
||||||
raise Exception
|
logging.error('Error parsing command output as %s: %s' % (outputType,e))
|
||||||
|
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("Output data mismatch for " + outputFn + " (format " + outputType + ")")
|
||||||
|
raise Exception
|
||||||
|
# Compare formatting
|
||||||
|
if outs[0] != outputData:
|
||||||
|
error_message = "Output formatting mismatch for " + outputFn + ":\n"
|
||||||
|
error_message += "".join(difflib.context_diff(outputData.splitlines(True),
|
||||||
|
outs[0].splitlines(True),
|
||||||
|
fromfile=outputFn,
|
||||||
|
tofile="returned"))
|
||||||
|
logging.error(error_message)
|
||||||
|
raise Exception
|
||||||
|
|
||||||
|
# Compare the return code to the expected return code
|
||||||
|
wantRC = 0
|
||||||
|
if "return_code" in testObj:
|
||||||
|
wantRC = testObj['return_code']
|
||||||
|
if proc.returncode != wantRC:
|
||||||
|
logging.error("Return code mismatch for " + outputFn)
|
||||||
|
raise Exception
|
||||||
|
|
||||||
def bctester(testDir, input_basename, buildenv):
|
def bctester(testDir, input_basename, buildenv):
|
||||||
input_filename = testDir + "/" + input_basename
|
""" Loads and parses the input file, runs all tests and reports results"""
|
||||||
raw_data = open(input_filename).read()
|
input_filename = testDir + "/" + input_basename
|
||||||
input_data = json.loads(raw_data)
|
raw_data = open(input_filename).read()
|
||||||
|
input_data = json.loads(raw_data)
|
||||||
|
|
||||||
failed_testcases = []
|
failed_testcases = []
|
||||||
|
|
||||||
for testObj in input_data:
|
for testObj in input_data:
|
||||||
try:
|
try:
|
||||||
bctest(testDir, testObj, buildenv.exeext)
|
bctest(testDir, testObj, buildenv.exeext)
|
||||||
logging.info("PASSED: " + testObj["description"])
|
logging.info("PASSED: " + testObj["description"])
|
||||||
except:
|
except:
|
||||||
logging.info("FAILED: " + testObj["description"])
|
logging.info("FAILED: " + testObj["description"])
|
||||||
failed_testcases.append(testObj["description"])
|
failed_testcases.append(testObj["description"])
|
||||||
|
|
||||||
if failed_testcases:
|
|
||||||
logging.error("FAILED TESTCASES: [" + ", ".join(failed_testcases) + "]")
|
|
||||||
sys.exit(1)
|
|
||||||
else:
|
|
||||||
sys.exit(0)
|
|
||||||
|
|
||||||
|
if failed_testcases:
|
||||||
|
logging.error("FAILED TESTCASES: [" + ", ".join(failed_testcases) + "]")
|
||||||
|
sys.exit(1)
|
||||||
|
else:
|
||||||
|
sys.exit(0)
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
#!/usr/bin/env python
|
#!/usr/bin/env python
|
||||||
# Copyright 2014 BitPay, Inc.
|
# Copyright 2014 BitPay, Inc.
|
||||||
|
# Copyright 2016 The Bitcoin Core developers
|
||||||
# Distributed under the MIT software license, see the accompanying
|
# Distributed under the MIT software license, see the accompanying
|
||||||
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
||||||
from __future__ import division,print_function,unicode_literals
|
from __future__ import division,print_function,unicode_literals
|
||||||
|
@ -16,11 +17,13 @@ Runs automatically during `make check`.
|
||||||
|
|
||||||
Can also be run manually from the src directory by specifiying the source directory:
|
Can also be run manually from the src directory by specifiying the source directory:
|
||||||
|
|
||||||
test/bitcoin-util-test.py --src=[srcdir]
|
test/bitcoin-util-test.py --srcdir='srcdir' [--verbose]
|
||||||
"""
|
"""
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
if __name__ == '__main__':
|
||||||
|
# Try to get the source directory from the environment variables. This will
|
||||||
|
# be set for `make check` automated runs. If environment variable is not set,
|
||||||
|
# then get the source directory from command line args.
|
||||||
try:
|
try:
|
||||||
srcdir = os.environ["srcdir"]
|
srcdir = os.environ["srcdir"]
|
||||||
verbose = False
|
verbose = False
|
||||||
|
|
Loading…
Reference in a new issue