在Linux上使用systemd設(shè)置定時(shí)器
學(xué)習(xí)使用 systemd 創(chuàng)建啟動(dòng)你的游戲服務(wù)器的定時(shí)器。
之前,我們看到了如何手動(dòng)的、在開機(jī)與關(guān)機(jī)時(shí)、在啟用某個(gè)設(shè)備時(shí)、在文件系統(tǒng)發(fā)生改變時(shí) 啟用與禁用 systemd 服務(wù)。
定時(shí)器增加了另一種啟動(dòng)服務(wù)的方式,基于……時(shí)間。盡管與定時(shí)任務(wù)很相似,但 systemd 定時(shí)器稍微地靈活一些。讓我們看看它是怎么工作的。
“定時(shí)運(yùn)行”
讓我們展開本系列前兩篇文章中你所設(shè)置的 Minetest 服務(wù)器作為如何使用定時(shí)器單元的***個(gè)例子。如果你還沒有讀過那幾篇文章,可以現(xiàn)在去看看。
你將通過創(chuàng)建一個(gè)定時(shí)器來“改進(jìn)” Minetest 服務(wù)器,使得在服務(wù)器啟動(dòng) 1 分鐘后運(yùn)行游戲服務(wù)器而不是立即運(yùn)行。這樣做的原因可能是,在啟動(dòng)之前可能會(huì)用到其他的服務(wù),例如發(fā)郵件給其他玩家告訴他們游戲已經(jīng)準(zhǔn)備就緒,你要確保其他的服務(wù)(例如網(wǎng)絡(luò))在開始前完全啟動(dòng)并運(yùn)行。
最終,你的 minetest.timer 單元看起來就像這樣:
# minetest.timer[Unit]Description=Runs the minetest.service 1 minute after boot up[Timer]OnBootSec=1 mUnit=minetest.service[Install]WantedBy=basic.target
一點(diǎn)也不難吧。
如以往一般,開頭是 [Unit] 和一段描述單元作用的信息,這兒沒什么新東西。[Timer] 這一節(jié)是新出現(xiàn)的,但它的作用不言自明:它包含了何時(shí)啟動(dòng)服務(wù),啟動(dòng)哪個(gè)服務(wù)的信息。在這個(gè)例子當(dāng)中,OnBootSec 是告訴 systemd 在系統(tǒng)啟動(dòng)后運(yùn)行服務(wù)的指令。
其他的指令有:
OnActiveSec=,告訴 systemd 在定時(shí)器啟動(dòng)后多長時(shí)間運(yùn)行服務(wù)。OnStartupSec=,同樣的,它告訴 systemd 在 systemd 進(jìn)程啟動(dòng)后多長時(shí)間運(yùn)行服務(wù)。OnUnitActiveSec=,告訴 systemd 在上次由定時(shí)器激活的服務(wù)啟動(dòng)后多長時(shí)間運(yùn)行服務(wù)。OnUnitInactiveSec=,告訴 systemd 在上次由定時(shí)器激活的服務(wù)停用后多長時(shí)間運(yùn)行服務(wù)。
繼續(xù) minetest.timer 單元,basic.target 通常用作后期引導(dǎo)服務(wù)的同步點(diǎn)。這就意味著它可以讓 minetest.timer 單元運(yùn)行在安裝完本地掛載點(diǎn)或交換設(shè)備,套接字、定時(shí)器、路徑單元和其他基本的初始化進(jìn)程之后。就像在第二篇文章中 systemd 單元里解釋的那樣,targets 就像舊的運(yùn)行等級(jí)一樣,可以將你的計(jì)算機(jī)置于某個(gè)狀態(tài),或像這樣告訴你的服務(wù)在達(dá)到某個(gè)狀態(tài)后開始運(yùn)行。
在前兩篇文章中你配置的 minetest.service 文件最終看起來就像這樣:
# minetest.service[Unit]Description= Minetest serverDocumentation= https://wiki.minetest.net/Main_Page[Service]Type= simpleUser=ExecStart= /usr/games/minetest --serverExecStartPost= /home//bin/mtsendmail.sh "Ready to rumble?" "Minetest Starting up"TimeoutStopSec= 180ExecStop= /home//bin/mtsendmail.sh "Off to bed. Nightie night!" "Minetest Stopping in 2 minutes"ExecStop= /bin/sleep 120ExecStop= /bin/kill -2 $MAINPID[Install]WantedBy= multi-user.target
這兒沒什么需要修改的。但是你需要將 mtsendmail.sh(發(fā)送你的 email 的腳本)從:
#!/bin/bash# mtsendmailsleep 20echo $1 | mutt -F /home/<username>/.muttrc -s "$2" my_minetest@mailing_list.comsleep 10
改成:
#!/bin/bash# mtsendmail.shecho $1 | mutt -F /home/paul/.muttrc -s "$2" pbrown@mykolab.com
你做的事是去除掉 Bash 腳本中那些蹩腳的停頓。Systemd 現(xiàn)在來做等待。
讓它運(yùn)行起來
確保一切運(yùn)作正常,禁用 minetest.service:
sudo systemctl disable minetest
這使得系統(tǒng)啟動(dòng)時(shí)它不會(huì)一同啟動(dòng);然后,相反地,啟用 minetest.timer:
sudo systemctl enable minetest.timer
現(xiàn)在你就可以重啟服務(wù)器了,當(dāng)運(yùn)行 sudo journalctl -u minetest.* 后,你就會(huì)看到 minetest.timer 單元執(zhí)行后大約一分鐘,minetest.service 單元開始運(yùn)行。

圖 1:minetest.timer 運(yùn)行大約 1 分鐘后 minetest.service 開始運(yùn)行
時(shí)間的問題
minetest.timer 在 systemd 的日志里顯示的啟動(dòng)時(shí)間為 09:08:33 而 minetest.service 啟動(dòng)時(shí)間是 09:09:18,它們之間少于 1 分鐘,關(guān)于這件事有幾點(diǎn)需要說明一下:首先,請(qǐng)記住我們說過 OnBootSec= 指令是從引導(dǎo)完成后開始計(jì)算服務(wù)啟動(dòng)的時(shí)間。當(dāng) minetest.timer 的時(shí)間到來時(shí),引導(dǎo)已經(jīng)在幾秒之前完成了。
另一件事情是 systemd 給自己設(shè)置了一個(gè)誤差幅度(默認(rèn)是 1 分鐘)來運(yùn)行東西。這有助于在多個(gè)資源密集型進(jìn)程同時(shí)運(yùn)行時(shí)分配負(fù)載:通過分配 1 分鐘的時(shí)間,systemd 可以等待某些進(jìn)程關(guān)閉。這也意味著 minetest.service 會(huì)在引導(dǎo)完成后的 1~2 分鐘之間啟動(dòng)。但精確的時(shí)間誰也不知道。
順便一提,你可以用 AccuracySec= 指令修改誤差幅度。
你也可以檢查系統(tǒng)上所有的定時(shí)器何時(shí)運(yùn)行或是上次運(yùn)行的時(shí)間:
systemctl list-timers --all

圖 2:檢查定時(shí)器何時(shí)運(yùn)行或上次運(yùn)行的時(shí)間
***一件值得思考的事就是你應(yīng)該用怎樣的格式去表示一段時(shí)間。Systemd 在這方面非常靈活:2 h,2 hours 或 2hr 都可以用來表示 2 個(gè)小時(shí)。對(duì)于“秒”,你可以用 seconds,second,sec 和 s。“分”也是同樣的方式:minutes,minute,min 和 m。你可以檢查 man systemd.time 來查看 systemd 能夠理解的所有時(shí)間單元。
下一次
下次你會(huì)看到如何使用日歷中的日期和時(shí)間來定期運(yùn)行服務(wù),以及如何通過組合定時(shí)器與設(shè)備單元在插入某些硬件時(shí)運(yùn)行服務(wù)。






















