我正在运行一个python脚本demo.py,如下所示:
#!/usr/bin/env python
from subprocess import Popen, PIPE, CalledProcessError
try:
process = Popen(["/root/script.sh"], stdout = PIPE, stderr = PIPE)
process_out, process_err = process.communicate()
return_code = process.returncode
if process_out:
print "output:", process_out
if process_err:
print "error:", process_err
print "return code:", return_code
except CalledProcessError as e:
print "CalledProcessError:", e
except Exception, fault:
print "fault:", fault
script.sh如下:
#!/bin/bash
cd /root/
mkdir foo
cd foo
cat << EOF > bar.txt
random text
EOF
目录foo已经存在。因此,script.sh应该失败并且确实失败了。 demo.py正确捕获process_err中的错误并打印出来:
错误:mkdir:无法创建目录`foo':文件存在
但是return_code的值仍为0 (表示成功运行)。
如果我的script.sh如下:
#!/bin/bash
cd /root/
mkdir foo
process_err打印相同的错误消息但现在return_code的值为1.
问题出在哪里?
还请建议process.returncode采用0和1以外的值的方案。
答案 0 :(得分:2)
这不是Python的问题,而是您的Bash脚本在mkdir
之后继续执行。这就是Bash默认工作的方式,你必须告诉它在遇到错误时退出。
使用:
#!/bin/bash
set -e
或者,如果您无法更改Bash脚本,并且只能更改Python代码:
process = Popen(['bash', '-e', '/root/script.sh'], stdout = PIPE, stderr = PIPE)
来自help set
:
-e
如果命令以非零状态退出,则立即退出。