/** * 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. */ #include "ob_all_virtual_sys_event.h" #include "share/ob_tenant_mgr.h" using namespace oceanbase::common; namespace oceanbase { namespace observer { ObAllVirtualSysEvent::ObAllVirtualSysEvent() : ObVirtualTableScannerIterator(), ObMultiTenantOperator(), addr_(NULL), ipstr_(), port_(0), event_iter_(0), tenant_id_(OB_INVALID_ID), diag_info_(allocator_) { } ObAllVirtualSysEvent::~ObAllVirtualSysEvent() { reset(); } void ObAllVirtualSysEvent::reset() { omt::ObMultiTenantOperator::reset(); ObVirtualTableScannerIterator::reset(); addr_ = NULL; port_ = 0; ipstr_.reset(); event_iter_ = 0; tenant_id_ = OB_INVALID_ID; diag_info_.reset(); } int ObAllVirtualSysEvent::set_ip(common::ObAddr *addr) { int ret = OB_SUCCESS; char ipbuf[common::OB_IP_STR_BUFF]; if (NULL == addr){ ret = OB_ENTRY_NOT_EXIST; } else if (!addr_->ip_to_string(ipbuf, sizeof(ipbuf))) { SERVER_LOG(ERROR, "ip to string failed"); ret = OB_ERR_UNEXPECTED; } else { ipstr_ = ObString::make_string(ipbuf); if (OB_FAIL(ob_write_string(*allocator_, ipstr_, ipstr_))) { SERVER_LOG(WARN, "failed to write string", K(ret)); } port_ = addr_->get_port(); } return ret; } int ObAllVirtualSysEvent::get_the_diag_info( const uint64_t tenant_id, common::ObDiagnoseTenantInfo &diag_info) { int ret = OB_SUCCESS; diag_info_.reset(); if (OB_FAIL(common::ObDIGlobalTenantCache::get_instance().get_the_diag_info(tenant_id, diag_info))) { if (OB_ENTRY_NOT_EXIST == ret) { ret = OB_SUCCESS; } else { SERVER_LOG(WARN, "Fail to get tenant sys event", KR(ret), K(tenant_id)); } } return ret; } void ObAllVirtualSysEvent::release_last_tenant() { diag_info_.reset(); tenant_id_ = OB_INVALID_TENANT_ID; } bool ObAllVirtualSysEvent::is_need_process(uint64_t tenant_id) { return (is_sys_tenant(effective_tenant_id_) || tenant_id == effective_tenant_id_); } int ObAllVirtualSysEvent::inner_get_next_row(ObNewRow *&row) { int ret = OB_SUCCESS; if (OB_FAIL(execute(row))) { SERVER_LOG(WARN, "execute fail", K(ret)); } return ret; } int ObAllVirtualSysEvent::process_curr_tenant(ObNewRow *&row) { int ret = OB_SUCCESS; ObObj *cells = cur_row_.cells_; const int64_t col_count = output_column_ids_.count(); if (OB_UNLIKELY(NULL == allocator_)) { ret = OB_NOT_INIT; SERVER_LOG(WARN, "allocator is NULL", K(ret)); } else { if (MTL_ID() != tenant_id_) { tenant_id_ = MTL_ID(); // new tenant, init diag info if (OB_FAIL(set_ip(addr_))){ SERVER_LOG(WARN, "can't get ip", K(ret)); } else if (OB_FAIL(get_the_diag_info(tenant_id_, diag_info_))) { SERVER_LOG(WARN, "get diag info fail", K(ret), K(tenant_id_)); } else { event_iter_ = 0; } } if (event_iter_ >= WAIT_EVENTS_TOTAL) { ret = OB_ITER_END; } if (OB_SUCC(ret)) { uint64_t cell_idx = 0; double value = 0; for (int64_t i = 0; OB_SUCC(ret) && i < col_count; ++i) { uint64_t col_id = output_column_ids_.at(i); switch(col_id) { case TENANT_ID: { cells[cell_idx].set_int(tenant_id_); break; } case SVR_IP: { cells[cell_idx].set_varchar(ipstr_); cells[cell_idx].set_collation_type(ObCharset::get_default_collation(ObCharset::get_default_charset())); break; } case SVR_PORT: { cells[cell_idx].set_int(port_); break; } case EVENT_ID: { cells[cell_idx].set_int(OB_WAIT_EVENTS[event_iter_].event_id_); break; } case EVENT: { cells[cell_idx].set_varchar(OB_WAIT_EVENTS[event_iter_].event_name_); cells[cell_idx].set_collation_type(ObCharset::get_default_collation(ObCharset::get_default_charset())); break; } case WAIT_CLASS_ID: { cells[cell_idx].set_int(OB_WAIT_CLASSES[OB_WAIT_EVENTS[event_iter_].wait_class_].wait_class_id_); break; } case WAIT_CLASS_NO: { cells[cell_idx].set_int(OB_WAIT_EVENTS[event_iter_].wait_class_); break; } case WAIT_CLASS: { cells[cell_idx].set_varchar(OB_WAIT_CLASSES[OB_WAIT_EVENTS[event_iter_].wait_class_].wait_class_); cells[cell_idx].set_collation_type(ObCharset::get_default_collation(ObCharset::get_default_charset())); break; } case TOTAL_WAITS: { cells[cell_idx].set_int(diag_info_.get_event_stats().get(event_iter_)->total_waits_); break; } case TOTAL_TIMEOUTS: { cells[cell_idx].set_int(diag_info_.get_event_stats().get(event_iter_)->total_timeouts_); break; } case TIME_WAITED: { value = static_cast(diag_info_.get_event_stats().get(event_iter_)->time_waited_) / 10000; cells[cell_idx].set_double(value); break; } case MAX_WAIT: { value = static_cast(diag_info_.get_event_stats().get(event_iter_)->max_wait_) / 10000; cells[cell_idx].set_double(value); break; } case AVERAGE_WAIT: { if (0 != diag_info_.get_event_stats().get(event_iter_)->total_waits_) { value = (static_cast(diag_info_.get_event_stats().get(event_iter_)->time_waited_) / static_cast(diag_info_.get_event_stats().get(event_iter_)->total_waits_)) / 10000; cells[cell_idx].set_double(value); } else { cells[cell_idx].set_double(0); } break; } case TIME_WAITED_MICRO: { cells[cell_idx].set_int(diag_info_.get_event_stats().get(event_iter_)->time_waited_); break; } default: { ret = OB_ERR_UNEXPECTED; SERVER_LOG(WARN, "invalid column id", K(ret), K(cell_idx), K(output_column_ids_), K(col_id)); break; } } if (OB_SUCC(ret)) { cell_idx++; } } } if (OB_SUCC(ret)) { event_iter_++; row = &cur_row_; } } return ret; } int ObAllVirtualSysEventI1::get_the_diag_info( const uint64_t tenant_id, common::ObDiagnoseTenantInfo &diag_info) { int ret = OB_SUCCESS; diag_info.reset(); if (!is_contain(get_index_ids(), (int64_t)tenant_id)) { ret = OB_ITER_END; } else if (OB_FAIL(common::ObDIGlobalTenantCache::get_instance().get_the_diag_info(tenant_id, diag_info))) { if (OB_ENTRY_NOT_EXIST == ret) { ret = OB_SUCCESS; } else { SERVER_LOG(WARN, "Fail to get tenant stat event", KR(ret), K(tenant_id)); } } return ret; } } /* namespace observer */ } /* namespace oceanbase */