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.
192 lines
5.5 KiB
192 lines
5.5 KiB
// ArduinoJson - https://arduinojson.org
|
|
// Copyright © 2014-2024, Benoit BLANCHON
|
|
// MIT License
|
|
|
|
#include <Arduino.h>
|
|
#include <catch.hpp>
|
|
|
|
#define ARDUINOJSON_ENABLE_ARDUINO_STREAM 1
|
|
#include <ArduinoJson.h>
|
|
|
|
#include "Allocators.hpp"
|
|
|
|
using ArduinoJson::detail::sizeofArray;
|
|
|
|
struct PrintOneCharacterAtATime {
|
|
static size_t printStringTo(const std::string& s, Print& p) {
|
|
size_t result = 0;
|
|
for (std::string::const_iterator it = s.begin(); it != s.end(); ++it) {
|
|
size_t n = p.write(uint8_t(*it));
|
|
if (n == 0)
|
|
break;
|
|
result += n;
|
|
}
|
|
return result;
|
|
}
|
|
};
|
|
|
|
struct PrintAllAtOnce {
|
|
static size_t printStringTo(const std::string& s, Print& p) {
|
|
return p.write(s.data(), s.size());
|
|
}
|
|
};
|
|
|
|
template <typename PrintPolicy>
|
|
struct PrintableString : public Printable {
|
|
PrintableString(const char* s) : str_(s), total_(0) {}
|
|
|
|
virtual size_t printTo(Print& p) const {
|
|
size_t result = PrintPolicy::printStringTo(str_, p);
|
|
total_ += result;
|
|
return result;
|
|
}
|
|
|
|
size_t totalBytesWritten() const {
|
|
return total_;
|
|
}
|
|
|
|
private:
|
|
std::string str_;
|
|
mutable size_t total_;
|
|
};
|
|
|
|
TEST_CASE("Printable") {
|
|
SECTION("Doesn't overflow") {
|
|
SpyingAllocator spy;
|
|
JsonDocument doc(&spy);
|
|
const char* value = "example";
|
|
|
|
doc.set(666); // to make sure we override the value
|
|
|
|
SECTION("Via Print::write(char)") {
|
|
PrintableString<PrintOneCharacterAtATime> printable(value);
|
|
CHECK(doc.set(printable) == true);
|
|
CHECK(doc.as<std::string>() == value);
|
|
CHECK(printable.totalBytesWritten() == 7);
|
|
CHECK(doc.overflowed() == false);
|
|
CHECK(spy.log() ==
|
|
AllocatorLog{
|
|
Allocate(sizeofStringBuffer()),
|
|
Reallocate(sizeofStringBuffer(), sizeofString("example")),
|
|
});
|
|
}
|
|
|
|
SECTION("Via Print::write(const char* size_t)") {
|
|
PrintableString<PrintAllAtOnce> printable(value);
|
|
CHECK(doc.set(printable) == true);
|
|
CHECK(doc.as<std::string>() == value);
|
|
CHECK(printable.totalBytesWritten() == 7);
|
|
CHECK(doc.overflowed() == false);
|
|
CHECK(spy.log() ==
|
|
AllocatorLog{
|
|
Allocate(sizeofStringBuffer()),
|
|
Reallocate(sizeofStringBuffer(), sizeofString("example")),
|
|
});
|
|
}
|
|
}
|
|
|
|
SECTION("First allocation fails") {
|
|
SpyingAllocator spy(FailingAllocator::instance());
|
|
JsonDocument doc(&spy);
|
|
const char* value = "hello world";
|
|
|
|
doc.set(666); // to make sure we override the value
|
|
|
|
SECTION("Via Print::write(char)") {
|
|
PrintableString<PrintOneCharacterAtATime> printable(value);
|
|
|
|
bool success = doc.set(printable);
|
|
|
|
CHECK(success == false);
|
|
CHECK(doc.isNull());
|
|
CHECK(printable.totalBytesWritten() == 0);
|
|
CHECK(doc.overflowed() == true);
|
|
CHECK(spy.log() == AllocatorLog{
|
|
AllocateFail(sizeofStringBuffer()),
|
|
});
|
|
}
|
|
|
|
SECTION("Via Print::write(const char*, size_t)") {
|
|
PrintableString<PrintAllAtOnce> printable(value);
|
|
|
|
bool success = doc.set(printable);
|
|
|
|
CHECK(success == false);
|
|
CHECK(doc.isNull());
|
|
CHECK(printable.totalBytesWritten() == 0);
|
|
CHECK(doc.overflowed() == true);
|
|
CHECK(spy.log() == AllocatorLog{
|
|
AllocateFail(sizeofStringBuffer()),
|
|
});
|
|
}
|
|
}
|
|
|
|
SECTION("Reallocation fails") {
|
|
TimebombAllocator timebomb(1);
|
|
SpyingAllocator spy(&timebomb);
|
|
JsonDocument doc(&spy);
|
|
const char* value = "Lorem ipsum dolor sit amet, cons"; // > 31 chars
|
|
|
|
doc.set(666); // to make sure we override the value
|
|
|
|
SECTION("Via Print::write(char)") {
|
|
PrintableString<PrintOneCharacterAtATime> printable(value);
|
|
|
|
bool success = doc.set(printable);
|
|
|
|
CHECK(success == false);
|
|
CHECK(doc.isNull());
|
|
CHECK(printable.totalBytesWritten() == 31);
|
|
CHECK(doc.overflowed() == true);
|
|
CHECK(spy.log() ==
|
|
AllocatorLog{
|
|
Allocate(sizeofStringBuffer()),
|
|
ReallocateFail(sizeofStringBuffer(), sizeofStringBuffer(2)),
|
|
Deallocate(sizeofStringBuffer()),
|
|
});
|
|
}
|
|
|
|
SECTION("Via Print::write(const char*, size_t)") {
|
|
PrintableString<PrintAllAtOnce> printable(value);
|
|
|
|
bool success = doc.set(printable);
|
|
|
|
CHECK(success == false);
|
|
CHECK(doc.isNull());
|
|
CHECK(printable.totalBytesWritten() == 31);
|
|
CHECK(doc.overflowed() == true);
|
|
CHECK(spy.log() ==
|
|
AllocatorLog{
|
|
Allocate(sizeofStringBuffer()),
|
|
ReallocateFail(sizeofStringBuffer(), sizeofStringBuffer(2)),
|
|
Deallocate(sizeofStringBuffer()),
|
|
});
|
|
}
|
|
}
|
|
|
|
SECTION("Null variant") {
|
|
JsonVariant var;
|
|
PrintableString<PrintOneCharacterAtATime> printable = "Hello World!";
|
|
CHECK(var.set(printable) == false);
|
|
CHECK(var.isNull());
|
|
CHECK(printable.totalBytesWritten() == 0);
|
|
}
|
|
|
|
SECTION("String deduplication") {
|
|
SpyingAllocator spy;
|
|
JsonDocument doc(&spy);
|
|
doc.add(PrintableString<PrintOneCharacterAtATime>("Hello World!"));
|
|
doc.add(PrintableString<PrintAllAtOnce>("Hello World!"));
|
|
REQUIRE(doc.size() == 2);
|
|
CHECK(doc[0] == "Hello World!");
|
|
CHECK(doc[1] == "Hello World!");
|
|
CHECK(spy.log() ==
|
|
AllocatorLog{
|
|
Allocate(sizeofPool()),
|
|
Allocate(sizeofStringBuffer()),
|
|
Reallocate(sizeofStringBuffer(), sizeofString("Hello World!")),
|
|
Allocate(sizeofStringBuffer()),
|
|
Deallocate(sizeofStringBuffer()),
|
|
});
|
|
}
|
|
}
|
|
|