リロードを使用した Systemd アップグレードバイナリ

リロードを使用した Systemd アップグレードバイナリ

SIGUSR2 を受け取ると、ダウンタイムなしでフォークして新しいプロセスを適切に作成する (既存のすべての親ソケットを子に渡し、親を強制終了する) プログラムがあります。通常、systemd を使用しない場合は次のようになります。

cp newbinary coredns
kill -s USR2 oldpid

SystemdはExecReloadこれによると答え

言い換えれば、systemd は、基礎となるサービスが真のリロード機能をサポートしている場合にのみ「リロード」を実装することを望んでいます。つまり、サービスを強制終了して再起動したり、サービスの PID を変更したりしないリロードです。言い換えれば、systemd は存在する機能のみを反映することを望んでいます。

systemd ファイルは次のようになります:

PIDFile=/tmp/coredns.pid
Type=forking
ExecStart=./coredns -pidfile /tmp/coredns.pid
ExecStop=kill `/tmp/coredns.pid`
ExecReload=/bin/kill -s USR2 `cat /tmp/coredns.pid`

問題は、 を実行するときにsudo systemctl reload coredns、systemd は次の操作を実行するかどうかです。

  1. pidが変更されたことを知っていますか?
  2. ログとプロセスの監視を適切に処理していますか?

答え1

OK、forkingまたはoneoff起動時にスタックする場合は、タイプが である必要がありsimple、正常に動作します。

echo '[Unit]
Description=Custom CoreDNS DNS server
After=network.target
[Service]
PermissionsStartOnly=true
LimitNOFILE=1048576
LimitNPROC=512
CapabilityBoundingSet=CAP_NET_BIND_SERVICE
AmbientCapabilities=CAP_NET_BIND_SERVICE
WorkingDirectory=/tmp/1
Type=simple
RemainAfterExit=yes
ExecStart=/tmp/1/start.sh
ExecStop=/tmp/1/kill.sh
ExecReload=/tmp/1/reload.sh
Restart=on-failure
[Install]
WantedBy=multi-user.target 
' | sudo tee /usr/lib/systemd/system/coredns.service
sudo systemctl daemon-reload 

スクリプトを作成します:

mkdir -p /tmp/1
echo '#!/usr/bin/bash
kill -s USR2 `cat /tmp/1/coredns.pid`
' > /tmp/1/reload.sh
echo '#!/usr/bin/bash
kill -s USR2 `cat /tmp/1/coredns.pid`
' > /tmp/1/kill.sh
echo '#!/usr/bin/bash
/tmp/1/coredns -pidfile /tmp/1/coredns.pid -conf=/tmp/1/Corefile
' > /tmp/1/start.sh
chmod a+x /tmp/1/*.sh
chmod 777 /tmp/1 # just so that no need to set user permission on systemd
cp Corefile coredns /tmp/1/

正常に起動できます:

sudo systemctl start coredns

# on journalctl:
Apr 20 18:50:06 pop2204 systemd[1]: Started Custom CoreDNS DNS server.
Apr 20 18:50:06 pop2204 start.sh[1553313]: whoami Name called
Apr 20 18:50:06 pop2204 start.sh[1553313]: whoami Name called
Apr 20 18:50:06 pop2204 start.sh[1553313]: whoami Name called
Apr 20 18:50:06 pop2204 start.sh[1553313]: .:1053
Apr 20 18:50:06 pop2204 start.sh[1553313]: CoreDNS-1.9.4
Apr 20 18:50:06 pop2204 start.sh[1553313]: linux/amd64, go1.20.3, 81159d2f-dirty

古いバイナリを削除し(直接コピーするとテキストファイルがビジーエラーになるので)、新しいバイナリをコピーします。

rm /tmp/1/coredns
cp coredns /tmp/1/coredns
sudo systemctl reload coredns

# from journalctl:
Apr 20 18:52:03 pop2204 systemd[1]: Reloading Custom CoreDNS DNS server...
Apr 20 18:52:03 pop2204 start.sh[1553313]: [INFO] SIGUSR2: Upgrading
Apr 20 18:52:03 pop2204 start.sh[1553313]: [INFO] Upgrading
Apr 20 18:52:03 pop2204 systemd[1]: Reloaded Custom CoreDNS DNS server.
Apr 20 18:52:03 pop2204 start.sh[1553726]: whoami Name called 2
Apr 20 18:52:03 pop2204 start.sh[1553726]: whoami Name called 2
Apr 20 18:52:03 pop2204 start.sh[1553726]: whoami Name called 2
Apr 20 18:52:03 pop2204 start.sh[1553726]: .:1053
Apr 20 18:52:03 pop2204 start.sh[1553313]: [INFO] Upgrade finished
Apr 20 18:52:03 pop2204 start.sh[1553726]: CoreDNS-1.9.4
Apr 20 18:52:03 pop2204 start.sh[1553726]: linux/amd64, go1.20.3, 81159d2f-dirty

関連情報