From 082547143c33b228555ef0216dfc4754ce4b8a53 Mon Sep 17 00:00:00 2001 From: joprietoe Date: Thu, 7 Apr 2016 10:33:48 -0300 Subject: [PATCH] v01 --- .vscode/tasks.json | 29 ++ GPATH | Bin 0 -> 16384 bytes GRTAGS | Bin 0 -> 57344 bytes GTAGS | Bin 0 -> 16384 bytes Makefile | 39 ++ client.c | 146 ++++++ client.h | 48 ++ gdbus-example-client.c | 132 +++++ gdbus-example-server.c | 391 +++++++++++++++ gdbus-example-unix-fd-client.c | 132 +++++ gdbus-message.c | 156 ++++++ gdbus-tests.c | 207 ++++++++ gdbus-tests.h | 122 +++++ gdbus-testserver.c | 893 +++++++++++++++++++++++++++++++++ server.c | 145 ++++++ server.h | 48 ++ servicename-infos.xml | 12 + 17 files changed, 2500 insertions(+) create mode 100644 .vscode/tasks.json create mode 100644 GPATH create mode 100644 GRTAGS create mode 100644 GTAGS create mode 100755 Makefile create mode 100755 client.c create mode 100755 client.h create mode 100644 gdbus-example-client.c create mode 100755 gdbus-example-server.c create mode 100755 gdbus-example-unix-fd-client.c create mode 100755 gdbus-message.c create mode 100755 gdbus-tests.c create mode 100755 gdbus-tests.h create mode 100755 gdbus-testserver.c create mode 100755 server.c create mode 100755 server.h create mode 100755 servicename-infos.xml diff --git a/.vscode/tasks.json b/.vscode/tasks.json new file mode 100644 index 0000000..cc7ff0b --- /dev/null +++ b/.vscode/tasks.json @@ -0,0 +1,29 @@ +{ + "version": "0.1.0", + "command": "make", + "isShellCommand": true, + "tasks": [ + { + "taskName": "all", + // Make this the default build command. + "isBuildCommand": true, + // Show the output window only if unrecognized errors occur. + "showOutput": "always", + // No args + "args": [], + // Use the standard less compilation problem matcher. + "problemMatcher": { + "owner": "cpp", + "fileLocation": ["relative", "${workspaceRoot}"], + "pattern": { + "regexp": "^(.*):(\\d+):(\\d+):\\s+(warning|error):\\s+(.*)$", + "file": 1, + "line": 2, + "column": 3, + "severity": 4, + "message": 5 + } + } + } + ] +} \ No newline at end of file diff --git a/GPATH b/GPATH new file mode 100644 index 0000000000000000000000000000000000000000..16aad2786d6b5cc8e9568d08d38c78aaa3f7ddfd GIT binary patch literal 16384 zcmeI(O-{l<6bJB8apgyhiHVwaiU(-5h#w>_WnuhE)EG@n6mgJ7Kts{6^Z*{gCC6~# z!XvnF;nJl`58!*#GPI%xF!_@JllT5JOn*#QgmZsb3<-K<`6YT}y92yK00Izz00bZa z0SG_<0uX=z1Rwwb2tWV=5P$##AOHafKmY;|fB*y_@Gk@g61yCv@s{XV9qZl6IF{_l zd&jcRg^Pd4mOL&!q4m41kdmn zg*ShBaef=REfuyxuPV5%=lpC?mg_TmUe3%@?Na@wrrc09?^M$(bDJ}%HTpt+j&qzD z;eqeddhrU?s20yRMT+SBtj;gQ`Jk*CrMhy1M&&y6+Vz^arv5ZNhdCE5+8^>4bdGy}#jeb4BWSyo3)kd1 zm-_5KlPkS_Ecf<7jM_H6kE&RX*VX%8qI}|uOYGZJm*V^wy^B2Mw@X*564Vq4az4ZL zL^V%!{>(|P_l{Rxg<GXaBv|Px5 literal 0 HcmV?d00001 diff --git a/GRTAGS b/GRTAGS new file mode 100644 index 0000000000000000000000000000000000000000..debcce534bdf45f66e9697bb1f438b45fe62935a GIT binary patch literal 57344 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 literal 0 HcmV?d00001 diff --git a/GTAGS b/GTAGS new file mode 100644 index 0000000000000000000000000000000000000000..2d8218380c04eb5476120557a6c7701aa28fe416 GIT binary patch literal 16384 zcmeHLy>k;+5WghDBxLZn@fXH^BqWg(k+HrkF_~oK2#xIkw#QO3Fd_4DmJa!Xe5c%< z9X#L}3d$7x0Vye%3HFRN$=;>9^tjH|^1}XY=>wZ_l5ee>it;j?HOv&raSv zId$^)+3IX;_Q{D$C!Wtdp1C`7YoQl2*&nE9r)+b*~JejyX zQJff_czS&0_?_bq$6t=09&e7{8GAZ*Y4nfL`=h^%#>TXaABl5 z{9yR`P;B^zp}Rw|p}z+24Q>yz!9NFn9f%FQIQHmRh^V?%3Md7X0!jg;fKosypcGIF zC;h#>k)+vk?;nOFCtJhItCX&{HW?ihkY zMo6)|zV)L~^NQJoO*cvK843ByuuZ#m&8TY`QZY|aU-d-4gp6tFI|kp=Ez{iwKk`T| zkb7+T+*iA9$+j$rYuXk|)PnmY#qN$lE7%vtKqrr;?Rwf-WL@yn?!^K9sTQGzC=c=% zC6vkakp%do7F1L|hTdu!y5nn+69v>$iXIrGMc)_^J@wj8yC(7wdZhaxg1JQW2r#53 z*+o$iy+j2J+=qNFTe~g~U<8$vXo}vNAcoH+f+4A-MN9Mq!S%)cM%yu5S8o^`t6S7| zMr0!P!B^~Rc57eQofTY#-4THLi12jNlDcL^nc$ABJB?bBi5(H{_rCic3Ilxv2lSyu zNTnB<+`;4Yyp1XPCna+06O;NomCi&lrvsiuFqIYC0Ae!1RB&s@7Jb)G?2iL%xfqK= zrNao1gFUU~wrV()X&5cJ#0v|wp9ROeoDgsyq#3N%Gz|+=N56fmCi=IK?;KG|WCwZECmjdK z^A!xe4U+EzB$($Uk6@mo8i$VWnCCF&23(r7r+LmXTFrgA@yL?KN?hJ>?Cbk10X24g zKiIV__h-Xt&uuie!B7_-qohU{i~2{#3e9oE4ys;GtNt8UB5^wBxTAoB>#Fev%wIfM zK~bVX7Yj;k-~+uV}_S3#B%WQenF@kT0rKt%#yXzRdPXC9w2oH*h@uyAk%u#gwcacw-m z{r!~V>uiHC#qKPEoIF(7l z!z6=e@uPK{-C*Wb7CQyMv70wp3_jy$z7&q^)E8vNAbEV}!FAO*ua+y-3SZ$|0>ev- z0LI5?7sH$C<&840lsCUEZ)&2ZVH^;4Bxp?T2Y$oE;?$GqPd!Qgr3ygK!?x$rRWDY$u-!p2C)1(5U1kPY_BNy&K?>jeML<`xw|4 zq=`tAuSg3x-vU8!pQ^&8Nt@+zm6t9oRVuzHXZ=-y(u)@hv1Lw~udY?9OKYVv4;)x5 zh9*fCQZ~sdZTci2&PkU;@glMEek}2d2J5hJ;0G+NluPFYHC|e(R3V(O>Kza@T5rem zv AC@XBBXq+HD87h=n(eksxgm6vqJyP3{FJEi)?$ut?Ot2iZ7OQBxT9K6(mD*nx znOHF&KkemqPJGZ;6F@lh!&cAZoPMjwiEv!`2d6Q9{8wG0s1OeY&!` zzNUrl*`lUl&UEJC@1^yN8=Ga=yTV^sy>47JErWUZ|5$KOKIj*itu0+FN55$Q0G?!8 AF8}}l literal 0 HcmV?d00001 diff --git a/Makefile b/Makefile new file mode 100755 index 0000000..0b663a8 --- /dev/null +++ b/Makefile @@ -0,0 +1,39 @@ + +CFLAGS_DBUS = $(shell pkg-config --cflags --libs dbus-1) +CFLAGS_DBUS_GLIB = $(shell pkg-config --cflags --libs dbus-glib-1) +CFLAGS_GIO = $(shell pkg-config --cflags --libs gio-2.0) +CFLAGS_GUNIX = $(shell pkg-config --cflags --libs gio-unix-2.0) + +CFLAGS = -g -Wall -Werror + + +all: gdbus-example-server gdbus-example-client + +#dbus-server: dbus-server.c +# gcc $< -o $@ $(CFLAGS) $(CFLAGS_DBUS) $(CFLAGS_DBUS_GLIB) + +#dbus-client: dbus-client.c +# gcc $< -o $@ $(CFLAGS) $(CFLAGS_GIO) + +gdbus-example-server: gdbus-example-server.c + gcc $< -o $@ $(CFLAGS) $(CFLAGS_DBUS) $(CFLAGS_GIO) $(CFLAGS_GUNIX) + +gdbus-example-client: gdbus-example-client.c + gcc $< -o $@ $(CFLAGS) $(CFLAGS_GIO) + +#gdbus-testserver: gdbus-testserver.c +# gcc $< -o $@ $(CFLAGS) $(CFLAGS_GIO) + +#gdbus-example-unix-fd-client: gdbus-example-unix-fd-client.c +# gcc $< -o $@ $(CFLAGS) $(CFLAGS_DBUS) $(CFLAGS_GIO) $(CFLAGS_GUNIX) + +clean: + rm -f dbus-server + rm -f dbus-client + rm -f gdbus-example-server + rm -f gdbus-testserver + rm -f gdbus-example-unix-fd-client + rm -f gdbus-example-client + + +.PHONY: all clean diff --git a/client.c b/client.c new file mode 100755 index 0000000..49fcace --- /dev/null +++ b/client.c @@ -0,0 +1,146 @@ +/* + * Source Code from the DBus Activation Tutorial + * from Raphael Slinckx + * + * This code illustrates how to requrest the dbus daemon + * to automatically start a program that provides a given + * service. For more detailed information refer also to + * http://raphael.slinckx.net/blog/documents/dbus-tutorial + * where all source has taken from. + * + * Provision of all glue code to form compilable application + * by Otto Linnemann + * + * Implementations for Client + */ + +#include "client.h" +#include "client-bindings.h" + +static void client_class_init(ClientClass *klass); +static void client_class_release(ClientClass *klass); + +GType client_get_type(void) +{ + static GType client_type = 0; + + if(!client_type) + { + static const GTypeInfo client_info = { + sizeof(ClientClass), /* class structure size */ + NULL, /* base class initializer */ + NULL, /* base class finalizer */ + (GClassInitFunc)client_class_init, /* class initializer */ + NULL, /* class finalizer */ + NULL, /* class data */ + sizeof(Client), /* instance structure size */ + 1, /* preallocated instances */ + NULL, /* instance initializers */ + NULL + }; + + client_type = g_type_register_static( + G_TYPE_OBJECT, /* parent class */ + "Client", + &client_info, + 0); + } + + return client_type; +} + +static void +client_echo_reply (DBusGProxy *proxy, char *answer, GError *error, gpointer userdata); + +static void client_class_init(ClientClass *klass) +{ + GError *error = NULL; + + /* Init the DBus connection, per-klass */ + klass->connection = dbus_g_bus_get (DBUS_BUS_SESSION, &error); + if (klass->connection == NULL) + { + g_warning("Unable to connect to dbus: %s", error->message); + g_error_free (error); + return; + } + +/* This won't trigger activation! */ + klass->proxy = dbus_g_proxy_new_for_name (klass->connection, + "org.gnome.ServiceName", + "/org/gnome/ServiceName", + "org.gnome.ServiceName"); + + g_message("ClientClass successfully initialized"); +} + +static void client_send_async(Client *client) +{ + ClientClass *klass = CLIENT_GET_CLASS( client ); + +/* The method call will trigger activation and prints results asynchronously */ + org_gnome_ServiceName_echo_string_async(klass->proxy, + "The async request string we want echo-ed", + client_echo_reply, client); +} + +static void client_send_sync(Client *client) +{ + GError *error = NULL; + ClientClass *klass = CLIENT_GET_CLASS( client ); + gchar *result; + +/* The method call will trigger activation and blocks until result arives */ + if (!org_gnome_ServiceName_echo_string (klass->proxy, + "The sync request string we want echo-ed", + &result, &error)) + { + /* Method failed, the GError is set, let's warn everyone */ + g_warning ("Woops remote method failed: %s", error->message); + g_error_free (error); + return; + } + + g_print ("We got the folowing result from synchronous request: %s\n", result); + +/* Cleanup */ + g_free (result); +} + +static void +client_echo_reply (DBusGProxy *proxy, char *answer, GError *error, gpointer userdata) +{ + Client *client = CLIENT(userdata); + + if (error!= NULL) + { + g_warning ("An error occured while calling echo_string remote method: %s", error->message); + g_error_free (error); + return; + } + + g_print ("We got an echo reply, result: %s\n", answer); +} + + +int main() +{ + Client *client; + GMainLoop *loop; + + g_type_init(); /* initialize type system */ + + if (!g_thread_supported ()) + g_thread_init (NULL); + + dbus_g_thread_init (); + + loop = g_main_loop_new (NULL, FALSE); + client = g_object_new( TYPE_CLIENT, NULL ); + + client_send_async( client ); + client_send_sync( client ); + + g_main_loop_run (loop); + return 0; +} diff --git a/client.h b/client.h new file mode 100755 index 0000000..6a6a7eb --- /dev/null +++ b/client.h @@ -0,0 +1,48 @@ +/* + * Source Code from the DBus Activation Tutorial + * from Raphael Slinckx + * + * This code illustrates how to requrest the dbus daemon + * to automatically start a program that provides a given + * service. For more detailed information refer also to + * http://raphael.slinckx.net/blog/documents/dbus-tutorial + * where all source has taken from. + * + * Provision of all glue code to form compilable application + * by Otto Linnemann + * + * Declarations for Client + */ + +#ifndef CLIENT_H +#define CLIENT_H + +#include + +/* Standard GObject class structures, etc */ + +#define TYPE_CLIENT (client_get_type()) +#define CLIENT(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), TYPE_CLIENT, Client)) +#define IS_CLIENT(obj) (G_TYPE_CHECK_INSTANCE_TYPE ((obj), TYPE_CLIENT)) +#define CLIENT_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), TYPE_CLIENT, ClientClass)) +#define IS_CLIENT_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), TYPE_CLIENT)) +#define CLIENT_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), TYPE_CLIENT, ClientClass)) + +typedef struct +{ + GObjectClass parent_class; + + DBusGConnection *connection; + DBusGProxy *proxy; +} ClientClass; + +typedef struct +{ + GObject parnet_instance; + +} Client; + + +GType client_get_type(void); + +#endif /* #ifndef CLIENT_H */ diff --git a/gdbus-example-client.c b/gdbus-example-client.c new file mode 100644 index 0000000..9c510ad --- /dev/null +++ b/gdbus-example-client.c @@ -0,0 +1,132 @@ +#include +#include + +#include +#include + +#include + +#include +#include + +/* see gdbus-example-server.c for the server implementation */ +static gint +get_server_stdout (GDBusConnection *connection, + const gchar *name_owner, + GError **error) +{ + GDBusMessage *method_call_message; + GDBusMessage *method_reply_message; + GUnixFDList *fd_list; + gint fd; + + fd = -1; + method_call_message = NULL; + method_reply_message = NULL; + + method_call_message = g_dbus_message_new_method_call (name_owner, + "/org/gtk/GDBus/TestObject", + "org.gtk.GDBus.TestInterface", + "GimmeStdout"); + method_reply_message = g_dbus_connection_send_message_with_reply_sync (connection, + method_call_message, + G_DBUS_SEND_MESSAGE_FLAGS_NONE, + -1, + NULL, /* out_serial */ + NULL, /* cancellable */ + error); + if (method_reply_message == NULL) + goto out; + + if (g_dbus_message_get_message_type (method_reply_message) == G_DBUS_MESSAGE_TYPE_ERROR) + { + g_dbus_message_to_gerror (method_reply_message, error); + goto out; + } + + fd_list = g_dbus_message_get_unix_fd_list (method_reply_message); + fd = g_unix_fd_list_get (fd_list, 0, error); + + out: + g_object_unref (method_call_message); + g_object_unref (method_reply_message); + + return fd; +} + +static void +on_name_appeared (GDBusConnection *connection, + const gchar *name, + const gchar *name_owner, + gpointer user_data) +{ + gint fd; + GError *error; + + error = NULL; + fd = get_server_stdout (connection, name_owner, &error); + if (fd == -1) + { + g_printerr ("Error invoking GimmeStdout(): %s\n", + error->message); + g_error_free (error); + exit (1); + } + else + { + gchar now_buf[256]; + time_t now; + gssize len; + gchar *str; + + now = time (NULL); + strftime (now_buf, + sizeof now_buf, + "%c", + localtime (&now)); + + str = g_strdup_printf ("On %s, gdbus-example-unix-fd-client with pid %d was here!\n", + now_buf, + (gint) getpid ()); + len = strlen (str); + g_warn_if_fail (write (fd, str, len) == len); + close (fd); + + g_print ("Wrote the following on server's stdout:\n%s", str); + + g_free (str); + exit (0); + } +} + +static void +on_name_vanished (GDBusConnection *connection, + const gchar *name, + gpointer user_data) +{ + g_printerr ("Failed to get name owner for %s\n" + "Is ./gdbus-example-server running?\n", + name); + exit (1); +} + +int +main (int argc, char *argv[]) +{ + guint watcher_id; + GMainLoop *loop; + + watcher_id = g_bus_watch_name (G_BUS_TYPE_SESSION, + "org.gtk.GDBus.TestServer", + G_BUS_NAME_WATCHER_FLAGS_NONE, + on_name_appeared, + on_name_vanished, + NULL, + NULL); + + loop = g_main_loop_new (NULL, FALSE); + g_main_loop_run (loop); + + g_bus_unwatch_name (watcher_id); + return 0; +} diff --git a/gdbus-example-server.c b/gdbus-example-server.c new file mode 100755 index 0000000..edcfaca --- /dev/null +++ b/gdbus-example-server.c @@ -0,0 +1,391 @@ +#include +#include + +#ifdef G_OS_UNIX +#include +/* For STDOUT_FILENO */ +#include +#endif + +/* ---------------------------------------------------------------------------------------------------- */ + +static GDBusNodeInfo *introspection_data = NULL; + +/* Introspection data for the service we are exporting */ +static const gchar introspection_xml[] = + "" + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + ""; + +/* ---------------------------------------------------------------------------------------------------- */ + +static void +handle_method_call (GDBusConnection *connection, + const gchar *sender, + const gchar *object_path, + const gchar *interface_name, + const gchar *method_name, + GVariant *parameters, + GDBusMethodInvocation *invocation, + gpointer user_data) +{ + if (g_strcmp0 (method_name, "HelloWorld") == 0) + { + const gchar *greeting; + + g_variant_get (parameters, "(&s)", &greeting); + + if (g_strcmp0 (greeting, "Return Unregistered") == 0) + { + g_dbus_method_invocation_return_error (invocation, + G_IO_ERROR, + G_IO_ERROR_FAILED_HANDLED, + "As requested, here's a GError not registered (G_IO_ERROR_FAILED_HANDLED)"); + } + else if (g_strcmp0 (greeting, "Return Registered") == 0) + { + g_dbus_method_invocation_return_error (invocation, + G_DBUS_ERROR, + G_DBUS_ERROR_MATCH_RULE_NOT_FOUND, + "As requested, here's a GError that is registered (G_DBUS_ERROR_MATCH_RULE_NOT_FOUND)"); + } + else if (g_strcmp0 (greeting, "Return Raw") == 0) + { + g_dbus_method_invocation_return_dbus_error (invocation, + "org.gtk.GDBus.SomeErrorName", + "As requested, here's a raw D-Bus error"); + } + else + { + gchar *response; + response = g_strdup_printf ("You greeted me with '%s'. Thanks!", greeting); + g_dbus_method_invocation_return_value (invocation, + g_variant_new ("(s)", response)); + g_free (response); + } + } + else if (g_strcmp0 (method_name, "EmitSignal") == 0) + { + GError *local_error; + gdouble speed_in_mph; + gchar *speed_as_string; + + g_variant_get (parameters, "(d)", &speed_in_mph); + speed_as_string = g_strdup_printf ("%g mph!", speed_in_mph); + + local_error = NULL; + g_dbus_connection_emit_signal (connection, + NULL, + object_path, + interface_name, + "VelocityChanged", + g_variant_new ("(ds)", + speed_in_mph, + speed_as_string), + &local_error); + g_assert_no_error (local_error); + g_free (speed_as_string); + + g_dbus_method_invocation_return_value (invocation, NULL); + } + else if (g_strcmp0 (method_name, "GimmeStdout") == 0) + { +#ifdef G_OS_UNIX + if (g_dbus_connection_get_capabilities (connection) & G_DBUS_CAPABILITY_FLAGS_UNIX_FD_PASSING) + { + GDBusMessage *reply; + GUnixFDList *fd_list; + GError *error; + + fd_list = g_unix_fd_list_new (); + error = NULL; + g_unix_fd_list_append (fd_list, STDOUT_FILENO, &error); + g_assert_no_error (error); + + reply = g_dbus_message_new_method_reply (g_dbus_method_invocation_get_message (invocation)); + g_dbus_message_set_unix_fd_list (reply, fd_list); + + error = NULL; + g_dbus_connection_send_message (connection, + reply, + G_DBUS_SEND_MESSAGE_FLAGS_NONE, + NULL, /* out_serial */ + &error); + g_assert_no_error (error); + + g_object_unref (invocation); + g_object_unref (fd_list); + g_object_unref (reply); + } + else + { + g_dbus_method_invocation_return_dbus_error (invocation, + "org.gtk.GDBus.Failed", + "Your message bus daemon does not support file descriptor passing (need D-Bus >= 1.3.0)"); + } +#else + g_dbus_method_invocation_return_dbus_error (invocation, + "org.gtk.GDBus.NotOnUnix", + "Your OS does not support file descriptor passing"); +#endif + } +} + +static gchar *_global_title = NULL; + +static gboolean swap_a_and_b = FALSE; + +static GVariant * +handle_get_property (GDBusConnection *connection, + const gchar *sender, + const gchar *object_path, + const gchar *interface_name, + const gchar *property_name, + GError **error, + gpointer user_data) +{ + GVariant *ret; + + ret = NULL; + if (g_strcmp0 (property_name, "FluxCapicitorName") == 0) + { + ret = g_variant_new_string ("DeLorean"); + } + else if (g_strcmp0 (property_name, "Title") == 0) + { + if (_global_title == NULL) + _global_title = g_strdup ("Back To C!"); + ret = g_variant_new_string (_global_title); + } + else if (g_strcmp0 (property_name, "ReadingAlwaysThrowsError") == 0) + { + g_set_error (error, + G_IO_ERROR, + G_IO_ERROR_FAILED, + "Hello %s. I thought I said reading this property " + "always results in an error. kthxbye", + sender); + } + else if (g_strcmp0 (property_name, "WritingAlwaysThrowsError") == 0) + { + ret = g_variant_new_string ("There's no home like home"); + } + else if (g_strcmp0 (property_name, "Foo") == 0) + { + ret = g_variant_new_string (swap_a_and_b ? "Tock" : "Tick"); + } + else if (g_strcmp0 (property_name, "Bar") == 0) + { + ret = g_variant_new_string (swap_a_and_b ? "Tick" : "Tock"); + } + + return ret; +} + +static gboolean +handle_set_property (GDBusConnection *connection, + const gchar *sender, + const gchar *object_path, + const gchar *interface_name, + const gchar *property_name, + GVariant *value, + GError **error, + gpointer user_data) +{ + if (g_strcmp0 (property_name, "Title") == 0) + { + if (g_strcmp0 (_global_title, g_variant_get_string (value, NULL)) != 0) + { + GVariantBuilder *builder; + GError *local_error; + + g_free (_global_title); + _global_title = g_variant_dup_string (value, NULL); + + local_error = NULL; + builder = g_variant_builder_new (G_VARIANT_TYPE_ARRAY); + g_variant_builder_add (builder, + "{sv}", + "Title", + g_variant_new_string (_global_title)); + g_dbus_connection_emit_signal (connection, + NULL, + object_path, + "org.freedesktop.DBus.Properties", + "PropertiesChanged", + g_variant_new ("(sa{sv}as)", + interface_name, + builder, + NULL), + &local_error); + g_assert_no_error (local_error); + } + } + else if (g_strcmp0 (property_name, "ReadingAlwaysThrowsError") == 0) + { + /* do nothing - they can't read it after all! */ + } + else if (g_strcmp0 (property_name, "WritingAlwaysThrowsError") == 0) + { + g_set_error (error, + G_IO_ERROR, + G_IO_ERROR_FAILED, + "Hello AGAIN %s. I thought I said writing this property " + "always results in an error. kthxbye", + sender); + } + + return *error == NULL; +} + + +/* for now */ +static const GDBusInterfaceVTable interface_vtable = +{ + handle_method_call, + handle_get_property, + handle_set_property +}; + +/* ---------------------------------------------------------------------------------------------------- */ + +static gboolean +on_timeout_cb (gpointer user_data) +{ + GDBusConnection *connection = G_DBUS_CONNECTION (user_data); + GVariantBuilder *builder; + GVariantBuilder *invalidated_builder; + GError *error; + + swap_a_and_b = !swap_a_and_b; + + error = NULL; + builder = g_variant_builder_new (G_VARIANT_TYPE_ARRAY); + invalidated_builder = g_variant_builder_new (G_VARIANT_TYPE ("as")); + g_variant_builder_add (builder, + "{sv}", + "Foo", + g_variant_new_string (swap_a_and_b ? "Tock" : "Tick")); + g_variant_builder_add (builder, + "{sv}", + "Bar", + g_variant_new_string (swap_a_and_b ? "Tick" : "Tock")); + g_dbus_connection_emit_signal (connection, + NULL, + "/org/gtk/GDBus/TestObject", + "org.freedesktop.DBus.Properties", + "PropertiesChanged", + g_variant_new ("(sa{sv}as)", + "org.gtk.GDBus.TestInterface", + builder, + invalidated_builder), + &error); + g_assert_no_error (error); + + + return TRUE; +} + +/* ---------------------------------------------------------------------------------------------------- */ + +static void +on_bus_acquired (GDBusConnection *connection, + const gchar *name, + gpointer user_data) +{ + guint registration_id; + + registration_id = g_dbus_connection_register_object (connection, + "/org/gtk/GDBus/TestObject", + introspection_data->interfaces[0], + &interface_vtable, + NULL, /* user_data */ + NULL, /* user_data_free_func */ + NULL); /* GError** */ + g_assert (registration_id > 0); + + /* swap value of properties Foo and Bar every two seconds */ + g_timeout_add_seconds (2, + on_timeout_cb, + connection); +} + +static void +on_name_acquired (GDBusConnection *connection, + const gchar *name, + gpointer user_data) +{ +} + +static void +on_name_lost (GDBusConnection *connection, + const gchar *name, + gpointer user_data) +{ + exit (1); +} + +int +main (int argc, char *argv[]) +{ + guint owner_id; + GMainLoop *loop; + + /* We are lazy here - we don't want to manually provide + * the introspection data structures - so we just build + * them from XML. + */ + introspection_data = g_dbus_node_info_new_for_xml (introspection_xml, NULL); + g_assert (introspection_data != NULL); + + owner_id = g_bus_own_name (G_BUS_TYPE_SESSION, + "org.gtk.GDBus.TestServer", + G_BUS_NAME_OWNER_FLAGS_NONE, + on_bus_acquired, + on_name_acquired, + on_name_lost, + NULL, + NULL); + + loop = g_main_loop_new (NULL, FALSE); + g_main_loop_run (loop); + + g_bus_unown_name (owner_id); + + g_dbus_node_info_unref (introspection_data); + + return 0; +} + diff --git a/gdbus-example-unix-fd-client.c b/gdbus-example-unix-fd-client.c new file mode 100755 index 0000000..e3ab015 --- /dev/null +++ b/gdbus-example-unix-fd-client.c @@ -0,0 +1,132 @@ +#include +#include + +#include +#include + +#include + +#include +#include + +/* see gdbus-example-server.c for the server implementation */ +static gint +get_server_stdout (GDBusConnection *connection, + const gchar *name_owner, + GError **error) +{ + GDBusMessage *method_call_message; + GDBusMessage *method_reply_message; + GUnixFDList *fd_list; + gint fd; + + fd = -1; + method_call_message = NULL; + method_reply_message = NULL; + + method_call_message = g_dbus_message_new_method_call (name_owner, + "/org/gtk/GDBus/TestObject", + "org.gtk.GDBus.TestInterface", + "GimmeStdout"); + method_reply_message = g_dbus_connection_send_message_with_reply_sync (connection, + method_call_message, + G_DBUS_SEND_MESSAGE_FLAGS_NONE, + -1, + NULL, /* out_serial */ + NULL, /* cancellable */ + error); + if (method_reply_message == NULL) + goto out; + + if (g_dbus_message_get_message_type (method_reply_message) == G_DBUS_MESSAGE_TYPE_ERROR) + { + g_dbus_message_to_gerror (method_reply_message, error); + goto out; + } + + fd_list = g_dbus_message_get_unix_fd_list (method_reply_message); + fd = g_unix_fd_list_get (fd_list, 0, error); + + out: + g_object_unref (method_call_message); + g_object_unref (method_reply_message); + + return fd; +} + +static void +on_name_appeared (GDBusConnection *connection, + const gchar *name, + const gchar *name_owner, + gpointer user_data) +{ + gint fd; + GError *error; + + error = NULL; + fd = get_server_stdout (connection, name_owner, &error); + if (fd == -1) + { + g_printerr ("Error invoking GimmeStdout(): %s\n", + error->message); + g_error_free (error); + exit (1); + } + else + { + gchar now_buf[256]; + time_t now; + gssize len; + gchar *str; + + now = time (NULL); + strftime (now_buf, + sizeof now_buf, + "%c", + localtime (&now)); + + str = g_strdup_printf ("On %s, gdbus-example-unix-fd-client with pid %d was here!\n", + now_buf, + (gint) getpid ()); + len = strlen (str); + g_warn_if_fail (write (fd, str, len) == len); + close (fd); + + g_print ("Wrote the following on server's stdout:\n%s", str); + + g_free (str); + exit (0); + } +} + +static void +on_name_vanished (GDBusConnection *connection, + const gchar *name, + gpointer user_data) +{ + g_printerr ("Failed to get name owner for %s\n" + "Is ./gdbus-example-server running?\n", + name); + exit (1); +} + +int +main (int argc, char *argv[]) +{ + guint watcher_id; + GMainLoop *loop; + + watcher_id = g_bus_watch_name (G_BUS_TYPE_SESSION, + "org.gtk.GDBus.TestServer", + G_BUS_NAME_WATCHER_FLAGS_NONE, + on_name_appeared, + on_name_vanished, + NULL, + NULL); + + loop = g_main_loop_new (NULL, FALSE); + g_main_loop_run (loop); + + g_bus_unwatch_name (watcher_id); + return 0; +} diff --git a/gdbus-message.c b/gdbus-message.c new file mode 100755 index 0000000..7a88bbf --- /dev/null +++ b/gdbus-message.c @@ -0,0 +1,156 @@ +/* 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, see . + * + * Author: David Zeuthen + */ + +#include +#include + +/* ---------------------------------------------------------------------------------------------------- */ + +static void +on_notify_locked (GObject *object, + GParamSpec *pspec, + gpointer user_data) +{ + gint *count = user_data; + *count += 1; +} + +static void +message_lock (void) +{ + GDBusMessage *m; + gint count; + + count = 0; + m = g_dbus_message_new (); + g_signal_connect (m, + "notify::locked", + G_CALLBACK (on_notify_locked), + &count); + g_assert (!g_dbus_message_get_locked (m)); + g_dbus_message_lock (m); + g_assert (g_dbus_message_get_locked (m)); + g_assert_cmpint (count, ==, 1); + g_dbus_message_lock (m); + g_assert (g_dbus_message_get_locked (m)); + g_assert_cmpint (count, ==, 1); + + g_test_expect_message (G_LOG_DOMAIN, G_LOG_LEVEL_WARNING, + "*Attempted to modify a locked message*"); + g_dbus_message_set_serial (m, 42); + g_test_assert_expected_messages (); + + g_test_expect_message (G_LOG_DOMAIN, G_LOG_LEVEL_WARNING, + "*Attempted to modify a locked message*"); + g_dbus_message_set_byte_order (m, G_DBUS_MESSAGE_BYTE_ORDER_BIG_ENDIAN); + g_test_assert_expected_messages (); + + g_test_expect_message (G_LOG_DOMAIN, G_LOG_LEVEL_WARNING, + "*Attempted to modify a locked message*"); + g_dbus_message_set_message_type (m, G_DBUS_MESSAGE_TYPE_METHOD_CALL); + g_test_assert_expected_messages (); + + g_test_expect_message (G_LOG_DOMAIN, G_LOG_LEVEL_WARNING, + "*Attempted to modify a locked message*"); + g_dbus_message_set_flags (m, G_DBUS_MESSAGE_FLAGS_NONE); + g_test_assert_expected_messages (); + + g_test_expect_message (G_LOG_DOMAIN, G_LOG_LEVEL_WARNING, + "*Attempted to modify a locked message*"); + g_dbus_message_set_body (m, NULL); + g_test_assert_expected_messages (); + + g_test_expect_message (G_LOG_DOMAIN, G_LOG_LEVEL_WARNING, + "*Attempted to modify a locked message*"); + g_dbus_message_set_header (m, 0, NULL); + g_test_assert_expected_messages (); + + g_object_unref (m); +} + +/* ---------------------------------------------------------------------------------------------------- */ + +static void +message_copy (void) +{ + GDBusMessage *m; + GDBusMessage *copy; + GError *error; + guchar *m_headers; + guchar *copy_headers; + guint n; + + m = g_dbus_message_new_method_call ("org.example.Name", + "/org/example/Object", + "org.example.Interface", + "Method"); + g_dbus_message_set_serial (m, 42); + g_dbus_message_set_byte_order (m, G_DBUS_MESSAGE_BYTE_ORDER_BIG_ENDIAN); + + error = NULL; + copy = g_dbus_message_copy (m, &error); + g_assert_no_error (error); + g_assert (G_IS_DBUS_MESSAGE (copy)); + g_assert (m != copy); + g_assert_cmpint (G_OBJECT (m)->ref_count, ==, 1); + g_assert_cmpint (G_OBJECT (copy)->ref_count, ==, 1); + + g_assert_cmpint (g_dbus_message_get_serial (copy), ==, g_dbus_message_get_serial (m)); + g_assert_cmpint (g_dbus_message_get_byte_order (copy), ==, g_dbus_message_get_byte_order (m)); + g_assert_cmpint (g_dbus_message_get_flags (copy), ==, g_dbus_message_get_flags (m)); + g_assert_cmpint (g_dbus_message_get_message_type (copy), ==, g_dbus_message_get_message_type (m)); + m_headers = g_dbus_message_get_header_fields (m); + copy_headers = g_dbus_message_get_header_fields (copy); + g_assert (m_headers != NULL); + g_assert (copy_headers != NULL); + for (n = 0; m_headers[n] != 0; n++) + { + GVariant *m_val; + GVariant *copy_val; + m_val = g_dbus_message_get_header (m, m_headers[n]); + copy_val = g_dbus_message_get_header (m, m_headers[n]); + g_assert (m_val != NULL); + g_assert (copy_val != NULL); + g_assert (g_variant_equal (m_val, copy_val)); + } + g_assert_cmpint (n, >, 0); /* make sure we actually compared headers etc. */ + g_assert_cmpint (copy_headers[n], ==, 0); + g_free (m_headers); + g_free (copy_headers); + + g_object_unref (copy); + g_object_unref (m); +} + +/* ---------------------------------------------------------------------------------------------------- */ + +int +main (int argc, + char *argv[]) +{ + setlocale (LC_ALL, "C"); + + g_test_init (&argc, &argv, NULL); + + g_test_add_func ("/gdbus/message/lock", message_lock); + g_test_add_func ("/gdbus/message/copy", message_copy); + return g_test_run(); +} + diff --git a/gdbus-tests.c b/gdbus-tests.c new file mode 100755 index 0000000..2770354 --- /dev/null +++ b/gdbus-tests.c @@ -0,0 +1,207 @@ +/* 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, see . + * + * Author: David Zeuthen + */ + +#include +#include + +#include "gdbus-tests.h" + +/* ---------------------------------------------------------------------------------------------------- */ + +typedef struct +{ + GMainLoop *loop; + gboolean timed_out; +} PropertyNotifyData; + +static void +on_property_notify (GObject *object, + GParamSpec *pspec, + gpointer user_data) +{ + PropertyNotifyData *data = user_data; + g_main_loop_quit (data->loop); +} + +static gboolean +on_property_notify_timeout (gpointer user_data) +{ + PropertyNotifyData *data = user_data; + data->timed_out = TRUE; + g_main_loop_quit (data->loop); + return TRUE; +} + +gboolean +_g_assert_property_notify_run (gpointer object, + const gchar *property_name) +{ + gchar *s; + gulong handler_id; + guint timeout_id; + PropertyNotifyData data; + + data.loop = g_main_loop_new (g_main_context_get_thread_default (), FALSE); + data.timed_out = FALSE; + s = g_strdup_printf ("notify::%s", property_name); + handler_id = g_signal_connect (object, + s, + G_CALLBACK (on_property_notify), + &data); + g_free (s); + timeout_id = g_timeout_add_seconds (30, + on_property_notify_timeout, + &data); + g_main_loop_run (data.loop); + g_signal_handler_disconnect (object, handler_id); + g_source_remove (timeout_id); + g_main_loop_unref (data.loop); + + return data.timed_out; +} + +static gboolean +_give_up (gpointer data) +{ + g_error ("%s", (const gchar *) data); + g_return_val_if_reached (TRUE); +} + +void +ensure_gdbus_testserver_up (void) +{ + guint id; + gchar *name_owner; + GDBusConnection *connection; + GDBusProxy *proxy; + GError *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); + + id = g_timeout_add_seconds (60, _give_up, + "waited more than ~ 60s for gdbus-testserver to take its bus name"); + + while (TRUE) + { + name_owner = g_dbus_proxy_get_name_owner (proxy); + + if (name_owner != NULL) + break; + + g_main_context_iteration (NULL, TRUE); + } + + g_source_remove (id); + g_free (name_owner); + g_object_unref (proxy); + g_object_unref (connection); +} + +/* ---------------------------------------------------------------------------------------------------- */ + +typedef struct +{ + GMainLoop *loop; + gboolean timed_out; +} SignalReceivedData; + +static void +on_signal_received (gpointer user_data) +{ + SignalReceivedData *data = user_data; + g_main_loop_quit (data->loop); +} + +static gboolean +on_signal_received_timeout (gpointer user_data) +{ + SignalReceivedData *data = user_data; + data->timed_out = TRUE; + g_main_loop_quit (data->loop); + return TRUE; +} + +gboolean +_g_assert_signal_received_run (gpointer object, + const gchar *signal_name) +{ + gulong handler_id; + guint timeout_id; + SignalReceivedData data; + + data.loop = g_main_loop_new (g_main_context_get_thread_default (), FALSE); + data.timed_out = FALSE; + handler_id = g_signal_connect_swapped (object, + signal_name, + G_CALLBACK (on_signal_received), + &data); + timeout_id = g_timeout_add_seconds (30, + on_signal_received_timeout, + &data); + g_main_loop_run (data.loop); + g_signal_handler_disconnect (object, handler_id); + g_source_remove (timeout_id); + g_main_loop_unref (data.loop); + + return data.timed_out; +} + +/* ---------------------------------------------------------------------------------------------------- */ + +GDBusConnection * +_g_bus_get_priv (GBusType bus_type, + GCancellable *cancellable, + GError **error) +{ + gchar *address; + GDBusConnection *ret; + + ret = NULL; + + address = g_dbus_address_get_for_bus_sync (bus_type, cancellable, error); + if (address == NULL) + goto out; + + ret = g_dbus_connection_new_for_address_sync (address, + G_DBUS_CONNECTION_FLAGS_AUTHENTICATION_CLIENT | + G_DBUS_CONNECTION_FLAGS_MESSAGE_BUS_CONNECTION, + NULL, /* GDBusAuthObserver */ + cancellable, + error); + g_free (address); + + out: + return ret; +} \ No newline at end of file diff --git a/gdbus-tests.h b/gdbus-tests.h new file mode 100755 index 0000000..2d7b8a1 --- /dev/null +++ b/gdbus-tests.h @@ -0,0 +1,122 @@ +/* GLib testing framework examples and tests + * + * Copyright (C) 2008-2009 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, see . + * + * Author: David Zeuthen + */ + +#ifndef __TESTS_H__ +#define __TESTS_H__ + +#include +#include "gdbus-sessionbus.h" + +G_BEGIN_DECLS + +/* TODO: clean up and move to gtestutils.c + * + * This is needed because libdbus-1 does not give predictable error messages - e.g. you + * get a different error message on connecting to a bus if the socket file is there vs + * if the socket file is missing. + */ + +#define _g_assert_error_domain(err, dom) do { if (!err || (err)->domain != dom) \ + g_assertion_message_error (G_LOG_DOMAIN, __FILE__, __LINE__, G_STRFUNC, \ + #err, err, dom, -1); } while (0) + +#define _g_assert_property_notify(object, property_name) \ + do \ + { \ + if (!G_IS_OBJECT (object)) \ + { \ + g_assertion_message (G_LOG_DOMAIN, \ + __FILE__, \ + __LINE__, \ + G_STRFUNC, \ + "Not a GObject instance"); \ + } \ + if (g_object_class_find_property (G_OBJECT_GET_CLASS (object), \ + property_name) == NULL) \ + { \ + g_assertion_message (G_LOG_DOMAIN, \ + __FILE__, \ + __LINE__, \ + G_STRFUNC, \ + "Property " property_name " does not " \ + "exist on object"); \ + } \ + if (_g_assert_property_notify_run (object, property_name)) \ + { \ + g_assertion_message (G_LOG_DOMAIN, \ + __FILE__, \ + __LINE__, \ + G_STRFUNC, \ + "Timed out waiting for notification " \ + "on property " property_name); \ + } \ + } \ + while (FALSE) + +#define _g_assert_signal_received(object, signal_name) \ + do \ + { \ + if (!G_IS_OBJECT (object)) \ + { \ + g_assertion_message (G_LOG_DOMAIN, \ + __FILE__, \ + __LINE__, \ + G_STRFUNC, \ + "Not a GObject instance"); \ + } \ + if (g_signal_lookup (signal_name, \ + G_TYPE_FROM_INSTANCE (object)) == 0) \ + { \ + g_assertion_message (G_LOG_DOMAIN, \ + __FILE__, \ + __LINE__, \ + G_STRFUNC, \ + "Signal '" signal_name "' does not " \ + "exist on object"); \ + } \ + if (_g_assert_signal_received_run (object, signal_name)) \ + { \ + g_assertion_message (G_LOG_DOMAIN, \ + __FILE__, \ + __LINE__, \ + G_STRFUNC, \ + "Timed out waiting for signal '" \ + signal_name "'"); \ + } \ + } \ + while (FALSE) + +gboolean _g_assert_property_notify_run (gpointer object, + const gchar *property_name); + + +gboolean _g_assert_signal_received_run (gpointer object, + const gchar *signal_name); + +GDBusConnection *_g_bus_get_priv (GBusType bus_type, + GCancellable *cancellable, + GError **error); + +void ensure_gdbus_testserver_up (void); + +G_END_DECLS + +#endif /* __TESTS_H__ */ + diff --git a/gdbus-testserver.c b/gdbus-testserver.c new file mode 100755 index 0000000..1ceb638 --- /dev/null +++ b/gdbus-testserver.c @@ -0,0 +1,893 @@ +#include +#include + +static GDBusNodeInfo *introspection_data = NULL; +static GMainLoop *loop = NULL; +static GHashTable *properties = NULL; + +static const gchar introspection_xml[] = + "" + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + " " + ""; + +static gboolean +end_sleep (gpointer data) +{ + GDBusMethodInvocation *invocation = data; + + g_dbus_method_invocation_return_value (invocation, NULL); + g_object_unref (invocation); + + return G_SOURCE_REMOVE; +} + +static void +handle_method_call (GDBusConnection *connection, + const gchar *sender, + const gchar *object_path, + const gchar *interface_name, + const gchar *method_name, + GVariant *parameters, + GDBusMethodInvocation *invocation, + gpointer user_data) +{ + if (g_strcmp0 (method_name, "HelloWorld") == 0) + { + const gchar *greeting; + + g_variant_get (parameters, "(&s)", &greeting); + if (g_strcmp0 (greeting, "Yo") == 0) + { + g_dbus_method_invocation_return_dbus_error (invocation, + "com.example.TestException", + "Yo is not a proper greeting"); + } + else + { + gchar *response; + response = g_strdup_printf ("You greeted me with '%s'. Thanks!", greeting); + g_dbus_method_invocation_return_value (invocation, + g_variant_new ("(s)", response)); + g_free ( response); + } + } + else if (g_strcmp0 (method_name, "DoubleHelloWorld") == 0) + { + const gchar *hello1, *hello2; + gchar *reply1, *reply2; + + g_variant_get (parameters, "(&s&s)", &hello1, &hello2); + reply1 = g_strdup_printf ("You greeted me with '%s'. Thanks!", hello1); + reply2 = g_strdup_printf ("Yo dawg, you uttered '%s'. Thanks!", hello2); + g_dbus_method_invocation_return_value (invocation, + g_variant_new ("(ss)", reply1, reply2)); + g_free (reply1); + g_free (reply2); + } + else if (g_strcmp0 (method_name, "PairReturn") == 0) + { + g_dbus_method_invocation_return_value (invocation, + g_variant_new ("(su)", "foo", 42)); + } + else if (g_strcmp0 (method_name, "TestPrimitiveTypes") == 0) + { + guchar val_byte; + gboolean val_boolean; + gint16 val_int16; + guint16 val_uint16; + gint32 val_int32; + guint32 val_uint32; + gint64 val_int64; + guint64 val_uint64; + gdouble val_double; + const gchar *val_string; + const gchar *val_objpath; + const gchar *val_signature; + gchar *ret_string; + gchar *ret_objpath; + gchar *ret_signature; + + g_variant_get (parameters, "(ybnqiuxtd&s&o&g)", + &val_byte, + &val_boolean, + &val_int16, + &val_uint16, + &val_int32, + &val_uint32, + &val_int64, + &val_uint64, + &val_double, + &val_string, + &val_objpath, + &val_signature); + + ret_string = g_strconcat (val_string, val_string, NULL); + ret_objpath = g_strconcat (val_objpath, "/modified", NULL); + ret_signature = g_strconcat (val_signature, val_signature, NULL); + + g_dbus_method_invocation_return_value (invocation, + g_variant_new ("(ybnqiuxtdsog)", + val_byte + 1, + !val_boolean, + val_int16 + 1, + val_uint16 + 1, + val_int32 + 1, + val_uint32 + 1, + val_int64 + 1, + val_uint64 + 1, + - val_double + 0.123, + ret_string, + ret_objpath, + ret_signature)); + + g_free (ret_string); + g_free (ret_objpath); + g_free (ret_signature); + } + else if (g_strcmp0 (method_name, "TestArrayOfPrimitiveTypes") == 0) + { + GVariant *v; + const guchar *bytes; + const gint16 *int16s; + const guint16 *uint16s; + const gint32 *int32s; + const guint32 *uint32s; + const gint64 *int64s; + const guint64 *uint64s; + const gdouble *doubles; + gsize n_elts; + gint i, j; + GVariantBuilder ret; + + g_variant_builder_init (&ret, G_VARIANT_TYPE ("(ayabanaqaiauaxatad)")); + + v = g_variant_get_child_value (parameters, 0); + bytes = g_variant_get_fixed_array (v, &n_elts, 1); + g_variant_builder_open (&ret, G_VARIANT_TYPE ("ay")); + for (j = 0; j < 2; j++) + for (i = 0; i < n_elts; i++) + g_variant_builder_add (&ret, "y", bytes[i]); + g_variant_builder_close (&ret); + g_variant_unref (v); + + v = g_variant_get_child_value (parameters, 1); + bytes = g_variant_get_fixed_array (v, &n_elts, 1); + g_variant_builder_open (&ret, G_VARIANT_TYPE ("ab")); + for (j = 0; j < 2; j++) + for (i = 0; i < n_elts; i++) + g_variant_builder_add (&ret, "b", (gboolean)bytes[i]); + g_variant_builder_close (&ret); + g_variant_unref (v); + + v = g_variant_get_child_value (parameters, 2); + int16s = g_variant_get_fixed_array (v, &n_elts, 2); + g_variant_builder_open (&ret, G_VARIANT_TYPE ("an")); + for (j = 0; j < 2; j++) + for (i = 0; i < n_elts; i++) + g_variant_builder_add (&ret, "n", int16s[i]); + g_variant_builder_close (&ret); + g_variant_unref (v); + + v = g_variant_get_child_value (parameters, 3); + uint16s = g_variant_get_fixed_array (v, &n_elts, 2); + g_variant_builder_open (&ret, G_VARIANT_TYPE ("aq")); + for (j = 0; j < 2; j++) + for (i = 0; i < n_elts; i++) + g_variant_builder_add (&ret, "q", uint16s[i]); + g_variant_builder_close (&ret); + g_variant_unref (v); + + v = g_variant_get_child_value (parameters, 4); + int32s = g_variant_get_fixed_array (v, &n_elts, 4); + g_variant_builder_open (&ret, G_VARIANT_TYPE ("ai")); + for (j = 0; j < 2; j++) + for (i = 0; i < n_elts; i++) + g_variant_builder_add (&ret, "i", int32s[i]); + g_variant_builder_close (&ret); + g_variant_unref (v); + + v = g_variant_get_child_value (parameters, 5); + uint32s = g_variant_get_fixed_array (v, &n_elts, 4); + g_variant_builder_open (&ret, G_VARIANT_TYPE ("au")); + for (j = 0; j < 2; j++) + for (i = 0; i < n_elts; i++) + g_variant_builder_add (&ret, "u", uint32s[i]); + g_variant_builder_close (&ret); + g_variant_unref (v); + + v = g_variant_get_child_value (parameters, 6); + int64s = g_variant_get_fixed_array (v, &n_elts, 8); + g_variant_builder_open (&ret, G_VARIANT_TYPE ("ax")); + for (j = 0; j < 2; j++) + for (i = 0; i < n_elts; i++) + g_variant_builder_add (&ret, "x", int64s[i]); + g_variant_builder_close (&ret); + g_variant_unref (v); + + v = g_variant_get_child_value (parameters, 7); + uint64s = g_variant_get_fixed_array (v, &n_elts, 8); + g_variant_builder_open (&ret, G_VARIANT_TYPE ("at")); + for (j = 0; j < 2; j++) + for (i = 0; i < n_elts; i++) + g_variant_builder_add (&ret, "t", uint64s[i]); + g_variant_builder_close (&ret); + g_variant_unref (v); + + v = g_variant_get_child_value (parameters, 8); + doubles = g_variant_get_fixed_array (v, &n_elts, sizeof (gdouble)); + g_variant_builder_open (&ret, G_VARIANT_TYPE ("ad")); + for (j = 0; j < 2; j++) + for (i = 0; i < n_elts; i++) + g_variant_builder_add (&ret, "d", doubles[i]); + g_variant_builder_close (&ret); + g_variant_unref (v); + + g_dbus_method_invocation_return_value (invocation, + g_variant_builder_end (&ret)); + } + else if (g_strcmp0 (method_name, "TestArrayOfStringTypes") == 0) + { + GVariantIter *iter1; + GVariantIter *iter2; + GVariantIter *iter3; + GVariantIter *iter; + GVariantBuilder ret; + const gchar *s; + gint i; + + g_variant_builder_init (&ret, G_VARIANT_TYPE ("(asaoag)")); + g_variant_get (parameters, "(asaoag)", &iter1, &iter2, &iter3); + + g_variant_builder_open (&ret, G_VARIANT_TYPE ("as")); + for (i = 0; i < 2; i++) + { + iter = g_variant_iter_copy (iter1); + while (g_variant_iter_loop (iter, "s", &s)) + g_variant_builder_add (&ret, "s", s); + g_variant_iter_free (iter); + } + g_variant_builder_close (&ret); + + g_variant_builder_open (&ret, G_VARIANT_TYPE ("ao")); + for (i = 0; i < 2; i++) + { + iter = g_variant_iter_copy (iter1); + while (g_variant_iter_loop (iter, "o", &s)) + g_variant_builder_add (&ret, "o", s); + g_variant_iter_free (iter); + } + g_variant_builder_close (&ret); + + g_variant_builder_open (&ret, G_VARIANT_TYPE ("ag")); + for (i = 0; i < 2; i++) + { + iter = g_variant_iter_copy (iter1); + while (g_variant_iter_loop (iter, "g", &s)) + g_variant_builder_add (&ret, "g", s); + g_variant_iter_free (iter); + } + g_variant_builder_close (&ret); + + g_variant_iter_free (iter1); + g_variant_iter_free (iter2); + g_variant_iter_free (iter3); + + g_dbus_method_invocation_return_value (invocation, + g_variant_builder_end (&ret)); + } + else if (g_strcmp0 (method_name, "TestHashTables") == 0) + { + GVariant *v; + GVariantIter iter; + GVariantBuilder ret; + guint8 y1, y2; + gboolean b1, b2; + gint16 n1, n2; + guint16 q1, q2; + gint i1, i2; + guint u1, u2; + gint64 x1, x2; + guint64 t1, t2; + gdouble d1, d2; + gchar *s1, *s2; + + g_print("Hello\n"); + + g_variant_builder_init (&ret, G_VARIANT_TYPE ("(a{yy}a{bb}a{nn}a{qq}a{ii}a{uu}a{xx}a{tt}a{dd}a{ss}a{oo}a{gg})")); + + g_variant_builder_open (&ret, G_VARIANT_TYPE ("a{yy}")); + v = g_variant_get_child_value (parameters, 0); + g_variant_iter_init (&iter, v); + while (g_variant_iter_loop (&iter, "yy", &y1, &y2)) + g_variant_builder_add (&ret, "{yy}", y1 * 2, (y2 * 3) & 255); + g_variant_unref (v); + g_variant_builder_close (&ret); + + g_variant_builder_open (&ret, G_VARIANT_TYPE ("a{bb}")); + v = g_variant_get_child_value (parameters, 1); + g_variant_iter_init (&iter, v); + while (g_variant_iter_loop (&iter, "bb", &b1, &b2)) + g_variant_builder_add (&ret, "{bb}", b1, TRUE); + g_variant_unref (v); + g_variant_builder_close (&ret); + + g_variant_builder_open (&ret, G_VARIANT_TYPE ("a{nn}")); + v = g_variant_get_child_value (parameters, 2); + g_variant_iter_init (&iter, v); + while (g_variant_iter_loop (&iter, "nn", &n1, &n2)) + g_variant_builder_add (&ret, "{nn}", n1 * 2, n2 * 3); + g_variant_unref (v); + g_variant_builder_close (&ret); + + g_variant_builder_open (&ret, G_VARIANT_TYPE ("a{qq}")); + v = g_variant_get_child_value (parameters, 3); + g_variant_iter_init (&iter, v); + while (g_variant_iter_loop (&iter, "qq", &q1, &q2)) + g_variant_builder_add (&ret, "{qq}", q1 * 2, q2 * 3); + g_variant_unref (v); + g_variant_builder_close (&ret); + + g_variant_builder_open (&ret, G_VARIANT_TYPE ("a{ii}")); + v = g_variant_get_child_value (parameters, 4); + g_variant_iter_init (&iter, v); + while (g_variant_iter_loop (&iter, "ii", &i1, &i2)) + g_variant_builder_add (&ret, "{ii}", i1 * 2, i2 * 3); + g_variant_unref (v); + g_variant_builder_close (&ret); + + g_variant_builder_open (&ret, G_VARIANT_TYPE ("a{uu}")); + v = g_variant_get_child_value (parameters, 5); + g_variant_iter_init (&iter, v); + while (g_variant_iter_loop (&iter, "uu", &u1, &u2)) + g_variant_builder_add (&ret, "{uu}", u1 * 2, u2 * 3); + g_variant_unref (v); + g_variant_builder_close (&ret); + + g_variant_builder_open (&ret, G_VARIANT_TYPE ("a{xx}")); + v = g_variant_get_child_value (parameters, 6); + g_variant_iter_init (&iter, v); + while (g_variant_iter_loop (&iter, "xx", &x1, &x2)) + g_variant_builder_add (&ret, "{xx}", x1 + 2, x2 + 1); + g_variant_unref (v); + g_variant_builder_close (&ret); + + g_variant_builder_open (&ret, G_VARIANT_TYPE ("a{tt}")); + v = g_variant_get_child_value (parameters, 7); + g_variant_iter_init (&iter, v); + while (g_variant_iter_loop (&iter, "tt", &t1, &t2)) + g_variant_builder_add (&ret, "{tt}", t1 + 2, t2 + 1); + g_variant_unref (v); + g_variant_builder_close (&ret); + + g_variant_builder_open (&ret, G_VARIANT_TYPE ("a{dd}")); + v = g_variant_get_child_value (parameters, 8); + g_variant_iter_init (&iter, v); + while (g_variant_iter_loop (&iter, "dd", &d1, &d2)) + g_variant_builder_add (&ret, "{dd}", d1 + 2.5, d2 + 5.0); + g_variant_unref (v); + g_variant_builder_close (&ret); + + g_variant_builder_open (&ret, G_VARIANT_TYPE ("a{ss}")); + v = g_variant_get_child_value (parameters, 9); + g_variant_iter_init (&iter, v); + while (g_variant_iter_loop (&iter, "ss", &s1, &s2)) + { + gchar *tmp1, *tmp2; + tmp1 = g_strconcat (s1, "mod", NULL); + tmp2 = g_strconcat (s2, s2, NULL); + g_variant_builder_add (&ret, "{ss}", tmp1, tmp2); + g_free (tmp1); + g_free (tmp2); + } + g_variant_unref (v); + g_variant_builder_close (&ret); + + g_variant_builder_open (&ret, G_VARIANT_TYPE ("a{oo}")); + v = g_variant_get_child_value (parameters, 10); + g_variant_iter_init (&iter, v); + while (g_variant_iter_loop (&iter, "oo", &s1, &s2)) + { + gchar *tmp1, *tmp2; + tmp1 = g_strconcat (s1, "/mod", NULL); + tmp2 = g_strconcat (s2, "/mod2", NULL); + g_variant_builder_add (&ret, "{oo}", tmp1, tmp2); + g_free (tmp1); + g_free (tmp2); + } + g_variant_unref (v); + g_variant_builder_close (&ret); + + g_variant_builder_open (&ret, G_VARIANT_TYPE ("a{gg}")); + v = g_variant_get_child_value (parameters, 11); + g_variant_iter_init (&iter, v); + while (g_variant_iter_loop (&iter, "gg", &s1, &s2)) + { + gchar *tmp1, *tmp2; + tmp1 = g_strconcat (s1, "assgit", NULL); + tmp2 = g_strconcat (s2, s2, NULL); + g_variant_builder_add (&ret, "{gg}", tmp1, tmp2); + g_free (tmp1); + g_free (tmp2); + } + g_variant_unref (v); + g_variant_builder_close (&ret); + + g_dbus_method_invocation_return_value (invocation, + g_variant_builder_end (&ret)); + } + else if (g_strcmp0 (method_name, "TestStructureTypes") == 0) + { + gint x, y, x1, y1; + const gchar *desc; + GVariantIter *iter1, *iter2; + gchar *desc_ret; + GVariantBuilder ret1, ret2; + GVariantIter *iter; + GVariant *v; + gchar *s1, *s2; + + g_variant_get (parameters, "((ii)(&s(ii)aya{ss}))", + &x, &y, &desc, &x1, &y1, &iter1, &iter2); + + desc_ret = g_strconcat (desc, "... in bed!", NULL); + + g_variant_builder_init (&ret1, G_VARIANT_TYPE ("ay")); + iter = g_variant_iter_copy (iter1); + while (g_variant_iter_loop (iter1, "y", &v)) + g_variant_builder_add (&ret1, "y", v); + while (g_variant_iter_loop (iter, "y", &v)) + g_variant_builder_add (&ret1, "y", v); + g_variant_iter_free (iter); + g_variant_iter_free (iter1); + + g_variant_builder_init (&ret2, G_VARIANT_TYPE ("a{ss}")); + while (g_variant_iter_loop (iter1, "ss", &s1, &s2)) + { + gchar *tmp; + tmp = g_strconcat (s2, " ... in bed!", NULL); + g_variant_builder_add (&ret1, "{ss}", s1, tmp); + g_free (tmp); + } + g_variant_iter_free (iter2); + + g_dbus_method_invocation_return_value (invocation, + g_variant_new ("((ii)(&s(ii)aya{ss}))", + x + 1, y + 1, desc_ret, x1 + 2, y1 + 2, + &ret1, &ret2)); + + g_free (desc_ret); + } + else if (g_strcmp0 (method_name, "TestVariant") == 0) + { + GVariant *v; + gboolean modify; + GVariant *ret; + + g_variant_get (parameters, "(vb)", &v, &modify); + + /* FIXME handle more cases */ + if (modify) + { + if (g_variant_is_of_type (v, G_VARIANT_TYPE_BOOLEAN)) + { + ret = g_variant_new_boolean (FALSE); + } + else if (g_variant_is_of_type (v, G_VARIANT_TYPE_TUPLE)) + { + ret = g_variant_new ("(si)", "other struct", 100); + } + else + g_assert_not_reached (); + } + else + ret = v; + + g_dbus_method_invocation_return_value (invocation, ret); + g_variant_unref (v); + } + else if (g_strcmp0 (method_name, "TestComplexArrays") == 0) + { + /* FIXME */ + g_dbus_method_invocation_return_value (invocation, parameters); + } + else if (g_strcmp0 (method_name, "TestComplexHashTables") == 0) + { + /* FIXME */ + g_dbus_method_invocation_return_value (invocation, parameters); + } + else if (g_strcmp0 (method_name, "FrobSetProperty") == 0) + { + gchar *name; + GVariant *value; + g_variant_get (parameters, "(sv)", &name, &value); + g_hash_table_replace (properties, name, value); + g_dbus_connection_emit_signal (connection, + NULL, + "/com/example/TestObject", + "org.freedesktop.DBus.Properties", + "PropertiesChanged", + g_variant_new_parsed ("('com.example.Frob', [{%s, %v}], @as [])", name, value), + NULL); + g_dbus_method_invocation_return_value (invocation, NULL); + } + else if (g_strcmp0 (method_name, "FrobInvalidateProperty") == 0) + { + const gchar *value; + g_variant_get (parameters, "(&s)", &value); + g_hash_table_replace (properties, g_strdup ("PropertyThatWillBeInvalidated"), g_variant_ref_sink (g_variant_new_string (value))); + + g_dbus_connection_emit_signal (connection, + NULL, + "/com/example/TestObject", + "org.freedesktop.DBus.Properties", + "PropertiesChanged", + g_variant_new_parsed ("('com.example.Frob', @a{sv} [], ['PropertyThatWillBeInvalidated'])"), + NULL); + g_dbus_method_invocation_return_value (invocation, NULL); + } + else if (g_strcmp0 (method_name, "EmitSignal") == 0) + { + const gchar *str; + const gchar *path; + gchar *str_ret; + gchar *path_ret; + g_variant_get (parameters, "(&s&o)", &str, &path); + str_ret = g_strconcat (str, " .. in bed!", NULL); + path_ret = g_strconcat (path, "/in/bed", NULL); + g_dbus_connection_emit_signal (connection, + NULL, + "/com/example/TestObject", + "com.example.Frob", + "TestSignal", + g_variant_new_parsed ("(%s, %o, <'a variant'>)", str_ret, path_ret), + NULL); + g_free (str_ret); + g_free (path_ret); + g_dbus_method_invocation_return_value (invocation, NULL); + } + else if (g_strcmp0 (method_name, "EmitSignal2") == 0) + { + g_dbus_connection_emit_signal (connection, + NULL, + "/com/example/TestObject", + "com.example.Frob", + "TestSignal2", + g_variant_new_parsed ("(42, )"), + NULL); + g_dbus_method_invocation_return_value (invocation, NULL); + } + else if (g_strcmp0 (method_name, "Sleep") == 0) + { + gint msec; + + g_variant_get (parameters, "(i)", &msec); + + g_timeout_add ((guint)msec, end_sleep, g_object_ref (invocation)); + } + else if (g_strcmp0 (method_name, "Quit") == 0) + { + g_dbus_method_invocation_return_value (invocation, NULL); + g_main_loop_quit (loop); + } +} + +static GVariant * +handle_get_property (GDBusConnection *connection, + const gchar *sender, + const gchar *object_path, + const gchar *interface_name, + const gchar *property_name, + GError **error, + gpointer user_data) +{ + GVariant *ret; + + ret = g_hash_table_lookup (properties, property_name); + if (ret) + { + g_assert (!g_variant_is_floating (ret)); + g_variant_ref (ret); + } + else + { + g_set_error (error, + G_DBUS_ERROR, G_DBUS_ERROR_FAILED, + "no such property: %s", property_name); + } + + return ret; +} + +static gboolean +handle_set_property (GDBusConnection *connection, + const gchar *sender, + const gchar *object_path, + const gchar *interface_name, + const gchar *property_name, + GVariant *value, + GError **error, + gpointer user_data) +{ + g_set_error (error, + G_DBUS_ERROR, G_DBUS_ERROR_FAILED, + "SetProperty not implemented"); + return FALSE; +} + +static const GDBusInterfaceVTable interface_vtable = +{ + handle_method_call, + handle_get_property, + handle_set_property +}; + +static void +on_bus_acquired (GDBusConnection *connection, + const gchar *name, + gpointer user_data) +{ + guint id; + + id = g_dbus_connection_register_object (connection, + "/com/example/TestObject", + introspection_data->interfaces[0], + &interface_vtable, + NULL, + NULL, + NULL); + g_assert (id > 0); +} + +static void +on_name_acquired (GDBusConnection *connection, + const gchar *name, + gpointer user_data) +{ +} + +static void +on_name_lost (GDBusConnection *connection, + const gchar *name, + gpointer user_data) +{ + exit (1); +} + +int +main (int argc, char *argv[]) +{ + guint owner_id; + + introspection_data = g_dbus_node_info_new_for_xml (introspection_xml, NULL); + properties = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, (GDestroyNotify)g_variant_unref); + g_hash_table_insert (properties, g_strdup ("y"), g_variant_ref_sink (g_variant_new_byte (1))); + g_hash_table_insert (properties, g_strdup ("b"), g_variant_ref_sink (g_variant_new_boolean (TRUE))); + g_hash_table_insert (properties, g_strdup ("n"), g_variant_ref_sink (g_variant_new_int16 (2))); + g_hash_table_insert (properties, g_strdup ("q"), g_variant_ref_sink (g_variant_new_uint16 (3))); + g_hash_table_insert (properties, g_strdup ("i"), g_variant_ref_sink (g_variant_new_int32 (4))); + g_hash_table_insert (properties, g_strdup ("u"), g_variant_ref_sink (g_variant_new_uint32 (5))); + g_hash_table_insert (properties, g_strdup ("x"), g_variant_ref_sink (g_variant_new_int64 (6))); + g_hash_table_insert (properties, g_strdup ("t"), g_variant_ref_sink (g_variant_new_uint64 (7))); + g_hash_table_insert (properties, g_strdup ("d"), g_variant_ref_sink (g_variant_new_double (7.5))); + g_hash_table_insert (properties, g_strdup ("s"), g_variant_ref_sink (g_variant_new_string ("a string"))); + g_hash_table_insert (properties, g_strdup ("o"), g_variant_ref_sink (g_variant_new_object_path ("/some/path"))); + g_hash_table_insert (properties, g_strdup ("ay"), g_variant_ref_sink (g_variant_new_parsed ("[@y 1, @y 11]"))); + g_hash_table_insert (properties, g_strdup ("ab"), g_variant_ref_sink (g_variant_new_parsed ("[true, false]"))); + g_hash_table_insert (properties, g_strdup ("an"), g_variant_ref_sink (g_variant_new_parsed ("[@n 2, @n 12]"))); + g_hash_table_insert (properties, g_strdup ("aq"), g_variant_ref_sink (g_variant_new_parsed ("[@q 3, @q 13]"))); + g_hash_table_insert (properties, g_strdup ("ai"), g_variant_ref_sink (g_variant_new_parsed ("[@i 4, @i 14]"))); + g_hash_table_insert (properties, g_strdup ("au"), g_variant_ref_sink (g_variant_new_parsed ("[@u 5, @u 15]"))); + g_hash_table_insert (properties, g_strdup ("ax"), g_variant_ref_sink (g_variant_new_parsed ("[@x 6, @x 16]"))); + g_hash_table_insert (properties, g_strdup ("at"), g_variant_ref_sink (g_variant_new_parsed ("[@t 7, @t 17]"))); + g_hash_table_insert (properties, g_strdup ("ad"), g_variant_ref_sink (g_variant_new_parsed ("[7.5, 17.5]"))); + g_hash_table_insert (properties, g_strdup ("as"), g_variant_ref_sink (g_variant_new_parsed ("['a string', 'another string']"))); + g_hash_table_insert (properties, g_strdup ("ao"), g_variant_ref_sink (g_variant_new_parsed ("[@o '/some/path', @o '/another/path']"))); + g_hash_table_insert (properties, g_strdup ("foo"), g_variant_ref_sink (g_variant_new_string ("a frobbed string"))); + g_hash_table_insert (properties, g_strdup ("PropertyThatWillBeInvalidated"), g_variant_ref_sink (g_variant_new_string ("InitialValue"))); + + owner_id = g_bus_own_name (G_BUS_TYPE_SESSION, + "com.example.TestService", + G_BUS_NAME_OWNER_FLAGS_NONE, + on_bus_acquired, + on_name_acquired, + on_name_lost, + NULL, + NULL); + + loop = g_main_loop_new (NULL, FALSE); + g_main_loop_run (loop); + + g_bus_unown_name (owner_id); + + g_dbus_node_info_unref (introspection_data); + + return 0; +} + + diff --git a/server.c b/server.c new file mode 100755 index 0000000..e884eed --- /dev/null +++ b/server.c @@ -0,0 +1,145 @@ +/* + * Source Code from the DBus Activation Tutorial + * from Raphael Slinckx + * + * This code illustrates how to requrest the dbus daemon + * to automatically start a program that provides a given + * service. For more detailed information refer also to + * http://raphael.slinckx.net/blog/documents/dbus-tutorial + * where all source has taken from. + * + * Provision of all glue code to form compilable application + * by Otto Linnemann + * + * Implementations for Server + */ + +#include "server.h" +#include "server-bindings.h" + +static void server_class_init(ServerClass *klass); + +GType server_get_type(void) +{ + static GType server_type = 0; + + if(!server_type) + { + static const GTypeInfo server_info = { + sizeof(ServerClass), /* class structure size */ + NULL, /* base class initializer */ + NULL, /* base class finalizer */ + (GClassInitFunc)server_class_init, /* class initializer */ + NULL, /* class finalizer */ + NULL, /* class data */ + sizeof(Server), /* instance structure size */ + 1, /* preallocated instances */ + NULL, /* instance initializers */ + NULL + }; + + server_type = g_type_register_static( + G_TYPE_OBJECT, /* parent class */ + "Server", + &server_info, + 0); + } + + return server_type; +} + +static void server_class_init(ServerClass *klass) +{ + GError *error = NULL; + + /* Init the DBus connection, per-klass */ + klass->connection = dbus_g_bus_get (DBUS_BUS_SESSION, &error); + if (klass->connection == NULL) + { + g_warning("Unable to connect to dbus: %s", error->message); + g_error_free (error); + return; + } + + /* &dbus_glib__object_info is provided in the server-bindings.h file */ + /* OBJECT_TYPE_SERVER is the GType of your server object */ + dbus_g_object_type_install_info (TYPE_SERVER, &dbus_glib_server_object_object_info); + + g_message("ServerClass successfully initialized"); +} + + +static gint server_init(Server *server) +{ + GError *error = NULL; + DBusGProxy *driver_proxy; + ServerClass *klass = SERVER_GET_CLASS (server); + guint request_ret; + gint retcode = 1; + + /* Register DBUS path */ + dbus_g_connection_register_g_object (klass->connection, + "/org/gnome/ServiceName", + G_OBJECT (server)); + + /* Register the service name, the constant here are defined in dbus-glib-bindings.h */ + driver_proxy = dbus_g_proxy_new_for_name (klass->connection, + DBUS_SERVICE_DBUS, + DBUS_PATH_DBUS, + DBUS_INTERFACE_DBUS); + + if(!org_freedesktop_DBus_request_name (driver_proxy, + "org.gnome.ServiceName", + 0, &request_ret, /* See tutorial for more infos about these */ + &error)) + { + g_warning("Unable to register service: %s", error->message); + g_error_free (error); + retcode = 0; + } + g_object_unref (driver_proxy); + + return( retcode ); +} + + +gboolean server_echo_string (Server *server, gchar *original, gchar **echo, GError **error) +{ + *echo = g_strdup (original); + gboolean problem = FALSE; + + if (problem) + { + /* We have an error, set the gerror */ + g_set_error (error, g_quark_from_static_string ("echo"), + 0xdeadbeef, + "Some random problem occured, you're screwed"); + return FALSE; + } + + return TRUE; +} + +int main() +{ + Server *server; + GMainLoop *loop; + + g_type_init(); /* initialize type system */ + + if (!g_thread_supported ()) + g_thread_init (NULL); + + dbus_g_thread_init (); + + loop = g_main_loop_new (NULL, FALSE); + + server = g_object_new( TYPE_SERVER, NULL ); + + if( server_init( server ) ) + { + g_main_loop_run (loop); + } + + return 0; +} diff --git a/server.h b/server.h new file mode 100755 index 0000000..21266d3 --- /dev/null +++ b/server.h @@ -0,0 +1,48 @@ +/* + * Source Code from the DBus Activation Tutorial + * from Raphael Slinckx + * + * This code illustrates how to requrest the dbus daemon + * to automatically start a program that provides a given + * service. For more detailed information refer also to + * http://raphael.slinckx.net/blog/documents/dbus-tutorial + * where all source has taken from. + * + * Provision of all glue code to form compilable application + * by Otto Linnemann + * + * Declarations for Server + */ + +#ifndef SERVER_H +#define SERVER_H + +#include + +/* Standard GObject class structures, etc */ + +#define TYPE_SERVER (server_get_type()) +#define SERVER(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), TYPE_SERVER, Server)) +#define IS_SERVER(obj) (G_TYPE_CHECK_INSTANCE_TYPE ((obj), TYPE_SERVER)) +#define SERVER_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), TYPE_SERVER, ServerClass)) +#define IS_SERVER_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), TYPE_SERVER)) +#define SERVER_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), TYPE_SERVER, ServerClass)) + +typedef struct +{ + GObjectClass parent_class; + + DBusGConnection *connection; +} ServerClass; + +typedef struct +{ + GObject parnet_instance; + +} Server; + + +GType server_get_type(void); +gboolean server_echo_string(Server *server, gchar *original, gchar **echo, GError **error); + +#endif /* #ifndef SERVER_H */ diff --git a/servicename-infos.xml b/servicename-infos.xml new file mode 100755 index 0000000..119253c --- /dev/null +++ b/servicename-infos.xml @@ -0,0 +1,12 @@ + + + + + + + + + + + +