blob: 7f4d5b9ec786e1f688d6541d0e121ef4b61aa7b7 [file] [log] [blame]
Guido van Rossum0e548712002-08-09 16:14:33 +00001# tempfile.py unit tests.
Tim Petersc57a2852001-10-29 21:46:08 +00002import tempfile
Serhiy Storchaka7451a722013-02-09 22:25:49 +02003import errno
Guido van Rossum0e548712002-08-09 16:14:33 +00004import os
Antoine Pitrou4558bad2011-11-25 21:28:15 +01005import signal
Guido van Rossum0e548712002-08-09 16:14:33 +00006import sys
7import re
Guido van Rossum0e548712002-08-09 16:14:33 +00008import warnings
Tim Petersc57a2852001-10-29 21:46:08 +00009
Guido van Rossum0e548712002-08-09 16:14:33 +000010import unittest
Benjamin Petersonee8712c2008-05-20 21:35:26 +000011from test import support
Guido van Rossum0e548712002-08-09 16:14:33 +000012
Fred Drake7633d232002-10-17 22:09:03 +000013
Guido van Rossum0e548712002-08-09 16:14:33 +000014if hasattr(os, 'stat'):
15 import stat
16 has_stat = 1
17else:
18 has_stat = 0
19
20has_textmode = (tempfile._text_openflags != tempfile._bin_openflags)
Guido van Rossum78741062002-08-17 11:41:01 +000021has_spawnl = hasattr(os, 'spawnl')
Guido van Rossum0e548712002-08-09 16:14:33 +000022
Neal Norwitz68ee0122002-08-16 19:28:59 +000023# TEST_FILES may need to be tweaked for systems depending on the maximum
24# number of files that can be opened at one time (see ulimit -n)
Victor Stinner9c3de4a2011-08-17 20:49:41 +020025if sys.platform.startswith('openbsd'):
Martin v. Löwis99968282004-09-15 06:02:54 +000026 TEST_FILES = 48
Jack Jansence921472003-01-08 16:30:34 +000027else:
28 TEST_FILES = 100
Neal Norwitz68ee0122002-08-16 19:28:59 +000029
Guido van Rossum0e548712002-08-09 16:14:33 +000030# This is organized as one test for each chunk of code in tempfile.py,
31# in order of their appearance in the file. Testing which requires
32# threads is not done here.
33
34# Common functionality.
Antoine Pitroueab2a502012-03-10 16:34:40 +010035class BaseTestCase(unittest.TestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +000036
37 str_check = re.compile(r"[a-zA-Z0-9_-]{6}$")
38
Brett Cannone1adece2010-03-20 22:19:55 +000039 def setUp(self):
40 self._warnings_manager = support.check_warnings()
41 self._warnings_manager.__enter__()
42 warnings.filterwarnings("ignore", category=RuntimeWarning,
43 message="mktemp", module=__name__)
44
45 def tearDown(self):
46 self._warnings_manager.__exit__(None, None, None)
47
48
Guido van Rossum0e548712002-08-09 16:14:33 +000049 def nameCheck(self, name, dir, pre, suf):
50 (ndir, nbase) = os.path.split(name)
51 npre = nbase[:len(pre)]
52 nsuf = nbase[len(nbase)-len(suf):]
53
Martin v. Löwisd6625482003-10-12 17:37:01 +000054 # check for equality of the absolute paths!
55 self.assertEqual(os.path.abspath(ndir), os.path.abspath(dir),
Guido van Rossum0e548712002-08-09 16:14:33 +000056 "file '%s' not in directory '%s'" % (name, dir))
57 self.assertEqual(npre, pre,
58 "file '%s' does not begin with '%s'" % (nbase, pre))
59 self.assertEqual(nsuf, suf,
60 "file '%s' does not end with '%s'" % (nbase, suf))
61
62 nbase = nbase[len(pre):len(nbase)-len(suf)]
Benjamin Petersonc9c0f202009-06-30 23:06:06 +000063 self.assertTrue(self.str_check.match(nbase),
Guido van Rossum0e548712002-08-09 16:14:33 +000064 "random string '%s' does not match /^[a-zA-Z0-9_-]{6}$/"
65 % nbase)
66
Guido van Rossum0e548712002-08-09 16:14:33 +000067
Antoine Pitroueab2a502012-03-10 16:34:40 +010068class TestExports(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +000069 def test_exports(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +000070 # There are no surprising symbols in the tempfile module
Guido van Rossum0e548712002-08-09 16:14:33 +000071 dict = tempfile.__dict__
72
73 expected = {
74 "NamedTemporaryFile" : 1,
75 "TemporaryFile" : 1,
76 "mkstemp" : 1,
77 "mkdtemp" : 1,
78 "mktemp" : 1,
79 "TMP_MAX" : 1,
80 "gettempprefix" : 1,
81 "gettempdir" : 1,
82 "tempdir" : 1,
Guido van Rossumd8faa362007-04-27 19:54:29 +000083 "template" : 1,
Nick Coghlan543af752010-10-24 11:23:25 +000084 "SpooledTemporaryFile" : 1,
85 "TemporaryDirectory" : 1,
Guido van Rossum0e548712002-08-09 16:14:33 +000086 }
87
88 unexp = []
89 for key in dict:
90 if key[0] != '_' and key not in expected:
91 unexp.append(key)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +000092 self.assertTrue(len(unexp) == 0,
Guido van Rossum0e548712002-08-09 16:14:33 +000093 "unexpected keys: %s" % unexp)
94
Guido van Rossum0e548712002-08-09 16:14:33 +000095
Antoine Pitroueab2a502012-03-10 16:34:40 +010096class TestRandomNameSequence(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +000097 """Test the internal iterator object _RandomNameSequence."""
98
99 def setUp(self):
100 self.r = tempfile._RandomNameSequence()
Brett Cannone1adece2010-03-20 22:19:55 +0000101 super().setUp()
Guido van Rossum0e548712002-08-09 16:14:33 +0000102
103 def test_get_six_char_str(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000104 # _RandomNameSequence returns a six-character string
Georg Brandla18af4e2007-04-21 15:47:16 +0000105 s = next(self.r)
Guido van Rossum0e548712002-08-09 16:14:33 +0000106 self.nameCheck(s, '', '', '')
107
108 def test_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000109 # _RandomNameSequence returns no duplicate strings (stochastic)
Guido van Rossum0e548712002-08-09 16:14:33 +0000110
111 dict = {}
112 r = self.r
Guido van Rossum805365e2007-05-07 22:24:25 +0000113 for i in range(TEST_FILES):
Georg Brandla18af4e2007-04-21 15:47:16 +0000114 s = next(r)
Guido van Rossum0e548712002-08-09 16:14:33 +0000115 self.nameCheck(s, '', '', '')
Ezio Melottib58e0bd2010-01-23 15:40:09 +0000116 self.assertNotIn(s, dict)
Guido van Rossum0e548712002-08-09 16:14:33 +0000117 dict[s] = 1
118
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000119 def supports_iter(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000120 # _RandomNameSequence supports the iterator protocol
Guido van Rossum0e548712002-08-09 16:14:33 +0000121
122 i = 0
123 r = self.r
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100124 for s in r:
125 i += 1
126 if i == 20:
127 break
Guido van Rossum0e548712002-08-09 16:14:33 +0000128
Antoine Pitrou4558bad2011-11-25 21:28:15 +0100129 @unittest.skipUnless(hasattr(os, 'fork'),
130 "os.fork is required for this test")
131 def test_process_awareness(self):
132 # ensure that the random source differs between
133 # child and parent.
134 read_fd, write_fd = os.pipe()
135 pid = None
136 try:
137 pid = os.fork()
138 if not pid:
139 os.close(read_fd)
140 os.write(write_fd, next(self.r).encode("ascii"))
141 os.close(write_fd)
142 # bypass the normal exit handlers- leave those to
143 # the parent.
144 os._exit(0)
145 parent_value = next(self.r)
146 child_value = os.read(read_fd, len(parent_value)).decode("ascii")
147 finally:
148 if pid:
149 # best effort to ensure the process can't bleed out
150 # via any bugs above
151 try:
152 os.kill(pid, signal.SIGKILL)
153 except EnvironmentError:
154 pass
155 os.close(read_fd)
156 os.close(write_fd)
157 self.assertNotEqual(child_value, parent_value)
158
159
Guido van Rossum0e548712002-08-09 16:14:33 +0000160
Antoine Pitroueab2a502012-03-10 16:34:40 +0100161class TestCandidateTempdirList(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000162 """Test the internal function _candidate_tempdir_list."""
163
164 def test_nonempty_list(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000165 # _candidate_tempdir_list returns a nonempty list of strings
Guido van Rossum0e548712002-08-09 16:14:33 +0000166
167 cand = tempfile._candidate_tempdir_list()
168
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000169 self.assertFalse(len(cand) == 0)
Guido van Rossum0e548712002-08-09 16:14:33 +0000170 for c in cand:
Ezio Melottie9615932010-01-24 19:26:24 +0000171 self.assertIsInstance(c, str)
Guido van Rossum0e548712002-08-09 16:14:33 +0000172
173 def test_wanted_dirs(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000174 # _candidate_tempdir_list contains the expected directories
Guido van Rossum0e548712002-08-09 16:14:33 +0000175
176 # Make sure the interesting environment variables are all set.
Hirokazu Yamamoto71959632009-04-27 01:44:28 +0000177 with support.EnvironmentVarGuard() as env:
Guido van Rossum0e548712002-08-09 16:14:33 +0000178 for envname in 'TMPDIR', 'TEMP', 'TMP':
179 dirname = os.getenv(envname)
180 if not dirname:
Walter Dörwald155374d2009-05-01 19:58:58 +0000181 env[envname] = os.path.abspath(envname)
Guido van Rossum0e548712002-08-09 16:14:33 +0000182
183 cand = tempfile._candidate_tempdir_list()
184
185 for envname in 'TMPDIR', 'TEMP', 'TMP':
186 dirname = os.getenv(envname)
187 if not dirname: raise ValueError
Benjamin Peterson577473f2010-01-19 00:09:57 +0000188 self.assertIn(dirname, cand)
Guido van Rossum0e548712002-08-09 16:14:33 +0000189
190 try:
191 dirname = os.getcwd()
192 except (AttributeError, os.error):
193 dirname = os.curdir
194
Benjamin Peterson577473f2010-01-19 00:09:57 +0000195 self.assertIn(dirname, cand)
Guido van Rossum0e548712002-08-09 16:14:33 +0000196
197 # Not practical to try to verify the presence of OS-specific
198 # paths in this list.
Guido van Rossum0e548712002-08-09 16:14:33 +0000199
Guido van Rossum0e548712002-08-09 16:14:33 +0000200
201# We test _get_default_tempdir by testing gettempdir.
202
203
Antoine Pitroueab2a502012-03-10 16:34:40 +0100204class TestGetCandidateNames(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000205 """Test the internal function _get_candidate_names."""
206
207 def test_retval(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000208 # _get_candidate_names returns a _RandomNameSequence object
Guido van Rossum0e548712002-08-09 16:14:33 +0000209 obj = tempfile._get_candidate_names()
Ezio Melottie9615932010-01-24 19:26:24 +0000210 self.assertIsInstance(obj, tempfile._RandomNameSequence)
Guido van Rossum0e548712002-08-09 16:14:33 +0000211
212 def test_same_thing(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000213 # _get_candidate_names always returns the same object
Guido van Rossum0e548712002-08-09 16:14:33 +0000214 a = tempfile._get_candidate_names()
215 b = tempfile._get_candidate_names()
216
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000217 self.assertTrue(a is b)
Guido van Rossum0e548712002-08-09 16:14:33 +0000218
Guido van Rossum0e548712002-08-09 16:14:33 +0000219
Antoine Pitroueab2a502012-03-10 16:34:40 +0100220class TestMkstempInner(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000221 """Test the internal function _mkstemp_inner."""
222
223 class mkstemped:
224 _bflags = tempfile._bin_openflags
225 _tflags = tempfile._text_openflags
226 _close = os.close
227 _unlink = os.unlink
228
229 def __init__(self, dir, pre, suf, bin):
230 if bin: flags = self._bflags
231 else: flags = self._tflags
232
233 (self.fd, self.name) = tempfile._mkstemp_inner(dir, pre, suf, flags)
234
235 def write(self, str):
236 os.write(self.fd, str)
237
238 def __del__(self):
239 self._close(self.fd)
240 self._unlink(self.name)
Tim Petersa0d55de2002-08-09 18:01:01 +0000241
Guido van Rossum0e548712002-08-09 16:14:33 +0000242 def do_create(self, dir=None, pre="", suf="", bin=1):
243 if dir is None:
244 dir = tempfile.gettempdir()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100245 file = self.mkstemped(dir, pre, suf, bin)
Guido van Rossum0e548712002-08-09 16:14:33 +0000246
247 self.nameCheck(file.name, dir, pre, suf)
248 return file
249
250 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000251 # _mkstemp_inner can create files
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000252 self.do_create().write(b"blat")
253 self.do_create(pre="a").write(b"blat")
254 self.do_create(suf="b").write(b"blat")
255 self.do_create(pre="a", suf="b").write(b"blat")
256 self.do_create(pre="aa", suf=".txt").write(b"blat")
Guido van Rossum0e548712002-08-09 16:14:33 +0000257
258 def test_basic_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000259 # _mkstemp_inner can create many files (stochastic)
Guido van Rossum805365e2007-05-07 22:24:25 +0000260 extant = list(range(TEST_FILES))
Guido van Rossum0e548712002-08-09 16:14:33 +0000261 for i in extant:
262 extant[i] = self.do_create(pre="aa")
263
264 def test_choose_directory(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000265 # _mkstemp_inner can create files in a user-selected directory
Guido van Rossum0e548712002-08-09 16:14:33 +0000266 dir = tempfile.mkdtemp()
267 try:
Antoine Pitrou9cadb1b2008-09-15 23:02:56 +0000268 self.do_create(dir=dir).write(b"blat")
Guido van Rossum0e548712002-08-09 16:14:33 +0000269 finally:
270 os.rmdir(dir)
271
272 def test_file_mode(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000273 # _mkstemp_inner creates files with the proper mode
Guido van Rossum0e548712002-08-09 16:14:33 +0000274 if not has_stat:
Benjamin Petersone549ead2009-03-28 21:42:05 +0000275 return # ugh, can't use SkipTest.
Guido van Rossum0e548712002-08-09 16:14:33 +0000276
277 file = self.do_create()
278 mode = stat.S_IMODE(os.stat(file.name).st_mode)
Guido van Rossumcd16bf62007-06-13 18:07:49 +0000279 expected = 0o600
Ronald Oussoren94f25282010-05-05 19:11:21 +0000280 if sys.platform in ('win32', 'os2emx'):
Tim Petersca3ac7f2002-08-09 18:13:51 +0000281 # There's no distinction among 'user', 'group' and 'world';
282 # replicate the 'user' bits.
283 user = expected >> 6
284 expected = user * (1 + 8 + 64)
285 self.assertEqual(mode, expected)
Guido van Rossum0e548712002-08-09 16:14:33 +0000286
287 def test_noinherit(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000288 # _mkstemp_inner file handles are not inherited by child processes
Guido van Rossum78741062002-08-17 11:41:01 +0000289 if not has_spawnl:
Benjamin Petersone549ead2009-03-28 21:42:05 +0000290 return # ugh, can't use SkipTest.
Guido van Rossum0e548712002-08-09 16:14:33 +0000291
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000292 if support.verbose:
Guido van Rossum78741062002-08-17 11:41:01 +0000293 v="v"
294 else:
295 v="q"
296
Guido van Rossum0e548712002-08-09 16:14:33 +0000297 file = self.do_create()
Guido van Rossum78741062002-08-17 11:41:01 +0000298 fd = "%d" % file.fd
299
300 try:
301 me = __file__
302 except NameError:
303 me = sys.argv[0]
Guido van Rossum0e548712002-08-09 16:14:33 +0000304
305 # We have to exec something, so that FD_CLOEXEC will take
Guido van Rossum78741062002-08-17 11:41:01 +0000306 # effect. The core of this test is therefore in
307 # tf_inherit_check.py, which see.
308 tester = os.path.join(os.path.dirname(os.path.abspath(me)),
309 "tf_inherit_check.py")
Guido van Rossum0e548712002-08-09 16:14:33 +0000310
Martin v. Löwisd4210bc2003-10-23 15:55:28 +0000311 # On Windows a spawn* /path/ with embedded spaces shouldn't be quoted,
312 # but an arg with embedded spaces should be decorated with double
313 # quotes on each end
Christian Heimes895627f2007-12-08 17:28:33 +0000314 if sys.platform in ('win32',):
Martin v. Löwisd4210bc2003-10-23 15:55:28 +0000315 decorated = '"%s"' % sys.executable
316 tester = '"%s"' % tester
317 else:
318 decorated = sys.executable
319
320 retval = os.spawnl(os.P_WAIT, sys.executable, decorated, tester, v, fd)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000321 self.assertFalse(retval < 0,
Guido van Rossum78741062002-08-17 11:41:01 +0000322 "child process caught fatal signal %d" % -retval)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000323 self.assertFalse(retval > 0, "child process reports failure %d"%retval)
Guido van Rossum0e548712002-08-09 16:14:33 +0000324
325 def test_textmode(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000326 # _mkstemp_inner can create files in text mode
Guido van Rossum0e548712002-08-09 16:14:33 +0000327 if not has_textmode:
Benjamin Petersone549ead2009-03-28 21:42:05 +0000328 return # ugh, can't use SkipTest.
Guido van Rossum0e548712002-08-09 16:14:33 +0000329
Amaury Forgeot d'Arc7d0bddd2009-11-30 00:08:56 +0000330 # A text file is truncated at the first Ctrl+Z byte
331 f = self.do_create(bin=0)
332 f.write(b"blat\x1a")
333 f.write(b"extra\n")
334 os.lseek(f.fd, 0, os.SEEK_SET)
Ezio Melottib3aedd42010-11-20 19:04:17 +0000335 self.assertEqual(os.read(f.fd, 20), b"blat")
Guido van Rossum0e548712002-08-09 16:14:33 +0000336
Guido van Rossum0e548712002-08-09 16:14:33 +0000337
Antoine Pitroueab2a502012-03-10 16:34:40 +0100338class TestGetTempPrefix(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000339 """Test gettempprefix()."""
340
341 def test_sane_template(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000342 # gettempprefix returns a nonempty prefix string
Guido van Rossum0e548712002-08-09 16:14:33 +0000343 p = tempfile.gettempprefix()
344
Ezio Melottie9615932010-01-24 19:26:24 +0000345 self.assertIsInstance(p, str)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000346 self.assertTrue(len(p) > 0)
Guido van Rossum0e548712002-08-09 16:14:33 +0000347
348 def test_usable_template(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000349 # gettempprefix returns a usable prefix string
Guido van Rossum0e548712002-08-09 16:14:33 +0000350
351 # Create a temp directory, avoiding use of the prefix.
352 # Then attempt to create a file whose name is
353 # prefix + 'xxxxxx.xxx' in that directory.
354 p = tempfile.gettempprefix() + "xxxxxx.xxx"
355 d = tempfile.mkdtemp(prefix="")
356 try:
357 p = os.path.join(d, p)
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100358 fd = os.open(p, os.O_RDWR | os.O_CREAT)
Guido van Rossum0e548712002-08-09 16:14:33 +0000359 os.close(fd)
360 os.unlink(p)
361 finally:
362 os.rmdir(d)
363
Guido van Rossum0e548712002-08-09 16:14:33 +0000364
Antoine Pitroueab2a502012-03-10 16:34:40 +0100365class TestGetTempDir(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000366 """Test gettempdir()."""
367
368 def test_directory_exists(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000369 # gettempdir returns a directory which exists
Guido van Rossum0e548712002-08-09 16:14:33 +0000370
371 dir = tempfile.gettempdir()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000372 self.assertTrue(os.path.isabs(dir) or dir == os.curdir,
Guido van Rossum0e548712002-08-09 16:14:33 +0000373 "%s is not an absolute path" % dir)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000374 self.assertTrue(os.path.isdir(dir),
Guido van Rossum0e548712002-08-09 16:14:33 +0000375 "%s is not a directory" % dir)
376
377 def test_directory_writable(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000378 # gettempdir returns a directory writable by the user
Guido van Rossum0e548712002-08-09 16:14:33 +0000379
380 # sneaky: just instantiate a NamedTemporaryFile, which
381 # defaults to writing into the directory returned by
382 # gettempdir.
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100383 file = tempfile.NamedTemporaryFile()
384 file.write(b"blat")
385 file.close()
Guido van Rossum0e548712002-08-09 16:14:33 +0000386
387 def test_same_thing(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000388 # gettempdir always returns the same object
Guido van Rossum0e548712002-08-09 16:14:33 +0000389 a = tempfile.gettempdir()
390 b = tempfile.gettempdir()
391
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000392 self.assertTrue(a is b)
Guido van Rossum0e548712002-08-09 16:14:33 +0000393
Guido van Rossum0e548712002-08-09 16:14:33 +0000394
Antoine Pitroueab2a502012-03-10 16:34:40 +0100395class TestMkstemp(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000396 """Test mkstemp()."""
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000397
Thomas Wouters73e5a5b2006-06-08 15:35:45 +0000398 def do_create(self, dir=None, pre="", suf=""):
Guido van Rossum0e548712002-08-09 16:14:33 +0000399 if dir is None:
400 dir = tempfile.gettempdir()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100401 (fd, name) = tempfile.mkstemp(dir=dir, prefix=pre, suffix=suf)
402 (ndir, nbase) = os.path.split(name)
403 adir = os.path.abspath(dir)
404 self.assertEqual(adir, ndir,
405 "Directory '%s' incorrectly returned as '%s'" % (adir, ndir))
Guido van Rossum0e548712002-08-09 16:14:33 +0000406
407 try:
408 self.nameCheck(name, dir, pre, suf)
409 finally:
410 os.close(fd)
411 os.unlink(name)
412
413 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000414 # mkstemp can create files
Guido van Rossum0e548712002-08-09 16:14:33 +0000415 self.do_create()
416 self.do_create(pre="a")
417 self.do_create(suf="b")
418 self.do_create(pre="a", suf="b")
419 self.do_create(pre="aa", suf=".txt")
Martin v. Löwisd6625482003-10-12 17:37:01 +0000420 self.do_create(dir=".")
Guido van Rossum0e548712002-08-09 16:14:33 +0000421
422 def test_choose_directory(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000423 # mkstemp can create directories in a user-selected directory
Guido van Rossum0e548712002-08-09 16:14:33 +0000424 dir = tempfile.mkdtemp()
425 try:
426 self.do_create(dir=dir)
427 finally:
428 os.rmdir(dir)
429
Guido van Rossum0e548712002-08-09 16:14:33 +0000430
Antoine Pitroueab2a502012-03-10 16:34:40 +0100431class TestMkdtemp(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000432 """Test mkdtemp()."""
433
434 def do_create(self, dir=None, pre="", suf=""):
435 if dir is None:
436 dir = tempfile.gettempdir()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100437 name = tempfile.mkdtemp(dir=dir, prefix=pre, suffix=suf)
Guido van Rossum0e548712002-08-09 16:14:33 +0000438
439 try:
440 self.nameCheck(name, dir, pre, suf)
441 return name
442 except:
443 os.rmdir(name)
444 raise
445
446 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000447 # mkdtemp can create directories
Guido van Rossum0e548712002-08-09 16:14:33 +0000448 os.rmdir(self.do_create())
449 os.rmdir(self.do_create(pre="a"))
450 os.rmdir(self.do_create(suf="b"))
451 os.rmdir(self.do_create(pre="a", suf="b"))
452 os.rmdir(self.do_create(pre="aa", suf=".txt"))
Tim Petersa0d55de2002-08-09 18:01:01 +0000453
Guido van Rossum0e548712002-08-09 16:14:33 +0000454 def test_basic_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000455 # mkdtemp can create many directories (stochastic)
Guido van Rossum805365e2007-05-07 22:24:25 +0000456 extant = list(range(TEST_FILES))
Guido van Rossum0e548712002-08-09 16:14:33 +0000457 try:
458 for i in extant:
459 extant[i] = self.do_create(pre="aa")
460 finally:
461 for i in extant:
Guido van Rossum3172c5d2007-10-16 18:12:55 +0000462 if(isinstance(i, str)):
Guido van Rossum0e548712002-08-09 16:14:33 +0000463 os.rmdir(i)
464
465 def test_choose_directory(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000466 # mkdtemp can create directories in a user-selected directory
Guido van Rossum0e548712002-08-09 16:14:33 +0000467 dir = tempfile.mkdtemp()
468 try:
469 os.rmdir(self.do_create(dir=dir))
470 finally:
471 os.rmdir(dir)
472
473 def test_mode(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000474 # mkdtemp creates directories with the proper mode
Guido van Rossum0e548712002-08-09 16:14:33 +0000475 if not has_stat:
Benjamin Petersone549ead2009-03-28 21:42:05 +0000476 return # ugh, can't use SkipTest.
Guido van Rossum0e548712002-08-09 16:14:33 +0000477
478 dir = self.do_create()
479 try:
480 mode = stat.S_IMODE(os.stat(dir).st_mode)
Guido van Rossumcd16bf62007-06-13 18:07:49 +0000481 mode &= 0o777 # Mask off sticky bits inherited from /tmp
482 expected = 0o700
Ronald Oussoren94f25282010-05-05 19:11:21 +0000483 if sys.platform in ('win32', 'os2emx'):
Tim Petersca3ac7f2002-08-09 18:13:51 +0000484 # There's no distinction among 'user', 'group' and 'world';
485 # replicate the 'user' bits.
486 user = expected >> 6
487 expected = user * (1 + 8 + 64)
488 self.assertEqual(mode, expected)
Guido van Rossum0e548712002-08-09 16:14:33 +0000489 finally:
490 os.rmdir(dir)
491
Guido van Rossum0e548712002-08-09 16:14:33 +0000492
Antoine Pitroueab2a502012-03-10 16:34:40 +0100493class TestMktemp(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000494 """Test mktemp()."""
495
496 # For safety, all use of mktemp must occur in a private directory.
497 # We must also suppress the RuntimeWarning it generates.
498 def setUp(self):
499 self.dir = tempfile.mkdtemp()
Brett Cannone1adece2010-03-20 22:19:55 +0000500 super().setUp()
Guido van Rossum0e548712002-08-09 16:14:33 +0000501
502 def tearDown(self):
503 if self.dir:
504 os.rmdir(self.dir)
505 self.dir = None
Brett Cannone1adece2010-03-20 22:19:55 +0000506 super().tearDown()
Guido van Rossum0e548712002-08-09 16:14:33 +0000507
508 class mktemped:
509 _unlink = os.unlink
510 _bflags = tempfile._bin_openflags
511
512 def __init__(self, dir, pre, suf):
513 self.name = tempfile.mktemp(dir=dir, prefix=pre, suffix=suf)
514 # Create the file. This will raise an exception if it's
515 # mysteriously appeared in the meanwhile.
Guido van Rossumcd16bf62007-06-13 18:07:49 +0000516 os.close(os.open(self.name, self._bflags, 0o600))
Guido van Rossum0e548712002-08-09 16:14:33 +0000517
518 def __del__(self):
519 self._unlink(self.name)
520
521 def do_create(self, pre="", suf=""):
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100522 file = self.mktemped(self.dir, pre, suf)
Guido van Rossum0e548712002-08-09 16:14:33 +0000523
524 self.nameCheck(file.name, self.dir, pre, suf)
525 return file
526
527 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000528 # mktemp can choose usable file names
Guido van Rossum0e548712002-08-09 16:14:33 +0000529 self.do_create()
530 self.do_create(pre="a")
531 self.do_create(suf="b")
532 self.do_create(pre="a", suf="b")
533 self.do_create(pre="aa", suf=".txt")
534
535 def test_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000536 # mktemp can choose many usable file names (stochastic)
Guido van Rossum805365e2007-05-07 22:24:25 +0000537 extant = list(range(TEST_FILES))
Guido van Rossum0e548712002-08-09 16:14:33 +0000538 for i in extant:
539 extant[i] = self.do_create(pre="aa")
540
Fred Drake8bec4832002-11-22 20:13:43 +0000541## def test_warning(self):
542## # mktemp issues a warning when used
543## warnings.filterwarnings("error",
544## category=RuntimeWarning,
545## message="mktemp")
546## self.assertRaises(RuntimeWarning,
547## tempfile.mktemp, dir=self.dir)
Guido van Rossum0e548712002-08-09 16:14:33 +0000548
Guido van Rossum0e548712002-08-09 16:14:33 +0000549
550# We test _TemporaryFileWrapper by testing NamedTemporaryFile.
551
552
Antoine Pitroueab2a502012-03-10 16:34:40 +0100553class TestNamedTemporaryFile(BaseTestCase):
Guido van Rossum0e548712002-08-09 16:14:33 +0000554 """Test NamedTemporaryFile()."""
555
Guido van Rossumd8faa362007-04-27 19:54:29 +0000556 def do_create(self, dir=None, pre="", suf="", delete=True):
Guido van Rossum0e548712002-08-09 16:14:33 +0000557 if dir is None:
558 dir = tempfile.gettempdir()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100559 file = tempfile.NamedTemporaryFile(dir=dir, prefix=pre, suffix=suf,
560 delete=delete)
Guido van Rossum0e548712002-08-09 16:14:33 +0000561
562 self.nameCheck(file.name, dir, pre, suf)
563 return file
564
565
566 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000567 # NamedTemporaryFile can create files
Guido van Rossum0e548712002-08-09 16:14:33 +0000568 self.do_create()
569 self.do_create(pre="a")
570 self.do_create(suf="b")
571 self.do_create(pre="a", suf="b")
572 self.do_create(pre="aa", suf=".txt")
573
574 def test_creates_named(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000575 # NamedTemporaryFile creates files with names
Guido van Rossum0e548712002-08-09 16:14:33 +0000576 f = tempfile.NamedTemporaryFile()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000577 self.assertTrue(os.path.exists(f.name),
Guido van Rossum0e548712002-08-09 16:14:33 +0000578 "NamedTemporaryFile %s does not exist" % f.name)
579
580 def test_del_on_close(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000581 # A NamedTemporaryFile is deleted when closed
Guido van Rossum0e548712002-08-09 16:14:33 +0000582 dir = tempfile.mkdtemp()
583 try:
584 f = tempfile.NamedTemporaryFile(dir=dir)
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000585 f.write(b'blat')
Guido van Rossum0e548712002-08-09 16:14:33 +0000586 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000587 self.assertFalse(os.path.exists(f.name),
Guido van Rossum0e548712002-08-09 16:14:33 +0000588 "NamedTemporaryFile %s exists after close" % f.name)
589 finally:
590 os.rmdir(dir)
591
Guido van Rossumd8faa362007-04-27 19:54:29 +0000592 def test_dis_del_on_close(self):
593 # Tests that delete-on-close can be disabled
594 dir = tempfile.mkdtemp()
595 tmp = None
596 try:
597 f = tempfile.NamedTemporaryFile(dir=dir, delete=False)
598 tmp = f.name
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000599 f.write(b'blat')
Guido van Rossumd8faa362007-04-27 19:54:29 +0000600 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000601 self.assertTrue(os.path.exists(f.name),
Guido van Rossumd8faa362007-04-27 19:54:29 +0000602 "NamedTemporaryFile %s missing after close" % f.name)
603 finally:
604 if tmp is not None:
605 os.unlink(tmp)
606 os.rmdir(dir)
607
Guido van Rossum0e548712002-08-09 16:14:33 +0000608 def test_multiple_close(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000609 # A NamedTemporaryFile can be closed many times without error
Guido van Rossum0e548712002-08-09 16:14:33 +0000610 f = tempfile.NamedTemporaryFile()
Guido van Rossumec42ffd2007-08-27 23:40:36 +0000611 f.write(b'abc\n')
Guido van Rossum0e548712002-08-09 16:14:33 +0000612 f.close()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100613 f.close()
614 f.close()
Guido van Rossum0e548712002-08-09 16:14:33 +0000615
Christian Heimes3ecfea712008-02-09 20:51:34 +0000616 def test_context_manager(self):
617 # A NamedTemporaryFile can be used as a context manager
618 with tempfile.NamedTemporaryFile() as f:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000619 self.assertTrue(os.path.exists(f.name))
620 self.assertFalse(os.path.exists(f.name))
Christian Heimes3ecfea712008-02-09 20:51:34 +0000621 def use_closed():
622 with f:
623 pass
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000624 self.assertRaises(ValueError, use_closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000625
Guido van Rossum0e548712002-08-09 16:14:33 +0000626 # How to test the mode and bufsize parameters?
627
Guido van Rossum0e548712002-08-09 16:14:33 +0000628
Antoine Pitroueab2a502012-03-10 16:34:40 +0100629class TestSpooledTemporaryFile(BaseTestCase):
Guido van Rossumd8faa362007-04-27 19:54:29 +0000630 """Test SpooledTemporaryFile()."""
631
632 def do_create(self, max_size=0, dir=None, pre="", suf=""):
633 if dir is None:
634 dir = tempfile.gettempdir()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100635 file = tempfile.SpooledTemporaryFile(max_size=max_size, dir=dir, prefix=pre, suffix=suf)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000636
637 return file
638
639
640 def test_basic(self):
641 # SpooledTemporaryFile can create files
642 f = self.do_create()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000643 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000644 f = self.do_create(max_size=100, pre="a", suf=".txt")
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000645 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000646
647 def test_del_on_close(self):
648 # A SpooledTemporaryFile is deleted when closed
649 dir = tempfile.mkdtemp()
650 try:
651 f = tempfile.SpooledTemporaryFile(max_size=10, dir=dir)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000652 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +0000653 f.write(b'blat ' * 5)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000654 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000655 filename = f.name
656 f.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000657 self.assertFalse(isinstance(filename, str) and os.path.exists(filename),
Guido van Rossumd8faa362007-04-27 19:54:29 +0000658 "SpooledTemporaryFile %s exists after close" % filename)
659 finally:
660 os.rmdir(dir)
661
662 def test_rewrite_small(self):
663 # A SpooledTemporaryFile can be written to multiple within the max_size
664 f = self.do_create(max_size=30)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000665 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000666 for i in range(5):
667 f.seek(0, 0)
Guido van Rossum39478e82007-08-27 17:23:59 +0000668 f.write(b'x' * 20)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000669 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000670
671 def test_write_sequential(self):
672 # A SpooledTemporaryFile should hold exactly max_size bytes, and roll
673 # over afterward
674 f = self.do_create(max_size=30)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000675 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +0000676 f.write(b'x' * 20)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000677 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +0000678 f.write(b'x' * 10)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000679 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +0000680 f.write(b'x')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000681 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000682
R David Murrayd89ee792011-03-14 09:55:46 -0400683 def test_writelines(self):
684 # Verify writelines with a SpooledTemporaryFile
685 f = self.do_create()
686 f.writelines((b'x', b'y', b'z'))
687 f.seek(0)
688 buf = f.read()
689 self.assertEqual(buf, b'xyz')
690
691 def test_writelines_sequential(self):
692 # A SpooledTemporaryFile should hold exactly max_size bytes, and roll
693 # over afterward
694 f = self.do_create(max_size=35)
695 f.writelines((b'x' * 20, b'x' * 10, b'x' * 5))
696 self.assertFalse(f._rolled)
697 f.write(b'x')
698 self.assertTrue(f._rolled)
699
Guido van Rossumd8faa362007-04-27 19:54:29 +0000700 def test_sparse(self):
701 # A SpooledTemporaryFile that is written late in the file will extend
702 # when that occurs
703 f = self.do_create(max_size=30)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000704 self.assertFalse(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000705 f.seek(100, 0)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000706 self.assertFalse(f._rolled)
Guido van Rossum39478e82007-08-27 17:23:59 +0000707 f.write(b'x')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000708 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000709
710 def test_fileno(self):
711 # A SpooledTemporaryFile should roll over to a real file on fileno()
712 f = self.do_create(max_size=30)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000713 self.assertFalse(f._rolled)
714 self.assertTrue(f.fileno() > 0)
715 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000716
Christian Heimes3ecfea712008-02-09 20:51:34 +0000717 def test_multiple_close_before_rollover(self):
Guido van Rossumd8faa362007-04-27 19:54:29 +0000718 # A SpooledTemporaryFile can be closed many times without error
719 f = tempfile.SpooledTemporaryFile()
Guido van Rossum39478e82007-08-27 17:23:59 +0000720 f.write(b'abc\n')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000721 self.assertFalse(f._rolled)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000722 f.close()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100723 f.close()
724 f.close()
Christian Heimes3ecfea712008-02-09 20:51:34 +0000725
726 def test_multiple_close_after_rollover(self):
727 # A SpooledTemporaryFile can be closed many times without error
728 f = tempfile.SpooledTemporaryFile(max_size=1)
729 f.write(b'abc\n')
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000730 self.assertTrue(f._rolled)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000731 f.close()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100732 f.close()
733 f.close()
Guido van Rossumd8faa362007-04-27 19:54:29 +0000734
735 def test_bound_methods(self):
736 # It should be OK to steal a bound method from a SpooledTemporaryFile
737 # and use it independently; when the file rolls over, those bound
738 # methods should continue to function
739 f = self.do_create(max_size=30)
740 read = f.read
741 write = f.write
742 seek = f.seek
743
Guido van Rossum39478e82007-08-27 17:23:59 +0000744 write(b"a" * 35)
745 write(b"b" * 35)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000746 seek(0, 0)
Guido van Rossum9a634702007-07-09 10:24:45 +0000747 self.assertEqual(read(70), b'a'*35 + b'b'*35)
748
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +0200749 def test_properties(self):
750 f = tempfile.SpooledTemporaryFile(max_size=10)
751 f.write(b'x' * 10)
752 self.assertFalse(f._rolled)
753 self.assertEqual(f.mode, 'w+b')
754 self.assertIsNone(f.name)
755 with self.assertRaises(AttributeError):
756 f.newlines
757 with self.assertRaises(AttributeError):
758 f.encoding
759
760 f.write(b'x')
761 self.assertTrue(f._rolled)
762 self.assertEqual(f.mode, 'rb+')
763 self.assertIsNotNone(f.name)
764 with self.assertRaises(AttributeError):
765 f.newlines
766 with self.assertRaises(AttributeError):
767 f.encoding
768
Guido van Rossum9a634702007-07-09 10:24:45 +0000769 def test_text_mode(self):
770 # Creating a SpooledTemporaryFile with a text mode should produce
771 # a file object reading and writing (Unicode) text strings.
772 f = tempfile.SpooledTemporaryFile(mode='w+', max_size=10)
773 f.write("abc\n")
774 f.seek(0)
775 self.assertEqual(f.read(), "abc\n")
776 f.write("def\n")
777 f.seek(0)
778 self.assertEqual(f.read(), "abc\ndef\n")
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +0200779 self.assertFalse(f._rolled)
780 self.assertEqual(f.mode, 'w+')
781 self.assertIsNone(f.name)
782 self.assertIsNone(f.newlines)
783 self.assertIsNone(f.encoding)
784
Guido van Rossum9a634702007-07-09 10:24:45 +0000785 f.write("xyzzy\n")
786 f.seek(0)
787 self.assertEqual(f.read(), "abc\ndef\nxyzzy\n")
Amaury Forgeot d'Arc7d0bddd2009-11-30 00:08:56 +0000788 # Check that Ctrl+Z doesn't truncate the file
789 f.write("foo\x1abar\n")
790 f.seek(0)
791 self.assertEqual(f.read(), "abc\ndef\nxyzzy\nfoo\x1abar\n")
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +0200792 self.assertTrue(f._rolled)
793 self.assertEqual(f.mode, 'w+')
794 self.assertIsNotNone(f.name)
795 self.assertEqual(f.newlines, '\n')
796 self.assertIsNotNone(f.encoding)
Guido van Rossum9a634702007-07-09 10:24:45 +0000797
Guido van Rossumf0c74162007-08-28 03:29:45 +0000798 def test_text_newline_and_encoding(self):
799 f = tempfile.SpooledTemporaryFile(mode='w+', max_size=10,
800 newline='', encoding='utf-8')
801 f.write("\u039B\r\n")
802 f.seek(0)
803 self.assertEqual(f.read(), "\u039B\r\n")
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000804 self.assertFalse(f._rolled)
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +0200805 self.assertEqual(f.mode, 'w+')
806 self.assertIsNone(f.name)
807 self.assertIsNone(f.newlines)
808 self.assertIsNone(f.encoding)
Guido van Rossumf0c74162007-08-28 03:29:45 +0000809
810 f.write("\u039B" * 20 + "\r\n")
811 f.seek(0)
812 self.assertEqual(f.read(), "\u039B\r\n" + ("\u039B" * 20) + "\r\n")
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000813 self.assertTrue(f._rolled)
Serhiy Storchakabbbbe8e2013-02-09 12:21:14 +0200814 self.assertEqual(f.mode, 'w+')
815 self.assertIsNotNone(f.name)
816 self.assertIsNotNone(f.newlines)
817 self.assertEqual(f.encoding, 'utf-8')
Guido van Rossumd8faa362007-04-27 19:54:29 +0000818
Christian Heimes3ecfea712008-02-09 20:51:34 +0000819 def test_context_manager_before_rollover(self):
820 # A SpooledTemporaryFile can be used as a context manager
821 with tempfile.SpooledTemporaryFile(max_size=1) as f:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000822 self.assertFalse(f._rolled)
823 self.assertFalse(f.closed)
824 self.assertTrue(f.closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000825 def use_closed():
826 with f:
827 pass
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000828 self.assertRaises(ValueError, use_closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000829
830 def test_context_manager_during_rollover(self):
831 # A SpooledTemporaryFile can be used as a context manager
832 with tempfile.SpooledTemporaryFile(max_size=1) as f:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000833 self.assertFalse(f._rolled)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000834 f.write(b'abc\n')
835 f.flush()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000836 self.assertTrue(f._rolled)
837 self.assertFalse(f.closed)
838 self.assertTrue(f.closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000839 def use_closed():
840 with f:
841 pass
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000842 self.assertRaises(ValueError, use_closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000843
844 def test_context_manager_after_rollover(self):
845 # A SpooledTemporaryFile can be used as a context manager
846 f = tempfile.SpooledTemporaryFile(max_size=1)
847 f.write(b'abc\n')
848 f.flush()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000849 self.assertTrue(f._rolled)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000850 with f:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000851 self.assertFalse(f.closed)
852 self.assertTrue(f.closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000853 def use_closed():
854 with f:
855 pass
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000856 self.assertRaises(ValueError, use_closed)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000857
Antoine Pitrou0e86a582011-11-25 18:03:09 +0100858 def test_truncate_with_size_parameter(self):
859 # A SpooledTemporaryFile can be truncated to zero size
860 f = tempfile.SpooledTemporaryFile(max_size=10)
861 f.write(b'abcdefg\n')
862 f.seek(0)
863 f.truncate()
864 self.assertFalse(f._rolled)
865 self.assertEqual(f._file.getvalue(), b'')
866 # A SpooledTemporaryFile can be truncated to a specific size
867 f = tempfile.SpooledTemporaryFile(max_size=10)
868 f.write(b'abcdefg\n')
869 f.truncate(4)
870 self.assertFalse(f._rolled)
871 self.assertEqual(f._file.getvalue(), b'abcd')
872 # A SpooledTemporaryFile rolls over if truncated to large size
873 f = tempfile.SpooledTemporaryFile(max_size=10)
874 f.write(b'abcdefg\n')
875 f.truncate(20)
876 self.assertTrue(f._rolled)
877 if has_stat:
878 self.assertEqual(os.fstat(f.fileno()).st_size, 20)
Christian Heimes3ecfea712008-02-09 20:51:34 +0000879
Guido van Rossum0e548712002-08-09 16:14:33 +0000880
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000881if tempfile.NamedTemporaryFile is not tempfile.TemporaryFile:
Antoine Pitroueab2a502012-03-10 16:34:40 +0100882
883 class TestTemporaryFile(BaseTestCase):
884 """Test TemporaryFile()."""
885
886 def test_basic(self):
887 # TemporaryFile can create files
888 # No point in testing the name params - the file has no name.
889 tempfile.TemporaryFile()
890
891 def test_has_no_name(self):
892 # TemporaryFile creates files with no names (on this system)
893 dir = tempfile.mkdtemp()
894 f = tempfile.TemporaryFile(dir=dir)
895 f.write(b'blat')
896
897 # Sneaky: because this file has no name, it should not prevent
898 # us from removing the directory it was created in.
899 try:
900 os.rmdir(dir)
901 except:
902 # cleanup
903 f.close()
904 os.rmdir(dir)
905 raise
906
907 def test_multiple_close(self):
908 # A TemporaryFile can be closed many times without error
909 f = tempfile.TemporaryFile()
910 f.write(b'abc\n')
911 f.close()
912 f.close()
913 f.close()
914
915 # How to test the mode and bufsize parameters?
916 def test_mode_and_encoding(self):
917
918 def roundtrip(input, *args, **kwargs):
919 with tempfile.TemporaryFile(*args, **kwargs) as fileobj:
920 fileobj.write(input)
921 fileobj.seek(0)
922 self.assertEqual(input, fileobj.read())
923
924 roundtrip(b"1234", "w+b")
925 roundtrip("abdc\n", "w+")
926 roundtrip("\u039B", "w+", encoding="utf-16")
927 roundtrip("foo\r\n", "w+", newline="")
Guido van Rossum0e548712002-08-09 16:14:33 +0000928
Nick Coghlan543af752010-10-24 11:23:25 +0000929
930# Helper for test_del_on_shutdown
931class NulledModules:
932 def __init__(self, *modules):
933 self.refs = [mod.__dict__ for mod in modules]
934 self.contents = [ref.copy() for ref in self.refs]
935
936 def __enter__(self):
937 for d in self.refs:
938 for key in d:
939 d[key] = None
940
941 def __exit__(self, *exc_info):
942 for d, c in zip(self.refs, self.contents):
943 d.clear()
944 d.update(c)
945
Antoine Pitroueab2a502012-03-10 16:34:40 +0100946class TestTemporaryDirectory(BaseTestCase):
Nick Coghlan543af752010-10-24 11:23:25 +0000947 """Test TemporaryDirectory()."""
948
949 def do_create(self, dir=None, pre="", suf="", recurse=1):
950 if dir is None:
951 dir = tempfile.gettempdir()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +0100952 tmp = tempfile.TemporaryDirectory(dir=dir, prefix=pre, suffix=suf)
Nick Coghlan543af752010-10-24 11:23:25 +0000953 self.nameCheck(tmp.name, dir, pre, suf)
954 # Create a subdirectory and some files
955 if recurse:
956 self.do_create(tmp.name, pre, suf, recurse-1)
957 with open(os.path.join(tmp.name, "test.txt"), "wb") as f:
958 f.write(b"Hello world!")
959 return tmp
960
Nick Coghlan6b22f3f2010-12-12 15:24:21 +0000961 def test_mkdtemp_failure(self):
962 # Check no additional exception if mkdtemp fails
963 # Previously would raise AttributeError instead
Nick Coghlan3c54ea62010-12-13 03:02:43 +0000964 # (noted as part of Issue #10188)
965 with tempfile.TemporaryDirectory() as nonexistent:
966 pass
Serhiy Storchaka7451a722013-02-09 22:25:49 +0200967 with self.assertRaises(FileNotFoundError) as cm:
Nick Coghlan3c54ea62010-12-13 03:02:43 +0000968 tempfile.TemporaryDirectory(dir=nonexistent)
Serhiy Storchaka7451a722013-02-09 22:25:49 +0200969 self.assertEqual(cm.exception.errno, errno.ENOENT)
Nick Coghlan6b22f3f2010-12-12 15:24:21 +0000970
Nick Coghlan543af752010-10-24 11:23:25 +0000971 def test_explicit_cleanup(self):
972 # A TemporaryDirectory is deleted when cleaned up
973 dir = tempfile.mkdtemp()
974 try:
975 d = self.do_create(dir=dir)
976 self.assertTrue(os.path.exists(d.name),
977 "TemporaryDirectory %s does not exist" % d.name)
978 d.cleanup()
979 self.assertFalse(os.path.exists(d.name),
980 "TemporaryDirectory %s exists after cleanup" % d.name)
981 finally:
982 os.rmdir(dir)
983
Charles-François Natalidef35432011-07-29 18:59:24 +0200984 @support.skip_unless_symlink
985 def test_cleanup_with_symlink_to_a_directory(self):
986 # cleanup() should not follow symlinks to directories (issue #12464)
987 d1 = self.do_create()
988 d2 = self.do_create()
989
990 # Symlink d1/foo -> d2
991 os.symlink(d2.name, os.path.join(d1.name, "foo"))
992
993 # This call to cleanup() should not follow the "foo" symlink
994 d1.cleanup()
995
996 self.assertFalse(os.path.exists(d1.name),
997 "TemporaryDirectory %s exists after cleanup" % d1.name)
998 self.assertTrue(os.path.exists(d2.name),
999 "Directory pointed to by a symlink was deleted")
1000 self.assertEqual(os.listdir(d2.name), ['test.txt'],
1001 "Contents of the directory pointed to by a symlink "
1002 "were deleted")
1003 d2.cleanup()
1004
Nick Coghlan543af752010-10-24 11:23:25 +00001005 @support.cpython_only
1006 def test_del_on_collection(self):
1007 # A TemporaryDirectory is deleted when garbage collected
1008 dir = tempfile.mkdtemp()
1009 try:
1010 d = self.do_create(dir=dir)
1011 name = d.name
1012 del d # Rely on refcounting to invoke __del__
1013 self.assertFalse(os.path.exists(name),
1014 "TemporaryDirectory %s exists after __del__" % name)
1015 finally:
1016 os.rmdir(dir)
1017
1018 @unittest.expectedFailure # See issue #10188
1019 def test_del_on_shutdown(self):
1020 # A TemporaryDirectory may be cleaned up during shutdown
1021 # Make sure it works with the relevant modules nulled out
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001022 with self.do_create() as dir:
Nick Coghlan543af752010-10-24 11:23:25 +00001023 d = self.do_create(dir=dir)
1024 # Mimic the nulling out of modules that
1025 # occurs during system shutdown
1026 modules = [os, os.path]
1027 if has_stat:
1028 modules.append(stat)
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001029 # Currently broken, so suppress the warning
1030 # that is otherwise emitted on stdout
1031 with support.captured_stderr() as err:
1032 with NulledModules(*modules):
1033 d.cleanup()
1034 # Currently broken, so stop spurious exception by
1035 # indicating the object has already been closed
1036 d._closed = True
1037 # And this assert will fail, as expected by the
1038 # unittest decorator...
Nick Coghlan543af752010-10-24 11:23:25 +00001039 self.assertFalse(os.path.exists(d.name),
1040 "TemporaryDirectory %s exists after cleanup" % d.name)
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001041
1042 def test_warnings_on_cleanup(self):
1043 # Two kinds of warning on shutdown
1044 # Issue 10888: may write to stderr if modules are nulled out
1045 # ResourceWarning will be triggered by __del__
1046 with self.do_create() as dir:
Nick Coghlane98e8a32010-12-13 16:32:51 +00001047 if os.sep != '\\':
1048 # Embed a backslash in order to make sure string escaping
1049 # in the displayed error message is dealt with correctly
1050 suffix = '\\check_backslash_handling'
1051 else:
1052 suffix = ''
1053 d = self.do_create(dir=dir, suf=suffix)
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001054
1055 #Check for the Issue 10888 message
1056 modules = [os, os.path]
1057 if has_stat:
1058 modules.append(stat)
1059 with support.captured_stderr() as err:
1060 with NulledModules(*modules):
1061 d.cleanup()
Nick Coghlane98e8a32010-12-13 16:32:51 +00001062 message = err.getvalue().replace('\\\\', '\\')
Nick Coghlan6b22f3f2010-12-12 15:24:21 +00001063 self.assertIn("while cleaning up", message)
1064 self.assertIn(d.name, message)
1065
1066 # Check for the resource warning
1067 with support.check_warnings(('Implicitly', ResourceWarning), quiet=False):
1068 warnings.filterwarnings("always", category=ResourceWarning)
1069 d.__del__()
1070 self.assertFalse(os.path.exists(d.name),
1071 "TemporaryDirectory %s exists after __del__" % d.name)
Nick Coghlan543af752010-10-24 11:23:25 +00001072
1073 def test_multiple_close(self):
1074 # Can be cleaned-up many times without error
1075 d = self.do_create()
1076 d.cleanup()
Antoine Pitrou8cd8d5e2012-03-10 16:20:24 +01001077 d.cleanup()
1078 d.cleanup()
Nick Coghlan543af752010-10-24 11:23:25 +00001079
1080 def test_context_manager(self):
1081 # Can be used as a context manager
1082 d = self.do_create()
1083 with d as name:
1084 self.assertTrue(os.path.exists(name))
1085 self.assertEqual(name, d.name)
1086 self.assertFalse(os.path.exists(name))
1087
1088
Guido van Rossum0e548712002-08-09 16:14:33 +00001089def test_main():
Antoine Pitroueab2a502012-03-10 16:34:40 +01001090 support.run_unittest(__name__)
Guido van Rossum0e548712002-08-09 16:14:33 +00001091
1092if __name__ == "__main__":
1093 test_main()