init.dスクリプトから起動すると、kill -s SIGINTが期待どおりに機能しません。

init.dスクリプトから起動すると、kill -s SIGINTが期待どおりに機能しません。

起動時に(venvを使用して)いくつかのPythonスクリプトを起動し、問題が発生した場合は自動的に再起動できるように状態を監視したいと思います。

prg.py私は次のようにしました: - 永遠に眠り、sigintシグナルに従って終了する単純なPythonの例です。

  • prg.sh監視用のbashスクリプトprg.py。プロセスが停止すると、スクリプトはプログラムを再起動します。スクリプトが sigint を受け取ると、prg.pysigint をプログラムに送信して終了します。

  • bash script() を起動するために使用される Script() init.d。コードは私のものではありません。 GitHubで見つけて修正しました。リンクを検索していますが、見つかったら投稿を編集します。prgprg.shprg

prg.py:

import signal
import time

def signal_handler(sig, frame):
  print('sigint')
  quit(0)

signal.signal(signal.SIGINT, signal_handler)
print ('aaa')
while (1):
  time.sleep(1.5)

prg.sh:

#!/bin/bash

#set -x

DIR="/home/pi"
PRESCRIPT="source venv/bin/activate;python --version"
SCRIPT="python prg.py"

EXITDO=0
PID=-1
trap ctrl_c INT
function ctrl_c() {
  echo 'SIGINT intercepted.'
  EXITDO=1
}

#Main loop
while [ $EXITDO -eq 0 ]; do
  echo 'Starting process…'
  cd "$DIR"
  eval $PRESCRIPT
  $SCRIPT &
  PID=$(echo $!)
  echo "Process started with PID: $PID"
  while $(kill -0 $PID) && [ $EXITDO -eq 0 ]; do
    sleep 1
  done
  if [ $EXITDO -eq 0 ]; then
    echo 'Process terminated. Restart.'
  else
    echo 'Kill service.'
    kill -2 $PID
  fi
done

prg:

#!/bin/sh
### BEGIN INIT INFO
# Provides:          prg
# Required-Start:    $local_fs $network $named $time $syslog
# Required-Stop:     $local_fs $network $named $time $syslog
# Default-Start:     2 3 4 5
# Default-Stop:      0 1 6
# Description:       <DESCRIPTION>
### END INIT INFO
NAME=prg
SCRIPT="/home/pi/prg.sh"

PIDFILE=/var/run/$NAME.pid

start() {
  if [ -f $PIDFILE ] && [ -s $PIDFILE ] && kill -0 $(cat $PIDFILE); then
    echo 'Service already running' >&2
    return 1
  fi
  echo 'Starting service…' >&2
  #local CMD="$SCRIPT &> \"$LOGFILE\" & echo \$!"
  #su -c "$CMD" $RUNAS > "$PIDFILE"
  # Try with this command line instead of above if not workable
  # su -s /bin/sh $RUNAS -c "$CMD" > "$PIDFILE"
  $SCRIPT &
  echo $! > "$PIDFILE"
  PID=$(cat $PIDFILE)
  if kill -0 $(cat $PIDFILE) > /dev/null
  then
    echo "$NAME is now running, the PID is $PID"
  else
    echo ''
    echo "Error! Could not start $NAME!"
  fi
}

stop() {
  if [ ! -f "$PIDFILE" ] || ! kill -0 $(cat "$PIDFILE"); then
    echo 'Service not running' >&2
    return 1
  fi
  echo 'Stopping service…' >&2
  #Kill by pid
  kill -2 $(cat "$PIDFILE")
  #remove .pid file
  sleep 3
  if ! kill -0 $(cat "$PIDFILE"); then
    rm -f "$PIDFILE"
    echo 'Service stopped' >&2
  fi
}

uninstall() {
  echo -n "Are you really sure you want to uninstall this service? That cannot be undone. [yes|No] "
  local SURE
  read SURE
  if [ "$SURE" = "yes" ]; then
    stop
    rm -f "$PIDFILE"
    echo "Notice: log file was not removed: $LOGFILE" >&2
    update-rc.d -f $NAME remove
    rm -fv "$0"
  else
    echo "Abort!"
  fi
}

status() {
    printf "%-50s" "Checking $NAME..."
    if [ -f $PIDFILE ] && [ -s $PIDFILE ]; then
        PID=$(cat $PIDFILE)
            if [ -z "$(ps axf | grep ${PID} | grep -v grep)" ]; then
                printf "%s\n" "The process appears to be dead but pidfile still exists"
            else
                echo "Running, the PID is $PID"
            fi
    else
        printf "%s\n" "Service not running"
    fi
}


case "$1" in
  start)
    start
    ;;
  stop)
    stop
    ;;
  status)
    status
    ;;
  uninstall)
    uninstall
    ;;
  restart)
    stop
    start
    ;;
  *)
    echo "Usage: $0 {start|stop|status|restart|uninstall}"
esac
  • prg.shプログラムを起動すると、期待どおりに実行されます。

  • 自動的にシャットダウンしprg.pyて再起動できます。prg.shprg.py

  • sigintを送信、停止、閉じることができますprg.shprg.shprg.py

  • prgコマンドで始めると、sudo service prg startすべてがうまくいきます。

しかし、

  • sigintこの場合を送信するとprg.sh何も起こりません。

  • 簡単に言うとうまくservice prg stopいかず、その理由を理解していません。

よろしくお願いします。

関連情報