Important changes to repositories hosted on mbed.com
Mbed hosted mercurial repositories are deprecated and are due to be permanently deleted in July 2026.
To keep a copy of this software download the repository Zip archive or clone locally using Mercurial.
It is also possible to export all your personal repositories from the account settings page.
deps/parson/parson.c@0:f7f1f0d76dd6, 2018-08-23 (annotated)
- Committer:
- XinZhangMS
- Date:
- Thu Aug 23 06:52:14 2018 +0000
- Revision:
- 0:f7f1f0d76dd6
azure-c-sdk for mbed os supporting NUCLEO_F767ZI
Who changed what in which revision?
| User | Revision | Line number | New contents of line |
|---|---|---|---|
| XinZhangMS | 0:f7f1f0d76dd6 | 1 | /* |
| XinZhangMS | 0:f7f1f0d76dd6 | 2 | Parson ( http://kgabis.github.com/parson/ ) |
| XinZhangMS | 0:f7f1f0d76dd6 | 3 | Copyright (c) 2012 - 2017 Krzysztof Gabis |
| XinZhangMS | 0:f7f1f0d76dd6 | 4 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 5 | Permission is hereby granted, free of charge, to any person obtaining a copy |
| XinZhangMS | 0:f7f1f0d76dd6 | 6 | of this software and associated documentation files (the "Software"), to deal |
| XinZhangMS | 0:f7f1f0d76dd6 | 7 | in the Software without restriction, including without limitation the rights |
| XinZhangMS | 0:f7f1f0d76dd6 | 8 | to use, copy, modify, merge, publish, distribute, sublicense, and/or sell |
| XinZhangMS | 0:f7f1f0d76dd6 | 9 | copies of the Software, and to permit persons to whom the Software is |
| XinZhangMS | 0:f7f1f0d76dd6 | 10 | furnished to do so, subject to the following conditions: |
| XinZhangMS | 0:f7f1f0d76dd6 | 11 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 12 | The above copyright notice and this permission notice shall be included in |
| XinZhangMS | 0:f7f1f0d76dd6 | 13 | all copies or substantial portions of the Software. |
| XinZhangMS | 0:f7f1f0d76dd6 | 14 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 15 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |
| XinZhangMS | 0:f7f1f0d76dd6 | 16 | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
| XinZhangMS | 0:f7f1f0d76dd6 | 17 | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE |
| XinZhangMS | 0:f7f1f0d76dd6 | 18 | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |
| XinZhangMS | 0:f7f1f0d76dd6 | 19 | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, |
| XinZhangMS | 0:f7f1f0d76dd6 | 20 | OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN |
| XinZhangMS | 0:f7f1f0d76dd6 | 21 | THE SOFTWARE. |
| XinZhangMS | 0:f7f1f0d76dd6 | 22 | */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 23 | #ifdef _MSC_VER |
| XinZhangMS | 0:f7f1f0d76dd6 | 24 | #ifndef _CRT_SECURE_NO_WARNINGS |
| XinZhangMS | 0:f7f1f0d76dd6 | 25 | #define _CRT_SECURE_NO_WARNINGS |
| XinZhangMS | 0:f7f1f0d76dd6 | 26 | #endif /* _CRT_SECURE_NO_WARNINGS */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 27 | #endif /* _MSC_VER */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 28 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 29 | #include "parson.h" |
| XinZhangMS | 0:f7f1f0d76dd6 | 30 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 31 | #include <stdio.h> |
| XinZhangMS | 0:f7f1f0d76dd6 | 32 | #include <stdlib.h> |
| XinZhangMS | 0:f7f1f0d76dd6 | 33 | #include <string.h> |
| XinZhangMS | 0:f7f1f0d76dd6 | 34 | #include <ctype.h> |
| XinZhangMS | 0:f7f1f0d76dd6 | 35 | #include <math.h> |
| XinZhangMS | 0:f7f1f0d76dd6 | 36 | #include <errno.h> |
| XinZhangMS | 0:f7f1f0d76dd6 | 37 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 38 | /* Apparently sscanf is not implemented in some "standard" libraries, so don't use it, if you |
| XinZhangMS | 0:f7f1f0d76dd6 | 39 | * don't have to. */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 40 | #define sscanf THINK_TWICE_ABOUT_USING_SSCANF |
| XinZhangMS | 0:f7f1f0d76dd6 | 41 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 42 | #define STARTING_CAPACITY 16 |
| XinZhangMS | 0:f7f1f0d76dd6 | 43 | #define MAX_NESTING 2048 |
| XinZhangMS | 0:f7f1f0d76dd6 | 44 | #define FLOAT_FORMAT "%1.17g" |
| XinZhangMS | 0:f7f1f0d76dd6 | 45 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 46 | #define SIZEOF_TOKEN(a) (sizeof(a) - 1) |
| XinZhangMS | 0:f7f1f0d76dd6 | 47 | #define SKIP_CHAR(str) ((*str)++) |
| XinZhangMS | 0:f7f1f0d76dd6 | 48 | #define SKIP_WHITESPACES(str) while (isspace((unsigned char)(**str))) { SKIP_CHAR(str); } |
| XinZhangMS | 0:f7f1f0d76dd6 | 49 | #define MAX(a, b) ((a) > (b) ? (a) : (b)) |
| XinZhangMS | 0:f7f1f0d76dd6 | 50 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 51 | #undef malloc |
| XinZhangMS | 0:f7f1f0d76dd6 | 52 | #undef free |
| XinZhangMS | 0:f7f1f0d76dd6 | 53 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 54 | static JSON_Malloc_Function parson_malloc = malloc; |
| XinZhangMS | 0:f7f1f0d76dd6 | 55 | static JSON_Free_Function parson_free = free; |
| XinZhangMS | 0:f7f1f0d76dd6 | 56 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 57 | #define IS_CONT(b) (((unsigned char)(b) & 0xC0) == 0x80) /* is utf-8 continuation byte */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 58 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 59 | /* Type definitions */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 60 | typedef union json_value_value { |
| XinZhangMS | 0:f7f1f0d76dd6 | 61 | char *string; |
| XinZhangMS | 0:f7f1f0d76dd6 | 62 | double number; |
| XinZhangMS | 0:f7f1f0d76dd6 | 63 | JSON_Object *object; |
| XinZhangMS | 0:f7f1f0d76dd6 | 64 | JSON_Array *array; |
| XinZhangMS | 0:f7f1f0d76dd6 | 65 | int boolean; |
| XinZhangMS | 0:f7f1f0d76dd6 | 66 | int null; |
| XinZhangMS | 0:f7f1f0d76dd6 | 67 | } JSON_Value_Value; |
| XinZhangMS | 0:f7f1f0d76dd6 | 68 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 69 | struct json_value_t { |
| XinZhangMS | 0:f7f1f0d76dd6 | 70 | JSON_Value *parent; |
| XinZhangMS | 0:f7f1f0d76dd6 | 71 | JSON_Value_Type type; |
| XinZhangMS | 0:f7f1f0d76dd6 | 72 | JSON_Value_Value value; |
| XinZhangMS | 0:f7f1f0d76dd6 | 73 | }; |
| XinZhangMS | 0:f7f1f0d76dd6 | 74 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 75 | struct json_object_t { |
| XinZhangMS | 0:f7f1f0d76dd6 | 76 | JSON_Value *wrapping_value; |
| XinZhangMS | 0:f7f1f0d76dd6 | 77 | char **names; |
| XinZhangMS | 0:f7f1f0d76dd6 | 78 | JSON_Value **values; |
| XinZhangMS | 0:f7f1f0d76dd6 | 79 | size_t count; |
| XinZhangMS | 0:f7f1f0d76dd6 | 80 | size_t capacity; |
| XinZhangMS | 0:f7f1f0d76dd6 | 81 | }; |
| XinZhangMS | 0:f7f1f0d76dd6 | 82 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 83 | struct json_array_t { |
| XinZhangMS | 0:f7f1f0d76dd6 | 84 | JSON_Value *wrapping_value; |
| XinZhangMS | 0:f7f1f0d76dd6 | 85 | JSON_Value **items; |
| XinZhangMS | 0:f7f1f0d76dd6 | 86 | size_t count; |
| XinZhangMS | 0:f7f1f0d76dd6 | 87 | size_t capacity; |
| XinZhangMS | 0:f7f1f0d76dd6 | 88 | }; |
| XinZhangMS | 0:f7f1f0d76dd6 | 89 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 90 | /* Various */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 91 | static char * read_file(const char *filename); |
| XinZhangMS | 0:f7f1f0d76dd6 | 92 | static void remove_comments(char *string, const char *start_token, const char *end_token); |
| XinZhangMS | 0:f7f1f0d76dd6 | 93 | static char * parson_strndup(const char *string, size_t n); |
| XinZhangMS | 0:f7f1f0d76dd6 | 94 | static char * parson_strdup(const char *string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 95 | static int hex_char_to_int(char c); |
| XinZhangMS | 0:f7f1f0d76dd6 | 96 | static int parse_utf16_hex(const char *string, unsigned int *result); |
| XinZhangMS | 0:f7f1f0d76dd6 | 97 | static int num_bytes_in_utf8_sequence(unsigned char c); |
| XinZhangMS | 0:f7f1f0d76dd6 | 98 | static int verify_utf8_sequence(const unsigned char *string, int *len); |
| XinZhangMS | 0:f7f1f0d76dd6 | 99 | static int is_valid_utf8(const char *string, size_t string_len); |
| XinZhangMS | 0:f7f1f0d76dd6 | 100 | static int is_decimal(const char *string, size_t length); |
| XinZhangMS | 0:f7f1f0d76dd6 | 101 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 102 | /* JSON Object */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 103 | static JSON_Object * json_object_init(JSON_Value *wrapping_value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 104 | static JSON_Status json_object_add(JSON_Object *object, const char *name, JSON_Value *value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 105 | static JSON_Status json_object_resize(JSON_Object *object, size_t new_capacity); |
| XinZhangMS | 0:f7f1f0d76dd6 | 106 | static JSON_Value * json_object_nget_value(const JSON_Object *object, const char *name, size_t n); |
| XinZhangMS | 0:f7f1f0d76dd6 | 107 | static void json_object_free(JSON_Object *object); |
| XinZhangMS | 0:f7f1f0d76dd6 | 108 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 109 | /* JSON Array */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 110 | static JSON_Array * json_array_init(JSON_Value *wrapping_value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 111 | static JSON_Status json_array_add(JSON_Array *array, JSON_Value *value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 112 | static JSON_Status json_array_resize(JSON_Array *array, size_t new_capacity); |
| XinZhangMS | 0:f7f1f0d76dd6 | 113 | static void json_array_free(JSON_Array *array); |
| XinZhangMS | 0:f7f1f0d76dd6 | 114 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 115 | /* JSON Value */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 116 | static JSON_Value * json_value_init_string_no_copy(char *string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 117 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 118 | /* Parser */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 119 | static JSON_Status skip_quotes(const char **string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 120 | static int parse_utf16(const char **unprocessed, char **processed); |
| XinZhangMS | 0:f7f1f0d76dd6 | 121 | static char * process_string(const char *input, size_t len); |
| XinZhangMS | 0:f7f1f0d76dd6 | 122 | static char * get_quoted_string(const char **string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 123 | static JSON_Value * parse_object_value(const char **string, size_t nesting); |
| XinZhangMS | 0:f7f1f0d76dd6 | 124 | static JSON_Value * parse_array_value(const char **string, size_t nesting); |
| XinZhangMS | 0:f7f1f0d76dd6 | 125 | static JSON_Value * parse_string_value(const char **string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 126 | static JSON_Value * parse_boolean_value(const char **string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 127 | static JSON_Value * parse_number_value(const char **string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 128 | static JSON_Value * parse_null_value(const char **string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 129 | static JSON_Value * parse_value(const char **string, size_t nesting); |
| XinZhangMS | 0:f7f1f0d76dd6 | 130 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 131 | /* Serialization */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 132 | static int json_serialize_to_buffer_r(const JSON_Value *value, char *buf, int level, int is_pretty, char *num_buf); |
| XinZhangMS | 0:f7f1f0d76dd6 | 133 | static int json_serialize_string(const char *string, char *buf); |
| XinZhangMS | 0:f7f1f0d76dd6 | 134 | static int append_indent(char *buf, int level); |
| XinZhangMS | 0:f7f1f0d76dd6 | 135 | static int append_string(char *buf, const char *string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 136 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 137 | /* Various */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 138 | static char * parson_strndup(const char *string, size_t n) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 139 | char *output_string = (char*)parson_malloc(n + 1); |
| XinZhangMS | 0:f7f1f0d76dd6 | 140 | if (!output_string) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 141 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 142 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 143 | output_string[n] = '\0'; |
| XinZhangMS | 0:f7f1f0d76dd6 | 144 | strncpy(output_string, string, n); |
| XinZhangMS | 0:f7f1f0d76dd6 | 145 | return output_string; |
| XinZhangMS | 0:f7f1f0d76dd6 | 146 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 147 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 148 | static char * parson_strdup(const char *string) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 149 | return parson_strndup(string, strlen(string)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 150 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 151 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 152 | static int hex_char_to_int(char c) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 153 | if (c >= '0' && c <= '9') { |
| XinZhangMS | 0:f7f1f0d76dd6 | 154 | return c - '0'; |
| XinZhangMS | 0:f7f1f0d76dd6 | 155 | } else if (c >= 'a' && c <= 'f') { |
| XinZhangMS | 0:f7f1f0d76dd6 | 156 | return c - 'a' + 10; |
| XinZhangMS | 0:f7f1f0d76dd6 | 157 | } else if (c >= 'A' && c <= 'F') { |
| XinZhangMS | 0:f7f1f0d76dd6 | 158 | return c - 'A' + 10; |
| XinZhangMS | 0:f7f1f0d76dd6 | 159 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 160 | return -1; |
| XinZhangMS | 0:f7f1f0d76dd6 | 161 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 162 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 163 | static int parse_utf16_hex(const char *s, unsigned int *result) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 164 | int x1, x2, x3, x4; |
| XinZhangMS | 0:f7f1f0d76dd6 | 165 | if (s[0] == '\0' || s[1] == '\0' || s[2] == '\0' || s[3] == '\0') { |
| XinZhangMS | 0:f7f1f0d76dd6 | 166 | return 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 167 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 168 | x1 = hex_char_to_int(s[0]); |
| XinZhangMS | 0:f7f1f0d76dd6 | 169 | x2 = hex_char_to_int(s[1]); |
| XinZhangMS | 0:f7f1f0d76dd6 | 170 | x3 = hex_char_to_int(s[2]); |
| XinZhangMS | 0:f7f1f0d76dd6 | 171 | x4 = hex_char_to_int(s[3]); |
| XinZhangMS | 0:f7f1f0d76dd6 | 172 | if (x1 == -1 || x2 == -1 || x3 == -1 || x4 == -1) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 173 | return 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 174 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 175 | *result = (unsigned int)((x1 << 12) | (x2 << 8) | (x3 << 4) | x4); |
| XinZhangMS | 0:f7f1f0d76dd6 | 176 | return 1; |
| XinZhangMS | 0:f7f1f0d76dd6 | 177 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 178 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 179 | static int num_bytes_in_utf8_sequence(unsigned char c) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 180 | if (c == 0xC0 || c == 0xC1 || c > 0xF4 || IS_CONT(c)) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 181 | return 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 182 | } else if ((c & 0x80) == 0) { /* 0xxxxxxx */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 183 | return 1; |
| XinZhangMS | 0:f7f1f0d76dd6 | 184 | } else if ((c & 0xE0) == 0xC0) { /* 110xxxxx */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 185 | return 2; |
| XinZhangMS | 0:f7f1f0d76dd6 | 186 | } else if ((c & 0xF0) == 0xE0) { /* 1110xxxx */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 187 | return 3; |
| XinZhangMS | 0:f7f1f0d76dd6 | 188 | } else if ((c & 0xF8) == 0xF0) { /* 11110xxx */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 189 | return 4; |
| XinZhangMS | 0:f7f1f0d76dd6 | 190 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 191 | return 0; /* won't happen */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 192 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 193 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 194 | static int verify_utf8_sequence(const unsigned char *string, int *len) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 195 | unsigned int cp = 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 196 | *len = num_bytes_in_utf8_sequence(string[0]); |
| XinZhangMS | 0:f7f1f0d76dd6 | 197 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 198 | if (*len == 1) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 199 | cp = string[0]; |
| XinZhangMS | 0:f7f1f0d76dd6 | 200 | } else if (*len == 2 && IS_CONT(string[1])) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 201 | cp = string[0] & 0x1F; |
| XinZhangMS | 0:f7f1f0d76dd6 | 202 | cp = (cp << 6) | (string[1] & 0x3F); |
| XinZhangMS | 0:f7f1f0d76dd6 | 203 | } else if (*len == 3 && IS_CONT(string[1]) && IS_CONT(string[2])) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 204 | cp = ((unsigned char)string[0]) & 0xF; |
| XinZhangMS | 0:f7f1f0d76dd6 | 205 | cp = (cp << 6) | (string[1] & 0x3F); |
| XinZhangMS | 0:f7f1f0d76dd6 | 206 | cp = (cp << 6) | (string[2] & 0x3F); |
| XinZhangMS | 0:f7f1f0d76dd6 | 207 | } else if (*len == 4 && IS_CONT(string[1]) && IS_CONT(string[2]) && IS_CONT(string[3])) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 208 | cp = string[0] & 0x7; |
| XinZhangMS | 0:f7f1f0d76dd6 | 209 | cp = (cp << 6) | (string[1] & 0x3F); |
| XinZhangMS | 0:f7f1f0d76dd6 | 210 | cp = (cp << 6) | (string[2] & 0x3F); |
| XinZhangMS | 0:f7f1f0d76dd6 | 211 | cp = (cp << 6) | (string[3] & 0x3F); |
| XinZhangMS | 0:f7f1f0d76dd6 | 212 | } else { |
| XinZhangMS | 0:f7f1f0d76dd6 | 213 | return 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 214 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 215 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 216 | /* overlong encodings */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 217 | if ((cp < 0x80 && *len > 1) || |
| XinZhangMS | 0:f7f1f0d76dd6 | 218 | (cp < 0x800 && *len > 2) || |
| XinZhangMS | 0:f7f1f0d76dd6 | 219 | (cp < 0x10000 && *len > 3)) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 220 | return 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 221 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 222 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 223 | /* invalid unicode */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 224 | if (cp > 0x10FFFF) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 225 | return 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 226 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 227 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 228 | /* surrogate halves */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 229 | if (cp >= 0xD800 && cp <= 0xDFFF) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 230 | return 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 231 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 232 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 233 | return 1; |
| XinZhangMS | 0:f7f1f0d76dd6 | 234 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 235 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 236 | static int is_valid_utf8(const char *string, size_t string_len) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 237 | int len = 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 238 | const char *string_end = string + string_len; |
| XinZhangMS | 0:f7f1f0d76dd6 | 239 | while (string < string_end) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 240 | if (!verify_utf8_sequence((const unsigned char*)string, &len)) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 241 | return 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 242 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 243 | string += len; |
| XinZhangMS | 0:f7f1f0d76dd6 | 244 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 245 | return 1; |
| XinZhangMS | 0:f7f1f0d76dd6 | 246 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 247 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 248 | static int is_decimal(const char *string, size_t length) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 249 | if (length > 1 && string[0] == '0' && string[1] != '.') { |
| XinZhangMS | 0:f7f1f0d76dd6 | 250 | return 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 251 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 252 | if (length > 2 && !strncmp(string, "-0", 2) && string[2] != '.') { |
| XinZhangMS | 0:f7f1f0d76dd6 | 253 | return 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 254 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 255 | while (length--) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 256 | if (strchr("xX", string[length])) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 257 | return 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 258 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 259 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 260 | return 1; |
| XinZhangMS | 0:f7f1f0d76dd6 | 261 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 262 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 263 | static char * read_file(const char * filename) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 264 | FILE *fp = fopen(filename, "r"); |
| XinZhangMS | 0:f7f1f0d76dd6 | 265 | size_t file_size; |
| XinZhangMS | 0:f7f1f0d76dd6 | 266 | long pos; |
| XinZhangMS | 0:f7f1f0d76dd6 | 267 | char *file_contents; |
| XinZhangMS | 0:f7f1f0d76dd6 | 268 | if (!fp) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 269 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 270 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 271 | fseek(fp, 0L, SEEK_END); |
| XinZhangMS | 0:f7f1f0d76dd6 | 272 | pos = ftell(fp); |
| XinZhangMS | 0:f7f1f0d76dd6 | 273 | if (pos < 0) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 274 | fclose(fp); |
| XinZhangMS | 0:f7f1f0d76dd6 | 275 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 276 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 277 | file_size = pos; |
| XinZhangMS | 0:f7f1f0d76dd6 | 278 | rewind(fp); |
| XinZhangMS | 0:f7f1f0d76dd6 | 279 | file_contents = (char*)parson_malloc(sizeof(char) * (file_size + 1)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 280 | if (!file_contents) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 281 | fclose(fp); |
| XinZhangMS | 0:f7f1f0d76dd6 | 282 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 283 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 284 | if (fread(file_contents, file_size, 1, fp) < 1) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 285 | if (ferror(fp)) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 286 | fclose(fp); |
| XinZhangMS | 0:f7f1f0d76dd6 | 287 | parson_free(file_contents); |
| XinZhangMS | 0:f7f1f0d76dd6 | 288 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 289 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 290 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 291 | fclose(fp); |
| XinZhangMS | 0:f7f1f0d76dd6 | 292 | file_contents[file_size] = '\0'; |
| XinZhangMS | 0:f7f1f0d76dd6 | 293 | return file_contents; |
| XinZhangMS | 0:f7f1f0d76dd6 | 294 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 295 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 296 | static void remove_comments(char *string, const char *start_token, const char *end_token) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 297 | int in_string = 0, escaped = 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 298 | size_t i; |
| XinZhangMS | 0:f7f1f0d76dd6 | 299 | char *ptr = NULL, current_char; |
| XinZhangMS | 0:f7f1f0d76dd6 | 300 | size_t start_token_len = strlen(start_token); |
| XinZhangMS | 0:f7f1f0d76dd6 | 301 | size_t end_token_len = strlen(end_token); |
| XinZhangMS | 0:f7f1f0d76dd6 | 302 | if (start_token_len == 0 || end_token_len == 0) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 303 | return; |
| XinZhangMS | 0:f7f1f0d76dd6 | 304 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 305 | while ((current_char = *string) != '\0') { |
| XinZhangMS | 0:f7f1f0d76dd6 | 306 | if (current_char == '\\' && !escaped) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 307 | escaped = 1; |
| XinZhangMS | 0:f7f1f0d76dd6 | 308 | string++; |
| XinZhangMS | 0:f7f1f0d76dd6 | 309 | continue; |
| XinZhangMS | 0:f7f1f0d76dd6 | 310 | } else if (current_char == '\"' && !escaped) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 311 | in_string = !in_string; |
| XinZhangMS | 0:f7f1f0d76dd6 | 312 | } else if (!in_string && strncmp(string, start_token, start_token_len) == 0) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 313 | for(i = 0; i < start_token_len; i++) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 314 | string[i] = ' '; |
| XinZhangMS | 0:f7f1f0d76dd6 | 315 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 316 | string = string + start_token_len; |
| XinZhangMS | 0:f7f1f0d76dd6 | 317 | ptr = strstr(string, end_token); |
| XinZhangMS | 0:f7f1f0d76dd6 | 318 | if (!ptr) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 319 | return; |
| XinZhangMS | 0:f7f1f0d76dd6 | 320 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 321 | for (i = 0; i < (ptr - string) + end_token_len; i++) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 322 | string[i] = ' '; |
| XinZhangMS | 0:f7f1f0d76dd6 | 323 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 324 | string = ptr + end_token_len - 1; |
| XinZhangMS | 0:f7f1f0d76dd6 | 325 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 326 | escaped = 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 327 | string++; |
| XinZhangMS | 0:f7f1f0d76dd6 | 328 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 329 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 330 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 331 | /* JSON Object */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 332 | static JSON_Object * json_object_init(JSON_Value *wrapping_value) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 333 | JSON_Object *new_obj = (JSON_Object*)parson_malloc(sizeof(JSON_Object)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 334 | if (new_obj == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 335 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 336 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 337 | new_obj->wrapping_value = wrapping_value; |
| XinZhangMS | 0:f7f1f0d76dd6 | 338 | new_obj->names = (char**)NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 339 | new_obj->values = (JSON_Value**)NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 340 | new_obj->capacity = 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 341 | new_obj->count = 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 342 | return new_obj; |
| XinZhangMS | 0:f7f1f0d76dd6 | 343 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 344 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 345 | static JSON_Status json_object_add(JSON_Object *object, const char *name, JSON_Value *value) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 346 | size_t index = 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 347 | if (object == NULL || name == NULL || value == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 348 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 349 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 350 | if (json_object_get_value(object, name) != NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 351 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 352 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 353 | if (object->count >= object->capacity) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 354 | size_t new_capacity = MAX(object->capacity * 2, STARTING_CAPACITY); |
| XinZhangMS | 0:f7f1f0d76dd6 | 355 | if (json_object_resize(object, new_capacity) == JSONFailure) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 356 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 357 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 358 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 359 | index = object->count; |
| XinZhangMS | 0:f7f1f0d76dd6 | 360 | object->names[index] = parson_strdup(name); |
| XinZhangMS | 0:f7f1f0d76dd6 | 361 | if (object->names[index] == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 362 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 363 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 364 | value->parent = json_object_get_wrapping_value(object); |
| XinZhangMS | 0:f7f1f0d76dd6 | 365 | object->values[index] = value; |
| XinZhangMS | 0:f7f1f0d76dd6 | 366 | object->count++; |
| XinZhangMS | 0:f7f1f0d76dd6 | 367 | return JSONSuccess; |
| XinZhangMS | 0:f7f1f0d76dd6 | 368 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 369 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 370 | static JSON_Status json_object_resize(JSON_Object *object, size_t new_capacity) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 371 | char **temp_names = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 372 | JSON_Value **temp_values = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 373 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 374 | if ((object->names == NULL && object->values != NULL) || |
| XinZhangMS | 0:f7f1f0d76dd6 | 375 | (object->names != NULL && object->values == NULL) || |
| XinZhangMS | 0:f7f1f0d76dd6 | 376 | new_capacity == 0) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 377 | return JSONFailure; /* Shouldn't happen */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 378 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 379 | temp_names = (char**)parson_malloc(new_capacity * sizeof(char*)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 380 | if (temp_names == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 381 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 382 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 383 | temp_values = (JSON_Value**)parson_malloc(new_capacity * sizeof(JSON_Value*)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 384 | if (temp_values == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 385 | parson_free(temp_names); |
| XinZhangMS | 0:f7f1f0d76dd6 | 386 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 387 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 388 | if (object->names != NULL && object->values != NULL && object->count > 0) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 389 | memcpy(temp_names, object->names, object->count * sizeof(char*)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 390 | memcpy(temp_values, object->values, object->count * sizeof(JSON_Value*)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 391 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 392 | parson_free(object->names); |
| XinZhangMS | 0:f7f1f0d76dd6 | 393 | parson_free(object->values); |
| XinZhangMS | 0:f7f1f0d76dd6 | 394 | object->names = temp_names; |
| XinZhangMS | 0:f7f1f0d76dd6 | 395 | object->values = temp_values; |
| XinZhangMS | 0:f7f1f0d76dd6 | 396 | object->capacity = new_capacity; |
| XinZhangMS | 0:f7f1f0d76dd6 | 397 | return JSONSuccess; |
| XinZhangMS | 0:f7f1f0d76dd6 | 398 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 399 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 400 | static JSON_Value * json_object_nget_value(const JSON_Object *object, const char *name, size_t n) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 401 | size_t i, name_length; |
| XinZhangMS | 0:f7f1f0d76dd6 | 402 | for (i = 0; i < json_object_get_count(object); i++) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 403 | name_length = strlen(object->names[i]); |
| XinZhangMS | 0:f7f1f0d76dd6 | 404 | if (name_length != n) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 405 | continue; |
| XinZhangMS | 0:f7f1f0d76dd6 | 406 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 407 | if (strncmp(object->names[i], name, n) == 0) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 408 | return object->values[i]; |
| XinZhangMS | 0:f7f1f0d76dd6 | 409 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 410 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 411 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 412 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 413 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 414 | static void json_object_free(JSON_Object *object) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 415 | size_t i; |
| XinZhangMS | 0:f7f1f0d76dd6 | 416 | for (i = 0; i < object->count; i++) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 417 | parson_free(object->names[i]); |
| XinZhangMS | 0:f7f1f0d76dd6 | 418 | json_value_free(object->values[i]); |
| XinZhangMS | 0:f7f1f0d76dd6 | 419 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 420 | parson_free(object->names); |
| XinZhangMS | 0:f7f1f0d76dd6 | 421 | parson_free(object->values); |
| XinZhangMS | 0:f7f1f0d76dd6 | 422 | parson_free(object); |
| XinZhangMS | 0:f7f1f0d76dd6 | 423 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 424 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 425 | /* JSON Array */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 426 | static JSON_Array * json_array_init(JSON_Value *wrapping_value) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 427 | JSON_Array *new_array = (JSON_Array*)parson_malloc(sizeof(JSON_Array)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 428 | if (new_array == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 429 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 430 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 431 | new_array->wrapping_value = wrapping_value; |
| XinZhangMS | 0:f7f1f0d76dd6 | 432 | new_array->items = (JSON_Value**)NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 433 | new_array->capacity = 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 434 | new_array->count = 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 435 | return new_array; |
| XinZhangMS | 0:f7f1f0d76dd6 | 436 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 437 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 438 | static JSON_Status json_array_add(JSON_Array *array, JSON_Value *value) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 439 | if (array->count >= array->capacity) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 440 | size_t new_capacity = MAX(array->capacity * 2, STARTING_CAPACITY); |
| XinZhangMS | 0:f7f1f0d76dd6 | 441 | if (json_array_resize(array, new_capacity) == JSONFailure) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 442 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 443 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 444 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 445 | value->parent = json_array_get_wrapping_value(array); |
| XinZhangMS | 0:f7f1f0d76dd6 | 446 | array->items[array->count] = value; |
| XinZhangMS | 0:f7f1f0d76dd6 | 447 | array->count++; |
| XinZhangMS | 0:f7f1f0d76dd6 | 448 | return JSONSuccess; |
| XinZhangMS | 0:f7f1f0d76dd6 | 449 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 450 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 451 | static JSON_Status json_array_resize(JSON_Array *array, size_t new_capacity) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 452 | JSON_Value **new_items = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 453 | if (new_capacity == 0) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 454 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 455 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 456 | new_items = (JSON_Value**)parson_malloc(new_capacity * sizeof(JSON_Value*)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 457 | if (new_items == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 458 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 459 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 460 | if (array->items != NULL && array->count > 0) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 461 | memcpy(new_items, array->items, array->count * sizeof(JSON_Value*)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 462 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 463 | parson_free(array->items); |
| XinZhangMS | 0:f7f1f0d76dd6 | 464 | array->items = new_items; |
| XinZhangMS | 0:f7f1f0d76dd6 | 465 | array->capacity = new_capacity; |
| XinZhangMS | 0:f7f1f0d76dd6 | 466 | return JSONSuccess; |
| XinZhangMS | 0:f7f1f0d76dd6 | 467 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 468 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 469 | static void json_array_free(JSON_Array *array) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 470 | size_t i; |
| XinZhangMS | 0:f7f1f0d76dd6 | 471 | for (i = 0; i < array->count; i++) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 472 | json_value_free(array->items[i]); |
| XinZhangMS | 0:f7f1f0d76dd6 | 473 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 474 | parson_free(array->items); |
| XinZhangMS | 0:f7f1f0d76dd6 | 475 | parson_free(array); |
| XinZhangMS | 0:f7f1f0d76dd6 | 476 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 477 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 478 | /* JSON Value */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 479 | static JSON_Value * json_value_init_string_no_copy(char *string) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 480 | JSON_Value *new_value = (JSON_Value*)parson_malloc(sizeof(JSON_Value)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 481 | if (!new_value) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 482 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 483 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 484 | new_value->parent = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 485 | new_value->type = JSONString; |
| XinZhangMS | 0:f7f1f0d76dd6 | 486 | new_value->value.string = string; |
| XinZhangMS | 0:f7f1f0d76dd6 | 487 | return new_value; |
| XinZhangMS | 0:f7f1f0d76dd6 | 488 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 489 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 490 | /* Parser */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 491 | static JSON_Status skip_quotes(const char **string) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 492 | if (**string != '\"') { |
| XinZhangMS | 0:f7f1f0d76dd6 | 493 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 494 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 495 | SKIP_CHAR(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 496 | while (**string != '\"') { |
| XinZhangMS | 0:f7f1f0d76dd6 | 497 | if (**string == '\0') { |
| XinZhangMS | 0:f7f1f0d76dd6 | 498 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 499 | } else if (**string == '\\') { |
| XinZhangMS | 0:f7f1f0d76dd6 | 500 | SKIP_CHAR(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 501 | if (**string == '\0') { |
| XinZhangMS | 0:f7f1f0d76dd6 | 502 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 503 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 504 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 505 | SKIP_CHAR(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 506 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 507 | SKIP_CHAR(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 508 | return JSONSuccess; |
| XinZhangMS | 0:f7f1f0d76dd6 | 509 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 510 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 511 | static int parse_utf16(const char **unprocessed, char **processed) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 512 | unsigned int cp, lead, trail; |
| XinZhangMS | 0:f7f1f0d76dd6 | 513 | int parse_succeeded = 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 514 | char *processed_ptr = *processed; |
| XinZhangMS | 0:f7f1f0d76dd6 | 515 | const char *unprocessed_ptr = *unprocessed; |
| XinZhangMS | 0:f7f1f0d76dd6 | 516 | unprocessed_ptr++; /* skips u */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 517 | parse_succeeded = parse_utf16_hex(unprocessed_ptr, &cp); |
| XinZhangMS | 0:f7f1f0d76dd6 | 518 | if (!parse_succeeded) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 519 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 520 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 521 | if (cp < 0x80) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 522 | processed_ptr[0] = (char)cp; /* 0xxxxxxx */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 523 | } else if (cp < 0x800) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 524 | processed_ptr[0] = ((cp >> 6) & 0x1F) | 0xC0; /* 110xxxxx */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 525 | processed_ptr[1] = ((cp) & 0x3F) | 0x80; /* 10xxxxxx */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 526 | processed_ptr += 1; |
| XinZhangMS | 0:f7f1f0d76dd6 | 527 | } else if (cp < 0xD800 || cp > 0xDFFF) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 528 | processed_ptr[0] = ((cp >> 12) & 0x0F) | 0xE0; /* 1110xxxx */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 529 | processed_ptr[1] = ((cp >> 6) & 0x3F) | 0x80; /* 10xxxxxx */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 530 | processed_ptr[2] = ((cp) & 0x3F) | 0x80; /* 10xxxxxx */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 531 | processed_ptr += 2; |
| XinZhangMS | 0:f7f1f0d76dd6 | 532 | } else if (cp >= 0xD800 && cp <= 0xDBFF) { /* lead surrogate (0xD800..0xDBFF) */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 533 | lead = cp; |
| XinZhangMS | 0:f7f1f0d76dd6 | 534 | unprocessed_ptr += 4; /* should always be within the buffer, otherwise previous sscanf would fail */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 535 | if (*unprocessed_ptr++ != '\\' || *unprocessed_ptr++ != 'u') { |
| XinZhangMS | 0:f7f1f0d76dd6 | 536 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 537 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 538 | parse_succeeded = parse_utf16_hex(unprocessed_ptr, &trail); |
| XinZhangMS | 0:f7f1f0d76dd6 | 539 | if (!parse_succeeded || trail < 0xDC00 || trail > 0xDFFF) { /* valid trail surrogate? (0xDC00..0xDFFF) */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 540 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 541 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 542 | cp = ((((lead - 0xD800) & 0x3FF) << 10) | ((trail - 0xDC00) & 0x3FF)) + 0x010000; |
| XinZhangMS | 0:f7f1f0d76dd6 | 543 | processed_ptr[0] = (((cp >> 18) & 0x07) | 0xF0); /* 11110xxx */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 544 | processed_ptr[1] = (((cp >> 12) & 0x3F) | 0x80); /* 10xxxxxx */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 545 | processed_ptr[2] = (((cp >> 6) & 0x3F) | 0x80); /* 10xxxxxx */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 546 | processed_ptr[3] = (((cp) & 0x3F) | 0x80); /* 10xxxxxx */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 547 | processed_ptr += 3; |
| XinZhangMS | 0:f7f1f0d76dd6 | 548 | } else { /* trail surrogate before lead surrogate */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 549 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 550 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 551 | unprocessed_ptr += 3; |
| XinZhangMS | 0:f7f1f0d76dd6 | 552 | *processed = processed_ptr; |
| XinZhangMS | 0:f7f1f0d76dd6 | 553 | *unprocessed = unprocessed_ptr; |
| XinZhangMS | 0:f7f1f0d76dd6 | 554 | return JSONSuccess; |
| XinZhangMS | 0:f7f1f0d76dd6 | 555 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 556 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 557 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 558 | /* Copies and processes passed string up to supplied length. |
| XinZhangMS | 0:f7f1f0d76dd6 | 559 | Example: "\u006Corem ipsum" -> lorem ipsum */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 560 | static char* process_string(const char *input, size_t len) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 561 | const char *input_ptr = input; |
| XinZhangMS | 0:f7f1f0d76dd6 | 562 | size_t initial_size = (len + 1) * sizeof(char); |
| XinZhangMS | 0:f7f1f0d76dd6 | 563 | size_t final_size = 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 564 | char *output = NULL, *output_ptr = NULL, *resized_output = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 565 | output = (char*)parson_malloc(initial_size); |
| XinZhangMS | 0:f7f1f0d76dd6 | 566 | if (output == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 567 | goto error; |
| XinZhangMS | 0:f7f1f0d76dd6 | 568 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 569 | output_ptr = output; |
| XinZhangMS | 0:f7f1f0d76dd6 | 570 | while ((*input_ptr != '\0') && (size_t)(input_ptr - input) < len) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 571 | if (*input_ptr == '\\') { |
| XinZhangMS | 0:f7f1f0d76dd6 | 572 | input_ptr++; |
| XinZhangMS | 0:f7f1f0d76dd6 | 573 | switch (*input_ptr) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 574 | case '\"': *output_ptr = '\"'; break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 575 | case '\\': *output_ptr = '\\'; break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 576 | case '/': *output_ptr = '/'; break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 577 | case 'b': *output_ptr = '\b'; break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 578 | case 'f': *output_ptr = '\f'; break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 579 | case 'n': *output_ptr = '\n'; break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 580 | case 'r': *output_ptr = '\r'; break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 581 | case 't': *output_ptr = '\t'; break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 582 | case 'u': |
| XinZhangMS | 0:f7f1f0d76dd6 | 583 | if (parse_utf16(&input_ptr, &output_ptr) == JSONFailure) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 584 | goto error; |
| XinZhangMS | 0:f7f1f0d76dd6 | 585 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 586 | break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 587 | default: |
| XinZhangMS | 0:f7f1f0d76dd6 | 588 | goto error; |
| XinZhangMS | 0:f7f1f0d76dd6 | 589 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 590 | } else if ((unsigned char)*input_ptr < 0x20) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 591 | goto error; /* 0x00-0x19 are invalid characters for json string (http://www.ietf.org/rfc/rfc4627.txt) */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 592 | } else { |
| XinZhangMS | 0:f7f1f0d76dd6 | 593 | *output_ptr = *input_ptr; |
| XinZhangMS | 0:f7f1f0d76dd6 | 594 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 595 | output_ptr++; |
| XinZhangMS | 0:f7f1f0d76dd6 | 596 | input_ptr++; |
| XinZhangMS | 0:f7f1f0d76dd6 | 597 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 598 | *output_ptr = '\0'; |
| XinZhangMS | 0:f7f1f0d76dd6 | 599 | /* resize to new length */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 600 | final_size = (size_t)(output_ptr-output) + 1; |
| XinZhangMS | 0:f7f1f0d76dd6 | 601 | /* todo: don't resize if final_size == initial_size */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 602 | resized_output = (char*)parson_malloc(final_size); |
| XinZhangMS | 0:f7f1f0d76dd6 | 603 | if (resized_output == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 604 | goto error; |
| XinZhangMS | 0:f7f1f0d76dd6 | 605 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 606 | memcpy(resized_output, output, final_size); |
| XinZhangMS | 0:f7f1f0d76dd6 | 607 | parson_free(output); |
| XinZhangMS | 0:f7f1f0d76dd6 | 608 | return resized_output; |
| XinZhangMS | 0:f7f1f0d76dd6 | 609 | error: |
| XinZhangMS | 0:f7f1f0d76dd6 | 610 | parson_free(output); |
| XinZhangMS | 0:f7f1f0d76dd6 | 611 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 612 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 613 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 614 | /* Return processed contents of a string between quotes and |
| XinZhangMS | 0:f7f1f0d76dd6 | 615 | skips passed argument to a matching quote. */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 616 | static char * get_quoted_string(const char **string) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 617 | const char *string_start = *string; |
| XinZhangMS | 0:f7f1f0d76dd6 | 618 | size_t string_len = 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 619 | JSON_Status status = skip_quotes(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 620 | if (status != JSONSuccess) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 621 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 622 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 623 | string_len = *string - string_start - 2; /* length without quotes */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 624 | return process_string(string_start + 1, string_len); |
| XinZhangMS | 0:f7f1f0d76dd6 | 625 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 626 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 627 | static JSON_Value * parse_value(const char **string, size_t nesting) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 628 | if (nesting > MAX_NESTING) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 629 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 630 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 631 | SKIP_WHITESPACES(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 632 | switch (**string) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 633 | case '{': |
| XinZhangMS | 0:f7f1f0d76dd6 | 634 | return parse_object_value(string, nesting + 1); |
| XinZhangMS | 0:f7f1f0d76dd6 | 635 | case '[': |
| XinZhangMS | 0:f7f1f0d76dd6 | 636 | return parse_array_value(string, nesting + 1); |
| XinZhangMS | 0:f7f1f0d76dd6 | 637 | case '\"': |
| XinZhangMS | 0:f7f1f0d76dd6 | 638 | return parse_string_value(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 639 | case 'f': case 't': |
| XinZhangMS | 0:f7f1f0d76dd6 | 640 | return parse_boolean_value(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 641 | case '-': |
| XinZhangMS | 0:f7f1f0d76dd6 | 642 | case '0': case '1': case '2': case '3': case '4': |
| XinZhangMS | 0:f7f1f0d76dd6 | 643 | case '5': case '6': case '7': case '8': case '9': |
| XinZhangMS | 0:f7f1f0d76dd6 | 644 | return parse_number_value(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 645 | case 'n': |
| XinZhangMS | 0:f7f1f0d76dd6 | 646 | return parse_null_value(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 647 | default: |
| XinZhangMS | 0:f7f1f0d76dd6 | 648 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 649 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 650 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 651 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 652 | static JSON_Value * parse_object_value(const char **string, size_t nesting) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 653 | JSON_Value *output_value = json_value_init_object(), *new_value = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 654 | JSON_Object *output_object = json_value_get_object(output_value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 655 | char *new_key = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 656 | if (output_value == NULL || **string != '{') { |
| XinZhangMS | 0:f7f1f0d76dd6 | 657 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 658 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 659 | SKIP_CHAR(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 660 | SKIP_WHITESPACES(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 661 | if (**string == '}') { /* empty object */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 662 | SKIP_CHAR(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 663 | return output_value; |
| XinZhangMS | 0:f7f1f0d76dd6 | 664 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 665 | while (**string != '\0') { |
| XinZhangMS | 0:f7f1f0d76dd6 | 666 | new_key = get_quoted_string(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 667 | if (new_key == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 668 | json_value_free(output_value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 669 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 670 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 671 | SKIP_WHITESPACES(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 672 | if (**string != ':') { |
| XinZhangMS | 0:f7f1f0d76dd6 | 673 | parson_free(new_key); |
| XinZhangMS | 0:f7f1f0d76dd6 | 674 | json_value_free(output_value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 675 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 676 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 677 | SKIP_CHAR(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 678 | new_value = parse_value(string, nesting); |
| XinZhangMS | 0:f7f1f0d76dd6 | 679 | if (new_value == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 680 | parson_free(new_key); |
| XinZhangMS | 0:f7f1f0d76dd6 | 681 | json_value_free(output_value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 682 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 683 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 684 | if (json_object_add(output_object, new_key, new_value) == JSONFailure) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 685 | parson_free(new_key); |
| XinZhangMS | 0:f7f1f0d76dd6 | 686 | json_value_free(new_value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 687 | json_value_free(output_value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 688 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 689 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 690 | parson_free(new_key); |
| XinZhangMS | 0:f7f1f0d76dd6 | 691 | SKIP_WHITESPACES(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 692 | if (**string != ',') { |
| XinZhangMS | 0:f7f1f0d76dd6 | 693 | break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 694 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 695 | SKIP_CHAR(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 696 | SKIP_WHITESPACES(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 697 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 698 | SKIP_WHITESPACES(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 699 | if (**string != '}' || /* Trim object after parsing is over */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 700 | json_object_resize(output_object, json_object_get_count(output_object)) == JSONFailure) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 701 | json_value_free(output_value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 702 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 703 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 704 | SKIP_CHAR(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 705 | return output_value; |
| XinZhangMS | 0:f7f1f0d76dd6 | 706 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 707 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 708 | static JSON_Value * parse_array_value(const char **string, size_t nesting) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 709 | JSON_Value *output_value = json_value_init_array(), *new_array_value = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 710 | JSON_Array *output_array = json_value_get_array(output_value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 711 | if (!output_value || **string != '[') { |
| XinZhangMS | 0:f7f1f0d76dd6 | 712 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 713 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 714 | SKIP_CHAR(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 715 | SKIP_WHITESPACES(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 716 | if (**string == ']') { /* empty array */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 717 | SKIP_CHAR(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 718 | return output_value; |
| XinZhangMS | 0:f7f1f0d76dd6 | 719 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 720 | while (**string != '\0') { |
| XinZhangMS | 0:f7f1f0d76dd6 | 721 | new_array_value = parse_value(string, nesting); |
| XinZhangMS | 0:f7f1f0d76dd6 | 722 | if (new_array_value == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 723 | json_value_free(output_value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 724 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 725 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 726 | if (json_array_add(output_array, new_array_value) == JSONFailure) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 727 | json_value_free(new_array_value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 728 | json_value_free(output_value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 729 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 730 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 731 | SKIP_WHITESPACES(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 732 | if (**string != ',') { |
| XinZhangMS | 0:f7f1f0d76dd6 | 733 | break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 734 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 735 | SKIP_CHAR(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 736 | SKIP_WHITESPACES(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 737 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 738 | SKIP_WHITESPACES(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 739 | if (**string != ']' || /* Trim array after parsing is over */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 740 | json_array_resize(output_array, json_array_get_count(output_array)) == JSONFailure) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 741 | json_value_free(output_value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 742 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 743 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 744 | SKIP_CHAR(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 745 | return output_value; |
| XinZhangMS | 0:f7f1f0d76dd6 | 746 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 747 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 748 | static JSON_Value * parse_string_value(const char **string) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 749 | JSON_Value *value = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 750 | char *new_string = get_quoted_string(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 751 | if (new_string == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 752 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 753 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 754 | value = json_value_init_string_no_copy(new_string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 755 | if (value == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 756 | parson_free(new_string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 757 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 758 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 759 | return value; |
| XinZhangMS | 0:f7f1f0d76dd6 | 760 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 761 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 762 | static JSON_Value * parse_boolean_value(const char **string) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 763 | size_t true_token_size = SIZEOF_TOKEN("true"); |
| XinZhangMS | 0:f7f1f0d76dd6 | 764 | size_t false_token_size = SIZEOF_TOKEN("false"); |
| XinZhangMS | 0:f7f1f0d76dd6 | 765 | if (strncmp("true", *string, true_token_size) == 0) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 766 | *string += true_token_size; |
| XinZhangMS | 0:f7f1f0d76dd6 | 767 | return json_value_init_boolean(1); |
| XinZhangMS | 0:f7f1f0d76dd6 | 768 | } else if (strncmp("false", *string, false_token_size) == 0) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 769 | *string += false_token_size; |
| XinZhangMS | 0:f7f1f0d76dd6 | 770 | return json_value_init_boolean(0); |
| XinZhangMS | 0:f7f1f0d76dd6 | 771 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 772 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 773 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 774 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 775 | static JSON_Value * parse_number_value(const char **string) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 776 | char *end; |
| XinZhangMS | 0:f7f1f0d76dd6 | 777 | double number = 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 778 | errno = 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 779 | number = strtod(*string, &end); |
| XinZhangMS | 0:f7f1f0d76dd6 | 780 | if (errno || !is_decimal(*string, end - *string)) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 781 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 782 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 783 | *string = end; |
| XinZhangMS | 0:f7f1f0d76dd6 | 784 | return json_value_init_number(number); |
| XinZhangMS | 0:f7f1f0d76dd6 | 785 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 786 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 787 | static JSON_Value * parse_null_value(const char **string) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 788 | size_t token_size = SIZEOF_TOKEN("null"); |
| XinZhangMS | 0:f7f1f0d76dd6 | 789 | if (strncmp("null", *string, token_size) == 0) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 790 | *string += token_size; |
| XinZhangMS | 0:f7f1f0d76dd6 | 791 | return json_value_init_null(); |
| XinZhangMS | 0:f7f1f0d76dd6 | 792 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 793 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 794 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 795 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 796 | /* Serialization */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 797 | #define APPEND_STRING(str) do { written = append_string(buf, (str));\ |
| XinZhangMS | 0:f7f1f0d76dd6 | 798 | if (written < 0) { return -1; }\ |
| XinZhangMS | 0:f7f1f0d76dd6 | 799 | if (buf != NULL) { buf += written; }\ |
| XinZhangMS | 0:f7f1f0d76dd6 | 800 | written_total += written; } while(0) |
| XinZhangMS | 0:f7f1f0d76dd6 | 801 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 802 | #define APPEND_INDENT(level) do { written = append_indent(buf, (level));\ |
| XinZhangMS | 0:f7f1f0d76dd6 | 803 | if (written < 0) { return -1; }\ |
| XinZhangMS | 0:f7f1f0d76dd6 | 804 | if (buf != NULL) { buf += written; }\ |
| XinZhangMS | 0:f7f1f0d76dd6 | 805 | written_total += written; } while(0) |
| XinZhangMS | 0:f7f1f0d76dd6 | 806 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 807 | static int json_serialize_to_buffer_r(const JSON_Value *value, char *buf, int level, int is_pretty, char *num_buf) |
| XinZhangMS | 0:f7f1f0d76dd6 | 808 | { |
| XinZhangMS | 0:f7f1f0d76dd6 | 809 | const char *key = NULL, *string = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 810 | JSON_Value *temp_value = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 811 | JSON_Array *array = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 812 | JSON_Object *object = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 813 | size_t i = 0, count = 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 814 | double num = 0.0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 815 | int written = -1, written_total = 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 816 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 817 | switch (json_value_get_type(value)) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 818 | case JSONArray: |
| XinZhangMS | 0:f7f1f0d76dd6 | 819 | array = json_value_get_array(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 820 | count = json_array_get_count(array); |
| XinZhangMS | 0:f7f1f0d76dd6 | 821 | APPEND_STRING("["); |
| XinZhangMS | 0:f7f1f0d76dd6 | 822 | if (count > 0 && is_pretty) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 823 | APPEND_STRING("\n"); |
| XinZhangMS | 0:f7f1f0d76dd6 | 824 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 825 | for (i = 0; i < count; i++) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 826 | if (is_pretty) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 827 | APPEND_INDENT(level+1); |
| XinZhangMS | 0:f7f1f0d76dd6 | 828 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 829 | temp_value = json_array_get_value(array, i); |
| XinZhangMS | 0:f7f1f0d76dd6 | 830 | written = json_serialize_to_buffer_r(temp_value, buf, level+1, is_pretty, num_buf); |
| XinZhangMS | 0:f7f1f0d76dd6 | 831 | if (written < 0) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 832 | return -1; |
| XinZhangMS | 0:f7f1f0d76dd6 | 833 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 834 | if (buf != NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 835 | buf += written; |
| XinZhangMS | 0:f7f1f0d76dd6 | 836 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 837 | written_total += written; |
| XinZhangMS | 0:f7f1f0d76dd6 | 838 | if (i < (count - 1)) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 839 | APPEND_STRING(","); |
| XinZhangMS | 0:f7f1f0d76dd6 | 840 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 841 | if (is_pretty) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 842 | APPEND_STRING("\n"); |
| XinZhangMS | 0:f7f1f0d76dd6 | 843 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 844 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 845 | if (count > 0 && is_pretty) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 846 | APPEND_INDENT(level); |
| XinZhangMS | 0:f7f1f0d76dd6 | 847 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 848 | APPEND_STRING("]"); |
| XinZhangMS | 0:f7f1f0d76dd6 | 849 | return written_total; |
| XinZhangMS | 0:f7f1f0d76dd6 | 850 | case JSONObject: |
| XinZhangMS | 0:f7f1f0d76dd6 | 851 | object = json_value_get_object(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 852 | count = json_object_get_count(object); |
| XinZhangMS | 0:f7f1f0d76dd6 | 853 | APPEND_STRING("{"); |
| XinZhangMS | 0:f7f1f0d76dd6 | 854 | if (count > 0 && is_pretty) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 855 | APPEND_STRING("\n"); |
| XinZhangMS | 0:f7f1f0d76dd6 | 856 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 857 | for (i = 0; i < count; i++) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 858 | key = json_object_get_name(object, i); |
| XinZhangMS | 0:f7f1f0d76dd6 | 859 | if (key == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 860 | return -1; |
| XinZhangMS | 0:f7f1f0d76dd6 | 861 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 862 | if (is_pretty) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 863 | APPEND_INDENT(level+1); |
| XinZhangMS | 0:f7f1f0d76dd6 | 864 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 865 | written = json_serialize_string(key, buf); |
| XinZhangMS | 0:f7f1f0d76dd6 | 866 | if (written < 0) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 867 | return -1; |
| XinZhangMS | 0:f7f1f0d76dd6 | 868 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 869 | if (buf != NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 870 | buf += written; |
| XinZhangMS | 0:f7f1f0d76dd6 | 871 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 872 | written_total += written; |
| XinZhangMS | 0:f7f1f0d76dd6 | 873 | APPEND_STRING(":"); |
| XinZhangMS | 0:f7f1f0d76dd6 | 874 | if (is_pretty) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 875 | APPEND_STRING(" "); |
| XinZhangMS | 0:f7f1f0d76dd6 | 876 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 877 | temp_value = json_object_get_value(object, key); |
| XinZhangMS | 0:f7f1f0d76dd6 | 878 | written = json_serialize_to_buffer_r(temp_value, buf, level+1, is_pretty, num_buf); |
| XinZhangMS | 0:f7f1f0d76dd6 | 879 | if (written < 0) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 880 | return -1; |
| XinZhangMS | 0:f7f1f0d76dd6 | 881 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 882 | if (buf != NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 883 | buf += written; |
| XinZhangMS | 0:f7f1f0d76dd6 | 884 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 885 | written_total += written; |
| XinZhangMS | 0:f7f1f0d76dd6 | 886 | if (i < (count - 1)) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 887 | APPEND_STRING(","); |
| XinZhangMS | 0:f7f1f0d76dd6 | 888 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 889 | if (is_pretty) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 890 | APPEND_STRING("\n"); |
| XinZhangMS | 0:f7f1f0d76dd6 | 891 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 892 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 893 | if (count > 0 && is_pretty) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 894 | APPEND_INDENT(level); |
| XinZhangMS | 0:f7f1f0d76dd6 | 895 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 896 | APPEND_STRING("}"); |
| XinZhangMS | 0:f7f1f0d76dd6 | 897 | return written_total; |
| XinZhangMS | 0:f7f1f0d76dd6 | 898 | case JSONString: |
| XinZhangMS | 0:f7f1f0d76dd6 | 899 | string = json_value_get_string(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 900 | if (string == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 901 | return -1; |
| XinZhangMS | 0:f7f1f0d76dd6 | 902 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 903 | written = json_serialize_string(string, buf); |
| XinZhangMS | 0:f7f1f0d76dd6 | 904 | if (written < 0) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 905 | return -1; |
| XinZhangMS | 0:f7f1f0d76dd6 | 906 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 907 | if (buf != NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 908 | buf += written; |
| XinZhangMS | 0:f7f1f0d76dd6 | 909 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 910 | written_total += written; |
| XinZhangMS | 0:f7f1f0d76dd6 | 911 | return written_total; |
| XinZhangMS | 0:f7f1f0d76dd6 | 912 | case JSONBoolean: |
| XinZhangMS | 0:f7f1f0d76dd6 | 913 | if (json_value_get_boolean(value)) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 914 | APPEND_STRING("true"); |
| XinZhangMS | 0:f7f1f0d76dd6 | 915 | } else { |
| XinZhangMS | 0:f7f1f0d76dd6 | 916 | APPEND_STRING("false"); |
| XinZhangMS | 0:f7f1f0d76dd6 | 917 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 918 | return written_total; |
| XinZhangMS | 0:f7f1f0d76dd6 | 919 | case JSONNumber: |
| XinZhangMS | 0:f7f1f0d76dd6 | 920 | num = json_value_get_number(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 921 | if (buf != NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 922 | num_buf = buf; |
| XinZhangMS | 0:f7f1f0d76dd6 | 923 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 924 | written = sprintf(num_buf, FLOAT_FORMAT, num); |
| XinZhangMS | 0:f7f1f0d76dd6 | 925 | if (written < 0) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 926 | return -1; |
| XinZhangMS | 0:f7f1f0d76dd6 | 927 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 928 | if (buf != NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 929 | buf += written; |
| XinZhangMS | 0:f7f1f0d76dd6 | 930 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 931 | written_total += written; |
| XinZhangMS | 0:f7f1f0d76dd6 | 932 | return written_total; |
| XinZhangMS | 0:f7f1f0d76dd6 | 933 | case JSONNull: |
| XinZhangMS | 0:f7f1f0d76dd6 | 934 | APPEND_STRING("null"); |
| XinZhangMS | 0:f7f1f0d76dd6 | 935 | return written_total; |
| XinZhangMS | 0:f7f1f0d76dd6 | 936 | case JSONError: |
| XinZhangMS | 0:f7f1f0d76dd6 | 937 | return -1; |
| XinZhangMS | 0:f7f1f0d76dd6 | 938 | default: |
| XinZhangMS | 0:f7f1f0d76dd6 | 939 | return -1; |
| XinZhangMS | 0:f7f1f0d76dd6 | 940 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 941 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 942 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 943 | static int json_serialize_string(const char *string, char *buf) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 944 | size_t i = 0, len = strlen(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 945 | char c = '\0'; |
| XinZhangMS | 0:f7f1f0d76dd6 | 946 | int written = -1, written_total = 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 947 | APPEND_STRING("\""); |
| XinZhangMS | 0:f7f1f0d76dd6 | 948 | for (i = 0; i < len; i++) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 949 | c = string[i]; |
| XinZhangMS | 0:f7f1f0d76dd6 | 950 | switch (c) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 951 | case '\"': APPEND_STRING("\\\""); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 952 | case '\\': APPEND_STRING("\\\\"); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 953 | case '/': APPEND_STRING("\\/"); break; /* to make json embeddable in xml\/html */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 954 | case '\b': APPEND_STRING("\\b"); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 955 | case '\f': APPEND_STRING("\\f"); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 956 | case '\n': APPEND_STRING("\\n"); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 957 | case '\r': APPEND_STRING("\\r"); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 958 | case '\t': APPEND_STRING("\\t"); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 959 | case '\x00': APPEND_STRING("\\u0000"); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 960 | case '\x01': APPEND_STRING("\\u0001"); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 961 | case '\x02': APPEND_STRING("\\u0002"); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 962 | case '\x03': APPEND_STRING("\\u0003"); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 963 | case '\x04': APPEND_STRING("\\u0004"); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 964 | case '\x05': APPEND_STRING("\\u0005"); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 965 | case '\x06': APPEND_STRING("\\u0006"); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 966 | case '\x07': APPEND_STRING("\\u0007"); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 967 | /* '\x08' duplicate: '\b' */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 968 | /* '\x09' duplicate: '\t' */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 969 | /* '\x0a' duplicate: '\n' */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 970 | case '\x0b': APPEND_STRING("\\u000b"); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 971 | /* '\x0c' duplicate: '\f' */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 972 | /* '\x0d' duplicate: '\r' */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 973 | case '\x0e': APPEND_STRING("\\u000e"); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 974 | case '\x0f': APPEND_STRING("\\u000f"); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 975 | case '\x10': APPEND_STRING("\\u0010"); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 976 | case '\x11': APPEND_STRING("\\u0011"); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 977 | case '\x12': APPEND_STRING("\\u0012"); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 978 | case '\x13': APPEND_STRING("\\u0013"); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 979 | case '\x14': APPEND_STRING("\\u0014"); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 980 | case '\x15': APPEND_STRING("\\u0015"); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 981 | case '\x16': APPEND_STRING("\\u0016"); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 982 | case '\x17': APPEND_STRING("\\u0017"); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 983 | case '\x18': APPEND_STRING("\\u0018"); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 984 | case '\x19': APPEND_STRING("\\u0019"); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 985 | case '\x1a': APPEND_STRING("\\u001a"); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 986 | case '\x1b': APPEND_STRING("\\u001b"); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 987 | case '\x1c': APPEND_STRING("\\u001c"); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 988 | case '\x1d': APPEND_STRING("\\u001d"); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 989 | case '\x1e': APPEND_STRING("\\u001e"); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 990 | case '\x1f': APPEND_STRING("\\u001f"); break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 991 | default: |
| XinZhangMS | 0:f7f1f0d76dd6 | 992 | if (buf != NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 993 | buf[0] = c; |
| XinZhangMS | 0:f7f1f0d76dd6 | 994 | buf += 1; |
| XinZhangMS | 0:f7f1f0d76dd6 | 995 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 996 | written_total += 1; |
| XinZhangMS | 0:f7f1f0d76dd6 | 997 | break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 998 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 999 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1000 | APPEND_STRING("\""); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1001 | return written_total; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1002 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1003 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1004 | static int append_indent(char *buf, int level) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1005 | int i; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1006 | int written = -1, written_total = 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1007 | for (i = 0; i < level; i++) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1008 | APPEND_STRING(" "); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1009 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1010 | return written_total; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1011 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1012 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1013 | static int append_string(char *buf, const char *string) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1014 | if (buf == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1015 | return (int)strlen(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1016 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1017 | return sprintf(buf, "%s", string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1018 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1019 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1020 | #undef APPEND_STRING |
| XinZhangMS | 0:f7f1f0d76dd6 | 1021 | #undef APPEND_INDENT |
| XinZhangMS | 0:f7f1f0d76dd6 | 1022 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1023 | /* Parser API */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 1024 | JSON_Value * json_parse_file(const char *filename) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1025 | char *file_contents = read_file(filename); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1026 | JSON_Value *output_value = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1027 | if (file_contents == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1028 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1029 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1030 | output_value = json_parse_string(file_contents); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1031 | parson_free(file_contents); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1032 | return output_value; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1033 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1034 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1035 | JSON_Value * json_parse_file_with_comments(const char *filename) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1036 | char *file_contents = read_file(filename); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1037 | JSON_Value *output_value = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1038 | if (file_contents == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1039 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1040 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1041 | output_value = json_parse_string_with_comments(file_contents); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1042 | parson_free(file_contents); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1043 | return output_value; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1044 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1045 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1046 | JSON_Value * json_parse_string(const char *string) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1047 | if (string == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1048 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1049 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1050 | if (string[0] == '\xEF' && string[1] == '\xBB' && string[2] == '\xBF') { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1051 | string = string + 3; /* Support for UTF-8 BOM */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 1052 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1053 | return parse_value((const char**)&string, 0); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1054 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1055 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1056 | JSON_Value * json_parse_string_with_comments(const char *string) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1057 | JSON_Value *result = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1058 | char *string_mutable_copy = NULL, *string_mutable_copy_ptr = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1059 | string_mutable_copy = parson_strdup(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1060 | if (string_mutable_copy == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1061 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1062 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1063 | remove_comments(string_mutable_copy, "/*", "*/"); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1064 | remove_comments(string_mutable_copy, "//", "\n"); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1065 | string_mutable_copy_ptr = string_mutable_copy; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1066 | result = parse_value((const char**)&string_mutable_copy_ptr, 0); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1067 | parson_free(string_mutable_copy); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1068 | return result; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1069 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1070 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1071 | /* JSON Object API */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 1072 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1073 | JSON_Value * json_object_get_value(const JSON_Object *object, const char *name) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1074 | if (object == NULL || name == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1075 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1076 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1077 | return json_object_nget_value(object, name, strlen(name)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1078 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1079 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1080 | const char * json_object_get_string(const JSON_Object *object, const char *name) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1081 | return json_value_get_string(json_object_get_value(object, name)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1082 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1083 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1084 | double json_object_get_number(const JSON_Object *object, const char *name) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1085 | return json_value_get_number(json_object_get_value(object, name)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1086 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1087 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1088 | JSON_Object * json_object_get_object(const JSON_Object *object, const char *name) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1089 | return json_value_get_object(json_object_get_value(object, name)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1090 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1091 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1092 | JSON_Array * json_object_get_array(const JSON_Object *object, const char *name) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1093 | return json_value_get_array(json_object_get_value(object, name)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1094 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1095 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1096 | int json_object_get_boolean(const JSON_Object *object, const char *name) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1097 | return json_value_get_boolean(json_object_get_value(object, name)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1098 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1099 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1100 | JSON_Value * json_object_dotget_value(const JSON_Object *object, const char *name) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1101 | const char *dot_position = strchr(name, '.'); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1102 | if (!dot_position) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1103 | return json_object_get_value(object, name); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1104 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1105 | object = json_value_get_object(json_object_nget_value(object, name, dot_position - name)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1106 | return json_object_dotget_value(object, dot_position + 1); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1107 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1108 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1109 | const char * json_object_dotget_string(const JSON_Object *object, const char *name) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1110 | return json_value_get_string(json_object_dotget_value(object, name)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1111 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1112 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1113 | double json_object_dotget_number(const JSON_Object *object, const char *name) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1114 | return json_value_get_number(json_object_dotget_value(object, name)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1115 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1116 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1117 | JSON_Object * json_object_dotget_object(const JSON_Object *object, const char *name) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1118 | return json_value_get_object(json_object_dotget_value(object, name)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1119 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1120 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1121 | JSON_Array * json_object_dotget_array(const JSON_Object *object, const char *name) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1122 | return json_value_get_array(json_object_dotget_value(object, name)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1123 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1124 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1125 | int json_object_dotget_boolean(const JSON_Object *object, const char *name) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1126 | return json_value_get_boolean(json_object_dotget_value(object, name)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1127 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1128 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1129 | size_t json_object_get_count(const JSON_Object *object) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1130 | return object ? object->count : 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1131 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1132 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1133 | const char * json_object_get_name(const JSON_Object *object, size_t index) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1134 | if (object == NULL || index >= json_object_get_count(object)) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1135 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1136 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1137 | return object->names[index]; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1138 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1139 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1140 | JSON_Value * json_object_get_value_at(const JSON_Object *object, size_t index) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1141 | if (object == NULL || index >= json_object_get_count(object)) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1142 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1143 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1144 | return object->values[index]; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1145 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1146 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1147 | JSON_Value *json_object_get_wrapping_value(const JSON_Object *object) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1148 | return object->wrapping_value; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1149 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1150 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1151 | int json_object_has_value (const JSON_Object *object, const char *name) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1152 | return json_object_get_value(object, name) != NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1153 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1154 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1155 | int json_object_has_value_of_type(const JSON_Object *object, const char *name, JSON_Value_Type type) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1156 | JSON_Value *val = json_object_get_value(object, name); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1157 | return val != NULL && json_value_get_type(val) == type; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1158 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1159 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1160 | int json_object_dothas_value (const JSON_Object *object, const char *name) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1161 | return json_object_dotget_value(object, name) != NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1162 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1163 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1164 | int json_object_dothas_value_of_type(const JSON_Object *object, const char *name, JSON_Value_Type type) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1165 | JSON_Value *val = json_object_dotget_value(object, name); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1166 | return val != NULL && json_value_get_type(val) == type; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1167 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1168 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1169 | /* JSON Array API */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 1170 | JSON_Value * json_array_get_value(const JSON_Array *array, size_t index) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1171 | if (array == NULL || index >= json_array_get_count(array)) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1172 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1173 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1174 | return array->items[index]; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1175 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1176 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1177 | const char * json_array_get_string(const JSON_Array *array, size_t index) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1178 | return json_value_get_string(json_array_get_value(array, index)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1179 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1180 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1181 | double json_array_get_number(const JSON_Array *array, size_t index) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1182 | return json_value_get_number(json_array_get_value(array, index)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1183 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1184 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1185 | JSON_Object * json_array_get_object(const JSON_Array *array, size_t index) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1186 | return json_value_get_object(json_array_get_value(array, index)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1187 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1188 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1189 | JSON_Array * json_array_get_array(const JSON_Array *array, size_t index) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1190 | return json_value_get_array(json_array_get_value(array, index)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1191 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1192 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1193 | int json_array_get_boolean(const JSON_Array *array, size_t index) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1194 | return json_value_get_boolean(json_array_get_value(array, index)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1195 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1196 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1197 | size_t json_array_get_count(const JSON_Array *array) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1198 | return array ? array->count : 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1199 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1200 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1201 | JSON_Value * json_array_get_wrapping_value(const JSON_Array *array) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1202 | return array->wrapping_value; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1203 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1204 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1205 | /* JSON Value API */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 1206 | JSON_Value_Type json_value_get_type(const JSON_Value *value) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1207 | return value ? value->type : JSONError; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1208 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1209 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1210 | JSON_Object * json_value_get_object(const JSON_Value *value) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1211 | return json_value_get_type(value) == JSONObject ? value->value.object : NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1212 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1213 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1214 | JSON_Array * json_value_get_array(const JSON_Value *value) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1215 | return json_value_get_type(value) == JSONArray ? value->value.array : NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1216 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1217 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1218 | const char * json_value_get_string(const JSON_Value *value) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1219 | return json_value_get_type(value) == JSONString ? value->value.string : NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1220 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1221 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1222 | double json_value_get_number(const JSON_Value *value) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1223 | return json_value_get_type(value) == JSONNumber ? value->value.number : 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1224 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1225 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1226 | int json_value_get_boolean(const JSON_Value *value) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1227 | return json_value_get_type(value) == JSONBoolean ? value->value.boolean : -1; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1228 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1229 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1230 | JSON_Value * json_value_get_parent (const JSON_Value *value) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1231 | return value ? value->parent : NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1232 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1233 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1234 | void json_value_free(JSON_Value *value) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1235 | switch (json_value_get_type(value)) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1236 | case JSONObject: |
| XinZhangMS | 0:f7f1f0d76dd6 | 1237 | json_object_free(value->value.object); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1238 | break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1239 | case JSONString: |
| XinZhangMS | 0:f7f1f0d76dd6 | 1240 | parson_free(value->value.string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1241 | break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1242 | case JSONArray: |
| XinZhangMS | 0:f7f1f0d76dd6 | 1243 | json_array_free(value->value.array); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1244 | break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1245 | default: |
| XinZhangMS | 0:f7f1f0d76dd6 | 1246 | break; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1247 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1248 | parson_free(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1249 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1250 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1251 | JSON_Value * json_value_init_object(void) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1252 | JSON_Value *new_value = (JSON_Value*)parson_malloc(sizeof(JSON_Value)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1253 | if (!new_value) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1254 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1255 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1256 | new_value->parent = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1257 | new_value->type = JSONObject; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1258 | new_value->value.object = json_object_init(new_value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1259 | if (!new_value->value.object) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1260 | parson_free(new_value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1261 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1262 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1263 | return new_value; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1264 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1265 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1266 | JSON_Value * json_value_init_array(void) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1267 | JSON_Value *new_value = (JSON_Value*)parson_malloc(sizeof(JSON_Value)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1268 | if (!new_value) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1269 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1270 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1271 | new_value->parent = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1272 | new_value->type = JSONArray; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1273 | new_value->value.array = json_array_init(new_value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1274 | if (!new_value->value.array) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1275 | parson_free(new_value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1276 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1277 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1278 | return new_value; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1279 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1280 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1281 | JSON_Value * json_value_init_string(const char *string) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1282 | char *copy = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1283 | JSON_Value *value; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1284 | size_t string_len = 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1285 | if (string == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1286 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1287 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1288 | string_len = strlen(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1289 | if (!is_valid_utf8(string, string_len)) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1290 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1291 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1292 | copy = parson_strndup(string, string_len); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1293 | if (copy == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1294 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1295 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1296 | value = json_value_init_string_no_copy(copy); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1297 | if (value == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1298 | parson_free(copy); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1299 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1300 | return value; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1301 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1302 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1303 | JSON_Value * json_value_init_number(double number) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1304 | JSON_Value *new_value = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1305 | if ((number * 0.0) != 0.0) { /* nan and inf test */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 1306 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1307 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1308 | new_value = (JSON_Value*)parson_malloc(sizeof(JSON_Value)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1309 | if (new_value == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1310 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1311 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1312 | new_value->parent = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1313 | new_value->type = JSONNumber; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1314 | new_value->value.number = number; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1315 | return new_value; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1316 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1317 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1318 | JSON_Value * json_value_init_boolean(int boolean) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1319 | JSON_Value *new_value = (JSON_Value*)parson_malloc(sizeof(JSON_Value)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1320 | if (!new_value) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1321 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1322 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1323 | new_value->parent = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1324 | new_value->type = JSONBoolean; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1325 | new_value->value.boolean = boolean ? 1 : 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1326 | return new_value; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1327 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1328 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1329 | JSON_Value * json_value_init_null(void) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1330 | JSON_Value *new_value = (JSON_Value*)parson_malloc(sizeof(JSON_Value)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1331 | if (!new_value) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1332 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1333 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1334 | new_value->parent = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1335 | new_value->type = JSONNull; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1336 | return new_value; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1337 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1338 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1339 | JSON_Value * json_value_deep_copy(const JSON_Value *value) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1340 | size_t i = 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1341 | JSON_Value *return_value = NULL, *temp_value_copy = NULL, *temp_value = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1342 | const char *temp_string = NULL, *temp_key = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1343 | char *temp_string_copy = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1344 | JSON_Array *temp_array = NULL, *temp_array_copy = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1345 | JSON_Object *temp_object = NULL, *temp_object_copy = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1346 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1347 | switch (json_value_get_type(value)) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1348 | case JSONArray: |
| XinZhangMS | 0:f7f1f0d76dd6 | 1349 | temp_array = json_value_get_array(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1350 | return_value = json_value_init_array(); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1351 | if (return_value == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1352 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1353 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1354 | temp_array_copy = json_value_get_array(return_value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1355 | for (i = 0; i < json_array_get_count(temp_array); i++) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1356 | temp_value = json_array_get_value(temp_array, i); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1357 | temp_value_copy = json_value_deep_copy(temp_value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1358 | if (temp_value_copy == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1359 | json_value_free(return_value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1360 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1361 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1362 | if (json_array_add(temp_array_copy, temp_value_copy) == JSONFailure) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1363 | json_value_free(return_value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1364 | json_value_free(temp_value_copy); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1365 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1366 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1367 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1368 | return return_value; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1369 | case JSONObject: |
| XinZhangMS | 0:f7f1f0d76dd6 | 1370 | temp_object = json_value_get_object(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1371 | return_value = json_value_init_object(); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1372 | if (return_value == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1373 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1374 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1375 | temp_object_copy = json_value_get_object(return_value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1376 | for (i = 0; i < json_object_get_count(temp_object); i++) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1377 | temp_key = json_object_get_name(temp_object, i); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1378 | temp_value = json_object_get_value(temp_object, temp_key); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1379 | temp_value_copy = json_value_deep_copy(temp_value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1380 | if (temp_value_copy == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1381 | json_value_free(return_value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1382 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1383 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1384 | if (json_object_add(temp_object_copy, temp_key, temp_value_copy) == JSONFailure) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1385 | json_value_free(return_value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1386 | json_value_free(temp_value_copy); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1387 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1388 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1389 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1390 | return return_value; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1391 | case JSONBoolean: |
| XinZhangMS | 0:f7f1f0d76dd6 | 1392 | return json_value_init_boolean(json_value_get_boolean(value)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1393 | case JSONNumber: |
| XinZhangMS | 0:f7f1f0d76dd6 | 1394 | return json_value_init_number(json_value_get_number(value)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1395 | case JSONString: |
| XinZhangMS | 0:f7f1f0d76dd6 | 1396 | temp_string = json_value_get_string(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1397 | if (temp_string == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1398 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1399 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1400 | temp_string_copy = parson_strdup(temp_string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1401 | if (temp_string_copy == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1402 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1403 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1404 | return_value = json_value_init_string_no_copy(temp_string_copy); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1405 | if (return_value == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1406 | parson_free(temp_string_copy); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1407 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1408 | return return_value; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1409 | case JSONNull: |
| XinZhangMS | 0:f7f1f0d76dd6 | 1410 | return json_value_init_null(); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1411 | case JSONError: |
| XinZhangMS | 0:f7f1f0d76dd6 | 1412 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1413 | default: |
| XinZhangMS | 0:f7f1f0d76dd6 | 1414 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1415 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1416 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1417 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1418 | size_t json_serialization_size(const JSON_Value *value) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1419 | char num_buf[1100]; /* recursively allocating buffer on stack is a bad idea, so let's do it only once */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 1420 | int res = json_serialize_to_buffer_r(value, NULL, 0, 0, num_buf); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1421 | return res < 0 ? 0 : (size_t)(res + 1); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1422 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1423 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1424 | JSON_Status json_serialize_to_buffer(const JSON_Value *value, char *buf, size_t buf_size_in_bytes) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1425 | int written = -1; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1426 | size_t needed_size_in_bytes = json_serialization_size(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1427 | if (needed_size_in_bytes == 0 || buf_size_in_bytes < needed_size_in_bytes) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1428 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1429 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1430 | written = json_serialize_to_buffer_r(value, buf, 0, 0, NULL); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1431 | if (written < 0) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1432 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1433 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1434 | return JSONSuccess; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1435 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1436 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1437 | JSON_Status json_serialize_to_file(const JSON_Value *value, const char *filename) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1438 | JSON_Status return_code = JSONSuccess; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1439 | FILE *fp = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1440 | char *serialized_string = json_serialize_to_string(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1441 | if (serialized_string == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1442 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1443 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1444 | fp = fopen (filename, "w"); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1445 | if (fp == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1446 | json_free_serialized_string(serialized_string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1447 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1448 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1449 | if (fputs(serialized_string, fp) == EOF) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1450 | return_code = JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1451 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1452 | if (fclose(fp) == EOF) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1453 | return_code = JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1454 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1455 | json_free_serialized_string(serialized_string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1456 | return return_code; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1457 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1458 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1459 | char * json_serialize_to_string(const JSON_Value *value) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1460 | JSON_Status serialization_result = JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1461 | size_t buf_size_bytes = json_serialization_size(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1462 | char *buf = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1463 | if (buf_size_bytes == 0) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1464 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1465 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1466 | buf = (char*)parson_malloc(buf_size_bytes); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1467 | if (buf == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1468 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1469 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1470 | serialization_result = json_serialize_to_buffer(value, buf, buf_size_bytes); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1471 | if (serialization_result == JSONFailure) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1472 | json_free_serialized_string(buf); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1473 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1474 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1475 | return buf; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1476 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1477 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1478 | size_t json_serialization_size_pretty(const JSON_Value *value) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1479 | char num_buf[1100]; /* recursively allocating buffer on stack is a bad idea, so let's do it only once */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 1480 | int res = json_serialize_to_buffer_r(value, NULL, 0, 1, num_buf); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1481 | return res < 0 ? 0 : (size_t)(res + 1); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1482 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1483 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1484 | JSON_Status json_serialize_to_buffer_pretty(const JSON_Value *value, char *buf, size_t buf_size_in_bytes) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1485 | int written = -1; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1486 | size_t needed_size_in_bytes = json_serialization_size_pretty(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1487 | if (needed_size_in_bytes == 0 || buf_size_in_bytes < needed_size_in_bytes) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1488 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1489 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1490 | written = json_serialize_to_buffer_r(value, buf, 0, 1, NULL); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1491 | if (written < 0) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1492 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1493 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1494 | return JSONSuccess; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1495 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1496 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1497 | JSON_Status json_serialize_to_file_pretty(const JSON_Value *value, const char *filename) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1498 | JSON_Status return_code = JSONSuccess; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1499 | FILE *fp = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1500 | char *serialized_string = json_serialize_to_string_pretty(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1501 | if (serialized_string == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1502 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1503 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1504 | fp = fopen (filename, "w"); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1505 | if (fp == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1506 | json_free_serialized_string(serialized_string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1507 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1508 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1509 | if (fputs(serialized_string, fp) == EOF) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1510 | return_code = JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1511 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1512 | if (fclose(fp) == EOF) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1513 | return_code = JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1514 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1515 | json_free_serialized_string(serialized_string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1516 | return return_code; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1517 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1518 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1519 | char * json_serialize_to_string_pretty(const JSON_Value *value) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1520 | JSON_Status serialization_result = JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1521 | size_t buf_size_bytes = json_serialization_size_pretty(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1522 | char *buf = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1523 | if (buf_size_bytes == 0) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1524 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1525 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1526 | buf = (char*)parson_malloc(buf_size_bytes); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1527 | if (buf == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1528 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1529 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1530 | serialization_result = json_serialize_to_buffer_pretty(value, buf, buf_size_bytes); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1531 | if (serialization_result == JSONFailure) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1532 | json_free_serialized_string(buf); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1533 | return NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1534 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1535 | return buf; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1536 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1537 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1538 | void json_free_serialized_string(char *string) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1539 | parson_free(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1540 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1541 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1542 | JSON_Status json_array_remove(JSON_Array *array, size_t ix) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1543 | size_t to_move_bytes = 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1544 | if (array == NULL || ix >= json_array_get_count(array)) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1545 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1546 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1547 | json_value_free(json_array_get_value(array, ix)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1548 | to_move_bytes = (json_array_get_count(array) - 1 - ix) * sizeof(JSON_Value*); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1549 | memmove(array->items + ix, array->items + ix + 1, to_move_bytes); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1550 | array->count -= 1; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1551 | return JSONSuccess; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1552 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1553 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1554 | JSON_Status json_array_replace_value(JSON_Array *array, size_t ix, JSON_Value *value) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1555 | if (array == NULL || value == NULL || value->parent != NULL || ix >= json_array_get_count(array)) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1556 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1557 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1558 | json_value_free(json_array_get_value(array, ix)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1559 | value->parent = json_array_get_wrapping_value(array); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1560 | array->items[ix] = value; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1561 | return JSONSuccess; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1562 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1563 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1564 | JSON_Status json_array_replace_string(JSON_Array *array, size_t i, const char* string) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1565 | JSON_Value *value = json_value_init_string(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1566 | if (value == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1567 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1568 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1569 | if (json_array_replace_value(array, i, value) == JSONFailure) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1570 | json_value_free(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1571 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1572 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1573 | return JSONSuccess; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1574 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1575 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1576 | JSON_Status json_array_replace_number(JSON_Array *array, size_t i, double number) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1577 | JSON_Value *value = json_value_init_number(number); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1578 | if (value == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1579 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1580 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1581 | if (json_array_replace_value(array, i, value) == JSONFailure) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1582 | json_value_free(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1583 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1584 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1585 | return JSONSuccess; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1586 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1587 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1588 | JSON_Status json_array_replace_boolean(JSON_Array *array, size_t i, int boolean) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1589 | JSON_Value *value = json_value_init_boolean(boolean); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1590 | if (value == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1591 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1592 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1593 | if (json_array_replace_value(array, i, value) == JSONFailure) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1594 | json_value_free(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1595 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1596 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1597 | return JSONSuccess; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1598 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1599 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1600 | JSON_Status json_array_replace_null(JSON_Array *array, size_t i) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1601 | JSON_Value *value = json_value_init_null(); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1602 | if (value == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1603 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1604 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1605 | if (json_array_replace_value(array, i, value) == JSONFailure) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1606 | json_value_free(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1607 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1608 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1609 | return JSONSuccess; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1610 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1611 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1612 | JSON_Status json_array_clear(JSON_Array *array) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1613 | size_t i = 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1614 | if (array == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1615 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1616 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1617 | for (i = 0; i < json_array_get_count(array); i++) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1618 | json_value_free(json_array_get_value(array, i)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1619 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1620 | array->count = 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1621 | return JSONSuccess; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1622 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1623 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1624 | JSON_Status json_array_append_value(JSON_Array *array, JSON_Value *value) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1625 | if (array == NULL || value == NULL || value->parent != NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1626 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1627 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1628 | return json_array_add(array, value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1629 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1630 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1631 | JSON_Status json_array_append_string(JSON_Array *array, const char *string) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1632 | JSON_Value *value = json_value_init_string(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1633 | if (value == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1634 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1635 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1636 | if (json_array_append_value(array, value) == JSONFailure) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1637 | json_value_free(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1638 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1639 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1640 | return JSONSuccess; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1641 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1642 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1643 | JSON_Status json_array_append_number(JSON_Array *array, double number) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1644 | JSON_Value *value = json_value_init_number(number); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1645 | if (value == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1646 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1647 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1648 | if (json_array_append_value(array, value) == JSONFailure) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1649 | json_value_free(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1650 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1651 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1652 | return JSONSuccess; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1653 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1654 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1655 | JSON_Status json_array_append_boolean(JSON_Array *array, int boolean) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1656 | JSON_Value *value = json_value_init_boolean(boolean); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1657 | if (value == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1658 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1659 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1660 | if (json_array_append_value(array, value) == JSONFailure) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1661 | json_value_free(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1662 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1663 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1664 | return JSONSuccess; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1665 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1666 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1667 | JSON_Status json_array_append_null(JSON_Array *array) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1668 | JSON_Value *value = json_value_init_null(); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1669 | if (value == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1670 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1671 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1672 | if (json_array_append_value(array, value) == JSONFailure) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1673 | json_value_free(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1674 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1675 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1676 | return JSONSuccess; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1677 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1678 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1679 | JSON_Status json_object_set_value(JSON_Object *object, const char *name, JSON_Value *value) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1680 | size_t i = 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1681 | JSON_Value *old_value; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1682 | if (object == NULL || name == NULL || value == NULL || value->parent != NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1683 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1684 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1685 | old_value = json_object_get_value(object, name); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1686 | if (old_value != NULL) { /* free and overwrite old value */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 1687 | json_value_free(old_value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1688 | for (i = 0; i < json_object_get_count(object); i++) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1689 | if (strcmp(object->names[i], name) == 0) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1690 | value->parent = json_object_get_wrapping_value(object); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1691 | object->values[i] = value; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1692 | return JSONSuccess; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1693 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1694 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1695 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1696 | /* add new key value pair */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 1697 | return json_object_add(object, name, value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1698 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1699 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1700 | JSON_Status json_object_set_string(JSON_Object *object, const char *name, const char *string) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1701 | return json_object_set_value(object, name, json_value_init_string(string)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1702 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1703 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1704 | JSON_Status json_object_set_number(JSON_Object *object, const char *name, double number) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1705 | return json_object_set_value(object, name, json_value_init_number(number)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1706 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1707 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1708 | JSON_Status json_object_set_boolean(JSON_Object *object, const char *name, int boolean) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1709 | return json_object_set_value(object, name, json_value_init_boolean(boolean)); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1710 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1711 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1712 | JSON_Status json_object_set_null(JSON_Object *object, const char *name) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1713 | return json_object_set_value(object, name, json_value_init_null()); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1714 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1715 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1716 | JSON_Status json_object_dotset_value(JSON_Object *object, const char *name, JSON_Value *value) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1717 | const char *dot_pos = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1718 | char *current_name = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1719 | JSON_Object *temp_obj = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1720 | JSON_Value *new_value = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1721 | if (object == NULL || name == NULL || value == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1722 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1723 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1724 | dot_pos = strchr(name, '.'); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1725 | if (dot_pos == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1726 | return json_object_set_value(object, name, value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1727 | } else { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1728 | current_name = parson_strndup(name, dot_pos - name); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1729 | temp_obj = json_object_get_object(object, current_name); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1730 | if (temp_obj == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1731 | new_value = json_value_init_object(); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1732 | if (new_value == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1733 | parson_free(current_name); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1734 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1735 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1736 | if (json_object_add(object, current_name, new_value) == JSONFailure) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1737 | json_value_free(new_value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1738 | parson_free(current_name); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1739 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1740 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1741 | temp_obj = json_object_get_object(object, current_name); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1742 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1743 | parson_free(current_name); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1744 | return json_object_dotset_value(temp_obj, dot_pos + 1, value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1745 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1746 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1747 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1748 | JSON_Status json_object_dotset_string(JSON_Object *object, const char *name, const char *string) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1749 | JSON_Value *value = json_value_init_string(string); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1750 | if (value == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1751 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1752 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1753 | if (json_object_dotset_value(object, name, value) == JSONFailure) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1754 | json_value_free(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1755 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1756 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1757 | return JSONSuccess; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1758 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1759 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1760 | JSON_Status json_object_dotset_number(JSON_Object *object, const char *name, double number) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1761 | JSON_Value *value = json_value_init_number(number); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1762 | if (value == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1763 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1764 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1765 | if (json_object_dotset_value(object, name, value) == JSONFailure) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1766 | json_value_free(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1767 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1768 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1769 | return JSONSuccess; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1770 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1771 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1772 | JSON_Status json_object_dotset_boolean(JSON_Object *object, const char *name, int boolean) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1773 | JSON_Value *value = json_value_init_boolean(boolean); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1774 | if (value == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1775 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1776 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1777 | if (json_object_dotset_value(object, name, value) == JSONFailure) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1778 | json_value_free(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1779 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1780 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1781 | return JSONSuccess; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1782 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1783 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1784 | JSON_Status json_object_dotset_null(JSON_Object *object, const char *name) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1785 | JSON_Value *value = json_value_init_null(); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1786 | if (value == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1787 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1788 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1789 | if (json_object_dotset_value(object, name, value) == JSONFailure) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1790 | json_value_free(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1791 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1792 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1793 | return JSONSuccess; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1794 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1795 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1796 | JSON_Status json_object_remove(JSON_Object *object, const char *name) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1797 | size_t i = 0, last_item_index = 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1798 | if (object == NULL || json_object_get_value(object, name) == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1799 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1800 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1801 | last_item_index = json_object_get_count(object) - 1; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1802 | for (i = 0; i < json_object_get_count(object); i++) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1803 | if (strcmp(object->names[i], name) == 0) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1804 | parson_free(object->names[i]); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1805 | json_value_free(object->values[i]); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1806 | if (i != last_item_index) { /* Replace key value pair with one from the end */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 1807 | object->names[i] = object->names[last_item_index]; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1808 | object->values[i] = object->values[last_item_index]; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1809 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1810 | object->count -= 1; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1811 | return JSONSuccess; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1812 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1813 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1814 | return JSONFailure; /* No execution path should end here */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 1815 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1816 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1817 | JSON_Status json_object_dotremove(JSON_Object *object, const char *name) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1818 | const char *dot_pos = strchr(name, '.'); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1819 | char *current_name = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1820 | JSON_Object *temp_obj = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1821 | if (dot_pos == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1822 | return json_object_remove(object, name); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1823 | } else { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1824 | current_name = parson_strndup(name, dot_pos - name); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1825 | temp_obj = json_object_get_object(object, current_name); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1826 | parson_free(current_name); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1827 | if (temp_obj == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1828 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1829 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1830 | return json_object_dotremove(temp_obj, dot_pos + 1); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1831 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1832 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1833 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1834 | JSON_Status json_object_clear(JSON_Object *object) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1835 | size_t i = 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1836 | if (object == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1837 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1838 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1839 | for (i = 0; i < json_object_get_count(object); i++) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1840 | parson_free(object->names[i]); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1841 | json_value_free(object->values[i]); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1842 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1843 | object->count = 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1844 | return JSONSuccess; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1845 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1846 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1847 | JSON_Status json_validate(const JSON_Value *schema, const JSON_Value *value) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1848 | JSON_Value *temp_schema_value = NULL, *temp_value = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1849 | JSON_Array *schema_array = NULL, *value_array = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1850 | JSON_Object *schema_object = NULL, *value_object = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1851 | JSON_Value_Type schema_type = JSONError, value_type = JSONError; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1852 | const char *key = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1853 | size_t i = 0, count = 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1854 | if (schema == NULL || value == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1855 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1856 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1857 | schema_type = json_value_get_type(schema); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1858 | value_type = json_value_get_type(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1859 | if (schema_type != value_type && schema_type != JSONNull) { /* null represents all values */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 1860 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1861 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1862 | switch (schema_type) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1863 | case JSONArray: |
| XinZhangMS | 0:f7f1f0d76dd6 | 1864 | schema_array = json_value_get_array(schema); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1865 | value_array = json_value_get_array(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1866 | count = json_array_get_count(schema_array); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1867 | if (count == 0) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1868 | return JSONSuccess; /* Empty array allows all types */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 1869 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1870 | /* Get first value from array, rest is ignored */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 1871 | temp_schema_value = json_array_get_value(schema_array, 0); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1872 | for (i = 0; i < json_array_get_count(value_array); i++) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1873 | temp_value = json_array_get_value(value_array, i); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1874 | if (json_validate(temp_schema_value, temp_value) == JSONFailure) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1875 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1876 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1877 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1878 | return JSONSuccess; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1879 | case JSONObject: |
| XinZhangMS | 0:f7f1f0d76dd6 | 1880 | schema_object = json_value_get_object(schema); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1881 | value_object = json_value_get_object(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1882 | count = json_object_get_count(schema_object); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1883 | if (count == 0) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1884 | return JSONSuccess; /* Empty object allows all objects */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 1885 | } else if (json_object_get_count(value_object) < count) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1886 | return JSONFailure; /* Tested object mustn't have less name-value pairs than schema */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 1887 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1888 | for (i = 0; i < count; i++) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1889 | key = json_object_get_name(schema_object, i); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1890 | temp_schema_value = json_object_get_value(schema_object, key); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1891 | temp_value = json_object_get_value(value_object, key); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1892 | if (temp_value == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1893 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1894 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1895 | if (json_validate(temp_schema_value, temp_value) == JSONFailure) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1896 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1897 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1898 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1899 | return JSONSuccess; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1900 | case JSONString: case JSONNumber: case JSONBoolean: case JSONNull: |
| XinZhangMS | 0:f7f1f0d76dd6 | 1901 | return JSONSuccess; /* equality already tested before switch */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 1902 | case JSONError: default: |
| XinZhangMS | 0:f7f1f0d76dd6 | 1903 | return JSONFailure; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1904 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1905 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1906 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1907 | int json_value_equals(const JSON_Value *a, const JSON_Value *b) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1908 | JSON_Object *a_object = NULL, *b_object = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1909 | JSON_Array *a_array = NULL, *b_array = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1910 | const char *a_string = NULL, *b_string = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1911 | const char *key = NULL; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1912 | size_t a_count = 0, b_count = 0, i = 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1913 | JSON_Value_Type a_type, b_type; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1914 | a_type = json_value_get_type(a); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1915 | b_type = json_value_get_type(b); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1916 | if (a_type != b_type) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1917 | return 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1918 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1919 | switch (a_type) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1920 | case JSONArray: |
| XinZhangMS | 0:f7f1f0d76dd6 | 1921 | a_array = json_value_get_array(a); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1922 | b_array = json_value_get_array(b); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1923 | a_count = json_array_get_count(a_array); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1924 | b_count = json_array_get_count(b_array); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1925 | if (a_count != b_count) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1926 | return 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1927 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1928 | for (i = 0; i < a_count; i++) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1929 | if (!json_value_equals(json_array_get_value(a_array, i), |
| XinZhangMS | 0:f7f1f0d76dd6 | 1930 | json_array_get_value(b_array, i))) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1931 | return 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1932 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1933 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1934 | return 1; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1935 | case JSONObject: |
| XinZhangMS | 0:f7f1f0d76dd6 | 1936 | a_object = json_value_get_object(a); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1937 | b_object = json_value_get_object(b); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1938 | a_count = json_object_get_count(a_object); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1939 | b_count = json_object_get_count(b_object); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1940 | if (a_count != b_count) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1941 | return 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1942 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1943 | for (i = 0; i < a_count; i++) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1944 | key = json_object_get_name(a_object, i); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1945 | if (!json_value_equals(json_object_get_value(a_object, key), |
| XinZhangMS | 0:f7f1f0d76dd6 | 1946 | json_object_get_value(b_object, key))) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1947 | return 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1948 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1949 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1950 | return 1; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1951 | case JSONString: |
| XinZhangMS | 0:f7f1f0d76dd6 | 1952 | a_string = json_value_get_string(a); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1953 | b_string = json_value_get_string(b); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1954 | if (a_string == NULL || b_string == NULL) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1955 | return 0; /* shouldn't happen */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 1956 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1957 | return strcmp(a_string, b_string) == 0; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1958 | case JSONBoolean: |
| XinZhangMS | 0:f7f1f0d76dd6 | 1959 | return json_value_get_boolean(a) == json_value_get_boolean(b); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1960 | case JSONNumber: |
| XinZhangMS | 0:f7f1f0d76dd6 | 1961 | return fabs(json_value_get_number(a) - json_value_get_number(b)) < 0.000001; /* EPSILON */ |
| XinZhangMS | 0:f7f1f0d76dd6 | 1962 | case JSONError: |
| XinZhangMS | 0:f7f1f0d76dd6 | 1963 | return 1; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1964 | case JSONNull: |
| XinZhangMS | 0:f7f1f0d76dd6 | 1965 | return 1; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1966 | default: |
| XinZhangMS | 0:f7f1f0d76dd6 | 1967 | return 1; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1968 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1969 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1970 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1971 | JSON_Value_Type json_type(const JSON_Value *value) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1972 | return json_value_get_type(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1973 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1974 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1975 | JSON_Object * json_object (const JSON_Value *value) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1976 | return json_value_get_object(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1977 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1978 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1979 | JSON_Array * json_array (const JSON_Value *value) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1980 | return json_value_get_array(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1981 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1982 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1983 | const char * json_string (const JSON_Value *value) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1984 | return json_value_get_string(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1985 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1986 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1987 | double json_number (const JSON_Value *value) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1988 | return json_value_get_number(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1989 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1990 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1991 | int json_boolean(const JSON_Value *value) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1992 | return json_value_get_boolean(value); |
| XinZhangMS | 0:f7f1f0d76dd6 | 1993 | } |
| XinZhangMS | 0:f7f1f0d76dd6 | 1994 | |
| XinZhangMS | 0:f7f1f0d76dd6 | 1995 | void json_set_allocation_functions(JSON_Malloc_Function malloc_fun, JSON_Free_Function free_fun) { |
| XinZhangMS | 0:f7f1f0d76dd6 | 1996 | parson_malloc = malloc_fun; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1997 | parson_free = free_fun; |
| XinZhangMS | 0:f7f1f0d76dd6 | 1998 | } |