diff options
author | Eli Bendersky <eliben@gmail.com> | 2013-09-06 13:11:19 (GMT) |
---|---|---|
committer | Eli Bendersky <eliben@gmail.com> | 2013-09-06 13:11:19 (GMT) |
commit | f315df31bd8a927768bb94c3342d155cdc87d997 (patch) | |
tree | 25a027aa12346e1fff608ea693a61ff3b2cd2aa2 /Lib | |
parent | 43c6ef189955474001aad75d3b47d895180b2d42 (diff) | |
download | cpython-f315df31bd8a927768bb94c3342d155cdc87d997.zip cpython-f315df31bd8a927768bb94c3342d155cdc87d997.tar.gz cpython-f315df31bd8a927768bb94c3342d155cdc87d997.tar.bz2 |
Issue #18849: Fixed a Windows-specific tempfile bug where collision with an
existing directory caused mkstemp and related APIs to fail instead of
retrying. Report and fix by Vlad Shcherbina.
Diffstat (limited to 'Lib')
-rw-r--r-- | Lib/tempfile.py | 7 | ||||
-rw-r--r-- | Lib/test/test_tempfile.py | 26 |
2 files changed, 33 insertions, 0 deletions
diff --git a/Lib/tempfile.py b/Lib/tempfile.py index 8a165ed..91332b6 100644 --- a/Lib/tempfile.py +++ b/Lib/tempfile.py @@ -219,6 +219,13 @@ def _mkstemp_inner(dir, pre, suf, flags): return (fd, _os.path.abspath(file)) except FileExistsError: continue # try again + except PermissionError: + # This exception is thrown when a directory with the chosen name + # already exists on windows. + if _os.name == 'nt': + continue + else: + raise raise FileExistsError(_errno.EEXIST, "No usable temporary file name found") diff --git a/Lib/test/test_tempfile.py b/Lib/test/test_tempfile.py index 2962939..b56d648 100644 --- a/Lib/test/test_tempfile.py +++ b/Lib/test/test_tempfile.py @@ -372,6 +372,32 @@ class TestMkstempInner(BaseTestCase): os.lseek(f.fd, 0, os.SEEK_SET) self.assertEqual(os.read(f.fd, 20), b"blat") + def test_collision_with_existing_directory(self): + # _mkstemp_inner tries another name when a directory with + # the chosen name already exists + container_dir = tempfile.mkdtemp() + try: + def mock_get_candidate_names(): + 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')) + + flags = tempfile._bin_openflags + (fd, name) = tempfile._mkstemp_inner(container_dir, + tempfile.template, + '', + flags) + try: + self.assertTrue(name.endswith('bbb')) + finally: + os.close(fd) + os.unlink(name) + finally: + support.rmtree(container_dir) + class TestGetTempPrefix(BaseTestCase): """Test gettempprefix().""" |