
The SERVER_REF and DCB members of the Backend class are now private. Access to the stored SERVER_REF is provided with the backend() function. No accompanying setter function is provided as the backend server should not change during the lifetime of the session. The creation of the internal DCB is hidden behind the connect() function. It simplifies the process of connecting to a server by removing the need to manually do the bookkeeping of the server reference connection counts. Access to the DCB is provided by the dcb() function. The closing of the backend is done with the close() function which contains the code that was previously in closeSession. If the backend isn't closed when the destructor is called, it will be done automatically. This should prevent connection leakage. The pending command queues and the methods used to write them are now also internal to the backends. They are simple wrappers around dcb->func.write and the interfaces provided by the Buffer class. The mapping command queue is still public. It needs to be combined with the generic command queue.
195 lines
3.9 KiB
C++
195 lines
3.9 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: 2019-07-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 "schemarouter.hh"
|
|
|
|
#include <maxscale/protocol/mysql.h>
|
|
|
|
using namespace schemarouter;
|
|
|
|
Backend::Backend(SERVER_REF *ref):
|
|
m_closed(false),
|
|
m_backend(ref),
|
|
m_dcb(NULL),
|
|
m_map_queue(NULL),
|
|
m_mapped(false),
|
|
m_num_mapping_eof(0),
|
|
m_num_result_wait(0),
|
|
m_state(0)
|
|
{
|
|
}
|
|
|
|
Backend::~Backend()
|
|
{
|
|
ss_dassert(m_closed);
|
|
|
|
if (!m_closed)
|
|
{
|
|
close();
|
|
}
|
|
|
|
gwbuf_free(m_map_queue);
|
|
}
|
|
|
|
void Backend::close()
|
|
{
|
|
if (!m_closed)
|
|
{
|
|
m_closed = true;
|
|
|
|
if (BREF_IS_IN_USE(this))
|
|
{
|
|
CHK_DCB(m_dcb);
|
|
|
|
/** Clean operation counter in bref and in SERVER */
|
|
while (BREF_IS_WAITING_RESULT(this))
|
|
{
|
|
clear_state(BREF_WAITING_RESULT);
|
|
}
|
|
clear_state(BREF_IN_USE);
|
|
set_state(BREF_CLOSED);
|
|
|
|
dcb_close(m_dcb);
|
|
|
|
/** decrease server current connection counters */
|
|
atomic_add(&m_backend->connections, -1);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
ss_dassert(false);
|
|
}
|
|
}
|
|
|
|
bool Backend::execute_sescmd()
|
|
{
|
|
if (BREF_IS_CLOSED(this) || m_session_commands.size() == 0)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
CHK_DCB(m_dcb);
|
|
|
|
int rc = 0;
|
|
|
|
/** Return if there are no pending ses commands */
|
|
if (m_session_commands.size() == 0)
|
|
{
|
|
MXS_INFO("Cursor had no pending session commands.");
|
|
return false;
|
|
}
|
|
|
|
SessionCommandList::iterator iter = m_session_commands.begin();
|
|
GWBUF *buffer = iter->copy_buffer().release();
|
|
|
|
switch (iter->get_command())
|
|
{
|
|
case MYSQL_COM_CHANGE_USER:
|
|
/** This makes it possible to handle replies correctly */
|
|
gwbuf_set_type(buffer, GWBUF_TYPE_SESCMD);
|
|
rc = m_dcb->func.auth(m_dcb, NULL, m_dcb->session, buffer);
|
|
break;
|
|
|
|
case MYSQL_COM_QUERY:
|
|
default:
|
|
/**
|
|
* Mark session command buffer, it triggers writing
|
|
* MySQL command to protocol
|
|
*/
|
|
gwbuf_set_type(buffer, GWBUF_TYPE_SESCMD);
|
|
rc = m_dcb->func.write(m_dcb, buffer);
|
|
break;
|
|
}
|
|
|
|
return rc == 1;
|
|
}
|
|
|
|
void Backend::clear_state(enum bref_state state)
|
|
{
|
|
if (state != BREF_WAITING_RESULT)
|
|
{
|
|
m_state &= ~state;
|
|
}
|
|
else
|
|
{
|
|
/** Decrease global operation count */
|
|
ss_debug(int prev2 = )atomic_add(&m_backend->server->stats.n_current_ops, -1);
|
|
ss_dassert(prev2 > 0);
|
|
}
|
|
}
|
|
|
|
void Backend::set_state(enum bref_state state)
|
|
{
|
|
if (state != BREF_WAITING_RESULT)
|
|
{
|
|
m_state |= state;
|
|
}
|
|
else
|
|
{
|
|
/** Increase global operation count */
|
|
ss_debug(int prev2 = )atomic_add(&m_backend->server->stats.n_current_ops, 1);
|
|
ss_dassert(prev2 >= 0);
|
|
}
|
|
}
|
|
|
|
SERVER_REF* Backend::backend() const
|
|
{
|
|
return m_backend;
|
|
}
|
|
|
|
bool Backend::connect(MXS_SESSION* session)
|
|
{
|
|
bool rval = false;
|
|
|
|
if ((m_dcb = dcb_connect(m_backend->server, session, m_backend->server->protocol)))
|
|
{
|
|
m_state = BREF_IN_USE;
|
|
atomic_add(&m_backend->connections, 1);
|
|
rval = true;
|
|
}
|
|
|
|
return rval;
|
|
}
|
|
|
|
DCB* Backend::dcb() const
|
|
{
|
|
return m_dcb;
|
|
}
|
|
|
|
bool Backend::write(GWBUF* buffer)
|
|
{
|
|
return m_dcb->func.write(m_dcb, buffer) != 0;
|
|
}
|
|
|
|
void Backend::store_command(GWBUF* buffer)
|
|
{
|
|
m_pending_cmd.reset(buffer);
|
|
}
|
|
|
|
bool Backend::write_stored_command()
|
|
{
|
|
bool rval = false;
|
|
|
|
if (m_pending_cmd.length())
|
|
{
|
|
rval = write(m_pending_cmd.release());
|
|
|
|
if (!rval)
|
|
{
|
|
MXS_ERROR("Routing of pending query failed.");
|
|
}
|
|
}
|
|
|
|
return rval;
|
|
}
|