159 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			159 lines
		
	
	
		
			3.9 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: 2022-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.
 | 
						|
 */
 | 
						|
 | 
						|
/**
 | 
						|
 * @file A MySQL Group Replication cluster monitor
 | 
						|
 */
 | 
						|
 | 
						|
#define MXS_MODULE_NAME "grmon"
 | 
						|
 | 
						|
#include "grmon.hh"
 | 
						|
 | 
						|
#include <new>
 | 
						|
#include <string>
 | 
						|
 | 
						|
#include <maxscale/protocol/mysql.h>
 | 
						|
#include <mysqld_error.h>
 | 
						|
 | 
						|
 | 
						|
GRMon::GRMon(MXS_MONITOR* monitor)
 | 
						|
    : MonitorInstanceSimple(monitor)
 | 
						|
{
 | 
						|
}
 | 
						|
 | 
						|
GRMon::~GRMon()
 | 
						|
{
 | 
						|
}
 | 
						|
 | 
						|
GRMon* GRMon::create(MXS_MONITOR* monitor)
 | 
						|
{
 | 
						|
    return new GRMon(monitor);
 | 
						|
}
 | 
						|
 | 
						|
bool GRMon::has_sufficient_permissions() const
 | 
						|
{
 | 
						|
    return true;
 | 
						|
}
 | 
						|
 | 
						|
static inline bool is_false(const char* value)
 | 
						|
{
 | 
						|
    return strcasecmp(value, "0") == 0
 | 
						|
           || strcasecmp(value, "no") == 0
 | 
						|
           || strcasecmp(value, "off") == 0
 | 
						|
           || strcasecmp(value, "false") == 0;
 | 
						|
}
 | 
						|
 | 
						|
static bool is_master(MXS_MONITORED_SERVER* server)
 | 
						|
{
 | 
						|
    bool rval = false;
 | 
						|
    MYSQL_RES* result;
 | 
						|
    const char* master_query =
 | 
						|
        "SELECT VARIABLE_VALUE, @@server_uuid, @@read_only FROM performance_schema.global_status "
 | 
						|
        "WHERE VARIABLE_NAME= 'group_replication_primary_member'";
 | 
						|
 | 
						|
    if (mysql_query(server->con, master_query) == 0 && (result = mysql_store_result(server->con)))
 | 
						|
    {
 | 
						|
        for (MYSQL_ROW row = mysql_fetch_row(result); row; row = mysql_fetch_row(result))
 | 
						|
        {
 | 
						|
            if (strcasecmp(row[0], row[1]) == 0 && is_false(row[2]))
 | 
						|
            {
 | 
						|
                rval = true;
 | 
						|
            }
 | 
						|
        }
 | 
						|
        mysql_free_result(result);
 | 
						|
    }
 | 
						|
    else
 | 
						|
    {
 | 
						|
        mon_report_query_error(server);
 | 
						|
    }
 | 
						|
 | 
						|
    return rval;
 | 
						|
}
 | 
						|
 | 
						|
static bool is_slave(MXS_MONITORED_SERVER* server)
 | 
						|
{
 | 
						|
    bool rval = false;
 | 
						|
    MYSQL_RES* result;
 | 
						|
    const char slave_query[] = "SELECT MEMBER_STATE FROM "
 | 
						|
                               "performance_schema.replication_group_members "
 | 
						|
                               "WHERE MEMBER_ID = @@server_uuid";
 | 
						|
 | 
						|
    if (mysql_query(server->con, slave_query) == 0 && (result = mysql_store_result(server->con)))
 | 
						|
    {
 | 
						|
        for (MYSQL_ROW row = mysql_fetch_row(result); row; row = mysql_fetch_row(result))
 | 
						|
        {
 | 
						|
            if (strcasecmp(row[0], "ONLINE") == 0)
 | 
						|
            {
 | 
						|
                rval = true;
 | 
						|
            }
 | 
						|
        }
 | 
						|
        mysql_free_result(result);
 | 
						|
    }
 | 
						|
    else
 | 
						|
    {
 | 
						|
        mon_report_query_error(server);
 | 
						|
    }
 | 
						|
 | 
						|
    return rval;
 | 
						|
}
 | 
						|
 | 
						|
void GRMon::update_server_status(MXS_MONITORED_SERVER* monitored_server)
 | 
						|
{
 | 
						|
    if (is_master(monitored_server))
 | 
						|
    {
 | 
						|
        monitor_set_pending_status(monitored_server, SERVER_MASTER);
 | 
						|
        monitor_clear_pending_status(monitored_server, SERVER_SLAVE);
 | 
						|
    }
 | 
						|
    else if (is_slave(monitored_server))
 | 
						|
    {
 | 
						|
        monitor_set_pending_status(monitored_server, SERVER_SLAVE);
 | 
						|
        monitor_clear_pending_status(monitored_server, SERVER_MASTER);
 | 
						|
    }
 | 
						|
    else
 | 
						|
    {
 | 
						|
        monitor_clear_pending_status(monitored_server, SERVER_SLAVE);
 | 
						|
        monitor_clear_pending_status(monitored_server, SERVER_MASTER);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * The module entry point routine. It is this routine that
 | 
						|
 * must populate the structure that is referred to as the
 | 
						|
 * "module object", this is a structure with the set of
 | 
						|
 * external entry points for this module.
 | 
						|
 *
 | 
						|
 * @return The module object
 | 
						|
 */
 | 
						|
extern "C" MXS_MODULE* MXS_CREATE_MODULE()
 | 
						|
{
 | 
						|
    static MXS_MODULE info =
 | 
						|
    {
 | 
						|
        MXS_MODULE_API_MONITOR,
 | 
						|
        MXS_MODULE_GA,
 | 
						|
        MXS_MONITOR_VERSION,
 | 
						|
        "A Group Replication cluster monitor",
 | 
						|
        "V1.0.0",
 | 
						|
        MXS_NO_MODULE_CAPABILITIES,
 | 
						|
        &maxscale::MonitorApi<GRMon>::s_api,
 | 
						|
        NULL,       /* Process init. */
 | 
						|
        NULL,       /* Process finish. */
 | 
						|
        NULL,       /* Thread init. */
 | 
						|
        NULL,       /* Thread finish. */
 | 
						|
        {
 | 
						|
            {MXS_END_MODULE_PARAMS}
 | 
						|
        }
 | 
						|
    };
 | 
						|
 | 
						|
    return &info;
 | 
						|
}
 |