当任何服务的状态发生变化时,有没有办法运行脚本?

当任何服务的状态发生变化时,有没有办法运行脚本?

我想挂钩正在运行的服务的状态,以便当重要服务停止时,我可以运行脚本来通知我或采取预定义的操作来恢复服务。

答案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或。.SubStateorg.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,以便于解析。

相关内容