blob: 3c0b9a74e5c8eab541d71554990043edaca15bec [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
Antoine Pitrou4558bad2011-11-25 21:28:15 +01006import signal
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
Antoine Pitrou17c93262013-12-21 22:14:56 +010012import weakref
Victor Stinner1f99f9d2014-03-25 09:18:04 +010013from unittest import mock
Tim Petersc57a2852001-10-29 21:46:08 +000014
Guido van Rossum0e548712002-08-09 16:14:33 +000015import unittest
Berker Peksagce643912015-05-06 06:33:17 +030016from test import support
17from test.support import script_helper
Guido van Rossum0e548712002-08-09 16:14:33 +000018
Fred Drake7633d232002-10-17 22:09:03 +000019
Guido van Rossum0e548712002-08-09 16:14:33 +000020has_textmode = (tempfile._text_openflags != tempfile._bin_openflags)
Guido van Rossum78741062002-08-17 11:41:01 +000021has_spawnl = hasattr(os, 'spawnl')
Guido van Rossum0e548712002-08-09 16:14:33 +000022
Neal Norwitz68ee0122002-08-16 19:28:59 +000023# TEST_FILES may need to be tweaked for systems depending on the maximum
24# number of files that can be opened at one time (see ulimit -n)
Victor Stinner9c3de4a2011-08-17 20:49:41 +020025if sys.platform.startswith('openbsd'):
Martin v. Löwis99968282004-09-15 06:02:54 +000026 TEST_FILES = 48
Jack Jansence921472003-01-08 16:30:34 +000027else:
28 TEST_FILES = 100
Neal Norwitz68ee0122002-08-16 19:28:59 +000029
Guido van Rossum0e548712002-08-09 16:14:33 +000030# This is organized as one test for each chunk of code in tempfile.py,
31# in order of their appearance in the file. Testing which requires
32# threads is not done here.
33
Gregory P. Smithad577b92015-05-22 16:18:14 -070034class TestLowLevelInternals(unittest.TestCase):
35 def test_infer_return_type_singles(self):
36 self.assertIs(str, tempfile._infer_return_type(''))
37 self.assertIs(bytes, tempfile._infer_return_type(b''))
38 self.assertIs(str, tempfile._infer_return_type(None))
39
40 def test_infer_return_type_multiples(self):
41 self.assertIs(str, tempfile._infer_return_type('', ''))
42 self.assertIs(bytes, tempfile._infer_return_type(b'', b''))
43 with self.assertRaises(TypeError):
44 tempfile._infer_return_type('', b'')
45 with self.assertRaises(TypeError):
46 tempfile._infer_return_type(b'', '')
47
48 def test_infer_return_type_multiples_and_none(self):
49 self.assertIs(str, tempfile._infer_return_type(None, ''))
50 self.assertIs(str, tempfile._infer_return_type('', None))
51 self.assertIs(str, tempfile._infer_return_type(None, None))
52 self.assertIs(bytes, tempfile._infer_return_type(b'', None))
53 self.assertIs(bytes, tempfile._infer_return_type(None, b''))
54 with self.assertRaises(TypeError):
55 tempfile._infer_return_type('', None, b'')
56 with self.assertRaises(TypeError):
57 tempfile._infer_return_type(b'', None, '')
58
59
Guido van Rossum0e548712002-08-09 16:14:33 +000060# Common functionality.
Gregory P. Smithad577b92015-05-22 16:18:14 -070061
Antoine Pitroueab2a502012-03-10 16:34:40 +010062class BaseTestCase(unittest.TestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +000063
Victor Stinner97869102013-08-14 01:28:28 +020064 str_check = re.compile(r"^[a-z0-9_-]{8}$")
Gregory P. Smithad577b92015-05-22 16:18:14 -070065 b_check = re.compile(br"^[a-z0-9_-]{8}$")
Guido van Rossum0e548712002-08-09 16:14:33 +000066
Brett Cannone1adece2010-03-20 22:19:55 +000067 def setUp(self):
68 self._warnings_manager = support.check_warnings()
69 self._warnings_manager.__enter__()
70 warnings.filterwarnings("ignore", category=RuntimeWarning,
71 message="mktemp", module=__name__)
72
73 def tearDown(self):
74 self._warnings_manager.__exit__(None, None, None)
75
Guido van Rossum0e548712002-08-09 16:14:33 +000076 def nameCheck(self, name, dir, pre, suf):
77 (ndir, nbase) = os.path.split(name)
78 npre = nbase[:len(pre)]
79 nsuf = nbase[len(nbase)-len(suf):]
80
Gregory P. Smithad577b92015-05-22 16:18:14 -070081 if dir is not None:
82 self.assertIs(type(name), str if type(dir) is str else bytes,
83 "unexpected return type")
84 if pre is not None:
85 self.assertIs(type(name), str if type(pre) is str else bytes,
86 "unexpected return type")
87 if suf is not None:
88 self.assertIs(type(name), str if type(suf) is str else bytes,
89 "unexpected return type")
90 if (dir, pre, suf) == (None, None, None):
91 self.assertIs(type(name), str, "default return type must be str")
92
Martin v. Löwisd6625482003-10-12 17:37:01 +000093 # check for equality of the absolute paths!
94 self.assertEqual(os.path.abspath(ndir), os.path.abspath(dir),
Gregory P. Smithad577b92015-05-22 16:18:14 -070095 "file %r not in directory %r" % (name, dir))
Guido van Rossum0e548712002-08-09 16:14:33 +000096 self.assertEqual(npre, pre,
Gregory P. Smithad577b92015-05-22 16:18:14 -070097 "file %r does not begin with %r" % (nbase, pre))
Guido van Rossum0e548712002-08-09 16:14:33 +000098 self.assertEqual(nsuf, suf,
Gregory P. Smithad577b92015-05-22 16:18:14 -070099 "file %r does not end with %r" % (nbase, suf))
Guido van Rossum0e548712002-08-09 16:14:33 +0000100
101 nbase = nbase[len(pre):len(nbase)-len(suf)]
Gregory P. Smithad577b92015-05-22 16:18:14 -0700102 check = self.str_check if isinstance(nbase, str) else self.b_check
103 self.assertTrue(check.match(nbase),
104 "random characters %r do not match %r"
105 % (nbase, check.pattern))
Guido van Rossum0e548712002-08-09 16:14:33 +0000106
Guido van Rossum0e548712002-08-09 16:14:33 +0000107
Antoine Pitroueab2a502012-03-10 16:34:40 +0100108class TestExports(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000109 def test_exports(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000110 # There are no surprising symbols in the tempfile module
Guido van Rossum0e548712002-08-09 16:14:33 +0000111 dict = tempfile.__dict__
112
113 expected = {
114 "NamedTemporaryFile" : 1,
115 "TemporaryFile" : 1,
116 "mkstemp" : 1,
117 "mkdtemp" : 1,
118 "mktemp" : 1,
119 "TMP_MAX" : 1,
120 "gettempprefix" : 1,
Gregory P. Smithad577b92015-05-22 16:18:14 -0700121 "gettempprefixb" : 1,
Guido van Rossum0e548712002-08-09 16:14:33 +0000122 "gettempdir" : 1,
Gregory P. Smithad577b92015-05-22 16:18:14 -0700123 "gettempdirb" : 1,
Guido van Rossum0e548712002-08-09 16:14:33 +0000124 "tempdir" : 1,
Guido van Rossumd8faa362007-04-27 19:54:29 +0000125 "template" : 1,
Nick Coghlan543af752010-10-24 11:23:25 +0000126 "SpooledTemporaryFile" : 1,
127 "TemporaryDirectory" : 1,
Guido van Rossum0e548712002-08-09 16:14:33 +0000128 }
129
130 unexp = []
131 for key in dict:
132 if key[0] != '_' and key not in expected:
133 unexp.append(key)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000134 self.assertTrue(len(unexp) == 0,
Guido van Rossum0e548712002-08-09 16:14:33 +0000135 "unexpected keys: %s" % unexp)
136
Guido van Rossum0e548712002-08-09 16:14:33 +0000137
Antoine Pitroueab2a502012-03-10 16:34:40 +0100138class TestRandomNameSequence(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000139 """Test the internal iterator object _RandomNameSequence."""
140
141 def setUp(self):
142 self.r = tempfile._RandomNameSequence()
Brett Cannone1adece2010-03-20 22:19:55 +0000143 super().setUp()
Guido van Rossum0e548712002-08-09 16:14:33 +0000144
145 def test_get_six_char_str(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000146 # _RandomNameSequence returns a six-character string
Georg Brandla18af4e2007-04-21 15:47:16 +0000147 s = next(self.r)
Guido van Rossum0e548712002-08-09 16:14:33 +0000148 self.nameCheck(s, '', '', '')
149
150 def test_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000151 # _RandomNameSequence returns no duplicate strings (stochastic)
Guido van Rossum0e548712002-08-09 16:14:33 +0000152
153 dict = {}
154 r = self.r
Guido van Rossum805365e2007-05-07 22:24:25 +0000155 for i in range(TEST_FILES):
Georg Brandla18af4e2007-04-21 15:47:16 +0000156 s = next(r)
Guido van Rossum0e548712002-08-09 16:14:33 +0000157 self.nameCheck(s, '', '', '')
Ezio Melottib58e0bd2010-01-23 15:40:09 +0000158 self.assertNotIn(s, dict)
Guido van Rossum0e548712002-08-09 16:14:33 +0000159 dict[s] = 1
160
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000161 def supports_iter(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000162 # _RandomNameSequence supports the iterator protocol
Guido van Rossum0e548712002-08-09 16:14:33 +0000163
164 i = 0
165 r = self.r
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100166 for s in r:
167 i += 1
168 if i == 20:
169 break
Guido van Rossum0e548712002-08-09 16:14:33 +0000170
Antoine Pitrou4558bad2011-11-25 21:28:15 +0100171 @unittest.skipUnless(hasattr(os, 'fork'),
172 "os.fork is required for this test")
173 def test_process_awareness(self):
174 # ensure that the random source differs between
175 # child and parent.
176 read_fd, write_fd = os.pipe()
177 pid = None
178 try:
179 pid = os.fork()
180 if not pid:
Victor Stinner6c8c2942017-08-10 13:05:06 +0200181 # child process
Antoine Pitrou4558bad2011-11-25 21:28:15 +0100182 os.close(read_fd)
183 os.write(write_fd, next(self.r).encode("ascii"))
184 os.close(write_fd)
185 # bypass the normal exit handlers- leave those to
186 # the parent.
187 os._exit(0)
Victor Stinner6c8c2942017-08-10 13:05:06 +0200188
189 # parent process
Antoine Pitrou4558bad2011-11-25 21:28:15 +0100190 parent_value = next(self.r)
191 child_value = os.read(read_fd, len(parent_value)).decode("ascii")
192 finally:
193 if pid:
194 # best effort to ensure the process can't bleed out
195 # via any bugs above
196 try:
197 os.kill(pid, signal.SIGKILL)
Andrew Svetlov3438fa42012-12-17 23:35:18 +0200198 except OSError:
Antoine Pitrou4558bad2011-11-25 21:28:15 +0100199 pass
Victor Stinner6c8c2942017-08-10 13:05:06 +0200200
201 # Read the process exit status to avoid zombie process
202 os.waitpid(pid, 0)
203
Antoine Pitrou4558bad2011-11-25 21:28:15 +0100204 os.close(read_fd)
205 os.close(write_fd)
206 self.assertNotEqual(child_value, parent_value)
207
208
Guido van Rossum0e548712002-08-09 16:14:33 +0000209
Antoine Pitroueab2a502012-03-10 16:34:40 +0100210class TestCandidateTempdirList(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000211 """Test the internal function _candidate_tempdir_list."""
212
213 def test_nonempty_list(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000214 # _candidate_tempdir_list returns a nonempty list of strings
Guido van Rossum0e548712002-08-09 16:14:33 +0000215
216 cand = tempfile._candidate_tempdir_list()
217
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000218 self.assertFalse(len(cand) == 0)
Guido van Rossum0e548712002-08-09 16:14:33 +0000219 for c in cand:
Ezio Melottie9615932010-01-24 19:26:24 +0000220 self.assertIsInstance(c, str)
Guido van Rossum0e548712002-08-09 16:14:33 +0000221
222 def test_wanted_dirs(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000223 # _candidate_tempdir_list contains the expected directories
Guido van Rossum0e548712002-08-09 16:14:33 +0000224
225 # Make sure the interesting environment variables are all set.
Hirokazu Yamamoto71959632009-04-27 01:44:28 +0000226 with support.EnvironmentVarGuard() as env:
Guido van Rossum0e548712002-08-09 16:14:33 +0000227 for envname in 'TMPDIR', 'TEMP', 'TMP':
228 dirname = os.getenv(envname)
229 if not dirname:
Walter Dörwald155374d2009-05-01 19:58:58 +0000230 env[envname] = os.path.abspath(envname)
Guido van Rossum0e548712002-08-09 16:14:33 +0000231
232 cand = tempfile._candidate_tempdir_list()
233
234 for envname in 'TMPDIR', 'TEMP', 'TMP':
235 dirname = os.getenv(envname)
236 if not dirname: raise ValueError
Benjamin Peterson577473f2010-01-19 00:09:57 +0000237 self.assertIn(dirname, cand)
Guido van Rossum0e548712002-08-09 16:14:33 +0000238
239 try:
240 dirname = os.getcwd()
Andrew Svetlovad28c7f2012-12-18 22:02:39 +0200241 except (AttributeError, OSError):
Guido van Rossum0e548712002-08-09 16:14:33 +0000242 dirname = os.curdir
243
Benjamin Peterson577473f2010-01-19 00:09:57 +0000244 self.assertIn(dirname, cand)
Guido van Rossum0e548712002-08-09 16:14:33 +0000245
246 # Not practical to try to verify the presence of OS-specific
247 # paths in this list.
Guido van Rossum0e548712002-08-09 16:14:33 +0000248
Guido van Rossum0e548712002-08-09 16:14:33 +0000249
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200250# We test _get_default_tempdir some more by testing gettempdir.
Guido van Rossum0e548712002-08-09 16:14:33 +0000251
Serhiy Storchakaff7fef92013-02-13 00:37:29 +0200252class TestGetDefaultTempdir(BaseTestCase):
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200253 """Test _get_default_tempdir()."""
254
255 def test_no_files_left_behind(self):
256 # use a private empty directory
257 with tempfile.TemporaryDirectory() as our_temp_directory:
258 # force _get_default_tempdir() to consider our empty directory
259 def our_candidate_list():
260 return [our_temp_directory]
261
262 with support.swap_attr(tempfile, "_candidate_tempdir_list",
263 our_candidate_list):
264 # verify our directory is empty after _get_default_tempdir()
265 tempfile._get_default_tempdir()
266 self.assertEqual(os.listdir(our_temp_directory), [])
267
268 def raise_OSError(*args, **kwargs):
Serhiy Storchakaff7fef92013-02-13 00:37:29 +0200269 raise OSError()
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200270
271 with support.swap_attr(io, "open", raise_OSError):
272 # test again with failing io.open()
Serhiy Storchakaff7fef92013-02-13 00:37:29 +0200273 with self.assertRaises(FileNotFoundError):
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200274 tempfile._get_default_tempdir()
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200275 self.assertEqual(os.listdir(our_temp_directory), [])
276
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200277 def bad_writer(*args, **kwargs):
Serhiy Storchakad1a1def2017-04-28 19:17:26 +0300278 fp = orig_open(*args, **kwargs)
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200279 fp.write = raise_OSError
280 return fp
281
Serhiy Storchakad1a1def2017-04-28 19:17:26 +0300282 with support.swap_attr(io, "open", bad_writer) as orig_open:
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200283 # test again with failing write()
Serhiy Storchakaff7fef92013-02-13 00:37:29 +0200284 with self.assertRaises(FileNotFoundError):
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200285 tempfile._get_default_tempdir()
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200286 self.assertEqual(os.listdir(our_temp_directory), [])
Guido van Rossum0e548712002-08-09 16:14:33 +0000287
288
Antoine Pitroueab2a502012-03-10 16:34:40 +0100289class TestGetCandidateNames(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000290 """Test the internal function _get_candidate_names."""
291
292 def test_retval(self):
Victor Stinner1e62bf12017-04-19 22:59:51 +0200293 # _get_candidate_names returns a _RandomNameSequence object
Guido van Rossum0e548712002-08-09 16:14:33 +0000294 obj = tempfile._get_candidate_names()
Victor Stinner1e62bf12017-04-19 22:59:51 +0200295 self.assertIsInstance(obj, tempfile._RandomNameSequence)
Guido van Rossum0e548712002-08-09 16:14:33 +0000296
297 def test_same_thing(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000298 # _get_candidate_names always returns the same object
Guido van Rossum0e548712002-08-09 16:14:33 +0000299 a = tempfile._get_candidate_names()
300 b = tempfile._get_candidate_names()
301
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000302 self.assertTrue(a is b)
Guido van Rossum0e548712002-08-09 16:14:33 +0000303
Guido van Rossum0e548712002-08-09 16:14:33 +0000304
Eli Benderskyaa04f9a2013-09-13 05:28:20 -0700305@contextlib.contextmanager
306def _inside_empty_temp_dir():
307 dir = tempfile.mkdtemp()
308 try:
309 with support.swap_attr(tempfile, 'tempdir', dir):
310 yield
311 finally:
312 support.rmtree(dir)
313
314
315def _mock_candidate_names(*names):
316 return support.swap_attr(tempfile,
317 '_get_candidate_names',
318 lambda: iter(names))
319
320
Serhiy Storchaka5d6b7b12015-05-20 00:11:48 +0300321class TestBadTempdir:
322
323 def test_read_only_directory(self):
324 with _inside_empty_temp_dir():
325 oldmode = mode = os.stat(tempfile.tempdir).st_mode
326 mode &= ~(stat.S_IWUSR | stat.S_IWGRP | stat.S_IWOTH)
327 os.chmod(tempfile.tempdir, mode)
328 try:
329 if os.access(tempfile.tempdir, os.W_OK):
330 self.skipTest("can't set the directory read-only")
331 with self.assertRaises(PermissionError):
332 self.make_temp()
333 self.assertEqual(os.listdir(tempfile.tempdir), [])
334 finally:
335 os.chmod(tempfile.tempdir, oldmode)
336
337 def test_nonexisting_directory(self):
338 with _inside_empty_temp_dir():
339 tempdir = os.path.join(tempfile.tempdir, 'nonexistent')
340 with support.swap_attr(tempfile, 'tempdir', tempdir):
341 with self.assertRaises(FileNotFoundError):
342 self.make_temp()
343
344 def test_non_directory(self):
345 with _inside_empty_temp_dir():
346 tempdir = os.path.join(tempfile.tempdir, 'file')
347 open(tempdir, 'wb').close()
348 with support.swap_attr(tempfile, 'tempdir', tempdir):
349 with self.assertRaises((NotADirectoryError, FileNotFoundError)):
350 self.make_temp()
351
352
353class TestMkstempInner(TestBadTempdir, BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000354 """Test the internal function _mkstemp_inner."""
355
356 class mkstemped:
357 _bflags = tempfile._bin_openflags
358 _tflags = tempfile._text_openflags
359 _close = os.close
360 _unlink = os.unlink
361
362 def __init__(self, dir, pre, suf, bin):
363 if bin: flags = self._bflags
364 else: flags = self._tflags
365
Gregory P. Smithad577b92015-05-22 16:18:14 -0700366 output_type = tempfile._infer_return_type(dir, pre, suf)
367 (self.fd, self.name) = tempfile._mkstemp_inner(dir, pre, suf, flags, output_type)
Guido van Rossum0e548712002-08-09 16:14:33 +0000368
369 def write(self, str):
370 os.write(self.fd, str)
371
372 def __del__(self):
373 self._close(self.fd)
374 self._unlink(self.name)
Tim Petersa0d55de2002-08-09 18:01:01 +0000375
Gregory P. Smithad577b92015-05-22 16:18:14 -0700376 def do_create(self, dir=None, pre=None, suf=None, bin=1):
377 output_type = tempfile._infer_return_type(dir, pre, suf)
Guido van Rossum0e548712002-08-09 16:14:33 +0000378 if dir is None:
Gregory P. Smithad577b92015-05-22 16:18:14 -0700379 if output_type is str:
380 dir = tempfile.gettempdir()
381 else:
382 dir = tempfile.gettempdirb()
383 if pre is None:
384 pre = output_type()
385 if suf is None:
386 suf = output_type()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100387 file = self.mkstemped(dir, pre, suf, bin)
Guido van Rossum0e548712002-08-09 16:14:33 +0000388
389 self.nameCheck(file.name, dir, pre, suf)
390 return file
391
392 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000393 # _mkstemp_inner can create files
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000394 self.do_create().write(b"blat")
395 self.do_create(pre="a").write(b"blat")
396 self.do_create(suf="b").write(b"blat")
397 self.do_create(pre="a", suf="b").write(b"blat")
398 self.do_create(pre="aa", suf=".txt").write(b"blat")
Guido van Rossum0e548712002-08-09 16:14:33 +0000399
Gregory P. Smithad577b92015-05-22 16:18:14 -0700400 def test_basic_with_bytes_names(self):
401 # _mkstemp_inner can create files when given name parts all
402 # specified as bytes.
403 dir_b = tempfile.gettempdirb()
404 self.do_create(dir=dir_b, suf=b"").write(b"blat")
405 self.do_create(dir=dir_b, pre=b"a").write(b"blat")
406 self.do_create(dir=dir_b, suf=b"b").write(b"blat")
407 self.do_create(dir=dir_b, pre=b"a", suf=b"b").write(b"blat")
408 self.do_create(dir=dir_b, pre=b"aa", suf=b".txt").write(b"blat")
409 # Can't mix str & binary types in the args.
410 with self.assertRaises(TypeError):
411 self.do_create(dir="", suf=b"").write(b"blat")
412 with self.assertRaises(TypeError):
413 self.do_create(dir=dir_b, pre="").write(b"blat")
414 with self.assertRaises(TypeError):
415 self.do_create(dir=dir_b, pre=b"", suf="").write(b"blat")
416
Guido van Rossum0e548712002-08-09 16:14:33 +0000417 def test_basic_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000418 # _mkstemp_inner can create many files (stochastic)
Guido van Rossum805365e2007-05-07 22:24:25 +0000419 extant = list(range(TEST_FILES))
Guido van Rossum0e548712002-08-09 16:14:33 +0000420 for i in extant:
421 extant[i] = self.do_create(pre="aa")
422
423 def test_choose_directory(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000424 # _mkstemp_inner can create files in a user-selected directory
Guido van Rossum0e548712002-08-09 16:14:33 +0000425 dir = tempfile.mkdtemp()
426 try:
Antoine Pitrou9cadb1b2008-09-15 23:02:56 +0000427 self.do_create(dir=dir).write(b"blat")
Guido van Rossum0e548712002-08-09 16:14:33 +0000428 finally:
429 os.rmdir(dir)
430
431 def test_file_mode(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000432 # _mkstemp_inner creates files with the proper mode
Guido van Rossum0e548712002-08-09 16:14:33 +0000433
434 file = self.do_create()
435 mode = stat.S_IMODE(os.stat(file.name).st_mode)
Guido van Rossumcd16bf62007-06-13 18:07:49 +0000436 expected = 0o600
Jesus Cea4791a242012-10-05 03:15:39 +0200437 if sys.platform == 'win32':
Tim Petersca3ac7f2002-08-09 18:13:51 +0000438 # There's no distinction among 'user', 'group' and 'world';
439 # replicate the 'user' bits.
440 user = expected >> 6
441 expected = user * (1 + 8 + 64)
442 self.assertEqual(mode, expected)
Guido van Rossum0e548712002-08-09 16:14:33 +0000443
Zachary Ware9fe6d862013-12-08 00:20:35 -0600444 @unittest.skipUnless(has_spawnl, 'os.spawnl not available')
Guido van Rossum0e548712002-08-09 16:14:33 +0000445 def test_noinherit(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000446 # _mkstemp_inner file handles are not inherited by child processes
Guido van Rossum0e548712002-08-09 16:14:33 +0000447
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000448 if support.verbose:
Guido van Rossum78741062002-08-17 11:41:01 +0000449 v="v"
450 else:
451 v="q"
452
Guido van Rossum0e548712002-08-09 16:14:33 +0000453 file = self.do_create()
Victor Stinnerdaf45552013-08-28 00:53:59 +0200454 self.assertEqual(os.get_inheritable(file.fd), False)
Guido van Rossum78741062002-08-17 11:41:01 +0000455 fd = "%d" % file.fd
456
457 try:
458 me = __file__
459 except NameError:
460 me = sys.argv[0]
Guido van Rossum0e548712002-08-09 16:14:33 +0000461
462 # We have to exec something, so that FD_CLOEXEC will take
Guido van Rossum78741062002-08-17 11:41:01 +0000463 # effect. The core of this test is therefore in
464 # tf_inherit_check.py, which see.
465 tester = os.path.join(os.path.dirname(os.path.abspath(me)),
466 "tf_inherit_check.py")
Guido van Rossum0e548712002-08-09 16:14:33 +0000467
Martin v. Löwisd4210bc2003-10-23 15:55:28 +0000468 # On Windows a spawn* /path/ with embedded spaces shouldn't be quoted,
469 # but an arg with embedded spaces should be decorated with double
470 # quotes on each end
Jesus Cea4791a242012-10-05 03:15:39 +0200471 if sys.platform == 'win32':
Martin v. Löwisd4210bc2003-10-23 15:55:28 +0000472 decorated = '"%s"' % sys.executable
473 tester = '"%s"' % tester
474 else:
475 decorated = sys.executable
476
477 retval = os.spawnl(os.P_WAIT, sys.executable, decorated, tester, v, fd)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000478 self.assertFalse(retval < 0,
Guido van Rossum78741062002-08-17 11:41:01 +0000479 "child process caught fatal signal %d" % -retval)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000480 self.assertFalse(retval > 0, "child process reports failure %d"%retval)
Guido van Rossum0e548712002-08-09 16:14:33 +0000481
Zachary Ware9fe6d862013-12-08 00:20:35 -0600482 @unittest.skipUnless(has_textmode, "text mode not available")
Guido van Rossum0e548712002-08-09 16:14:33 +0000483 def test_textmode(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000484 # _mkstemp_inner can create files in text mode
Guido van Rossum0e548712002-08-09 16:14:33 +0000485
Amaury Forgeot d'Arc7d0bddd2009-11-30 00:08:56 +0000486 # A text file is truncated at the first Ctrl+Z byte
487 f = self.do_create(bin=0)
488 f.write(b"blat\x1a")
489 f.write(b"extra\n")
490 os.lseek(f.fd, 0, os.SEEK_SET)
Ezio Melottib3aedd42010-11-20 19:04:17 +0000491 self.assertEqual(os.read(f.fd, 20), b"blat")
Guido van Rossum0e548712002-08-09 16:14:33 +0000492
Serhiy Storchaka5d6b7b12015-05-20 00:11:48 +0300493 def make_temp(self):
Eli Benderskyaa04f9a2013-09-13 05:28:20 -0700494 return tempfile._mkstemp_inner(tempfile.gettempdir(),
Gregory P. Smithad577b92015-05-22 16:18:14 -0700495 tempfile.gettempprefix(),
Eli Benderskyaa04f9a2013-09-13 05:28:20 -0700496 '',
Gregory P. Smithad577b92015-05-22 16:18:14 -0700497 tempfile._bin_openflags,
498 str)
Eli Benderskyaa04f9a2013-09-13 05:28:20 -0700499
500 def test_collision_with_existing_file(self):
501 # _mkstemp_inner tries another name when a file with
502 # the chosen name already exists
503 with _inside_empty_temp_dir(), \
504 _mock_candidate_names('aaa', 'aaa', 'bbb'):
Serhiy Storchaka5d6b7b12015-05-20 00:11:48 +0300505 (fd1, name1) = self.make_temp()
Eli Benderskyaa04f9a2013-09-13 05:28:20 -0700506 os.close(fd1)
507 self.assertTrue(name1.endswith('aaa'))
508
Serhiy Storchaka5d6b7b12015-05-20 00:11:48 +0300509 (fd2, name2) = self.make_temp()
Eli Benderskyaa04f9a2013-09-13 05:28:20 -0700510 os.close(fd2)
511 self.assertTrue(name2.endswith('bbb'))
512
Eli Benderskyf315df32013-09-06 06:11:19 -0700513 def test_collision_with_existing_directory(self):
514 # _mkstemp_inner tries another name when a directory with
515 # the chosen name already exists
Eli Benderskyaa04f9a2013-09-13 05:28:20 -0700516 with _inside_empty_temp_dir(), \
517 _mock_candidate_names('aaa', 'aaa', 'bbb'):
518 dir = tempfile.mkdtemp()
519 self.assertTrue(dir.endswith('aaa'))
Eli Benderskyf315df32013-09-06 06:11:19 -0700520
Serhiy Storchaka5d6b7b12015-05-20 00:11:48 +0300521 (fd, name) = self.make_temp()
Eli Benderskyaa04f9a2013-09-13 05:28:20 -0700522 os.close(fd)
523 self.assertTrue(name.endswith('bbb'))
Eli Benderskyf315df32013-09-06 06:11:19 -0700524
Guido van Rossum0e548712002-08-09 16:14:33 +0000525
Antoine Pitroueab2a502012-03-10 16:34:40 +0100526class TestGetTempPrefix(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000527 """Test gettempprefix()."""
528
529 def test_sane_template(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000530 # gettempprefix returns a nonempty prefix string
Guido van Rossum0e548712002-08-09 16:14:33 +0000531 p = tempfile.gettempprefix()
532
Ezio Melottie9615932010-01-24 19:26:24 +0000533 self.assertIsInstance(p, str)
Gregory P. Smithad577b92015-05-22 16:18:14 -0700534 self.assertGreater(len(p), 0)
535
536 pb = tempfile.gettempprefixb()
537
538 self.assertIsInstance(pb, bytes)
539 self.assertGreater(len(pb), 0)
Guido van Rossum0e548712002-08-09 16:14:33 +0000540
541 def test_usable_template(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000542 # gettempprefix returns a usable prefix string
Guido van Rossum0e548712002-08-09 16:14:33 +0000543
544 # Create a temp directory, avoiding use of the prefix.
545 # Then attempt to create a file whose name is
546 # prefix + 'xxxxxx.xxx' in that directory.
547 p = tempfile.gettempprefix() + "xxxxxx.xxx"
548 d = tempfile.mkdtemp(prefix="")
549 try:
550 p = os.path.join(d, p)
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100551 fd = os.open(p, os.O_RDWR | os.O_CREAT)
Guido van Rossum0e548712002-08-09 16:14:33 +0000552 os.close(fd)
553 os.unlink(p)
554 finally:
555 os.rmdir(d)
556
Guido van Rossum0e548712002-08-09 16:14:33 +0000557
Antoine Pitroueab2a502012-03-10 16:34:40 +0100558class TestGetTempDir(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000559 """Test gettempdir()."""
560
561 def test_directory_exists(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000562 # gettempdir returns a directory which exists
Guido van Rossum0e548712002-08-09 16:14:33 +0000563
Gregory P. Smithad577b92015-05-22 16:18:14 -0700564 for d in (tempfile.gettempdir(), tempfile.gettempdirb()):
565 self.assertTrue(os.path.isabs(d) or d == os.curdir,
566 "%r is not an absolute path" % d)
567 self.assertTrue(os.path.isdir(d),
568 "%r is not a directory" % d)
Guido van Rossum0e548712002-08-09 16:14:33 +0000569
570 def test_directory_writable(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000571 # gettempdir returns a directory writable by the user
Guido van Rossum0e548712002-08-09 16:14:33 +0000572
573 # sneaky: just instantiate a NamedTemporaryFile, which
574 # defaults to writing into the directory returned by
575 # gettempdir.
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100576 file = tempfile.NamedTemporaryFile()
577 file.write(b"blat")
578 file.close()
Guido van Rossum0e548712002-08-09 16:14:33 +0000579
580 def test_same_thing(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000581 # gettempdir always returns the same object
Guido van Rossum0e548712002-08-09 16:14:33 +0000582 a = tempfile.gettempdir()
583 b = tempfile.gettempdir()
Gregory P. Smithad577b92015-05-22 16:18:14 -0700584 c = tempfile.gettempdirb()
Guido van Rossum0e548712002-08-09 16:14:33 +0000585
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000586 self.assertTrue(a is b)
Gregory P. Smithad577b92015-05-22 16:18:14 -0700587 self.assertNotEqual(type(a), type(c))
588 self.assertEqual(a, os.fsdecode(c))
Guido van Rossum0e548712002-08-09 16:14:33 +0000589
Tim Golden6d09f092013-10-25 18:38:16 +0100590 def test_case_sensitive(self):
591 # gettempdir should not flatten its case
592 # even on a case-insensitive file system
593 case_sensitive_tempdir = tempfile.mkdtemp("-Temp")
594 _tempdir, tempfile.tempdir = tempfile.tempdir, None
595 try:
596 with support.EnvironmentVarGuard() as env:
597 # Fake the first env var which is checked as a candidate
598 env["TMPDIR"] = case_sensitive_tempdir
599 self.assertEqual(tempfile.gettempdir(), case_sensitive_tempdir)
600 finally:
601 tempfile.tempdir = _tempdir
602 support.rmdir(case_sensitive_tempdir)
603
Guido van Rossum0e548712002-08-09 16:14:33 +0000604
Antoine Pitroueab2a502012-03-10 16:34:40 +0100605class TestMkstemp(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000606 """Test mkstemp()."""
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000607
Gregory P. Smithad577b92015-05-22 16:18:14 -0700608 def do_create(self, dir=None, pre=None, suf=None):
609 output_type = tempfile._infer_return_type(dir, pre, suf)
Guido van Rossum0e548712002-08-09 16:14:33 +0000610 if dir is None:
Gregory P. Smithad577b92015-05-22 16:18:14 -0700611 if output_type is str:
612 dir = tempfile.gettempdir()
613 else:
614 dir = tempfile.gettempdirb()
615 if pre is None:
616 pre = output_type()
617 if suf is None:
618 suf = output_type()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100619 (fd, name) = tempfile.mkstemp(dir=dir, prefix=pre, suffix=suf)
620 (ndir, nbase) = os.path.split(name)
621 adir = os.path.abspath(dir)
622 self.assertEqual(adir, ndir,
623 "Directory '%s' incorrectly returned as '%s'" % (adir, ndir))
Guido van Rossum0e548712002-08-09 16:14:33 +0000624
625 try:
626 self.nameCheck(name, dir, pre, suf)
627 finally:
628 os.close(fd)
629 os.unlink(name)
630
631 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000632 # mkstemp can create files
Guido van Rossum0e548712002-08-09 16:14:33 +0000633 self.do_create()
634 self.do_create(pre="a")
635 self.do_create(suf="b")
636 self.do_create(pre="a", suf="b")
637 self.do_create(pre="aa", suf=".txt")
Martin v. Löwisd6625482003-10-12 17:37:01 +0000638 self.do_create(dir=".")
Guido van Rossum0e548712002-08-09 16:14:33 +0000639
Gregory P. Smithad577b92015-05-22 16:18:14 -0700640 def test_basic_with_bytes_names(self):
641 # mkstemp can create files when given name parts all
642 # specified as bytes.
643 d = tempfile.gettempdirb()
644 self.do_create(dir=d, suf=b"")
645 self.do_create(dir=d, pre=b"a")
646 self.do_create(dir=d, suf=b"b")
647 self.do_create(dir=d, pre=b"a", suf=b"b")
648 self.do_create(dir=d, pre=b"aa", suf=b".txt")
649 self.do_create(dir=b".")
650 with self.assertRaises(TypeError):
651 self.do_create(dir=".", pre=b"aa", suf=b".txt")
652 with self.assertRaises(TypeError):
653 self.do_create(dir=b".", pre="aa", suf=b".txt")
654 with self.assertRaises(TypeError):
655 self.do_create(dir=b".", pre=b"aa", suf=".txt")
656
657
Guido van Rossum0e548712002-08-09 16:14:33 +0000658 def test_choose_directory(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000659 # mkstemp can create directories in a user-selected directory
Guido van Rossum0e548712002-08-09 16:14:33 +0000660 dir = tempfile.mkdtemp()
661 try:
662 self.do_create(dir=dir)
663 finally:
664 os.rmdir(dir)
665
Guido van Rossum0e548712002-08-09 16:14:33 +0000666
Serhiy Storchaka5d6b7b12015-05-20 00:11:48 +0300667class TestMkdtemp(TestBadTempdir, BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000668 """Test mkdtemp()."""
669
Serhiy Storchaka5d6b7b12015-05-20 00:11:48 +0300670 def make_temp(self):
671 return tempfile.mkdtemp()
672
Gregory P. Smithad577b92015-05-22 16:18:14 -0700673 def do_create(self, dir=None, pre=None, suf=None):
674 output_type = tempfile._infer_return_type(dir, pre, suf)
Guido van Rossum0e548712002-08-09 16:14:33 +0000675 if dir is None:
Gregory P. Smithad577b92015-05-22 16:18:14 -0700676 if output_type is str:
677 dir = tempfile.gettempdir()
678 else:
679 dir = tempfile.gettempdirb()
680 if pre is None:
681 pre = output_type()
682 if suf is None:
683 suf = output_type()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100684 name = tempfile.mkdtemp(dir=dir, prefix=pre, suffix=suf)
Guido van Rossum0e548712002-08-09 16:14:33 +0000685
686 try:
687 self.nameCheck(name, dir, pre, suf)
688 return name
689 except:
690 os.rmdir(name)
691 raise
692
693 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000694 # mkdtemp can create directories
Guido van Rossum0e548712002-08-09 16:14:33 +0000695 os.rmdir(self.do_create())
696 os.rmdir(self.do_create(pre="a"))
697 os.rmdir(self.do_create(suf="b"))
698 os.rmdir(self.do_create(pre="a", suf="b"))
699 os.rmdir(self.do_create(pre="aa", suf=".txt"))
Tim Petersa0d55de2002-08-09 18:01:01 +0000700
Gregory P. Smithad577b92015-05-22 16:18:14 -0700701 def test_basic_with_bytes_names(self):
702 # mkdtemp can create directories when given all binary parts
703 d = tempfile.gettempdirb()
704 os.rmdir(self.do_create(dir=d))
705 os.rmdir(self.do_create(dir=d, pre=b"a"))
706 os.rmdir(self.do_create(dir=d, suf=b"b"))
707 os.rmdir(self.do_create(dir=d, pre=b"a", suf=b"b"))
708 os.rmdir(self.do_create(dir=d, pre=b"aa", suf=b".txt"))
709 with self.assertRaises(TypeError):
710 os.rmdir(self.do_create(dir=d, pre="aa", suf=b".txt"))
711 with self.assertRaises(TypeError):
712 os.rmdir(self.do_create(dir=d, pre=b"aa", suf=".txt"))
713 with self.assertRaises(TypeError):
714 os.rmdir(self.do_create(dir="", pre=b"aa", suf=b".txt"))
715
Guido van Rossum0e548712002-08-09 16:14:33 +0000716 def test_basic_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000717 # mkdtemp can create many directories (stochastic)
Guido van Rossum805365e2007-05-07 22:24:25 +0000718 extant = list(range(TEST_FILES))
Guido van Rossum0e548712002-08-09 16:14:33 +0000719 try:
720 for i in extant:
721 extant[i] = self.do_create(pre="aa")
722 finally:
723 for i in extant:
Guido van Rossum3172c5d2007-10-16 18:12:55 +0000724 if(isinstance(i, str)):
Guido van Rossum0e548712002-08-09 16:14:33 +0000725 os.rmdir(i)
726
727 def test_choose_directory(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000728 # mkdtemp can create directories in a user-selected directory
Guido van Rossum0e548712002-08-09 16:14:33 +0000729 dir = tempfile.mkdtemp()
730 try:
731 os.rmdir(self.do_create(dir=dir))
732 finally:
733 os.rmdir(dir)
734
735 def test_mode(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000736 # mkdtemp creates directories with the proper mode
Guido van Rossum0e548712002-08-09 16:14:33 +0000737
738 dir = self.do_create()
739 try:
740 mode = stat.S_IMODE(os.stat(dir).st_mode)
Guido van Rossumcd16bf62007-06-13 18:07:49 +0000741 mode &= 0o777 # Mask off sticky bits inherited from /tmp
742 expected = 0o700
Jesus Cea4791a242012-10-05 03:15:39 +0200743 if sys.platform == 'win32':
Tim Petersca3ac7f2002-08-09 18:13:51 +0000744 # There's no distinction among 'user', 'group' and 'world';
745 # replicate the 'user' bits.
746 user = expected >> 6
747 expected = user * (1 + 8 + 64)
748 self.assertEqual(mode, expected)
Guido van Rossum0e548712002-08-09 16:14:33 +0000749 finally:
750 os.rmdir(dir)
751
Eli Benderskyaa04f9a2013-09-13 05:28:20 -0700752 def test_collision_with_existing_file(self):
753 # mkdtemp tries another name when a file with
754 # the chosen name already exists
755 with _inside_empty_temp_dir(), \
756 _mock_candidate_names('aaa', 'aaa', 'bbb'):
757 file = tempfile.NamedTemporaryFile(delete=False)
758 file.close()
759 self.assertTrue(file.name.endswith('aaa'))
760 dir = tempfile.mkdtemp()
761 self.assertTrue(dir.endswith('bbb'))
762
763 def test_collision_with_existing_directory(self):
764 # mkdtemp tries another name when a directory with
765 # the chosen name already exists
766 with _inside_empty_temp_dir(), \
767 _mock_candidate_names('aaa', 'aaa', 'bbb'):
768 dir1 = tempfile.mkdtemp()
769 self.assertTrue(dir1.endswith('aaa'))
770 dir2 = tempfile.mkdtemp()
771 self.assertTrue(dir2.endswith('bbb'))
772
Guido van Rossum0e548712002-08-09 16:14:33 +0000773
Antoine Pitroueab2a502012-03-10 16:34:40 +0100774class TestMktemp(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000775 """Test mktemp()."""
776
777 # For safety, all use of mktemp must occur in a private directory.
778 # We must also suppress the RuntimeWarning it generates.
779 def setUp(self):
780 self.dir = tempfile.mkdtemp()
Brett Cannone1adece2010-03-20 22:19:55 +0000781 super().setUp()
Guido van Rossum0e548712002-08-09 16:14:33 +0000782
783 def tearDown(self):
784 if self.dir:
785 os.rmdir(self.dir)
786 self.dir = None
Brett Cannone1adece2010-03-20 22:19:55 +0000787 super().tearDown()
Guido van Rossum0e548712002-08-09 16:14:33 +0000788
789 class mktemped:
790 _unlink = os.unlink
791 _bflags = tempfile._bin_openflags
792
793 def __init__(self, dir, pre, suf):
794 self.name = tempfile.mktemp(dir=dir, prefix=pre, suffix=suf)
795 # Create the file. This will raise an exception if it's
796 # mysteriously appeared in the meanwhile.
Guido van Rossumcd16bf62007-06-13 18:07:49 +0000797 os.close(os.open(self.name, self._bflags, 0o600))
Guido van Rossum0e548712002-08-09 16:14:33 +0000798
799 def __del__(self):
800 self._unlink(self.name)
801
802 def do_create(self, pre="", suf=""):
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100803 file = self.mktemped(self.dir, pre, suf)
Guido van Rossum0e548712002-08-09 16:14:33 +0000804
805 self.nameCheck(file.name, self.dir, pre, suf)
806 return file
807
808 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000809 # mktemp can choose usable file names
Guido van Rossum0e548712002-08-09 16:14:33 +0000810 self.do_create()
811 self.do_create(pre="a")
812 self.do_create(suf="b")
813 self.do_create(pre="a", suf="b")
814 self.do_create(pre="aa", suf=".txt")
815
816 def test_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000817 # mktemp can choose many usable file names (stochastic)
Guido van Rossum805365e2007-05-07 22:24:25 +0000818 extant = list(range(TEST_FILES))
Guido van Rossum0e548712002-08-09 16:14:33 +0000819 for i in extant:
820 extant[i] = self.do_create(pre="aa")
821
Fred Drake8bec4832002-11-22 20:13:43 +0000822## def test_warning(self):
823## # mktemp issues a warning when used
824## warnings.filterwarnings("error",
825## category=RuntimeWarning,
826## message="mktemp")
827## self.assertRaises(RuntimeWarning,
828## tempfile.mktemp, dir=self.dir)
Guido van Rossum0e548712002-08-09 16:14:33 +0000829
Guido van Rossum0e548712002-08-09 16:14:33 +0000830
831# We test _TemporaryFileWrapper by testing NamedTemporaryFile.
832
833
Antoine Pitroueab2a502012-03-10 16:34:40 +0100834class TestNamedTemporaryFile(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000835 """Test NamedTemporaryFile()."""
836
Guido van Rossumd8faa362007-04-27 19:54:29 +0000837 def do_create(self, dir=None, pre="", suf="", delete=True):
Guido van Rossum0e548712002-08-09 16:14:33 +0000838 if dir is None:
839 dir = tempfile.gettempdir()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100840 file = tempfile.NamedTemporaryFile(dir=dir, prefix=pre, suffix=suf,
841 delete=delete)
Guido van Rossum0e548712002-08-09 16:14:33 +0000842
843 self.nameCheck(file.name, dir, pre, suf)
844 return file
845
846
847 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000848 # NamedTemporaryFile can create files
Guido van Rossum0e548712002-08-09 16:14:33 +0000849 self.do_create()
850 self.do_create(pre="a")
851 self.do_create(suf="b")
852 self.do_create(pre="a", suf="b")
853 self.do_create(pre="aa", suf=".txt")
854
Antoine Pitrou17c93262013-12-21 22:14:56 +0100855 def test_method_lookup(self):
856 # Issue #18879: Looking up a temporary file method should keep it
857 # alive long enough.
858 f = self.do_create()
859 wr = weakref.ref(f)
860 write = f.write
861 write2 = f.write
862 del f
863 write(b'foo')
864 del write
865 write2(b'bar')
866 del write2
867 if support.check_impl_detail(cpython=True):
868 # No reference cycle was created.
869 self.assertIsNone(wr())
870
Serhiy Storchaka56cefa62015-03-19 15:23:15 +0200871 def test_iter(self):
872 # Issue #23700: getting iterator from a temporary file should keep
873 # it alive as long as it's being iterated over
874 lines = [b'spam\n', b'eggs\n', b'beans\n']
875 def make_file():
876 f = tempfile.NamedTemporaryFile(mode='w+b')
877 f.write(b''.join(lines))
878 f.seek(0)
879 return f
880 for i, l in enumerate(make_file()):
881 self.assertEqual(l, lines[i])
882 self.assertEqual(i, len(lines) - 1)
883
Guido van Rossum0e548712002-08-09 16:14:33 +0000884 def test_creates_named(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000885 # NamedTemporaryFile creates files with names
Guido van Rossum0e548712002-08-09 16:14:33 +0000886 f = tempfile.NamedTemporaryFile()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000887 self.assertTrue(os.path.exists(f.name),
Guido van Rossum0e548712002-08-09 16:14:33 +0000888 "NamedTemporaryFile %s does not exist" % f.name)
889
890 def test_del_on_close(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000891 # A NamedTemporaryFile is deleted when closed
Guido van Rossum0e548712002-08-09 16:14:33 +0000892 dir = tempfile.mkdtemp()
893 try:
894 f = tempfile.NamedTemporaryFile(dir=dir)
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000895 f.write(b'blat')
Guido van Rossum0e548712002-08-09 16:14:33 +0000896 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000897 self.assertFalse(os.path.exists(f.name),
Guido van Rossum0e548712002-08-09 16:14:33 +0000898 "NamedTemporaryFile %s exists after close" % f.name)
899 finally:
900 os.rmdir(dir)
901
Guido van Rossumd8faa362007-04-27 19:54:29 +0000902 def test_dis_del_on_close(self):
903 # Tests that delete-on-close can be disabled
904 dir = tempfile.mkdtemp()
905 tmp = None
906 try:
907 f = tempfile.NamedTemporaryFile(dir=dir, delete=False)
908 tmp = f.name
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000909 f.write(b'blat')
Guido van Rossumd8faa362007-04-27 19:54:29 +0000910 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000911 self.assertTrue(os.path.exists(f.name),
Guido van Rossumd8faa362007-04-27 19:54:29 +0000912 "NamedTemporaryFile %s missing after close" % f.name)
913 finally:
914 if tmp is not None:
915 os.unlink(tmp)
916 os.rmdir(dir)
917
Guido van Rossum0e548712002-08-09 16:14:33 +0000918 def test_multiple_close(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000919 # A NamedTemporaryFile can be closed many times without error
Guido van Rossum0e548712002-08-09 16:14:33 +0000920 f = tempfile.NamedTemporaryFile()
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000921 f.write(b'abc\n')
Guido van Rossum0e548712002-08-09 16:14:33 +0000922 f.close()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100923 f.close()
924 f.close()
Guido van Rossum0e548712002-08-09 16:14:33 +0000925
Christian Heimes3ecfea712008-02-09 20:51:34 +0000926 def test_context_manager(self):
927 # A NamedTemporaryFile can be used as a context manager
928 with tempfile.NamedTemporaryFile() as f:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000929 self.assertTrue(os.path.exists(f.name))
930 self.assertFalse(os.path.exists(f.name))
Christian Heimes3ecfea712008-02-09 20:51:34 +0000931 def use_closed():
932 with f:
933 pass
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000934 self.assertRaises(ValueError, use_closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000935
Victor Stinner1f99f9d2014-03-25 09:18:04 +0100936 def test_no_leak_fd(self):
Victor Stinner87d13ea2014-03-25 18:19:17 +0100937 # Issue #21058: don't leak file descriptor when io.open() fails
Victor Stinner1f99f9d2014-03-25 09:18:04 +0100938 closed = []
Victor Stinnerc61c1702014-07-29 01:13:39 +0200939 os_close = os.close
Victor Stinner1f99f9d2014-03-25 09:18:04 +0100940 def close(fd):
941 closed.append(fd)
Victor Stinnerc61c1702014-07-29 01:13:39 +0200942 os_close(fd)
Victor Stinner1f99f9d2014-03-25 09:18:04 +0100943
944 with mock.patch('os.close', side_effect=close):
945 with mock.patch('io.open', side_effect=ValueError):
946 self.assertRaises(ValueError, tempfile.NamedTemporaryFile)
947 self.assertEqual(len(closed), 1)
948
Martin Panter7869a222016-02-28 05:22:20 +0000949 def test_bad_mode(self):
950 dir = tempfile.mkdtemp()
951 self.addCleanup(support.rmtree, dir)
952 with self.assertRaises(ValueError):
953 tempfile.NamedTemporaryFile(mode='wr', dir=dir)
954 with self.assertRaises(TypeError):
955 tempfile.NamedTemporaryFile(mode=2, dir=dir)
956 self.assertEqual(os.listdir(dir), [])
Guido van Rossum0e548712002-08-09 16:14:33 +0000957
Martin Panter7869a222016-02-28 05:22:20 +0000958 # How to test the mode and bufsize parameters?
Guido van Rossum0e548712002-08-09 16:14:33 +0000959
Antoine Pitroueab2a502012-03-10 16:34:40 +0100960class TestSpooledTemporaryFile(BaseTestCase):
Guido van Rossumd8faa362007-04-27 19:54:29 +0000961 """Test SpooledTemporaryFile()."""
962
963 def do_create(self, max_size=0, dir=None, pre="", suf=""):
964 if dir is None:
965 dir = tempfile.gettempdir()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100966 file = tempfile.SpooledTemporaryFile(max_size=max_size, dir=dir, prefix=pre, suffix=suf)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000967
968 return file
969
970
971 def test_basic(self):
972 # SpooledTemporaryFile can create files
973 f = self.do_create()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000974 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000975 f = self.do_create(max_size=100, pre="a", suf=".txt")
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000976 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000977
978 def test_del_on_close(self):
979 # A SpooledTemporaryFile is deleted when closed
980 dir = tempfile.mkdtemp()
981 try:
982 f = tempfile.SpooledTemporaryFile(max_size=10, dir=dir)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000983 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +0000984 f.write(b'blat ' * 5)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000985 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000986 filename = f.name
987 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000988 self.assertFalse(isinstance(filename, str) and os.path.exists(filename),
Guido van Rossumd8faa362007-04-27 19:54:29 +0000989 "SpooledTemporaryFile %s exists after close" % filename)
990 finally:
991 os.rmdir(dir)
992
993 def test_rewrite_small(self):
994 # A SpooledTemporaryFile can be written to multiple within the max_size
995 f = self.do_create(max_size=30)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000996 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000997 for i in range(5):
998 f.seek(0, 0)
Guido van Rossum39478e82007-08-27 17:23:59 +0000999 f.write(b'x' * 20)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001000 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +00001001
1002 def test_write_sequential(self):
1003 # A SpooledTemporaryFile should hold exactly max_size bytes, and roll
1004 # over afterward
1005 f = self.do_create(max_size=30)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001006 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +00001007 f.write(b'x' * 20)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001008 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +00001009 f.write(b'x' * 10)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001010 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +00001011 f.write(b'x')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001012 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +00001013
R David Murrayd89ee792011-03-14 09:55:46 -04001014 def test_writelines(self):
1015 # Verify writelines with a SpooledTemporaryFile
1016 f = self.do_create()
1017 f.writelines((b'x', b'y', b'z'))
1018 f.seek(0)
1019 buf = f.read()
1020 self.assertEqual(buf, b'xyz')
1021
1022 def test_writelines_sequential(self):
1023 # A SpooledTemporaryFile should hold exactly max_size bytes, and roll
1024 # over afterward
1025 f = self.do_create(max_size=35)
1026 f.writelines((b'x' * 20, b'x' * 10, b'x' * 5))
1027 self.assertFalse(f._rolled)
1028 f.write(b'x')
1029 self.assertTrue(f._rolled)
1030
Guido van Rossumd8faa362007-04-27 19:54:29 +00001031 def test_sparse(self):
1032 # A SpooledTemporaryFile that is written late in the file will extend
1033 # when that occurs
1034 f = self.do_create(max_size=30)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001035 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +00001036 f.seek(100, 0)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001037 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +00001038 f.write(b'x')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001039 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +00001040
1041 def test_fileno(self):
1042 # A SpooledTemporaryFile should roll over to a real file on fileno()
1043 f = self.do_create(max_size=30)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001044 self.assertFalse(f._rolled)
1045 self.assertTrue(f.fileno() > 0)
1046 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +00001047
Christian Heimes3ecfea712008-02-09 20:51:34 +00001048 def test_multiple_close_before_rollover(self):
Guido van Rossumd8faa362007-04-27 19:54:29 +00001049 # A SpooledTemporaryFile can be closed many times without error
1050 f = tempfile.SpooledTemporaryFile()
Guido van Rossum39478e82007-08-27 17:23:59 +00001051 f.write(b'abc\n')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001052 self.assertFalse(f._rolled)
Christian Heimes3ecfea712008-02-09 20:51:34 +00001053 f.close()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +01001054 f.close()
1055 f.close()
Christian Heimes3ecfea712008-02-09 20:51:34 +00001056
1057 def test_multiple_close_after_rollover(self):
1058 # A SpooledTemporaryFile can be closed many times without error
1059 f = tempfile.SpooledTemporaryFile(max_size=1)
1060 f.write(b'abc\n')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001061 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +00001062 f.close()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +01001063 f.close()
1064 f.close()
Guido van Rossumd8faa362007-04-27 19:54:29 +00001065
1066 def test_bound_methods(self):
1067 # It should be OK to steal a bound method from a SpooledTemporaryFile
1068 # and use it independently; when the file rolls over, those bound
1069 # methods should continue to function
1070 f = self.do_create(max_size=30)
1071 read = f.read
1072 write = f.write
1073 seek = f.seek
1074
Guido van Rossum39478e82007-08-27 17:23:59 +00001075 write(b"a" * 35)
1076 write(b"b" * 35)
Guido van Rossumd8faa362007-04-27 19:54:29 +00001077 seek(0, 0)
Guido van Rossum9a634702007-07-09 10:24:45 +00001078 self.assertEqual(read(70), b'a'*35 + b'b'*35)
1079
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +02001080 def test_properties(self):
1081 f = tempfile.SpooledTemporaryFile(max_size=10)
1082 f.write(b'x' * 10)
1083 self.assertFalse(f._rolled)
1084 self.assertEqual(f.mode, 'w+b')
1085 self.assertIsNone(f.name)
1086 with self.assertRaises(AttributeError):
1087 f.newlines
1088 with self.assertRaises(AttributeError):
1089 f.encoding
sth825aab92018-05-23 07:07:01 +02001090 with self.assertRaises(AttributeError):
1091 f.errors
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +02001092
1093 f.write(b'x')
1094 self.assertTrue(f._rolled)
1095 self.assertEqual(f.mode, 'rb+')
1096 self.assertIsNotNone(f.name)
1097 with self.assertRaises(AttributeError):
1098 f.newlines
1099 with self.assertRaises(AttributeError):
1100 f.encoding
sth825aab92018-05-23 07:07:01 +02001101 with self.assertRaises(AttributeError):
1102 f.errors
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +02001103
Guido van Rossum9a634702007-07-09 10:24:45 +00001104 def test_text_mode(self):
1105 # Creating a SpooledTemporaryFile with a text mode should produce
1106 # a file object reading and writing (Unicode) text strings.
1107 f = tempfile.SpooledTemporaryFile(mode='w+', max_size=10)
1108 f.write("abc\n")
1109 f.seek(0)
1110 self.assertEqual(f.read(), "abc\n")
1111 f.write("def\n")
1112 f.seek(0)
1113 self.assertEqual(f.read(), "abc\ndef\n")
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +02001114 self.assertFalse(f._rolled)
1115 self.assertEqual(f.mode, 'w+')
1116 self.assertIsNone(f.name)
1117 self.assertIsNone(f.newlines)
1118 self.assertIsNone(f.encoding)
sth825aab92018-05-23 07:07:01 +02001119 self.assertIsNone(f.errors)
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +02001120
Guido van Rossum9a634702007-07-09 10:24:45 +00001121 f.write("xyzzy\n")
1122 f.seek(0)
1123 self.assertEqual(f.read(), "abc\ndef\nxyzzy\n")
Amaury Forgeot d'Arc7d0bddd2009-11-30 00:08:56 +00001124 # Check that Ctrl+Z doesn't truncate the file
1125 f.write("foo\x1abar\n")
1126 f.seek(0)
1127 self.assertEqual(f.read(), "abc\ndef\nxyzzy\nfoo\x1abar\n")
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +02001128 self.assertTrue(f._rolled)
1129 self.assertEqual(f.mode, 'w+')
1130 self.assertIsNotNone(f.name)
Serhiy Storchaka497cee42013-02-10 14:43:46 +02001131 self.assertEqual(f.newlines, os.linesep)
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +02001132 self.assertIsNotNone(f.encoding)
sth825aab92018-05-23 07:07:01 +02001133 self.assertIsNotNone(f.errors)
Guido van Rossum9a634702007-07-09 10:24:45 +00001134
Guido van Rossumf0c74162007-08-28 03:29:45 +00001135 def test_text_newline_and_encoding(self):
1136 f = tempfile.SpooledTemporaryFile(mode='w+', max_size=10,
sth825aab92018-05-23 07:07:01 +02001137 newline='', encoding='utf-8',
1138 errors='ignore')
Guido van Rossumf0c74162007-08-28 03:29:45 +00001139 f.write("\u039B\r\n")
1140 f.seek(0)
1141 self.assertEqual(f.read(), "\u039B\r\n")
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001142 self.assertFalse(f._rolled)
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +02001143 self.assertEqual(f.mode, 'w+')
1144 self.assertIsNone(f.name)
1145 self.assertIsNone(f.newlines)
1146 self.assertIsNone(f.encoding)
sth825aab92018-05-23 07:07:01 +02001147 self.assertIsNone(f.errors)
Guido van Rossumf0c74162007-08-28 03:29:45 +00001148
1149 f.write("\u039B" * 20 + "\r\n")
1150 f.seek(0)
1151 self.assertEqual(f.read(), "\u039B\r\n" + ("\u039B" * 20) + "\r\n")
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001152 self.assertTrue(f._rolled)
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +02001153 self.assertEqual(f.mode, 'w+')
1154 self.assertIsNotNone(f.name)
1155 self.assertIsNotNone(f.newlines)
1156 self.assertEqual(f.encoding, 'utf-8')
sth825aab92018-05-23 07:07:01 +02001157 self.assertEqual(f.errors, 'ignore')
Guido van Rossumd8faa362007-04-27 19:54:29 +00001158
Christian Heimes3ecfea712008-02-09 20:51:34 +00001159 def test_context_manager_before_rollover(self):
1160 # A SpooledTemporaryFile can be used as a context manager
1161 with tempfile.SpooledTemporaryFile(max_size=1) as f:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001162 self.assertFalse(f._rolled)
1163 self.assertFalse(f.closed)
1164 self.assertTrue(f.closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +00001165 def use_closed():
1166 with f:
1167 pass
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001168 self.assertRaises(ValueError, use_closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +00001169
1170 def test_context_manager_during_rollover(self):
1171 # A SpooledTemporaryFile can be used as a context manager
1172 with tempfile.SpooledTemporaryFile(max_size=1) as f:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001173 self.assertFalse(f._rolled)
Christian Heimes3ecfea712008-02-09 20:51:34 +00001174 f.write(b'abc\n')
1175 f.flush()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001176 self.assertTrue(f._rolled)
1177 self.assertFalse(f.closed)
1178 self.assertTrue(f.closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +00001179 def use_closed():
1180 with f:
1181 pass
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001182 self.assertRaises(ValueError, use_closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +00001183
1184 def test_context_manager_after_rollover(self):
1185 # A SpooledTemporaryFile can be used as a context manager
1186 f = tempfile.SpooledTemporaryFile(max_size=1)
1187 f.write(b'abc\n')
1188 f.flush()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001189 self.assertTrue(f._rolled)
Christian Heimes3ecfea712008-02-09 20:51:34 +00001190 with f:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001191 self.assertFalse(f.closed)
1192 self.assertTrue(f.closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +00001193 def use_closed():
1194 with f:
1195 pass
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001196 self.assertRaises(ValueError, use_closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +00001197
Antoine Pitrou0e86a582011-11-25 18:03:09 +01001198 def test_truncate_with_size_parameter(self):
1199 # A SpooledTemporaryFile can be truncated to zero size
1200 f = tempfile.SpooledTemporaryFile(max_size=10)
1201 f.write(b'abcdefg\n')
1202 f.seek(0)
1203 f.truncate()
1204 self.assertFalse(f._rolled)
1205 self.assertEqual(f._file.getvalue(), b'')
1206 # A SpooledTemporaryFile can be truncated to a specific size
1207 f = tempfile.SpooledTemporaryFile(max_size=10)
1208 f.write(b'abcdefg\n')
1209 f.truncate(4)
1210 self.assertFalse(f._rolled)
1211 self.assertEqual(f._file.getvalue(), b'abcd')
1212 # A SpooledTemporaryFile rolls over if truncated to large size
1213 f = tempfile.SpooledTemporaryFile(max_size=10)
1214 f.write(b'abcdefg\n')
1215 f.truncate(20)
1216 self.assertTrue(f._rolled)
Anthony Sottile8377cd42019-02-25 14:32:27 -08001217 self.assertEqual(os.fstat(f.fileno()).st_size, 20)
Christian Heimes3ecfea712008-02-09 20:51:34 +00001218
Guido van Rossum0e548712002-08-09 16:14:33 +00001219
Guido van Rossum8ccd9b62002-08-22 20:02:03 +00001220if tempfile.NamedTemporaryFile is not tempfile.TemporaryFile:
Antoine Pitroueab2a502012-03-10 16:34:40 +01001221
1222 class TestTemporaryFile(BaseTestCase):
1223 """Test TemporaryFile()."""
1224
1225 def test_basic(self):
1226 # TemporaryFile can create files
1227 # No point in testing the name params - the file has no name.
1228 tempfile.TemporaryFile()
1229
1230 def test_has_no_name(self):
1231 # TemporaryFile creates files with no names (on this system)
1232 dir = tempfile.mkdtemp()
1233 f = tempfile.TemporaryFile(dir=dir)
1234 f.write(b'blat')
1235
1236 # Sneaky: because this file has no name, it should not prevent
1237 # us from removing the directory it was created in.
1238 try:
1239 os.rmdir(dir)
1240 except:
1241 # cleanup
1242 f.close()
1243 os.rmdir(dir)
1244 raise
1245
1246 def test_multiple_close(self):
1247 # A TemporaryFile can be closed many times without error
1248 f = tempfile.TemporaryFile()
1249 f.write(b'abc\n')
1250 f.close()
1251 f.close()
1252 f.close()
1253
1254 # How to test the mode and bufsize parameters?
1255 def test_mode_and_encoding(self):
1256
1257 def roundtrip(input, *args, **kwargs):
1258 with tempfile.TemporaryFile(*args, **kwargs) as fileobj:
1259 fileobj.write(input)
1260 fileobj.seek(0)
1261 self.assertEqual(input, fileobj.read())
1262
1263 roundtrip(b"1234", "w+b")
1264 roundtrip("abdc\n", "w+")
1265 roundtrip("\u039B", "w+", encoding="utf-16")
1266 roundtrip("foo\r\n", "w+", newline="")
Guido van Rossum0e548712002-08-09 16:14:33 +00001267
Victor Stinner1f99f9d2014-03-25 09:18:04 +01001268 def test_no_leak_fd(self):
1269 # Issue #21058: don't leak file descriptor when io.open() fails
1270 closed = []
Victor Stinnerc61c1702014-07-29 01:13:39 +02001271 os_close = os.close
Victor Stinner1f99f9d2014-03-25 09:18:04 +01001272 def close(fd):
1273 closed.append(fd)
Victor Stinnerc61c1702014-07-29 01:13:39 +02001274 os_close(fd)
Victor Stinner1f99f9d2014-03-25 09:18:04 +01001275
1276 with mock.patch('os.close', side_effect=close):
1277 with mock.patch('io.open', side_effect=ValueError):
1278 self.assertRaises(ValueError, tempfile.TemporaryFile)
1279 self.assertEqual(len(closed), 1)
1280
1281
Nick Coghlan543af752010-10-24 11:23:25 +00001282
1283# Helper for test_del_on_shutdown
1284class NulledModules:
1285 def __init__(self, *modules):
1286 self.refs = [mod.__dict__ for mod in modules]
1287 self.contents = [ref.copy() for ref in self.refs]
1288
1289 def __enter__(self):
1290 for d in self.refs:
1291 for key in d:
1292 d[key] = None
1293
1294 def __exit__(self, *exc_info):
1295 for d, c in zip(self.refs, self.contents):
1296 d.clear()
1297 d.update(c)
1298
Antoine Pitroueab2a502012-03-10 16:34:40 +01001299class TestTemporaryDirectory(BaseTestCase):
Nick Coghlan543af752010-10-24 11:23:25 +00001300 """Test TemporaryDirectory()."""
1301
1302 def do_create(self, dir=None, pre="", suf="", recurse=1):
1303 if dir is None:
1304 dir = tempfile.gettempdir()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +01001305 tmp = tempfile.TemporaryDirectory(dir=dir, prefix=pre, suffix=suf)
Nick Coghlan543af752010-10-24 11:23:25 +00001306 self.nameCheck(tmp.name, dir, pre, suf)
1307 # Create a subdirectory and some files
1308 if recurse:
Serhiy Storchaka99e033b2014-01-27 11:18:27 +02001309 d1 = self.do_create(tmp.name, pre, suf, recurse-1)
1310 d1.name = None
Nick Coghlan543af752010-10-24 11:23:25 +00001311 with open(os.path.join(tmp.name, "test.txt"), "wb") as f:
1312 f.write(b"Hello world!")
1313 return tmp
1314
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001315 def test_mkdtemp_failure(self):
1316 # Check no additional exception if mkdtemp fails
1317 # Previously would raise AttributeError instead
Nick Coghlan3c54ea62010-12-13 03:02:43 +00001318 # (noted as part of Issue #10188)
1319 with tempfile.TemporaryDirectory() as nonexistent:
1320 pass
Serhiy Storchaka7451a722013-02-09 22:25:49 +02001321 with self.assertRaises(FileNotFoundError) as cm:
Nick Coghlan3c54ea62010-12-13 03:02:43 +00001322 tempfile.TemporaryDirectory(dir=nonexistent)
Serhiy Storchaka7451a722013-02-09 22:25:49 +02001323 self.assertEqual(cm.exception.errno, errno.ENOENT)
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001324
Nick Coghlan543af752010-10-24 11:23:25 +00001325 def test_explicit_cleanup(self):
1326 # A TemporaryDirectory is deleted when cleaned up
1327 dir = tempfile.mkdtemp()
1328 try:
1329 d = self.do_create(dir=dir)
1330 self.assertTrue(os.path.exists(d.name),
1331 "TemporaryDirectory %s does not exist" % d.name)
1332 d.cleanup()
1333 self.assertFalse(os.path.exists(d.name),
1334 "TemporaryDirectory %s exists after cleanup" % d.name)
1335 finally:
1336 os.rmdir(dir)
1337
Charles-François Natalidef35432011-07-29 18:59:24 +02001338 @support.skip_unless_symlink
1339 def test_cleanup_with_symlink_to_a_directory(self):
1340 # cleanup() should not follow symlinks to directories (issue #12464)
1341 d1 = self.do_create()
Serhiy Storchaka99e033b2014-01-27 11:18:27 +02001342 d2 = self.do_create(recurse=0)
Charles-François Natalidef35432011-07-29 18:59:24 +02001343
1344 # Symlink d1/foo -> d2
1345 os.symlink(d2.name, os.path.join(d1.name, "foo"))
1346
1347 # This call to cleanup() should not follow the "foo" symlink
1348 d1.cleanup()
1349
1350 self.assertFalse(os.path.exists(d1.name),
1351 "TemporaryDirectory %s exists after cleanup" % d1.name)
1352 self.assertTrue(os.path.exists(d2.name),
1353 "Directory pointed to by a symlink was deleted")
1354 self.assertEqual(os.listdir(d2.name), ['test.txt'],
1355 "Contents of the directory pointed to by a symlink "
1356 "were deleted")
1357 d2.cleanup()
1358
Nick Coghlan543af752010-10-24 11:23:25 +00001359 @support.cpython_only
1360 def test_del_on_collection(self):
1361 # A TemporaryDirectory is deleted when garbage collected
1362 dir = tempfile.mkdtemp()
1363 try:
1364 d = self.do_create(dir=dir)
1365 name = d.name
1366 del d # Rely on refcounting to invoke __del__
1367 self.assertFalse(os.path.exists(name),
1368 "TemporaryDirectory %s exists after __del__" % name)
1369 finally:
1370 os.rmdir(dir)
1371
Nick Coghlan543af752010-10-24 11:23:25 +00001372 def test_del_on_shutdown(self):
1373 # A TemporaryDirectory may be cleaned up during shutdown
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001374 with self.do_create() as dir:
Serhiy Storchakaa28632b2014-01-27 11:21:54 +02001375 for mod in ('builtins', 'os', 'shutil', 'sys', 'tempfile', 'warnings'):
Serhiy Storchaka99e033b2014-01-27 11:18:27 +02001376 code = """if True:
Serhiy Storchakaa28632b2014-01-27 11:21:54 +02001377 import builtins
Serhiy Storchaka99e033b2014-01-27 11:18:27 +02001378 import os
1379 import shutil
1380 import sys
1381 import tempfile
1382 import warnings
1383
1384 tmp = tempfile.TemporaryDirectory(dir={dir!r})
1385 sys.stdout.buffer.write(tmp.name.encode())
1386
1387 tmp2 = os.path.join(tmp.name, 'test_dir')
1388 os.mkdir(tmp2)
1389 with open(os.path.join(tmp2, "test.txt"), "w") as f:
1390 f.write("Hello world!")
1391
1392 {mod}.tmp = tmp
1393
1394 warnings.filterwarnings("always", category=ResourceWarning)
1395 """.format(dir=dir, mod=mod)
1396 rc, out, err = script_helper.assert_python_ok("-c", code)
1397 tmp_name = out.decode().strip()
1398 self.assertFalse(os.path.exists(tmp_name),
1399 "TemporaryDirectory %s exists after cleanup" % tmp_name)
1400 err = err.decode('utf-8', 'backslashreplace')
1401 self.assertNotIn("Exception ", err)
Serhiy Storchakaa28632b2014-01-27 11:21:54 +02001402 self.assertIn("ResourceWarning: Implicitly cleaning up", err)
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001403
Serhiy Storchaka5e193ac2014-09-24 13:26:25 +03001404 def test_exit_on_shutdown(self):
1405 # Issue #22427
1406 with self.do_create() as dir:
1407 code = """if True:
1408 import sys
1409 import tempfile
1410 import warnings
1411
1412 def generator():
1413 with tempfile.TemporaryDirectory(dir={dir!r}) as tmp:
1414 yield tmp
1415 g = generator()
1416 sys.stdout.buffer.write(next(g).encode())
1417
1418 warnings.filterwarnings("always", category=ResourceWarning)
1419 """.format(dir=dir)
1420 rc, out, err = script_helper.assert_python_ok("-c", code)
1421 tmp_name = out.decode().strip()
1422 self.assertFalse(os.path.exists(tmp_name),
1423 "TemporaryDirectory %s exists after cleanup" % tmp_name)
1424 err = err.decode('utf-8', 'backslashreplace')
1425 self.assertNotIn("Exception ", err)
1426 self.assertIn("ResourceWarning: Implicitly cleaning up", err)
1427
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001428 def test_warnings_on_cleanup(self):
Serhiy Storchaka99e033b2014-01-27 11:18:27 +02001429 # ResourceWarning will be triggered by __del__
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001430 with self.do_create() as dir:
Serhiy Storchaka99e033b2014-01-27 11:18:27 +02001431 d = self.do_create(dir=dir, recurse=3)
1432 name = d.name
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001433
1434 # Check for the resource warning
1435 with support.check_warnings(('Implicitly', ResourceWarning), quiet=False):
1436 warnings.filterwarnings("always", category=ResourceWarning)
Serhiy Storchaka99e033b2014-01-27 11:18:27 +02001437 del d
1438 support.gc_collect()
1439 self.assertFalse(os.path.exists(name),
1440 "TemporaryDirectory %s exists after __del__" % name)
Nick Coghlan543af752010-10-24 11:23:25 +00001441
1442 def test_multiple_close(self):
1443 # Can be cleaned-up many times without error
1444 d = self.do_create()
1445 d.cleanup()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +01001446 d.cleanup()
1447 d.cleanup()
Nick Coghlan543af752010-10-24 11:23:25 +00001448
1449 def test_context_manager(self):
1450 # Can be used as a context manager
1451 d = self.do_create()
1452 with d as name:
1453 self.assertTrue(os.path.exists(name))
1454 self.assertEqual(name, d.name)
1455 self.assertFalse(os.path.exists(name))
1456
1457
Guido van Rossum0e548712002-08-09 16:14:33 +00001458if __name__ == "__main__":
Zachary Ware38c707e2015-04-13 15:00:43 -05001459 unittest.main()