1 /*
2 Copyright (c) 2009-2017 Dave Gamble and cJSON contributors
3
4 Permission is hereby granted, free of charge, to any person obtaining a copy
5 of this software and associated documentation files (the "Software"), to deal
6 in the Software without restriction, including without limitation the rights
7 to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
8 copies of the Software, and to permit persons to whom the Software is
9 furnished to do so, subject to the following conditions:
10
11 The above copyright notice and this permission notice shall be included in
12 all copies or substantial portions of the Software.
13
14 THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15 IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16 FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
17 AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
18 LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
19 OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
20 THE SOFTWARE.
21 */
22
23 #include <stdio.h>
24 #include <stdlib.h>
25 #include <string.h>
26
27 #include "unity/examples/unity_config.h"
28 #include "unity/src/unity.h"
29 #include "common.h"
30 #include "../cJSON_Utils.h"
31
parse_test_file(const char * const filename)32 static cJSON *parse_test_file(const char * const filename)
33 {
34 char *file = NULL;
35 cJSON *json = NULL;
36
37 file = read_file(filename);
38 TEST_ASSERT_NOT_NULL_MESSAGE(file, "Failed to read file.");
39
40 json = cJSON_Parse(file);
41 TEST_ASSERT_NOT_NULL_MESSAGE(json, "Failed to parse test json.");
42 TEST_ASSERT_TRUE_MESSAGE(cJSON_IsArray(json), "Json is not an array.");
43
44 free(file);
45
46 return json;
47 }
48
test_apply_patch(const cJSON * const test)49 static cJSON_bool test_apply_patch(const cJSON * const test)
50 {
51 cJSON *doc = NULL;
52 cJSON *patch = NULL;
53 cJSON *expected = NULL;
54 cJSON *error_element = NULL;
55 cJSON *comment = NULL;
56 cJSON *disabled = NULL;
57
58 cJSON *object = NULL;
59 cJSON_bool successful = false;
60
61 /* extract all the data out of the test */
62 comment = cJSON_GetObjectItemCaseSensitive(test, "comment");
63 if (cJSON_IsString(comment))
64 {
65 printf("Testing \"%s\"\n", comment->valuestring);
66 }
67 else
68 {
69 printf("Testing unknown\n");
70 }
71
72 disabled = cJSON_GetObjectItemCaseSensitive(test, "disabled");
73 if (cJSON_IsTrue(disabled))
74 {
75 printf("SKIPPED\n");
76 return true;
77 }
78
79 doc = cJSON_GetObjectItemCaseSensitive(test, "doc");
80 TEST_ASSERT_NOT_NULL_MESSAGE(doc, "No \"doc\" in the test.");
81 patch = cJSON_GetObjectItemCaseSensitive(test, "patch");
82 TEST_ASSERT_NOT_NULL_MESSAGE(patch, "No \"patch\"in the test.");
83 /* Make a working copy of 'doc' */
84 object = cJSON_Duplicate(doc, true);
85 TEST_ASSERT_NOT_NULL(object);
86
87 expected = cJSON_GetObjectItemCaseSensitive(test, "expected");
88 error_element = cJSON_GetObjectItemCaseSensitive(test, "error");
89 if (error_element != NULL)
90 {
91 /* excepting an error */
92 TEST_ASSERT_TRUE_MESSAGE(0 != cJSONUtils_ApplyPatchesCaseSensitive(object, patch), "Test didn't fail as it's supposed to.");
93
94 successful = true;
95 }
96 else
97 {
98 /* apply the patch */
99 TEST_ASSERT_EQUAL_INT_MESSAGE(0, cJSONUtils_ApplyPatchesCaseSensitive(object, patch), "Failed to apply patches.");
100 successful = true;
101
102 if (expected != NULL)
103 {
104 successful = cJSON_Compare(object, expected, true);
105 }
106 }
107
108 cJSON_Delete(object);
109
110 if (successful)
111 {
112 printf("OK\n");
113 }
114 else
115 {
116 printf("FAILED\n");
117 }
118
119 return successful;
120 }
121
test_generate_test(cJSON * test)122 static cJSON_bool test_generate_test(cJSON *test)
123 {
124 cJSON *doc = NULL;
125 cJSON *patch = NULL;
126 cJSON *expected = NULL;
127 cJSON *disabled = NULL;
128
129 cJSON *object = NULL;
130 cJSON_bool successful = false;
131
132 char *printed_patch = NULL;
133
134 disabled = cJSON_GetObjectItemCaseSensitive(test, "disabled");
135 if (cJSON_IsTrue(disabled))
136 {
137 printf("SKIPPED\n");
138 return true;
139 }
140
141 doc = cJSON_GetObjectItemCaseSensitive(test, "doc");
142 TEST_ASSERT_NOT_NULL_MESSAGE(doc, "No \"doc\" in the test.");
143
144 /* Make a working copy of 'doc' */
145 object = cJSON_Duplicate(doc, true);
146 TEST_ASSERT_NOT_NULL(object);
147
148 expected = cJSON_GetObjectItemCaseSensitive(test, "expected");
149 if (expected == NULL)
150 {
151 cJSON_Delete(object);
152 /* if there is no expected output, this test doesn't make sense */
153 return true;
154 }
155
156 patch = cJSONUtils_GeneratePatchesCaseSensitive(doc, expected);
157 TEST_ASSERT_NOT_NULL_MESSAGE(patch, "Failed to generate patches.");
158
159 printed_patch = cJSON_Print(patch);
160 printf("%s\n", printed_patch);
161 free(printed_patch);
162
163 /* apply the generated patch */
164 TEST_ASSERT_EQUAL_INT_MESSAGE(0, cJSONUtils_ApplyPatchesCaseSensitive(object, patch), "Failed to apply generated patch.");
165
166 successful = cJSON_Compare(object, expected, true);
167
168 cJSON_Delete(patch);
169 cJSON_Delete(object);
170
171 if (successful)
172 {
173 printf("generated patch: OK\n");
174 }
175 else
176 {
177 printf("generated patch: FAILED\n");
178 }
179
180 return successful;
181 }
182
cjson_utils_should_pass_json_patch_test_tests(void)183 static void cjson_utils_should_pass_json_patch_test_tests(void)
184 {
185 cJSON *tests = parse_test_file("json-patch-tests/tests.json");
186 cJSON *test = NULL;
187
188 cJSON_bool failed = false;
189 cJSON_ArrayForEach(test, tests)
190 {
191 failed |= !test_apply_patch(test);
192 failed |= !test_generate_test(test);
193 }
194
195 cJSON_Delete(tests);
196
197 TEST_ASSERT_FALSE_MESSAGE(failed, "Some tests failed.");
198 }
199
cjson_utils_should_pass_json_patch_test_spec_tests(void)200 static void cjson_utils_should_pass_json_patch_test_spec_tests(void)
201 {
202 cJSON *tests = parse_test_file("json-patch-tests/spec_tests.json");
203 cJSON *test = NULL;
204
205 cJSON_bool failed = false;
206 cJSON_ArrayForEach(test, tests)
207 {
208 failed |= !test_apply_patch(test);
209 failed |= !test_generate_test(test);
210 }
211
212 cJSON_Delete(tests);
213
214 TEST_ASSERT_FALSE_MESSAGE(failed, "Some tests failed.");
215 }
216
cjson_utils_should_pass_json_patch_test_cjson_utils_tests(void)217 static void cjson_utils_should_pass_json_patch_test_cjson_utils_tests(void)
218 {
219 cJSON *tests = parse_test_file("json-patch-tests/cjson-utils-tests.json");
220 cJSON *test = NULL;
221
222 cJSON_bool failed = false;
223 cJSON_ArrayForEach(test, tests)
224 {
225 failed |= !test_apply_patch(test);
226 failed |= !test_generate_test(test);
227 }
228
229 cJSON_Delete(tests);
230
231 TEST_ASSERT_FALSE_MESSAGE(failed, "Some tests failed.");
232 }
233
main(void)234 int main(void)
235 {
236 UNITY_BEGIN();
237
238 RUN_TEST(cjson_utils_should_pass_json_patch_test_tests);
239 RUN_TEST(cjson_utils_should_pass_json_patch_test_spec_tests);
240 RUN_TEST(cjson_utils_should_pass_json_patch_test_cjson_utils_tests);
241
242 return UNITY_END();
243 }
244