mirror of
https://github.com/MariaDB/server.git
synced 2025-08-01 03:47:19 +03:00
merge feedback plugin
This commit is contained in:
11
plugin/feedback/CMakeLists.txt
Normal file
11
plugin/feedback/CMakeLists.txt
Normal file
@ -0,0 +1,11 @@
|
||||
INCLUDE("${PROJECT_SOURCE_DIR}/storage/mysql_storage_engine.cmake")
|
||||
|
||||
INCLUDE_DIRECTORIES(${CMAKE_SOURCE_DIR}/sql ${CMAKE_SOURCE_DIR}/regex
|
||||
${CMAKE_SOURCE_DIR}/extra/yassl/include)
|
||||
|
||||
SET(FEEDBACK_SOURCES feedback.cc sender_thread.cc
|
||||
url_base.cc url_http.cc utils.cc)
|
||||
|
||||
SET(FEEDBACK_LIBS Ws2_32)
|
||||
|
||||
MYSQL_PLUGIN(FEEDBACK)
|
18
plugin/feedback/Makefile.am
Normal file
18
plugin/feedback/Makefile.am
Normal file
@ -0,0 +1,18 @@
|
||||
pkgplugindir = $(pkglibdir)/plugin
|
||||
INCLUDES = -I$(top_srcdir)/include -I$(top_builddir)/include \
|
||||
-I$(top_srcdir)/regex -I$(top_srcdir)/sql
|
||||
|
||||
EXTRA_LTLIBRARIES = feedback.la libfeedback.la
|
||||
pkgplugin_LTLIBRARIES = @plugin_feedback_shared_target@
|
||||
feedback_la_LDFLAGS = -module -rpath $(pkgplugindir)
|
||||
feedback_la_CXXFLAGS = -shared -DMYSQL_DYNAMIC_PLUGIN
|
||||
feedback_la_SOURCES = feedback.cc utils.cc url_base.cc url_http.cc \
|
||||
sender_thread.cc
|
||||
|
||||
noinst_LTLIBRARIES = @plugin_feedback_static_target@
|
||||
libfeedback_la_SOURCES= feedback.cc utils.cc url_base.cc url_http.cc \
|
||||
sender_thread.cc
|
||||
|
||||
noinst_HEADERS = feedback.h
|
||||
EXTRA_DIST = CMakeLists.txt plug.in
|
||||
|
355
plugin/feedback/feedback.cc
Normal file
355
plugin/feedback/feedback.cc
Normal file
@ -0,0 +1,355 @@
|
||||
/* Copyright (C) 2010 Sergei Golubchik and Monty Program Ab
|
||||
|
||||
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 */
|
||||
|
||||
#include "feedback.h"
|
||||
|
||||
/* MySQL functions/variables not declared in mysql_priv.h */
|
||||
int fill_variables(THD *thd, TABLE_LIST *tables, COND *cond);
|
||||
int fill_status(THD *thd, TABLE_LIST *tables, COND *cond);
|
||||
extern ST_SCHEMA_TABLE schema_tables[];
|
||||
|
||||
namespace feedback {
|
||||
|
||||
char server_uid_buf[SERVER_UID_SIZE+1]; ///< server uid will be written here
|
||||
|
||||
/* backing store for system variables */
|
||||
static char *server_uid= server_uid_buf, *url;
|
||||
char *user_info;
|
||||
ulong send_timeout, send_retry_wait;
|
||||
|
||||
/**
|
||||
these three are used to communicate the shutdown signal to the
|
||||
background thread
|
||||
*/
|
||||
pthread_mutex_t sleep_mutex;
|
||||
pthread_cond_t sleep_condition;
|
||||
volatile bool shutdown_plugin;
|
||||
static pthread_t sender_thread;
|
||||
|
||||
Url **urls; ///< list of urls to send the report to
|
||||
uint url_count;
|
||||
|
||||
ST_SCHEMA_TABLE *i_s_feedback; ///< table descriptor for our I_S table
|
||||
|
||||
/*
|
||||
the column names *must* match column names in GLOBAL_VARIABLES and
|
||||
GLOBAL_STATUS tables otherwise condition pushdown below will not work
|
||||
*/
|
||||
static ST_FIELD_INFO feedback_fields[] =
|
||||
{
|
||||
{"VARIABLE_NAME", 255, MYSQL_TYPE_STRING, 0, 0, 0, 0},
|
||||
{"VARIABLE_VALUE", 1024, MYSQL_TYPE_STRING, 0, 0, 0, 0},
|
||||
{0, 0, MYSQL_TYPE_NULL, 0, 0, 0, 0}
|
||||
};
|
||||
|
||||
static COND * const OOM= (COND*)1;
|
||||
|
||||
/**
|
||||
Generate the COND tree for the condition pushdown
|
||||
|
||||
This function takes a list of strings and generates an Item tree
|
||||
corresponding to the following expression:
|
||||
|
||||
field LIKE str1 OR field LIKE str2 OR field LIKE str3 OR ...
|
||||
|
||||
where 'field' is the first field in the table - VARIABLE_NAME field -
|
||||
and str1, str2... are strings from the list.
|
||||
|
||||
This condition is used to filter the selected rows, emulating
|
||||
|
||||
SELECT * FROM INFORMATION_SCHEMA.GLOBAL_VARIABLES WHERE ...
|
||||
*/
|
||||
static COND* make_cond(THD *thd, TABLE_LIST *tables, LEX_STRING *filter)
|
||||
{
|
||||
Item_cond_or *res= NULL;
|
||||
Name_resolution_context nrc;
|
||||
const char *db= tables->db, *table= tables->alias,
|
||||
*field= tables->table->field[0]->field_name;
|
||||
CHARSET_INFO *cs= &my_charset_latin1;
|
||||
|
||||
if (!filter->str)
|
||||
return 0;
|
||||
|
||||
nrc.init();
|
||||
nrc.resolve_in_table_list_only(tables);
|
||||
|
||||
res= new Item_cond_or();
|
||||
if (!res)
|
||||
return OOM;
|
||||
|
||||
for (; filter->str; filter++)
|
||||
{
|
||||
Item_field *fld= new Item_field(&nrc, db, table, field);
|
||||
Item_string *pattern= new Item_string(filter->str, filter->length, cs);
|
||||
Item_string *escape= new Item_string("\\", 1, cs);
|
||||
|
||||
if (!fld || !pattern || !escape)
|
||||
return OOM;
|
||||
|
||||
Item_func_like *like= new Item_func_like(fld, pattern, escape, 0);
|
||||
|
||||
if (!like)
|
||||
return OOM;
|
||||
|
||||
res->add(like);
|
||||
}
|
||||
|
||||
if (res->fix_fields(thd, (Item**)&res))
|
||||
return OOM;
|
||||
|
||||
return res;
|
||||
}
|
||||
|
||||
/**
|
||||
System variables that we want to see in the feedback report
|
||||
*/
|
||||
static LEX_STRING vars_filter[]= {
|
||||
{C_STRING_WITH_LEN("auto\\_increment%")},
|
||||
{C_STRING_WITH_LEN("binlog\\_format")},
|
||||
{C_STRING_WITH_LEN("character\\_set\\_%")},
|
||||
{C_STRING_WITH_LEN("collation%")},
|
||||
{C_STRING_WITH_LEN("engine\\_condition\\_pushdown")},
|
||||
{C_STRING_WITH_LEN("event\\_scheduler")},
|
||||
{C_STRING_WITH_LEN("feedback\\_%")},
|
||||
{C_STRING_WITH_LEN("ft\\_m%")},
|
||||
{C_STRING_WITH_LEN("have\\_%")},
|
||||
{C_STRING_WITH_LEN("%\\_size")},
|
||||
{C_STRING_WITH_LEN("%\\_length%")},
|
||||
{C_STRING_WITH_LEN("%\\_timeout")},
|
||||
{C_STRING_WITH_LEN("large\\_%")},
|
||||
{C_STRING_WITH_LEN("lc_time_names")},
|
||||
{C_STRING_WITH_LEN("log")},
|
||||
{C_STRING_WITH_LEN("log_bin")},
|
||||
{C_STRING_WITH_LEN("log_output")},
|
||||
{C_STRING_WITH_LEN("log_slow_queries")},
|
||||
{C_STRING_WITH_LEN("log_slow_time")},
|
||||
{C_STRING_WITH_LEN("lower_case%")},
|
||||
{C_STRING_WITH_LEN("max_allowed_packet")},
|
||||
{C_STRING_WITH_LEN("max_connections")},
|
||||
{C_STRING_WITH_LEN("max_prepared_stmt_count")},
|
||||
{C_STRING_WITH_LEN("max_sp_recursion_depth")},
|
||||
{C_STRING_WITH_LEN("max_user_connections")},
|
||||
{C_STRING_WITH_LEN("max_write_lock_count")},
|
||||
{C_STRING_WITH_LEN("myisam_recover_options")},
|
||||
{C_STRING_WITH_LEN("myisam_repair_threads")},
|
||||
{C_STRING_WITH_LEN("myisam_stats_method")},
|
||||
{C_STRING_WITH_LEN("myisam_use_mmap")},
|
||||
{C_STRING_WITH_LEN("net\\_%")},
|
||||
{C_STRING_WITH_LEN("new")},
|
||||
{C_STRING_WITH_LEN("old%")},
|
||||
{C_STRING_WITH_LEN("optimizer%")},
|
||||
{C_STRING_WITH_LEN("profiling")},
|
||||
{C_STRING_WITH_LEN("query_cache%")},
|
||||
{C_STRING_WITH_LEN("secure_auth")},
|
||||
{C_STRING_WITH_LEN("slow_launch_time")},
|
||||
{C_STRING_WITH_LEN("sql%")},
|
||||
{C_STRING_WITH_LEN("storage_engine")},
|
||||
{C_STRING_WITH_LEN("sync_binlog")},
|
||||
{C_STRING_WITH_LEN("table_definition_cache")},
|
||||
{C_STRING_WITH_LEN("table_open_cache")},
|
||||
{C_STRING_WITH_LEN("thread_handling")},
|
||||
{C_STRING_WITH_LEN("time_zone")},
|
||||
{C_STRING_WITH_LEN("timed_mutexes")},
|
||||
{C_STRING_WITH_LEN("version%")},
|
||||
{0, 0}
|
||||
};
|
||||
|
||||
/**
|
||||
Status variables that we want to see in the feedback report
|
||||
|
||||
(empty list = no WHERE condition)
|
||||
*/
|
||||
static LEX_STRING status_filter[]= {{0, 0}};
|
||||
|
||||
/**
|
||||
Fill our I_S table with data
|
||||
|
||||
This function works by invoking fill_variables() and
|
||||
fill_status() of the corresponding I_S tables - to have
|
||||
their data UNION-ed in the same target table.
|
||||
After that it invokes our own fill_* functions
|
||||
from the utils.cc - to get the data that aren't available in the
|
||||
I_S.GLOBAL_VARIABLES and I_S.GLOBAL_STATUS.
|
||||
*/
|
||||
int fill_feedback(THD *thd, TABLE_LIST *tables, COND *unused)
|
||||
{
|
||||
int res;
|
||||
COND *cond;
|
||||
|
||||
tables->schema_table= schema_tables + SCH_GLOBAL_VARIABLES;
|
||||
cond= make_cond(thd, tables, vars_filter);
|
||||
res= (cond == OOM) ? 1 : fill_variables(thd, tables, cond);
|
||||
|
||||
tables->schema_table= schema_tables + SCH_GLOBAL_STATUS;
|
||||
if (!res)
|
||||
{
|
||||
cond= make_cond(thd, tables, status_filter);
|
||||
res= (cond == OOM) ? 1 : fill_status(thd, tables, cond);
|
||||
}
|
||||
|
||||
tables->schema_table= i_s_feedback;
|
||||
res= res || fill_plugin_version(thd, tables)
|
||||
|| fill_misc_data(thd, tables)
|
||||
|| fill_linux_info(thd, tables);
|
||||
|
||||
return res;
|
||||
}
|
||||
|
||||
/**
|
||||
plugin initialization function
|
||||
*/
|
||||
static int init(void *p)
|
||||
{
|
||||
i_s_feedback= (ST_SCHEMA_TABLE*) p;
|
||||
/* initialize the I_S descriptor structure */
|
||||
i_s_feedback->fields_info= feedback_fields; ///< field descriptor
|
||||
i_s_feedback->fill_table= fill_feedback; ///< how to fill the I_S table
|
||||
i_s_feedback->idx_field1 = 0; ///< virtual index on the 1st col
|
||||
|
||||
if (calculate_server_uid(server_uid_buf))
|
||||
return 1;
|
||||
|
||||
prepare_linux_info();
|
||||
|
||||
url_count= 0;
|
||||
if (*url)
|
||||
{
|
||||
// now we split url on spaces and store them in Url objects
|
||||
int slot;
|
||||
char *s, *e;
|
||||
|
||||
for (s= url, url_count= 1; *s; s++)
|
||||
if (*s == ' ')
|
||||
url_count++;
|
||||
|
||||
urls= (Url **)my_malloc(url_count*sizeof(Url*), MYF(MY_WME));
|
||||
if (!urls)
|
||||
return 1;
|
||||
|
||||
for (s= url, e = url+1, slot= 0; e[-1]; e++)
|
||||
if (*e == 0 || *e == ' ')
|
||||
{
|
||||
if (e > s && (urls[slot]= Url::create(s, e - s)))
|
||||
slot++;
|
||||
else
|
||||
{
|
||||
if (e > s)
|
||||
sql_print_error("feedback plugin: invalid url '%.*s'", (int)(e-s), s);
|
||||
url_count--;
|
||||
}
|
||||
s= e + 1;
|
||||
}
|
||||
|
||||
// create a background thread to handle urls, if any
|
||||
if (url_count)
|
||||
{
|
||||
pthread_mutex_init(&sleep_mutex, 0);
|
||||
pthread_cond_init(&sleep_condition, 0);
|
||||
shutdown_plugin= false;
|
||||
|
||||
pthread_attr_t attr;
|
||||
pthread_attr_init(&attr);
|
||||
pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_JOINABLE);
|
||||
if (pthread_create(&sender_thread, &attr, background_thread, 0) != 0)
|
||||
{
|
||||
sql_print_error("feedback plugin: failed to start a background thread");
|
||||
return 1;
|
||||
}
|
||||
}
|
||||
else
|
||||
my_free(urls, MYF(0));
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
plugin deinitialization function
|
||||
*/
|
||||
static int free(void *p)
|
||||
{
|
||||
if (url_count)
|
||||
{
|
||||
pthread_mutex_lock(&sleep_mutex);
|
||||
shutdown_plugin= true;
|
||||
pthread_cond_signal(&sleep_condition);
|
||||
pthread_mutex_unlock(&sleep_mutex);
|
||||
pthread_join(sender_thread, NULL);
|
||||
|
||||
pthread_mutex_destroy(&sleep_mutex);
|
||||
pthread_cond_destroy(&sleep_condition);
|
||||
|
||||
for (uint i= 0; i < url_count; i++)
|
||||
delete urls[i];
|
||||
my_free(urls, MYF(0));
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
#ifdef HAVE_OPENSSL
|
||||
#define DEFAULT_PROTO "https://"
|
||||
#else
|
||||
#define DEFAULT_PROTO "http://"
|
||||
#endif
|
||||
|
||||
static MYSQL_SYSVAR_STR(server_uid, server_uid,
|
||||
PLUGIN_VAR_READONLY | PLUGIN_VAR_NOCMDOPT,
|
||||
"Automatically calculated server unique id hash.", NULL, NULL, 0);
|
||||
static MYSQL_SYSVAR_STR(user_info, user_info,
|
||||
PLUGIN_VAR_READONLY | PLUGIN_VAR_RQCMDARG,
|
||||
"User specified string that will be included in the feedback report.",
|
||||
NULL, NULL, "");
|
||||
static MYSQL_SYSVAR_STR(url, url, PLUGIN_VAR_READONLY | PLUGIN_VAR_RQCMDARG,
|
||||
"Space separated URLs to send the feedback report to.", NULL, NULL,
|
||||
DEFAULT_PROTO "mariadb.org/feedback_plugin/post");
|
||||
static MYSQL_SYSVAR_ULONG(send_timeout, send_timeout, PLUGIN_VAR_RQCMDARG,
|
||||
"Timeout (in seconds) for the sending the report.",
|
||||
NULL, NULL, 60, 1, 60*60*24, 1);
|
||||
static MYSQL_SYSVAR_ULONG(send_retry_wait, send_retry_wait, PLUGIN_VAR_RQCMDARG,
|
||||
"Wait this many seconds before retrying a failed send.",
|
||||
NULL, NULL, 60, 1, 60*60*24, 1);
|
||||
|
||||
static struct st_mysql_sys_var* settings[] = {
|
||||
MYSQL_SYSVAR(server_uid),
|
||||
MYSQL_SYSVAR(user_info),
|
||||
MYSQL_SYSVAR(url),
|
||||
MYSQL_SYSVAR(send_timeout),
|
||||
MYSQL_SYSVAR(send_retry_wait),
|
||||
NULL
|
||||
};
|
||||
|
||||
|
||||
static struct st_mysql_information_schema feedback =
|
||||
{ MYSQL_INFORMATION_SCHEMA_INTERFACE_VERSION };
|
||||
|
||||
} // namespace feedback
|
||||
|
||||
mysql_declare_plugin(feedback)
|
||||
{
|
||||
MYSQL_INFORMATION_SCHEMA_PLUGIN,
|
||||
&feedback::feedback,
|
||||
"FEEDBACK",
|
||||
"Sergei Golubchik",
|
||||
"MariaDB User Feedback Plugin",
|
||||
PLUGIN_LICENSE_GPL,
|
||||
feedback::init,
|
||||
feedback::free,
|
||||
0x0100,
|
||||
NULL,
|
||||
feedback::settings,
|
||||
NULL
|
||||
}
|
||||
mysql_declare_plugin_end;
|
||||
|
67
plugin/feedback/feedback.h
Normal file
67
plugin/feedback/feedback.h
Normal file
@ -0,0 +1,67 @@
|
||||
/* Copyright (C) 2010 Sergei Golubchik and Monty Program Ab
|
||||
|
||||
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
|
||||
#include <mysql_priv.h>
|
||||
|
||||
namespace feedback {
|
||||
|
||||
int fill_feedback(THD *thd, TABLE_LIST *tables, COND *cond);
|
||||
int fill_plugin_version(THD *thd, TABLE_LIST *tables);
|
||||
int fill_misc_data(THD *thd, TABLE_LIST *tables);
|
||||
int fill_linux_info(THD *thd, TABLE_LIST *tables);
|
||||
|
||||
static const int SERVER_UID_SIZE= 29;
|
||||
extern char server_uid_buf[SERVER_UID_SIZE+1], *user_info;
|
||||
int calculate_server_uid(char *);
|
||||
int prepare_linux_info();
|
||||
|
||||
extern ST_SCHEMA_TABLE *i_s_feedback;
|
||||
|
||||
extern ulong send_timeout, send_retry_wait;
|
||||
|
||||
pthread_handler_t background_thread(void *arg);
|
||||
|
||||
/**
|
||||
The class for storing urls to send report data to.
|
||||
|
||||
Constructors are private, the object should be created with create() method.
|
||||
send() method does the actual sending.
|
||||
*/
|
||||
class Url {
|
||||
protected:
|
||||
Url(LEX_STRING &url_arg) : full_url(url_arg) {}
|
||||
const LEX_STRING full_url;
|
||||
|
||||
public:
|
||||
virtual ~Url() { my_free(full_url.str, MYF(0)); }
|
||||
|
||||
const char *url() { return full_url.str; }
|
||||
size_t url_length() { return full_url.length; }
|
||||
virtual int send(const char* data, size_t data_length) = 0;
|
||||
|
||||
static Url* create(const char *url, size_t url_length);
|
||||
};
|
||||
|
||||
extern Url **urls;
|
||||
extern uint url_count;
|
||||
|
||||
/* these are used to communicate with the background thread */
|
||||
extern pthread_mutex_t sleep_mutex;
|
||||
extern pthread_cond_t sleep_condition;
|
||||
extern volatile bool shutdown_plugin;
|
||||
|
||||
} // namespace feedback
|
||||
|
25
plugin/feedback/plug.in
Normal file
25
plugin/feedback/plug.in
Normal file
@ -0,0 +1,25 @@
|
||||
MYSQL_PLUGIN(feedback,[MariaDB User Feedback Plugin],
|
||||
[MariaDB User Feedback Plugin])
|
||||
|
||||
dnl Although it's not exactly obvious, top-level CMakeLists.txt parses plug.in
|
||||
dnl files, in particular looking for what the library name should be. It uses
|
||||
dnl regexp that matches MYSQL_PLUGIN_DYNAMIC or MYSQL_PLUGIN_STATIC, followed
|
||||
dnl by an open parenthesys, and the plugin name. Having engine name enclosed in
|
||||
dnl square brackets below causes this regexp to fail and as a result feedback
|
||||
dnl plugin will not be considered for dynamic builds on Windows.
|
||||
dnl Unfortunately, feedback cannot be built dynamically on Windows, because it
|
||||
dnl needs to access server internals that aren't designed for plugin use and
|
||||
dnl aren't marked with MYSQL_PLUGIN_IMPORT.
|
||||
MYSQL_PLUGIN_DYNAMIC([feedback], [feedback.la])
|
||||
ifelse(index(AC_PACKAGE_NAME, [MariaDB]), -1, [], [
|
||||
|
||||
dnl MariaDB and MySQL define static plugins differently.
|
||||
dnl I only support MariaDB here, for now.
|
||||
MYSQL_PLUGIN_STATIC(feedback, [libfeedback.la])
|
||||
|
||||
])
|
||||
|
||||
MYSQL_PLUGIN_ACTIONS(feedback, [
|
||||
AC_CHECK_HEADERS([netdb.h sys/utsname.h])
|
||||
])
|
||||
|
301
plugin/feedback/sender_thread.cc
Normal file
301
plugin/feedback/sender_thread.cc
Normal file
@ -0,0 +1,301 @@
|
||||
/* Copyright (C) 2010 Sergei Golubchik and Monty Program Ab
|
||||
|
||||
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 */
|
||||
|
||||
#include "feedback.h"
|
||||
#include <time.h>
|
||||
|
||||
namespace feedback {
|
||||
|
||||
static THD *thd= 0; ///< background thread thd
|
||||
static my_thread_id thd_thread_id; ///< its thread_id
|
||||
|
||||
static size_t needed_size= 20480;
|
||||
|
||||
static const time_t startup_interval= 60*5; ///< in seconds (5 minutes)
|
||||
static const time_t first_interval= 60*60*24; ///< in seconds (one day)
|
||||
static const time_t interval= 60*60*24*7; ///< in seconds (one week)
|
||||
|
||||
/**
|
||||
reads the rows from a table and puts them, concatenated, in a String
|
||||
|
||||
@note
|
||||
1. only supports two column tables - no less, no more.
|
||||
2. it emulates mysql -e "select * from..." and thus it separates
|
||||
columns with \t and starts the output with column names.
|
||||
*/
|
||||
static int table_to_string(TABLE *table, String *result)
|
||||
{
|
||||
bool res;
|
||||
char buff1[MAX_FIELD_WIDTH], buff2[MAX_FIELD_WIDTH];
|
||||
String str1(buff1, sizeof(buff1), system_charset_info);
|
||||
String str2(buff2, sizeof(buff2), system_charset_info);
|
||||
|
||||
res= table->file->ha_rnd_init(1);
|
||||
|
||||
dbug_tmp_use_all_columns(table, table->read_set);
|
||||
|
||||
while(!res && !table->file->rnd_next(table->record[0]))
|
||||
{
|
||||
table->field[0]->val_str(&str1);
|
||||
table->field[1]->val_str(&str2);
|
||||
if (result->reserve(str1.length() + str2.length() + 3))
|
||||
res= 1;
|
||||
else
|
||||
{
|
||||
result->qs_append(str1.ptr(), str1.length());
|
||||
result->qs_append('\t');
|
||||
result->qs_append(str2.ptr(), str2.length());
|
||||
result->qs_append('\n');
|
||||
}
|
||||
}
|
||||
|
||||
res = res || result->append('\n');
|
||||
|
||||
/*
|
||||
Note, "|=" and not "||" - because we want to call ha_rnd_end()
|
||||
even if res is already 1.
|
||||
*/
|
||||
res |= table->file->ha_rnd_end();
|
||||
|
||||
return res;
|
||||
}
|
||||
|
||||
/**
|
||||
Initialize the THD and TABLE_LIST
|
||||
|
||||
The structures must be sufficiently initialized for create_tmp_table()
|
||||
and fill_feedback() to work.
|
||||
*/
|
||||
static int prepare_for_fill(TABLE_LIST *tables)
|
||||
{
|
||||
/*
|
||||
Add our thd to the list, for it to be visible in SHOW PROCESSLIST.
|
||||
But don't generate thread_id every time - use the saved value
|
||||
(every increment of global thread_id counts as a new connection
|
||||
in SHOW STATUS and we want to avoid skewing the statistics)
|
||||
*/
|
||||
thd->thread_id= thd->variables.pseudo_thread_id= thd_thread_id;
|
||||
pthread_mutex_lock(&LOCK_thread_count);
|
||||
thread_count++;
|
||||
threads.append(thd);
|
||||
pthread_mutex_unlock(&LOCK_thread_count);
|
||||
thd->thread_stack= (char*) &tables;
|
||||
if (thd->store_globals())
|
||||
return 1;
|
||||
|
||||
thd->mysys_var->current_cond= &sleep_condition;
|
||||
thd->mysys_var->current_mutex= &sleep_mutex;
|
||||
thd->proc_info="feedback";
|
||||
thd->command=COM_SLEEP;
|
||||
thd->version=refresh_version;
|
||||
thd->system_thread= SYSTEM_THREAD_EVENT_WORKER; // whatever
|
||||
thd->set_time();
|
||||
thd->init_for_queries();
|
||||
thd->real_id= pthread_self();
|
||||
thd->db= NULL;
|
||||
thd->db_length= 0;
|
||||
thd->security_ctx->host_or_ip= "";
|
||||
thd->security_ctx->db_access= DB_ACLS;
|
||||
thd->security_ctx->master_access= ~NO_ACCESS;
|
||||
bzero((char*) &thd->net, sizeof(thd->net));
|
||||
lex_start(thd);
|
||||
mysql_init_select(thd->lex);
|
||||
|
||||
tables->init_one_table(INFORMATION_SCHEMA_NAME.str,
|
||||
i_s_feedback->table_name, TL_READ);
|
||||
tables->schema_table= i_s_feedback;
|
||||
tables->table= i_s_feedback->create_table(thd, tables);
|
||||
if (!tables->table)
|
||||
return 1;
|
||||
|
||||
tables->table->pos_in_table_list= tables;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
Try to detect if this thread is going down
|
||||
|
||||
which can happen for different reasons:
|
||||
* plugin is being unloaded
|
||||
* mysqld server is being shut down
|
||||
* the thread is being killed
|
||||
|
||||
*/
|
||||
static bool going_down()
|
||||
{
|
||||
return shutdown_plugin || shutdown_in_progress || (thd && thd->killed);
|
||||
}
|
||||
|
||||
/**
|
||||
just like sleep, but waits on a condition and checks "plugin shutdown" status
|
||||
*/
|
||||
static int slept_ok(time_t sec)
|
||||
{
|
||||
struct timespec abstime;
|
||||
int ret= 0;
|
||||
|
||||
set_timespec(abstime, sec);
|
||||
|
||||
pthread_mutex_lock(&sleep_mutex);
|
||||
while (!going_down() && ret != ETIMEDOUT)
|
||||
ret= pthread_cond_timedwait(&sleep_condition, &sleep_mutex, &abstime);
|
||||
pthread_mutex_unlock(&sleep_mutex);
|
||||
|
||||
return !going_down();
|
||||
}
|
||||
|
||||
/**
|
||||
create a feedback report and send it to all specified urls
|
||||
|
||||
If "when" argument is not null, only it and the server uid are sent.
|
||||
Otherwise a full report is generated.
|
||||
*/
|
||||
static void send_report(const char *when)
|
||||
{
|
||||
TABLE_LIST tables;
|
||||
String str;
|
||||
int i, last_todo;
|
||||
Url **todo= (Url**)alloca(url_count*sizeof(Url*));
|
||||
|
||||
str.alloc(needed_size); // preallocate it to avoid many small mallocs
|
||||
|
||||
/*
|
||||
on startup and shutdown the server may not be completely
|
||||
initialized, and full report won't work.
|
||||
We send a short status notice only.
|
||||
*/
|
||||
if (when)
|
||||
{
|
||||
str.length(0);
|
||||
str.append(STRING_WITH_LEN("FEEDBACK_SERVER_UID"));
|
||||
str.append('\t');
|
||||
str.append(server_uid_buf);
|
||||
str.append('\n');
|
||||
str.append(STRING_WITH_LEN("FEEDBACK_WHEN"));
|
||||
str.append('\t');
|
||||
str.append(when);
|
||||
str.append('\n');
|
||||
str.append(STRING_WITH_LEN("FEEDBACK_USER_INFO"));
|
||||
str.append('\t');
|
||||
str.append(user_info);
|
||||
str.append('\n');
|
||||
str.append('\n');
|
||||
}
|
||||
else
|
||||
{
|
||||
/*
|
||||
otherwise, prepare the THD and TABLE_LIST,
|
||||
create and fill the temporary table with data just like
|
||||
SELECT * FROM IFROEMATION_SCHEMA.feedback is doing,
|
||||
read and concatenate table data into a String.
|
||||
*/
|
||||
if (!(thd= new THD()))
|
||||
return;
|
||||
|
||||
if (prepare_for_fill(&tables))
|
||||
goto ret;
|
||||
|
||||
if (fill_feedback(thd, &tables, NULL))
|
||||
goto ret;
|
||||
|
||||
if (table_to_string(tables.table, &str))
|
||||
goto ret;
|
||||
|
||||
needed_size= (size_t)(str.length() * 1.1);
|
||||
|
||||
free_tmp_table(thd, tables.table);
|
||||
tables.table= 0;
|
||||
}
|
||||
|
||||
/*
|
||||
Try to send the report on every url from the list, remove url on success,
|
||||
keep failed in the list. Repeat until the list is empty.
|
||||
*/
|
||||
memcpy(todo, urls, url_count*sizeof(Url*));
|
||||
last_todo= url_count - 1;
|
||||
do
|
||||
{
|
||||
for (i= 0; i <= last_todo;)
|
||||
{
|
||||
Url *url= todo[i];
|
||||
|
||||
if (thd) // for nicer SHOW PROCESSLIST
|
||||
thd->set_query(const_cast<char*>(url->url()), url->url_length());
|
||||
|
||||
if (url->send(str.ptr(), str.length()))
|
||||
i++;
|
||||
else
|
||||
todo[i]= todo[last_todo--];
|
||||
}
|
||||
if (last_todo < 0)
|
||||
break;
|
||||
} while (slept_ok(send_retry_wait)); // wait a little bit before retrying
|
||||
|
||||
ret:
|
||||
if (thd)
|
||||
{
|
||||
if (tables.table)
|
||||
free_tmp_table(thd, tables.table);
|
||||
/*
|
||||
clean up, free the thd.
|
||||
reset all thread local status variables to minimize
|
||||
the effect of the background thread on SHOW STATUS.
|
||||
*/
|
||||
pthread_mutex_lock(&LOCK_thread_count);
|
||||
bzero(&thd->status_var, sizeof(thd->status_var));
|
||||
thread_count--;
|
||||
thd->killed= THD::KILL_CONNECTION;
|
||||
pthread_cond_broadcast(&COND_thread_count);
|
||||
pthread_mutex_unlock(&LOCK_thread_count);
|
||||
delete thd;
|
||||
thd= 0;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
background sending thread
|
||||
*/
|
||||
pthread_handler_t background_thread(void *arg __attribute__((unused)))
|
||||
{
|
||||
if (my_thread_init())
|
||||
return 0;
|
||||
|
||||
pthread_mutex_lock(&LOCK_thread_count);
|
||||
thd_thread_id= thread_id++;
|
||||
pthread_mutex_unlock(&LOCK_thread_count);
|
||||
|
||||
if (slept_ok(startup_interval))
|
||||
{
|
||||
send_report("startup");
|
||||
|
||||
if (slept_ok(first_interval))
|
||||
{
|
||||
send_report(NULL);
|
||||
|
||||
while(slept_ok(interval))
|
||||
send_report(NULL);
|
||||
}
|
||||
|
||||
send_report("shutdown");
|
||||
}
|
||||
|
||||
my_thread_end();
|
||||
pthread_exit(0);
|
||||
return 0;
|
||||
}
|
||||
|
||||
} // namespace feedback
|
||||
|
51
plugin/feedback/url_base.cc
Normal file
51
plugin/feedback/url_base.cc
Normal file
@ -0,0 +1,51 @@
|
||||
/* Copyright (C) 2010 Sergei Golubchik and Monty Program Ab
|
||||
|
||||
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 */
|
||||
|
||||
#include "feedback.h"
|
||||
|
||||
namespace feedback {
|
||||
|
||||
Url* http_create(const char *url, size_t url_length);
|
||||
|
||||
/**
|
||||
creates an Url object out of an url, if possible.
|
||||
|
||||
This is done by invoking corresponding creator functions
|
||||
of the derived classes, until the first not NULL result.
|
||||
*/
|
||||
Url* Url::create(const char *url, size_t url_length)
|
||||
{
|
||||
url= my_strndup(url, url_length, MYF(MY_WME));
|
||||
|
||||
if (!url)
|
||||
return NULL;
|
||||
|
||||
Url *self= http_create(url, url_length);
|
||||
|
||||
/*
|
||||
here we can add
|
||||
|
||||
if (!self) self= smtp_create(url, url_length);
|
||||
if (!self) self= tftp_create(url, url_length);
|
||||
etc
|
||||
*/
|
||||
|
||||
if (!self)
|
||||
my_free(const_cast<char*>(url), MYF(0));
|
||||
|
||||
return self;
|
||||
}
|
||||
|
||||
} // namespace feedback
|
303
plugin/feedback/url_http.cc
Normal file
303
plugin/feedback/url_http.cc
Normal file
@ -0,0 +1,303 @@
|
||||
/* Copyright (C) 2010 Sergei Golubchik and Monty Program Ab
|
||||
|
||||
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 */
|
||||
|
||||
#include "feedback.h"
|
||||
|
||||
#ifdef HAVE_NETDB_H
|
||||
#include <netdb.h>
|
||||
#endif
|
||||
|
||||
#ifdef _WIN32
|
||||
#undef VOID
|
||||
#define VOID void
|
||||
#include <ws2tcpip.h>
|
||||
#define addrinfo ADDRINFOA
|
||||
#endif
|
||||
|
||||
namespace feedback {
|
||||
|
||||
static const uint FOR_READING= 0;
|
||||
static const uint FOR_WRITING= 1;
|
||||
|
||||
#ifdef MARIADB_BASE_VERSION
|
||||
#define ssl_connect(A,B,C,D) sslconnect(A,B,C,D)
|
||||
#else
|
||||
#define ssl_connect(A,B,C,D) sslconnect(A,B,C)
|
||||
#endif
|
||||
|
||||
/**
|
||||
implementation of the Url class that sends the data via HTTP POST request.
|
||||
|
||||
Both http:// and https:// protocols are supported.
|
||||
*/
|
||||
class Url_http: public Url {
|
||||
protected:
|
||||
const LEX_STRING host, port, path;
|
||||
bool ssl;
|
||||
|
||||
Url_http(LEX_STRING &url_arg, LEX_STRING &host_arg,
|
||||
LEX_STRING &port_arg, LEX_STRING &path_arg, bool ssl_arg) :
|
||||
Url(url_arg), host(host_arg), port(port_arg), path(path_arg), ssl(ssl_arg)
|
||||
{}
|
||||
~Url_http()
|
||||
{
|
||||
my_free(host.str, MYF(0));
|
||||
my_free(port.str, MYF(0));
|
||||
my_free(path.str, MYF(0));
|
||||
}
|
||||
|
||||
public:
|
||||
int send(const char* data, size_t data_length);
|
||||
|
||||
friend Url* http_create(const char *url, size_t url_length);
|
||||
};
|
||||
|
||||
/**
|
||||
create a Url_http object out of the url, if possible.
|
||||
|
||||
@note
|
||||
Arbitrary limitations here.
|
||||
|
||||
The url must be http[s]://hostname[:port]/path
|
||||
No username:password@ or ?script=parameters are supported.
|
||||
|
||||
But it's ok. This is not a generic purpose www browser - it only needs to be
|
||||
good enough to POST the data to mariadb.org.
|
||||
*/
|
||||
Url* http_create(const char *url, size_t url_length)
|
||||
{
|
||||
const char *s;
|
||||
LEX_STRING full_url= {const_cast<char*>(url), url_length};
|
||||
LEX_STRING host, port, path;
|
||||
bool ssl= false;
|
||||
|
||||
if (is_prefix(url, "http://"))
|
||||
s= url + 7;
|
||||
#ifdef HAVE_OPENSSL
|
||||
else if (is_prefix(url, "https://"))
|
||||
{
|
||||
ssl= true;
|
||||
s= url + 8;
|
||||
}
|
||||
#endif
|
||||
else
|
||||
return NULL;
|
||||
|
||||
for (url= s; *s && *s != ':' && *s != '/'; s++) /* no-op */;
|
||||
host.str= const_cast<char*>(url);
|
||||
host.length= s-url;
|
||||
|
||||
if (*s == ':')
|
||||
{
|
||||
for (url= ++s; *s && *s >= '0' && *s <= '9'; s++) /* no-op */;
|
||||
port.str= const_cast<char*>(url);
|
||||
port.length= s-url;
|
||||
}
|
||||
else
|
||||
{
|
||||
if (ssl)
|
||||
{
|
||||
port.str= const_cast<char*>("443");
|
||||
port.length=3;
|
||||
}
|
||||
else
|
||||
{
|
||||
port.str= const_cast<char*>("80");
|
||||
port.length=2;
|
||||
}
|
||||
}
|
||||
|
||||
if (*s == 0)
|
||||
{
|
||||
path.str= const_cast<char*>("/");
|
||||
path.length= 1;
|
||||
}
|
||||
else
|
||||
{
|
||||
path.str= const_cast<char*>(s);
|
||||
path.length= strlen(s);
|
||||
}
|
||||
if (!host.length || !port.length || path.str[0] != '/')
|
||||
return NULL;
|
||||
|
||||
host.str= my_strndup(host.str, host.length, MYF(MY_WME));
|
||||
port.str= my_strndup(port.str, port.length, MYF(MY_WME));
|
||||
path.str= my_strndup(path.str, path.length, MYF(MY_WME));
|
||||
|
||||
if (!host.str || !port.str || !path.str)
|
||||
{
|
||||
my_free(host.str, MYF(MY_ALLOW_ZERO_PTR));
|
||||
my_free(port.str, MYF(MY_ALLOW_ZERO_PTR));
|
||||
my_free(path.str, MYF(MY_ALLOW_ZERO_PTR));
|
||||
return NULL;
|
||||
}
|
||||
|
||||
return new Url_http(full_url, host, port, path, ssl);
|
||||
}
|
||||
|
||||
/* do the vio_write and check that all data were sent ok */
|
||||
#define write_check(VIO, DATA, LEN) \
|
||||
(vio_write((VIO), (uchar*)(DATA), (LEN)) != (LEN))
|
||||
|
||||
int Url_http::send(const char* data, size_t data_length)
|
||||
{
|
||||
my_socket fd= INVALID_SOCKET;
|
||||
char buf[1024];
|
||||
uint len;
|
||||
|
||||
addrinfo *addrs, *addr, filter= {0, AF_UNSPEC, SOCK_STREAM, 6, 0, 0, 0, 0};
|
||||
int res= getaddrinfo(host.str, port.str, &filter, &addrs);
|
||||
|
||||
if (res)
|
||||
{
|
||||
sql_print_error("feedback plugin: getaddrinfo() failed for url '%s': %s",
|
||||
full_url.str, gai_strerror(res));
|
||||
return 1;
|
||||
}
|
||||
|
||||
for (addr= addrs; addr != NULL; addr= addr->ai_next)
|
||||
{
|
||||
fd= socket(addr->ai_family, addr->ai_socktype, addr->ai_protocol);
|
||||
if (fd == INVALID_SOCKET)
|
||||
continue;
|
||||
|
||||
if (connect(fd, addr->ai_addr, addr->ai_addrlen) == 0)
|
||||
break;
|
||||
|
||||
closesocket(fd);
|
||||
}
|
||||
|
||||
freeaddrinfo(addrs);
|
||||
|
||||
if (fd == INVALID_SOCKET)
|
||||
{
|
||||
sql_print_error("feedback plugin: could not connect for url '%s'",
|
||||
full_url.str);
|
||||
return 1;
|
||||
}
|
||||
|
||||
Vio *vio= vio_new(fd, VIO_TYPE_TCPIP, 0);
|
||||
if (!vio)
|
||||
{
|
||||
sql_print_error("feedback plugin: vio_new failed for url '%s'",
|
||||
full_url.str);
|
||||
closesocket(fd);
|
||||
return 1;
|
||||
}
|
||||
|
||||
#ifdef HAVE_OPENSSL
|
||||
struct st_VioSSLFd *ssl_fd;
|
||||
if (ssl)
|
||||
{
|
||||
buf[0]= 0;
|
||||
if (!(ssl_fd= new_VioSSLConnectorFd(0, 0, 0, 0, 0)) ||
|
||||
ssl_connect(ssl_fd, vio, send_timeout, buf))
|
||||
{
|
||||
sql_print_error("feedback plugin: ssl failed for url '%s' %s",
|
||||
full_url.str, buf);
|
||||
if (ssl_fd)
|
||||
free_vio_ssl_acceptor_fd(ssl_fd);
|
||||
closesocket(fd);
|
||||
vio_delete(vio);
|
||||
return 1;
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
static const LEX_STRING boundary=
|
||||
{ C_STRING_WITH_LEN("----------------------------ba4f3696b39f") };
|
||||
static const LEX_STRING header=
|
||||
{ C_STRING_WITH_LEN("\r\n"
|
||||
"Content-Disposition: form-data; name=\"data\"; filename=\"-\"\r\n"
|
||||
"Content-Type: application/octet-stream\r\n\r\n")
|
||||
};
|
||||
|
||||
len= my_snprintf(buf, sizeof(buf),
|
||||
"POST %s HTTP/1.0\r\n"
|
||||
"User-Agent: MariaDB User Feedback Plugin\r\n"
|
||||
"Host: %s:%s\r\n"
|
||||
"Accept: */*\r\n"
|
||||
"Content-Length: %u\r\n"
|
||||
"Content-Type: multipart/form-data; boundary=%s\r\n"
|
||||
"\r\n",
|
||||
path.str, host.str, port.str,
|
||||
(uint)(2*boundary.length + header.length + data_length + 4),
|
||||
boundary.str + 2);
|
||||
|
||||
vio_timeout(vio, FOR_READING, send_timeout);
|
||||
vio_timeout(vio, FOR_WRITING, send_timeout);
|
||||
res = write_check(vio, buf, len)
|
||||
|| write_check(vio, boundary.str, boundary.length)
|
||||
|| write_check(vio, header.str, header.length)
|
||||
|| write_check(vio, data, data_length)
|
||||
|| write_check(vio, boundary.str, boundary.length)
|
||||
|| write_check(vio, "--\r\n", 4);
|
||||
|
||||
if (res)
|
||||
sql_print_error("feedback plugin: failed to send report to '%s'",
|
||||
full_url.str);
|
||||
else
|
||||
{
|
||||
sql_print_information("feedback plugin: report to '%s' was sent",
|
||||
full_url.str);
|
||||
|
||||
/*
|
||||
if the data were send successfully, read the reply.
|
||||
Extract the first string between <h1>...</h1> tags
|
||||
and put it as a server reply into the error log.
|
||||
*/
|
||||
len= vio_read(vio, (uchar*)buf, sizeof(buf)-1);
|
||||
if (len && len < sizeof(buf))
|
||||
{
|
||||
char *from;
|
||||
|
||||
buf[len+1]= 0; // safety
|
||||
|
||||
if ((from= strstr(buf, "<h1>")))
|
||||
{
|
||||
from+= 4;
|
||||
char *to= strstr(from, "</h1>");
|
||||
if (to)
|
||||
*to= 0;
|
||||
else
|
||||
from= NULL;
|
||||
}
|
||||
if (from)
|
||||
sql_print_information("feedback plugin: server replied '%s'", from);
|
||||
else
|
||||
sql_print_warning("feedback plugin: failed to parse server reply");
|
||||
}
|
||||
else
|
||||
{
|
||||
res= 1;
|
||||
sql_print_error("feedback plugin: failed to read server reply");
|
||||
}
|
||||
}
|
||||
|
||||
vio_delete(vio);
|
||||
|
||||
#ifdef HAVE_OPENSSL
|
||||
if (ssl)
|
||||
{
|
||||
SSL_CTX_free(ssl_fd->ssl_context);
|
||||
my_free(ssl_fd, MYF(0));
|
||||
}
|
||||
#endif
|
||||
|
||||
return res;
|
||||
}
|
||||
|
||||
} // namespace feedback
|
||||
|
299
plugin/feedback/utils.cc
Normal file
299
plugin/feedback/utils.cc
Normal file
@ -0,0 +1,299 @@
|
||||
/* Copyright (C) 2010 Sergei Golubchik and Monty Program Ab
|
||||
|
||||
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 */
|
||||
|
||||
#include "feedback.h"
|
||||
|
||||
#ifdef HAVE_UNISTD_H
|
||||
#include <unistd.h>
|
||||
#endif
|
||||
|
||||
#include <base64.h>
|
||||
#include <sha1.h>
|
||||
|
||||
#ifdef HAVE_SYS_UTSNAME_H
|
||||
#include <sys/utsname.h>
|
||||
static bool have_ubuf= false;
|
||||
static struct utsname ubuf;
|
||||
#endif
|
||||
|
||||
#ifdef TARGET_OS_LINUX
|
||||
#include <glob.h>
|
||||
static bool have_distribution= false;
|
||||
static char distribution[256];
|
||||
|
||||
static const char *masks[]= {
|
||||
"/etc/*-version", "/etc/*-release",
|
||||
"/etc/*_version", "/etc/*_release"
|
||||
};
|
||||
#endif
|
||||
|
||||
bool schema_table_store_record(THD *thd, TABLE *table);
|
||||
|
||||
namespace feedback {
|
||||
|
||||
/*
|
||||
convenience macros for inserting rows into I_S table.
|
||||
*/
|
||||
#define INSERT2(NAME,LEN,VALUE) \
|
||||
do { \
|
||||
table->field[0]->store(NAME, LEN, system_charset_info); \
|
||||
table->field[1]->store VALUE; \
|
||||
if (schema_table_store_record(thd, table)) \
|
||||
return 1; \
|
||||
} while (0)
|
||||
|
||||
#define INSERT1(NAME,VALUE) \
|
||||
do { \
|
||||
table->field[0]->store(NAME, sizeof(NAME)-1, system_charset_info); \
|
||||
table->field[1]->store VALUE; \
|
||||
if (schema_table_store_record(thd, table)) \
|
||||
return 1; \
|
||||
} while (0)
|
||||
|
||||
static const bool UNSIGNED= true; ///< used below when inserting integers
|
||||
|
||||
/**
|
||||
callback for fill_plugin_version() - insert a plugin name and its version
|
||||
*/
|
||||
static my_bool show_plugins(THD *thd, plugin_ref plugin, void *arg)
|
||||
{
|
||||
TABLE *table= (TABLE*) arg;
|
||||
char version[20];
|
||||
size_t version_len;
|
||||
|
||||
version_len= my_snprintf(version, sizeof(version), "%d.%d",
|
||||
(plugin_decl(plugin)->version) >> 8,
|
||||
(plugin_decl(plugin)->version) & 0xff);
|
||||
|
||||
INSERT2(plugin_name(plugin)->str, plugin_name(plugin)->length,
|
||||
(version, version_len, system_charset_info));
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
inserts all plugins and their versions into I_S.FEEDBACK
|
||||
*/
|
||||
int fill_plugin_version(THD *thd, TABLE_LIST *tables)
|
||||
{
|
||||
return plugin_foreach_with_mask(thd, show_plugins, MYSQL_ANY_PLUGIN,
|
||||
~PLUGIN_IS_FREED, tables->table);
|
||||
}
|
||||
|
||||
#if defined(_SC_PAGE_SIZE) && !defined(_SC_PAGESIZE)
|
||||
#define _SC_PAGESIZE _SC_PAGE_SIZE
|
||||
#endif
|
||||
|
||||
/**
|
||||
return the amount of physical memory
|
||||
*/
|
||||
static ulonglong my_getphysmem()
|
||||
{
|
||||
ulonglong pages= 0;
|
||||
#ifdef _SC_PHYS_PAGES
|
||||
pages= sysconf(_SC_PHYS_PAGES);
|
||||
#else
|
||||
return 0;
|
||||
#endif
|
||||
|
||||
#ifdef _SC_PAGESIZE
|
||||
return pages * sysconf(_SC_PAGESIZE);
|
||||
#else
|
||||
return pages * my_getpagesize();
|
||||
#endif
|
||||
}
|
||||
|
||||
/* get the number of (online) CPUs */
|
||||
int my_getncpus()
|
||||
{
|
||||
#ifdef _SC_NPROCESSORS_ONLN
|
||||
return sysconf(_SC_NPROCESSORS_ONLN);
|
||||
#elif defined(__WIN__)
|
||||
SYSTEM_INFO sysinfo;
|
||||
GetSystemInfo(&sysinfo);
|
||||
return sysinfo.dwNumberOfProcessors;
|
||||
#else
|
||||
return 0;
|
||||
#endif
|
||||
}
|
||||
|
||||
/**
|
||||
Find the version of the kernel and the linux distribution
|
||||
*/
|
||||
int prepare_linux_info()
|
||||
{
|
||||
#ifdef HAVE_SYS_UTSNAME_H
|
||||
have_ubuf= (uname(&ubuf) != -1);
|
||||
#endif
|
||||
|
||||
#ifdef TARGET_OS_LINUX
|
||||
/*
|
||||
let's try to find what linux distribution it is
|
||||
we read *[-_]{release,version} file in /etc.
|
||||
|
||||
Either it will be /etc/lsb-release, such as
|
||||
|
||||
==> /etc/lsb-release <==
|
||||
DISTRIB_ID=Ubuntu
|
||||
DISTRIB_RELEASE=8.04
|
||||
DISTRIB_CODENAME=hardy
|
||||
DISTRIB_DESCRIPTION="Ubuntu 8.04.4 LTS"
|
||||
|
||||
Or a one-liner with the description (/etc/SuSE-release has more
|
||||
than one line, but the description is the first, so it can be
|
||||
treated as a one-liner).
|
||||
|
||||
We'll read lsb-release first, and if it's not found will search
|
||||
for other files (*-version *-release *_version *_release)
|
||||
*/
|
||||
int fd;
|
||||
have_distribution= false;
|
||||
if ((fd= my_open("/etc/lsb-release", O_RDONLY, MYF(0))) != -1)
|
||||
{
|
||||
/* Cool, LSB-compliant distribution! */
|
||||
size_t len= my_read(fd, (uchar*)distribution, sizeof(distribution)-1, MYF(0));
|
||||
my_close(fd, MYF(0));
|
||||
if (len != (size_t)-1)
|
||||
{
|
||||
distribution[len]= 0; // safety
|
||||
char *found= strstr(distribution, "DISTRIB_DESCRIPTION=");
|
||||
if (found)
|
||||
{
|
||||
have_distribution= true;
|
||||
char *end= strstr(found, "\n");
|
||||
if (end == NULL)
|
||||
end= distribution + len;
|
||||
found+= 20;
|
||||
|
||||
if (*found == '"' && end[-1] == '"')
|
||||
{
|
||||
found++;
|
||||
end--;
|
||||
}
|
||||
*end= 0;
|
||||
|
||||
char *to= strmov(distribution, "lsb: ");
|
||||
memmove(to, found, end - found + 1);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/* if not an LSB-compliant distribution */
|
||||
for (uint i= 0; !have_distribution && i < array_elements(masks); i++)
|
||||
{
|
||||
glob_t found;
|
||||
if (glob(masks[i], GLOB_NOSORT, NULL, &found) == 0)
|
||||
{
|
||||
int fd;
|
||||
if ((fd= my_open(found.gl_pathv[0], O_RDONLY, MYF(0))) != -1)
|
||||
{
|
||||
/*
|
||||
+5 and -8 below cut the file name part out of the
|
||||
full pathname that corresponds to the mask as above.
|
||||
*/
|
||||
char *to= strmov(distribution, found.gl_pathv[0] + 5) - 8;
|
||||
*to++= ':';
|
||||
*to++= ' ';
|
||||
|
||||
size_t to_len= distribution + sizeof(distribution) - 1 - to;
|
||||
size_t len= my_read(fd, (uchar*)to, to_len, MYF(0));
|
||||
my_close(fd, MYF(0));
|
||||
if (len != (size_t)-1)
|
||||
{
|
||||
to[len]= 0; // safety
|
||||
char *end= strstr(to, "\n");
|
||||
if (end)
|
||||
*end= 0;
|
||||
have_distribution= true;
|
||||
}
|
||||
}
|
||||
}
|
||||
globfree(&found);
|
||||
}
|
||||
#endif
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
Add the linux distribution and the kernel version
|
||||
*/
|
||||
int fill_linux_info(THD *thd, TABLE_LIST *tables)
|
||||
{
|
||||
TABLE *table= tables->table;
|
||||
CHARSET_INFO *cs= system_charset_info;
|
||||
|
||||
#ifdef HAVE_SYS_UTSNAME_H
|
||||
if (have_ubuf)
|
||||
{
|
||||
INSERT1("Uname_sysname", (ubuf.sysname, strlen(ubuf.sysname), cs));
|
||||
INSERT1("Uname_release", (ubuf.release, strlen(ubuf.release), cs));
|
||||
INSERT1("Uname_version", (ubuf.version, strlen(ubuf.version), cs));
|
||||
INSERT1("Uname_machine", (ubuf.machine, strlen(ubuf.machine), cs));
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef TARGET_OS_LINUX
|
||||
if (have_distribution)
|
||||
INSERT1("Uname_distribution", (distribution, strlen(distribution), cs));
|
||||
#endif
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
Adds varios bits of information to the I_S.FEEDBACK
|
||||
*/
|
||||
int fill_misc_data(THD *thd, TABLE_LIST *tables)
|
||||
{
|
||||
TABLE *table= tables->table;
|
||||
|
||||
#ifdef MY_ATOMIC_OK
|
||||
INSERT1("Cpu_count", (my_getncpus(), UNSIGNED));
|
||||
#endif
|
||||
INSERT1("Mem_total", (my_getphysmem(), UNSIGNED));
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
calculates the server unique identifier
|
||||
|
||||
UID is a base64 encoded SHA1 hash of the MAC address of one of
|
||||
the interfaces, and the tcp port that the server is listening on
|
||||
*/
|
||||
int calculate_server_uid(char *dest)
|
||||
{
|
||||
uchar rawbuf[2 + 6];
|
||||
uchar shabuf[SHA1_HASH_SIZE];
|
||||
SHA1_CONTEXT ctx;
|
||||
|
||||
int2store(rawbuf, mysqld_port);
|
||||
if (my_gethwaddr(rawbuf + 2))
|
||||
{
|
||||
sql_print_error("feedback plugin: failed to retrieve the MAC address");
|
||||
return 1;
|
||||
}
|
||||
|
||||
mysql_sha1_reset(&ctx);
|
||||
mysql_sha1_input(&ctx, rawbuf, sizeof(rawbuf));
|
||||
mysql_sha1_result(&ctx, shabuf);
|
||||
|
||||
assert(base64_needed_encoded_length(sizeof(shabuf)) <= SERVER_UID_SIZE);
|
||||
base64_encode(shabuf, sizeof(shabuf), dest);
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
} // namespace feedback
|
Reference in New Issue
Block a user