在systemd下如何编写我们的守护进程,利用systemd管理我们的守护进程

本文介绍了如何在systemd环境下编写和管理守护进程,包括抛弃传统的两次fork和setsid方法,避免维护pid文件,利用systemd的日志功能,正确处理SIGTERM和SIGHUP信号,并给出了python服务示例及service文件配置与启动命令。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

       前言:在sysv init与upstart方式下,我们编写守护进程的方法可参见我写的其它文章,但思想与方法来源都是那本圣经《unix环境高级编程》,基本思路就是两次fork,调用setsid()脱离终端(如果有的话),标准输出入、标准输出、标准出错重定向等等;

     最近的一些linux distribution开始采用systemd作为sysv init和upstart的替代,如果我们想让自己的程序(服务)被systemd管理,则编写守护进程则要遵循一些systemd下的规则了....


一:systemd简单介绍

        http://www.ibm.com/developerworks/cn/linux/1407_liuming_init3/index.html

二:systemd下编写守护进程遵循的几点规则

        开发人员需要了解 systemd 的更多细节。比如您打算开发一个新的系统服务,就必须了解如何让这个服务能够被 systemd 管理。这需要您注意以下这些要点:

  • 后台服务进程代码不需要执行两次派生来实现后台精灵进程,只需要实现服务本身的主循环即可。(传统编写守护进程要至少调用一次fork,然后停止父进程)
  • 不要调用 setsid(),交给 systemd 处理
  • 不再需要维护 pid 文件。(传统编写守护进程,会自己在某个目录下,生成pid文件,一是记录本守护进程pid,另外一点是防止本守护进程被多次重启而导致出错或者导致多个实例在运行)
  • Systemd 提供了日志功能,服务进程只需要输出到 stderr 即可,无需使用 syslog。(传统编写守护进程我们要将标准输出、出错、输入关闭或者重定向,日志信息都是发往rsyslog)
  • 处理信号 SIGTERM,这个信号的唯一正确作用就是停止当前服务,不要做其他的事情。(传统守护进程一般SIGTERM也是用来做这种事情的)
  • SIGHUP 信号的作用是重启服务。(传统数据进程一般SIGHUP也是做这种事情的)
  • 需要套接字的服务,不要自己创建套接字,让 systemd 传入套接字。(这个承接systemd快速启动优点而设立的,可以实现这个特点,也可以不实现)
  • 使用 sd_notify()函数通知 systemd 服务自己的状态改变。一般地,当服务初始化结束,进入服务就绪状态时,可以调用它。(没用过)

三:实例编写

      1.python代码:

#! /usr/bin/python
import sys, os, socket, signal
serv = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
addr = '', 9090
serv.bind(addr)
serv.listen(5)

def sighup(num1, num2):
    sys.stderr.write("We receive SIGHUP signal...\nwe will reload the configuration file...\n")
def sigterm(num1, num2):
    sys.stderr.write("We kill us greacefully...\n")
    os._exit(0)

#handle signal SIGHUB
signal.signal(signal.SIGHUP, sighup)

#handler signal SIGTERM
signal.signal(signal.SIGTERM, sigterm)

sys.stderr.write("My self daemon start...\n")
while True:
    try:
        conn, addr = serv.accept()
    except Exception:
        continue
    try:
        pid = os.fork()
        if pid == 0:
           #child process close serv_socket_object of parent
           serv.close()
           time.sleep(5)
           os._exit(0)
        else:
           #parent process close conn_socket_object of child
           conn.close()
    except OSError:
        #if we use conn.close(), we must wait for python gc to 
        #really close the socket connection
        conn.shutdown(socket.SHUT_RDWR)

这就是一个普通的python程序,我们编写了信号处理函数,分别针对SIGHUB(用于重新读取配置文件)、SIGTERM(用于杀死自己)

    2.编写service文件

[root@localhost system]# pwd
/etc/systemd/system
[root@localhost system]# cat mydaemon.service 
[Unit]
Description=myDaemon 
[Service]
ExecStart=/root/mydaemon.py
ExecStop=/bin/kill -TERM $MAINPID
ExecReload=/bin/kill -HUP $MAINPID
killMode=process
After=network.target
[Install]
WantedBy=multi-user.target


我们在/etc/systemd/system新建mydaemon.service文件,文件内容如上

  3. 命令执行

接着我们执行:

[root@localhost system]# systemctl daemon-reload


再执行:

[root@localhost system]# systemctl start mydaemon

查看一下状态:

[root@localhost system]# systemctl status mydaemon
mydaemon.service - myDaemon
   Loaded: <span style="color:#ff0000;">loaded</span> (/etc/systemd/system/mydaemon.service; <span style="color:#ff0000;">disabled</span>)
   Active: <span style="color:#ff0000;">active</span> (running) since 三 2015-11-18 16:35:15 CST; 45min ago
 Main PID: 6873 (mydaemon.py)
   CGroup: /system.slice/mydaemon.service
           └─<span style="color:#ff0000;">6873 /usr/bin/python /root/mydaemon.py</span>

11月 18 16:35:15 localhost.localdomain systemd[1]: Started myDaemon.
11月 18 16:35:15 localhost.localdomain mydaemon.py[6873]: My self daemon start...
11月 18 16:35:36 localhost.localdomain systemd[1]: Reloading myDaemon.
11月 18 16:35:36 localhost.localdomain mydaemon.py[6873]: We receive SIGHUP signal...and we will reload the...ile
11月 18 16:35:36 localhost.localdomain systemd[1]: Reloaded myDaemon.
11月 18 17:19:00 localhost.localdomain systemd[1]: [/etc/systemd/system/mydaemon.service:7] Unknown lvalue...ice'
11月 18 17:19:00 localhost.localdomain systemd[1]: [/etc/systemd/system/mydaemon.service:8] Unknown lvalue...ice'
11月 18 17:20:12 localhost.localdomain systemd[1]: Started myDaemon.
Hint: Some lines were ellipsized, use -l to show in full.
正常启动了,不过貌似我配置文件写的有点问题


现在我们验证下systemd提供的日志功能:

[root@localhost system]# systemctl reload mydaemon
日志内容(命令内容: journalctl --unit mydaemon):

11月 18 17:20:12 localhost.localdomain systemd[1]: Started myDaemon.
11月 18 17:23:56 localhost.localdomain systemd[1]: Reloading myDaemon.
11月 18 17:23:56 localhost.localdomain mydaemon.py[6873]: <span style="color:#ff0000;">We receive SIGHUP signal...and we will reload the config</span>
11月 18 17:23:56 localhost.localdomain systemd[1]: Reloaded myDaemon.
红色内容就是我们在程序中sys.stderr.write()的内容


当然,我们也可以设置为开机启动,命令

[root@localhost system]# systemctl enable mydaemon









评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值