aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--libempathy/empathy-dispatcher.c176
1 files changed, 92 insertions, 84 deletions
diff --git a/libempathy/empathy-dispatcher.c b/libempathy/empathy-dispatcher.c
index 9b8c997bd..fe0981737 100644
--- a/libempathy/empathy-dispatcher.c
+++ b/libempathy/empathy-dispatcher.c
@@ -1415,37 +1415,85 @@ empathy_dispatcher_create_channel (EmpathyDispatcher *dispatcher,
G_OBJECT (request_data->dispatcher));
}
-typedef struct {
- gboolean mismatch;
- gboolean not_generic;
- const char *channel_namespace;
- GArray *properties;
-} PropertiesMatcherData;
-
-static void
-match_with_properties (gpointer key,
- gpointer val,
- gpointer user_data)
+static gboolean
+channel_class_matches (GValueArray *class,
+ const char *channel_type,
+ guint handle_type,
+ GArray *fixed_properties)
{
- int idx;
- PropertiesMatcherData *data = user_data;
+ GHashTable *fprops;
+ GValue *v;
+ const char *c_type;
+ guint h_type;
+ gboolean valid;
- if (data->mismatch)
- return;
+ v = g_value_array_get_nth (class, 0);
- /* discard generic properties, as we already checked them */
- if (!g_str_has_prefix ((char *) key, data->channel_namespace))
- return;
+ /* if the class doesn't match channel type discard it. */
+ fprops = g_value_get_boxed (v);
+ c_type = tp_asv_get_string (fprops, TP_IFACE_CHANNEL ".ChannelType");
+
+ if (tp_strdiff (channel_type, c_type))
+ return FALSE;
+
+ /* we have the right channel type, see if the handle type matches */
+ h_type = tp_asv_get_uint32 (fprops,
+ TP_IFACE_CHANNEL ".TargetHandleType", &valid);
- data->not_generic = TRUE;
+ if (!valid || handle_type != h_type)
+ return FALSE;
- for (idx = 0; idx < data->properties->len; idx++)
+ if (fixed_properties != NULL)
{
- /* if |key| exists in the properties, it's fine */
- if (!tp_strdiff ((char *) key, g_array_index (data->properties,
- char *, idx)))
- return;
+ gpointer h_key, h_val;
+ int idx;
+ GHashTableIter iter;
+ gboolean found;
+
+ g_hash_table_iter_init (&iter, fprops);
+
+ while (g_hash_table_iter_next (&iter, &h_key, &h_val))
+ {
+ /* discard ChannelType and TargetHandleType, as we already
+ * checked them.
+ */
+ if (!tp_strdiff ((char *) h_key, TP_IFACE_CHANNEL ".ChannelType") ||
+ !tp_strdiff
+ ((char *) h_key, TP_IFACE_CHANNEL ".TargetHandleType"))
+ continue;
+
+ found = FALSE;
+
+ for (idx = 0; idx < fixed_properties->len; idx++)
+ {
+ /* if |key| doesn't exist in |fixed_properties|, discard
+ * the class.
+ */
+ if (!tp_strdiff
+ ((char *) h_key,
+ g_array_index (fixed_properties, char *, idx)))
+ {
+ found = TRUE;
+ /* exit the for() loop */
+ break;
+ }
+ }
+
+ if (!found)
+ return FALSE;
+ }
+ }
+ else
+ {
+ /* if no fixed_properties are specified, discard the classes
+ * with some fixed properties other than the two we already
+ * checked.
+ */
+ if (g_hash_table_size (fprops) > 2)
+ return FALSE;
}
+
+ return TRUE;
}
static GList *
@@ -1457,13 +1505,8 @@ empathy_dispatcher_find_channel_classes (EmpathyDispatcher *dispatcher,
{
EmpathyDispatcherPriv *priv = GET_PRIV (dispatcher);
GValueArray *class;
- GHashTable *fixed_props;
- GValue *val;
GPtrArray *classes;
GList *matching_classes;
- const gchar *c_type;
- guint32 h_type;
- gboolean valid, found;
int i;
ConnectionData *cd;
@@ -1479,59 +1522,20 @@ empathy_dispatcher_find_channel_classes (EmpathyDispatcher *dispatcher,
if (classes == NULL)
return NULL;
- found = TRUE;
matching_classes = NULL;
for (i = 0; i < classes->len; i++)
{
class = g_ptr_array_index (classes, i);
- val = g_value_array_get_nth (class, 0);
-
- /* if the class doesn't match channel type discard it. */
- fixed_props = g_value_get_boxed (val);
- c_type = tp_asv_get_string (fixed_props,
- TP_IFACE_CHANNEL ".ChannelType");
-
- if (tp_strdiff (channel_type, c_type))
- continue;
-
- /* we have the right channel type, see if the handle type matches */
- h_type = tp_asv_get_uint32 (fixed_props,
- TP_IFACE_CHANNEL ".TargetHandleType", &valid);
- if (!valid || handle_type != h_type)
+ if (!channel_class_matches
+ (class, channel_type, handle_type, fixed_properties))
continue;
-
- /* now we should ensure that the fixed props that we specified
- * are the only values inside the hash table.
- */
- if (fixed_properties != NULL)
- {
- PropertiesMatcherData *data;
-
- data = g_slice_new0 (PropertiesMatcherData);
- data->mismatch = FALSE;
- data->not_generic = FALSE;
- data->channel_namespace = channel_type;
- data->properties = fixed_properties;
- g_hash_table_foreach (fixed_props, match_with_properties, data);
-
- found = (!data->mismatch && data->not_generic);
-
- g_slice_free (PropertiesMatcherData, data);
-
- if (!found)
- continue;
- }
-
- if (found)
- matching_classes = g_list_prepend (matching_classes, class);
+
+ matching_classes = g_list_prepend (matching_classes, class);
}
- if (matching_classes != NULL)
- return g_list_reverse (matching_classes);
- else
- return NULL;
+ return matching_classes;
}
static gboolean
@@ -1593,7 +1597,7 @@ setup_varargs (va_list var_args,
while (name != NULL)
{
- name_full = g_strconcat (channel_namespace, ".", name, NULL);
+ name_full = g_strdup (name);
properties = g_array_append_val (properties, name_full);
name = va_arg (var_args, char *);
}
@@ -1613,16 +1617,17 @@ setup_varargs (va_list var_args,
* Returns all the channel classes that a client can request for the connection
* @connection, of the type identified by @channel_type, @handle_type and the
* fixed properties list.
- * If @first_property_name is %NULL, no additional fixed properties will be
- * specified and the function will return all the requestable classes for
- * the specified channel type and handle type.
+ * If @first_property_name is %NULL, only the classes with no other fixed
+ * properties than ChannelType and TargetHandleType will be returned.
* Note that this function may return %NULL without performing any lookup if
* @connection is not ready. To ensure that @connection is always ready,
* use the empathy_dispatcher_find_requestable_channel_classes_async() variant.
*
* Return value: a #GList of #GValueArray objects, where the first element in
* the array is a #GHashTable of the fixed properties, and the second is
- * a #GStrv of the allowed properties for the class.
+ * a #GStrv of the allowed properties for the class. The list should be free'd
+ * with g_list_free() when done, but the objects inside the list are owned
+ * by the #EmpathyDispatcher and must not be modified.
*/
GList *
empathy_dispatcher_find_requestable_channel_classes
@@ -1656,14 +1661,17 @@ empathy_dispatcher_find_requestable_channel_classes
retval = empathy_dispatcher_find_channel_classes (dispatcher, connection,
channel_type, handle_type, properties);
- /* free the properties array */
- for (idx = 0; idx < properties->len ; idx++)
+ if (properties != NULL)
{
- str = g_array_index (properties, char *, idx);
- g_free (str);
- }
+ /* free the properties array */
+ for (idx = 0; idx < properties->len ; idx++)
+ {
+ str = g_array_index (properties, char *, idx);
+ g_free (str);
+ }
- g_array_free (properties, TRUE);
+ g_array_free (properties, TRUE);
+ }
return retval;
}