123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187 |
- // ArduinoJson - https://arduinojson.org
- // Copyright © 2014-2022, Benoit BLANCHON
- // MIT License
- #include <ArduinoJson.h>
- #include <stdlib.h> // malloc, free
- #include <catch.hpp>
- #include <sstream>
- #include <utility>
- using ARDUINOJSON_NAMESPACE::addPadding;
- 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(std::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 larger") {
- {
- 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");
- }
- SECTION("Copy assign smaller") {
- {
- BasicJsonDocument<SpyingAllocator> doc1(1024, log);
- doc1.set(std::string("The size of this string is 32!!"));
- BasicJsonDocument<SpyingAllocator> doc2(4096, 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() == 1024);
- }
- REQUIRE(log.str() == "A1024A4096FA1024FF");
- }
- SECTION("Copy assign same size") {
- {
- BasicJsonDocument<SpyingAllocator> doc1(1024, log);
- doc1.set(std::string("The size of this string is 32!!"));
- BasicJsonDocument<SpyingAllocator> doc2(1024, 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() == 1024);
- }
- REQUIRE(log.str() == "A1024A1024FF");
- }
- #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 = std::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}");
- }
- }
- }
|