有没有办法检查是不是有人收听 dbus 信号?
Posted
技术标签:
【中文标题】有没有办法检查是不是有人收听 dbus 信号?【英文标题】:Is there a way to check if someone listens to dbus signal?有没有办法检查是否有人收听 dbus 信号? 【发布时间】:2015-07-23 16:05:42 【问题描述】:有没有办法在 DBus 中检查监听客户端?
有可能吗?我正在使用 gdbus。
背景
我正在创建与串行端口接口的服务,如果有人在监听,我想隐式打开串行端口,如果最后一个客户端断开连接,我想自动关闭它。我可以使用打开/关闭方法来做到这一点,但存在一个客户端在其他客户端仍在侦听时关闭连接的风险。
我的问题的另一个解决方案是连接计数,但也存在客户端忘记关闭端口或崩溃的风险。
您还有其他想法如何实现吗?
我的代码(缩短)
基于: https://github.com/bratsche/glib/blob/master/gio/tests/gdbus-example-server.c
#include <gio/gio.h>
#include <stdlib.h>
#ifdef G_OS_UNIX
#include <unistd.h>
#endif
/* ---------------------------------------------------------------------------------------------------- */
static GDBusNodeInfo *introspection_data = NULL;
/* Introspection data for the service we are exporting */
static const gchar introspection_xml[] =
"<node>"
" <interface name='info.skorepa.serial.port'>"
" <signal name='DataRecieved'>"
" <arg type='ay' name='data'/>"
" </signal>"
" </interface>"
"</node>";
/* ---------------------------------------------------------------------------------------------------- */
static void
handle_method_call (GDBusConnection *connection,
const gchar *sender,
const gchar *object_path,
const gchar *interface_name,
const gchar *method_name,
GVariant *parameters,
GDBusMethodInvocation *invocation,
gpointer user_data)
// nothing - signal only
static GVariant *
handle_get_property (GDBusConnection *connection,
const gchar *sender,
const gchar *object_path,
const gchar *interface_name,
const gchar *property_name,
GError **error,
gpointer user_data)
// nothing - signal only
static gboolean
handle_set_property (GDBusConnection *connection,
const gchar *sender,
const gchar *object_path,
const gchar *interface_name,
const gchar *property_name,
GVariant *value,
GError **error,
gpointer user_data)
// nothing - no properties
/* for now */
static const GDBusInterfaceVTable interface_vtable =
handle_method_call,
handle_get_property,
handle_set_property
;
/* ---------------------------------------------------------------------------------------------------- */
// Here I emit signal - for now I just emit every 2 seconds
static gboolean
on_timeout_cb (gpointer user_data)
GDBusConnection *connection = G_DBUS_CONNECTION (user_data);
GVariantBuilder *builder;
GVariantBuilder *invalidated_builder;
GError *error;
error = NULL;
printf("Constructing array\n");
builder = g_variant_builder_new (G_VARIANT_TYPE ("ay"));
printf("Adding 65\n");
g_variant_builder_add (builder,
"y",
65);
printf("Adding 66\n");
g_variant_builder_add (builder,
"y",
66);
printf("Emitting signal\n");
g_dbus_connection_emit_signal (connection,
NULL,
"/info/skorepa/TestObject",
"info.skorepa.serial.port",
"DataRecieved",
g_variant_new ("(ay)",
builder),
&error);
printf("Checking for errors\n");
g_assert_no_error (error);
return TRUE;
/* ---------------------------------------------------------------------------------------------------- */
static void
on_bus_acquired (GDBusConnection *connection,
const gchar *name,
gpointer user_data)
guint registration_id;
registration_id = g_dbus_connection_register_object (connection,
"/info/skorepa/TestObject",
introspection_data->interfaces[0],
&interface_vtable,
NULL, /* user_data */
NULL, /* user_data_free_func */
NULL); /* GError** */
g_assert (registration_id > 0);
/* swap value of properties Foo and Bar every two seconds */
g_timeout_add_seconds (2,
on_timeout_cb,
connection);
static void
on_name_acquired (GDBusConnection *connection,
const gchar *name,
gpointer user_data)
static void
on_name_lost (GDBusConnection *connection,
const gchar *name,
gpointer user_data)
exit (1);
int
main (int argc, char *argv[])
guint owner_id;
GMainLoop *loop;
g_type_init ();
introspection_data = g_dbus_node_info_new_for_xml (introspection_xml, NULL);
g_assert (introspection_data != NULL);
owner_id = g_bus_own_name (G_BUS_TYPE_SESSION,
"info.skorepa.serial",
G_BUS_NAME_OWNER_FLAGS_NONE,
on_bus_acquired,
on_name_acquired,
on_name_lost,
NULL,
NULL);
loop = g_main_loop_new (NULL, FALSE);
g_main_loop_run (loop);
g_bus_unown_name (owner_id);
g_dbus_node_info_unref (introspection_data);
return 0;
编译使用:
gcc signal-sample.c `pkg-config --cflags --libs glib-2.0 gio-2.0` -o test
谢谢
【问题讨论】:
为什么要投反对票?我想知道如何改进我的问题。 我建议将您的代码添加到您的问题中。这样您就有更好的机会获得有用的回复。 我也不确定您为什么也投了反对票。我不同意 Ted 的观点,因为我认为代码不会有帮助。显然,您对这个想法的探索还很早。但是,最好展示您为尝试解决问题所做的一些研究。您是否咨询过任何没有答案的 DBus 资源? 尝试阅读this DBus tutorial。 【参考方案1】:有没有办法在 DBus 中检查监听客户端?
不,这是不可能的,因为 D-Bus 的设计方式。
当客户端想要订阅一个信号时,他们会发送一个AddMatch
method call 给 D-Bus 守护进程,后者会在内部注册该状态。当您的服务发出信号时,它会将信号发送到 D-Bus 守护程序,然后由 D-Bus 守护程序将其转发给已订阅该信号的客户端(受有关广播和权限的各种策略规则的约束)。您的服务无法知道 D-Bus 守护程序中的内部订阅状态。
处理此类事情的模式是让您的服务显式公开subscribe
或open
方法,客户端必须调用这些方法才能打开串行端口。当客户端完成后,您可以使用第二种方法关闭串行端口;您还可以侦听客户端断开连接以自动关闭端口。 (在 GDBus 中,这是使用 g_bus_watch_name()
完成的,并将客户端的唯一名称传递给它,类似于 :1.5
。)
【讨论】:
以上是关于有没有办法检查是不是有人收听 dbus 信号?的主要内容,如果未能解决你的问题,请参考以下文章