
When LEAST_BEHIND_MASTER routing criteria was used, the info level logging function would fall through to the default case. In debug builds, this would trigger a debug assertion.
433 lines
13 KiB
C++
433 lines
13 KiB
C++
/*
|
|
* 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: 2020-01-01
|
|
*
|
|
* 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.
|
|
*/
|
|
|
|
#include "readwritesplit.hh"
|
|
#include "rwsplit_internal.hh"
|
|
|
|
#include <stdio.h>
|
|
#include <strings.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
#include <stdint.h>
|
|
|
|
#include <maxscale/router.h>
|
|
|
|
/**
|
|
* The functions that implement back end selection for the read write
|
|
* split router. All of these functions are internal to that router and
|
|
* not intended to be called from elsewhere.
|
|
*/
|
|
|
|
/**
|
|
* Check whether it's possible to use this server as a slave
|
|
*
|
|
* @param server The slave candidate
|
|
* @param master The master server or NULL if no master is available
|
|
*
|
|
* @return True if this server is a valid slave candidate
|
|
*/
|
|
static bool valid_for_slave(const SERVER *server, const SERVER *master)
|
|
{
|
|
return (SERVER_IS_SLAVE(server) || SERVER_IS_RELAY_SERVER(server)) &&
|
|
(master == NULL || (server != master));
|
|
}
|
|
|
|
/**
|
|
* @brief Find the best slave candidate
|
|
*
|
|
* This function iterates through @c backend and tries to find the best backend
|
|
* reference that is not in use. @c cmpfun will be called to compare the backends.
|
|
*
|
|
* @param rses Router client session
|
|
* @param master The master server
|
|
* @param cmpfun qsort() compatible comparison function
|
|
*
|
|
* @return The best slave backend reference or NULL if no candidates could be found
|
|
*/
|
|
SRWBackend get_slave_candidate(const SRWBackendList& backends, const SERVER *master,
|
|
int (*cmpfun)(const SRWBackend&, const SRWBackend&))
|
|
{
|
|
SRWBackend candidate;
|
|
|
|
for (SRWBackendList::const_iterator it = backends.begin();
|
|
it != backends.end(); it++)
|
|
{
|
|
const SRWBackend& backend = *it;
|
|
|
|
if (!backend->in_use() && backend->can_connect() &&
|
|
valid_for_slave(backend->server(), master))
|
|
{
|
|
if (candidate)
|
|
{
|
|
if (cmpfun(candidate, backend) > 0)
|
|
{
|
|
candidate = backend;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
candidate = backend;
|
|
}
|
|
}
|
|
}
|
|
|
|
return candidate;
|
|
}
|
|
|
|
/** Compare number of connections from this router in backend servers */
|
|
static int backend_cmp_router_conn(const SRWBackend& a, const SRWBackend& b)
|
|
{
|
|
SERVER_REF *first = a->backend();
|
|
SERVER_REF *second = b->backend();
|
|
|
|
if (first->weight == 0 && second->weight == 0)
|
|
{
|
|
return first->connections - second->connections;
|
|
}
|
|
else if (first->weight == 0)
|
|
{
|
|
return 1;
|
|
}
|
|
else if (second->weight == 0)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
return ((1000 + 1000 * first->connections) / first->weight) -
|
|
((1000 + 1000 * second->connections) / second->weight);
|
|
}
|
|
|
|
/** Compare number of global connections in backend servers */
|
|
static int backend_cmp_global_conn(const SRWBackend& a, const SRWBackend& b)
|
|
{
|
|
SERVER_REF *first = a->backend();
|
|
SERVER_REF *second = b->backend();
|
|
|
|
if (first->weight == 0 && second->weight == 0)
|
|
{
|
|
return first->server->stats.n_current -
|
|
second->server->stats.n_current;
|
|
}
|
|
else if (first->weight == 0)
|
|
{
|
|
return 1;
|
|
}
|
|
else if (second->weight == 0)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
return ((1000 + 1000 * first->server->stats.n_current) / first->weight) -
|
|
((1000 + 1000 * second->server->stats.n_current) / second->weight);
|
|
}
|
|
|
|
/** Compare replication lag between backend servers */
|
|
static int backend_cmp_behind_master(const SRWBackend& a, const SRWBackend& b)
|
|
{
|
|
SERVER_REF *first = a->backend();
|
|
SERVER_REF *second = b->backend();
|
|
|
|
if (first->weight == 0 && second->weight == 0)
|
|
{
|
|
return first->server->rlag -
|
|
second->server->rlag;
|
|
}
|
|
else if (first->weight == 0)
|
|
{
|
|
return 1;
|
|
}
|
|
else if (second->weight == 0)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
return ((1000 + 1000 * first->server->rlag) / first->weight) -
|
|
((1000 + 1000 * second->server->rlag) / second->weight);
|
|
}
|
|
|
|
/** Compare number of current operations in backend servers */
|
|
static int backend_cmp_current_load(const SRWBackend& a, const SRWBackend& b)
|
|
{
|
|
SERVER_REF *first = a->backend();
|
|
SERVER_REF *second = b->backend();
|
|
|
|
if (first->weight == 0 && second->weight == 0)
|
|
{
|
|
return first->server->stats.n_current_ops - second->server->stats.n_current_ops;
|
|
}
|
|
else if (first->weight == 0)
|
|
{
|
|
return 1;
|
|
}
|
|
else if (second->weight == 0)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
return ((1000 + 1000 * first->server->stats.n_current_ops) / first->weight) -
|
|
((1000 + 1000 * second->server->stats.n_current_ops) / second->weight);
|
|
}
|
|
|
|
/**
|
|
* The order of functions _must_ match with the order the select criteria are
|
|
* listed in select_criteria_t definition in readwritesplit.h
|
|
*/
|
|
int (*criteria_cmpfun[LAST_CRITERIA])(const SRWBackend&, const SRWBackend&) =
|
|
{
|
|
NULL,
|
|
backend_cmp_global_conn,
|
|
backend_cmp_router_conn,
|
|
backend_cmp_behind_master,
|
|
backend_cmp_current_load
|
|
};
|
|
|
|
/**
|
|
* @brief Log server connections
|
|
*
|
|
* @param criteria Slave selection criteria
|
|
* @param rses Router client session
|
|
*/
|
|
static void log_server_connections(select_criteria_t criteria, const SRWBackendList& backends)
|
|
{
|
|
MXS_INFO("Servers and %s connection counts:",
|
|
criteria == LEAST_GLOBAL_CONNECTIONS ? "all MaxScale" : "router");
|
|
|
|
for (SRWBackendList::const_iterator it = backends.begin(); it != backends.end(); it++)
|
|
{
|
|
SERVER_REF* b = (*it)->backend();
|
|
|
|
switch (criteria)
|
|
{
|
|
case LEAST_GLOBAL_CONNECTIONS:
|
|
MXS_INFO("MaxScale connections : %d in \t[%s]:%d %s",
|
|
b->server->stats.n_current, b->server->name,
|
|
b->server->port, STRSRVSTATUS(b->server));
|
|
break;
|
|
|
|
case LEAST_ROUTER_CONNECTIONS:
|
|
MXS_INFO("RWSplit connections : %d in \t[%s]:%d %s",
|
|
b->connections, b->server->name,
|
|
b->server->port, STRSRVSTATUS(b->server));
|
|
break;
|
|
|
|
case LEAST_CURRENT_OPERATIONS:
|
|
MXS_INFO("current operations : %d in \t[%s]:%d %s",
|
|
b->server->stats.n_current_ops,
|
|
b->server->name, b->server->port,
|
|
STRSRVSTATUS(b->server));
|
|
break;
|
|
|
|
case LEAST_BEHIND_MASTER:
|
|
MXS_INFO("replication lag : %d in \t[%s]:%d %s",
|
|
b->server->rlag, b->server->name,
|
|
b->server->port, STRSRVSTATUS(b->server));
|
|
break;
|
|
|
|
default:
|
|
ss_dassert(!true);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
/**
|
|
* @brief Find the master server that is at the root of the replication tree
|
|
*
|
|
* @param rses Router client session
|
|
*
|
|
* @return The root master reference or NULL if no master is found
|
|
*/
|
|
static SERVER_REF* get_root_master(const SRWBackendList& backends)
|
|
{
|
|
SERVER_REF *master_host = NULL;
|
|
|
|
for (SRWBackendList::const_iterator it = backends.begin();
|
|
it != backends.end(); it++)
|
|
{
|
|
SERVER_REF* b = (*it)->backend();
|
|
|
|
if (SERVER_IS_MASTER(b->server))
|
|
{
|
|
if (master_host == NULL ||
|
|
(b->server->depth < master_host->server->depth))
|
|
{
|
|
master_host = b;
|
|
}
|
|
}
|
|
}
|
|
|
|
return master_host;
|
|
}
|
|
|
|
/**
|
|
* @brief Search suitable backend servers from those of router instance
|
|
*
|
|
* It is assumed that there is only one master among servers of a router instance.
|
|
* As a result, the first master found is chosen. There will possibly be more
|
|
* backend references than connected backends because only those in correct state
|
|
* are connected to.
|
|
*
|
|
* @param router_nservers Number of backend servers
|
|
* @param max_nslaves Upper limit for the number of slaves
|
|
* @param select_criteria Slave selection criteria
|
|
* @param session Client session
|
|
* @param router Router instance
|
|
* @param rses Router client session
|
|
* @param type Connection type, ALL for all types, SLAVE for slaves only
|
|
*
|
|
* @return True if at least one master and one slave was found
|
|
*/
|
|
bool select_connect_backend_servers(int router_nservers,
|
|
int max_nslaves,
|
|
MXS_SESSION *session,
|
|
const Config& config,
|
|
SRWBackendList& backends,
|
|
SRWBackend& current_master,
|
|
mxs::SessionCommandList* sescmd_list,
|
|
int* expected_responses,
|
|
connection_type type)
|
|
{
|
|
/* get the root Master */
|
|
SERVER_REF *master_backend = get_root_master(backends);
|
|
SERVER *master_host = master_backend ? master_backend->server : NULL;
|
|
|
|
if (config.master_failure_mode == RW_FAIL_INSTANTLY &&
|
|
(master_host == NULL || SERVER_IS_DOWN(master_host)))
|
|
{
|
|
MXS_ERROR("Couldn't find suitable Master from %d candidates.", router_nservers);
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* New session:
|
|
*
|
|
* Connect to both master and slaves
|
|
*
|
|
* Existing session:
|
|
*
|
|
* Master is already connected or we don't have a master. The function was
|
|
* called because new slaves must be selected to replace failed ones.
|
|
*/
|
|
bool master_connected = type == SLAVE || current_master;
|
|
|
|
/** Check slave selection criteria and set compare function */
|
|
select_criteria_t select_criteria = config.slave_selection_criteria;
|
|
int (*cmpfun)(const SRWBackend&, const SRWBackend&) = criteria_cmpfun[select_criteria];
|
|
ss_dassert(cmpfun);
|
|
|
|
if (MXS_LOG_PRIORITY_IS_ENABLED(LOG_INFO))
|
|
{
|
|
log_server_connections(select_criteria, backends);
|
|
}
|
|
|
|
int slaves_found = 0;
|
|
int slaves_connected = 0;
|
|
const int min_nslaves = 0; /*< not configurable at the time */
|
|
bool succp = false;
|
|
|
|
if (!master_connected)
|
|
{
|
|
/** Find a master server */
|
|
for (SRWBackendList::const_iterator it = backends.begin(); it != backends.end(); it++)
|
|
{
|
|
const SRWBackend& backend = *it;
|
|
|
|
if (backend->can_connect() && master_host && backend->server() == master_host)
|
|
{
|
|
if (backend->connect(session))
|
|
{
|
|
current_master = backend;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/** Calculate how many connections we already have */
|
|
for (SRWBackendList::const_iterator it = backends.begin(); it != backends.end(); it++)
|
|
{
|
|
const SRWBackend& backend = *it;
|
|
|
|
if (backend->can_connect() && valid_for_slave(backend->server(), master_host))
|
|
{
|
|
slaves_found += 1;
|
|
|
|
if (backend->in_use())
|
|
{
|
|
slaves_connected += 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
ss_dassert(slaves_connected < max_nslaves || max_nslaves == 0);
|
|
|
|
/** Connect to all possible slaves */
|
|
for (SRWBackend backend(get_slave_candidate(backends, master_host, cmpfun));
|
|
backend && slaves_connected < max_nslaves;
|
|
backend = get_slave_candidate(backends, master_host, cmpfun))
|
|
{
|
|
if (backend->can_connect() && backend->connect(session))
|
|
{
|
|
if (sescmd_list && sescmd_list->size())
|
|
{
|
|
backend->append_session_command(*sescmd_list);
|
|
|
|
if (backend->execute_session_command())
|
|
{
|
|
if (expected_responses)
|
|
{
|
|
(*expected_responses)++;
|
|
}
|
|
slaves_connected++;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
slaves_connected++;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (slaves_connected >= min_nslaves && slaves_connected <= max_nslaves)
|
|
{
|
|
succp = true;
|
|
|
|
if (MXS_LOG_PRIORITY_IS_ENABLED(LOG_INFO))
|
|
{
|
|
if (slaves_connected < max_nslaves)
|
|
{
|
|
MXS_INFO("Couldn't connect to maximum number of "
|
|
"slaves. Connected successfully to %d slaves "
|
|
"of %d of them.", slaves_connected, slaves_found);
|
|
}
|
|
|
|
for (SRWBackendList::const_iterator it = backends.begin(); it != backends.end(); it++)
|
|
{
|
|
const SRWBackend& backend = *it;
|
|
if (backend->in_use())
|
|
{
|
|
MXS_INFO("Selected %s in \t%s", STRSRVSTATUS(backend->server()),
|
|
backend->uri());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
MXS_ERROR("Couldn't establish required amount of slave connections for "
|
|
"router session. Would need between %d and %d slaves but only have %d.",
|
|
min_nslaves, max_nslaves, slaves_connected);
|
|
close_all_connections(backends);
|
|
}
|
|
|
|
return succp;
|
|
}
|