Christopher Dunn | bd1e895 | 2014-11-19 23:30:47 -0600 | [diff] [blame] | 1 | from __future__ import print_function |
Christopher Dunn | cd140b5 | 2015-01-16 13:44:27 -0600 | [diff] [blame] | 2 | from __future__ import unicode_literals |
| 3 | from io import open |
| 4 | from glob import glob |
Christopher Dunn | f986423 | 2007-06-14 21:01:26 +0000 | [diff] [blame] | 5 | import sys |
| 6 | import os |
Christopher Dunn | 4ca9d25 | 2015-01-09 22:28:20 -0600 | [diff] [blame] | 7 | import os.path |
Baptiste Lepilleur | 932cfc7 | 2009-11-19 20:16:59 +0000 | [diff] [blame] | 8 | import optparse |
Christopher Dunn | f986423 | 2007-06-14 21:01:26 +0000 | [diff] [blame] | 9 | |
Baptiste Lepilleur | 932cfc7 | 2009-11-19 20:16:59 +0000 | [diff] [blame] | 10 | VALGRIND_CMD = 'valgrind --tool=memcheck --leak-check=yes --undef-value-errors=yes ' |
Christopher Dunn | f986423 | 2007-06-14 21:01:26 +0000 | [diff] [blame] | 11 | |
Christopher Dunn | cd140b5 | 2015-01-16 13:44:27 -0600 | [diff] [blame] | 12 | def getStatusOutput(cmd): |
| 13 | """ |
| 14 | Return int, unicode (for both Python 2 and 3). |
| 15 | Note: os.popen().close() would return None for 0. |
| 16 | """ |
| 17 | pipe = os.popen(cmd) |
| 18 | process_output = pipe.read() |
| 19 | try: |
| 20 | # We have been using os.popen(). When we read() the result |
| 21 | # we get 'str' (bytes) in py2, and 'str' (unicode) in py3. |
| 22 | # Ugh! There must be a better way to handle this. |
| 23 | process_output = process_output.decode('utf-8') |
| 24 | except AttributeError: |
| 25 | pass # python3 |
| 26 | status = pipe.close() |
| 27 | return status, process_output |
Christopher Dunn | f986423 | 2007-06-14 21:01:26 +0000 | [diff] [blame] | 28 | def compareOutputs( expected, actual, message ): |
| 29 | expected = expected.strip().replace('\r','').split('\n') |
| 30 | actual = actual.strip().replace('\r','').split('\n') |
| 31 | diff_line = 0 |
| 32 | max_line_to_compare = min( len(expected), len(actual) ) |
Christopher Dunn | bd1e895 | 2014-11-19 23:30:47 -0600 | [diff] [blame] | 33 | for index in range(0,max_line_to_compare): |
Christopher Dunn | f986423 | 2007-06-14 21:01:26 +0000 | [diff] [blame] | 34 | if expected[index].strip() != actual[index].strip(): |
| 35 | diff_line = index + 1 |
| 36 | break |
| 37 | if diff_line == 0 and len(expected) != len(actual): |
| 38 | diff_line = max_line_to_compare+1 |
| 39 | if diff_line == 0: |
| 40 | return None |
| 41 | def safeGetLine( lines, index ): |
| 42 | index += -1 |
| 43 | if index >= len(lines): |
| 44 | return '' |
| 45 | return lines[index].strip() |
| 46 | return """ Difference in %s at line %d: |
| 47 | Expected: '%s' |
| 48 | Actual: '%s' |
| 49 | """ % (message, diff_line, |
| 50 | safeGetLine(expected,diff_line), |
| 51 | safeGetLine(actual,diff_line) ) |
| 52 | |
| 53 | def safeReadFile( path ): |
| 54 | try: |
datadiode | 01aee4a | 2015-01-11 10:39:24 +0100 | [diff] [blame] | 55 | return open( path, 'rt', encoding = 'utf-8' ).read() |
Christopher Dunn | 9aa6144 | 2014-11-19 23:10:02 -0600 | [diff] [blame] | 56 | except IOError as e: |
Christopher Dunn | f986423 | 2007-06-14 21:01:26 +0000 | [diff] [blame] | 57 | return '<File "%s" is missing: %s>' % (path,e) |
| 58 | |
Baptiste Lepilleur | 932cfc7 | 2009-11-19 20:16:59 +0000 | [diff] [blame] | 59 | def runAllTests( jsontest_executable_path, input_dir = None, |
Baptiste Lepilleur | 7c66ac2 | 2010-02-21 14:26:08 +0000 | [diff] [blame] | 60 | use_valgrind=False, with_json_checker=False ): |
Christopher Dunn | f986423 | 2007-06-14 21:01:26 +0000 | [diff] [blame] | 61 | if not input_dir: |
Baptiste Lepilleur | 7dec64f | 2009-11-21 18:20:25 +0000 | [diff] [blame] | 62 | input_dir = os.path.join( os.getcwd(), 'data' ) |
Baptiste Lepilleur | 64e07e5 | 2009-11-18 21:27:06 +0000 | [diff] [blame] | 63 | tests = glob( os.path.join( input_dir, '*.json' ) ) |
Baptiste Lepilleur | 7c66ac2 | 2010-02-21 14:26:08 +0000 | [diff] [blame] | 64 | if with_json_checker: |
| 65 | test_jsonchecker = glob( os.path.join( input_dir, '../jsonchecker', '*.json' ) ) |
Baptiste Lepilleur | 8868147 | 2009-11-18 21:38:54 +0000 | [diff] [blame] | 66 | else: |
| 67 | test_jsonchecker = [] |
Christopher Dunn | f986423 | 2007-06-14 21:01:26 +0000 | [diff] [blame] | 68 | failed_tests = [] |
Baptiste Lepilleur | 932cfc7 | 2009-11-19 20:16:59 +0000 | [diff] [blame] | 69 | valgrind_path = use_valgrind and VALGRIND_CMD or '' |
Baptiste Lepilleur | 64e07e5 | 2009-11-18 21:27:06 +0000 | [diff] [blame] | 70 | for input_path in tests + test_jsonchecker: |
Baptiste Lepilleur | 9c98f22 | 2011-05-01 15:40:47 +0000 | [diff] [blame] | 71 | expect_failure = os.path.basename( input_path ).startswith( 'fail' ) |
| 72 | is_json_checker_test = (input_path in test_jsonchecker) or expect_failure |
Christopher Dunn | bd1e895 | 2014-11-19 23:30:47 -0600 | [diff] [blame] | 73 | print('TESTING:', input_path, end=' ') |
Baptiste Lepilleur | 64e07e5 | 2009-11-18 21:27:06 +0000 | [diff] [blame] | 74 | options = is_json_checker_test and '--json-checker' or '' |
Christopher Dunn | 26c5286 | 2015-01-23 11:53:16 -0600 | [diff] [blame^] | 75 | options += ' --json-writer StyledWriter' |
Christopher Dunn | cd140b5 | 2015-01-16 13:44:27 -0600 | [diff] [blame] | 76 | cmd = '%s%s %s "%s"' % ( |
Baptiste Lepilleur | 932cfc7 | 2009-11-19 20:16:59 +0000 | [diff] [blame] | 77 | valgrind_path, jsontest_executable_path, options, |
Christopher Dunn | cd140b5 | 2015-01-16 13:44:27 -0600 | [diff] [blame] | 78 | input_path) |
| 79 | status, process_output = getStatusOutput(cmd) |
Baptiste Lepilleur | 64e07e5 | 2009-11-18 21:27:06 +0000 | [diff] [blame] | 80 | if is_json_checker_test: |
Baptiste Lepilleur | 64e07e5 | 2009-11-18 21:27:06 +0000 | [diff] [blame] | 81 | if expect_failure: |
Christopher Dunn | cd140b5 | 2015-01-16 13:44:27 -0600 | [diff] [blame] | 82 | if not status: |
Christopher Dunn | bd1e895 | 2014-11-19 23:30:47 -0600 | [diff] [blame] | 83 | print('FAILED') |
Baptiste Lepilleur | 8868147 | 2009-11-18 21:38:54 +0000 | [diff] [blame] | 84 | failed_tests.append( (input_path, 'Parsing should have failed:\n%s' % |
| 85 | safeReadFile(input_path)) ) |
Baptiste Lepilleur | 64e07e5 | 2009-11-18 21:27:06 +0000 | [diff] [blame] | 86 | else: |
Christopher Dunn | bd1e895 | 2014-11-19 23:30:47 -0600 | [diff] [blame] | 87 | print('OK') |
Christopher Dunn | f986423 | 2007-06-14 21:01:26 +0000 | [diff] [blame] | 88 | else: |
Christopher Dunn | cd140b5 | 2015-01-16 13:44:27 -0600 | [diff] [blame] | 89 | if status: |
Christopher Dunn | bd1e895 | 2014-11-19 23:30:47 -0600 | [diff] [blame] | 90 | print('FAILED') |
Baptiste Lepilleur | 64e07e5 | 2009-11-18 21:27:06 +0000 | [diff] [blame] | 91 | failed_tests.append( (input_path, 'Parsing failed:\n' + process_output) ) |
| 92 | else: |
Christopher Dunn | bd1e895 | 2014-11-19 23:30:47 -0600 | [diff] [blame] | 93 | print('OK') |
Baptiste Lepilleur | 64e07e5 | 2009-11-18 21:27:06 +0000 | [diff] [blame] | 94 | else: |
| 95 | base_path = os.path.splitext(input_path)[0] |
| 96 | actual_output = safeReadFile( base_path + '.actual' ) |
| 97 | actual_rewrite_output = safeReadFile( base_path + '.actual-rewrite' ) |
datadiode | 01aee4a | 2015-01-11 10:39:24 +0100 | [diff] [blame] | 98 | open(base_path + '.process-output', 'wt', encoding = 'utf-8').write( process_output ) |
Baptiste Lepilleur | 64e07e5 | 2009-11-18 21:27:06 +0000 | [diff] [blame] | 99 | if status: |
Christopher Dunn | bd1e895 | 2014-11-19 23:30:47 -0600 | [diff] [blame] | 100 | print('parsing failed') |
Baptiste Lepilleur | 64e07e5 | 2009-11-18 21:27:06 +0000 | [diff] [blame] | 101 | failed_tests.append( (input_path, 'Parsing failed:\n' + process_output) ) |
| 102 | else: |
| 103 | expected_output_path = os.path.splitext(input_path)[0] + '.expected' |
datadiode | 01aee4a | 2015-01-11 10:39:24 +0100 | [diff] [blame] | 104 | expected_output = open( expected_output_path, 'rt', encoding = 'utf-8' ).read() |
Baptiste Lepilleur | 64e07e5 | 2009-11-18 21:27:06 +0000 | [diff] [blame] | 105 | detail = ( compareOutputs( expected_output, actual_output, 'input' ) |
| 106 | or compareOutputs( expected_output, actual_rewrite_output, 'rewrite' ) ) |
| 107 | if detail: |
Christopher Dunn | bd1e895 | 2014-11-19 23:30:47 -0600 | [diff] [blame] | 108 | print('FAILED') |
Baptiste Lepilleur | 64e07e5 | 2009-11-18 21:27:06 +0000 | [diff] [blame] | 109 | failed_tests.append( (input_path, detail) ) |
| 110 | else: |
Christopher Dunn | bd1e895 | 2014-11-19 23:30:47 -0600 | [diff] [blame] | 111 | print('OK') |
Christopher Dunn | f986423 | 2007-06-14 21:01:26 +0000 | [diff] [blame] | 112 | |
| 113 | if failed_tests: |
Christopher Dunn | bd1e895 | 2014-11-19 23:30:47 -0600 | [diff] [blame] | 114 | print() |
| 115 | print('Failure details:') |
Christopher Dunn | f986423 | 2007-06-14 21:01:26 +0000 | [diff] [blame] | 116 | for failed_test in failed_tests: |
Christopher Dunn | bd1e895 | 2014-11-19 23:30:47 -0600 | [diff] [blame] | 117 | print('* Test', failed_test[0]) |
| 118 | print(failed_test[1]) |
| 119 | print() |
| 120 | print('Test results: %d passed, %d failed.' % (len(tests)-len(failed_tests), |
| 121 | len(failed_tests) )) |
Christopher Dunn | f986423 | 2007-06-14 21:01:26 +0000 | [diff] [blame] | 122 | return 1 |
| 123 | else: |
Christopher Dunn | bd1e895 | 2014-11-19 23:30:47 -0600 | [diff] [blame] | 124 | print('All %d tests passed.' % len(tests)) |
Christopher Dunn | f986423 | 2007-06-14 21:01:26 +0000 | [diff] [blame] | 125 | return 0 |
| 126 | |
Baptiste Lepilleur | 932cfc7 | 2009-11-19 20:16:59 +0000 | [diff] [blame] | 127 | def main(): |
| 128 | from optparse import OptionParser |
| 129 | parser = OptionParser( usage="%prog [options] <path to jsontestrunner.exe> [test case directory]" ) |
| 130 | parser.add_option("--valgrind", |
| 131 | action="store_true", dest="valgrind", default=False, |
| 132 | help="run all the tests using valgrind to detect memory leaks") |
Baptiste Lepilleur | 7c66ac2 | 2010-02-21 14:26:08 +0000 | [diff] [blame] | 133 | parser.add_option("-c", "--with-json-checker", |
| 134 | action="store_true", dest="with_json_checker", default=False, |
| 135 | help="run all the tests from the official JSONChecker test suite of json.org") |
Baptiste Lepilleur | 932cfc7 | 2009-11-19 20:16:59 +0000 | [diff] [blame] | 136 | parser.enable_interspersed_args() |
| 137 | options, args = parser.parse_args() |
| 138 | |
| 139 | if len(args) < 1 or len(args) > 2: |
Baptiste Lepilleur | 45c499d | 2009-11-21 18:07:09 +0000 | [diff] [blame] | 140 | parser.error( 'Must provides at least path to jsontestrunner executable.' ) |
Christopher Dunn | f986423 | 2007-06-14 21:01:26 +0000 | [diff] [blame] | 141 | sys.exit( 1 ) |
| 142 | |
Baptiste Lepilleur | 932cfc7 | 2009-11-19 20:16:59 +0000 | [diff] [blame] | 143 | jsontest_executable_path = os.path.normpath( os.path.abspath( args[0] ) ) |
| 144 | if len(args) > 1: |
| 145 | input_path = os.path.normpath( os.path.abspath( args[1] ) ) |
Christopher Dunn | f986423 | 2007-06-14 21:01:26 +0000 | [diff] [blame] | 146 | else: |
| 147 | input_path = None |
Baptiste Lepilleur | 932cfc7 | 2009-11-19 20:16:59 +0000 | [diff] [blame] | 148 | status = runAllTests( jsontest_executable_path, input_path, |
Baptiste Lepilleur | 7c66ac2 | 2010-02-21 14:26:08 +0000 | [diff] [blame] | 149 | use_valgrind=options.valgrind, with_json_checker=options.with_json_checker ) |
Baptiste Lepilleur | 932cfc7 | 2009-11-19 20:16:59 +0000 | [diff] [blame] | 150 | sys.exit( status ) |
| 151 | |
| 152 | if __name__ == '__main__': |
| 153 | main() |