mirror of
https://gitlab.gnome.org/GNOME/glib.git
synced 2025-01-15 00:36:19 +01:00
c3371efcaa
Rework filter functions as per https://bugzilla.gnome.org/show_bug.cgi?id=624546#c8 This commit breaks ABI. However, this ABI break affects only applications using filter functions. The only known user of is dconf. Signed-off-by: David Zeuthen <davidz@redhat.com>
1188 lines
41 KiB
C
1188 lines
41 KiB
C
/* GLib testing framework examples and tests
|
|
*
|
|
* Copyright (C) 2008-2010 Red Hat, Inc.
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General
|
|
* Public License along with this library; if not, write to the
|
|
* Free Software Foundation, Inc., 59 Temple Place, Suite 330,
|
|
* Boston, MA 02111-1307, USA.
|
|
*
|
|
* Author: David Zeuthen <davidz@redhat.com>
|
|
*/
|
|
|
|
#include <gio/gio.h>
|
|
#include <unistd.h>
|
|
#include <string.h>
|
|
|
|
#include <sys/types.h>
|
|
#include <sys/wait.h>
|
|
|
|
#include "gdbus-tests.h"
|
|
|
|
/* all tests rely on a shared mainloop */
|
|
static GMainLoop *loop = NULL;
|
|
|
|
static gboolean
|
|
test_connection_quit_mainloop (gpointer user_data)
|
|
{
|
|
gboolean *quit_mainloop_fired = user_data;
|
|
*quit_mainloop_fired = TRUE;
|
|
g_main_loop_quit (loop);
|
|
return TRUE;
|
|
}
|
|
|
|
/* ---------------------------------------------------------------------------------------------------- */
|
|
/* Connection life-cycle testing */
|
|
/* ---------------------------------------------------------------------------------------------------- */
|
|
|
|
static const GDBusInterfaceInfo boo_interface_info =
|
|
{
|
|
-1,
|
|
"org.example.Boo",
|
|
(GDBusMethodInfo **) NULL,
|
|
(GDBusSignalInfo **) NULL,
|
|
(GDBusPropertyInfo **) NULL,
|
|
NULL,
|
|
};
|
|
|
|
static const GDBusInterfaceVTable boo_vtable =
|
|
{
|
|
NULL, /* _method_call */
|
|
NULL, /* _get_property */
|
|
NULL /* _set_property */
|
|
};
|
|
|
|
static GDBusMessage *
|
|
some_filter_func (GDBusConnection *connection,
|
|
GDBusMessage *message,
|
|
gboolean incoming,
|
|
gpointer user_data)
|
|
{
|
|
return message;
|
|
}
|
|
|
|
static void
|
|
on_name_owner_changed (GDBusConnection *connection,
|
|
const gchar *sender_name,
|
|
const gchar *object_path,
|
|
const gchar *interface_name,
|
|
const gchar *signal_name,
|
|
GVariant *parameters,
|
|
gpointer user_data)
|
|
{
|
|
}
|
|
|
|
static void
|
|
a_gdestroynotify_that_sets_a_gboolean_to_true_and_quits_loop (gpointer user_data)
|
|
{
|
|
gboolean *val = user_data;
|
|
*val = TRUE;
|
|
|
|
g_main_loop_quit (loop);
|
|
}
|
|
|
|
static void
|
|
test_connection_life_cycle (void)
|
|
{
|
|
gboolean ret;
|
|
GDBusConnection *c;
|
|
GDBusConnection *c2;
|
|
GError *error;
|
|
gboolean on_signal_registration_freed_called;
|
|
gboolean on_filter_freed_called;
|
|
gboolean on_register_object_freed_called;
|
|
gboolean quit_mainloop_fired;
|
|
guint quit_mainloop_id;
|
|
guint registration_id;
|
|
|
|
error = NULL;
|
|
|
|
/*
|
|
* Check for correct behavior when no bus is present
|
|
*
|
|
*/
|
|
c = g_bus_get_sync (G_BUS_TYPE_SESSION, NULL, &error);
|
|
_g_assert_error_domain (error, G_IO_ERROR);
|
|
g_assert (!g_dbus_error_is_remote_error (error));
|
|
g_assert (c == NULL);
|
|
g_error_free (error);
|
|
error = NULL;
|
|
|
|
/*
|
|
* Check for correct behavior when a bus is present
|
|
*/
|
|
session_bus_up ();
|
|
/* case 1 */
|
|
c = g_bus_get_sync (G_BUS_TYPE_SESSION, NULL, &error);
|
|
g_assert_no_error (error);
|
|
g_assert (c != NULL);
|
|
g_assert (!g_dbus_connection_is_closed (c));
|
|
|
|
/*
|
|
* Check that singleton handling work
|
|
*/
|
|
c2 = g_bus_get_sync (G_BUS_TYPE_SESSION, NULL, &error);
|
|
g_assert_no_error (error);
|
|
g_assert (c2 != NULL);
|
|
g_assert (c == c2);
|
|
g_object_unref (c2);
|
|
|
|
/*
|
|
* Check that private connections work
|
|
*/
|
|
c2 = _g_bus_get_priv (G_BUS_TYPE_SESSION, NULL, &error);
|
|
g_assert_no_error (error);
|
|
g_assert (c2 != NULL);
|
|
g_assert (c != c2);
|
|
g_object_unref (c2);
|
|
|
|
c2 = _g_bus_get_priv (G_BUS_TYPE_SESSION, NULL, &error);
|
|
g_assert_no_error (error);
|
|
g_assert (c2 != NULL);
|
|
g_assert (!g_dbus_connection_is_closed (c2));
|
|
ret = g_dbus_connection_close_sync (c2, NULL, &error);
|
|
g_assert_no_error (error);
|
|
g_assert (ret);
|
|
_g_assert_signal_received (c2, "closed");
|
|
g_assert (g_dbus_connection_is_closed (c2));
|
|
ret = g_dbus_connection_close_sync (c2, NULL, &error);
|
|
g_assert_error (error, G_IO_ERROR, G_IO_ERROR_CLOSED);
|
|
g_error_free (error);
|
|
g_assert (!ret);
|
|
g_object_unref (c2);
|
|
|
|
/*
|
|
* Check that the finalization code works
|
|
*
|
|
* (and that the GDestroyNotify for filters and objects and signal
|
|
* registrations are run as expected)
|
|
*/
|
|
error = NULL;
|
|
c2 = _g_bus_get_priv (G_BUS_TYPE_SESSION, NULL, &error);
|
|
g_assert_no_error (error);
|
|
g_assert (c2 != NULL);
|
|
/* signal registration */
|
|
on_signal_registration_freed_called = FALSE;
|
|
g_dbus_connection_signal_subscribe (c2,
|
|
"org.freedesktop.DBus", /* bus name */
|
|
"org.freedesktop.DBus", /* interface */
|
|
"NameOwnerChanged", /* member */
|
|
"/org/freesktop/DBus", /* path */
|
|
NULL, /* arg0 */
|
|
G_DBUS_SIGNAL_FLAGS_NONE,
|
|
on_name_owner_changed,
|
|
&on_signal_registration_freed_called,
|
|
a_gdestroynotify_that_sets_a_gboolean_to_true_and_quits_loop);
|
|
/* filter func */
|
|
on_filter_freed_called = FALSE;
|
|
g_dbus_connection_add_filter (c2,
|
|
some_filter_func,
|
|
&on_filter_freed_called,
|
|
a_gdestroynotify_that_sets_a_gboolean_to_true_and_quits_loop);
|
|
/* object registration */
|
|
on_register_object_freed_called = FALSE;
|
|
error = NULL;
|
|
registration_id = g_dbus_connection_register_object (c2,
|
|
"/foo",
|
|
(GDBusInterfaceInfo *) &boo_interface_info,
|
|
&boo_vtable,
|
|
&on_register_object_freed_called,
|
|
a_gdestroynotify_that_sets_a_gboolean_to_true_and_quits_loop,
|
|
&error);
|
|
g_assert_no_error (error);
|
|
g_assert (registration_id > 0);
|
|
/* ok, finalize the connection and check that all the GDestroyNotify functions are invoked as expected */
|
|
g_object_unref (c2);
|
|
quit_mainloop_fired = FALSE;
|
|
quit_mainloop_id = g_timeout_add (1000, test_connection_quit_mainloop, &quit_mainloop_fired);
|
|
while (TRUE)
|
|
{
|
|
if (on_signal_registration_freed_called &&
|
|
on_filter_freed_called &&
|
|
on_register_object_freed_called)
|
|
break;
|
|
if (quit_mainloop_fired)
|
|
break;
|
|
g_main_loop_run (loop);
|
|
}
|
|
g_source_remove (quit_mainloop_id);
|
|
g_assert (on_signal_registration_freed_called);
|
|
g_assert (on_filter_freed_called);
|
|
g_assert (on_register_object_freed_called);
|
|
g_assert (!quit_mainloop_fired);
|
|
|
|
/*
|
|
* Check for correct behavior when the bus goes away
|
|
*
|
|
*/
|
|
g_assert (!g_dbus_connection_is_closed (c));
|
|
g_dbus_connection_set_exit_on_close (c, FALSE);
|
|
session_bus_down ();
|
|
if (!g_dbus_connection_is_closed (c))
|
|
_g_assert_signal_received (c, "closed");
|
|
g_assert (g_dbus_connection_is_closed (c));
|
|
|
|
_g_object_wait_for_single_ref (c);
|
|
g_object_unref (c);
|
|
}
|
|
|
|
/* ---------------------------------------------------------------------------------------------------- */
|
|
/* Test that sending and receiving messages work as expected */
|
|
/* ---------------------------------------------------------------------------------------------------- */
|
|
|
|
static void
|
|
msg_cb_expect_error_disconnected (GDBusConnection *connection,
|
|
GAsyncResult *res,
|
|
gpointer user_data)
|
|
{
|
|
GError *error;
|
|
GVariant *result;
|
|
|
|
error = NULL;
|
|
result = g_dbus_connection_call_finish (connection,
|
|
res,
|
|
&error);
|
|
g_assert_error (error, G_IO_ERROR, G_IO_ERROR_CLOSED);
|
|
g_assert (!g_dbus_error_is_remote_error (error));
|
|
g_error_free (error);
|
|
g_assert (result == NULL);
|
|
|
|
g_main_loop_quit (loop);
|
|
}
|
|
|
|
static void
|
|
msg_cb_expect_error_unknown_method (GDBusConnection *connection,
|
|
GAsyncResult *res,
|
|
gpointer user_data)
|
|
{
|
|
GError *error;
|
|
GVariant *result;
|
|
|
|
error = NULL;
|
|
result = g_dbus_connection_call_finish (connection,
|
|
res,
|
|
&error);
|
|
g_assert_error (error, G_DBUS_ERROR, G_DBUS_ERROR_UNKNOWN_METHOD);
|
|
g_assert (g_dbus_error_is_remote_error (error));
|
|
g_error_free (error);
|
|
g_assert (result == NULL);
|
|
|
|
g_main_loop_quit (loop);
|
|
}
|
|
|
|
static void
|
|
msg_cb_expect_success (GDBusConnection *connection,
|
|
GAsyncResult *res,
|
|
gpointer user_data)
|
|
{
|
|
GError *error;
|
|
GVariant *result;
|
|
|
|
error = NULL;
|
|
result = g_dbus_connection_call_finish (connection,
|
|
res,
|
|
&error);
|
|
g_assert_no_error (error);
|
|
g_assert (result != NULL);
|
|
g_variant_unref (result);
|
|
|
|
g_main_loop_quit (loop);
|
|
}
|
|
|
|
static void
|
|
msg_cb_expect_error_cancelled (GDBusConnection *connection,
|
|
GAsyncResult *res,
|
|
gpointer user_data)
|
|
{
|
|
GError *error;
|
|
GVariant *result;
|
|
|
|
error = NULL;
|
|
result = g_dbus_connection_call_finish (connection,
|
|
res,
|
|
&error);
|
|
g_assert_error (error, G_IO_ERROR, G_IO_ERROR_CANCELLED);
|
|
g_assert (!g_dbus_error_is_remote_error (error));
|
|
g_error_free (error);
|
|
g_assert (result == NULL);
|
|
|
|
g_main_loop_quit (loop);
|
|
}
|
|
|
|
static void
|
|
msg_cb_expect_error_cancelled_2 (GDBusConnection *connection,
|
|
GAsyncResult *res,
|
|
gpointer user_data)
|
|
{
|
|
GError *error;
|
|
GVariant *result;
|
|
|
|
error = NULL;
|
|
result = g_dbus_connection_call_finish (connection,
|
|
res,
|
|
&error);
|
|
g_assert_error (error, G_IO_ERROR, G_IO_ERROR_CANCELLED);
|
|
g_assert (!g_dbus_error_is_remote_error (error));
|
|
g_error_free (error);
|
|
g_assert (result == NULL);
|
|
|
|
g_main_loop_quit (loop);
|
|
}
|
|
|
|
/* ---------------------------------------------------------------------------------------------------- */
|
|
|
|
static void
|
|
test_connection_send (void)
|
|
{
|
|
GDBusConnection *c;
|
|
GCancellable *ca;
|
|
|
|
session_bus_up ();
|
|
|
|
/* First, get an unopened connection */
|
|
c = g_bus_get_sync (G_BUS_TYPE_SESSION, NULL, NULL);
|
|
g_assert (c != NULL);
|
|
g_assert (!g_dbus_connection_is_closed (c));
|
|
|
|
/*
|
|
* Check that we never actually send a message if the GCancellable
|
|
* is already cancelled - i.e. we should get #G_IO_ERROR_CANCELLED
|
|
* when the actual connection is not up.
|
|
*/
|
|
ca = g_cancellable_new ();
|
|
g_cancellable_cancel (ca);
|
|
g_dbus_connection_call (c,
|
|
"org.freedesktop.DBus", /* bus_name */
|
|
"/org/freedesktop/DBus", /* object path */
|
|
"org.freedesktop.DBus", /* interface name */
|
|
"GetId", /* method name */
|
|
NULL, NULL,
|
|
G_DBUS_CALL_FLAGS_NONE,
|
|
-1,
|
|
ca,
|
|
(GAsyncReadyCallback) msg_cb_expect_error_cancelled,
|
|
NULL);
|
|
g_main_loop_run (loop);
|
|
g_object_unref (ca);
|
|
|
|
/*
|
|
* Check that we get a reply to the GetId() method call.
|
|
*/
|
|
g_dbus_connection_call (c,
|
|
"org.freedesktop.DBus", /* bus_name */
|
|
"/org/freedesktop/DBus", /* object path */
|
|
"org.freedesktop.DBus", /* interface name */
|
|
"GetId", /* method name */
|
|
NULL, NULL,
|
|
G_DBUS_CALL_FLAGS_NONE,
|
|
-1,
|
|
NULL,
|
|
(GAsyncReadyCallback) msg_cb_expect_success,
|
|
NULL);
|
|
g_main_loop_run (loop);
|
|
|
|
/*
|
|
* Check that we get an error reply to the NonExistantMethod() method call.
|
|
*/
|
|
g_dbus_connection_call (c,
|
|
"org.freedesktop.DBus", /* bus_name */
|
|
"/org/freedesktop/DBus", /* object path */
|
|
"org.freedesktop.DBus", /* interface name */
|
|
"NonExistantMethod", /* method name */
|
|
NULL, NULL,
|
|
G_DBUS_CALL_FLAGS_NONE,
|
|
-1,
|
|
NULL,
|
|
(GAsyncReadyCallback) msg_cb_expect_error_unknown_method,
|
|
NULL);
|
|
g_main_loop_run (loop);
|
|
|
|
/*
|
|
* Check that cancellation works when the message is already in flight.
|
|
*/
|
|
ca = g_cancellable_new ();
|
|
g_dbus_connection_call (c,
|
|
"org.freedesktop.DBus", /* bus_name */
|
|
"/org/freedesktop/DBus", /* object path */
|
|
"org.freedesktop.DBus", /* interface name */
|
|
"GetId", /* method name */
|
|
NULL, NULL,
|
|
G_DBUS_CALL_FLAGS_NONE,
|
|
-1,
|
|
ca,
|
|
(GAsyncReadyCallback) msg_cb_expect_error_cancelled_2,
|
|
NULL);
|
|
g_cancellable_cancel (ca);
|
|
g_main_loop_run (loop);
|
|
g_object_unref (ca);
|
|
|
|
/*
|
|
* Check that we get an error when sending to a connection that is disconnected.
|
|
*/
|
|
g_dbus_connection_set_exit_on_close (c, FALSE);
|
|
session_bus_down ();
|
|
_g_assert_signal_received (c, "closed");
|
|
g_assert (g_dbus_connection_is_closed (c));
|
|
|
|
g_dbus_connection_call (c,
|
|
"org.freedesktop.DBus", /* bus_name */
|
|
"/org/freedesktop/DBus", /* object path */
|
|
"org.freedesktop.DBus", /* interface name */
|
|
"GetId", /* method name */
|
|
NULL, NULL,
|
|
G_DBUS_CALL_FLAGS_NONE,
|
|
-1,
|
|
NULL,
|
|
(GAsyncReadyCallback) msg_cb_expect_error_disconnected,
|
|
NULL);
|
|
g_main_loop_run (loop);
|
|
|
|
_g_object_wait_for_single_ref (c);
|
|
g_object_unref (c);
|
|
}
|
|
|
|
/* ---------------------------------------------------------------------------------------------------- */
|
|
/* Connection signal tests */
|
|
/* ---------------------------------------------------------------------------------------------------- */
|
|
|
|
static void
|
|
test_connection_signal_handler (GDBusConnection *connection,
|
|
const gchar *sender_name,
|
|
const gchar *object_path,
|
|
const gchar *interface_name,
|
|
const gchar *signal_name,
|
|
GVariant *parameters,
|
|
gpointer user_data)
|
|
{
|
|
gint *counter = user_data;
|
|
*counter += 1;
|
|
|
|
/*g_debug ("in test_connection_signal_handler (sender=%s path=%s interface=%s member=%s)",
|
|
sender_name,
|
|
object_path,
|
|
interface_name,
|
|
signal_name);*/
|
|
|
|
g_main_loop_quit (loop);
|
|
}
|
|
|
|
static void
|
|
test_connection_signals (void)
|
|
{
|
|
GDBusConnection *c1;
|
|
GDBusConnection *c2;
|
|
GDBusConnection *c3;
|
|
guint s1;
|
|
guint s1b;
|
|
guint s2;
|
|
guint s3;
|
|
gint count_s1;
|
|
gint count_s1b;
|
|
gint count_s2;
|
|
gint count_name_owner_changed;
|
|
GError *error;
|
|
gboolean ret;
|
|
GVariant *result;
|
|
|
|
error = NULL;
|
|
|
|
/*
|
|
* Bring up first separate connections
|
|
*/
|
|
session_bus_up ();
|
|
/* if running with dbus-monitor, it claims the name :1.0 - so if we don't run with the monitor
|
|
* emulate this
|
|
*/
|
|
if (g_getenv ("G_DBUS_MONITOR") == NULL)
|
|
{
|
|
c1 = _g_bus_get_priv (G_BUS_TYPE_SESSION, NULL, NULL);
|
|
g_assert (c1 != NULL);
|
|
g_assert (!g_dbus_connection_is_closed (c1));
|
|
g_object_unref (c1);
|
|
}
|
|
c1 = g_bus_get_sync (G_BUS_TYPE_SESSION, NULL, NULL);
|
|
g_assert (c1 != NULL);
|
|
g_assert (!g_dbus_connection_is_closed (c1));
|
|
g_assert_cmpstr (g_dbus_connection_get_unique_name (c1), ==, ":1.1");
|
|
|
|
/*
|
|
* Install two signal handlers for the first connection
|
|
*
|
|
* - Listen to the signal "Foo" from :1.2 (e.g. c2)
|
|
* - Listen to the signal "Foo" from anyone (e.g. both c2 and c3)
|
|
*
|
|
* and then count how many times this signal handler was invoked.
|
|
*/
|
|
s1 = g_dbus_connection_signal_subscribe (c1,
|
|
":1.2",
|
|
"org.gtk.GDBus.ExampleInterface",
|
|
"Foo",
|
|
"/org/gtk/GDBus/ExampleInterface",
|
|
NULL,
|
|
G_DBUS_SIGNAL_FLAGS_NONE,
|
|
test_connection_signal_handler,
|
|
&count_s1,
|
|
NULL);
|
|
s2 = g_dbus_connection_signal_subscribe (c1,
|
|
NULL, /* match any sender */
|
|
"org.gtk.GDBus.ExampleInterface",
|
|
"Foo",
|
|
"/org/gtk/GDBus/ExampleInterface",
|
|
NULL,
|
|
G_DBUS_SIGNAL_FLAGS_NONE,
|
|
test_connection_signal_handler,
|
|
&count_s2,
|
|
NULL);
|
|
s3 = g_dbus_connection_signal_subscribe (c1,
|
|
"org.freedesktop.DBus", /* sender */
|
|
"org.freedesktop.DBus", /* interface */
|
|
"NameOwnerChanged", /* member */
|
|
"/org/freedesktop/DBus", /* path */
|
|
NULL,
|
|
G_DBUS_SIGNAL_FLAGS_NONE,
|
|
test_connection_signal_handler,
|
|
&count_name_owner_changed,
|
|
NULL);
|
|
/* Note that s1b is *just like* s1 - this is to catch a bug where N
|
|
* subscriptions of the same rule causes N calls to each of the N
|
|
* subscriptions instead of just 1 call to each of the N subscriptions.
|
|
*/
|
|
s1b = g_dbus_connection_signal_subscribe (c1,
|
|
":1.2",
|
|
"org.gtk.GDBus.ExampleInterface",
|
|
"Foo",
|
|
"/org/gtk/GDBus/ExampleInterface",
|
|
NULL,
|
|
G_DBUS_SIGNAL_FLAGS_NONE,
|
|
test_connection_signal_handler,
|
|
&count_s1b,
|
|
NULL);
|
|
g_assert (s1 != 0);
|
|
g_assert (s1b != 0);
|
|
g_assert (s2 != 0);
|
|
g_assert (s3 != 0);
|
|
|
|
count_s1 = 0;
|
|
count_s1b = 0;
|
|
count_s2 = 0;
|
|
count_name_owner_changed = 0;
|
|
|
|
/*
|
|
* Bring up two other connections
|
|
*/
|
|
c2 = _g_bus_get_priv (G_BUS_TYPE_SESSION, NULL, NULL);
|
|
g_assert (c2 != NULL);
|
|
g_assert (!g_dbus_connection_is_closed (c2));
|
|
g_assert_cmpstr (g_dbus_connection_get_unique_name (c2), ==, ":1.2");
|
|
c3 = _g_bus_get_priv (G_BUS_TYPE_SESSION, NULL, NULL);
|
|
g_assert (c3 != NULL);
|
|
g_assert (!g_dbus_connection_is_closed (c3));
|
|
g_assert_cmpstr (g_dbus_connection_get_unique_name (c3), ==, ":1.3");
|
|
|
|
/*
|
|
* Make c2 emit "Foo" - we should catch it twice
|
|
*
|
|
* Note that there is no way to be sure that the signal subscriptions
|
|
* on c1 are effective yet - for all we know, the AddMatch() messages
|
|
* could sit waiting in a buffer somewhere between this process and
|
|
* the message bus. And emitting signals on c2 (a completely other
|
|
* socket!) will not necessarily change this.
|
|
*
|
|
* To ensure this is not the case, do a synchronous call on c1.
|
|
*/
|
|
result = g_dbus_connection_call_sync (c1,
|
|
"org.freedesktop.DBus", /* bus name */
|
|
"/org/freedesktop/DBus", /* object path */
|
|
"org.freedesktop.DBus", /* interface name */
|
|
"GetId", /* method name */
|
|
NULL, /* parameters */
|
|
NULL, /* return type */
|
|
G_DBUS_CALL_FLAGS_NONE,
|
|
-1,
|
|
NULL,
|
|
&error);
|
|
g_assert_no_error (error);
|
|
g_assert (result != NULL);
|
|
g_variant_unref (result);
|
|
/* now, emit the signal on c2 */
|
|
ret = g_dbus_connection_emit_signal (c2,
|
|
NULL, /* destination bus name */
|
|
"/org/gtk/GDBus/ExampleInterface",
|
|
"org.gtk.GDBus.ExampleInterface",
|
|
"Foo",
|
|
NULL,
|
|
&error);
|
|
g_assert_no_error (error);
|
|
g_assert (ret);
|
|
while (!(count_s1 >= 1 && count_s2 >= 1))
|
|
g_main_loop_run (loop);
|
|
g_assert_cmpint (count_s1, ==, 1);
|
|
g_assert_cmpint (count_s2, ==, 1);
|
|
|
|
/*
|
|
* Make c3 emit "Foo" - we should catch it only once
|
|
*/
|
|
ret = g_dbus_connection_emit_signal (c3,
|
|
NULL, /* destination bus name */
|
|
"/org/gtk/GDBus/ExampleInterface",
|
|
"org.gtk.GDBus.ExampleInterface",
|
|
"Foo",
|
|
NULL,
|
|
&error);
|
|
g_assert_no_error (error);
|
|
g_assert (ret);
|
|
while (!(count_s1 == 1 && count_s2 == 2))
|
|
g_main_loop_run (loop);
|
|
g_assert_cmpint (count_s1, ==, 1);
|
|
g_assert_cmpint (count_s2, ==, 2);
|
|
|
|
/*
|
|
* Also to check the total amount of NameOwnerChanged signals - use a 5 second ceiling
|
|
* to avoid spinning forever
|
|
*/
|
|
gboolean quit_mainloop_fired;
|
|
guint quit_mainloop_id;
|
|
quit_mainloop_fired = FALSE;
|
|
quit_mainloop_id = g_timeout_add (5000, test_connection_quit_mainloop, &quit_mainloop_fired);
|
|
while (count_name_owner_changed < 2 && !quit_mainloop_fired)
|
|
g_main_loop_run (loop);
|
|
g_source_remove (quit_mainloop_id);
|
|
g_assert_cmpint (count_s1, ==, 1);
|
|
g_assert_cmpint (count_s2, ==, 2);
|
|
g_assert_cmpint (count_name_owner_changed, ==, 2);
|
|
|
|
g_dbus_connection_signal_unsubscribe (c1, s1);
|
|
g_dbus_connection_signal_unsubscribe (c1, s2);
|
|
g_dbus_connection_signal_unsubscribe (c1, s3);
|
|
g_dbus_connection_signal_unsubscribe (c1, s1b);
|
|
|
|
_g_object_wait_for_single_ref (c1);
|
|
_g_object_wait_for_single_ref (c2);
|
|
_g_object_wait_for_single_ref (c3);
|
|
|
|
g_object_unref (c1);
|
|
g_object_unref (c2);
|
|
g_object_unref (c3);
|
|
|
|
session_bus_down ();
|
|
}
|
|
|
|
/* ---------------------------------------------------------------------------------------------------- */
|
|
|
|
typedef struct
|
|
{
|
|
guint num_handled;
|
|
guint num_outgoing;
|
|
guint32 serial;
|
|
} FilterData;
|
|
|
|
static GDBusMessage *
|
|
filter_func (GDBusConnection *connection,
|
|
GDBusMessage *message,
|
|
gboolean incoming,
|
|
gpointer user_data)
|
|
{
|
|
FilterData *data = user_data;
|
|
guint32 reply_serial;
|
|
|
|
if (incoming)
|
|
{
|
|
reply_serial = g_dbus_message_get_reply_serial (message);
|
|
if (reply_serial == data->serial)
|
|
data->num_handled += 1;
|
|
}
|
|
else
|
|
{
|
|
data->num_outgoing += 1;
|
|
}
|
|
|
|
return message;
|
|
}
|
|
|
|
|
|
typedef struct
|
|
{
|
|
gboolean alter_incoming;
|
|
gboolean alter_outgoing;
|
|
} FilterEffects;
|
|
|
|
static GDBusMessage *
|
|
other_filter_func (GDBusConnection *connection,
|
|
GDBusMessage *message,
|
|
gboolean incoming,
|
|
gpointer user_data)
|
|
{
|
|
FilterEffects *effects = user_data;
|
|
GDBusMessage *ret;
|
|
gboolean alter;
|
|
|
|
if (incoming)
|
|
alter = effects->alter_incoming;
|
|
else
|
|
alter = effects->alter_outgoing;
|
|
|
|
if (alter)
|
|
{
|
|
GDBusMessage *copy;
|
|
GVariant *body;
|
|
gchar *s;
|
|
gchar *s2;
|
|
|
|
copy = g_dbus_message_copy (message, NULL);
|
|
g_object_unref (message);
|
|
|
|
body = g_dbus_message_get_body (copy);
|
|
g_variant_get (body, "(s)", &s);
|
|
s2 = g_strdup_printf ("MOD: %s", s);
|
|
g_dbus_message_set_body (copy, g_variant_new ("(s)", s2));
|
|
g_free (s2);
|
|
g_free (s);
|
|
|
|
ret = copy;
|
|
}
|
|
else
|
|
{
|
|
ret = message;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static void
|
|
test_connection_filter_name_owner_changed_signal_handler (GDBusConnection *connection,
|
|
const gchar *sender_name,
|
|
const gchar *object_path,
|
|
const gchar *interface_name,
|
|
const gchar *signal_name,
|
|
GVariant *parameters,
|
|
gpointer user_data)
|
|
{
|
|
const gchar *name;
|
|
const gchar *old_owner;
|
|
const gchar *new_owner;
|
|
|
|
g_variant_get (parameters,
|
|
"(&s&s&s)",
|
|
&name,
|
|
&old_owner,
|
|
&new_owner);
|
|
|
|
if (g_strcmp0 (name, "com.example.TestService") == 0 && strlen (new_owner) > 0)
|
|
{
|
|
g_main_loop_quit (loop);
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
test_connection_filter_on_timeout (gpointer user_data)
|
|
{
|
|
g_printerr ("Timeout waiting 1000 msec on service\n");
|
|
g_assert_not_reached ();
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
test_connection_filter (void)
|
|
{
|
|
GDBusConnection *c;
|
|
FilterData data;
|
|
GDBusMessage *m;
|
|
GDBusMessage *m2;
|
|
GDBusMessage *r;
|
|
GError *error;
|
|
guint filter_id;
|
|
guint timeout_mainloop_id;
|
|
guint signal_handler_id;
|
|
FilterEffects effects;
|
|
GVariant *result;
|
|
const gchar *s;
|
|
|
|
memset (&data, '\0', sizeof (FilterData));
|
|
|
|
session_bus_up ();
|
|
|
|
error = NULL;
|
|
c = g_bus_get_sync (G_BUS_TYPE_SESSION, NULL, &error);
|
|
g_assert_no_error (error);
|
|
g_assert (c != NULL);
|
|
|
|
filter_id = g_dbus_connection_add_filter (c,
|
|
filter_func,
|
|
&data,
|
|
NULL);
|
|
|
|
m = g_dbus_message_new_method_call ("org.freedesktop.DBus", /* name */
|
|
"/org/freedesktop/DBus", /* path */
|
|
"org.freedesktop.DBus", /* interface */
|
|
"GetNameOwner");
|
|
g_dbus_message_set_body (m, g_variant_new ("(s)", "org.freedesktop.DBus"));
|
|
error = NULL;
|
|
g_dbus_connection_send_message (c, m, G_DBUS_SEND_MESSAGE_FLAGS_NONE, &data.serial, &error);
|
|
g_assert_no_error (error);
|
|
|
|
while (data.num_handled == 0)
|
|
g_thread_yield ();
|
|
|
|
m2 = g_dbus_message_copy (m, &error);
|
|
g_assert_no_error (error);
|
|
g_dbus_connection_send_message (c, m2, G_DBUS_SEND_MESSAGE_FLAGS_NONE, &data.serial, &error);
|
|
g_object_unref (m2);
|
|
g_assert_no_error (error);
|
|
|
|
while (data.num_handled == 1)
|
|
g_thread_yield ();
|
|
|
|
m2 = g_dbus_message_copy (m, &error);
|
|
g_assert_no_error (error);
|
|
r = g_dbus_connection_send_message_with_reply_sync (c,
|
|
m2,
|
|
G_DBUS_SEND_MESSAGE_FLAGS_NONE,
|
|
-1,
|
|
&data.serial,
|
|
NULL, /* GCancellable */
|
|
&error);
|
|
g_object_unref (m2);
|
|
g_assert_no_error (error);
|
|
g_assert (r != NULL);
|
|
g_object_unref (r);
|
|
g_assert_cmpint (data.num_handled, ==, 3);
|
|
|
|
g_dbus_connection_remove_filter (c, filter_id);
|
|
|
|
m2 = g_dbus_message_copy (m, &error);
|
|
g_assert_no_error (error);
|
|
r = g_dbus_connection_send_message_with_reply_sync (c,
|
|
m2,
|
|
G_DBUS_SEND_MESSAGE_FLAGS_NONE,
|
|
-1,
|
|
&data.serial,
|
|
NULL, /* GCancellable */
|
|
&error);
|
|
g_object_unref (m2);
|
|
g_assert_no_error (error);
|
|
g_assert (r != NULL);
|
|
g_object_unref (r);
|
|
g_assert_cmpint (data.num_handled, ==, 3);
|
|
g_assert_cmpint (data.num_outgoing, ==, 3);
|
|
|
|
/* this is safe; testserver will exit once the bus goes away */
|
|
g_assert (g_spawn_command_line_async (SRCDIR "/gdbus-testserver.py", NULL));
|
|
/* wait for service to be available */
|
|
signal_handler_id = g_dbus_connection_signal_subscribe (c,
|
|
"org.freedesktop.DBus", /* sender */
|
|
"org.freedesktop.DBus",
|
|
"NameOwnerChanged",
|
|
"/org/freedesktop/DBus",
|
|
NULL, /* arg0 */
|
|
G_DBUS_SIGNAL_FLAGS_NONE,
|
|
test_connection_filter_name_owner_changed_signal_handler,
|
|
NULL,
|
|
NULL);
|
|
g_assert_cmpint (signal_handler_id, !=, 0);
|
|
timeout_mainloop_id = g_timeout_add (1000, test_connection_filter_on_timeout, NULL);
|
|
g_main_loop_run (loop);
|
|
g_source_remove (timeout_mainloop_id);
|
|
g_dbus_connection_signal_unsubscribe (c, signal_handler_id);
|
|
|
|
/* now test some combinations... */
|
|
filter_id = g_dbus_connection_add_filter (c,
|
|
other_filter_func,
|
|
&effects,
|
|
NULL);
|
|
/* -- */
|
|
effects.alter_incoming = FALSE;
|
|
effects.alter_outgoing = FALSE;
|
|
error = NULL;
|
|
result = g_dbus_connection_call_sync (c,
|
|
"com.example.TestService", /* bus name */
|
|
"/com/example/TestObject", /* object path */
|
|
"com.example.Frob", /* interface name */
|
|
"HelloWorld", /* method name */
|
|
g_variant_new ("(s)", "Cat"), /* parameters */
|
|
G_VARIANT_TYPE ("(s)"), /* return type */
|
|
G_DBUS_CALL_FLAGS_NONE,
|
|
-1,
|
|
NULL,
|
|
&error);
|
|
g_assert_no_error (error);
|
|
g_variant_get (result, "(&s)", &s);
|
|
g_assert_cmpstr (s, ==, "You greeted me with 'Cat'. Thanks!");
|
|
g_variant_unref (result);
|
|
/* -- */
|
|
effects.alter_incoming = TRUE;
|
|
effects.alter_outgoing = TRUE;
|
|
error = NULL;
|
|
result = g_dbus_connection_call_sync (c,
|
|
"com.example.TestService", /* bus name */
|
|
"/com/example/TestObject", /* object path */
|
|
"com.example.Frob", /* interface name */
|
|
"HelloWorld", /* method name */
|
|
g_variant_new ("(s)", "Cat"), /* parameters */
|
|
G_VARIANT_TYPE ("(s)"), /* return type */
|
|
G_DBUS_CALL_FLAGS_NONE,
|
|
-1,
|
|
NULL,
|
|
&error);
|
|
g_assert_no_error (error);
|
|
g_variant_get (result, "(&s)", &s);
|
|
g_assert_cmpstr (s, ==, "MOD: You greeted me with 'MOD: Cat'. Thanks!");
|
|
g_variant_unref (result);
|
|
|
|
|
|
g_dbus_connection_remove_filter (c, filter_id);
|
|
|
|
_g_object_wait_for_single_ref (c);
|
|
g_object_unref (c);
|
|
g_object_unref (m);
|
|
|
|
session_bus_down ();
|
|
}
|
|
|
|
/* ---------------------------------------------------------------------------------------------------- */
|
|
|
|
static void
|
|
test_connection_flush_signal_handler (GDBusConnection *connection,
|
|
const gchar *sender_name,
|
|
const gchar *object_path,
|
|
const gchar *interface_name,
|
|
const gchar *signal_name,
|
|
GVariant *parameters,
|
|
gpointer user_data)
|
|
{
|
|
g_main_loop_quit (loop);
|
|
}
|
|
|
|
static gboolean
|
|
test_connection_flush_on_timeout (gpointer user_data)
|
|
{
|
|
guint iteration = GPOINTER_TO_UINT (user_data);
|
|
g_printerr ("Timeout waiting 1000 msec on iteration %d\n", iteration);
|
|
g_assert_not_reached ();
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
test_connection_flush (void)
|
|
{
|
|
GDBusConnection *connection;
|
|
GError *error;
|
|
guint n;
|
|
guint signal_handler_id;
|
|
|
|
session_bus_up ();
|
|
|
|
error = NULL;
|
|
connection = g_bus_get_sync (G_BUS_TYPE_SESSION, NULL, &error);
|
|
g_assert_no_error (error);
|
|
g_assert (connection != NULL);
|
|
|
|
signal_handler_id = g_dbus_connection_signal_subscribe (connection,
|
|
NULL, /* sender */
|
|
"org.gtk.GDBus.FlushInterface",
|
|
"SomeSignal",
|
|
"/org/gtk/GDBus/FlushObject",
|
|
NULL,
|
|
G_DBUS_SIGNAL_FLAGS_NONE,
|
|
test_connection_flush_signal_handler,
|
|
NULL,
|
|
NULL);
|
|
g_assert_cmpint (signal_handler_id, !=, 0);
|
|
|
|
for (n = 0; n < 50; n++)
|
|
{
|
|
gboolean ret;
|
|
gint exit_status;
|
|
guint timeout_mainloop_id;
|
|
|
|
error = NULL;
|
|
ret = g_spawn_command_line_sync ("./gdbus-connection-flush-helper",
|
|
NULL, /* stdout */
|
|
NULL, /* stderr */
|
|
&exit_status,
|
|
&error);
|
|
g_assert_no_error (error);
|
|
g_assert (WIFEXITED (exit_status));
|
|
g_assert_cmpint (WEXITSTATUS (exit_status), ==, 0);
|
|
g_assert (ret);
|
|
|
|
timeout_mainloop_id = g_timeout_add (1000, test_connection_flush_on_timeout, GUINT_TO_POINTER (n));
|
|
g_main_loop_run (loop);
|
|
g_source_remove (timeout_mainloop_id);
|
|
}
|
|
|
|
g_dbus_connection_signal_unsubscribe (connection, signal_handler_id);
|
|
_g_object_wait_for_single_ref (connection);
|
|
g_object_unref (connection);
|
|
|
|
session_bus_down ();
|
|
}
|
|
|
|
static void
|
|
test_connection_basic (void)
|
|
{
|
|
GDBusConnection *connection;
|
|
GError *error;
|
|
GDBusCapabilityFlags flags;
|
|
gchar *guid;
|
|
gchar *name;
|
|
gboolean closed;
|
|
gboolean exit_on_close;
|
|
GIOStream *stream;
|
|
GCredentials *credentials;
|
|
|
|
session_bus_up ();
|
|
|
|
error = NULL;
|
|
connection = g_bus_get_sync (G_BUS_TYPE_SESSION, NULL, &error);
|
|
g_assert_no_error (error);
|
|
g_assert (connection != NULL);
|
|
|
|
flags = g_dbus_connection_get_capabilities (connection);
|
|
g_assert (flags == G_DBUS_CAPABILITY_FLAGS_NONE ||
|
|
flags == G_DBUS_CAPABILITY_FLAGS_UNIX_FD_PASSING);
|
|
|
|
credentials = g_dbus_connection_get_peer_credentials (connection);
|
|
g_assert (credentials == NULL);
|
|
|
|
g_object_get (connection,
|
|
"stream", &stream,
|
|
"guid", &guid,
|
|
"unique-name", &name,
|
|
"closed", &closed,
|
|
"exit-on-close", &exit_on_close,
|
|
"capabilities", &flags,
|
|
NULL);
|
|
|
|
g_assert (G_IS_IO_STREAM (stream));
|
|
g_assert (g_dbus_is_guid (guid));
|
|
g_assert (g_dbus_is_unique_name (name));
|
|
g_assert (!closed);
|
|
g_assert (exit_on_close);
|
|
g_assert (flags == G_DBUS_CAPABILITY_FLAGS_NONE ||
|
|
flags == G_DBUS_CAPABILITY_FLAGS_UNIX_FD_PASSING);
|
|
|
|
g_object_unref (stream);
|
|
g_object_unref (connection);
|
|
g_free (name);
|
|
g_free (guid);
|
|
|
|
session_bus_down ();
|
|
}
|
|
|
|
/* ---------------------------------------------------------------------------------------------------- */
|
|
|
|
/* Message size > 20MiB ... should be enough to make sure the message
|
|
* is fragmented when shoved across any transport
|
|
*/
|
|
#define LARGE_MESSAGE_STRING_LENGTH (20*1024*1024)
|
|
|
|
static void
|
|
large_message_on_name_appeared (GDBusConnection *connection,
|
|
const gchar *name,
|
|
const gchar *name_owner,
|
|
gpointer user_data)
|
|
{
|
|
GError *error;
|
|
gchar *request;
|
|
const gchar *reply;
|
|
GVariant *result;
|
|
guint n;
|
|
|
|
request = g_new (gchar, LARGE_MESSAGE_STRING_LENGTH + 1);
|
|
for (n = 0; n < LARGE_MESSAGE_STRING_LENGTH; n++)
|
|
request[n] = '0' + (n%10);
|
|
request[n] = '\0';
|
|
|
|
error = NULL;
|
|
result = g_dbus_connection_call_sync (connection,
|
|
"com.example.TestService", /* bus name */
|
|
"/com/example/TestObject", /* object path */
|
|
"com.example.Frob", /* interface name */
|
|
"HelloWorld", /* method name */
|
|
g_variant_new ("(s)", request), /* parameters */
|
|
G_VARIANT_TYPE ("(s)"), /* return type */
|
|
G_DBUS_CALL_FLAGS_NONE,
|
|
-1,
|
|
NULL,
|
|
&error);
|
|
g_assert_no_error (error);
|
|
g_assert (result != NULL);
|
|
g_variant_get (result, "(&s)", &reply);
|
|
g_assert_cmpint (strlen (reply), >, LARGE_MESSAGE_STRING_LENGTH);
|
|
g_assert (g_str_has_prefix (reply, "You greeted me with '01234567890123456789012"));
|
|
g_assert (g_str_has_suffix (reply, "6789'. Thanks!"));
|
|
g_variant_unref (result);
|
|
|
|
g_free (request);
|
|
|
|
g_main_loop_quit (loop);
|
|
}
|
|
|
|
static void
|
|
large_message_on_name_vanished (GDBusConnection *connection,
|
|
const gchar *name,
|
|
gpointer user_data)
|
|
{
|
|
}
|
|
|
|
static void
|
|
test_connection_large_message (void)
|
|
{
|
|
guint watcher_id;
|
|
|
|
session_bus_up ();
|
|
|
|
/* this is safe; testserver will exit once the bus goes away */
|
|
g_assert (g_spawn_command_line_async (SRCDIR "/gdbus-testserver.py", NULL));
|
|
|
|
watcher_id = g_bus_watch_name (G_BUS_TYPE_SESSION,
|
|
"com.example.TestService",
|
|
G_BUS_NAME_WATCHER_FLAGS_NONE,
|
|
large_message_on_name_appeared,
|
|
large_message_on_name_vanished,
|
|
NULL, /* user_data */
|
|
NULL); /* GDestroyNotify */
|
|
g_main_loop_run (loop);
|
|
g_bus_unwatch_name (watcher_id);
|
|
|
|
session_bus_down ();
|
|
}
|
|
|
|
/* ---------------------------------------------------------------------------------------------------- */
|
|
|
|
int
|
|
main (int argc,
|
|
char *argv[])
|
|
{
|
|
g_type_init ();
|
|
g_test_init (&argc, &argv, NULL);
|
|
|
|
/* all the tests rely on a shared main loop */
|
|
loop = g_main_loop_new (NULL, FALSE);
|
|
|
|
/* all the tests use a session bus with a well-known address that we can bring up and down
|
|
* using session_bus_up() and session_bus_down().
|
|
*/
|
|
g_unsetenv ("DISPLAY");
|
|
g_setenv ("DBUS_SESSION_BUS_ADDRESS", session_bus_get_temporary_address (), TRUE);
|
|
|
|
g_test_add_func ("/gdbus/connection/basic", test_connection_basic);
|
|
g_test_add_func ("/gdbus/connection/life-cycle", test_connection_life_cycle);
|
|
g_test_add_func ("/gdbus/connection/send", test_connection_send);
|
|
g_test_add_func ("/gdbus/connection/signals", test_connection_signals);
|
|
g_test_add_func ("/gdbus/connection/filter", test_connection_filter);
|
|
g_test_add_func ("/gdbus/connection/flush", test_connection_flush);
|
|
g_test_add_func ("/gdbus/connection/large_message", test_connection_large_message);
|
|
return g_test_run();
|
|
}
|