/** * 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_EXE #include "sql/executor/ob_direct_receive_op.h" #include "sql/engine/ob_exec_context.h" #include "sql/engine/ob_physical_plan_ctx.h" #include "sql/executor/ob_task_executor_ctx.h" #include "sql/executor/ob_executor_rpc_impl.h" #include "share/ob_scanner.h" #include "sql/session/ob_sql_session_info.h" #include "sql/monitor/ob_exec_stat_collector.h" using namespace oceanbase::common; namespace oceanbase { namespace sql { OB_SERIALIZE_MEMBER((ObDirectReceiveSpec, ObOpSpec)); ObDirectReceiveOp::ObDirectReceiveOp(ObExecContext& exec_ctx, const ObOpSpec& spec, ObOpInput* input) : ObReceiveOp(exec_ctx, spec, input), scanner_(NULL), scanner_iter_(), all_data_empty_(false), cur_data_empty_(true), first_request_received_(false), found_rows_(0) {} int ObDirectReceiveOp::inner_open() { int ret = OB_SUCCESS; all_data_empty_ = false; /* Whether all the scanner data has been read */ cur_data_empty_ = true; /* Whether the current scanner data has been read */ first_request_received_ = false; /* Whether the plan has been sent to the remote server */ // Receive the first scanner. For DML such as insert and update, need to get affected_rows etc. if (OB_FAIL(setup_next_scanner())) { if (OB_UNLIKELY(OB_ITER_END != ret)) { LOG_WARN("failed to setup first scanner", K(ret)); } else { all_data_empty_ = true; /* no more scanner */ } } else { cur_data_empty_ = false; /* scanner is filled up again, can continue reading */ } return ret; } /* * state:cur_data_empty_, all_data_empty_ */ int ObDirectReceiveOp::inner_get_next_row() { int ret = OB_SUCCESS; bool has_got_a_row = false; RemoteExecuteStreamHandle* resp_handler = NULL; // Sometimes we need send user variables to remote end to complete query successfully // And, of course, we will get the new value for user variable. // Scanner contains the updated values. // so we update the user variables in terms of scanners here. if (OB_FAIL(ObTaskExecutorCtxUtil::get_stream_handler(ctx_, resp_handler))) { LOG_WARN("fail get task response handler", K(ret)); } else if (OB_ISNULL(resp_handler)) { ret = OB_ERR_UNEXPECTED; LOG_WARN("resp_handler is NULL", K(ret)); } else if (OB_FAIL(THIS_WORKER.check_status())) { LOG_WARN("check physical plan status failed", K(ret)); } else if (OB_ERR_TASK_SKIPPED == resp_handler->get_result_code()) { // skip ret = OB_ITER_END; LOG_WARN("this remote task is skipped", K(ret)); } /* It will be easier to understand the following code with state machine thinking */ while (OB_SUCC(ret) && false == has_got_a_row) { if (all_data_empty_) { /* All data has been read */ ret = OB_ITER_END; } else if (cur_data_empty_) { /* The current scanner has been read over */ /* send RPC req and remote server response Scanner */ if (OB_FAIL(setup_next_scanner())) { if (OB_UNLIKELY(OB_ITER_END != ret)) { LOG_WARN("fail to setup next scanner", K(ret)); } else { all_data_empty_ = true; /* no more scanner */ } } else { cur_data_empty_ = false; /* scanner is filled up again, can continue reading */ } } else { /* current scanner is readable */ if (OB_FAIL(get_next_row_from_cur_scanner())) { if (OB_UNLIKELY(OB_ITER_END != ret)) { LOG_WARN("fail to get next row from cur scanner", K(ret)); } else { // curr scanner read over cur_data_empty_ = true; ret = OB_SUCCESS; // set ret be OB_SUCCESS for loop } } else { // get one row then break loop has_got_a_row = true; } } } if (OB_ITER_END == ret) { ObPhysicalPlanCtx* plan_ctx = GET_PHY_PLAN_CTX(ctx_); plan_ctx->set_found_rows(found_rows_); } return ret; } int ObDirectReceiveOp::inner_close() { int ret = OB_SUCCESS; RemoteExecuteStreamHandle* resp_handler = NULL; if (OB_FAIL(ObTaskExecutorCtxUtil::get_stream_handler(ctx_, resp_handler))) { LOG_WARN("fail get task response handler", K(ret)); } else if (OB_ISNULL(resp_handler)) { ret = OB_ERR_UNEXPECTED; LOG_WARN("resp_handler is NULL", K(ret)); } else { if (resp_handler->has_more()) { if (OB_FAIL(resp_handler->abort())) { LOG_WARN("fail to abort", K(ret)); } else { ObSQLSessionInfo* session = ctx_.get_my_session(); ObPhysicalPlanCtx* plan_ctx = ctx_.get_physical_plan_ctx(); ObExecutorRpcImpl* rpc = NULL; if (OB_FAIL(ObTaskExecutorCtxUtil::get_task_executor_rpc(ctx_, rpc))) { LOG_WARN("get task executor rpc failed", K(ret)); } else if (OB_ISNULL(session) || OB_ISNULL(plan_ctx) || OB_ISNULL(rpc)) { ret = OB_ERR_UNEXPECTED; LOG_WARN("session or plan ctx or rpc is NULL", K(ret)); } else { ObQueryRetryInfo retry_info; ObExecutorRpcCtx rpc_ctx(session->get_effective_tenant_id(), plan_ctx->get_timeout_timestamp(), ctx_.get_task_exec_ctx().get_min_cluster_version(), &retry_info, session, plan_ctx->is_plain_select_stmt()); int tmp_ret = rpc->task_kill(rpc_ctx, resp_handler->get_task_id(), resp_handler->get_dst_addr()); if (OB_SUCCESS != tmp_ret) { LOG_WARN("kill task failed", K(tmp_ret), K(resp_handler->get_task_id()), K(resp_handler->get_dst_addr())); } } } } else { } } return ret; } int ObDirectReceiveOp::setup_next_scanner() { int ret = OB_SUCCESS; ObPhysicalPlanCtx* plan_ctx = GET_PHY_PLAN_CTX(ctx_); RemoteExecuteStreamHandle* resp_handler = NULL; ObSQLSessionInfo* my_session = GET_MY_SESSION(ctx_); if (OB_FAIL(ObTaskExecutorCtxUtil::get_stream_handler(ctx_, resp_handler))) { LOG_WARN("fail get task response handler", K(ret)); } else if (OB_ISNULL(resp_handler)) { ret = OB_ERR_UNEXPECTED; LOG_WARN("resp_handler is NULL", K(ret)); } if (OB_SUCC(ret)) { // Read the data the first time, and the result has been obtained when task_submit() is called by the Scheduler if (!first_request_received_) { ObScanner* scanner = resp_handler->get_result(); if (OB_ISNULL(scanner)) { ret = OB_ERR_UNEXPECTED; } else { // set last_insert_id no matter success or fail plan_ctx->set_last_insert_id_to_client(scanner->get_last_insert_id_to_client()); plan_ctx->set_last_insert_id_session(scanner->get_last_insert_id_session()); plan_ctx->set_last_insert_id_changed(scanner->get_last_insert_id_changed()); int tmp_ret = OB_SUCCESS; ObExecStatCollector& collector = ctx_.get_exec_stat_collector(); if (OB_SUCCESS != (tmp_ret = collector.add_raw_stat(scanner->get_extend_info()))) { LOG_WARN("fail to collected raw extend info in scanner", K(tmp_ret)); } if (OB_FAIL(scanner->get_err_code())) { int add_ret = OB_SUCCESS; const char* err_msg = scanner->get_err_msg(); // after FORWARD_USER_ERROR(ret, err_msg),if err_msg length > 0, // then return err_msg // Otherwise, the err_msg returned is the default error message corresponding to ret. // use FORWARD_USER_ERROR(ret, err_msg) can qualify. FORWARD_USER_ERROR(ret, err_msg); LOG_WARN("while fetching first scanner, the remote rcode is not OB_SUCCESS", K(ret), K(err_msg), "dst_addr", to_cstring(resp_handler->get_dst_addr())); if (is_data_not_readable_err(ret)) { ObQueryRetryInfo& retry_info = my_session->get_retry_info_for_update(); if (OB_UNLIKELY(OB_SUCCESS != (add_ret = retry_info.add_invalid_server_distinctly(resp_handler->get_dst_addr(), true)))) { LOG_WARN("fail to add remote addr to invalid servers distinctly", K(ret), K(add_ret), K(resp_handler->get_dst_addr()), K(retry_info)); } } } else { scanner_ = scanner; first_request_received_ = true; // INSERT,UPDATE,DELETE,The Scanner returned for the first time contains the affected row plan_ctx->set_affected_rows(scanner->get_affected_rows()); found_rows_ += scanner->get_found_rows(); if (OB_FAIL(scanner->get_datum_store().begin(scanner_iter_))) { LOG_WARN("fail to init datum store iter", K(ret)); } else if (OB_FAIL(plan_ctx->set_row_matched_count(scanner->get_row_matched_count()))) { LOG_WARN("fail to set row matched count", K(ret), K(scanner->get_row_matched_count())); } else if (OB_FAIL(plan_ctx->set_row_duplicated_count(scanner->get_row_duplicated_count()))) { LOG_WARN("fail to set row duplicate count", K(ret), K(scanner->get_row_duplicated_count())); /** * ObRemoteTaskExecutor::execute() has called merge_result() before here, that is a * better place to call merge_result(), especially when any operation failed between * there and here. */ } else if (OB_FAIL(plan_ctx->merge_implicit_cursors(scanner->get_implicit_cursors()))) { LOG_WARN("merge implicit cursors failed", K(ret), K(scanner->get_implicit_cursors())); } } } } else { /* Subsequent request, send SESSION_NEXT to the remote end through Handle */ ObScanner* result_scanner = NULL; if (resp_handler->has_more()) { if (OB_FAIL(resp_handler->reset_and_init_result())) { LOG_WARN("fail reset and init result", K(ret)); } else if (OB_ISNULL(result_scanner = resp_handler->get_result())) { ret = OB_ERR_UNEXPECTED; LOG_WARN("succ to alloc result, but result scanner is NULL", K(ret)); } else if (OB_FAIL(resp_handler->get_more(*result_scanner))) { LOG_WARN("fail wait response", K(ret), "dst_addr", to_cstring(resp_handler->get_dst_addr())); } else if (OB_FAIL(result_scanner->get_err_code())) { int add_ret = OB_SUCCESS; const char* err_msg = result_scanner->get_err_msg(); FORWARD_USER_ERROR(ret, err_msg); LOG_WARN("while getting more scanner, the remote rcode is not OB_SUCCESS", K(ret), K(err_msg), "dst_addr", to_cstring(resp_handler->get_dst_addr())); if (is_data_not_readable_err(ret)) { // Add the remote observer to the invalid servers of retry info ObQueryRetryInfo& retry_info = my_session->get_retry_info_for_update(); if (OB_UNLIKELY(OB_SUCCESS != (add_ret = retry_info.add_invalid_server_distinctly(resp_handler->get_dst_addr(), true)))) { LOG_WARN("fail to add remote addr to invalid servers distinctly", K(ret), K(add_ret), K(resp_handler->get_dst_addr()), K(retry_info)); } } } else { scanner_ = result_scanner; found_rows_ += scanner_->get_found_rows(); if (OB_FAIL(scanner_->get_datum_store().begin(scanner_iter_))) { LOG_WARN("fail to init datum store iter", K(ret)); } } } else { ret = OB_ITER_END; // only successful select affect last_insert_id // for select, last_insert_id may changed because last_insert_id(#) called // last_insert_id values should be the last row calling last_insert_id(#) plan_ctx->set_last_insert_id_session(scanner_->get_last_insert_id_session()); plan_ctx->set_last_insert_id_changed(scanner_->get_last_insert_id_changed()); int tmp_ret = OB_SUCCESS; ObExecStatCollector& collector = ctx_.get_exec_stat_collector(); if (OB_SUCCESS != (tmp_ret = collector.add_raw_stat(scanner_->get_extend_info()))) { LOG_WARN("fail to collected raw extend info in scanner", K(tmp_ret)); } if (OB_SUCCESS != (tmp_ret = plan_ctx->get_table_row_count_list().assign(scanner_->get_table_row_counts()))) { LOG_WARN("fail to set table row count", K(ret), K(scanner_->get_table_row_counts())); } LOG_DEBUG( "remote table row counts", K(scanner_->get_table_row_counts()), K(plan_ctx->get_table_row_count_list())); } } } if (OB_SUCC(ret)) { if (OB_FAIL(my_session->replace_user_variables(scanner_->get_session_var_map()))) { LOG_WARN("replace user variables failed", K(ret)); } } return ret; } int ObDirectReceiveOp::get_next_row_from_cur_scanner() { int ret = OB_SUCCESS; if (OB_FAIL(scanner_iter_.get_next_row(MY_SPEC.output_, eval_ctx_))) { if (OB_UNLIKELY(OB_ITER_END != ret)) { LOG_WARN("fail get next row", K(ret)); } else { } } else { LOG_DEBUG("direct receive next row", "row", ROWEXPR2STR(eval_ctx_, MY_SPEC.output_)); } return ret; } int ObDirectReceiveOp::rescan() { // not support the rescan operation of the remote operator int ret = OB_NOT_SUPPORTED; LOG_USER_ERROR(OB_NOT_SUPPORTED, "Distributed rescan"); return ret; } } // namespace sql } // namespace oceanbase