/** * Copyright (c) 2021 OceanBase * OceanBase CE is licensed under Mulan PubL v2. * You can use this software according to the terms and conditions of the Mulan PubL v2. * You may obtain a copy of Mulan PubL v2 at: * http://license.coscl.org.cn/MulanPubL-2.0 * THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND, * EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT, * MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE. * See the Mulan PubL v2 for more details. */ #define USING_LOG_PREFIX SQL_REWRITE #include "sql/rewrite/ob_transform_query_push_down.h" #include "sql/resolver/dml/ob_insert_stmt.h" #include "sql/rewrite/ob_transform_utils.h" #include "sql/optimizer/ob_optimizer_util.h" #include "common/ob_smart_call.h" namespace oceanbase { using namespace common; namespace sql { /** * @brief ObTransformQueryPushDown::transform_one_stmt * 1.where condition * select * from (select * from t1 order by c3) where c2 > 2; * ==> * select * from t1 where c2 > 2 order by c3; * * select c2, c3 from (select c2, c3 from t1 group by c2, c3) where c2 > 2; * ==> * select c2, c3 from t1 group by c2, c3 having c2 > 2; * * 2.group by(having) * select c2 from (select c2 from t1) group by c2; * ==> * select c2 from t1 group by c2; * select c3 from (select c3 from t1) group by c3 having c3 > 5; * ==> * select c3 from t1 group by c3 having c3 > 5; * * 3.window function * select row_number() OVER() from (select * from t1); * ==> * select row_number() OVER() from t1; * * 4.aggr * select sum(c3) from (select * from t1) group by c3; * ==> * select sum(c3) from t1 group by c3; * * 5.distinct * select distinct * from (select * from t1 order by c2); * ==> * select distinct * from t1 order by c2; * * 6.order by * select * from (select * from t1 order by c2) order by c3; * ==> * select * from t1 order by c3; * * 7.limit * select * from (select t1.a, sum(b) from t1 group by t1.a) limit 5 * ==> * select t1.a, sum(b) from t1 group by t1.a limit 5 * * 8.rownum * select * from (select * from t1 where rownum = 1) * ==> * select * from t1 where rownum = 1 * */ int ObTransformQueryPushDown::transform_one_stmt( common::ObIArray& parent_stmts, ObDMLStmt*& stmt, bool& trans_happened) { int ret = OB_SUCCESS; UNUSED(parent_stmts); trans_happened = false; ObSelectStmt* select_stmt = NULL; // outer select stmt ObSelectStmt* view_stmt = NULL; // inner select stmt bool can_transform = false; // can rewrite bool need_distinct = false; // need distinct bool transform_having = false; // can push where condition to having condition ObSEArray select_offset; // record select item offset postition, used to adjust set-op stmt output // postition ObSEArray const_select_items; // record const select item, used to adjust set-op stmt output postition if (OB_ISNULL(stmt)) { ret = OB_INVALID_ARGUMENT; LOG_WARN("stmt is NULL", K(ret)); } else if (!stmt->is_select_stmt()) { // do nothing } else if (FALSE_IT(select_stmt = static_cast(stmt))) { // do nothing } else if (1 == select_stmt->get_from_item_size() && // only one table reference !select_stmt->get_from_item(0).is_joined_ && select_stmt->get_semi_infos().empty()) { const FromItem& cur_from = select_stmt->get_from_item(0); const TableItem* view_table_item = NULL; if (OB_ISNULL(view_table_item = select_stmt->get_table_item_by_id(cur_from.table_id_))) { ret = OB_ERR_UNEXPECTED; LOG_WARN("failed to get table item", K(ret), K(cur_from)); } else if (!view_table_item->is_generated_table()) { // do nothing } else if (OB_ISNULL(view_stmt = view_table_item->ref_query_)) { ret = OB_ERR_UNEXPECTED; LOG_WARN("view stmt is null", K(ret)); } else if (OB_FAIL(check_transform_validity(select_stmt, view_stmt, can_transform, need_distinct, transform_having, select_offset, const_select_items))) { LOG_WARN("gather transform information failed", K(ret)); } else if (!can_transform) { /*do nothing*/ } else if (OB_FAIL(do_transform(select_stmt, view_stmt, need_distinct, transform_having, view_table_item, select_offset, const_select_items))) { LOG_WARN("do transform failed", K(ret), K(stmt)); } else if (OB_ISNULL(stmt = view_stmt)) { ret = OB_ERR_UNEXPECTED; LOG_WARN("view stmt is null", K(ret), K(view_stmt)); } else { trans_happened = true; } } else { /*do nothing*/ } LOG_TRACE("succeed to push query down", K(trans_happened)); return ret; } int ObTransformQueryPushDown::check_transform_validity(ObSelectStmt* select_stmt, ObSelectStmt* view_stmt, bool& can_transform, bool& need_distinct, bool& transform_having, ObIArray& select_offset, ObIArray& const_select_items) { int ret = OB_SUCCESS; bool check_status = false; can_transform = false; need_distinct = false; transform_having = false; if (OB_ISNULL(select_stmt) || OB_ISNULL(view_stmt)) { ret = OB_INVALID_ARGUMENT; LOG_WARN("stmt is NULL", K(select_stmt), K(view_stmt), K(ret)); } else if (view_stmt->get_stmt_hint().enable_no_view_merge() || select_stmt->is_recursive_union() || view_stmt->is_hierarchical_query() || (view_stmt->is_recursive_union() && !select_stmt->is_spj()) || select_stmt->has_set_op() || (select_stmt->has_sequence() && !view_stmt->is_spj()) || select_stmt->need_temp_table_trans()) { can_transform = false; } else if (OB_FAIL(check_rownum_push_down(select_stmt, view_stmt, check_status))) { LOG_WARN("check rownum push down failed", K(check_status), K(ret)); } else if (!check_status) { can_transform = false; } else if (OB_FAIL(check_select_item_push_down( select_stmt, view_stmt, select_offset, const_select_items, check_status))) { LOG_WARN("check select item push down failed"); } else if (!check_status) { can_transform = false; } else if (select_stmt->get_condition_size() > 0 && OB_FAIL(check_where_condition_push_down(select_stmt, view_stmt, transform_having, check_status))) { LOG_WARN("check where condition push down failed", K(check_status), K(ret)); } else if (!check_status) { can_transform = false; } else if ((select_stmt->has_group_by() || select_stmt->has_rollup()) && !view_stmt->is_spj()) { can_transform = false; } else if (select_stmt->has_window_function() && OB_FAIL(check_window_function_push_down(view_stmt, check_status))) { LOG_WARN("check window function push down failed", K(check_status), K(ret)); } else if (!check_status) { can_transform = false; } else if (select_stmt->has_distinct() && OB_FAIL(check_distinct_push_down(view_stmt, need_distinct, check_status))) { LOG_WARN("check distinct push down failed", K(check_status), K(ret)); } else if (!check_status) { can_transform = false; } else if (select_stmt->has_order_by() && view_stmt->has_limit()) { can_transform = false; } else if (select_stmt->has_limit()) { can_transform = (!select_stmt->is_calc_found_rows() && !view_stmt->is_contains_assignment() && can_limit_merge(*select_stmt, *view_stmt)); } else { can_transform = true; } return ret; } // select ... from (select ... limit a1 offset b1) limit a2 offset b2; // select ... from ... limit min(a1 - b2, a2) offset b1 + b2; bool ObTransformQueryPushDown::can_limit_merge(ObSelectStmt& upper_stmt, ObSelectStmt& view_stmt) { bool can_merge = false; if (!view_stmt.has_limit()) { can_merge = true; } else if (NULL == upper_stmt.get_limit_percent_expr() && NULL == view_stmt.get_limit_percent_expr() && !upper_stmt.is_fetch_with_ties() && !view_stmt.is_fetch_with_ties()) { can_merge = true; } else { can_merge = false; } return can_merge; } int ObTransformQueryPushDown::do_limit_merge(ObSelectStmt& upper_stmt, ObSelectStmt& view_stmt) { int ret = OB_SUCCESS; ObRawExpr* limit_expr = NULL; ObRawExpr* offset_expr = NULL; if (!upper_stmt.has_limit()) { /*do nothing*/ } else if (!can_limit_merge(upper_stmt, view_stmt)) { ret = OB_ERR_UNEXPECTED; LOG_WARN("unexpected stmt input", K(ret), K(upper_stmt), K(view_stmt)); } else if (!view_stmt.has_limit()) { view_stmt.set_limit_offset(upper_stmt.get_limit_expr(), upper_stmt.get_offset_expr()); view_stmt.set_limit_percent_expr(upper_stmt.get_limit_percent_expr()); view_stmt.set_fetch_with_ties(upper_stmt.is_fetch_with_ties()); view_stmt.set_has_fetch(upper_stmt.has_fetch()); } else if (OB_FAIL(ObTransformUtils::merge_limit_offset(ctx_, view_stmt.get_limit_expr(), upper_stmt.get_limit_expr(), view_stmt.get_offset_expr(), upper_stmt.get_offset_expr(), limit_expr, offset_expr))) { LOG_WARN("failed to merge limit offset", K(ret)); } else { view_stmt.set_limit_offset(limit_expr, offset_expr); } return ret; } int ObTransformQueryPushDown::is_select_item_same(ObSelectStmt* select_stmt, ObSelectStmt* view_stmt, bool& is_same, ObIArray& select_offset, ObIArray& const_select_items) { int ret = OB_SUCCESS; is_same = false; if (OB_ISNULL(select_stmt) || OB_ISNULL(view_stmt)) { ret = OB_INVALID_ARGUMENT; LOG_WARN("stmt is NULL", K(select_stmt), K(view_stmt), K(ret)); } else if (select_stmt->get_select_item_size() < view_stmt->get_select_item_size()) { is_same = false; } else { is_same = true; ObBitSet<> column_id; bool is_same_exactly = true; int64_t column_count = 0; for (int64_t i = 0; OB_SUCC(ret) && is_same && i < select_stmt->get_select_item_size(); ++i) { const ObRawExpr* sel_expr = NULL; const ObColumnRefRawExpr* column_expr = NULL; if (OB_ISNULL(sel_expr = select_stmt->get_select_item(i).expr_)) { ret = OB_ERR_UNEXPECTED; LOG_WARN("select expr is null", K(ret), K(sel_expr)); } else if (sel_expr->has_const_or_const_expr_flag()) { if (OB_FAIL(const_select_items.push_back(select_stmt->get_select_item(i)))) { LOG_WARN("failed to push back select item"); } else if (OB_FAIL(select_offset.push_back(-1))) { //-1 meanings const expr LOG_WARN("failed to push back select offset"); } else { is_same_exactly = false; } } else if (!sel_expr->is_column_ref_expr() || sel_expr->get_expr_level() != select_stmt->get_current_level()) { is_same = false; } else if (OB_FAIL(select_offset.push_back( static_cast(sel_expr)->get_column_id() - OB_APP_MIN_COLUMN_ID))) { LOG_WARN("push back location offset failed", K(ret)); } else { ++column_count; column_expr = static_cast(sel_expr); if (column_id.has_member(column_expr->get_column_id())) { is_same = false; is_same_exactly = false; } else { column_id.add_member(column_expr->get_column_id()); } is_same_exactly &= (static_cast(sel_expr)->get_column_id() == i + OB_APP_MIN_COLUMN_ID); } } if (OB_SUCC(ret) && is_same) { if (column_count == view_stmt->get_select_item_size()) { /*do nothing*/ // if outer output is const expr, then inner output must be const expr, eg:select 1 from (select 2 from t1) } else if (const_select_items.count() == select_stmt->get_select_item_size()) { for (int64_t i = 0; OB_SUCC(ret) && is_same && i < view_stmt->get_select_item_size(); ++i) { ObRawExpr* select_expr = NULL; if (OB_ISNULL(select_expr = view_stmt->get_select_item(i).expr_)) { ret = OB_ERR_UNEXPECTED; LOG_WARN("select expr is null", K(ret), K(select_expr)); } else if (select_expr->is_set_op_expr()) { if (OB_FAIL(check_set_op_is_const_expr(view_stmt, select_expr, is_same))) { LOG_WARN("failed to check set op is const expr", K(ret)); } } else { is_same = select_expr->has_const_or_const_expr_flag(); } } } else { is_same = false; } if (OB_SUCC(ret) && is_same && is_same_exactly) { select_offset.reset(); const_select_items.reset(); } } } return ret; } int ObTransformQueryPushDown::check_rownum_push_down(ObSelectStmt* select_stmt, ObSelectStmt* view_stmt, bool& can_be) { int ret = OB_SUCCESS; can_be = false; bool select_has_rownum = false; bool view_has_rownum = false; if (OB_ISNULL(select_stmt) || OB_ISNULL(view_stmt)) { ret = OB_INVALID_ARGUMENT; LOG_WARN("stmt is NULL", K(select_stmt), K(view_stmt), K(ret)); } else if (OB_FAIL(select_stmt->has_rownum(select_has_rownum))) { LOG_WARN("check stmt has rownum failed", K(select_stmt), K(ret)); } else if (OB_FAIL(view_stmt->has_rownum(view_has_rownum))) { LOG_WARN("check stmt has rownum failed", K(view_stmt), K(ret)); } else if (select_has_rownum) { can_be = false; } else if (view_has_rownum && (select_stmt->get_condition_size() > 0 || select_stmt->has_group_by() || select_stmt->has_rollup() || select_stmt->has_window_function() || select_stmt->has_set_op() || select_stmt->has_order_by())) { can_be = false; } else { can_be = true; } return ret; } int ObTransformQueryPushDown::check_select_item_push_down(ObSelectStmt* select_stmt, ObSelectStmt* view_stmt, ObIArray& select_offset, ObIArray& const_select_items, bool& can_be) { int ret = OB_SUCCESS; can_be = false; bool check_status = false; if (OB_ISNULL(select_stmt) || OB_ISNULL(view_stmt)) { ret = OB_INVALID_ARGUMENT; LOG_WARN("stmt is NULL", K(select_stmt), K(view_stmt), K(ret)); } else if (OB_FAIL(check_select_item_subquery(*select_stmt, *view_stmt, check_status))) { LOG_WARN("failed to check select item has subquery", K(ret)); } else if (!check_status) { can_be = false; } else if (view_stmt->is_contains_assignment() || select_stmt->is_contains_assignment()) { can_be = false; } else if (OB_FAIL(is_select_item_same(select_stmt, view_stmt, check_status, select_offset, const_select_items))) { LOG_WARN("check select item same failed", K(ret)); } else if (check_status && !view_stmt->is_recursive_union()) { can_be = true; } else if (view_stmt->is_scala_group_by() || view_stmt->has_distinct() || (view_stmt->is_recursive_union() && (!check_status || !select_offset.empty())) || (view_stmt->has_set_op() && !view_stmt->is_recursive_union())) { can_be = false; } else { can_be = true; } return ret; } int ObTransformQueryPushDown::check_select_item_subquery(ObSelectStmt &select_stmt, ObSelectStmt &view, bool &can_be) { int ret = OB_SUCCESS; can_be = true; ObSEArray column_exprs_from_subquery; ObRawExpr *expr = NULL; TableItem *table = NULL; ObSqlBitSet<> table_set; if (OB_UNLIKELY(1 != select_stmt.get_table_items().count()) || OB_ISNULL(table = select_stmt.get_table_item(0))) { ret = OB_ERR_UNEXPECTED; LOG_WARN("unexpect select stmt", K(ret), K(select_stmt.get_from_item_size()), K(table)); } for (int64_t i = 0; OB_SUCC(ret) && i < view.get_select_item_size(); ++i) { if (OB_ISNULL(expr = view.get_select_item(i).expr_)) { ret = OB_ERR_UNEXPECTED; LOG_WARN("unexpect null expr", K(ret)); } else if (!expr->has_flag(CNT_SUB_QUERY)) { /* do nothing */ } else if (OB_ISNULL(expr = select_stmt.get_column_expr_by_id(table->table_id_, i + OB_APP_MIN_COLUMN_ID))) { /* do nothing */ } else if (OB_FAIL(column_exprs_from_subquery.push_back(expr))) { LOG_WARN("failed to push back column expr", K(ret)); } } if (OB_FAIL(ret)) { } else if (column_exprs_from_subquery.empty()) { /* do nothing */ } else if (OB_FAIL(select_stmt.get_table_rel_ids(*table, table_set))) { LOG_WARN("failed to get rel ids", K(ret)); } else { ObIArray &subquery_exprs = select_stmt.get_subquery_exprs(); ObSEArray ignore_stmts; ObSEArray column_exprs; for (int64_t i = 0; OB_SUCC(ret) && can_be && i < subquery_exprs.count(); ++i) { column_exprs.reuse(); if(OB_FAIL(ObTransformUtils::extract_column_exprs(subquery_exprs.at(i), select_stmt.get_current_level(), table_set, ignore_stmts, column_exprs))) { LOG_WARN("extract column exprs failed", K(ret)); } else if (ObOptimizerUtil::overlap(column_exprs, column_exprs_from_subquery)) { can_be = false; } } } return ret; } int ObTransformQueryPushDown::check_where_condition_push_down( ObSelectStmt* select_stmt, ObSelectStmt* view_stmt, bool& transform_having, bool& can_be) { int ret = OB_SUCCESS; can_be = false; transform_having = false; if (OB_ISNULL(select_stmt) || OB_ISNULL(view_stmt)) { ret = OB_INVALID_ARGUMENT; LOG_WARN("stmt is NULL", K(view_stmt), K(ret)); } else if (view_stmt->has_set_op() || view_stmt->has_limit() || view_stmt->has_window_function() || view_stmt->is_contains_assignment()) { can_be = false; } else if (view_stmt->has_group_by() || select_stmt->has_rollup()) { bool is_invalid = false; for (int64_t i = 0; OB_SUCC(ret) && !is_invalid && i < select_stmt->get_condition_size(); ++i) { const ObRawExpr* cond_expr = NULL; if (OB_ISNULL(cond_expr = select_stmt->get_condition_expr(i))) { ret = OB_ERR_UNEXPECTED; LOG_WARN("invalid argument", K(ret)); } else if (cond_expr->has_flag(CNT_SUB_QUERY)) { is_invalid = true; can_be = false; } else { /*do nothing*/ } } if (OB_SUCC(ret) && !is_invalid) { transform_having = true; can_be = true; } } else { can_be = true; } return ret; } int ObTransformQueryPushDown::check_window_function_push_down(ObSelectStmt* view_stmt, bool& can_be) { int ret = OB_SUCCESS; can_be = false; if (OB_ISNULL(view_stmt)) { ret = OB_INVALID_ARGUMENT; LOG_WARN("stmt is NULL", K(view_stmt), K(ret)); } else if (view_stmt->has_window_function() || view_stmt->has_distinct() || view_stmt->has_set_op() || view_stmt->has_order_by() || view_stmt->has_limit() || view_stmt->is_contains_assignment()) { can_be = false; } else { can_be = true; } return ret; } int ObTransformQueryPushDown::check_distinct_push_down(ObSelectStmt* view_stmt, bool& need_distinct, bool& can_be) { int ret = OB_SUCCESS; can_be = false; if (OB_ISNULL(view_stmt)) { ret = OB_INVALID_ARGUMENT; LOG_WARN("stmt is NULL", K(view_stmt), K(ret)); } else if (view_stmt->has_limit()) { can_be = false; } else if (view_stmt->has_set_op()) { need_distinct = true; can_be = true; } else { can_be = true; } return ret; } int ObTransformQueryPushDown::do_transform(ObSelectStmt* select_stmt, ObSelectStmt* view_stmt, bool need_distinct, bool transform_having, const TableItem* view_table_item, ObIArray& select_offset, ObIArray& const_select_items) { int ret = OB_SUCCESS; if (OB_ISNULL(select_stmt) || OB_ISNULL(view_stmt) || OB_ISNULL(view_table_item) || OB_ISNULL(ctx_)) { ret = OB_INVALID_ARGUMENT; LOG_WARN("invalid argument", K(select_stmt), K(view_stmt), K(view_table_item), K(ctx_), K(ret)); } else if (OB_FAIL(replace_stmt_exprs(select_stmt, view_stmt, view_table_item->table_id_))) { LOG_WARN("failed to replace stmt exprs", K(ret)); } else if (OB_FAIL(push_down_stmt_exprs( select_stmt, view_stmt, need_distinct, transform_having, select_offset, const_select_items))) { LOG_WARN("push down stmt exprs failed", K(ret)); } else { /*do nothing*/ } return ret; } int ObTransformQueryPushDown::push_down_stmt_exprs(ObSelectStmt* select_stmt, ObSelectStmt* view_stmt, bool need_distinct, bool transform_having, ObIArray& select_offset, ObIArray& const_select_items) { int ret = OB_SUCCESS; if (OB_ISNULL(select_stmt) || OB_ISNULL(view_stmt)) { ret = OB_INVALID_ARGUMENT; LOG_WARN("invalid argument", K(select_stmt), K(view_stmt), K(ret)); } else if (!transform_having && OB_FAIL(append(view_stmt->get_condition_exprs(), select_stmt->get_condition_exprs()))) { LOG_WARN("append select_stmt condition exprs to view stmt failed", K(ret)); } else if (transform_having && OB_FAIL(append(view_stmt->get_having_exprs(), select_stmt->get_condition_exprs()))) { LOG_WARN("append select_stmt condition exprs to view stmt having expr failed", K(ret)); } else if (OB_FAIL(append(view_stmt->get_group_exprs(), select_stmt->get_group_exprs()))) { LOG_WARN("append select_stmt window func exprs to view stmt window func exprs failed", K(ret)); } else if (OB_FAIL(append(view_stmt->get_rollup_exprs(), select_stmt->get_rollup_exprs()))) { LOG_WARN("append select_stmt rollup exprs failed", K(ret)); } else if (OB_FAIL(append(view_stmt->get_rollup_dirs(), select_stmt->get_rollup_dirs()))) { LOG_WARN("append select_stmt rollup directions failed", K(ret)); } else if (OB_FAIL(append(view_stmt->get_aggr_items(), select_stmt->get_aggr_items()))) { LOG_WARN("append aggr items failed", K(ret)); } else if (OB_FAIL(append(view_stmt->get_having_exprs(), select_stmt->get_having_exprs()))) { LOG_WARN("append select_stmt having exprs to view stmt having exprs failed", K(ret)); } else if (OB_FAIL(append(view_stmt->get_window_func_exprs(), select_stmt->get_window_func_exprs()))) { LOG_WARN("append select_stmt window func exprs to view stmt window func exprs failed", K(ret)); } else { if (select_stmt->has_rollup()) { view_stmt->assign_rollup(); } if (!view_stmt->is_from_pivot()) { view_stmt->set_from_pivot(select_stmt->is_from_pivot()); } if (need_distinct && !view_stmt->is_set_distinct()) { view_stmt->assign_set_distinct(); } else if (select_stmt->has_distinct() && !view_stmt->has_distinct()) { view_stmt->assign_distinct(); } if (select_stmt->has_select_into()) { view_stmt->set_select_into(select_stmt->get_select_into()); } if (select_stmt->has_order_by()) { view_stmt->get_order_items().reset(); if (OB_FAIL(append(view_stmt->get_order_items(), select_stmt->get_order_items()))) { LOG_WARN("append select_stmt order items to view stmt order items failed", K(ret)); } } if (OB_FAIL(ret)) { } else if (select_stmt->has_limit() && OB_FAIL(do_limit_merge(*select_stmt, *view_stmt))) { LOG_WARN("failed to merge limit", K(ret)); } else if (select_stmt->has_sequence() && OB_FAIL(append(view_stmt->get_nextval_sequence_ids(), select_stmt->get_nextval_sequence_ids()))) { LOG_WARN("failed to append nextval sequence ids", K(ret)); } else if (select_stmt->has_sequence() && OB_FAIL(append(view_stmt->get_currval_sequence_ids(), select_stmt->get_currval_sequence_ids()))) { LOG_WARN("failed to append currval sequence ids", K(ret)); } else if (view_stmt->has_set_op()) { if (select_offset.empty()) { /*do nothing*/ } else if (OB_FAIL(recursive_adjust_select_item(view_stmt, select_offset, const_select_items))) { LOG_WARN("recursive adjust select item location failed", K(ret)); } else { /*do nothing*/ } } else { view_stmt->get_select_items().reset(); if (OB_FAIL(append(view_stmt->get_select_items(), select_stmt->get_select_items()))) { LOG_WARN("view stmt replace select items failed", K(ret)); } } if (OB_FAIL(ret)) { } else if (OB_FAIL(append(view_stmt->get_subquery_exprs(), select_stmt->get_subquery_exprs()))) { LOG_WARN("view stmt append subquery failed", K(ret)); } else if (OB_FAIL(adjust_stmt_hints(select_stmt, view_stmt))) { LOG_WARN("failed to adjust hints", K(ret)); } else if (OB_FAIL(view_stmt->adjust_subquery_list())) { LOG_WARN("failed to adjust subquery list", K(ret)); } else if (OB_FAIL(view_stmt->adjust_subquery_stmt_parent(select_stmt, view_stmt))) { LOG_WARN("failed to adjust subquery stmt parent", K(ret)); } else { view_stmt->set_current_level(select_stmt->get_current_level()); view_stmt->set_parent_namespace_stmt(select_stmt->get_parent_namespace_stmt()); // add all the view store info const ObDMLStmt::ObViewTableIds& view_ids = select_stmt->get_view_table_id_store(); for (int64_t i = 0; OB_SUCC(ret) && i < view_ids.count(); i++) { if (OB_FAIL(view_stmt->add_view_table_id(view_ids.at(i)))) { LOG_WARN("fail to add view table id", K(ret)); } } if (OB_SUCC(ret)) { bool is_from_show_stmt = select_stmt->is_from_show_stmt(); stmt::StmtType literal_stmt_type = select_stmt->get_literal_stmt_type(); view_stmt->set_literal_stmt_type(literal_stmt_type); view_stmt->set_is_from_show_stmt(is_from_show_stmt); } } } return ret; } int ObTransformQueryPushDown::adjust_stmt_hints(ObSelectStmt* select_stmt, ObSelectStmt* view_stmt) { int ret = OB_SUCCESS; if (OB_ISNULL(select_stmt) || OB_ISNULL(view_stmt)) { ret = OB_INVALID_ARGUMENT; LOG_WARN("invalid argument", K(select_stmt), K(view_stmt), K(ret)); } else if (select_stmt->has_group_by()) { view_stmt->get_stmt_hint().aggregate_ = select_stmt->get_stmt_hint().aggregate_; view_stmt->get_stmt_hint().use_place_groupby_ = select_stmt->get_stmt_hint().use_place_groupby_; if (OB_FAIL(view_stmt->get_stmt_hint().place_groupby_.assign(select_stmt->get_stmt_hint().place_groupby_))) { LOG_WARN("place groupby expand array assign failed.", K(ret)); } else if (OB_FAIL(view_stmt->get_stmt_hint().no_place_groupby_.assign( select_stmt->get_stmt_hint().no_place_groupby_))) { LOG_WARN("no place groupby expand array assign failed.", K(ret)); } else { /*do nothing*/ } } if (OB_SUCC(ret) && select_stmt->get_condition_size() > 0) { view_stmt->get_stmt_hint().use_expand_ = select_stmt->get_stmt_hint().use_expand_; if (OB_FAIL(view_stmt->get_stmt_hint().no_expand_.assign(select_stmt->get_stmt_hint().no_expand_))) { LOG_WARN("no expand array assign failed.", K(ret)); } else if (OB_FAIL(view_stmt->get_stmt_hint().use_concat_.assign(select_stmt->get_stmt_hint().use_concat_))) { LOG_WARN("use concat array assign failed.", K(ret)); } else { /*do nothing*/ } } if (OB_SUCC(ret)) { view_stmt->get_stmt_hint().use_unnest_ = select_stmt->get_stmt_hint().use_unnest_; if (OB_FAIL(view_stmt->get_stmt_hint().no_unnest_.assign(select_stmt->get_stmt_hint().no_unnest_))) { LOG_WARN("no unnest array assign failed.", K(ret)); } else if (OB_FAIL(view_stmt->get_stmt_hint().unnest_.assign(select_stmt->get_stmt_hint().unnest_))) { LOG_WARN("unnest array assign failed.", K(ret)); } else { /*do nothing*/ } } if (OB_SUCC(ret)) { view_stmt->get_stmt_hint().use_view_merge_ = select_stmt->get_stmt_hint().use_view_merge_; if (OB_FAIL(view_stmt->get_stmt_hint().no_v_merge_.assign(select_stmt->get_stmt_hint().no_v_merge_))) { LOG_WARN("no unnest array assign failed.", K(ret)); } else if (OB_FAIL(view_stmt->get_stmt_hint().v_merge_.assign(select_stmt->get_stmt_hint().v_merge_))) { LOG_WARN("unnest array assign failed.", K(ret)); } else { /*do nothing*/ } } return ret; } int ObTransformQueryPushDown::replace_stmt_exprs(ObDMLStmt* parent_stmt, ObSelectStmt* child_stmt, uint64_t table_id) { int ret = OB_SUCCESS; ObSEArray old_column_exprs; ObSEArray new_column_exprs; ObSEArray temp_exprs; if (OB_ISNULL(parent_stmt) || OB_ISNULL(child_stmt)) { ret = OB_ERR_UNEXPECTED; LOG_WARN("get unexpected null", K(ret)); } else if (OB_FAIL(parent_stmt->get_column_exprs(table_id, temp_exprs))) { LOG_WARN("failed to get column exprs", K(ret)); } else if (OB_FAIL(append(old_column_exprs, temp_exprs))) { LOG_WARN("failed to append exprs", K(ret)); } else if (OB_FAIL(ObTransformUtils::convert_column_expr_to_select_expr( old_column_exprs, *child_stmt, new_column_exprs))) { LOG_WARN("failed to convert column expr to select expr", K(ret)); } else { ret = parent_stmt->replace_inner_stmt_expr(old_column_exprs, new_column_exprs); } return ret; } int ObTransformQueryPushDown::recursive_adjust_select_item( ObSelectStmt* select_stmt, ObIArray& select_offset, ObIArray& const_select_items) { int ret = OB_SUCCESS; bool is_stack_overflow = false; if (OB_ISNULL(select_stmt) || OB_ISNULL(ctx_) || OB_ISNULL(ctx_->expr_factory_)) { ret = OB_INVALID_ARGUMENT; LOG_WARN("stmt is NULL", K(select_stmt), K(ctx_), K(ctx_->expr_factory_), K(ret)); } else if (OB_FAIL(check_stack_overflow(is_stack_overflow))) { LOG_WARN("check stack overflow failed", K(is_stack_overflow), K(ret)); } else if (is_stack_overflow) { ret = OB_SIZE_OVERFLOW; LOG_WARN("too deep recursive", K(is_stack_overflow), K(ret)); } else if (select_stmt->has_set_op()) { ObIArray& child_stmts = static_cast(select_stmt)->get_set_query(); for (int64_t i = 0; OB_SUCC(ret) && i < child_stmts.count(); ++i) { ret = SMART_CALL(recursive_adjust_select_item(child_stmts.at(i), select_offset, const_select_items)); } if (OB_SUCC(ret)) { if (OB_FAIL(reset_set_stmt_select_list(select_stmt, select_offset))) { LOG_WARN("failed to reset set stmt select list", K(ret)); } } } else { ObSEArray new_select_item; ObSEArray old_select_item; ObSEArray new_const_select_items; if (OB_FAIL(old_select_item.assign(select_stmt->get_select_items()))) { LOG_WARN("failed to assign a new select item", K(ret)); } else if (OB_FAIL(ObTransformUtils::deep_copy_select_items( *ctx_->expr_factory_, const_select_items, new_const_select_items, COPY_REF_DEFAULT))) { LOG_WARN("deep copy select items failed", K(ret)); } else { int64_t k = 0; for (int64_t i = 0; OB_SUCC(ret) && i < select_offset.count(); ++i) { if (select_offset.at(i) == -1) { //-1 meanings upper stmt has const select item if (OB_UNLIKELY(k >= new_const_select_items.count())) { ret = OB_ERR_UNEXPECTED; LOG_WARN("get unexpected error", K(ret), K(k), K(new_const_select_items.count())); } else if (OB_FAIL(new_select_item.push_back(new_const_select_items.at(k)))) { LOG_WARN("push back select item error", K(ret)); } else { ++k; } } else if (OB_UNLIKELY(select_offset.at(i) < 0 || select_offset.at(i) >= old_select_item.count())) { ret = OB_ERR_UNEXPECTED; LOG_WARN("get unexpected error", K(select_offset.at(i)), K(old_select_item.count()), K(ret)); } else if (OB_FAIL(new_select_item.push_back(old_select_item.at(select_offset.at(i))))) { LOG_WARN("push back select item error", K(ret)); } else { /*do nothing*/ } } if (OB_SUCC(ret)) { select_stmt->get_select_items().reset(); if (OB_FAIL(append(select_stmt->get_select_items(), new_select_item))) { LOG_WARN("view stmt replace select items failed", K(ret)); } } } } return ret; } int ObTransformQueryPushDown::reset_set_stmt_select_list(ObSelectStmt* select_stmt, ObIArray& select_offset) { int ret = OB_SUCCESS; ObSEArray old_select_exprs; ObSEArray adjust_old_select_exprs; ObSEArray new_select_exprs; ObSEArray adjust_new_select_exprs; if (OB_ISNULL(select_stmt) || OB_ISNULL(ctx_)) { ret = OB_INVALID_ARGUMENT; LOG_WARN("stmt is NULL", K(select_stmt), K(ctx_), K(ret)); } else if (!select_stmt->is_set_stmt()) { /*do nothing*/ } else if (OB_FAIL(select_stmt->get_select_exprs(old_select_exprs))) { LOG_WARN("failed to get select exprs", K(ret)); } else { select_stmt->get_select_items().reset(); if (OB_FAIL(ObOptimizerUtil::gen_set_target_list( ctx_->allocator_, ctx_->session_info_, ctx_->expr_factory_, select_stmt))) { LOG_WARN("failed to create select list for union", K(ret)); } else if (OB_FAIL(select_stmt->get_select_exprs(new_select_exprs))) { LOG_WARN("failed to get select exprs", K(ret)); } else { for (int64_t i = 0; OB_SUCC(ret) && i < select_offset.count(); ++i) { if (select_offset.at(i) == -1) { //-1 meanings upper stmt has const select item /*do nothing */ } else if (OB_UNLIKELY(select_offset.at(i) < 0 || select_offset.at(i) >= old_select_exprs.count() || i >= new_select_exprs.count())) { ret = OB_ERR_UNEXPECTED; LOG_WARN("get unexpected error", K(select_offset.at(i)), K(old_select_exprs.count()), K(new_select_exprs.count()), K(i), K(ret)); } else if (OB_FAIL(adjust_old_select_exprs.push_back(old_select_exprs.at(select_offset.at(i)))) || OB_FAIL(adjust_new_select_exprs.push_back(new_select_exprs.at(i)))) { LOG_WARN("failed to push back expr", K(ret)); } else { /*do nothing*/ } } if (OB_SUCC(ret) && adjust_old_select_exprs.count() > 0) { if (OB_FAIL(select_stmt->replace_inner_stmt_expr(adjust_old_select_exprs, adjust_new_select_exprs))) { LOG_WARN("failed to replace inner stmt expr", K(ret)); } else { /*do nothing*/ } } } } return ret; } int ObTransformQueryPushDown::check_set_op_is_const_expr(ObSelectStmt* select_stmt, ObRawExpr* expr, bool& is_const) { int ret = OB_SUCCESS; if (OB_ISNULL(expr) || OB_ISNULL(select_stmt) || OB_UNLIKELY(!expr->is_set_op_expr() || !select_stmt->is_set_stmt() || select_stmt->get_set_query().empty())) { ret = OB_ERR_UNEXPECTED; LOG_WARN("get unexpected null", K(ret), K(expr), K(select_stmt)); } else if (!is_const) { /*do nothing*/ } else if (select_stmt->is_set_distinct()) { is_const = false; } else { int64_t idx = static_cast(expr)->get_idx(); for (int64_t i = 0; OB_SUCC(ret) && is_const && i < select_stmt->get_set_query().count(); ++i) { ObRawExpr* select_expr = NULL; if (OB_ISNULL(select_stmt->get_set_query(i)) || OB_UNLIKELY(idx < 0 || idx >= select_stmt->get_set_query(i)->get_select_item_size()) || OB_ISNULL(select_expr = select_stmt->get_set_query(i)->get_select_item(idx).expr_)) { ret = OB_ERR_UNEXPECTED; LOG_WARN("get unexpected null", K(i), K(select_stmt->get_set_query(i)), K(idx), K(select_expr), K(ret)); } else if (select_expr->is_set_op_expr()) { if (OB_FAIL(SMART_CALL(check_set_op_is_const_expr(select_stmt->get_set_query(i), select_expr, is_const)))) { LOG_WARN("failed to check set op is const expr", K(ret)); } } else { is_const &= select_expr->has_const_or_const_expr_flag(); } } } return ret; } } // namespace sql } // namespace oceanbase