/* * Copyright (c) 2016 MariaDB Corporation Ab * * Use of this software is governed by the Business Source License included * in the LICENSE.TXT file and at www.mariadb.com/bsl11. * * Change Date: 2024-06-02 * * On the date above, in accordance with the Business Source License, use * of this software will be governed by version 2 or later of the General * Public License. */ /** * @file A MariaDB replication cluster monitor */ #include "mariadbmon.hh" #include #include #include #include #include #include #include #include #include #include using std::string; using maxscale::string_printf; // Config parameter names const char* const CN_AUTO_FAILOVER = "auto_failover"; const char* const CN_SWITCHOVER_ON_LOW_DISK_SPACE = "switchover_on_low_disk_space"; const char* const CN_PROMOTION_SQL_FILE = "promotion_sql_file"; const char* const CN_DEMOTION_SQL_FILE = "demotion_sql_file"; const char* const CN_HANDLE_EVENTS = "handle_events"; static const char CN_AUTO_REJOIN[] = "auto_rejoin"; static const char CN_FAILCOUNT[] = "failcount"; static const char CN_ENFORCE_READONLY[] = "enforce_read_only_slaves"; static const char CN_NO_PROMOTE_SERVERS[] = "servers_no_promotion"; static const char CN_FAILOVER_TIMEOUT[] = "failover_timeout"; static const char CN_SWITCHOVER_TIMEOUT[] = "switchover_timeout"; static const char CN_DETECT_STANDALONE_MASTER[] = "detect_standalone_master"; static const char CN_MAINTENANCE_ON_LOW_DISK_SPACE[] = "maintenance_on_low_disk_space"; static const char CN_ASSUME_UNIQUE_HOSTNAMES[] = "assume_unique_hostnames"; // Parameters for master failure verification and timeout static const char CN_VERIFY_MASTER_FAILURE[] = "verify_master_failure"; static const char CN_MASTER_FAILURE_TIMEOUT[] = "master_failure_timeout"; // Replication credentials parameters for failover/switchover/join static const char CN_REPLICATION_USER[] = "replication_user"; static const char CN_REPLICATION_PASSWORD[] = "replication_password"; static const char CN_REPLICATION_MASTER_SSL[] = "replication_master_ssl"; MariaDBMonitor::MariaDBMonitor(MXS_MONITOR* monitor) : maxscale::MonitorInstance(monitor) { } MariaDBMonitor::~MariaDBMonitor() { clear_server_info(); } /** * Reset and initialize server arrays and related data. */ void MariaDBMonitor::reset_server_info() { // If this monitor is being restarted, the server data needs to be freed. clear_server_info(); // Next, initialize the data. for (auto mon_server = m_monitor->monitored_servers; mon_server; mon_server = mon_server->next) { m_servers.push_back(new MariaDBServer(mon_server, m_servers.size(), m_assume_unique_hostnames, m_handle_event_scheduler)); } } void MariaDBMonitor::clear_server_info() { for (auto iter = m_servers.begin(); iter != m_servers.end(); iter++) { delete *iter; } // All MariaDBServer*:s are now invalid, as well as any dependant data. m_servers.clear(); m_servers_by_id.clear(); m_excluded_servers.clear(); assign_new_master(NULL); m_next_master = NULL; m_master_gtid_domain = GTID_DOMAIN_UNKNOWN; m_external_master_host.clear(); m_external_master_port = PORT_UNKNOWN; } void MariaDBMonitor::reset_node_index_info() { for (auto iter = m_servers.begin(); iter != m_servers.end(); iter++) { (*iter)->m_node.reset_indexes(); } } MariaDBServer* MariaDBMonitor::get_server(const std::string& host, int port) { // TODO: Do this with a map lookup MariaDBServer* found = NULL; for (MariaDBServer* server : m_servers) { SERVER* srv = server->m_server_base->server; if (host == srv->address && srv->port == port) { found = server; break; } } return found; } MariaDBServer* MariaDBMonitor::get_server(int64_t id) { auto found = m_servers_by_id.find(id); return (found != m_servers_by_id.end()) ? (*found).second : NULL; } MariaDBServer* MariaDBMonitor::get_server(MXS_MONITORED_SERVER* mon_server) { return get_server(mon_server->server); } MariaDBServer* MariaDBMonitor::get_server(SERVER* server) { for (auto iter : m_servers) { if (iter->m_server_base->server == server) { return iter; } } return NULL; } bool MariaDBMonitor::set_replication_credentials(const MXS_CONFIG_PARAMETER* params) { MXS_CONFIG_PARAMETER* mutable_params = const_cast(params); bool repl_user_exists = (config_get_param(mutable_params, CN_REPLICATION_USER) != nullptr); bool repl_pw_exists = (config_get_param(mutable_params, CN_REPLICATION_PASSWORD) != nullptr); // Because runtime modifications are performed 1-by-1, we must be less strict here and allow // partial setups. Password is not required even if username is set. This is contrary to the // general monitor username & pw, which are both required. Even this assumes that the username // is given first in a "maxadmin alter monitor"-command. string repl_user; string repl_pw; if (repl_user_exists) { repl_user = config_get_string(params, CN_REPLICATION_USER); if (repl_pw_exists) { // Ok, both set. repl_pw = config_get_string(params, CN_REPLICATION_PASSWORD); } // Password not set is ok. This needs to be accepted so that runtime modifications work. // Hopefully the password is set later on. } else { if (repl_pw_exists) { MXS_ERROR("'%s' is defined while '%s' is not. If performing an \"alter monitor\"-command, " "give '%s' first.", CN_REPLICATION_PASSWORD, CN_REPLICATION_USER, CN_REPLICATION_USER); return false; } else { // Ok, neither is set. Use monitor credentials. repl_user = m_monitor->user; repl_pw = m_monitor->password; } } m_replication_user = repl_user; char* decrypted = decrypt_password(repl_pw.c_str()); m_replication_password = decrypted; MXS_FREE(decrypted); return true; } MariaDBMonitor* MariaDBMonitor::create(MXS_MONITOR* monitor) { return new MariaDBMonitor(monitor); } /** * Load config parameters * * @param params Config parameters * @return True if settings are ok */ bool MariaDBMonitor::configure(const MXS_CONFIG_PARAMETER* params) { m_detect_stale_master = config_get_bool(params, "detect_stale_master"); m_detect_stale_slave = config_get_bool(params, "detect_stale_slave"); m_ignore_external_masters = config_get_bool(params, "ignore_external_masters"); m_detect_standalone_master = config_get_bool(params, CN_DETECT_STANDALONE_MASTER); m_assume_unique_hostnames = config_get_bool(params, CN_ASSUME_UNIQUE_HOSTNAMES); m_failcount = config_get_integer(params, CN_FAILCOUNT); m_failover_timeout = config_get_integer(params, CN_FAILOVER_TIMEOUT); m_switchover_timeout = config_get_integer(params, CN_SWITCHOVER_TIMEOUT); m_auto_failover = config_get_bool(params, CN_AUTO_FAILOVER); m_auto_rejoin = config_get_bool(params, CN_AUTO_REJOIN); m_enforce_read_only_slaves = config_get_bool(params, CN_ENFORCE_READONLY); m_verify_master_failure = config_get_bool(params, CN_VERIFY_MASTER_FAILURE); m_master_failure_timeout = config_get_integer(params, CN_MASTER_FAILURE_TIMEOUT); m_promote_sql_file = config_get_string(params, CN_PROMOTION_SQL_FILE); m_demote_sql_file = config_get_string(params, CN_DEMOTION_SQL_FILE); m_switchover_on_low_disk_space = config_get_bool(params, CN_SWITCHOVER_ON_LOW_DISK_SPACE); m_maintenance_on_low_disk_space = config_get_bool(params, CN_MAINTENANCE_ON_LOW_DISK_SPACE); m_handle_event_scheduler = config_get_bool(params, CN_HANDLE_EVENTS); m_replication_ssl = config_get_bool(params, CN_REPLICATION_MASTER_SSL); /* Reset all monitored state info. The server dependent values must be reset as servers could have been * added, removed and modified. */ reset_server_info(); m_excluded_servers.clear(); MXS_MONITORED_SERVER** excluded_array = NULL; int n_excluded = mon_config_get_servers(params, CN_NO_PROMOTE_SERVERS, m_monitor, &excluded_array); for (int i = 0; i < n_excluded; i++) { m_excluded_servers.push_back(get_server(excluded_array[i])); } MXS_FREE(excluded_array); bool settings_ok = true; if (!check_sql_files()) { settings_ok = false; } if (!set_replication_credentials(params)) { settings_ok = false; } if (!m_assume_unique_hostnames) { const char requires[] = "%s requires that %s is on."; if (m_auto_failover) { MXB_ERROR(requires, CN_AUTO_FAILOVER, CN_ASSUME_UNIQUE_HOSTNAMES); settings_ok = false; } if (m_switchover_on_low_disk_space) { MXB_ERROR(requires, CN_SWITCHOVER_ON_LOW_DISK_SPACE, CN_ASSUME_UNIQUE_HOSTNAMES); settings_ok = false; } if (m_auto_rejoin) { MXB_ERROR(requires, CN_AUTO_REJOIN, CN_ASSUME_UNIQUE_HOSTNAMES); settings_ok = false; } } return settings_ok; } void MariaDBMonitor::diagnostics(DCB* dcb) const { /* The problem with diagnostic printing is that some of the printed elements are array-like and their * length could change during a monitor loop. Such variables are protected by mutexes. Locking is * only required when the monitor thread writes to such a variable and when the admin thread is * reading it. */ mxb_assert(mxs_rworker_get_current() == mxs_rworker_get(MXS_RWORKER_MAIN)); dcb_printf(dcb, "%s", diagnostics_to_string().c_str()); } string MariaDBMonitor::diagnostics_to_string() const { using maxscale::string_printf; string rval; rval.reserve(1000); // Enough for basic output. rval += string_printf("Automatic failover: %s\n", m_auto_failover ? "Enabled" : "Disabled"); rval += string_printf("Failcount: %d\n", m_failcount); rval += string_printf("Failover timeout: %u\n", m_failover_timeout); rval += string_printf("Switchover timeout: %u\n", m_switchover_timeout); rval += string_printf("Automatic rejoin: %s\n", m_auto_rejoin ? "Enabled" : "Disabled"); rval += string_printf("Enforce read-only: %s\n", m_enforce_read_only_slaves ? "Enabled" : "Disabled"); rval += string_printf("Detect stale master: %s\n", m_detect_stale_master ? "Enabled" : "Disabled"); if (m_excluded_servers.size() > 0) { rval += string_printf("Non-promotable servers (failover): "); rval += string_printf("%s\n", monitored_servers_to_string(m_excluded_servers).c_str()); } rval += string_printf("\nServer information:\n-------------------\n\n"); for (auto iter = m_servers.begin(); iter != m_servers.end(); iter++) { rval += (*iter)->diagnostics() + "\n"; } return rval; } json_t* MariaDBMonitor::diagnostics_json() const { mxb_assert(mxs_rworker_get_current() == mxs_rworker_get(MXS_RWORKER_MAIN)); return to_json(); } json_t* MariaDBMonitor::to_json() const { json_t* rval = MonitorInstance::diagnostics_json(); // The m_master-pointer can be modified during a tick, but the pointed object cannot be deleted. auto master = mxb::atomic::load(&m_master, mxb::atomic::RELAXED); json_object_set_new(rval, "master", master == nullptr ? json_null() : json_string(master->name())); json_object_set_new(rval, "master_gtid_domain_id", m_master_gtid_domain == GTID_DOMAIN_UNKNOWN ? json_null() : json_integer(m_master_gtid_domain)); json_t* server_info = json_array(); for (MariaDBServer* server : m_servers) { json_array_append_new(server_info, server->to_json()); } json_object_set_new(rval, "server_info", server_info); return rval; } /** * Connect to and query/update a server. * * @param server The server to update */ void MariaDBMonitor::update_server(MariaDBServer* server) { MXS_MONITORED_SERVER* mon_srv = server->m_server_base; mxs_connect_result_t conn_status = mon_ping_or_connect_to_db(m_monitor, mon_srv); MYSQL* conn = mon_srv->con; // mon_ping_or_connect_to_db() may have reallocated the MYSQL struct. if (mon_connection_is_ok(conn_status)) { server->set_status(SERVER_RUNNING); if (conn_status == MONITOR_CONN_NEWCONN_OK) { // Is a new connection or a reconnection. Check server version. server->update_server_version(); } if (server->m_capabilities.basic_support || server->m_srv_type == MariaDBServer::server_type::BINLOG_ROUTER) { // Check permissions if permissions failed last time or if this is a new connection. if (server->had_status(SERVER_AUTH_ERROR) || conn_status == MONITOR_CONN_NEWCONN_OK) { server->check_permissions(); } // If permissions are ok, continue. if (!server->has_status(SERVER_AUTH_ERROR)) { if (should_update_disk_space_status(mon_srv)) { update_disk_space_status(mon_srv); } // Query MariaDBServer specific data server->monitor_server(); } } } else { // The server is not running. Clear some of the bits. User-set bits and some long-term bits // can stay. server->clear_status(SERVER_DOWN_CLEAR_BITS); if (conn_status == MONITOR_CONN_ACCESS_DENIED) { server->set_status(SERVER_AUTH_ERROR); } /* Log connect failure only once, that is, if server was RUNNING or MAINTENANCE during last * iteration. */ if (server->had_status(SERVER_RUNNING) || server->had_status(SERVER_MAINT)) { mon_log_connect_error(mon_srv, conn_status); } } /** Increase or reset the error count of the server. */ bool is_running = server->is_running(); bool in_maintenance = server->is_in_maintenance(); mon_srv->mon_err_count = (is_running || in_maintenance) ? 0 : mon_srv->mon_err_count + 1; } void MariaDBMonitor::pre_loop() { // MonitorInstance reads the journal and has the last known master in its m_master member variable. // Write the corresponding MariaDBServer into the class-specific m_master variable. auto journal_master = MonitorInstance::m_master; if (journal_master) { // This is somewhat questionable, as the journal only contains status bits but no actual topology // info. In a fringe case the actual queried topology may not match the journal data, freezing the // master to a suboptimal choice. assign_new_master(get_server(journal_master)); } /* This loop can be removed if/once the replication check code is inside tick. It's required so that * the monitor makes new connections when starting. */ for (MariaDBServer* server : m_servers) { if (server->m_server_base->con) { mysql_close(server->m_server_base->con); server->m_server_base->con = NULL; } } } void MariaDBMonitor::tick() { /* Update MXS_MONITORED_SERVER->pending_status. This is where the monitor loop writes it's findings. * Also, backup current status so that it can be compared to any deduced state. */ for (auto mon_srv = m_monitor->monitored_servers; mon_srv; mon_srv = mon_srv->next) { auto status = mon_srv->server->status; mon_srv->pending_status = status; mon_srv->mon_prev_status = status; } if (cluster_operation_disable_timer > 0) { cluster_operation_disable_timer--; } // Query all servers for their status. for (MariaDBServer* server : m_servers) { update_server(server); if (server->m_topology_changed) { m_cluster_topology_changed = true; server->m_topology_changed = false; } } // Topology needs to be rechecked if it has changed or if master is down. if (m_cluster_topology_changed || (m_master && m_master->is_down())) { update_topology(); m_cluster_topology_changed = false; // If cluster operations are enabled, check topology support and disable if needed. if (m_auto_failover || m_switchover_on_low_disk_space || m_auto_rejoin) { check_cluster_operations_support(); } } // Always re-assign master, slave etc bits as these depend on other factors outside topology // (e.g. slave sql state). assign_server_roles(); if (m_master != NULL && m_master->is_master()) { // Update cluster-wide values dependant on the current master. update_gtid_domain(); update_external_master(); } /* Set low disk space slaves to maintenance. This needs to happen after roles have been assigned. * Is not a real cluster operation, since nothing on the actual backends is changed. */ if (m_maintenance_on_low_disk_space) { set_low_disk_slaves_maintenance(); } // Sanity check. Master may not be both slave and master. mxb_assert(m_master == NULL || !m_master->has_status(SERVER_SLAVE | SERVER_MASTER)); // Update shared status. for (auto server : m_servers) { SERVER* srv = server->m_server_base->server; srv->rlag = server->m_replication_lag; srv->status = server->m_server_base->pending_status; } log_master_changes(); // Before exiting, we need to store the current master into the m_master // member variable of MonitorInstance so that the right server will be // stored to the journal. MonitorInstance::m_master = m_master ? m_master->m_server_base : NULL; } void MariaDBMonitor::process_state_changes() { MonitorInstance::process_state_changes(); m_cluster_modified = false; // Check for manual commands if (m_manual_cmd.command_waiting_exec) { // Looks like a command is waiting. Lock mutex, check again and wait for the condition variable. std::unique_lock lock(m_manual_cmd.mutex); if (m_manual_cmd.command_waiting_exec) { m_manual_cmd.has_command.wait(lock, [this] { return m_manual_cmd.command_waiting_exec; }); m_manual_cmd.method(); m_manual_cmd.command_waiting_exec = false; m_manual_cmd.result_waiting = true; // Manual command ran, signal the sender to continue. lock.unlock(); m_manual_cmd.has_result.notify_one(); } else { // There was no command after all. lock.unlock(); } } if (can_perform_cluster_ops()) { if (m_auto_failover) { handle_auto_failover(); } // Do not auto-join servers on this monitor loop if a failover (or any other cluster modification) // has been performed, as server states have not been updated yet. It will happen next iteration. if (m_auto_rejoin && cluster_can_be_joined() && can_perform_cluster_ops()) { // Check if any servers should be autojoined to the cluster and try to join them. handle_auto_rejoin(); } /* Check if any slave servers have read-only off and turn it on if user so wishes. Again, do not * perform this if cluster has been modified this loop since it may not be clear which server * should be a slave. */ if (m_enforce_read_only_slaves && can_perform_cluster_ops()) { enforce_read_only_on_slaves(); } /* Check if the master server is on low disk space and act on it. */ if (m_switchover_on_low_disk_space && can_perform_cluster_ops()) { handle_low_disk_space_master(); } } } /** * Save info on the master server's multimaster group, if any. This is required when checking for changes * in the topology. */ void MariaDBMonitor::update_master_cycle_info() { if (m_master) { int new_cycle_id = m_master->m_node.cycle; m_master_cycle_status.cycle_id = new_cycle_id; if (new_cycle_id == NodeData::CYCLE_NONE) { m_master_cycle_status.cycle_members.clear(); } else { m_master_cycle_status.cycle_members = m_cycles[new_cycle_id]; } } else { m_master_cycle_status.cycle_id = NodeData::CYCLE_NONE; m_master_cycle_status.cycle_members.clear(); } } void MariaDBMonitor::update_gtid_domain() { int64_t domain = m_master->m_gtid_domain_id; if (m_master_gtid_domain != GTID_DOMAIN_UNKNOWN && domain != m_master_gtid_domain) { MXS_NOTICE("Gtid domain id of master has changed: %" PRId64 " -> %" PRId64 ".", m_master_gtid_domain, domain); } m_master_gtid_domain = domain; } void MariaDBMonitor::update_external_master() { if (m_master->is_slave_of_ext_master()) { mxb_assert(!m_master->m_slave_status.empty() && !m_master->m_node.external_masters.empty()); // TODO: Add support for multiple external masters. auto& master_sstatus = m_master->m_slave_status[0]; if (master_sstatus.master_host != m_external_master_host || master_sstatus.master_port != m_external_master_port) { const string new_ext_host = master_sstatus.master_host; const int new_ext_port = master_sstatus.master_port; if (m_external_master_port == PORT_UNKNOWN) { MXS_NOTICE("Cluster master server is replicating from an external master: %s:%d", new_ext_host.c_str(), new_ext_port); } else { MXS_NOTICE("The external master of the cluster has changed: %s:%d -> %s:%d.", m_external_master_host.c_str(), m_external_master_port, new_ext_host.c_str(), new_ext_port); } m_external_master_host = new_ext_host; m_external_master_port = new_ext_port; } } else { if (m_external_master_port != PORT_UNKNOWN) { MXS_NOTICE("Cluster lost the external master. Previous one was at: [%s]:%d", m_external_master_host.c_str(), m_external_master_port); } m_external_master_host.clear(); m_external_master_port = PORT_UNKNOWN; } } void MariaDBMonitor::log_master_changes() { MXS_MONITORED_SERVER* root_master = m_master ? m_master->m_server_base : NULL; if (root_master && mon_status_changed(root_master) && !(root_master->pending_status & SERVER_WAS_MASTER)) { if ((root_master->pending_status & SERVER_MASTER) && m_master->is_running()) { if (!(root_master->mon_prev_status & SERVER_WAS_MASTER) && !(root_master->pending_status & SERVER_MAINT)) { MXS_NOTICE("A Master Server is now available: %s:%i", root_master->server->address, root_master->server->port); } } else { MXS_ERROR("No Master can be determined. Last known was %s:%i", root_master->server->address, root_master->server->port); } m_log_no_master = true; } else { if (!root_master && m_log_no_master) { MXS_ERROR("No Master can be determined"); m_log_no_master = false; } } } void MariaDBMonitor::assign_new_master(MariaDBServer* new_master) { mxb::atomic::store(&m_master, new_master, mxb::atomic::RELAXED); update_master_cycle_info(); m_warn_current_master_invalid = true; m_warn_have_better_master = true; } /** * Check sql text file parameters. A parameter should either be empty or a valid file which can be opened. * * @return True if no errors occurred when opening the files */ bool MariaDBMonitor::check_sql_files() { const char ERRMSG[] = "%s ('%s') does not exist or cannot be accessed for reading: '%s'."; bool rval = true; if (!m_promote_sql_file.empty() && access(m_promote_sql_file.c_str(), R_OK) != 0) { rval = false; MXS_ERROR(ERRMSG, CN_PROMOTION_SQL_FILE, m_promote_sql_file.c_str(), mxs_strerror(errno)); } if (!m_demote_sql_file.empty() && access(m_demote_sql_file.c_str(), R_OK) != 0) { rval = false; MXS_ERROR(ERRMSG, CN_DEMOTION_SQL_FILE, m_demote_sql_file.c_str(), mxs_strerror(errno)); } return rval; } /** * Schedule a manual command for execution. It will be ran during the next monitor loop. This method waits * for the command to have finished running. * * @param command Function object containing the method the monitor should execute: switchover, failover or * rejoin. * @param error_out Json error output * @return True if command execution was attempted. False if monitor was in an invalid state * to run the command. */ bool MariaDBMonitor::execute_manual_command(std::function command, json_t** error_out) { bool rval = false; if (monitor_state() != MONITOR_STATE_RUNNING) { PRINT_MXS_JSON_ERROR(error_out, "The monitor is not running, cannot execute manual command."); } else if (m_manual_cmd.command_waiting_exec) { PRINT_MXS_JSON_ERROR(error_out, "Previous command has not been executed, cannot send another command."); mxb_assert(!true); } else { rval = true; // Write the command. std::unique_lock lock(m_manual_cmd.mutex); m_manual_cmd.method = command; m_manual_cmd.command_waiting_exec = true; // Signal the monitor thread to start running the command. lock.unlock(); m_manual_cmd.has_command.notify_one(); // Wait for the result. lock.lock(); m_manual_cmd.has_result.wait(lock, [this] { return m_manual_cmd.result_waiting; }); m_manual_cmd.result_waiting = false; } return rval; } bool MariaDBMonitor::immediate_tick_required() const { return m_manual_cmd.command_waiting_exec; } bool MariaDBMonitor::run_manual_switchover(SERVER* promotion_server, SERVER* demotion_server, json_t** error_out) { bool rval = false; bool send_ok = execute_manual_command([this, &rval, promotion_server, demotion_server, error_out]() { rval = manual_switchover(promotion_server, demotion_server, error_out); }, error_out); return send_ok && rval; } bool MariaDBMonitor::run_manual_failover(json_t** error_out) { bool rval = false; bool send_ok = execute_manual_command([this, &rval, error_out]() { rval = manual_failover(error_out); }, error_out); return send_ok && rval; } bool MariaDBMonitor::run_manual_rejoin(SERVER* rejoin_server, json_t** error_out) { bool rval = false; bool send_ok = execute_manual_command([this, &rval, rejoin_server, error_out]() { rval = manual_rejoin(rejoin_server, error_out); }, error_out); return send_ok && rval; } bool MariaDBMonitor::run_manual_reset_replication(SERVER* master_server, json_t** error_out) { bool rval = false; bool send_ok = execute_manual_command([this, &rval, master_server, error_out]() { rval = manual_reset_replication(master_server, error_out); }, error_out); return send_ok && rval; } /** * Command handler for 'switchover' * * @param args The provided arguments. * @param output Pointer where to place output object. * * @return True, if the command was executed, false otherwise. */ bool handle_manual_switchover(const MODULECMD_ARG* args, json_t** error_out) { mxb_assert((args->argc >= 1) && (args->argc <= 3)); mxb_assert(MODULECMD_GET_TYPE(&args->argv[0].type) == MODULECMD_ARG_MONITOR); mxb_assert((args->argc < 2) || (MODULECMD_GET_TYPE(&args->argv[1].type) == MODULECMD_ARG_SERVER)); mxb_assert((args->argc < 3) || (MODULECMD_GET_TYPE(&args->argv[2].type) == MODULECMD_ARG_SERVER)); bool rval = false; if (config_get_global_options()->passive) { const char* const MSG = "Switchover requested but not performed, as MaxScale is in passive mode."; PRINT_MXS_JSON_ERROR(error_out, MSG); } else { MXS_MONITOR* mon = args->argv[0].value.monitor; auto handle = static_cast(mon->instance); SERVER* promotion_server = (args->argc >= 2) ? args->argv[1].value.server : NULL; SERVER* demotion_server = (args->argc == 3) ? args->argv[2].value.server : NULL; rval = handle->run_manual_switchover(promotion_server, demotion_server, error_out); } return rval; } /** * Command handler for 'failover' * * @param args Arguments given by user * @param output Json error output * @return True on success */ bool handle_manual_failover(const MODULECMD_ARG* args, json_t** output) { mxb_assert(args->argc == 1); mxb_assert(MODULECMD_GET_TYPE(&args->argv[0].type) == MODULECMD_ARG_MONITOR); bool rv = false; if (config_get_global_options()->passive) { PRINT_MXS_JSON_ERROR(output, "Failover requested but not performed, as MaxScale is in passive mode."); } else { MXS_MONITOR* mon = args->argv[0].value.monitor; auto handle = static_cast(mon->instance); rv = handle->run_manual_failover(output); } return rv; } /** * Command handler for 'rejoin' * * @param args Arguments given by user * @param output Json error output * @return True on success */ bool handle_manual_rejoin(const MODULECMD_ARG* args, json_t** output) { mxb_assert(args->argc == 2); mxb_assert(MODULECMD_GET_TYPE(&args->argv[0].type) == MODULECMD_ARG_MONITOR); mxb_assert(MODULECMD_GET_TYPE(&args->argv[1].type) == MODULECMD_ARG_SERVER); bool rv = false; if (config_get_global_options()->passive) { PRINT_MXS_JSON_ERROR(output, "Rejoin requested but not performed, as MaxScale is in passive mode."); } else { MXS_MONITOR* mon = args->argv[0].value.monitor; SERVER* server = args->argv[1].value.server; auto handle = static_cast(mon->instance); rv = handle->run_manual_rejoin(server, output); } return rv; } bool handle_manual_reset_replication(const MODULECMD_ARG* args, json_t** output) { mxb_assert(args->argc >= 1); mxb_assert(MODULECMD_GET_TYPE(&args->argv[0].type) == MODULECMD_ARG_MONITOR); mxb_assert(args->argc == 1 || MODULECMD_GET_TYPE(&args->argv[1].type) == MODULECMD_ARG_SERVER); bool rv = false; if (config_get_global_options()->passive) { PRINT_MXS_JSON_ERROR(output, "Replication reset requested but not performed, as MaxScale is in " "passive mode."); } else { MXS_MONITOR* mon = args->argv[0].value.monitor; SERVER* server = args->argv[1].value.server; auto handle = static_cast(mon->instance); rv = handle->run_manual_reset_replication(server, output); } return rv; } string monitored_servers_to_string(const ServerArray& servers) { string rval; size_t array_size = servers.size(); if (array_size > 0) { const char* separator = ""; for (size_t i = 0; i < array_size; i++) { rval += separator; rval += string("'") + servers[i]->name() + "'"; separator = ", "; } } return rval; } /** * The module entry point routine. This routine populates the module object structure. * * @return The module object */ extern "C" MXS_MODULE* MXS_CREATE_MODULE() { MXS_NOTICE("Initialise the MariaDB Monitor module."); static const char ARG_MONITOR_DESC[] = "Monitor name (from configuration file)"; static modulecmd_arg_type_t switchover_argv[] = { { MODULECMD_ARG_MONITOR | MODULECMD_ARG_NAME_MATCHES_DOMAIN, ARG_MONITOR_DESC }, {MODULECMD_ARG_SERVER | MODULECMD_ARG_OPTIONAL, "New master (optional)" }, {MODULECMD_ARG_SERVER | MODULECMD_ARG_OPTIONAL, "Current master (optional)"} }; modulecmd_register_command(MXS_MODULE_NAME, "switchover", MODULECMD_TYPE_ACTIVE, handle_manual_switchover, MXS_ARRAY_NELEMS(switchover_argv), switchover_argv, "Perform master switchover"); static modulecmd_arg_type_t failover_argv[] = { { MODULECMD_ARG_MONITOR | MODULECMD_ARG_NAME_MATCHES_DOMAIN, ARG_MONITOR_DESC }, }; modulecmd_register_command(MXS_MODULE_NAME, "failover", MODULECMD_TYPE_ACTIVE, handle_manual_failover, MXS_ARRAY_NELEMS(failover_argv), failover_argv, "Perform master failover"); static modulecmd_arg_type_t rejoin_argv[] = { { MODULECMD_ARG_MONITOR | MODULECMD_ARG_NAME_MATCHES_DOMAIN, ARG_MONITOR_DESC }, {MODULECMD_ARG_SERVER, "Joining server"} }; modulecmd_register_command(MXS_MODULE_NAME, "rejoin", MODULECMD_TYPE_ACTIVE, handle_manual_rejoin, MXS_ARRAY_NELEMS(rejoin_argv), rejoin_argv, "Rejoin server to a cluster"); static modulecmd_arg_type_t reset_gtid_argv[] = { { MODULECMD_ARG_MONITOR | MODULECMD_ARG_NAME_MATCHES_DOMAIN, ARG_MONITOR_DESC }, {MODULECMD_ARG_SERVER | MODULECMD_ARG_OPTIONAL, "Master server (optional)"} }; modulecmd_register_command(MXS_MODULE_NAME, "reset-replication", MODULECMD_TYPE_ACTIVE, handle_manual_reset_replication, MXS_ARRAY_NELEMS(reset_gtid_argv), reset_gtid_argv, "Delete slave connections, delete binary logs and " "set up replication (dangerous)"); static MXS_MODULE info = { MXS_MODULE_API_MONITOR, MXS_MODULE_GA, MXS_MONITOR_VERSION, "A MariaDB Master/Slave replication monitor", "V1.5.0", MXS_NO_MODULE_CAPABILITIES, &maxscale::MonitorApi::s_api, NULL, /* Process init. */ NULL, /* Process finish. */ NULL, /* Thread init. */ NULL, /* Thread finish. */ { { "detect_replication_lag", MXS_MODULE_PARAM_BOOL, "false" }, { "detect_stale_master", MXS_MODULE_PARAM_BOOL, "true" }, { "detect_stale_slave", MXS_MODULE_PARAM_BOOL, "true" }, { "mysql51_replication", MXS_MODULE_PARAM_BOOL, "false", MXS_MODULE_OPT_DEPRECATED }, { "multimaster", MXS_MODULE_PARAM_BOOL, "false", MXS_MODULE_OPT_DEPRECATED }, { CN_DETECT_STANDALONE_MASTER, MXS_MODULE_PARAM_BOOL, "true" }, { CN_FAILCOUNT, MXS_MODULE_PARAM_COUNT, "5" }, { "allow_cluster_recovery", MXS_MODULE_PARAM_BOOL, "true", MXS_MODULE_OPT_DEPRECATED }, { "ignore_external_masters", MXS_MODULE_PARAM_BOOL, "false" }, { CN_AUTO_FAILOVER, MXS_MODULE_PARAM_BOOL, "false" }, { CN_FAILOVER_TIMEOUT, MXS_MODULE_PARAM_COUNT, "90" }, { CN_SWITCHOVER_TIMEOUT, MXS_MODULE_PARAM_COUNT, "90" }, { CN_REPLICATION_USER, MXS_MODULE_PARAM_STRING }, { CN_REPLICATION_PASSWORD, MXS_MODULE_PARAM_PASSWORD }, { CN_REPLICATION_MASTER_SSL, MXS_MODULE_PARAM_BOOL, "false" }, { CN_VERIFY_MASTER_FAILURE, MXS_MODULE_PARAM_BOOL, "true" }, { CN_MASTER_FAILURE_TIMEOUT, MXS_MODULE_PARAM_COUNT, "10" }, { CN_AUTO_REJOIN, MXS_MODULE_PARAM_BOOL, "false" }, { CN_ENFORCE_READONLY, MXS_MODULE_PARAM_BOOL, "false" }, { CN_NO_PROMOTE_SERVERS, MXS_MODULE_PARAM_SERVERLIST }, { CN_PROMOTION_SQL_FILE, MXS_MODULE_PARAM_PATH }, { CN_DEMOTION_SQL_FILE, MXS_MODULE_PARAM_PATH }, { CN_SWITCHOVER_ON_LOW_DISK_SPACE, MXS_MODULE_PARAM_BOOL, "false" }, { CN_MAINTENANCE_ON_LOW_DISK_SPACE, MXS_MODULE_PARAM_BOOL, "true" }, { CN_HANDLE_EVENTS, MXS_MODULE_PARAM_BOOL, "true" }, { CN_ASSUME_UNIQUE_HOSTNAMES, MXS_MODULE_PARAM_BOOL, "true" }, {MXS_END_MODULE_PARAMS} } }; return &info; }