225 lines
		
	
	
		
			5.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			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: 2025-02-16
 | |
|  *
 | |
|  * 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();
 | 
