语言:Python v2.6.2
操作系统:AIX 5.3
我正在使用Python将一些文件从备份恢复到测试系统 - 所有命令都以下面的方式调用,但是有些命令只是简单的不想工作。
#!/usr/bin/python
import subprocess, shlex
cmd = 'sudo rm -rf /work/TEST/*'
arg = shlex.split(cmd)
# This does not work
p = subprocess.Popen(arg)
# This, however, works just fine
p = subprocess.Popen(cmd, shell=True)
如果我从命令中删除*,它们可以正常工作(好吧,它们可以正常工作而不使用通配符,这很不幸地不是我想要的)。
出于明显的安全原因,我真的不想使用shell = True,但是有一些其他命令基本上做同样的事情。如果命令中有通配符,它将无法正常工作 - 它会毫无错误地执行,只是不执行任何操作。
有趣的是以下命令(通过shlex解析):
sudo mv /work/testrestore/production/* /work/TESTC
产生以下内容:
mv:0653-401无法将/ work / testrestore / production / *重命名为/ work / TESTC / *:路径名中的文件或目录不存在。
就好像unix现在正在尝试移动名为*的文件,而不是使用*作为通配符。这是shlex的典型行为吗?
编辑:我尝试使用\来转义*,也尝试从单引号转换为双引号。不是我希望它能做任何事情。
答案 0 :(得分:5)
要将*
替换为含义,您需要shell或者需要glob
模块。所以最简单的方法是shell=True
(如果命令是常量,我看不到任何安全漏洞)。
另一种方法是
#!/usr/bin/python
import subprocess
import shlex
import glob
cmd = 'sudo rm -rf /work/TEST/*'
arg = shlex.split(cmd)
arg = arg[:-1] + glob.glob(arg[-1])
# This should work now
p = subprocess.Popen(arg)
或者,如果您仍然自己追加路径,
cmd = 'sudo rm -rf'
basearg = shlex.split(cmd)
arg = basearg + glob.glob(path+"/*")