blob: 8b883b030c5afaa61cd1d0f258298cc5e75d216d [file] [log] [blame]
Guido van Rossum0e548712002-08-09 16:14:33 +00001# tempfile.py unit tests.
Tim Petersc57a2852001-10-29 21:46:08 +00002import tempfile
Guido van Rossum0e548712002-08-09 16:14:33 +00003import os
4import sys
5import re
6import errno
7import warnings
Tim Petersc57a2852001-10-29 21:46:08 +00008
Guido van Rossum0e548712002-08-09 16:14:33 +00009import unittest
10from test import test_support
11
Fred Drake7633d232002-10-17 22:09:03 +000012warnings.filterwarnings("ignore",
13 category=RuntimeWarning,
14 message="mktemp", module=__name__)
15
Guido van Rossum0e548712002-08-09 16:14:33 +000016if hasattr(os, 'stat'):
17 import stat
18 has_stat = 1
19else:
20 has_stat = 0
21
22has_textmode = (tempfile._text_openflags != tempfile._bin_openflags)
Guido van Rossum78741062002-08-17 11:41:01 +000023has_spawnl = hasattr(os, 'spawnl')
Guido van Rossum0e548712002-08-09 16:14:33 +000024
Neal Norwitz68ee0122002-08-16 19:28:59 +000025# TEST_FILES may need to be tweaked for systems depending on the maximum
26# number of files that can be opened at one time (see ulimit -n)
Jack Jansence921472003-01-08 16:30:34 +000027if sys.platform == 'mac':
28 TEST_FILES = 32
Neal Norwitz4bc2c092006-09-05 02:57:01 +000029elif sys.platform in ('openbsd3', 'openbsd4'):
Martin v. Löwis99968282004-09-15 06:02:54 +000030 TEST_FILES = 48
Jack Jansence921472003-01-08 16:30:34 +000031else:
32 TEST_FILES = 100
Neal Norwitz68ee0122002-08-16 19:28:59 +000033
Guido van Rossum0e548712002-08-09 16:14:33 +000034# This is organized as one test for each chunk of code in tempfile.py,
35# in order of their appearance in the file. Testing which requires
36# threads is not done here.
37
38# Common functionality.
39class TC(unittest.TestCase):
40
41 str_check = re.compile(r"[a-zA-Z0-9_-]{6}$")
42
43 def failOnException(self, what, ei=None):
44 if ei is None:
45 ei = sys.exc_info()
46 self.fail("%s raised %s: %s" % (what, ei[0], ei[1]))
47
48 def nameCheck(self, name, dir, pre, suf):
49 (ndir, nbase) = os.path.split(name)
50 npre = nbase[:len(pre)]
51 nsuf = nbase[len(nbase)-len(suf):]
52
Martin v. Löwisd6625482003-10-12 17:37:01 +000053 # check for equality of the absolute paths!
54 self.assertEqual(os.path.abspath(ndir), os.path.abspath(dir),
Guido van Rossum0e548712002-08-09 16:14:33 +000055 "file '%s' not in directory '%s'" % (name, dir))
56 self.assertEqual(npre, pre,
57 "file '%s' does not begin with '%s'" % (nbase, pre))
58 self.assertEqual(nsuf, suf,
59 "file '%s' does not end with '%s'" % (nbase, suf))
60
61 nbase = nbase[len(pre):len(nbase)-len(suf)]
62 self.assert_(self.str_check.match(nbase),
63 "random string '%s' does not match /^[a-zA-Z0-9_-]{6}$/"
64 % nbase)
65
66test_classes = []
67
68class test_exports(TC):
69 def test_exports(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +000070 # There are no surprising symbols in the tempfile module
Guido van Rossum0e548712002-08-09 16:14:33 +000071 dict = tempfile.__dict__
72
73 expected = {
74 "NamedTemporaryFile" : 1,
75 "TemporaryFile" : 1,
76 "mkstemp" : 1,
77 "mkdtemp" : 1,
78 "mktemp" : 1,
79 "TMP_MAX" : 1,
80 "gettempprefix" : 1,
81 "gettempdir" : 1,
82 "tempdir" : 1,
Collin Wintera8785cc2007-03-19 18:52:08 +000083 "template" : 1,
84 "SpooledTemporaryFile" : 1
Guido van Rossum0e548712002-08-09 16:14:33 +000085 }
86
87 unexp = []
88 for key in dict:
89 if key[0] != '_' and key not in expected:
90 unexp.append(key)
91 self.failUnless(len(unexp) == 0,
92 "unexpected keys: %s" % unexp)
93
94test_classes.append(test_exports)
95
96
Guido van Rossum0e548712002-08-09 16:14:33 +000097class test__RandomNameSequence(TC):
98 """Test the internal iterator object _RandomNameSequence."""
99
100 def setUp(self):
101 self.r = tempfile._RandomNameSequence()
102
103 def test_get_six_char_str(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000104 # _RandomNameSequence returns a six-character string
Guido van Rossum0e548712002-08-09 16:14:33 +0000105 s = self.r.next()
106 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
Neal Norwitz68ee0122002-08-16 19:28:59 +0000113 for i in xrange(TEST_FILES):
Guido van Rossum0e548712002-08-09 16:14:33 +0000114 s = r.next()
115 self.nameCheck(s, '', '', '')
116 self.failIf(s in dict)
117 dict[s] = 1
118
119 def test_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
124 try:
125 for s in r:
126 i += 1
127 if i == 20:
128 break
129 except:
130 failOnException("iteration")
131
132test_classes.append(test__RandomNameSequence)
133
134
135class test__candidate_tempdir_list(TC):
136 """Test the internal function _candidate_tempdir_list."""
137
138 def test_nonempty_list(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000139 # _candidate_tempdir_list returns a nonempty list of strings
Guido van Rossum0e548712002-08-09 16:14:33 +0000140
141 cand = tempfile._candidate_tempdir_list()
142
143 self.failIf(len(cand) == 0)
144 for c in cand:
145 self.assert_(isinstance(c, basestring),
146 "%s is not a string" % c)
147
148 def test_wanted_dirs(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000149 # _candidate_tempdir_list contains the expected directories
Guido van Rossum0e548712002-08-09 16:14:33 +0000150
151 # Make sure the interesting environment variables are all set.
Walter Dörwald4b965f62009-04-26 20:51:44 +0000152 with test_support.EnvironmentVarGuard() as env:
Guido van Rossum0e548712002-08-09 16:14:33 +0000153 for envname in 'TMPDIR', 'TEMP', 'TMP':
154 dirname = os.getenv(envname)
155 if not dirname:
Walter Dörwald6733bed2009-05-01 17:35:37 +0000156 env[envname] = os.path.abspath(envname)
Guido van Rossum0e548712002-08-09 16:14:33 +0000157
158 cand = tempfile._candidate_tempdir_list()
159
160 for envname in 'TMPDIR', 'TEMP', 'TMP':
161 dirname = os.getenv(envname)
162 if not dirname: raise ValueError
163 self.assert_(dirname in cand)
164
165 try:
166 dirname = os.getcwd()
167 except (AttributeError, os.error):
168 dirname = os.curdir
169
170 self.assert_(dirname in cand)
171
172 # Not practical to try to verify the presence of OS-specific
173 # paths in this list.
Guido van Rossum0e548712002-08-09 16:14:33 +0000174
175test_classes.append(test__candidate_tempdir_list)
176
177
178# We test _get_default_tempdir by testing gettempdir.
179
180
181class test__get_candidate_names(TC):
182 """Test the internal function _get_candidate_names."""
183
184 def test_retval(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000185 # _get_candidate_names returns a _RandomNameSequence object
Guido van Rossum0e548712002-08-09 16:14:33 +0000186 obj = tempfile._get_candidate_names()
187 self.assert_(isinstance(obj, tempfile._RandomNameSequence))
188
189 def test_same_thing(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000190 # _get_candidate_names always returns the same object
Guido van Rossum0e548712002-08-09 16:14:33 +0000191 a = tempfile._get_candidate_names()
192 b = tempfile._get_candidate_names()
193
194 self.assert_(a is b)
195
196test_classes.append(test__get_candidate_names)
197
198
199class test__mkstemp_inner(TC):
200 """Test the internal function _mkstemp_inner."""
201
202 class mkstemped:
203 _bflags = tempfile._bin_openflags
204 _tflags = tempfile._text_openflags
205 _close = os.close
206 _unlink = os.unlink
207
208 def __init__(self, dir, pre, suf, bin):
209 if bin: flags = self._bflags
210 else: flags = self._tflags
211
212 (self.fd, self.name) = tempfile._mkstemp_inner(dir, pre, suf, flags)
213
214 def write(self, str):
215 os.write(self.fd, str)
216
217 def __del__(self):
218 self._close(self.fd)
219 self._unlink(self.name)
Tim Petersa0d55de2002-08-09 18:01:01 +0000220
Guido van Rossum0e548712002-08-09 16:14:33 +0000221 def do_create(self, dir=None, pre="", suf="", bin=1):
222 if dir is None:
223 dir = tempfile.gettempdir()
224 try:
225 file = self.mkstemped(dir, pre, suf, bin)
226 except:
227 self.failOnException("_mkstemp_inner")
228
229 self.nameCheck(file.name, dir, pre, suf)
230 return file
231
232 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000233 # _mkstemp_inner can create files
Guido van Rossum0e548712002-08-09 16:14:33 +0000234 self.do_create().write("blat")
235 self.do_create(pre="a").write("blat")
236 self.do_create(suf="b").write("blat")
237 self.do_create(pre="a", suf="b").write("blat")
238 self.do_create(pre="aa", suf=".txt").write("blat")
239
240 def test_basic_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000241 # _mkstemp_inner can create many files (stochastic)
Neal Norwitz68ee0122002-08-16 19:28:59 +0000242 extant = range(TEST_FILES)
Guido van Rossum0e548712002-08-09 16:14:33 +0000243 for i in extant:
244 extant[i] = self.do_create(pre="aa")
245
246 def test_choose_directory(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000247 # _mkstemp_inner can create files in a user-selected directory
Guido van Rossum0e548712002-08-09 16:14:33 +0000248 dir = tempfile.mkdtemp()
249 try:
250 self.do_create(dir=dir).write("blat")
251 finally:
252 os.rmdir(dir)
253
254 def test_file_mode(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000255 # _mkstemp_inner creates files with the proper mode
Guido van Rossum0e548712002-08-09 16:14:33 +0000256 if not has_stat:
Benjamin Peterson888a39b2009-03-26 20:48:25 +0000257 return # ugh, can't use SkipTest.
Guido van Rossum0e548712002-08-09 16:14:33 +0000258
259 file = self.do_create()
260 mode = stat.S_IMODE(os.stat(file.name).st_mode)
Tim Petersca3ac7f2002-08-09 18:13:51 +0000261 expected = 0600
Jack Jansence921472003-01-08 16:30:34 +0000262 if sys.platform in ('win32', 'os2emx', 'mac'):
Tim Petersca3ac7f2002-08-09 18:13:51 +0000263 # There's no distinction among 'user', 'group' and 'world';
264 # replicate the 'user' bits.
265 user = expected >> 6
266 expected = user * (1 + 8 + 64)
267 self.assertEqual(mode, expected)
Guido van Rossum0e548712002-08-09 16:14:33 +0000268
269 def test_noinherit(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000270 # _mkstemp_inner file handles are not inherited by child processes
Guido van Rossum78741062002-08-17 11:41:01 +0000271 if not has_spawnl:
Benjamin Peterson888a39b2009-03-26 20:48:25 +0000272 return # ugh, can't use SkipTest.
Guido van Rossum0e548712002-08-09 16:14:33 +0000273
Guido van Rossum78741062002-08-17 11:41:01 +0000274 if test_support.verbose:
275 v="v"
276 else:
277 v="q"
278
Guido van Rossum0e548712002-08-09 16:14:33 +0000279 file = self.do_create()
Guido van Rossum78741062002-08-17 11:41:01 +0000280 fd = "%d" % file.fd
281
282 try:
283 me = __file__
284 except NameError:
285 me = sys.argv[0]
Guido van Rossum0e548712002-08-09 16:14:33 +0000286
287 # We have to exec something, so that FD_CLOEXEC will take
Guido van Rossum78741062002-08-17 11:41:01 +0000288 # effect. The core of this test is therefore in
289 # tf_inherit_check.py, which see.
290 tester = os.path.join(os.path.dirname(os.path.abspath(me)),
291 "tf_inherit_check.py")
Guido van Rossum0e548712002-08-09 16:14:33 +0000292
Martin v. Löwisd4210bc2003-10-23 15:55:28 +0000293 # On Windows a spawn* /path/ with embedded spaces shouldn't be quoted,
294 # but an arg with embedded spaces should be decorated with double
295 # quotes on each end
Armin Rigo66d41b22007-12-07 19:19:55 +0000296 if sys.platform in ('win32',):
Martin v. Löwisd4210bc2003-10-23 15:55:28 +0000297 decorated = '"%s"' % sys.executable
298 tester = '"%s"' % tester
299 else:
300 decorated = sys.executable
301
302 retval = os.spawnl(os.P_WAIT, sys.executable, decorated, tester, v, fd)
Guido van Rossum78741062002-08-17 11:41:01 +0000303 self.failIf(retval < 0,
304 "child process caught fatal signal %d" % -retval)
Michael W. Hudsona1fb4c82005-02-15 15:22:37 +0000305 self.failIf(retval > 0, "child process reports failure %d"%retval)
Guido van Rossum0e548712002-08-09 16:14:33 +0000306
307 def test_textmode(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000308 # _mkstemp_inner can create files in text mode
Guido van Rossum0e548712002-08-09 16:14:33 +0000309 if not has_textmode:
Benjamin Peterson888a39b2009-03-26 20:48:25 +0000310 return # ugh, can't use SkipTest.
Guido van Rossum0e548712002-08-09 16:14:33 +0000311
312 self.do_create(bin=0).write("blat\n")
313 # XXX should test that the file really is a text file
314
315test_classes.append(test__mkstemp_inner)
316
317
318class test_gettempprefix(TC):
319 """Test gettempprefix()."""
320
321 def test_sane_template(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000322 # gettempprefix returns a nonempty prefix string
Guido van Rossum0e548712002-08-09 16:14:33 +0000323 p = tempfile.gettempprefix()
324
325 self.assert_(isinstance(p, basestring))
326 self.assert_(len(p) > 0)
327
328 def test_usable_template(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000329 # gettempprefix returns a usable prefix string
Guido van Rossum0e548712002-08-09 16:14:33 +0000330
331 # Create a temp directory, avoiding use of the prefix.
332 # Then attempt to create a file whose name is
333 # prefix + 'xxxxxx.xxx' in that directory.
334 p = tempfile.gettempprefix() + "xxxxxx.xxx"
335 d = tempfile.mkdtemp(prefix="")
336 try:
337 p = os.path.join(d, p)
338 try:
339 fd = os.open(p, os.O_RDWR | os.O_CREAT)
340 except:
341 self.failOnException("os.open")
342 os.close(fd)
343 os.unlink(p)
344 finally:
345 os.rmdir(d)
346
347test_classes.append(test_gettempprefix)
348
349
350class test_gettempdir(TC):
351 """Test gettempdir()."""
352
353 def test_directory_exists(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000354 # gettempdir returns a directory which exists
Guido van Rossum0e548712002-08-09 16:14:33 +0000355
356 dir = tempfile.gettempdir()
357 self.assert_(os.path.isabs(dir) or dir == os.curdir,
358 "%s is not an absolute path" % dir)
359 self.assert_(os.path.isdir(dir),
360 "%s is not a directory" % dir)
361
362 def test_directory_writable(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000363 # gettempdir returns a directory writable by the user
Guido van Rossum0e548712002-08-09 16:14:33 +0000364
365 # sneaky: just instantiate a NamedTemporaryFile, which
366 # defaults to writing into the directory returned by
367 # gettempdir.
368 try:
369 file = tempfile.NamedTemporaryFile()
370 file.write("blat")
371 file.close()
372 except:
373 self.failOnException("create file in %s" % tempfile.gettempdir())
374
375 def test_same_thing(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000376 # gettempdir always returns the same object
Guido van Rossum0e548712002-08-09 16:14:33 +0000377 a = tempfile.gettempdir()
378 b = tempfile.gettempdir()
379
380 self.assert_(a is b)
381
382test_classes.append(test_gettempdir)
383
384
385class test_mkstemp(TC):
386 """Test mkstemp()."""
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000387
Martin Blais215f13d2006-06-06 12:46:55 +0000388 def do_create(self, dir=None, pre="", suf=""):
Guido van Rossum0e548712002-08-09 16:14:33 +0000389 if dir is None:
390 dir = tempfile.gettempdir()
391 try:
392 (fd, name) = tempfile.mkstemp(dir=dir, prefix=pre, suffix=suf)
Martin v. Löwisd6625482003-10-12 17:37:01 +0000393 (ndir, nbase) = os.path.split(name)
394 adir = os.path.abspath(dir)
395 self.assertEqual(adir, ndir,
396 "Directory '%s' incorrectly returned as '%s'" % (adir, ndir))
Guido van Rossum0e548712002-08-09 16:14:33 +0000397 except:
398 self.failOnException("mkstemp")
399
400 try:
401 self.nameCheck(name, dir, pre, suf)
402 finally:
403 os.close(fd)
404 os.unlink(name)
405
406 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000407 # mkstemp can create files
Guido van Rossum0e548712002-08-09 16:14:33 +0000408 self.do_create()
409 self.do_create(pre="a")
410 self.do_create(suf="b")
411 self.do_create(pre="a", suf="b")
412 self.do_create(pre="aa", suf=".txt")
Martin v. Löwisd6625482003-10-12 17:37:01 +0000413 self.do_create(dir=".")
Guido van Rossum0e548712002-08-09 16:14:33 +0000414
415 def test_choose_directory(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000416 # mkstemp can create directories in a user-selected directory
Guido van Rossum0e548712002-08-09 16:14:33 +0000417 dir = tempfile.mkdtemp()
418 try:
419 self.do_create(dir=dir)
420 finally:
421 os.rmdir(dir)
422
423test_classes.append(test_mkstemp)
424
425
426class test_mkdtemp(TC):
427 """Test mkdtemp()."""
428
429 def do_create(self, dir=None, pre="", suf=""):
430 if dir is None:
431 dir = tempfile.gettempdir()
432 try:
433 name = tempfile.mkdtemp(dir=dir, prefix=pre, suffix=suf)
434 except:
435 self.failOnException("mkdtemp")
436
437 try:
438 self.nameCheck(name, dir, pre, suf)
439 return name
440 except:
441 os.rmdir(name)
442 raise
443
444 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000445 # mkdtemp can create directories
Guido van Rossum0e548712002-08-09 16:14:33 +0000446 os.rmdir(self.do_create())
447 os.rmdir(self.do_create(pre="a"))
448 os.rmdir(self.do_create(suf="b"))
449 os.rmdir(self.do_create(pre="a", suf="b"))
450 os.rmdir(self.do_create(pre="aa", suf=".txt"))
Tim Petersa0d55de2002-08-09 18:01:01 +0000451
Guido van Rossum0e548712002-08-09 16:14:33 +0000452 def test_basic_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000453 # mkdtemp can create many directories (stochastic)
Neal Norwitz68ee0122002-08-16 19:28:59 +0000454 extant = range(TEST_FILES)
Guido van Rossum0e548712002-08-09 16:14:33 +0000455 try:
456 for i in extant:
457 extant[i] = self.do_create(pre="aa")
458 finally:
459 for i in extant:
460 if(isinstance(i, basestring)):
461 os.rmdir(i)
462
463 def test_choose_directory(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000464 # mkdtemp can create directories in a user-selected directory
Guido van Rossum0e548712002-08-09 16:14:33 +0000465 dir = tempfile.mkdtemp()
466 try:
467 os.rmdir(self.do_create(dir=dir))
468 finally:
469 os.rmdir(dir)
470
471 def test_mode(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000472 # mkdtemp creates directories with the proper mode
Guido van Rossum0e548712002-08-09 16:14:33 +0000473 if not has_stat:
Benjamin Peterson888a39b2009-03-26 20:48:25 +0000474 return # ugh, can't use SkipTest.
Guido van Rossum0e548712002-08-09 16:14:33 +0000475
476 dir = self.do_create()
477 try:
478 mode = stat.S_IMODE(os.stat(dir).st_mode)
Guido van Rossum59db96f2004-03-31 18:53:29 +0000479 mode &= 0777 # Mask off sticky bits inherited from /tmp
Tim Petersca3ac7f2002-08-09 18:13:51 +0000480 expected = 0700
Jack Jansence921472003-01-08 16:30:34 +0000481 if sys.platform in ('win32', 'os2emx', 'mac'):
Tim Petersca3ac7f2002-08-09 18:13:51 +0000482 # There's no distinction among 'user', 'group' and 'world';
483 # replicate the 'user' bits.
484 user = expected >> 6
485 expected = user * (1 + 8 + 64)
486 self.assertEqual(mode, expected)
Guido van Rossum0e548712002-08-09 16:14:33 +0000487 finally:
488 os.rmdir(dir)
489
490test_classes.append(test_mkdtemp)
491
492
493class test_mktemp(TC):
494 """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()
Guido van Rossum0e548712002-08-09 16:14:33 +0000500
501 def tearDown(self):
502 if self.dir:
503 os.rmdir(self.dir)
504 self.dir = None
Guido van Rossum0e548712002-08-09 16:14:33 +0000505
506 class mktemped:
507 _unlink = os.unlink
508 _bflags = tempfile._bin_openflags
509
510 def __init__(self, dir, pre, suf):
511 self.name = tempfile.mktemp(dir=dir, prefix=pre, suffix=suf)
512 # Create the file. This will raise an exception if it's
513 # mysteriously appeared in the meanwhile.
514 os.close(os.open(self.name, self._bflags, 0600))
515
516 def __del__(self):
517 self._unlink(self.name)
518
519 def do_create(self, pre="", suf=""):
520 try:
521 file = self.mktemped(self.dir, pre, suf)
522 except:
523 self.failOnException("mktemp")
524
525 self.nameCheck(file.name, self.dir, pre, suf)
526 return file
527
528 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000529 # mktemp can choose usable file names
Guido van Rossum0e548712002-08-09 16:14:33 +0000530 self.do_create()
531 self.do_create(pre="a")
532 self.do_create(suf="b")
533 self.do_create(pre="a", suf="b")
534 self.do_create(pre="aa", suf=".txt")
535
536 def test_many(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000537 # mktemp can choose many usable file names (stochastic)
Neal Norwitz68ee0122002-08-16 19:28:59 +0000538 extant = range(TEST_FILES)
Guido van Rossum0e548712002-08-09 16:14:33 +0000539 for i in extant:
540 extant[i] = self.do_create(pre="aa")
541
Fred Drake8bec4832002-11-22 20:13:43 +0000542## def test_warning(self):
543## # mktemp issues a warning when used
544## warnings.filterwarnings("error",
545## category=RuntimeWarning,
546## message="mktemp")
547## self.assertRaises(RuntimeWarning,
548## tempfile.mktemp, dir=self.dir)
Guido van Rossum0e548712002-08-09 16:14:33 +0000549
550test_classes.append(test_mktemp)
551
552
553# We test _TemporaryFileWrapper by testing NamedTemporaryFile.
554
555
556class test_NamedTemporaryFile(TC):
557 """Test NamedTemporaryFile()."""
558
Georg Brandl35ef9c12007-03-13 18:31:49 +0000559 def do_create(self, dir=None, pre="", suf="", delete=True):
Guido van Rossum0e548712002-08-09 16:14:33 +0000560 if dir is None:
561 dir = tempfile.gettempdir()
562 try:
Georg Brandl35ef9c12007-03-13 18:31:49 +0000563 file = tempfile.NamedTemporaryFile(dir=dir, prefix=pre, suffix=suf,
Georg Brandl4168c042007-03-13 19:18:18 +0000564 delete=delete)
Guido van Rossum0e548712002-08-09 16:14:33 +0000565 except:
566 self.failOnException("NamedTemporaryFile")
567
568 self.nameCheck(file.name, dir, pre, suf)
569 return file
570
571
572 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000573 # NamedTemporaryFile can create files
Guido van Rossum0e548712002-08-09 16:14:33 +0000574 self.do_create()
575 self.do_create(pre="a")
576 self.do_create(suf="b")
577 self.do_create(pre="a", suf="b")
578 self.do_create(pre="aa", suf=".txt")
579
580 def test_creates_named(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000581 # NamedTemporaryFile creates files with names
Guido van Rossum0e548712002-08-09 16:14:33 +0000582 f = tempfile.NamedTemporaryFile()
583 self.failUnless(os.path.exists(f.name),
584 "NamedTemporaryFile %s does not exist" % f.name)
585
586 def test_del_on_close(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000587 # A NamedTemporaryFile is deleted when closed
Guido van Rossum0e548712002-08-09 16:14:33 +0000588 dir = tempfile.mkdtemp()
589 try:
590 f = tempfile.NamedTemporaryFile(dir=dir)
591 f.write('blat')
592 f.close()
593 self.failIf(os.path.exists(f.name),
594 "NamedTemporaryFile %s exists after close" % f.name)
595 finally:
596 os.rmdir(dir)
597
Georg Brandl35ef9c12007-03-13 18:31:49 +0000598 def test_dis_del_on_close(self):
599 # Tests that delete-on-close can be disabled
600 dir = tempfile.mkdtemp()
Georg Brandl4168c042007-03-13 19:18:18 +0000601 tmp = None
Georg Brandl35ef9c12007-03-13 18:31:49 +0000602 try:
603 f = tempfile.NamedTemporaryFile(dir=dir, delete=False)
604 tmp = f.name
605 f.write('blat')
606 f.close()
607 self.failUnless(os.path.exists(f.name),
608 "NamedTemporaryFile %s missing after close" % f.name)
609 finally:
610 if tmp is not None:
611 os.unlink(tmp)
612 os.rmdir(dir)
613
Guido van Rossum0e548712002-08-09 16:14:33 +0000614 def test_multiple_close(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000615 # A NamedTemporaryFile can be closed many times without error
Guido van Rossum0e548712002-08-09 16:14:33 +0000616 f = tempfile.NamedTemporaryFile()
617 f.write('abc\n')
618 f.close()
619 try:
620 f.close()
621 f.close()
622 except:
623 self.failOnException("close")
624
Nick Coghlan97fac3e2008-02-09 15:28:09 +0000625 def test_context_manager(self):
626 # A NamedTemporaryFile can be used as a context manager
627 with tempfile.NamedTemporaryFile() as f:
628 self.failUnless(os.path.exists(f.name))
629 self.failIf(os.path.exists(f.name))
630 def use_closed():
631 with f:
632 pass
633 self.failUnlessRaises(ValueError, use_closed)
634
Guido van Rossum0e548712002-08-09 16:14:33 +0000635 # How to test the mode and bufsize parameters?
636
637test_classes.append(test_NamedTemporaryFile)
638
Collin Wintera8785cc2007-03-19 18:52:08 +0000639class test_SpooledTemporaryFile(TC):
640 """Test SpooledTemporaryFile()."""
641
642 def do_create(self, max_size=0, dir=None, pre="", suf=""):
643 if dir is None:
644 dir = tempfile.gettempdir()
645 try:
646 file = tempfile.SpooledTemporaryFile(max_size=max_size, dir=dir, prefix=pre, suffix=suf)
647 except:
648 self.failOnException("SpooledTemporaryFile")
649
650 return file
651
652
653 def test_basic(self):
654 # SpooledTemporaryFile can create files
655 f = self.do_create()
656 self.failIf(f._rolled)
657 f = self.do_create(max_size=100, pre="a", suf=".txt")
658 self.failIf(f._rolled)
659
660 def test_del_on_close(self):
661 # A SpooledTemporaryFile is deleted when closed
662 dir = tempfile.mkdtemp()
663 try:
664 f = tempfile.SpooledTemporaryFile(max_size=10, dir=dir)
665 self.failIf(f._rolled)
666 f.write('blat ' * 5)
667 self.failUnless(f._rolled)
668 filename = f.name
669 f.close()
670 self.failIf(os.path.exists(filename),
671 "SpooledTemporaryFile %s exists after close" % filename)
672 finally:
673 os.rmdir(dir)
674
675 def test_rewrite_small(self):
676 # A SpooledTemporaryFile can be written to multiple within the max_size
677 f = self.do_create(max_size=30)
678 self.failIf(f._rolled)
679 for i in range(5):
680 f.seek(0, 0)
681 f.write('x' * 20)
682 self.failIf(f._rolled)
683
684 def test_write_sequential(self):
685 # A SpooledTemporaryFile should hold exactly max_size bytes, and roll
686 # over afterward
687 f = self.do_create(max_size=30)
688 self.failIf(f._rolled)
689 f.write('x' * 20)
690 self.failIf(f._rolled)
691 f.write('x' * 10)
692 self.failIf(f._rolled)
693 f.write('x')
694 self.failUnless(f._rolled)
695
696 def test_sparse(self):
697 # A SpooledTemporaryFile that is written late in the file will extend
698 # when that occurs
699 f = self.do_create(max_size=30)
700 self.failIf(f._rolled)
701 f.seek(100, 0)
702 self.failIf(f._rolled)
703 f.write('x')
704 self.failUnless(f._rolled)
705
706 def test_fileno(self):
707 # A SpooledTemporaryFile should roll over to a real file on fileno()
708 f = self.do_create(max_size=30)
709 self.failIf(f._rolled)
710 self.failUnless(f.fileno() > 0)
711 self.failUnless(f._rolled)
712
Nick Coghlan97fac3e2008-02-09 15:28:09 +0000713 def test_multiple_close_before_rollover(self):
Collin Wintera8785cc2007-03-19 18:52:08 +0000714 # A SpooledTemporaryFile can be closed many times without error
715 f = tempfile.SpooledTemporaryFile()
716 f.write('abc\n')
Nick Coghlan97fac3e2008-02-09 15:28:09 +0000717 self.failIf(f._rolled)
718 f.close()
719 try:
720 f.close()
721 f.close()
722 except:
723 self.failOnException("close")
724
725 def test_multiple_close_after_rollover(self):
726 # A SpooledTemporaryFile can be closed many times without error
727 f = tempfile.SpooledTemporaryFile(max_size=1)
728 f.write('abc\n')
729 self.failUnless(f._rolled)
Collin Wintera8785cc2007-03-19 18:52:08 +0000730 f.close()
731 try:
732 f.close()
733 f.close()
734 except:
735 self.failOnException("close")
736
737 def test_bound_methods(self):
738 # It should be OK to steal a bound method from a SpooledTemporaryFile
739 # and use it independently; when the file rolls over, those bound
740 # methods should continue to function
741 f = self.do_create(max_size=30)
742 read = f.read
743 write = f.write
744 seek = f.seek
745
746 write("a" * 35)
747 write("b" * 35)
748 seek(0, 0)
749 self.failUnless(read(70) == 'a'*35 + 'b'*35)
750
Nick Coghlan97fac3e2008-02-09 15:28:09 +0000751 def test_context_manager_before_rollover(self):
752 # A SpooledTemporaryFile can be used as a context manager
753 with tempfile.SpooledTemporaryFile(max_size=1) as f:
754 self.failIf(f._rolled)
755 self.failIf(f.closed)
756 self.failUnless(f.closed)
757 def use_closed():
758 with f:
759 pass
760 self.failUnlessRaises(ValueError, use_closed)
761
762 def test_context_manager_during_rollover(self):
763 # A SpooledTemporaryFile can be used as a context manager
764 with tempfile.SpooledTemporaryFile(max_size=1) as f:
765 self.failIf(f._rolled)
766 f.write('abc\n')
767 f.flush()
768 self.failUnless(f._rolled)
769 self.failIf(f.closed)
770 self.failUnless(f.closed)
771 def use_closed():
772 with f:
773 pass
774 self.failUnlessRaises(ValueError, use_closed)
775
776 def test_context_manager_after_rollover(self):
777 # A SpooledTemporaryFile can be used as a context manager
778 f = tempfile.SpooledTemporaryFile(max_size=1)
779 f.write('abc\n')
780 f.flush()
781 self.failUnless(f._rolled)
782 with f:
783 self.failIf(f.closed)
784 self.failUnless(f.closed)
785 def use_closed():
786 with f:
787 pass
788 self.failUnlessRaises(ValueError, use_closed)
789
790
Collin Wintera8785cc2007-03-19 18:52:08 +0000791test_classes.append(test_SpooledTemporaryFile)
792
Guido van Rossum0e548712002-08-09 16:14:33 +0000793
794class test_TemporaryFile(TC):
795 """Test TemporaryFile()."""
796
797 def test_basic(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000798 # TemporaryFile can create files
Guido van Rossum0e548712002-08-09 16:14:33 +0000799 # No point in testing the name params - the file has no name.
800 try:
801 tempfile.TemporaryFile()
802 except:
803 self.failOnException("TemporaryFile")
804
805 def test_has_no_name(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000806 # TemporaryFile creates files with no names (on this system)
Guido van Rossum0e548712002-08-09 16:14:33 +0000807 dir = tempfile.mkdtemp()
808 f = tempfile.TemporaryFile(dir=dir)
809 f.write('blat')
810
811 # Sneaky: because this file has no name, it should not prevent
812 # us from removing the directory it was created in.
813 try:
814 os.rmdir(dir)
815 except:
816 ei = sys.exc_info()
817 # cleanup
818 f.close()
819 os.rmdir(dir)
820 self.failOnException("rmdir", ei)
821
822 def test_multiple_close(self):
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000823 # A TemporaryFile can be closed many times without error
Guido van Rossum0e548712002-08-09 16:14:33 +0000824 f = tempfile.TemporaryFile()
825 f.write('abc\n')
826 f.close()
827 try:
828 f.close()
829 f.close()
830 except:
831 self.failOnException("close")
832
833 # How to test the mode and bufsize parameters?
834
Guido van Rossum0e548712002-08-09 16:14:33 +0000835
Guido van Rossum8ccd9b62002-08-22 20:02:03 +0000836if tempfile.NamedTemporaryFile is not tempfile.TemporaryFile:
Guido van Rossum0e548712002-08-09 16:14:33 +0000837 test_classes.append(test_TemporaryFile)
838
839def test_main():
Walter Dörwald21d3a322003-05-01 17:45:56 +0000840 test_support.run_unittest(*test_classes)
Guido van Rossum0e548712002-08-09 16:14:33 +0000841
842if __name__ == "__main__":
843 test_main()