0

我正在尝试使用 DBus 调用来管理 systemd 服务。但是,我似乎无法找出进行方法调用的正确方法,因为无论我做什么都会引发 DBus 错误

// Create connection
auto connection = Gio::DBus::Connection::get_sync(Gio::DBus::BusType::BUS_TYPE_SESSION);
// Create the parameters vector
std::vector<std::string> param_vec;
param_vec.push_back("my_unit_name");
auto param_vec_variant = Glib::Variant<std::vector<std::string>>::create(param_vec);
auto params = Glib::Variant<std::vector<std::string>>::create_tuple(param_vec_variant);

try {
    // Make the "GetUnit" dbus call to systemd
    auto response = connection->call_sync(
        "/org/freedesktop/systemd1",
        "org.freedesktop.systemd1.Manager",
        "GetUnit",
        params
    );
}
catch (Gio::DBus::Error err) {
    // org.freedesktop.DBus.Error.UnknownMethod being thrown
    std::cerr << Gio::DBus::ErrorUtils::get_remote_error(err) << std::endl;
}

谁能指出我哪里出错了?

4

1 回答 1

1

正如菲利普指出的那样,我的问题在于采用connection->call_sync默认参数bus_name = ""docs)的方法。

对于bus_namesystemd 是org.freedesktop.systemd1,所以这个问题的正确解决方案是添加这个参数:

auto response = connection->call_sync(
    "/org/freedesktop/systemd1",
    "org.freedesktop.systemd1.Manager",
    "GetUnit",
    params,
    "org.freedesktop.systemd1" // bus_name
);

但是,经过进一步挖掘,我发现实际上不建议将该Connection对象用于 DBus 调用,而我实际上应该使用Proxy.

最终,我使用可变参数概括了我的代码并选择了这个函数(为了便于阅读,删除了 try-catch 块):

template <typename... Types>
Glib::VariantContainerBase dbus_call(const Glib::ustring& method, const Types&... params) {
    auto proxy = Gio::DBus::Proxy::create_for_bus_sync(Gio::DBus::BusType::BUS_TYPE_SESSION, "org.freedesktop.systemd1", "/org/freedesktop/systemd1", "org.freedesktop.systemd1.Manager");
    auto params_variant = Glib::Variant<std::tuple<Types...>>::create(std::tuple<Types...>(params...));

    return proxy->call_sync(method, params_variant);
}

这可以这样调用:

auto response = dbus_call("GetUnit", "my_unit_name");
于 2021-06-30T09:09:11.140 回答