Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 1 | #! /usr/bin/env python |
| 2 | |
| 3 | """Regression test. |
| 4 | |
| 5 | This will find all modules whose name is "test_*" in the test |
| 6 | directory, and run them. Various command line options provide |
| 7 | additional facilities. |
| 8 | |
| 9 | Command line options: |
| 10 | |
Barry Warsaw | a873b03 | 2000-08-03 15:50:37 +0000 | [diff] [blame] | 11 | -v: verbose -- run tests in verbose mode with output to stdout |
| 12 | -q: quiet -- don't print anything except if a test fails |
| 13 | -g: generate -- write the output file for a test instead of comparing it |
| 14 | -x: exclude -- arguments are tests to *exclude* |
| 15 | -s: single -- run only a single test (see below) |
| 16 | -r: random -- randomize test execution order |
| 17 | -l: leakdebug -- if cycle garbage collection is enabled, run with DEBUG_LEAK |
Trent Mick | f29f47b | 2000-08-11 19:02:59 +0000 | [diff] [blame] | 18 | --have-resources -- run tests that require large resources (time/space) |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 19 | |
| 20 | If non-option arguments are present, they are names for tests to run, |
| 21 | unless -x is given, in which case they are names for tests not to run. |
| 22 | If no test names are given, all tests are run. |
Guido van Rossum | f58ed25 | 1997-03-07 21:04:33 +0000 | [diff] [blame] | 23 | |
Guido van Rossum | a412220 | 1997-08-18 20:08:24 +0000 | [diff] [blame] | 24 | -v is incompatible with -g and does not compare test output files. |
Barry Warsaw | e11e3de | 1999-01-28 19:51:51 +0000 | [diff] [blame] | 25 | |
| 26 | -s means to run only a single test and exit. This is useful when Purifying |
| 27 | the Python interpreter. The file /tmp/pynexttest is read to find the next |
| 28 | test to run. If this file is missing, the first test_*.py file in testdir or |
| 29 | on the command line is used. (actually tempfile.gettempdir() is used instead |
| 30 | of /tmp). |
| 31 | |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 32 | """ |
| 33 | |
| 34 | import sys |
| 35 | import string |
| 36 | import os |
| 37 | import getopt |
Guido van Rossum | 9e48b27 | 1997-07-16 01:56:13 +0000 | [diff] [blame] | 38 | import traceback |
Skip Montanaro | ab1c791 | 2000-06-30 16:39:27 +0000 | [diff] [blame] | 39 | import random |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 40 | |
| 41 | import test_support |
| 42 | |
Skip Montanaro | ab1c791 | 2000-06-30 16:39:27 +0000 | [diff] [blame] | 43 | def main(tests=None, testdir=None, verbose=0, quiet=0, generate=0, |
Trent Mick | f29f47b | 2000-08-11 19:02:59 +0000 | [diff] [blame] | 44 | exclude=0, single=0, randomize=0, leakdebug=0, |
| 45 | use_large_resources=0): |
Guido van Rossum | 6fd83b7 | 1998-08-01 17:04:08 +0000 | [diff] [blame] | 46 | """Execute a test suite. |
| 47 | |
Thomas Wouters | 7e47402 | 2000-07-16 12:04:32 +0000 | [diff] [blame] | 48 | This also parses command-line options and modifies its behavior |
Guido van Rossum | 6fd83b7 | 1998-08-01 17:04:08 +0000 | [diff] [blame] | 49 | accordingly. |
| 50 | |
| 51 | tests -- a list of strings containing test names (optional) |
| 52 | testdir -- the directory in which to look for tests (optional) |
| 53 | |
| 54 | Users other than the Python test suite will certainly want to |
| 55 | specify testdir; if it's omitted, the directory containing the |
| 56 | Python test suite is searched for. |
| 57 | |
| 58 | If the tests argument is omitted, the tests listed on the |
| 59 | command-line will be used. If that's empty, too, then all *.py |
| 60 | files beginning with test_ will be used. |
Skip Montanaro | ab1c791 | 2000-06-30 16:39:27 +0000 | [diff] [blame] | 61 | |
Barry Warsaw | a873b03 | 2000-08-03 15:50:37 +0000 | [diff] [blame] | 62 | The other seven default arguments (verbose, quiet, generate, exclude, |
| 63 | single, randomize, and leakdebug) allow programmers calling main() |
| 64 | directly to set the values that would normally be set by flags on the |
| 65 | command line. |
| 66 | |
Guido van Rossum | 6fd83b7 | 1998-08-01 17:04:08 +0000 | [diff] [blame] | 67 | """ |
| 68 | |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 69 | try: |
Trent Mick | f29f47b | 2000-08-11 19:02:59 +0000 | [diff] [blame] | 70 | opts, args = getopt.getopt(sys.argv[1:], 'vgqxsrl', ['have-resources']) |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 71 | except getopt.error, msg: |
Guido van Rossum | 41360a4 | 1998-03-26 19:42:58 +0000 | [diff] [blame] | 72 | print msg |
| 73 | print __doc__ |
| 74 | return 2 |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 75 | for o, a in opts: |
Guido van Rossum | 41360a4 | 1998-03-26 19:42:58 +0000 | [diff] [blame] | 76 | if o == '-v': verbose = verbose+1 |
| 77 | if o == '-q': quiet = 1; verbose = 0 |
| 78 | if o == '-g': generate = 1 |
| 79 | if o == '-x': exclude = 1 |
Barry Warsaw | e11e3de | 1999-01-28 19:51:51 +0000 | [diff] [blame] | 80 | if o == '-s': single = 1 |
Skip Montanaro | ab1c791 | 2000-06-30 16:39:27 +0000 | [diff] [blame] | 81 | if o == '-r': randomize = 1 |
Barry Warsaw | a873b03 | 2000-08-03 15:50:37 +0000 | [diff] [blame] | 82 | if o == '-l': leakdebug = 1 |
Trent Mick | f29f47b | 2000-08-11 19:02:59 +0000 | [diff] [blame] | 83 | if o == '--have-resources': use_large_resources = 1 |
Guido van Rossum | a412220 | 1997-08-18 20:08:24 +0000 | [diff] [blame] | 84 | if generate and verbose: |
Guido van Rossum | 41360a4 | 1998-03-26 19:42:58 +0000 | [diff] [blame] | 85 | print "-g and -v don't go together!" |
| 86 | return 2 |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 87 | good = [] |
| 88 | bad = [] |
| 89 | skipped = [] |
Barry Warsaw | e11e3de | 1999-01-28 19:51:51 +0000 | [diff] [blame] | 90 | |
Barry Warsaw | a873b03 | 2000-08-03 15:50:37 +0000 | [diff] [blame] | 91 | if leakdebug: |
| 92 | try: |
| 93 | import gc |
| 94 | except ImportError: |
| 95 | print 'cycle garbage collection not available' |
| 96 | else: |
| 97 | gc.set_debug(gc.DEBUG_LEAK) |
| 98 | |
Barry Warsaw | e11e3de | 1999-01-28 19:51:51 +0000 | [diff] [blame] | 99 | if single: |
| 100 | from tempfile import gettempdir |
| 101 | filename = os.path.join(gettempdir(), 'pynexttest') |
| 102 | try: |
| 103 | fp = open(filename, 'r') |
| 104 | next = string.strip(fp.read()) |
| 105 | tests = [next] |
| 106 | fp.close() |
| 107 | except IOError: |
| 108 | pass |
Guido van Rossum | a412220 | 1997-08-18 20:08:24 +0000 | [diff] [blame] | 109 | for i in range(len(args)): |
Guido van Rossum | 41360a4 | 1998-03-26 19:42:58 +0000 | [diff] [blame] | 110 | # Strip trailing ".py" from arguments |
| 111 | if args[i][-3:] == '.py': |
| 112 | args[i] = args[i][:-3] |
Guido van Rossum | 6c74fea | 1998-08-25 12:29:08 +0000 | [diff] [blame] | 113 | stdtests = STDTESTS[:] |
| 114 | nottests = NOTTESTS[:] |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 115 | if exclude: |
Guido van Rossum | 6c74fea | 1998-08-25 12:29:08 +0000 | [diff] [blame] | 116 | for arg in args: |
| 117 | if arg in stdtests: |
| 118 | stdtests.remove(arg) |
| 119 | nottests[:0] = args |
Guido van Rossum | 41360a4 | 1998-03-26 19:42:58 +0000 | [diff] [blame] | 120 | args = [] |
Guido van Rossum | 747e1ca | 1998-08-24 13:48:36 +0000 | [diff] [blame] | 121 | tests = tests or args or findtests(testdir, stdtests, nottests) |
Barry Warsaw | e11e3de | 1999-01-28 19:51:51 +0000 | [diff] [blame] | 122 | if single: |
| 123 | tests = tests[:1] |
Skip Montanaro | ab1c791 | 2000-06-30 16:39:27 +0000 | [diff] [blame] | 124 | if randomize: |
| 125 | random.shuffle(tests) |
Guido van Rossum | 41360a4 | 1998-03-26 19:42:58 +0000 | [diff] [blame] | 126 | test_support.verbose = verbose # Tell tests to be moderately quiet |
Trent Mick | f29f47b | 2000-08-11 19:02:59 +0000 | [diff] [blame] | 127 | test_support.use_large_resources = use_large_resources |
Guido van Rossum | 5796d26 | 2000-04-21 21:35:06 +0000 | [diff] [blame] | 128 | save_modules = sys.modules.keys() |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 129 | for test in tests: |
Guido van Rossum | 41360a4 | 1998-03-26 19:42:58 +0000 | [diff] [blame] | 130 | if not quiet: |
| 131 | print test |
Trent Mick | f29f47b | 2000-08-11 19:02:59 +0000 | [diff] [blame] | 132 | ok = runtest(test, generate, verbose, quiet, testdir) |
Guido van Rossum | 41360a4 | 1998-03-26 19:42:58 +0000 | [diff] [blame] | 133 | if ok > 0: |
| 134 | good.append(test) |
| 135 | elif ok == 0: |
| 136 | bad.append(test) |
| 137 | else: |
Guido van Rossum | 41360a4 | 1998-03-26 19:42:58 +0000 | [diff] [blame] | 138 | skipped.append(test) |
Guido van Rossum | 5796d26 | 2000-04-21 21:35:06 +0000 | [diff] [blame] | 139 | # Unload the newly imported modules (best effort finalization) |
| 140 | for module in sys.modules.keys(): |
Guido van Rossum | 5193114 | 2000-05-05 14:27:39 +0000 | [diff] [blame] | 141 | if module not in save_modules and module.startswith("test."): |
Guido van Rossum | 5796d26 | 2000-04-21 21:35:06 +0000 | [diff] [blame] | 142 | test_support.unload(module) |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 143 | if good and not quiet: |
Guido van Rossum | 41360a4 | 1998-03-26 19:42:58 +0000 | [diff] [blame] | 144 | if not bad and not skipped and len(good) > 1: |
| 145 | print "All", |
| 146 | print count(len(good), "test"), "OK." |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 147 | if bad: |
Guido van Rossum | 41360a4 | 1998-03-26 19:42:58 +0000 | [diff] [blame] | 148 | print count(len(bad), "test"), "failed:", |
| 149 | print string.join(bad) |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 150 | if skipped and not quiet: |
Guido van Rossum | 41360a4 | 1998-03-26 19:42:58 +0000 | [diff] [blame] | 151 | print count(len(skipped), "test"), "skipped:", |
| 152 | print string.join(skipped) |
Barry Warsaw | e11e3de | 1999-01-28 19:51:51 +0000 | [diff] [blame] | 153 | |
| 154 | if single: |
| 155 | alltests = findtests(testdir, stdtests, nottests) |
| 156 | for i in range(len(alltests)): |
| 157 | if tests[0] == alltests[i]: |
| 158 | if i == len(alltests) - 1: |
| 159 | os.unlink(filename) |
| 160 | else: |
| 161 | fp = open(filename, 'w') |
| 162 | fp.write(alltests[i+1] + '\n') |
| 163 | fp.close() |
| 164 | break |
| 165 | else: |
| 166 | os.unlink(filename) |
| 167 | |
Guido van Rossum | e838701 | 1997-08-14 19:40:34 +0000 | [diff] [blame] | 168 | return len(bad) > 0 |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 169 | |
Guido van Rossum | 6fd83b7 | 1998-08-01 17:04:08 +0000 | [diff] [blame] | 170 | STDTESTS = [ |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 171 | 'test_grammar', |
| 172 | 'test_opcodes', |
| 173 | 'test_operations', |
| 174 | 'test_builtin', |
| 175 | 'test_exceptions', |
| 176 | 'test_types', |
| 177 | ] |
| 178 | |
Guido van Rossum | 6fd83b7 | 1998-08-01 17:04:08 +0000 | [diff] [blame] | 179 | NOTTESTS = [ |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 180 | 'test_support', |
| 181 | 'test_b1', |
| 182 | 'test_b2', |
| 183 | ] |
| 184 | |
Guido van Rossum | 6fd83b7 | 1998-08-01 17:04:08 +0000 | [diff] [blame] | 185 | def findtests(testdir=None, stdtests=STDTESTS, nottests=NOTTESTS): |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 186 | """Return a list of all applicable test modules.""" |
Guido van Rossum | 6fd83b7 | 1998-08-01 17:04:08 +0000 | [diff] [blame] | 187 | if not testdir: testdir = findtestdir() |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 188 | names = os.listdir(testdir) |
| 189 | tests = [] |
| 190 | for name in names: |
Guido van Rossum | 41360a4 | 1998-03-26 19:42:58 +0000 | [diff] [blame] | 191 | if name[:5] == "test_" and name[-3:] == ".py": |
| 192 | modname = name[:-3] |
| 193 | if modname not in stdtests and modname not in nottests: |
| 194 | tests.append(modname) |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 195 | tests.sort() |
| 196 | return stdtests + tests |
| 197 | |
Trent Mick | f29f47b | 2000-08-11 19:02:59 +0000 | [diff] [blame] | 198 | def runtest(test, generate, verbose, quiet, testdir = None): |
Guido van Rossum | 6fd83b7 | 1998-08-01 17:04:08 +0000 | [diff] [blame] | 199 | """Run a single test. |
| 200 | test -- the name of the test |
| 201 | generate -- if true, generate output, instead of running the test |
| 202 | and comparing it to a previously created output file |
| 203 | verbose -- if true, print more messages |
Trent Mick | f29f47b | 2000-08-11 19:02:59 +0000 | [diff] [blame] | 204 | quiet -- if true, don't print 'skipped' messages (probably redundant) |
Guido van Rossum | 6fd83b7 | 1998-08-01 17:04:08 +0000 | [diff] [blame] | 205 | testdir -- test directory |
| 206 | """ |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 207 | test_support.unload(test) |
Guido van Rossum | 6fd83b7 | 1998-08-01 17:04:08 +0000 | [diff] [blame] | 208 | if not testdir: testdir = findtestdir() |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 209 | outputdir = os.path.join(testdir, "output") |
| 210 | outputfile = os.path.join(outputdir, test) |
| 211 | try: |
Guido van Rossum | 41360a4 | 1998-03-26 19:42:58 +0000 | [diff] [blame] | 212 | if generate: |
| 213 | cfp = open(outputfile, "w") |
| 214 | elif verbose: |
| 215 | cfp = sys.stdout |
| 216 | else: |
| 217 | cfp = Compare(outputfile) |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 218 | except IOError: |
Guido van Rossum | 41360a4 | 1998-03-26 19:42:58 +0000 | [diff] [blame] | 219 | cfp = None |
| 220 | print "Warning: can't open", outputfile |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 221 | try: |
Guido van Rossum | 41360a4 | 1998-03-26 19:42:58 +0000 | [diff] [blame] | 222 | save_stdout = sys.stdout |
| 223 | try: |
| 224 | if cfp: |
| 225 | sys.stdout = cfp |
| 226 | print test # Output file starts with test name |
| 227 | __import__(test, globals(), locals(), []) |
Jeremy Hylton | fff9e20 | 2000-07-11 15:15:31 +0000 | [diff] [blame] | 228 | if cfp and not (generate or verbose): |
| 229 | cfp.close() |
Guido van Rossum | 41360a4 | 1998-03-26 19:42:58 +0000 | [diff] [blame] | 230 | finally: |
| 231 | sys.stdout = save_stdout |
Thomas Wouters | 3af826e | 2000-08-04 13:17:51 +0000 | [diff] [blame] | 232 | except (ImportError, test_support.TestSkipped), msg: |
Trent Mick | f29f47b | 2000-08-11 19:02:59 +0000 | [diff] [blame] | 233 | if not quiet: |
| 234 | print "test", test, |
| 235 | print "skipped -- ", msg |
Guido van Rossum | 41360a4 | 1998-03-26 19:42:58 +0000 | [diff] [blame] | 236 | return -1 |
Fred Drake | fe5c22a | 2000-08-18 16:04:05 +0000 | [diff] [blame] | 237 | except KeyboardInterrupt: |
| 238 | raise |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 239 | except test_support.TestFailed, msg: |
Guido van Rossum | 41360a4 | 1998-03-26 19:42:58 +0000 | [diff] [blame] | 240 | print "test", test, "failed --", msg |
| 241 | return 0 |
Guido van Rossum | 9e48b27 | 1997-07-16 01:56:13 +0000 | [diff] [blame] | 242 | except: |
Guido van Rossum | 41360a4 | 1998-03-26 19:42:58 +0000 | [diff] [blame] | 243 | type, value = sys.exc_info()[:2] |
Fred Drake | 27c4b39 | 2000-08-23 20:34:40 +0000 | [diff] [blame] | 244 | print "test", test, "crashed --", str(type) + ":", value |
Guido van Rossum | 41360a4 | 1998-03-26 19:42:58 +0000 | [diff] [blame] | 245 | if verbose: |
| 246 | traceback.print_exc(file=sys.stdout) |
| 247 | return 0 |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 248 | else: |
Guido van Rossum | 41360a4 | 1998-03-26 19:42:58 +0000 | [diff] [blame] | 249 | return 1 |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 250 | |
| 251 | def findtestdir(): |
| 252 | if __name__ == '__main__': |
Guido van Rossum | 41360a4 | 1998-03-26 19:42:58 +0000 | [diff] [blame] | 253 | file = sys.argv[0] |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 254 | else: |
Guido van Rossum | 41360a4 | 1998-03-26 19:42:58 +0000 | [diff] [blame] | 255 | file = __file__ |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 256 | testdir = os.path.dirname(file) or os.curdir |
| 257 | return testdir |
| 258 | |
| 259 | def count(n, word): |
| 260 | if n == 1: |
Guido van Rossum | 41360a4 | 1998-03-26 19:42:58 +0000 | [diff] [blame] | 261 | return "%d %s" % (n, word) |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 262 | else: |
Guido van Rossum | 41360a4 | 1998-03-26 19:42:58 +0000 | [diff] [blame] | 263 | return "%d %ss" % (n, word) |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 264 | |
| 265 | class Compare: |
| 266 | |
| 267 | def __init__(self, filename): |
Guido van Rossum | 41360a4 | 1998-03-26 19:42:58 +0000 | [diff] [blame] | 268 | self.fp = open(filename, 'r') |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 269 | |
| 270 | def write(self, data): |
Guido van Rossum | 41360a4 | 1998-03-26 19:42:58 +0000 | [diff] [blame] | 271 | expected = self.fp.read(len(data)) |
| 272 | if data <> expected: |
| 273 | raise test_support.TestFailed, \ |
| 274 | 'Writing: '+`data`+', expected: '+`expected` |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 275 | |
Guido van Rossum | e87ed5f | 1998-04-23 13:33:21 +0000 | [diff] [blame] | 276 | def writelines(self, listoflines): |
| 277 | map(self.write, listoflines) |
| 278 | |
Guido van Rossum | 75fce30 | 1997-07-17 14:51:37 +0000 | [diff] [blame] | 279 | def flush(self): |
Guido van Rossum | 41360a4 | 1998-03-26 19:42:58 +0000 | [diff] [blame] | 280 | pass |
Guido van Rossum | 75fce30 | 1997-07-17 14:51:37 +0000 | [diff] [blame] | 281 | |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 282 | def close(self): |
Guido van Rossum | 41360a4 | 1998-03-26 19:42:58 +0000 | [diff] [blame] | 283 | leftover = self.fp.read() |
| 284 | if leftover: |
| 285 | raise test_support.TestFailed, 'Unread: '+`leftover` |
| 286 | self.fp.close() |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 287 | |
| 288 | def isatty(self): |
Guido van Rossum | 41360a4 | 1998-03-26 19:42:58 +0000 | [diff] [blame] | 289 | return 0 |
Guido van Rossum | 152494a | 1996-12-20 03:12:20 +0000 | [diff] [blame] | 290 | |
| 291 | if __name__ == '__main__': |
Guido van Rossum | e838701 | 1997-08-14 19:40:34 +0000 | [diff] [blame] | 292 | sys.exit(main()) |