diff --git a/src/graph/optimizer/OptimizerUtils.cpp b/src/graph/optimizer/OptimizerUtils.cpp index 53f7e89c240..764f109f8b6 100644 --- a/src/graph/optimizer/OptimizerUtils.cpp +++ b/src/graph/optimizer/OptimizerUtils.cpp @@ -642,7 +642,12 @@ StatusOr selectRelExprIndex(const ColumnDef& field, } auto right = expr->right(); - DCHECK(right->kind() == Expression::Kind::kConstant); + if (expr->kind() == Expression::Kind::kRelIn) { // container expressions + DCHECK(right->isContainerExpr()); + } else { // other expressions + DCHECK(right->kind() == Expression::Kind::kConstant); + } + const auto& value = static_cast(right)->value(); ScoredColumnHint hint; @@ -912,6 +917,32 @@ bool OptimizerUtils::findOptimalIndex(const Expression* condition, return true; } +// Check if the relational expression has a valid index +// The left operand should either be a kEdgeProperty or kTagProperty expr +bool OptimizerUtils::relExprHasIndex( + const Expression* expr, + const std::vector>& indexItems) { + DCHECK(expr->isRelExpr()); + + for (auto& index : indexItems) { + const auto& fields = index->get_fields(); + if (fields.empty()) { + return false; + } + + auto left = static_cast(expr)->left(); + DCHECK(left->kind() == Expression::Kind::kEdgeProperty || + left->kind() == Expression::Kind::kTagProperty); + + auto propExpr = static_cast(left); + if (propExpr->prop() == fields[0].get_name()) { + return true; + } + } + + return false; +} + void OptimizerUtils::copyIndexScanData(const nebula::graph::IndexScan* from, nebula::graph::IndexScan* to) { to->setEmptyResultSet(from->isEmptyResultSet()); diff --git a/src/graph/optimizer/OptimizerUtils.h b/src/graph/optimizer/OptimizerUtils.h index 02d8efe1bee..4ae562b6e0e 100644 --- a/src/graph/optimizer/OptimizerUtils.h +++ b/src/graph/optimizer/OptimizerUtils.h @@ -95,6 +95,10 @@ class OptimizerUtils { bool* isPrefixScan, nebula::storage::cpp2::IndexQueryContext* ictx); + static bool relExprHasIndex( + const Expression* expr, + const std::vector>& indexItems); + static void copyIndexScanData(const nebula::graph::IndexScan* from, nebula::graph::IndexScan* to); }; diff --git a/src/graph/optimizer/rule/OptimizeTagIndexScanByFilterRule.cpp b/src/graph/optimizer/rule/OptimizeTagIndexScanByFilterRule.cpp index 58ac2c7047c..b06a149c0aa 100644 --- a/src/graph/optimizer/rule/OptimizeTagIndexScanByFilterRule.cpp +++ b/src/graph/optimizer/rule/OptimizeTagIndexScanByFilterRule.cpp @@ -46,6 +46,15 @@ const Pattern& OptimizeTagIndexScanByFilterRule::pattern() const { return pattern; } +// Match 2 kinds of expressions: +// +// 1. Relational expr. If it is an IN expr, its list MUST have only 1 element, so it could always be +// transformed to an relEQ expr. i.g. A in [B] => A == B +// It the list has more than 1 element, the expr will be matched with UnionAllIndexScanBaseRule. +// +// 2. Logical AND expr. If the AND expr contains an operand that is an IN expr, the label attribute +// in the IN expr SHOULD NOT have a valid index, otherwise the expression should be matched with +// UnionAllIndexScanBaseRule. bool OptimizeTagIndexScanByFilterRule::match(OptContext* ctx, const MatchedResult& matched) const { if (!OptRule::match(ctx, matched)) { return false; @@ -58,16 +67,23 @@ bool OptimizeTagIndexScanByFilterRule::match(OptContext* ctx, const MatchedResul } } auto condition = filter->condition(); + + // Case1: relational expr if (condition->isRelExpr()) { auto relExpr = static_cast(condition); + // If the container in the IN expr has only 1 element, it will be converted to an relEQ + // expr. If more than 1 element found in the container, UnionAllIndexScanBaseRule will be + // applied. + if (relExpr->kind() == ExprKind::kRelIn && relExpr->right()->isContainerExpr()) { + auto ContainerOperands = graph::ExpressionUtils::getContainerExprOperands(relExpr->right()); + return ContainerOperands.size() == 1; + } return relExpr->left()->kind() == ExprKind::kTagProperty && relExpr->right()->kind() == ExprKind::kConstant; } - if (condition->isLogicalExpr()) { - return condition->kind() == Expression::Kind::kLogicalAnd; - } - return false; + // Case2: logical AND expr + return condition->kind() == ExprKind::kLogicalAnd; } TagIndexScan* makeTagIndexScan(QueryContext* qctx, const TagIndexScan* scan, bool isPrefixScan) { @@ -94,9 +110,38 @@ StatusOr OptimizeTagIndexScanByFilterRule::transform( OptimizerUtils::eraseInvalidIndexItems(scan->schemaId(), &indexItems); + auto condition = filter->condition(); + auto conditionType = condition->kind(); + Expression* transformedExpr = condition->clone(); + + // Stand alone IN expr with only 1 element in the list, no need to check index + if (conditionType == ExprKind::kRelIn) { + transformedExpr = graph::ExpressionUtils::rewriteInExpr(condition); + DCHECK(transformedExpr->kind() == ExprKind::kRelEQ); + } + + // case2: logical AND expr + if (condition->kind() == ExprKind::kLogicalAnd) { + for (auto& operand : static_cast(condition)->operands()) { + if (operand->kind() == ExprKind::kRelIn) { + auto inExpr = static_cast(operand); + // Do not apply this rule if the IN expr has a valid index or it has only 1 element in the + // list + if (static_cast(inExpr->right())->size() > 1) { + return TransformResult::noTransform(); + } else { + transformedExpr = graph::ExpressionUtils::rewriteInExpr(condition); + } + if (OptimizerUtils::relExprHasIndex(inExpr, indexItems)) { + return TransformResult::noTransform(); + } + } + } + } + IndexQueryContext ictx; bool isPrefixScan = false; - if (!OptimizerUtils::findOptimalIndex(filter->condition(), indexItems, &isPrefixScan, &ictx)) { + if (!OptimizerUtils::findOptimalIndex(transformedExpr, indexItems, &isPrefixScan, &ictx)) { return TransformResult::noTransform(); } diff --git a/src/graph/optimizer/rule/UnionAllIndexScanBaseRule.cpp b/src/graph/optimizer/rule/UnionAllIndexScanBaseRule.cpp index 16a5003cf43..2333ba672f4 100644 --- a/src/graph/optimizer/rule/UnionAllIndexScanBaseRule.cpp +++ b/src/graph/optimizer/rule/UnionAllIndexScanBaseRule.cpp @@ -15,6 +15,7 @@ #include "graph/planner/plan/PlanNode.h" #include "graph/planner/plan/Query.h" #include "graph/planner/plan/Scan.h" +#include "graph/util/ExpressionUtils.h" #include "interface/gen-cpp2/storage_types.h" using nebula::graph::Filter; @@ -24,11 +25,23 @@ using nebula::graph::TagIndexFullScan; using nebula::storage::cpp2::IndexQueryContext; using Kind = nebula::graph::PlanNode::Kind; +using ExprKind = nebula::Expression::Kind; using TransformResult = nebula::opt::OptRule::TransformResult; namespace nebula { namespace opt { +// The matched expression should be either a OR expression or an expression that could be +// rewrote to a OR expression. There are 3 senarios. +// +// 1. OR expr. If OR expr has an IN expr operand that has a valid index, expand it to OR expr. +// +// 2. AND expr such as A in [a, b] AND B when A has a valid index, because it can be transformed to +// (A==a AND B) OR (A==b AND B) +// +// 3. IN expr with its list size > 1, such as A in [a, b] since it can be transformed to (A==a) OR +// (A==b). +// If the list has a size of 1, the expr will be matched with OptimizeTagIndexScanByFilterRule. bool UnionAllIndexScanBaseRule::match(OptContext* ctx, const MatchedResult& matched) const { if (!OptRule::match(ctx, matched)) { return false; @@ -36,13 +49,34 @@ bool UnionAllIndexScanBaseRule::match(OptContext* ctx, const MatchedResult& matc auto filter = static_cast(matched.planNode()); auto scan = static_cast(matched.planNode({0, 0})); auto condition = filter->condition(); - if (!condition->isLogicalExpr() || condition->kind() != Expression::Kind::kLogicalOr) { - return false; + auto conditionType = condition->kind(); + + if (condition->isLogicalExpr()) { + // Case1: OR Expr + if (conditionType == ExprKind::kLogicalOr) { + return true; + } + // Case2: AND Expr + if (conditionType == ExprKind::kLogicalAnd && + graph::ExpressionUtils::findAny(static_cast(condition), + {ExprKind::kRelIn})) { + return true; + } + // Check logical operands + for (auto operand : static_cast(condition)->operands()) { + if (!operand->isRelExpr() || !operand->isLogicalExpr()) { + return false; + } + } } - for (auto operand : static_cast(condition)->operands()) { - if (!operand->isRelExpr()) { - return false; + // If the number of elements is less or equal than 1, the IN expr will be transformed into a + // relEQ expr by the OptimizeTagIndexScanByFilterRule. + if (condition->isRelExpr()) { + auto relExpr = static_cast(condition); + if (relExpr->kind() == ExprKind::kRelIn && relExpr->right()->isContainerExpr()) { + auto operandsVec = graph::ExpressionUtils::getContainerExprOperands(relExpr->right()); + return operandsVec.size() > 1; } } @@ -52,7 +86,7 @@ bool UnionAllIndexScanBaseRule::match(OptContext* ctx, const MatchedResult& matc } } - return true; + return false; } StatusOr UnionAllIndexScanBaseRule::transform(OptContext* ctx, @@ -62,20 +96,77 @@ StatusOr UnionAllIndexScanBaseRule::transform(OptContext* ctx, auto scan = static_cast(node); auto metaClient = ctx->qctx()->getMetaClient(); - StatusOr>> status; - if (node->kind() == graph::PlanNode::Kind::kTagIndexFullScan) { - status = metaClient->getTagIndexesFromCache(scan->space()); - } else { - status = metaClient->getEdgeIndexesFromCache(scan->space()); - } + auto status = node->kind() == graph::PlanNode::Kind::kTagIndexFullScan + ? metaClient->getTagIndexesFromCache(scan->space()) + : metaClient->getEdgeIndexesFromCache(scan->space()); + NG_RETURN_IF_ERROR(status); auto indexItems = std::move(status).value(); OptimizerUtils::eraseInvalidIndexItems(scan->schemaId(), &indexItems); + // Check whether the prop has index. + // Rewrite if the property in the IN expr has a valid index + if (indexItems.empty()) { + return TransformResult::noTransform(); + } + + auto condition = filter->condition(); + auto conditionType = condition->kind(); + Expression* transformedExpr = condition->clone(); + + switch (conditionType) { + // Stand alone IN expr + // If it has multiple elements in the list, check valid index before expanding to OR expr + case ExprKind::kRelIn: { + if (!OptimizerUtils::relExprHasIndex(condition, indexItems)) { + return TransformResult::noTransform(); + } + transformedExpr = graph::ExpressionUtils::rewriteInExpr(condition); + break; + } + + // AND expr containing IN expr operand + case ExprKind::kLogicalAnd: { + // Iterate all operands and expand IN exprs if possible + for (auto& expr : static_cast(transformedExpr)->operands()) { + if (expr->kind() == ExprKind::kRelIn) { + if (OptimizerUtils::relExprHasIndex(expr, indexItems)) { + expr = graph::ExpressionUtils::rewriteInExpr(expr); + } + } + } + + // Reconstruct AND expr using distributive law + transformedExpr = graph::ExpressionUtils::rewriteLogicalAndToLogicalOr(transformedExpr); + break; + } + + // OR expr + case ExprKind::kLogicalOr: { + // Iterate all operands and expand IN exprs if possible + for (auto& expr : static_cast(transformedExpr)->operands()) { + if (expr->kind() == ExprKind::kRelIn) { + if (OptimizerUtils::relExprHasIndex(expr, indexItems)) { + expr = graph::ExpressionUtils::rewriteInExpr(expr); + } + } + } + // Flatten OR exprs + graph::ExpressionUtils::pullOrs(transformedExpr); + + break; + } + default: + LOG(FATAL) << "Invalid expression kind: " << static_cast(conditionType); + break; + } + + DCHECK(transformedExpr->kind() == ExprKind::kLogicalOr || + transformedExpr->kind() == ExprKind::kRelEQ); std::vector idxCtxs; - auto condition = static_cast(filter->condition()); - for (auto operand : condition->operands()) { + auto logicalExpr = static_cast(transformedExpr); + for (auto operand : logicalExpr->operands()) { IndexQueryContext ictx; bool isPrefixScan = false; if (!OptimizerUtils::findOptimalIndex(operand, indexItems, &isPrefixScan, &ictx)) { diff --git a/src/graph/util/ExpressionUtils.cpp b/src/graph/util/ExpressionUtils.cpp index a4150564b06..932a638d561 100644 --- a/src/graph/util/ExpressionUtils.cpp +++ b/src/graph/util/ExpressionUtils.cpp @@ -170,6 +170,130 @@ Expression *ExpressionUtils::rewriteAgg2VarProp(const Expression *expr) { return RewriteVisitor::transform(expr, std::move(matcher), std::move(rewriter)); } +// Rewrite the IN expr to a relEQ expr if the right operand has only 1 element. +// Rewrite the IN expr to an OR expr if the right operand has more than 1 element. +Expression *ExpressionUtils::rewriteInExpr(const Expression *expr) { + DCHECK(expr->kind() == Expression::Kind::kRelIn); + auto pool = expr->getObjPool(); + auto inExpr = static_cast(expr->clone()); + auto containerOperands = getContainerExprOperands(inExpr->right()); + + auto operandSize = containerOperands.size(); + // container has only 1 element, no need to transform to logical expression + if (operandSize == 1) { + return RelationalExpression::makeEQ(pool, inExpr->left(), containerOperands[0]); + } + + std::vector orExprOperands; + orExprOperands.reserve(operandSize); + // A in [B, C, D] => (A == B) or (A == C) or (A == D) + for (auto *operand : containerOperands) { + orExprOperands.emplace_back(RelationalExpression::makeEQ(pool, inExpr->left(), operand)); + } + auto orExpr = LogicalExpression::makeOr(pool); + orExpr->setOperands(orExprOperands); + + return orExpr; +} + +Expression *ExpressionUtils::rewriteLogicalAndToLogicalOr(const Expression *expr) { + DCHECK(expr->kind() == Expression::Kind::kLogicalAnd); + auto pool = expr->getObjPool(); + auto logicalAndExpr = static_cast(expr->clone()); + auto logicalAndExprSize = (logicalAndExpr->operands()).size(); + + // Extract all OR expr + auto orExprList = collectAll(logicalAndExpr, {Expression::Kind::kLogicalOr}); + auto orExprListSize = orExprList.size(); + + // Extract all non-OR expr + std::vector nonOrExprList; + bool isAllRelOr = logicalAndExprSize == orExprListSize; + + // If logical expression has operand that is not an OR expr, add into nonOrExprList + if (!isAllRelOr) { + nonOrExprList.reserve(logicalAndExprSize - orExprListSize); + for (const auto &operand : logicalAndExpr->operands()) { + if (operand->kind() != Expression::Kind::kLogicalOr) { + nonOrExprList.emplace_back(std::move(operand)); + } + } + } + + DCHECK_GT(orExprListSize, 0); + std::vector> orExprOperands{{}}; + orExprOperands.reserve(orExprListSize); + + // Merge the elements of vec2 into each subVec of vec1 + // [[A], [B]] and [C, D] => [[A, C], [A, D], [B, C], [B,D]] + auto mergeVecs = [](std::vector> &vec1, + const std::vector vec2) { + std::vector> res; + for (auto &ele1 : vec1) { + for (const auto &ele2 : vec2) { + auto tempSubVec = ele1; + tempSubVec.emplace_back(std::move(ele2)); + res.emplace_back(std::move(tempSubVec)); + } + } + return res; + }; + + // Iterate all OR exprs and construct the operand list + for (auto curExpr : orExprList) { + auto curLogicalOrExpr = static_cast(const_cast(curExpr)); + auto curOrOperands = curLogicalOrExpr->operands(); + + orExprOperands = mergeVecs(orExprOperands, curOrOperands); + } + + // orExprOperands is a 2D vector where each sub-vector is the operands of AND expression. + // [[A, C], [A, D], [B, C], [B,D]] => (A and C) or (A and D) or (B and C) or (B and D) + std::vector andExprList; + andExprList.reserve(orExprOperands.size()); + for (auto &operand : orExprOperands) { + auto andExpr = LogicalExpression::makeAnd(pool); + // if nonOrExprList is not empty, append it to operand + if (!isAllRelOr) { + operand.insert(operand.end(), nonOrExprList.begin(), nonOrExprList.end()); + } + andExpr->setOperands(operand); + andExprList.emplace_back(std::move(andExpr)); + } + + auto orExpr = LogicalExpression::makeOr(pool); + orExpr->setOperands(andExprList); + return orExpr; +} + +std::vector ExpressionUtils::getContainerExprOperands(const Expression *expr) { + DCHECK(expr->isContainerExpr()); + auto pool = expr->getObjPool(); + auto containerExpr = expr->clone(); + + std::vector containerOperands; + switch (containerExpr->kind()) { + case Expression::Kind::kList: + containerOperands = static_cast(containerExpr)->get(); + break; + case Expression::Kind::kSet: { + containerOperands = static_cast(containerExpr)->get(); + break; + } + case Expression::Kind::kMap: { + auto mapItems = static_cast(containerExpr)->get(); + // iterate map and add key into containerOperands + for (auto &item : mapItems) { + containerOperands.emplace_back(ConstantExpression::make(pool, std::move(item.first))); + } + break; + } + default: + LOG(FATAL) << "Invalid expression type " << containerExpr->kind(); + } + return containerOperands; +} + StatusOr ExpressionUtils::foldConstantExpr(const Expression *expr) { ObjectPool *objPool = expr->getObjPool(); auto newExpr = expr->clone(); diff --git a/src/graph/util/ExpressionUtils.h b/src/graph/util/ExpressionUtils.h index ef62ecfccba..a4faeaf5777 100644 --- a/src/graph/util/ExpressionUtils.h +++ b/src/graph/util/ExpressionUtils.h @@ -66,6 +66,20 @@ class ExpressionUtils { static Expression* rewriteRelExpr(const Expression* expr); static Expression* rewriteRelExprHelper(const Expression* expr, Expression*& relRightOperandExpr); + // Rewrite IN expression into OR expression or relEQ expression + static Expression* rewriteInExpr(const Expression* expr); + + // Rewrite Logical AND expr to Logical OR expr using distributive law + // Examples: + // A and (B or C) => (A and B) or (A and C) + // (A or B) and (C or D) => (A and C) or (A and D) or (B and C) or (B or D) + static Expression* rewriteLogicalAndToLogicalOr(const Expression* expr); + + // Return the operands of container expressions + // For list and set, return the operands + // For map, return the keys + static std::vector getContainerExprOperands(const Expression* expr); + // Clone and fold constant expression static StatusOr foldConstantExpr(const Expression* expr); @@ -73,6 +87,10 @@ class ExpressionUtils { static Expression* reduceUnaryNotExpr(const Expression* expr); // Transform filter using multiple expression rewrite strategies + // 1. rewrite relational expressions containing arithmetic operands so that + // all constants are on the right side of relExpr. + // 2. fold constant + // 3. reduce unary expression e.g. !(A and B) => !A or !B static StatusOr filterTransform(const Expression* expr); // Negate the given logical expr: (A && B) -> (!A || !B) diff --git a/src/graph/util/test/ExpressionUtilsTest.cpp b/src/graph/util/test/ExpressionUtilsTest.cpp index 95c3033ad26..fa6963000b8 100644 --- a/src/graph/util/test/ExpressionUtilsTest.cpp +++ b/src/graph/util/test/ExpressionUtilsTest.cpp @@ -494,6 +494,81 @@ TEST_F(ExpressionUtilsTest, flattenInnerLogicalExpr) { } } +TEST_F(ExpressionUtilsTest, rewriteInExpr) { + auto elist1 = ExpressionList::make(pool); + (*elist1).add(ConstantExpression::make(pool, 10)).add(ConstantExpression::make(pool, 20)); + auto listExpr1 = ListExpression::make(pool, elist1); + + auto elist2 = ExpressionList::make(pool); + (*elist2).add(ConstantExpression::make(pool, "a")).add(ConstantExpression::make(pool, "b")); + auto listExpr2 = ListExpression::make(pool, elist2); + + auto elist3 = ExpressionList::make(pool); + (*elist3).add(ConstantExpression::make(pool, 100)); + auto listExpr3 = ListExpression::make(pool, elist3); + + // a IN [b,c] -> a==b OR a==c + { + auto inExpr1 = + RelationalExpression::makeIn(pool, ConstantExpression::make(pool, 10), listExpr1); + auto orExpr1 = ExpressionUtils::rewriteInExpr(inExpr1); + auto expected1 = LogicalExpression::makeOr( + pool, + RelationalExpression::makeEQ( + pool, ConstantExpression::make(pool, 10), ConstantExpression::make(pool, 10)), + RelationalExpression::makeEQ( + pool, ConstantExpression::make(pool, 10), ConstantExpression::make(pool, 20))); + ASSERT_EQ(*expected1, *orExpr1); + + auto inExpr2 = + RelationalExpression::makeIn(pool, ConstantExpression::make(pool, "abc"), listExpr2); + auto orExpr2 = ExpressionUtils::rewriteInExpr(inExpr2); + auto expected2 = LogicalExpression::makeOr( + pool, + RelationalExpression::makeEQ( + pool, ConstantExpression::make(pool, "abc"), ConstantExpression::make(pool, "a")), + RelationalExpression::makeEQ( + pool, ConstantExpression::make(pool, "abc"), ConstantExpression::make(pool, "b"))); + ASSERT_EQ(*expected2, *orExpr2); + } + + // a IN [b] -> a == b + { + auto inExpr = RelationalExpression::makeIn(pool, ConstantExpression::make(pool, 10), listExpr3); + auto expected = RelationalExpression::makeEQ( + pool, ConstantExpression::make(pool, 10), ConstantExpression::make(pool, 100)); + ASSERT_EQ(*expected, *ExpressionUtils::rewriteInExpr(inExpr)); + } +} + +TEST_F(ExpressionUtilsTest, rewriteLogicalAndToLogicalOr) { + auto orExpr1 = LogicalExpression::makeOr( + pool, ConstantExpression::make(pool, 10), ConstantExpression::make(pool, 20)); + auto orExpr2 = LogicalExpression::makeOr( + pool, ConstantExpression::make(pool, "a"), ConstantExpression::make(pool, "b")); + + // (a OR b) AND (c OR d) -> (a AND c) OR (a AND d) OR (b AND c) OR (b AND d) + { + auto andExpr = LogicalExpression::makeAnd(pool, orExpr1, orExpr2); + auto transformedExpr = ExpressionUtils::rewriteLogicalAndToLogicalOr(andExpr); + + std::vector orOperands = { + LogicalExpression::makeAnd( + pool, ConstantExpression::make(pool, 10), ConstantExpression::make(pool, "a")), + LogicalExpression::makeAnd( + pool, ConstantExpression::make(pool, 10), ConstantExpression::make(pool, "b")), + LogicalExpression::makeAnd( + pool, ConstantExpression::make(pool, 20), ConstantExpression::make(pool, "a")), + LogicalExpression::makeAnd( + pool, ConstantExpression::make(pool, 20), ConstantExpression::make(pool, "b"))}; + + auto expected = LogicalExpression::makeOr(pool); + expected->setOperands(orOperands); + + ASSERT_EQ(*expected, *transformedExpr); + } +} + TEST_F(ExpressionUtilsTest, splitFilter) { using Kind = Expression::Kind; { diff --git a/src/graph/validator/LookupValidator.cpp b/src/graph/validator/LookupValidator.cpp index c93a49e878c..b96a0f6cafe 100644 --- a/src/graph/validator/LookupValidator.cpp +++ b/src/graph/validator/LookupValidator.cpp @@ -21,6 +21,8 @@ using nebula::meta::NebulaSchemaProvider; using std::shared_ptr; using std::unique_ptr; +using ExprKind = nebula::Expression::Kind; + namespace nebula { namespace graph { @@ -186,6 +188,7 @@ Status LookupValidator::validateYield() { return Status::OK(); } lookupCtx_->dedup = yieldClause->isDistinct(); + if (lookupCtx_->isEdge) { NG_RETURN_IF_ERROR(validateYieldEdge()); } else { @@ -231,12 +234,9 @@ StatusOr LookupValidator::handleLogicalExprOperands(LogicalExpressi auto& operands = lExpr->operands(); for (auto i = 0u; i < operands.size(); i++) { auto operand = lExpr->operand(i); - if (operand->isLogicalExpr()) { - // Not allow different logical expression to use: A AND B OR C - return Status::SemanticError("Not supported filter: %s", lExpr->toString().c_str()); - } auto ret = checkFilter(operand); NG_RETURN_IF_ERROR(ret); + auto newOperand = ret.value(); if (operand != newOperand) { lExpr->setOperand(i, newOperand); @@ -248,14 +248,25 @@ StatusOr LookupValidator::handleLogicalExprOperands(LogicalExpressi StatusOr LookupValidator::checkFilter(Expression* expr) { // TODO: Support IN expression push down if (expr->isRelExpr()) { - return checkRelExpr(static_cast(expr)); + // Only starts with can be pushed down as a range scan, so forbid other string-related relExpr + if (expr->kind() == ExprKind::kRelREG || expr->kind() == ExprKind::kContains || + expr->kind() == ExprKind::kNotContains || expr->kind() == ExprKind::kEndsWith || + expr->kind() == ExprKind::kNotStartsWith || expr->kind() == ExprKind::kNotEndsWith) { + return Status::SemanticError( + "Expression %s is not supported, please use full-text index as an optimal solution", + expr->toString().c_str()); + } + + auto relExpr = static_cast(expr); + NG_RETURN_IF_ERROR(checkRelExpr(relExpr)); + return rewriteRelExpr(relExpr); } switch (expr->kind()) { - case Expression::Kind::kLogicalOr: { + case ExprKind::kLogicalOr: { ExpressionUtils::pullOrs(expr); return handleLogicalExprOperands(static_cast(expr)); } - case Expression::Kind::kLogicalAnd: { + case ExprKind::kLogicalAnd: { ExpressionUtils::pullAnds(expr); return handleLogicalExprOperands(static_cast(expr)); } @@ -265,17 +276,15 @@ StatusOr LookupValidator::checkFilter(Expression* expr) { } } -StatusOr LookupValidator::checkRelExpr(RelationalExpression* expr) { +Status LookupValidator::checkRelExpr(RelationalExpression* expr) { auto* left = expr->left(); auto* right = expr->right(); // Does not support filter : schema.col1 > schema.col2 - if (left->kind() == Expression::Kind::kLabelAttribute && - right->kind() == Expression::Kind::kLabelAttribute) { + if (left->kind() == ExprKind::kLabelAttribute && right->kind() == ExprKind::kLabelAttribute) { return Status::SemanticError("Expression %s not supported yet", expr->toString().c_str()); } - if (left->kind() == Expression::Kind::kLabelAttribute || - right->kind() == Expression::Kind::kLabelAttribute) { - return rewriteRelExpr(expr); + if (left->kind() == ExprKind::kLabelAttribute || right->kind() == ExprKind::kLabelAttribute) { + return Status::OK(); } return Status::SemanticError("Expression %s not supported yet", expr->toString().c_str()); } @@ -283,42 +292,41 @@ StatusOr LookupValidator::checkRelExpr(RelationalExpression* expr) StatusOr LookupValidator::rewriteRelExpr(RelationalExpression* expr) { // swap LHS and RHS of relExpr if LabelAttributeExpr in on the right, // so that LabelAttributeExpr is always on the left - auto right = expr->right(); - if (right->kind() == Expression::Kind::kLabelAttribute) { + auto rightOperand = expr->right(); + if (rightOperand->kind() == ExprKind::kLabelAttribute) { expr = static_cast(reverseRelKind(expr)); } - auto left = expr->left(); - auto* la = static_cast(left); + auto leftOperand = expr->left(); + auto* la = static_cast(leftOperand); if (la->left()->name() != sentence()->from()) { return Status::SemanticError("Schema name error: %s", la->left()->name().c_str()); } // fold constant expression - auto pool = qctx_->objPool(); auto foldRes = ExpressionUtils::foldConstantExpr(expr); NG_RETURN_IF_ERROR(foldRes); expr = static_cast(foldRes.value()); - DCHECK_EQ(expr->left()->kind(), Expression::Kind::kLabelAttribute); + DCHECK_EQ(expr->left()->kind(), ExprKind::kLabelAttribute); + // Check schema and value type std::string prop = la->right()->value().getStr(); auto relExprType = expr->kind(); auto c = checkConstExpr(expr->right(), prop, relExprType); NG_RETURN_IF_ERROR(c); - expr->setRight(ConstantExpression::make(pool, std::move(c).value())); + expr->setRight(std::move(c).value()); // rewrite PropertyExpression - if (lookupCtx_->isEdge) { - expr->setLeft(ExpressionUtils::rewriteLabelAttr2EdgeProp(la)); - } else { - expr->setLeft(ExpressionUtils::rewriteLabelAttr2TagProp(la)); - } + auto propExpr = lookupCtx_->isEdge ? ExpressionUtils::rewriteLabelAttr2EdgeProp(la) + : ExpressionUtils::rewriteLabelAttr2TagProp(la); + expr->setLeft(propExpr); return expr; } -StatusOr LookupValidator::checkConstExpr(Expression* expr, - const std::string& prop, - const Expression::Kind kind) { +StatusOr LookupValidator::checkConstExpr(Expression* expr, + const std::string& prop, + const ExprKind kind) { + auto* pool = expr->getObjPool(); if (!evaluableExpr(expr)) { return Status::SemanticError("'%s' is not an evaluable expression.", expr->toString().c_str()); } @@ -336,29 +344,30 @@ StatusOr LookupValidator::checkConstExpr(Expression* expr, // Allow different numeric type to compare if (graph::SchemaUtil::propTypeToValueType(type) == Value::Type::FLOAT && v.isInt()) { - return v.toFloat(); + return ConstantExpression::make(pool, v.toFloat()); } else if (graph::SchemaUtil::propTypeToValueType(type) == Value::Type::INT && v.isFloat()) { // col1 < 10.5 range: [min, 11), col1 < 10 range: [min, 10) double f = v.getFloat(); int iCeil = ceil(f); int iFloor = floor(f); - if (kind == Expression::Kind::kRelGE || kind == Expression::Kind::kRelLT) { + if (kind == ExprKind::kRelGE || kind == ExprKind::kRelLT) { // edge case col1 >= 40.0, no need to round up if (std::abs(f - iCeil) < kEpsilon) { - return iFloor; + return ConstantExpression::make(pool, iFloor); } - return iCeil; + return ConstantExpression::make(pool, iCeil); } - return iFloor; + return ConstantExpression::make(pool, iFloor); } + // Check prop type if (v.type() != SchemaUtil::propTypeToValueType(type)) { // allow diffrent types in the IN expression, such as "abc" IN ["abc"] - if (v.type() != Value::Type::LIST) { + if (!expr->isContainerExpr()) { return Status::SemanticError("Column type error : %s", prop.c_str()); } } - return v; + return expr; } StatusOr LookupValidator::checkTSExpr(Expression* expr) { @@ -381,27 +390,28 @@ StatusOr LookupValidator::checkTSExpr(Expression* expr) { } return tsName; } + // Transform (A > B) to (B < A) Expression* LookupValidator::reverseRelKind(RelationalExpression* expr) { auto kind = expr->kind(); auto reversedKind = kind; switch (kind) { - case Expression::Kind::kRelEQ: + case ExprKind::kRelEQ: break; - case Expression::Kind::kRelNE: + case ExprKind::kRelNE: break; - case Expression::Kind::kRelLT: - reversedKind = Expression::Kind::kRelGT; + case ExprKind::kRelLT: + reversedKind = ExprKind::kRelGT; break; - case Expression::Kind::kRelLE: - reversedKind = Expression::Kind::kRelGE; + case ExprKind::kRelLE: + reversedKind = ExprKind::kRelGE; break; - case Expression::Kind::kRelGT: - reversedKind = Expression::Kind::kRelLT; + case ExprKind::kRelGT: + reversedKind = ExprKind::kRelLT; break; - case Expression::Kind::kRelGE: - reversedKind = Expression::Kind::kRelLE; + case ExprKind::kRelGE: + reversedKind = ExprKind::kRelLE; break; default: LOG(FATAL) << "Invalid relational expression kind: " << static_cast(kind); diff --git a/src/graph/validator/LookupValidator.h b/src/graph/validator/LookupValidator.h index 3b9c6ab715f..97c900d9d36 100644 --- a/src/graph/validator/LookupValidator.h +++ b/src/graph/validator/LookupValidator.h @@ -37,12 +37,11 @@ class LookupValidator final : public Validator { Status validateYieldEdge(); StatusOr checkFilter(Expression* expr); - StatusOr checkRelExpr(RelationalExpression* expr); + Status checkRelExpr(RelationalExpression* expr); StatusOr checkTSExpr(Expression* expr); - StatusOr checkConstExpr(Expression* expr, - const std::string& prop, - const Expression::Kind kind); - + StatusOr checkConstExpr(Expression* expr, + const std::string& prop, + const Expression::Kind kind); StatusOr rewriteRelExpr(RelationalExpression* expr); Expression* reverseRelKind(RelationalExpression* expr); diff --git a/src/graph/visitor/FindVisitor.cpp b/src/graph/visitor/FindVisitor.cpp index ec0d5505975..5f8142a5003 100644 --- a/src/graph/visitor/FindVisitor.cpp +++ b/src/graph/visitor/FindVisitor.cpp @@ -121,6 +121,15 @@ void FindVisitor::visit(ListComprehensionExpression* expr) { } } +void FindVisitor::visit(LogicalExpression* expr) { + findInCurrentExpr(expr); + if (!needFindAll_ && !foundExprs_.empty()) return; + for (const auto& operand : expr->operands()) { + operand->accept(this); + if (!needFindAll_ && !foundExprs_.empty()) return; + } +} + void FindVisitor::visit(ConstantExpression* expr) { findInCurrentExpr(expr); } void FindVisitor::visit(EdgePropertyExpression* expr) { findInCurrentExpr(expr); } diff --git a/src/graph/visitor/FindVisitor.h b/src/graph/visitor/FindVisitor.h index 64936264416..3553c551734 100644 --- a/src/graph/visitor/FindVisitor.h +++ b/src/graph/visitor/FindVisitor.h @@ -66,6 +66,7 @@ class FindVisitor final : public ExprVisitorImpl { void visit(ColumnExpression* expr) override; void visit(ListComprehensionExpression* expr) override; void visit(SubscriptRangeExpression* expr) override; + void visit(LogicalExpression* expr) override; void visitBinaryExpr(BinaryExpression* expr) override; void findInCurrentExpr(Expression* expr); diff --git a/src/graph/visitor/FoldConstantExprVisitor.cpp b/src/graph/visitor/FoldConstantExprVisitor.cpp index 93b0430b964..a075d7eef6b 100644 --- a/src/graph/visitor/FoldConstantExprVisitor.cpp +++ b/src/graph/visitor/FoldConstantExprVisitor.cpp @@ -338,6 +338,11 @@ void FoldConstantExprVisitor::visitBinaryExpr(BinaryExpression *expr) { } Expression *FoldConstantExprVisitor::fold(Expression *expr) { + // Container expresison should remain the same type after being folded + if (expr->isContainerExpr()) { + return expr; + } + QueryExpressionContext ctx; auto value = expr->eval(ctx(nullptr)); if (value.type() == Value::Type::NULLVALUE) { diff --git a/src/graph/visitor/VidExtractVisitor.cpp b/src/graph/visitor/VidExtractVisitor.cpp index 66d8684a79e..d37a81c18b5 100644 --- a/src/graph/visitor/VidExtractVisitor.cpp +++ b/src/graph/visitor/VidExtractVisitor.cpp @@ -135,7 +135,7 @@ void VidExtractVisitor::visit(ArithmeticExpression *expr) { void VidExtractVisitor::visit(RelationalExpression *expr) { if (expr->kind() == Expression::Kind::kRelIn) { - // const auto *inExpr = static_cast(expr); + // id(V) IN [List] if (expr->left()->kind() == Expression::Kind::kLabelAttribute) { const auto *labelExpr = static_cast(expr->left()); const auto &label = labelExpr->left()->toString(); @@ -143,29 +143,30 @@ void VidExtractVisitor::visit(RelationalExpression *expr) { {{label, {VidPattern::Vids::Kind::kOtherSource, {}}}}}; return; } + if (expr->left()->kind() != Expression::Kind::kFunctionCall || - expr->right()->kind() != Expression::Kind::kConstant) { + expr->right()->kind() != Expression::Kind::kList || + !ExpressionUtils::isEvaluableExpr(expr->right())) { vidPattern_ = VidPattern{}; return; } + const auto *fCallExpr = static_cast(expr->left()); if (fCallExpr->name() != "id" && fCallExpr->args()->numArgs() != 1 && fCallExpr->args()->args().front()->kind() != Expression::Kind::kLabel) { vidPattern_ = VidPattern{}; return; } - const auto *constExpr = static_cast(expr->right()); - if (constExpr->value().type() != Value::Type::LIST) { - vidPattern_ = VidPattern{}; - return; - } - vidPattern_ = VidPattern{VidPattern::Special::kInUsed, - {{fCallExpr->args()->args().front()->toString(), - {VidPattern::Vids::Kind::kIn, constExpr->value().getList()}}}}; + + auto *listExpr = static_cast(expr->right()); + QueryExpressionContext ctx; + vidPattern_ = + VidPattern{VidPattern::Special::kInUsed, + {{fCallExpr->args()->args().front()->toString(), + {VidPattern::Vids::Kind::kIn, listExpr->eval(ctx(nullptr)).getList()}}}}; return; } else if (expr->kind() == Expression::Kind::kRelEQ) { - // const auto *eqExpr = static_cast(expr); + // id(V) == vid if (expr->left()->kind() == Expression::Kind::kLabelAttribute) { const auto *labelExpr = static_cast(expr->left()); const auto &label = labelExpr->left()->toString(); diff --git a/src/graph/visitor/test/FoldConstantExprVisitorTest.cpp b/src/graph/visitor/test/FoldConstantExprVisitorTest.cpp index 7535592e920..90724f2cf95 100644 --- a/src/graph/visitor/test/FoldConstantExprVisitorTest.cpp +++ b/src/graph/visitor/test/FoldConstantExprVisitorTest.cpp @@ -104,6 +104,8 @@ TEST_F(FoldConstantExprVisitorTest, TestListExpr) { expr->accept(&visitor); ASSERT_EQ(*expr, *expected) << expr->toString() << " vs. " << expected->toString(); ASSERT(visitor.canBeFolded()); + // type should remain the same after folding + ASSERT_EQ(expr->kind(), Expression::Kind::kList); } TEST_F(FoldConstantExprVisitorTest, TestSetExpr) { @@ -116,6 +118,8 @@ TEST_F(FoldConstantExprVisitorTest, TestSetExpr) { expr->accept(&visitor); ASSERT_EQ(*expr, *expected) << expr->toString() << " vs. " << expected->toString(); ASSERT(visitor.canBeFolded()); + // type should remain the same after folding + ASSERT_EQ(expr->kind(), Expression::Kind::kSet); } TEST_F(FoldConstantExprVisitorTest, TestMapExpr) { @@ -131,6 +135,8 @@ TEST_F(FoldConstantExprVisitorTest, TestMapExpr) { expr->accept(&visitor); ASSERT_EQ(*expr, *expected) << expr->toString() << " vs. " << expected->toString(); ASSERT(visitor.canBeFolded()); + // type should remain the same after folding + ASSERT_EQ(expr->kind(), Expression::Kind::kMap); } TEST_F(FoldConstantExprVisitorTest, TestCaseExpr) { diff --git a/tests/tck/features/lookup/EdgeIndexFullScan.feature b/tests/tck/features/lookup/EdgeIndexFullScan.feature index 293eaa339e3..d07348979c3 100644 --- a/tests/tck/features/lookup/EdgeIndexFullScan.feature +++ b/tests/tck/features/lookup/EdgeIndexFullScan.feature @@ -33,51 +33,12 @@ Feature: Lookup edge index full scan '103'->'101':('Blue', 33); """ - Scenario: Edge with relational RegExp filter[1] + Scenario: Edge with relational RegExp filter When executing query: """ LOOKUP ON edge_1 WHERE edge_1.col1_str =~ "\\w+\\d+" YIELD edge_1.col1_str """ - Then the result should be, in any order: - | SrcVID | DstVID | Ranking | edge_1.col1_str | - | "101" | "102" | 0 | "Red1" | - When executing query: - """ - LOOKUP ON edge_1 WHERE edge_1.col1_str =~ "\\w+ll\\w+" YIELD edge_1.col1_str - """ - Then the result should be, in any order: - | SrcVID | DstVID | Ranking | edge_1.col1_str | - | "102" | "103" | 0 | "Yellow" | - - # skip because `make fmt` will delete '\' in the operator info and causes tests fail - @skip - Scenario: Edge with relational RegExp filter[2] - When profiling query: - """ - LOOKUP ON edge_1 where edge_1.col1_str =~ "\\d+\\w+" YIELD edge_1.col1_str - """ - Then the result should be, in any order: - | SrcVID | DstVID | Ranking | edge_1.col1_str | - | "101" | "102" | 0 | "Red1" | - And the execution plan should be: - | id | name | dependencies | operator info | - | 3 | Project | 2 | | - | 2 | Filter | 4 | {"condition": "(edge_1.col1_str=~\"\w+\d+\")"} | - | 4 | EdgeIndexFullScan | 0 | | - | 0 | Start | | | - When profiling query: - """ - LOOKUP ON edge_1 where edge_1.col1_str =~ "\\w+ea\\w+" YIELD edge_1.col1_str - """ - Then the result should be, in any order: - | SrcVID | DstVID | Ranking | edge_1.col1_str | - | "102" | "103" | 0 | "Yellow" | - And the execution plan should be: - | id | name | dependencies | operator info | - | 3 | Project | 2 | | - | 2 | Filter | 4 | {"condition": "(edge_1.col1_str=~\"\w+ea\w+\")"} | - | 4 | EdgeIndexFullScan | 0 | | - | 0 | Start | | | + Then a SemanticError should be raised at runtime: Expression (edge_1.col1_str=~"\w+\d+") is not supported, please use full-text index as an optimal solution Scenario: Edge with relational NE filter When profiling query: @@ -109,7 +70,7 @@ Feature: Lookup edge index full scan | 4 | EdgeIndexFullScan | 0 | | | 0 | Start | | | - Scenario: Edge with relational IN/NOT IN filter + Scenario: Edge with simple relational IN filter When profiling query: """ LOOKUP ON edge_1 WHERE edge_1.col1_str IN ["Red", "Yellow"] YIELD edge_1.col1_str @@ -118,11 +79,10 @@ Feature: Lookup edge index full scan | SrcVID | DstVID | Ranking | edge_1.col1_str | | "102" | "103" | 0 | "Yellow" | And the execution plan should be: - | id | name | dependencies | operator info | - | 3 | Project | 2 | | - | 2 | Filter | 4 | {"condition": "(edge_1.col1_str IN [\"Red\",\"Yellow\"])"} | - | 4 | EdgeIndexFullScan | 0 | | - | 0 | Start | | | + | id | name | dependencies | operator info | + | 3 | Project | 4 | | + | 4 | IndexScan | 0 | | + | 0 | Start | | | When executing query: """ LOOKUP ON edge_1 WHERE edge_1.col1_str IN ["non-existed-name"] YIELD edge_1.col1_str @@ -138,11 +98,165 @@ Feature: Lookup edge index full scan | "103" | "101" | 0 | 33 | | "102" | "103" | 0 | 22 | And the execution plan should be: - | id | name | dependencies | operator info | - | 3 | Project | 2 | | - | 2 | Filter | 4 | {"condition": "(edge_1.col2_int IN [22,33])"} | - | 4 | EdgeIndexFullScan | 0 | | - | 0 | Start | | | + | id | name | dependencies | operator info | + | 3 | Project | 4 | | + | 4 | IndexScan | 0 | | + | 0 | Start | | | + # a IN b OR c + When profiling query: + """ + LOOKUP ON edge_1 + WHERE edge_1.col2_int IN [23 - 1 , 66/2] OR edge_1.col2_int==11 + YIELD edge_1.col2_int + """ + Then the result should be, in any order: + | SrcVID | DstVID | Ranking | edge_1.col2_int | + | "101" | "102" | 0 | 11 | + | "102" | "103" | 0 | 22 | + | "103" | "101" | 0 | 33 | + And the execution plan should be: + | id | name | dependencies | operator info | + | 3 | Project | 4 | | + | 4 | IndexScan | 0 | | + | 0 | Start | | | + # a IN b OR c IN d + When profiling query: + """ + LOOKUP ON edge_1 + WHERE edge_1.col2_int IN [23 - 1 , 66/2] OR edge_1.col1_str IN [toUpper("r")+"ed1"] + YIELD edge_1.col1_str, edge_1.col2_int + """ + Then the result should be, in any order: + | SrcVID | DstVID | Ranking | edge_1.col1_str | edge_1.col2_int | + | "101" | "102" | 0 | "Red1" | 11 | + | "102" | "103" | 0 | "Yellow" | 22 | + | "103" | "101" | 0 | "Blue" | 33 | + And the execution plan should be: + | id | name | dependencies | operator info | + | 3 | Project | 4 | | + | 4 | IndexScan | 0 | | + | 0 | Start | | | + # a IN b AND c (EdgeIndexPrefixScan) + When profiling query: + """ + LOOKUP ON edge_1 + WHERE edge_1.col2_int IN [11 , 66/2] AND edge_1.col2_int==11 + YIELD edge_1.col2_int + """ + Then the result should be, in any order: + | SrcVID | DstVID | Ranking | edge_1.col2_int | + | "101" | "102" | 0 | 11 | + And the execution plan should be: + | id | name | dependencies | operator info | + | 3 | Project | 4 | | + | 4 | EdgeIndexPrefixScan | 0 | | + | 0 | Start | | | + + Scenario: Edge with complex relational IN filter + # (a IN b) AND (c IN d) + # List has only 1 element, so prefixScan is applied + When profiling query: + """ + LOOKUP ON edge_1 + WHERE edge_1.col2_int IN [11 , 33] AND edge_1.col1_str IN ["Red1"] + YIELD edge_1.col1_str, edge_1.col2_int + """ + Then the result should be, in any order: + | SrcVID | DstVID | Ranking | edge_1.col1_str | edge_1.col2_int | + | "101" | "102" | 0 | "Red1" | 11 | + And the execution plan should be: + | id | name | dependencies | operator info | + | 3 | Project | 4 | | + | 4 | IndexScan | 0 | | + | 0 | Start | | | + # (a IN b) AND (c IN d) + # a, c both have indexes (4 prefixScan will be executed) + When profiling query: + """ + LOOKUP ON edge_1 + WHERE edge_1.col2_int IN [11 , 33] AND edge_1.col1_str IN ["Red1", "ABC"] + YIELD edge_1.col1_str, edge_1.col2_int + """ + Then the result should be, in any order: + | SrcVID | DstVID | Ranking | edge_1.col1_str | edge_1.col2_int | + | "101" | "102" | 0 | "Red1" | 11 | + And the execution plan should be: + | id | name | dependencies | operator info | + | 3 | Project | 4 | | + | 4 | IndexScan | 0 | | + | 0 | Start | | | + # (a IN b) AND (c IN d) + # a, c have a composite index + When executing query: + """ + CREATE EDGE INDEX composite_edge_index ON edge_1(col1_str(20), col2_int); + """ + Then the execution should be successful + And wait 6 seconds + When submit a job: + """ + REBUILD EDGE INDEX composite_edge_index + """ + Then wait the job to finish + When profiling query: + """ + LOOKUP ON edge_1 + WHERE edge_1.col2_int IN [11 , 33] AND edge_1.col1_str IN ["Red1", "ABC"] + YIELD edge_1.col1_str, edge_1.col2_int + """ + Then the result should be, in any order: + | SrcVID | DstVID | Ranking | edge_1.col1_str | edge_1.col2_int | + | "101" | "102" | 0 | "Red1" | 11 | + And the execution plan should be: + | id | name | dependencies | operator info | + | 3 | Project | 4 | | + | 4 | IndexScan | 0 | | + | 0 | Start | | | + # (a IN b) AND (c IN d) while only a has index + # first drop tag index + When executing query: + """ + DROP EDGE INDEX composite_edge_index + """ + Then the execution should be successful + When executing query: + """ + DROP EDGE INDEX col1_str_index + """ + Then the execution should be successful + And wait 6 seconds + # since the edge index has been dropped, here an EdgeIndexFullScan should be performed + When profiling query: + """ + LOOKUP ON edge_1 + WHERE edge_1.col1_str IN ["Red1", "ABC"] + YIELD edge_1.col1_str, edge_1.col2_int + """ + Then the result should be, in any order: + | SrcVID | DstVID | Ranking | edge_1.col1_str | edge_1.col2_int | + | "101" | "102" | 0 | "Red1" | 11 | + And the execution plan should be: + | id | name | dependencies | operator info | + | 3 | Project | 2 | | + | 2 | Filter | 4 | | + | 4 | EdgeIndexFullScan | 0 | | + | 0 | Start | | | + When profiling query: + """ + LOOKUP ON edge_1 + WHERE edge_1.col2_int IN [11 , 33] AND edge_1.col1_str IN ["Red1", "ABC"] + YIELD edge_1.col1_str, edge_1.col2_int + """ + Then the result should be, in any order: + | SrcVID | DstVID | Ranking | edge_1.col1_str | edge_1.col2_int | + | "101" | "102" | 0 | "Red1" | 11 | + And the execution plan should be: + | id | name | dependencies | operator info | + | 3 | Project | 4 | | + | 4 | IndexScan | 0 | | + | 0 | Start | | | + + Scenario: Edge with relational NOT IN filter When profiling query: """ LOOKUP ON edge_1 WHERE edge_1.col1_str NOT IN ["Blue"] YIELD edge_1.col1_str @@ -172,39 +286,16 @@ Feature: Lookup edge index full scan | 0 | Start | | | Scenario: Edge with relational CONTAINS/NOT CONTAINS filter - When profiling query: - """ - LOOKUP ON edge_1 WHERE edge_1.col1_str CONTAINS toLower("L") YIELD edge_1.col1_str - """ - Then the result should be, in any order: - | SrcVID | DstVID | Ranking | edge_1.col1_str | - | "103" | "101" | 0 | "Blue" | - | "102" | "103" | 0 | "Yellow" | - And the execution plan should be: - | id | name | dependencies | operator info | - | 3 | Project | 2 | | - | 2 | Filter | 4 | {"condition": "(edge_1.col1_str CONTAINS \"l\")"} | - | 4 | EdgeIndexFullScan | 0 | | - | 0 | Start | | | When executing query: """ LOOKUP ON edge_1 WHERE edge_1.col1_str CONTAINS "ABC" YIELD edge_1.col1_str """ - Then the result should be, in any order: - | SrcVID | DstVID | Ranking | edge_1.col1_str | - When profiling query: + Then a SemanticError should be raised at runtime: Expression (edge_1.col1_str CONTAINS "ABC") is not supported, please use full-text index as an optimal solution + When executing query: """ - LOOKUP ON edge_1 WHERE edge_1.col1_str NOT CONTAINS toLower("L") YIELD edge_1.col1_str + LOOKUP ON edge_1 WHERE edge_1.col1_str NOT CONTAINS "ABC" YIELD edge_1.col1_str """ - Then the result should be, in any order: - | SrcVID | DstVID | Ranking | edge_1.col1_str | - | "101" | "102" | 0 | "Red1" | - And the execution plan should be: - | id | name | dependencies | operator info | - | 3 | Project | 2 | | - | 2 | Filter | 4 | {"condition": "(edge_1.col1_str NOT CONTAINS \"l\")"} | - | 4 | EdgeIndexFullScan | 0 | | - | 0 | Start | | | + Then a SemanticError should be raised at runtime: Expression (edge_1.col1_str NOT CONTAINS "ABC") is not supported, please use full-text index as an optimal solution Scenario: Edge with relational STARTS/NOT STARTS WITH filter When profiling query: @@ -233,55 +324,18 @@ Feature: Lookup edge index full scan Then a SemanticError should be raised at runtime: Column type error : col1_str When profiling query: """ - LOOKUP ON edge_1 WHERE edge_1.col1_str NOT STARTS WITH toUpper("r") YIELD edge_1.col1_str + LOOKUP ON edge_1 WHERE edge_1.col1_str NOT STARTS WITH "R" YIELD edge_1.col1_str """ - Then the result should be, in any order: - | SrcVID | DstVID | Ranking | edge_1.col1_str | - | "103" | "101" | 0 | "Blue" | - | "102" | "103" | 0 | "Yellow" | - And the execution plan should be: - | id | name | dependencies | operator info | - | 3 | Project | 2 | | - | 2 | Filter | 4 | {"condition": "(edge_1.col1_str NOT STARTS WITH \"R\")"} | - | 4 | EdgeIndexFullScan | 0 | | - | 0 | Start | | | + Then a SemanticError should be raised at runtime: Expression (edge_1.col1_str NOT STARTS WITH "R") is not supported, please use full-text index as an optimal solution Scenario: Edge with relational ENDS/NOT ENDS WITH filter - When profiling query: - """ - LOOKUP ON edge_1 WHERE edge_1.col1_str ENDS WITH toLower("E") YIELD edge_1.col1_str - """ - Then the result should be, in any order: - | SrcVID | DstVID | Ranking | edge_1.col1_str | - | "103" | "101" | 0 | "Blue" | - And the execution plan should be: - | id | name | dependencies | operator info | - | 3 | Project | 2 | | - | 2 | Filter | 4 | {"condition": "(edge_1.col1_str ENDS WITH \"e\")"} | - | 4 | EdgeIndexFullScan | 0 | | - | 0 | Start | | | When executing query: """ LOOKUP ON edge_1 WHERE edge_1.col1_str ENDS WITH "ABC" YIELD edge_1.col1_str """ - Then the result should be, in any order: - | SrcVID | DstVID | Ranking | edge_1.col1_str | + Then a SemanticError should be raised at runtime: Expression (edge_1.col1_str ENDS WITH "ABC") is not supported, please use full-text index as an optimal solution When executing query: - """ - LOOKUP ON edge_1 WHERE edge_1.col1_str ENDS WITH 123 YIELD edge_1.col1_str - """ - Then a SemanticError should be raised at runtime: Column type error : col1_str - When profiling query: """ LOOKUP ON edge_1 WHERE edge_1.col1_str NOT ENDS WITH toLower("E") YIELD edge_1.col1_str """ - Then the result should be, in any order: - | SrcVID | DstVID | Ranking | edge_1.col1_str | - | "101" | "102" | 0 | "Red1" | - | "102" | "103" | 0 | "Yellow" | - And the execution plan should be: - | id | name | dependencies | operator info | - | 3 | Project | 2 | | - | 2 | Filter | 4 | {"condition": "(edge_1.col1_str NOT ENDS WITH \"e\")"} | - | 4 | EdgeIndexFullScan | 0 | | - | 0 | Start | | | + Then a SemanticError should be raised at runtime: Expression (edge_1.col1_str NOT ENDS WITH toLower("E")) is not supported, please use full-text index as an optimal solution diff --git a/tests/tck/features/lookup/LookupEdge2.feature b/tests/tck/features/lookup/LookupEdge2.feature index 1e7f6a536ed..aeac34348f3 100644 --- a/tests/tck/features/lookup/LookupEdge2.feature +++ b/tests/tck/features/lookup/LookupEdge2.feature @@ -27,6 +27,11 @@ Feature: Test lookup on edge index 2 """ Scenario Outline: [edge] Simple test cases + When executing query: + """ + LOOKUP ON lookup_edge_1 WHERE lookup_edge_1.col1 == 201 OR lookup_edge_1.col2 == 201 AND lookup_edge_1.col3 == 202 + """ + Then the execution should be successful When executing query: """ LOOKUP ON lookup_edge_1 WHERE col1 == 201 @@ -37,11 +42,6 @@ Feature: Test lookup on edge index 2 LOOKUP ON lookup_edge_1 WHERE lookup_edge_1.col1 == 201 OR lookup_edge_1.col5 == 201 """ Then a SemanticError should be raised at runtime: Invalid column: col5 - When executing query: - """ - LOOKUP ON lookup_edge_1 WHERE lookup_edge_1.col1 == 201 OR lookup_edge_1.col2 == 201 AND lookup_edge_1.col3 == 202 - """ - Then a SemanticError should be raised at runtime: Not supported filter When executing query: """ LOOKUP ON lookup_edge_1 WHERE lookup_edge_1.col1 == 300 diff --git a/tests/tck/features/lookup/LookupTag2.feature b/tests/tck/features/lookup/LookupTag2.feature index 46b594b0aa1..50616a8a4ef 100644 --- a/tests/tck/features/lookup/LookupTag2.feature +++ b/tests/tck/features/lookup/LookupTag2.feature @@ -28,6 +28,11 @@ Feature: Test lookup on tag index 2 """ Scenario Outline: [tag] simple tag test cases + When executing query: + """ + LOOKUP ON lookup_tag_1 WHERE lookup_tag_1.col1 == 201 OR lookup_tag_1.col2 == 201 AND lookup_tag_1.col3 == 202 + """ + Then the execution should be successful When executing query: """ LOOKUP ON lookup_tag_1 WHERE col1 == 200; @@ -38,11 +43,6 @@ Feature: Test lookup on tag index 2 LOOKUP ON lookup_tag_1 WHERE lookup_tag_1.col1 == 200 OR lookup_tag_1.col5 == 20; """ Then a SemanticError should be raised at runtime: Invalid column: col5 - When executing query: - """ - LOOKUP ON lookup_tag_1 WHERE lookup_tag_1.col1 == 201 OR lookup_tag_1.col2 == 201 AND lookup_tag_1.col3 == 202 - """ - Then a SemanticError should be raised at runtime: Not supported filter When executing query: """ LOOKUP ON lookup_tag_1 WHERE lookup_tag_1.col1 == 300 diff --git a/tests/tck/features/lookup/TagIndexFullScan.feature b/tests/tck/features/lookup/TagIndexFullScan.feature index 22fb26b9738..85e189d9b0c 100644 --- a/tests/tck/features/lookup/TagIndexFullScan.feature +++ b/tests/tck/features/lookup/TagIndexFullScan.feature @@ -8,46 +8,7 @@ Feature: Lookup tag index full scan """ LOOKUP ON team where team.name =~ "\\d+\\w+" """ - Then the result should be, in any order: - | VertexID | - | "76ers" | - When executing query: - """ - LOOKUP ON team where team.name =~ "\\w+ea\\w+" - """ - Then the result should be, in any order: - | VertexID | - | "Heat" | - - # skip because `make fmt` will delete '\' in the operator info and causes tests fail - @skip - Scenario: Tag with relational RegExp filter[2] - When profiling query: - """ - LOOKUP ON team where team.name =~ "\\d+\\w+" - """ - Then the result should be, in any order: - | VertexID | - | "76ers" | - And the execution plan should be: - | id | name | dependencies | operator info | - | 3 | Project | 2 | | - | 2 | Filter | 4 | {"condition": "(team.name=~\"\d+\w+\")"} | - | 4 | TagIndexFullScan | 0 | | - | 0 | Start | | | - When profiling query: - """ - LOOKUP ON team where team.name =~ "\\w+ea\\w+" - """ - Then the result should be, in any order: - | VertexID | - | "Heat" | - And the execution plan should be: - | id | name | dependencies | operator info | - | 3 | Project | 2 | | - | 2 | Filter | 4 | {"condition": "(team.name=~\"\w+ea\w+\")"} | - | 4 | TagIndexFullScan | 0 | | - | 0 | Start | | | + Then a SemanticError should be raised at runtime: Expression (team.name=~"\d+\w+") is not supported, please use full-text index as an optimal solution Scenario: Tag with relational NE filter When profiling query: @@ -92,7 +53,8 @@ Feature: Lookup tag index full scan | 4 | TagIndexFullScan | 0 | | | 0 | Start | | | - Scenario: Tag with relational IN/NOT IN filter + # TODO: Support compare operator info that has multiple column hints + Scenario: Tag with simple relational IN filter When profiling query: """ LOOKUP ON team WHERE team.name IN ["Hornets", "Jazz"] @@ -102,11 +64,10 @@ Feature: Lookup tag index full scan | "Jazz" | | "Hornets" | And the execution plan should be: - | id | name | dependencies | operator info | - | 3 | Project | 2 | | - | 2 | Filter | 4 | {"condition": "(team.name IN [\"Hornets\",\"Jazz\"])"} | - | 4 | TagIndexFullScan | 0 | | - | 0 | Start | | | + | id | name | dependencies | operator info | + | 3 | Project | 4 | | + | 4 | IndexScan | 0 | | + | 0 | Start | | | When executing query: """ LOOKUP ON team WHERE team.name IN ["non-existed-name"] @@ -124,11 +85,164 @@ Feature: Lookup tag index full scan | "Tony Parker" | 36 | | "Boris Diaw" | 36 | And the execution plan should be: - | id | name | dependencies | operator info | - | 3 | Project | 2 | | - | 2 | Filter | 4 | {"condition": "(player.age IN [39,36])"} | - | 4 | TagIndexFullScan | 0 | | - | 0 | Start | | | + | id | name | dependencies | operator info | + | 3 | Project | 4 | | + | 4 | IndexScan | 0 | | + | 0 | Start | | | + # (a IN b) OR c + When profiling query: + """ + LOOKUP ON player WHERE player.age IN [40, 25] OR player.name == "ABC" YIELD player.age + """ + Then the result should be, in any order: + | VertexID | player.age | + | "Dirk Nowitzki" | 40 | + | "Joel Embiid" | 25 | + | "Kobe Bryant" | 40 | + | "Kyle Anderson" | 25 | + And the execution plan should be: + | id | name | dependencies | operator info | + | 3 | Project | 4 | | + | 4 | IndexScan | 0 | | + | 0 | Start | | | + # (a IN b) OR (c IN d) + When profiling query: + """ + LOOKUP ON player WHERE player.age IN [40, 25] OR player.name IN ["Kobe Bryant"] YIELD player.age + """ + Then the result should be, in any order: + | VertexID | player.age | + | "Dirk Nowitzki" | 40 | + | "Joel Embiid" | 25 | + | "Kobe Bryant" | 40 | + | "Kyle Anderson" | 25 | + And the execution plan should be: + | id | name | dependencies | operator info | + | 3 | Project | 4 | | + | 4 | IndexScan | 0 | | + | 0 | Start | | | + # (a IN b) AND c + When profiling query: + """ + LOOKUP ON player WHERE player.age IN [40, 25] AND player.name == "Kobe Bryant" YIELD player.age + """ + Then the result should be, in any order: + | VertexID | player.age | + | "Kobe Bryant" | 40 | + And the execution plan should be: + | id | name | dependencies | operator info | + | 3 | Project | 4 | | + | 4 | IndexScan | 0 | | + | 0 | Start | | | + When profiling query: + """ + LOOKUP ON player WHERE player.name IN ["Kobe Bryant", "Tim Duncan"] AND player.age > 30 + """ + Then the result should be, in any order: + | VertexID | + | "Kobe Bryant" | + | "Tim Duncan" | + And the execution plan should be: + | id | name | dependencies | operator info | + | 3 | Project | 4 | | + | 4 | IndexScan | 0 | | + | 0 | Start | | | + # c AND (a IN b) + When profiling query: + """ + LOOKUP ON player WHERE player.age IN [40, 25] AND player.name == "Kobe Bryant" YIELD player.age + """ + Then the result should be, in any order: + | VertexID | player.age | + | "Kobe Bryant" | 40 | + And the execution plan should be: + | id | name | dependencies | operator info | + | 3 | Project | 4 | | + | 4 | IndexScan | 0 | | + | 0 | Start | | | + + Scenario: Tag with complex relational IN filter + Given an empty graph + And load "nba" csv data to a new space + # (a IN b) AND (c IN d) while a, c both have indexes + When profiling query: + """ + LOOKUP ON player WHERE player.age IN [40, 25] AND player.name IN ["ABC", "Kobe Bryant"] YIELD player.age + """ + Then the result should be, in any order: + | VertexID | player.age | + | "Kobe Bryant" | 40 | + And the execution plan should be: + | id | name | dependencies | operator info | + | 3 | Project | 4 | | + | 4 | IndexScan | 0 | | + | 0 | Start | | | + # (a IN b) AND (c IN d) while a, c have a composite index + When executing query: + """ + CREATE TAG INDEX composite_player_name_age_index ON player(name(64), age); + """ + Then the execution should be successful + And wait 6 seconds + When submit a job: + """ + REBUILD TAG INDEX composite_player_name_age_index + """ + Then wait the job to finish + When profiling query: + """ + LOOKUP ON player WHERE player.age IN [40, 25] AND player.name IN ["ABC", "Kobe Bryant"] YIELD player.age + """ + Then the result should be, in any order: + | VertexID | player.age | + | "Kobe Bryant" | 40 | + And the execution plan should be: + | id | name | dependencies | operator info | + | 3 | Project | 4 | | + | 4 | IndexScan | 0 | | + | 0 | Start | | | + # (a IN b) AND (c IN d) while only a has index + # first drop tag index + When executing query: + """ + DROP TAG INDEX composite_player_name_age_index + """ + Then the execution should be successful + When executing query: + """ + DROP TAG INDEX player_name_index + """ + Then the execution should be successful + And wait 6 seconds + # since the tag index has been dropped, here a TagIndexFullScan should be performed + When profiling query: + """ + LOOKUP ON player WHERE player.name IN ["ABC", "Kobe Bryant"] YIELD player.age + """ + Then the result should be, in any order: + | VertexID | player.age | + | "Kobe Bryant" | 40 | + And the execution plan should be: + | id | name | dependencies | operator info | + | 3 | Project | 2 | | + | 2 | Filter | 4 | | + | 4 | TagIndexFullScan | 0 | | + | 0 | Start | | | + When profiling query: + """ + LOOKUP ON player WHERE player.age IN [40, 25] AND player.name IN ["ABC", "Kobe Bryant"] YIELD player.age + """ + Then the result should be, in any order: + | VertexID | player.age | + | "Kobe Bryant" | 40 | + And the execution plan should be: + | id | name | dependencies | operator info | + | 3 | Project | 4 | | + | 4 | IndexScan | 0 | | + | 0 | Start | | | + Then drop the used space + + Scenario: Tag with relational NOT IN filter When profiling query: """ LOOKUP ON team WHERE team.name NOT IN ["Hornets", "Jazz"] @@ -235,68 +349,18 @@ Feature: Lookup tag index full scan | 0 | Start | | | Scenario: Tag with relational CONTAINS/NOT CONTAINS filter - When profiling query: - """ - LOOKUP ON team WHERE team.name CONTAINS toLower("ER") - """ - Then the result should be, in any order: - | VertexID | - | "76ers" | - | "Trail Blazers" | - | "Timberwolves" | - | "Cavaliers" | - | "Thunders" | - | "Clippers" | - | "Pacers" | - | "Mavericks" | - | "Lakers" | - And the execution plan should be: - | id | name | dependencies | operator info | - | 3 | Project | 2 | | - | 2 | Filter | 4 | {"condition": "(team.name CONTAINS \"er\")"} | - | 4 | TagIndexFullScan | 0 | | - | 0 | Start | | | When executing query: """ LOOKUP ON team WHERE team.name CONTAINS "ABC" """ - Then the result should be, in any order: - | VertexID | - When profiling query: + Then a SemanticError should be raised at runtime: Expression (team.name CONTAINS "ABC") is not supported, please use full-text index as an optimal solution + When executing query: """ - LOOKUP ON team WHERE team.name NOT CONTAINS toLower("ER") + LOOKUP ON team WHERE team.name NOT CONTAINS "ABC" """ - Then the result should be, in any order: - | VertexID | - | "Wizards" | - | "Bucks" | - | "Bulls" | - | "Warriors" | - | "Celtics" | - | "Suns" | - | "Grizzlies" | - | "Hawks" | - | "Heat" | - | "Hornets" | - | "Jazz" | - | "Kings" | - | "Knicks" | - | "Spurs" | - | "Magic" | - | "Rockets" | - | "Nets" | - | "Nuggets" | - | "Raptors" | - | "Pelicans" | - | "Pistons" | - And the execution plan should be: - | id | name | dependencies | operator info | - | 3 | Project | 2 | | - | 2 | Filter | 4 | {"condition": "(team.name NOT CONTAINS \"er\")"} | - | 4 | TagIndexFullScan | 0 | | - | 0 | Start | | | + Then a SemanticError should be raised at runtime: Expression (team.name NOT CONTAINS "ABC") is not supported, please use full-text index as an optimal solution - Scenario: Tag with relational STARTS/NOT STARTS WITH filter + Scenario: Tag with relational STARTS WITH filter When profiling query: """ LOOKUP ON team WHERE team.name STARTS WITH toUpper("t") @@ -327,105 +391,16 @@ Feature: Lookup tag index full scan """ LOOKUP ON team WHERE team.name NOT STARTS WITH toUpper("t") """ - Then the result should be, in any order: - | VertexID | - | "76ers" | - | "Bucks" | - | "Bulls" | - | "Cavaliers" | - | "Celtics" | - | "Clippers" | - | "Grizzlies" | - | "Hawks" | - | "Heat" | - | "Hornets" | - | "Jazz" | - | "Kings" | - | "Knicks" | - | "Lakers" | - | "Magic" | - | "Mavericks" | - | "Nets" | - | "Nuggets" | - | "Pacers" | - | "Pelicans" | - | "Pistons" | - | "Raptors" | - | "Rockets" | - | "Spurs" | - | "Suns" | - | "Wizards" | - | "Warriors" | - And the execution plan should be: - | id | name | dependencies | operator info | - | 3 | Project | 2 | | - | 2 | Filter | 4 | {"condition": "(team.name NOT STARTS WITH \"T\")"} | - | 4 | TagIndexFullScan | 0 | | - | 0 | Start | | | + Then a SemanticError should be raised at runtime: Expression (team.name NOT STARTS WITH toUpper("t")) is not supported, please use full-text index as an optimal solution Scenario: Tag with relational ENDS/NOT ENDS WITH filter - When profiling query: - """ - LOOKUP ON team WHERE team.name ENDS WITH toLower("S") - """ - Then the result should be, in any order: - | VertexID | - | "76ers" | - | "Bucks" | - | "Bulls" | - | "Cavaliers" | - | "Celtics" | - | "Clippers" | - | "Grizzlies" | - | "Hawks" | - | "Wizards" | - | "Hornets" | - | "Warriors" | - | "Kings" | - | "Knicks" | - | "Lakers" | - | "Trail Blazers" | - | "Mavericks" | - | "Nets" | - | "Nuggets" | - | "Pacers" | - | "Pelicans" | - | "Pistons" | - | "Raptors" | - | "Rockets" | - | "Spurs" | - | "Suns" | - | "Thunders" | - | "Timberwolves" | - And the execution plan should be: - | id | name | dependencies | operator info | - | 3 | Project | 2 | | - | 2 | Filter | 4 | {"condition": "(team.name ENDS WITH \"s\")"} | - | 4 | TagIndexFullScan | 0 | | - | 0 | Start | | | When executing query: """ - LOOKUP ON team WHERE team.name ENDS WITH "ABC" + LOOKUP ON team WHERE team.name ENDS WITH toLower("S") """ - Then the result should be, in any order: - | VertexID | + Then a SemanticError should be raised at runtime: Expression (team.name ENDS WITH toLower("S")) is not supported, please use full-text index as an optimal solution When executing query: - """ - LOOKUP ON team WHERE team.name ENDS WITH 123 - """ - Then a SemanticError should be raised at runtime: Column type error : name - When profiling query: """ LOOKUP ON team WHERE team.name NOT ENDS WITH toLower("S") """ - Then the result should be, in any order: - | VertexID | - | "Magic" | - | "Jazz" | - | "Heat" | - And the execution plan should be: - | id | name | dependencies | operator info | - | 3 | Project | 2 | | - | 2 | Filter | 4 | {"condition": "(team.name NOT ENDS WITH \"s\")"} | - | 4 | TagIndexFullScan | 0 | | - | 0 | Start | | | + Then a SemanticError should be raised at runtime: Expression (team.name NOT ENDS WITH toLower("S")) is not supported, please use full-text index as an optimal solution diff --git a/tests/tck/features/match/SeekById.feature b/tests/tck/features/match/SeekById.feature index ebdba05960a..e4793ecea7b 100644 --- a/tests/tck/features/match/SeekById.feature +++ b/tests/tck/features/match/SeekById.feature @@ -138,6 +138,15 @@ Feature: Match seek by id Then the result should be, in any order: | Name | | 'James Harden' | + When executing query: + """ + MATCH (v:player) + WHERE id(v) IN ['James Harden', v.age] + RETURN v.name AS Name + """ + Then the result should be, in any order: + | Name | + | 'James Harden' | Scenario: complicate logical When executing query: @@ -251,6 +260,13 @@ Feature: Match seek by id RETURN v.name AS Name """ Then a SemanticError should be raised at runtime: + When executing query: + """ + MATCH (v) + WHERE id(v) IN ['James Harden', v.name] + RETURN v.name AS Name + """ + Then a SemanticError should be raised at runtime: Scenario: Start from end When executing query: diff --git a/tests/tck/features/match/SeekById.intVid.feature b/tests/tck/features/match/SeekById.intVid.feature index d1d427c3229..02a9f94c806 100644 --- a/tests/tck/features/match/SeekById.intVid.feature +++ b/tests/tck/features/match/SeekById.intVid.feature @@ -138,6 +138,15 @@ Feature: Match seek by id Then the result should be, in any order: | Name | | 'James Harden' | + When executing query: + """ + MATCH (v:player) + WHERE id(v) IN [hash('James Harden'), v.age] + RETURN v.name AS Name + """ + Then the result should be, in any order: + | Name | + | 'James Harden' | Scenario: complicate logical When executing query: @@ -244,6 +253,13 @@ Feature: Match seek by id RETURN v.name AS Name """ Then a SemanticError should be raised at runtime: + When executing query: + """ + MATCH (v) + WHERE id(v) IN [hash('James Harden'), v.name] + RETURN v.name AS Name + """ + Then a SemanticError should be raised at runtime: Scenario: with arithmetic When executing query: