如何使用 python 2.7.6 使 subprocess.call 超时?
Posted
技术标签:
【中文标题】如何使用 python 2.7.6 使 subprocess.call 超时?【英文标题】:How to make a subprocess.call timeout using python 2.7.6? 【发布时间】:2014-06-10 20:20:22 【问题描述】:这可能已经被问过了,但我在使用 python 2.7 时找不到任何关于 subprocess.call 超时的信息
【问题讨论】:
看看:***.com/questions/1191374/subprocess-with-timeout 【参考方案1】:您可以使用subprocess32mentioned by @gps,它是 Python 3.2 - 3.5 的子进程标准库模块的反向移植,可用于 Python 2。
首先,安装 subprocess32 模块:
pip install subprocess32
这是一个代码sn-p:
>>> import subprocess32
>>> print subprocess32.check_output(["python", "--version"])
Python 2.7.12
>>> subprocess32.check_output(["sleep", "infinity"], timeout=3)
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "/usr/local/lib/python2.7/dist-packages/subprocess32.py", line 340, in check_output
raise TimeoutExpired(process.args, timeout, output=output)
subprocess32.TimeoutExpired: Command '['sleep', 'infinity']' timed out after 3 seconds
注意,默认timeout=None
,表示永不超时。
【讨论】:
【参考方案2】:我一直使用 2.7 超时的一种简单方法是使用 subprocess.poll()
和 time.sleep()
延迟。这是一个非常基本的例子:
import subprocess
import time
x = #some amount of seconds
delay = 1.0
timeout = int(x / delay)
args = #a string or array of arguments
task = subprocess.Popen(args)
#while the process is still executing and we haven't timed-out yet
while task.poll() is None and timeout > 0:
#do other things too if necessary e.g. print, check resources, etc.
time.sleep(delay)
timeout -= delay
如果您设置x = 600
,那么您的超时将达到 10 分钟。而task.poll()
会查询进程是否已经终止。 time.sleep(delay)
在这种情况下会休眠 1 秒,然后将超时时间减少 1 秒。您可以随心所欲地玩弄这部分,但基本概念始终相同。
希望这会有所帮助!
subprocess.poll()
https://docs.python.org/2/library/subprocess.html#popen-objects
【讨论】:
这不会终止进程。你需要添加 os.killpg(os.getpgid(task.pid), signal.SIGTERM) @AaronS 我认为task.terminate()
的作用几乎相同。【参考方案3】:
您可以安装 subprocess32
module mentioned by @gps -- Python 3.2/3.3 中 subprocess
模块的反向端口,用于 2.x。它适用于 Python 2.7,并且包含来自 Python 3.3 的超时支持。
subprocess.call()
is just Popen().wait()
从而在timeout
秒内中断一个长时间运行的进程:
#!/usr/bin/env python
import time
from subprocess import Popen
p = Popen(*call_args)
time.sleep(timeout)
try:
p.kill()
except OSError:
pass # ignore
p.wait()
如果子进程可能更早结束,那么可移植的解决方案是use Timer()
as suggested in @sussudio's answer:
#!/usr/bin/env python
from subprocess import Popen
from threading import Timer
def kill(p):
try:
p.kill()
except OSError:
pass # ignore
p = Popen(*call_args)
t = Timer(timeout, kill, [p])
t.start()
p.wait()
t.cancel()
在 Unix 上,你可以use SIGALRM
as suggested in @Alex Martelli's answer:
#!/usr/bin/env python
import signal
from subprocess import Popen
class Alarm(Exception):
pass
def alarm_handler(signum, frame):
raise Alarm
signal.signal(signal.SIGALRM, alarm_handler)
p = Popen(*call_args)
signal.alarm(timeout) # raise Alarm in 5 minutes
try:
p.wait()
signal.alarm(0) # reset the alarm
except Alarm:
p.kill()
p.wait()
为避免在此处使用线程和信号,Python 3 上的subprocess
模块使用busy loop with waitpid(WNOHANG)
calls on Unix 和winapi.WaitForSingleObject()
on Windows。
【讨论】:
【参考方案4】:您可以尝试使用“easyprocess”:
https://github.com/ponty/EasyProcess
它有许多您需要的功能,例如“超时”。
【讨论】:
【参考方案5】:在 python 3.3 中添加了 timeout 参数。
https://docs.python.org/3/library/subprocess.html#subprocess.call
【讨论】:
感谢您的回复。我知道我们使用的是python 2.7.6以上是关于如何使用 python 2.7.6 使 subprocess.call 超时?的主要内容,如果未能解决你的问题,请参考以下文章
带有 Python 2.7.6 和 Virtualenv 12.0.7 (OSX10,10.2) 的 Django 1.7.6 - 模板呈现模板语法错误
使用 EB CLI 3.4.5 (Python 2.7.6) 发布 ElasticBeanStalk/Tomcat 战争部署