Blame gio/tests/gdbus-example-server.c

Packit ae235b
#include <gio/gio.h>
Packit ae235b
#include <stdlib.h>
Packit ae235b
Packit ae235b
#ifdef G_OS_UNIX
Packit ae235b
#include <gio/gunixfdlist.h>
Packit ae235b
/* For STDOUT_FILENO */
Packit ae235b
#include <unistd.h>
Packit ae235b
#endif
Packit ae235b
Packit ae235b
/* ---------------------------------------------------------------------------------------------------- */
Packit ae235b
Packit ae235b
static GDBusNodeInfo *introspection_data = NULL;
Packit ae235b
Packit ae235b
/* Introspection data for the service we are exporting */
Packit ae235b
static const gchar introspection_xml[] =
Packit ae235b
  "<node>"
Packit ae235b
  "  <interface name='org.gtk.GDBus.TestInterface'>"
Packit ae235b
  "    <annotation name='org.gtk.GDBus.Annotation' value='OnInterface'/>"
Packit ae235b
  "    <annotation name='org.gtk.GDBus.Annotation' value='AlsoOnInterface'/>"
Packit ae235b
  "    <method name='HelloWorld'>"
Packit ae235b
  "      <annotation name='org.gtk.GDBus.Annotation' value='OnMethod'/>"
Packit ae235b
  "      <arg type='s' name='greeting' direction='in'/>"
Packit ae235b
  "      <arg type='s' name='response' direction='out'/>"
Packit ae235b
  "    </method>"
Packit ae235b
  "    <method name='EmitSignal'>"
Packit ae235b
  "      <arg type='d' name='speed_in_mph' direction='in'>"
Packit ae235b
  "        <annotation name='org.gtk.GDBus.Annotation' value='OnArg'/>"
Packit ae235b
  "      </arg>"
Packit ae235b
  "    </method>"
Packit ae235b
  "    <method name='GimmeStdout'/>"
Packit ae235b
  "    <signal name='VelocityChanged'>"
Packit ae235b
  "      <annotation name='org.gtk.GDBus.Annotation' value='Onsignal'/>"
Packit ae235b
  "      <arg type='d' name='speed_in_mph'/>"
Packit ae235b
  "      <arg type='s' name='speed_as_string'>"
Packit ae235b
  "        <annotation name='org.gtk.GDBus.Annotation' value='OnArg_NonFirst'/>"
Packit ae235b
  "      </arg>"
Packit ae235b
  "    </signal>"
Packit ae235b
  "    <property type='s' name='FluxCapicitorName' access='read'>"
Packit ae235b
  "      <annotation name='org.gtk.GDBus.Annotation' value='OnProperty'>"
Packit ae235b
  "        <annotation name='org.gtk.GDBus.Annotation' value='OnAnnotation_YesThisIsCrazy'/>"
Packit ae235b
  "      </annotation>"
Packit ae235b
  "    </property>"
Packit ae235b
  "    <property type='s' name='Title' access='readwrite'/>"
Packit ae235b
  "    <property type='s' name='ReadingAlwaysThrowsError' access='read'/>"
Packit ae235b
  "    <property type='s' name='WritingAlwaysThrowsError' access='readwrite'/>"
Packit ae235b
  "    <property type='s' name='OnlyWritable' access='write'/>"
Packit ae235b
  "    <property type='s' name='Foo' access='read'/>"
Packit ae235b
  "    <property type='s' name='Bar' access='read'/>"
Packit ae235b
  "  </interface>"
Packit ae235b
  "</node>";
Packit ae235b
Packit ae235b
/* ---------------------------------------------------------------------------------------------------- */
Packit ae235b
Packit ae235b
static void
Packit ae235b
handle_method_call (GDBusConnection       *connection,
Packit ae235b
                    const gchar           *sender,
Packit ae235b
                    const gchar           *object_path,
Packit ae235b
                    const gchar           *interface_name,
Packit ae235b
                    const gchar           *method_name,
Packit ae235b
                    GVariant              *parameters,
Packit ae235b
                    GDBusMethodInvocation *invocation,
Packit ae235b
                    gpointer               user_data)
Packit ae235b
{
Packit ae235b
  if (g_strcmp0 (method_name, "HelloWorld") == 0)
Packit ae235b
    {
Packit ae235b
      const gchar *greeting;
Packit ae235b
Packit ae235b
      g_variant_get (parameters, "(&s)", &greeting);
Packit ae235b
Packit ae235b
      if (g_strcmp0 (greeting, "Return Unregistered") == 0)
Packit ae235b
        {
Packit ae235b
          g_dbus_method_invocation_return_error (invocation,
Packit ae235b
                                                 G_IO_ERROR,
Packit ae235b
                                                 G_IO_ERROR_FAILED_HANDLED,
Packit ae235b
                                                 "As requested, here's a GError not registered (G_IO_ERROR_FAILED_HANDLED)");
Packit ae235b
        }
Packit ae235b
      else if (g_strcmp0 (greeting, "Return Registered") == 0)
Packit ae235b
        {
Packit ae235b
          g_dbus_method_invocation_return_error (invocation,
Packit ae235b
                                                 G_DBUS_ERROR,
Packit ae235b
                                                 G_DBUS_ERROR_MATCH_RULE_NOT_FOUND,
Packit ae235b
                                                 "As requested, here's a GError that is registered (G_DBUS_ERROR_MATCH_RULE_NOT_FOUND)");
Packit ae235b
        }
Packit ae235b
      else if (g_strcmp0 (greeting, "Return Raw") == 0)
Packit ae235b
        {
Packit ae235b
          g_dbus_method_invocation_return_dbus_error (invocation,
Packit ae235b
                                                      "org.gtk.GDBus.SomeErrorName",
Packit ae235b
                                                      "As requested, here's a raw D-Bus error");
Packit ae235b
        }
Packit ae235b
      else
Packit ae235b
        {
Packit ae235b
          gchar *response;
Packit ae235b
          response = g_strdup_printf ("You greeted me with '%s'. Thanks!", greeting);
Packit ae235b
          g_dbus_method_invocation_return_value (invocation,
Packit ae235b
                                                 g_variant_new ("(s)", response));
Packit ae235b
          g_free (response);
Packit ae235b
        }
Packit ae235b
    }
Packit ae235b
  else if (g_strcmp0 (method_name, "EmitSignal") == 0)
Packit ae235b
    {
Packit ae235b
      GError *local_error;
Packit ae235b
      gdouble speed_in_mph;
Packit ae235b
      gchar *speed_as_string;
Packit ae235b
Packit ae235b
      g_variant_get (parameters, "(d)", &speed_in_mph);
Packit ae235b
      speed_as_string = g_strdup_printf ("%g mph!", speed_in_mph);
Packit ae235b
Packit ae235b
      local_error = NULL;
Packit ae235b
      g_dbus_connection_emit_signal (connection,
Packit ae235b
                                     NULL,
Packit ae235b
                                     object_path,
Packit ae235b
                                     interface_name,
Packit ae235b
                                     "VelocityChanged",
Packit ae235b
                                     g_variant_new ("(ds)",
Packit ae235b
                                                    speed_in_mph,
Packit ae235b
                                                    speed_as_string),
Packit ae235b
                                     &local_error);
Packit ae235b
      g_assert_no_error (local_error);
Packit ae235b
      g_free (speed_as_string);
Packit ae235b
Packit ae235b
      g_dbus_method_invocation_return_value (invocation, NULL);
Packit ae235b
    }
Packit ae235b
  else if (g_strcmp0 (method_name, "GimmeStdout") == 0)
Packit ae235b
    {
Packit ae235b
#ifdef G_OS_UNIX
Packit ae235b
      if (g_dbus_connection_get_capabilities (connection) & G_DBUS_CAPABILITY_FLAGS_UNIX_FD_PASSING)
Packit ae235b
        {
Packit ae235b
          GDBusMessage *reply;
Packit ae235b
          GUnixFDList *fd_list;
Packit ae235b
          GError *error;
Packit ae235b
Packit ae235b
          fd_list = g_unix_fd_list_new ();
Packit ae235b
          error = NULL;
Packit ae235b
          g_unix_fd_list_append (fd_list, STDOUT_FILENO, &error);
Packit ae235b
          g_assert_no_error (error);
Packit ae235b
Packit ae235b
          reply = g_dbus_message_new_method_reply (g_dbus_method_invocation_get_message (invocation));
Packit ae235b
          g_dbus_message_set_unix_fd_list (reply, fd_list);
Packit ae235b
Packit ae235b
          error = NULL;
Packit ae235b
          g_dbus_connection_send_message (connection,
Packit ae235b
                                          reply,
Packit ae235b
                                          G_DBUS_SEND_MESSAGE_FLAGS_NONE,
Packit ae235b
                                          NULL, /* out_serial */
Packit ae235b
                                          &error);
Packit ae235b
          g_assert_no_error (error);
Packit ae235b
Packit ae235b
          g_object_unref (invocation);
Packit ae235b
          g_object_unref (fd_list);
Packit ae235b
          g_object_unref (reply);
Packit ae235b
        }
Packit ae235b
      else
Packit ae235b
        {
Packit ae235b
          g_dbus_method_invocation_return_dbus_error (invocation,
Packit ae235b
                                                      "org.gtk.GDBus.Failed",
Packit ae235b
                                                      "Your message bus daemon does not support file descriptor passing (need D-Bus >= 1.3.0)");
Packit ae235b
        }
Packit ae235b
#else
Packit ae235b
      g_dbus_method_invocation_return_dbus_error (invocation,
Packit ae235b
                                                  "org.gtk.GDBus.NotOnUnix",
Packit ae235b
                                                  "Your OS does not support file descriptor passing");
Packit ae235b
#endif
Packit ae235b
    }
Packit ae235b
}
Packit ae235b
Packit ae235b
static gchar *_global_title = NULL;
Packit ae235b
Packit ae235b
static gboolean swap_a_and_b = FALSE;
Packit ae235b
Packit ae235b
static GVariant *
Packit ae235b
handle_get_property (GDBusConnection  *connection,
Packit ae235b
                     const gchar      *sender,
Packit ae235b
                     const gchar      *object_path,
Packit ae235b
                     const gchar      *interface_name,
Packit ae235b
                     const gchar      *property_name,
Packit ae235b
                     GError          **error,
Packit ae235b
                     gpointer          user_data)
Packit ae235b
{
Packit ae235b
  GVariant *ret;
Packit ae235b
Packit ae235b
  ret = NULL;
Packit ae235b
  if (g_strcmp0 (property_name, "FluxCapicitorName") == 0)
Packit ae235b
    {
Packit ae235b
      ret = g_variant_new_string ("DeLorean");
Packit ae235b
    }
Packit ae235b
  else if (g_strcmp0 (property_name, "Title") == 0)
Packit ae235b
    {
Packit ae235b
      if (_global_title == NULL)
Packit ae235b
        _global_title = g_strdup ("Back To C!");
Packit ae235b
      ret = g_variant_new_string (_global_title);
Packit ae235b
    }
Packit ae235b
  else if (g_strcmp0 (property_name, "ReadingAlwaysThrowsError") == 0)
Packit ae235b
    {
Packit ae235b
      g_set_error (error,
Packit ae235b
                   G_IO_ERROR,
Packit ae235b
                   G_IO_ERROR_FAILED,
Packit ae235b
                   "Hello %s. I thought I said reading this property "
Packit ae235b
                   "always results in an error. kthxbye",
Packit ae235b
                   sender);
Packit ae235b
    }
Packit ae235b
  else if (g_strcmp0 (property_name, "WritingAlwaysThrowsError") == 0)
Packit ae235b
    {
Packit ae235b
      ret = g_variant_new_string ("There's no home like home");
Packit ae235b
    }
Packit ae235b
  else if (g_strcmp0 (property_name, "Foo") == 0)
Packit ae235b
    {
Packit ae235b
      ret = g_variant_new_string (swap_a_and_b ? "Tock" : "Tick");
Packit ae235b
    }
Packit ae235b
  else if (g_strcmp0 (property_name, "Bar") == 0)
Packit ae235b
    {
Packit ae235b
      ret = g_variant_new_string (swap_a_and_b ? "Tick" : "Tock");
Packit ae235b
    }
Packit ae235b
Packit ae235b
  return ret;
Packit ae235b
}
Packit ae235b
Packit ae235b
static gboolean
Packit ae235b
handle_set_property (GDBusConnection  *connection,
Packit ae235b
                     const gchar      *sender,
Packit ae235b
                     const gchar      *object_path,
Packit ae235b
                     const gchar      *interface_name,
Packit ae235b
                     const gchar      *property_name,
Packit ae235b
                     GVariant         *value,
Packit ae235b
                     GError          **error,
Packit ae235b
                     gpointer          user_data)
Packit ae235b
{
Packit ae235b
  if (g_strcmp0 (property_name, "Title") == 0)
Packit ae235b
    {
Packit ae235b
      if (g_strcmp0 (_global_title, g_variant_get_string (value, NULL)) != 0)
Packit ae235b
        {
Packit ae235b
          GVariantBuilder *builder;
Packit ae235b
          GError *local_error;
Packit ae235b
Packit ae235b
          g_free (_global_title);
Packit ae235b
          _global_title = g_variant_dup_string (value, NULL);
Packit ae235b
Packit ae235b
          local_error = NULL;
Packit ae235b
          builder = g_variant_builder_new (G_VARIANT_TYPE_ARRAY);
Packit ae235b
          g_variant_builder_add (builder,
Packit ae235b
                                 "{sv}",
Packit ae235b
                                 "Title",
Packit ae235b
                                 g_variant_new_string (_global_title));
Packit ae235b
          g_dbus_connection_emit_signal (connection,
Packit ae235b
                                         NULL,
Packit ae235b
                                         object_path,
Packit ae235b
                                         "org.freedesktop.DBus.Properties",
Packit ae235b
                                         "PropertiesChanged",
Packit ae235b
                                         g_variant_new ("(sa{sv}as)",
Packit ae235b
                                                        interface_name,
Packit ae235b
                                                        builder,
Packit ae235b
                                                        NULL),
Packit ae235b
                                         &local_error);
Packit ae235b
          g_assert_no_error (local_error);
Packit ae235b
        }
Packit ae235b
    }
Packit ae235b
  else if (g_strcmp0 (property_name, "ReadingAlwaysThrowsError") == 0)
Packit ae235b
    {
Packit ae235b
      /* do nothing - they can't read it after all! */
Packit ae235b
    }
Packit ae235b
  else if (g_strcmp0 (property_name, "WritingAlwaysThrowsError") == 0)
Packit ae235b
    {
Packit ae235b
      g_set_error (error,
Packit ae235b
                   G_IO_ERROR,
Packit ae235b
                   G_IO_ERROR_FAILED,
Packit ae235b
                   "Hello AGAIN %s. I thought I said writing this property "
Packit ae235b
                   "always results in an error. kthxbye",
Packit ae235b
                   sender);
Packit ae235b
    }
Packit ae235b
Packit ae235b
  return *error == NULL;
Packit ae235b
}
Packit ae235b
Packit ae235b
Packit ae235b
/* for now */
Packit ae235b
static const GDBusInterfaceVTable interface_vtable =
Packit ae235b
{
Packit ae235b
  handle_method_call,
Packit ae235b
  handle_get_property,
Packit ae235b
  handle_set_property
Packit ae235b
};
Packit ae235b
Packit ae235b
/* ---------------------------------------------------------------------------------------------------- */
Packit ae235b
Packit ae235b
static gboolean
Packit ae235b
on_timeout_cb (gpointer user_data)
Packit ae235b
{
Packit ae235b
  GDBusConnection *connection = G_DBUS_CONNECTION (user_data);
Packit ae235b
  GVariantBuilder *builder;
Packit ae235b
  GVariantBuilder *invalidated_builder;
Packit ae235b
  GError *error;
Packit ae235b
Packit ae235b
  swap_a_and_b = !swap_a_and_b;
Packit ae235b
Packit ae235b
  error = NULL;
Packit ae235b
  builder = g_variant_builder_new (G_VARIANT_TYPE_ARRAY);
Packit ae235b
  invalidated_builder = g_variant_builder_new (G_VARIANT_TYPE ("as"));
Packit ae235b
  g_variant_builder_add (builder,
Packit ae235b
                         "{sv}",
Packit ae235b
                         "Foo",
Packit ae235b
                         g_variant_new_string (swap_a_and_b ? "Tock" : "Tick"));
Packit ae235b
  g_variant_builder_add (builder,
Packit ae235b
                         "{sv}",
Packit ae235b
                         "Bar",
Packit ae235b
                         g_variant_new_string (swap_a_and_b ? "Tick" : "Tock"));
Packit ae235b
  g_dbus_connection_emit_signal (connection,
Packit ae235b
                                 NULL,
Packit ae235b
                                 "/org/gtk/GDBus/TestObject",
Packit ae235b
                                 "org.freedesktop.DBus.Properties",
Packit ae235b
                                 "PropertiesChanged",
Packit ae235b
                                 g_variant_new ("(sa{sv}as)",
Packit ae235b
                                                "org.gtk.GDBus.TestInterface",
Packit ae235b
                                                builder,
Packit ae235b
                                                invalidated_builder),
Packit ae235b
                                 &error);
Packit ae235b
  g_assert_no_error (error);
Packit ae235b
Packit ae235b
Packit ae235b
  return TRUE;
Packit ae235b
}
Packit ae235b
Packit ae235b
/* ---------------------------------------------------------------------------------------------------- */
Packit ae235b
Packit ae235b
static void
Packit ae235b
on_bus_acquired (GDBusConnection *connection,
Packit ae235b
                 const gchar     *name,
Packit ae235b
                 gpointer         user_data)
Packit ae235b
{
Packit ae235b
  guint registration_id;
Packit ae235b
Packit ae235b
  registration_id = g_dbus_connection_register_object (connection,
Packit ae235b
                                                       "/org/gtk/GDBus/TestObject",
Packit ae235b
                                                       introspection_data->interfaces[0],
Packit ae235b
                                                       &interface_vtable,
Packit ae235b
                                                       NULL,  /* user_data */
Packit ae235b
                                                       NULL,  /* user_data_free_func */
Packit ae235b
                                                       NULL); /* GError** */
Packit ae235b
  g_assert (registration_id > 0);
Packit ae235b
Packit ae235b
  /* swap value of properties Foo and Bar every two seconds */
Packit ae235b
  g_timeout_add_seconds (2,
Packit ae235b
                         on_timeout_cb,
Packit ae235b
                         connection);
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
on_name_acquired (GDBusConnection *connection,
Packit ae235b
                  const gchar     *name,
Packit ae235b
                  gpointer         user_data)
Packit ae235b
{
Packit ae235b
}
Packit ae235b
Packit ae235b
static void
Packit ae235b
on_name_lost (GDBusConnection *connection,
Packit ae235b
              const gchar     *name,
Packit ae235b
              gpointer         user_data)
Packit ae235b
{
Packit ae235b
  exit (1);
Packit ae235b
}
Packit ae235b
Packit ae235b
int
Packit ae235b
main (int argc, char *argv[])
Packit ae235b
{
Packit ae235b
  guint owner_id;
Packit ae235b
  GMainLoop *loop;
Packit ae235b
Packit ae235b
  /* We are lazy here - we don't want to manually provide
Packit ae235b
   * the introspection data structures - so we just build
Packit ae235b
   * them from XML.
Packit ae235b
   */
Packit ae235b
  introspection_data = g_dbus_node_info_new_for_xml (introspection_xml, NULL);
Packit ae235b
  g_assert (introspection_data != NULL);
Packit ae235b
Packit ae235b
  owner_id = g_bus_own_name (G_BUS_TYPE_SESSION,
Packit ae235b
                             "org.gtk.GDBus.TestServer",
Packit ae235b
                             G_BUS_NAME_OWNER_FLAGS_NONE,
Packit ae235b
                             on_bus_acquired,
Packit ae235b
                             on_name_acquired,
Packit ae235b
                             on_name_lost,
Packit ae235b
                             NULL,
Packit ae235b
                             NULL);
Packit ae235b
Packit ae235b
  loop = g_main_loop_new (NULL, FALSE);
Packit ae235b
  g_main_loop_run (loop);
Packit ae235b
Packit ae235b
  g_bus_unown_name (owner_id);
Packit ae235b
Packit ae235b
  g_dbus_node_info_unref (introspection_data);
Packit ae235b
Packit ae235b
  return 0;
Packit ae235b
}