Skip to content

Commit 5751530

Browse files
gh-94315: Check for DAC override capability (GH-94316)
``os.geteuid() == 0`` is not a reliable check whether the current user has the capability to bypass permission checks. Tests now probe for DAC override. (cherry picked from commit 7e0d98e) Co-authored-by: Christian Heimes <christian@python.org>
1 parent 56f5f90 commit 5751530

File tree

6 files changed

+58
-22
lines changed

6 files changed

+58
-22
lines changed

Lib/test/support/os_helper.py

+43-1
Original file line numberDiff line numberDiff line change
@@ -258,7 +258,7 @@ def can_chmod():
258258
else:
259259
can = stat.S_IMODE(mode1) != stat.S_IMODE(mode2)
260260
finally:
261-
os.unlink(TESTFN)
261+
unlink(TESTFN)
262262
_can_chmod = can
263263
return can
264264

@@ -273,6 +273,48 @@ def skip_unless_working_chmod(test):
273273
return test if ok else unittest.skip(msg)(test)
274274

275275

276+
# Check whether the current effective user has the capability to override
277+
# DAC (discretionary access control). Typically user root is able to
278+
# bypass file read, write, and execute permission checks. The capability
279+
# is independent of the effective user. See capabilities(7).
280+
_can_dac_override = None
281+
282+
def can_dac_override():
283+
global _can_dac_override
284+
285+
if not can_chmod():
286+
_can_dac_override = False
287+
if _can_dac_override is not None:
288+
return _can_dac_override
289+
290+
try:
291+
with open(TESTFN, "wb") as f:
292+
os.chmod(TESTFN, 0o400)
293+
try:
294+
with open(TESTFN, "wb"):
295+
pass
296+
except OSError:
297+
_can_dac_override = False
298+
else:
299+
_can_dac_override = True
300+
finally:
301+
unlink(TESTFN)
302+
303+
return _can_dac_override
304+
305+
306+
def skip_if_dac_override(test):
307+
ok = not can_dac_override()
308+
msg = "incompatible with CAP_DAC_OVERRIDE"
309+
return test if ok else unittest.skip(msg)(test)
310+
311+
312+
def skip_unless_dac_override(test):
313+
ok = can_dac_override()
314+
msg = "requires CAP_DAC_OVERRIDE"
315+
return test if ok else unittest.skip(msg)(test)
316+
317+
276318
def unlink(filename):
277319
try:
278320
_unlink(filename)

Lib/test/test_argparse.py

+5-8
Original file line numberDiff line numberDiff line change
@@ -1723,8 +1723,7 @@ def __eq__(self, other):
17231723
return self.name == other.name
17241724

17251725

1726-
@unittest.skipIf(hasattr(os, 'geteuid') and os.geteuid() == 0,
1727-
"non-root user required")
1726+
@os_helper.skip_if_dac_override
17281727
class TestFileTypeW(TempDirMixin, ParserTestCase):
17291728
"""Test the FileType option/argument type for writing files"""
17301729

@@ -1746,8 +1745,8 @@ def setUp(self):
17461745
('-x - -', NS(x=eq_stdout, spam=eq_stdout)),
17471746
]
17481747

1749-
@unittest.skipIf(hasattr(os, 'geteuid') and os.geteuid() == 0,
1750-
"non-root user required")
1748+
1749+
@os_helper.skip_if_dac_override
17511750
class TestFileTypeX(TempDirMixin, ParserTestCase):
17521751
"""Test the FileType option/argument type for writing new files only"""
17531752

@@ -1767,8 +1766,7 @@ def setUp(self):
17671766
]
17681767

17691768

1770-
@unittest.skipIf(hasattr(os, 'geteuid') and os.geteuid() == 0,
1771-
"non-root user required")
1769+
@os_helper.skip_if_dac_override
17721770
class TestFileTypeWB(TempDirMixin, ParserTestCase):
17731771
"""Test the FileType option/argument type for writing binary files"""
17741772

@@ -1785,8 +1783,7 @@ class TestFileTypeWB(TempDirMixin, ParserTestCase):
17851783
]
17861784

17871785

1788-
@unittest.skipIf(hasattr(os, 'geteuid') and os.geteuid() == 0,
1789-
"non-root user required")
1786+
@os_helper.skip_if_dac_override
17901787
class TestFileTypeXB(TestFileTypeX):
17911788
"Test the FileType option/argument type for writing new binary files only"
17921789

Lib/test/test_import/__init__.py

+1-2
Original file line numberDiff line numberDiff line change
@@ -885,10 +885,9 @@ def test_import_pyc_path(self):
885885

886886
@unittest.skipUnless(os.name == 'posix',
887887
"test meaningful only on posix systems")
888-
@unittest.skipIf(hasattr(os, 'geteuid') and os.geteuid() == 0,
889-
"due to varying filesystem permission semantics (issue #11956)")
890888
@skip_if_dont_write_bytecode
891889
@os_helper.skip_unless_working_chmod
890+
@os_helper.skip_if_dac_override
892891
@unittest.skipIf(is_emscripten, "umask is a stub")
893892
def test_unwritable_directory(self):
894893
# When the umask causes the new __pycache__ directory to be

Lib/test/test_py_compile.py

+1-2
Original file line numberDiff line numberDiff line change
@@ -115,8 +115,7 @@ def test_relative_path(self):
115115
self.assertTrue(os.path.exists(self.pyc_path))
116116
self.assertFalse(os.path.exists(self.cache_path))
117117

118-
@unittest.skipIf(hasattr(os, 'geteuid') and os.geteuid() == 0,
119-
'non-root user required')
118+
@os_helper.skip_if_dac_override
120119
@unittest.skipIf(os.name == 'nt',
121120
'cannot control directory permissions on Windows')
122121
@os_helper.skip_unless_working_chmod

Lib/test/test_shutil.py

+6-9
Original file line numberDiff line numberDiff line change
@@ -312,8 +312,7 @@ def onerror(*args):
312312

313313
@unittest.skipIf(sys.platform[:6] == 'cygwin',
314314
"This test can't be run on Cygwin (issue #1071513).")
315-
@unittest.skipIf(hasattr(os, 'geteuid') and os.geteuid() == 0,
316-
"This test can't be run reliably as root (issue #1076467).")
315+
@os_helper.skip_if_dac_override
317316
@os_helper.skip_unless_working_chmod
318317
def test_on_error(self):
319318
self.errorState = 0
@@ -1033,8 +1032,7 @@ def _raise_on_src(fname, *, follow_symlinks=True):
10331032

10341033
@os_helper.skip_unless_symlink
10351034
@os_helper.skip_unless_xattr
1036-
@unittest.skipUnless(hasattr(os, 'geteuid') and os.geteuid() == 0,
1037-
'root privileges required')
1035+
@os_helper.skip_unless_dac_override
10381036
def test_copyxattr_symlinks(self):
10391037
# On Linux, it's only possible to access non-user xattr for symlinks;
10401038
# which in turn require root privileges. This test should be expanded
@@ -1830,8 +1828,7 @@ def test_cwd(self):
18301828
# Other platforms: shouldn't match in the current directory.
18311829
self.assertIsNone(rv)
18321830

1833-
@unittest.skipIf(hasattr(os, 'geteuid') and os.geteuid() == 0,
1834-
'non-root user required')
1831+
@os_helper.skip_if_dac_override
18351832
def test_non_matching_mode(self):
18361833
# Set the file read-only and ask for writeable files.
18371834
os.chmod(self.temp_file.name, stat.S_IREAD)
@@ -2182,11 +2179,11 @@ def test_move_dir_caseinsensitive(self):
21822179
os.rmdir(dst_dir)
21832180

21842181

2185-
@unittest.skipUnless(hasattr(os, 'geteuid') and os.geteuid() == 0
2186-
and hasattr(os, 'lchflags')
2182+
@os_helper.skip_unless_dac_override
2183+
@unittest.skipUnless(hasattr(os, 'lchflags')
21872184
and hasattr(stat, 'SF_IMMUTABLE')
21882185
and hasattr(stat, 'UF_OPAQUE'),
2189-
'root privileges required')
2186+
'requires lchflags')
21902187
def test_move_dir_permission_denied(self):
21912188
# bpo-42782: shutil.move should not create destination directories
21922189
# if the source directory cannot be removed.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
Tests now check for DAC override capability instead of relying on
2+
:func:`os.geteuid`.

0 commit comments

Comments
 (0)