Skip to content

Commit

Permalink
Add MsgPack bin8/bin16/bin32 support
Browse files Browse the repository at this point in the history
Closes #2078
Closes #922
  • Loading branch information
Sanae6 authored and bblanchon committed Apr 29, 2024
1 parent cd4bf33 commit 18a9a5b
Show file tree
Hide file tree
Showing 16 changed files with 358 additions and 11 deletions.
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ ArduinoJson: change log
=======================

* Add `ARDUINOJSON_STRING_LENGTH_SIZE` to the namespace name
* Add MsgPack bin8/bin16/bin32 support (PR #2078 by @Sanae6)

v7.0.4 (2024-03-12)
------
Expand Down
36 changes: 36 additions & 0 deletions extras/tests/JsonVariant/compare.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -113,6 +113,42 @@ TEST_CASE("Compare JsonVariant with JsonVariant") {
CHECK_FALSE(a == b);
}

SECTION("MsgPackBinary('abc') vs MsgPackBinary('abc')") {
a.set(MsgPackBinary("abc", 4));
b.set(MsgPackBinary("abc", 4));

CHECK(a == b);
CHECK(a <= b);
CHECK(a >= b);
CHECK_FALSE(a != b);
CHECK_FALSE(a < b);
CHECK_FALSE(a > b);
}

SECTION("MsgPackBinary('abc') vs MsgPackBinary('bcd')") {
a.set(MsgPackBinary("abc", 4));
b.set(MsgPackBinary("bcd", 4));

CHECK(a != b);
CHECK(a < b);
CHECK(a <= b);
CHECK_FALSE(a == b);
CHECK_FALSE(a > b);
CHECK_FALSE(a >= b);
}

SECTION("MsgPackBinary('bcd') vs MsgPackBinary('abc')") {
a.set(MsgPackBinary("bcd", 4));
b.set(MsgPackBinary("abc", 4));

CHECK(a != b);
CHECK(a > b);
CHECK(a >= b);
CHECK_FALSE(a < b);
CHECK_FALSE(a <= b);
CHECK_FALSE(a == b);
}

SECTION("false vs true") {
a.set(false);
b.set(true);
Expand Down
3 changes: 3 additions & 0 deletions extras/tests/JsonVariant/unbound.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,8 @@ TEST_CASE("Unbound JsonVariant") {
CHECK(variant.as<JsonObject>().isNull());
CHECK(variant.as<JsonObjectConst>().isNull());
CHECK(variant.as<JsonString>().isNull());
CHECK(variant.as<MsgPackBinary>().data() == nullptr);
CHECK(variant.as<MsgPackBinary>().size() == 0);
}

SECTION("is<T>()") {
Expand All @@ -46,6 +48,7 @@ TEST_CASE("Unbound JsonVariant") {
CHECK_FALSE(variant.set(serialized("42")));
CHECK_FALSE(variant.set(serialized(std::string("42"))));
CHECK_FALSE(variant.set(true));
CHECK_FALSE(variant.set(MsgPackBinary("hello", 5)));
}

SECTION("add()") {
Expand Down
37 changes: 35 additions & 2 deletions extras/tests/MixedConfiguration/string_length_size_1.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,22 @@ TEST_CASE("ARDUINOJSON_STRING_LENGTH_SIZE == 1") {
REQUIRE(doc.overflowed() == true);
}

SECTION("set() returns true if binary has 255 characters") {
auto str = std::string(255, '?');
auto result = doc.set(MsgPackBinary(str.data(), str.size()));

REQUIRE(result == true);
REQUIRE(doc.overflowed() == false);
}

SECTION("set() returns false if binary has 256 characters") {
auto str = std::string(256, '?');
auto result = doc.set(MsgPackBinary(str.data(), str.size()));

REQUIRE(result == false);
REQUIRE(doc.overflowed() == true);
}

SECTION("deserializeJson() returns Ok if string has 255 characters") {
auto input = "\"" + std::string(255, '?') + "\"";

Expand All @@ -37,7 +53,7 @@ TEST_CASE("ARDUINOJSON_STRING_LENGTH_SIZE == 1") {
REQUIRE(err == DeserializationError::NoMemory);
}

SECTION("deserializeMsgPack() returns Ok of string has 255 characters") {
SECTION("deserializeMsgPack() returns Ok if string has 255 characters") {
auto input = "\xd9\xff" + std::string(255, '?');

auto err = deserializeMsgPack(doc, input);
Expand All @@ -46,11 +62,28 @@ TEST_CASE("ARDUINOJSON_STRING_LENGTH_SIZE == 1") {
}

SECTION(
"deserializeMsgPack() returns NoMemory of string has 256 characters") {
"deserializeMsgPack() returns NoMemory if string has 256 characters") {
auto input = std::string("\xda\x01\x00", 3) + std::string(256, '?');

auto err = deserializeMsgPack(doc, input);

REQUIRE(err == DeserializationError::NoMemory);
}

SECTION("deserializeMsgPack() returns Ok if binary has 255 characters") {
auto input = "\xc4\xff" + std::string(255, '?');

auto err = deserializeMsgPack(doc, input);

REQUIRE(err == DeserializationError::Ok);
}

SECTION(
"deserializeMsgPack() returns NoMemory if binary has 256 characters") {
auto input = std::string("\xc5\x01\x00", 3) + std::string(256, '?');

auto err = deserializeMsgPack(doc, input);

REQUIRE(err == DeserializationError::NoMemory);
}
}
38 changes: 36 additions & 2 deletions extras/tests/MixedConfiguration/string_length_size_2.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,22 @@ TEST_CASE("ARDUINOJSON_STRING_LENGTH_SIZE == 2") {
REQUIRE(doc.overflowed() == true);
}

SECTION("set() returns true if string has 65535 characters") {
auto str = std::string(65535, '?');
auto result = doc.set(MsgPackBinary(str.data(), str.size()));

REQUIRE(result == true);
REQUIRE(doc.overflowed() == false);
}

SECTION("set() returns false if string has 65536 characters") {
auto str = std::string(65536, '?');
auto result = doc.set(MsgPackBinary(str.data(), str.size()));

REQUIRE(result == false);
REQUIRE(doc.overflowed() == true);
}

SECTION("deserializeJson() returns Ok if string has 65535 characters") {
auto input = "\"" + std::string(65535, '?') + "\"";

Expand All @@ -37,7 +53,7 @@ TEST_CASE("ARDUINOJSON_STRING_LENGTH_SIZE == 2") {
REQUIRE(err == DeserializationError::NoMemory);
}

SECTION("deserializeMsgPack() returns Ok of string has 65535 characters") {
SECTION("deserializeMsgPack() returns Ok if string has 65535 characters") {
auto input = "\xda\xff\xff" + std::string(65535, '?');

auto err = deserializeMsgPack(doc, input);
Expand All @@ -46,12 +62,30 @@ TEST_CASE("ARDUINOJSON_STRING_LENGTH_SIZE == 2") {
}

SECTION(
"deserializeMsgPack() returns NoMemory of string has 65536 characters") {
"deserializeMsgPack() returns NoMemory if string has 65536 characters") {
auto input =
std::string("\xdb\x00\x01\x00\x00", 5) + std::string(65536, '?');

auto err = deserializeMsgPack(doc, input);

REQUIRE(err == DeserializationError::NoMemory);
}

SECTION("deserializeMsgPack() returns Ok if binary has 65535 characters") {
auto input = "\xc5\xff\xff" + std::string(65535, '?');

auto err = deserializeMsgPack(doc, input);

REQUIRE(err == DeserializationError::Ok);
}

SECTION(
"deserializeMsgPack() returns NoMemory of binary has 65536 characters") {
auto input =
std::string("\xc6\x00\x01\x00\x00", 5) + std::string(65536, '?');

auto err = deserializeMsgPack(doc, input);

REQUIRE(err == DeserializationError::NoMemory);
}
}
44 changes: 43 additions & 1 deletion extras/tests/MixedConfiguration/string_length_size_4.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,14 @@ TEST_CASE("ARDUINOJSON_STRING_LENGTH_SIZE == 4") {
REQUIRE(doc.overflowed() == false);
}

SECTION("set() returns true if binary has 65536 characters") {
auto str = std::string(65536, '?');
auto result = doc.set(MsgPackBinary(str.data(), str.size()));

REQUIRE(result == true);
REQUIRE(doc.overflowed() == false);
}

SECTION("deserializeJson() returns Ok if string has 65536 characters") {
auto input = "\"" + std::string(65536, '?') + "\"";

Expand All @@ -22,11 +30,45 @@ TEST_CASE("ARDUINOJSON_STRING_LENGTH_SIZE == 4") {
REQUIRE(err == DeserializationError::Ok);
}

SECTION("deserializeMsgPack() returns Ok of string has 65536 characters") {
SECTION("deserializeMsgPack() returns Ok if string has 65536 characters") {
auto input = "\xda\xff\xff" + std::string(65536, '?');

auto err = deserializeMsgPack(doc, input);

REQUIRE(err == DeserializationError::Ok);
}

SECTION("deserializeMsgPack() returns Ok if binary has 65536 characters") {
auto input = "\xc5\xff\xff" + std::string(65536, '?');

auto err = deserializeMsgPack(doc, input);

REQUIRE(err == DeserializationError::Ok);
}

SECTION("bin 32 deserialization") {
auto str = std::string(65536, '?');
auto input = std::string("\xc6\x00\x01\x00\x00", 5) + str;

auto err = deserializeMsgPack(doc, input);

REQUIRE(err == DeserializationError::Ok);
REQUIRE(doc.is<MsgPackBinary>());
auto binary = doc.as<MsgPackBinary>();
REQUIRE(binary.size() == 65536);
REQUIRE(binary.data() != nullptr);
REQUIRE(std::string(reinterpret_cast<const char*>(binary.data()),
binary.size()) == str);
}

SECTION("bin 32 serialization") {
auto str = std::string(65536, '?');
doc.set(MsgPackBinary(str.data(), str.size()));

std::string output;
auto result = serializeMsgPack(doc, output);

REQUIRE(result == 5 + str.size());
REQUIRE(output == std::string("\xc6\x00\x01\x00\x00", 5) + str);
}
}
31 changes: 31 additions & 0 deletions extras/tests/MsgPackDeserializer/deserializeVariant.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@
// MIT License

#include <ArduinoJson.h>
#include <array>
#include <catch.hpp>

#include "Allocators.hpp"
Expand Down Expand Up @@ -139,6 +140,36 @@ TEST_CASE("deserialize MsgPack value") {
SECTION("str 32") {
checkValue<std::string>("\xdb\x00\x00\x00\x05hello", std::string("hello"));
}

SECTION("bin 8") {
JsonDocument doc;

DeserializationError error = deserializeMsgPack(doc, "\xc4\x01?");

REQUIRE(error == DeserializationError::Ok);
REQUIRE(doc.is<MsgPackBinary>());
auto binary = doc.as<MsgPackBinary>();
REQUIRE(binary.size() == 1);
REQUIRE(binary.data() != nullptr);
REQUIRE(reinterpret_cast<const char*>(binary.data())[0] == '?');
}

SECTION("bin 16") {
JsonDocument doc;

auto str = std::string(256, '?');
auto input = std::string("\xc5\x01\x00", 3) + str;

DeserializationError error = deserializeMsgPack(doc, input);

REQUIRE(error == DeserializationError::Ok);
REQUIRE(doc.is<MsgPackBinary>());
auto binary = doc.as<MsgPackBinary>();
REQUIRE(binary.size() == 0x100);
REQUIRE(binary.data() != nullptr);
REQUIRE(std::string(reinterpret_cast<const char*>(binary.data()),
binary.size()) == str);
}
}

TEST_CASE("deserializeMsgPack() under memory constaints") {
Expand Down
12 changes: 12 additions & 0 deletions extras/tests/MsgPackSerializer/serializeVariant.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@
// MIT License

#include <ArduinoJson.h>
#include <array>
#include <catch.hpp>

template <typename T>
Expand Down Expand Up @@ -146,6 +147,17 @@ TEST_CASE("serialize MsgPack value") {
checkVariant(serialized("\xDB\x00\x01\x00\x00", 5), "\xDB\x00\x01\x00\x00");
}

SECTION("bin 8") {
auto str = std::string(1, 1);
checkVariant(MsgPackBinary(str.data(), str.size()), "\xC4\x01\x01");
}

SECTION("bin 16") {
auto str = std::string(256, 1);
checkVariant(MsgPackBinary(str.data(), str.size()),
std::string("\xC5\x01\x00", 3) + str);
}

SECTION("serialize round double as integer") { // Issue #1718
checkVariant(-32768.0, "\xD1\x80\x00");
checkVariant(-129.0, "\xD1\xFF\x7F");
Expand Down
4 changes: 4 additions & 0 deletions src/ArduinoJson/Json/JsonSerializer.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -81,6 +81,10 @@ class JsonSerializer : public VariantDataVisitor<size_t> {
return bytesWritten();
}

size_t visit(MsgPackBinary) {
return visit(nullptr);
}

size_t visit(JsonInteger value) {
formatter_.writeInteger(value);
return bytesWritten();
Expand Down
23 changes: 23 additions & 0 deletions src/ArduinoJson/MsgPack/MsgPackBinary.hpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
#pragma once

ARDUINOJSON_BEGIN_PUBLIC_NAMESPACE

class MsgPackBinary {
public:
MsgPackBinary() : data_(nullptr), size_(0) {}
explicit MsgPackBinary(const void* c, size_t size) : data_(c), size_(size) {}

const void* data() const {
return data_;
}

size_t size() const {
return size_;
}

private:
const void* data_;
size_t size_;
};

ARDUINOJSON_END_PUBLIC_NAMESPACE

0 comments on commit 18a9a5b

Please sign in to comment.