Commit f0a9a9b5 authored by Gregory P. Smith's avatar Gregory P. Smith

Fixes Issue #14992: os.makedirs(path, exist_ok=True) would raise an OSError

when the path existed and had the S_ISGID mode bit set when it was
not explicitly asked for.  This is no longer an exception as mkdir
cannot control if the OS sets that bit for it or not.
parents 82ffabdf a81c8564
...@@ -160,8 +160,20 @@ def makedirs(name, mode=0o777, exist_ok=False): ...@@ -160,8 +160,20 @@ def makedirs(name, mode=0o777, exist_ok=False):
try: try:
mkdir(name, mode) mkdir(name, mode)
except OSError as e: except OSError as e:
if not (e.errno == errno.EEXIST and exist_ok and path.isdir(name) and dir_exists = path.isdir(name)
st.S_IMODE(lstat(name).st_mode) == _get_masked_mode(mode)): expected_mode = _get_masked_mode(mode)
if dir_exists:
# S_ISGID is automatically copied by the OS from parent to child
# directories on mkdir. Don't consider it being set to be a mode
# mismatch as mkdir does not unset it when not specified in mode.
actual_mode = st.S_IMODE(lstat(name).st_mode) & ~st.S_ISGID
else:
actual_mode = -1
if not (e.errno == errno.EEXIST and exist_ok and dir_exists and
actual_mode == expected_mode):
if dir_exists and actual_mode != expected_mode:
e.strerror += ' (mode %o != expected mode %o)' % (
actual_mode, expected_mode)
raise raise
def removedirs(name): def removedirs(name):
......
...@@ -838,6 +838,31 @@ class MakedirTests(unittest.TestCase): ...@@ -838,6 +838,31 @@ class MakedirTests(unittest.TestCase):
os.makedirs(path, mode=mode, exist_ok=True) os.makedirs(path, mode=mode, exist_ok=True)
os.umask(old_mask) os.umask(old_mask)
def test_exist_ok_s_isgid_directory(self):
path = os.path.join(support.TESTFN, 'dir1')
S_ISGID = stat.S_ISGID
mode = 0o777
old_mask = os.umask(0o022)
try:
existing_testfn_mode = stat.S_IMODE(
os.lstat(support.TESTFN).st_mode)
os.chmod(support.TESTFN, existing_testfn_mode | S_ISGID)
if (os.lstat(support.TESTFN).st_mode & S_ISGID != S_ISGID):
raise unittest.SkipTest('No support for S_ISGID dir mode.')
# The os should apply S_ISGID from the parent dir for us, but
# this test need not depend on that behavior. Be explicit.
os.makedirs(path, mode | S_ISGID)
# http://bugs.python.org/issue14992
# Should not fail when the bit is already set.
os.makedirs(path, mode, exist_ok=True)
# remove the bit.
os.chmod(path, stat.S_IMODE(os.lstat(path).st_mode) & ~S_ISGID)
with self.assertRaises(OSError):
# Should fail when the bit is not already set when demanded.
os.makedirs(path, mode | S_ISGID, exist_ok=True)
finally:
os.umask(old_mask)
def test_exist_ok_existing_regular_file(self): def test_exist_ok_existing_regular_file(self):
base = support.TESTFN base = support.TESTFN
path = os.path.join(support.TESTFN, 'dir1') path = os.path.join(support.TESTFN, 'dir1')
......
...@@ -10,6 +10,11 @@ What's New in Python 3.3.0 Beta 1? ...@@ -10,6 +10,11 @@ What's New in Python 3.3.0 Beta 1?
Library Library
------- -------
- Issue #14992: os.makedirs(path, exist_ok=True) would raise an OSError
when the path existed and had the S_ISGID mode bit set when it was
not explicitly asked for. This is no longer an exception as mkdir
cannot control if the OS sets that bit for it or not.
- Issue #14989: Make the CGI enable option to http.server available via command - Issue #14989: Make the CGI enable option to http.server available via command
line. line.
......
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