blob: cf2ae080bed9269d4cb4c40d27e964fefb779988 [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
Tim Petersc57a2852001-10-29 21:46:08 +000012
Guido van Rossum0e548712002-08-09 16:14:33 +000013import unittest
Serhiy Storchaka99e033b2014-01-27 11:18:27 +020014from test import support, script_helper
Guido van Rossum0e548712002-08-09 16:14:33 +000015
Fred Drake7633d232002-10-17 22:09:03 +000016
Guido van Rossum0e548712002-08-09 16:14:33 +000017if hasattr(os, 'stat'):
18 import stat
19 has_stat = 1
20else:
21 has_stat = 0
22
23has_textmode = (tempfile._text_openflags != tempfile._bin_openflags)
Guido van Rossum78741062002-08-17 11:41:01 +000024has_spawnl = hasattr(os, 'spawnl')
Guido van Rossum0e548712002-08-09 16:14:33 +000025
Neal Norwitz68ee0122002-08-16 19:28:59 +000026# TEST_FILES may need to be tweaked for systems depending on the maximum
27# number of files that can be opened at one time (see ulimit -n)
Victor Stinner9c3de4a2011-08-17 20:49:41 +020028if sys.platform.startswith('openbsd'):
Martin v. Löwis99968282004-09-15 06:02:54 +000029 TEST_FILES = 48
Jack Jansence921472003-01-08 16:30:34 +000030else:
31 TEST_FILES = 100
Neal Norwitz68ee0122002-08-16 19:28:59 +000032
Guido van Rossum0e548712002-08-09 16:14:33 +000033# This is organized as one test for each chunk of code in tempfile.py,
34# in order of their appearance in the file. Testing which requires
35# threads is not done here.
36
37# Common functionality.
Antoine Pitroueab2a502012-03-10 16:34:40 +010038class BaseTestCase(unittest.TestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +000039
Victor Stinner97869102013-08-14 01:28:28 +020040 str_check = re.compile(r"^[a-z0-9_-]{8}$")
Guido van Rossum0e548712002-08-09 16:14:33 +000041
Brett Cannone1adece2010-03-20 22:19:55 +000042 def setUp(self):
43 self._warnings_manager = support.check_warnings()
44 self._warnings_manager.__enter__()
45 warnings.filterwarnings("ignore", category=RuntimeWarning,
46 message="mktemp", module=__name__)
47
48 def tearDown(self):
49 self._warnings_manager.__exit__(None, None, None)
50
51
Guido van Rossum0e548712002-08-09 16:14:33 +000052 def nameCheck(self, name, dir, pre, suf):
53 (ndir, nbase) = os.path.split(name)
54 npre = nbase[:len(pre)]
55 nsuf = nbase[len(nbase)-len(suf):]
56
Martin v. Löwisd6625482003-10-12 17:37:01 +000057 # check for equality of the absolute paths!
58 self.assertEqual(os.path.abspath(ndir), os.path.abspath(dir),
Guido van Rossum0e548712002-08-09 16:14:33 +000059 "file '%s' not in directory '%s'" % (name, dir))
60 self.assertEqual(npre, pre,
61 "file '%s' does not begin with '%s'" % (nbase, pre))
62 self.assertEqual(nsuf, suf,
63 "file '%s' does not end with '%s'" % (nbase, suf))
64
65 nbase = nbase[len(pre):len(nbase)-len(suf)]
Benjamin Petersonc9c0f202009-06-30 23:06:06 +000066 self.assertTrue(self.str_check.match(nbase),
Victor Stinner97869102013-08-14 01:28:28 +020067 "random string '%s' does not match ^[a-z0-9_-]{8}$"
Guido van Rossum0e548712002-08-09 16:14:33 +000068 % nbase)
69
Guido van Rossum0e548712002-08-09 16:14:33 +000070
Antoine Pitroueab2a502012-03-10 16:34:40 +010071class TestExports(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +000072 def test_exports(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +000073 # There are no surprising symbols in the tempfile module
Guido van Rossum0e548712002-08-09 16:14:33 +000074 dict = tempfile.__dict__
75
76 expected = {
77 "NamedTemporaryFile" : 1,
78 "TemporaryFile" : 1,
79 "mkstemp" : 1,
80 "mkdtemp" : 1,
81 "mktemp" : 1,
82 "TMP_MAX" : 1,
83 "gettempprefix" : 1,
84 "gettempdir" : 1,
85 "tempdir" : 1,
Guido van Rossumd8faa362007-04-27 19:54:29 +000086 "template" : 1,
Nick Coghlan543af752010-10-24 11:23:25 +000087 "SpooledTemporaryFile" : 1,
88 "TemporaryDirectory" : 1,
Guido van Rossum0e548712002-08-09 16:14:33 +000089 }
90
91 unexp = []
92 for key in dict:
93 if key[0] != '_' and key not in expected:
94 unexp.append(key)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +000095 self.assertTrue(len(unexp) == 0,
Guido van Rossum0e548712002-08-09 16:14:33 +000096 "unexpected keys: %s" % unexp)
97
Guido van Rossum0e548712002-08-09 16:14:33 +000098
Antoine Pitroueab2a502012-03-10 16:34:40 +010099class TestRandomNameSequence(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000100 """Test the internal iterator object _RandomNameSequence."""
101
102 def setUp(self):
103 self.r = tempfile._RandomNameSequence()
Brett Cannone1adece2010-03-20 22:19:55 +0000104 super().setUp()
Guido van Rossum0e548712002-08-09 16:14:33 +0000105
106 def test_get_six_char_str(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000107 # _RandomNameSequence returns a six-character string
Georg Brandla18af4e2007-04-21 15:47:16 +0000108 s = next(self.r)
Guido van Rossum0e548712002-08-09 16:14:33 +0000109 self.nameCheck(s, '', '', '')
110
111 def test_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000112 # _RandomNameSequence returns no duplicate strings (stochastic)
Guido van Rossum0e548712002-08-09 16:14:33 +0000113
114 dict = {}
115 r = self.r
Guido van Rossum805365e2007-05-07 22:24:25 +0000116 for i in range(TEST_FILES):
Georg Brandla18af4e2007-04-21 15:47:16 +0000117 s = next(r)
Guido van Rossum0e548712002-08-09 16:14:33 +0000118 self.nameCheck(s, '', '', '')
Ezio Melottib58e0bd2010-01-23 15:40:09 +0000119 self.assertNotIn(s, dict)
Guido van Rossum0e548712002-08-09 16:14:33 +0000120 dict[s] = 1
121
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000122 def supports_iter(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000123 # _RandomNameSequence supports the iterator protocol
Guido van Rossum0e548712002-08-09 16:14:33 +0000124
125 i = 0
126 r = self.r
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100127 for s in r:
128 i += 1
129 if i == 20:
130 break
Guido van Rossum0e548712002-08-09 16:14:33 +0000131
Antoine Pitrou4558bad2011-11-25 21:28:15 +0100132 @unittest.skipUnless(hasattr(os, 'fork'),
133 "os.fork is required for this test")
134 def test_process_awareness(self):
135 # ensure that the random source differs between
136 # child and parent.
137 read_fd, write_fd = os.pipe()
138 pid = None
139 try:
140 pid = os.fork()
141 if not pid:
142 os.close(read_fd)
143 os.write(write_fd, next(self.r).encode("ascii"))
144 os.close(write_fd)
145 # bypass the normal exit handlers- leave those to
146 # the parent.
147 os._exit(0)
148 parent_value = next(self.r)
149 child_value = os.read(read_fd, len(parent_value)).decode("ascii")
150 finally:
151 if pid:
152 # best effort to ensure the process can't bleed out
153 # via any bugs above
154 try:
155 os.kill(pid, signal.SIGKILL)
Andrew Svetlov3438fa42012-12-17 23:35:18 +0200156 except OSError:
Antoine Pitrou4558bad2011-11-25 21:28:15 +0100157 pass
158 os.close(read_fd)
159 os.close(write_fd)
160 self.assertNotEqual(child_value, parent_value)
161
162
Guido van Rossum0e548712002-08-09 16:14:33 +0000163
Antoine Pitroueab2a502012-03-10 16:34:40 +0100164class TestCandidateTempdirList(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000165 """Test the internal function _candidate_tempdir_list."""
166
167 def test_nonempty_list(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000168 # _candidate_tempdir_list returns a nonempty list of strings
Guido van Rossum0e548712002-08-09 16:14:33 +0000169
170 cand = tempfile._candidate_tempdir_list()
171
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000172 self.assertFalse(len(cand) == 0)
Guido van Rossum0e548712002-08-09 16:14:33 +0000173 for c in cand:
Ezio Melottie9615932010-01-24 19:26:24 +0000174 self.assertIsInstance(c, str)
Guido van Rossum0e548712002-08-09 16:14:33 +0000175
176 def test_wanted_dirs(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000177 # _candidate_tempdir_list contains the expected directories
Guido van Rossum0e548712002-08-09 16:14:33 +0000178
179 # Make sure the interesting environment variables are all set.
Hirokazu Yamamoto71959632009-04-27 01:44:28 +0000180 with support.EnvironmentVarGuard() as env:
Guido van Rossum0e548712002-08-09 16:14:33 +0000181 for envname in 'TMPDIR', 'TEMP', 'TMP':
182 dirname = os.getenv(envname)
183 if not dirname:
Walter Dörwald155374d2009-05-01 19:58:58 +0000184 env[envname] = os.path.abspath(envname)
Guido van Rossum0e548712002-08-09 16:14:33 +0000185
186 cand = tempfile._candidate_tempdir_list()
187
188 for envname in 'TMPDIR', 'TEMP', 'TMP':
189 dirname = os.getenv(envname)
190 if not dirname: raise ValueError
Benjamin Peterson577473f2010-01-19 00:09:57 +0000191 self.assertIn(dirname, cand)
Guido van Rossum0e548712002-08-09 16:14:33 +0000192
193 try:
194 dirname = os.getcwd()
Andrew Svetlovad28c7f2012-12-18 22:02:39 +0200195 except (AttributeError, OSError):
Guido van Rossum0e548712002-08-09 16:14:33 +0000196 dirname = os.curdir
197
Benjamin Peterson577473f2010-01-19 00:09:57 +0000198 self.assertIn(dirname, cand)
Guido van Rossum0e548712002-08-09 16:14:33 +0000199
200 # Not practical to try to verify the presence of OS-specific
201 # paths in this list.
Guido van Rossum0e548712002-08-09 16:14:33 +0000202
Guido van Rossum0e548712002-08-09 16:14:33 +0000203
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200204# We test _get_default_tempdir some more by testing gettempdir.
Guido van Rossum0e548712002-08-09 16:14:33 +0000205
Serhiy Storchakaff7fef92013-02-13 00:37:29 +0200206class TestGetDefaultTempdir(BaseTestCase):
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200207 """Test _get_default_tempdir()."""
208
209 def test_no_files_left_behind(self):
210 # use a private empty directory
211 with tempfile.TemporaryDirectory() as our_temp_directory:
212 # force _get_default_tempdir() to consider our empty directory
213 def our_candidate_list():
214 return [our_temp_directory]
215
216 with support.swap_attr(tempfile, "_candidate_tempdir_list",
217 our_candidate_list):
218 # verify our directory is empty after _get_default_tempdir()
219 tempfile._get_default_tempdir()
220 self.assertEqual(os.listdir(our_temp_directory), [])
221
222 def raise_OSError(*args, **kwargs):
Serhiy Storchakaff7fef92013-02-13 00:37:29 +0200223 raise OSError()
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200224
225 with support.swap_attr(io, "open", raise_OSError):
226 # test again with failing io.open()
Serhiy Storchakaff7fef92013-02-13 00:37:29 +0200227 with self.assertRaises(FileNotFoundError):
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200228 tempfile._get_default_tempdir()
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200229 self.assertEqual(os.listdir(our_temp_directory), [])
230
231 open = io.open
232 def bad_writer(*args, **kwargs):
233 fp = open(*args, **kwargs)
234 fp.write = raise_OSError
235 return fp
236
237 with support.swap_attr(io, "open", bad_writer):
238 # test again with failing write()
Serhiy Storchakaff7fef92013-02-13 00:37:29 +0200239 with self.assertRaises(FileNotFoundError):
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200240 tempfile._get_default_tempdir()
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200241 self.assertEqual(os.listdir(our_temp_directory), [])
Guido van Rossum0e548712002-08-09 16:14:33 +0000242
243
Antoine Pitroueab2a502012-03-10 16:34:40 +0100244class TestGetCandidateNames(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000245 """Test the internal function _get_candidate_names."""
246
247 def test_retval(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000248 # _get_candidate_names returns a _RandomNameSequence object
Guido van Rossum0e548712002-08-09 16:14:33 +0000249 obj = tempfile._get_candidate_names()
Ezio Melottie9615932010-01-24 19:26:24 +0000250 self.assertIsInstance(obj, tempfile._RandomNameSequence)
Guido van Rossum0e548712002-08-09 16:14:33 +0000251
252 def test_same_thing(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000253 # _get_candidate_names always returns the same object
Guido van Rossum0e548712002-08-09 16:14:33 +0000254 a = tempfile._get_candidate_names()
255 b = tempfile._get_candidate_names()
256
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000257 self.assertTrue(a is b)
Guido van Rossum0e548712002-08-09 16:14:33 +0000258
Guido van Rossum0e548712002-08-09 16:14:33 +0000259
Eli Benderskyaa04f9a2013-09-13 05:28:20 -0700260@contextlib.contextmanager
261def _inside_empty_temp_dir():
262 dir = tempfile.mkdtemp()
263 try:
264 with support.swap_attr(tempfile, 'tempdir', dir):
265 yield
266 finally:
267 support.rmtree(dir)
268
269
270def _mock_candidate_names(*names):
271 return support.swap_attr(tempfile,
272 '_get_candidate_names',
273 lambda: iter(names))
274
275
Antoine Pitroueab2a502012-03-10 16:34:40 +0100276class TestMkstempInner(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000277 """Test the internal function _mkstemp_inner."""
278
279 class mkstemped:
280 _bflags = tempfile._bin_openflags
281 _tflags = tempfile._text_openflags
282 _close = os.close
283 _unlink = os.unlink
284
285 def __init__(self, dir, pre, suf, bin):
286 if bin: flags = self._bflags
287 else: flags = self._tflags
288
289 (self.fd, self.name) = tempfile._mkstemp_inner(dir, pre, suf, flags)
290
291 def write(self, str):
292 os.write(self.fd, str)
293
294 def __del__(self):
295 self._close(self.fd)
296 self._unlink(self.name)
Tim Petersa0d55de2002-08-09 18:01:01 +0000297
Guido van Rossum0e548712002-08-09 16:14:33 +0000298 def do_create(self, dir=None, pre="", suf="", bin=1):
299 if dir is None:
300 dir = tempfile.gettempdir()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100301 file = self.mkstemped(dir, pre, suf, bin)
Guido van Rossum0e548712002-08-09 16:14:33 +0000302
303 self.nameCheck(file.name, dir, pre, suf)
304 return file
305
306 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000307 # _mkstemp_inner can create files
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000308 self.do_create().write(b"blat")
309 self.do_create(pre="a").write(b"blat")
310 self.do_create(suf="b").write(b"blat")
311 self.do_create(pre="a", suf="b").write(b"blat")
312 self.do_create(pre="aa", suf=".txt").write(b"blat")
Guido van Rossum0e548712002-08-09 16:14:33 +0000313
314 def test_basic_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000315 # _mkstemp_inner can create many files (stochastic)
Guido van Rossum805365e2007-05-07 22:24:25 +0000316 extant = list(range(TEST_FILES))
Guido van Rossum0e548712002-08-09 16:14:33 +0000317 for i in extant:
318 extant[i] = self.do_create(pre="aa")
319
320 def test_choose_directory(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000321 # _mkstemp_inner can create files in a user-selected directory
Guido van Rossum0e548712002-08-09 16:14:33 +0000322 dir = tempfile.mkdtemp()
323 try:
Antoine Pitrou9cadb1b2008-09-15 23:02:56 +0000324 self.do_create(dir=dir).write(b"blat")
Guido van Rossum0e548712002-08-09 16:14:33 +0000325 finally:
326 os.rmdir(dir)
327
Zachary Ware9fe6d862013-12-08 00:20:35 -0600328 @unittest.skipUnless(has_stat, 'os.stat not available')
Guido van Rossum0e548712002-08-09 16:14:33 +0000329 def test_file_mode(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000330 # _mkstemp_inner creates files with the proper mode
Guido van Rossum0e548712002-08-09 16:14:33 +0000331
332 file = self.do_create()
333 mode = stat.S_IMODE(os.stat(file.name).st_mode)
Guido van Rossumcd16bf62007-06-13 18:07:49 +0000334 expected = 0o600
Jesus Cea4791a242012-10-05 03:15:39 +0200335 if sys.platform == 'win32':
Tim Petersca3ac7f2002-08-09 18:13:51 +0000336 # There's no distinction among 'user', 'group' and 'world';
337 # replicate the 'user' bits.
338 user = expected >> 6
339 expected = user * (1 + 8 + 64)
340 self.assertEqual(mode, expected)
Guido van Rossum0e548712002-08-09 16:14:33 +0000341
Zachary Ware9fe6d862013-12-08 00:20:35 -0600342 @unittest.skipUnless(has_spawnl, 'os.spawnl not available')
Guido van Rossum0e548712002-08-09 16:14:33 +0000343 def test_noinherit(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000344 # _mkstemp_inner file handles are not inherited by child processes
Guido van Rossum0e548712002-08-09 16:14:33 +0000345
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000346 if support.verbose:
Guido van Rossum78741062002-08-17 11:41:01 +0000347 v="v"
348 else:
349 v="q"
350
Guido van Rossum0e548712002-08-09 16:14:33 +0000351 file = self.do_create()
Victor Stinnerdaf45552013-08-28 00:53:59 +0200352 self.assertEqual(os.get_inheritable(file.fd), False)
Guido van Rossum78741062002-08-17 11:41:01 +0000353 fd = "%d" % file.fd
354
355 try:
356 me = __file__
357 except NameError:
358 me = sys.argv[0]
Guido van Rossum0e548712002-08-09 16:14:33 +0000359
360 # We have to exec something, so that FD_CLOEXEC will take
Guido van Rossum78741062002-08-17 11:41:01 +0000361 # effect. The core of this test is therefore in
362 # tf_inherit_check.py, which see.
363 tester = os.path.join(os.path.dirname(os.path.abspath(me)),
364 "tf_inherit_check.py")
Guido van Rossum0e548712002-08-09 16:14:33 +0000365
Martin v. Löwisd4210bc2003-10-23 15:55:28 +0000366 # On Windows a spawn* /path/ with embedded spaces shouldn't be quoted,
367 # but an arg with embedded spaces should be decorated with double
368 # quotes on each end
Jesus Cea4791a242012-10-05 03:15:39 +0200369 if sys.platform == 'win32':
Martin v. Löwisd4210bc2003-10-23 15:55:28 +0000370 decorated = '"%s"' % sys.executable
371 tester = '"%s"' % tester
372 else:
373 decorated = sys.executable
374
375 retval = os.spawnl(os.P_WAIT, sys.executable, decorated, tester, v, fd)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000376 self.assertFalse(retval < 0,
Guido van Rossum78741062002-08-17 11:41:01 +0000377 "child process caught fatal signal %d" % -retval)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000378 self.assertFalse(retval > 0, "child process reports failure %d"%retval)
Guido van Rossum0e548712002-08-09 16:14:33 +0000379
Zachary Ware9fe6d862013-12-08 00:20:35 -0600380 @unittest.skipUnless(has_textmode, "text mode not available")
Guido van Rossum0e548712002-08-09 16:14:33 +0000381 def test_textmode(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000382 # _mkstemp_inner can create files in text mode
Guido van Rossum0e548712002-08-09 16:14:33 +0000383
Amaury Forgeot d'Arc7d0bddd2009-11-30 00:08:56 +0000384 # A text file is truncated at the first Ctrl+Z byte
385 f = self.do_create(bin=0)
386 f.write(b"blat\x1a")
387 f.write(b"extra\n")
388 os.lseek(f.fd, 0, os.SEEK_SET)
Ezio Melottib3aedd42010-11-20 19:04:17 +0000389 self.assertEqual(os.read(f.fd, 20), b"blat")
Guido van Rossum0e548712002-08-09 16:14:33 +0000390
Eli Benderskyaa04f9a2013-09-13 05:28:20 -0700391 def default_mkstemp_inner(self):
392 return tempfile._mkstemp_inner(tempfile.gettempdir(),
393 tempfile.template,
394 '',
395 tempfile._bin_openflags)
396
397 def test_collision_with_existing_file(self):
398 # _mkstemp_inner tries another name when a file with
399 # the chosen name already exists
400 with _inside_empty_temp_dir(), \
401 _mock_candidate_names('aaa', 'aaa', 'bbb'):
402 (fd1, name1) = self.default_mkstemp_inner()
403 os.close(fd1)
404 self.assertTrue(name1.endswith('aaa'))
405
406 (fd2, name2) = self.default_mkstemp_inner()
407 os.close(fd2)
408 self.assertTrue(name2.endswith('bbb'))
409
Eli Benderskyf315df32013-09-06 06:11:19 -0700410 def test_collision_with_existing_directory(self):
411 # _mkstemp_inner tries another name when a directory with
412 # the chosen name already exists
Eli Benderskyaa04f9a2013-09-13 05:28:20 -0700413 with _inside_empty_temp_dir(), \
414 _mock_candidate_names('aaa', 'aaa', 'bbb'):
415 dir = tempfile.mkdtemp()
416 self.assertTrue(dir.endswith('aaa'))
Eli Benderskyf315df32013-09-06 06:11:19 -0700417
Eli Benderskyaa04f9a2013-09-13 05:28:20 -0700418 (fd, name) = self.default_mkstemp_inner()
419 os.close(fd)
420 self.assertTrue(name.endswith('bbb'))
Eli Benderskyf315df32013-09-06 06:11:19 -0700421
Guido van Rossum0e548712002-08-09 16:14:33 +0000422
Antoine Pitroueab2a502012-03-10 16:34:40 +0100423class TestGetTempPrefix(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000424 """Test gettempprefix()."""
425
426 def test_sane_template(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000427 # gettempprefix returns a nonempty prefix string
Guido van Rossum0e548712002-08-09 16:14:33 +0000428 p = tempfile.gettempprefix()
429
Ezio Melottie9615932010-01-24 19:26:24 +0000430 self.assertIsInstance(p, str)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000431 self.assertTrue(len(p) > 0)
Guido van Rossum0e548712002-08-09 16:14:33 +0000432
433 def test_usable_template(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000434 # gettempprefix returns a usable prefix string
Guido van Rossum0e548712002-08-09 16:14:33 +0000435
436 # Create a temp directory, avoiding use of the prefix.
437 # Then attempt to create a file whose name is
438 # prefix + 'xxxxxx.xxx' in that directory.
439 p = tempfile.gettempprefix() + "xxxxxx.xxx"
440 d = tempfile.mkdtemp(prefix="")
441 try:
442 p = os.path.join(d, p)
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100443 fd = os.open(p, os.O_RDWR | os.O_CREAT)
Guido van Rossum0e548712002-08-09 16:14:33 +0000444 os.close(fd)
445 os.unlink(p)
446 finally:
447 os.rmdir(d)
448
Guido van Rossum0e548712002-08-09 16:14:33 +0000449
Antoine Pitroueab2a502012-03-10 16:34:40 +0100450class TestGetTempDir(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000451 """Test gettempdir()."""
452
453 def test_directory_exists(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000454 # gettempdir returns a directory which exists
Guido van Rossum0e548712002-08-09 16:14:33 +0000455
456 dir = tempfile.gettempdir()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000457 self.assertTrue(os.path.isabs(dir) or dir == os.curdir,
Guido van Rossum0e548712002-08-09 16:14:33 +0000458 "%s is not an absolute path" % dir)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000459 self.assertTrue(os.path.isdir(dir),
Guido van Rossum0e548712002-08-09 16:14:33 +0000460 "%s is not a directory" % dir)
461
462 def test_directory_writable(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000463 # gettempdir returns a directory writable by the user
Guido van Rossum0e548712002-08-09 16:14:33 +0000464
465 # sneaky: just instantiate a NamedTemporaryFile, which
466 # defaults to writing into the directory returned by
467 # gettempdir.
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100468 file = tempfile.NamedTemporaryFile()
469 file.write(b"blat")
470 file.close()
Guido van Rossum0e548712002-08-09 16:14:33 +0000471
472 def test_same_thing(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000473 # gettempdir always returns the same object
Guido van Rossum0e548712002-08-09 16:14:33 +0000474 a = tempfile.gettempdir()
475 b = tempfile.gettempdir()
476
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000477 self.assertTrue(a is b)
Guido van Rossum0e548712002-08-09 16:14:33 +0000478
Tim Golden6d09f092013-10-25 18:38:16 +0100479 def test_case_sensitive(self):
480 # gettempdir should not flatten its case
481 # even on a case-insensitive file system
482 case_sensitive_tempdir = tempfile.mkdtemp("-Temp")
483 _tempdir, tempfile.tempdir = tempfile.tempdir, None
484 try:
485 with support.EnvironmentVarGuard() as env:
486 # Fake the first env var which is checked as a candidate
487 env["TMPDIR"] = case_sensitive_tempdir
488 self.assertEqual(tempfile.gettempdir(), case_sensitive_tempdir)
489 finally:
490 tempfile.tempdir = _tempdir
491 support.rmdir(case_sensitive_tempdir)
492
Guido van Rossum0e548712002-08-09 16:14:33 +0000493
Antoine Pitroueab2a502012-03-10 16:34:40 +0100494class TestMkstemp(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000495 """Test mkstemp()."""
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000496
Thomas Wouters73e5a5b2006-06-08 15:35:45 +0000497 def do_create(self, dir=None, pre="", suf=""):
Guido van Rossum0e548712002-08-09 16:14:33 +0000498 if dir is None:
499 dir = tempfile.gettempdir()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100500 (fd, name) = tempfile.mkstemp(dir=dir, prefix=pre, suffix=suf)
501 (ndir, nbase) = os.path.split(name)
502 adir = os.path.abspath(dir)
503 self.assertEqual(adir, ndir,
504 "Directory '%s' incorrectly returned as '%s'" % (adir, ndir))
Guido van Rossum0e548712002-08-09 16:14:33 +0000505
506 try:
507 self.nameCheck(name, dir, pre, suf)
508 finally:
509 os.close(fd)
510 os.unlink(name)
511
512 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000513 # mkstemp can create files
Guido van Rossum0e548712002-08-09 16:14:33 +0000514 self.do_create()
515 self.do_create(pre="a")
516 self.do_create(suf="b")
517 self.do_create(pre="a", suf="b")
518 self.do_create(pre="aa", suf=".txt")
Martin v. Löwisd6625482003-10-12 17:37:01 +0000519 self.do_create(dir=".")
Guido van Rossum0e548712002-08-09 16:14:33 +0000520
521 def test_choose_directory(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000522 # mkstemp can create directories in a user-selected directory
Guido van Rossum0e548712002-08-09 16:14:33 +0000523 dir = tempfile.mkdtemp()
524 try:
525 self.do_create(dir=dir)
526 finally:
527 os.rmdir(dir)
528
Guido van Rossum0e548712002-08-09 16:14:33 +0000529
Antoine Pitroueab2a502012-03-10 16:34:40 +0100530class TestMkdtemp(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000531 """Test mkdtemp()."""
532
533 def do_create(self, dir=None, pre="", suf=""):
534 if dir is None:
535 dir = tempfile.gettempdir()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100536 name = tempfile.mkdtemp(dir=dir, prefix=pre, suffix=suf)
Guido van Rossum0e548712002-08-09 16:14:33 +0000537
538 try:
539 self.nameCheck(name, dir, pre, suf)
540 return name
541 except:
542 os.rmdir(name)
543 raise
544
545 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000546 # mkdtemp can create directories
Guido van Rossum0e548712002-08-09 16:14:33 +0000547 os.rmdir(self.do_create())
548 os.rmdir(self.do_create(pre="a"))
549 os.rmdir(self.do_create(suf="b"))
550 os.rmdir(self.do_create(pre="a", suf="b"))
551 os.rmdir(self.do_create(pre="aa", suf=".txt"))
Tim Petersa0d55de2002-08-09 18:01:01 +0000552
Guido van Rossum0e548712002-08-09 16:14:33 +0000553 def test_basic_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000554 # mkdtemp can create many directories (stochastic)
Guido van Rossum805365e2007-05-07 22:24:25 +0000555 extant = list(range(TEST_FILES))
Guido van Rossum0e548712002-08-09 16:14:33 +0000556 try:
557 for i in extant:
558 extant[i] = self.do_create(pre="aa")
559 finally:
560 for i in extant:
Guido van Rossum3172c5d2007-10-16 18:12:55 +0000561 if(isinstance(i, str)):
Guido van Rossum0e548712002-08-09 16:14:33 +0000562 os.rmdir(i)
563
564 def test_choose_directory(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000565 # mkdtemp can create directories in a user-selected directory
Guido van Rossum0e548712002-08-09 16:14:33 +0000566 dir = tempfile.mkdtemp()
567 try:
568 os.rmdir(self.do_create(dir=dir))
569 finally:
570 os.rmdir(dir)
571
Zachary Ware9fe6d862013-12-08 00:20:35 -0600572 @unittest.skipUnless(has_stat, 'os.stat not available')
Guido van Rossum0e548712002-08-09 16:14:33 +0000573 def test_mode(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000574 # mkdtemp creates directories with the proper mode
Guido van Rossum0e548712002-08-09 16:14:33 +0000575
576 dir = self.do_create()
577 try:
578 mode = stat.S_IMODE(os.stat(dir).st_mode)
Guido van Rossumcd16bf62007-06-13 18:07:49 +0000579 mode &= 0o777 # Mask off sticky bits inherited from /tmp
580 expected = 0o700
Jesus Cea4791a242012-10-05 03:15:39 +0200581 if sys.platform == 'win32':
Tim Petersca3ac7f2002-08-09 18:13:51 +0000582 # There's no distinction among 'user', 'group' and 'world';
583 # replicate the 'user' bits.
584 user = expected >> 6
585 expected = user * (1 + 8 + 64)
586 self.assertEqual(mode, expected)
Guido van Rossum0e548712002-08-09 16:14:33 +0000587 finally:
588 os.rmdir(dir)
589
Eli Benderskyaa04f9a2013-09-13 05:28:20 -0700590 def test_collision_with_existing_file(self):
591 # mkdtemp tries another name when a file with
592 # the chosen name already exists
593 with _inside_empty_temp_dir(), \
594 _mock_candidate_names('aaa', 'aaa', 'bbb'):
595 file = tempfile.NamedTemporaryFile(delete=False)
596 file.close()
597 self.assertTrue(file.name.endswith('aaa'))
598 dir = tempfile.mkdtemp()
599 self.assertTrue(dir.endswith('bbb'))
600
601 def test_collision_with_existing_directory(self):
602 # mkdtemp tries another name when a directory with
603 # the chosen name already exists
604 with _inside_empty_temp_dir(), \
605 _mock_candidate_names('aaa', 'aaa', 'bbb'):
606 dir1 = tempfile.mkdtemp()
607 self.assertTrue(dir1.endswith('aaa'))
608 dir2 = tempfile.mkdtemp()
609 self.assertTrue(dir2.endswith('bbb'))
610
Guido van Rossum0e548712002-08-09 16:14:33 +0000611
Antoine Pitroueab2a502012-03-10 16:34:40 +0100612class TestMktemp(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000613 """Test mktemp()."""
614
615 # For safety, all use of mktemp must occur in a private directory.
616 # We must also suppress the RuntimeWarning it generates.
617 def setUp(self):
618 self.dir = tempfile.mkdtemp()
Brett Cannone1adece2010-03-20 22:19:55 +0000619 super().setUp()
Guido van Rossum0e548712002-08-09 16:14:33 +0000620
621 def tearDown(self):
622 if self.dir:
623 os.rmdir(self.dir)
624 self.dir = None
Brett Cannone1adece2010-03-20 22:19:55 +0000625 super().tearDown()
Guido van Rossum0e548712002-08-09 16:14:33 +0000626
627 class mktemped:
628 _unlink = os.unlink
629 _bflags = tempfile._bin_openflags
630
631 def __init__(self, dir, pre, suf):
632 self.name = tempfile.mktemp(dir=dir, prefix=pre, suffix=suf)
633 # Create the file. This will raise an exception if it's
634 # mysteriously appeared in the meanwhile.
Guido van Rossumcd16bf62007-06-13 18:07:49 +0000635 os.close(os.open(self.name, self._bflags, 0o600))
Guido van Rossum0e548712002-08-09 16:14:33 +0000636
637 def __del__(self):
638 self._unlink(self.name)
639
640 def do_create(self, pre="", suf=""):
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100641 file = self.mktemped(self.dir, pre, suf)
Guido van Rossum0e548712002-08-09 16:14:33 +0000642
643 self.nameCheck(file.name, self.dir, pre, suf)
644 return file
645
646 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000647 # mktemp can choose usable file names
Guido van Rossum0e548712002-08-09 16:14:33 +0000648 self.do_create()
649 self.do_create(pre="a")
650 self.do_create(suf="b")
651 self.do_create(pre="a", suf="b")
652 self.do_create(pre="aa", suf=".txt")
653
654 def test_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000655 # mktemp can choose many usable file names (stochastic)
Guido van Rossum805365e2007-05-07 22:24:25 +0000656 extant = list(range(TEST_FILES))
Guido van Rossum0e548712002-08-09 16:14:33 +0000657 for i in extant:
658 extant[i] = self.do_create(pre="aa")
659
Fred Drake8bec4832002-11-22 20:13:43 +0000660## def test_warning(self):
661## # mktemp issues a warning when used
662## warnings.filterwarnings("error",
663## category=RuntimeWarning,
664## message="mktemp")
665## self.assertRaises(RuntimeWarning,
666## tempfile.mktemp, dir=self.dir)
Guido van Rossum0e548712002-08-09 16:14:33 +0000667
Guido van Rossum0e548712002-08-09 16:14:33 +0000668
669# We test _TemporaryFileWrapper by testing NamedTemporaryFile.
670
671
Antoine Pitroueab2a502012-03-10 16:34:40 +0100672class TestNamedTemporaryFile(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000673 """Test NamedTemporaryFile()."""
674
Guido van Rossumd8faa362007-04-27 19:54:29 +0000675 def do_create(self, dir=None, pre="", suf="", delete=True):
Guido van Rossum0e548712002-08-09 16:14:33 +0000676 if dir is None:
677 dir = tempfile.gettempdir()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100678 file = tempfile.NamedTemporaryFile(dir=dir, prefix=pre, suffix=suf,
679 delete=delete)
Guido van Rossum0e548712002-08-09 16:14:33 +0000680
681 self.nameCheck(file.name, dir, pre, suf)
682 return file
683
684
685 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000686 # NamedTemporaryFile can create files
Guido van Rossum0e548712002-08-09 16:14:33 +0000687 self.do_create()
688 self.do_create(pre="a")
689 self.do_create(suf="b")
690 self.do_create(pre="a", suf="b")
691 self.do_create(pre="aa", suf=".txt")
692
Antoine Pitrou17c93262013-12-21 22:14:56 +0100693 def test_method_lookup(self):
694 # Issue #18879: Looking up a temporary file method should keep it
695 # alive long enough.
696 f = self.do_create()
697 wr = weakref.ref(f)
698 write = f.write
699 write2 = f.write
700 del f
701 write(b'foo')
702 del write
703 write2(b'bar')
704 del write2
705 if support.check_impl_detail(cpython=True):
706 # No reference cycle was created.
707 self.assertIsNone(wr())
708
Guido van Rossum0e548712002-08-09 16:14:33 +0000709 def test_creates_named(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000710 # NamedTemporaryFile creates files with names
Guido van Rossum0e548712002-08-09 16:14:33 +0000711 f = tempfile.NamedTemporaryFile()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000712 self.assertTrue(os.path.exists(f.name),
Guido van Rossum0e548712002-08-09 16:14:33 +0000713 "NamedTemporaryFile %s does not exist" % f.name)
714
715 def test_del_on_close(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000716 # A NamedTemporaryFile is deleted when closed
Guido van Rossum0e548712002-08-09 16:14:33 +0000717 dir = tempfile.mkdtemp()
718 try:
719 f = tempfile.NamedTemporaryFile(dir=dir)
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000720 f.write(b'blat')
Guido van Rossum0e548712002-08-09 16:14:33 +0000721 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000722 self.assertFalse(os.path.exists(f.name),
Guido van Rossum0e548712002-08-09 16:14:33 +0000723 "NamedTemporaryFile %s exists after close" % f.name)
724 finally:
725 os.rmdir(dir)
726
Guido van Rossumd8faa362007-04-27 19:54:29 +0000727 def test_dis_del_on_close(self):
728 # Tests that delete-on-close can be disabled
729 dir = tempfile.mkdtemp()
730 tmp = None
731 try:
732 f = tempfile.NamedTemporaryFile(dir=dir, delete=False)
733 tmp = f.name
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000734 f.write(b'blat')
Guido van Rossumd8faa362007-04-27 19:54:29 +0000735 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000736 self.assertTrue(os.path.exists(f.name),
Guido van Rossumd8faa362007-04-27 19:54:29 +0000737 "NamedTemporaryFile %s missing after close" % f.name)
738 finally:
739 if tmp is not None:
740 os.unlink(tmp)
741 os.rmdir(dir)
742
Guido van Rossum0e548712002-08-09 16:14:33 +0000743 def test_multiple_close(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000744 # A NamedTemporaryFile can be closed many times without error
Guido van Rossum0e548712002-08-09 16:14:33 +0000745 f = tempfile.NamedTemporaryFile()
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000746 f.write(b'abc\n')
Guido van Rossum0e548712002-08-09 16:14:33 +0000747 f.close()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100748 f.close()
749 f.close()
Guido van Rossum0e548712002-08-09 16:14:33 +0000750
Christian Heimes3ecfea712008-02-09 20:51:34 +0000751 def test_context_manager(self):
752 # A NamedTemporaryFile can be used as a context manager
753 with tempfile.NamedTemporaryFile() as f:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000754 self.assertTrue(os.path.exists(f.name))
755 self.assertFalse(os.path.exists(f.name))
Christian Heimes3ecfea712008-02-09 20:51:34 +0000756 def use_closed():
757 with f:
758 pass
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000759 self.assertRaises(ValueError, use_closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000760
Guido van Rossum0e548712002-08-09 16:14:33 +0000761 # How to test the mode and bufsize parameters?
762
Guido van Rossum0e548712002-08-09 16:14:33 +0000763
Antoine Pitroueab2a502012-03-10 16:34:40 +0100764class TestSpooledTemporaryFile(BaseTestCase):
Guido van Rossumd8faa362007-04-27 19:54:29 +0000765 """Test SpooledTemporaryFile()."""
766
767 def do_create(self, max_size=0, dir=None, pre="", suf=""):
768 if dir is None:
769 dir = tempfile.gettempdir()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100770 file = tempfile.SpooledTemporaryFile(max_size=max_size, dir=dir, prefix=pre, suffix=suf)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000771
772 return file
773
774
775 def test_basic(self):
776 # SpooledTemporaryFile can create files
777 f = self.do_create()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000778 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000779 f = self.do_create(max_size=100, pre="a", suf=".txt")
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000780 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000781
782 def test_del_on_close(self):
783 # A SpooledTemporaryFile is deleted when closed
784 dir = tempfile.mkdtemp()
785 try:
786 f = tempfile.SpooledTemporaryFile(max_size=10, dir=dir)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000787 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +0000788 f.write(b'blat ' * 5)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000789 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000790 filename = f.name
791 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000792 self.assertFalse(isinstance(filename, str) and os.path.exists(filename),
Guido van Rossumd8faa362007-04-27 19:54:29 +0000793 "SpooledTemporaryFile %s exists after close" % filename)
794 finally:
795 os.rmdir(dir)
796
797 def test_rewrite_small(self):
798 # A SpooledTemporaryFile can be written to multiple within the max_size
799 f = self.do_create(max_size=30)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000800 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000801 for i in range(5):
802 f.seek(0, 0)
Guido van Rossum39478e82007-08-27 17:23:59 +0000803 f.write(b'x' * 20)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000804 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000805
806 def test_write_sequential(self):
807 # A SpooledTemporaryFile should hold exactly max_size bytes, and roll
808 # over afterward
809 f = self.do_create(max_size=30)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000810 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +0000811 f.write(b'x' * 20)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000812 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +0000813 f.write(b'x' * 10)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000814 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +0000815 f.write(b'x')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000816 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000817
R David Murrayd89ee792011-03-14 09:55:46 -0400818 def test_writelines(self):
819 # Verify writelines with a SpooledTemporaryFile
820 f = self.do_create()
821 f.writelines((b'x', b'y', b'z'))
822 f.seek(0)
823 buf = f.read()
824 self.assertEqual(buf, b'xyz')
825
826 def test_writelines_sequential(self):
827 # A SpooledTemporaryFile should hold exactly max_size bytes, and roll
828 # over afterward
829 f = self.do_create(max_size=35)
830 f.writelines((b'x' * 20, b'x' * 10, b'x' * 5))
831 self.assertFalse(f._rolled)
832 f.write(b'x')
833 self.assertTrue(f._rolled)
834
Guido van Rossumd8faa362007-04-27 19:54:29 +0000835 def test_sparse(self):
836 # A SpooledTemporaryFile that is written late in the file will extend
837 # when that occurs
838 f = self.do_create(max_size=30)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000839 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000840 f.seek(100, 0)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000841 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +0000842 f.write(b'x')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000843 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000844
845 def test_fileno(self):
846 # A SpooledTemporaryFile should roll over to a real file on fileno()
847 f = self.do_create(max_size=30)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000848 self.assertFalse(f._rolled)
849 self.assertTrue(f.fileno() > 0)
850 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000851
Christian Heimes3ecfea712008-02-09 20:51:34 +0000852 def test_multiple_close_before_rollover(self):
Guido van Rossumd8faa362007-04-27 19:54:29 +0000853 # A SpooledTemporaryFile can be closed many times without error
854 f = tempfile.SpooledTemporaryFile()
Guido van Rossum39478e82007-08-27 17:23:59 +0000855 f.write(b'abc\n')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000856 self.assertFalse(f._rolled)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000857 f.close()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100858 f.close()
859 f.close()
Christian Heimes3ecfea712008-02-09 20:51:34 +0000860
861 def test_multiple_close_after_rollover(self):
862 # A SpooledTemporaryFile can be closed many times without error
863 f = tempfile.SpooledTemporaryFile(max_size=1)
864 f.write(b'abc\n')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000865 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000866 f.close()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100867 f.close()
868 f.close()
Guido van Rossumd8faa362007-04-27 19:54:29 +0000869
870 def test_bound_methods(self):
871 # It should be OK to steal a bound method from a SpooledTemporaryFile
872 # and use it independently; when the file rolls over, those bound
873 # methods should continue to function
874 f = self.do_create(max_size=30)
875 read = f.read
876 write = f.write
877 seek = f.seek
878
Guido van Rossum39478e82007-08-27 17:23:59 +0000879 write(b"a" * 35)
880 write(b"b" * 35)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000881 seek(0, 0)
Guido van Rossum9a634702007-07-09 10:24:45 +0000882 self.assertEqual(read(70), b'a'*35 + b'b'*35)
883
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +0200884 def test_properties(self):
885 f = tempfile.SpooledTemporaryFile(max_size=10)
886 f.write(b'x' * 10)
887 self.assertFalse(f._rolled)
888 self.assertEqual(f.mode, 'w+b')
889 self.assertIsNone(f.name)
890 with self.assertRaises(AttributeError):
891 f.newlines
892 with self.assertRaises(AttributeError):
893 f.encoding
894
895 f.write(b'x')
896 self.assertTrue(f._rolled)
897 self.assertEqual(f.mode, 'rb+')
898 self.assertIsNotNone(f.name)
899 with self.assertRaises(AttributeError):
900 f.newlines
901 with self.assertRaises(AttributeError):
902 f.encoding
903
Guido van Rossum9a634702007-07-09 10:24:45 +0000904 def test_text_mode(self):
905 # Creating a SpooledTemporaryFile with a text mode should produce
906 # a file object reading and writing (Unicode) text strings.
907 f = tempfile.SpooledTemporaryFile(mode='w+', max_size=10)
908 f.write("abc\n")
909 f.seek(0)
910 self.assertEqual(f.read(), "abc\n")
911 f.write("def\n")
912 f.seek(0)
913 self.assertEqual(f.read(), "abc\ndef\n")
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +0200914 self.assertFalse(f._rolled)
915 self.assertEqual(f.mode, 'w+')
916 self.assertIsNone(f.name)
917 self.assertIsNone(f.newlines)
918 self.assertIsNone(f.encoding)
919
Guido van Rossum9a634702007-07-09 10:24:45 +0000920 f.write("xyzzy\n")
921 f.seek(0)
922 self.assertEqual(f.read(), "abc\ndef\nxyzzy\n")
Amaury Forgeot d'Arc7d0bddd2009-11-30 00:08:56 +0000923 # Check that Ctrl+Z doesn't truncate the file
924 f.write("foo\x1abar\n")
925 f.seek(0)
926 self.assertEqual(f.read(), "abc\ndef\nxyzzy\nfoo\x1abar\n")
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +0200927 self.assertTrue(f._rolled)
928 self.assertEqual(f.mode, 'w+')
929 self.assertIsNotNone(f.name)
Serhiy Storchaka497cee42013-02-10 14:43:46 +0200930 self.assertEqual(f.newlines, os.linesep)
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +0200931 self.assertIsNotNone(f.encoding)
Guido van Rossum9a634702007-07-09 10:24:45 +0000932
Guido van Rossumf0c74162007-08-28 03:29:45 +0000933 def test_text_newline_and_encoding(self):
934 f = tempfile.SpooledTemporaryFile(mode='w+', max_size=10,
935 newline='', encoding='utf-8')
936 f.write("\u039B\r\n")
937 f.seek(0)
938 self.assertEqual(f.read(), "\u039B\r\n")
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000939 self.assertFalse(f._rolled)
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +0200940 self.assertEqual(f.mode, 'w+')
941 self.assertIsNone(f.name)
942 self.assertIsNone(f.newlines)
943 self.assertIsNone(f.encoding)
Guido van Rossumf0c74162007-08-28 03:29:45 +0000944
945 f.write("\u039B" * 20 + "\r\n")
946 f.seek(0)
947 self.assertEqual(f.read(), "\u039B\r\n" + ("\u039B" * 20) + "\r\n")
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000948 self.assertTrue(f._rolled)
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +0200949 self.assertEqual(f.mode, 'w+')
950 self.assertIsNotNone(f.name)
951 self.assertIsNotNone(f.newlines)
952 self.assertEqual(f.encoding, 'utf-8')
Guido van Rossumd8faa362007-04-27 19:54:29 +0000953
Christian Heimes3ecfea712008-02-09 20:51:34 +0000954 def test_context_manager_before_rollover(self):
955 # A SpooledTemporaryFile can be used as a context manager
956 with tempfile.SpooledTemporaryFile(max_size=1) as f:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000957 self.assertFalse(f._rolled)
958 self.assertFalse(f.closed)
959 self.assertTrue(f.closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000960 def use_closed():
961 with f:
962 pass
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000963 self.assertRaises(ValueError, use_closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000964
965 def test_context_manager_during_rollover(self):
966 # A SpooledTemporaryFile can be used as a context manager
967 with tempfile.SpooledTemporaryFile(max_size=1) as f:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000968 self.assertFalse(f._rolled)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000969 f.write(b'abc\n')
970 f.flush()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000971 self.assertTrue(f._rolled)
972 self.assertFalse(f.closed)
973 self.assertTrue(f.closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000974 def use_closed():
975 with f:
976 pass
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000977 self.assertRaises(ValueError, use_closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000978
979 def test_context_manager_after_rollover(self):
980 # A SpooledTemporaryFile can be used as a context manager
981 f = tempfile.SpooledTemporaryFile(max_size=1)
982 f.write(b'abc\n')
983 f.flush()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000984 self.assertTrue(f._rolled)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000985 with f:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000986 self.assertFalse(f.closed)
987 self.assertTrue(f.closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000988 def use_closed():
989 with f:
990 pass
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000991 self.assertRaises(ValueError, use_closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000992
Antoine Pitrou0e86a582011-11-25 18:03:09 +0100993 def test_truncate_with_size_parameter(self):
994 # A SpooledTemporaryFile can be truncated to zero size
995 f = tempfile.SpooledTemporaryFile(max_size=10)
996 f.write(b'abcdefg\n')
997 f.seek(0)
998 f.truncate()
999 self.assertFalse(f._rolled)
1000 self.assertEqual(f._file.getvalue(), b'')
1001 # A SpooledTemporaryFile can be truncated to a specific size
1002 f = tempfile.SpooledTemporaryFile(max_size=10)
1003 f.write(b'abcdefg\n')
1004 f.truncate(4)
1005 self.assertFalse(f._rolled)
1006 self.assertEqual(f._file.getvalue(), b'abcd')
1007 # A SpooledTemporaryFile rolls over if truncated to large size
1008 f = tempfile.SpooledTemporaryFile(max_size=10)
1009 f.write(b'abcdefg\n')
1010 f.truncate(20)
1011 self.assertTrue(f._rolled)
1012 if has_stat:
1013 self.assertEqual(os.fstat(f.fileno()).st_size, 20)
Christian Heimes3ecfea712008-02-09 20:51:34 +00001014
Guido van Rossum0e548712002-08-09 16:14:33 +00001015
Guido van Rossum8ccd9b62002-08-22 20:02:03 +00001016if tempfile.NamedTemporaryFile is not tempfile.TemporaryFile:
Antoine Pitroueab2a502012-03-10 16:34:40 +01001017
1018 class TestTemporaryFile(BaseTestCase):
1019 """Test TemporaryFile()."""
1020
1021 def test_basic(self):
1022 # TemporaryFile can create files
1023 # No point in testing the name params - the file has no name.
1024 tempfile.TemporaryFile()
1025
1026 def test_has_no_name(self):
1027 # TemporaryFile creates files with no names (on this system)
1028 dir = tempfile.mkdtemp()
1029 f = tempfile.TemporaryFile(dir=dir)
1030 f.write(b'blat')
1031
1032 # Sneaky: because this file has no name, it should not prevent
1033 # us from removing the directory it was created in.
1034 try:
1035 os.rmdir(dir)
1036 except:
1037 # cleanup
1038 f.close()
1039 os.rmdir(dir)
1040 raise
1041
1042 def test_multiple_close(self):
1043 # A TemporaryFile can be closed many times without error
1044 f = tempfile.TemporaryFile()
1045 f.write(b'abc\n')
1046 f.close()
1047 f.close()
1048 f.close()
1049
1050 # How to test the mode and bufsize parameters?
1051 def test_mode_and_encoding(self):
1052
1053 def roundtrip(input, *args, **kwargs):
1054 with tempfile.TemporaryFile(*args, **kwargs) as fileobj:
1055 fileobj.write(input)
1056 fileobj.seek(0)
1057 self.assertEqual(input, fileobj.read())
1058
1059 roundtrip(b"1234", "w+b")
1060 roundtrip("abdc\n", "w+")
1061 roundtrip("\u039B", "w+", encoding="utf-16")
1062 roundtrip("foo\r\n", "w+", newline="")
Guido van Rossum0e548712002-08-09 16:14:33 +00001063
Nick Coghlan543af752010-10-24 11:23:25 +00001064
1065# Helper for test_del_on_shutdown
1066class NulledModules:
1067 def __init__(self, *modules):
1068 self.refs = [mod.__dict__ for mod in modules]
1069 self.contents = [ref.copy() for ref in self.refs]
1070
1071 def __enter__(self):
1072 for d in self.refs:
1073 for key in d:
1074 d[key] = None
1075
1076 def __exit__(self, *exc_info):
1077 for d, c in zip(self.refs, self.contents):
1078 d.clear()
1079 d.update(c)
1080
Antoine Pitroueab2a502012-03-10 16:34:40 +01001081class TestTemporaryDirectory(BaseTestCase):
Nick Coghlan543af752010-10-24 11:23:25 +00001082 """Test TemporaryDirectory()."""
1083
1084 def do_create(self, dir=None, pre="", suf="", recurse=1):
1085 if dir is None:
1086 dir = tempfile.gettempdir()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +01001087 tmp = tempfile.TemporaryDirectory(dir=dir, prefix=pre, suffix=suf)
Nick Coghlan543af752010-10-24 11:23:25 +00001088 self.nameCheck(tmp.name, dir, pre, suf)
1089 # Create a subdirectory and some files
1090 if recurse:
Serhiy Storchaka99e033b2014-01-27 11:18:27 +02001091 d1 = self.do_create(tmp.name, pre, suf, recurse-1)
1092 d1.name = None
Nick Coghlan543af752010-10-24 11:23:25 +00001093 with open(os.path.join(tmp.name, "test.txt"), "wb") as f:
1094 f.write(b"Hello world!")
1095 return tmp
1096
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001097 def test_mkdtemp_failure(self):
1098 # Check no additional exception if mkdtemp fails
1099 # Previously would raise AttributeError instead
Nick Coghlan3c54ea62010-12-13 03:02:43 +00001100 # (noted as part of Issue #10188)
1101 with tempfile.TemporaryDirectory() as nonexistent:
1102 pass
Serhiy Storchaka7451a722013-02-09 22:25:49 +02001103 with self.assertRaises(FileNotFoundError) as cm:
Nick Coghlan3c54ea62010-12-13 03:02:43 +00001104 tempfile.TemporaryDirectory(dir=nonexistent)
Serhiy Storchaka7451a722013-02-09 22:25:49 +02001105 self.assertEqual(cm.exception.errno, errno.ENOENT)
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001106
Nick Coghlan543af752010-10-24 11:23:25 +00001107 def test_explicit_cleanup(self):
1108 # A TemporaryDirectory is deleted when cleaned up
1109 dir = tempfile.mkdtemp()
1110 try:
1111 d = self.do_create(dir=dir)
1112 self.assertTrue(os.path.exists(d.name),
1113 "TemporaryDirectory %s does not exist" % d.name)
1114 d.cleanup()
1115 self.assertFalse(os.path.exists(d.name),
1116 "TemporaryDirectory %s exists after cleanup" % d.name)
1117 finally:
1118 os.rmdir(dir)
1119
Charles-François Natalidef35432011-07-29 18:59:24 +02001120 @support.skip_unless_symlink
1121 def test_cleanup_with_symlink_to_a_directory(self):
1122 # cleanup() should not follow symlinks to directories (issue #12464)
1123 d1 = self.do_create()
Serhiy Storchaka99e033b2014-01-27 11:18:27 +02001124 d2 = self.do_create(recurse=0)
Charles-François Natalidef35432011-07-29 18:59:24 +02001125
1126 # Symlink d1/foo -> d2
1127 os.symlink(d2.name, os.path.join(d1.name, "foo"))
1128
1129 # This call to cleanup() should not follow the "foo" symlink
1130 d1.cleanup()
1131
1132 self.assertFalse(os.path.exists(d1.name),
1133 "TemporaryDirectory %s exists after cleanup" % d1.name)
1134 self.assertTrue(os.path.exists(d2.name),
1135 "Directory pointed to by a symlink was deleted")
1136 self.assertEqual(os.listdir(d2.name), ['test.txt'],
1137 "Contents of the directory pointed to by a symlink "
1138 "were deleted")
1139 d2.cleanup()
1140
Nick Coghlan543af752010-10-24 11:23:25 +00001141 @support.cpython_only
1142 def test_del_on_collection(self):
1143 # A TemporaryDirectory is deleted when garbage collected
1144 dir = tempfile.mkdtemp()
1145 try:
1146 d = self.do_create(dir=dir)
1147 name = d.name
1148 del d # Rely on refcounting to invoke __del__
1149 self.assertFalse(os.path.exists(name),
1150 "TemporaryDirectory %s exists after __del__" % name)
1151 finally:
1152 os.rmdir(dir)
1153
Nick Coghlan543af752010-10-24 11:23:25 +00001154 def test_del_on_shutdown(self):
1155 # A TemporaryDirectory may be cleaned up during shutdown
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001156 with self.do_create() as dir:
Serhiy Storchakaa28632b2014-01-27 11:21:54 +02001157 for mod in ('builtins', 'os', 'shutil', 'sys', 'tempfile', 'warnings'):
Serhiy Storchaka99e033b2014-01-27 11:18:27 +02001158 code = """if True:
Serhiy Storchakaa28632b2014-01-27 11:21:54 +02001159 import builtins
Serhiy Storchaka99e033b2014-01-27 11:18:27 +02001160 import os
1161 import shutil
1162 import sys
1163 import tempfile
1164 import warnings
1165
1166 tmp = tempfile.TemporaryDirectory(dir={dir!r})
1167 sys.stdout.buffer.write(tmp.name.encode())
1168
1169 tmp2 = os.path.join(tmp.name, 'test_dir')
1170 os.mkdir(tmp2)
1171 with open(os.path.join(tmp2, "test.txt"), "w") as f:
1172 f.write("Hello world!")
1173
1174 {mod}.tmp = tmp
1175
1176 warnings.filterwarnings("always", category=ResourceWarning)
1177 """.format(dir=dir, mod=mod)
1178 rc, out, err = script_helper.assert_python_ok("-c", code)
1179 tmp_name = out.decode().strip()
1180 self.assertFalse(os.path.exists(tmp_name),
1181 "TemporaryDirectory %s exists after cleanup" % tmp_name)
1182 err = err.decode('utf-8', 'backslashreplace')
1183 self.assertNotIn("Exception ", err)
Serhiy Storchakaa28632b2014-01-27 11:21:54 +02001184 self.assertIn("ResourceWarning: Implicitly cleaning up", err)
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001185
1186 def test_warnings_on_cleanup(self):
Serhiy Storchaka99e033b2014-01-27 11:18:27 +02001187 # ResourceWarning will be triggered by __del__
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001188 with self.do_create() as dir:
Serhiy Storchaka99e033b2014-01-27 11:18:27 +02001189 d = self.do_create(dir=dir, recurse=3)
1190 name = d.name
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001191
1192 # Check for the resource warning
1193 with support.check_warnings(('Implicitly', ResourceWarning), quiet=False):
1194 warnings.filterwarnings("always", category=ResourceWarning)
Serhiy Storchaka99e033b2014-01-27 11:18:27 +02001195 del d
1196 support.gc_collect()
1197 self.assertFalse(os.path.exists(name),
1198 "TemporaryDirectory %s exists after __del__" % name)
Nick Coghlan543af752010-10-24 11:23:25 +00001199
1200 def test_multiple_close(self):
1201 # Can be cleaned-up many times without error
1202 d = self.do_create()
1203 d.cleanup()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +01001204 d.cleanup()
1205 d.cleanup()
Nick Coghlan543af752010-10-24 11:23:25 +00001206
1207 def test_context_manager(self):
1208 # Can be used as a context manager
1209 d = self.do_create()
1210 with d as name:
1211 self.assertTrue(os.path.exists(name))
1212 self.assertEqual(name, d.name)
1213 self.assertFalse(os.path.exists(name))
1214
1215
Guido van Rossum0e548712002-08-09 16:14:33 +00001216def test_main():
Antoine Pitroueab2a502012-03-10 16:34:40 +01001217 support.run_unittest(__name__)
Guido van Rossum0e548712002-08-09 16:14:33 +00001218
1219if __name__ == "__main__":
1220 test_main()