Python script как служба/демон linux

Алло,

Я пытаюсь запустить python script как службу (daemon) на (ubuntu) linux.

В Интернете существует несколько решений, таких как:

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

Хорошо продуманный процесс демона Unix с трудом подходит для правильной работы, но требуемые шаги для всех программ-демонов одинаковы. Экземпляр DaemonContext содержит поведение и настроенную среду процесса для программы; используйте экземпляр в качестве менеджера контекста для входа в состояние демона.

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

Однако, поскольку я хочу интегрировать свой python script специально с ubuntu linux, мое решение представляет собой комбинацию с init.d script

#!/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 __name__ == '__main__':
    while not stop_event.is_set():
        time.sleep(3)

Теперь мой вопрос: правильно ли этот подход. Должен ли я обрабатывать любые дополнительные сигналы? Будет ли это "хорошо зарекомендовавший себя демон-процесс Unix"?

Ответ 1

Предполагая, что у вашего демона есть какой-то способ непрерывного запуска (какой-то цикл событий, скрученный, что угодно), вы можете попробовать использовать upstart.

Здесь приведен пример конфигурации выскочки для гипотетической службы Python:

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, вы просто сделаете одноразовый

$ sudo initctl reload-configuration
$ sudo start script

Вы можете остановить его с помощью stop script. То, что сказал выше, говорит о том, чтобы запустить эту службу при перезагрузках, а также перезапустить ее, если она умирает.

Что касается обработки сигналов - ваш процесс должен нормально реагировать на SIGTERM. По умолчанию это нужно обрабатывать, если вы специально не установили свой собственный обработчик сигналов.

Ответ 2

Ответ Rloton хороший. Вот легкая утонченность, просто потому, что я потратил много времени на отладку. И мне нужно сделать новый ответ, чтобы я мог правильно отформатировать.

Несколько других моментов, которые потребовали от меня отладки:

  • Когда это не удается, сначала проверьте /var/log/upstart/.log
  • Если ваш script реализует демон с python-daemon, вы НЕ используете строфу 'expect daemon'. Отсутствие "ожидаемых" работ. Я не знаю почему. (Если кто знает, почему - пожалуйста, напишите!)
  • Кроме того, продолжайте проверять статус initctl 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