blob: a90cd0ef0ff1f927a7f1473cbd678ed107d38ec0 [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)
Ronald Oussoren94f25282010-05-05 19:11:21 +000024if sys.platform in ('openbsd3', 'openbsd4'):
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 failOnException(self, what, ei=None):
49 if ei is None:
50 ei = sys.exc_info()
51 self.fail("%s raised %s: %s" % (what, ei[0], ei[1]))
52
53 def nameCheck(self, name, dir, pre, suf):
54 (ndir, nbase) = os.path.split(name)
55 npre = nbase[:len(pre)]
56 nsuf = nbase[len(nbase)-len(suf):]
57
Martin v. Löwisd6625482003-10-12 17:37:01 +000058 # check for equality of the absolute paths!
59 self.assertEqual(os.path.abspath(ndir), os.path.abspath(dir),
Guido van Rossum0e548712002-08-09 16:14:33 +000060 "file '%s' not in directory '%s'" % (name, dir))
61 self.assertEqual(npre, pre,
62 "file '%s' does not begin with '%s'" % (nbase, pre))
63 self.assertEqual(nsuf, suf,
64 "file '%s' does not end with '%s'" % (nbase, suf))
65
66 nbase = nbase[len(pre):len(nbase)-len(suf)]
Benjamin Petersonc9c0f202009-06-30 23:06:06 +000067 self.assertTrue(self.str_check.match(nbase),
Guido van Rossum0e548712002-08-09 16:14:33 +000068 "random string '%s' does not match /^[a-zA-Z0-9_-]{6}$/"
69 % nbase)
70
71test_classes = []
72
73class test_exports(TC):
74 def test_exports(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +000075 # There are no surprising symbols in the tempfile module
Guido van Rossum0e548712002-08-09 16:14:33 +000076 dict = tempfile.__dict__
77
78 expected = {
79 "NamedTemporaryFile" : 1,
80 "TemporaryFile" : 1,
81 "mkstemp" : 1,
82 "mkdtemp" : 1,
83 "mktemp" : 1,
84 "TMP_MAX" : 1,
85 "gettempprefix" : 1,
86 "gettempdir" : 1,
87 "tempdir" : 1,
Guido van Rossumd8faa362007-04-27 19:54:29 +000088 "template" : 1,
Nick Coghlan543af752010-10-24 11:23:25 +000089 "SpooledTemporaryFile" : 1,
90 "TemporaryDirectory" : 1,
Guido van Rossum0e548712002-08-09 16:14:33 +000091 }
92
93 unexp = []
94 for key in dict:
95 if key[0] != '_' and key not in expected:
96 unexp.append(key)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +000097 self.assertTrue(len(unexp) == 0,
Guido van Rossum0e548712002-08-09 16:14:33 +000098 "unexpected keys: %s" % unexp)
99
100test_classes.append(test_exports)
101
102
Guido van Rossum0e548712002-08-09 16:14:33 +0000103class test__RandomNameSequence(TC):
104 """Test the internal iterator object _RandomNameSequence."""
105
106 def setUp(self):
107 self.r = tempfile._RandomNameSequence()
Brett Cannone1adece2010-03-20 22:19:55 +0000108 super().setUp()
Guido van Rossum0e548712002-08-09 16:14:33 +0000109
110 def test_get_six_char_str(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000111 # _RandomNameSequence returns a six-character string
Georg Brandla18af4e2007-04-21 15:47:16 +0000112 s = next(self.r)
Guido van Rossum0e548712002-08-09 16:14:33 +0000113 self.nameCheck(s, '', '', '')
114
115 def test_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000116 # _RandomNameSequence returns no duplicate strings (stochastic)
Guido van Rossum0e548712002-08-09 16:14:33 +0000117
118 dict = {}
119 r = self.r
Guido van Rossum805365e2007-05-07 22:24:25 +0000120 for i in range(TEST_FILES):
Georg Brandla18af4e2007-04-21 15:47:16 +0000121 s = next(r)
Guido van Rossum0e548712002-08-09 16:14:33 +0000122 self.nameCheck(s, '', '', '')
Ezio Melottib58e0bd2010-01-23 15:40:09 +0000123 self.assertNotIn(s, dict)
Guido van Rossum0e548712002-08-09 16:14:33 +0000124 dict[s] = 1
125
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000126 def supports_iter(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000127 # _RandomNameSequence supports the iterator protocol
Guido van Rossum0e548712002-08-09 16:14:33 +0000128
129 i = 0
130 r = self.r
131 try:
132 for s in r:
133 i += 1
134 if i == 20:
135 break
136 except:
Georg Brandl89fad142010-03-14 10:23:39 +0000137 self.failOnException("iteration")
Guido van Rossum0e548712002-08-09 16:14:33 +0000138
Antoine Pitrou4558bad2011-11-25 21:28:15 +0100139 @unittest.skipUnless(hasattr(os, 'fork'),
140 "os.fork is required for this test")
141 def test_process_awareness(self):
142 # ensure that the random source differs between
143 # child and parent.
144 read_fd, write_fd = os.pipe()
145 pid = None
146 try:
147 pid = os.fork()
148 if not pid:
149 os.close(read_fd)
150 os.write(write_fd, next(self.r).encode("ascii"))
151 os.close(write_fd)
152 # bypass the normal exit handlers- leave those to
153 # the parent.
154 os._exit(0)
155 parent_value = next(self.r)
156 child_value = os.read(read_fd, len(parent_value)).decode("ascii")
157 finally:
158 if pid:
159 # best effort to ensure the process can't bleed out
160 # via any bugs above
161 try:
162 os.kill(pid, signal.SIGKILL)
163 except EnvironmentError:
164 pass
165 os.close(read_fd)
166 os.close(write_fd)
167 self.assertNotEqual(child_value, parent_value)
168
169
Guido van Rossum0e548712002-08-09 16:14:33 +0000170test_classes.append(test__RandomNameSequence)
171
172
173class test__candidate_tempdir_list(TC):
174 """Test the internal function _candidate_tempdir_list."""
175
176 def test_nonempty_list(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000177 # _candidate_tempdir_list returns a nonempty list of strings
Guido van Rossum0e548712002-08-09 16:14:33 +0000178
179 cand = tempfile._candidate_tempdir_list()
180
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000181 self.assertFalse(len(cand) == 0)
Guido van Rossum0e548712002-08-09 16:14:33 +0000182 for c in cand:
Ezio Melottie9615932010-01-24 19:26:24 +0000183 self.assertIsInstance(c, str)
Guido van Rossum0e548712002-08-09 16:14:33 +0000184
185 def test_wanted_dirs(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000186 # _candidate_tempdir_list contains the expected directories
Guido van Rossum0e548712002-08-09 16:14:33 +0000187
188 # Make sure the interesting environment variables are all set.
Hirokazu Yamamoto71959632009-04-27 01:44:28 +0000189 with support.EnvironmentVarGuard() as env:
Guido van Rossum0e548712002-08-09 16:14:33 +0000190 for envname in 'TMPDIR', 'TEMP', 'TMP':
191 dirname = os.getenv(envname)
192 if not dirname:
Walter Dörwald155374d2009-05-01 19:58:58 +0000193 env[envname] = os.path.abspath(envname)
Guido van Rossum0e548712002-08-09 16:14:33 +0000194
195 cand = tempfile._candidate_tempdir_list()
196
197 for envname in 'TMPDIR', 'TEMP', 'TMP':
198 dirname = os.getenv(envname)
199 if not dirname: raise ValueError
Benjamin Peterson577473f2010-01-19 00:09:57 +0000200 self.assertIn(dirname, cand)
Guido van Rossum0e548712002-08-09 16:14:33 +0000201
202 try:
203 dirname = os.getcwd()
204 except (AttributeError, os.error):
205 dirname = os.curdir
206
Benjamin Peterson577473f2010-01-19 00:09:57 +0000207 self.assertIn(dirname, cand)
Guido van Rossum0e548712002-08-09 16:14:33 +0000208
209 # Not practical to try to verify the presence of OS-specific
210 # paths in this list.
Guido van Rossum0e548712002-08-09 16:14:33 +0000211
212test_classes.append(test__candidate_tempdir_list)
213
214
215# We test _get_default_tempdir by testing gettempdir.
216
217
218class test__get_candidate_names(TC):
219 """Test the internal function _get_candidate_names."""
220
221 def test_retval(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000222 # _get_candidate_names returns a _RandomNameSequence object
Guido van Rossum0e548712002-08-09 16:14:33 +0000223 obj = tempfile._get_candidate_names()
Ezio Melottie9615932010-01-24 19:26:24 +0000224 self.assertIsInstance(obj, tempfile._RandomNameSequence)
Guido van Rossum0e548712002-08-09 16:14:33 +0000225
226 def test_same_thing(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000227 # _get_candidate_names always returns the same object
Guido van Rossum0e548712002-08-09 16:14:33 +0000228 a = tempfile._get_candidate_names()
229 b = tempfile._get_candidate_names()
230
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000231 self.assertTrue(a is b)
Guido van Rossum0e548712002-08-09 16:14:33 +0000232
233test_classes.append(test__get_candidate_names)
234
235
236class test__mkstemp_inner(TC):
237 """Test the internal function _mkstemp_inner."""
238
239 class mkstemped:
240 _bflags = tempfile._bin_openflags
241 _tflags = tempfile._text_openflags
242 _close = os.close
243 _unlink = os.unlink
244
245 def __init__(self, dir, pre, suf, bin):
246 if bin: flags = self._bflags
247 else: flags = self._tflags
248
249 (self.fd, self.name) = tempfile._mkstemp_inner(dir, pre, suf, flags)
250
251 def write(self, str):
252 os.write(self.fd, str)
253
254 def __del__(self):
255 self._close(self.fd)
256 self._unlink(self.name)
Tim Petersa0d55de2002-08-09 18:01:01 +0000257
Guido van Rossum0e548712002-08-09 16:14:33 +0000258 def do_create(self, dir=None, pre="", suf="", bin=1):
259 if dir is None:
260 dir = tempfile.gettempdir()
261 try:
262 file = self.mkstemped(dir, pre, suf, bin)
263 except:
264 self.failOnException("_mkstemp_inner")
265
266 self.nameCheck(file.name, dir, pre, suf)
267 return file
268
269 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000270 # _mkstemp_inner can create files
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000271 self.do_create().write(b"blat")
272 self.do_create(pre="a").write(b"blat")
273 self.do_create(suf="b").write(b"blat")
274 self.do_create(pre="a", suf="b").write(b"blat")
275 self.do_create(pre="aa", suf=".txt").write(b"blat")
Guido van Rossum0e548712002-08-09 16:14:33 +0000276
277 def test_basic_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000278 # _mkstemp_inner can create many files (stochastic)
Guido van Rossum805365e2007-05-07 22:24:25 +0000279 extant = list(range(TEST_FILES))
Guido van Rossum0e548712002-08-09 16:14:33 +0000280 for i in extant:
281 extant[i] = self.do_create(pre="aa")
282
283 def test_choose_directory(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000284 # _mkstemp_inner can create files in a user-selected directory
Guido van Rossum0e548712002-08-09 16:14:33 +0000285 dir = tempfile.mkdtemp()
286 try:
Antoine Pitrou9cadb1b2008-09-15 23:02:56 +0000287 self.do_create(dir=dir).write(b"blat")
Guido van Rossum0e548712002-08-09 16:14:33 +0000288 finally:
289 os.rmdir(dir)
290
291 def test_file_mode(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000292 # _mkstemp_inner creates files with the proper mode
Guido van Rossum0e548712002-08-09 16:14:33 +0000293 if not has_stat:
Benjamin Petersone549ead2009-03-28 21:42:05 +0000294 return # ugh, can't use SkipTest.
Guido van Rossum0e548712002-08-09 16:14:33 +0000295
296 file = self.do_create()
297 mode = stat.S_IMODE(os.stat(file.name).st_mode)
Guido van Rossumcd16bf62007-06-13 18:07:49 +0000298 expected = 0o600
Ronald Oussoren94f25282010-05-05 19:11:21 +0000299 if sys.platform in ('win32', 'os2emx'):
Tim Petersca3ac7f2002-08-09 18:13:51 +0000300 # There's no distinction among 'user', 'group' and 'world';
301 # replicate the 'user' bits.
302 user = expected >> 6
303 expected = user * (1 + 8 + 64)
304 self.assertEqual(mode, expected)
Guido van Rossum0e548712002-08-09 16:14:33 +0000305
306 def test_noinherit(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000307 # _mkstemp_inner file handles are not inherited by child processes
Guido van Rossum78741062002-08-17 11:41:01 +0000308 if not has_spawnl:
Benjamin Petersone549ead2009-03-28 21:42:05 +0000309 return # ugh, can't use SkipTest.
Guido van Rossum0e548712002-08-09 16:14:33 +0000310
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000311 if support.verbose:
Guido van Rossum78741062002-08-17 11:41:01 +0000312 v="v"
313 else:
314 v="q"
315
Guido van Rossum0e548712002-08-09 16:14:33 +0000316 file = self.do_create()
Guido van Rossum78741062002-08-17 11:41:01 +0000317 fd = "%d" % file.fd
318
319 try:
320 me = __file__
321 except NameError:
322 me = sys.argv[0]
Guido van Rossum0e548712002-08-09 16:14:33 +0000323
324 # We have to exec something, so that FD_CLOEXEC will take
Guido van Rossum78741062002-08-17 11:41:01 +0000325 # effect. The core of this test is therefore in
326 # tf_inherit_check.py, which see.
327 tester = os.path.join(os.path.dirname(os.path.abspath(me)),
328 "tf_inherit_check.py")
Guido van Rossum0e548712002-08-09 16:14:33 +0000329
Martin v. Löwisd4210bc2003-10-23 15:55:28 +0000330 # On Windows a spawn* /path/ with embedded spaces shouldn't be quoted,
331 # but an arg with embedded spaces should be decorated with double
332 # quotes on each end
Christian Heimes895627f2007-12-08 17:28:33 +0000333 if sys.platform in ('win32',):
Martin v. Löwisd4210bc2003-10-23 15:55:28 +0000334 decorated = '"%s"' % sys.executable
335 tester = '"%s"' % tester
336 else:
337 decorated = sys.executable
338
339 retval = os.spawnl(os.P_WAIT, sys.executable, decorated, tester, v, fd)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000340 self.assertFalse(retval < 0,
Guido van Rossum78741062002-08-17 11:41:01 +0000341 "child process caught fatal signal %d" % -retval)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000342 self.assertFalse(retval > 0, "child process reports failure %d"%retval)
Guido van Rossum0e548712002-08-09 16:14:33 +0000343
344 def test_textmode(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000345 # _mkstemp_inner can create files in text mode
Guido van Rossum0e548712002-08-09 16:14:33 +0000346 if not has_textmode:
Benjamin Petersone549ead2009-03-28 21:42:05 +0000347 return # ugh, can't use SkipTest.
Guido van Rossum0e548712002-08-09 16:14:33 +0000348
Amaury Forgeot d'Arc7d0bddd2009-11-30 00:08:56 +0000349 # A text file is truncated at the first Ctrl+Z byte
350 f = self.do_create(bin=0)
351 f.write(b"blat\x1a")
352 f.write(b"extra\n")
353 os.lseek(f.fd, 0, os.SEEK_SET)
Ezio Melottib3aedd42010-11-20 19:04:17 +0000354 self.assertEqual(os.read(f.fd, 20), b"blat")
Guido van Rossum0e548712002-08-09 16:14:33 +0000355
356test_classes.append(test__mkstemp_inner)
357
358
359class test_gettempprefix(TC):
360 """Test gettempprefix()."""
361
362 def test_sane_template(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000363 # gettempprefix returns a nonempty prefix string
Guido van Rossum0e548712002-08-09 16:14:33 +0000364 p = tempfile.gettempprefix()
365
Ezio Melottie9615932010-01-24 19:26:24 +0000366 self.assertIsInstance(p, str)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000367 self.assertTrue(len(p) > 0)
Guido van Rossum0e548712002-08-09 16:14:33 +0000368
369 def test_usable_template(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000370 # gettempprefix returns a usable prefix string
Guido van Rossum0e548712002-08-09 16:14:33 +0000371
372 # Create a temp directory, avoiding use of the prefix.
373 # Then attempt to create a file whose name is
374 # prefix + 'xxxxxx.xxx' in that directory.
375 p = tempfile.gettempprefix() + "xxxxxx.xxx"
376 d = tempfile.mkdtemp(prefix="")
377 try:
378 p = os.path.join(d, p)
379 try:
380 fd = os.open(p, os.O_RDWR | os.O_CREAT)
381 except:
382 self.failOnException("os.open")
383 os.close(fd)
384 os.unlink(p)
385 finally:
386 os.rmdir(d)
387
388test_classes.append(test_gettempprefix)
389
390
391class test_gettempdir(TC):
392 """Test gettempdir()."""
393
394 def test_directory_exists(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000395 # gettempdir returns a directory which exists
Guido van Rossum0e548712002-08-09 16:14:33 +0000396
397 dir = tempfile.gettempdir()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000398 self.assertTrue(os.path.isabs(dir) or dir == os.curdir,
Guido van Rossum0e548712002-08-09 16:14:33 +0000399 "%s is not an absolute path" % dir)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000400 self.assertTrue(os.path.isdir(dir),
Guido van Rossum0e548712002-08-09 16:14:33 +0000401 "%s is not a directory" % dir)
402
403 def test_directory_writable(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000404 # gettempdir returns a directory writable by the user
Guido van Rossum0e548712002-08-09 16:14:33 +0000405
406 # sneaky: just instantiate a NamedTemporaryFile, which
407 # defaults to writing into the directory returned by
408 # gettempdir.
409 try:
410 file = tempfile.NamedTemporaryFile()
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000411 file.write(b"blat")
Guido van Rossum0e548712002-08-09 16:14:33 +0000412 file.close()
413 except:
414 self.failOnException("create file in %s" % tempfile.gettempdir())
415
416 def test_same_thing(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000417 # gettempdir always returns the same object
Guido van Rossum0e548712002-08-09 16:14:33 +0000418 a = tempfile.gettempdir()
419 b = tempfile.gettempdir()
420
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000421 self.assertTrue(a is b)
Guido van Rossum0e548712002-08-09 16:14:33 +0000422
423test_classes.append(test_gettempdir)
424
425
426class test_mkstemp(TC):
427 """Test mkstemp()."""
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000428
Thomas Wouters73e5a5b2006-06-08 15:35:45 +0000429 def do_create(self, dir=None, pre="", suf=""):
Guido van Rossum0e548712002-08-09 16:14:33 +0000430 if dir is None:
431 dir = tempfile.gettempdir()
432 try:
433 (fd, name) = tempfile.mkstemp(dir=dir, prefix=pre, suffix=suf)
Martin v. Löwisd6625482003-10-12 17:37:01 +0000434 (ndir, nbase) = os.path.split(name)
435 adir = os.path.abspath(dir)
436 self.assertEqual(adir, ndir,
437 "Directory '%s' incorrectly returned as '%s'" % (adir, ndir))
Guido van Rossum0e548712002-08-09 16:14:33 +0000438 except:
439 self.failOnException("mkstemp")
440
441 try:
442 self.nameCheck(name, dir, pre, suf)
443 finally:
444 os.close(fd)
445 os.unlink(name)
446
447 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000448 # mkstemp can create files
Guido van Rossum0e548712002-08-09 16:14:33 +0000449 self.do_create()
450 self.do_create(pre="a")
451 self.do_create(suf="b")
452 self.do_create(pre="a", suf="b")
453 self.do_create(pre="aa", suf=".txt")
Martin v. Löwisd6625482003-10-12 17:37:01 +0000454 self.do_create(dir=".")
Guido van Rossum0e548712002-08-09 16:14:33 +0000455
456 def test_choose_directory(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000457 # mkstemp can create directories in a user-selected directory
Guido van Rossum0e548712002-08-09 16:14:33 +0000458 dir = tempfile.mkdtemp()
459 try:
460 self.do_create(dir=dir)
461 finally:
462 os.rmdir(dir)
463
464test_classes.append(test_mkstemp)
465
466
467class test_mkdtemp(TC):
468 """Test mkdtemp()."""
469
470 def do_create(self, dir=None, pre="", suf=""):
471 if dir is None:
472 dir = tempfile.gettempdir()
473 try:
474 name = tempfile.mkdtemp(dir=dir, prefix=pre, suffix=suf)
475 except:
476 self.failOnException("mkdtemp")
477
478 try:
479 self.nameCheck(name, dir, pre, suf)
480 return name
481 except:
482 os.rmdir(name)
483 raise
484
485 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000486 # mkdtemp can create directories
Guido van Rossum0e548712002-08-09 16:14:33 +0000487 os.rmdir(self.do_create())
488 os.rmdir(self.do_create(pre="a"))
489 os.rmdir(self.do_create(suf="b"))
490 os.rmdir(self.do_create(pre="a", suf="b"))
491 os.rmdir(self.do_create(pre="aa", suf=".txt"))
Tim Petersa0d55de2002-08-09 18:01:01 +0000492
Guido van Rossum0e548712002-08-09 16:14:33 +0000493 def test_basic_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000494 # mkdtemp can create many directories (stochastic)
Guido van Rossum805365e2007-05-07 22:24:25 +0000495 extant = list(range(TEST_FILES))
Guido van Rossum0e548712002-08-09 16:14:33 +0000496 try:
497 for i in extant:
498 extant[i] = self.do_create(pre="aa")
499 finally:
500 for i in extant:
Guido van Rossum3172c5d2007-10-16 18:12:55 +0000501 if(isinstance(i, str)):
Guido van Rossum0e548712002-08-09 16:14:33 +0000502 os.rmdir(i)
503
504 def test_choose_directory(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000505 # mkdtemp can create directories in a user-selected directory
Guido van Rossum0e548712002-08-09 16:14:33 +0000506 dir = tempfile.mkdtemp()
507 try:
508 os.rmdir(self.do_create(dir=dir))
509 finally:
510 os.rmdir(dir)
511
512 def test_mode(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000513 # mkdtemp creates directories with the proper mode
Guido van Rossum0e548712002-08-09 16:14:33 +0000514 if not has_stat:
Benjamin Petersone549ead2009-03-28 21:42:05 +0000515 return # ugh, can't use SkipTest.
Guido van Rossum0e548712002-08-09 16:14:33 +0000516
517 dir = self.do_create()
518 try:
519 mode = stat.S_IMODE(os.stat(dir).st_mode)
Guido van Rossumcd16bf62007-06-13 18:07:49 +0000520 mode &= 0o777 # Mask off sticky bits inherited from /tmp
521 expected = 0o700
Ronald Oussoren94f25282010-05-05 19:11:21 +0000522 if sys.platform in ('win32', 'os2emx'):
Tim Petersca3ac7f2002-08-09 18:13:51 +0000523 # There's no distinction among 'user', 'group' and 'world';
524 # replicate the 'user' bits.
525 user = expected >> 6
526 expected = user * (1 + 8 + 64)
527 self.assertEqual(mode, expected)
Guido van Rossum0e548712002-08-09 16:14:33 +0000528 finally:
529 os.rmdir(dir)
530
531test_classes.append(test_mkdtemp)
532
533
534class test_mktemp(TC):
535 """Test mktemp()."""
536
537 # For safety, all use of mktemp must occur in a private directory.
538 # We must also suppress the RuntimeWarning it generates.
539 def setUp(self):
540 self.dir = tempfile.mkdtemp()
Brett Cannone1adece2010-03-20 22:19:55 +0000541 super().setUp()
Guido van Rossum0e548712002-08-09 16:14:33 +0000542
543 def tearDown(self):
544 if self.dir:
545 os.rmdir(self.dir)
546 self.dir = None
Brett Cannone1adece2010-03-20 22:19:55 +0000547 super().tearDown()
Guido van Rossum0e548712002-08-09 16:14:33 +0000548
549 class mktemped:
550 _unlink = os.unlink
551 _bflags = tempfile._bin_openflags
552
553 def __init__(self, dir, pre, suf):
554 self.name = tempfile.mktemp(dir=dir, prefix=pre, suffix=suf)
555 # Create the file. This will raise an exception if it's
556 # mysteriously appeared in the meanwhile.
Guido van Rossumcd16bf62007-06-13 18:07:49 +0000557 os.close(os.open(self.name, self._bflags, 0o600))
Guido van Rossum0e548712002-08-09 16:14:33 +0000558
559 def __del__(self):
560 self._unlink(self.name)
561
562 def do_create(self, pre="", suf=""):
563 try:
564 file = self.mktemped(self.dir, pre, suf)
565 except:
566 self.failOnException("mktemp")
567
568 self.nameCheck(file.name, self.dir, pre, suf)
569 return file
570
571 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000572 # mktemp can choose usable file names
Guido van Rossum0e548712002-08-09 16:14:33 +0000573 self.do_create()
574 self.do_create(pre="a")
575 self.do_create(suf="b")
576 self.do_create(pre="a", suf="b")
577 self.do_create(pre="aa", suf=".txt")
578
579 def test_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000580 # mktemp can choose many usable file names (stochastic)
Guido van Rossum805365e2007-05-07 22:24:25 +0000581 extant = list(range(TEST_FILES))
Guido van Rossum0e548712002-08-09 16:14:33 +0000582 for i in extant:
583 extant[i] = self.do_create(pre="aa")
584
Fred Drake8bec4832002-11-22 20:13:43 +0000585## def test_warning(self):
586## # mktemp issues a warning when used
587## warnings.filterwarnings("error",
588## category=RuntimeWarning,
589## message="mktemp")
590## self.assertRaises(RuntimeWarning,
591## tempfile.mktemp, dir=self.dir)
Guido van Rossum0e548712002-08-09 16:14:33 +0000592
593test_classes.append(test_mktemp)
594
595
596# We test _TemporaryFileWrapper by testing NamedTemporaryFile.
597
598
599class test_NamedTemporaryFile(TC):
600 """Test NamedTemporaryFile()."""
601
Guido van Rossumd8faa362007-04-27 19:54:29 +0000602 def do_create(self, dir=None, pre="", suf="", delete=True):
Guido van Rossum0e548712002-08-09 16:14:33 +0000603 if dir is None:
604 dir = tempfile.gettempdir()
605 try:
Guido van Rossumd8faa362007-04-27 19:54:29 +0000606 file = tempfile.NamedTemporaryFile(dir=dir, prefix=pre, suffix=suf,
607 delete=delete)
Guido van Rossum0e548712002-08-09 16:14:33 +0000608 except:
609 self.failOnException("NamedTemporaryFile")
610
611 self.nameCheck(file.name, dir, pre, suf)
612 return file
613
614
615 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000616 # NamedTemporaryFile can create files
Guido van Rossum0e548712002-08-09 16:14:33 +0000617 self.do_create()
618 self.do_create(pre="a")
619 self.do_create(suf="b")
620 self.do_create(pre="a", suf="b")
621 self.do_create(pre="aa", suf=".txt")
622
623 def test_creates_named(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000624 # NamedTemporaryFile creates files with names
Guido van Rossum0e548712002-08-09 16:14:33 +0000625 f = tempfile.NamedTemporaryFile()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000626 self.assertTrue(os.path.exists(f.name),
Guido van Rossum0e548712002-08-09 16:14:33 +0000627 "NamedTemporaryFile %s does not exist" % f.name)
628
629 def test_del_on_close(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000630 # A NamedTemporaryFile is deleted when closed
Guido van Rossum0e548712002-08-09 16:14:33 +0000631 dir = tempfile.mkdtemp()
632 try:
633 f = tempfile.NamedTemporaryFile(dir=dir)
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000634 f.write(b'blat')
Guido van Rossum0e548712002-08-09 16:14:33 +0000635 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000636 self.assertFalse(os.path.exists(f.name),
Guido van Rossum0e548712002-08-09 16:14:33 +0000637 "NamedTemporaryFile %s exists after close" % f.name)
638 finally:
639 os.rmdir(dir)
640
Guido van Rossumd8faa362007-04-27 19:54:29 +0000641 def test_dis_del_on_close(self):
642 # Tests that delete-on-close can be disabled
643 dir = tempfile.mkdtemp()
644 tmp = None
645 try:
646 f = tempfile.NamedTemporaryFile(dir=dir, delete=False)
647 tmp = f.name
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000648 f.write(b'blat')
Guido van Rossumd8faa362007-04-27 19:54:29 +0000649 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000650 self.assertTrue(os.path.exists(f.name),
Guido van Rossumd8faa362007-04-27 19:54:29 +0000651 "NamedTemporaryFile %s missing after close" % f.name)
652 finally:
653 if tmp is not None:
654 os.unlink(tmp)
655 os.rmdir(dir)
656
Guido van Rossum0e548712002-08-09 16:14:33 +0000657 def test_multiple_close(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000658 # A NamedTemporaryFile can be closed many times without error
Guido van Rossum0e548712002-08-09 16:14:33 +0000659 f = tempfile.NamedTemporaryFile()
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000660 f.write(b'abc\n')
Guido van Rossum0e548712002-08-09 16:14:33 +0000661 f.close()
662 try:
663 f.close()
664 f.close()
665 except:
666 self.failOnException("close")
667
Christian Heimes3ecfea712008-02-09 20:51:34 +0000668 def test_context_manager(self):
669 # A NamedTemporaryFile can be used as a context manager
670 with tempfile.NamedTemporaryFile() as f:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000671 self.assertTrue(os.path.exists(f.name))
672 self.assertFalse(os.path.exists(f.name))
Christian Heimes3ecfea712008-02-09 20:51:34 +0000673 def use_closed():
674 with f:
675 pass
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000676 self.assertRaises(ValueError, use_closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000677
Guido van Rossum0e548712002-08-09 16:14:33 +0000678 # How to test the mode and bufsize parameters?
679
680test_classes.append(test_NamedTemporaryFile)
681
Guido van Rossumd8faa362007-04-27 19:54:29 +0000682class test_SpooledTemporaryFile(TC):
683 """Test SpooledTemporaryFile()."""
684
685 def do_create(self, max_size=0, dir=None, pre="", suf=""):
686 if dir is None:
687 dir = tempfile.gettempdir()
688 try:
689 file = tempfile.SpooledTemporaryFile(max_size=max_size, dir=dir, prefix=pre, suffix=suf)
690 except:
691 self.failOnException("SpooledTemporaryFile")
692
693 return file
694
695
696 def test_basic(self):
697 # SpooledTemporaryFile can create files
698 f = self.do_create()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000699 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000700 f = self.do_create(max_size=100, pre="a", suf=".txt")
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000701 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000702
703 def test_del_on_close(self):
704 # A SpooledTemporaryFile is deleted when closed
705 dir = tempfile.mkdtemp()
706 try:
707 f = tempfile.SpooledTemporaryFile(max_size=10, dir=dir)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000708 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +0000709 f.write(b'blat ' * 5)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000710 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000711 filename = f.name
712 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000713 self.assertFalse(isinstance(filename, str) and os.path.exists(filename),
Guido van Rossumd8faa362007-04-27 19:54:29 +0000714 "SpooledTemporaryFile %s exists after close" % filename)
715 finally:
716 os.rmdir(dir)
717
718 def test_rewrite_small(self):
719 # A SpooledTemporaryFile can be written to multiple within the max_size
720 f = self.do_create(max_size=30)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000721 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000722 for i in range(5):
723 f.seek(0, 0)
Guido van Rossum39478e82007-08-27 17:23:59 +0000724 f.write(b'x' * 20)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000725 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000726
727 def test_write_sequential(self):
728 # A SpooledTemporaryFile should hold exactly max_size bytes, and roll
729 # over afterward
730 f = self.do_create(max_size=30)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000731 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +0000732 f.write(b'x' * 20)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000733 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +0000734 f.write(b'x' * 10)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000735 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +0000736 f.write(b'x')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000737 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000738
R David Murrayd89ee792011-03-14 09:55:46 -0400739 def test_writelines(self):
740 # Verify writelines with a SpooledTemporaryFile
741 f = self.do_create()
742 f.writelines((b'x', b'y', b'z'))
743 f.seek(0)
744 buf = f.read()
745 self.assertEqual(buf, b'xyz')
746
747 def test_writelines_sequential(self):
748 # A SpooledTemporaryFile should hold exactly max_size bytes, and roll
749 # over afterward
750 f = self.do_create(max_size=35)
751 f.writelines((b'x' * 20, b'x' * 10, b'x' * 5))
752 self.assertFalse(f._rolled)
753 f.write(b'x')
754 self.assertTrue(f._rolled)
755
Guido van Rossumd8faa362007-04-27 19:54:29 +0000756 def test_sparse(self):
757 # A SpooledTemporaryFile that is written late in the file will extend
758 # when that occurs
759 f = self.do_create(max_size=30)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000760 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000761 f.seek(100, 0)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000762 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +0000763 f.write(b'x')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000764 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000765
766 def test_fileno(self):
767 # A SpooledTemporaryFile should roll over to a real file on fileno()
768 f = self.do_create(max_size=30)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000769 self.assertFalse(f._rolled)
770 self.assertTrue(f.fileno() > 0)
771 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000772
Christian Heimes3ecfea712008-02-09 20:51:34 +0000773 def test_multiple_close_before_rollover(self):
Guido van Rossumd8faa362007-04-27 19:54:29 +0000774 # A SpooledTemporaryFile can be closed many times without error
775 f = tempfile.SpooledTemporaryFile()
Guido van Rossum39478e82007-08-27 17:23:59 +0000776 f.write(b'abc\n')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000777 self.assertFalse(f._rolled)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000778 f.close()
779 try:
780 f.close()
781 f.close()
782 except:
783 self.failOnException("close")
784
785 def test_multiple_close_after_rollover(self):
786 # A SpooledTemporaryFile can be closed many times without error
787 f = tempfile.SpooledTemporaryFile(max_size=1)
788 f.write(b'abc\n')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000789 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000790 f.close()
791 try:
792 f.close()
793 f.close()
794 except:
795 self.failOnException("close")
796
797 def test_bound_methods(self):
798 # It should be OK to steal a bound method from a SpooledTemporaryFile
799 # and use it independently; when the file rolls over, those bound
800 # methods should continue to function
801 f = self.do_create(max_size=30)
802 read = f.read
803 write = f.write
804 seek = f.seek
805
Guido van Rossum39478e82007-08-27 17:23:59 +0000806 write(b"a" * 35)
807 write(b"b" * 35)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000808 seek(0, 0)
Guido van Rossum9a634702007-07-09 10:24:45 +0000809 self.assertEqual(read(70), b'a'*35 + b'b'*35)
810
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +0200811 def test_properties(self):
812 f = tempfile.SpooledTemporaryFile(max_size=10)
813 f.write(b'x' * 10)
814 self.assertFalse(f._rolled)
815 self.assertEqual(f.mode, 'w+b')
816 self.assertIsNone(f.name)
817 with self.assertRaises(AttributeError):
818 f.newlines
819 with self.assertRaises(AttributeError):
820 f.encoding
821
822 f.write(b'x')
823 self.assertTrue(f._rolled)
824 self.assertEqual(f.mode, 'rb+')
825 self.assertIsNotNone(f.name)
826 with self.assertRaises(AttributeError):
827 f.newlines
828 with self.assertRaises(AttributeError):
829 f.encoding
830
Guido van Rossum9a634702007-07-09 10:24:45 +0000831 def test_text_mode(self):
832 # Creating a SpooledTemporaryFile with a text mode should produce
833 # a file object reading and writing (Unicode) text strings.
834 f = tempfile.SpooledTemporaryFile(mode='w+', max_size=10)
835 f.write("abc\n")
836 f.seek(0)
837 self.assertEqual(f.read(), "abc\n")
838 f.write("def\n")
839 f.seek(0)
840 self.assertEqual(f.read(), "abc\ndef\n")
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +0200841 self.assertFalse(f._rolled)
842 self.assertEqual(f.mode, 'w+')
843 self.assertIsNone(f.name)
844 self.assertIsNone(f.newlines)
845 self.assertIsNone(f.encoding)
846
Guido van Rossum9a634702007-07-09 10:24:45 +0000847 f.write("xyzzy\n")
848 f.seek(0)
849 self.assertEqual(f.read(), "abc\ndef\nxyzzy\n")
Amaury Forgeot d'Arc7d0bddd2009-11-30 00:08:56 +0000850 # Check that Ctrl+Z doesn't truncate the file
851 f.write("foo\x1abar\n")
852 f.seek(0)
853 self.assertEqual(f.read(), "abc\ndef\nxyzzy\nfoo\x1abar\n")
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +0200854 self.assertTrue(f._rolled)
855 self.assertEqual(f.mode, 'w+')
856 self.assertIsNotNone(f.name)
Serhiy Storchaka497cee42013-02-10 14:43:46 +0200857 self.assertEqual(f.newlines, os.linesep)
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +0200858 self.assertIsNotNone(f.encoding)
Guido van Rossum9a634702007-07-09 10:24:45 +0000859
Guido van Rossumf0c74162007-08-28 03:29:45 +0000860 def test_text_newline_and_encoding(self):
861 f = tempfile.SpooledTemporaryFile(mode='w+', max_size=10,
862 newline='', encoding='utf-8')
863 f.write("\u039B\r\n")
864 f.seek(0)
865 self.assertEqual(f.read(), "\u039B\r\n")
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000866 self.assertFalse(f._rolled)
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +0200867 self.assertEqual(f.mode, 'w+')
868 self.assertIsNone(f.name)
869 self.assertIsNone(f.newlines)
870 self.assertIsNone(f.encoding)
Guido van Rossumf0c74162007-08-28 03:29:45 +0000871
872 f.write("\u039B" * 20 + "\r\n")
873 f.seek(0)
874 self.assertEqual(f.read(), "\u039B\r\n" + ("\u039B" * 20) + "\r\n")
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000875 self.assertTrue(f._rolled)
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +0200876 self.assertEqual(f.mode, 'w+')
877 self.assertIsNotNone(f.name)
878 self.assertIsNotNone(f.newlines)
879 self.assertEqual(f.encoding, 'utf-8')
Guido van Rossumd8faa362007-04-27 19:54:29 +0000880
Christian Heimes3ecfea712008-02-09 20:51:34 +0000881 def test_context_manager_before_rollover(self):
882 # A SpooledTemporaryFile can be used as a context manager
883 with tempfile.SpooledTemporaryFile(max_size=1) as f:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000884 self.assertFalse(f._rolled)
885 self.assertFalse(f.closed)
886 self.assertTrue(f.closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000887 def use_closed():
888 with f:
889 pass
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000890 self.assertRaises(ValueError, use_closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000891
892 def test_context_manager_during_rollover(self):
893 # A SpooledTemporaryFile can be used as a context manager
894 with tempfile.SpooledTemporaryFile(max_size=1) as f:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000895 self.assertFalse(f._rolled)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000896 f.write(b'abc\n')
897 f.flush()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000898 self.assertTrue(f._rolled)
899 self.assertFalse(f.closed)
900 self.assertTrue(f.closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000901 def use_closed():
902 with f:
903 pass
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000904 self.assertRaises(ValueError, use_closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000905
906 def test_context_manager_after_rollover(self):
907 # A SpooledTemporaryFile can be used as a context manager
908 f = tempfile.SpooledTemporaryFile(max_size=1)
909 f.write(b'abc\n')
910 f.flush()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000911 self.assertTrue(f._rolled)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000912 with f:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000913 self.assertFalse(f.closed)
914 self.assertTrue(f.closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000915 def use_closed():
916 with f:
917 pass
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000918 self.assertRaises(ValueError, use_closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000919
920
Guido van Rossumd8faa362007-04-27 19:54:29 +0000921test_classes.append(test_SpooledTemporaryFile)
922
Guido van Rossum0e548712002-08-09 16:14:33 +0000923
924class test_TemporaryFile(TC):
925 """Test TemporaryFile()."""
926
927 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000928 # TemporaryFile can create files
Guido van Rossum0e548712002-08-09 16:14:33 +0000929 # No point in testing the name params - the file has no name.
930 try:
931 tempfile.TemporaryFile()
932 except:
933 self.failOnException("TemporaryFile")
934
935 def test_has_no_name(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000936 # TemporaryFile creates files with no names (on this system)
Guido van Rossum0e548712002-08-09 16:14:33 +0000937 dir = tempfile.mkdtemp()
938 f = tempfile.TemporaryFile(dir=dir)
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000939 f.write(b'blat')
Guido van Rossum0e548712002-08-09 16:14:33 +0000940
941 # Sneaky: because this file has no name, it should not prevent
942 # us from removing the directory it was created in.
943 try:
944 os.rmdir(dir)
945 except:
946 ei = sys.exc_info()
947 # cleanup
948 f.close()
949 os.rmdir(dir)
950 self.failOnException("rmdir", ei)
951
952 def test_multiple_close(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000953 # A TemporaryFile can be closed many times without error
Guido van Rossum0e548712002-08-09 16:14:33 +0000954 f = tempfile.TemporaryFile()
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000955 f.write(b'abc\n')
Guido van Rossum0e548712002-08-09 16:14:33 +0000956 f.close()
957 try:
958 f.close()
959 f.close()
960 except:
961 self.failOnException("close")
962
963 # How to test the mode and bufsize parameters?
Guido van Rossumf0c74162007-08-28 03:29:45 +0000964 def test_mode_and_encoding(self):
965
966 def roundtrip(input, *args, **kwargs):
967 with tempfile.TemporaryFile(*args, **kwargs) as fileobj:
968 fileobj.write(input)
969 fileobj.seek(0)
Ezio Melottib3aedd42010-11-20 19:04:17 +0000970 self.assertEqual(input, fileobj.read())
Guido van Rossumf0c74162007-08-28 03:29:45 +0000971
972 roundtrip(b"1234", "w+b")
973 roundtrip("abdc\n", "w+")
974 roundtrip("\u039B", "w+", encoding="utf-16")
975 roundtrip("foo\r\n", "w+", newline="")
Guido van Rossum0e548712002-08-09 16:14:33 +0000976
Guido van Rossum0e548712002-08-09 16:14:33 +0000977
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000978if tempfile.NamedTemporaryFile is not tempfile.TemporaryFile:
Guido van Rossum0e548712002-08-09 16:14:33 +0000979 test_classes.append(test_TemporaryFile)
980
Nick Coghlan543af752010-10-24 11:23:25 +0000981
982# Helper for test_del_on_shutdown
983class NulledModules:
984 def __init__(self, *modules):
985 self.refs = [mod.__dict__ for mod in modules]
986 self.contents = [ref.copy() for ref in self.refs]
987
988 def __enter__(self):
989 for d in self.refs:
990 for key in d:
991 d[key] = None
992
993 def __exit__(self, *exc_info):
994 for d, c in zip(self.refs, self.contents):
995 d.clear()
996 d.update(c)
997
998class test_TemporaryDirectory(TC):
999 """Test TemporaryDirectory()."""
1000
1001 def do_create(self, dir=None, pre="", suf="", recurse=1):
1002 if dir is None:
1003 dir = tempfile.gettempdir()
1004 try:
1005 tmp = tempfile.TemporaryDirectory(dir=dir, prefix=pre, suffix=suf)
1006 except:
1007 self.failOnException("TemporaryDirectory")
1008 self.nameCheck(tmp.name, dir, pre, suf)
1009 # Create a subdirectory and some files
1010 if recurse:
1011 self.do_create(tmp.name, pre, suf, recurse-1)
1012 with open(os.path.join(tmp.name, "test.txt"), "wb") as f:
1013 f.write(b"Hello world!")
1014 return tmp
1015
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001016 def test_mkdtemp_failure(self):
1017 # Check no additional exception if mkdtemp fails
1018 # Previously would raise AttributeError instead
Nick Coghlan3c54ea62010-12-13 03:02:43 +00001019 # (noted as part of Issue #10188)
1020 with tempfile.TemporaryDirectory() as nonexistent:
1021 pass
1022 with self.assertRaises(os.error):
1023 tempfile.TemporaryDirectory(dir=nonexistent)
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001024
Nick Coghlan543af752010-10-24 11:23:25 +00001025 def test_explicit_cleanup(self):
1026 # A TemporaryDirectory is deleted when cleaned up
1027 dir = tempfile.mkdtemp()
1028 try:
1029 d = self.do_create(dir=dir)
1030 self.assertTrue(os.path.exists(d.name),
1031 "TemporaryDirectory %s does not exist" % d.name)
1032 d.cleanup()
1033 self.assertFalse(os.path.exists(d.name),
1034 "TemporaryDirectory %s exists after cleanup" % d.name)
1035 finally:
1036 os.rmdir(dir)
1037
Charles-François Natalidef35432011-07-29 18:59:24 +02001038 @support.skip_unless_symlink
1039 def test_cleanup_with_symlink_to_a_directory(self):
1040 # cleanup() should not follow symlinks to directories (issue #12464)
1041 d1 = self.do_create()
1042 d2 = self.do_create()
1043
1044 # Symlink d1/foo -> d2
1045 os.symlink(d2.name, os.path.join(d1.name, "foo"))
1046
1047 # This call to cleanup() should not follow the "foo" symlink
1048 d1.cleanup()
1049
1050 self.assertFalse(os.path.exists(d1.name),
1051 "TemporaryDirectory %s exists after cleanup" % d1.name)
1052 self.assertTrue(os.path.exists(d2.name),
1053 "Directory pointed to by a symlink was deleted")
1054 self.assertEqual(os.listdir(d2.name), ['test.txt'],
1055 "Contents of the directory pointed to by a symlink "
1056 "were deleted")
1057 d2.cleanup()
1058
Nick Coghlan543af752010-10-24 11:23:25 +00001059 @support.cpython_only
1060 def test_del_on_collection(self):
1061 # A TemporaryDirectory is deleted when garbage collected
1062 dir = tempfile.mkdtemp()
1063 try:
1064 d = self.do_create(dir=dir)
1065 name = d.name
1066 del d # Rely on refcounting to invoke __del__
1067 self.assertFalse(os.path.exists(name),
1068 "TemporaryDirectory %s exists after __del__" % name)
1069 finally:
1070 os.rmdir(dir)
1071
1072 @unittest.expectedFailure # See issue #10188
1073 def test_del_on_shutdown(self):
1074 # A TemporaryDirectory may be cleaned up during shutdown
1075 # Make sure it works with the relevant modules nulled out
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001076 with self.do_create() as dir:
Nick Coghlan543af752010-10-24 11:23:25 +00001077 d = self.do_create(dir=dir)
1078 # Mimic the nulling out of modules that
1079 # occurs during system shutdown
1080 modules = [os, os.path]
1081 if has_stat:
1082 modules.append(stat)
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001083 # Currently broken, so suppress the warning
1084 # that is otherwise emitted on stdout
1085 with support.captured_stderr() as err:
1086 with NulledModules(*modules):
1087 d.cleanup()
1088 # Currently broken, so stop spurious exception by
1089 # indicating the object has already been closed
1090 d._closed = True
1091 # And this assert will fail, as expected by the
1092 # unittest decorator...
Nick Coghlan543af752010-10-24 11:23:25 +00001093 self.assertFalse(os.path.exists(d.name),
1094 "TemporaryDirectory %s exists after cleanup" % d.name)
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001095
1096 def test_warnings_on_cleanup(self):
1097 # Two kinds of warning on shutdown
1098 # Issue 10888: may write to stderr if modules are nulled out
1099 # ResourceWarning will be triggered by __del__
1100 with self.do_create() as dir:
Nick Coghlane98e8a32010-12-13 16:32:51 +00001101 if os.sep != '\\':
1102 # Embed a backslash in order to make sure string escaping
1103 # in the displayed error message is dealt with correctly
1104 suffix = '\\check_backslash_handling'
1105 else:
1106 suffix = ''
1107 d = self.do_create(dir=dir, suf=suffix)
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001108
1109 #Check for the Issue 10888 message
1110 modules = [os, os.path]
1111 if has_stat:
1112 modules.append(stat)
1113 with support.captured_stderr() as err:
1114 with NulledModules(*modules):
1115 d.cleanup()
Nick Coghlane98e8a32010-12-13 16:32:51 +00001116 message = err.getvalue().replace('\\\\', '\\')
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001117 self.assertIn("while cleaning up", message)
1118 self.assertIn(d.name, message)
1119
1120 # Check for the resource warning
1121 with support.check_warnings(('Implicitly', ResourceWarning), quiet=False):
1122 warnings.filterwarnings("always", category=ResourceWarning)
1123 d.__del__()
1124 self.assertFalse(os.path.exists(d.name),
1125 "TemporaryDirectory %s exists after __del__" % d.name)
Nick Coghlan543af752010-10-24 11:23:25 +00001126
1127 def test_multiple_close(self):
1128 # Can be cleaned-up many times without error
1129 d = self.do_create()
1130 d.cleanup()
1131 try:
1132 d.cleanup()
1133 d.cleanup()
1134 except:
1135 self.failOnException("cleanup")
1136
1137 def test_context_manager(self):
1138 # Can be used as a context manager
1139 d = self.do_create()
1140 with d as name:
1141 self.assertTrue(os.path.exists(name))
1142 self.assertEqual(name, d.name)
1143 self.assertFalse(os.path.exists(name))
1144
1145
1146test_classes.append(test_TemporaryDirectory)
1147
Guido van Rossum0e548712002-08-09 16:14:33 +00001148def test_main():
Benjamin Petersonee8712c2008-05-20 21:35:26 +00001149 support.run_unittest(*test_classes)
Guido van Rossum0e548712002-08-09 16:14:33 +00001150
1151if __name__ == "__main__":
1152 test_main()