blob: 2e1e2c349c8d09dc828a6377a36febe6020c0297 [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
pxinwrb2d0c662020-12-01 16:20:50 +08005from test.support import os_helper
Serhiy Storchaka16994912020-04-25 10:06:29 +03006from test.support import socket_helper
Hai Shi46605972020-08-04 00:49:18 +08007from test.support.import_helper import import_fresh_module
8from test.support.os_helper import TESTFN
9
Christian Heimesc77d9f32013-06-22 21:05:02 +020010
11c_stat = import_fresh_module('stat', fresh=['_stat'])
12py_stat = import_fresh_module('stat', blocked=['_stat'])
Giampaolo Rodola'ffa1d0b2012-05-15 15:30:25 +020013
Antoine Pitrou3a5053b2013-06-29 12:58:57 +020014class TestFilemode:
Christian Heimesc77d9f32013-06-22 21:05:02 +020015 statmod = None
16
Christian Heimesf678b312013-06-21 18:25:56 +020017 file_flags = {'SF_APPEND', 'SF_ARCHIVED', 'SF_IMMUTABLE', 'SF_NOUNLINK',
18 'SF_SNAPSHOT', 'UF_APPEND', 'UF_COMPRESSED', 'UF_HIDDEN',
19 'UF_IMMUTABLE', 'UF_NODUMP', 'UF_NOUNLINK', 'UF_OPAQUE'}
20
21 formats = {'S_IFBLK', 'S_IFCHR', 'S_IFDIR', 'S_IFIFO', 'S_IFLNK',
Ronan Lamy7bb14312019-10-10 09:34:46 +020022 'S_IFREG', 'S_IFSOCK', 'S_IFDOOR', 'S_IFPORT', 'S_IFWHT'}
Christian Heimesf678b312013-06-21 18:25:56 +020023
24 format_funcs = {'S_ISBLK', 'S_ISCHR', 'S_ISDIR', 'S_ISFIFO', 'S_ISLNK',
Ronan Lamy7bb14312019-10-10 09:34:46 +020025 'S_ISREG', 'S_ISSOCK', 'S_ISDOOR', 'S_ISPORT', 'S_ISWHT'}
Christian Heimesf678b312013-06-21 18:25:56 +020026
27 stat_struct = {
28 'ST_MODE': 0,
29 'ST_INO': 1,
30 'ST_DEV': 2,
31 'ST_NLINK': 3,
32 'ST_UID': 4,
33 'ST_GID': 5,
34 'ST_SIZE': 6,
35 'ST_ATIME': 7,
36 'ST_MTIME': 8,
37 'ST_CTIME': 9}
38
39 # permission bit value are defined by POSIX
40 permission_bits = {
41 'S_ISUID': 0o4000,
42 'S_ISGID': 0o2000,
43 'S_ENFMT': 0o2000,
44 'S_ISVTX': 0o1000,
45 'S_IRWXU': 0o700,
46 'S_IRUSR': 0o400,
47 'S_IREAD': 0o400,
48 'S_IWUSR': 0o200,
49 'S_IWRITE': 0o200,
50 'S_IXUSR': 0o100,
51 'S_IEXEC': 0o100,
52 'S_IRWXG': 0o070,
53 'S_IRGRP': 0o040,
54 'S_IWGRP': 0o020,
55 'S_IXGRP': 0o010,
56 'S_IRWXO': 0o007,
57 'S_IROTH': 0o004,
58 'S_IWOTH': 0o002,
59 'S_IXOTH': 0o001}
Giampaolo Rodola'ffa1d0b2012-05-15 15:30:25 +020060
Zachary Ware63f277b2014-06-19 09:46:37 -050061 # defined by the Windows API documentation
62 file_attributes = {
63 'FILE_ATTRIBUTE_ARCHIVE': 32,
64 'FILE_ATTRIBUTE_COMPRESSED': 2048,
65 'FILE_ATTRIBUTE_DEVICE': 64,
66 'FILE_ATTRIBUTE_DIRECTORY': 16,
67 'FILE_ATTRIBUTE_ENCRYPTED': 16384,
68 'FILE_ATTRIBUTE_HIDDEN': 2,
69 'FILE_ATTRIBUTE_INTEGRITY_STREAM': 32768,
70 'FILE_ATTRIBUTE_NORMAL': 128,
71 'FILE_ATTRIBUTE_NOT_CONTENT_INDEXED': 8192,
72 'FILE_ATTRIBUTE_NO_SCRUB_DATA': 131072,
73 'FILE_ATTRIBUTE_OFFLINE': 4096,
74 'FILE_ATTRIBUTE_READONLY': 1,
75 'FILE_ATTRIBUTE_REPARSE_POINT': 1024,
76 'FILE_ATTRIBUTE_SPARSE_FILE': 512,
77 'FILE_ATTRIBUTE_SYSTEM': 4,
78 'FILE_ATTRIBUTE_TEMPORARY': 256,
79 'FILE_ATTRIBUTE_VIRTUAL': 65536}
80
Giampaolo Rodola'ffa1d0b2012-05-15 15:30:25 +020081 def setUp(self):
82 try:
83 os.remove(TESTFN)
84 except OSError:
85 try:
86 os.rmdir(TESTFN)
87 except OSError:
88 pass
89 tearDown = setUp
90
Christian Heimes36a7e4f2013-06-23 16:10:29 +020091 def get_mode(self, fname=TESTFN, lstat=True):
92 if lstat:
93 st_mode = os.lstat(fname).st_mode
94 else:
95 st_mode = os.stat(fname).st_mode
Christian Heimesc77d9f32013-06-22 21:05:02 +020096 modestr = self.statmod.filemode(st_mode)
Christian Heimesf678b312013-06-21 18:25:56 +020097 return st_mode, modestr
98
99 def assertS_IS(self, name, mode):
100 # test format, lstrip is for S_IFIFO
Christian Heimesc77d9f32013-06-22 21:05:02 +0200101 fmt = getattr(self.statmod, "S_IF" + name.lstrip("F"))
102 self.assertEqual(self.statmod.S_IFMT(mode), fmt)
Christian Heimesf678b312013-06-21 18:25:56 +0200103 # test that just one function returns true
104 testname = "S_IS" + name
105 for funcname in self.format_funcs:
Christian Heimesc77d9f32013-06-22 21:05:02 +0200106 func = getattr(self.statmod, funcname, None)
Christian Heimesf678b312013-06-21 18:25:56 +0200107 if func is None:
108 if funcname == testname:
109 raise ValueError(funcname)
110 continue
111 if funcname == testname:
112 self.assertTrue(func(mode))
113 else:
114 self.assertFalse(func(mode))
115
Giampaolo Rodola'ffa1d0b2012-05-15 15:30:25 +0200116 def test_mode(self):
117 with open(TESTFN, 'w'):
118 pass
Giampaolo Rodola'e1266782012-05-16 16:01:59 +0200119 if os.name == 'posix':
120 os.chmod(TESTFN, 0o700)
Christian Heimesf678b312013-06-21 18:25:56 +0200121 st_mode, modestr = self.get_mode()
122 self.assertEqual(modestr, '-rwx------')
123 self.assertS_IS("REG", st_mode)
Christian Heimesc77d9f32013-06-22 21:05:02 +0200124 self.assertEqual(self.statmod.S_IMODE(st_mode),
125 self.statmod.S_IRWXU)
Christian Heimesf678b312013-06-21 18:25:56 +0200126
Giampaolo Rodola'e1266782012-05-16 16:01:59 +0200127 os.chmod(TESTFN, 0o070)
Christian Heimesf678b312013-06-21 18:25:56 +0200128 st_mode, modestr = self.get_mode()
129 self.assertEqual(modestr, '----rwx---')
130 self.assertS_IS("REG", st_mode)
Christian Heimesc77d9f32013-06-22 21:05:02 +0200131 self.assertEqual(self.statmod.S_IMODE(st_mode),
132 self.statmod.S_IRWXG)
Christian Heimesf678b312013-06-21 18:25:56 +0200133
Giampaolo Rodola'e1266782012-05-16 16:01:59 +0200134 os.chmod(TESTFN, 0o007)
Christian Heimesf678b312013-06-21 18:25:56 +0200135 st_mode, modestr = self.get_mode()
136 self.assertEqual(modestr, '-------rwx')
137 self.assertS_IS("REG", st_mode)
Christian Heimesc77d9f32013-06-22 21:05:02 +0200138 self.assertEqual(self.statmod.S_IMODE(st_mode),
139 self.statmod.S_IRWXO)
Christian Heimesf678b312013-06-21 18:25:56 +0200140
Giampaolo Rodola'e1266782012-05-16 16:01:59 +0200141 os.chmod(TESTFN, 0o444)
Christian Heimesf678b312013-06-21 18:25:56 +0200142 st_mode, modestr = self.get_mode()
143 self.assertS_IS("REG", st_mode)
144 self.assertEqual(modestr, '-r--r--r--')
Christian Heimesc77d9f32013-06-22 21:05:02 +0200145 self.assertEqual(self.statmod.S_IMODE(st_mode), 0o444)
Giampaolo Rodola'e1266782012-05-16 16:01:59 +0200146 else:
147 os.chmod(TESTFN, 0o700)
Christian Heimesf678b312013-06-21 18:25:56 +0200148 st_mode, modestr = self.get_mode()
149 self.assertEqual(modestr[:3], '-rw')
150 self.assertS_IS("REG", st_mode)
Christian Heimesc77d9f32013-06-22 21:05:02 +0200151 self.assertEqual(self.statmod.S_IFMT(st_mode),
152 self.statmod.S_IFREG)
Giampaolo Rodola'ffa1d0b2012-05-15 15:30:25 +0200153
154 def test_directory(self):
155 os.mkdir(TESTFN)
156 os.chmod(TESTFN, 0o700)
Christian Heimesf678b312013-06-21 18:25:56 +0200157 st_mode, modestr = self.get_mode()
158 self.assertS_IS("DIR", st_mode)
Giampaolo Rodola'e1266782012-05-16 16:01:59 +0200159 if os.name == 'posix':
Christian Heimesf678b312013-06-21 18:25:56 +0200160 self.assertEqual(modestr, 'drwx------')
Giampaolo Rodola'e1266782012-05-16 16:01:59 +0200161 else:
Christian Heimesf678b312013-06-21 18:25:56 +0200162 self.assertEqual(modestr[0], 'd')
Giampaolo Rodola'ffa1d0b2012-05-15 15:30:25 +0200163
164 @unittest.skipUnless(hasattr(os, 'symlink'), 'os.symlink not available')
165 def test_link(self):
Giampaolo Rodola'e1266782012-05-16 16:01:59 +0200166 try:
167 os.symlink(os.getcwd(), TESTFN)
168 except (OSError, NotImplementedError) as err:
169 raise unittest.SkipTest(str(err))
170 else:
Christian Heimesf678b312013-06-21 18:25:56 +0200171 st_mode, modestr = self.get_mode()
172 self.assertEqual(modestr[0], 'l')
173 self.assertS_IS("LNK", st_mode)
Giampaolo Rodola'ffa1d0b2012-05-15 15:30:25 +0200174
175 @unittest.skipUnless(hasattr(os, 'mkfifo'), 'os.mkfifo not available')
176 def test_fifo(self):
pxinwrb2d0c662020-12-01 16:20:50 +0800177 if sys.platform == "vxworks":
178 fifo_path = os.path.join("/fifos/", TESTFN)
179 else:
180 fifo_path = TESTFN
181 self.addCleanup(os_helper.unlink, fifo_path)
xdegaye92c2ca72017-11-12 17:31:07 +0100182 try:
pxinwrb2d0c662020-12-01 16:20:50 +0800183 os.mkfifo(fifo_path, 0o700)
xdegaye92c2ca72017-11-12 17:31:07 +0100184 except PermissionError as e:
185 self.skipTest('os.mkfifo(): %s' % e)
pxinwrb2d0c662020-12-01 16:20:50 +0800186 st_mode, modestr = self.get_mode(fifo_path)
Christian Heimesf678b312013-06-21 18:25:56 +0200187 self.assertEqual(modestr, 'prwx------')
188 self.assertS_IS("FIFO", st_mode)
189
190 @unittest.skipUnless(os.name == 'posix', 'requires Posix')
191 def test_devices(self):
192 if os.path.exists(os.devnull):
Christian Heimes36a7e4f2013-06-23 16:10:29 +0200193 st_mode, modestr = self.get_mode(os.devnull, lstat=False)
Christian Heimesf678b312013-06-21 18:25:56 +0200194 self.assertEqual(modestr[0], 'c')
195 self.assertS_IS("CHR", st_mode)
Christian Heimes45d94932013-06-22 14:48:32 +0200196 # Linux block devices, BSD has no block devices anymore
Christian Heimes60a95932013-06-21 18:53:13 +0200197 for blockdev in ("/dev/sda", "/dev/hda"):
Christian Heimesf678b312013-06-21 18:25:56 +0200198 if os.path.exists(blockdev):
Christian Heimes36a7e4f2013-06-23 16:10:29 +0200199 st_mode, modestr = self.get_mode(blockdev, lstat=False)
Christian Heimesf678b312013-06-21 18:25:56 +0200200 self.assertEqual(modestr[0], 'b')
201 self.assertS_IS("BLK", st_mode)
202 break
203
Serhiy Storchaka16994912020-04-25 10:06:29 +0300204 @socket_helper.skip_unless_bind_unix_socket
GPeryb92c5262018-08-10 08:12:08 +0300205 def test_socket(self):
206 with socket.socket(socket.AF_UNIX) as s:
207 s.bind(TESTFN)
208 st_mode, modestr = self.get_mode()
209 self.assertEqual(modestr[0], 's')
210 self.assertS_IS("SOCK", st_mode)
211
Christian Heimesf678b312013-06-21 18:25:56 +0200212 def test_module_attributes(self):
213 for key, value in self.stat_struct.items():
Christian Heimesc77d9f32013-06-22 21:05:02 +0200214 modvalue = getattr(self.statmod, key)
Christian Heimesf678b312013-06-21 18:25:56 +0200215 self.assertEqual(value, modvalue, key)
216 for key, value in self.permission_bits.items():
Christian Heimesc77d9f32013-06-22 21:05:02 +0200217 modvalue = getattr(self.statmod, key)
Christian Heimesf678b312013-06-21 18:25:56 +0200218 self.assertEqual(value, modvalue, key)
219 for key in self.file_flags:
Christian Heimesc77d9f32013-06-22 21:05:02 +0200220 modvalue = getattr(self.statmod, key)
Christian Heimesf678b312013-06-21 18:25:56 +0200221 self.assertIsInstance(modvalue, int)
222 for key in self.formats:
Christian Heimesc77d9f32013-06-22 21:05:02 +0200223 modvalue = getattr(self.statmod, key)
Christian Heimesf678b312013-06-21 18:25:56 +0200224 self.assertIsInstance(modvalue, int)
225 for key in self.format_funcs:
Christian Heimesc77d9f32013-06-22 21:05:02 +0200226 func = getattr(self.statmod, key)
Christian Heimesf678b312013-06-21 18:25:56 +0200227 self.assertTrue(callable(func))
228 self.assertEqual(func(0), 0)
Giampaolo Rodola'ffa1d0b2012-05-15 15:30:25 +0200229
Zachary Ware63f277b2014-06-19 09:46:37 -0500230 @unittest.skipUnless(sys.platform == "win32",
231 "FILE_ATTRIBUTE_* constants are Win32 specific")
232 def test_file_attribute_constants(self):
233 for key, value in sorted(self.file_attributes.items()):
234 self.assertTrue(hasattr(self.statmod, key), key)
235 modvalue = getattr(self.statmod, key)
236 self.assertEqual(value, modvalue, key)
237
Giampaolo Rodola'ffa1d0b2012-05-15 15:30:25 +0200238
Antoine Pitrou3a5053b2013-06-29 12:58:57 +0200239class TestFilemodeCStat(TestFilemode, unittest.TestCase):
Christian Heimesc77d9f32013-06-22 21:05:02 +0200240 statmod = c_stat
241
Christian Heimesc77d9f32013-06-22 21:05:02 +0200242
Antoine Pitrou3a5053b2013-06-29 12:58:57 +0200243class TestFilemodePyStat(TestFilemode, unittest.TestCase):
Christian Heimesc77d9f32013-06-22 21:05:02 +0200244 statmod = py_stat
245
246
Giampaolo Rodola'ffa1d0b2012-05-15 15:30:25 +0200247if __name__ == '__main__':
Antoine Pitrou3a5053b2013-06-29 12:58:57 +0200248 unittest.main()