web-dev-qa-db-ja.com

Python Linuxサービス/デーモンとしてのスクリプト

こんにちは

(ubuntu)linux上でサービス(デーモン)としてpythonスクリプトを実行させようとしています。

Webには、次のようなソリューションがいくつかあります。

http://pypi.python.org/pypi/python-daemon/

正常に動作するUnixデーモンプロセスは正しく動作するのが難しいですが、必要な手順はすべてのデーモンプログラムでほぼ同じです。 DaemonContextインスタンスは、プログラムの動作と構成されたプロセス環境を保持します。インスタンスをコンテキストマネージャとして使用して、デーモン状態に入ります。

http://www.jejik.com/articles/2007/02/a_simple_unix_linux_daemon_in_python/

ただし、pythonスクリプトとubuntu linuxを具体的に統合したいので、私のソリューションはinit.dスクリプトとの組み合わせです

#!/bin/bash

WORK_DIR="/var/lib/foo"
DAEMON="/usr/bin/python"
ARGS="/opt/foo/linux_service.py"
PIDFILE="/var/run/foo.pid"
USER="foo"

case "$1" in
  start)
    echo "Starting server"
    mkdir -p "$WORK_DIR"
    /sbin/start-stop-daemon --start --pidfile $PIDFILE \
        --user $USER --group $USER \
        -b --make-pidfile \
        --chuid $USER \
        --exec $DAEMON $ARGS
    ;;
  stop)
    echo "Stopping server"
    /sbin/start-stop-daemon --stop --pidfile $PIDFILE --verbose
    ;;
  *)
    echo "Usage: /etc/init.d/$USER {start|stop}"
    exit 1
    ;;
esac

exit 0

そしてPythonで:

import signal
import time
import multiprocessing

stop_event = multiprocessing.Event()

def stop(signum, frame):
    stop_event.set()

signal.signal(signal.SIGTERM, stop)

if __== '__main__':
    while not stop_event.is_set():
        time.sleep(3)

私の質問は、このアプローチが正しいかどうかです。追加の信号を処理する必要がありますか? 「行儀の良いUnixデーモンプロセス」でしょうか。

67
tauran

デーモンに継続的に実行する何らかの方法(何らかのイベントループ、ツイストなど)があると仮定すると、upstartを使用してみることができます。

以下は、架空のPythonサービスのupstart configの例です。

description "My service"
author  "Some Dude <[email protected]>"

start on runlevel [234]
stop on runlevel [0156]

chdir /some/dir
exec /some/dir/script.py
respawn

これをscript.confとして/etc/initに保存すると、単純に1回だけ

$ Sudo initctl reload-configuration
$ Sudo start script

stop scriptで停止できます。上記のupstart confには、再起動時にこのサービスを開始し、サービスが停止した場合は再起動することも記載されています。

シグナル処理については、プロセスは当然SIGTERMに応答する必要があります。デフォルトでは、独自のシグナルハンドラを特にインストールしていない限り、これを処理する必要があります。

87
rlotun

ロトンの答えは良いです。デバッグに多くの時間を費やしたからといって、ここに少し改良を加えました。そして、適切にフォーマットできるように、新しい回答を行う必要があります。

デバッグに永遠にかかった他のいくつかのポイント:

  1. 失敗した場合、最初に/var/log/upstart/.logを確認します
  2. スクリプトが python-daemon でデーモンを実装する場合、「expect daemon」スタンザは使用しません。 「期待」が機能しない。理由はわかりません。 (誰かが理由を知っているなら-投稿してください!)
  3. また、「initctl status script」をチェックして、起動していることを確認してください(開始/実行)。 (そして、confファイルを更新するときにリロードを行います)

これが私のバージョンです:

description "My service"
author  "Some Dude <[email protected]>"

env PYTHON_HOME=/<pathtovirtualenv>
env PATH=$PYTHON_HOME:$PATH

start on runlevel [2345]
stop on runlevel [016]

chdir <directory>

# NO expect stanza if your script uses python-daemon
exec $PYTHON_HOME/bin/python script.py

# Only turn on respawn after you've debugged getting it to start and stop properly
respawn
9
Ross R