2017-02-21 11:17:49 +03:00
|
|
|
/*
|
|
|
|
Copyright (c) 2009-2017 Dave Gamble and cJSON contributors
|
|
|
|
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
of this software and associated documentation files (the "Software"), to deal
|
|
|
|
in the Software without restriction, including without limitation the rights
|
|
|
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
copies of the Software, and to permit persons to whom the Software is
|
|
|
|
furnished to do so, subject to the following conditions:
|
|
|
|
|
|
|
|
The above copyright notice and this permission notice shall be included in
|
|
|
|
all copies or substantial portions of the Software.
|
|
|
|
|
|
|
|
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 OR COPYRIGHT HOLDERS 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "unity/examples/unity_config.h"
|
|
|
|
#include "unity/src/unity.h"
|
|
|
|
#include "common.h"
|
|
|
|
|
|
|
|
|
|
|
|
static void cjson_array_foreach_should_loop_over_arrays(void)
|
|
|
|
{
|
|
|
|
cJSON array[1];
|
|
|
|
cJSON elements[10];
|
|
|
|
cJSON *element_pointer = NULL;
|
|
|
|
size_t i = 0;
|
|
|
|
|
|
|
|
memset(array, 0, sizeof(array));
|
|
|
|
memset(elements, 0, sizeof(elements));
|
|
|
|
|
|
|
|
/* create array */
|
|
|
|
array[0].child = &elements[0];
|
|
|
|
elements[0].prev = NULL;
|
|
|
|
elements[9].next = NULL;
|
|
|
|
for (i = 0; i < 9; i++)
|
|
|
|
{
|
|
|
|
elements[i].next = &elements[i + 1];
|
|
|
|
elements[i + 1].prev = &elements[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
i = 0;
|
|
|
|
cJSON_ArrayForEach(element_pointer, array)
|
|
|
|
{
|
|
|
|
TEST_ASSERT_TRUE_MESSAGE(element_pointer == &elements[i], "Not iterating over array properly");
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void cjson_array_foreach_should_not_dereference_null_pointer(void)
|
|
|
|
{
|
|
|
|
cJSON *array = NULL;
|
|
|
|
cJSON *element = NULL;
|
|
|
|
cJSON_ArrayForEach(element, array);
|
|
|
|
}
|
|
|
|
|
2017-02-21 12:45:22 +03:00
|
|
|
static void cjson_get_object_item_should_get_object_items(void)
|
|
|
|
{
|
|
|
|
cJSON *item = NULL;
|
|
|
|
cJSON *found = NULL;
|
|
|
|
|
|
|
|
item = cJSON_Parse("{\"one\":1, \"Two\":2, \"tHree\":3}");
|
|
|
|
|
|
|
|
found = cJSON_GetObjectItem(NULL, "test");
|
|
|
|
TEST_ASSERT_NULL_MESSAGE(found, "Failed to fail on NULL pointer.");
|
|
|
|
|
|
|
|
found = cJSON_GetObjectItem(item, NULL);
|
|
|
|
TEST_ASSERT_NULL_MESSAGE(found, "Failed to fail on NULL string.");
|
|
|
|
|
|
|
|
|
|
|
|
found = cJSON_GetObjectItem(item, "one");
|
|
|
|
TEST_ASSERT_NOT_NULL_MESSAGE(found, "Failed to find first item.");
|
|
|
|
TEST_ASSERT_EQUAL_DOUBLE(found->valuedouble, 1);
|
|
|
|
|
|
|
|
found = cJSON_GetObjectItem(item, "tWo");
|
|
|
|
TEST_ASSERT_NOT_NULL_MESSAGE(found, "Failed to find first item.");
|
|
|
|
TEST_ASSERT_EQUAL_DOUBLE(found->valuedouble, 2);
|
|
|
|
|
|
|
|
found = cJSON_GetObjectItem(item, "three");
|
|
|
|
TEST_ASSERT_NOT_NULL_MESSAGE(found, "Failed to find item.");
|
|
|
|
TEST_ASSERT_EQUAL_DOUBLE(found->valuedouble, 3);
|
|
|
|
|
|
|
|
found = cJSON_GetObjectItem(item, "four");
|
|
|
|
TEST_ASSERT_NULL_MESSAGE(found, "Should not find something that isn't there.");
|
|
|
|
|
|
|
|
cJSON_Delete(item);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void cjson_get_object_item_case_sensitive_should_get_object_items(void)
|
|
|
|
{
|
|
|
|
cJSON *item = NULL;
|
|
|
|
cJSON *found = NULL;
|
|
|
|
|
|
|
|
item = cJSON_Parse("{\"one\":1, \"Two\":2, \"tHree\":3}");
|
|
|
|
|
|
|
|
found = cJSON_GetObjectItemCaseSensitive(NULL, "test");
|
|
|
|
TEST_ASSERT_NULL_MESSAGE(found, "Failed to fail on NULL pointer.");
|
|
|
|
|
|
|
|
found = cJSON_GetObjectItemCaseSensitive(item, NULL);
|
|
|
|
TEST_ASSERT_NULL_MESSAGE(found, "Failed to fail on NULL string.");
|
|
|
|
|
|
|
|
found = cJSON_GetObjectItemCaseSensitive(item, "one");
|
|
|
|
TEST_ASSERT_NOT_NULL_MESSAGE(found, "Failed to find first item.");
|
|
|
|
TEST_ASSERT_EQUAL_DOUBLE(found->valuedouble, 1);
|
|
|
|
|
|
|
|
found = cJSON_GetObjectItemCaseSensitive(item, "Two");
|
|
|
|
TEST_ASSERT_NOT_NULL_MESSAGE(found, "Failed to find first item.");
|
|
|
|
TEST_ASSERT_EQUAL_DOUBLE(found->valuedouble, 2);
|
|
|
|
|
|
|
|
found = cJSON_GetObjectItemCaseSensitive(item, "tHree");
|
|
|
|
TEST_ASSERT_NOT_NULL_MESSAGE(found, "Failed to find item.");
|
|
|
|
TEST_ASSERT_EQUAL_DOUBLE(found->valuedouble, 3);
|
|
|
|
|
|
|
|
found = cJSON_GetObjectItemCaseSensitive(item, "One");
|
|
|
|
TEST_ASSERT_NULL_MESSAGE(found, "Should not find something that isn't there.");
|
|
|
|
|
|
|
|
cJSON_Delete(item);
|
|
|
|
}
|
|
|
|
|
2017-02-26 23:54:01 +03:00
|
|
|
static void typecheck_functions_should_check_type(void)
|
|
|
|
{
|
|
|
|
cJSON invalid[1];
|
|
|
|
cJSON item[1];
|
|
|
|
invalid->type = cJSON_Invalid;
|
|
|
|
invalid->type |= cJSON_StringIsConst;
|
|
|
|
item->type = cJSON_False;
|
|
|
|
item->type |= cJSON_StringIsConst;
|
|
|
|
|
|
|
|
TEST_ASSERT_FALSE(cJSON_IsInvalid(NULL));
|
|
|
|
TEST_ASSERT_FALSE(cJSON_IsInvalid(item));
|
|
|
|
TEST_ASSERT_TRUE(cJSON_IsInvalid(invalid));
|
|
|
|
|
|
|
|
item->type = cJSON_False | cJSON_StringIsConst;
|
|
|
|
TEST_ASSERT_FALSE(cJSON_IsFalse(NULL));
|
|
|
|
TEST_ASSERT_FALSE(cJSON_IsFalse(invalid));
|
|
|
|
TEST_ASSERT_TRUE(cJSON_IsFalse(item));
|
|
|
|
TEST_ASSERT_TRUE(cJSON_IsBool(item));
|
|
|
|
|
|
|
|
item->type = cJSON_True | cJSON_StringIsConst;
|
|
|
|
TEST_ASSERT_FALSE(cJSON_IsTrue(NULL));
|
|
|
|
TEST_ASSERT_FALSE(cJSON_IsTrue(invalid));
|
|
|
|
TEST_ASSERT_TRUE(cJSON_IsTrue(item));
|
|
|
|
TEST_ASSERT_TRUE(cJSON_IsBool(item));
|
|
|
|
|
|
|
|
item->type = cJSON_NULL | cJSON_StringIsConst;
|
|
|
|
TEST_ASSERT_FALSE(cJSON_IsNull(NULL));
|
|
|
|
TEST_ASSERT_FALSE(cJSON_IsNull(invalid));
|
|
|
|
TEST_ASSERT_TRUE(cJSON_IsNull(item));
|
|
|
|
|
|
|
|
item->type = cJSON_Number | cJSON_StringIsConst;
|
|
|
|
TEST_ASSERT_FALSE(cJSON_IsNumber(NULL));
|
|
|
|
TEST_ASSERT_FALSE(cJSON_IsNumber(invalid));
|
|
|
|
TEST_ASSERT_TRUE(cJSON_IsNumber(item));
|
|
|
|
|
|
|
|
item->type = cJSON_String | cJSON_StringIsConst;
|
|
|
|
TEST_ASSERT_FALSE(cJSON_IsString(NULL));
|
|
|
|
TEST_ASSERT_FALSE(cJSON_IsString(invalid));
|
|
|
|
TEST_ASSERT_TRUE(cJSON_IsString(item));
|
|
|
|
|
|
|
|
item->type = cJSON_Array | cJSON_StringIsConst;
|
|
|
|
TEST_ASSERT_FALSE(cJSON_IsArray(NULL));
|
|
|
|
TEST_ASSERT_FALSE(cJSON_IsArray(invalid));
|
|
|
|
TEST_ASSERT_TRUE(cJSON_IsArray(item));
|
|
|
|
|
|
|
|
item->type = cJSON_Object | cJSON_StringIsConst;
|
|
|
|
TEST_ASSERT_FALSE(cJSON_IsObject(NULL));
|
|
|
|
TEST_ASSERT_FALSE(cJSON_IsObject(invalid));
|
|
|
|
TEST_ASSERT_TRUE(cJSON_IsObject(item));
|
|
|
|
|
|
|
|
item->type = cJSON_Raw | cJSON_StringIsConst;
|
|
|
|
TEST_ASSERT_FALSE(cJSON_IsRaw(NULL));
|
|
|
|
TEST_ASSERT_FALSE(cJSON_IsRaw(invalid));
|
|
|
|
TEST_ASSERT_TRUE(cJSON_IsRaw(item));
|
|
|
|
}
|
|
|
|
|
2017-04-27 02:48:40 +03:00
|
|
|
static void cjson_should_not_parse_to_deeply_nested_jsons(void)
|
|
|
|
{
|
|
|
|
char deep_json[CJSON_NESTING_LIMIT + 1];
|
|
|
|
size_t position = 0;
|
|
|
|
|
|
|
|
for (position = 0; position < sizeof(deep_json); position++)
|
|
|
|
{
|
|
|
|
deep_json[position] = '[';
|
|
|
|
}
|
|
|
|
deep_json[sizeof(deep_json) - 1] = '\0';
|
|
|
|
|
|
|
|
TEST_ASSERT_NULL_MESSAGE(cJSON_Parse(deep_json), "To deep JSONs should not be parsed.");
|
|
|
|
}
|
|
|
|
|
2017-04-28 15:41:24 +03:00
|
|
|
static void cjson_set_number_value_should_set_numbers(void)
|
|
|
|
{
|
|
|
|
cJSON number[1] = {{NULL, NULL, NULL, cJSON_Number, NULL, 0, 0, NULL}};
|
|
|
|
|
|
|
|
cJSON_SetNumberValue(number, 1.5);
|
|
|
|
TEST_ASSERT_EQUAL(1, number->valueint);
|
|
|
|
TEST_ASSERT_EQUAL_DOUBLE(1.5, number->valuedouble);
|
|
|
|
|
|
|
|
cJSON_SetNumberValue(number, -1.5);
|
|
|
|
TEST_ASSERT_EQUAL(-1, number->valueint);
|
|
|
|
TEST_ASSERT_EQUAL_DOUBLE(-1.5, number->valuedouble);
|
|
|
|
|
|
|
|
cJSON_SetNumberValue(number, 1 + (double)INT_MAX);
|
|
|
|
TEST_ASSERT_EQUAL(INT_MAX, number->valueint);
|
|
|
|
TEST_ASSERT_EQUAL_DOUBLE(1 + (double)INT_MAX, number->valuedouble);
|
|
|
|
|
|
|
|
cJSON_SetNumberValue(number, -1 + (double)INT_MIN);
|
|
|
|
TEST_ASSERT_EQUAL(INT_MIN, number->valueint);
|
|
|
|
TEST_ASSERT_EQUAL_DOUBLE(-1 + (double)INT_MIN, number->valuedouble);
|
|
|
|
}
|
|
|
|
|
2017-05-02 00:15:00 +03:00
|
|
|
static void cjson_detach_item_via_pointer_should_detach_items(void)
|
|
|
|
{
|
|
|
|
cJSON list[4];
|
|
|
|
cJSON parent[1];
|
|
|
|
|
|
|
|
memset(list, '\0', sizeof(list));
|
|
|
|
|
|
|
|
/* link the list */
|
|
|
|
list[0].next = &(list[1]);
|
|
|
|
list[1].next = &(list[2]);
|
|
|
|
list[2].next = &(list[3]);
|
|
|
|
|
|
|
|
list[3].prev = &(list[2]);
|
|
|
|
list[2].prev = &(list[1]);
|
|
|
|
list[1].prev = &(list[0]);
|
|
|
|
|
|
|
|
parent->child = &list[0];
|
|
|
|
|
|
|
|
/* detach in the middle (list[1]) */
|
|
|
|
TEST_ASSERT_TRUE_MESSAGE(cJSON_DetachItemViaPointer(parent, &(list[1])) == &(list[1]), "Failed to detach in the middle.");
|
|
|
|
TEST_ASSERT_TRUE_MESSAGE((list[1].prev == NULL) && (list[1].next == NULL), "Didn't set pointers of detached item to NULL.");
|
|
|
|
TEST_ASSERT_TRUE((list[0].next == &(list[2])) && (list[2].prev == &(list[0])));
|
|
|
|
|
|
|
|
/* detach beginning (list[0]) */
|
|
|
|
TEST_ASSERT_TRUE_MESSAGE(cJSON_DetachItemViaPointer(parent, &(list[0])) == &(list[0]), "Failed to detach beginning.");
|
|
|
|
TEST_ASSERT_TRUE_MESSAGE((list[0].prev == NULL) && (list[0].next == NULL), "Didn't set pointers of detached item to NULL.");
|
|
|
|
TEST_ASSERT_TRUE_MESSAGE((list[2].prev == NULL) && (parent->child == &(list[2])), "Didn't set the new beginning.");
|
|
|
|
|
|
|
|
/* detach end (list[3])*/
|
|
|
|
TEST_ASSERT_TRUE_MESSAGE(cJSON_DetachItemViaPointer(parent, &(list[3])) == &(list[3]), "Failed to detach end.");
|
|
|
|
TEST_ASSERT_TRUE_MESSAGE((list[3].prev == NULL) && (list[3].next == NULL), "Didn't set pointers of detached item to NULL.");
|
|
|
|
TEST_ASSERT_TRUE_MESSAGE((list[2].next == NULL) && (parent->child == &(list[2])), "Didn't set the new end");
|
|
|
|
|
|
|
|
/* detach single item (list[2]) */
|
|
|
|
TEST_ASSERT_TRUE_MESSAGE(cJSON_DetachItemViaPointer(parent, &list[2]) == &list[2], "Failed to detach single item.");
|
|
|
|
TEST_ASSERT_TRUE_MESSAGE((list[2].prev == NULL) && (list[2].next == NULL), "Didn't set pointers of detached item to NULL.");
|
|
|
|
TEST_ASSERT_NULL_MESSAGE(parent->child, "Child of the parent wasn't set to NULL.");
|
|
|
|
}
|
|
|
|
|
2017-05-02 01:49:46 +03:00
|
|
|
static void cjson_replace_item_via_pointer_should_replace_items(void)
|
|
|
|
{
|
|
|
|
cJSON replacements[3];
|
|
|
|
cJSON *beginning = NULL;
|
|
|
|
cJSON *middle = NULL;
|
|
|
|
cJSON *end = NULL;
|
|
|
|
cJSON *array = NULL;
|
|
|
|
|
|
|
|
beginning = cJSON_CreateNull();
|
|
|
|
TEST_ASSERT_NOT_NULL(beginning);
|
|
|
|
middle = cJSON_CreateNull();
|
|
|
|
TEST_ASSERT_NOT_NULL(middle);
|
|
|
|
end = cJSON_CreateNull();
|
|
|
|
TEST_ASSERT_NOT_NULL(end);
|
|
|
|
|
|
|
|
array = cJSON_CreateArray();
|
|
|
|
TEST_ASSERT_NOT_NULL(array);
|
|
|
|
|
|
|
|
cJSON_AddItemToArray(array, beginning);
|
|
|
|
cJSON_AddItemToArray(array, middle);
|
|
|
|
cJSON_AddItemToArray(array, end);
|
|
|
|
|
|
|
|
|
|
|
|
memset(replacements, '\0', sizeof(replacements));
|
|
|
|
|
|
|
|
/* replace beginning */
|
|
|
|
TEST_ASSERT_TRUE(cJSON_ReplaceItemViaPointer(array, beginning, &(replacements[0])));
|
|
|
|
TEST_ASSERT_NULL(replacements[0].prev);
|
|
|
|
TEST_ASSERT_TRUE(replacements[0].next == middle);
|
|
|
|
TEST_ASSERT_TRUE(middle->prev == &(replacements[0]));
|
|
|
|
TEST_ASSERT_TRUE(array->child == &(replacements[0]));
|
|
|
|
|
|
|
|
/* replace middle */
|
|
|
|
TEST_ASSERT_TRUE(cJSON_ReplaceItemViaPointer(array, middle, &(replacements[1])));
|
|
|
|
TEST_ASSERT_TRUE(replacements[1].prev == &(replacements[0]));
|
|
|
|
TEST_ASSERT_TRUE(replacements[1].next == end);
|
|
|
|
TEST_ASSERT_TRUE(end->prev == &(replacements[1]));
|
|
|
|
|
|
|
|
/* replace end */
|
|
|
|
TEST_ASSERT_TRUE(cJSON_ReplaceItemViaPointer(array, end, &(replacements[2])));
|
|
|
|
TEST_ASSERT_TRUE(replacements[2].prev == &(replacements[1]));
|
|
|
|
TEST_ASSERT_NULL(replacements[2].next);
|
|
|
|
TEST_ASSERT_TRUE(replacements[1].next == &(replacements[2]));
|
|
|
|
|
|
|
|
cJSON_free(array);
|
|
|
|
}
|
|
|
|
|
2017-02-21 11:17:49 +03:00
|
|
|
int main(void)
|
|
|
|
{
|
|
|
|
UNITY_BEGIN();
|
|
|
|
|
|
|
|
RUN_TEST(cjson_array_foreach_should_loop_over_arrays);
|
|
|
|
RUN_TEST(cjson_array_foreach_should_not_dereference_null_pointer);
|
2017-02-21 12:45:22 +03:00
|
|
|
RUN_TEST(cjson_get_object_item_should_get_object_items);
|
|
|
|
RUN_TEST(cjson_get_object_item_case_sensitive_should_get_object_items);
|
2017-02-26 23:54:01 +03:00
|
|
|
RUN_TEST(typecheck_functions_should_check_type);
|
2017-04-27 02:48:40 +03:00
|
|
|
RUN_TEST(cjson_should_not_parse_to_deeply_nested_jsons);
|
2017-04-28 15:41:24 +03:00
|
|
|
RUN_TEST(cjson_set_number_value_should_set_numbers);
|
2017-05-02 00:15:00 +03:00
|
|
|
RUN_TEST(cjson_detach_item_via_pointer_should_detach_items);
|
2017-05-02 01:49:46 +03:00
|
|
|
RUN_TEST(cjson_replace_item_via_pointer_should_replace_items);
|
2017-02-21 11:17:49 +03:00
|
|
|
|
|
|
|
return UNITY_END();
|
|
|
|
}
|