2017-01-19 22:48:23 +01:00
|
|
|
#!/usr/bin/env python
|
|
|
|
|
2018-08-28 16:14:21 +02:00
|
|
|
from __future__ import print_function
|
2014-03-08 11:31:38 +01:00
|
|
|
|
2018-04-19 22:02:31 +02:00
|
|
|
import io
|
2012-11-29 09:41:17 +01:00
|
|
|
import os
|
|
|
|
import sys
|
|
|
|
import subprocess
|
|
|
|
import re
|
2017-01-19 22:48:23 +01:00
|
|
|
import difflib
|
2012-11-29 09:41:17 +01:00
|
|
|
|
2017-01-09 15:12:12 +01:00
|
|
|
import scriptCommon
|
2019-07-18 16:33:23 +02:00
|
|
|
from scriptCommon import catchPathSymbolic
|
2013-04-24 19:58:57 +02:00
|
|
|
|
2018-05-02 07:25:30 +02:00
|
|
|
if os.name == 'nt':
|
2018-08-28 16:14:21 +02:00
|
|
|
# Enable console colours on windows
|
|
|
|
os.system('')
|
2018-05-02 07:25:30 +02:00
|
|
|
|
2019-07-18 16:33:23 +02:00
|
|
|
rootPath = os.path.join(catchPathSymbolic, 'projects/SelfTest/Baselines')
|
2013-09-30 08:39:06 +02:00
|
|
|
|
2018-10-19 12:46:06 +02:00
|
|
|
langFilenameParser = re.compile(r'(.+\.[ch]pp)')
|
2017-01-19 23:56:31 +01:00
|
|
|
filelocParser = re.compile(r'''
|
|
|
|
.*/
|
|
|
|
(.+\.[ch]pp) # filename
|
|
|
|
(?::|\() # : is starting separator between filename and line number on Linux, ( on Windows
|
|
|
|
([0-9]*) # line number
|
|
|
|
\)? # Windows also has an ending separator, )
|
|
|
|
''', re.VERBOSE)
|
2017-01-19 15:15:06 +01:00
|
|
|
lineNumberParser = re.compile(r' line="[0-9]*"')
|
|
|
|
hexParser = re.compile(r'\b(0[xX][0-9a-fA-F]+)\b')
|
|
|
|
durationsParser = re.compile(r' time="[0-9]*\.[0-9]*"')
|
2017-10-09 12:31:22 +02:00
|
|
|
timestampsParser = re.compile(r'\d{4}-\d{2}-\d{2}T\d{2}\:\d{2}\:\d{2}Z')
|
2017-01-19 15:15:06 +01:00
|
|
|
versionParser = re.compile(r'Catch v[0-9]+\.[0-9]+\.[0-9]+(-develop\.[0-9]+)?')
|
|
|
|
nullParser = re.compile(r'\b(__null|nullptr)\b')
|
2017-01-19 23:56:31 +01:00
|
|
|
exeNameParser = re.compile(r'''
|
|
|
|
\b
|
|
|
|
(CatchSelfTest|SelfTest) # Expected executable name
|
|
|
|
(?:.exe)? # Executable name contains .exe on Windows.
|
|
|
|
\b
|
|
|
|
''', re.VERBOSE)
|
|
|
|
# This is a hack until something more reasonable is figured out
|
|
|
|
specialCaseParser = re.compile(r'file\((\d+)\)')
|
2012-11-29 09:41:17 +01:00
|
|
|
|
2017-04-29 17:52:12 +02:00
|
|
|
# errno macro expands into various names depending on platform, so we need to fix them up as well
|
|
|
|
errnoParser = re.compile(r'''
|
|
|
|
\(\*__errno_location\ \(\)\)
|
|
|
|
|
|
|
|
|
\(\*__error\(\)\)
|
2017-09-01 20:28:49 +02:00
|
|
|
|
|
|
|
|
\(\*_errno\(\)\)
|
2017-04-29 17:52:12 +02:00
|
|
|
''', re.VERBOSE)
|
2017-10-09 12:31:22 +02:00
|
|
|
sinceEpochParser = re.compile(r'\d+ .+ since epoch')
|
2017-11-01 07:30:11 +01:00
|
|
|
infParser = re.compile(r'''
|
|
|
|
\(\(float\)\(1e\+300\ \*\ 1e\+300\)\) # MSVC INFINITY macro
|
|
|
|
|
|
2017-11-01 08:46:49 +01:00
|
|
|
\(__builtin_inff\(\)\) # Linux (ubuntu) INFINITY macro
|
|
|
|
|
|
2018-01-14 17:04:36 +01:00
|
|
|
\(__builtin_inff\ \(\)\) # Fedora INFINITY macro
|
|
|
|
|
|
2017-11-01 08:46:49 +01:00
|
|
|
__builtin_huge_valf\(\) # OSX macro
|
2017-11-01 07:30:11 +01:00
|
|
|
''', re.VERBOSE)
|
2017-11-10 18:14:42 +01:00
|
|
|
nanParser = re.compile(r'''
|
|
|
|
\(\(float\)\(\(\(float\)\(1e\+300\ \*\ 1e\+300\)\)\ \*\ 0\.0F\)\) # MSVC NAN macro
|
|
|
|
|
|
2017-11-10 19:52:43 +01:00
|
|
|
\(\(float\)\(INFINITY\ \*\ 0\.0F\)\) # Yet another MSVC NAN macro
|
|
|
|
|
|
2017-11-10 21:43:23 +01:00
|
|
|
\(__builtin_nanf\ \(""\)\) # Linux (ubuntu) NAN macro
|
|
|
|
|
|
|
|
|
__builtin_nanf\("0x<hex\ digits>"\) # The weird content of the brackets is there because a different parser has already ran before this one
|
2017-11-10 18:14:42 +01:00
|
|
|
''', re.VERBOSE)
|
|
|
|
|
2017-04-29 17:52:12 +02:00
|
|
|
|
2013-02-02 20:58:04 +01:00
|
|
|
if len(sys.argv) == 2:
|
2017-01-09 15:12:12 +01:00
|
|
|
cmdPath = sys.argv[1]
|
2013-02-02 20:58:04 +01:00
|
|
|
else:
|
2017-01-19 22:08:51 +01:00
|
|
|
cmdPath = os.path.join(catchPath, scriptCommon.getBuildExecutable())
|
2012-11-29 09:41:17 +01:00
|
|
|
|
2013-09-27 20:01:14 +02:00
|
|
|
overallResult = 0
|
2012-11-29 09:41:17 +01:00
|
|
|
|
2018-04-06 14:59:08 +02:00
|
|
|
|
2017-01-19 22:48:23 +01:00
|
|
|
def diffFiles(fileA, fileB):
|
2018-04-19 22:02:31 +02:00
|
|
|
with io.open(fileA, 'r', encoding='utf-8', errors='surrogateescape') as file:
|
2017-04-29 18:06:36 +02:00
|
|
|
aLines = [line.rstrip() for line in file.readlines()]
|
2018-04-19 22:02:31 +02:00
|
|
|
with io.open(fileB, 'r', encoding='utf-8', errors='surrogateescape') as file:
|
2017-04-29 18:06:36 +02:00
|
|
|
bLines = [line.rstrip() for line in file.readlines()]
|
2017-01-19 22:48:23 +01:00
|
|
|
|
|
|
|
shortenedFilenameA = fileA.rsplit(os.sep, 1)[-1]
|
|
|
|
shortenedFilenameB = fileB.rsplit(os.sep, 1)[-1]
|
|
|
|
|
|
|
|
diff = difflib.unified_diff(aLines, bLines, fromfile=shortenedFilenameA, tofile=shortenedFilenameB, n=0)
|
|
|
|
return [line for line in diff if line[0] in ('+', '-')]
|
|
|
|
|
|
|
|
|
2018-10-19 12:46:06 +02:00
|
|
|
def normalizeFilepath(line):
|
2019-07-19 13:29:16 +02:00
|
|
|
if catchPathSymbolic in line or catchPathSymbolic.lower() in line:
|
2017-01-19 23:56:31 +01:00
|
|
|
# make paths relative to Catch root
|
2019-07-18 16:33:23 +02:00
|
|
|
line = line.replace(catchPathSymbolic + os.sep, '')
|
2019-07-19 13:29:16 +02:00
|
|
|
line = line.replace(catchPathSymbolic.lower() + os.sep, '')
|
2018-10-19 12:46:06 +02:00
|
|
|
|
|
|
|
m = langFilenameParser.match(line)
|
|
|
|
if m:
|
|
|
|
filepath = m.group(0)
|
2017-01-19 23:56:31 +01:00
|
|
|
# go from \ in windows paths to /
|
2018-10-19 12:46:06 +02:00
|
|
|
filepath = filepath.replace('\\', '/')
|
|
|
|
# remove start of relative path
|
|
|
|
filepath = filepath.replace('../', '')
|
|
|
|
line = line[:m.start()] + filepath + line[m.end():]
|
|
|
|
|
|
|
|
return line
|
|
|
|
|
|
|
|
def filterLine(line, isCompact):
|
|
|
|
line = normalizeFilepath(line)
|
2017-01-19 23:56:31 +01:00
|
|
|
|
2017-01-19 15:15:06 +01:00
|
|
|
# strip source line numbers
|
|
|
|
m = filelocParser.match(line)
|
2017-01-09 15:12:12 +01:00
|
|
|
if m:
|
2017-01-19 15:15:06 +01:00
|
|
|
# note that this also strips directories, leaving only the filename
|
|
|
|
filename, lnum = m.groups()
|
2017-01-19 23:56:31 +01:00
|
|
|
lnum = ":<line number>" if lnum else ""
|
2017-01-19 15:15:06 +01:00
|
|
|
line = filename + lnum + line[m.end():]
|
2017-01-09 15:12:12 +01:00
|
|
|
else:
|
2017-01-19 15:15:06 +01:00
|
|
|
line = lineNumberParser.sub(" ", line)
|
2017-01-09 15:12:12 +01:00
|
|
|
|
2018-06-04 20:28:33 +02:00
|
|
|
if isCompact:
|
|
|
|
line = line.replace(': FAILED', ': failed')
|
|
|
|
line = line.replace(': PASSED', ': passed')
|
2018-08-28 16:14:21 +02:00
|
|
|
|
2017-01-19 15:15:06 +01:00
|
|
|
# strip Catch version number
|
|
|
|
line = versionParser.sub("<version>", line)
|
2017-01-09 15:12:12 +01:00
|
|
|
|
2017-01-19 15:15:06 +01:00
|
|
|
# replace *null* with 0
|
|
|
|
line = nullParser.sub("0", line)
|
|
|
|
|
|
|
|
# strip executable name
|
|
|
|
line = exeNameParser.sub("<exe-name>", line)
|
|
|
|
|
|
|
|
# strip hexadecimal numbers (presumably pointer values)
|
|
|
|
line = hexParser.sub("0x<hex digits>", line)
|
|
|
|
|
|
|
|
# strip durations and timestamps
|
|
|
|
line = durationsParser.sub(' time="{duration}"', line)
|
2017-10-09 12:31:22 +02:00
|
|
|
line = timestampsParser.sub('{iso8601-timestamp}', line)
|
2017-01-19 23:56:31 +01:00
|
|
|
line = specialCaseParser.sub('file:\g<1>', line)
|
2017-04-29 17:52:12 +02:00
|
|
|
line = errnoParser.sub('errno', line)
|
2017-10-09 12:31:22 +02:00
|
|
|
line = sinceEpochParser.sub('{since-epoch-report}', line)
|
2017-11-01 07:30:11 +01:00
|
|
|
line = infParser.sub('INFINITY', line)
|
2017-11-10 18:14:42 +01:00
|
|
|
line = nanParser.sub('NAN', line)
|
2017-01-09 15:12:12 +01:00
|
|
|
return line
|
|
|
|
|
|
|
|
|
|
|
|
def approve(baseName, args):
|
|
|
|
global overallResult
|
|
|
|
args[0:0] = [cmdPath]
|
|
|
|
if not os.path.exists(cmdPath):
|
|
|
|
raise Exception("Executable doesn't exist at " + cmdPath)
|
|
|
|
baselinesPath = os.path.join(rootPath, '{0}.approved.txt'.format(baseName))
|
|
|
|
rawResultsPath = os.path.join(rootPath, '_{0}.tmp'.format(baseName))
|
|
|
|
filteredResultsPath = os.path.join(rootPath, '{0}.unapproved.txt'.format(baseName))
|
|
|
|
|
|
|
|
f = open(rawResultsPath, 'w')
|
|
|
|
subprocess.call(args, stdout=f, stderr=f)
|
|
|
|
f.close()
|
|
|
|
|
2018-04-19 22:02:31 +02:00
|
|
|
rawFile = io.open(rawResultsPath, 'r', encoding='utf-8', errors='surrogateescape')
|
|
|
|
filteredFile = io.open(filteredResultsPath, 'w', encoding='utf-8', errors='surrogateescape')
|
2017-01-09 15:12:12 +01:00
|
|
|
for line in rawFile:
|
2018-06-04 20:28:33 +02:00
|
|
|
filteredFile.write(filterLine(line, 'compact' in baseName).rstrip() + "\n")
|
2017-01-09 15:12:12 +01:00
|
|
|
filteredFile.close()
|
|
|
|
rawFile.close()
|
|
|
|
|
|
|
|
os.remove(rawResultsPath)
|
|
|
|
print()
|
|
|
|
print(baseName + ":")
|
|
|
|
if os.path.exists(baselinesPath):
|
2017-01-19 22:48:23 +01:00
|
|
|
diffResult = diffFiles(baselinesPath, filteredResultsPath)
|
|
|
|
if diffResult:
|
2017-04-29 18:06:36 +02:00
|
|
|
print('\n'.join(diffResult))
|
2017-01-19 22:48:23 +01:00
|
|
|
print(" \n****************************\n \033[91mResults differed")
|
|
|
|
if len(diffResult) > overallResult:
|
|
|
|
overallResult = len(diffResult)
|
|
|
|
else:
|
2017-01-09 15:12:12 +01:00
|
|
|
os.remove(filteredResultsPath)
|
|
|
|
print(" \033[92mResults matched")
|
|
|
|
print("\033[0m")
|
|
|
|
else:
|
|
|
|
print(" first approval")
|
|
|
|
if overallResult == 0:
|
|
|
|
overallResult = 1
|
|
|
|
|
|
|
|
|
|
|
|
print("Running approvals against executable:")
|
|
|
|
print(" " + cmdPath)
|
2013-09-27 20:01:14 +02:00
|
|
|
|
2017-12-06 15:48:46 +01:00
|
|
|
|
2018-08-28 16:14:21 +02:00
|
|
|
# ## Keep default reporters here ##
|
2013-09-27 20:01:14 +02:00
|
|
|
# Standard console reporter
|
2018-10-13 15:30:15 +02:00
|
|
|
approve("console.std", ["~[!nonportable]~[!benchmark]~[approvals]", "--order", "lex", "--rng-seed", "1"])
|
2013-09-27 20:01:14 +02:00
|
|
|
# console reporter, include passes, warn about No Assertions
|
2018-10-13 15:30:15 +02:00
|
|
|
approve("console.sw", ["~[!nonportable]~[!benchmark]~[approvals]", "-s", "-w", "NoAssertions", "--order", "lex", "--rng-seed", "1"])
|
2013-09-27 20:01:14 +02:00
|
|
|
# console reporter, include passes, warn about No Assertions, limit failures to first 4
|
2018-10-13 15:30:15 +02:00
|
|
|
approve("console.swa4", ["~[!nonportable]~[!benchmark]~[approvals]", "-s", "-w", "NoAssertions", "-x", "4", "--order", "lex", "--rng-seed", "1"])
|
2013-09-27 20:01:14 +02:00
|
|
|
# junit reporter, include passes, warn about No Assertions
|
2018-10-13 15:30:15 +02:00
|
|
|
approve("junit.sw", ["~[!nonportable]~[!benchmark]~[approvals]", "-s", "-w", "NoAssertions", "-r", "junit", "--order", "lex", "--rng-seed", "1"])
|
2013-09-27 20:01:14 +02:00
|
|
|
# xml reporter, include passes, warn about No Assertions
|
2018-10-13 15:30:15 +02:00
|
|
|
approve("xml.sw", ["~[!nonportable]~[!benchmark]~[approvals]", "-s", "-w", "NoAssertions", "-r", "xml", "--order", "lex", "--rng-seed", "1"])
|
2017-12-06 15:48:46 +01:00
|
|
|
# compact reporter, include passes, warn about No Assertions
|
2018-10-13 15:30:15 +02:00
|
|
|
approve('compact.sw', ['~[!nonportable]~[!benchmark]~[approvals]', '-s', '-w', 'NoAssertions', '-r', 'compact', '--order', 'lex', "--rng-seed", "1"])
|
2013-09-27 20:01:14 +02:00
|
|
|
|
2014-03-08 11:31:11 +01:00
|
|
|
if overallResult != 0:
|
2017-02-24 15:56:26 +01:00
|
|
|
print("If these differences are expected, run approve.py to approve new baselines.")
|
2017-01-09 15:12:12 +01:00
|
|
|
exit(overallResult)
|