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