blob: 17443bed0738b59e7724860f94d47b5d6f90ca01 [file] [log] [blame]
Giampaolo Rodola'ffa1d0b2012-05-15 15:30:25 +02001import unittest
2import os
GPeryb92c5262018-08-10 08:12:08 +03003import socket
Zachary Ware63f277b2014-06-19 09:46:37 -05004import sys
xdegaye4461d702019-05-03 17:09:17 +02005from test.support import (TESTFN, import_fresh_module,
6 skip_unless_bind_unix_socket)
Christian Heimesc77d9f32013-06-22 21:05:02 +02007
8c_stat = import_fresh_module('stat', fresh=['_stat'])
9py_stat = import_fresh_module('stat', blocked=['_stat'])
Giampaolo Rodola'ffa1d0b2012-05-15 15:30:25 +020010
Antoine Pitrou3a5053b2013-06-29 12:58:57 +020011class TestFilemode:
Christian Heimesc77d9f32013-06-22 21:05:02 +020012 statmod = None
13
Christian Heimesf678b312013-06-21 18:25:56 +020014 file_flags = {'SF_APPEND', 'SF_ARCHIVED', 'SF_IMMUTABLE', 'SF_NOUNLINK',
15 'SF_SNAPSHOT', 'UF_APPEND', 'UF_COMPRESSED', 'UF_HIDDEN',
16 'UF_IMMUTABLE', 'UF_NODUMP', 'UF_NOUNLINK', 'UF_OPAQUE'}
17
18 formats = {'S_IFBLK', 'S_IFCHR', 'S_IFDIR', 'S_IFIFO', 'S_IFLNK',
19 'S_IFREG', 'S_IFSOCK'}
20
21 format_funcs = {'S_ISBLK', 'S_ISCHR', 'S_ISDIR', 'S_ISFIFO', 'S_ISLNK',
22 'S_ISREG', 'S_ISSOCK'}
23
24 stat_struct = {
25 'ST_MODE': 0,
26 'ST_INO': 1,
27 'ST_DEV': 2,
28 'ST_NLINK': 3,
29 'ST_UID': 4,
30 'ST_GID': 5,
31 'ST_SIZE': 6,
32 'ST_ATIME': 7,
33 'ST_MTIME': 8,
34 'ST_CTIME': 9}
35
36 # permission bit value are defined by POSIX
37 permission_bits = {
38 'S_ISUID': 0o4000,
39 'S_ISGID': 0o2000,
40 'S_ENFMT': 0o2000,
41 'S_ISVTX': 0o1000,
42 'S_IRWXU': 0o700,
43 'S_IRUSR': 0o400,
44 'S_IREAD': 0o400,
45 'S_IWUSR': 0o200,
46 'S_IWRITE': 0o200,
47 'S_IXUSR': 0o100,
48 'S_IEXEC': 0o100,
49 'S_IRWXG': 0o070,
50 'S_IRGRP': 0o040,
51 'S_IWGRP': 0o020,
52 'S_IXGRP': 0o010,
53 'S_IRWXO': 0o007,
54 'S_IROTH': 0o004,
55 'S_IWOTH': 0o002,
56 'S_IXOTH': 0o001}
Giampaolo Rodola'ffa1d0b2012-05-15 15:30:25 +020057
Zachary Ware63f277b2014-06-19 09:46:37 -050058 # defined by the Windows API documentation
59 file_attributes = {
60 'FILE_ATTRIBUTE_ARCHIVE': 32,
61 'FILE_ATTRIBUTE_COMPRESSED': 2048,
62 'FILE_ATTRIBUTE_DEVICE': 64,
63 'FILE_ATTRIBUTE_DIRECTORY': 16,
64 'FILE_ATTRIBUTE_ENCRYPTED': 16384,
65 'FILE_ATTRIBUTE_HIDDEN': 2,
66 'FILE_ATTRIBUTE_INTEGRITY_STREAM': 32768,
67 'FILE_ATTRIBUTE_NORMAL': 128,
68 'FILE_ATTRIBUTE_NOT_CONTENT_INDEXED': 8192,
69 'FILE_ATTRIBUTE_NO_SCRUB_DATA': 131072,
70 'FILE_ATTRIBUTE_OFFLINE': 4096,
71 'FILE_ATTRIBUTE_READONLY': 1,
72 'FILE_ATTRIBUTE_REPARSE_POINT': 1024,
73 'FILE_ATTRIBUTE_SPARSE_FILE': 512,
74 'FILE_ATTRIBUTE_SYSTEM': 4,
75 'FILE_ATTRIBUTE_TEMPORARY': 256,
76 'FILE_ATTRIBUTE_VIRTUAL': 65536}
77
Giampaolo Rodola'ffa1d0b2012-05-15 15:30:25 +020078 def setUp(self):
79 try:
80 os.remove(TESTFN)
81 except OSError:
82 try:
83 os.rmdir(TESTFN)
84 except OSError:
85 pass
86 tearDown = setUp
87
Christian Heimes36a7e4f2013-06-23 16:10:29 +020088 def get_mode(self, fname=TESTFN, lstat=True):
89 if lstat:
90 st_mode = os.lstat(fname).st_mode
91 else:
92 st_mode = os.stat(fname).st_mode
Christian Heimesc77d9f32013-06-22 21:05:02 +020093 modestr = self.statmod.filemode(st_mode)
Christian Heimesf678b312013-06-21 18:25:56 +020094 return st_mode, modestr
95
96 def assertS_IS(self, name, mode):
97 # test format, lstrip is for S_IFIFO
Christian Heimesc77d9f32013-06-22 21:05:02 +020098 fmt = getattr(self.statmod, "S_IF" + name.lstrip("F"))
99 self.assertEqual(self.statmod.S_IFMT(mode), fmt)
Christian Heimesf678b312013-06-21 18:25:56 +0200100 # test that just one function returns true
101 testname = "S_IS" + name
102 for funcname in self.format_funcs:
Christian Heimesc77d9f32013-06-22 21:05:02 +0200103 func = getattr(self.statmod, funcname, None)
Christian Heimesf678b312013-06-21 18:25:56 +0200104 if func is None:
105 if funcname == testname:
106 raise ValueError(funcname)
107 continue
108 if funcname == testname:
109 self.assertTrue(func(mode))
110 else:
111 self.assertFalse(func(mode))
112
Giampaolo Rodola'ffa1d0b2012-05-15 15:30:25 +0200113 def test_mode(self):
114 with open(TESTFN, 'w'):
115 pass
Giampaolo Rodola'e1266782012-05-16 16:01:59 +0200116 if os.name == 'posix':
117 os.chmod(TESTFN, 0o700)
Christian Heimesf678b312013-06-21 18:25:56 +0200118 st_mode, modestr = self.get_mode()
119 self.assertEqual(modestr, '-rwx------')
120 self.assertS_IS("REG", st_mode)
Christian Heimesc77d9f32013-06-22 21:05:02 +0200121 self.assertEqual(self.statmod.S_IMODE(st_mode),
122 self.statmod.S_IRWXU)
Christian Heimesf678b312013-06-21 18:25:56 +0200123
Giampaolo Rodola'e1266782012-05-16 16:01:59 +0200124 os.chmod(TESTFN, 0o070)
Christian Heimesf678b312013-06-21 18:25:56 +0200125 st_mode, modestr = self.get_mode()
126 self.assertEqual(modestr, '----rwx---')
127 self.assertS_IS("REG", st_mode)
Christian Heimesc77d9f32013-06-22 21:05:02 +0200128 self.assertEqual(self.statmod.S_IMODE(st_mode),
129 self.statmod.S_IRWXG)
Christian Heimesf678b312013-06-21 18:25:56 +0200130
Giampaolo Rodola'e1266782012-05-16 16:01:59 +0200131 os.chmod(TESTFN, 0o007)
Christian Heimesf678b312013-06-21 18:25:56 +0200132 st_mode, modestr = self.get_mode()
133 self.assertEqual(modestr, '-------rwx')
134 self.assertS_IS("REG", st_mode)
Christian Heimesc77d9f32013-06-22 21:05:02 +0200135 self.assertEqual(self.statmod.S_IMODE(st_mode),
136 self.statmod.S_IRWXO)
Christian Heimesf678b312013-06-21 18:25:56 +0200137
Giampaolo Rodola'e1266782012-05-16 16:01:59 +0200138 os.chmod(TESTFN, 0o444)
Christian Heimesf678b312013-06-21 18:25:56 +0200139 st_mode, modestr = self.get_mode()
140 self.assertS_IS("REG", st_mode)
141 self.assertEqual(modestr, '-r--r--r--')
Christian Heimesc77d9f32013-06-22 21:05:02 +0200142 self.assertEqual(self.statmod.S_IMODE(st_mode), 0o444)
Giampaolo Rodola'e1266782012-05-16 16:01:59 +0200143 else:
144 os.chmod(TESTFN, 0o700)
Christian Heimesf678b312013-06-21 18:25:56 +0200145 st_mode, modestr = self.get_mode()
146 self.assertEqual(modestr[:3], '-rw')
147 self.assertS_IS("REG", st_mode)
Christian Heimesc77d9f32013-06-22 21:05:02 +0200148 self.assertEqual(self.statmod.S_IFMT(st_mode),
149 self.statmod.S_IFREG)
Giampaolo Rodola'ffa1d0b2012-05-15 15:30:25 +0200150
151 def test_directory(self):
152 os.mkdir(TESTFN)
153 os.chmod(TESTFN, 0o700)
Christian Heimesf678b312013-06-21 18:25:56 +0200154 st_mode, modestr = self.get_mode()
155 self.assertS_IS("DIR", st_mode)
Giampaolo Rodola'e1266782012-05-16 16:01:59 +0200156 if os.name == 'posix':
Christian Heimesf678b312013-06-21 18:25:56 +0200157 self.assertEqual(modestr, 'drwx------')
Giampaolo Rodola'e1266782012-05-16 16:01:59 +0200158 else:
Christian Heimesf678b312013-06-21 18:25:56 +0200159 self.assertEqual(modestr[0], 'd')
Giampaolo Rodola'ffa1d0b2012-05-15 15:30:25 +0200160
161 @unittest.skipUnless(hasattr(os, 'symlink'), 'os.symlink not available')
162 def test_link(self):
Giampaolo Rodola'e1266782012-05-16 16:01:59 +0200163 try:
164 os.symlink(os.getcwd(), TESTFN)
165 except (OSError, NotImplementedError) as err:
166 raise unittest.SkipTest(str(err))
167 else:
Christian Heimesf678b312013-06-21 18:25:56 +0200168 st_mode, modestr = self.get_mode()
169 self.assertEqual(modestr[0], 'l')
170 self.assertS_IS("LNK", st_mode)
Giampaolo Rodola'ffa1d0b2012-05-15 15:30:25 +0200171
172 @unittest.skipUnless(hasattr(os, 'mkfifo'), 'os.mkfifo not available')
173 def test_fifo(self):
xdegaye92c2ca72017-11-12 17:31:07 +0100174 try:
175 os.mkfifo(TESTFN, 0o700)
176 except PermissionError as e:
177 self.skipTest('os.mkfifo(): %s' % e)
Christian Heimesf678b312013-06-21 18:25:56 +0200178 st_mode, modestr = self.get_mode()
179 self.assertEqual(modestr, 'prwx------')
180 self.assertS_IS("FIFO", st_mode)
181
182 @unittest.skipUnless(os.name == 'posix', 'requires Posix')
183 def test_devices(self):
184 if os.path.exists(os.devnull):
Christian Heimes36a7e4f2013-06-23 16:10:29 +0200185 st_mode, modestr = self.get_mode(os.devnull, lstat=False)
Christian Heimesf678b312013-06-21 18:25:56 +0200186 self.assertEqual(modestr[0], 'c')
187 self.assertS_IS("CHR", st_mode)
Christian Heimes45d94932013-06-22 14:48:32 +0200188 # Linux block devices, BSD has no block devices anymore
Christian Heimes60a95932013-06-21 18:53:13 +0200189 for blockdev in ("/dev/sda", "/dev/hda"):
Christian Heimesf678b312013-06-21 18:25:56 +0200190 if os.path.exists(blockdev):
Christian Heimes36a7e4f2013-06-23 16:10:29 +0200191 st_mode, modestr = self.get_mode(blockdev, lstat=False)
Christian Heimesf678b312013-06-21 18:25:56 +0200192 self.assertEqual(modestr[0], 'b')
193 self.assertS_IS("BLK", st_mode)
194 break
195
xdegaye4461d702019-05-03 17:09:17 +0200196 @skip_unless_bind_unix_socket
GPeryb92c5262018-08-10 08:12:08 +0300197 def test_socket(self):
198 with socket.socket(socket.AF_UNIX) as s:
199 s.bind(TESTFN)
200 st_mode, modestr = self.get_mode()
201 self.assertEqual(modestr[0], 's')
202 self.assertS_IS("SOCK", st_mode)
203
Christian Heimesf678b312013-06-21 18:25:56 +0200204 def test_module_attributes(self):
205 for key, value in self.stat_struct.items():
Christian Heimesc77d9f32013-06-22 21:05:02 +0200206 modvalue = getattr(self.statmod, key)
Christian Heimesf678b312013-06-21 18:25:56 +0200207 self.assertEqual(value, modvalue, key)
208 for key, value in self.permission_bits.items():
Christian Heimesc77d9f32013-06-22 21:05:02 +0200209 modvalue = getattr(self.statmod, key)
Christian Heimesf678b312013-06-21 18:25:56 +0200210 self.assertEqual(value, modvalue, key)
211 for key in self.file_flags:
Christian Heimesc77d9f32013-06-22 21:05:02 +0200212 modvalue = getattr(self.statmod, key)
Christian Heimesf678b312013-06-21 18:25:56 +0200213 self.assertIsInstance(modvalue, int)
214 for key in self.formats:
Christian Heimesc77d9f32013-06-22 21:05:02 +0200215 modvalue = getattr(self.statmod, key)
Christian Heimesf678b312013-06-21 18:25:56 +0200216 self.assertIsInstance(modvalue, int)
217 for key in self.format_funcs:
Christian Heimesc77d9f32013-06-22 21:05:02 +0200218 func = getattr(self.statmod, key)
Christian Heimesf678b312013-06-21 18:25:56 +0200219 self.assertTrue(callable(func))
220 self.assertEqual(func(0), 0)
Giampaolo Rodola'ffa1d0b2012-05-15 15:30:25 +0200221
Zachary Ware63f277b2014-06-19 09:46:37 -0500222 @unittest.skipUnless(sys.platform == "win32",
223 "FILE_ATTRIBUTE_* constants are Win32 specific")
224 def test_file_attribute_constants(self):
225 for key, value in sorted(self.file_attributes.items()):
226 self.assertTrue(hasattr(self.statmod, key), key)
227 modvalue = getattr(self.statmod, key)
228 self.assertEqual(value, modvalue, key)
229
Giampaolo Rodola'ffa1d0b2012-05-15 15:30:25 +0200230
Antoine Pitrou3a5053b2013-06-29 12:58:57 +0200231class TestFilemodeCStat(TestFilemode, unittest.TestCase):
Christian Heimesc77d9f32013-06-22 21:05:02 +0200232 statmod = c_stat
233
234 formats = TestFilemode.formats | {'S_IFDOOR', 'S_IFPORT', 'S_IFWHT'}
Antoine Pitrou3a5053b2013-06-29 12:58:57 +0200235 format_funcs = TestFilemode.format_funcs | {'S_ISDOOR', 'S_ISPORT',
236 'S_ISWHT'}
Christian Heimesc77d9f32013-06-22 21:05:02 +0200237
238
Antoine Pitrou3a5053b2013-06-29 12:58:57 +0200239class TestFilemodePyStat(TestFilemode, unittest.TestCase):
Christian Heimesc77d9f32013-06-22 21:05:02 +0200240 statmod = py_stat
241
242
Giampaolo Rodola'ffa1d0b2012-05-15 15:30:25 +0200243if __name__ == '__main__':
Antoine Pitrou3a5053b2013-06-29 12:58:57 +0200244 unittest.main()