memgraph/tests/e2e/triggers/on_delete_triggers.cpp
2021-10-26 08:53:56 +02:00

310 lines
12 KiB
C++

// 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 <string>
#include <string_view>
#include <unordered_set>
#include <gflags/gflags.h>
#include <mgclient.hpp>
#include "common.hpp"
#include "utils/logging.hpp"
constexpr std::string_view kTriggerDeletedVertexLabel{"DELETED_VERTEX"};
constexpr std::string_view kTriggerDeletedEdgeLabel{"DELETED_EDGE"};
constexpr std::string_view kTriggerDeletedObjectLabel{"DELETED_OBJECT"};
enum class AllowedTriggerType : uint8_t {
VERTEX,
EDGE,
OBJECT,
};
void DetachDeleteVertex(mg::Client &client, int vertex_id) {
mg::Map parameters{{"id", mg::Value{vertex_id}}};
client.Execute(fmt::format("MATCH (n: {} {{id: $id}}) DETACH DELETE n", kVertexLabel),
mg::ConstMap{parameters.ptr()});
client.DiscardAll();
}
void DeleteEdge(mg::Client &client, int edge_id) {
mg::Map parameters{{"id", mg::Value{edge_id}}};
client.Execute(fmt::format("MATCH ()-[r: {} {{id: $id}}]->() DELETE r", kEdgeLabel), mg::ConstMap{parameters.ptr()});
client.DiscardAll();
}
void CreateOnDeleteTriggers(mg::Client &client, bool is_before,
const std::unordered_set<AllowedTriggerType> &allowed_trigger_types) {
const std::string_view before_or_after = is_before ? "BEFORE" : "AFTER";
const auto create_on_vertex_delete_trigger = [&, before_or_after] {
client.Execute(
fmt::format("CREATE TRIGGER DeletedVerticesTrigger ON () DELETE "
"{} COMMIT "
"EXECUTE "
"UNWIND deletedVertices as deletedVertex "
"CREATE (n: {} {{ id: deletedVertex.id }})",
before_or_after, kTriggerDeletedVertexLabel));
client.DiscardAll();
};
const auto create_on_edge_delete_trigger = [&, before_or_after] {
client.Execute(
fmt::format("CREATE TRIGGER DeletedEdgesTrigger ON --> DELETE "
"{} COMMIT "
"EXECUTE "
"UNWIND deletedEdges as deletedEdge "
"CREATE (n: {} {{ id: deletedEdge.id }})",
before_or_after, kTriggerDeletedEdgeLabel));
client.DiscardAll();
};
const auto create_on_object_delete_trigger = [&, before_or_after] {
client.Execute(
fmt::format("CREATE TRIGGER DeletedObjectsTrigger ON DELETE "
"{} COMMIT "
"EXECUTE "
"UNWIND deletedObjects as deletedObjectEvent "
"WITH CASE deletedObjectEvent.event_type WHEN \"deleted_vertex\" THEN deletedObjectEvent.vertex.id "
"ELSE deletedObjectEvent.edge.id END as id "
"CREATE (n: {} {{ id: id }})",
before_or_after, kTriggerDeletedObjectLabel));
client.DiscardAll();
};
for (const auto allowed_trigger_type : allowed_trigger_types) {
switch (allowed_trigger_type) {
case AllowedTriggerType::VERTEX:
create_on_vertex_delete_trigger();
break;
case AllowedTriggerType::EDGE:
create_on_edge_delete_trigger();
break;
case AllowedTriggerType::OBJECT:
create_on_object_delete_trigger();
break;
}
}
}
void DropOnDeleteTriggers(mg::Client &client, const std::unordered_set<AllowedTriggerType> &allowed_trigger_types) {
for (const auto allowed_trigger_type : allowed_trigger_types) {
switch (allowed_trigger_type) {
case AllowedTriggerType::VERTEX: {
client.Execute("DROP TRIGGER DeletedVerticesTrigger");
client.DiscardAll();
}
case AllowedTriggerType::EDGE: {
client.Execute("DROP TRIGGER DeletedEdgesTrigger");
client.DiscardAll();
}
case AllowedTriggerType::OBJECT: {
client.Execute("DROP TRIGGER DeletedObjectsTrigger");
client.DiscardAll();
}
}
}
}
struct EdgeInfo {
int from_vertex;
int to_vertex;
int edge_id;
};
void ValidateVertexExistance(mg::Client &client, const bool should_exist, const std::string_view label, const int id) {
should_exist ? CheckVertexExists(client, label, id) : CheckVertexMissing(client, label, id);
};
int main(int argc, char **argv) {
gflags::SetUsageMessage("Memgraph E2E ON DELETE Triggers");
gflags::ParseCommandLineFlags(&argc, &argv, true);
logging::RedirectToStderr();
mg::Client::Init();
auto client = Connect();
const auto run_delete_trigger_tests = [&](const bool is_before,
const std::unordered_set<AllowedTriggerType> &allowed_trigger_types) {
constexpr std::array vertex_ids{1, 2, 3, 4};
constexpr std::array edges{EdgeInfo{vertex_ids[0], vertex_ids[1], 5}, EdgeInfo{vertex_ids[2], vertex_ids[3], 6}};
{
CreateOnDeleteTriggers(*client, is_before, allowed_trigger_types);
client->BeginTransaction();
for (const auto vertex_id : vertex_ids) {
CreateVertex(*client, vertex_id);
}
for (const auto &edge : edges) {
CreateEdge(*client, edge.from_vertex, edge.to_vertex, edge.edge_id);
}
client->CommitTransaction();
CheckNumberOfAllVertices(*client, vertex_ids.size());
client->BeginTransaction();
DetachDeleteVertex(*client, vertex_ids[0]);
DeleteEdge(*client, edges[1].edge_id);
client->CommitTransaction();
// :VERTEX x 4
// deleted :VERTEX x -1
auto number_of_expected_vertices = 3;
for (const auto allowed_trigger_type : allowed_trigger_types) {
switch (allowed_trigger_type) {
case AllowedTriggerType::VERTEX:
number_of_expected_vertices += 1;
break;
case AllowedTriggerType::EDGE:
number_of_expected_vertices += 2;
break;
case AllowedTriggerType::OBJECT:
number_of_expected_vertices += 3;
break;
}
}
if (is_before) {
CheckNumberOfAllVertices(*client, number_of_expected_vertices);
} else {
WaitForNumberOfAllVertices(*client, number_of_expected_vertices);
}
ValidateVertexExistance(*client, allowed_trigger_types.contains(AllowedTriggerType::VERTEX),
kTriggerDeletedVertexLabel, vertex_ids[0]);
ValidateVertexExistance(*client, allowed_trigger_types.contains(AllowedTriggerType::OBJECT),
kTriggerDeletedObjectLabel, vertex_ids[0]);
for (const auto &edge : edges) {
ValidateVertexExistance(*client, allowed_trigger_types.contains(AllowedTriggerType::EDGE),
kTriggerDeletedEdgeLabel, edge.edge_id);
ValidateVertexExistance(*client, allowed_trigger_types.contains(AllowedTriggerType::OBJECT),
kTriggerDeletedObjectLabel, edge.edge_id);
}
DropOnDeleteTriggers(*client, allowed_trigger_types);
client->Execute("MATCH (n) DETACH DELETE n;");
client->DiscardAll();
}
};
const auto run_delete_trigger_write_procedure_tests =
[&](const std::unordered_set<AllowedTriggerType> &allowed_trigger_types) {
ExecuteCreateVertex(*client, 2);
ExecuteCreateVertex(*client, 3);
client->Execute("MATCH (n {id:2}), (m {id:3}) CALL write.create_edge(n, m, 'edge') YIELD e RETURN e");
client->DiscardAll();
CreateOnDeleteTriggers(*client, true, allowed_trigger_types);
client->Execute("MATCH ()-[e]->() CALL write.delete_edge(e)");
client->DiscardAll();
client->Execute("MATCH (n {id:2}) CALL write.delete_vertex(n)");
client->DiscardAll();
auto number_of_expected_vertices = 1;
for (const auto allowed_trigger_type : allowed_trigger_types) {
switch (allowed_trigger_type) {
case AllowedTriggerType::VERTEX:
number_of_expected_vertices += 1;
break;
case AllowedTriggerType::EDGE:
number_of_expected_vertices += 1;
break;
case AllowedTriggerType::OBJECT:
number_of_expected_vertices += 2;
break;
}
}
CheckNumberOfAllVertices(*client, number_of_expected_vertices);
ValidateVertexExistance(*client, allowed_trigger_types.contains(AllowedTriggerType::EDGE),
kTriggerDeletedEdgeLabel, 1);
ValidateVertexExistance(*client, allowed_trigger_types.contains(AllowedTriggerType::OBJECT),
kTriggerDeletedObjectLabel, 1);
ValidateVertexExistance(*client, allowed_trigger_types.contains(AllowedTriggerType::VERTEX),
kTriggerDeletedVertexLabel, 2);
ValidateVertexExistance(*client, allowed_trigger_types.contains(AllowedTriggerType::OBJECT),
kTriggerDeletedObjectLabel, 2);
DropOnDeleteTriggers(*client, allowed_trigger_types);
client->Execute("MATCH (n) DETACH DELETE n;");
client->DiscardAll();
};
const auto run_delete_trigger_write_procedure_delete_detach_test =
[&](const std::unordered_set<AllowedTriggerType> &allowed_trigger_types) {
ExecuteCreateVertex(*client, 2);
ExecuteCreateVertex(*client, 3);
client->Execute("MATCH (n {id:2}), (m {id:3}) CALL write.create_edge(n, m, 'edge') YIELD e RETURN e");
client->DiscardAll();
CreateOnDeleteTriggers(*client, true, allowed_trigger_types);
client->Execute("MATCH (v {id:2}) CALL write.detach_delete_vertex(v)");
client->DiscardAll();
auto number_of_expected_vertices = 1;
for (const auto allowed_trigger_type : allowed_trigger_types) {
switch (allowed_trigger_type) {
case AllowedTriggerType::VERTEX:
number_of_expected_vertices += 1;
break;
case AllowedTriggerType::EDGE:
number_of_expected_vertices += 1;
break;
case AllowedTriggerType::OBJECT:
number_of_expected_vertices += 2;
break;
}
}
CheckNumberOfAllVertices(*client, number_of_expected_vertices);
ValidateVertexExistance(*client, allowed_trigger_types.contains(AllowedTriggerType::EDGE),
kTriggerDeletedEdgeLabel, 1);
ValidateVertexExistance(*client, allowed_trigger_types.contains(AllowedTriggerType::OBJECT),
kTriggerDeletedObjectLabel, 1);
ValidateVertexExistance(*client, allowed_trigger_types.contains(AllowedTriggerType::VERTEX),
kTriggerDeletedVertexLabel, 2);
ValidateVertexExistance(*client, allowed_trigger_types.contains(AllowedTriggerType::OBJECT),
kTriggerDeletedObjectLabel, 2);
DropOnDeleteTriggers(*client, allowed_trigger_types);
client->Execute("MATCH (n) DETACH DELETE n;");
client->DiscardAll();
};
const auto run_for_trigger_combinations = [&](const bool is_before) {
const std::array trigger_type_combinations{
std::unordered_set{AllowedTriggerType::VERTEX},
std::unordered_set{AllowedTriggerType::EDGE},
std::unordered_set{AllowedTriggerType::OBJECT},
std::unordered_set{AllowedTriggerType::VERTEX, AllowedTriggerType::EDGE},
std::unordered_set{AllowedTriggerType::VERTEX, AllowedTriggerType::OBJECT},
std::unordered_set{AllowedTriggerType::EDGE, AllowedTriggerType::OBJECT},
std::unordered_set{AllowedTriggerType::VERTEX, AllowedTriggerType::EDGE, AllowedTriggerType::OBJECT},
};
for (const auto &allowed_trigger_types : trigger_type_combinations) {
run_delete_trigger_tests(is_before, allowed_trigger_types);
run_delete_trigger_write_procedure_tests(allowed_trigger_types);
run_delete_trigger_write_procedure_delete_detach_test(allowed_trigger_types);
}
};
constexpr bool kBeforeCommit = true;
constexpr bool kAfterCommit = false;
run_for_trigger_combinations(kBeforeCommit);
run_for_trigger_combinations(kAfterCommit);
return 0;
}