memgraph/tests/e2e/triggers/on_delete_triggers.cpp

354 lines
14 KiB
C++
Raw Normal View History

// Copyright 2023 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.
2021-05-19 00:28:17 +08:00
#include <string>
#include <string_view>
2021-10-12 22:21:28 +08:00
#include <unordered_set>
2021-05-19 00:28:17 +08:00
#include <gflags/gflags.h>
#include <mgclient.hpp>
#include "common.hpp"
#include "utils/logging.hpp"
inline constexpr std::string_view kTriggerDeletedVertexLabel{"DELETED_VERTEX"};
inline constexpr std::string_view kTriggerDeletedEdgeLabel{"DELETED_EDGE"};
inline constexpr std::string_view kTriggerDeletedObjectLabel{"DELETED_OBJECT"};
2021-05-19 00:28:17 +08:00
inline constexpr std::string_view kTriggerNameOnVertexDeleteTrigger{"DeletedVerticesTrigger"};
inline constexpr std::string_view kTriggerNameOnEdgeDeleteTrigger{"DeletedEdgesTrigger"};
inline constexpr std::string_view kTriggerNameOnObjectDeleteTrigger{"DeletedObjectsTrigger"};
inline constexpr std::string_view kTriggerNameOnObjectDeleteTriggerProc{"DeletedObjectsTriggerProc"};
enum class AllowedTriggerType : uint8_t { VERTEX, EDGE, OBJECT, OBJECT_PROC };
2021-10-12 22:21:28 +08:00
2021-05-19 00:28:17 +08:00
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();
}
2021-10-12 22:21:28 +08:00
void CreateOnDeleteTriggers(mg::Client &client, bool is_before,
const std::unordered_set<AllowedTriggerType> &allowed_trigger_types) {
2021-05-19 00:28:17 +08:00
const std::string_view before_or_after = is_before ? "BEFORE" : "AFTER";
2021-10-12 22:21:28 +08:00
const auto create_on_vertex_delete_trigger = [&, before_or_after] {
client.Execute(
fmt::format("CREATE TRIGGER {} ON () DELETE "
2021-10-12 22:21:28 +08:00
"{} COMMIT "
"EXECUTE "
"UNWIND deletedVertices as deletedVertex "
"CREATE (n: {} {{ id: deletedVertex.id }})",
kTriggerNameOnVertexDeleteTrigger, before_or_after, kTriggerDeletedVertexLabel));
2021-10-12 22:21:28 +08:00
client.DiscardAll();
};
const auto create_on_edge_delete_trigger = [&, before_or_after] {
client.Execute(
fmt::format("CREATE TRIGGER {} ON --> DELETE "
2021-10-12 22:21:28 +08:00
"{} COMMIT "
"EXECUTE "
"UNWIND deletedEdges as deletedEdge "
"CREATE (n: {} {{ id: deletedEdge.id }})",
kTriggerNameOnEdgeDeleteTrigger, before_or_after, kTriggerDeletedEdgeLabel));
2021-10-12 22:21:28 +08:00
client.DiscardAll();
};
const auto create_on_object_delete_trigger = [&, before_or_after] {
client.Execute(
fmt::format("CREATE TRIGGER {} ON DELETE "
2021-10-12 22:21:28 +08:00
"{} 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 }})",
kTriggerNameOnObjectDeleteTrigger, before_or_after, kTriggerDeletedObjectLabel));
client.DiscardAll();
};
const auto create_on_object_delete_trigger_proc = [&, before_or_after] {
client.Execute(
fmt::format("CREATE TRIGGER {} ON DELETE "
"{} COMMIT "
"EXECUTE "
"CALL write.access_deleted(deletedObjects) YIELD status RETURN status;",
kTriggerNameOnObjectDeleteTriggerProc, before_or_after));
2021-10-12 22:21:28 +08:00
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;
case AllowedTriggerType::OBJECT_PROC:
create_on_object_delete_trigger_proc();
2021-10-12 22:21:28 +08:00
}
}
2021-05-19 00:28:17 +08:00
}
2021-10-12 22:21:28 +08:00
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(fmt::format("DROP TRIGGER {}", kTriggerNameOnVertexDeleteTrigger));
2021-10-12 22:21:28 +08:00
client.DiscardAll();
2022-02-11 18:29:41 +08:00
break;
2021-10-12 22:21:28 +08:00
}
case AllowedTriggerType::EDGE: {
client.Execute(fmt::format("DROP TRIGGER {}", kTriggerNameOnEdgeDeleteTrigger));
2021-10-12 22:21:28 +08:00
client.DiscardAll();
2022-02-11 18:29:41 +08:00
break;
2021-10-12 22:21:28 +08:00
}
case AllowedTriggerType::OBJECT: {
client.Execute(fmt::format("DROP TRIGGER {}", kTriggerNameOnObjectDeleteTrigger));
client.DiscardAll();
break;
}
case AllowedTriggerType::OBJECT_PROC: {
client.Execute(fmt::format("DROP TRIGGER {}", kTriggerNameOnObjectDeleteTriggerProc));
2021-10-12 22:21:28 +08:00
client.DiscardAll();
2022-02-11 18:29:41 +08:00
break;
2021-10-12 22:21:28 +08:00
}
}
}
2021-05-19 00:28:17 +08:00
}
struct EdgeInfo {
int from_vertex;
int to_vertex;
int edge_id;
};
2021-10-12 22:21:28 +08:00
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);
};
2021-05-19 00:28:17 +08:00
int main(int argc, char **argv) {
gflags::SetUsageMessage("Memgraph E2E ON DELETE Triggers");
gflags::ParseCommandLineFlags(&argc, &argv, true);
2022-02-22 20:33:45 +08:00
memgraph::logging::RedirectToStderr();
2021-05-19 00:28:17 +08:00
mg::Client::Init();
auto client = Connect();
2021-10-12 22:21:28 +08:00
const auto run_delete_trigger_tests = [&](const bool is_before,
const std::unordered_set<AllowedTriggerType> &allowed_trigger_types) {
static constexpr std::array vertex_ids{1, 2, 3, 4};
static constexpr std::array edges{EdgeInfo{vertex_ids[0], vertex_ids[1], 5},
EdgeInfo{vertex_ids[2], vertex_ids[3], 6}};
2021-05-19 00:28:17 +08:00
{
2021-10-12 22:21:28 +08:00
CreateOnDeleteTriggers(*client, is_before, allowed_trigger_types);
2021-05-19 00:28:17 +08:00
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
2021-10-12 22:21:28 +08:00
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;
case AllowedTriggerType::OBJECT_PROC:
break;
2021-10-12 22:21:28 +08:00
}
}
2021-05-19 00:28:17 +08:00
if (is_before) {
2021-10-12 22:21:28 +08:00
CheckNumberOfAllVertices(*client, number_of_expected_vertices);
2021-05-19 00:28:17 +08:00
} else {
2021-10-12 22:21:28 +08:00
WaitForNumberOfAllVertices(*client, number_of_expected_vertices);
2021-05-19 00:28:17 +08:00
}
2021-10-12 22:21:28 +08:00
ValidateVertexExistance(*client, allowed_trigger_types.contains(AllowedTriggerType::VERTEX),
kTriggerDeletedVertexLabel, vertex_ids[0]);
ValidateVertexExistance(*client, allowed_trigger_types.contains(AllowedTriggerType::OBJECT),
kTriggerDeletedObjectLabel, vertex_ids[0]);
2021-05-19 00:28:17 +08:00
for (const auto &edge : edges) {
2021-10-12 22:21:28 +08:00
ValidateVertexExistance(*client, allowed_trigger_types.contains(AllowedTriggerType::EDGE),
kTriggerDeletedEdgeLabel, edge.edge_id);
ValidateVertexExistance(*client, allowed_trigger_types.contains(AllowedTriggerType::OBJECT),
kTriggerDeletedObjectLabel, edge.edge_id);
2021-05-19 00:28:17 +08:00
}
2021-10-12 22:21:28 +08:00
DropOnDeleteTriggers(*client, allowed_trigger_types);
2021-05-19 00:28:17 +08:00
client->Execute("MATCH (n) DETACH DELETE n;");
client->DiscardAll();
}
};
2021-10-12 22:21:28 +08:00
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;
case AllowedTriggerType::OBJECT_PROC:
break;
break;
2021-10-12 22:21:28 +08:00
}
}
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;
case AllowedTriggerType::OBJECT_PROC:
break;
2021-10-12 22:21:28 +08:00
}
}
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);
}
};
static constexpr bool kBeforeCommit = true;
static constexpr bool kAfterCommit = false;
2021-10-12 22:21:28 +08:00
run_for_trigger_combinations(kBeforeCommit);
run_for_trigger_combinations(kAfterCommit);
const auto run_delete_trigger_access_deleted_object_in_proc = [&client]() {
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, false, {AllowedTriggerType::OBJECT_PROC});
client->Execute("MATCH (n) DETACH DELETE n;");
DropOnDeleteTriggers(*client, {AllowedTriggerType::OBJECT_PROC});
};
run_delete_trigger_access_deleted_object_in_proc();
2021-05-19 00:28:17 +08:00
return 0;
}