mirror of
https://github.com/MariaDB/server.git
synced 2025-01-25 00:04:33 +01:00
1500 lines
49 KiB
C++
1500 lines
49 KiB
C++
/* Copyright (C) 2009-2014 Kentoku Shiba
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; version 2 of the License.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software
|
|
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */
|
|
|
|
#define MYSQL_SERVER 1
|
|
#include "mysql_version.h"
|
|
#if MYSQL_VERSION_ID < 50500
|
|
#include "mysql_priv.h"
|
|
#include <mysql/plugin.h>
|
|
#else
|
|
#include "sql_priv.h"
|
|
#include "probes_mysql.h"
|
|
#include "sql_class.h"
|
|
#include "sql_partition.h"
|
|
#include "sql_acl.h"
|
|
#endif
|
|
#include "spd_err.h"
|
|
#include "spd_param.h"
|
|
#include "spd_db_include.h"
|
|
#include "spd_include.h"
|
|
#include "ha_spider.h"
|
|
#include "spd_db_conn.h"
|
|
#include "spd_trx.h"
|
|
#include "spd_conn.h"
|
|
#include "spd_sys_table.h"
|
|
#include "spd_table.h"
|
|
#include "spd_ping_table.h"
|
|
#include "spd_direct_sql.h"
|
|
#include "spd_udf.h"
|
|
#include "spd_malloc.h"
|
|
|
|
extern bool volatile *spd_abort_loop;
|
|
|
|
extern handlerton *spider_hton_ptr;
|
|
|
|
#ifdef HAVE_PSI_INTERFACE
|
|
extern PSI_mutex_key spd_key_mutex_mon_list_caller;
|
|
extern PSI_mutex_key spd_key_mutex_mon_list_receptor;
|
|
extern PSI_mutex_key spd_key_mutex_mon_list_monitor;
|
|
extern PSI_mutex_key spd_key_mutex_mon_list_update_status;
|
|
extern PSI_mutex_key spd_key_mutex_mon_table_cache;
|
|
#endif
|
|
|
|
#ifndef WITHOUT_SPIDER_BG_SEARCH
|
|
extern pthread_mutex_t spider_global_trx_mutex;
|
|
extern SPIDER_TRX *spider_global_trx;
|
|
#endif
|
|
|
|
HASH *spider_udf_table_mon_list_hash;
|
|
uint spider_udf_table_mon_list_hash_id;
|
|
const char *spider_udf_table_mon_list_hash_func_name;
|
|
const char *spider_udf_table_mon_list_hash_file_name;
|
|
ulong spider_udf_table_mon_list_hash_line_no;
|
|
pthread_mutex_t *spider_udf_table_mon_mutexes;
|
|
pthread_cond_t *spider_udf_table_mon_conds;
|
|
|
|
pthread_mutex_t spider_mon_table_cache_mutex;
|
|
DYNAMIC_ARRAY spider_mon_table_cache;
|
|
uint spider_mon_table_cache_id;
|
|
const char *spider_mon_table_cache_func_name;
|
|
const char *spider_mon_table_cache_file_name;
|
|
ulong spider_mon_table_cache_line_no;
|
|
volatile ulonglong spider_mon_table_cache_version = 0;
|
|
volatile ulonglong spider_mon_table_cache_version_req = 1;
|
|
|
|
SPIDER_TABLE_MON_LIST *spider_get_ping_table_mon_list(
|
|
SPIDER_TRX *trx,
|
|
THD *thd,
|
|
spider_string *str,
|
|
uint conv_name_length,
|
|
int link_idx,
|
|
uint32 server_id,
|
|
bool need_lock,
|
|
int *error_num
|
|
) {
|
|
uint mutex_hash;
|
|
SPIDER_TABLE_MON_LIST *table_mon_list;
|
|
MEM_ROOT mem_root;
|
|
ulonglong mon_table_cache_version;
|
|
#ifdef SPIDER_HAS_HASH_VALUE_TYPE
|
|
my_hash_value_type hash_value;
|
|
#endif
|
|
DBUG_ENTER("spider_get_ping_table_mon_list");
|
|
if (spider_mon_table_cache_version != spider_mon_table_cache_version_req)
|
|
{
|
|
SPD_INIT_ALLOC_ROOT(&mem_root, 4096, 0, MYF(MY_WME));
|
|
if ((*error_num = spider_init_ping_table_mon_cache(thd, &mem_root,
|
|
need_lock)))
|
|
{
|
|
free_root(&mem_root, MYF(0));
|
|
goto error;
|
|
}
|
|
free_root(&mem_root, MYF(0));
|
|
}
|
|
|
|
mutex_hash = spider_udf_calc_hash(str->c_ptr(),
|
|
spider_param_udf_table_mon_mutex_count());
|
|
DBUG_PRINT("info",("spider hash key=%s", str->c_ptr()));
|
|
DBUG_PRINT("info",("spider hash key length=%u", str->length()));
|
|
#ifdef SPIDER_HAS_HASH_VALUE_TYPE
|
|
hash_value = my_calc_hash(
|
|
&spider_udf_table_mon_list_hash[mutex_hash],
|
|
(uchar*) str->c_ptr(), str->length());
|
|
#endif
|
|
pthread_mutex_lock(&spider_udf_table_mon_mutexes[mutex_hash]);
|
|
mon_table_cache_version = (ulonglong) spider_mon_table_cache_version;
|
|
#ifdef SPIDER_HAS_HASH_VALUE_TYPE
|
|
if (!(table_mon_list = (SPIDER_TABLE_MON_LIST *)
|
|
my_hash_search_using_hash_value(
|
|
&spider_udf_table_mon_list_hash[mutex_hash], hash_value,
|
|
(uchar*) str->c_ptr(), str->length())) ||
|
|
table_mon_list->mon_table_cache_version != mon_table_cache_version
|
|
)
|
|
#else
|
|
if (!(table_mon_list = (SPIDER_TABLE_MON_LIST *) my_hash_search(
|
|
&spider_udf_table_mon_list_hash[mutex_hash],
|
|
(uchar*) str->c_ptr(), str->length())) ||
|
|
table_mon_list->mon_table_cache_version != mon_table_cache_version
|
|
)
|
|
#endif
|
|
{
|
|
DBUG_ASSERT(trx != spider_global_trx);
|
|
if (
|
|
table_mon_list &&
|
|
table_mon_list->mon_table_cache_version != mon_table_cache_version
|
|
)
|
|
spider_release_ping_table_mon_list_loop(mutex_hash, table_mon_list);
|
|
|
|
if (!(table_mon_list = spider_get_ping_table_tgt(thd, str->c_ptr(),
|
|
conv_name_length, link_idx, server_id, str, need_lock, error_num)))
|
|
{
|
|
pthread_mutex_unlock(&spider_udf_table_mon_mutexes[mutex_hash]);
|
|
goto error;
|
|
}
|
|
table_mon_list->mutex_hash = mutex_hash;
|
|
table_mon_list->mon_table_cache_version = mon_table_cache_version;
|
|
uint old_elements =
|
|
spider_udf_table_mon_list_hash[mutex_hash].array.max_element;
|
|
#ifdef SPIDER_HAS_HASH_VALUE_TYPE
|
|
table_mon_list->key_hash_value = hash_value;
|
|
#endif
|
|
#ifdef HASH_UPDATE_WITH_HASH_VALUE
|
|
if (my_hash_insert_with_hash_value(
|
|
&spider_udf_table_mon_list_hash[mutex_hash],
|
|
hash_value, (uchar*) table_mon_list))
|
|
#else
|
|
if (my_hash_insert(&spider_udf_table_mon_list_hash[mutex_hash],
|
|
(uchar*) table_mon_list))
|
|
#endif
|
|
{
|
|
spider_ping_table_free_mon_list(table_mon_list);
|
|
*error_num = HA_ERR_OUT_OF_MEM;
|
|
my_error(HA_ERR_OUT_OF_MEM, MYF(0));
|
|
pthread_mutex_unlock(&spider_udf_table_mon_mutexes[mutex_hash]);
|
|
goto error;
|
|
}
|
|
if (spider_udf_table_mon_list_hash[mutex_hash].array.max_element >
|
|
old_elements)
|
|
{
|
|
spider_alloc_calc_mem(spider_current_trx,
|
|
spider_udf_table_mon_list_hash,
|
|
(spider_udf_table_mon_list_hash[mutex_hash].array.max_element -
|
|
old_elements) *
|
|
spider_udf_table_mon_list_hash[mutex_hash].array.size_of_element);
|
|
}
|
|
}
|
|
table_mon_list->use_count++;
|
|
DBUG_PRINT("info",("spider table_mon_list->use_count=%d",
|
|
table_mon_list->use_count));
|
|
pthread_mutex_unlock(&spider_udf_table_mon_mutexes[mutex_hash]);
|
|
DBUG_RETURN(table_mon_list);
|
|
|
|
error:
|
|
DBUG_RETURN(NULL);
|
|
}
|
|
|
|
void spider_free_ping_table_mon_list(
|
|
SPIDER_TABLE_MON_LIST *table_mon_list
|
|
) {
|
|
DBUG_ENTER("spider_free_ping_table_mon_list");
|
|
pthread_mutex_lock(&spider_udf_table_mon_mutexes[
|
|
table_mon_list->mutex_hash]);
|
|
table_mon_list->use_count--;
|
|
DBUG_PRINT("info",("spider table_mon_list->use_count=%d", table_mon_list->use_count));
|
|
if (!table_mon_list->use_count)
|
|
pthread_cond_broadcast(&spider_udf_table_mon_conds[
|
|
table_mon_list->mutex_hash]);
|
|
pthread_mutex_unlock(&spider_udf_table_mon_mutexes[
|
|
table_mon_list->mutex_hash]);
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
void spider_release_ping_table_mon_list_loop(
|
|
uint mutex_hash,
|
|
SPIDER_TABLE_MON_LIST *table_mon_list
|
|
) {
|
|
DBUG_ENTER("spider_release_ping_table_mon_list_loop");
|
|
#ifdef HASH_UPDATE_WITH_HASH_VALUE
|
|
my_hash_delete_with_hash_value(&spider_udf_table_mon_list_hash[mutex_hash],
|
|
table_mon_list->key_hash_value, (uchar*) table_mon_list);
|
|
#else
|
|
my_hash_delete(&spider_udf_table_mon_list_hash[mutex_hash],
|
|
(uchar*) table_mon_list);
|
|
#endif
|
|
while (TRUE)
|
|
{
|
|
if (table_mon_list->use_count)
|
|
pthread_cond_wait(&spider_udf_table_mon_conds[mutex_hash],
|
|
&spider_udf_table_mon_mutexes[mutex_hash]);
|
|
else {
|
|
spider_ping_table_free_mon_list(table_mon_list);
|
|
break;
|
|
}
|
|
}
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
void spider_release_ping_table_mon_list(
|
|
const char *conv_name,
|
|
uint conv_name_length,
|
|
int link_idx
|
|
) {
|
|
uint mutex_hash;
|
|
SPIDER_TABLE_MON_LIST *table_mon_list;
|
|
char link_idx_str[SPIDER_SQL_INT_LEN];
|
|
int link_idx_str_length;
|
|
DBUG_ENTER("spider_release_ping_table_mon_list");
|
|
DBUG_PRINT("info", ("spider conv_name=%s", conv_name));
|
|
DBUG_PRINT("info", ("spider conv_name_length=%u", conv_name_length));
|
|
DBUG_PRINT("info", ("spider link_idx=%d", link_idx));
|
|
link_idx_str_length = my_sprintf(link_idx_str, (link_idx_str, "%010d",
|
|
link_idx));
|
|
#ifdef _MSC_VER
|
|
spider_string conv_name_str(conv_name_length + link_idx_str_length + 1);
|
|
conv_name_str.set_charset(system_charset_info);
|
|
#else
|
|
char buf[conv_name_length + link_idx_str_length + 1];
|
|
spider_string conv_name_str(buf, conv_name_length + link_idx_str_length + 1,
|
|
system_charset_info);
|
|
#endif
|
|
conv_name_str.init_calc_mem(134);
|
|
conv_name_str.length(0);
|
|
conv_name_str.q_append(conv_name, conv_name_length);
|
|
conv_name_str.q_append(link_idx_str, link_idx_str_length);
|
|
|
|
mutex_hash = spider_udf_calc_hash(conv_name_str.c_ptr_safe(),
|
|
spider_param_udf_table_mon_mutex_count());
|
|
#ifdef SPIDER_HAS_HASH_VALUE_TYPE
|
|
my_hash_value_type hash_value = my_calc_hash(
|
|
&spider_udf_table_mon_list_hash[mutex_hash],
|
|
(uchar*) conv_name_str.c_ptr(), conv_name_str.length());
|
|
#endif
|
|
pthread_mutex_lock(&spider_udf_table_mon_mutexes[mutex_hash]);
|
|
#ifdef SPIDER_HAS_HASH_VALUE_TYPE
|
|
if ((table_mon_list = (SPIDER_TABLE_MON_LIST *)
|
|
my_hash_search_using_hash_value(
|
|
&spider_udf_table_mon_list_hash[mutex_hash], hash_value,
|
|
(uchar*) conv_name_str.c_ptr(), conv_name_str.length())))
|
|
#else
|
|
if ((table_mon_list = (SPIDER_TABLE_MON_LIST *) my_hash_search(
|
|
&spider_udf_table_mon_list_hash[mutex_hash],
|
|
(uchar*) conv_name_str.c_ptr(), conv_name_str.length())))
|
|
#endif
|
|
spider_release_ping_table_mon_list_loop(mutex_hash, table_mon_list);
|
|
pthread_mutex_unlock(&spider_udf_table_mon_mutexes[mutex_hash]);
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
int spider_get_ping_table_mon(
|
|
THD *thd,
|
|
SPIDER_TABLE_MON_LIST *table_mon_list,
|
|
char *name,
|
|
uint name_length,
|
|
int link_idx,
|
|
uint32 server_id,
|
|
MEM_ROOT *mem_root,
|
|
bool need_lock
|
|
) {
|
|
int error_num;
|
|
TABLE *table_link_mon = NULL;
|
|
#if MYSQL_VERSION_ID < 50500
|
|
Open_tables_state open_tables_backup;
|
|
#else
|
|
Open_tables_backup open_tables_backup;
|
|
#endif
|
|
char table_key[MAX_KEY_LENGTH];
|
|
SPIDER_TABLE_MON *table_mon, *table_mon_prev = NULL;
|
|
SPIDER_SHARE *tmp_share;
|
|
char **tmp_connect_info, *tmp_ptr;
|
|
uint *tmp_connect_info_length;
|
|
long *tmp_long;
|
|
longlong *tmp_longlong;
|
|
int list_size = 0;
|
|
DBUG_ENTER("spider_get_ping_table_mon");
|
|
|
|
if (
|
|
!(table_link_mon = spider_open_sys_table(
|
|
thd, SPIDER_SYS_LINK_MON_TABLE_NAME_STR,
|
|
SPIDER_SYS_LINK_MON_TABLE_NAME_LEN, FALSE, &open_tables_backup,
|
|
need_lock, &error_num))
|
|
) {
|
|
my_error(error_num, MYF(0));
|
|
goto error;
|
|
}
|
|
spider_store_tables_name(table_link_mon, name, name_length);
|
|
spider_store_tables_link_idx(table_link_mon, link_idx);
|
|
if (!(error_num = spider_ping_table_cache_compare(table_link_mon, mem_root)))
|
|
goto create_table_mon;
|
|
if (error_num == HA_ERR_OUT_OF_MEM)
|
|
goto error;
|
|
if ((tmp_ptr = strstr(name, "#P#")))
|
|
{
|
|
*tmp_ptr = '\0';
|
|
spider_store_tables_name(table_link_mon, name, strlen(name));
|
|
*tmp_ptr = '#';
|
|
if (!(error_num = spider_ping_table_cache_compare(table_link_mon,
|
|
mem_root)))
|
|
goto create_table_mon;
|
|
if (error_num == HA_ERR_OUT_OF_MEM)
|
|
goto error;
|
|
}
|
|
error_num = HA_ERR_KEY_NOT_FOUND;
|
|
table_link_mon->file->print_error(error_num, MYF(0));
|
|
goto error;
|
|
|
|
create_table_mon:
|
|
if ((error_num = spider_get_sys_table_by_idx(table_link_mon, table_key,
|
|
table_link_mon->s->primary_key, 3)))
|
|
{
|
|
table_link_mon->file->print_error(error_num, MYF(0));
|
|
goto error;
|
|
}
|
|
|
|
do {
|
|
if (!(table_mon = (SPIDER_TABLE_MON *)
|
|
spider_bulk_malloc(spider_current_trx, 35, MYF(MY_WME | MY_ZEROFILL),
|
|
&table_mon, sizeof(SPIDER_TABLE_MON),
|
|
&tmp_share, sizeof(SPIDER_SHARE),
|
|
&tmp_connect_info, sizeof(char *) * SPIDER_TMP_SHARE_CHAR_PTR_COUNT,
|
|
&tmp_connect_info_length, sizeof(uint) * SPIDER_TMP_SHARE_UINT_COUNT,
|
|
&tmp_long, sizeof(long) * SPIDER_TMP_SHARE_LONG_COUNT,
|
|
&tmp_longlong, sizeof(longlong) * SPIDER_TMP_SHARE_LONGLONG_COUNT,
|
|
NullS))
|
|
) {
|
|
spider_sys_index_end(table_link_mon);
|
|
error_num = HA_ERR_OUT_OF_MEM;
|
|
my_error(HA_ERR_OUT_OF_MEM, MYF(0));
|
|
goto error;
|
|
}
|
|
spider_set_tmp_share_pointer(tmp_share, tmp_connect_info,
|
|
tmp_connect_info_length, tmp_long, tmp_longlong);
|
|
tmp_share->link_statuses[0] = -1;
|
|
table_mon->share = tmp_share;
|
|
if (table_mon_prev)
|
|
table_mon_prev->next = table_mon;
|
|
else
|
|
table_mon_list->first = table_mon;
|
|
table_mon_prev = table_mon;
|
|
if (
|
|
(error_num = spider_get_sys_link_mon_server_id(
|
|
table_link_mon, &table_mon->server_id, mem_root)) ||
|
|
(error_num = spider_get_sys_link_mon_connect_info(
|
|
table_link_mon, tmp_share, 0, mem_root))
|
|
) {
|
|
table_link_mon->file->print_error(error_num, MYF(0));
|
|
spider_sys_index_end(table_link_mon);
|
|
goto error;
|
|
}
|
|
if (
|
|
(error_num = spider_set_connect_info_default(
|
|
tmp_share,
|
|
#ifdef WITH_PARTITION_STORAGE_ENGINE
|
|
NULL,
|
|
NULL,
|
|
#endif
|
|
NULL
|
|
)) ||
|
|
(error_num = spider_set_connect_info_default_dbtable(
|
|
tmp_share, name, name_length
|
|
)) ||
|
|
(error_num = spider_create_conn_keys(tmp_share))
|
|
) {
|
|
spider_sys_index_end(table_link_mon);
|
|
goto error;
|
|
}
|
|
DBUG_PRINT("info",("spider table_mon->server_id=%u",
|
|
table_mon->server_id));
|
|
DBUG_PRINT("info",("spider server_id=%u", server_id));
|
|
if (table_mon->server_id == server_id)
|
|
table_mon_list->current = table_mon;
|
|
list_size++;
|
|
error_num = spider_sys_index_next_same(table_link_mon, table_key);
|
|
} while (error_num == 0);
|
|
spider_sys_index_end(table_link_mon);
|
|
spider_close_sys_table(thd, table_link_mon,
|
|
&open_tables_backup, need_lock);
|
|
table_link_mon = NULL;
|
|
table_mon_list->list_size = list_size;
|
|
|
|
if (!table_mon_list->current)
|
|
{
|
|
error_num = ER_SPIDER_UDF_PING_TABLE_NO_SERVER_ID_NUM;
|
|
my_printf_error(ER_SPIDER_UDF_PING_TABLE_NO_SERVER_ID_NUM,
|
|
ER_SPIDER_UDF_PING_TABLE_NO_SERVER_ID_STR, MYF(0));
|
|
goto error;
|
|
}
|
|
|
|
DBUG_RETURN(0);
|
|
|
|
error:
|
|
if (table_link_mon)
|
|
spider_close_sys_table(thd, table_link_mon,
|
|
&open_tables_backup, need_lock);
|
|
table_mon = table_mon_list->first;
|
|
table_mon_list->first = NULL;
|
|
table_mon_list->current = NULL;
|
|
while (table_mon)
|
|
{
|
|
spider_free_tmp_share_alloc(table_mon->share);
|
|
table_mon_prev = table_mon->next;
|
|
spider_free(spider_current_trx, table_mon, MYF(0));
|
|
table_mon = table_mon_prev;
|
|
}
|
|
DBUG_RETURN(error_num);
|
|
}
|
|
|
|
SPIDER_TABLE_MON_LIST *spider_get_ping_table_tgt(
|
|
THD *thd,
|
|
char *name,
|
|
uint name_length,
|
|
int link_idx,
|
|
uint32 server_id,
|
|
spider_string *str,
|
|
bool need_lock,
|
|
int *error_num
|
|
) {
|
|
TABLE *table_tables = NULL;
|
|
#if MYSQL_VERSION_ID < 50500
|
|
Open_tables_state open_tables_backup;
|
|
#else
|
|
Open_tables_backup open_tables_backup;
|
|
#endif
|
|
char table_key[MAX_KEY_LENGTH];
|
|
|
|
SPIDER_TABLE_MON_LIST *table_mon_list = NULL;
|
|
SPIDER_SHARE *tmp_share;
|
|
char **tmp_connect_info;
|
|
uint *tmp_connect_info_length;
|
|
long *tmp_long;
|
|
longlong *tmp_longlong;
|
|
char *key_str;
|
|
MEM_ROOT mem_root;
|
|
DBUG_ENTER("spider_get_ping_table_tgt");
|
|
|
|
SPD_INIT_ALLOC_ROOT(&mem_root, 4096, 0, MYF(MY_WME));
|
|
if (!(table_mon_list = (SPIDER_TABLE_MON_LIST *)
|
|
spider_bulk_malloc(spider_current_trx, 36, MYF(MY_WME | MY_ZEROFILL),
|
|
&table_mon_list, sizeof(SPIDER_TABLE_MON_LIST),
|
|
&tmp_share, sizeof(SPIDER_SHARE),
|
|
&tmp_connect_info, sizeof(char *) * SPIDER_TMP_SHARE_CHAR_PTR_COUNT,
|
|
&tmp_connect_info_length, sizeof(uint) * SPIDER_TMP_SHARE_UINT_COUNT,
|
|
&tmp_long, sizeof(long) * SPIDER_TMP_SHARE_LONG_COUNT,
|
|
&tmp_longlong, sizeof(longlong) * SPIDER_TMP_SHARE_LONGLONG_COUNT,
|
|
&key_str, str->length() + 1,
|
|
NullS))
|
|
) {
|
|
my_error(HA_ERR_OUT_OF_MEM, MYF(0));
|
|
goto error;
|
|
}
|
|
spider_set_tmp_share_pointer(tmp_share, tmp_connect_info,
|
|
tmp_connect_info_length, tmp_long, tmp_longlong);
|
|
table_mon_list->share = tmp_share;
|
|
table_mon_list->key = key_str;
|
|
table_mon_list->key_length = str->length();
|
|
memcpy(key_str, str->ptr(), table_mon_list->key_length);
|
|
tmp_share->access_charset = thd->variables.character_set_client;
|
|
|
|
if (
|
|
!(table_tables = spider_open_sys_table(
|
|
thd, SPIDER_SYS_TABLES_TABLE_NAME_STR,
|
|
SPIDER_SYS_TABLES_TABLE_NAME_LEN, FALSE, &open_tables_backup, need_lock,
|
|
error_num))
|
|
) {
|
|
my_error(*error_num, MYF(0));
|
|
goto error;
|
|
}
|
|
spider_store_tables_name(table_tables, name, name_length);
|
|
spider_store_tables_link_idx(table_tables, link_idx);
|
|
if (
|
|
(*error_num = spider_check_sys_table(table_tables, table_key)) ||
|
|
(*error_num = spider_get_sys_tables_connect_info(
|
|
table_tables, tmp_share, 0, &mem_root)) ||
|
|
(*error_num = spider_get_sys_tables_link_status(
|
|
table_tables, tmp_share, 0, &mem_root))
|
|
) {
|
|
table_tables->file->print_error(*error_num, MYF(0));
|
|
goto error;
|
|
}
|
|
spider_close_sys_table(thd, table_tables,
|
|
&open_tables_backup, need_lock);
|
|
table_tables = NULL;
|
|
|
|
if (
|
|
(*error_num = spider_set_connect_info_default(
|
|
tmp_share,
|
|
#ifdef WITH_PARTITION_STORAGE_ENGINE
|
|
NULL,
|
|
NULL,
|
|
#endif
|
|
NULL
|
|
)) ||
|
|
(*error_num = spider_set_connect_info_default_dbtable(
|
|
tmp_share, name, name_length
|
|
)) ||
|
|
(*error_num = spider_create_conn_keys(tmp_share)) ||
|
|
/*
|
|
(*error_num = spider_db_create_table_names_str(tmp_share)) ||
|
|
*/
|
|
(*error_num = spider_get_ping_table_mon(
|
|
thd, table_mon_list, name, name_length, link_idx, server_id, &mem_root,
|
|
need_lock))
|
|
)
|
|
goto error;
|
|
|
|
if (tmp_share->link_statuses[0] == SPIDER_LINK_STATUS_NG)
|
|
table_mon_list->mon_status = SPIDER_LINK_MON_NG;
|
|
|
|
#if MYSQL_VERSION_ID < 50500
|
|
if (pthread_mutex_init(&table_mon_list->caller_mutex, MY_MUTEX_INIT_FAST))
|
|
#else
|
|
if (mysql_mutex_init(spd_key_mutex_mon_list_caller,
|
|
&table_mon_list->caller_mutex, MY_MUTEX_INIT_FAST))
|
|
#endif
|
|
{
|
|
*error_num = HA_ERR_OUT_OF_MEM;
|
|
goto error_caller_mutex_init;
|
|
}
|
|
#if MYSQL_VERSION_ID < 50500
|
|
if (pthread_mutex_init(&table_mon_list->receptor_mutex, MY_MUTEX_INIT_FAST))
|
|
#else
|
|
if (mysql_mutex_init(spd_key_mutex_mon_list_receptor,
|
|
&table_mon_list->receptor_mutex, MY_MUTEX_INIT_FAST))
|
|
#endif
|
|
{
|
|
*error_num = HA_ERR_OUT_OF_MEM;
|
|
goto error_receptor_mutex_init;
|
|
}
|
|
#if MYSQL_VERSION_ID < 50500
|
|
if (pthread_mutex_init(&table_mon_list->monitor_mutex, MY_MUTEX_INIT_FAST))
|
|
#else
|
|
if (mysql_mutex_init(spd_key_mutex_mon_list_monitor,
|
|
&table_mon_list->monitor_mutex, MY_MUTEX_INIT_FAST))
|
|
#endif
|
|
{
|
|
*error_num = HA_ERR_OUT_OF_MEM;
|
|
goto error_monitor_mutex_init;
|
|
}
|
|
#if MYSQL_VERSION_ID < 50500
|
|
if (pthread_mutex_init(&table_mon_list->update_status_mutex,
|
|
MY_MUTEX_INIT_FAST))
|
|
#else
|
|
if (mysql_mutex_init(spd_key_mutex_mon_list_update_status,
|
|
&table_mon_list->update_status_mutex, MY_MUTEX_INIT_FAST))
|
|
#endif
|
|
{
|
|
*error_num = HA_ERR_OUT_OF_MEM;
|
|
goto error_update_status_mutex_init;
|
|
}
|
|
|
|
free_root(&mem_root, MYF(0));
|
|
DBUG_RETURN(table_mon_list);
|
|
|
|
error_update_status_mutex_init:
|
|
pthread_mutex_destroy(&table_mon_list->monitor_mutex);
|
|
error_monitor_mutex_init:
|
|
pthread_mutex_destroy(&table_mon_list->receptor_mutex);
|
|
error_receptor_mutex_init:
|
|
pthread_mutex_destroy(&table_mon_list->caller_mutex);
|
|
error_caller_mutex_init:
|
|
error:
|
|
if (table_tables)
|
|
spider_close_sys_table(thd, table_tables,
|
|
&open_tables_backup, need_lock);
|
|
free_root(&mem_root, MYF(0));
|
|
if (table_mon_list)
|
|
{
|
|
spider_free_tmp_share_alloc(table_mon_list->share);
|
|
spider_free(spider_current_trx, table_mon_list, MYF(0));
|
|
}
|
|
DBUG_RETURN(NULL);
|
|
}
|
|
|
|
SPIDER_CONN *spider_get_ping_table_tgt_conn(
|
|
SPIDER_TRX *trx,
|
|
SPIDER_SHARE *share,
|
|
int *error_num
|
|
) {
|
|
SPIDER_CONN *conn;
|
|
DBUG_ENTER("spider_get_ping_table_tgt_conn");
|
|
#ifndef WITHOUT_SPIDER_BG_SEARCH
|
|
if (trx == spider_global_trx)
|
|
pthread_mutex_lock(&spider_global_trx_mutex);
|
|
#endif
|
|
if (
|
|
!(conn = spider_get_conn(
|
|
share, 0, share->conn_keys[0], trx, NULL, FALSE, FALSE,
|
|
SPIDER_CONN_KIND_MYSQL, error_num))
|
|
) {
|
|
#ifndef WITHOUT_SPIDER_BG_SEARCH
|
|
if (trx == spider_global_trx)
|
|
pthread_mutex_unlock(&spider_global_trx_mutex);
|
|
#endif
|
|
my_error(ER_CONNECT_TO_FOREIGN_DATA_SOURCE, MYF(0),
|
|
share->server_names[0]);
|
|
*error_num = ER_CONNECT_TO_FOREIGN_DATA_SOURCE;
|
|
goto error;
|
|
}
|
|
conn->error_mode = 0;
|
|
#ifndef WITHOUT_SPIDER_BG_SEARCH
|
|
if (trx == spider_global_trx)
|
|
pthread_mutex_unlock(&spider_global_trx_mutex);
|
|
#endif
|
|
DBUG_RETURN(conn);
|
|
|
|
error:
|
|
DBUG_RETURN(NULL);
|
|
}
|
|
|
|
int spider_init_ping_table_mon_cache(
|
|
THD *thd,
|
|
MEM_ROOT *mem_root,
|
|
bool need_lock
|
|
) {
|
|
int error_num, same;
|
|
TABLE *table_link_mon = NULL;
|
|
#if MYSQL_VERSION_ID < 50500
|
|
Open_tables_state open_tables_backup;
|
|
#else
|
|
Open_tables_backup open_tables_backup;
|
|
#endif
|
|
SPIDER_MON_KEY mon_key;
|
|
DBUG_ENTER("spider_init_ping_table_mon_cache");
|
|
|
|
if (
|
|
!(table_link_mon = spider_open_sys_table(
|
|
thd, SPIDER_SYS_LINK_MON_TABLE_NAME_STR,
|
|
SPIDER_SYS_LINK_MON_TABLE_NAME_LEN, FALSE, &open_tables_backup,
|
|
need_lock, &error_num))
|
|
) {
|
|
my_error(error_num, MYF(0));
|
|
goto error_open_sys_table;
|
|
}
|
|
|
|
pthread_mutex_lock(&spider_mon_table_cache_mutex);
|
|
if (spider_mon_table_cache_version != spider_mon_table_cache_version_req)
|
|
{
|
|
/* reset */
|
|
spider_mon_table_cache.elements = 0;
|
|
|
|
if ((error_num = spider_sys_index_first(table_link_mon,
|
|
table_link_mon->s->primary_key)))
|
|
{
|
|
if (error_num != HA_ERR_KEY_NOT_FOUND && error_num != HA_ERR_END_OF_FILE)
|
|
{
|
|
table_link_mon->file->print_error(error_num, MYF(0));
|
|
goto error_sys_index_first;
|
|
}
|
|
}
|
|
|
|
if (!error_num)
|
|
{
|
|
mon_key.db_name_length = SPIDER_SYS_LINK_MON_TABLE_DB_NAME_SIZE + 1;
|
|
mon_key.table_name_length = SPIDER_SYS_LINK_MON_TABLE_TABLE_NAME_SIZE + 1;
|
|
mon_key.link_id_length = SPIDER_SYS_LINK_MON_TABLE_LINK_ID_SIZE + 1;
|
|
do {
|
|
if ((error_num = spider_get_sys_link_mon_key(table_link_mon, &mon_key,
|
|
mem_root, &same)))
|
|
goto error_get_sys_link_mon_key;
|
|
|
|
if (!same)
|
|
{
|
|
mon_key.sort = spider_calc_for_sort(3, mon_key.db_name,
|
|
mon_key.table_name, mon_key.link_id);
|
|
if (push_dynamic(&spider_mon_table_cache, (uchar *) &mon_key))
|
|
{
|
|
error_num = HA_ERR_OUT_OF_MEM;
|
|
goto error_push_dynamic;
|
|
}
|
|
}
|
|
|
|
if ((error_num = spider_sys_index_next(table_link_mon)))
|
|
{
|
|
if (
|
|
error_num != HA_ERR_KEY_NOT_FOUND &&
|
|
error_num != HA_ERR_END_OF_FILE
|
|
) {
|
|
table_link_mon->file->print_error(error_num, MYF(0));
|
|
goto error_sys_index_next;
|
|
}
|
|
}
|
|
} while (!error_num);
|
|
spider_sys_index_end(table_link_mon);
|
|
}
|
|
my_qsort(
|
|
(uchar *) dynamic_element(&spider_mon_table_cache, 0, SPIDER_MON_KEY *),
|
|
spider_mon_table_cache.elements, sizeof(SPIDER_MON_KEY),
|
|
(qsort_cmp) spider_compare_for_sort);
|
|
uint old_elements = spider_mon_table_cache.max_element;
|
|
freeze_size(&spider_mon_table_cache);
|
|
if (spider_mon_table_cache.max_element < old_elements)
|
|
{
|
|
spider_free_mem_calc(spider_current_trx,
|
|
spider_mon_table_cache_id,
|
|
spider_mon_table_cache.max_element *
|
|
spider_mon_table_cache.size_of_element);
|
|
}
|
|
spider_mon_table_cache_version = spider_mon_table_cache_version_req;
|
|
}
|
|
pthread_mutex_unlock(&spider_mon_table_cache_mutex);
|
|
spider_close_sys_table(thd, table_link_mon, &open_tables_backup, need_lock);
|
|
DBUG_RETURN(0);
|
|
|
|
error_push_dynamic:
|
|
error_get_sys_link_mon_key:
|
|
error_sys_index_next:
|
|
spider_sys_index_end(table_link_mon);
|
|
error_sys_index_first:
|
|
pthread_mutex_unlock(&spider_mon_table_cache_mutex);
|
|
spider_close_sys_table(thd, table_link_mon, &open_tables_backup, need_lock);
|
|
error_open_sys_table:
|
|
DBUG_RETURN(error_num);
|
|
}
|
|
|
|
int spider_ping_table_cache_compare(
|
|
TABLE *table,
|
|
MEM_ROOT *mem_root
|
|
) {
|
|
uint32 roop_count;
|
|
SPIDER_MON_KEY *mon_key;
|
|
char *db_name, *table_name, *link_id;
|
|
DBUG_ENTER("spider_ping_table_cache_compare");
|
|
|
|
if (
|
|
!(db_name = get_field(mem_root, table->field[0])) ||
|
|
!(table_name = get_field(mem_root, table->field[1])) ||
|
|
!(link_id = get_field(mem_root, table->field[2]))
|
|
)
|
|
DBUG_RETURN(HA_ERR_OUT_OF_MEM);
|
|
DBUG_PRINT("info", ("spider db_name=%s", db_name));
|
|
DBUG_PRINT("info", ("spider table_name=%s", table_name));
|
|
DBUG_PRINT("info", ("spider link_id=%s", link_id));
|
|
|
|
pthread_mutex_lock(&spider_mon_table_cache_mutex);
|
|
for (roop_count = 0; roop_count < spider_mon_table_cache.elements;
|
|
roop_count++)
|
|
{
|
|
mon_key = dynamic_element(&spider_mon_table_cache, roop_count,
|
|
SPIDER_MON_KEY *);
|
|
DBUG_PRINT("info", ("spider roop_count=%d", roop_count));
|
|
DBUG_PRINT("info", ("spider mon_key.db_name=%s", mon_key->db_name));
|
|
DBUG_PRINT("info", ("spider mon_key.table_name=%s", mon_key->table_name));
|
|
DBUG_PRINT("info", ("spider mon_key.link_id=%s", mon_key->link_id));
|
|
if (
|
|
!wild_case_compare(system_charset_info, db_name, mon_key->db_name) &&
|
|
!wild_case_compare(system_charset_info, table_name,
|
|
mon_key->table_name) &&
|
|
!wild_case_compare(system_charset_info, link_id, mon_key->link_id)
|
|
) {
|
|
spider_store_db_and_table_name(
|
|
table,
|
|
mon_key->db_name,
|
|
mon_key->db_name_length,
|
|
mon_key->table_name,
|
|
mon_key->table_name_length
|
|
);
|
|
spider_store_tables_link_idx_str(
|
|
table,
|
|
mon_key->link_id,
|
|
mon_key->link_id_length
|
|
);
|
|
pthread_mutex_unlock(&spider_mon_table_cache_mutex);
|
|
DBUG_PRINT("info", ("spider found"));
|
|
DBUG_RETURN(0);
|
|
}
|
|
}
|
|
pthread_mutex_unlock(&spider_mon_table_cache_mutex);
|
|
DBUG_PRINT("info", ("spider not found"));
|
|
DBUG_RETURN(1);
|
|
}
|
|
|
|
long long spider_ping_table_body(
|
|
UDF_INIT *initid,
|
|
UDF_ARGS *args,
|
|
char *is_null,
|
|
char *error
|
|
) {
|
|
int error_num = 0, link_idx, flags, full_mon_count, current_mon_count,
|
|
success_count, fault_count, tmp_error_num = 0;
|
|
uint32 first_sid;
|
|
longlong limit, tmp_sid = -1;
|
|
SPIDER_MON_TABLE_RESULT *mon_table_result =
|
|
(SPIDER_MON_TABLE_RESULT *) initid->ptr;
|
|
SPIDER_TRX *trx = mon_table_result->trx;
|
|
THD *thd = trx->thd;
|
|
SPIDER_CONN *ping_conn = NULL, *mon_conn;
|
|
char *where_clause;
|
|
SPIDER_TABLE_MON_LIST *table_mon_list;
|
|
SPIDER_TABLE_MON *table_mon;
|
|
|
|
char buf[MAX_FIELD_WIDTH];
|
|
spider_string conv_name(buf, sizeof(buf), system_charset_info);
|
|
int conv_name_length;
|
|
char link_idx_str[SPIDER_SQL_INT_LEN];
|
|
int link_idx_str_length;
|
|
bool get_lock = FALSE;
|
|
DBUG_ENTER("spider_ping_table_body");
|
|
conv_name.init_calc_mem(135);
|
|
conv_name.length(0);
|
|
if (
|
|
thd->open_tables != 0 ||
|
|
thd->handler_tables_hash.records != 0 ||
|
|
thd->derived_tables != 0 ||
|
|
thd->lock != 0 ||
|
|
#if MYSQL_VERSION_ID < 50500
|
|
thd->locked_tables != 0 ||
|
|
thd->prelocked_mode != NON_PRELOCKED
|
|
#else
|
|
thd->locked_tables_list.locked_tables() ||
|
|
thd->locked_tables_mode != LTM_NONE
|
|
#endif
|
|
) {
|
|
if (thd->open_tables != 0)
|
|
{
|
|
my_printf_error(ER_SPIDER_UDF_CANT_USE_IF_OPEN_TABLE_NUM,
|
|
ER_SPIDER_UDF_CANT_USE_IF_OPEN_TABLE_STR_WITH_PTR, MYF(0),
|
|
"thd->open_tables", thd->open_tables);
|
|
} else if (thd->handler_tables_hash.records != 0)
|
|
{
|
|
my_printf_error(ER_SPIDER_UDF_CANT_USE_IF_OPEN_TABLE_NUM,
|
|
ER_SPIDER_UDF_CANT_USE_IF_OPEN_TABLE_STR_WITH_NUM, MYF(0),
|
|
"thd->handler_tables_hash.records",
|
|
(longlong) thd->handler_tables_hash.records);
|
|
} else if (thd->derived_tables != 0)
|
|
{
|
|
my_printf_error(ER_SPIDER_UDF_CANT_USE_IF_OPEN_TABLE_NUM,
|
|
ER_SPIDER_UDF_CANT_USE_IF_OPEN_TABLE_STR_WITH_PTR, MYF(0),
|
|
"thd->derived_tables", thd->derived_tables);
|
|
} else if (thd->lock != 0)
|
|
{
|
|
my_printf_error(ER_SPIDER_UDF_CANT_USE_IF_OPEN_TABLE_NUM,
|
|
ER_SPIDER_UDF_CANT_USE_IF_OPEN_TABLE_STR_WITH_PTR, MYF(0),
|
|
"thd->lock", thd->lock);
|
|
#if MYSQL_VERSION_ID < 50500
|
|
} else if (thd->locked_tables != 0)
|
|
{
|
|
my_printf_error(ER_SPIDER_UDF_CANT_USE_IF_OPEN_TABLE_NUM,
|
|
ER_SPIDER_UDF_CANT_USE_IF_OPEN_TABLE_STR_WITH_PTR, MYF(0),
|
|
"thd->locked_tables", thd->locked_tables);
|
|
} else if (thd->prelocked_mode != NON_PRELOCKED)
|
|
{
|
|
my_printf_error(ER_SPIDER_UDF_CANT_USE_IF_OPEN_TABLE_NUM,
|
|
ER_SPIDER_UDF_CANT_USE_IF_OPEN_TABLE_STR_WITH_NUM, MYF(0),
|
|
"thd->prelocked_mode", (longlong) thd->prelocked_mode);
|
|
#else
|
|
} else if (thd->locked_tables_list.locked_tables())
|
|
{
|
|
my_printf_error(ER_SPIDER_UDF_CANT_USE_IF_OPEN_TABLE_NUM,
|
|
ER_SPIDER_UDF_CANT_USE_IF_OPEN_TABLE_STR_WITH_PTR, MYF(0),
|
|
"thd->locked_tables_list.locked_tables()",
|
|
thd->locked_tables_list.locked_tables());
|
|
} else if (thd->locked_tables_mode != LTM_NONE)
|
|
{
|
|
my_printf_error(ER_SPIDER_UDF_CANT_USE_IF_OPEN_TABLE_NUM,
|
|
ER_SPIDER_UDF_CANT_USE_IF_OPEN_TABLE_STR_WITH_NUM, MYF(0),
|
|
"thd->locked_tables_mode", (longlong) thd->locked_tables_mode);
|
|
#endif
|
|
}
|
|
goto error;
|
|
}
|
|
|
|
if (
|
|
args->lengths[0] > SPIDER_CONNECT_INFO_MAX_LEN
|
|
) {
|
|
my_printf_error(ER_SPIDER_UDF_PING_TABLE_PARAM_TOO_LONG_NUM,
|
|
ER_SPIDER_UDF_PING_TABLE_PARAM_TOO_LONG_STR, MYF(0));
|
|
goto error;
|
|
}
|
|
if (
|
|
args->lengths[0] == 0
|
|
) {
|
|
my_printf_error(ER_SPIDER_UDF_PING_TABLE_PARAM_REQIRED_NUM,
|
|
ER_SPIDER_UDF_PING_TABLE_PARAM_REQIRED_STR, MYF(0));
|
|
goto error;
|
|
}
|
|
|
|
link_idx = (int) (args->args[1] ? *((longlong *) args->args[1]) : 0);
|
|
flags = (int) (args->args[2] ? *((longlong *) args->args[2]) : 0);
|
|
limit = args->args[3] ? *((longlong *) args->args[3]) : 0;
|
|
where_clause = args->args[4] ? args->args[4] : (char *) "";
|
|
|
|
link_idx_str_length = my_sprintf(link_idx_str, (link_idx_str, "%010d",
|
|
link_idx));
|
|
|
|
if (conv_name.append(args->args[0], args->lengths[0],
|
|
trx->thd->variables.character_set_client))
|
|
{
|
|
my_error(HA_ERR_OUT_OF_MEM, MYF(0));
|
|
goto error;
|
|
}
|
|
conv_name_length = conv_name.length();
|
|
if (conv_name.reserve(link_idx_str_length + 1))
|
|
{
|
|
my_error(HA_ERR_OUT_OF_MEM, MYF(0));
|
|
goto error;
|
|
}
|
|
conv_name.q_append(link_idx_str, link_idx_str_length + 1);
|
|
conv_name.length(conv_name.length() - 1);
|
|
|
|
#if defined(MARIADB_BASE_VERSION) && MYSQL_VERSION_ID >= 100002
|
|
if (!(table_mon_list = spider_get_ping_table_mon_list(trx, trx->thd,
|
|
&conv_name, conv_name_length, link_idx, global_system_variables.server_id,
|
|
TRUE, &error_num)))
|
|
#else
|
|
if (!(table_mon_list = spider_get_ping_table_mon_list(trx, trx->thd,
|
|
&conv_name, conv_name_length, link_idx, thd->server_id, TRUE, &error_num)))
|
|
#endif
|
|
goto error;
|
|
|
|
if (table_mon_list->mon_status == SPIDER_LINK_MON_NG)
|
|
{
|
|
mon_table_result->result_status = SPIDER_LINK_MON_NG;
|
|
DBUG_PRINT("info",
|
|
("spider mon_table_result->result_status=SPIDER_LINK_MON_NG 1"));
|
|
goto end;
|
|
}
|
|
|
|
if (args->args[5])
|
|
tmp_sid = *((longlong *) args->args[5]);
|
|
|
|
if (tmp_sid >= 0)
|
|
{
|
|
first_sid = (uint32) tmp_sid;
|
|
full_mon_count = (int) (args->args[6] ? *((longlong *) args->args[6]) : 0);
|
|
current_mon_count =
|
|
(int) (args->args[7] ? *((longlong *) args->args[7]) + 1 : 1);
|
|
if (full_mon_count != table_mon_list->list_size)
|
|
{
|
|
my_printf_error(ER_SPIDER_UDF_PING_TABLE_DIFFERENT_MON_NUM,
|
|
ER_SPIDER_UDF_PING_TABLE_DIFFERENT_MON_STR, MYF(0));
|
|
goto error_with_free_table_mon_list;
|
|
}
|
|
} else {
|
|
#if defined(MARIADB_BASE_VERSION) && MYSQL_VERSION_ID >= 100002
|
|
first_sid = global_system_variables.server_id;
|
|
#else
|
|
first_sid = thd->server_id;
|
|
#endif
|
|
full_mon_count = table_mon_list->list_size;
|
|
current_mon_count = 1;
|
|
}
|
|
|
|
success_count = (int) (args->args[8] ? *((longlong *) args->args[8]) : 0);
|
|
fault_count = (int) (args->args[9] ? *((longlong *) args->args[9]) : 0);
|
|
if (
|
|
table_mon_list->mon_status != SPIDER_LINK_MON_NG &&
|
|
!(ping_conn = spider_get_ping_table_tgt_conn(trx,
|
|
table_mon_list->share, &error_num))
|
|
) {
|
|
if (error_num == HA_ERR_OUT_OF_MEM)
|
|
goto error_with_free_table_mon_list;
|
|
else
|
|
thd->clear_error();
|
|
}
|
|
if (
|
|
table_mon_list->mon_status == SPIDER_LINK_MON_NG ||
|
|
error_num ||
|
|
(tmp_error_num = spider_db_udf_ping_table(table_mon_list, table_mon_list->share, trx,
|
|
ping_conn, where_clause, args->lengths[4],
|
|
(flags & SPIDER_UDF_PING_TABLE_PING_ONLY),
|
|
(flags & SPIDER_UDF_PING_TABLE_USE_WHERE),
|
|
limit
|
|
))
|
|
) {
|
|
if (tmp_error_num == HA_ERR_OUT_OF_MEM)
|
|
goto error_with_free_table_mon_list;
|
|
else if(tmp_error_num)
|
|
thd->clear_error();
|
|
fault_count++;
|
|
error_num = 0;
|
|
if (fault_count > full_mon_count / 2)
|
|
{
|
|
mon_table_result->result_status = SPIDER_LINK_MON_NG;
|
|
DBUG_PRINT("info",("spider mon_table_result->result_status=SPIDER_LINK_MON_NG 2"));
|
|
if (table_mon_list->mon_status != SPIDER_LINK_MON_NG)
|
|
{
|
|
pthread_mutex_lock(&table_mon_list->update_status_mutex);
|
|
if (table_mon_list->mon_status != SPIDER_LINK_MON_NG)
|
|
{
|
|
table_mon_list->mon_status = SPIDER_LINK_MON_NG;
|
|
table_mon_list->share->link_statuses[0] = SPIDER_LINK_STATUS_NG;
|
|
spider_sys_update_tables_link_status(trx->thd,
|
|
conv_name.c_ptr(), conv_name_length, link_idx,
|
|
SPIDER_LINK_STATUS_NG, TRUE);
|
|
spider_sys_log_tables_link_failed(trx->thd,
|
|
conv_name.c_ptr(), conv_name_length, link_idx, TRUE);
|
|
}
|
|
pthread_mutex_unlock(&table_mon_list->update_status_mutex);
|
|
}
|
|
goto end;
|
|
}
|
|
} else {
|
|
success_count++;
|
|
if (success_count > full_mon_count / 2)
|
|
{
|
|
mon_table_result->result_status = SPIDER_LINK_MON_OK;
|
|
DBUG_PRINT("info",("spider mon_table_result->result_status=SPIDER_LINK_MON_OK 1"));
|
|
goto end;
|
|
}
|
|
}
|
|
|
|
if (tmp_sid < 0)
|
|
{
|
|
if (!pthread_mutex_trylock(&table_mon_list->receptor_mutex))
|
|
get_lock = TRUE;
|
|
}
|
|
|
|
if (
|
|
tmp_sid >= 0 ||
|
|
get_lock
|
|
) {
|
|
table_mon = table_mon_list->current->next;
|
|
while (TRUE)
|
|
{
|
|
if (!table_mon)
|
|
table_mon = table_mon_list->first;
|
|
if (
|
|
table_mon->server_id == first_sid ||
|
|
current_mon_count > full_mon_count
|
|
) {
|
|
if (success_count + fault_count > full_mon_count / 2)
|
|
{
|
|
mon_table_result->result_status = SPIDER_LINK_MON_DRAW;
|
|
DBUG_PRINT("info",(
|
|
"spider mon_table_result->result_status=SPIDER_LINK_MON_DRAW 1"));
|
|
} else {
|
|
mon_table_result->result_status = SPIDER_LINK_MON_DRAW_FEW_MON;
|
|
DBUG_PRINT("info",(
|
|
"spider mon_table_result->result_status=SPIDER_LINK_MON_DRAW_FEW_MON 1"));
|
|
}
|
|
table_mon_list->last_receptor_result = mon_table_result->result_status;
|
|
break;
|
|
}
|
|
if ((mon_conn = spider_get_ping_table_tgt_conn(trx,
|
|
table_mon->share, &error_num))
|
|
) {
|
|
if (!spider_db_udf_ping_table_mon_next(
|
|
thd, table_mon, mon_conn, mon_table_result, args->args[0],
|
|
args->lengths[0], link_idx,
|
|
where_clause, args->lengths[4], first_sid, full_mon_count,
|
|
current_mon_count, success_count, fault_count, flags, limit))
|
|
{
|
|
if (
|
|
mon_table_result->result_status == SPIDER_LINK_MON_NG &&
|
|
table_mon_list->mon_status != SPIDER_LINK_MON_NG
|
|
) {
|
|
pthread_mutex_lock(&table_mon_list->update_status_mutex);
|
|
if (table_mon_list->mon_status != SPIDER_LINK_MON_NG)
|
|
{
|
|
table_mon_list->mon_status = SPIDER_LINK_MON_NG;
|
|
table_mon_list->share->link_statuses[0] = SPIDER_LINK_STATUS_NG;
|
|
spider_sys_update_tables_link_status(trx->thd,
|
|
conv_name.c_ptr(), conv_name_length, link_idx,
|
|
SPIDER_LINK_STATUS_NG, TRUE);
|
|
spider_sys_log_tables_link_failed(trx->thd,
|
|
conv_name.c_ptr(), conv_name_length, link_idx, TRUE);
|
|
}
|
|
pthread_mutex_unlock(&table_mon_list->update_status_mutex);
|
|
}
|
|
table_mon_list->last_receptor_result =
|
|
mon_table_result->result_status;
|
|
break;
|
|
}
|
|
}
|
|
thd->clear_error();
|
|
table_mon = table_mon->next;
|
|
current_mon_count++;
|
|
}
|
|
if (get_lock)
|
|
pthread_mutex_unlock(&table_mon_list->receptor_mutex);
|
|
} else {
|
|
pthread_mutex_lock(&table_mon_list->receptor_mutex);
|
|
mon_table_result->result_status = table_mon_list->last_receptor_result;
|
|
DBUG_PRINT("info",("spider mon_table_result->result_status=%d 1",
|
|
table_mon_list->last_receptor_result));
|
|
pthread_mutex_unlock(&table_mon_list->receptor_mutex);
|
|
}
|
|
|
|
end:
|
|
spider_free_ping_table_mon_list(table_mon_list);
|
|
DBUG_RETURN(mon_table_result->result_status);
|
|
|
|
error_with_free_table_mon_list:
|
|
spider_free_ping_table_mon_list(table_mon_list);
|
|
error:
|
|
*error = 1;
|
|
DBUG_RETURN(0);
|
|
}
|
|
|
|
my_bool spider_ping_table_init_body(
|
|
UDF_INIT *initid,
|
|
UDF_ARGS *args,
|
|
char *message
|
|
) {
|
|
int error_num;
|
|
THD *thd = current_thd;
|
|
SPIDER_TRX *trx;
|
|
SPIDER_MON_TABLE_RESULT *mon_table_result = NULL;
|
|
DBUG_ENTER("spider_ping_table_init_body");
|
|
if (args->arg_count != 10)
|
|
{
|
|
strcpy(message, "spider_ping_table() requires 10 arguments");
|
|
goto error;
|
|
}
|
|
if (
|
|
args->arg_type[0] != STRING_RESULT ||
|
|
args->arg_type[4] != STRING_RESULT
|
|
) {
|
|
strcpy(message, "spider_ping_table() requires string 1st "
|
|
"and 5th arguments");
|
|
goto error;
|
|
}
|
|
if (
|
|
args->arg_type[1] != INT_RESULT ||
|
|
args->arg_type[2] != INT_RESULT ||
|
|
args->arg_type[3] != INT_RESULT ||
|
|
args->arg_type[5] != INT_RESULT ||
|
|
args->arg_type[6] != INT_RESULT ||
|
|
args->arg_type[7] != INT_RESULT ||
|
|
args->arg_type[8] != INT_RESULT ||
|
|
args->arg_type[9] != INT_RESULT
|
|
) {
|
|
strcpy(message, "spider_ping_table() requires integer 2nd, 3rd, 4,6,7,8,"
|
|
"9th and 10th argument");
|
|
goto error;
|
|
}
|
|
|
|
if (!(trx = spider_get_trx(thd, TRUE, &error_num)))
|
|
{
|
|
my_error(error_num, MYF(0));
|
|
strcpy(message, spider_stmt_da_message(thd));
|
|
goto error;
|
|
}
|
|
|
|
if (!(mon_table_result = (SPIDER_MON_TABLE_RESULT *)
|
|
spider_malloc(spider_current_trx, 11, sizeof(SPIDER_MON_TABLE_RESULT),
|
|
MYF(MY_WME | MY_ZEROFILL)))
|
|
) {
|
|
strcpy(message, "spider_ping_table() out of memory");
|
|
goto error;
|
|
}
|
|
mon_table_result->trx = trx;
|
|
initid->ptr = (char *) mon_table_result;
|
|
DBUG_RETURN(FALSE);
|
|
|
|
error:
|
|
if (mon_table_result)
|
|
{
|
|
spider_free(spider_current_trx, mon_table_result, MYF(0));
|
|
}
|
|
DBUG_RETURN(TRUE);
|
|
}
|
|
|
|
void spider_ping_table_deinit_body(
|
|
UDF_INIT *initid
|
|
) {
|
|
SPIDER_MON_TABLE_RESULT *mon_table_result =
|
|
(SPIDER_MON_TABLE_RESULT *) initid->ptr;
|
|
DBUG_ENTER("spider_ping_table_deinit_body");
|
|
if (mon_table_result)
|
|
{
|
|
spider_free(spider_current_trx, mon_table_result, MYF(0));
|
|
}
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
long long spider_flush_table_mon_cache_body()
|
|
{
|
|
DBUG_ENTER("spider_flush_table_mon_cache_body");
|
|
spider_mon_table_cache_version_req++;
|
|
DBUG_RETURN(1);
|
|
}
|
|
|
|
void spider_ping_table_free_mon_list(
|
|
SPIDER_TABLE_MON_LIST *table_mon_list
|
|
) {
|
|
DBUG_ENTER("spider_ping_table_free_mon_list");
|
|
if (table_mon_list)
|
|
{
|
|
spider_ping_table_free_mon(table_mon_list->first);
|
|
spider_free_tmp_share_alloc(table_mon_list->share);
|
|
pthread_mutex_destroy(&table_mon_list->update_status_mutex);
|
|
pthread_mutex_destroy(&table_mon_list->monitor_mutex);
|
|
pthread_mutex_destroy(&table_mon_list->receptor_mutex);
|
|
pthread_mutex_destroy(&table_mon_list->caller_mutex);
|
|
spider_free(spider_current_trx, table_mon_list, MYF(0));
|
|
}
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
void spider_ping_table_free_mon(
|
|
SPIDER_TABLE_MON *table_mon
|
|
) {
|
|
SPIDER_TABLE_MON *table_mon_next;
|
|
DBUG_ENTER("spider_ping_table_free_mon");
|
|
while (table_mon)
|
|
{
|
|
spider_free_tmp_share_alloc(table_mon->share);
|
|
table_mon_next = table_mon->next;
|
|
spider_free(spider_current_trx, table_mon, MYF(0));
|
|
table_mon = table_mon_next;
|
|
}
|
|
DBUG_VOID_RETURN;
|
|
}
|
|
|
|
int spider_ping_table_mon_from_table(
|
|
SPIDER_TRX *trx,
|
|
THD *thd,
|
|
SPIDER_SHARE *share,
|
|
uint32 server_id,
|
|
char *conv_name,
|
|
uint conv_name_length,
|
|
int link_idx,
|
|
char *where_clause,
|
|
uint where_clause_length,
|
|
long monitoring_kind,
|
|
longlong monitoring_limit,
|
|
bool need_lock
|
|
) {
|
|
int error_num = 0, current_mon_count, flags;
|
|
uint32 first_sid;
|
|
/*
|
|
THD *thd = trx->thd;
|
|
*/
|
|
SPIDER_TABLE_MON_LIST *table_mon_list;
|
|
SPIDER_TABLE_MON *table_mon;
|
|
SPIDER_MON_TABLE_RESULT mon_table_result;
|
|
SPIDER_CONN *mon_conn;
|
|
TABLE_SHARE *table_share = share->table_share;
|
|
char link_idx_str[SPIDER_SQL_INT_LEN];
|
|
int link_idx_str_length;
|
|
uint sql_command = thd_sql_command(thd);
|
|
DBUG_ENTER("spider_ping_table_mon_from_table");
|
|
if (table_share->tmp_table != NO_TMP_TABLE)
|
|
{
|
|
my_printf_error(ER_SPIDER_TMP_TABLE_MON_NUM,
|
|
ER_SPIDER_TMP_TABLE_MON_STR, MYF(0));
|
|
DBUG_RETURN(ER_SPIDER_TMP_TABLE_MON_NUM);
|
|
}
|
|
if (
|
|
sql_command == SQLCOM_DROP_TABLE ||
|
|
sql_command == SQLCOM_ALTER_TABLE
|
|
) {
|
|
my_printf_error(ER_SPIDER_MON_AT_ALTER_TABLE_NUM,
|
|
ER_SPIDER_MON_AT_ALTER_TABLE_STR, MYF(0));
|
|
DBUG_RETURN(ER_SPIDER_MON_AT_ALTER_TABLE_NUM);
|
|
}
|
|
DBUG_PRINT("info",("spider thd->killed=%s",
|
|
thd ? (thd->killed ? "TRUE" : "FALSE") : "NULL"));
|
|
DBUG_PRINT("info",("spider abort_loop=%s",
|
|
*spd_abort_loop ? "TRUE" : "FALSE"));
|
|
if (
|
|
(thd && thd->killed) ||
|
|
*spd_abort_loop
|
|
) {
|
|
DBUG_RETURN(ER_SPIDER_COND_SKIP_NUM);
|
|
}
|
|
|
|
link_idx_str_length = my_sprintf(link_idx_str, (link_idx_str, "%010d",
|
|
link_idx));
|
|
#ifdef _MSC_VER
|
|
spider_string conv_name_str(conv_name_length + link_idx_str_length + 1);
|
|
conv_name_str.set_charset(system_charset_info);
|
|
*((char *)(conv_name_str.ptr() + conv_name_length + link_idx_str_length)) =
|
|
'\0';
|
|
#else
|
|
char buf[conv_name_length + link_idx_str_length + 1];
|
|
buf[conv_name_length + link_idx_str_length] = '\0';
|
|
spider_string conv_name_str(buf, conv_name_length + link_idx_str_length + 1,
|
|
system_charset_info);
|
|
#endif
|
|
conv_name_str.init_calc_mem(136);
|
|
conv_name_str.length(0);
|
|
conv_name_str.q_append(conv_name, conv_name_length);
|
|
conv_name_str.q_append(link_idx_str, link_idx_str_length + 1);
|
|
conv_name_str.length(conv_name_str.length() - 1);
|
|
|
|
if (monitoring_kind == 1)
|
|
flags = SPIDER_UDF_PING_TABLE_PING_ONLY;
|
|
else if (monitoring_kind == 3)
|
|
flags = SPIDER_UDF_PING_TABLE_USE_WHERE;
|
|
else
|
|
flags = 0;
|
|
|
|
if (!(table_mon_list = spider_get_ping_table_mon_list(trx, thd,
|
|
&conv_name_str, conv_name_length, link_idx, server_id, need_lock,
|
|
&error_num)))
|
|
goto end;
|
|
|
|
if (table_mon_list->mon_status == SPIDER_LINK_MON_NG)
|
|
{
|
|
DBUG_PRINT("info",
|
|
("spider share->link_statuses[%d]=SPIDER_LINK_STATUS_NG", link_idx));
|
|
share->link_statuses[link_idx] = SPIDER_LINK_STATUS_NG;
|
|
error_num = ER_SPIDER_LINK_MON_NG_NUM;
|
|
my_printf_error(error_num,
|
|
ER_SPIDER_LINK_MON_NG_STR, MYF(0),
|
|
table_mon_list->share->tgt_dbs[0],
|
|
table_mon_list->share->tgt_table_names[0]);
|
|
goto end_with_free_table_mon_list;
|
|
}
|
|
|
|
if (!pthread_mutex_trylock(&table_mon_list->caller_mutex))
|
|
{
|
|
table_mon = table_mon_list->current;
|
|
first_sid = table_mon->server_id;
|
|
current_mon_count = 1;
|
|
while (TRUE)
|
|
{
|
|
DBUG_PRINT("info",("spider thd->killed=%s",
|
|
thd ? (thd->killed ? "TRUE" : "FALSE") : "NULL"));
|
|
DBUG_PRINT("info",("spider abort_loop=%s",
|
|
*spd_abort_loop ? "TRUE" : "FALSE"));
|
|
if (
|
|
(thd && thd->killed) ||
|
|
*spd_abort_loop
|
|
) {
|
|
error_num = ER_SPIDER_COND_SKIP_NUM;
|
|
break;
|
|
} else {
|
|
if (!table_mon)
|
|
table_mon = table_mon_list->first;
|
|
if (
|
|
current_mon_count > table_mon_list->list_size ||
|
|
(current_mon_count > 1 && table_mon->server_id == first_sid)
|
|
) {
|
|
table_mon_list->last_caller_result = SPIDER_LINK_MON_DRAW_FEW_MON;
|
|
mon_table_result.result_status = SPIDER_LINK_MON_DRAW_FEW_MON;
|
|
DBUG_PRINT("info",(
|
|
"spider mon_table_result->result_status=SPIDER_LINK_MON_DRAW_FEW_MON 1"));
|
|
error_num = ER_SPIDER_LINK_MON_DRAW_FEW_MON_NUM;
|
|
my_printf_error(error_num,
|
|
ER_SPIDER_LINK_MON_DRAW_FEW_MON_STR, MYF(0),
|
|
table_mon_list->share->tgt_dbs[0],
|
|
table_mon_list->share->tgt_table_names[0]);
|
|
break;
|
|
}
|
|
int prev_error = 0;
|
|
char prev_error_msg[MYSQL_ERRMSG_SIZE];
|
|
if (thd->is_error())
|
|
{
|
|
prev_error = spider_stmt_da_sql_errno(thd);
|
|
strmov(prev_error_msg, spider_stmt_da_message(thd));
|
|
thd->clear_error();
|
|
}
|
|
if ((mon_conn = spider_get_ping_table_tgt_conn(trx,
|
|
table_mon->share, &error_num))
|
|
) {
|
|
if (!spider_db_udf_ping_table_mon_next(
|
|
thd, table_mon, mon_conn, &mon_table_result, conv_name,
|
|
conv_name_length, link_idx,
|
|
where_clause, where_clause_length, -1, table_mon_list->list_size,
|
|
0, 0, 0, flags, monitoring_limit))
|
|
{
|
|
if (
|
|
mon_table_result.result_status == SPIDER_LINK_MON_NG &&
|
|
table_mon_list->mon_status != SPIDER_LINK_MON_NG
|
|
) {
|
|
pthread_mutex_lock(&table_mon_list->update_status_mutex);
|
|
if (table_mon_list->mon_status != SPIDER_LINK_MON_NG)
|
|
{
|
|
table_mon_list->mon_status = SPIDER_LINK_MON_NG;
|
|
table_mon_list->share->link_statuses[0] = SPIDER_LINK_STATUS_NG;
|
|
DBUG_PRINT("info", (
|
|
"spider share->link_statuses[%d]=SPIDER_LINK_STATUS_NG",
|
|
link_idx));
|
|
share->link_statuses[link_idx] = SPIDER_LINK_STATUS_NG;
|
|
spider_sys_update_tables_link_status(thd, conv_name,
|
|
conv_name_length, link_idx, SPIDER_LINK_STATUS_NG, need_lock);
|
|
spider_sys_log_tables_link_failed(thd, conv_name,
|
|
conv_name_length, link_idx, need_lock);
|
|
}
|
|
pthread_mutex_unlock(&table_mon_list->update_status_mutex);
|
|
}
|
|
table_mon_list->last_caller_result = mon_table_result.result_status;
|
|
if (mon_table_result.result_status == SPIDER_LINK_MON_OK)
|
|
{
|
|
if (prev_error)
|
|
my_message(prev_error, prev_error_msg, MYF(0));
|
|
error_num = ER_SPIDER_LINK_MON_OK_NUM;
|
|
my_printf_error(error_num,
|
|
ER_SPIDER_LINK_MON_OK_STR, MYF(0),
|
|
table_mon_list->share->tgt_dbs[0],
|
|
table_mon_list->share->tgt_table_names[0]);
|
|
break;
|
|
}
|
|
if (mon_table_result.result_status == SPIDER_LINK_MON_NG)
|
|
{
|
|
error_num = ER_SPIDER_LINK_MON_NG_NUM;
|
|
my_printf_error(error_num,
|
|
ER_SPIDER_LINK_MON_NG_STR, MYF(0),
|
|
table_mon_list->share->tgt_dbs[0],
|
|
table_mon_list->share->tgt_table_names[0]);
|
|
break;
|
|
}
|
|
if (mon_table_result.result_status ==
|
|
SPIDER_LINK_MON_DRAW_FEW_MON)
|
|
{
|
|
error_num = ER_SPIDER_LINK_MON_DRAW_FEW_MON_NUM;
|
|
my_printf_error(error_num,
|
|
ER_SPIDER_LINK_MON_DRAW_FEW_MON_STR, MYF(0),
|
|
table_mon_list->share->tgt_dbs[0],
|
|
table_mon_list->share->tgt_table_names[0]);
|
|
break;
|
|
}
|
|
error_num = ER_SPIDER_LINK_MON_DRAW_NUM;
|
|
my_printf_error(error_num,
|
|
ER_SPIDER_LINK_MON_DRAW_STR, MYF(0),
|
|
table_mon_list->share->tgt_dbs[0],
|
|
table_mon_list->share->tgt_table_names[0]);
|
|
break;
|
|
}
|
|
}
|
|
table_mon = table_mon->next;
|
|
current_mon_count++;
|
|
}
|
|
}
|
|
pthread_mutex_unlock(&table_mon_list->caller_mutex);
|
|
} else {
|
|
pthread_mutex_lock(&table_mon_list->caller_mutex);
|
|
DBUG_PRINT("info",("spider thd->killed=%s",
|
|
thd ? (thd->killed ? "TRUE" : "FALSE") : "NULL"));
|
|
DBUG_PRINT("info",("spider abort_loop=%s",
|
|
*spd_abort_loop ? "TRUE" : "FALSE"));
|
|
if (
|
|
(thd && thd->killed) ||
|
|
*spd_abort_loop
|
|
) {
|
|
error_num = ER_SPIDER_COND_SKIP_NUM;
|
|
} else {
|
|
switch (table_mon_list->last_caller_result)
|
|
{
|
|
case SPIDER_LINK_MON_OK:
|
|
error_num = ER_SPIDER_LINK_MON_OK_NUM;
|
|
my_printf_error(error_num,
|
|
ER_SPIDER_LINK_MON_OK_STR, MYF(0),
|
|
table_mon_list->share->tgt_dbs[0],
|
|
table_mon_list->share->tgt_table_names[0]);
|
|
break;
|
|
case SPIDER_LINK_MON_NG:
|
|
error_num = ER_SPIDER_LINK_MON_NG_NUM;
|
|
my_printf_error(error_num,
|
|
ER_SPIDER_LINK_MON_NG_STR, MYF(0),
|
|
table_mon_list->share->tgt_dbs[0],
|
|
table_mon_list->share->tgt_table_names[0]);
|
|
break;
|
|
case SPIDER_LINK_MON_DRAW_FEW_MON:
|
|
error_num = ER_SPIDER_LINK_MON_DRAW_FEW_MON_NUM;
|
|
my_printf_error(error_num,
|
|
ER_SPIDER_LINK_MON_DRAW_FEW_MON_STR, MYF(0),
|
|
table_mon_list->share->tgt_dbs[0],
|
|
table_mon_list->share->tgt_table_names[0]);
|
|
break;
|
|
default:
|
|
error_num = ER_SPIDER_LINK_MON_DRAW_NUM;
|
|
my_printf_error(error_num,
|
|
ER_SPIDER_LINK_MON_DRAW_STR, MYF(0),
|
|
table_mon_list->share->tgt_dbs[0],
|
|
table_mon_list->share->tgt_table_names[0]);
|
|
break;
|
|
}
|
|
}
|
|
pthread_mutex_unlock(&table_mon_list->caller_mutex);
|
|
}
|
|
|
|
end_with_free_table_mon_list:
|
|
spider_free_ping_table_mon_list(table_mon_list);
|
|
end:
|
|
DBUG_RETURN(error_num);
|
|
}
|