如何在python cgi脚本中捕获子进程中的错误?

时间:2011-04-13 17:48:42

标签: python apache cgi subprocess

我正在尝试编写一个Python CGI脚本,它将使用子进程调用sox(音频处理程序)。问题是当我从sox调用中得到错误时,一切都崩溃了,我从Apache那里得到了“格式错误的标题”。

相关位:

def downsample(in_file, in_location, sox, out_location):
  """ run sox """
...
  sox = shlex.split(sox)
  retcode = subprocess.check_call(sox)
  if not retcode == 0:
    print '<b>something went wrong in sox: returned error code ' +\
          retcode + ', but we are continuing anyway...</b>'
  """p = subprocess.Popen(sox)
  if p.stdout:
    print '<b>something happened in sox: returned ' +\
          p.stdout.read() + ', but we will keep going...</b>'
  if p.stderr:
    print '<b>something happened in sox: returned ' +\
          p.stderr.read() + ', but we will keep going...</b>'"""
...

def main():
  print "Content-Type: text/html\n\n"

...
    downsample(in_file, in_location, command, out_location)
...

if __name__ == '__main__':
  main()

我正在使用check_call来允许cgi错误处理程序立即打印堆栈跟踪(以避免500页),但我真的想抓住错误,自己处理它,继续使用脚本。我尝试通过在check:中包装check_call来实现这一点:除了CalledProcessError:语句,但这只是再次导致了500页。那部分被注释掉也不适合我。

来自/ var / www / apache2 / error_log:

Wed Apr 13 10:08:21 2011] [error] [client ::1] sox FAIL formats: can't open input file `/tmp/wavs/haha/f.wav': WAVE: RIFF header not found, referer: http://localhost/~Valkyrie_savage/
[Wed Apr 13 10:08:21 2011] [error] [client ::1] malformed header from script. Bad header=\x1f\x8b\b: downsample.py, referer: http://localhost/~Valkyrie_savage/

我无法理解为什么它似乎是在标题打印之前运行sox命令。或者,如果是,为什么说标题格式错误?

3 个答案:

答案 0 :(得分:2)

import subprocess
p = subprocess.Popen(cmd)  
   p.wait()  
   if p.returncode:  
      print "failed with code: %s" % str(p.returncode) 

答案 1 :(得分:1)

听起来你正在使用缓冲输出。

如果是这种情况,那么命令的输出会在之前打印,因此Web浏览器会感到困惑。

您可以在进行系统调用之前调用sys.stdout.flush(),以确保实际打印输出缓冲区中的所有标题和html。

希望这有帮助。

答案 2 :(得分:0)

您可以使用subprocess.check_call并抓住CalledProcessError例外。

try:
    retcode = subprocess.check_call(sox)
except CalledProcessError:
    print '<b>something went wrong in sox: returned error code ' +\
      retcode + ', but we are continuing anyway...</b>'