// This is free and unencumbered software released into the public domain. // // Anyone is free to copy, modify, publish, use, compile, sell, or // distribute this software, either in source code form or as a compiled // binary, for any purpose, commercial or non-commercial, and by any // means. // // In jurisdictions that recognize copyright laws, the author or authors // of this software dedicate any and all copyright interest in the // software to the public domain. We make this dedication for the benefit // of the public at large and to the detriment of our heirs and // successors. We intend this dedication to be an overt act of // relinquishment in perpetuity of all present and future rights to this // software under copyright law. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. // IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR // OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, // ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR // OTHER DEALINGS IN THE SOFTWARE. // // For more information, please refer to #include "utest.h" #include "json.h" UTEST(allow_trailing_comma, object_no_element) { const char payload[] = "{,}"; struct json_parse_result_s result; struct json_value_s *value = json_parse_ex(payload, strlen(payload), json_parse_flags_allow_trailing_comma, 0, 0, &result); ASSERT_FALSE(value); ASSERT_EQ(json_parse_error_invalid_string, result.error); ASSERT_EQ(1, result.error_offset); ASSERT_EQ(1, result.error_line_no); ASSERT_EQ(1, result.error_row_no); } UTEST(allow_trailing_comma, object_one_element) { const char payload[] = "{\"foo\" : true, }"; struct json_value_s *value = json_parse_ex( payload, strlen(payload), json_parse_flags_allow_trailing_comma, 0, 0, 0); struct json_object_s *object = 0; ASSERT_TRUE(value); ASSERT_EQ(json_type_object, value->type); object = (struct json_object_s *)value->payload; ASSERT_TRUE(object->start); ASSERT_EQ(1, object->length); ASSERT_TRUE(object->start->name->string); ASSERT_STREQ("foo", (char *)object->start->name->string); ASSERT_EQ(strlen("foo"), object->start->name->string_size); ASSERT_EQ(strlen((char *)object->start->name->string), object->start->name->string_size); ASSERT_TRUE(object->start->value); ASSERT_FALSE(object->start->value->payload); ASSERT_EQ(json_type_true, object->start->value->type); ASSERT_FALSE(object->start->next); free(value); } UTEST(allow_trailing_comma, object_two_elements) { const char payload[] = "{\"foo\" : true, \"bar\" : false, }"; struct json_value_s *value = json_parse_ex( payload, strlen(payload), json_parse_flags_allow_trailing_comma, 0, 0, 0); struct json_object_s *object = 0; struct json_object_element_s *element = 0; ASSERT_TRUE(value); ASSERT_EQ(json_type_object, value->type); object = (struct json_object_s *)value->payload; ASSERT_TRUE(object->start); ASSERT_EQ(2, object->length); element = object->start; ASSERT_TRUE(element->name->string); ASSERT_STREQ("foo", (char *)element->name->string); ASSERT_EQ(strlen("foo"), element->name->string_size); ASSERT_EQ(strlen((char *)element->name->string), element->name->string_size); ASSERT_TRUE(element->value); ASSERT_FALSE(element->value->payload); ASSERT_EQ(json_type_true, element->value->type); element = element->next; ASSERT_TRUE(element->name->string); ASSERT_STREQ("bar", (char *)element->name->string); ASSERT_EQ(strlen("bar"), element->name->string_size); ASSERT_EQ(strlen((char *)element->name->string), element->name->string_size); ASSERT_TRUE(element->value); ASSERT_FALSE(element->value->payload); ASSERT_EQ(json_type_false, element->value->type); ASSERT_FALSE(element->next); free(value); } UTEST(allow_trailing_comma, array_no_element) { const char payload[] = "[,]"; struct json_parse_result_s result; struct json_value_s *value = json_parse_ex(payload, strlen(payload), json_parse_flags_allow_trailing_comma, 0, 0, &result); ASSERT_FALSE(value); ASSERT_EQ(json_parse_error_invalid_value, result.error); ASSERT_EQ(1, result.error_offset); ASSERT_EQ(1, result.error_line_no); ASSERT_EQ(1, result.error_row_no); } UTEST(allow_trailing_comma, array_one_element) { const char payload[] = "[ true, ]"; struct json_value_s *value = json_parse_ex( payload, strlen(payload), json_parse_flags_allow_trailing_comma, 0, 0, 0); struct json_array_s *object = 0; ASSERT_TRUE(value); ASSERT_EQ(json_type_array, value->type); object = (struct json_array_s *)value->payload; ASSERT_TRUE(object->start); ASSERT_EQ(1, object->length); ASSERT_TRUE(object->start->value); ASSERT_FALSE(object->start->value->payload); ASSERT_EQ(json_type_true, object->start->value->type); ASSERT_FALSE(object->start->next); free(value); } UTEST(allow_trailing_comma, array_two_elements) { const char payload[] = "[ true, false, ]"; struct json_value_s *value = json_parse_ex( payload, strlen(payload), json_parse_flags_allow_trailing_comma, 0, 0, 0); struct json_array_s *object = 0; struct json_array_element_s *element = 0; ASSERT_TRUE(value); ASSERT_EQ(json_type_array, value->type); object = (struct json_array_s *)value->payload; ASSERT_TRUE(object->start); ASSERT_EQ(2, object->length); element = object->start; ASSERT_TRUE(element->value); ASSERT_FALSE(element->value->payload); ASSERT_EQ(json_type_true, element->value->type); element = element->next; ASSERT_TRUE(element->value); ASSERT_FALSE(element->value->payload); ASSERT_EQ(json_type_false, element->value->type); ASSERT_FALSE(element->next); free(value); } struct allow_trailing_comma { struct json_value_s *value; }; UTEST_F_SETUP(allow_trailing_comma) { const char payload[] = "{\"foo\" : \"Heyo, gaia?\", }"; utest_fixture->value = json_parse_ex( payload, strlen(payload), json_parse_flags_allow_trailing_comma, 0, 0, 0); ASSERT_TRUE(utest_fixture->value); } UTEST_F_TEARDOWN(allow_trailing_comma) { struct json_value_s *value = utest_fixture->value; struct json_object_s *object = 0; struct json_value_s *value2 = 0; struct json_string_s *string = 0; ASSERT_TRUE(value); ASSERT_TRUE(value->payload); ASSERT_EQ(json_type_object, value->type); object = (struct json_object_s *)value->payload; ASSERT_TRUE(object->start); ASSERT_EQ(1, object->length); ASSERT_TRUE(object->start->name); ASSERT_TRUE(object->start->value); ASSERT_FALSE(object->start->next); // we have only one element ASSERT_TRUE(object->start->name->string); ASSERT_STREQ("foo", object->start->name->string); ASSERT_EQ(strlen("foo"), object->start->name->string_size); ASSERT_EQ(strlen(object->start->name->string), object->start->name->string_size); value2 = object->start->value; ASSERT_TRUE(value2->payload); ASSERT_EQ(json_type_string, value2->type); string = (struct json_string_s *)value2->payload; ASSERT_TRUE(string->string); ASSERT_STREQ("Heyo, gaia?", string->string); ASSERT_EQ(strlen("Heyo, gaia?"), string->string_size); ASSERT_EQ(strlen(string->string), string->string_size); free(value); } UTEST_F(allow_trailing_comma, read_write_pretty_read) { size_t size = 0; void *json = json_write_pretty(utest_fixture->value, " ", "\n", &size); free(utest_fixture->value); utest_fixture->value = json_parse(json, size - 1); free(json); } UTEST_F(allow_trailing_comma, read_write_minified_read) { size_t size = 0; void *json = json_write_minified(utest_fixture->value, &size); free(utest_fixture->value); utest_fixture->value = json_parse(json, size - 1); free(json); }