Systemd定時器:三種使用場景
繼續(xù) systemd 教程,這些特殊的例子可以展示給你如何更好的利用 systemd 定時器單元。
在這個 systemd 系列教程中,我們已經(jīng)在某種程度上討論了 systemd 定時器單元。不過,在我們開始討論 sockets 之前,我們先來看三個例子,這些例子展示了如何***化利用這些單元。
簡單的類 cron 行為
我每周都要去收集 Debian popcon 數(shù)據(jù),如果每次都能在同一時間收集更好,這樣我就能看到某些應用程序的下載趨勢。這是一個可以使用 cron 任務來完成的典型事例,但 systemd 定時器同樣能做到:
# 類 cron 的 popcon.timer
[Unit]
Description= 這里描述了下載并處理 popcon 數(shù)據(jù)的時刻
[Timer]
OnCalendar= Thu *-*-* 05:32:07
Unit= popcon.service
[Install]
WantedBy= basic.target
實際的 popcon.service
會執(zhí)行一個常規(guī)的 wget
任務,并沒有什么特別之處。這里的新內(nèi)容是 OnCalendar=
指令。這個指令可以讓你在一個特定日期的特定時刻來運行某個服務。在這個例子中,Thu
表示 “在周四運行”,*-*-*
表示“具體年份、月份和日期無關緊要”,這些可以翻譯成 “不管年月日,只在每周四運行”。
這樣,你就設置了這個服務的運行時間。我選擇在歐洲中部夏令時區(qū)的上午 5:30 左右運行,那個時候服務器不是很忙。
如果你的服務器關閉了,而且剛好錯過了每周的截止時間,你還可以在同一個計時器中使用像 anacron 一樣的功能。
# 具備類似 anacron 功能的 popcon.timer
[Unit]
Description= 這里描述了下載并處理 popcon 數(shù)據(jù)的時刻
[Timer]
Unit=popcon.service
OnCalendar=Thu *-*-* 05:32:07
Persistent=true
[Install]
WantedBy=basic.target
當你將 Persistent=
指令設為真值時,它會告訴 systemd,如果服務器在本該它運行的時候關閉了,那么在啟動后就要立刻運行服務。這意味著,如果機器在周四凌晨停機了(比如說維護),一旦它再次啟動后,popcon.service
將會立刻執(zhí)行。在這之后,它的運行時間將會回到例行性的每周四早上 5:32.
到目前為止,就是這么簡單直白。
延遲執(zhí)行
但是,我們提升一個檔次,來“改進”這個基于 systemd 的監(jiān)控系統(tǒng)。你應該記得,當你接入攝像頭的時候,系統(tǒng)就會開始拍照。假設你并不希望它在你安裝攝像頭的時候拍下你的臉。你希望將拍照服務的啟動時間向后推遲一兩分鐘,這樣你就有時間接入攝像頭,然后走到畫框外面。
為了完成這件事,首先你要更改 Udev 規(guī)則,將它指向一個定時器:
ACTION=="add", SUBSYSTEM=="video4linux", ATTRS{idVendor}=="03f0",
ATTRS{idProduct}=="e207", TAG+="systemd", ENV{SYSTEMD_WANTS}="picchanged.timer",
SYMLINK+="mywebcam", MODE="0666"
這個定時器看起來像這樣:
# picchanged.timer
[Unit]
Description= 在攝像頭接入的一分鐘后,開始運行 picchanged
[Timer]
OnActiveSec= 1 m
Unit= picchanged.path
[Install]
WantedBy= basic.target
在你接入攝像頭后,Udev 規(guī)則被觸發(fā),它會調(diào)用定時器。這個定時器啟動后會等上一分鐘(OnActiveSec= 1 m
),然后運行 picchanged.path
,它會監(jiān)視主圖片的變化。picchanged.path
還會負責接觸 webcan.service
,這個實際用來拍照的服務。
在每天的特定時刻啟停 Minetest 服務器
在***一個例子中,我們認為你決定用 systemd 作為唯一的依賴。講真,不管怎么樣,systemd 差不多要接管你的生活了。為什么不擁抱這個必然性呢?
你有個為你的孩子設置的 Minetest 服務。不過,你還想要假裝關心一下他們的教育和成長,要讓他們做作業(yè)和家務活。所以你要確保 Minetest 只在每天晚上的一段時間內(nèi)可用,比如五點到七點。
這個跟之前的“在特定時間啟動服務”不太一樣。寫個定時器在下午五點啟動服務很簡單…:
# minetest.timer
[Unit]
Description= 在每天下午五點運行 minetest.service
[Timer]
OnCalendar= *-*-* 17:00:00
Unit= minetest.service
[Install]
WantedBy= basic.target
…可是編寫一個對應的定時器,讓它在特定時刻關閉服務,則需要更大劑量的橫向思維。
我們從最明顯的東西開始 —— 設置定時器:
# stopminetest.timer
[Unit]
Description= 每天晚上七點停止 minetest.service
[Timer]
OnCalendar= *-*-* 19:05:00
Unit= stopminetest.service
[Install]
WantedBy= basic.target
這里棘手的部分是如何去告訴 stopminetest.service
去 —— 你知道的 —— 停止 Minetest. 我們無法從 minetest.service
中傳遞 Minetest 服務器的 PID. 而且 systemd 的單元詞匯表中也沒有明顯的命令來停止或禁用正在運行的服務。
我們的訣竅是使用 systemd 的 Conflicts=
指令。它和 systemd 的 Wants=
指令類似,不過它所做的事情正相反。如果你有一個 b.service
單元,其中包含一個 Wants=a.service
指令,在這個單元啟動時,如果 a.service
沒有運行,則 b.service
會運行它。同樣,如果你的 b.service
單元中有一行寫著 Conflicts= a.service
,那么在 b.service
啟動時,systemd 會停止 a.service
.
這種機制用于兩個服務在嘗試同時控制同一資源時會發(fā)生沖突的場景,例如當兩個服務要同時訪問打印機的時候。通過在***服務中設置 Conflicts=
,你就可以確保它會覆蓋掉最不重要的服務。
不過,你會在一個稍微不同的場景中來使用 Conflicts=
. 你將使用 Conflicts=
來干凈地關閉 minetest.service
:
# stopminetest.service
[Unit]
Description= 關閉 Minetest 服務
Conflicts= minetest.service
[Service]
Type= oneshot
ExecStart= /bin/echo "Closing down minetest.service"
stopminetest.service
并不會做特別的東西。事實上,它什么都不會做。不過因為它包含那行 Conflicts=
,所以在它啟動時,systemd 會關掉 minetest.service
.
在你***的 Minetest 設置中,還有***一點漣漪:你下班晚了,錯過了服務器的開機時間,可當你開機的時候游戲時間還沒結(jié)束,這該怎么辦?Persistent=
指令(如上所述)在錯過開始時間后仍然可以運行服務,但這個方案還是不行。如果你在早上十一點把服務器打開,它就會啟動 Minetest,而這不是你想要的。你真正需要的是一個確保 systemd 只在晚上五到七點啟動 Minetest 的方法:
# minetest.timer
[Unit]
Description= 在下午五到七點內(nèi)的每分鐘都運行 minetest.service
[Timer]
OnCalendar= *-*-* 17..19:*:00
Unit= minetest.service
[Install]
WantedBy= basic.target
OnCalendar= *-*-* 17..19:*:00
這一行有兩個有趣的地方:(1) 17..19
并不是一個時間點,而是一個時間段,在這個場景中是 17 到 19 點;以及,(2) 分鐘字段中的 *
表示服務每分鐘都要運行。因此,你會把它讀做 “在下午五到七點間的每分鐘,運行 minetest.service”
不過還有一個問題:一旦 minetest.service
啟動并運行,你會希望 minetest.timer
不要再次嘗試運行它。你可以在 minetest.service
中包含一條 Conflicts=
指令:
# minetest.service
[Unit]
Description= 運行 Minetest 服務器
Conflicts= minetest.timer
[Service]
Type= simple
User= <your user name>
ExecStart= /usr/bin/minetest --server
ExecStop= /bin/kill -2 $MAINPID
[Install]
WantedBy= multi-user.targe
上面的 Conflicts=
指令會保證在 minstest.service
成功運行后,minetest.timer
就會立即停止。
現(xiàn)在,啟用并啟動 minetest.timer
:
systemctl enable minetest.timer
systemctl start minetest.timer
而且,如果你在六點鐘啟動了服務器,minetest.timer
會啟用;到了五到七點,minetest.timer
每分鐘都會嘗試啟動 minetest.service
。不過,一旦 minetest.service
開始運行,systemd 會停止 minetest.timer
,因為它會與 minetest.service
“沖突”,從而避免計時器在服務已經(jīng)運行的情況下還會不斷嘗試啟動服務。
在首先啟動某個服務時殺死啟動它的計時器,這么做有點反直覺,但它是有效的。
總結(jié)
你可能會認為,有更好的方式來做上面這些事。我在很多文章中看到過“過度設計”這個術語,尤其是在用 systemd 定時器來代替 cron 的時候。
但是,這個系列文章的目的不是為任何具體問題提供***解決方案。它的目的是為了盡可能多地使用 systemd 來解決問題,甚至會到荒唐的程度。它的目的是展示大量的例子,來說明如何利用不同類型的單位及其包含的指令。我們的讀者,也就是你,可以從這篇文章中找到所有這些的可實踐范例。
盡管如此,我們還有一件事要做:下回中,我們會關注 sockets 和 targets,然后我們將完成對 systemd 單元的介紹。