CentOS7下systemctl添加自定义服务

CentOS7下systemctl添加自定义服务

一、服务文件位置

CentOS7的服务systemctl脚本存放在/usr/lib/systemd/下,有系统和用户之分

需要开机不登陆就能运行的程序,存在系统服务里,路径:/usr/lib/systemd/system

登陆后才可以运行的程序,存在用户服务里,路径:/usr/lib/systemd/user

二、服务的基础组成

服务以.service结尾,一般会分为3部分:[Unit]、[Service]和[Install],每个部分有自己的配置。

1、[Unit]

Description : 服务的简单描述

Documentation : 服务文档

Before:代表本服务在xxx.service启动之前启动。

After:代表本服务在xxx.service之后启动。

Requires:这个服务启动了,它需要的服务也会被启动;它需要的服务被停止了,这个服务也停止了。

Wants:这个服务启动了,它需要的服务也会被启动;它需要的服务被停止了,对本服务没有影响。

2、[Service]

Type:定义服务的启动方式,有下面的值可以选

取值 含义
Type=oneshot 这一选项适用于只执行一项任务、随后立即退出的服务。可能需要同时设置 RemainAfterExit=yes 使得 systemd 在服务进程退出之后仍然认为服务处于激活状态。
Type=notify 与 Type=simple 相同,但约定服务会在就绪后向 systemd 发送一个信号。这一通知的实现由 libsystemd-daemon.so 提供。
Type=dbus 若以此方式启动,当指定的 BusName 出现在DBus系统总线上时,systemd认为服务就绪。
Type=idle systemd会等待所有任务处理完成后,才开始执行 idle 类型的单元。其他行为与 Type=simple 类似。
Type=forking systemd认为当该服务进程fork,且父进程退出后服务启动成功。对于常规的守护进程(daemon),除非你确定此启动方式无法满足需求,使用此类型启动即可。使用此启动类型应同时指定 PIDFile=,以便 systemd 能够跟踪服务的主进程
Type=simple (默认值) systemd认为该服务将立即启动。服务进程不会 fork 。如果该服务要启动其他服务,不要使用此类型启动,除非该服务是socket 激活型。

PIDFile:pid文件路径

ExecStart:启动服务的命令或者脚本,

ExecStartPre:在ExecStart之前用户自定义执行的脚本

ExecStartPost:在ExecStart之后用户自定义执行的脚本。Type=oneshot允许指定多个希望顺序执行的用户自定义命令。

ExecReload:指定服务停止时执行的命令或者脚本。

ExecStop:指定服务停止时执行的命令或者脚本。

PrivateTmp:True表示给服务分配独立的临时空间

Restart:这个选项如果被允许,服务重启的时候进程会退出,会通过systemctl命令执行清除并重启的操作。

RemainAfterExit:如果设置这个选择为真,服务会被认为是在激活状态,即使所以的进程已经退出,默认的为假,这个选项只有在Type=oneshot时需要被配置。

Restart:退出后的重启方式,有下面的值可以选

取值 含义
no(默认值) 退出后不会重启
on-success 只有正常退出时(退出状态码为0),才会重启
on-failure 非正常退出时(退出状态码非0),包括被信号终止和超时,才会重启
on-abnormal 只有被信号终止和超时,才会重启
on-abort 只有在收到没有捕捉到的信号终止时,才会重启
on-watchdog 超时退出,才会重启
always 不管是什么退出原因,总是重启

3、[Install]

Alias:为单元提供一个空间分离的附加名字。

RequiredBy:单元被允许运行需要的一系列依赖单元,RequiredBy列表从Require获得依赖信息。

WantBy:单元被允许运行需要的弱依赖性单元,Wantby从Want列表获得依赖信息。

Also:指出和单元一起安装或者被协助的单元。

DefaultInstance:实例单元的限制,这个选项指定如果单元被允许运行默认的实例。

三、服务文件实例

可以看看原始的服务都是怎么写的cat /usr/lib/systemd/system/某某服务.service

nginx服务

[root@localhost ~]# cat /usr/lib/systemd/system/nginx.service
[Unit]
Description=The nginx HTTP and reverse proxy server
After=network-online.target remote-fs.target nss-lookup.target
Wants=network-online.target

[Service]
Type=forking
PIDFile=/run/nginx.pid
# Nginx will fail to start if /run/nginx.pid already exists but has the wrong
# SELinux context. This might happen when running `nginx -t` from the cmdline.
# https://bugzilla.redhat.com/show_bug.cgi?id=1268621
ExecStartPre=/usr/bin/rm -f /run/nginx.pid
ExecStartPre=/usr/sbin/nginx -t
ExecStart=/usr/sbin/nginx
ExecReload=/usr/sbin/nginx -s reload
KillSignal=SIGQUIT
TimeoutStopSec=5
KillMode=process
PrivateTmp=true

[Install]
WantedBy=multi-user.target
[root@localhost ~]#

sshd服务

[root@localhost ~]# cat /usr/lib/systemd/system/sshd.service
[Unit]
Description=OpenSSH server daemon
Documentation=man:sshd(8) man:sshd_config(5)
After=network.target sshd-keygen.service
Wants=sshd-keygen.service

[Service]
Type=notify
EnvironmentFile=/etc/sysconfig/sshd
ExecStart=/usr/sbin/sshd -D $OPTIONS
ExecReload=/bin/kill -HUP $MAINPID
KillMode=process
Restart=on-failure
RestartSec=42s

[Install]
WantedBy=multi-user.target
[root@localhost ~]#

四、自定义服务

使用go编写了一个web服务

package main

import (
   "fmt"
   "net/http"
)

func main() {
   http.HandleFunc("/", func(writer http.ResponseWriter, request *http.Request) {
      n, err := fmt.Fprintln(writer, "我启动成功了")
      if err != nil {
         return
      } else {
         fmt.Println(n)
      }
   })
   err := http.ListenAndServe(":8080", nil)
   if err != nil {
      return
   }
}

编译后放在了/home/tmp下,名字就叫s

image-20220705174330981

部署为服务

①创建并编辑服务文件

vi /usr/lib/systemd/system/mygoweb.service

输入以下内容

[Unit]
Description=MY-GO-WEB-SERVER

[Service]
Type=simple
ExecStart=/home/tmp/s

[Install]
WantedBy=multi-user.target

③加载配置

systemctl daemon-reload

④启动服务

systemctl start mygoweb.service

重启服务

systemctl restart mygoweb.service

停止服务

systemctl stop mygoweb.service

查看服务

systemctl status mygoweb.service

五、遇到的错误

1、执行systemctl start 某某服务,然后卡住了,一般是Type设置错误(应该设置Type=simple但设置成了Type=forking),可参考:centos7编写自己的服务,运行systemctl后卡住了(即shell阻塞了)

2、上面提及到:

Type=forking  systemd认为当该服务进程fork,且父进程退出后服务启动成功。对于常规的守护进程(daemon),除非你确定此启动方式无法满足需求,使用此类型启动即可。使用此启动类型应同时指定 PIDFile=,以便 systemd 能够跟踪服务的主进程

但是我遇到使用Type=forking 然后指定PIDFile,然后systemctl start 某某服务,也是卡住了,解决办法是不写PIDFile

卡住现象:

image-20220707141118170

错误的配置(PIDFile文件写错了)

image-20220707141219220

正确的配置(去掉了PIDFile)

image-20220707141840912

正确的配置2(写你的程序输出的PID文件,如果没有,那就不写PIDFile)

image-20220707142306491

六、常用的命令

查看开机自启动的服务

systemctl list-unit-files --type service | grep enabled

image-20220824100537681

查看某个服务是否是开机自启动

例如查看防火墙服务是否开启自启动,enabled就是自启动 disabled就是非自启动

systemctl list-unit-files --type service | grep firewalld

image-20220824100717847

设置开机自启动

systemctl enable service_name

禁止开机自启动

systemctl disable service_name

七、日志重定向到文件

如果服务的日志是控制台输出,当我们需要重定向日志到文件时,可以这么做。

①在自定义 xxx.service 文件中 [Service] 增加

StandardOutput=syslog

StandardError=syslog

SyslogIdentifier=identifier_for-me

identifier_for-me是随意写的字符串,表示唯一标识日志,可以理解是唯一ID

②创建 /etc/rsyslog.d/<new_file>.conf 文件,其内容为

if $programname == 'identifier_for-me' then /home/log/mylog.log
& stop

③重启 rsyslog

重新启动 rsyslog (sudo systemctl restart rsyslog)

④重新启动服务 systemctl restart xxx

参考:https://www.codenong.com/37585758/https://my.oschina.net/u/3625745/blog/4693099

八、参考的链接

Linux——systemctl 添加自定义服务

centos查看自启动服务

posted @ 2022-07-05 18:23  南风丶轻语  阅读(5052)  评论(0编辑  收藏  举报