blob: cd8a1b973eef1d124d3baf79809598447d3cd41c [file] [log] [blame]
Fred Drake38c2ef02001-07-17 20:52:51 +00001# As a test suite for the os module, this is woefully inadequate, but this
2# does add tests for a few functions which have been determined to be more
Walter Dörwaldf0dfc7a2003-10-20 14:01:56 +00003# portable than they had been thought to be.
Fred Drake38c2ef02001-07-17 20:52:51 +00004
5import os
Benjamin Peterson5c6d7872009-02-06 02:40:07 +00006import errno
Fred Drake38c2ef02001-07-17 20:52:51 +00007import unittest
Jeremy Hyltona7fc21b2001-08-20 20:10:01 +00008import warnings
Thomas Wouters477c8d52006-05-27 19:21:47 +00009import sys
Brian Curtineb24d742010-04-12 17:16:38 +000010import signal
11import subprocess
12import time
Martin v. Löwis011e8422009-05-05 04:43:17 +000013import shutil
Benjamin Petersonee8712c2008-05-20 21:35:26 +000014from test import support
Victor Stinnerc2d095f2010-05-17 00:14:53 +000015import contextlib
Fred Drake38c2ef02001-07-17 20:52:51 +000016
Mark Dickinson7cf03892010-04-16 13:45:35 +000017# Detect whether we're on a Linux system that uses the (now outdated
18# and unmaintained) linuxthreads threading library. There's an issue
19# when combining linuxthreads with a failed execv call: see
20# http://bugs.python.org/issue4970.
Mark Dickinson89589c92010-04-16 13:51:27 +000021if (hasattr(os, "confstr_names") and
22 "CS_GNU_LIBPTHREAD_VERSION" in os.confstr_names):
Mark Dickinson7cf03892010-04-16 13:45:35 +000023 libpthread = os.confstr("CS_GNU_LIBPTHREAD_VERSION")
24 USING_LINUXTHREADS= libpthread.startswith("linuxthreads")
25else:
26 USING_LINUXTHREADS= False
Brian Curtineb24d742010-04-12 17:16:38 +000027
Thomas Wouters0e3f5912006-08-11 14:57:12 +000028# Tests creating TESTFN
29class FileTests(unittest.TestCase):
30 def setUp(self):
Benjamin Petersonee8712c2008-05-20 21:35:26 +000031 if os.path.exists(support.TESTFN):
32 os.unlink(support.TESTFN)
Thomas Wouters0e3f5912006-08-11 14:57:12 +000033 tearDown = setUp
34
35 def test_access(self):
Benjamin Petersonee8712c2008-05-20 21:35:26 +000036 f = os.open(support.TESTFN, os.O_CREAT|os.O_RDWR)
Thomas Wouters0e3f5912006-08-11 14:57:12 +000037 os.close(f)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +000038 self.assertTrue(os.access(support.TESTFN, os.W_OK))
Thomas Wouters0e3f5912006-08-11 14:57:12 +000039
Christian Heimesfdab48e2008-01-20 09:06:41 +000040 def test_closerange(self):
Antoine Pitroub9ee06c2008-08-16 22:03:17 +000041 first = os.open(support.TESTFN, os.O_CREAT|os.O_RDWR)
42 # We must allocate two consecutive file descriptors, otherwise
43 # it will mess up other file descriptors (perhaps even the three
44 # standard ones).
45 second = os.dup(first)
46 try:
47 retries = 0
48 while second != first + 1:
49 os.close(first)
50 retries += 1
51 if retries > 10:
52 # XXX test skipped
Benjamin Petersonfa0d7032009-06-01 22:42:33 +000053 self.skipTest("couldn't allocate two consecutive fds")
Antoine Pitroub9ee06c2008-08-16 22:03:17 +000054 first, second = second, os.dup(second)
55 finally:
56 os.close(second)
Christian Heimesfdab48e2008-01-20 09:06:41 +000057 # close a fd that is open, and one that isn't
Antoine Pitroub9ee06c2008-08-16 22:03:17 +000058 os.closerange(first, first + 2)
Antoine Pitrou9cadb1b2008-09-15 23:02:56 +000059 self.assertRaises(OSError, os.write, first, b"a")
Thomas Wouters0e3f5912006-08-11 14:57:12 +000060
Benjamin Peterson1cc6df92010-06-30 17:39:45 +000061 @support.cpython_only
Hirokazu Yamamoto4c19e6e2008-09-08 23:41:21 +000062 def test_rename(self):
63 path = support.TESTFN
64 old = sys.getrefcount(path)
65 self.assertRaises(TypeError, os.rename, path, 0)
66 new = sys.getrefcount(path)
67 self.assertEqual(old, new)
68
Antoine Pitrou9cadb1b2008-09-15 23:02:56 +000069 def test_read(self):
70 with open(support.TESTFN, "w+b") as fobj:
71 fobj.write(b"spam")
72 fobj.flush()
73 fd = fobj.fileno()
74 os.lseek(fd, 0, 0)
75 s = os.read(fd, 4)
76 self.assertEqual(type(s), bytes)
77 self.assertEqual(s, b"spam")
78
79 def test_write(self):
80 # os.write() accepts bytes- and buffer-like objects but not strings
81 fd = os.open(support.TESTFN, os.O_CREAT | os.O_WRONLY)
82 self.assertRaises(TypeError, os.write, fd, "beans")
83 os.write(fd, b"bacon\n")
84 os.write(fd, bytearray(b"eggs\n"))
85 os.write(fd, memoryview(b"spam\n"))
86 os.close(fd)
87 with open(support.TESTFN, "rb") as fobj:
Antoine Pitroud62269f2008-09-15 23:54:52 +000088 self.assertEqual(fobj.read().splitlines(),
89 [b"bacon", b"eggs", b"spam"])
Antoine Pitrou9cadb1b2008-09-15 23:02:56 +000090
91
Christian Heimesdd15f6c2008-03-16 00:07:10 +000092class TemporaryFileTests(unittest.TestCase):
93 def setUp(self):
94 self.files = []
Benjamin Petersonee8712c2008-05-20 21:35:26 +000095 os.mkdir(support.TESTFN)
Christian Heimesdd15f6c2008-03-16 00:07:10 +000096
97 def tearDown(self):
98 for name in self.files:
99 os.unlink(name)
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000100 os.rmdir(support.TESTFN)
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000101
102 def check_tempfile(self, name):
103 # make sure it doesn't already exist:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000104 self.assertFalse(os.path.exists(name),
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000105 "file already exists for temporary file")
106 # make sure we can create the file
107 open(name, "w")
108 self.files.append(name)
109
110 def test_tempnam(self):
111 if not hasattr(os, "tempnam"):
112 return
113 warnings.filterwarnings("ignore", "tempnam", RuntimeWarning,
114 r"test_os$")
115 self.check_tempfile(os.tempnam())
116
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000117 name = os.tempnam(support.TESTFN)
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000118 self.check_tempfile(name)
119
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000120 name = os.tempnam(support.TESTFN, "pfx")
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000121 self.assertTrue(os.path.basename(name)[:3] == "pfx")
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000122 self.check_tempfile(name)
123
124 def test_tmpfile(self):
125 if not hasattr(os, "tmpfile"):
126 return
127 # As with test_tmpnam() below, the Windows implementation of tmpfile()
128 # attempts to create a file in the root directory of the current drive.
129 # On Vista and Server 2008, this test will always fail for normal users
130 # as writing to the root directory requires elevated privileges. With
131 # XP and below, the semantics of tmpfile() are the same, but the user
132 # running the test is more likely to have administrative privileges on
133 # their account already. If that's the case, then os.tmpfile() should
134 # work. In order to make this test as useful as possible, rather than
135 # trying to detect Windows versions or whether or not the user has the
136 # right permissions, just try and create a file in the root directory
137 # and see if it raises a 'Permission denied' OSError. If it does, then
138 # test that a subsequent call to os.tmpfile() raises the same error. If
139 # it doesn't, assume we're on XP or below and the user running the test
140 # has administrative privileges, and proceed with the test as normal.
141 if sys.platform == 'win32':
142 name = '\\python_test_os_test_tmpfile.txt'
143 if os.path.exists(name):
144 os.remove(name)
145 try:
146 fp = open(name, 'w')
147 except IOError as first:
148 # open() failed, assert tmpfile() fails in the same way.
149 # Although open() raises an IOError and os.tmpfile() raises an
150 # OSError(), 'args' will be (13, 'Permission denied') in both
151 # cases.
152 try:
153 fp = os.tmpfile()
154 except OSError as second:
155 self.assertEqual(first.args, second.args)
156 else:
157 self.fail("expected os.tmpfile() to raise OSError")
158 return
159 else:
160 # open() worked, therefore, tmpfile() should work. Close our
161 # dummy file and proceed with the test as normal.
162 fp.close()
163 os.remove(name)
164
165 fp = os.tmpfile()
166 fp.write("foobar")
167 fp.seek(0,0)
168 s = fp.read()
169 fp.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000170 self.assertTrue(s == "foobar")
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000171
172 def test_tmpnam(self):
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000173 if not hasattr(os, "tmpnam"):
174 return
175 warnings.filterwarnings("ignore", "tmpnam", RuntimeWarning,
176 r"test_os$")
177 name = os.tmpnam()
178 if sys.platform in ("win32",):
179 # The Windows tmpnam() seems useless. From the MS docs:
180 #
181 # The character string that tmpnam creates consists of
182 # the path prefix, defined by the entry P_tmpdir in the
183 # file STDIO.H, followed by a sequence consisting of the
184 # digit characters '0' through '9'; the numerical value
185 # of this string is in the range 1 - 65,535. Changing the
186 # definitions of L_tmpnam or P_tmpdir in STDIO.H does not
187 # change the operation of tmpnam.
188 #
189 # The really bizarre part is that, at least under MSVC6,
190 # P_tmpdir is "\\". That is, the path returned refers to
191 # the root of the current drive. That's a terrible place to
192 # put temp files, and, depending on privileges, the user
193 # may not even be able to open a file in the root directory.
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000194 self.assertFalse(os.path.exists(name),
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000195 "file already exists for temporary file")
196 else:
197 self.check_tempfile(name)
198
Amaury Forgeot d'Arce2e36ba2008-08-01 00:14:22 +0000199 def fdopen_helper(self, *args):
200 fd = os.open(support.TESTFN, os.O_RDONLY)
201 fp2 = os.fdopen(fd, *args)
202 fp2.close()
203
204 def test_fdopen(self):
205 self.fdopen_helper()
206 self.fdopen_helper('r')
207 self.fdopen_helper('r', 100)
208
Guido van Rossum98bf58f2001-10-18 20:34:25 +0000209# Test attributes on return values from os.*stat* family.
210class StatAttributeTests(unittest.TestCase):
211 def setUp(self):
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000212 os.mkdir(support.TESTFN)
213 self.fname = os.path.join(support.TESTFN, "f1")
Guido van Rossum98bf58f2001-10-18 20:34:25 +0000214 f = open(self.fname, 'wb')
Guido van Rossum26d95c32007-08-27 23:18:54 +0000215 f.write(b"ABC")
Guido van Rossum98bf58f2001-10-18 20:34:25 +0000216 f.close()
Tim Peterse0c446b2001-10-18 21:57:37 +0000217
Guido van Rossum98bf58f2001-10-18 20:34:25 +0000218 def tearDown(self):
219 os.unlink(self.fname)
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000220 os.rmdir(support.TESTFN)
Guido van Rossum98bf58f2001-10-18 20:34:25 +0000221
222 def test_stat_attributes(self):
223 if not hasattr(os, "stat"):
224 return
225
226 import stat
227 result = os.stat(self.fname)
228
229 # Make sure direct access works
230 self.assertEquals(result[stat.ST_SIZE], 3)
231 self.assertEquals(result.st_size, 3)
232
Guido van Rossum98bf58f2001-10-18 20:34:25 +0000233 # Make sure all the attributes are there
234 members = dir(result)
235 for name in dir(stat):
236 if name[:3] == 'ST_':
237 attr = name.lower()
Martin v. Löwis4d394df2005-01-23 09:19:22 +0000238 if name.endswith("TIME"):
239 def trunc(x): return int(x)
240 else:
241 def trunc(x): return x
242 self.assertEquals(trunc(getattr(result, attr)),
Guido van Rossum98bf58f2001-10-18 20:34:25 +0000243 result[getattr(stat, name)])
Benjamin Peterson577473f2010-01-19 00:09:57 +0000244 self.assertIn(attr, members)
Guido van Rossum98bf58f2001-10-18 20:34:25 +0000245
246 try:
247 result[200]
248 self.fail("No exception thrown")
249 except IndexError:
250 pass
251
252 # Make sure that assignment fails
253 try:
254 result.st_mode = 1
255 self.fail("No exception thrown")
Collin Winter42dae6a2007-03-28 21:44:53 +0000256 except AttributeError:
Guido van Rossum98bf58f2001-10-18 20:34:25 +0000257 pass
258
259 try:
260 result.st_rdev = 1
261 self.fail("No exception thrown")
Guido van Rossum1fff8782001-10-18 21:19:31 +0000262 except (AttributeError, TypeError):
Guido van Rossum98bf58f2001-10-18 20:34:25 +0000263 pass
264
265 try:
266 result.parrot = 1
267 self.fail("No exception thrown")
268 except AttributeError:
269 pass
270
271 # Use the stat_result constructor with a too-short tuple.
272 try:
273 result2 = os.stat_result((10,))
274 self.fail("No exception thrown")
275 except TypeError:
276 pass
277
278 # Use the constructr with a too-long tuple.
279 try:
280 result2 = os.stat_result((0,1,2,3,4,5,6,7,8,9,10,11,12,13,14))
281 except TypeError:
282 pass
283
Tim Peterse0c446b2001-10-18 21:57:37 +0000284
Guido van Rossum98bf58f2001-10-18 20:34:25 +0000285 def test_statvfs_attributes(self):
286 if not hasattr(os, "statvfs"):
287 return
288
Martin v. Löwisf90ae202002-06-11 06:22:31 +0000289 try:
290 result = os.statvfs(self.fname)
Guido van Rossumb940e112007-01-10 16:19:56 +0000291 except OSError as e:
Martin v. Löwisf90ae202002-06-11 06:22:31 +0000292 # On AtheOS, glibc always returns ENOSYS
Martin v. Löwisf90ae202002-06-11 06:22:31 +0000293 if e.errno == errno.ENOSYS:
294 return
Guido van Rossum98bf58f2001-10-18 20:34:25 +0000295
296 # Make sure direct access works
Brett Cannoncfaf10c2008-05-16 00:45:35 +0000297 self.assertEquals(result.f_bfree, result[3])
Guido van Rossum98bf58f2001-10-18 20:34:25 +0000298
Brett Cannoncfaf10c2008-05-16 00:45:35 +0000299 # Make sure all the attributes are there.
300 members = ('bsize', 'frsize', 'blocks', 'bfree', 'bavail', 'files',
301 'ffree', 'favail', 'flag', 'namemax')
302 for value, member in enumerate(members):
303 self.assertEquals(getattr(result, 'f_' + member), result[value])
Guido van Rossum98bf58f2001-10-18 20:34:25 +0000304
305 # Make sure that assignment really fails
306 try:
307 result.f_bfree = 1
308 self.fail("No exception thrown")
Collin Winter42dae6a2007-03-28 21:44:53 +0000309 except AttributeError:
Guido van Rossum98bf58f2001-10-18 20:34:25 +0000310 pass
311
312 try:
313 result.parrot = 1
314 self.fail("No exception thrown")
315 except AttributeError:
316 pass
317
318 # Use the constructor with a too-short tuple.
319 try:
320 result2 = os.statvfs_result((10,))
321 self.fail("No exception thrown")
322 except TypeError:
323 pass
324
325 # Use the constructr with a too-long tuple.
326 try:
327 result2 = os.statvfs_result((0,1,2,3,4,5,6,7,8,9,10,11,12,13,14))
328 except TypeError:
329 pass
Fred Drake38c2ef02001-07-17 20:52:51 +0000330
Thomas Wouters89f507f2006-12-13 04:49:30 +0000331 def test_utime_dir(self):
332 delta = 1000000
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000333 st = os.stat(support.TESTFN)
Thomas Wouters89f507f2006-12-13 04:49:30 +0000334 # round to int, because some systems may support sub-second
335 # time stamps in stat, but not in utime.
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000336 os.utime(support.TESTFN, (st.st_atime, int(st.st_mtime-delta)))
337 st2 = os.stat(support.TESTFN)
Thomas Wouters89f507f2006-12-13 04:49:30 +0000338 self.assertEquals(st2.st_mtime, int(st.st_mtime-delta))
339
340 # Restrict test to Win32, since there is no guarantee other
341 # systems support centiseconds
342 if sys.platform == 'win32':
Thomas Wouters47b49bf2007-08-30 22:15:33 +0000343 def get_file_system(path):
Hirokazu Yamamoto5ef6d182008-08-20 04:17:24 +0000344 root = os.path.splitdrive(os.path.abspath(path))[0] + '\\'
Thomas Wouters47b49bf2007-08-30 22:15:33 +0000345 import ctypes
Hirokazu Yamamotoca765d52008-08-20 16:18:19 +0000346 kernel32 = ctypes.windll.kernel32
Hirokazu Yamamoto5ef6d182008-08-20 04:17:24 +0000347 buf = ctypes.create_unicode_buffer("", 100)
Hirokazu Yamamotoca765d52008-08-20 16:18:19 +0000348 if kernel32.GetVolumeInformationW(root, None, 0, None, None, None, buf, len(buf)):
Thomas Wouters47b49bf2007-08-30 22:15:33 +0000349 return buf.value
350
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000351 if get_file_system(support.TESTFN) == "NTFS":
Thomas Wouters47b49bf2007-08-30 22:15:33 +0000352 def test_1565150(self):
353 t1 = 1159195039.25
354 os.utime(self.fname, (t1, t1))
355 self.assertEquals(os.stat(self.fname).st_mtime, t1)
Thomas Wouters89f507f2006-12-13 04:49:30 +0000356
Guido van Rossumd8faa362007-04-27 19:54:29 +0000357 def test_1686475(self):
358 # Verify that an open file can be stat'ed
359 try:
360 os.stat(r"c:\pagefile.sys")
361 except WindowsError as e:
Benjamin Petersonc4fe6f32008-08-19 18:57:56 +0000362 if e.errno == 2: # file does not exist; cannot run test
Guido van Rossumd8faa362007-04-27 19:54:29 +0000363 return
364 self.fail("Could not stat pagefile.sys")
365
Walter Dörwald0a6d0ff2004-05-31 16:29:04 +0000366from test import mapping_tests
Raymond Hettinger2c2d3222003-03-09 07:05:43 +0000367
Walter Dörwald0a6d0ff2004-05-31 16:29:04 +0000368class EnvironTests(mapping_tests.BasicTestMappingProtocol):
Raymond Hettinger2c2d3222003-03-09 07:05:43 +0000369 """check that os.environ object conform to mapping protocol"""
Walter Dörwald118f9312004-06-02 18:42:25 +0000370 type2test = None
Christian Heimes90333392007-11-01 19:08:42 +0000371
Raymond Hettinger2c2d3222003-03-09 07:05:43 +0000372 def setUp(self):
373 self.__save = dict(os.environ)
Victor Stinnerb745a742010-05-18 17:17:23 +0000374 if os.supports_bytes_environ:
Victor Stinner208d28c2010-05-07 00:54:14 +0000375 self.__saveb = dict(os.environb)
Christian Heimes90333392007-11-01 19:08:42 +0000376 for key, value in self._reference().items():
377 os.environ[key] = value
378
Raymond Hettinger2c2d3222003-03-09 07:05:43 +0000379 def tearDown(self):
380 os.environ.clear()
381 os.environ.update(self.__save)
Victor Stinnerb745a742010-05-18 17:17:23 +0000382 if os.supports_bytes_environ:
Victor Stinner208d28c2010-05-07 00:54:14 +0000383 os.environb.clear()
384 os.environb.update(self.__saveb)
Raymond Hettinger2c2d3222003-03-09 07:05:43 +0000385
Christian Heimes90333392007-11-01 19:08:42 +0000386 def _reference(self):
387 return {"KEY1":"VALUE1", "KEY2":"VALUE2", "KEY3":"VALUE3"}
388
389 def _empty_mapping(self):
390 os.environ.clear()
391 return os.environ
392
Martin v. Löwis1d11de62005-01-29 13:29:23 +0000393 # Bug 1110478
Martin v. Löwis5510f652005-02-17 21:23:20 +0000394 def test_update2(self):
Christian Heimes90333392007-11-01 19:08:42 +0000395 os.environ.clear()
Martin v. Löwis1d11de62005-01-29 13:29:23 +0000396 if os.path.exists("/bin/sh"):
397 os.environ.update(HELLO="World")
398 value = os.popen("/bin/sh -c 'echo $HELLO'").read().strip()
399 self.assertEquals(value, "World")
400
Christian Heimes1a13d592007-11-08 14:16:55 +0000401 def test_os_popen_iter(self):
402 if os.path.exists("/bin/sh"):
403 popen = os.popen("/bin/sh -c 'echo \"line1\nline2\nline3\"'")
404 it = iter(popen)
405 self.assertEquals(next(it), "line1\n")
406 self.assertEquals(next(it), "line2\n")
407 self.assertEquals(next(it), "line3\n")
408 self.assertRaises(StopIteration, next, it)
409
Guido van Rossum67aca9e2007-06-13 21:51:27 +0000410 # Verify environ keys and values from the OS are of the
411 # correct str type.
412 def test_keyvalue_types(self):
413 for key, val in os.environ.items():
414 self.assertEquals(type(key), str)
415 self.assertEquals(type(val), str)
416
Christian Heimes90333392007-11-01 19:08:42 +0000417 def test_items(self):
418 for key, value in self._reference().items():
419 self.assertEqual(os.environ.get(key), value)
420
Ezio Melotti19e4acf2010-02-22 15:59:01 +0000421 # Issue 7310
422 def test___repr__(self):
423 """Check that the repr() of os.environ looks like environ({...})."""
424 env = os.environ
425 self.assertTrue(isinstance(env.data, dict))
Victor Stinner96f0de92010-07-29 00:29:00 +0000426 self.assertEqual(repr(env), 'environ({{{}}})'.format(', '.join(
427 '{!r}: {!r}'.format(key, value)
428 for key, value in env.items())))
Ezio Melotti19e4acf2010-02-22 15:59:01 +0000429
Gregory P. Smithb6e8c7e2010-02-27 07:22:22 +0000430 def test_get_exec_path(self):
431 defpath_list = os.defpath.split(os.pathsep)
432 test_path = ['/monty', '/python', '', '/flying/circus']
433 test_env = {'PATH': os.pathsep.join(test_path)}
434
435 saved_environ = os.environ
436 try:
437 os.environ = dict(test_env)
438 # Test that defaulting to os.environ works.
439 self.assertSequenceEqual(test_path, os.get_exec_path())
440 self.assertSequenceEqual(test_path, os.get_exec_path(env=None))
441 finally:
442 os.environ = saved_environ
443
444 # No PATH environment variable
445 self.assertSequenceEqual(defpath_list, os.get_exec_path({}))
446 # Empty PATH environment variable
447 self.assertSequenceEqual(('',), os.get_exec_path({'PATH':''}))
448 # Supplied PATH environment variable
449 self.assertSequenceEqual(test_path, os.get_exec_path(test_env))
450
Victor Stinnerb745a742010-05-18 17:17:23 +0000451 if os.supports_bytes_environ:
452 # env cannot contain 'PATH' and b'PATH' keys
453 self.assertRaises(ValueError,
454 os.get_exec_path, {'PATH': '1', b'PATH': b'2'})
455
456 # bytes key and/or value
457 self.assertSequenceEqual(os.get_exec_path({b'PATH': b'abc'}),
458 ['abc'])
459 self.assertSequenceEqual(os.get_exec_path({b'PATH': 'abc'}),
460 ['abc'])
461 self.assertSequenceEqual(os.get_exec_path({'PATH': b'abc'}),
462 ['abc'])
463
464 @unittest.skipUnless(os.supports_bytes_environ,
465 "os.environb required for this test.")
Victor Stinner84ae1182010-05-06 22:05:07 +0000466 def test_environb(self):
467 # os.environ -> os.environb
468 value = 'euro\u20ac'
469 try:
Benjamin Peterson180799d2010-05-06 22:25:42 +0000470 value_bytes = value.encode(sys.getfilesystemencoding(),
471 'surrogateescape')
Victor Stinner84ae1182010-05-06 22:05:07 +0000472 except UnicodeEncodeError:
Benjamin Peterson180799d2010-05-06 22:25:42 +0000473 msg = "U+20AC character is not encodable to %s" % (
474 sys.getfilesystemencoding(),)
Benjamin Peterson932d3f42010-05-06 22:26:31 +0000475 self.skipTest(msg)
Victor Stinner84ae1182010-05-06 22:05:07 +0000476 os.environ['unicode'] = value
477 self.assertEquals(os.environ['unicode'], value)
478 self.assertEquals(os.environb[b'unicode'], value_bytes)
479
480 # os.environb -> os.environ
481 value = b'\xff'
482 os.environb[b'bytes'] = value
483 self.assertEquals(os.environb[b'bytes'], value)
484 value_str = value.decode(sys.getfilesystemencoding(), 'surrogateescape')
485 self.assertEquals(os.environ['bytes'], value_str)
Ezio Melotti19e4acf2010-02-22 15:59:01 +0000486
Tim Petersc4e09402003-04-25 07:11:48 +0000487class WalkTests(unittest.TestCase):
488 """Tests for os.walk()."""
489
490 def test_traversal(self):
491 import os
492 from os.path import join
493
494 # Build:
Guido van Rossumd8faa362007-04-27 19:54:29 +0000495 # TESTFN/
496 # TEST1/ a file kid and two directory kids
Tim Petersc4e09402003-04-25 07:11:48 +0000497 # tmp1
498 # SUB1/ a file kid and a directory kid
Guido van Rossumd8faa362007-04-27 19:54:29 +0000499 # tmp2
500 # SUB11/ no kids
501 # SUB2/ a file kid and a dirsymlink kid
502 # tmp3
503 # link/ a symlink to TESTFN.2
504 # TEST2/
505 # tmp4 a lone file
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000506 walk_path = join(support.TESTFN, "TEST1")
Guido van Rossumd8faa362007-04-27 19:54:29 +0000507 sub1_path = join(walk_path, "SUB1")
Tim Petersc4e09402003-04-25 07:11:48 +0000508 sub11_path = join(sub1_path, "SUB11")
Guido van Rossumd8faa362007-04-27 19:54:29 +0000509 sub2_path = join(walk_path, "SUB2")
510 tmp1_path = join(walk_path, "tmp1")
Tim Petersc4e09402003-04-25 07:11:48 +0000511 tmp2_path = join(sub1_path, "tmp2")
512 tmp3_path = join(sub2_path, "tmp3")
Guido van Rossumd8faa362007-04-27 19:54:29 +0000513 link_path = join(sub2_path, "link")
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000514 t2_path = join(support.TESTFN, "TEST2")
515 tmp4_path = join(support.TESTFN, "TEST2", "tmp4")
Tim Petersc4e09402003-04-25 07:11:48 +0000516
517 # Create stuff.
518 os.makedirs(sub11_path)
519 os.makedirs(sub2_path)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000520 os.makedirs(t2_path)
521 for path in tmp1_path, tmp2_path, tmp3_path, tmp4_path:
Alex Martelli01c77c62006-08-24 02:58:11 +0000522 f = open(path, "w")
Tim Petersc4e09402003-04-25 07:11:48 +0000523 f.write("I'm " + path + " and proud of it. Blame test_os.\n")
524 f.close()
Brian Curtind40e6f72010-07-08 21:39:08 +0000525 if support.can_symlink():
Guido van Rossumd8faa362007-04-27 19:54:29 +0000526 os.symlink(os.path.abspath(t2_path), link_path)
527 sub2_tree = (sub2_path, ["link"], ["tmp3"])
528 else:
529 sub2_tree = (sub2_path, [], ["tmp3"])
Tim Petersc4e09402003-04-25 07:11:48 +0000530
531 # Walk top-down.
Guido van Rossumd8faa362007-04-27 19:54:29 +0000532 all = list(os.walk(walk_path))
Tim Petersc4e09402003-04-25 07:11:48 +0000533 self.assertEqual(len(all), 4)
534 # We can't know which order SUB1 and SUB2 will appear in.
535 # Not flipped: TESTFN, SUB1, SUB11, SUB2
536 # flipped: TESTFN, SUB2, SUB1, SUB11
537 flipped = all[0][1][0] != "SUB1"
538 all[0][1].sort()
Guido van Rossumd8faa362007-04-27 19:54:29 +0000539 self.assertEqual(all[0], (walk_path, ["SUB1", "SUB2"], ["tmp1"]))
Tim Petersc4e09402003-04-25 07:11:48 +0000540 self.assertEqual(all[1 + flipped], (sub1_path, ["SUB11"], ["tmp2"]))
541 self.assertEqual(all[2 + flipped], (sub11_path, [], []))
Guido van Rossumd8faa362007-04-27 19:54:29 +0000542 self.assertEqual(all[3 - 2 * flipped], sub2_tree)
Tim Petersc4e09402003-04-25 07:11:48 +0000543
544 # Prune the search.
545 all = []
Guido van Rossumd8faa362007-04-27 19:54:29 +0000546 for root, dirs, files in os.walk(walk_path):
Tim Petersc4e09402003-04-25 07:11:48 +0000547 all.append((root, dirs, files))
548 # Don't descend into SUB1.
549 if 'SUB1' in dirs:
550 # Note that this also mutates the dirs we appended to all!
551 dirs.remove('SUB1')
552 self.assertEqual(len(all), 2)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000553 self.assertEqual(all[0], (walk_path, ["SUB2"], ["tmp1"]))
554 self.assertEqual(all[1], sub2_tree)
Tim Petersc4e09402003-04-25 07:11:48 +0000555
556 # Walk bottom-up.
Guido van Rossumd8faa362007-04-27 19:54:29 +0000557 all = list(os.walk(walk_path, topdown=False))
Tim Petersc4e09402003-04-25 07:11:48 +0000558 self.assertEqual(len(all), 4)
559 # We can't know which order SUB1 and SUB2 will appear in.
560 # Not flipped: SUB11, SUB1, SUB2, TESTFN
561 # flipped: SUB2, SUB11, SUB1, TESTFN
562 flipped = all[3][1][0] != "SUB1"
563 all[3][1].sort()
Guido van Rossumd8faa362007-04-27 19:54:29 +0000564 self.assertEqual(all[3], (walk_path, ["SUB1", "SUB2"], ["tmp1"]))
Tim Petersc4e09402003-04-25 07:11:48 +0000565 self.assertEqual(all[flipped], (sub11_path, [], []))
566 self.assertEqual(all[flipped + 1], (sub1_path, ["SUB11"], ["tmp2"]))
Guido van Rossumd8faa362007-04-27 19:54:29 +0000567 self.assertEqual(all[2 - 2 * flipped], sub2_tree)
Tim Petersc4e09402003-04-25 07:11:48 +0000568
Brian Curtind40e6f72010-07-08 21:39:08 +0000569 if support.can_symlink():
Guido van Rossumd8faa362007-04-27 19:54:29 +0000570 # Walk, following symlinks.
571 for root, dirs, files in os.walk(walk_path, followlinks=True):
572 if root == link_path:
573 self.assertEqual(dirs, [])
574 self.assertEqual(files, ["tmp4"])
575 break
576 else:
577 self.fail("Didn't follow symlink with followlinks=True")
578
579 def tearDown(self):
Tim Petersc4e09402003-04-25 07:11:48 +0000580 # Tear everything down. This is a decent use for bottom-up on
581 # Windows, which doesn't have a recursive delete command. The
582 # (not so) subtlety is that rmdir will fail unless the dir's
583 # kids are removed first, so bottom up is essential.
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000584 for root, dirs, files in os.walk(support.TESTFN, topdown=False):
Tim Petersc4e09402003-04-25 07:11:48 +0000585 for name in files:
Guido van Rossumd8faa362007-04-27 19:54:29 +0000586 os.remove(os.path.join(root, name))
Tim Petersc4e09402003-04-25 07:11:48 +0000587 for name in dirs:
Guido van Rossumd8faa362007-04-27 19:54:29 +0000588 dirname = os.path.join(root, name)
589 if not os.path.islink(dirname):
590 os.rmdir(dirname)
591 else:
592 os.remove(dirname)
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000593 os.rmdir(support.TESTFN)
Tim Petersc4e09402003-04-25 07:11:48 +0000594
Guido van Rossume7ba4952007-06-06 23:52:48 +0000595class MakedirTests(unittest.TestCase):
Andrew M. Kuchlingb386f6a2003-12-23 16:36:11 +0000596 def setUp(self):
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000597 os.mkdir(support.TESTFN)
Andrew M. Kuchlingb386f6a2003-12-23 16:36:11 +0000598
599 def test_makedir(self):
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000600 base = support.TESTFN
Andrew M. Kuchlingb386f6a2003-12-23 16:36:11 +0000601 path = os.path.join(base, 'dir1', 'dir2', 'dir3')
602 os.makedirs(path) # Should work
603 path = os.path.join(base, 'dir1', 'dir2', 'dir3', 'dir4')
604 os.makedirs(path)
605
606 # Try paths with a '.' in them
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000607 self.assertRaises(OSError, os.makedirs, os.curdir)
Andrew M. Kuchlingb386f6a2003-12-23 16:36:11 +0000608 path = os.path.join(base, 'dir1', 'dir2', 'dir3', 'dir4', 'dir5', os.curdir)
609 os.makedirs(path)
610 path = os.path.join(base, 'dir1', os.curdir, 'dir2', 'dir3', 'dir4',
611 'dir5', 'dir6')
612 os.makedirs(path)
Andrew M. Kuchlingb386f6a2003-12-23 16:36:11 +0000613
Andrew M. Kuchlingb386f6a2003-12-23 16:36:11 +0000614 def tearDown(self):
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000615 path = os.path.join(support.TESTFN, 'dir1', 'dir2', 'dir3',
Andrew M. Kuchlingb386f6a2003-12-23 16:36:11 +0000616 'dir4', 'dir5', 'dir6')
617 # If the tests failed, the bottom-most directory ('../dir6')
618 # may not have been created, so we look for the outermost directory
619 # that exists.
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000620 while not os.path.exists(path) and path != support.TESTFN:
Andrew M. Kuchlingb386f6a2003-12-23 16:36:11 +0000621 path = os.path.dirname(path)
622
623 os.removedirs(path)
624
Guido van Rossume7ba4952007-06-06 23:52:48 +0000625class DevNullTests(unittest.TestCase):
Martin v. Löwisbdec50f2004-06-08 08:29:33 +0000626 def test_devnull(self):
Alex Martelli01c77c62006-08-24 02:58:11 +0000627 f = open(os.devnull, 'w')
Martin v. Löwisbdec50f2004-06-08 08:29:33 +0000628 f.write('hello')
629 f.close()
Alex Martelli01c77c62006-08-24 02:58:11 +0000630 f = open(os.devnull, 'r')
Tim Peters4182cfd2004-06-08 20:34:34 +0000631 self.assertEqual(f.read(), '')
Martin v. Löwisbdec50f2004-06-08 08:29:33 +0000632 f.close()
Andrew M. Kuchlingb386f6a2003-12-23 16:36:11 +0000633
Guido van Rossume7ba4952007-06-06 23:52:48 +0000634class URandomTests(unittest.TestCase):
Martin v. Löwisdc3883f2004-08-29 15:46:35 +0000635 def test_urandom(self):
636 try:
637 self.assertEqual(len(os.urandom(1)), 1)
638 self.assertEqual(len(os.urandom(10)), 10)
639 self.assertEqual(len(os.urandom(100)), 100)
640 self.assertEqual(len(os.urandom(1000)), 1000)
641 except NotImplementedError:
642 pass
643
Victor Stinnerc2d095f2010-05-17 00:14:53 +0000644@contextlib.contextmanager
645def _execvpe_mockup(defpath=None):
646 """
647 Stubs out execv and execve functions when used as context manager.
648 Records exec calls. The mock execv and execve functions always raise an
649 exception as they would normally never return.
650 """
651 # A list of tuples containing (function name, first arg, args)
652 # of calls to execv or execve that have been made.
653 calls = []
654
655 def mock_execv(name, *args):
656 calls.append(('execv', name, args))
657 raise RuntimeError("execv called")
658
659 def mock_execve(name, *args):
660 calls.append(('execve', name, args))
661 raise OSError(errno.ENOTDIR, "execve called")
662
663 try:
664 orig_execv = os.execv
665 orig_execve = os.execve
666 orig_defpath = os.defpath
667 os.execv = mock_execv
668 os.execve = mock_execve
669 if defpath is not None:
670 os.defpath = defpath
671 yield calls
672 finally:
673 os.execv = orig_execv
674 os.execve = orig_execve
675 os.defpath = orig_defpath
676
Guido van Rossume7ba4952007-06-06 23:52:48 +0000677class ExecTests(unittest.TestCase):
Mark Dickinson7cf03892010-04-16 13:45:35 +0000678 @unittest.skipIf(USING_LINUXTHREADS,
679 "avoid triggering a linuxthreads bug: see issue #4970")
Guido van Rossume7ba4952007-06-06 23:52:48 +0000680 def test_execvpe_with_bad_program(self):
Mark Dickinson7cf03892010-04-16 13:45:35 +0000681 self.assertRaises(OSError, os.execvpe, 'no such app-',
682 ['no such app-'], None)
Guido van Rossume7ba4952007-06-06 23:52:48 +0000683
Thomas Heller6790d602007-08-30 17:15:14 +0000684 def test_execvpe_with_bad_arglist(self):
685 self.assertRaises(ValueError, os.execvpe, 'notepad', [], None)
686
Gregory P. Smith4ae37772010-05-08 18:05:46 +0000687 @unittest.skipUnless(hasattr(os, '_execvpe'),
688 "No internal os._execvpe function to test.")
Victor Stinnerb745a742010-05-18 17:17:23 +0000689 def _test_internal_execvpe(self, test_type):
690 program_path = os.sep + 'absolutepath'
691 if test_type is bytes:
692 program = b'executable'
693 fullpath = os.path.join(os.fsencode(program_path), program)
694 native_fullpath = fullpath
695 arguments = [b'progname', 'arg1', 'arg2']
696 else:
697 program = 'executable'
698 arguments = ['progname', 'arg1', 'arg2']
699 fullpath = os.path.join(program_path, program)
700 if os.name != "nt":
701 native_fullpath = os.fsencode(fullpath)
702 else:
703 native_fullpath = fullpath
Victor Stinnerc2d095f2010-05-17 00:14:53 +0000704 env = {'spam': 'beans'}
705
Victor Stinnerb745a742010-05-18 17:17:23 +0000706 # test os._execvpe() with an absolute path
Victor Stinnerc2d095f2010-05-17 00:14:53 +0000707 with _execvpe_mockup() as calls:
Victor Stinnerb745a742010-05-18 17:17:23 +0000708 self.assertRaises(RuntimeError,
709 os._execvpe, fullpath, arguments)
Victor Stinnerc2d095f2010-05-17 00:14:53 +0000710 self.assertEqual(len(calls), 1)
711 self.assertEqual(calls[0], ('execv', fullpath, (arguments,)))
712
Victor Stinnerb745a742010-05-18 17:17:23 +0000713 # test os._execvpe() with a relative path:
714 # os.get_exec_path() returns defpath
Victor Stinnerc2d095f2010-05-17 00:14:53 +0000715 with _execvpe_mockup(defpath=program_path) as calls:
Victor Stinnerb745a742010-05-18 17:17:23 +0000716 self.assertRaises(OSError,
717 os._execvpe, program, arguments, env=env)
Victor Stinnerc2d095f2010-05-17 00:14:53 +0000718 self.assertEqual(len(calls), 1)
Victor Stinnerb745a742010-05-18 17:17:23 +0000719 self.assertSequenceEqual(calls[0],
720 ('execve', native_fullpath, (arguments, env)))
721
722 # test os._execvpe() with a relative path:
723 # os.get_exec_path() reads the 'PATH' variable
724 with _execvpe_mockup() as calls:
725 env_path = env.copy()
726 env_path['PATH'] = program_path
727 self.assertRaises(OSError,
728 os._execvpe, program, arguments, env=env_path)
729 self.assertEqual(len(calls), 1)
730 self.assertSequenceEqual(calls[0],
731 ('execve', native_fullpath, (arguments, env_path)))
732
733 def test_internal_execvpe_str(self):
734 self._test_internal_execvpe(str)
735 if os.name != "nt":
736 self._test_internal_execvpe(bytes)
Victor Stinnerc2d095f2010-05-17 00:14:53 +0000737
Gregory P. Smith4ae37772010-05-08 18:05:46 +0000738
Thomas Wouters477c8d52006-05-27 19:21:47 +0000739class Win32ErrorTests(unittest.TestCase):
740 def test_rename(self):
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000741 self.assertRaises(WindowsError, os.rename, support.TESTFN, support.TESTFN+".bak")
Thomas Wouters477c8d52006-05-27 19:21:47 +0000742
743 def test_remove(self):
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000744 self.assertRaises(WindowsError, os.remove, support.TESTFN)
Thomas Wouters477c8d52006-05-27 19:21:47 +0000745
746 def test_chdir(self):
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000747 self.assertRaises(WindowsError, os.chdir, support.TESTFN)
Thomas Wouters477c8d52006-05-27 19:21:47 +0000748
749 def test_mkdir(self):
Amaury Forgeot d'Arc2fc224f2009-02-19 23:23:47 +0000750 f = open(support.TESTFN, "w")
Benjamin Petersonf91df042009-02-13 02:50:59 +0000751 try:
752 self.assertRaises(WindowsError, os.mkdir, support.TESTFN)
753 finally:
754 f.close()
Amaury Forgeot d'Arc2fc224f2009-02-19 23:23:47 +0000755 os.unlink(support.TESTFN)
Thomas Wouters477c8d52006-05-27 19:21:47 +0000756
757 def test_utime(self):
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000758 self.assertRaises(WindowsError, os.utime, support.TESTFN, None)
Thomas Wouters477c8d52006-05-27 19:21:47 +0000759
Thomas Wouters477c8d52006-05-27 19:21:47 +0000760 def test_chmod(self):
Benjamin Petersonf91df042009-02-13 02:50:59 +0000761 self.assertRaises(WindowsError, os.chmod, support.TESTFN, 0)
Thomas Wouters477c8d52006-05-27 19:21:47 +0000762
Benjamin Petersone1cdfd72009-01-18 21:02:37 +0000763class TestInvalidFD(unittest.TestCase):
Benjamin Peterson05e782f2009-01-19 15:15:02 +0000764 singles = ["fchdir", "dup", "fdopen", "fdatasync", "fstat",
Benjamin Petersone1cdfd72009-01-18 21:02:37 +0000765 "fstatvfs", "fsync", "tcgetpgrp", "ttyname"]
766 #singles.append("close")
767 #We omit close because it doesn'r raise an exception on some platforms
768 def get_single(f):
769 def helper(self):
Benjamin Peterson7522c742009-01-19 21:00:09 +0000770 if hasattr(os, f):
771 self.check(getattr(os, f))
Benjamin Petersone1cdfd72009-01-18 21:02:37 +0000772 return helper
773 for f in singles:
774 locals()["test_"+f] = get_single(f)
775
Benjamin Peterson7522c742009-01-19 21:00:09 +0000776 def check(self, f, *args):
Benjamin Peterson5c6d7872009-02-06 02:40:07 +0000777 try:
778 f(support.make_bad_fd(), *args)
779 except OSError as e:
780 self.assertEqual(e.errno, errno.EBADF)
781 else:
782 self.fail("%r didn't raise a OSError with a bad file descriptor"
783 % f)
Benjamin Peterson7522c742009-01-19 21:00:09 +0000784
Benjamin Petersone1cdfd72009-01-18 21:02:37 +0000785 def test_isatty(self):
786 if hasattr(os, "isatty"):
Benjamin Peterson7522c742009-01-19 21:00:09 +0000787 self.assertEqual(os.isatty(support.make_bad_fd()), False)
Benjamin Petersone1cdfd72009-01-18 21:02:37 +0000788
789 def test_closerange(self):
790 if hasattr(os, "closerange"):
Benjamin Peterson7522c742009-01-19 21:00:09 +0000791 fd = support.make_bad_fd()
R. David Murray630cc482009-07-22 15:20:27 +0000792 # Make sure none of the descriptors we are about to close are
793 # currently valid (issue 6542).
794 for i in range(10):
795 try: os.fstat(fd+i)
796 except OSError:
797 pass
798 else:
799 break
800 if i < 2:
801 raise unittest.SkipTest(
802 "Unable to acquire a range of invalid file descriptors")
803 self.assertEqual(os.closerange(fd, fd + i-1), None)
Benjamin Petersone1cdfd72009-01-18 21:02:37 +0000804
805 def test_dup2(self):
806 if hasattr(os, "dup2"):
Benjamin Peterson7522c742009-01-19 21:00:09 +0000807 self.check(os.dup2, 20)
Benjamin Petersone1cdfd72009-01-18 21:02:37 +0000808
809 def test_fchmod(self):
810 if hasattr(os, "fchmod"):
Benjamin Peterson7522c742009-01-19 21:00:09 +0000811 self.check(os.fchmod, 0)
Benjamin Petersone1cdfd72009-01-18 21:02:37 +0000812
813 def test_fchown(self):
814 if hasattr(os, "fchown"):
Benjamin Peterson7522c742009-01-19 21:00:09 +0000815 self.check(os.fchown, -1, -1)
Benjamin Petersone1cdfd72009-01-18 21:02:37 +0000816
817 def test_fpathconf(self):
818 if hasattr(os, "fpathconf"):
Benjamin Peterson7522c742009-01-19 21:00:09 +0000819 self.check(os.fpathconf, "PC_NAME_MAX")
Benjamin Petersone1cdfd72009-01-18 21:02:37 +0000820
Benjamin Petersone1cdfd72009-01-18 21:02:37 +0000821 def test_ftruncate(self):
822 if hasattr(os, "ftruncate"):
Benjamin Peterson7522c742009-01-19 21:00:09 +0000823 self.check(os.ftruncate, 0)
Benjamin Petersone1cdfd72009-01-18 21:02:37 +0000824
825 def test_lseek(self):
826 if hasattr(os, "lseek"):
Benjamin Peterson7522c742009-01-19 21:00:09 +0000827 self.check(os.lseek, 0, 0)
Benjamin Petersone1cdfd72009-01-18 21:02:37 +0000828
829 def test_read(self):
830 if hasattr(os, "read"):
Benjamin Peterson7522c742009-01-19 21:00:09 +0000831 self.check(os.read, 1)
Benjamin Petersone1cdfd72009-01-18 21:02:37 +0000832
833 def test_tcsetpgrpt(self):
834 if hasattr(os, "tcsetpgrp"):
Benjamin Peterson7522c742009-01-19 21:00:09 +0000835 self.check(os.tcsetpgrp, 0)
Benjamin Petersone1cdfd72009-01-18 21:02:37 +0000836
837 def test_write(self):
838 if hasattr(os, "write"):
Benjamin Peterson7522c742009-01-19 21:00:09 +0000839 self.check(os.write, b" ")
Benjamin Petersone1cdfd72009-01-18 21:02:37 +0000840
Thomas Wouters477c8d52006-05-27 19:21:47 +0000841if sys.platform != 'win32':
842 class Win32ErrorTests(unittest.TestCase):
843 pass
844
Benjamin Petersonef3e4c22009-04-11 19:48:14 +0000845 class PosixUidGidTests(unittest.TestCase):
846 if hasattr(os, 'setuid'):
847 def test_setuid(self):
848 if os.getuid() != 0:
849 self.assertRaises(os.error, os.setuid, 0)
850 self.assertRaises(OverflowError, os.setuid, 1<<32)
851
852 if hasattr(os, 'setgid'):
853 def test_setgid(self):
854 if os.getuid() != 0:
855 self.assertRaises(os.error, os.setgid, 0)
856 self.assertRaises(OverflowError, os.setgid, 1<<32)
857
858 if hasattr(os, 'seteuid'):
859 def test_seteuid(self):
860 if os.getuid() != 0:
861 self.assertRaises(os.error, os.seteuid, 0)
862 self.assertRaises(OverflowError, os.seteuid, 1<<32)
863
864 if hasattr(os, 'setegid'):
865 def test_setegid(self):
866 if os.getuid() != 0:
867 self.assertRaises(os.error, os.setegid, 0)
868 self.assertRaises(OverflowError, os.setegid, 1<<32)
869
870 if hasattr(os, 'setreuid'):
871 def test_setreuid(self):
872 if os.getuid() != 0:
873 self.assertRaises(os.error, os.setreuid, 0, 0)
874 self.assertRaises(OverflowError, os.setreuid, 1<<32, 0)
875 self.assertRaises(OverflowError, os.setreuid, 0, 1<<32)
Benjamin Petersonebe87ba2010-03-06 20:34:24 +0000876
877 def test_setreuid_neg1(self):
878 # Needs to accept -1. We run this in a subprocess to avoid
879 # altering the test runner's process state (issue8045).
Benjamin Petersonebe87ba2010-03-06 20:34:24 +0000880 subprocess.check_call([
881 sys.executable, '-c',
882 'import os,sys;os.setreuid(-1,-1);sys.exit(0)'])
Benjamin Petersonef3e4c22009-04-11 19:48:14 +0000883
884 if hasattr(os, 'setregid'):
885 def test_setregid(self):
886 if os.getuid() != 0:
887 self.assertRaises(os.error, os.setregid, 0, 0)
888 self.assertRaises(OverflowError, os.setregid, 1<<32, 0)
889 self.assertRaises(OverflowError, os.setregid, 0, 1<<32)
Benjamin Petersonebe87ba2010-03-06 20:34:24 +0000890
891 def test_setregid_neg1(self):
892 # Needs to accept -1. We run this in a subprocess to avoid
893 # altering the test runner's process state (issue8045).
Benjamin Petersonebe87ba2010-03-06 20:34:24 +0000894 subprocess.check_call([
895 sys.executable, '-c',
896 'import os,sys;os.setregid(-1,-1);sys.exit(0)'])
Martin v. Löwis011e8422009-05-05 04:43:17 +0000897
898 class Pep383Tests(unittest.TestCase):
Martin v. Löwis011e8422009-05-05 04:43:17 +0000899 def setUp(self):
Victor Stinnerd91df1a2010-08-18 10:56:19 +0000900 if support.TESTFN_UNENCODABLE:
901 self.dir = support.TESTFN_UNENCODABLE
902 else:
903 self.dir = support.TESTFN
904 self.bdir = os.fsencode(self.dir)
905
906 bytesfn = []
907 def add_filename(fn):
908 try:
909 fn = os.fsencode(fn)
910 except UnicodeEncodeError:
911 return
912 bytesfn.append(fn)
913 add_filename(support.TESTFN_UNICODE)
914 if support.TESTFN_UNENCODABLE:
915 add_filename(support.TESTFN_UNENCODABLE)
916 if not bytesfn:
917 self.skipTest("couldn't create any non-ascii filename")
918
919 self.unicodefn = set()
Martin v. Löwis011e8422009-05-05 04:43:17 +0000920 os.mkdir(self.dir)
Victor Stinnerd91df1a2010-08-18 10:56:19 +0000921 try:
922 for fn in bytesfn:
923 f = open(os.path.join(self.bdir, fn), "w")
924 f.close()
Victor Stinnere8d51452010-08-19 01:05:19 +0000925 fn = os.fsdecode(fn)
Victor Stinnerd91df1a2010-08-18 10:56:19 +0000926 if fn in self.unicodefn:
927 raise ValueError("duplicate filename")
928 self.unicodefn.add(fn)
929 except:
930 shutil.rmtree(self.dir)
931 raise
Martin v. Löwis011e8422009-05-05 04:43:17 +0000932
933 def tearDown(self):
934 shutil.rmtree(self.dir)
Martin v. Löwis011e8422009-05-05 04:43:17 +0000935
936 def test_listdir(self):
Victor Stinnerd91df1a2010-08-18 10:56:19 +0000937 expected = self.unicodefn
938 found = set(os.listdir(self.dir))
Martin v. Löwis011e8422009-05-05 04:43:17 +0000939 self.assertEquals(found, expected)
940
941 def test_open(self):
942 for fn in self.unicodefn:
943 f = open(os.path.join(self.dir, fn))
944 f.close()
945
946 def test_stat(self):
947 for fn in self.unicodefn:
948 os.stat(os.path.join(self.dir, fn))
Benjamin Petersonef3e4c22009-04-11 19:48:14 +0000949else:
950 class PosixUidGidTests(unittest.TestCase):
951 pass
Martin v. Löwis011e8422009-05-05 04:43:17 +0000952 class Pep383Tests(unittest.TestCase):
953 pass
Benjamin Petersonef3e4c22009-04-11 19:48:14 +0000954
Brian Curtineb24d742010-04-12 17:16:38 +0000955@unittest.skipUnless(sys.platform == "win32", "Win32 specific tests")
956class Win32KillTests(unittest.TestCase):
Brian Curtinc3acbc32010-05-28 16:08:40 +0000957 def _kill(self, sig):
958 # Start sys.executable as a subprocess and communicate from the
959 # subprocess to the parent that the interpreter is ready. When it
960 # becomes ready, send *sig* via os.kill to the subprocess and check
961 # that the return code is equal to *sig*.
962 import ctypes
963 from ctypes import wintypes
964 import msvcrt
965
966 # Since we can't access the contents of the process' stdout until the
967 # process has exited, use PeekNamedPipe to see what's inside stdout
968 # without waiting. This is done so we can tell that the interpreter
969 # is started and running at a point where it could handle a signal.
970 PeekNamedPipe = ctypes.windll.kernel32.PeekNamedPipe
971 PeekNamedPipe.restype = wintypes.BOOL
972 PeekNamedPipe.argtypes = (wintypes.HANDLE, # Pipe handle
973 ctypes.POINTER(ctypes.c_char), # stdout buf
974 wintypes.DWORD, # Buffer size
975 ctypes.POINTER(wintypes.DWORD), # bytes read
976 ctypes.POINTER(wintypes.DWORD), # bytes avail
977 ctypes.POINTER(wintypes.DWORD)) # bytes left
978 msg = "running"
979 proc = subprocess.Popen([sys.executable, "-c",
980 "import sys;"
981 "sys.stdout.write('{}');"
982 "sys.stdout.flush();"
983 "input()".format(msg)],
984 stdout=subprocess.PIPE,
985 stderr=subprocess.PIPE,
986 stdin=subprocess.PIPE)
987
988 count, max = 0, 100
989 while count < max and proc.poll() is None:
990 # Create a string buffer to store the result of stdout from the pipe
991 buf = ctypes.create_string_buffer(len(msg))
992 # Obtain the text currently in proc.stdout
993 # Bytes read/avail/left are left as NULL and unused
994 rslt = PeekNamedPipe(msvcrt.get_osfhandle(proc.stdout.fileno()),
995 buf, ctypes.sizeof(buf), None, None, None)
996 self.assertNotEqual(rslt, 0, "PeekNamedPipe failed")
997 if buf.value:
998 self.assertEqual(msg, buf.value.decode())
999 break
1000 time.sleep(0.1)
1001 count += 1
1002 else:
1003 self.fail("Did not receive communication from the subprocess")
1004
Brian Curtineb24d742010-04-12 17:16:38 +00001005 os.kill(proc.pid, sig)
1006 self.assertEqual(proc.wait(), sig)
1007
1008 def test_kill_sigterm(self):
1009 # SIGTERM doesn't mean anything special, but make sure it works
Brian Curtinc3acbc32010-05-28 16:08:40 +00001010 self._kill(signal.SIGTERM)
Brian Curtineb24d742010-04-12 17:16:38 +00001011
1012 def test_kill_int(self):
1013 # os.kill on Windows can take an int which gets set as the exit code
Brian Curtinc3acbc32010-05-28 16:08:40 +00001014 self._kill(100)
Brian Curtineb24d742010-04-12 17:16:38 +00001015
1016 def _kill_with_event(self, event, name):
1017 # Run a script which has console control handling enabled.
1018 proc = subprocess.Popen([sys.executable,
1019 os.path.join(os.path.dirname(__file__),
1020 "win_console_handler.py")],
1021 creationflags=subprocess.CREATE_NEW_PROCESS_GROUP)
1022 # Let the interpreter startup before we send signals. See #3137.
1023 time.sleep(0.5)
1024 os.kill(proc.pid, event)
1025 # proc.send_signal(event) could also be done here.
1026 # Allow time for the signal to be passed and the process to exit.
1027 time.sleep(0.5)
1028 if not proc.poll():
1029 # Forcefully kill the process if we weren't able to signal it.
1030 os.kill(proc.pid, signal.SIGINT)
1031 self.fail("subprocess did not stop on {}".format(name))
1032
1033 @unittest.skip("subprocesses aren't inheriting CTRL+C property")
1034 def test_CTRL_C_EVENT(self):
1035 from ctypes import wintypes
1036 import ctypes
1037
1038 # Make a NULL value by creating a pointer with no argument.
1039 NULL = ctypes.POINTER(ctypes.c_int)()
1040 SetConsoleCtrlHandler = ctypes.windll.kernel32.SetConsoleCtrlHandler
1041 SetConsoleCtrlHandler.argtypes = (ctypes.POINTER(ctypes.c_int),
1042 wintypes.BOOL)
1043 SetConsoleCtrlHandler.restype = wintypes.BOOL
1044
1045 # Calling this with NULL and FALSE causes the calling process to
1046 # handle CTRL+C, rather than ignore it. This property is inherited
1047 # by subprocesses.
1048 SetConsoleCtrlHandler(NULL, 0)
1049
1050 self._kill_with_event(signal.CTRL_C_EVENT, "CTRL_C_EVENT")
1051
1052 def test_CTRL_BREAK_EVENT(self):
1053 self._kill_with_event(signal.CTRL_BREAK_EVENT, "CTRL_BREAK_EVENT")
1054
1055
Brian Curtind40e6f72010-07-08 21:39:08 +00001056def skipUnlessWindows6(test):
Brian Curtin74e45612010-07-09 15:58:59 +00001057 if (hasattr(sys, 'getwindowsversion')
1058 and sys.getwindowsversion().major >= 6):
Brian Curtind40e6f72010-07-08 21:39:08 +00001059 return test
1060 return unittest.skip("Requires Windows Vista or later")(test)
1061
1062@unittest.skipUnless(sys.platform == "win32", "Win32 specific tests")
1063@support.skip_unless_symlink
1064class Win32SymlinkTests(unittest.TestCase):
1065 filelink = 'filelinktest'
1066 filelink_target = os.path.abspath(__file__)
1067 dirlink = 'dirlinktest'
1068 dirlink_target = os.path.dirname(filelink_target)
1069 missing_link = 'missing link'
1070
1071 def setUp(self):
1072 assert os.path.exists(self.dirlink_target)
1073 assert os.path.exists(self.filelink_target)
1074 assert not os.path.exists(self.dirlink)
1075 assert not os.path.exists(self.filelink)
1076 assert not os.path.exists(self.missing_link)
1077
1078 def tearDown(self):
1079 if os.path.exists(self.filelink):
1080 os.remove(self.filelink)
1081 if os.path.exists(self.dirlink):
1082 os.rmdir(self.dirlink)
1083 if os.path.lexists(self.missing_link):
1084 os.remove(self.missing_link)
1085
1086 def test_directory_link(self):
1087 os.symlink(self.dirlink_target, self.dirlink)
1088 self.assertTrue(os.path.exists(self.dirlink))
1089 self.assertTrue(os.path.isdir(self.dirlink))
1090 self.assertTrue(os.path.islink(self.dirlink))
1091 self.check_stat(self.dirlink, self.dirlink_target)
1092
1093 def test_file_link(self):
1094 os.symlink(self.filelink_target, self.filelink)
1095 self.assertTrue(os.path.exists(self.filelink))
1096 self.assertTrue(os.path.isfile(self.filelink))
1097 self.assertTrue(os.path.islink(self.filelink))
1098 self.check_stat(self.filelink, self.filelink_target)
1099
1100 def _create_missing_dir_link(self):
1101 'Create a "directory" link to a non-existent target'
1102 linkname = self.missing_link
1103 if os.path.lexists(linkname):
1104 os.remove(linkname)
1105 target = r'c:\\target does not exist.29r3c740'
1106 assert not os.path.exists(target)
1107 target_is_dir = True
1108 os.symlink(target, linkname, target_is_dir)
1109
1110 def test_remove_directory_link_to_missing_target(self):
1111 self._create_missing_dir_link()
1112 # For compatibility with Unix, os.remove will check the
1113 # directory status and call RemoveDirectory if the symlink
1114 # was created with target_is_dir==True.
1115 os.remove(self.missing_link)
1116
1117 @unittest.skip("currently fails; consider for improvement")
1118 def test_isdir_on_directory_link_to_missing_target(self):
1119 self._create_missing_dir_link()
1120 # consider having isdir return true for directory links
1121 self.assertTrue(os.path.isdir(self.missing_link))
1122
1123 @unittest.skip("currently fails; consider for improvement")
1124 def test_rmdir_on_directory_link_to_missing_target(self):
1125 self._create_missing_dir_link()
1126 # consider allowing rmdir to remove directory links
1127 os.rmdir(self.missing_link)
1128
1129 def check_stat(self, link, target):
1130 self.assertEqual(os.stat(link), os.stat(target))
1131 self.assertNotEqual(os.lstat(link), os.stat(link))
1132
1133
Victor Stinnere8d51452010-08-19 01:05:19 +00001134class FSEncodingTests(unittest.TestCase):
1135 def test_nop(self):
1136 self.assertEquals(os.fsencode(b'abc\xff'), b'abc\xff')
1137 self.assertEquals(os.fsdecode('abc\u0141'), 'abc\u0141')
Benjamin Peterson31191a92010-05-09 03:22:58 +00001138
Victor Stinnere8d51452010-08-19 01:05:19 +00001139 def test_identity(self):
1140 # assert fsdecode(fsencode(x)) == x
1141 for fn in ('unicode\u0141', 'latin\xe9', 'ascii'):
1142 try:
1143 bytesfn = os.fsencode(fn)
1144 except UnicodeEncodeError:
1145 continue
1146 self.assertEquals(os.fsdecode(bytesfn), fn)
1147
1148 def get_output(self, fs_encoding, func):
1149 env = os.environ.copy()
1150 env['PYTHONIOENCODING'] = 'utf-8'
1151 env['PYTHONFSENCODING'] = fs_encoding
1152 code = 'import os; print(%s, end="")' % func
1153 process = subprocess.Popen(
1154 [sys.executable, "-c", code],
1155 stdout=subprocess.PIPE, env=env)
1156 stdout, stderr = process.communicate()
1157 self.assertEqual(process.returncode, 0)
1158 return stdout.decode('utf-8')
1159
1160 def test_encodings(self):
1161 def check(encoding, bytesfn, unicodefn):
1162 encoded = self.get_output(encoding, 'repr(os.fsencode(%a))' % unicodefn)
1163 self.assertEqual(encoded, repr(bytesfn))
1164
1165 decoded = self.get_output(encoding, 'repr(os.fsdecode(%a))' % bytesfn)
1166 self.assertEqual(decoded, repr(unicodefn))
1167
1168 check('ascii', b'abc\xff', 'abc\udcff')
1169 check('utf-8', b'\xc3\xa9\x80', '\xe9\udc80')
1170 check('iso-8859-15', b'\xef\xa4', '\xef\u20ac')
Victor Stinnerbf9bcab2010-05-09 03:15:33 +00001171
1172
Fred Drake2e2be372001-09-20 21:33:42 +00001173def test_main():
Benjamin Petersonee8712c2008-05-20 21:35:26 +00001174 support.run_unittest(
Thomas Wouters0e3f5912006-08-11 14:57:12 +00001175 FileTests,
Walter Dörwald21d3a322003-05-01 17:45:56 +00001176 StatAttributeTests,
1177 EnvironTests,
Andrew M. Kuchlingb386f6a2003-12-23 16:36:11 +00001178 WalkTests,
1179 MakedirTests,
Martin v. Löwisbdec50f2004-06-08 08:29:33 +00001180 DevNullTests,
Thomas Wouters477c8d52006-05-27 19:21:47 +00001181 URandomTests,
Guido van Rossume7ba4952007-06-06 23:52:48 +00001182 ExecTests,
Benjamin Petersone1cdfd72009-01-18 21:02:37 +00001183 Win32ErrorTests,
Benjamin Petersonef3e4c22009-04-11 19:48:14 +00001184 TestInvalidFD,
Martin v. Löwis011e8422009-05-05 04:43:17 +00001185 PosixUidGidTests,
Brian Curtineb24d742010-04-12 17:16:38 +00001186 Pep383Tests,
Victor Stinnerbf9bcab2010-05-09 03:15:33 +00001187 Win32KillTests,
Brian Curtind40e6f72010-07-08 21:39:08 +00001188 Win32SymlinkTests,
Victor Stinnere8d51452010-08-19 01:05:19 +00001189 FSEncodingTests,
Walter Dörwald21d3a322003-05-01 17:45:56 +00001190 )
Fred Drake2e2be372001-09-20 21:33:42 +00001191
1192if __name__ == "__main__":
1193 test_main()