You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
363 lines
7.9 KiB
363 lines
7.9 KiB
// ArduinoJson - https://arduinojson.org
|
|
// Copyright © 2014-2024, Benoit BLANCHON
|
|
// MIT License
|
|
|
|
#define ARDUINOJSON_ENABLE_ARDUINO_STRING 1
|
|
#define ARDUINOJSON_ENABLE_PROGMEM 1
|
|
#include <ArduinoJson.h>
|
|
|
|
#include <catch.hpp>
|
|
|
|
#include "Allocators.hpp"
|
|
|
|
using ArduinoJson::detail::sizeofArray;
|
|
using ArduinoJson::detail::sizeofObject;
|
|
|
|
typedef ArduinoJson::detail::MemberProxy<JsonDocument&, const char*>
|
|
MemberProxy;
|
|
|
|
TEST_CASE("MemberProxy::add()") {
|
|
JsonDocument doc;
|
|
MemberProxy mp = doc["hello"];
|
|
|
|
SECTION("add(int)") {
|
|
mp.add(42);
|
|
|
|
REQUIRE(doc.as<std::string>() == "{\"hello\":[42]}");
|
|
}
|
|
|
|
SECTION("add(const char*)") {
|
|
mp.add("world");
|
|
|
|
REQUIRE(doc.as<std::string>() == "{\"hello\":[\"world\"]}");
|
|
}
|
|
}
|
|
|
|
TEST_CASE("MemberProxy::clear()") {
|
|
JsonDocument doc;
|
|
MemberProxy mp = doc["hello"];
|
|
|
|
SECTION("size goes back to zero") {
|
|
mp.add(42);
|
|
mp.clear();
|
|
|
|
REQUIRE(mp.size() == 0);
|
|
}
|
|
|
|
SECTION("isNull() return true") {
|
|
mp.add("hello");
|
|
mp.clear();
|
|
|
|
REQUIRE(mp.isNull() == true);
|
|
}
|
|
}
|
|
|
|
TEST_CASE("MemberProxy::operator==()") {
|
|
JsonDocument doc;
|
|
|
|
SECTION("1 vs 1") {
|
|
doc["a"] = 1;
|
|
doc["b"] = 1;
|
|
|
|
REQUIRE(doc["a"] <= doc["b"]);
|
|
REQUIRE(doc["a"] == doc["b"]);
|
|
REQUIRE(doc["a"] >= doc["b"]);
|
|
REQUIRE_FALSE(doc["a"] != doc["b"]);
|
|
REQUIRE_FALSE(doc["a"] < doc["b"]);
|
|
REQUIRE_FALSE(doc["a"] > doc["b"]);
|
|
}
|
|
|
|
SECTION("1 vs 2") {
|
|
doc["a"] = 1;
|
|
doc["b"] = 2;
|
|
|
|
REQUIRE(doc["a"] != doc["b"]);
|
|
REQUIRE(doc["a"] < doc["b"]);
|
|
REQUIRE(doc["a"] <= doc["b"]);
|
|
REQUIRE_FALSE(doc["a"] == doc["b"]);
|
|
REQUIRE_FALSE(doc["a"] > doc["b"]);
|
|
REQUIRE_FALSE(doc["a"] >= doc["b"]);
|
|
}
|
|
|
|
SECTION("'abc' vs 'bcd'") {
|
|
doc["a"] = "abc";
|
|
doc["b"] = "bcd";
|
|
|
|
REQUIRE(doc["a"] != doc["b"]);
|
|
REQUIRE(doc["a"] < doc["b"]);
|
|
REQUIRE(doc["a"] <= doc["b"]);
|
|
REQUIRE_FALSE(doc["a"] == doc["b"]);
|
|
REQUIRE_FALSE(doc["a"] > doc["b"]);
|
|
REQUIRE_FALSE(doc["a"] >= doc["b"]);
|
|
}
|
|
}
|
|
|
|
TEST_CASE("MemberProxy::containsKey()") {
|
|
JsonDocument doc;
|
|
MemberProxy mp = doc["hello"];
|
|
|
|
SECTION("containsKey(const char*)") {
|
|
mp["key"] = "value";
|
|
|
|
REQUIRE(mp.containsKey("key") == true);
|
|
REQUIRE(mp.containsKey("key") == true);
|
|
}
|
|
|
|
SECTION("containsKey(std::string)") {
|
|
mp["key"] = "value";
|
|
|
|
REQUIRE(mp.containsKey(std::string("key")) == true);
|
|
REQUIRE(mp.containsKey(std::string("key")) == true);
|
|
}
|
|
}
|
|
|
|
TEST_CASE("MemberProxy::operator|()") {
|
|
JsonDocument doc;
|
|
|
|
SECTION("const char*") {
|
|
doc["a"] = "hello";
|
|
|
|
REQUIRE((doc["a"] | "world") == std::string("hello"));
|
|
REQUIRE((doc["b"] | "world") == std::string("world"));
|
|
}
|
|
|
|
SECTION("Issue #1411") {
|
|
doc["sensor"] = "gps";
|
|
|
|
const char* test = "test"; // <- the literal must be captured in a variable
|
|
// to trigger the bug
|
|
const char* sensor = doc["sensor"] | test; // "gps"
|
|
|
|
REQUIRE(sensor == std::string("gps"));
|
|
}
|
|
|
|
SECTION("Issue #1415") {
|
|
JsonObject object = doc.to<JsonObject>();
|
|
object["hello"] = "world";
|
|
|
|
JsonDocument emptyDoc;
|
|
JsonObject anotherObject = object["hello"] | emptyDoc.to<JsonObject>();
|
|
|
|
REQUIRE(anotherObject.isNull() == false);
|
|
REQUIRE(anotherObject.size() == 0);
|
|
}
|
|
}
|
|
|
|
TEST_CASE("MemberProxy::remove()") {
|
|
JsonDocument doc;
|
|
MemberProxy mp = doc["hello"];
|
|
|
|
SECTION("remove(int)") {
|
|
mp.add(1);
|
|
mp.add(2);
|
|
mp.add(3);
|
|
|
|
mp.remove(1);
|
|
|
|
REQUIRE(mp.as<std::string>() == "[1,3]");
|
|
}
|
|
|
|
SECTION("remove(const char *)") {
|
|
mp["a"] = 1;
|
|
mp["b"] = 2;
|
|
|
|
mp.remove("a");
|
|
|
|
REQUIRE(mp.as<std::string>() == "{\"b\":2}");
|
|
}
|
|
|
|
SECTION("remove(std::string)") {
|
|
mp["a"] = 1;
|
|
mp["b"] = 2;
|
|
|
|
mp.remove(std::string("b"));
|
|
|
|
REQUIRE(mp.as<std::string>() == "{\"a\":1}");
|
|
}
|
|
|
|
#ifdef HAS_VARIABLE_LENGTH_ARRAY
|
|
SECTION("remove(vla)") {
|
|
mp["a"] = 1;
|
|
mp["b"] = 2;
|
|
|
|
size_t i = 4;
|
|
char vla[i];
|
|
strcpy(vla, "b");
|
|
mp.remove(vla);
|
|
|
|
REQUIRE(mp.as<std::string>() == "{\"a\":1}");
|
|
}
|
|
#endif
|
|
}
|
|
|
|
TEST_CASE("MemberProxy::set()") {
|
|
JsonDocument doc;
|
|
MemberProxy mp = doc["hello"];
|
|
|
|
SECTION("set(int)") {
|
|
mp.set(42);
|
|
|
|
REQUIRE(doc.as<std::string>() == "{\"hello\":42}");
|
|
}
|
|
|
|
SECTION("set(const char*)") {
|
|
mp.set("world");
|
|
|
|
REQUIRE(doc.as<std::string>() == "{\"hello\":\"world\"}");
|
|
}
|
|
|
|
SECTION("set(char[])") { // issue #1191
|
|
char s[] = "world";
|
|
mp.set(s);
|
|
strcpy(s, "!!!!!");
|
|
|
|
REQUIRE(doc.as<std::string>() == "{\"hello\":\"world\"}");
|
|
}
|
|
}
|
|
|
|
TEST_CASE("MemberProxy::size()") {
|
|
JsonDocument doc;
|
|
MemberProxy mp = doc["hello"];
|
|
|
|
SECTION("returns 0") {
|
|
REQUIRE(mp.size() == 0);
|
|
}
|
|
|
|
SECTION("as an array, return 2") {
|
|
mp.add(1);
|
|
mp.add(2);
|
|
|
|
REQUIRE(mp.size() == 2);
|
|
}
|
|
|
|
SECTION("as an object, return 2") {
|
|
mp["a"] = 1;
|
|
mp["b"] = 2;
|
|
|
|
REQUIRE(mp.size() == 2);
|
|
}
|
|
}
|
|
|
|
TEST_CASE("MemberProxy::operator[]") {
|
|
JsonDocument doc;
|
|
MemberProxy mp = doc["hello"];
|
|
|
|
SECTION("set member") {
|
|
mp["world"] = 42;
|
|
|
|
REQUIRE(doc.as<std::string>() == "{\"hello\":{\"world\":42}}");
|
|
}
|
|
|
|
SECTION("set element") {
|
|
mp[2] = 42;
|
|
|
|
REQUIRE(doc.as<std::string>() == "{\"hello\":[null,null,42]}");
|
|
}
|
|
}
|
|
|
|
TEST_CASE("MemberProxy cast to JsonVariantConst") {
|
|
JsonDocument doc;
|
|
doc["hello"] = "world";
|
|
|
|
const MemberProxy mp = doc["hello"];
|
|
|
|
JsonVariantConst var = mp;
|
|
|
|
CHECK(var.as<std::string>() == "world");
|
|
}
|
|
|
|
TEST_CASE("MemberProxy cast to JsonVariant") {
|
|
JsonDocument doc;
|
|
doc["hello"] = "world";
|
|
|
|
MemberProxy mp = doc["hello"];
|
|
|
|
JsonVariant var = mp;
|
|
|
|
CHECK(var.as<std::string>() == "world");
|
|
|
|
var.set("toto");
|
|
|
|
CHECK(doc.as<std::string>() == "{\"hello\":\"toto\"}");
|
|
}
|
|
|
|
TEST_CASE("Deduplicate keys") {
|
|
SpyingAllocator spy;
|
|
JsonDocument doc(&spy);
|
|
|
|
SECTION("std::string") {
|
|
doc[0][std::string("example")] = 1;
|
|
doc[1][std::string("example")] = 2;
|
|
|
|
const char* key1 = doc[0].as<JsonObject>().begin()->key().c_str();
|
|
const char* key2 = doc[1].as<JsonObject>().begin()->key().c_str();
|
|
CHECK(key1 == key2);
|
|
|
|
REQUIRE(spy.log() == AllocatorLog{
|
|
Allocate(sizeofPool()),
|
|
Allocate(sizeofString("example")),
|
|
});
|
|
}
|
|
|
|
SECTION("char*") {
|
|
char key[] = "example";
|
|
doc[0][key] = 1;
|
|
doc[1][key] = 2;
|
|
|
|
const char* key1 = doc[0].as<JsonObject>().begin()->key().c_str();
|
|
const char* key2 = doc[1].as<JsonObject>().begin()->key().c_str();
|
|
CHECK(key1 == key2);
|
|
|
|
REQUIRE(spy.log() == AllocatorLog{
|
|
Allocate(sizeofPool()),
|
|
Allocate(sizeofString("example")),
|
|
});
|
|
}
|
|
|
|
SECTION("Arduino String") {
|
|
doc[0][String("example")] = 1;
|
|
doc[1][String("example")] = 2;
|
|
|
|
const char* key1 = doc[0].as<JsonObject>().begin()->key().c_str();
|
|
const char* key2 = doc[1].as<JsonObject>().begin()->key().c_str();
|
|
CHECK(key1 == key2);
|
|
|
|
REQUIRE(spy.log() == AllocatorLog{
|
|
Allocate(sizeofPool()),
|
|
Allocate(sizeofString("example")),
|
|
});
|
|
}
|
|
|
|
SECTION("Flash string") {
|
|
doc[0][F("example")] = 1;
|
|
doc[1][F("example")] = 2;
|
|
|
|
const char* key1 = doc[0].as<JsonObject>().begin()->key().c_str();
|
|
const char* key2 = doc[1].as<JsonObject>().begin()->key().c_str();
|
|
CHECK(key1 == key2);
|
|
|
|
REQUIRE(spy.log() == AllocatorLog{
|
|
Allocate(sizeofPool()),
|
|
Allocate(sizeofString("example")),
|
|
});
|
|
}
|
|
}
|
|
|
|
TEST_CASE("MemberProxy under memory constraints") {
|
|
KillswitchAllocator killswitch;
|
|
SpyingAllocator spy(&killswitch);
|
|
JsonDocument doc(&spy);
|
|
|
|
SECTION("key allocation fails") {
|
|
killswitch.on();
|
|
|
|
doc[std::string("hello")] = "world";
|
|
|
|
REQUIRE(doc.is<JsonObject>());
|
|
REQUIRE(doc.size() == 0);
|
|
REQUIRE(doc.overflowed() == true);
|
|
REQUIRE(spy.log() == AllocatorLog{
|
|
AllocateFail(sizeofString("hello")),
|
|
});
|
|
}
|
|
}
|
|
|