CentOS7使用systemctl新增自定義服務

weixin_33912445發表於2018-02-22

一、簡介

Centos7開機第一個程式從init完全換成了systemd這種啟動方式,同centos 5 6已經是實質差別。systemd是靠管理unit的方式來控制開機服務,開機級別等功能。
在/usr/lib/systemd/system目錄下包含了各種unit檔案,有service字尾的服務unit,有target字尾的開機級別unit等,這裡介紹關於service字尾的檔案。因為systemd在開機要想執行自啟動,都是通過這些*.service 的unit控制的,服務又分為系統服務(system)和使用者服務(user)。

  • 系統服務:開機不登陸就能執行的程式(常用於開機自啟)。
    使用者服務:需要登陸以後才能執行的程式。

二、配置檔案說明:

  • [Unit] 區塊:啟動順序與依賴關係
    Description欄位:給出當前服務的簡單描述。
    Documentation欄位:給出文件位置。
    After欄位:如果network.target或sshd-keygen.service需要啟動,那麼sshd.service應該在它們之後啟動。
    Before欄位:定義sshd.service應該在哪些服務之前啟動。
    注:After和Before欄位只涉及啟動順序,不涉及依賴關係。

    舉例來說,某 Web 應用需要 postgresql 資料庫儲存資料。在配置檔案中,它只定義要在 postgresql 之後啟動,而沒有定義依賴 postgresql 。上線後,由於某種原因,postgresql 需要重新啟動,在停止服務期間,該 Web 應用就會無法建立資料庫連線。
    設定依賴關係,需要使用Wants欄位和Requires欄位。
    Wants欄位:表示sshd.service與sshd-keygen.service之間存在"弱依賴"關係,即如果"sshd-keygen.service"啟動失敗或停止執行,不影響sshd.service繼續執行。
    Requires欄位則表示"強依賴"關係,即如果該服務啟動失敗或異常退出,那麼sshd.service也必須退出。
    注意,Wants欄位與Requires欄位只涉及依賴關係,與啟動順序無關,預設情況下是同時啟動的。

  • [Service] 區塊:啟動行為

    • 啟動命令
      ExecStart欄位:定義啟動程式時執行的命令
      ExecReload欄位:重啟服務時執行的命令
      ExecStop欄位:停止服務時執行的命令
      ExecStartPre欄位:啟動服務之前執行的命令
      ExecStartPost欄位:啟動服務之後執行的命令
      ExecStopPost欄位:停止服務之後執行的命令

      注:所有的啟動設定之前,都可以加上一個連詞號(-),表示"抑制錯誤",即發生錯誤的時候,不影響其他命令的執行。比如EnvironmentFile=-/etc/sysconfig/sshd(注意等號後面的那個連詞號),就表示即使/etc/sysconfig/sshd檔案不存在,也不會丟擲錯誤。
      注意:[Service]中的啟動、重啟、停止命令全部要求使用絕對路徑!

    • 啟動型別
      Type欄位定義啟動型別。它可以設定的值如下:
      simple(預設值):ExecStart欄位啟動的程式為主程式
      forking:ExecStart欄位將以fork()方式啟動,此時父程式將會退出,子程式將成為主程式(後臺執行)
      oneshot:類似於simple,但只執行一次,Systemd 會等它執行完,才啟動其他服務
      dbus:類似於simple,但會等待 D-Bus 訊號後啟動
      notify:類似於simple,啟動結束後會發出通知訊號,然後 Systemd 再啟動其他服務
      idle:類似於simple,但是要等到其他任務都執行完,才會啟動該服務。一種使用場合是為讓該服務的輸出,不與其他服務的輸出相混合

    • 重啟行為
      Service區塊有一些欄位,定義了重啟行為:
      KillMode欄位:定義 Systemd 如何停止 sshd 服務:
      control-group(預設值):當前控制組裡面的所有子程式,都會被殺掉
      process:只殺主程式
      mixed:主程式將收到 SIGTERM 訊號,子程式收到 SIGKILL 訊號
      none:沒有程式會被殺掉,只是執行服務的 stop 命令。
      Restart欄位:定義了 sshd 退出後,Systemd 的重啟方式
      上面的例子中,Restart設為on-failure,表示任何意外的失敗,就將重啟sshd。如果 sshd 正常停止(比如執行systemctl stop命令),它就不會重啟。
      Restart欄位可以設定的值如下。
      no(預設值):退出後不會重啟
      on-success:只有正常退出時(退出狀態碼為0),才會重啟
      on-failure:非正常退出時(退出狀態碼非0),包括被訊號終止和超時,才會重啟
      on-abnormal:只有被訊號終止和超時,才會重啟
      on-abort:只有在收到沒有捕捉到的訊號終止時,才會重啟
      on-watchdog:超時退出,才會重啟
      always:不管是什麼退出原因,總是重啟
      注:對於守護程式,推薦設為on-failure。對於那些允許發生錯誤退出的服務,可以設為on-abnormal。
      RestartSec欄位:表示 Systemd 重啟服務之前,需要等待的秒數。
      上面的例子設為等待42秒。

  • [Install] 區塊
    Install區塊,定義如何安裝這個配置檔案,即怎樣做到開機啟動。
    WantedBy欄位:表示該服務所在的 Target。
    Target的含義是服務組,表示一組服務。
    WantedBy=multi-user.target指的是:sshd 所在的 Target 是multi-user.target。
    這個設定非常重要,因為執行systemctl enable sshd.service命令時,sshd.service的一個符號連結,就會放在/etc/systemd/system目錄下面的multi-user.target.wants子目錄之中。
    Systemd 有預設的啟動 Target。

    systemctl get-default
    #輸出multi-user.target
    

    上面的結果表示,預設的啟動 Target 是multi-user.target。在這個組裡的所有服務,都將開機啟動。這就是為什麼systemctl enable命令能設定開機啟動的原因。
    使用 Target 的時候,systemctl list-dependencies命令和systemctl isolate命令也很有用。

    #檢視 multi-user.target 包含的所有服務
    systemctl list-dependencies multi-user.target
    
    #切換到另一個 target
    #shutdown.target 就是關機狀態
    systemctl isolate shutdown.target
    

    一般來說,常用的 Target 有兩個:
    multi-user.target:表示多使用者命令列狀態;
    graphical.target:表示圖形使用者狀態,它依賴於multi-user.target。

三、註冊服務例項

  • 配置檔案目錄
    systemctl指令碼目錄:/usr/lib/systemd/
    系統服務目錄:/usr/lib/systemd/system/
    使用者服務目錄:/usr/lib/systemd/system/

  • 在/usr/lib/systemd/system目錄下新建service-name.service檔案:

    [UNIT]
    #服務描述
    Description=Media wanager Service
    #指定了在systemd在執行完那些target之後再啟動該服務
    After=network.target
    
    [Service]
    #定義Service的執行型別,一般是forking(後臺執行)   
    Type=forking
    
    #定義systemctl start|stop|reload *.service 的執行方法(具體命令需要寫絕對路徑)
    #注:ExecStartPre為啟動前執行的命令
    ExecStartPre=/usr/bin/test "x${NETWORKMANAGER}" = xyes
    ExecStart=/home/mobileoa/apps/shMediaManager.sh -start
    ExecReload=
    ExecStop=/home/mobileoa/apps/shMediaManager.sh -stop
    
    #建立私有的記憶體臨時空間
    PrivateTmp=True
    
    [Install]
    #多使用者
    WantedBy=multi-user.target
    

    過載系統服務:systemctl daemon-reload
    設定開機啟動:systemctl enable *.service
    啟動服務:systemctl start *.service
    停止服務:systemctl stop *.service
    重啟服務:systemctl reload *.service

注:修改完配置檔案要過載配置檔案。

相關文章