Type check values used for indexed lookup in range

Reviewers: mferencevic, ipaljak

Reviewed By: mferencevic

Subscribers: pullbot

Differential Revision: https://phabricator.memgraph.io/D2558
This commit is contained in:
Teon Banek 2019-11-21 11:17:13 +01:00
parent f61a8c3358
commit 3c615759b6
2 changed files with 59 additions and 20 deletions

View File

@ -402,7 +402,7 @@ ScanAllByLabelPropertyRange::ScanAllByLabelPropertyRange(
property_name_(property_name), property_name_(property_name),
lower_bound_(lower_bound), lower_bound_(lower_bound),
upper_bound_(upper_bound) { upper_bound_(upper_bound) {
DCHECK(lower_bound_ || upper_bound_) << "Only one bound can be left out"; CHECK(lower_bound_ || upper_bound_) << "Only one bound can be left out";
} }
ACCEPT_WITH_INPUT(ScanAllByLabelPropertyRange) ACCEPT_WITH_INPUT(ScanAllByLabelPropertyRange)
@ -420,10 +420,28 @@ UniqueCursorPtr ScanAllByLabelPropertyRange::MakeCursor(
[&evaluator]( [&evaluator](
const auto &bound) -> std::optional<utils::Bound<PropertyValue>> { const auto &bound) -> std::optional<utils::Bound<PropertyValue>> {
if (!bound) return std::nullopt; if (!bound) return std::nullopt;
auto value = bound->value()->Accept(evaluator); const auto &value = bound->value()->Accept(evaluator);
try { try {
return std::make_optional( const auto &property_value = PropertyValue(value);
utils::Bound<PropertyValue>(PropertyValue(value), bound->type())); switch (property_value.type()) {
case storage::PropertyValue::Type::Bool:
case storage::PropertyValue::Type::List:
case storage::PropertyValue::Type::Map:
// Prevent indexed lookup with something that would fail if we did
// the original filter with `operator<`. Note, for some reason,
// Cypher does not support comparing boolean values.
throw QueryRuntimeException("Invalid type {} for '<'.",
value.type());
case storage::PropertyValue::Type::Null:
case storage::PropertyValue::Type::Int:
case storage::PropertyValue::Type::Double:
case storage::PropertyValue::Type::String:
// These are all fine, there's also Point, Date and Time data types
// which were added to Cypher, but we don't have support for those
// yet.
return std::make_optional(
utils::Bound<PropertyValue>(property_value, bound->type()));
}
} catch (const TypedValueException &) { } catch (const TypedValueException &) {
throw QueryRuntimeException("'{}' cannot be used as a property value.", throw QueryRuntimeException("'{}' cannot be used as a property value.",
value.type()); value.type());

View File

@ -1656,9 +1656,8 @@ TEST(QueryPlan, ScanAllByLabelProperty) {
auto dba = db.Access(); auto dba = db.Access();
ASSERT_EQ(14, CountIterable(dba.Vertices(false))); ASSERT_EQ(14, CountIterable(dba.Vertices(false)));
auto check = [&dba, label, prop](TypedValue lower, Bound::Type lower_type, auto run_scan_all = [&](const TypedValue &lower, Bound::Type lower_type,
TypedValue upper, Bound::Type upper_type, const TypedValue &upper, Bound::Type upper_type) {
const std::vector<TypedValue> &expected) {
AstStorage storage; AstStorage storage;
SymbolTable symbol_table; SymbolTable symbol_table;
auto scan_all = MakeScanAllByLabelPropertyRange( auto scan_all = MakeScanAllByLabelPropertyRange(
@ -1670,7 +1669,13 @@ TEST(QueryPlan, ScanAllByLabelProperty) {
auto produce = MakeProduce(scan_all.op_, output); auto produce = MakeProduce(scan_all.op_, output);
query::DbAccessor execution_dba(&dba); query::DbAccessor execution_dba(&dba);
auto context = MakeContext(storage, symbol_table, &execution_dba); auto context = MakeContext(storage, symbol_table, &execution_dba);
auto results = CollectProduce(*produce, &context); return CollectProduce(*produce, &context);
};
auto check = [&](TypedValue lower, Bound::Type lower_type, TypedValue upper,
Bound::Type upper_type,
const std::vector<TypedValue> &expected) {
auto results = run_scan_all(lower, lower_type, upper, upper_type);
ASSERT_EQ(results.size(), expected.size()); ASSERT_EQ(results.size(), expected.size());
for (size_t i = 0; i < expected.size(); i++) { for (size_t i = 0; i < expected.size(); i++) {
TypedValue equal = TypedValue equal =
@ -1682,10 +1687,6 @@ TEST(QueryPlan, ScanAllByLabelProperty) {
}; };
// normal ranges that return something // normal ranges that return something
check(TypedValue(false), Bound::Type::INCLUSIVE, TypedValue(true),
Bound::Type::EXCLUSIVE, {TypedValue(false)});
check(TypedValue(false), Bound::Type::EXCLUSIVE, TypedValue(true),
Bound::Type::INCLUSIVE, {TypedValue(true)});
check(TypedValue("a"), Bound::Type::EXCLUSIVE, TypedValue("c"), check(TypedValue("a"), Bound::Type::EXCLUSIVE, TypedValue("c"),
Bound::Type::EXCLUSIVE, {TypedValue("b")}); Bound::Type::EXCLUSIVE, {TypedValue("b")});
check(TypedValue(0), Bound::Type::EXCLUSIVE, TypedValue(2), check(TypedValue(0), Bound::Type::EXCLUSIVE, TypedValue(2),
@ -1693,11 +1694,6 @@ TEST(QueryPlan, ScanAllByLabelProperty) {
{TypedValue(0.5), TypedValue(1), TypedValue(1.5), TypedValue(2)}); {TypedValue(0.5), TypedValue(1), TypedValue(1.5), TypedValue(2)});
check(TypedValue(1.5), Bound::Type::EXCLUSIVE, TypedValue(2.5), check(TypedValue(1.5), Bound::Type::EXCLUSIVE, TypedValue(2.5),
Bound::Type::INCLUSIVE, {TypedValue(2), TypedValue(2.5)}); Bound::Type::INCLUSIVE, {TypedValue(2), TypedValue(2.5)});
check(TypedValue(std::vector<TypedValue>{TypedValue(0.5)}),
Bound::Type::EXCLUSIVE,
TypedValue(std::vector<TypedValue>{TypedValue(1.5)}),
Bound::Type::INCLUSIVE,
{TypedValue(std::vector<TypedValue>{TypedValue(1)})});
auto are_comparable = [](PropertyValue::Type a, PropertyValue::Type b) { auto are_comparable = [](PropertyValue::Type a, PropertyValue::Type b) {
auto is_numeric = [](const PropertyValue::Type t) { auto is_numeric = [](const PropertyValue::Type t) {
@ -1707,15 +1703,40 @@ TEST(QueryPlan, ScanAllByLabelProperty) {
return a == b || (is_numeric(a) && is_numeric(b)); return a == b || (is_numeric(a) && is_numeric(b));
}; };
auto is_orderable = [](const PropertyValue &t) {
return t.IsNull() || t.IsInt() || t.IsDouble() || t.IsString();
};
// when a range contains different types, nothing should get returned // when a range contains different types, nothing should get returned
for (const auto &value_a : values) for (const auto &value_a : values) {
for (const auto &value_b : values) { for (const auto &value_b : values) {
if (are_comparable(static_cast<PropertyValue>(value_a).type(), if (are_comparable(static_cast<PropertyValue>(value_a).type(),
static_cast<PropertyValue>(value_b).type())) static_cast<PropertyValue>(value_b).type()))
continue; continue;
check(TypedValue(value_a), Bound::Type::INCLUSIVE, TypedValue(value_b), if (is_orderable(value_a) && is_orderable(value_b)) {
Bound::Type::INCLUSIVE, {}); check(TypedValue(value_a), Bound::Type::INCLUSIVE, TypedValue(value_b),
Bound::Type::INCLUSIVE, {});
} else {
EXPECT_THROW(run_scan_all(TypedValue(value_a), Bound::Type::INCLUSIVE,
TypedValue(value_b), Bound::Type::INCLUSIVE),
QueryRuntimeException);
}
} }
}
// These should all raise an exception due to type mismatch when using
// `operator<`.
EXPECT_THROW(run_scan_all(TypedValue(false), Bound::Type::INCLUSIVE,
TypedValue(true), Bound::Type::EXCLUSIVE),
QueryRuntimeException);
EXPECT_THROW(run_scan_all(TypedValue(false), Bound::Type::EXCLUSIVE,
TypedValue(true), Bound::Type::INCLUSIVE),
QueryRuntimeException);
EXPECT_THROW(
run_scan_all(TypedValue(std::vector<TypedValue>{TypedValue(0.5)}),
Bound::Type::EXCLUSIVE,
TypedValue(std::vector<TypedValue>{TypedValue(1.5)}),
Bound::Type::INCLUSIVE),
QueryRuntimeException);
} }
TEST(QueryPlan, ScanAllByLabelPropertyEqualityNoError) { TEST(QueryPlan, ScanAllByLabelPropertyEqualityNoError) {