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