blob: 2b61bce99cc82dc6143d886cac1d3e1de703f459 [file] [log] [blame]
Guido van Rossum0e548712002-08-09 16:14:33 +00001# tempfile.py unit tests.
Tim Petersc57a2852001-10-29 21:46:08 +00002
3import tempfile
Guido van Rossum0e548712002-08-09 16:14:33 +00004import os
5import sys
6import re
7import errno
8import warnings
Tim Petersc57a2852001-10-29 21:46:08 +00009
Guido van Rossum0e548712002-08-09 16:14:33 +000010import unittest
11from test import test_support
12
13if hasattr(os, 'stat'):
14 import stat
15 has_stat = 1
16else:
17 has_stat = 0
18
19has_textmode = (tempfile._text_openflags != tempfile._bin_openflags)
Guido van Rossum78741062002-08-17 11:41:01 +000020has_spawnl = hasattr(os, 'spawnl')
Guido van Rossum0e548712002-08-09 16:14:33 +000021
Neal Norwitz68ee0122002-08-16 19:28:59 +000022# TEST_FILES may need to be tweaked for systems depending on the maximum
23# number of files that can be opened at one time (see ulimit -n)
24TEST_FILES = 100
25
Guido van Rossum0e548712002-08-09 16:14:33 +000026# This is organized as one test for each chunk of code in tempfile.py,
27# in order of their appearance in the file. Testing which requires
28# threads is not done here.
29
30# Common functionality.
31class TC(unittest.TestCase):
32
33 str_check = re.compile(r"[a-zA-Z0-9_-]{6}$")
34
35 def failOnException(self, what, ei=None):
36 if ei is None:
37 ei = sys.exc_info()
38 self.fail("%s raised %s: %s" % (what, ei[0], ei[1]))
39
40 def nameCheck(self, name, dir, pre, suf):
41 (ndir, nbase) = os.path.split(name)
42 npre = nbase[:len(pre)]
43 nsuf = nbase[len(nbase)-len(suf):]
44
45 self.assertEqual(ndir, dir,
46 "file '%s' not in directory '%s'" % (name, dir))
47 self.assertEqual(npre, pre,
48 "file '%s' does not begin with '%s'" % (nbase, pre))
49 self.assertEqual(nsuf, suf,
50 "file '%s' does not end with '%s'" % (nbase, suf))
51
52 nbase = nbase[len(pre):len(nbase)-len(suf)]
53 self.assert_(self.str_check.match(nbase),
54 "random string '%s' does not match /^[a-zA-Z0-9_-]{6}$/"
55 % nbase)
56
57test_classes = []
58
59class test_exports(TC):
60 def test_exports(self):
61 """There are no surprising symbols in the tempfile module"""
62 dict = tempfile.__dict__
63
64 expected = {
65 "NamedTemporaryFile" : 1,
66 "TemporaryFile" : 1,
67 "mkstemp" : 1,
68 "mkdtemp" : 1,
69 "mktemp" : 1,
70 "TMP_MAX" : 1,
71 "gettempprefix" : 1,
72 "gettempdir" : 1,
73 "tempdir" : 1,
74 "template" : 1
75 }
76
77 unexp = []
78 for key in dict:
79 if key[0] != '_' and key not in expected:
80 unexp.append(key)
81 self.failUnless(len(unexp) == 0,
82 "unexpected keys: %s" % unexp)
83
84test_classes.append(test_exports)
85
86
Guido van Rossum0e548712002-08-09 16:14:33 +000087class test__RandomNameSequence(TC):
88 """Test the internal iterator object _RandomNameSequence."""
89
90 def setUp(self):
91 self.r = tempfile._RandomNameSequence()
92
93 def test_get_six_char_str(self):
94 """_RandomNameSequence returns a six-character string"""
95 s = self.r.next()
96 self.nameCheck(s, '', '', '')
97
98 def test_many(self):
99 """_RandomNameSequence returns no duplicate strings (stochastic)"""
100
101 dict = {}
102 r = self.r
Neal Norwitz68ee0122002-08-16 19:28:59 +0000103 for i in xrange(TEST_FILES):
Guido van Rossum0e548712002-08-09 16:14:33 +0000104 s = r.next()
105 self.nameCheck(s, '', '', '')
106 self.failIf(s in dict)
107 dict[s] = 1
108
109 def test_supports_iter(self):
110 """_RandomNameSequence supports the iterator protocol"""
111
112 i = 0
113 r = self.r
114 try:
115 for s in r:
116 i += 1
117 if i == 20:
118 break
119 except:
120 failOnException("iteration")
121
122test_classes.append(test__RandomNameSequence)
123
124
125class test__candidate_tempdir_list(TC):
126 """Test the internal function _candidate_tempdir_list."""
127
128 def test_nonempty_list(self):
129 """_candidate_tempdir_list returns a nonempty list of strings"""
130
131 cand = tempfile._candidate_tempdir_list()
132
133 self.failIf(len(cand) == 0)
134 for c in cand:
135 self.assert_(isinstance(c, basestring),
136 "%s is not a string" % c)
137
138 def test_wanted_dirs(self):
139 """_candidate_tempdir_list contains the expected directories"""
140
141 # Make sure the interesting environment variables are all set.
142 added = []
143 try:
144 for envname in 'TMPDIR', 'TEMP', 'TMP':
145 dirname = os.getenv(envname)
146 if not dirname:
147 os.environ[envname] = os.path.abspath(envname)
148 added.append(envname)
149
150 cand = tempfile._candidate_tempdir_list()
151
152 for envname in 'TMPDIR', 'TEMP', 'TMP':
153 dirname = os.getenv(envname)
154 if not dirname: raise ValueError
155 self.assert_(dirname in cand)
156
157 try:
158 dirname = os.getcwd()
159 except (AttributeError, os.error):
160 dirname = os.curdir
161
162 self.assert_(dirname in cand)
163
164 # Not practical to try to verify the presence of OS-specific
165 # paths in this list.
166 finally:
167 for p in added:
168 del os.environ[p]
169
170test_classes.append(test__candidate_tempdir_list)
171
172
173# We test _get_default_tempdir by testing gettempdir.
174
175
176class test__get_candidate_names(TC):
177 """Test the internal function _get_candidate_names."""
178
179 def test_retval(self):
180 """_get_candidate_names returns a _RandomNameSequence object"""
181 obj = tempfile._get_candidate_names()
182 self.assert_(isinstance(obj, tempfile._RandomNameSequence))
183
184 def test_same_thing(self):
185 """_get_candidate_names always returns the same object"""
186 a = tempfile._get_candidate_names()
187 b = tempfile._get_candidate_names()
188
189 self.assert_(a is b)
190
191test_classes.append(test__get_candidate_names)
192
193
194class test__mkstemp_inner(TC):
195 """Test the internal function _mkstemp_inner."""
196
197 class mkstemped:
198 _bflags = tempfile._bin_openflags
199 _tflags = tempfile._text_openflags
200 _close = os.close
201 _unlink = os.unlink
202
203 def __init__(self, dir, pre, suf, bin):
204 if bin: flags = self._bflags
205 else: flags = self._tflags
206
207 (self.fd, self.name) = tempfile._mkstemp_inner(dir, pre, suf, flags)
208
209 def write(self, str):
210 os.write(self.fd, str)
211
212 def __del__(self):
213 self._close(self.fd)
214 self._unlink(self.name)
Tim Petersa0d55de2002-08-09 18:01:01 +0000215
Guido van Rossum0e548712002-08-09 16:14:33 +0000216 def do_create(self, dir=None, pre="", suf="", bin=1):
217 if dir is None:
218 dir = tempfile.gettempdir()
219 try:
220 file = self.mkstemped(dir, pre, suf, bin)
221 except:
222 self.failOnException("_mkstemp_inner")
223
224 self.nameCheck(file.name, dir, pre, suf)
225 return file
226
227 def test_basic(self):
228 """_mkstemp_inner can create files"""
229 self.do_create().write("blat")
230 self.do_create(pre="a").write("blat")
231 self.do_create(suf="b").write("blat")
232 self.do_create(pre="a", suf="b").write("blat")
233 self.do_create(pre="aa", suf=".txt").write("blat")
234
235 def test_basic_many(self):
236 """_mkstemp_inner can create many files (stochastic)"""
Neal Norwitz68ee0122002-08-16 19:28:59 +0000237 extant = range(TEST_FILES)
Guido van Rossum0e548712002-08-09 16:14:33 +0000238 for i in extant:
239 extant[i] = self.do_create(pre="aa")
240
241 def test_choose_directory(self):
242 """_mkstemp_inner can create files in a user-selected directory"""
243 dir = tempfile.mkdtemp()
244 try:
245 self.do_create(dir=dir).write("blat")
246 finally:
247 os.rmdir(dir)
248
249 def test_file_mode(self):
250 """_mkstemp_inner creates files with the proper mode"""
251 if not has_stat:
252 return # ugh, can't use TestSkipped.
253
254 file = self.do_create()
255 mode = stat.S_IMODE(os.stat(file.name).st_mode)
Tim Petersca3ac7f2002-08-09 18:13:51 +0000256 expected = 0600
Andrew MacIntyre1d0eeec2002-08-18 06:47:19 +0000257 if sys.platform in ('win32', 'os2emx'):
Tim Petersca3ac7f2002-08-09 18:13:51 +0000258 # There's no distinction among 'user', 'group' and 'world';
259 # replicate the 'user' bits.
260 user = expected >> 6
261 expected = user * (1 + 8 + 64)
262 self.assertEqual(mode, expected)
Guido van Rossum0e548712002-08-09 16:14:33 +0000263
264 def test_noinherit(self):
265 """_mkstemp_inner file handles are not inherited by child processes"""
Guido van Rossum78741062002-08-17 11:41:01 +0000266 if not has_spawnl:
Guido van Rossum0e548712002-08-09 16:14:33 +0000267 return # ugh, can't use TestSkipped.
268
Guido van Rossum78741062002-08-17 11:41:01 +0000269 if test_support.verbose:
270 v="v"
271 else:
272 v="q"
273
Guido van Rossum0e548712002-08-09 16:14:33 +0000274 file = self.do_create()
Guido van Rossum78741062002-08-17 11:41:01 +0000275 fd = "%d" % file.fd
276
277 try:
278 me = __file__
279 except NameError:
280 me = sys.argv[0]
Guido van Rossum0e548712002-08-09 16:14:33 +0000281
282 # We have to exec something, so that FD_CLOEXEC will take
Guido van Rossum78741062002-08-17 11:41:01 +0000283 # effect. The core of this test is therefore in
284 # tf_inherit_check.py, which see.
285 tester = os.path.join(os.path.dirname(os.path.abspath(me)),
286 "tf_inherit_check.py")
Guido van Rossum0e548712002-08-09 16:14:33 +0000287
Guido van Rossum78741062002-08-17 11:41:01 +0000288 retval = os.spawnl(os.P_WAIT, sys.executable,
289 sys.executable, tester, v, fd)
290 self.failIf(retval < 0,
291 "child process caught fatal signal %d" % -retval)
292 self.failIf(retval > 0, "child process reports failure")
Guido van Rossum0e548712002-08-09 16:14:33 +0000293
294 def test_textmode(self):
295 """_mkstemp_inner can create files in text mode"""
296 if not has_textmode:
297 return # ugh, can't use TestSkipped.
298
299 self.do_create(bin=0).write("blat\n")
300 # XXX should test that the file really is a text file
301
302test_classes.append(test__mkstemp_inner)
303
304
305class test_gettempprefix(TC):
306 """Test gettempprefix()."""
307
308 def test_sane_template(self):
309 """gettempprefix returns a nonempty prefix string"""
310 p = tempfile.gettempprefix()
311
312 self.assert_(isinstance(p, basestring))
313 self.assert_(len(p) > 0)
314
315 def test_usable_template(self):
316 """gettempprefix returns a usable prefix string"""
317
318 # Create a temp directory, avoiding use of the prefix.
319 # Then attempt to create a file whose name is
320 # prefix + 'xxxxxx.xxx' in that directory.
321 p = tempfile.gettempprefix() + "xxxxxx.xxx"
322 d = tempfile.mkdtemp(prefix="")
323 try:
324 p = os.path.join(d, p)
325 try:
326 fd = os.open(p, os.O_RDWR | os.O_CREAT)
327 except:
328 self.failOnException("os.open")
329 os.close(fd)
330 os.unlink(p)
331 finally:
332 os.rmdir(d)
333
334test_classes.append(test_gettempprefix)
335
336
337class test_gettempdir(TC):
338 """Test gettempdir()."""
339
340 def test_directory_exists(self):
341 """gettempdir returns a directory which exists"""
342
343 dir = tempfile.gettempdir()
344 self.assert_(os.path.isabs(dir) or dir == os.curdir,
345 "%s is not an absolute path" % dir)
346 self.assert_(os.path.isdir(dir),
347 "%s is not a directory" % dir)
348
349 def test_directory_writable(self):
350 """gettempdir returns a directory writable by the user"""
351
352 # sneaky: just instantiate a NamedTemporaryFile, which
353 # defaults to writing into the directory returned by
354 # gettempdir.
355 try:
356 file = tempfile.NamedTemporaryFile()
357 file.write("blat")
358 file.close()
359 except:
360 self.failOnException("create file in %s" % tempfile.gettempdir())
361
362 def test_same_thing(self):
363 """gettempdir always returns the same object"""
364 a = tempfile.gettempdir()
365 b = tempfile.gettempdir()
366
367 self.assert_(a is b)
368
369test_classes.append(test_gettempdir)
370
371
372class test_mkstemp(TC):
373 """Test mkstemp()."""
374 def do_create(self, dir=None, pre="", suf="", ):
375 if dir is None:
376 dir = tempfile.gettempdir()
377 try:
378 (fd, name) = tempfile.mkstemp(dir=dir, prefix=pre, suffix=suf)
379 except:
380 self.failOnException("mkstemp")
381
382 try:
383 self.nameCheck(name, dir, pre, suf)
384 finally:
385 os.close(fd)
386 os.unlink(name)
387
388 def test_basic(self):
389 """mkstemp can create files"""
390 self.do_create()
391 self.do_create(pre="a")
392 self.do_create(suf="b")
393 self.do_create(pre="a", suf="b")
394 self.do_create(pre="aa", suf=".txt")
395
396 def test_choose_directory(self):
397 """mkstemp can create directories in a user-selected directory"""
398 dir = tempfile.mkdtemp()
399 try:
400 self.do_create(dir=dir)
401 finally:
402 os.rmdir(dir)
403
404test_classes.append(test_mkstemp)
405
406
407class test_mkdtemp(TC):
408 """Test mkdtemp()."""
409
410 def do_create(self, dir=None, pre="", suf=""):
411 if dir is None:
412 dir = tempfile.gettempdir()
413 try:
414 name = tempfile.mkdtemp(dir=dir, prefix=pre, suffix=suf)
415 except:
416 self.failOnException("mkdtemp")
417
418 try:
419 self.nameCheck(name, dir, pre, suf)
420 return name
421 except:
422 os.rmdir(name)
423 raise
424
425 def test_basic(self):
426 """mkdtemp can create directories"""
427 os.rmdir(self.do_create())
428 os.rmdir(self.do_create(pre="a"))
429 os.rmdir(self.do_create(suf="b"))
430 os.rmdir(self.do_create(pre="a", suf="b"))
431 os.rmdir(self.do_create(pre="aa", suf=".txt"))
Tim Petersa0d55de2002-08-09 18:01:01 +0000432
Guido van Rossum0e548712002-08-09 16:14:33 +0000433 def test_basic_many(self):
434 """mkdtemp can create many directories (stochastic)"""
Neal Norwitz68ee0122002-08-16 19:28:59 +0000435 extant = range(TEST_FILES)
Guido van Rossum0e548712002-08-09 16:14:33 +0000436 try:
437 for i in extant:
438 extant[i] = self.do_create(pre="aa")
439 finally:
440 for i in extant:
441 if(isinstance(i, basestring)):
442 os.rmdir(i)
443
444 def test_choose_directory(self):
445 """mkdtemp can create directories in a user-selected directory"""
446 dir = tempfile.mkdtemp()
447 try:
448 os.rmdir(self.do_create(dir=dir))
449 finally:
450 os.rmdir(dir)
451
452 def test_mode(self):
453 """mkdtemp creates directories with the proper mode"""
454 if not has_stat:
455 return # ugh, can't use TestSkipped.
456
457 dir = self.do_create()
458 try:
459 mode = stat.S_IMODE(os.stat(dir).st_mode)
Tim Petersca3ac7f2002-08-09 18:13:51 +0000460 expected = 0700
Andrew MacIntyre1d0eeec2002-08-18 06:47:19 +0000461 if sys.platform in ('win32', 'os2emx'):
Tim Petersca3ac7f2002-08-09 18:13:51 +0000462 # There's no distinction among 'user', 'group' and 'world';
463 # replicate the 'user' bits.
464 user = expected >> 6
465 expected = user * (1 + 8 + 64)
466 self.assertEqual(mode, expected)
Guido van Rossum0e548712002-08-09 16:14:33 +0000467 finally:
468 os.rmdir(dir)
469
470test_classes.append(test_mkdtemp)
471
472
473class test_mktemp(TC):
474 """Test mktemp()."""
475
476 # For safety, all use of mktemp must occur in a private directory.
477 # We must also suppress the RuntimeWarning it generates.
478 def setUp(self):
479 self.dir = tempfile.mkdtemp()
480 warnings.filterwarnings("ignore",
481 category=RuntimeWarning,
482 message="mktemp")
483
484 def tearDown(self):
485 if self.dir:
486 os.rmdir(self.dir)
487 self.dir = None
488 # XXX This clobbers any -W options.
489 warnings.resetwarnings()
490
491 class mktemped:
492 _unlink = os.unlink
493 _bflags = tempfile._bin_openflags
494
495 def __init__(self, dir, pre, suf):
496 self.name = tempfile.mktemp(dir=dir, prefix=pre, suffix=suf)
497 # Create the file. This will raise an exception if it's
498 # mysteriously appeared in the meanwhile.
499 os.close(os.open(self.name, self._bflags, 0600))
500
501 def __del__(self):
502 self._unlink(self.name)
503
504 def do_create(self, pre="", suf=""):
505 try:
506 file = self.mktemped(self.dir, pre, suf)
507 except:
508 self.failOnException("mktemp")
509
510 self.nameCheck(file.name, self.dir, pre, suf)
511 return file
512
513 def test_basic(self):
514 """mktemp can choose usable file names"""
515 self.do_create()
516 self.do_create(pre="a")
517 self.do_create(suf="b")
518 self.do_create(pre="a", suf="b")
519 self.do_create(pre="aa", suf=".txt")
520
521 def test_many(self):
522 """mktemp can choose many usable file names (stochastic)"""
Neal Norwitz68ee0122002-08-16 19:28:59 +0000523 extant = range(TEST_FILES)
Guido van Rossum0e548712002-08-09 16:14:33 +0000524 for i in extant:
525 extant[i] = self.do_create(pre="aa")
526
527 def test_warning(self):
528 """mktemp issues a warning when used"""
529 warnings.filterwarnings("error",
530 category=RuntimeWarning,
531 message="mktemp")
532 self.assertRaises(RuntimeWarning,
533 tempfile.mktemp, (), { 'dir': self.dir })
534
535test_classes.append(test_mktemp)
536
537
538# We test _TemporaryFileWrapper by testing NamedTemporaryFile.
539
540
541class test_NamedTemporaryFile(TC):
542 """Test NamedTemporaryFile()."""
543
544 def do_create(self, dir=None, pre="", suf=""):
545 if dir is None:
546 dir = tempfile.gettempdir()
547 try:
548 file = tempfile.NamedTemporaryFile(dir=dir, prefix=pre, suffix=suf)
549 except:
550 self.failOnException("NamedTemporaryFile")
551
552 self.nameCheck(file.name, dir, pre, suf)
553 return file
554
555
556 def test_basic(self):
557 """NamedTemporaryFile can create files"""
558 self.do_create()
559 self.do_create(pre="a")
560 self.do_create(suf="b")
561 self.do_create(pre="a", suf="b")
562 self.do_create(pre="aa", suf=".txt")
563
564 def test_creates_named(self):
565 """NamedTemporaryFile creates files with names"""
566 f = tempfile.NamedTemporaryFile()
567 self.failUnless(os.path.exists(f.name),
568 "NamedTemporaryFile %s does not exist" % f.name)
569
570 def test_del_on_close(self):
571 """A NamedTemporaryFile is deleted when closed"""
572 dir = tempfile.mkdtemp()
573 try:
574 f = tempfile.NamedTemporaryFile(dir=dir)
575 f.write('blat')
576 f.close()
577 self.failIf(os.path.exists(f.name),
578 "NamedTemporaryFile %s exists after close" % f.name)
579 finally:
580 os.rmdir(dir)
581
582 def test_multiple_close(self):
583 """A NamedTemporaryFile can be closed many times without error"""
584
585 f = tempfile.NamedTemporaryFile()
586 f.write('abc\n')
587 f.close()
588 try:
589 f.close()
590 f.close()
591 except:
592 self.failOnException("close")
593
594 # How to test the mode and bufsize parameters?
595
596test_classes.append(test_NamedTemporaryFile)
597
598
599class test_TemporaryFile(TC):
600 """Test TemporaryFile()."""
601
602 def test_basic(self):
603 """TemporaryFile can create files"""
604 # No point in testing the name params - the file has no name.
605 try:
606 tempfile.TemporaryFile()
607 except:
608 self.failOnException("TemporaryFile")
609
610 def test_has_no_name(self):
611 """TemporaryFile creates files with no names (on this system)"""
612 dir = tempfile.mkdtemp()
613 f = tempfile.TemporaryFile(dir=dir)
614 f.write('blat')
615
616 # Sneaky: because this file has no name, it should not prevent
617 # us from removing the directory it was created in.
618 try:
619 os.rmdir(dir)
620 except:
621 ei = sys.exc_info()
622 # cleanup
623 f.close()
624 os.rmdir(dir)
625 self.failOnException("rmdir", ei)
626
627 def test_multiple_close(self):
628 """A TemporaryFile can be closed many times without error"""
629 f = tempfile.TemporaryFile()
630 f.write('abc\n')
631 f.close()
632 try:
633 f.close()
634 f.close()
635 except:
636 self.failOnException("close")
637
638 # How to test the mode and bufsize parameters?
639
640class dummy_test_TemporaryFile(TC):
641 def test_dummy(self):
642 """TemporaryFile and NamedTemporaryFile are the same (on this system)"""
643 pass
644
645if tempfile.NamedTemporaryFile is tempfile.TemporaryFile:
646 test_classes.append(dummy_test_TemporaryFile)
647else:
648 test_classes.append(test_TemporaryFile)
649
650def test_main():
651 suite = unittest.TestSuite()
652 for c in test_classes:
653 suite.addTest(unittest.makeSuite(c))
654 test_support.run_suite(suite)
655
656if __name__ == "__main__":
657 test_main()