/** * 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_RESV #include "sql/resolver/ddl/ob_set_comment_resolver.h" #include "sql/resolver/ddl/ob_ddl_resolver.h" #include "share/ob_rpc_struct.h" #include "share/schema/ob_schema_service.h" namespace oceanbase { using namespace common; namespace sql { ObSetCommentResolver::ObSetCommentResolver(ObResolverParams ¶ms) : ObDDLResolver(params), table_schema_(NULL), collation_type_(CS_TYPE_INVALID), charset_type_(CHARSET_INVALID) { } ObSetCommentResolver::~ObSetCommentResolver() { } int ObSetCommentResolver::resolve(const ParseNode &parse_tree) { int ret = OB_SUCCESS; CHECK_COMPATIBILITY_MODE(session_info_); if (OB_ISNULL(session_info_)) { ret = OB_ERR_UNEXPECTED; SQL_RESV_LOG(WARN, "session_info should not be null", K(ret)); } else if (OB_ISNULL(parse_tree.children_) || ((T_SET_TABLE_COMMENT != parse_tree.type_) && (T_SET_COLUMN_COMMENT != parse_tree.type_))) { ret = OB_ERR_UNEXPECTED; SQL_RESV_LOG(WARN, "invalid parse tree", K(ret)); } else if (!lib::is_oracle_mode()) { // do-nothing for non-oracle mode } else { const uint64_t tenant_id = session_info_->get_effective_tenant_id(); ObString database_name; ObString table_name; ObString col_name; ObAlterTableStmt *alter_table_stmt = NULL; const share::schema::ObTableSchema *table_schema = NULL; if (NULL == (alter_table_stmt = create_stmt())) { ret = OB_ALLOCATE_MEMORY_FAILED; SQL_RESV_LOG(ERROR, "failed to create alter table stmt", K(ret)); } else if (OB_FAIL(alter_table_stmt->set_tz_info_wrap(session_info_->get_tz_info_wrap()))) { LOG_WARN("failed to set_tz_info_wrap", "tz_info_wrap", session_info_->get_tz_info_wrap(), K(ret)); } else if (OB_FAIL(alter_table_stmt->set_nls_formats(session_info_->get_local_nls_date_format(), session_info_->get_local_nls_timestamp_format(), session_info_->get_local_nls_timestamp_tz_format()))) { SQL_RESV_LOG(WARN, "failed to set_nls_formats", K(ret)); } else { stmt_ = alter_table_stmt; alter_table_stmt->set_tenant_id(tenant_id); alter_table_stmt->set_is_comment_table(true); } // resolve string_value if (OB_FAIL(ret)) { // do-nothing } else if (OB_ISNULL(parse_tree.children_[1])) { ret = OB_ERR_UNEXPECTED; SQL_RESV_LOG(WARN, "invalid parse tree for comment string", K(ret)); } else { int64_t comment_length = 0; const char *comment_ptr = NULL; comment_length = parse_tree.children_[1]->str_len_; comment_ptr = parse_tree.children_[1]->str_value_; if (T_SET_TABLE_COMMENT == parse_tree.type_ && comment_length <= MAX_ORACLE_COMMENT_LENGTH) { comment_.assign_ptr((char *)(comment_ptr), static_cast(comment_length)); } else if (T_SET_COLUMN_COMMENT == parse_tree.type_ && comment_length <= MAX_ORACLE_COMMENT_LENGTH) { comment_.assign_ptr((char *)(comment_ptr), static_cast(comment_length)); } else { comment_ = ""; if (T_SET_TABLE_COMMENT == parse_tree.type_) { ret = OB_ERR_TOO_LONG_TABLE_COMMENT; LOG_USER_ERROR(OB_ERR_TOO_LONG_TABLE_COMMENT, MAX_ORACLE_COMMENT_LENGTH); } else { ret = OB_ERR_TOO_LONG_FIELD_COMMENT; LOG_USER_ERROR(OB_ERR_TOO_LONG_FIELD_COMMENT, MAX_ORACLE_COMMENT_LENGTH); } } OZ (ObSQLUtils::convert_sql_text_to_schema_for_storing( *allocator_, session_info_->get_dtc_params(), comment_)); } if (OB_SUCC(ret)) { if (T_SET_TABLE_COMMENT == parse_tree.type_) { // COMMENT ON TABLE bool is_exists = false; if (2 != parse_tree.num_child_) { ret = OB_ERR_UNEXPECTED; SQL_RESV_LOG(WARN, "invalid parse tree num", K(ret), K(parse_tree.num_child_)); } else if (OB_ISNULL(parse_tree.children_[0])) { ret = OB_ERR_UNEXPECTED; SQL_RESV_LOG(WARN, "invalid parse tree", K(ret)); } else if (OB_FAIL(resolve_table_relation_node(parse_tree.children_[0], table_name, database_name))) { SQL_RESV_LOG(WARN, "failed to resolve table name.", K(table_name), K(database_name), K(ret)); } else if (OB_FAIL(schema_checker_->check_table_exists( tenant_id, database_name, table_name, false /*index*/, false/*is_hidden*/, is_exists))) { SQL_RESV_LOG(WARN, "failed to check_table_exist", K(ret), K(tenant_id), K(database_name), K(table_name)); } else if (!is_exists) { ret = OB_TABLE_NOT_EXIST; SQL_RESV_LOG(WARN, "failed to check_table_exist", K(ret), K(tenant_id), K(database_name), K(table_name)); } else if (OB_FAIL(schema_checker_->get_table_schema( tenant_id, database_name, table_name, false/*not index table*/, table_schema))) { SQL_RESV_LOG(WARN, "failed to get table schema", K(ret), K(database_name), K(table_name)); } else if (OB_ISNULL(table_schema)) { ret = OB_ERR_UNEXPECTED; SQL_RESV_LOG(WARN, "table schema is null", K(ret), K(database_name)); } else { alter_table_stmt->set_table_id(table_schema->get_table_id()); } } else if (T_SET_COLUMN_COMMENT == parse_tree.type_) { // COMMENT ON COLUMN // resolve dabase_name if (2 != parse_tree.num_child_) { ret = OB_ERR_UNEXPECTED; SQL_RESV_LOG(WARN, "invalid parse tree num", K(ret), K(parse_tree.num_child_)); } else if (OB_ISNULL(parse_tree.children_[0])) { ret = OB_ERR_UNEXPECTED; SQL_RESV_LOG(WARN, "invalid parse tree", K(ret)); } else { bool is_exists = false; ObQualifiedName column_ref; ObNameCaseMode case_mode = OB_NAME_CASE_INVALID; ParseNode *column_ref_node = parse_tree.children_[0]; if (OB_UNLIKELY(T_COLUMN_REF != column_ref_node->type_)) { ret = OB_ERR_UNEXPECTED; SQL_RESV_LOG(WARN, "node type is not T_COLUMN_LIST", K(ret), K(column_ref_node->type_)); } else if (OB_FAIL(session_info_->get_name_case_mode(case_mode))) { SQL_RESV_LOG(WARN, "fail to get name case mode", K(ret)); } else if (OB_FAIL(ObResolverUtils::resolve_column_ref(column_ref_node, case_mode, column_ref))) { SQL_RESV_LOG(WARN, "failed to resolve column def", K(ret)); } else { table_name = column_ref.tbl_name_; col_name = column_ref.col_name_; if (column_ref.tbl_name_.empty() || column_ref.col_name_.empty()) { ret = OB_ERR_PARSER_SYNTAX; SQL_RESV_LOG(WARN, "syntax error", K(ret)); } else if (column_ref.database_name_.empty()) { if (session_info_->get_database_name().empty()) { ret = OB_ERR_NO_DB_SELECTED; SQL_RESV_LOG(WARN, "No database selected"); } else { database_name = session_info_->get_database_name(); } } else { database_name = column_ref.database_name_; } } if (OB_FAIL(ret)) { } else if (OB_FAIL(schema_checker_->get_table_schema( tenant_id, database_name, table_name, false/*not index table*/, table_schema))) { SQL_RESV_LOG(WARN, "failed to get table schema", K(ret), K(database_name), K(table_name)); } else if (OB_ISNULL(table_schema)) { ret = OB_ERR_UNEXPECTED; SQL_RESV_LOG(WARN, "table schema is null", K(ret), K(database_name)); } else if (OB_FAIL(schema_checker_->check_column_exists( tenant_id, table_schema->get_table_id(), col_name, is_exists))) { SQL_RESV_LOG(WARN, "failed to check_column_exists", K(ret), K(table_schema), K(col_name)); } else if (!is_exists) { ret = OB_ERR_COLUMN_NOT_FOUND; SQL_RESV_LOG(WARN, "column doesn't exist", K(ret), K(col_name)); } else { alter_table_stmt->set_table_id(table_schema->get_table_id()); } } } else { ret = OB_ERR_UNEXPECTED; SQL_RESV_LOG(WARN, "invalid parse tree", K(ret), K(parse_tree.type_)); } } if (OB_SUCC(ret)) { if (T_SET_TABLE_COMMENT == parse_tree.type_) { share::schema::AlterTableSchema &alter_table_schema = alter_table_stmt->get_alter_table_arg().alter_table_schema_; if (OB_FAIL(alter_table_schema.set_comment(comment_))) { SQL_RESV_LOG(WARN, "Write comment_ to alter_table_schema failed!", K(ret)); } else if (OB_FAIL(alter_table_bitset_.add_member(obrpc::ObAlterTableArg::COMMENT))) { SQL_RESV_LOG(WARN, "failed to add member to bitset!", K(ret)); } else { alter_table_schema.alter_option_bitset_ = alter_table_bitset_; } if (OB_FAIL(ret)) { alter_table_schema.reset(); SQL_RESV_LOG(WARN, "Set table options error!", K(ret)); } } else if (T_SET_COLUMN_COMMENT == parse_tree.type_) { const share::schema::ObColumnSchemaV2 *column_schema = NULL; if (OB_FAIL(schema_checker_->get_column_schema( table_schema->get_tenant_id(), table_schema->get_table_id(), col_name, column_schema, false))) { SQL_RESV_LOG(WARN, "fail to get origin column schema", K(ret)); } else if (OB_ISNULL(column_schema)) { ret = OB_ERR_UNEXPECTED; SQL_RESV_LOG(WARN, "column schema is NULL", K(ret)); } else { share::schema::AlterColumnSchema alter_column_schema; alter_column_schema.assign(*column_schema); alter_column_schema.set_comment(comment_); alter_table_stmt->set_alter_table_column(); alter_column_schema.alter_type_ = share::schema::OB_DDL_MODIFY_COLUMN; if (OB_FAIL(alter_column_schema.set_origin_column_name(col_name))) { SQL_RESV_LOG(WARN, "failed to set origin column name", K(col_name), K(ret)); } else if (OB_FAIL(alter_table_stmt->add_column(alter_column_schema))) { SQL_RESV_LOG(WARN, "Add alter column schema failed!", K(ret)); } if (OB_FAIL(ret)) { alter_column_schema.reset(); SQL_RESV_LOG(WARN, "Set column options error!", K(ret)); } } } else { ret = OB_ERR_UNEXPECTED; SQL_RESV_LOG(WARN, "Unkonwn operation type", K(ret), K(parse_tree.type_)); } if (OB_SUCC(ret)) { if (OB_FAIL(alter_table_stmt->set_origin_database_name(database_name))) { SQL_RESV_LOG(WARN, "failed to set origin database name", K(ret)); } else if (OB_FAIL(alter_table_stmt->set_origin_table_name(table_name))) { SQL_RESV_LOG(WARN, "failed to set origin table name", K(ret)); } } } } return ret; } } //namespace sql } //namespace oceanbase