2017-04-18 21:19:42 +08:00
|
|
|
//
|
|
|
|
// Copyright 2017 Memgraph
|
|
|
|
// Created by Florijan Stamenkovic on 14.03.17.
|
|
|
|
//
|
|
|
|
|
2017-04-20 19:31:18 +08:00
|
|
|
#include <algorithm>
|
2017-04-18 21:19:42 +08:00
|
|
|
#include <iterator>
|
|
|
|
#include <memory>
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
#include "gmock/gmock.h"
|
|
|
|
#include "gtest/gtest.h"
|
|
|
|
|
|
|
|
#include "query/context.hpp"
|
|
|
|
#include "query/exceptions.hpp"
|
|
|
|
#include "query/plan/operator.hpp"
|
|
|
|
|
|
|
|
#include "query_plan_common.hpp"
|
|
|
|
|
|
|
|
using namespace query;
|
|
|
|
using namespace query::plan;
|
|
|
|
|
|
|
|
TEST(QueryPlan, Skip) {
|
2019-11-22 01:25:25 +08:00
|
|
|
storage::Storage db;
|
|
|
|
auto storage_dba = db.Access();
|
|
|
|
query::DbAccessor dba(&storage_dba);
|
2017-04-18 21:19:42 +08:00
|
|
|
|
2018-05-22 22:45:52 +08:00
|
|
|
AstStorage storage;
|
2017-04-18 21:19:42 +08:00
|
|
|
SymbolTable symbol_table;
|
|
|
|
|
|
|
|
auto n = MakeScanAll(storage, symbol_table, "n1");
|
|
|
|
auto skip = std::make_shared<plan::Skip>(n.op_, LITERAL(2));
|
|
|
|
|
2019-11-22 01:25:25 +08:00
|
|
|
auto context = MakeContext(storage, symbol_table, &dba);
|
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
|
|
|
EXPECT_EQ(0, PullAll(*skip, &context));
|
2017-04-18 21:19:42 +08:00
|
|
|
|
2019-04-15 17:36:43 +08:00
|
|
|
dba.InsertVertex();
|
|
|
|
dba.AdvanceCommand();
|
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
|
|
|
EXPECT_EQ(0, PullAll(*skip, &context));
|
2017-04-18 21:19:42 +08:00
|
|
|
|
2019-04-15 17:36:43 +08:00
|
|
|
dba.InsertVertex();
|
|
|
|
dba.AdvanceCommand();
|
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
|
|
|
EXPECT_EQ(0, PullAll(*skip, &context));
|
2017-04-18 21:19:42 +08:00
|
|
|
|
2019-04-15 17:36:43 +08:00
|
|
|
dba.InsertVertex();
|
|
|
|
dba.AdvanceCommand();
|
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
|
|
|
EXPECT_EQ(1, PullAll(*skip, &context));
|
2017-04-18 21:19:42 +08:00
|
|
|
|
2019-04-15 17:36:43 +08:00
|
|
|
for (int i = 0; i < 10; ++i) dba.InsertVertex();
|
|
|
|
dba.AdvanceCommand();
|
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
|
|
|
EXPECT_EQ(11, PullAll(*skip, &context));
|
2017-04-18 21:19:42 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(QueryPlan, Limit) {
|
2019-11-22 01:25:25 +08:00
|
|
|
storage::Storage db;
|
|
|
|
auto storage_dba = db.Access();
|
|
|
|
query::DbAccessor dba(&storage_dba);
|
2017-04-18 21:19:42 +08:00
|
|
|
|
2018-05-22 22:45:52 +08:00
|
|
|
AstStorage storage;
|
2017-04-18 21:19:42 +08:00
|
|
|
SymbolTable symbol_table;
|
|
|
|
|
|
|
|
auto n = MakeScanAll(storage, symbol_table, "n1");
|
|
|
|
auto skip = std::make_shared<plan::Limit>(n.op_, LITERAL(2));
|
|
|
|
|
2019-11-22 01:25:25 +08:00
|
|
|
auto context = MakeContext(storage, symbol_table, &dba);
|
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
|
|
|
EXPECT_EQ(0, PullAll(*skip, &context));
|
2017-04-18 21:19:42 +08:00
|
|
|
|
2019-04-15 17:36:43 +08:00
|
|
|
dba.InsertVertex();
|
|
|
|
dba.AdvanceCommand();
|
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
|
|
|
EXPECT_EQ(1, PullAll(*skip, &context));
|
2017-04-18 21:19:42 +08:00
|
|
|
|
2019-04-15 17:36:43 +08:00
|
|
|
dba.InsertVertex();
|
|
|
|
dba.AdvanceCommand();
|
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
|
|
|
EXPECT_EQ(2, PullAll(*skip, &context));
|
2017-04-18 21:19:42 +08:00
|
|
|
|
2019-04-15 17:36:43 +08:00
|
|
|
dba.InsertVertex();
|
|
|
|
dba.AdvanceCommand();
|
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
|
|
|
EXPECT_EQ(2, PullAll(*skip, &context));
|
2017-04-18 21:19:42 +08:00
|
|
|
|
2019-04-15 17:36:43 +08:00
|
|
|
for (int i = 0; i < 10; ++i) dba.InsertVertex();
|
|
|
|
dba.AdvanceCommand();
|
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
|
|
|
EXPECT_EQ(2, PullAll(*skip, &context));
|
2017-04-18 21:19:42 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(QueryPlan, CreateLimit) {
|
|
|
|
// CREATE (n), (m)
|
|
|
|
// MATCH (n) CREATE (m) LIMIT 1
|
|
|
|
// in the end we need to have 3 vertices in the db
|
2019-11-22 01:25:25 +08:00
|
|
|
storage::Storage db;
|
|
|
|
auto storage_dba = db.Access();
|
|
|
|
query::DbAccessor dba(&storage_dba);
|
2019-04-15 17:36:43 +08:00
|
|
|
dba.InsertVertex();
|
|
|
|
dba.InsertVertex();
|
|
|
|
dba.AdvanceCommand();
|
2017-04-18 21:19:42 +08:00
|
|
|
|
2018-05-22 22:45:52 +08:00
|
|
|
AstStorage storage;
|
2017-04-18 21:19:42 +08:00
|
|
|
SymbolTable symbol_table;
|
|
|
|
|
|
|
|
auto n = MakeScanAll(storage, symbol_table, "n1");
|
2018-12-20 16:38:23 +08:00
|
|
|
NodeCreationInfo m;
|
|
|
|
m.symbol = symbol_table.CreateSymbol("m", true);
|
2018-08-30 19:31:50 +08:00
|
|
|
auto c = std::make_shared<CreateNode>(n.op_, m);
|
2017-04-18 21:19:42 +08:00
|
|
|
auto skip = std::make_shared<plan::Limit>(c, LITERAL(1));
|
|
|
|
|
2019-11-22 01:25:25 +08:00
|
|
|
auto context = MakeContext(storage, symbol_table, &dba);
|
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
|
|
|
EXPECT_EQ(1, PullAll(*skip, &context));
|
2019-04-15 17:36:43 +08:00
|
|
|
dba.AdvanceCommand();
|
2019-11-22 01:25:25 +08:00
|
|
|
EXPECT_EQ(3, CountIterable(dba.Vertices(storage::View::OLD)));
|
2017-04-18 21:19:42 +08:00
|
|
|
}
|
2017-04-20 19:31:18 +08:00
|
|
|
|
|
|
|
TEST(QueryPlan, OrderBy) {
|
2019-11-22 01:25:25 +08:00
|
|
|
storage::Storage db;
|
|
|
|
auto storage_dba = db.Access();
|
|
|
|
query::DbAccessor dba(&storage_dba);
|
2018-05-22 22:45:52 +08:00
|
|
|
AstStorage storage;
|
2017-04-20 19:31:18 +08:00
|
|
|
SymbolTable symbol_table;
|
2019-11-22 01:25:25 +08:00
|
|
|
auto prop = dba.NameToProperty("prop");
|
2017-04-20 19:31:18 +08:00
|
|
|
|
|
|
|
// contains a series of tests
|
|
|
|
// each test defines the ordering a vector of values in the desired order
|
2020-01-22 23:20:13 +08:00
|
|
|
auto Null = storage::PropertyValue();
|
|
|
|
std::vector<std::pair<Ordering, std::vector<storage::PropertyValue>>>
|
|
|
|
orderable{
|
|
|
|
{Ordering::ASC,
|
|
|
|
{storage::PropertyValue(0), storage::PropertyValue(0),
|
|
|
|
storage::PropertyValue(0.5), storage::PropertyValue(1),
|
|
|
|
storage::PropertyValue(2), storage::PropertyValue(12.6),
|
|
|
|
storage::PropertyValue(42), Null, Null}},
|
|
|
|
{Ordering::ASC,
|
|
|
|
{storage::PropertyValue(false), storage::PropertyValue(false),
|
|
|
|
storage::PropertyValue(true), storage::PropertyValue(true), Null,
|
|
|
|
Null}},
|
|
|
|
{Ordering::ASC,
|
|
|
|
{storage::PropertyValue("A"), storage::PropertyValue("B"),
|
|
|
|
storage::PropertyValue("a"), storage::PropertyValue("a"),
|
|
|
|
storage::PropertyValue("aa"), storage::PropertyValue("ab"),
|
|
|
|
storage::PropertyValue("aba"), Null, Null}},
|
|
|
|
{Ordering::DESC,
|
|
|
|
{Null, Null, storage::PropertyValue(33), storage::PropertyValue(33),
|
|
|
|
storage::PropertyValue(32.5), storage::PropertyValue(32),
|
|
|
|
storage::PropertyValue(2.2), storage::PropertyValue(2.1),
|
|
|
|
storage::PropertyValue(0)}},
|
|
|
|
{Ordering::DESC,
|
|
|
|
{Null, storage::PropertyValue(true), storage::PropertyValue(false)}},
|
|
|
|
{Ordering::DESC,
|
|
|
|
{Null, storage::PropertyValue("zorro"),
|
|
|
|
storage::PropertyValue("borro")}}};
|
2017-04-20 19:31:18 +08:00
|
|
|
|
|
|
|
for (const auto &order_value_pair : orderable) {
|
2019-06-06 19:16:37 +08:00
|
|
|
std::vector<TypedValue> values;
|
|
|
|
values.reserve(order_value_pair.second.size());
|
|
|
|
for (const auto &v : order_value_pair.second) values.emplace_back(v);
|
2017-04-20 19:31:18 +08:00
|
|
|
// empty database
|
2019-11-22 01:25:25 +08:00
|
|
|
for (auto vertex : dba.Vertices(storage::View::OLD))
|
2020-01-14 17:02:09 +08:00
|
|
|
ASSERT_TRUE(dba.DetachRemoveVertex(&vertex).HasValue());
|
2017-10-30 17:43:25 +08:00
|
|
|
dba.AdvanceCommand();
|
2019-11-22 01:25:25 +08:00
|
|
|
ASSERT_EQ(0, CountIterable(dba.Vertices(storage::View::OLD)));
|
2017-04-20 19:31:18 +08:00
|
|
|
|
|
|
|
// take some effort to shuffle the values
|
|
|
|
// because we are testing that something not ordered gets ordered
|
|
|
|
// and need to take care it does not happen by accident
|
2019-06-06 19:16:37 +08:00
|
|
|
auto shuffled = values;
|
2017-04-20 19:31:18 +08:00
|
|
|
auto order_equal = [&values, &shuffled]() {
|
|
|
|
return std::equal(values.begin(), values.end(), shuffled.begin(),
|
|
|
|
TypedValue::BoolEqual{});
|
|
|
|
};
|
|
|
|
for (int i = 0; i < 50 && order_equal(); ++i) {
|
|
|
|
std::random_shuffle(shuffled.begin(), shuffled.end());
|
|
|
|
}
|
|
|
|
ASSERT_FALSE(order_equal());
|
|
|
|
|
|
|
|
// create the vertices
|
2019-06-06 19:16:37 +08:00
|
|
|
for (const auto &value : shuffled)
|
2019-11-22 01:25:25 +08:00
|
|
|
ASSERT_TRUE(dba.InsertVertex()
|
2020-01-22 23:20:13 +08:00
|
|
|
.SetProperty(prop, storage::PropertyValue(value))
|
2019-11-22 01:25:25 +08:00
|
|
|
.HasValue());
|
2017-10-30 17:43:25 +08:00
|
|
|
dba.AdvanceCommand();
|
2017-04-20 19:31:18 +08:00
|
|
|
|
|
|
|
// order by and collect results
|
|
|
|
auto n = MakeScanAll(storage, symbol_table, "n");
|
2019-02-05 20:16:07 +08:00
|
|
|
auto n_p = PROPERTY_LOOKUP(IDENT("n")->MapTo(n.sym_), prop);
|
2017-04-20 19:31:18 +08:00
|
|
|
auto order_by = std::make_shared<plan::OrderBy>(
|
2018-10-04 17:57:23 +08:00
|
|
|
n.op_, std::vector<SortItem>{{order_value_pair.first, n_p}},
|
2017-04-20 19:31:18 +08:00
|
|
|
std::vector<Symbol>{n.sym_});
|
2019-02-05 20:16:07 +08:00
|
|
|
auto n_p_ne =
|
|
|
|
NEXPR("n.p", n_p)->MapTo(symbol_table.CreateSymbol("n.p", true));
|
2017-04-20 19:31:18 +08:00
|
|
|
auto produce = MakeProduce(order_by, n_p_ne);
|
2019-11-22 01:25:25 +08:00
|
|
|
auto context = MakeContext(storage, symbol_table, &dba);
|
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 results = CollectProduce(*produce, &context);
|
2017-04-20 19:31:18 +08:00
|
|
|
ASSERT_EQ(values.size(), results.size());
|
|
|
|
for (int j = 0; j < results.size(); ++j)
|
|
|
|
EXPECT_TRUE(TypedValue::BoolEqual{}(results[j][0], values[j]));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(QueryPlan, OrderByMultiple) {
|
2019-11-22 01:25:25 +08:00
|
|
|
storage::Storage db;
|
|
|
|
auto storage_dba = db.Access();
|
|
|
|
query::DbAccessor dba(&storage_dba);
|
2018-05-22 22:45:52 +08:00
|
|
|
AstStorage storage;
|
2017-04-20 19:31:18 +08:00
|
|
|
SymbolTable symbol_table;
|
|
|
|
|
2019-11-22 01:25:25 +08:00
|
|
|
auto p1 = dba.NameToProperty("p1");
|
|
|
|
auto p2 = dba.NameToProperty("p2");
|
2017-04-20 19:31:18 +08:00
|
|
|
|
|
|
|
// create a bunch of vertices that in two properties
|
|
|
|
// have all the variations (with repetition) of N values.
|
|
|
|
// ensure that those vertices are not created in the
|
|
|
|
// "right" sequence, but randomized
|
|
|
|
const int N = 20;
|
|
|
|
std::vector<std::pair<int, int>> prop_values;
|
|
|
|
for (int i = 0; i < N * N; ++i) prop_values.emplace_back(i % N, i / N);
|
|
|
|
std::random_shuffle(prop_values.begin(), prop_values.end());
|
|
|
|
for (const auto &pair : prop_values) {
|
2017-10-30 17:43:25 +08:00
|
|
|
auto v = dba.InsertVertex();
|
2020-01-22 23:20:13 +08:00
|
|
|
ASSERT_TRUE(
|
|
|
|
v.SetProperty(p1, storage::PropertyValue(pair.first)).HasValue());
|
|
|
|
ASSERT_TRUE(
|
|
|
|
v.SetProperty(p2, storage::PropertyValue(pair.second)).HasValue());
|
2017-04-20 19:31:18 +08:00
|
|
|
}
|
2017-10-30 17:43:25 +08:00
|
|
|
dba.AdvanceCommand();
|
2017-04-20 19:31:18 +08:00
|
|
|
|
|
|
|
// order by and collect results
|
|
|
|
auto n = MakeScanAll(storage, symbol_table, "n");
|
2019-02-05 20:16:07 +08:00
|
|
|
auto n_p1 = PROPERTY_LOOKUP(IDENT("n")->MapTo(n.sym_), p1);
|
|
|
|
auto n_p2 = PROPERTY_LOOKUP(IDENT("n")->MapTo(n.sym_), p2);
|
2017-04-20 19:31:18 +08:00
|
|
|
// order the results so we get
|
|
|
|
// (p1: 0, p2: N-1)
|
|
|
|
// (p1: 0, p2: N-2)
|
|
|
|
// ...
|
|
|
|
// (p1: N-1, p2:0)
|
2018-10-04 17:57:23 +08:00
|
|
|
auto order_by = std::make_shared<plan::OrderBy>(n.op_,
|
|
|
|
std::vector<SortItem>{
|
|
|
|
{Ordering::ASC, n_p1},
|
|
|
|
{Ordering::DESC, n_p2},
|
|
|
|
},
|
|
|
|
std::vector<Symbol>{n.sym_});
|
2019-02-05 20:16:07 +08:00
|
|
|
auto n_p1_ne =
|
|
|
|
NEXPR("n.p1", n_p1)->MapTo(symbol_table.CreateSymbol("n.p1", true));
|
|
|
|
auto n_p2_ne =
|
|
|
|
NEXPR("n.p2", n_p2)->MapTo(symbol_table.CreateSymbol("n.p2", true));
|
2017-04-20 19:31:18 +08:00
|
|
|
auto produce = MakeProduce(order_by, n_p1_ne, n_p2_ne);
|
2019-11-22 01:25:25 +08:00
|
|
|
auto context = MakeContext(storage, symbol_table, &dba);
|
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 results = CollectProduce(*produce, &context);
|
2017-04-20 19:31:18 +08:00
|
|
|
ASSERT_EQ(N * N, results.size());
|
|
|
|
for (int j = 0; j < N * N; ++j) {
|
|
|
|
ASSERT_EQ(results[j][0].type(), TypedValue::Type::Int);
|
2019-08-22 20:50:57 +08:00
|
|
|
EXPECT_EQ(results[j][0].ValueInt(), j / N);
|
2017-04-20 19:31:18 +08:00
|
|
|
ASSERT_EQ(results[j][1].type(), TypedValue::Type::Int);
|
2019-08-22 20:50:57 +08:00
|
|
|
EXPECT_EQ(results[j][1].ValueInt(), N - 1 - j % N);
|
2017-04-20 19:31:18 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(QueryPlan, OrderByExceptions) {
|
2019-11-22 01:25:25 +08:00
|
|
|
storage::Storage db;
|
|
|
|
auto storage_dba = db.Access();
|
|
|
|
query::DbAccessor dba(&storage_dba);
|
2018-05-22 22:45:52 +08:00
|
|
|
AstStorage storage;
|
2017-04-20 19:31:18 +08:00
|
|
|
SymbolTable symbol_table;
|
2019-11-22 01:25:25 +08:00
|
|
|
auto prop = dba.NameToProperty("prop");
|
2017-04-20 19:31:18 +08:00
|
|
|
|
|
|
|
// a vector of pairs of typed values that should result
|
|
|
|
// in an exception when trying to order on them
|
2020-01-22 23:20:13 +08:00
|
|
|
std::vector<std::pair<storage::PropertyValue, storage::PropertyValue>>
|
|
|
|
exception_pairs{
|
|
|
|
{storage::PropertyValue(42), storage::PropertyValue(true)},
|
|
|
|
{storage::PropertyValue(42), storage::PropertyValue("bla")},
|
|
|
|
{storage::PropertyValue(42),
|
|
|
|
storage::PropertyValue(std::vector<storage::PropertyValue>{
|
|
|
|
storage::PropertyValue(42)})},
|
|
|
|
{storage::PropertyValue(true), storage::PropertyValue("bla")},
|
|
|
|
{storage::PropertyValue(true),
|
|
|
|
storage::PropertyValue(std::vector<storage::PropertyValue>{
|
|
|
|
storage::PropertyValue(true)})},
|
|
|
|
{storage::PropertyValue("bla"),
|
|
|
|
storage::PropertyValue(std::vector<storage::PropertyValue>{
|
|
|
|
storage::PropertyValue("bla")})},
|
|
|
|
// illegal comparisons of same-type values
|
|
|
|
{storage::PropertyValue(
|
|
|
|
std::vector<storage::PropertyValue>{storage::PropertyValue(42)}),
|
|
|
|
storage::PropertyValue(std::vector<storage::PropertyValue>{
|
|
|
|
storage::PropertyValue(42)})}};
|
2017-04-20 19:31:18 +08:00
|
|
|
|
|
|
|
for (const auto &pair : exception_pairs) {
|
|
|
|
// empty database
|
2019-11-22 01:25:25 +08:00
|
|
|
for (auto vertex : dba.Vertices(storage::View::OLD))
|
2020-01-14 17:02:09 +08:00
|
|
|
ASSERT_TRUE(dba.DetachRemoveVertex(&vertex).HasValue());
|
2017-10-30 17:43:25 +08:00
|
|
|
dba.AdvanceCommand();
|
2019-11-22 01:25:25 +08:00
|
|
|
ASSERT_EQ(0, CountIterable(dba.Vertices(storage::View::OLD)));
|
2017-04-20 19:31:18 +08:00
|
|
|
|
|
|
|
// make two vertices, and set values
|
2019-11-22 01:25:25 +08:00
|
|
|
ASSERT_TRUE(dba.InsertVertex().SetProperty(prop, pair.first).HasValue());
|
|
|
|
ASSERT_TRUE(dba.InsertVertex().SetProperty(prop, pair.second).HasValue());
|
2017-10-30 17:43:25 +08:00
|
|
|
dba.AdvanceCommand();
|
2019-11-22 01:25:25 +08:00
|
|
|
ASSERT_EQ(2, CountIterable(dba.Vertices(storage::View::OLD)));
|
|
|
|
for (const auto &va : dba.Vertices(storage::View::OLD))
|
|
|
|
ASSERT_NE(va.GetProperty(storage::View::OLD, prop).GetValue().type(),
|
2020-01-22 23:20:13 +08:00
|
|
|
storage::PropertyValue::Type::Null);
|
2017-04-20 19:31:18 +08:00
|
|
|
|
|
|
|
// order by and expect an exception
|
|
|
|
auto n = MakeScanAll(storage, symbol_table, "n");
|
2019-02-05 20:16:07 +08:00
|
|
|
auto n_p = PROPERTY_LOOKUP(IDENT("n")->MapTo(n.sym_), prop);
|
2017-04-20 19:31:18 +08:00
|
|
|
auto order_by = std::make_shared<plan::OrderBy>(
|
2018-10-04 17:57:23 +08:00
|
|
|
n.op_, std::vector<SortItem>{{Ordering::ASC, n_p}},
|
2017-04-20 19:31:18 +08:00
|
|
|
std::vector<Symbol>{});
|
2019-11-22 01:25:25 +08:00
|
|
|
auto context = MakeContext(storage, symbol_table, &dba);
|
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
|
|
|
EXPECT_THROW(PullAll(*order_by, &context), QueryRuntimeException);
|
2017-04-20 19:31:18 +08:00
|
|
|
}
|
|
|
|
}
|