Files
oceanbase/src/share/ob_inner_config_root_addr.cpp
2023-07-27 03:13:14 +00:00

270 lines
9.2 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 SHARE
#include "lib/string/ob_sql_string.h"
#include "lib/mysqlclient/ob_mysql_proxy.h"
#include "config/ob_server_config.h"
#include "share/ob_inner_config_root_addr.h"
#include "observer/ob_server_struct.h"
#include "common/ob_timeout_ctx.h"
#include "rootserver/ob_root_utils.h"
namespace oceanbase
{
using namespace common;
namespace share
{
int ObInnerConfigRootAddr::check_inner_stat() const
{
int ret = OB_SUCCESS;
if (!inited_) {
ret = OB_NOT_INIT;
LOG_WARN("not init", KR(ret));
} else if (OB_ISNULL(config_) || OB_ISNULL(proxy_)) {
ret = OB_ERR_UNEXPECTED;
LOG_WARN("ptr is null", KR(ret), KP_(config), KP_(proxy));
}
return ret;
}
int ObInnerConfigRootAddr::init(ObMySQLProxy &sql_proxy, ObServerConfig &config)
{
int ret = OB_SUCCESS;
if (inited_) {
ret = OB_INIT_TWICE;
LOG_WARN("init twice", K(ret));
} else if (OB_FAIL(ObRootAddrAgent::init(config))) {
LOG_WARN("init root addr agent failed", K(ret));
} else {
proxy_ = &sql_proxy;
inited_ = true;
}
return ret;
}
// update root server list if %addr_list not same with config_->rootservice_list
int ObInnerConfigRootAddr::store(const ObIAddrList &addr_list, const ObIAddrList &readonly_addr_list,
const bool force, const common::ObClusterRole cluster_role,
const int64_t timestamp)
{
// TODO: Currently there is no local cache of the read copy location of __all_core_table
UNUSED(readonly_addr_list);
UNUSED(cluster_role);
UNUSED(timestamp);
UNUSED(force);
int ret = OB_SUCCESS;
bool need_update = false;
char addr_buf[MAX_IP_PORT_LENGTH] = "";
if (OB_FAIL(check_inner_stat())) {
LOG_WARN("fail to check inner stat", KR(ret));
} else if (addr_list.count() <= 0) {
ret = OB_INVALID_ARGUMENT;
LOG_WARN("invalid argument", K(ret), "addr count", addr_list.count());
} else {
if (addr_list.count() != config_->rootservice_list.size()) {
need_update = true;
} else {
ObAddr addr;
for (int64_t i = 0; OB_SUCC(ret) && !need_update
&& i < config_->rootservice_list.size(); ++i) {
addr.reset();
int64_t sql_port = 0;
if (OB_FAIL(config_->rootservice_list.get(
static_cast<int32_t>(i), addr_buf, sizeof(addr_buf)))) {
LOG_WARN("rootservice_list get failed", K(i), K(ret));
} else if (OB_FAIL(parse_rs_addr(addr_buf, addr, sql_port))) {
LOG_WARN("parse_rs_addr failed", KCSTRING(addr_buf), K(ret));
} else {
bool found = false;
FOREACH_CNT_X(a, addr_list, !found) {
if (addr == a->get_server() && sql_port == a->get_sql_port()) {
found = true;
}
}
if (!found) {
need_update = true;
}
}
}
}
}
if (OB_SUCC(ret)) {
if (need_update) {
ObSqlString sql;
if (OB_FAIL(sql.assign("ALTER SYSTEM SET rootservice_list = '"))) {
LOG_WARN("assign sql failed", K(ret));
} else if (OB_FAIL(format_rootservice_list(addr_list, sql))) {
LOG_WARN("fail to format rootservice list", KR(ret), K(addr_list), K(sql));
}
int64_t affected_rows = 0;
ObTimeoutCtx ctx;
if (OB_FAIL(ret)) {
} else if (OB_FAIL(sql.append_fmt("'"))) {
LOG_WARN("append sql failed", K(ret));
} else if (OB_FAIL(rootserver::ObRootUtils::get_rs_default_timeout_ctx(ctx))) {
LOG_WARN("fail to get timeout ctx", K(ret), K(ctx));
} else if (OB_FAIL(proxy_->write(OB_SYS_TENANT_ID, sql.ptr(), affected_rows))) {
LOG_WARN("execute sql failed", K(ret), K(sql));
} else {
LOG_INFO("ALTER SYSTEM SET rootservice_list succeed", K(addr_list), K(force));
}
}
}
return ret;
}
int ObInnerConfigRootAddr::format_rootservice_list(const ObIAddrList &addr_list, ObSqlString &str)
{
int ret = OB_SUCCESS;
for (int64_t i = 0; OB_SUCC(ret) && i < addr_list.count(); ++i) {
char ip_buf[OB_IP_STR_BUFF] = "";
const ObRootAddr &rs = addr_list.at(i);
ObAddr addr = rs.get_server();
// If it is not the last RS, add a separator at the end
const bool need_append_delimiter = (i != addr_list.count() - 1);
if (OB_UNLIKELY(!addr.ip_to_string(ip_buf, sizeof(ip_buf)))) {
ret = OB_ERR_UNEXPECTED;
LOG_WARN("convert ip to string failed", KR(ret), K(rs));
} else if (OB_FAIL(str.append_fmt("%s%s%s:%d:%ld%s",
addr.using_ipv4() ? "" : "[",
ip_buf,
addr.using_ipv4() ? "" : "]",
addr.get_port(),
rs.get_sql_port(),
need_append_delimiter ? ";" : ""))) {
LOG_WARN("fail to append_fmt", KR(ret), K(str));
}
}
return ret;
}
int ObInnerConfigRootAddr::fetch(
ObIAddrList &addr_list,
ObIAddrList &readonly_addr_list)
{
int ret = OB_SUCCESS;
char addr_buf[MAX_IP_PORT_LENGTH] = "";
const int64_t local_cluster_id = GCONF.cluster_id;
if (OB_FAIL(check_inner_stat())) {
LOG_WARN("fail to check inner stat", KR(ret));
} else {
addr_list.reuse();
// TODO: The location of the read-only copy of __all_core_table is not cached locally, so it cannot be read
readonly_addr_list.reuse();
ObAddr addr;
ObRootAddr rs_addr;
for (int64_t i = 0; OB_SUCC(ret) && i < config_->rootservice_list.size(); ++i) {
addr.reset();
int64_t sql_port = 0;
if (OB_FAIL(config_->rootservice_list.get(
static_cast<int>(i), addr_buf, sizeof(addr_buf)))) {
LOG_WARN("get rs failed", K(ret), K(i));
} else if (OB_FAIL(parse_rs_addr(addr_buf, addr, sql_port))) {
LOG_WARN("parse_rs_addr failed", K(addr_buf), K(ret));
} else if (OB_FAIL(rs_addr.init(addr, FOLLOWER, sql_port))) {
LOG_WARN("failed to simple init rs_addr", KR(ret), K(addr), K(sql_port));
} else if (OB_FAIL(addr_list.push_back(rs_addr))) {
LOG_WARN("add rs address to array failed", K(ret));
}
}
LOG_TRACE("fetch addr_list &readonly_addr_list", K(ret),
K(addr_list), K(readonly_addr_list));
}
return ret;
}
int ObInnerConfigRootAddr::parse_rs_addr(char *addr_buf, ObAddr &addr, int64_t &sql_port)
{
int ret = OB_SUCCESS;
if (nullptr == addr_buf) {
ret = OB_INVALID_ARGUMENT;
LOG_WARN("invalid argument", K(ret), KP(addr_buf));
} else {
int64_t rpc_port = 0;
bool use_ipv6 = false;
char *ip_str = nullptr;
char *port_str = nullptr;
char *sql_port_str = nullptr;
char *save_ptr = nullptr;
/*
* ipv4格式: a.b.c.d:port1:port2, proxy只使用一个port
* ipv6格式: [a:b:c:d:e:f:g:h]:port1:port2, proxy只使用一个port
*
*/
if ('[' != addr_buf[0]) { /* ipv4 */
if (OB_ISNULL(ip_str = strtok_r(addr_buf, ":", &save_ptr))) {
ret = OB_INVALID_ARGUMENT;
LOG_WARN("parse ipv4 failed", K(addr_buf), K(ret));
} else if (OB_ISNULL(port_str = strtok_r(nullptr, ":", &save_ptr))) {
ret = OB_INVALID_ARGUMENT;
LOG_WARN("parse port failed", K(addr_buf), K(ret));
}
} else { /* ipv6 */
use_ipv6 = true;
ip_str = addr_buf + 1;
port_str = strrchr(addr_buf, ']');
if (OB_NOT_NULL(port_str)) {
*(port_str++) = '\0';
if (':' != *port_str) {
ret = OB_INVALID_ARGUMENT;
LOG_WARN("parse port failed", K(addr_buf), K(ret));
} else {
port_str++;
if (OB_ISNULL(port_str = strtok_r(port_str, ":", &save_ptr))) {
ret = OB_INVALID_ARGUMENT;
LOG_WARN("parse port failed", K(addr_buf), K(ret));
}
}
} else {
ret = OB_INVALID_ARGUMENT;
LOG_WARN("parse ipv6 failed", K(addr_buf), K(ret));
}
}
if (OB_SUCC(ret)) {
// for compatible with obproxy
// observer rs list format-->ip:rpc_port:sql_port
// obporoxy rs list format-->ip:sql_port
if (OB_ISNULL(sql_port_str = strtok_r(nullptr, ":", &save_ptr))) {
sql_port_str = port_str;
port_str = nullptr;
LOG_INFO("only has one port, used for obproxy", K(sql_port_str), K(addr_buf));
} else if (OB_NOT_NULL(strtok_r(nullptr, ":", &save_ptr))) {
ret = OB_INVALID_ARGUMENT;
LOG_WARN("addr in rs_list not in right format", K(addr_buf), K(ret));
}
if (OB_SUCC(ret)) {
if (nullptr != port_str) {
rpc_port = atoi(port_str);
} else {
rpc_port = 1; // fake rpc port, for obproxy, will never be used
}
sql_port = atoi(sql_port_str);
if (!addr.set_ip_addr(ip_str, static_cast<int32_t>(rpc_port))) {
ret = OB_ERR_UNEXPECTED;
LOG_WARN("set ip address failed", K(ret), K(ip_str), K(rpc_port));
}
}
}
}
return ret;
}
} // end namespace share
} // end oceanbase