blob: 3a3f6a999ce0af132c8b7103b95dbd7d41d3817e [file] [log] [blame]
Guido van Rossum0e548712002-08-09 16:14:33 +00001# tempfile.py unit tests.
Tim Petersc57a2852001-10-29 21:46:08 +00002import tempfile
Serhiy Storchaka7451a722013-02-09 22:25:49 +02003import errno
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +02004import io
Guido van Rossum0e548712002-08-09 16:14:33 +00005import os
Anthony Sottile370138b2019-09-09 08:54:34 -07006import pathlib
Guido van Rossum0e548712002-08-09 16:14:33 +00007import sys
8import re
Guido van Rossum0e548712002-08-09 16:14:33 +00009import warnings
Eli Benderskyaa04f9a2013-09-13 05:28:20 -070010import contextlib
Anthony Sottile8377cd42019-02-25 14:32:27 -080011import stat
Guido van Rossum48b069a2020-04-07 09:50:06 -070012import types
Antoine Pitrou17c93262013-12-21 22:14:56 +010013import weakref
Victor Stinner1f99f9d2014-03-25 09:18:04 +010014from unittest import mock
Tim Petersc57a2852001-10-29 21:46:08 +000015
Guido van Rossum0e548712002-08-09 16:14:33 +000016import unittest
Berker Peksagce643912015-05-06 06:33:17 +030017from test import support
Hai Shi0c4f0f32020-06-30 21:46:31 +080018from test.support import os_helper
Berker Peksagce643912015-05-06 06:33:17 +030019from test.support import script_helper
Hai Shi0c4f0f32020-06-30 21:46:31 +080020from test.support import warnings_helper
Guido van Rossum0e548712002-08-09 16:14:33 +000021
Fred Drake7633d232002-10-17 22:09:03 +000022
Guido van Rossum0e548712002-08-09 16:14:33 +000023has_textmode = (tempfile._text_openflags != tempfile._bin_openflags)
Guido van Rossum78741062002-08-17 11:41:01 +000024has_spawnl = hasattr(os, 'spawnl')
Guido van Rossum0e548712002-08-09 16:14:33 +000025
Neal Norwitz68ee0122002-08-16 19:28:59 +000026# TEST_FILES may need to be tweaked for systems depending on the maximum
27# number of files that can be opened at one time (see ulimit -n)
Victor Stinner9c3de4a2011-08-17 20:49:41 +020028if sys.platform.startswith('openbsd'):
Martin v. Löwis99968282004-09-15 06:02:54 +000029 TEST_FILES = 48
Jack Jansence921472003-01-08 16:30:34 +000030else:
31 TEST_FILES = 100
Neal Norwitz68ee0122002-08-16 19:28:59 +000032
Guido van Rossum0e548712002-08-09 16:14:33 +000033# This is organized as one test for each chunk of code in tempfile.py,
34# in order of their appearance in the file. Testing which requires
35# threads is not done here.
36
Gregory P. Smithad577b92015-05-22 16:18:14 -070037class TestLowLevelInternals(unittest.TestCase):
38 def test_infer_return_type_singles(self):
39 self.assertIs(str, tempfile._infer_return_type(''))
40 self.assertIs(bytes, tempfile._infer_return_type(b''))
41 self.assertIs(str, tempfile._infer_return_type(None))
42
43 def test_infer_return_type_multiples(self):
44 self.assertIs(str, tempfile._infer_return_type('', ''))
45 self.assertIs(bytes, tempfile._infer_return_type(b'', b''))
46 with self.assertRaises(TypeError):
47 tempfile._infer_return_type('', b'')
48 with self.assertRaises(TypeError):
49 tempfile._infer_return_type(b'', '')
50
51 def test_infer_return_type_multiples_and_none(self):
52 self.assertIs(str, tempfile._infer_return_type(None, ''))
53 self.assertIs(str, tempfile._infer_return_type('', None))
54 self.assertIs(str, tempfile._infer_return_type(None, None))
55 self.assertIs(bytes, tempfile._infer_return_type(b'', None))
56 self.assertIs(bytes, tempfile._infer_return_type(None, b''))
57 with self.assertRaises(TypeError):
58 tempfile._infer_return_type('', None, b'')
59 with self.assertRaises(TypeError):
60 tempfile._infer_return_type(b'', None, '')
61
Anthony Sottile370138b2019-09-09 08:54:34 -070062 def test_infer_return_type_pathlib(self):
63 self.assertIs(str, tempfile._infer_return_type(pathlib.Path('/')))
64
Gregory P. Smithad577b92015-05-22 16:18:14 -070065
Guido van Rossum0e548712002-08-09 16:14:33 +000066# Common functionality.
Gregory P. Smithad577b92015-05-22 16:18:14 -070067
Antoine Pitroueab2a502012-03-10 16:34:40 +010068class BaseTestCase(unittest.TestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +000069
Victor Stinner97869102013-08-14 01:28:28 +020070 str_check = re.compile(r"^[a-z0-9_-]{8}$")
Gregory P. Smithad577b92015-05-22 16:18:14 -070071 b_check = re.compile(br"^[a-z0-9_-]{8}$")
Guido van Rossum0e548712002-08-09 16:14:33 +000072
Brett Cannone1adece2010-03-20 22:19:55 +000073 def setUp(self):
Hai Shi0c4f0f32020-06-30 21:46:31 +080074 self._warnings_manager = warnings_helper.check_warnings()
Brett Cannone1adece2010-03-20 22:19:55 +000075 self._warnings_manager.__enter__()
76 warnings.filterwarnings("ignore", category=RuntimeWarning,
77 message="mktemp", module=__name__)
78
79 def tearDown(self):
80 self._warnings_manager.__exit__(None, None, None)
81
Guido van Rossum0e548712002-08-09 16:14:33 +000082 def nameCheck(self, name, dir, pre, suf):
83 (ndir, nbase) = os.path.split(name)
84 npre = nbase[:len(pre)]
85 nsuf = nbase[len(nbase)-len(suf):]
86
Gregory P. Smithad577b92015-05-22 16:18:14 -070087 if dir is not None:
Anthony Sottile370138b2019-09-09 08:54:34 -070088 self.assertIs(
89 type(name),
90 str
91 if type(dir) is str or isinstance(dir, os.PathLike) else
92 bytes,
93 "unexpected return type",
94 )
Gregory P. Smithad577b92015-05-22 16:18:14 -070095 if pre is not None:
96 self.assertIs(type(name), str if type(pre) is str else bytes,
97 "unexpected return type")
98 if suf is not None:
99 self.assertIs(type(name), str if type(suf) is str else bytes,
100 "unexpected return type")
101 if (dir, pre, suf) == (None, None, None):
102 self.assertIs(type(name), str, "default return type must be str")
103
Martin v. Löwisd6625482003-10-12 17:37:01 +0000104 # check for equality of the absolute paths!
105 self.assertEqual(os.path.abspath(ndir), os.path.abspath(dir),
Gregory P. Smithad577b92015-05-22 16:18:14 -0700106 "file %r not in directory %r" % (name, dir))
Guido van Rossum0e548712002-08-09 16:14:33 +0000107 self.assertEqual(npre, pre,
Gregory P. Smithad577b92015-05-22 16:18:14 -0700108 "file %r does not begin with %r" % (nbase, pre))
Guido van Rossum0e548712002-08-09 16:14:33 +0000109 self.assertEqual(nsuf, suf,
Gregory P. Smithad577b92015-05-22 16:18:14 -0700110 "file %r does not end with %r" % (nbase, suf))
Guido van Rossum0e548712002-08-09 16:14:33 +0000111
112 nbase = nbase[len(pre):len(nbase)-len(suf)]
Gregory P. Smithad577b92015-05-22 16:18:14 -0700113 check = self.str_check if isinstance(nbase, str) else self.b_check
114 self.assertTrue(check.match(nbase),
115 "random characters %r do not match %r"
116 % (nbase, check.pattern))
Guido van Rossum0e548712002-08-09 16:14:33 +0000117
Guido van Rossum0e548712002-08-09 16:14:33 +0000118
Antoine Pitroueab2a502012-03-10 16:34:40 +0100119class TestExports(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000120 def test_exports(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000121 # There are no surprising symbols in the tempfile module
Guido van Rossum0e548712002-08-09 16:14:33 +0000122 dict = tempfile.__dict__
123
124 expected = {
125 "NamedTemporaryFile" : 1,
126 "TemporaryFile" : 1,
127 "mkstemp" : 1,
128 "mkdtemp" : 1,
129 "mktemp" : 1,
130 "TMP_MAX" : 1,
131 "gettempprefix" : 1,
Gregory P. Smithad577b92015-05-22 16:18:14 -0700132 "gettempprefixb" : 1,
Guido van Rossum0e548712002-08-09 16:14:33 +0000133 "gettempdir" : 1,
Gregory P. Smithad577b92015-05-22 16:18:14 -0700134 "gettempdirb" : 1,
Guido van Rossum0e548712002-08-09 16:14:33 +0000135 "tempdir" : 1,
Guido van Rossumd8faa362007-04-27 19:54:29 +0000136 "template" : 1,
Nick Coghlan543af752010-10-24 11:23:25 +0000137 "SpooledTemporaryFile" : 1,
138 "TemporaryDirectory" : 1,
Guido van Rossum0e548712002-08-09 16:14:33 +0000139 }
140
141 unexp = []
142 for key in dict:
143 if key[0] != '_' and key not in expected:
144 unexp.append(key)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000145 self.assertTrue(len(unexp) == 0,
Guido van Rossum0e548712002-08-09 16:14:33 +0000146 "unexpected keys: %s" % unexp)
147
Guido van Rossum0e548712002-08-09 16:14:33 +0000148
Antoine Pitroueab2a502012-03-10 16:34:40 +0100149class TestRandomNameSequence(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000150 """Test the internal iterator object _RandomNameSequence."""
151
152 def setUp(self):
153 self.r = tempfile._RandomNameSequence()
Brett Cannone1adece2010-03-20 22:19:55 +0000154 super().setUp()
Guido van Rossum0e548712002-08-09 16:14:33 +0000155
Inada Naokid2810052020-11-01 20:02:03 +0900156 def test_get_eight_char_str(self):
157 # _RandomNameSequence returns a eight-character string
Georg Brandla18af4e2007-04-21 15:47:16 +0000158 s = next(self.r)
Guido van Rossum0e548712002-08-09 16:14:33 +0000159 self.nameCheck(s, '', '', '')
160
161 def test_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000162 # _RandomNameSequence returns no duplicate strings (stochastic)
Guido van Rossum0e548712002-08-09 16:14:33 +0000163
164 dict = {}
165 r = self.r
Guido van Rossum805365e2007-05-07 22:24:25 +0000166 for i in range(TEST_FILES):
Georg Brandla18af4e2007-04-21 15:47:16 +0000167 s = next(r)
Guido van Rossum0e548712002-08-09 16:14:33 +0000168 self.nameCheck(s, '', '', '')
Ezio Melottib58e0bd2010-01-23 15:40:09 +0000169 self.assertNotIn(s, dict)
Guido van Rossum0e548712002-08-09 16:14:33 +0000170 dict[s] = 1
171
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000172 def supports_iter(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000173 # _RandomNameSequence supports the iterator protocol
Guido van Rossum0e548712002-08-09 16:14:33 +0000174
175 i = 0
176 r = self.r
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100177 for s in r:
178 i += 1
179 if i == 20:
180 break
Guido van Rossum0e548712002-08-09 16:14:33 +0000181
Antoine Pitrou4558bad2011-11-25 21:28:15 +0100182 @unittest.skipUnless(hasattr(os, 'fork'),
183 "os.fork is required for this test")
184 def test_process_awareness(self):
185 # ensure that the random source differs between
186 # child and parent.
187 read_fd, write_fd = os.pipe()
188 pid = None
189 try:
190 pid = os.fork()
191 if not pid:
Victor Stinner6c8c2942017-08-10 13:05:06 +0200192 # child process
Antoine Pitrou4558bad2011-11-25 21:28:15 +0100193 os.close(read_fd)
194 os.write(write_fd, next(self.r).encode("ascii"))
195 os.close(write_fd)
196 # bypass the normal exit handlers- leave those to
197 # the parent.
198 os._exit(0)
Victor Stinner6c8c2942017-08-10 13:05:06 +0200199
200 # parent process
Antoine Pitrou4558bad2011-11-25 21:28:15 +0100201 parent_value = next(self.r)
202 child_value = os.read(read_fd, len(parent_value)).decode("ascii")
203 finally:
204 if pid:
Victor Stinner278c1e12020-03-31 20:08:12 +0200205 support.wait_process(pid, exitcode=0)
Victor Stinner6c8c2942017-08-10 13:05:06 +0200206
Antoine Pitrou4558bad2011-11-25 21:28:15 +0100207 os.close(read_fd)
208 os.close(write_fd)
209 self.assertNotEqual(child_value, parent_value)
210
211
Guido van Rossum0e548712002-08-09 16:14:33 +0000212
Antoine Pitroueab2a502012-03-10 16:34:40 +0100213class TestCandidateTempdirList(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000214 """Test the internal function _candidate_tempdir_list."""
215
216 def test_nonempty_list(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000217 # _candidate_tempdir_list returns a nonempty list of strings
Guido van Rossum0e548712002-08-09 16:14:33 +0000218
219 cand = tempfile._candidate_tempdir_list()
220
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000221 self.assertFalse(len(cand) == 0)
Guido van Rossum0e548712002-08-09 16:14:33 +0000222 for c in cand:
Ezio Melottie9615932010-01-24 19:26:24 +0000223 self.assertIsInstance(c, str)
Guido van Rossum0e548712002-08-09 16:14:33 +0000224
225 def test_wanted_dirs(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000226 # _candidate_tempdir_list contains the expected directories
Guido van Rossum0e548712002-08-09 16:14:33 +0000227
228 # Make sure the interesting environment variables are all set.
Hai Shi0c4f0f32020-06-30 21:46:31 +0800229 with os_helper.EnvironmentVarGuard() as env:
Guido van Rossum0e548712002-08-09 16:14:33 +0000230 for envname in 'TMPDIR', 'TEMP', 'TMP':
231 dirname = os.getenv(envname)
232 if not dirname:
Walter Dörwald155374d2009-05-01 19:58:58 +0000233 env[envname] = os.path.abspath(envname)
Guido van Rossum0e548712002-08-09 16:14:33 +0000234
235 cand = tempfile._candidate_tempdir_list()
236
237 for envname in 'TMPDIR', 'TEMP', 'TMP':
238 dirname = os.getenv(envname)
239 if not dirname: raise ValueError
Benjamin Peterson577473f2010-01-19 00:09:57 +0000240 self.assertIn(dirname, cand)
Guido van Rossum0e548712002-08-09 16:14:33 +0000241
242 try:
243 dirname = os.getcwd()
Andrew Svetlovad28c7f2012-12-18 22:02:39 +0200244 except (AttributeError, OSError):
Guido van Rossum0e548712002-08-09 16:14:33 +0000245 dirname = os.curdir
246
Benjamin Peterson577473f2010-01-19 00:09:57 +0000247 self.assertIn(dirname, cand)
Guido van Rossum0e548712002-08-09 16:14:33 +0000248
249 # Not practical to try to verify the presence of OS-specific
250 # paths in this list.
Guido van Rossum0e548712002-08-09 16:14:33 +0000251
Guido van Rossum0e548712002-08-09 16:14:33 +0000252
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200253# We test _get_default_tempdir some more by testing gettempdir.
Guido van Rossum0e548712002-08-09 16:14:33 +0000254
Serhiy Storchakaff7fef92013-02-13 00:37:29 +0200255class TestGetDefaultTempdir(BaseTestCase):
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200256 """Test _get_default_tempdir()."""
257
258 def test_no_files_left_behind(self):
259 # use a private empty directory
260 with tempfile.TemporaryDirectory() as our_temp_directory:
261 # force _get_default_tempdir() to consider our empty directory
262 def our_candidate_list():
263 return [our_temp_directory]
264
265 with support.swap_attr(tempfile, "_candidate_tempdir_list",
266 our_candidate_list):
267 # verify our directory is empty after _get_default_tempdir()
268 tempfile._get_default_tempdir()
269 self.assertEqual(os.listdir(our_temp_directory), [])
270
271 def raise_OSError(*args, **kwargs):
Serhiy Storchakaff7fef92013-02-13 00:37:29 +0200272 raise OSError()
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200273
274 with support.swap_attr(io, "open", raise_OSError):
275 # test again with failing io.open()
Serhiy Storchakaff7fef92013-02-13 00:37:29 +0200276 with self.assertRaises(FileNotFoundError):
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200277 tempfile._get_default_tempdir()
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200278 self.assertEqual(os.listdir(our_temp_directory), [])
279
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200280 def bad_writer(*args, **kwargs):
Serhiy Storchakad1a1def2017-04-28 19:17:26 +0300281 fp = orig_open(*args, **kwargs)
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200282 fp.write = raise_OSError
283 return fp
284
Serhiy Storchakad1a1def2017-04-28 19:17:26 +0300285 with support.swap_attr(io, "open", bad_writer) as orig_open:
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200286 # test again with failing write()
Serhiy Storchakaff7fef92013-02-13 00:37:29 +0200287 with self.assertRaises(FileNotFoundError):
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200288 tempfile._get_default_tempdir()
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200289 self.assertEqual(os.listdir(our_temp_directory), [])
Guido van Rossum0e548712002-08-09 16:14:33 +0000290
291
Antoine Pitroueab2a502012-03-10 16:34:40 +0100292class TestGetCandidateNames(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000293 """Test the internal function _get_candidate_names."""
294
295 def test_retval(self):
Victor Stinner1e62bf12017-04-19 22:59:51 +0200296 # _get_candidate_names returns a _RandomNameSequence object
Guido van Rossum0e548712002-08-09 16:14:33 +0000297 obj = tempfile._get_candidate_names()
Victor Stinner1e62bf12017-04-19 22:59:51 +0200298 self.assertIsInstance(obj, tempfile._RandomNameSequence)
Guido van Rossum0e548712002-08-09 16:14:33 +0000299
300 def test_same_thing(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000301 # _get_candidate_names always returns the same object
Guido van Rossum0e548712002-08-09 16:14:33 +0000302 a = tempfile._get_candidate_names()
303 b = tempfile._get_candidate_names()
304
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000305 self.assertTrue(a is b)
Guido van Rossum0e548712002-08-09 16:14:33 +0000306
Guido van Rossum0e548712002-08-09 16:14:33 +0000307
Eli Benderskyaa04f9a2013-09-13 05:28:20 -0700308@contextlib.contextmanager
309def _inside_empty_temp_dir():
310 dir = tempfile.mkdtemp()
311 try:
312 with support.swap_attr(tempfile, 'tempdir', dir):
313 yield
314 finally:
Hai Shi0c4f0f32020-06-30 21:46:31 +0800315 os_helper.rmtree(dir)
Eli Benderskyaa04f9a2013-09-13 05:28:20 -0700316
317
318def _mock_candidate_names(*names):
319 return support.swap_attr(tempfile,
320 '_get_candidate_names',
321 lambda: iter(names))
322
323
Serhiy Storchaka5d6b7b12015-05-20 00:11:48 +0300324class TestBadTempdir:
325
326 def test_read_only_directory(self):
327 with _inside_empty_temp_dir():
328 oldmode = mode = os.stat(tempfile.tempdir).st_mode
329 mode &= ~(stat.S_IWUSR | stat.S_IWGRP | stat.S_IWOTH)
330 os.chmod(tempfile.tempdir, mode)
331 try:
332 if os.access(tempfile.tempdir, os.W_OK):
333 self.skipTest("can't set the directory read-only")
334 with self.assertRaises(PermissionError):
335 self.make_temp()
336 self.assertEqual(os.listdir(tempfile.tempdir), [])
337 finally:
338 os.chmod(tempfile.tempdir, oldmode)
339
340 def test_nonexisting_directory(self):
341 with _inside_empty_temp_dir():
342 tempdir = os.path.join(tempfile.tempdir, 'nonexistent')
343 with support.swap_attr(tempfile, 'tempdir', tempdir):
344 with self.assertRaises(FileNotFoundError):
345 self.make_temp()
346
347 def test_non_directory(self):
348 with _inside_empty_temp_dir():
349 tempdir = os.path.join(tempfile.tempdir, 'file')
350 open(tempdir, 'wb').close()
351 with support.swap_attr(tempfile, 'tempdir', tempdir):
352 with self.assertRaises((NotADirectoryError, FileNotFoundError)):
353 self.make_temp()
354
355
356class TestMkstempInner(TestBadTempdir, BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000357 """Test the internal function _mkstemp_inner."""
358
359 class mkstemped:
360 _bflags = tempfile._bin_openflags
361 _tflags = tempfile._text_openflags
362 _close = os.close
363 _unlink = os.unlink
364
365 def __init__(self, dir, pre, suf, bin):
366 if bin: flags = self._bflags
367 else: flags = self._tflags
368
Gregory P. Smithad577b92015-05-22 16:18:14 -0700369 output_type = tempfile._infer_return_type(dir, pre, suf)
370 (self.fd, self.name) = tempfile._mkstemp_inner(dir, pre, suf, flags, output_type)
Guido van Rossum0e548712002-08-09 16:14:33 +0000371
372 def write(self, str):
373 os.write(self.fd, str)
374
375 def __del__(self):
376 self._close(self.fd)
377 self._unlink(self.name)
Tim Petersa0d55de2002-08-09 18:01:01 +0000378
Gregory P. Smithad577b92015-05-22 16:18:14 -0700379 def do_create(self, dir=None, pre=None, suf=None, bin=1):
380 output_type = tempfile._infer_return_type(dir, pre, suf)
Guido van Rossum0e548712002-08-09 16:14:33 +0000381 if dir is None:
Gregory P. Smithad577b92015-05-22 16:18:14 -0700382 if output_type is str:
383 dir = tempfile.gettempdir()
384 else:
385 dir = tempfile.gettempdirb()
386 if pre is None:
387 pre = output_type()
388 if suf is None:
389 suf = output_type()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100390 file = self.mkstemped(dir, pre, suf, bin)
Guido van Rossum0e548712002-08-09 16:14:33 +0000391
392 self.nameCheck(file.name, dir, pre, suf)
393 return file
394
395 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000396 # _mkstemp_inner can create files
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000397 self.do_create().write(b"blat")
398 self.do_create(pre="a").write(b"blat")
399 self.do_create(suf="b").write(b"blat")
400 self.do_create(pre="a", suf="b").write(b"blat")
401 self.do_create(pre="aa", suf=".txt").write(b"blat")
Guido van Rossum0e548712002-08-09 16:14:33 +0000402
Gregory P. Smithad577b92015-05-22 16:18:14 -0700403 def test_basic_with_bytes_names(self):
404 # _mkstemp_inner can create files when given name parts all
405 # specified as bytes.
406 dir_b = tempfile.gettempdirb()
407 self.do_create(dir=dir_b, suf=b"").write(b"blat")
408 self.do_create(dir=dir_b, pre=b"a").write(b"blat")
409 self.do_create(dir=dir_b, suf=b"b").write(b"blat")
410 self.do_create(dir=dir_b, pre=b"a", suf=b"b").write(b"blat")
411 self.do_create(dir=dir_b, pre=b"aa", suf=b".txt").write(b"blat")
412 # Can't mix str & binary types in the args.
413 with self.assertRaises(TypeError):
414 self.do_create(dir="", suf=b"").write(b"blat")
415 with self.assertRaises(TypeError):
416 self.do_create(dir=dir_b, pre="").write(b"blat")
417 with self.assertRaises(TypeError):
418 self.do_create(dir=dir_b, pre=b"", suf="").write(b"blat")
419
Guido van Rossum0e548712002-08-09 16:14:33 +0000420 def test_basic_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000421 # _mkstemp_inner can create many files (stochastic)
Guido van Rossum805365e2007-05-07 22:24:25 +0000422 extant = list(range(TEST_FILES))
Guido van Rossum0e548712002-08-09 16:14:33 +0000423 for i in extant:
424 extant[i] = self.do_create(pre="aa")
425
426 def test_choose_directory(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000427 # _mkstemp_inner can create files in a user-selected directory
Guido van Rossum0e548712002-08-09 16:14:33 +0000428 dir = tempfile.mkdtemp()
429 try:
Antoine Pitrou9cadb1b2008-09-15 23:02:56 +0000430 self.do_create(dir=dir).write(b"blat")
Anthony Sottile370138b2019-09-09 08:54:34 -0700431 self.do_create(dir=pathlib.Path(dir)).write(b"blat")
Guido van Rossum0e548712002-08-09 16:14:33 +0000432 finally:
433 os.rmdir(dir)
434
435 def test_file_mode(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000436 # _mkstemp_inner creates files with the proper mode
Guido van Rossum0e548712002-08-09 16:14:33 +0000437
438 file = self.do_create()
439 mode = stat.S_IMODE(os.stat(file.name).st_mode)
Guido van Rossumcd16bf62007-06-13 18:07:49 +0000440 expected = 0o600
Jesus Cea4791a242012-10-05 03:15:39 +0200441 if sys.platform == 'win32':
Tim Petersca3ac7f2002-08-09 18:13:51 +0000442 # There's no distinction among 'user', 'group' and 'world';
443 # replicate the 'user' bits.
444 user = expected >> 6
445 expected = user * (1 + 8 + 64)
446 self.assertEqual(mode, expected)
Guido van Rossum0e548712002-08-09 16:14:33 +0000447
Zachary Ware9fe6d862013-12-08 00:20:35 -0600448 @unittest.skipUnless(has_spawnl, 'os.spawnl not available')
Guido van Rossum0e548712002-08-09 16:14:33 +0000449 def test_noinherit(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000450 # _mkstemp_inner file handles are not inherited by child processes
Guido van Rossum0e548712002-08-09 16:14:33 +0000451
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000452 if support.verbose:
Guido van Rossum78741062002-08-17 11:41:01 +0000453 v="v"
454 else:
455 v="q"
456
Guido van Rossum0e548712002-08-09 16:14:33 +0000457 file = self.do_create()
Victor Stinnerdaf45552013-08-28 00:53:59 +0200458 self.assertEqual(os.get_inheritable(file.fd), False)
Guido van Rossum78741062002-08-17 11:41:01 +0000459 fd = "%d" % file.fd
460
461 try:
462 me = __file__
463 except NameError:
464 me = sys.argv[0]
Guido van Rossum0e548712002-08-09 16:14:33 +0000465
466 # We have to exec something, so that FD_CLOEXEC will take
Guido van Rossum78741062002-08-17 11:41:01 +0000467 # effect. The core of this test is therefore in
468 # tf_inherit_check.py, which see.
469 tester = os.path.join(os.path.dirname(os.path.abspath(me)),
470 "tf_inherit_check.py")
Guido van Rossum0e548712002-08-09 16:14:33 +0000471
Martin v. Löwisd4210bc2003-10-23 15:55:28 +0000472 # On Windows a spawn* /path/ with embedded spaces shouldn't be quoted,
473 # but an arg with embedded spaces should be decorated with double
474 # quotes on each end
Jesus Cea4791a242012-10-05 03:15:39 +0200475 if sys.platform == 'win32':
Martin v. Löwisd4210bc2003-10-23 15:55:28 +0000476 decorated = '"%s"' % sys.executable
477 tester = '"%s"' % tester
478 else:
479 decorated = sys.executable
480
481 retval = os.spawnl(os.P_WAIT, sys.executable, decorated, tester, v, fd)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000482 self.assertFalse(retval < 0,
Guido van Rossum78741062002-08-17 11:41:01 +0000483 "child process caught fatal signal %d" % -retval)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000484 self.assertFalse(retval > 0, "child process reports failure %d"%retval)
Guido van Rossum0e548712002-08-09 16:14:33 +0000485
Zachary Ware9fe6d862013-12-08 00:20:35 -0600486 @unittest.skipUnless(has_textmode, "text mode not available")
Guido van Rossum0e548712002-08-09 16:14:33 +0000487 def test_textmode(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000488 # _mkstemp_inner can create files in text mode
Guido van Rossum0e548712002-08-09 16:14:33 +0000489
Amaury Forgeot d'Arc7d0bddd2009-11-30 00:08:56 +0000490 # A text file is truncated at the first Ctrl+Z byte
491 f = self.do_create(bin=0)
492 f.write(b"blat\x1a")
493 f.write(b"extra\n")
494 os.lseek(f.fd, 0, os.SEEK_SET)
Ezio Melottib3aedd42010-11-20 19:04:17 +0000495 self.assertEqual(os.read(f.fd, 20), b"blat")
Guido van Rossum0e548712002-08-09 16:14:33 +0000496
Serhiy Storchaka5d6b7b12015-05-20 00:11:48 +0300497 def make_temp(self):
Eli Benderskyaa04f9a2013-09-13 05:28:20 -0700498 return tempfile._mkstemp_inner(tempfile.gettempdir(),
Gregory P. Smithad577b92015-05-22 16:18:14 -0700499 tempfile.gettempprefix(),
Eli Benderskyaa04f9a2013-09-13 05:28:20 -0700500 '',
Gregory P. Smithad577b92015-05-22 16:18:14 -0700501 tempfile._bin_openflags,
502 str)
Eli Benderskyaa04f9a2013-09-13 05:28:20 -0700503
504 def test_collision_with_existing_file(self):
505 # _mkstemp_inner tries another name when a file with
506 # the chosen name already exists
507 with _inside_empty_temp_dir(), \
508 _mock_candidate_names('aaa', 'aaa', 'bbb'):
Serhiy Storchaka5d6b7b12015-05-20 00:11:48 +0300509 (fd1, name1) = self.make_temp()
Eli Benderskyaa04f9a2013-09-13 05:28:20 -0700510 os.close(fd1)
511 self.assertTrue(name1.endswith('aaa'))
512
Serhiy Storchaka5d6b7b12015-05-20 00:11:48 +0300513 (fd2, name2) = self.make_temp()
Eli Benderskyaa04f9a2013-09-13 05:28:20 -0700514 os.close(fd2)
515 self.assertTrue(name2.endswith('bbb'))
516
Eli Benderskyf315df32013-09-06 06:11:19 -0700517 def test_collision_with_existing_directory(self):
518 # _mkstemp_inner tries another name when a directory with
519 # the chosen name already exists
Eli Benderskyaa04f9a2013-09-13 05:28:20 -0700520 with _inside_empty_temp_dir(), \
521 _mock_candidate_names('aaa', 'aaa', 'bbb'):
522 dir = tempfile.mkdtemp()
523 self.assertTrue(dir.endswith('aaa'))
Eli Benderskyf315df32013-09-06 06:11:19 -0700524
Serhiy Storchaka5d6b7b12015-05-20 00:11:48 +0300525 (fd, name) = self.make_temp()
Eli Benderskyaa04f9a2013-09-13 05:28:20 -0700526 os.close(fd)
527 self.assertTrue(name.endswith('bbb'))
Eli Benderskyf315df32013-09-06 06:11:19 -0700528
Guido van Rossum0e548712002-08-09 16:14:33 +0000529
Antoine Pitroueab2a502012-03-10 16:34:40 +0100530class TestGetTempPrefix(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000531 """Test gettempprefix()."""
532
533 def test_sane_template(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000534 # gettempprefix returns a nonempty prefix string
Guido van Rossum0e548712002-08-09 16:14:33 +0000535 p = tempfile.gettempprefix()
536
Ezio Melottie9615932010-01-24 19:26:24 +0000537 self.assertIsInstance(p, str)
Gregory P. Smithad577b92015-05-22 16:18:14 -0700538 self.assertGreater(len(p), 0)
539
540 pb = tempfile.gettempprefixb()
541
542 self.assertIsInstance(pb, bytes)
543 self.assertGreater(len(pb), 0)
Guido van Rossum0e548712002-08-09 16:14:33 +0000544
545 def test_usable_template(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000546 # gettempprefix returns a usable prefix string
Guido van Rossum0e548712002-08-09 16:14:33 +0000547
548 # Create a temp directory, avoiding use of the prefix.
549 # Then attempt to create a file whose name is
550 # prefix + 'xxxxxx.xxx' in that directory.
551 p = tempfile.gettempprefix() + "xxxxxx.xxx"
552 d = tempfile.mkdtemp(prefix="")
553 try:
554 p = os.path.join(d, p)
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100555 fd = os.open(p, os.O_RDWR | os.O_CREAT)
Guido van Rossum0e548712002-08-09 16:14:33 +0000556 os.close(fd)
557 os.unlink(p)
558 finally:
559 os.rmdir(d)
560
Guido van Rossum0e548712002-08-09 16:14:33 +0000561
Antoine Pitroueab2a502012-03-10 16:34:40 +0100562class TestGetTempDir(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000563 """Test gettempdir()."""
564
565 def test_directory_exists(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000566 # gettempdir returns a directory which exists
Guido van Rossum0e548712002-08-09 16:14:33 +0000567
Gregory P. Smithad577b92015-05-22 16:18:14 -0700568 for d in (tempfile.gettempdir(), tempfile.gettempdirb()):
569 self.assertTrue(os.path.isabs(d) or d == os.curdir,
570 "%r is not an absolute path" % d)
571 self.assertTrue(os.path.isdir(d),
572 "%r is not a directory" % d)
Guido van Rossum0e548712002-08-09 16:14:33 +0000573
574 def test_directory_writable(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000575 # gettempdir returns a directory writable by the user
Guido van Rossum0e548712002-08-09 16:14:33 +0000576
577 # sneaky: just instantiate a NamedTemporaryFile, which
578 # defaults to writing into the directory returned by
579 # gettempdir.
Serhiy Storchaka5b10b982019-03-05 10:06:26 +0200580 with tempfile.NamedTemporaryFile() as file:
581 file.write(b"blat")
Guido van Rossum0e548712002-08-09 16:14:33 +0000582
583 def test_same_thing(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000584 # gettempdir always returns the same object
Guido van Rossum0e548712002-08-09 16:14:33 +0000585 a = tempfile.gettempdir()
586 b = tempfile.gettempdir()
Gregory P. Smithad577b92015-05-22 16:18:14 -0700587 c = tempfile.gettempdirb()
Guido van Rossum0e548712002-08-09 16:14:33 +0000588
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000589 self.assertTrue(a is b)
Gregory P. Smithad577b92015-05-22 16:18:14 -0700590 self.assertNotEqual(type(a), type(c))
591 self.assertEqual(a, os.fsdecode(c))
Guido van Rossum0e548712002-08-09 16:14:33 +0000592
Tim Golden6d09f092013-10-25 18:38:16 +0100593 def test_case_sensitive(self):
594 # gettempdir should not flatten its case
595 # even on a case-insensitive file system
596 case_sensitive_tempdir = tempfile.mkdtemp("-Temp")
597 _tempdir, tempfile.tempdir = tempfile.tempdir, None
598 try:
Hai Shi0c4f0f32020-06-30 21:46:31 +0800599 with os_helper.EnvironmentVarGuard() as env:
Tim Golden6d09f092013-10-25 18:38:16 +0100600 # Fake the first env var which is checked as a candidate
601 env["TMPDIR"] = case_sensitive_tempdir
602 self.assertEqual(tempfile.gettempdir(), case_sensitive_tempdir)
603 finally:
604 tempfile.tempdir = _tempdir
Hai Shi0c4f0f32020-06-30 21:46:31 +0800605 os_helper.rmdir(case_sensitive_tempdir)
Tim Golden6d09f092013-10-25 18:38:16 +0100606
Guido van Rossum0e548712002-08-09 16:14:33 +0000607
Antoine Pitroueab2a502012-03-10 16:34:40 +0100608class TestMkstemp(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000609 """Test mkstemp()."""
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000610
Gregory P. Smithad577b92015-05-22 16:18:14 -0700611 def do_create(self, dir=None, pre=None, suf=None):
612 output_type = tempfile._infer_return_type(dir, pre, suf)
Guido van Rossum0e548712002-08-09 16:14:33 +0000613 if dir is None:
Gregory P. Smithad577b92015-05-22 16:18:14 -0700614 if output_type is str:
615 dir = tempfile.gettempdir()
616 else:
617 dir = tempfile.gettempdirb()
618 if pre is None:
619 pre = output_type()
620 if suf is None:
621 suf = output_type()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100622 (fd, name) = tempfile.mkstemp(dir=dir, prefix=pre, suffix=suf)
623 (ndir, nbase) = os.path.split(name)
624 adir = os.path.abspath(dir)
625 self.assertEqual(adir, ndir,
626 "Directory '%s' incorrectly returned as '%s'" % (adir, ndir))
Guido van Rossum0e548712002-08-09 16:14:33 +0000627
628 try:
629 self.nameCheck(name, dir, pre, suf)
630 finally:
631 os.close(fd)
632 os.unlink(name)
633
634 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000635 # mkstemp can create files
Guido van Rossum0e548712002-08-09 16:14:33 +0000636 self.do_create()
637 self.do_create(pre="a")
638 self.do_create(suf="b")
639 self.do_create(pre="a", suf="b")
640 self.do_create(pre="aa", suf=".txt")
Martin v. Löwisd6625482003-10-12 17:37:01 +0000641 self.do_create(dir=".")
Guido van Rossum0e548712002-08-09 16:14:33 +0000642
Gregory P. Smithad577b92015-05-22 16:18:14 -0700643 def test_basic_with_bytes_names(self):
644 # mkstemp can create files when given name parts all
645 # specified as bytes.
646 d = tempfile.gettempdirb()
647 self.do_create(dir=d, suf=b"")
648 self.do_create(dir=d, pre=b"a")
649 self.do_create(dir=d, suf=b"b")
650 self.do_create(dir=d, pre=b"a", suf=b"b")
651 self.do_create(dir=d, pre=b"aa", suf=b".txt")
652 self.do_create(dir=b".")
653 with self.assertRaises(TypeError):
654 self.do_create(dir=".", pre=b"aa", suf=b".txt")
655 with self.assertRaises(TypeError):
656 self.do_create(dir=b".", pre="aa", suf=b".txt")
657 with self.assertRaises(TypeError):
658 self.do_create(dir=b".", pre=b"aa", suf=".txt")
659
660
Guido van Rossum0e548712002-08-09 16:14:33 +0000661 def test_choose_directory(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000662 # mkstemp can create directories in a user-selected directory
Guido van Rossum0e548712002-08-09 16:14:33 +0000663 dir = tempfile.mkdtemp()
664 try:
665 self.do_create(dir=dir)
Anthony Sottile370138b2019-09-09 08:54:34 -0700666 self.do_create(dir=pathlib.Path(dir))
Guido van Rossum0e548712002-08-09 16:14:33 +0000667 finally:
668 os.rmdir(dir)
669
Eric L9c792742021-03-03 21:36:22 +0100670 def test_for_tempdir_is_bytes_issue40701_api_warts(self):
671 orig_tempdir = tempfile.tempdir
672 self.assertIsInstance(tempfile.tempdir, (str, type(None)))
673 try:
674 fd, path = tempfile.mkstemp()
675 os.close(fd)
676 os.unlink(path)
677 self.assertIsInstance(path, str)
678 tempfile.tempdir = tempfile.gettempdirb()
679 self.assertIsInstance(tempfile.tempdir, bytes)
680 self.assertIsInstance(tempfile.gettempdir(), str)
681 self.assertIsInstance(tempfile.gettempdirb(), bytes)
682 fd, path = tempfile.mkstemp()
683 os.close(fd)
684 os.unlink(path)
685 self.assertIsInstance(path, bytes)
686 fd, path = tempfile.mkstemp(suffix='.txt')
687 os.close(fd)
688 os.unlink(path)
689 self.assertIsInstance(path, str)
690 fd, path = tempfile.mkstemp(prefix='test-temp-')
691 os.close(fd)
692 os.unlink(path)
693 self.assertIsInstance(path, str)
694 fd, path = tempfile.mkstemp(dir=tempfile.gettempdir())
695 os.close(fd)
696 os.unlink(path)
697 self.assertIsInstance(path, str)
698 finally:
699 tempfile.tempdir = orig_tempdir
700
Guido van Rossum0e548712002-08-09 16:14:33 +0000701
Serhiy Storchaka5d6b7b12015-05-20 00:11:48 +0300702class TestMkdtemp(TestBadTempdir, BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000703 """Test mkdtemp()."""
704
Serhiy Storchaka5d6b7b12015-05-20 00:11:48 +0300705 def make_temp(self):
706 return tempfile.mkdtemp()
707
Gregory P. Smithad577b92015-05-22 16:18:14 -0700708 def do_create(self, dir=None, pre=None, suf=None):
709 output_type = tempfile._infer_return_type(dir, pre, suf)
Guido van Rossum0e548712002-08-09 16:14:33 +0000710 if dir is None:
Gregory P. Smithad577b92015-05-22 16:18:14 -0700711 if output_type is str:
712 dir = tempfile.gettempdir()
713 else:
714 dir = tempfile.gettempdirb()
715 if pre is None:
716 pre = output_type()
717 if suf is None:
718 suf = output_type()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100719 name = tempfile.mkdtemp(dir=dir, prefix=pre, suffix=suf)
Guido van Rossum0e548712002-08-09 16:14:33 +0000720
721 try:
722 self.nameCheck(name, dir, pre, suf)
723 return name
724 except:
725 os.rmdir(name)
726 raise
727
728 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000729 # mkdtemp can create directories
Guido van Rossum0e548712002-08-09 16:14:33 +0000730 os.rmdir(self.do_create())
731 os.rmdir(self.do_create(pre="a"))
732 os.rmdir(self.do_create(suf="b"))
733 os.rmdir(self.do_create(pre="a", suf="b"))
734 os.rmdir(self.do_create(pre="aa", suf=".txt"))
Tim Petersa0d55de2002-08-09 18:01:01 +0000735
Gregory P. Smithad577b92015-05-22 16:18:14 -0700736 def test_basic_with_bytes_names(self):
737 # mkdtemp can create directories when given all binary parts
738 d = tempfile.gettempdirb()
739 os.rmdir(self.do_create(dir=d))
740 os.rmdir(self.do_create(dir=d, pre=b"a"))
741 os.rmdir(self.do_create(dir=d, suf=b"b"))
742 os.rmdir(self.do_create(dir=d, pre=b"a", suf=b"b"))
743 os.rmdir(self.do_create(dir=d, pre=b"aa", suf=b".txt"))
744 with self.assertRaises(TypeError):
745 os.rmdir(self.do_create(dir=d, pre="aa", suf=b".txt"))
746 with self.assertRaises(TypeError):
747 os.rmdir(self.do_create(dir=d, pre=b"aa", suf=".txt"))
748 with self.assertRaises(TypeError):
749 os.rmdir(self.do_create(dir="", pre=b"aa", suf=b".txt"))
750
Guido van Rossum0e548712002-08-09 16:14:33 +0000751 def test_basic_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000752 # mkdtemp can create many directories (stochastic)
Guido van Rossum805365e2007-05-07 22:24:25 +0000753 extant = list(range(TEST_FILES))
Guido van Rossum0e548712002-08-09 16:14:33 +0000754 try:
755 for i in extant:
756 extant[i] = self.do_create(pre="aa")
757 finally:
758 for i in extant:
Guido van Rossum3172c5d2007-10-16 18:12:55 +0000759 if(isinstance(i, str)):
Guido van Rossum0e548712002-08-09 16:14:33 +0000760 os.rmdir(i)
761
762 def test_choose_directory(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000763 # mkdtemp can create directories in a user-selected directory
Guido van Rossum0e548712002-08-09 16:14:33 +0000764 dir = tempfile.mkdtemp()
765 try:
766 os.rmdir(self.do_create(dir=dir))
Anthony Sottile370138b2019-09-09 08:54:34 -0700767 os.rmdir(self.do_create(dir=pathlib.Path(dir)))
Guido van Rossum0e548712002-08-09 16:14:33 +0000768 finally:
769 os.rmdir(dir)
770
771 def test_mode(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000772 # mkdtemp creates directories with the proper mode
Guido van Rossum0e548712002-08-09 16:14:33 +0000773
774 dir = self.do_create()
775 try:
776 mode = stat.S_IMODE(os.stat(dir).st_mode)
Guido van Rossumcd16bf62007-06-13 18:07:49 +0000777 mode &= 0o777 # Mask off sticky bits inherited from /tmp
778 expected = 0o700
Jesus Cea4791a242012-10-05 03:15:39 +0200779 if sys.platform == 'win32':
Tim Petersca3ac7f2002-08-09 18:13:51 +0000780 # There's no distinction among 'user', 'group' and 'world';
781 # replicate the 'user' bits.
782 user = expected >> 6
783 expected = user * (1 + 8 + 64)
784 self.assertEqual(mode, expected)
Guido van Rossum0e548712002-08-09 16:14:33 +0000785 finally:
786 os.rmdir(dir)
787
Eli Benderskyaa04f9a2013-09-13 05:28:20 -0700788 def test_collision_with_existing_file(self):
789 # mkdtemp tries another name when a file with
790 # the chosen name already exists
791 with _inside_empty_temp_dir(), \
792 _mock_candidate_names('aaa', 'aaa', 'bbb'):
793 file = tempfile.NamedTemporaryFile(delete=False)
794 file.close()
795 self.assertTrue(file.name.endswith('aaa'))
796 dir = tempfile.mkdtemp()
797 self.assertTrue(dir.endswith('bbb'))
798
799 def test_collision_with_existing_directory(self):
800 # mkdtemp tries another name when a directory with
801 # the chosen name already exists
802 with _inside_empty_temp_dir(), \
803 _mock_candidate_names('aaa', 'aaa', 'bbb'):
804 dir1 = tempfile.mkdtemp()
805 self.assertTrue(dir1.endswith('aaa'))
806 dir2 = tempfile.mkdtemp()
807 self.assertTrue(dir2.endswith('bbb'))
808
Eric L9c792742021-03-03 21:36:22 +0100809 def test_for_tempdir_is_bytes_issue40701_api_warts(self):
810 orig_tempdir = tempfile.tempdir
811 self.assertIsInstance(tempfile.tempdir, (str, type(None)))
812 try:
813 path = tempfile.mkdtemp()
814 os.rmdir(path)
815 self.assertIsInstance(path, str)
816 tempfile.tempdir = tempfile.gettempdirb()
817 self.assertIsInstance(tempfile.tempdir, bytes)
818 self.assertIsInstance(tempfile.gettempdir(), str)
819 self.assertIsInstance(tempfile.gettempdirb(), bytes)
820 path = tempfile.mkdtemp()
821 os.rmdir(path)
822 self.assertIsInstance(path, bytes)
823 path = tempfile.mkdtemp(suffix='-dir')
824 os.rmdir(path)
825 self.assertIsInstance(path, str)
826 path = tempfile.mkdtemp(prefix='test-mkdtemp-')
827 os.rmdir(path)
828 self.assertIsInstance(path, str)
829 path = tempfile.mkdtemp(dir=tempfile.gettempdir())
830 os.rmdir(path)
831 self.assertIsInstance(path, str)
832 finally:
833 tempfile.tempdir = orig_tempdir
834
Guido van Rossum0e548712002-08-09 16:14:33 +0000835
Antoine Pitroueab2a502012-03-10 16:34:40 +0100836class TestMktemp(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000837 """Test mktemp()."""
838
839 # For safety, all use of mktemp must occur in a private directory.
840 # We must also suppress the RuntimeWarning it generates.
841 def setUp(self):
842 self.dir = tempfile.mkdtemp()
Brett Cannone1adece2010-03-20 22:19:55 +0000843 super().setUp()
Guido van Rossum0e548712002-08-09 16:14:33 +0000844
845 def tearDown(self):
846 if self.dir:
847 os.rmdir(self.dir)
848 self.dir = None
Brett Cannone1adece2010-03-20 22:19:55 +0000849 super().tearDown()
Guido van Rossum0e548712002-08-09 16:14:33 +0000850
851 class mktemped:
852 _unlink = os.unlink
853 _bflags = tempfile._bin_openflags
854
855 def __init__(self, dir, pre, suf):
856 self.name = tempfile.mktemp(dir=dir, prefix=pre, suffix=suf)
857 # Create the file. This will raise an exception if it's
858 # mysteriously appeared in the meanwhile.
Guido van Rossumcd16bf62007-06-13 18:07:49 +0000859 os.close(os.open(self.name, self._bflags, 0o600))
Guido van Rossum0e548712002-08-09 16:14:33 +0000860
861 def __del__(self):
862 self._unlink(self.name)
863
864 def do_create(self, pre="", suf=""):
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100865 file = self.mktemped(self.dir, pre, suf)
Guido van Rossum0e548712002-08-09 16:14:33 +0000866
867 self.nameCheck(file.name, self.dir, pre, suf)
868 return file
869
870 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000871 # mktemp can choose usable file names
Guido van Rossum0e548712002-08-09 16:14:33 +0000872 self.do_create()
873 self.do_create(pre="a")
874 self.do_create(suf="b")
875 self.do_create(pre="a", suf="b")
876 self.do_create(pre="aa", suf=".txt")
877
878 def test_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000879 # mktemp can choose many usable file names (stochastic)
Guido van Rossum805365e2007-05-07 22:24:25 +0000880 extant = list(range(TEST_FILES))
Guido van Rossum0e548712002-08-09 16:14:33 +0000881 for i in extant:
882 extant[i] = self.do_create(pre="aa")
883
Fred Drake8bec4832002-11-22 20:13:43 +0000884## def test_warning(self):
885## # mktemp issues a warning when used
886## warnings.filterwarnings("error",
887## category=RuntimeWarning,
888## message="mktemp")
889## self.assertRaises(RuntimeWarning,
890## tempfile.mktemp, dir=self.dir)
Guido van Rossum0e548712002-08-09 16:14:33 +0000891
Guido van Rossum0e548712002-08-09 16:14:33 +0000892
893# We test _TemporaryFileWrapper by testing NamedTemporaryFile.
894
895
Antoine Pitroueab2a502012-03-10 16:34:40 +0100896class TestNamedTemporaryFile(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000897 """Test NamedTemporaryFile()."""
898
Guido van Rossumd8faa362007-04-27 19:54:29 +0000899 def do_create(self, dir=None, pre="", suf="", delete=True):
Guido van Rossum0e548712002-08-09 16:14:33 +0000900 if dir is None:
901 dir = tempfile.gettempdir()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100902 file = tempfile.NamedTemporaryFile(dir=dir, prefix=pre, suffix=suf,
903 delete=delete)
Guido van Rossum0e548712002-08-09 16:14:33 +0000904
905 self.nameCheck(file.name, dir, pre, suf)
906 return file
907
908
909 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000910 # NamedTemporaryFile can create files
Guido van Rossum0e548712002-08-09 16:14:33 +0000911 self.do_create()
912 self.do_create(pre="a")
913 self.do_create(suf="b")
914 self.do_create(pre="a", suf="b")
915 self.do_create(pre="aa", suf=".txt")
916
Antoine Pitrou17c93262013-12-21 22:14:56 +0100917 def test_method_lookup(self):
918 # Issue #18879: Looking up a temporary file method should keep it
919 # alive long enough.
920 f = self.do_create()
921 wr = weakref.ref(f)
922 write = f.write
923 write2 = f.write
924 del f
925 write(b'foo')
926 del write
927 write2(b'bar')
928 del write2
929 if support.check_impl_detail(cpython=True):
930 # No reference cycle was created.
931 self.assertIsNone(wr())
932
Serhiy Storchaka56cefa62015-03-19 15:23:15 +0200933 def test_iter(self):
934 # Issue #23700: getting iterator from a temporary file should keep
935 # it alive as long as it's being iterated over
936 lines = [b'spam\n', b'eggs\n', b'beans\n']
937 def make_file():
938 f = tempfile.NamedTemporaryFile(mode='w+b')
939 f.write(b''.join(lines))
940 f.seek(0)
941 return f
942 for i, l in enumerate(make_file()):
943 self.assertEqual(l, lines[i])
944 self.assertEqual(i, len(lines) - 1)
945
Guido van Rossum0e548712002-08-09 16:14:33 +0000946 def test_creates_named(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000947 # NamedTemporaryFile creates files with names
Guido van Rossum0e548712002-08-09 16:14:33 +0000948 f = tempfile.NamedTemporaryFile()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000949 self.assertTrue(os.path.exists(f.name),
Guido van Rossum0e548712002-08-09 16:14:33 +0000950 "NamedTemporaryFile %s does not exist" % f.name)
951
952 def test_del_on_close(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000953 # A NamedTemporaryFile is deleted when closed
Guido van Rossum0e548712002-08-09 16:14:33 +0000954 dir = tempfile.mkdtemp()
955 try:
Serhiy Storchaka5b10b982019-03-05 10:06:26 +0200956 with tempfile.NamedTemporaryFile(dir=dir) as f:
957 f.write(b'blat')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000958 self.assertFalse(os.path.exists(f.name),
Guido van Rossum0e548712002-08-09 16:14:33 +0000959 "NamedTemporaryFile %s exists after close" % f.name)
960 finally:
961 os.rmdir(dir)
962
Guido van Rossumd8faa362007-04-27 19:54:29 +0000963 def test_dis_del_on_close(self):
964 # Tests that delete-on-close can be disabled
965 dir = tempfile.mkdtemp()
966 tmp = None
967 try:
968 f = tempfile.NamedTemporaryFile(dir=dir, delete=False)
969 tmp = f.name
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000970 f.write(b'blat')
Guido van Rossumd8faa362007-04-27 19:54:29 +0000971 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000972 self.assertTrue(os.path.exists(f.name),
Guido van Rossumd8faa362007-04-27 19:54:29 +0000973 "NamedTemporaryFile %s missing after close" % f.name)
974 finally:
975 if tmp is not None:
976 os.unlink(tmp)
977 os.rmdir(dir)
978
Guido van Rossum0e548712002-08-09 16:14:33 +0000979 def test_multiple_close(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000980 # A NamedTemporaryFile can be closed many times without error
Guido van Rossum0e548712002-08-09 16:14:33 +0000981 f = tempfile.NamedTemporaryFile()
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000982 f.write(b'abc\n')
Guido van Rossum0e548712002-08-09 16:14:33 +0000983 f.close()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100984 f.close()
985 f.close()
Guido van Rossum0e548712002-08-09 16:14:33 +0000986
Christian Heimes3ecfea712008-02-09 20:51:34 +0000987 def test_context_manager(self):
988 # A NamedTemporaryFile can be used as a context manager
989 with tempfile.NamedTemporaryFile() as f:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000990 self.assertTrue(os.path.exists(f.name))
991 self.assertFalse(os.path.exists(f.name))
Christian Heimes3ecfea712008-02-09 20:51:34 +0000992 def use_closed():
993 with f:
994 pass
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000995 self.assertRaises(ValueError, use_closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000996
Victor Stinner1f99f9d2014-03-25 09:18:04 +0100997 def test_no_leak_fd(self):
Victor Stinner87d13ea2014-03-25 18:19:17 +0100998 # Issue #21058: don't leak file descriptor when io.open() fails
Victor Stinner1f99f9d2014-03-25 09:18:04 +0100999 closed = []
Victor Stinnerc61c1702014-07-29 01:13:39 +02001000 os_close = os.close
Victor Stinner1f99f9d2014-03-25 09:18:04 +01001001 def close(fd):
1002 closed.append(fd)
Victor Stinnerc61c1702014-07-29 01:13:39 +02001003 os_close(fd)
Victor Stinner1f99f9d2014-03-25 09:18:04 +01001004
1005 with mock.patch('os.close', side_effect=close):
1006 with mock.patch('io.open', side_effect=ValueError):
1007 self.assertRaises(ValueError, tempfile.NamedTemporaryFile)
1008 self.assertEqual(len(closed), 1)
1009
Martin Panter7869a222016-02-28 05:22:20 +00001010 def test_bad_mode(self):
1011 dir = tempfile.mkdtemp()
Hai Shi0c4f0f32020-06-30 21:46:31 +08001012 self.addCleanup(os_helper.rmtree, dir)
Martin Panter7869a222016-02-28 05:22:20 +00001013 with self.assertRaises(ValueError):
1014 tempfile.NamedTemporaryFile(mode='wr', dir=dir)
1015 with self.assertRaises(TypeError):
1016 tempfile.NamedTemporaryFile(mode=2, dir=dir)
1017 self.assertEqual(os.listdir(dir), [])
Guido van Rossum0e548712002-08-09 16:14:33 +00001018
Martin Panter7869a222016-02-28 05:22:20 +00001019 # How to test the mode and bufsize parameters?
Guido van Rossum0e548712002-08-09 16:14:33 +00001020
Antoine Pitroueab2a502012-03-10 16:34:40 +01001021class TestSpooledTemporaryFile(BaseTestCase):
Guido van Rossumd8faa362007-04-27 19:54:29 +00001022 """Test SpooledTemporaryFile()."""
1023
1024 def do_create(self, max_size=0, dir=None, pre="", suf=""):
1025 if dir is None:
1026 dir = tempfile.gettempdir()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +01001027 file = tempfile.SpooledTemporaryFile(max_size=max_size, dir=dir, prefix=pre, suffix=suf)
Guido van Rossumd8faa362007-04-27 19:54:29 +00001028
1029 return file
1030
1031
1032 def test_basic(self):
1033 # SpooledTemporaryFile can create files
1034 f = self.do_create()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001035 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +00001036 f = self.do_create(max_size=100, pre="a", suf=".txt")
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001037 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +00001038
1039 def test_del_on_close(self):
1040 # A SpooledTemporaryFile is deleted when closed
1041 dir = tempfile.mkdtemp()
1042 try:
1043 f = tempfile.SpooledTemporaryFile(max_size=10, dir=dir)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001044 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +00001045 f.write(b'blat ' * 5)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001046 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +00001047 filename = f.name
1048 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001049 self.assertFalse(isinstance(filename, str) and os.path.exists(filename),
Guido van Rossumd8faa362007-04-27 19:54:29 +00001050 "SpooledTemporaryFile %s exists after close" % filename)
1051 finally:
1052 os.rmdir(dir)
1053
1054 def test_rewrite_small(self):
1055 # A SpooledTemporaryFile can be written to multiple within the max_size
1056 f = self.do_create(max_size=30)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001057 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +00001058 for i in range(5):
1059 f.seek(0, 0)
Guido van Rossum39478e82007-08-27 17:23:59 +00001060 f.write(b'x' * 20)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001061 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +00001062
1063 def test_write_sequential(self):
1064 # A SpooledTemporaryFile should hold exactly max_size bytes, and roll
1065 # over afterward
1066 f = self.do_create(max_size=30)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001067 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +00001068 f.write(b'x' * 20)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001069 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +00001070 f.write(b'x' * 10)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001071 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +00001072 f.write(b'x')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001073 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +00001074
R David Murrayd89ee792011-03-14 09:55:46 -04001075 def test_writelines(self):
1076 # Verify writelines with a SpooledTemporaryFile
1077 f = self.do_create()
1078 f.writelines((b'x', b'y', b'z'))
Inada Naoki485e7152020-04-17 15:56:35 +09001079 pos = f.seek(0)
1080 self.assertEqual(pos, 0)
R David Murrayd89ee792011-03-14 09:55:46 -04001081 buf = f.read()
1082 self.assertEqual(buf, b'xyz')
1083
1084 def test_writelines_sequential(self):
1085 # A SpooledTemporaryFile should hold exactly max_size bytes, and roll
1086 # over afterward
1087 f = self.do_create(max_size=35)
1088 f.writelines((b'x' * 20, b'x' * 10, b'x' * 5))
1089 self.assertFalse(f._rolled)
1090 f.write(b'x')
1091 self.assertTrue(f._rolled)
1092
Guido van Rossumd8faa362007-04-27 19:54:29 +00001093 def test_sparse(self):
1094 # A SpooledTemporaryFile that is written late in the file will extend
1095 # when that occurs
1096 f = self.do_create(max_size=30)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001097 self.assertFalse(f._rolled)
Inada Naoki485e7152020-04-17 15:56:35 +09001098 pos = f.seek(100, 0)
1099 self.assertEqual(pos, 100)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001100 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +00001101 f.write(b'x')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001102 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +00001103
1104 def test_fileno(self):
1105 # A SpooledTemporaryFile should roll over to a real file on fileno()
1106 f = self.do_create(max_size=30)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001107 self.assertFalse(f._rolled)
1108 self.assertTrue(f.fileno() > 0)
1109 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +00001110
Christian Heimes3ecfea712008-02-09 20:51:34 +00001111 def test_multiple_close_before_rollover(self):
Guido van Rossumd8faa362007-04-27 19:54:29 +00001112 # A SpooledTemporaryFile can be closed many times without error
1113 f = tempfile.SpooledTemporaryFile()
Guido van Rossum39478e82007-08-27 17:23:59 +00001114 f.write(b'abc\n')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001115 self.assertFalse(f._rolled)
Christian Heimes3ecfea712008-02-09 20:51:34 +00001116 f.close()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +01001117 f.close()
1118 f.close()
Christian Heimes3ecfea712008-02-09 20:51:34 +00001119
1120 def test_multiple_close_after_rollover(self):
1121 # A SpooledTemporaryFile can be closed many times without error
1122 f = tempfile.SpooledTemporaryFile(max_size=1)
1123 f.write(b'abc\n')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001124 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +00001125 f.close()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +01001126 f.close()
1127 f.close()
Guido van Rossumd8faa362007-04-27 19:54:29 +00001128
1129 def test_bound_methods(self):
1130 # It should be OK to steal a bound method from a SpooledTemporaryFile
1131 # and use it independently; when the file rolls over, those bound
1132 # methods should continue to function
1133 f = self.do_create(max_size=30)
1134 read = f.read
1135 write = f.write
1136 seek = f.seek
1137
Guido van Rossum39478e82007-08-27 17:23:59 +00001138 write(b"a" * 35)
1139 write(b"b" * 35)
Guido van Rossumd8faa362007-04-27 19:54:29 +00001140 seek(0, 0)
Guido van Rossum9a634702007-07-09 10:24:45 +00001141 self.assertEqual(read(70), b'a'*35 + b'b'*35)
1142
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +02001143 def test_properties(self):
1144 f = tempfile.SpooledTemporaryFile(max_size=10)
1145 f.write(b'x' * 10)
1146 self.assertFalse(f._rolled)
1147 self.assertEqual(f.mode, 'w+b')
1148 self.assertIsNone(f.name)
1149 with self.assertRaises(AttributeError):
1150 f.newlines
1151 with self.assertRaises(AttributeError):
1152 f.encoding
sth825aab92018-05-23 07:07:01 +02001153 with self.assertRaises(AttributeError):
1154 f.errors
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +02001155
1156 f.write(b'x')
1157 self.assertTrue(f._rolled)
1158 self.assertEqual(f.mode, 'rb+')
1159 self.assertIsNotNone(f.name)
1160 with self.assertRaises(AttributeError):
1161 f.newlines
1162 with self.assertRaises(AttributeError):
1163 f.encoding
sth825aab92018-05-23 07:07:01 +02001164 with self.assertRaises(AttributeError):
1165 f.errors
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +02001166
Guido van Rossum9a634702007-07-09 10:24:45 +00001167 def test_text_mode(self):
1168 # Creating a SpooledTemporaryFile with a text mode should produce
1169 # a file object reading and writing (Unicode) text strings.
Inada Naokiea9835c2019-11-27 22:22:06 +09001170 f = tempfile.SpooledTemporaryFile(mode='w+', max_size=10,
1171 encoding="utf-8")
Guido van Rossum9a634702007-07-09 10:24:45 +00001172 f.write("abc\n")
1173 f.seek(0)
1174 self.assertEqual(f.read(), "abc\n")
1175 f.write("def\n")
1176 f.seek(0)
1177 self.assertEqual(f.read(), "abc\ndef\n")
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +02001178 self.assertFalse(f._rolled)
1179 self.assertEqual(f.mode, 'w+')
1180 self.assertIsNone(f.name)
Inada Naokiea9835c2019-11-27 22:22:06 +09001181 self.assertEqual(f.newlines, os.linesep)
1182 self.assertEqual(f.encoding, "utf-8")
1183 self.assertEqual(f.errors, "strict")
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +02001184
Guido van Rossum9a634702007-07-09 10:24:45 +00001185 f.write("xyzzy\n")
1186 f.seek(0)
1187 self.assertEqual(f.read(), "abc\ndef\nxyzzy\n")
Amaury Forgeot d'Arc7d0bddd2009-11-30 00:08:56 +00001188 # Check that Ctrl+Z doesn't truncate the file
1189 f.write("foo\x1abar\n")
1190 f.seek(0)
1191 self.assertEqual(f.read(), "abc\ndef\nxyzzy\nfoo\x1abar\n")
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +02001192 self.assertTrue(f._rolled)
1193 self.assertEqual(f.mode, 'w+')
1194 self.assertIsNotNone(f.name)
Serhiy Storchaka497cee42013-02-10 14:43:46 +02001195 self.assertEqual(f.newlines, os.linesep)
Inada Naokiea9835c2019-11-27 22:22:06 +09001196 self.assertEqual(f.encoding, "utf-8")
1197 self.assertEqual(f.errors, "strict")
Guido van Rossum9a634702007-07-09 10:24:45 +00001198
Guido van Rossumf0c74162007-08-28 03:29:45 +00001199 def test_text_newline_and_encoding(self):
1200 f = tempfile.SpooledTemporaryFile(mode='w+', max_size=10,
sth825aab92018-05-23 07:07:01 +02001201 newline='', encoding='utf-8',
1202 errors='ignore')
Guido van Rossumf0c74162007-08-28 03:29:45 +00001203 f.write("\u039B\r\n")
1204 f.seek(0)
1205 self.assertEqual(f.read(), "\u039B\r\n")
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001206 self.assertFalse(f._rolled)
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +02001207 self.assertEqual(f.mode, 'w+')
1208 self.assertIsNone(f.name)
Inada Naokiea9835c2019-11-27 22:22:06 +09001209 self.assertIsNotNone(f.newlines)
1210 self.assertEqual(f.encoding, "utf-8")
1211 self.assertEqual(f.errors, "ignore")
Guido van Rossumf0c74162007-08-28 03:29:45 +00001212
Inada Naokiea9835c2019-11-27 22:22:06 +09001213 f.write("\u039C" * 10 + "\r\n")
1214 f.write("\u039D" * 20)
Guido van Rossumf0c74162007-08-28 03:29:45 +00001215 f.seek(0)
Inada Naokiea9835c2019-11-27 22:22:06 +09001216 self.assertEqual(f.read(),
1217 "\u039B\r\n" + ("\u039C" * 10) + "\r\n" + ("\u039D" * 20))
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001218 self.assertTrue(f._rolled)
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +02001219 self.assertEqual(f.mode, 'w+')
1220 self.assertIsNotNone(f.name)
1221 self.assertIsNotNone(f.newlines)
1222 self.assertEqual(f.encoding, 'utf-8')
sth825aab92018-05-23 07:07:01 +02001223 self.assertEqual(f.errors, 'ignore')
Guido van Rossumd8faa362007-04-27 19:54:29 +00001224
Christian Heimes3ecfea712008-02-09 20:51:34 +00001225 def test_context_manager_before_rollover(self):
1226 # A SpooledTemporaryFile can be used as a context manager
1227 with tempfile.SpooledTemporaryFile(max_size=1) as f:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001228 self.assertFalse(f._rolled)
1229 self.assertFalse(f.closed)
1230 self.assertTrue(f.closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +00001231 def use_closed():
1232 with f:
1233 pass
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001234 self.assertRaises(ValueError, use_closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +00001235
1236 def test_context_manager_during_rollover(self):
1237 # A SpooledTemporaryFile can be used as a context manager
1238 with tempfile.SpooledTemporaryFile(max_size=1) as f:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001239 self.assertFalse(f._rolled)
Christian Heimes3ecfea712008-02-09 20:51:34 +00001240 f.write(b'abc\n')
1241 f.flush()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001242 self.assertTrue(f._rolled)
1243 self.assertFalse(f.closed)
1244 self.assertTrue(f.closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +00001245 def use_closed():
1246 with f:
1247 pass
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001248 self.assertRaises(ValueError, use_closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +00001249
1250 def test_context_manager_after_rollover(self):
1251 # A SpooledTemporaryFile can be used as a context manager
1252 f = tempfile.SpooledTemporaryFile(max_size=1)
1253 f.write(b'abc\n')
1254 f.flush()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001255 self.assertTrue(f._rolled)
Christian Heimes3ecfea712008-02-09 20:51:34 +00001256 with f:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001257 self.assertFalse(f.closed)
1258 self.assertTrue(f.closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +00001259 def use_closed():
1260 with f:
1261 pass
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001262 self.assertRaises(ValueError, use_closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +00001263
Antoine Pitrou0e86a582011-11-25 18:03:09 +01001264 def test_truncate_with_size_parameter(self):
1265 # A SpooledTemporaryFile can be truncated to zero size
1266 f = tempfile.SpooledTemporaryFile(max_size=10)
1267 f.write(b'abcdefg\n')
1268 f.seek(0)
1269 f.truncate()
1270 self.assertFalse(f._rolled)
1271 self.assertEqual(f._file.getvalue(), b'')
1272 # A SpooledTemporaryFile can be truncated to a specific size
1273 f = tempfile.SpooledTemporaryFile(max_size=10)
1274 f.write(b'abcdefg\n')
1275 f.truncate(4)
1276 self.assertFalse(f._rolled)
1277 self.assertEqual(f._file.getvalue(), b'abcd')
1278 # A SpooledTemporaryFile rolls over if truncated to large size
1279 f = tempfile.SpooledTemporaryFile(max_size=10)
1280 f.write(b'abcdefg\n')
1281 f.truncate(20)
1282 self.assertTrue(f._rolled)
Anthony Sottile8377cd42019-02-25 14:32:27 -08001283 self.assertEqual(os.fstat(f.fileno()).st_size, 20)
Christian Heimes3ecfea712008-02-09 20:51:34 +00001284
Batuhan TaĹźkaya09c482f2019-12-30 19:08:08 +03001285 def test_class_getitem(self):
Guido van Rossum48b069a2020-04-07 09:50:06 -07001286 self.assertIsInstance(tempfile.SpooledTemporaryFile[bytes],
1287 types.GenericAlias)
Guido van Rossum0e548712002-08-09 16:14:33 +00001288
Guido van Rossum8ccd9b62002-08-22 20:02:03 +00001289if tempfile.NamedTemporaryFile is not tempfile.TemporaryFile:
Antoine Pitroueab2a502012-03-10 16:34:40 +01001290
1291 class TestTemporaryFile(BaseTestCase):
1292 """Test TemporaryFile()."""
1293
1294 def test_basic(self):
1295 # TemporaryFile can create files
1296 # No point in testing the name params - the file has no name.
1297 tempfile.TemporaryFile()
1298
1299 def test_has_no_name(self):
1300 # TemporaryFile creates files with no names (on this system)
1301 dir = tempfile.mkdtemp()
1302 f = tempfile.TemporaryFile(dir=dir)
1303 f.write(b'blat')
1304
1305 # Sneaky: because this file has no name, it should not prevent
1306 # us from removing the directory it was created in.
1307 try:
1308 os.rmdir(dir)
1309 except:
1310 # cleanup
1311 f.close()
1312 os.rmdir(dir)
1313 raise
1314
1315 def test_multiple_close(self):
1316 # A TemporaryFile can be closed many times without error
1317 f = tempfile.TemporaryFile()
1318 f.write(b'abc\n')
1319 f.close()
1320 f.close()
1321 f.close()
1322
1323 # How to test the mode and bufsize parameters?
1324 def test_mode_and_encoding(self):
1325
1326 def roundtrip(input, *args, **kwargs):
1327 with tempfile.TemporaryFile(*args, **kwargs) as fileobj:
1328 fileobj.write(input)
1329 fileobj.seek(0)
1330 self.assertEqual(input, fileobj.read())
1331
1332 roundtrip(b"1234", "w+b")
1333 roundtrip("abdc\n", "w+")
1334 roundtrip("\u039B", "w+", encoding="utf-16")
1335 roundtrip("foo\r\n", "w+", newline="")
Guido van Rossum0e548712002-08-09 16:14:33 +00001336
Victor Stinner1f99f9d2014-03-25 09:18:04 +01001337 def test_no_leak_fd(self):
1338 # Issue #21058: don't leak file descriptor when io.open() fails
1339 closed = []
Victor Stinnerc61c1702014-07-29 01:13:39 +02001340 os_close = os.close
Victor Stinner1f99f9d2014-03-25 09:18:04 +01001341 def close(fd):
1342 closed.append(fd)
Victor Stinnerc61c1702014-07-29 01:13:39 +02001343 os_close(fd)
Victor Stinner1f99f9d2014-03-25 09:18:04 +01001344
1345 with mock.patch('os.close', side_effect=close):
1346 with mock.patch('io.open', side_effect=ValueError):
1347 self.assertRaises(ValueError, tempfile.TemporaryFile)
1348 self.assertEqual(len(closed), 1)
1349
1350
Nick Coghlan543af752010-10-24 11:23:25 +00001351
1352# Helper for test_del_on_shutdown
1353class NulledModules:
1354 def __init__(self, *modules):
1355 self.refs = [mod.__dict__ for mod in modules]
1356 self.contents = [ref.copy() for ref in self.refs]
1357
1358 def __enter__(self):
1359 for d in self.refs:
1360 for key in d:
1361 d[key] = None
1362
1363 def __exit__(self, *exc_info):
1364 for d, c in zip(self.refs, self.contents):
1365 d.clear()
1366 d.update(c)
1367
CAM Gerlachbd2fa3c2021-03-14 13:06:56 -05001368
Antoine Pitroueab2a502012-03-10 16:34:40 +01001369class TestTemporaryDirectory(BaseTestCase):
Nick Coghlan543af752010-10-24 11:23:25 +00001370 """Test TemporaryDirectory()."""
1371
CAM Gerlachbd2fa3c2021-03-14 13:06:56 -05001372 def do_create(self, dir=None, pre="", suf="", recurse=1, dirs=1, files=1,
1373 ignore_cleanup_errors=False):
Nick Coghlan543af752010-10-24 11:23:25 +00001374 if dir is None:
1375 dir = tempfile.gettempdir()
CAM Gerlachbd2fa3c2021-03-14 13:06:56 -05001376 tmp = tempfile.TemporaryDirectory(
1377 dir=dir, prefix=pre, suffix=suf,
1378 ignore_cleanup_errors=ignore_cleanup_errors)
Nick Coghlan543af752010-10-24 11:23:25 +00001379 self.nameCheck(tmp.name, dir, pre, suf)
Serhiy Storchakae9b51c02019-05-31 11:30:37 +03001380 self.do_create2(tmp.name, recurse, dirs, files)
Nick Coghlan543af752010-10-24 11:23:25 +00001381 return tmp
1382
Serhiy Storchakae9b51c02019-05-31 11:30:37 +03001383 def do_create2(self, path, recurse=1, dirs=1, files=1):
1384 # Create subdirectories and some files
1385 if recurse:
1386 for i in range(dirs):
1387 name = os.path.join(path, "dir%d" % i)
1388 os.mkdir(name)
1389 self.do_create2(name, recurse-1, dirs, files)
1390 for i in range(files):
1391 with open(os.path.join(path, "test%d.txt" % i), "wb") as f:
1392 f.write(b"Hello world!")
1393
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001394 def test_mkdtemp_failure(self):
1395 # Check no additional exception if mkdtemp fails
1396 # Previously would raise AttributeError instead
Nick Coghlan3c54ea62010-12-13 03:02:43 +00001397 # (noted as part of Issue #10188)
1398 with tempfile.TemporaryDirectory() as nonexistent:
1399 pass
Serhiy Storchaka7451a722013-02-09 22:25:49 +02001400 with self.assertRaises(FileNotFoundError) as cm:
Nick Coghlan3c54ea62010-12-13 03:02:43 +00001401 tempfile.TemporaryDirectory(dir=nonexistent)
Serhiy Storchaka7451a722013-02-09 22:25:49 +02001402 self.assertEqual(cm.exception.errno, errno.ENOENT)
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001403
Nick Coghlan543af752010-10-24 11:23:25 +00001404 def test_explicit_cleanup(self):
1405 # A TemporaryDirectory is deleted when cleaned up
1406 dir = tempfile.mkdtemp()
1407 try:
1408 d = self.do_create(dir=dir)
1409 self.assertTrue(os.path.exists(d.name),
1410 "TemporaryDirectory %s does not exist" % d.name)
1411 d.cleanup()
1412 self.assertFalse(os.path.exists(d.name),
1413 "TemporaryDirectory %s exists after cleanup" % d.name)
1414 finally:
1415 os.rmdir(dir)
1416
CAM Gerlachbd2fa3c2021-03-14 13:06:56 -05001417 def test_explict_cleanup_ignore_errors(self):
1418 """Test that cleanup doesn't return an error when ignoring them."""
1419 with tempfile.TemporaryDirectory() as working_dir:
1420 temp_dir = self.do_create(
1421 dir=working_dir, ignore_cleanup_errors=True)
1422 temp_path = pathlib.Path(temp_dir.name)
1423 self.assertTrue(temp_path.exists(),
1424 f"TemporaryDirectory {temp_path!s} does not exist")
1425 with open(temp_path / "a_file.txt", "w+t") as open_file:
1426 open_file.write("Hello world!\n")
1427 temp_dir.cleanup()
1428 self.assertEqual(len(list(temp_path.glob("*"))),
1429 int(sys.platform.startswith("win")),
1430 "Unexpected number of files in "
1431 f"TemporaryDirectory {temp_path!s}")
1432 self.assertEqual(
1433 temp_path.exists(),
1434 sys.platform.startswith("win"),
1435 f"TemporaryDirectory {temp_path!s} existance state unexpected")
1436 temp_dir.cleanup()
1437 self.assertFalse(
1438 temp_path.exists(),
1439 f"TemporaryDirectory {temp_path!s} exists after cleanup")
1440
Hai Shi0c4f0f32020-06-30 21:46:31 +08001441 @os_helper.skip_unless_symlink
Charles-François Natalidef35432011-07-29 18:59:24 +02001442 def test_cleanup_with_symlink_to_a_directory(self):
1443 # cleanup() should not follow symlinks to directories (issue #12464)
1444 d1 = self.do_create()
Serhiy Storchaka99e033b2014-01-27 11:18:27 +02001445 d2 = self.do_create(recurse=0)
Charles-François Natalidef35432011-07-29 18:59:24 +02001446
1447 # Symlink d1/foo -> d2
1448 os.symlink(d2.name, os.path.join(d1.name, "foo"))
1449
1450 # This call to cleanup() should not follow the "foo" symlink
1451 d1.cleanup()
1452
1453 self.assertFalse(os.path.exists(d1.name),
1454 "TemporaryDirectory %s exists after cleanup" % d1.name)
1455 self.assertTrue(os.path.exists(d2.name),
1456 "Directory pointed to by a symlink was deleted")
Serhiy Storchakae9b51c02019-05-31 11:30:37 +03001457 self.assertEqual(os.listdir(d2.name), ['test0.txt'],
Charles-François Natalidef35432011-07-29 18:59:24 +02001458 "Contents of the directory pointed to by a symlink "
1459 "were deleted")
1460 d2.cleanup()
1461
Nick Coghlan543af752010-10-24 11:23:25 +00001462 @support.cpython_only
1463 def test_del_on_collection(self):
1464 # A TemporaryDirectory is deleted when garbage collected
1465 dir = tempfile.mkdtemp()
1466 try:
1467 d = self.do_create(dir=dir)
1468 name = d.name
1469 del d # Rely on refcounting to invoke __del__
1470 self.assertFalse(os.path.exists(name),
1471 "TemporaryDirectory %s exists after __del__" % name)
1472 finally:
1473 os.rmdir(dir)
1474
CAM Gerlachbd2fa3c2021-03-14 13:06:56 -05001475 @support.cpython_only
1476 def test_del_on_collection_ignore_errors(self):
1477 """Test that ignoring errors works when TemporaryDirectory is gced."""
1478 with tempfile.TemporaryDirectory() as working_dir:
1479 temp_dir = self.do_create(
1480 dir=working_dir, ignore_cleanup_errors=True)
1481 temp_path = pathlib.Path(temp_dir.name)
1482 self.assertTrue(temp_path.exists(),
1483 f"TemporaryDirectory {temp_path!s} does not exist")
1484 with open(temp_path / "a_file.txt", "w+t") as open_file:
1485 open_file.write("Hello world!\n")
1486 del temp_dir
1487 self.assertEqual(len(list(temp_path.glob("*"))),
1488 int(sys.platform.startswith("win")),
1489 "Unexpected number of files in "
1490 f"TemporaryDirectory {temp_path!s}")
1491 self.assertEqual(
1492 temp_path.exists(),
1493 sys.platform.startswith("win"),
1494 f"TemporaryDirectory {temp_path!s} existance state unexpected")
1495
Nick Coghlan543af752010-10-24 11:23:25 +00001496 def test_del_on_shutdown(self):
1497 # A TemporaryDirectory may be cleaned up during shutdown
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001498 with self.do_create() as dir:
Serhiy Storchakaa28632b2014-01-27 11:21:54 +02001499 for mod in ('builtins', 'os', 'shutil', 'sys', 'tempfile', 'warnings'):
Serhiy Storchaka99e033b2014-01-27 11:18:27 +02001500 code = """if True:
Serhiy Storchakaa28632b2014-01-27 11:21:54 +02001501 import builtins
Serhiy Storchaka99e033b2014-01-27 11:18:27 +02001502 import os
1503 import shutil
1504 import sys
1505 import tempfile
1506 import warnings
1507
1508 tmp = tempfile.TemporaryDirectory(dir={dir!r})
1509 sys.stdout.buffer.write(tmp.name.encode())
1510
1511 tmp2 = os.path.join(tmp.name, 'test_dir')
1512 os.mkdir(tmp2)
Serhiy Storchakae9b51c02019-05-31 11:30:37 +03001513 with open(os.path.join(tmp2, "test0.txt"), "w") as f:
Serhiy Storchaka99e033b2014-01-27 11:18:27 +02001514 f.write("Hello world!")
1515
1516 {mod}.tmp = tmp
1517
1518 warnings.filterwarnings("always", category=ResourceWarning)
1519 """.format(dir=dir, mod=mod)
1520 rc, out, err = script_helper.assert_python_ok("-c", code)
1521 tmp_name = out.decode().strip()
1522 self.assertFalse(os.path.exists(tmp_name),
1523 "TemporaryDirectory %s exists after cleanup" % tmp_name)
1524 err = err.decode('utf-8', 'backslashreplace')
1525 self.assertNotIn("Exception ", err)
Serhiy Storchakaa28632b2014-01-27 11:21:54 +02001526 self.assertIn("ResourceWarning: Implicitly cleaning up", err)
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001527
CAM Gerlachbd2fa3c2021-03-14 13:06:56 -05001528 def test_del_on_shutdown_ignore_errors(self):
1529 """Test ignoring errors works when a tempdir is gc'ed on shutdown."""
1530 with tempfile.TemporaryDirectory() as working_dir:
1531 code = """if True:
1532 import pathlib
1533 import sys
1534 import tempfile
1535 import warnings
1536
1537 temp_dir = tempfile.TemporaryDirectory(
1538 dir={working_dir!r}, ignore_cleanup_errors=True)
1539 sys.stdout.buffer.write(temp_dir.name.encode())
1540
1541 temp_dir_2 = pathlib.Path(temp_dir.name) / "test_dir"
1542 temp_dir_2.mkdir()
1543 with open(temp_dir_2 / "test0.txt", "w") as test_file:
1544 test_file.write("Hello world!")
1545 open_file = open(temp_dir_2 / "open_file.txt", "w")
1546 open_file.write("Hello world!")
1547
1548 warnings.filterwarnings("always", category=ResourceWarning)
1549 """.format(working_dir=working_dir)
1550 __, out, err = script_helper.assert_python_ok("-c", code)
1551 temp_path = pathlib.Path(out.decode().strip())
1552 self.assertEqual(len(list(temp_path.glob("*"))),
1553 int(sys.platform.startswith("win")),
1554 "Unexpected number of files in "
1555 f"TemporaryDirectory {temp_path!s}")
1556 self.assertEqual(
1557 temp_path.exists(),
1558 sys.platform.startswith("win"),
1559 f"TemporaryDirectory {temp_path!s} existance state unexpected")
1560 err = err.decode('utf-8', 'backslashreplace')
1561 self.assertNotIn("Exception", err)
1562 self.assertNotIn("Error", err)
1563 self.assertIn("ResourceWarning: Implicitly cleaning up", err)
1564
Serhiy Storchaka5e193ac2014-09-24 13:26:25 +03001565 def test_exit_on_shutdown(self):
1566 # Issue #22427
1567 with self.do_create() as dir:
1568 code = """if True:
1569 import sys
1570 import tempfile
1571 import warnings
1572
1573 def generator():
1574 with tempfile.TemporaryDirectory(dir={dir!r}) as tmp:
1575 yield tmp
1576 g = generator()
1577 sys.stdout.buffer.write(next(g).encode())
1578
1579 warnings.filterwarnings("always", category=ResourceWarning)
1580 """.format(dir=dir)
1581 rc, out, err = script_helper.assert_python_ok("-c", code)
1582 tmp_name = out.decode().strip()
1583 self.assertFalse(os.path.exists(tmp_name),
1584 "TemporaryDirectory %s exists after cleanup" % tmp_name)
1585 err = err.decode('utf-8', 'backslashreplace')
1586 self.assertNotIn("Exception ", err)
1587 self.assertIn("ResourceWarning: Implicitly cleaning up", err)
1588
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001589 def test_warnings_on_cleanup(self):
Serhiy Storchaka99e033b2014-01-27 11:18:27 +02001590 # ResourceWarning will be triggered by __del__
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001591 with self.do_create() as dir:
Serhiy Storchaka99e033b2014-01-27 11:18:27 +02001592 d = self.do_create(dir=dir, recurse=3)
1593 name = d.name
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001594
1595 # Check for the resource warning
Hai Shi0c4f0f32020-06-30 21:46:31 +08001596 with warnings_helper.check_warnings(('Implicitly',
1597 ResourceWarning),
1598 quiet=False):
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001599 warnings.filterwarnings("always", category=ResourceWarning)
Serhiy Storchaka99e033b2014-01-27 11:18:27 +02001600 del d
1601 support.gc_collect()
1602 self.assertFalse(os.path.exists(name),
1603 "TemporaryDirectory %s exists after __del__" % name)
Nick Coghlan543af752010-10-24 11:23:25 +00001604
1605 def test_multiple_close(self):
1606 # Can be cleaned-up many times without error
1607 d = self.do_create()
1608 d.cleanup()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +01001609 d.cleanup()
1610 d.cleanup()
Nick Coghlan543af752010-10-24 11:23:25 +00001611
1612 def test_context_manager(self):
1613 # Can be used as a context manager
1614 d = self.do_create()
1615 with d as name:
1616 self.assertTrue(os.path.exists(name))
1617 self.assertEqual(name, d.name)
1618 self.assertFalse(os.path.exists(name))
1619
Serhiy Storchakae9b51c02019-05-31 11:30:37 +03001620 def test_modes(self):
1621 for mode in range(8):
1622 mode <<= 6
1623 with self.subTest(mode=format(mode, '03o')):
1624 d = self.do_create(recurse=3, dirs=2, files=2)
1625 with d:
1626 # Change files and directories mode recursively.
1627 for root, dirs, files in os.walk(d.name, topdown=False):
1628 for name in files:
1629 os.chmod(os.path.join(root, name), mode)
1630 os.chmod(root, mode)
1631 d.cleanup()
1632 self.assertFalse(os.path.exists(d.name))
1633
1634 @unittest.skipUnless(hasattr(os, 'chflags'), 'requires os.lchflags')
1635 def test_flags(self):
1636 flags = stat.UF_IMMUTABLE | stat.UF_NOUNLINK
1637 d = self.do_create(recurse=3, dirs=2, files=2)
1638 with d:
1639 # Change files and directories flags recursively.
1640 for root, dirs, files in os.walk(d.name, topdown=False):
1641 for name in files:
1642 os.chflags(os.path.join(root, name), flags)
1643 os.chflags(root, flags)
1644 d.cleanup()
1645 self.assertFalse(os.path.exists(d.name))
1646
Nick Coghlan543af752010-10-24 11:23:25 +00001647
Guido van Rossum0e548712002-08-09 16:14:33 +00001648if __name__ == "__main__":
Zachary Ware38c707e2015-04-13 15:00:43 -05001649 unittest.main()