| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  | #! /usr/bin/env python | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | """Regression test.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | This will find all modules whose name is "test_*" in the test | 
					
						
							|  |  |  | directory, and run them.  Various command line options provide | 
					
						
							|  |  |  | additional facilities. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Command line options: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-01-28 19:51:51 +00:00
										 |  |  | -v: verbose  -- run tests in verbose mode with output to stdout | 
					
						
							|  |  |  | -q: quiet    -- don't print anything except if a test fails | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  | -g: generate -- write the output file for a test instead of comparing it | 
					
						
							| 
									
										
										
										
											1999-01-28 19:51:51 +00:00
										 |  |  | -x: exclude  -- arguments are tests to *exclude* | 
					
						
							|  |  |  | -s: single   -- run only a single test (see below) | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | If non-option arguments are present, they are names for tests to run, | 
					
						
							|  |  |  | unless -x is given, in which case they are names for tests not to run. | 
					
						
							|  |  |  | If no test names are given, all tests are run. | 
					
						
							| 
									
										
										
										
											1997-03-07 21:04:33 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-08-18 20:08:24 +00:00
										 |  |  | -v is incompatible with -g and does not compare test output files. | 
					
						
							| 
									
										
										
										
											1999-01-28 19:51:51 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | -s means to run only a single test and exit.  This is useful when Purifying | 
					
						
							|  |  |  | the Python interpreter.  The file /tmp/pynexttest is read to find the next | 
					
						
							|  |  |  | test to run.  If this file is missing, the first test_*.py file in testdir or | 
					
						
							|  |  |  | on the command line is used.  (actually tempfile.gettempdir() is used instead | 
					
						
							|  |  |  | of /tmp). | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  | """
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import sys | 
					
						
							|  |  |  | import string | 
					
						
							|  |  |  | import os | 
					
						
							|  |  |  | import getopt | 
					
						
							| 
									
										
										
										
											1997-07-16 01:56:13 +00:00
										 |  |  | import traceback | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | import test_support | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-08-01 17:04:08 +00:00
										 |  |  | def main(tests=None, testdir=None): | 
					
						
							|  |  |  |     """Execute a test suite.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     This also parses command-line options and modifies its behaviour | 
					
						
							|  |  |  |     accordingly.  | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     tests -- a list of strings containing test names (optional) | 
					
						
							|  |  |  |     testdir -- the directory in which to look for tests (optional) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Users other than the Python test suite will certainly want to | 
					
						
							|  |  |  |     specify testdir; if it's omitted, the directory containing the | 
					
						
							|  |  |  |     Python test suite is searched for.   | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     If the tests argument is omitted, the tests listed on the | 
					
						
							|  |  |  |     command-line will be used.  If that's empty, too, then all *.py | 
					
						
							|  |  |  |     files beginning with test_ will be used. | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     try: | 
					
						
							| 
									
										
										
										
											1999-01-28 19:51:51 +00:00
										 |  |  |         opts, args = getopt.getopt(sys.argv[1:], 'vgqxs') | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     except getopt.error, msg: | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         print msg | 
					
						
							|  |  |  |         print __doc__ | 
					
						
							|  |  |  |         return 2 | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     verbose = 0 | 
					
						
							|  |  |  |     quiet = 0 | 
					
						
							|  |  |  |     generate = 0 | 
					
						
							|  |  |  |     exclude = 0 | 
					
						
							| 
									
										
										
										
											1999-01-28 19:51:51 +00:00
										 |  |  |     single = 0 | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     for o, a in opts: | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         if o == '-v': verbose = verbose+1 | 
					
						
							|  |  |  |         if o == '-q': quiet = 1; verbose = 0 | 
					
						
							|  |  |  |         if o == '-g': generate = 1 | 
					
						
							|  |  |  |         if o == '-x': exclude = 1 | 
					
						
							| 
									
										
										
										
											1999-01-28 19:51:51 +00:00
										 |  |  |         if o == '-s': single = 1 | 
					
						
							| 
									
										
										
										
											1997-08-18 20:08:24 +00:00
										 |  |  |     if generate and verbose: | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         print "-g and -v don't go together!" | 
					
						
							|  |  |  |         return 2 | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     good = [] | 
					
						
							|  |  |  |     bad = [] | 
					
						
							|  |  |  |     skipped = [] | 
					
						
							| 
									
										
										
										
											1999-01-28 19:51:51 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if single: | 
					
						
							|  |  |  |         from tempfile import gettempdir | 
					
						
							|  |  |  |         filename = os.path.join(gettempdir(), 'pynexttest') | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             fp = open(filename, 'r') | 
					
						
							|  |  |  |             next = string.strip(fp.read()) | 
					
						
							|  |  |  |             tests = [next] | 
					
						
							|  |  |  |             fp.close() | 
					
						
							|  |  |  |         except IOError: | 
					
						
							|  |  |  |             pass | 
					
						
							| 
									
										
										
										
											1997-08-18 20:08:24 +00:00
										 |  |  |     for i in range(len(args)): | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         # Strip trailing ".py" from arguments | 
					
						
							|  |  |  |         if args[i][-3:] == '.py': | 
					
						
							|  |  |  |             args[i] = args[i][:-3] | 
					
						
							| 
									
										
										
										
											1998-08-25 12:29:08 +00:00
										 |  |  |     stdtests = STDTESTS[:] | 
					
						
							|  |  |  |     nottests = NOTTESTS[:] | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     if exclude: | 
					
						
							| 
									
										
										
										
											1998-08-25 12:29:08 +00:00
										 |  |  |         for arg in args: | 
					
						
							|  |  |  |             if arg in stdtests: | 
					
						
							|  |  |  |                 stdtests.remove(arg) | 
					
						
							|  |  |  |         nottests[:0] = args | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         args = [] | 
					
						
							| 
									
										
										
										
											1998-08-24 13:48:36 +00:00
										 |  |  |     tests = tests or args or findtests(testdir, stdtests, nottests) | 
					
						
							| 
									
										
										
										
											1999-01-28 19:51:51 +00:00
										 |  |  |     if single: | 
					
						
							|  |  |  |         tests = tests[:1] | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |     test_support.verbose = verbose      # Tell tests to be moderately quiet | 
					
						
							| 
									
										
										
										
											2000-04-21 21:35:06 +00:00
										 |  |  |     save_modules = sys.modules.keys() | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     for test in tests: | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         if not quiet: | 
					
						
							|  |  |  |             print test | 
					
						
							| 
									
										
										
										
											1998-08-01 17:04:08 +00:00
										 |  |  |         ok = runtest(test, generate, verbose, testdir) | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         if ok > 0: | 
					
						
							|  |  |  |             good.append(test) | 
					
						
							|  |  |  |         elif ok == 0: | 
					
						
							|  |  |  |             bad.append(test) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             if not quiet: | 
					
						
							|  |  |  |                 print "test", test, | 
					
						
							|  |  |  |                 print "skipped -- an optional feature could not be imported" | 
					
						
							|  |  |  |             skipped.append(test) | 
					
						
							| 
									
										
										
										
											2000-04-21 21:35:06 +00:00
										 |  |  |         # Unload the newly imported modules (best effort finalization) | 
					
						
							|  |  |  |         for module in sys.modules.keys(): | 
					
						
							| 
									
										
										
										
											2000-05-05 14:27:39 +00:00
										 |  |  |             if module not in save_modules and module.startswith("test."): | 
					
						
							| 
									
										
										
										
											2000-04-21 21:35:06 +00:00
										 |  |  |                 test_support.unload(module) | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     if good and not quiet: | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         if not bad and not skipped and len(good) > 1: | 
					
						
							|  |  |  |             print "All", | 
					
						
							|  |  |  |         print count(len(good), "test"), "OK." | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     if bad: | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         print count(len(bad), "test"), "failed:", | 
					
						
							|  |  |  |         print string.join(bad) | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     if skipped and not quiet: | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         print count(len(skipped), "test"), "skipped:", | 
					
						
							|  |  |  |         print string.join(skipped) | 
					
						
							| 
									
										
										
										
											1999-01-28 19:51:51 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if single: | 
					
						
							|  |  |  |         alltests = findtests(testdir, stdtests, nottests) | 
					
						
							|  |  |  |         for i in range(len(alltests)): | 
					
						
							|  |  |  |             if tests[0] == alltests[i]: | 
					
						
							|  |  |  |                 if i == len(alltests) - 1: | 
					
						
							|  |  |  |                     os.unlink(filename) | 
					
						
							|  |  |  |                 else: | 
					
						
							|  |  |  |                     fp = open(filename, 'w') | 
					
						
							|  |  |  |                     fp.write(alltests[i+1] + '\n') | 
					
						
							|  |  |  |                     fp.close() | 
					
						
							|  |  |  |                 break | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             os.unlink(filename) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-08-14 19:40:34 +00:00
										 |  |  |     return len(bad) > 0 | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-08-01 17:04:08 +00:00
										 |  |  | STDTESTS = [ | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     'test_grammar', | 
					
						
							|  |  |  |     'test_opcodes', | 
					
						
							|  |  |  |     'test_operations', | 
					
						
							|  |  |  |     'test_builtin', | 
					
						
							|  |  |  |     'test_exceptions', | 
					
						
							|  |  |  |     'test_types', | 
					
						
							|  |  |  |    ] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-08-01 17:04:08 +00:00
										 |  |  | NOTTESTS = [ | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     'test_support', | 
					
						
							|  |  |  |     'test_b1', | 
					
						
							|  |  |  |     'test_b2', | 
					
						
							|  |  |  |     ] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-08-01 17:04:08 +00:00
										 |  |  | def findtests(testdir=None, stdtests=STDTESTS, nottests=NOTTESTS): | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     """Return a list of all applicable test modules.""" | 
					
						
							| 
									
										
										
										
											1998-08-01 17:04:08 +00:00
										 |  |  |     if not testdir: testdir = findtestdir() | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     names = os.listdir(testdir) | 
					
						
							|  |  |  |     tests = [] | 
					
						
							|  |  |  |     for name in names: | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         if name[:5] == "test_" and name[-3:] == ".py": | 
					
						
							|  |  |  |             modname = name[:-3] | 
					
						
							|  |  |  |             if modname not in stdtests and modname not in nottests: | 
					
						
							|  |  |  |                 tests.append(modname) | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     tests.sort() | 
					
						
							|  |  |  |     return stdtests + tests | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-08-01 17:04:08 +00:00
										 |  |  | def runtest(test, generate, verbose, testdir = None): | 
					
						
							|  |  |  |     """Run a single test.
 | 
					
						
							|  |  |  |     test -- the name of the test | 
					
						
							|  |  |  |     generate -- if true, generate output, instead of running the test | 
					
						
							|  |  |  |     and comparing it to a previously created output file | 
					
						
							|  |  |  |     verbose -- if true, print more messages | 
					
						
							|  |  |  |     testdir -- test directory | 
					
						
							|  |  |  |     """
 | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     test_support.unload(test) | 
					
						
							| 
									
										
										
										
											1998-08-01 17:04:08 +00:00
										 |  |  |     if not testdir: testdir = findtestdir() | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     outputdir = os.path.join(testdir, "output") | 
					
						
							|  |  |  |     outputfile = os.path.join(outputdir, test) | 
					
						
							|  |  |  |     try: | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         if generate: | 
					
						
							|  |  |  |             cfp = open(outputfile, "w") | 
					
						
							|  |  |  |         elif verbose: | 
					
						
							|  |  |  |             cfp = sys.stdout | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             cfp = Compare(outputfile) | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     except IOError: | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         cfp = None | 
					
						
							|  |  |  |         print "Warning: can't open", outputfile | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     try: | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         save_stdout = sys.stdout | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             if cfp: | 
					
						
							|  |  |  |                 sys.stdout = cfp | 
					
						
							|  |  |  |                 print test              # Output file starts with test name | 
					
						
							|  |  |  |             __import__(test, globals(), locals(), []) | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             sys.stdout = save_stdout | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     except ImportError, msg: | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         return -1 | 
					
						
							| 
									
										
										
										
											1997-10-20 23:46:54 +00:00
										 |  |  |     except KeyboardInterrupt, v: | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         raise KeyboardInterrupt, v, sys.exc_info()[2] | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     except test_support.TestFailed, msg: | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         print "test", test, "failed --", msg | 
					
						
							|  |  |  |         return 0 | 
					
						
							| 
									
										
										
										
											1997-07-16 01:56:13 +00:00
										 |  |  |     except: | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         type, value = sys.exc_info()[:2] | 
					
						
							|  |  |  |         print "test", test, "crashed --", type, ":", value | 
					
						
							|  |  |  |         if verbose: | 
					
						
							|  |  |  |             traceback.print_exc(file=sys.stdout) | 
					
						
							|  |  |  |         return 0 | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     else: | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         return 1 | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | def findtestdir(): | 
					
						
							|  |  |  |     if __name__ == '__main__': | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         file = sys.argv[0] | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     else: | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         file = __file__ | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     testdir = os.path.dirname(file) or os.curdir | 
					
						
							|  |  |  |     return testdir | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def count(n, word): | 
					
						
							|  |  |  |     if n == 1: | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         return "%d %s" % (n, word) | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     else: | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         return "%d %ss" % (n, word) | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | class Compare: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, filename): | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         self.fp = open(filename, 'r') | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def write(self, data): | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         expected = self.fp.read(len(data)) | 
					
						
							|  |  |  |         if data <> expected: | 
					
						
							|  |  |  |             raise test_support.TestFailed, \ | 
					
						
							|  |  |  |                     'Writing: '+`data`+', expected: '+`expected` | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-23 13:33:21 +00:00
										 |  |  |     def writelines(self, listoflines): | 
					
						
							|  |  |  |         map(self.write, listoflines) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-07-17 14:51:37 +00:00
										 |  |  |     def flush(self): | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         pass | 
					
						
							| 
									
										
										
										
											1997-07-17 14:51:37 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     def close(self): | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         leftover = self.fp.read() | 
					
						
							|  |  |  |         if leftover: | 
					
						
							|  |  |  |             raise test_support.TestFailed, 'Unread: '+`leftover` | 
					
						
							|  |  |  |         self.fp.close() | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def isatty(self): | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         return 0 | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | if __name__ == '__main__': | 
					
						
							| 
									
										
										
										
											1997-08-14 19:40:34 +00:00
										 |  |  |     sys.exit(main()) |