
The id has now been moved from mxs::Worker to mxs::RoutingWorker and the implications are felt in many places. The primary need for the id was to be able to access worker specfic data, maintained outside of a routing worker, when given a worker (the id is used to index into an array). Slightly related to that was the need to be able to iterate over all workers. That obviously implies some kind of collection. That causes all sorts of issues if there is a need for being able to create and destroy a worker at runtime. With the id removed from mxs::Worker all those issues are gone, and its perfectly ok to create and destory mxs::Workers as needed. Further, while there is a need to broadcast a particular message to all _routing_ workers, it hardly makes sense to broadcast a particular message too _all_ workers. Consequently, only routing workers are kept in a collection and all static member functions dealing with all workers (e.g. broadcast) have now been moved to mxs::RoutingWorker. Now, instead of passing the id around we instead deal directly with the worker pointer. Later the data in all those external arrays will be moved into mxs::[Worker|RoutingWorker] so that worker related data is maintained in exactly one place.
297 lines
7.1 KiB
C++
297 lines
7.1 KiB
C++
/*
|
|
* Copyright (c) 2016 MariaDB Corporation Ab
|
|
*
|
|
* Use of this software is governed by the Business Source License included
|
|
* in the LICENSE.TXT file and at www.mariadb.com/bsl11.
|
|
*
|
|
* Change Date: 2022-01-01
|
|
*
|
|
* On the date above, in accordance with the Business Source License, use
|
|
* of this software will be governed by version 2 or later of the General
|
|
* Public License.
|
|
*/
|
|
|
|
#include <maxscale/protocol/mariadb_client.hh>
|
|
#include <maxscale/utils.h>
|
|
|
|
// TODO: Find a way to cleanly expose this
|
|
#include "../../../core/internal/routingworker.hh"
|
|
|
|
#ifdef EPOLLRDHUP
|
|
#define ERROR_EVENTS (EPOLLRDHUP | EPOLLHUP | EPOLLERR)
|
|
#else
|
|
#define ERROR_EVENTS (EPOLLHUP | EPOLLERR)
|
|
#endif
|
|
|
|
static const uint32_t poll_events = EPOLLIN | EPOLLOUT | EPOLLET | ERROR_EVENTS;
|
|
|
|
LocalClient::LocalClient(MXS_SESSION* session, int fd):
|
|
m_state(VC_WAITING_HANDSHAKE),
|
|
m_sock(fd),
|
|
m_expected_bytes(0),
|
|
m_client({}),
|
|
m_protocol({}),
|
|
m_self_destruct(false)
|
|
{
|
|
MXS_POLL_DATA::handler = LocalClient::poll_handler;
|
|
MySQLProtocol* client = (MySQLProtocol*)session->client_dcb->protocol;
|
|
m_protocol.charset = client->charset;
|
|
m_protocol.client_capabilities = client->client_capabilities;
|
|
m_protocol.extra_capabilities = client->extra_capabilities;
|
|
gw_get_shared_session_auth_info(session->client_dcb, &m_client);
|
|
}
|
|
|
|
LocalClient::~LocalClient()
|
|
{
|
|
if (m_state != VC_ERROR)
|
|
{
|
|
close();
|
|
}
|
|
}
|
|
|
|
bool LocalClient::queue_query(GWBUF* buffer)
|
|
{
|
|
GWBUF* my_buf = NULL;
|
|
|
|
if (m_state != VC_ERROR && (my_buf = gwbuf_deep_clone(buffer)))
|
|
{
|
|
m_queue.push_back(my_buf);
|
|
|
|
if (m_state == VC_OK)
|
|
{
|
|
drain_queue();
|
|
}
|
|
}
|
|
|
|
return my_buf != NULL;
|
|
}
|
|
|
|
void LocalClient::self_destruct()
|
|
{
|
|
GWBUF* buffer = mysql_create_com_quit(NULL, 0);
|
|
queue_query(buffer);
|
|
gwbuf_free(buffer);
|
|
m_self_destruct = true;
|
|
}
|
|
|
|
void LocalClient::close()
|
|
{
|
|
mxs::Worker* worker = mxs::Worker::get_current();
|
|
ss_dassert(worker);
|
|
worker->remove_fd(m_sock);
|
|
::close(m_sock);
|
|
}
|
|
|
|
void LocalClient::error()
|
|
{
|
|
if (m_state != VC_ERROR)
|
|
{
|
|
close();
|
|
m_state = VC_ERROR;
|
|
}
|
|
}
|
|
|
|
void LocalClient::process(uint32_t events)
|
|
{
|
|
|
|
if (events & EPOLLIN)
|
|
{
|
|
GWBUF* buf = read_complete_packet();
|
|
|
|
if (buf)
|
|
{
|
|
if (m_state == VC_WAITING_HANDSHAKE)
|
|
{
|
|
if (gw_decode_mysql_server_handshake(&m_protocol, GWBUF_DATA(buf) + MYSQL_HEADER_LEN) == 0)
|
|
{
|
|
GWBUF* response = gw_generate_auth_response(&m_client, &m_protocol, false, false, 0);
|
|
m_queue.push_front(response);
|
|
m_state = VC_RESPONSE_SENT;
|
|
}
|
|
else
|
|
{
|
|
error();
|
|
}
|
|
}
|
|
else if (m_state == VC_RESPONSE_SENT)
|
|
{
|
|
if (mxs_mysql_is_ok_packet(buf))
|
|
{
|
|
m_state = VC_OK;
|
|
}
|
|
else
|
|
{
|
|
error();
|
|
}
|
|
}
|
|
|
|
gwbuf_free(buf);
|
|
}
|
|
}
|
|
|
|
if (events & EPOLLOUT)
|
|
{
|
|
/** Queue is drained */
|
|
}
|
|
|
|
if (events & ERROR_EVENTS)
|
|
{
|
|
error();
|
|
}
|
|
|
|
if (m_queue.size() && m_state != VC_ERROR && m_state != VC_WAITING_HANDSHAKE)
|
|
{
|
|
drain_queue();
|
|
}
|
|
else if (m_state == VC_ERROR && m_self_destruct)
|
|
{
|
|
delete this;
|
|
}
|
|
}
|
|
|
|
GWBUF* LocalClient::read_complete_packet()
|
|
{
|
|
GWBUF* rval = NULL;
|
|
|
|
while (true)
|
|
{
|
|
uint8_t buffer[1024];
|
|
int rc = read(m_sock, buffer, sizeof(buffer));
|
|
|
|
if (rc == -1)
|
|
{
|
|
if (errno != EAGAIN && errno != EWOULDBLOCK)
|
|
{
|
|
MXS_ERROR("Failed to read from backend: %d, %s", errno, mxs_strerror(errno));
|
|
error();
|
|
}
|
|
break;
|
|
}
|
|
|
|
mxs::Buffer chunk(buffer, rc);
|
|
m_partial.append(chunk);
|
|
size_t len = m_partial.length();
|
|
|
|
if (m_expected_bytes == 0 && len >= 3)
|
|
{
|
|
mxs::Buffer::iterator iter = m_partial.begin();
|
|
m_expected_bytes = MYSQL_HEADER_LEN;
|
|
m_expected_bytes += *iter++;
|
|
m_expected_bytes += (*iter++ << 8);
|
|
m_expected_bytes += (*iter++ << 16);
|
|
}
|
|
|
|
if (len >= m_expected_bytes)
|
|
{
|
|
/** Read complete packet. Reset expected byte count and make
|
|
* the buffer contiguous. */
|
|
m_expected_bytes = 0;
|
|
m_partial.make_contiguous();
|
|
rval = m_partial.release();
|
|
break;
|
|
}
|
|
|
|
}
|
|
|
|
return rval;
|
|
}
|
|
|
|
void LocalClient::drain_queue()
|
|
{
|
|
bool more = true;
|
|
|
|
while (m_queue.size() && more)
|
|
{
|
|
/** Grab a buffer from the queue */
|
|
GWBUF* buf = m_queue.front().release();
|
|
m_queue.pop_front();
|
|
|
|
while (buf)
|
|
{
|
|
int rc = write(m_sock, GWBUF_DATA(buf), GWBUF_LENGTH(buf));
|
|
|
|
if (rc > 0)
|
|
{
|
|
buf = gwbuf_consume(buf, rc);
|
|
}
|
|
else
|
|
{
|
|
if (rc == -1 && errno != EAGAIN && errno != EWOULDBLOCK)
|
|
{
|
|
MXS_ERROR("Failed to write to backend: %d, %s", errno, mxs_strerror(errno));
|
|
error();
|
|
}
|
|
|
|
m_queue.push_front(buf);
|
|
more = false;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
uint32_t LocalClient::poll_handler(struct mxs_poll_data* data, void* worker, uint32_t events)
|
|
{
|
|
LocalClient* client = static_cast<LocalClient*>(data);
|
|
client->process(events);
|
|
return 0;
|
|
}
|
|
|
|
LocalClient* LocalClient::create(MXS_SESSION* session, const char* ip, uint64_t port)
|
|
{
|
|
LocalClient* rval = NULL;
|
|
sockaddr_storage addr;
|
|
int fd = open_network_socket(MXS_SOCKET_NETWORK, &addr, ip, port);
|
|
|
|
if (fd > 0 && (connect(fd, (struct sockaddr*)&addr, sizeof(addr)) == 0 || errno == EINPROGRESS))
|
|
{
|
|
LocalClient* relay = new (std::nothrow) LocalClient(session, fd);
|
|
|
|
if (relay)
|
|
{
|
|
mxs::Worker* worker = mxs::Worker::get_current();
|
|
|
|
if (worker->add_fd(fd, poll_events, (MXS_POLL_DATA*)relay))
|
|
{
|
|
rval = relay;
|
|
}
|
|
else
|
|
{
|
|
relay->m_state = VC_ERROR;
|
|
delete rval;
|
|
rval = NULL;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (rval == NULL && fd > 0)
|
|
{
|
|
::close(fd);
|
|
}
|
|
return rval;
|
|
}
|
|
|
|
LocalClient* LocalClient::create(MXS_SESSION* session, SERVICE* service)
|
|
{
|
|
LocalClient* rval = NULL;
|
|
LISTENER_ITERATOR iter;
|
|
|
|
for (SERV_LISTENER* listener = listener_iterator_init(service, &iter);
|
|
listener; listener = listener_iterator_next(&iter))
|
|
{
|
|
if (listener->port > 0)
|
|
{
|
|
/** Pick the first network listener */
|
|
rval = create(session, "127.0.0.1", service->ports->port);
|
|
break;
|
|
}
|
|
}
|
|
|
|
return rval;
|
|
}
|
|
|
|
LocalClient* LocalClient::create(MXS_SESSION* session, SERVER* server)
|
|
{
|
|
return create(session, server->address, server->port);
|
|
}
|