[英]How to send a SIGINT to Python from a bash script?
我想從bash腳本啟動后台Python作業,然后使用SIGINT優雅地終止它。 這可以從shell工作得很好,但我似乎無法讓它在腳本中工作。
loop.py:
#! /usr/bin/env python
if __name__ == "__main__":
try:
print 'starting loop'
while True:
pass
except KeyboardInterrupt:
print 'quitting loop'
從shell我可以打斷它:
$ python loop.py &
[1] 15420
starting loop
$ kill -SIGINT 15420
quitting loop
[1]+ Done python loop.py
kill.sh:
#! /bin/bash
python loop.py &
PID=$!
echo "sending SIGINT to process $PID"
kill -SIGINT $PID
但是從腳本我不能:
$ ./kill.sh
starting loop
sending SIGINT to process 15452
$ ps ax | grep loop.py | grep -v grep
15452 pts/3 R 0:08 python loop.py
並且,如果它是從腳本啟動的,我就不能再從shell中刪除它了:
$ kill -SIGINT 15452
$ ps ax | grep loop.py | grep -v grep
15452 pts/3 R 0:34 python loop.py
我假設我錯過了一些關於bash工作控制的好點。
你沒有注冊信號處理程序。 試試下面的內容。 它看起來相當可靠。 我認為罕見的例外是它在Python注冊腳本的處理程序之前捕獲信號。 請注意,只有當用戶點擊中斷鍵時才會引發KeyboardInterrupt。 我認為它適用於顯式(例如,通過殺死)SIGINT的事實是實施的意外。
import signal
def quit_gracefully(*args):
print 'quitting loop'
exit(0);
if __name__ == "__main__":
signal.signal(signal.SIGINT, quit_gracefully)
try:
print 'starting loop'
while True:
pass
except KeyboardInterrupt:
quit_gracefully()
我同意Matthew Flaschen; 問題出在python上,當沒有從交互式shell調用時,它顯然沒有用SIGINT注冊KeyboardInterrupt異常。
當然,沒有什么能阻止你像這樣注冊你的信號處理程序:
def signal_handler(signum, frame):
raise KeyboardInterrupt, "Signal handler"
除了@ matthew-flaschen的答案之外,你可以在bash腳本中使用exec
來有效地將范圍替換為正在打開的進程:
#!/bin/bash
exec python loop.py &
PID=$!
sleep 5 # waiting for the python process to come up
echo "sending SIGINT to process $PID"
kill -SIGINT $PID
使用&在后台運行命令時,將忽略SIGINT。 這是man bash的相關部分:
由bash運行的非內置命令將信號處理程序設置為shell從其父級繼承的值。 當作業控制不起作用時,除了這些繼承的處理程序之外,異步命令還會忽略SIGINT和SIGQUIT。 由於命令替換而運行的命令忽略鍵盤生成的作業控制信號SIGTTIN,SIGTTOU和SIGTSTP。
我認為您需要明確設置信號處理程序,如Matthew所評論的那樣。
腳本kill.sh也有問題。 由於loop.py被發送到后台,因此不能保證在python loop.py之后kill會運行。
#! /bin/bash
python loop.py &
PID=$!
#
# NEED TO WAIT ON EXISTENCE OF python loop.py PROCESS HERE.
#
echo "sending SIGINT to process $PID"
kill -SIGINT $PID
試過@ Steen的方法,但是唉,它顯然不適用於Mac。
另一個解決方案,與上面幾乎相同,但更一般的是,如果忽略SIGINT
,只需重新安裝默認處理程序:
def _ensure_sigint_handler():
# On Mac, even using `exec <cmd>` in `bash` still yields an ignored SIGINT.
sig = signal.getsignal(signal.SIGINT)
if signal.getsignal(signal.SIGINT) == signal.SIG_IGN:
signal.signal(signal.SIGINT, signal.default_int_handler)
# ...
_ensure_sigint_handler()
聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.