27
27
#include <algorithm>
28
28
#include <boost/program_options.hpp>
29
29
#include <drizzled/module/option_map.h>
30
#include "drizzled/util/tokenize.h"
31
30
#include "errmsg.h"
32
31
#include "mysql_protocol.h"
33
32
#include "mysql_password.h"
34
33
#include "options.h"
35
34
#include "table_function.h"
37
#define PROTOCOL_VERSION 10
39
36
namespace po= boost::program_options;
40
37
using namespace std;
41
38
using namespace drizzled;
43
namespace drizzle_plugin
40
#define PROTOCOL_VERSION 10
46
std::vector<std::string> ClientMySQLProtocol::mysql_admin_ip_addresses;
47
42
static const unsigned int PACKET_BUFFER_EXTRA_ALLOC= 1024;
49
static port_constraint port;
50
static timeout_constraint connect_timeout;
51
static timeout_constraint read_timeout;
52
static timeout_constraint write_timeout;
53
static retry_constraint retry_count;
54
static buffer_constraint buffer_length;
44
static uint32_t connect_timeout;
45
static uint32_t read_timeout;
46
static uint32_t write_timeout;
47
static uint32_t retry_count;
48
static uint32_t buffer_length;
49
static char* bind_address;
56
50
static uint32_t random_seed1;
57
51
static uint32_t random_seed2;
58
52
static const uint32_t random_max= 0x3FFFFFFF;
59
53
static const double random_max_double= (double)0x3FFFFFFF;
62
ProtocolCounters *ListenMySQLProtocol::mysql_counters= new ProtocolCounters();
55
static plugin::TableFunction* mysql_status_table_function_ptr= NULL;
64
57
ListenMySQLProtocol::~ListenMySQLProtocol()
59
/* This is strdup'd from the options */
67
const std::string ListenMySQLProtocol::getHost(void) const
63
const char* ListenMySQLProtocol::getHost(void) const
72
68
in_port_t ListenMySQLProtocol::getPort(void) const
70
return (in_port_t) port;
77
73
plugin::Client *ListenMySQLProtocol::getClient(int fd)
84
return new ClientMySQLProtocol(new_fd, _using_mysql41_protocol, getCounters());
80
return new (nothrow) ClientMySQLProtocol(new_fd, using_mysql41_protocol);
87
ClientMySQLProtocol::ClientMySQLProtocol(int fd, bool using_mysql41_protocol, ProtocolCounters *set_counters):
88
is_admin_connection(false),
89
_using_mysql41_protocol(using_mysql41_protocol),
90
counters(set_counters)
83
drizzled::atomic<uint64_t> ClientMySQLProtocol::connectionCount;
84
drizzled::atomic<uint64_t> ClientMySQLProtocol::failedConnections;
85
drizzled::atomic<uint64_t> ClientMySQLProtocol::connected;
87
ClientMySQLProtocol::ClientMySQLProtocol(int fd, bool using_mysql41_protocol_arg):
88
using_mysql41_protocol(using_mysql41_protocol_arg)
98
if (drizzleclient_net_init_sock(&net, fd, buffer_length.get()))
95
if (drizzleclient_net_init_sock(&net, fd, buffer_length))
101
drizzleclient_net_set_read_timeout(&net, read_timeout.get());
102
drizzleclient_net_set_write_timeout(&net, write_timeout.get());
103
net.retry_count=retry_count.get();
98
drizzleclient_net_set_read_timeout(&net, read_timeout);
99
drizzleclient_net_set_write_timeout(&net, write_timeout);
100
net.retry_count=retry_count;
106
103
ClientMySQLProtocol::~ClientMySQLProtocol()
112
109
int ClientMySQLProtocol::getFileDescriptor(void)
146
143
drizzleclient_net_close(&net);
147
144
drizzleclient_net_end(&net);
148
if (is_admin_connection)
149
counters->adminConnected.decrement();
151
counters->connected.decrement();
145
connected.decrement();
155
149
bool ClientMySQLProtocol::authenticate()
157
151
bool connection_is_valid;
158
if (is_admin_connection)
160
counters->adminConnectionCount.increment();
161
counters->adminConnected.increment();
165
counters->connectionCount.increment();
166
counters->connected.increment();
153
connectionCount.increment();
154
connected.increment();
169
156
/* Use "connect_timeout" value during connection phase */
170
drizzleclient_net_set_read_timeout(&net, connect_timeout.get());
171
drizzleclient_net_set_write_timeout(&net, connect_timeout.get());
157
drizzleclient_net_set_read_timeout(&net, connect_timeout);
158
drizzleclient_net_set_write_timeout(&net, connect_timeout);
173
160
connection_is_valid= checkConnection();
175
162
if (connection_is_valid)
177
if (not is_admin_connection and (counters->connected > counters->max_connections))
179
std::string errmsg(ER(ER_CON_COUNT_ERROR));
180
sendError(ER_CON_COUNT_ERROR, errmsg.c_str());
181
counters->failedConnections.increment();
190
166
sendError(session->main_da.sql_errno(), session->main_da.message());
191
counters->failedConnections.increment();
167
failedConnections.increment();
195
170
/* Connect completed, set read/write timeouts back to default */
196
drizzleclient_net_set_read_timeout(&net, read_timeout.get());
197
drizzleclient_net_set_write_timeout(&net, write_timeout.get());
171
drizzleclient_net_set_read_timeout(&net, read_timeout);
172
drizzleclient_net_set_write_timeout(&net, write_timeout);
947
void ClientMySQLProtocol::mysql_compose_ip_addresses(vector<string> options)
949
for (vector<string>::iterator it= options.begin();
953
tokenize(*it, mysql_admin_ip_addresses, ",", true);
957
897
static ListenMySQLProtocol *listen_obj= NULL;
958
898
plugin::Create_function<MySQLPassword> *mysql_password= NULL;
960
900
static int init(drizzled::module::Context &context)
962
context.add(new MysqlProtocolStatus);
902
mysql_status_table_function_ptr= new MysqlProtocolStatus;
904
context.add(mysql_status_table_function_ptr);
964
905
/* Initialize random seeds for the MySQL algorithm with minimal changes. */
965
906
time_t seed_time= time(NULL);
966
907
random_seed1= seed_time % random_max;
967
908
random_seed2= (seed_time / 2) % random_max;
969
910
const module::option_map &vm= context.getOptions();
911
if (vm.count("port"))
915
errmsg_printf(ERRMSG_LVL_ERROR, _("Invalid value of port\n"));
920
if (vm.count("connect-timeout"))
922
if (connect_timeout < 1 || connect_timeout > 300)
924
errmsg_printf(ERRMSG_LVL_ERROR, _("Invalid value for connect_timeout\n"));
929
if (vm.count("read-timeout"))
931
if (read_timeout < 1 || read_timeout > 300)
933
errmsg_printf(ERRMSG_LVL_ERROR, _("Invalid value for read_timeout\n"));
938
if (vm.count("write-timeout"))
940
if (write_timeout < 1 || write_timeout > 300)
942
errmsg_printf(ERRMSG_LVL_ERROR, _("Invalid value for write_timeout\n"));
947
if (vm.count("retry-count"))
949
if (retry_count < 1 || retry_count > 100)
951
errmsg_printf(ERRMSG_LVL_ERROR, _("Invalid value for retry_count"));
956
if (vm.count("buffer-length"))
958
if (buffer_length < 1024 || buffer_length > 1024*1024)
960
errmsg_printf(ERRMSG_LVL_ERROR, _("Invalid value for buffer_length\n"));
965
if (vm.count("bind-address"))
967
bind_address= strdup(vm["bind-address"].as<string>().c_str());
971
975
mysql_password= new plugin::Create_function<MySQLPassword>(MySQLPasswordName);
972
976
context.add(mysql_password);
974
listen_obj= new ListenMySQLProtocol("mysql_protocol", vm["bind-address"].as<std::string>(), true);
978
listen_obj= new ListenMySQLProtocol("mysql_protocol", true);
975
979
context.add(listen_obj);
976
context.registerVariable(new sys_var_constrained_value_readonly<in_port_t>("port", port));
977
context.registerVariable(new sys_var_constrained_value<uint32_t>("connect_timeout", connect_timeout));
978
context.registerVariable(new sys_var_constrained_value<uint32_t>("read_timeout", read_timeout));
979
context.registerVariable(new sys_var_constrained_value<uint32_t>("write_timeout", write_timeout));
980
context.registerVariable(new sys_var_constrained_value<uint32_t>("retry_count", retry_count));
981
context.registerVariable(new sys_var_constrained_value<uint32_t>("buffer_length", buffer_length));
982
context.registerVariable(new sys_var_const_string_val("bind_address",
983
vm["bind-address"].as<std::string>()));
985
context.registerVariable(new sys_var_uint32_t_ptr("max-connections", &ListenMySQLProtocol::mysql_counters->max_connections));
984
static DRIZZLE_SYSVAR_UINT(port, port, PLUGIN_VAR_RQCMDARG,
985
N_("Port number to use for connection or 0 for default to with MySQL "
987
NULL, NULL, 3306, 0, 65535, 0);
988
static DRIZZLE_SYSVAR_UINT(connect_timeout, connect_timeout,
989
PLUGIN_VAR_RQCMDARG, N_("Connect Timeout."),
990
NULL, NULL, 10, 1, 300, 0);
991
static DRIZZLE_SYSVAR_UINT(read_timeout, read_timeout, PLUGIN_VAR_RQCMDARG,
992
N_("Read Timeout."), NULL, NULL, 30, 1, 300, 0);
993
static DRIZZLE_SYSVAR_UINT(write_timeout, write_timeout, PLUGIN_VAR_RQCMDARG,
994
N_("Write Timeout."), NULL, NULL, 60, 1, 300, 0);
995
static DRIZZLE_SYSVAR_UINT(retry_count, retry_count, PLUGIN_VAR_RQCMDARG,
996
N_("Retry Count."), NULL, NULL, 10, 1, 100, 0);
997
static DRIZZLE_SYSVAR_UINT(buffer_length, buffer_length, PLUGIN_VAR_RQCMDARG,
998
N_("Buffer length."), NULL, NULL, 16384, 1024,
1000
static DRIZZLE_SYSVAR_STR(bind_address, bind_address, PLUGIN_VAR_READONLY,
1001
N_("Address to bind to."), NULL, NULL, NULL);
990
1003
static void init_options(drizzled::module::option_context &context)
993
po::value<port_constraint>(&port)->default_value(3306),
1006
po::value<uint32_t>(&port)->default_value(3306),
994
1007
N_("Port number to use for connection or 0 for default to with MySQL "
996
1009
context("connect-timeout",
997
po::value<timeout_constraint>(&connect_timeout)->default_value(10),
1010
po::value<uint32_t>(&connect_timeout)->default_value(10),
998
1011
N_("Connect Timeout."));
999
1012
context("read-timeout",
1000
po::value<timeout_constraint>(&read_timeout)->default_value(30),
1013
po::value<uint32_t>(&read_timeout)->default_value(30),
1001
1014
N_("Read Timeout."));
1002
1015
context("write-timeout",
1003
po::value<timeout_constraint>(&write_timeout)->default_value(60),
1016
po::value<uint32_t>(&write_timeout)->default_value(60),
1004
1017
N_("Write Timeout."));
1005
1018
context("retry-count",
1006
po::value<retry_constraint>(&retry_count)->default_value(10),
1019
po::value<uint32_t>(&retry_count)->default_value(10),
1007
1020
N_("Retry Count."));
1008
1021
context("buffer-length",
1009
po::value<buffer_constraint>(&buffer_length)->default_value(16384),
1022
po::value<uint32_t>(&buffer_length)->default_value(16384),
1010
1023
N_("Buffer length."));
1011
1024
context("bind-address",
1012
po::value<string>()->default_value(""),
1025
po::value<string>(),
1013
1026
N_("Address to bind to."));
1014
context("max-connections",
1015
po::value<uint32_t>(&ListenMySQLProtocol::mysql_counters->max_connections)->default_value(1000),
1016
N_("Maximum simultaneous connections."));
1017
context("admin-ip-addresses",
1018
po::value<vector<string> >()->composing()->notifier(&ClientMySQLProtocol::mysql_compose_ip_addresses),
1019
N_("A restrictive IP address list for incoming admin connections."));
1029
static drizzle_sys_var* sys_variables[]= {
1030
DRIZZLE_SYSVAR(port),
1031
DRIZZLE_SYSVAR(connect_timeout),
1032
DRIZZLE_SYSVAR(read_timeout),
1033
DRIZZLE_SYSVAR(write_timeout),
1034
DRIZZLE_SYSVAR(retry_count),
1035
DRIZZLE_SYSVAR(buffer_length),
1036
DRIZZLE_SYSVAR(bind_address),
1022
1040
static int mysql_protocol_connection_count_func(drizzle_show_var *var, char *buff)
1024
1042
var->type= SHOW_LONGLONG;
1025
1043
var->value= buff;
1026
*((uint64_t *)buff)= ListenMySQLProtocol::mysql_counters->connectionCount;
1044
*((uint64_t *)buff)= ClientMySQLProtocol::connectionCount;