504 lines
15 KiB
C++
504 lines
15 KiB
C++
/*
|
|
* Copyright (c) 2015, 2019, Oracle and/or its affiliates. All rights reserved.
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License, version 2.0,
|
|
* as published by the Free Software Foundation.
|
|
*
|
|
* This program is also distributed with certain software (including
|
|
* but not limited to OpenSSL) that is licensed under separate terms,
|
|
* as designated in a particular file or component or in included license
|
|
* documentation. The authors of MySQL hereby grant you an additional
|
|
* permission to link the program and your derivative works with the
|
|
* separately licensed software that they have included with MySQL.
|
|
*
|
|
* 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, version 2.0, 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., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*/
|
|
|
|
// MySQL DB access module, for use by plugins and others
|
|
// For the module that implements interactive DB functionality see mod_db
|
|
|
|
#include "../../global_defines.h"
|
|
#ifndef MYSQL8
|
|
#define MYSQL_SERVER
|
|
#include "sql/sql_class.h"
|
|
#endif
|
|
|
|
#include "xconnection_impl.h"
|
|
|
|
#include "my_config.h"
|
|
#include "my_dbug.h"
|
|
|
|
#include <errno.h>
|
|
#ifdef HAVE_NETINET_IN_H
|
|
#include <netinet/in.h>
|
|
#endif // HAVE_NETINET_IN_H
|
|
#ifdef HAVE_OPENSSL
|
|
#include <openssl/x509v3.h>
|
|
#endif // HAVE_OPENSSL
|
|
#include <cassert>
|
|
#include <chrono>
|
|
#include <future>
|
|
#include <limits>
|
|
#include <sstream>
|
|
#include <string>
|
|
|
|
#include "errmsg.h"
|
|
#ifdef MYSQL8
|
|
#include "my_macros.h"
|
|
#endif
|
|
|
|
#include "context/xconnection_config.h"
|
|
|
|
#ifndef WIN32
|
|
#include <netdb.h>
|
|
#include <sys/socket.h>
|
|
#endif // WIN32
|
|
#ifdef HAVE_SYS_UN_H
|
|
#include <sys/un.h>
|
|
#endif // HAVE_SYS_UN_H
|
|
|
|
#ifdef HAVE_OPENSSL
|
|
#define HAVE_SSL(Y, N) Y
|
|
#else
|
|
#define HAVE_SSL(Y, N) N
|
|
#endif // HAVE_OPENSSL
|
|
|
|
#ifdef WIN32
|
|
#define SHUT_RD SD_RECEIVE
|
|
#define SHUT_WR SD_SEND
|
|
#endif
|
|
|
|
namespace xcl {
|
|
|
|
const char *const ER_TEXT_SERVER_GONE = "MySQL server has gone away";
|
|
const char *const ER_TEXT_CANT_SET_TIMEOUT_WHEN_NOT_CONNECTED =
|
|
"Can't set a timeout, socket not connected.";
|
|
const char *const ER_TEXT_TLS_ALREADY_ACTIVATED = "TLS already activated";
|
|
const char *const ER_TEXT_TLS_NOT_CONFIGURATED = "TLS not configured";
|
|
const char *const ER_TEXT_INVALID_SOCKET = "Invalid socket";
|
|
const char *const ER_TEXT_UN_SOCKET_FILE_NOT_SET =
|
|
"UNIX Socket file was not specified";
|
|
const char *const ER_TEXT_CANT_TIMEOUT_WHILE_READING =
|
|
"Read operation failed because of a timeout";
|
|
const char *const ER_TEXT_CANT_TIMEOUT_WHILE_WRITTING =
|
|
"Write operation failed because of a timeout";
|
|
|
|
#if !defined(HAVE_SYS_UN_H)
|
|
const char *const ER_TEXT_UN_SOCKET_NOT_SUPPORTED =
|
|
"UNIX sockets aren't supported on current OS";
|
|
#endif // !defined(HAVE_SYS_UN_H)
|
|
|
|
namespace details {
|
|
|
|
class Connection_state : public XConnection::State {
|
|
public:
|
|
Connection_state(Vio *vio, const bool is_ssl_configured,
|
|
const bool is_ssl_active, const bool is_connected,
|
|
const Connection_type connection_type)
|
|
: m_vio(vio),
|
|
m_is_ssl_configured(is_ssl_configured),
|
|
m_is_ssl_active(is_ssl_active),
|
|
m_is_connected(is_connected),
|
|
m_connection_type(connection_type) {}
|
|
|
|
bool is_ssl_configured() const override { return m_is_ssl_configured; }
|
|
bool is_ssl_activated() const override { return m_is_ssl_active; }
|
|
bool is_connected() const override { return m_is_connected; }
|
|
|
|
Connection_type get_connection_type() const override {
|
|
return m_connection_type;
|
|
}
|
|
|
|
bool has_data() const override {
|
|
bool res = m_vio->has_data(m_vio);
|
|
if (res) return true;
|
|
|
|
return vio_io_wait(m_vio, VIO_IO_EVENT_READ, 0) != 0;
|
|
}
|
|
|
|
Vio *m_vio;
|
|
bool m_is_ssl_configured;
|
|
bool m_is_ssl_active;
|
|
bool m_is_connected;
|
|
Connection_type m_connection_type;
|
|
};
|
|
|
|
const char *null_when_empty(const std::string &value) {
|
|
if (value.empty()) return nullptr;
|
|
|
|
return value.c_str();
|
|
}
|
|
|
|
int make_vio_timeout(const int64_t value) {
|
|
if (value > 0) {
|
|
if (std::numeric_limits<int>::max() < value)
|
|
return std::numeric_limits<int>::max();
|
|
|
|
return static_cast<int>(value);
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
inline int get_shutdown_consts(const Shutdown_type type) {
|
|
switch (type) {
|
|
case Shutdown_type::Send:
|
|
return SHUT_WR;
|
|
case Shutdown_type::Recv:
|
|
return SHUT_RD;
|
|
case Shutdown_type::Both:
|
|
return SHUT_RDWR;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
} // namespace details
|
|
|
|
Connection_impl::Connection_impl(std::shared_ptr<Context> context)
|
|
: m_vio(nullptr),
|
|
m_ssl_active(false),
|
|
m_connected(false),
|
|
m_context(context) {}
|
|
|
|
Connection_impl::~Connection_impl() { close(); }
|
|
|
|
XError Connection_impl::connect_to_localhost(const std::string &unix_socket) {
|
|
m_connection_type = Connection_type::Unix_socket;
|
|
m_hostname = "localhost";
|
|
#if defined(HAVE_SYS_UN_H)
|
|
sockaddr_un addr;
|
|
|
|
if (unix_socket.empty())
|
|
return XError(CR_UNKNOWN_HOST, ER_TEXT_UN_SOCKET_FILE_NOT_SET, true);
|
|
|
|
if (unix_socket.length() > (sizeof(addr.sun_path) - 1)) {
|
|
std::stringstream stream;
|
|
|
|
stream << "UNIX Socket file name too long, size should be less or equal "
|
|
<< sizeof(addr.sun_path) - 1;
|
|
return XError(CR_UNKNOWN_HOST, stream.str(), true);
|
|
}
|
|
|
|
memset(&addr, 0, sizeof(addr));
|
|
addr.sun_family = AF_UNIX;
|
|
strncpy(addr.sun_path, unix_socket.c_str(), sizeof(addr.sun_path) - 1);
|
|
addr.sun_path[sizeof(addr.sun_path) - 1] = 0;
|
|
|
|
auto error = connect(reinterpret_cast<sockaddr *>(&addr), sizeof(addr));
|
|
|
|
if (error) {
|
|
return XError(
|
|
CR_CONNECTION_ERROR,
|
|
std::string(error.what()) + ", while connecting to " + unix_socket,
|
|
true);
|
|
}
|
|
|
|
m_connected = true;
|
|
return {};
|
|
#else
|
|
return XError(CR_SOCKET_CREATE_ERROR, ER_TEXT_UN_SOCKET_NOT_SUPPORTED, true);
|
|
#endif // defined(HAVE_SYS_UN_H)
|
|
}
|
|
|
|
XError Connection_impl::connect(const std::string &host, const uint16_t port,
|
|
const Internet_protocol ip_mode) {
|
|
m_connection_type = Connection_type::Tcp;
|
|
m_hostname = host;
|
|
|
|
char port_buf[NI_MAXSERV];
|
|
snprintf(port_buf, NI_MAXSERV, "%d", port);
|
|
|
|
addrinfo hints;
|
|
memset(&hints, 0, sizeof(hints));
|
|
hints.ai_socktype = SOCK_STREAM;
|
|
hints.ai_protocol = IPPROTO_TCP;
|
|
hints.ai_family = AF_UNSPEC;
|
|
|
|
if (Internet_protocol::V6 == ip_mode)
|
|
hints.ai_family = AF_INET6;
|
|
else if (Internet_protocol::V4 == ip_mode)
|
|
hints.ai_family = AF_INET;
|
|
|
|
auto addr_future =
|
|
std::async(std::launch::async, [&host, &port_buf, &hints]() {
|
|
auto addr_cleanup = [](addrinfo *addr) {
|
|
if (addr) freeaddrinfo(addr);
|
|
};
|
|
std::shared_ptr<addrinfo> result_list(nullptr, addr_cleanup);
|
|
addrinfo *temp_res_lst;
|
|
auto gai_errno =
|
|
getaddrinfo(host.c_str(), port_buf, &hints, &temp_res_lst);
|
|
if (gai_errno == 0) result_list.reset(temp_res_lst, addr_cleanup);
|
|
return result_list;
|
|
});
|
|
|
|
const auto timeout = m_context->m_connection_config.m_timeout_session_connect;
|
|
const auto delay = std::chrono::milliseconds(
|
|
timeout > 0 ? timeout : std::numeric_limits<std::int32_t>::max());
|
|
if (addr_future.wait_for(delay) == std::future_status::timeout) {
|
|
return XError(CR_X_SESSION_CONNECT_TIMEOUT,
|
|
"Session_connect_timeout limit exceeded", true);
|
|
}
|
|
|
|
auto resolved_addr_list_ptr = addr_future.get();
|
|
if (!resolved_addr_list_ptr)
|
|
return XError(CR_UNKNOWN_HOST, "No such host is known '" + host + "'",
|
|
true);
|
|
|
|
XError error;
|
|
const auto &ns = m_context->m_connection_config.m_network_namespace;
|
|
if (!ns.empty()) {
|
|
return XError(CR_SOCKET_CREATE_ERROR,
|
|
"Network namespace not supported by the platform", true);
|
|
}
|
|
for (const auto *t_res = resolved_addr_list_ptr.get(); t_res;
|
|
t_res = t_res->ai_next) {
|
|
error = connect(reinterpret_cast<sockaddr *>(t_res->ai_addr),
|
|
t_res->ai_addrlen);
|
|
|
|
if (!error) break;
|
|
}
|
|
|
|
if (error) {
|
|
std::string error_description = error.what();
|
|
return XError(CR_CONNECTION_ERROR,
|
|
error_description + " connecting to " + host + ":" + port_buf,
|
|
true);
|
|
}
|
|
|
|
m_connected = true;
|
|
|
|
return XError();
|
|
}
|
|
|
|
XError Connection_impl::connect(sockaddr *addr, const std::size_t addr_size) {
|
|
enum_vio_type type = VIO_TYPE_TCPIP;
|
|
int err = 0;
|
|
const bool is_unix_socket = AF_UNIX == addr->sa_family;
|
|
my_socket s =
|
|
::socket(addr->sa_family, SOCK_STREAM, is_unix_socket ? 0 : IPPROTO_TCP);
|
|
|
|
if (is_unix_socket) type = VIO_TYPE_SOCKET;
|
|
|
|
if (INVALID_SOCKET == s)
|
|
return XError(CR_SOCKET_CREATE_ERROR, ER_TEXT_INVALID_SOCKET);
|
|
|
|
auto vio = vio_new(s, type, 0);
|
|
auto error =
|
|
#ifdef MYSQL8
|
|
vio_socket_connect(vio, addr, static_cast<socklen_t>(addr_size), false,
|
|
details::make_vio_timeout(
|
|
m_context->m_connection_config.m_timeout_connect));
|
|
#else
|
|
vio_socket_connect(vio, addr, static_cast<socklen_t>(addr_size),
|
|
details::make_vio_timeout(
|
|
m_context->m_connection_config.m_timeout_connect));
|
|
#endif
|
|
|
|
if (error) {
|
|
err = socket_errno;
|
|
vio_delete(vio);
|
|
|
|
return get_socket_error(err);
|
|
}
|
|
|
|
m_vio = vio;
|
|
// Enable TCP_NODELAY
|
|
vio_fastsend(m_vio);
|
|
|
|
const auto read_timeout = m_context->m_connection_config.m_timeout_read;
|
|
set_read_timeout(
|
|
details::make_vio_timeout((read_timeout < 0) ? -1 : read_timeout / 1000));
|
|
const auto write_timeout = m_context->m_connection_config.m_timeout_write;
|
|
set_write_timeout(details::make_vio_timeout(
|
|
(write_timeout < 0) ? -1 : write_timeout / 1000));
|
|
|
|
return XError();
|
|
}
|
|
|
|
my_socket Connection_impl::get_socket_fd() {
|
|
if (nullptr == m_vio) return INVALID_SOCKET;
|
|
|
|
return vio_fd(m_vio);
|
|
}
|
|
|
|
std::string Connection_impl::get_socket_error_description(const int error_id) {
|
|
std::string strerr;
|
|
#ifdef _WIN32
|
|
char *s = nullptr;
|
|
if (0 == FormatMessage(
|
|
FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM |
|
|
FORMAT_MESSAGE_IGNORE_INSERTS,
|
|
nullptr, error_id, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),
|
|
(LPSTR)&s, 0, nullptr)) {
|
|
char text[256];
|
|
snprintf(text, sizeof(text), "Error %i", error_id);
|
|
strerr = text;
|
|
} else {
|
|
strerr = s;
|
|
LocalFree(s);
|
|
|
|
while (strerr.size() && (strerr[strerr.size() - 1] == '\n' ||
|
|
strerr[strerr.size() - 1] == '\r')) {
|
|
strerr.erase(strerr.size() - 1);
|
|
}
|
|
|
|
if (strerr.size() && strerr[strerr.size() - 1] == '.') {
|
|
strerr.erase(strerr.size() - 1);
|
|
}
|
|
}
|
|
#else
|
|
strerr = strerror(error_id);
|
|
#endif
|
|
|
|
return strerr;
|
|
}
|
|
|
|
#define SOCKET_EPIPE IF_WIN(ERROR_BROKEN_PIPE, EPIPE)
|
|
#define SOCKET_ECONNABORTED IF_WIN(WSAECONNABORTED, ECONNABORTED)
|
|
|
|
XError Connection_impl::get_socket_error(const int error_id) {
|
|
switch (error_id) {
|
|
#if defined(__APPLE__)
|
|
// OSX return this undocumented error in case of kernel race-condition
|
|
// lets ignore it and next call to any 'IO' function should return correct
|
|
// error
|
|
case EPROTOTYPE:
|
|
return XError();
|
|
#endif // defined(__APPLE__)
|
|
|
|
case SOCKET_ECONNABORTED:
|
|
case SOCKET_ECONNRESET:
|
|
return XError(CR_SERVER_GONE_ERROR, ER_TEXT_SERVER_GONE, true);
|
|
// For now pipe is not fatal, still in future this might change
|
|
// in some cases
|
|
case SOCKET_EPIPE:
|
|
return XError(CR_SERVER_GONE_ERROR, ER_TEXT_SERVER_GONE);
|
|
default:
|
|
return XError(CR_UNKNOWN_ERROR, get_socket_error_description(error_id),
|
|
true);
|
|
}
|
|
}
|
|
|
|
XError Connection_impl::shutdown(const Shutdown_type how_to_shutdown) {
|
|
if (0 !=
|
|
::shutdown(vio_fd(m_vio), details::get_shutdown_consts(how_to_shutdown)))
|
|
return get_socket_error(socket_errno);
|
|
|
|
m_connected = false;
|
|
return XError();
|
|
}
|
|
|
|
XError Connection_impl::write(const uint8_t *data,
|
|
const std::size_t data_length) {
|
|
std::size_t left_data_to_write = data_length;
|
|
const unsigned char *data_to_send = (const unsigned char *)data;
|
|
|
|
if (nullptr == m_vio) return get_socket_error(SOCKET_ECONNRESET);
|
|
|
|
do {
|
|
const int result =
|
|
static_cast<int>(vio_write(m_vio, data_to_send, left_data_to_write));
|
|
|
|
if (-1 == result) {
|
|
const int vio_error = vio_errno(m_vio);
|
|
|
|
if (SOCKET_EWOULDBLOCK == vio_error || vio_was_timeout(m_vio)) {
|
|
return XError{CR_X_WRITE_TIMEOUT, ER_TEXT_CANT_TIMEOUT_WHILE_WRITTING};
|
|
}
|
|
|
|
return get_socket_error(0 != vio_error ? vio_error : SOCKET_ECONNRESET);
|
|
} else if (0 == result) {
|
|
return get_socket_error(SOCKET_ECONNRESET);
|
|
}
|
|
|
|
left_data_to_write -= result;
|
|
data_to_send += result;
|
|
} while (left_data_to_write > 0);
|
|
|
|
return XError();
|
|
}
|
|
|
|
XError Connection_impl::read(uint8_t *data_head,
|
|
const std::size_t data_length) {
|
|
int result = 0;
|
|
std::size_t data_to_send = data_length;
|
|
char *data = reinterpret_cast<char *>(data_head);
|
|
|
|
if (nullptr == m_vio) return get_socket_error(SOCKET_ECONNRESET);
|
|
|
|
do {
|
|
result = static_cast<int>(
|
|
vio_read(m_vio, reinterpret_cast<uint8_t *>(data), data_to_send));
|
|
|
|
if (-1 == result) {
|
|
int vio_error = vio_errno(m_vio);
|
|
|
|
if (SOCKET_EWOULDBLOCK == vio_error || vio_was_timeout(m_vio)) {
|
|
if (!vio_is_connected(m_vio))
|
|
return get_socket_error(SOCKET_ECONNRESET);
|
|
|
|
return XError{CR_X_READ_TIMEOUT, ER_TEXT_CANT_TIMEOUT_WHILE_READING};
|
|
}
|
|
|
|
vio_error = vio_error == 0 ? SOCKET_ECONNRESET : vio_error;
|
|
return get_socket_error(0 != vio_error ? vio_error : SOCKET_ECONNRESET);
|
|
} else if (0 == result) {
|
|
return get_socket_error(SOCKET_ECONNRESET);
|
|
}
|
|
|
|
data_to_send -= result;
|
|
data += result;
|
|
} while (data_to_send != 0);
|
|
|
|
return XError();
|
|
}
|
|
|
|
XError Connection_impl::set_read_timeout(const int deadline_seconds) {
|
|
if (nullptr == m_vio) {
|
|
return XError{CR_INVALID_CONN_HANDLE,
|
|
ER_TEXT_CANT_SET_TIMEOUT_WHEN_NOT_CONNECTED, true};
|
|
}
|
|
|
|
vio_timeout(m_vio, 0, deadline_seconds);
|
|
return {};
|
|
}
|
|
|
|
XError Connection_impl::set_write_timeout(const int deadline_seconds) {
|
|
if (nullptr == m_vio) {
|
|
return XError{CR_INVALID_CONN_HANDLE,
|
|
ER_TEXT_CANT_SET_TIMEOUT_WHEN_NOT_CONNECTED, true};
|
|
}
|
|
|
|
vio_timeout(m_vio, 1, deadline_seconds);
|
|
return {};
|
|
}
|
|
|
|
const XConnection::State &Connection_impl::state() {
|
|
m_state.reset(new details::Connection_state(
|
|
m_vio, false, m_ssl_active, m_connected,
|
|
m_connection_type));
|
|
return *m_state;
|
|
}
|
|
|
|
void Connection_impl::close() {
|
|
if (m_vio) {
|
|
::closesocket(vio_fd(m_vio));
|
|
vio_delete(m_vio);
|
|
m_connected = false;
|
|
m_ssl_active = false;
|
|
m_vio = nullptr;
|
|
}
|
|
}
|
|
|
|
} // namespace xcl
|