Commit e207e38c authored by Eli Bendersky's avatar Eli Bendersky

Close #18945: Add tests for tempfile name collision handling.

Patch by Vlad Shcherbina
parent 35532c86
...@@ -8,6 +8,7 @@ import shutil ...@@ -8,6 +8,7 @@ import shutil
import sys import sys
import re import re
import warnings import warnings
import contextlib
import unittest import unittest
from test import test_support as support from test import test_support as support
...@@ -270,6 +271,22 @@ class test__get_candidate_names(TC): ...@@ -270,6 +271,22 @@ class test__get_candidate_names(TC):
test_classes.append(test__get_candidate_names) test_classes.append(test__get_candidate_names)
@contextlib.contextmanager
def _inside_empty_temp_dir():
dir = tempfile.mkdtemp()
try:
with support.swap_attr(tempfile, 'tempdir', dir):
yield
finally:
support.rmtree(dir)
def _mock_candidate_names(*names):
return support.swap_attr(tempfile,
'_get_candidate_names',
lambda: iter(names))
class test__mkstemp_inner(TC): class test__mkstemp_inner(TC):
"""Test the internal function _mkstemp_inner.""" """Test the internal function _mkstemp_inner."""
...@@ -386,31 +403,36 @@ class test__mkstemp_inner(TC): ...@@ -386,31 +403,36 @@ class test__mkstemp_inner(TC):
self.do_create(bin=0).write("blat\n") self.do_create(bin=0).write("blat\n")
# XXX should test that the file really is a text file # XXX should test that the file really is a text file
def default_mkstemp_inner(self):
return tempfile._mkstemp_inner(tempfile.gettempdir(),
tempfile.template,
'',
tempfile._bin_openflags)
def test_collision_with_existing_file(self):
# _mkstemp_inner tries another name when a file with
# the chosen name already exists
with _inside_empty_temp_dir(), \
_mock_candidate_names('aaa', 'aaa', 'bbb'):
(fd1, name1) = self.default_mkstemp_inner()
os.close(fd1)
self.assertTrue(name1.endswith('aaa'))
(fd2, name2) = self.default_mkstemp_inner()
os.close(fd2)
self.assertTrue(name2.endswith('bbb'))
def test_collision_with_existing_directory(self): def test_collision_with_existing_directory(self):
# _mkstemp_inner tries another name when a directory with # _mkstemp_inner tries another name when a directory with
# the chosen name already exists # the chosen name already exists
container_dir = tempfile.mkdtemp() with _inside_empty_temp_dir(), \
try: _mock_candidate_names('aaa', 'aaa', 'bbb'):
def mock_get_candidate_names(): dir = tempfile.mkdtemp()
return iter(['aaa', 'aaa', 'bbb'])
with support.swap_attr(tempfile,
'_get_candidate_names',
mock_get_candidate_names):
dir = tempfile.mkdtemp(dir=container_dir)
self.assertTrue(dir.endswith('aaa')) self.assertTrue(dir.endswith('aaa'))
flags = tempfile._bin_openflags (fd, name) = self.default_mkstemp_inner()
(fd, name) = tempfile._mkstemp_inner(container_dir,
tempfile.template,
'',
flags)
try:
self.assertTrue(name.endswith('bbb'))
finally:
os.close(fd) os.close(fd)
os.unlink(name) self.assertTrue(name.endswith('bbb'))
finally:
support.rmtree(container_dir)
test_classes.append(test__mkstemp_inner) test_classes.append(test__mkstemp_inner)
...@@ -587,6 +609,27 @@ class test_mkdtemp(TC): ...@@ -587,6 +609,27 @@ class test_mkdtemp(TC):
finally: finally:
os.rmdir(dir) os.rmdir(dir)
def test_collision_with_existing_file(self):
# mkdtemp tries another name when a file with
# the chosen name already exists
with _inside_empty_temp_dir(), \
_mock_candidate_names('aaa', 'aaa', 'bbb'):
file = tempfile.NamedTemporaryFile(delete=False)
file.close()
self.assertTrue(file.name.endswith('aaa'))
dir = tempfile.mkdtemp()
self.assertTrue(dir.endswith('bbb'))
def test_collision_with_existing_directory(self):
# mkdtemp tries another name when a directory with
# the chosen name already exists
with _inside_empty_temp_dir(), \
_mock_candidate_names('aaa', 'aaa', 'bbb'):
dir1 = tempfile.mkdtemp()
self.assertTrue(dir1.endswith('aaa'))
dir2 = tempfile.mkdtemp()
self.assertTrue(dir2.endswith('bbb'))
test_classes.append(test_mkdtemp) test_classes.append(test_mkdtemp)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment