问题:您如何在python中执行简单的“ chmod + x”操作?
我想从可执行的python脚本中创建文件。
import os
import stat
os.chmod('somefile', stat.S_IEXEC)
它似乎os.chmod
没有像unix chmod
那样“添加”权限。在最后一行注释掉的情况下,文件具有filemode -rw-r--r--
,而在未注释掉的情况下,文件模式为---x------
。如何u+x
在保持其余模式不变的同时添加标志?
回答 0
使用os.stat()
得到当前的权限,使用|
或位在一起,并使用os.chmod()
设置更新的权限。
例:
import os
import stat
st = os.stat('somefile')
os.chmod('somefile', st.st_mode | stat.S_IEXEC)
回答 1
对于生成可执行文件的工具(例如脚本),以下代码可能会有所帮助:
def make_executable(path):
mode = os.stat(path).st_mode
mode |= (mode & 0o444) >> 2 # copy R bits to X
os.chmod(path, mode)
这使它(或多或少)尊重umask
创建文件时的效果:仅为可读取的文件设置可执行文件。
用法:
path = 'foo.sh'
with open(path, 'w') as f: # umask in effect when file is created
f.write('#!/bin/sh\n')
f.write('echo "hello world"\n')
make_executable(path)
回答 2
如果知道所需的权限,则可以使用以下示例来简化操作。
Python 2:
os.chmod("/somedir/somefile", 0775)
Python 3:
os.chmod("/somedir/somefile", 0o775)
兼容(八进制转换):
os.chmod("/somedir/somefile", 509)
参考权限示例
回答 3
您也可以这样做
>>> import os
>>> st = os.stat("hello.txt")
当前文件清单
$ ls -l hello.txt
-rw-r--r-- 1 morrison staff 17 Jan 13 2014 hello.txt
现在做。
>>> os.chmod("hello.txt", st.st_mode | 0o111)
然后您将在终端中看到此内容。
ls -l hello.txt
-rwxr-xr-x 1 morrison staff 17 Jan 13 2014 hello.txt
您可以按位或0o111使所有可执行文件,0o222使所有可写,和0o444使所有可读。
回答 4
尊重umask
喜欢chmod +x
man chmod
说如果augo
没有给出,如:
chmod +x mypath
然后a
使用,但与umask
:
字母ugoa的组合控制将更改哪些用户对该文件的访问权限:拥有该文件的用户(u),该文件组中的其他用户(g),不在该文件组中的其他用户(o)或全部用户(a)。如果没有给出这些,则效果就好像给出了(a)一样,但是不影响umask中设置的位。
这是一个完全模拟该行为的版本:
#!/usr/bin/env python3
import os
import stat
def get_umask():
umask = os.umask(0)
os.umask(umask)
return umask
def chmod_plus_x(path):
os.chmod(
path,
os.stat(path).st_mode |
(
(
stat.S_IXUSR |
stat.S_IXGRP |
stat.S_IXOTH
)
& ~get_umask()
)
)
chmod_plus_x('.gitignore')
另请参阅:如何获取Python中的默认文件权限?
已在Ubuntu 16.04,Python 3.5.2中进行了测试。
回答 5
在python3中:
import os
os.chmod("somefile", 0o664)
请记住要添加0o
前缀,因为权限设置为八进制整数,Python会自动将前导零的任何整数视为八进制。否则,您os.chmod("somefile", 1230)
的确通过了,这是的八进制664
。
回答 6
如果您使用的是Python 3.4+,则可以使用标准库的便捷pathlib。
from pathlib import Path
f = Path("/path/to/file.txt")
f.chmod(f.stat().st_mode | stat.S_IEXEC)