memgraph/tests/unit/storage_rocks.cpp
Gareth Andrew Lloyd a282542666
Optimise ORDER BY, RANGE, UNWIND (#1781)
* Optimise frame change

* Optimise distinct + orderby memory usage

- dispose collections as earlier as possible
- move values rather than copy

* Better perf, ORDER BY

* Optimise RANGE and UNWIND

* ConstraintVerificationInfo only if at least one constraint

* Optimise TypeValue

* Clang-tidy fix
2024-03-12 00:26:11 +00:00

287 lines
12 KiB
C++

// Copyright 2024 Memgraph Ltd.
//
// Use of this software is governed by the Business Source License
// included in the file licenses/BSL.txt; by using this file, you agree to be bound by the terms of the Business Source
// License, and you may not use this file except in compliance with the Business Source License.
//
// As of the Change Date specified in that file, in accordance with
// the Business Source License, use of this software will be governed
// by the Apache License, Version 2.0, included in the file
// licenses/APL.txt.
#include <gmock/gmock.h>
#include <gtest/gtest.h>
#include <cassert>
#include <exception>
#include <string>
#include <unordered_set>
#include "disk_test_utils.hpp"
#include "storage/v2/delta.hpp"
#include "storage/v2/disk/storage.hpp"
#include "storage/v2/id_types.hpp"
#include "storage/v2/isolation_level.hpp"
#include "storage/v2/property_store.hpp"
#include "storage/v2/property_value.hpp"
#include "storage/v2/storage.hpp"
#include "storage/v2/vertex_accessor.hpp"
#include "storage/v2/view.hpp"
#include "utils/rocksdb_serialization.hpp"
using memgraph::replication_coordination_glue::ReplicationRole;
// NOLINTNEXTLINE(google-build-using-namespace)
using namespace memgraph::storage;
/* Tests that serialization of vertices and edges to RocksDB works correctly.
*/
class RocksDBStorageTest : public ::testing::TestWithParam<bool> {
public:
const std::string testSuite = "storage_rocks";
RocksDBStorageTest() {
config_ = disk_test_utils::GenerateOnDiskConfig(testSuite);
storage = std::make_unique<DiskStorage>(config_);
}
void TearDown() override {
storage.reset(nullptr);
disk_test_utils::RemoveRocksDbDirs(testSuite);
}
~RocksDBStorageTest() override = default;
protected:
std::unique_ptr<Storage> storage;
Config config_;
};
TEST_F(RocksDBStorageTest, SerializeVertexGID) {
auto acc = storage->Access(ReplicationRole::MAIN);
auto vertex = acc->CreateVertex();
auto gid = vertex.Gid();
ASSERT_EQ(memgraph::utils::SerializeVertex(*vertex.vertex_), "|" + gid.ToString());
}
TEST_F(RocksDBStorageTest, SerializeVertexGIDLabels) {
auto acc = storage->Access(ReplicationRole::MAIN);
auto vertex = acc->CreateVertex();
auto ser_player_label = acc->NameToLabel("Player");
auto ser_user_label = acc->NameToLabel("User");
ASSERT_FALSE(vertex.AddLabel(ser_player_label).HasError());
ASSERT_FALSE(vertex.AddLabel(ser_user_label).HasError());
auto gid = vertex.Gid();
ASSERT_EQ(memgraph::utils::SerializeVertex(*vertex.vertex_),
ser_player_label.ToString() + "," + ser_user_label.ToString() + "|" + gid.ToString());
}
TEST_F(RocksDBStorageTest, SerializePropertiesLocalBuffer) {
PropertyStore props;
auto id = PropertyId::FromInt(0);
auto id_value = PropertyValue(1);
auto completion_percentage = PropertyId::FromInt(1);
auto completion_percentage_value = PropertyValue(14);
auto gender = PropertyId::FromInt(2);
auto gender_value = PropertyValue("man");
auto age = PropertyId::FromInt(3);
auto age_value = PropertyValue(26);
ASSERT_TRUE(props.SetProperty(id, id_value));
ASSERT_TRUE(props.SetProperty(age, age_value));
ASSERT_TRUE(props.SetProperty(completion_percentage, completion_percentage_value));
ASSERT_TRUE(props.SetProperty(gender, gender_value));
std::string serialized_props = memgraph::utils::SerializeProperties(props);
PropertyStore deserialized_props = PropertyStore::CreateFromBuffer(serialized_props);
for (const auto &[prop_id, prop_value] : props.Properties()) {
ASSERT_TRUE(deserialized_props.IsPropertyEqual(prop_id, prop_value));
}
}
TEST_F(RocksDBStorageTest, SerializePropertiesExternalBuffer) {
PropertyStore props;
auto id = PropertyId::FromInt(0);
auto id_value = PropertyValue(1);
auto completion_percentage = PropertyId::FromInt(1);
auto completion_percentage_value = PropertyValue(14);
auto gender = PropertyId::FromInt(2);
// Use big value so that memory for properties is allocated on the heap not on the stack
auto gender_value = PropertyValue("man167863816386826");
auto age = PropertyId::FromInt(3);
auto age_value = PropertyValue(26);
ASSERT_TRUE(props.SetProperty(id, id_value));
ASSERT_TRUE(props.SetProperty(age, age_value));
ASSERT_TRUE(props.SetProperty(completion_percentage, completion_percentage_value));
ASSERT_TRUE(props.SetProperty(gender, gender_value));
std::string serialized_props = memgraph::utils::SerializeProperties(props);
PropertyStore deserialized_props = PropertyStore::CreateFromBuffer(serialized_props);
for (const auto &[prop_id, prop_value] : props.Properties()) {
ASSERT_TRUE(deserialized_props.IsPropertyEqual(prop_id, prop_value));
}
}
TEST(RocksDbSerDeSuite, ExtractVertexGidFromVertexKeyNoLabels) {
auto gid = Gid::FromInt(1);
Vertex vertex(gid, nullptr);
std::string serializedVertex = memgraph::utils::SerializeVertex(vertex);
ASSERT_EQ(memgraph::utils::ExtractGidFromKey(serializedVertex), "1");
}
TEST(RocksDbSerDeSuite, ExtractVertexGidFromVertexKeyWithOneLabel) {
auto gid = Gid::FromInt(1);
Vertex vertex(gid, nullptr);
vertex.labels.push_back(LabelId::FromInt(2));
std::string serializedVertex = memgraph::utils::SerializeVertex(vertex);
ASSERT_EQ(memgraph::utils::ExtractGidFromKey(serializedVertex), "1");
}
TEST(RocksDbSerDeSuite, ExtractVertexGidFromVertexKeyWithMultipleLabels) {
auto gid = Gid::FromInt(1);
Vertex vertex(gid, nullptr);
std::vector<unsigned> labels = {2, 3, 4};
for (unsigned label : labels) {
vertex.labels.push_back(LabelId::FromInt(label));
}
std::string serializedVertex = memgraph::utils::SerializeVertex(vertex);
ASSERT_EQ(memgraph::utils::ExtractGidFromKey(serializedVertex), "1");
}
TEST(RocksDbSerDeSuite, ExtractLabelsFromMainDiskStorageWhenOnlyOneLabel) {
auto gid = Gid::FromInt(1);
Vertex vertex(gid, nullptr);
std::vector<unsigned> labels = {2};
std::vector<std::string> expectedLabelsStr = {"2"};
for (unsigned label : labels) {
vertex.labels.push_back(LabelId::FromInt(label));
}
std::string serializedVertex = memgraph::utils::SerializeVertex(vertex);
ASSERT_EQ(memgraph::utils::ExtractLabelsFromMainDiskStorage(serializedVertex), expectedLabelsStr);
}
TEST(RocksDbSerDeSuite, ExtractLabelsFromMainDiskStorageWhenMultipleLabels) {
auto gid = Gid::FromInt(1);
Vertex vertex(gid, nullptr);
std::vector<unsigned> labels = {2, 3, 4};
std::vector<std::string> expectedLabelsStr = {"2", "3", "4"};
for (unsigned label : labels) {
vertex.labels.push_back(LabelId::FromInt(label));
}
std::string serializedVertex = memgraph::utils::SerializeVertex(vertex);
ASSERT_EQ(memgraph::utils::ExtractLabelsFromMainDiskStorage(serializedVertex), expectedLabelsStr);
}
TEST(RocksDbSerDeSuite, ExtractVertexGidFromMainDiskStorageNoLabels) {
auto gid = Gid::FromInt(1);
Vertex vertex(gid, nullptr);
std::string serializedVertex = memgraph::utils::SerializeVertex(vertex);
ASSERT_EQ(memgraph::utils::ExtractGidFromMainDiskStorage(serializedVertex), "1");
}
TEST(RocksDbSerDeSuite, ExtractVertexGidFromMainDiskStorageWithOneLabel) {
auto gid = Gid::FromInt(1);
Vertex vertex(gid, nullptr);
vertex.labels.push_back(LabelId::FromInt(2));
std::string serializedVertex = memgraph::utils::SerializeVertex(vertex);
ASSERT_EQ(memgraph::utils::ExtractGidFromMainDiskStorage(serializedVertex), "1");
}
TEST(RocksDbSerDeSuite, ExtractVertexGidFromMainDiskStorageWithMultipleLabels) {
auto gid = Gid::FromInt(1);
Vertex vertex(gid, nullptr);
vertex.labels.push_back(LabelId::FromInt(2));
vertex.labels.push_back(LabelId::FromInt(3));
vertex.labels.push_back(LabelId::FromInt(4));
std::string serializedVertex = memgraph::utils::SerializeVertex(vertex);
ASSERT_EQ(memgraph::utils::ExtractGidFromLabelIndexStorage(serializedVertex), "1");
}
TEST(RocksDbSerDeSuite, ExtractGidFromLabelIndexStorageKey) {
auto gid = Gid::FromInt(1);
LabelId label = LabelId::FromInt(2);
std::string serializedVertex = memgraph::utils::SerializeVertexAsKeyForLabelIndex(label, gid);
ASSERT_EQ(memgraph::utils::ExtractGidFromLabelIndexStorage(serializedVertex), "1");
}
TEST(RocksDbSerDeSuite, DeserializePropertiesFromLabelIndexStorage) {
std::string expectedGid = "1";
LabelId indexingLabel = LabelId::FromInt(2);
std::vector<LabelId> labels = {LabelId::FromInt(3), LabelId::FromInt(4)};
std::map<PropertyId, PropertyValue> properties = {{PropertyId::FromInt(5), PropertyValue("5")},
{PropertyId::FromInt(6), PropertyValue("6")}};
PropertyStore propertyStore;
propertyStore.InitProperties(properties);
std::string serializedVertex =
memgraph::utils::SerializeVertexAsValueForLabelIndex(indexingLabel, labels, propertyStore);
ASSERT_EQ(memgraph::utils::DeserializePropertiesFromLabelIndexStorage(serializedVertex).StringBuffer(),
propertyStore.StringBuffer());
}
TEST(RocksDbSerDeSuite, DeserializePropertiesFromLabelPropertyIndexStorage) {
std::string expectedGid = "1";
LabelId indexingLabel = LabelId::FromInt(2);
std::vector<LabelId> labels = {LabelId::FromInt(3), LabelId::FromInt(4)};
std::map<PropertyId, PropertyValue> properties = {{PropertyId::FromInt(5), PropertyValue("5")},
{PropertyId::FromInt(6), PropertyValue("6")}};
PropertyStore propertyStore;
propertyStore.InitProperties(properties);
std::string serializedVertex =
memgraph::utils::SerializeVertexAsValueForLabelPropertyIndex(indexingLabel, labels, propertyStore);
ASSERT_EQ(memgraph::utils::DeserializePropertiesFromLabelPropertyIndexStorage(serializedVertex).StringBuffer(),
propertyStore.StringBuffer());
}
TEST(RocksDbSerDeSuite, ExtractGidFromLabelPropertyIndexStorageKey) {
auto gid = Gid::FromInt(1);
LabelId label = LabelId::FromInt(2);
PropertyId property = PropertyId::FromInt(3);
std::string serializedVertex = memgraph::utils::SerializeVertexAsKeyForLabelPropertyIndex(label, property, gid);
ASSERT_EQ(memgraph::utils::ExtractGidFromLabelPropertyIndexStorage(serializedVertex), "1");
}
TEST(RocksDbSerDeSuite, ExtractGidFromUniqueConstraintStorageKey) {
std::string expectedGid = "1";
LabelId constraintLabel = LabelId::FromInt(2);
std::set<PropertyId> properties = {PropertyId::FromInt(3), PropertyId::FromInt(4)};
std::string serializedVertex =
memgraph::utils::SerializeVertexAsKeyForUniqueConstraint(constraintLabel, properties, expectedGid);
ASSERT_EQ(memgraph::utils::ExtractGidFromUniqueConstraintStorage(serializedVertex), expectedGid);
}
TEST(RocksDbSerDeSuite, DeserializeConstraintLabelFromUniqueConstraintStorage) {
std::string expectedGid = "1";
LabelId constraintLabel = LabelId::FromInt(2);
std::set<PropertyId> properties = {PropertyId::FromInt(3), PropertyId::FromInt(4)};
std::string serializedVertex =
memgraph::utils::SerializeVertexAsKeyForUniqueConstraint(constraintLabel, properties, expectedGid);
ASSERT_EQ(memgraph::utils::DeserializeConstraintLabelFromUniqueConstraintStorage(serializedVertex), constraintLabel);
}
TEST(RocksDbSerDeSuite, DeserializePropertiesFromUniqueConstraintStorage) {
std::string expectedGid = "1";
LabelId constraintLabel = LabelId::FromInt(2);
std::vector<LabelId> labels = {LabelId::FromInt(3), LabelId::FromInt(4)};
std::map<PropertyId, PropertyValue> properties = {{PropertyId::FromInt(5), PropertyValue("5")},
{PropertyId::FromInt(6), PropertyValue("6")}};
PropertyStore propertyStore;
propertyStore.InitProperties(properties);
std::string serializedVertex =
memgraph::utils::SerializeVertexAsValueForUniqueConstraint(constraintLabel, labels, propertyStore);
ASSERT_EQ(memgraph::utils::DeserializePropertiesFromUniqueConstraintStorage(serializedVertex).StringBuffer(),
propertyStore.StringBuffer());
}