// Copyright 2021 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 #include #include "storage/v2/constraints.hpp" #include "storage/v2/storage.hpp" const int kNumThreads = 8; #define ASSERT_OK(x) ASSERT_FALSE((x).HasError()) using storage::LabelId; using storage::PropertyId; using storage::PropertyValue; class StorageUniqueConstraints : public ::testing::Test { protected: StorageUniqueConstraints() : label(storage.NameToLabel("label")), prop1(storage.NameToProperty("prop1")), prop2(storage.NameToProperty("prop2")), prop3(storage.NameToProperty("prop3")) {} void SetUp() override { // Create initial vertices. auto acc = storage.Access(); // NOLINTNEXTLINE(modernize-loop-convert) for (int i = 0; i < kNumThreads; ++i) { auto vertex = acc.CreateVertex(); gids[i] = vertex.Gid(); } ASSERT_OK(acc.Commit()); } storage::Storage storage; LabelId label; PropertyId prop1; PropertyId prop2; PropertyId prop3; storage::Gid gids[kNumThreads]; }; void SetProperties(storage::Storage *storage, storage::Gid gid, const std::vector &properties, const std::vector &values, bool *commit_status) { ASSERT_EQ(properties.size(), values.size()); auto acc = storage->Access(); auto vertex = acc.FindVertex(gid, storage::View::OLD); ASSERT_TRUE(vertex); int value = 0; for (int iter = 0; iter < 40000; ++iter) { for (const auto &property : properties) { ASSERT_OK(vertex->SetProperty(property, PropertyValue(value++))); } } for (size_t i = 0; i < properties.size(); ++i) { ASSERT_OK(vertex->SetProperty(properties[i], values[i])); } *commit_status = !acc.Commit().HasError(); } void AddLabel(storage::Storage *storage, storage::Gid gid, LabelId label, bool *commit_status) { auto acc = storage->Access(); auto vertex = acc.FindVertex(gid, storage::View::OLD); ASSERT_TRUE(vertex); for (int iter = 0; iter < 40000; ++iter) { ASSERT_OK(vertex->AddLabel(label)); ASSERT_OK(vertex->RemoveLabel(label)); } ASSERT_OK(vertex->AddLabel(label)); *commit_status = !acc.Commit().HasError(); } TEST_F(StorageUniqueConstraints, ChangeProperties) { { auto res = storage.CreateUniqueConstraint(label, {prop1, prop2, prop3}); ASSERT_TRUE(res.HasValue()); ASSERT_EQ(res.GetValue(), storage::UniqueConstraints::CreationStatus::SUCCESS); } { auto acc = storage.Access(); // NOLINTNEXTLINE(modernize-loop-convert) for (int i = 0; i < kNumThreads; ++i) { auto vertex = acc.FindVertex(gids[i], storage::View::OLD); ASSERT_TRUE(vertex); ASSERT_OK(vertex->AddLabel(label)); } ASSERT_OK(acc.Commit()); } std::vector properties{prop1, prop2, prop3}; // There is fixed set of property values that is tried to be set to all // vertices in all iterations. { std::vector values{PropertyValue(1), PropertyValue(2), PropertyValue(3)}; for (int iter = 0; iter < 20; ++iter) { bool status[kNumThreads]; std::vector threads; threads.reserve(kNumThreads); for (int i = 0; i < kNumThreads; ++i) { threads.emplace_back(SetProperties, &storage, gids[i], properties, values, &status[i]); } int count_ok = 0; for (int i = 0; i < kNumThreads; ++i) { threads[i].join(); count_ok += status[i]; } ASSERT_EQ(count_ok, 1); } } // The same set of property values is tried to be set to all vertices in // every iteration, but each iteration uses a different set of values. { for (int iter = 0; iter < 20; ++iter) { bool status[kNumThreads]; std::vector values{PropertyValue(iter), PropertyValue(iter + 1), PropertyValue(iter + 2)}; std::vector threads; threads.reserve(kNumThreads); for (int i = 0; i < kNumThreads; ++i) { threads.emplace_back(SetProperties, &storage, gids[i], properties, values, &status[i]); } int count_ok = 0; for (int i = 0; i < kNumThreads; ++i) { threads[i].join(); count_ok += status[i]; } ASSERT_EQ(count_ok, 1); } } // Different property values are being set to vertices. In that case, all // transactions should succeed. { int value = 1000000; for (int iter = 0; iter < 20; ++iter) { bool status[kNumThreads]; std::vector threads; threads.reserve(kNumThreads); for (int i = 0; i < kNumThreads; ++i) { std::vector values{PropertyValue(value++), PropertyValue(value++), PropertyValue(value++)}; threads.emplace_back(SetProperties, &storage, gids[i], properties, values, &status[i]); } int count_ok = 0; for (int i = 0; i < kNumThreads; ++i) { threads[i].join(); count_ok += status[i]; } ASSERT_EQ(count_ok, kNumThreads); } } } TEST_F(StorageUniqueConstraints, ChangeLabels) { { auto res = storage.CreateUniqueConstraint(label, {prop1, prop2, prop3}); ASSERT_TRUE(res.HasValue()); ASSERT_EQ(res.GetValue(), storage::UniqueConstraints::CreationStatus::SUCCESS); } // In the first part of the test, each transaction tries to add the same label // to different vertices, assuming that each vertex had the same set of // property values initially. In that case, exactly one transaction should // succeed, as the others should result with constraint violation. { auto acc = storage.Access(); // NOLINTNEXTLINE(modernize-loop-convert) for (int i = 0; i < kNumThreads; ++i) { auto vertex = acc.FindVertex(gids[i], storage::View::OLD); ASSERT_TRUE(vertex); ASSERT_OK(vertex->SetProperty(prop1, PropertyValue(1))); ASSERT_OK(vertex->SetProperty(prop2, PropertyValue(2))); ASSERT_OK(vertex->SetProperty(prop3, PropertyValue(3))); } ASSERT_OK(acc.Commit()); } for (int iter = 0; iter < 20; ++iter) { // Clear labels. { auto acc = storage.Access(); // NOLINTNEXTLINE(modernize-loop-convert) for (int i = 0; i < kNumThreads; ++i) { auto vertex = acc.FindVertex(gids[i], storage::View::OLD); ASSERT_TRUE(vertex); ASSERT_OK(vertex->RemoveLabel(label)); } ASSERT_OK(acc.Commit()); } bool status[kNumThreads]; std::vector threads; threads.reserve(kNumThreads); // NOLINTNEXTLINE(modernize-loop-convert) for (int i = 0; i < kNumThreads; ++i) { threads.emplace_back(AddLabel, &storage, gids[i], label, &status[i]); } int count_ok = 0; // NOLINTNEXTLINE(modernize-loop-convert) for (int i = 0; i < kNumThreads; ++i) { threads[i].join(); count_ok += status[i]; } ASSERT_EQ(count_ok, 1); } // In the second part of the test, it's assumed that all vertices has // different set of property values initially. In that case, all transactions // should succeed. { auto acc = storage.Access(); // NOLINTNEXTLINE(modernize-loop-convert) for (int i = 0; i < kNumThreads; ++i) { auto vertex = acc.FindVertex(gids[i], storage::View::OLD); ASSERT_TRUE(vertex); ASSERT_OK(vertex->SetProperty(prop1, PropertyValue(3 * i))); ASSERT_OK(vertex->SetProperty(prop2, PropertyValue(3 * i + 1))); ASSERT_OK(vertex->SetProperty(prop3, PropertyValue(3 * i + 2))); } ASSERT_OK(acc.Commit()); } for (int iter = 0; iter < 20; ++iter) { // Clear labels. { auto acc = storage.Access(); // NOLINTNEXTLINE(modernize-loop-convert) for (int i = 0; i < kNumThreads; ++i) { auto vertex = acc.FindVertex(gids[i], storage::View::OLD); ASSERT_TRUE(vertex); ASSERT_OK(vertex->RemoveLabel(label)); } ASSERT_OK(acc.Commit()); } bool status[kNumThreads]; std::vector threads; threads.reserve(kNumThreads); for (int i = 0; i < kNumThreads; ++i) { threads.emplace_back(AddLabel, &storage, gids[i], label, &status[i]); } int count_ok = 0; for (int i = 0; i < kNumThreads; ++i) { threads[i].join(); count_ok += status[i]; } ASSERT_EQ(count_ok, kNumThreads); } } int main(int argc, char **argv) { ::testing::InitGoogleTest(&argc, argv); return RUN_ALL_TESTS(); }