Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 1 | "Test posix functions" |
| 2 | |
Benjamin Peterson | ee8712c | 2008-05-20 21:35:26 +0000 | [diff] [blame] | 3 | from test import support |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 4 | from test.support import import_helper |
| 5 | from test.support import os_helper |
| 6 | from test.support import warnings_helper |
Antoine Pitrou | 346cbd3 | 2017-05-27 17:50:54 +0200 | [diff] [blame] | 7 | from test.support.script_helper import assert_python_ok |
R. David Murray | eb3615d | 2009-04-22 02:24:39 +0000 | [diff] [blame] | 8 | |
| 9 | # Skip these tests if there is no posix module. |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 10 | posix = import_helper.import_module('posix') |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 11 | |
Antoine Pitrou | b7572f0 | 2009-12-02 20:46:48 +0000 | [diff] [blame] | 12 | import errno |
Ronald Oussoren | b6ee4f5 | 2010-07-23 13:53:51 +0000 | [diff] [blame] | 13 | import sys |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 14 | import signal |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 15 | import time |
| 16 | import os |
Charles-François Natali | ab2d58e | 2012-04-17 19:48:35 +0200 | [diff] [blame] | 17 | import platform |
Christian Heimes | d5e2b6f | 2008-03-19 21:50:51 +0000 | [diff] [blame] | 18 | import pwd |
Benjamin Peterson | 052a02b | 2010-08-17 01:27:09 +0000 | [diff] [blame] | 19 | import stat |
Ned Deily | ba2eab2 | 2011-07-26 13:53:55 -0700 | [diff] [blame] | 20 | import tempfile |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 21 | import unittest |
| 22 | import warnings |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 23 | import textwrap |
R. David Murray | a21e4ca | 2009-03-31 23:16:50 +0000 | [diff] [blame] | 24 | |
Ned Deily | ba2eab2 | 2011-07-26 13:53:55 -0700 | [diff] [blame] | 25 | _DUMMY_SYMLINK = os.path.join(tempfile.gettempdir(), |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 26 | os_helper.TESTFN + '-dummy-symlink') |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 27 | |
Serhiy Storchaka | 9d57273 | 2018-07-31 10:24:54 +0300 | [diff] [blame] | 28 | requires_32b = unittest.skipUnless(sys.maxsize < 2**32, |
| 29 | 'test is only meaningful on 32-bit builds') |
| 30 | |
Benjamin Peterson | c704222 | 2018-09-12 15:12:24 -0700 | [diff] [blame] | 31 | def _supports_sched(): |
| 32 | if not hasattr(posix, 'sched_getscheduler'): |
| 33 | return False |
| 34 | try: |
| 35 | posix.sched_getscheduler(0) |
| 36 | except OSError as e: |
| 37 | if e.errno == errno.ENOSYS: |
| 38 | return False |
| 39 | return True |
| 40 | |
| 41 | requires_sched = unittest.skipUnless(_supports_sched(), 'requires POSIX scheduler API') |
| 42 | |
Victor Stinner | 278c1e1 | 2020-03-31 20:08:12 +0200 | [diff] [blame] | 43 | |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 44 | class PosixTester(unittest.TestCase): |
| 45 | |
| 46 | def setUp(self): |
| 47 | # create empty file |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 48 | fp = open(os_helper.TESTFN, 'w+') |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 49 | fp.close() |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 50 | self.teardown_files = [ os_helper.TESTFN ] |
| 51 | self._warnings_manager = warnings_helper.check_warnings() |
Brett Cannon | c8d502e | 2010-03-20 21:53:28 +0000 | [diff] [blame] | 52 | self._warnings_manager.__enter__() |
| 53 | warnings.filterwarnings('ignore', '.* potential security risk .*', |
| 54 | RuntimeWarning) |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 55 | |
| 56 | def tearDown(self): |
Ned Deily | 3eb67d5 | 2011-06-28 00:00:28 -0700 | [diff] [blame] | 57 | for teardown_file in self.teardown_files: |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 58 | os_helper.unlink(teardown_file) |
Brett Cannon | c8d502e | 2010-03-20 21:53:28 +0000 | [diff] [blame] | 59 | self._warnings_manager.__exit__(None, None, None) |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 60 | |
| 61 | def testNoArgFunctions(self): |
| 62 | # test posix functions which take no arguments and have |
| 63 | # no side-effects which we need to cleanup (e.g., fork, wait, abort) |
Guido van Rossum | f0af3e3 | 2008-10-02 18:55:37 +0000 | [diff] [blame] | 64 | NO_ARG_FUNCTIONS = [ "ctermid", "getcwd", "getcwdb", "uname", |
Guido van Rossum | 687b9c0 | 2007-10-25 23:18:51 +0000 | [diff] [blame] | 65 | "times", "getloadavg", |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 66 | "getegid", "geteuid", "getgid", "getgroups", |
Ross Lagerwall | 7807c35 | 2011-03-17 20:20:30 +0200 | [diff] [blame] | 67 | "getpid", "getpgrp", "getppid", "getuid", "sync", |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 68 | ] |
Neal Norwitz | 71b13e8 | 2003-02-23 22:12:24 +0000 | [diff] [blame] | 69 | |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 70 | for name in NO_ARG_FUNCTIONS: |
| 71 | posix_func = getattr(posix, name, None) |
| 72 | if posix_func is not None: |
| 73 | posix_func() |
Neal Norwitz | 2ff51a8 | 2003-02-17 22:40:31 +0000 | [diff] [blame] | 74 | self.assertRaises(TypeError, posix_func, 1) |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 75 | |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 76 | @unittest.skipUnless(hasattr(posix, 'getresuid'), |
| 77 | 'test needs posix.getresuid()') |
| 78 | def test_getresuid(self): |
| 79 | user_ids = posix.getresuid() |
| 80 | self.assertEqual(len(user_ids), 3) |
| 81 | for val in user_ids: |
| 82 | self.assertGreaterEqual(val, 0) |
Martin v. Löwis | 7aed61a | 2009-11-27 14:09:49 +0000 | [diff] [blame] | 83 | |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 84 | @unittest.skipUnless(hasattr(posix, 'getresgid'), |
| 85 | 'test needs posix.getresgid()') |
| 86 | def test_getresgid(self): |
| 87 | group_ids = posix.getresgid() |
| 88 | self.assertEqual(len(group_ids), 3) |
| 89 | for val in group_ids: |
| 90 | self.assertGreaterEqual(val, 0) |
Martin v. Löwis | 7aed61a | 2009-11-27 14:09:49 +0000 | [diff] [blame] | 91 | |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 92 | @unittest.skipUnless(hasattr(posix, 'setresuid'), |
| 93 | 'test needs posix.setresuid()') |
| 94 | def test_setresuid(self): |
| 95 | current_user_ids = posix.getresuid() |
| 96 | self.assertIsNone(posix.setresuid(*current_user_ids)) |
| 97 | # -1 means don't change that value. |
| 98 | self.assertIsNone(posix.setresuid(-1, -1, -1)) |
Martin v. Löwis | 7aed61a | 2009-11-27 14:09:49 +0000 | [diff] [blame] | 99 | |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 100 | @unittest.skipUnless(hasattr(posix, 'setresuid'), |
| 101 | 'test needs posix.setresuid()') |
| 102 | def test_setresuid_exception(self): |
| 103 | # Don't do this test if someone is silly enough to run us as root. |
| 104 | current_user_ids = posix.getresuid() |
| 105 | if 0 not in current_user_ids: |
| 106 | new_user_ids = (current_user_ids[0]+1, -1, -1) |
| 107 | self.assertRaises(OSError, posix.setresuid, *new_user_ids) |
Martin v. Löwis | 7aed61a | 2009-11-27 14:09:49 +0000 | [diff] [blame] | 108 | |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 109 | @unittest.skipUnless(hasattr(posix, 'setresgid'), |
| 110 | 'test needs posix.setresgid()') |
| 111 | def test_setresgid(self): |
| 112 | current_group_ids = posix.getresgid() |
| 113 | self.assertIsNone(posix.setresgid(*current_group_ids)) |
| 114 | # -1 means don't change that value. |
| 115 | self.assertIsNone(posix.setresgid(-1, -1, -1)) |
Martin v. Löwis | 7aed61a | 2009-11-27 14:09:49 +0000 | [diff] [blame] | 116 | |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 117 | @unittest.skipUnless(hasattr(posix, 'setresgid'), |
| 118 | 'test needs posix.setresgid()') |
| 119 | def test_setresgid_exception(self): |
| 120 | # Don't do this test if someone is silly enough to run us as root. |
| 121 | current_group_ids = posix.getresgid() |
| 122 | if 0 not in current_group_ids: |
| 123 | new_group_ids = (current_group_ids[0]+1, -1, -1) |
| 124 | self.assertRaises(OSError, posix.setresgid, *new_group_ids) |
Martin v. Löwis | 7aed61a | 2009-11-27 14:09:49 +0000 | [diff] [blame] | 125 | |
Antoine Pitrou | b7572f0 | 2009-12-02 20:46:48 +0000 | [diff] [blame] | 126 | @unittest.skipUnless(hasattr(posix, 'initgroups'), |
| 127 | "test needs os.initgroups()") |
| 128 | def test_initgroups(self): |
| 129 | # It takes a string and an integer; check that it raises a TypeError |
| 130 | # for other argument lists. |
| 131 | self.assertRaises(TypeError, posix.initgroups) |
| 132 | self.assertRaises(TypeError, posix.initgroups, None) |
| 133 | self.assertRaises(TypeError, posix.initgroups, 3, "foo") |
| 134 | self.assertRaises(TypeError, posix.initgroups, "foo", 3, object()) |
| 135 | |
| 136 | # If a non-privileged user invokes it, it should fail with OSError |
| 137 | # EPERM. |
| 138 | if os.getuid() != 0: |
Charles-François Natali | e8a255a | 2012-05-02 20:01:38 +0200 | [diff] [blame] | 139 | try: |
| 140 | name = pwd.getpwuid(posix.getuid()).pw_name |
| 141 | except KeyError: |
| 142 | # the current UID may not have a pwd entry |
| 143 | raise unittest.SkipTest("need a pwd entry") |
Antoine Pitrou | b7572f0 | 2009-12-02 20:46:48 +0000 | [diff] [blame] | 144 | try: |
| 145 | posix.initgroups(name, 13) |
| 146 | except OSError as e: |
Ezio Melotti | b3aedd4 | 2010-11-20 19:04:17 +0000 | [diff] [blame] | 147 | self.assertEqual(e.errno, errno.EPERM) |
Antoine Pitrou | b7572f0 | 2009-12-02 20:46:48 +0000 | [diff] [blame] | 148 | else: |
| 149 | self.fail("Expected OSError to be raised by initgroups") |
| 150 | |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 151 | @unittest.skipUnless(hasattr(posix, 'statvfs'), |
| 152 | 'test needs posix.statvfs()') |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 153 | def test_statvfs(self): |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 154 | self.assertTrue(posix.statvfs(os.curdir)) |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 155 | |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 156 | @unittest.skipUnless(hasattr(posix, 'fstatvfs'), |
| 157 | 'test needs posix.fstatvfs()') |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 158 | def test_fstatvfs(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 159 | fp = open(os_helper.TESTFN) |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 160 | try: |
| 161 | self.assertTrue(posix.fstatvfs(fp.fileno())) |
| 162 | self.assertTrue(posix.statvfs(fp.fileno())) |
| 163 | finally: |
| 164 | fp.close() |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 165 | |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 166 | @unittest.skipUnless(hasattr(posix, 'ftruncate'), |
| 167 | 'test needs posix.ftruncate()') |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 168 | def test_ftruncate(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 169 | fp = open(os_helper.TESTFN, 'w+') |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 170 | try: |
| 171 | # we need to have some data to truncate |
| 172 | fp.write('test') |
| 173 | fp.flush() |
| 174 | posix.ftruncate(fp.fileno(), 0) |
| 175 | finally: |
| 176 | fp.close() |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 177 | |
Ross Lagerwall | 7807c35 | 2011-03-17 20:20:30 +0200 | [diff] [blame] | 178 | @unittest.skipUnless(hasattr(posix, 'truncate'), "test needs posix.truncate()") |
| 179 | def test_truncate(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 180 | with open(os_helper.TESTFN, 'w') as fp: |
Ross Lagerwall | 7807c35 | 2011-03-17 20:20:30 +0200 | [diff] [blame] | 181 | fp.write('test') |
| 182 | fp.flush() |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 183 | posix.truncate(os_helper.TESTFN, 0) |
Ross Lagerwall | 7807c35 | 2011-03-17 20:20:30 +0200 | [diff] [blame] | 184 | |
Larry Hastings | 9cf065c | 2012-06-22 16:30:09 -0700 | [diff] [blame] | 185 | @unittest.skipUnless(getattr(os, 'execve', None) in os.supports_fd, "test needs execve() to support the fd parameter") |
Ross Lagerwall | 7807c35 | 2011-03-17 20:20:30 +0200 | [diff] [blame] | 186 | @unittest.skipUnless(hasattr(os, 'fork'), "test needs os.fork()") |
Ross Lagerwall | 7807c35 | 2011-03-17 20:20:30 +0200 | [diff] [blame] | 187 | def test_fexecve(self): |
| 188 | fp = os.open(sys.executable, os.O_RDONLY) |
| 189 | try: |
| 190 | pid = os.fork() |
| 191 | if pid == 0: |
| 192 | os.chdir(os.path.split(sys.executable)[0]) |
Larry Hastings | 9cf065c | 2012-06-22 16:30:09 -0700 | [diff] [blame] | 193 | posix.execve(fp, [sys.executable, '-c', 'pass'], os.environ) |
Ross Lagerwall | 7807c35 | 2011-03-17 20:20:30 +0200 | [diff] [blame] | 194 | else: |
Victor Stinner | 278c1e1 | 2020-03-31 20:08:12 +0200 | [diff] [blame] | 195 | support.wait_process(pid, exitcode=0) |
Ross Lagerwall | 7807c35 | 2011-03-17 20:20:30 +0200 | [diff] [blame] | 196 | finally: |
| 197 | os.close(fp) |
| 198 | |
Pablo Galindo | 6c6ddf9 | 2018-01-29 01:56:10 +0000 | [diff] [blame] | 199 | |
Ross Lagerwall | 7807c35 | 2011-03-17 20:20:30 +0200 | [diff] [blame] | 200 | @unittest.skipUnless(hasattr(posix, 'waitid'), "test needs posix.waitid()") |
| 201 | @unittest.skipUnless(hasattr(os, 'fork'), "test needs os.fork()") |
| 202 | def test_waitid(self): |
| 203 | pid = os.fork() |
| 204 | if pid == 0: |
| 205 | os.chdir(os.path.split(sys.executable)[0]) |
| 206 | posix.execve(sys.executable, [sys.executable, '-c', 'pass'], os.environ) |
| 207 | else: |
| 208 | res = posix.waitid(posix.P_PID, pid, posix.WEXITED) |
| 209 | self.assertEqual(pid, res.si_pid) |
| 210 | |
Antoine Pitrou | 346cbd3 | 2017-05-27 17:50:54 +0200 | [diff] [blame] | 211 | @unittest.skipUnless(hasattr(os, 'fork'), "test needs os.fork()") |
Gregory P. Smith | 163468a | 2017-05-29 10:03:41 -0700 | [diff] [blame] | 212 | def test_register_at_fork(self): |
| 213 | with self.assertRaises(TypeError, msg="Positional args not allowed"): |
| 214 | os.register_at_fork(lambda: None) |
| 215 | with self.assertRaises(TypeError, msg="Args must be callable"): |
| 216 | os.register_at_fork(before=2) |
| 217 | with self.assertRaises(TypeError, msg="Args must be callable"): |
| 218 | os.register_at_fork(after_in_child="three") |
| 219 | with self.assertRaises(TypeError, msg="Args must be callable"): |
| 220 | os.register_at_fork(after_in_parent=b"Five") |
| 221 | with self.assertRaises(TypeError, msg="Args must not be None"): |
| 222 | os.register_at_fork(before=None) |
| 223 | with self.assertRaises(TypeError, msg="Args must not be None"): |
| 224 | os.register_at_fork(after_in_child=None) |
| 225 | with self.assertRaises(TypeError, msg="Args must not be None"): |
| 226 | os.register_at_fork(after_in_parent=None) |
| 227 | with self.assertRaises(TypeError, msg="Invalid arg was allowed"): |
| 228 | # Ensure a combination of valid and invalid is an error. |
| 229 | os.register_at_fork(before=None, after_in_parent=lambda: 3) |
| 230 | with self.assertRaises(TypeError, msg="Invalid arg was allowed"): |
| 231 | # Ensure a combination of valid and invalid is an error. |
| 232 | os.register_at_fork(before=lambda: None, after_in_child='') |
| 233 | # We test actual registrations in their own process so as not to |
| 234 | # pollute this one. There is no way to unregister for cleanup. |
Antoine Pitrou | 346cbd3 | 2017-05-27 17:50:54 +0200 | [diff] [blame] | 235 | code = """if 1: |
| 236 | import os |
| 237 | |
| 238 | r, w = os.pipe() |
| 239 | fin_r, fin_w = os.pipe() |
| 240 | |
Gregory P. Smith | 163468a | 2017-05-29 10:03:41 -0700 | [diff] [blame] | 241 | os.register_at_fork(before=lambda: os.write(w, b'A')) |
| 242 | os.register_at_fork(after_in_parent=lambda: os.write(w, b'C')) |
| 243 | os.register_at_fork(after_in_child=lambda: os.write(w, b'E')) |
| 244 | os.register_at_fork(before=lambda: os.write(w, b'B'), |
| 245 | after_in_parent=lambda: os.write(w, b'D'), |
| 246 | after_in_child=lambda: os.write(w, b'F')) |
Antoine Pitrou | 346cbd3 | 2017-05-27 17:50:54 +0200 | [diff] [blame] | 247 | |
| 248 | pid = os.fork() |
| 249 | if pid == 0: |
| 250 | # At this point, after-forkers have already been executed |
| 251 | os.close(w) |
| 252 | # Wait for parent to tell us to exit |
| 253 | os.read(fin_r, 1) |
| 254 | os._exit(0) |
| 255 | else: |
| 256 | try: |
| 257 | os.close(w) |
| 258 | with open(r, "rb") as f: |
| 259 | data = f.read() |
| 260 | assert len(data) == 6, data |
| 261 | # Check before-fork callbacks |
| 262 | assert data[:2] == b'BA', data |
| 263 | # Check after-fork callbacks |
| 264 | assert sorted(data[2:]) == list(b'CDEF'), data |
| 265 | assert data.index(b'C') < data.index(b'D'), data |
| 266 | assert data.index(b'E') < data.index(b'F'), data |
| 267 | finally: |
| 268 | os.write(fin_w, b'!') |
| 269 | """ |
| 270 | assert_python_ok('-c', code) |
| 271 | |
Ross Lagerwall | 7807c35 | 2011-03-17 20:20:30 +0200 | [diff] [blame] | 272 | @unittest.skipUnless(hasattr(posix, 'lockf'), "test needs posix.lockf()") |
| 273 | def test_lockf(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 274 | fd = os.open(os_helper.TESTFN, os.O_WRONLY | os.O_CREAT) |
Ross Lagerwall | 7807c35 | 2011-03-17 20:20:30 +0200 | [diff] [blame] | 275 | try: |
| 276 | os.write(fd, b'test') |
| 277 | os.lseek(fd, 0, os.SEEK_SET) |
| 278 | posix.lockf(fd, posix.F_LOCK, 4) |
| 279 | # section is locked |
| 280 | posix.lockf(fd, posix.F_ULOCK, 4) |
| 281 | finally: |
| 282 | os.close(fd) |
| 283 | |
| 284 | @unittest.skipUnless(hasattr(posix, 'pread'), "test needs posix.pread()") |
| 285 | def test_pread(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 286 | fd = os.open(os_helper.TESTFN, os.O_RDWR | os.O_CREAT) |
Ross Lagerwall | 7807c35 | 2011-03-17 20:20:30 +0200 | [diff] [blame] | 287 | try: |
| 288 | os.write(fd, b'test') |
| 289 | os.lseek(fd, 0, os.SEEK_SET) |
| 290 | self.assertEqual(b'es', posix.pread(fd, 2, 1)) |
Florent Xicluna | e41f0de | 2011-11-11 19:39:25 +0100 | [diff] [blame] | 291 | # the first pread() shouldn't disturb the file offset |
Ross Lagerwall | 7807c35 | 2011-03-17 20:20:30 +0200 | [diff] [blame] | 292 | self.assertEqual(b'te', posix.read(fd, 2)) |
| 293 | finally: |
| 294 | os.close(fd) |
| 295 | |
Pablo Galindo | 4defba3 | 2018-01-27 16:16:37 +0000 | [diff] [blame] | 296 | @unittest.skipUnless(hasattr(posix, 'preadv'), "test needs posix.preadv()") |
| 297 | def test_preadv(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 298 | fd = os.open(os_helper.TESTFN, os.O_RDWR | os.O_CREAT) |
Pablo Galindo | 4defba3 | 2018-01-27 16:16:37 +0000 | [diff] [blame] | 299 | try: |
| 300 | os.write(fd, b'test1tt2t3t5t6t6t8') |
| 301 | buf = [bytearray(i) for i in [5, 3, 2]] |
| 302 | self.assertEqual(posix.preadv(fd, buf, 3), 10) |
| 303 | self.assertEqual([b't1tt2', b't3t', b'5t'], list(buf)) |
| 304 | finally: |
| 305 | os.close(fd) |
| 306 | |
Serhiy Storchaka | 9d57273 | 2018-07-31 10:24:54 +0300 | [diff] [blame] | 307 | @unittest.skipUnless(hasattr(posix, 'preadv'), "test needs posix.preadv()") |
Pablo Galindo | 4defba3 | 2018-01-27 16:16:37 +0000 | [diff] [blame] | 308 | @unittest.skipUnless(hasattr(posix, 'RWF_HIPRI'), "test needs posix.RWF_HIPRI") |
| 309 | def test_preadv_flags(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 310 | fd = os.open(os_helper.TESTFN, os.O_RDWR | os.O_CREAT) |
Pablo Galindo | 4defba3 | 2018-01-27 16:16:37 +0000 | [diff] [blame] | 311 | try: |
| 312 | os.write(fd, b'test1tt2t3t5t6t6t8') |
| 313 | buf = [bytearray(i) for i in [5, 3, 2]] |
| 314 | self.assertEqual(posix.preadv(fd, buf, 3, os.RWF_HIPRI), 10) |
| 315 | self.assertEqual([b't1tt2', b't3t', b'5t'], list(buf)) |
Benjamin Peterson | 44867bb | 2019-06-11 10:15:31 -0700 | [diff] [blame] | 316 | except NotImplementedError: |
| 317 | self.skipTest("preadv2 not available") |
Pablo Galindo | 46544f6 | 2019-04-13 17:06:03 +0100 | [diff] [blame] | 318 | except OSError as inst: |
| 319 | # Is possible that the macro RWF_HIPRI was defined at compilation time |
| 320 | # but the option is not supported by the kernel or the runtime libc shared |
| 321 | # library. |
| 322 | if inst.errno in {errno.EINVAL, errno.ENOTSUP}: |
| 323 | raise unittest.SkipTest("RWF_HIPRI is not supported by the current system") |
| 324 | else: |
| 325 | raise |
Pablo Galindo | 4defba3 | 2018-01-27 16:16:37 +0000 | [diff] [blame] | 326 | finally: |
| 327 | os.close(fd) |
| 328 | |
Serhiy Storchaka | 9d57273 | 2018-07-31 10:24:54 +0300 | [diff] [blame] | 329 | @unittest.skipUnless(hasattr(posix, 'preadv'), "test needs posix.preadv()") |
| 330 | @requires_32b |
| 331 | def test_preadv_overflow_32bits(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 332 | fd = os.open(os_helper.TESTFN, os.O_RDWR | os.O_CREAT) |
Serhiy Storchaka | 9d57273 | 2018-07-31 10:24:54 +0300 | [diff] [blame] | 333 | try: |
| 334 | buf = [bytearray(2**16)] * 2**15 |
| 335 | with self.assertRaises(OSError) as cm: |
| 336 | os.preadv(fd, buf, 0) |
| 337 | self.assertEqual(cm.exception.errno, errno.EINVAL) |
| 338 | self.assertEqual(bytes(buf[0]), b'\0'* 2**16) |
| 339 | finally: |
| 340 | os.close(fd) |
| 341 | |
Ross Lagerwall | 7807c35 | 2011-03-17 20:20:30 +0200 | [diff] [blame] | 342 | @unittest.skipUnless(hasattr(posix, 'pwrite'), "test needs posix.pwrite()") |
| 343 | def test_pwrite(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 344 | fd = os.open(os_helper.TESTFN, os.O_RDWR | os.O_CREAT) |
Ross Lagerwall | 7807c35 | 2011-03-17 20:20:30 +0200 | [diff] [blame] | 345 | try: |
| 346 | os.write(fd, b'test') |
| 347 | os.lseek(fd, 0, os.SEEK_SET) |
| 348 | posix.pwrite(fd, b'xx', 1) |
| 349 | self.assertEqual(b'txxt', posix.read(fd, 4)) |
| 350 | finally: |
| 351 | os.close(fd) |
| 352 | |
Pablo Galindo | 4defba3 | 2018-01-27 16:16:37 +0000 | [diff] [blame] | 353 | @unittest.skipUnless(hasattr(posix, 'pwritev'), "test needs posix.pwritev()") |
| 354 | def test_pwritev(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 355 | fd = os.open(os_helper.TESTFN, os.O_RDWR | os.O_CREAT) |
Pablo Galindo | 4defba3 | 2018-01-27 16:16:37 +0000 | [diff] [blame] | 356 | try: |
| 357 | os.write(fd, b"xx") |
| 358 | os.lseek(fd, 0, os.SEEK_SET) |
| 359 | n = os.pwritev(fd, [b'test1', b'tt2', b't3'], 2) |
| 360 | self.assertEqual(n, 10) |
| 361 | |
| 362 | os.lseek(fd, 0, os.SEEK_SET) |
| 363 | self.assertEqual(b'xxtest1tt2t3', posix.read(fd, 100)) |
| 364 | finally: |
| 365 | os.close(fd) |
| 366 | |
Serhiy Storchaka | 9d57273 | 2018-07-31 10:24:54 +0300 | [diff] [blame] | 367 | @unittest.skipUnless(hasattr(posix, 'pwritev'), "test needs posix.pwritev()") |
Pablo Galindo | 4defba3 | 2018-01-27 16:16:37 +0000 | [diff] [blame] | 368 | @unittest.skipUnless(hasattr(posix, 'os.RWF_SYNC'), "test needs os.RWF_SYNC") |
| 369 | def test_pwritev_flags(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 370 | fd = os.open(os_helper.TESTFN, os.O_RDWR | os.O_CREAT) |
Pablo Galindo | 4defba3 | 2018-01-27 16:16:37 +0000 | [diff] [blame] | 371 | try: |
| 372 | os.write(fd,b"xx") |
| 373 | os.lseek(fd, 0, os.SEEK_SET) |
| 374 | n = os.pwritev(fd, [b'test1', b'tt2', b't3'], 2, os.RWF_SYNC) |
| 375 | self.assertEqual(n, 10) |
| 376 | |
| 377 | os.lseek(fd, 0, os.SEEK_SET) |
| 378 | self.assertEqual(b'xxtest1tt2', posix.read(fd, 100)) |
| 379 | finally: |
| 380 | os.close(fd) |
| 381 | |
Serhiy Storchaka | 9d57273 | 2018-07-31 10:24:54 +0300 | [diff] [blame] | 382 | @unittest.skipUnless(hasattr(posix, 'pwritev'), "test needs posix.pwritev()") |
| 383 | @requires_32b |
| 384 | def test_pwritev_overflow_32bits(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 385 | fd = os.open(os_helper.TESTFN, os.O_RDWR | os.O_CREAT) |
Serhiy Storchaka | 9d57273 | 2018-07-31 10:24:54 +0300 | [diff] [blame] | 386 | try: |
| 387 | with self.assertRaises(OSError) as cm: |
| 388 | os.pwritev(fd, [b"x" * 2**16] * 2**15, 0) |
| 389 | self.assertEqual(cm.exception.errno, errno.EINVAL) |
| 390 | finally: |
| 391 | os.close(fd) |
| 392 | |
Ross Lagerwall | 7807c35 | 2011-03-17 20:20:30 +0200 | [diff] [blame] | 393 | @unittest.skipUnless(hasattr(posix, 'posix_fallocate'), |
| 394 | "test needs posix.posix_fallocate()") |
| 395 | def test_posix_fallocate(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 396 | fd = os.open(os_helper.TESTFN, os.O_WRONLY | os.O_CREAT) |
Ross Lagerwall | 7807c35 | 2011-03-17 20:20:30 +0200 | [diff] [blame] | 397 | try: |
| 398 | posix.posix_fallocate(fd, 0, 10) |
| 399 | except OSError as inst: |
| 400 | # issue10812, ZFS doesn't appear to support posix_fallocate, |
| 401 | # so skip Solaris-based since they are likely to have ZFS. |
Ned Deily | 09c4a7d | 2018-05-26 16:30:46 -0400 | [diff] [blame] | 402 | # issue33655: Also ignore EINVAL on *BSD since ZFS is also |
| 403 | # often used there. |
| 404 | if inst.errno == errno.EINVAL and sys.platform.startswith( |
| 405 | ('sunos', 'freebsd', 'netbsd', 'openbsd', 'gnukfreebsd')): |
| 406 | raise unittest.SkipTest("test may fail on ZFS filesystems") |
| 407 | else: |
Ross Lagerwall | 7807c35 | 2011-03-17 20:20:30 +0200 | [diff] [blame] | 408 | raise |
| 409 | finally: |
| 410 | os.close(fd) |
| 411 | |
Коренберг Марк | d4b93e2 | 2017-08-14 18:55:16 +0500 | [diff] [blame] | 412 | # issue31106 - posix_fallocate() does not set error in errno. |
| 413 | @unittest.skipUnless(hasattr(posix, 'posix_fallocate'), |
| 414 | "test needs posix.posix_fallocate()") |
| 415 | def test_posix_fallocate_errno(self): |
| 416 | try: |
| 417 | posix.posix_fallocate(-42, 0, 10) |
| 418 | except OSError as inst: |
| 419 | if inst.errno != errno.EBADF: |
| 420 | raise |
| 421 | |
Ross Lagerwall | 7807c35 | 2011-03-17 20:20:30 +0200 | [diff] [blame] | 422 | @unittest.skipUnless(hasattr(posix, 'posix_fadvise'), |
| 423 | "test needs posix.posix_fadvise()") |
| 424 | def test_posix_fadvise(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 425 | fd = os.open(os_helper.TESTFN, os.O_RDONLY) |
Ross Lagerwall | 7807c35 | 2011-03-17 20:20:30 +0200 | [diff] [blame] | 426 | try: |
| 427 | posix.posix_fadvise(fd, 0, 0, posix.POSIX_FADV_WILLNEED) |
| 428 | finally: |
| 429 | os.close(fd) |
| 430 | |
Коренберг Марк | d4b93e2 | 2017-08-14 18:55:16 +0500 | [diff] [blame] | 431 | @unittest.skipUnless(hasattr(posix, 'posix_fadvise'), |
| 432 | "test needs posix.posix_fadvise()") |
| 433 | def test_posix_fadvise_errno(self): |
| 434 | try: |
| 435 | posix.posix_fadvise(-42, 0, 0, posix.POSIX_FADV_WILLNEED) |
| 436 | except OSError as inst: |
| 437 | if inst.errno != errno.EBADF: |
| 438 | raise |
| 439 | |
Larry Hastings | 9cf065c | 2012-06-22 16:30:09 -0700 | [diff] [blame] | 440 | @unittest.skipUnless(os.utime in os.supports_fd, "test needs fd support in os.utime") |
| 441 | def test_utime_with_fd(self): |
Ross Lagerwall | 7807c35 | 2011-03-17 20:20:30 +0200 | [diff] [blame] | 442 | now = time.time() |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 443 | fd = os.open(os_helper.TESTFN, os.O_RDONLY) |
Ross Lagerwall | 7807c35 | 2011-03-17 20:20:30 +0200 | [diff] [blame] | 444 | try: |
Larry Hastings | 9cf065c | 2012-06-22 16:30:09 -0700 | [diff] [blame] | 445 | posix.utime(fd) |
| 446 | posix.utime(fd, None) |
| 447 | self.assertRaises(TypeError, posix.utime, fd, (None, None)) |
| 448 | self.assertRaises(TypeError, posix.utime, fd, (now, None)) |
| 449 | self.assertRaises(TypeError, posix.utime, fd, (None, now)) |
| 450 | posix.utime(fd, (int(now), int(now))) |
| 451 | posix.utime(fd, (now, now)) |
| 452 | self.assertRaises(ValueError, posix.utime, fd, (now, now), ns=(now, now)) |
| 453 | self.assertRaises(ValueError, posix.utime, fd, (now, 0), ns=(None, None)) |
| 454 | self.assertRaises(ValueError, posix.utime, fd, (None, None), ns=(now, 0)) |
| 455 | posix.utime(fd, (int(now), int((now - int(now)) * 1e9))) |
| 456 | posix.utime(fd, ns=(int(now), int((now - int(now)) * 1e9))) |
| 457 | |
Ross Lagerwall | 7807c35 | 2011-03-17 20:20:30 +0200 | [diff] [blame] | 458 | finally: |
| 459 | os.close(fd) |
| 460 | |
Larry Hastings | 9cf065c | 2012-06-22 16:30:09 -0700 | [diff] [blame] | 461 | @unittest.skipUnless(os.utime in os.supports_follow_symlinks, "test needs follow_symlinks support in os.utime") |
| 462 | def test_utime_nofollow_symlinks(self): |
Ross Lagerwall | 7807c35 | 2011-03-17 20:20:30 +0200 | [diff] [blame] | 463 | now = time.time() |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 464 | posix.utime(os_helper.TESTFN, None, follow_symlinks=False) |
| 465 | self.assertRaises(TypeError, posix.utime, os_helper.TESTFN, |
| 466 | (None, None), follow_symlinks=False) |
| 467 | self.assertRaises(TypeError, posix.utime, os_helper.TESTFN, |
| 468 | (now, None), follow_symlinks=False) |
| 469 | self.assertRaises(TypeError, posix.utime, os_helper.TESTFN, |
| 470 | (None, now), follow_symlinks=False) |
| 471 | posix.utime(os_helper.TESTFN, (int(now), int(now)), |
| 472 | follow_symlinks=False) |
| 473 | posix.utime(os_helper.TESTFN, (now, now), follow_symlinks=False) |
| 474 | posix.utime(os_helper.TESTFN, follow_symlinks=False) |
Ross Lagerwall | 7807c35 | 2011-03-17 20:20:30 +0200 | [diff] [blame] | 475 | |
| 476 | @unittest.skipUnless(hasattr(posix, 'writev'), "test needs posix.writev()") |
| 477 | def test_writev(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 478 | fd = os.open(os_helper.TESTFN, os.O_RDWR | os.O_CREAT) |
Ross Lagerwall | 7807c35 | 2011-03-17 20:20:30 +0200 | [diff] [blame] | 479 | try: |
Victor Stinner | 57ddf78 | 2014-01-08 15:21:28 +0100 | [diff] [blame] | 480 | n = os.writev(fd, (b'test1', b'tt2', b't3')) |
| 481 | self.assertEqual(n, 10) |
| 482 | |
Ross Lagerwall | 7807c35 | 2011-03-17 20:20:30 +0200 | [diff] [blame] | 483 | os.lseek(fd, 0, os.SEEK_SET) |
| 484 | self.assertEqual(b'test1tt2t3', posix.read(fd, 10)) |
Victor Stinner | 57ddf78 | 2014-01-08 15:21:28 +0100 | [diff] [blame] | 485 | |
| 486 | # Issue #20113: empty list of buffers should not crash |
Victor Stinner | cd5ca6a | 2014-01-08 16:01:31 +0100 | [diff] [blame] | 487 | try: |
| 488 | size = posix.writev(fd, []) |
| 489 | except OSError: |
| 490 | # writev(fd, []) raises OSError(22, "Invalid argument") |
| 491 | # on OpenIndiana |
| 492 | pass |
| 493 | else: |
| 494 | self.assertEqual(size, 0) |
Ross Lagerwall | 7807c35 | 2011-03-17 20:20:30 +0200 | [diff] [blame] | 495 | finally: |
| 496 | os.close(fd) |
| 497 | |
Serhiy Storchaka | 9d57273 | 2018-07-31 10:24:54 +0300 | [diff] [blame] | 498 | @unittest.skipUnless(hasattr(posix, 'writev'), "test needs posix.writev()") |
| 499 | @requires_32b |
| 500 | def test_writev_overflow_32bits(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 501 | fd = os.open(os_helper.TESTFN, os.O_RDWR | os.O_CREAT) |
Serhiy Storchaka | 9d57273 | 2018-07-31 10:24:54 +0300 | [diff] [blame] | 502 | try: |
| 503 | with self.assertRaises(OSError) as cm: |
| 504 | os.writev(fd, [b"x" * 2**16] * 2**15) |
| 505 | self.assertEqual(cm.exception.errno, errno.EINVAL) |
| 506 | finally: |
| 507 | os.close(fd) |
| 508 | |
Ross Lagerwall | 7807c35 | 2011-03-17 20:20:30 +0200 | [diff] [blame] | 509 | @unittest.skipUnless(hasattr(posix, 'readv'), "test needs posix.readv()") |
| 510 | def test_readv(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 511 | fd = os.open(os_helper.TESTFN, os.O_RDWR | os.O_CREAT) |
Ross Lagerwall | 7807c35 | 2011-03-17 20:20:30 +0200 | [diff] [blame] | 512 | try: |
| 513 | os.write(fd, b'test1tt2t3') |
| 514 | os.lseek(fd, 0, os.SEEK_SET) |
| 515 | buf = [bytearray(i) for i in [5, 3, 2]] |
| 516 | self.assertEqual(posix.readv(fd, buf), 10) |
| 517 | self.assertEqual([b'test1', b'tt2', b't3'], [bytes(i) for i in buf]) |
Victor Stinner | 57ddf78 | 2014-01-08 15:21:28 +0100 | [diff] [blame] | 518 | |
| 519 | # Issue #20113: empty list of buffers should not crash |
Victor Stinner | cd5ca6a | 2014-01-08 16:01:31 +0100 | [diff] [blame] | 520 | try: |
| 521 | size = posix.readv(fd, []) |
| 522 | except OSError: |
| 523 | # readv(fd, []) raises OSError(22, "Invalid argument") |
| 524 | # on OpenIndiana |
| 525 | pass |
| 526 | else: |
| 527 | self.assertEqual(size, 0) |
Ross Lagerwall | 7807c35 | 2011-03-17 20:20:30 +0200 | [diff] [blame] | 528 | finally: |
| 529 | os.close(fd) |
| 530 | |
Serhiy Storchaka | 9d57273 | 2018-07-31 10:24:54 +0300 | [diff] [blame] | 531 | @unittest.skipUnless(hasattr(posix, 'readv'), "test needs posix.readv()") |
| 532 | @requires_32b |
| 533 | def test_readv_overflow_32bits(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 534 | fd = os.open(os_helper.TESTFN, os.O_RDWR | os.O_CREAT) |
Serhiy Storchaka | 9d57273 | 2018-07-31 10:24:54 +0300 | [diff] [blame] | 535 | try: |
| 536 | buf = [bytearray(2**16)] * 2**15 |
| 537 | with self.assertRaises(OSError) as cm: |
| 538 | os.readv(fd, buf) |
| 539 | self.assertEqual(cm.exception.errno, errno.EINVAL) |
| 540 | self.assertEqual(bytes(buf[0]), b'\0'* 2**16) |
| 541 | finally: |
| 542 | os.close(fd) |
| 543 | |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 544 | @unittest.skipUnless(hasattr(posix, 'dup'), |
| 545 | 'test needs posix.dup()') |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 546 | def test_dup(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 547 | fp = open(os_helper.TESTFN) |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 548 | try: |
| 549 | fd = posix.dup(fp.fileno()) |
| 550 | self.assertIsInstance(fd, int) |
| 551 | os.close(fd) |
| 552 | finally: |
| 553 | fp.close() |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 554 | |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 555 | @unittest.skipUnless(hasattr(posix, 'confstr'), |
| 556 | 'test needs posix.confstr()') |
Thomas Wouters | 49fd7fa | 2006-04-21 10:40:58 +0000 | [diff] [blame] | 557 | def test_confstr(self): |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 558 | self.assertRaises(ValueError, posix.confstr, "CS_garbage") |
| 559 | self.assertEqual(len(posix.confstr("CS_PATH")) > 0, True) |
Thomas Wouters | 49fd7fa | 2006-04-21 10:40:58 +0000 | [diff] [blame] | 560 | |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 561 | @unittest.skipUnless(hasattr(posix, 'dup2'), |
| 562 | 'test needs posix.dup2()') |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 563 | def test_dup2(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 564 | fp1 = open(os_helper.TESTFN) |
| 565 | fp2 = open(os_helper.TESTFN) |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 566 | try: |
| 567 | posix.dup2(fp1.fileno(), fp2.fileno()) |
| 568 | finally: |
| 569 | fp1.close() |
| 570 | fp2.close() |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 571 | |
Charles-François Natali | 1e045b1 | 2011-05-22 20:42:32 +0200 | [diff] [blame] | 572 | @unittest.skipUnless(hasattr(os, 'O_CLOEXEC'), "needs os.O_CLOEXEC") |
Charles-François Natali | 239bb96 | 2011-06-03 12:55:15 +0200 | [diff] [blame] | 573 | @support.requires_linux_version(2, 6, 23) |
Charles-François Natali | 1e045b1 | 2011-05-22 20:42:32 +0200 | [diff] [blame] | 574 | def test_oscloexec(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 575 | fd = os.open(os_helper.TESTFN, os.O_RDONLY|os.O_CLOEXEC) |
Charles-François Natali | 1e045b1 | 2011-05-22 20:42:32 +0200 | [diff] [blame] | 576 | self.addCleanup(os.close, fd) |
Victor Stinner | 1db9e7b | 2014-07-29 22:32:47 +0200 | [diff] [blame] | 577 | self.assertFalse(os.get_inheritable(fd)) |
Charles-François Natali | 1e045b1 | 2011-05-22 20:42:32 +0200 | [diff] [blame] | 578 | |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 579 | @unittest.skipUnless(hasattr(posix, 'O_EXLOCK'), |
| 580 | 'test needs posix.O_EXLOCK') |
Skip Montanaro | 9847000 | 2005-06-17 01:14:49 +0000 | [diff] [blame] | 581 | def test_osexlock(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 582 | fd = os.open(os_helper.TESTFN, |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 583 | os.O_WRONLY|os.O_EXLOCK|os.O_CREAT) |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 584 | self.assertRaises(OSError, os.open, os_helper.TESTFN, |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 585 | os.O_WRONLY|os.O_EXLOCK|os.O_NONBLOCK) |
| 586 | os.close(fd) |
| 587 | |
| 588 | if hasattr(posix, "O_SHLOCK"): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 589 | fd = os.open(os_helper.TESTFN, |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 590 | os.O_WRONLY|os.O_SHLOCK|os.O_CREAT) |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 591 | self.assertRaises(OSError, os.open, os_helper.TESTFN, |
Skip Montanaro | 9847000 | 2005-06-17 01:14:49 +0000 | [diff] [blame] | 592 | os.O_WRONLY|os.O_EXLOCK|os.O_NONBLOCK) |
| 593 | os.close(fd) |
| 594 | |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 595 | @unittest.skipUnless(hasattr(posix, 'O_SHLOCK'), |
| 596 | 'test needs posix.O_SHLOCK') |
Skip Montanaro | 9847000 | 2005-06-17 01:14:49 +0000 | [diff] [blame] | 597 | def test_osshlock(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 598 | fd1 = os.open(os_helper.TESTFN, |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 599 | os.O_WRONLY|os.O_SHLOCK|os.O_CREAT) |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 600 | fd2 = os.open(os_helper.TESTFN, |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 601 | os.O_WRONLY|os.O_SHLOCK|os.O_CREAT) |
| 602 | os.close(fd2) |
| 603 | os.close(fd1) |
| 604 | |
| 605 | if hasattr(posix, "O_EXLOCK"): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 606 | fd = os.open(os_helper.TESTFN, |
Skip Montanaro | 9847000 | 2005-06-17 01:14:49 +0000 | [diff] [blame] | 607 | os.O_WRONLY|os.O_SHLOCK|os.O_CREAT) |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 608 | self.assertRaises(OSError, os.open, os_helper.TESTFN, |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 609 | os.O_RDONLY|os.O_EXLOCK|os.O_NONBLOCK) |
| 610 | os.close(fd) |
Skip Montanaro | 9847000 | 2005-06-17 01:14:49 +0000 | [diff] [blame] | 611 | |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 612 | @unittest.skipUnless(hasattr(posix, 'fstat'), |
| 613 | 'test needs posix.fstat()') |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 614 | def test_fstat(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 615 | fp = open(os_helper.TESTFN) |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 616 | try: |
| 617 | self.assertTrue(posix.fstat(fp.fileno())) |
| 618 | self.assertTrue(posix.stat(fp.fileno())) |
Serhiy Storchaka | a2ad5c3 | 2013-01-07 23:13:46 +0200 | [diff] [blame] | 619 | |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 620 | self.assertRaisesRegex(TypeError, |
Brett Cannon | 3f9183b | 2016-08-26 14:44:48 -0700 | [diff] [blame] | 621 | 'should be string, bytes, os.PathLike or integer, not', |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 622 | posix.stat, float(fp.fileno())) |
| 623 | finally: |
| 624 | fp.close() |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 625 | |
| 626 | def test_stat(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 627 | self.assertTrue(posix.stat(os_helper.TESTFN)) |
| 628 | self.assertTrue(posix.stat(os.fsencode(os_helper.TESTFN))) |
Serhiy Storchaka | a2ad5c3 | 2013-01-07 23:13:46 +0200 | [diff] [blame] | 629 | |
Serhiy Storchaka | d73c318 | 2016-08-06 23:22:08 +0300 | [diff] [blame] | 630 | self.assertWarnsRegex(DeprecationWarning, |
Brett Cannon | 3f9183b | 2016-08-26 14:44:48 -0700 | [diff] [blame] | 631 | 'should be string, bytes, os.PathLike or integer, not', |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 632 | posix.stat, bytearray(os.fsencode(os_helper.TESTFN))) |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 633 | self.assertRaisesRegex(TypeError, |
Brett Cannon | 3f9183b | 2016-08-26 14:44:48 -0700 | [diff] [blame] | 634 | 'should be string, bytes, os.PathLike or integer, not', |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 635 | posix.stat, None) |
| 636 | self.assertRaisesRegex(TypeError, |
Brett Cannon | 3f9183b | 2016-08-26 14:44:48 -0700 | [diff] [blame] | 637 | 'should be string, bytes, os.PathLike or integer, not', |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 638 | posix.stat, list(os_helper.TESTFN)) |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 639 | self.assertRaisesRegex(TypeError, |
Brett Cannon | 3f9183b | 2016-08-26 14:44:48 -0700 | [diff] [blame] | 640 | 'should be string, bytes, os.PathLike or integer, not', |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 641 | posix.stat, list(os.fsencode(os_helper.TESTFN))) |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 642 | |
Benjamin Peterson | 052a02b | 2010-08-17 01:27:09 +0000 | [diff] [blame] | 643 | @unittest.skipUnless(hasattr(posix, 'mkfifo'), "don't have mkfifo()") |
| 644 | def test_mkfifo(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 645 | os_helper.unlink(os_helper.TESTFN) |
xdegaye | 92c2ca7 | 2017-11-12 17:31:07 +0100 | [diff] [blame] | 646 | try: |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 647 | posix.mkfifo(os_helper.TESTFN, stat.S_IRUSR | stat.S_IWUSR) |
xdegaye | 92c2ca7 | 2017-11-12 17:31:07 +0100 | [diff] [blame] | 648 | except PermissionError as e: |
| 649 | self.skipTest('posix.mkfifo(): %s' % e) |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 650 | self.assertTrue(stat.S_ISFIFO(posix.stat(os_helper.TESTFN).st_mode)) |
Benjamin Peterson | 052a02b | 2010-08-17 01:27:09 +0000 | [diff] [blame] | 651 | |
| 652 | @unittest.skipUnless(hasattr(posix, 'mknod') and hasattr(stat, 'S_IFIFO'), |
| 653 | "don't have mknod()/S_IFIFO") |
| 654 | def test_mknod(self): |
| 655 | # Test using mknod() to create a FIFO (the only use specified |
| 656 | # by POSIX). |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 657 | os_helper.unlink(os_helper.TESTFN) |
Benjamin Peterson | 052a02b | 2010-08-17 01:27:09 +0000 | [diff] [blame] | 658 | mode = stat.S_IFIFO | stat.S_IRUSR | stat.S_IWUSR |
| 659 | try: |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 660 | posix.mknod(os_helper.TESTFN, mode, 0) |
Benjamin Peterson | 052a02b | 2010-08-17 01:27:09 +0000 | [diff] [blame] | 661 | except OSError as e: |
| 662 | # Some old systems don't allow unprivileged users to use |
| 663 | # mknod(), or only support creating device nodes. |
xdegaye | 92c2ca7 | 2017-11-12 17:31:07 +0100 | [diff] [blame] | 664 | self.assertIn(e.errno, (errno.EPERM, errno.EINVAL, errno.EACCES)) |
Benjamin Peterson | 052a02b | 2010-08-17 01:27:09 +0000 | [diff] [blame] | 665 | else: |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 666 | self.assertTrue(stat.S_ISFIFO(posix.stat(os_helper.TESTFN).st_mode)) |
Benjamin Peterson | 052a02b | 2010-08-17 01:27:09 +0000 | [diff] [blame] | 667 | |
Martin Panter | bf19d16 | 2015-09-09 01:01:13 +0000 | [diff] [blame] | 668 | # Keyword arguments are also supported |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 669 | os_helper.unlink(os_helper.TESTFN) |
Martin Panter | bf19d16 | 2015-09-09 01:01:13 +0000 | [diff] [blame] | 670 | try: |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 671 | posix.mknod(path=os_helper.TESTFN, mode=mode, device=0, |
Martin Panter | bf19d16 | 2015-09-09 01:01:13 +0000 | [diff] [blame] | 672 | dir_fd=None) |
| 673 | except OSError as e: |
xdegaye | 92c2ca7 | 2017-11-12 17:31:07 +0100 | [diff] [blame] | 674 | self.assertIn(e.errno, (errno.EPERM, errno.EINVAL, errno.EACCES)) |
Martin Panter | bf19d16 | 2015-09-09 01:01:13 +0000 | [diff] [blame] | 675 | |
Serhiy Storchaka | 16b2e4f | 2015-04-20 09:22:13 +0300 | [diff] [blame] | 676 | @unittest.skipUnless(hasattr(posix, 'makedev'), 'test needs posix.makedev()') |
| 677 | def test_makedev(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 678 | st = posix.stat(os_helper.TESTFN) |
Serhiy Storchaka | 16b2e4f | 2015-04-20 09:22:13 +0300 | [diff] [blame] | 679 | dev = st.st_dev |
| 680 | self.assertIsInstance(dev, int) |
| 681 | self.assertGreaterEqual(dev, 0) |
| 682 | |
| 683 | major = posix.major(dev) |
| 684 | self.assertIsInstance(major, int) |
| 685 | self.assertGreaterEqual(major, 0) |
| 686 | self.assertEqual(posix.major(dev), major) |
| 687 | self.assertRaises(TypeError, posix.major, float(dev)) |
| 688 | self.assertRaises(TypeError, posix.major) |
| 689 | self.assertRaises((ValueError, OverflowError), posix.major, -1) |
| 690 | |
| 691 | minor = posix.minor(dev) |
| 692 | self.assertIsInstance(minor, int) |
| 693 | self.assertGreaterEqual(minor, 0) |
| 694 | self.assertEqual(posix.minor(dev), minor) |
| 695 | self.assertRaises(TypeError, posix.minor, float(dev)) |
| 696 | self.assertRaises(TypeError, posix.minor) |
| 697 | self.assertRaises((ValueError, OverflowError), posix.minor, -1) |
| 698 | |
| 699 | self.assertEqual(posix.makedev(major, minor), dev) |
| 700 | self.assertRaises(TypeError, posix.makedev, float(major), minor) |
| 701 | self.assertRaises(TypeError, posix.makedev, major, float(minor)) |
| 702 | self.assertRaises(TypeError, posix.makedev, major) |
| 703 | self.assertRaises(TypeError, posix.makedev) |
| 704 | |
Serhiy Storchaka | 7cf5599 | 2013-02-10 21:56:49 +0200 | [diff] [blame] | 705 | def _test_all_chown_common(self, chown_func, first_param, stat_func): |
Benjamin Peterson | 1baf465 | 2009-12-31 03:11:23 +0000 | [diff] [blame] | 706 | """Common code for chown, fchown and lchown tests.""" |
Serhiy Storchaka | 54db2fd | 2013-02-20 19:40:25 +0200 | [diff] [blame] | 707 | def check_stat(uid, gid): |
Serhiy Storchaka | 7cf5599 | 2013-02-10 21:56:49 +0200 | [diff] [blame] | 708 | if stat_func is not None: |
| 709 | stat = stat_func(first_param) |
Serhiy Storchaka | 54db2fd | 2013-02-20 19:40:25 +0200 | [diff] [blame] | 710 | self.assertEqual(stat.st_uid, uid) |
| 711 | self.assertEqual(stat.st_gid, gid) |
| 712 | uid = os.getuid() |
| 713 | gid = os.getgid() |
Charles-François Natali | ab2d58e | 2012-04-17 19:48:35 +0200 | [diff] [blame] | 714 | # test a successful chown call |
Serhiy Storchaka | 54db2fd | 2013-02-20 19:40:25 +0200 | [diff] [blame] | 715 | chown_func(first_param, uid, gid) |
| 716 | check_stat(uid, gid) |
| 717 | chown_func(first_param, -1, gid) |
| 718 | check_stat(uid, gid) |
| 719 | chown_func(first_param, uid, -1) |
| 720 | check_stat(uid, gid) |
Charles-François Natali | ab2d58e | 2012-04-17 19:48:35 +0200 | [diff] [blame] | 721 | |
Serhiy Storchaka | 54db2fd | 2013-02-20 19:40:25 +0200 | [diff] [blame] | 722 | if uid == 0: |
| 723 | # Try an amusingly large uid/gid to make sure we handle |
| 724 | # large unsigned values. (chown lets you use any |
| 725 | # uid/gid you like, even if they aren't defined.) |
| 726 | # |
| 727 | # This problem keeps coming up: |
| 728 | # http://bugs.python.org/issue1747858 |
| 729 | # http://bugs.python.org/issue4591 |
| 730 | # http://bugs.python.org/issue15301 |
| 731 | # Hopefully the fix in 4591 fixes it for good! |
| 732 | # |
| 733 | # This part of the test only runs when run as root. |
| 734 | # Only scary people run their tests as root. |
| 735 | |
| 736 | big_value = 2**31 |
| 737 | chown_func(first_param, big_value, big_value) |
| 738 | check_stat(big_value, big_value) |
| 739 | chown_func(first_param, -1, -1) |
| 740 | check_stat(big_value, big_value) |
| 741 | chown_func(first_param, uid, gid) |
| 742 | check_stat(uid, gid) |
Charles-François Natali | ab2d58e | 2012-04-17 19:48:35 +0200 | [diff] [blame] | 743 | elif platform.system() in ('HP-UX', 'SunOS'): |
| 744 | # HP-UX and Solaris can allow a non-root user to chown() to root |
| 745 | # (issue #5113) |
| 746 | raise unittest.SkipTest("Skipping because of non-standard chown() " |
| 747 | "behavior") |
Benjamin Peterson | 1baf465 | 2009-12-31 03:11:23 +0000 | [diff] [blame] | 748 | else: |
| 749 | # non-root cannot chown to root, raises OSError |
Serhiy Storchaka | 7cf5599 | 2013-02-10 21:56:49 +0200 | [diff] [blame] | 750 | self.assertRaises(OSError, chown_func, first_param, 0, 0) |
Serhiy Storchaka | 54db2fd | 2013-02-20 19:40:25 +0200 | [diff] [blame] | 751 | check_stat(uid, gid) |
Serhiy Storchaka | 7cf5599 | 2013-02-10 21:56:49 +0200 | [diff] [blame] | 752 | self.assertRaises(OSError, chown_func, first_param, 0, -1) |
Serhiy Storchaka | 54db2fd | 2013-02-20 19:40:25 +0200 | [diff] [blame] | 753 | check_stat(uid, gid) |
Serhiy Storchaka | a2964b3 | 2013-02-21 14:34:36 +0200 | [diff] [blame] | 754 | if 0 not in os.getgroups(): |
Serhiy Storchaka | b3d62ce | 2013-02-20 19:48:22 +0200 | [diff] [blame] | 755 | self.assertRaises(OSError, chown_func, first_param, -1, 0) |
| 756 | check_stat(uid, gid) |
Serhiy Storchaka | 54db2fd | 2013-02-20 19:40:25 +0200 | [diff] [blame] | 757 | # test illegal types |
| 758 | for t in str, float: |
| 759 | self.assertRaises(TypeError, chown_func, first_param, t(uid), gid) |
| 760 | check_stat(uid, gid) |
| 761 | self.assertRaises(TypeError, chown_func, first_param, uid, t(gid)) |
| 762 | check_stat(uid, gid) |
Christian Heimes | d5e2b6f | 2008-03-19 21:50:51 +0000 | [diff] [blame] | 763 | |
Benjamin Peterson | 1baf465 | 2009-12-31 03:11:23 +0000 | [diff] [blame] | 764 | @unittest.skipUnless(hasattr(posix, 'chown'), "test needs os.chown()") |
| 765 | def test_chown(self): |
| 766 | # raise an OSError if the file does not exist |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 767 | os.unlink(os_helper.TESTFN) |
| 768 | self.assertRaises(OSError, posix.chown, os_helper.TESTFN, -1, -1) |
Christian Heimes | d5e2b6f | 2008-03-19 21:50:51 +0000 | [diff] [blame] | 769 | |
Benjamin Peterson | 1baf465 | 2009-12-31 03:11:23 +0000 | [diff] [blame] | 770 | # re-create the file |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 771 | os_helper.create_empty_file(os_helper.TESTFN) |
| 772 | self._test_all_chown_common(posix.chown, os_helper.TESTFN, posix.stat) |
Benjamin Peterson | 1baf465 | 2009-12-31 03:11:23 +0000 | [diff] [blame] | 773 | |
| 774 | @unittest.skipUnless(hasattr(posix, 'fchown'), "test needs os.fchown()") |
| 775 | def test_fchown(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 776 | os.unlink(os_helper.TESTFN) |
Benjamin Peterson | 1baf465 | 2009-12-31 03:11:23 +0000 | [diff] [blame] | 777 | |
| 778 | # re-create the file |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 779 | test_file = open(os_helper.TESTFN, 'w') |
Benjamin Peterson | 1baf465 | 2009-12-31 03:11:23 +0000 | [diff] [blame] | 780 | try: |
| 781 | fd = test_file.fileno() |
Serhiy Storchaka | 7cf5599 | 2013-02-10 21:56:49 +0200 | [diff] [blame] | 782 | self._test_all_chown_common(posix.fchown, fd, |
| 783 | getattr(posix, 'fstat', None)) |
Benjamin Peterson | 1baf465 | 2009-12-31 03:11:23 +0000 | [diff] [blame] | 784 | finally: |
| 785 | test_file.close() |
| 786 | |
| 787 | @unittest.skipUnless(hasattr(posix, 'lchown'), "test needs os.lchown()") |
| 788 | def test_lchown(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 789 | os.unlink(os_helper.TESTFN) |
Benjamin Peterson | 1baf465 | 2009-12-31 03:11:23 +0000 | [diff] [blame] | 790 | # create a symlink |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 791 | os.symlink(_DUMMY_SYMLINK, os_helper.TESTFN) |
| 792 | self._test_all_chown_common(posix.lchown, os_helper.TESTFN, |
Serhiy Storchaka | 7cf5599 | 2013-02-10 21:56:49 +0200 | [diff] [blame] | 793 | getattr(posix, 'lstat', None)) |
Christian Heimes | d5e2b6f | 2008-03-19 21:50:51 +0000 | [diff] [blame] | 794 | |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 795 | @unittest.skipUnless(hasattr(posix, 'chdir'), 'test needs posix.chdir()') |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 796 | def test_chdir(self): |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 797 | posix.chdir(os.curdir) |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 798 | self.assertRaises(OSError, posix.chdir, os_helper.TESTFN) |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 799 | |
Martin v. Löwis | c9e1c7d | 2010-07-23 12:16:41 +0000 | [diff] [blame] | 800 | def test_listdir(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 801 | self.assertIn(os_helper.TESTFN, posix.listdir(os.curdir)) |
Martin v. Löwis | c9e1c7d | 2010-07-23 12:16:41 +0000 | [diff] [blame] | 802 | |
| 803 | def test_listdir_default(self): |
Larry Hastings | fdaea06 | 2012-06-25 04:42:23 -0700 | [diff] [blame] | 804 | # When listdir is called without argument, |
| 805 | # it's the same as listdir(os.curdir). |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 806 | self.assertIn(os_helper.TESTFN, posix.listdir()) |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 807 | |
Larry Hastings | fdaea06 | 2012-06-25 04:42:23 -0700 | [diff] [blame] | 808 | def test_listdir_bytes(self): |
| 809 | # When listdir is called with a bytes object, |
| 810 | # the returned strings are of type bytes. |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 811 | self.assertIn(os.fsencode(os_helper.TESTFN), posix.listdir(b'.')) |
Serhiy Storchaka | 1180e5a | 2017-07-11 06:36:46 +0300 | [diff] [blame] | 812 | |
| 813 | def test_listdir_bytes_like(self): |
| 814 | for cls in bytearray, memoryview: |
| 815 | with self.assertWarns(DeprecationWarning): |
| 816 | names = posix.listdir(cls(b'.')) |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 817 | self.assertIn(os.fsencode(os_helper.TESTFN), names) |
Serhiy Storchaka | 1180e5a | 2017-07-11 06:36:46 +0300 | [diff] [blame] | 818 | for name in names: |
| 819 | self.assertIs(type(name), bytes) |
Larry Hastings | fdaea06 | 2012-06-25 04:42:23 -0700 | [diff] [blame] | 820 | |
| 821 | @unittest.skipUnless(posix.listdir in os.supports_fd, |
| 822 | "test needs fd support for posix.listdir()") |
| 823 | def test_listdir_fd(self): |
Antoine Pitrou | 8250e23 | 2011-02-25 23:41:16 +0000 | [diff] [blame] | 824 | f = posix.open(posix.getcwd(), posix.O_RDONLY) |
Charles-François Natali | 7546ad3 | 2012-01-08 18:34:06 +0100 | [diff] [blame] | 825 | self.addCleanup(posix.close, f) |
Antoine Pitrou | 8250e23 | 2011-02-25 23:41:16 +0000 | [diff] [blame] | 826 | self.assertEqual( |
| 827 | sorted(posix.listdir('.')), |
Larry Hastings | 9cf065c | 2012-06-22 16:30:09 -0700 | [diff] [blame] | 828 | sorted(posix.listdir(f)) |
Antoine Pitrou | 8250e23 | 2011-02-25 23:41:16 +0000 | [diff] [blame] | 829 | ) |
Charles-François Natali | 7546ad3 | 2012-01-08 18:34:06 +0100 | [diff] [blame] | 830 | # Check that the fd offset was reset (issue #13739) |
Charles-François Natali | 7546ad3 | 2012-01-08 18:34:06 +0100 | [diff] [blame] | 831 | self.assertEqual( |
| 832 | sorted(posix.listdir('.')), |
Larry Hastings | 9cf065c | 2012-06-22 16:30:09 -0700 | [diff] [blame] | 833 | sorted(posix.listdir(f)) |
Charles-François Natali | 7546ad3 | 2012-01-08 18:34:06 +0100 | [diff] [blame] | 834 | ) |
Antoine Pitrou | 8250e23 | 2011-02-25 23:41:16 +0000 | [diff] [blame] | 835 | |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 836 | @unittest.skipUnless(hasattr(posix, 'access'), 'test needs posix.access()') |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 837 | def test_access(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 838 | self.assertTrue(posix.access(os_helper.TESTFN, os.R_OK)) |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 839 | |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 840 | @unittest.skipUnless(hasattr(posix, 'umask'), 'test needs posix.umask()') |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 841 | def test_umask(self): |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 842 | old_mask = posix.umask(0) |
| 843 | self.assertIsInstance(old_mask, int) |
| 844 | posix.umask(old_mask) |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 845 | |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 846 | @unittest.skipUnless(hasattr(posix, 'strerror'), |
| 847 | 'test needs posix.strerror()') |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 848 | def test_strerror(self): |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 849 | self.assertTrue(posix.strerror(0)) |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 850 | |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 851 | @unittest.skipUnless(hasattr(posix, 'pipe'), 'test needs posix.pipe()') |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 852 | def test_pipe(self): |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 853 | reader, writer = posix.pipe() |
| 854 | os.close(reader) |
| 855 | os.close(writer) |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 856 | |
Charles-François Natali | daafdd5 | 2011-05-29 20:07:40 +0200 | [diff] [blame] | 857 | @unittest.skipUnless(hasattr(os, 'pipe2'), "test needs os.pipe2()") |
Charles-François Natali | 239bb96 | 2011-06-03 12:55:15 +0200 | [diff] [blame] | 858 | @support.requires_linux_version(2, 6, 27) |
Charles-François Natali | daafdd5 | 2011-05-29 20:07:40 +0200 | [diff] [blame] | 859 | def test_pipe2(self): |
| 860 | self.assertRaises(TypeError, os.pipe2, 'DEADBEEF') |
| 861 | self.assertRaises(TypeError, os.pipe2, 0, 0) |
| 862 | |
Charles-François Natali | 368f34b | 2011-06-06 19:49:47 +0200 | [diff] [blame] | 863 | # try calling with flags = 0, like os.pipe() |
| 864 | r, w = os.pipe2(0) |
Charles-François Natali | daafdd5 | 2011-05-29 20:07:40 +0200 | [diff] [blame] | 865 | os.close(r) |
| 866 | os.close(w) |
| 867 | |
| 868 | # test flags |
| 869 | r, w = os.pipe2(os.O_CLOEXEC|os.O_NONBLOCK) |
| 870 | self.addCleanup(os.close, r) |
| 871 | self.addCleanup(os.close, w) |
Victor Stinner | bff989e | 2013-08-28 12:25:40 +0200 | [diff] [blame] | 872 | self.assertFalse(os.get_inheritable(r)) |
| 873 | self.assertFalse(os.get_inheritable(w)) |
Victor Stinner | 1db9e7b | 2014-07-29 22:32:47 +0200 | [diff] [blame] | 874 | self.assertFalse(os.get_blocking(r)) |
| 875 | self.assertFalse(os.get_blocking(w)) |
Charles-François Natali | daafdd5 | 2011-05-29 20:07:40 +0200 | [diff] [blame] | 876 | # try reading from an empty pipe: this should fail, not block |
| 877 | self.assertRaises(OSError, os.read, r, 1) |
| 878 | # try a write big enough to fill-up the pipe: this should either |
| 879 | # fail or perform a partial write, not block |
| 880 | try: |
| 881 | os.write(w, b'x' * support.PIPE_MAX_SIZE) |
| 882 | except OSError: |
| 883 | pass |
| 884 | |
Serhiy Storchaka | 5cfc79d | 2014-02-07 10:06:39 +0200 | [diff] [blame] | 885 | @support.cpython_only |
| 886 | @unittest.skipUnless(hasattr(os, 'pipe2'), "test needs os.pipe2()") |
| 887 | @support.requires_linux_version(2, 6, 27) |
| 888 | def test_pipe2_c_limits(self): |
Serhiy Storchaka | 7898043 | 2013-01-15 01:12:17 +0200 | [diff] [blame] | 889 | # Issue 15989 |
Serhiy Storchaka | 5cfc79d | 2014-02-07 10:06:39 +0200 | [diff] [blame] | 890 | import _testcapi |
Serhiy Storchaka | 7898043 | 2013-01-15 01:12:17 +0200 | [diff] [blame] | 891 | self.assertRaises(OverflowError, os.pipe2, _testcapi.INT_MAX + 1) |
| 892 | self.assertRaises(OverflowError, os.pipe2, _testcapi.UINT_MAX + 1) |
| 893 | |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 894 | @unittest.skipUnless(hasattr(posix, 'utime'), 'test needs posix.utime()') |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 895 | def test_utime(self): |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 896 | now = time.time() |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 897 | posix.utime(os_helper.TESTFN, None) |
| 898 | self.assertRaises(TypeError, posix.utime, |
| 899 | os_helper.TESTFN, (None, None)) |
| 900 | self.assertRaises(TypeError, posix.utime, |
| 901 | os_helper.TESTFN, (now, None)) |
| 902 | self.assertRaises(TypeError, posix.utime, |
| 903 | os_helper.TESTFN, (None, now)) |
| 904 | posix.utime(os_helper.TESTFN, (int(now), int(now))) |
| 905 | posix.utime(os_helper.TESTFN, (now, now)) |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 906 | |
Larry Hastings | 9cf065c | 2012-06-22 16:30:09 -0700 | [diff] [blame] | 907 | def _test_chflags_regular_file(self, chflags_func, target_file, **kwargs): |
Ned Deily | 3eb67d5 | 2011-06-28 00:00:28 -0700 | [diff] [blame] | 908 | st = os.stat(target_file) |
| 909 | self.assertTrue(hasattr(st, 'st_flags')) |
Trent Nelson | 75959cf | 2012-08-21 23:59:31 +0000 | [diff] [blame] | 910 | |
| 911 | # ZFS returns EOPNOTSUPP when attempting to set flag UF_IMMUTABLE. |
| 912 | flags = st.st_flags | stat.UF_IMMUTABLE |
| 913 | try: |
| 914 | chflags_func(target_file, flags, **kwargs) |
| 915 | except OSError as err: |
| 916 | if err.errno != errno.EOPNOTSUPP: |
| 917 | raise |
| 918 | msg = 'chflag UF_IMMUTABLE not supported by underlying fs' |
| 919 | self.skipTest(msg) |
| 920 | |
Ned Deily | 3eb67d5 | 2011-06-28 00:00:28 -0700 | [diff] [blame] | 921 | try: |
| 922 | new_st = os.stat(target_file) |
| 923 | self.assertEqual(st.st_flags | stat.UF_IMMUTABLE, new_st.st_flags) |
| 924 | try: |
| 925 | fd = open(target_file, 'w+') |
Andrew Svetlov | f7a17b4 | 2012-12-25 16:47:37 +0200 | [diff] [blame] | 926 | except OSError as e: |
Ned Deily | 3eb67d5 | 2011-06-28 00:00:28 -0700 | [diff] [blame] | 927 | self.assertEqual(e.errno, errno.EPERM) |
| 928 | finally: |
| 929 | posix.chflags(target_file, st.st_flags) |
Thomas Wouters | cf297e4 | 2007-02-23 15:07:44 +0000 | [diff] [blame] | 930 | |
Ned Deily | 3eb67d5 | 2011-06-28 00:00:28 -0700 | [diff] [blame] | 931 | @unittest.skipUnless(hasattr(posix, 'chflags'), 'test needs os.chflags()') |
| 932 | def test_chflags(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 933 | self._test_chflags_regular_file(posix.chflags, os_helper.TESTFN) |
Ned Deily | 3eb67d5 | 2011-06-28 00:00:28 -0700 | [diff] [blame] | 934 | |
| 935 | @unittest.skipUnless(hasattr(posix, 'lchflags'), 'test needs os.lchflags()') |
| 936 | def test_lchflags_regular_file(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 937 | self._test_chflags_regular_file(posix.lchflags, os_helper.TESTFN) |
| 938 | self._test_chflags_regular_file(posix.chflags, os_helper.TESTFN, |
| 939 | follow_symlinks=False) |
Ned Deily | 3eb67d5 | 2011-06-28 00:00:28 -0700 | [diff] [blame] | 940 | |
| 941 | @unittest.skipUnless(hasattr(posix, 'lchflags'), 'test needs os.lchflags()') |
| 942 | def test_lchflags_symlink(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 943 | testfn_st = os.stat(os_helper.TESTFN) |
Ned Deily | 3eb67d5 | 2011-06-28 00:00:28 -0700 | [diff] [blame] | 944 | |
| 945 | self.assertTrue(hasattr(testfn_st, 'st_flags')) |
| 946 | |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 947 | os.symlink(os_helper.TESTFN, _DUMMY_SYMLINK) |
Ned Deily | 3eb67d5 | 2011-06-28 00:00:28 -0700 | [diff] [blame] | 948 | self.teardown_files.append(_DUMMY_SYMLINK) |
| 949 | dummy_symlink_st = os.lstat(_DUMMY_SYMLINK) |
| 950 | |
Larry Hastings | 9cf065c | 2012-06-22 16:30:09 -0700 | [diff] [blame] | 951 | def chflags_nofollow(path, flags): |
| 952 | return posix.chflags(path, flags, follow_symlinks=False) |
Ned Deily | 3eb67d5 | 2011-06-28 00:00:28 -0700 | [diff] [blame] | 953 | |
Larry Hastings | 9cf065c | 2012-06-22 16:30:09 -0700 | [diff] [blame] | 954 | for fn in (posix.lchflags, chflags_nofollow): |
Trent Nelson | 75959cf | 2012-08-21 23:59:31 +0000 | [diff] [blame] | 955 | # ZFS returns EOPNOTSUPP when attempting to set flag UF_IMMUTABLE. |
| 956 | flags = dummy_symlink_st.st_flags | stat.UF_IMMUTABLE |
| 957 | try: |
| 958 | fn(_DUMMY_SYMLINK, flags) |
| 959 | except OSError as err: |
| 960 | if err.errno != errno.EOPNOTSUPP: |
| 961 | raise |
| 962 | msg = 'chflag UF_IMMUTABLE not supported by underlying fs' |
| 963 | self.skipTest(msg) |
Larry Hastings | 9cf065c | 2012-06-22 16:30:09 -0700 | [diff] [blame] | 964 | try: |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 965 | new_testfn_st = os.stat(os_helper.TESTFN) |
Larry Hastings | 9cf065c | 2012-06-22 16:30:09 -0700 | [diff] [blame] | 966 | new_dummy_symlink_st = os.lstat(_DUMMY_SYMLINK) |
| 967 | |
| 968 | self.assertEqual(testfn_st.st_flags, new_testfn_st.st_flags) |
| 969 | self.assertEqual(dummy_symlink_st.st_flags | stat.UF_IMMUTABLE, |
| 970 | new_dummy_symlink_st.st_flags) |
| 971 | finally: |
| 972 | fn(_DUMMY_SYMLINK, dummy_symlink_st.st_flags) |
Thomas Wouters | cf297e4 | 2007-02-23 15:07:44 +0000 | [diff] [blame] | 973 | |
Guido van Rossum | 98297ee | 2007-11-06 21:34:58 +0000 | [diff] [blame] | 974 | def test_environ(self): |
Victor Stinner | 17b490d | 2010-05-06 22:19:30 +0000 | [diff] [blame] | 975 | if os.name == "nt": |
| 976 | item_type = str |
| 977 | else: |
| 978 | item_type = bytes |
Guido van Rossum | 98297ee | 2007-11-06 21:34:58 +0000 | [diff] [blame] | 979 | for k, v in posix.environ.items(): |
Victor Stinner | 17b490d | 2010-05-06 22:19:30 +0000 | [diff] [blame] | 980 | self.assertEqual(type(k), item_type) |
| 981 | self.assertEqual(type(v), item_type) |
Guido van Rossum | 98297ee | 2007-11-06 21:34:58 +0000 | [diff] [blame] | 982 | |
Serhiy Storchaka | 7770394 | 2017-06-25 07:33:01 +0300 | [diff] [blame] | 983 | def test_putenv(self): |
| 984 | with self.assertRaises(ValueError): |
| 985 | os.putenv('FRUIT\0VEGETABLE', 'cabbage') |
| 986 | with self.assertRaises(ValueError): |
| 987 | os.putenv(b'FRUIT\0VEGETABLE', b'cabbage') |
| 988 | with self.assertRaises(ValueError): |
| 989 | os.putenv('FRUIT', 'orange\0VEGETABLE=cabbage') |
| 990 | with self.assertRaises(ValueError): |
| 991 | os.putenv(b'FRUIT', b'orange\0VEGETABLE=cabbage') |
| 992 | with self.assertRaises(ValueError): |
| 993 | os.putenv('FRUIT=ORANGE', 'lemon') |
| 994 | with self.assertRaises(ValueError): |
| 995 | os.putenv(b'FRUIT=ORANGE', b'lemon') |
| 996 | |
Serhiy Storchaka | 4376763 | 2013-11-03 21:31:38 +0200 | [diff] [blame] | 997 | @unittest.skipUnless(hasattr(posix, 'getcwd'), 'test needs posix.getcwd()') |
Benjamin Peterson | dcf97b9 | 2008-07-02 17:30:14 +0000 | [diff] [blame] | 998 | def test_getcwd_long_pathnames(self): |
Benjamin Peterson | 3a7dffa | 2013-08-23 21:01:48 -0500 | [diff] [blame] | 999 | dirname = 'getcwd-test-directory-0123456789abcdef-01234567890abcdef' |
| 1000 | curdir = os.getcwd() |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1001 | base_path = os.path.abspath(os_helper.TESTFN) + '.getcwd' |
Benjamin Peterson | dcf97b9 | 2008-07-02 17:30:14 +0000 | [diff] [blame] | 1002 | |
Benjamin Peterson | 3a7dffa | 2013-08-23 21:01:48 -0500 | [diff] [blame] | 1003 | try: |
| 1004 | os.mkdir(base_path) |
| 1005 | os.chdir(base_path) |
| 1006 | except: |
| 1007 | # Just returning nothing instead of the SkipTest exception, because |
| 1008 | # the test results in Error in that case. Is that ok? |
| 1009 | # raise unittest.SkipTest("cannot create directory for testing") |
| 1010 | return |
Benjamin Peterson | dcf97b9 | 2008-07-02 17:30:14 +0000 | [diff] [blame] | 1011 | |
Benjamin Peterson | 3a7dffa | 2013-08-23 21:01:48 -0500 | [diff] [blame] | 1012 | def _create_and_do_getcwd(dirname, current_path_length = 0): |
| 1013 | try: |
| 1014 | os.mkdir(dirname) |
| 1015 | except: |
| 1016 | raise unittest.SkipTest("mkdir cannot create directory sufficiently deep for getcwd test") |
Benjamin Peterson | dcf97b9 | 2008-07-02 17:30:14 +0000 | [diff] [blame] | 1017 | |
Benjamin Peterson | 3a7dffa | 2013-08-23 21:01:48 -0500 | [diff] [blame] | 1018 | os.chdir(dirname) |
| 1019 | try: |
| 1020 | os.getcwd() |
| 1021 | if current_path_length < 1027: |
| 1022 | _create_and_do_getcwd(dirname, current_path_length + len(dirname) + 1) |
| 1023 | finally: |
| 1024 | os.chdir('..') |
| 1025 | os.rmdir(dirname) |
Benjamin Peterson | dcf97b9 | 2008-07-02 17:30:14 +0000 | [diff] [blame] | 1026 | |
Benjamin Peterson | 3a7dffa | 2013-08-23 21:01:48 -0500 | [diff] [blame] | 1027 | _create_and_do_getcwd(dirname) |
Benjamin Peterson | dcf97b9 | 2008-07-02 17:30:14 +0000 | [diff] [blame] | 1028 | |
Benjamin Peterson | 3a7dffa | 2013-08-23 21:01:48 -0500 | [diff] [blame] | 1029 | finally: |
| 1030 | os.chdir(curdir) |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1031 | os_helper.rmtree(base_path) |
Benjamin Peterson | dcf97b9 | 2008-07-02 17:30:14 +0000 | [diff] [blame] | 1032 | |
Ross Lagerwall | b0ae53d | 2011-06-10 07:30:30 +0200 | [diff] [blame] | 1033 | @unittest.skipUnless(hasattr(posix, 'getgrouplist'), "test needs posix.getgrouplist()") |
| 1034 | @unittest.skipUnless(hasattr(pwd, 'getpwuid'), "test needs pwd.getpwuid()") |
| 1035 | @unittest.skipUnless(hasattr(os, 'getuid'), "test needs os.getuid()") |
| 1036 | def test_getgrouplist(self): |
Ross Lagerwall | a0b315f | 2012-12-13 15:20:26 +0000 | [diff] [blame] | 1037 | user = pwd.getpwuid(os.getuid())[0] |
| 1038 | group = pwd.getpwuid(os.getuid())[3] |
| 1039 | self.assertIn(group, posix.getgrouplist(user, group)) |
Ross Lagerwall | b0ae53d | 2011-06-10 07:30:30 +0200 | [diff] [blame] | 1040 | |
Ross Lagerwall | b0ae53d | 2011-06-10 07:30:30 +0200 | [diff] [blame] | 1041 | |
Antoine Pitrou | 318b8f3 | 2011-01-12 18:45:27 +0000 | [diff] [blame] | 1042 | @unittest.skipUnless(hasattr(os, 'getegid'), "test needs os.getegid()") |
Ronald Oussoren | b6ee4f5 | 2010-07-23 13:53:51 +0000 | [diff] [blame] | 1043 | def test_getgroups(self): |
Jesus Cea | 61f32cb | 2014-06-28 18:39:35 +0200 | [diff] [blame] | 1044 | with os.popen('id -G 2>/dev/null') as idg: |
Ronald Oussoren | b6ee4f5 | 2010-07-23 13:53:51 +0000 | [diff] [blame] | 1045 | groups = idg.read().strip() |
Charles-François Natali | e8a255a | 2012-05-02 20:01:38 +0200 | [diff] [blame] | 1046 | ret = idg.close() |
Ronald Oussoren | b6ee4f5 | 2010-07-23 13:53:51 +0000 | [diff] [blame] | 1047 | |
Xavier de Gaye | 24c3b49 | 2016-10-19 11:00:26 +0200 | [diff] [blame] | 1048 | try: |
| 1049 | idg_groups = set(int(g) for g in groups.split()) |
| 1050 | except ValueError: |
| 1051 | idg_groups = set() |
| 1052 | if ret is not None or not idg_groups: |
Ronald Oussoren | b6ee4f5 | 2010-07-23 13:53:51 +0000 | [diff] [blame] | 1053 | raise unittest.SkipTest("need working 'id -G'") |
| 1054 | |
Ned Deily | 028915e | 2013-02-02 15:08:52 -0800 | [diff] [blame] | 1055 | # Issues 16698: OS X ABIs prior to 10.6 have limits on getgroups() |
| 1056 | if sys.platform == 'darwin': |
| 1057 | import sysconfig |
| 1058 | dt = sysconfig.get_config_var('MACOSX_DEPLOYMENT_TARGET') or '10.0' |
Ned Deily | 04cdfa1 | 2014-06-25 13:36:14 -0700 | [diff] [blame] | 1059 | if tuple(int(n) for n in dt.split('.')[0:2]) < (10, 6): |
Ned Deily | 028915e | 2013-02-02 15:08:52 -0800 | [diff] [blame] | 1060 | raise unittest.SkipTest("getgroups(2) is broken prior to 10.6") |
| 1061 | |
Ronald Oussoren | 7fb6f51 | 2010-08-01 19:18:13 +0000 | [diff] [blame] | 1062 | # 'id -G' and 'os.getgroups()' should return the same |
Xavier de Gaye | 24c3b49 | 2016-10-19 11:00:26 +0200 | [diff] [blame] | 1063 | # groups, ignoring order, duplicates, and the effective gid. |
| 1064 | # #10822/#26944 - It is implementation defined whether |
| 1065 | # posix.getgroups() includes the effective gid. |
| 1066 | symdiff = idg_groups.symmetric_difference(posix.getgroups()) |
| 1067 | self.assertTrue(not symdiff or symdiff == {posix.getegid()}) |
Ronald Oussoren | b6ee4f5 | 2010-07-23 13:53:51 +0000 | [diff] [blame] | 1068 | |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1069 | # tests for the posix *at functions follow |
| 1070 | |
Larry Hastings | 9cf065c | 2012-06-22 16:30:09 -0700 | [diff] [blame] | 1071 | @unittest.skipUnless(os.access in os.supports_dir_fd, "test needs dir_fd support for os.access()") |
| 1072 | def test_access_dir_fd(self): |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1073 | f = posix.open(posix.getcwd(), posix.O_RDONLY) |
| 1074 | try: |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1075 | self.assertTrue(posix.access(os_helper.TESTFN, os.R_OK, dir_fd=f)) |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1076 | finally: |
| 1077 | posix.close(f) |
| 1078 | |
Larry Hastings | 9cf065c | 2012-06-22 16:30:09 -0700 | [diff] [blame] | 1079 | @unittest.skipUnless(os.chmod in os.supports_dir_fd, "test needs dir_fd support in os.chmod()") |
| 1080 | def test_chmod_dir_fd(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1081 | os.chmod(os_helper.TESTFN, stat.S_IRUSR) |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1082 | |
| 1083 | f = posix.open(posix.getcwd(), posix.O_RDONLY) |
| 1084 | try: |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1085 | posix.chmod(os_helper.TESTFN, stat.S_IRUSR | stat.S_IWUSR, dir_fd=f) |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1086 | |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1087 | s = posix.stat(os_helper.TESTFN) |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1088 | self.assertEqual(s[0] & stat.S_IRWXU, stat.S_IRUSR | stat.S_IWUSR) |
| 1089 | finally: |
| 1090 | posix.close(f) |
| 1091 | |
Larry Hastings | 9cf065c | 2012-06-22 16:30:09 -0700 | [diff] [blame] | 1092 | @unittest.skipUnless(os.chown in os.supports_dir_fd, "test needs dir_fd support in os.chown()") |
| 1093 | def test_chown_dir_fd(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1094 | os_helper.unlink(os_helper.TESTFN) |
| 1095 | os_helper.create_empty_file(os_helper.TESTFN) |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1096 | |
| 1097 | f = posix.open(posix.getcwd(), posix.O_RDONLY) |
| 1098 | try: |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1099 | posix.chown(os_helper.TESTFN, os.getuid(), os.getgid(), dir_fd=f) |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1100 | finally: |
| 1101 | posix.close(f) |
| 1102 | |
Larry Hastings | 9cf065c | 2012-06-22 16:30:09 -0700 | [diff] [blame] | 1103 | @unittest.skipUnless(os.stat in os.supports_dir_fd, "test needs dir_fd support in os.stat()") |
| 1104 | def test_stat_dir_fd(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1105 | os_helper.unlink(os_helper.TESTFN) |
| 1106 | with open(os_helper.TESTFN, 'w') as outfile: |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1107 | outfile.write("testline\n") |
| 1108 | |
| 1109 | f = posix.open(posix.getcwd(), posix.O_RDONLY) |
| 1110 | try: |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1111 | s1 = posix.stat(os_helper.TESTFN) |
| 1112 | s2 = posix.stat(os_helper.TESTFN, dir_fd=f) |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1113 | self.assertEqual(s1, s2) |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1114 | s2 = posix.stat(os_helper.TESTFN, dir_fd=None) |
Serhiy Storchaka | a2ad5c3 | 2013-01-07 23:13:46 +0200 | [diff] [blame] | 1115 | self.assertEqual(s1, s2) |
Serhiy Storchaka | 7155b88 | 2016-04-08 08:48:20 +0300 | [diff] [blame] | 1116 | self.assertRaisesRegex(TypeError, 'should be integer or None, not', |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1117 | posix.stat, os_helper.TESTFN, dir_fd=posix.getcwd()) |
Serhiy Storchaka | 7155b88 | 2016-04-08 08:48:20 +0300 | [diff] [blame] | 1118 | self.assertRaisesRegex(TypeError, 'should be integer or None, not', |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1119 | posix.stat, os_helper.TESTFN, dir_fd=float(f)) |
Serhiy Storchaka | a2ad5c3 | 2013-01-07 23:13:46 +0200 | [diff] [blame] | 1120 | self.assertRaises(OverflowError, |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1121 | posix.stat, os_helper.TESTFN, dir_fd=10**20) |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1122 | finally: |
| 1123 | posix.close(f) |
| 1124 | |
Larry Hastings | 9cf065c | 2012-06-22 16:30:09 -0700 | [diff] [blame] | 1125 | @unittest.skipUnless(os.utime in os.supports_dir_fd, "test needs dir_fd support in os.utime()") |
| 1126 | def test_utime_dir_fd(self): |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1127 | f = posix.open(posix.getcwd(), posix.O_RDONLY) |
| 1128 | try: |
| 1129 | now = time.time() |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1130 | posix.utime(os_helper.TESTFN, None, dir_fd=f) |
| 1131 | posix.utime(os_helper.TESTFN, dir_fd=f) |
| 1132 | self.assertRaises(TypeError, posix.utime, os_helper.TESTFN, |
| 1133 | now, dir_fd=f) |
| 1134 | self.assertRaises(TypeError, posix.utime, os_helper.TESTFN, |
| 1135 | (None, None), dir_fd=f) |
| 1136 | self.assertRaises(TypeError, posix.utime, os_helper.TESTFN, |
| 1137 | (now, None), dir_fd=f) |
| 1138 | self.assertRaises(TypeError, posix.utime, os_helper.TESTFN, |
| 1139 | (None, now), dir_fd=f) |
| 1140 | self.assertRaises(TypeError, posix.utime, os_helper.TESTFN, |
| 1141 | (now, "x"), dir_fd=f) |
| 1142 | posix.utime(os_helper.TESTFN, (int(now), int(now)), dir_fd=f) |
| 1143 | posix.utime(os_helper.TESTFN, (now, now), dir_fd=f) |
| 1144 | posix.utime(os_helper.TESTFN, |
Larry Hastings | 9cf065c | 2012-06-22 16:30:09 -0700 | [diff] [blame] | 1145 | (int(now), int((now - int(now)) * 1e9)), dir_fd=f) |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1146 | posix.utime(os_helper.TESTFN, dir_fd=f, |
Larry Hastings | 9cf065c | 2012-06-22 16:30:09 -0700 | [diff] [blame] | 1147 | times=(int(now), int((now - int(now)) * 1e9))) |
| 1148 | |
Larry Hastings | 90867a5 | 2012-06-22 17:01:41 -0700 | [diff] [blame] | 1149 | # try dir_fd and follow_symlinks together |
Larry Hastings | 9cf065c | 2012-06-22 16:30:09 -0700 | [diff] [blame] | 1150 | if os.utime in os.supports_follow_symlinks: |
Larry Hastings | 90867a5 | 2012-06-22 17:01:41 -0700 | [diff] [blame] | 1151 | try: |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1152 | posix.utime(os_helper.TESTFN, follow_symlinks=False, |
| 1153 | dir_fd=f) |
Georg Brandl | 969288e | 2012-06-26 09:25:44 +0200 | [diff] [blame] | 1154 | except ValueError: |
Larry Hastings | 90867a5 | 2012-06-22 17:01:41 -0700 | [diff] [blame] | 1155 | # whoops! using both together not supported on this platform. |
| 1156 | pass |
Larry Hastings | 9cf065c | 2012-06-22 16:30:09 -0700 | [diff] [blame] | 1157 | |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1158 | finally: |
| 1159 | posix.close(f) |
| 1160 | |
Larry Hastings | 9cf065c | 2012-06-22 16:30:09 -0700 | [diff] [blame] | 1161 | @unittest.skipUnless(os.link in os.supports_dir_fd, "test needs dir_fd support in os.link()") |
| 1162 | def test_link_dir_fd(self): |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1163 | f = posix.open(posix.getcwd(), posix.O_RDONLY) |
| 1164 | try: |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1165 | posix.link(os_helper.TESTFN, os_helper.TESTFN + 'link', |
| 1166 | src_dir_fd=f, dst_dir_fd=f) |
xdegaye | 92c2ca7 | 2017-11-12 17:31:07 +0100 | [diff] [blame] | 1167 | except PermissionError as e: |
| 1168 | self.skipTest('posix.link(): %s' % e) |
| 1169 | else: |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1170 | # should have same inodes |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1171 | self.assertEqual(posix.stat(os_helper.TESTFN)[1], |
| 1172 | posix.stat(os_helper.TESTFN + 'link')[1]) |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1173 | finally: |
| 1174 | posix.close(f) |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1175 | os_helper.unlink(os_helper.TESTFN + 'link') |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1176 | |
Larry Hastings | 9cf065c | 2012-06-22 16:30:09 -0700 | [diff] [blame] | 1177 | @unittest.skipUnless(os.mkdir in os.supports_dir_fd, "test needs dir_fd support in os.mkdir()") |
| 1178 | def test_mkdir_dir_fd(self): |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1179 | f = posix.open(posix.getcwd(), posix.O_RDONLY) |
| 1180 | try: |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1181 | posix.mkdir(os_helper.TESTFN + 'dir', dir_fd=f) |
| 1182 | posix.stat(os_helper.TESTFN + 'dir') # should not raise exception |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1183 | finally: |
| 1184 | posix.close(f) |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1185 | os_helper.rmtree(os_helper.TESTFN + 'dir') |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1186 | |
Larry Hastings | 9cf065c | 2012-06-22 16:30:09 -0700 | [diff] [blame] | 1187 | @unittest.skipUnless((os.mknod in os.supports_dir_fd) and hasattr(stat, 'S_IFIFO'), |
| 1188 | "test requires both stat.S_IFIFO and dir_fd support for os.mknod()") |
| 1189 | def test_mknod_dir_fd(self): |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1190 | # Test using mknodat() to create a FIFO (the only use specified |
| 1191 | # by POSIX). |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1192 | os_helper.unlink(os_helper.TESTFN) |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1193 | mode = stat.S_IFIFO | stat.S_IRUSR | stat.S_IWUSR |
| 1194 | f = posix.open(posix.getcwd(), posix.O_RDONLY) |
| 1195 | try: |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1196 | posix.mknod(os_helper.TESTFN, mode, 0, dir_fd=f) |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1197 | except OSError as e: |
| 1198 | # Some old systems don't allow unprivileged users to use |
| 1199 | # mknod(), or only support creating device nodes. |
xdegaye | 92c2ca7 | 2017-11-12 17:31:07 +0100 | [diff] [blame] | 1200 | self.assertIn(e.errno, (errno.EPERM, errno.EINVAL, errno.EACCES)) |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1201 | else: |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1202 | self.assertTrue(stat.S_ISFIFO(posix.stat(os_helper.TESTFN).st_mode)) |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1203 | finally: |
| 1204 | posix.close(f) |
| 1205 | |
Larry Hastings | 9cf065c | 2012-06-22 16:30:09 -0700 | [diff] [blame] | 1206 | @unittest.skipUnless(os.open in os.supports_dir_fd, "test needs dir_fd support in os.open()") |
| 1207 | def test_open_dir_fd(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1208 | os_helper.unlink(os_helper.TESTFN) |
| 1209 | with open(os_helper.TESTFN, 'w') as outfile: |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1210 | outfile.write("testline\n") |
| 1211 | a = posix.open(posix.getcwd(), posix.O_RDONLY) |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1212 | b = posix.open(os_helper.TESTFN, posix.O_RDONLY, dir_fd=a) |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1213 | try: |
| 1214 | res = posix.read(b, 9).decode(encoding="utf-8") |
| 1215 | self.assertEqual("testline\n", res) |
| 1216 | finally: |
| 1217 | posix.close(a) |
| 1218 | posix.close(b) |
| 1219 | |
Larry Hastings | 9cf065c | 2012-06-22 16:30:09 -0700 | [diff] [blame] | 1220 | @unittest.skipUnless(os.readlink in os.supports_dir_fd, "test needs dir_fd support in os.readlink()") |
| 1221 | def test_readlink_dir_fd(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1222 | os.symlink(os_helper.TESTFN, os_helper.TESTFN + 'link') |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1223 | f = posix.open(posix.getcwd(), posix.O_RDONLY) |
| 1224 | try: |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1225 | self.assertEqual(posix.readlink(os_helper.TESTFN + 'link'), |
| 1226 | posix.readlink(os_helper.TESTFN + 'link', dir_fd=f)) |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1227 | finally: |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1228 | os_helper.unlink(os_helper.TESTFN + 'link') |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1229 | posix.close(f) |
| 1230 | |
Larry Hastings | 9cf065c | 2012-06-22 16:30:09 -0700 | [diff] [blame] | 1231 | @unittest.skipUnless(os.rename in os.supports_dir_fd, "test needs dir_fd support in os.rename()") |
| 1232 | def test_rename_dir_fd(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1233 | os_helper.unlink(os_helper.TESTFN) |
| 1234 | os_helper.create_empty_file(os_helper.TESTFN + 'ren') |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1235 | f = posix.open(posix.getcwd(), posix.O_RDONLY) |
| 1236 | try: |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1237 | posix.rename(os_helper.TESTFN + 'ren', os_helper.TESTFN, src_dir_fd=f, dst_dir_fd=f) |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1238 | except: |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1239 | posix.rename(os_helper.TESTFN + 'ren', os_helper.TESTFN) |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1240 | raise |
| 1241 | else: |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1242 | posix.stat(os_helper.TESTFN) # should not raise exception |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1243 | finally: |
| 1244 | posix.close(f) |
| 1245 | |
Dong-hee Na | 2eba6ad | 2019-10-21 16:01:05 +0900 | [diff] [blame] | 1246 | @unittest.skipUnless(hasattr(signal, 'SIGCHLD'), 'CLD_XXXX be placed in si_code for a SIGCHLD signal') |
| 1247 | @unittest.skipUnless(hasattr(os, 'waitid_result'), "test needs os.waitid_result") |
| 1248 | def test_cld_xxxx_constants(self): |
| 1249 | os.CLD_EXITED |
| 1250 | os.CLD_KILLED |
| 1251 | os.CLD_DUMPED |
| 1252 | os.CLD_TRAPPED |
| 1253 | os.CLD_STOPPED |
| 1254 | os.CLD_CONTINUED |
| 1255 | |
Larry Hastings | 9cf065c | 2012-06-22 16:30:09 -0700 | [diff] [blame] | 1256 | @unittest.skipUnless(os.symlink in os.supports_dir_fd, "test needs dir_fd support in os.symlink()") |
| 1257 | def test_symlink_dir_fd(self): |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1258 | f = posix.open(posix.getcwd(), posix.O_RDONLY) |
| 1259 | try: |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1260 | posix.symlink(os_helper.TESTFN, os_helper.TESTFN + 'link', |
| 1261 | dir_fd=f) |
| 1262 | self.assertEqual(posix.readlink(os_helper.TESTFN + 'link'), |
| 1263 | os_helper.TESTFN) |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1264 | finally: |
| 1265 | posix.close(f) |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1266 | os_helper.unlink(os_helper.TESTFN + 'link') |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1267 | |
Larry Hastings | 9cf065c | 2012-06-22 16:30:09 -0700 | [diff] [blame] | 1268 | @unittest.skipUnless(os.unlink in os.supports_dir_fd, "test needs dir_fd support in os.unlink()") |
| 1269 | def test_unlink_dir_fd(self): |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1270 | f = posix.open(posix.getcwd(), posix.O_RDONLY) |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1271 | os_helper.create_empty_file(os_helper.TESTFN + 'del') |
| 1272 | posix.stat(os_helper.TESTFN + 'del') # should not raise exception |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1273 | try: |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1274 | posix.unlink(os_helper.TESTFN + 'del', dir_fd=f) |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1275 | except: |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1276 | os_helper.unlink(os_helper.TESTFN + 'del') |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1277 | raise |
| 1278 | else: |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1279 | self.assertRaises(OSError, posix.stat, os_helper.TESTFN + 'link') |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1280 | finally: |
| 1281 | posix.close(f) |
| 1282 | |
Larry Hastings | 9cf065c | 2012-06-22 16:30:09 -0700 | [diff] [blame] | 1283 | @unittest.skipUnless(os.mkfifo in os.supports_dir_fd, "test needs dir_fd support in os.mkfifo()") |
| 1284 | def test_mkfifo_dir_fd(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1285 | os_helper.unlink(os_helper.TESTFN) |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1286 | f = posix.open(posix.getcwd(), posix.O_RDONLY) |
| 1287 | try: |
xdegaye | 92c2ca7 | 2017-11-12 17:31:07 +0100 | [diff] [blame] | 1288 | try: |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1289 | posix.mkfifo(os_helper.TESTFN, |
xdegaye | 92c2ca7 | 2017-11-12 17:31:07 +0100 | [diff] [blame] | 1290 | stat.S_IRUSR | stat.S_IWUSR, dir_fd=f) |
| 1291 | except PermissionError as e: |
| 1292 | self.skipTest('posix.mkfifo(): %s' % e) |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1293 | self.assertTrue(stat.S_ISFIFO(posix.stat(os_helper.TESTFN).st_mode)) |
Antoine Pitrou | f65132d | 2011-02-25 23:25:17 +0000 | [diff] [blame] | 1294 | finally: |
| 1295 | posix.close(f) |
| 1296 | |
Benjamin Peterson | 94b580d | 2011-08-02 17:30:04 -0500 | [diff] [blame] | 1297 | requires_sched_h = unittest.skipUnless(hasattr(posix, 'sched_yield'), |
| 1298 | "don't have scheduling support") |
Antoine Pitrou | 8486987 | 2012-08-04 16:16:35 +0200 | [diff] [blame] | 1299 | requires_sched_affinity = unittest.skipUnless(hasattr(posix, 'sched_setaffinity'), |
Benjamin Peterson | 50ba271 | 2011-08-02 22:15:40 -0500 | [diff] [blame] | 1300 | "don't have sched affinity support") |
Benjamin Peterson | 94b580d | 2011-08-02 17:30:04 -0500 | [diff] [blame] | 1301 | |
| 1302 | @requires_sched_h |
| 1303 | def test_sched_yield(self): |
| 1304 | # This has no error conditions (at least on Linux). |
| 1305 | posix.sched_yield() |
| 1306 | |
| 1307 | @requires_sched_h |
Charles-François Natali | ea0d5fc | 2011-09-06 19:03:35 +0200 | [diff] [blame] | 1308 | @unittest.skipUnless(hasattr(posix, 'sched_get_priority_max'), |
| 1309 | "requires sched_get_priority_max()") |
Benjamin Peterson | 94b580d | 2011-08-02 17:30:04 -0500 | [diff] [blame] | 1310 | def test_sched_priority(self): |
| 1311 | # Round-robin usually has interesting priorities. |
| 1312 | pol = posix.SCHED_RR |
| 1313 | lo = posix.sched_get_priority_min(pol) |
| 1314 | hi = posix.sched_get_priority_max(pol) |
| 1315 | self.assertIsInstance(lo, int) |
| 1316 | self.assertIsInstance(hi, int) |
| 1317 | self.assertGreaterEqual(hi, lo) |
Benjamin Peterson | 539b6c4 | 2011-08-02 22:09:37 -0500 | [diff] [blame] | 1318 | # OSX evidently just returns 15 without checking the argument. |
| 1319 | if sys.platform != "darwin": |
Benjamin Peterson | c158158 | 2011-08-02 22:10:55 -0500 | [diff] [blame] | 1320 | self.assertRaises(OSError, posix.sched_get_priority_min, -23) |
| 1321 | self.assertRaises(OSError, posix.sched_get_priority_max, -23) |
Benjamin Peterson | 94b580d | 2011-08-02 17:30:04 -0500 | [diff] [blame] | 1322 | |
Benjamin Peterson | c704222 | 2018-09-12 15:12:24 -0700 | [diff] [blame] | 1323 | @requires_sched |
Benjamin Peterson | 94b580d | 2011-08-02 17:30:04 -0500 | [diff] [blame] | 1324 | def test_get_and_set_scheduler_and_param(self): |
| 1325 | possible_schedulers = [sched for name, sched in posix.__dict__.items() |
| 1326 | if name.startswith("SCHED_")] |
| 1327 | mine = posix.sched_getscheduler(0) |
| 1328 | self.assertIn(mine, possible_schedulers) |
| 1329 | try: |
Jesus Cea | ceb5d16 | 2011-09-10 01:16:55 +0200 | [diff] [blame] | 1330 | parent = posix.sched_getscheduler(os.getppid()) |
Benjamin Peterson | 94b580d | 2011-08-02 17:30:04 -0500 | [diff] [blame] | 1331 | except OSError as e: |
Jesus Cea | ceb5d16 | 2011-09-10 01:16:55 +0200 | [diff] [blame] | 1332 | if e.errno != errno.EPERM: |
Benjamin Peterson | 94b580d | 2011-08-02 17:30:04 -0500 | [diff] [blame] | 1333 | raise |
| 1334 | else: |
Jesus Cea | ceb5d16 | 2011-09-10 01:16:55 +0200 | [diff] [blame] | 1335 | self.assertIn(parent, possible_schedulers) |
Benjamin Peterson | 94b580d | 2011-08-02 17:30:04 -0500 | [diff] [blame] | 1336 | self.assertRaises(OSError, posix.sched_getscheduler, -1) |
| 1337 | self.assertRaises(OSError, posix.sched_getparam, -1) |
| 1338 | param = posix.sched_getparam(0) |
| 1339 | self.assertIsInstance(param.sched_priority, int) |
Charles-François Natali | 7b911cb | 2011-08-21 12:41:43 +0200 | [diff] [blame] | 1340 | |
Charles-François Natali | b402a5c | 2013-01-13 14:13:25 +0100 | [diff] [blame] | 1341 | # POSIX states that calling sched_setparam() or sched_setscheduler() on |
| 1342 | # a process with a scheduling policy other than SCHED_FIFO or SCHED_RR |
| 1343 | # is implementation-defined: NetBSD and FreeBSD can return EINVAL. |
| 1344 | if not sys.platform.startswith(('freebsd', 'netbsd')): |
| 1345 | try: |
| 1346 | posix.sched_setscheduler(0, mine, param) |
| 1347 | posix.sched_setparam(0, param) |
| 1348 | except OSError as e: |
| 1349 | if e.errno != errno.EPERM: |
| 1350 | raise |
Charles-François Natali | 7b911cb | 2011-08-21 12:41:43 +0200 | [diff] [blame] | 1351 | self.assertRaises(OSError, posix.sched_setparam, -1, param) |
| 1352 | |
Benjamin Peterson | 94b580d | 2011-08-02 17:30:04 -0500 | [diff] [blame] | 1353 | self.assertRaises(OSError, posix.sched_setscheduler, -1, mine, param) |
| 1354 | self.assertRaises(TypeError, posix.sched_setscheduler, 0, mine, None) |
| 1355 | self.assertRaises(TypeError, posix.sched_setparam, 0, 43) |
| 1356 | param = posix.sched_param(None) |
| 1357 | self.assertRaises(TypeError, posix.sched_setparam, 0, param) |
| 1358 | large = 214748364700 |
| 1359 | param = posix.sched_param(large) |
| 1360 | self.assertRaises(OverflowError, posix.sched_setparam, 0, param) |
| 1361 | param = posix.sched_param(sched_priority=-large) |
| 1362 | self.assertRaises(OverflowError, posix.sched_setparam, 0, param) |
| 1363 | |
Benjamin Peterson | c5fce4d | 2011-08-02 18:07:32 -0500 | [diff] [blame] | 1364 | @unittest.skipUnless(hasattr(posix, "sched_rr_get_interval"), "no function") |
Benjamin Peterson | 94b580d | 2011-08-02 17:30:04 -0500 | [diff] [blame] | 1365 | def test_sched_rr_get_interval(self): |
Benjamin Peterson | 43234ab | 2011-08-02 22:19:14 -0500 | [diff] [blame] | 1366 | try: |
| 1367 | interval = posix.sched_rr_get_interval(0) |
| 1368 | except OSError as e: |
| 1369 | # This likely means that sched_rr_get_interval is only valid for |
| 1370 | # processes with the SCHED_RR scheduler in effect. |
| 1371 | if e.errno != errno.EINVAL: |
| 1372 | raise |
| 1373 | self.skipTest("only works on SCHED_RR processes") |
Benjamin Peterson | 94b580d | 2011-08-02 17:30:04 -0500 | [diff] [blame] | 1374 | self.assertIsInstance(interval, float) |
| 1375 | # Reasonable constraints, I think. |
| 1376 | self.assertGreaterEqual(interval, 0.) |
| 1377 | self.assertLess(interval, 1.) |
| 1378 | |
Benjamin Peterson | 2740af8 | 2011-08-02 17:41:34 -0500 | [diff] [blame] | 1379 | @requires_sched_affinity |
Antoine Pitrou | 8486987 | 2012-08-04 16:16:35 +0200 | [diff] [blame] | 1380 | def test_sched_getaffinity(self): |
| 1381 | mask = posix.sched_getaffinity(0) |
| 1382 | self.assertIsInstance(mask, set) |
| 1383 | self.assertGreaterEqual(len(mask), 1) |
| 1384 | self.assertRaises(OSError, posix.sched_getaffinity, -1) |
| 1385 | for cpu in mask: |
| 1386 | self.assertIsInstance(cpu, int) |
| 1387 | self.assertGreaterEqual(cpu, 0) |
| 1388 | self.assertLess(cpu, 1 << 32) |
| 1389 | |
| 1390 | @requires_sched_affinity |
| 1391 | def test_sched_setaffinity(self): |
| 1392 | mask = posix.sched_getaffinity(0) |
| 1393 | if len(mask) > 1: |
| 1394 | # Empty masks are forbidden |
| 1395 | mask.pop() |
Benjamin Peterson | 94b580d | 2011-08-02 17:30:04 -0500 | [diff] [blame] | 1396 | posix.sched_setaffinity(0, mask) |
Antoine Pitrou | 8486987 | 2012-08-04 16:16:35 +0200 | [diff] [blame] | 1397 | self.assertEqual(posix.sched_getaffinity(0), mask) |
| 1398 | self.assertRaises(OSError, posix.sched_setaffinity, 0, []) |
| 1399 | self.assertRaises(ValueError, posix.sched_setaffinity, 0, [-10]) |
Brandt Bucher | 45a30af | 2019-06-27 09:10:57 -0700 | [diff] [blame] | 1400 | self.assertRaises(ValueError, posix.sched_setaffinity, 0, map(int, "0X")) |
Antoine Pitrou | 8486987 | 2012-08-04 16:16:35 +0200 | [diff] [blame] | 1401 | self.assertRaises(OverflowError, posix.sched_setaffinity, 0, [1<<128]) |
Benjamin Peterson | 94b580d | 2011-08-02 17:30:04 -0500 | [diff] [blame] | 1402 | self.assertRaises(OSError, posix.sched_setaffinity, -1, mask) |
| 1403 | |
Victor Stinner | 8b905bd | 2011-10-25 13:34:04 +0200 | [diff] [blame] | 1404 | def test_rtld_constants(self): |
| 1405 | # check presence of major RTLD_* constants |
| 1406 | posix.RTLD_LAZY |
| 1407 | posix.RTLD_NOW |
| 1408 | posix.RTLD_GLOBAL |
| 1409 | posix.RTLD_LOCAL |
| 1410 | |
Jesus Cea | 60c13dd | 2012-06-23 02:58:14 +0200 | [diff] [blame] | 1411 | @unittest.skipUnless(hasattr(os, 'SEEK_HOLE'), |
| 1412 | "test needs an OS that reports file holes") |
Hynek Schlawack | f841e42 | 2012-06-24 09:51:46 +0200 | [diff] [blame] | 1413 | def test_fs_holes(self): |
Jesus Cea | 9436361 | 2012-06-22 18:32:07 +0200 | [diff] [blame] | 1414 | # Even if the filesystem doesn't report holes, |
| 1415 | # if the OS supports it the SEEK_* constants |
| 1416 | # will be defined and will have a consistent |
| 1417 | # behaviour: |
| 1418 | # os.SEEK_DATA = current position |
| 1419 | # os.SEEK_HOLE = end of file position |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1420 | with open(os_helper.TESTFN, 'r+b') as fp: |
Jesus Cea | 9436361 | 2012-06-22 18:32:07 +0200 | [diff] [blame] | 1421 | fp.write(b"hello") |
| 1422 | fp.flush() |
| 1423 | size = fp.tell() |
| 1424 | fno = fp.fileno() |
Jesus Cea | d46f7d2 | 2012-07-07 14:56:04 +0200 | [diff] [blame] | 1425 | try : |
| 1426 | for i in range(size): |
| 1427 | self.assertEqual(i, os.lseek(fno, i, os.SEEK_DATA)) |
| 1428 | self.assertLessEqual(size, os.lseek(fno, i, os.SEEK_HOLE)) |
| 1429 | self.assertRaises(OSError, os.lseek, fno, size, os.SEEK_DATA) |
| 1430 | self.assertRaises(OSError, os.lseek, fno, size, os.SEEK_HOLE) |
| 1431 | except OSError : |
| 1432 | # Some OSs claim to support SEEK_HOLE/SEEK_DATA |
| 1433 | # but it is not true. |
| 1434 | # For instance: |
| 1435 | # http://lists.freebsd.org/pipermail/freebsd-amd64/2012-January/014332.html |
| 1436 | raise unittest.SkipTest("OSError raised!") |
Jesus Cea | 9436361 | 2012-06-22 18:32:07 +0200 | [diff] [blame] | 1437 | |
Larry Hastings | b082731 | 2014-02-09 22:05:19 -0800 | [diff] [blame] | 1438 | def test_path_error2(self): |
| 1439 | """ |
| 1440 | Test functions that call path_error2(), providing two filenames in their exceptions. |
| 1441 | """ |
Victor Stinner | 047b7ae | 2014-10-05 17:37:41 +0200 | [diff] [blame] | 1442 | for name in ("rename", "replace", "link"): |
Larry Hastings | b082731 | 2014-02-09 22:05:19 -0800 | [diff] [blame] | 1443 | function = getattr(os, name, None) |
Victor Stinner | bed04a7 | 2014-10-05 17:37:59 +0200 | [diff] [blame] | 1444 | if function is None: |
| 1445 | continue |
Larry Hastings | b082731 | 2014-02-09 22:05:19 -0800 | [diff] [blame] | 1446 | |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1447 | for dst in ("noodly2", os_helper.TESTFN): |
Victor Stinner | bed04a7 | 2014-10-05 17:37:59 +0200 | [diff] [blame] | 1448 | try: |
| 1449 | function('doesnotexistfilename', dst) |
| 1450 | except OSError as e: |
| 1451 | self.assertIn("'doesnotexistfilename' -> '{}'".format(dst), str(e)) |
| 1452 | break |
| 1453 | else: |
| 1454 | self.fail("No valid path_error2() test for os." + name) |
Larry Hastings | b082731 | 2014-02-09 22:05:19 -0800 | [diff] [blame] | 1455 | |
Serhiy Storchaka | 2b0d200 | 2015-04-20 09:53:58 +0300 | [diff] [blame] | 1456 | def test_path_with_null_character(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1457 | fn = os_helper.TESTFN |
Serhiy Storchaka | 2b0d200 | 2015-04-20 09:53:58 +0300 | [diff] [blame] | 1458 | fn_with_NUL = fn + '\0' |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1459 | self.addCleanup(os_helper.unlink, fn) |
| 1460 | os_helper.unlink(fn) |
Serhiy Storchaka | 2b0d200 | 2015-04-20 09:53:58 +0300 | [diff] [blame] | 1461 | fd = None |
| 1462 | try: |
Serhiy Storchaka | 7e9d1d1 | 2015-04-20 10:12:28 +0300 | [diff] [blame] | 1463 | with self.assertRaises(ValueError): |
Serhiy Storchaka | 2b0d200 | 2015-04-20 09:53:58 +0300 | [diff] [blame] | 1464 | fd = os.open(fn_with_NUL, os.O_WRONLY | os.O_CREAT) # raises |
| 1465 | finally: |
| 1466 | if fd is not None: |
| 1467 | os.close(fd) |
| 1468 | self.assertFalse(os.path.exists(fn)) |
Serhiy Storchaka | 7e9d1d1 | 2015-04-20 10:12:28 +0300 | [diff] [blame] | 1469 | self.assertRaises(ValueError, os.mkdir, fn_with_NUL) |
Serhiy Storchaka | 2b0d200 | 2015-04-20 09:53:58 +0300 | [diff] [blame] | 1470 | self.assertFalse(os.path.exists(fn)) |
| 1471 | open(fn, 'wb').close() |
Serhiy Storchaka | 7e9d1d1 | 2015-04-20 10:12:28 +0300 | [diff] [blame] | 1472 | self.assertRaises(ValueError, os.stat, fn_with_NUL) |
Serhiy Storchaka | 2b0d200 | 2015-04-20 09:53:58 +0300 | [diff] [blame] | 1473 | |
| 1474 | def test_path_with_null_byte(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1475 | fn = os.fsencode(os_helper.TESTFN) |
Serhiy Storchaka | 2b0d200 | 2015-04-20 09:53:58 +0300 | [diff] [blame] | 1476 | fn_with_NUL = fn + b'\0' |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1477 | self.addCleanup(os_helper.unlink, fn) |
| 1478 | os_helper.unlink(fn) |
Serhiy Storchaka | 2b0d200 | 2015-04-20 09:53:58 +0300 | [diff] [blame] | 1479 | fd = None |
| 1480 | try: |
| 1481 | with self.assertRaises(ValueError): |
| 1482 | fd = os.open(fn_with_NUL, os.O_WRONLY | os.O_CREAT) # raises |
| 1483 | finally: |
| 1484 | if fd is not None: |
| 1485 | os.close(fd) |
| 1486 | self.assertFalse(os.path.exists(fn)) |
| 1487 | self.assertRaises(ValueError, os.mkdir, fn_with_NUL) |
| 1488 | self.assertFalse(os.path.exists(fn)) |
| 1489 | open(fn, 'wb').close() |
| 1490 | self.assertRaises(ValueError, os.stat, fn_with_NUL) |
| 1491 | |
Benjamin Peterson | 6c4c45e | 2019-11-05 19:21:29 -0800 | [diff] [blame] | 1492 | @unittest.skipUnless(hasattr(os, "pidfd_open"), "pidfd_open unavailable") |
| 1493 | def test_pidfd_open(self): |
| 1494 | with self.assertRaises(OSError) as cm: |
| 1495 | os.pidfd_open(-1) |
| 1496 | if cm.exception.errno == errno.ENOSYS: |
| 1497 | self.skipTest("system does not support pidfd_open") |
Victor Stinner | 3ab479a | 2019-11-21 12:54:54 +0100 | [diff] [blame] | 1498 | if isinstance(cm.exception, PermissionError): |
| 1499 | self.skipTest(f"pidfd_open syscall blocked: {cm.exception!r}") |
Benjamin Peterson | 6c4c45e | 2019-11-05 19:21:29 -0800 | [diff] [blame] | 1500 | self.assertEqual(cm.exception.errno, errno.EINVAL) |
| 1501 | os.close(os.pidfd_open(os.getpid(), 0)) |
| 1502 | |
Ronald Oussoren | b6ee4f5 | 2010-07-23 13:53:51 +0000 | [diff] [blame] | 1503 | class PosixGroupsTester(unittest.TestCase): |
| 1504 | |
| 1505 | def setUp(self): |
| 1506 | if posix.getuid() != 0: |
| 1507 | raise unittest.SkipTest("not enough privileges") |
| 1508 | if not hasattr(posix, 'getgroups'): |
| 1509 | raise unittest.SkipTest("need posix.getgroups") |
| 1510 | if sys.platform == 'darwin': |
| 1511 | raise unittest.SkipTest("getgroups(2) is broken on OSX") |
| 1512 | self.saved_groups = posix.getgroups() |
| 1513 | |
| 1514 | def tearDown(self): |
| 1515 | if hasattr(posix, 'setgroups'): |
| 1516 | posix.setgroups(self.saved_groups) |
| 1517 | elif hasattr(posix, 'initgroups'): |
| 1518 | name = pwd.getpwuid(posix.getuid()).pw_name |
| 1519 | posix.initgroups(name, self.saved_groups[0]) |
| 1520 | |
| 1521 | @unittest.skipUnless(hasattr(posix, 'initgroups'), |
| 1522 | "test needs posix.initgroups()") |
| 1523 | def test_initgroups(self): |
| 1524 | # find missing group |
| 1525 | |
Benjamin Peterson | 659a6f5 | 2014-03-01 19:14:12 -0500 | [diff] [blame] | 1526 | g = max(self.saved_groups or [0]) + 1 |
Ronald Oussoren | b6ee4f5 | 2010-07-23 13:53:51 +0000 | [diff] [blame] | 1527 | name = pwd.getpwuid(posix.getuid()).pw_name |
| 1528 | posix.initgroups(name, g) |
| 1529 | self.assertIn(g, posix.getgroups()) |
| 1530 | |
| 1531 | @unittest.skipUnless(hasattr(posix, 'setgroups'), |
| 1532 | "test needs posix.setgroups()") |
| 1533 | def test_setgroups(self): |
Antoine Pitrou | e5a9101 | 2010-09-04 17:32:06 +0000 | [diff] [blame] | 1534 | for groups in [[0], list(range(16))]: |
Ronald Oussoren | b6ee4f5 | 2010-07-23 13:53:51 +0000 | [diff] [blame] | 1535 | posix.setgroups(groups) |
| 1536 | self.assertListEqual(groups, posix.getgroups()) |
| 1537 | |
Serhiy Storchaka | ef34753 | 2018-05-01 16:45:04 +0300 | [diff] [blame] | 1538 | |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1539 | class _PosixSpawnMixin: |
| 1540 | # Program which does nothing and exits with status 0 (success) |
Victor Stinner | 0382406 | 2018-08-30 01:21:11 +0200 | [diff] [blame] | 1541 | NOOP_PROGRAM = (sys.executable, '-I', '-S', '-c', 'pass') |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1542 | spawn_func = None |
Victor Stinner | 0382406 | 2018-08-30 01:21:11 +0200 | [diff] [blame] | 1543 | |
| 1544 | def python_args(self, *args): |
| 1545 | # Disable site module to avoid side effects. For example, |
| 1546 | # on Fedora 28, if the HOME environment variable is not set, |
| 1547 | # site._getuserbase() calls pwd.getpwuid() which opens |
| 1548 | # /var/lib/sss/mc/passwd but then leaves the file open which makes |
| 1549 | # test_close_file() to fail. |
| 1550 | return (sys.executable, '-I', '-S', *args) |
| 1551 | |
Serhiy Storchaka | ef34753 | 2018-05-01 16:45:04 +0300 | [diff] [blame] | 1552 | def test_returns_pid(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1553 | pidfile = os_helper.TESTFN |
| 1554 | self.addCleanup(os_helper.unlink, pidfile) |
Serhiy Storchaka | ef34753 | 2018-05-01 16:45:04 +0300 | [diff] [blame] | 1555 | script = f"""if 1: |
| 1556 | import os |
| 1557 | with open({pidfile!r}, "w") as pidfile: |
| 1558 | pidfile.write(str(os.getpid())) |
| 1559 | """ |
Victor Stinner | 0382406 | 2018-08-30 01:21:11 +0200 | [diff] [blame] | 1560 | args = self.python_args('-c', script) |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1561 | pid = self.spawn_func(args[0], args, os.environ) |
Victor Stinner | 278c1e1 | 2020-03-31 20:08:12 +0200 | [diff] [blame] | 1562 | support.wait_process(pid, exitcode=0) |
Serhiy Storchaka | ef34753 | 2018-05-01 16:45:04 +0300 | [diff] [blame] | 1563 | with open(pidfile) as f: |
| 1564 | self.assertEqual(f.read(), str(pid)) |
| 1565 | |
| 1566 | def test_no_such_executable(self): |
| 1567 | no_such_executable = 'no_such_executable' |
| 1568 | try: |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1569 | pid = self.spawn_func(no_such_executable, |
| 1570 | [no_such_executable], |
| 1571 | os.environ) |
Pablo Galindo | e9b185f2a | 2019-01-21 14:35:43 +0000 | [diff] [blame] | 1572 | # bpo-35794: PermissionError can be raised if there are |
| 1573 | # directories in the $PATH that are not accessible. |
| 1574 | except (FileNotFoundError, PermissionError) as exc: |
Serhiy Storchaka | ef34753 | 2018-05-01 16:45:04 +0300 | [diff] [blame] | 1575 | self.assertEqual(exc.filename, no_such_executable) |
| 1576 | else: |
| 1577 | pid2, status = os.waitpid(pid, 0) |
| 1578 | self.assertEqual(pid2, pid) |
| 1579 | self.assertNotEqual(status, 0) |
| 1580 | |
| 1581 | def test_specify_environment(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1582 | envfile = os_helper.TESTFN |
| 1583 | self.addCleanup(os_helper.unlink, envfile) |
Serhiy Storchaka | ef34753 | 2018-05-01 16:45:04 +0300 | [diff] [blame] | 1584 | script = f"""if 1: |
| 1585 | import os |
| 1586 | with open({envfile!r}, "w") as envfile: |
| 1587 | envfile.write(os.environ['foo']) |
| 1588 | """ |
Victor Stinner | 0382406 | 2018-08-30 01:21:11 +0200 | [diff] [blame] | 1589 | args = self.python_args('-c', script) |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1590 | pid = self.spawn_func(args[0], args, |
| 1591 | {**os.environ, 'foo': 'bar'}) |
Victor Stinner | 278c1e1 | 2020-03-31 20:08:12 +0200 | [diff] [blame] | 1592 | support.wait_process(pid, exitcode=0) |
Serhiy Storchaka | ef34753 | 2018-05-01 16:45:04 +0300 | [diff] [blame] | 1593 | with open(envfile) as f: |
| 1594 | self.assertEqual(f.read(), 'bar') |
| 1595 | |
Anthony Shaw | 948ed8c | 2019-05-10 12:00:06 +1000 | [diff] [blame] | 1596 | def test_none_file_actions(self): |
| 1597 | pid = self.spawn_func( |
| 1598 | self.NOOP_PROGRAM[0], |
| 1599 | self.NOOP_PROGRAM, |
| 1600 | os.environ, |
| 1601 | file_actions=None |
| 1602 | ) |
Victor Stinner | 278c1e1 | 2020-03-31 20:08:12 +0200 | [diff] [blame] | 1603 | support.wait_process(pid, exitcode=0) |
Anthony Shaw | 948ed8c | 2019-05-10 12:00:06 +1000 | [diff] [blame] | 1604 | |
Serhiy Storchaka | ef34753 | 2018-05-01 16:45:04 +0300 | [diff] [blame] | 1605 | def test_empty_file_actions(self): |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1606 | pid = self.spawn_func( |
Victor Stinner | 0382406 | 2018-08-30 01:21:11 +0200 | [diff] [blame] | 1607 | self.NOOP_PROGRAM[0], |
| 1608 | self.NOOP_PROGRAM, |
Serhiy Storchaka | ef34753 | 2018-05-01 16:45:04 +0300 | [diff] [blame] | 1609 | os.environ, |
Serhiy Storchaka | d700f97 | 2018-09-08 14:48:18 +0300 | [diff] [blame] | 1610 | file_actions=[] |
Serhiy Storchaka | ef34753 | 2018-05-01 16:45:04 +0300 | [diff] [blame] | 1611 | ) |
Victor Stinner | 278c1e1 | 2020-03-31 20:08:12 +0200 | [diff] [blame] | 1612 | support.wait_process(pid, exitcode=0) |
Serhiy Storchaka | ef34753 | 2018-05-01 16:45:04 +0300 | [diff] [blame] | 1613 | |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 1614 | def test_resetids_explicit_default(self): |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1615 | pid = self.spawn_func( |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 1616 | sys.executable, |
| 1617 | [sys.executable, '-c', 'pass'], |
| 1618 | os.environ, |
| 1619 | resetids=False |
| 1620 | ) |
Victor Stinner | 278c1e1 | 2020-03-31 20:08:12 +0200 | [diff] [blame] | 1621 | support.wait_process(pid, exitcode=0) |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 1622 | |
| 1623 | def test_resetids(self): |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1624 | pid = self.spawn_func( |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 1625 | sys.executable, |
| 1626 | [sys.executable, '-c', 'pass'], |
| 1627 | os.environ, |
| 1628 | resetids=True |
| 1629 | ) |
Victor Stinner | 278c1e1 | 2020-03-31 20:08:12 +0200 | [diff] [blame] | 1630 | support.wait_process(pid, exitcode=0) |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 1631 | |
| 1632 | def test_resetids_wrong_type(self): |
| 1633 | with self.assertRaises(TypeError): |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1634 | self.spawn_func(sys.executable, |
| 1635 | [sys.executable, "-c", "pass"], |
| 1636 | os.environ, resetids=None) |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 1637 | |
| 1638 | def test_setpgroup(self): |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1639 | pid = self.spawn_func( |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 1640 | sys.executable, |
| 1641 | [sys.executable, '-c', 'pass'], |
| 1642 | os.environ, |
| 1643 | setpgroup=os.getpgrp() |
| 1644 | ) |
Victor Stinner | 278c1e1 | 2020-03-31 20:08:12 +0200 | [diff] [blame] | 1645 | support.wait_process(pid, exitcode=0) |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 1646 | |
| 1647 | def test_setpgroup_wrong_type(self): |
| 1648 | with self.assertRaises(TypeError): |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1649 | self.spawn_func(sys.executable, |
| 1650 | [sys.executable, "-c", "pass"], |
| 1651 | os.environ, setpgroup="023") |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 1652 | |
| 1653 | @unittest.skipUnless(hasattr(signal, 'pthread_sigmask'), |
| 1654 | 'need signal.pthread_sigmask()') |
| 1655 | def test_setsigmask(self): |
| 1656 | code = textwrap.dedent("""\ |
Vladimir Matveev | c24c6c2 | 2019-01-08 01:58:25 -0800 | [diff] [blame] | 1657 | import signal |
| 1658 | signal.raise_signal(signal.SIGUSR1)""") |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 1659 | |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1660 | pid = self.spawn_func( |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 1661 | sys.executable, |
| 1662 | [sys.executable, '-c', code], |
| 1663 | os.environ, |
| 1664 | setsigmask=[signal.SIGUSR1] |
| 1665 | ) |
Victor Stinner | 278c1e1 | 2020-03-31 20:08:12 +0200 | [diff] [blame] | 1666 | support.wait_process(pid, exitcode=0) |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 1667 | |
| 1668 | def test_setsigmask_wrong_type(self): |
| 1669 | with self.assertRaises(TypeError): |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1670 | self.spawn_func(sys.executable, |
| 1671 | [sys.executable, "-c", "pass"], |
| 1672 | os.environ, setsigmask=34) |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 1673 | with self.assertRaises(TypeError): |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1674 | self.spawn_func(sys.executable, |
| 1675 | [sys.executable, "-c", "pass"], |
| 1676 | os.environ, setsigmask=["j"]) |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 1677 | with self.assertRaises(ValueError): |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1678 | self.spawn_func(sys.executable, |
| 1679 | [sys.executable, "-c", "pass"], |
| 1680 | os.environ, setsigmask=[signal.NSIG, |
| 1681 | signal.NSIG+1]) |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 1682 | |
Victor Stinner | 5884043 | 2019-06-14 19:31:43 +0200 | [diff] [blame] | 1683 | def test_setsid(self): |
| 1684 | rfd, wfd = os.pipe() |
| 1685 | self.addCleanup(os.close, rfd) |
Joannah Nanjekye | 80c5dfe | 2019-02-01 13:05:22 +0300 | [diff] [blame] | 1686 | try: |
Victor Stinner | 5884043 | 2019-06-14 19:31:43 +0200 | [diff] [blame] | 1687 | os.set_inheritable(wfd, True) |
| 1688 | |
| 1689 | code = textwrap.dedent(f""" |
| 1690 | import os |
| 1691 | fd = {wfd} |
| 1692 | sid = os.getsid(0) |
| 1693 | os.write(fd, str(sid).encode()) |
| 1694 | """) |
| 1695 | |
| 1696 | try: |
| 1697 | pid = self.spawn_func(sys.executable, |
| 1698 | [sys.executable, "-c", code], |
| 1699 | os.environ, setsid=True) |
| 1700 | except NotImplementedError as exc: |
| 1701 | self.skipTest(f"setsid is not supported: {exc!r}") |
| 1702 | except PermissionError as exc: |
| 1703 | self.skipTest(f"setsid failed with: {exc!r}") |
| 1704 | finally: |
| 1705 | os.close(wfd) |
| 1706 | |
Victor Stinner | 278c1e1 | 2020-03-31 20:08:12 +0200 | [diff] [blame] | 1707 | support.wait_process(pid, exitcode=0) |
| 1708 | |
Victor Stinner | 5884043 | 2019-06-14 19:31:43 +0200 | [diff] [blame] | 1709 | output = os.read(rfd, 100) |
| 1710 | child_sid = int(output) |
| 1711 | parent_sid = os.getsid(os.getpid()) |
| 1712 | self.assertNotEqual(parent_sid, child_sid) |
Joannah Nanjekye | 80c5dfe | 2019-02-01 13:05:22 +0300 | [diff] [blame] | 1713 | |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 1714 | @unittest.skipUnless(hasattr(signal, 'pthread_sigmask'), |
| 1715 | 'need signal.pthread_sigmask()') |
| 1716 | def test_setsigdef(self): |
| 1717 | original_handler = signal.signal(signal.SIGUSR1, signal.SIG_IGN) |
| 1718 | code = textwrap.dedent("""\ |
Vladimir Matveev | c24c6c2 | 2019-01-08 01:58:25 -0800 | [diff] [blame] | 1719 | import signal |
| 1720 | signal.raise_signal(signal.SIGUSR1)""") |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 1721 | try: |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1722 | pid = self.spawn_func( |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 1723 | sys.executable, |
| 1724 | [sys.executable, '-c', code], |
| 1725 | os.environ, |
| 1726 | setsigdef=[signal.SIGUSR1] |
| 1727 | ) |
| 1728 | finally: |
| 1729 | signal.signal(signal.SIGUSR1, original_handler) |
| 1730 | |
Victor Stinner | 278c1e1 | 2020-03-31 20:08:12 +0200 | [diff] [blame] | 1731 | support.wait_process(pid, exitcode=-signal.SIGUSR1) |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 1732 | |
| 1733 | def test_setsigdef_wrong_type(self): |
| 1734 | with self.assertRaises(TypeError): |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1735 | self.spawn_func(sys.executable, |
| 1736 | [sys.executable, "-c", "pass"], |
| 1737 | os.environ, setsigdef=34) |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 1738 | with self.assertRaises(TypeError): |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1739 | self.spawn_func(sys.executable, |
| 1740 | [sys.executable, "-c", "pass"], |
| 1741 | os.environ, setsigdef=["j"]) |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 1742 | with self.assertRaises(ValueError): |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1743 | self.spawn_func(sys.executable, |
| 1744 | [sys.executable, "-c", "pass"], |
| 1745 | os.environ, setsigdef=[signal.NSIG, signal.NSIG+1]) |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 1746 | |
Benjamin Peterson | c704222 | 2018-09-12 15:12:24 -0700 | [diff] [blame] | 1747 | @requires_sched |
Pablo Galindo | 3d18b50 | 2018-09-14 15:12:22 -0700 | [diff] [blame] | 1748 | @unittest.skipIf(sys.platform.startswith(('freebsd', 'netbsd')), |
| 1749 | "bpo-34685: test can fail on BSD") |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 1750 | def test_setscheduler_only_param(self): |
| 1751 | policy = os.sched_getscheduler(0) |
| 1752 | priority = os.sched_get_priority_min(policy) |
| 1753 | code = textwrap.dedent(f"""\ |
Pablo Galindo | 3d18b50 | 2018-09-14 15:12:22 -0700 | [diff] [blame] | 1754 | import os, sys |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 1755 | if os.sched_getscheduler(0) != {policy}: |
Pablo Galindo | 3d18b50 | 2018-09-14 15:12:22 -0700 | [diff] [blame] | 1756 | sys.exit(101) |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 1757 | if os.sched_getparam(0).sched_priority != {priority}: |
Pablo Galindo | 3d18b50 | 2018-09-14 15:12:22 -0700 | [diff] [blame] | 1758 | sys.exit(102)""") |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1759 | pid = self.spawn_func( |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 1760 | sys.executable, |
| 1761 | [sys.executable, '-c', code], |
| 1762 | os.environ, |
| 1763 | scheduler=(None, os.sched_param(priority)) |
| 1764 | ) |
Victor Stinner | 278c1e1 | 2020-03-31 20:08:12 +0200 | [diff] [blame] | 1765 | support.wait_process(pid, exitcode=0) |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 1766 | |
Benjamin Peterson | c704222 | 2018-09-12 15:12:24 -0700 | [diff] [blame] | 1767 | @requires_sched |
Pablo Galindo | 3d18b50 | 2018-09-14 15:12:22 -0700 | [diff] [blame] | 1768 | @unittest.skipIf(sys.platform.startswith(('freebsd', 'netbsd')), |
| 1769 | "bpo-34685: test can fail on BSD") |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 1770 | def test_setscheduler_with_policy(self): |
| 1771 | policy = os.sched_getscheduler(0) |
| 1772 | priority = os.sched_get_priority_min(policy) |
| 1773 | code = textwrap.dedent(f"""\ |
Pablo Galindo | 3d18b50 | 2018-09-14 15:12:22 -0700 | [diff] [blame] | 1774 | import os, sys |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 1775 | if os.sched_getscheduler(0) != {policy}: |
Pablo Galindo | 3d18b50 | 2018-09-14 15:12:22 -0700 | [diff] [blame] | 1776 | sys.exit(101) |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 1777 | if os.sched_getparam(0).sched_priority != {priority}: |
Pablo Galindo | 3d18b50 | 2018-09-14 15:12:22 -0700 | [diff] [blame] | 1778 | sys.exit(102)""") |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1779 | pid = self.spawn_func( |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 1780 | sys.executable, |
| 1781 | [sys.executable, '-c', code], |
| 1782 | os.environ, |
| 1783 | scheduler=(policy, os.sched_param(priority)) |
| 1784 | ) |
Victor Stinner | 278c1e1 | 2020-03-31 20:08:12 +0200 | [diff] [blame] | 1785 | support.wait_process(pid, exitcode=0) |
Pablo Galindo | 254a466 | 2018-09-07 16:44:24 +0100 | [diff] [blame] | 1786 | |
Serhiy Storchaka | ef34753 | 2018-05-01 16:45:04 +0300 | [diff] [blame] | 1787 | def test_multiple_file_actions(self): |
| 1788 | file_actions = [ |
| 1789 | (os.POSIX_SPAWN_OPEN, 3, os.path.realpath(__file__), os.O_RDONLY, 0), |
| 1790 | (os.POSIX_SPAWN_CLOSE, 0), |
| 1791 | (os.POSIX_SPAWN_DUP2, 1, 4), |
| 1792 | ] |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1793 | pid = self.spawn_func(self.NOOP_PROGRAM[0], |
| 1794 | self.NOOP_PROGRAM, |
| 1795 | os.environ, |
| 1796 | file_actions=file_actions) |
Victor Stinner | 278c1e1 | 2020-03-31 20:08:12 +0200 | [diff] [blame] | 1797 | support.wait_process(pid, exitcode=0) |
Serhiy Storchaka | ef34753 | 2018-05-01 16:45:04 +0300 | [diff] [blame] | 1798 | |
| 1799 | def test_bad_file_actions(self): |
Victor Stinner | 0382406 | 2018-08-30 01:21:11 +0200 | [diff] [blame] | 1800 | args = self.NOOP_PROGRAM |
Serhiy Storchaka | ef34753 | 2018-05-01 16:45:04 +0300 | [diff] [blame] | 1801 | with self.assertRaises(TypeError): |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1802 | self.spawn_func(args[0], args, os.environ, |
| 1803 | file_actions=[None]) |
Serhiy Storchaka | ef34753 | 2018-05-01 16:45:04 +0300 | [diff] [blame] | 1804 | with self.assertRaises(TypeError): |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1805 | self.spawn_func(args[0], args, os.environ, |
| 1806 | file_actions=[()]) |
Serhiy Storchaka | ef34753 | 2018-05-01 16:45:04 +0300 | [diff] [blame] | 1807 | with self.assertRaises(TypeError): |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1808 | self.spawn_func(args[0], args, os.environ, |
| 1809 | file_actions=[(None,)]) |
Serhiy Storchaka | ef34753 | 2018-05-01 16:45:04 +0300 | [diff] [blame] | 1810 | with self.assertRaises(TypeError): |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1811 | self.spawn_func(args[0], args, os.environ, |
| 1812 | file_actions=[(12345,)]) |
Serhiy Storchaka | ef34753 | 2018-05-01 16:45:04 +0300 | [diff] [blame] | 1813 | with self.assertRaises(TypeError): |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1814 | self.spawn_func(args[0], args, os.environ, |
| 1815 | file_actions=[(os.POSIX_SPAWN_CLOSE,)]) |
Serhiy Storchaka | ef34753 | 2018-05-01 16:45:04 +0300 | [diff] [blame] | 1816 | with self.assertRaises(TypeError): |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1817 | self.spawn_func(args[0], args, os.environ, |
| 1818 | file_actions=[(os.POSIX_SPAWN_CLOSE, 1, 2)]) |
Serhiy Storchaka | ef34753 | 2018-05-01 16:45:04 +0300 | [diff] [blame] | 1819 | with self.assertRaises(TypeError): |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1820 | self.spawn_func(args[0], args, os.environ, |
| 1821 | file_actions=[(os.POSIX_SPAWN_CLOSE, None)]) |
Serhiy Storchaka | ef34753 | 2018-05-01 16:45:04 +0300 | [diff] [blame] | 1822 | with self.assertRaises(ValueError): |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1823 | self.spawn_func(args[0], args, os.environ, |
| 1824 | file_actions=[(os.POSIX_SPAWN_OPEN, |
| 1825 | 3, __file__ + '\0', |
| 1826 | os.O_RDONLY, 0)]) |
Serhiy Storchaka | ef34753 | 2018-05-01 16:45:04 +0300 | [diff] [blame] | 1827 | |
| 1828 | def test_open_file(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1829 | outfile = os_helper.TESTFN |
| 1830 | self.addCleanup(os_helper.unlink, outfile) |
Serhiy Storchaka | ef34753 | 2018-05-01 16:45:04 +0300 | [diff] [blame] | 1831 | script = """if 1: |
| 1832 | import sys |
| 1833 | sys.stdout.write("hello") |
| 1834 | """ |
| 1835 | file_actions = [ |
| 1836 | (os.POSIX_SPAWN_OPEN, 1, outfile, |
| 1837 | os.O_WRONLY | os.O_CREAT | os.O_TRUNC, |
| 1838 | stat.S_IRUSR | stat.S_IWUSR), |
| 1839 | ] |
Victor Stinner | 0382406 | 2018-08-30 01:21:11 +0200 | [diff] [blame] | 1840 | args = self.python_args('-c', script) |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1841 | pid = self.spawn_func(args[0], args, os.environ, |
| 1842 | file_actions=file_actions) |
Victor Stinner | 278c1e1 | 2020-03-31 20:08:12 +0200 | [diff] [blame] | 1843 | |
| 1844 | support.wait_process(pid, exitcode=0) |
Serhiy Storchaka | ef34753 | 2018-05-01 16:45:04 +0300 | [diff] [blame] | 1845 | with open(outfile) as f: |
| 1846 | self.assertEqual(f.read(), 'hello') |
| 1847 | |
| 1848 | def test_close_file(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1849 | closefile = os_helper.TESTFN |
| 1850 | self.addCleanup(os_helper.unlink, closefile) |
Serhiy Storchaka | ef34753 | 2018-05-01 16:45:04 +0300 | [diff] [blame] | 1851 | script = f"""if 1: |
| 1852 | import os |
| 1853 | try: |
| 1854 | os.fstat(0) |
| 1855 | except OSError as e: |
| 1856 | with open({closefile!r}, 'w') as closefile: |
| 1857 | closefile.write('is closed %d' % e.errno) |
| 1858 | """ |
Victor Stinner | 0382406 | 2018-08-30 01:21:11 +0200 | [diff] [blame] | 1859 | args = self.python_args('-c', script) |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1860 | pid = self.spawn_func(args[0], args, os.environ, |
| 1861 | file_actions=[(os.POSIX_SPAWN_CLOSE, 0)]) |
Victor Stinner | 278c1e1 | 2020-03-31 20:08:12 +0200 | [diff] [blame] | 1862 | |
| 1863 | support.wait_process(pid, exitcode=0) |
Serhiy Storchaka | ef34753 | 2018-05-01 16:45:04 +0300 | [diff] [blame] | 1864 | with open(closefile) as f: |
| 1865 | self.assertEqual(f.read(), 'is closed %d' % errno.EBADF) |
| 1866 | |
| 1867 | def test_dup2(self): |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1868 | dupfile = os_helper.TESTFN |
| 1869 | self.addCleanup(os_helper.unlink, dupfile) |
Serhiy Storchaka | ef34753 | 2018-05-01 16:45:04 +0300 | [diff] [blame] | 1870 | script = """if 1: |
| 1871 | import sys |
| 1872 | sys.stdout.write("hello") |
| 1873 | """ |
| 1874 | with open(dupfile, "wb") as childfile: |
| 1875 | file_actions = [ |
| 1876 | (os.POSIX_SPAWN_DUP2, childfile.fileno(), 1), |
| 1877 | ] |
Victor Stinner | 0382406 | 2018-08-30 01:21:11 +0200 | [diff] [blame] | 1878 | args = self.python_args('-c', script) |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1879 | pid = self.spawn_func(args[0], args, os.environ, |
| 1880 | file_actions=file_actions) |
Victor Stinner | 278c1e1 | 2020-03-31 20:08:12 +0200 | [diff] [blame] | 1881 | support.wait_process(pid, exitcode=0) |
Serhiy Storchaka | ef34753 | 2018-05-01 16:45:04 +0300 | [diff] [blame] | 1882 | with open(dupfile) as f: |
| 1883 | self.assertEqual(f.read(), 'hello') |
| 1884 | |
| 1885 | |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1886 | @unittest.skipUnless(hasattr(os, 'posix_spawn'), "test needs os.posix_spawn") |
| 1887 | class TestPosixSpawn(unittest.TestCase, _PosixSpawnMixin): |
| 1888 | spawn_func = getattr(posix, 'posix_spawn', None) |
| 1889 | |
| 1890 | |
| 1891 | @unittest.skipUnless(hasattr(os, 'posix_spawnp'), "test needs os.posix_spawnp") |
| 1892 | class TestPosixSpawnP(unittest.TestCase, _PosixSpawnMixin): |
| 1893 | spawn_func = getattr(posix, 'posix_spawnp', None) |
| 1894 | |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1895 | @os_helper.skip_unless_symlink |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1896 | def test_posix_spawnp(self): |
| 1897 | # Use a symlink to create a program in its own temporary directory |
| 1898 | temp_dir = tempfile.mkdtemp() |
Hai Shi | bb0424b | 2020-08-04 00:47:42 +0800 | [diff] [blame] | 1899 | self.addCleanup(os_helper.rmtree, temp_dir) |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1900 | |
| 1901 | program = 'posix_spawnp_test_program.exe' |
| 1902 | program_fullpath = os.path.join(temp_dir, program) |
| 1903 | os.symlink(sys.executable, program_fullpath) |
| 1904 | |
| 1905 | try: |
| 1906 | path = os.pathsep.join((temp_dir, os.environ['PATH'])) |
| 1907 | except KeyError: |
| 1908 | path = temp_dir # PATH is not set |
| 1909 | |
| 1910 | spawn_args = (program, '-I', '-S', '-c', 'pass') |
| 1911 | code = textwrap.dedent(""" |
| 1912 | import os |
Victor Stinner | 278c1e1 | 2020-03-31 20:08:12 +0200 | [diff] [blame] | 1913 | from test import support |
| 1914 | |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1915 | args = %a |
| 1916 | pid = os.posix_spawnp(args[0], args, os.environ) |
Victor Stinner | 278c1e1 | 2020-03-31 20:08:12 +0200 | [diff] [blame] | 1917 | |
| 1918 | support.wait_process(pid, exitcode=0) |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 1919 | """ % (spawn_args,)) |
| 1920 | |
| 1921 | # Use a subprocess to test os.posix_spawnp() with a modified PATH |
| 1922 | # environment variable: posix_spawnp() uses the current environment |
| 1923 | # to locate the program, not its environment argument. |
| 1924 | args = ('-c', code) |
| 1925 | assert_python_ok(*args, PATH=path) |
| 1926 | |
| 1927 | |
Ronald Oussoren | 4176193 | 2020-11-08 10:05:27 +0100 | [diff] [blame^] | 1928 | @unittest.skipUnless(sys.platform == "darwin", "test weak linking on macOS") |
| 1929 | class TestPosixWeaklinking(unittest.TestCase): |
| 1930 | # These test cases verify that weak linking support on macOS works |
| 1931 | # as expected. These cases only test new behaviour introduced by weak linking, |
| 1932 | # regular behaviour is tested by the normal test cases. |
| 1933 | # |
| 1934 | # See the section on Weak Linking in Mac/README.txt for more information. |
| 1935 | def setUp(self): |
| 1936 | import sysconfig |
| 1937 | import platform |
| 1938 | |
| 1939 | config_vars = sysconfig.get_config_vars() |
| 1940 | self.available = { nm for nm in config_vars if nm.startswith("HAVE_") and config_vars[nm] } |
| 1941 | self.mac_ver = tuple(int(part) for part in platform.mac_ver()[0].split(".")) |
| 1942 | |
| 1943 | def _verify_available(self, name): |
| 1944 | if name not in self.available: |
| 1945 | raise unittest.SkipTest(f"{name} not weak-linked") |
| 1946 | |
| 1947 | def test_pwritev(self): |
| 1948 | self._verify_available("HAVE_PWRITEV") |
| 1949 | if self.mac_ver >= (10, 16): |
| 1950 | self.assertTrue(hasattr(os, "pwritev"), "os.pwritev is not available") |
| 1951 | self.assertTrue(hasattr(os, "preadv"), "os.readv is not available") |
| 1952 | |
| 1953 | else: |
| 1954 | self.assertFalse(hasattr(os, "pwritev"), "os.pwritev is available") |
| 1955 | self.assertFalse(hasattr(os, "preadv"), "os.readv is available") |
| 1956 | |
| 1957 | def test_stat(self): |
| 1958 | self._verify_available("HAVE_FSTATAT") |
| 1959 | if self.mac_ver >= (10, 10): |
| 1960 | self.assertIn("HAVE_FSTATAT", posix._have_functions) |
| 1961 | |
| 1962 | else: |
| 1963 | self.assertNotIn("HAVE_FSTATAT", posix._have_functions) |
| 1964 | |
| 1965 | with self.assertRaisesRegex(NotImplementedError, "dir_fd unavailable"): |
| 1966 | os.stat("file", dir_fd=0) |
| 1967 | |
| 1968 | def test_access(self): |
| 1969 | self._verify_available("HAVE_FACCESSAT") |
| 1970 | if self.mac_ver >= (10, 10): |
| 1971 | self.assertIn("HAVE_FACCESSAT", posix._have_functions) |
| 1972 | |
| 1973 | else: |
| 1974 | self.assertNotIn("HAVE_FACCESSAT", posix._have_functions) |
| 1975 | |
| 1976 | with self.assertRaisesRegex(NotImplementedError, "dir_fd unavailable"): |
| 1977 | os.access("file", os.R_OK, dir_fd=0) |
| 1978 | |
| 1979 | with self.assertRaisesRegex(NotImplementedError, "follow_symlinks unavailable"): |
| 1980 | os.access("file", os.R_OK, follow_symlinks=False) |
| 1981 | |
| 1982 | with self.assertRaisesRegex(NotImplementedError, "effective_ids unavailable"): |
| 1983 | os.access("file", os.R_OK, effective_ids=True) |
| 1984 | |
| 1985 | def test_chmod(self): |
| 1986 | self._verify_available("HAVE_FCHMODAT") |
| 1987 | if self.mac_ver >= (10, 10): |
| 1988 | self.assertIn("HAVE_FCHMODAT", posix._have_functions) |
| 1989 | |
| 1990 | else: |
| 1991 | self.assertNotIn("HAVE_FCHMODAT", posix._have_functions) |
| 1992 | self.assertIn("HAVE_LCHMOD", posix._have_functions) |
| 1993 | |
| 1994 | with self.assertRaisesRegex(NotImplementedError, "dir_fd unavailable"): |
| 1995 | os.chmod("file", 0o644, dir_fd=0) |
| 1996 | |
| 1997 | def test_chown(self): |
| 1998 | self._verify_available("HAVE_FCHOWNAT") |
| 1999 | if self.mac_ver >= (10, 10): |
| 2000 | self.assertIn("HAVE_FCHOWNAT", posix._have_functions) |
| 2001 | |
| 2002 | else: |
| 2003 | self.assertNotIn("HAVE_FCHOWNAT", posix._have_functions) |
| 2004 | self.assertIn("HAVE_LCHOWN", posix._have_functions) |
| 2005 | |
| 2006 | with self.assertRaisesRegex(NotImplementedError, "dir_fd unavailable"): |
| 2007 | os.chown("file", 0, 0, dir_fd=0) |
| 2008 | |
| 2009 | def test_link(self): |
| 2010 | self._verify_available("HAVE_LINKAT") |
| 2011 | if self.mac_ver >= (10, 10): |
| 2012 | self.assertIn("HAVE_LINKAT", posix._have_functions) |
| 2013 | |
| 2014 | else: |
| 2015 | self.assertNotIn("HAVE_LINKAT", posix._have_functions) |
| 2016 | |
| 2017 | with self.assertRaisesRegex(NotImplementedError, "src_dir_fd unavailable"): |
| 2018 | os.link("source", "target", src_dir_fd=0) |
| 2019 | |
| 2020 | with self.assertRaisesRegex(NotImplementedError, "dst_dir_fd unavailable"): |
| 2021 | os.link("source", "target", dst_dir_fd=0) |
| 2022 | |
| 2023 | with self.assertRaisesRegex(NotImplementedError, "src_dir_fd unavailable"): |
| 2024 | os.link("source", "target", src_dir_fd=0, dst_dir_fd=0) |
| 2025 | |
| 2026 | # issue 41355: !HAVE_LINKAT code path ignores the follow_symlinks flag |
| 2027 | with os_helper.temp_dir() as base_path: |
| 2028 | link_path = os.path.join(base_path, "link") |
| 2029 | target_path = os.path.join(base_path, "target") |
| 2030 | source_path = os.path.join(base_path, "source") |
| 2031 | |
| 2032 | with open(source_path, "w") as fp: |
| 2033 | fp.write("data") |
| 2034 | |
| 2035 | os.symlink("target", link_path) |
| 2036 | |
| 2037 | # Calling os.link should fail in the link(2) call, and |
| 2038 | # should not reject *follow_symlinks* (to match the |
| 2039 | # behaviour you'd get when building on a platform without |
| 2040 | # linkat) |
| 2041 | with self.assertRaises(FileExistsError): |
| 2042 | os.link(source_path, link_path, follow_symlinks=True) |
| 2043 | |
| 2044 | with self.assertRaises(FileExistsError): |
| 2045 | os.link(source_path, link_path, follow_symlinks=False) |
| 2046 | |
| 2047 | |
| 2048 | def test_listdir_scandir(self): |
| 2049 | self._verify_available("HAVE_FDOPENDIR") |
| 2050 | if self.mac_ver >= (10, 10): |
| 2051 | self.assertIn("HAVE_FDOPENDIR", posix._have_functions) |
| 2052 | |
| 2053 | else: |
| 2054 | self.assertNotIn("HAVE_FDOPENDIR", posix._have_functions) |
| 2055 | |
| 2056 | with self.assertRaisesRegex(TypeError, "listdir: path should be string, bytes, os.PathLike or None, not int"): |
| 2057 | os.listdir(0) |
| 2058 | |
| 2059 | with self.assertRaisesRegex(TypeError, "scandir: path should be string, bytes, os.PathLike or None, not int"): |
| 2060 | os.scandir(0) |
| 2061 | |
| 2062 | def test_mkdir(self): |
| 2063 | self._verify_available("HAVE_MKDIRAT") |
| 2064 | if self.mac_ver >= (10, 10): |
| 2065 | self.assertIn("HAVE_MKDIRAT", posix._have_functions) |
| 2066 | |
| 2067 | else: |
| 2068 | self.assertNotIn("HAVE_MKDIRAT", posix._have_functions) |
| 2069 | |
| 2070 | with self.assertRaisesRegex(NotImplementedError, "dir_fd unavailable"): |
| 2071 | os.mkdir("dir", dir_fd=0) |
| 2072 | |
| 2073 | def test_rename_replace(self): |
| 2074 | self._verify_available("HAVE_RENAMEAT") |
| 2075 | if self.mac_ver >= (10, 10): |
| 2076 | self.assertIn("HAVE_RENAMEAT", posix._have_functions) |
| 2077 | |
| 2078 | else: |
| 2079 | self.assertNotIn("HAVE_RENAMEAT", posix._have_functions) |
| 2080 | |
| 2081 | with self.assertRaisesRegex(NotImplementedError, "src_dir_fd and dst_dir_fd unavailable"): |
| 2082 | os.rename("a", "b", src_dir_fd=0) |
| 2083 | |
| 2084 | with self.assertRaisesRegex(NotImplementedError, "src_dir_fd and dst_dir_fd unavailable"): |
| 2085 | os.rename("a", "b", dst_dir_fd=0) |
| 2086 | |
| 2087 | with self.assertRaisesRegex(NotImplementedError, "src_dir_fd and dst_dir_fd unavailable"): |
| 2088 | os.replace("a", "b", src_dir_fd=0) |
| 2089 | |
| 2090 | with self.assertRaisesRegex(NotImplementedError, "src_dir_fd and dst_dir_fd unavailable"): |
| 2091 | os.replace("a", "b", dst_dir_fd=0) |
| 2092 | |
| 2093 | def test_unlink_rmdir(self): |
| 2094 | self._verify_available("HAVE_UNLINKAT") |
| 2095 | if self.mac_ver >= (10, 10): |
| 2096 | self.assertIn("HAVE_UNLINKAT", posix._have_functions) |
| 2097 | |
| 2098 | else: |
| 2099 | self.assertNotIn("HAVE_UNLINKAT", posix._have_functions) |
| 2100 | |
| 2101 | with self.assertRaisesRegex(NotImplementedError, "dir_fd unavailable"): |
| 2102 | os.unlink("path", dir_fd=0) |
| 2103 | |
| 2104 | with self.assertRaisesRegex(NotImplementedError, "dir_fd unavailable"): |
| 2105 | os.rmdir("path", dir_fd=0) |
| 2106 | |
| 2107 | def test_open(self): |
| 2108 | self._verify_available("HAVE_OPENAT") |
| 2109 | if self.mac_ver >= (10, 10): |
| 2110 | self.assertIn("HAVE_OPENAT", posix._have_functions) |
| 2111 | |
| 2112 | else: |
| 2113 | self.assertNotIn("HAVE_OPENAT", posix._have_functions) |
| 2114 | |
| 2115 | with self.assertRaisesRegex(NotImplementedError, "dir_fd unavailable"): |
| 2116 | os.open("path", os.O_RDONLY, dir_fd=0) |
| 2117 | |
| 2118 | def test_readlink(self): |
| 2119 | self._verify_available("HAVE_READLINKAT") |
| 2120 | if self.mac_ver >= (10, 10): |
| 2121 | self.assertIn("HAVE_READLINKAT", posix._have_functions) |
| 2122 | |
| 2123 | else: |
| 2124 | self.assertNotIn("HAVE_READLINKAT", posix._have_functions) |
| 2125 | |
| 2126 | with self.assertRaisesRegex(NotImplementedError, "dir_fd unavailable"): |
| 2127 | os.readlink("path", dir_fd=0) |
| 2128 | |
| 2129 | def test_symlink(self): |
| 2130 | self._verify_available("HAVE_SYMLINKAT") |
| 2131 | if self.mac_ver >= (10, 10): |
| 2132 | self.assertIn("HAVE_SYMLINKAT", posix._have_functions) |
| 2133 | |
| 2134 | else: |
| 2135 | self.assertNotIn("HAVE_SYMLINKAT", posix._have_functions) |
| 2136 | |
| 2137 | with self.assertRaisesRegex(NotImplementedError, "dir_fd unavailable"): |
| 2138 | os.symlink("a", "b", dir_fd=0) |
| 2139 | |
| 2140 | def test_utime(self): |
| 2141 | self._verify_available("HAVE_FUTIMENS") |
| 2142 | self._verify_available("HAVE_UTIMENSAT") |
| 2143 | if self.mac_ver >= (10, 13): |
| 2144 | self.assertIn("HAVE_FUTIMENS", posix._have_functions) |
| 2145 | self.assertIn("HAVE_UTIMENSAT", posix._have_functions) |
| 2146 | |
| 2147 | else: |
| 2148 | self.assertNotIn("HAVE_FUTIMENS", posix._have_functions) |
| 2149 | self.assertNotIn("HAVE_UTIMENSAT", posix._have_functions) |
| 2150 | |
| 2151 | with self.assertRaisesRegex(NotImplementedError, "dir_fd unavailable"): |
| 2152 | os.utime("path", dir_fd=0) |
| 2153 | |
| 2154 | |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 2155 | def test_main(): |
Antoine Pitrou | 68c9592 | 2011-03-20 17:33:57 +0100 | [diff] [blame] | 2156 | try: |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 2157 | support.run_unittest( |
| 2158 | PosixTester, |
| 2159 | PosixGroupsTester, |
| 2160 | TestPosixSpawn, |
| 2161 | TestPosixSpawnP, |
Ronald Oussoren | 4176193 | 2020-11-08 10:05:27 +0100 | [diff] [blame^] | 2162 | TestPosixWeaklinking |
Joannah Nanjekye | 92b8322 | 2019-01-16 16:29:26 +0300 | [diff] [blame] | 2163 | ) |
Antoine Pitrou | 68c9592 | 2011-03-20 17:33:57 +0100 | [diff] [blame] | 2164 | finally: |
| 2165 | support.reap_children() |
Neal Norwitz | e241ce8 | 2003-02-17 18:17:05 +0000 | [diff] [blame] | 2166 | |
| 2167 | if __name__ == '__main__': |
| 2168 | test_main() |