mirror of
https://github.com/DaveGamble/cJSON.git
synced 2023-08-10 21:13:26 +03:00
Added simple load & save fns for cJSON objects, with unit tests
This commit is contained in:
parent
5437b79086
commit
0b7c916252
145
cJSON.c
145
cJSON.c
@ -2988,3 +2988,148 @@ CJSON_PUBLIC(void) cJSON_free(void *object)
|
|||||||
{
|
{
|
||||||
global_hooks.deallocate(object);
|
global_hooks.deallocate(object);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* This will write out a json object to the specified file name
|
||||||
|
|
||||||
|
Args:
|
||||||
|
filename - the name of the file to load
|
||||||
|
**item - a ptr to the cJSON structure to store the parsed root item in
|
||||||
|
**errorMessage - an optional field to return an error message
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
1 - on success
|
||||||
|
0 - The provided object was NULL
|
||||||
|
-1 - on failure to open the file to write
|
||||||
|
-2 - on failure to stringify the cJSON object
|
||||||
|
*/
|
||||||
|
CJSON_PUBLIC(int) cJSON_saveJSONfile(const char *filename, cJSON *item, char *errorMessage) {
|
||||||
|
FILE *fptr = NULL;
|
||||||
|
char *json_str = NULL;
|
||||||
|
|
||||||
|
/* Null check the incoming object */
|
||||||
|
if (item == NULL) { return 0; }
|
||||||
|
|
||||||
|
/* Open the target file for writting */
|
||||||
|
fptr = fopen(filename, "w");
|
||||||
|
if (fptr == NULL) {
|
||||||
|
if (errorMessage != NULL)
|
||||||
|
{
|
||||||
|
sprintf(errorMessage, "Cannot open the file '%s' to write.", filename);
|
||||||
|
}
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Print out the json string to the file */
|
||||||
|
json_str = cJSON_Print(item);
|
||||||
|
if (json_str != NULL)
|
||||||
|
{
|
||||||
|
fprintf(fptr, "%s", json_str);
|
||||||
|
fprintf(fptr, "\n");
|
||||||
|
free(json_str);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
if (errorMessage != NULL)
|
||||||
|
{
|
||||||
|
sprintf(errorMessage, "Failed to print json object to a string.");
|
||||||
|
}
|
||||||
|
fclose(fptr);
|
||||||
|
return -2;
|
||||||
|
}
|
||||||
|
fclose(fptr);
|
||||||
|
|
||||||
|
/* Success */
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* This will load and parse a json file.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
filename - the name of the file to load
|
||||||
|
**item - a ptr to the cJSON structure to store the parsed root item in
|
||||||
|
**errorMessage - an optional field to return an error message
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
1 - on success
|
||||||
|
0 - file not found
|
||||||
|
-1 - on intial memory allocation failure
|
||||||
|
-2 - on failure to reallocate more memory
|
||||||
|
-3 - bad filename arg
|
||||||
|
-4 - bad item arg
|
||||||
|
*/
|
||||||
|
#define cJSON_LOAD_BUFFER_INCREMENT 2048
|
||||||
|
CJSON_PUBLIC(int) cJSON_loadJSONfile(const char *filename, cJSON **item, char *errorMessage) {
|
||||||
|
|
||||||
|
FILE *fptr;
|
||||||
|
char *buffer;
|
||||||
|
size_t bufferSize = cJSON_LOAD_BUFFER_INCREMENT;
|
||||||
|
size_t bytesRead = 0;
|
||||||
|
int i = 0;
|
||||||
|
|
||||||
|
/* Sanity check the item arg*/
|
||||||
|
if (item == NULL) {
|
||||||
|
return -4;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Sanity check the filename arg */
|
||||||
|
if (filename == NULL) {
|
||||||
|
if (errorMessage != NULL)
|
||||||
|
{
|
||||||
|
sprintf(errorMessage, "The filename was NULL");
|
||||||
|
}
|
||||||
|
return -3;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Open the file */
|
||||||
|
fptr = fopen(filename, "r");
|
||||||
|
if (fptr == NULL) {
|
||||||
|
if (errorMessage != NULL)
|
||||||
|
{
|
||||||
|
sprintf(errorMessage, "Cannot open the file '%s' to read.", filename);
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Allocate the buffer to it's initial size */
|
||||||
|
buffer = (char *)malloc(bufferSize + cJSON_LOAD_BUFFER_INCREMENT);
|
||||||
|
if (buffer == NULL) {
|
||||||
|
if (errorMessage != NULL)
|
||||||
|
{
|
||||||
|
sprintf(errorMessage, "Memory allocation error while reading '%s'.", filename);
|
||||||
|
}
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
/* Read in the file until there's nothing left to read
|
||||||
|
Keep reading in the file, expanding our buffer if needed */
|
||||||
|
while ((bytesRead = fread(buffer + (cJSON_LOAD_BUFFER_INCREMENT * i), 1, cJSON_LOAD_BUFFER_INCREMENT, fptr)) > 0) {
|
||||||
|
if (bytesRead == sizeof(buffer))
|
||||||
|
{
|
||||||
|
/* We filled the buffer, realloc the final buffer so it's larger */
|
||||||
|
buffer = (char *)realloc(buffer, bufferSize + cJSON_LOAD_BUFFER_INCREMENT);
|
||||||
|
if (buffer == NULL)
|
||||||
|
{
|
||||||
|
if (errorMessage != NULL)
|
||||||
|
{
|
||||||
|
sprintf(errorMessage, "Memory allocation error while reading '%s'.", filename);
|
||||||
|
}
|
||||||
|
return -2;
|
||||||
|
}
|
||||||
|
/* Keep track of where we are in our enlarged buffer */
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* We're done with the file */
|
||||||
|
fclose(fptr);
|
||||||
|
|
||||||
|
/* Clear any existing structure in memory */
|
||||||
|
if (*item != NULL)
|
||||||
|
{
|
||||||
|
cJSON_Delete(*item);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Parse the final buffer */
|
||||||
|
*item = cJSON_Parse(buffer);
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
33
cJSON.h
33
cJSON.h
@ -273,6 +273,39 @@ CJSON_PUBLIC(double) cJSON_SetNumberHelper(cJSON *object, double number);
|
|||||||
CJSON_PUBLIC(void *) cJSON_malloc(size_t size);
|
CJSON_PUBLIC(void *) cJSON_malloc(size_t size);
|
||||||
CJSON_PUBLIC(void) cJSON_free(void *object);
|
CJSON_PUBLIC(void) cJSON_free(void *object);
|
||||||
|
|
||||||
|
/* This will write out a json object to the specified file name
|
||||||
|
|
||||||
|
Args:
|
||||||
|
filename - the name of the file to load
|
||||||
|
**item - a ptr to the cJSON structure to store the parsed root item in
|
||||||
|
**errorMessage - an optional field to return an error message
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
1 - on success
|
||||||
|
-1 - on failure to open the file to write
|
||||||
|
-2 - on failure to stringify the cJSON object
|
||||||
|
*/
|
||||||
|
CJSON_PUBLIC(int) cJSON_saveJSONfile(const char *filename, cJSON *item, char *errorMessage);
|
||||||
|
|
||||||
|
/* This will load and parse a json file.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
filename - the name of the file to load
|
||||||
|
**item - a ptr to the cJSON structure to store the parsed root item in.
|
||||||
|
If an object already exists at this ptr value, it will be deleted
|
||||||
|
and overwritten.
|
||||||
|
**errorMessage - an optional field to return an error message
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
1 - on success
|
||||||
|
0 - file not found
|
||||||
|
-1 - on failure to open the file to read
|
||||||
|
-2 - on intial memory allocation failure
|
||||||
|
-3 - on failure to reallocate more memory
|
||||||
|
-4 - bad filename arg
|
||||||
|
*/
|
||||||
|
CJSON_PUBLIC(int) cJSON_loadJSONfile(const char *filename, cJSON **item, char *errorMessage);
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
@ -57,6 +57,7 @@ if(ENABLE_CJSON_TEST)
|
|||||||
compare_tests
|
compare_tests
|
||||||
cjson_add
|
cjson_add
|
||||||
readme_examples
|
readme_examples
|
||||||
|
file_functions
|
||||||
)
|
)
|
||||||
|
|
||||||
option(ENABLE_VALGRIND OFF "Enable the valgrind memory checker for the tests.")
|
option(ENABLE_VALGRIND OFF "Enable the valgrind memory checker for the tests.")
|
||||||
|
183
tests/file_functions.c
Normal file
183
tests/file_functions.c
Normal file
@ -0,0 +1,183 @@
|
|||||||
|
/*
|
||||||
|
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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* These are unit tests for the cJSON_saveJSONfile & cJSON_loadJSONfile functions */
|
||||||
|
#include "unity/examples/unity_config.h"
|
||||||
|
#include "unity/src/unity.h"
|
||||||
|
#include "common.h"
|
||||||
|
|
||||||
|
#define TEST_VALUE "test value"
|
||||||
|
|
||||||
|
static void save_file_should_succeed(void)
|
||||||
|
{
|
||||||
|
char *errorMessage = NULL;
|
||||||
|
cJSON *object;
|
||||||
|
int rc = 0;
|
||||||
|
|
||||||
|
errorMessage = (char *)malloc(512);
|
||||||
|
object = cJSON_CreateObject();
|
||||||
|
cJSON_AddNumberToObject(object, TEST_VALUE, 99);
|
||||||
|
|
||||||
|
rc = cJSON_saveJSONfile("./testfile", object, errorMessage);
|
||||||
|
TEST_ASSERT_EQUAL(1, rc);
|
||||||
|
free(errorMessage);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void save_file_with_bad_path(void)
|
||||||
|
{
|
||||||
|
char *errorMessage = NULL;
|
||||||
|
cJSON *object;
|
||||||
|
int rc = 0;
|
||||||
|
|
||||||
|
errorMessage = (char *)malloc(512);
|
||||||
|
object = cJSON_CreateObject();
|
||||||
|
|
||||||
|
rc = cJSON_saveJSONfile("/!@#$!this doesn't exist/testfile", object, errorMessage);
|
||||||
|
TEST_ASSERT_EQUAL(-1, rc);
|
||||||
|
TEST_ASSERT_EQUAL_STRING("Cannot open the file '/!@#$!this doesn't exist/testfile' to write.", errorMessage);
|
||||||
|
free(errorMessage);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void save_file_with_bad_path_no_error_msg(void)
|
||||||
|
{
|
||||||
|
char *errorMessage = NULL;
|
||||||
|
cJSON *object;
|
||||||
|
int rc = 0;
|
||||||
|
|
||||||
|
object = cJSON_CreateObject();
|
||||||
|
|
||||||
|
rc = cJSON_saveJSONfile("/!@#$!this doesn't exist/testfile", object, errorMessage);
|
||||||
|
TEST_ASSERT_EQUAL(-1, rc);
|
||||||
|
TEST_ASSERT_EQUAL_STRING(NULL, errorMessage);
|
||||||
|
free(errorMessage);
|
||||||
|
cJSON_Delete(object);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void save_file_with_NULL_object(void)
|
||||||
|
{
|
||||||
|
char *errorMessage = NULL;
|
||||||
|
int rc = 0;
|
||||||
|
|
||||||
|
errorMessage = (char *)malloc(512);
|
||||||
|
rc = cJSON_saveJSONfile("./testfile", NULL, errorMessage);
|
||||||
|
TEST_ASSERT_EQUAL(0, rc);
|
||||||
|
free(errorMessage);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void load_file_successs_preexisting_object(void)
|
||||||
|
{
|
||||||
|
char *errorMessage = NULL;
|
||||||
|
cJSON *object = NULL;
|
||||||
|
int rc = 0;
|
||||||
|
double testValue = 0;
|
||||||
|
cJSON *numberJson = NULL;
|
||||||
|
|
||||||
|
object = cJSON_CreateObject();
|
||||||
|
|
||||||
|
errorMessage = (char *)malloc(512);
|
||||||
|
rc = cJSON_loadJSONfile("./testfile", &object, errorMessage);
|
||||||
|
|
||||||
|
numberJson = cJSON_GetObjectItem(object, TEST_VALUE);
|
||||||
|
testValue = cJSON_GetNumberValue(numberJson);
|
||||||
|
|
||||||
|
TEST_ASSERT_EQUAL(1, rc);
|
||||||
|
TEST_ASSERT_EQUAL(99, testValue);
|
||||||
|
free(errorMessage);
|
||||||
|
cJSON_Delete(object);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void load_file_success(void)
|
||||||
|
{
|
||||||
|
char *errorMessage = NULL;
|
||||||
|
cJSON *object = NULL;
|
||||||
|
int rc = 0;
|
||||||
|
double testValue = 0;
|
||||||
|
cJSON *numberJson = NULL;
|
||||||
|
|
||||||
|
errorMessage = (char *)malloc(512);
|
||||||
|
rc = cJSON_loadJSONfile("./testfile", &object, errorMessage);
|
||||||
|
|
||||||
|
numberJson = cJSON_GetObjectItem(object, TEST_VALUE);
|
||||||
|
testValue = cJSON_GetNumberValue(numberJson);
|
||||||
|
|
||||||
|
TEST_ASSERT_EQUAL(1, rc);
|
||||||
|
TEST_ASSERT_EQUAL(99, testValue);
|
||||||
|
free(errorMessage);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void load_file_bad_filename(void)
|
||||||
|
{
|
||||||
|
char *errorMessage = NULL;
|
||||||
|
cJSON *object = NULL;
|
||||||
|
int rc = 0;
|
||||||
|
|
||||||
|
errorMessage = (char *)malloc(512);
|
||||||
|
rc = cJSON_loadJSONfile("./notfound", &object, errorMessage);
|
||||||
|
TEST_ASSERT_EQUAL_STRING("Cannot open the file './notfound' to read.",errorMessage);
|
||||||
|
TEST_ASSERT_EQUAL(0, rc);
|
||||||
|
free(errorMessage);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void load_file_NULL_filename(void)
|
||||||
|
{
|
||||||
|
char *errorMessage = NULL;
|
||||||
|
cJSON *object = NULL;
|
||||||
|
int rc = 0;
|
||||||
|
|
||||||
|
errorMessage = (char *)malloc(512);
|
||||||
|
rc = cJSON_loadJSONfile(NULL, &object, errorMessage);
|
||||||
|
|
||||||
|
TEST_ASSERT_EQUAL(-3, rc);
|
||||||
|
TEST_ASSERT_EQUAL_STRING("The filename was NULL",errorMessage);
|
||||||
|
free(errorMessage);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void load_file_NULL_object(void)
|
||||||
|
{
|
||||||
|
char *errorMessage = NULL;
|
||||||
|
int rc = 0;
|
||||||
|
|
||||||
|
errorMessage = (char *)malloc(512);
|
||||||
|
rc = cJSON_loadJSONfile("./testfile", NULL, errorMessage);
|
||||||
|
|
||||||
|
TEST_ASSERT_EQUAL(-4, rc);
|
||||||
|
free(errorMessage);
|
||||||
|
}
|
||||||
|
|
||||||
|
int main(void)
|
||||||
|
{
|
||||||
|
/* initialize cJSON item */
|
||||||
|
UNITY_BEGIN();
|
||||||
|
|
||||||
|
RUN_TEST(save_file_with_bad_path);
|
||||||
|
RUN_TEST(save_file_with_bad_path_no_error_msg);
|
||||||
|
RUN_TEST(save_file_with_NULL_object);
|
||||||
|
RUN_TEST(save_file_should_succeed);
|
||||||
|
|
||||||
|
RUN_TEST(load_file_success);
|
||||||
|
RUN_TEST(load_file_successs_preexisting_object);
|
||||||
|
RUN_TEST(load_file_bad_filename);
|
||||||
|
RUN_TEST(load_file_NULL_filename);
|
||||||
|
RUN_TEST(load_file_NULL_object);
|
||||||
|
|
||||||
|
return UNITY_END();
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user