我想挂钩正在运行的服务的状态,以便当重要服务停止时,我可以运行脚本来通知我或采取预定义的操作来恢复服务。
答案1
不是任何服务。 systemd 管理器的 dbus 接口不提供这种类型的全局通知。如果您正在寻找将在全球范围内触发的信号,则经理的信号有点有限,可能是出于性能原因:
$ busctl introspect org.freedesktop.systemd1 \
/org/freedesktop/systemd1 \
org.freedesktop.systemd1.Manager \
| grep signal
.JobNew signal uos - -
.JobRemoved signal uoss - -
.Reloading signal b - -
.StartupFinished signal tttttt - -
.UnitFilesChanged signal - - -
.UnitNew signal so - -
.UnitRemoved signal so - -
完整界面可在此处查看:
https://www.freedesktop.org/software/systemd/man/org.freedesktop.systemd1.html
但是,如果您想知道特定服务是否已更改状态,您可以订阅该特定服务。示例:sshd.service:
$ busctl introspect \
org.freedesktop.systemd1 \
/org/freedesktop/systemd1/unit/sshd_2eservice \
org.freedesktop.DBus.Properties
NAME TYPE SIGNATURE RESULT/VALUE FLAGS
.Get method ss v -
.GetAll method s a{sv} -
.Set method ssv - -
.PropertiesChanged signal sa{sv}as - -
.PropertiesChanged
每当属性发生变化时就会发出信号。
您可能感兴趣的属性是接口上的.ActiveState
或。.SubState
org.freedesktop.systemd1.Unit
$ busctl introspect \
org.freedesktop.systemd1 \
/org/freedesktop/systemd1/unit/sshd_2eservice \
org.freedesktop.systemd1.Unit
...
.ActiveState property s "active" emits-change
.SubState property s "running" emits-change
那么您将如何订阅并获取这些数据呢?我不知道如何编写脚本,但我会使用 sd-bus API ( #include <systemd/sd-bus.h>
) 用 C 编写一些东西。这需要执行以下步骤:
- 开通巴士:
int r = sd_bus_default_system(&m_bus);
- 订阅某些单位的更改:
sd_bus_match_signal(
m_bus,
NULL, // slot
NULL, // sender
"/org/freedesktop/systemd1/unit/sshd_2eservice", // path
"org.freedesktop.DBus.Properties", // interface
"PropertiesChanged", // member
callback, // callback
NULL // userdata
);
sd_bus_match_signal(
m_bus,
NULL,
NULL,
"/org/freedesktop/systemd1/unit/network_2etarget",
"org.freedesktop.DBus.Properties",
"PropertiesChanged",
callback,
NULL
);
- 对变化做出反应并获取新状态:
int callback(sd_bus_message *m, void *userdata, sd_bus_error *ret_error)
{
sd_bus_error err = SD_BUS_ERROR_NULL;
char* msg = NULL;
int r;
r = sd_bus_get_property_string(m_bus,
"org.freedesktop.systemd1",
sd_bus_message_get_path(m), // (/org/freedesktop/systemd1/unit/sshd_2eservice)
"org.freedesktop.systemd1.Unit",
"ActiveState",
&err,
&msg);
if (r < 0) { /* Error */ }
sd_bus_error_free(&err);
printf("Property change, state is: %s\n", msg);
free (msg);
}
这里有一个相关的答案: https://stackoverflow.com/questions/61940461/how-to-get-the-state-of-a-service-with-sd-bus/62038047#62038047
答案2
一个简单的可能性是只尾部日志输出,journalctl -f
。有多种输出格式选项,包括 json,以便于解析。