mirror of
				https://github.com/python/cpython.git
				synced 2025-10-26 03:04:41 +00:00 
			
		
		
		
	gh-108834: regrtest reruns failed tests in subprocesses (#108839)
When using --rerun option, regrtest now re-runs failed tests in verbose mode in fresh worker processes to have more deterministic behavior. So it can write its final report even if a test killed a worker progress. Add --fail-rerun option to regrtest: exit with non-zero exit code if a test failed pass passed when re-run in verbose mode (in a fresh process). That's now more useful since tests can pass when re-run in a fresh worker progress, whereas they failed when run after other tests when tests are run sequentially. Rename --verbose2 option (-w) to --rerun. Keep --verbose2 as a deprecated alias. Changes: * Fix and enhance statistics in regrtest summary. Add "(filtered)" when --match and/or --ignore options are used. * Add RunTests class. * Add TestResult.get_rerun_match_tests() method * Rewrite code to serialize/deserialize worker arguments as JSON using a new WorkerJob class. * Fix stats when a test is run with --forever --rerun. * If failed test names cannot be parsed, log a warning and don't filter tests. * test_regrtest.test_rerun_success() now uses a marker file, since the test is re-run in a separated process. * Add tests on normalize_test_name() function. * Add test_success() and test_skip() tests to test_regrtest.
This commit is contained in:
		
							parent
							
								
									c2ec174d24
								
							
						
					
					
						commit
						31c2945f14
					
				
					 12 changed files with 819 additions and 478 deletions
				
			
		|  | @ -109,8 +109,9 @@ def parse_args(): | ||||||
| 
 | 
 | ||||||
| def main(): | def main(): | ||||||
|     args = parse_args() |     args = parse_args() | ||||||
|     if '-w' in args.test_args or '--verbose2' in args.test_args: |     for opt in ('-w', '--rerun', '--verbose2'): | ||||||
|         print("WARNING: -w/--verbose2 option should not be used to bisect!") |         if opt in args.test_args: | ||||||
|  |             print(f"WARNING: {opt} option should not be used to bisect!") | ||||||
|             print() |             print() | ||||||
| 
 | 
 | ||||||
|     if args.input: |     if args.input: | ||||||
|  |  | ||||||
|  | @ -156,7 +156,7 @@ def __init__(self, **kwargs) -> None: | ||||||
|         self.coverdir = 'coverage' |         self.coverdir = 'coverage' | ||||||
|         self.runleaks = False |         self.runleaks = False | ||||||
|         self.huntrleaks = False |         self.huntrleaks = False | ||||||
|         self.verbose2 = False |         self.rerun = False | ||||||
|         self.verbose3 = False |         self.verbose3 = False | ||||||
|         self.print_slow = False |         self.print_slow = False | ||||||
|         self.random_seed = None |         self.random_seed = None | ||||||
|  | @ -213,8 +213,10 @@ def _create_parser(): | ||||||
|     group = parser.add_argument_group('Verbosity') |     group = parser.add_argument_group('Verbosity') | ||||||
|     group.add_argument('-v', '--verbose', action='count', |     group.add_argument('-v', '--verbose', action='count', | ||||||
|                        help='run tests in verbose mode with output to stdout') |                        help='run tests in verbose mode with output to stdout') | ||||||
|     group.add_argument('-w', '--verbose2', action='store_true', |     group.add_argument('-w', '--rerun', action='store_true', | ||||||
|                        help='re-run failed tests in verbose mode') |                        help='re-run failed tests in verbose mode') | ||||||
|  |     group.add_argument('--verbose2', action='store_true', dest='rerun', | ||||||
|  |                        help='deprecated alias to --rerun') | ||||||
|     group.add_argument('-W', '--verbose3', action='store_true', |     group.add_argument('-W', '--verbose3', action='store_true', | ||||||
|                        help='display test output on failure') |                        help='display test output on failure') | ||||||
|     group.add_argument('-q', '--quiet', action='store_true', |     group.add_argument('-q', '--quiet', action='store_true', | ||||||
|  | @ -309,6 +311,9 @@ def _create_parser(): | ||||||
|     group.add_argument('--fail-env-changed', action='store_true', |     group.add_argument('--fail-env-changed', action='store_true', | ||||||
|                        help='if a test file alters the environment, mark ' |                        help='if a test file alters the environment, mark ' | ||||||
|                             'the test as failed') |                             'the test as failed') | ||||||
|  |     group.add_argument('--fail-rerun', action='store_true', | ||||||
|  |                        help='if a test failed and then passed when re-run, ' | ||||||
|  |                             'mark the tests as failed') | ||||||
| 
 | 
 | ||||||
|     group.add_argument('--junit-xml', dest='xmlpath', metavar='FILENAME', |     group.add_argument('--junit-xml', dest='xmlpath', metavar='FILENAME', | ||||||
|                        help='writes JUnit-style XML results to the specified ' |                        help='writes JUnit-style XML results to the specified ' | ||||||
|  | @ -380,7 +385,7 @@ def _parse_args(args, **kwargs): | ||||||
|         ns.python = shlex.split(ns.python) |         ns.python = shlex.split(ns.python) | ||||||
|     if ns.failfast and not (ns.verbose or ns.verbose3): |     if ns.failfast and not (ns.verbose or ns.verbose3): | ||||||
|         parser.error("-G/--failfast needs either -v or -W") |         parser.error("-G/--failfast needs either -v or -W") | ||||||
|     if ns.pgo and (ns.verbose or ns.verbose2 or ns.verbose3): |     if ns.pgo and (ns.verbose or ns.rerun or ns.verbose3): | ||||||
|         parser.error("--pgo/-v don't go together!") |         parser.error("--pgo/-v don't go together!") | ||||||
|     if ns.pgo_extended: |     if ns.pgo_extended: | ||||||
|         ns.pgo = True  # pgo_extended implies pgo |         ns.pgo = True  # pgo_extended implies pgo | ||||||
|  |  | ||||||
|  | @ -11,11 +11,11 @@ | ||||||
| import unittest | import unittest | ||||||
| from test.libregrtest.cmdline import _parse_args | from test.libregrtest.cmdline import _parse_args | ||||||
| from test.libregrtest.runtest import ( | from test.libregrtest.runtest import ( | ||||||
|     findtests, split_test_packages, runtest, get_abs_module, |     findtests, split_test_packages, runtest, abs_module_name, | ||||||
|     PROGRESS_MIN_TIME, State) |     PROGRESS_MIN_TIME, State, MatchTestsDict, RunTests) | ||||||
| from test.libregrtest.setup import setup_tests | from test.libregrtest.setup import setup_tests | ||||||
| from test.libregrtest.pgo import setup_pgo_tests | from test.libregrtest.pgo import setup_pgo_tests | ||||||
| from test.libregrtest.utils import (removepy, count, format_duration, | from test.libregrtest.utils import (strip_py_suffix, count, format_duration, | ||||||
|                                     printlist, get_build_info) |                                     printlist, get_build_info) | ||||||
| from test import support | from test import support | ||||||
| from test.support import TestStats | from test.support import TestStats | ||||||
|  | @ -28,14 +28,6 @@ | ||||||
| # Must be smaller than buildbot "1200 seconds without output" limit. | # Must be smaller than buildbot "1200 seconds without output" limit. | ||||||
| EXIT_TIMEOUT = 120.0 | EXIT_TIMEOUT = 120.0 | ||||||
| 
 | 
 | ||||||
| # gh-90681: When rerunning tests, we might need to rerun the whole |  | ||||||
| # class or module suite if some its life-cycle hooks fail. |  | ||||||
| # Test level hooks are not affected. |  | ||||||
| _TEST_LIFECYCLE_HOOKS = frozenset(( |  | ||||||
|     'setUpClass', 'tearDownClass', |  | ||||||
|     'setUpModule', 'tearDownModule', |  | ||||||
| )) |  | ||||||
| 
 |  | ||||||
| EXITCODE_BAD_TEST = 2 | EXITCODE_BAD_TEST = 2 | ||||||
| EXITCODE_INTERRUPTED = 130 | EXITCODE_INTERRUPTED = 130 | ||||||
| EXITCODE_ENV_CHANGED = 3 | EXITCODE_ENV_CHANGED = 3 | ||||||
|  | @ -72,19 +64,22 @@ def __init__(self): | ||||||
|         # tests |         # tests | ||||||
|         self.tests = [] |         self.tests = [] | ||||||
|         self.selected = [] |         self.selected = [] | ||||||
|  |         self.all_runtests: list[RunTests] = [] | ||||||
| 
 | 
 | ||||||
|         # test results |         # test results | ||||||
|         self.good = [] |         self.good: list[str] = [] | ||||||
|         self.bad = [] |         self.bad: list[str] = [] | ||||||
|         self.skipped = [] |         self.rerun_bad: list[str] = [] | ||||||
|         self.resource_denied = [] |         self.skipped: list[str] = [] | ||||||
|         self.environment_changed = [] |         self.resource_denied: list[str] = [] | ||||||
|         self.run_no_tests = [] |         self.environment_changed: list[str] = [] | ||||||
|         self.need_rerun = [] |         self.run_no_tests: list[str] = [] | ||||||
|         self.rerun = [] |         self.rerun: list[str] = [] | ||||||
|         self.first_result = None | 
 | ||||||
|  |         self.need_rerun: list[TestResult] = [] | ||||||
|  |         self.first_state: str | None = None | ||||||
|         self.interrupted = False |         self.interrupted = False | ||||||
|         self.stats_dict: dict[str, TestStats] = {} |         self.total_stats = TestStats() | ||||||
| 
 | 
 | ||||||
|         # used by --slow |         # used by --slow | ||||||
|         self.test_times = [] |         self.test_times = [] | ||||||
|  | @ -94,7 +89,7 @@ def __init__(self): | ||||||
| 
 | 
 | ||||||
|         # used to display the progress bar "[ 3/100]" |         # used to display the progress bar "[ 3/100]" | ||||||
|         self.start_time = time.perf_counter() |         self.start_time = time.perf_counter() | ||||||
|         self.test_count = '' |         self.test_count_text = '' | ||||||
|         self.test_count_width = 1 |         self.test_count_width = 1 | ||||||
| 
 | 
 | ||||||
|         # used by --single |         # used by --single | ||||||
|  | @ -107,7 +102,6 @@ def __init__(self): | ||||||
|         # misc |         # misc | ||||||
|         self.win_load_tracker = None |         self.win_load_tracker = None | ||||||
|         self.tmp_dir = None |         self.tmp_dir = None | ||||||
|         self.worker_test_name = None |  | ||||||
| 
 | 
 | ||||||
|     def get_executed(self): |     def get_executed(self): | ||||||
|         return (set(self.good) | set(self.bad) | set(self.skipped) |         return (set(self.good) | set(self.bad) | set(self.skipped) | ||||||
|  | @ -115,11 +109,9 @@ def get_executed(self): | ||||||
|                 | set(self.run_no_tests)) |                 | set(self.run_no_tests)) | ||||||
| 
 | 
 | ||||||
|     def accumulate_result(self, result, rerun=False): |     def accumulate_result(self, result, rerun=False): | ||||||
|  |         fail_env_changed = self.ns.fail_env_changed | ||||||
|         test_name = result.test_name |         test_name = result.test_name | ||||||
| 
 | 
 | ||||||
|         if result.has_meaningful_duration() and not rerun: |  | ||||||
|             self.test_times.append((result.duration, test_name)) |  | ||||||
| 
 |  | ||||||
|         match result.state: |         match result.state: | ||||||
|             case State.PASSED: |             case State.PASSED: | ||||||
|                 self.good.append(test_name) |                 self.good.append(test_name) | ||||||
|  | @ -128,25 +120,24 @@ def accumulate_result(self, result, rerun=False): | ||||||
|             case State.SKIPPED: |             case State.SKIPPED: | ||||||
|                 self.skipped.append(test_name) |                 self.skipped.append(test_name) | ||||||
|             case State.RESOURCE_DENIED: |             case State.RESOURCE_DENIED: | ||||||
|                 self.skipped.append(test_name) |  | ||||||
|                 self.resource_denied.append(test_name) |                 self.resource_denied.append(test_name) | ||||||
|             case State.INTERRUPTED: |             case State.INTERRUPTED: | ||||||
|                 self.interrupted = True |                 self.interrupted = True | ||||||
|             case State.DID_NOT_RUN: |             case State.DID_NOT_RUN: | ||||||
|                 self.run_no_tests.append(test_name) |                 self.run_no_tests.append(test_name) | ||||||
|             case _: |             case _: | ||||||
|                 if result.is_failed(self.ns.fail_env_changed): |                 if result.is_failed(fail_env_changed): | ||||||
|                     if not rerun: |  | ||||||
|                     self.bad.append(test_name) |                     self.bad.append(test_name) | ||||||
|                     self.need_rerun.append(result) |                     self.need_rerun.append(result) | ||||||
|                 else: |                 else: | ||||||
|                     raise ValueError(f"invalid test state: {state!r}") |                     raise ValueError(f"invalid test state: {result.state!r}") | ||||||
| 
 | 
 | ||||||
|  |         if result.has_meaningful_duration() and not rerun: | ||||||
|  |             self.test_times.append((result.duration, test_name)) | ||||||
|         if result.stats is not None: |         if result.stats is not None: | ||||||
|             self.stats_dict[result.test_name] = result.stats |             self.total_stats.accumulate(result.stats) | ||||||
| 
 |         if rerun: | ||||||
|         if rerun and not(result.is_failed(False) or result.state == State.INTERRUPTED): |             self.rerun.append(test_name) | ||||||
|             self.bad.remove(test_name) |  | ||||||
| 
 | 
 | ||||||
|         xml_data = result.xml_data |         xml_data = result.xml_data | ||||||
|         if xml_data: |         if xml_data: | ||||||
|  | @ -180,13 +171,15 @@ def log(self, line=''): | ||||||
|         print(line, flush=True) |         print(line, flush=True) | ||||||
| 
 | 
 | ||||||
|     def display_progress(self, test_index, text): |     def display_progress(self, test_index, text): | ||||||
|         if self.ns.quiet: |         quiet = self.ns.quiet | ||||||
|  |         pgo = self.ns.pgo | ||||||
|  |         if quiet: | ||||||
|             return |             return | ||||||
| 
 | 
 | ||||||
|         # "[ 51/405/1] test_tcl passed" |         # "[ 51/405/1] test_tcl passed" | ||||||
|         line = f"{test_index:{self.test_count_width}}{self.test_count}" |         line = f"{test_index:{self.test_count_width}}{self.test_count_text}" | ||||||
|         fails = len(self.bad) + len(self.environment_changed) |         fails = len(self.bad) + len(self.environment_changed) | ||||||
|         if fails and not self.ns.pgo: |         if fails and not pgo: | ||||||
|             line = f"{line}/{fails}" |             line = f"{line}/{fails}" | ||||||
|         self.log(f"[{line}] {text}") |         self.log(f"[{line}] {text}") | ||||||
| 
 | 
 | ||||||
|  | @ -196,15 +189,7 @@ def parse_args(self, kwargs): | ||||||
|         if ns.xmlpath: |         if ns.xmlpath: | ||||||
|             support.junit_xml_list = self.testsuite_xml = [] |             support.junit_xml_list = self.testsuite_xml = [] | ||||||
| 
 | 
 | ||||||
|         worker_args = ns.worker_args |         strip_py_suffix(ns.args) | ||||||
|         if worker_args is not None: |  | ||||||
|             from test.libregrtest.runtest_mp import parse_worker_args |  | ||||||
|             ns, test_name = parse_worker_args(ns.worker_args) |  | ||||||
|             ns.worker_args = worker_args |  | ||||||
|             self.worker_test_name = test_name |  | ||||||
| 
 |  | ||||||
|         # Strip .py extensions. |  | ||||||
|         removepy(ns.args) |  | ||||||
| 
 | 
 | ||||||
|         if ns.huntrleaks: |         if ns.huntrleaks: | ||||||
|             warmup, repetitions, _ = ns.huntrleaks |             warmup, repetitions, _ = ns.huntrleaks | ||||||
|  | @ -221,9 +206,18 @@ def parse_args(self, kwargs): | ||||||
|         self.ns = ns |         self.ns = ns | ||||||
| 
 | 
 | ||||||
|     def find_tests(self, tests): |     def find_tests(self, tests): | ||||||
|  |         ns = self.ns | ||||||
|  |         single = ns.single | ||||||
|  |         fromfile = ns.fromfile | ||||||
|  |         pgo = ns.pgo | ||||||
|  |         exclude = ns.exclude | ||||||
|  |         test_dir = ns.testdir | ||||||
|  |         starting_test = ns.start | ||||||
|  |         randomize = ns.randomize | ||||||
|  | 
 | ||||||
|         self.tests = tests |         self.tests = tests | ||||||
| 
 | 
 | ||||||
|         if self.ns.single: |         if single: | ||||||
|             self.next_single_filename = os.path.join(self.tmp_dir, 'pynexttest') |             self.next_single_filename = os.path.join(self.tmp_dir, 'pynexttest') | ||||||
|             try: |             try: | ||||||
|                 with open(self.next_single_filename, 'r') as fp: |                 with open(self.next_single_filename, 'r') as fp: | ||||||
|  | @ -232,12 +226,12 @@ def find_tests(self, tests): | ||||||
|             except OSError: |             except OSError: | ||||||
|                 pass |                 pass | ||||||
| 
 | 
 | ||||||
|         if self.ns.fromfile: |         if fromfile: | ||||||
|             self.tests = [] |             self.tests = [] | ||||||
|             # regex to match 'test_builtin' in line: |             # regex to match 'test_builtin' in line: | ||||||
|             # '0:00:00 [  4/400] test_builtin -- test_dict took 1 sec' |             # '0:00:00 [  4/400] test_builtin -- test_dict took 1 sec' | ||||||
|             regex = re.compile(r'\btest_[a-zA-Z0-9_]+\b') |             regex = re.compile(r'\btest_[a-zA-Z0-9_]+\b') | ||||||
|             with open(os.path.join(os_helper.SAVEDCWD, self.ns.fromfile)) as fp: |             with open(os.path.join(os_helper.SAVEDCWD, fromfile)) as fp: | ||||||
|                 for line in fp: |                 for line in fp: | ||||||
|                     line = line.split('#', 1)[0] |                     line = line.split('#', 1)[0] | ||||||
|                     line = line.strip() |                     line = line.strip() | ||||||
|  | @ -245,22 +239,22 @@ def find_tests(self, tests): | ||||||
|                     if match is not None: |                     if match is not None: | ||||||
|                         self.tests.append(match.group()) |                         self.tests.append(match.group()) | ||||||
| 
 | 
 | ||||||
|         removepy(self.tests) |         strip_py_suffix(self.tests) | ||||||
| 
 | 
 | ||||||
|         if self.ns.pgo: |         if pgo: | ||||||
|             # add default PGO tests if no tests are specified |             # add default PGO tests if no tests are specified | ||||||
|             setup_pgo_tests(self.ns) |             setup_pgo_tests(ns) | ||||||
| 
 | 
 | ||||||
|         exclude = set() |         exclude_tests = set() | ||||||
|         if self.ns.exclude: |         if exclude: | ||||||
|             for arg in self.ns.args: |             for arg in ns.args: | ||||||
|                 exclude.add(arg) |                 exclude_tests.add(arg) | ||||||
|             self.ns.args = [] |             ns.args = [] | ||||||
| 
 | 
 | ||||||
|         alltests = findtests(testdir=self.ns.testdir, exclude=exclude) |         alltests = findtests(testdir=test_dir, exclude=exclude_tests) | ||||||
| 
 | 
 | ||||||
|         if not self.ns.fromfile: |         if not fromfile: | ||||||
|             self.selected = self.tests or self.ns.args |             self.selected = self.tests or ns.args | ||||||
|             if self.selected: |             if self.selected: | ||||||
|                 self.selected = split_test_packages(self.selected) |                 self.selected = split_test_packages(self.selected) | ||||||
|             else: |             else: | ||||||
|  | @ -268,7 +262,7 @@ def find_tests(self, tests): | ||||||
|         else: |         else: | ||||||
|             self.selected = self.tests |             self.selected = self.tests | ||||||
| 
 | 
 | ||||||
|         if self.ns.single: |         if single: | ||||||
|             self.selected = self.selected[:1] |             self.selected = self.selected[:1] | ||||||
|             try: |             try: | ||||||
|                 pos = alltests.index(self.selected[0]) |                 pos = alltests.index(self.selected[0]) | ||||||
|  | @ -277,17 +271,17 @@ def find_tests(self, tests): | ||||||
|                 pass |                 pass | ||||||
| 
 | 
 | ||||||
|         # Remove all the selected tests that precede start if it's set. |         # Remove all the selected tests that precede start if it's set. | ||||||
|         if self.ns.start: |         if starting_test: | ||||||
|             try: |             try: | ||||||
|                 del self.selected[:self.selected.index(self.ns.start)] |                 del self.selected[:self.selected.index(starting_test)] | ||||||
|             except ValueError: |             except ValueError: | ||||||
|                 print("Couldn't find starting test (%s), using all tests" |                 print(f"Cannot find starting test: {starting_test}") | ||||||
|                       % self.ns.start, file=sys.stderr) |                 sys.exit(1) | ||||||
| 
 | 
 | ||||||
|         if self.ns.randomize: |         if randomize: | ||||||
|             if self.ns.random_seed is None: |             if ns.random_seed is None: | ||||||
|                 self.ns.random_seed = random.randrange(10000000) |                 ns.random_seed = random.randrange(10000000) | ||||||
|             random.seed(self.ns.random_seed) |             random.seed(ns.random_seed) | ||||||
|             random.shuffle(self.selected) |             random.shuffle(self.selected) | ||||||
| 
 | 
 | ||||||
|     def list_tests(self): |     def list_tests(self): | ||||||
|  | @ -305,25 +299,63 @@ def _list_cases(self, suite): | ||||||
|                     print(test.id()) |                     print(test.id()) | ||||||
| 
 | 
 | ||||||
|     def list_cases(self): |     def list_cases(self): | ||||||
|  |         ns = self.ns | ||||||
|  |         test_dir = ns.testdir | ||||||
|         support.verbose = False |         support.verbose = False | ||||||
|         support.set_match_tests(self.ns.match_tests, self.ns.ignore_tests) |         support.set_match_tests(ns.match_tests, ns.ignore_tests) | ||||||
| 
 | 
 | ||||||
|  |         skipped = [] | ||||||
|         for test_name in self.selected: |         for test_name in self.selected: | ||||||
|             abstest = get_abs_module(self.ns, test_name) |             module_name = abs_module_name(test_name, test_dir) | ||||||
|             try: |             try: | ||||||
|                 suite = unittest.defaultTestLoader.loadTestsFromName(abstest) |                 suite = unittest.defaultTestLoader.loadTestsFromName(module_name) | ||||||
|                 self._list_cases(suite) |                 self._list_cases(suite) | ||||||
|             except unittest.SkipTest: |             except unittest.SkipTest: | ||||||
|                 self.skipped.append(test_name) |                 skipped.append(test_name) | ||||||
| 
 | 
 | ||||||
|         if self.skipped: |         if skipped: | ||||||
|             print(file=sys.stderr) |             sys.stdout.flush() | ||||||
|             print(count(len(self.skipped), "test"), "skipped:", file=sys.stderr) |             stderr = sys.stderr | ||||||
|             printlist(self.skipped, file=sys.stderr) |             print(file=stderr) | ||||||
|  |             print(count(len(skipped), "test"), "skipped:", file=stderr) | ||||||
|  |             printlist(skipped, file=stderr) | ||||||
| 
 | 
 | ||||||
|     def rerun_failed_tests(self): |     def get_rerun_match(self, rerun_list) -> MatchTestsDict: | ||||||
|         self.log() |         rerun_match_tests = {} | ||||||
|  |         for result in rerun_list: | ||||||
|  |             match_tests = result.get_rerun_match_tests() | ||||||
|  |             # ignore empty match list | ||||||
|  |             if match_tests: | ||||||
|  |                 rerun_match_tests[result.test_name] = match_tests | ||||||
|  |         return rerun_match_tests | ||||||
| 
 | 
 | ||||||
|  |     def _rerun_failed_tests(self, need_rerun): | ||||||
|  |         # Configure the runner to re-run tests | ||||||
|  |         ns = self.ns | ||||||
|  |         ns.verbose = True | ||||||
|  |         ns.failfast = False | ||||||
|  |         ns.verbose3 = False | ||||||
|  |         ns.forever = False | ||||||
|  |         if ns.use_mp is None: | ||||||
|  |             ns.use_mp = 1 | ||||||
|  | 
 | ||||||
|  |         # Get tests to re-run | ||||||
|  |         tests = [result.test_name for result in need_rerun] | ||||||
|  |         match_tests = self.get_rerun_match(need_rerun) | ||||||
|  |         self.set_tests(tests) | ||||||
|  | 
 | ||||||
|  |         # Clear previously failed tests | ||||||
|  |         self.rerun_bad.extend(self.bad) | ||||||
|  |         self.bad.clear() | ||||||
|  |         self.need_rerun.clear() | ||||||
|  | 
 | ||||||
|  |         # Re-run failed tests | ||||||
|  |         self.log(f"Re-running {len(tests)} failed tests in verbose mode in subprocesses") | ||||||
|  |         runtests = RunTests(tests, match_tests=match_tests, rerun=True) | ||||||
|  |         self.all_runtests.append(runtests) | ||||||
|  |         self._run_tests_mp(runtests) | ||||||
|  | 
 | ||||||
|  |     def rerun_failed_tests(self, need_rerun): | ||||||
|         if self.ns.python: |         if self.ns.python: | ||||||
|             # Temp patch for https://github.com/python/cpython/issues/94052 |             # Temp patch for https://github.com/python/cpython/issues/94052 | ||||||
|             self.log( |             self.log( | ||||||
|  | @ -332,45 +364,10 @@ def rerun_failed_tests(self): | ||||||
|             ) |             ) | ||||||
|             return |             return | ||||||
| 
 | 
 | ||||||
|         self.ns.verbose = True |         self.first_state = self.get_tests_state() | ||||||
|         self.ns.failfast = False |  | ||||||
|         self.ns.verbose3 = False |  | ||||||
| 
 | 
 | ||||||
|         self.first_result = self.get_tests_result() |         print() | ||||||
| 
 |         self._rerun_failed_tests(need_rerun) | ||||||
|         self.log("Re-running failed tests in verbose mode") |  | ||||||
|         rerun_list = list(self.need_rerun) |  | ||||||
|         self.need_rerun.clear() |  | ||||||
|         for result in rerun_list: |  | ||||||
|             test_name = result.test_name |  | ||||||
|             self.rerun.append(test_name) |  | ||||||
| 
 |  | ||||||
|             errors = result.errors or [] |  | ||||||
|             failures = result.failures or [] |  | ||||||
|             error_names = [ |  | ||||||
|                 self.normalize_test_name(test_full_name, is_error=True) |  | ||||||
|                 for (test_full_name, *_) in errors] |  | ||||||
|             failure_names = [ |  | ||||||
|                 self.normalize_test_name(test_full_name) |  | ||||||
|                 for (test_full_name, *_) in failures] |  | ||||||
|             self.ns.verbose = True |  | ||||||
|             orig_match_tests = self.ns.match_tests |  | ||||||
|             if errors or failures: |  | ||||||
|                 if self.ns.match_tests is None: |  | ||||||
|                     self.ns.match_tests = [] |  | ||||||
|                 self.ns.match_tests.extend(error_names) |  | ||||||
|                 self.ns.match_tests.extend(failure_names) |  | ||||||
|                 matching = "matching: " + ", ".join(self.ns.match_tests) |  | ||||||
|                 self.log(f"Re-running {test_name} in verbose mode ({matching})") |  | ||||||
|             else: |  | ||||||
|                 self.log(f"Re-running {test_name} in verbose mode") |  | ||||||
|             result = runtest(self.ns, test_name) |  | ||||||
|             self.ns.match_tests = orig_match_tests |  | ||||||
| 
 |  | ||||||
|             self.accumulate_result(result, rerun=True) |  | ||||||
| 
 |  | ||||||
|             if result.state == State.INTERRUPTED: |  | ||||||
|                 break |  | ||||||
| 
 | 
 | ||||||
|         if self.bad: |         if self.bad: | ||||||
|             print(count(len(self.bad), 'test'), "failed again:") |             print(count(len(self.bad), 'test'), "failed again:") | ||||||
|  | @ -378,28 +375,17 @@ def rerun_failed_tests(self): | ||||||
| 
 | 
 | ||||||
|         self.display_result() |         self.display_result() | ||||||
| 
 | 
 | ||||||
|     def normalize_test_name(self, test_full_name, *, is_error=False): |  | ||||||
|         short_name = test_full_name.split(" ")[0] |  | ||||||
|         if is_error and short_name in _TEST_LIFECYCLE_HOOKS: |  | ||||||
|             # This means that we have a failure in a life-cycle hook, |  | ||||||
|             # we need to rerun the whole module or class suite. |  | ||||||
|             # Basically the error looks like this: |  | ||||||
|             #    ERROR: setUpClass (test.test_reg_ex.RegTest) |  | ||||||
|             # or |  | ||||||
|             #    ERROR: setUpModule (test.test_reg_ex) |  | ||||||
|             # So, we need to parse the class / module name. |  | ||||||
|             lpar = test_full_name.index('(') |  | ||||||
|             rpar = test_full_name.index(')') |  | ||||||
|             return test_full_name[lpar + 1: rpar].split('.')[-1] |  | ||||||
|         return short_name |  | ||||||
| 
 |  | ||||||
|     def display_result(self): |     def display_result(self): | ||||||
|  |         pgo = self.ns.pgo | ||||||
|  |         quiet = self.ns.quiet | ||||||
|  |         print_slow = self.ns.print_slow | ||||||
|  | 
 | ||||||
|         # If running the test suite for PGO then no one cares about results. |         # If running the test suite for PGO then no one cares about results. | ||||||
|         if self.ns.pgo: |         if pgo: | ||||||
|             return |             return | ||||||
| 
 | 
 | ||||||
|         print() |         print() | ||||||
|         print("== Tests result: %s ==" % self.get_tests_result()) |         print("== Tests result: %s ==" % self.get_tests_state()) | ||||||
| 
 | 
 | ||||||
|         if self.interrupted: |         if self.interrupted: | ||||||
|             print("Test suite interrupted by signal SIGINT.") |             print("Test suite interrupted by signal SIGINT.") | ||||||
|  | @ -410,7 +396,7 @@ def display_result(self): | ||||||
|             print(count(len(omitted), "test"), "omitted:") |             print(count(len(omitted), "test"), "omitted:") | ||||||
|             printlist(omitted) |             printlist(omitted) | ||||||
| 
 | 
 | ||||||
|         if self.good and not self.ns.quiet: |         if self.good and not quiet: | ||||||
|             print() |             print() | ||||||
|             if (not self.bad |             if (not self.bad | ||||||
|                 and not self.skipped |                 and not self.skipped | ||||||
|  | @ -419,7 +405,7 @@ def display_result(self): | ||||||
|                 print("All", end=' ') |                 print("All", end=' ') | ||||||
|             print(count(len(self.good), "test"), "OK.") |             print(count(len(self.good), "test"), "OK.") | ||||||
| 
 | 
 | ||||||
|         if self.ns.print_slow: |         if print_slow: | ||||||
|             self.test_times.sort(reverse=True) |             self.test_times.sort(reverse=True) | ||||||
|             print() |             print() | ||||||
|             print("10 slowest tests:") |             print("10 slowest tests:") | ||||||
|  | @ -437,11 +423,16 @@ def display_result(self): | ||||||
|                      count(len(self.environment_changed), "test"))) |                      count(len(self.environment_changed), "test"))) | ||||||
|             printlist(self.environment_changed) |             printlist(self.environment_changed) | ||||||
| 
 | 
 | ||||||
|         if self.skipped and not self.ns.quiet: |         if self.skipped and not quiet: | ||||||
|             print() |             print() | ||||||
|             print(count(len(self.skipped), "test"), "skipped:") |             print(count(len(self.skipped), "test"), "skipped:") | ||||||
|             printlist(self.skipped) |             printlist(self.skipped) | ||||||
| 
 | 
 | ||||||
|  |         if self.resource_denied and not quiet: | ||||||
|  |             print() | ||||||
|  |             print(count(len(self.resource_denied), "test"), "skipped (resource denied):") | ||||||
|  |             printlist(self.resource_denied) | ||||||
|  | 
 | ||||||
|         if self.rerun: |         if self.rerun: | ||||||
|             print() |             print() | ||||||
|             print("%s:" % count(len(self.rerun), "re-run test")) |             print("%s:" % count(len(self.rerun), "re-run test")) | ||||||
|  | @ -452,22 +443,7 @@ def display_result(self): | ||||||
|             print(count(len(self.run_no_tests), "test"), "run no tests:") |             print(count(len(self.run_no_tests), "test"), "run no tests:") | ||||||
|             printlist(self.run_no_tests) |             printlist(self.run_no_tests) | ||||||
| 
 | 
 | ||||||
|     def run_tests_sequential(self): |     def run_test(self, test_index, test_name, previous_test, save_modules): | ||||||
|         if self.ns.trace: |  | ||||||
|             import trace |  | ||||||
|             self.tracer = trace.Trace(trace=False, count=True) |  | ||||||
| 
 |  | ||||||
|         save_modules = sys.modules.keys() |  | ||||||
| 
 |  | ||||||
|         msg = "Run tests sequentially" |  | ||||||
|         if self.ns.timeout: |  | ||||||
|             msg += " (timeout: %s)" % format_duration(self.ns.timeout) |  | ||||||
|         self.log(msg) |  | ||||||
| 
 |  | ||||||
|         previous_test = None |  | ||||||
|         for test_index, test_name in enumerate(self.tests, 1): |  | ||||||
|             start_time = time.perf_counter() |  | ||||||
| 
 |  | ||||||
|         text = test_name |         text = test_name | ||||||
|         if previous_test: |         if previous_test: | ||||||
|             text = '%s -- %s' % (text, previous_test) |             text = '%s -- %s' % (text, previous_test) | ||||||
|  | @ -485,7 +461,40 @@ def run_tests_sequential(self): | ||||||
|             result = runtest(self.ns, test_name) |             result = runtest(self.ns, test_name) | ||||||
|             self.accumulate_result(result) |             self.accumulate_result(result) | ||||||
| 
 | 
 | ||||||
|             if result.state == State.INTERRUPTED: |         # Unload the newly imported modules (best effort finalization) | ||||||
|  |         for module in sys.modules.keys(): | ||||||
|  |             if module not in save_modules and module.startswith("test."): | ||||||
|  |                 support.unload(module) | ||||||
|  | 
 | ||||||
|  |         return result | ||||||
|  | 
 | ||||||
|  |     def run_tests_sequentially(self, runtests): | ||||||
|  |         ns = self.ns | ||||||
|  |         coverage = ns.trace | ||||||
|  |         fail_fast = ns.failfast | ||||||
|  |         fail_env_changed = ns.fail_env_changed | ||||||
|  |         timeout = ns.timeout | ||||||
|  | 
 | ||||||
|  |         if coverage: | ||||||
|  |             import trace | ||||||
|  |             self.tracer = trace.Trace(trace=False, count=True) | ||||||
|  | 
 | ||||||
|  |         save_modules = sys.modules.keys() | ||||||
|  | 
 | ||||||
|  |         msg = "Run tests sequentially" | ||||||
|  |         if timeout: | ||||||
|  |             msg += " (timeout: %s)" % format_duration(timeout) | ||||||
|  |         self.log(msg) | ||||||
|  | 
 | ||||||
|  |         previous_test = None | ||||||
|  |         tests_iter = runtests.iter_tests() | ||||||
|  |         for test_index, test_name in enumerate(tests_iter, 1): | ||||||
|  |             start_time = time.perf_counter() | ||||||
|  | 
 | ||||||
|  |             result = self.run_test(test_index, test_name, | ||||||
|  |                                    previous_test, save_modules) | ||||||
|  | 
 | ||||||
|  |             if result.must_stop(fail_fast, fail_env_changed): | ||||||
|                 break |                 break | ||||||
| 
 | 
 | ||||||
|             previous_test = str(result) |             previous_test = str(result) | ||||||
|  | @ -496,26 +505,9 @@ def run_tests_sequential(self): | ||||||
|                 # be quiet: say nothing if the test passed shortly |                 # be quiet: say nothing if the test passed shortly | ||||||
|                 previous_test = None |                 previous_test = None | ||||||
| 
 | 
 | ||||||
|             # Unload the newly imported modules (best effort finalization) |  | ||||||
|             for module in sys.modules.keys(): |  | ||||||
|                 if module not in save_modules and module.startswith("test."): |  | ||||||
|                     support.unload(module) |  | ||||||
| 
 |  | ||||||
|             if self.ns.failfast and result.is_failed(self.ns.fail_env_changed): |  | ||||||
|                 break |  | ||||||
| 
 |  | ||||||
|         if previous_test: |         if previous_test: | ||||||
|             print(previous_test) |             print(previous_test) | ||||||
| 
 | 
 | ||||||
|     def _test_forever(self, tests): |  | ||||||
|         while True: |  | ||||||
|             for test_name in tests: |  | ||||||
|                 yield test_name |  | ||||||
|                 if self.bad: |  | ||||||
|                     return |  | ||||||
|                 if self.ns.fail_env_changed and self.environment_changed: |  | ||||||
|                     return |  | ||||||
| 
 |  | ||||||
|     def display_header(self): |     def display_header(self): | ||||||
|         # Print basic platform information |         # Print basic platform information | ||||||
|         print("==", platform.python_implementation(), *sys.version.split()) |         print("==", platform.python_implementation(), *sys.version.split()) | ||||||
|  | @ -560,11 +552,13 @@ def no_tests_run(self): | ||||||
|         return not any((self.good, self.bad, self.skipped, self.interrupted, |         return not any((self.good, self.bad, self.skipped, self.interrupted, | ||||||
|                         self.environment_changed)) |                         self.environment_changed)) | ||||||
| 
 | 
 | ||||||
|     def get_tests_result(self): |     def get_tests_state(self): | ||||||
|  |         fail_env_changed = self.ns.fail_env_changed | ||||||
|  | 
 | ||||||
|         result = [] |         result = [] | ||||||
|         if self.bad: |         if self.bad: | ||||||
|             result.append("FAILURE") |             result.append("FAILURE") | ||||||
|         elif self.ns.fail_env_changed and self.environment_changed: |         elif fail_env_changed and self.environment_changed: | ||||||
|             result.append("ENV CHANGED") |             result.append("ENV CHANGED") | ||||||
|         elif self.no_tests_run(): |         elif self.no_tests_run(): | ||||||
|             result.append("NO TESTS RAN") |             result.append("NO TESTS RAN") | ||||||
|  | @ -576,10 +570,40 @@ def get_tests_result(self): | ||||||
|             result.append("SUCCESS") |             result.append("SUCCESS") | ||||||
| 
 | 
 | ||||||
|         result = ', '.join(result) |         result = ', '.join(result) | ||||||
|         if self.first_result: |         if self.first_state: | ||||||
|             result = '%s then %s' % (self.first_result, result) |             result = '%s then %s' % (self.first_state, result) | ||||||
|         return result |         return result | ||||||
| 
 | 
 | ||||||
|  |     def _run_tests_mp(self, runtests: RunTests) -> None: | ||||||
|  |         from test.libregrtest.runtest_mp import run_tests_multiprocess | ||||||
|  |         # If we're on windows and this is the parent runner (not a worker), | ||||||
|  |         # track the load average. | ||||||
|  |         if sys.platform == 'win32': | ||||||
|  |             from test.libregrtest.win_utils import WindowsLoadTracker | ||||||
|  | 
 | ||||||
|  |             try: | ||||||
|  |                 self.win_load_tracker = WindowsLoadTracker() | ||||||
|  |             except PermissionError as error: | ||||||
|  |                 # Standard accounts may not have access to the performance | ||||||
|  |                 # counters. | ||||||
|  |                 print(f'Failed to create WindowsLoadTracker: {error}') | ||||||
|  | 
 | ||||||
|  |         try: | ||||||
|  |             run_tests_multiprocess(self, runtests) | ||||||
|  |         finally: | ||||||
|  |             if self.win_load_tracker is not None: | ||||||
|  |                 self.win_load_tracker.close() | ||||||
|  |                 self.win_load_tracker = None | ||||||
|  | 
 | ||||||
|  |     def set_tests(self, tests): | ||||||
|  |         self.tests = tests | ||||||
|  |         if self.ns.forever: | ||||||
|  |             self.test_count_text = '' | ||||||
|  |             self.test_count_width = 3 | ||||||
|  |         else: | ||||||
|  |             self.test_count_text = '/{}'.format(len(self.tests)) | ||||||
|  |             self.test_count_width = len(self.test_count_text) - 1 | ||||||
|  | 
 | ||||||
|     def run_tests(self): |     def run_tests(self): | ||||||
|         # For a partial run, we do not need to clutter the output. |         # For a partial run, we do not need to clutter the output. | ||||||
|         if (self.ns.header |         if (self.ns.header | ||||||
|  | @ -597,37 +621,14 @@ def run_tests(self): | ||||||
|         if self.ns.randomize: |         if self.ns.randomize: | ||||||
|             print("Using random seed", self.ns.random_seed) |             print("Using random seed", self.ns.random_seed) | ||||||
| 
 | 
 | ||||||
|         if self.ns.forever: |         tests = self.selected | ||||||
|             self.tests = self._test_forever(list(self.selected)) |         self.set_tests(tests) | ||||||
|             self.test_count = '' |         runtests = RunTests(tests, forever=self.ns.forever) | ||||||
|             self.test_count_width = 3 |         self.all_runtests.append(runtests) | ||||||
|         else: |  | ||||||
|             self.tests = iter(self.selected) |  | ||||||
|             self.test_count = '/{}'.format(len(self.selected)) |  | ||||||
|             self.test_count_width = len(self.test_count) - 1 |  | ||||||
| 
 |  | ||||||
|         if self.ns.use_mp: |         if self.ns.use_mp: | ||||||
|             from test.libregrtest.runtest_mp import run_tests_multiprocess |             self._run_tests_mp(runtests) | ||||||
|             # If we're on windows and this is the parent runner (not a worker), |  | ||||||
|             # track the load average. |  | ||||||
|             if sys.platform == 'win32' and self.worker_test_name is None: |  | ||||||
|                 from test.libregrtest.win_utils import WindowsLoadTracker |  | ||||||
| 
 |  | ||||||
|                 try: |  | ||||||
|                     self.win_load_tracker = WindowsLoadTracker() |  | ||||||
|                 except PermissionError as error: |  | ||||||
|                     # Standard accounts may not have access to the performance |  | ||||||
|                     # counters. |  | ||||||
|                     print(f'Failed to create WindowsLoadTracker: {error}') |  | ||||||
| 
 |  | ||||||
|             try: |  | ||||||
|                 run_tests_multiprocess(self) |  | ||||||
|             finally: |  | ||||||
|                 if self.win_load_tracker is not None: |  | ||||||
|                     self.win_load_tracker.close() |  | ||||||
|                     self.win_load_tracker = None |  | ||||||
|         else: |         else: | ||||||
|             self.run_tests_sequential() |             self.run_tests_sequentially(runtests) | ||||||
| 
 | 
 | ||||||
|     def finalize(self): |     def finalize(self): | ||||||
|         if self.next_single_filename: |         if self.next_single_filename: | ||||||
|  | @ -642,23 +643,29 @@ def finalize(self): | ||||||
|             r.write_results(show_missing=True, summary=True, |             r.write_results(show_missing=True, summary=True, | ||||||
|                             coverdir=self.ns.coverdir) |                             coverdir=self.ns.coverdir) | ||||||
| 
 | 
 | ||||||
|         print() |  | ||||||
|         self.display_summary() |  | ||||||
| 
 |  | ||||||
|         if self.ns.runleaks: |         if self.ns.runleaks: | ||||||
|             os.system("leaks %d" % os.getpid()) |             os.system("leaks %d" % os.getpid()) | ||||||
| 
 | 
 | ||||||
|  |         self.save_xml_result() | ||||||
|  | 
 | ||||||
|     def display_summary(self): |     def display_summary(self): | ||||||
|         duration = time.perf_counter() - self.start_time |         duration = time.perf_counter() - self.start_time | ||||||
|  |         first_runtests = self.all_runtests[0] | ||||||
|  |         # the second runtests (re-run failed tests) disables forever, | ||||||
|  |         # use the first runtests | ||||||
|  |         forever = first_runtests.forever | ||||||
|  |         filtered = bool(self.ns.match_tests) or bool(self.ns.ignore_tests) | ||||||
| 
 | 
 | ||||||
|         # Total duration |         # Total duration | ||||||
|  |         print() | ||||||
|         print("Total duration: %s" % format_duration(duration)) |         print("Total duration: %s" % format_duration(duration)) | ||||||
| 
 | 
 | ||||||
|         # Total tests |         # Total tests | ||||||
|         total = TestStats() |         total = self.total_stats | ||||||
|         for stats in self.stats_dict.values(): |         text = f'run={total.tests_run:,}' | ||||||
|             total.accumulate(stats) |         if filtered: | ||||||
|         stats = [f'run={total.tests_run:,}'] |             text = f"{text} (filtered)" | ||||||
|  |         stats = [text] | ||||||
|         if total.failures: |         if total.failures: | ||||||
|             stats.append(f'failures={total.failures:,}') |             stats.append(f'failures={total.failures:,}') | ||||||
|         if total.skipped: |         if total.skipped: | ||||||
|  | @ -666,23 +673,31 @@ def display_summary(self): | ||||||
|         print(f"Total tests: {' '.join(stats)}") |         print(f"Total tests: {' '.join(stats)}") | ||||||
| 
 | 
 | ||||||
|         # Total test files |         # Total test files | ||||||
|         report = [f'success={len(self.good)}'] |         all_tests = [self.good, self.bad, self.rerun, | ||||||
|         if self.bad: |                      self.skipped, | ||||||
|             report.append(f'failed={len(self.bad)}') |                      self.environment_changed, self.run_no_tests] | ||||||
|         if self.environment_changed: |         run = sum(map(len, all_tests)) | ||||||
|             report.append(f'env_changed={len(self.environment_changed)}') |         text = f'run={run}' | ||||||
|         if self.skipped: |         if not forever: | ||||||
|             report.append(f'skipped={len(self.skipped)}') |             ntest = len(first_runtests.tests) | ||||||
|         if self.resource_denied: |             text = f"{text}/{ntest}" | ||||||
|             report.append(f'resource_denied={len(self.resource_denied)}') |         if filtered: | ||||||
|         if self.rerun: |             text = f"{text} (filtered)" | ||||||
|             report.append(f'rerun={len(self.rerun)}') |         report = [text] | ||||||
|         if self.run_no_tests: |         for name, tests in ( | ||||||
|             report.append(f'run_no_tests={len(self.run_no_tests)}') |             ('failed', self.bad), | ||||||
|  |             ('env_changed', self.environment_changed), | ||||||
|  |             ('skipped', self.skipped), | ||||||
|  |             ('resource_denied', self.resource_denied), | ||||||
|  |             ('rerun', self.rerun), | ||||||
|  |             ('run_no_tests', self.run_no_tests), | ||||||
|  |         ): | ||||||
|  |             if tests: | ||||||
|  |                 report.append(f'{name}={len(tests)}') | ||||||
|         print(f"Total test files: {' '.join(report)}") |         print(f"Total test files: {' '.join(report)}") | ||||||
| 
 | 
 | ||||||
|         # Result |         # Result | ||||||
|         result = self.get_tests_result() |         result = self.get_tests_state() | ||||||
|         print(f"Result: {result}") |         print(f"Result: {result}") | ||||||
| 
 | 
 | ||||||
|     def save_xml_result(self): |     def save_xml_result(self): | ||||||
|  | @ -742,6 +757,9 @@ def set_temp_dir(self): | ||||||
| 
 | 
 | ||||||
|         self.tmp_dir = os.path.abspath(self.tmp_dir) |         self.tmp_dir = os.path.abspath(self.tmp_dir) | ||||||
| 
 | 
 | ||||||
|  |     def is_worker(self): | ||||||
|  |         return (self.ns.worker_args is not None) | ||||||
|  | 
 | ||||||
|     def create_temp_dir(self): |     def create_temp_dir(self): | ||||||
|         os.makedirs(self.tmp_dir, exist_ok=True) |         os.makedirs(self.tmp_dir, exist_ok=True) | ||||||
| 
 | 
 | ||||||
|  | @ -754,7 +772,8 @@ def create_temp_dir(self): | ||||||
|             nounce = random.randint(0, 1_000_000) |             nounce = random.randint(0, 1_000_000) | ||||||
|         else: |         else: | ||||||
|             nounce = os.getpid() |             nounce = os.getpid() | ||||||
|         if self.worker_test_name is not None: | 
 | ||||||
|  |         if self.is_worker(): | ||||||
|             test_cwd = 'test_python_worker_{}'.format(nounce) |             test_cwd = 'test_python_worker_{}'.format(nounce) | ||||||
|         else: |         else: | ||||||
|             test_cwd = 'test_python_{}'.format(nounce) |             test_cwd = 'test_python_{}'.format(nounce) | ||||||
|  | @ -817,48 +836,53 @@ def getloadavg(self): | ||||||
| 
 | 
 | ||||||
|         return None |         return None | ||||||
| 
 | 
 | ||||||
|  |     def get_exitcode(self): | ||||||
|  |         exitcode = 0 | ||||||
|  |         if self.bad: | ||||||
|  |             exitcode = EXITCODE_BAD_TEST | ||||||
|  |         elif self.interrupted: | ||||||
|  |             exitcode = EXITCODE_INTERRUPTED | ||||||
|  |         elif self.ns.fail_env_changed and self.environment_changed: | ||||||
|  |             exitcode = EXITCODE_ENV_CHANGED | ||||||
|  |         elif self.no_tests_run(): | ||||||
|  |             exitcode = EXITCODE_NO_TESTS_RAN | ||||||
|  |         elif self.rerun and self.ns.fail_rerun: | ||||||
|  |             exitcode = EXITCODE_BAD_TEST | ||||||
|  |         return exitcode | ||||||
|  | 
 | ||||||
|  |     def action_run_tests(self): | ||||||
|  |         self.run_tests() | ||||||
|  |         self.display_result() | ||||||
|  | 
 | ||||||
|  |         need_rerun = self.need_rerun | ||||||
|  |         if self.ns.rerun and need_rerun: | ||||||
|  |             self.rerun_failed_tests(need_rerun) | ||||||
|  | 
 | ||||||
|  |         self.display_summary() | ||||||
|  |         self.finalize() | ||||||
|  | 
 | ||||||
|     def _main(self, tests, kwargs): |     def _main(self, tests, kwargs): | ||||||
|         if self.worker_test_name is not None: |         if self.is_worker(): | ||||||
|             from test.libregrtest.runtest_mp import run_tests_worker |             from test.libregrtest.runtest_mp import run_tests_worker | ||||||
|             run_tests_worker(self.ns, self.worker_test_name) |             run_tests_worker(self.ns.worker_args) | ||||||
|  |             return | ||||||
| 
 | 
 | ||||||
|         if self.ns.wait: |         if self.ns.wait: | ||||||
|             input("Press any key to continue...") |             input("Press any key to continue...") | ||||||
| 
 | 
 | ||||||
|         support.PGO = self.ns.pgo |  | ||||||
|         support.PGO_EXTENDED = self.ns.pgo_extended |  | ||||||
| 
 |  | ||||||
|         setup_tests(self.ns) |         setup_tests(self.ns) | ||||||
| 
 |  | ||||||
|         self.find_tests(tests) |         self.find_tests(tests) | ||||||
| 
 | 
 | ||||||
|  |         exitcode = 0 | ||||||
|         if self.ns.list_tests: |         if self.ns.list_tests: | ||||||
|             self.list_tests() |             self.list_tests() | ||||||
|             sys.exit(0) |         elif self.ns.list_cases: | ||||||
| 
 |  | ||||||
|         if self.ns.list_cases: |  | ||||||
|             self.list_cases() |             self.list_cases() | ||||||
|             sys.exit(0) |         else: | ||||||
|  |             self.action_run_tests() | ||||||
|  |             exitcode = self.get_exitcode() | ||||||
| 
 | 
 | ||||||
|         self.run_tests() |         sys.exit(exitcode) | ||||||
|         self.display_result() |  | ||||||
| 
 |  | ||||||
|         if self.ns.verbose2 and self.bad: |  | ||||||
|             self.rerun_failed_tests() |  | ||||||
| 
 |  | ||||||
|         self.finalize() |  | ||||||
| 
 |  | ||||||
|         self.save_xml_result() |  | ||||||
| 
 |  | ||||||
|         if self.bad: |  | ||||||
|             sys.exit(EXITCODE_BAD_TEST) |  | ||||||
|         if self.interrupted: |  | ||||||
|             sys.exit(EXITCODE_INTERRUPTED) |  | ||||||
|         if self.ns.fail_env_changed and self.environment_changed: |  | ||||||
|             sys.exit(EXITCODE_ENV_CHANGED) |  | ||||||
|         if self.no_tests_run(): |  | ||||||
|             sys.exit(EXITCODE_NO_TESTS_RAN) |  | ||||||
|         sys.exit(0) |  | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def main(tests=None, **kwargs): | def main(tests=None, **kwargs): | ||||||
|  |  | ||||||
|  | @ -1,7 +1,6 @@ | ||||||
| import dataclasses | import dataclasses | ||||||
| import doctest | import doctest | ||||||
| import faulthandler | import faulthandler | ||||||
| import functools |  | ||||||
| import gc | import gc | ||||||
| import importlib | import importlib | ||||||
| import io | import io | ||||||
|  | @ -20,6 +19,10 @@ | ||||||
| from test.libregrtest.utils import clear_caches, format_duration, print_warning | from test.libregrtest.utils import clear_caches, format_duration, print_warning | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | MatchTests = list[str] | ||||||
|  | MatchTestsDict = dict[str, MatchTests] | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| # Avoid enum.Enum to reduce the number of imports when tests are run | # Avoid enum.Enum to reduce the number of imports when tests are run | ||||||
| class State: | class State: | ||||||
|     PASSED = "PASSED" |     PASSED = "PASSED" | ||||||
|  | @ -56,6 +59,41 @@ def has_meaningful_duration(state): | ||||||
|             State.MULTIPROCESSING_ERROR, |             State.MULTIPROCESSING_ERROR, | ||||||
|             State.DID_NOT_RUN} |             State.DID_NOT_RUN} | ||||||
| 
 | 
 | ||||||
|  |     @staticmethod | ||||||
|  |     def must_stop(state): | ||||||
|  |         return state in { | ||||||
|  |             State.INTERRUPTED, | ||||||
|  |             State.MULTIPROCESSING_ERROR} | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | # gh-90681: When rerunning tests, we might need to rerun the whole | ||||||
|  | # class or module suite if some its life-cycle hooks fail. | ||||||
|  | # Test level hooks are not affected. | ||||||
|  | _TEST_LIFECYCLE_HOOKS = frozenset(( | ||||||
|  |     'setUpClass', 'tearDownClass', | ||||||
|  |     'setUpModule', 'tearDownModule', | ||||||
|  | )) | ||||||
|  | 
 | ||||||
|  | def normalize_test_name(test_full_name, *, is_error=False): | ||||||
|  |     short_name = test_full_name.split(" ")[0] | ||||||
|  |     if is_error and short_name in _TEST_LIFECYCLE_HOOKS: | ||||||
|  |         if test_full_name.startswith(('setUpModule (', 'tearDownModule (')): | ||||||
|  |             # if setUpModule() or tearDownModule() failed, don't filter | ||||||
|  |             # tests with the test file name, don't use use filters. | ||||||
|  |             return None | ||||||
|  | 
 | ||||||
|  |         # This means that we have a failure in a life-cycle hook, | ||||||
|  |         # we need to rerun the whole module or class suite. | ||||||
|  |         # Basically the error looks like this: | ||||||
|  |         #    ERROR: setUpClass (test.test_reg_ex.RegTest) | ||||||
|  |         # or | ||||||
|  |         #    ERROR: setUpModule (test.test_reg_ex) | ||||||
|  |         # So, we need to parse the class / module name. | ||||||
|  |         lpar = test_full_name.index('(') | ||||||
|  |         rpar = test_full_name.index(')') | ||||||
|  |         return test_full_name[lpar + 1: rpar].split('.')[-1] | ||||||
|  |     return short_name | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
| @dataclasses.dataclass(slots=True) | @dataclasses.dataclass(slots=True) | ||||||
| class TestResult: | class TestResult: | ||||||
|  | @ -129,6 +167,58 @@ def set_env_changed(self): | ||||||
|         if self.state is None or self.state == State.PASSED: |         if self.state is None or self.state == State.PASSED: | ||||||
|             self.state = State.ENV_CHANGED |             self.state = State.ENV_CHANGED | ||||||
| 
 | 
 | ||||||
|  |     def must_stop(self, fail_fast: bool, fail_env_changed: bool) -> bool: | ||||||
|  |         if State.must_stop(self.state): | ||||||
|  |             return True | ||||||
|  |         if fail_fast and self.is_failed(fail_env_changed): | ||||||
|  |             return True | ||||||
|  |         return False | ||||||
|  | 
 | ||||||
|  |     def get_rerun_match_tests(self): | ||||||
|  |         match_tests = [] | ||||||
|  | 
 | ||||||
|  |         errors = self.errors or [] | ||||||
|  |         failures = self.failures or [] | ||||||
|  |         for error_list, is_error in ( | ||||||
|  |             (errors, True), | ||||||
|  |             (failures, False), | ||||||
|  |         ): | ||||||
|  |             for full_name, *_ in error_list: | ||||||
|  |                 match_name = normalize_test_name(full_name, is_error=is_error) | ||||||
|  |                 if match_name is None: | ||||||
|  |                     # 'setUpModule (test.test_sys)': don't filter tests | ||||||
|  |                     return None | ||||||
|  |                 if not match_name: | ||||||
|  |                     error_type = "ERROR" if is_error else "FAIL" | ||||||
|  |                     print_warning(f"rerun failed to parse {error_type} test name: " | ||||||
|  |                                   f"{full_name!r}: don't filter tests") | ||||||
|  |                     return None | ||||||
|  |                 match_tests.append(match_name) | ||||||
|  | 
 | ||||||
|  |         return match_tests | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | @dataclasses.dataclass(slots=True, frozen=True) | ||||||
|  | class RunTests: | ||||||
|  |     tests: list[str] | ||||||
|  |     match_tests: MatchTestsDict | None = None | ||||||
|  |     rerun: bool = False | ||||||
|  |     forever: bool = False | ||||||
|  | 
 | ||||||
|  |     def get_match_tests(self, test_name) -> MatchTests | None: | ||||||
|  |         if self.match_tests is not None: | ||||||
|  |             return self.match_tests.get(test_name, None) | ||||||
|  |         else: | ||||||
|  |             return None | ||||||
|  | 
 | ||||||
|  |     def iter_tests(self): | ||||||
|  |         tests = tuple(self.tests) | ||||||
|  |         if self.forever: | ||||||
|  |             while True: | ||||||
|  |                 yield from tests | ||||||
|  |         else: | ||||||
|  |             yield from tests | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
| # Minimum duration of a test to display its duration or to mention that | # Minimum duration of a test to display its duration or to mention that | ||||||
| # the test is running in background | # the test is running in background | ||||||
|  | @ -147,9 +237,6 @@ def set_env_changed(self): | ||||||
|     "test_multiprocessing_spawn", |     "test_multiprocessing_spawn", | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| # Storage of uncollectable objects |  | ||||||
| FOUND_GARBAGE = [] |  | ||||||
| 
 |  | ||||||
| 
 | 
 | ||||||
| def findtestdir(path=None): | def findtestdir(path=None): | ||||||
|     return path or os.path.dirname(os.path.dirname(__file__)) or os.curdir |     return path or os.path.dirname(os.path.dirname(__file__)) or os.curdir | ||||||
|  | @ -189,31 +276,41 @@ def split_test_packages(tests, *, testdir=None, exclude=(), | ||||||
|     return splitted |     return splitted | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def get_abs_module(ns: Namespace, test_name: str) -> str: | def abs_module_name(test_name: str, test_dir: str | None) -> str: | ||||||
|     if test_name.startswith('test.') or ns.testdir: |     if test_name.startswith('test.') or test_dir: | ||||||
|         return test_name |         return test_name | ||||||
|     else: |     else: | ||||||
|         # Import it from the test package |         # Import it from the test package | ||||||
|         return 'test.' + test_name |         return 'test.' + test_name | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def _runtest_capture_output_timeout_junit(result: TestResult, ns: Namespace) -> None: | def setup_support(ns: Namespace): | ||||||
|  |     support.PGO = ns.pgo | ||||||
|  |     support.PGO_EXTENDED = ns.pgo_extended | ||||||
|  |     support.set_match_tests(ns.match_tests, ns.ignore_tests) | ||||||
|  |     support.failfast = ns.failfast | ||||||
|  |     support.verbose = ns.verbose | ||||||
|  |     if ns.xmlpath: | ||||||
|  |         support.junit_xml_list = [] | ||||||
|  |     else: | ||||||
|  |         support.junit_xml_list = None | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def _runtest(result: TestResult, ns: Namespace) -> None: | ||||||
|     # Capture stdout and stderr, set faulthandler timeout, |     # Capture stdout and stderr, set faulthandler timeout, | ||||||
|     # and create JUnit XML report. |     # and create JUnit XML report. | ||||||
| 
 |     verbose = ns.verbose | ||||||
|     output_on_failure = ns.verbose3 |     output_on_failure = ns.verbose3 | ||||||
|  |     timeout = ns.timeout | ||||||
| 
 | 
 | ||||||
|     use_timeout = ( |     use_timeout = ( | ||||||
|         ns.timeout is not None and threading_helper.can_start_thread |         timeout is not None and threading_helper.can_start_thread | ||||||
|     ) |     ) | ||||||
|     if use_timeout: |     if use_timeout: | ||||||
|         faulthandler.dump_traceback_later(ns.timeout, exit=True) |         faulthandler.dump_traceback_later(timeout, exit=True) | ||||||
| 
 | 
 | ||||||
|     try: |     try: | ||||||
|         support.set_match_tests(ns.match_tests, ns.ignore_tests) |         setup_support(ns) | ||||||
|         support.junit_xml_list = xml_list = [] if ns.xmlpath else None |  | ||||||
|         if ns.failfast: |  | ||||||
|             support.failfast = True |  | ||||||
| 
 | 
 | ||||||
|         if output_on_failure: |         if output_on_failure: | ||||||
|             support.verbose = True |             support.verbose = True | ||||||
|  | @ -247,11 +344,10 @@ def _runtest_capture_output_timeout_junit(result: TestResult, ns: Namespace) -> | ||||||
|                 sys.stderr.flush() |                 sys.stderr.flush() | ||||||
|         else: |         else: | ||||||
|             # Tell tests to be moderately quiet |             # Tell tests to be moderately quiet | ||||||
|             support.verbose = ns.verbose |             support.verbose = verbose | ||||||
| 
 |             _runtest_env_changed_exc(result, ns, display_failure=not verbose) | ||||||
|             _runtest_env_changed_exc(result, ns, |  | ||||||
|                                      display_failure=not ns.verbose) |  | ||||||
| 
 | 
 | ||||||
|  |         xml_list = support.junit_xml_list | ||||||
|         if xml_list: |         if xml_list: | ||||||
|             import xml.etree.ElementTree as ET |             import xml.etree.ElementTree as ET | ||||||
|             result.xml_data = [ET.tostring(x).decode('us-ascii') |             result.xml_data = [ET.tostring(x).decode('us-ascii') | ||||||
|  | @ -276,7 +372,7 @@ def runtest(ns: Namespace, test_name: str) -> TestResult: | ||||||
|     start_time = time.perf_counter() |     start_time = time.perf_counter() | ||||||
|     result = TestResult(test_name) |     result = TestResult(test_name) | ||||||
|     try: |     try: | ||||||
|         _runtest_capture_output_timeout_junit(result, ns) |         _runtest(result, ns) | ||||||
|     except: |     except: | ||||||
|         if not ns.pgo: |         if not ns.pgo: | ||||||
|             msg = traceback.format_exc() |             msg = traceback.format_exc() | ||||||
|  | @ -287,9 +383,9 @@ def runtest(ns: Namespace, test_name: str) -> TestResult: | ||||||
|     return result |     return result | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def _test_module(the_module): | def run_unittest(test_mod): | ||||||
|     loader = unittest.TestLoader() |     loader = unittest.TestLoader() | ||||||
|     tests = loader.loadTestsFromModule(the_module) |     tests = loader.loadTestsFromModule(test_mod) | ||||||
|     for error in loader.errors: |     for error in loader.errors: | ||||||
|         print(error, file=sys.stderr) |         print(error, file=sys.stderr) | ||||||
|     if loader.errors: |     if loader.errors: | ||||||
|  | @ -304,7 +400,6 @@ def save_env(ns: Namespace, test_name: str): | ||||||
| def regrtest_runner(result, test_func, ns) -> None: | def regrtest_runner(result, test_func, ns) -> None: | ||||||
|     # Run test_func(), collect statistics, and detect reference and memory |     # Run test_func(), collect statistics, and detect reference and memory | ||||||
|     # leaks. |     # leaks. | ||||||
| 
 |  | ||||||
|     if ns.huntrleaks: |     if ns.huntrleaks: | ||||||
|         from test.libregrtest.refleak import dash_R |         from test.libregrtest.refleak import dash_R | ||||||
|         refleak, test_result = dash_R(ns, result.test_name, test_func) |         refleak, test_result = dash_R(ns, result.test_name, test_func) | ||||||
|  | @ -332,24 +427,27 @@ def regrtest_runner(result, test_func, ns) -> None: | ||||||
|     result.stats = stats |     result.stats = stats | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | # Storage of uncollectable objects | ||||||
|  | FOUND_GARBAGE = [] | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| def _load_run_test(result: TestResult, ns: Namespace) -> None: | def _load_run_test(result: TestResult, ns: Namespace) -> None: | ||||||
|     # Load the test function, run the test function. |     # Load the test function, run the test function. | ||||||
|  |     module_name = abs_module_name(result.test_name, ns.testdir) | ||||||
| 
 | 
 | ||||||
|     abstest = get_abs_module(ns, result.test_name) |     # Remove the module from sys.module to reload it if it was already imported | ||||||
|  |     sys.modules.pop(module_name, None) | ||||||
| 
 | 
 | ||||||
|     # remove the module from sys.module to reload it if it was already imported |     test_mod = importlib.import_module(module_name) | ||||||
|     try: |  | ||||||
|         del sys.modules[abstest] |  | ||||||
|     except KeyError: |  | ||||||
|         pass |  | ||||||
| 
 |  | ||||||
|     the_module = importlib.import_module(abstest) |  | ||||||
| 
 | 
 | ||||||
|     # If the test has a test_main, that will run the appropriate |     # If the test has a test_main, that will run the appropriate | ||||||
|     # tests.  If not, use normal unittest test loading. |     # tests.  If not, use normal unittest test runner. | ||||||
|     test_func = getattr(the_module, "test_main", None) |     test_main = getattr(test_mod, "test_main", None) | ||||||
|     if test_func is None: |     if test_main is not None: | ||||||
|         test_func = functools.partial(_test_module, the_module) |         test_func = test_main | ||||||
|  |     else: | ||||||
|  |         def test_func(): | ||||||
|  |             return run_unittest(test_mod) | ||||||
| 
 | 
 | ||||||
|     try: |     try: | ||||||
|         with save_env(ns, result.test_name): |         with save_env(ns, result.test_name): | ||||||
|  | @ -361,12 +459,12 @@ def _load_run_test(result: TestResult, ns: Namespace) -> None: | ||||||
|         # failures. |         # failures. | ||||||
|         support.gc_collect() |         support.gc_collect() | ||||||
| 
 | 
 | ||||||
|         cleanup_test_droppings(result.test_name, ns.verbose) |         remove_testfn(result.test_name, ns.verbose) | ||||||
| 
 | 
 | ||||||
|     if gc.garbage: |     if gc.garbage: | ||||||
|         support.environment_altered = True |         support.environment_altered = True | ||||||
|         print_warning(f"{result.test_name} created {len(gc.garbage)} " |         print_warning(f"{result.test_name} created {len(gc.garbage)} " | ||||||
|                       f"uncollectable object(s).") |                       f"uncollectable object(s)") | ||||||
| 
 | 
 | ||||||
|         # move the uncollectable objects somewhere, |         # move the uncollectable objects somewhere, | ||||||
|         # so we don't see them again |         # so we don't see them again | ||||||
|  | @ -444,16 +542,18 @@ def _runtest_env_changed_exc(result: TestResult, ns: Namespace, | ||||||
|         result.state = State.PASSED |         result.state = State.PASSED | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def cleanup_test_droppings(test_name: str, verbose: int) -> None: | def remove_testfn(test_name: str, verbose: int) -> None: | ||||||
|     # Try to clean up junk commonly left behind.  While tests shouldn't leave |     # Try to clean up os_helper.TESTFN if left behind. | ||||||
|     # any files or directories behind, when a test fails that can be tedious |     # | ||||||
|     # for it to arrange.  The consequences can be especially nasty on Windows, |     # While tests shouldn't leave any files or directories behind, when a test | ||||||
|     # since if a test leaves a file open, it cannot be deleted by name (while |     # fails that can be tedious for it to arrange.  The consequences can be | ||||||
|     # there's nothing we can do about that here either, we can display the |     # especially nasty on Windows, since if a test leaves a file open, it | ||||||
|     # name of the offending test, which is a real help). |     # cannot be deleted by name (while there's nothing we can do about that | ||||||
|     for name in (os_helper.TESTFN,): |     # here either, we can display the name of the offending test, which is a | ||||||
|  |     # real help). | ||||||
|  |     name = os_helper.TESTFN | ||||||
|     if not os.path.exists(name): |     if not os.path.exists(name): | ||||||
|             continue |         return | ||||||
| 
 | 
 | ||||||
|     if os.path.isdir(name): |     if os.path.isdir(name): | ||||||
|         import shutil |         import shutil | ||||||
|  |  | ||||||
|  | @ -19,8 +19,8 @@ | ||||||
| from test.libregrtest.cmdline import Namespace | from test.libregrtest.cmdline import Namespace | ||||||
| from test.libregrtest.main import Regrtest | from test.libregrtest.main import Regrtest | ||||||
| from test.libregrtest.runtest import ( | from test.libregrtest.runtest import ( | ||||||
|     runtest, TestResult, State, |     runtest, TestResult, State, PROGRESS_MIN_TIME, | ||||||
|     PROGRESS_MIN_TIME) |     MatchTests, RunTests) | ||||||
| from test.libregrtest.setup import setup_tests | from test.libregrtest.setup import setup_tests | ||||||
| from test.libregrtest.utils import format_duration, print_warning | from test.libregrtest.utils import format_duration, print_warning | ||||||
| 
 | 
 | ||||||
|  | @ -44,26 +44,54 @@ | ||||||
| USE_PROCESS_GROUP = (hasattr(os, "setsid") and hasattr(os, "killpg")) | USE_PROCESS_GROUP = (hasattr(os, "setsid") and hasattr(os, "killpg")) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def must_stop(result: TestResult, ns: Namespace) -> bool: | @dataclasses.dataclass(slots=True) | ||||||
|     if result.state == State.INTERRUPTED: | class WorkerJob: | ||||||
|         return True |     test_name: str | ||||||
|     if ns.failfast and result.is_failed(ns.fail_env_changed): |     namespace: Namespace | ||||||
|         return True |     rerun: bool = False | ||||||
|     return False |     match_tests: MatchTests | None = None | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def parse_worker_args(worker_args) -> tuple[Namespace, str]: | class _EncodeWorkerJob(json.JSONEncoder): | ||||||
|     ns_dict, test_name = json.loads(worker_args) |     def default(self, o: Any) -> dict[str, Any]: | ||||||
|     ns = Namespace(**ns_dict) |         match o: | ||||||
|     return (ns, test_name) |             case WorkerJob(): | ||||||
|  |                 result = dataclasses.asdict(o) | ||||||
|  |                 result["__worker_job__"] = True | ||||||
|  |                 return result | ||||||
|  |             case Namespace(): | ||||||
|  |                 result = vars(o) | ||||||
|  |                 result["__namespace__"] = True | ||||||
|  |                 return result | ||||||
|  |             case _: | ||||||
|  |                 return super().default(o) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def run_test_in_subprocess(testname: str, ns: Namespace, tmp_dir: str, stdout_fh: TextIO) -> subprocess.Popen: | def _decode_worker_job(d: dict[str, Any]) -> WorkerJob | dict[str, Any]: | ||||||
|     ns_dict = vars(ns) |     if "__worker_job__" in d: | ||||||
|     worker_args = (ns_dict, testname) |         d.pop('__worker_job__') | ||||||
|     worker_args = json.dumps(worker_args) |         return WorkerJob(**d) | ||||||
|     if ns.python is not None: |     if "__namespace__" in d: | ||||||
|         executable = ns.python |         d.pop('__namespace__') | ||||||
|  |         return Namespace(**d) | ||||||
|  |     else: | ||||||
|  |         return d | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def _parse_worker_args(worker_json: str) -> tuple[Namespace, str]: | ||||||
|  |     return json.loads(worker_json, | ||||||
|  |                       object_hook=_decode_worker_job) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def run_test_in_subprocess(worker_job: WorkerJob, | ||||||
|  |                            output_file: TextIO, | ||||||
|  |                            tmp_dir: str | None = None) -> subprocess.Popen: | ||||||
|  |     ns = worker_job.namespace | ||||||
|  |     python = ns.python | ||||||
|  |     worker_args = json.dumps(worker_job, cls=_EncodeWorkerJob) | ||||||
|  | 
 | ||||||
|  |     if python is not None: | ||||||
|  |         executable = python | ||||||
|     else: |     else: | ||||||
|         executable = [sys.executable] |         executable = [sys.executable] | ||||||
|     cmd = [*executable, *support.args_from_interpreter_flags(), |     cmd = [*executable, *support.args_from_interpreter_flags(), | ||||||
|  | @ -82,9 +110,9 @@ def run_test_in_subprocess(testname: str, ns: Namespace, tmp_dir: str, stdout_fh | ||||||
|     # sysconfig.is_python_build() is true. See issue 15300. |     # sysconfig.is_python_build() is true. See issue 15300. | ||||||
|     kw = dict( |     kw = dict( | ||||||
|         env=env, |         env=env, | ||||||
|         stdout=stdout_fh, |         stdout=output_file, | ||||||
|         # bpo-45410: Write stderr into stdout to keep messages order |         # bpo-45410: Write stderr into stdout to keep messages order | ||||||
|         stderr=stdout_fh, |         stderr=output_file, | ||||||
|         text=True, |         text=True, | ||||||
|         close_fds=(os.name != 'nt'), |         close_fds=(os.name != 'nt'), | ||||||
|         cwd=os_helper.SAVEDCWD, |         cwd=os_helper.SAVEDCWD, | ||||||
|  | @ -94,11 +122,27 @@ def run_test_in_subprocess(testname: str, ns: Namespace, tmp_dir: str, stdout_fh | ||||||
|     return subprocess.Popen(cmd, **kw) |     return subprocess.Popen(cmd, **kw) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def run_tests_worker(ns: Namespace, test_name: str) -> NoReturn: | def run_tests_worker(worker_json: str) -> NoReturn: | ||||||
|  |     worker_job = _parse_worker_args(worker_json) | ||||||
|  |     ns = worker_job.namespace | ||||||
|  |     test_name = worker_job.test_name | ||||||
|  |     rerun = worker_job.rerun | ||||||
|  |     match_tests = worker_job.match_tests | ||||||
|  | 
 | ||||||
|     setup_tests(ns) |     setup_tests(ns) | ||||||
| 
 | 
 | ||||||
|     result = runtest(ns, test_name) |     if rerun: | ||||||
|  |         if match_tests: | ||||||
|  |             matching = "matching: " + ", ".join(match_tests) | ||||||
|  |             print(f"Re-running {test_name} in verbose mode ({matching})", flush=True) | ||||||
|  |         else: | ||||||
|  |             print(f"Re-running {test_name} in verbose mode", flush=True) | ||||||
|  |         ns.verbose = True | ||||||
| 
 | 
 | ||||||
|  |     if match_tests is not None: | ||||||
|  |         ns.match_tests = match_tests | ||||||
|  | 
 | ||||||
|  |     result = runtest(ns, test_name) | ||||||
|     print()   # Force a newline (just in case) |     print()   # Force a newline (just in case) | ||||||
| 
 | 
 | ||||||
|     # Serialize TestResult as dict in JSON |     # Serialize TestResult as dict in JSON | ||||||
|  | @ -148,11 +192,13 @@ class TestWorkerProcess(threading.Thread): | ||||||
|     def __init__(self, worker_id: int, runner: "MultiprocessTestRunner") -> None: |     def __init__(self, worker_id: int, runner: "MultiprocessTestRunner") -> None: | ||||||
|         super().__init__() |         super().__init__() | ||||||
|         self.worker_id = worker_id |         self.worker_id = worker_id | ||||||
|  |         self.runtests = runner.runtests | ||||||
|         self.pending = runner.pending |         self.pending = runner.pending | ||||||
|         self.output = runner.output |         self.output = runner.output | ||||||
|         self.ns = runner.ns |         self.ns = runner.ns | ||||||
|         self.timeout = runner.worker_timeout |         self.timeout = runner.worker_timeout | ||||||
|         self.regrtest = runner.regrtest |         self.regrtest = runner.regrtest | ||||||
|  |         self.rerun = runner.rerun | ||||||
|         self.current_test_name = None |         self.current_test_name = None | ||||||
|         self.start_time = None |         self.start_time = None | ||||||
|         self._popen = None |         self._popen = None | ||||||
|  | @ -216,10 +262,11 @@ def mp_result_error( | ||||||
|     ) -> MultiprocessResult: |     ) -> MultiprocessResult: | ||||||
|         return MultiprocessResult(test_result, stdout, err_msg) |         return MultiprocessResult(test_result, stdout, err_msg) | ||||||
| 
 | 
 | ||||||
|     def _run_process(self, test_name: str, tmp_dir: str, stdout_fh: TextIO) -> int: |     def _run_process(self, worker_job, output_file: TextIO, | ||||||
|         self.current_test_name = test_name |                      tmp_dir: str | None = None) -> int: | ||||||
|  |         self.current_test_name = worker_job.test_name | ||||||
|         try: |         try: | ||||||
|             popen = run_test_in_subprocess(test_name, self.ns, tmp_dir, stdout_fh) |             popen = run_test_in_subprocess(worker_job, output_file, tmp_dir) | ||||||
| 
 | 
 | ||||||
|             self._killed = False |             self._killed = False | ||||||
|             self._popen = popen |             self._popen = popen | ||||||
|  | @ -277,9 +324,15 @@ def _runtest(self, test_name: str) -> MultiprocessResult: | ||||||
|         else: |         else: | ||||||
|             encoding = sys.stdout.encoding |             encoding = sys.stdout.encoding | ||||||
| 
 | 
 | ||||||
|  |         match_tests = self.runtests.get_match_tests(test_name) | ||||||
|  | 
 | ||||||
|         # gh-94026: Write stdout+stderr to a tempfile as workaround for |         # gh-94026: Write stdout+stderr to a tempfile as workaround for | ||||||
|         # non-blocking pipes on Emscripten with NodeJS. |         # non-blocking pipes on Emscripten with NodeJS. | ||||||
|         with tempfile.TemporaryFile('w+', encoding=encoding) as stdout_fh: |         with tempfile.TemporaryFile('w+', encoding=encoding) as stdout_file: | ||||||
|  |             worker_job = WorkerJob(test_name, | ||||||
|  |                                    namespace=self.ns, | ||||||
|  |                                    rerun=self.rerun, | ||||||
|  |                                    match_tests=match_tests) | ||||||
|             # gh-93353: Check for leaked temporary files in the parent process, |             # gh-93353: Check for leaked temporary files in the parent process, | ||||||
|             # since the deletion of temporary files can happen late during |             # since the deletion of temporary files can happen late during | ||||||
|             # Python finalization: too late for libregrtest. |             # Python finalization: too late for libregrtest. | ||||||
|  | @ -290,17 +343,17 @@ def _runtest(self, test_name: str) -> MultiprocessResult: | ||||||
|                 tmp_dir = tempfile.mkdtemp(prefix="test_python_") |                 tmp_dir = tempfile.mkdtemp(prefix="test_python_") | ||||||
|                 tmp_dir = os.path.abspath(tmp_dir) |                 tmp_dir = os.path.abspath(tmp_dir) | ||||||
|                 try: |                 try: | ||||||
|                     retcode = self._run_process(test_name, tmp_dir, stdout_fh) |                     retcode = self._run_process(worker_job, stdout_file, tmp_dir) | ||||||
|                 finally: |                 finally: | ||||||
|                     tmp_files = os.listdir(tmp_dir) |                     tmp_files = os.listdir(tmp_dir) | ||||||
|                     os_helper.rmtree(tmp_dir) |                     os_helper.rmtree(tmp_dir) | ||||||
|             else: |             else: | ||||||
|                 retcode = self._run_process(test_name, None, stdout_fh) |                 retcode = self._run_process(worker_job, stdout_file) | ||||||
|                 tmp_files = () |                 tmp_files = () | ||||||
|             stdout_fh.seek(0) |             stdout_file.seek(0) | ||||||
| 
 | 
 | ||||||
|             try: |             try: | ||||||
|                 stdout = stdout_fh.read().strip() |                 stdout = stdout_file.read().strip() | ||||||
|             except Exception as exc: |             except Exception as exc: | ||||||
|                 # gh-101634: Catch UnicodeDecodeError if stdout cannot be |                 # gh-101634: Catch UnicodeDecodeError if stdout cannot be | ||||||
|                 # decoded from encoding |                 # decoded from encoding | ||||||
|  | @ -342,6 +395,8 @@ def _runtest(self, test_name: str) -> MultiprocessResult: | ||||||
|         return MultiprocessResult(result, stdout) |         return MultiprocessResult(result, stdout) | ||||||
| 
 | 
 | ||||||
|     def run(self) -> None: |     def run(self) -> None: | ||||||
|  |         fail_fast = self.ns.failfast | ||||||
|  |         fail_env_changed = self.ns.fail_env_changed | ||||||
|         while not self._stopped: |         while not self._stopped: | ||||||
|             try: |             try: | ||||||
|                 try: |                 try: | ||||||
|  | @ -354,7 +409,7 @@ def run(self) -> None: | ||||||
|                 mp_result.result.duration = time.monotonic() - self.start_time |                 mp_result.result.duration = time.monotonic() - self.start_time | ||||||
|                 self.output.put((False, mp_result)) |                 self.output.put((False, mp_result)) | ||||||
| 
 | 
 | ||||||
|                 if must_stop(mp_result.result, self.ns): |                 if mp_result.result.must_stop(fail_fast, fail_env_changed): | ||||||
|                     break |                     break | ||||||
|             except ExitThread: |             except ExitThread: | ||||||
|                 break |                 break | ||||||
|  | @ -410,29 +465,36 @@ def get_running(workers: list[TestWorkerProcess]) -> list[TestWorkerProcess]: | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| class MultiprocessTestRunner: | class MultiprocessTestRunner: | ||||||
|     def __init__(self, regrtest: Regrtest) -> None: |     def __init__(self, regrtest: Regrtest, runtests: RunTests) -> None: | ||||||
|  |         ns = regrtest.ns | ||||||
|  |         timeout = ns.timeout | ||||||
|  | 
 | ||||||
|         self.regrtest = regrtest |         self.regrtest = regrtest | ||||||
|  |         self.runtests = runtests | ||||||
|  |         self.rerun = runtests.rerun | ||||||
|         self.log = self.regrtest.log |         self.log = self.regrtest.log | ||||||
|         self.ns = regrtest.ns |         self.ns = ns | ||||||
|         self.output: queue.Queue[QueueOutput] = queue.Queue() |         self.output: queue.Queue[QueueOutput] = queue.Queue() | ||||||
|         self.pending = MultiprocessIterator(self.regrtest.tests) |         tests_iter = runtests.iter_tests() | ||||||
|         if self.ns.timeout is not None: |         self.pending = MultiprocessIterator(tests_iter) | ||||||
|  |         if timeout is not None: | ||||||
|             # Rely on faulthandler to kill a worker process. This timouet is |             # Rely on faulthandler to kill a worker process. This timouet is | ||||||
|             # when faulthandler fails to kill a worker process. Give a maximum |             # when faulthandler fails to kill a worker process. Give a maximum | ||||||
|             # of 5 minutes to faulthandler to kill the worker. |             # of 5 minutes to faulthandler to kill the worker. | ||||||
|             self.worker_timeout = min(self.ns.timeout * 1.5, |             self.worker_timeout = min(timeout * 1.5, timeout + 5 * 60) | ||||||
|                                       self.ns.timeout + 5 * 60) |  | ||||||
|         else: |         else: | ||||||
|             self.worker_timeout = None |             self.worker_timeout = None | ||||||
|         self.workers = None |         self.workers = None | ||||||
| 
 | 
 | ||||||
|     def start_workers(self) -> None: |     def start_workers(self) -> None: | ||||||
|  |         use_mp = self.ns.use_mp | ||||||
|  |         timeout = self.ns.timeout | ||||||
|         self.workers = [TestWorkerProcess(index, self) |         self.workers = [TestWorkerProcess(index, self) | ||||||
|                         for index in range(1, self.ns.use_mp + 1)] |                         for index in range(1, use_mp + 1)] | ||||||
|         msg = f"Run tests in parallel using {len(self.workers)} child processes" |         msg = f"Run tests in parallel using {len(self.workers)} child processes" | ||||||
|         if self.ns.timeout: |         if timeout: | ||||||
|             msg += (" (timeout: %s, worker timeout: %s)" |             msg += (" (timeout: %s, worker timeout: %s)" | ||||||
|                     % (format_duration(self.ns.timeout), |                     % (format_duration(timeout), | ||||||
|                        format_duration(self.worker_timeout))) |                        format_duration(self.worker_timeout))) | ||||||
|         self.log(msg) |         self.log(msg) | ||||||
|         for worker in self.workers: |         for worker in self.workers: | ||||||
|  | @ -446,6 +508,7 @@ def stop_workers(self) -> None: | ||||||
|             worker.wait_stopped(start_time) |             worker.wait_stopped(start_time) | ||||||
| 
 | 
 | ||||||
|     def _get_result(self) -> QueueOutput | None: |     def _get_result(self) -> QueueOutput | None: | ||||||
|  |         pgo = self.ns.pgo | ||||||
|         use_faulthandler = (self.ns.timeout is not None) |         use_faulthandler = (self.ns.timeout is not None) | ||||||
|         timeout = PROGRESS_UPDATE |         timeout = PROGRESS_UPDATE | ||||||
| 
 | 
 | ||||||
|  | @ -464,7 +527,7 @@ def _get_result(self) -> QueueOutput | None: | ||||||
| 
 | 
 | ||||||
|             # display progress |             # display progress | ||||||
|             running = get_running(self.workers) |             running = get_running(self.workers) | ||||||
|             if running and not self.ns.pgo: |             if running and not pgo: | ||||||
|                 self.log('running: %s' % ', '.join(running)) |                 self.log('running: %s' % ', '.join(running)) | ||||||
| 
 | 
 | ||||||
|         # all worker threads are done: consume pending results |         # all worker threads are done: consume pending results | ||||||
|  | @ -475,42 +538,46 @@ def _get_result(self) -> QueueOutput | None: | ||||||
| 
 | 
 | ||||||
|     def display_result(self, mp_result: MultiprocessResult) -> None: |     def display_result(self, mp_result: MultiprocessResult) -> None: | ||||||
|         result = mp_result.result |         result = mp_result.result | ||||||
|  |         pgo = self.ns.pgo | ||||||
| 
 | 
 | ||||||
|         text = str(result) |         text = str(result) | ||||||
|         if mp_result.err_msg: |         if mp_result.err_msg: | ||||||
|             # MULTIPROCESSING_ERROR |             # MULTIPROCESSING_ERROR | ||||||
|             text += ' (%s)' % mp_result.err_msg |             text += ' (%s)' % mp_result.err_msg | ||||||
|         elif (result.duration >= PROGRESS_MIN_TIME and not self.ns.pgo): |         elif (result.duration >= PROGRESS_MIN_TIME and not pgo): | ||||||
|             text += ' (%s)' % format_duration(result.duration) |             text += ' (%s)' % format_duration(result.duration) | ||||||
|         running = get_running(self.workers) |         running = get_running(self.workers) | ||||||
|         if running and not self.ns.pgo: |         if running and not pgo: | ||||||
|             text += ' -- running: %s' % ', '.join(running) |             text += ' -- running: %s' % ', '.join(running) | ||||||
|         self.regrtest.display_progress(self.test_index, text) |         self.regrtest.display_progress(self.test_index, text) | ||||||
| 
 | 
 | ||||||
|     def _process_result(self, item: QueueOutput) -> bool: |     def _process_result(self, item: QueueOutput) -> bool: | ||||||
|         """Returns True if test runner must stop.""" |         """Returns True if test runner must stop.""" | ||||||
|  |         rerun = self.runtests.rerun | ||||||
|         if item[0]: |         if item[0]: | ||||||
|             # Thread got an exception |             # Thread got an exception | ||||||
|             format_exc = item[1] |             format_exc = item[1] | ||||||
|             print_warning(f"regrtest worker thread failed: {format_exc}") |             print_warning(f"regrtest worker thread failed: {format_exc}") | ||||||
|             result = TestResult("<regrtest worker>", state=State.MULTIPROCESSING_ERROR) |             result = TestResult("<regrtest worker>", state=State.MULTIPROCESSING_ERROR) | ||||||
|             self.regrtest.accumulate_result(result) |             self.regrtest.accumulate_result(result, rerun=rerun) | ||||||
|             return True |             return result | ||||||
| 
 | 
 | ||||||
|         self.test_index += 1 |         self.test_index += 1 | ||||||
|         mp_result = item[1] |         mp_result = item[1] | ||||||
|         self.regrtest.accumulate_result(mp_result.result) |         result = mp_result.result | ||||||
|  |         self.regrtest.accumulate_result(result, rerun=rerun) | ||||||
|         self.display_result(mp_result) |         self.display_result(mp_result) | ||||||
| 
 | 
 | ||||||
|         if mp_result.worker_stdout: |         if mp_result.worker_stdout: | ||||||
|             print(mp_result.worker_stdout, flush=True) |             print(mp_result.worker_stdout, flush=True) | ||||||
| 
 | 
 | ||||||
|         if must_stop(mp_result.result, self.ns): |         return result | ||||||
|             return True |  | ||||||
| 
 |  | ||||||
|         return False |  | ||||||
| 
 | 
 | ||||||
|     def run_tests(self) -> None: |     def run_tests(self) -> None: | ||||||
|  |         fail_fast = self.ns.failfast | ||||||
|  |         fail_env_changed = self.ns.fail_env_changed | ||||||
|  |         timeout = self.ns.timeout | ||||||
|  | 
 | ||||||
|         self.start_workers() |         self.start_workers() | ||||||
| 
 | 
 | ||||||
|         self.test_index = 0 |         self.test_index = 0 | ||||||
|  | @ -520,14 +587,14 @@ def run_tests(self) -> None: | ||||||
|                 if item is None: |                 if item is None: | ||||||
|                     break |                     break | ||||||
| 
 | 
 | ||||||
|                 stop = self._process_result(item) |                 result = self._process_result(item) | ||||||
|                 if stop: |                 if result.must_stop(fail_fast, fail_env_changed): | ||||||
|                     break |                     break | ||||||
|         except KeyboardInterrupt: |         except KeyboardInterrupt: | ||||||
|             print() |             print() | ||||||
|             self.regrtest.interrupted = True |             self.regrtest.interrupted = True | ||||||
|         finally: |         finally: | ||||||
|             if self.ns.timeout is not None: |             if timeout is not None: | ||||||
|                 faulthandler.cancel_dump_traceback_later() |                 faulthandler.cancel_dump_traceback_later() | ||||||
| 
 | 
 | ||||||
|             # Always ensure that all worker processes are no longer |             # Always ensure that all worker processes are no longer | ||||||
|  | @ -536,8 +603,8 @@ def run_tests(self) -> None: | ||||||
|             self.stop_workers() |             self.stop_workers() | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def run_tests_multiprocess(regrtest: Regrtest) -> None: | def run_tests_multiprocess(regrtest: Regrtest, runtests: RunTests) -> None: | ||||||
|     MultiprocessTestRunner(regrtest).run_tests() |     MultiprocessTestRunner(regrtest, runtests).run_tests() | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| class EncodeTestResult(json.JSONEncoder): | class EncodeTestResult(json.JSONEncoder): | ||||||
|  | @ -552,7 +619,7 @@ def default(self, o: Any) -> dict[str, Any]: | ||||||
|         return super().default(o) |         return super().default(o) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def decode_test_result(d: dict[str, Any]) -> TestResult | TestStats | dict[str, Any]: | def decode_test_result(d: dict[str, Any]) -> TestResult | dict[str, Any]: | ||||||
|     """Decode a TestResult (sub)class object from a JSON dict.""" |     """Decode a TestResult (sub)class object from a JSON dict.""" | ||||||
| 
 | 
 | ||||||
|     if "__test_result__" not in d: |     if "__test_result__" not in d: | ||||||
|  |  | ||||||
|  | @ -31,7 +31,7 @@ def format_duration(seconds): | ||||||
|     return ' '.join(parts) |     return ' '.join(parts) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def removepy(names): | def strip_py_suffix(names: list[str]): | ||||||
|     if not names: |     if not names: | ||||||
|         return |         return | ||||||
|     for idx, name in enumerate(names): |     for idx, name in enumerate(names): | ||||||
|  |  | ||||||
|  | @ -1189,7 +1189,6 @@ def _is_full_match_test(pattern): | ||||||
| def set_match_tests(accept_patterns=None, ignore_patterns=None): | def set_match_tests(accept_patterns=None, ignore_patterns=None): | ||||||
|     global _match_test_func, _accept_test_patterns, _ignore_test_patterns |     global _match_test_func, _accept_test_patterns, _ignore_test_patterns | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
|     if accept_patterns is None: |     if accept_patterns is None: | ||||||
|         accept_patterns = () |         accept_patterns = () | ||||||
|     if ignore_patterns is None: |     if ignore_patterns is None: | ||||||
|  |  | ||||||
|  | @ -8,6 +8,7 @@ | ||||||
| import time | import time | ||||||
| import traceback | import traceback | ||||||
| import unittest | import unittest | ||||||
|  | from test import support | ||||||
| 
 | 
 | ||||||
| class RegressionTestResult(unittest.TextTestResult): | class RegressionTestResult(unittest.TextTestResult): | ||||||
|     USE_XML = False |     USE_XML = False | ||||||
|  | @ -112,6 +113,8 @@ def addExpectedFailure(self, test, err): | ||||||
|     def addFailure(self, test, err): |     def addFailure(self, test, err): | ||||||
|         self._add_result(test, True, failure=self.__makeErrorDict(*err)) |         self._add_result(test, True, failure=self.__makeErrorDict(*err)) | ||||||
|         super().addFailure(test, err) |         super().addFailure(test, err) | ||||||
|  |         if support.failfast: | ||||||
|  |             self.stop() | ||||||
| 
 | 
 | ||||||
|     def addSkip(self, test, reason): |     def addSkip(self, test, reason): | ||||||
|         self._add_result(test, skipped=reason) |         self._add_result(test, skipped=reason) | ||||||
|  |  | ||||||
|  | @ -5,6 +5,7 @@ | ||||||
| """ | """ | ||||||
| 
 | 
 | ||||||
| import contextlib | import contextlib | ||||||
|  | import dataclasses | ||||||
| import glob | import glob | ||||||
| import io | import io | ||||||
| import locale | import locale | ||||||
|  | @ -21,6 +22,7 @@ | ||||||
| from test import support | from test import support | ||||||
| from test.support import os_helper, TestStats | from test.support import os_helper, TestStats | ||||||
| from test.libregrtest import utils, setup | from test.libregrtest import utils, setup | ||||||
|  | from test.libregrtest.runtest import normalize_test_name | ||||||
| 
 | 
 | ||||||
| if not support.has_subprocess_support: | if not support.has_subprocess_support: | ||||||
|     raise unittest.SkipTest("test module requires subprocess") |     raise unittest.SkipTest("test module requires subprocess") | ||||||
|  | @ -96,11 +98,11 @@ def test_verbose(self): | ||||||
|         ns = libregrtest._parse_args([]) |         ns = libregrtest._parse_args([]) | ||||||
|         self.assertEqual(ns.verbose, 0) |         self.assertEqual(ns.verbose, 0) | ||||||
| 
 | 
 | ||||||
|     def test_verbose2(self): |     def test_rerun(self): | ||||||
|         for opt in '-w', '--verbose2': |         for opt in '-w', '--rerun', '--verbose2': | ||||||
|             with self.subTest(opt=opt): |             with self.subTest(opt=opt): | ||||||
|                 ns = libregrtest._parse_args([opt]) |                 ns = libregrtest._parse_args([opt]) | ||||||
|                 self.assertTrue(ns.verbose2) |                 self.assertTrue(ns.rerun) | ||||||
| 
 | 
 | ||||||
|     def test_verbose3(self): |     def test_verbose3(self): | ||||||
|         for opt in '-W', '--verbose3': |         for opt in '-W', '--verbose3': | ||||||
|  | @ -362,6 +364,13 @@ def test_unknown_option(self): | ||||||
|                         'unrecognized arguments: --unknown-option') |                         'unrecognized arguments: --unknown-option') | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | @dataclasses.dataclass(slots=True) | ||||||
|  | class Rerun: | ||||||
|  |     name: str | ||||||
|  |     match: str | None | ||||||
|  |     success: bool | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
| class BaseTestCase(unittest.TestCase): | class BaseTestCase(unittest.TestCase): | ||||||
|     TEST_UNIQUE_ID = 1 |     TEST_UNIQUE_ID = 1 | ||||||
|     TESTNAME_PREFIX = 'test_regrtest_' |     TESTNAME_PREFIX = 'test_regrtest_' | ||||||
|  | @ -423,11 +432,11 @@ def parse_executed_tests(self, output): | ||||||
| 
 | 
 | ||||||
|     def check_executed_tests(self, output, tests, skipped=(), failed=(), |     def check_executed_tests(self, output, tests, skipped=(), failed=(), | ||||||
|                              env_changed=(), omitted=(), |                              env_changed=(), omitted=(), | ||||||
|                              rerun={}, run_no_tests=(), |                              rerun=None, run_no_tests=(), | ||||||
|                              resource_denied=(), |                              resource_denied=(), | ||||||
|                              randomize=False, interrupted=False, |                              randomize=False, interrupted=False, | ||||||
|                              fail_env_changed=False, |                              fail_env_changed=False, | ||||||
|                              *, stats): |                              *, stats, forever=False, filtered=False): | ||||||
|         if isinstance(tests, str): |         if isinstance(tests, str): | ||||||
|             tests = [tests] |             tests = [tests] | ||||||
|         if isinstance(skipped, str): |         if isinstance(skipped, str): | ||||||
|  | @ -445,11 +454,20 @@ def check_executed_tests(self, output, tests, skipped=(), failed=(), | ||||||
|         if isinstance(stats, int): |         if isinstance(stats, int): | ||||||
|             stats = TestStats(stats) |             stats = TestStats(stats) | ||||||
| 
 | 
 | ||||||
|  |         rerun_failed = [] | ||||||
|  |         if rerun is not None: | ||||||
|  |             failed = [rerun.name] | ||||||
|  |             if not rerun.success: | ||||||
|  |                 rerun_failed.append(rerun.name) | ||||||
|  | 
 | ||||||
|         executed = self.parse_executed_tests(output) |         executed = self.parse_executed_tests(output) | ||||||
|  |         total_tests = list(tests) | ||||||
|  |         if rerun is not None: | ||||||
|  |             total_tests.append(rerun.name) | ||||||
|         if randomize: |         if randomize: | ||||||
|             self.assertEqual(set(executed), set(tests), output) |             self.assertEqual(set(executed), set(total_tests), output) | ||||||
|         else: |         else: | ||||||
|             self.assertEqual(executed, tests, output) |             self.assertEqual(executed, total_tests, output) | ||||||
| 
 | 
 | ||||||
|         def plural(count): |         def plural(count): | ||||||
|             return 's' if count != 1 else '' |             return 's' if count != 1 else '' | ||||||
|  | @ -465,6 +483,10 @@ def list_regex(line_format, tests): | ||||||
|             regex = list_regex('%s test%s skipped', skipped) |             regex = list_regex('%s test%s skipped', skipped) | ||||||
|             self.check_line(output, regex) |             self.check_line(output, regex) | ||||||
| 
 | 
 | ||||||
|  |         if resource_denied: | ||||||
|  |             regex = list_regex(r'%s test%s skipped \(resource denied\)', resource_denied) | ||||||
|  |             self.check_line(output, regex) | ||||||
|  | 
 | ||||||
|         if failed: |         if failed: | ||||||
|             regex = list_regex('%s test%s failed', failed) |             regex = list_regex('%s test%s failed', failed) | ||||||
|             self.check_line(output, regex) |             self.check_line(output, regex) | ||||||
|  | @ -478,32 +500,36 @@ def list_regex(line_format, tests): | ||||||
|             regex = list_regex('%s test%s omitted', omitted) |             regex = list_regex('%s test%s omitted', omitted) | ||||||
|             self.check_line(output, regex) |             self.check_line(output, regex) | ||||||
| 
 | 
 | ||||||
|         if rerun: |         if rerun is not None: | ||||||
|             regex = list_regex('%s re-run test%s', rerun.keys()) |             regex = list_regex('%s re-run test%s', [rerun.name]) | ||||||
|             self.check_line(output, regex) |             self.check_line(output, regex) | ||||||
|             regex = LOG_PREFIX + r"Re-running failed tests in verbose mode" |             regex = LOG_PREFIX + fr"Re-running 1 failed tests in verbose mode" | ||||||
|             self.check_line(output, regex) |             self.check_line(output, regex) | ||||||
|             for name, match in rerun.items(): |             regex = fr"Re-running {rerun.name} in verbose mode" | ||||||
|                 regex = LOG_PREFIX + f"Re-running {name} in verbose mode \\(matching: {match}\\)" |             if rerun.match: | ||||||
|  |                 regex = fr"{regex} \(matching: {rerun.match}\)" | ||||||
|             self.check_line(output, regex) |             self.check_line(output, regex) | ||||||
| 
 | 
 | ||||||
|         if run_no_tests: |         if run_no_tests: | ||||||
|             regex = list_regex('%s test%s run no tests', run_no_tests) |             regex = list_regex('%s test%s run no tests', run_no_tests) | ||||||
|             self.check_line(output, regex) |             self.check_line(output, regex) | ||||||
| 
 | 
 | ||||||
|         good = (len(tests) - len(skipped) - len(failed) |         good = (len(tests) - len(skipped) - len(resource_denied) - len(failed) | ||||||
|                 - len(omitted) - len(env_changed) - len(run_no_tests)) |                 - len(omitted) - len(env_changed) - len(run_no_tests)) | ||||||
|         if good: |         if good: | ||||||
|             regex = r'%s test%s OK\.$' % (good, plural(good)) |             regex = r'%s test%s OK\.' % (good, plural(good)) | ||||||
|             if not skipped and not failed and good > 1: |             if not skipped and not failed and (rerun is None or rerun.success) and good > 1: | ||||||
|                 regex = 'All %s' % regex |                 regex = 'All %s' % regex | ||||||
|             self.check_line(output, regex) |             self.check_line(output, regex, full=True) | ||||||
| 
 | 
 | ||||||
|         if interrupted: |         if interrupted: | ||||||
|             self.check_line(output, 'Test suite interrupted by signal SIGINT.') |             self.check_line(output, 'Test suite interrupted by signal SIGINT.') | ||||||
| 
 | 
 | ||||||
|         # Total tests |         # Total tests | ||||||
|         parts = [f'run={stats.tests_run:,}'] |         text = f'run={stats.tests_run:,}' | ||||||
|  |         if filtered: | ||||||
|  |             text = fr'{text} \(filtered\)' | ||||||
|  |         parts = [text] | ||||||
|         if stats.failures: |         if stats.failures: | ||||||
|             parts.append(f'failures={stats.failures:,}') |             parts.append(f'failures={stats.failures:,}') | ||||||
|         if stats.skipped: |         if stats.skipped: | ||||||
|  | @ -512,39 +538,52 @@ def list_regex(line_format, tests): | ||||||
|         self.check_line(output, line, full=True) |         self.check_line(output, line, full=True) | ||||||
| 
 | 
 | ||||||
|         # Total test files |         # Total test files | ||||||
|         report = [f'success={good}'] |         run = len(total_tests) - len(resource_denied) | ||||||
|         if failed: |         if rerun is not None: | ||||||
|             report.append(f'failed={len(failed)}') |             total_failed = len(rerun_failed) | ||||||
|         if env_changed: |             total_rerun = 1 | ||||||
|             report.append(f'env_changed={len(env_changed)}') |         else: | ||||||
|         if skipped: |             total_failed = len(failed) | ||||||
|             report.append(f'skipped={len(skipped)}') |             total_rerun = 0 | ||||||
|         if resource_denied: |         if interrupted: | ||||||
|             report.append(f'resource_denied={len(resource_denied)}') |             run = 0 | ||||||
|         if rerun: |         text = f'run={run}' | ||||||
|             report.append(f'rerun={len(rerun)}') |         if not forever: | ||||||
|         if run_no_tests: |             text = f'{text}/{len(tests)}' | ||||||
|             report.append(f'run_no_tests={len(run_no_tests)}') |         if filtered: | ||||||
|  |             text = fr'{text} \(filtered\)' | ||||||
|  |         report = [text] | ||||||
|  |         for name, ntest in ( | ||||||
|  |             ('failed', total_failed), | ||||||
|  |             ('env_changed', len(env_changed)), | ||||||
|  |             ('skipped', len(skipped)), | ||||||
|  |             ('resource_denied', len(resource_denied)), | ||||||
|  |             ('rerun', total_rerun), | ||||||
|  |             ('run_no_tests', len(run_no_tests)), | ||||||
|  |         ): | ||||||
|  |             if ntest: | ||||||
|  |                 report.append(f'{name}={ntest}') | ||||||
|         line = fr'Total test files: {" ".join(report)}' |         line = fr'Total test files: {" ".join(report)}' | ||||||
|         self.check_line(output, line, full=True) |         self.check_line(output, line, full=True) | ||||||
| 
 | 
 | ||||||
|         # Result |         # Result | ||||||
|         result = [] |         state = [] | ||||||
|         if failed: |         if failed: | ||||||
|             result.append('FAILURE') |             state.append('FAILURE') | ||||||
|         elif fail_env_changed and env_changed: |         elif fail_env_changed and env_changed: | ||||||
|             result.append('ENV CHANGED') |             state.append('ENV CHANGED') | ||||||
|         if interrupted: |         if interrupted: | ||||||
|             result.append('INTERRUPTED') |             state.append('INTERRUPTED') | ||||||
|         if not any((good, result, failed, interrupted, skipped, |         if not any((good, failed, interrupted, skipped, | ||||||
|                     env_changed, fail_env_changed)): |                     env_changed, fail_env_changed)): | ||||||
|             result.append("NO TESTS RAN") |             state.append("NO TESTS RAN") | ||||||
|         elif not result: |         elif not state: | ||||||
|             result.append('SUCCESS') |             state.append('SUCCESS') | ||||||
|         result = ', '.join(result) |         state = ', '.join(state) | ||||||
|         if rerun: |         if rerun is not None: | ||||||
|             result = 'FAILURE then %s' % result |             new_state = 'SUCCESS' if rerun.success else 'FAILURE' | ||||||
|         self.check_line(output, f'Result: {result}', full=True) |             state = 'FAILURE then ' + new_state | ||||||
|  |         self.check_line(output, f'Result: {state}', full=True) | ||||||
| 
 | 
 | ||||||
|     def parse_random_seed(self, output): |     def parse_random_seed(self, output): | ||||||
|         match = self.regex_search(r'Using random seed ([0-9]+)', output) |         match = self.regex_search(r'Using random seed ([0-9]+)', output) | ||||||
|  | @ -563,13 +602,13 @@ def run_command(self, args, input=None, exitcode=0, **kw): | ||||||
|                               stdout=subprocess.PIPE, |                               stdout=subprocess.PIPE, | ||||||
|                               **kw) |                               **kw) | ||||||
|         if proc.returncode != exitcode: |         if proc.returncode != exitcode: | ||||||
|             msg = ("Command %s failed with exit code %s\n" |             msg = ("Command %s failed with exit code %s, but exit code %s expected!\n" | ||||||
|                    "\n" |                    "\n" | ||||||
|                    "stdout:\n" |                    "stdout:\n" | ||||||
|                    "---\n" |                    "---\n" | ||||||
|                    "%s\n" |                    "%s\n" | ||||||
|                    "---\n" |                    "---\n" | ||||||
|                    % (str(args), proc.returncode, proc.stdout)) |                    % (str(args), proc.returncode, exitcode, proc.stdout)) | ||||||
|             if proc.stderr: |             if proc.stderr: | ||||||
|                 msg += ("\n" |                 msg += ("\n" | ||||||
|                         "stderr:\n" |                         "stderr:\n" | ||||||
|  | @ -738,6 +777,40 @@ def run_tests(self, *testargs, **kw): | ||||||
|         cmdargs = ['-m', 'test', '--testdir=%s' % self.tmptestdir, *testargs] |         cmdargs = ['-m', 'test', '--testdir=%s' % self.tmptestdir, *testargs] | ||||||
|         return self.run_python(cmdargs, **kw) |         return self.run_python(cmdargs, **kw) | ||||||
| 
 | 
 | ||||||
|  |     def test_success(self): | ||||||
|  |         code = textwrap.dedent(""" | ||||||
|  |             import unittest | ||||||
|  | 
 | ||||||
|  |             class PassingTests(unittest.TestCase): | ||||||
|  |                 def test_test1(self): | ||||||
|  |                     pass | ||||||
|  | 
 | ||||||
|  |                 def test_test2(self): | ||||||
|  |                     pass | ||||||
|  | 
 | ||||||
|  |                 def test_test3(self): | ||||||
|  |                     pass | ||||||
|  |         """) | ||||||
|  |         tests = [self.create_test(f'ok{i}', code=code) for i in range(1, 6)] | ||||||
|  | 
 | ||||||
|  |         output = self.run_tests(*tests) | ||||||
|  |         self.check_executed_tests(output, tests, | ||||||
|  |                                   stats=3 * len(tests)) | ||||||
|  | 
 | ||||||
|  |     def test_skip(self): | ||||||
|  |         code = textwrap.dedent(""" | ||||||
|  |             import unittest | ||||||
|  |             raise unittest.SkipTest("nope") | ||||||
|  |         """) | ||||||
|  |         test_ok = self.create_test('ok') | ||||||
|  |         test_skip = self.create_test('skip', code=code) | ||||||
|  |         tests = [test_ok, test_skip] | ||||||
|  | 
 | ||||||
|  |         output = self.run_tests(*tests) | ||||||
|  |         self.check_executed_tests(output, tests, | ||||||
|  |                                   skipped=[test_skip], | ||||||
|  |                                   stats=1) | ||||||
|  | 
 | ||||||
|     def test_failing_test(self): |     def test_failing_test(self): | ||||||
|         # test a failing test |         # test a failing test | ||||||
|         code = textwrap.dedent(""" |         code = textwrap.dedent(""" | ||||||
|  | @ -777,14 +850,12 @@ def test_pass(self): | ||||||
|         # -u audio: 1 resource enabled |         # -u audio: 1 resource enabled | ||||||
|         output = self.run_tests('-uaudio', *test_names) |         output = self.run_tests('-uaudio', *test_names) | ||||||
|         self.check_executed_tests(output, test_names, |         self.check_executed_tests(output, test_names, | ||||||
|                                   skipped=tests['network'], |  | ||||||
|                                   resource_denied=tests['network'], |                                   resource_denied=tests['network'], | ||||||
|                                   stats=1) |                                   stats=1) | ||||||
| 
 | 
 | ||||||
|         # no option: 0 resources enabled |         # no option: 0 resources enabled | ||||||
|         output = self.run_tests(*test_names) |         output = self.run_tests(*test_names, exitcode=EXITCODE_NO_TESTS_RAN) | ||||||
|         self.check_executed_tests(output, test_names, |         self.check_executed_tests(output, test_names, | ||||||
|                                   skipped=test_names, |  | ||||||
|                                   resource_denied=test_names, |                                   resource_denied=test_names, | ||||||
|                                   stats=0) |                                   stats=0) | ||||||
| 
 | 
 | ||||||
|  | @ -930,9 +1001,21 @@ def test_run(self): | ||||||
|                         builtins.__dict__['RUN'] = 1 |                         builtins.__dict__['RUN'] = 1 | ||||||
|         """) |         """) | ||||||
|         test = self.create_test('forever', code=code) |         test = self.create_test('forever', code=code) | ||||||
|  | 
 | ||||||
|  |         # --forever | ||||||
|         output = self.run_tests('--forever', test, exitcode=EXITCODE_BAD_TEST) |         output = self.run_tests('--forever', test, exitcode=EXITCODE_BAD_TEST) | ||||||
|         self.check_executed_tests(output, [test]*3, failed=test, |         self.check_executed_tests(output, [test]*3, failed=test, | ||||||
|                                   stats=TestStats(1, 1)) |                                   stats=TestStats(3, 1), | ||||||
|  |                                   forever=True) | ||||||
|  | 
 | ||||||
|  |         # --forever --rerun | ||||||
|  |         output = self.run_tests('--forever', '--rerun', test, exitcode=0) | ||||||
|  |         self.check_executed_tests(output, [test]*3, | ||||||
|  |                                   rerun=Rerun(test, | ||||||
|  |                                               match='test_run', | ||||||
|  |                                               success=True), | ||||||
|  |                                   stats=TestStats(4, 1), | ||||||
|  |                                   forever=True) | ||||||
| 
 | 
 | ||||||
|     def check_leak(self, code, what): |     def check_leak(self, code, what): | ||||||
|         test = self.create_test('huntrleaks', code=code) |         test = self.create_test('huntrleaks', code=code) | ||||||
|  | @ -1143,33 +1226,55 @@ def test_fail_always(self): | ||||||
|         """) |         """) | ||||||
|         testname = self.create_test(code=code) |         testname = self.create_test(code=code) | ||||||
| 
 | 
 | ||||||
|         output = self.run_tests("-w", testname, exitcode=EXITCODE_BAD_TEST) |         output = self.run_tests("--rerun", testname, exitcode=EXITCODE_BAD_TEST) | ||||||
|         self.check_executed_tests(output, [testname], |         self.check_executed_tests(output, [testname], | ||||||
|                                   failed=testname, |                                   rerun=Rerun(testname, | ||||||
|                                   rerun={testname: "test_fail_always"}, |                                               "test_fail_always", | ||||||
|                                   stats=TestStats(1, 1)) |                                               success=False), | ||||||
|  |                                   stats=TestStats(3, 2)) | ||||||
| 
 | 
 | ||||||
|     def test_rerun_success(self): |     def test_rerun_success(self): | ||||||
|         # FAILURE then SUCCESS |         # FAILURE then SUCCESS | ||||||
|         code = textwrap.dedent(""" |         marker_filename = os.path.abspath("regrtest_marker_filename") | ||||||
|             import builtins |         self.addCleanup(os_helper.unlink, marker_filename) | ||||||
|  |         self.assertFalse(os.path.exists(marker_filename)) | ||||||
|  | 
 | ||||||
|  |         code = textwrap.dedent(f""" | ||||||
|  |             import os.path | ||||||
|             import unittest |             import unittest | ||||||
| 
 | 
 | ||||||
|  |             marker_filename = {marker_filename!r} | ||||||
|  | 
 | ||||||
|             class Tests(unittest.TestCase): |             class Tests(unittest.TestCase): | ||||||
|                 def test_succeed(self): |                 def test_succeed(self): | ||||||
|                     return |                     return | ||||||
| 
 | 
 | ||||||
|                 def test_fail_once(self): |                 def test_fail_once(self): | ||||||
|                     if not hasattr(builtins, '_test_failed'): |                     if not os.path.exists(marker_filename): | ||||||
|                         builtins._test_failed = True |                         open(marker_filename, "w").close() | ||||||
|                         self.fail("bug") |                         self.fail("bug") | ||||||
|         """) |         """) | ||||||
|         testname = self.create_test(code=code) |         testname = self.create_test(code=code) | ||||||
| 
 | 
 | ||||||
|         output = self.run_tests("-w", testname, exitcode=0) |         # FAILURE then SUCCESS => exit code 0 | ||||||
|  |         output = self.run_tests("--rerun", testname, exitcode=0) | ||||||
|         self.check_executed_tests(output, [testname], |         self.check_executed_tests(output, [testname], | ||||||
|                                   rerun={testname: "test_fail_once"}, |                                   rerun=Rerun(testname, | ||||||
|                                   stats=1) |                                               match="test_fail_once", | ||||||
|  |                                               success=True), | ||||||
|  |                                   stats=TestStats(3, 1)) | ||||||
|  |         os_helper.unlink(marker_filename) | ||||||
|  | 
 | ||||||
|  |         # with --fail-rerun, exit code EXITCODE_BAD_TEST | ||||||
|  |         # on "FAILURE then SUCCESS" state. | ||||||
|  |         output = self.run_tests("--rerun", "--fail-rerun", testname, | ||||||
|  |                                 exitcode=EXITCODE_BAD_TEST) | ||||||
|  |         self.check_executed_tests(output, [testname], | ||||||
|  |                                   rerun=Rerun(testname, | ||||||
|  |                                               match="test_fail_once", | ||||||
|  |                                               success=True), | ||||||
|  |                                   stats=TestStats(3, 1)) | ||||||
|  |         os_helper.unlink(marker_filename) | ||||||
| 
 | 
 | ||||||
|     def test_rerun_setup_class_hook_failure(self): |     def test_rerun_setup_class_hook_failure(self): | ||||||
|         # FAILURE then FAILURE |         # FAILURE then FAILURE | ||||||
|  | @ -1186,10 +1291,12 @@ def test_success(self): | ||||||
|         """) |         """) | ||||||
|         testname = self.create_test(code=code) |         testname = self.create_test(code=code) | ||||||
| 
 | 
 | ||||||
|         output = self.run_tests("-w", testname, exitcode=EXITCODE_BAD_TEST) |         output = self.run_tests("--rerun", testname, exitcode=EXITCODE_BAD_TEST) | ||||||
|         self.check_executed_tests(output, testname, |         self.check_executed_tests(output, testname, | ||||||
|                                   failed=[testname], |                                   failed=[testname], | ||||||
|                                   rerun={testname: "ExampleTests"}, |                                   rerun=Rerun(testname, | ||||||
|  |                                               match="ExampleTests", | ||||||
|  |                                               success=False), | ||||||
|                                   stats=0) |                                   stats=0) | ||||||
| 
 | 
 | ||||||
|     def test_rerun_teardown_class_hook_failure(self): |     def test_rerun_teardown_class_hook_failure(self): | ||||||
|  | @ -1207,11 +1314,13 @@ def test_success(self): | ||||||
|         """) |         """) | ||||||
|         testname = self.create_test(code=code) |         testname = self.create_test(code=code) | ||||||
| 
 | 
 | ||||||
|         output = self.run_tests("-w", testname, exitcode=EXITCODE_BAD_TEST) |         output = self.run_tests("--rerun", testname, exitcode=EXITCODE_BAD_TEST) | ||||||
|         self.check_executed_tests(output, testname, |         self.check_executed_tests(output, testname, | ||||||
|                                   failed=[testname], |                                   failed=[testname], | ||||||
|                                   rerun={testname: "ExampleTests"}, |                                   rerun=Rerun(testname, | ||||||
|                                   stats=1) |                                               match="ExampleTests", | ||||||
|  |                                               success=False), | ||||||
|  |                                   stats=2) | ||||||
| 
 | 
 | ||||||
|     def test_rerun_setup_module_hook_failure(self): |     def test_rerun_setup_module_hook_failure(self): | ||||||
|         # FAILURE then FAILURE |         # FAILURE then FAILURE | ||||||
|  | @ -1227,10 +1336,12 @@ def test_success(self): | ||||||
|         """) |         """) | ||||||
|         testname = self.create_test(code=code) |         testname = self.create_test(code=code) | ||||||
| 
 | 
 | ||||||
|         output = self.run_tests("-w", testname, exitcode=EXITCODE_BAD_TEST) |         output = self.run_tests("--rerun", testname, exitcode=EXITCODE_BAD_TEST) | ||||||
|         self.check_executed_tests(output, testname, |         self.check_executed_tests(output, testname, | ||||||
|                                   failed=[testname], |                                   failed=[testname], | ||||||
|                                   rerun={testname: testname}, |                                   rerun=Rerun(testname, | ||||||
|  |                                               match=None, | ||||||
|  |                                               success=False), | ||||||
|                                   stats=0) |                                   stats=0) | ||||||
| 
 | 
 | ||||||
|     def test_rerun_teardown_module_hook_failure(self): |     def test_rerun_teardown_module_hook_failure(self): | ||||||
|  | @ -1247,11 +1358,13 @@ def test_success(self): | ||||||
|         """) |         """) | ||||||
|         testname = self.create_test(code=code) |         testname = self.create_test(code=code) | ||||||
| 
 | 
 | ||||||
|         output = self.run_tests("-w", testname, exitcode=EXITCODE_BAD_TEST) |         output = self.run_tests("--rerun", testname, exitcode=EXITCODE_BAD_TEST) | ||||||
|         self.check_executed_tests(output, testname, |         self.check_executed_tests(output, [testname], | ||||||
|                                   failed=[testname], |                                   failed=[testname], | ||||||
|                                   rerun={testname: testname}, |                                   rerun=Rerun(testname, | ||||||
|                                   stats=1) |                                               match=None, | ||||||
|  |                                               success=False), | ||||||
|  |                                   stats=2) | ||||||
| 
 | 
 | ||||||
|     def test_rerun_setup_hook_failure(self): |     def test_rerun_setup_hook_failure(self): | ||||||
|         # FAILURE then FAILURE |         # FAILURE then FAILURE | ||||||
|  | @ -1267,11 +1380,13 @@ def test_success(self): | ||||||
|         """) |         """) | ||||||
|         testname = self.create_test(code=code) |         testname = self.create_test(code=code) | ||||||
| 
 | 
 | ||||||
|         output = self.run_tests("-w", testname, exitcode=EXITCODE_BAD_TEST) |         output = self.run_tests("--rerun", testname, exitcode=EXITCODE_BAD_TEST) | ||||||
|         self.check_executed_tests(output, testname, |         self.check_executed_tests(output, testname, | ||||||
|                                   failed=[testname], |                                   failed=[testname], | ||||||
|                                   rerun={testname: "test_success"}, |                                   rerun=Rerun(testname, | ||||||
|                                   stats=1) |                                               match="test_success", | ||||||
|  |                                               success=False), | ||||||
|  |                                   stats=2) | ||||||
| 
 | 
 | ||||||
|     def test_rerun_teardown_hook_failure(self): |     def test_rerun_teardown_hook_failure(self): | ||||||
|         # FAILURE then FAILURE |         # FAILURE then FAILURE | ||||||
|  | @ -1287,11 +1402,13 @@ def test_success(self): | ||||||
|         """) |         """) | ||||||
|         testname = self.create_test(code=code) |         testname = self.create_test(code=code) | ||||||
| 
 | 
 | ||||||
|         output = self.run_tests("-w", testname, exitcode=EXITCODE_BAD_TEST) |         output = self.run_tests("--rerun", testname, exitcode=EXITCODE_BAD_TEST) | ||||||
|         self.check_executed_tests(output, testname, |         self.check_executed_tests(output, testname, | ||||||
|                                   failed=[testname], |                                   failed=[testname], | ||||||
|                                   rerun={testname: "test_success"}, |                                   rerun=Rerun(testname, | ||||||
|                                   stats=1) |                                               match="test_success", | ||||||
|  |                                               success=False), | ||||||
|  |                                   stats=2) | ||||||
| 
 | 
 | ||||||
|     def test_rerun_async_setup_hook_failure(self): |     def test_rerun_async_setup_hook_failure(self): | ||||||
|         # FAILURE then FAILURE |         # FAILURE then FAILURE | ||||||
|  | @ -1307,11 +1424,12 @@ async def test_success(self): | ||||||
|         """) |         """) | ||||||
|         testname = self.create_test(code=code) |         testname = self.create_test(code=code) | ||||||
| 
 | 
 | ||||||
|         output = self.run_tests("-w", testname, exitcode=EXITCODE_BAD_TEST) |         output = self.run_tests("--rerun", testname, exitcode=EXITCODE_BAD_TEST) | ||||||
|         self.check_executed_tests(output, testname, |         self.check_executed_tests(output, testname, | ||||||
|                                   failed=[testname], |                                   rerun=Rerun(testname, | ||||||
|                                   rerun={testname: "test_success"}, |                                               match="test_success", | ||||||
|                                   stats=1) |                                               success=False), | ||||||
|  |                                   stats=2) | ||||||
| 
 | 
 | ||||||
|     def test_rerun_async_teardown_hook_failure(self): |     def test_rerun_async_teardown_hook_failure(self): | ||||||
|         # FAILURE then FAILURE |         # FAILURE then FAILURE | ||||||
|  | @ -1327,11 +1445,13 @@ async def test_success(self): | ||||||
|         """) |         """) | ||||||
|         testname = self.create_test(code=code) |         testname = self.create_test(code=code) | ||||||
| 
 | 
 | ||||||
|         output = self.run_tests("-w", testname, exitcode=EXITCODE_BAD_TEST) |         output = self.run_tests("--rerun", testname, exitcode=EXITCODE_BAD_TEST) | ||||||
|         self.check_executed_tests(output, testname, |         self.check_executed_tests(output, testname, | ||||||
|                                   failed=[testname], |                                   failed=[testname], | ||||||
|                                   rerun={testname: "test_success"}, |                                   rerun=Rerun(testname, | ||||||
|                                   stats=1) |                                               match="test_success", | ||||||
|  |                                               success=False), | ||||||
|  |                                   stats=2) | ||||||
| 
 | 
 | ||||||
|     def test_no_tests_ran(self): |     def test_no_tests_ran(self): | ||||||
|         code = textwrap.dedent(""" |         code = textwrap.dedent(""" | ||||||
|  | @ -1347,7 +1467,7 @@ def test_bug(self): | ||||||
|                                 exitcode=EXITCODE_NO_TESTS_RAN) |                                 exitcode=EXITCODE_NO_TESTS_RAN) | ||||||
|         self.check_executed_tests(output, [testname], |         self.check_executed_tests(output, [testname], | ||||||
|                                   run_no_tests=testname, |                                   run_no_tests=testname, | ||||||
|                                   stats=0) |                                   stats=0, filtered=True) | ||||||
| 
 | 
 | ||||||
|     def test_no_tests_ran_skip(self): |     def test_no_tests_ran_skip(self): | ||||||
|         code = textwrap.dedent(""" |         code = textwrap.dedent(""" | ||||||
|  | @ -1378,7 +1498,7 @@ def test_bug(self): | ||||||
|                                 exitcode=EXITCODE_NO_TESTS_RAN) |                                 exitcode=EXITCODE_NO_TESTS_RAN) | ||||||
|         self.check_executed_tests(output, [testname, testname2], |         self.check_executed_tests(output, [testname, testname2], | ||||||
|                                   run_no_tests=[testname, testname2], |                                   run_no_tests=[testname, testname2], | ||||||
|                                   stats=0) |                                   stats=0, filtered=True) | ||||||
| 
 | 
 | ||||||
|     def test_no_test_ran_some_test_exist_some_not(self): |     def test_no_test_ran_some_test_exist_some_not(self): | ||||||
|         code = textwrap.dedent(""" |         code = textwrap.dedent(""" | ||||||
|  | @ -1402,7 +1522,7 @@ def test_other_bug(self): | ||||||
|                                 "-m", "test_other_bug", exitcode=0) |                                 "-m", "test_other_bug", exitcode=0) | ||||||
|         self.check_executed_tests(output, [testname, testname2], |         self.check_executed_tests(output, [testname, testname2], | ||||||
|                                   run_no_tests=[testname], |                                   run_no_tests=[testname], | ||||||
|                                   stats=1) |                                   stats=1, filtered=True) | ||||||
| 
 | 
 | ||||||
|     @support.cpython_only |     @support.cpython_only | ||||||
|     def test_uncollectable(self): |     def test_uncollectable(self): | ||||||
|  | @ -1719,6 +1839,17 @@ def test_format_duration(self): | ||||||
|         self.assertEqual(utils.format_duration(3 * 3600 + 1), |         self.assertEqual(utils.format_duration(3 * 3600 + 1), | ||||||
|                          '3 hour 1 sec') |                          '3 hour 1 sec') | ||||||
| 
 | 
 | ||||||
|  |     def test_normalize_test_name(self): | ||||||
|  |         normalize = normalize_test_name | ||||||
|  |         self.assertEqual(normalize('test_access (test.test_os.FileTests.test_access)'), | ||||||
|  |                          'test_access') | ||||||
|  |         self.assertEqual(normalize('setUpClass (test.test_os.ChownFileTests)', is_error=True), | ||||||
|  |                          'ChownFileTests') | ||||||
|  |         self.assertEqual(normalize('test_success (test.test_bug.ExampleTests.test_success)', is_error=True), | ||||||
|  |                          'test_success') | ||||||
|  |         self.assertIsNone(normalize('setUpModule (test.test_x)', is_error=True)) | ||||||
|  |         self.assertIsNone(normalize('tearDownModule (test.test_module)', is_error=True)) | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
| if __name__ == '__main__': | if __name__ == '__main__': | ||||||
|     unittest.main() |     unittest.main() | ||||||
|  |  | ||||||
|  | @ -0,0 +1,6 @@ | ||||||
|  | When regrtest reruns failed tests in verbose mode (``./python -m test | ||||||
|  | --rerun``), tests are now rerun in fresh worker processes rather than being | ||||||
|  | executed in the main process. If a test does crash or is killed by a timeout, | ||||||
|  | the main process can detect and handle the killed worker process. Tests are | ||||||
|  | rerun in parallel if the ``-jN`` option is used to run tests in parallel. | ||||||
|  | Patch by Victor Stinner. | ||||||
|  | @ -0,0 +1,2 @@ | ||||||
|  | Rename regrtest ``--verbose2`` option (``-w``) to ``--rerun``. Keep | ||||||
|  | ``--verbose2`` as a deprecated alias. Patch by Victor Stinner. | ||||||
|  | @ -0,0 +1,3 @@ | ||||||
|  | Add ``--fail-rerun option`` option to regrtest: if a test failed when then | ||||||
|  | passed when rerun in verbose mode, exit the process with exit code 2 | ||||||
|  | (error), instead of exit code 0 (success). Patch by Victor Stinner. | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Victor Stinner
						Victor Stinner