/* Copyright (c) 2016, 2019, Oracle and/or its affiliates. All rights reserved. This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License, version 2.0, as published by the Free Software Foundation. This program is also distributed with certain software (including but not limited to OpenSSL) that is licensed under separate terms, as designated in a particular file or component or in included license documentation. The authors of MySQL hereby grant you an additional permission to link the program and your derivative works with the separately licensed software that they have included with MySQL. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License, version 2.0, for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */ #include "sql/dd/info_schema/show.h" #include #include "lex_string.h" #include "m_string.h" #include "my_sqlcommand.h" #include "sql/dd/info_schema/show_query_builder.h" // Select_lex_builder #include "sql/dd/info_schema/table_stats.h" #include "sql/dd/string_type.h" #include "sql/item_cmpfunc.h" // Item_func_case #include "sql/key.h" #include "sql/parse_tree_node_base.h" #include "sql/sql_class.h" #include "sql/sql_lex.h" #include "sql/table.h" #include "sql_string.h" namespace dd { namespace info_schema { // Build a substitute query for SHOW CHARSETS. SELECT_LEX *build_show_character_set_query(const POS &pos, THD *thd, const String *wild, Item *where_cond) { static const LEX_CSTRING system_view_name = { STRING_WITH_LEN("CHARACTER_SETS")}; // Define field name literal used in query to be built. static const LEX_CSTRING field_charset = { STRING_WITH_LEN("CHARACTER_SET_NAME")}; static const LEX_CSTRING alias_charset = {STRING_WITH_LEN("Charset")}; static const LEX_CSTRING field_desc = {STRING_WITH_LEN("DESCRIPTION")}; static const LEX_CSTRING alias_desc = {STRING_WITH_LEN("Description")}; static const LEX_CSTRING field_collate = { STRING_WITH_LEN("DEFAULT_COLLATE_NAME")}; static const LEX_CSTRING alias_collate = { STRING_WITH_LEN("Default collation")}; static const LEX_CSTRING field_maxlen = {STRING_WITH_LEN("MAXLEN")}; static const LEX_CSTRING alias_maxlen = {STRING_WITH_LEN("Maxlen")}; /* Build sub query. ... SELECT CHARACTER_SET_NAME as Charset, DESCRIPTION as Description, DEFAULT_COLLATE_NAME as `Default collation`, MAXLEN as `Maxlen` ... */ Select_lex_builder sub_query(&pos, thd); if (sub_query.add_select_item(field_charset, alias_charset) || sub_query.add_select_item(field_desc, alias_desc)) return nullptr; if (thd->variables.default_collation_for_utf8mb4 == &my_charset_utf8mb4_0900_ai_ci) { if (sub_query.add_select_item(field_collate, alias_collate)) return nullptr; } else { MEM_ROOT *const mem_root = thd->mem_root; // ... CHARACTER_SETS ... Item *charset_item = new (mem_root) Item_field(pos, NullS, NullS, field_charset.str); if (charset_item == nullptr) return nullptr; // ... 'utf8mb4' ... Item *utf8mb4_item = new (mem_root) Item_string(STRING_WITH_LEN("utf8mb4"), system_charset_info); if (utf8mb4_item == nullptr) return nullptr; // ... = ... Item *if_cond = new (mem_root) Item_func_eq(pos, charset_item, utf8mb4_item); if (if_cond == nullptr) return nullptr; // ... 'utf8mb4_general_ci' ... Item *if_then = new (mem_root) Item_string(STRING_WITH_LEN("utf8mb4_general_ci"), system_charset_info); if (if_then == nullptr) return nullptr; // ... DEFAULT_COLLATE_NAME ... Item *if_else = new (mem_root) Item_field(pos, NullS, NullS, field_collate.str); if (if_else == nullptr) return nullptr; /* IF(CHARACTER_SETS = 'utf8mb4', 'utf8mb4_general_ci', DEFAULT_COLLATE_NAME) */ Item *if_func = new (mem_root) Item_func_if(pos, if_cond, if_then, if_else); if (if_func == nullptr || sub_query.add_select_expr(if_func, alias_collate)) return nullptr; } if (sub_query.add_select_item(field_maxlen, alias_maxlen)) return nullptr; // ... FROM information_schema. ... if (sub_query.add_from_item(INFORMATION_SCHEMA_NAME, system_view_name)) return nullptr; /* Build the top level query */ Select_lex_builder top_query(&pos, thd); // SELECT * FROM ... if (top_query.add_star_select_item() || top_query.add_from_item( sub_query.prepare_derived_table(system_view_name))) return nullptr; // SELECT * FROM WHERE Charset LIKE ... if (wild) { Item *like = top_query.prepare_like_item(alias_charset, wild); if (!like || top_query.add_condition(like)) return nullptr; } else if (where_cond && top_query.add_condition(where_cond)) return nullptr; // ... ORDER BY 'Charset' ... if (top_query.add_order_by(alias_charset)) return nullptr; SELECT_LEX *sl = top_query.prepare_select_lex(); // sql_command is set to SQL_QUERY after above call, so. thd->lex->sql_command = SQLCOM_SHOW_CHARSETS; return sl; } // Build a substitute query for SHOW COLLATION. SELECT_LEX *build_show_collation_query(const POS &pos, THD *thd, const String *wild, Item *where_cond) { static const LEX_CSTRING system_view_name = {STRING_WITH_LEN("COLLATIONS")}; // Define field name literal used in query to be built. static const LEX_CSTRING field_collation = { STRING_WITH_LEN("COLLATION_NAME")}; static const LEX_CSTRING alias_collation = {STRING_WITH_LEN("Collation")}; static const LEX_CSTRING field_charset = { STRING_WITH_LEN("CHARACTER_SET_NAME")}; static const LEX_CSTRING alias_charset = {STRING_WITH_LEN("Charset")}; static const LEX_CSTRING field_id = {STRING_WITH_LEN("ID")}; static const LEX_CSTRING alias_id = {STRING_WITH_LEN("Id")}; static const LEX_CSTRING field_default = {STRING_WITH_LEN("IS_DEFAULT")}; static const LEX_CSTRING alias_default = {STRING_WITH_LEN("Default")}; static const LEX_CSTRING field_compiled = {STRING_WITH_LEN("IS_COMPILED")}; static const LEX_CSTRING alias_compiled = {STRING_WITH_LEN("Compiled")}; static const LEX_CSTRING field_sortlen = {STRING_WITH_LEN("SORTLEN")}; static const LEX_CSTRING alias_sortlen = {STRING_WITH_LEN("Sortlen")}; static const LEX_CSTRING field_pad_attribute = { STRING_WITH_LEN("PAD_ATTRIBUTE")}; static const LEX_CSTRING alias_pad_attribute = { STRING_WITH_LEN("Pad_attribute")}; /* Build sub query. ... SELECT COLLATION_NAME as `Collation`, CHARACTER_SET_NAME as `Charset`, ID as `Id`, IS_DEFAULT as `Default`, IS_COMPILED as `Compiled`, SORTLEN as `Sortlen`, PAD_ATTRIBUTE AS `Pad_attribute` ... */ Select_lex_builder sub_query(&pos, thd); if (sub_query.add_select_item(field_collation, alias_collation) || sub_query.add_select_item(field_charset, alias_charset) || sub_query.add_select_item(field_id, alias_id)) return nullptr; if (thd->variables.default_collation_for_utf8mb4 == &my_charset_utf8mb4_0900_ai_ci) { if (sub_query.add_select_item(field_default, alias_default)) return nullptr; } else { MEM_ROOT *const mem_root = thd->mem_root; // ... `ID` ... Item *case_value_item = new (mem_root) Item_field(pos, NullS, NullS, field_id.str); if (case_value_item == nullptr) return nullptr; // OOM List case_when_list; // ... WHEN ... Item *old_default_collation = new (mem_root) Item_uint(my_charset_utf8mb4_general_ci.number); if (old_default_collation == nullptr || case_when_list.push_back(old_default_collation)) return nullptr; // OOM // ... THEN 'Yes' ... Item *force_old_default_collation = new (mem_root) Item_string(STRING_WITH_LEN("Yes"), system_charset_info); if (force_old_default_collation == nullptr || case_when_list.push_back(force_old_default_collation)) return nullptr; // OOM // ... WHEN ... Item *new_default_collation = new (mem_root) Item_uint(my_charset_utf8mb4_0900_ai_ci.number); if (new_default_collation == nullptr || case_when_list.push_back(new_default_collation)) return nullptr; // OOM // ... THEN '' ... Item *suppress_new_default_collation = new (mem_root) Item_string(STRING_WITH_LEN(""), system_charset_info); if (suppress_new_default_collation == nullptr || case_when_list.push_back(suppress_new_default_collation)) return nullptr; // OOM // ... ELSE IS_DEFAULT ... Item *case_else_item = new (mem_root) Item_field(pos, NullS, NullS, field_default.str); if (case_else_item == nullptr) return nullptr; // CASE `ID` // WHEN THEN TRUE // WHEN THEN FALSE // ELSE `IS_DEFAULT` Item *case_func = new (mem_root) Item_func_case(pos, case_when_list, case_value_item, case_else_item); if (case_func == nullptr || sub_query.add_select_expr(case_func, alias_default)) return nullptr; } if (sub_query.add_select_item(field_compiled, alias_compiled) || sub_query.add_select_item(field_sortlen, alias_sortlen) || sub_query.add_select_item(field_pad_attribute, alias_pad_attribute)) return nullptr; // ... FROM information_schema. ... if (sub_query.add_from_item(INFORMATION_SCHEMA_NAME, system_view_name)) return nullptr; /* Build the top level query */ Select_lex_builder top_query(&pos, thd); // SELECT * FROM ... if (top_query.add_star_select_item() || top_query.add_from_item( sub_query.prepare_derived_table(system_view_name))) return nullptr; // SELECT * FROM WHERE Collation LIKE ... if (wild) { Item *like = top_query.prepare_like_item(alias_collation, wild); if (!like || top_query.add_condition(like)) return nullptr; } else if (where_cond && top_query.add_condition(where_cond)) return nullptr; // ... ORDER BY 'Collation' ... if (top_query.add_order_by(alias_collation)) return nullptr; SELECT_LEX *sl = top_query.prepare_select_lex(); // sql_command is set to SQL_QUERY after above call, so. thd->lex->sql_command = SQLCOM_SHOW_COLLATIONS; return sl; } // Build a substitute query for SHOW DATABASES. SELECT_LEX *build_show_databases_query(const POS &pos, THD *thd, String *wild, Item *where_cond) { static const LEX_CSTRING system_view_name = {STRING_WITH_LEN("SCHEMATA")}; // Define field name literal used in query to be built. static const LEX_CSTRING field_schema_name = {STRING_WITH_LEN("SCHEMA_NAME")}; static const LEX_CSTRING alias_database = {STRING_WITH_LEN("Database")}; // Build the alias 'Database (%)' String_type alias; alias.append(alias_database.str); if (wild) { alias.append(" ("); alias.append(wild->ptr(), wild->length()); alias.append(")"); } char *tmp = static_cast(thd->alloc(alias.length() + 1)); memcpy(tmp, alias.c_str(), alias.length()); *(tmp + alias.length()) = '\0'; LEX_CSTRING alias_lex_string = {tmp, alias.length()}; /* Build sub query. ... SELECT SCHEMA_NAME as `Database`, FROM information_schema.schemata; ... */ Select_lex_builder sub_query(&pos, thd); if (sub_query.add_select_item(field_schema_name, alias_database) || sub_query.add_from_item(INFORMATION_SCHEMA_NAME, system_view_name)) return nullptr; /* Build the top level query */ Select_lex_builder top_query(&pos, thd); // SELECT * FROM ... if (top_query.add_select_item(alias_database, alias_lex_string) || top_query.add_from_item( sub_query.prepare_derived_table(system_view_name))) return nullptr; // SELECT * FROM WHERE Database LIKE ... if (wild) { // Convert IS db and table name to desired form. dd::info_schema::convert_table_name_case(wild->ptr(), nullptr); Item *like = top_query.prepare_like_item(alias_database, wild); if (!like || top_query.add_condition(like)) return nullptr; } else if (where_cond && top_query.add_condition(where_cond)) return nullptr; // ... ORDER BY 'Database' ... if (top_query.add_order_by(alias_database)) return nullptr; SELECT_LEX *sl = top_query.prepare_select_lex(); // sql_command is set to SQL_QUERY after above call, so. thd->lex->sql_command = SQLCOM_SHOW_DATABASES; return sl; } /** Add fields required by SHOW TABLE STATUS. @param query - Select_lex_builder to which we add fields. @param alias_as_alias - Add select items of the form, 'alias AS alias'. This is required for top level query, as the real field names would be valied only the sub query that we are building for SHOW TABLE STATUS. Check comments in sql/dd/show.h for more details. @returns false on success. true on failure. */ static bool add_table_status_fields(Select_lex_builder *query, bool alias_as_alias) { // Field for SHOW TABLE STATUS static const LEX_CSTRING field_engine = {STRING_WITH_LEN("ENGINE")}; static const LEX_CSTRING alias_engine = {STRING_WITH_LEN("Engine")}; static const LEX_CSTRING field_version = {STRING_WITH_LEN("VERSION")}; static const LEX_CSTRING alias_version = {STRING_WITH_LEN("Version")}; static const LEX_CSTRING field_row_format = {STRING_WITH_LEN("ROW_FORMAT")}; static const LEX_CSTRING alias_row_format = {STRING_WITH_LEN("Row_format")}; static const LEX_CSTRING field_rows = {STRING_WITH_LEN("TABLE_ROWS")}; static const LEX_CSTRING alias_rows = {STRING_WITH_LEN("Rows")}; static const LEX_CSTRING field_avg_row_length = { STRING_WITH_LEN("AVG_ROW_LENGTH")}; static const LEX_CSTRING alias_avg_row_length = { STRING_WITH_LEN("Avg_row_length")}; static const LEX_CSTRING field_data_length = {STRING_WITH_LEN("DATA_LENGTH")}; static const LEX_CSTRING alias_data_length = {STRING_WITH_LEN("Data_length")}; static const LEX_CSTRING field_max_data_length = { STRING_WITH_LEN("MAX_DATA_LENGTH")}; static const LEX_CSTRING alias_max_data_length = { STRING_WITH_LEN("Max_data_length")}; static const LEX_CSTRING field_index_length = { STRING_WITH_LEN("INDEX_LENGTH")}; static const LEX_CSTRING alias_index_length = { STRING_WITH_LEN("Index_length")}; static const LEX_CSTRING field_data_free = {STRING_WITH_LEN("DATA_FREE")}; static const LEX_CSTRING alias_data_free = {STRING_WITH_LEN("Data_free")}; static const LEX_CSTRING field_auto_increment = { STRING_WITH_LEN("AUTO_INCREMENT")}; static const LEX_CSTRING alias_auto_increment = { STRING_WITH_LEN("Auto_increment")}; static const LEX_CSTRING field_create_time = {STRING_WITH_LEN("CREATE_TIME")}; static const LEX_CSTRING alias_create_time = {STRING_WITH_LEN("Create_time")}; static const LEX_CSTRING field_update_time = {STRING_WITH_LEN("UPDATE_TIME")}; static const LEX_CSTRING alias_update_time = {STRING_WITH_LEN("Update_time")}; static const LEX_CSTRING field_check_time = {STRING_WITH_LEN("CHECK_TIME")}; static const LEX_CSTRING alias_check_time = {STRING_WITH_LEN("Check_time")}; static const LEX_CSTRING field_collation = { STRING_WITH_LEN("TABLE_COLLATION")}; static const LEX_CSTRING alias_collation = {STRING_WITH_LEN("Collation")}; static const LEX_CSTRING field_checksum = {STRING_WITH_LEN("CHECKSUM")}; static const LEX_CSTRING alias_checksum = {STRING_WITH_LEN("Checksum")}; static const LEX_CSTRING field_create_options = { STRING_WITH_LEN("CREATE_OPTIONS")}; static const LEX_CSTRING alias_create_options = { STRING_WITH_LEN("Create_options")}; static const LEX_CSTRING field_comment = {STRING_WITH_LEN("TABLE_COMMENT")}; static const LEX_CSTRING alias_comment = {STRING_WITH_LEN("Comment")}; if (alias_as_alias == false) { if (query->add_select_item(field_engine, alias_engine) || query->add_select_item(field_version, alias_version) || query->add_select_item(field_row_format, alias_row_format) || query->add_select_item(field_rows, alias_rows) || query->add_select_item(field_avg_row_length, alias_avg_row_length) || query->add_select_item(field_data_length, alias_data_length) || query->add_select_item(field_max_data_length, alias_max_data_length) || query->add_select_item(field_index_length, alias_index_length) || query->add_select_item(field_data_free, alias_data_free) || query->add_select_item(field_auto_increment, alias_auto_increment) || query->add_select_item(field_create_time, alias_create_time) || query->add_select_item(field_update_time, alias_update_time) || query->add_select_item(field_check_time, alias_check_time) || query->add_select_item(field_collation, alias_collation) || query->add_select_item(field_checksum, alias_checksum) || query->add_select_item(field_create_options, alias_create_options) || query->add_select_item(field_comment, alias_comment)) return true; } else { if (query->add_select_item(alias_engine, alias_engine) || query->add_select_item(alias_version, alias_version) || query->add_select_item(alias_row_format, alias_row_format) || query->add_select_item(alias_rows, alias_rows) || query->add_select_item(alias_avg_row_length, alias_avg_row_length) || query->add_select_item(alias_data_length, alias_data_length) || query->add_select_item(alias_max_data_length, alias_max_data_length) || query->add_select_item(alias_index_length, alias_index_length) || query->add_select_item(alias_data_free, alias_data_free) || query->add_select_item(alias_auto_increment, alias_auto_increment) || query->add_select_item(alias_create_time, alias_create_time) || query->add_select_item(alias_update_time, alias_update_time) || query->add_select_item(alias_check_time, alias_check_time) || query->add_select_item(alias_collation, alias_collation) || query->add_select_item(alias_checksum, alias_checksum) || query->add_select_item(alias_create_options, alias_create_options) || query->add_select_item(alias_comment, alias_comment)) return true; } return false; } // Build a substitute query for SHOW TABLES / TABLE STATUS. SELECT_LEX *build_show_tables_query(const POS &pos, THD *thd, String *wild, Item *where_cond, bool include_status_fields) { static const LEX_CSTRING system_view_name = {STRING_WITH_LEN("TABLES")}; // Define field name literal used in query to be built. static const LEX_CSTRING field_table = {STRING_WITH_LEN("TABLE_NAME")}; static const LEX_CSTRING alias_table = {STRING_WITH_LEN("Name")}; static const LEX_CSTRING field_database = {STRING_WITH_LEN("TABLE_SCHEMA")}; static const LEX_CSTRING alias_database = {STRING_WITH_LEN("Database")}; static const LEX_CSTRING field_table_type = {STRING_WITH_LEN("TABLE_TYPE")}; static const LEX_CSTRING alias_table_type = {STRING_WITH_LEN("Table_type")}; // Get the current logged in schema name size_t dummy; if (thd->lex->select_lex->db == nullptr && thd->lex->copy_db_to(&thd->lex->select_lex->db, &dummy)) return nullptr; // Convert IS db and table name to desired form. dd::info_schema::convert_table_name_case(thd->lex->select_lex->db, wild ? wild->ptr() : nullptr); LEX_STRING cur_db = {thd->lex->select_lex->db, strlen(thd->lex->select_lex->db)}; if (check_and_convert_db_name(&cur_db, false) != Ident_name_check::OK) return nullptr; /* Build the alias 'Tables_in_ %' we are building SHOW TABLES and not SHOW TABLE STATUS */ String_type alias; if (include_status_fields) { // Set the output alias for first column of SHOW TABLE STATUS as 'Name' alias.append(alias_table.str); } else { // Build the output alias for SHOW TABLES alias.append("Tables_in_"); alias.append(cur_db.str); if (wild) { alias.append(" ("); alias.append(wild->ptr()); alias.append(")"); } } char *tmp = static_cast(thd->alloc(alias.length() + 1)); memcpy(tmp, alias.c_str(), alias.length()); *(tmp + alias.length()) = '\0'; LEX_CSTRING alias_lex_string = {tmp, alias.length()}; /* Build sub query. ... */ Select_lex_builder sub_query(&pos, thd); if (sub_query.add_select_item(field_database, alias_database) || sub_query.add_select_item(field_table, alias_lex_string) || (include_status_fields == true && add_table_status_fields(&sub_query, false)) || (include_status_fields == false && thd->lex->verbose && sub_query.add_select_item(field_table_type, alias_table_type))) return nullptr; // ... FROM information_schema.tables ... if (sub_query.add_from_item(INFORMATION_SCHEMA_NAME, system_view_name)) return nullptr; /* Build the top level query */ Select_lex_builder top_query(&pos, thd); // SELECT * FROM ... if (top_query.add_select_item(alias_lex_string, alias_lex_string) || (include_status_fields == true && add_table_status_fields(&top_query, true)) || (include_status_fields == false && thd->lex->verbose && top_query.add_select_item(alias_table_type, alias_table_type)) || top_query.add_from_item( sub_query.prepare_derived_table(system_view_name))) return nullptr; // ... WHERE 'Database' = ... Item *database_condition = top_query.prepare_equal_item(alias_database, to_lex_cstring(cur_db)); if (top_query.add_condition(database_condition)) return nullptr; // ... [ AND ] Table LIKE ... if (wild) { Item *like = top_query.prepare_like_item(alias_lex_string, wild); if (!like || top_query.add_condition(like)) return nullptr; } // ... [ AND ] ... if (where_cond && top_query.add_condition(where_cond)) return nullptr; // ... ORDER BY 'Table' ... if (top_query.add_order_by(alias_lex_string)) return nullptr; SELECT_LEX *sl = top_query.prepare_select_lex(); // sql_command is set to SQL_QUERY after above call, so. if (include_status_fields) thd->lex->sql_command = SQLCOM_SHOW_TABLE_STATUS; else thd->lex->sql_command = SQLCOM_SHOW_TABLES; return sl; } // Build a substitute query for SHOW COLUMNS/FIELDS OR DESCRIBE. SELECT_LEX *build_show_columns_query(const POS &pos, THD *thd, Table_ident *table_ident, const String *wild, Item *where_cond) { static const LEX_CSTRING system_view_name = {STRING_WITH_LEN("COLUMNS")}; // Define field name literal used in query to be built. static const LEX_CSTRING field_database = {STRING_WITH_LEN("TABLE_SCHEMA")}; static const LEX_CSTRING alias_database = {STRING_WITH_LEN("Database")}; static const LEX_CSTRING field_table = {STRING_WITH_LEN("TABLE_NAME")}; static const LEX_CSTRING alias_table = {STRING_WITH_LEN("Table")}; static const LEX_CSTRING field_field = {STRING_WITH_LEN("COLUMN_NAME")}; static const LEX_CSTRING alias_field = {STRING_WITH_LEN("Field")}; static const LEX_CSTRING field_column_type = {STRING_WITH_LEN("COLUMN_TYPE")}; static const LEX_CSTRING alias_column_type = {STRING_WITH_LEN("Type")}; static const LEX_CSTRING field_collation = { STRING_WITH_LEN("COLLATION_NAME")}; static const LEX_CSTRING alias_collation = {STRING_WITH_LEN("Collation")}; static const LEX_CSTRING field_null = {STRING_WITH_LEN("IS_NULLABLE")}; static const LEX_CSTRING alias_null = {STRING_WITH_LEN("Null")}; static const LEX_CSTRING field_key = {STRING_WITH_LEN("COLUMN_KEY")}; static const LEX_CSTRING alias_key = {STRING_WITH_LEN("Key")}; static const LEX_CSTRING field_default = {STRING_WITH_LEN("COLUMN_DEFAULT")}; static const LEX_CSTRING alias_default = {STRING_WITH_LEN("Default")}; static const LEX_CSTRING field_extra = {STRING_WITH_LEN("EXTRA")}; static const LEX_CSTRING alias_extra = {STRING_WITH_LEN("Extra")}; static const LEX_CSTRING field_privileges = {STRING_WITH_LEN("PRIVILEGES")}; static const LEX_CSTRING alias_privileges = {STRING_WITH_LEN("Privileges")}; static const LEX_CSTRING field_comment = {STRING_WITH_LEN("COLUMN_COMMENT")}; static const LEX_CSTRING alias_comment = {STRING_WITH_LEN("Comment")}; static const LEX_CSTRING field_ordinal_position = { STRING_WITH_LEN("ORDINAL_POSITION")}; static const LEX_CSTRING alias_ordinal_position = { STRING_WITH_LEN("Ordinal_position")}; // Get the current logged in schema name LEX_CSTRING cur_db; if (table_ident->db.str) { cur_db.str = table_ident->db.str; cur_db.length = table_ident->db.length; } else if (thd->lex->copy_db_to(&cur_db.str, &cur_db.length)) return nullptr; // Convert IS db and table name to desired form. dd::info_schema::convert_table_name_case( const_cast(cur_db.str), const_cast(table_ident->table.str)); /* Build sub query. ... */ Select_lex_builder sub_query(&pos, thd); if (sub_query.add_select_item(field_database, alias_database) || sub_query.add_select_item(field_table, alias_table) || sub_query.add_select_item(field_field, alias_field) || sub_query.add_select_item(field_column_type, alias_column_type) || (thd->lex->verbose == true && sub_query.add_select_item(field_collation, alias_collation)) || sub_query.add_select_item(field_null, alias_null) || sub_query.add_select_item(field_key, alias_key) || sub_query.add_select_item(field_default, alias_default) || sub_query.add_select_item(field_extra, alias_extra) || (thd->lex->verbose == true && sub_query.add_select_item(field_privileges, alias_privileges)) || (thd->lex->verbose == true && sub_query.add_select_item(field_comment, alias_comment)) || sub_query.add_select_item(field_ordinal_position, alias_ordinal_position)) return nullptr; // ... FROM information_schema.columns ... if (sub_query.add_from_item(INFORMATION_SCHEMA_NAME, system_view_name)) return nullptr; /* Build the top level query */ Select_lex_builder top_query(&pos, thd); // SELECT * FROM ... if (top_query.add_select_item(alias_field, alias_field) || top_query.add_select_item(alias_column_type, alias_column_type) || (thd->lex->verbose == true && top_query.add_select_item(alias_collation, alias_collation)) || top_query.add_select_item(alias_null, alias_null) || top_query.add_select_item(alias_key, alias_key) || top_query.add_select_item(alias_default, alias_default) || top_query.add_select_item(alias_extra, alias_extra) || (thd->lex->verbose == true && top_query.add_select_item(alias_privileges, alias_privileges)) || (thd->lex->verbose == true && top_query.add_select_item(alias_comment, alias_comment)) || top_query.add_from_item( sub_query.prepare_derived_table(system_view_name))) return nullptr; // ... WHERE 'Database' = ... Item *database_condition = top_query.prepare_equal_item(alias_database, cur_db); if (!database_condition || top_query.add_condition(database_condition)) return nullptr; // ... AND Table = table_ident->table->str ... Item *table_condition = top_query.prepare_equal_item(alias_table, table_ident->table); if (!table_condition || top_query.add_condition(table_condition)) return nullptr; // ... [ AND ] Field LIKE ... if (wild) { Item *like = top_query.prepare_like_item(alias_field, wild); if (!like || top_query.add_condition(like)) return nullptr; } // ... [ AND ] ... if (where_cond && top_query.add_condition(where_cond)) return nullptr; // ... ORDER BY 'Ordinal_position' ... if (top_query.add_order_by(alias_ordinal_position)) return nullptr; // Prepare the SELECT_LEX SELECT_LEX *sl = top_query.prepare_select_lex(); if (sl == nullptr) return nullptr; // sql_command is set to SQL_QUERY after above call, so. thd->lex->sql_command = SQLCOM_SHOW_FIELDS; return sl; } // Build a substitute query for SHOW INDEX|KEYS|INDEXES SELECT_LEX *build_show_keys_query(const POS &pos, THD *thd, Table_ident *table_ident, Item *where_cond) { LEX_CSTRING system_view_name = {STRING_WITH_LEN("SHOW_STATISTICS")}; // Define field name literal used in query to be built. static const LEX_CSTRING field_database = {STRING_WITH_LEN("TABLE_SCHEMA")}; static const LEX_CSTRING alias_database = {STRING_WITH_LEN("Database")}; static const LEX_CSTRING field_table = {STRING_WITH_LEN("TABLE_NAME")}; static const LEX_CSTRING alias_table = {STRING_WITH_LEN("Table")}; static const LEX_CSTRING field_non_unique = {STRING_WITH_LEN("NON_UNIQUE")}; static const LEX_CSTRING alias_non_unique = {STRING_WITH_LEN("Non_unique")}; static const LEX_CSTRING field_key = {STRING_WITH_LEN("INDEX_NAME")}; static const LEX_CSTRING alias_key = {STRING_WITH_LEN("Key_name")}; static const LEX_CSTRING field_seq_in_index = { STRING_WITH_LEN("SEQ_IN_INDEX")}; static const LEX_CSTRING alias_seq_in_index = { STRING_WITH_LEN("Seq_in_index")}; static const LEX_CSTRING field_column_name = {STRING_WITH_LEN("COLUMN_NAME")}; static const LEX_CSTRING alias_column_name = {STRING_WITH_LEN("Column_name")}; static const LEX_CSTRING field_collation = {STRING_WITH_LEN("COLLATION")}; static const LEX_CSTRING alias_collation = {STRING_WITH_LEN("Collation")}; static const LEX_CSTRING field_cardinality = {STRING_WITH_LEN("CARDINALITY")}; static const LEX_CSTRING alias_cardinality = {STRING_WITH_LEN("Cardinality")}; static const LEX_CSTRING field_sub_part = {STRING_WITH_LEN("SUB_PART")}; static const LEX_CSTRING alias_sub_part = {STRING_WITH_LEN("Sub_part")}; static const LEX_CSTRING field_packed = {STRING_WITH_LEN("PACKED")}; static const LEX_CSTRING alias_packed = {STRING_WITH_LEN("Packed")}; static const LEX_CSTRING field_null = {STRING_WITH_LEN("NULLABLE")}; static const LEX_CSTRING alias_null = {STRING_WITH_LEN("Null")}; static const LEX_CSTRING field_type = {STRING_WITH_LEN("INDEX_TYPE")}; static const LEX_CSTRING alias_type = {STRING_WITH_LEN("Index_type")}; static const LEX_CSTRING field_comment = {STRING_WITH_LEN("COMMENT")}; static const LEX_CSTRING alias_comment = {STRING_WITH_LEN("Comment")}; static const LEX_CSTRING field_index_comment = { STRING_WITH_LEN("INDEX_COMMENT")}; static const LEX_CSTRING alias_index_comment = { STRING_WITH_LEN("Index_comment")}; static const LEX_CSTRING field_is_visible = {STRING_WITH_LEN("IS_VISIBLE")}; static const LEX_CSTRING alias_visible = {STRING_WITH_LEN("Visible")}; static const LEX_CSTRING alias_index_pos = { STRING_WITH_LEN("INDEX_ORDINAL_POSITION")}; static const LEX_CSTRING alias_column_pos = { STRING_WITH_LEN("COLUMN_ORDINAL_POSITION")}; static const LEX_CSTRING field_expression = {STRING_WITH_LEN("EXPRESSION")}; static const LEX_CSTRING alias_expression = {STRING_WITH_LEN("Expression")}; // Get the current logged in schema name LEX_CSTRING cur_db; if (table_ident->db.str) { cur_db.str = table_ident->db.str; cur_db.length = table_ident->db.length; } else if (thd->lex->copy_db_to(&cur_db.str, &cur_db.length)) return nullptr; // Convert IS db and table name to desired form. dd::info_schema::convert_table_name_case( const_cast(cur_db.str), const_cast(table_ident->table.str)); /* Build sub query. ... */ Select_lex_builder sub_query(&pos, thd); if (sub_query.add_select_item(field_database, alias_database) || sub_query.add_select_item(field_table, alias_table) || sub_query.add_select_item(field_non_unique, alias_non_unique) || sub_query.add_select_item(field_key, alias_key) || sub_query.add_select_item(field_seq_in_index, alias_seq_in_index) || sub_query.add_select_item(field_column_name, alias_column_name) || sub_query.add_select_item(field_collation, alias_collation) || sub_query.add_select_item(field_cardinality, alias_cardinality) || sub_query.add_select_item(field_sub_part, alias_sub_part) || sub_query.add_select_item(field_packed, alias_packed) || sub_query.add_select_item(field_null, alias_null) || sub_query.add_select_item(field_type, alias_type) || sub_query.add_select_item(field_comment, alias_comment) || sub_query.add_select_item(field_index_comment, alias_index_comment) || sub_query.add_select_item(field_is_visible, alias_visible) || sub_query.add_select_item(field_expression, alias_expression) || sub_query.add_select_item(alias_index_pos, alias_index_pos) || sub_query.add_select_item(alias_column_pos, alias_column_pos)) return nullptr; // ... FROM information_schema.columns ... if (sub_query.add_from_item(INFORMATION_SCHEMA_NAME, system_view_name)) return nullptr; /* Build the top level query */ Select_lex_builder top_query(&pos, thd); // SELECT * FROM ... if (top_query.add_select_item(alias_table, alias_table) || top_query.add_select_item(alias_non_unique, alias_non_unique) || top_query.add_select_item(alias_key, alias_key) || top_query.add_select_item(alias_seq_in_index, alias_seq_in_index) || top_query.add_select_item(alias_column_name, alias_column_name) || top_query.add_select_item(alias_collation, alias_collation) || top_query.add_select_item(alias_cardinality, alias_cardinality) || top_query.add_select_item(alias_sub_part, alias_sub_part) || top_query.add_select_item(alias_packed, alias_packed) || top_query.add_select_item(alias_null, alias_null) || top_query.add_select_item(alias_type, alias_type) || top_query.add_select_item(alias_comment, alias_comment) || top_query.add_select_item(alias_index_comment, alias_index_comment) || top_query.add_select_item(alias_visible, alias_visible) || top_query.add_select_item(alias_expression, alias_expression) || top_query.add_from_item( sub_query.prepare_derived_table(system_view_name))) return nullptr; // ... WHERE 'Database' = ... Item *database_condition = top_query.prepare_equal_item(alias_database, cur_db); if (!database_condition || top_query.add_condition(database_condition)) return nullptr; // ... AND Table = table_ident->table->str ... Item *table_condition = top_query.prepare_equal_item(alias_table, table_ident->table); if (!table_condition || top_query.add_condition(table_condition)) return nullptr; // ... [ AND ] ... if (where_cond && top_query.add_condition(where_cond)) return nullptr; // ... ORDER BY 'INDEX_ORDINAL_POSITION, COLUMN_ORDINAL_POSITION' ... if (top_query.add_order_by(alias_index_pos) || top_query.add_order_by(alias_column_pos)) return nullptr; // Prepare the SELECT_LEX SELECT_LEX *sl = top_query.prepare_select_lex(); if (sl == nullptr) return nullptr; // sql_command is set to SQL_QUERY after above call, so. thd->lex->sql_command = SQLCOM_SHOW_KEYS; return sl; } // Build a substitute query for SHOW TRIGGERS SELECT_LEX *build_show_triggers_query(const POS &pos, THD *thd, String *wild, Item *where_cond) { static const LEX_CSTRING system_view_name = {STRING_WITH_LEN("TRIGGERS")}; // Define field name literal used in query to be built. static const LEX_CSTRING field_database = { STRING_WITH_LEN("EVENT_OBJECT_SCHEMA")}; static const LEX_CSTRING alias_database = {STRING_WITH_LEN("Database")}; static const LEX_CSTRING field_trigger = {STRING_WITH_LEN("TRIGGER_NAME")}; static const LEX_CSTRING alias_trigger = {STRING_WITH_LEN("Trigger")}; static const LEX_CSTRING field_manipulation = { STRING_WITH_LEN("EVENT_MANIPULATION")}; static const LEX_CSTRING alias_manipulation = {STRING_WITH_LEN("Event")}; static const LEX_CSTRING field_table = { STRING_WITH_LEN("EVENT_OBJECT_TABLE")}; static const LEX_CSTRING alias_table = {STRING_WITH_LEN("Table")}; static const LEX_CSTRING field_statement = { STRING_WITH_LEN("ACTION_STATEMENT")}; static const LEX_CSTRING alias_statement = {STRING_WITH_LEN("Statement")}; static const LEX_CSTRING field_timing = {STRING_WITH_LEN("ACTION_TIMING")}; static const LEX_CSTRING alias_timing = {STRING_WITH_LEN("Timing")}; static const LEX_CSTRING field_created = {STRING_WITH_LEN("CREATED")}; static const LEX_CSTRING alias_created = {STRING_WITH_LEN("Created")}; static const LEX_CSTRING field_sql_mode = {STRING_WITH_LEN("SQL_MODE")}; static const LEX_CSTRING alias_sql_mode = {STRING_WITH_LEN("sql_mode")}; static const LEX_CSTRING field_definer = {STRING_WITH_LEN("DEFINER")}; static const LEX_CSTRING alias_definer = {STRING_WITH_LEN("Definer")}; static const LEX_CSTRING field_client_cs = { STRING_WITH_LEN("CHARACTER_SET_CLIENT")}; static const LEX_CSTRING alias_client_cs = { STRING_WITH_LEN("character_set_client")}; static const LEX_CSTRING field_conn_coll = { STRING_WITH_LEN("COLLATION_CONNECTION")}; static const LEX_CSTRING alias_conn_coll = { STRING_WITH_LEN("collation_connection")}; static const LEX_CSTRING field_db_coll = { STRING_WITH_LEN("DATABASE_COLLATION")}; static const LEX_CSTRING alias_db_coll = { STRING_WITH_LEN("Database Collation")}; static const LEX_CSTRING field_action_order = { STRING_WITH_LEN("ACTION_ORDER")}; static const LEX_CSTRING alias_action_order = { STRING_WITH_LEN("action_order")}; // Get the current logged in schema name size_t dummy; if (thd->lex->select_lex->db == nullptr && thd->lex->copy_db_to(&thd->lex->select_lex->db, &dummy)) return nullptr; // Convert IS db and table name to desired form. dd::info_schema::convert_table_name_case(thd->lex->select_lex->db, wild ? wild->ptr() : nullptr); LEX_STRING cur_db = {thd->lex->select_lex->db, strlen(thd->lex->select_lex->db)}; if (check_and_convert_db_name(&cur_db, false) != Ident_name_check::OK) return nullptr; /* Build sub query. ... */ Select_lex_builder sub_query(&pos, thd); if (sub_query.add_select_item(field_database, alias_database) || sub_query.add_select_item(field_trigger, alias_trigger) || sub_query.add_select_item(field_manipulation, alias_manipulation) || sub_query.add_select_item(field_table, alias_table) || sub_query.add_select_item(field_statement, alias_statement) || sub_query.add_select_item(field_timing, alias_timing) || sub_query.add_select_item(field_created, alias_created) || sub_query.add_select_item(field_sql_mode, alias_sql_mode) || sub_query.add_select_item(field_definer, alias_definer) || sub_query.add_select_item(field_client_cs, alias_client_cs) || sub_query.add_select_item(field_conn_coll, alias_conn_coll) || sub_query.add_select_item(field_db_coll, alias_db_coll) || sub_query.add_select_item(field_action_order, alias_action_order)) return nullptr; // ... FROM information_schema.tables ... if (sub_query.add_from_item(INFORMATION_SCHEMA_NAME, system_view_name)) return nullptr; /* Build the top level query */ Select_lex_builder top_query(&pos, thd); // SELECT * FROM ... if (top_query.add_select_item(alias_trigger, alias_trigger) || top_query.add_select_item(alias_manipulation, alias_manipulation) || top_query.add_select_item(alias_table, alias_table) || top_query.add_select_item(alias_statement, alias_statement) || top_query.add_select_item(alias_timing, alias_timing) || top_query.add_select_item(alias_created, alias_created) || top_query.add_select_item(alias_sql_mode, alias_sql_mode) || top_query.add_select_item(alias_definer, alias_definer) || top_query.add_select_item(alias_client_cs, alias_client_cs) || top_query.add_select_item(alias_conn_coll, alias_conn_coll) || top_query.add_select_item(alias_db_coll, alias_db_coll) || top_query.add_from_item( sub_query.prepare_derived_table(system_view_name))) return nullptr; // ... WHERE 'Database' = ... Item *database_condition = top_query.prepare_equal_item(alias_database, to_lex_cstring(cur_db)); if (top_query.add_condition(database_condition)) return nullptr; // ... [ AND ] Table LIKE ... if (wild) { Item *like = top_query.prepare_like_item(alias_table, wild); if (!like || top_query.add_condition(like)) return nullptr; } // ... [ AND ] ... if (where_cond && top_query.add_condition(where_cond)) return nullptr; // ... ORDER BY 'Table, Event, Timing, Action order' ... if (top_query.add_order_by(alias_table) || top_query.add_order_by(alias_manipulation) || top_query.add_order_by(alias_timing) || top_query.add_order_by(alias_action_order)) return nullptr; SELECT_LEX *sl = top_query.prepare_select_lex(); // sql_command is set to SQL_QUERY after above call, so. thd->lex->sql_command = SQLCOM_SHOW_TRIGGERS; return sl; } // Build a substitute query for SHOW PROCEDURES / FUNCTIONS SELECT_LEX *build_show_procedures_query(const POS &pos, THD *thd, String *wild, Item *where_cond) { enum_sql_command current_cmd = thd->lex->sql_command; static const LEX_CSTRING system_view_name = {STRING_WITH_LEN("ROUTINES")}; // Define field name literal used in query to be built. static const LEX_CSTRING field_db = {STRING_WITH_LEN("ROUTINE_SCHEMA")}; static const LEX_CSTRING alias_db = {STRING_WITH_LEN("Db")}; static const LEX_CSTRING field_name = {STRING_WITH_LEN("ROUTINE_NAME")}; static const LEX_CSTRING alias_name = {STRING_WITH_LEN("Name")}; static const LEX_CSTRING field_type = {STRING_WITH_LEN("ROUTINE_TYPE")}; static const LEX_CSTRING alias_type = {STRING_WITH_LEN("Type")}; static const LEX_CSTRING field_definer = {STRING_WITH_LEN("DEFINER")}; static const LEX_CSTRING alias_definer = {STRING_WITH_LEN("Definer")}; static const LEX_CSTRING field_modified = {STRING_WITH_LEN("LAST_ALTERED")}; static const LEX_CSTRING alias_modified = {STRING_WITH_LEN("Modified")}; static const LEX_CSTRING field_created = {STRING_WITH_LEN("CREATED")}; static const LEX_CSTRING alias_created = {STRING_WITH_LEN("Created")}; static const LEX_CSTRING field_security_type = { STRING_WITH_LEN("SECURITY_TYPE")}; static const LEX_CSTRING alias_security_type = { STRING_WITH_LEN("Security_type")}; static const LEX_CSTRING field_comment = {STRING_WITH_LEN("ROUTINE_COMMENT")}; static const LEX_CSTRING alias_comment = {STRING_WITH_LEN("Comment")}; static const LEX_CSTRING field_client_cs = { STRING_WITH_LEN("CHARACTER_SET_CLIENT")}; static const LEX_CSTRING alias_client_cs = { STRING_WITH_LEN("character_set_client")}; static const LEX_CSTRING field_conn_coll = { STRING_WITH_LEN("COLLATION_CONNECTION")}; static const LEX_CSTRING alias_conn_coll = { STRING_WITH_LEN("collation_connection")}; static const LEX_CSTRING field_db_coll = { STRING_WITH_LEN("DATABASE_COLLATION")}; static const LEX_CSTRING alias_db_coll = { STRING_WITH_LEN("Database Collation")}; /* Build sub query. ... */ Select_lex_builder sub_query(&pos, thd); if (sub_query.add_select_item(field_db, alias_db) || sub_query.add_select_item(field_name, alias_name) || sub_query.add_select_item(field_type, alias_type) || sub_query.add_select_item(field_definer, alias_definer) || sub_query.add_select_item(field_modified, alias_modified) || sub_query.add_select_item(field_created, alias_created) || sub_query.add_select_item(field_security_type, alias_security_type) || sub_query.add_select_item(field_comment, alias_comment) || sub_query.add_select_item(field_client_cs, alias_client_cs) || sub_query.add_select_item(field_conn_coll, alias_conn_coll) || sub_query.add_select_item(field_db_coll, alias_db_coll)) return nullptr; // ... FROM information_schema.tables ... if (sub_query.add_from_item(INFORMATION_SCHEMA_NAME, system_view_name)) return nullptr; /* Build the top level query */ Select_lex_builder top_query(&pos, thd); // SELECT * FROM ... if (top_query.add_star_select_item() || top_query.add_from_item( sub_query.prepare_derived_table(system_view_name))) return nullptr; // ... WHERE 'Type' = 'PROCEDURE | FUNCTION' const char *current_type = thd->lex->sql_command == SQLCOM_SHOW_STATUS_PROC ? "PROCEDURE" : "FUNCTION"; LEX_CSTRING tmp_lex_string = {current_type, strlen(current_type)}; Item *type_condition = top_query.prepare_equal_item(alias_type, tmp_lex_string); if (top_query.add_condition(type_condition)) return nullptr; // ... [ AND ] Name LIKE ... if (wild) { Item *like = top_query.prepare_like_item(alias_name, wild); if (!like || top_query.add_condition(like)) return nullptr; } // ... [ AND ] ... if (where_cond && top_query.add_condition(where_cond)) return nullptr; // ... ORDER BY 'Db, Name' ... if (top_query.add_order_by(alias_db) || top_query.add_order_by(alias_name)) return nullptr; SELECT_LEX *sl = top_query.prepare_select_lex(); // sql_command is set to SQL_QUERY after above call, so. thd->lex->sql_command = current_cmd; return sl; } // Build a substitute query for SHOW EVENTS SELECT_LEX *build_show_events_query(const POS &pos, THD *thd, String *wild, Item *where_cond) { static const LEX_CSTRING system_view_name = {STRING_WITH_LEN("EVENTS")}; // Define field name literal used in query to be built. static const LEX_CSTRING field_db = {STRING_WITH_LEN("EVENT_SCHEMA")}; static const LEX_CSTRING alias_db = {STRING_WITH_LEN("Db")}; static const LEX_CSTRING field_name = {STRING_WITH_LEN("EVENT_NAME")}; static const LEX_CSTRING alias_name = {STRING_WITH_LEN("Name")}; static const LEX_CSTRING field_definer = {STRING_WITH_LEN("DEFINER")}; static const LEX_CSTRING alias_definer = {STRING_WITH_LEN("Definer")}; static const LEX_CSTRING field_zone = {STRING_WITH_LEN("TIME_ZONE")}; static const LEX_CSTRING alias_zone = {STRING_WITH_LEN("Time zone")}; static const LEX_CSTRING field_type = {STRING_WITH_LEN("EVENT_TYPE")}; static const LEX_CSTRING alias_type = {STRING_WITH_LEN("Type")}; static const LEX_CSTRING field_execute_at = {STRING_WITH_LEN("EXECUTE_AT")}; static const LEX_CSTRING alias_execute_at = {STRING_WITH_LEN("Execute at")}; static const LEX_CSTRING field_interval_value = { STRING_WITH_LEN("INTERVAL_VALUE")}; static const LEX_CSTRING alias_interval_value = { STRING_WITH_LEN("Interval value")}; static const LEX_CSTRING field_interval_field = { STRING_WITH_LEN("INTERVAL_FIELD")}; static const LEX_CSTRING alias_interval_field = { STRING_WITH_LEN("Interval field")}; static const LEX_CSTRING field_starts = {STRING_WITH_LEN("STARTS")}; static const LEX_CSTRING alias_starts = {STRING_WITH_LEN("Starts")}; static const LEX_CSTRING field_ends = {STRING_WITH_LEN("ENDS")}; static const LEX_CSTRING alias_ends = {STRING_WITH_LEN("Ends")}; static const LEX_CSTRING field_status = {STRING_WITH_LEN("STATUS")}; static const LEX_CSTRING alias_status = {STRING_WITH_LEN("Status")}; static const LEX_CSTRING field_originator = {STRING_WITH_LEN("ORIGINATOR")}; static const LEX_CSTRING alias_originator = {STRING_WITH_LEN("Originator")}; static const LEX_CSTRING field_client_cs = { STRING_WITH_LEN("CHARACTER_SET_CLIENT")}; static const LEX_CSTRING alias_client_cs = { STRING_WITH_LEN("character_set_client")}; static const LEX_CSTRING field_conn_coll = { STRING_WITH_LEN("COLLATION_CONNECTION")}; static const LEX_CSTRING alias_conn_coll = { STRING_WITH_LEN("collation_connection")}; static const LEX_CSTRING field_db_coll = { STRING_WITH_LEN("DATABASE_COLLATION")}; static const LEX_CSTRING alias_db_coll = { STRING_WITH_LEN("Database Collation")}; // Get the current logged in schema name size_t dummy; if (thd->lex->select_lex->db == nullptr && thd->lex->copy_db_to(&thd->lex->select_lex->db, &dummy)) return nullptr; // Convert IS db and table name to desired form. dd::info_schema::convert_table_name_case(thd->lex->select_lex->db, wild ? wild->ptr() : nullptr); LEX_STRING cur_db = {thd->lex->select_lex->db, strlen(thd->lex->select_lex->db)}; if (check_and_convert_db_name(&cur_db, false) != Ident_name_check::OK) return nullptr; /* Build sub query. ... */ Select_lex_builder sub_query(&pos, thd); if (sub_query.add_select_item(field_db, alias_db) || sub_query.add_select_item(field_name, alias_name) || sub_query.add_select_item(field_definer, alias_definer) || sub_query.add_select_item(field_zone, alias_zone) || sub_query.add_select_item(field_type, alias_type) || sub_query.add_select_item(field_execute_at, alias_execute_at) || sub_query.add_select_item(field_interval_value, alias_interval_value) || sub_query.add_select_item(field_interval_field, alias_interval_field) || sub_query.add_select_item(field_starts, alias_starts) || sub_query.add_select_item(field_ends, alias_ends) || sub_query.add_select_item(field_status, alias_status) || sub_query.add_select_item(field_originator, alias_originator) || sub_query.add_select_item(field_client_cs, alias_client_cs) || sub_query.add_select_item(field_conn_coll, alias_conn_coll) || sub_query.add_select_item(field_db_coll, alias_db_coll)) return nullptr; // ... FROM information_schema.tables ... if (sub_query.add_from_item(INFORMATION_SCHEMA_NAME, system_view_name)) return nullptr; /* Build the top level query */ Select_lex_builder top_query(&pos, thd); // SELECT * FROM ... if (top_query.add_star_select_item() || top_query.add_from_item( sub_query.prepare_derived_table(system_view_name))) return nullptr; // ... WHERE 'Database' = ... Item *database_condition = top_query.prepare_equal_item(alias_db, to_lex_cstring(cur_db)); if (top_query.add_condition(database_condition)) return nullptr; // ... [ AND ] Name LIKE ... if (wild) { Item *like = top_query.prepare_like_item(alias_name, wild); if (!like || top_query.add_condition(like)) return nullptr; } // ... [ AND ] ... if (where_cond && top_query.add_condition(where_cond)) return nullptr; // ... ORDER BY 'Db, Name' ... if (top_query.add_order_by(alias_db) || top_query.add_order_by(alias_name)) return nullptr; SELECT_LEX *sl = top_query.prepare_select_lex(); // sql_command is set to SQL_QUERY after above call, so. thd->lex->sql_command = SQLCOM_SHOW_EVENTS; return sl; } } // namespace info_schema } // namespace dd