Files
MaxScale/server/core/internal/session.hh
2019-12-18 13:25:03 +02:00

225 lines
5.8 KiB
C++

/*
* Copyright (c) 2018 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: 2023-12-18
*
* 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.
*/
#pragma once
#include <maxscale/ccdefs.hh>
#include <deque>
#include <string>
#include <unordered_map>
#include <unordered_set>
#include <vector>
#include <maxscale/buffer.hh>
#include <maxscale/session.hh>
#include <maxscale/resultset.hh>
#include <maxscale/utils.hh>
#include "filter.hh"
#include "service.hh"
#define SESSION_STATS_INIT {0}
#define MXS_DOWNSTREAM_INIT {0}
#define MXS_UPSTREAM_INIT {0}
#define SESSION_FILTER_INIT {0}
#define SESSION_PROTOCOL(x, type) DCB_PROTOCOL((x)->client_dcb, type)
MXS_BEGIN_DECLS
/**
* Filter type for the sessionGetList call
*/
typedef enum
{
SESSION_LIST_ALL,
SESSION_LIST_CONNECTION
} SESSIONLISTFILTER;
int session_isvalid(MXS_SESSION*);
/**
* Link a session to a backend DCB.
*
* @param session The session to link with the dcb
* @param dcb The backend DCB to be linked
*/
void session_link_backend_dcb(MXS_SESSION* session, DCB* dcb);
/**
* Unlink a session to a backend DCB.
*
* @param session The session to unlink with the dcb
* @param dcb The backend DCB to be unlinked
*/
void session_unlink_backend_dcb(MXS_SESSION* session, DCB* dcb);
void printAllSessions();
void printSession(MXS_SESSION*);
void dprintSessionList(DCB* pdcb);
void dprintAllSessions(DCB*);
void dprintSession(DCB*, MXS_SESSION*);
void dListSessions(DCB*);
/**
* @brief Get a session reference
*
* This creates an additional reference to a session which allows it to live
* as long as it is needed.
*
* @param session Session reference to get
* @return Reference to a MXS_SESSION
*
* @note The caller must free the session reference by calling session_put_ref
*/
MXS_SESSION* session_get_ref(MXS_SESSION* sessoin);
MXS_END_DECLS
namespace maxscale
{
typedef struct SESSION_VARIABLE
{
session_variable_handler_t handler;
void* context;
} SESSION_VARIABLE;
typedef std::unordered_map<std::string, SESSION_VARIABLE> SessionVarsByName;
typedef std::unordered_set<DCB*> DCBSet;
// Class that holds the session specific filter data
class SessionFilter
{
public:
SessionFilter(const SFilterDef& f)
: filter(f)
, instance(nullptr)
, session(nullptr)
{
}
SFilterDef filter;
MXS_FILTER* instance;
MXS_FILTER_SESSION* session;
};
class Session : public MXS_SESSION
{
public:
class QueryInfo
{
public:
QueryInfo(const std::shared_ptr<GWBUF>& sQuery);
json_t* as_json() const;
bool complete() const
{
return m_complete;
}
const std::shared_ptr<GWBUF>& query() const
{
return m_sQuery;
}
timespec time_completed() const
{
return m_completed;
}
void book_server_response(SERVER* pServer, bool final_response);
void book_as_complete();
void reset_server_bookkeeping();
struct ServerInfo
{
SERVER* pServer;
timespec processed;
};
private:
std::shared_ptr<GWBUF> m_sQuery; /*< The packet, COM_QUERY *or* something else. */
timespec m_received; /*< When was it received. */
timespec m_completed; /*< When was it completed. */
std::vector<ServerInfo> m_server_infos; /*< When different servers responded. */
bool m_complete = false; /*< Is this information complete? */
};
typedef std::deque<QueryInfo> QueryInfos;
using Log = std::deque<std::string>;
using FilterList = std::vector<SessionFilter>;
Session(const SListener& listener);
~Session();
// Links a client DCB to a session
void set_client_dcb(DCB* dcb);
bool setup_filters(Service* service);
const FilterList& get_filters() const
{
return m_filters;
}
bool add_variable(const char* name, session_variable_handler_t handler, void* context);
char* set_variable_value(const char* name_begin,
const char* name_end,
const char* value_begin,
const char* value_end);
bool remove_variable(const char* name, void** context);
void retain_statement(GWBUF* pBuffer);
void dump_statements() const;
void book_server_response(SERVER* pServer, bool final_response);
void book_last_as_complete();
void reset_server_bookkeeping();
void append_session_log(std::string);
void dump_session_log();
json_t* queries_as_json() const;
json_t* log_as_json() const;
void link_backend_dcb(DCB* dcb)
{
mxb_assert(m_dcb_set.count(dcb) == 0);
m_dcb_set.insert(dcb);
}
void unlink_backend_dcb(DCB* dcb)
{
mxb_assert(m_dcb_set.count(dcb) == 1);
m_dcb_set.erase(dcb);
}
const DCBSet& dcb_set() const
{
return m_dcb_set;
}
private:
FilterList m_filters;
SessionVarsByName m_variables;
QueryInfos m_last_queries; /*< The N last queries by the client */
int m_current_query = -1; /*< The index of the current query */
DCBSet m_dcb_set; /*< Set of associated backend DCBs */
uint32_t m_retain_last_statements; /*< How many statements be retained */
Log m_log; /*< Session specific in-memory log */
};
}
std::unique_ptr<ResultSet> sessionGetList();