小编典典

Python守护程序和systemd服务

python

我有一个简单的Python脚本充当守护程序。我试图创建systemd脚本,以便能够在启动过程中启动此脚本。

当前的systemd脚本:

[Unit]
Description=Text
After=syslog.target

[Service]
Type=forking
User=node
Group=node
WorkingDirectory=/home/node/Node/
PIDFile=/var/run/zebra.pid
ExecStart=/home/node/Node/node.py

[Install]
WantedBy=multi-user.target

node.py:

if __name__ == '__main__':
    with daemon.DaemonContext():
        check = Node()
        check.run()

run包含while True循环。

我尝试使用运行该服务systemctl start zebra-node.service。不幸的是,服务从未完成说明序列-我必须按Ctrl + C。脚本正在运行,但是状态为激活,一段时间后变为停用状态。现在我正在使用python-daemon(但是在我尝试不使用它之前,症状是相似的)。

我应该为脚本实现一些其他功能还是systemd文件不正确?


阅读 170

收藏
2020-12-20

共1个答案

小编典典

原因是,它没有完成启动顺序,是因为对于Type,forking您的启动过程需要分叉并退出(请参阅$ man systemd.service-搜索分叉)。

只需使用主过程,不守护
一种选择是减少工作量。使用systemd时,通常无需创建守护程序,您可以直接运行代码而不进行守护程序。

#!/usr/bin/python -u
from somewhere import Node
check = Node()
check.run()

这允许使用称为的更简单的服务类型simple,因此您的单位文件将看起来像。

[Unit]
Description=Simplified simple zebra service
After=syslog.target

[Service]
Type=simple
User=node
Group=node
WorkingDirectory=/home/node/Node/
ExecStart=/home/node/Node/node.py
StandardOutput=syslog
StandardError=syslog

[Install]
WantedBy=multi-user.target

请注意,-u不需要使用in python shebang,但是如果您将某些内容输出到stdout或stderr,请-u确保没有适当的输出缓冲,并且系统会立即捕获打印的行并将其记录在日志中。没有它,它会出现一些延迟。

为此,我将行StandardOutput=syslog和添加到了单元文件中StandardError=syslog。如果您不关心日记本中的打印输出,则不必关心这些行(不必存在)。

systemd 使守护进程过时

虽然您的问题的标题明确询问了有关守护程序的问题,但我想,问题的核心是“如何使我的服务运行”,而使用主进程似乎要简单得多(您根本不必关心守护程序),可以被视为您问题的答案。

我认为,许多人之所以使用守护进程,只是因为“每个人都这样做”。使用systemd时,守护进程的原因通常已过时。使用守护程序可能有一些原因,但是现在很少见。

编辑:固定python -p为正确python -u

2020-12-20