blob: 83d09e17f93c563c45e2131153dfb445ef2d00aa [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
Serhiy Storchaka16994912020-04-25 10:06:29 +03005from test.support import socket_helper
Hai Shi46605972020-08-04 00:49:18 +08006from test.support.import_helper import import_fresh_module
7from test.support.os_helper import TESTFN
8
Christian Heimesc77d9f32013-06-22 21:05:02 +02009
10c_stat = import_fresh_module('stat', fresh=['_stat'])
11py_stat = import_fresh_module('stat', blocked=['_stat'])
Giampaolo Rodola'ffa1d0b2012-05-15 15:30:25 +020012
Antoine Pitrou3a5053b2013-06-29 12:58:57 +020013class TestFilemode:
Christian Heimesc77d9f32013-06-22 21:05:02 +020014 statmod = None
15
Christian Heimesf678b312013-06-21 18:25:56 +020016 file_flags = {'SF_APPEND', 'SF_ARCHIVED', 'SF_IMMUTABLE', 'SF_NOUNLINK',
17 'SF_SNAPSHOT', 'UF_APPEND', 'UF_COMPRESSED', 'UF_HIDDEN',
18 'UF_IMMUTABLE', 'UF_NODUMP', 'UF_NOUNLINK', 'UF_OPAQUE'}
19
20 formats = {'S_IFBLK', 'S_IFCHR', 'S_IFDIR', 'S_IFIFO', 'S_IFLNK',
Ronan Lamy7bb14312019-10-10 09:34:46 +020021 'S_IFREG', 'S_IFSOCK', 'S_IFDOOR', 'S_IFPORT', 'S_IFWHT'}
Christian Heimesf678b312013-06-21 18:25:56 +020022
23 format_funcs = {'S_ISBLK', 'S_ISCHR', 'S_ISDIR', 'S_ISFIFO', 'S_ISLNK',
Ronan Lamy7bb14312019-10-10 09:34:46 +020024 'S_ISREG', 'S_ISSOCK', 'S_ISDOOR', 'S_ISPORT', 'S_ISWHT'}
Christian Heimesf678b312013-06-21 18:25:56 +020025
26 stat_struct = {
27 'ST_MODE': 0,
28 'ST_INO': 1,
29 'ST_DEV': 2,
30 'ST_NLINK': 3,
31 'ST_UID': 4,
32 'ST_GID': 5,
33 'ST_SIZE': 6,
34 'ST_ATIME': 7,
35 'ST_MTIME': 8,
36 'ST_CTIME': 9}
37
38 # permission bit value are defined by POSIX
39 permission_bits = {
40 'S_ISUID': 0o4000,
41 'S_ISGID': 0o2000,
42 'S_ENFMT': 0o2000,
43 'S_ISVTX': 0o1000,
44 'S_IRWXU': 0o700,
45 'S_IRUSR': 0o400,
46 'S_IREAD': 0o400,
47 'S_IWUSR': 0o200,
48 'S_IWRITE': 0o200,
49 'S_IXUSR': 0o100,
50 'S_IEXEC': 0o100,
51 'S_IRWXG': 0o070,
52 'S_IRGRP': 0o040,
53 'S_IWGRP': 0o020,
54 'S_IXGRP': 0o010,
55 'S_IRWXO': 0o007,
56 'S_IROTH': 0o004,
57 'S_IWOTH': 0o002,
58 'S_IXOTH': 0o001}
Giampaolo Rodola'ffa1d0b2012-05-15 15:30:25 +020059
Zachary Ware63f277b2014-06-19 09:46:37 -050060 # defined by the Windows API documentation
61 file_attributes = {
62 'FILE_ATTRIBUTE_ARCHIVE': 32,
63 'FILE_ATTRIBUTE_COMPRESSED': 2048,
64 'FILE_ATTRIBUTE_DEVICE': 64,
65 'FILE_ATTRIBUTE_DIRECTORY': 16,
66 'FILE_ATTRIBUTE_ENCRYPTED': 16384,
67 'FILE_ATTRIBUTE_HIDDEN': 2,
68 'FILE_ATTRIBUTE_INTEGRITY_STREAM': 32768,
69 'FILE_ATTRIBUTE_NORMAL': 128,
70 'FILE_ATTRIBUTE_NOT_CONTENT_INDEXED': 8192,
71 'FILE_ATTRIBUTE_NO_SCRUB_DATA': 131072,
72 'FILE_ATTRIBUTE_OFFLINE': 4096,
73 'FILE_ATTRIBUTE_READONLY': 1,
74 'FILE_ATTRIBUTE_REPARSE_POINT': 1024,
75 'FILE_ATTRIBUTE_SPARSE_FILE': 512,
76 'FILE_ATTRIBUTE_SYSTEM': 4,
77 'FILE_ATTRIBUTE_TEMPORARY': 256,
78 'FILE_ATTRIBUTE_VIRTUAL': 65536}
79
Giampaolo Rodola'ffa1d0b2012-05-15 15:30:25 +020080 def setUp(self):
81 try:
82 os.remove(TESTFN)
83 except OSError:
84 try:
85 os.rmdir(TESTFN)
86 except OSError:
87 pass
88 tearDown = setUp
89
Christian Heimes36a7e4f2013-06-23 16:10:29 +020090 def get_mode(self, fname=TESTFN, lstat=True):
91 if lstat:
92 st_mode = os.lstat(fname).st_mode
93 else:
94 st_mode = os.stat(fname).st_mode
Christian Heimesc77d9f32013-06-22 21:05:02 +020095 modestr = self.statmod.filemode(st_mode)
Christian Heimesf678b312013-06-21 18:25:56 +020096 return st_mode, modestr
97
98 def assertS_IS(self, name, mode):
99 # test format, lstrip is for S_IFIFO
Christian Heimesc77d9f32013-06-22 21:05:02 +0200100 fmt = getattr(self.statmod, "S_IF" + name.lstrip("F"))
101 self.assertEqual(self.statmod.S_IFMT(mode), fmt)
Christian Heimesf678b312013-06-21 18:25:56 +0200102 # test that just one function returns true
103 testname = "S_IS" + name
104 for funcname in self.format_funcs:
Christian Heimesc77d9f32013-06-22 21:05:02 +0200105 func = getattr(self.statmod, funcname, None)
Christian Heimesf678b312013-06-21 18:25:56 +0200106 if func is None:
107 if funcname == testname:
108 raise ValueError(funcname)
109 continue
110 if funcname == testname:
111 self.assertTrue(func(mode))
112 else:
113 self.assertFalse(func(mode))
114
Giampaolo Rodola'ffa1d0b2012-05-15 15:30:25 +0200115 def test_mode(self):
116 with open(TESTFN, 'w'):
117 pass
Giampaolo Rodola'e1266782012-05-16 16:01:59 +0200118 if os.name == 'posix':
119 os.chmod(TESTFN, 0o700)
Christian Heimesf678b312013-06-21 18:25:56 +0200120 st_mode, modestr = self.get_mode()
121 self.assertEqual(modestr, '-rwx------')
122 self.assertS_IS("REG", st_mode)
Christian Heimesc77d9f32013-06-22 21:05:02 +0200123 self.assertEqual(self.statmod.S_IMODE(st_mode),
124 self.statmod.S_IRWXU)
Christian Heimesf678b312013-06-21 18:25:56 +0200125
Giampaolo Rodola'e1266782012-05-16 16:01:59 +0200126 os.chmod(TESTFN, 0o070)
Christian Heimesf678b312013-06-21 18:25:56 +0200127 st_mode, modestr = self.get_mode()
128 self.assertEqual(modestr, '----rwx---')
129 self.assertS_IS("REG", st_mode)
Christian Heimesc77d9f32013-06-22 21:05:02 +0200130 self.assertEqual(self.statmod.S_IMODE(st_mode),
131 self.statmod.S_IRWXG)
Christian Heimesf678b312013-06-21 18:25:56 +0200132
Giampaolo Rodola'e1266782012-05-16 16:01:59 +0200133 os.chmod(TESTFN, 0o007)
Christian Heimesf678b312013-06-21 18:25:56 +0200134 st_mode, modestr = self.get_mode()
135 self.assertEqual(modestr, '-------rwx')
136 self.assertS_IS("REG", st_mode)
Christian Heimesc77d9f32013-06-22 21:05:02 +0200137 self.assertEqual(self.statmod.S_IMODE(st_mode),
138 self.statmod.S_IRWXO)
Christian Heimesf678b312013-06-21 18:25:56 +0200139
Giampaolo Rodola'e1266782012-05-16 16:01:59 +0200140 os.chmod(TESTFN, 0o444)
Christian Heimesf678b312013-06-21 18:25:56 +0200141 st_mode, modestr = self.get_mode()
142 self.assertS_IS("REG", st_mode)
143 self.assertEqual(modestr, '-r--r--r--')
Christian Heimesc77d9f32013-06-22 21:05:02 +0200144 self.assertEqual(self.statmod.S_IMODE(st_mode), 0o444)
Giampaolo Rodola'e1266782012-05-16 16:01:59 +0200145 else:
146 os.chmod(TESTFN, 0o700)
Christian Heimesf678b312013-06-21 18:25:56 +0200147 st_mode, modestr = self.get_mode()
148 self.assertEqual(modestr[:3], '-rw')
149 self.assertS_IS("REG", st_mode)
Christian Heimesc77d9f32013-06-22 21:05:02 +0200150 self.assertEqual(self.statmod.S_IFMT(st_mode),
151 self.statmod.S_IFREG)
Giampaolo Rodola'ffa1d0b2012-05-15 15:30:25 +0200152
153 def test_directory(self):
154 os.mkdir(TESTFN)
155 os.chmod(TESTFN, 0o700)
Christian Heimesf678b312013-06-21 18:25:56 +0200156 st_mode, modestr = self.get_mode()
157 self.assertS_IS("DIR", st_mode)
Giampaolo Rodola'e1266782012-05-16 16:01:59 +0200158 if os.name == 'posix':
Christian Heimesf678b312013-06-21 18:25:56 +0200159 self.assertEqual(modestr, 'drwx------')
Giampaolo Rodola'e1266782012-05-16 16:01:59 +0200160 else:
Christian Heimesf678b312013-06-21 18:25:56 +0200161 self.assertEqual(modestr[0], 'd')
Giampaolo Rodola'ffa1d0b2012-05-15 15:30:25 +0200162
163 @unittest.skipUnless(hasattr(os, 'symlink'), 'os.symlink not available')
164 def test_link(self):
Giampaolo Rodola'e1266782012-05-16 16:01:59 +0200165 try:
166 os.symlink(os.getcwd(), TESTFN)
167 except (OSError, NotImplementedError) as err:
168 raise unittest.SkipTest(str(err))
169 else:
Christian Heimesf678b312013-06-21 18:25:56 +0200170 st_mode, modestr = self.get_mode()
171 self.assertEqual(modestr[0], 'l')
172 self.assertS_IS("LNK", st_mode)
Giampaolo Rodola'ffa1d0b2012-05-15 15:30:25 +0200173
174 @unittest.skipUnless(hasattr(os, 'mkfifo'), 'os.mkfifo not available')
175 def test_fifo(self):
xdegaye92c2ca72017-11-12 17:31:07 +0100176 try:
177 os.mkfifo(TESTFN, 0o700)
178 except PermissionError as e:
179 self.skipTest('os.mkfifo(): %s' % e)
Christian Heimesf678b312013-06-21 18:25:56 +0200180 st_mode, modestr = self.get_mode()
181 self.assertEqual(modestr, 'prwx------')
182 self.assertS_IS("FIFO", st_mode)
183
184 @unittest.skipUnless(os.name == 'posix', 'requires Posix')
185 def test_devices(self):
186 if os.path.exists(os.devnull):
Christian Heimes36a7e4f2013-06-23 16:10:29 +0200187 st_mode, modestr = self.get_mode(os.devnull, lstat=False)
Christian Heimesf678b312013-06-21 18:25:56 +0200188 self.assertEqual(modestr[0], 'c')
189 self.assertS_IS("CHR", st_mode)
Christian Heimes45d94932013-06-22 14:48:32 +0200190 # Linux block devices, BSD has no block devices anymore
Christian Heimes60a95932013-06-21 18:53:13 +0200191 for blockdev in ("/dev/sda", "/dev/hda"):
Christian Heimesf678b312013-06-21 18:25:56 +0200192 if os.path.exists(blockdev):
Christian Heimes36a7e4f2013-06-23 16:10:29 +0200193 st_mode, modestr = self.get_mode(blockdev, lstat=False)
Christian Heimesf678b312013-06-21 18:25:56 +0200194 self.assertEqual(modestr[0], 'b')
195 self.assertS_IS("BLK", st_mode)
196 break
197
Serhiy Storchaka16994912020-04-25 10:06:29 +0300198 @socket_helper.skip_unless_bind_unix_socket
GPeryb92c5262018-08-10 08:12:08 +0300199 def test_socket(self):
200 with socket.socket(socket.AF_UNIX) as s:
201 s.bind(TESTFN)
202 st_mode, modestr = self.get_mode()
203 self.assertEqual(modestr[0], 's')
204 self.assertS_IS("SOCK", st_mode)
205
Christian Heimesf678b312013-06-21 18:25:56 +0200206 def test_module_attributes(self):
207 for key, value in self.stat_struct.items():
Christian Heimesc77d9f32013-06-22 21:05:02 +0200208 modvalue = getattr(self.statmod, key)
Christian Heimesf678b312013-06-21 18:25:56 +0200209 self.assertEqual(value, modvalue, key)
210 for key, value in self.permission_bits.items():
Christian Heimesc77d9f32013-06-22 21:05:02 +0200211 modvalue = getattr(self.statmod, key)
Christian Heimesf678b312013-06-21 18:25:56 +0200212 self.assertEqual(value, modvalue, key)
213 for key in self.file_flags:
Christian Heimesc77d9f32013-06-22 21:05:02 +0200214 modvalue = getattr(self.statmod, key)
Christian Heimesf678b312013-06-21 18:25:56 +0200215 self.assertIsInstance(modvalue, int)
216 for key in self.formats:
Christian Heimesc77d9f32013-06-22 21:05:02 +0200217 modvalue = getattr(self.statmod, key)
Christian Heimesf678b312013-06-21 18:25:56 +0200218 self.assertIsInstance(modvalue, int)
219 for key in self.format_funcs:
Christian Heimesc77d9f32013-06-22 21:05:02 +0200220 func = getattr(self.statmod, key)
Christian Heimesf678b312013-06-21 18:25:56 +0200221 self.assertTrue(callable(func))
222 self.assertEqual(func(0), 0)
Giampaolo Rodola'ffa1d0b2012-05-15 15:30:25 +0200223
Zachary Ware63f277b2014-06-19 09:46:37 -0500224 @unittest.skipUnless(sys.platform == "win32",
225 "FILE_ATTRIBUTE_* constants are Win32 specific")
226 def test_file_attribute_constants(self):
227 for key, value in sorted(self.file_attributes.items()):
228 self.assertTrue(hasattr(self.statmod, key), key)
229 modvalue = getattr(self.statmod, key)
230 self.assertEqual(value, modvalue, key)
231
Giampaolo Rodola'ffa1d0b2012-05-15 15:30:25 +0200232
Antoine Pitrou3a5053b2013-06-29 12:58:57 +0200233class TestFilemodeCStat(TestFilemode, unittest.TestCase):
Christian Heimesc77d9f32013-06-22 21:05:02 +0200234 statmod = c_stat
235
Christian Heimesc77d9f32013-06-22 21:05:02 +0200236
Antoine Pitrou3a5053b2013-06-29 12:58:57 +0200237class TestFilemodePyStat(TestFilemode, unittest.TestCase):
Christian Heimesc77d9f32013-06-22 21:05:02 +0200238 statmod = py_stat
239
240
Giampaolo Rodola'ffa1d0b2012-05-15 15:30:25 +0200241if __name__ == '__main__':
Antoine Pitrou3a5053b2013-06-29 12:58:57 +0200242 unittest.main()