blob: df114a2188f86410284c86706a9cda7e99ac2a5b [file] [log] [blame]
Brett Cannonf1cfb622003-05-04 21:15:27 +00001"""Supporting definitions for the Python regression tests."""
Guido van Rossum3bead091992-01-27 17:00:37 +00002
Benjamin Petersonee8712c2008-05-20 21:35:26 +00003if __name__ != 'test.support':
4 raise ImportError('support must be imported from the test package')
Barry Warsaw408b6d32002-07-30 23:27:12 +00005
Guido van Rossumd8faa362007-04-27 19:54:29 +00006import contextlib
7import errno
8import socket
Fred Drakecd1b1dd2001-03-21 18:26:33 +00009import sys
Guido van Rossumd8faa362007-04-27 19:54:29 +000010import os
Benjamin Petersone549ead2009-03-28 21:42:05 +000011import platform
Christian Heimes23daade02008-02-25 12:39:23 +000012import shutil
Thomas Wouters902d6eb2007-01-09 23:18:33 +000013import warnings
Guido van Rossumd8faa362007-04-27 19:54:29 +000014import unittest
R. David Murraya21e4ca2009-03-31 23:16:50 +000015import importlib
Walter Dörwald155374d2009-05-01 19:58:58 +000016import collections
Fred Drakecd1b1dd2001-03-21 18:26:33 +000017
Benjamin Petersone549ead2009-03-28 21:42:05 +000018__all__ = ["Error", "TestFailed", "ResourceDenied", "import_module",
Benjamin Peterson744c2cd2008-05-26 16:26:37 +000019 "verbose", "use_resources", "max_memuse", "record_original_stdout",
20 "get_original_stdout", "unload", "unlink", "rmtree", "forget",
21 "is_resource_enabled", "requires", "find_unused_port", "bind_port",
Benjamin Peterson79e48032008-05-26 17:44:33 +000022 "fcmp", "is_jython", "TESTFN", "HOST", "FUZZ", "findfile", "verify",
23 "vereq", "sortdict", "check_syntax_error", "open_urlresource",
Benjamin Petersonfcf5d632008-10-16 23:24:44 +000024 "check_warnings", "CleanImport", "EnvironmentVarGuard",
Benjamin Peterson79e48032008-05-26 17:44:33 +000025 "TransientResource", "captured_output", "captured_stdout",
26 "TransientResource", "transient_internet", "run_with_locale",
27 "set_memlimit", "bigmemtest", "bigaddrspacetest", "BasicTestRunner",
28 "run_unittest", "run_doctest", "threading_setup", "threading_cleanup",
R. David Murraya21e4ca2009-03-31 23:16:50 +000029 "reap_children", "cpython_only", "check_impl_detail", "get_attribute"]
Benjamin Peterson744c2cd2008-05-26 16:26:37 +000030
Fred Drake1790dd42000-07-24 06:55:00 +000031class Error(Exception):
Fred Drake004d5e62000-10-23 17:22:08 +000032 """Base class for regression test exceptions."""
Fred Drake1790dd42000-07-24 06:55:00 +000033
34class TestFailed(Error):
Fred Drake004d5e62000-10-23 17:22:08 +000035 """Test failed."""
Fred Drake1790dd42000-07-24 06:55:00 +000036
Benjamin Petersone549ead2009-03-28 21:42:05 +000037class ResourceDenied(unittest.SkipTest):
Fred Drake9a0db072003-02-03 15:19:30 +000038 """Test skipped because it requested a disallowed resource.
39
40 This is raised when a test calls requires() for a resource that
41 has not be enabled. It is used to distinguish between expected
42 and unexpected skips.
43 """
44
Nick Coghlanfce769e2009-04-11 14:30:59 +000045@contextlib.contextmanager
46def _ignore_deprecated_imports(ignore=True):
47 """Context manager to suppress package and module deprecation
48 warnings when importing them.
49
50 If ignore is False, this context manager has no effect."""
51 if ignore:
52 with warnings.catch_warnings():
53 warnings.filterwarnings("ignore", ".+ (module|package)",
54 DeprecationWarning)
55 yield
56 else:
57 yield
58
59
Benjamin Peterson699adb92008-05-08 22:27:58 +000060def import_module(name, deprecated=False):
R. David Murraya21e4ca2009-03-31 23:16:50 +000061 """Import and return the module to be tested, raising SkipTest if
62 it is not available.
63
64 If deprecated is True, any module or package deprecation messages
65 will be suppressed."""
Nick Coghlanfce769e2009-04-11 14:30:59 +000066 with _ignore_deprecated_imports(deprecated):
Benjamin Peterson699adb92008-05-08 22:27:58 +000067 try:
Nick Coghlanfce769e2009-04-11 14:30:59 +000068 return importlib.import_module(name)
R. David Murraya21e4ca2009-03-31 23:16:50 +000069 except ImportError as msg:
70 raise unittest.SkipTest(str(msg))
Nick Coghlanfce769e2009-04-11 14:30:59 +000071
72
Nick Coghlan47384702009-04-22 16:13:36 +000073def _save_and_remove_module(name, orig_modules):
74 """Helper function to save and remove a module from sys.modules
75
76 Return value is True if the module was in sys.modules and
77 False otherwise."""
78 saved = True
79 try:
80 orig_modules[name] = sys.modules[name]
81 except KeyError:
82 saved = False
83 else:
84 del sys.modules[name]
85 return saved
86
87
88def _save_and_block_module(name, orig_modules):
89 """Helper function to save and block a module in sys.modules
90
91 Return value is True if the module was in sys.modules and
92 False otherwise."""
93 saved = True
94 try:
95 orig_modules[name] = sys.modules[name]
96 except KeyError:
97 saved = False
98 sys.modules[name] = 0
99 return saved
100
101
102def import_fresh_module(name, fresh=(), blocked=(), deprecated=False):
Nick Coghlanfce769e2009-04-11 14:30:59 +0000103 """Imports and returns a module, deliberately bypassing the sys.modules cache
104 and importing a fresh copy of the module. Once the import is complete,
105 the sys.modules cache is restored to its original state.
106
Nick Coghlan47384702009-04-22 16:13:36 +0000107 Modules named in fresh are also imported anew if needed by the import.
108
109 Importing of modules named in blocked is prevented while the fresh import
Nick Coghlanfce769e2009-04-11 14:30:59 +0000110 takes place.
111
112 If deprecated is True, any module or package deprecation messages
113 will be suppressed."""
114 # NOTE: test_heapq and test_warnings include extra sanity checks to make
115 # sure that this utility function is working as expected
116 with _ignore_deprecated_imports(deprecated):
Nick Coghlan47384702009-04-22 16:13:36 +0000117 # Keep track of modules saved for later restoration as well
118 # as those which just need a blocking entry removed
Nick Coghlanfce769e2009-04-11 14:30:59 +0000119 orig_modules = {}
Nick Coghlan47384702009-04-22 16:13:36 +0000120 names_to_remove = []
121 _save_and_remove_module(name, orig_modules)
Nick Coghlanfce769e2009-04-11 14:30:59 +0000122 try:
Nick Coghlan47384702009-04-22 16:13:36 +0000123 for fresh_name in fresh:
124 _save_and_remove_module(fresh_name, orig_modules)
125 for blocked_name in blocked:
126 if not _save_and_block_module(blocked_name, orig_modules):
127 names_to_remove.append(blocked_name)
128 fresh_module = importlib.import_module(name)
Nick Coghlanfce769e2009-04-11 14:30:59 +0000129 finally:
Nick Coghlan47384702009-04-22 16:13:36 +0000130 for orig_name, module in orig_modules.items():
131 sys.modules[orig_name] = module
132 for name_to_remove in names_to_remove:
133 del sys.modules[name_to_remove]
134 return fresh_module
Nick Coghlanfce769e2009-04-11 14:30:59 +0000135
Benjamin Peterson699adb92008-05-08 22:27:58 +0000136
R. David Murraya21e4ca2009-03-31 23:16:50 +0000137def get_attribute(obj, name):
138 """Get an attribute, raising SkipTest if AttributeError is raised."""
139 try:
140 attribute = getattr(obj, name)
141 except AttributeError:
142 raise unittest.SkipTest("module %s has no attribute %s" % (
143 obj.__name__, name))
144 else:
145 return attribute
146
Barry Warsawc0fb6052001-08-20 22:29:23 +0000147verbose = 1 # Flag set to 0 by regrtest.py
Thomas Wouters477c8d52006-05-27 19:21:47 +0000148use_resources = None # Flag set to [] by regrtest.py
149max_memuse = 0 # Disable bigmem tests (they will still be run with
150 # small sizes, to make sure they work.)
Neal Norwitz3ce5d922008-08-24 07:08:55 +0000151real_max_memuse = 0
Guido van Rossum531661c1996-12-20 02:58:22 +0000152
Tim Peters8dee8092001-09-25 20:05:11 +0000153# _original_stdout is meant to hold stdout at the time regrtest began.
154# This may be "the real" stdout, or IDLE's emulation of stdout, or whatever.
155# The point is to have some flavor of stdout the user can actually see.
156_original_stdout = None
157def record_original_stdout(stdout):
158 global _original_stdout
159 _original_stdout = stdout
160
161def get_original_stdout():
162 return _original_stdout or sys.stdout
163
Guido van Rossum3bead091992-01-27 17:00:37 +0000164def unload(name):
Fred Drake004d5e62000-10-23 17:22:08 +0000165 try:
166 del sys.modules[name]
167 except KeyError:
168 pass
Guido van Rossum3bead091992-01-27 17:00:37 +0000169
Neal Norwitz0e17f8c2006-01-23 07:51:27 +0000170def unlink(filename):
Neal Norwitz0e17f8c2006-01-23 07:51:27 +0000171 try:
172 os.unlink(filename)
173 except OSError:
174 pass
175
Christian Heimes23daade02008-02-25 12:39:23 +0000176def rmtree(path):
177 try:
178 shutil.rmtree(path)
179 except OSError as e:
180 # Unix returns ENOENT, Windows returns ESRCH.
181 if e.errno not in (errno.ENOENT, errno.ESRCH):
182 raise
183
Guido van Rossum3bead091992-01-27 17:00:37 +0000184def forget(modname):
Brett Cannonf1cfb622003-05-04 21:15:27 +0000185 '''"Forget" a module was ever imported by removing it from sys.modules and
186 deleting any .pyc and .pyo files.'''
Fred Drake004d5e62000-10-23 17:22:08 +0000187 unload(modname)
Fred Drake004d5e62000-10-23 17:22:08 +0000188 for dirname in sys.path:
Skip Montanaro7a98be22007-08-16 14:35:24 +0000189 unlink(os.path.join(dirname, modname + '.pyc'))
Brett Cannonf1cfb622003-05-04 21:15:27 +0000190 # Deleting the .pyo file cannot be within the 'try' for the .pyc since
191 # the chance exists that there is no .pyc (and thus the 'try' statement
192 # is exited) but there is a .pyo file.
Skip Montanaro7a98be22007-08-16 14:35:24 +0000193 unlink(os.path.join(dirname, modname + '.pyo'))
Guido van Rossum3bead091992-01-27 17:00:37 +0000194
Tim Petersb4ee4eb2002-12-04 03:26:57 +0000195def is_resource_enabled(resource):
Brett Cannonf1cfb622003-05-04 21:15:27 +0000196 """Test whether a resource is enabled. Known resources are set by
197 regrtest.py."""
Tim Petersb4ee4eb2002-12-04 03:26:57 +0000198 return use_resources is not None and resource in use_resources
199
Barry Warsawc0fb6052001-08-20 22:29:23 +0000200def requires(resource, msg=None):
Brett Cannonf1cfb622003-05-04 21:15:27 +0000201 """Raise ResourceDenied if the specified resource is not available.
202
203 If the caller's module is __main__ then automatically return True. The
204 possibility of False being returned occurs when regrtest.py is executing."""
Skip Montanarod839ecd2003-04-24 19:06:57 +0000205 # see if the caller's module is __main__ - if so, treat as if
206 # the resource was set
Benjamin Petersone549ead2009-03-28 21:42:05 +0000207 if sys._getframe(1).f_globals.get("__name__") == "__main__":
Skip Montanarod839ecd2003-04-24 19:06:57 +0000208 return
Tim Petersb4ee4eb2002-12-04 03:26:57 +0000209 if not is_resource_enabled(resource):
Barry Warsawc0fb6052001-08-20 22:29:23 +0000210 if msg is None:
211 msg = "Use of the `%s' resource not enabled" % resource
Fred Drake9a0db072003-02-03 15:19:30 +0000212 raise ResourceDenied(msg)
Barry Warsawc0fb6052001-08-20 22:29:23 +0000213
Christian Heimes5e696852008-04-09 08:37:03 +0000214HOST = 'localhost'
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000215
Christian Heimes5e696852008-04-09 08:37:03 +0000216def find_unused_port(family=socket.AF_INET, socktype=socket.SOCK_STREAM):
217 """Returns an unused port that should be suitable for binding. This is
218 achieved by creating a temporary socket with the same family and type as
219 the 'sock' parameter (default is AF_INET, SOCK_STREAM), and binding it to
220 the specified host address (defaults to 0.0.0.0) with the port set to 0,
221 eliciting an unused ephemeral port from the OS. The temporary socket is
222 then closed and deleted, and the ephemeral port is returned.
223
224 Either this method or bind_port() should be used for any tests where a
225 server socket needs to be bound to a particular port for the duration of
226 the test. Which one to use depends on whether the calling code is creating
227 a python socket, or if an unused port needs to be provided in a constructor
228 or passed to an external program (i.e. the -accept argument to openssl's
229 s_server mode). Always prefer bind_port() over find_unused_port() where
230 possible. Hard coded ports should *NEVER* be used. As soon as a server
231 socket is bound to a hard coded port, the ability to run multiple instances
232 of the test simultaneously on the same host is compromised, which makes the
233 test a ticking time bomb in a buildbot environment. On Unix buildbots, this
234 may simply manifest as a failed test, which can be recovered from without
235 intervention in most cases, but on Windows, the entire python process can
236 completely and utterly wedge, requiring someone to log in to the buildbot
237 and manually kill the affected process.
238
239 (This is easy to reproduce on Windows, unfortunately, and can be traced to
240 the SO_REUSEADDR socket option having different semantics on Windows versus
241 Unix/Linux. On Unix, you can't have two AF_INET SOCK_STREAM sockets bind,
242 listen and then accept connections on identical host/ports. An EADDRINUSE
243 socket.error will be raised at some point (depending on the platform and
244 the order bind and listen were called on each socket).
245
246 However, on Windows, if SO_REUSEADDR is set on the sockets, no EADDRINUSE
247 will ever be raised when attempting to bind two identical host/ports. When
248 accept() is called on each socket, the second caller's process will steal
249 the port from the first caller, leaving them both in an awkwardly wedged
250 state where they'll no longer respond to any signals or graceful kills, and
251 must be forcibly killed via OpenProcess()/TerminateProcess().
252
253 The solution on Windows is to use the SO_EXCLUSIVEADDRUSE socket option
254 instead of SO_REUSEADDR, which effectively affords the same semantics as
255 SO_REUSEADDR on Unix. Given the propensity of Unix developers in the Open
256 Source world compared to Windows ones, this is a common mistake. A quick
257 look over OpenSSL's 0.9.8g source shows that they use SO_REUSEADDR when
258 openssl.exe is called with the 's_server' option, for example. See
259 http://bugs.python.org/issue2550 for more info. The following site also
260 has a very thorough description about the implications of both REUSEADDR
261 and EXCLUSIVEADDRUSE on Windows:
262 http://msdn2.microsoft.com/en-us/library/ms740621(VS.85).aspx)
263
264 XXX: although this approach is a vast improvement on previous attempts to
265 elicit unused ports, it rests heavily on the assumption that the ephemeral
266 port returned to us by the OS won't immediately be dished back out to some
267 other process when we close and delete our temporary socket but before our
268 calling code has a chance to bind the returned port. We can deal with this
269 issue if/when we come across it.
270 """
271
272 tempsock = socket.socket(family, socktype)
273 port = bind_port(tempsock)
274 tempsock.close()
275 del tempsock
276 return port
277
278def bind_port(sock, host=HOST):
279 """Bind the socket to a free port and return the port number. Relies on
280 ephemeral ports in order to ensure we are using an unbound port. This is
281 important as many tests may be running simultaneously, especially in a
282 buildbot environment. This method raises an exception if the sock.family
283 is AF_INET and sock.type is SOCK_STREAM, *and* the socket has SO_REUSEADDR
284 or SO_REUSEPORT set on it. Tests should *never* set these socket options
285 for TCP/IP sockets. The only case for setting these options is testing
286 multicasting via multiple UDP sockets.
287
288 Additionally, if the SO_EXCLUSIVEADDRUSE socket option is available (i.e.
289 on Windows), it will be set on the socket. This will prevent anyone else
290 from bind()'ing to our host/port for the duration of the test.
291 """
292
293 if sock.family == socket.AF_INET and sock.type == socket.SOCK_STREAM:
294 if hasattr(socket, 'SO_REUSEADDR'):
295 if sock.getsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR) == 1:
296 raise TestFailed("tests should never set the SO_REUSEADDR " \
297 "socket option on TCP/IP sockets!")
298 if hasattr(socket, 'SO_REUSEPORT'):
299 if sock.getsockopt(socket.SOL_SOCKET, socket.SO_REUSEPORT) == 1:
300 raise TestFailed("tests should never set the SO_REUSEPORT " \
301 "socket option on TCP/IP sockets!")
302 if hasattr(socket, 'SO_EXCLUSIVEADDRUSE'):
303 sock.setsockopt(socket.SOL_SOCKET, socket.SO_EXCLUSIVEADDRUSE, 1)
304
305 sock.bind((host, 0))
306 port = sock.getsockname()[1]
307 return port
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000308
Guido van Rossum35fb82a1993-01-26 13:04:43 +0000309FUZZ = 1e-6
310
311def fcmp(x, y): # fuzzy comparison function
Neal Norwitz79212992006-08-21 16:27:31 +0000312 if isinstance(x, float) or isinstance(y, float):
Fred Drake004d5e62000-10-23 17:22:08 +0000313 try:
Fred Drake004d5e62000-10-23 17:22:08 +0000314 fuzz = (abs(x) + abs(y)) * FUZZ
315 if abs(x-y) <= fuzz:
316 return 0
317 except:
318 pass
Neal Norwitz79212992006-08-21 16:27:31 +0000319 elif type(x) == type(y) and isinstance(x, (tuple, list)):
Fred Drake004d5e62000-10-23 17:22:08 +0000320 for i in range(min(len(x), len(y))):
321 outcome = fcmp(x[i], y[i])
Fred Drake132dce22000-12-12 23:11:42 +0000322 if outcome != 0:
Fred Drake004d5e62000-10-23 17:22:08 +0000323 return outcome
Guido van Rossum47b9ff62006-08-24 00:41:19 +0000324 return (len(x) > len(y)) - (len(x) < len(y))
325 return (x > y) - (x < y)
Guido van Rossum35fb82a1993-01-26 13:04:43 +0000326
Finn Bock57bc5fa2002-11-01 18:02:03 +0000327is_jython = sys.platform.startswith('java')
328
Barry Warsaw559f6682001-03-23 18:04:02 +0000329# Filename used for testing
330if os.name == 'java':
331 # Jython disallows @ in module names
332 TESTFN = '$test'
Martin v. Löwisa94568a2003-05-10 07:36:56 +0000333else:
Barry Warsaw559f6682001-03-23 18:04:02 +0000334 TESTFN = '@test'
Walter Dörwald9b775532007-06-08 14:30:53 +0000335
336 # Assuming sys.getfilesystemencoding()!=sys.getdefaultencoding()
337 # TESTFN_UNICODE is a filename that can be encoded using the
338 # file system encoding, but *not* with the default (ascii) encoding
339 TESTFN_UNICODE = "@test-\xe0\xf2"
340 TESTFN_ENCODING = sys.getfilesystemencoding()
341 # TESTFN_UNICODE_UNENCODEABLE is a filename that should *not* be
342 # able to be encoded by *either* the default or filesystem encoding.
343 # This test really only makes sense on Windows NT platforms
344 # which have special Unicode support in posixmodule.
345 if (not hasattr(sys, "getwindowsversion") or
346 sys.getwindowsversion()[3] < 2): # 0=win32s or 1=9x/ME
347 TESTFN_UNICODE_UNENCODEABLE = None
348 else:
349 # Japanese characters (I think - from bug 846133)
350 TESTFN_UNICODE_UNENCODEABLE = "@test-\u5171\u6709\u3055\u308c\u308b"
351 try:
352 # XXX - Note - should be using TESTFN_ENCODING here - but for
353 # Windows, "mbcs" currently always operates as if in
354 # errors=ignore' mode - hence we get '?' characters rather than
355 # the exception. 'Latin1' operates as we expect - ie, fails.
356 # See [ 850997 ] mbcs encoding ignores errors
357 TESTFN_UNICODE_UNENCODEABLE.encode("Latin1")
358 except UnicodeEncodeError:
359 pass
Martin v. Löwis2411a2d2002-11-09 19:57:26 +0000360 else:
Georg Brandldb028442008-02-05 20:48:58 +0000361 print('WARNING: The filename %r CAN be encoded by the filesystem. '
362 'Unicode filename tests may not be effective'
363 % TESTFN_UNICODE_UNENCODEABLE)
Neal Norwitz26a1eef2002-11-03 00:35:53 +0000364
365# Make sure we can write to TESTFN, try in /tmp if we can't
366fp = None
367try:
368 fp = open(TESTFN, 'w+')
369except IOError:
370 TMP_TESTFN = os.path.join('/tmp', TESTFN)
371 try:
372 fp = open(TMP_TESTFN, 'w+')
373 TESTFN = TMP_TESTFN
374 del TMP_TESTFN
375 except IOError:
Guido van Rossumbe19ed72007-02-09 05:37:30 +0000376 print(('WARNING: tests will fail, unable to write to: %s or %s' %
377 (TESTFN, TMP_TESTFN)))
Neal Norwitz26a1eef2002-11-03 00:35:53 +0000378if fp is not None:
379 fp.close()
Neal Norwitz0e17f8c2006-01-23 07:51:27 +0000380 unlink(TESTFN)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000381del fp
Guido van Rossuma8f7e592001-03-13 09:31:07 +0000382
Guido van Rossume26132c1998-04-23 20:13:30 +0000383def findfile(file, here=__file__):
Brett Cannonf1cfb622003-05-04 21:15:27 +0000384 """Try to find a file on sys.path and the working directory. If it is not
385 found the argument passed to the function is returned (this does not
386 necessarily signal failure; could still be the legitimate path)."""
Fred Drake004d5e62000-10-23 17:22:08 +0000387 if os.path.isabs(file):
388 return file
Fred Drake004d5e62000-10-23 17:22:08 +0000389 path = sys.path
390 path = [os.path.dirname(here)] + path
391 for dn in path:
392 fn = os.path.join(dn, file)
393 if os.path.exists(fn): return fn
394 return file
Marc-André Lemburg36619082001-01-17 19:11:13 +0000395
396def verify(condition, reason='test failed'):
Guido van Rossuma1374e42001-01-19 19:01:56 +0000397 """Verify that condition is true. If not, raise TestFailed.
Marc-André Lemburg36619082001-01-17 19:11:13 +0000398
Skip Montanaroc955f892001-01-20 19:12:54 +0000399 The optional argument reason can be given to provide
Tim Peters983874d2001-01-19 05:59:21 +0000400 a better error text.
Tim Petersd2bf3b72001-01-18 02:22:22 +0000401 """
Tim Peters983874d2001-01-19 05:59:21 +0000402
Tim Petersd2bf3b72001-01-18 02:22:22 +0000403 if not condition:
Guido van Rossuma1374e42001-01-19 19:01:56 +0000404 raise TestFailed(reason)
Jeremy Hylton47793992001-02-19 15:35:26 +0000405
Tim Petersc2fe6182001-10-30 23:20:46 +0000406def vereq(a, b):
Tim Peters77902972001-12-29 17:34:57 +0000407 """Raise TestFailed if a == b is false.
408
409 This is better than verify(a == b) because, in case of failure, the
410 error message incorporates repr(a) and repr(b) so you can see the
411 inputs.
412
413 Note that "not (a == b)" isn't necessarily the same as "a != b"; the
414 former is tested.
415 """
416
Tim Petersc2fe6182001-10-30 23:20:46 +0000417 if not (a == b):
Guido van Rossumd8faa362007-04-27 19:54:29 +0000418 raise TestFailed("%r == %r" % (a, b))
Tim Petersc2fe6182001-10-30 23:20:46 +0000419
Tim Peters2f228e72001-05-13 00:19:31 +0000420def sortdict(dict):
421 "Like repr(dict), but in sorted order."
Guido van Rossumcc2b0162007-02-11 06:12:03 +0000422 items = sorted(dict.items())
Tim Peters2f228e72001-05-13 00:19:31 +0000423 reprpairs = ["%r: %r" % pair for pair in items]
424 withcommas = ", ".join(reprpairs)
425 return "{%s}" % withcommas
426
Benjamin Peterson7522c742009-01-19 21:00:09 +0000427def make_bad_fd():
428 """
429 Create an invalid file descriptor by opening and closing a file and return
430 its fd.
431 """
432 file = open(TESTFN, "wb")
433 try:
434 return file.fileno()
435 finally:
436 file.close()
437 unlink(TESTFN)
438
Thomas Wouters89f507f2006-12-13 04:49:30 +0000439def check_syntax_error(testcase, statement):
Benjamin Petersone549ead2009-03-28 21:42:05 +0000440 testcase.assertRaises(SyntaxError, compile, statement,
441 '<test string>', 'exec')
Fred Drakecd1b1dd2001-03-21 18:26:33 +0000442
Martin v. Löwis234a34a2007-08-30 20:58:02 +0000443def open_urlresource(url, *args, **kw):
Jeremy Hylton1afc1692008-06-18 20:49:58 +0000444 import urllib.request, urllib.parse
Fred Drakecd1b1dd2001-03-21 18:26:33 +0000445
Guido van Rossum360e4b82007-05-14 22:51:27 +0000446 requires('urlfetch')
Jeremy Hylton1afc1692008-06-18 20:49:58 +0000447 filename = urllib.parse.urlparse(url)[2].split('/')[-1] # '/': it's URL!
Hye-Shik Changaaa2f1d2005-12-10 17:44:27 +0000448
449 for path in [os.path.curdir, os.path.pardir]:
450 fn = os.path.join(path, filename)
451 if os.path.exists(fn):
Martin v. Löwis234a34a2007-08-30 20:58:02 +0000452 return open(fn, *args, **kw)
Hye-Shik Changaaa2f1d2005-12-10 17:44:27 +0000453
Guido van Rossumbe19ed72007-02-09 05:37:30 +0000454 print('\tfetching %s ...' % url, file=get_original_stdout())
Jeremy Hylton1afc1692008-06-18 20:49:58 +0000455 fn, _ = urllib.request.urlretrieve(url, filename)
Martin v. Löwis234a34a2007-08-30 20:58:02 +0000456 return open(fn, *args, **kw)
Thomas Wouters9fe394c2007-02-05 01:24:16 +0000457
Thomas Wouters902d6eb2007-01-09 23:18:33 +0000458
Benjamin Petersonfcf5d632008-10-16 23:24:44 +0000459class WarningsRecorder(object):
460 """Convenience wrapper for the warnings list returned on
461 entry to the warnings.catch_warnings() context manager.
Guido van Rossumd8faa362007-04-27 19:54:29 +0000462 """
Benjamin Petersonfcf5d632008-10-16 23:24:44 +0000463 def __init__(self, warnings_list):
464 self.warnings = warnings_list
465
466 def __getattr__(self, attr):
467 if self.warnings:
468 return getattr(self.warnings[-1], attr)
469 elif attr in warnings.WarningMessage._WARNING_DETAILS:
470 return None
471 raise AttributeError("%r has no attribute %r" % (self, attr))
472
473 def reset(self):
474 del self.warnings[:]
475
476@contextlib.contextmanager
477def check_warnings():
478 with warnings.catch_warnings(record=True) as w:
479 yield WarningsRecorder(w)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000480
Alexandre Vassalotti5f8ced22008-05-16 00:03:33 +0000481
482class CleanImport(object):
483 """Context manager to force import to return a new module reference.
484
485 This is useful for testing module-level behaviours, such as
Nick Coghlanb1304932008-07-13 12:25:08 +0000486 the emission of a DeprecationWarning on import.
Alexandre Vassalotti5f8ced22008-05-16 00:03:33 +0000487
488 Use like this:
489
490 with CleanImport("foo"):
491 __import__("foo") # new reference
492 """
493
494 def __init__(self, *module_names):
495 self.original_modules = sys.modules.copy()
496 for module_name in module_names:
497 if module_name in sys.modules:
498 module = sys.modules[module_name]
499 # It is possible that module_name is just an alias for
500 # another module (e.g. stub for modules renamed in 3.x).
501 # In that case, we also need delete the real module to clear
502 # the import cache.
503 if module.__name__ != module_name:
504 del sys.modules[module.__name__]
505 del sys.modules[module_name]
506
507 def __enter__(self):
508 return self
509
510 def __exit__(self, *ignore_exc):
511 sys.modules.update(self.original_modules)
512
513
Walter Dörwald155374d2009-05-01 19:58:58 +0000514class EnvironmentVarGuard(collections.MutableMapping):
Thomas Wouters902d6eb2007-01-09 23:18:33 +0000515
516 """Class to help protect the environment variable properly. Can be used as
517 a context manager."""
518
519 def __init__(self):
Walter Dörwald155374d2009-05-01 19:58:58 +0000520 self._environ = os.environ
Walter Dörwald4ba80132009-04-25 12:48:43 +0000521 self._changed = {}
Thomas Wouters902d6eb2007-01-09 23:18:33 +0000522
Walter Dörwald155374d2009-05-01 19:58:58 +0000523 def __getitem__(self, envvar):
524 return self._environ[envvar]
525
526 def __setitem__(self, envvar, value):
Walter Dörwald4ba80132009-04-25 12:48:43 +0000527 # Remember the initial value on the first access
528 if envvar not in self._changed:
Walter Dörwald155374d2009-05-01 19:58:58 +0000529 self._changed[envvar] = self._environ.get(envvar)
530 self._environ[envvar] = value
531
532 def __delitem__(self, envvar):
533 # Remember the initial value on the first access
534 if envvar not in self._changed:
535 self._changed[envvar] = self._environ.get(envvar)
536 if envvar in self._environ:
537 del self._environ[envvar]
538
539 def keys(self):
540 return self._environ.keys()
541
542 def __iter__(self):
543 return iter(self._environ)
544
545 def __len__(self):
546 return len(self._environ)
547
548 def set(self, envvar, value):
549 self[envvar] = value
Thomas Wouters902d6eb2007-01-09 23:18:33 +0000550
551 def unset(self, envvar):
Walter Dörwald155374d2009-05-01 19:58:58 +0000552 del self[envvar]
Thomas Wouters902d6eb2007-01-09 23:18:33 +0000553
554 def __enter__(self):
555 return self
556
557 def __exit__(self, *ignore_exc):
Walter Dörwald4ba80132009-04-25 12:48:43 +0000558 for (k, v) in self._changed.items():
559 if v is None:
Walter Dörwald155374d2009-05-01 19:58:58 +0000560 if k in self._environ:
561 del self._environ[k]
Walter Dörwald4ba80132009-04-25 12:48:43 +0000562 else:
Walter Dörwald155374d2009-05-01 19:58:58 +0000563 self._environ[k] = v
564
Thomas Wouters902d6eb2007-01-09 23:18:33 +0000565
Guido van Rossumd8faa362007-04-27 19:54:29 +0000566class TransientResource(object):
567
568 """Raise ResourceDenied if an exception is raised while the context manager
569 is in effect that matches the specified exception and attributes."""
570
571 def __init__(self, exc, **kwargs):
572 self.exc = exc
573 self.attrs = kwargs
574
575 def __enter__(self):
576 return self
577
578 def __exit__(self, type_=None, value=None, traceback=None):
579 """If type_ is a subclass of self.exc and value has attributes matching
580 self.attrs, raise ResourceDenied. Otherwise let the exception
581 propagate (if any)."""
582 if type_ is not None and issubclass(self.exc, type_):
583 for attr, attr_value in self.attrs.items():
584 if not hasattr(value, attr):
585 break
586 if getattr(value, attr) != attr_value:
587 break
588 else:
589 raise ResourceDenied("an optional resource is not available")
590
591
592def transient_internet():
593 """Return a context manager that raises ResourceDenied when various issues
594 with the Internet connection manifest themselves as exceptions."""
595 time_out = TransientResource(IOError, errno=errno.ETIMEDOUT)
596 socket_peer_reset = TransientResource(socket.error, errno=errno.ECONNRESET)
597 ioerror_peer_reset = TransientResource(IOError, errno=errno.ECONNRESET)
598 return contextlib.nested(time_out, socket_peer_reset, ioerror_peer_reset)
599
Fred Drakecd1b1dd2001-03-21 18:26:33 +0000600
Thomas Woutersed03b412007-08-28 21:37:11 +0000601@contextlib.contextmanager
Benjamin Petersonad9d48d2008-04-02 21:49:44 +0000602def captured_output(stream_name):
603 """Run the 'with' statement body using a StringIO object in place of a
604 specific attribute on the sys module.
605 Example use (with 'stream_name=stdout')::
Thomas Woutersed03b412007-08-28 21:37:11 +0000606
607 with captured_stdout() as s:
Neal Norwitz752abd02008-05-13 04:55:24 +0000608 print("hello")
Thomas Woutersed03b412007-08-28 21:37:11 +0000609 assert s.getvalue() == "hello"
610 """
611 import io
Benjamin Petersonad9d48d2008-04-02 21:49:44 +0000612 orig_stdout = getattr(sys, stream_name)
613 setattr(sys, stream_name, io.StringIO())
Christian Heimes81ee3ef2008-05-04 22:42:01 +0000614 try:
615 yield getattr(sys, stream_name)
616 finally:
617 setattr(sys, stream_name, orig_stdout)
Benjamin Petersonad9d48d2008-04-02 21:49:44 +0000618
619def captured_stdout():
620 return captured_output("stdout")
Thomas Woutersed03b412007-08-28 21:37:11 +0000621
Benjamin Petersone549ead2009-03-28 21:42:05 +0000622def gc_collect():
623 """Force as many objects as possible to be collected.
624
625 In non-CPython implementations of Python, this is needed because timely
626 deallocation is not guaranteed by the garbage collector. (Even in CPython
627 this can be the case in case of reference cycles.) This means that __del__
628 methods may be called later than expected and weakrefs may remain alive for
629 longer than expected. This function tries its best to force all garbage
630 objects to disappear.
631 """
632 import gc
633 gc.collect()
634 gc.collect()
635 gc.collect()
636
Thomas Woutersed03b412007-08-28 21:37:11 +0000637
Fred Drakecd1b1dd2001-03-21 18:26:33 +0000638#=======================================================================
Thomas Wouters477c8d52006-05-27 19:21:47 +0000639# Decorator for running a function in a different locale, correctly resetting
640# it afterwards.
641
642def run_with_locale(catstr, *locales):
643 def decorator(func):
644 def inner(*args, **kwds):
645 try:
646 import locale
647 category = getattr(locale, catstr)
648 orig_locale = locale.setlocale(category)
649 except AttributeError:
650 # if the test author gives us an invalid category string
651 raise
652 except:
653 # cannot retrieve original locale, so do nothing
654 locale = orig_locale = None
655 else:
656 for loc in locales:
657 try:
658 locale.setlocale(category, loc)
659 break
660 except:
661 pass
662
663 # now run the function, resetting the locale on exceptions
664 try:
665 return func(*args, **kwds)
666 finally:
667 if locale and orig_locale:
668 locale.setlocale(category, orig_locale)
Neal Norwitz221085d2007-02-25 20:55:47 +0000669 inner.__name__ = func.__name__
Thomas Wouters477c8d52006-05-27 19:21:47 +0000670 inner.__doc__ = func.__doc__
671 return inner
672 return decorator
673
674#=======================================================================
Georg Brandldb028442008-02-05 20:48:58 +0000675# Big-memory-test support. Separate from 'resources' because memory use
676# should be configurable.
Thomas Wouters477c8d52006-05-27 19:21:47 +0000677
678# Some handy shorthands. Note that these are used for byte-limits as well
679# as size-limits, in the various bigmem tests
680_1M = 1024*1024
681_1G = 1024 * _1M
682_2G = 2 * _1G
Neal Norwitz3ce5d922008-08-24 07:08:55 +0000683_4G = 4 * _1G
Thomas Wouters477c8d52006-05-27 19:21:47 +0000684
Thomas Woutersd2cf20e2007-08-30 22:57:53 +0000685MAX_Py_ssize_t = sys.maxsize
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000686
Thomas Wouters477c8d52006-05-27 19:21:47 +0000687def set_memlimit(limit):
688 import re
689 global max_memuse
Neal Norwitz3ce5d922008-08-24 07:08:55 +0000690 global real_max_memuse
Thomas Wouters477c8d52006-05-27 19:21:47 +0000691 sizes = {
692 'k': 1024,
693 'm': _1M,
694 'g': _1G,
695 't': 1024*_1G,
696 }
697 m = re.match(r'(\d+(\.\d+)?) (K|M|G|T)b?$', limit,
698 re.IGNORECASE | re.VERBOSE)
699 if m is None:
700 raise ValueError('Invalid memory limit %r' % (limit,))
701 memlimit = int(float(m.group(1)) * sizes[m.group(3).lower()])
Neal Norwitz3ce5d922008-08-24 07:08:55 +0000702 real_max_memuse = memlimit
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000703 if memlimit > MAX_Py_ssize_t:
704 memlimit = MAX_Py_ssize_t
705 if memlimit < _2G - 1:
Thomas Wouters477c8d52006-05-27 19:21:47 +0000706 raise ValueError('Memory limit %r too low to be useful' % (limit,))
707 max_memuse = memlimit
708
709def bigmemtest(minsize, memuse, overhead=5*_1M):
710 """Decorator for bigmem tests.
711
712 'minsize' is the minimum useful size for the test (in arbitrary,
713 test-interpreted units.) 'memuse' is the number of 'bytes per size' for
714 the test, or a good estimate of it. 'overhead' specifies fixed overhead,
Christian Heimes33fe8092008-04-13 13:53:33 +0000715 independent of the testsize, and defaults to 5Mb.
Thomas Wouters477c8d52006-05-27 19:21:47 +0000716
717 The decorator tries to guess a good value for 'size' and passes it to
718 the decorated test function. If minsize * memuse is more than the
719 allowed memory use (as defined by max_memuse), the test is skipped.
720 Otherwise, minsize is adjusted upward to use up to max_memuse.
721 """
722 def decorator(f):
723 def wrapper(self):
Antoine Pitrou7cdb4952009-03-07 23:40:49 +0000724 # Retrieve values in case someone decided to adjust them
725 minsize = wrapper.minsize
726 memuse = wrapper.memuse
727 overhead = wrapper.overhead
Thomas Wouters477c8d52006-05-27 19:21:47 +0000728 if not max_memuse:
729 # If max_memuse is 0 (the default),
730 # we still want to run the tests with size set to a few kb,
731 # to make sure they work. We still want to avoid using
732 # too much memory, though, but we do that noisily.
733 maxsize = 5147
734 self.failIf(maxsize * memuse + overhead > 20 * _1M)
735 else:
736 maxsize = int((max_memuse - overhead) / memuse)
737 if maxsize < minsize:
738 # Really ought to print 'test skipped' or something
739 if verbose:
740 sys.stderr.write("Skipping %s because of memory "
741 "constraint\n" % (f.__name__,))
742 return
743 # Try to keep some breathing room in memory use
744 maxsize = max(maxsize - 50 * _1M, minsize)
745 return f(self, maxsize)
746 wrapper.minsize = minsize
747 wrapper.memuse = memuse
748 wrapper.overhead = overhead
749 return wrapper
750 return decorator
751
Neal Norwitz3ce5d922008-08-24 07:08:55 +0000752def precisionbigmemtest(size, memuse, overhead=5*_1M):
753 def decorator(f):
754 def wrapper(self):
Antoine Pitrou7cdb4952009-03-07 23:40:49 +0000755 size = wrapper.size
756 memuse = wrapper.memuse
757 overhead = wrapper.overhead
Neal Norwitz3ce5d922008-08-24 07:08:55 +0000758 if not real_max_memuse:
759 maxsize = 5147
760 else:
761 maxsize = size
762
763 if real_max_memuse and real_max_memuse < maxsize * memuse:
764 if verbose:
765 sys.stderr.write("Skipping %s because of memory "
766 "constraint\n" % (f.__name__,))
767 return
768
769 return f(self, maxsize)
770 wrapper.size = size
771 wrapper.memuse = memuse
772 wrapper.overhead = overhead
773 return wrapper
774 return decorator
775
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000776def bigaddrspacetest(f):
777 """Decorator for tests that fill the address space."""
778 def wrapper(self):
779 if max_memuse < MAX_Py_ssize_t:
780 if verbose:
781 sys.stderr.write("Skipping %s because of memory "
782 "constraint\n" % (f.__name__,))
783 else:
784 return f(self)
785 return wrapper
786
Thomas Wouters477c8d52006-05-27 19:21:47 +0000787#=======================================================================
Guido van Rossumd8faa362007-04-27 19:54:29 +0000788# unittest integration.
Fred Drakecd1b1dd2001-03-21 18:26:33 +0000789
Steve Purcell5ddd1a82001-03-22 08:45:36 +0000790class BasicTestRunner:
Fred Drakecd1b1dd2001-03-21 18:26:33 +0000791 def run(self, test):
Steve Purcell5ddd1a82001-03-22 08:45:36 +0000792 result = unittest.TestResult()
Fred Drakecd1b1dd2001-03-21 18:26:33 +0000793 test(result)
794 return result
795
Benjamin Petersone549ead2009-03-28 21:42:05 +0000796def _id(obj):
797 return obj
798
799def requires_resource(resource):
800 if resource_is_enabled(resource):
801 return _id
802 else:
803 return unittest.skip("resource {0!r} is not enabled".format(resource))
804
805def cpython_only(test):
806 """
807 Decorator for tests only applicable on CPython.
808 """
809 return impl_detail(cpython=True)(test)
810
811def impl_detail(msg=None, **guards):
812 if check_impl_detail(**guards):
813 return _id
814 if msg is None:
815 guardnames, default = _parse_guards(guards)
816 if default:
817 msg = "implementation detail not available on {0}"
818 else:
819 msg = "implementation detail specific to {0}"
820 guardnames = sorted(guardnames.keys())
821 msg = msg.format(' or '.join(guardnames))
822 return unittest.skip(msg)
823
824def _parse_guards(guards):
825 # Returns a tuple ({platform_name: run_me}, default_value)
826 if not guards:
827 return ({'cpython': True}, False)
Eric Smith886b40a2009-04-26 21:26:45 +0000828 is_true = list(guards.values())[0]
829 assert list(guards.values()) == [is_true] * len(guards) # all True or all False
Benjamin Petersone549ead2009-03-28 21:42:05 +0000830 return (guards, not is_true)
831
832# Use the following check to guard CPython's implementation-specific tests --
833# or to run them only on the implementation(s) guarded by the arguments.
834def check_impl_detail(**guards):
835 """This function returns True or False depending on the host platform.
836 Examples:
837 if check_impl_detail(): # only on CPython (default)
838 if check_impl_detail(jython=True): # only on Jython
839 if check_impl_detail(cpython=False): # everywhere except on CPython
840 """
841 guards, default = _parse_guards(guards)
842 return guards.get(platform.python_implementation().lower(), default)
843
844
Fred Drakecd1b1dd2001-03-21 18:26:33 +0000845
Guido van Rossumd8faa362007-04-27 19:54:29 +0000846def _run_suite(suite):
Barry Warsawc88425e2001-09-20 06:31:22 +0000847 """Run tests from a unittest.TestSuite-derived class."""
Fred Drakecd1b1dd2001-03-21 18:26:33 +0000848 if verbose:
Fred Drake84a59342001-03-23 04:21:17 +0000849 runner = unittest.TextTestRunner(sys.stdout, verbosity=2)
Fred Drakecd1b1dd2001-03-21 18:26:33 +0000850 else:
Steve Purcell5ddd1a82001-03-22 08:45:36 +0000851 runner = BasicTestRunner()
Fred Drakecd1b1dd2001-03-21 18:26:33 +0000852
Steve Purcell5ddd1a82001-03-22 08:45:36 +0000853 result = runner.run(suite)
854 if not result.wasSuccessful():
Fred Drake14f6c182001-07-16 18:51:32 +0000855 if len(result.errors) == 1 and not result.failures:
856 err = result.errors[0][1]
857 elif len(result.failures) == 1 and not result.errors:
858 err = result.failures[0][1]
859 else:
Guido van Rossum8ce8a782007-11-01 19:42:39 +0000860 err = "errors occurred; run in verbose mode for details"
Tim Peters2d84f2c2001-09-08 03:37:56 +0000861 raise TestFailed(err)
Tim Petersa0a62222001-09-09 06:12:01 +0000862
Barry Warsawc10d6902001-09-20 06:30:41 +0000863
Walter Dörwald21d3a322003-05-01 17:45:56 +0000864def run_unittest(*classes):
865 """Run tests from unittest.TestCase-derived classes."""
Guido van Rossumd8faa362007-04-27 19:54:29 +0000866 valid_types = (unittest.TestSuite, unittest.TestCase)
Raymond Hettinger9dcbbea2003-04-27 07:54:23 +0000867 suite = unittest.TestSuite()
Walter Dörwald21d3a322003-05-01 17:45:56 +0000868 for cls in classes:
Guido van Rossum3172c5d2007-10-16 18:12:55 +0000869 if isinstance(cls, str):
Guido van Rossumd8faa362007-04-27 19:54:29 +0000870 if cls in sys.modules:
871 suite.addTest(unittest.findTestCases(sys.modules[cls]))
872 else:
873 raise ValueError("str arguments must be keys in sys.modules")
874 elif isinstance(cls, valid_types):
Raymond Hettinger21d99872003-07-16 02:59:32 +0000875 suite.addTest(cls)
876 else:
877 suite.addTest(unittest.makeSuite(cls))
Guido van Rossumd8faa362007-04-27 19:54:29 +0000878 _run_suite(suite)
Raymond Hettinger9dcbbea2003-04-27 07:54:23 +0000879
Barry Warsawc10d6902001-09-20 06:30:41 +0000880
Tim Petersa0a62222001-09-09 06:12:01 +0000881#=======================================================================
882# doctest driver.
883
884def run_doctest(module, verbosity=None):
Tim Peters17111f32001-10-03 04:08:26 +0000885 """Run doctest on the given module. Return (#failures, #tests).
Tim Petersa0a62222001-09-09 06:12:01 +0000886
887 If optional argument verbosity is not specified (or is None), pass
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000888 support's belief about verbosity on to doctest. Else doctest's
Tim Petersbea3fb82001-09-10 01:39:21 +0000889 usual behavior is used (it searches sys.argv for -v).
Tim Petersa0a62222001-09-09 06:12:01 +0000890 """
891
892 import doctest
893
894 if verbosity is None:
895 verbosity = verbose
896 else:
897 verbosity = None
898
Tim Peters342ca752001-09-25 19:13:20 +0000899 # Direct doctest output (normally just errors) to real stdout; doctest
900 # output shouldn't be compared by regrtest.
901 save_stdout = sys.stdout
Tim Peters8dee8092001-09-25 20:05:11 +0000902 sys.stdout = get_original_stdout()
Tim Peters342ca752001-09-25 19:13:20 +0000903 try:
904 f, t = doctest.testmod(module, verbose=verbosity)
905 if f:
906 raise TestFailed("%d of %d doctests failed" % (f, t))
907 finally:
908 sys.stdout = save_stdout
Raymond Hettinger35b34bd2003-05-17 00:58:33 +0000909 if verbose:
Georg Brandldb028442008-02-05 20:48:58 +0000910 print('doctest (%s) ... %d tests with zero failures' %
911 (module.__name__, t))
Raymond Hettinger35b34bd2003-05-17 00:58:33 +0000912 return f, t
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000913
914#=======================================================================
915# Threading support to prevent reporting refleaks when running regrtest.py -R
916
917def threading_setup():
918 import threading
919 return len(threading._active), len(threading._limbo)
920
921def threading_cleanup(num_active, num_limbo):
922 import threading
923 import time
924
925 _MAX_COUNT = 10
926 count = 0
927 while len(threading._active) != num_active and count < _MAX_COUNT:
928 count += 1
929 time.sleep(0.1)
930
931 count = 0
932 while len(threading._limbo) != num_limbo and count < _MAX_COUNT:
933 count += 1
934 time.sleep(0.1)
935
936def reap_children():
937 """Use this function at the end of test_main() whenever sub-processes
938 are started. This will help ensure that no extra children (zombies)
939 stick around to hog resources and create problems when looking
940 for refleaks.
941 """
942
943 # Reap all our dead child processes so we don't leave zombies around.
944 # These hog resources and might be causing some of the buildbots to die.
Thomas Wouters0e3f5912006-08-11 14:57:12 +0000945 if hasattr(os, 'waitpid'):
946 any_process = -1
947 while True:
948 try:
949 # This will raise an exception on Windows. That's ok.
950 pid, status = os.waitpid(any_process, os.WNOHANG)
951 if pid == 0:
952 break
953 except:
954 break