592 lines
12 KiB
C
592 lines
12 KiB
C
#include "xed-message.h"
|
|
#include "xed-message-type.h"
|
|
|
|
#include <string.h>
|
|
#include <gobject/gvaluecollector.h>
|
|
|
|
/**
|
|
* SECTION:xed-message
|
|
* @short_description: message bus message object
|
|
* @include: xed/xed-message.h
|
|
*
|
|
* Communication on a #XedMessageBus is done through messages. Messages are
|
|
* sent over the bus and received by connecting callbacks on the message bus.
|
|
* A #XedMessage is an instantiation of a #XedMessageType, containing
|
|
* values for the arguments as specified in the message type.
|
|
*
|
|
* A message can be seen as a method call, or signal emission depending on
|
|
* who is the sender and who is the receiver. There is no explicit distinction
|
|
* between methods and signals.
|
|
*/
|
|
#define XED_MESSAGE_GET_PRIVATE(object)(G_TYPE_INSTANCE_GET_PRIVATE((object), XED_TYPE_MESSAGE, XedMessagePrivate))
|
|
|
|
enum {
|
|
PROP_0,
|
|
|
|
PROP_OBJECT_PATH,
|
|
PROP_METHOD,
|
|
PROP_TYPE
|
|
};
|
|
|
|
struct _XedMessagePrivate
|
|
{
|
|
XedMessageType *type;
|
|
gboolean valid;
|
|
|
|
GHashTable *values;
|
|
};
|
|
|
|
G_DEFINE_TYPE (XedMessage, xed_message, G_TYPE_OBJECT)
|
|
|
|
static void
|
|
xed_message_finalize (GObject *object)
|
|
{
|
|
XedMessage *message = XED_MESSAGE (object);
|
|
|
|
xed_message_type_unref (message->priv->type);
|
|
g_hash_table_destroy (message->priv->values);
|
|
|
|
G_OBJECT_CLASS (xed_message_parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
xed_message_get_property (GObject *object,
|
|
guint prop_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
XedMessage *msg = XED_MESSAGE (object);
|
|
|
|
switch (prop_id)
|
|
{
|
|
case PROP_OBJECT_PATH:
|
|
g_value_set_string (value, xed_message_type_get_object_path (msg->priv->type));
|
|
break;
|
|
case PROP_METHOD:
|
|
g_value_set_string (value, xed_message_type_get_method (msg->priv->type));
|
|
break;
|
|
case PROP_TYPE:
|
|
g_value_set_boxed (value, msg->priv->type);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
xed_message_set_property (GObject *object,
|
|
guint prop_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
XedMessage *msg = XED_MESSAGE (object);
|
|
|
|
switch (prop_id)
|
|
{
|
|
case PROP_TYPE:
|
|
msg->priv->type = XED_MESSAGE_TYPE (g_value_dup_boxed (value));
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static GValue *
|
|
add_value (XedMessage *message,
|
|
const gchar *key)
|
|
{
|
|
GValue *value;
|
|
GType type = xed_message_type_lookup (message->priv->type, key);
|
|
|
|
if (type == G_TYPE_INVALID)
|
|
return NULL;
|
|
|
|
value = g_new0 (GValue, 1);
|
|
g_value_init (value, type);
|
|
g_value_reset (value);
|
|
|
|
g_hash_table_insert (message->priv->values, g_strdup (key), value);
|
|
|
|
return value;
|
|
}
|
|
|
|
static void
|
|
xed_message_class_init (XedMessageClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS(klass);
|
|
|
|
object_class->finalize = xed_message_finalize;
|
|
object_class->get_property = xed_message_get_property;
|
|
object_class->set_property = xed_message_set_property;
|
|
|
|
/**
|
|
* XedMessage:object_path:
|
|
*
|
|
* The messages object path (e.g. /xed/object/path).
|
|
*
|
|
*/
|
|
g_object_class_install_property (object_class, PROP_OBJECT_PATH,
|
|
g_param_spec_string ("object-path",
|
|
"OBJECT_PATH",
|
|
"The message object path",
|
|
NULL,
|
|
G_PARAM_READABLE |
|
|
G_PARAM_STATIC_STRINGS));
|
|
|
|
/**
|
|
* XedMessage:method:
|
|
*
|
|
* The messages method.
|
|
*
|
|
*/
|
|
g_object_class_install_property (object_class, PROP_METHOD,
|
|
g_param_spec_string ("method",
|
|
"METHOD",
|
|
"The message method",
|
|
NULL,
|
|
G_PARAM_READABLE |
|
|
G_PARAM_STATIC_STRINGS));
|
|
|
|
/**
|
|
* XedMEssage:type:
|
|
*
|
|
* The message type.
|
|
*
|
|
*/
|
|
g_object_class_install_property (object_class, PROP_TYPE,
|
|
g_param_spec_boxed ("type",
|
|
"TYPE",
|
|
"The message type",
|
|
XED_TYPE_MESSAGE_TYPE,
|
|
G_PARAM_READWRITE |
|
|
G_PARAM_CONSTRUCT_ONLY |
|
|
G_PARAM_STATIC_STRINGS));
|
|
|
|
g_type_class_add_private (object_class, sizeof(XedMessagePrivate));
|
|
}
|
|
|
|
static void
|
|
destroy_value (GValue *value)
|
|
{
|
|
g_value_unset (value);
|
|
g_free (value);
|
|
}
|
|
|
|
static void
|
|
xed_message_init (XedMessage *self)
|
|
{
|
|
self->priv = XED_MESSAGE_GET_PRIVATE (self);
|
|
|
|
self->priv->values = g_hash_table_new_full (g_str_hash,
|
|
g_str_equal,
|
|
(GDestroyNotify)g_free,
|
|
(GDestroyNotify)destroy_value);
|
|
}
|
|
|
|
static gboolean
|
|
set_value_real (GValue *to,
|
|
const GValue *from)
|
|
{
|
|
GType from_type;
|
|
GType to_type;
|
|
|
|
from_type = G_VALUE_TYPE (from);
|
|
to_type = G_VALUE_TYPE (to);
|
|
|
|
if (!g_type_is_a (from_type, to_type))
|
|
{
|
|
if (!g_value_transform (from, to))
|
|
{
|
|
g_warning ("%s: Unable to make conversion from %s to %s",
|
|
G_STRLOC,
|
|
g_type_name (from_type),
|
|
g_type_name (to_type));
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
g_value_copy (from, to);
|
|
return TRUE;
|
|
}
|
|
|
|
inline static GValue *
|
|
value_lookup (XedMessage *message,
|
|
const gchar *key,
|
|
gboolean create)
|
|
{
|
|
GValue *ret = (GValue *)g_hash_table_lookup (message->priv->values, key);
|
|
|
|
if (!ret && create)
|
|
ret = add_value (message, key);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* xed_message_get_method:
|
|
* @message: the #XedMessage
|
|
*
|
|
* Get the message method.
|
|
*
|
|
* Return value: the message method
|
|
*
|
|
*/
|
|
const gchar *
|
|
xed_message_get_method (XedMessage *message)
|
|
{
|
|
g_return_val_if_fail (XED_IS_MESSAGE (message), NULL);
|
|
|
|
return xed_message_type_get_method (message->priv->type);
|
|
}
|
|
|
|
/**
|
|
* xed_message_get_object_path:
|
|
* @message: the #XedMessage
|
|
*
|
|
* Get the message object path.
|
|
*
|
|
* Return value: the message object path
|
|
*
|
|
*/
|
|
const gchar *
|
|
xed_message_get_object_path (XedMessage *message)
|
|
{
|
|
g_return_val_if_fail (XED_IS_MESSAGE (message), NULL);
|
|
|
|
return xed_message_type_get_object_path (message->priv->type);
|
|
}
|
|
|
|
/**
|
|
* xed_message_set:
|
|
* @message: the #XedMessage
|
|
* @...: a %NULL terminated variable list of key/value pairs
|
|
*
|
|
* Set values of message arguments. The supplied @var_args should contain
|
|
* pairs of keys and argument values.
|
|
*
|
|
*/
|
|
void
|
|
xed_message_set (XedMessage *message,
|
|
...)
|
|
{
|
|
va_list ap;
|
|
|
|
g_return_if_fail (XED_IS_MESSAGE (message));
|
|
|
|
va_start (ap, message);
|
|
xed_message_set_valist (message, ap);
|
|
va_end (ap);
|
|
}
|
|
|
|
/**
|
|
* xed_message_set_valist:
|
|
* @message: the #XedMessage
|
|
* @var_args: a %NULL terminated variable list of key/value pairs
|
|
*
|
|
* Set values of message arguments. The supplied @var_args should contain
|
|
* pairs of keys and argument values.
|
|
*
|
|
*/
|
|
void
|
|
xed_message_set_valist (XedMessage *message,
|
|
va_list var_args)
|
|
{
|
|
const gchar *key;
|
|
|
|
g_return_if_fail (XED_IS_MESSAGE (message));
|
|
|
|
while ((key = va_arg (var_args, const gchar *)) != NULL)
|
|
{
|
|
/* lookup the key */
|
|
GValue *container = value_lookup (message, key, TRUE);
|
|
GValue value = {0,};
|
|
gchar *error = NULL;
|
|
|
|
if (!container)
|
|
{
|
|
g_warning ("%s: Cannot set value for %s, does not exist",
|
|
G_STRLOC,
|
|
key);
|
|
|
|
/* skip value */
|
|
va_arg (var_args, gpointer);
|
|
continue;
|
|
}
|
|
|
|
g_value_init (&value, G_VALUE_TYPE (container));
|
|
G_VALUE_COLLECT (&value, var_args, 0, &error);
|
|
|
|
if (error)
|
|
{
|
|
g_warning ("%s: %s", G_STRLOC, error);
|
|
continue;
|
|
}
|
|
|
|
set_value_real (container, &value);
|
|
g_value_unset (&value);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* xed_message_set_value:
|
|
* @message: the #XedMessage
|
|
* @key: the argument key
|
|
* @value: (out): the argument value
|
|
*
|
|
* Set value of message argument @key to @value.
|
|
*
|
|
*/
|
|
void
|
|
xed_message_set_value (XedMessage *message,
|
|
const gchar *key,
|
|
GValue *value)
|
|
{
|
|
GValue *container;
|
|
g_return_if_fail (XED_IS_MESSAGE (message));
|
|
|
|
container = value_lookup (message, key, TRUE);
|
|
|
|
if (!container)
|
|
{
|
|
g_warning ("%s: Cannot set value for %s, does not exist",
|
|
G_STRLOC,
|
|
key);
|
|
return;
|
|
}
|
|
|
|
set_value_real (container, value);
|
|
}
|
|
|
|
/**
|
|
* xed_message_set_valuesv:
|
|
* @message: the #XedMessage
|
|
* @keys: (array length=n_values): keys to set values for
|
|
* @values: (array length=n_values): values to set
|
|
* @n_values: number of arguments to set values for
|
|
*
|
|
* Set message argument values.
|
|
*
|
|
*/
|
|
void
|
|
xed_message_set_valuesv (XedMessage *message,
|
|
const gchar **keys,
|
|
GValue *values,
|
|
gint n_values)
|
|
{
|
|
gint i;
|
|
|
|
g_return_if_fail (XED_IS_MESSAGE (message));
|
|
|
|
for (i = 0; i < n_values; i++)
|
|
{
|
|
xed_message_set_value (message, keys[i], &values[i]);
|
|
}
|
|
}
|
|
|
|
/* FIXME this is an issue for introspection */
|
|
/**
|
|
* xed_message_get:
|
|
* @message: the #XedMessage
|
|
* @...: a %NULL variable argument list of key/value container pairs
|
|
*
|
|
* Get values of message arguments. The supplied @var_args should contain
|
|
* pairs of keys and pointers to variables which are set to the argument
|
|
* value for the specified key.
|
|
*
|
|
*/
|
|
void
|
|
xed_message_get (XedMessage *message,
|
|
...)
|
|
{
|
|
va_list ap;
|
|
|
|
g_return_if_fail (XED_IS_MESSAGE (message));
|
|
|
|
va_start (ap, message);
|
|
xed_message_get_valist (message, ap);
|
|
va_end (ap);
|
|
}
|
|
|
|
/**
|
|
* xed_message_get_valist:
|
|
* @message: the #XedMessage
|
|
* @var_args: a %NULL variable argument list of key/value container pairs
|
|
*
|
|
* Get values of message arguments. The supplied @var_args should contain
|
|
* pairs of keys and pointers to variables which are set to the argument
|
|
* value for the specified key.
|
|
*
|
|
*/
|
|
void
|
|
xed_message_get_valist (XedMessage *message,
|
|
va_list var_args)
|
|
{
|
|
const gchar *key;
|
|
|
|
g_return_if_fail (XED_IS_MESSAGE (message));
|
|
|
|
while ((key = va_arg (var_args, const gchar *)) != NULL)
|
|
{
|
|
GValue *container;
|
|
GValue copy = {0,};
|
|
gchar *error = NULL;
|
|
|
|
container = value_lookup (message, key, FALSE);
|
|
|
|
if (!container)
|
|
{
|
|
/* skip value */
|
|
va_arg (var_args, gpointer);
|
|
continue;
|
|
}
|
|
|
|
/* copy the value here, to be sure it isn't tainted */
|
|
g_value_init (©, G_VALUE_TYPE (container));
|
|
g_value_copy (container, ©);
|
|
|
|
G_VALUE_LCOPY (©, var_args, 0, &error);
|
|
|
|
if (error)
|
|
{
|
|
g_warning ("%s: %s", G_STRLOC, error);
|
|
g_free (error);
|
|
|
|
/* purposely leak the value here, because it might
|
|
be in a bad state */
|
|
continue;
|
|
}
|
|
|
|
g_value_unset (©);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* xed_message_get_value:
|
|
* @message: the #XedMessage
|
|
* @key: the argument key
|
|
* @value: (out): value return container
|
|
*
|
|
* Get the value of a specific message argument. @value will be initialized
|
|
* with the correct type.
|
|
*
|
|
*/
|
|
void
|
|
xed_message_get_value (XedMessage *message,
|
|
const gchar *key,
|
|
GValue *value)
|
|
{
|
|
GValue *container;
|
|
|
|
g_return_if_fail (XED_IS_MESSAGE (message));
|
|
|
|
container = value_lookup (message, key, FALSE);
|
|
|
|
if (!container)
|
|
{
|
|
g_warning ("%s: Invalid key `%s'",
|
|
G_STRLOC,
|
|
key);
|
|
return;
|
|
}
|
|
|
|
g_value_init (value, G_VALUE_TYPE (container));
|
|
set_value_real (value, container);
|
|
}
|
|
|
|
/**
|
|
* xed_message_get_key_type:
|
|
* @message: the #XedMessage
|
|
* @key: the argument key
|
|
*
|
|
* Get the type of a message argument.
|
|
*
|
|
* Return value: the type of @key
|
|
*
|
|
*/
|
|
GType
|
|
xed_message_get_key_type (XedMessage *message,
|
|
const gchar *key)
|
|
{
|
|
g_return_val_if_fail (XED_IS_MESSAGE (message), G_TYPE_INVALID);
|
|
g_return_val_if_fail (message->priv->type != NULL, G_TYPE_INVALID);
|
|
|
|
return xed_message_type_lookup (message->priv->type, key);
|
|
}
|
|
|
|
/**
|
|
* xed_message_has_key:
|
|
* @message: the #XedMessage
|
|
* @key: the argument key
|
|
*
|
|
* Check whether the message has a specific key.
|
|
*
|
|
* Return value: %TRUE if @message has argument @key
|
|
*
|
|
*/
|
|
gboolean
|
|
xed_message_has_key (XedMessage *message,
|
|
const gchar *key)
|
|
{
|
|
g_return_val_if_fail (XED_IS_MESSAGE (message), FALSE);
|
|
|
|
return value_lookup (message, key, FALSE) != NULL;
|
|
}
|
|
|
|
typedef struct
|
|
{
|
|
XedMessage *message;
|
|
gboolean valid;
|
|
} ValidateInfo;
|
|
|
|
static void
|
|
validate_key (const gchar *key,
|
|
GType type,
|
|
gboolean required,
|
|
ValidateInfo *info)
|
|
{
|
|
GValue *value;
|
|
|
|
if (!info->valid || !required)
|
|
return;
|
|
|
|
value = value_lookup (info->message, key, FALSE);
|
|
|
|
if (!value)
|
|
info->valid = FALSE;
|
|
}
|
|
|
|
/**
|
|
* xed_message_validate:
|
|
* @message: the #XedMessage
|
|
*
|
|
* Validates the message arguments according to the message type.
|
|
*
|
|
* Return value: %TRUE if the message is valid
|
|
*
|
|
*/
|
|
gboolean
|
|
xed_message_validate (XedMessage *message)
|
|
{
|
|
ValidateInfo info = {message, TRUE};
|
|
|
|
g_return_val_if_fail (XED_IS_MESSAGE (message), FALSE);
|
|
g_return_val_if_fail (message->priv->type != NULL, FALSE);
|
|
|
|
if (!message->priv->valid)
|
|
{
|
|
xed_message_type_foreach (message->priv->type,
|
|
(XedMessageTypeForeach)validate_key,
|
|
&info);
|
|
|
|
message->priv->valid = info.valid;
|
|
}
|
|
|
|
return message->priv->valid;
|
|
}
|
|
|
|
// ex:ts=8:noet:
|