Commit da57599a authored by Victor Stinner's avatar Victor Stinner Committed by GitHub

bpo-38203: regrtest: put a 2 min timeout on Python exit (GH-16250)

parent 272d0d01
...@@ -22,6 +22,12 @@ from test.libregrtest.utils import removepy, count, format_duration, printlist ...@@ -22,6 +22,12 @@ from test.libregrtest.utils import removepy, count, format_duration, printlist
from test import support from test import support
# bpo-38203: Maximum delay in seconds to exit Python (call Py_Finalize()).
# Used to protect against threading._shutdown() hang.
# Must be smaller than buildbot "1200 seconds without output" limit.
EXIT_TIMEOUT = 120.0
class Regrtest: class Regrtest:
"""Execute a test suite. """Execute a test suite.
...@@ -616,16 +622,24 @@ class Regrtest: ...@@ -616,16 +622,24 @@ class Regrtest:
test_cwd = self.create_temp_dir() test_cwd = self.create_temp_dir()
# Run the tests in a context manager that temporarily changes the CWD try:
# to a temporary and writable directory. If it's not possible to # Run the tests in a context manager that temporarily changes the CWD
# create or change the CWD, the original CWD will be used. # to a temporary and writable directory. If it's not possible to
# The original CWD is available from support.SAVEDCWD. # create or change the CWD, the original CWD will be used.
with support.temp_cwd(test_cwd, quiet=True): # The original CWD is available from support.SAVEDCWD.
# When using multiprocessing, worker processes will use test_cwd with support.temp_cwd(test_cwd, quiet=True):
# as their parent temporary directory. So when the main process # When using multiprocessing, worker processes will use test_cwd
# exit, it removes also subdirectories of worker processes. # as their parent temporary directory. So when the main process
self.ns.tempdir = test_cwd # exit, it removes also subdirectories of worker processes.
self._main(tests, kwargs) self.ns.tempdir = test_cwd
self._main(tests, kwargs)
except SystemExit as exc:
# bpo-38203: Python can hang at exit in Py_Finalize(), especially
# on threading._shutdown() call: put a timeout
faulthandler.dump_traceback_later(EXIT_TIMEOUT, exit=True)
sys.exit(exc.code)
def getloadavg(self): def getloadavg(self):
if self.win_load_tracker is not None: if self.win_load_tracker is not None:
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment