testLoadSuccess.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323
  1. #include "testBase.h"
  2. static void testLoadRootZero(void)
  3. {
  4. RyanJson_t json = RyanJsonParse("0");
  5. TEST_ASSERT_NOT_NULL_MESSAGE(json, "Parse 数字 \"0\" 作为根节点 应正常");
  6. if (json)
  7. {
  8. TEST_ASSERT_TRUE(RyanJsonIsInt(json));
  9. TEST_ASSERT_EQUAL_INT(0, RyanJsonGetIntValue(json));
  10. RyanJsonDelete(json);
  11. }
  12. json = RyanJsonParse("-0");
  13. TEST_ASSERT_NOT_NULL_MESSAGE(json, "Parse 数字 \"-0\" 作为根节点 应正常");
  14. if (json)
  15. {
  16. TEST_ASSERT_TRUE(RyanJsonIsInt(json));
  17. TEST_ASSERT_EQUAL_INT(0, RyanJsonGetIntValue(json));
  18. RyanJsonDelete(json);
  19. }
  20. }
  21. static void testLoadUtf8(void)
  22. {
  23. // UTF-8 边界测试
  24. // 双字节字符(© -> \xC2\xA9)
  25. RyanJson_t json = RyanJsonParse("{\"c\":\"\xC2\xA9\"}");
  26. TEST_ASSERT_NOT_NULL(json);
  27. TEST_ASSERT_EQUAL_STRING("\xC2\xA9", RyanJsonGetStringValue(RyanJsonGetObjectToKey(json, "c")));
  28. RyanJsonDelete(json);
  29. // 三字节字符(中 -> \xE4\xB8\xAD)
  30. json = RyanJsonParse("{\"z\":\"\xE4\xB8\xAD\"}");
  31. TEST_ASSERT_NOT_NULL(json);
  32. TEST_ASSERT_EQUAL_STRING("\xE4\xB8\xAD", RyanJsonGetStringValue(RyanJsonGetObjectToKey(json, "z")));
  33. RyanJsonDelete(json);
  34. // 四字节字符(Emoji 🐂 -> \xF0\x9F\x90\x82)
  35. json = RyanJsonParse("{\"e\":\"\xF0\x9F\x90\x82\"}");
  36. TEST_ASSERT_NOT_NULL(json);
  37. TEST_ASSERT_EQUAL_STRING("\xF0\x9F\x90\x82", RyanJsonGetStringValue(RyanJsonGetObjectToKey(json, "e")));
  38. RyanJsonDelete(json);
  39. }
  40. static void testLoadStandardObject(void)
  41. {
  42. char *str = NULL;
  43. RyanJson_t json;
  44. char *jsonstr = "{\"inter\":16,\"double\":16.89,\"string\":\"hello\",\"boolTrue\":true,\"boolFalse\":false,\"null\":null,\"item\":"
  45. "{\"inter\":16,\"double\":16."
  46. "89,\"string\":\"hello\","
  47. "\"boolTrue\":true,\"boolFalse\":false,\"null\":null},\"arrayInt\":[16,16,16,16,16],\"arrayDouble\":[16.89,16.89,"
  48. "16.89,16.89,16.89],"
  49. "\"arrayString\":[\"hello\",\"hello\","
  50. "\"hello\",\"hello\",\"hello\"],\"array\":[16,16.89,\"hello\",true,false,null],\"arrayItem\":[{\"inter\":16,"
  51. "\"double\":16.89,\"string\":"
  52. "\"hello\",\"boolTrue\":true,"
  53. "\"boolFalse\":false,\"null\":null},{\"inter\":16,\"double\":16.89,\"string\":\"hello\",\"boolTrue\":true,"
  54. "\"boolFalse\":false,\"null\":null}],\"unicode\":\"😀\"}";
  55. // 标准对象加载测试
  56. json = RyanJsonParse(jsonstr);
  57. TEST_ASSERT_NOT_NULL_MESSAGE(json, "解析基础 Json 失败");
  58. str = RyanJsonPrint(json, 250, RyanJsonFalse, NULL);
  59. TEST_ASSERT_EQUAL_STRING_MESSAGE(jsonstr, str, "打印生成的字符串与原始字符串不匹配");
  60. RyanJsonFree(str);
  61. // 使用公共验证函数进一步检查
  62. testCheckRoot(json);
  63. RyanJsonDelete(json);
  64. }
  65. static void testLoadUnicodeValid(void)
  66. {
  67. char printfBuf[256] = {0};
  68. char *str = NULL;
  69. RyanJson_t json;
  70. // Emoji 测试
  71. json = RyanJsonParse("{\"emoji\":\"\\uD83D\\uDE00\"}");
  72. TEST_ASSERT_NOT_NULL_MESSAGE(json, "解析 Unicode Emoji 失败");
  73. str = RyanJsonPrintPreallocated(json, printfBuf, sizeof(printfBuf), RyanJsonFalse, NULL);
  74. TEST_ASSERT_NOT_NULL_MESSAGE(str, "打印 Unicode Emoji 失败");
  75. RyanJsonDelete(json);
  76. // 测试数字 0-9 分支: \u0030 = '0', \u0039 = '9'
  77. json = RyanJsonParse("{\"num\":\"\\u0030\\u0039\"}");
  78. TEST_ASSERT_NOT_NULL_MESSAGE(json, "解析 Unicode 数字失败");
  79. str = RyanJsonPrintPreallocated(json, printfBuf, sizeof(printfBuf), RyanJsonFalse, NULL);
  80. TEST_ASSERT_EQUAL_STRING_MESSAGE("{\"num\":\"09\"}", str, "Unicode 数字解析/打印错误");
  81. RyanJsonDelete(json);
  82. // 测试小写 a-f 分支: \u0061 = 'a', \u0066 = 'f'
  83. json = RyanJsonParse("{\"lower\":\"\\u0061\\u0062\\u0063\\u0064\\u0065\\u0066\"}");
  84. TEST_ASSERT_NOT_NULL_MESSAGE(json, "解析 Unicode 小写字母失败");
  85. str = RyanJsonPrintPreallocated(json, printfBuf, sizeof(printfBuf), RyanJsonFalse, NULL);
  86. TEST_ASSERT_EQUAL_STRING_MESSAGE("{\"lower\":\"abcdef\"}", str, "Unicode 小写字母解析/打印错误");
  87. RyanJsonDelete(json);
  88. // 测试大写 A-F 分支: \u0041 = 'A', \u0046 = 'F'
  89. json = RyanJsonParse("{\"upper\":\"\\u0041\\u0042\\u0043\\u0044\\u0045\\u0046\"}");
  90. TEST_ASSERT_NOT_NULL_MESSAGE(json, "解析 Unicode 大写字母失败");
  91. str = RyanJsonPrintPreallocated(json, printfBuf, sizeof(printfBuf), RyanJsonFalse, NULL);
  92. TEST_ASSERT_EQUAL_STRING_MESSAGE("{\"upper\":\"ABCDEF\"}", str, "Unicode 大写字母解析/打印错误");
  93. RyanJsonDelete(json);
  94. // 测试混合大小写: \uAbCd
  95. json = RyanJsonParse("{\"mixed\":\"\\uAbCd\"}");
  96. TEST_ASSERT_NOT_NULL_MESSAGE(json, "解析 Unicode 混合大小写失败");
  97. RyanJsonFree(RyanJsonPrint(json, 50, RyanJsonFalse, NULL));
  98. RyanJsonDelete(json);
  99. }
  100. static void testLoadBoundaryConditionsSuccess(void)
  101. {
  102. RyanJson_t json;
  103. // 空结构
  104. json = RyanJsonParse("{}");
  105. TEST_ASSERT_NOT_NULL(json);
  106. TEST_ASSERT_TRUE(RyanJsonIsObject(json));
  107. TEST_ASSERT_NULL(RyanJsonGetObjectValue(json));
  108. RyanJsonDelete(json);
  109. json = RyanJsonParse("[]");
  110. TEST_ASSERT_NOT_NULL(json);
  111. TEST_ASSERT_TRUE(RyanJsonIsArray(json));
  112. TEST_ASSERT_NULL(RyanJsonGetArrayValue(json));
  113. RyanJsonDelete(json);
  114. // 极端空白字符
  115. const char *wsJson = " \n\t { \r\n \"key\" : [ \t ] \n } \r ";
  116. json = RyanJsonParse(wsJson);
  117. TEST_ASSERT_NOT_NULL_MESSAGE(json, "极端空白字符解析失败");
  118. TEST_ASSERT_NOT_NULL(RyanJsonGetObjectToKey(json, "key"));
  119. RyanJsonDelete(json);
  120. // 空 key 和空 strValue
  121. json = RyanJsonParse("{\"\": \"\"}");
  122. TEST_ASSERT_NOT_NULL(json);
  123. RyanJson_t emptyNode = RyanJsonGetObjectValue(json);
  124. TEST_ASSERT_NOT_NULL(emptyNode);
  125. TEST_ASSERT_EQUAL_STRING("", RyanJsonGetKey(emptyNode));
  126. TEST_ASSERT_EQUAL_STRING("", RyanJsonGetStringValue(emptyNode));
  127. RyanJsonDelete(json);
  128. // 极长 key
  129. char longKeyJson[1024];
  130. snprintf(longKeyJson, sizeof(longKeyJson), "{\"%s\":1}",
  131. "a_very_long_key_padding_........................................................................");
  132. json = RyanJsonParse(longKeyJson);
  133. TEST_ASSERT_NOT_NULL_MESSAGE(json, "Parse(长 Key) 失败");
  134. RyanJsonDelete(json);
  135. // 纯标量测试
  136. json = RyanJsonParse("\"just a string\"");
  137. TEST_ASSERT_NOT_NULL(json);
  138. TEST_ASSERT_TRUE(RyanJsonIsString(json));
  139. TEST_ASSERT_EQUAL_STRING("just a string", RyanJsonGetStringValue(json));
  140. RyanJsonDelete(json);
  141. json = RyanJsonParse("123.456");
  142. TEST_ASSERT_NOT_NULL(json);
  143. TEST_ASSERT_TRUE(RyanJsonIsDouble(json));
  144. RyanJsonDelete(json);
  145. // 包含 \0 的输入 (应在 \0 处停止或报错,取决于解析逻辑)
  146. // RyanJsonParse 使用 strlen 确定长度,所以会自动在第一个 \0 处截断
  147. json = RyanJsonParse("{\"a\":1}\0{\"b\":2}");
  148. TEST_ASSERT_NOT_NULL(json);
  149. TEST_ASSERT_NULL(RyanJsonGetObjectToKey(json, "b"));
  150. RyanJsonDelete(json);
  151. }
  152. static void testLoadParseOptionsSuccess(void)
  153. {
  154. const char *end = NULL;
  155. // 允许尾部内容:requireNullTerminator = false
  156. const char *text = " {\"a\":1} trailing";
  157. RyanJson_t json = RyanJsonParseOptions(text, (uint32_t)strlen(text), RyanJsonFalse, &end);
  158. TEST_ASSERT_NOT_NULL_MESSAGE(json, "ParseOptions(允许尾部) 失败");
  159. TEST_ASSERT_NOT_NULL_MESSAGE(end, "parseEndPtr 不应为 NULL");
  160. TEST_ASSERT_EQUAL_STRING_MESSAGE(" trailing", end, "parseEndPtr 位置错误");
  161. RyanJsonDelete(json);
  162. // 仅包含空白尾部:应成功,parseEndPtr 应指向末尾
  163. text = "{\"a\":1} \t\r\n";
  164. json = RyanJsonParseOptions(text, (uint32_t)strlen(text), RyanJsonTrue, &end);
  165. TEST_ASSERT_NOT_NULL_MESSAGE(json, "ParseOptions(空白尾部) 失败");
  166. TEST_ASSERT_NOT_NULL(end);
  167. TEST_ASSERT_EQUAL_CHAR('\0', *end);
  168. RyanJsonDelete(json);
  169. // 限长解析:仅解析前半段
  170. const char *concat = "{\"a\":1}{\"b\":2}";
  171. uint32_t firstLen = (uint32_t)strlen("{\"a\":1}");
  172. end = NULL;
  173. json = RyanJsonParseOptions(concat, firstLen, RyanJsonTrue, &end);
  174. TEST_ASSERT_NOT_NULL_MESSAGE(json, "ParseOptions(限长解析) 失败");
  175. TEST_ASSERT_NOT_NULL(end);
  176. TEST_ASSERT_EQUAL_STRING_MESSAGE("{\"b\":2}", end, "限长解析 parseEndPtr 错误");
  177. RyanJsonDelete(json);
  178. }
  179. static void testLoadParseOptionsBinaryTail(void)
  180. {
  181. // 包含内嵌 '\0' 与后续数据,验证 size 驱动的解析行为
  182. const char rawText[] = {'{', '\"', 'a', '\"', ':', '1', '}', '\0', '{', '\"', 'b', '\"', ':', '2', '}', '\0'};
  183. const char *end = NULL;
  184. RyanJson_t json = RyanJsonParseOptions(rawText, (uint32_t)(sizeof(rawText) - 1U), RyanJsonFalse, &end);
  185. TEST_ASSERT_NOT_NULL_MESSAGE(json, "ParseOptions(含二进制尾部, 允许尾部) 应成功");
  186. TEST_ASSERT_NOT_NULL(end);
  187. TEST_ASSERT_EQUAL_PTR(rawText + 7, end);
  188. TEST_ASSERT_EQUAL_CHAR('\0', *end);
  189. TEST_ASSERT_EQUAL_INT(1, RyanJsonGetIntValue(RyanJsonGetObjectToKey(json, "a")));
  190. RyanJsonDelete(json);
  191. // requireNullTerminator=true 时,内嵌 '\0' 后仍有剩余数据,应失败
  192. json = RyanJsonParseOptions(rawText, (uint32_t)(sizeof(rawText) - 1U), RyanJsonTrue, &end);
  193. TEST_ASSERT_NULL_MESSAGE(json, "ParseOptions(含二进制尾部, 强制结尾) 应失败");
  194. }
  195. static void testLoadNumberBoundaries(void)
  196. {
  197. RyanJson_t json = RyanJsonParse("{\"i\":2147483647,\"i2\":-2147483648,\"i3\":2147483648,\"n\":-0}");
  198. TEST_ASSERT_NOT_NULL(json);
  199. RyanJson_t i = RyanJsonGetObjectToKey(json, "i");
  200. RyanJson_t i2 = RyanJsonGetObjectToKey(json, "i2");
  201. RyanJson_t i3 = RyanJsonGetObjectToKey(json, "i3");
  202. RyanJson_t n = RyanJsonGetObjectToKey(json, "n");
  203. TEST_ASSERT_TRUE_MESSAGE(RyanJsonIsInt(i), "2147483647 应解析为 int32_t");
  204. TEST_ASSERT_TRUE_MESSAGE(RyanJsonIsInt(i2), "-2147483648 应解析为 int32_t");
  205. TEST_ASSERT_TRUE_MESSAGE(RyanJsonIsDouble(i3), "2147483648 应解析为 double");
  206. TEST_ASSERT_TRUE_MESSAGE(RyanJsonIsInt(n), "-0 应解析为 int32_t");
  207. TEST_ASSERT_EQUAL_INT_MESSAGE(0, RyanJsonGetIntValue(n), "-0 值错误");
  208. RyanJsonDelete(json);
  209. // 极大负指数会下溢到 0(有限数),应作为合法数字解析成功
  210. json = RyanJsonParse("1e-2147483647");
  211. TEST_ASSERT_NOT_NULL_MESSAGE(json, "1e-2147483647 应解析成功");
  212. if (json)
  213. {
  214. TEST_ASSERT_TRUE_MESSAGE(RyanJsonIsDouble(json), "1e-2147483647 应解析为 double");
  215. RyanJsonDelete(json);
  216. }
  217. }
  218. static void testLoadScientificNumberRoundtrip(void)
  219. {
  220. const char *jsonText = "{\"a\":1e0,\"b\":1E+2,\"c\":-2.5e-3,\"d\":0e+1}";
  221. RyanJson_t json = RyanJsonParse(jsonText);
  222. TEST_ASSERT_NOT_NULL_MESSAGE(json, "科学计数法解析失败");
  223. RyanJson_t a = RyanJsonGetObjectToKey(json, "a");
  224. RyanJson_t b = RyanJsonGetObjectToKey(json, "b");
  225. RyanJson_t c = RyanJsonGetObjectToKey(json, "c");
  226. RyanJson_t d = RyanJsonGetObjectToKey(json, "d");
  227. TEST_ASSERT_TRUE(RyanJsonIsDouble(a));
  228. TEST_ASSERT_TRUE(RyanJsonIsDouble(b));
  229. TEST_ASSERT_TRUE(RyanJsonIsDouble(c));
  230. TEST_ASSERT_TRUE(RyanJsonIsDouble(d));
  231. TEST_ASSERT_TRUE(RyanJsonCompareDouble(1.0, RyanJsonGetDoubleValue(a)));
  232. TEST_ASSERT_TRUE(RyanJsonCompareDouble(100.0, RyanJsonGetDoubleValue(b)));
  233. TEST_ASSERT_TRUE(RyanJsonCompareDouble(-0.0025, RyanJsonGetDoubleValue(c)));
  234. TEST_ASSERT_TRUE(RyanJsonCompareDouble(0.0, RyanJsonGetDoubleValue(d)));
  235. char *printed = RyanJsonPrint(json, 128, RyanJsonFalse, NULL);
  236. TEST_ASSERT_NOT_NULL(printed);
  237. RyanJson_t roundtrip = RyanJsonParse(printed);
  238. RyanJsonFree(printed);
  239. TEST_ASSERT_NOT_NULL_MESSAGE(roundtrip, "科学计数法往返解析失败");
  240. TEST_ASSERT_TRUE_MESSAGE(RyanJsonCompare(json, roundtrip), "科学计数法往返后 Compare 应相等");
  241. RyanJsonDelete(roundtrip);
  242. RyanJsonDelete(json);
  243. }
  244. static void testLoadDuplicateKeyScopeIsolation(void)
  245. {
  246. // 允许不同作用域使用同名 key(仅同一 Object 作用域内禁止重复)
  247. const char *jsonText = "{\"a\":1,\"obj\":{\"a\":2},\"arr\":[{\"a\":3},{\"a\":4}]}";
  248. RyanJson_t json = RyanJsonParse(jsonText);
  249. TEST_ASSERT_NOT_NULL_MESSAGE(json, "不同作用域同名 key 应解析成功");
  250. if (json)
  251. {
  252. RyanJson_t arr = RyanJsonGetObjectToKey(json, "arr");
  253. RyanJson_t arrObj0 = RyanJsonGetObjectByIndex(arr, 0);
  254. RyanJson_t arrObj1 = RyanJsonGetObjectByIndex(arr, 1);
  255. TEST_ASSERT_EQUAL_INT(1, RyanJsonGetIntValue(RyanJsonGetObjectToKey(json, "a")));
  256. TEST_ASSERT_EQUAL_INT(2, RyanJsonGetIntValue(RyanJsonGetObjectToKey(json, "obj", "a")));
  257. TEST_ASSERT_EQUAL_INT(3, RyanJsonGetIntValue(RyanJsonGetObjectToKey(arrObj0, "a")));
  258. TEST_ASSERT_EQUAL_INT(4, RyanJsonGetIntValue(RyanJsonGetObjectToKey(arrObj1, "a")));
  259. RyanJsonDelete(json);
  260. }
  261. }
  262. void testLoadSuccessRunner(void)
  263. {
  264. UnitySetTestFile(__FILE__);
  265. RUN_TEST(testLoadRootZero);
  266. RUN_TEST(testLoadUtf8);
  267. RUN_TEST(testLoadStandardObject);
  268. RUN_TEST(testLoadUnicodeValid);
  269. RUN_TEST(testLoadBoundaryConditionsSuccess);
  270. RUN_TEST(testLoadParseOptionsSuccess);
  271. RUN_TEST(testLoadParseOptionsBinaryTail);
  272. RUN_TEST(testLoadNumberBoundaries);
  273. RUN_TEST(testLoadScientificNumberRoundtrip);
  274. RUN_TEST(testLoadDuplicateKeyScopeIsolation);
  275. }