mirror of
				https://gitlab.gnome.org/GNOME/glib.git
				synced 2025-10-31 00:12:19 +01:00 
			
		
		
		
	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>
		
			
				
	
	
		
			328 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			328 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /* GLib testing framework examples and tests
 | |
|  *
 | |
|  * Copyright (C) 2008-2010 Red Hat, Inc.
 | |
|  *
 | |
|  * 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/>.
 | |
|  *
 | |
|  * Author: David Zeuthen <davidz@redhat.com>
 | |
|  */
 | |
| 
 | |
| #include <gio/gio.h>
 | |
| #include <unistd.h>
 | |
| #include <string.h>
 | |
| 
 | |
| #include "gdbusprivate.h"
 | |
| #include "gdbus-tests.h"
 | |
| 
 | |
| /* all tests rely on a shared mainloop */
 | |
| static GMainLoop *loop = NULL;
 | |
| 
 | |
| /* ---------------------------------------------------------------------------------------------------- */
 | |
| /* Test introspection parser */
 | |
| /* ---------------------------------------------------------------------------------------------------- */
 | |
| 
 | |
| static void
 | |
| test_introspection (GDBusProxy *proxy)
 | |
| {
 | |
|   GError *error;
 | |
|   const gchar *xml_data;
 | |
|   GDBusNodeInfo *node_info;
 | |
|   GDBusInterfaceInfo *interface_info;
 | |
|   GDBusMethodInfo *method_info;
 | |
|   GDBusSignalInfo *signal_info;
 | |
|   GVariant *result;
 | |
| 
 | |
|   error = NULL;
 | |
| 
 | |
|   /*
 | |
|    * Invoke Introspect(), then parse the output.
 | |
|    */
 | |
|   result = g_dbus_proxy_call_sync (proxy,
 | |
|                                    DBUS_INTERFACE_INTROSPECTABLE ".Introspect",
 | |
|                                    NULL,
 | |
|                                    G_DBUS_CALL_FLAGS_NONE,
 | |
|                                    -1,
 | |
|                                    NULL,
 | |
|                                    &error);
 | |
|   g_assert_no_error (error);
 | |
|   g_assert (result != NULL);
 | |
|   g_variant_get (result, "(&s)", &xml_data);
 | |
| 
 | |
|   node_info = g_dbus_node_info_new_for_xml (xml_data, &error);
 | |
|   g_assert_no_error (error);
 | |
|   g_assert (node_info != NULL);
 | |
| 
 | |
|   /* for now we only check a couple of things. TODO: check more things */
 | |
| 
 | |
|   interface_info = g_dbus_node_info_lookup_interface (node_info, "com.example.NonExistantInterface");
 | |
|   g_assert (interface_info == NULL);
 | |
| 
 | |
|   interface_info = g_dbus_node_info_lookup_interface (node_info, DBUS_INTERFACE_INTROSPECTABLE);
 | |
|   g_assert (interface_info != NULL);
 | |
|   method_info = g_dbus_interface_info_lookup_method (interface_info, "NonExistantMethod");
 | |
|   g_assert (method_info == NULL);
 | |
|   method_info = g_dbus_interface_info_lookup_method (interface_info, "Introspect");
 | |
|   g_assert (method_info != NULL);
 | |
|   g_assert (method_info->in_args != NULL);
 | |
|   g_assert (method_info->in_args[0] == NULL);
 | |
|   g_assert (method_info->out_args != NULL);
 | |
|   g_assert (method_info->out_args[0] != NULL);
 | |
|   g_assert (method_info->out_args[1] == NULL);
 | |
|   g_assert_cmpstr (method_info->out_args[0]->signature, ==, "s");
 | |
| 
 | |
|   interface_info = g_dbus_node_info_lookup_interface (node_info, "com.example.Frob");
 | |
|   g_assert (interface_info != NULL);
 | |
|   signal_info = g_dbus_interface_info_lookup_signal (interface_info, "TestSignal");
 | |
|   g_assert (signal_info != NULL);
 | |
|   g_assert (signal_info->args != NULL);
 | |
|   g_assert (signal_info->args[0] != NULL);
 | |
|   g_assert_cmpstr (signal_info->args[0]->signature, ==, "s");
 | |
|   g_assert (signal_info->args[1] != NULL);
 | |
|   g_assert_cmpstr (signal_info->args[1]->signature, ==, "o");
 | |
|   g_assert (signal_info->args[2] != NULL);
 | |
|   g_assert_cmpstr (signal_info->args[2]->signature, ==, "v");
 | |
|   g_assert (signal_info->args[3] == NULL);
 | |
| 
 | |
|   g_dbus_node_info_unref (node_info);
 | |
|   g_variant_unref (result);
 | |
| 
 | |
|   g_main_loop_quit (loop);
 | |
| }
 | |
| 
 | |
| static void
 | |
| test_introspection_parser (void)
 | |
| {
 | |
|   GDBusProxy *proxy;
 | |
|   GDBusConnection *connection;
 | |
|   GError *error;
 | |
| 
 | |
|   error = NULL;
 | |
|   connection = g_bus_get_sync (G_BUS_TYPE_SESSION,
 | |
|                                NULL,
 | |
|                                &error);
 | |
|   g_assert_no_error (error);
 | |
|   error = NULL;
 | |
|   proxy = g_dbus_proxy_new_sync (connection,
 | |
|                                  G_DBUS_PROXY_FLAGS_NONE,
 | |
|                                  NULL,                      /* GDBusInterfaceInfo */
 | |
|                                  "com.example.TestService", /* name */
 | |
|                                  "/com/example/TestObject", /* object path */
 | |
|                                  "com.example.Frob",        /* interface */
 | |
|                                  NULL, /* GCancellable */
 | |
|                                  &error);
 | |
|   g_assert_no_error (error);
 | |
| 
 | |
|   /* this is safe; testserver will exit once the bus goes away */
 | |
|   g_assert (g_spawn_command_line_async (g_test_get_filename (G_TEST_BUILT, "gdbus-testserver", NULL), NULL));
 | |
| 
 | |
|   _g_assert_property_notify (proxy, "g-name-owner");
 | |
| 
 | |
|   test_introspection (proxy);
 | |
| 
 | |
|   g_object_unref (proxy);
 | |
|   g_object_unref (connection);
 | |
| }
 | |
| 
 | |
| /* check that a parse-generate roundtrip produces identical results
 | |
|  */
 | |
| static void
 | |
| test_generate (void)
 | |
| {
 | |
|   GDBusNodeInfo *info;
 | |
|   GDBusNodeInfo *info2;
 | |
|   GDBusInterfaceInfo *iinfo;
 | |
|   GDBusMethodInfo *minfo;
 | |
|   GDBusSignalInfo *sinfo;
 | |
|   GDBusArgInfo *arginfo;
 | |
|   GDBusPropertyInfo *pinfo;
 | |
|   GDBusAnnotationInfo *aninfo;
 | |
|   const gchar *data =
 | |
|   "  <node>"
 | |
|   "    <interface name='com.example.Frob'>"
 | |
|   "      <annotation name='foo' value='bar'/>"
 | |
|   "      <method name='PairReturn'>"
 | |
|   "        <annotation name='org.freedesktop.DBus.GLib.Async' value=''/>"
 | |
|   "        <arg type='u' name='somenumber' direction='in'/>"
 | |
|   "        <arg type='s' name='somestring' direction='out'/>"
 | |
|   "      </method>"
 | |
|   "      <signal name='HelloWorld'>"
 | |
|   "        <arg type='s' name='greeting' direction='out'/>"
 | |
|   "      </signal>"
 | |
|   "      <method name='Sleep'>"
 | |
|   "        <arg type='i' name='timeout' direction='in'/>"
 | |
|   "      </method>"
 | |
|   "      <property name='y' type='y' access='readwrite'>"
 | |
|   "        <annotation name='needs-escaping' value='bar<>'"'/>"
 | |
|   "      </property>"
 | |
|   "    </interface>"
 | |
|   "  </node>";
 | |
| 
 | |
|   GString *string;
 | |
|   GString *string2;
 | |
|   GError *error;
 | |
| 
 | |
|   error = NULL;
 | |
|   info = g_dbus_node_info_new_for_xml (data, &error);
 | |
|   g_assert_no_error (error);
 | |
| 
 | |
|   iinfo = g_dbus_node_info_lookup_interface (info, "com.example.Frob");
 | |
|   aninfo = iinfo->annotations[0];
 | |
|   g_assert_cmpstr (aninfo->key, ==, "foo");
 | |
|   g_assert_cmpstr (aninfo->value, ==, "bar");
 | |
|   g_assert (iinfo->annotations[1] == NULL);
 | |
|   minfo = g_dbus_interface_info_lookup_method (iinfo, "PairReturn");
 | |
|   g_assert_cmpstr (g_dbus_annotation_info_lookup (minfo->annotations, "org.freedesktop.DBus.GLib.Async"), ==, "");
 | |
|   arginfo = minfo->in_args[0];
 | |
|   g_assert_cmpstr (arginfo->name, ==, "somenumber");
 | |
|   g_assert_cmpstr (arginfo->signature, ==, "u");
 | |
|   g_assert (minfo->in_args[1] == NULL);
 | |
|   arginfo = minfo->out_args[0];
 | |
|   g_assert_cmpstr (arginfo->name, ==, "somestring");
 | |
|   g_assert_cmpstr (arginfo->signature, ==, "s");
 | |
|   g_assert (minfo->out_args[1] == NULL);
 | |
|   sinfo = g_dbus_interface_info_lookup_signal (iinfo, "HelloWorld");
 | |
|   arginfo = sinfo->args[0];
 | |
|   g_assert_cmpstr (arginfo->name, ==, "greeting");
 | |
|   g_assert_cmpstr (arginfo->signature, ==, "s");
 | |
|   g_assert (sinfo->args[1] == NULL);
 | |
|   pinfo = g_dbus_interface_info_lookup_property (iinfo, "y");
 | |
|   g_assert_cmpstr (pinfo->signature, ==, "y");
 | |
|   g_assert_cmpint (pinfo->flags, ==, G_DBUS_PROPERTY_INFO_FLAGS_READABLE |
 | |
|                                      G_DBUS_PROPERTY_INFO_FLAGS_WRITABLE);
 | |
| 
 | |
|   string = g_string_new ("");
 | |
|   g_dbus_node_info_generate_xml (info, 2, string);
 | |
| 
 | |
|   info2 = g_dbus_node_info_new_for_xml (string->str, &error);
 | |
|   string2 = g_string_new ("");
 | |
|   g_dbus_node_info_generate_xml (info2, 2, string2);
 | |
| 
 | |
|   g_assert_cmpstr (string->str, ==, string2->str);
 | |
|   g_string_free (string, TRUE);
 | |
|   g_string_free (string2, TRUE);
 | |
| 
 | |
|   g_dbus_node_info_unref (info);
 | |
|   g_dbus_node_info_unref (info2);
 | |
| }
 | |
| 
 | |
| /* test that omitted direction attributes default to 'out' for signals,
 | |
|  * and 'in' for methods.
 | |
|  */
 | |
| static void
 | |
| test_default_direction (void)
 | |
| {
 | |
|   GDBusNodeInfo *info;
 | |
|   GDBusInterfaceInfo *iinfo;
 | |
|   GDBusMethodInfo *minfo;
 | |
|   GDBusSignalInfo *sinfo;
 | |
|   GDBusArgInfo *arginfo;
 | |
|   const gchar *data =
 | |
|   "  <node>"
 | |
|   "    <interface name='com.example.Frob'>"
 | |
|   "      <signal name='HelloWorld'>"
 | |
|   "        <arg type='s' name='greeting'/>"
 | |
|   "      </signal>"
 | |
|   "      <method name='Sleep'>"
 | |
|   "        <arg type='i' name='timeout'/>"
 | |
|   "      </method>"
 | |
|   "    </interface>"
 | |
|   "  </node>";
 | |
| 
 | |
|   GError *error;
 | |
| 
 | |
|   error = NULL;
 | |
|   info = g_dbus_node_info_new_for_xml (data, &error);
 | |
|   g_assert_no_error (error);
 | |
| 
 | |
|   iinfo = g_dbus_node_info_lookup_interface (info, "com.example.Frob");
 | |
|   sinfo = g_dbus_interface_info_lookup_signal (iinfo, "HelloWorld");
 | |
|   g_assert (sinfo->args != NULL);
 | |
|   arginfo = sinfo->args[0];
 | |
|   g_assert_cmpstr (arginfo->name, ==, "greeting");
 | |
|   g_assert (sinfo->args[1] == NULL);
 | |
|   minfo = g_dbus_interface_info_lookup_method (iinfo, "Sleep");
 | |
|   g_assert (minfo->in_args != NULL);
 | |
|   arginfo = minfo->in_args[0];
 | |
|   g_assert_cmpstr (arginfo->name, ==, "timeout");
 | |
|   g_assert (minfo->in_args[1] == NULL);
 | |
| 
 | |
|   g_dbus_node_info_unref (info);
 | |
| }
 | |
| 
 | |
| static void
 | |
| test_extra_data (void)
 | |
| {
 | |
|   GDBusNodeInfo *info;
 | |
|   const gchar *data =
 | |
|   "  <node>"
 | |
|   "    <interface name='com.example.Frob' version='1.0'>"
 | |
|   "      <doc:doc><doc:description><doc:para>Blah blah</doc:para></doc:description></doc:doc>"
 | |
|   "      <method name='DownloadPackages'>"
 | |
|   "        <arg type='u' name='somenumber' direction='in'>"
 | |
|   "          <doc:doc><doc:summary><doc:para>"
 | |
|   "            See <doc:ulink url='http:///example.com'>example</doc:ulink>"
 | |
|   "          </doc:para></doc:summary></doc:doc>"
 | |
|   "        </arg>"
 | |
|   "        <arg type='s' name='somestring' direction='out'>"
 | |
|   "          <doc:doc><doc:summary><doc:para>"
 | |
|   "            More docs"
 | |
|   "          </doc:para></doc:summary></doc:doc>"
 | |
|   "        </arg>"
 | |
|   "      </method>"
 | |
|   "      <signal name='HelloWorld'>"
 | |
|   "        <arg type='s' name='somestring'/>"
 | |
|   "      </signal>"
 | |
|   "      <method name='Sleep'>"
 | |
|   "        <arg type='i' name='timeout' direction='in'/>"
 | |
|   "      </method>"
 | |
|   "      <property name='y' type='y' access='readwrite'/>"
 | |
|   "    </interface>"
 | |
|   "  </node>";
 | |
|   GError *error;
 | |
| 
 | |
|   error = NULL;
 | |
|   info = g_dbus_node_info_new_for_xml (data, &error);
 | |
|   g_assert_no_error (error);
 | |
| 
 | |
|   g_dbus_node_info_unref (info);
 | |
| }
 | |
| 
 | |
| /* ---------------------------------------------------------------------------------------------------- */
 | |
| 
 | |
| int
 | |
| main (int   argc,
 | |
|       char *argv[])
 | |
| {
 | |
|   gint ret;
 | |
| 
 | |
|   g_test_init (&argc, &argv, G_TEST_OPTION_ISOLATE_DIRS, NULL);
 | |
| 
 | |
|   /* all the tests rely on a shared main loop */
 | |
|   loop = g_main_loop_new (NULL, FALSE);
 | |
| 
 | |
|   g_test_add_func ("/gdbus/introspection-parser", test_introspection_parser);
 | |
|   g_test_add_func ("/gdbus/introspection-generate", test_generate);
 | |
|   g_test_add_func ("/gdbus/introspection-default-direction", test_default_direction);
 | |
|   g_test_add_func ("/gdbus/introspection-extra-data", test_extra_data);
 | |
| 
 | |
|   ret = session_bus_run ();
 | |
| 
 | |
|   while (g_main_context_iteration (NULL, FALSE));
 | |
|   g_main_loop_unref (loop);
 | |
| 
 | |
|   return ret;
 | |
| }
 |