/* 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 */ #include #include #include #include #include #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 GDBusMessageFilterResult some_filter_func (GDBusConnection *connection, GDBusMessage *message, gboolean incoming, gpointer user_data) { return G_DBUS_MESSAGE_FILTER_RESULT_NO_EFFECT; } 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 GDBusMessageFilterResult 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 G_DBUS_MESSAGE_FILTER_RESULT_NO_EFFECT; } typedef struct { GDBusMessageFilterResult incoming; GDBusMessageFilterResult outgoing; } FilterEffects; static GDBusMessageFilterResult other_filter_func (GDBusConnection *connection, GDBusMessage *message, gboolean incoming, gpointer user_data) { FilterEffects *effects = user_data; GDBusMessageFilterResult ret; if (incoming) ret = effects->incoming; else ret = effects->outgoing; if (ret == G_DBUS_MESSAGE_FILTER_RESULT_MESSAGE_ALTERED) { GVariant *body; gchar *s; gchar *s2; body = g_dbus_message_get_body (message); g_variant_get (body, "(s)", &s); s2 = g_strdup_printf ("MOD: %s", s); g_dbus_message_set_body (message, g_variant_new ("(s)", s2)); g_free (s2); g_free (s); } 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 *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 (); g_dbus_message_set_serial (m, 0); 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 == 1) g_thread_yield (); g_dbus_message_set_serial (m, 0); r = g_dbus_connection_send_message_with_reply_sync (c, m, G_DBUS_SEND_MESSAGE_FLAGS_NONE, -1, &data.serial, NULL, /* GCancellable */ &error); 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); g_dbus_message_set_serial (m, 0); r = g_dbus_connection_send_message_with_reply_sync (c, m, G_DBUS_SEND_MESSAGE_FLAGS_NONE, -1, &data.serial, NULL, /* GCancellable */ &error); 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 all nine combinations... */ filter_id = g_dbus_connection_add_filter (c, other_filter_func, &effects, NULL); /* -- */ effects.incoming = G_DBUS_MESSAGE_FILTER_RESULT_NO_EFFECT; effects.outgoing = G_DBUS_MESSAGE_FILTER_RESULT_NO_EFFECT; 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.incoming = G_DBUS_MESSAGE_FILTER_RESULT_MESSAGE_ALTERED; effects.outgoing = G_DBUS_MESSAGE_FILTER_RESULT_MESSAGE_ALTERED; 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(); }