blob: cf8b35ac80e0f2b93dd9b161077a589f61cf1101 [file] [log] [blame]
Guido van Rossum0e548712002-08-09 16:14:33 +00001# tempfile.py unit tests.
Tim Petersc57a2852001-10-29 21:46:08 +00002import tempfile
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +02003import errno
4import io
Guido van Rossum0e548712002-08-09 16:14:33 +00005import os
Antoine Pitrou4558bad2011-11-25 21:28:15 +01006import signal
Guido van Rossum0e548712002-08-09 16:14:33 +00007import sys
8import re
Guido van Rossum0e548712002-08-09 16:14:33 +00009import warnings
Tim Petersc57a2852001-10-29 21:46:08 +000010
Guido van Rossum0e548712002-08-09 16:14:33 +000011import unittest
Benjamin Petersonee8712c2008-05-20 21:35:26 +000012from test import support
Guido van Rossum0e548712002-08-09 16:14:33 +000013
Fred Drake7633d232002-10-17 22:09:03 +000014
Guido van Rossum0e548712002-08-09 16:14:33 +000015if hasattr(os, 'stat'):
16 import stat
17 has_stat = 1
18else:
19 has_stat = 0
20
21has_textmode = (tempfile._text_openflags != tempfile._bin_openflags)
Guido van Rossum78741062002-08-17 11:41:01 +000022has_spawnl = hasattr(os, 'spawnl')
Guido van Rossum0e548712002-08-09 16:14:33 +000023
Neal Norwitz68ee0122002-08-16 19:28:59 +000024# TEST_FILES may need to be tweaked for systems depending on the maximum
25# number of files that can be opened at one time (see ulimit -n)
Ronald Oussoren94f25282010-05-05 19:11:21 +000026if sys.platform in ('openbsd3', 'openbsd4'):
Martin v. Löwis99968282004-09-15 06:02:54 +000027 TEST_FILES = 48
Jack Jansence921472003-01-08 16:30:34 +000028else:
29 TEST_FILES = 100
Neal Norwitz68ee0122002-08-16 19:28:59 +000030
Guido van Rossum0e548712002-08-09 16:14:33 +000031# This is organized as one test for each chunk of code in tempfile.py,
32# in order of their appearance in the file. Testing which requires
33# threads is not done here.
34
35# Common functionality.
36class TC(unittest.TestCase):
37
38 str_check = re.compile(r"[a-zA-Z0-9_-]{6}$")
39
Brett Cannone1adece2010-03-20 22:19:55 +000040 def setUp(self):
41 self._warnings_manager = support.check_warnings()
42 self._warnings_manager.__enter__()
43 warnings.filterwarnings("ignore", category=RuntimeWarning,
44 message="mktemp", module=__name__)
45
46 def tearDown(self):
47 self._warnings_manager.__exit__(None, None, None)
48
49
Guido van Rossum0e548712002-08-09 16:14:33 +000050 def failOnException(self, what, ei=None):
51 if ei is None:
52 ei = sys.exc_info()
53 self.fail("%s raised %s: %s" % (what, ei[0], ei[1]))
54
55 def nameCheck(self, name, dir, pre, suf):
56 (ndir, nbase) = os.path.split(name)
57 npre = nbase[:len(pre)]
58 nsuf = nbase[len(nbase)-len(suf):]
59
Martin v. Löwisd6625482003-10-12 17:37:01 +000060 # check for equality of the absolute paths!
61 self.assertEqual(os.path.abspath(ndir), os.path.abspath(dir),
Guido van Rossum0e548712002-08-09 16:14:33 +000062 "file '%s' not in directory '%s'" % (name, dir))
63 self.assertEqual(npre, pre,
64 "file '%s' does not begin with '%s'" % (nbase, pre))
65 self.assertEqual(nsuf, suf,
66 "file '%s' does not end with '%s'" % (nbase, suf))
67
68 nbase = nbase[len(pre):len(nbase)-len(suf)]
Benjamin Petersonc9c0f202009-06-30 23:06:06 +000069 self.assertTrue(self.str_check.match(nbase),
Guido van Rossum0e548712002-08-09 16:14:33 +000070 "random string '%s' does not match /^[a-zA-Z0-9_-]{6}$/"
71 % nbase)
72
73test_classes = []
74
75class test_exports(TC):
76 def test_exports(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +000077 # There are no surprising symbols in the tempfile module
Guido van Rossum0e548712002-08-09 16:14:33 +000078 dict = tempfile.__dict__
79
80 expected = {
81 "NamedTemporaryFile" : 1,
82 "TemporaryFile" : 1,
83 "mkstemp" : 1,
84 "mkdtemp" : 1,
85 "mktemp" : 1,
86 "TMP_MAX" : 1,
87 "gettempprefix" : 1,
88 "gettempdir" : 1,
89 "tempdir" : 1,
Guido van Rossumd8faa362007-04-27 19:54:29 +000090 "template" : 1,
Nick Coghlan543af752010-10-24 11:23:25 +000091 "SpooledTemporaryFile" : 1,
92 "TemporaryDirectory" : 1,
Guido van Rossum0e548712002-08-09 16:14:33 +000093 }
94
95 unexp = []
96 for key in dict:
97 if key[0] != '_' and key not in expected:
98 unexp.append(key)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +000099 self.assertTrue(len(unexp) == 0,
Guido van Rossum0e548712002-08-09 16:14:33 +0000100 "unexpected keys: %s" % unexp)
101
102test_classes.append(test_exports)
103
104
Guido van Rossum0e548712002-08-09 16:14:33 +0000105class test__RandomNameSequence(TC):
106 """Test the internal iterator object _RandomNameSequence."""
107
108 def setUp(self):
109 self.r = tempfile._RandomNameSequence()
Brett Cannone1adece2010-03-20 22:19:55 +0000110 super().setUp()
Guido van Rossum0e548712002-08-09 16:14:33 +0000111
112 def test_get_six_char_str(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000113 # _RandomNameSequence returns a six-character string
Georg Brandla18af4e2007-04-21 15:47:16 +0000114 s = next(self.r)
Guido van Rossum0e548712002-08-09 16:14:33 +0000115 self.nameCheck(s, '', '', '')
116
117 def test_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000118 # _RandomNameSequence returns no duplicate strings (stochastic)
Guido van Rossum0e548712002-08-09 16:14:33 +0000119
120 dict = {}
121 r = self.r
Guido van Rossum805365e2007-05-07 22:24:25 +0000122 for i in range(TEST_FILES):
Georg Brandla18af4e2007-04-21 15:47:16 +0000123 s = next(r)
Guido van Rossum0e548712002-08-09 16:14:33 +0000124 self.nameCheck(s, '', '', '')
Ezio Melottib58e0bd2010-01-23 15:40:09 +0000125 self.assertNotIn(s, dict)
Guido van Rossum0e548712002-08-09 16:14:33 +0000126 dict[s] = 1
127
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000128 def supports_iter(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000129 # _RandomNameSequence supports the iterator protocol
Guido van Rossum0e548712002-08-09 16:14:33 +0000130
131 i = 0
132 r = self.r
133 try:
134 for s in r:
135 i += 1
136 if i == 20:
137 break
138 except:
Georg Brandl89fad142010-03-14 10:23:39 +0000139 self.failOnException("iteration")
Guido van Rossum0e548712002-08-09 16:14:33 +0000140
Antoine Pitrou4558bad2011-11-25 21:28:15 +0100141 @unittest.skipUnless(hasattr(os, 'fork'),
142 "os.fork is required for this test")
143 def test_process_awareness(self):
144 # ensure that the random source differs between
145 # child and parent.
146 read_fd, write_fd = os.pipe()
147 pid = None
148 try:
149 pid = os.fork()
150 if not pid:
151 os.close(read_fd)
152 os.write(write_fd, next(self.r).encode("ascii"))
153 os.close(write_fd)
154 # bypass the normal exit handlers- leave those to
155 # the parent.
156 os._exit(0)
157 parent_value = next(self.r)
158 child_value = os.read(read_fd, len(parent_value)).decode("ascii")
159 finally:
160 if pid:
161 # best effort to ensure the process can't bleed out
162 # via any bugs above
163 try:
164 os.kill(pid, signal.SIGKILL)
165 except EnvironmentError:
166 pass
167 os.close(read_fd)
168 os.close(write_fd)
169 self.assertNotEqual(child_value, parent_value)
170
171
Guido van Rossum0e548712002-08-09 16:14:33 +0000172test_classes.append(test__RandomNameSequence)
173
174
175class test__candidate_tempdir_list(TC):
176 """Test the internal function _candidate_tempdir_list."""
177
178 def test_nonempty_list(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000179 # _candidate_tempdir_list returns a nonempty list of strings
Guido van Rossum0e548712002-08-09 16:14:33 +0000180
181 cand = tempfile._candidate_tempdir_list()
182
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000183 self.assertFalse(len(cand) == 0)
Guido van Rossum0e548712002-08-09 16:14:33 +0000184 for c in cand:
Ezio Melottie9615932010-01-24 19:26:24 +0000185 self.assertIsInstance(c, str)
Guido van Rossum0e548712002-08-09 16:14:33 +0000186
187 def test_wanted_dirs(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000188 # _candidate_tempdir_list contains the expected directories
Guido van Rossum0e548712002-08-09 16:14:33 +0000189
190 # Make sure the interesting environment variables are all set.
Hirokazu Yamamoto71959632009-04-27 01:44:28 +0000191 with support.EnvironmentVarGuard() as env:
Guido van Rossum0e548712002-08-09 16:14:33 +0000192 for envname in 'TMPDIR', 'TEMP', 'TMP':
193 dirname = os.getenv(envname)
194 if not dirname:
Walter Dörwald155374d2009-05-01 19:58:58 +0000195 env[envname] = os.path.abspath(envname)
Guido van Rossum0e548712002-08-09 16:14:33 +0000196
197 cand = tempfile._candidate_tempdir_list()
198
199 for envname in 'TMPDIR', 'TEMP', 'TMP':
200 dirname = os.getenv(envname)
201 if not dirname: raise ValueError
Benjamin Peterson577473f2010-01-19 00:09:57 +0000202 self.assertIn(dirname, cand)
Guido van Rossum0e548712002-08-09 16:14:33 +0000203
204 try:
205 dirname = os.getcwd()
206 except (AttributeError, os.error):
207 dirname = os.curdir
208
Benjamin Peterson577473f2010-01-19 00:09:57 +0000209 self.assertIn(dirname, cand)
Guido van Rossum0e548712002-08-09 16:14:33 +0000210
211 # Not practical to try to verify the presence of OS-specific
212 # paths in this list.
Guido van Rossum0e548712002-08-09 16:14:33 +0000213
214test_classes.append(test__candidate_tempdir_list)
215
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200216# We test _get_default_tempdir some more by testing gettempdir.
Guido van Rossum0e548712002-08-09 16:14:33 +0000217
Serhiy Storchakaf6b361e2013-02-13 00:35:30 +0200218class TestGetDefaultTempdir(TC):
219 """Test _get_default_tempdir()."""
220
221 def test_no_files_left_behind(self):
222 # use a private empty directory
223 with tempfile.TemporaryDirectory() as our_temp_directory:
224 # force _get_default_tempdir() to consider our empty directory
225 def our_candidate_list():
226 return [our_temp_directory]
227
228 with support.swap_attr(tempfile, "_candidate_tempdir_list",
229 our_candidate_list):
230 # verify our directory is empty after _get_default_tempdir()
231 tempfile._get_default_tempdir()
232 self.assertEqual(os.listdir(our_temp_directory), [])
233
234 def raise_OSError(*args, **kwargs):
235 raise OSError(-1)
236
237 with support.swap_attr(io, "open", raise_OSError):
238 # test again with failing io.open()
239 with self.assertRaises(IOError) as cm:
240 tempfile._get_default_tempdir()
241 self.assertEqual(cm.exception.args[0], errno.ENOENT)
242 self.assertEqual(os.listdir(our_temp_directory), [])
243
244 open = io.open
245 def bad_writer(*args, **kwargs):
246 fp = open(*args, **kwargs)
247 fp.write = raise_OSError
248 return fp
249
250 with support.swap_attr(io, "open", bad_writer):
251 # test again with failing write()
252 with self.assertRaises(IOError) as cm:
253 tempfile._get_default_tempdir()
254 self.assertEqual(cm.exception.errno, errno.ENOENT)
255 self.assertEqual(os.listdir(our_temp_directory), [])
256
257test_classes.append(TestGetDefaultTempdir)
Guido van Rossum0e548712002-08-09 16:14:33 +0000258
259
260class test__get_candidate_names(TC):
261 """Test the internal function _get_candidate_names."""
262
263 def test_retval(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000264 # _get_candidate_names returns a _RandomNameSequence object
Guido van Rossum0e548712002-08-09 16:14:33 +0000265 obj = tempfile._get_candidate_names()
Ezio Melottie9615932010-01-24 19:26:24 +0000266 self.assertIsInstance(obj, tempfile._RandomNameSequence)
Guido van Rossum0e548712002-08-09 16:14:33 +0000267
268 def test_same_thing(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000269 # _get_candidate_names always returns the same object
Guido van Rossum0e548712002-08-09 16:14:33 +0000270 a = tempfile._get_candidate_names()
271 b = tempfile._get_candidate_names()
272
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000273 self.assertTrue(a is b)
Guido van Rossum0e548712002-08-09 16:14:33 +0000274
275test_classes.append(test__get_candidate_names)
276
277
278class test__mkstemp_inner(TC):
279 """Test the internal function _mkstemp_inner."""
280
281 class mkstemped:
282 _bflags = tempfile._bin_openflags
283 _tflags = tempfile._text_openflags
284 _close = os.close
285 _unlink = os.unlink
286
287 def __init__(self, dir, pre, suf, bin):
288 if bin: flags = self._bflags
289 else: flags = self._tflags
290
291 (self.fd, self.name) = tempfile._mkstemp_inner(dir, pre, suf, flags)
292
293 def write(self, str):
294 os.write(self.fd, str)
295
296 def __del__(self):
297 self._close(self.fd)
298 self._unlink(self.name)
Tim Petersa0d55de2002-08-09 18:01:01 +0000299
Guido van Rossum0e548712002-08-09 16:14:33 +0000300 def do_create(self, dir=None, pre="", suf="", bin=1):
301 if dir is None:
302 dir = tempfile.gettempdir()
303 try:
304 file = self.mkstemped(dir, pre, suf, bin)
305 except:
306 self.failOnException("_mkstemp_inner")
307
308 self.nameCheck(file.name, dir, pre, suf)
309 return file
310
311 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000312 # _mkstemp_inner can create files
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000313 self.do_create().write(b"blat")
314 self.do_create(pre="a").write(b"blat")
315 self.do_create(suf="b").write(b"blat")
316 self.do_create(pre="a", suf="b").write(b"blat")
317 self.do_create(pre="aa", suf=".txt").write(b"blat")
Guido van Rossum0e548712002-08-09 16:14:33 +0000318
319 def test_basic_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000320 # _mkstemp_inner can create many files (stochastic)
Guido van Rossum805365e2007-05-07 22:24:25 +0000321 extant = list(range(TEST_FILES))
Guido van Rossum0e548712002-08-09 16:14:33 +0000322 for i in extant:
323 extant[i] = self.do_create(pre="aa")
324
325 def test_choose_directory(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000326 # _mkstemp_inner can create files in a user-selected directory
Guido van Rossum0e548712002-08-09 16:14:33 +0000327 dir = tempfile.mkdtemp()
328 try:
Antoine Pitrou9cadb1b2008-09-15 23:02:56 +0000329 self.do_create(dir=dir).write(b"blat")
Guido van Rossum0e548712002-08-09 16:14:33 +0000330 finally:
331 os.rmdir(dir)
332
333 def test_file_mode(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000334 # _mkstemp_inner creates files with the proper mode
Guido van Rossum0e548712002-08-09 16:14:33 +0000335 if not has_stat:
Benjamin Petersone549ead2009-03-28 21:42:05 +0000336 return # ugh, can't use SkipTest.
Guido van Rossum0e548712002-08-09 16:14:33 +0000337
338 file = self.do_create()
339 mode = stat.S_IMODE(os.stat(file.name).st_mode)
Guido van Rossumcd16bf62007-06-13 18:07:49 +0000340 expected = 0o600
Ronald Oussoren94f25282010-05-05 19:11:21 +0000341 if sys.platform in ('win32', 'os2emx'):
Tim Petersca3ac7f2002-08-09 18:13:51 +0000342 # There's no distinction among 'user', 'group' and 'world';
343 # replicate the 'user' bits.
344 user = expected >> 6
345 expected = user * (1 + 8 + 64)
346 self.assertEqual(mode, expected)
Guido van Rossum0e548712002-08-09 16:14:33 +0000347
348 def test_noinherit(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000349 # _mkstemp_inner file handles are not inherited by child processes
Guido van Rossum78741062002-08-17 11:41:01 +0000350 if not has_spawnl:
Benjamin Petersone549ead2009-03-28 21:42:05 +0000351 return # ugh, can't use SkipTest.
Guido van Rossum0e548712002-08-09 16:14:33 +0000352
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000353 if support.verbose:
Guido van Rossum78741062002-08-17 11:41:01 +0000354 v="v"
355 else:
356 v="q"
357
Guido van Rossum0e548712002-08-09 16:14:33 +0000358 file = self.do_create()
Guido van Rossum78741062002-08-17 11:41:01 +0000359 fd = "%d" % file.fd
360
361 try:
362 me = __file__
363 except NameError:
364 me = sys.argv[0]
Guido van Rossum0e548712002-08-09 16:14:33 +0000365
366 # We have to exec something, so that FD_CLOEXEC will take
Guido van Rossum78741062002-08-17 11:41:01 +0000367 # effect. The core of this test is therefore in
368 # tf_inherit_check.py, which see.
369 tester = os.path.join(os.path.dirname(os.path.abspath(me)),
370 "tf_inherit_check.py")
Guido van Rossum0e548712002-08-09 16:14:33 +0000371
Martin v. Löwisd4210bc2003-10-23 15:55:28 +0000372 # On Windows a spawn* /path/ with embedded spaces shouldn't be quoted,
373 # but an arg with embedded spaces should be decorated with double
374 # quotes on each end
Christian Heimes895627f2007-12-08 17:28:33 +0000375 if sys.platform in ('win32',):
Martin v. Löwisd4210bc2003-10-23 15:55:28 +0000376 decorated = '"%s"' % sys.executable
377 tester = '"%s"' % tester
378 else:
379 decorated = sys.executable
380
381 retval = os.spawnl(os.P_WAIT, sys.executable, decorated, tester, v, fd)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000382 self.assertFalse(retval < 0,
Guido van Rossum78741062002-08-17 11:41:01 +0000383 "child process caught fatal signal %d" % -retval)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000384 self.assertFalse(retval > 0, "child process reports failure %d"%retval)
Guido van Rossum0e548712002-08-09 16:14:33 +0000385
386 def test_textmode(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000387 # _mkstemp_inner can create files in text mode
Guido van Rossum0e548712002-08-09 16:14:33 +0000388 if not has_textmode:
Benjamin Petersone549ead2009-03-28 21:42:05 +0000389 return # ugh, can't use SkipTest.
Guido van Rossum0e548712002-08-09 16:14:33 +0000390
Amaury Forgeot d'Arc7d0bddd2009-11-30 00:08:56 +0000391 # A text file is truncated at the first Ctrl+Z byte
392 f = self.do_create(bin=0)
393 f.write(b"blat\x1a")
394 f.write(b"extra\n")
395 os.lseek(f.fd, 0, os.SEEK_SET)
Ezio Melottib3aedd42010-11-20 19:04:17 +0000396 self.assertEqual(os.read(f.fd, 20), b"blat")
Guido van Rossum0e548712002-08-09 16:14:33 +0000397
398test_classes.append(test__mkstemp_inner)
399
400
401class test_gettempprefix(TC):
402 """Test gettempprefix()."""
403
404 def test_sane_template(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000405 # gettempprefix returns a nonempty prefix string
Guido van Rossum0e548712002-08-09 16:14:33 +0000406 p = tempfile.gettempprefix()
407
Ezio Melottie9615932010-01-24 19:26:24 +0000408 self.assertIsInstance(p, str)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000409 self.assertTrue(len(p) > 0)
Guido van Rossum0e548712002-08-09 16:14:33 +0000410
411 def test_usable_template(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000412 # gettempprefix returns a usable prefix string
Guido van Rossum0e548712002-08-09 16:14:33 +0000413
414 # Create a temp directory, avoiding use of the prefix.
415 # Then attempt to create a file whose name is
416 # prefix + 'xxxxxx.xxx' in that directory.
417 p = tempfile.gettempprefix() + "xxxxxx.xxx"
418 d = tempfile.mkdtemp(prefix="")
419 try:
420 p = os.path.join(d, p)
421 try:
422 fd = os.open(p, os.O_RDWR | os.O_CREAT)
423 except:
424 self.failOnException("os.open")
425 os.close(fd)
426 os.unlink(p)
427 finally:
428 os.rmdir(d)
429
430test_classes.append(test_gettempprefix)
431
432
433class test_gettempdir(TC):
434 """Test gettempdir()."""
435
436 def test_directory_exists(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000437 # gettempdir returns a directory which exists
Guido van Rossum0e548712002-08-09 16:14:33 +0000438
439 dir = tempfile.gettempdir()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000440 self.assertTrue(os.path.isabs(dir) or dir == os.curdir,
Guido van Rossum0e548712002-08-09 16:14:33 +0000441 "%s is not an absolute path" % dir)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000442 self.assertTrue(os.path.isdir(dir),
Guido van Rossum0e548712002-08-09 16:14:33 +0000443 "%s is not a directory" % dir)
444
445 def test_directory_writable(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000446 # gettempdir returns a directory writable by the user
Guido van Rossum0e548712002-08-09 16:14:33 +0000447
448 # sneaky: just instantiate a NamedTemporaryFile, which
449 # defaults to writing into the directory returned by
450 # gettempdir.
451 try:
452 file = tempfile.NamedTemporaryFile()
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000453 file.write(b"blat")
Guido van Rossum0e548712002-08-09 16:14:33 +0000454 file.close()
455 except:
456 self.failOnException("create file in %s" % tempfile.gettempdir())
457
458 def test_same_thing(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000459 # gettempdir always returns the same object
Guido van Rossum0e548712002-08-09 16:14:33 +0000460 a = tempfile.gettempdir()
461 b = tempfile.gettempdir()
462
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000463 self.assertTrue(a is b)
Guido van Rossum0e548712002-08-09 16:14:33 +0000464
465test_classes.append(test_gettempdir)
466
467
468class test_mkstemp(TC):
469 """Test mkstemp()."""
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000470
Thomas Wouters73e5a5b2006-06-08 15:35:45 +0000471 def do_create(self, dir=None, pre="", suf=""):
Guido van Rossum0e548712002-08-09 16:14:33 +0000472 if dir is None:
473 dir = tempfile.gettempdir()
474 try:
475 (fd, name) = tempfile.mkstemp(dir=dir, prefix=pre, suffix=suf)
Martin v. Löwisd6625482003-10-12 17:37:01 +0000476 (ndir, nbase) = os.path.split(name)
477 adir = os.path.abspath(dir)
478 self.assertEqual(adir, ndir,
479 "Directory '%s' incorrectly returned as '%s'" % (adir, ndir))
Guido van Rossum0e548712002-08-09 16:14:33 +0000480 except:
481 self.failOnException("mkstemp")
482
483 try:
484 self.nameCheck(name, dir, pre, suf)
485 finally:
486 os.close(fd)
487 os.unlink(name)
488
489 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000490 # mkstemp can create files
Guido van Rossum0e548712002-08-09 16:14:33 +0000491 self.do_create()
492 self.do_create(pre="a")
493 self.do_create(suf="b")
494 self.do_create(pre="a", suf="b")
495 self.do_create(pre="aa", suf=".txt")
Martin v. Löwisd6625482003-10-12 17:37:01 +0000496 self.do_create(dir=".")
Guido van Rossum0e548712002-08-09 16:14:33 +0000497
498 def test_choose_directory(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000499 # mkstemp can create directories in a user-selected directory
Guido van Rossum0e548712002-08-09 16:14:33 +0000500 dir = tempfile.mkdtemp()
501 try:
502 self.do_create(dir=dir)
503 finally:
504 os.rmdir(dir)
505
506test_classes.append(test_mkstemp)
507
508
509class test_mkdtemp(TC):
510 """Test mkdtemp()."""
511
512 def do_create(self, dir=None, pre="", suf=""):
513 if dir is None:
514 dir = tempfile.gettempdir()
515 try:
516 name = tempfile.mkdtemp(dir=dir, prefix=pre, suffix=suf)
517 except:
518 self.failOnException("mkdtemp")
519
520 try:
521 self.nameCheck(name, dir, pre, suf)
522 return name
523 except:
524 os.rmdir(name)
525 raise
526
527 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000528 # mkdtemp can create directories
Guido van Rossum0e548712002-08-09 16:14:33 +0000529 os.rmdir(self.do_create())
530 os.rmdir(self.do_create(pre="a"))
531 os.rmdir(self.do_create(suf="b"))
532 os.rmdir(self.do_create(pre="a", suf="b"))
533 os.rmdir(self.do_create(pre="aa", suf=".txt"))
Tim Petersa0d55de2002-08-09 18:01:01 +0000534
Guido van Rossum0e548712002-08-09 16:14:33 +0000535 def test_basic_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000536 # mkdtemp can create many directories (stochastic)
Guido van Rossum805365e2007-05-07 22:24:25 +0000537 extant = list(range(TEST_FILES))
Guido van Rossum0e548712002-08-09 16:14:33 +0000538 try:
539 for i in extant:
540 extant[i] = self.do_create(pre="aa")
541 finally:
542 for i in extant:
Guido van Rossum3172c5d2007-10-16 18:12:55 +0000543 if(isinstance(i, str)):
Guido van Rossum0e548712002-08-09 16:14:33 +0000544 os.rmdir(i)
545
546 def test_choose_directory(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000547 # mkdtemp can create directories in a user-selected directory
Guido van Rossum0e548712002-08-09 16:14:33 +0000548 dir = tempfile.mkdtemp()
549 try:
550 os.rmdir(self.do_create(dir=dir))
551 finally:
552 os.rmdir(dir)
553
554 def test_mode(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000555 # mkdtemp creates directories with the proper mode
Guido van Rossum0e548712002-08-09 16:14:33 +0000556 if not has_stat:
Benjamin Petersone549ead2009-03-28 21:42:05 +0000557 return # ugh, can't use SkipTest.
Guido van Rossum0e548712002-08-09 16:14:33 +0000558
559 dir = self.do_create()
560 try:
561 mode = stat.S_IMODE(os.stat(dir).st_mode)
Guido van Rossumcd16bf62007-06-13 18:07:49 +0000562 mode &= 0o777 # Mask off sticky bits inherited from /tmp
563 expected = 0o700
Ronald Oussoren94f25282010-05-05 19:11:21 +0000564 if sys.platform in ('win32', 'os2emx'):
Tim Petersca3ac7f2002-08-09 18:13:51 +0000565 # There's no distinction among 'user', 'group' and 'world';
566 # replicate the 'user' bits.
567 user = expected >> 6
568 expected = user * (1 + 8 + 64)
569 self.assertEqual(mode, expected)
Guido van Rossum0e548712002-08-09 16:14:33 +0000570 finally:
571 os.rmdir(dir)
572
573test_classes.append(test_mkdtemp)
574
575
576class test_mktemp(TC):
577 """Test mktemp()."""
578
579 # For safety, all use of mktemp must occur in a private directory.
580 # We must also suppress the RuntimeWarning it generates.
581 def setUp(self):
582 self.dir = tempfile.mkdtemp()
Brett Cannone1adece2010-03-20 22:19:55 +0000583 super().setUp()
Guido van Rossum0e548712002-08-09 16:14:33 +0000584
585 def tearDown(self):
586 if self.dir:
587 os.rmdir(self.dir)
588 self.dir = None
Brett Cannone1adece2010-03-20 22:19:55 +0000589 super().tearDown()
Guido van Rossum0e548712002-08-09 16:14:33 +0000590
591 class mktemped:
592 _unlink = os.unlink
593 _bflags = tempfile._bin_openflags
594
595 def __init__(self, dir, pre, suf):
596 self.name = tempfile.mktemp(dir=dir, prefix=pre, suffix=suf)
597 # Create the file. This will raise an exception if it's
598 # mysteriously appeared in the meanwhile.
Guido van Rossumcd16bf62007-06-13 18:07:49 +0000599 os.close(os.open(self.name, self._bflags, 0o600))
Guido van Rossum0e548712002-08-09 16:14:33 +0000600
601 def __del__(self):
602 self._unlink(self.name)
603
604 def do_create(self, pre="", suf=""):
605 try:
606 file = self.mktemped(self.dir, pre, suf)
607 except:
608 self.failOnException("mktemp")
609
610 self.nameCheck(file.name, self.dir, pre, suf)
611 return file
612
613 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000614 # mktemp can choose usable file names
Guido van Rossum0e548712002-08-09 16:14:33 +0000615 self.do_create()
616 self.do_create(pre="a")
617 self.do_create(suf="b")
618 self.do_create(pre="a", suf="b")
619 self.do_create(pre="aa", suf=".txt")
620
621 def test_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000622 # mktemp can choose many usable file names (stochastic)
Guido van Rossum805365e2007-05-07 22:24:25 +0000623 extant = list(range(TEST_FILES))
Guido van Rossum0e548712002-08-09 16:14:33 +0000624 for i in extant:
625 extant[i] = self.do_create(pre="aa")
626
Fred Drake8bec4832002-11-22 20:13:43 +0000627## def test_warning(self):
628## # mktemp issues a warning when used
629## warnings.filterwarnings("error",
630## category=RuntimeWarning,
631## message="mktemp")
632## self.assertRaises(RuntimeWarning,
633## tempfile.mktemp, dir=self.dir)
Guido van Rossum0e548712002-08-09 16:14:33 +0000634
635test_classes.append(test_mktemp)
636
637
638# We test _TemporaryFileWrapper by testing NamedTemporaryFile.
639
640
641class test_NamedTemporaryFile(TC):
642 """Test NamedTemporaryFile()."""
643
Guido van Rossumd8faa362007-04-27 19:54:29 +0000644 def do_create(self, dir=None, pre="", suf="", delete=True):
Guido van Rossum0e548712002-08-09 16:14:33 +0000645 if dir is None:
646 dir = tempfile.gettempdir()
647 try:
Guido van Rossumd8faa362007-04-27 19:54:29 +0000648 file = tempfile.NamedTemporaryFile(dir=dir, prefix=pre, suffix=suf,
649 delete=delete)
Guido van Rossum0e548712002-08-09 16:14:33 +0000650 except:
651 self.failOnException("NamedTemporaryFile")
652
653 self.nameCheck(file.name, dir, pre, suf)
654 return file
655
656
657 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000658 # NamedTemporaryFile can create files
Guido van Rossum0e548712002-08-09 16:14:33 +0000659 self.do_create()
660 self.do_create(pre="a")
661 self.do_create(suf="b")
662 self.do_create(pre="a", suf="b")
663 self.do_create(pre="aa", suf=".txt")
664
665 def test_creates_named(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000666 # NamedTemporaryFile creates files with names
Guido van Rossum0e548712002-08-09 16:14:33 +0000667 f = tempfile.NamedTemporaryFile()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000668 self.assertTrue(os.path.exists(f.name),
Guido van Rossum0e548712002-08-09 16:14:33 +0000669 "NamedTemporaryFile %s does not exist" % f.name)
670
671 def test_del_on_close(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000672 # A NamedTemporaryFile is deleted when closed
Guido van Rossum0e548712002-08-09 16:14:33 +0000673 dir = tempfile.mkdtemp()
674 try:
675 f = tempfile.NamedTemporaryFile(dir=dir)
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000676 f.write(b'blat')
Guido van Rossum0e548712002-08-09 16:14:33 +0000677 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000678 self.assertFalse(os.path.exists(f.name),
Guido van Rossum0e548712002-08-09 16:14:33 +0000679 "NamedTemporaryFile %s exists after close" % f.name)
680 finally:
681 os.rmdir(dir)
682
Guido van Rossumd8faa362007-04-27 19:54:29 +0000683 def test_dis_del_on_close(self):
684 # Tests that delete-on-close can be disabled
685 dir = tempfile.mkdtemp()
686 tmp = None
687 try:
688 f = tempfile.NamedTemporaryFile(dir=dir, delete=False)
689 tmp = f.name
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000690 f.write(b'blat')
Guido van Rossumd8faa362007-04-27 19:54:29 +0000691 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000692 self.assertTrue(os.path.exists(f.name),
Guido van Rossumd8faa362007-04-27 19:54:29 +0000693 "NamedTemporaryFile %s missing after close" % f.name)
694 finally:
695 if tmp is not None:
696 os.unlink(tmp)
697 os.rmdir(dir)
698
Guido van Rossum0e548712002-08-09 16:14:33 +0000699 def test_multiple_close(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000700 # A NamedTemporaryFile can be closed many times without error
Guido van Rossum0e548712002-08-09 16:14:33 +0000701 f = tempfile.NamedTemporaryFile()
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000702 f.write(b'abc\n')
Guido van Rossum0e548712002-08-09 16:14:33 +0000703 f.close()
704 try:
705 f.close()
706 f.close()
707 except:
708 self.failOnException("close")
709
Christian Heimes3ecfea712008-02-09 20:51:34 +0000710 def test_context_manager(self):
711 # A NamedTemporaryFile can be used as a context manager
712 with tempfile.NamedTemporaryFile() as f:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000713 self.assertTrue(os.path.exists(f.name))
714 self.assertFalse(os.path.exists(f.name))
Christian Heimes3ecfea712008-02-09 20:51:34 +0000715 def use_closed():
716 with f:
717 pass
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000718 self.assertRaises(ValueError, use_closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000719
Guido van Rossum0e548712002-08-09 16:14:33 +0000720 # How to test the mode and bufsize parameters?
721
722test_classes.append(test_NamedTemporaryFile)
723
Guido van Rossumd8faa362007-04-27 19:54:29 +0000724class test_SpooledTemporaryFile(TC):
725 """Test SpooledTemporaryFile()."""
726
727 def do_create(self, max_size=0, dir=None, pre="", suf=""):
728 if dir is None:
729 dir = tempfile.gettempdir()
730 try:
731 file = tempfile.SpooledTemporaryFile(max_size=max_size, dir=dir, prefix=pre, suffix=suf)
732 except:
733 self.failOnException("SpooledTemporaryFile")
734
735 return file
736
737
738 def test_basic(self):
739 # SpooledTemporaryFile can create files
740 f = self.do_create()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000741 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000742 f = self.do_create(max_size=100, pre="a", suf=".txt")
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000743 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000744
745 def test_del_on_close(self):
746 # A SpooledTemporaryFile is deleted when closed
747 dir = tempfile.mkdtemp()
748 try:
749 f = tempfile.SpooledTemporaryFile(max_size=10, dir=dir)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000750 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +0000751 f.write(b'blat ' * 5)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000752 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000753 filename = f.name
754 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000755 self.assertFalse(isinstance(filename, str) and os.path.exists(filename),
Guido van Rossumd8faa362007-04-27 19:54:29 +0000756 "SpooledTemporaryFile %s exists after close" % filename)
757 finally:
758 os.rmdir(dir)
759
760 def test_rewrite_small(self):
761 # A SpooledTemporaryFile can be written to multiple within the max_size
762 f = self.do_create(max_size=30)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000763 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000764 for i in range(5):
765 f.seek(0, 0)
Guido van Rossum39478e82007-08-27 17:23:59 +0000766 f.write(b'x' * 20)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000767 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000768
769 def test_write_sequential(self):
770 # A SpooledTemporaryFile should hold exactly max_size bytes, and roll
771 # over afterward
772 f = self.do_create(max_size=30)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000773 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +0000774 f.write(b'x' * 20)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000775 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +0000776 f.write(b'x' * 10)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000777 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +0000778 f.write(b'x')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000779 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000780
R David Murrayd89ee792011-03-14 09:55:46 -0400781 def test_writelines(self):
782 # Verify writelines with a SpooledTemporaryFile
783 f = self.do_create()
784 f.writelines((b'x', b'y', b'z'))
785 f.seek(0)
786 buf = f.read()
787 self.assertEqual(buf, b'xyz')
788
789 def test_writelines_sequential(self):
790 # A SpooledTemporaryFile should hold exactly max_size bytes, and roll
791 # over afterward
792 f = self.do_create(max_size=35)
793 f.writelines((b'x' * 20, b'x' * 10, b'x' * 5))
794 self.assertFalse(f._rolled)
795 f.write(b'x')
796 self.assertTrue(f._rolled)
797
Guido van Rossumd8faa362007-04-27 19:54:29 +0000798 def test_sparse(self):
799 # A SpooledTemporaryFile that is written late in the file will extend
800 # when that occurs
801 f = self.do_create(max_size=30)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000802 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000803 f.seek(100, 0)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000804 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +0000805 f.write(b'x')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000806 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000807
808 def test_fileno(self):
809 # A SpooledTemporaryFile should roll over to a real file on fileno()
810 f = self.do_create(max_size=30)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000811 self.assertFalse(f._rolled)
812 self.assertTrue(f.fileno() > 0)
813 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000814
Christian Heimes3ecfea712008-02-09 20:51:34 +0000815 def test_multiple_close_before_rollover(self):
Guido van Rossumd8faa362007-04-27 19:54:29 +0000816 # A SpooledTemporaryFile can be closed many times without error
817 f = tempfile.SpooledTemporaryFile()
Guido van Rossum39478e82007-08-27 17:23:59 +0000818 f.write(b'abc\n')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000819 self.assertFalse(f._rolled)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000820 f.close()
821 try:
822 f.close()
823 f.close()
824 except:
825 self.failOnException("close")
826
827 def test_multiple_close_after_rollover(self):
828 # A SpooledTemporaryFile can be closed many times without error
829 f = tempfile.SpooledTemporaryFile(max_size=1)
830 f.write(b'abc\n')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000831 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000832 f.close()
833 try:
834 f.close()
835 f.close()
836 except:
837 self.failOnException("close")
838
839 def test_bound_methods(self):
840 # It should be OK to steal a bound method from a SpooledTemporaryFile
841 # and use it independently; when the file rolls over, those bound
842 # methods should continue to function
843 f = self.do_create(max_size=30)
844 read = f.read
845 write = f.write
846 seek = f.seek
847
Guido van Rossum39478e82007-08-27 17:23:59 +0000848 write(b"a" * 35)
849 write(b"b" * 35)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000850 seek(0, 0)
Guido van Rossum9a634702007-07-09 10:24:45 +0000851 self.assertEqual(read(70), b'a'*35 + b'b'*35)
852
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +0200853 def test_properties(self):
854 f = tempfile.SpooledTemporaryFile(max_size=10)
855 f.write(b'x' * 10)
856 self.assertFalse(f._rolled)
857 self.assertEqual(f.mode, 'w+b')
858 self.assertIsNone(f.name)
859 with self.assertRaises(AttributeError):
860 f.newlines
861 with self.assertRaises(AttributeError):
862 f.encoding
863
864 f.write(b'x')
865 self.assertTrue(f._rolled)
866 self.assertEqual(f.mode, 'rb+')
867 self.assertIsNotNone(f.name)
868 with self.assertRaises(AttributeError):
869 f.newlines
870 with self.assertRaises(AttributeError):
871 f.encoding
872
Guido van Rossum9a634702007-07-09 10:24:45 +0000873 def test_text_mode(self):
874 # Creating a SpooledTemporaryFile with a text mode should produce
875 # a file object reading and writing (Unicode) text strings.
876 f = tempfile.SpooledTemporaryFile(mode='w+', max_size=10)
877 f.write("abc\n")
878 f.seek(0)
879 self.assertEqual(f.read(), "abc\n")
880 f.write("def\n")
881 f.seek(0)
882 self.assertEqual(f.read(), "abc\ndef\n")
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +0200883 self.assertFalse(f._rolled)
884 self.assertEqual(f.mode, 'w+')
885 self.assertIsNone(f.name)
886 self.assertIsNone(f.newlines)
887 self.assertIsNone(f.encoding)
888
Guido van Rossum9a634702007-07-09 10:24:45 +0000889 f.write("xyzzy\n")
890 f.seek(0)
891 self.assertEqual(f.read(), "abc\ndef\nxyzzy\n")
Amaury Forgeot d'Arc7d0bddd2009-11-30 00:08:56 +0000892 # Check that Ctrl+Z doesn't truncate the file
893 f.write("foo\x1abar\n")
894 f.seek(0)
895 self.assertEqual(f.read(), "abc\ndef\nxyzzy\nfoo\x1abar\n")
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +0200896 self.assertTrue(f._rolled)
897 self.assertEqual(f.mode, 'w+')
898 self.assertIsNotNone(f.name)
Serhiy Storchaka497cee42013-02-10 14:43:46 +0200899 self.assertEqual(f.newlines, os.linesep)
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +0200900 self.assertIsNotNone(f.encoding)
Guido van Rossum9a634702007-07-09 10:24:45 +0000901
Guido van Rossumf0c74162007-08-28 03:29:45 +0000902 def test_text_newline_and_encoding(self):
903 f = tempfile.SpooledTemporaryFile(mode='w+', max_size=10,
904 newline='', encoding='utf-8')
905 f.write("\u039B\r\n")
906 f.seek(0)
907 self.assertEqual(f.read(), "\u039B\r\n")
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000908 self.assertFalse(f._rolled)
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +0200909 self.assertEqual(f.mode, 'w+')
910 self.assertIsNone(f.name)
911 self.assertIsNone(f.newlines)
912 self.assertIsNone(f.encoding)
Guido van Rossumf0c74162007-08-28 03:29:45 +0000913
914 f.write("\u039B" * 20 + "\r\n")
915 f.seek(0)
916 self.assertEqual(f.read(), "\u039B\r\n" + ("\u039B" * 20) + "\r\n")
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000917 self.assertTrue(f._rolled)
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +0200918 self.assertEqual(f.mode, 'w+')
919 self.assertIsNotNone(f.name)
920 self.assertIsNotNone(f.newlines)
921 self.assertEqual(f.encoding, 'utf-8')
Guido van Rossumd8faa362007-04-27 19:54:29 +0000922
Christian Heimes3ecfea712008-02-09 20:51:34 +0000923 def test_context_manager_before_rollover(self):
924 # A SpooledTemporaryFile can be used as a context manager
925 with tempfile.SpooledTemporaryFile(max_size=1) as f:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000926 self.assertFalse(f._rolled)
927 self.assertFalse(f.closed)
928 self.assertTrue(f.closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000929 def use_closed():
930 with f:
931 pass
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000932 self.assertRaises(ValueError, use_closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000933
934 def test_context_manager_during_rollover(self):
935 # A SpooledTemporaryFile can be used as a context manager
936 with tempfile.SpooledTemporaryFile(max_size=1) as f:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000937 self.assertFalse(f._rolled)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000938 f.write(b'abc\n')
939 f.flush()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000940 self.assertTrue(f._rolled)
941 self.assertFalse(f.closed)
942 self.assertTrue(f.closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000943 def use_closed():
944 with f:
945 pass
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000946 self.assertRaises(ValueError, use_closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000947
948 def test_context_manager_after_rollover(self):
949 # A SpooledTemporaryFile can be used as a context manager
950 f = tempfile.SpooledTemporaryFile(max_size=1)
951 f.write(b'abc\n')
952 f.flush()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000953 self.assertTrue(f._rolled)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000954 with f:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000955 self.assertFalse(f.closed)
956 self.assertTrue(f.closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000957 def use_closed():
958 with f:
959 pass
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000960 self.assertRaises(ValueError, use_closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000961
962
Guido van Rossumd8faa362007-04-27 19:54:29 +0000963test_classes.append(test_SpooledTemporaryFile)
964
Guido van Rossum0e548712002-08-09 16:14:33 +0000965
966class test_TemporaryFile(TC):
967 """Test TemporaryFile()."""
968
969 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000970 # TemporaryFile can create files
Guido van Rossum0e548712002-08-09 16:14:33 +0000971 # No point in testing the name params - the file has no name.
972 try:
973 tempfile.TemporaryFile()
974 except:
975 self.failOnException("TemporaryFile")
976
977 def test_has_no_name(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000978 # TemporaryFile creates files with no names (on this system)
Guido van Rossum0e548712002-08-09 16:14:33 +0000979 dir = tempfile.mkdtemp()
980 f = tempfile.TemporaryFile(dir=dir)
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000981 f.write(b'blat')
Guido van Rossum0e548712002-08-09 16:14:33 +0000982
983 # Sneaky: because this file has no name, it should not prevent
984 # us from removing the directory it was created in.
985 try:
986 os.rmdir(dir)
987 except:
988 ei = sys.exc_info()
989 # cleanup
990 f.close()
991 os.rmdir(dir)
992 self.failOnException("rmdir", ei)
993
994 def test_multiple_close(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000995 # A TemporaryFile can be closed many times without error
Guido van Rossum0e548712002-08-09 16:14:33 +0000996 f = tempfile.TemporaryFile()
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000997 f.write(b'abc\n')
Guido van Rossum0e548712002-08-09 16:14:33 +0000998 f.close()
999 try:
1000 f.close()
1001 f.close()
1002 except:
1003 self.failOnException("close")
1004
1005 # How to test the mode and bufsize parameters?
Guido van Rossumf0c74162007-08-28 03:29:45 +00001006 def test_mode_and_encoding(self):
1007
1008 def roundtrip(input, *args, **kwargs):
1009 with tempfile.TemporaryFile(*args, **kwargs) as fileobj:
1010 fileobj.write(input)
1011 fileobj.seek(0)
Ezio Melottib3aedd42010-11-20 19:04:17 +00001012 self.assertEqual(input, fileobj.read())
Guido van Rossumf0c74162007-08-28 03:29:45 +00001013
1014 roundtrip(b"1234", "w+b")
1015 roundtrip("abdc\n", "w+")
1016 roundtrip("\u039B", "w+", encoding="utf-16")
1017 roundtrip("foo\r\n", "w+", newline="")
Guido van Rossum0e548712002-08-09 16:14:33 +00001018
Guido van Rossum0e548712002-08-09 16:14:33 +00001019
Guido van Rossum8ccd9b62002-08-22 20:02:03 +00001020if tempfile.NamedTemporaryFile is not tempfile.TemporaryFile:
Guido van Rossum0e548712002-08-09 16:14:33 +00001021 test_classes.append(test_TemporaryFile)
1022
Nick Coghlan543af752010-10-24 11:23:25 +00001023
1024# Helper for test_del_on_shutdown
1025class NulledModules:
1026 def __init__(self, *modules):
1027 self.refs = [mod.__dict__ for mod in modules]
1028 self.contents = [ref.copy() for ref in self.refs]
1029
1030 def __enter__(self):
1031 for d in self.refs:
1032 for key in d:
1033 d[key] = None
1034
1035 def __exit__(self, *exc_info):
1036 for d, c in zip(self.refs, self.contents):
1037 d.clear()
1038 d.update(c)
1039
1040class test_TemporaryDirectory(TC):
1041 """Test TemporaryDirectory()."""
1042
1043 def do_create(self, dir=None, pre="", suf="", recurse=1):
1044 if dir is None:
1045 dir = tempfile.gettempdir()
1046 try:
1047 tmp = tempfile.TemporaryDirectory(dir=dir, prefix=pre, suffix=suf)
1048 except:
1049 self.failOnException("TemporaryDirectory")
1050 self.nameCheck(tmp.name, dir, pre, suf)
1051 # Create a subdirectory and some files
1052 if recurse:
1053 self.do_create(tmp.name, pre, suf, recurse-1)
1054 with open(os.path.join(tmp.name, "test.txt"), "wb") as f:
1055 f.write(b"Hello world!")
1056 return tmp
1057
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001058 def test_mkdtemp_failure(self):
1059 # Check no additional exception if mkdtemp fails
1060 # Previously would raise AttributeError instead
Nick Coghlan3c54ea62010-12-13 03:02:43 +00001061 # (noted as part of Issue #10188)
1062 with tempfile.TemporaryDirectory() as nonexistent:
1063 pass
1064 with self.assertRaises(os.error):
1065 tempfile.TemporaryDirectory(dir=nonexistent)
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001066
Nick Coghlan543af752010-10-24 11:23:25 +00001067 def test_explicit_cleanup(self):
1068 # A TemporaryDirectory is deleted when cleaned up
1069 dir = tempfile.mkdtemp()
1070 try:
1071 d = self.do_create(dir=dir)
1072 self.assertTrue(os.path.exists(d.name),
1073 "TemporaryDirectory %s does not exist" % d.name)
1074 d.cleanup()
1075 self.assertFalse(os.path.exists(d.name),
1076 "TemporaryDirectory %s exists after cleanup" % d.name)
1077 finally:
1078 os.rmdir(dir)
1079
Charles-François Natalidef35432011-07-29 18:59:24 +02001080 @support.skip_unless_symlink
1081 def test_cleanup_with_symlink_to_a_directory(self):
1082 # cleanup() should not follow symlinks to directories (issue #12464)
1083 d1 = self.do_create()
1084 d2 = self.do_create()
1085
1086 # Symlink d1/foo -> d2
1087 os.symlink(d2.name, os.path.join(d1.name, "foo"))
1088
1089 # This call to cleanup() should not follow the "foo" symlink
1090 d1.cleanup()
1091
1092 self.assertFalse(os.path.exists(d1.name),
1093 "TemporaryDirectory %s exists after cleanup" % d1.name)
1094 self.assertTrue(os.path.exists(d2.name),
1095 "Directory pointed to by a symlink was deleted")
1096 self.assertEqual(os.listdir(d2.name), ['test.txt'],
1097 "Contents of the directory pointed to by a symlink "
1098 "were deleted")
1099 d2.cleanup()
1100
Nick Coghlan543af752010-10-24 11:23:25 +00001101 @support.cpython_only
1102 def test_del_on_collection(self):
1103 # A TemporaryDirectory is deleted when garbage collected
1104 dir = tempfile.mkdtemp()
1105 try:
1106 d = self.do_create(dir=dir)
1107 name = d.name
1108 del d # Rely on refcounting to invoke __del__
1109 self.assertFalse(os.path.exists(name),
1110 "TemporaryDirectory %s exists after __del__" % name)
1111 finally:
1112 os.rmdir(dir)
1113
1114 @unittest.expectedFailure # See issue #10188
1115 def test_del_on_shutdown(self):
1116 # A TemporaryDirectory may be cleaned up during shutdown
1117 # Make sure it works with the relevant modules nulled out
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001118 with self.do_create() as dir:
Nick Coghlan543af752010-10-24 11:23:25 +00001119 d = self.do_create(dir=dir)
1120 # Mimic the nulling out of modules that
1121 # occurs during system shutdown
1122 modules = [os, os.path]
1123 if has_stat:
1124 modules.append(stat)
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001125 # Currently broken, so suppress the warning
1126 # that is otherwise emitted on stdout
1127 with support.captured_stderr() as err:
1128 with NulledModules(*modules):
1129 d.cleanup()
1130 # Currently broken, so stop spurious exception by
1131 # indicating the object has already been closed
1132 d._closed = True
1133 # And this assert will fail, as expected by the
1134 # unittest decorator...
Nick Coghlan543af752010-10-24 11:23:25 +00001135 self.assertFalse(os.path.exists(d.name),
1136 "TemporaryDirectory %s exists after cleanup" % d.name)
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001137
1138 def test_warnings_on_cleanup(self):
1139 # Two kinds of warning on shutdown
1140 # Issue 10888: may write to stderr if modules are nulled out
1141 # ResourceWarning will be triggered by __del__
1142 with self.do_create() as dir:
Nick Coghlane98e8a32010-12-13 16:32:51 +00001143 if os.sep != '\\':
1144 # Embed a backslash in order to make sure string escaping
1145 # in the displayed error message is dealt with correctly
1146 suffix = '\\check_backslash_handling'
1147 else:
1148 suffix = ''
1149 d = self.do_create(dir=dir, suf=suffix)
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001150
1151 #Check for the Issue 10888 message
1152 modules = [os, os.path]
1153 if has_stat:
1154 modules.append(stat)
1155 with support.captured_stderr() as err:
1156 with NulledModules(*modules):
1157 d.cleanup()
Nick Coghlane98e8a32010-12-13 16:32:51 +00001158 message = err.getvalue().replace('\\\\', '\\')
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001159 self.assertIn("while cleaning up", message)
1160 self.assertIn(d.name, message)
1161
1162 # Check for the resource warning
1163 with support.check_warnings(('Implicitly', ResourceWarning), quiet=False):
1164 warnings.filterwarnings("always", category=ResourceWarning)
1165 d.__del__()
1166 self.assertFalse(os.path.exists(d.name),
1167 "TemporaryDirectory %s exists after __del__" % d.name)
Nick Coghlan543af752010-10-24 11:23:25 +00001168
1169 def test_multiple_close(self):
1170 # Can be cleaned-up many times without error
1171 d = self.do_create()
1172 d.cleanup()
1173 try:
1174 d.cleanup()
1175 d.cleanup()
1176 except:
1177 self.failOnException("cleanup")
1178
1179 def test_context_manager(self):
1180 # Can be used as a context manager
1181 d = self.do_create()
1182 with d as name:
1183 self.assertTrue(os.path.exists(name))
1184 self.assertEqual(name, d.name)
1185 self.assertFalse(os.path.exists(name))
1186
1187
1188test_classes.append(test_TemporaryDirectory)
1189
Guido van Rossum0e548712002-08-09 16:14:33 +00001190def test_main():
Benjamin Petersonee8712c2008-05-20 21:35:26 +00001191 support.run_unittest(*test_classes)
Guido van Rossum0e548712002-08-09 16:14:33 +00001192
1193if __name__ == "__main__":
1194 test_main()