mirror of
https://github.com/joprietoe/gdbus.git
synced 2025-04-04 16:06:50 +02:00
v01
This commit is contained in:
commit
082547143c
29
.vscode/tasks.json
vendored
Normal file
29
.vscode/tasks.json
vendored
Normal file
@ -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
|
||||
}
|
||||
}
|
||||
}
|
||||
]
|
||||
}
|
39
Makefile
Executable file
39
Makefile
Executable file
@ -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
|
146
client.c
Executable file
146
client.c
Executable file
@ -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;
|
||||
}
|
48
client.h
Executable file
48
client.h
Executable file
@ -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 <dbus/dbus-glib-bindings.h>
|
||||
|
||||
/* 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 */
|
132
gdbus-example-client.c
Normal file
132
gdbus-example-client.c
Normal file
@ -0,0 +1,132 @@
|
||||
#include <string.h>
|
||||
#include <stdlib.h>
|
||||
|
||||
#include <sys/types.h>
|
||||
#include <unistd.h>
|
||||
|
||||
#include <time.h>
|
||||
|
||||
#include <gio/gio.h>
|
||||
#include <gio-unix-2.0/gio/gunixfdlist.h>
|
||||
|
||||
/* 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;
|
||||
}
|
391
gdbus-example-server.c
Executable file
391
gdbus-example-server.c
Executable file
@ -0,0 +1,391 @@
|
||||
#include <stdlib.h>
|
||||
#include <gio/gio.h>
|
||||
|
||||
#ifdef G_OS_UNIX
|
||||
#include <gio/gunixfdlist.h>
|
||||
/* For STDOUT_FILENO */
|
||||
#include <unistd.h>
|
||||
#endif
|
||||
|
||||
/* ---------------------------------------------------------------------------------------------------- */
|
||||
|
||||
static GDBusNodeInfo *introspection_data = NULL;
|
||||
|
||||
/* Introspection data for the service we are exporting */
|
||||
static const gchar introspection_xml[] =
|
||||
"<node>"
|
||||
" <interface name='org.gtk.GDBus.TestInterface'>"
|
||||
" <annotation name='org.gtk.GDBus.Annotation' value='OnInterface'/>"
|
||||
" <annotation name='org.gtk.GDBus.Annotation' value='AlsoOnInterface'/>"
|
||||
" <method name='HelloWorld'>"
|
||||
" <annotation name='org.gtk.GDBus.Annotation' value='OnMethod'/>"
|
||||
" <arg type='s' name='greeting' direction='in'/>"
|
||||
" <arg type='s' name='response' direction='out'/>"
|
||||
" </method>"
|
||||
" <method name='EmitSignal'>"
|
||||
" <arg type='d' name='speed_in_mph' direction='in'>"
|
||||
" <annotation name='org.gtk.GDBus.Annotation' value='OnArg'/>"
|
||||
" </arg>"
|
||||
" </method>"
|
||||
" <method name='GimmeStdout'/>"
|
||||
" <signal name='VelocityChanged'>"
|
||||
" <annotation name='org.gtk.GDBus.Annotation' value='Onsignal'/>"
|
||||
" <arg type='d' name='speed_in_mph'/>"
|
||||
" <arg type='s' name='speed_as_string'>"
|
||||
" <annotation name='org.gtk.GDBus.Annotation' value='OnArg_NonFirst'/>"
|
||||
" </arg>"
|
||||
" </signal>"
|
||||
" <property type='s' name='FluxCapicitorName' access='read'>"
|
||||
" <annotation name='org.gtk.GDBus.Annotation' value='OnProperty'>"
|
||||
" <annotation name='org.gtk.GDBus.Annotation' value='OnAnnotation_YesThisIsCrazy'/>"
|
||||
" </annotation>"
|
||||
" </property>"
|
||||
" <property type='s' name='Title' access='readwrite'/>"
|
||||
" <property type='s' name='ReadingAlwaysThrowsError' access='read'/>"
|
||||
" <property type='s' name='WritingAlwaysThrowsError' access='readwrite'/>"
|
||||
" <property type='s' name='OnlyWritable' access='write'/>"
|
||||
" <property type='s' name='Foo' access='read'/>"
|
||||
" <property type='s' name='Bar' access='read'/>"
|
||||
" </interface>"
|
||||
"</node>";
|
||||
|
||||
/* ---------------------------------------------------------------------------------------------------- */
|
||||
|
||||
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;
|
||||
}
|
||||
|
132
gdbus-example-unix-fd-client.c
Executable file
132
gdbus-example-unix-fd-client.c
Executable file
@ -0,0 +1,132 @@
|
||||
#include <string.h>
|
||||
#include <stdlib.h>
|
||||
|
||||
#include <sys/types.h>
|
||||
#include <unistd.h>
|
||||
|
||||
#include <time.h>
|
||||
|
||||
#include <gio/gio.h>
|
||||
#include <gio/gunixfdlist.h>
|
||||
|
||||
/* 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;
|
||||
}
|
156
gdbus-message.c
Executable file
156
gdbus-message.c
Executable file
@ -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 <http://www.gnu.org/licenses/>.
|
||||
*
|
||||
* Author: David Zeuthen <davidz@redhat.com>
|
||||
*/
|
||||
|
||||
#include <locale.h>
|
||||
#include <gio/gio.h>
|
||||
|
||||
/* ---------------------------------------------------------------------------------------------------- */
|
||||
|
||||
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();
|
||||
}
|
||||
|
207
gdbus-tests.c
Executable file
207
gdbus-tests.c
Executable file
@ -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 <http://www.gnu.org/licenses/>.
|
||||
*
|
||||
* Author: David Zeuthen <davidz@redhat.com>
|
||||
*/
|
||||
|
||||
#include <gio/gio.h>
|
||||
#include <unistd.h>
|
||||
|
||||
#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;
|
||||
}
|
122
gdbus-tests.h
Executable file
122
gdbus-tests.h
Executable file
@ -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 <http://www.gnu.org/licenses/>.
|
||||
*
|
||||
* Author: David Zeuthen <davidz@redhat.com>
|
||||
*/
|
||||
|
||||
#ifndef __TESTS_H__
|
||||
#define __TESTS_H__
|
||||
|
||||
#include <gio/gio.h>
|
||||
#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__ */
|
||||
|
893
gdbus-testserver.c
Executable file
893
gdbus-testserver.c
Executable file
@ -0,0 +1,893 @@
|
||||
#include <gio/gio.h>
|
||||
#include <stdlib.h>
|
||||
|
||||
static GDBusNodeInfo *introspection_data = NULL;
|
||||
static GMainLoop *loop = NULL;
|
||||
static GHashTable *properties = NULL;
|
||||
|
||||
static const gchar introspection_xml[] =
|
||||
"<node>"
|
||||
" <interface name='com.example.Frob'>"
|
||||
" <method name='Quit'>"
|
||||
" </method>"
|
||||
" <method name='TestArrayOfStringTypes'>"
|
||||
" <arg direction='in' type='as' name='val_string' />"
|
||||
" <arg direction='in' type='ao' name='val_objpath' />"
|
||||
" <arg direction='in' type='ag' name='val_signature' />"
|
||||
" <arg direction='out' type='as' />"
|
||||
" <arg direction='out' type='ao' />"
|
||||
" <arg direction='out' type='ag' />"
|
||||
" </method>"
|
||||
" <method name='TestPrimitiveTypes'>"
|
||||
" <arg direction='in' type='y' name='val_byte' />"
|
||||
" <arg direction='in' type='b' name='val_boolean' />"
|
||||
" <arg direction='in' type='n' name='val_int16' />"
|
||||
" <arg direction='in' type='q' name='val_uint16' />"
|
||||
" <arg direction='in' type='i' name='val_int32' />"
|
||||
" <arg direction='in' type='u' name='val_uint32' />"
|
||||
" <arg direction='in' type='x' name='val_int64' />"
|
||||
" <arg direction='in' type='t' name='val_uint64' />"
|
||||
" <arg direction='in' type='d' name='val_double' />"
|
||||
" <arg direction='in' type='s' name='val_string' />"
|
||||
" <arg direction='in' type='o' name='val_objpath' />"
|
||||
" <arg direction='in' type='g' name='val_signature' />"
|
||||
" <arg direction='out' type='y' />"
|
||||
" <arg direction='out' type='b' />"
|
||||
" <arg direction='out' type='n' />"
|
||||
" <arg direction='out' type='q' />"
|
||||
" <arg direction='out' type='i' />"
|
||||
" <arg direction='out' type='u' />"
|
||||
" <arg direction='out' type='x' />"
|
||||
" <arg direction='out' type='t' />"
|
||||
" <arg direction='out' type='d' />"
|
||||
" <arg direction='out' type='s' />"
|
||||
" <arg direction='out' type='o' />"
|
||||
" <arg direction='out' type='g' />"
|
||||
" </method>"
|
||||
" <method name='EmitSignal'>"
|
||||
" <arg direction='in' type='s' name='str1' />"
|
||||
" <arg direction='in' type='o' name='objpath1' />"
|
||||
" </method>"
|
||||
" <method name='TestArrayOfPrimitiveTypes'>"
|
||||
" <arg direction='in' type='ay' name='val_byte' />"
|
||||
" <arg direction='in' type='ab' name='val_boolean' />"
|
||||
" <arg direction='in' type='an' name='val_int16' />"
|
||||
" <arg direction='in' type='aq' name='val_uint16' />"
|
||||
" <arg direction='in' type='ai' name='val_int32' />"
|
||||
" <arg direction='in' type='au' name='val_uint32' />"
|
||||
" <arg direction='in' type='ax' name='val_int64' />"
|
||||
" <arg direction='in' type='at' name='val_uint64' />"
|
||||
" <arg direction='in' type='ad' name='val_double' />"
|
||||
" <arg direction='out' type='ay' />"
|
||||
" <arg direction='out' type='ab' />"
|
||||
" <arg direction='out' type='an' />"
|
||||
" <arg direction='out' type='aq' />"
|
||||
" <arg direction='out' type='ai' />"
|
||||
" <arg direction='out' type='au' />"
|
||||
" <arg direction='out' type='ax' />"
|
||||
" <arg direction='out' type='at' />"
|
||||
" <arg direction='out' type='ad' />"
|
||||
" </method>"
|
||||
" <method name='FrobSetProperty'>"
|
||||
" <arg direction='in' type='s' name='prop_name' />"
|
||||
" <arg direction='in' type='v' name='prop_value' />"
|
||||
" </method>"
|
||||
" <signal name='TestSignal'>"
|
||||
" <arg type='s' name='str1' />"
|
||||
" <arg type='o' name='objpath1' />"
|
||||
" <arg type='v' name='variant1' />"
|
||||
" </signal>"
|
||||
" <method name='TestComplexArrays'>"
|
||||
" <arg direction='in' type='a(ii)' name='aii' />"
|
||||
" <arg direction='in' type='aa(ii)' name='aaii' />"
|
||||
" <arg direction='in' type='aas' name='aas' />"
|
||||
" <arg direction='in' type='aa{ss}' name='ahashes' />"
|
||||
" <arg direction='in' type='aay' name='aay' />"
|
||||
" <arg direction='in' type='av' name='av' />"
|
||||
" <arg direction='in' type='aav' name='aav' />"
|
||||
" <arg direction='out' type='a(ii)' />"
|
||||
" <arg direction='out' type='aa(ii)' />"
|
||||
" <arg direction='out' type='aas' />"
|
||||
" <arg direction='out' type='aa{ss}' />"
|
||||
" <arg direction='out' type='aay' />"
|
||||
" <arg direction='out' type='av' />"
|
||||
" <arg direction='out' type='aav' />"
|
||||
" </method>"
|
||||
" <method name='TestVariant'>"
|
||||
" <arg direction='in' type='v' name='v' />"
|
||||
" <arg direction='in' type='b' name='modify' />"
|
||||
" <arg direction='out' type='v' />"
|
||||
" </method>"
|
||||
" <method name='FrobInvalidateProperty'>"
|
||||
" <arg direction='in' type='s' name='new_value' />"
|
||||
" </method>"
|
||||
" <method name='HelloWorld'>"
|
||||
" <arg direction='in' type='s' name='hello_message' />"
|
||||
" <arg direction='out' type='s' />"
|
||||
" </method>"
|
||||
" <method name='PairReturn'>"
|
||||
" <arg direction='out' type='s' />"
|
||||
" <arg direction='out' type='u' />"
|
||||
" </method>"
|
||||
" <method name='TestStructureTypes'>"
|
||||
" <arg direction='in' type='(ii)' name='s1' />"
|
||||
" <arg direction='in' type='(s(ii)aya{ss})' name='s2' />"
|
||||
" <arg direction='out' type='(ii)' />"
|
||||
" <arg direction='out' type='(s(ii)aya{ss})' />"
|
||||
" </method>"
|
||||
" <method name='EmitSignal2'>"
|
||||
" </method>"
|
||||
" <method name='DoubleHelloWorld'>"
|
||||
" <arg direction='in' type='s' name='hello1' />"
|
||||
" <arg direction='in' type='s' name='hello2' />"
|
||||
" <arg direction='out' type='s' />"
|
||||
" <arg direction='out' type='s' />"
|
||||
" </method>"
|
||||
" <method name='Sleep'>"
|
||||
" <arg direction='in' type='i' name='msec' />"
|
||||
" </method>"
|
||||
" <method name='TestHashTables'>"
|
||||
" <arg direction='in' type='a{yy}' name='hyy' />"
|
||||
" <arg direction='in' type='a{bb}' name='hbb' />"
|
||||
" <arg direction='in' type='a{nn}' name='hnn' />"
|
||||
" <arg direction='in' type='a{qq}' name='hqq' />"
|
||||
" <arg direction='in' type='a{ii}' name='hii' />"
|
||||
" <arg direction='in' type='a{uu}' name='huu' />"
|
||||
" <arg direction='in' type='a{xx}' name='hxx' />"
|
||||
" <arg direction='in' type='a{tt}' name='htt' />"
|
||||
" <arg direction='in' type='a{dd}' name='hdd' />"
|
||||
" <arg direction='in' type='a{ss}' name='hss' />"
|
||||
" <arg direction='in' type='a{oo}' name='hoo' />"
|
||||
" <arg direction='in' type='a{gg}' name='hgg' />"
|
||||
" <arg direction='out' type='a{yy}' />"
|
||||
" <arg direction='out' type='a{bb}' />"
|
||||
" <arg direction='out' type='a{nn}' />"
|
||||
" <arg direction='out' type='a{qq}' />"
|
||||
" <arg direction='out' type='a{ii}' />"
|
||||
" <arg direction='out' type='a{uu}' />"
|
||||
" <arg direction='out' type='a{xx}' />"
|
||||
" <arg direction='out' type='a{tt}' />"
|
||||
" <arg direction='out' type='a{dd}' />"
|
||||
" <arg direction='out' type='a{ss}' />"
|
||||
" <arg direction='out' type='a{oo}' />"
|
||||
" <arg direction='out' type='a{gg}' />"
|
||||
" </method>"
|
||||
" <signal name='TestSignal2'>"
|
||||
" <arg type='i' name='int1' />"
|
||||
" </signal>"
|
||||
" <method name='TestComplexHashTables'>"
|
||||
" <arg direction='in' type='a{s(ii)}' name='h_str_to_pair' />"
|
||||
" <arg direction='in' type='a{sv}' name='h_str_to_variant' />"
|
||||
" <arg direction='in' type='a{sav}' name='h_str_to_av' />"
|
||||
" <arg direction='in' type='a{saav}' name='h_str_to_aav' />"
|
||||
" <arg direction='in' type='a{sa(ii)}' name='h_str_to_array_of_pairs' />"
|
||||
" <arg direction='in' type='a{sa{ss}}' name='hash_of_hashes' />"
|
||||
" <arg direction='out' type='a{s(ii)}' />"
|
||||
" <arg direction='out' type='a{sv}' />"
|
||||
" <arg direction='out' type='a{sav}' />"
|
||||
" <arg direction='out' type='a{saav}' />"
|
||||
" <arg direction='out' type='a{sa(ii)}' />"
|
||||
" <arg direction='out' type='a{sa{ss}}' />"
|
||||
" </method>"
|
||||
" <property type='y' name='y' access='readwrite' />"
|
||||
" <property type='b' name='b' access='readwrite' />"
|
||||
" <property type='n' name='n' access='readwrite' />"
|
||||
" <property type='q' name='q' access='readwrite' />"
|
||||
" <property type='i' name='i' access='readwrite' />"
|
||||
" <property type='u' name='u' access='readwrite' />"
|
||||
" <property type='x' name='x' access='readwrite' />"
|
||||
" <property type='t' name='t' access='readwrite' />"
|
||||
" <property type='d' name='d' access='readwrite' />"
|
||||
" <property type='s' name='s' access='readwrite' />"
|
||||
" <property type='o' name='o' access='readwrite' />"
|
||||
" <property type='ay' name='ay' access='readwrite' />"
|
||||
" <property type='ab' name='ab' access='readwrite' />"
|
||||
" <property type='an' name='an' access='readwrite' />"
|
||||
" <property type='aq' name='aq' access='readwrite' />"
|
||||
" <property type='ai' name='ai' access='readwrite' />"
|
||||
" <property type='au' name='au' access='readwrite' />"
|
||||
" <property type='ax' name='ax' access='readwrite' />"
|
||||
" <property type='at' name='at' access='readwrite' />"
|
||||
" <property type='ad' name='ad' access='readwrite' />"
|
||||
" <property type='as' name='as' access='readwrite' />"
|
||||
" <property type='ao' name='ao' access='readwrite' />"
|
||||
" <property type='s' name='foo' access='readwrite' />"
|
||||
" <property type='s' name='PropertyThatWillBeInvalidated' access='readwrite' />"
|
||||
" </interface>"
|
||||
"</node>";
|
||||
|
||||
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;
|
||||
}
|
||||
|
||||
|
145
server.c
Executable file
145
server.c
Executable file
@ -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;
|
||||
}
|
48
server.h
Executable file
48
server.h
Executable file
@ -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 <dbus/dbus-glib-bindings.h>
|
||||
|
||||
/* 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 */
|
12
servicename-infos.xml
Executable file
12
servicename-infos.xml
Executable file
@ -0,0 +1,12 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
|
||||
<node name="/org/gnome/ServiceName">
|
||||
<interface name="org.gnome.ServiceName">
|
||||
<annotation name="org.freedesktop.DBus.GLib.CSymbol" value="server"/>
|
||||
<method name="EchoString">
|
||||
<arg type="s" name="original" direction="in" />
|
||||
<arg type="s" name="echo" direction="out" />
|
||||
</method>
|
||||
<!-- Add more methods/signals if you want -->
|
||||
</interface>
|
||||
</node>
|
Loading…
x
Reference in New Issue
Block a user