| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400 |
- // ArduinoJson - https://arduinojson.org
- // Copyright © 2014-2025, Benoit BLANCHON
- // MIT License
- #include <ArduinoJson.h>
- #include <catch.hpp>
- #include "Allocators.hpp"
- #include "Literals.hpp"
- using ArduinoJson::detail::sizeofObject;
- TEST_CASE("deserialize JSON object") {
- SpyingAllocator spy;
- JsonDocument doc(&spy);
- SECTION("An empty object") {
- DeserializationError err = deserializeJson(doc, "{}");
- JsonObject obj = doc.as<JsonObject>();
- REQUIRE(err == DeserializationError::Ok);
- REQUIRE(doc.is<JsonObject>());
- REQUIRE(obj.size() == 0);
- }
- SECTION("Quotes") {
- SECTION("Double quotes") {
- DeserializationError err = deserializeJson(doc, "{\"key\":\"value\"}");
- JsonObject obj = doc.as<JsonObject>();
- REQUIRE(err == DeserializationError::Ok);
- REQUIRE(doc.is<JsonObject>());
- REQUIRE(obj.size() == 1);
- REQUIRE(obj["key"] == "value");
- }
- SECTION("Single quotes") {
- DeserializationError err = deserializeJson(doc, "{'key':'value'}");
- JsonObject obj = doc.as<JsonObject>();
- REQUIRE(err == DeserializationError::Ok);
- REQUIRE(doc.is<JsonObject>());
- REQUIRE(obj.size() == 1);
- REQUIRE(obj["key"] == "value");
- }
- SECTION("No quotes") {
- DeserializationError err = deserializeJson(doc, "{key:'value'}");
- JsonObject obj = doc.as<JsonObject>();
- REQUIRE(err == DeserializationError::Ok);
- REQUIRE(doc.is<JsonObject>());
- REQUIRE(obj.size() == 1);
- REQUIRE(obj["key"] == "value");
- }
- SECTION("No quotes, allow underscore in key") {
- DeserializationError err = deserializeJson(doc, "{_k_e_y_:42}");
- JsonObject obj = doc.as<JsonObject>();
- REQUIRE(err == DeserializationError::Ok);
- REQUIRE(doc.is<JsonObject>());
- REQUIRE(obj.size() == 1);
- REQUIRE(obj["_k_e_y_"] == 42);
- }
- }
- SECTION("Spaces") {
- SECTION("Before the key") {
- DeserializationError err = deserializeJson(doc, "{ \"key\":\"value\"}");
- JsonObject obj = doc.as<JsonObject>();
- REQUIRE(err == DeserializationError::Ok);
- REQUIRE(doc.is<JsonObject>());
- REQUIRE(obj.size() == 1);
- REQUIRE(obj["key"] == "value");
- }
- SECTION("After the key") {
- DeserializationError err = deserializeJson(doc, "{\"key\" :\"value\"}");
- JsonObject obj = doc.as<JsonObject>();
- REQUIRE(err == DeserializationError::Ok);
- REQUIRE(doc.is<JsonObject>());
- REQUIRE(obj.size() == 1);
- REQUIRE(obj["key"] == "value");
- }
- SECTION("Before the value") {
- DeserializationError err = deserializeJson(doc, "{\"key\": \"value\"}");
- JsonObject obj = doc.as<JsonObject>();
- REQUIRE(err == DeserializationError::Ok);
- REQUIRE(doc.is<JsonObject>());
- REQUIRE(obj.size() == 1);
- REQUIRE(obj["key"] == "value");
- }
- SECTION("After the value") {
- DeserializationError err = deserializeJson(doc, "{\"key\":\"value\" }");
- JsonObject obj = doc.as<JsonObject>();
- REQUIRE(err == DeserializationError::Ok);
- REQUIRE(doc.is<JsonObject>());
- REQUIRE(obj.size() == 1);
- REQUIRE(obj["key"] == "value");
- }
- SECTION("Before the comma") {
- DeserializationError err =
- deserializeJson(doc, "{\"key1\":\"value1\" ,\"key2\":\"value2\"}");
- JsonObject obj = doc.as<JsonObject>();
- REQUIRE(err == DeserializationError::Ok);
- REQUIRE(doc.is<JsonObject>());
- REQUIRE(obj.size() == 2);
- REQUIRE(obj["key1"] == "value1");
- REQUIRE(obj["key2"] == "value2");
- }
- SECTION("After the comma") {
- DeserializationError err =
- deserializeJson(doc, "{\"key1\":\"value1\", \"key2\":\"value2\"}");
- JsonObject obj = doc.as<JsonObject>();
- REQUIRE(err == DeserializationError::Ok);
- REQUIRE(doc.is<JsonObject>());
- REQUIRE(obj.size() == 2);
- REQUIRE(obj["key1"] == "value1");
- REQUIRE(obj["key2"] == "value2");
- }
- }
- SECTION("Values types") {
- SECTION("String") {
- DeserializationError err =
- deserializeJson(doc, "{\"key1\":\"value1\",\"key2\":\"value2\"}");
- JsonObject obj = doc.as<JsonObject>();
- REQUIRE(err == DeserializationError::Ok);
- REQUIRE(doc.is<JsonObject>());
- REQUIRE(obj.size() == 2);
- REQUIRE(obj["key1"] == "value1");
- REQUIRE(obj["key2"] == "value2");
- }
- SECTION("Integer") {
- DeserializationError err =
- deserializeJson(doc, "{\"key1\":42,\"key2\":-42}");
- JsonObject obj = doc.as<JsonObject>();
- REQUIRE(err == DeserializationError::Ok);
- REQUIRE(doc.is<JsonObject>());
- REQUIRE(obj.size() == 2);
- REQUIRE(obj["key1"] == 42);
- REQUIRE(obj["key2"] == -42);
- }
- SECTION("Float") {
- DeserializationError err =
- deserializeJson(doc, "{\"key1\":12.345,\"key2\":-7E3}");
- JsonObject obj = doc.as<JsonObject>();
- REQUIRE(err == DeserializationError::Ok);
- REQUIRE(doc.is<JsonObject>());
- REQUIRE(obj.size() == 2);
- REQUIRE(obj["key1"].as<float>() == Approx(12.345f));
- REQUIRE(obj["key2"] == -7E3f);
- }
- SECTION("Double") {
- DeserializationError err =
- deserializeJson(doc, "{\"key1\":12.3456789,\"key2\":-7E89}");
- JsonObject obj = doc.as<JsonObject>();
- REQUIRE(err == DeserializationError::Ok);
- REQUIRE(doc.is<JsonObject>());
- REQUIRE(obj.size() == 2);
- REQUIRE(obj["key1"].as<double>() == Approx(12.3456789));
- REQUIRE(obj["key2"] == -7E89);
- }
- SECTION("Booleans") {
- DeserializationError err =
- deserializeJson(doc, "{\"key1\":true,\"key2\":false}");
- JsonObject obj = doc.as<JsonObject>();
- REQUIRE(err == DeserializationError::Ok);
- REQUIRE(doc.is<JsonObject>());
- REQUIRE(obj.size() == 2);
- REQUIRE(obj["key1"] == true);
- REQUIRE(obj["key2"] == false);
- }
- SECTION("Null") {
- DeserializationError err =
- deserializeJson(doc, "{\"key1\":null,\"key2\":null}");
- JsonObject obj = doc.as<JsonObject>();
- REQUIRE(err == DeserializationError::Ok);
- REQUIRE(doc.is<JsonObject>());
- REQUIRE(obj.size() == 2);
- REQUIRE(obj["key1"].as<const char*>() == 0);
- REQUIRE(obj["key2"].as<const char*>() == 0);
- }
- SECTION("Array") {
- char jsonString[] = " { \"ab\" : [ 1 , 2 ] , \"cd\" : [ 3 , 4 ] } ";
- DeserializationError err = deserializeJson(doc, jsonString);
- JsonObject obj = doc.as<JsonObject>();
- JsonArray array1 = obj["ab"];
- const JsonArray array2 = obj["cd"];
- JsonArray array3 = obj["ef"];
- REQUIRE(err == DeserializationError::Ok);
- REQUIRE(array1.isNull() == false);
- REQUIRE(array2.isNull() == false);
- REQUIRE(array3.isNull() == true);
- REQUIRE(2 == array1.size());
- REQUIRE(2 == array2.size());
- REQUIRE(0 == array3.size());
- REQUIRE(1 == array1[0].as<int>());
- REQUIRE(2 == array1[1].as<int>());
- REQUIRE(3 == array2[0].as<int>());
- REQUIRE(4 == array2[1].as<int>());
- REQUIRE(0 == array3[0].as<int>());
- }
- }
- SECTION("Premature null terminator") {
- SECTION("After opening brace") {
- DeserializationError err = deserializeJson(doc, "{");
- REQUIRE(err == DeserializationError::IncompleteInput);
- }
- SECTION("After key") {
- DeserializationError err = deserializeJson(doc, "{\"hello\"");
- REQUIRE(err == DeserializationError::IncompleteInput);
- }
- SECTION("After colon") {
- DeserializationError err = deserializeJson(doc, "{\"hello\":");
- REQUIRE(err == DeserializationError::IncompleteInput);
- }
- SECTION("After value") {
- DeserializationError err = deserializeJson(doc, "{\"hello\":\"world\"");
- REQUIRE(err == DeserializationError::IncompleteInput);
- }
- SECTION("After comma") {
- DeserializationError err = deserializeJson(doc, "{\"hello\":\"world\",");
- REQUIRE(err == DeserializationError::IncompleteInput);
- }
- }
- SECTION("Misc") {
- SECTION("A quoted key without value") {
- DeserializationError err = deserializeJson(doc, "{\"key\"}");
- REQUIRE(err == DeserializationError::InvalidInput);
- }
- SECTION("A non-quoted key without value") {
- DeserializationError err = deserializeJson(doc, "{key}");
- REQUIRE(err == DeserializationError::InvalidInput);
- }
- SECTION("A dangling comma") {
- DeserializationError err = deserializeJson(doc, "{\"key1\":\"value1\",}");
- REQUIRE(err == DeserializationError::InvalidInput);
- }
- SECTION("null as a key") {
- DeserializationError err = deserializeJson(doc, "{null:\"value\"}");
- REQUIRE(err == DeserializationError::Ok);
- }
- SECTION("Repeated key") {
- DeserializationError err =
- deserializeJson(doc, "{alfa:{bravo:{charlie:1}},alfa:2}");
- REQUIRE(err == DeserializationError::Ok);
- REQUIRE(doc.as<std::string>() == "{\"alfa\":2}");
- REQUIRE(spy.log() ==
- AllocatorLog{
- Allocate(sizeofStringBuffer()),
- Allocate(sizeofPool()),
- Reallocate(sizeofStringBuffer(), sizeofString("alfa")),
- Allocate(sizeofStringBuffer()),
- Reallocate(sizeofStringBuffer(), sizeofString("bravo")),
- Allocate(sizeofStringBuffer()),
- Reallocate(sizeofStringBuffer(), sizeofString("charlie")),
- Allocate(sizeofStringBuffer()),
- Deallocate(sizeofString("bravo")),
- Deallocate(sizeofString("charlie")),
- Deallocate(sizeofStringBuffer()),
- Reallocate(sizeofPool(), sizeofObject(2) + sizeofObject(1)),
- });
- }
- SECTION("Repeated key with zero copy mode") { // issue #1697
- char input[] = "{a:{b:{c:1}},a:2}";
- DeserializationError err = deserializeJson(doc, input);
- REQUIRE(err == DeserializationError::Ok);
- REQUIRE(doc["a"] == 2);
- }
- SECTION("NUL in keys") {
- DeserializationError err =
- deserializeJson(doc, "{\"x\":0,\"x\\u0000a\":1,\"x\\u0000b\":2}");
- REQUIRE(err == DeserializationError::Ok);
- REQUIRE(doc.as<std::string>() ==
- "{\"x\":0,\"x\\u0000a\":1,\"x\\u0000b\":2}");
- }
- }
- SECTION("Should clear the JsonObject") {
- deserializeJson(doc, "{\"hello\":\"world\"}");
- spy.clearLog();
- deserializeJson(doc, "{}");
- REQUIRE(doc.is<JsonObject>());
- REQUIRE(doc.size() == 0);
- REQUIRE(spy.log() == AllocatorLog{
- Deallocate(sizeofObject(1)),
- Deallocate(sizeofString("hello")),
- Deallocate(sizeofString("world")),
- });
- }
- SECTION("Issue #1335") {
- std::string json("{\"a\":{},\"b\":{}}");
- deserializeJson(doc, json);
- CHECK(doc.as<std::string>() == json);
- }
- }
- TEST_CASE("deserialize JSON object under memory constraints") {
- TimebombAllocator timebomb(1024);
- JsonDocument doc(&timebomb);
- SECTION("empty object requires no allocation") {
- timebomb.setCountdown(0);
- char input[] = "{}";
- DeserializationError err = deserializeJson(doc, input);
- REQUIRE(err == DeserializationError::Ok);
- REQUIRE(doc.as<std::string>() == "{}");
- }
- SECTION("key allocation fails") {
- timebomb.setCountdown(0);
- char input[] = "{\"a\":1}";
- DeserializationError err = deserializeJson(doc, input);
- REQUIRE(err == DeserializationError::NoMemory);
- REQUIRE(doc.as<std::string>() == "{}");
- }
- SECTION("pool allocation fails") {
- timebomb.setCountdown(1);
- char input[] = "{\"a\":1}";
- DeserializationError err = deserializeJson(doc, input);
- REQUIRE(err == DeserializationError::NoMemory);
- REQUIRE(doc.as<std::string>() == "{}");
- }
- SECTION("string allocation fails") {
- timebomb.setCountdown(3);
- char input[] = "{\"alfa\":\"bravo\"}";
- DeserializationError err = deserializeJson(doc, input);
- REQUIRE(err == DeserializationError::NoMemory);
- REQUIRE(doc.as<std::string>() == "{\"alfa\":null}");
- }
- }
|