efl/src/lib/eldbus/eldbus_model_object.c

293 lines
8.1 KiB
C

#ifdef HAVE_CONFIG_H
# include <config.h>
#endif
#include "eldbus_model_object_private.h"
#include "eldbus_model_private.h"
#include <Ecore.h>
#include <Eina.h>
#define MY_CLASS ELDBUS_MODEL_OBJECT_CLASS
#define MY_CLASS_NAME "Eldbus_Model_Object"
static void _eldbus_model_object_introspect_cb(void *, const Eldbus_Message *, Eldbus_Pending *);
static void _eldbus_model_object_create_children(Eldbus_Model_Object_Data *, Eldbus_Object *, Eina_List *);
static Efl_Object*
_eldbus_model_object_efl_object_constructor(Eo *obj, Eldbus_Model_Object_Data *pd)
{
obj = efl_constructor(efl_super(obj, MY_CLASS));
pd->obj = obj;
return obj;
}
static void
_eldbus_model_object_bus_set(Eo *obj EINA_UNUSED,
Eldbus_Model_Object_Data *pd,
const char *bus)
{
pd->bus = eina_stringshare_add(bus);
}
static void
_eldbus_model_object_path_set(Eo *obj EINA_UNUSED,
Eldbus_Model_Object_Data *pd,
const char *path)
{
pd->path = eina_stringshare_add(path);
}
static Efl_Object*
_eldbus_model_object_efl_object_finalize(Eo *obj, Eldbus_Model_Object_Data *pd)
{
if (!pd->bus || !pd->path)
return NULL;
return efl_finalize(efl_super(obj, MY_CLASS));
}
static void
_eldbus_model_object_efl_object_invalidate(Eo *obj, Eldbus_Model_Object_Data *pd)
{
Eldbus_Pending *pending;
Eldbus_Object *object;
Eo *child;
EINA_LIST_FREE(pd->childrens, child)
efl_unref(child);
EINA_LIST_FREE(pd->pendings, pending)
eldbus_pending_cancel(pending);
EINA_LIST_FREE(pd->objects, object)
eldbus_object_unref(object);
if (pd->introspection)
{
eldbus_introspection_node_free(pd->introspection);
pd->introspection = NULL;
}
efl_invalidate(efl_super(obj, MY_CLASS));
}
static void
_eldbus_model_object_efl_object_destructor(Eo *obj, Eldbus_Model_Object_Data *pd)
{
eina_stringshare_del(pd->bus);
eina_stringshare_del(pd->path);
efl_destructor(efl_super(obj, MY_CLASS));
}
static Eina_Bool
_eldbus_model_object_introspect(const Eo *obj,
Eldbus_Model_Object_Data *pd,
const char *bus,
const char *path)
{
Eldbus_Pending *pending;
Eldbus_Object *object;
DBG("(%p) Introspecting: bus = %s, path = %s", pd->obj, bus, path);
object = eldbus_object_get(eldbus_model_connection_get(obj),
bus, path);
if (!object)
{
ERR("(%p): Cannot get object: bus=%s, path=%s", pd->obj, bus, path);
return EINA_FALSE;
}
pd->objects = eina_list_append(pd->objects, object);
// TODO: Register for interface added/removed event
pending = eldbus_object_introspect(object, &_eldbus_model_object_introspect_cb, pd);
eldbus_pending_data_set(pending, "object", object);
pd->pendings = eina_list_append(pd->pendings, pending);
return EINA_TRUE;
}
static Eina_Future *
_eldbus_model_object_efl_model_children_slice_get(Eo *obj EINA_UNUSED,
Eldbus_Model_Object_Data *pd,
unsigned start,
unsigned count)
{
Eldbus_Children_Slice_Promise *slice;
Eina_Promise *p;
if (pd->is_listed)
{
Eina_Value v;
v = efl_model_list_value_get(pd->childrens, start, count);
return efl_loop_future_resolved(obj, v);
}
p = efl_loop_promise_new(obj);
slice = calloc(1, sizeof(struct _Eldbus_Children_Slice_Promise));
slice->p = p;
slice->start = start;
slice->count = count;
pd->requests = eina_list_prepend(pd->requests, slice);
if (!pd->pendings)
_eldbus_model_object_introspect(obj, pd, pd->bus, pd->path);
return efl_future_then(obj, eina_future_new(p));;
}
static unsigned int
_eldbus_model_object_efl_model_children_count_get(const Eo *obj EINA_UNUSED,
Eldbus_Model_Object_Data *pd)
{
if (!pd->is_listed && !pd->pendings)
_eldbus_model_object_introspect(obj, pd, pd->bus, pd->path);
return eina_list_count(pd->childrens);
}
static const char *
_eldbus_model_object_bus_get(const Eo *obj EINA_UNUSED, Eldbus_Model_Object_Data *pd)
{
return pd->bus;
}
static const char *
_eldbus_model_object_path_get(const Eo *obj EINA_UNUSED, Eldbus_Model_Object_Data *pd)
{
return pd->path;
}
static char *
_eldbus_model_object_concatenate_path(const char *root_path,
const char *relative_path)
{
Eina_Strbuf *buffer;
const char *format = (!eina_streq(root_path, "/")) ? "%s/%s" : "%s%s";
char *absolute_path = NULL;
buffer = eina_strbuf_new();
eina_strbuf_append_printf(buffer, format, root_path, relative_path);
absolute_path = eina_strbuf_string_steal(buffer);
eina_strbuf_free(buffer);
return absolute_path;
}
static void
_eldbus_model_object_introspect_nodes(Eldbus_Model_Object_Data *pd,
const char *current_path,
Eina_List *nodes)
{
Eldbus_Introspection_Node *node;
Eina_List *it;
EINA_LIST_FOREACH(nodes, it, node)
{
const char *relative_path;
char *absolute_path;
relative_path = node->name;
if (!relative_path) continue;
absolute_path = _eldbus_model_object_concatenate_path(current_path, relative_path);
if (!absolute_path) continue;
_eldbus_model_object_introspect(pd->obj, pd, pd->bus, absolute_path);
free(absolute_path);
}
}
static void
_eldbus_model_object_create_children(Eldbus_Model_Object_Data *pd, Eldbus_Object *object, Eina_List *interfaces)
{
Eldbus_Introspection_Interface *interface;
const char *current_path;
Eina_List *l;
current_path = eldbus_object_path_get(object);
if (!current_path) return ;
EINA_LIST_FOREACH(interfaces, l, interface)
{
Eo *child;
DBG("(%p) Creating child: bus = %s, path = %s, interface = %s",
pd->obj, pd->bus, current_path, interface->name);
// TODO: increment reference to keep 'interface' in memory
child = efl_add_ref(ELDBUS_MODEL_PROXY_CLASS, pd->obj,
eldbus_model_proxy_object_set(efl_added, object),
eldbus_model_proxy_interface_set(efl_added, interface));
if (child) pd->childrens = eina_list_append(pd->childrens, child);
}
}
static void
_eldbus_model_object_introspect_cb(void *data,
const Eldbus_Message *msg,
Eldbus_Pending *pending)
{
Eldbus_Model_Object_Data *pd = (Eldbus_Model_Object_Data*)data;
Eldbus_Children_Slice_Promise* slice;
Eldbus_Object *object;
const char *error_name;
const char *error_text;
const char *xml = NULL;
const char *current_path;
pd->pendings = eina_list_remove(pd->pendings, pending);
object = eldbus_pending_data_get(pending, "object");
if (eldbus_message_error_get(msg, &error_name, &error_text))
{
ERR("%s: %s", error_name, error_text);
//efl_model_error_notify(pd->obj);
return;
}
if (!eldbus_message_arguments_get(msg, "s", &xml))
{
ERR("Error getting arguments.");
return;
}
if (!xml)
{
ERR("No XML.");
return ;
}
current_path = eldbus_object_path_get(object);
pd->introspection = eldbus_introspection_parse(xml);
DBG("(%p): introspect of bus = %s, path = %s =>\n%s", pd->obj, pd->bus, current_path, xml);
_eldbus_model_object_introspect_nodes(pd, current_path, pd->introspection->nodes);
_eldbus_model_object_create_children(pd, object, pd->introspection->interfaces);
if (eina_list_count(pd->pendings) != 0) return ;
efl_event_callback_call(pd->obj, EFL_MODEL_EVENT_CHILDREN_COUNT_CHANGED, NULL);
pd->is_listed = EINA_TRUE;
EINA_LIST_FREE(pd->requests, slice)
{
Eina_Value v;
v = efl_model_list_value_get(pd->childrens, slice->start, slice->count);
eina_promise_resolve(slice->p, v);
free(slice);
}
}
#include "eldbus_model_object.eo.c"