| 
									
										
										
										
											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: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-03 15:50:37 +00:00
										 |  |  | -v: verbose   -- run tests in verbose mode with output to stdout | 
					
						
							|  |  |  | -q: quiet     -- don't print anything except if a test fails | 
					
						
							|  |  |  | -g: generate  -- write the output file for a test instead of comparing it | 
					
						
							|  |  |  | -x: exclude   -- arguments are tests to *exclude* | 
					
						
							|  |  |  | -s: single    -- run only a single test (see below) | 
					
						
							|  |  |  | -r: random    -- randomize test execution order | 
					
						
							| 
									
										
										
										
											2000-10-13 01:32:42 +00:00
										 |  |  | -l: findleaks -- if GC is available detect tests that leak memory | 
					
						
							| 
									
										
										
										
											2001-08-20 22:33:46 +00:00
										 |  |  | -u: use       -- specify which special resource intensive tests to run | 
					
						
							|  |  |  | -h: help      -- print this text and exit | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-02-23 18:31:40 +00:00
										 |  |  | -s means to run only a single test and exit.  This is useful when doing memory | 
					
						
							|  |  |  | analysis on the Python interpreter (which tend to consume to many resources to | 
					
						
							|  |  |  | run the full regression test non-stop).  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). | 
					
						
							| 
									
										
										
										
											1999-01-28 19:51:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-08-20 22:33:46 +00:00
										 |  |  | -u is used to specify which special resource intensive tests to run, such as | 
					
						
							|  |  |  | those requiring large file support or network connectivity.  The argument is a | 
					
						
							|  |  |  | comma-separated list of words indicating the resources to test.  Currently | 
					
						
							|  |  |  | only the following are defined: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     largefile - It is okay to run some test that may create huge files.  These | 
					
						
							|  |  |  |                 tests can take a long time and may consume >2GB of disk space | 
					
						
							|  |  |  |                 temporarily. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     network -   It is okay to run tests that use external network resource, | 
					
						
							|  |  |  |                 e.g. testing SSL support for sockets. | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  | """
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import sys | 
					
						
							|  |  |  | import os | 
					
						
							|  |  |  | import getopt | 
					
						
							| 
									
										
										
										
											1997-07-16 01:56:13 +00:00
										 |  |  | import traceback | 
					
						
							| 
									
										
										
										
											2000-06-30 16:39:27 +00:00
										 |  |  | import random | 
					
						
							| 
									
										
										
										
											2001-05-21 21:08:12 +00:00
										 |  |  | import StringIO | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | import test_support | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-08-20 22:33:46 +00:00
										 |  |  | def usage(code, msg=''): | 
					
						
							|  |  |  |     print __doc__ | 
					
						
							|  |  |  |     if msg: print msg | 
					
						
							|  |  |  |     sys.exit(code) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-06-30 16:39:27 +00:00
										 |  |  | def main(tests=None, testdir=None, verbose=0, quiet=0, generate=0, | 
					
						
							| 
									
										
										
										
											2000-09-22 15:29:28 +00:00
										 |  |  |          exclude=0, single=0, randomize=0, findleaks=0, | 
					
						
							| 
									
										
										
										
											2001-08-20 22:33:46 +00:00
										 |  |  |          use_resources=None): | 
					
						
							| 
									
										
										
										
											1998-08-01 17:04:08 +00:00
										 |  |  |     """Execute a test suite.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-07-16 12:04:32 +00:00
										 |  |  |     This also parses command-line options and modifies its behavior | 
					
						
							| 
									
										
										
										
											2000-10-23 17:22:08 +00:00
										 |  |  |     accordingly. | 
					
						
							| 
									
										
										
										
											1998-08-01 17:04:08 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     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 | 
					
						
							| 
									
										
										
										
											2000-10-23 17:22:08 +00:00
										 |  |  |     Python test suite is searched for. | 
					
						
							| 
									
										
										
										
											1998-08-01 17:04:08 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     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. | 
					
						
							| 
									
										
										
										
											2000-06-30 16:39:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-08-20 22:33:46 +00:00
										 |  |  |     The other default arguments (verbose, quiet, generate, exclude, single, | 
					
						
							|  |  |  |     randomize, findleaks, and use_resources) allow programmers calling main() | 
					
						
							| 
									
										
										
										
											2000-08-03 15:50:37 +00:00
										 |  |  |     directly to set the values that would normally be set by flags on the | 
					
						
							|  |  |  |     command line. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-08-01 17:04:08 +00:00
										 |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2000-10-23 17:22:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     try: | 
					
						
							| 
									
										
										
										
											2001-08-20 22:33:46 +00:00
										 |  |  |         opts, args = getopt.getopt(sys.argv[1:], 'hvgqxsrlu:', | 
					
						
							|  |  |  |                                    ['help', 'verbose', 'quiet', 'generate', | 
					
						
							|  |  |  |                                     'exclude', 'single', 'random', | 
					
						
							|  |  |  |                                     'findleaks', 'use=']) | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     except getopt.error, msg: | 
					
						
							| 
									
										
										
										
											2001-08-20 22:33:46 +00:00
										 |  |  |         usage(2, msg) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Defaults | 
					
						
							|  |  |  |     if use_resources is None: | 
					
						
							|  |  |  |         use_resources = [] | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     for o, a in opts: | 
					
						
							| 
									
										
										
										
											2001-08-20 22:33:46 +00:00
										 |  |  |         if o in ('-h', '--help'): | 
					
						
							|  |  |  |             usage(0) | 
					
						
							|  |  |  |         elif o in ('-v', '--verbose'): | 
					
						
							|  |  |  |             verbose += 1 | 
					
						
							|  |  |  |         elif o in ('-q', '--quiet'): | 
					
						
							|  |  |  |             quiet = 1; | 
					
						
							|  |  |  |             verbose = 0 | 
					
						
							|  |  |  |         elif o in ('-g', '--generate'): | 
					
						
							|  |  |  |             generate = 1 | 
					
						
							|  |  |  |         elif o in ('-x', '--exclude'): | 
					
						
							|  |  |  |             exclude = 1 | 
					
						
							|  |  |  |         elif o in ('-s', '--single'): | 
					
						
							|  |  |  |             single = 1 | 
					
						
							|  |  |  |         elif o in ('-r', '--randomize'): | 
					
						
							|  |  |  |             randomize = 1 | 
					
						
							|  |  |  |         elif o in ('-l', '--findleaks'): | 
					
						
							|  |  |  |             findleaks = 1 | 
					
						
							|  |  |  |         elif o in ('-u', '--use'): | 
					
						
							| 
									
										
										
										
											2001-09-06 16:09:41 +00:00
										 |  |  |             u = [x.lower() for x in a.split(',')] | 
					
						
							|  |  |  |             for r in u: | 
					
						
							| 
									
										
										
										
											2001-08-20 22:33:46 +00:00
										 |  |  |                 if r not in ('largefile', 'network'): | 
					
						
							|  |  |  |                     usage(1, 'Invalid -u/--use option: %s' % a) | 
					
						
							| 
									
										
										
										
											2001-09-06 16:09:41 +00:00
										 |  |  |             use_resources.extend(u) | 
					
						
							| 
									
										
										
										
											1997-08-18 20:08:24 +00:00
										 |  |  |     if generate and verbose: | 
					
						
							| 
									
										
										
										
											2001-08-20 22:33:46 +00:00
										 |  |  |         usage(2, "-g and -v don't go together!") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     good = [] | 
					
						
							|  |  |  |     bad = [] | 
					
						
							|  |  |  |     skipped = [] | 
					
						
							| 
									
										
										
										
											1999-01-28 19:51:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-09-22 15:29:28 +00:00
										 |  |  |     if findleaks: | 
					
						
							| 
									
										
										
										
											2000-08-03 15:50:37 +00:00
										 |  |  |         try: | 
					
						
							|  |  |  |             import gc | 
					
						
							|  |  |  |         except ImportError: | 
					
						
							| 
									
										
										
										
											2000-10-13 01:32:42 +00:00
										 |  |  |             print 'No GC available, disabling findleaks.' | 
					
						
							| 
									
										
										
										
											2000-09-22 15:29:28 +00:00
										 |  |  |             findleaks = 0 | 
					
						
							| 
									
										
										
										
											2000-08-03 15:50:37 +00:00
										 |  |  |         else: | 
					
						
							| 
									
										
										
										
											2000-10-13 01:32:42 +00:00
										 |  |  |             # Uncomment the line below to report garbage that is not | 
					
						
							|  |  |  |             # freeable by reference counting alone.  By default only | 
					
						
							|  |  |  |             # garbage that is not collectable by the GC is reported. | 
					
						
							|  |  |  |             #gc.set_debug(gc.DEBUG_SAVEALL) | 
					
						
							| 
									
										
										
										
											2000-09-22 15:29:28 +00:00
										 |  |  |             found_garbage = [] | 
					
						
							| 
									
										
										
										
											2000-08-03 15:50:37 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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') | 
					
						
							| 
									
										
										
										
											2001-02-09 11:51:27 +00:00
										 |  |  |             next = fp.read().strip() | 
					
						
							| 
									
										
										
										
											1999-01-28 19:51:51 +00:00
										 |  |  |             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] | 
					
						
							| 
									
										
										
										
											2000-06-30 16:39:27 +00:00
										 |  |  |     if randomize: | 
					
						
							|  |  |  |         random.shuffle(tests) | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |     test_support.verbose = verbose      # Tell tests to be moderately quiet | 
					
						
							| 
									
										
										
										
											2001-08-20 22:33:46 +00:00
										 |  |  |     test_support.use_resources = use_resources | 
					
						
							| 
									
										
										
										
											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 | 
					
						
							| 
									
										
										
										
											2000-08-11 19:02:59 +00:00
										 |  |  |         ok = runtest(test, generate, verbose, quiet, testdir) | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         if ok > 0: | 
					
						
							|  |  |  |             good.append(test) | 
					
						
							|  |  |  |         elif ok == 0: | 
					
						
							|  |  |  |             bad.append(test) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             skipped.append(test) | 
					
						
							| 
									
										
										
										
											2000-09-22 15:29:28 +00:00
										 |  |  |         if findleaks: | 
					
						
							|  |  |  |             gc.collect() | 
					
						
							|  |  |  |             if gc.garbage: | 
					
						
							| 
									
										
										
										
											2000-10-13 01:32:42 +00:00
										 |  |  |                 print "Warning: test created", len(gc.garbage), | 
					
						
							|  |  |  |                 print "uncollectable object(s)." | 
					
						
							|  |  |  |                 # move the uncollectable objects somewhere so we don't see | 
					
						
							|  |  |  |                 # them again | 
					
						
							| 
									
										
										
										
											2000-09-22 15:29:28 +00:00
										 |  |  |                 found_garbage.extend(gc.garbage) | 
					
						
							|  |  |  |                 del gc.garbage[:] | 
					
						
							| 
									
										
										
										
											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." | 
					
						
							| 
									
										
										
										
											2000-12-30 22:21:22 +00:00
										 |  |  |         if verbose: | 
					
						
							|  |  |  |             print "CAUTION:  stdout isn't compared in verbose mode:  a test" | 
					
						
							|  |  |  |             print "that passes in verbose mode may fail without it." | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     if bad: | 
					
						
							| 
									
										
										
										
											2001-08-12 03:45:50 +00:00
										 |  |  |         print count(len(bad), "test"), "failed:" | 
					
						
							|  |  |  |         printlist(bad) | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     if skipped and not quiet: | 
					
						
							| 
									
										
										
										
											2001-08-12 03:45:50 +00:00
										 |  |  |         print count(len(skipped), "test"), "skipped:" | 
					
						
							|  |  |  |         printlist(skipped) | 
					
						
							| 
									
										
										
										
											1999-01-28 19:51:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-08-12 01:20:39 +00:00
										 |  |  |         e = _ExpectedSkips() | 
					
						
							| 
									
										
										
										
											2001-08-12 02:01:09 +00:00
										 |  |  |         plat = sys.platform | 
					
						
							| 
									
										
										
										
											2001-08-12 01:20:39 +00:00
										 |  |  |         if e.isvalid(): | 
					
						
							|  |  |  |             surprise = _Set(skipped) - e.getexpected() | 
					
						
							|  |  |  |             if surprise: | 
					
						
							|  |  |  |                 print count(len(surprise), "skip"), \ | 
					
						
							| 
									
										
										
										
											2001-08-12 03:45:50 +00:00
										 |  |  |                       "unexpected on", plat + ":" | 
					
						
							|  |  |  |                 printlist(surprise) | 
					
						
							| 
									
										
										
										
											2001-08-12 01:20:39 +00:00
										 |  |  |             else: | 
					
						
							|  |  |  |                 print "Those skips are all expected on", plat + "." | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             print "Ask someone to teach regrtest.py about which tests are" | 
					
						
							|  |  |  |             print "expected to get skipped on", plat + "." | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-08-20 22:33:46 +00:00
										 |  |  |     sys.exit(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', | 
					
						
							| 
									
										
										
										
											2001-02-28 17:48:06 +00:00
										 |  |  |     'test_future1', | 
					
						
							|  |  |  |     'test_future2', | 
					
						
							| 
									
										
										
										
											2001-08-20 20:33:42 +00:00
										 |  |  |     'test_future3', | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     ] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-08-11 19:02:59 +00:00
										 |  |  | def runtest(test, generate, verbose, quiet, testdir = None): | 
					
						
							| 
									
										
										
										
											1998-08-01 17:04:08 +00:00
										 |  |  |     """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 | 
					
						
							| 
									
										
										
										
											2000-08-11 19:02:59 +00:00
										 |  |  |     quiet -- if true, don't print 'skipped' messages (probably redundant) | 
					
						
							| 
									
										
										
										
											1998-08-01 17:04:08 +00:00
										 |  |  |     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: | 
					
						
							| 
									
										
										
										
											2001-05-29 17:10:51 +00:00
										 |  |  |             cfp = StringIO.StringIO() | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         elif verbose: | 
					
						
							|  |  |  |             cfp = sys.stdout | 
					
						
							|  |  |  |         else: | 
					
						
							| 
									
										
										
										
											2001-09-09 06:12:01 +00:00
										 |  |  |             cfp = Compare(outputfile, sys.stdout) | 
					
						
							| 
									
										
										
										
											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 | 
					
						
							| 
									
										
										
										
											2001-05-22 18:28:25 +00:00
										 |  |  |             the_module = __import__(test, globals(), locals(), []) | 
					
						
							|  |  |  |             # Most tests run to completion simply as a side-effect of | 
					
						
							|  |  |  |             # being imported.  For the benefit of tests that can't run | 
					
						
							|  |  |  |             # that way (like test_threaded_import), explicitly invoke | 
					
						
							|  |  |  |             # their test_main() function (if it exists). | 
					
						
							|  |  |  |             indirect_test = getattr(the_module, "test_main", None) | 
					
						
							|  |  |  |             if indirect_test is not None: | 
					
						
							|  |  |  |                 indirect_test() | 
					
						
							| 
									
										
										
										
											2000-07-11 15:15:31 +00:00
										 |  |  |             if cfp and not (generate or verbose): | 
					
						
							|  |  |  |                 cfp.close() | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         finally: | 
					
						
							|  |  |  |             sys.stdout = save_stdout | 
					
						
							| 
									
										
										
										
											2000-08-04 13:17:51 +00:00
										 |  |  |     except (ImportError, test_support.TestSkipped), msg: | 
					
						
							| 
									
										
										
										
											2000-08-11 19:02:59 +00:00
										 |  |  |         if not quiet: | 
					
						
							|  |  |  |             print "test", test, | 
					
						
							|  |  |  |             print "skipped -- ", msg | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         return -1 | 
					
						
							| 
									
										
										
										
											2000-08-18 16:04:05 +00:00
										 |  |  |     except KeyboardInterrupt: | 
					
						
							|  |  |  |         raise | 
					
						
							| 
									
										
										
										
											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] | 
					
						
							| 
									
										
										
										
											2000-08-23 20:34:40 +00:00
										 |  |  |         print "test", test, "crashed --", str(type) + ":", value | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         if verbose: | 
					
						
							|  |  |  |             traceback.print_exc(file=sys.stdout) | 
					
						
							|  |  |  |         return 0 | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  |     else: | 
					
						
							| 
									
										
										
										
											2001-05-29 17:10:51 +00:00
										 |  |  |         if generate: | 
					
						
							|  |  |  |             output = cfp.getvalue() | 
					
						
							|  |  |  |             if output == test + "\n": | 
					
						
							|  |  |  |                 if os.path.exists(outputfile): | 
					
						
							|  |  |  |                     # Write it since it already exists (and the contents | 
					
						
							|  |  |  |                     # may have changed), but let the user know it isn't | 
					
						
							|  |  |  |                     # needed: | 
					
						
							|  |  |  |                     fp = open(outputfile, "w") | 
					
						
							|  |  |  |                     fp.write(output) | 
					
						
							|  |  |  |                     fp.close() | 
					
						
							|  |  |  |                     print "output file", outputfile, \ | 
					
						
							|  |  |  |                           "is no longer needed; consider removing it" | 
					
						
							|  |  |  |                 # else: | 
					
						
							|  |  |  |                 #     We don't need it, so don't create it. | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 fp = open(outputfile, "w") | 
					
						
							|  |  |  |                 fp.write(output) | 
					
						
							|  |  |  |                 fp.close() | 
					
						
							| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-08-12 03:45:50 +00:00
										 |  |  | def printlist(x, width=70, indent=4): | 
					
						
							|  |  |  |     """Print the elements of a sequence to stdout.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Optional arg width (default 70) is the maximum line length. | 
					
						
							|  |  |  |     Optional arg indent (default 4) is the number of blanks with which to | 
					
						
							|  |  |  |     begin each line. | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     line = ' ' * indent | 
					
						
							|  |  |  |     for one in map(str, x): | 
					
						
							|  |  |  |         w = len(line) + len(one) | 
					
						
							|  |  |  |         if line[-1:] == ' ': | 
					
						
							|  |  |  |             pad = '' | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             pad = ' ' | 
					
						
							|  |  |  |             w += 1 | 
					
						
							|  |  |  |         if w > width: | 
					
						
							|  |  |  |             print line | 
					
						
							|  |  |  |             line = ' ' * indent + one | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             line += pad + one | 
					
						
							|  |  |  |     if len(line) > indent: | 
					
						
							|  |  |  |         print line | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  | class Compare: | 
					
						
							| 
									
										
										
										
											2001-09-09 06:12:01 +00:00
										 |  |  |     def __init__(self, filename, origstdout): | 
					
						
							|  |  |  |         self.origstdout = origstdout | 
					
						
							| 
									
										
										
										
											2001-05-21 21:08:12 +00:00
										 |  |  |         if os.path.exists(filename): | 
					
						
							|  |  |  |             self.fp = open(filename, 'r') | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self.fp = StringIO.StringIO( | 
					
						
							|  |  |  |                 os.path.basename(filename) + "\n") | 
					
						
							| 
									
										
										
										
											2000-12-30 22:21:22 +00:00
										 |  |  |         self.stuffthatmatched = [] | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def write(self, data): | 
					
						
							| 
									
										
										
										
											2001-09-09 06:12:01 +00:00
										 |  |  |         if test_support.suppress_output_comparison(): | 
					
						
							|  |  |  |             self.origstdout.write(data) | 
					
						
							|  |  |  |             return | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         expected = self.fp.read(len(data)) | 
					
						
							| 
									
										
										
										
											2000-12-30 22:21:22 +00:00
										 |  |  |         if data == expected: | 
					
						
							|  |  |  |             self.stuffthatmatched.append(expected) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             # This Compare instance is spoofing stdout, so we need to write | 
					
						
							|  |  |  |             # to stderr instead. | 
					
						
							|  |  |  |             from sys import stderr as e | 
					
						
							|  |  |  |             print >> e, "The actual stdout doesn't match the expected stdout." | 
					
						
							|  |  |  |             if self.stuffthatmatched: | 
					
						
							|  |  |  |                 print >> e, "This much did match (between asterisk lines):" | 
					
						
							|  |  |  |                 print >> e, "*" * 70 | 
					
						
							|  |  |  |                 good = "".join(self.stuffthatmatched) | 
					
						
							|  |  |  |                 e.write(good) | 
					
						
							|  |  |  |                 if not good.endswith("\n"): | 
					
						
							|  |  |  |                     e.write("\n") | 
					
						
							|  |  |  |                 print >> e, "*" * 70 | 
					
						
							|  |  |  |                 print >> e, "Then ..." | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 print >> e, "The first write to stdout clashed:" | 
					
						
							|  |  |  |             # Note that the prompts are the same length in next two lines. | 
					
						
							|  |  |  |             # This is so what we expected and what we got line up. | 
					
						
							|  |  |  |             print >> e, "We expected (repr):", `expected` | 
					
						
							|  |  |  |             print >> e, "But instead we got:", `data` | 
					
						
							|  |  |  |             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: | 
					
						
							| 
									
										
										
										
											2000-12-30 22:21:22 +00:00
										 |  |  |             raise test_support.TestFailed('Tail of expected stdout unseen: ' + | 
					
						
							|  |  |  |                                           `leftover`) | 
					
						
							| 
									
										
										
										
											1998-03-26 19:42:58 +00:00
										 |  |  |         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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-08-12 01:20:39 +00:00
										 |  |  | class _Set: | 
					
						
							|  |  |  |     def __init__(self, seq=[]): | 
					
						
							|  |  |  |         data = self.data = {} | 
					
						
							|  |  |  |         for x in seq: | 
					
						
							|  |  |  |             data[x] = 1 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __len__(self): | 
					
						
							|  |  |  |         return len(self.data) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __sub__(self, other): | 
					
						
							|  |  |  |         "Return set of all elements in self not in other." | 
					
						
							|  |  |  |         result = _Set() | 
					
						
							|  |  |  |         data = result.data = self.data.copy() | 
					
						
							|  |  |  |         for x in other.data: | 
					
						
							|  |  |  |             if x in data: | 
					
						
							|  |  |  |                 del data[x] | 
					
						
							|  |  |  |         return result | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-08-12 21:53:08 +00:00
										 |  |  |     def __iter__(self): | 
					
						
							|  |  |  |         return iter(self.data) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-08-12 01:20:39 +00:00
										 |  |  |     def tolist(self, sorted=1): | 
					
						
							|  |  |  |         "Return _Set elements as a list." | 
					
						
							|  |  |  |         data = self.data.keys() | 
					
						
							|  |  |  |         if sorted: | 
					
						
							|  |  |  |             data.sort() | 
					
						
							|  |  |  |         return data | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-08-12 02:22:19 +00:00
										 |  |  | _expectations = { | 
					
						
							|  |  |  |     'win32': | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         test_al | 
					
						
							|  |  |  |         test_cd | 
					
						
							|  |  |  |         test_cl | 
					
						
							|  |  |  |         test_commands | 
					
						
							|  |  |  |         test_crypt | 
					
						
							|  |  |  |         test_dbm | 
					
						
							|  |  |  |         test_dl | 
					
						
							|  |  |  |         test_fcntl | 
					
						
							|  |  |  |         test_fork1 | 
					
						
							|  |  |  |         test_gdbm | 
					
						
							|  |  |  |         test_gl | 
					
						
							|  |  |  |         test_grp | 
					
						
							|  |  |  |         test_imgfile | 
					
						
							|  |  |  |         test_largefile | 
					
						
							|  |  |  |         test_linuxaudiodev | 
					
						
							|  |  |  |         test_mhlib | 
					
						
							|  |  |  |         test_nis | 
					
						
							|  |  |  |         test_openpty | 
					
						
							|  |  |  |         test_poll | 
					
						
							|  |  |  |         test_pty | 
					
						
							|  |  |  |         test_pwd | 
					
						
							|  |  |  |         test_signal | 
					
						
							| 
									
										
										
										
											2001-08-20 22:33:46 +00:00
										 |  |  |         test_socket_ssl | 
					
						
							| 
									
										
										
										
											2001-09-18 02:18:57 +00:00
										 |  |  |         test_socketserver | 
					
						
							| 
									
										
										
										
											2001-08-12 02:22:19 +00:00
										 |  |  |         test_sunaudiodev | 
					
						
							|  |  |  |         test_timing | 
					
						
							|  |  |  |         """,
 | 
					
						
							|  |  |  |     'linux2': | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         test_al | 
					
						
							|  |  |  |         test_cd | 
					
						
							|  |  |  |         test_cl | 
					
						
							|  |  |  |         test_dl | 
					
						
							|  |  |  |         test_gl | 
					
						
							|  |  |  |         test_imgfile | 
					
						
							|  |  |  |         test_largefile | 
					
						
							|  |  |  |         test_nis | 
					
						
							|  |  |  |         test_ntpath | 
					
						
							| 
									
										
										
										
											2001-08-20 22:33:46 +00:00
										 |  |  |         test_socket_ssl | 
					
						
							| 
									
										
										
										
											2001-08-12 02:22:19 +00:00
										 |  |  |         test_socketserver | 
					
						
							|  |  |  |         test_sunaudiodev | 
					
						
							|  |  |  |         test_unicode_file | 
					
						
							|  |  |  |         test_winreg | 
					
						
							|  |  |  |         test_winsound | 
					
						
							|  |  |  |         """,
 | 
					
						
							| 
									
										
										
										
											2001-08-28 14:49:00 +00:00
										 |  |  |    'mac': | 
					
						
							| 
									
										
										
										
											2001-09-02 03:58:41 +00:00
										 |  |  |         """
 | 
					
						
							|  |  |  |         test_al | 
					
						
							|  |  |  |         test_bsddb | 
					
						
							|  |  |  |         test_cd | 
					
						
							|  |  |  |         test_cl | 
					
						
							|  |  |  |         test_commands | 
					
						
							|  |  |  |         test_crypt | 
					
						
							|  |  |  |         test_dbm | 
					
						
							|  |  |  |         test_dl | 
					
						
							|  |  |  |         test_fcntl | 
					
						
							|  |  |  |         test_fork1 | 
					
						
							|  |  |  |         test_gl | 
					
						
							|  |  |  |         test_grp | 
					
						
							|  |  |  |         test_imgfile | 
					
						
							|  |  |  |         test_largefile | 
					
						
							|  |  |  |         test_linuxaudiodev | 
					
						
							|  |  |  |         test_locale | 
					
						
							|  |  |  |         test_mmap | 
					
						
							|  |  |  |         test_nis | 
					
						
							|  |  |  |         test_ntpath | 
					
						
							|  |  |  |         test_openpty | 
					
						
							|  |  |  |         test_poll | 
					
						
							|  |  |  |         test_popen2 | 
					
						
							|  |  |  |         test_pty | 
					
						
							|  |  |  |         test_pwd | 
					
						
							|  |  |  |         test_signal | 
					
						
							|  |  |  |         test_socket_ssl | 
					
						
							|  |  |  |         test_socketserver | 
					
						
							|  |  |  |         test_sunaudiodev | 
					
						
							|  |  |  |         test_sundry | 
					
						
							|  |  |  |         test_timing | 
					
						
							|  |  |  |         test_unicode_file | 
					
						
							|  |  |  |         test_winreg | 
					
						
							|  |  |  |         test_winsound | 
					
						
							|  |  |  |         """,
 | 
					
						
							| 
									
										
										
										
											2001-09-05 14:38:48 +00:00
										 |  |  |     'unixware5': | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         test_al | 
					
						
							|  |  |  |         test_bsddb | 
					
						
							|  |  |  |         test_cd | 
					
						
							|  |  |  |         test_cl | 
					
						
							|  |  |  |         test_dl | 
					
						
							|  |  |  |         test_gl | 
					
						
							|  |  |  |         test_imgfile | 
					
						
							|  |  |  |         test_largefile | 
					
						
							|  |  |  |         test_linuxaudiodev | 
					
						
							|  |  |  |         test_minidom | 
					
						
							|  |  |  |         test_nis | 
					
						
							|  |  |  |         test_ntpath | 
					
						
							|  |  |  |         test_openpty | 
					
						
							|  |  |  |         test_pyexpat | 
					
						
							|  |  |  |         test_sax | 
					
						
							|  |  |  |         test_socketserver | 
					
						
							|  |  |  |         test_sunaudiodev | 
					
						
							|  |  |  |         test_sundry | 
					
						
							|  |  |  |         test_unicode_file | 
					
						
							|  |  |  |         test_winreg | 
					
						
							|  |  |  |         test_winsound | 
					
						
							|  |  |  |         """,
 | 
					
						
							| 
									
										
										
										
											2001-08-12 02:22:19 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-08-12 01:20:39 +00:00
										 |  |  | class _ExpectedSkips: | 
					
						
							|  |  |  |     def __init__(self): | 
					
						
							|  |  |  |         self.valid = 0 | 
					
						
							| 
									
										
										
										
											2001-08-12 02:22:19 +00:00
										 |  |  |         if _expectations.has_key(sys.platform): | 
					
						
							|  |  |  |             s = _expectations[sys.platform] | 
					
						
							| 
									
										
										
										
											2001-08-12 01:20:39 +00:00
										 |  |  |             self.expected = _Set(s.split()) | 
					
						
							| 
									
										
										
										
											2001-08-12 02:22:19 +00:00
										 |  |  |             self.valid = 1 | 
					
						
							| 
									
										
										
										
											2001-08-12 01:20:39 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def isvalid(self): | 
					
						
							|  |  |  |         "Return true iff _ExpectedSkips knows about the current platform." | 
					
						
							|  |  |  |         return self.valid | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def getexpected(self): | 
					
						
							|  |  |  |         """Return set of test names we expect to skip on current platform.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.isvalid() must be true. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         assert self.isvalid() | 
					
						
							|  |  |  |         return self.expected | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-20 03:12:20 +00:00
										 |  |  | if __name__ == '__main__': | 
					
						
							| 
									
										
										
										
											2001-08-20 22:33:46 +00:00
										 |  |  |     main() |