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.
158 lines
4.1 KiB
158 lines
4.1 KiB
4 years ago
|
// ArduinoJson - https://arduinojson.org
|
||
|
// Copyright Benoit Blanchon 2014-2021
|
||
|
// MIT License
|
||
|
|
||
|
#include <ArduinoJson.h>
|
||
|
#include <stdlib.h> // malloc, free
|
||
|
#include <catch.hpp>
|
||
|
#include <sstream>
|
||
|
|
||
|
using ARDUINOJSON_NAMESPACE::addPadding;
|
||
|
using ARDUINOJSON_NAMESPACE::move;
|
||
|
|
||
|
class SpyingAllocator {
|
||
|
public:
|
||
|
SpyingAllocator(const SpyingAllocator& src) : _log(src._log) {}
|
||
|
SpyingAllocator(std::ostream& log) : _log(log) {}
|
||
|
|
||
|
void* allocate(size_t n) {
|
||
|
_log << "A" << n;
|
||
|
return malloc(n);
|
||
|
}
|
||
|
void deallocate(void* p) {
|
||
|
_log << "F";
|
||
|
free(p);
|
||
|
}
|
||
|
|
||
|
private:
|
||
|
SpyingAllocator& operator=(const SpyingAllocator& src);
|
||
|
|
||
|
std::ostream& _log;
|
||
|
};
|
||
|
|
||
|
class ControllableAllocator {
|
||
|
public:
|
||
|
ControllableAllocator() : _enabled(true) {}
|
||
|
|
||
|
void* allocate(size_t n) {
|
||
|
return _enabled ? malloc(n) : 0;
|
||
|
}
|
||
|
|
||
|
void deallocate(void* p) {
|
||
|
free(p);
|
||
|
}
|
||
|
|
||
|
void disable() {
|
||
|
_enabled = false;
|
||
|
}
|
||
|
|
||
|
private:
|
||
|
bool _enabled;
|
||
|
};
|
||
|
|
||
|
TEST_CASE("BasicJsonDocument") {
|
||
|
std::stringstream log;
|
||
|
|
||
|
SECTION("Construct/Destruct") {
|
||
|
{ BasicJsonDocument<SpyingAllocator> doc(4096, log); }
|
||
|
REQUIRE(log.str() == "A4096F");
|
||
|
}
|
||
|
|
||
|
SECTION("Copy construct") {
|
||
|
{
|
||
|
BasicJsonDocument<SpyingAllocator> doc1(4096, log);
|
||
|
doc1.set(std::string("The size of this string is 32!!"));
|
||
|
|
||
|
BasicJsonDocument<SpyingAllocator> doc2(doc1);
|
||
|
|
||
|
REQUIRE(doc1.as<std::string>() == "The size of this string is 32!!");
|
||
|
REQUIRE(doc2.as<std::string>() == "The size of this string is 32!!");
|
||
|
REQUIRE(doc2.capacity() == 4096);
|
||
|
}
|
||
|
REQUIRE(log.str() == "A4096A4096FF");
|
||
|
}
|
||
|
|
||
|
#if ARDUINOJSON_HAS_RVALUE_REFERENCES
|
||
|
SECTION("Move construct") {
|
||
|
{
|
||
|
BasicJsonDocument<SpyingAllocator> doc1(4096, log);
|
||
|
doc1.set(std::string("The size of this string is 32!!"));
|
||
|
|
||
|
BasicJsonDocument<SpyingAllocator> doc2(move(doc1));
|
||
|
|
||
|
REQUIRE(doc2.as<std::string>() == "The size of this string is 32!!");
|
||
|
REQUIRE(doc1.as<std::string>() == "null");
|
||
|
REQUIRE(doc1.capacity() == 0);
|
||
|
REQUIRE(doc2.capacity() == 4096);
|
||
|
}
|
||
|
REQUIRE(log.str() == "A4096F");
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
SECTION("Copy assign") {
|
||
|
{
|
||
|
BasicJsonDocument<SpyingAllocator> doc1(4096, log);
|
||
|
doc1.set(std::string("The size of this string is 32!!"));
|
||
|
BasicJsonDocument<SpyingAllocator> doc2(8, log);
|
||
|
|
||
|
doc2 = doc1;
|
||
|
|
||
|
REQUIRE(doc1.as<std::string>() == "The size of this string is 32!!");
|
||
|
REQUIRE(doc2.as<std::string>() == "The size of this string is 32!!");
|
||
|
REQUIRE(doc2.capacity() == 4096);
|
||
|
}
|
||
|
REQUIRE(log.str() == "A4096A8FA4096FF");
|
||
|
}
|
||
|
|
||
|
#if ARDUINOJSON_HAS_RVALUE_REFERENCES
|
||
|
SECTION("Move assign") {
|
||
|
{
|
||
|
BasicJsonDocument<SpyingAllocator> doc1(4096, log);
|
||
|
doc1.set(std::string("The size of this string is 32!!"));
|
||
|
BasicJsonDocument<SpyingAllocator> doc2(8, log);
|
||
|
|
||
|
doc2 = move(doc1);
|
||
|
|
||
|
REQUIRE(doc2.as<std::string>() == "The size of this string is 32!!");
|
||
|
REQUIRE(doc1.as<std::string>() == "null");
|
||
|
REQUIRE(doc1.capacity() == 0);
|
||
|
REQUIRE(doc2.capacity() == 4096);
|
||
|
}
|
||
|
REQUIRE(log.str() == "A4096A8FF");
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
SECTION("garbageCollect()") {
|
||
|
BasicJsonDocument<ControllableAllocator> doc(4096);
|
||
|
|
||
|
SECTION("when allocation succeeds") {
|
||
|
deserializeJson(doc, "{\"blanket\":1,\"dancing\":2}");
|
||
|
REQUIRE(doc.capacity() == 4096);
|
||
|
REQUIRE(doc.memoryUsage() == JSON_OBJECT_SIZE(2) + 16);
|
||
|
doc.remove("blanket");
|
||
|
|
||
|
bool result = doc.garbageCollect();
|
||
|
|
||
|
REQUIRE(result == true);
|
||
|
REQUIRE(doc.memoryUsage() == JSON_OBJECT_SIZE(1) + 8);
|
||
|
REQUIRE(doc.capacity() == 4096);
|
||
|
REQUIRE(doc.as<std::string>() == "{\"dancing\":2}");
|
||
|
}
|
||
|
|
||
|
SECTION("when allocation fails") {
|
||
|
deserializeJson(doc, "{\"blanket\":1,\"dancing\":2}");
|
||
|
REQUIRE(doc.capacity() == 4096);
|
||
|
REQUIRE(doc.memoryUsage() == JSON_OBJECT_SIZE(2) + 16);
|
||
|
doc.remove("blanket");
|
||
|
doc.allocator().disable();
|
||
|
|
||
|
bool result = doc.garbageCollect();
|
||
|
|
||
|
REQUIRE(result == false);
|
||
|
REQUIRE(doc.memoryUsage() == JSON_OBJECT_SIZE(2) + 16);
|
||
|
REQUIRE(doc.capacity() == 4096);
|
||
|
REQUIRE(doc.as<std::string>() == "{\"dancing\":2}");
|
||
|
}
|
||
|
}
|
||
|
}
|