// ArduinoJson - arduinojson.org // Copyright Benoit Blanchon 2014-2020 // MIT License #include #include template void checkEquals(T a, T b) { DynamicJsonDocument doc(4096); JsonVariant variant = doc.to(); variant.set(a); REQUIRE(b == variant); REQUIRE(variant == b); REQUIRE(b <= variant); REQUIRE(variant <= b); REQUIRE(b >= variant); REQUIRE(variant >= b); REQUIRE_FALSE(b != variant); REQUIRE_FALSE(variant != b); REQUIRE_FALSE(b > variant); REQUIRE_FALSE(variant > b); REQUIRE_FALSE(b < variant); REQUIRE_FALSE(variant < b); } template void checkGreater(T a, T b) { DynamicJsonDocument doc(4096); JsonVariant variant = doc.to(); variant.set(a); REQUIRE(variant > b); REQUIRE(b < variant); REQUIRE(variant != b); REQUIRE(b != variant); REQUIRE_FALSE(variant < b); REQUIRE_FALSE(b > variant); REQUIRE_FALSE(variant == b); REQUIRE_FALSE(b == variant); } template void checkLower(T a, T b) { DynamicJsonDocument doc(4096); JsonVariant variant = doc.to(); variant.set(a); REQUIRE(variant < b); REQUIRE(b > variant); REQUIRE(variant != b); REQUIRE(b != variant); REQUIRE_FALSE(variant > b); REQUIRE_FALSE(b < variant); REQUIRE_FALSE(variant == b); REQUIRE_FALSE(b == variant); } template void checkComparisons(T low, T mid, T high) { checkEquals(mid, mid); checkGreater(mid, low); checkLower(mid, high); } TEST_CASE("JsonVariant comparisons") { static const char* null = 0; SECTION("Double") { checkComparisons(123.44, 123.45, 123.46); } SECTION("Float") { checkComparisons(123.44f, 123.45f, 123.46f); } SECTION("SChar") { checkComparisons(122, 123, 124); } SECTION("SInt") { checkComparisons(122, 123, 124); } SECTION("SLong") { checkComparisons(122L, 123L, 124L); } SECTION("SShort") { checkComparisons(122, 123, 124); } SECTION("UChar") { checkComparisons(122, 123, 124); } SECTION("UInt") { checkComparisons(122, 123, 124); } SECTION("ULong") { checkComparisons(122L, 123L, 124L); } SECTION("UShort") { checkComparisons(122, 123, 124); } SECTION("null") { DynamicJsonDocument doc(4096); JsonVariant variant = doc.to(); variant.set(null); REQUIRE(variant == variant); REQUIRE_FALSE(variant != variant); REQUIRE(variant == null); REQUIRE_FALSE(variant != null); REQUIRE(variant != "null"); REQUIRE_FALSE(variant == "null"); } SECTION("StringLiteral") { DynamicJsonDocument doc(4096); deserializeJson(doc, "\"hello\""); JsonVariant variant = doc.as(); REQUIRE(variant == variant); REQUIRE_FALSE(variant != variant); REQUIRE(variant == "hello"); REQUIRE_FALSE(variant != "hello"); REQUIRE(variant != "world"); REQUIRE_FALSE(variant == "world"); REQUIRE(variant != null); REQUIRE_FALSE(variant == null); REQUIRE("hello" == variant); REQUIRE_FALSE("hello" != variant); REQUIRE("world" != variant); REQUIRE_FALSE("world" == variant); REQUIRE(null != variant); REQUIRE_FALSE(null == variant); } SECTION("String") { DynamicJsonDocument doc(4096); JsonVariant variant = doc.to(); variant.set("hello"); REQUIRE(variant == variant); REQUIRE_FALSE(variant != variant); REQUIRE(variant == std::string("hello")); REQUIRE_FALSE(variant != std::string("hello")); REQUIRE(variant != std::string("world")); REQUIRE_FALSE(variant == std::string("world")); REQUIRE(variant != null); REQUIRE_FALSE(variant == null); REQUIRE(std::string("hello") == variant); REQUIRE_FALSE(std::string("hello") != variant); REQUIRE(std::string("world") != variant); REQUIRE_FALSE(std::string("world") == variant); REQUIRE(null != variant); REQUIRE_FALSE(null == variant); } #ifdef HAS_VARIABLE_LENGTH_ARRAY SECTION("VLA equals") { int i = 16; char vla[i]; strcpy(vla, "hello"); DynamicJsonDocument doc(4096); JsonVariant variant = doc.to(); variant.set("hello"); REQUIRE((vla == variant)); REQUIRE((variant == vla)); REQUIRE_FALSE((vla != variant)); REQUIRE_FALSE((variant != vla)); } SECTION("VLA differs") { int i = 16; char vla[i]; strcpy(vla, "hello"); DynamicJsonDocument doc(4096); JsonVariant variant = doc.to(); variant.set("world"); REQUIRE((vla != variant)); REQUIRE((variant != vla)); REQUIRE_FALSE((vla == variant)); REQUIRE_FALSE((variant == vla)); } #endif DynamicJsonDocument doc1(4096), doc2(4096), doc3(4096); JsonVariant variant1 = doc1.to(); JsonVariant variant2 = doc2.to(); JsonVariant variant3 = doc3.to(); SECTION("Variants containing integers") { variant1.set(42); variant2.set(42); variant3.set(666); REQUIRE(variant1 == variant2); REQUIRE_FALSE(variant1 != variant2); REQUIRE(variant1 != variant3); REQUIRE_FALSE(variant1 == variant3); } SECTION("Variants containing linked strings") { // create two identical strings at different addresses char hello1[] = "hello"; char hello2[] = "hello"; REQUIRE(hello1 != hello2); variant1.set(hello1); variant2.set(hello2); variant3.set("world"); REQUIRE(variant1 == variant2); REQUIRE_FALSE(variant1 != variant2); REQUIRE(variant1 != variant3); REQUIRE_FALSE(variant1 == variant3); } SECTION("Variants containing owned strings") { variant1.set(std::string("hello")); variant2.set(std::string("hello")); variant3.set(std::string("world")); REQUIRE(variant1 == variant2); REQUIRE_FALSE(variant1 != variant2); REQUIRE(variant1 != variant3); REQUIRE_FALSE(variant1 == variant3); } SECTION("Variants containing linked raws") { // create two identical strings at different addresses char hello1[] = "hello"; char hello2[] = "hello"; REQUIRE(hello1 != hello2); variant1.set(serialized(hello1)); variant2.set(serialized(hello2)); variant3.set(serialized("world")); REQUIRE(variant1 == variant2); REQUIRE_FALSE(variant1 != variant2); REQUIRE(variant1 != variant3); REQUIRE_FALSE(variant1 == variant3); } SECTION("Variants containing owned raws") { variant1.set(serialized(std::string("hello"))); variant2.set(serialized(std::string("hello"))); variant3.set(serialized(std::string("world"))); REQUIRE(variant1 == variant2); REQUIRE_FALSE(variant1 != variant2); REQUIRE(variant1 != variant3); REQUIRE_FALSE(variant1 == variant3); } SECTION("Variants containing mixed strings (issue #1051)") { variant1.set("hello"); variant2.set(std::string("hello")); REQUIRE(variant1 == variant2); REQUIRE_FALSE(variant1 != variant2); REQUIRE(variant2 == variant1); REQUIRE_FALSE(variant2 != variant1); } SECTION("Variants containing double") { variant1.set(42.0); variant2.set(42.0); variant3.set(666.0); REQUIRE(variant1 == variant2); REQUIRE_FALSE(variant1 != variant2); REQUIRE(variant1 != variant3); REQUIRE_FALSE(variant1 == variant3); } SECTION("BoolInVariant") { variant1.set(true); variant2.set(true); variant3.set(false); REQUIRE(variant1 == variant2); REQUIRE_FALSE(variant1 != variant2); REQUIRE(variant1 != variant3); REQUIRE_FALSE(variant1 == variant3); } SECTION("ArrayInVariant") { JsonArray array1 = variant1.to(); JsonArray array2 = variant2.to(); array1.add(42); array2.add(42); REQUIRE(variant1 == variant2); REQUIRE_FALSE(variant1 != variant2); REQUIRE(variant1 != variant3); REQUIRE_FALSE(variant1 == variant3); } SECTION("ObjectInVariant") { JsonObject obj1 = variant1.to(); JsonObject obj2 = variant2.to(); obj1["hello"] = "world"; obj2["hello"] = "world"; REQUIRE(variant1 == variant2); REQUIRE_FALSE(variant1 != variant2); REQUIRE(variant1 != variant3); REQUIRE_FALSE(variant1 == variant3); } } class VariantComparisionFixture { private: StaticJsonDocument<256> doc; JsonVariant variant; public: VariantComparisionFixture() : variant(doc.to()) {} protected: template void setValue(const T& value) { variant.set(value); } template void assertEqualsTo(const T& value) { REQUIRE(variant == value); REQUIRE(value == variant); REQUIRE_FALSE(variant != value); REQUIRE_FALSE(value != variant); } template void assertDiffersFrom(const T& value) { REQUIRE(variant != value); REQUIRE(value != variant); REQUIRE_FALSE(variant == value); REQUIRE_FALSE(value == variant); } template void assertGreaterThan(const T& value) { REQUIRE((variant > value)); REQUIRE((variant >= value)); REQUIRE(value < variant); REQUIRE(value <= variant); REQUIRE_FALSE((variant < value)); REQUIRE_FALSE((variant <= value)); REQUIRE_FALSE(value > variant); REQUIRE_FALSE(value >= variant); } template void assertLowerThan(const T& value) { REQUIRE(variant < value); REQUIRE(variant <= value); REQUIRE(value > variant); REQUIRE(value >= variant); REQUIRE_FALSE(variant > value); REQUIRE_FALSE(variant >= value); REQUIRE_FALSE(value < variant); REQUIRE_FALSE(value <= variant); } }; TEST_CASE_METHOD(VariantComparisionFixture, "Compare variant with another type") { SECTION("null") { assertDiffersFrom(3); assertDiffersFrom("world"); } SECTION("string") { setValue("hello"); assertEqualsTo("hello"); assertDiffersFrom(3); assertDiffersFrom("world"); assertGreaterThan("helln"); assertLowerThan("hellp"); } SECTION("positive integer") { setValue(42); assertEqualsTo(42); assertDiffersFrom(43); assertGreaterThan(41); assertLowerThan(43); assertDiffersFrom("world"); } SECTION("negative integer") { setValue(-42); assertEqualsTo(-42); assertDiffersFrom(42); assertGreaterThan(-43); assertLowerThan(-41); assertDiffersFrom("world"); } SECTION("double") { setValue(42.0); assertEqualsTo(42.0); assertDiffersFrom(42.1); assertGreaterThan(41.0); assertLowerThan(43.0); assertDiffersFrom("42.0"); } SECTION("true") { setValue(true); assertEqualsTo(true); assertDiffersFrom(false); assertDiffersFrom(1); assertDiffersFrom("true"); assertDiffersFrom(1.0); assertGreaterThan(false); } }