// ArduinoJson - https://arduinojson.org // Copyright © 2014-2025, Benoit BLANCHON // MIT License #include #include #include "Allocators.hpp" #include "Literals.hpp" using ArduinoJson::detail::sizeofArray; TEST_CASE("JsonArray::add(T)") { SpyingAllocator spy; JsonDocument doc(&spy); JsonArray array = doc.to(); SECTION("int") { array.add(123); REQUIRE(123 == array[0].as()); REQUIRE(array[0].is()); REQUIRE(array[0].is()); REQUIRE(spy.log() == AllocatorLog{ Allocate(sizeofPool()), }); } SECTION("double") { array.add(123.45); REQUIRE(123.45 == array[0].as()); REQUIRE(array[0].is()); REQUIRE_FALSE(array[0].is()); REQUIRE(spy.log() == AllocatorLog{ Allocate(sizeofPool()), }); } SECTION("bool") { array.add(true); REQUIRE(array[0].as() == true); REQUIRE(array[0].is()); REQUIRE_FALSE(array[0].is()); REQUIRE(spy.log() == AllocatorLog{ Allocate(sizeofPool()), }); } SECTION("string literal") { array.add("hello"); REQUIRE(array[0].as() == "hello"); REQUIRE(array[0].is()); REQUIRE(array[0].is() == false); REQUIRE(spy.log() == AllocatorLog{ Allocate(sizeofPool()), Allocate(sizeofStaticStringPool()), }); } SECTION("std::string") { array.add("hello"_s); REQUIRE(array[0].as() == "hello"); REQUIRE(array[0].is() == true); REQUIRE(array[0].is() == false); REQUIRE(spy.log() == AllocatorLog{ Allocate(sizeofPool()), Allocate(sizeofString("hello")), }); } SECTION("const char*") { const char* str = "hello"; array.add(str); REQUIRE(array[0].as() == "hello"); REQUIRE(array[0].as() != str); REQUIRE(array[0].is() == true); REQUIRE(array[0].is() == false); REQUIRE(spy.log() == AllocatorLog{ Allocate(sizeofPool()), Allocate(sizeofString("hello")), }); } SECTION("serialized(const char*)") { array.add(serialized("{}")); REQUIRE(doc.as() == "[{}]"); REQUIRE(spy.log() == AllocatorLog{ Allocate(sizeofPool()), Allocate(sizeofString("{}")), }); } SECTION("serialized(char*)") { array.add(serialized(const_cast("{}"))); REQUIRE(doc.as() == "[{}]"); REQUIRE(spy.log() == AllocatorLog{ Allocate(sizeofPool()), Allocate(sizeofString("{}")), }); } SECTION("serialized(std::string)") { array.add(serialized("{}"_s)); REQUIRE(doc.as() == "[{}]"); REQUIRE(spy.log() == AllocatorLog{ Allocate(sizeofPool()), Allocate(sizeofString("{}")), }); } SECTION("serialized(std::string)") { array.add(serialized("\0XX"_s)); REQUIRE(doc.as() == "[\0XX]"_s); REQUIRE(spy.log() == AllocatorLog{ Allocate(sizeofPool()), Allocate(sizeofString(" XX")), }); } #ifdef HAS_VARIABLE_LENGTH_ARRAY SECTION("vla") { size_t i = 16; char vla[i]; strcpy(vla, "world"); array.add(vla); strcpy(vla, "hello"); REQUIRE(array[0] == "world"_s); REQUIRE(spy.log() == AllocatorLog{ Allocate(sizeofPool()), Allocate(sizeofString("hello")), }); } #endif SECTION("nested array") { JsonDocument doc2; JsonArray arr = doc2.to(); array.add(arr); REQUIRE(arr == array[0].as()); REQUIRE(array[0].is()); REQUIRE_FALSE(array[0].is()); } SECTION("nested object") { JsonDocument doc2; JsonObject obj = doc2.to(); array.add(obj); REQUIRE(obj == array[0].as()); REQUIRE(array[0].is()); REQUIRE_FALSE(array[0].is()); } SECTION("array subscript") { const char* str = "hello"; JsonDocument doc2; JsonArray arr = doc2.to(); arr.add(str); array.add(arr[0]); REQUIRE(str == array[0]); } SECTION("object subscript") { const char* str = "hello"; JsonDocument doc2; JsonObject obj = doc2.to(); obj["x"] = str; array.add(obj["x"]); REQUIRE(str == array[0]); } } TEST_CASE("JsonArray::add()") { JsonDocument doc; JsonArray array = doc.to(); SECTION("add()") { JsonArray nestedArray = array.add(); nestedArray.add(1); nestedArray.add(2); REQUIRE(doc.as() == "[[1,2]]"); } SECTION("add()") { JsonObject nestedObject = array.add(); nestedObject["a"] = 1; nestedObject["b"] = 2; REQUIRE(doc.as() == "[{\"a\":1,\"b\":2}]"); } SECTION("add()") { JsonVariant nestedVariant = array.add(); nestedVariant.set(42); REQUIRE(doc.as() == "[42]"); } } TEST_CASE("JsonObject::add(JsonObject) ") { JsonDocument doc1; doc1["key1"_s] = "value1"_s; TimebombAllocator allocator(10); SpyingAllocator spy(&allocator); JsonDocument doc2(&spy); JsonArray array = doc2.to(); SECTION("success") { bool result = array.add(doc1.as()); REQUIRE(result == true); REQUIRE(doc2.as() == "[{\"key1\":\"value1\"}]"); REQUIRE(spy.log() == AllocatorLog{ Allocate(sizeofPool()), Allocate(sizeofString("key1")), Allocate(sizeofString("value1")), }); } SECTION("partial failure") { // issue #2081 allocator.setCountdown(2); bool result = array.add(doc1.as()); REQUIRE(result == false); REQUIRE(doc2.as() == "[]"); REQUIRE(spy.log() == AllocatorLog{ Allocate(sizeofPool()), Allocate(sizeofString("key1")), AllocateFail(sizeofString("value1")), Deallocate(sizeofString("key1")), }); } SECTION("complete failure") { allocator.setCountdown(0); bool result = array.add(doc1.as()); REQUIRE(result == false); REQUIRE(doc2.as() == "[]"); REQUIRE(spy.log() == AllocatorLog{ AllocateFail(sizeofPool()), }); } }