Files
oceanbase/src/sql/rewrite/ob_transform_query_push_down.cpp
2021-08-19 17:53:21 +08:00

819 lines
35 KiB
C++

/**
* 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<ObParentDMLStmt>& 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<int64_t, 4> select_offset; // record select item offset postition, used to adjust set-op stmt output
// postition
ObSEArray<SelectItem, 4> 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<ObSelectStmt*>(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<int64_t>& select_offset,
ObIArray<SelectItem>& 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<int64_t>& select_offset, ObIArray<SelectItem>& 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<const ObColumnRefRawExpr*>(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<const ObColumnRefRawExpr*>(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<const ObColumnRefRawExpr*>(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<int64_t>& select_offset, ObIArray<SelectItem>& 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<ObRawExpr*, 4> 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<ObQueryRefRawExpr*> &subquery_exprs = select_stmt.get_subquery_exprs();
ObSEArray<ObDMLStmt *, 4> ignore_stmts;
ObSEArray<ObRawExpr*, 4> 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<int64_t>& select_offset,
ObIArray<SelectItem>& 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<int64_t>& select_offset,
ObIArray<SelectItem>& 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_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<ObRawExpr*, 16> old_column_exprs;
ObSEArray<ObRawExpr*, 16> new_column_exprs;
ObSEArray<ObColumnRefRawExpr*, 16> 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<int64_t>& select_offset, ObIArray<SelectItem>& 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<ObSelectStmt*>& child_stmts = static_cast<ObSelectStmt*>(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<SelectItem, 4> new_select_item;
ObSEArray<SelectItem, 4> old_select_item;
ObSEArray<SelectItem, 4> 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<int64_t>& select_offset)
{
int ret = OB_SUCCESS;
ObSEArray<ObRawExpr*, 4> old_select_exprs;
ObSEArray<ObRawExpr*, 4> adjust_old_select_exprs;
ObSEArray<ObRawExpr*, 4> new_select_exprs;
ObSEArray<ObRawExpr*, 4> 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<ObSetOpRawExpr*>(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