Add router template
Using the class RouterSession and the template Router, a router module can be defined. The way they are intended to be used are as follows: class MyRouterSession : public maxscale::RouterSession { ... }; class MyRouter : public maxscale::Router<MyRouter, MyRouterSession> { ... } ... extern "C" MXS_MODULE* MXS_CREATE_MODULE() { static MXS_MODULE module = { ... &MyRouter::s_object, ... }; return &module; }
This commit is contained in:
248
include/maxscale/router.hh
Normal file
248
include/maxscale/router.hh
Normal file
@ -0,0 +1,248 @@
|
||||
#pragma once
|
||||
/*
|
||||
* 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 <maxscale/cppdefs.hh>
|
||||
#include <maxscale/router.h>
|
||||
|
||||
namespace maxscale
|
||||
{
|
||||
|
||||
/**
|
||||
* @class RouterSession router.hh <maxscale/router.hh>
|
||||
*
|
||||
* RouterSession is a base class for router sessions. A concrete router session
|
||||
* class should be derived from this class and override all relevant functions.
|
||||
*
|
||||
* Note that even though this class is intended to be derived from, no functions
|
||||
* are virtual. That is by design, as the class will be used in a context where
|
||||
* the concrete class is known. That is, there is no need for the virtual mechanism.
|
||||
*/
|
||||
class RouterSession : public MXS_ROUTER_SESSION
|
||||
{
|
||||
public:
|
||||
/**
|
||||
* The RouterSession instance will be deleted when a client session
|
||||
* has terminated. Will be called only after @c close() has been called.
|
||||
*/
|
||||
~RouterSession();
|
||||
|
||||
/**
|
||||
* Called when a client session has been closed.
|
||||
*/
|
||||
void close();
|
||||
|
||||
/**
|
||||
* Called when a packet being is routed to the backend. The router should
|
||||
* forward the packet to the appropriate server(s).
|
||||
*
|
||||
* @param pPacket A client packet.
|
||||
*/
|
||||
int32_t routeQuery(GWBUF* pPacket);
|
||||
|
||||
/**
|
||||
* Called when a packet is routed to the client. The router should
|
||||
* forward the packet to the client using `MXS_SESSION_ROUTE_REPLY`.
|
||||
*
|
||||
* @param pPacket A client packet.
|
||||
* @param pBackend The backend the packet is coming from.
|
||||
*/
|
||||
void clientReply(GWBUF* pPacket, DCB* pBackend);
|
||||
|
||||
/**
|
||||
*
|
||||
* @param pMessage The rror message.
|
||||
* @param pProblem The DCB on which the error occurred.
|
||||
* @param action The context.
|
||||
* @param pSuccess On output, if false, the session will be terminated.
|
||||
*/
|
||||
void handleError(GWBUF* pMessage,
|
||||
DCB* pProblem,
|
||||
mxs_error_action_t action,
|
||||
bool* pSuccess);
|
||||
|
||||
protected:
|
||||
RouterSession(MXS_SESSION* pSession);
|
||||
|
||||
protected:
|
||||
MXS_SESSION* m_pSession; /*< The MXS_SESSION this router session is associated with. */
|
||||
};
|
||||
|
||||
|
||||
/**
|
||||
* @class Router router.hh <maxscale/router.hh>
|
||||
*
|
||||
* An instantiation of the Router template is used for creating a router.
|
||||
* Router is an example of the "Curiously recurring template pattern"
|
||||
* https://en.wikipedia.org/wiki/Curiously_recurring_template_pattern
|
||||
* that is used for compile time polymorfism.
|
||||
*
|
||||
* The typical way for using the template is as follows:
|
||||
*
|
||||
* @code
|
||||
* class MyRouterSession : public maxscale::RouterSession
|
||||
* {
|
||||
* // Override the relevant functions.
|
||||
* };
|
||||
*
|
||||
* class MyRouter : public maxscale::Router<MyRouter, MyRouterSession>
|
||||
* {
|
||||
* public:
|
||||
* static MyRouter* create(SERVICE* pService, char** pzOptions);
|
||||
*
|
||||
* MyRouterSession* newSession(MXS_SESSION* pSession);
|
||||
*
|
||||
* void diagnostics(DCB* pDcb);
|
||||
* uint64_t getCapabilities();
|
||||
* };
|
||||
* @endcode
|
||||
*
|
||||
* The concrete router class must implement the methods @c create, @c newSession,
|
||||
* @c diagnostics and @c getCapabilities, with the prototypes as shown above.
|
||||
*
|
||||
* The plugin function @c GetModuleObject is then implemented as follows:
|
||||
*
|
||||
* @code
|
||||
* extern "C" MXS_MODULE* MXS_CREATE_MODULE()
|
||||
* {
|
||||
* static MXS_MODULE module_object =
|
||||
* {
|
||||
* ...
|
||||
* &MyRouter::s_object,
|
||||
* ...
|
||||
* };
|
||||
*
|
||||
* return &module_object;
|
||||
* }
|
||||
* @endcode
|
||||
*/
|
||||
template<class RouterType, class RouterSessionType>
|
||||
class Router : public MXS_ROUTER
|
||||
{
|
||||
public:
|
||||
static MXS_ROUTER* createInstance(SERVICE* pService, char** pzOptions)
|
||||
{
|
||||
RouterType* pRouter = NULL;
|
||||
|
||||
MXS_EXCEPTION_GUARD(pRouter = RouterType::create(pService, pzOptions));
|
||||
|
||||
return pRouter;
|
||||
}
|
||||
|
||||
static MXS_ROUTER_SESSION* newSession(MXS_ROUTER* pInstance, MXS_SESSION* pSession)
|
||||
{
|
||||
RouterType* pRouter = static_cast<RouterType*>(pInstance);
|
||||
RouterSessionType* pRouter_session;
|
||||
|
||||
MXS_EXCEPTION_GUARD(pRouter_session = pRouter->newSession(pSession));
|
||||
|
||||
return pRouter_session;
|
||||
}
|
||||
|
||||
static void closeSession(MXS_ROUTER*, MXS_ROUTER_SESSION* pData)
|
||||
{
|
||||
RouterSessionType* pRouter_session = static_cast<RouterSessionType*>(pData);
|
||||
|
||||
MXS_EXCEPTION_GUARD(pRouter_session->close());
|
||||
}
|
||||
|
||||
static void freeSession(MXS_ROUTER*, MXS_ROUTER_SESSION* pData)
|
||||
{
|
||||
RouterSessionType* pRouter_session = static_cast<RouterSessionType*>(pData);
|
||||
|
||||
MXS_EXCEPTION_GUARD(delete pRouter_session);
|
||||
}
|
||||
|
||||
static int32_t routeQuery(MXS_ROUTER*, MXS_ROUTER_SESSION* pData, GWBUF* pPacket)
|
||||
{
|
||||
RouterSessionType* pRouter_session = static_cast<RouterSessionType*>(pData);
|
||||
|
||||
int32_t rv = 0;
|
||||
MXS_EXCEPTION_GUARD(rv = pRouter_session->routeQuery(pPacket));
|
||||
|
||||
return rv;
|
||||
}
|
||||
|
||||
static void diagnostics(MXS_ROUTER* pInstance, DCB* pDcb)
|
||||
{
|
||||
RouterType* pRouter = static_cast<RouterType*>(pInstance);
|
||||
|
||||
MXS_EXCEPTION_GUARD(pRouter->diagnostics(pDcb));
|
||||
}
|
||||
|
||||
static void clientReply(MXS_ROUTER*, MXS_ROUTER_SESSION* pData, GWBUF* pPacket, DCB* pBackend)
|
||||
{
|
||||
RouterSessionType* pRouter_session = static_cast<RouterSessionType*>(pData);
|
||||
|
||||
MXS_EXCEPTION_GUARD(pRouter_session->clientReply(pPacket, pBackend));
|
||||
}
|
||||
|
||||
static void handleError(MXS_ROUTER* pInstance,
|
||||
MXS_ROUTER_SESSION* pData,
|
||||
GWBUF* pMessage,
|
||||
DCB* pProblem,
|
||||
mxs_error_action_t action,
|
||||
bool* pSuccess)
|
||||
{
|
||||
RouterSessionType* pRouter_session = static_cast<RouterSessionType*>(pData);
|
||||
|
||||
MXS_EXCEPTION_GUARD(pRouter_session->handleError(pMessage, pProblem, action, pSuccess));
|
||||
}
|
||||
|
||||
static uint64_t getCapabilities(MXS_ROUTER* pInstance)
|
||||
{
|
||||
uint64_t rv = 0;
|
||||
|
||||
RouterType* pRouter = static_cast<RouterType*>(pInstance);
|
||||
|
||||
MXS_EXCEPTION_GUARD(rv = pRouter->getCapabilities());
|
||||
|
||||
return rv;
|
||||
}
|
||||
|
||||
static void destroyInstance(MXS_ROUTER* pInstance)
|
||||
{
|
||||
RouterType* pRouter = static_cast<RouterType*>(pInstance);
|
||||
|
||||
MXS_EXCEPTION_GUARD(delete pRouter);
|
||||
}
|
||||
|
||||
static MXS_ROUTER_OBJECT s_object;
|
||||
|
||||
protected:
|
||||
Router(SERVICE *pService)
|
||||
: m_pService(pService)
|
||||
{
|
||||
}
|
||||
|
||||
SERVICE* m_pService;
|
||||
};
|
||||
|
||||
|
||||
template<class RouterType, class RouterSessionType>
|
||||
MXS_ROUTER_OBJECT Router<RouterType, RouterSessionType>::s_object =
|
||||
{
|
||||
&Router<RouterType, RouterSessionType>::createInstance,
|
||||
&Router<RouterType, RouterSessionType>::newSession,
|
||||
&Router<RouterType, RouterSessionType>::closeSession,
|
||||
&Router<RouterType, RouterSessionType>::freeSession,
|
||||
&Router<RouterType, RouterSessionType>::routeQuery,
|
||||
&Router<RouterType, RouterSessionType>::diagnostics,
|
||||
&Router<RouterType, RouterSessionType>::clientReply,
|
||||
&Router<RouterType, RouterSessionType>::handleError,
|
||||
&Router<RouterType, RouterSessionType>::getCapabilities,
|
||||
&Router<RouterType, RouterSessionType>::destroyInstance,
|
||||
};
|
||||
|
||||
|
||||
}
|
Reference in New Issue
Block a user