summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorPhilip Withnall <philip@tecnocode.co.uk>2013-08-29 23:04:55 -0600
committerPhilip Withnall <philip@tecnocode.co.uk>2013-08-29 23:04:55 -0600
commitadfc260f9bb69adfce623882276b8578279baf63 (patch)
tree7ef7500de33f243d6a4e6dd192f22f57f235a216
parente18bec0f183b660cb7c4066dcc9e6577a819a895 (diff)
downloadlibgdata-tasks-integration.tar.gz
core: Finish JSON supporttasks-integration
This adds test cases which cover all of the new code, and fixes a number of small bugs. extra_json support is now complete.
-rw-r--r--gdata/gdata-entry.c22
-rw-r--r--gdata/gdata-parsable.c135
-rw-r--r--gdata/gdata-parser.c14
-rw-r--r--gdata/tests/general.c95
4 files changed, 223 insertions, 43 deletions
diff --git a/gdata/gdata-entry.c b/gdata/gdata-entry.c
index c1d3ce95..c3939808 100644
--- a/gdata/gdata-entry.c
+++ b/gdata/gdata-entry.c
@@ -602,13 +602,31 @@ parse_json (GDataParsable *parsable, JsonReader *reader, gpointer user_data, GEr
gdata_parser_string_from_json_member (reader, "etag", P_NON_EMPTY | P_NO_DUPES, &(priv->etag), &success, error) == TRUE) {
return success;
} else if (g_strcmp0 (json_reader_get_member_name (reader), "selfLink") == 0) {
- GDataLink *_link = gdata_link_new (json_reader_get_string_value (reader), GDATA_LINK_SELF);
+ GDataLink *_link;
+ const gchar *uri;
+
+ /* Empty URI? */
+ uri = json_reader_get_string_value (reader);
+ if (uri == NULL || *uri == '\0') {
+ return gdata_parser_error_required_json_content_missing (reader, error);
+ }
+
+ _link = gdata_link_new (uri, GDATA_LINK_SELF);
gdata_entry_add_link (GDATA_ENTRY (parsable), _link);
g_object_unref (_link);
return TRUE;
} else if (g_strcmp0 (json_reader_get_member_name (reader), "kind") == 0) {
- GDataCategory *category = gdata_category_new (json_reader_get_string_value (reader), "http://schemas.google.com/g/2005#kind", NULL);
+ GDataCategory *category;
+ const gchar *kind;
+
+ /* Empty kind? */
+ kind = json_reader_get_string_value (reader);
+ if (kind == NULL || *kind == '\0') {
+ return gdata_parser_error_required_json_content_missing (reader, error);
+ }
+
+ category = gdata_category_new (kind, "http://schemas.google.com/g/2005#kind", NULL);
gdata_entry_add_category (GDATA_ENTRY (parsable), category);
g_object_unref (category);
diff --git a/gdata/gdata-parsable.c b/gdata/gdata-parsable.c
index 29b9e69f..f9e1dfb1 100644
--- a/gdata/gdata-parsable.c
+++ b/gdata/gdata-parsable.c
@@ -55,8 +55,13 @@ static gboolean real_parse_xml (GDataParsable *parsable, xmlDoc *doc, xmlNode *n
static gboolean real_parse_json (GDataParsable *parsable, JsonReader *reader, gpointer user_data, GError **error);
struct _GDataParsablePrivate {
+ /* XML stuff. */
GString *extra_xml;
GHashTable *extra_namespaces;
+
+ /* JSON stuff. */
+ GHashTable/*<gchar*, owned JsonNode*>*/ *extra_json;
+
gboolean constructed_from_xml;
};
@@ -101,6 +106,9 @@ gdata_parsable_init (GDataParsable *self)
self->priv->extra_xml = g_string_new ("");
self->priv->extra_namespaces = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, g_free);
+
+ self->priv->extra_json = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, (GDestroyNotify) json_node_free);
+
self->priv->constructed_from_xml = FALSE;
}
@@ -145,6 +153,8 @@ gdata_parsable_finalize (GObject *object)
g_string_free (priv->extra_xml, TRUE);
g_hash_table_destroy (priv->extra_namespaces);
+ g_hash_table_destroy (priv->extra_json);
+
/* Chain up to the parent class */
G_OBJECT_CLASS (gdata_parsable_parent_class)->finalize (object);
}
@@ -181,28 +191,87 @@ real_parse_xml (GDataParsable *parsable, xmlDoc *doc, xmlNode *node, gpointer us
return TRUE;
}
+/* Extract the member node. This would be a lot easier if JsonReader had an API to return
+ * the current node (regardless of whether it's a value, object or array). FIXME: bgo#707100. */
+static JsonNode * /* transfer full */
+_json_reader_dup_current_node (JsonReader *reader)
+{
+ JsonNode *value;
+
+ if (json_reader_is_value (reader) == TRUE) {
+ /* Value nodes are easy. Well, ignoring the complication of nulls. */
+ if (json_reader_get_null_value (reader) == TRUE) {
+ value = json_node_new (JSON_NODE_NULL);
+ } else {
+ value = json_node_copy (json_reader_get_value (reader));
+ }
+ } else if (json_reader_is_object (reader) == TRUE) {
+ /* Object nodes require deep copies. */
+ gint i, members;
+ JsonObject *obj;
+
+ obj = json_object_new ();
+
+ for (i = 0, members = json_reader_count_members (reader); i < members; i++) {
+ json_reader_read_element (reader, i);
+ json_object_set_member (obj, json_reader_get_member_name (reader), _json_reader_dup_current_node (reader));
+ json_reader_end_element (reader);
+ }
+
+ value = json_node_new (JSON_NODE_OBJECT);
+ json_node_take_object (value, obj);
+ } else if (json_reader_is_array (reader) == TRUE) {
+ /* Array nodes require deep copies. */
+ gint i, elements;
+ JsonArray *arr;
+
+ arr = json_array_new ();
+
+ for (i = 0, elements = json_reader_count_elements (reader); i < elements; i++) {
+ json_reader_read_element (reader, i);
+ json_array_add_element (arr, _json_reader_dup_current_node (reader));
+ json_reader_end_element (reader);
+ }
+
+ value = json_node_new (JSON_NODE_ARRAY);
+ json_node_take_array (value, arr);
+ } else {
+ /* Uh-oh. */
+ g_assert_not_reached ();
+ }
+
+ return value;
+}
+
static gboolean
real_parse_json (GDataParsable *parsable, JsonReader *reader, gpointer user_data, GError **error)
{
- gchar *json;
- JsonNode *root;
+ gchar *json, *member_name;
JsonGenerator *generator;
+ JsonNode *value;
- /* Unhandled JSON. Save it to ->extra_xml so that it's not lost if we
+ /* Unhandled JSON member. Save it and its value to ->extra_xml so that it's not lost if we
* re-upload this Parsable to the server. */
- generator = json_generator_new ();
- g_object_get (G_OBJECT (reader), "root", &root, NULL);
- json_generator_set_root (generator, root);
- json_node_free (root);
+ member_name = g_strdup (json_reader_get_member_name (reader));
+ g_assert (member_name != NULL);
- json = json_generator_to_data (generator, NULL);
- g_string_append (parsable->priv->extra_xml, json);
+ /* Extract a copy of the current node. */
+ value = _json_reader_dup_current_node (reader);
+ g_assert (value != NULL);
- g_debug ("Unhandled JSON in %s: %s", G_OBJECT_TYPE_NAME (parsable), json);
+ /* Serialise the value for debugging. */
+ generator = json_generator_new ();
+ json_generator_set_root (generator, value);
+ json = json_generator_to_data (generator, NULL);
+ g_debug ("Unhandled JSON member ā€˜%sā€™ in %s: %s", member_name, G_OBJECT_TYPE_NAME (parsable), json);
g_free (json);
+
g_object_unref (generator);
+ /* Save the value. Transfer ownership of the member_name and value. */
+ g_hash_table_replace (parsable->priv->extra_json, (gpointer) member_name, (gpointer) value);
+
return TRUE;
}
@@ -357,12 +426,12 @@ _gdata_parsable_new_from_xml_node (GType parsable_type, xmlDoc *doc, xmlNode *no
*
* Creates a new #GDataParsable subclass (of the given @parsable_type) from the given @json.
*
- * An object of the given @parsable_type is created, and its <function>pre_parse_json</function>, <function>parse_json</function> and
- * <function>post_parse_json</function> class functions called on the JSON node obtained from @json. <function>pre_parse_json</function> and
- * <function>post_parse_json</function> are called once each on the root node, while <function>parse_json</function> is called for
- * each of the child nodes. TODO: Check me.
+ * An object of the given @parsable_type is created, and its <function>parse_json</function> and
+ * <function>post_parse_json</function> class functions called on the JSON node obtained from @json.
+ * <function>post_parse_json</function> is called once on the root node, while <function>parse_json</function> is called for
+ * each of the node's members.
*
- * If @length is -1, @json will be assumed to be null-terminated.
+ * If @length is -1, @json will be assumed to be nul-terminated.
*
* If an error occurs during parsing, a suitable error from #GDataParserError will be returned.
*
@@ -387,6 +456,7 @@ _gdata_parsable_new_from_json (GType parsable_type, const gchar *json, gint leng
JsonParser *parser;
JsonReader *reader;
GDataParsable *parsable;
+ GError *child_error = NULL;
g_return_val_if_fail (g_type_is_a (parsable_type, GDATA_TYPE_PARSABLE), NULL);
g_return_val_if_fail (json != NULL && *json != '\0', NULL);
@@ -397,8 +467,13 @@ _gdata_parsable_new_from_json (GType parsable_type, const gchar *json, gint leng
length = strlen (json);
parser = json_parser_new ();
- if (!json_parser_load_from_data (parser, json, length, error)) {
- g_assert (error == NULL || *error != NULL); /* safety check */
+ if (!json_parser_load_from_data (parser, json, length, &child_error)) {
+ g_set_error (error, GDATA_PARSER_ERROR, GDATA_PARSER_ERROR_PARSING_STRING,
+ /* Translators: the parameter is an error message */
+ _("Error parsing JSON: %s"), child_error->message);
+ g_error_free (child_error);
+ g_object_unref (parser);
+
return NULL;
}
@@ -427,7 +502,14 @@ _gdata_parsable_new_from_json_node (GType parsable_type, JsonReader *reader, gpo
parsable = g_object_new (parsable_type, "constructed-from-xml", TRUE, NULL);
klass = GDATA_PARSABLE_GET_CLASS (parsable);
- if (klass->parse_json == NULL) {
+ g_assert (klass->parse_json != NULL);
+
+ /* Check that the outermost node is an object. */
+ if (json_reader_is_object (reader) == FALSE) {
+ g_set_error (error, GDATA_PARSER_ERROR, GDATA_PARSER_ERROR_PARSING_STRING,
+ /* Translators: the parameter is an error message */
+ _("Error parsing JSON: %s"),
+ _("Outermost JSON node is not an object."));
g_object_unref (parsable);
return NULL;
}
@@ -578,8 +660,6 @@ _gdata_parsable_get_xml (GDataParsable *self, GString *xml_string, gboolean decl
* Builds a JSON representation of the #GDataParsable in its current state, such that it could be inserted on the server. The JSON
* is valid for stand-alone use.
*
- * TODO: How to enforce mutual exclusion between get_json and get_xml?
- *
* Return value: the object's JSON; free with g_free()
*
* Since: UNRELEASED
@@ -624,6 +704,9 @@ void
_gdata_parsable_get_json (GDataParsable *self, JsonBuilder *builder)
{
GDataParsableClass *klass;
+ GHashTableIter iter;
+ gchar *member_name;
+ JsonNode *value;
g_return_if_fail (GDATA_IS_PARSABLE (self));
g_return_if_fail (JSON_IS_BUILDER (builder));
@@ -636,12 +719,12 @@ _gdata_parsable_get_json (GDataParsable *self, JsonBuilder *builder)
if (klass->get_json != NULL)
klass->get_json (self, builder);
-#if 0
-TODO
- /* Any extra JSON? Note: The use of extra_xml is intended; the variable is re-used and hence is mis-named. */
- if (self->priv->extra_xml != NULL && self->priv->extra_xml->str != NULL)
- g_string_append (json_string, self->priv->extra_xml->str);
-#endif
+ /* Any extra JSON which we couldn't parse before? */
+ g_hash_table_iter_init (&iter, self->priv->extra_json);
+ while (g_hash_table_iter_next (&iter, (gpointer *) &member_name, (gpointer *) &value) == TRUE) {
+ json_builder_set_member_name (builder, member_name);
+ json_builder_add_value (builder, json_node_copy (value)); /* transfers ownership */
+ }
json_builder_end_object (builder);
}
diff --git a/gdata/gdata-parser.c b/gdata/gdata-parser.c
index 02222934..1bb926bf 100644
--- a/gdata/gdata-parser.c
+++ b/gdata/gdata-parser.c
@@ -769,11 +769,8 @@ gdata_parser_string_from_json_member (JsonReader *reader, const gchar *member_na
return FALSE;
}
- /* Check if the output string has already been set */
- if (options & P_NO_DUPES && *output != NULL) {
- *success = gdata_parser_error_duplicate_json_element (reader, error);
- return TRUE;
- }
+ /* Check if the output string has already been set. The JSON parser guarantees this can't happen. */
+ g_assert (!(options & P_NO_DUPES) || *output == NULL);
/* Get the string and check it for NULLness or emptiness. Check for parser errors first. */
text = json_reader_get_string_value (reader);
@@ -838,11 +835,8 @@ gdata_parser_int64_time_from_json_member (JsonReader *reader, const gchar *membe
return FALSE;
}
- /* Check if the output time val has already been set */
- if (options & P_NO_DUPES && *output != -1) {
- *success = gdata_parser_error_duplicate_json_element (reader, error);
- return TRUE;
- }
+ /* Check if the output time val has already been set. The JSON parser guarantees this can't happen. */
+ g_assert (!(options & P_NO_DUPES) || *output == -1);
/* Get the string and check it for NULLness. Check for errors first. */
text = json_reader_get_string_value (reader);
diff --git a/gdata/tests/general.c b/gdata/tests/general.c
index f8f2c787..88831e86 100644
--- a/gdata/tests/general.c
+++ b/gdata/tests/general.c
@@ -553,11 +553,23 @@ test_entry_parse_json (void)
"\"etag\":\"some-etag\","
"\"id\":\"some-id\","
"\"kind\":\"kind#kind\","
- "\"unhandled-member\":false"
+ "\"unhandled-boolean\":false,"
+ "\"unhandled-string\":\"this-is-a-string---sometimes\","
+ "\"unhandled-int\":15,"
+ "\"unhandled-double\":42.42,"
+ "\"unhandled-object\":{"
+ "\"a\":true,"
+ "\"b\":true"
+ "},"
+ "\"unhandled-array\":["
+ "1,"
+ "2,"
+ "3"
+ "],"
+ "\"unhandled-null\":null"
"}", -1, &error));
g_assert_no_error (error);
g_assert (GDATA_IS_ENTRY (entry));
- g_clear_error (&error);
/* Now check the outputted JSON from the entry still has the unhandled nodes. */
gdata_test_assert_json (entry,
@@ -568,13 +580,35 @@ test_entry_parse_json (void)
"\"etag\":\"some-etag\","
"\"selfLink\":\"http://example.com/\","
"\"kind\":\"kind#kind\","
- "\"unhandled-member\":false"
+ "\"unhandled-boolean\":false,"
+ "\"unhandled-string\":\"this-is-a-string---sometimes\","
+ "\"unhandled-int\":15,"
+ "\"unhandled-double\":42.42,"
+ "\"unhandled-object\":{"
+ "\"a\":true,"
+ "\"b\":true"
+ "},"
+ "\"unhandled-array\":["
+ "1,"
+ "2,"
+ "3"
+ "],"
+ "\"unhandled-null\":null"
"}");
g_object_unref (entry);
+
+ /* Test parsing of empty titles. */
+ entry = GDATA_ENTRY (gdata_parsable_new_from_json (GDATA_TYPE_ENTRY,
+ "{"
+ "\"title\":\"\""
+ "}", -1, &error));
+ g_assert_no_error (error);
+ g_assert (GDATA_IS_ENTRY (entry));
+ g_object_unref (entry);
}
static void
-test_entry_error_handling (void)
+test_entry_error_handling_xml (void)
{
GDataEntry *entry;
GError *error = NULL;
@@ -606,6 +640,56 @@ test_entry_error_handling (void)
}
static void
+test_entry_error_handling_json (void)
+{
+ GDataEntry *entry;
+ GError *error = NULL;
+
+#define TEST_JSON_ERROR_HANDLING(x) \
+ entry = GDATA_ENTRY (gdata_parsable_new_from_json (GDATA_TYPE_ENTRY,x, -1, &error));\
+ g_assert_error (error, GDATA_SERVICE_ERROR, GDATA_SERVICE_ERROR_PROTOCOL_ERROR);\
+ g_assert (entry == NULL);\
+ g_clear_error (&error)
+#define TEST_JSON_ERROR_HANDLING_PARSER(x) \
+ entry = GDATA_ENTRY (gdata_parsable_new_from_json (GDATA_TYPE_ENTRY,x, -1, &error));\
+ g_assert_error (error, GDATA_PARSER_ERROR, GDATA_PARSER_ERROR_PARSING_STRING);\
+ g_assert (entry == NULL);\
+ g_clear_error (&error)
+
+ /* General structure. */
+ TEST_JSON_ERROR_HANDLING_PARSER ("[true,false,true]"); /* root node is not an object */
+ TEST_JSON_ERROR_HANDLING_PARSER ("false"); /* root node is not an object */
+ TEST_JSON_ERROR_HANDLING_PARSER ("invalid json"); /* totally invalid JSON */
+
+ /* id */
+ TEST_JSON_ERROR_HANDLING ("{\"id\":\"\"}"); /* empty */
+ TEST_JSON_ERROR_HANDLING ("{\"id\":null}"); /* invalid type */
+
+ /* updated */
+ TEST_JSON_ERROR_HANDLING ("{\"updated\":\"not correct\"}"); /* incorrect format */
+ TEST_JSON_ERROR_HANDLING ("{\"updated\":false}"); /* invalid type */
+ TEST_JSON_ERROR_HANDLING ("{\"updated\":\"\"}"); /* empty */
+
+ /* title */
+ TEST_JSON_ERROR_HANDLING ("{\"title\":false}"); /* invalid type */
+
+ /* etag */
+ TEST_JSON_ERROR_HANDLING ("{\"etag\":\"\"}"); /* empty */
+ TEST_JSON_ERROR_HANDLING ("{\"etag\":false}"); /* invalid type */
+
+ /* selfLink */
+ TEST_JSON_ERROR_HANDLING ("{\"selfLink\":\"\"}"); /* empty */
+ TEST_JSON_ERROR_HANDLING ("{\"selfLink\":false}"); /* invalid type */
+
+ /* kind */
+ TEST_JSON_ERROR_HANDLING ("{\"kind\":\"\"}"); /* empty */
+ TEST_JSON_ERROR_HANDLING ("{\"kind\":false}"); /* invalid type */
+
+#undef TEST_JSON_ERROR_HANDLING_PARSER
+#undef TEST_JSON_ERROR_HANDLING
+}
+
+static void
test_entry_escaping (void)
{
GDataEntry *entry;
@@ -4082,7 +4166,8 @@ main (int argc, char *argv[])
g_test_add_func ("/entry/get_json", test_entry_get_json);
g_test_add_func ("/entry/parse_xml", test_entry_parse_xml);
g_test_add_func ("/entry/parse_json", test_entry_parse_json);
- g_test_add_func ("/entry/error_handling", test_entry_error_handling);
+ g_test_add_func ("/entry/error_handling/xml", test_entry_error_handling_xml);
+ g_test_add_func ("/entry/error_handling/json", test_entry_error_handling_json);
g_test_add_func ("/entry/escaping", test_entry_escaping);
g_test_add_func ("/entry/links/remove", test_entry_links_remove);