[gupnp-av/wip/didl-lite-fragments: 6/19] wip: some fixes.
- From: Krzesimir Nowak <krnowak src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [gupnp-av/wip/didl-lite-fragments: 6/19] wip: some fixes.
- Date: Tue, 23 Oct 2012 14:34:52 +0000 (UTC)
commit 8a48fb3f2d3942e291fc530b6b31485894b8b868
Author: Krzesimir Nowak <krnowak openismus com>
Date: Thu Oct 18 16:28:57 2012 +0200
wip: some fixes.
turned off the xsd validation - its fishy and slow.
Makefile.am | 3 +-
libgupnp-av/gupnp-didl-lite-object.c | 100 +++++++++++++++++++++++++---------
tests/Makefile.am | 2 +-
tests/fragments.c | 85 +++++++++++++++++++----------
4 files changed, 131 insertions(+), 59 deletions(-)
---
diff --git a/Makefile.am b/Makefile.am
index 92b6bde..f259863 100644
--- a/Makefile.am
+++ b/Makefile.am
@@ -1,4 +1,5 @@
-SUBDIRS = libgupnp-av tests doc vala data
+SUBDIRS = libgupnp-av tests doc vala
+DIST_SUBDIRS = $(SUBDIRS) data
pkgconfig_DATA= gupnp-av-1.0.pc
pkgconfigdir = $(libdir)/pkgconfig
diff --git a/libgupnp-av/gupnp-didl-lite-object.c b/libgupnp-av/gupnp-didl-lite-object.c
index 65729cf..345e86b 100644
--- a/libgupnp-av/gupnp-didl-lite-object.c
+++ b/libgupnp-av/gupnp-didl-lite-object.c
@@ -2135,14 +2135,20 @@ static gboolean
node_deep_equal (xmlNodePtr first,
xmlNodePtr second)
{
- GHashTable *first_attributes = g_hash_table_new (g_str_hash,
- g_str_equal);
+ GHashTable *first_attributes;
xmlAttrPtr attribute;
- gboolean equal = FALSE;
+ gboolean equal;
+
+ if (!first && !second)
+ return TRUE;
+ if (!first || !second)
+ return FALSE;
if (xmlStrcmp (first->name, second->name))
return FALSE;
+ equal = FALSE;
+ first_attributes = g_hash_table_new (g_str_hash, g_str_equal);
/* compare attributes */
for (attribute = first->properties;
attribute;
@@ -2225,7 +2231,14 @@ is_current_doc_part_of_this_doc (xmlDocPtr this_doc,
xmlDocPtr current_doc)
{
xmlNodePtr current_node = current_doc->children->children;
- xmlNodePtr this_node = find_node (this_doc->children, current_node);
+ xmlNodePtr this_node;
+
+ /* No current node means that we want to add new elements to
+ the document. */
+ if (!current_node)
+ return TRUE;
+
+ this_node = find_node (this_doc->children, current_node);
if (!this_node)
return FALSE;
@@ -2317,7 +2330,7 @@ typedef struct {
GHashTable* required_indep_props; /* string to indep prop */
} IndependentProperty;
-void
+static void
independent_property_free (IndependentProperty *indep)
{
if (indep) {
@@ -2327,7 +2340,7 @@ independent_property_free (IndependentProperty *indep)
}
}
-IndependentProperty *
+static IndependentProperty *
independent_property_new (gboolean required)
{
IndependentProperty *indep = g_slice_new (IndependentProperty);
@@ -2369,7 +2382,7 @@ add_dep_prop (IndependentProperty *indep,
g_hash_table_add (indep->required_dep_props, g_strdup (name));
}
-IndependentProperty *
+static IndependentProperty *
create_prop_with_required_dep_props (gboolean required,
gchar *dep_prop,
...)
@@ -2391,7 +2404,7 @@ create_prop_with_required_dep_props (gboolean required,
return indep;
}
-IndependentProperty *
+static IndependentProperty *
create_foreign_metadata_props (void)
{
IndependentProperty *fm = independent_property_new (FALSE);
@@ -2429,7 +2442,7 @@ get_required_properties (void)
(GDestroyNotify) independent_property_free);
insert_indep_prop (required_props,
- NULL,
+ "",
create_prop_with_required_dep_props
(FALSE,
"id",
@@ -2524,7 +2537,7 @@ is_required (const xmlChar *changed_element,
if (changed_element) {
IndependentProperty *toplevel_prop = g_hash_table_lookup
(required_props,
- NULL);
+ "");
IndependentProperty *this_prop = g_hash_table_lookup
(required_props,
(gpointer) changed_element);
@@ -2637,7 +2650,7 @@ typedef struct {
xmlSchemaValidCtxtPtr valid_context;
} XSDValidateData;
-void
+static void
xsd_validate_data_free (XSDValidateData *data)
{
if (!data)
@@ -2653,13 +2666,15 @@ xsd_validate_data_free (XSDValidateData *data)
g_slice_free (XSDValidateData, data);
}
-XSDValidateData *
+static XSDValidateData *
xsd_validate_data_new (const gchar *xsd_file)
{
XSDValidateData *data = g_slice_new0 (XSDValidateData);
gboolean failed = TRUE;
- data->schema_doc = xmlReadFile (xsd_file, NULL, 0);
+ return data;
+
+ data->schema_doc = xmlReadFile (xsd_file, NULL, XML_PARSE_NONET);
if (!data->schema_doc) {
/* the schema cannot be loaded or is not well-formed */
goto out;
@@ -2693,6 +2708,14 @@ static gboolean
validate_temporary_modification (xmlDocPtr modified_doc,
XSDValidateData *vdata)
{
+ xmlChar *dump = NULL;
+
+ xmlDocDumpMemory (modified_doc, &dump, NULL);
+ g_debug ("Modified doc dump:\n%s", dump);
+ xmlFree (dump);
+
+ return TRUE;
+
return (xmlSchemaValidateDoc (vdata->valid_context, modified_doc) == 0);
}
@@ -2725,16 +2748,18 @@ apply_temporary_addition (xmlDocPtr modified_doc,
xmlNodePtr new_node,
XSDValidateData *vdata)
{
- xmlNodePtr mod_sibling = find_node (modified_doc->children,
- sibling);
+ xmlNodePtr mod_sibling;
+
+ if (sibling->doc == modified_doc)
+ mod_sibling = sibling;
+ else
+ mod_sibling = find_node (modified_doc->children, sibling);
- if (!mod_sibling || (xmlAddSibling (mod_sibling, new_node) != NULL)) {
+ if (!mod_sibling || !xmlAddSibling (mod_sibling, new_node))
return GUPNP_DIDL_LITE_FRAGMENT_RESULT_UNKNOWN_ERROR;
- }
- if (validate_temporary_modification (modified_doc, vdata)) {
+ if (!validate_temporary_modification (modified_doc, vdata))
return GUPNP_DIDL_LITE_FRAGMENT_RESULT_NEW_INVALID;
- }
return GUPNP_DIDL_LITE_FRAGMENT_RESULT_OK;
}
@@ -2752,7 +2777,7 @@ apply_temporary_removal (xmlDocPtr modified_doc,
xmlUnlinkNode (mod_cur_node);
xmlFreeNode (mod_cur_node);
- if (validate_temporary_modification (modified_doc, vdata)) {
+ if (!validate_temporary_modification (modified_doc, vdata)) {
/* not sure if this is correct */
return GUPNP_DIDL_LITE_FRAGMENT_RESULT_REQUIRED_TAG;
}
@@ -2767,7 +2792,7 @@ new_doc_is_valid_modification (xmlDocPtr modified_doc,
XSDValidateData *vdata) {
xmlNodePtr current_node;
xmlNodePtr new_node;
- xmlNodePtr last_sibling;
+ xmlNodePtr last_sibling = NULL;
for (current_node = current_doc->children->children,
new_node = new_doc->children->children;
@@ -2778,6 +2803,7 @@ new_doc_is_valid_modification (xmlDocPtr modified_doc,
if (node_deep_equal (current_node, new_node)) {
/* this is just a context, skip the checks. */
last_sibling = current_node;
+ new_node = new_node->next;
continue;
}
if (xmlStrcmp (current_node->name, new_node->name)) {
@@ -2786,6 +2812,12 @@ new_doc_is_valid_modification (xmlDocPtr modified_doc,
if (is_any_change_read_only (current_node, new_node))
return GUPNP_DIDL_LITE_FRAGMENT_RESULT_READONLY_TAG;
last_sibling = new_node;
+ /* we can't put this line into for instruction,
+ * because new_node is unlinked from its document and
+ * put into another one in
+ * apply_temporary_modification. We have to get its
+ * sibling before that happens.
+ */
new_node = new_node->next;
result = apply_temporary_modification (modified_doc,
current_node,
@@ -2794,6 +2826,10 @@ new_doc_is_valid_modification (xmlDocPtr modified_doc,
if (result != GUPNP_DIDL_LITE_FRAGMENT_RESULT_OK)
return result;
}
+ if (!last_sibling &&
+ modified_doc->children &&
+ modified_doc->children->children)
+ last_sibling = modified_doc->children->children->children;
/* If there are some more nodes in current fragment then it
* means they are going to be removed. Check against required
* or read-only tag removal.
@@ -2826,8 +2862,9 @@ new_doc_is_valid_modification (xmlDocPtr modified_doc,
* they are going to be added. Check against read-only tags
* addition and general sanity check.
*/
- for (; new_node; new_node = new_node->next) {
+ while (new_node) {
GUPnPDIDLLiteFragmentResult result;
+ xmlNodePtr temp_node;
if (is_read_only ((gchar *) new_node->name, NULL)) {
return GUPNP_DIDL_LITE_FRAGMENT_RESULT_READONLY_TAG;
@@ -2836,9 +2873,11 @@ new_doc_is_valid_modification (xmlDocPtr modified_doc,
* has all required properties. Maybe XSD check could
* do that for us.
*/
+ temp_node = new_node;
+ new_node = new_node->next;
result = apply_temporary_addition (modified_doc,
last_sibling,
- new_node,
+ temp_node,
vdata);
if (result != GUPNP_DIDL_LITE_FRAGMENT_RESULT_OK)
return result;
@@ -2850,11 +2889,18 @@ new_doc_is_valid_modification (xmlDocPtr modified_doc,
static gchar *
fix_fragment (const gchar *fragment)
{
- return g_strdup_printf ("<DIDLLiteFragment>%s</DIDLLiteFragment>",
- fragment);
+ return g_strdup_printf
+ ("<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"
+ "<DIDLLiteFragment\n"
+ "xmlns:dc=\"http://purl.org/dc/elements/1.1/\"\n"
+ "xmlns=\"urn:schemas-upnp-org:metadata-1-0/DIDL-Lite/\"\n"
+ "xmlns:upnp=\"urn:schemas-upnp-org:metadata-1-0/upnp/\"\n"
+ "xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\"\n"
+ ">%s</DIDLLiteFragment>\n",
+ fragment);
}
-GUPnPDIDLLiteFragmentResult
+static GUPnPDIDLLiteFragmentResult
check_fragments (xmlDocPtr this_doc,
xmlDocPtr modified_doc,
const gchar *current_fragment,
@@ -2922,7 +2968,7 @@ get_data_dir (void)
return datadir;
}
-XSDValidateData *
+static XSDValidateData *
get_xsd_validate_data (void)
{
gchar *path = g_strdup_printf
diff --git a/tests/Makefile.am b/tests/Makefile.am
index 9831b28..9ae71cf 100644
--- a/tests/Makefile.am
+++ b/tests/Makefile.am
@@ -18,6 +18,6 @@ check_search_CFLAGS = $(common_cflags)
fragments_SOURCES = fragments.c
fragments_LDADD = $(top_builddir)/libgupnp-av/libgupnp-av-1.0.la \
$(LIBGUPNP_LIBS)
-fragments_CFLAGS = $(common_cflags) -DTOP_SRCDIR="\"$(top_srcdir)\""
+fragments_CFLAGS = $(common_cflags) -DABS_TOP_SRCDIR="\"$(abs_top_srcdir)\""
TESTS = $(check_PROGRAMS)
diff --git a/tests/fragments.c b/tests/fragments.c
index 6007232..3bb1e83 100644
--- a/tests/fragments.c
+++ b/tests/fragments.c
@@ -20,37 +20,38 @@
*/
#include <glib-object.h>
-#include "gupnp-didl-lite-object.h"
-#include "gupnp-didl-lite-object-private.h"
+#include <libgupnp-av/gupnp-didl-lite-object.h>
+#include <libgupnp-av/gupnp-didl-lite-writer.h>
+#include <libgupnp-av/gupnp-didl-lite-item.h>
/* creates an item described by:
static const gchar * const didllite =
- "<?xml version=\"1.0\" encoding=\"UTF-8\"?>"
- "<DIDL-Lite"
- "xmlns:dc=\"http://purl.org/dc/elements/1.1/\""
- "xmlns=\"urn:schemas-upnp-org:metadata-1-0/DIDL-Lite/\""
- "xmlns:upnp=\"urn:schemas-upnp-org:metadata-1-0/upnp/\""
- "xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\""
- "xsi:schemaLocation=\""
- "urn:schemas-upnp-org:metadata-1-0/DIDL-Lite/"
- "http://www.upnp.org/schemas/av/didl-lite.xsd"
- "urn:schemas-upnp-org:metadata-1-0/upnp/"
- "http://www.upnp.org/schemas/av/upnp.xsd\">"
- "<item id=\"18\" parentID=\"13\" restricted=\"0\">"
- "<dc:title>Try a little tenderness</dc:title>"
- "<upnp:class>object.item.audioItem.musicTrack</upnp:class>"
- "<res protocolInfo=\"http-get:*:audio/mpeg:*\" size=\"3558000\">"
- "http://168.192.1.1/audio197.mp3"
- "</res>"
- "<upnp:artist>Unknown</upnp:artist>"
- "</item>"
- "</DIDL-Lite>";
+ "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n"
+ "<DIDL-Lite\n"
+ "xmlns:dc=\"http://purl.org/dc/elements/1.1/\"\n"
+ "xmlns=\"urn:schemas-upnp-org:metadata-1-0/DIDL-Lite/\"\n"
+ "xmlns:upnp=\"urn:schemas-upnp-org:metadata-1-0/upnp/\"\n"
+ "xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\"\n"
+ "xsi:schemaLocation=\"\n"
+ "urn:schemas-upnp-org:metadata-1-0/DIDL-Lite/\n"
+ "http://www.upnp.org/schemas/av/didl-lite.xsd\n"
+ "urn:schemas-upnp-org:metadata-1-0/upnp/\n"
+ "http://www.upnp.org/schemas/av/upnp.xsd\">\n"
+ "<item id=\"18\" parentID=\"13\" restricted=\"0\">\n"
+ "<dc:title>Try a little tenderness</dc:title>\n"
+ "<upnp:class>object.item.audioItem.musicTrack</upnp:class>\n"
+ "<res protocolInfo=\"http-get:*:audio/mpeg:*\" size=\"3558000\">\n"
+ "http://168.192.1.1/audio197.mp3\n"
+ "</res>\n"
+ "<upnp:artist>Unknown</upnp:artist>\n"
+ "</item>\n"
+ "</DIDL-Lite>\n";
*/
static GUPnPDIDLLiteObject *
get_item (void)
{
- GUPnPDIDLiteWriter *writer = gupnp_didl_lite_writer_new ();
- GUPnPDIDLiteItem *item = gupnp_didl_lite_writer_add_item (writer);
+ GUPnPDIDLLiteWriter *writer = gupnp_didl_lite_writer_new (NULL);
+ GUPnPDIDLLiteItem *item = gupnp_didl_lite_writer_add_item (writer);
GUPnPDIDLLiteObject *object = GUPNP_DIDL_LITE_OBJECT (item);
GUPnPDIDLLiteContributor *artist;
GUPnPDIDLLiteResource *resource;
@@ -83,17 +84,25 @@ get_item (void)
}
static const gchar * const current_fragments[] = {
+ /* 1 */
"<upnp:class>object.item.audioItem.musicTrack</upnp:class>",
+ /* 2 */
"",
+ /* 3 */
"<upnp:artist>Unknown</upnp:artist>",
+ /* 4 */
"<dc:title>Try a little tenderness</dc:title>"
};
static const gchar * const new_fragments[] = {
+ /* 1 */
"<upnp:class>object.item.audioItem.musicTrack</upnp:class>"
"<upnp:genre>Obscure</upnp:genre>",
+ /* 2 */
"<upnp:genre>Even more obscure</upnp:genre>",
+ /* 3 */
"",
+ /* 4 */
"<dc:title>Cthulhu fhtagn</dc:title>"
};
@@ -105,15 +114,27 @@ create_fragment_list (const gchar * const * const fragments,
guint iter;
for (iter = 0; iter < count; ++iter) {
- list = g_list_prepend (list, fragments[iter]);
+ list = g_list_prepend (list, (gpointer) fragments[iter]);
}
return g_list_reverse (list);
}
+static void
+debug_dump (GUPnPDIDLLiteObject *object)
+{
+ xmlChar *dump = NULL;
+ xmlNodePtr node = gupnp_didl_lite_object_get_xml_node (object);
+ xmlDocPtr doc = node->doc;
+
+ xmlDocDumpMemory (doc, &dump, NULL);
+ g_debug ("Obj dump:\n%s", dump);
+ xmlFree (dump);
+}
+
int main (void)
{
- GUPnPDIDLLiteObject *obj;
+ GUPnPDIDLLiteObject *object;
GList *current = create_fragment_list
(current_fragments,
G_N_ELEMENTS (current_fragments));
@@ -122,11 +143,15 @@ int main (void)
GUPnPDIDLLiteFragmentResult result;
g_type_init ();
- g_setenv ("GUPNP_AV_DATADIR", TOP_SRCDIR G_DIR_SEPARATOR_S "data", FALSE);
-
- obj = get_item ();
- result = gupnp_didl_lite_object_apply_fragments (obj, current, new);
+ g_setenv ("GUPNP_AV_DATADIR", ABS_TOP_SRCDIR G_DIR_SEPARATOR_S "data", FALSE);
+ object = get_item ();
+ debug_dump (object);
+ result = gupnp_didl_lite_object_apply_fragments (object, current, new);
+ g_list_free (new);
+ g_list_free (current);
+ debug_dump (object);
+ g_object_unref (object);
if (result != GUPNP_DIDL_LITE_FRAGMENT_RESULT_OK)
return 1;
return 0;
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]