memgraph/tests/unit/custom_serialization.cpp
Matej Ferencevic 0493470f98 Add support for std::pair in SLK
Reviewers: teon.banek

Reviewed By: teon.banek

Subscribers: pullbot

Differential Revision: https://phabricator.memgraph.io/D1717
2018-10-31 15:21:36 +01:00

474 lines
16 KiB
C++

#include <gtest/gtest.h>
#include "communication/rpc/serialization.hpp"
#define CREATE_PRIMITIVE_TEST(primitive_type, original_value, decoded_value) \
{ \
ASSERT_NE(original_value, decoded_value); \
primitive_type original = original_value; \
slk::Builder builder; \
slk::Save(original, &builder); \
ASSERT_EQ(builder.size(), sizeof(primitive_type)); \
primitive_type decoded = decoded_value; \
slk::Reader reader(builder.data(), builder.size()); \
slk::Load(&decoded, &reader); \
ASSERT_EQ(original, decoded); \
}
TEST(CustomSerialization, Primitive) {
CREATE_PRIMITIVE_TEST(bool, true, false);
CREATE_PRIMITIVE_TEST(int8_t, 0x12, 0);
CREATE_PRIMITIVE_TEST(uint8_t, 0x12, 0);
CREATE_PRIMITIVE_TEST(int16_t, 0x1234, 0);
CREATE_PRIMITIVE_TEST(uint16_t, 0x1234, 0);
CREATE_PRIMITIVE_TEST(int32_t, 0x12345678, 0);
CREATE_PRIMITIVE_TEST(uint32_t, 0x12345678, 0);
CREATE_PRIMITIVE_TEST(int64_t, 0x1234567890abcdef, 0);
CREATE_PRIMITIVE_TEST(uint64_t, 0x1234567890abcdef, 0);
CREATE_PRIMITIVE_TEST(float, 1.23456789, 0);
CREATE_PRIMITIVE_TEST(double, 1234567890.1234567890, 0);
}
TEST(CustomSerialization, String) {
std::string original = "hello world";
slk::Builder builder;
slk::Save(original, &builder);
ASSERT_EQ(builder.size(), sizeof(uint64_t) + original.size());
std::string decoded;
slk::Reader reader(builder.data(), builder.size());
slk::Load(&decoded, &reader);
ASSERT_EQ(original, decoded);
}
TEST(CustomSerialization, VectorPrimitive) {
std::vector<int> original{1, 2, 3, 4, 5};
slk::Builder builder;
slk::Save(original, &builder);
ASSERT_EQ(builder.size(), sizeof(uint64_t) + original.size() * sizeof(int));
std::vector<int> decoded;
slk::Reader reader(builder.data(), builder.size());
slk::Load(&decoded, &reader);
ASSERT_EQ(original, decoded);
}
TEST(CustomSerialization, VectorString) {
std::vector<std::string> original{"hai hai hai", "nandare!"};
slk::Builder builder;
slk::Save(original, &builder);
uint64_t size = sizeof(uint64_t);
for (const auto &item : original) {
size += sizeof(uint64_t) + item.size();
}
ASSERT_EQ(builder.size(), size);
std::vector<std::string> decoded;
slk::Reader reader(builder.data(), builder.size());
slk::Load(&decoded, &reader);
ASSERT_EQ(original, decoded);
}
TEST(CustomSerialization, SetPrimitive) {
std::set<int> original{1, 2, 3, 4, 5};
slk::Builder builder;
slk::Save(original, &builder);
ASSERT_EQ(builder.size(), sizeof(uint64_t) + original.size() * sizeof(int));
std::set<int> decoded;
slk::Reader reader(builder.data(), builder.size());
slk::Load(&decoded, &reader);
ASSERT_EQ(original, decoded);
}
TEST(CustomSerialization, SetString) {
std::set<std::string> original{"hai hai hai", "nandare!"};
slk::Builder builder;
slk::Save(original, &builder);
uint64_t size = sizeof(uint64_t);
for (const auto &item : original) {
size += sizeof(uint64_t) + item.size();
}
ASSERT_EQ(builder.size(), size);
std::set<std::string> decoded;
slk::Reader reader(builder.data(), builder.size());
slk::Load(&decoded, &reader);
ASSERT_EQ(original, decoded);
}
TEST(CustomSerialization, MapPrimitive) {
std::map<int, int> original{{1, 2}, {3, 4}, {5, 6}};
slk::Builder builder;
slk::Save(original, &builder);
ASSERT_EQ(builder.size(),
sizeof(uint64_t) + original.size() * sizeof(int) * 2);
std::map<int, int> decoded;
slk::Reader reader(builder.data(), builder.size());
slk::Load(&decoded, &reader);
ASSERT_EQ(original, decoded);
}
TEST(CustomSerialization, MapString) {
std::map<std::string, std::string> original{{"hai hai hai", "nandare!"}};
slk::Builder builder;
slk::Save(original, &builder);
uint64_t size = sizeof(uint64_t);
for (const auto &item : original) {
size += sizeof(uint64_t) + item.first.size();
size += sizeof(uint64_t) + item.second.size();
}
ASSERT_EQ(builder.size(), size);
std::map<std::string, std::string> decoded;
slk::Reader reader(builder.data(), builder.size());
slk::Load(&decoded, &reader);
ASSERT_EQ(original, decoded);
}
TEST(CustomSerialization, UnorderedMapPrimitive) {
std::unordered_map<int, int> original{{1, 2}, {3, 4}, {5, 6}};
slk::Builder builder;
slk::Save(original, &builder);
ASSERT_EQ(builder.size(),
sizeof(uint64_t) + original.size() * sizeof(int) * 2);
std::unordered_map<int, int> decoded;
slk::Reader reader(builder.data(), builder.size());
slk::Load(&decoded, &reader);
ASSERT_EQ(original, decoded);
}
TEST(CustomSerialization, UnorderedMapString) {
std::unordered_map<std::string, std::string> original{
{"hai hai hai", "nandare!"}};
slk::Builder builder;
slk::Save(original, &builder);
uint64_t size = sizeof(uint64_t);
for (const auto &item : original) {
size += sizeof(uint64_t) + item.first.size();
size += sizeof(uint64_t) + item.second.size();
}
ASSERT_EQ(builder.size(), size);
std::unordered_map<std::string, std::string> decoded;
slk::Reader reader(builder.data(), builder.size());
slk::Load(&decoded, &reader);
ASSERT_EQ(original, decoded);
}
TEST(CustomSerialization, UniquePtrEmpty) {
std::unique_ptr<std::string> original;
slk::Builder builder;
slk::Save(original, &builder);
ASSERT_EQ(builder.size(), sizeof(bool));
std::unique_ptr<std::string> decoded =
std::make_unique<std::string>("nandare!");
ASSERT_NE(decoded.get(), nullptr);
slk::Reader reader(builder.data(), builder.size());
slk::Load(&decoded, &reader);
ASSERT_EQ(decoded.get(), nullptr);
}
TEST(CustomSerialization, UniquePtrFull) {
std::unique_ptr<std::string> original =
std::make_unique<std::string>("nandare!");
slk::Builder builder;
slk::Save(original, &builder);
ASSERT_EQ(builder.size(),
sizeof(bool) + sizeof(uint64_t) + original.get()->size());
std::unique_ptr<std::string> decoded;
ASSERT_EQ(decoded.get(), nullptr);
slk::Reader reader(builder.data(), builder.size());
slk::Load(&decoded, &reader);
ASSERT_NE(decoded.get(), nullptr);
ASSERT_EQ(*original.get(), *decoded.get());
}
TEST(CustomSerialization, OptionalPrimitiveEmpty) {
std::experimental::optional<int> original;
slk::Builder builder;
slk::Save(original, &builder);
ASSERT_EQ(builder.size(), sizeof(bool));
std::experimental::optional<int> decoded = 5;
ASSERT_NE(decoded, std::experimental::nullopt);
slk::Reader reader(builder.data(), builder.size());
slk::Load(&decoded, &reader);
ASSERT_EQ(decoded, std::experimental::nullopt);
}
TEST(CustomSerialization, OptionalPrimitiveFull) {
std::experimental::optional<int> original = 5;
slk::Builder builder;
slk::Save(original, &builder);
ASSERT_EQ(builder.size(), sizeof(bool) + sizeof(int));
std::experimental::optional<int> decoded;
ASSERT_EQ(decoded, std::experimental::nullopt);
slk::Reader reader(builder.data(), builder.size());
slk::Load(&decoded, &reader);
ASSERT_NE(decoded, std::experimental::nullopt);
ASSERT_EQ(*original, *decoded);
}
TEST(CustomSerialization, OptionalStringEmpty) {
std::experimental::optional<std::string> original;
slk::Builder builder;
slk::Save(original, &builder);
ASSERT_EQ(builder.size(), sizeof(bool));
std::experimental::optional<std::string> decoded = "nandare!";
ASSERT_NE(decoded, std::experimental::nullopt);
slk::Reader reader(builder.data(), builder.size());
slk::Load(&decoded, &reader);
ASSERT_EQ(decoded, std::experimental::nullopt);
}
TEST(CustomSerialization, OptionalStringFull) {
std::experimental::optional<std::string> original = "nandare!";
slk::Builder builder;
slk::Save(original, &builder);
ASSERT_EQ(builder.size(), sizeof(bool) + sizeof(uint64_t) + original->size());
std::experimental::optional<std::string> decoded;
ASSERT_EQ(decoded, std::experimental::nullopt);
slk::Reader reader(builder.data(), builder.size());
slk::Load(&decoded, &reader);
ASSERT_NE(decoded, std::experimental::nullopt);
ASSERT_EQ(*original, *decoded);
}
TEST(CustomSerialization, Pair) {
std::pair<std::string, int> original{"nandare!", 5};
slk::Builder builder;
slk::Save(original, &builder);
ASSERT_EQ(builder.size(),
sizeof(uint64_t) + original.first.size() + sizeof(int));
std::pair<std::string, int> decoded;
slk::Reader reader(builder.data(), builder.size());
slk::Load(&decoded, &reader);
ASSERT_EQ(original, decoded);
}
TEST(CustomSerialization, SharedPtrEmpty) {
std::shared_ptr<std::string> original;
std::vector<std::string *> saved;
slk::Builder builder;
slk::Save(original, &builder, &saved);
ASSERT_EQ(builder.size(), sizeof(bool));
std::shared_ptr<std::string> decoded =
std::make_shared<std::string>("nandare!");
std::vector<std::shared_ptr<std::string>> loaded;
ASSERT_NE(decoded.get(), nullptr);
slk::Reader reader(builder.data(), builder.size());
slk::Load(&decoded, &reader, &loaded);
ASSERT_EQ(decoded.get(), nullptr);
ASSERT_EQ(saved.size(), 0);
ASSERT_EQ(loaded.size(), 0);
}
TEST(CustomSerialization, SharedPtrFull) {
std::shared_ptr<std::string> original =
std::make_shared<std::string>("nandare!");
std::vector<std::string *> saved;
slk::Builder builder;
slk::Save(original, &builder, &saved);
ASSERT_EQ(builder.size(),
sizeof(bool) * 2 + sizeof(uint64_t) + original.get()->size());
std::shared_ptr<std::string> decoded;
std::vector<std::shared_ptr<std::string>> loaded;
ASSERT_EQ(decoded.get(), nullptr);
slk::Reader reader(builder.data(), builder.size());
slk::Load(&decoded, &reader, &loaded);
ASSERT_NE(decoded.get(), nullptr);
ASSERT_EQ(*original.get(), *decoded.get());
ASSERT_EQ(saved.size(), 1);
ASSERT_EQ(loaded.size(), 1);
}
TEST(CustomSerialization, SharedPtrMultiple) {
std::shared_ptr<std::string> ptr1 = std::make_shared<std::string>("nandare!");
std::shared_ptr<std::string> ptr2;
std::shared_ptr<std::string> ptr3 =
std::make_shared<std::string>("hai hai hai");
std::vector<std::string *> saved;
slk::Builder builder;
slk::Save(ptr1, &builder, &saved);
slk::Save(ptr2, &builder, &saved);
slk::Save(ptr3, &builder, &saved);
slk::Save(ptr1, &builder, &saved);
slk::Save(ptr3, &builder, &saved);
// clang-format off
ASSERT_EQ(builder.size(),
sizeof(bool) * 2 + sizeof(uint64_t) + ptr1.get()->size() +
sizeof(bool) +
sizeof(bool) * 2 + sizeof(uint64_t) + ptr3.get()->size() +
sizeof(bool) * 2 + sizeof(uint64_t) +
sizeof(bool) * 2 + sizeof(uint64_t));
// clang-format on
std::shared_ptr<std::string> dec1, dec2, dec3, dec4, dec5;
std::vector<std::shared_ptr<std::string>> loaded;
slk::Reader reader(builder.data(), builder.size());
slk::Load(&dec1, &reader, &loaded);
slk::Load(&dec2, &reader, &loaded);
slk::Load(&dec3, &reader, &loaded);
slk::Load(&dec4, &reader, &loaded);
slk::Load(&dec5, &reader, &loaded);
ASSERT_EQ(saved.size(), 2);
ASSERT_EQ(loaded.size(), 2);
ASSERT_NE(dec1.get(), nullptr);
ASSERT_EQ(dec2.get(), nullptr);
ASSERT_NE(dec3.get(), nullptr);
ASSERT_NE(dec4.get(), nullptr);
ASSERT_NE(dec5.get(), nullptr);
ASSERT_EQ(*dec1.get(), *ptr1.get());
ASSERT_EQ(*dec3.get(), *ptr3.get());
ASSERT_EQ(*dec4.get(), *ptr1.get());
ASSERT_EQ(*dec5.get(), *ptr3.get());
ASSERT_NE(dec1.get(), dec3.get());
ASSERT_EQ(dec4.get(), dec1.get());
ASSERT_EQ(dec5.get(), dec3.get());
}
TEST(CustomSerialization, Complex) {
std::unique_ptr<std::vector<std::experimental::optional<std::string>>>
original = std::make_unique<
std::vector<std::experimental::optional<std::string>>>();
original.get()->push_back("nandare!");
original.get()->push_back(std::experimental::nullopt);
original.get()->push_back("hai hai hai");
slk::Builder builder;
slk::Save(original, &builder);
// clang-format off
ASSERT_EQ(builder.size(),
sizeof(bool) +
sizeof(uint64_t) +
sizeof(bool) + sizeof(uint64_t) + (*original.get())[0]->size() +
sizeof(bool) +
sizeof(bool) + sizeof(uint64_t) + (*original.get())[2]->size());
// clang-format on
std::unique_ptr<std::vector<std::experimental::optional<std::string>>>
decoded;
ASSERT_EQ(decoded.get(), nullptr);
slk::Reader reader(builder.data(), builder.size());
slk::Load(&decoded, &reader);
ASSERT_NE(decoded.get(), nullptr);
ASSERT_EQ(*original.get(), *decoded.get());
}
struct Foo {
std::string name;
std::experimental::optional<int> value;
};
bool operator==(const Foo &a, const Foo &b) {
return a.name == b.name && a.value == b.value;
}
namespace slk {
void Save(const Foo &obj, Builder *builder) {
Save(obj.name, builder);
Save(obj.value, builder);
}
void Load(Foo *obj, Reader *reader) {
Load(&obj->name, reader);
Load(&obj->value, reader);
}
} // namespace slk
TEST(CustomSerialization, VectorStruct) {
std::vector<Foo> original;
original.push_back({"hai hai hai", 5});
original.push_back({"nandare!", std::experimental::nullopt});
slk::Builder builder;
slk::Save(original, &builder);
// clang-format off
ASSERT_EQ(builder.size(),
sizeof(uint64_t) +
sizeof(uint64_t) + original[0].name.size() + sizeof(bool) + sizeof(int) +
sizeof(uint64_t) + original[1].name.size() + sizeof(bool));
// clang-format on
std::vector<Foo> decoded;
slk::Reader reader(builder.data(), builder.size());
slk::Load(&decoded, &reader);
ASSERT_EQ(original, decoded);
}
TEST(CustomSerialization, VectorSharedPtr) {
std::shared_ptr<std::string> ptr1 = std::make_shared<std::string>("nandare!");
std::shared_ptr<std::string> ptr2;
std::shared_ptr<std::string> ptr3 =
std::make_shared<std::string>("hai hai hai");
std::vector<std::shared_ptr<std::string>> original;
std::vector<std::string *> saved;
original.push_back(ptr1);
original.push_back(ptr2);
original.push_back(ptr3);
original.push_back(ptr1);
original.push_back(ptr3);
slk::Builder builder;
slk::Save<std::shared_ptr<std::string>>(
original, &builder, [&saved](const auto &item, auto *builder) {
Save(item, builder, &saved);
});
std::vector<std::shared_ptr<std::string>> decoded;
std::vector<std::shared_ptr<std::string>> loaded;
slk::Reader reader(builder.data(), builder.size());
slk::Load<std::shared_ptr<std::string>>(
&decoded, &reader,
[&loaded](auto *item, auto *reader) { Load(item, reader, &loaded); });
ASSERT_EQ(decoded.size(), original.size());
ASSERT_EQ(saved.size(), 2);
ASSERT_EQ(loaded.size(), saved.size());
ASSERT_EQ(*decoded[0].get(), *ptr1.get());
ASSERT_EQ(decoded[1].get(), nullptr);
ASSERT_EQ(*decoded[2].get(), *ptr3.get());
ASSERT_EQ(*decoded[3].get(), *ptr1.get());
ASSERT_EQ(*decoded[4].get(), *ptr3.get());
ASSERT_NE(decoded[0].get(), decoded[2].get());
ASSERT_EQ(decoded[3].get(), decoded[0].get());
ASSERT_EQ(decoded[4].get(), decoded[2].get());
}
TEST(CustomSerialization, OptionalSharedPtr) {
std::experimental::optional<std::shared_ptr<std::string>> original =
std::make_shared<std::string>("nandare!");
std::vector<std::string *> saved;
slk::Builder builder;
slk::Save<std::shared_ptr<std::string>>(
original, &builder, [&saved](const auto &item, auto *builder) {
Save(item, builder, &saved);
});
std::experimental::optional<std::shared_ptr<std::string>> decoded;
std::vector<std::shared_ptr<std::string>> loaded;
slk::Reader reader(builder.data(), builder.size());
slk::Load<std::shared_ptr<std::string>>(
&decoded, &reader,
[&loaded](auto *item, auto *reader) { Load(item, reader, &loaded); });
ASSERT_NE(decoded, std::experimental::nullopt);
ASSERT_EQ(saved.size(), 1);
ASSERT_EQ(loaded.size(), 1);
ASSERT_EQ(*decoded->get(), *original->get());
}