mirror of
https://github.com/MariaDB/server.git
synced 2025-01-26 16:54:15 +01:00
577 lines
18 KiB
C++
577 lines
18 KiB
C++
/* Copyright (c) 2010, 2015, Oracle and/or its affiliates.
|
|
Copyright (c) 2012, 2018, MariaDB
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; version 2 of the License.
|
|
|
|
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 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-1335 USA */
|
|
|
|
#include "mariadb.h"
|
|
#include "debug_sync.h" // DEBUG_SYNC
|
|
#include "table.h" // TABLE, FOREIGN_KEY_INFO
|
|
#include "sql_class.h" // THD
|
|
#include "sql_base.h" // open_and_lock_tables
|
|
#include "sql_table.h" // write_bin_log
|
|
#include "datadict.h" // dd_recreate_table()
|
|
#include "lock.h" // MYSQL_OPEN_* flags
|
|
#include "sql_acl.h" // DROP_ACL
|
|
#include "sql_parse.h" // check_one_table_access()
|
|
#include "sql_truncate.h"
|
|
#include "wsrep_mysqld.h"
|
|
#include "sql_show.h" //append_identifier()
|
|
#include "sql_select.h"
|
|
#include "sql_delete.h"
|
|
|
|
/**
|
|
Append a list of field names to a string.
|
|
|
|
@param str The string.
|
|
@param fields The list of field names.
|
|
|
|
@return TRUE on failure, FALSE otherwise.
|
|
*/
|
|
|
|
static bool fk_info_append_fields(THD *thd, String *str,
|
|
List<LEX_CSTRING> *fields)
|
|
{
|
|
bool res= FALSE;
|
|
LEX_CSTRING *field;
|
|
List_iterator_fast<LEX_CSTRING> it(*fields);
|
|
|
|
while ((field= it++))
|
|
{
|
|
res|= append_identifier(thd, str, field);
|
|
res|= str->append(STRING_WITH_LEN(", "));
|
|
}
|
|
|
|
str->chop();
|
|
str->chop();
|
|
|
|
return res;
|
|
}
|
|
|
|
|
|
/**
|
|
Generate a foreign key description suitable for a error message.
|
|
|
|
@param thd Thread context.
|
|
@param fk_info The foreign key information.
|
|
|
|
@return A human-readable string describing the foreign key.
|
|
*/
|
|
|
|
static const char *fk_info_str(THD *thd, FOREIGN_KEY_INFO *fk_info)
|
|
{
|
|
bool res= FALSE;
|
|
char buffer[STRING_BUFFER_USUAL_SIZE*2];
|
|
String str(buffer, sizeof(buffer), system_charset_info);
|
|
|
|
str.length(0);
|
|
|
|
/*
|
|
`db`.`tbl`, CONSTRAINT `id` FOREIGN KEY (`fk`) REFERENCES `db`.`tbl` (`fk`)
|
|
*/
|
|
|
|
res|= append_identifier(thd, &str, fk_info->foreign_db);
|
|
res|= str.append('.');
|
|
res|= append_identifier(thd, &str, fk_info->foreign_table);
|
|
res|= str.append(STRING_WITH_LEN(", CONSTRAINT "));
|
|
res|= append_identifier(thd, &str, fk_info->foreign_id);
|
|
res|= str.append(STRING_WITH_LEN(" FOREIGN KEY ("));
|
|
res|= fk_info_append_fields(thd, &str, &fk_info->foreign_fields);
|
|
res|= str.append(STRING_WITH_LEN(") REFERENCES "));
|
|
res|= append_identifier(thd, &str, fk_info->referenced_db);
|
|
res|= str.append('.');
|
|
res|= append_identifier(thd, &str, fk_info->referenced_table);
|
|
res|= str.append(STRING_WITH_LEN(" ("));
|
|
res|= fk_info_append_fields(thd, &str, &fk_info->referenced_fields);
|
|
res|= str.append(')');
|
|
|
|
return res ? NULL : thd->strmake(str.ptr(), str.length());
|
|
}
|
|
|
|
|
|
/**
|
|
Check and emit a fatal error if the table which is going to be
|
|
affected by TRUNCATE TABLE is a parent table in some non-self-
|
|
referencing foreign key.
|
|
|
|
@remark The intention is to allow truncate only for tables that
|
|
are not dependent on other tables.
|
|
|
|
@param thd Thread context.
|
|
@param table Table handle.
|
|
|
|
@retval FALSE This table is not parent in a non-self-referencing foreign
|
|
key. Statement can proceed.
|
|
@retval TRUE This table is parent in a non-self-referencing foreign key,
|
|
error was emitted.
|
|
*/
|
|
|
|
static bool
|
|
fk_truncate_illegal_if_parent(THD *thd, TABLE *table)
|
|
{
|
|
FOREIGN_KEY_INFO *fk_info;
|
|
List<FOREIGN_KEY_INFO> fk_list;
|
|
List_iterator_fast<FOREIGN_KEY_INFO> it;
|
|
|
|
/*
|
|
Bail out early if the table is not referenced by a foreign key.
|
|
In this case, the table could only be, if at all, a child table.
|
|
*/
|
|
if (! table->file->referenced_by_foreign_key())
|
|
return FALSE;
|
|
|
|
/*
|
|
This table _is_ referenced by a foreign key. At this point, only
|
|
self-referencing keys are acceptable. For this reason, get the list
|
|
of foreign keys referencing this table in order to check the name
|
|
of the child (dependent) tables.
|
|
*/
|
|
table->file->get_parent_foreign_key_list(thd, &fk_list);
|
|
|
|
/* Out of memory when building list. */
|
|
if (unlikely(thd->is_error()))
|
|
return TRUE;
|
|
|
|
it.init(fk_list);
|
|
|
|
/* Loop over the set of foreign keys for which this table is a parent. */
|
|
while ((fk_info= it++))
|
|
{
|
|
if (lex_string_cmp(system_charset_info, fk_info->referenced_db,
|
|
&table->s->db) ||
|
|
lex_string_cmp(system_charset_info, fk_info->referenced_table,
|
|
&table->s->table_name) ||
|
|
lex_string_cmp(system_charset_info, fk_info->foreign_db,
|
|
&table->s->db) ||
|
|
lex_string_cmp(system_charset_info, fk_info->foreign_table,
|
|
&table->s->table_name))
|
|
break;
|
|
}
|
|
|
|
/* Table is parent in a non-self-referencing foreign key. */
|
|
if (fk_info)
|
|
{
|
|
my_error(ER_TRUNCATE_ILLEGAL_FK, MYF(0), fk_info_str(thd, fk_info));
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
/*
|
|
Open and truncate a locked table.
|
|
|
|
@param thd Thread context.
|
|
@param table_ref Table list element for the table to be truncated.
|
|
@param is_tmp_table True if element refers to a temp table.
|
|
|
|
@retval TRUNCATE_OK Truncate was successful and statement can be safely
|
|
binlogged.
|
|
@retval TRUNCATE_FAILED_BUT_BINLOG Truncate failed but still go ahead with
|
|
binlogging as in case of non transactional tables
|
|
partial truncation is possible.
|
|
|
|
@retval TRUNCATE_FAILED_SKIP_BINLOG Truncate was not successful hence donot
|
|
binlong the statement.
|
|
*/
|
|
|
|
enum Sql_cmd_truncate_table::truncate_result
|
|
Sql_cmd_truncate_table::handler_truncate(THD *thd, TABLE_LIST *table_ref,
|
|
bool is_tmp_table)
|
|
{
|
|
int error= 0;
|
|
uint flags= 0;
|
|
TABLE *table;
|
|
DBUG_ENTER("Sql_cmd_truncate_table::handler_truncate");
|
|
|
|
/*
|
|
Can't recreate, the engine must mechanically delete all rows
|
|
in the table. Use open_and_lock_tables() to open a write cursor.
|
|
*/
|
|
|
|
/* If it is a temporary table, no need to take locks. */
|
|
if (!is_tmp_table)
|
|
{
|
|
/* We don't need to load triggers. */
|
|
DBUG_ASSERT(table_ref->trg_event_map == 0);
|
|
/*
|
|
Our metadata lock guarantees that no transaction is reading
|
|
or writing into the table. Yet, to open a write cursor we need
|
|
a thr_lock lock. Allow to open base tables only.
|
|
*/
|
|
table_ref->required_type= TABLE_TYPE_NORMAL;
|
|
/*
|
|
Ignore pending FLUSH TABLES since we don't want to release
|
|
the MDL lock taken above and otherwise there is no way to
|
|
wait for FLUSH TABLES in deadlock-free fashion.
|
|
*/
|
|
flags= MYSQL_OPEN_IGNORE_FLUSH;
|
|
/*
|
|
Even though we have an MDL lock on the table here, we don't
|
|
pass MYSQL_OPEN_HAS_MDL_LOCK to open_and_lock_tables
|
|
since to truncate a MERGE table, we must open and lock
|
|
merge children, and on those we don't have an MDL lock.
|
|
Thus clear the ticket to satisfy MDL asserts.
|
|
*/
|
|
table_ref->mdl_request.ticket= NULL;
|
|
}
|
|
|
|
/* Open the table as it will handle some required preparations. */
|
|
if (open_and_lock_tables(thd, table_ref, FALSE, flags))
|
|
DBUG_RETURN(TRUNCATE_FAILED_SKIP_BINLOG);
|
|
|
|
/* Whether to truncate regardless of foreign keys. */
|
|
if (! (thd->variables.option_bits & OPTION_NO_FOREIGN_KEY_CHECKS))
|
|
if (fk_truncate_illegal_if_parent(thd, table_ref->table))
|
|
DBUG_RETURN(TRUNCATE_FAILED_SKIP_BINLOG);
|
|
|
|
table= table_ref->table;
|
|
|
|
if ((table->file->ht->flags & HTON_TRUNCATE_REQUIRES_EXCLUSIVE_USE) &&
|
|
!is_tmp_table)
|
|
{
|
|
if (wait_while_table_is_used(thd, table, HA_EXTRA_FORCE_REOPEN))
|
|
DBUG_RETURN(TRUNCATE_FAILED_SKIP_BINLOG);
|
|
/*
|
|
Get rid of all TABLE instances belonging to this thread
|
|
except one to be used for TRUNCATE
|
|
*/
|
|
close_all_tables_for_name(thd, table->s,
|
|
HA_EXTRA_NOT_USED,
|
|
table);
|
|
}
|
|
|
|
error= table->file->ha_truncate();
|
|
|
|
if (!is_tmp_table && !error)
|
|
{
|
|
backup_log_info ddl_log;
|
|
bzero(&ddl_log, sizeof(ddl_log));
|
|
ddl_log.query= { C_STRING_WITH_LEN("TRUNCATE") };
|
|
ddl_log.org_partitioned= table->file->partition_engine();
|
|
lex_string_set(&ddl_log.org_storage_engine_name,
|
|
table->file->real_table_type());
|
|
ddl_log.org_database= table->s->db;
|
|
ddl_log.org_table= table->s->table_name;
|
|
ddl_log.org_table_id= table->s->tabledef_version;
|
|
backup_log_ddl(&ddl_log);
|
|
}
|
|
|
|
if (unlikely(error))
|
|
{
|
|
table->file->print_error(error, MYF(0));
|
|
/*
|
|
If truncate method is not implemented then we don't binlog the
|
|
statement. If truncation has failed in a transactional engine then also
|
|
we don't binlog the statment. Only in non transactional engine we binlog
|
|
inspite of errors.
|
|
*/
|
|
if (error == HA_ERR_WRONG_COMMAND ||
|
|
table->file->has_transactions_and_rollback())
|
|
DBUG_RETURN(TRUNCATE_FAILED_SKIP_BINLOG);
|
|
else
|
|
DBUG_RETURN(TRUNCATE_FAILED_BUT_BINLOG);
|
|
}
|
|
DBUG_RETURN(TRUNCATE_OK);
|
|
}
|
|
|
|
|
|
/*
|
|
Handle locking a base table for truncate.
|
|
|
|
@param[in] thd Thread context.
|
|
@param[in] table_ref Table list element for the table to
|
|
be truncated.
|
|
@param[out] hton_can_recreate Set to TRUE if table can be dropped
|
|
and recreated.
|
|
|
|
@retval FALSE Success.
|
|
@retval TRUE Error.
|
|
*/
|
|
|
|
bool Sql_cmd_truncate_table::lock_table(THD *thd, TABLE_LIST *table_ref,
|
|
bool *hton_can_recreate)
|
|
{
|
|
handlerton *hton;
|
|
bool versioned;
|
|
bool sequence= false;
|
|
TABLE *table= NULL;
|
|
DBUG_ENTER("Sql_cmd_truncate_table::lock_table");
|
|
|
|
/* Lock types are set in the parser. */
|
|
DBUG_ASSERT(table_ref->lock_type == TL_WRITE);
|
|
/* The handler truncate protocol dictates a exclusive lock. */
|
|
DBUG_ASSERT(table_ref->mdl_request.type == MDL_EXCLUSIVE);
|
|
|
|
/*
|
|
Before doing anything else, acquire a metadata lock on the table,
|
|
or ensure we have one. We don't use open_and_lock_tables()
|
|
right away because we want to be able to truncate (and recreate)
|
|
corrupted tables, those that we can't fully open.
|
|
|
|
MySQL manual documents that TRUNCATE can be used to repair a
|
|
damaged table, i.e. a table that can not be fully "opened".
|
|
In particular MySQL manual says: As long as the table format
|
|
file tbl_name.frm is valid, the table can be re-created as
|
|
an empty table with TRUNCATE TABLE, even if the data or index
|
|
files have become corrupted.
|
|
*/
|
|
if (thd->locked_tables_mode)
|
|
{
|
|
if (!(table= find_table_for_mdl_upgrade(thd, table_ref->db.str,
|
|
table_ref->table_name.str, NULL)))
|
|
DBUG_RETURN(TRUE);
|
|
|
|
versioned= table->versioned();
|
|
hton= table->file->ht;
|
|
#ifdef WITH_WSREP
|
|
if (WSREP(thd) &&
|
|
!wsrep_should_replicate_ddl(thd, hton))
|
|
DBUG_RETURN(TRUE);
|
|
#endif
|
|
|
|
table_ref->mdl_request.ticket= table->mdl_ticket;
|
|
}
|
|
else
|
|
{
|
|
DBUG_ASSERT(table_ref->next_global == NULL);
|
|
if (lock_table_names(thd, table_ref, NULL,
|
|
thd->variables.lock_wait_timeout, 0))
|
|
DBUG_RETURN(TRUE);
|
|
|
|
TABLE_SHARE *share= tdc_acquire_share(thd, table_ref, GTS_TABLE | GTS_VIEW);
|
|
if (share == NULL)
|
|
DBUG_RETURN(TRUE);
|
|
DBUG_ASSERT(share != UNUSABLE_TABLE_SHARE);
|
|
|
|
versioned= share->versioned;
|
|
sequence= share->table_type == TABLE_TYPE_SEQUENCE;
|
|
hton= share->db_type();
|
|
#ifdef WITH_WSREP
|
|
if (WSREP(thd) &&
|
|
hton != view_pseudo_hton &&
|
|
!wsrep_should_replicate_ddl(thd, hton))
|
|
{
|
|
tdc_release_share(share);
|
|
DBUG_RETURN(TRUE);
|
|
}
|
|
#endif
|
|
|
|
if (!versioned)
|
|
tdc_remove_referenced_share(thd, share);
|
|
else
|
|
tdc_release_share(share);
|
|
|
|
if (hton == view_pseudo_hton)
|
|
{
|
|
my_error(ER_NO_SUCH_TABLE, MYF(0), table_ref->db.str,
|
|
table_ref->table_name.str);
|
|
DBUG_RETURN(TRUE);
|
|
}
|
|
}
|
|
|
|
*hton_can_recreate= (!sequence &&
|
|
ha_check_storage_engine_flag(hton, HTON_CAN_RECREATE));
|
|
|
|
if (versioned)
|
|
{
|
|
my_error(ER_VERS_NOT_SUPPORTED, MYF(0), "TRUNCATE TABLE");
|
|
DBUG_RETURN(TRUE);
|
|
}
|
|
|
|
/*
|
|
A storage engine can recreate or truncate the table only if there
|
|
are no references to it from anywhere, i.e. no cached TABLE in the
|
|
table cache.
|
|
*/
|
|
if (thd->locked_tables_mode)
|
|
{
|
|
DEBUG_SYNC(thd, "upgrade_lock_for_truncate");
|
|
/* To remove the table from the cache we need an exclusive lock. */
|
|
if (wait_while_table_is_used(thd, table,
|
|
*hton_can_recreate ? HA_EXTRA_PREPARE_FOR_DROP : HA_EXTRA_NOT_USED))
|
|
DBUG_RETURN(TRUE);
|
|
m_ticket_downgrade= table->mdl_ticket;
|
|
/* Close if table is going to be recreated. */
|
|
if (*hton_can_recreate)
|
|
close_all_tables_for_name(thd, table->s, HA_EXTRA_NOT_USED, NULL);
|
|
}
|
|
DBUG_RETURN(FALSE);
|
|
}
|
|
|
|
|
|
/*
|
|
Optimized delete of all rows by doing a full generate of the table.
|
|
|
|
@remark Will work even if the .MYI and .MYD files are destroyed.
|
|
In other words, it works as long as the .FRM is intact and
|
|
the engine supports re-create.
|
|
|
|
@param thd Thread context.
|
|
@param table_ref Table list element for the table to be truncated.
|
|
|
|
@retval FALSE Success.
|
|
@retval TRUE Error.
|
|
*/
|
|
|
|
bool Sql_cmd_truncate_table::truncate_table(THD *thd, TABLE_LIST *table_ref)
|
|
{
|
|
int error;
|
|
bool binlog_stmt;
|
|
DBUG_ENTER("Sql_cmd_truncate_table::truncate_table");
|
|
|
|
DBUG_ASSERT((!table_ref->table) ||
|
|
(table_ref->table && table_ref->table->s));
|
|
|
|
/* Initialize, or reinitialize in case of reexecution (SP). */
|
|
m_ticket_downgrade= NULL;
|
|
|
|
/* If it is a temporary table, no need to take locks. */
|
|
if (is_temporary_table(table_ref))
|
|
{
|
|
/* In RBR, the statement is not binlogged if the table is temporary. */
|
|
binlog_stmt= !thd->is_current_stmt_binlog_format_row();
|
|
|
|
thd->close_unused_temporary_table_instances(table_ref);
|
|
|
|
error= handler_truncate(thd, table_ref, TRUE);
|
|
|
|
/*
|
|
No need to invalidate the query cache, queries with temporary
|
|
tables are not in the cache. No need to write to the binary
|
|
log a failed row-by-row delete even if under RBR as the table
|
|
might not exist on the slave.
|
|
*/
|
|
}
|
|
else /* It's not a temporary table. */
|
|
{
|
|
bool hton_can_recreate;
|
|
|
|
#ifdef WITH_WSREP
|
|
if (WSREP(thd) && wsrep_thd_is_local(thd))
|
|
{
|
|
wsrep::key_array keys;
|
|
/* Do not start TOI if table is not found */
|
|
if (!wsrep_append_fk_parent_table(thd, table_ref, &keys))
|
|
{
|
|
if (keys.empty())
|
|
{
|
|
if (wsrep_to_isolation_begin(thd, table_ref->db.str, table_ref->table_name.str, NULL))
|
|
DBUG_RETURN(TRUE);
|
|
}
|
|
else
|
|
{
|
|
if (wsrep_to_isolation_begin(thd, NULL, NULL, table_ref, NULL, &keys))
|
|
DBUG_RETURN(TRUE);
|
|
}
|
|
}
|
|
}
|
|
#endif /* WITH_WSREP */
|
|
|
|
if (lock_table(thd, table_ref, &hton_can_recreate))
|
|
DBUG_RETURN(TRUE);
|
|
|
|
if (hton_can_recreate)
|
|
{
|
|
/*
|
|
The storage engine can truncate the table by creating an
|
|
empty table with the same structure.
|
|
*/
|
|
error= dd_recreate_table(thd, table_ref->db.str, table_ref->table_name.str);
|
|
|
|
if (thd->locked_tables_mode && thd->locked_tables_list.reopen_tables(thd, false))
|
|
{
|
|
thd->locked_tables_list.unlink_all_closed_tables(thd, NULL, 0);
|
|
error= 1;
|
|
}
|
|
/* No need to binlog a failed truncate-by-recreate. */
|
|
binlog_stmt= !error;
|
|
}
|
|
else
|
|
{
|
|
/*
|
|
The engine does not support truncate-by-recreate.
|
|
Attempt to use the handler truncate method.
|
|
*/
|
|
error= handler_truncate(thd, table_ref, FALSE);
|
|
|
|
if (error == TRUNCATE_OK && thd->locked_tables_mode &&
|
|
(table_ref->table->file->ht->flags &
|
|
(HTON_REQUIRES_CLOSE_AFTER_TRUNCATE |
|
|
HTON_TRUNCATE_REQUIRES_EXCLUSIVE_USE)))
|
|
{
|
|
thd->locked_tables_list.mark_table_for_reopen(thd, table_ref->table);
|
|
if (unlikely(thd->locked_tables_list.reopen_tables(thd, false)))
|
|
thd->locked_tables_list.unlink_all_closed_tables(thd, NULL, 0);
|
|
}
|
|
|
|
/*
|
|
All effects of a TRUNCATE TABLE operation are committed even if
|
|
truncation fails in the case of non transactional tables. Thus, the
|
|
query must be written to the binary log. The only exception is a
|
|
unimplemented truncate method.
|
|
*/
|
|
if (unlikely(error == TRUNCATE_OK || error == TRUNCATE_FAILED_BUT_BINLOG))
|
|
binlog_stmt= true;
|
|
else
|
|
binlog_stmt= false;
|
|
}
|
|
|
|
/*
|
|
If we tried to open a MERGE table and failed due to problems with the
|
|
children tables, the table will have been closed and table_ref->table
|
|
will be invalid. Reset the pointer here in any case as
|
|
query_cache_invalidate does not need a valid TABLE object.
|
|
*/
|
|
table_ref->table= NULL;
|
|
query_cache_invalidate3(thd, table_ref, FALSE);
|
|
}
|
|
|
|
/* DDL is logged in statement format, regardless of binlog format. */
|
|
if (binlog_stmt)
|
|
error|= write_bin_log(thd, !error, thd->query(), thd->query_length());
|
|
|
|
/*
|
|
A locked table ticket was upgraded to a exclusive lock. After the
|
|
the query has been written to the binary log, downgrade the lock
|
|
to a shared one.
|
|
*/
|
|
if (m_ticket_downgrade)
|
|
m_ticket_downgrade->downgrade_lock(MDL_SHARED_NO_READ_WRITE);
|
|
|
|
DBUG_RETURN(error);
|
|
}
|
|
|
|
/**
|
|
Execute a TRUNCATE statement at runtime.
|
|
|
|
@param thd The current thread.
|
|
|
|
@return FALSE on success.
|
|
*/
|
|
|
|
bool Sql_cmd_truncate_table::execute(THD *thd)
|
|
{
|
|
bool res= TRUE;
|
|
TABLE_LIST *table= thd->lex->first_select_lex()->table_list.first;
|
|
DBUG_ENTER("Sql_cmd_truncate_table::execute");
|
|
|
|
if (check_one_table_access(thd, DROP_ACL, table))
|
|
DBUG_RETURN(res);
|
|
|
|
if (! (res= truncate_table(thd, table)))
|
|
my_ok(thd);
|
|
|
|
DBUG_RETURN(res);
|
|
}
|