.service服务简介

上文介绍了systemctl命令是基于Systemd系统的,本文主要简单的介绍一下如何编写一个可以使用systemctl命令管理的后台进程(daemon进程),使用systemctl命令管理进程(服务)主要有以下好处:
1、可以方便的查看进程的状态
2、可以方便的启动、停止、重启、reload进程
3、可以方便的配置进程是否随系统启动而启动,并且可以配置复杂的依赖关系(如:在某个服务启动后启动)
4、可以方便的指定以某个用户的身份启动
等等。

service文件一般分为三个部分:

Unit

用来定义 Unit 的元数据,以及配置与其他 Unit 的关系:

  1. Description:简短描述
  2. Documentation:文档地址
  3. Requires:当前 Unit 依赖的其他 Unit,如果它们没有运行,当前 Unit 会启动失败
  4. Wants:与当前 Unit 配合的其他 Unit,如果它们没有运行,当前 Unit 不会启动失败
  5. BindsTo:与Requires类似,它指定的 Unit 如果退出,会导致当前 Unit 停止运行
  6. Before:如果该字段指定的 Unit 也要启动,那么必须在当前 Unit 之后启动
  7. After:如果该字段指定的 Unit 也要启动,那么必须在当前 Unit 之前启动
  8. Conflicts:这里指定的 Unit 不能与当前 Unit 同时运行
  9. Condition…:当前 Unit 运行必须满足的条件,否则不会运行
  10. Assert…:当前 Unit 运行必须满足的条件,否则会报启动失败

Service

用来定义Service的配置,只有 Service 类型的 Unit 才有这个部分:

  1. Type:定义启动时的进程行为。它有以下几种值。
Type=simple:默认值,执行ExecStart指定的命令,启动主进程
Type=forking:以 fork 方式从父进程创建子进程,创建后父进程会立即退出
Type=oneshot:一次性进程,Systemd 会等当前服务退出,再继续往下执行
Type=dbus:当前服务通过D-Bus启动
Type=notify:当前服务启动完毕,会通知Systemd,再继续往下执行
Type=idle:若有其他任务执行完毕,当前服务才会运行
  1. ExecStart:启动当前服务的命令
  2. ExecStartPre:启动当前服务之前执行的命令
  3. ExecStartPost:启动当前服务之后执行的命令
  4. ExecReload:重启当前服务时执行的命令
  5. ExecStop:停止当前服务时执行的命令
  6. ExecStopPost:停止当其服务之后执行的命令
  7. RestartSec:自动重启当前服务间隔的秒数
  8. Restart:定义何种情况 Systemd 会自动重启当前服务,可能的值包括always(总是重启)、on-success、on-failure、on-abnormal、on-abort、on-watchdog
  9. TimeoutSec:定义 Systemd 停止当前服务之前等待的秒数
  10. Environment:指定环境变量

Install

用来定义如何启动,以及是否开机启动:

  1. WantedBy:它的值是一个或多个 Target,当前 Unit 激活时(enable)符号链接会放入/etc/systemd/system目录下面以 Target 名 + .wants后缀构成的子目录中
  2. RequiredBy:它的值是一个或多个 Target,当前 Unit 激活时,符号链接会放入/etc/systemd/system目录下面以 Target 名 + .required后缀构成的子目录中
  3. Alias:当前 Unit 可用于启动的别名
  4. Also:当前 Unit 激活(enable)时,会被同时激活的其他 Unit

后记——自动启动

对于那些支持 Systemd 的软件,安装的时候,会自动在/usr/lib/systemd/system目录添加一个配置文件,如: systemctl enable httpd.service 上面的命令相当于在 /etc/systemd/system 目录添加一个符号链接,指向 /usr/lib/systemd/system 里面的 httpd.service 文件。
这是因为开机时, Systemd 只执行 /etc/systemd/system 目录里面的配置文件。这也意味着,如果把修改后的配置文件放在该目录,就可以达到覆盖原始配置的效果。

示例

示例1

[Unit]
Description=Foo

[Service]
ExecStart=/usr/sbin/foo-daemon

[Install]
WantedBy=multi-user.target

示例2

[Unit]
Description=Simple firewall

[Service]
Type=oneshot
RemainAfterExit=yes
ExecStart=/usr/local/sbin/simple-firewall-start
ExecStop=/usr/local/sbin/simple-firewall-stop

[Install]
WantedBy=multi-user.target

示例3

[Unit]
Description=mdpb
After=mariadb.service

[Service]
User=ubuntu
Group=ubuntu
Type=simple
PIDFile=/deploy/mdpb/1.0/debug/bin/mdpb.pid
ExecStart=/deploy/mdpb/1.0/debug/bin/mdpb 
ExecReload=/deploy/quant-mdpb/1.0/debug/bin/mdpb
Restart=always
RestartSec=10
KillSignal=SIGINT
PrivateTmp=true

[Install]
WantedBy=multi-user.target