memgraph/tests/unit/storage_v2.cpp
Matej Ferencevic d17d463884 Add NONEXISTENT_OBJECT error to storage v2
Summary:
Previously, when accessing the labels/properties/edges of a vertex/edge that
was just created the NEW view would correctly display the change, but the OLD
view would be invalid and would crash the database. With this change the OLD
view of a freshly created vertex/edge won't cause a crash, but will instead
report an error.

Reviewers: teon.banek

Reviewed By: teon.banek

Subscribers: pullbot

Differential Revision: https://phabricator.memgraph.io/D2549
2019-11-15 10:01:22 +01:00

2336 lines
79 KiB
C++

#include <gmock/gmock.h>
#include <gtest/gtest.h>
#include <limits>
#include "storage/v2/storage.hpp"
using testing::UnorderedElementsAre;
size_t CountVertices(storage::Storage::Accessor *storage_accessor,
storage::View view) {
auto vertices = storage_accessor->Vertices(view);
size_t count = 0U;
for (auto it = vertices.begin(); it != vertices.end(); ++it) ++count;
return count;
}
// NOLINTNEXTLINE(hicpp-special-member-functions)
TEST(StorageV2, Commit) {
storage::Storage store;
storage::Gid gid =
storage::Gid::FromUint(std::numeric_limits<uint64_t>::max());
{
auto acc = store.Access();
auto vertex = acc.CreateVertex();
gid = vertex.Gid();
ASSERT_FALSE(acc.FindVertex(gid, storage::View::OLD).has_value());
EXPECT_EQ(CountVertices(&acc, storage::View::OLD), 0U);
ASSERT_TRUE(acc.FindVertex(gid, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc, storage::View::NEW), 1U);
ASSERT_FALSE(acc.Commit().HasError());
}
{
auto acc = store.Access();
ASSERT_TRUE(acc.FindVertex(gid, storage::View::OLD).has_value());
EXPECT_EQ(CountVertices(&acc, storage::View::OLD), 1U);
ASSERT_TRUE(acc.FindVertex(gid, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc, storage::View::NEW), 1U);
acc.Abort();
}
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::NEW);
ASSERT_TRUE(vertex);
auto res = acc.DeleteVertex(&*vertex);
ASSERT_FALSE(res.HasError());
EXPECT_EQ(CountVertices(&acc, storage::View::OLD), 1U);
EXPECT_EQ(CountVertices(&acc, storage::View::NEW), 0U);
ASSERT_FALSE(acc.Commit().HasError());
}
{
auto acc = store.Access();
ASSERT_FALSE(acc.FindVertex(gid, storage::View::OLD).has_value());
EXPECT_EQ(CountVertices(&acc, storage::View::OLD), 0U);
ASSERT_FALSE(acc.FindVertex(gid, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc, storage::View::NEW), 0U);
acc.Abort();
}
}
// NOLINTNEXTLINE(hicpp-special-member-functions)
TEST(StorageV2, Abort) {
storage::Storage store;
storage::Gid gid =
storage::Gid::FromUint(std::numeric_limits<uint64_t>::max());
{
auto acc = store.Access();
auto vertex = acc.CreateVertex();
gid = vertex.Gid();
ASSERT_FALSE(acc.FindVertex(gid, storage::View::OLD).has_value());
EXPECT_EQ(CountVertices(&acc, storage::View::OLD), 0U);
ASSERT_TRUE(acc.FindVertex(gid, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc, storage::View::NEW), 1U);
acc.Abort();
}
{
auto acc = store.Access();
ASSERT_FALSE(acc.FindVertex(gid, storage::View::OLD).has_value());
EXPECT_EQ(CountVertices(&acc, storage::View::OLD), 0U);
ASSERT_FALSE(acc.FindVertex(gid, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc, storage::View::NEW), 0U);
acc.Abort();
}
}
// NOLINTNEXTLINE(hicpp-special-member-functions)
TEST(StorageV2, AdvanceCommandCommit) {
storage::Storage store;
storage::Gid gid1 =
storage::Gid::FromUint(std::numeric_limits<uint64_t>::max());
storage::Gid gid2 =
storage::Gid::FromUint(std::numeric_limits<uint64_t>::max());
{
auto acc = store.Access();
auto vertex1 = acc.CreateVertex();
gid1 = vertex1.Gid();
ASSERT_FALSE(acc.FindVertex(gid1, storage::View::OLD).has_value());
EXPECT_EQ(CountVertices(&acc, storage::View::OLD), 0U);
ASSERT_TRUE(acc.FindVertex(gid1, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc, storage::View::NEW), 1U);
acc.AdvanceCommand();
auto vertex2 = acc.CreateVertex();
gid2 = vertex2.Gid();
ASSERT_FALSE(acc.FindVertex(gid2, storage::View::OLD).has_value());
EXPECT_EQ(CountVertices(&acc, storage::View::OLD), 1U);
ASSERT_TRUE(acc.FindVertex(gid2, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc, storage::View::NEW), 2U);
ASSERT_TRUE(acc.FindVertex(gid1, storage::View::OLD).has_value());
ASSERT_TRUE(acc.FindVertex(gid1, storage::View::NEW).has_value());
ASSERT_FALSE(acc.Commit().HasError());
}
{
auto acc = store.Access();
ASSERT_TRUE(acc.FindVertex(gid1, storage::View::OLD).has_value());
ASSERT_TRUE(acc.FindVertex(gid1, storage::View::NEW).has_value());
ASSERT_TRUE(acc.FindVertex(gid2, storage::View::OLD).has_value());
ASSERT_TRUE(acc.FindVertex(gid2, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc, storage::View::OLD), 2U);
EXPECT_EQ(CountVertices(&acc, storage::View::NEW), 2U);
acc.Abort();
}
}
// NOLINTNEXTLINE(hicpp-special-member-functions)
TEST(StorageV2, AdvanceCommandAbort) {
storage::Storage store;
storage::Gid gid1 =
storage::Gid::FromUint(std::numeric_limits<uint64_t>::max());
storage::Gid gid2 =
storage::Gid::FromUint(std::numeric_limits<uint64_t>::max());
{
auto acc = store.Access();
auto vertex1 = acc.CreateVertex();
gid1 = vertex1.Gid();
ASSERT_FALSE(acc.FindVertex(gid1, storage::View::OLD).has_value());
EXPECT_EQ(CountVertices(&acc, storage::View::OLD), 0U);
ASSERT_TRUE(acc.FindVertex(gid1, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc, storage::View::NEW), 1U);
acc.AdvanceCommand();
auto vertex2 = acc.CreateVertex();
gid2 = vertex2.Gid();
ASSERT_FALSE(acc.FindVertex(gid2, storage::View::OLD).has_value());
EXPECT_EQ(CountVertices(&acc, storage::View::OLD), 1U);
ASSERT_TRUE(acc.FindVertex(gid2, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc, storage::View::NEW), 2U);
ASSERT_TRUE(acc.FindVertex(gid1, storage::View::OLD).has_value());
ASSERT_TRUE(acc.FindVertex(gid1, storage::View::NEW).has_value());
acc.Abort();
}
{
auto acc = store.Access();
ASSERT_FALSE(acc.FindVertex(gid1, storage::View::OLD).has_value());
ASSERT_FALSE(acc.FindVertex(gid1, storage::View::NEW).has_value());
ASSERT_FALSE(acc.FindVertex(gid2, storage::View::OLD).has_value());
ASSERT_FALSE(acc.FindVertex(gid2, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc, storage::View::OLD), 0U);
EXPECT_EQ(CountVertices(&acc, storage::View::NEW), 0U);
acc.Abort();
}
}
// NOLINTNEXTLINE(hicpp-special-member-functions)
TEST(StorageV2, SnapshotIsolation) {
storage::Storage store;
auto acc1 = store.Access();
auto acc2 = store.Access();
auto vertex = acc1.CreateVertex();
auto gid = vertex.Gid();
ASSERT_FALSE(acc2.FindVertex(gid, storage::View::OLD).has_value());
EXPECT_EQ(CountVertices(&acc1, storage::View::OLD), 0U);
EXPECT_EQ(CountVertices(&acc2, storage::View::OLD), 0U);
ASSERT_FALSE(acc2.FindVertex(gid, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc1, storage::View::NEW), 1U);
EXPECT_EQ(CountVertices(&acc2, storage::View::NEW), 0U);
ASSERT_FALSE(acc1.Commit().HasError());
ASSERT_FALSE(acc2.FindVertex(gid, storage::View::OLD).has_value());
EXPECT_EQ(CountVertices(&acc2, storage::View::OLD), 0U);
ASSERT_FALSE(acc2.FindVertex(gid, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc2, storage::View::NEW), 0U);
acc2.Abort();
auto acc3 = store.Access();
ASSERT_TRUE(acc3.FindVertex(gid, storage::View::OLD).has_value());
EXPECT_EQ(CountVertices(&acc3, storage::View::OLD), 1U);
ASSERT_TRUE(acc3.FindVertex(gid, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc3, storage::View::NEW), 1U);
acc3.Abort();
}
// NOLINTNEXTLINE(hicpp-special-member-functions)
TEST(StorageV2, AccessorMove) {
storage::Storage store;
storage::Gid gid =
storage::Gid::FromUint(std::numeric_limits<uint64_t>::max());
{
auto acc = store.Access();
auto vertex = acc.CreateVertex();
gid = vertex.Gid();
ASSERT_FALSE(acc.FindVertex(gid, storage::View::OLD).has_value());
EXPECT_EQ(CountVertices(&acc, storage::View::OLD), 0U);
ASSERT_TRUE(acc.FindVertex(gid, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc, storage::View::NEW), 1U);
storage::Storage::Accessor moved(std::move(acc));
ASSERT_FALSE(moved.FindVertex(gid, storage::View::OLD).has_value());
EXPECT_EQ(CountVertices(&moved, storage::View::OLD), 0U);
ASSERT_TRUE(moved.FindVertex(gid, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&moved, storage::View::NEW), 1U);
ASSERT_FALSE(moved.Commit().HasError());
}
{
auto acc = store.Access();
ASSERT_TRUE(acc.FindVertex(gid, storage::View::OLD).has_value());
EXPECT_EQ(CountVertices(&acc, storage::View::OLD), 1U);
ASSERT_TRUE(acc.FindVertex(gid, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc, storage::View::NEW), 1U);
acc.Abort();
}
}
// NOLINTNEXTLINE(hicpp-special-member-functions)
TEST(StorageV2, VertexDeleteCommit) {
storage::Storage store;
storage::Gid gid =
storage::Gid::FromUint(std::numeric_limits<uint64_t>::max());
auto acc1 = store.Access(); // read transaction
auto acc2 = store.Access(); // write transaction
// Create the vertex in transaction 2
{
auto vertex = acc2.CreateVertex();
gid = vertex.Gid();
ASSERT_FALSE(acc2.FindVertex(gid, storage::View::OLD).has_value());
EXPECT_EQ(CountVertices(&acc2, storage::View::OLD), 0U);
ASSERT_TRUE(acc2.FindVertex(gid, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc2, storage::View::NEW), 1U);
ASSERT_FALSE(acc2.Commit().HasError());
}
auto acc3 = store.Access(); // read transaction
auto acc4 = store.Access(); // write transaction
// Check whether the vertex exists in transaction 1
ASSERT_FALSE(acc1.FindVertex(gid, storage::View::OLD).has_value());
EXPECT_EQ(CountVertices(&acc1, storage::View::OLD), 0U);
ASSERT_FALSE(acc1.FindVertex(gid, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc1, storage::View::NEW), 0U);
// Check whether the vertex exists in transaction 3
ASSERT_TRUE(acc3.FindVertex(gid, storage::View::OLD).has_value());
EXPECT_EQ(CountVertices(&acc3, storage::View::OLD), 1U);
ASSERT_TRUE(acc3.FindVertex(gid, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc3, storage::View::NEW), 1U);
// Delete the vertex in transaction 4
{
auto vertex = acc4.FindVertex(gid, storage::View::NEW);
ASSERT_TRUE(vertex);
EXPECT_EQ(CountVertices(&acc4, storage::View::OLD), 1U);
EXPECT_EQ(CountVertices(&acc4, storage::View::NEW), 1U);
auto res = acc4.DeleteVertex(&*vertex);
ASSERT_TRUE(res.HasValue());
EXPECT_EQ(CountVertices(&acc4, storage::View::OLD), 1U);
EXPECT_EQ(CountVertices(&acc4, storage::View::NEW), 0U);
ASSERT_FALSE(acc4.Commit().HasError());
}
auto acc5 = store.Access(); // read transaction
// Check whether the vertex exists in transaction 1
ASSERT_FALSE(acc1.FindVertex(gid, storage::View::OLD).has_value());
EXPECT_EQ(CountVertices(&acc1, storage::View::OLD), 0U);
ASSERT_FALSE(acc1.FindVertex(gid, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc1, storage::View::NEW), 0U);
// Check whether the vertex exists in transaction 3
ASSERT_TRUE(acc3.FindVertex(gid, storage::View::OLD).has_value());
EXPECT_EQ(CountVertices(&acc3, storage::View::OLD), 1U);
ASSERT_TRUE(acc3.FindVertex(gid, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc3, storage::View::NEW), 1U);
// Check whether the vertex exists in transaction 5
ASSERT_FALSE(acc5.FindVertex(gid, storage::View::OLD).has_value());
EXPECT_EQ(CountVertices(&acc5, storage::View::OLD), 0U);
ASSERT_FALSE(acc5.FindVertex(gid, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc5, storage::View::NEW), 0U);
}
// NOLINTNEXTLINE(hicpp-special-member-functions)
TEST(StorageV2, VertexDeleteAbort) {
storage::Storage store;
storage::Gid gid =
storage::Gid::FromUint(std::numeric_limits<uint64_t>::max());
auto acc1 = store.Access(); // read transaction
auto acc2 = store.Access(); // write transaction
// Create the vertex in transaction 2
{
auto vertex = acc2.CreateVertex();
gid = vertex.Gid();
ASSERT_FALSE(acc2.FindVertex(gid, storage::View::OLD).has_value());
EXPECT_EQ(CountVertices(&acc2, storage::View::OLD), 0U);
ASSERT_TRUE(acc2.FindVertex(gid, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc2, storage::View::NEW), 1U);
ASSERT_FALSE(acc2.Commit().HasError());
}
auto acc3 = store.Access(); // read transaction
auto acc4 = store.Access(); // write transaction (aborted)
// Check whether the vertex exists in transaction 1
ASSERT_FALSE(acc1.FindVertex(gid, storage::View::OLD).has_value());
EXPECT_EQ(CountVertices(&acc1, storage::View::OLD), 0U);
ASSERT_FALSE(acc1.FindVertex(gid, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc1, storage::View::NEW), 0U);
// Check whether the vertex exists in transaction 3
ASSERT_TRUE(acc3.FindVertex(gid, storage::View::OLD).has_value());
EXPECT_EQ(CountVertices(&acc3, storage::View::OLD), 1U);
ASSERT_TRUE(acc3.FindVertex(gid, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc3, storage::View::NEW), 1U);
// Delete the vertex in transaction 4, but abort the transaction
{
auto vertex = acc4.FindVertex(gid, storage::View::NEW);
ASSERT_TRUE(vertex);
EXPECT_EQ(CountVertices(&acc4, storage::View::OLD), 1U);
EXPECT_EQ(CountVertices(&acc4, storage::View::NEW), 1U);
auto res = acc4.DeleteVertex(&*vertex);
ASSERT_TRUE(res.HasValue());
EXPECT_EQ(CountVertices(&acc4, storage::View::OLD), 1U);
EXPECT_EQ(CountVertices(&acc4, storage::View::NEW), 0U);
acc4.Abort();
}
auto acc5 = store.Access(); // read transaction
auto acc6 = store.Access(); // write transaction
// Check whether the vertex exists in transaction 1
ASSERT_FALSE(acc1.FindVertex(gid, storage::View::OLD).has_value());
EXPECT_EQ(CountVertices(&acc1, storage::View::OLD), 0U);
ASSERT_FALSE(acc1.FindVertex(gid, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc1, storage::View::NEW), 0U);
// Check whether the vertex exists in transaction 3
ASSERT_TRUE(acc3.FindVertex(gid, storage::View::OLD).has_value());
EXPECT_EQ(CountVertices(&acc3, storage::View::OLD), 1U);
ASSERT_TRUE(acc3.FindVertex(gid, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc3, storage::View::NEW), 1U);
// Check whether the vertex exists in transaction 5
ASSERT_TRUE(acc5.FindVertex(gid, storage::View::OLD).has_value());
EXPECT_EQ(CountVertices(&acc5, storage::View::OLD), 1U);
ASSERT_TRUE(acc5.FindVertex(gid, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc5, storage::View::NEW), 1U);
// Delete the vertex in transaction 6
{
auto vertex = acc6.FindVertex(gid, storage::View::NEW);
ASSERT_TRUE(vertex);
EXPECT_EQ(CountVertices(&acc6, storage::View::OLD), 1U);
EXPECT_EQ(CountVertices(&acc6, storage::View::NEW), 1U);
auto res = acc6.DeleteVertex(&*vertex);
ASSERT_TRUE(res.HasValue());
EXPECT_EQ(CountVertices(&acc6, storage::View::OLD), 1U);
EXPECT_EQ(CountVertices(&acc6, storage::View::NEW), 0U);
ASSERT_FALSE(acc6.Commit().HasError());
}
auto acc7 = store.Access(); // read transaction
// Check whether the vertex exists in transaction 1
ASSERT_FALSE(acc1.FindVertex(gid, storage::View::OLD).has_value());
EXPECT_EQ(CountVertices(&acc1, storage::View::OLD), 0U);
ASSERT_FALSE(acc1.FindVertex(gid, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc1, storage::View::NEW), 0U);
// Check whether the vertex exists in transaction 3
ASSERT_TRUE(acc3.FindVertex(gid, storage::View::OLD).has_value());
EXPECT_EQ(CountVertices(&acc3, storage::View::OLD), 1U);
ASSERT_TRUE(acc3.FindVertex(gid, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc3, storage::View::NEW), 1U);
// Check whether the vertex exists in transaction 5
ASSERT_TRUE(acc5.FindVertex(gid, storage::View::OLD).has_value());
EXPECT_EQ(CountVertices(&acc5, storage::View::OLD), 1U);
ASSERT_TRUE(acc5.FindVertex(gid, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc5, storage::View::NEW), 1U);
// Check whether the vertex exists in transaction 7
ASSERT_FALSE(acc7.FindVertex(gid, storage::View::OLD).has_value());
EXPECT_EQ(CountVertices(&acc7, storage::View::OLD), 0U);
ASSERT_FALSE(acc7.FindVertex(gid, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc7, storage::View::NEW), 0U);
// Commit all accessors
ASSERT_FALSE(acc1.Commit().HasError());
ASSERT_FALSE(acc3.Commit().HasError());
ASSERT_FALSE(acc5.Commit().HasError());
ASSERT_FALSE(acc7.Commit().HasError());
}
// NOLINTNEXTLINE(hicpp-special-member-functions)
TEST(StorageV2, VertexDeleteSerializationError) {
storage::Storage store;
storage::Gid gid =
storage::Gid::FromUint(std::numeric_limits<uint64_t>::max());
// Create vertex
{
auto acc = store.Access();
auto vertex = acc.CreateVertex();
gid = vertex.Gid();
ASSERT_FALSE(acc.Commit().HasError());
}
auto acc1 = store.Access();
auto acc2 = store.Access();
// Delete vertex in accessor 1
{
auto vertex = acc1.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
EXPECT_EQ(CountVertices(&acc1, storage::View::OLD), 1U);
EXPECT_EQ(CountVertices(&acc1, storage::View::NEW), 1U);
{
auto res = acc1.DeleteVertex(&*vertex);
ASSERT_TRUE(res.HasValue());
ASSERT_TRUE(res.GetValue());
EXPECT_EQ(CountVertices(&acc1, storage::View::OLD), 1U);
EXPECT_EQ(CountVertices(&acc1, storage::View::NEW), 0U);
}
{
auto res = acc1.DeleteVertex(&*vertex);
ASSERT_TRUE(res.HasValue());
ASSERT_FALSE(res.GetValue());
EXPECT_EQ(CountVertices(&acc1, storage::View::OLD), 1U);
EXPECT_EQ(CountVertices(&acc1, storage::View::NEW), 0U);
}
}
// Delete vertex in accessor 2
{
auto vertex = acc2.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
EXPECT_EQ(CountVertices(&acc2, storage::View::OLD), 1U);
EXPECT_EQ(CountVertices(&acc2, storage::View::NEW), 1U);
auto res = acc2.DeleteVertex(&*vertex);
ASSERT_TRUE(res.HasError());
ASSERT_EQ(res.GetError(), storage::Error::SERIALIZATION_ERROR);
EXPECT_EQ(CountVertices(&acc2, storage::View::OLD), 1U);
EXPECT_EQ(CountVertices(&acc2, storage::View::NEW), 1U);
}
// Finalize both accessors
ASSERT_FALSE(acc1.Commit().HasError());
acc2.Abort();
// Check whether the vertex exists
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::OLD);
ASSERT_FALSE(vertex);
EXPECT_EQ(CountVertices(&acc, storage::View::OLD), 0U);
EXPECT_EQ(CountVertices(&acc, storage::View::NEW), 0U);
ASSERT_FALSE(acc.Commit().HasError());
}
}
// NOLINTNEXTLINE(hicpp-special-member-functions)
TEST(StorageV2, VertexDeleteSpecialCases) {
storage::Storage store;
storage::Gid gid1 =
storage::Gid::FromUint(std::numeric_limits<uint64_t>::max());
storage::Gid gid2 =
storage::Gid::FromUint(std::numeric_limits<uint64_t>::max());
// Create vertex and delete it in the same transaction, but abort the
// transaction
{
auto acc = store.Access();
auto vertex = acc.CreateVertex();
gid1 = vertex.Gid();
ASSERT_FALSE(acc.FindVertex(gid1, storage::View::OLD).has_value());
EXPECT_EQ(CountVertices(&acc, storage::View::OLD), 0U);
ASSERT_TRUE(acc.FindVertex(gid1, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc, storage::View::NEW), 1U);
auto res = acc.DeleteVertex(&vertex);
ASSERT_TRUE(res.HasValue());
ASSERT_TRUE(res.GetValue());
EXPECT_EQ(CountVertices(&acc, storage::View::OLD), 0U);
EXPECT_EQ(CountVertices(&acc, storage::View::NEW), 0U);
acc.Abort();
}
// Create vertex and delete it in the same transaction
{
auto acc = store.Access();
auto vertex = acc.CreateVertex();
gid2 = vertex.Gid();
ASSERT_FALSE(acc.FindVertex(gid2, storage::View::OLD).has_value());
EXPECT_EQ(CountVertices(&acc, storage::View::OLD), 0U);
ASSERT_TRUE(acc.FindVertex(gid2, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc, storage::View::NEW), 1U);
auto res = acc.DeleteVertex(&vertex);
ASSERT_TRUE(res.HasValue());
ASSERT_TRUE(res.GetValue());
EXPECT_EQ(CountVertices(&acc, storage::View::OLD), 0U);
EXPECT_EQ(CountVertices(&acc, storage::View::NEW), 0U);
ASSERT_FALSE(acc.Commit().HasError());
}
// Check whether the vertices exist
{
auto acc = store.Access();
ASSERT_FALSE(acc.FindVertex(gid1, storage::View::OLD).has_value());
ASSERT_FALSE(acc.FindVertex(gid1, storage::View::NEW).has_value());
ASSERT_FALSE(acc.FindVertex(gid2, storage::View::OLD).has_value());
ASSERT_FALSE(acc.FindVertex(gid2, storage::View::NEW).has_value());
EXPECT_EQ(CountVertices(&acc, storage::View::OLD), 0U);
EXPECT_EQ(CountVertices(&acc, storage::View::NEW), 0U);
acc.Abort();
}
}
// NOLINTNEXTLINE(hicpp-special-member-functions)
TEST(StorageV2, VertexDeleteLabel) {
storage::Storage store;
storage::Gid gid =
storage::Gid::FromUint(std::numeric_limits<uint64_t>::max());
// Create the vertex
{
auto acc = store.Access();
auto vertex = acc.CreateVertex();
gid = vertex.Gid();
ASSERT_FALSE(acc.FindVertex(gid, storage::View::OLD).has_value());
ASSERT_TRUE(acc.FindVertex(gid, storage::View::NEW).has_value());
ASSERT_FALSE(acc.Commit().HasError());
}
// Add label, delete the vertex and check the label API (same command)
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::NEW);
ASSERT_TRUE(vertex);
auto label = acc.NameToLabel("label5");
// Check whether label 5 exists
ASSERT_FALSE(vertex->HasLabel(label, storage::View::OLD).GetValue());
ASSERT_FALSE(vertex->HasLabel(label, storage::View::NEW).GetValue());
ASSERT_EQ(vertex->Labels(storage::View::OLD)->size(), 0);
ASSERT_EQ(vertex->Labels(storage::View::NEW)->size(), 0);
// Add label 5
ASSERT_TRUE(vertex->AddLabel(label).GetValue());
// Check whether label 5 exists
ASSERT_FALSE(vertex->HasLabel(label, storage::View::OLD).GetValue());
ASSERT_TRUE(vertex->HasLabel(label, storage::View::NEW).GetValue());
ASSERT_EQ(vertex->Labels(storage::View::OLD)->size(), 0);
{
auto labels = vertex->Labels(storage::View::NEW).GetValue();
ASSERT_EQ(labels.size(), 1);
ASSERT_EQ(labels[0], label);
}
// Delete the vertex
ASSERT_TRUE(acc.DeleteVertex(&*vertex).GetValue());
// Check whether label 5 exists
ASSERT_FALSE(vertex->HasLabel(label, storage::View::OLD).GetValue());
ASSERT_EQ(vertex->HasLabel(label, storage::View::NEW).GetError(),
storage::Error::DELETED_OBJECT);
ASSERT_EQ(vertex->Labels(storage::View::OLD)->size(), 0);
ASSERT_EQ(vertex->Labels(storage::View::NEW).GetError(),
storage::Error::DELETED_OBJECT);
// Try to add the label
{
auto ret = vertex->AddLabel(label);
ASSERT_TRUE(ret.HasError());
ASSERT_EQ(ret.GetError(), storage::Error::DELETED_OBJECT);
}
// Try to remove the label
{
auto ret = vertex->RemoveLabel(label);
ASSERT_TRUE(ret.HasError());
ASSERT_EQ(ret.GetError(), storage::Error::DELETED_OBJECT);
}
acc.Abort();
}
// Add label, delete the vertex and check the label API (different command)
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::NEW);
ASSERT_TRUE(vertex);
auto label = acc.NameToLabel("label5");
// Check whether label 5 exists
ASSERT_FALSE(vertex->HasLabel(label, storage::View::OLD).GetValue());
ASSERT_FALSE(vertex->HasLabel(label, storage::View::NEW).GetValue());
ASSERT_EQ(vertex->Labels(storage::View::OLD)->size(), 0);
ASSERT_EQ(vertex->Labels(storage::View::NEW)->size(), 0);
// Add label 5
ASSERT_TRUE(vertex->AddLabel(label).GetValue());
// Check whether label 5 exists
ASSERT_FALSE(vertex->HasLabel(label, storage::View::OLD).GetValue());
ASSERT_TRUE(vertex->HasLabel(label, storage::View::NEW).GetValue());
ASSERT_EQ(vertex->Labels(storage::View::OLD)->size(), 0);
{
auto labels = vertex->Labels(storage::View::NEW).GetValue();
ASSERT_EQ(labels.size(), 1);
ASSERT_EQ(labels[0], label);
}
// Advance command
acc.AdvanceCommand();
// Check whether label 5 exists
ASSERT_TRUE(vertex->HasLabel(label, storage::View::OLD).GetValue());
ASSERT_TRUE(vertex->HasLabel(label, storage::View::NEW).GetValue());
{
auto labels = vertex->Labels(storage::View::OLD).GetValue();
ASSERT_EQ(labels.size(), 1);
ASSERT_EQ(labels[0], label);
}
{
auto labels = vertex->Labels(storage::View::NEW).GetValue();
ASSERT_EQ(labels.size(), 1);
ASSERT_EQ(labels[0], label);
}
// Delete the vertex
ASSERT_TRUE(acc.DeleteVertex(&*vertex).GetValue());
// Check whether label 5 exists
ASSERT_TRUE(vertex->HasLabel(label, storage::View::OLD).GetValue());
ASSERT_EQ(vertex->HasLabel(label, storage::View::NEW).GetError(),
storage::Error::DELETED_OBJECT);
{
auto labels = vertex->Labels(storage::View::OLD).GetValue();
ASSERT_EQ(labels.size(), 1);
ASSERT_EQ(labels[0], label);
}
ASSERT_EQ(vertex->Labels(storage::View::NEW).GetError(),
storage::Error::DELETED_OBJECT);
// Advance command
acc.AdvanceCommand();
// Check whether label 5 exists
ASSERT_EQ(vertex->HasLabel(label, storage::View::OLD).GetError(),
storage::Error::DELETED_OBJECT);
ASSERT_EQ(vertex->HasLabel(label, storage::View::NEW).GetError(),
storage::Error::DELETED_OBJECT);
ASSERT_EQ(vertex->Labels(storage::View::OLD).GetError(),
storage::Error::DELETED_OBJECT);
ASSERT_EQ(vertex->Labels(storage::View::NEW).GetError(),
storage::Error::DELETED_OBJECT);
// Try to add the label
{
auto ret = vertex->AddLabel(label);
ASSERT_TRUE(ret.HasError());
ASSERT_EQ(ret.GetError(), storage::Error::DELETED_OBJECT);
}
// Try to remove the label
{
auto ret = vertex->RemoveLabel(label);
ASSERT_TRUE(ret.HasError());
ASSERT_EQ(ret.GetError(), storage::Error::DELETED_OBJECT);
}
acc.Abort();
}
}
// NOLINTNEXTLINE(hicpp-special-member-functions)
TEST(StorageV2, VertexDeleteProperty) {
storage::Storage store;
storage::Gid gid =
storage::Gid::FromUint(std::numeric_limits<uint64_t>::max());
// Create the vertex
{
auto acc = store.Access();
auto vertex = acc.CreateVertex();
gid = vertex.Gid();
ASSERT_FALSE(acc.FindVertex(gid, storage::View::OLD).has_value());
ASSERT_TRUE(acc.FindVertex(gid, storage::View::NEW).has_value());
ASSERT_FALSE(acc.Commit().HasError());
}
// Set property, delete the vertex and check the property API (same command)
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::NEW);
ASSERT_TRUE(vertex);
auto property = acc.NameToProperty("property5");
// Check whether property 5 exists
ASSERT_TRUE(vertex->GetProperty(property, storage::View::OLD)->IsNull());
ASSERT_TRUE(vertex->GetProperty(property, storage::View::NEW)->IsNull());
ASSERT_EQ(vertex->Properties(storage::View::OLD)->size(), 0);
ASSERT_EQ(vertex->Properties(storage::View::NEW)->size(), 0);
// Set property 5 to "nandare"
ASSERT_TRUE(vertex->SetProperty(property, storage::PropertyValue("nandare"))
.GetValue());
// Check whether property 5 exists
ASSERT_TRUE(vertex->GetProperty(property, storage::View::OLD)->IsNull());
ASSERT_EQ(vertex->GetProperty(property, storage::View::NEW)->ValueString(),
"nandare");
ASSERT_EQ(vertex->Properties(storage::View::OLD)->size(), 0);
{
auto properties = vertex->Properties(storage::View::NEW).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "nandare");
}
// Delete the vertex
ASSERT_TRUE(acc.DeleteVertex(&*vertex).GetValue());
// Check whether label 5 exists
ASSERT_TRUE(vertex->GetProperty(property, storage::View::OLD)->IsNull());
ASSERT_EQ(vertex->GetProperty(property, storage::View::NEW).GetError(),
storage::Error::DELETED_OBJECT);
ASSERT_EQ(vertex->Properties(storage::View::OLD)->size(), 0);
ASSERT_EQ(vertex->Properties(storage::View::NEW).GetError(),
storage::Error::DELETED_OBJECT);
// Try to set the property
{
auto ret =
vertex->SetProperty(property, storage::PropertyValue("haihai"));
ASSERT_TRUE(ret.HasError());
ASSERT_EQ(ret.GetError(), storage::Error::DELETED_OBJECT);
}
acc.Abort();
}
// Set property, delete the vertex and check the property API (different
// command)
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::NEW);
ASSERT_TRUE(vertex);
auto property = acc.NameToProperty("property5");
// Check whether property 5 exists
ASSERT_TRUE(vertex->GetProperty(property, storage::View::OLD)->IsNull());
ASSERT_TRUE(vertex->GetProperty(property, storage::View::NEW)->IsNull());
ASSERT_EQ(vertex->Properties(storage::View::OLD)->size(), 0);
ASSERT_EQ(vertex->Properties(storage::View::NEW)->size(), 0);
// Set property 5 to "nandare"
ASSERT_TRUE(vertex->SetProperty(property, storage::PropertyValue("nandare"))
.GetValue());
// Check whether property 5 exists
ASSERT_TRUE(vertex->GetProperty(property, storage::View::OLD)->IsNull());
ASSERT_EQ(vertex->GetProperty(property, storage::View::NEW)->ValueString(),
"nandare");
ASSERT_EQ(vertex->Properties(storage::View::OLD)->size(), 0);
{
auto properties = vertex->Properties(storage::View::NEW).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "nandare");
}
// Advance command
acc.AdvanceCommand();
// Check whether property 5 exists
ASSERT_EQ(vertex->GetProperty(property, storage::View::OLD)->ValueString(),
"nandare");
ASSERT_EQ(vertex->GetProperty(property, storage::View::NEW)->ValueString(),
"nandare");
{
auto properties = vertex->Properties(storage::View::OLD).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "nandare");
}
{
auto properties = vertex->Properties(storage::View::NEW).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "nandare");
}
// Delete the vertex
ASSERT_TRUE(acc.DeleteVertex(&*vertex).GetValue());
// Check whether property 5 exists
ASSERT_EQ(vertex->GetProperty(property, storage::View::OLD)->ValueString(),
"nandare");
ASSERT_EQ(vertex->GetProperty(property, storage::View::NEW).GetError(),
storage::Error::DELETED_OBJECT);
{
auto properties = vertex->Properties(storage::View::OLD).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "nandare");
}
ASSERT_EQ(vertex->Properties(storage::View::NEW).GetError(),
storage::Error::DELETED_OBJECT);
// Advance command
acc.AdvanceCommand();
// Check whether property 5 exists
ASSERT_EQ(vertex->GetProperty(property, storage::View::OLD).GetError(),
storage::Error::DELETED_OBJECT);
ASSERT_EQ(vertex->GetProperty(property, storage::View::NEW).GetError(),
storage::Error::DELETED_OBJECT);
ASSERT_EQ(vertex->Properties(storage::View::OLD).GetError(),
storage::Error::DELETED_OBJECT);
ASSERT_EQ(vertex->Properties(storage::View::NEW).GetError(),
storage::Error::DELETED_OBJECT);
// Try to set the property
{
auto ret =
vertex->SetProperty(property, storage::PropertyValue("haihai"));
ASSERT_TRUE(ret.HasError());
ASSERT_EQ(ret.GetError(), storage::Error::DELETED_OBJECT);
}
acc.Abort();
}
}
// NOLINTNEXTLINE(hicpp-special-member-functions)
TEST(StorageV2, VertexLabelCommit) {
storage::Storage store;
storage::Gid gid =
storage::Gid::FromUint(std::numeric_limits<uint64_t>::max());
{
auto acc = store.Access();
auto vertex = acc.CreateVertex();
gid = vertex.Gid();
auto label = acc.NameToLabel("label5");
ASSERT_FALSE(vertex.HasLabel(label, storage::View::NEW).GetValue());
ASSERT_EQ(vertex.Labels(storage::View::NEW)->size(), 0);
{
auto res = vertex.AddLabel(label);
ASSERT_TRUE(res.HasValue());
ASSERT_TRUE(res.GetValue());
}
ASSERT_TRUE(vertex.HasLabel(label, storage::View::NEW).GetValue());
{
auto labels = vertex.Labels(storage::View::NEW).GetValue();
ASSERT_EQ(labels.size(), 1);
ASSERT_EQ(labels[0], label);
}
{
auto res = vertex.AddLabel(label);
ASSERT_TRUE(res.HasValue());
ASSERT_FALSE(res.GetValue());
}
ASSERT_FALSE(acc.Commit().HasError());
}
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
auto label = acc.NameToLabel("label5");
ASSERT_TRUE(vertex->HasLabel(label, storage::View::OLD).GetValue());
{
auto labels = vertex->Labels(storage::View::OLD).GetValue();
ASSERT_EQ(labels.size(), 1);
ASSERT_EQ(labels[0], label);
}
ASSERT_TRUE(vertex->HasLabel(label, storage::View::NEW).GetValue());
{
auto labels = vertex->Labels(storage::View::NEW).GetValue();
ASSERT_EQ(labels.size(), 1);
ASSERT_EQ(labels[0], label);
}
auto other_label = acc.NameToLabel("other");
ASSERT_FALSE(vertex->HasLabel(other_label, storage::View::OLD).GetValue());
ASSERT_FALSE(vertex->HasLabel(other_label, storage::View::NEW).GetValue());
acc.Abort();
}
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
auto label = acc.NameToLabel("label5");
{
auto res = vertex->RemoveLabel(label);
ASSERT_TRUE(res.HasValue());
ASSERT_TRUE(res.GetValue());
}
ASSERT_TRUE(vertex->HasLabel(label, storage::View::OLD).GetValue());
{
auto labels = vertex->Labels(storage::View::OLD).GetValue();
ASSERT_EQ(labels.size(), 1);
ASSERT_EQ(labels[0], label);
}
ASSERT_FALSE(vertex->HasLabel(label, storage::View::NEW).GetValue());
ASSERT_EQ(vertex->Labels(storage::View::NEW)->size(), 0);
{
auto res = vertex->RemoveLabel(label);
ASSERT_TRUE(res.HasValue());
ASSERT_FALSE(res.GetValue());
}
ASSERT_FALSE(acc.Commit().HasError());
}
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
auto label = acc.NameToLabel("label5");
ASSERT_FALSE(vertex->HasLabel(label, storage::View::OLD).GetValue());
ASSERT_FALSE(vertex->HasLabel(label, storage::View::NEW).GetValue());
ASSERT_EQ(vertex->Labels(storage::View::OLD)->size(), 0);
ASSERT_EQ(vertex->Labels(storage::View::NEW)->size(), 0);
auto other_label = acc.NameToLabel("other");
ASSERT_FALSE(vertex->HasLabel(other_label, storage::View::OLD).GetValue());
ASSERT_FALSE(vertex->HasLabel(other_label, storage::View::NEW).GetValue());
acc.Abort();
}
}
// NOLINTNEXTLINE(hicpp-special-member-functions)
TEST(StorageV2, VertexLabelAbort) {
storage::Storage store;
storage::Gid gid =
storage::Gid::FromUint(std::numeric_limits<uint64_t>::max());
// Create the vertex.
{
auto acc = store.Access();
auto vertex = acc.CreateVertex();
gid = vertex.Gid();
ASSERT_FALSE(acc.Commit().HasError());
}
// Add label 5, but abort the transaction.
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
auto label = acc.NameToLabel("label5");
ASSERT_FALSE(vertex->HasLabel(label, storage::View::NEW).GetValue());
ASSERT_EQ(vertex->Labels(storage::View::NEW)->size(), 0);
{
auto res = vertex->AddLabel(label);
ASSERT_TRUE(res.HasValue());
ASSERT_TRUE(res.GetValue());
}
ASSERT_TRUE(vertex->HasLabel(label, storage::View::NEW).GetValue());
{
auto labels = vertex->Labels(storage::View::NEW).GetValue();
ASSERT_EQ(labels.size(), 1);
ASSERT_EQ(labels[0], label);
}
{
auto res = vertex->AddLabel(label);
ASSERT_TRUE(res.HasValue());
ASSERT_FALSE(res.GetValue());
}
acc.Abort();
}
// Check that label 5 doesn't exist.
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
auto label = acc.NameToLabel("label5");
ASSERT_FALSE(vertex->HasLabel(label, storage::View::OLD).GetValue());
ASSERT_FALSE(vertex->HasLabel(label, storage::View::NEW).GetValue());
ASSERT_EQ(vertex->Labels(storage::View::OLD)->size(), 0);
ASSERT_EQ(vertex->Labels(storage::View::NEW)->size(), 0);
auto other_label = acc.NameToLabel("other");
ASSERT_FALSE(vertex->HasLabel(other_label, storage::View::OLD).GetValue());
ASSERT_FALSE(vertex->HasLabel(other_label, storage::View::NEW).GetValue());
acc.Abort();
}
// Add label 5.
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
auto label = acc.NameToLabel("label5");
ASSERT_FALSE(vertex->HasLabel(label, storage::View::NEW).GetValue());
ASSERT_EQ(vertex->Labels(storage::View::NEW)->size(), 0);
{
auto res = vertex->AddLabel(label);
ASSERT_TRUE(res.HasValue());
ASSERT_TRUE(res.GetValue());
}
ASSERT_TRUE(vertex->HasLabel(label, storage::View::NEW).GetValue());
{
auto labels = vertex->Labels(storage::View::NEW).GetValue();
ASSERT_EQ(labels.size(), 1);
ASSERT_EQ(labels[0], label);
}
{
auto res = vertex->AddLabel(label);
ASSERT_TRUE(res.HasValue());
ASSERT_FALSE(res.GetValue());
}
ASSERT_FALSE(acc.Commit().HasError());
}
// Check that label 5 exists.
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
auto label = acc.NameToLabel("label5");
ASSERT_TRUE(vertex->HasLabel(label, storage::View::OLD).GetValue());
{
auto labels = vertex->Labels(storage::View::OLD).GetValue();
ASSERT_EQ(labels.size(), 1);
ASSERT_EQ(labels[0], label);
}
ASSERT_TRUE(vertex->HasLabel(label, storage::View::NEW).GetValue());
{
auto labels = vertex->Labels(storage::View::NEW).GetValue();
ASSERT_EQ(labels.size(), 1);
ASSERT_EQ(labels[0], label);
}
auto other_label = acc.NameToLabel("other");
ASSERT_FALSE(vertex->HasLabel(other_label, storage::View::OLD).GetValue());
ASSERT_FALSE(vertex->HasLabel(other_label, storage::View::NEW).GetValue());
acc.Abort();
}
// Remove label 5, but abort the transaction.
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
auto label = acc.NameToLabel("label5");
{
auto res = vertex->RemoveLabel(label);
ASSERT_TRUE(res.HasValue());
ASSERT_TRUE(res.GetValue());
}
ASSERT_TRUE(vertex->HasLabel(label, storage::View::OLD).GetValue());
{
auto labels = vertex->Labels(storage::View::OLD).GetValue();
ASSERT_EQ(labels.size(), 1);
ASSERT_EQ(labels[0], label);
}
ASSERT_FALSE(vertex->HasLabel(label, storage::View::NEW).GetValue());
ASSERT_EQ(vertex->Labels(storage::View::NEW)->size(), 0);
{
auto res = vertex->RemoveLabel(label);
ASSERT_TRUE(res.HasValue());
ASSERT_FALSE(res.GetValue());
}
acc.Abort();
}
// Check that label 5 exists.
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
auto label = acc.NameToLabel("label5");
ASSERT_TRUE(vertex->HasLabel(label, storage::View::OLD).GetValue());
{
auto labels = vertex->Labels(storage::View::OLD).GetValue();
ASSERT_EQ(labels.size(), 1);
ASSERT_EQ(labels[0], label);
}
ASSERT_TRUE(vertex->HasLabel(label, storage::View::NEW).GetValue());
{
auto labels = vertex->Labels(storage::View::NEW).GetValue();
ASSERT_EQ(labels.size(), 1);
ASSERT_EQ(labels[0], label);
}
auto other_label = acc.NameToLabel("other");
ASSERT_FALSE(vertex->HasLabel(other_label, storage::View::OLD).GetValue());
ASSERT_FALSE(vertex->HasLabel(other_label, storage::View::NEW).GetValue());
acc.Abort();
}
// Remove label 5.
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
auto label = acc.NameToLabel("label5");
{
auto res = vertex->RemoveLabel(label);
ASSERT_TRUE(res.HasValue());
ASSERT_TRUE(res.GetValue());
}
ASSERT_TRUE(vertex->HasLabel(label, storage::View::OLD).GetValue());
{
auto labels = vertex->Labels(storage::View::OLD).GetValue();
ASSERT_EQ(labels.size(), 1);
ASSERT_EQ(labels[0], label);
}
ASSERT_FALSE(vertex->HasLabel(label, storage::View::NEW).GetValue());
ASSERT_EQ(vertex->Labels(storage::View::NEW)->size(), 0);
{
auto res = vertex->RemoveLabel(label);
ASSERT_TRUE(res.HasValue());
ASSERT_FALSE(res.GetValue());
}
ASSERT_FALSE(acc.Commit().HasError());
}
// Check that label 5 doesn't exist.
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
auto label = acc.NameToLabel("label5");
ASSERT_FALSE(vertex->HasLabel(label, storage::View::OLD).GetValue());
ASSERT_FALSE(vertex->HasLabel(label, storage::View::NEW).GetValue());
ASSERT_EQ(vertex->Labels(storage::View::OLD)->size(), 0);
ASSERT_EQ(vertex->Labels(storage::View::NEW)->size(), 0);
auto other_label = acc.NameToLabel("other");
ASSERT_FALSE(vertex->HasLabel(other_label, storage::View::OLD).GetValue());
ASSERT_FALSE(vertex->HasLabel(other_label, storage::View::NEW).GetValue());
acc.Abort();
}
}
// NOLINTNEXTLINE(hicpp-special-member-functions)
TEST(StorageV2, VertexLabelSerializationError) {
storage::Storage store;
storage::Gid gid =
storage::Gid::FromUint(std::numeric_limits<uint64_t>::max());
{
auto acc = store.Access();
auto vertex = acc.CreateVertex();
gid = vertex.Gid();
ASSERT_FALSE(acc.Commit().HasError());
}
auto acc1 = store.Access();
auto acc2 = store.Access();
// Add label 1 in accessor 1.
{
auto vertex = acc1.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
auto label1 = acc1.NameToLabel("label1");
auto label2 = acc1.NameToLabel("label2");
ASSERT_FALSE(vertex->HasLabel(label1, storage::View::OLD).GetValue());
ASSERT_FALSE(vertex->HasLabel(label1, storage::View::NEW).GetValue());
ASSERT_FALSE(vertex->HasLabel(label2, storage::View::OLD).GetValue());
ASSERT_FALSE(vertex->HasLabel(label2, storage::View::NEW).GetValue());
ASSERT_EQ(vertex->Labels(storage::View::OLD)->size(), 0);
ASSERT_EQ(vertex->Labels(storage::View::NEW)->size(), 0);
{
auto res = vertex->AddLabel(label1);
ASSERT_TRUE(res.HasValue());
ASSERT_TRUE(res.GetValue());
}
ASSERT_FALSE(vertex->HasLabel(label1, storage::View::OLD).GetValue());
ASSERT_TRUE(vertex->HasLabel(label1, storage::View::NEW).GetValue());
ASSERT_FALSE(vertex->HasLabel(label2, storage::View::OLD).GetValue());
ASSERT_FALSE(vertex->HasLabel(label2, storage::View::NEW).GetValue());
ASSERT_EQ(vertex->Labels(storage::View::OLD)->size(), 0);
{
auto labels = vertex->Labels(storage::View::NEW).GetValue();
ASSERT_EQ(labels.size(), 1);
ASSERT_EQ(labels[0], label1);
}
{
auto res = vertex->AddLabel(label1);
ASSERT_TRUE(res.HasValue());
ASSERT_FALSE(res.GetValue());
}
}
// Add label 2 in accessor 2.
{
auto vertex = acc2.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
auto label1 = acc2.NameToLabel("label1");
auto label2 = acc2.NameToLabel("label2");
ASSERT_FALSE(vertex->HasLabel(label1, storage::View::OLD).GetValue());
ASSERT_FALSE(vertex->HasLabel(label1, storage::View::NEW).GetValue());
ASSERT_FALSE(vertex->HasLabel(label2, storage::View::OLD).GetValue());
ASSERT_FALSE(vertex->HasLabel(label2, storage::View::NEW).GetValue());
ASSERT_EQ(vertex->Labels(storage::View::OLD)->size(), 0);
ASSERT_EQ(vertex->Labels(storage::View::NEW)->size(), 0);
{
auto res = vertex->AddLabel(label1);
ASSERT_TRUE(res.HasError());
ASSERT_EQ(res.GetError(), storage::Error::SERIALIZATION_ERROR);
}
}
// Finalize both accessors.
ASSERT_FALSE(acc1.Commit().HasError());
acc2.Abort();
// Check which labels exist.
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
auto label1 = acc.NameToLabel("label1");
auto label2 = acc.NameToLabel("label2");
ASSERT_TRUE(vertex->HasLabel(label1, storage::View::OLD).GetValue());
ASSERT_FALSE(vertex->HasLabel(label2, storage::View::OLD).GetValue());
{
auto labels = vertex->Labels(storage::View::OLD).GetValue();
ASSERT_EQ(labels.size(), 1);
ASSERT_EQ(labels[0], label1);
}
ASSERT_TRUE(vertex->HasLabel(label1, storage::View::NEW).GetValue());
ASSERT_FALSE(vertex->HasLabel(label2, storage::View::NEW).GetValue());
{
auto labels = vertex->Labels(storage::View::NEW).GetValue();
ASSERT_EQ(labels.size(), 1);
ASSERT_EQ(labels[0], label1);
}
acc.Abort();
}
}
// NOLINTNEXTLINE(hicpp-special-member-functions)
TEST(StorageV2, VertexPropertyCommit) {
storage::Storage store;
storage::Gid gid =
storage::Gid::FromUint(std::numeric_limits<uint64_t>::max());
{
auto acc = store.Access();
auto vertex = acc.CreateVertex();
gid = vertex.Gid();
auto property = acc.NameToProperty("property5");
ASSERT_TRUE(vertex.GetProperty(property, storage::View::NEW)->IsNull());
ASSERT_EQ(vertex.Properties(storage::View::NEW)->size(), 0);
{
auto res =
vertex.SetProperty(property, storage::PropertyValue("temporary"));
ASSERT_TRUE(res.HasValue());
ASSERT_TRUE(res.GetValue());
}
ASSERT_EQ(vertex.GetProperty(property, storage::View::NEW)->ValueString(),
"temporary");
{
auto properties = vertex.Properties(storage::View::NEW).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "temporary");
}
{
auto res =
vertex.SetProperty(property, storage::PropertyValue("nandare"));
ASSERT_TRUE(res.HasValue());
ASSERT_FALSE(res.GetValue());
}
ASSERT_EQ(vertex.GetProperty(property, storage::View::NEW)->ValueString(),
"nandare");
{
auto properties = vertex.Properties(storage::View::NEW).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "nandare");
}
ASSERT_FALSE(acc.Commit().HasError());
}
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
auto property = acc.NameToProperty("property5");
ASSERT_EQ(vertex->GetProperty(property, storage::View::OLD)->ValueString(),
"nandare");
{
auto properties = vertex->Properties(storage::View::OLD).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "nandare");
}
ASSERT_EQ(vertex->GetProperty(property, storage::View::NEW)->ValueString(),
"nandare");
{
auto properties = vertex->Properties(storage::View::NEW).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "nandare");
}
auto other_property = acc.NameToProperty("other");
ASSERT_TRUE(
vertex->GetProperty(other_property, storage::View::OLD)->IsNull());
ASSERT_TRUE(
vertex->GetProperty(other_property, storage::View::NEW)->IsNull());
acc.Abort();
}
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
auto property = acc.NameToProperty("property5");
{
auto res = vertex->SetProperty(property, storage::PropertyValue());
ASSERT_TRUE(res.HasValue());
ASSERT_FALSE(res.GetValue());
}
ASSERT_EQ(vertex->GetProperty(property, storage::View::OLD)->ValueString(),
"nandare");
{
auto properties = vertex->Properties(storage::View::OLD).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "nandare");
}
ASSERT_TRUE(vertex->GetProperty(property, storage::View::NEW)->IsNull());
ASSERT_EQ(vertex->Properties(storage::View::NEW)->size(), 0);
{
auto res = vertex->SetProperty(property, storage::PropertyValue());
ASSERT_TRUE(res.HasValue());
ASSERT_TRUE(res.GetValue());
}
ASSERT_FALSE(acc.Commit().HasError());
}
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
auto property = acc.NameToProperty("property5");
ASSERT_TRUE(vertex->GetProperty(property, storage::View::OLD)->IsNull());
ASSERT_TRUE(vertex->GetProperty(property, storage::View::NEW)->IsNull());
ASSERT_EQ(vertex->Properties(storage::View::OLD)->size(), 0);
ASSERT_EQ(vertex->Properties(storage::View::NEW)->size(), 0);
auto other_property = acc.NameToProperty("other");
ASSERT_TRUE(
vertex->GetProperty(other_property, storage::View::OLD)->IsNull());
ASSERT_TRUE(
vertex->GetProperty(other_property, storage::View::NEW)->IsNull());
acc.Abort();
}
}
// NOLINTNEXTLINE(hicpp-special-member-functions)
TEST(StorageV2, VertexPropertyAbort) {
storage::Storage store;
storage::Gid gid =
storage::Gid::FromUint(std::numeric_limits<uint64_t>::max());
// Create the vertex.
{
auto acc = store.Access();
auto vertex = acc.CreateVertex();
gid = vertex.Gid();
ASSERT_FALSE(acc.Commit().HasError());
}
// Set property 5 to "nandare", but abort the transaction.
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
auto property = acc.NameToProperty("property5");
ASSERT_TRUE(vertex->GetProperty(property, storage::View::NEW)->IsNull());
ASSERT_EQ(vertex->Properties(storage::View::NEW)->size(), 0);
{
auto res =
vertex->SetProperty(property, storage::PropertyValue("temporary"));
ASSERT_TRUE(res.HasValue());
ASSERT_TRUE(res.GetValue());
}
ASSERT_EQ(vertex->GetProperty(property, storage::View::NEW)->ValueString(),
"temporary");
{
auto properties = vertex->Properties(storage::View::NEW).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "temporary");
}
{
auto res =
vertex->SetProperty(property, storage::PropertyValue("nandare"));
ASSERT_TRUE(res.HasValue());
ASSERT_FALSE(res.GetValue());
}
ASSERT_EQ(vertex->GetProperty(property, storage::View::NEW)->ValueString(),
"nandare");
{
auto properties = vertex->Properties(storage::View::NEW).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "nandare");
}
acc.Abort();
}
// Check that property 5 is null.
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
auto property = acc.NameToProperty("property5");
ASSERT_TRUE(vertex->GetProperty(property, storage::View::OLD)->IsNull());
ASSERT_TRUE(vertex->GetProperty(property, storage::View::NEW)->IsNull());
ASSERT_EQ(vertex->Properties(storage::View::OLD)->size(), 0);
ASSERT_EQ(vertex->Properties(storage::View::NEW)->size(), 0);
auto other_property = acc.NameToProperty("other");
ASSERT_TRUE(
vertex->GetProperty(other_property, storage::View::OLD)->IsNull());
ASSERT_TRUE(
vertex->GetProperty(other_property, storage::View::NEW)->IsNull());
acc.Abort();
}
// Set property 5 to "nandare".
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
auto property = acc.NameToProperty("property5");
ASSERT_TRUE(vertex->GetProperty(property, storage::View::NEW)->IsNull());
ASSERT_EQ(vertex->Properties(storage::View::NEW)->size(), 0);
{
auto res =
vertex->SetProperty(property, storage::PropertyValue("temporary"));
ASSERT_TRUE(res.HasValue());
ASSERT_TRUE(res.GetValue());
}
ASSERT_EQ(vertex->GetProperty(property, storage::View::NEW)->ValueString(),
"temporary");
{
auto properties = vertex->Properties(storage::View::NEW).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "temporary");
}
{
auto res =
vertex->SetProperty(property, storage::PropertyValue("nandare"));
ASSERT_TRUE(res.HasValue());
ASSERT_FALSE(res.GetValue());
}
ASSERT_EQ(vertex->GetProperty(property, storage::View::NEW)->ValueString(),
"nandare");
{
auto properties = vertex->Properties(storage::View::NEW).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "nandare");
}
ASSERT_FALSE(acc.Commit().HasError());
}
// Check that property 5 is "nandare".
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
auto property = acc.NameToProperty("property5");
ASSERT_EQ(vertex->GetProperty(property, storage::View::OLD)->ValueString(),
"nandare");
{
auto properties = vertex->Properties(storage::View::OLD).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "nandare");
}
ASSERT_EQ(vertex->GetProperty(property, storage::View::NEW)->ValueString(),
"nandare");
{
auto properties = vertex->Properties(storage::View::NEW).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "nandare");
}
auto other_property = acc.NameToProperty("other");
ASSERT_TRUE(
vertex->GetProperty(other_property, storage::View::OLD)->IsNull());
ASSERT_TRUE(
vertex->GetProperty(other_property, storage::View::NEW)->IsNull());
acc.Abort();
}
// Set property 5 to null, but abort the transaction.
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
auto property = acc.NameToProperty("property5");
ASSERT_EQ(vertex->GetProperty(property, storage::View::OLD)->ValueString(),
"nandare");
{
auto properties = vertex->Properties(storage::View::OLD).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "nandare");
}
ASSERT_EQ(vertex->GetProperty(property, storage::View::NEW)->ValueString(),
"nandare");
{
auto properties = vertex->Properties(storage::View::NEW).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "nandare");
}
{
auto res = vertex->SetProperty(property, storage::PropertyValue());
ASSERT_TRUE(res.HasValue());
ASSERT_FALSE(res.GetValue());
}
ASSERT_EQ(vertex->GetProperty(property, storage::View::OLD)->ValueString(),
"nandare");
{
auto properties = vertex->Properties(storage::View::OLD).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "nandare");
}
ASSERT_TRUE(vertex->GetProperty(property, storage::View::NEW)->IsNull());
ASSERT_EQ(vertex->Properties(storage::View::NEW)->size(), 0);
acc.Abort();
}
// Check that property 5 is "nandare".
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
auto property = acc.NameToProperty("property5");
ASSERT_EQ(vertex->GetProperty(property, storage::View::OLD)->ValueString(),
"nandare");
{
auto properties = vertex->Properties(storage::View::OLD).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "nandare");
}
ASSERT_EQ(vertex->GetProperty(property, storage::View::NEW)->ValueString(),
"nandare");
{
auto properties = vertex->Properties(storage::View::NEW).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "nandare");
}
auto other_property = acc.NameToProperty("other");
ASSERT_TRUE(
vertex->GetProperty(other_property, storage::View::OLD)->IsNull());
ASSERT_TRUE(
vertex->GetProperty(other_property, storage::View::NEW)->IsNull());
acc.Abort();
}
// Set property 5 to null.
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
auto property = acc.NameToProperty("property5");
ASSERT_EQ(vertex->GetProperty(property, storage::View::OLD)->ValueString(),
"nandare");
{
auto properties = vertex->Properties(storage::View::OLD).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "nandare");
}
ASSERT_EQ(vertex->GetProperty(property, storage::View::NEW)->ValueString(),
"nandare");
{
auto properties = vertex->Properties(storage::View::NEW).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "nandare");
}
{
auto res = vertex->SetProperty(property, storage::PropertyValue());
ASSERT_TRUE(res.HasValue());
ASSERT_FALSE(res.GetValue());
}
ASSERT_EQ(vertex->GetProperty(property, storage::View::OLD)->ValueString(),
"nandare");
{
auto properties = vertex->Properties(storage::View::OLD).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "nandare");
}
ASSERT_TRUE(vertex->GetProperty(property, storage::View::NEW)->IsNull());
ASSERT_EQ(vertex->Properties(storage::View::NEW)->size(), 0);
ASSERT_FALSE(acc.Commit().HasError());
}
// Check that property 5 is null.
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
auto property = acc.NameToProperty("property5");
ASSERT_TRUE(vertex->GetProperty(property, storage::View::OLD)->IsNull());
ASSERT_TRUE(vertex->GetProperty(property, storage::View::NEW)->IsNull());
ASSERT_EQ(vertex->Properties(storage::View::OLD)->size(), 0);
ASSERT_EQ(vertex->Properties(storage::View::NEW)->size(), 0);
auto other_property = acc.NameToProperty("other");
ASSERT_TRUE(
vertex->GetProperty(other_property, storage::View::OLD)->IsNull());
ASSERT_TRUE(
vertex->GetProperty(other_property, storage::View::NEW)->IsNull());
acc.Abort();
}
}
// NOLINTNEXTLINE(hicpp-special-member-functions)
TEST(StorageV2, VertexPropertySerializationError) {
storage::Storage store;
storage::Gid gid =
storage::Gid::FromUint(std::numeric_limits<uint64_t>::max());
{
auto acc = store.Access();
auto vertex = acc.CreateVertex();
gid = vertex.Gid();
ASSERT_FALSE(acc.Commit().HasError());
}
auto acc1 = store.Access();
auto acc2 = store.Access();
// Set property 1 to 123 in accessor 1.
{
auto vertex = acc1.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
auto property1 = acc1.NameToProperty("property1");
auto property2 = acc1.NameToProperty("property2");
ASSERT_TRUE(vertex->GetProperty(property1, storage::View::OLD)->IsNull());
ASSERT_TRUE(vertex->GetProperty(property1, storage::View::NEW)->IsNull());
ASSERT_TRUE(vertex->GetProperty(property2, storage::View::OLD)->IsNull());
ASSERT_TRUE(vertex->GetProperty(property2, storage::View::NEW)->IsNull());
ASSERT_EQ(vertex->Properties(storage::View::OLD)->size(), 0);
ASSERT_EQ(vertex->Properties(storage::View::NEW)->size(), 0);
{
auto res = vertex->SetProperty(property1, storage::PropertyValue(123));
ASSERT_TRUE(res.HasValue());
ASSERT_TRUE(res.GetValue());
}
ASSERT_TRUE(vertex->GetProperty(property1, storage::View::OLD)->IsNull());
ASSERT_EQ(vertex->GetProperty(property1, storage::View::NEW)->ValueInt(),
123);
ASSERT_TRUE(vertex->GetProperty(property2, storage::View::OLD)->IsNull());
ASSERT_TRUE(vertex->GetProperty(property2, storage::View::NEW)->IsNull());
ASSERT_EQ(vertex->Properties(storage::View::OLD)->size(), 0);
{
auto properties = vertex->Properties(storage::View::NEW).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property1].ValueInt(), 123);
}
}
// Set property 2 to "nandare" in accessor 2.
{
auto vertex = acc2.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
auto property1 = acc2.NameToProperty("property1");
auto property2 = acc2.NameToProperty("property2");
ASSERT_TRUE(vertex->GetProperty(property1, storage::View::OLD)->IsNull());
ASSERT_TRUE(vertex->GetProperty(property1, storage::View::NEW)->IsNull());
ASSERT_TRUE(vertex->GetProperty(property2, storage::View::OLD)->IsNull());
ASSERT_TRUE(vertex->GetProperty(property2, storage::View::NEW)->IsNull());
ASSERT_EQ(vertex->Properties(storage::View::OLD)->size(), 0);
ASSERT_EQ(vertex->Properties(storage::View::NEW)->size(), 0);
{
auto res =
vertex->SetProperty(property2, storage::PropertyValue("nandare"));
ASSERT_TRUE(res.HasError());
ASSERT_EQ(res.GetError(), storage::Error::SERIALIZATION_ERROR);
}
}
// Finalize both accessors.
ASSERT_FALSE(acc1.Commit().HasError());
acc2.Abort();
// Check which properties exist.
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
auto property1 = acc.NameToProperty("property1");
auto property2 = acc.NameToProperty("property2");
ASSERT_EQ(vertex->GetProperty(property1, storage::View::OLD)->ValueInt(),
123);
ASSERT_TRUE(vertex->GetProperty(property2, storage::View::OLD)->IsNull());
{
auto properties = vertex->Properties(storage::View::OLD).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property1].ValueInt(), 123);
}
ASSERT_EQ(vertex->GetProperty(property1, storage::View::NEW)->ValueInt(),
123);
ASSERT_TRUE(vertex->GetProperty(property2, storage::View::NEW)->IsNull());
{
auto properties = vertex->Properties(storage::View::NEW).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property1].ValueInt(), 123);
}
acc.Abort();
}
}
// NOLINTNEXTLINE(hicpp-special-member-functions)
TEST(StorageV2, VertexLabelPropertyMixed) {
storage::Storage store;
auto acc = store.Access();
auto vertex = acc.CreateVertex();
auto label = acc.NameToLabel("label5");
auto property = acc.NameToProperty("property5");
// Check whether label 5 and property 5 exist
ASSERT_FALSE(vertex.HasLabel(label, storage::View::NEW).GetValue());
ASSERT_EQ(vertex.Labels(storage::View::NEW)->size(), 0);
ASSERT_TRUE(vertex.GetProperty(property, storage::View::NEW)->IsNull());
ASSERT_EQ(vertex.Properties(storage::View::NEW)->size(), 0);
// Add label 5
ASSERT_TRUE(vertex.AddLabel(label).GetValue());
// Check whether label 5 and property 5 exist
ASSERT_TRUE(vertex.HasLabel(label, storage::View::NEW).GetValue());
{
auto labels = vertex.Labels(storage::View::NEW).GetValue();
ASSERT_EQ(labels.size(), 1);
ASSERT_EQ(labels[0], label);
}
ASSERT_TRUE(vertex.GetProperty(property, storage::View::NEW)->IsNull());
ASSERT_EQ(vertex.Properties(storage::View::NEW)->size(), 0);
// Advance command
acc.AdvanceCommand();
// Check whether label 5 and property 5 exist
ASSERT_TRUE(vertex.HasLabel(label, storage::View::OLD).GetValue());
ASSERT_TRUE(vertex.HasLabel(label, storage::View::NEW).GetValue());
{
auto labels = vertex.Labels(storage::View::OLD).GetValue();
ASSERT_EQ(labels.size(), 1);
ASSERT_EQ(labels[0], label);
}
{
auto labels = vertex.Labels(storage::View::NEW).GetValue();
ASSERT_EQ(labels.size(), 1);
ASSERT_EQ(labels[0], label);
}
ASSERT_TRUE(vertex.GetProperty(property, storage::View::OLD)->IsNull());
ASSERT_TRUE(vertex.GetProperty(property, storage::View::NEW)->IsNull());
ASSERT_EQ(vertex.Properties(storage::View::OLD)->size(), 0);
ASSERT_EQ(vertex.Properties(storage::View::NEW)->size(), 0);
// Set property 5 to "nandare"
ASSERT_TRUE(vertex.SetProperty(property, storage::PropertyValue("nandare"))
.GetValue());
// Check whether label 5 and property 5 exist
ASSERT_TRUE(vertex.HasLabel(label, storage::View::OLD).GetValue());
ASSERT_TRUE(vertex.HasLabel(label, storage::View::NEW).GetValue());
{
auto labels = vertex.Labels(storage::View::OLD).GetValue();
ASSERT_EQ(labels.size(), 1);
ASSERT_EQ(labels[0], label);
}
{
auto labels = vertex.Labels(storage::View::NEW).GetValue();
ASSERT_EQ(labels.size(), 1);
ASSERT_EQ(labels[0], label);
}
ASSERT_TRUE(vertex.GetProperty(property, storage::View::OLD)->IsNull());
ASSERT_EQ(vertex.GetProperty(property, storage::View::NEW)->ValueString(),
"nandare");
ASSERT_EQ(vertex.Properties(storage::View::OLD)->size(), 0);
{
auto properties = vertex.Properties(storage::View::NEW).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "nandare");
}
// Advance command
acc.AdvanceCommand();
// Check whether label 5 and property 5 exist
ASSERT_TRUE(vertex.HasLabel(label, storage::View::OLD).GetValue());
ASSERT_TRUE(vertex.HasLabel(label, storage::View::NEW).GetValue());
{
auto labels = vertex.Labels(storage::View::OLD).GetValue();
ASSERT_EQ(labels.size(), 1);
ASSERT_EQ(labels[0], label);
}
{
auto labels = vertex.Labels(storage::View::NEW).GetValue();
ASSERT_EQ(labels.size(), 1);
ASSERT_EQ(labels[0], label);
}
ASSERT_EQ(vertex.GetProperty(property, storage::View::OLD)->ValueString(),
"nandare");
ASSERT_EQ(vertex.GetProperty(property, storage::View::NEW)->ValueString(),
"nandare");
{
auto properties = vertex.Properties(storage::View::OLD).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "nandare");
}
{
auto properties = vertex.Properties(storage::View::NEW).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "nandare");
}
// Set property 5 to "haihai"
ASSERT_FALSE(vertex.SetProperty(property, storage::PropertyValue("haihai"))
.GetValue());
// Check whether label 5 and property 5 exist
ASSERT_TRUE(vertex.HasLabel(label, storage::View::OLD).GetValue());
ASSERT_TRUE(vertex.HasLabel(label, storage::View::NEW).GetValue());
{
auto labels = vertex.Labels(storage::View::OLD).GetValue();
ASSERT_EQ(labels.size(), 1);
ASSERT_EQ(labels[0], label);
}
{
auto labels = vertex.Labels(storage::View::NEW).GetValue();
ASSERT_EQ(labels.size(), 1);
ASSERT_EQ(labels[0], label);
}
ASSERT_EQ(vertex.GetProperty(property, storage::View::OLD)->ValueString(),
"nandare");
ASSERT_EQ(vertex.GetProperty(property, storage::View::NEW)->ValueString(),
"haihai");
{
auto properties = vertex.Properties(storage::View::OLD).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "nandare");
}
{
auto properties = vertex.Properties(storage::View::NEW).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "haihai");
}
// Advance command
acc.AdvanceCommand();
// Check whether label 5 and property 5 exist
ASSERT_TRUE(vertex.HasLabel(label, storage::View::OLD).GetValue());
ASSERT_TRUE(vertex.HasLabel(label, storage::View::NEW).GetValue());
{
auto labels = vertex.Labels(storage::View::OLD).GetValue();
ASSERT_EQ(labels.size(), 1);
ASSERT_EQ(labels[0], label);
}
{
auto labels = vertex.Labels(storage::View::NEW).GetValue();
ASSERT_EQ(labels.size(), 1);
ASSERT_EQ(labels[0], label);
}
ASSERT_EQ(vertex.GetProperty(property, storage::View::OLD)->ValueString(),
"haihai");
ASSERT_EQ(vertex.GetProperty(property, storage::View::NEW)->ValueString(),
"haihai");
{
auto properties = vertex.Properties(storage::View::OLD).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "haihai");
}
{
auto properties = vertex.Properties(storage::View::NEW).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "haihai");
}
// Remove label 5
ASSERT_TRUE(vertex.RemoveLabel(label).GetValue());
// Check whether label 5 and property 5 exist
ASSERT_TRUE(vertex.HasLabel(label, storage::View::OLD).GetValue());
ASSERT_FALSE(vertex.HasLabel(label, storage::View::NEW).GetValue());
{
auto labels = vertex.Labels(storage::View::OLD).GetValue();
ASSERT_EQ(labels.size(), 1);
ASSERT_EQ(labels[0], label);
}
ASSERT_EQ(vertex.Labels(storage::View::NEW)->size(), 0);
ASSERT_EQ(vertex.GetProperty(property, storage::View::OLD)->ValueString(),
"haihai");
ASSERT_EQ(vertex.GetProperty(property, storage::View::NEW)->ValueString(),
"haihai");
{
auto properties = vertex.Properties(storage::View::OLD).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "haihai");
}
{
auto properties = vertex.Properties(storage::View::NEW).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "haihai");
}
// Advance command
acc.AdvanceCommand();
// Check whether label 5 and property 5 exist
ASSERT_FALSE(vertex.HasLabel(label, storage::View::OLD).GetValue());
ASSERT_FALSE(vertex.HasLabel(label, storage::View::NEW).GetValue());
ASSERT_EQ(vertex.Labels(storage::View::OLD)->size(), 0);
ASSERT_EQ(vertex.Labels(storage::View::NEW)->size(), 0);
ASSERT_EQ(vertex.GetProperty(property, storage::View::OLD)->ValueString(),
"haihai");
ASSERT_EQ(vertex.GetProperty(property, storage::View::NEW)->ValueString(),
"haihai");
{
auto properties = vertex.Properties(storage::View::OLD).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "haihai");
}
{
auto properties = vertex.Properties(storage::View::NEW).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "haihai");
}
// Set property 5 to null
ASSERT_FALSE(
vertex.SetProperty(property, storage::PropertyValue()).GetValue());
// Check whether label 5 and property 5 exist
ASSERT_FALSE(vertex.HasLabel(label, storage::View::OLD).GetValue());
ASSERT_FALSE(vertex.HasLabel(label, storage::View::NEW).GetValue());
ASSERT_EQ(vertex.Labels(storage::View::OLD)->size(), 0);
ASSERT_EQ(vertex.Labels(storage::View::NEW)->size(), 0);
ASSERT_EQ(vertex.GetProperty(property, storage::View::OLD)->ValueString(),
"haihai");
ASSERT_TRUE(vertex.GetProperty(property, storage::View::NEW)->IsNull());
{
auto properties = vertex.Properties(storage::View::OLD).GetValue();
ASSERT_EQ(properties.size(), 1);
ASSERT_EQ(properties[property].ValueString(), "haihai");
}
ASSERT_EQ(vertex.Properties(storage::View::NEW)->size(), 0);
// Advance command
acc.AdvanceCommand();
// Check whether label 5 and property 5 exist
ASSERT_FALSE(vertex.HasLabel(label, storage::View::OLD).GetValue());
ASSERT_FALSE(vertex.HasLabel(label, storage::View::NEW).GetValue());
ASSERT_EQ(vertex.Labels(storage::View::OLD)->size(), 0);
ASSERT_EQ(vertex.Labels(storage::View::NEW)->size(), 0);
ASSERT_TRUE(vertex.GetProperty(property, storage::View::NEW)->IsNull());
ASSERT_TRUE(vertex.GetProperty(property, storage::View::NEW)->IsNull());
ASSERT_EQ(vertex.Properties(storage::View::OLD)->size(), 0);
ASSERT_EQ(vertex.Properties(storage::View::NEW)->size(), 0);
ASSERT_FALSE(acc.Commit().HasError());
}
TEST(StorageV2, VertexPropertyClear) {
storage::Storage store;
storage::Gid gid;
auto property1 = store.NameToProperty("property1");
auto property2 = store.NameToProperty("property2");
{
auto acc = store.Access();
auto vertex = acc.CreateVertex();
gid = vertex.Gid();
auto res = vertex.SetProperty(property1, storage::PropertyValue("value"));
ASSERT_TRUE(res.HasValue());
ASSERT_TRUE(res.GetValue());
ASSERT_FALSE(acc.Commit().HasError());
}
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
ASSERT_EQ(vertex->GetProperty(property1, storage::View::OLD)->ValueString(),
"value");
ASSERT_TRUE(vertex->GetProperty(property2, storage::View::OLD)->IsNull());
ASSERT_THAT(vertex->Properties(storage::View::OLD).GetValue(),
UnorderedElementsAre(
std::pair(property1, storage::PropertyValue("value"))));
{
auto ret = vertex->ClearProperties();
ASSERT_TRUE(ret.HasValue());
ASSERT_TRUE(ret.GetValue());
}
ASSERT_TRUE(vertex->GetProperty(property1, storage::View::NEW)->IsNull());
ASSERT_TRUE(vertex->GetProperty(property2, storage::View::NEW)->IsNull());
ASSERT_EQ(vertex->Properties(storage::View::NEW).GetValue().size(), 0);
{
auto ret = vertex->ClearProperties();
ASSERT_TRUE(ret.HasValue());
ASSERT_FALSE(ret.GetValue());
}
ASSERT_TRUE(vertex->GetProperty(property1, storage::View::NEW)->IsNull());
ASSERT_TRUE(vertex->GetProperty(property2, storage::View::NEW)->IsNull());
ASSERT_EQ(vertex->Properties(storage::View::NEW).GetValue().size(), 0);
acc.Abort();
}
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
auto res = vertex->SetProperty(property2, storage::PropertyValue(42));
ASSERT_TRUE(res.HasValue());
ASSERT_TRUE(res.GetValue());
ASSERT_FALSE(acc.Commit().HasError());
}
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
ASSERT_EQ(vertex->GetProperty(property1, storage::View::OLD)->ValueString(),
"value");
ASSERT_EQ(vertex->GetProperty(property2, storage::View::OLD)->ValueInt(),
42);
ASSERT_THAT(vertex->Properties(storage::View::OLD).GetValue(),
UnorderedElementsAre(
std::pair(property1, storage::PropertyValue("value")),
std::pair(property2, storage::PropertyValue(42))));
{
auto ret = vertex->ClearProperties();
ASSERT_TRUE(ret.HasValue());
ASSERT_TRUE(ret.GetValue());
}
ASSERT_TRUE(vertex->GetProperty(property1, storage::View::NEW)->IsNull());
ASSERT_TRUE(vertex->GetProperty(property2, storage::View::NEW)->IsNull());
ASSERT_EQ(vertex->Properties(storage::View::NEW).GetValue().size(), 0);
{
auto ret = vertex->ClearProperties();
ASSERT_TRUE(ret.HasValue());
ASSERT_FALSE(ret.GetValue());
}
ASSERT_TRUE(vertex->GetProperty(property1, storage::View::NEW)->IsNull());
ASSERT_TRUE(vertex->GetProperty(property2, storage::View::NEW)->IsNull());
ASSERT_EQ(vertex->Properties(storage::View::NEW).GetValue().size(), 0);
ASSERT_FALSE(acc.Commit().HasError());
}
{
auto acc = store.Access();
auto vertex = acc.FindVertex(gid, storage::View::OLD);
ASSERT_TRUE(vertex);
ASSERT_TRUE(vertex->GetProperty(property1, storage::View::NEW)->IsNull());
ASSERT_TRUE(vertex->GetProperty(property2, storage::View::NEW)->IsNull());
ASSERT_EQ(vertex->Properties(storage::View::NEW).GetValue().size(), 0);
acc.Abort();
}
}
TEST(StorageV2, VertexNonexistentLabelPropertyEdgeAPI) {
storage::Storage store;
auto label = store.NameToLabel("label");
auto property = store.NameToProperty("property");
auto acc = store.Access();
auto vertex = acc.CreateVertex();
// Check state before (OLD view).
ASSERT_EQ(vertex.Labels(storage::View::OLD).GetError(),
storage::Error::NONEXISTENT_OBJECT);
ASSERT_EQ(vertex.HasLabel(label, storage::View::OLD).GetError(),
storage::Error::NONEXISTENT_OBJECT);
ASSERT_EQ(vertex.Properties(storage::View::OLD).GetError(),
storage::Error::NONEXISTENT_OBJECT);
ASSERT_EQ(vertex.GetProperty(property, storage::View::OLD).GetError(),
storage::Error::NONEXISTENT_OBJECT);
ASSERT_EQ(vertex.InEdges({}, storage::View::OLD).GetError(),
storage::Error::NONEXISTENT_OBJECT);
ASSERT_EQ(vertex.OutEdges({}, storage::View::OLD).GetError(),
storage::Error::NONEXISTENT_OBJECT);
ASSERT_EQ(vertex.InDegree(storage::View::OLD).GetError(),
storage::Error::NONEXISTENT_OBJECT);
ASSERT_EQ(vertex.OutDegree(storage::View::OLD).GetError(),
storage::Error::NONEXISTENT_OBJECT);
// Check state before (NEW view).
ASSERT_EQ(vertex.Labels(storage::View::NEW)->size(), 0);
ASSERT_EQ(*vertex.HasLabel(label, storage::View::NEW), false);
ASSERT_EQ(vertex.Properties(storage::View::NEW)->size(), 0);
ASSERT_EQ(*vertex.GetProperty(property, storage::View::NEW),
storage::PropertyValue());
ASSERT_EQ(vertex.InEdges({}, storage::View::NEW)->size(), 0);
ASSERT_EQ(vertex.OutEdges({}, storage::View::NEW)->size(), 0);
ASSERT_EQ(*vertex.InDegree(storage::View::NEW), 0);
ASSERT_EQ(*vertex.OutDegree(storage::View::NEW), 0);
// Modify vertex.
ASSERT_TRUE(vertex.AddLabel(label).HasValue());
ASSERT_TRUE(
vertex.SetProperty(property, storage::PropertyValue("value")).HasValue());
ASSERT_TRUE(
acc.CreateEdge(&vertex, &vertex, acc.NameToEdgeType("edge")).HasValue());
// Check state after (OLD view).
ASSERT_EQ(vertex.Labels(storage::View::OLD).GetError(),
storage::Error::NONEXISTENT_OBJECT);
ASSERT_EQ(vertex.HasLabel(label, storage::View::OLD).GetError(),
storage::Error::NONEXISTENT_OBJECT);
ASSERT_EQ(vertex.Properties(storage::View::OLD).GetError(),
storage::Error::NONEXISTENT_OBJECT);
ASSERT_EQ(vertex.GetProperty(property, storage::View::OLD).GetError(),
storage::Error::NONEXISTENT_OBJECT);
ASSERT_EQ(vertex.InEdges({}, storage::View::OLD).GetError(),
storage::Error::NONEXISTENT_OBJECT);
ASSERT_EQ(vertex.OutEdges({}, storage::View::OLD).GetError(),
storage::Error::NONEXISTENT_OBJECT);
ASSERT_EQ(vertex.InDegree(storage::View::OLD).GetError(),
storage::Error::NONEXISTENT_OBJECT);
ASSERT_EQ(vertex.OutDegree(storage::View::OLD).GetError(),
storage::Error::NONEXISTENT_OBJECT);
// Check state after (NEW view).
ASSERT_EQ(vertex.Labels(storage::View::NEW)->size(), 1);
ASSERT_EQ(*vertex.HasLabel(label, storage::View::NEW), true);
ASSERT_EQ(vertex.Properties(storage::View::NEW)->size(), 1);
ASSERT_EQ(*vertex.GetProperty(property, storage::View::NEW),
storage::PropertyValue("value"));
ASSERT_EQ(vertex.InEdges({}, storage::View::NEW)->size(), 1);
ASSERT_EQ(vertex.OutEdges({}, storage::View::NEW)->size(), 1);
ASSERT_EQ(*vertex.InDegree(storage::View::NEW), 1);
ASSERT_EQ(*vertex.OutDegree(storage::View::NEW), 1);
ASSERT_FALSE(acc.Commit().HasError());
}