Python 脚本作为 linux 服务/守护进程

你好,

我试图让一个 python 脚本在(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 脚本特别集成到 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

还有蟒蛇:

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 守护进程”吗?

74986 次浏览

假设您的守护进程有某种方式可以持续运行(某个事件循环、双向循环等等) ,您可以尝试使用 upstart

下面是一个假设的 Python 服务的 upstart 配置示例:

description "My service"
author  "Some Dude <blah@foo.com>"


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阻止它。上面的 upstart conf 说的是在重新引导时启动这个服务,如果死机也要重新启动它。

至于信号处理-您的进程应该自然地响应 SIGTERM。默认情况下,除非您专门安装了自己的信号处理程序,否则应该处理这个问题。

罗顿的回答很好。这是一个轻微的改进,只是因为我花了大量的时间调试。我需要做一个新的答案,这样我就可以正确的格式。

还有几点我花了很长时间调试:

  1. 当它失败时,首先检查/var/log/upstart/. log
  2. 如果您的脚本使用 巨蟒守护神实现了一个守护进程,那么不要使用“ Expect 守护进程”节。没有“期望”是有用的。我不知道为什么。(如果有人知道原因,请发帖!)
  3. 另外,继续检查“ initctl status script”以确保已经启动(启动/运行)。(并在更新 conf 文件时重新加载)

我的版本是这样的:

description "My service"
author  "Some Dude <blah@foo.com>"


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