簡體   English   中英

如何從bash腳本向Python發送SIGINT?

[英]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.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM