我正在尝试创建一个 systemctl 单元,只有当另一个单元处于活动状态时,该单元才会启动。据我所知,可以通过指定一个Requisite
字段来完成。因此,我已按如下方式配置我的单元:
[Unit]
Description=Force clients with mismatching IPs to reconnect
[email protected]
[Service]
Type=oneshot
ExecStart=/etc/openvpn/kicker/kicker.py
TimeoutSec=15
但是,当我使用systemctl stop openvpn@server
和时systemctl start my-unit
,我收到以下日志:
Starting openvpn-kicker.service - Force clients with mismatching IPs to reconnect...
Dependency failed for openvpn-kicker.service - Force clients with mismatching IPs to reconnect.
openvpn-kicker.service: Job openvpn-kicker.service/start failed with result 'dependency'.
kicker.py[1026]: Traceback (most recent call last):
kicker.py[1026]: File "/etc/openvpn/kicker/kicker.py", line 16, in <module>
kicker.py[1026]: manage.connect(Manage_Path)
kicker.py[1026]: FileNotFoundError: [Errno 2] No such file or directory
Main process exited, code=exited, status=1/FAILURE
openvpn-kicker.service: Failed with result 'exit-code'.
因此,基本上,单元在需求检查时失败,但无论如何都会在“ExecStart”中执行脚本 - 因此出现 python 错误。它是否打算以这种方式工作,而我弄错了什么,或者这是一个错误?如果相关,操作系统是 Ubuntu 22.04
首先,请注意,在 systemd 中,需求依赖关系和顺序依赖关系是正交的。一个单元可以对另一个单元有弱/中等/强需求(
Wants
、Requires
、Requisite
等),但顺序必须使用Before
/单独指定After
。可以这样想:Wants=foo
:当我开始时,foo
也要求开始,但我并不真正关心它会发生什么。Requires=foo
:当我启动时,foo
也请求启动,如果foo
的启动失败,则认为我的启动也失败,而不管对我的启动所采取的实际操作的状态如何。Requisite=foo
:当我启动时,foo
必须已经启动。如果没有,则认为我的启动失败,无论对我的启动采取的实际操作的状态如何。因此,我可以拥有服务 A 和
Requires
B,但可能有 B 的排序Before
。当 B 正在做一些可以独立于 A 完成的一般性事情时,可能会出现这种情况,但 A 总是希望 B 在 A 启动后再做那件事。特别是对于
Requisite
,文档特别建议将其与以下一起使用After
:第二件要理解的事情是,作业和单元在 systemd 中是独立但相关的概念。有关更多信息,请参阅例如此答案或此答案。因此,当文档说:
这意味着该单元的启动工作将失败。但实际启动将继续进行。其工作方式如下:
systemctl start foo
,或者其他导致foo
启动的操作(例如,另一个单元、目标等的依赖)。Requisite
[在 (5) 之前的某个时间] 对非活动单元的查询导致作业状态设置为失败。Before
顺序排列的单元的作业。Required
或Requisite
更严重),则它将继续实际运行ExecStart*
命令。请注意,这意味着如果排序中它之前没有其他作业,则systemd 将立即继续执行此步骤。Requisite
如果某个单元未处于活动状态或者由于其他一系列原因(例如实际运行命令失败等),此操作将会失败。为了使 (2.1) 中的检查对 (4) 中的命令执行有任何意义,systemd 应该在 (3) 中考虑一个顺序。在本例中,没有。
引用Lennart Poettering 的话: