为什么我的 DBus 调用给出 UnknownMethod DBus 错误?

问题描述

我正在尝试使用 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;
}

谁能指出我哪里出错了?

解决方法

正如 Philip 指出的那样,我的问题出在 connection->call_sync 方法上,该方法采用默认参数 bus_name = "" (docs)。

systemd 的 bus_nameorg.freedesktop.systemd1,因此解决此问题的正确方法是添加此参数:

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

但是,在进一步挖掘之后,我发现实际上是 not recommended 使用 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");