| 1 | #include "duckdb/parser/expression/operator_expression.hpp" |
|---|---|
| 2 | #include "duckdb/parser/expression/subquery_expression.hpp" |
| 3 | #include "duckdb/parser/transformer.hpp" |
| 4 | |
| 5 | using namespace duckdb; |
| 6 | using namespace std; |
| 7 | |
| 8 | unique_ptr<ParsedExpression> Transformer::TransformSubquery(PGSubLink *root) { |
| 9 | if (!root) { |
| 10 | return nullptr; |
| 11 | } |
| 12 | auto subquery_expr = make_unique<SubqueryExpression>(); |
| 13 | subquery_expr->subquery = TransformSelectNode((PGSelectStmt *)root->subselect); |
| 14 | if (!subquery_expr->subquery) { |
| 15 | return nullptr; |
| 16 | } |
| 17 | assert(subquery_expr->subquery->GetSelectList().size() > 0); |
| 18 | |
| 19 | switch (root->subLinkType) { |
| 20 | case PG_EXISTS_SUBLINK: { |
| 21 | subquery_expr->subquery_type = SubqueryType::EXISTS; |
| 22 | break; |
| 23 | } |
| 24 | case PG_ANY_SUBLINK: |
| 25 | case PG_ALL_SUBLINK: { |
| 26 | // comparison with ANY() or ALL() |
| 27 | subquery_expr->subquery_type = SubqueryType::ANY; |
| 28 | subquery_expr->child = TransformExpression(root->testexpr); |
| 29 | // get the operator name |
| 30 | if (!root->operName) { |
| 31 | // simple IN |
| 32 | subquery_expr->comparison_type = ExpressionType::COMPARE_EQUAL; |
| 33 | } else { |
| 34 | auto operator_name = string((reinterpret_cast<PGValue *>(root->operName->head->data.ptr_value))->val.str); |
| 35 | subquery_expr->comparison_type = OperatorToExpressionType(operator_name); |
| 36 | } |
| 37 | assert(subquery_expr->comparison_type == ExpressionType::COMPARE_EQUAL || |
| 38 | subquery_expr->comparison_type == ExpressionType::COMPARE_NOTEQUAL || |
| 39 | subquery_expr->comparison_type == ExpressionType::COMPARE_GREATERTHAN || |
| 40 | subquery_expr->comparison_type == ExpressionType::COMPARE_GREATERTHANOREQUALTO || |
| 41 | subquery_expr->comparison_type == ExpressionType::COMPARE_LESSTHAN || |
| 42 | subquery_expr->comparison_type == ExpressionType::COMPARE_LESSTHANOREQUALTO); |
| 43 | if (root->subLinkType == PG_ALL_SUBLINK) { |
| 44 | // ALL sublink is equivalent to NOT(ANY) with inverted comparison |
| 45 | // e.g. [= ALL()] is equivalent to [NOT(<> ANY())] |
| 46 | // first invert the comparison type |
| 47 | subquery_expr->comparison_type = NegateComparisionExpression(subquery_expr->comparison_type); |
| 48 | return make_unique<OperatorExpression>(ExpressionType::OPERATOR_NOT, move(subquery_expr)); |
| 49 | } |
| 50 | break; |
| 51 | } |
| 52 | case PG_EXPR_SUBLINK: { |
| 53 | // return a single scalar value from the subquery |
| 54 | // no child expression to compare to |
| 55 | subquery_expr->subquery_type = SubqueryType::SCALAR; |
| 56 | break; |
| 57 | } |
| 58 | default: |
| 59 | throw NotImplementedException("Subquery of type %d not implemented\n", (int)root->subLinkType); |
| 60 | } |
| 61 | return move(subquery_expr); |
| 62 | } |
| 63 |