| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548 |
- #include "testBase.h"
- static void testChangeEdgeCases(void)
- {
- // 测试 NULL 输入处理
- TEST_ASSERT_FALSE_MESSAGE(RyanJsonChangeBoolValue(NULL, true), "ChangeBoolValue(NULL) 应返回 False");
- TEST_ASSERT_FALSE_MESSAGE(RyanJsonChangeIntValue(NULL, 1), "ChangeIntValue(NULL) 应返回 False");
- TEST_ASSERT_FALSE_MESSAGE(RyanJsonChangeDoubleValue(NULL, 1.0), "ChangeDoubleValue(NULL) 应返回 False");
- TEST_ASSERT_FALSE_MESSAGE(RyanJsonChangeStringValue(NULL, "test"), "ChangeStringValue(NULL) 应返回 False");
- TEST_ASSERT_FALSE_MESSAGE(RyanJsonChangeKey(NULL, "key"), "ChangeKey(NULL) 应返回 False");
- // 测试类型不匹配处理
- RyanJson_t intNode = RyanJsonCreateInt(NULL, 1);
- RyanJson_t doubleNode = RyanJsonCreateDouble(NULL, 1.0);
- RyanJson_t boolNode = RyanJsonCreateBool(NULL, RyanJsonTrue);
- RyanJson_t strNode = RyanJsonCreateString(NULL, "s");
- RyanJson_t keyedStrNode = RyanJsonCreateString("k", "v");
- RyanJson_t keyedBoolNode = RyanJsonCreateBool("flag", RyanJsonTrue);
- TEST_ASSERT_NOT_NULL(intNode);
- TEST_ASSERT_NOT_NULL(doubleNode);
- TEST_ASSERT_NOT_NULL(boolNode);
- TEST_ASSERT_NOT_NULL(strNode);
- TEST_ASSERT_NOT_NULL(keyedStrNode);
- TEST_ASSERT_NOT_NULL(keyedBoolNode);
- TEST_ASSERT_FALSE_MESSAGE(RyanJsonChangeIntValue(doubleNode, 3), "ChangeIntValue(非 Int 节点) 应返回 False");
- TEST_ASSERT_FALSE_MESSAGE(RyanJsonChangeDoubleValue(intNode, 3.14), "ChangeDoubleValue(非 Double 节点) 应返回 False");
- TEST_ASSERT_FALSE_MESSAGE(RyanJsonChangeBoolValue(intNode, RyanJsonTrue), "ChangeBoolValue(非 Bool 节点) 应返回 False");
- TEST_ASSERT_FALSE_MESSAGE(RyanJsonChangeStringValue(intNode, "x"), "ChangeStringValue(非 String 节点) 应返回 False");
- TEST_ASSERT_FALSE_MESSAGE(RyanJsonChangeKey(strNode, "k2"), "ChangeKey(无Key的 String 节点) 应返回 False");
- TEST_ASSERT_FALSE_MESSAGE(RyanJsonChangeKey(keyedStrNode, NULL), "ChangeKey(key!=NULL 前置条件) 应返回 False");
- // 失败后原值应保持不变
- TEST_ASSERT_TRUE_MESSAGE(RyanJsonGetBoolValue(boolNode), "bool 初始值错误");
- TEST_ASSERT_FALSE_MESSAGE(RyanJsonChangeIntValue(boolNode, 9), "ChangeIntValue(Bool 节点) 应返回 False");
- TEST_ASSERT_TRUE_MESSAGE(RyanJsonGetBoolValue(boolNode), "失败后 bool 值不应变化");
- TEST_ASSERT_EQUAL_INT_MESSAGE(1, RyanJsonGetIntValue(intNode), "int 初始值错误");
- TEST_ASSERT_FALSE_MESSAGE(RyanJsonChangeDoubleValue(intNode, 8.8), "ChangeDoubleValue(Int 节点) 应返回 False");
- TEST_ASSERT_EQUAL_INT_MESSAGE(1, RyanJsonGetIntValue(intNode), "失败后 int 值不应变化");
- TEST_ASSERT_EQUAL_STRING_MESSAGE("v", RyanJsonGetStringValue(keyedStrNode), "string 初始值错误");
- TEST_ASSERT_FALSE_MESSAGE(RyanJsonChangeBoolValue(keyedStrNode, RyanJsonFalse), "ChangeBoolValue(String 节点) 应返回 False");
- TEST_ASSERT_EQUAL_STRING_MESSAGE("v", RyanJsonGetStringValue(keyedStrNode), "失败后 string 值不应变化");
- TEST_ASSERT_EQUAL_STRING_MESSAGE("k", RyanJsonGetKey(keyedStrNode), "失败后 key 不应变化");
- // 有 key 的非 String 节点允许改 key,且值不应改变
- TEST_ASSERT_TRUE_MESSAGE(RyanJsonChangeKey(keyedBoolNode, "flag2"), "ChangeKey(Bool+Key 节点) 应成功");
- TEST_ASSERT_EQUAL_STRING_MESSAGE("flag2", RyanJsonGetKey(keyedBoolNode), "Bool 节点改 key 失败");
- TEST_ASSERT_TRUE_MESSAGE(RyanJsonGetBoolValue(keyedBoolNode), "Bool 节点改 key 后 value 不应变化");
- RyanJsonDelete(intNode);
- RyanJsonDelete(doubleNode);
- RyanJsonDelete(boolNode);
- RyanJsonDelete(strNode);
- RyanJsonDelete(keyedStrNode);
- RyanJsonDelete(keyedBoolNode);
- }
- static void testChangeKeyDuplicatePolicy(void)
- {
- // 这个用例只证明“改成已存在 key”时的策略差异:
- // strict 必须拒绝,non-strict 可以成功;不重复覆盖后续查询/roundtrip 语义。
- RyanJson_t obj = RyanJsonCreateObject();
- TEST_ASSERT_NOT_NULL(obj);
- TEST_ASSERT_TRUE(RyanJsonAddIntToObject(obj, "a", 1));
- TEST_ASSERT_TRUE(RyanJsonAddIntToObject(obj, "b", 2));
- RyanJson_t bNode = RyanJsonGetObjectByKey(obj, "b");
- TEST_ASSERT_NOT_NULL_MESSAGE(bNode, "前置条件失败:缺少 key=b");
- #if true == RyanJsonStrictObjectKeyCheck
- TEST_ASSERT_FALSE_MESSAGE(RyanJsonChangeKey(bNode, "a"), "严格模式下 ChangeKey 重复 key 应失败");
- TEST_ASSERT_NOT_NULL_MESSAGE(RyanJsonGetObjectByKey(obj, "b"), "严格模式失败后原 key 应保持不变");
- TEST_ASSERT_EQUAL_INT_MESSAGE(2, RyanJsonGetIntValue(RyanJsonGetObjectByKey(obj, "b")), "严格模式失败后节点值不应变化");
- #else
- TEST_ASSERT_TRUE_MESSAGE(RyanJsonChangeKey(bNode, "a"), "非严格模式下 ChangeKey 重复 key 应成功");
- TEST_ASSERT_NULL_MESSAGE(RyanJsonGetObjectByKey(obj, "b"), "非严格模式成功后旧 key 应不存在");
- TEST_ASSERT_EQUAL_UINT32_MESSAGE(2U, RyanJsonGetSize(obj), "非严格模式成功后 Object 元素数量应保持不变");
- #endif
- RyanJsonDelete(obj);
- }
- static void testChangeKeySameTextNoOp(void)
- {
- // 同文本改 key 应视为 no-op:
- // 既不能被 strict 重复 key 检查误伤,也不能替换节点身份或污染兄弟节点。
- RyanJson_t obj = RyanJsonCreateObject();
- TEST_ASSERT_NOT_NULL(obj);
- TEST_ASSERT_TRUE(RyanJsonAddIntToObject(obj, "a", 1));
- TEST_ASSERT_TRUE(RyanJsonAddStringToObject(obj, "b", "keep"));
- RyanJson_t aNode = RyanJsonGetObjectToKey(obj, "a");
- RyanJson_t bNode = RyanJsonGetObjectToKey(obj, "b");
- TEST_ASSERT_NOT_NULL(aNode);
- TEST_ASSERT_NOT_NULL(bNode);
- TEST_ASSERT_TRUE_MESSAGE(RyanJsonChangeKey(aNode, "a"), "ChangeKey(同文本 key) 应成功");
- TEST_ASSERT_EQUAL_PTR_MESSAGE(aNode, RyanJsonGetObjectToKey(obj, "a"), "同文本 ChangeKey 不应替换节点身份");
- TEST_ASSERT_EQUAL_INT_MESSAGE(1, RyanJsonGetIntValue(RyanJsonGetObjectToKey(obj, "a")), "同文本 ChangeKey 后值不应变化");
- TEST_ASSERT_EQUAL_STRING_MESSAGE("keep", RyanJsonGetStringValue(RyanJsonGetObjectToKey(obj, "b")),
- "同文本 ChangeKey 不应影响其他兄弟节点");
- TEST_ASSERT_EQUAL_UINT32_MESSAGE(2U, RyanJsonGetSize(obj), "同文本 ChangeKey 不应改变 Object 元素数量");
- RyanJsonDelete(obj);
- }
- static void testChangeValueStress(void)
- {
- RyanJson_t root = RyanJsonCreateObject();
- RyanJsonAddStringToObject(root, "k", "v");
- // 高频修改 strValue
- // 变为超长
- char *longStr = (char *)malloc(1000);
- memset(longStr, 'A', 999);
- longStr[999] = '\0';
- TEST_ASSERT_TRUE(RyanJsonChangeStringValue(RyanJsonGetObjectByKey(root, "k"), longStr));
- TEST_ASSERT_EQUAL_STRING(longStr, RyanJsonGetStringValue(RyanJsonGetObjectByKey(root, "k")));
- // 变为特殊字符
- const char *specials = "\t\n\r\b\f\"\\/";
- TEST_ASSERT_TRUE(RyanJsonChangeStringValue(RyanJsonGetObjectByKey(root, "k"), specials));
- TEST_ASSERT_EQUAL_STRING(specials, RyanJsonGetStringValue(RyanJsonGetObjectByKey(root, "k")));
- // 变为空串
- TEST_ASSERT_TRUE(RyanJsonChangeStringValue(RyanJsonGetObjectByKey(root, "k"), ""));
- TEST_ASSERT_EQUAL_STRING("", RyanJsonGetStringValue(RyanJsonGetObjectByKey(root, "k")));
- // 高频修改 key
- // 变为超长
- TEST_ASSERT_TRUE(RyanJsonChangeKey(RyanJsonGetObjectByKey(root, "k"), longStr));
- TEST_ASSERT_TRUE(RyanJsonHasObjectByKey(root, longStr));
- // 变为空 key
- TEST_ASSERT_TRUE(RyanJsonChangeKey(RyanJsonGetObjectByKey(root, longStr), ""));
- TEST_ASSERT_TRUE(RyanJsonHasObjectByKey(root, ""));
- free(longStr);
- RyanJsonDelete(root);
- }
- static void testChangeScalarAndStorageMode(void)
- {
- char jsonstr[] =
- "{\"inter\":16,\"double\":16.89,\"string\":\"hello\",\"boolTrue\":true,\"boolFalse\":false,\"null\":null,"
- "\"item\":{\"inter\":16,\"double\":16.89,\"string\":\"hello\",\"boolTrue\":true,\"boolFalse\":false,\"null\":null},"
- "\"arrayInt\":[16,16,16,16,16],\"arrayDouble\":[16.89,16.89,16.89,16.89,16.89],"
- "\"arrayString\":[\"hello\",\"hello\",\"hello\",\"hello\",\"hello\"],"
- "\"array\":[16,16.89,\"hello\",true,false,null],"
- "\"arrayItem\":[{\"inter\":16,\"double\":16.89,\"string\":\"hello\",\"boolTrue\":true,\"boolFalse\":false,\"null\":null},"
- "{\"inter\":16,\"double\":16.89,\"string\":\"hello\",\"boolTrue\":true,\"boolFalse\":false,\"null\":null}],"
- "\"string2222\":\"hello\",\"0\":\"1\",\"nameaaaaaaaaaaaaaaaaaaaaaaaaaaaa\":\"Mash\",\"2\":\"3\",\"name\":"
- "\"Mashaaaaaaaaaaaaaaaaaaaaaaaa\"}";
- RyanJson_t jsonRoot = RyanJsonParse(jsonstr);
- TEST_ASSERT_NOT_NULL_MESSAGE(jsonRoot, "解析基础 Json 失败");
- /**
- * @brief 修改基本类型
- */
- // 修改 Int
- RyanJsonChangeIntValue(RyanJsonGetObjectToKey(jsonRoot, "inter"), 20);
- TEST_ASSERT_TRUE_MESSAGE(RyanJsonIsInt(RyanJsonGetObjectToKey(jsonRoot, "inter")), "字段 'inter' 修改后不是 Int");
- TEST_ASSERT_EQUAL_INT_MESSAGE(20, RyanJsonGetIntValue(RyanJsonGetObjectToKey(jsonRoot, "inter")), "字段 'inter' 值不匹配");
- // 修改 Double
- RyanJsonChangeDoubleValue(RyanJsonGetObjectToKey(jsonRoot, "double"), 20.89);
- TEST_ASSERT_TRUE_MESSAGE(RyanJsonIsDouble(RyanJsonGetObjectToKey(jsonRoot, "double")), "字段 'double' 修改后不是 Double");
- TEST_ASSERT_TRUE_MESSAGE(RyanJsonCompareDouble(RyanJsonGetDoubleValue(RyanJsonGetObjectToKey(jsonRoot, "double")), 20.89),
- "字段 'double' 值不匹配");
- // 修改 Key (inline 模式,不超过长度)
- RyanJsonChangeKey(RyanJsonGetObjectByKey(jsonRoot, "0"), "type");
- TEST_ASSERT_EQUAL_STRING_MESSAGE("type", RyanJsonGetKey(RyanJsonGetObjectToKey(jsonRoot, "type")), "字段 'type' 的 Key 不匹配");
- TEST_ASSERT_EQUAL_STRING_MESSAGE("1", RyanJsonGetStringValue(RyanJsonGetObjectToKey(jsonRoot, "type")), "字段 'type' 的内容不匹配");
- // 修改 Key (从 inline 转 ptr 模式)
- RyanJsonChangeKey(RyanJsonGetObjectByKey(jsonRoot, "type"), "type000000000000000");
- TEST_ASSERT_EQUAL_STRING_MESSAGE("type000000000000000", RyanJsonGetKey(RyanJsonGetObjectToKey(jsonRoot, "type000000000000000")),
- "长 Key 模式下的 Key 不匹配");
- TEST_ASSERT_EQUAL_STRING_MESSAGE("1", RyanJsonGetStringValue(RyanJsonGetObjectToKey(jsonRoot, "type000000000000000")),
- "长 Key 模式下的内容不匹配");
- // 修改 Key (从 ptr 转 inline 模式)
- RyanJsonChangeKey(RyanJsonGetObjectByKey(jsonRoot, "nameaaaaaaaaaaaaaaaaaaaaaaaaaaaa"), "na");
- TEST_ASSERT_EQUAL_STRING_MESSAGE("na", RyanJsonGetKey(RyanJsonGetObjectToKey(jsonRoot, "na")), "回退到 inline 模式的 Key 不匹配");
- TEST_ASSERT_EQUAL_STRING_MESSAGE("Mash", RyanJsonGetStringValue(RyanJsonGetObjectToKey(jsonRoot, "na")),
- "回退到 inline 模式的内容不匹配");
- // 修改 Value (inline 模式,不超过长度)
- RyanJsonChangeStringValue(RyanJsonGetObjectByKey(jsonRoot, "2"), "type");
- TEST_ASSERT_EQUAL_STRING_MESSAGE("2", RyanJsonGetKey(RyanJsonGetObjectToKey(jsonRoot, "2")), "字段 '2' 的 Key 不匹配");
- TEST_ASSERT_EQUAL_STRING_MESSAGE("type", RyanJsonGetStringValue(RyanJsonGetObjectToKey(jsonRoot, "2")), "字段 '2' 的新内容不匹配");
- // 修改 Value (从 ptr 转 inline 模式)
- RyanJsonChangeStringValue(RyanJsonGetObjectByKey(jsonRoot, "name"), "Ma");
- TEST_ASSERT_EQUAL_STRING_MESSAGE("name", RyanJsonGetKey(RyanJsonGetObjectToKey(jsonRoot, "name")), "字段 'name' 的 Key 不匹配");
- TEST_ASSERT_EQUAL_STRING_MESSAGE("Ma", RyanJsonGetStringValue(RyanJsonGetObjectToKey(jsonRoot, "name")),
- "回退到 inline 模式的 Value 不匹配");
- // 修改 Value (从 ptr 转 ptr 模式)
- RyanJsonChangeStringValue(RyanJsonGetObjectByKey(jsonRoot, "name"), "Mashaaaaaaaaaaaaaaaaaaaaaaaa");
- TEST_ASSERT_EQUAL_STRING_MESSAGE("name", RyanJsonGetKey(RyanJsonGetObjectToKey(jsonRoot, "name")), "字段 'name' 的 Key 不匹配");
- TEST_ASSERT_EQUAL_STRING_MESSAGE("Mashaaaaaaaaaaaaaaaaaaaaaaaa", RyanJsonGetStringValue(RyanJsonGetObjectToKey(jsonRoot, "name")),
- "长 Value 模式下的内容不匹配");
- // 修改 String
- RyanJsonChangeStringValue(RyanJsonGetObjectToKey(jsonRoot, "string"), "world");
- TEST_ASSERT_TRUE_MESSAGE(RyanJsonIsString(RyanJsonGetObjectToKey(jsonRoot, "string")), "字段 'string' 修改后不是 String");
- TEST_ASSERT_EQUAL_STRING_MESSAGE("world", RyanJsonGetStringValue(RyanJsonGetObjectToKey(jsonRoot, "string")),
- "字段 'string' 内容不匹配");
- // 修改 boolValue (true -> false)
- RyanJsonChangeBoolValue(RyanJsonGetObjectToKey(jsonRoot, "boolTrue"), RyanJsonFalse);
- TEST_ASSERT_TRUE_MESSAGE(RyanJsonIsBool(RyanJsonGetObjectToKey(jsonRoot, "boolTrue")), "字段 'boolTrue' 类型错误");
- TEST_ASSERT_EQUAL_INT_MESSAGE(RyanJsonFalse, RyanJsonGetBoolValue(RyanJsonGetObjectToKey(jsonRoot, "boolTrue")),
- "字段 'boolTrue' 值错误");
- // 修改 boolValue (false -> true)
- RyanJsonChangeBoolValue(RyanJsonGetObjectToKey(jsonRoot, "boolFalse"), RyanJsonTrue);
- TEST_ASSERT_TRUE_MESSAGE(RyanJsonIsBool(RyanJsonGetObjectToKey(jsonRoot, "boolFalse")), "字段 'boolFalse' 类型错误");
- TEST_ASSERT_EQUAL_INT_MESSAGE(RyanJsonTrue, RyanJsonGetBoolValue(RyanJsonGetObjectToKey(jsonRoot, "boolFalse")),
- "字段 'boolFalse' 值错误");
- /**
- * @brief 修改 Array 元素 (arrayInt)
- */
- RyanJsonChangeIntValue(RyanJsonGetObjectToIndex(RyanJsonGetObjectToKey(jsonRoot, "arrayInt"), 0), 99);
- TEST_ASSERT_EQUAL_INT_MESSAGE(99, RyanJsonGetIntValue(RyanJsonGetObjectToIndex(RyanJsonGetObjectToKey(jsonRoot, "arrayInt"), 0)),
- "Array 元素修改失败");
- /**
- * @brief 修改 Array 元素 (arrayDouble)
- */
- RyanJsonChangeDoubleValue(RyanJsonGetObjectToIndex(RyanJsonGetObjectToKey(jsonRoot, "arrayDouble"), 1), 99.99);
- TEST_ASSERT_TRUE_MESSAGE(
- RyanJsonCompareDouble(RyanJsonGetDoubleValue(RyanJsonGetObjectToIndex(RyanJsonGetObjectToKey(jsonRoot, "arrayDouble"), 1)),
- 99.99),
- "Array 浮点元素修改失败");
- /**
- * @brief 修改 Array 元素 (arrayString)
- */
- RyanJsonChangeStringValue(RyanJsonGetObjectToIndex(RyanJsonGetObjectToKey(jsonRoot, "arrayString"), 2), "changedString");
- TEST_ASSERT_EQUAL_STRING_MESSAGE(
- "changedString", RyanJsonGetStringValue(RyanJsonGetObjectToIndex(RyanJsonGetObjectToKey(jsonRoot, "arrayString"), 2)),
- "ArrayString 元素修改失败");
- /**
- * @brief 修改嵌套 Object
- */
- RyanJson_t nestedObj = RyanJsonGetObjectToKey(jsonRoot, "item");
- RyanJsonChangeStringValue(RyanJsonGetObjectToKey(nestedObj, "string"), "nestedWorld");
- TEST_ASSERT_EQUAL_STRING_MESSAGE("nestedWorld", RyanJsonGetStringValue(RyanJsonGetObjectToKey(nestedObj, "string")),
- "嵌套 Object 修改失败");
- /**
- * @brief 修改 Array 中 Object 的字段 (arrayItem[0].inter -> 123)
- */
- RyanJson_t arrayItem0 = RyanJsonGetObjectToIndex(RyanJsonGetObjectToKey(jsonRoot, "arrayItem"), 0);
- RyanJsonChangeIntValue(RyanJsonGetObjectToKey(arrayItem0, "inter"), 123);
- TEST_ASSERT_EQUAL_INT_MESSAGE(123, RyanJsonGetIntValue(RyanJsonGetObjectToKey(arrayItem0, "inter")), "Array 中 Object 的字段修改失败");
- char *str = RyanJsonPrint(jsonRoot, 1024, RyanJsonTrue, NULL);
- RyanJsonFree(str);
- RyanJsonDelete(jsonRoot);
- /**
- * @brief 验证创建后的内存模式库切换 (Create -> Change)
- */
- RyanJson_t json = RyanJsonCreateString("key", "val");
- TEST_ASSERT_NOT_NULL(json);
- TEST_ASSERT_EQUAL_STRING_MESSAGE("val", RyanJsonGetStringValue(json), "Inline 模式初次设置失败");
- // 修改为长 String (触发自动切换到 ptr 模式)
- const char *longVal = "This is a very long string that definitely exceeds the inline threshold of RyanJson structure.";
- RyanJsonChangeStringValue(json, longVal);
- TEST_ASSERT_EQUAL_STRING_MESSAGE(longVal, RyanJsonGetStringValue(json), "切换到 Ptr 模式后值错误");
- // 修改回短 String
- RyanJsonChangeStringValue(json, "new");
- TEST_ASSERT_EQUAL_STRING_MESSAGE("new", RyanJsonGetStringValue(json), "切回短 String 后值错误");
- // 修改 Key 为长 Key
- const char *longKey = "a_very_long_key_name_to_trigger_ptr_mode_for_key_storage_in_ryanjson";
- RyanJsonChangeKey(json, longKey);
- TEST_ASSERT_EQUAL_STRING_MESSAGE(longKey, RyanJsonGetKey(json), "Key 切换到 Ptr 模式后错误");
- RyanJsonDelete(json);
- }
- static void testChangeInlineCalcBoundary(void)
- {
- /**
- * @brief 这个测试的目标:
- * @details
- * 验证 RyanJsonInternalChangeString 的容量判断在“临界长度”处是否正确;
- * 验证 value 和 key 两条路径都会发生 inline <-> ptr 的双向切换;
- * 不依赖固定常量(例如 keyLen=255),而是基于当前编译配置动态计算边界。
- *
- * 背景:
- * RyanJsonInternalChangeString 的判定核心是:
- * if ((mallocSize + keyLenFieldBytes) <= RyanJsonInlineStringSize) => inline
- * else => ptr
- *
- * 因此这里分别构造:
- * - 恰好满足 <= 的输入(应保持/切回 inline)
- * - 超过 1 字节的输入(应切到 ptr)
- */
- uint32_t inlineSize = RyanJsonInlineStringSize;
- uint32_t fixedKeyLen = 1;
- uint32_t fixedValueLen = 1;
- /**
- * @brief value 边界计算(固定 key="k")。
- * @details
- * 需要占用的总 inline 空间 = keyFieldLen + keyLen + '\0' + valueLen + '\0'
- * 先算出不含 value 内容本体时的固定开销 baseNeedForValue,
- * 再反推出 value 能放进 inline 的最大长度 maxInlineValueLen。
- */
- uint32_t keyFieldLenForValue = RyanJsonInternalDecodeKeyLenField(RyanJsonInternalCalcLenBytes(fixedKeyLen));
- uint32_t baseNeedForValue = keyFieldLenForValue + fixedKeyLen + 1 + 1; // keyField + key + '\0' + value '\0'
- TEST_ASSERT_TRUE_MESSAGE(inlineSize >= baseNeedForValue, "inline 大小异常,无法完成边界测试");
- uint32_t maxInlineValueLen = inlineSize - baseNeedForValue;
- // valueInline: 恰好命中 inline 临界;valuePtr: 比临界多 1 字节,必须走 ptr
- char *valueInline = (char *)malloc((size_t)maxInlineValueLen + 1U);
- char *valuePtr = (char *)malloc((size_t)maxInlineValueLen + 2U);
- TEST_ASSERT_NOT_NULL(valueInline);
- TEST_ASSERT_NOT_NULL(valuePtr);
- memset(valueInline, 'v', maxInlineValueLen);
- valueInline[maxInlineValueLen] = '\0';
- memset(valuePtr, 'p', maxInlineValueLen + 1U);
- valuePtr[maxInlineValueLen + 1U] = '\0';
- RyanJson_t valueNode = RyanJsonCreateString("k", "");
- TEST_ASSERT_NOT_NULL(valueNode);
- // 临界长度:应为 inline
- TEST_ASSERT_TRUE(RyanJsonChangeStringValue(valueNode, valueInline));
- TEST_ASSERT_FALSE_MESSAGE(RyanJsonGetPayloadStrIsPtrByFlag(valueNode), "value 临界长度应为 inline");
- // 超临界:应切到 ptr
- TEST_ASSERT_TRUE(RyanJsonChangeStringValue(valueNode, valuePtr));
- TEST_ASSERT_TRUE_MESSAGE(RyanJsonGetPayloadStrIsPtrByFlag(valueNode), "value 超过临界长度应切到 ptr");
- // 回退到临界:应能从 ptr 回到 inline(覆盖回切逻辑)
- TEST_ASSERT_TRUE(RyanJsonChangeStringValue(valueNode, valueInline));
- TEST_ASSERT_FALSE_MESSAGE(RyanJsonGetPayloadStrIsPtrByFlag(valueNode), "value 回退到临界长度应切回 inline");
- /**
- * @brief key 边界计算(固定 value="v")。
- * @details
- * keyLen 变化时,keyLenField 可能从 1/2/4 字节变化,
- * 所以不能硬编码某个 keyLen,必须遍历找到“当前配置下最大可 inline 的 key 长度”。
- */
- uint32_t maxInlineKeyLen = 0;
- for (uint32_t keyLen = 0; keyLen <= inlineSize; keyLen++)
- {
- uint32_t keyFieldLen = RyanJsonInternalDecodeKeyLenField(RyanJsonInternalCalcLenBytes(keyLen));
- uint32_t need = keyFieldLen + keyLen + 1 + fixedValueLen + 1;
- if (need <= inlineSize) { maxInlineKeyLen = keyLen; }
- }
- // 交叉校验:maxInlineKeyLen 应满足 inline;maxInlineKeyLen+1 应超过 inline
- uint32_t inlineNeed =
- RyanJsonInternalDecodeKeyLenField(RyanJsonInternalCalcLenBytes(maxInlineKeyLen)) + maxInlineKeyLen + 1 + fixedValueLen + 1;
- uint32_t ptrNeed = RyanJsonInternalDecodeKeyLenField(RyanJsonInternalCalcLenBytes(maxInlineKeyLen + 1U)) + (maxInlineKeyLen + 1U) +
- 1 + fixedValueLen + 1;
- TEST_ASSERT_TRUE_MESSAGE(inlineNeed <= inlineSize, "inline key 临界值计算错误");
- TEST_ASSERT_TRUE_MESSAGE(ptrNeed > inlineSize, "ptr key 临界值计算错误");
- // keyInline: 恰好临界;keyPtr: 超临界 1 字节
- char *keyInline = (char *)malloc((size_t)maxInlineKeyLen + 1U);
- char *keyPtr = (char *)malloc((size_t)maxInlineKeyLen + 2U);
- TEST_ASSERT_NOT_NULL(keyInline);
- TEST_ASSERT_NOT_NULL(keyPtr);
- memset(keyInline, 'k', maxInlineKeyLen);
- keyInline[maxInlineKeyLen] = '\0';
- memset(keyPtr, 'K', maxInlineKeyLen + 1U);
- keyPtr[maxInlineKeyLen + 1U] = '\0';
- RyanJson_t keyNode = RyanJsonCreateString("", "v");
- TEST_ASSERT_NOT_NULL(keyNode);
- // key 临界长度:应为 inline
- TEST_ASSERT_TRUE(RyanJsonChangeKey(keyNode, keyInline));
- TEST_ASSERT_FALSE_MESSAGE(RyanJsonGetPayloadStrIsPtrByFlag(keyNode), "key 临界长度应为 inline");
- // key 超临界:应切到 ptr
- TEST_ASSERT_TRUE(RyanJsonChangeKey(keyNode, keyPtr));
- TEST_ASSERT_TRUE_MESSAGE(RyanJsonGetPayloadStrIsPtrByFlag(keyNode), "key 超过临界长度应切到 ptr");
- // key 回退临界:应切回 inline
- TEST_ASSERT_TRUE(RyanJsonChangeKey(keyNode, keyInline));
- TEST_ASSERT_FALSE_MESSAGE(RyanJsonGetPayloadStrIsPtrByFlag(keyNode), "key 回退到临界长度应切回 inline");
- // 释放临时资源,避免单测内存泄漏
- RyanJsonDelete(valueNode);
- RyanJsonDelete(keyNode);
- free(valueInline);
- free(valuePtr);
- free(keyInline);
- free(keyPtr);
- }
- static void testChangeNumericStringIdFidelity(void)
- {
- // 覆盖“NumberString ID”保真:
- // - 解析后仍为 String 类型,且保留前导零与大 Int 文本;
- // - Print/Parse 往返不数值化;
- // - ChangeStringValue 不会触发数值化或去零。
- const char *text = "{\"id\":\"00123\",\"big\":\"9007199254740993\",\"arr\":[\"00123\",\"9007199254740993\"]}";
- RyanJson_t root = RyanJsonParse(text);
- TEST_ASSERT_NOT_NULL_MESSAGE(root, "Number String 样本解析失败");
- RyanJson_t idNode = RyanJsonGetObjectByKey(root, "id");
- RyanJson_t bigNode = RyanJsonGetObjectByKey(root, "big");
- RyanJson_t arr = RyanJsonGetObjectByKey(root, "arr");
- TEST_ASSERT_NOT_NULL(idNode);
- TEST_ASSERT_NOT_NULL(bigNode);
- TEST_ASSERT_NOT_NULL(arr);
- TEST_ASSERT_TRUE(RyanJsonIsArray(arr));
- TEST_ASSERT_TRUE(RyanJsonIsString(idNode));
- TEST_ASSERT_TRUE(RyanJsonIsString(bigNode));
- TEST_ASSERT_EQUAL_STRING("00123", RyanJsonGetStringValue(idNode));
- TEST_ASSERT_EQUAL_STRING("9007199254740993", RyanJsonGetStringValue(bigNode));
- TEST_ASSERT_TRUE(RyanJsonIsString(RyanJsonGetObjectByIndex(arr, 0)));
- TEST_ASSERT_TRUE(RyanJsonIsString(RyanJsonGetObjectByIndex(arr, 1)));
- TEST_ASSERT_EQUAL_STRING("00123", RyanJsonGetStringValue(RyanJsonGetObjectByIndex(arr, 0)));
- TEST_ASSERT_EQUAL_STRING("9007199254740993", RyanJsonGetStringValue(RyanJsonGetObjectByIndex(arr, 1)));
- char *printed = RyanJsonPrint(root, 160, RyanJsonFalse, NULL);
- TEST_ASSERT_NOT_NULL(printed);
- RyanJson_t roundtrip = RyanJsonParse(printed);
- TEST_ASSERT_NOT_NULL(roundtrip);
- TEST_ASSERT_EQUAL_STRING("00123", RyanJsonGetStringValue(RyanJsonGetObjectByKey(roundtrip, "id")));
- TEST_ASSERT_EQUAL_STRING("9007199254740993", RyanJsonGetStringValue(RyanJsonGetObjectByKey(roundtrip, "big")));
- TEST_ASSERT_TRUE(RyanJsonIsString(RyanJsonGetObjectByKey(roundtrip, "id")));
- TEST_ASSERT_TRUE(RyanJsonIsString(RyanJsonGetObjectByKey(roundtrip, "big")));
- TEST_ASSERT_TRUE(RyanJsonIsString(RyanJsonGetObjectByIndex(RyanJsonGetObjectByKey(roundtrip, "arr"), 0)));
- TEST_ASSERT_TRUE(RyanJsonIsString(RyanJsonGetObjectByIndex(RyanJsonGetObjectByKey(roundtrip, "arr"), 1)));
- TEST_ASSERT_EQUAL_STRING("00123", RyanJsonGetStringValue(RyanJsonGetObjectByIndex(RyanJsonGetObjectByKey(roundtrip, "arr"), 0)));
- TEST_ASSERT_EQUAL_STRING("9007199254740993",
- RyanJsonGetStringValue(RyanJsonGetObjectByIndex(RyanJsonGetObjectByKey(roundtrip, "arr"), 1)));
- TEST_ASSERT_TRUE(RyanJsonChangeStringValue(idNode, "00001"));
- TEST_ASSERT_TRUE(RyanJsonChangeStringValue(bigNode, "90071992547409930"));
- TEST_ASSERT_TRUE(RyanJsonChangeStringValue(RyanJsonGetObjectByIndex(arr, 0), "00001"));
- TEST_ASSERT_TRUE(RyanJsonChangeStringValue(RyanJsonGetObjectByIndex(arr, 1), "90071992547409930"));
- TEST_ASSERT_TRUE(RyanJsonIsString(RyanJsonGetObjectByKey(root, "id")));
- TEST_ASSERT_TRUE(RyanJsonIsString(RyanJsonGetObjectByKey(root, "big")));
- TEST_ASSERT_EQUAL_STRING("00001", RyanJsonGetStringValue(RyanJsonGetObjectByKey(root, "id")));
- TEST_ASSERT_EQUAL_STRING("90071992547409930", RyanJsonGetStringValue(RyanJsonGetObjectByKey(root, "big")));
- TEST_ASSERT_EQUAL_STRING("00001", RyanJsonGetStringValue(RyanJsonGetObjectByIndex(arr, 0)));
- TEST_ASSERT_EQUAL_STRING("90071992547409930", RyanJsonGetStringValue(RyanJsonGetObjectByIndex(arr, 1)));
- char *printed2 = RyanJsonPrint(root, 160, RyanJsonFalse, NULL);
- TEST_ASSERT_NOT_NULL(printed2);
- RyanJson_t roundtrip2 = RyanJsonParse(printed2);
- TEST_ASSERT_NOT_NULL(roundtrip2);
- TEST_ASSERT_EQUAL_STRING("00001", RyanJsonGetStringValue(RyanJsonGetObjectByKey(roundtrip2, "id")));
- TEST_ASSERT_EQUAL_STRING("90071992547409930", RyanJsonGetStringValue(RyanJsonGetObjectByKey(roundtrip2, "big")));
- TEST_ASSERT_TRUE(RyanJsonIsString(RyanJsonGetObjectByKey(roundtrip2, "id")));
- TEST_ASSERT_TRUE(RyanJsonIsString(RyanJsonGetObjectByKey(roundtrip2, "big")));
- TEST_ASSERT_TRUE(RyanJsonIsString(RyanJsonGetObjectByIndex(RyanJsonGetObjectByKey(roundtrip2, "arr"), 0)));
- TEST_ASSERT_TRUE(RyanJsonIsString(RyanJsonGetObjectByIndex(RyanJsonGetObjectByKey(roundtrip2, "arr"), 1)));
- TEST_ASSERT_EQUAL_STRING("00001", RyanJsonGetStringValue(RyanJsonGetObjectByIndex(RyanJsonGetObjectByKey(roundtrip2, "arr"), 0)));
- TEST_ASSERT_EQUAL_STRING("90071992547409930",
- RyanJsonGetStringValue(RyanJsonGetObjectByIndex(RyanJsonGetObjectByKey(roundtrip2, "arr"), 1)));
- RyanJsonDelete(roundtrip2);
- RyanJsonFree(printed2);
- RyanJsonDelete(roundtrip);
- RyanJsonFree(printed);
- RyanJsonDelete(root);
- }
- static void testChangeFailureThenSuccessChainAgainstExpected(void)
- {
- // 复杂链路:
- // Parse -> Duplicate(snapshot) -> 连续失败 Change -> Compare(snapshot)
- // -> 连续成功 Change(含 ChangeKey) -> CompareOnlyKey 差异 -> 对齐期望文档 -> Roundtrip。
- // 目标:
- // - 验证失败 Change 不会污染原树;
- // - 验证失败与成功交错后,结构和值仍可稳定收敛到期望语义;
- // - 覆盖 ChangeKey 引起的 key 结构变化对 CompareOnlyKey 的影响。
- const char *source = "{\"cfg\":{\"mode\":\"a\",\"retry\":1},\"arr\":[{\"k\":\"x\"},{\"k\":\"y\"}],\"flag\":true}";
- const char *expectText = "{\"cfg\":{\"mode\":\"b\",\"retry2\":9},\"arr\":[{\"k\":\"x\"},{\"k\":\"z\"}],\"flag\":false}";
- RyanJson_t root = RyanJsonParse(source);
- TEST_ASSERT_NOT_NULL_MESSAGE(root, "Change 链路样本解析失败");
- RyanJson_t snapshot = RyanJsonDuplicate(root);
- TEST_ASSERT_NOT_NULL_MESSAGE(snapshot, "Change 链路快照构造失败");
- RyanJson_t cfg = RyanJsonGetObjectToKey(root, "cfg");
- RyanJson_t arr = RyanJsonGetObjectToKey(root, "arr");
- RyanJson_t arr0 = RyanJsonGetObjectToIndex(arr, 0);
- RyanJson_t arr1 = RyanJsonGetObjectToIndex(arr, 1);
- TEST_ASSERT_NOT_NULL(cfg);
- TEST_ASSERT_NOT_NULL(arr);
- TEST_ASSERT_NOT_NULL(arr0);
- TEST_ASSERT_NOT_NULL(arr1);
- // 失败路径:类型不匹配 + Array 元素无 key,不应改变文档语义。
- TEST_ASSERT_FALSE_MESSAGE(RyanJsonChangeIntValue(RyanJsonGetObjectToKey(cfg, "mode"), 1), "String 节点 ChangeIntValue 应失败");
- TEST_ASSERT_FALSE_MESSAGE(RyanJsonChangeBoolValue(RyanJsonGetObjectToKey(arr0, "k"), RyanJsonTrue),
- "String 节点 ChangeBoolValue 应失败");
- TEST_ASSERT_FALSE_MESSAGE(RyanJsonChangeKey(arr0, "arr0"), "Array 元素无 key,ChangeKey 应失败");
- TEST_ASSERT_TRUE_MESSAGE(RyanJsonCompare(root, snapshot), "仅发生失败 Change 时,文档语义应与 snapshot 完全一致");
- // 成功路径:值修改 + key 修改。
- TEST_ASSERT_TRUE(RyanJsonChangeStringValue(RyanJsonGetObjectToKey(cfg, "mode"), "b"));
- TEST_ASSERT_TRUE(RyanJsonChangeIntValue(RyanJsonGetObjectToKey(cfg, "retry"), 9));
- TEST_ASSERT_TRUE(RyanJsonChangeStringValue(RyanJsonGetObjectToKey(arr1, "k"), "z"));
- TEST_ASSERT_TRUE(RyanJsonChangeBoolValue(RyanJsonGetObjectToKey(root, "flag"), RyanJsonFalse));
- TEST_ASSERT_TRUE_MESSAGE(RyanJsonChangeKey(RyanJsonGetObjectToKey(cfg, "retry"), "retry2"), "将 cfg.retry 重命名为 retry2 失败");
- TEST_ASSERT_FALSE(RyanJsonHasObjectToKey(cfg, "retry"));
- TEST_ASSERT_TRUE(RyanJsonHasObjectToKey(cfg, "retry2"));
- TEST_ASSERT_FALSE_MESSAGE(RyanJsonCompareOnlyKey(root, snapshot), "发生 key 结构变化后 CompareOnlyKey 应返回 False");
- RyanJson_t expect = RyanJsonParse(expectText);
- TEST_ASSERT_NOT_NULL(expect);
- TEST_ASSERT_TRUE_MESSAGE(RyanJsonCompare(root, expect), "Change 链路最终结果与期望文档不一致");
- char *printed = RyanJsonPrint(root, 160, RyanJsonFalse, NULL);
- TEST_ASSERT_NOT_NULL(printed);
- RyanJson_t roundtrip = RyanJsonParse(printed);
- TEST_ASSERT_NOT_NULL(roundtrip);
- TEST_ASSERT_TRUE(RyanJsonCompare(root, roundtrip));
- RyanJsonDelete(roundtrip);
- RyanJsonFree(printed);
- RyanJsonDelete(expect);
- RyanJsonDelete(snapshot);
- RyanJsonDelete(root);
- }
- void testChangeRunner(void)
- {
- UnitySetTestFile(__FILE__);
- RUN_TEST(testChangeEdgeCases);
- RUN_TEST(testChangeKeyDuplicatePolicy);
- RUN_TEST(testChangeKeySameTextNoOp);
- RUN_TEST(testChangeValueStress);
- RUN_TEST(testChangeScalarAndStorageMode);
- RUN_TEST(testChangeInlineCalcBoundary);
- RUN_TEST(testChangeNumericStringIdFidelity);
- RUN_TEST(testChangeFailureThenSuccessChainAgainstExpected);
- }
|