1 /* GLib testing framework examples and tests
3 * Copyright (C) 2008-2010 Red Hat, Inc.
5 * This library is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU Lesser General Public
7 * License as published by the Free Software Foundation; either
8 * version 2.1 of the License, or (at your option) any later version.
10 * This library is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * Lesser General Public License for more details.
15 * You should have received a copy of the GNU Lesser General
16 * Public License along with this library; if not, see <http://www.gnu.org/licenses/>.
18 * Author: David Zeuthen <davidz@redhat.com>
25 #include "gdbus-tests.h"
27 /* all tests rely on a global connection */
28 static GDBusConnection
*c
= NULL
;
30 /* ---------------------------------------------------------------------------------------------------- */
31 /* Ensure that signal and method replies are delivered in the right thread */
32 /* ---------------------------------------------------------------------------------------------------- */
36 GMainLoop
*thread_loop
;
41 msg_cb_expect_success (GDBusConnection
*connection
,
45 DeliveryData
*data
= user_data
;
50 result
= g_dbus_connection_call_finish (connection
,
53 g_assert_no_error (error
);
54 g_assert (result
!= NULL
);
55 g_variant_unref (result
);
57 g_assert (g_thread_self () == data
->thread
);
59 g_main_loop_quit (data
->thread_loop
);
63 msg_cb_expect_error_cancelled (GDBusConnection
*connection
,
67 DeliveryData
*data
= user_data
;
72 result
= g_dbus_connection_call_finish (connection
,
75 g_assert_error (error
, G_IO_ERROR
, G_IO_ERROR_CANCELLED
);
76 g_assert (!g_dbus_error_is_remote_error (error
));
78 g_assert (result
== NULL
);
80 g_assert (g_thread_self () == data
->thread
);
82 g_main_loop_quit (data
->thread_loop
);
86 signal_handler (GDBusConnection
*connection
,
87 const gchar
*sender_name
,
88 const gchar
*object_path
,
89 const gchar
*interface_name
,
90 const gchar
*signal_name
,
94 DeliveryData
*data
= user_data
;
96 g_assert (g_thread_self () == data
->thread
);
100 g_main_loop_quit (data
->thread_loop
);
104 test_delivery_in_thread_func (gpointer _data
)
106 GMainLoop
*thread_loop
;
107 GMainContext
*thread_context
;
110 guint subscription_id
;
111 GDBusConnection
*priv_c
;
116 thread_context
= g_main_context_new ();
117 thread_loop
= g_main_loop_new (thread_context
, FALSE
);
118 g_main_context_push_thread_default (thread_context
);
120 data
.thread
= g_thread_self ();
121 data
.thread_loop
= thread_loop
;
122 data
.signal_count
= 0;
124 /* ---------------------------------------------------------------------------------------------------- */
127 * Check that we get a reply to the GetId() method call.
129 g_dbus_connection_call (c
,
130 "org.freedesktop.DBus", /* bus_name */
131 "/org/freedesktop/DBus", /* object path */
132 "org.freedesktop.DBus", /* interface name */
133 "GetId", /* method name */
135 G_DBUS_CALL_FLAGS_NONE
,
138 (GAsyncReadyCallback
) msg_cb_expect_success
,
140 g_main_loop_run (thread_loop
);
143 * Check that we never actually send a message if the GCancellable
144 * is already cancelled - i.e. we should get #G_IO_ERROR_CANCELLED
145 * when the actual connection is not up.
147 ca
= g_cancellable_new ();
148 g_cancellable_cancel (ca
);
149 g_dbus_connection_call (c
,
150 "org.freedesktop.DBus", /* bus_name */
151 "/org/freedesktop/DBus", /* object path */
152 "org.freedesktop.DBus", /* interface name */
153 "GetId", /* method name */
155 G_DBUS_CALL_FLAGS_NONE
,
158 (GAsyncReadyCallback
) msg_cb_expect_error_cancelled
,
160 g_main_loop_run (thread_loop
);
164 * Check that cancellation works when the message is already in flight.
166 ca
= g_cancellable_new ();
167 g_dbus_connection_call (c
,
168 "org.freedesktop.DBus", /* bus_name */
169 "/org/freedesktop/DBus", /* object path */
170 "org.freedesktop.DBus", /* interface name */
171 "GetId", /* method name */
173 G_DBUS_CALL_FLAGS_NONE
,
176 (GAsyncReadyCallback
) msg_cb_expect_error_cancelled
,
178 g_cancellable_cancel (ca
);
179 g_main_loop_run (thread_loop
);
183 * Check that signals are delivered to the correct thread.
185 * First we subscribe to the signal, then we create a a private
186 * connection. This should cause a NameOwnerChanged message from
189 subscription_id
= g_dbus_connection_signal_subscribe (c
,
190 "org.freedesktop.DBus", /* sender */
191 "org.freedesktop.DBus", /* interface */
192 "NameOwnerChanged", /* member */
193 "/org/freedesktop/DBus", /* path */
195 G_DBUS_SIGNAL_FLAGS_NONE
,
199 g_assert (subscription_id
!= 0);
200 g_assert (data
.signal_count
== 0);
202 priv_c
= _g_bus_get_priv (G_BUS_TYPE_SESSION
, NULL
, &error
);
203 g_assert_no_error (error
);
204 g_assert (priv_c
!= NULL
);
206 g_main_loop_run (thread_loop
);
207 g_assert (data
.signal_count
== 1);
209 g_object_unref (priv_c
);
211 g_dbus_connection_signal_unsubscribe (c
, subscription_id
);
213 /* ---------------------------------------------------------------------------------------------------- */
215 g_main_context_pop_thread_default (thread_context
);
216 g_main_loop_unref (thread_loop
);
217 g_main_context_unref (thread_context
);
223 test_delivery_in_thread (void)
227 thread
= g_thread_new ("deliver",
228 test_delivery_in_thread_func
,
231 g_thread_join (thread
);
234 /* ---------------------------------------------------------------------------------------------------- */
242 GMainLoop
*thread_loop
;
247 sleep_cb (GDBusProxy
*proxy
,
251 SyncThreadData
*data
= user_data
;
256 result
= g_dbus_proxy_call_finish (proxy
,
259 g_assert_no_error (error
);
260 g_assert (result
!= NULL
);
261 g_assert_cmpstr (g_variant_get_type_string (result
), ==, "()");
262 g_variant_unref (result
);
264 g_assert (data
->thread
== g_thread_self ());
266 g_main_loop_quit (data
->thread_loop
);
268 //g_debug ("async cb (%p)", g_thread_self ());
272 test_sleep_in_thread_func (gpointer _data
)
274 SyncThreadData
*data
= _data
;
275 GMainContext
*thread_context
;
278 thread_context
= g_main_context_new ();
279 data
->thread_loop
= g_main_loop_new (thread_context
, FALSE
);
280 g_main_context_push_thread_default (thread_context
);
282 data
->thread
= g_thread_self ();
284 for (n
= 0; n
< data
->num
; n
++)
288 //g_debug ("invoking async (%p)", g_thread_self ());
289 g_dbus_proxy_call (data
->proxy
,
291 g_variant_new ("(i)", data
->msec
),
292 G_DBUS_CALL_FLAGS_NONE
,
295 (GAsyncReadyCallback
) sleep_cb
,
297 g_main_loop_run (data
->thread_loop
);
298 if (g_test_verbose ())
300 //g_debug ("done invoking async (%p)", g_thread_self ());
308 //g_debug ("invoking sync (%p)", g_thread_self ());
309 result
= g_dbus_proxy_call_sync (data
->proxy
,
311 g_variant_new ("(i)", data
->msec
),
312 G_DBUS_CALL_FLAGS_NONE
,
316 if (g_test_verbose ())
318 //g_debug ("done invoking sync (%p)", g_thread_self ());
319 g_assert_no_error (error
);
320 g_assert (result
!= NULL
);
321 g_assert_cmpstr (g_variant_get_type_string (result
), ==, "()");
322 g_variant_unref (result
);
326 g_main_context_pop_thread_default (thread_context
);
327 g_main_loop_unref (data
->thread_loop
);
328 g_main_context_unref (thread_context
);
334 test_method_calls_on_proxy (GDBusProxy
*proxy
)
339 * Check that multiple threads can do calls without interferring with
340 * each other. We do this by creating three threads that call the
341 * Sleep() method on the server (which handles it asynchronously, e.g.
342 * it won't block other requests) with different sleep durations and
343 * a number of times. We do this so each set of calls add up to 4000
346 * The dbus test server that this code calls into uses glib timeouts
347 * to do the sleeping which have only a granularity of 1ms. It is
348 * therefore possible to lose as much as 40ms; the test could finish
349 * in slightly less than 4 seconds.
351 * We run this test twice - first with async calls in each thread, then
352 * again with sync calls
355 for (n
= 0; n
< 2; n
++)
361 SyncThreadData data1
;
362 SyncThreadData data2
;
363 SyncThreadData data3
;
370 g_get_current_time (&start_time
);
375 data1
.async
= do_async
;
376 thread1
= g_thread_new ("sleep",
377 test_sleep_in_thread_func
,
383 data2
.async
= do_async
;
384 thread2
= g_thread_new ("sleep2",
385 test_sleep_in_thread_func
,
391 data3
.async
= do_async
;
392 thread3
= g_thread_new ("sleep3",
393 test_sleep_in_thread_func
,
396 g_thread_join (thread1
);
397 g_thread_join (thread2
);
398 g_thread_join (thread3
);
400 g_get_current_time (&end_time
);
402 elapsed_msec
= ((end_time
.tv_sec
* G_USEC_PER_SEC
+ end_time
.tv_usec
) -
403 (start_time
.tv_sec
* G_USEC_PER_SEC
+ start_time
.tv_usec
)) / 1000;
405 //g_debug ("Elapsed time for %s = %d msec", n == 0 ? "async" : "sync", elapsed_msec);
407 /* elapsed_msec should be 4000 msec +/- change for overhead/inaccuracy */
408 g_assert_cmpint (elapsed_msec
, >=, 3950);
409 g_assert_cmpint (elapsed_msec
, <, 30000);
411 if (g_test_verbose ())
417 test_method_calls_in_thread (void)
420 GDBusConnection
*connection
;
424 connection
= g_bus_get_sync (G_BUS_TYPE_SESSION
,
427 g_assert_no_error (error
);
429 proxy
= g_dbus_proxy_new_sync (connection
,
430 G_DBUS_PROXY_FLAGS_NONE
,
431 NULL
, /* GDBusInterfaceInfo */
432 "com.example.TestService", /* name */
433 "/com/example/TestObject", /* object path */
434 "com.example.Frob", /* interface */
435 NULL
, /* GCancellable */
437 g_assert_no_error (error
);
439 test_method_calls_on_proxy (proxy
);
441 g_object_unref (proxy
);
442 g_object_unref (connection
);
444 if (g_test_verbose ())
448 #define SLEEP_MIN_USEC 1
449 #define SLEEP_MAX_USEC 10
451 /* Can run in any thread */
453 ensure_connection_works (GDBusConnection
*conn
)
456 GError
*error
= NULL
;
458 v
= g_dbus_connection_call_sync (conn
, "org.freedesktop.DBus",
459 "/org/freedesktop/DBus", "org.freedesktop.DBus", "GetId", NULL
, NULL
, 0, -1,
461 g_assert_no_error (error
);
462 g_assert (v
!= NULL
);
463 g_assert (g_variant_is_of_type (v
, G_VARIANT_TYPE ("(s)")));
468 * get_sync_in_thread:
469 * @data: (type guint): delay in microseconds
471 * Sleep for a short time, then get a session bus connection and call
474 * Runs in a non-main thread.
476 * Returns: (transfer full): the connection
479 get_sync_in_thread (gpointer data
)
481 guint delay
= GPOINTER_TO_UINT (data
);
482 GError
*error
= NULL
;
483 GDBusConnection
*conn
;
487 conn
= g_bus_get_sync (G_BUS_TYPE_SESSION
, NULL
, &error
);
488 g_assert_no_error (error
);
490 ensure_connection_works (conn
);
496 test_threaded_singleton (void)
499 guint unref_wins
= 0;
502 if (g_test_thorough ())
507 for (i
= 0; i
< n
; i
++)
511 guint unref_delay
, get_delay
;
512 GDBusConnection
*new_conn
;
514 /* We want to be the last ref, so let it finish setting up */
515 for (j
= 0; j
< 100; j
++)
517 guint r
= g_atomic_int_get (&G_OBJECT (c
)->ref_count
);
522 g_debug ("run %u: refcount is %u, sleeping", i
, r
);
527 g_error ("connection had too many refs");
529 if (g_test_verbose () && (i
% (n
/50)) == 0)
530 g_printerr ("%u%%\n", ((i
* 100) / n
));
532 /* Delay for a random time on each side of the race, to perturb the
533 * timing. Ideally, we want each side to win half the races; these
534 * timings are about right on smcv's laptop.
536 unref_delay
= g_random_int_range (SLEEP_MIN_USEC
, SLEEP_MAX_USEC
);
537 get_delay
= g_random_int_range (SLEEP_MIN_USEC
/ 2, SLEEP_MAX_USEC
/ 2);
539 /* One half of the race is to call g_bus_get_sync... */
540 thread
= g_thread_new ("get_sync_in_thread", get_sync_in_thread
,
541 GUINT_TO_POINTER (get_delay
));
543 /* ... and the other half is to unref the shared connection, which must
544 * have exactly one ref at this point
546 g_usleep (unref_delay
);
549 /* Wait for the thread to run; see what it got */
550 new_conn
= g_thread_join (thread
);
552 /* If the thread won the race, it will have kept the same connection,
553 * and it'll have one ref
562 /* c is invalid now, but new_conn is suitable for the
568 ensure_connection_works (c
);
571 if (g_test_verbose ())
572 g_printerr ("Unref won %u races; Get won %u races\n", unref_wins
, get_wins
);
575 /* ---------------------------------------------------------------------------------------------------- */
585 g_test_init (&argc
, &argv
, NULL
);
589 /* this is safe; testserver will exit once the bus goes away */
590 path
= g_test_build_filename (G_TEST_BUILT
, "gdbus-testserver", NULL
);
591 g_assert (g_spawn_command_line_async (path
, NULL
));
594 ensure_gdbus_testserver_up ();
596 /* Create the connection in the main thread */
598 c
= g_bus_get_sync (G_BUS_TYPE_SESSION
, NULL
, &error
);
599 g_assert_no_error (error
);
600 g_assert (c
!= NULL
);
602 g_test_add_func ("/gdbus/delivery-in-thread", test_delivery_in_thread
);
603 g_test_add_func ("/gdbus/method-calls-in-thread", test_method_calls_in_thread
);
604 g_test_add_func ("/gdbus/threaded-singleton", test_threaded_singleton
);