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()), |
|
}); |
|
} |
|
}
|
|
|