Как я могу получить маску разрешения файла?
Как я могу получить маску разрешения файла, например 644 или 755 на *nix, используя python? Есть ли какая-либо функция или класс для этого? Не могли бы вы, ребята, помочь мне? Большое спасибо!
6 ответов
os.stat
является оберткой для интерфейса системных вызовов stat(2)
>>> import os
>>> from stat import *
>>> os.stat("test.txt") # returns 10-tupel, you really want the 0th element ...
posix.stat_result(st_mode=33188, st_ino=57197013, \
st_dev=234881026L, st_nlink=1, st_uid=501, st_gid=20, st_size=0, \
st_atime=1300354697, st_mtime=1300354697, st_ctime=1300354697)
>>> os.stat("test.txt")[ST_MODE] # this is an int, but we like octal ...
33188
>>> oct(os.stat("test.txt")[ST_MODE])
'0100644'
Отсюда вы узнаете типичные восьмеричные разрешения.
S_IRWXU 00700 mask for file owner permissions
S_IRUSR 00400 owner has read permission
S_IWUSR 00200 owner has write permission
S_IXUSR 00100 owner has execute permission
S_IRWXG 00070 mask for group permissions
S_IRGRP 00040 group has read permission
S_IWGRP 00020 group has write permission
S_IXGRP 00010 group has execute permission
S_IRWXO 00007 mask for permissions for others (not in group)
S_IROTH 00004 others have read permission
S_IWOTH 00002 others have write permission
S_IXOTH 00001 others have execute permission
Вас действительно интересуют только младшие биты, поэтому вы можете отрубить все остальное:
>>> oct(os.stat("test.txt")[ST_MODE])[-3:]
'644'
>>> # or better
>>> oct(os.stat("test.txt").st_mode & 0o777)
Sidenote: верхние части определяют тип файла, например:
S_IFMT 0170000 bitmask for the file type bitfields
S_IFSOCK 0140000 socket
S_IFLNK 0120000 symbolic link
S_IFREG 0100000 regular file
S_IFBLK 0060000 block device
S_IFDIR 0040000 directory
S_IFCHR 0020000 character device
S_IFIFO 0010000 FIFO
S_ISUID 0004000 set UID bit
S_ISGID 0002000 set-group-ID bit (see below)
S_ISVTX 0001000 sticky bit (see below)
Я думаю, что это самый простой способ получить файл с битами разрешения:
stat.S_IMODE(os.lstat("file").st_mode)
Функция os.lstat, в случае, если файл является символической ссылкой, предоставляет вам режим самой ссылки, тогда как os.stat разыменовывает ссылку. Поэтому я считаю os.lstat наиболее полезным.
Вот пример случая с заданным обычным файлом "testfile" и символической ссылкой на последний "testlink":
import stat
import os
print oct(stat.S_IMODE(os.lstat("testlink").st_mode))
print oct(stat.S_IMODE(os.stat("testlink").st_mode))
Этот скрипт выводит для меня следующее:
0777
0666
Другой способ сделать это, если вы не хотите выяснить, что означает stat, это использовать команду os.access http://docs.python.org/library/os.html,НО читать документы о возможных проблемы с безопасностью
Например, для проверки прав доступа к файлу test.dat, который имеет права на чтение / запись.
os.access("test.dat",os.R_OK)
>>> True
#Execute permissions
os.access("test.dat",os.X_OK)
>>> False
#And Combinations thereof
os.access("test.dat",os.R_OK or os.X_OK)
>>> True
os.access("test.dat",os.R_OK and os.X_OK)
>>> False
Метод os.access(path, mode) возвращает True, если доступ разрешен по пути, и False, если нет.
Доступные режимы:
- os.F_OK - проверить наличие пути.
- os.R_OK - проверить читаемость пути.
- os.W_OK - проверить возможность записи пути.
- os.X_OK - проверить, можно ли выполнить путь.
например, проверка файла /tmp/test.sh имеет разрешение на выполнение
ls -l /tmp/temp.sh
-rw-r--r-- 1 * * 0 Mar 2 12:05 /tmp/temp.sh
os.access('/tmp/temp.sh',os.X_OK)
False
after changing the file permission to +x
chmod +x /tmp/temp.sh
ls -l /tmp/temp.sh
-rwxr-xr-x 1 * * 0 Mar 2 12:05 /tmp/temp.sh
os.access('/tmp/temp.sh',os.X_OK)
True
Here is a simple way to check the permissions of a directory.
import os
import stat
mode = os.stat("path_of_directory").st_mode
if not ((mode & stat.S_IWUSR):
print ('not writable by user')
if not ((mode & stat.S_IWUSR) and (mode & stat.S_IWGRP) and (mode & stat.S_IWOTH)):
print ('not writable by all')
The flag list is herebelow:
S_IRWXU 00700 mask for file owner permissions
S_IRUSR 00400 owner has read permission
S_IWUSR 00200 owner has write permission
S_IXUSR 00100 owner has execute permission
S_IRWXG 00070 mask for group permissions
S_IRGRP 00040 group has read permission
S_IWGRP 00020 group has write permission
S_IXGRP 00010 group has execute permission
S_IRWXO 00007 mask for permissions for others (not in group)
S_IROTH 00004 others have read permission
S_IWOTH 00002 others have write permission
S_IXOTH 00001 others have execute permission
В модуле os есть много файловых функций. Если вы бежите os.stat(filename)
Вы всегда можете интерпретировать результаты.
os.stat аналогичен статистике c-lib (для просмотра информации смотрите статистику man 2 в linux)
stats = os.stat('file.txt')
print stats.st_mode
Вы можете просто запустить команду Bash stat с Popen, если хотите:
Обычная команда Bash:
jlc@server:~/NetBeansProjects/LineReverse$ stat -c '%A %a %n' revline.c
-rw-rw-r-- 664 revline.c
А потом с Python:
>>> from subprocess import Popen, PIPE
>>> fname = 'revline.c'
>>> cmd = "stat -c '%A %a %n' " + fname
>>> out = Popen(cmd, shell=True, stdout=PIPE).communicate()[0].split()[1].decode()
>>> out
'664'
И вот еще один способ, если вам хочется искать в каталоге:
>>> from os import popen
>>> cmd = "stat -c '%A %a %n' *"
>>> fname = 'revline.c'
>>> for i in popen(cmd):
... p, m, n = i.split()
... if n != fname:
... continue
... print(m)
break
...
664
>>>