编写systemctl自启动服务

Systemd简介

Centos7 以上使用Systemd进行系统初始化,Systemd是Linux系统中最新的初始化工具,它可以有效的提高系统的启动速度。

Systemd 默认从目录/etc/systemd/system/读取配置文件。然而,里面存放的大部分文件都是符号链接,它们指向目录/usr/lib/systemd/system/,真正的配置文件(以 .service 结尾的文件)存放在那个目录。

我们可以将自定义程序注册为systemd service进程管理交给系统管理,可以方便启动停止,也可以实现服务异常退出重启,开机自启动,减少自定义程序服务管理的时间消耗。

Systemd配置与命令

service文件参数详解

service文件主要包含三个部分,分别是:[Unit]、[Service]、[Install]

如以下的nginx.service配置文件

# 配置文件内容
[Unit]                                          
Description=nginx 
After=network.target 
 
[Service] 
Type=forking 
ExecStart=/usr/local/nginx/sbin/nginx 
ExecReload=/usr/local/nginx/sbin/nginx -s reload 
ExecStop=/usr/local/nginx/sbin/nginx -s quit
PrivateTmp=true 
 
[Install] 
WantedBy=multi-user.target

参数说明如下

参数名称 参数说明
[Unit]
Description 一段描述这个 Unit 文件的文字,通常只是简短的一句话。
Documentation 指定服务的文档,可以是一个或多个文档的URL路径。
Requires 依赖的其他 Unit 列表,列在其中的 Unit 模块会在这个服务启动的同时被启动,并且如果其中有任意一个服务启动失败,这个服务也会被终止
After 与 Requires 相似,但会在后面列出的所有模块全部启动完成以后,才会启动当前的服务。
Before 与 After 相反,在启动指定的任一个模块之前,都会首先确保当前服务已经运行。
Wants 与 Requires 相似,但只是在被配置的这个 Unit 启动时,触发启动列出的每个 Unit 模块,而不去考虑这些模块启动是否成功。
Conflicts 与这个模块有冲突的模块,如果列出模块中有已经在运行的,这个服务就不能启动,反之亦然。
OnFailure 当这个模块启动失败时,就自动启动列出的每个模块。
[Service]
Type 服务的类型,常用的有 simple(默认类型) 和 forking。默认的 simple 类型可以适应于绝大多数的场景,因此一般可以忽略这个参数的配置。而如果服务程序启动后会通过 fork 系统调用创建子进程,然后关闭应用程序本身进程的情况,则应该将 Type 的值设置为 forking,否则 systemd 将不会跟踪子进程的行为,而认为服务已经退出。
Environment 为服务添加环境变量
EnvironmentFile 指定加载一个包含服务所需的环境变量列表的文件,文件中的每一行都是一个环境变量的定义。
ExecStart 这个参数是几乎每个 .service 文件都会有的,指定服务启动的主要命令,在每个配置文件中只能使用一次。(需要使用绝对路径)
ExecStartPre 指定在启动执行 ExecStart 的命令前的准备工作,可以有多个,所有命令会按照文件中书写的顺序依次被执行。
ExecStartPost 指定在启动执行 ExecStart 的命令后的收尾工作,也可以有多个。
ExecStop 停止服务所需要执行的主要命令。(需要使用绝对路径)
ExecStopPost 指定在 ExecStop 命令执行后的收尾工作,也可以有多个。
ExecReload 重新加载服务所需执行的主要命令。(需要使用绝对路径)
Restart 这个值用于指定在什么情况下需要重启服务进程。常用的值有 no,on-success,on-failure,on-abnormal,on-abort 和 always。默认值为 no,即不会自动重启服务。这些不同的值分别表示了在哪些情况下,服务会被重新启动
RestartSec 如果服务需要被重启,这个参数的值为服务被重启前的等待秒数。
Nice 服务的进程优先级,值越小优先级越高,默认为0。-20为最高优先级,19为最低优先级。
WorkingDirectory 指定服务的工作目录。
RootDirectory 指定服务进程的根目录( / 目录),如果配置了这个参数后,服务将无法访问指定目录以外的任何文件。
User 指定运行服务的用户,会影响服务对本地文件系统的访问权限。
Group 指定运行服务的用户组,会影响服务对本地文件系统的访问权限。
PrivateTmp 是否给服务分配独立的临时空间(true/false)
[Install]
WantedBy 和前面的 Wants 作用相似,只是后面列出的不是服务所依赖的模块,而是依赖当前服务的模块。“WantedBy=multi-user.target” 表明当系统以多用户方式(默认的运行级别)启动时,这个服务需要被自动运行。当然还需要 systemctl enable 激活这个服务以后自动运行才会生效。
RequiredBy 和前面的 Requires 作用相似,同样后面列出的不是服务所依赖的模块,而是依赖当前服务的模块。
Also 当这个服务被 enable/disable 时,将自动 enable/disable 后面列出的每个模块。

注意:[Service]的启动、重启、停止命令全部要求使用绝对路径

想了解更详细的参数说明可以在systemd.service 中文手册中查看

systemctl常用命令

以nginx.service为例

# 启动nginx服务
systemctl start nginx.service
# 停止nginx服务
systemctl stop nginx.service
# 重启nginx服务
systemctl restart nginx.service
# 查看nginx服务状态
systemctl status nginx.service
# 在开机时启用一个服务
systemctl enable nginx.service
# 在开机时禁用一个服务
systemctl disable nginx.service
# 查看服务是否开机启动
systemctl is-enabled nginx.service
# 重新读取nginx配置(不用停止nginx服务就能使修改的配置生效)
systemctl reload nginx.service
# 重新加载systemd(一般修改完配置文件执行该命令使配置生效)
systemctl daemon-reload
# 查看已启动的服务列表
systemctl list-unit-files | grep enabled
# 查看启动失败的服务列表
systemctl --failed

其中,systemctl enable命令用于/etc/systemd/system//usr/lib/systemd/system/两个目录之间,建立符号链接关系。

systemctl enable nginx.service
# 等同于
ln -s '/usr/lib/systemd/system/nginx.service' '/etc/systemd/system/multi-user.target.wants/nginx.service'

systemd 默认从目录/etc/systemd/system/读取配置文件,所以systemctl enable命令相当于激活开机启动。

同理,systemctl disable用于撤销两个目录之间的符号链接关系,也就撤销了开机启动。

编写systemd自启动服务

在上面内容的基础下,我们尝试编写一个自己的systemd服务,本次的systemd服务主要是执行一段python脚本。

在/root/pyfile路径(这里可以改为你自己的路径)下创建run.py,内容如下

import time
import string
import random

f = open('/root/pyfile/a.txt', 'w')
for i in range(100):
    words = ''.join(random.sample(string.ascii_lowercase, 5))
    row = '--- No.{}, {} ---\n'.format(i, words)
    f.write(row)
    f.flush()
    time.sleep(2)

f.close()

以上脚本的作用是在100次遍历中,每2s生成一个随机的单词,并将单词写入到a.txt中。

接下来我们来编写自定义的systemd服务,在/usr/lib/systemd/system路径下新建pyscript.service服务,内容如下

[Unit]
Description=python script

[Service]
Type=simple
WorkingDirectory=/root/pyfile
ExecStart=python run.py

[Install]
WantedBy=multi-user.target

编写完成后,保存并执行以下命令

systemctl daemon-reload
systemctl start pyscript.service

进入/root/pyfile目录下执行tail -f a.txt可以看出每2s打印一个随机单词,如下

--- No.0, jkqdr ---
--- No.1, beqvu ---
--- No.2, eaqrw ---
--- No.3, kuljf ---
--- No.4, qmzhl ---

接下来我们设置开机启动pyscript.service服务

systemctl start pyscript.service

执行后系统生成了一个符号链接,如下

Created symlink /etc/systemd/system/multi-user.target.wants/pyscript.service → /usr/lib/systemd/system/pyscript.service.

然后重启服务器,等待重启完毕后,执行以下命令查看python脚本是否执行

ps -ef | grep python

可以发现多了如下进程

root   738  1  0 18:00 ?    00:00:00 /usr/bin/python run.py

再次进入/root/pyfile目录,执行tail -f a.txt,可以发现文本正在打印。

自此,我们已经成功完成了自定义systemd服务的编写和启动。

posted @ 2021-04-06 18:22  蓝莓薄荷  阅读(4304)  评论(0编辑  收藏  举报