2017-04-18 18:43:20 +00:00
|
|
|
/******************************************************
|
|
|
|
hot backup tool for InnoDB
|
|
|
|
(c) 2009-2015 Percona LLC and/or its affiliates
|
|
|
|
Originally Created 3/3/2009 Yasufumi Kinoshita
|
|
|
|
Written by Alexey Kopytov, Aleksandr Kuzminsky, Stewart Smith, Vadim Tkachenko,
|
|
|
|
Yasufumi Kinoshita, Ignacio Nin and Baron Schwartz.
|
|
|
|
|
|
|
|
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
|
2019-05-11 22:19:05 +03:00
|
|
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1335 USA
|
2017-04-18 18:43:20 +00:00
|
|
|
|
|
|
|
*******************************************************
|
|
|
|
|
|
|
|
This file incorporates work covered by the following copyright and
|
|
|
|
permission notice:
|
|
|
|
|
|
|
|
Copyright (c) 2000, 2011, MySQL AB & Innobase Oy. 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
|
2019-05-11 19:25:02 +03:00
|
|
|
this program; if not, write to the Free Software Foundation, Inc., 51 Franklin
|
|
|
|
Street, Fifth Floor, Boston, MA 02110-1335 USA
|
2017-04-18 18:43:20 +00:00
|
|
|
|
|
|
|
*******************************************************/
|
2017-04-18 19:05:57 +00:00
|
|
|
#define MYSQL_CLIENT
|
2017-04-18 18:43:20 +00:00
|
|
|
|
|
|
|
#include <my_global.h>
|
|
|
|
#include <mysql.h>
|
|
|
|
#include <mysqld.h>
|
|
|
|
#include <my_sys.h>
|
2021-04-11 17:07:36 +02:00
|
|
|
#include <stdlib.h>
|
2017-04-18 18:43:20 +00:00
|
|
|
#include <string.h>
|
|
|
|
#include <limits>
|
|
|
|
#include "common.h"
|
|
|
|
#include "xtrabackup.h"
|
2018-04-29 09:41:42 +03:00
|
|
|
#include "srv0srv.h"
|
2017-04-18 19:05:57 +00:00
|
|
|
#include "mysql_version.h"
|
2017-04-18 18:43:20 +00:00
|
|
|
#include "backup_copy.h"
|
|
|
|
#include "backup_mysql.h"
|
|
|
|
#include "mysqld.h"
|
2021-10-22 10:14:57 +02:00
|
|
|
#include "xb_plugin.h"
|
2017-04-18 19:05:57 +00:00
|
|
|
#include <sstream>
|
MDEV-12548 Initial implementation of Mariabackup for MariaDB 10.2
InnoDB I/O and buffer pool interfaces and the redo log format
have been changed between MariaDB 10.1 and 10.2, and the backup
code has to be adjusted accordingly.
The code has been simplified, and many memory leaks have been fixed.
Instead of the file name xtrabackup_logfile, the file name ib_logfile0
is being used for the copy of the redo log. Unnecessary InnoDB startup and
shutdown and some unnecessary threads have been removed.
Some help was provided by Vladislav Vaintroub.
Parameters have been cleaned up and aligned with those of MariaDB 10.2.
The --dbug option has been added, so that in debug builds,
--dbug=d,ib_log can be specified to enable diagnostic messages
for processing redo log entries.
By default, innodb_doublewrite=OFF, so that --prepare works faster.
If more crash-safety for --prepare is needed, double buffering
can be enabled.
The parameter innodb_log_checksums=OFF can be used to ignore redo log
checksums in --backup.
Some messages have been cleaned up.
Unless --export is specified, Mariabackup will not deal with undo log.
The InnoDB mini-transaction redo log is not only about user-level
transactions; it is actually about mini-transactions. To avoid confusion,
call it the redo log, not transaction log.
We disable any undo log processing in --prepare.
Because MariaDB 10.2 supports indexed virtual columns, the
undo log processing would need to be able to evaluate virtual column
expressions. To reduce the amount of code dependencies, we will not
process any undo log in prepare.
This means that the --export option must be disabled for now.
This also means that the following options are redundant
and have been removed:
xtrabackup --apply-log-only
innobackupex --redo-only
In addition to disabling any undo log processing, we will disable any
further changes to data pages during --prepare, including the change
buffer merge. This means that restoring incremental backups should
reliably work even when change buffering is being used on the server.
Because of this, preparing a backup will not generate any further
redo log, and the redo log file can be safely deleted. (If the
--export option is enabled in the future, it must generate redo log
when processing undo logs and buffered changes.)
In --prepare, we cannot easily know if a partial backup was used,
especially when restoring a series of incremental backups. So, we
simply warn about any missing files, and ignore the redo log for them.
FIXME: Enable the --export option.
FIXME: Improve the handling of the MLOG_INDEX_LOAD record, and write
a test that initiates a backup while an ALGORITHM=INPLACE operation
is creating indexes or rebuilding a table. An error should be detected
when preparing the backup.
FIXME: In --incremental --prepare, xtrabackup_apply_delta() should
ensure that if FSP_SIZE is modified, the file size will be adjusted
accordingly.
2017-06-30 10:49:37 +03:00
|
|
|
#include <sql_error.h>
|
2019-03-08 16:00:08 +05:30
|
|
|
#include "page0zip.h"
|
2017-04-18 18:43:20 +00:00
|
|
|
|
|
|
|
char *tool_name;
|
|
|
|
char tool_args[2048];
|
|
|
|
|
2022-05-25 09:20:04 +03:00
|
|
|
ulong mysql_server_version;
|
2017-04-18 18:43:20 +00:00
|
|
|
|
|
|
|
/* server capabilities */
|
|
|
|
bool have_changed_page_bitmaps = false;
|
|
|
|
bool have_backup_locks = false;
|
|
|
|
bool have_lock_wait_timeout = false;
|
|
|
|
bool have_galera_enabled = false;
|
|
|
|
bool have_multi_threaded_slave = false;
|
|
|
|
bool have_gtid_slave = false;
|
|
|
|
|
|
|
|
/* Kill long selects */
|
2020-12-04 16:18:04 +02:00
|
|
|
static mysql_mutex_t kill_query_thread_mutex;
|
|
|
|
static bool kill_query_thread_running, kill_query_thread_stopping;
|
|
|
|
static mysql_cond_t kill_query_thread_stopped;
|
|
|
|
static mysql_cond_t kill_query_thread_stop;
|
2017-04-18 18:43:20 +00:00
|
|
|
|
|
|
|
bool sql_thread_started = false;
|
|
|
|
char *mysql_slave_position = NULL;
|
|
|
|
char *mysql_binlog_position = NULL;
|
2022-06-28 09:16:31 +02:00
|
|
|
char *buffer_pool_filename = NULL;
|
2017-04-18 18:43:20 +00:00
|
|
|
|
|
|
|
/* History on server */
|
|
|
|
time_t history_start_time;
|
|
|
|
time_t history_end_time;
|
|
|
|
time_t history_lock_time;
|
|
|
|
|
|
|
|
MYSQL *mysql_connection;
|
|
|
|
|
2018-05-08 15:08:08 +01:00
|
|
|
extern my_bool opt_ssl_verify_server_cert, opt_use_ssl;
|
2017-04-18 18:43:20 +00:00
|
|
|
|
|
|
|
MYSQL *
|
|
|
|
xb_mysql_connect()
|
|
|
|
{
|
|
|
|
MYSQL *connection = mysql_init(NULL);
|
|
|
|
char mysql_port_str[std::numeric_limits<int>::digits10 + 3];
|
|
|
|
|
|
|
|
sprintf(mysql_port_str, "%d", opt_port);
|
|
|
|
|
|
|
|
if (connection == NULL) {
|
2021-03-18 13:17:30 +05:30
|
|
|
msg("Failed to init MariaDB struct: %s.",
|
2017-04-18 18:43:20 +00:00
|
|
|
mysql_error(connection));
|
|
|
|
return(NULL);
|
|
|
|
}
|
|
|
|
|
2021-04-11 17:07:36 +02:00
|
|
|
#if !defined(DONT_USE_MYSQL_PWD)
|
|
|
|
if (!opt_password)
|
|
|
|
{
|
|
|
|
opt_password=getenv("MYSQL_PWD");
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2017-04-18 18:43:20 +00:00
|
|
|
if (!opt_secure_auth) {
|
|
|
|
mysql_options(connection, MYSQL_SECURE_AUTH,
|
|
|
|
(char *) &opt_secure_auth);
|
|
|
|
}
|
|
|
|
|
2017-08-21 17:16:12 +00:00
|
|
|
if (xb_plugin_dir && *xb_plugin_dir){
|
|
|
|
mysql_options(connection, MYSQL_PLUGIN_DIR, xb_plugin_dir);
|
|
|
|
}
|
|
|
|
mysql_options(connection, MYSQL_OPT_PROTOCOL, &opt_protocol);
|
2017-09-11 16:45:36 +00:00
|
|
|
mysql_options(connection,MYSQL_SET_CHARSET_NAME, "utf8");
|
2017-08-21 17:16:12 +00:00
|
|
|
|
2021-03-18 13:17:30 +05:30
|
|
|
msg("Connecting to MariaDB server host: %s, user: %s, password: %s, "
|
2019-01-14 22:28:23 +01:00
|
|
|
"port: %s, socket: %s", opt_host ? opt_host : "localhost",
|
2017-04-18 18:43:20 +00:00
|
|
|
opt_user ? opt_user : "not set",
|
|
|
|
opt_password ? "set" : "not set",
|
|
|
|
opt_port != 0 ? mysql_port_str : "not set",
|
|
|
|
opt_socket ? opt_socket : "not set");
|
|
|
|
|
|
|
|
#ifdef HAVE_OPENSSL
|
2022-06-17 21:20:43 +02:00
|
|
|
if (opt_use_ssl && opt_protocol <= MYSQL_PROTOCOL_SOCKET)
|
2017-04-18 18:43:20 +00:00
|
|
|
{
|
|
|
|
mysql_ssl_set(connection, opt_ssl_key, opt_ssl_cert,
|
|
|
|
opt_ssl_ca, opt_ssl_capath,
|
|
|
|
opt_ssl_cipher);
|
|
|
|
mysql_options(connection, MYSQL_OPT_SSL_CRL, opt_ssl_crl);
|
|
|
|
mysql_options(connection, MYSQL_OPT_SSL_CRLPATH,
|
|
|
|
opt_ssl_crlpath);
|
|
|
|
}
|
|
|
|
mysql_options(connection,MYSQL_OPT_SSL_VERIFY_SERVER_CERT,
|
|
|
|
(char*)&opt_ssl_verify_server_cert);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (!mysql_real_connect(connection,
|
|
|
|
opt_host ? opt_host : "localhost",
|
|
|
|
opt_user,
|
|
|
|
opt_password,
|
|
|
|
"" /*database*/, opt_port,
|
|
|
|
opt_socket, 0)) {
|
2021-03-18 13:17:30 +05:30
|
|
|
msg("Failed to connect to MariaDB server: %s.", mysql_error(connection));
|
2017-04-18 18:43:20 +00:00
|
|
|
mysql_close(connection);
|
|
|
|
return(NULL);
|
|
|
|
}
|
|
|
|
|
2019-01-29 13:15:59 +01:00
|
|
|
xb_mysql_query(connection, "SET SESSION wait_timeout=2147483, max_statement_time=0",
|
2017-04-18 18:43:20 +00:00
|
|
|
false, true);
|
|
|
|
|
|
|
|
return(connection);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*********************************************************************//**
|
|
|
|
Execute mysql query. */
|
|
|
|
MYSQL_RES *
|
|
|
|
xb_mysql_query(MYSQL *connection, const char *query, bool use_result,
|
|
|
|
bool die_on_error)
|
|
|
|
{
|
|
|
|
MYSQL_RES *mysql_result = NULL;
|
|
|
|
|
|
|
|
if (mysql_query(connection, query)) {
|
|
|
|
if (die_on_error) {
|
2019-01-15 22:47:54 +01:00
|
|
|
die("failed to execute query %s: %s", query, mysql_error(connection));
|
|
|
|
} else {
|
|
|
|
msg("Error: failed to execute query %s: %s", query, mysql_error(connection));
|
2017-04-18 18:43:20 +00:00
|
|
|
}
|
|
|
|
return(NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* store result set on client if there is a result */
|
|
|
|
if (mysql_field_count(connection) > 0) {
|
|
|
|
if ((mysql_result = mysql_store_result(connection)) == NULL) {
|
2019-01-15 22:47:54 +01:00
|
|
|
die("failed to fetch query result %s: %s",
|
2017-04-18 18:43:20 +00:00
|
|
|
query, mysql_error(connection));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!use_result) {
|
|
|
|
mysql_free_result(mysql_result);
|
2018-06-22 15:24:09 +01:00
|
|
|
mysql_result = NULL;
|
2017-04-18 18:43:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return mysql_result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
struct mysql_variable {
|
|
|
|
const char *name;
|
|
|
|
char **value;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
static
|
|
|
|
void
|
|
|
|
read_mysql_variables(MYSQL *connection, const char *query, mysql_variable *vars,
|
|
|
|
bool vertical_result)
|
|
|
|
{
|
|
|
|
MYSQL_RES *mysql_result;
|
|
|
|
MYSQL_ROW row;
|
|
|
|
mysql_variable *var;
|
|
|
|
|
|
|
|
mysql_result = xb_mysql_query(connection, query, true);
|
|
|
|
|
|
|
|
ut_ad(!vertical_result || mysql_num_fields(mysql_result) == 2);
|
|
|
|
|
|
|
|
if (vertical_result) {
|
|
|
|
while ((row = mysql_fetch_row(mysql_result))) {
|
|
|
|
char *name = row[0];
|
|
|
|
char *value = row[1];
|
|
|
|
for (var = vars; var->name; var++) {
|
|
|
|
if (strcmp(var->name, name) == 0
|
|
|
|
&& value != NULL) {
|
|
|
|
*(var->value) = strdup(value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
MYSQL_FIELD *field;
|
|
|
|
|
|
|
|
if ((row = mysql_fetch_row(mysql_result)) != NULL) {
|
|
|
|
int i = 0;
|
|
|
|
while ((field = mysql_fetch_field(mysql_result))
|
|
|
|
!= NULL) {
|
|
|
|
char *name = field->name;
|
|
|
|
char *value = row[i];
|
|
|
|
for (var = vars; var->name; var++) {
|
|
|
|
if (strcmp(var->name, name) == 0
|
|
|
|
&& value != NULL) {
|
|
|
|
*(var->value) = strdup(value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
++i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
mysql_free_result(mysql_result);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static
|
|
|
|
void
|
|
|
|
free_mysql_variables(mysql_variable *vars)
|
|
|
|
{
|
|
|
|
mysql_variable *var;
|
|
|
|
|
|
|
|
for (var = vars; var->name; var++) {
|
|
|
|
free(*(var->value));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static
|
|
|
|
char *
|
2022-04-18 19:43:25 +04:00
|
|
|
read_mysql_one_value(MYSQL *connection, const char *query,
|
|
|
|
uint column, uint expect_columns)
|
2017-04-18 18:43:20 +00:00
|
|
|
{
|
|
|
|
MYSQL_RES *mysql_result;
|
|
|
|
MYSQL_ROW row;
|
|
|
|
char *result = NULL;
|
|
|
|
|
|
|
|
mysql_result = xb_mysql_query(connection, query, true);
|
|
|
|
|
2022-04-18 19:43:25 +04:00
|
|
|
ut_ad(mysql_num_fields(mysql_result) == expect_columns);
|
2017-04-18 18:43:20 +00:00
|
|
|
|
|
|
|
if ((row = mysql_fetch_row(mysql_result))) {
|
2022-04-18 19:43:25 +04:00
|
|
|
result = strdup(row[column]);
|
2017-04-18 18:43:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
mysql_free_result(mysql_result);
|
|
|
|
|
|
|
|
return(result);
|
|
|
|
}
|
|
|
|
|
2022-04-18 19:43:25 +04:00
|
|
|
|
|
|
|
static
|
|
|
|
char *
|
|
|
|
read_mysql_one_value(MYSQL *mysql, const char *query)
|
|
|
|
{
|
|
|
|
return read_mysql_one_value(mysql, query, 0/*offset*/, 1/*total columns*/);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-04-18 18:43:20 +00:00
|
|
|
static
|
|
|
|
bool
|
2022-05-25 09:20:04 +03:00
|
|
|
check_server_version(ulong version_number, const char *version_string)
|
2017-04-18 18:43:20 +00:00
|
|
|
{
|
2022-05-25 09:20:04 +03:00
|
|
|
if (strstr(version_string, "MariaDB") && version_number >= 100800)
|
|
|
|
return true;
|
2017-04-18 18:43:20 +00:00
|
|
|
|
2022-05-25 09:20:04 +03:00
|
|
|
msg("Error: Unsupported server version: '%s'.", version_string);
|
|
|
|
return false;
|
2017-04-18 18:43:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*********************************************************************//**
|
2022-08-26 10:20:26 +10:00
|
|
|
Receive options important for XtraBackup from server.
|
2017-04-18 18:43:20 +00:00
|
|
|
@return true on success. */
|
2020-04-20 17:23:43 +03:00
|
|
|
bool get_mysql_vars(MYSQL *connection)
|
2017-04-18 18:43:20 +00:00
|
|
|
{
|
2020-04-20 17:23:43 +03:00
|
|
|
char *gtid_mode_var= NULL;
|
|
|
|
char *version_var= NULL;
|
|
|
|
char *have_backup_locks_var= NULL;
|
|
|
|
char *log_bin_var= NULL;
|
|
|
|
char *lock_wait_timeout_var= NULL;
|
|
|
|
char *wsrep_on_var= NULL;
|
|
|
|
char *slave_parallel_workers_var= NULL;
|
|
|
|
char *gtid_slave_pos_var= NULL;
|
|
|
|
char *innodb_buffer_pool_filename_var= NULL;
|
|
|
|
char *datadir_var= NULL;
|
|
|
|
char *innodb_log_group_home_dir_var= NULL;
|
|
|
|
char *innodb_log_file_size_var= NULL;
|
|
|
|
char *innodb_log_files_in_group_var= NULL;
|
|
|
|
char *innodb_data_file_path_var= NULL;
|
|
|
|
char *innodb_data_home_dir_var= NULL;
|
|
|
|
char *innodb_undo_directory_var= NULL;
|
|
|
|
char *innodb_page_size_var= NULL;
|
|
|
|
char *innodb_undo_tablespaces_var= NULL;
|
2023-04-13 15:42:53 +04:00
|
|
|
char *aria_log_dir_path_var= NULL;
|
2020-04-20 17:23:43 +03:00
|
|
|
char *page_zip_level_var= NULL;
|
|
|
|
char *ignore_db_dirs= NULL;
|
|
|
|
char *endptr;
|
2022-05-25 09:20:04 +03:00
|
|
|
ulong server_version= mysql_get_server_version(connection);
|
2020-04-20 17:23:43 +03:00
|
|
|
|
|
|
|
bool ret= true;
|
|
|
|
|
|
|
|
mysql_variable mysql_vars[]= {
|
|
|
|
{"have_backup_locks", &have_backup_locks_var},
|
|
|
|
{"log_bin", &log_bin_var},
|
|
|
|
{"lock_wait_timeout", &lock_wait_timeout_var},
|
|
|
|
{"gtid_mode", >id_mode_var},
|
|
|
|
{"version", &version_var},
|
|
|
|
{"wsrep_on", &wsrep_on_var},
|
|
|
|
{"slave_parallel_workers", &slave_parallel_workers_var},
|
|
|
|
{"gtid_slave_pos", >id_slave_pos_var},
|
|
|
|
{"innodb_buffer_pool_filename", &innodb_buffer_pool_filename_var},
|
|
|
|
{"datadir", &datadir_var},
|
|
|
|
{"innodb_log_group_home_dir", &innodb_log_group_home_dir_var},
|
|
|
|
{"innodb_log_file_size", &innodb_log_file_size_var},
|
|
|
|
{"innodb_log_files_in_group", &innodb_log_files_in_group_var},
|
|
|
|
{"innodb_data_file_path", &innodb_data_file_path_var},
|
|
|
|
{"innodb_data_home_dir", &innodb_data_home_dir_var},
|
|
|
|
{"innodb_undo_directory", &innodb_undo_directory_var},
|
|
|
|
{"innodb_page_size", &innodb_page_size_var},
|
|
|
|
{"innodb_undo_tablespaces", &innodb_undo_tablespaces_var},
|
|
|
|
{"innodb_compression_level", &page_zip_level_var},
|
|
|
|
{"ignore_db_dirs", &ignore_db_dirs},
|
2023-04-13 15:42:53 +04:00
|
|
|
{"aria_log_dir_path", &aria_log_dir_path_var},
|
2020-04-20 17:23:43 +03:00
|
|
|
{NULL, NULL}};
|
|
|
|
|
|
|
|
read_mysql_variables(connection, "SHOW VARIABLES", mysql_vars, true);
|
|
|
|
|
|
|
|
if (have_backup_locks_var != NULL && !opt_no_backup_locks)
|
|
|
|
{
|
|
|
|
have_backup_locks= true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (opt_binlog_info == BINLOG_INFO_AUTO)
|
|
|
|
{
|
|
|
|
if (log_bin_var != NULL && !strcmp(log_bin_var, "ON"))
|
|
|
|
opt_binlog_info= BINLOG_INFO_ON;
|
|
|
|
else
|
|
|
|
opt_binlog_info= BINLOG_INFO_OFF;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (lock_wait_timeout_var != NULL)
|
|
|
|
{
|
|
|
|
have_lock_wait_timeout= true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (wsrep_on_var != NULL)
|
|
|
|
{
|
|
|
|
have_galera_enabled= true;
|
|
|
|
}
|
|
|
|
|
2022-05-25 09:20:04 +03:00
|
|
|
/* Check server version compatibility */
|
|
|
|
if (!(ret= check_server_version(server_version, version_var)))
|
2020-04-20 17:23:43 +03:00
|
|
|
{
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2022-05-25 09:20:04 +03:00
|
|
|
mysql_server_version= server_version;
|
2020-04-20 17:23:43 +03:00
|
|
|
|
|
|
|
if (slave_parallel_workers_var != NULL &&
|
|
|
|
atoi(slave_parallel_workers_var) > 0)
|
|
|
|
{
|
|
|
|
have_multi_threaded_slave= true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (innodb_buffer_pool_filename_var != NULL)
|
|
|
|
{
|
|
|
|
buffer_pool_filename= strdup(innodb_buffer_pool_filename_var);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((gtid_mode_var && strcmp(gtid_mode_var, "ON") == 0) ||
|
|
|
|
(gtid_slave_pos_var && *gtid_slave_pos_var))
|
|
|
|
{
|
|
|
|
have_gtid_slave= true;
|
|
|
|
}
|
|
|
|
|
|
|
|
msg("Using server version %s", version_var);
|
|
|
|
|
|
|
|
if (!(ret= detect_mysql_capabilities_for_backup()))
|
|
|
|
{
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* make sure datadir value is the same in configuration file */
|
|
|
|
if (check_if_param_set("datadir"))
|
|
|
|
{
|
|
|
|
if (!directory_exists(mysql_data_home, false))
|
|
|
|
{
|
|
|
|
msg("Warning: option 'datadir' points to "
|
|
|
|
"nonexistent directory '%s'",
|
|
|
|
mysql_data_home);
|
|
|
|
}
|
|
|
|
if (!directory_exists(datadir_var, false))
|
|
|
|
{
|
2021-03-18 13:17:30 +05:30
|
|
|
msg("Warning: MariaDB variable 'datadir' points to "
|
2020-04-20 17:23:43 +03:00
|
|
|
"nonexistent directory '%s'",
|
|
|
|
datadir_var);
|
|
|
|
}
|
|
|
|
if (!equal_paths(mysql_data_home, datadir_var))
|
|
|
|
{
|
|
|
|
msg("Warning: option 'datadir' has different "
|
|
|
|
"values:\n"
|
|
|
|
" '%s' in defaults file\n"
|
|
|
|
" '%s' in SHOW VARIABLES",
|
|
|
|
mysql_data_home, datadir_var);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* get some default values is they are missing from my.cnf */
|
|
|
|
if (datadir_var && *datadir_var)
|
|
|
|
{
|
|
|
|
strmake(mysql_real_data_home, datadir_var, FN_REFLEN - 1);
|
|
|
|
mysql_data_home= mysql_real_data_home;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (innodb_data_file_path_var && *innodb_data_file_path_var)
|
2020-04-25 21:57:52 +03:00
|
|
|
innobase_data_file_path= my_strdup(PSI_NOT_INSTRUMENTED,
|
|
|
|
innodb_data_file_path_var, MYF(MY_FAE));
|
2020-04-20 17:23:43 +03:00
|
|
|
|
|
|
|
if (innodb_data_home_dir_var)
|
2020-04-25 21:57:52 +03:00
|
|
|
innobase_data_home_dir= my_strdup(PSI_NOT_INSTRUMENTED,
|
|
|
|
innodb_data_home_dir_var, MYF(MY_FAE));
|
2020-04-20 17:23:43 +03:00
|
|
|
|
|
|
|
if (innodb_log_group_home_dir_var && *innodb_log_group_home_dir_var)
|
2020-04-25 21:57:52 +03:00
|
|
|
srv_log_group_home_dir= my_strdup(PSI_NOT_INSTRUMENTED,
|
|
|
|
innodb_log_group_home_dir_var,
|
|
|
|
MYF(MY_FAE));
|
2020-04-20 17:23:43 +03:00
|
|
|
|
|
|
|
if (innodb_undo_directory_var && *innodb_undo_directory_var)
|
2020-04-25 21:57:52 +03:00
|
|
|
srv_undo_dir= my_strdup(PSI_NOT_INSTRUMENTED, innodb_undo_directory_var,
|
|
|
|
MYF(MY_FAE));
|
2020-04-20 17:23:43 +03:00
|
|
|
|
|
|
|
if (innodb_log_file_size_var)
|
|
|
|
{
|
|
|
|
srv_log_file_size= strtoll(innodb_log_file_size_var, &endptr, 10);
|
|
|
|
ut_ad(*endptr == 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (innodb_page_size_var)
|
|
|
|
{
|
|
|
|
innobase_page_size= strtoll(innodb_page_size_var, &endptr, 10);
|
|
|
|
ut_ad(*endptr == 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (innodb_undo_tablespaces_var)
|
|
|
|
{
|
2021-07-22 11:22:47 +03:00
|
|
|
srv_undo_tablespaces= static_cast<uint32_t>
|
|
|
|
(strtoul(innodb_undo_tablespaces_var, &endptr, 10));
|
2020-04-20 17:23:43 +03:00
|
|
|
ut_ad(*endptr == 0);
|
|
|
|
}
|
|
|
|
|
2023-04-13 15:42:53 +04:00
|
|
|
if (aria_log_dir_path_var)
|
|
|
|
{
|
2023-04-24 12:43:47 +02:00
|
|
|
aria_log_dir_path= my_strdup(PSI_NOT_INSTRUMENTED,
|
|
|
|
aria_log_dir_path_var, MYF(MY_FAE));
|
2023-04-13 15:42:53 +04:00
|
|
|
}
|
|
|
|
|
2020-04-20 17:23:43 +03:00
|
|
|
if (page_zip_level_var != NULL)
|
|
|
|
{
|
2020-04-25 21:57:52 +03:00
|
|
|
page_zip_level= static_cast<uint>(strtoul(page_zip_level_var, &endptr,
|
|
|
|
10));
|
2020-04-20 17:23:43 +03:00
|
|
|
ut_ad(*endptr == 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ignore_db_dirs)
|
|
|
|
xb_load_list_string(ignore_db_dirs, ",", register_ignore_db_dirs_filter);
|
2017-06-19 17:00:09 +00:00
|
|
|
|
2017-04-18 18:43:20 +00:00
|
|
|
out:
|
2020-04-20 17:23:43 +03:00
|
|
|
free_mysql_variables(mysql_vars);
|
2017-04-18 18:43:20 +00:00
|
|
|
|
2020-04-20 17:23:43 +03:00
|
|
|
return (ret);
|
2017-04-18 18:43:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*********************************************************************//**
|
|
|
|
Query the server to find out what backup capabilities it supports.
|
|
|
|
@return true on success. */
|
|
|
|
bool
|
|
|
|
detect_mysql_capabilities_for_backup()
|
|
|
|
{
|
|
|
|
const char *query = "SELECT 'INNODB_CHANGED_PAGES', COUNT(*) FROM "
|
|
|
|
"INFORMATION_SCHEMA.PLUGINS "
|
|
|
|
"WHERE PLUGIN_NAME LIKE 'INNODB_CHANGED_PAGES'";
|
|
|
|
char *innodb_changed_pages = NULL;
|
|
|
|
mysql_variable vars[] = {
|
|
|
|
{"INNODB_CHANGED_PAGES", &innodb_changed_pages}, {NULL, NULL}};
|
|
|
|
|
|
|
|
if (xtrabackup_incremental) {
|
|
|
|
|
|
|
|
read_mysql_variables(mysql_connection, query, vars, true);
|
|
|
|
|
|
|
|
ut_ad(innodb_changed_pages != NULL);
|
|
|
|
|
|
|
|
have_changed_page_bitmaps = (atoi(innodb_changed_pages) == 1);
|
|
|
|
|
|
|
|
free_mysql_variables(vars);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* do some sanity checks */
|
|
|
|
if (opt_galera_info && !have_galera_enabled) {
|
|
|
|
msg("--galera-info is specified on the command "
|
|
|
|
"line, but the server does not support Galera "
|
2019-01-14 22:28:23 +01:00
|
|
|
"replication. Ignoring the option.");
|
2017-04-18 18:43:20 +00:00
|
|
|
opt_galera_info = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (opt_slave_info && have_multi_threaded_slave &&
|
|
|
|
!have_gtid_slave) {
|
|
|
|
msg("The --slave-info option requires GTID enabled for a "
|
2019-01-14 22:28:23 +01:00
|
|
|
"multi-threaded slave.");
|
2017-04-18 18:43:20 +00:00
|
|
|
return(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
return(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
static
|
|
|
|
bool
|
|
|
|
select_incremental_lsn_from_history(lsn_t *incremental_lsn)
|
|
|
|
{
|
|
|
|
MYSQL_RES *mysql_result;
|
|
|
|
char query[1000];
|
|
|
|
char buf[100];
|
|
|
|
|
|
|
|
if (opt_incremental_history_name) {
|
|
|
|
mysql_real_escape_string(mysql_connection, buf,
|
|
|
|
opt_incremental_history_name,
|
2017-04-18 19:05:57 +00:00
|
|
|
(unsigned long)strlen(opt_incremental_history_name));
|
2017-11-11 23:07:24 +02:00
|
|
|
snprintf(query, sizeof(query),
|
2017-04-18 18:43:20 +00:00
|
|
|
"SELECT innodb_to_lsn "
|
2022-09-09 12:02:16 +04:00
|
|
|
"FROM " XB_HISTORY_TABLE " "
|
2017-04-18 18:43:20 +00:00
|
|
|
"WHERE name = '%s' "
|
|
|
|
"AND innodb_to_lsn IS NOT NULL "
|
|
|
|
"ORDER BY innodb_to_lsn DESC LIMIT 1",
|
|
|
|
buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (opt_incremental_history_uuid) {
|
|
|
|
mysql_real_escape_string(mysql_connection, buf,
|
|
|
|
opt_incremental_history_uuid,
|
2017-04-18 19:05:57 +00:00
|
|
|
(unsigned long)strlen(opt_incremental_history_uuid));
|
2017-11-11 23:07:24 +02:00
|
|
|
snprintf(query, sizeof(query),
|
2017-04-18 18:43:20 +00:00
|
|
|
"SELECT innodb_to_lsn "
|
2022-09-09 12:02:16 +04:00
|
|
|
"FROM " XB_HISTORY_TABLE " "
|
2017-04-18 18:43:20 +00:00
|
|
|
"WHERE uuid = '%s' "
|
|
|
|
"AND innodb_to_lsn IS NOT NULL "
|
|
|
|
"ORDER BY innodb_to_lsn DESC LIMIT 1",
|
|
|
|
buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
mysql_result = xb_mysql_query(mysql_connection, query, true);
|
|
|
|
|
|
|
|
ut_ad(mysql_num_fields(mysql_result) == 1);
|
2017-08-18 10:07:11 +03:00
|
|
|
const MYSQL_ROW row = mysql_fetch_row(mysql_result);
|
|
|
|
if (row) {
|
|
|
|
*incremental_lsn = strtoull(row[0], NULL, 10);
|
2019-01-14 22:28:23 +01:00
|
|
|
msg("Found and using lsn: " LSN_PF " for %s %s",
|
2017-08-18 10:07:11 +03:00
|
|
|
*incremental_lsn,
|
|
|
|
opt_incremental_history_uuid ? "uuid" : "name",
|
|
|
|
opt_incremental_history_uuid ?
|
|
|
|
opt_incremental_history_uuid :
|
|
|
|
opt_incremental_history_name);
|
|
|
|
} else {
|
2017-04-18 18:43:20 +00:00
|
|
|
msg("Error while attempting to find history record "
|
2019-01-14 22:28:23 +01:00
|
|
|
"for %s %s",
|
2017-04-18 18:43:20 +00:00
|
|
|
opt_incremental_history_uuid ? "uuid" : "name",
|
|
|
|
opt_incremental_history_uuid ?
|
|
|
|
opt_incremental_history_uuid :
|
|
|
|
opt_incremental_history_name);
|
|
|
|
}
|
|
|
|
|
|
|
|
mysql_free_result(mysql_result);
|
|
|
|
|
2017-08-18 10:07:11 +03:00
|
|
|
return(row != NULL);
|
2017-04-18 18:43:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static
|
|
|
|
const char *
|
|
|
|
eat_sql_whitespace(const char *query)
|
|
|
|
{
|
|
|
|
bool comment = false;
|
|
|
|
|
|
|
|
while (*query) {
|
|
|
|
if (comment) {
|
|
|
|
if (query[0] == '*' && query[1] == '/') {
|
|
|
|
query += 2;
|
|
|
|
comment = false;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
++query;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (query[0] == '/' && query[1] == '*') {
|
|
|
|
query += 2;
|
|
|
|
comment = true;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (strchr("\t\n\r (", query[0])) {
|
|
|
|
++query;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return(query);
|
|
|
|
}
|
|
|
|
|
|
|
|
static
|
|
|
|
bool
|
|
|
|
is_query_from_list(const char *query, const char **list)
|
|
|
|
{
|
|
|
|
const char **item;
|
|
|
|
|
|
|
|
query = eat_sql_whitespace(query);
|
|
|
|
|
|
|
|
item = list;
|
|
|
|
while (*item) {
|
|
|
|
if (strncasecmp(query, *item, strlen(*item)) == 0) {
|
|
|
|
return(true);
|
|
|
|
}
|
|
|
|
++item;
|
|
|
|
}
|
|
|
|
|
|
|
|
return(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
static
|
|
|
|
bool
|
|
|
|
is_query(const char *query)
|
|
|
|
{
|
|
|
|
const char *query_list[] = {"insert", "update", "delete", "replace",
|
|
|
|
"alter", "load", "select", "do", "handler", "call", "execute",
|
|
|
|
"begin", NULL};
|
|
|
|
|
|
|
|
return is_query_from_list(query, query_list);
|
|
|
|
}
|
|
|
|
|
|
|
|
static
|
|
|
|
bool
|
|
|
|
is_select_query(const char *query)
|
|
|
|
{
|
|
|
|
const char *query_list[] = {"select", NULL};
|
|
|
|
|
|
|
|
return is_query_from_list(query, query_list);
|
|
|
|
}
|
|
|
|
|
|
|
|
static
|
|
|
|
bool
|
|
|
|
is_update_query(const char *query)
|
|
|
|
{
|
|
|
|
const char *query_list[] = {"insert", "update", "delete", "replace",
|
|
|
|
"alter", "load", NULL};
|
|
|
|
|
|
|
|
return is_query_from_list(query, query_list);
|
|
|
|
}
|
|
|
|
|
|
|
|
static
|
|
|
|
bool
|
|
|
|
have_queries_to_wait_for(MYSQL *connection, uint threshold)
|
|
|
|
{
|
2017-08-18 10:00:56 +03:00
|
|
|
MYSQL_RES *result = xb_mysql_query(connection, "SHOW FULL PROCESSLIST",
|
|
|
|
true);
|
|
|
|
const bool all_queries = (opt_lock_wait_query_type == QUERY_TYPE_ALL);
|
2017-08-18 10:07:11 +03:00
|
|
|
bool have_to_wait = false;
|
2017-04-18 18:43:20 +00:00
|
|
|
|
2017-08-18 10:00:56 +03:00
|
|
|
while (MYSQL_ROW row = mysql_fetch_row(result)) {
|
2017-04-18 18:43:20 +00:00
|
|
|
const char *info = row[7];
|
2017-07-17 17:04:18 +00:00
|
|
|
int duration = row[5] ? atoi(row[5]) : 0;
|
2017-04-18 18:43:20 +00:00
|
|
|
char *id = row[0];
|
|
|
|
|
|
|
|
if (info != NULL
|
|
|
|
&& duration >= (int)threshold
|
|
|
|
&& ((all_queries && is_query(info))
|
|
|
|
|| is_update_query(info))) {
|
2019-01-14 22:28:23 +01:00
|
|
|
msg("Waiting for query %s (duration %d sec): %s",
|
2017-04-18 18:43:20 +00:00
|
|
|
id, duration, info);
|
2017-08-18 10:07:11 +03:00
|
|
|
have_to_wait = true;
|
|
|
|
break;
|
2017-04-18 18:43:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-18 10:07:11 +03:00
|
|
|
mysql_free_result(result);
|
|
|
|
return(have_to_wait);
|
2017-04-18 18:43:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static
|
|
|
|
void
|
2017-04-18 19:05:57 +00:00
|
|
|
kill_long_queries(MYSQL *connection, time_t timeout)
|
2017-04-18 18:43:20 +00:00
|
|
|
{
|
|
|
|
char kill_stmt[100];
|
|
|
|
|
2017-08-18 10:00:56 +03:00
|
|
|
MYSQL_RES *result = xb_mysql_query(connection, "SHOW FULL PROCESSLIST",
|
|
|
|
true);
|
|
|
|
const bool all_queries = (opt_kill_long_query_type == QUERY_TYPE_ALL);
|
|
|
|
while (MYSQL_ROW row = mysql_fetch_row(result)) {
|
2017-04-18 18:43:20 +00:00
|
|
|
const char *info = row[7];
|
2017-07-17 17:04:18 +00:00
|
|
|
long long duration = row[5]? atoll(row[5]) : 0;
|
2017-04-18 18:43:20 +00:00
|
|
|
char *id = row[0];
|
|
|
|
|
|
|
|
if (info != NULL &&
|
2017-04-18 19:05:57 +00:00
|
|
|
(time_t)duration >= timeout &&
|
2017-04-18 18:43:20 +00:00
|
|
|
((all_queries && is_query(info)) ||
|
|
|
|
is_select_query(info))) {
|
2019-01-14 22:28:23 +01:00
|
|
|
msg("Killing query %s (duration %d sec): %s",
|
2017-04-18 19:05:57 +00:00
|
|
|
id, (int)duration, info);
|
2017-11-11 23:07:24 +02:00
|
|
|
snprintf(kill_stmt, sizeof(kill_stmt),
|
2017-04-18 18:43:20 +00:00
|
|
|
"KILL %s", id);
|
|
|
|
xb_mysql_query(connection, kill_stmt, false, false);
|
|
|
|
}
|
|
|
|
}
|
2017-08-18 10:07:11 +03:00
|
|
|
|
|
|
|
mysql_free_result(result);
|
2017-04-18 18:43:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static
|
|
|
|
bool
|
|
|
|
wait_for_no_updates(MYSQL *connection, uint timeout, uint threshold)
|
|
|
|
{
|
|
|
|
time_t start_time;
|
|
|
|
|
|
|
|
start_time = time(NULL);
|
|
|
|
|
2019-01-14 22:28:23 +01:00
|
|
|
msg("Waiting %u seconds for queries running longer than %u seconds "
|
|
|
|
"to finish", timeout, threshold);
|
2017-04-18 18:43:20 +00:00
|
|
|
|
|
|
|
while (time(NULL) <= (time_t)(start_time + timeout)) {
|
|
|
|
if (!have_queries_to_wait_for(connection, threshold)) {
|
|
|
|
return(true);
|
|
|
|
}
|
2021-03-16 16:09:41 +03:00
|
|
|
std::this_thread::sleep_for(std::chrono::seconds(1));
|
2017-04-18 18:43:20 +00:00
|
|
|
}
|
|
|
|
|
2019-01-14 22:28:23 +01:00
|
|
|
msg("Unable to obtain lock. Please try again later.");
|
2017-04-18 18:43:20 +00:00
|
|
|
|
|
|
|
return(false);
|
|
|
|
}
|
|
|
|
|
2021-03-17 10:03:06 +03:00
|
|
|
static void kill_query_thread()
|
2017-04-18 18:43:20 +00:00
|
|
|
{
|
2020-12-04 16:18:04 +02:00
|
|
|
mysql_mutex_lock(&kill_query_thread_mutex);
|
2017-04-18 18:43:20 +00:00
|
|
|
|
2020-12-04 16:18:04 +02:00
|
|
|
msg("Kill query timeout %d seconds.", opt_kill_long_queries_timeout);
|
2017-04-18 18:43:20 +00:00
|
|
|
|
2020-12-04 16:18:04 +02:00
|
|
|
time_t start_time= time(nullptr);
|
|
|
|
timespec abstime;
|
|
|
|
set_timespec(abstime, opt_kill_long_queries_timeout);
|
2017-04-18 18:43:20 +00:00
|
|
|
|
2020-12-04 16:18:04 +02:00
|
|
|
while (!kill_query_thread_stopping)
|
|
|
|
if (!mysql_cond_timedwait(&kill_query_thread_stop,
|
|
|
|
&kill_query_thread_mutex, &abstime))
|
|
|
|
goto func_exit;
|
2017-04-18 18:43:20 +00:00
|
|
|
|
2020-12-04 16:18:04 +02:00
|
|
|
if (MYSQL *mysql= xb_mysql_connect())
|
|
|
|
{
|
|
|
|
do
|
|
|
|
{
|
|
|
|
kill_long_queries(mysql, time(nullptr) - start_time);
|
|
|
|
set_timespec(abstime, 1);
|
|
|
|
}
|
|
|
|
while (mysql_cond_timedwait(&kill_query_thread_stop,
|
|
|
|
&kill_query_thread_mutex, &abstime) &&
|
|
|
|
!kill_query_thread_stopping);
|
|
|
|
mysql_close(mysql);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
msg("Error: kill query thread failed");
|
2017-04-18 18:43:20 +00:00
|
|
|
|
2020-12-04 16:18:04 +02:00
|
|
|
func_exit:
|
|
|
|
msg("Kill query thread stopped");
|
2017-04-18 18:43:20 +00:00
|
|
|
|
2020-12-04 16:18:04 +02:00
|
|
|
kill_query_thread_running= false;
|
|
|
|
mysql_cond_signal(&kill_query_thread_stopped);
|
|
|
|
mysql_mutex_unlock(&kill_query_thread_mutex);
|
2017-04-18 18:43:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-12-04 16:18:04 +02:00
|
|
|
static void start_query_killer()
|
2017-04-18 18:43:20 +00:00
|
|
|
{
|
2020-12-04 16:18:04 +02:00
|
|
|
ut_ad(!kill_query_thread_running);
|
|
|
|
kill_query_thread_running= true;
|
|
|
|
kill_query_thread_stopping= false;
|
|
|
|
mysql_mutex_init(0, &kill_query_thread_mutex, nullptr);
|
|
|
|
mysql_cond_init(0, &kill_query_thread_stop, nullptr);
|
|
|
|
mysql_cond_init(0, &kill_query_thread_stopped, nullptr);
|
2021-03-17 10:03:06 +03:00
|
|
|
std::thread(kill_query_thread).detach();
|
2017-04-18 18:43:20 +00:00
|
|
|
}
|
|
|
|
|
2020-12-04 16:18:04 +02:00
|
|
|
static void stop_query_killer()
|
2017-04-18 18:43:20 +00:00
|
|
|
{
|
2020-12-04 16:18:04 +02:00
|
|
|
mysql_mutex_lock(&kill_query_thread_mutex);
|
|
|
|
kill_query_thread_stopping= true;
|
|
|
|
mysql_cond_signal(&kill_query_thread_stop);
|
|
|
|
|
|
|
|
do
|
|
|
|
mysql_cond_wait(&kill_query_thread_stopped, &kill_query_thread_mutex);
|
|
|
|
while (kill_query_thread_running);
|
|
|
|
|
|
|
|
mysql_cond_destroy(&kill_query_thread_stop);
|
|
|
|
mysql_cond_destroy(&kill_query_thread_stopped);
|
|
|
|
mysql_mutex_unlock(&kill_query_thread_mutex);
|
|
|
|
mysql_mutex_destroy(&kill_query_thread_mutex);
|
2017-04-18 18:43:20 +00:00
|
|
|
}
|
|
|
|
|
2018-03-29 22:13:01 +00:00
|
|
|
|
2017-04-18 18:43:20 +00:00
|
|
|
/*********************************************************************//**
|
|
|
|
Function acquires either a backup tables lock, if supported
|
|
|
|
by the server, or a global read lock (FLUSH TABLES WITH READ LOCK)
|
|
|
|
otherwise.
|
|
|
|
@returns true if lock acquired */
|
2020-04-27 21:46:05 +03:00
|
|
|
bool lock_tables(MYSQL *connection)
|
2017-04-18 18:43:20 +00:00
|
|
|
{
|
2020-04-27 21:46:05 +03:00
|
|
|
if (have_lock_wait_timeout || opt_lock_wait_timeout)
|
|
|
|
{
|
|
|
|
char buf[FN_REFLEN];
|
|
|
|
/* Set the maximum supported session value for
|
|
|
|
lock_wait_timeout if opt_lock_wait_timeout is not set to prevent
|
|
|
|
unnecessary timeouts when the global value is changed from the default */
|
|
|
|
snprintf(buf, sizeof(buf), "SET SESSION lock_wait_timeout=%u",
|
|
|
|
opt_lock_wait_timeout ? opt_lock_wait_timeout : 31536000);
|
|
|
|
xb_mysql_query(connection, buf, false);
|
|
|
|
}
|
2017-04-18 18:43:20 +00:00
|
|
|
|
2020-04-27 21:46:05 +03:00
|
|
|
if (have_backup_locks)
|
|
|
|
{
|
|
|
|
msg("Executing LOCK TABLES FOR BACKUP...");
|
|
|
|
xb_mysql_query(connection, "LOCK TABLES FOR BACKUP", false);
|
|
|
|
return (true);
|
|
|
|
}
|
2017-04-18 18:43:20 +00:00
|
|
|
|
2020-04-27 21:46:05 +03:00
|
|
|
if (opt_lock_wait_timeout)
|
|
|
|
{
|
|
|
|
if (!wait_for_no_updates(connection, opt_lock_wait_timeout,
|
|
|
|
opt_lock_wait_threshold))
|
|
|
|
{
|
|
|
|
return (false);
|
|
|
|
}
|
|
|
|
}
|
2017-04-18 18:43:20 +00:00
|
|
|
|
2020-04-29 09:05:15 +03:00
|
|
|
msg("Acquiring BACKUP LOCKS...");
|
2017-04-18 18:43:20 +00:00
|
|
|
|
2020-04-27 21:46:05 +03:00
|
|
|
if (opt_kill_long_queries_timeout)
|
|
|
|
{
|
|
|
|
start_query_killer();
|
|
|
|
}
|
2017-04-18 18:43:20 +00:00
|
|
|
|
2020-04-27 21:46:05 +03:00
|
|
|
if (have_galera_enabled)
|
|
|
|
{
|
2021-07-20 12:45:00 -03:00
|
|
|
xb_mysql_query(connection, "SET SESSION wsrep_sync_wait=0", false);
|
2020-04-27 21:46:05 +03:00
|
|
|
}
|
2017-04-18 18:43:20 +00:00
|
|
|
|
2020-04-29 09:05:15 +03:00
|
|
|
xb_mysql_query(connection, "BACKUP STAGE START", true);
|
2021-06-17 13:46:16 +03:00
|
|
|
DBUG_MARIABACKUP_EVENT("after_backup_stage_start", {});
|
2020-04-29 09:05:15 +03:00
|
|
|
xb_mysql_query(connection, "BACKUP STAGE BLOCK_COMMIT", true);
|
2021-06-17 13:46:16 +03:00
|
|
|
DBUG_MARIABACKUP_EVENT("after_backup_stage_block_commit", {});
|
2020-04-27 21:46:05 +03:00
|
|
|
/* Set the maximum supported session value for
|
|
|
|
lock_wait_timeout to prevent unnecessary timeouts when the
|
|
|
|
global value is changed from the default */
|
|
|
|
if (opt_lock_wait_timeout)
|
|
|
|
xb_mysql_query(connection, "SET SESSION lock_wait_timeout=31536000",
|
|
|
|
false);
|
2017-04-18 18:43:20 +00:00
|
|
|
|
2020-04-27 21:46:05 +03:00
|
|
|
if (opt_kill_long_queries_timeout)
|
|
|
|
{
|
|
|
|
stop_query_killer();
|
|
|
|
}
|
2017-04-18 18:43:20 +00:00
|
|
|
|
2020-04-27 21:46:05 +03:00
|
|
|
return (true);
|
2017-04-18 18:43:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*********************************************************************//**
|
|
|
|
If backup locks are used, execute LOCK BINLOG FOR BACKUP provided that we are
|
|
|
|
not in the --no-lock mode and the lock has not been acquired already.
|
|
|
|
@returns true if lock acquired */
|
|
|
|
bool
|
|
|
|
lock_binlog_maybe(MYSQL *connection)
|
|
|
|
{
|
|
|
|
if (have_backup_locks && !opt_no_lock && !binlog_locked) {
|
2019-01-14 22:28:23 +01:00
|
|
|
msg("Executing LOCK BINLOG FOR BACKUP...");
|
2017-04-18 18:43:20 +00:00
|
|
|
xb_mysql_query(connection, "LOCK BINLOG FOR BACKUP", false);
|
|
|
|
binlog_locked = true;
|
|
|
|
|
|
|
|
return(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
return(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*********************************************************************//**
|
|
|
|
Releases either global read lock acquired with FTWRL and the binlog
|
|
|
|
lock acquired with LOCK BINLOG FOR BACKUP, depending on
|
|
|
|
the locking strategy being used */
|
|
|
|
void
|
|
|
|
unlock_all(MYSQL *connection)
|
|
|
|
{
|
|
|
|
if (opt_debug_sleep_before_unlock) {
|
2019-01-14 22:28:23 +01:00
|
|
|
msg("Debug sleep for %u seconds",
|
2017-04-18 18:43:20 +00:00
|
|
|
opt_debug_sleep_before_unlock);
|
2021-03-16 16:09:41 +03:00
|
|
|
std::this_thread::sleep_for(
|
|
|
|
std::chrono::milliseconds(opt_debug_sleep_before_unlock));
|
|
|
|
}
|
2017-04-18 18:43:20 +00:00
|
|
|
|
2019-01-17 13:21:27 +02:00
|
|
|
msg("Executing BACKUP STAGE END");
|
2018-11-19 20:43:04 +01:00
|
|
|
xb_mysql_query(connection, "BACKUP STAGE END", false);
|
2017-04-18 18:43:20 +00:00
|
|
|
|
2019-01-14 22:28:23 +01:00
|
|
|
msg("All tables unlocked");
|
2017-04-18 18:43:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static
|
|
|
|
int
|
|
|
|
get_open_temp_tables(MYSQL *connection)
|
|
|
|
{
|
|
|
|
char *slave_open_temp_tables = NULL;
|
|
|
|
mysql_variable status[] = {
|
|
|
|
{"Slave_open_temp_tables", &slave_open_temp_tables},
|
|
|
|
{NULL, NULL}
|
|
|
|
};
|
|
|
|
int result = false;
|
|
|
|
|
|
|
|
read_mysql_variables(connection,
|
|
|
|
"SHOW STATUS LIKE 'slave_open_temp_tables'", status, true);
|
|
|
|
|
|
|
|
result = slave_open_temp_tables ? atoi(slave_open_temp_tables) : 0;
|
|
|
|
|
|
|
|
free_mysql_variables(status);
|
|
|
|
|
|
|
|
return(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*********************************************************************//**
|
|
|
|
Wait until it's safe to backup a slave. Returns immediately if
|
|
|
|
the host isn't a slave. Currently there's only one check:
|
|
|
|
Slave_open_temp_tables has to be zero. Dies on timeout. */
|
|
|
|
bool
|
|
|
|
wait_for_safe_slave(MYSQL *connection)
|
|
|
|
{
|
|
|
|
char *read_master_log_pos = NULL;
|
|
|
|
char *slave_sql_running = NULL;
|
|
|
|
int n_attempts = 1;
|
|
|
|
const int sleep_time = 3;
|
|
|
|
int open_temp_tables = 0;
|
|
|
|
bool result = true;
|
|
|
|
|
|
|
|
mysql_variable status[] = {
|
|
|
|
{"Read_Master_Log_Pos", &read_master_log_pos},
|
|
|
|
{"Slave_SQL_Running", &slave_sql_running},
|
|
|
|
{NULL, NULL}
|
|
|
|
};
|
|
|
|
|
|
|
|
sql_thread_started = false;
|
|
|
|
|
|
|
|
read_mysql_variables(connection, "SHOW SLAVE STATUS", status, false);
|
|
|
|
|
|
|
|
if (!(read_master_log_pos && slave_sql_running)) {
|
|
|
|
msg("Not checking slave open temp tables for "
|
2019-01-14 22:28:23 +01:00
|
|
|
"--safe-slave-backup because host is not a slave");
|
2017-04-18 18:43:20 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (strcmp(slave_sql_running, "Yes") == 0) {
|
|
|
|
sql_thread_started = true;
|
|
|
|
xb_mysql_query(connection, "STOP SLAVE SQL_THREAD", false);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (opt_safe_slave_backup_timeout > 0) {
|
|
|
|
n_attempts = opt_safe_slave_backup_timeout / sleep_time;
|
|
|
|
}
|
|
|
|
|
|
|
|
open_temp_tables = get_open_temp_tables(connection);
|
2019-01-14 22:28:23 +01:00
|
|
|
msg("Slave open temp tables: %d", open_temp_tables);
|
2017-04-18 18:43:20 +00:00
|
|
|
|
|
|
|
while (open_temp_tables && n_attempts--) {
|
2019-01-14 22:28:23 +01:00
|
|
|
msg("Starting slave SQL thread, waiting %d seconds, then "
|
2017-04-18 18:43:20 +00:00
|
|
|
"checking Slave_open_temp_tables again (%d attempts "
|
2019-01-14 22:28:23 +01:00
|
|
|
"remaining)...", sleep_time, n_attempts);
|
2017-04-18 18:43:20 +00:00
|
|
|
|
|
|
|
xb_mysql_query(connection, "START SLAVE SQL_THREAD", false);
|
2021-03-16 16:09:41 +03:00
|
|
|
std::this_thread::sleep_for(std::chrono::seconds(sleep_time));
|
2017-04-18 18:43:20 +00:00
|
|
|
xb_mysql_query(connection, "STOP SLAVE SQL_THREAD", false);
|
|
|
|
|
|
|
|
open_temp_tables = get_open_temp_tables(connection);
|
2019-01-14 22:28:23 +01:00
|
|
|
msg("Slave open temp tables: %d", open_temp_tables);
|
2017-04-18 18:43:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Restart the slave if it was running at start */
|
|
|
|
if (open_temp_tables == 0) {
|
2019-01-14 22:28:23 +01:00
|
|
|
msg("Slave is safe to backup");
|
2017-04-18 18:43:20 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
result = false;
|
|
|
|
|
|
|
|
if (sql_thread_started) {
|
2019-01-14 22:28:23 +01:00
|
|
|
msg("Restarting slave SQL thread.");
|
2017-04-18 18:43:20 +00:00
|
|
|
xb_mysql_query(connection, "START SLAVE SQL_THREAD", false);
|
|
|
|
}
|
|
|
|
|
2019-01-14 22:28:23 +01:00
|
|
|
msg("Slave_open_temp_tables did not become zero after "
|
|
|
|
"%d seconds", opt_safe_slave_backup_timeout);
|
2017-04-18 18:43:20 +00:00
|
|
|
|
|
|
|
cleanup:
|
|
|
|
free_mysql_variables(status);
|
|
|
|
|
|
|
|
return(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-04-18 19:43:25 +04:00
|
|
|
class Var
|
|
|
|
{
|
|
|
|
const char *m_name;
|
|
|
|
char *m_value;
|
|
|
|
/*
|
|
|
|
Disable copying constructors for safety, as the default binary copying
|
|
|
|
which would be wrong. If we ever want them, the m_value
|
|
|
|
member should be copied using an strdup()-alike function.
|
|
|
|
*/
|
|
|
|
Var(const Var &); // Disabled
|
|
|
|
Var(Var &); // Disabled
|
|
|
|
public:
|
|
|
|
~Var()
|
|
|
|
{
|
|
|
|
free(m_value);
|
|
|
|
}
|
|
|
|
Var(const char *name)
|
|
|
|
:m_name(name),
|
|
|
|
m_value(NULL)
|
|
|
|
{ }
|
|
|
|
// Init using a SHOW VARIABLES LIKE 'name' query
|
|
|
|
Var(const char *name, MYSQL *mysql)
|
|
|
|
:m_name(name)
|
|
|
|
{
|
|
|
|
char buf[128];
|
|
|
|
my_snprintf(buf, sizeof(buf), "SHOW VARIABLES LIKE '%s'", m_name);
|
|
|
|
m_value= read_mysql_one_value(mysql, buf, 1/*offset*/, 2/*total columns*/);
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
Init by name from a result set.
|
|
|
|
If the variable name is not found in the result set metadata field names,
|
|
|
|
it's value stays untouched.
|
|
|
|
*/
|
|
|
|
bool init(MYSQL_RES *mysql_result, MYSQL_ROW row)
|
|
|
|
{
|
|
|
|
MYSQL_FIELD *field= mysql_fetch_fields(mysql_result);
|
|
|
|
for (uint i= 0; i < mysql_num_fields(mysql_result); i++)
|
|
|
|
{
|
|
|
|
if (!strcmp(field[i].name, m_name))
|
|
|
|
{
|
|
|
|
free(m_value); // In case it was initialized earlier
|
|
|
|
m_value= row[i] ? strdup(row[i]) : NULL;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
void replace(char from, char to)
|
|
|
|
{
|
|
|
|
ut_ad(m_value);
|
|
|
|
for (char *ptr= strchr(m_value, from); ptr; ptr= strchr(ptr, from))
|
|
|
|
*ptr= to;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *value() const { return m_value; }
|
|
|
|
bool eq_value(const char *str, size_t length) const
|
|
|
|
{
|
|
|
|
return m_value && !strncmp(m_value, str, length) && m_value[length] == '\0';
|
|
|
|
}
|
|
|
|
bool is_null_or_empty() const { return !m_value || !m_value[0]; }
|
|
|
|
bool print(String *to) const
|
|
|
|
{
|
|
|
|
ut_ad(m_value);
|
2022-05-10 11:53:59 +02:00
|
|
|
return to->append(m_value, strlen(m_value));
|
2022-04-18 19:43:25 +04:00
|
|
|
}
|
|
|
|
bool print_quoted(String *to) const
|
|
|
|
{
|
|
|
|
ut_ad(m_value);
|
2022-05-10 11:53:59 +02:00
|
|
|
return to->append('\'') || to->append(m_value, strlen(m_value)) ||
|
|
|
|
to->append('\'');
|
2022-04-18 19:43:25 +04:00
|
|
|
}
|
|
|
|
bool print_set_global(String *to) const
|
|
|
|
{
|
|
|
|
ut_ad(m_value);
|
|
|
|
return
|
2022-05-10 11:53:59 +02:00
|
|
|
to->append(STRING_WITH_LEN("SET GLOBAL ")) ||
|
|
|
|
to->append(m_name, strlen(m_name)) ||
|
|
|
|
to->append(STRING_WITH_LEN(" = '")) ||
|
|
|
|
to->append(m_value, strlen(m_value)) ||
|
|
|
|
to->append(STRING_WITH_LEN("';\n"));
|
2022-04-18 19:43:25 +04:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class Show_slave_status
|
|
|
|
{
|
|
|
|
Var m_mariadb_connection_name; // MariaDB: e.g. 'master1'
|
|
|
|
Var m_master; // e.g. 'localhost'
|
|
|
|
Var m_filename; // e.g. 'source-bin.000002'
|
|
|
|
Var m_position; // a number
|
|
|
|
Var m_mysql_gtid_executed; // MySQL56: e.g. single '<UUID>:1-5" or multiline
|
|
|
|
// '<UUID1>:1-10,\n<UUID2>:1-20\n<UUID3>:1-30'
|
|
|
|
Var m_mariadb_using_gtid; // MariaDB: 'No','Slave_Pos','Current_Pos'
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
Show_slave_status()
|
|
|
|
:m_mariadb_connection_name("Connection_name"),
|
|
|
|
m_master("Master_Host"),
|
|
|
|
m_filename("Relay_Master_Log_File"),
|
|
|
|
m_position("Exec_Master_Log_Pos"),
|
|
|
|
m_mysql_gtid_executed("Executed_Gtid_Set"),
|
|
|
|
m_mariadb_using_gtid("Using_Gtid")
|
|
|
|
{ }
|
|
|
|
|
|
|
|
void init(MYSQL_RES *res, MYSQL_ROW row)
|
|
|
|
{
|
|
|
|
m_mariadb_connection_name.init(res, row);
|
|
|
|
m_master.init(res, row);
|
|
|
|
m_filename.init(res, row);
|
|
|
|
m_position.init(res, row);
|
|
|
|
m_mysql_gtid_executed.init(res, row);
|
|
|
|
m_mariadb_using_gtid.init(res, row);
|
|
|
|
// Normalize
|
|
|
|
if (m_mysql_gtid_executed.value())
|
|
|
|
m_mysql_gtid_executed.replace('\n', ' ');
|
|
|
|
}
|
|
|
|
|
|
|
|
static void msg_is_not_slave()
|
|
|
|
{
|
|
|
|
msg("Failed to get master binlog coordinates "
|
|
|
|
"from SHOW SLAVE STATUS.This means that the server is not a "
|
|
|
|
"replication slave. Ignoring the --slave-info option");
|
|
|
|
}
|
|
|
|
|
|
|
|
bool is_mariadb_using_gtid() const
|
|
|
|
{
|
|
|
|
return !m_mariadb_using_gtid.eq_value("No", 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool start_comment_chunk(String *to)
|
|
|
|
{
|
2022-05-10 11:53:59 +02:00
|
|
|
return to->length() ? to->append(STRING_WITH_LEN("; ")) : false;
|
2022-04-18 19:43:25 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
bool print_connection_name_if_set(String *to) const
|
|
|
|
{
|
|
|
|
if (!m_mariadb_connection_name.is_null_or_empty())
|
|
|
|
return m_mariadb_connection_name.print_quoted(to) || to->append(' ');
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool print_comment_master_identity(String *comment) const
|
|
|
|
{
|
2022-05-10 11:53:59 +02:00
|
|
|
if (comment->append(STRING_WITH_LEN("master ")))
|
2022-04-18 19:43:25 +04:00
|
|
|
return true;
|
|
|
|
if (!m_mariadb_connection_name.is_null_or_empty())
|
|
|
|
return m_mariadb_connection_name.print_quoted(comment);
|
2022-05-10 11:53:59 +02:00
|
|
|
return comment->append(STRING_WITH_LEN("''")); // Default not named master
|
2022-04-18 19:43:25 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
bool print_using_master_log_pos(String *sql, String *comment) const
|
|
|
|
{
|
|
|
|
return
|
2022-05-10 11:53:59 +02:00
|
|
|
sql->append(STRING_WITH_LEN("CHANGE MASTER ")) ||
|
2022-04-18 19:43:25 +04:00
|
|
|
print_connection_name_if_set(sql) ||
|
2022-05-10 11:53:59 +02:00
|
|
|
sql->append(STRING_WITH_LEN("TO MASTER_LOG_FILE=")) ||
|
|
|
|
m_filename.print_quoted(sql) ||
|
|
|
|
sql->append(STRING_WITH_LEN(", MASTER_LOG_POS=")) ||
|
|
|
|
m_position.print(sql) ||
|
|
|
|
sql->append(STRING_WITH_LEN(";\n")) ||
|
2022-04-18 19:43:25 +04:00
|
|
|
print_comment_master_identity(comment) ||
|
2022-05-10 11:53:59 +02:00
|
|
|
comment->append(STRING_WITH_LEN(" filename ")) ||
|
|
|
|
m_filename.print_quoted(comment) ||
|
|
|
|
comment->append(STRING_WITH_LEN(" position ")) ||
|
|
|
|
m_position.print_quoted(comment);
|
2022-04-18 19:43:25 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
bool print_mysql56(String *sql, String *comment) const
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
SET @@GLOBAL.gtid_purged = '2174B383-5441-11E8-B90A-C80AA9429562:1-1029, '
|
|
|
|
'224DA167-0C0C-11E8-8442-00059A3C7B00:1-2695';
|
|
|
|
CHANGE MASTER TO MASTER_AUTO_POSITION=1;
|
|
|
|
*/
|
|
|
|
return
|
2022-05-10 11:53:59 +02:00
|
|
|
sql->append(STRING_WITH_LEN("SET GLOBAL gtid_purged=")) ||
|
2022-04-18 19:43:25 +04:00
|
|
|
m_mysql_gtid_executed.print_quoted(sql) ||
|
2022-05-10 11:53:59 +02:00
|
|
|
sql->append(STRING_WITH_LEN(";\n")) ||
|
|
|
|
sql->append(STRING_WITH_LEN("CHANGE MASTER TO MASTER_AUTO_POSITION=1;\n")) ||
|
2022-04-18 19:43:25 +04:00
|
|
|
print_comment_master_identity(comment) ||
|
2022-05-10 11:53:59 +02:00
|
|
|
comment->append(STRING_WITH_LEN(" purge list ")) ||
|
2022-04-18 19:43:25 +04:00
|
|
|
m_mysql_gtid_executed.print_quoted(comment);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool print_mariadb10_using_gtid(String *sql, String *comment) const
|
|
|
|
{
|
|
|
|
return
|
2022-05-10 11:53:59 +02:00
|
|
|
sql->append(STRING_WITH_LEN("CHANGE MASTER ")) ||
|
2022-04-18 19:43:25 +04:00
|
|
|
print_connection_name_if_set(sql) ||
|
2022-05-10 11:53:59 +02:00
|
|
|
sql->append(STRING_WITH_LEN("TO master_use_gtid = slave_pos;\n")) ||
|
2022-04-18 19:43:25 +04:00
|
|
|
print_comment_master_identity(comment) ||
|
2022-05-10 11:53:59 +02:00
|
|
|
comment->append(STRING_WITH_LEN(" master_use_gtid = slave_pos"));
|
2022-04-18 19:43:25 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
bool print(String *sql, String *comment, const Var >id_slave_pos) const
|
|
|
|
{
|
|
|
|
if (!m_mysql_gtid_executed.is_null_or_empty())
|
|
|
|
{
|
|
|
|
/* MySQL >= 5.6 with GTID enabled */
|
|
|
|
return print_mysql56(sql, comment);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!gtid_slave_pos.is_null_or_empty() && is_mariadb_using_gtid())
|
|
|
|
{
|
|
|
|
/* MariaDB >= 10.0 with GTID enabled */
|
|
|
|
return print_mariadb10_using_gtid(sql, comment);
|
|
|
|
}
|
|
|
|
|
|
|
|
return print_using_master_log_pos(sql, comment);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
Get master info into strings "sql" and "comment" from a MYSQL_RES.
|
|
|
|
@return false on success
|
|
|
|
@return true on error
|
|
|
|
*/
|
|
|
|
static bool get_slave_info(MYSQL_RES *show_slave_info_result,
|
|
|
|
const Var >id_slave_pos,
|
|
|
|
String *sql, String *comment)
|
|
|
|
{
|
|
|
|
if (!gtid_slave_pos.is_null_or_empty())
|
|
|
|
{
|
|
|
|
// Print gtid_slave_pos if any of the masters really needs it.
|
|
|
|
while (MYSQL_ROW row= mysql_fetch_row(show_slave_info_result))
|
|
|
|
{
|
|
|
|
Show_slave_status status;
|
|
|
|
status.init(show_slave_info_result, row);
|
|
|
|
if (status.is_mariadb_using_gtid())
|
|
|
|
{
|
|
|
|
if (gtid_slave_pos.print_set_global(sql) ||
|
2022-05-10 11:53:59 +02:00
|
|
|
comment->append(STRING_WITH_LEN("gtid_slave_pos ")) ||
|
2022-04-18 19:43:25 +04:00
|
|
|
gtid_slave_pos.print_quoted(comment))
|
|
|
|
return true; // Error
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Print the list of masters
|
|
|
|
mysql_data_seek(show_slave_info_result, 0);
|
|
|
|
while (MYSQL_ROW row= mysql_fetch_row(show_slave_info_result))
|
|
|
|
{
|
|
|
|
Show_slave_status status;
|
|
|
|
status.init(show_slave_info_result, row);
|
|
|
|
if (start_comment_chunk(comment) ||
|
|
|
|
status.print(sql, comment, gtid_slave_pos))
|
|
|
|
return true; // Error
|
|
|
|
}
|
|
|
|
return false; // Success
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
Get master info into strings "sql" and "comment".
|
|
|
|
@return false on success
|
|
|
|
@return true on error
|
|
|
|
*/
|
|
|
|
static bool get_slave_info(MYSQL *mysql, bool show_all_slave_status,
|
|
|
|
String *sql, String *comment)
|
|
|
|
{
|
|
|
|
bool rc= false; // Success
|
|
|
|
// gtid_slave_pos - MariaDB variable : e.g. "0-1-1" or "1-10-100,2-20-500"
|
|
|
|
Var gtid_slave_pos("gtid_slave_pos", mysql);
|
|
|
|
const char *query= show_all_slave_status ? "SHOW ALL SLAVES STATUS" :
|
|
|
|
"SHOW SLAVE STATUS";
|
|
|
|
MYSQL_RES *mysql_result= xb_mysql_query(mysql, query, true);
|
|
|
|
if (!mysql_num_rows(mysql_result))
|
|
|
|
{
|
|
|
|
msg_is_not_slave();
|
|
|
|
// Don't change rc, we still want to continue the backup
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
rc= get_slave_info(mysql_result, gtid_slave_pos, sql, comment);
|
|
|
|
}
|
|
|
|
mysql_free_result(mysql_result);
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
2017-04-18 18:43:20 +00:00
|
|
|
/*********************************************************************//**
|
|
|
|
Retrieves MySQL binlog position of the master server in a replication
|
|
|
|
setup and saves it in a file. It also saves it in mysql_slave_position
|
2022-04-18 19:43:25 +04:00
|
|
|
variable.
|
|
|
|
@returns false on error
|
|
|
|
@returns true on success
|
|
|
|
*/
|
2017-04-18 18:43:20 +00:00
|
|
|
bool
|
2023-04-12 11:40:46 +04:00
|
|
|
write_slave_info(ds_ctxt *datasink, MYSQL *connection)
|
2017-04-18 18:43:20 +00:00
|
|
|
{
|
2022-04-18 19:43:25 +04:00
|
|
|
String sql, comment;
|
2017-04-18 18:43:20 +00:00
|
|
|
|
2022-05-25 09:20:04 +03:00
|
|
|
if (Show_slave_status::get_slave_info(connection, true, &sql, &comment))
|
2022-04-18 19:43:25 +04:00
|
|
|
return false; // Error
|
2017-04-18 18:43:20 +00:00
|
|
|
|
2022-04-18 19:43:25 +04:00
|
|
|
if (!sql.length())
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
SHOW [ALL] SLAVE STATUS returned no rows.
|
|
|
|
Don't create the file, but return success to continue the backup.
|
|
|
|
*/
|
|
|
|
return true; // Success
|
|
|
|
}
|
2017-04-18 18:43:20 +00:00
|
|
|
|
2022-04-18 19:43:25 +04:00
|
|
|
mysql_slave_position= strdup(comment.c_ptr());
|
2023-06-05 20:13:06 +02:00
|
|
|
return datasink->backup_file_print_buf(MB_SLAVE_INFO,
|
2023-04-12 11:40:46 +04:00
|
|
|
sql.ptr(), sql.length());
|
2017-04-18 18:43:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*********************************************************************//**
|
|
|
|
Retrieves MySQL Galera and
|
|
|
|
saves it in a file. It also prints it to stdout. */
|
|
|
|
bool
|
2023-04-12 11:40:46 +04:00
|
|
|
write_galera_info(ds_ctxt *datasink, MYSQL *connection)
|
2017-04-18 18:43:20 +00:00
|
|
|
{
|
|
|
|
char *state_uuid = NULL, *state_uuid55 = NULL;
|
|
|
|
char *last_committed = NULL, *last_committed55 = NULL;
|
|
|
|
bool result;
|
|
|
|
|
|
|
|
mysql_variable status[] = {
|
|
|
|
{"Wsrep_local_state_uuid", &state_uuid},
|
|
|
|
{"wsrep_local_state_uuid", &state_uuid55},
|
|
|
|
{"Wsrep_last_committed", &last_committed},
|
|
|
|
{"wsrep_last_committed", &last_committed55},
|
|
|
|
{NULL, NULL}
|
|
|
|
};
|
|
|
|
|
|
|
|
/* When backup locks are supported by the server, we should skip
|
2023-02-03 07:42:59 +04:00
|
|
|
creating MB_GALERA_INFO file on the backup stage, because
|
2017-04-18 18:43:20 +00:00
|
|
|
wsrep_local_state_uuid and wsrep_last_committed will be inconsistent
|
|
|
|
without blocking commits. The state file will be created on the prepare
|
|
|
|
stage using the WSREP recovery procedure. */
|
|
|
|
if (have_backup_locks) {
|
|
|
|
return(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
read_mysql_variables(connection, "SHOW STATUS", status, true);
|
|
|
|
|
|
|
|
if ((state_uuid == NULL && state_uuid55 == NULL)
|
|
|
|
|| (last_committed == NULL && last_committed55 == NULL)) {
|
2022-12-26 08:23:16 +01:00
|
|
|
msg("Warning: failed to get master wsrep state from SHOW STATUS.");
|
|
|
|
result = true;
|
2017-04-18 18:43:20 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2023-06-05 20:13:06 +02:00
|
|
|
result = datasink->backup_file_printf(MB_GALERA_INFO,
|
2017-04-18 18:43:20 +00:00
|
|
|
"%s:%s\n", state_uuid ? state_uuid : state_uuid55,
|
|
|
|
last_committed ? last_committed : last_committed55);
|
2022-12-26 08:23:16 +01:00
|
|
|
if (result)
|
|
|
|
{
|
2023-04-12 11:40:46 +04:00
|
|
|
write_current_binlog_file(datasink, connection);
|
2022-12-26 08:23:16 +01:00
|
|
|
}
|
2017-04-18 18:43:20 +00:00
|
|
|
|
|
|
|
cleanup:
|
|
|
|
free_mysql_variables(status);
|
|
|
|
|
|
|
|
return(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*********************************************************************//**
|
|
|
|
Flush and copy the current binary log file into the backup,
|
|
|
|
if GTID is enabled */
|
|
|
|
bool
|
2023-04-12 11:40:46 +04:00
|
|
|
write_current_binlog_file(ds_ctxt *datasink, MYSQL *connection)
|
2017-04-18 18:43:20 +00:00
|
|
|
{
|
|
|
|
char *executed_gtid_set = NULL;
|
|
|
|
char *gtid_binlog_state = NULL;
|
2022-06-28 09:16:31 +02:00
|
|
|
char *log_bin_file = NULL;
|
2017-04-18 18:43:20 +00:00
|
|
|
char *log_bin_dir = NULL;
|
|
|
|
bool gtid_exists;
|
|
|
|
bool result = true;
|
2022-06-28 09:16:31 +02:00
|
|
|
char filepath[FN_REFLEN];
|
2017-04-18 18:43:20 +00:00
|
|
|
|
2022-06-28 09:16:31 +02:00
|
|
|
mysql_variable status[] = {
|
|
|
|
{"Executed_Gtid_Set", &executed_gtid_set},
|
2017-04-18 18:43:20 +00:00
|
|
|
{NULL, NULL}
|
|
|
|
};
|
|
|
|
|
2022-06-28 09:16:31 +02:00
|
|
|
mysql_variable status_after_flush[] = {
|
|
|
|
{"File", &log_bin_file},
|
2017-04-18 18:43:20 +00:00
|
|
|
{NULL, NULL}
|
|
|
|
};
|
|
|
|
|
2022-06-28 09:16:31 +02:00
|
|
|
mysql_variable vars[] = {
|
|
|
|
{"gtid_binlog_state", >id_binlog_state},
|
|
|
|
{"log_bin_basename", &log_bin_dir},
|
MDEV-27524: Incorrect binlogs after Galera SST using rsync and mariabackup
This commit adds correct handling of binlogs for SST using rsync
or mariabackup. Before this fix, binlogs were handled incorrectly -
- only one (last) binary log file was transferred during SST, which
then led to various failures (for example, when trying to list all
events from the binary log). These bugs were long masked by flaws
in the primitive binlogs handling code in the SST scripts, which
causing binary logs files to be erased after transfer or not added
to the binlog index on the joiner node. Now the correct transfer
of all binary logs (not just the last of the binary log files) has
been implemented both for the rsync (at the script level) and for
the mariabackup (at the level of the main utility code).
This commit also adds a new sst_max_binlogs=<n> parameter, which
can be located in the [sst] section or in the [xtrabackup] section
(historically, supported for mariabackup only, not for rsync), or
in one of the server sections. This parameter specifies the number
of binary log files to be sent to the joiner node during SST. This
option is added for compatibility with old SST scripting behavior,
which can be emulated by setting the sst_max_binlogs=1 (although
in general this can cause problems for the reasons described above).
In addition, setting the sst_max_binlogs=0 can be used to suppress
the transmission of binary logs to the joiner nodes during SST
(although sometimes a single file with the current binary log can
still be transmitted to the joiner, even with sst_max_binlogs=0,
because this sometimes necessary in modes that involve the use of
GTIDs with Galera).
Also, this commit ensures correct handling of paths to various
innodb files and directories in the SST scripts, and fixes some
problems with this that existed in mariabackup utility (which
were associated with incorrect handling of the innodb_data_dir
parameter in some scenarios).
In addition, this commit contains the following enhancements:
1) Added tests for mtr, which check the correct work with binlogs
after SST (using rsync and mariabackup);
2) Added correct handling of slashes at the end of all paths that
the SST script receives as parameters;
3) Improved parsing code for --mysqld-args parameters. Now it
correctly processes the sequence "--" after the name of the
one-letter option;
4) Checking the secret signature during joiner authentication
is made independent of presence of bash (as a unix shell)
in the system and diff utility no longer needed to check
certificates compliance;
5) All directories that are necessary for the correct placement
of various logs are automatically created by SST scripts in
advance (before running mariabackup on the joiner node);
6) Removal of old binary logs on joiner is done using the binlog
index (if it exists) (not only by fixed pattern that based
on the current binlog name, as before);
7) Paths for placing binary logs are correctly processed if they
are set as relative paths (to the datadir);
8) SST scripts are made even more resistant to spaces in filenames
(now for binlogs);
9) In case of failure, SST scripts now always end with an exit
code other than zero;
10) SST script for rsync now correctly create a tar file with
the binlogs, even if the paths to them (in the binlog index
file) are specified as a mix of absolute and relative paths,
and even if they do not match with the datadir path specified
in the current configuration settings.
2022-02-22 10:45:06 +01:00
|
|
|
{NULL, NULL}
|
|
|
|
};
|
|
|
|
|
2017-04-18 18:43:20 +00:00
|
|
|
read_mysql_variables(connection, "SHOW MASTER STATUS", status, false);
|
|
|
|
read_mysql_variables(connection, "SHOW VARIABLES", vars, true);
|
|
|
|
|
|
|
|
gtid_exists = (executed_gtid_set && *executed_gtid_set)
|
|
|
|
|| (gtid_binlog_state && *gtid_binlog_state);
|
|
|
|
|
2022-06-28 09:16:31 +02:00
|
|
|
if (gtid_exists) {
|
|
|
|
size_t log_bin_dir_length;
|
|
|
|
|
|
|
|
lock_binlog_maybe(connection);
|
|
|
|
|
|
|
|
xb_mysql_query(connection, "FLUSH BINARY LOGS", false);
|
|
|
|
|
|
|
|
read_mysql_variables(connection, "SHOW MASTER STATUS",
|
|
|
|
status_after_flush, false);
|
2017-04-18 18:43:20 +00:00
|
|
|
|
|
|
|
if (opt_log_bin != NULL && strchr(opt_log_bin, FN_LIBCHAR)) {
|
|
|
|
/* If log_bin is set, it has priority */
|
|
|
|
if (log_bin_dir) {
|
|
|
|
free(log_bin_dir);
|
|
|
|
}
|
|
|
|
log_bin_dir = strdup(opt_log_bin);
|
|
|
|
} else if (log_bin_dir == NULL) {
|
|
|
|
/* Default location is MySQL datadir */
|
2022-06-28 09:16:31 +02:00
|
|
|
log_bin_dir = strdup("./");
|
2017-04-18 18:43:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
dirname_part(log_bin_dir, log_bin_dir, &log_bin_dir_length);
|
|
|
|
|
|
|
|
/* strip final slash if it is not the only path component */
|
2022-06-28 09:16:31 +02:00
|
|
|
if (log_bin_dir_length > 1 &&
|
|
|
|
log_bin_dir[log_bin_dir_length - 1] == FN_LIBCHAR) {
|
|
|
|
log_bin_dir[log_bin_dir_length - 1] = 0;
|
2017-04-18 18:43:20 +00:00
|
|
|
}
|
|
|
|
|
2022-06-28 09:16:31 +02:00
|
|
|
if (log_bin_dir == NULL || log_bin_file == NULL) {
|
|
|
|
msg("Failed to get master binlog coordinates from "
|
|
|
|
"SHOW MASTER STATUS");
|
2017-04-18 18:43:20 +00:00
|
|
|
result = false;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2022-06-28 09:16:31 +02:00
|
|
|
snprintf(filepath, sizeof(filepath), "%s%c%s",
|
|
|
|
log_bin_dir, FN_LIBCHAR, log_bin_file);
|
2023-04-12 11:40:46 +04:00
|
|
|
result = datasink->copy_file(filepath, log_bin_file, 0);
|
2017-04-18 18:43:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
cleanup:
|
2022-06-28 09:16:31 +02:00
|
|
|
free_mysql_variables(status_after_flush);
|
MDEV-27524: Incorrect binlogs after Galera SST using rsync and mariabackup
This commit adds correct handling of binlogs for SST using rsync
or mariabackup. Before this fix, binlogs were handled incorrectly -
- only one (last) binary log file was transferred during SST, which
then led to various failures (for example, when trying to list all
events from the binary log). These bugs were long masked by flaws
in the primitive binlogs handling code in the SST scripts, which
causing binary logs files to be erased after transfer or not added
to the binlog index on the joiner node. Now the correct transfer
of all binary logs (not just the last of the binary log files) has
been implemented both for the rsync (at the script level) and for
the mariabackup (at the level of the main utility code).
This commit also adds a new sst_max_binlogs=<n> parameter, which
can be located in the [sst] section or in the [xtrabackup] section
(historically, supported for mariabackup only, not for rsync), or
in one of the server sections. This parameter specifies the number
of binary log files to be sent to the joiner node during SST. This
option is added for compatibility with old SST scripting behavior,
which can be emulated by setting the sst_max_binlogs=1 (although
in general this can cause problems for the reasons described above).
In addition, setting the sst_max_binlogs=0 can be used to suppress
the transmission of binary logs to the joiner nodes during SST
(although sometimes a single file with the current binary log can
still be transmitted to the joiner, even with sst_max_binlogs=0,
because this sometimes necessary in modes that involve the use of
GTIDs with Galera).
Also, this commit ensures correct handling of paths to various
innodb files and directories in the SST scripts, and fixes some
problems with this that existed in mariabackup utility (which
were associated with incorrect handling of the innodb_data_dir
parameter in some scenarios).
In addition, this commit contains the following enhancements:
1) Added tests for mtr, which check the correct work with binlogs
after SST (using rsync and mariabackup);
2) Added correct handling of slashes at the end of all paths that
the SST script receives as parameters;
3) Improved parsing code for --mysqld-args parameters. Now it
correctly processes the sequence "--" after the name of the
one-letter option;
4) Checking the secret signature during joiner authentication
is made independent of presence of bash (as a unix shell)
in the system and diff utility no longer needed to check
certificates compliance;
5) All directories that are necessary for the correct placement
of various logs are automatically created by SST scripts in
advance (before running mariabackup on the joiner node);
6) Removal of old binary logs on joiner is done using the binlog
index (if it exists) (not only by fixed pattern that based
on the current binlog name, as before);
7) Paths for placing binary logs are correctly processed if they
are set as relative paths (to the datadir);
8) SST scripts are made even more resistant to spaces in filenames
(now for binlogs);
9) In case of failure, SST scripts now always end with an exit
code other than zero;
10) SST script for rsync now correctly create a tar file with
the binlogs, even if the paths to them (in the binlog index
file) are specified as a mix of absolute and relative paths,
and even if they do not match with the datadir path specified
in the current configuration settings.
2022-02-22 10:45:06 +01:00
|
|
|
free_mysql_variables(status);
|
2022-06-28 09:16:31 +02:00
|
|
|
free_mysql_variables(vars);
|
MDEV-27524: Incorrect binlogs after Galera SST using rsync and mariabackup
This commit adds correct handling of binlogs for SST using rsync
or mariabackup. Before this fix, binlogs were handled incorrectly -
- only one (last) binary log file was transferred during SST, which
then led to various failures (for example, when trying to list all
events from the binary log). These bugs were long masked by flaws
in the primitive binlogs handling code in the SST scripts, which
causing binary logs files to be erased after transfer or not added
to the binlog index on the joiner node. Now the correct transfer
of all binary logs (not just the last of the binary log files) has
been implemented both for the rsync (at the script level) and for
the mariabackup (at the level of the main utility code).
This commit also adds a new sst_max_binlogs=<n> parameter, which
can be located in the [sst] section or in the [xtrabackup] section
(historically, supported for mariabackup only, not for rsync), or
in one of the server sections. This parameter specifies the number
of binary log files to be sent to the joiner node during SST. This
option is added for compatibility with old SST scripting behavior,
which can be emulated by setting the sst_max_binlogs=1 (although
in general this can cause problems for the reasons described above).
In addition, setting the sst_max_binlogs=0 can be used to suppress
the transmission of binary logs to the joiner nodes during SST
(although sometimes a single file with the current binary log can
still be transmitted to the joiner, even with sst_max_binlogs=0,
because this sometimes necessary in modes that involve the use of
GTIDs with Galera).
Also, this commit ensures correct handling of paths to various
innodb files and directories in the SST scripts, and fixes some
problems with this that existed in mariabackup utility (which
were associated with incorrect handling of the innodb_data_dir
parameter in some scenarios).
In addition, this commit contains the following enhancements:
1) Added tests for mtr, which check the correct work with binlogs
after SST (using rsync and mariabackup);
2) Added correct handling of slashes at the end of all paths that
the SST script receives as parameters;
3) Improved parsing code for --mysqld-args parameters. Now it
correctly processes the sequence "--" after the name of the
one-letter option;
4) Checking the secret signature during joiner authentication
is made independent of presence of bash (as a unix shell)
in the system and diff utility no longer needed to check
certificates compliance;
5) All directories that are necessary for the correct placement
of various logs are automatically created by SST scripts in
advance (before running mariabackup on the joiner node);
6) Removal of old binary logs on joiner is done using the binlog
index (if it exists) (not only by fixed pattern that based
on the current binlog name, as before);
7) Paths for placing binary logs are correctly processed if they
are set as relative paths (to the datadir);
8) SST scripts are made even more resistant to spaces in filenames
(now for binlogs);
9) In case of failure, SST scripts now always end with an exit
code other than zero;
10) SST script for rsync now correctly create a tar file with
the binlogs, even if the paths to them (in the binlog index
file) are specified as a mix of absolute and relative paths,
and even if they do not match with the datadir path specified
in the current configuration settings.
2022-02-22 10:45:06 +01:00
|
|
|
|
2017-04-18 18:43:20 +00:00
|
|
|
return(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*********************************************************************//**
|
|
|
|
Retrieves MySQL binlog position and
|
|
|
|
saves it in a file. It also prints it to stdout. */
|
|
|
|
bool
|
2023-04-12 11:40:46 +04:00
|
|
|
write_binlog_info(ds_ctxt *datasink, MYSQL *connection)
|
2017-04-18 18:43:20 +00:00
|
|
|
{
|
|
|
|
char *filename = NULL;
|
|
|
|
char *position = NULL;
|
|
|
|
char *gtid_mode = NULL;
|
|
|
|
char *gtid_current_pos = NULL;
|
|
|
|
char *gtid_executed = NULL;
|
|
|
|
char *gtid = NULL;
|
2022-06-28 09:16:31 +02:00
|
|
|
bool result;
|
2017-04-18 18:43:20 +00:00
|
|
|
bool mysql_gtid;
|
|
|
|
bool mariadb_gtid;
|
|
|
|
|
|
|
|
mysql_variable status[] = {
|
|
|
|
{"File", &filename},
|
|
|
|
{"Position", &position},
|
|
|
|
{"Executed_Gtid_Set", >id_executed},
|
|
|
|
{NULL, NULL}
|
|
|
|
};
|
|
|
|
|
|
|
|
mysql_variable vars[] = {
|
|
|
|
{"gtid_mode", >id_mode},
|
|
|
|
{"gtid_current_pos", >id_current_pos},
|
|
|
|
{NULL, NULL}
|
|
|
|
};
|
|
|
|
|
|
|
|
read_mysql_variables(connection, "SHOW MASTER STATUS", status, false);
|
|
|
|
read_mysql_variables(connection, "SHOW VARIABLES", vars, true);
|
|
|
|
|
2022-06-28 09:16:31 +02:00
|
|
|
if (filename == NULL || position == NULL) {
|
2023-02-03 07:42:59 +04:00
|
|
|
/* Do not create MB_BINLOG_INFO if binary
|
2022-06-28 09:16:31 +02:00
|
|
|
log is disabled */
|
|
|
|
result = true;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
mysql_gtid = ((gtid_mode != NULL) && (strcmp(gtid_mode, "ON") == 0));
|
|
|
|
mariadb_gtid = (gtid_current_pos != NULL);
|
2017-04-18 18:43:20 +00:00
|
|
|
|
2022-06-28 09:16:31 +02:00
|
|
|
gtid = (gtid_executed != NULL ? gtid_executed : gtid_current_pos);
|
2017-04-18 18:43:20 +00:00
|
|
|
|
2022-06-28 09:16:31 +02:00
|
|
|
if (mariadb_gtid || mysql_gtid) {
|
2017-04-18 18:43:20 +00:00
|
|
|
ut_a(asprintf(&mysql_binlog_position,
|
|
|
|
"filename '%s', position '%s', "
|
|
|
|
"GTID of the last change '%s'",
|
|
|
|
filename, position, gtid) != -1);
|
2023-06-05 20:13:06 +02:00
|
|
|
result = datasink->backup_file_printf(MB_BINLOG_INFO,
|
2022-06-28 09:16:31 +02:00
|
|
|
"%s\t%s\t%s\n", filename, position,
|
|
|
|
gtid);
|
2017-04-18 18:43:20 +00:00
|
|
|
} else {
|
|
|
|
ut_a(asprintf(&mysql_binlog_position,
|
|
|
|
"filename '%s', position '%s'",
|
|
|
|
filename, position) != -1);
|
2023-06-05 20:13:06 +02:00
|
|
|
result = datasink->backup_file_printf(MB_BINLOG_INFO,
|
2022-06-28 09:16:31 +02:00
|
|
|
"%s\t%s\n", filename, position);
|
2017-04-18 18:43:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
cleanup:
|
MDEV-27524: Incorrect binlogs after Galera SST using rsync and mariabackup
This commit adds correct handling of binlogs for SST using rsync
or mariabackup. Before this fix, binlogs were handled incorrectly -
- only one (last) binary log file was transferred during SST, which
then led to various failures (for example, when trying to list all
events from the binary log). These bugs were long masked by flaws
in the primitive binlogs handling code in the SST scripts, which
causing binary logs files to be erased after transfer or not added
to the binlog index on the joiner node. Now the correct transfer
of all binary logs (not just the last of the binary log files) has
been implemented both for the rsync (at the script level) and for
the mariabackup (at the level of the main utility code).
This commit also adds a new sst_max_binlogs=<n> parameter, which
can be located in the [sst] section or in the [xtrabackup] section
(historically, supported for mariabackup only, not for rsync), or
in one of the server sections. This parameter specifies the number
of binary log files to be sent to the joiner node during SST. This
option is added for compatibility with old SST scripting behavior,
which can be emulated by setting the sst_max_binlogs=1 (although
in general this can cause problems for the reasons described above).
In addition, setting the sst_max_binlogs=0 can be used to suppress
the transmission of binary logs to the joiner nodes during SST
(although sometimes a single file with the current binary log can
still be transmitted to the joiner, even with sst_max_binlogs=0,
because this sometimes necessary in modes that involve the use of
GTIDs with Galera).
Also, this commit ensures correct handling of paths to various
innodb files and directories in the SST scripts, and fixes some
problems with this that existed in mariabackup utility (which
were associated with incorrect handling of the innodb_data_dir
parameter in some scenarios).
In addition, this commit contains the following enhancements:
1) Added tests for mtr, which check the correct work with binlogs
after SST (using rsync and mariabackup);
2) Added correct handling of slashes at the end of all paths that
the SST script receives as parameters;
3) Improved parsing code for --mysqld-args parameters. Now it
correctly processes the sequence "--" after the name of the
one-letter option;
4) Checking the secret signature during joiner authentication
is made independent of presence of bash (as a unix shell)
in the system and diff utility no longer needed to check
certificates compliance;
5) All directories that are necessary for the correct placement
of various logs are automatically created by SST scripts in
advance (before running mariabackup on the joiner node);
6) Removal of old binary logs on joiner is done using the binlog
index (if it exists) (not only by fixed pattern that based
on the current binlog name, as before);
7) Paths for placing binary logs are correctly processed if they
are set as relative paths (to the datadir);
8) SST scripts are made even more resistant to spaces in filenames
(now for binlogs);
9) In case of failure, SST scripts now always end with an exit
code other than zero;
10) SST script for rsync now correctly create a tar file with
the binlogs, even if the paths to them (in the binlog index
file) are specified as a mix of absolute and relative paths,
and even if they do not match with the datadir path specified
in the current configuration settings.
2022-02-22 10:45:06 +01:00
|
|
|
free_mysql_variables(status);
|
2022-06-28 09:16:31 +02:00
|
|
|
free_mysql_variables(vars);
|
2017-04-18 18:43:20 +00:00
|
|
|
|
|
|
|
return(result);
|
|
|
|
}
|
|
|
|
|
MDEV-12548 Initial implementation of Mariabackup for MariaDB 10.2
InnoDB I/O and buffer pool interfaces and the redo log format
have been changed between MariaDB 10.1 and 10.2, and the backup
code has to be adjusted accordingly.
The code has been simplified, and many memory leaks have been fixed.
Instead of the file name xtrabackup_logfile, the file name ib_logfile0
is being used for the copy of the redo log. Unnecessary InnoDB startup and
shutdown and some unnecessary threads have been removed.
Some help was provided by Vladislav Vaintroub.
Parameters have been cleaned up and aligned with those of MariaDB 10.2.
The --dbug option has been added, so that in debug builds,
--dbug=d,ib_log can be specified to enable diagnostic messages
for processing redo log entries.
By default, innodb_doublewrite=OFF, so that --prepare works faster.
If more crash-safety for --prepare is needed, double buffering
can be enabled.
The parameter innodb_log_checksums=OFF can be used to ignore redo log
checksums in --backup.
Some messages have been cleaned up.
Unless --export is specified, Mariabackup will not deal with undo log.
The InnoDB mini-transaction redo log is not only about user-level
transactions; it is actually about mini-transactions. To avoid confusion,
call it the redo log, not transaction log.
We disable any undo log processing in --prepare.
Because MariaDB 10.2 supports indexed virtual columns, the
undo log processing would need to be able to evaluate virtual column
expressions. To reduce the amount of code dependencies, we will not
process any undo log in prepare.
This means that the --export option must be disabled for now.
This also means that the following options are redundant
and have been removed:
xtrabackup --apply-log-only
innobackupex --redo-only
In addition to disabling any undo log processing, we will disable any
further changes to data pages during --prepare, including the change
buffer merge. This means that restoring incremental backups should
reliably work even when change buffering is being used on the server.
Because of this, preparing a backup will not generate any further
redo log, and the redo log file can be safely deleted. (If the
--export option is enabled in the future, it must generate redo log
when processing undo logs and buffered changes.)
In --prepare, we cannot easily know if a partial backup was used,
especially when restoring a series of incremental backups. So, we
simply warn about any missing files, and ignore the redo log for them.
FIXME: Enable the --export option.
FIXME: Improve the handling of the MLOG_INDEX_LOAD record, and write
a test that initiates a backup while an ALGORITHM=INPLACE operation
is creating indexes or rebuilding a table. An error should be detected
when preparing the backup.
FIXME: In --incremental --prepare, xtrabackup_apply_delta() should
ensure that if FSP_SIZE is modified, the file size will be adjusted
accordingly.
2017-06-30 10:49:37 +03:00
|
|
|
struct escape_and_quote
|
2017-04-18 19:05:57 +00:00
|
|
|
{
|
MDEV-12548 Initial implementation of Mariabackup for MariaDB 10.2
InnoDB I/O and buffer pool interfaces and the redo log format
have been changed between MariaDB 10.1 and 10.2, and the backup
code has to be adjusted accordingly.
The code has been simplified, and many memory leaks have been fixed.
Instead of the file name xtrabackup_logfile, the file name ib_logfile0
is being used for the copy of the redo log. Unnecessary InnoDB startup and
shutdown and some unnecessary threads have been removed.
Some help was provided by Vladislav Vaintroub.
Parameters have been cleaned up and aligned with those of MariaDB 10.2.
The --dbug option has been added, so that in debug builds,
--dbug=d,ib_log can be specified to enable diagnostic messages
for processing redo log entries.
By default, innodb_doublewrite=OFF, so that --prepare works faster.
If more crash-safety for --prepare is needed, double buffering
can be enabled.
The parameter innodb_log_checksums=OFF can be used to ignore redo log
checksums in --backup.
Some messages have been cleaned up.
Unless --export is specified, Mariabackup will not deal with undo log.
The InnoDB mini-transaction redo log is not only about user-level
transactions; it is actually about mini-transactions. To avoid confusion,
call it the redo log, not transaction log.
We disable any undo log processing in --prepare.
Because MariaDB 10.2 supports indexed virtual columns, the
undo log processing would need to be able to evaluate virtual column
expressions. To reduce the amount of code dependencies, we will not
process any undo log in prepare.
This means that the --export option must be disabled for now.
This also means that the following options are redundant
and have been removed:
xtrabackup --apply-log-only
innobackupex --redo-only
In addition to disabling any undo log processing, we will disable any
further changes to data pages during --prepare, including the change
buffer merge. This means that restoring incremental backups should
reliably work even when change buffering is being used on the server.
Because of this, preparing a backup will not generate any further
redo log, and the redo log file can be safely deleted. (If the
--export option is enabled in the future, it must generate redo log
when processing undo logs and buffered changes.)
In --prepare, we cannot easily know if a partial backup was used,
especially when restoring a series of incremental backups. So, we
simply warn about any missing files, and ignore the redo log for them.
FIXME: Enable the --export option.
FIXME: Improve the handling of the MLOG_INDEX_LOAD record, and write
a test that initiates a backup while an ALGORITHM=INPLACE operation
is creating indexes or rebuilding a table. An error should be detected
when preparing the backup.
FIXME: In --incremental --prepare, xtrabackup_apply_delta() should
ensure that if FSP_SIZE is modified, the file size will be adjusted
accordingly.
2017-06-30 10:49:37 +03:00
|
|
|
escape_and_quote(MYSQL *mysql, const char *str)
|
|
|
|
: mysql(mysql), str(str) {}
|
|
|
|
MYSQL * const mysql;
|
|
|
|
const char * const str;
|
|
|
|
};
|
|
|
|
|
|
|
|
static
|
|
|
|
std::ostream&
|
|
|
|
operator<<(std::ostream& s, const escape_and_quote& eq)
|
|
|
|
{
|
|
|
|
if (!eq.str)
|
|
|
|
return s << "NULL";
|
|
|
|
s << '\'';
|
|
|
|
size_t len = strlen(eq.str);
|
|
|
|
char* escaped = (char *)alloca(2 * len + 1);
|
2018-02-06 12:55:58 +00:00
|
|
|
len = mysql_real_escape_string(eq.mysql, escaped, eq.str, (ulong)len);
|
MDEV-12548 Initial implementation of Mariabackup for MariaDB 10.2
InnoDB I/O and buffer pool interfaces and the redo log format
have been changed between MariaDB 10.1 and 10.2, and the backup
code has to be adjusted accordingly.
The code has been simplified, and many memory leaks have been fixed.
Instead of the file name xtrabackup_logfile, the file name ib_logfile0
is being used for the copy of the redo log. Unnecessary InnoDB startup and
shutdown and some unnecessary threads have been removed.
Some help was provided by Vladislav Vaintroub.
Parameters have been cleaned up and aligned with those of MariaDB 10.2.
The --dbug option has been added, so that in debug builds,
--dbug=d,ib_log can be specified to enable diagnostic messages
for processing redo log entries.
By default, innodb_doublewrite=OFF, so that --prepare works faster.
If more crash-safety for --prepare is needed, double buffering
can be enabled.
The parameter innodb_log_checksums=OFF can be used to ignore redo log
checksums in --backup.
Some messages have been cleaned up.
Unless --export is specified, Mariabackup will not deal with undo log.
The InnoDB mini-transaction redo log is not only about user-level
transactions; it is actually about mini-transactions. To avoid confusion,
call it the redo log, not transaction log.
We disable any undo log processing in --prepare.
Because MariaDB 10.2 supports indexed virtual columns, the
undo log processing would need to be able to evaluate virtual column
expressions. To reduce the amount of code dependencies, we will not
process any undo log in prepare.
This means that the --export option must be disabled for now.
This also means that the following options are redundant
and have been removed:
xtrabackup --apply-log-only
innobackupex --redo-only
In addition to disabling any undo log processing, we will disable any
further changes to data pages during --prepare, including the change
buffer merge. This means that restoring incremental backups should
reliably work even when change buffering is being used on the server.
Because of this, preparing a backup will not generate any further
redo log, and the redo log file can be safely deleted. (If the
--export option is enabled in the future, it must generate redo log
when processing undo logs and buffered changes.)
In --prepare, we cannot easily know if a partial backup was used,
especially when restoring a series of incremental backups. So, we
simply warn about any missing files, and ignore the redo log for them.
FIXME: Enable the --export option.
FIXME: Improve the handling of the MLOG_INDEX_LOAD record, and write
a test that initiates a backup while an ALGORITHM=INPLACE operation
is creating indexes or rebuilding a table. An error should be detected
when preparing the backup.
FIXME: In --incremental --prepare, xtrabackup_apply_delta() should
ensure that if FSP_SIZE is modified, the file size will be adjusted
accordingly.
2017-06-30 10:49:37 +03:00
|
|
|
s << std::string(escaped, len);
|
|
|
|
s << '\'';
|
|
|
|
return s;
|
2017-04-18 19:05:57 +00:00
|
|
|
}
|
2017-04-18 18:43:20 +00:00
|
|
|
|
|
|
|
/*********************************************************************//**
|
2023-02-03 07:42:59 +04:00
|
|
|
Writes MB_INFO file and if backup_history is enable creates
|
2022-09-09 12:02:16 +04:00
|
|
|
mysql.mariabackup_history and writes a new history record to the
|
2017-04-18 18:43:20 +00:00
|
|
|
table containing all the history info particular to the just completed
|
|
|
|
backup. */
|
|
|
|
bool
|
2023-04-12 11:40:46 +04:00
|
|
|
write_xtrabackup_info(ds_ctxt *datasink,
|
|
|
|
MYSQL *connection, const char * filename, bool history,
|
|
|
|
bool stream)
|
2017-04-18 18:43:20 +00:00
|
|
|
{
|
2017-04-18 19:05:57 +00:00
|
|
|
|
2019-05-27 13:52:27 +02:00
|
|
|
bool result = true;
|
|
|
|
FILE *fp = NULL;
|
2017-04-18 18:43:20 +00:00
|
|
|
char *uuid = NULL;
|
|
|
|
char *server_version = NULL;
|
|
|
|
char buf_start_time[100];
|
|
|
|
char buf_end_time[100];
|
|
|
|
tm tm;
|
MDEV-12548 Initial implementation of Mariabackup for MariaDB 10.2
InnoDB I/O and buffer pool interfaces and the redo log format
have been changed between MariaDB 10.1 and 10.2, and the backup
code has to be adjusted accordingly.
The code has been simplified, and many memory leaks have been fixed.
Instead of the file name xtrabackup_logfile, the file name ib_logfile0
is being used for the copy of the redo log. Unnecessary InnoDB startup and
shutdown and some unnecessary threads have been removed.
Some help was provided by Vladislav Vaintroub.
Parameters have been cleaned up and aligned with those of MariaDB 10.2.
The --dbug option has been added, so that in debug builds,
--dbug=d,ib_log can be specified to enable diagnostic messages
for processing redo log entries.
By default, innodb_doublewrite=OFF, so that --prepare works faster.
If more crash-safety for --prepare is needed, double buffering
can be enabled.
The parameter innodb_log_checksums=OFF can be used to ignore redo log
checksums in --backup.
Some messages have been cleaned up.
Unless --export is specified, Mariabackup will not deal with undo log.
The InnoDB mini-transaction redo log is not only about user-level
transactions; it is actually about mini-transactions. To avoid confusion,
call it the redo log, not transaction log.
We disable any undo log processing in --prepare.
Because MariaDB 10.2 supports indexed virtual columns, the
undo log processing would need to be able to evaluate virtual column
expressions. To reduce the amount of code dependencies, we will not
process any undo log in prepare.
This means that the --export option must be disabled for now.
This also means that the following options are redundant
and have been removed:
xtrabackup --apply-log-only
innobackupex --redo-only
In addition to disabling any undo log processing, we will disable any
further changes to data pages during --prepare, including the change
buffer merge. This means that restoring incremental backups should
reliably work even when change buffering is being used on the server.
Because of this, preparing a backup will not generate any further
redo log, and the redo log file can be safely deleted. (If the
--export option is enabled in the future, it must generate redo log
when processing undo logs and buffered changes.)
In --prepare, we cannot easily know if a partial backup was used,
especially when restoring a series of incremental backups. So, we
simply warn about any missing files, and ignore the redo log for them.
FIXME: Enable the --export option.
FIXME: Improve the handling of the MLOG_INDEX_LOAD record, and write
a test that initiates a backup while an ALGORITHM=INPLACE operation
is creating indexes or rebuilding a table. An error should be detected
when preparing the backup.
FIXME: In --incremental --prepare, xtrabackup_apply_delta() should
ensure that if FSP_SIZE is modified, the file size will be adjusted
accordingly.
2017-06-30 10:49:37 +03:00
|
|
|
std::ostringstream oss;
|
2017-04-18 18:43:20 +00:00
|
|
|
const char *xb_stream_name[] = {"file", "tar", "xbstream"};
|
2017-04-18 19:05:57 +00:00
|
|
|
|
2017-04-18 18:43:20 +00:00
|
|
|
uuid = read_mysql_one_value(connection, "SELECT UUID()");
|
|
|
|
server_version = read_mysql_one_value(connection, "SELECT VERSION()");
|
|
|
|
localtime_r(&history_start_time, &tm);
|
|
|
|
strftime(buf_start_time, sizeof(buf_start_time),
|
|
|
|
"%Y-%m-%d %H:%M:%S", &tm);
|
|
|
|
history_end_time = time(NULL);
|
|
|
|
localtime_r(&history_end_time, &tm);
|
|
|
|
strftime(buf_end_time, sizeof(buf_end_time),
|
|
|
|
"%Y-%m-%d %H:%M:%S", &tm);
|
2017-04-18 19:05:57 +00:00
|
|
|
bool is_partial = (xtrabackup_tables
|
|
|
|
|| xtrabackup_tables_file
|
|
|
|
|| xtrabackup_databases
|
2017-04-19 13:09:03 +00:00
|
|
|
|| xtrabackup_databases_file
|
|
|
|
|| xtrabackup_tables_exclude
|
|
|
|
|| xtrabackup_databases_exclude
|
|
|
|
);
|
2017-04-18 19:05:57 +00:00
|
|
|
|
2019-05-27 13:52:27 +02:00
|
|
|
char *buf = NULL;
|
|
|
|
int buf_len = asprintf(&buf,
|
2017-04-18 18:43:20 +00:00
|
|
|
"uuid = %s\n"
|
|
|
|
"name = %s\n"
|
|
|
|
"tool_name = %s\n"
|
|
|
|
"tool_command = %s\n"
|
|
|
|
"tool_version = %s\n"
|
|
|
|
"ibbackup_version = %s\n"
|
|
|
|
"server_version = %s\n"
|
|
|
|
"start_time = %s\n"
|
|
|
|
"end_time = %s\n"
|
|
|
|
"lock_time = %d\n"
|
|
|
|
"binlog_pos = %s\n"
|
2019-05-27 13:52:27 +02:00
|
|
|
"innodb_from_lsn = " LSN_PF "\n"
|
|
|
|
"innodb_to_lsn = " LSN_PF "\n"
|
2017-04-18 18:43:20 +00:00
|
|
|
"partial = %s\n"
|
|
|
|
"incremental = %s\n"
|
|
|
|
"format = %s\n"
|
2017-05-26 12:13:48 +03:00
|
|
|
"compressed = %s\n",
|
2017-04-18 18:43:20 +00:00
|
|
|
uuid, /* uuid */
|
|
|
|
opt_history ? opt_history : "", /* name */
|
|
|
|
tool_name, /* tool_name */
|
|
|
|
tool_args, /* tool_command */
|
2017-04-18 19:05:57 +00:00
|
|
|
MYSQL_SERVER_VERSION, /* tool_version */
|
|
|
|
MYSQL_SERVER_VERSION, /* ibbackup_version */
|
2017-04-18 18:43:20 +00:00
|
|
|
server_version, /* server_version */
|
|
|
|
buf_start_time, /* start_time */
|
|
|
|
buf_end_time, /* end_time */
|
2017-04-18 19:05:57 +00:00
|
|
|
(int)history_lock_time, /* lock_time */
|
2017-04-18 18:43:20 +00:00
|
|
|
mysql_binlog_position ?
|
|
|
|
mysql_binlog_position : "", /* binlog_pos */
|
2019-05-27 13:52:27 +02:00
|
|
|
incremental_lsn,
|
|
|
|
/* innodb_from_lsn */
|
|
|
|
metadata_to_lsn,
|
|
|
|
/* innodb_to_lsn */
|
2017-04-18 19:05:57 +00:00
|
|
|
is_partial? "Y" : "N",
|
2017-04-18 18:43:20 +00:00
|
|
|
xtrabackup_incremental ? "Y" : "N", /* incremental */
|
|
|
|
xb_stream_name[xtrabackup_stream_fmt], /* format */
|
2017-05-26 12:13:48 +03:00
|
|
|
xtrabackup_compress ? "compressed" : "N"); /* compressed */
|
2019-05-27 13:52:27 +02:00
|
|
|
if (buf_len < 0) {
|
2023-02-03 07:42:59 +04:00
|
|
|
msg("Error: cannot generate " MB_INFO);
|
2019-05-27 13:52:27 +02:00
|
|
|
result = false;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (stream) {
|
2023-04-12 11:40:46 +04:00
|
|
|
datasink->backup_file_printf(filename, "%s", buf);
|
2019-05-27 13:52:27 +02:00
|
|
|
} else {
|
|
|
|
fp = fopen(filename, "w");
|
|
|
|
if (!fp) {
|
|
|
|
msg("Error: cannot open %s", filename);
|
|
|
|
result = false;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
if (fwrite(buf, buf_len, 1, fp) < 1) {
|
|
|
|
result = false;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
}
|
2017-04-18 18:43:20 +00:00
|
|
|
|
2018-02-15 21:08:18 +00:00
|
|
|
if (!history) {
|
2017-04-18 18:43:20 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
xb_mysql_query(connection,
|
2022-09-09 12:02:16 +04:00
|
|
|
"CREATE TABLE IF NOT EXISTS " XB_HISTORY_TABLE "("
|
2017-04-18 18:43:20 +00:00
|
|
|
"uuid VARCHAR(40) NOT NULL PRIMARY KEY,"
|
|
|
|
"name VARCHAR(255) DEFAULT NULL,"
|
|
|
|
"tool_name VARCHAR(255) DEFAULT NULL,"
|
|
|
|
"tool_command TEXT DEFAULT NULL,"
|
|
|
|
"tool_version VARCHAR(255) DEFAULT NULL,"
|
|
|
|
"ibbackup_version VARCHAR(255) DEFAULT NULL,"
|
|
|
|
"server_version VARCHAR(255) DEFAULT NULL,"
|
|
|
|
"start_time TIMESTAMP NULL DEFAULT NULL,"
|
|
|
|
"end_time TIMESTAMP NULL DEFAULT NULL,"
|
|
|
|
"lock_time BIGINT UNSIGNED DEFAULT NULL,"
|
|
|
|
"binlog_pos VARCHAR(128) DEFAULT NULL,"
|
|
|
|
"innodb_from_lsn BIGINT UNSIGNED DEFAULT NULL,"
|
|
|
|
"innodb_to_lsn BIGINT UNSIGNED DEFAULT NULL,"
|
|
|
|
"partial ENUM('Y', 'N') DEFAULT NULL,"
|
|
|
|
"incremental ENUM('Y', 'N') DEFAULT NULL,"
|
|
|
|
"format ENUM('file', 'tar', 'xbstream') DEFAULT NULL,"
|
2017-05-26 12:13:48 +03:00
|
|
|
"compressed ENUM('Y', 'N') DEFAULT NULL"
|
2017-04-18 18:43:20 +00:00
|
|
|
") CHARACTER SET utf8 ENGINE=innodb", false);
|
|
|
|
|
2017-04-18 19:05:57 +00:00
|
|
|
|
|
|
|
#define ESCAPE_BOOL(expr) ((expr)?"'Y'":"'N'")
|
|
|
|
|
2022-09-09 12:02:16 +04:00
|
|
|
oss << "insert into " XB_HISTORY_TABLE "("
|
2017-04-18 19:05:57 +00:00
|
|
|
<< "uuid, name, tool_name, tool_command, tool_version,"
|
|
|
|
<< "ibbackup_version, server_version, start_time, end_time,"
|
|
|
|
<< "lock_time, binlog_pos, innodb_from_lsn, innodb_to_lsn,"
|
2017-05-26 12:13:48 +03:00
|
|
|
<< "partial, incremental, format, compressed) "
|
|
|
|
<< "values("
|
2017-04-18 19:05:57 +00:00
|
|
|
<< escape_and_quote(connection, uuid) << ","
|
|
|
|
<< escape_and_quote(connection, opt_history) << ","
|
|
|
|
<< escape_and_quote(connection, tool_name) << ","
|
|
|
|
<< escape_and_quote(connection, tool_args) << ","
|
|
|
|
<< escape_and_quote(connection, MYSQL_SERVER_VERSION) << ","
|
|
|
|
<< escape_and_quote(connection, MYSQL_SERVER_VERSION) << ","
|
|
|
|
<< escape_and_quote(connection, server_version) << ","
|
|
|
|
<< "from_unixtime(" << history_start_time << "),"
|
|
|
|
<< "from_unixtime(" << history_end_time << "),"
|
|
|
|
<< history_lock_time << ","
|
|
|
|
<< escape_and_quote(connection, mysql_binlog_position) << ","
|
|
|
|
<< incremental_lsn << ","
|
|
|
|
<< metadata_to_lsn << ","
|
|
|
|
<< ESCAPE_BOOL(is_partial) << ","
|
|
|
|
<< ESCAPE_BOOL(xtrabackup_incremental)<< ","
|
|
|
|
<< escape_and_quote(connection,xb_stream_name[xtrabackup_stream_fmt]) <<","
|
2017-05-26 12:13:48 +03:00
|
|
|
<< ESCAPE_BOOL(xtrabackup_compress) << ")";
|
2017-04-18 19:05:57 +00:00
|
|
|
|
|
|
|
xb_mysql_query(mysql_connection, oss.str().c_str(), false);
|
2017-04-18 18:43:20 +00:00
|
|
|
|
|
|
|
cleanup:
|
|
|
|
|
|
|
|
free(uuid);
|
|
|
|
free(server_version);
|
2019-05-27 13:52:27 +02:00
|
|
|
free(buf);
|
|
|
|
if (fp)
|
|
|
|
fclose(fp);
|
2017-04-18 18:43:20 +00:00
|
|
|
|
2019-05-27 13:52:27 +02:00
|
|
|
return(result);
|
2017-04-18 18:43:20 +00:00
|
|
|
}
|
|
|
|
|
2017-04-18 19:05:57 +00:00
|
|
|
extern const char *innodb_checksum_algorithm_names[];
|
|
|
|
|
2018-04-11 23:07:23 +01:00
|
|
|
#ifdef _WIN32
|
|
|
|
#include <algorithm>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static std::string make_local_paths(const char *data_file_path)
|
|
|
|
{
|
|
|
|
if (strchr(data_file_path, '/') == 0
|
|
|
|
#ifdef _WIN32
|
|
|
|
&& strchr(data_file_path, '\\') == 0
|
|
|
|
#endif
|
|
|
|
){
|
|
|
|
return std::string(data_file_path);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::ostringstream buf;
|
|
|
|
|
|
|
|
char *dup = strdup(innobase_data_file_path);
|
|
|
|
ut_a(dup);
|
|
|
|
char *p;
|
|
|
|
char * token = strtok_r(dup, ";", &p);
|
|
|
|
while (token) {
|
|
|
|
if (buf.tellp())
|
|
|
|
buf << ";";
|
|
|
|
|
|
|
|
char *fname = strrchr(token, '/');
|
|
|
|
#ifdef _WIN32
|
|
|
|
fname = std::max(fname,strrchr(token, '\\'));
|
|
|
|
#endif
|
|
|
|
if (fname)
|
|
|
|
buf << fname + 1;
|
|
|
|
else
|
|
|
|
buf << token;
|
|
|
|
token = strtok_r(NULL, ";", &p);
|
|
|
|
}
|
|
|
|
free(dup);
|
|
|
|
return buf.str();
|
|
|
|
}
|
|
|
|
|
2023-04-12 11:40:46 +04:00
|
|
|
bool write_backup_config_file(ds_ctxt *datasink)
|
2017-04-18 18:43:20 +00:00
|
|
|
{
|
2023-04-12 11:40:46 +04:00
|
|
|
int rc= datasink->backup_file_printf("backup-my.cnf",
|
2022-08-26 10:20:26 +10:00
|
|
|
"# This options file was generated by innobackupex.\n\n"
|
|
|
|
"# The server\n"
|
2017-04-18 18:43:20 +00:00
|
|
|
"[mysqld]\n"
|
|
|
|
"innodb_checksum_algorithm=%s\n"
|
|
|
|
"innodb_data_file_path=%s\n"
|
MDEV-12548 Initial implementation of Mariabackup for MariaDB 10.2
InnoDB I/O and buffer pool interfaces and the redo log format
have been changed between MariaDB 10.1 and 10.2, and the backup
code has to be adjusted accordingly.
The code has been simplified, and many memory leaks have been fixed.
Instead of the file name xtrabackup_logfile, the file name ib_logfile0
is being used for the copy of the redo log. Unnecessary InnoDB startup and
shutdown and some unnecessary threads have been removed.
Some help was provided by Vladislav Vaintroub.
Parameters have been cleaned up and aligned with those of MariaDB 10.2.
The --dbug option has been added, so that in debug builds,
--dbug=d,ib_log can be specified to enable diagnostic messages
for processing redo log entries.
By default, innodb_doublewrite=OFF, so that --prepare works faster.
If more crash-safety for --prepare is needed, double buffering
can be enabled.
The parameter innodb_log_checksums=OFF can be used to ignore redo log
checksums in --backup.
Some messages have been cleaned up.
Unless --export is specified, Mariabackup will not deal with undo log.
The InnoDB mini-transaction redo log is not only about user-level
transactions; it is actually about mini-transactions. To avoid confusion,
call it the redo log, not transaction log.
We disable any undo log processing in --prepare.
Because MariaDB 10.2 supports indexed virtual columns, the
undo log processing would need to be able to evaluate virtual column
expressions. To reduce the amount of code dependencies, we will not
process any undo log in prepare.
This means that the --export option must be disabled for now.
This also means that the following options are redundant
and have been removed:
xtrabackup --apply-log-only
innobackupex --redo-only
In addition to disabling any undo log processing, we will disable any
further changes to data pages during --prepare, including the change
buffer merge. This means that restoring incremental backups should
reliably work even when change buffering is being used on the server.
Because of this, preparing a backup will not generate any further
redo log, and the redo log file can be safely deleted. (If the
--export option is enabled in the future, it must generate redo log
when processing undo logs and buffered changes.)
In --prepare, we cannot easily know if a partial backup was used,
especially when restoring a series of incremental backups. So, we
simply warn about any missing files, and ignore the redo log for them.
FIXME: Enable the --export option.
FIXME: Improve the handling of the MLOG_INDEX_LOAD record, and write
a test that initiates a backup while an ALGORITHM=INPLACE operation
is creating indexes or rebuilding a table. An error should be detected
when preparing the backup.
FIXME: In --incremental --prepare, xtrabackup_apply_delta() should
ensure that if FSP_SIZE is modified, the file size will be adjusted
accordingly.
2017-06-30 10:49:37 +03:00
|
|
|
"innodb_log_file_size=%llu\n"
|
2017-04-18 18:43:20 +00:00
|
|
|
"innodb_page_size=%lu\n"
|
|
|
|
"innodb_undo_directory=%s\n"
|
2021-07-24 21:42:03 +03:00
|
|
|
"innodb_undo_tablespaces=%u\n"
|
2019-03-08 16:00:08 +05:30
|
|
|
"innodb_compression_level=%u\n"
|
2017-04-18 18:43:20 +00:00
|
|
|
"%s%s\n"
|
2017-04-18 19:05:57 +00:00
|
|
|
"%s\n",
|
2017-04-18 18:43:20 +00:00
|
|
|
innodb_checksum_algorithm_names[srv_checksum_algorithm],
|
2018-04-11 23:07:23 +01:00
|
|
|
make_local_paths(innobase_data_file_path).c_str(),
|
MDEV-12548 Initial implementation of Mariabackup for MariaDB 10.2
InnoDB I/O and buffer pool interfaces and the redo log format
have been changed between MariaDB 10.1 and 10.2, and the backup
code has to be adjusted accordingly.
The code has been simplified, and many memory leaks have been fixed.
Instead of the file name xtrabackup_logfile, the file name ib_logfile0
is being used for the copy of the redo log. Unnecessary InnoDB startup and
shutdown and some unnecessary threads have been removed.
Some help was provided by Vladislav Vaintroub.
Parameters have been cleaned up and aligned with those of MariaDB 10.2.
The --dbug option has been added, so that in debug builds,
--dbug=d,ib_log can be specified to enable diagnostic messages
for processing redo log entries.
By default, innodb_doublewrite=OFF, so that --prepare works faster.
If more crash-safety for --prepare is needed, double buffering
can be enabled.
The parameter innodb_log_checksums=OFF can be used to ignore redo log
checksums in --backup.
Some messages have been cleaned up.
Unless --export is specified, Mariabackup will not deal with undo log.
The InnoDB mini-transaction redo log is not only about user-level
transactions; it is actually about mini-transactions. To avoid confusion,
call it the redo log, not transaction log.
We disable any undo log processing in --prepare.
Because MariaDB 10.2 supports indexed virtual columns, the
undo log processing would need to be able to evaluate virtual column
expressions. To reduce the amount of code dependencies, we will not
process any undo log in prepare.
This means that the --export option must be disabled for now.
This also means that the following options are redundant
and have been removed:
xtrabackup --apply-log-only
innobackupex --redo-only
In addition to disabling any undo log processing, we will disable any
further changes to data pages during --prepare, including the change
buffer merge. This means that restoring incremental backups should
reliably work even when change buffering is being used on the server.
Because of this, preparing a backup will not generate any further
redo log, and the redo log file can be safely deleted. (If the
--export option is enabled in the future, it must generate redo log
when processing undo logs and buffered changes.)
In --prepare, we cannot easily know if a partial backup was used,
especially when restoring a series of incremental backups. So, we
simply warn about any missing files, and ignore the redo log for them.
FIXME: Enable the --export option.
FIXME: Improve the handling of the MLOG_INDEX_LOAD record, and write
a test that initiates a backup while an ALGORITHM=INPLACE operation
is creating indexes or rebuilding a table. An error should be detected
when preparing the backup.
FIXME: In --incremental --prepare, xtrabackup_apply_delta() should
ensure that if FSP_SIZE is modified, the file size will be adjusted
accordingly.
2017-06-30 10:49:37 +03:00
|
|
|
srv_log_file_size,
|
2017-04-18 18:43:20 +00:00
|
|
|
srv_page_size,
|
|
|
|
srv_undo_dir,
|
|
|
|
srv_undo_tablespaces,
|
2019-03-08 16:00:08 +05:30
|
|
|
page_zip_level,
|
2017-04-18 18:43:20 +00:00
|
|
|
innobase_buffer_pool_filename ?
|
|
|
|
"innodb_buffer_pool_filename=" : "",
|
|
|
|
innobase_buffer_pool_filename ?
|
2017-04-18 19:05:57 +00:00
|
|
|
innobase_buffer_pool_filename : "",
|
2021-10-22 10:14:57 +02:00
|
|
|
xb_plugin_get_config());
|
2017-04-18 19:05:57 +00:00
|
|
|
return rc;
|
2017-04-18 18:43:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static
|
|
|
|
char *make_argv(char *buf, size_t len, int argc, char **argv)
|
|
|
|
{
|
|
|
|
size_t left= len;
|
|
|
|
const char *arg;
|
|
|
|
|
|
|
|
buf[0]= 0;
|
|
|
|
++argv; --argc;
|
|
|
|
while (argc > 0 && left > 0)
|
|
|
|
{
|
|
|
|
arg = *argv;
|
|
|
|
if (strncmp(*argv, "--password", strlen("--password")) == 0) {
|
|
|
|
arg = "--password=...";
|
|
|
|
}
|
2017-11-11 23:07:24 +02:00
|
|
|
left-= snprintf(buf + len - left, left,
|
|
|
|
"%s%c", arg, argc > 1 ? ' ' : 0);
|
2017-04-18 18:43:20 +00:00
|
|
|
++argv; --argc;
|
|
|
|
}
|
|
|
|
|
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
capture_tool_command(int argc, char **argv)
|
|
|
|
{
|
|
|
|
/* capture tool name tool args */
|
|
|
|
tool_name = strrchr(argv[0], '/');
|
|
|
|
tool_name = tool_name ? tool_name + 1 : argv[0];
|
|
|
|
|
|
|
|
make_argv(tool_args, sizeof(tool_args), argc, argv);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool
|
|
|
|
select_history()
|
|
|
|
{
|
|
|
|
if (opt_incremental_history_name || opt_incremental_history_uuid) {
|
|
|
|
if (!select_incremental_lsn_from_history(
|
|
|
|
&incremental_lsn)) {
|
|
|
|
return(false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
flush_changed_page_bitmaps()
|
|
|
|
{
|
|
|
|
if (xtrabackup_incremental && have_changed_page_bitmaps &&
|
|
|
|
!xtrabackup_incremental_force_scan) {
|
|
|
|
xb_mysql_query(mysql_connection,
|
|
|
|
"FLUSH NO_WRITE_TO_BINLOG CHANGED_PAGE_BITMAPS", false);
|
|
|
|
}
|
|
|
|
return(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*********************************************************************//**
|
2022-08-26 10:20:26 +10:00
|
|
|
Deallocate memory, disconnect from server, etc.
|
2017-04-18 18:43:20 +00:00
|
|
|
@return true on success. */
|
|
|
|
void
|
|
|
|
backup_cleanup()
|
|
|
|
{
|
|
|
|
free(mysql_slave_position);
|
|
|
|
free(mysql_binlog_position);
|
2022-06-28 09:16:31 +02:00
|
|
|
free(buffer_pool_filename);
|
2017-04-18 18:43:20 +00:00
|
|
|
|
|
|
|
if (mysql_connection) {
|
|
|
|
mysql_close(mysql_connection);
|
|
|
|
}
|
|
|
|
}
|
2017-09-11 16:45:36 +00:00
|
|
|
|
|
|
|
|
|
|
|
static MYSQL *mdl_con = NULL;
|
|
|
|
|
2018-10-31 17:19:25 +01:00
|
|
|
std::map<ulint, std::string> spaceid_to_tablename;
|
|
|
|
|
2017-09-11 16:45:36 +00:00
|
|
|
void
|
|
|
|
mdl_lock_init()
|
|
|
|
{
|
|
|
|
mdl_con = xb_mysql_connect();
|
2018-10-31 17:19:25 +01:00
|
|
|
if (!mdl_con)
|
2017-09-11 16:45:36 +00:00
|
|
|
{
|
2018-10-31 17:19:25 +01:00
|
|
|
msg("FATAL: cannot create connection for MDL locks");
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
const char *query =
|
|
|
|
"SELECT NAME, SPACE FROM INFORMATION_SCHEMA.INNODB_SYS_TABLES WHERE NAME LIKE '%%/%%'";
|
|
|
|
|
|
|
|
MYSQL_RES *mysql_result = xb_mysql_query(mdl_con, query, true, true);
|
|
|
|
while (MYSQL_ROW row = mysql_fetch_row(mysql_result)) {
|
|
|
|
int err;
|
|
|
|
ulint id = (ulint)my_strtoll10(row[1], 0, &err);
|
|
|
|
spaceid_to_tablename[id] = ut_get_name(0, row[0]);
|
2017-09-11 16:45:36 +00:00
|
|
|
}
|
2018-10-31 17:19:25 +01:00
|
|
|
mysql_free_result(mysql_result);
|
|
|
|
|
|
|
|
xb_mysql_query(mdl_con, "BEGIN", false, true);
|
2017-09-11 16:45:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
mdl_lock_table(ulint space_id)
|
|
|
|
{
|
2018-10-31 17:19:25 +01:00
|
|
|
if (space_id == 0)
|
|
|
|
return;
|
2017-09-11 16:45:36 +00:00
|
|
|
|
2018-10-31 17:19:25 +01:00
|
|
|
std::string full_table_name = spaceid_to_tablename[space_id];
|
2017-09-11 16:45:36 +00:00
|
|
|
|
2018-10-31 17:19:25 +01:00
|
|
|
DBUG_EXECUTE_IF("rename_during_mdl_lock_table",
|
|
|
|
if (full_table_name == "`test`.`t1`")
|
|
|
|
xb_mysql_query(mysql_connection, "RENAME TABLE test.t1 to test.t2", false, true);
|
|
|
|
);
|
2017-09-11 16:45:36 +00:00
|
|
|
|
2018-10-31 17:19:25 +01:00
|
|
|
std::ostringstream lock_query;
|
|
|
|
lock_query << "SELECT 1 FROM " << full_table_name << " LIMIT 0";
|
2019-01-14 22:28:23 +01:00
|
|
|
msg("Locking MDL for %s", full_table_name.c_str());
|
2018-10-31 17:19:25 +01:00
|
|
|
if (mysql_query(mdl_con, lock_query.str().c_str())) {
|
2019-01-14 22:28:23 +01:00
|
|
|
msg("Warning : locking MDL failed for space id %zu, name %s", space_id, full_table_name.c_str());
|
2018-10-31 17:19:25 +01:00
|
|
|
} else {
|
2018-08-09 15:06:52 +01:00
|
|
|
MYSQL_RES *r = mysql_store_result(mdl_con);
|
|
|
|
mysql_free_result(r);
|
2017-09-11 16:45:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
mdl_unlock_all()
|
|
|
|
{
|
2019-01-14 22:28:23 +01:00
|
|
|
msg("Unlocking MDL for all tables");
|
2017-09-11 16:45:36 +00:00
|
|
|
xb_mysql_query(mdl_con, "COMMIT", false, true);
|
|
|
|
mysql_close(mdl_con);
|
2018-10-31 17:19:25 +01:00
|
|
|
spaceid_to_tablename.clear();
|
2017-09-11 16:45:36 +00:00
|
|
|
}
|