Commit a1e41e32 authored by Jan Lindström's avatar Jan Lindström

MDEV-6470: Restrict number of error messages about persistent statictic tables not found

If mysql.innodb_table_stats or mysql.innodb_index_stats is not found or has
unexpected structure output that error only once and no other error for
every table trying to use them. If they do exists, then print fetch or
recalculation errors only once / table or index.
parent 15a529e1
...@@ -121,6 +121,11 @@ UNIV_INTERN mysql_pfs_key_t dict_foreign_err_mutex_key; ...@@ -121,6 +121,11 @@ UNIV_INTERN mysql_pfs_key_t dict_foreign_err_mutex_key;
/** Identifies generated InnoDB foreign key names */ /** Identifies generated InnoDB foreign key names */
static char dict_ibfk[] = "_ibfk_"; static char dict_ibfk[] = "_ibfk_";
bool innodb_table_stats_not_found = false;
bool innodb_index_stats_not_found = false;
static bool innodb_table_stats_not_found_reported = false;
static bool innodb_index_stats_not_found_reported = false;
/*******************************************************************//** /*******************************************************************//**
Tries to find column names for the index and sets the col field of the Tries to find column names for the index and sets the col field of the
index. index.
...@@ -5994,14 +5999,34 @@ dict_table_schema_check( ...@@ -5994,14 +5999,34 @@ dict_table_schema_check(
table = dict_table_get_low(req_schema->table_name); table = dict_table_get_low(req_schema->table_name);
if (table == NULL) { if (table == NULL) {
bool should_print=true;
/* no such table */ /* no such table */
ut_snprintf(errstr, errstr_sz, if (innobase_strcasecmp(req_schema->table_name, "mysql/innodb_table_stats") == 0) {
"Table %s not found.", if (innodb_table_stats_not_found_reported == false) {
ut_format_name(req_schema->table_name, innodb_table_stats_not_found = true;
TRUE, buf, sizeof(buf))); innodb_table_stats_not_found_reported = true;
} else {
should_print = false;
}
} else if (innobase_strcasecmp(req_schema->table_name, "mysql/innodb_index_stats") == 0 ) {
if (innodb_index_stats_not_found_reported == false) {
innodb_index_stats_not_found = true;
innodb_index_stats_not_found_reported = true;
} else {
should_print = false;
}
}
return(DB_TABLE_NOT_FOUND); if (should_print) {
ut_snprintf(errstr, errstr_sz,
"Table %s not found.",
ut_format_name(req_schema->table_name,
TRUE, buf, sizeof(buf)));
return(DB_TABLE_NOT_FOUND);
} else {
return(DB_STATS_DO_NOT_EXIST);
}
} }
if (table->ibd_file_missing) { if (table->ibd_file_missing) {
......
...@@ -268,10 +268,12 @@ dict_stats_persistent_storage_check( ...@@ -268,10 +268,12 @@ dict_stats_persistent_storage_check(
mutex_exit(&(dict_sys->mutex)); mutex_exit(&(dict_sys->mutex));
} }
if (ret != DB_SUCCESS) { if (ret != DB_SUCCESS && ret != DB_STATS_DO_NOT_EXIST) {
ut_print_timestamp(stderr); ut_print_timestamp(stderr);
fprintf(stderr, " InnoDB: Error: %s\n", errstr); fprintf(stderr, " InnoDB: Error: %s\n", errstr);
return(false); return(false);
} else if (ret == DB_STATS_DO_NOT_EXIST) {
return false;
} }
/* else */ /* else */
...@@ -2201,17 +2203,21 @@ dict_stats_save_index_stat( ...@@ -2201,17 +2203,21 @@ dict_stats_save_index_stat(
"END;", trx); "END;", trx);
if (ret != DB_SUCCESS) { if (ret != DB_SUCCESS) {
char buf_table[MAX_FULL_NAME_LEN]; if (innodb_index_stats_not_found == false &&
char buf_index[MAX_FULL_NAME_LEN]; index->stats_error_printed == false) {
ut_print_timestamp(stderr); char buf_table[MAX_FULL_NAME_LEN];
fprintf(stderr, char buf_index[MAX_FULL_NAME_LEN];
" InnoDB: Cannot save index statistics for table " ut_print_timestamp(stderr);
"%s, index %s, stat name \"%s\": %s\n", fprintf(stderr,
ut_format_name(index->table->name, TRUE, " InnoDB: Cannot save index statistics for table "
buf_table, sizeof(buf_table)), "%s, index %s, stat name \"%s\": %s\n",
ut_format_name(index->name, FALSE, ut_format_name(index->table->name, TRUE,
buf_index, sizeof(buf_index)), buf_table, sizeof(buf_table)),
stat_name, ut_strerr(ret)); ut_format_name(index->name, FALSE,
buf_index, sizeof(buf_index)),
stat_name, ut_strerr(ret));
index->stats_error_printed = true;
}
} }
return(ret); return(ret);
...@@ -2900,20 +2906,24 @@ dict_stats_update_for_index( ...@@ -2900,20 +2906,24 @@ dict_stats_update_for_index(
} }
/* else */ /* else */
/* Fall back to transient stats since the persistent if (innodb_index_stats_not_found == false &&
storage is not present or is corrupted */ index->stats_error_printed == false) {
char buf_table[MAX_FULL_NAME_LEN]; /* Fall back to transient stats since the persistent
char buf_index[MAX_FULL_NAME_LEN]; storage is not present or is corrupted */
ut_print_timestamp(stderr); char buf_table[MAX_FULL_NAME_LEN];
fprintf(stderr, char buf_index[MAX_FULL_NAME_LEN];
" InnoDB: Recalculation of persistent statistics " ut_print_timestamp(stderr);
"requested for table %s index %s but the required " fprintf(stderr,
"persistent statistics storage is not present or is " " InnoDB: Recalculation of persistent statistics "
"corrupted. Using transient stats instead.\n", "requested for table %s index %s but the required "
ut_format_name(index->table->name, TRUE, "persistent statistics storage is not present or is "
buf_table, sizeof(buf_table)), "corrupted. Using transient stats instead.\n",
ut_format_name(index->name, FALSE, ut_format_name(index->table->name, TRUE,
buf_index, sizeof(buf_index))); buf_table, sizeof(buf_table)),
ut_format_name(index->name, FALSE,
buf_index, sizeof(buf_index)));
index->stats_error_printed = false;
}
} }
dict_table_stats_lock(index->table, RW_X_LATCH); dict_table_stats_lock(index->table, RW_X_LATCH);
...@@ -2998,13 +3008,17 @@ dict_stats_update( ...@@ -2998,13 +3008,17 @@ dict_stats_update(
/* Fall back to transient stats since the persistent /* Fall back to transient stats since the persistent
storage is not present or is corrupted */ storage is not present or is corrupted */
ut_print_timestamp(stderr); if (innodb_table_stats_not_found == false &&
fprintf(stderr, table->stats_error_printed == false) {
" InnoDB: Recalculation of persistent statistics " ut_print_timestamp(stderr);
"requested for table %s but the required persistent " fprintf(stderr,
"statistics storage is not present or is corrupted. " " InnoDB: Recalculation of persistent statistics "
"Using transient stats instead.\n", "requested for table %s but the required persistent "
ut_format_name(table->name, TRUE, buf, sizeof(buf))); "statistics storage is not present or is corrupted. "
"Using transient stats instead.\n",
ut_format_name(table->name, TRUE, buf, sizeof(buf)));
table->stats_error_printed = true;
}
goto transient; goto transient;
...@@ -3048,17 +3062,21 @@ dict_stats_update( ...@@ -3048,17 +3062,21 @@ dict_stats_update(
/* persistent statistics storage does not exist /* persistent statistics storage does not exist
or is corrupted, calculate the transient stats */ or is corrupted, calculate the transient stats */
ut_print_timestamp(stderr); if (innodb_table_stats_not_found == false &&
fprintf(stderr, table->stats_error_printed == false) {
" InnoDB: Error: Fetch of persistent " ut_print_timestamp(stderr);
"statistics requested for table %s but the " fprintf(stderr,
"required system tables %s and %s are not " " InnoDB: Error: Fetch of persistent "
"present or have unexpected structure. " "statistics requested for table %s but the "
"Using transient stats instead.\n", "required system tables %s and %s are not "
ut_format_name(table->name, TRUE, "present or have unexpected structure. "
buf, sizeof(buf)), "Using transient stats instead.\n",
TABLE_STATS_NAME_PRINT, ut_format_name(table->name, TRUE,
INDEX_STATS_NAME_PRINT); buf, sizeof(buf)),
TABLE_STATS_NAME_PRINT,
INDEX_STATS_NAME_PRINT);
table->stats_error_printed = true;
}
goto transient; goto transient;
} }
...@@ -3128,16 +3146,19 @@ dict_stats_update( ...@@ -3128,16 +3146,19 @@ dict_stats_update(
dict_stats_table_clone_free(t); dict_stats_table_clone_free(t);
ut_print_timestamp(stderr); if (innodb_table_stats_not_found == false &&
fprintf(stderr, table->stats_error_printed == false) {
" InnoDB: Error fetching persistent statistics " ut_print_timestamp(stderr);
"for table %s from %s and %s: %s. " fprintf(stderr,
"Using transient stats method instead.\n", " InnoDB: Error fetching persistent statistics "
ut_format_name(table->name, TRUE, buf, "for table %s from %s and %s: %s. "
sizeof(buf)), "Using transient stats method instead.\n",
TABLE_STATS_NAME, ut_format_name(table->name, TRUE, buf,
INDEX_STATS_NAME, sizeof(buf)),
ut_strerr(err)); TABLE_STATS_NAME,
INDEX_STATS_NAME,
ut_strerr(err));
}
goto transient; goto transient;
} }
......
...@@ -43,6 +43,9 @@ Created 1/8/1996 Heikki Tuuri ...@@ -43,6 +43,9 @@ Created 1/8/1996 Heikki Tuuri
#include "trx0types.h" #include "trx0types.h"
#include "row0types.h" #include "row0types.h"
extern bool innodb_table_stats_not_found;
extern bool innodb_index_stats_not_found;
#ifndef UNIV_HOTBACKUP #ifndef UNIV_HOTBACKUP
# include "sync0sync.h" # include "sync0sync.h"
# include "sync0rw.h" # include "sync0rw.h"
......
...@@ -624,6 +624,9 @@ struct dict_index_t{ ...@@ -624,6 +624,9 @@ struct dict_index_t{
ulint stat_n_leaf_pages; ulint stat_n_leaf_pages;
/*!< approximate number of leaf pages in the /*!< approximate number of leaf pages in the
index tree */ index tree */
bool stats_error_printed;
/*!< has persistent statistics error printed
for this index ? */
/* @} */ /* @} */
rw_lock_t lock; /*!< read-write lock protecting the rw_lock_t lock; /*!< read-write lock protecting the
upper levels of the index tree */ upper levels of the index tree */
...@@ -947,6 +950,9 @@ struct dict_table_t{ ...@@ -947,6 +950,9 @@ struct dict_table_t{
/*!< see BG_STAT_* above. /*!< see BG_STAT_* above.
Writes are covered by dict_sys->mutex. Writes are covered by dict_sys->mutex.
Dirty reads are possible. */ Dirty reads are possible. */
bool stats_error_printed;
/*!< Has persistent stats error beein
already printed for this table ? */
/* @} */ /* @} */
/*----------------------*/ /*----------------------*/
/**!< The following fields are used by the /**!< The following fields are used by the
......
...@@ -121,6 +121,11 @@ UNIV_INTERN mysql_pfs_key_t dict_foreign_err_mutex_key; ...@@ -121,6 +121,11 @@ UNIV_INTERN mysql_pfs_key_t dict_foreign_err_mutex_key;
/** Identifies generated InnoDB foreign key names */ /** Identifies generated InnoDB foreign key names */
static char dict_ibfk[] = "_ibfk_"; static char dict_ibfk[] = "_ibfk_";
bool innodb_table_stats_not_found = false;
bool innodb_index_stats_not_found = false;
static bool innodb_table_stats_not_found_reported = false;
static bool innodb_index_stats_not_found_reported = false;
/*******************************************************************//** /*******************************************************************//**
Tries to find column names for the index and sets the col field of the Tries to find column names for the index and sets the col field of the
index. index.
...@@ -6037,14 +6042,34 @@ dict_table_schema_check( ...@@ -6037,14 +6042,34 @@ dict_table_schema_check(
table = dict_table_get_low(req_schema->table_name); table = dict_table_get_low(req_schema->table_name);
if (table == NULL) { if (table == NULL) {
bool should_print=true;
/* no such table */ /* no such table */
ut_snprintf(errstr, errstr_sz, if (innobase_strcasecmp(req_schema->table_name, "mysql/innodb_table_stats") == 0) {
"Table %s not found.", if (innodb_table_stats_not_found_reported == false) {
ut_format_name(req_schema->table_name, innodb_table_stats_not_found = true;
TRUE, buf, sizeof(buf))); innodb_table_stats_not_found_reported = true;
} else {
should_print = false;
}
} else if (innobase_strcasecmp(req_schema->table_name, "mysql/innodb_index_stats") == 0 ) {
if (innodb_index_stats_not_found_reported == false) {
innodb_index_stats_not_found = true;
innodb_index_stats_not_found_reported = true;
} else {
should_print = false;
}
}
return(DB_TABLE_NOT_FOUND); if (should_print) {
ut_snprintf(errstr, errstr_sz,
"Table %s not found.",
ut_format_name(req_schema->table_name,
TRUE, buf, sizeof(buf)));
return(DB_TABLE_NOT_FOUND);
} else {
return(DB_STATS_DO_NOT_EXIST);
}
} }
if (table->ibd_file_missing) { if (table->ibd_file_missing) {
......
...@@ -268,10 +268,12 @@ dict_stats_persistent_storage_check( ...@@ -268,10 +268,12 @@ dict_stats_persistent_storage_check(
mutex_exit(&(dict_sys->mutex)); mutex_exit(&(dict_sys->mutex));
} }
if (ret != DB_SUCCESS) { if (ret != DB_SUCCESS && ret != DB_STATS_DO_NOT_EXIST) {
ut_print_timestamp(stderr); ut_print_timestamp(stderr);
fprintf(stderr, " InnoDB: Error: %s\n", errstr); fprintf(stderr, " InnoDB: Error: %s\n", errstr);
return(false); return(false);
} else if (ret == DB_STATS_DO_NOT_EXIST) {
return false;
} }
/* else */ /* else */
...@@ -2201,17 +2203,21 @@ dict_stats_save_index_stat( ...@@ -2201,17 +2203,21 @@ dict_stats_save_index_stat(
"END;", trx); "END;", trx);
if (ret != DB_SUCCESS) { if (ret != DB_SUCCESS) {
char buf_table[MAX_FULL_NAME_LEN]; if (innodb_index_stats_not_found == false &&
char buf_index[MAX_FULL_NAME_LEN]; index->stats_error_printed == false) {
ut_print_timestamp(stderr); char buf_table[MAX_FULL_NAME_LEN];
fprintf(stderr, char buf_index[MAX_FULL_NAME_LEN];
" InnoDB: Cannot save index statistics for table " ut_print_timestamp(stderr);
"%s, index %s, stat name \"%s\": %s\n", fprintf(stderr,
ut_format_name(index->table->name, TRUE, " InnoDB: Cannot save index statistics for table "
buf_table, sizeof(buf_table)), "%s, index %s, stat name \"%s\": %s\n",
ut_format_name(index->name, FALSE, ut_format_name(index->table->name, TRUE,
buf_index, sizeof(buf_index)), buf_table, sizeof(buf_table)),
stat_name, ut_strerr(ret)); ut_format_name(index->name, FALSE,
buf_index, sizeof(buf_index)),
stat_name, ut_strerr(ret));
index->stats_error_printed = true;
}
} }
return(ret); return(ret);
...@@ -2900,20 +2906,24 @@ dict_stats_update_for_index( ...@@ -2900,20 +2906,24 @@ dict_stats_update_for_index(
} }
/* else */ /* else */
/* Fall back to transient stats since the persistent if (innodb_index_stats_not_found == false &&
storage is not present or is corrupted */ index->stats_error_printed == false) {
char buf_table[MAX_FULL_NAME_LEN]; /* Fall back to transient stats since the persistent
char buf_index[MAX_FULL_NAME_LEN]; storage is not present or is corrupted */
ut_print_timestamp(stderr); char buf_table[MAX_FULL_NAME_LEN];
fprintf(stderr, char buf_index[MAX_FULL_NAME_LEN];
" InnoDB: Recalculation of persistent statistics " ut_print_timestamp(stderr);
"requested for table %s index %s but the required " fprintf(stderr,
"persistent statistics storage is not present or is " " InnoDB: Recalculation of persistent statistics "
"corrupted. Using transient stats instead.\n", "requested for table %s index %s but the required "
ut_format_name(index->table->name, TRUE, "persistent statistics storage is not present or is "
buf_table, sizeof(buf_table)), "corrupted. Using transient stats instead.\n",
ut_format_name(index->name, FALSE, ut_format_name(index->table->name, TRUE,
buf_index, sizeof(buf_index))); buf_table, sizeof(buf_table)),
ut_format_name(index->name, FALSE,
buf_index, sizeof(buf_index)));
index->stats_error_printed = false;
}
} }
dict_table_stats_lock(index->table, RW_X_LATCH); dict_table_stats_lock(index->table, RW_X_LATCH);
...@@ -2998,13 +3008,17 @@ dict_stats_update( ...@@ -2998,13 +3008,17 @@ dict_stats_update(
/* Fall back to transient stats since the persistent /* Fall back to transient stats since the persistent
storage is not present or is corrupted */ storage is not present or is corrupted */
ut_print_timestamp(stderr); if (innodb_table_stats_not_found == false &&
fprintf(stderr, table->stats_error_printed == false) {
" InnoDB: Recalculation of persistent statistics " ut_print_timestamp(stderr);
"requested for table %s but the required persistent " fprintf(stderr,
"statistics storage is not present or is corrupted. " " InnoDB: Recalculation of persistent statistics "
"Using transient stats instead.\n", "requested for table %s but the required persistent "
ut_format_name(table->name, TRUE, buf, sizeof(buf))); "statistics storage is not present or is corrupted. "
"Using transient stats instead.\n",
ut_format_name(table->name, TRUE, buf, sizeof(buf)));
table->stats_error_printed = true;
}
goto transient; goto transient;
...@@ -3048,17 +3062,21 @@ dict_stats_update( ...@@ -3048,17 +3062,21 @@ dict_stats_update(
/* persistent statistics storage does not exist /* persistent statistics storage does not exist
or is corrupted, calculate the transient stats */ or is corrupted, calculate the transient stats */
ut_print_timestamp(stderr); if (innodb_table_stats_not_found == false &&
fprintf(stderr, table->stats_error_printed == false) {
" InnoDB: Error: Fetch of persistent " ut_print_timestamp(stderr);
"statistics requested for table %s but the " fprintf(stderr,
"required system tables %s and %s are not " " InnoDB: Error: Fetch of persistent "
"present or have unexpected structure. " "statistics requested for table %s but the "
"Using transient stats instead.\n", "required system tables %s and %s are not "
ut_format_name(table->name, TRUE, "present or have unexpected structure. "
buf, sizeof(buf)), "Using transient stats instead.\n",
TABLE_STATS_NAME_PRINT, ut_format_name(table->name, TRUE,
INDEX_STATS_NAME_PRINT); buf, sizeof(buf)),
TABLE_STATS_NAME_PRINT,
INDEX_STATS_NAME_PRINT);
table->stats_error_printed = true;
}
goto transient; goto transient;
} }
...@@ -3128,16 +3146,19 @@ dict_stats_update( ...@@ -3128,16 +3146,19 @@ dict_stats_update(
dict_stats_table_clone_free(t); dict_stats_table_clone_free(t);
ut_print_timestamp(stderr); if (innodb_table_stats_not_found == false &&
fprintf(stderr, table->stats_error_printed == false) {
" InnoDB: Error fetching persistent statistics " ut_print_timestamp(stderr);
"for table %s from %s and %s: %s. " fprintf(stderr,
"Using transient stats method instead.\n", " InnoDB: Error fetching persistent statistics "
ut_format_name(table->name, TRUE, buf, "for table %s from %s and %s: %s. "
sizeof(buf)), "Using transient stats method instead.\n",
TABLE_STATS_NAME, ut_format_name(table->name, TRUE, buf,
INDEX_STATS_NAME, sizeof(buf)),
ut_strerr(err)); TABLE_STATS_NAME,
INDEX_STATS_NAME,
ut_strerr(err));
}
goto transient; goto transient;
} }
......
...@@ -43,6 +43,9 @@ Created 1/8/1996 Heikki Tuuri ...@@ -43,6 +43,9 @@ Created 1/8/1996 Heikki Tuuri
#include "trx0types.h" #include "trx0types.h"
#include "row0types.h" #include "row0types.h"
extern bool innodb_table_stats_not_found;
extern bool innodb_index_stats_not_found;
#ifndef UNIV_HOTBACKUP #ifndef UNIV_HOTBACKUP
# include "sync0sync.h" # include "sync0sync.h"
# include "sync0rw.h" # include "sync0rw.h"
......
...@@ -638,6 +638,9 @@ struct dict_index_t{ ...@@ -638,6 +638,9 @@ struct dict_index_t{
ulint stat_n_leaf_pages; ulint stat_n_leaf_pages;
/*!< approximate number of leaf pages in the /*!< approximate number of leaf pages in the
index tree */ index tree */
bool stats_error_printed;
/*!< has persistent statistics error printed
for this index ? */
/* @} */ /* @} */
prio_rw_lock_t lock; /*!< read-write lock protecting the prio_rw_lock_t lock; /*!< read-write lock protecting the
upper levels of the index tree */ upper levels of the index tree */
...@@ -962,6 +965,9 @@ struct dict_table_t{ ...@@ -962,6 +965,9 @@ struct dict_table_t{
/*!< see BG_STAT_* above. /*!< see BG_STAT_* above.
Writes are covered by dict_sys->mutex. Writes are covered by dict_sys->mutex.
Dirty reads are possible. */ Dirty reads are possible. */
bool stats_error_printed;
/*!< Has persistent stats error beein
already printed for this table ? */
/* @} */ /* @} */
/*----------------------*/ /*----------------------*/
/**!< The following fields are used by the /**!< The following fields are used by the
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment