blob: 7540d73b1abe647dfaf2f63f83de960e82a35fe0 [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
Guido van Rossum0e548712002-08-09 16:14:33 +00003import os
Antoine Pitrou4558bad2011-11-25 21:28:15 +01004import signal
Guido van Rossum0e548712002-08-09 16:14:33 +00005import sys
6import re
Guido van Rossum0e548712002-08-09 16:14:33 +00007import warnings
Tim Petersc57a2852001-10-29 21:46:08 +00008
Guido van Rossum0e548712002-08-09 16:14:33 +00009import unittest
Benjamin Petersonee8712c2008-05-20 21:35:26 +000010from test import support
Guido van Rossum0e548712002-08-09 16:14:33 +000011
Fred Drake7633d232002-10-17 22:09:03 +000012
Guido van Rossum0e548712002-08-09 16:14:33 +000013if hasattr(os, 'stat'):
14 import stat
15 has_stat = 1
16else:
17 has_stat = 0
18
19has_textmode = (tempfile._text_openflags != tempfile._bin_openflags)
Guido van Rossum78741062002-08-17 11:41:01 +000020has_spawnl = hasattr(os, 'spawnl')
Guido van Rossum0e548712002-08-09 16:14:33 +000021
Neal Norwitz68ee0122002-08-16 19:28:59 +000022# TEST_FILES may need to be tweaked for systems depending on the maximum
23# number of files that can be opened at one time (see ulimit -n)
Victor Stinner9c3de4a2011-08-17 20:49:41 +020024if sys.platform.startswith('openbsd'):
Martin v. Löwis99968282004-09-15 06:02:54 +000025 TEST_FILES = 48
Jack Jansence921472003-01-08 16:30:34 +000026else:
27 TEST_FILES = 100
Neal Norwitz68ee0122002-08-16 19:28:59 +000028
Guido van Rossum0e548712002-08-09 16:14:33 +000029# This is organized as one test for each chunk of code in tempfile.py,
30# in order of their appearance in the file. Testing which requires
31# threads is not done here.
32
33# Common functionality.
34class TC(unittest.TestCase):
35
36 str_check = re.compile(r"[a-zA-Z0-9_-]{6}$")
37
Brett Cannone1adece2010-03-20 22:19:55 +000038 def setUp(self):
39 self._warnings_manager = support.check_warnings()
40 self._warnings_manager.__enter__()
41 warnings.filterwarnings("ignore", category=RuntimeWarning,
42 message="mktemp", module=__name__)
43
44 def tearDown(self):
45 self._warnings_manager.__exit__(None, None, None)
46
47
Guido van Rossum0e548712002-08-09 16:14:33 +000048 def nameCheck(self, name, dir, pre, suf):
49 (ndir, nbase) = os.path.split(name)
50 npre = nbase[:len(pre)]
51 nsuf = nbase[len(nbase)-len(suf):]
52
Martin v. Löwisd6625482003-10-12 17:37:01 +000053 # check for equality of the absolute paths!
54 self.assertEqual(os.path.abspath(ndir), os.path.abspath(dir),
Guido van Rossum0e548712002-08-09 16:14:33 +000055 "file '%s' not in directory '%s'" % (name, dir))
56 self.assertEqual(npre, pre,
57 "file '%s' does not begin with '%s'" % (nbase, pre))
58 self.assertEqual(nsuf, suf,
59 "file '%s' does not end with '%s'" % (nbase, suf))
60
61 nbase = nbase[len(pre):len(nbase)-len(suf)]
Benjamin Petersonc9c0f202009-06-30 23:06:06 +000062 self.assertTrue(self.str_check.match(nbase),
Guido van Rossum0e548712002-08-09 16:14:33 +000063 "random string '%s' does not match /^[a-zA-Z0-9_-]{6}$/"
64 % nbase)
65
66test_classes = []
67
68class test_exports(TC):
69 def test_exports(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +000070 # There are no surprising symbols in the tempfile module
Guido van Rossum0e548712002-08-09 16:14:33 +000071 dict = tempfile.__dict__
72
73 expected = {
74 "NamedTemporaryFile" : 1,
75 "TemporaryFile" : 1,
76 "mkstemp" : 1,
77 "mkdtemp" : 1,
78 "mktemp" : 1,
79 "TMP_MAX" : 1,
80 "gettempprefix" : 1,
81 "gettempdir" : 1,
82 "tempdir" : 1,
Guido van Rossumd8faa362007-04-27 19:54:29 +000083 "template" : 1,
Nick Coghlan543af752010-10-24 11:23:25 +000084 "SpooledTemporaryFile" : 1,
85 "TemporaryDirectory" : 1,
Guido van Rossum0e548712002-08-09 16:14:33 +000086 }
87
88 unexp = []
89 for key in dict:
90 if key[0] != '_' and key not in expected:
91 unexp.append(key)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +000092 self.assertTrue(len(unexp) == 0,
Guido van Rossum0e548712002-08-09 16:14:33 +000093 "unexpected keys: %s" % unexp)
94
95test_classes.append(test_exports)
96
97
Guido van Rossum0e548712002-08-09 16:14:33 +000098class test__RandomNameSequence(TC):
99 """Test the internal iterator object _RandomNameSequence."""
100
101 def setUp(self):
102 self.r = tempfile._RandomNameSequence()
Brett Cannone1adece2010-03-20 22:19:55 +0000103 super().setUp()
Guido van Rossum0e548712002-08-09 16:14:33 +0000104
105 def test_get_six_char_str(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000106 # _RandomNameSequence returns a six-character string
Georg Brandla18af4e2007-04-21 15:47:16 +0000107 s = next(self.r)
Guido van Rossum0e548712002-08-09 16:14:33 +0000108 self.nameCheck(s, '', '', '')
109
110 def test_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000111 # _RandomNameSequence returns no duplicate strings (stochastic)
Guido van Rossum0e548712002-08-09 16:14:33 +0000112
113 dict = {}
114 r = self.r
Guido van Rossum805365e2007-05-07 22:24:25 +0000115 for i in range(TEST_FILES):
Georg Brandla18af4e2007-04-21 15:47:16 +0000116 s = next(r)
Guido van Rossum0e548712002-08-09 16:14:33 +0000117 self.nameCheck(s, '', '', '')
Ezio Melottib58e0bd2010-01-23 15:40:09 +0000118 self.assertNotIn(s, dict)
Guido van Rossum0e548712002-08-09 16:14:33 +0000119 dict[s] = 1
120
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000121 def supports_iter(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000122 # _RandomNameSequence supports the iterator protocol
Guido van Rossum0e548712002-08-09 16:14:33 +0000123
124 i = 0
125 r = self.r
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100126 for s in r:
127 i += 1
128 if i == 20:
129 break
Guido van Rossum0e548712002-08-09 16:14:33 +0000130
Antoine Pitrou4558bad2011-11-25 21:28:15 +0100131 @unittest.skipUnless(hasattr(os, 'fork'),
132 "os.fork is required for this test")
133 def test_process_awareness(self):
134 # ensure that the random source differs between
135 # child and parent.
136 read_fd, write_fd = os.pipe()
137 pid = None
138 try:
139 pid = os.fork()
140 if not pid:
141 os.close(read_fd)
142 os.write(write_fd, next(self.r).encode("ascii"))
143 os.close(write_fd)
144 # bypass the normal exit handlers- leave those to
145 # the parent.
146 os._exit(0)
147 parent_value = next(self.r)
148 child_value = os.read(read_fd, len(parent_value)).decode("ascii")
149 finally:
150 if pid:
151 # best effort to ensure the process can't bleed out
152 # via any bugs above
153 try:
154 os.kill(pid, signal.SIGKILL)
155 except EnvironmentError:
156 pass
157 os.close(read_fd)
158 os.close(write_fd)
159 self.assertNotEqual(child_value, parent_value)
160
161
Guido van Rossum0e548712002-08-09 16:14:33 +0000162test_classes.append(test__RandomNameSequence)
163
164
165class test__candidate_tempdir_list(TC):
166 """Test the internal function _candidate_tempdir_list."""
167
168 def test_nonempty_list(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000169 # _candidate_tempdir_list returns a nonempty list of strings
Guido van Rossum0e548712002-08-09 16:14:33 +0000170
171 cand = tempfile._candidate_tempdir_list()
172
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000173 self.assertFalse(len(cand) == 0)
Guido van Rossum0e548712002-08-09 16:14:33 +0000174 for c in cand:
Ezio Melottie9615932010-01-24 19:26:24 +0000175 self.assertIsInstance(c, str)
Guido van Rossum0e548712002-08-09 16:14:33 +0000176
177 def test_wanted_dirs(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000178 # _candidate_tempdir_list contains the expected directories
Guido van Rossum0e548712002-08-09 16:14:33 +0000179
180 # Make sure the interesting environment variables are all set.
Hirokazu Yamamoto71959632009-04-27 01:44:28 +0000181 with support.EnvironmentVarGuard() as env:
Guido van Rossum0e548712002-08-09 16:14:33 +0000182 for envname in 'TMPDIR', 'TEMP', 'TMP':
183 dirname = os.getenv(envname)
184 if not dirname:
Walter Dörwald155374d2009-05-01 19:58:58 +0000185 env[envname] = os.path.abspath(envname)
Guido van Rossum0e548712002-08-09 16:14:33 +0000186
187 cand = tempfile._candidate_tempdir_list()
188
189 for envname in 'TMPDIR', 'TEMP', 'TMP':
190 dirname = os.getenv(envname)
191 if not dirname: raise ValueError
Benjamin Peterson577473f2010-01-19 00:09:57 +0000192 self.assertIn(dirname, cand)
Guido van Rossum0e548712002-08-09 16:14:33 +0000193
194 try:
195 dirname = os.getcwd()
196 except (AttributeError, os.error):
197 dirname = os.curdir
198
Benjamin Peterson577473f2010-01-19 00:09:57 +0000199 self.assertIn(dirname, cand)
Guido van Rossum0e548712002-08-09 16:14:33 +0000200
201 # Not practical to try to verify the presence of OS-specific
202 # paths in this list.
Guido van Rossum0e548712002-08-09 16:14:33 +0000203
204test_classes.append(test__candidate_tempdir_list)
205
206
207# We test _get_default_tempdir by testing gettempdir.
208
209
210class test__get_candidate_names(TC):
211 """Test the internal function _get_candidate_names."""
212
213 def test_retval(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000214 # _get_candidate_names returns a _RandomNameSequence object
Guido van Rossum0e548712002-08-09 16:14:33 +0000215 obj = tempfile._get_candidate_names()
Ezio Melottie9615932010-01-24 19:26:24 +0000216 self.assertIsInstance(obj, tempfile._RandomNameSequence)
Guido van Rossum0e548712002-08-09 16:14:33 +0000217
218 def test_same_thing(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000219 # _get_candidate_names always returns the same object
Guido van Rossum0e548712002-08-09 16:14:33 +0000220 a = tempfile._get_candidate_names()
221 b = tempfile._get_candidate_names()
222
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000223 self.assertTrue(a is b)
Guido van Rossum0e548712002-08-09 16:14:33 +0000224
225test_classes.append(test__get_candidate_names)
226
227
228class test__mkstemp_inner(TC):
229 """Test the internal function _mkstemp_inner."""
230
231 class mkstemped:
232 _bflags = tempfile._bin_openflags
233 _tflags = tempfile._text_openflags
234 _close = os.close
235 _unlink = os.unlink
236
237 def __init__(self, dir, pre, suf, bin):
238 if bin: flags = self._bflags
239 else: flags = self._tflags
240
241 (self.fd, self.name) = tempfile._mkstemp_inner(dir, pre, suf, flags)
242
243 def write(self, str):
244 os.write(self.fd, str)
245
246 def __del__(self):
247 self._close(self.fd)
248 self._unlink(self.name)
Tim Petersa0d55de2002-08-09 18:01:01 +0000249
Guido van Rossum0e548712002-08-09 16:14:33 +0000250 def do_create(self, dir=None, pre="", suf="", bin=1):
251 if dir is None:
252 dir = tempfile.gettempdir()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100253 file = self.mkstemped(dir, pre, suf, bin)
Guido van Rossum0e548712002-08-09 16:14:33 +0000254
255 self.nameCheck(file.name, dir, pre, suf)
256 return file
257
258 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000259 # _mkstemp_inner can create files
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000260 self.do_create().write(b"blat")
261 self.do_create(pre="a").write(b"blat")
262 self.do_create(suf="b").write(b"blat")
263 self.do_create(pre="a", suf="b").write(b"blat")
264 self.do_create(pre="aa", suf=".txt").write(b"blat")
Guido van Rossum0e548712002-08-09 16:14:33 +0000265
266 def test_basic_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000267 # _mkstemp_inner can create many files (stochastic)
Guido van Rossum805365e2007-05-07 22:24:25 +0000268 extant = list(range(TEST_FILES))
Guido van Rossum0e548712002-08-09 16:14:33 +0000269 for i in extant:
270 extant[i] = self.do_create(pre="aa")
271
272 def test_choose_directory(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000273 # _mkstemp_inner can create files in a user-selected directory
Guido van Rossum0e548712002-08-09 16:14:33 +0000274 dir = tempfile.mkdtemp()
275 try:
Antoine Pitrou9cadb1b2008-09-15 23:02:56 +0000276 self.do_create(dir=dir).write(b"blat")
Guido van Rossum0e548712002-08-09 16:14:33 +0000277 finally:
278 os.rmdir(dir)
279
280 def test_file_mode(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000281 # _mkstemp_inner creates files with the proper mode
Guido van Rossum0e548712002-08-09 16:14:33 +0000282 if not has_stat:
Benjamin Petersone549ead2009-03-28 21:42:05 +0000283 return # ugh, can't use SkipTest.
Guido van Rossum0e548712002-08-09 16:14:33 +0000284
285 file = self.do_create()
286 mode = stat.S_IMODE(os.stat(file.name).st_mode)
Guido van Rossumcd16bf62007-06-13 18:07:49 +0000287 expected = 0o600
Ronald Oussoren94f25282010-05-05 19:11:21 +0000288 if sys.platform in ('win32', 'os2emx'):
Tim Petersca3ac7f2002-08-09 18:13:51 +0000289 # There's no distinction among 'user', 'group' and 'world';
290 # replicate the 'user' bits.
291 user = expected >> 6
292 expected = user * (1 + 8 + 64)
293 self.assertEqual(mode, expected)
Guido van Rossum0e548712002-08-09 16:14:33 +0000294
295 def test_noinherit(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000296 # _mkstemp_inner file handles are not inherited by child processes
Guido van Rossum78741062002-08-17 11:41:01 +0000297 if not has_spawnl:
Benjamin Petersone549ead2009-03-28 21:42:05 +0000298 return # ugh, can't use SkipTest.
Guido van Rossum0e548712002-08-09 16:14:33 +0000299
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000300 if support.verbose:
Guido van Rossum78741062002-08-17 11:41:01 +0000301 v="v"
302 else:
303 v="q"
304
Guido van Rossum0e548712002-08-09 16:14:33 +0000305 file = self.do_create()
Guido van Rossum78741062002-08-17 11:41:01 +0000306 fd = "%d" % file.fd
307
308 try:
309 me = __file__
310 except NameError:
311 me = sys.argv[0]
Guido van Rossum0e548712002-08-09 16:14:33 +0000312
313 # We have to exec something, so that FD_CLOEXEC will take
Guido van Rossum78741062002-08-17 11:41:01 +0000314 # effect. The core of this test is therefore in
315 # tf_inherit_check.py, which see.
316 tester = os.path.join(os.path.dirname(os.path.abspath(me)),
317 "tf_inherit_check.py")
Guido van Rossum0e548712002-08-09 16:14:33 +0000318
Martin v. Löwisd4210bc2003-10-23 15:55:28 +0000319 # On Windows a spawn* /path/ with embedded spaces shouldn't be quoted,
320 # but an arg with embedded spaces should be decorated with double
321 # quotes on each end
Christian Heimes895627f2007-12-08 17:28:33 +0000322 if sys.platform in ('win32',):
Martin v. Löwisd4210bc2003-10-23 15:55:28 +0000323 decorated = '"%s"' % sys.executable
324 tester = '"%s"' % tester
325 else:
326 decorated = sys.executable
327
328 retval = os.spawnl(os.P_WAIT, sys.executable, decorated, tester, v, fd)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000329 self.assertFalse(retval < 0,
Guido van Rossum78741062002-08-17 11:41:01 +0000330 "child process caught fatal signal %d" % -retval)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000331 self.assertFalse(retval > 0, "child process reports failure %d"%retval)
Guido van Rossum0e548712002-08-09 16:14:33 +0000332
333 def test_textmode(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000334 # _mkstemp_inner can create files in text mode
Guido van Rossum0e548712002-08-09 16:14:33 +0000335 if not has_textmode:
Benjamin Petersone549ead2009-03-28 21:42:05 +0000336 return # ugh, can't use SkipTest.
Guido van Rossum0e548712002-08-09 16:14:33 +0000337
Amaury Forgeot d'Arc7d0bddd2009-11-30 00:08:56 +0000338 # A text file is truncated at the first Ctrl+Z byte
339 f = self.do_create(bin=0)
340 f.write(b"blat\x1a")
341 f.write(b"extra\n")
342 os.lseek(f.fd, 0, os.SEEK_SET)
Ezio Melottib3aedd42010-11-20 19:04:17 +0000343 self.assertEqual(os.read(f.fd, 20), b"blat")
Guido van Rossum0e548712002-08-09 16:14:33 +0000344
345test_classes.append(test__mkstemp_inner)
346
347
348class test_gettempprefix(TC):
349 """Test gettempprefix()."""
350
351 def test_sane_template(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000352 # gettempprefix returns a nonempty prefix string
Guido van Rossum0e548712002-08-09 16:14:33 +0000353 p = tempfile.gettempprefix()
354
Ezio Melottie9615932010-01-24 19:26:24 +0000355 self.assertIsInstance(p, str)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000356 self.assertTrue(len(p) > 0)
Guido van Rossum0e548712002-08-09 16:14:33 +0000357
358 def test_usable_template(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000359 # gettempprefix returns a usable prefix string
Guido van Rossum0e548712002-08-09 16:14:33 +0000360
361 # Create a temp directory, avoiding use of the prefix.
362 # Then attempt to create a file whose name is
363 # prefix + 'xxxxxx.xxx' in that directory.
364 p = tempfile.gettempprefix() + "xxxxxx.xxx"
365 d = tempfile.mkdtemp(prefix="")
366 try:
367 p = os.path.join(d, p)
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100368 fd = os.open(p, os.O_RDWR | os.O_CREAT)
Guido van Rossum0e548712002-08-09 16:14:33 +0000369 os.close(fd)
370 os.unlink(p)
371 finally:
372 os.rmdir(d)
373
374test_classes.append(test_gettempprefix)
375
376
377class test_gettempdir(TC):
378 """Test gettempdir()."""
379
380 def test_directory_exists(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000381 # gettempdir returns a directory which exists
Guido van Rossum0e548712002-08-09 16:14:33 +0000382
383 dir = tempfile.gettempdir()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000384 self.assertTrue(os.path.isabs(dir) or dir == os.curdir,
Guido van Rossum0e548712002-08-09 16:14:33 +0000385 "%s is not an absolute path" % dir)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000386 self.assertTrue(os.path.isdir(dir),
Guido van Rossum0e548712002-08-09 16:14:33 +0000387 "%s is not a directory" % dir)
388
389 def test_directory_writable(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000390 # gettempdir returns a directory writable by the user
Guido van Rossum0e548712002-08-09 16:14:33 +0000391
392 # sneaky: just instantiate a NamedTemporaryFile, which
393 # defaults to writing into the directory returned by
394 # gettempdir.
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100395 file = tempfile.NamedTemporaryFile()
396 file.write(b"blat")
397 file.close()
Guido van Rossum0e548712002-08-09 16:14:33 +0000398
399 def test_same_thing(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000400 # gettempdir always returns the same object
Guido van Rossum0e548712002-08-09 16:14:33 +0000401 a = tempfile.gettempdir()
402 b = tempfile.gettempdir()
403
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000404 self.assertTrue(a is b)
Guido van Rossum0e548712002-08-09 16:14:33 +0000405
406test_classes.append(test_gettempdir)
407
408
409class test_mkstemp(TC):
410 """Test mkstemp()."""
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000411
Thomas Wouters73e5a5b2006-06-08 15:35:45 +0000412 def do_create(self, dir=None, pre="", suf=""):
Guido van Rossum0e548712002-08-09 16:14:33 +0000413 if dir is None:
414 dir = tempfile.gettempdir()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100415 (fd, name) = tempfile.mkstemp(dir=dir, prefix=pre, suffix=suf)
416 (ndir, nbase) = os.path.split(name)
417 adir = os.path.abspath(dir)
418 self.assertEqual(adir, ndir,
419 "Directory '%s' incorrectly returned as '%s'" % (adir, ndir))
Guido van Rossum0e548712002-08-09 16:14:33 +0000420
421 try:
422 self.nameCheck(name, dir, pre, suf)
423 finally:
424 os.close(fd)
425 os.unlink(name)
426
427 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000428 # mkstemp can create files
Guido van Rossum0e548712002-08-09 16:14:33 +0000429 self.do_create()
430 self.do_create(pre="a")
431 self.do_create(suf="b")
432 self.do_create(pre="a", suf="b")
433 self.do_create(pre="aa", suf=".txt")
Martin v. Löwisd6625482003-10-12 17:37:01 +0000434 self.do_create(dir=".")
Guido van Rossum0e548712002-08-09 16:14:33 +0000435
436 def test_choose_directory(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000437 # mkstemp can create directories in a user-selected directory
Guido van Rossum0e548712002-08-09 16:14:33 +0000438 dir = tempfile.mkdtemp()
439 try:
440 self.do_create(dir=dir)
441 finally:
442 os.rmdir(dir)
443
444test_classes.append(test_mkstemp)
445
446
447class test_mkdtemp(TC):
448 """Test mkdtemp()."""
449
450 def do_create(self, dir=None, pre="", suf=""):
451 if dir is None:
452 dir = tempfile.gettempdir()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100453 name = tempfile.mkdtemp(dir=dir, prefix=pre, suffix=suf)
Guido van Rossum0e548712002-08-09 16:14:33 +0000454
455 try:
456 self.nameCheck(name, dir, pre, suf)
457 return name
458 except:
459 os.rmdir(name)
460 raise
461
462 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000463 # mkdtemp can create directories
Guido van Rossum0e548712002-08-09 16:14:33 +0000464 os.rmdir(self.do_create())
465 os.rmdir(self.do_create(pre="a"))
466 os.rmdir(self.do_create(suf="b"))
467 os.rmdir(self.do_create(pre="a", suf="b"))
468 os.rmdir(self.do_create(pre="aa", suf=".txt"))
Tim Petersa0d55de2002-08-09 18:01:01 +0000469
Guido van Rossum0e548712002-08-09 16:14:33 +0000470 def test_basic_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000471 # mkdtemp can create many directories (stochastic)
Guido van Rossum805365e2007-05-07 22:24:25 +0000472 extant = list(range(TEST_FILES))
Guido van Rossum0e548712002-08-09 16:14:33 +0000473 try:
474 for i in extant:
475 extant[i] = self.do_create(pre="aa")
476 finally:
477 for i in extant:
Guido van Rossum3172c5d2007-10-16 18:12:55 +0000478 if(isinstance(i, str)):
Guido van Rossum0e548712002-08-09 16:14:33 +0000479 os.rmdir(i)
480
481 def test_choose_directory(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000482 # mkdtemp can create directories in a user-selected directory
Guido van Rossum0e548712002-08-09 16:14:33 +0000483 dir = tempfile.mkdtemp()
484 try:
485 os.rmdir(self.do_create(dir=dir))
486 finally:
487 os.rmdir(dir)
488
489 def test_mode(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000490 # mkdtemp creates directories with the proper mode
Guido van Rossum0e548712002-08-09 16:14:33 +0000491 if not has_stat:
Benjamin Petersone549ead2009-03-28 21:42:05 +0000492 return # ugh, can't use SkipTest.
Guido van Rossum0e548712002-08-09 16:14:33 +0000493
494 dir = self.do_create()
495 try:
496 mode = stat.S_IMODE(os.stat(dir).st_mode)
Guido van Rossumcd16bf62007-06-13 18:07:49 +0000497 mode &= 0o777 # Mask off sticky bits inherited from /tmp
498 expected = 0o700
Ronald Oussoren94f25282010-05-05 19:11:21 +0000499 if sys.platform in ('win32', 'os2emx'):
Tim Petersca3ac7f2002-08-09 18:13:51 +0000500 # There's no distinction among 'user', 'group' and 'world';
501 # replicate the 'user' bits.
502 user = expected >> 6
503 expected = user * (1 + 8 + 64)
504 self.assertEqual(mode, expected)
Guido van Rossum0e548712002-08-09 16:14:33 +0000505 finally:
506 os.rmdir(dir)
507
508test_classes.append(test_mkdtemp)
509
510
511class test_mktemp(TC):
512 """Test mktemp()."""
513
514 # For safety, all use of mktemp must occur in a private directory.
515 # We must also suppress the RuntimeWarning it generates.
516 def setUp(self):
517 self.dir = tempfile.mkdtemp()
Brett Cannone1adece2010-03-20 22:19:55 +0000518 super().setUp()
Guido van Rossum0e548712002-08-09 16:14:33 +0000519
520 def tearDown(self):
521 if self.dir:
522 os.rmdir(self.dir)
523 self.dir = None
Brett Cannone1adece2010-03-20 22:19:55 +0000524 super().tearDown()
Guido van Rossum0e548712002-08-09 16:14:33 +0000525
526 class mktemped:
527 _unlink = os.unlink
528 _bflags = tempfile._bin_openflags
529
530 def __init__(self, dir, pre, suf):
531 self.name = tempfile.mktemp(dir=dir, prefix=pre, suffix=suf)
532 # Create the file. This will raise an exception if it's
533 # mysteriously appeared in the meanwhile.
Guido van Rossumcd16bf62007-06-13 18:07:49 +0000534 os.close(os.open(self.name, self._bflags, 0o600))
Guido van Rossum0e548712002-08-09 16:14:33 +0000535
536 def __del__(self):
537 self._unlink(self.name)
538
539 def do_create(self, pre="", suf=""):
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100540 file = self.mktemped(self.dir, pre, suf)
Guido van Rossum0e548712002-08-09 16:14:33 +0000541
542 self.nameCheck(file.name, self.dir, pre, suf)
543 return file
544
545 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000546 # mktemp can choose usable file names
Guido van Rossum0e548712002-08-09 16:14:33 +0000547 self.do_create()
548 self.do_create(pre="a")
549 self.do_create(suf="b")
550 self.do_create(pre="a", suf="b")
551 self.do_create(pre="aa", suf=".txt")
552
553 def test_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000554 # mktemp can choose many usable file names (stochastic)
Guido van Rossum805365e2007-05-07 22:24:25 +0000555 extant = list(range(TEST_FILES))
Guido van Rossum0e548712002-08-09 16:14:33 +0000556 for i in extant:
557 extant[i] = self.do_create(pre="aa")
558
Fred Drake8bec4832002-11-22 20:13:43 +0000559## def test_warning(self):
560## # mktemp issues a warning when used
561## warnings.filterwarnings("error",
562## category=RuntimeWarning,
563## message="mktemp")
564## self.assertRaises(RuntimeWarning,
565## tempfile.mktemp, dir=self.dir)
Guido van Rossum0e548712002-08-09 16:14:33 +0000566
567test_classes.append(test_mktemp)
568
569
570# We test _TemporaryFileWrapper by testing NamedTemporaryFile.
571
572
573class test_NamedTemporaryFile(TC):
574 """Test NamedTemporaryFile()."""
575
Guido van Rossumd8faa362007-04-27 19:54:29 +0000576 def do_create(self, dir=None, pre="", suf="", delete=True):
Guido van Rossum0e548712002-08-09 16:14:33 +0000577 if dir is None:
578 dir = tempfile.gettempdir()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100579 file = tempfile.NamedTemporaryFile(dir=dir, prefix=pre, suffix=suf,
580 delete=delete)
Guido van Rossum0e548712002-08-09 16:14:33 +0000581
582 self.nameCheck(file.name, dir, pre, suf)
583 return file
584
585
586 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000587 # NamedTemporaryFile can create files
Guido van Rossum0e548712002-08-09 16:14:33 +0000588 self.do_create()
589 self.do_create(pre="a")
590 self.do_create(suf="b")
591 self.do_create(pre="a", suf="b")
592 self.do_create(pre="aa", suf=".txt")
593
594 def test_creates_named(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000595 # NamedTemporaryFile creates files with names
Guido van Rossum0e548712002-08-09 16:14:33 +0000596 f = tempfile.NamedTemporaryFile()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000597 self.assertTrue(os.path.exists(f.name),
Guido van Rossum0e548712002-08-09 16:14:33 +0000598 "NamedTemporaryFile %s does not exist" % f.name)
599
600 def test_del_on_close(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000601 # A NamedTemporaryFile is deleted when closed
Guido van Rossum0e548712002-08-09 16:14:33 +0000602 dir = tempfile.mkdtemp()
603 try:
604 f = tempfile.NamedTemporaryFile(dir=dir)
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000605 f.write(b'blat')
Guido van Rossum0e548712002-08-09 16:14:33 +0000606 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000607 self.assertFalse(os.path.exists(f.name),
Guido van Rossum0e548712002-08-09 16:14:33 +0000608 "NamedTemporaryFile %s exists after close" % f.name)
609 finally:
610 os.rmdir(dir)
611
Guido van Rossumd8faa362007-04-27 19:54:29 +0000612 def test_dis_del_on_close(self):
613 # Tests that delete-on-close can be disabled
614 dir = tempfile.mkdtemp()
615 tmp = None
616 try:
617 f = tempfile.NamedTemporaryFile(dir=dir, delete=False)
618 tmp = f.name
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000619 f.write(b'blat')
Guido van Rossumd8faa362007-04-27 19:54:29 +0000620 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000621 self.assertTrue(os.path.exists(f.name),
Guido van Rossumd8faa362007-04-27 19:54:29 +0000622 "NamedTemporaryFile %s missing after close" % f.name)
623 finally:
624 if tmp is not None:
625 os.unlink(tmp)
626 os.rmdir(dir)
627
Guido van Rossum0e548712002-08-09 16:14:33 +0000628 def test_multiple_close(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000629 # A NamedTemporaryFile can be closed many times without error
Guido van Rossum0e548712002-08-09 16:14:33 +0000630 f = tempfile.NamedTemporaryFile()
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000631 f.write(b'abc\n')
Guido van Rossum0e548712002-08-09 16:14:33 +0000632 f.close()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100633 f.close()
634 f.close()
Guido van Rossum0e548712002-08-09 16:14:33 +0000635
Christian Heimes3ecfea712008-02-09 20:51:34 +0000636 def test_context_manager(self):
637 # A NamedTemporaryFile can be used as a context manager
638 with tempfile.NamedTemporaryFile() as f:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000639 self.assertTrue(os.path.exists(f.name))
640 self.assertFalse(os.path.exists(f.name))
Christian Heimes3ecfea712008-02-09 20:51:34 +0000641 def use_closed():
642 with f:
643 pass
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000644 self.assertRaises(ValueError, use_closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000645
Guido van Rossum0e548712002-08-09 16:14:33 +0000646 # How to test the mode and bufsize parameters?
647
648test_classes.append(test_NamedTemporaryFile)
649
Guido van Rossumd8faa362007-04-27 19:54:29 +0000650class test_SpooledTemporaryFile(TC):
651 """Test SpooledTemporaryFile()."""
652
653 def do_create(self, max_size=0, dir=None, pre="", suf=""):
654 if dir is None:
655 dir = tempfile.gettempdir()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100656 file = tempfile.SpooledTemporaryFile(max_size=max_size, dir=dir, prefix=pre, suffix=suf)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000657
658 return file
659
660
661 def test_basic(self):
662 # SpooledTemporaryFile can create files
663 f = self.do_create()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000664 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000665 f = self.do_create(max_size=100, pre="a", suf=".txt")
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000666 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000667
668 def test_del_on_close(self):
669 # A SpooledTemporaryFile is deleted when closed
670 dir = tempfile.mkdtemp()
671 try:
672 f = tempfile.SpooledTemporaryFile(max_size=10, dir=dir)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000673 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +0000674 f.write(b'blat ' * 5)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000675 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000676 filename = f.name
677 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000678 self.assertFalse(isinstance(filename, str) and os.path.exists(filename),
Guido van Rossumd8faa362007-04-27 19:54:29 +0000679 "SpooledTemporaryFile %s exists after close" % filename)
680 finally:
681 os.rmdir(dir)
682
683 def test_rewrite_small(self):
684 # A SpooledTemporaryFile can be written to multiple within the max_size
685 f = self.do_create(max_size=30)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000686 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000687 for i in range(5):
688 f.seek(0, 0)
Guido van Rossum39478e82007-08-27 17:23:59 +0000689 f.write(b'x' * 20)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000690 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000691
692 def test_write_sequential(self):
693 # A SpooledTemporaryFile should hold exactly max_size bytes, and roll
694 # over afterward
695 f = self.do_create(max_size=30)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000696 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +0000697 f.write(b'x' * 20)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000698 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +0000699 f.write(b'x' * 10)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000700 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +0000701 f.write(b'x')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000702 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000703
R David Murrayd89ee792011-03-14 09:55:46 -0400704 def test_writelines(self):
705 # Verify writelines with a SpooledTemporaryFile
706 f = self.do_create()
707 f.writelines((b'x', b'y', b'z'))
708 f.seek(0)
709 buf = f.read()
710 self.assertEqual(buf, b'xyz')
711
712 def test_writelines_sequential(self):
713 # A SpooledTemporaryFile should hold exactly max_size bytes, and roll
714 # over afterward
715 f = self.do_create(max_size=35)
716 f.writelines((b'x' * 20, b'x' * 10, b'x' * 5))
717 self.assertFalse(f._rolled)
718 f.write(b'x')
719 self.assertTrue(f._rolled)
720
Guido van Rossumd8faa362007-04-27 19:54:29 +0000721 def test_sparse(self):
722 # A SpooledTemporaryFile that is written late in the file will extend
723 # when that occurs
724 f = self.do_create(max_size=30)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000725 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000726 f.seek(100, 0)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000727 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +0000728 f.write(b'x')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000729 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000730
731 def test_fileno(self):
732 # A SpooledTemporaryFile should roll over to a real file on fileno()
733 f = self.do_create(max_size=30)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000734 self.assertFalse(f._rolled)
735 self.assertTrue(f.fileno() > 0)
736 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000737
Christian Heimes3ecfea712008-02-09 20:51:34 +0000738 def test_multiple_close_before_rollover(self):
Guido van Rossumd8faa362007-04-27 19:54:29 +0000739 # A SpooledTemporaryFile can be closed many times without error
740 f = tempfile.SpooledTemporaryFile()
Guido van Rossum39478e82007-08-27 17:23:59 +0000741 f.write(b'abc\n')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000742 self.assertFalse(f._rolled)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000743 f.close()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100744 f.close()
745 f.close()
Christian Heimes3ecfea712008-02-09 20:51:34 +0000746
747 def test_multiple_close_after_rollover(self):
748 # A SpooledTemporaryFile can be closed many times without error
749 f = tempfile.SpooledTemporaryFile(max_size=1)
750 f.write(b'abc\n')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000751 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000752 f.close()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100753 f.close()
754 f.close()
Guido van Rossumd8faa362007-04-27 19:54:29 +0000755
756 def test_bound_methods(self):
757 # It should be OK to steal a bound method from a SpooledTemporaryFile
758 # and use it independently; when the file rolls over, those bound
759 # methods should continue to function
760 f = self.do_create(max_size=30)
761 read = f.read
762 write = f.write
763 seek = f.seek
764
Guido van Rossum39478e82007-08-27 17:23:59 +0000765 write(b"a" * 35)
766 write(b"b" * 35)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000767 seek(0, 0)
Guido van Rossum9a634702007-07-09 10:24:45 +0000768 self.assertEqual(read(70), b'a'*35 + b'b'*35)
769
770 def test_text_mode(self):
771 # Creating a SpooledTemporaryFile with a text mode should produce
772 # a file object reading and writing (Unicode) text strings.
773 f = tempfile.SpooledTemporaryFile(mode='w+', max_size=10)
774 f.write("abc\n")
775 f.seek(0)
776 self.assertEqual(f.read(), "abc\n")
777 f.write("def\n")
778 f.seek(0)
779 self.assertEqual(f.read(), "abc\ndef\n")
780 f.write("xyzzy\n")
781 f.seek(0)
782 self.assertEqual(f.read(), "abc\ndef\nxyzzy\n")
Amaury Forgeot d'Arc7d0bddd2009-11-30 00:08:56 +0000783 # Check that Ctrl+Z doesn't truncate the file
784 f.write("foo\x1abar\n")
785 f.seek(0)
786 self.assertEqual(f.read(), "abc\ndef\nxyzzy\nfoo\x1abar\n")
Guido van Rossum9a634702007-07-09 10:24:45 +0000787
Guido van Rossumf0c74162007-08-28 03:29:45 +0000788 def test_text_newline_and_encoding(self):
789 f = tempfile.SpooledTemporaryFile(mode='w+', max_size=10,
790 newline='', encoding='utf-8')
791 f.write("\u039B\r\n")
792 f.seek(0)
793 self.assertEqual(f.read(), "\u039B\r\n")
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000794 self.assertFalse(f._rolled)
Guido van Rossumf0c74162007-08-28 03:29:45 +0000795
796 f.write("\u039B" * 20 + "\r\n")
797 f.seek(0)
798 self.assertEqual(f.read(), "\u039B\r\n" + ("\u039B" * 20) + "\r\n")
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000799 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000800
Christian Heimes3ecfea712008-02-09 20:51:34 +0000801 def test_context_manager_before_rollover(self):
802 # A SpooledTemporaryFile can be used as a context manager
803 with tempfile.SpooledTemporaryFile(max_size=1) as f:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000804 self.assertFalse(f._rolled)
805 self.assertFalse(f.closed)
806 self.assertTrue(f.closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000807 def use_closed():
808 with f:
809 pass
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000810 self.assertRaises(ValueError, use_closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000811
812 def test_context_manager_during_rollover(self):
813 # A SpooledTemporaryFile can be used as a context manager
814 with tempfile.SpooledTemporaryFile(max_size=1) as f:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000815 self.assertFalse(f._rolled)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000816 f.write(b'abc\n')
817 f.flush()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000818 self.assertTrue(f._rolled)
819 self.assertFalse(f.closed)
820 self.assertTrue(f.closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000821 def use_closed():
822 with f:
823 pass
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000824 self.assertRaises(ValueError, use_closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000825
826 def test_context_manager_after_rollover(self):
827 # A SpooledTemporaryFile can be used as a context manager
828 f = tempfile.SpooledTemporaryFile(max_size=1)
829 f.write(b'abc\n')
830 f.flush()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000831 self.assertTrue(f._rolled)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000832 with f:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000833 self.assertFalse(f.closed)
834 self.assertTrue(f.closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000835 def use_closed():
836 with f:
837 pass
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000838 self.assertRaises(ValueError, use_closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000839
Antoine Pitrou0e86a582011-11-25 18:03:09 +0100840 def test_truncate_with_size_parameter(self):
841 # A SpooledTemporaryFile can be truncated to zero size
842 f = tempfile.SpooledTemporaryFile(max_size=10)
843 f.write(b'abcdefg\n')
844 f.seek(0)
845 f.truncate()
846 self.assertFalse(f._rolled)
847 self.assertEqual(f._file.getvalue(), b'')
848 # A SpooledTemporaryFile can be truncated to a specific size
849 f = tempfile.SpooledTemporaryFile(max_size=10)
850 f.write(b'abcdefg\n')
851 f.truncate(4)
852 self.assertFalse(f._rolled)
853 self.assertEqual(f._file.getvalue(), b'abcd')
854 # A SpooledTemporaryFile rolls over if truncated to large size
855 f = tempfile.SpooledTemporaryFile(max_size=10)
856 f.write(b'abcdefg\n')
857 f.truncate(20)
858 self.assertTrue(f._rolled)
859 if has_stat:
860 self.assertEqual(os.fstat(f.fileno()).st_size, 20)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000861
Guido van Rossumd8faa362007-04-27 19:54:29 +0000862test_classes.append(test_SpooledTemporaryFile)
863
Guido van Rossum0e548712002-08-09 16:14:33 +0000864
865class test_TemporaryFile(TC):
866 """Test TemporaryFile()."""
867
868 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000869 # TemporaryFile can create files
Guido van Rossum0e548712002-08-09 16:14:33 +0000870 # No point in testing the name params - the file has no name.
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100871 tempfile.TemporaryFile()
Guido van Rossum0e548712002-08-09 16:14:33 +0000872
873 def test_has_no_name(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000874 # TemporaryFile creates files with no names (on this system)
Guido van Rossum0e548712002-08-09 16:14:33 +0000875 dir = tempfile.mkdtemp()
876 f = tempfile.TemporaryFile(dir=dir)
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000877 f.write(b'blat')
Guido van Rossum0e548712002-08-09 16:14:33 +0000878
879 # Sneaky: because this file has no name, it should not prevent
880 # us from removing the directory it was created in.
881 try:
882 os.rmdir(dir)
883 except:
Guido van Rossum0e548712002-08-09 16:14:33 +0000884 # cleanup
885 f.close()
886 os.rmdir(dir)
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100887 raise
Guido van Rossum0e548712002-08-09 16:14:33 +0000888
889 def test_multiple_close(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000890 # A TemporaryFile can be closed many times without error
Guido van Rossum0e548712002-08-09 16:14:33 +0000891 f = tempfile.TemporaryFile()
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000892 f.write(b'abc\n')
Guido van Rossum0e548712002-08-09 16:14:33 +0000893 f.close()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100894 f.close()
895 f.close()
Guido van Rossum0e548712002-08-09 16:14:33 +0000896
897 # How to test the mode and bufsize parameters?
Guido van Rossumf0c74162007-08-28 03:29:45 +0000898 def test_mode_and_encoding(self):
899
900 def roundtrip(input, *args, **kwargs):
901 with tempfile.TemporaryFile(*args, **kwargs) as fileobj:
902 fileobj.write(input)
903 fileobj.seek(0)
Ezio Melottib3aedd42010-11-20 19:04:17 +0000904 self.assertEqual(input, fileobj.read())
Guido van Rossumf0c74162007-08-28 03:29:45 +0000905
906 roundtrip(b"1234", "w+b")
907 roundtrip("abdc\n", "w+")
908 roundtrip("\u039B", "w+", encoding="utf-16")
909 roundtrip("foo\r\n", "w+", newline="")
Guido van Rossum0e548712002-08-09 16:14:33 +0000910
Guido van Rossum0e548712002-08-09 16:14:33 +0000911
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000912if tempfile.NamedTemporaryFile is not tempfile.TemporaryFile:
Guido van Rossum0e548712002-08-09 16:14:33 +0000913 test_classes.append(test_TemporaryFile)
914
Nick Coghlan543af752010-10-24 11:23:25 +0000915
916# Helper for test_del_on_shutdown
917class NulledModules:
918 def __init__(self, *modules):
919 self.refs = [mod.__dict__ for mod in modules]
920 self.contents = [ref.copy() for ref in self.refs]
921
922 def __enter__(self):
923 for d in self.refs:
924 for key in d:
925 d[key] = None
926
927 def __exit__(self, *exc_info):
928 for d, c in zip(self.refs, self.contents):
929 d.clear()
930 d.update(c)
931
932class test_TemporaryDirectory(TC):
933 """Test TemporaryDirectory()."""
934
935 def do_create(self, dir=None, pre="", suf="", recurse=1):
936 if dir is None:
937 dir = tempfile.gettempdir()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100938 tmp = tempfile.TemporaryDirectory(dir=dir, prefix=pre, suffix=suf)
Nick Coghlan543af752010-10-24 11:23:25 +0000939 self.nameCheck(tmp.name, dir, pre, suf)
940 # Create a subdirectory and some files
941 if recurse:
942 self.do_create(tmp.name, pre, suf, recurse-1)
943 with open(os.path.join(tmp.name, "test.txt"), "wb") as f:
944 f.write(b"Hello world!")
945 return tmp
946
Nick Coghlan6b22f3f2010-12-12 15:24:21 +0000947 def test_mkdtemp_failure(self):
948 # Check no additional exception if mkdtemp fails
949 # Previously would raise AttributeError instead
Nick Coghlan3c54ea62010-12-13 03:02:43 +0000950 # (noted as part of Issue #10188)
951 with tempfile.TemporaryDirectory() as nonexistent:
952 pass
953 with self.assertRaises(os.error):
954 tempfile.TemporaryDirectory(dir=nonexistent)
Nick Coghlan6b22f3f2010-12-12 15:24:21 +0000955
Nick Coghlan543af752010-10-24 11:23:25 +0000956 def test_explicit_cleanup(self):
957 # A TemporaryDirectory is deleted when cleaned up
958 dir = tempfile.mkdtemp()
959 try:
960 d = self.do_create(dir=dir)
961 self.assertTrue(os.path.exists(d.name),
962 "TemporaryDirectory %s does not exist" % d.name)
963 d.cleanup()
964 self.assertFalse(os.path.exists(d.name),
965 "TemporaryDirectory %s exists after cleanup" % d.name)
966 finally:
967 os.rmdir(dir)
968
Charles-François Natalidef35432011-07-29 18:59:24 +0200969 @support.skip_unless_symlink
970 def test_cleanup_with_symlink_to_a_directory(self):
971 # cleanup() should not follow symlinks to directories (issue #12464)
972 d1 = self.do_create()
973 d2 = self.do_create()
974
975 # Symlink d1/foo -> d2
976 os.symlink(d2.name, os.path.join(d1.name, "foo"))
977
978 # This call to cleanup() should not follow the "foo" symlink
979 d1.cleanup()
980
981 self.assertFalse(os.path.exists(d1.name),
982 "TemporaryDirectory %s exists after cleanup" % d1.name)
983 self.assertTrue(os.path.exists(d2.name),
984 "Directory pointed to by a symlink was deleted")
985 self.assertEqual(os.listdir(d2.name), ['test.txt'],
986 "Contents of the directory pointed to by a symlink "
987 "were deleted")
988 d2.cleanup()
989
Nick Coghlan543af752010-10-24 11:23:25 +0000990 @support.cpython_only
991 def test_del_on_collection(self):
992 # A TemporaryDirectory is deleted when garbage collected
993 dir = tempfile.mkdtemp()
994 try:
995 d = self.do_create(dir=dir)
996 name = d.name
997 del d # Rely on refcounting to invoke __del__
998 self.assertFalse(os.path.exists(name),
999 "TemporaryDirectory %s exists after __del__" % name)
1000 finally:
1001 os.rmdir(dir)
1002
1003 @unittest.expectedFailure # See issue #10188
1004 def test_del_on_shutdown(self):
1005 # A TemporaryDirectory may be cleaned up during shutdown
1006 # Make sure it works with the relevant modules nulled out
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001007 with self.do_create() as dir:
Nick Coghlan543af752010-10-24 11:23:25 +00001008 d = self.do_create(dir=dir)
1009 # Mimic the nulling out of modules that
1010 # occurs during system shutdown
1011 modules = [os, os.path]
1012 if has_stat:
1013 modules.append(stat)
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001014 # Currently broken, so suppress the warning
1015 # that is otherwise emitted on stdout
1016 with support.captured_stderr() as err:
1017 with NulledModules(*modules):
1018 d.cleanup()
1019 # Currently broken, so stop spurious exception by
1020 # indicating the object has already been closed
1021 d._closed = True
1022 # And this assert will fail, as expected by the
1023 # unittest decorator...
Nick Coghlan543af752010-10-24 11:23:25 +00001024 self.assertFalse(os.path.exists(d.name),
1025 "TemporaryDirectory %s exists after cleanup" % d.name)
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001026
1027 def test_warnings_on_cleanup(self):
1028 # Two kinds of warning on shutdown
1029 # Issue 10888: may write to stderr if modules are nulled out
1030 # ResourceWarning will be triggered by __del__
1031 with self.do_create() as dir:
Nick Coghlane98e8a32010-12-13 16:32:51 +00001032 if os.sep != '\\':
1033 # Embed a backslash in order to make sure string escaping
1034 # in the displayed error message is dealt with correctly
1035 suffix = '\\check_backslash_handling'
1036 else:
1037 suffix = ''
1038 d = self.do_create(dir=dir, suf=suffix)
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001039
1040 #Check for the Issue 10888 message
1041 modules = [os, os.path]
1042 if has_stat:
1043 modules.append(stat)
1044 with support.captured_stderr() as err:
1045 with NulledModules(*modules):
1046 d.cleanup()
Nick Coghlane98e8a32010-12-13 16:32:51 +00001047 message = err.getvalue().replace('\\\\', '\\')
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001048 self.assertIn("while cleaning up", message)
1049 self.assertIn(d.name, message)
1050
1051 # Check for the resource warning
1052 with support.check_warnings(('Implicitly', ResourceWarning), quiet=False):
1053 warnings.filterwarnings("always", category=ResourceWarning)
1054 d.__del__()
1055 self.assertFalse(os.path.exists(d.name),
1056 "TemporaryDirectory %s exists after __del__" % d.name)
Nick Coghlan543af752010-10-24 11:23:25 +00001057
1058 def test_multiple_close(self):
1059 # Can be cleaned-up many times without error
1060 d = self.do_create()
1061 d.cleanup()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +01001062 d.cleanup()
1063 d.cleanup()
Nick Coghlan543af752010-10-24 11:23:25 +00001064
1065 def test_context_manager(self):
1066 # Can be used as a context manager
1067 d = self.do_create()
1068 with d as name:
1069 self.assertTrue(os.path.exists(name))
1070 self.assertEqual(name, d.name)
1071 self.assertFalse(os.path.exists(name))
1072
1073
1074test_classes.append(test_TemporaryDirectory)
1075
Guido van Rossum0e548712002-08-09 16:14:33 +00001076def test_main():
Benjamin Petersonee8712c2008-05-20 21:35:26 +00001077 support.run_unittest(*test_classes)
Guido van Rossum0e548712002-08-09 16:14:33 +00001078
1079if __name__ == "__main__":
1080 test_main()