mysql5/mysql-5.7.27/sql/rpl_reporting.cc

178 lines
5.7 KiB
C++

/* Copyright (c) 2007, 2015, 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 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,
51 Franklin Street, Suite 500, Boston, MA 02110-1335 USA */
#include "rpl_reporting.h"
#include "log.h" // sql_print_warning
#include "mysqld.h" // current_thd
#include "sql_class.h" // THD
#include "sql_error.h" // Diagnostics_area
Slave_reporting_capability::Slave_reporting_capability(char const *thread_name)
: m_thread_name(thread_name)
{
mysql_mutex_init(key_mutex_slave_reporting_capability_err_lock,
&err_lock, MY_MUTEX_INIT_FAST);
}
#if !defined(EMBEDDED_LIBRARY)
/**
Check if the current error is of temporary nature or not.
Some errors are temporary in nature, such as
ER_LOCK_DEADLOCK and ER_LOCK_WAIT_TIMEOUT. Ndb also signals
that the error is temporary by pushing a warning with the error code
ER_GET_TEMPORARY_ERRMSG, if the originating error is temporary.
@param thd a THD instance, typically of the slave SQL thread's.
@error_arg the error code for assessment.
defaults to zero which makes the function check the top
of the reported errors stack.
@silent bool indicating whether the error should be silently handled.
@return 1 as the positive and 0 as the negative verdict
*/
int Slave_reporting_capability::has_temporary_error(THD *thd,
uint error_arg, bool* silent) const
{
uint error;
DBUG_ENTER("has_temporary_error");
DBUG_EXECUTE_IF("all_errors_are_temporary_errors",
if (thd->get_stmt_da()->is_error())
{
thd->clear_error();
my_error(ER_LOCK_DEADLOCK, MYF(0));
});
/*
The slave can't be regarded as experiencing a temporary failure in cases of
is_fatal_error is TRUE, or if no error is in THD and error_arg is not set.
*/
if (thd->is_fatal_error || (!thd->is_error() && error_arg == 0))
DBUG_RETURN(0);
error= (error_arg == 0)? thd->get_stmt_da()->mysql_errno() : error_arg;
/*
Temporary error codes:
currently, InnoDB deadlock detected by InnoDB or lock
wait timeout (innodb_lock_wait_timeout exceeded).
Notice, the temporary error requires slave_trans_retries != 0)
*/
if (slave_trans_retries &&
(error == ER_LOCK_DEADLOCK || error == ER_LOCK_WAIT_TIMEOUT))
DBUG_RETURN(1);
/*
currently temporary error set in ndbcluster
*/
Diagnostics_area::Sql_condition_iterator it=
thd->get_stmt_da()->sql_conditions();
const Sql_condition *err;
while ((err= it++))
{
DBUG_PRINT("info", ("has condition %d %s", err->mysql_errno(),
err->message_text()));
switch (err->mysql_errno())
{
case ER_GET_TEMPORARY_ERRMSG:
DBUG_RETURN(1);
case ER_SLAVE_SILENT_RETRY_TRANSACTION:
{
if (silent != NULL)
*silent= true;
DBUG_RETURN(1);
}
default:
break;
}
}
DBUG_RETURN(0);
}
#endif // EMBEDDED_LIBRARY
void
Slave_reporting_capability::report(loglevel level, int err_code,
const char *msg, ...) const
{
va_list args;
va_start(args, msg);
do_report(level, err_code, msg, args);
va_end(args);
}
void
Slave_reporting_capability::va_report(loglevel level, int err_code,
const char *prefix_msg,
const char *msg, va_list args) const
{
#if !defined(EMBEDDED_LIBRARY)
THD *thd= current_thd;
void (*report_function)(const char *, ...);
char buff[MAX_SLAVE_ERRMSG];
char *pbuff= buff;
char *curr_buff;
uint pbuffsize= sizeof(buff);
if (thd && level == ERROR_LEVEL && has_temporary_error(thd, err_code) &&
!thd->get_transaction()->cannot_safely_rollback(
Transaction_ctx::SESSION))
level= WARNING_LEVEL;
mysql_mutex_lock(&err_lock);
switch (level)
{
case ERROR_LEVEL:
/*
It's an error, it must be reported in Last_error and Last_errno in SHOW
SLAVE STATUS.
*/
pbuff= m_last_error.message;
pbuffsize= sizeof(m_last_error.message);
m_last_error.number = err_code;
m_last_error.update_timestamp();
report_function= sql_print_error;
break;
case WARNING_LEVEL:
report_function= sql_print_warning;
break;
case INFORMATION_LEVEL:
report_function= sql_print_information;
break;
default:
DBUG_ASSERT(0); // should not come here
return; // don't crash production builds, just do nothing
}
curr_buff= pbuff;
if (prefix_msg)
curr_buff += my_snprintf(curr_buff, pbuffsize, "%s; ", prefix_msg);
my_vsnprintf(curr_buff, pbuffsize - (curr_buff - pbuff), msg, args);
mysql_mutex_unlock(&err_lock);
/* If the msg string ends with '.', do not add a ',' it would be ugly */
report_function("Slave %s%s: %s%s Error_code: %d",
m_thread_name, get_for_channel_str(false), pbuff,
(curr_buff[0] && *(strend(curr_buff)-1) == '.') ? "" : ",",
err_code);
#endif
}
Slave_reporting_capability::~Slave_reporting_capability()
{
mysql_mutex_destroy(&err_lock);
}