blob: 710756bde64c04fd2758e38c545043e2e2726e8a [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
Antoine Pitrou17c93262013-12-21 22:14:56 +010011import weakref
Victor Stinner1f99f9d2014-03-25 09:18:04 +010012from unittest import mock
Tim Petersc57a2852001-10-29 21:46:08 +000013
Guido van Rossum0e548712002-08-09 16:14:33 +000014import unittest
Berker Peksagce643912015-05-06 06:33:17 +030015from test import support
16from test.support import script_helper
Guido van Rossum0e548712002-08-09 16:14:33 +000017
Fred Drake7633d232002-10-17 22:09:03 +000018
Guido van Rossum0e548712002-08-09 16:14:33 +000019if hasattr(os, 'stat'):
20 import stat
21 has_stat = 1
22else:
23 has_stat = 0
24
25has_textmode = (tempfile._text_openflags != tempfile._bin_openflags)
Guido van Rossum78741062002-08-17 11:41:01 +000026has_spawnl = hasattr(os, 'spawnl')
Guido van Rossum0e548712002-08-09 16:14:33 +000027
Neal Norwitz68ee0122002-08-16 19:28:59 +000028# TEST_FILES may need to be tweaked for systems depending on the maximum
29# number of files that can be opened at one time (see ulimit -n)
Victor Stinner9c3de4a2011-08-17 20:49:41 +020030if sys.platform.startswith('openbsd'):
Martin v. Löwis99968282004-09-15 06:02:54 +000031 TEST_FILES = 48
Jack Jansence921472003-01-08 16:30:34 +000032else:
33 TEST_FILES = 100
Neal Norwitz68ee0122002-08-16 19:28:59 +000034
Guido van Rossum0e548712002-08-09 16:14:33 +000035# This is organized as one test for each chunk of code in tempfile.py,
36# in order of their appearance in the file. Testing which requires
37# threads is not done here.
38
Gregory P. Smithad577b92015-05-22 16:18:14 -070039class TestLowLevelInternals(unittest.TestCase):
40 def test_infer_return_type_singles(self):
41 self.assertIs(str, tempfile._infer_return_type(''))
42 self.assertIs(bytes, tempfile._infer_return_type(b''))
43 self.assertIs(str, tempfile._infer_return_type(None))
44
45 def test_infer_return_type_multiples(self):
46 self.assertIs(str, tempfile._infer_return_type('', ''))
47 self.assertIs(bytes, tempfile._infer_return_type(b'', b''))
48 with self.assertRaises(TypeError):
49 tempfile._infer_return_type('', b'')
50 with self.assertRaises(TypeError):
51 tempfile._infer_return_type(b'', '')
52
53 def test_infer_return_type_multiples_and_none(self):
54 self.assertIs(str, tempfile._infer_return_type(None, ''))
55 self.assertIs(str, tempfile._infer_return_type('', None))
56 self.assertIs(str, tempfile._infer_return_type(None, None))
57 self.assertIs(bytes, tempfile._infer_return_type(b'', None))
58 self.assertIs(bytes, tempfile._infer_return_type(None, b''))
59 with self.assertRaises(TypeError):
60 tempfile._infer_return_type('', None, b'')
61 with self.assertRaises(TypeError):
62 tempfile._infer_return_type(b'', None, '')
63
64
Guido van Rossum0e548712002-08-09 16:14:33 +000065# Common functionality.
Gregory P. Smithad577b92015-05-22 16:18:14 -070066
Antoine Pitroueab2a502012-03-10 16:34:40 +010067class BaseTestCase(unittest.TestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +000068
Victor Stinner97869102013-08-14 01:28:28 +020069 str_check = re.compile(r"^[a-z0-9_-]{8}$")
Gregory P. Smithad577b92015-05-22 16:18:14 -070070 b_check = re.compile(br"^[a-z0-9_-]{8}$")
Guido van Rossum0e548712002-08-09 16:14:33 +000071
Brett Cannone1adece2010-03-20 22:19:55 +000072 def setUp(self):
73 self._warnings_manager = support.check_warnings()
74 self._warnings_manager.__enter__()
75 warnings.filterwarnings("ignore", category=RuntimeWarning,
76 message="mktemp", module=__name__)
77
78 def tearDown(self):
79 self._warnings_manager.__exit__(None, None, None)
80
Guido van Rossum0e548712002-08-09 16:14:33 +000081 def nameCheck(self, name, dir, pre, suf):
82 (ndir, nbase) = os.path.split(name)
83 npre = nbase[:len(pre)]
84 nsuf = nbase[len(nbase)-len(suf):]
85
Gregory P. Smithad577b92015-05-22 16:18:14 -070086 if dir is not None:
87 self.assertIs(type(name), str if type(dir) is str else bytes,
88 "unexpected return type")
89 if pre is not None:
90 self.assertIs(type(name), str if type(pre) is str else bytes,
91 "unexpected return type")
92 if suf is not None:
93 self.assertIs(type(name), str if type(suf) is str else bytes,
94 "unexpected return type")
95 if (dir, pre, suf) == (None, None, None):
96 self.assertIs(type(name), str, "default return type must be str")
97
Martin v. Löwisd6625482003-10-12 17:37:01 +000098 # check for equality of the absolute paths!
99 self.assertEqual(os.path.abspath(ndir), os.path.abspath(dir),
Gregory P. Smithad577b92015-05-22 16:18:14 -0700100 "file %r not in directory %r" % (name, dir))
Guido van Rossum0e548712002-08-09 16:14:33 +0000101 self.assertEqual(npre, pre,
Gregory P. Smithad577b92015-05-22 16:18:14 -0700102 "file %r does not begin with %r" % (nbase, pre))
Guido van Rossum0e548712002-08-09 16:14:33 +0000103 self.assertEqual(nsuf, suf,
Gregory P. Smithad577b92015-05-22 16:18:14 -0700104 "file %r does not end with %r" % (nbase, suf))
Guido van Rossum0e548712002-08-09 16:14:33 +0000105
106 nbase = nbase[len(pre):len(nbase)-len(suf)]
Gregory P. Smithad577b92015-05-22 16:18:14 -0700107 check = self.str_check if isinstance(nbase, str) else self.b_check
108 self.assertTrue(check.match(nbase),
109 "random characters %r do not match %r"
110 % (nbase, check.pattern))
Guido van Rossum0e548712002-08-09 16:14:33 +0000111
Guido van Rossum0e548712002-08-09 16:14:33 +0000112
Antoine Pitroueab2a502012-03-10 16:34:40 +0100113class TestExports(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000114 def test_exports(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000115 # There are no surprising symbols in the tempfile module
Guido van Rossum0e548712002-08-09 16:14:33 +0000116 dict = tempfile.__dict__
117
118 expected = {
119 "NamedTemporaryFile" : 1,
120 "TemporaryFile" : 1,
121 "mkstemp" : 1,
122 "mkdtemp" : 1,
123 "mktemp" : 1,
124 "TMP_MAX" : 1,
125 "gettempprefix" : 1,
Gregory P. Smithad577b92015-05-22 16:18:14 -0700126 "gettempprefixb" : 1,
Guido van Rossum0e548712002-08-09 16:14:33 +0000127 "gettempdir" : 1,
Gregory P. Smithad577b92015-05-22 16:18:14 -0700128 "gettempdirb" : 1,
Guido van Rossum0e548712002-08-09 16:14:33 +0000129 "tempdir" : 1,
Guido van Rossumd8faa362007-04-27 19:54:29 +0000130 "template" : 1,
Nick Coghlan543af752010-10-24 11:23:25 +0000131 "SpooledTemporaryFile" : 1,
132 "TemporaryDirectory" : 1,
Guido van Rossum0e548712002-08-09 16:14:33 +0000133 }
134
135 unexp = []
136 for key in dict:
137 if key[0] != '_' and key not in expected:
138 unexp.append(key)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000139 self.assertTrue(len(unexp) == 0,
Guido van Rossum0e548712002-08-09 16:14:33 +0000140 "unexpected keys: %s" % unexp)
141
Guido van Rossum0e548712002-08-09 16:14:33 +0000142
Antoine Pitroueab2a502012-03-10 16:34:40 +0100143class TestRandomNameSequence(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000144 """Test the internal iterator object _RandomNameSequence."""
145
146 def setUp(self):
147 self.r = tempfile._RandomNameSequence()
Brett Cannone1adece2010-03-20 22:19:55 +0000148 super().setUp()
Guido van Rossum0e548712002-08-09 16:14:33 +0000149
150 def test_get_six_char_str(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000151 # _RandomNameSequence returns a six-character string
Georg Brandla18af4e2007-04-21 15:47:16 +0000152 s = next(self.r)
Guido van Rossum0e548712002-08-09 16:14:33 +0000153 self.nameCheck(s, '', '', '')
154
155 def test_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000156 # _RandomNameSequence returns no duplicate strings (stochastic)
Guido van Rossum0e548712002-08-09 16:14:33 +0000157
158 dict = {}
159 r = self.r
Guido van Rossum805365e2007-05-07 22:24:25 +0000160 for i in range(TEST_FILES):
Georg Brandla18af4e2007-04-21 15:47:16 +0000161 s = next(r)
Guido van Rossum0e548712002-08-09 16:14:33 +0000162 self.nameCheck(s, '', '', '')
Ezio Melottib58e0bd2010-01-23 15:40:09 +0000163 self.assertNotIn(s, dict)
Guido van Rossum0e548712002-08-09 16:14:33 +0000164 dict[s] = 1
165
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000166 def supports_iter(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000167 # _RandomNameSequence supports the iterator protocol
Guido van Rossum0e548712002-08-09 16:14:33 +0000168
169 i = 0
170 r = self.r
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100171 for s in r:
172 i += 1
173 if i == 20:
174 break
Guido van Rossum0e548712002-08-09 16:14:33 +0000175
Antoine Pitrou4558bad2011-11-25 21:28:15 +0100176 @unittest.skipUnless(hasattr(os, 'fork'),
177 "os.fork is required for this test")
178 def test_process_awareness(self):
179 # ensure that the random source differs between
180 # child and parent.
181 read_fd, write_fd = os.pipe()
182 pid = None
183 try:
184 pid = os.fork()
185 if not pid:
Victor Stinner6c8c2942017-08-10 13:05:06 +0200186 # child process
Antoine Pitrou4558bad2011-11-25 21:28:15 +0100187 os.close(read_fd)
188 os.write(write_fd, next(self.r).encode("ascii"))
189 os.close(write_fd)
190 # bypass the normal exit handlers- leave those to
191 # the parent.
192 os._exit(0)
Victor Stinner6c8c2942017-08-10 13:05:06 +0200193
194 # parent process
Antoine Pitrou4558bad2011-11-25 21:28:15 +0100195 parent_value = next(self.r)
196 child_value = os.read(read_fd, len(parent_value)).decode("ascii")
197 finally:
198 if pid:
199 # best effort to ensure the process can't bleed out
200 # via any bugs above
201 try:
202 os.kill(pid, signal.SIGKILL)
Andrew Svetlov3438fa42012-12-17 23:35:18 +0200203 except OSError:
Antoine Pitrou4558bad2011-11-25 21:28:15 +0100204 pass
Victor Stinner6c8c2942017-08-10 13:05:06 +0200205
206 # Read the process exit status to avoid zombie process
207 os.waitpid(pid, 0)
208
Antoine Pitrou4558bad2011-11-25 21:28:15 +0100209 os.close(read_fd)
210 os.close(write_fd)
211 self.assertNotEqual(child_value, parent_value)
212
213
Guido van Rossum0e548712002-08-09 16:14:33 +0000214
Antoine Pitroueab2a502012-03-10 16:34:40 +0100215class TestCandidateTempdirList(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000216 """Test the internal function _candidate_tempdir_list."""
217
218 def test_nonempty_list(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000219 # _candidate_tempdir_list returns a nonempty list of strings
Guido van Rossum0e548712002-08-09 16:14:33 +0000220
221 cand = tempfile._candidate_tempdir_list()
222
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000223 self.assertFalse(len(cand) == 0)
Guido van Rossum0e548712002-08-09 16:14:33 +0000224 for c in cand:
Ezio Melottie9615932010-01-24 19:26:24 +0000225 self.assertIsInstance(c, str)
Guido van Rossum0e548712002-08-09 16:14:33 +0000226
227 def test_wanted_dirs(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000228 # _candidate_tempdir_list contains the expected directories
Guido van Rossum0e548712002-08-09 16:14:33 +0000229
230 # Make sure the interesting environment variables are all set.
Hirokazu Yamamoto71959632009-04-27 01:44:28 +0000231 with support.EnvironmentVarGuard() as env:
Guido van Rossum0e548712002-08-09 16:14:33 +0000232 for envname in 'TMPDIR', 'TEMP', 'TMP':
233 dirname = os.getenv(envname)
234 if not dirname:
Walter Dörwald155374d2009-05-01 19:58:58 +0000235 env[envname] = os.path.abspath(envname)
Guido van Rossum0e548712002-08-09 16:14:33 +0000236
237 cand = tempfile._candidate_tempdir_list()
238
239 for envname in 'TMPDIR', 'TEMP', 'TMP':
240 dirname = os.getenv(envname)
241 if not dirname: raise ValueError
Benjamin Peterson577473f2010-01-19 00:09:57 +0000242 self.assertIn(dirname, cand)
Guido van Rossum0e548712002-08-09 16:14:33 +0000243
244 try:
245 dirname = os.getcwd()
Andrew Svetlovad28c7f2012-12-18 22:02:39 +0200246 except (AttributeError, OSError):
Guido van Rossum0e548712002-08-09 16:14:33 +0000247 dirname = os.curdir
248
Benjamin Peterson577473f2010-01-19 00:09:57 +0000249 self.assertIn(dirname, cand)
Guido van Rossum0e548712002-08-09 16:14:33 +0000250
251 # Not practical to try to verify the presence of OS-specific
252 # paths in this list.
Guido van Rossum0e548712002-08-09 16:14:33 +0000253
Guido van Rossum0e548712002-08-09 16:14:33 +0000254
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200255# We test _get_default_tempdir some more by testing gettempdir.
Guido van Rossum0e548712002-08-09 16:14:33 +0000256
Serhiy Storchakaff7fef92013-02-13 00:37:29 +0200257class TestGetDefaultTempdir(BaseTestCase):
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200258 """Test _get_default_tempdir()."""
259
260 def test_no_files_left_behind(self):
261 # use a private empty directory
262 with tempfile.TemporaryDirectory() as our_temp_directory:
263 # force _get_default_tempdir() to consider our empty directory
264 def our_candidate_list():
265 return [our_temp_directory]
266
267 with support.swap_attr(tempfile, "_candidate_tempdir_list",
268 our_candidate_list):
269 # verify our directory is empty after _get_default_tempdir()
270 tempfile._get_default_tempdir()
271 self.assertEqual(os.listdir(our_temp_directory), [])
272
273 def raise_OSError(*args, **kwargs):
Serhiy Storchakaff7fef92013-02-13 00:37:29 +0200274 raise OSError()
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200275
276 with support.swap_attr(io, "open", raise_OSError):
277 # test again with failing io.open()
Serhiy Storchakaff7fef92013-02-13 00:37:29 +0200278 with self.assertRaises(FileNotFoundError):
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200279 tempfile._get_default_tempdir()
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200280 self.assertEqual(os.listdir(our_temp_directory), [])
281
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200282 def bad_writer(*args, **kwargs):
Serhiy Storchakad1a1def2017-04-28 19:17:26 +0300283 fp = orig_open(*args, **kwargs)
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200284 fp.write = raise_OSError
285 return fp
286
Serhiy Storchakad1a1def2017-04-28 19:17:26 +0300287 with support.swap_attr(io, "open", bad_writer) as orig_open:
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200288 # test again with failing write()
Serhiy Storchakaff7fef92013-02-13 00:37:29 +0200289 with self.assertRaises(FileNotFoundError):
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200290 tempfile._get_default_tempdir()
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200291 self.assertEqual(os.listdir(our_temp_directory), [])
Guido van Rossum0e548712002-08-09 16:14:33 +0000292
293
Antoine Pitroueab2a502012-03-10 16:34:40 +0100294class TestGetCandidateNames(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000295 """Test the internal function _get_candidate_names."""
296
297 def test_retval(self):
Victor Stinner1e62bf12017-04-19 22:59:51 +0200298 # _get_candidate_names returns a _RandomNameSequence object
Guido van Rossum0e548712002-08-09 16:14:33 +0000299 obj = tempfile._get_candidate_names()
Victor Stinner1e62bf12017-04-19 22:59:51 +0200300 self.assertIsInstance(obj, tempfile._RandomNameSequence)
Guido van Rossum0e548712002-08-09 16:14:33 +0000301
302 def test_same_thing(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000303 # _get_candidate_names always returns the same object
Guido van Rossum0e548712002-08-09 16:14:33 +0000304 a = tempfile._get_candidate_names()
305 b = tempfile._get_candidate_names()
306
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000307 self.assertTrue(a is b)
Guido van Rossum0e548712002-08-09 16:14:33 +0000308
Guido van Rossum0e548712002-08-09 16:14:33 +0000309
Eli Benderskyaa04f9a2013-09-13 05:28:20 -0700310@contextlib.contextmanager
311def _inside_empty_temp_dir():
312 dir = tempfile.mkdtemp()
313 try:
314 with support.swap_attr(tempfile, 'tempdir', dir):
315 yield
316 finally:
317 support.rmtree(dir)
318
319
320def _mock_candidate_names(*names):
321 return support.swap_attr(tempfile,
322 '_get_candidate_names',
323 lambda: iter(names))
324
325
Serhiy Storchaka5d6b7b12015-05-20 00:11:48 +0300326class TestBadTempdir:
327
328 def test_read_only_directory(self):
329 with _inside_empty_temp_dir():
330 oldmode = mode = os.stat(tempfile.tempdir).st_mode
331 mode &= ~(stat.S_IWUSR | stat.S_IWGRP | stat.S_IWOTH)
332 os.chmod(tempfile.tempdir, mode)
333 try:
334 if os.access(tempfile.tempdir, os.W_OK):
335 self.skipTest("can't set the directory read-only")
336 with self.assertRaises(PermissionError):
337 self.make_temp()
338 self.assertEqual(os.listdir(tempfile.tempdir), [])
339 finally:
340 os.chmod(tempfile.tempdir, oldmode)
341
342 def test_nonexisting_directory(self):
343 with _inside_empty_temp_dir():
344 tempdir = os.path.join(tempfile.tempdir, 'nonexistent')
345 with support.swap_attr(tempfile, 'tempdir', tempdir):
346 with self.assertRaises(FileNotFoundError):
347 self.make_temp()
348
349 def test_non_directory(self):
350 with _inside_empty_temp_dir():
351 tempdir = os.path.join(tempfile.tempdir, 'file')
352 open(tempdir, 'wb').close()
353 with support.swap_attr(tempfile, 'tempdir', tempdir):
354 with self.assertRaises((NotADirectoryError, FileNotFoundError)):
355 self.make_temp()
356
357
358class TestMkstempInner(TestBadTempdir, BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000359 """Test the internal function _mkstemp_inner."""
360
361 class mkstemped:
362 _bflags = tempfile._bin_openflags
363 _tflags = tempfile._text_openflags
364 _close = os.close
365 _unlink = os.unlink
366
367 def __init__(self, dir, pre, suf, bin):
368 if bin: flags = self._bflags
369 else: flags = self._tflags
370
Gregory P. Smithad577b92015-05-22 16:18:14 -0700371 output_type = tempfile._infer_return_type(dir, pre, suf)
372 (self.fd, self.name) = tempfile._mkstemp_inner(dir, pre, suf, flags, output_type)
Guido van Rossum0e548712002-08-09 16:14:33 +0000373
374 def write(self, str):
375 os.write(self.fd, str)
376
377 def __del__(self):
378 self._close(self.fd)
379 self._unlink(self.name)
Tim Petersa0d55de2002-08-09 18:01:01 +0000380
Gregory P. Smithad577b92015-05-22 16:18:14 -0700381 def do_create(self, dir=None, pre=None, suf=None, bin=1):
382 output_type = tempfile._infer_return_type(dir, pre, suf)
Guido van Rossum0e548712002-08-09 16:14:33 +0000383 if dir is None:
Gregory P. Smithad577b92015-05-22 16:18:14 -0700384 if output_type is str:
385 dir = tempfile.gettempdir()
386 else:
387 dir = tempfile.gettempdirb()
388 if pre is None:
389 pre = output_type()
390 if suf is None:
391 suf = output_type()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100392 file = self.mkstemped(dir, pre, suf, bin)
Guido van Rossum0e548712002-08-09 16:14:33 +0000393
394 self.nameCheck(file.name, dir, pre, suf)
395 return file
396
397 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000398 # _mkstemp_inner can create files
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000399 self.do_create().write(b"blat")
400 self.do_create(pre="a").write(b"blat")
401 self.do_create(suf="b").write(b"blat")
402 self.do_create(pre="a", suf="b").write(b"blat")
403 self.do_create(pre="aa", suf=".txt").write(b"blat")
Guido van Rossum0e548712002-08-09 16:14:33 +0000404
Gregory P. Smithad577b92015-05-22 16:18:14 -0700405 def test_basic_with_bytes_names(self):
406 # _mkstemp_inner can create files when given name parts all
407 # specified as bytes.
408 dir_b = tempfile.gettempdirb()
409 self.do_create(dir=dir_b, suf=b"").write(b"blat")
410 self.do_create(dir=dir_b, pre=b"a").write(b"blat")
411 self.do_create(dir=dir_b, suf=b"b").write(b"blat")
412 self.do_create(dir=dir_b, pre=b"a", suf=b"b").write(b"blat")
413 self.do_create(dir=dir_b, pre=b"aa", suf=b".txt").write(b"blat")
414 # Can't mix str & binary types in the args.
415 with self.assertRaises(TypeError):
416 self.do_create(dir="", suf=b"").write(b"blat")
417 with self.assertRaises(TypeError):
418 self.do_create(dir=dir_b, pre="").write(b"blat")
419 with self.assertRaises(TypeError):
420 self.do_create(dir=dir_b, pre=b"", suf="").write(b"blat")
421
Guido van Rossum0e548712002-08-09 16:14:33 +0000422 def test_basic_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000423 # _mkstemp_inner can create many files (stochastic)
Guido van Rossum805365e2007-05-07 22:24:25 +0000424 extant = list(range(TEST_FILES))
Guido van Rossum0e548712002-08-09 16:14:33 +0000425 for i in extant:
426 extant[i] = self.do_create(pre="aa")
427
428 def test_choose_directory(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000429 # _mkstemp_inner can create files in a user-selected directory
Guido van Rossum0e548712002-08-09 16:14:33 +0000430 dir = tempfile.mkdtemp()
431 try:
Antoine Pitrou9cadb1b2008-09-15 23:02:56 +0000432 self.do_create(dir=dir).write(b"blat")
Guido van Rossum0e548712002-08-09 16:14:33 +0000433 finally:
434 os.rmdir(dir)
435
Zachary Ware9fe6d862013-12-08 00:20:35 -0600436 @unittest.skipUnless(has_stat, 'os.stat not available')
Guido van Rossum0e548712002-08-09 16:14:33 +0000437 def test_file_mode(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000438 # _mkstemp_inner creates files with the proper mode
Guido van Rossum0e548712002-08-09 16:14:33 +0000439
440 file = self.do_create()
441 mode = stat.S_IMODE(os.stat(file.name).st_mode)
Guido van Rossumcd16bf62007-06-13 18:07:49 +0000442 expected = 0o600
Jesus Cea4791a242012-10-05 03:15:39 +0200443 if sys.platform == 'win32':
Tim Petersca3ac7f2002-08-09 18:13:51 +0000444 # There's no distinction among 'user', 'group' and 'world';
445 # replicate the 'user' bits.
446 user = expected >> 6
447 expected = user * (1 + 8 + 64)
448 self.assertEqual(mode, expected)
Guido van Rossum0e548712002-08-09 16:14:33 +0000449
Zachary Ware9fe6d862013-12-08 00:20:35 -0600450 @unittest.skipUnless(has_spawnl, 'os.spawnl not available')
Guido van Rossum0e548712002-08-09 16:14:33 +0000451 def test_noinherit(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000452 # _mkstemp_inner file handles are not inherited by child processes
Guido van Rossum0e548712002-08-09 16:14:33 +0000453
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000454 if support.verbose:
Guido van Rossum78741062002-08-17 11:41:01 +0000455 v="v"
456 else:
457 v="q"
458
Guido van Rossum0e548712002-08-09 16:14:33 +0000459 file = self.do_create()
Victor Stinnerdaf45552013-08-28 00:53:59 +0200460 self.assertEqual(os.get_inheritable(file.fd), False)
Guido van Rossum78741062002-08-17 11:41:01 +0000461 fd = "%d" % file.fd
462
463 try:
464 me = __file__
465 except NameError:
466 me = sys.argv[0]
Guido van Rossum0e548712002-08-09 16:14:33 +0000467
468 # We have to exec something, so that FD_CLOEXEC will take
Guido van Rossum78741062002-08-17 11:41:01 +0000469 # effect. The core of this test is therefore in
470 # tf_inherit_check.py, which see.
471 tester = os.path.join(os.path.dirname(os.path.abspath(me)),
472 "tf_inherit_check.py")
Guido van Rossum0e548712002-08-09 16:14:33 +0000473
Martin v. Löwisd4210bc2003-10-23 15:55:28 +0000474 # On Windows a spawn* /path/ with embedded spaces shouldn't be quoted,
475 # but an arg with embedded spaces should be decorated with double
476 # quotes on each end
Jesus Cea4791a242012-10-05 03:15:39 +0200477 if sys.platform == 'win32':
Martin v. Löwisd4210bc2003-10-23 15:55:28 +0000478 decorated = '"%s"' % sys.executable
479 tester = '"%s"' % tester
480 else:
481 decorated = sys.executable
482
483 retval = os.spawnl(os.P_WAIT, sys.executable, decorated, tester, v, fd)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000484 self.assertFalse(retval < 0,
Guido van Rossum78741062002-08-17 11:41:01 +0000485 "child process caught fatal signal %d" % -retval)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000486 self.assertFalse(retval > 0, "child process reports failure %d"%retval)
Guido van Rossum0e548712002-08-09 16:14:33 +0000487
Zachary Ware9fe6d862013-12-08 00:20:35 -0600488 @unittest.skipUnless(has_textmode, "text mode not available")
Guido van Rossum0e548712002-08-09 16:14:33 +0000489 def test_textmode(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000490 # _mkstemp_inner can create files in text mode
Guido van Rossum0e548712002-08-09 16:14:33 +0000491
Amaury Forgeot d'Arc7d0bddd2009-11-30 00:08:56 +0000492 # A text file is truncated at the first Ctrl+Z byte
493 f = self.do_create(bin=0)
494 f.write(b"blat\x1a")
495 f.write(b"extra\n")
496 os.lseek(f.fd, 0, os.SEEK_SET)
Ezio Melottib3aedd42010-11-20 19:04:17 +0000497 self.assertEqual(os.read(f.fd, 20), b"blat")
Guido van Rossum0e548712002-08-09 16:14:33 +0000498
Serhiy Storchaka5d6b7b12015-05-20 00:11:48 +0300499 def make_temp(self):
Eli Benderskyaa04f9a2013-09-13 05:28:20 -0700500 return tempfile._mkstemp_inner(tempfile.gettempdir(),
Gregory P. Smithad577b92015-05-22 16:18:14 -0700501 tempfile.gettempprefix(),
Eli Benderskyaa04f9a2013-09-13 05:28:20 -0700502 '',
Gregory P. Smithad577b92015-05-22 16:18:14 -0700503 tempfile._bin_openflags,
504 str)
Eli Benderskyaa04f9a2013-09-13 05:28:20 -0700505
506 def test_collision_with_existing_file(self):
507 # _mkstemp_inner tries another name when a file with
508 # the chosen name already exists
509 with _inside_empty_temp_dir(), \
510 _mock_candidate_names('aaa', 'aaa', 'bbb'):
Serhiy Storchaka5d6b7b12015-05-20 00:11:48 +0300511 (fd1, name1) = self.make_temp()
Eli Benderskyaa04f9a2013-09-13 05:28:20 -0700512 os.close(fd1)
513 self.assertTrue(name1.endswith('aaa'))
514
Serhiy Storchaka5d6b7b12015-05-20 00:11:48 +0300515 (fd2, name2) = self.make_temp()
Eli Benderskyaa04f9a2013-09-13 05:28:20 -0700516 os.close(fd2)
517 self.assertTrue(name2.endswith('bbb'))
518
Eli Benderskyf315df32013-09-06 06:11:19 -0700519 def test_collision_with_existing_directory(self):
520 # _mkstemp_inner tries another name when a directory with
521 # the chosen name already exists
Eli Benderskyaa04f9a2013-09-13 05:28:20 -0700522 with _inside_empty_temp_dir(), \
523 _mock_candidate_names('aaa', 'aaa', 'bbb'):
524 dir = tempfile.mkdtemp()
525 self.assertTrue(dir.endswith('aaa'))
Eli Benderskyf315df32013-09-06 06:11:19 -0700526
Serhiy Storchaka5d6b7b12015-05-20 00:11:48 +0300527 (fd, name) = self.make_temp()
Eli Benderskyaa04f9a2013-09-13 05:28:20 -0700528 os.close(fd)
529 self.assertTrue(name.endswith('bbb'))
Eli Benderskyf315df32013-09-06 06:11:19 -0700530
Guido van Rossum0e548712002-08-09 16:14:33 +0000531
Antoine Pitroueab2a502012-03-10 16:34:40 +0100532class TestGetTempPrefix(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000533 """Test gettempprefix()."""
534
535 def test_sane_template(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000536 # gettempprefix returns a nonempty prefix string
Guido van Rossum0e548712002-08-09 16:14:33 +0000537 p = tempfile.gettempprefix()
538
Ezio Melottie9615932010-01-24 19:26:24 +0000539 self.assertIsInstance(p, str)
Gregory P. Smithad577b92015-05-22 16:18:14 -0700540 self.assertGreater(len(p), 0)
541
542 pb = tempfile.gettempprefixb()
543
544 self.assertIsInstance(pb, bytes)
545 self.assertGreater(len(pb), 0)
Guido van Rossum0e548712002-08-09 16:14:33 +0000546
547 def test_usable_template(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000548 # gettempprefix returns a usable prefix string
Guido van Rossum0e548712002-08-09 16:14:33 +0000549
550 # Create a temp directory, avoiding use of the prefix.
551 # Then attempt to create a file whose name is
552 # prefix + 'xxxxxx.xxx' in that directory.
553 p = tempfile.gettempprefix() + "xxxxxx.xxx"
554 d = tempfile.mkdtemp(prefix="")
555 try:
556 p = os.path.join(d, p)
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100557 fd = os.open(p, os.O_RDWR | os.O_CREAT)
Guido van Rossum0e548712002-08-09 16:14:33 +0000558 os.close(fd)
559 os.unlink(p)
560 finally:
561 os.rmdir(d)
562
Guido van Rossum0e548712002-08-09 16:14:33 +0000563
Antoine Pitroueab2a502012-03-10 16:34:40 +0100564class TestGetTempDir(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000565 """Test gettempdir()."""
566
567 def test_directory_exists(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000568 # gettempdir returns a directory which exists
Guido van Rossum0e548712002-08-09 16:14:33 +0000569
Gregory P. Smithad577b92015-05-22 16:18:14 -0700570 for d in (tempfile.gettempdir(), tempfile.gettempdirb()):
571 self.assertTrue(os.path.isabs(d) or d == os.curdir,
572 "%r is not an absolute path" % d)
573 self.assertTrue(os.path.isdir(d),
574 "%r is not a directory" % d)
Guido van Rossum0e548712002-08-09 16:14:33 +0000575
576 def test_directory_writable(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000577 # gettempdir returns a directory writable by the user
Guido van Rossum0e548712002-08-09 16:14:33 +0000578
579 # sneaky: just instantiate a NamedTemporaryFile, which
580 # defaults to writing into the directory returned by
581 # gettempdir.
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100582 file = tempfile.NamedTemporaryFile()
583 file.write(b"blat")
584 file.close()
Guido van Rossum0e548712002-08-09 16:14:33 +0000585
586 def test_same_thing(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000587 # gettempdir always returns the same object
Guido van Rossum0e548712002-08-09 16:14:33 +0000588 a = tempfile.gettempdir()
589 b = tempfile.gettempdir()
Gregory P. Smithad577b92015-05-22 16:18:14 -0700590 c = tempfile.gettempdirb()
Guido van Rossum0e548712002-08-09 16:14:33 +0000591
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000592 self.assertTrue(a is b)
Gregory P. Smithad577b92015-05-22 16:18:14 -0700593 self.assertNotEqual(type(a), type(c))
594 self.assertEqual(a, os.fsdecode(c))
Guido van Rossum0e548712002-08-09 16:14:33 +0000595
Tim Golden6d09f092013-10-25 18:38:16 +0100596 def test_case_sensitive(self):
597 # gettempdir should not flatten its case
598 # even on a case-insensitive file system
599 case_sensitive_tempdir = tempfile.mkdtemp("-Temp")
600 _tempdir, tempfile.tempdir = tempfile.tempdir, None
601 try:
602 with support.EnvironmentVarGuard() as env:
603 # Fake the first env var which is checked as a candidate
604 env["TMPDIR"] = case_sensitive_tempdir
605 self.assertEqual(tempfile.gettempdir(), case_sensitive_tempdir)
606 finally:
607 tempfile.tempdir = _tempdir
608 support.rmdir(case_sensitive_tempdir)
609
Guido van Rossum0e548712002-08-09 16:14:33 +0000610
Antoine Pitroueab2a502012-03-10 16:34:40 +0100611class TestMkstemp(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000612 """Test mkstemp()."""
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000613
Gregory P. Smithad577b92015-05-22 16:18:14 -0700614 def do_create(self, dir=None, pre=None, suf=None):
615 output_type = tempfile._infer_return_type(dir, pre, suf)
Guido van Rossum0e548712002-08-09 16:14:33 +0000616 if dir is None:
Gregory P. Smithad577b92015-05-22 16:18:14 -0700617 if output_type is str:
618 dir = tempfile.gettempdir()
619 else:
620 dir = tempfile.gettempdirb()
621 if pre is None:
622 pre = output_type()
623 if suf is None:
624 suf = output_type()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100625 (fd, name) = tempfile.mkstemp(dir=dir, prefix=pre, suffix=suf)
626 (ndir, nbase) = os.path.split(name)
627 adir = os.path.abspath(dir)
628 self.assertEqual(adir, ndir,
629 "Directory '%s' incorrectly returned as '%s'" % (adir, ndir))
Guido van Rossum0e548712002-08-09 16:14:33 +0000630
631 try:
632 self.nameCheck(name, dir, pre, suf)
633 finally:
634 os.close(fd)
635 os.unlink(name)
636
637 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000638 # mkstemp can create files
Guido van Rossum0e548712002-08-09 16:14:33 +0000639 self.do_create()
640 self.do_create(pre="a")
641 self.do_create(suf="b")
642 self.do_create(pre="a", suf="b")
643 self.do_create(pre="aa", suf=".txt")
Martin v. Löwisd6625482003-10-12 17:37:01 +0000644 self.do_create(dir=".")
Guido van Rossum0e548712002-08-09 16:14:33 +0000645
Gregory P. Smithad577b92015-05-22 16:18:14 -0700646 def test_basic_with_bytes_names(self):
647 # mkstemp can create files when given name parts all
648 # specified as bytes.
649 d = tempfile.gettempdirb()
650 self.do_create(dir=d, suf=b"")
651 self.do_create(dir=d, pre=b"a")
652 self.do_create(dir=d, suf=b"b")
653 self.do_create(dir=d, pre=b"a", suf=b"b")
654 self.do_create(dir=d, pre=b"aa", suf=b".txt")
655 self.do_create(dir=b".")
656 with self.assertRaises(TypeError):
657 self.do_create(dir=".", pre=b"aa", suf=b".txt")
658 with self.assertRaises(TypeError):
659 self.do_create(dir=b".", pre="aa", suf=b".txt")
660 with self.assertRaises(TypeError):
661 self.do_create(dir=b".", pre=b"aa", suf=".txt")
662
663
Guido van Rossum0e548712002-08-09 16:14:33 +0000664 def test_choose_directory(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000665 # mkstemp can create directories in a user-selected directory
Guido van Rossum0e548712002-08-09 16:14:33 +0000666 dir = tempfile.mkdtemp()
667 try:
668 self.do_create(dir=dir)
669 finally:
670 os.rmdir(dir)
671
Guido van Rossum0e548712002-08-09 16:14:33 +0000672
Serhiy Storchaka5d6b7b12015-05-20 00:11:48 +0300673class TestMkdtemp(TestBadTempdir, BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000674 """Test mkdtemp()."""
675
Serhiy Storchaka5d6b7b12015-05-20 00:11:48 +0300676 def make_temp(self):
677 return tempfile.mkdtemp()
678
Gregory P. Smithad577b92015-05-22 16:18:14 -0700679 def do_create(self, dir=None, pre=None, suf=None):
680 output_type = tempfile._infer_return_type(dir, pre, suf)
Guido van Rossum0e548712002-08-09 16:14:33 +0000681 if dir is None:
Gregory P. Smithad577b92015-05-22 16:18:14 -0700682 if output_type is str:
683 dir = tempfile.gettempdir()
684 else:
685 dir = tempfile.gettempdirb()
686 if pre is None:
687 pre = output_type()
688 if suf is None:
689 suf = output_type()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100690 name = tempfile.mkdtemp(dir=dir, prefix=pre, suffix=suf)
Guido van Rossum0e548712002-08-09 16:14:33 +0000691
692 try:
693 self.nameCheck(name, dir, pre, suf)
694 return name
695 except:
696 os.rmdir(name)
697 raise
698
699 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000700 # mkdtemp can create directories
Guido van Rossum0e548712002-08-09 16:14:33 +0000701 os.rmdir(self.do_create())
702 os.rmdir(self.do_create(pre="a"))
703 os.rmdir(self.do_create(suf="b"))
704 os.rmdir(self.do_create(pre="a", suf="b"))
705 os.rmdir(self.do_create(pre="aa", suf=".txt"))
Tim Petersa0d55de2002-08-09 18:01:01 +0000706
Gregory P. Smithad577b92015-05-22 16:18:14 -0700707 def test_basic_with_bytes_names(self):
708 # mkdtemp can create directories when given all binary parts
709 d = tempfile.gettempdirb()
710 os.rmdir(self.do_create(dir=d))
711 os.rmdir(self.do_create(dir=d, pre=b"a"))
712 os.rmdir(self.do_create(dir=d, suf=b"b"))
713 os.rmdir(self.do_create(dir=d, pre=b"a", suf=b"b"))
714 os.rmdir(self.do_create(dir=d, pre=b"aa", suf=b".txt"))
715 with self.assertRaises(TypeError):
716 os.rmdir(self.do_create(dir=d, pre="aa", suf=b".txt"))
717 with self.assertRaises(TypeError):
718 os.rmdir(self.do_create(dir=d, pre=b"aa", suf=".txt"))
719 with self.assertRaises(TypeError):
720 os.rmdir(self.do_create(dir="", pre=b"aa", suf=b".txt"))
721
Guido van Rossum0e548712002-08-09 16:14:33 +0000722 def test_basic_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000723 # mkdtemp can create many directories (stochastic)
Guido van Rossum805365e2007-05-07 22:24:25 +0000724 extant = list(range(TEST_FILES))
Guido van Rossum0e548712002-08-09 16:14:33 +0000725 try:
726 for i in extant:
727 extant[i] = self.do_create(pre="aa")
728 finally:
729 for i in extant:
Guido van Rossum3172c5d2007-10-16 18:12:55 +0000730 if(isinstance(i, str)):
Guido van Rossum0e548712002-08-09 16:14:33 +0000731 os.rmdir(i)
732
733 def test_choose_directory(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000734 # mkdtemp can create directories in a user-selected directory
Guido van Rossum0e548712002-08-09 16:14:33 +0000735 dir = tempfile.mkdtemp()
736 try:
737 os.rmdir(self.do_create(dir=dir))
738 finally:
739 os.rmdir(dir)
740
Zachary Ware9fe6d862013-12-08 00:20:35 -0600741 @unittest.skipUnless(has_stat, 'os.stat not available')
Guido van Rossum0e548712002-08-09 16:14:33 +0000742 def test_mode(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000743 # mkdtemp creates directories with the proper mode
Guido van Rossum0e548712002-08-09 16:14:33 +0000744
745 dir = self.do_create()
746 try:
747 mode = stat.S_IMODE(os.stat(dir).st_mode)
Guido van Rossumcd16bf62007-06-13 18:07:49 +0000748 mode &= 0o777 # Mask off sticky bits inherited from /tmp
749 expected = 0o700
Jesus Cea4791a242012-10-05 03:15:39 +0200750 if sys.platform == 'win32':
Tim Petersca3ac7f2002-08-09 18:13:51 +0000751 # There's no distinction among 'user', 'group' and 'world';
752 # replicate the 'user' bits.
753 user = expected >> 6
754 expected = user * (1 + 8 + 64)
755 self.assertEqual(mode, expected)
Guido van Rossum0e548712002-08-09 16:14:33 +0000756 finally:
757 os.rmdir(dir)
758
Eli Benderskyaa04f9a2013-09-13 05:28:20 -0700759 def test_collision_with_existing_file(self):
760 # mkdtemp tries another name when a file with
761 # the chosen name already exists
762 with _inside_empty_temp_dir(), \
763 _mock_candidate_names('aaa', 'aaa', 'bbb'):
764 file = tempfile.NamedTemporaryFile(delete=False)
765 file.close()
766 self.assertTrue(file.name.endswith('aaa'))
767 dir = tempfile.mkdtemp()
768 self.assertTrue(dir.endswith('bbb'))
769
770 def test_collision_with_existing_directory(self):
771 # mkdtemp tries another name when a directory with
772 # the chosen name already exists
773 with _inside_empty_temp_dir(), \
774 _mock_candidate_names('aaa', 'aaa', 'bbb'):
775 dir1 = tempfile.mkdtemp()
776 self.assertTrue(dir1.endswith('aaa'))
777 dir2 = tempfile.mkdtemp()
778 self.assertTrue(dir2.endswith('bbb'))
779
Guido van Rossum0e548712002-08-09 16:14:33 +0000780
Antoine Pitroueab2a502012-03-10 16:34:40 +0100781class TestMktemp(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000782 """Test mktemp()."""
783
784 # For safety, all use of mktemp must occur in a private directory.
785 # We must also suppress the RuntimeWarning it generates.
786 def setUp(self):
787 self.dir = tempfile.mkdtemp()
Brett Cannone1adece2010-03-20 22:19:55 +0000788 super().setUp()
Guido van Rossum0e548712002-08-09 16:14:33 +0000789
790 def tearDown(self):
791 if self.dir:
792 os.rmdir(self.dir)
793 self.dir = None
Brett Cannone1adece2010-03-20 22:19:55 +0000794 super().tearDown()
Guido van Rossum0e548712002-08-09 16:14:33 +0000795
796 class mktemped:
797 _unlink = os.unlink
798 _bflags = tempfile._bin_openflags
799
800 def __init__(self, dir, pre, suf):
801 self.name = tempfile.mktemp(dir=dir, prefix=pre, suffix=suf)
802 # Create the file. This will raise an exception if it's
803 # mysteriously appeared in the meanwhile.
Guido van Rossumcd16bf62007-06-13 18:07:49 +0000804 os.close(os.open(self.name, self._bflags, 0o600))
Guido van Rossum0e548712002-08-09 16:14:33 +0000805
806 def __del__(self):
807 self._unlink(self.name)
808
809 def do_create(self, pre="", suf=""):
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100810 file = self.mktemped(self.dir, pre, suf)
Guido van Rossum0e548712002-08-09 16:14:33 +0000811
812 self.nameCheck(file.name, self.dir, pre, suf)
813 return file
814
815 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000816 # mktemp can choose usable file names
Guido van Rossum0e548712002-08-09 16:14:33 +0000817 self.do_create()
818 self.do_create(pre="a")
819 self.do_create(suf="b")
820 self.do_create(pre="a", suf="b")
821 self.do_create(pre="aa", suf=".txt")
822
823 def test_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000824 # mktemp can choose many usable file names (stochastic)
Guido van Rossum805365e2007-05-07 22:24:25 +0000825 extant = list(range(TEST_FILES))
Guido van Rossum0e548712002-08-09 16:14:33 +0000826 for i in extant:
827 extant[i] = self.do_create(pre="aa")
828
Fred Drake8bec4832002-11-22 20:13:43 +0000829## def test_warning(self):
830## # mktemp issues a warning when used
831## warnings.filterwarnings("error",
832## category=RuntimeWarning,
833## message="mktemp")
834## self.assertRaises(RuntimeWarning,
835## tempfile.mktemp, dir=self.dir)
Guido van Rossum0e548712002-08-09 16:14:33 +0000836
Guido van Rossum0e548712002-08-09 16:14:33 +0000837
838# We test _TemporaryFileWrapper by testing NamedTemporaryFile.
839
840
Antoine Pitroueab2a502012-03-10 16:34:40 +0100841class TestNamedTemporaryFile(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000842 """Test NamedTemporaryFile()."""
843
Guido van Rossumd8faa362007-04-27 19:54:29 +0000844 def do_create(self, dir=None, pre="", suf="", delete=True):
Guido van Rossum0e548712002-08-09 16:14:33 +0000845 if dir is None:
846 dir = tempfile.gettempdir()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100847 file = tempfile.NamedTemporaryFile(dir=dir, prefix=pre, suffix=suf,
848 delete=delete)
Guido van Rossum0e548712002-08-09 16:14:33 +0000849
850 self.nameCheck(file.name, dir, pre, suf)
851 return file
852
853
854 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000855 # NamedTemporaryFile can create files
Guido van Rossum0e548712002-08-09 16:14:33 +0000856 self.do_create()
857 self.do_create(pre="a")
858 self.do_create(suf="b")
859 self.do_create(pre="a", suf="b")
860 self.do_create(pre="aa", suf=".txt")
861
Antoine Pitrou17c93262013-12-21 22:14:56 +0100862 def test_method_lookup(self):
863 # Issue #18879: Looking up a temporary file method should keep it
864 # alive long enough.
865 f = self.do_create()
866 wr = weakref.ref(f)
867 write = f.write
868 write2 = f.write
869 del f
870 write(b'foo')
871 del write
872 write2(b'bar')
873 del write2
874 if support.check_impl_detail(cpython=True):
875 # No reference cycle was created.
876 self.assertIsNone(wr())
877
Serhiy Storchaka56cefa62015-03-19 15:23:15 +0200878 def test_iter(self):
879 # Issue #23700: getting iterator from a temporary file should keep
880 # it alive as long as it's being iterated over
881 lines = [b'spam\n', b'eggs\n', b'beans\n']
882 def make_file():
883 f = tempfile.NamedTemporaryFile(mode='w+b')
884 f.write(b''.join(lines))
885 f.seek(0)
886 return f
887 for i, l in enumerate(make_file()):
888 self.assertEqual(l, lines[i])
889 self.assertEqual(i, len(lines) - 1)
890
Guido van Rossum0e548712002-08-09 16:14:33 +0000891 def test_creates_named(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000892 # NamedTemporaryFile creates files with names
Guido van Rossum0e548712002-08-09 16:14:33 +0000893 f = tempfile.NamedTemporaryFile()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000894 self.assertTrue(os.path.exists(f.name),
Guido van Rossum0e548712002-08-09 16:14:33 +0000895 "NamedTemporaryFile %s does not exist" % f.name)
896
897 def test_del_on_close(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000898 # A NamedTemporaryFile is deleted when closed
Guido van Rossum0e548712002-08-09 16:14:33 +0000899 dir = tempfile.mkdtemp()
900 try:
901 f = tempfile.NamedTemporaryFile(dir=dir)
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000902 f.write(b'blat')
Guido van Rossum0e548712002-08-09 16:14:33 +0000903 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000904 self.assertFalse(os.path.exists(f.name),
Guido van Rossum0e548712002-08-09 16:14:33 +0000905 "NamedTemporaryFile %s exists after close" % f.name)
906 finally:
907 os.rmdir(dir)
908
Guido van Rossumd8faa362007-04-27 19:54:29 +0000909 def test_dis_del_on_close(self):
910 # Tests that delete-on-close can be disabled
911 dir = tempfile.mkdtemp()
912 tmp = None
913 try:
914 f = tempfile.NamedTemporaryFile(dir=dir, delete=False)
915 tmp = f.name
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000916 f.write(b'blat')
Guido van Rossumd8faa362007-04-27 19:54:29 +0000917 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000918 self.assertTrue(os.path.exists(f.name),
Guido van Rossumd8faa362007-04-27 19:54:29 +0000919 "NamedTemporaryFile %s missing after close" % f.name)
920 finally:
921 if tmp is not None:
922 os.unlink(tmp)
923 os.rmdir(dir)
924
Guido van Rossum0e548712002-08-09 16:14:33 +0000925 def test_multiple_close(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000926 # A NamedTemporaryFile can be closed many times without error
Guido van Rossum0e548712002-08-09 16:14:33 +0000927 f = tempfile.NamedTemporaryFile()
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000928 f.write(b'abc\n')
Guido van Rossum0e548712002-08-09 16:14:33 +0000929 f.close()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100930 f.close()
931 f.close()
Guido van Rossum0e548712002-08-09 16:14:33 +0000932
Christian Heimes3ecfea712008-02-09 20:51:34 +0000933 def test_context_manager(self):
934 # A NamedTemporaryFile can be used as a context manager
935 with tempfile.NamedTemporaryFile() as f:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000936 self.assertTrue(os.path.exists(f.name))
937 self.assertFalse(os.path.exists(f.name))
Christian Heimes3ecfea712008-02-09 20:51:34 +0000938 def use_closed():
939 with f:
940 pass
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000941 self.assertRaises(ValueError, use_closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000942
Victor Stinner1f99f9d2014-03-25 09:18:04 +0100943 def test_no_leak_fd(self):
Victor Stinner87d13ea2014-03-25 18:19:17 +0100944 # Issue #21058: don't leak file descriptor when io.open() fails
Victor Stinner1f99f9d2014-03-25 09:18:04 +0100945 closed = []
Victor Stinnerc61c1702014-07-29 01:13:39 +0200946 os_close = os.close
Victor Stinner1f99f9d2014-03-25 09:18:04 +0100947 def close(fd):
948 closed.append(fd)
Victor Stinnerc61c1702014-07-29 01:13:39 +0200949 os_close(fd)
Victor Stinner1f99f9d2014-03-25 09:18:04 +0100950
951 with mock.patch('os.close', side_effect=close):
952 with mock.patch('io.open', side_effect=ValueError):
953 self.assertRaises(ValueError, tempfile.NamedTemporaryFile)
954 self.assertEqual(len(closed), 1)
955
Martin Panter7869a222016-02-28 05:22:20 +0000956 def test_bad_mode(self):
957 dir = tempfile.mkdtemp()
958 self.addCleanup(support.rmtree, dir)
959 with self.assertRaises(ValueError):
960 tempfile.NamedTemporaryFile(mode='wr', dir=dir)
961 with self.assertRaises(TypeError):
962 tempfile.NamedTemporaryFile(mode=2, dir=dir)
963 self.assertEqual(os.listdir(dir), [])
Guido van Rossum0e548712002-08-09 16:14:33 +0000964
Martin Panter7869a222016-02-28 05:22:20 +0000965 # How to test the mode and bufsize parameters?
Guido van Rossum0e548712002-08-09 16:14:33 +0000966
Antoine Pitroueab2a502012-03-10 16:34:40 +0100967class TestSpooledTemporaryFile(BaseTestCase):
Guido van Rossumd8faa362007-04-27 19:54:29 +0000968 """Test SpooledTemporaryFile()."""
969
970 def do_create(self, max_size=0, dir=None, pre="", suf=""):
971 if dir is None:
972 dir = tempfile.gettempdir()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100973 file = tempfile.SpooledTemporaryFile(max_size=max_size, dir=dir, prefix=pre, suffix=suf)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000974
975 return file
976
977
978 def test_basic(self):
979 # SpooledTemporaryFile can create files
980 f = self.do_create()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000981 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000982 f = self.do_create(max_size=100, pre="a", suf=".txt")
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000983 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000984
985 def test_del_on_close(self):
986 # A SpooledTemporaryFile is deleted when closed
987 dir = tempfile.mkdtemp()
988 try:
989 f = tempfile.SpooledTemporaryFile(max_size=10, dir=dir)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000990 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +0000991 f.write(b'blat ' * 5)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000992 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000993 filename = f.name
994 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000995 self.assertFalse(isinstance(filename, str) and os.path.exists(filename),
Guido van Rossumd8faa362007-04-27 19:54:29 +0000996 "SpooledTemporaryFile %s exists after close" % filename)
997 finally:
998 os.rmdir(dir)
999
1000 def test_rewrite_small(self):
1001 # A SpooledTemporaryFile can be written to multiple within the max_size
1002 f = self.do_create(max_size=30)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001003 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +00001004 for i in range(5):
1005 f.seek(0, 0)
Guido van Rossum39478e82007-08-27 17:23:59 +00001006 f.write(b'x' * 20)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001007 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +00001008
1009 def test_write_sequential(self):
1010 # A SpooledTemporaryFile should hold exactly max_size bytes, and roll
1011 # over afterward
1012 f = self.do_create(max_size=30)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001013 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +00001014 f.write(b'x' * 20)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001015 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +00001016 f.write(b'x' * 10)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001017 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +00001018 f.write(b'x')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001019 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +00001020
R David Murrayd89ee792011-03-14 09:55:46 -04001021 def test_writelines(self):
1022 # Verify writelines with a SpooledTemporaryFile
1023 f = self.do_create()
1024 f.writelines((b'x', b'y', b'z'))
1025 f.seek(0)
1026 buf = f.read()
1027 self.assertEqual(buf, b'xyz')
1028
1029 def test_writelines_sequential(self):
1030 # A SpooledTemporaryFile should hold exactly max_size bytes, and roll
1031 # over afterward
1032 f = self.do_create(max_size=35)
1033 f.writelines((b'x' * 20, b'x' * 10, b'x' * 5))
1034 self.assertFalse(f._rolled)
1035 f.write(b'x')
1036 self.assertTrue(f._rolled)
1037
Guido van Rossumd8faa362007-04-27 19:54:29 +00001038 def test_sparse(self):
1039 # A SpooledTemporaryFile that is written late in the file will extend
1040 # when that occurs
1041 f = self.do_create(max_size=30)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001042 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +00001043 f.seek(100, 0)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001044 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +00001045 f.write(b'x')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001046 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +00001047
1048 def test_fileno(self):
1049 # A SpooledTemporaryFile should roll over to a real file on fileno()
1050 f = self.do_create(max_size=30)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001051 self.assertFalse(f._rolled)
1052 self.assertTrue(f.fileno() > 0)
1053 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +00001054
Christian Heimes3ecfea712008-02-09 20:51:34 +00001055 def test_multiple_close_before_rollover(self):
Guido van Rossumd8faa362007-04-27 19:54:29 +00001056 # A SpooledTemporaryFile can be closed many times without error
1057 f = tempfile.SpooledTemporaryFile()
Guido van Rossum39478e82007-08-27 17:23:59 +00001058 f.write(b'abc\n')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001059 self.assertFalse(f._rolled)
Christian Heimes3ecfea712008-02-09 20:51:34 +00001060 f.close()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +01001061 f.close()
1062 f.close()
Christian Heimes3ecfea712008-02-09 20:51:34 +00001063
1064 def test_multiple_close_after_rollover(self):
1065 # A SpooledTemporaryFile can be closed many times without error
1066 f = tempfile.SpooledTemporaryFile(max_size=1)
1067 f.write(b'abc\n')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001068 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +00001069 f.close()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +01001070 f.close()
1071 f.close()
Guido van Rossumd8faa362007-04-27 19:54:29 +00001072
1073 def test_bound_methods(self):
1074 # It should be OK to steal a bound method from a SpooledTemporaryFile
1075 # and use it independently; when the file rolls over, those bound
1076 # methods should continue to function
1077 f = self.do_create(max_size=30)
1078 read = f.read
1079 write = f.write
1080 seek = f.seek
1081
Guido van Rossum39478e82007-08-27 17:23:59 +00001082 write(b"a" * 35)
1083 write(b"b" * 35)
Guido van Rossumd8faa362007-04-27 19:54:29 +00001084 seek(0, 0)
Guido van Rossum9a634702007-07-09 10:24:45 +00001085 self.assertEqual(read(70), b'a'*35 + b'b'*35)
1086
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +02001087 def test_properties(self):
1088 f = tempfile.SpooledTemporaryFile(max_size=10)
1089 f.write(b'x' * 10)
1090 self.assertFalse(f._rolled)
1091 self.assertEqual(f.mode, 'w+b')
1092 self.assertIsNone(f.name)
1093 with self.assertRaises(AttributeError):
1094 f.newlines
1095 with self.assertRaises(AttributeError):
1096 f.encoding
1097
1098 f.write(b'x')
1099 self.assertTrue(f._rolled)
1100 self.assertEqual(f.mode, 'rb+')
1101 self.assertIsNotNone(f.name)
1102 with self.assertRaises(AttributeError):
1103 f.newlines
1104 with self.assertRaises(AttributeError):
1105 f.encoding
1106
Guido van Rossum9a634702007-07-09 10:24:45 +00001107 def test_text_mode(self):
1108 # Creating a SpooledTemporaryFile with a text mode should produce
1109 # a file object reading and writing (Unicode) text strings.
1110 f = tempfile.SpooledTemporaryFile(mode='w+', max_size=10)
1111 f.write("abc\n")
1112 f.seek(0)
1113 self.assertEqual(f.read(), "abc\n")
1114 f.write("def\n")
1115 f.seek(0)
1116 self.assertEqual(f.read(), "abc\ndef\n")
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +02001117 self.assertFalse(f._rolled)
1118 self.assertEqual(f.mode, 'w+')
1119 self.assertIsNone(f.name)
1120 self.assertIsNone(f.newlines)
1121 self.assertIsNone(f.encoding)
1122
Guido van Rossum9a634702007-07-09 10:24:45 +00001123 f.write("xyzzy\n")
1124 f.seek(0)
1125 self.assertEqual(f.read(), "abc\ndef\nxyzzy\n")
Amaury Forgeot d'Arc7d0bddd2009-11-30 00:08:56 +00001126 # Check that Ctrl+Z doesn't truncate the file
1127 f.write("foo\x1abar\n")
1128 f.seek(0)
1129 self.assertEqual(f.read(), "abc\ndef\nxyzzy\nfoo\x1abar\n")
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +02001130 self.assertTrue(f._rolled)
1131 self.assertEqual(f.mode, 'w+')
1132 self.assertIsNotNone(f.name)
Serhiy Storchaka497cee42013-02-10 14:43:46 +02001133 self.assertEqual(f.newlines, os.linesep)
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +02001134 self.assertIsNotNone(f.encoding)
Guido van Rossum9a634702007-07-09 10:24:45 +00001135
Guido van Rossumf0c74162007-08-28 03:29:45 +00001136 def test_text_newline_and_encoding(self):
1137 f = tempfile.SpooledTemporaryFile(mode='w+', max_size=10,
1138 newline='', encoding='utf-8')
1139 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)
Guido van Rossumf0c74162007-08-28 03:29:45 +00001147
1148 f.write("\u039B" * 20 + "\r\n")
1149 f.seek(0)
1150 self.assertEqual(f.read(), "\u039B\r\n" + ("\u039B" * 20) + "\r\n")
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001151 self.assertTrue(f._rolled)
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +02001152 self.assertEqual(f.mode, 'w+')
1153 self.assertIsNotNone(f.name)
1154 self.assertIsNotNone(f.newlines)
1155 self.assertEqual(f.encoding, 'utf-8')
Guido van Rossumd8faa362007-04-27 19:54:29 +00001156
Christian Heimes3ecfea712008-02-09 20:51:34 +00001157 def test_context_manager_before_rollover(self):
1158 # A SpooledTemporaryFile can be used as a context manager
1159 with tempfile.SpooledTemporaryFile(max_size=1) as f:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001160 self.assertFalse(f._rolled)
1161 self.assertFalse(f.closed)
1162 self.assertTrue(f.closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +00001163 def use_closed():
1164 with f:
1165 pass
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001166 self.assertRaises(ValueError, use_closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +00001167
1168 def test_context_manager_during_rollover(self):
1169 # A SpooledTemporaryFile can be used as a context manager
1170 with tempfile.SpooledTemporaryFile(max_size=1) as f:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001171 self.assertFalse(f._rolled)
Christian Heimes3ecfea712008-02-09 20:51:34 +00001172 f.write(b'abc\n')
1173 f.flush()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001174 self.assertTrue(f._rolled)
1175 self.assertFalse(f.closed)
1176 self.assertTrue(f.closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +00001177 def use_closed():
1178 with f:
1179 pass
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001180 self.assertRaises(ValueError, use_closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +00001181
1182 def test_context_manager_after_rollover(self):
1183 # A SpooledTemporaryFile can be used as a context manager
1184 f = tempfile.SpooledTemporaryFile(max_size=1)
1185 f.write(b'abc\n')
1186 f.flush()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001187 self.assertTrue(f._rolled)
Christian Heimes3ecfea712008-02-09 20:51:34 +00001188 with f:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001189 self.assertFalse(f.closed)
1190 self.assertTrue(f.closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +00001191 def use_closed():
1192 with f:
1193 pass
Benjamin Petersonc9c0f202009-06-30 23:06:06 +00001194 self.assertRaises(ValueError, use_closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +00001195
Antoine Pitrou0e86a582011-11-25 18:03:09 +01001196 def test_truncate_with_size_parameter(self):
1197 # A SpooledTemporaryFile can be truncated to zero size
1198 f = tempfile.SpooledTemporaryFile(max_size=10)
1199 f.write(b'abcdefg\n')
1200 f.seek(0)
1201 f.truncate()
1202 self.assertFalse(f._rolled)
1203 self.assertEqual(f._file.getvalue(), b'')
1204 # A SpooledTemporaryFile can be truncated to a specific size
1205 f = tempfile.SpooledTemporaryFile(max_size=10)
1206 f.write(b'abcdefg\n')
1207 f.truncate(4)
1208 self.assertFalse(f._rolled)
1209 self.assertEqual(f._file.getvalue(), b'abcd')
1210 # A SpooledTemporaryFile rolls over if truncated to large size
1211 f = tempfile.SpooledTemporaryFile(max_size=10)
1212 f.write(b'abcdefg\n')
1213 f.truncate(20)
1214 self.assertTrue(f._rolled)
1215 if has_stat:
1216 self.assertEqual(os.fstat(f.fileno()).st_size, 20)
Christian Heimes3ecfea712008-02-09 20:51:34 +00001217
Guido van Rossum0e548712002-08-09 16:14:33 +00001218
Guido van Rossum8ccd9b62002-08-22 20:02:03 +00001219if tempfile.NamedTemporaryFile is not tempfile.TemporaryFile:
Antoine Pitroueab2a502012-03-10 16:34:40 +01001220
1221 class TestTemporaryFile(BaseTestCase):
1222 """Test TemporaryFile()."""
1223
1224 def test_basic(self):
1225 # TemporaryFile can create files
1226 # No point in testing the name params - the file has no name.
1227 tempfile.TemporaryFile()
1228
1229 def test_has_no_name(self):
1230 # TemporaryFile creates files with no names (on this system)
1231 dir = tempfile.mkdtemp()
1232 f = tempfile.TemporaryFile(dir=dir)
1233 f.write(b'blat')
1234
1235 # Sneaky: because this file has no name, it should not prevent
1236 # us from removing the directory it was created in.
1237 try:
1238 os.rmdir(dir)
1239 except:
1240 # cleanup
1241 f.close()
1242 os.rmdir(dir)
1243 raise
1244
1245 def test_multiple_close(self):
1246 # A TemporaryFile can be closed many times without error
1247 f = tempfile.TemporaryFile()
1248 f.write(b'abc\n')
1249 f.close()
1250 f.close()
1251 f.close()
1252
1253 # How to test the mode and bufsize parameters?
1254 def test_mode_and_encoding(self):
1255
1256 def roundtrip(input, *args, **kwargs):
1257 with tempfile.TemporaryFile(*args, **kwargs) as fileobj:
1258 fileobj.write(input)
1259 fileobj.seek(0)
1260 self.assertEqual(input, fileobj.read())
1261
1262 roundtrip(b"1234", "w+b")
1263 roundtrip("abdc\n", "w+")
1264 roundtrip("\u039B", "w+", encoding="utf-16")
1265 roundtrip("foo\r\n", "w+", newline="")
Guido van Rossum0e548712002-08-09 16:14:33 +00001266
Victor Stinner1f99f9d2014-03-25 09:18:04 +01001267 def test_no_leak_fd(self):
1268 # Issue #21058: don't leak file descriptor when io.open() fails
1269 closed = []
Victor Stinnerc61c1702014-07-29 01:13:39 +02001270 os_close = os.close
Victor Stinner1f99f9d2014-03-25 09:18:04 +01001271 def close(fd):
1272 closed.append(fd)
Victor Stinnerc61c1702014-07-29 01:13:39 +02001273 os_close(fd)
Victor Stinner1f99f9d2014-03-25 09:18:04 +01001274
1275 with mock.patch('os.close', side_effect=close):
1276 with mock.patch('io.open', side_effect=ValueError):
1277 self.assertRaises(ValueError, tempfile.TemporaryFile)
1278 self.assertEqual(len(closed), 1)
1279
1280
Nick Coghlan543af752010-10-24 11:23:25 +00001281
1282# Helper for test_del_on_shutdown
1283class NulledModules:
1284 def __init__(self, *modules):
1285 self.refs = [mod.__dict__ for mod in modules]
1286 self.contents = [ref.copy() for ref in self.refs]
1287
1288 def __enter__(self):
1289 for d in self.refs:
1290 for key in d:
1291 d[key] = None
1292
1293 def __exit__(self, *exc_info):
1294 for d, c in zip(self.refs, self.contents):
1295 d.clear()
1296 d.update(c)
1297
Antoine Pitroueab2a502012-03-10 16:34:40 +01001298class TestTemporaryDirectory(BaseTestCase):
Nick Coghlan543af752010-10-24 11:23:25 +00001299 """Test TemporaryDirectory()."""
1300
1301 def do_create(self, dir=None, pre="", suf="", recurse=1):
1302 if dir is None:
1303 dir = tempfile.gettempdir()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +01001304 tmp = tempfile.TemporaryDirectory(dir=dir, prefix=pre, suffix=suf)
Nick Coghlan543af752010-10-24 11:23:25 +00001305 self.nameCheck(tmp.name, dir, pre, suf)
1306 # Create a subdirectory and some files
1307 if recurse:
Serhiy Storchaka99e033b2014-01-27 11:18:27 +02001308 d1 = self.do_create(tmp.name, pre, suf, recurse-1)
1309 d1.name = None
Nick Coghlan543af752010-10-24 11:23:25 +00001310 with open(os.path.join(tmp.name, "test.txt"), "wb") as f:
1311 f.write(b"Hello world!")
1312 return tmp
1313
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001314 def test_mkdtemp_failure(self):
1315 # Check no additional exception if mkdtemp fails
1316 # Previously would raise AttributeError instead
Nick Coghlan3c54ea62010-12-13 03:02:43 +00001317 # (noted as part of Issue #10188)
1318 with tempfile.TemporaryDirectory() as nonexistent:
1319 pass
Serhiy Storchaka7451a722013-02-09 22:25:49 +02001320 with self.assertRaises(FileNotFoundError) as cm:
Nick Coghlan3c54ea62010-12-13 03:02:43 +00001321 tempfile.TemporaryDirectory(dir=nonexistent)
Serhiy Storchaka7451a722013-02-09 22:25:49 +02001322 self.assertEqual(cm.exception.errno, errno.ENOENT)
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001323
Nick Coghlan543af752010-10-24 11:23:25 +00001324 def test_explicit_cleanup(self):
1325 # A TemporaryDirectory is deleted when cleaned up
1326 dir = tempfile.mkdtemp()
1327 try:
1328 d = self.do_create(dir=dir)
1329 self.assertTrue(os.path.exists(d.name),
1330 "TemporaryDirectory %s does not exist" % d.name)
1331 d.cleanup()
1332 self.assertFalse(os.path.exists(d.name),
1333 "TemporaryDirectory %s exists after cleanup" % d.name)
1334 finally:
1335 os.rmdir(dir)
1336
Charles-François Natalidef35432011-07-29 18:59:24 +02001337 @support.skip_unless_symlink
1338 def test_cleanup_with_symlink_to_a_directory(self):
1339 # cleanup() should not follow symlinks to directories (issue #12464)
1340 d1 = self.do_create()
Serhiy Storchaka99e033b2014-01-27 11:18:27 +02001341 d2 = self.do_create(recurse=0)
Charles-François Natalidef35432011-07-29 18:59:24 +02001342
1343 # Symlink d1/foo -> d2
1344 os.symlink(d2.name, os.path.join(d1.name, "foo"))
1345
1346 # This call to cleanup() should not follow the "foo" symlink
1347 d1.cleanup()
1348
1349 self.assertFalse(os.path.exists(d1.name),
1350 "TemporaryDirectory %s exists after cleanup" % d1.name)
1351 self.assertTrue(os.path.exists(d2.name),
1352 "Directory pointed to by a symlink was deleted")
1353 self.assertEqual(os.listdir(d2.name), ['test.txt'],
1354 "Contents of the directory pointed to by a symlink "
1355 "were deleted")
1356 d2.cleanup()
1357
Nick Coghlan543af752010-10-24 11:23:25 +00001358 @support.cpython_only
1359 def test_del_on_collection(self):
1360 # A TemporaryDirectory is deleted when garbage collected
1361 dir = tempfile.mkdtemp()
1362 try:
1363 d = self.do_create(dir=dir)
1364 name = d.name
1365 del d # Rely on refcounting to invoke __del__
1366 self.assertFalse(os.path.exists(name),
1367 "TemporaryDirectory %s exists after __del__" % name)
1368 finally:
1369 os.rmdir(dir)
1370
Nick Coghlan543af752010-10-24 11:23:25 +00001371 def test_del_on_shutdown(self):
1372 # A TemporaryDirectory may be cleaned up during shutdown
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001373 with self.do_create() as dir:
Serhiy Storchakaa28632b2014-01-27 11:21:54 +02001374 for mod in ('builtins', 'os', 'shutil', 'sys', 'tempfile', 'warnings'):
Serhiy Storchaka99e033b2014-01-27 11:18:27 +02001375 code = """if True:
Serhiy Storchakaa28632b2014-01-27 11:21:54 +02001376 import builtins
Serhiy Storchaka99e033b2014-01-27 11:18:27 +02001377 import os
1378 import shutil
1379 import sys
1380 import tempfile
1381 import warnings
1382
1383 tmp = tempfile.TemporaryDirectory(dir={dir!r})
1384 sys.stdout.buffer.write(tmp.name.encode())
1385
1386 tmp2 = os.path.join(tmp.name, 'test_dir')
1387 os.mkdir(tmp2)
1388 with open(os.path.join(tmp2, "test.txt"), "w") as f:
1389 f.write("Hello world!")
1390
1391 {mod}.tmp = tmp
1392
1393 warnings.filterwarnings("always", category=ResourceWarning)
1394 """.format(dir=dir, mod=mod)
1395 rc, out, err = script_helper.assert_python_ok("-c", code)
1396 tmp_name = out.decode().strip()
1397 self.assertFalse(os.path.exists(tmp_name),
1398 "TemporaryDirectory %s exists after cleanup" % tmp_name)
1399 err = err.decode('utf-8', 'backslashreplace')
1400 self.assertNotIn("Exception ", err)
Serhiy Storchakaa28632b2014-01-27 11:21:54 +02001401 self.assertIn("ResourceWarning: Implicitly cleaning up", err)
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001402
Serhiy Storchaka5e193ac2014-09-24 13:26:25 +03001403 def test_exit_on_shutdown(self):
1404 # Issue #22427
1405 with self.do_create() as dir:
1406 code = """if True:
1407 import sys
1408 import tempfile
1409 import warnings
1410
1411 def generator():
1412 with tempfile.TemporaryDirectory(dir={dir!r}) as tmp:
1413 yield tmp
1414 g = generator()
1415 sys.stdout.buffer.write(next(g).encode())
1416
1417 warnings.filterwarnings("always", category=ResourceWarning)
1418 """.format(dir=dir)
1419 rc, out, err = script_helper.assert_python_ok("-c", code)
1420 tmp_name = out.decode().strip()
1421 self.assertFalse(os.path.exists(tmp_name),
1422 "TemporaryDirectory %s exists after cleanup" % tmp_name)
1423 err = err.decode('utf-8', 'backslashreplace')
1424 self.assertNotIn("Exception ", err)
1425 self.assertIn("ResourceWarning: Implicitly cleaning up", err)
1426
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001427 def test_warnings_on_cleanup(self):
Serhiy Storchaka99e033b2014-01-27 11:18:27 +02001428 # ResourceWarning will be triggered by __del__
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001429 with self.do_create() as dir:
Serhiy Storchaka99e033b2014-01-27 11:18:27 +02001430 d = self.do_create(dir=dir, recurse=3)
1431 name = d.name
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001432
1433 # Check for the resource warning
1434 with support.check_warnings(('Implicitly', ResourceWarning), quiet=False):
1435 warnings.filterwarnings("always", category=ResourceWarning)
Serhiy Storchaka99e033b2014-01-27 11:18:27 +02001436 del d
1437 support.gc_collect()
1438 self.assertFalse(os.path.exists(name),
1439 "TemporaryDirectory %s exists after __del__" % name)
Nick Coghlan543af752010-10-24 11:23:25 +00001440
1441 def test_multiple_close(self):
1442 # Can be cleaned-up many times without error
1443 d = self.do_create()
1444 d.cleanup()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +01001445 d.cleanup()
1446 d.cleanup()
Nick Coghlan543af752010-10-24 11:23:25 +00001447
1448 def test_context_manager(self):
1449 # Can be used as a context manager
1450 d = self.do_create()
1451 with d as name:
1452 self.assertTrue(os.path.exists(name))
1453 self.assertEqual(name, d.name)
1454 self.assertFalse(os.path.exists(name))
1455
1456
Guido van Rossum0e548712002-08-09 16:14:33 +00001457if __name__ == "__main__":
Zachary Ware38c707e2015-04-13 15:00:43 -05001458 unittest.main()