Implement coalesce as a special operator

Reviewers: teon.banek, mtomic

Reviewed By: mtomic

Subscribers: pullbot

Differential Revision: https://phabricator.memgraph.io/D1652
This commit is contained in:
Lovro Lugovic 2018-10-11 15:04:49 +02:00
parent cbe100ef5a
commit 4b5c0d3426
3 changed files with 52 additions and 4 deletions

View File

@ -13,6 +13,7 @@ namespace {
const char kStartsWith[] = "STARTSWITH"; const char kStartsWith[] = "STARTSWITH";
const char kEndsWith[] = "ENDSWITH"; const char kEndsWith[] = "ENDSWITH";
const char kContains[] = "CONTAINS"; const char kContains[] = "CONTAINS";
const char kCoalesce[] = "COALESCE";
} // namespace } // namespace
/// Return the function implementation with the given name. /// Return the function implementation with the given name.

View File

@ -371,6 +371,24 @@ class ExpressionEvaluator : public TreeVisitor<TypedValue> {
} }
TypedValue Visit(Function &function) override { TypedValue Visit(Function &function) override {
// Handle COALESCE specially -- evaluate the arguments in order until one of
// them produces a non-null value.
if (function.function_name_ == kCoalesce) {
if (function.arguments_.size() == 0) {
throw QueryRuntimeException(
"'coalesce' requires at least one argument.");
}
for (int64_t i = 0; i < function.arguments_.size(); ++i) {
TypedValue val = function.arguments_[i]->Accept(*this);
if (val.type() != TypedValue::Type::Null) {
return val;
}
}
return TypedValue::Null;
}
// Stack allocate evaluated arguments when there's a small number of them. // Stack allocate evaluated arguments when there's a small number of them.
if (function.arguments_.size() <= 8) { if (function.arguments_.size() <= 8) {
TypedValue arguments[8]; TypedValue arguments[8];

View File

@ -860,20 +860,35 @@ TEST_F(ExpressionEvaluatorPropertyLookup, MapLiteral) {
class FunctionTest : public ExpressionEvaluatorTest { class FunctionTest : public ExpressionEvaluatorTest {
protected: protected:
TypedValue EvaluateFunction(const std::string &function_name, std::vector<Expression *> ExpressionsFromTypedValues(
const std::vector<TypedValue> &args) { const std::vector<TypedValue> &tvs) {
std::vector<Expression *> expressions; std::vector<Expression *> expressions;
for (size_t i = 0; i < args.size(); ++i) { expressions.reserve(tvs.size());
for (size_t i = 0; i < tvs.size(); ++i) {
auto *ident = auto *ident =
storage.Create<Identifier>("arg_" + std::to_string(i), true); storage.Create<Identifier>("arg_" + std::to_string(i), true);
auto sym = symbol_table.CreateSymbol("arg_" + std::to_string(i), true); auto sym = symbol_table.CreateSymbol("arg_" + std::to_string(i), true);
symbol_table[*ident] = sym; symbol_table[*ident] = sym;
frame[sym] = args[i]; frame[sym] = tvs[i];
expressions.push_back(ident); expressions.push_back(ident);
} }
return expressions;
}
TypedValue EvaluateFunctionWithExprs(
const std::string &function_name,
const std::vector<Expression *> &expressions) {
auto *op = storage.Create<Function>(function_name, expressions); auto *op = storage.Create<Function>(function_name, expressions);
return op->Accept(eval); return op->Accept(eval);
} }
TypedValue EvaluateFunction(const std::string &function_name,
const std::vector<TypedValue> &args) {
return EvaluateFunctionWithExprs(function_name,
ExpressionsFromTypedValues(args));
}
}; };
TEST_F(FunctionTest, Coalesce) { TEST_F(FunctionTest, Coalesce) {
@ -882,6 +897,20 @@ TEST_F(FunctionTest, Coalesce) {
.IsNull()); .IsNull());
ASSERT_EQ(EvaluateFunction("COALESCE", {TypedValue::Null, 2, 3}).ValueInt(), ASSERT_EQ(EvaluateFunction("COALESCE", {TypedValue::Null, 2, 3}).ValueInt(),
2); 2);
// (null, 2, assert(false), 3)
auto expressions1 = ExpressionsFromTypedValues({TypedValue::Null, 2, 3});
expressions1.insert(
expressions1.begin() + 2,
storage.Create<Function>("ASSERT", ExpressionsFromTypedValues({false})));
ASSERT_EQ(EvaluateFunctionWithExprs("COALESCE", expressions1).ValueInt(), 2);
// (null, assert(false))
auto expressions2 = ExpressionsFromTypedValues({TypedValue::Null});
expressions2.push_back(
storage.Create<Function>("ASSERT", ExpressionsFromTypedValues({false})));
ASSERT_THROW(EvaluateFunctionWithExprs("COALESCE", expressions2),
QueryRuntimeException);
} }
TEST_F(FunctionTest, EndNode) { TEST_F(FunctionTest, EndNode) {