mirror of
https://gitlab.gnome.org/GNOME/glib.git
synced 2025-01-07 13:16:18 +01:00
7c609f8142
Most D-Bus interfaces are domain-specific, but these interfaces from the D-Bus Specification are intended to be commonly used in any context for which they are found to be appropriate. Most of these use `gdbusprivate.h`. One exception is that `gio/tests/gdbus-example-*` redefine the constants locally: due to these files' dual role as part of the unit tests and as sample code, it seems desirable to ensure that they can still be compiled outside GLib. Signed-off-by: Simon McVittie <smcv@collabora.com>
413 lines
15 KiB
C
413 lines
15 KiB
C
/* GLib testing framework examples and tests
|
|
*
|
|
* Copyright © 2022 Endless OS Foundation, LLC
|
|
*
|
|
* SPDX-License-Identifier: LGPL-2.1-or-later
|
|
*
|
|
* 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.1 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, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
* SPDX-License-Identifier: LGPL-2.1-or-later
|
|
* Author: Philip Withnall <pwithnall@endlessos.org>
|
|
*/
|
|
|
|
#include <gio/gio.h>
|
|
#include <locale.h>
|
|
|
|
#include "gdbusprivate.h"
|
|
|
|
static void
|
|
test_dbus_basic (void)
|
|
{
|
|
GTestDBus *bus;
|
|
GDBusConnection *connection = NULL, *connection2 = NULL;
|
|
GDebugControllerDBus *controller = NULL;
|
|
gboolean old_value;
|
|
gboolean debug_enabled;
|
|
GError *local_error = NULL;
|
|
|
|
g_test_summary ("Smoketest for construction and setting of a #GDebugControllerDBus.");
|
|
|
|
/* Set up a test session bus and connection. */
|
|
bus = g_test_dbus_new (G_TEST_DBUS_NONE);
|
|
g_test_dbus_up (bus);
|
|
|
|
connection = g_bus_get_sync (G_BUS_TYPE_SESSION, NULL, &local_error);
|
|
g_assert_no_error (local_error);
|
|
|
|
/* Create a controller for this process. */
|
|
controller = g_debug_controller_dbus_new (connection, NULL, &local_error);
|
|
g_assert_no_error (local_error);
|
|
g_assert_nonnull (controller);
|
|
g_assert_true (G_IS_DEBUG_CONTROLLER_DBUS (controller));
|
|
|
|
/* Try enabling and disabling debug output from within the process. */
|
|
old_value = g_debug_controller_get_debug_enabled (G_DEBUG_CONTROLLER (controller));
|
|
|
|
g_debug_controller_set_debug_enabled (G_DEBUG_CONTROLLER (controller), TRUE);
|
|
g_assert_true (g_debug_controller_get_debug_enabled (G_DEBUG_CONTROLLER (controller)));
|
|
|
|
g_debug_controller_set_debug_enabled (G_DEBUG_CONTROLLER (controller), FALSE);
|
|
g_assert_false (g_debug_controller_get_debug_enabled (G_DEBUG_CONTROLLER (controller)));
|
|
|
|
/* Reset the debug state and check using g_object_get(), to exercise that. */
|
|
g_debug_controller_set_debug_enabled (G_DEBUG_CONTROLLER (controller), old_value);
|
|
|
|
g_object_get (G_OBJECT (controller),
|
|
"debug-enabled", &debug_enabled,
|
|
"connection", &connection2,
|
|
NULL);
|
|
g_assert_true (debug_enabled == old_value);
|
|
g_assert_true (connection2 == connection);
|
|
g_clear_object (&connection2);
|
|
|
|
g_debug_controller_dbus_stop (controller);
|
|
while (g_main_context_iteration (NULL, FALSE));
|
|
g_assert_finalize_object (controller);
|
|
g_clear_object (&connection);
|
|
|
|
g_test_dbus_down (bus);
|
|
g_clear_object (&bus);
|
|
}
|
|
|
|
static void
|
|
test_dbus_duplicate (void)
|
|
{
|
|
GTestDBus *bus;
|
|
GDBusConnection *connection = NULL;
|
|
GDebugControllerDBus *controller1 = NULL, *controller2 = NULL;
|
|
GError *local_error = NULL;
|
|
|
|
g_test_summary ("Test that creating a second #GDebugControllerDBus on the same D-Bus connection fails.");
|
|
|
|
/* Set up a test session bus and connection. */
|
|
bus = g_test_dbus_new (G_TEST_DBUS_NONE);
|
|
g_test_dbus_up (bus);
|
|
|
|
connection = g_bus_get_sync (G_BUS_TYPE_SESSION, NULL, &local_error);
|
|
g_assert_no_error (local_error);
|
|
|
|
/* Create a controller for this process. */
|
|
controller1 = g_debug_controller_dbus_new (connection, NULL, &local_error);
|
|
g_assert_no_error (local_error);
|
|
g_assert_nonnull (controller1);
|
|
|
|
/* And try creating a second one. */
|
|
controller2 = g_debug_controller_dbus_new (connection, NULL, &local_error);
|
|
g_assert_error (local_error, G_IO_ERROR, G_IO_ERROR_EXISTS);
|
|
g_assert_null (controller2);
|
|
g_clear_error (&local_error);
|
|
|
|
g_debug_controller_dbus_stop (controller1);
|
|
while (g_main_context_iteration (NULL, FALSE));
|
|
g_assert_finalize_object (controller1);
|
|
g_clear_object (&connection);
|
|
|
|
g_test_dbus_down (bus);
|
|
g_clear_object (&bus);
|
|
}
|
|
|
|
static void
|
|
async_result_cb (GObject *source_object,
|
|
GAsyncResult *result,
|
|
gpointer user_data)
|
|
{
|
|
GAsyncResult **result_out = user_data;
|
|
|
|
g_assert_null (*result_out);
|
|
*result_out = g_object_ref (result);
|
|
|
|
g_main_context_wakeup (g_main_context_get_thread_default ());
|
|
}
|
|
|
|
static gboolean
|
|
authorize_false_cb (GDebugControllerDBus *debug_controller,
|
|
GDBusMethodInvocation *invocation,
|
|
gpointer user_data)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
authorize_true_cb (GDebugControllerDBus *debug_controller,
|
|
GDBusMethodInvocation *invocation,
|
|
gpointer user_data)
|
|
{
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
notify_debug_enabled_cb (GObject *object,
|
|
GParamSpec *pspec,
|
|
gpointer user_data)
|
|
{
|
|
guint *notify_count_out = user_data;
|
|
|
|
*notify_count_out = *notify_count_out + 1;
|
|
}
|
|
|
|
static void
|
|
properties_changed_cb (GDBusConnection *connection,
|
|
const gchar *sender_name,
|
|
const gchar *object_path,
|
|
const gchar *interface_name,
|
|
const gchar *signal_name,
|
|
GVariant *parameters,
|
|
gpointer user_data)
|
|
{
|
|
guint *properties_changed_count_out = user_data;
|
|
|
|
*properties_changed_count_out = *properties_changed_count_out + 1;
|
|
g_main_context_wakeup (g_main_context_get_thread_default ());
|
|
}
|
|
|
|
static void
|
|
test_dbus_properties (void)
|
|
{
|
|
GTestDBus *bus;
|
|
GDBusConnection *controller_connection = NULL;
|
|
GDBusConnection *remote_connection = NULL;
|
|
GDebugControllerDBus *controller = NULL;
|
|
gboolean old_value;
|
|
GAsyncResult *result = NULL;
|
|
GVariant *reply = NULL;
|
|
GVariant *debug_enabled_variant = NULL;
|
|
gboolean debug_enabled;
|
|
GError *local_error = NULL;
|
|
gulong handler_id;
|
|
gulong notify_id;
|
|
guint notify_count = 0;
|
|
guint properties_changed_id;
|
|
guint properties_changed_count = 0;
|
|
|
|
g_test_summary ("Test getting and setting properties on a #GDebugControllerDBus.");
|
|
|
|
/* Set up a test session bus and connection. Set up a separate second
|
|
* connection to simulate a remote peer. */
|
|
bus = g_test_dbus_new (G_TEST_DBUS_NONE);
|
|
g_test_dbus_up (bus);
|
|
|
|
controller_connection = g_bus_get_sync (G_BUS_TYPE_SESSION, NULL, &local_error);
|
|
g_assert_no_error (local_error);
|
|
|
|
remote_connection = g_dbus_connection_new_for_address_sync (g_test_dbus_get_bus_address (bus),
|
|
G_DBUS_CONNECTION_FLAGS_AUTHENTICATION_CLIENT |
|
|
G_DBUS_CONNECTION_FLAGS_MESSAGE_BUS_CONNECTION,
|
|
NULL,
|
|
NULL,
|
|
&local_error);
|
|
g_assert_no_error (local_error);
|
|
|
|
/* Create a controller for this process. */
|
|
controller = g_debug_controller_dbus_new (controller_connection, NULL, &local_error);
|
|
g_assert_no_error (local_error);
|
|
g_assert_nonnull (controller);
|
|
g_assert_true (G_IS_DEBUG_CONTROLLER_DBUS (controller));
|
|
|
|
old_value = g_debug_controller_get_debug_enabled (G_DEBUG_CONTROLLER (controller));
|
|
notify_id = g_signal_connect (controller, "notify::debug-enabled", G_CALLBACK (notify_debug_enabled_cb), ¬ify_count);
|
|
|
|
properties_changed_id = g_dbus_connection_signal_subscribe (remote_connection,
|
|
g_dbus_connection_get_unique_name (controller_connection),
|
|
DBUS_INTERFACE_PROPERTIES,
|
|
"PropertiesChanged",
|
|
"/org/gtk/Debugging",
|
|
NULL,
|
|
G_DBUS_SIGNAL_FLAGS_NONE,
|
|
properties_changed_cb,
|
|
&properties_changed_count,
|
|
NULL);
|
|
|
|
/* Get the debug status remotely. */
|
|
g_dbus_connection_call (remote_connection,
|
|
g_dbus_connection_get_unique_name (controller_connection),
|
|
"/org/gtk/Debugging",
|
|
DBUS_INTERFACE_PROPERTIES,
|
|
"Get",
|
|
g_variant_new ("(ss)", "org.gtk.Debugging", "DebugEnabled"),
|
|
G_VARIANT_TYPE ("(v)"),
|
|
G_DBUS_CALL_FLAGS_NONE,
|
|
-1,
|
|
NULL,
|
|
async_result_cb,
|
|
&result);
|
|
g_assert_no_error (local_error);
|
|
|
|
while (result == NULL)
|
|
g_main_context_iteration (NULL, TRUE);
|
|
|
|
reply = g_dbus_connection_call_finish (remote_connection, result, &local_error);
|
|
g_assert_no_error (local_error);
|
|
g_clear_object (&result);
|
|
|
|
g_variant_get (reply, "(v)", &debug_enabled_variant);
|
|
debug_enabled = g_variant_get_boolean (debug_enabled_variant);
|
|
g_assert_true (debug_enabled == old_value);
|
|
g_assert_cmpuint (notify_count, ==, 0);
|
|
g_assert_cmpuint (properties_changed_count, ==, 0);
|
|
|
|
g_clear_pointer (&debug_enabled_variant, g_variant_unref);
|
|
g_clear_pointer (&reply, g_variant_unref);
|
|
|
|
/* Set the debug status remotely. The first attempt should fail due to no
|
|
* authorisation handler being connected. The second should fail due to the
|
|
* now-connected handler returning %FALSE. The third attempt should
|
|
* succeed. */
|
|
g_dbus_connection_call (remote_connection,
|
|
g_dbus_connection_get_unique_name (controller_connection),
|
|
"/org/gtk/Debugging",
|
|
"org.gtk.Debugging",
|
|
"SetDebugEnabled",
|
|
g_variant_new ("(b)", !old_value),
|
|
NULL,
|
|
G_DBUS_CALL_FLAGS_NONE,
|
|
-1,
|
|
NULL,
|
|
async_result_cb,
|
|
&result);
|
|
|
|
while (result == NULL)
|
|
g_main_context_iteration (NULL, TRUE);
|
|
|
|
reply = g_dbus_connection_call_finish (remote_connection, result, &local_error);
|
|
g_assert_error (local_error, G_DBUS_ERROR, G_DBUS_ERROR_ACCESS_DENIED);
|
|
g_clear_object (&result);
|
|
g_clear_error (&local_error);
|
|
|
|
g_assert_true (g_debug_controller_get_debug_enabled (G_DEBUG_CONTROLLER (controller)) == old_value);
|
|
g_assert_cmpuint (notify_count, ==, 0);
|
|
g_assert_cmpuint (properties_changed_count, ==, 0);
|
|
|
|
g_clear_pointer (&debug_enabled_variant, g_variant_unref);
|
|
g_clear_pointer (&reply, g_variant_unref);
|
|
|
|
/* Attach an authorisation handler and try again. */
|
|
handler_id = g_signal_connect (controller, "authorize", G_CALLBACK (authorize_false_cb), NULL);
|
|
|
|
g_dbus_connection_call (remote_connection,
|
|
g_dbus_connection_get_unique_name (controller_connection),
|
|
"/org/gtk/Debugging",
|
|
"org.gtk.Debugging",
|
|
"SetDebugEnabled",
|
|
g_variant_new ("(b)", !old_value),
|
|
NULL,
|
|
G_DBUS_CALL_FLAGS_NONE,
|
|
-1,
|
|
NULL,
|
|
async_result_cb,
|
|
&result);
|
|
|
|
while (result == NULL)
|
|
g_main_context_iteration (NULL, TRUE);
|
|
|
|
reply = g_dbus_connection_call_finish (remote_connection, result, &local_error);
|
|
g_assert_error (local_error, G_DBUS_ERROR, G_DBUS_ERROR_ACCESS_DENIED);
|
|
g_clear_object (&result);
|
|
g_clear_error (&local_error);
|
|
|
|
g_assert_true (g_debug_controller_get_debug_enabled (G_DEBUG_CONTROLLER (controller)) == old_value);
|
|
g_assert_cmpuint (notify_count, ==, 0);
|
|
g_assert_cmpuint (properties_changed_count, ==, 0);
|
|
|
|
g_clear_pointer (&debug_enabled_variant, g_variant_unref);
|
|
g_clear_pointer (&reply, g_variant_unref);
|
|
|
|
g_signal_handler_disconnect (controller, handler_id);
|
|
handler_id = 0;
|
|
|
|
/* Attach another signal handler which will grant access, and try again. */
|
|
handler_id = g_signal_connect (controller, "authorize", G_CALLBACK (authorize_true_cb), NULL);
|
|
|
|
g_dbus_connection_call (remote_connection,
|
|
g_dbus_connection_get_unique_name (controller_connection),
|
|
"/org/gtk/Debugging",
|
|
"org.gtk.Debugging",
|
|
"SetDebugEnabled",
|
|
g_variant_new ("(b)", !old_value),
|
|
NULL,
|
|
G_DBUS_CALL_FLAGS_NONE,
|
|
-1,
|
|
NULL,
|
|
async_result_cb,
|
|
&result);
|
|
|
|
while (result == NULL)
|
|
g_main_context_iteration (NULL, TRUE);
|
|
|
|
reply = g_dbus_connection_call_finish (remote_connection, result, &local_error);
|
|
g_assert_no_error (local_error);
|
|
g_clear_object (&result);
|
|
|
|
g_assert_true (g_debug_controller_get_debug_enabled (G_DEBUG_CONTROLLER (controller)) == !old_value);
|
|
g_assert_cmpuint (notify_count, ==, 1);
|
|
g_assert_cmpuint (properties_changed_count, ==, 1);
|
|
|
|
g_clear_pointer (&debug_enabled_variant, g_variant_unref);
|
|
g_clear_pointer (&reply, g_variant_unref);
|
|
|
|
g_signal_handler_disconnect (controller, handler_id);
|
|
handler_id = 0;
|
|
|
|
/* Set the debug status locally. */
|
|
g_debug_controller_set_debug_enabled (G_DEBUG_CONTROLLER (controller), old_value);
|
|
g_assert_true (g_debug_controller_get_debug_enabled (G_DEBUG_CONTROLLER (controller)) == old_value);
|
|
g_assert_cmpuint (notify_count, ==, 2);
|
|
|
|
while (properties_changed_count != 2)
|
|
g_main_context_iteration (NULL, TRUE);
|
|
|
|
g_assert_cmpuint (properties_changed_count, ==, 2);
|
|
|
|
g_signal_handler_disconnect (controller, notify_id);
|
|
notify_id = 0;
|
|
|
|
g_dbus_connection_signal_unsubscribe (remote_connection, properties_changed_id);
|
|
properties_changed_id = 0;
|
|
|
|
g_debug_controller_dbus_stop (controller);
|
|
while (g_main_context_iteration (NULL, FALSE));
|
|
g_assert_finalize_object (controller);
|
|
g_clear_object (&controller_connection);
|
|
g_clear_object (&remote_connection);
|
|
|
|
g_test_dbus_down (bus);
|
|
g_clear_object (&bus);
|
|
}
|
|
|
|
static GLogWriterOutput
|
|
noop_log_writer_cb (GLogLevelFlags log_level,
|
|
const GLogField *fields,
|
|
gsize n_fields,
|
|
gpointer user_data)
|
|
{
|
|
return G_LOG_WRITER_HANDLED;
|
|
}
|
|
|
|
int
|
|
main (int argc,
|
|
char *argv[])
|
|
{
|
|
setlocale (LC_ALL, "");
|
|
g_test_init (&argc, &argv, NULL);
|
|
|
|
/* Ignore the log messages, as the debug controller prints one when debug is
|
|
* enabled/disabled, and if debug is enabled then that will escape to stdout. */
|
|
g_log_set_writer_func (noop_log_writer_cb, NULL, NULL);
|
|
|
|
g_test_add_func ("/debug-controller/dbus/basic", test_dbus_basic);
|
|
g_test_add_func ("/debug-controller/dbus/duplicate", test_dbus_duplicate);
|
|
g_test_add_func ("/debug-controller/dbus/properties", test_dbus_properties);
|
|
|
|
return g_test_run ();
|
|
}
|