2017-08-22 22:24:40 +08:00
|
|
|
#include <string>
|
|
|
|
|
|
|
|
#include <benchmark/benchmark_api.h>
|
|
|
|
|
|
|
|
#include "query/frontend/semantic/symbol_generator.hpp"
|
2017-09-07 22:23:59 +08:00
|
|
|
#include "query/plan/cost_estimator.hpp"
|
2017-08-22 22:24:40 +08:00
|
|
|
#include "query/plan/planner.hpp"
|
2017-09-07 22:23:59 +08:00
|
|
|
#include "query/plan/vertex_count_cache.hpp"
|
2019-11-22 00:24:01 +08:00
|
|
|
#include "storage/v2/storage.hpp"
|
2017-08-22 22:24:40 +08:00
|
|
|
|
|
|
|
// Add chained MATCH (node1) -- (node2), MATCH (node2) -- (node3) ... clauses.
|
2018-10-19 22:18:44 +08:00
|
|
|
static query::CypherQuery *AddChainedMatches(int num_matches,
|
|
|
|
query::AstStorage &storage) {
|
|
|
|
auto *query = storage.Create<query::CypherQuery>();
|
2017-08-22 22:24:40 +08:00
|
|
|
for (int i = 0; i < num_matches; ++i) {
|
|
|
|
auto *match = storage.Create<query::Match>();
|
|
|
|
auto *pattern = storage.Create<query::Pattern>();
|
2017-11-29 20:55:02 +08:00
|
|
|
auto *single_query = storage.Create<query::SingleQuery>();
|
2017-08-22 22:24:40 +08:00
|
|
|
pattern->identifier_ = storage.Create<query::Identifier>("path");
|
|
|
|
match->patterns_.emplace_back(pattern);
|
|
|
|
std::string node1_name = "node" + std::to_string(i - 1);
|
|
|
|
pattern->atoms_.emplace_back(storage.Create<query::NodeAtom>(
|
|
|
|
storage.Create<query::Identifier>(node1_name)));
|
|
|
|
pattern->atoms_.emplace_back(storage.Create<query::EdgeAtom>(
|
|
|
|
storage.Create<query::Identifier>("edge" + std::to_string(i)),
|
2017-10-13 15:46:49 +08:00
|
|
|
query::EdgeAtom::Type::SINGLE, query::EdgeAtom::Direction::BOTH));
|
2017-08-22 22:24:40 +08:00
|
|
|
pattern->atoms_.emplace_back(storage.Create<query::NodeAtom>(
|
|
|
|
storage.Create<query::Identifier>("node" + std::to_string(i))));
|
2017-11-29 20:55:02 +08:00
|
|
|
single_query->clauses_.emplace_back(match);
|
2018-10-10 21:19:34 +08:00
|
|
|
query->single_query_ = single_query;
|
2017-08-22 22:24:40 +08:00
|
|
|
}
|
2018-10-10 21:19:34 +08:00
|
|
|
return query;
|
2017-08-22 22:24:40 +08:00
|
|
|
}
|
|
|
|
|
2017-09-07 22:23:59 +08:00
|
|
|
static void BM_PlanChainedMatches(benchmark::State &state) {
|
2019-11-22 00:24:01 +08:00
|
|
|
storage::Storage db;
|
|
|
|
auto storage_dba = db.Access();
|
|
|
|
query::DbAccessor dba(&storage_dba);
|
2017-08-22 22:24:40 +08:00
|
|
|
while (state.KeepRunning()) {
|
|
|
|
state.PauseTiming();
|
2018-05-22 22:45:52 +08:00
|
|
|
query::AstStorage storage;
|
2017-08-22 22:24:40 +08:00
|
|
|
int num_matches = state.range(0);
|
2018-10-10 21:19:34 +08:00
|
|
|
auto *query = AddChainedMatches(num_matches, storage);
|
2018-10-30 22:29:12 +08:00
|
|
|
auto symbol_table = query::MakeSymbolTable(query);
|
2019-11-22 00:24:01 +08:00
|
|
|
auto ctx =
|
|
|
|
query::plan::MakePlanningContext(&storage, &symbol_table, query, &dba);
|
2017-08-22 22:24:40 +08:00
|
|
|
state.ResumeTiming();
|
2018-10-10 21:19:34 +08:00
|
|
|
auto query_parts =
|
|
|
|
query::plan::CollectQueryParts(symbol_table, storage, query);
|
2017-11-29 20:55:02 +08:00
|
|
|
if (query_parts.query_parts.size() == 0) {
|
|
|
|
std::exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
auto single_query_parts = query_parts.query_parts.at(0).single_query_parts;
|
|
|
|
auto plans = query::plan::MakeLogicalPlanForSingleQuery<
|
Remove GraphDbAccessor and storage types from Ast
Summary:
This diff removes the need for a database when parsing a query and
creating an Ast. Instead of storing storage::{Label,Property,EdgeType}
in Ast nodes, we store the name and an index into all of the names. This
allows for easy creation of a map from {Label,Property,EdgeType} index
into the concrete storage type. Obviously, this comes with a performance
penalty during execution, but it should be minor. The upside is that the
query/frontend minimally depends on storage (PropertyValue), which makes
writing tests easier as well as running them a lot faster (there is no
database setup). This is most noticeable in the ast_serialization test
which took a long time due to start up of a distributed database.
Reviewers: mtomic, llugovic
Reviewed By: mtomic
Subscribers: mferencevic, pullbot
Differential Revision: https://phabricator.memgraph.io/D1774
2019-01-14 21:41:37 +08:00
|
|
|
query::plan::VariableStartPlanner>(single_query_parts, &ctx);
|
2017-10-13 15:46:49 +08:00
|
|
|
for (const auto &plan : plans) {
|
|
|
|
// Exhaust through all generated plans, since they are lazily generated.
|
2017-11-29 20:55:02 +08:00
|
|
|
benchmark::DoNotOptimize(plan.get());
|
2017-10-13 15:46:49 +08:00
|
|
|
}
|
2017-08-22 22:24:40 +08:00
|
|
|
}
|
2017-09-07 22:23:59 +08:00
|
|
|
}
|
2017-08-22 22:24:40 +08:00
|
|
|
|
2017-09-07 22:23:59 +08:00
|
|
|
BENCHMARK(BM_PlanChainedMatches)
|
2017-08-22 22:24:40 +08:00
|
|
|
->RangeMultiplier(2)
|
|
|
|
->Range(50, 400)
|
|
|
|
->Unit(benchmark::kMillisecond);
|
|
|
|
|
Remove GraphDbAccessor and storage types from Ast
Summary:
This diff removes the need for a database when parsing a query and
creating an Ast. Instead of storing storage::{Label,Property,EdgeType}
in Ast nodes, we store the name and an index into all of the names. This
allows for easy creation of a map from {Label,Property,EdgeType} index
into the concrete storage type. Obviously, this comes with a performance
penalty during execution, but it should be minor. The upside is that the
query/frontend minimally depends on storage (PropertyValue), which makes
writing tests easier as well as running them a lot faster (there is no
database setup). This is most noticeable in the ast_serialization test
which took a long time due to start up of a distributed database.
Reviewers: mtomic, llugovic
Reviewed By: mtomic
Subscribers: mferencevic, pullbot
Differential Revision: https://phabricator.memgraph.io/D1774
2019-01-14 21:41:37 +08:00
|
|
|
static query::CypherQuery *AddIndexedMatches(int num_matches,
|
|
|
|
const std::string &label,
|
|
|
|
const std::string &property,
|
|
|
|
query::AstStorage &storage) {
|
2018-10-19 22:18:44 +08:00
|
|
|
auto *query = storage.Create<query::CypherQuery>();
|
2017-09-07 22:23:59 +08:00
|
|
|
for (int i = 0; i < num_matches; ++i) {
|
|
|
|
auto *match = storage.Create<query::Match>();
|
|
|
|
auto *pattern = storage.Create<query::Pattern>();
|
2017-11-29 20:55:02 +08:00
|
|
|
auto *single_query = storage.Create<query::SingleQuery>();
|
2017-09-07 22:23:59 +08:00
|
|
|
pattern->identifier_ = storage.Create<query::Identifier>("path");
|
|
|
|
match->patterns_.emplace_back(pattern);
|
|
|
|
std::string node1_name = "node" + std::to_string(i - 1);
|
|
|
|
auto *node = storage.Create<query::NodeAtom>(
|
|
|
|
storage.Create<query::Identifier>(node1_name));
|
Remove GraphDbAccessor and storage types from Ast
Summary:
This diff removes the need for a database when parsing a query and
creating an Ast. Instead of storing storage::{Label,Property,EdgeType}
in Ast nodes, we store the name and an index into all of the names. This
allows for easy creation of a map from {Label,Property,EdgeType} index
into the concrete storage type. Obviously, this comes with a performance
penalty during execution, but it should be minor. The upside is that the
query/frontend minimally depends on storage (PropertyValue), which makes
writing tests easier as well as running them a lot faster (there is no
database setup). This is most noticeable in the ast_serialization test
which took a long time due to start up of a distributed database.
Reviewers: mtomic, llugovic
Reviewed By: mtomic
Subscribers: mferencevic, pullbot
Differential Revision: https://phabricator.memgraph.io/D1774
2019-01-14 21:41:37 +08:00
|
|
|
node->labels_.emplace_back(storage.GetLabelIx(label));
|
|
|
|
node->properties_[storage.GetPropertyIx(property)] =
|
|
|
|
storage.Create<query::PrimitiveLiteral>(i);
|
2017-09-07 22:23:59 +08:00
|
|
|
pattern->atoms_.emplace_back(node);
|
2017-11-29 20:55:02 +08:00
|
|
|
single_query->clauses_.emplace_back(match);
|
2018-10-10 21:19:34 +08:00
|
|
|
query->single_query_ = single_query;
|
2017-09-07 22:23:59 +08:00
|
|
|
}
|
2018-10-10 21:19:34 +08:00
|
|
|
return query;
|
2017-09-07 22:23:59 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static auto CreateIndexedVertices(int index_count, int vertex_count,
|
2019-11-22 00:24:01 +08:00
|
|
|
storage::Storage *db) {
|
|
|
|
auto label = db->NameToLabel("label");
|
|
|
|
auto prop = db->NameToProperty("prop");
|
|
|
|
db->CreateIndex(label, prop);
|
|
|
|
auto dba = db->Access();
|
2017-09-07 22:23:59 +08:00
|
|
|
for (int vi = 0; vi < vertex_count; ++vi) {
|
|
|
|
for (int index = 0; index < index_count; ++index) {
|
2019-11-22 00:24:01 +08:00
|
|
|
auto vertex = dba.CreateVertex();
|
|
|
|
CHECK(vertex.AddLabel(label).HasValue());
|
|
|
|
CHECK(vertex.SetProperty(prop, PropertyValue(index)).HasValue());
|
2017-09-07 22:23:59 +08:00
|
|
|
}
|
|
|
|
}
|
2019-11-22 00:24:01 +08:00
|
|
|
CHECK(!dba.Commit().HasError());
|
Remove GraphDbAccessor and storage types from Ast
Summary:
This diff removes the need for a database when parsing a query and
creating an Ast. Instead of storing storage::{Label,Property,EdgeType}
in Ast nodes, we store the name and an index into all of the names. This
allows for easy creation of a map from {Label,Property,EdgeType} index
into the concrete storage type. Obviously, this comes with a performance
penalty during execution, but it should be minor. The upside is that the
query/frontend minimally depends on storage (PropertyValue), which makes
writing tests easier as well as running them a lot faster (there is no
database setup). This is most noticeable in the ast_serialization test
which took a long time due to start up of a distributed database.
Reviewers: mtomic, llugovic
Reviewed By: mtomic
Subscribers: mferencevic, pullbot
Differential Revision: https://phabricator.memgraph.io/D1774
2019-01-14 21:41:37 +08:00
|
|
|
return std::make_pair("label", "prop");
|
2017-09-07 22:23:59 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void BM_PlanAndEstimateIndexedMatching(benchmark::State &state) {
|
2019-11-22 00:24:01 +08:00
|
|
|
storage::Storage db;
|
Remove GraphDbAccessor and storage types from Ast
Summary:
This diff removes the need for a database when parsing a query and
creating an Ast. Instead of storing storage::{Label,Property,EdgeType}
in Ast nodes, we store the name and an index into all of the names. This
allows for easy creation of a map from {Label,Property,EdgeType} index
into the concrete storage type. Obviously, this comes with a performance
penalty during execution, but it should be minor. The upside is that the
query/frontend minimally depends on storage (PropertyValue), which makes
writing tests easier as well as running them a lot faster (there is no
database setup). This is most noticeable in the ast_serialization test
which took a long time due to start up of a distributed database.
Reviewers: mtomic, llugovic
Reviewed By: mtomic
Subscribers: mferencevic, pullbot
Differential Revision: https://phabricator.memgraph.io/D1774
2019-01-14 21:41:37 +08:00
|
|
|
std::string label;
|
|
|
|
std::string prop;
|
2017-09-07 22:23:59 +08:00
|
|
|
int index_count = state.range(0);
|
|
|
|
int vertex_count = state.range(1);
|
2019-11-22 00:24:01 +08:00
|
|
|
std::tie(label, prop) = CreateIndexedVertices(index_count, vertex_count, &db);
|
|
|
|
auto storage_dba = db.Access();
|
|
|
|
query::DbAccessor dba(&storage_dba);
|
2018-09-11 19:32:25 +08:00
|
|
|
query::Parameters parameters;
|
2017-09-07 22:23:59 +08:00
|
|
|
while (state.KeepRunning()) {
|
|
|
|
state.PauseTiming();
|
2018-05-22 22:45:52 +08:00
|
|
|
query::AstStorage storage;
|
Remove GraphDbAccessor and storage types from Ast
Summary:
This diff removes the need for a database when parsing a query and
creating an Ast. Instead of storing storage::{Label,Property,EdgeType}
in Ast nodes, we store the name and an index into all of the names. This
allows for easy creation of a map from {Label,Property,EdgeType} index
into the concrete storage type. Obviously, this comes with a performance
penalty during execution, but it should be minor. The upside is that the
query/frontend minimally depends on storage (PropertyValue), which makes
writing tests easier as well as running them a lot faster (there is no
database setup). This is most noticeable in the ast_serialization test
which took a long time due to start up of a distributed database.
Reviewers: mtomic, llugovic
Reviewed By: mtomic
Subscribers: mferencevic, pullbot
Differential Revision: https://phabricator.memgraph.io/D1774
2019-01-14 21:41:37 +08:00
|
|
|
auto *query = AddIndexedMatches(index_count, label, prop, storage);
|
2018-10-30 22:29:12 +08:00
|
|
|
auto symbol_table = query::MakeSymbolTable(query);
|
2017-09-07 22:23:59 +08:00
|
|
|
state.ResumeTiming();
|
2019-11-22 00:24:01 +08:00
|
|
|
auto ctx =
|
|
|
|
query::plan::MakePlanningContext(&storage, &symbol_table, query, &dba);
|
2018-10-10 21:19:34 +08:00
|
|
|
auto query_parts =
|
|
|
|
query::plan::CollectQueryParts(symbol_table, storage, query);
|
2017-11-29 20:55:02 +08:00
|
|
|
if (query_parts.query_parts.size() == 0) {
|
|
|
|
std::exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
auto single_query_parts = query_parts.query_parts.at(0).single_query_parts;
|
|
|
|
auto plans = query::plan::MakeLogicalPlanForSingleQuery<
|
Remove GraphDbAccessor and storage types from Ast
Summary:
This diff removes the need for a database when parsing a query and
creating an Ast. Instead of storing storage::{Label,Property,EdgeType}
in Ast nodes, we store the name and an index into all of the names. This
allows for easy creation of a map from {Label,Property,EdgeType} index
into the concrete storage type. Obviously, this comes with a performance
penalty during execution, but it should be minor. The upside is that the
query/frontend minimally depends on storage (PropertyValue), which makes
writing tests easier as well as running them a lot faster (there is no
database setup). This is most noticeable in the ast_serialization test
which took a long time due to start up of a distributed database.
Reviewers: mtomic, llugovic
Reviewed By: mtomic
Subscribers: mferencevic, pullbot
Differential Revision: https://phabricator.memgraph.io/D1774
2019-01-14 21:41:37 +08:00
|
|
|
query::plan::VariableStartPlanner>(single_query_parts, &ctx);
|
2017-10-13 15:46:49 +08:00
|
|
|
for (auto plan : plans) {
|
2019-04-15 17:36:43 +08:00
|
|
|
query::plan::EstimatePlanCost(&dba, parameters, *plan);
|
2017-09-07 22:23:59 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void BM_PlanAndEstimateIndexedMatchingWithCachedCounts(
|
|
|
|
benchmark::State &state) {
|
2019-11-22 00:24:01 +08:00
|
|
|
storage::Storage db;
|
Remove GraphDbAccessor and storage types from Ast
Summary:
This diff removes the need for a database when parsing a query and
creating an Ast. Instead of storing storage::{Label,Property,EdgeType}
in Ast nodes, we store the name and an index into all of the names. This
allows for easy creation of a map from {Label,Property,EdgeType} index
into the concrete storage type. Obviously, this comes with a performance
penalty during execution, but it should be minor. The upside is that the
query/frontend minimally depends on storage (PropertyValue), which makes
writing tests easier as well as running them a lot faster (there is no
database setup). This is most noticeable in the ast_serialization test
which took a long time due to start up of a distributed database.
Reviewers: mtomic, llugovic
Reviewed By: mtomic
Subscribers: mferencevic, pullbot
Differential Revision: https://phabricator.memgraph.io/D1774
2019-01-14 21:41:37 +08:00
|
|
|
std::string label;
|
|
|
|
std::string prop;
|
2017-09-07 22:23:59 +08:00
|
|
|
int index_count = state.range(0);
|
|
|
|
int vertex_count = state.range(1);
|
2019-11-22 00:24:01 +08:00
|
|
|
std::tie(label, prop) = CreateIndexedVertices(index_count, vertex_count, &db);
|
|
|
|
auto storage_dba = db.Access();
|
|
|
|
query::DbAccessor dba(&storage_dba);
|
|
|
|
auto vertex_counts = query::plan::MakeVertexCountCache(&dba);
|
2018-09-11 19:32:25 +08:00
|
|
|
query::Parameters parameters;
|
2017-09-07 22:23:59 +08:00
|
|
|
while (state.KeepRunning()) {
|
|
|
|
state.PauseTiming();
|
2018-05-22 22:45:52 +08:00
|
|
|
query::AstStorage storage;
|
Remove GraphDbAccessor and storage types from Ast
Summary:
This diff removes the need for a database when parsing a query and
creating an Ast. Instead of storing storage::{Label,Property,EdgeType}
in Ast nodes, we store the name and an index into all of the names. This
allows for easy creation of a map from {Label,Property,EdgeType} index
into the concrete storage type. Obviously, this comes with a performance
penalty during execution, but it should be minor. The upside is that the
query/frontend minimally depends on storage (PropertyValue), which makes
writing tests easier as well as running them a lot faster (there is no
database setup). This is most noticeable in the ast_serialization test
which took a long time due to start up of a distributed database.
Reviewers: mtomic, llugovic
Reviewed By: mtomic
Subscribers: mferencevic, pullbot
Differential Revision: https://phabricator.memgraph.io/D1774
2019-01-14 21:41:37 +08:00
|
|
|
auto *query = AddIndexedMatches(index_count, label, prop, storage);
|
2018-10-30 22:29:12 +08:00
|
|
|
auto symbol_table = query::MakeSymbolTable(query);
|
2017-09-07 22:23:59 +08:00
|
|
|
state.ResumeTiming();
|
Remove GraphDbAccessor and storage types from Ast
Summary:
This diff removes the need for a database when parsing a query and
creating an Ast. Instead of storing storage::{Label,Property,EdgeType}
in Ast nodes, we store the name and an index into all of the names. This
allows for easy creation of a map from {Label,Property,EdgeType} index
into the concrete storage type. Obviously, this comes with a performance
penalty during execution, but it should be minor. The upside is that the
query/frontend minimally depends on storage (PropertyValue), which makes
writing tests easier as well as running them a lot faster (there is no
database setup). This is most noticeable in the ast_serialization test
which took a long time due to start up of a distributed database.
Reviewers: mtomic, llugovic
Reviewed By: mtomic
Subscribers: mferencevic, pullbot
Differential Revision: https://phabricator.memgraph.io/D1774
2019-01-14 21:41:37 +08:00
|
|
|
auto ctx = query::plan::MakePlanningContext(&storage, &symbol_table, query,
|
|
|
|
&vertex_counts);
|
2018-10-10 21:19:34 +08:00
|
|
|
auto query_parts =
|
|
|
|
query::plan::CollectQueryParts(symbol_table, storage, query);
|
2017-11-29 20:55:02 +08:00
|
|
|
if (query_parts.query_parts.size() == 0) {
|
|
|
|
std::exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
auto single_query_parts = query_parts.query_parts.at(0).single_query_parts;
|
|
|
|
auto plans = query::plan::MakeLogicalPlanForSingleQuery<
|
Remove GraphDbAccessor and storage types from Ast
Summary:
This diff removes the need for a database when parsing a query and
creating an Ast. Instead of storing storage::{Label,Property,EdgeType}
in Ast nodes, we store the name and an index into all of the names. This
allows for easy creation of a map from {Label,Property,EdgeType} index
into the concrete storage type. Obviously, this comes with a performance
penalty during execution, but it should be minor. The upside is that the
query/frontend minimally depends on storage (PropertyValue), which makes
writing tests easier as well as running them a lot faster (there is no
database setup). This is most noticeable in the ast_serialization test
which took a long time due to start up of a distributed database.
Reviewers: mtomic, llugovic
Reviewed By: mtomic
Subscribers: mferencevic, pullbot
Differential Revision: https://phabricator.memgraph.io/D1774
2019-01-14 21:41:37 +08:00
|
|
|
query::plan::VariableStartPlanner>(single_query_parts, &ctx);
|
2017-10-13 15:46:49 +08:00
|
|
|
for (auto plan : plans) {
|
Remove GraphDbAccessor and storage types from Ast
Summary:
This diff removes the need for a database when parsing a query and
creating an Ast. Instead of storing storage::{Label,Property,EdgeType}
in Ast nodes, we store the name and an index into all of the names. This
allows for easy creation of a map from {Label,Property,EdgeType} index
into the concrete storage type. Obviously, this comes with a performance
penalty during execution, but it should be minor. The upside is that the
query/frontend minimally depends on storage (PropertyValue), which makes
writing tests easier as well as running them a lot faster (there is no
database setup). This is most noticeable in the ast_serialization test
which took a long time due to start up of a distributed database.
Reviewers: mtomic, llugovic
Reviewed By: mtomic
Subscribers: mferencevic, pullbot
Differential Revision: https://phabricator.memgraph.io/D1774
2019-01-14 21:41:37 +08:00
|
|
|
query::plan::EstimatePlanCost(&vertex_counts, parameters, *plan);
|
2017-09-07 22:23:59 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
BENCHMARK(BM_PlanAndEstimateIndexedMatching)
|
|
|
|
->RangeMultiplier(4)
|
|
|
|
->Ranges({{1, 100}, {100, 1000}})
|
|
|
|
->Unit(benchmark::kMicrosecond);
|
|
|
|
|
|
|
|
BENCHMARK(BM_PlanAndEstimateIndexedMatchingWithCachedCounts)
|
|
|
|
->RangeMultiplier(4)
|
|
|
|
->Ranges({{1, 100}, {100, 1000}})
|
|
|
|
->Unit(benchmark::kMicrosecond);
|
|
|
|
|
2017-08-22 22:24:40 +08:00
|
|
|
BENCHMARK_MAIN();
|