1067 lines
		
	
	
		
			35 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			1067 lines
		
	
	
		
			35 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: 2024-10-14
 | |
|  *
 | |
|  * 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 <map>
 | |
| #include <maxscale/ccdefs.hh>
 | |
| #include <maxbase/jansson.h>
 | |
| #include <maxscale/buffer.hh>
 | |
| 
 | |
| #define MXS_QUERY_CLASSIFIER_VERSION {3, 0, 0}
 | |
| 
 | |
| /**
 | |
|  * qc_init_kind_t specifies what kind of initialization should be performed.
 | |
|  */
 | |
| enum qc_init_kind_t
 | |
| {
 | |
|     QC_INIT_SELF   = 0x01,  /*< Initialize/finalize the query classifier itself. */
 | |
|     QC_INIT_PLUGIN = 0x02,  /*< Initialize/finalize the plugin. */
 | |
|     QC_INIT_BOTH   = 0x03
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * qc_option_t defines options that affect the classification.
 | |
|  */
 | |
| enum qc_option_t
 | |
| {
 | |
|     QC_OPTION_STRING_ARG_AS_FIELD = (1 << 0),   /*< Report a string argument to a function as a field. */
 | |
|     QC_OPTION_STRING_AS_FIELD     = (1 << 1),   /*< Report strings as fields. */
 | |
| };
 | |
| 
 | |
| const uint32_t QC_OPTION_MASK = QC_OPTION_STRING_ARG_AS_FIELD | QC_OPTION_STRING_AS_FIELD;
 | |
| 
 | |
| /**
 | |
|  * qc_sql_mode_t specifies what should be assumed of the statements
 | |
|  * that will be parsed.
 | |
|  */
 | |
| enum qc_sql_mode_t
 | |
| {
 | |
|     QC_SQL_MODE_DEFAULT,    /*< Assume the statements are MariaDB SQL. */
 | |
|     QC_SQL_MODE_ORACLE      /*< Assume the statements are PL/SQL. */
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * @c qc_collect_info_t specifies what information should be collected during parsing.
 | |
|  */
 | |
| enum qc_collect_info_t
 | |
| {
 | |
|     QC_COLLECT_ESSENTIALS = 0x00,   /*< Collect only the base minimum. */
 | |
|     QC_COLLECT_TABLES     = 0x01,   /*< Collect table names. */
 | |
|     QC_COLLECT_DATABASES  = 0x02,   /*< Collect database names. */
 | |
|     QC_COLLECT_FIELDS     = 0x04,   /*< Collect field information. */
 | |
|     QC_COLLECT_FUNCTIONS  = 0x08,   /*< Collect function information. */
 | |
| 
 | |
|     QC_COLLECT_ALL = (QC_COLLECT_TABLES | QC_COLLECT_DATABASES | QC_COLLECT_FIELDS | QC_COLLECT_FUNCTIONS)
 | |
| };
 | |
| /**
 | |
|  * qc_query_type_t defines bits that provide information about a
 | |
|  * particular statement.
 | |
|  *
 | |
|  * Note that more than one bit may be set for a single statement.
 | |
|  */
 | |
| enum qc_query_type_t
 | |
| {
 | |
|     QUERY_TYPE_UNKNOWN       = 0x000000,    /*< Initial value, can't be tested bitwisely */
 | |
|     QUERY_TYPE_LOCAL_READ    = 0x000001,    /*< Read non-database data, execute in MaxScale:any */
 | |
|     QUERY_TYPE_READ          = 0x000002,    /*< Read database data:any */
 | |
|     QUERY_TYPE_WRITE         = 0x000004,    /*< Master data will be  modified:master */
 | |
|     QUERY_TYPE_MASTER_READ   = 0x000008,    /*< Read from the master:master */
 | |
|     QUERY_TYPE_SESSION_WRITE = 0x000010,    /*< Session data will be modified:master or all */
 | |
|     QUERY_TYPE_USERVAR_WRITE = 0x000020,    /*< Write a user variable:master or all */
 | |
|     QUERY_TYPE_USERVAR_READ  = 0x000040,    /*< Read a user variable:master or any */
 | |
|     QUERY_TYPE_SYSVAR_READ   = 0x000080,    /*< Read a system variable:master or any */
 | |
|     /** Not implemented yet */
 | |
|     // QUERY_TYPE_SYSVAR_WRITE       = 0x000100, /*< Write a system variable:master or all */
 | |
|     QUERY_TYPE_GSYSVAR_READ       = 0x000200,   /*< Read global system variable:master or any */
 | |
|     QUERY_TYPE_GSYSVAR_WRITE      = 0x000400,   /*< Write global system variable:master or all */
 | |
|     QUERY_TYPE_BEGIN_TRX          = 0x000800,   /*< BEGIN or START TRANSACTION */
 | |
|     QUERY_TYPE_ENABLE_AUTOCOMMIT  = 0x001000,   /*< SET autocommit=1 */
 | |
|     QUERY_TYPE_DISABLE_AUTOCOMMIT = 0x002000,   /*< SET autocommit=0 */
 | |
|     QUERY_TYPE_ROLLBACK           = 0x004000,   /*< ROLLBACK */
 | |
|     QUERY_TYPE_COMMIT             = 0x008000,   /*< COMMIT */
 | |
|     QUERY_TYPE_PREPARE_NAMED_STMT = 0x010000,   /*< Prepared stmt with name from user:all */
 | |
|     QUERY_TYPE_PREPARE_STMT       = 0x020000,   /*< Prepared stmt with id provided by server:all */
 | |
|     QUERY_TYPE_EXEC_STMT          = 0x040000,   /*< Execute prepared statement:master or any */
 | |
|     QUERY_TYPE_CREATE_TMP_TABLE   = 0x080000,   /*< Create temporary table:master (could be all) */
 | |
|     QUERY_TYPE_READ_TMP_TABLE     = 0x100000,   /*< Read temporary table:master (could be any) */
 | |
|     QUERY_TYPE_SHOW_DATABASES     = 0x200000,   /*< Show list of databases */
 | |
|     QUERY_TYPE_SHOW_TABLES        = 0x400000,   /*< Show list of tables */
 | |
|     QUERY_TYPE_DEALLOC_PREPARE    = 0x1000000   /*< Dealloc named prepare stmt:all */
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * qc_query_op_t defines the operations a particular statement can perform.
 | |
|  */
 | |
| enum qc_query_op_t
 | |
| {
 | |
|     QUERY_OP_UNDEFINED = 0,
 | |
| 
 | |
|     QUERY_OP_ALTER,
 | |
|     QUERY_OP_CALL,
 | |
|     QUERY_OP_CHANGE_DB,
 | |
|     QUERY_OP_CREATE,
 | |
|     QUERY_OP_DELETE,
 | |
|     QUERY_OP_DROP,
 | |
|     QUERY_OP_EXECUTE,
 | |
|     QUERY_OP_EXPLAIN,
 | |
|     QUERY_OP_GRANT,
 | |
|     QUERY_OP_INSERT,
 | |
|     QUERY_OP_LOAD_LOCAL,
 | |
|     QUERY_OP_LOAD,
 | |
|     QUERY_OP_REVOKE,
 | |
|     QUERY_OP_SELECT,
 | |
|     QUERY_OP_SET,
 | |
|     QUERY_OP_SHOW,
 | |
|     QUERY_OP_TRUNCATE,
 | |
|     QUERY_OP_UPDATE,
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * qc_parse_result_t defines the possible outcomes when a statement is parsed.
 | |
|  */
 | |
| enum qc_parse_result_t
 | |
| {
 | |
|     QC_QUERY_INVALID          = 0,  /*< The query was not recognized or could not be parsed. */
 | |
|     QC_QUERY_TOKENIZED        = 1,  /*< The query was classified based on tokens; incompletely classified. */
 | |
|     QC_QUERY_PARTIALLY_PARSED = 2,  /*< The query was only partially parsed; incompletely classified. */
 | |
|     QC_QUERY_PARSED           = 3   /*< The query was fully parsed; completely classified. */
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * qc_field_context_t defines the context where a field appears.
 | |
|  *
 | |
|  * NOTE: A particular bit does NOT mean that the field appears ONLY in the context,
 | |
|  *       but it may appear in other contexts as well.
 | |
|  */
 | |
| typedef enum qc_field_context
 | |
| {
 | |
|     QC_FIELD_UNION    = 1,  /** The field appears on the right hand side in a UNION. */
 | |
|     QC_FIELD_SUBQUERY = 2   /** The field appears in a subquery. */
 | |
| } qc_field_context_t;
 | |
| 
 | |
| struct QC_FIELD_INFO
 | |
| {
 | |
|     char*    database;  /** Present if the field is of the form "a.b.c", NULL otherwise. */
 | |
|     char*    table;     /** Present if the field is of the form "a.b", NULL otherwise. */
 | |
|     char*    column;    /** Always present. */
 | |
|     uint32_t context;   /** The context in which the field appears. */
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * QC_FUNCTION_INFO contains information about a function used in a statement.
 | |
|  */
 | |
| struct QC_FUNCTION_INFO
 | |
| {
 | |
|     char*          name;    /** Name of function. */
 | |
|     QC_FIELD_INFO* fields;  /** What fields the function accesses. */
 | |
|     uint32_t       n_fields;/** The number of fields in @c fields. */
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * Each API function returns @c QC_RESULT_OK if the actual parsing process
 | |
|  * succeeded, and some error code otherwise.
 | |
|  */
 | |
| enum qc_result_t
 | |
| {
 | |
|     QC_RESULT_OK,
 | |
|     QC_RESULT_ERROR
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * QC_STMT_INFO is an opaque type where the query classifier stores
 | |
|  * information about a statement.
 | |
|  */
 | |
| struct QC_STMT_INFO
 | |
| {
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * QC_STMT_RESULT contains limited information about a particular
 | |
|  * statement.
 | |
|  */
 | |
| struct QC_STMT_RESULT
 | |
| {
 | |
|     qc_parse_result_t status;
 | |
|     uint32_t          type_mask;
 | |
|     qc_query_op_t     op;
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * QUERY_CLASSIFIER defines the object a query classifier plugin must
 | |
|  * implement and return.
 | |
|  *
 | |
|  * To a user of the query classifier functionality, it can in general
 | |
|  * be ignored.
 | |
|  */
 | |
| struct QUERY_CLASSIFIER
 | |
| {
 | |
|     /**
 | |
|      * Called once to setup the query classifier
 | |
|      *
 | |
|      * @param sql_mode  The default sql mode.
 | |
|      * @param args      The value of `query_classifier_args` in the configuration file.
 | |
|      *
 | |
|      * @return QC_RESULT_OK, if the query classifier could be setup, otherwise
 | |
|      *         some specific error code.
 | |
|      */
 | |
|     int32_t (* qc_setup)(qc_sql_mode_t sql_mode, const char* args);
 | |
| 
 | |
|     /**
 | |
|      * Called once at process startup, after @c qc_setup has successfully
 | |
|      * been called.
 | |
|      *
 | |
|      * @return QC_RESULT_OK, if the process initialization succeeded.
 | |
|      */
 | |
|     int32_t (* qc_process_init)(void);
 | |
| 
 | |
|     /**
 | |
|      * Called once at process shutdown.
 | |
|      */
 | |
|     void (* qc_process_end)(void);
 | |
| 
 | |
|     /**
 | |
|      * Called once per each thread.
 | |
|      *
 | |
|      * @return QC_RESULT_OK, if the thread initialization succeeded.
 | |
|      */
 | |
|     int32_t (* qc_thread_init)(void);
 | |
| 
 | |
|     /**
 | |
|      * Called once when a thread finishes.
 | |
|      */
 | |
|     void (* qc_thread_end)(void);
 | |
| 
 | |
|     /**
 | |
|      * Called to explicitly parse a statement.
 | |
|      *
 | |
|      * @param stmt     The statement to be parsed.
 | |
|      * @param collect  A bitmask of @c qc_collect_info_t values. Specifies what information
 | |
|      *                 should be collected. Only a hint and must not restrict what information
 | |
|      *                 later can be queried.
 | |
|      * @param result   On return, the parse result, if @c QC_RESULT_OK is returned.
 | |
|      *
 | |
|      * @return QC_RESULT_OK, if the parsing was not aborted due to resource
 | |
|      *         exhaustion or equivalent.
 | |
|      */
 | |
|     int32_t (* qc_parse)(GWBUF* stmt, uint32_t collect, int32_t* result);
 | |
| 
 | |
|     /**
 | |
|      * Reports the type of the statement.
 | |
|      *
 | |
|      * @param stmt  A COM_QUERY or COM_STMT_PREPARE packet.
 | |
|      * @param type  On return, the type mask (combination of @c qc_query_type_t),
 | |
|      *              if @c QC_RESULT_OK is returned.
 | |
|      *
 | |
|      * @return QC_RESULT_OK, if the parsing was not aborted due to resource
 | |
|      *         exhaustion or equivalent.
 | |
|      */
 | |
|     int32_t (* qc_get_type_mask)(GWBUF* stmt, uint32_t* type);
 | |
| 
 | |
|     /**
 | |
|      * Reports the operation of the statement.
 | |
|      *
 | |
|      * @param stmt  A COM_QUERY or COM_STMT_PREPARE packet.
 | |
|      * @param type  On return, the operation (one of @c qc_query_op_t), if
 | |
|      *              @c QC_RESULT_OK is returned.
 | |
|      *
 | |
|      * @return QC_RESULT_OK, if the parsing was not aborted due to resource
 | |
|      *         exhaustion or equivalent.
 | |
|      */
 | |
|     int32_t (* qc_get_operation)(GWBUF* stmt, int32_t* op);
 | |
| 
 | |
|     /**
 | |
|      * Reports the name of a created table.
 | |
|      *
 | |
|      * @param stmt  A COM_QUERY or COM_STMT_PREPARE packet.
 | |
|      * @param name  On return, the name of the created table, if
 | |
|      *              @c QC_RESULT_OK is returned.
 | |
|      *
 | |
|      * @return QC_RESULT_OK, if the parsing was not aborted due to resource
 | |
|      *         exhaustion or equivalent.
 | |
|      */
 | |
|     int32_t (* qc_get_created_table_name)(GWBUF* stmt, char** name);
 | |
| 
 | |
|     /**
 | |
|      * Reports whether a statement is a "DROP TABLE ..." statement.
 | |
|      *
 | |
|      * @param stmt           A COM_QUERY or COM_STMT_PREPARE packet
 | |
|      * @param is_drop_table  On return, non-zero if the statement is a DROP TABLE
 | |
|      *                       statement, if @c QC_RESULT_OK is returned.
 | |
|      *
 | |
|      * @return QC_RESULT_OK, if the parsing was not aborted due to resource
 | |
|      *         exhaustion or equivalent.
 | |
|      */
 | |
|     int32_t (* qc_is_drop_table_query)(GWBUF* stmt, int32_t* is_drop_table);
 | |
| 
 | |
|     /**
 | |
|      * Returns all table names.
 | |
|      *
 | |
|      * @param stmt       A COM_QUERY or COM_STMT_PREPARE packet.
 | |
|      * @param fullnames  If non-zero, the full (i.e. qualified) names are returned.
 | |
|      * @param names      On return, the names of the statement, if @c QC_RESULT_OK
 | |
|      *                   is returned.
 | |
|      * @param n_names    On return, how many names were returned, if @c QC_RESULT_OK
 | |
|      *                   is returned.
 | |
|      *
 | |
|      * @return QC_RESULT_OK, if the parsing was not aborted due to resource
 | |
|      *         exhaustion or equivalent.
 | |
|      */
 | |
|     int32_t (* qc_get_table_names)(GWBUF* stmt, int32_t full_names, char*** names, int32_t* n_names);
 | |
| 
 | |
|     /**
 | |
|      * The canonical version of a statement.
 | |
|      *
 | |
|      * @param stmt       A COM_QUERY or COM_STMT_PREPARE packet.
 | |
|      * @param canonical  On return, the canonical version of the statement, if @c QC_RESULT_OK
 | |
|      *                   is returned.
 | |
|      *
 | |
|      * @return QC_RESULT_OK, if the parsing was not aborted due to resource
 | |
|      *         exhaustion or equivalent.
 | |
|      */
 | |
|     int32_t (* qc_get_canonical)(GWBUF* stmt, char** canonical);
 | |
| 
 | |
|     /**
 | |
|      * Reports whether the statement has a where clause.
 | |
|      *
 | |
|      * @param stmt        A COM_QUERY or COM_STMT_PREPARE packet.
 | |
|      * @param has_clause  On return, non-zero if the statement has a where clause, if
 | |
|      *                    @c QC_RESULT_OK is returned.
 | |
|      *
 | |
|      * @return QC_RESULT_OK, if the parsing was not aborted due to resource
 | |
|      *         exhaustion or equivalent.
 | |
|      */
 | |
|     int32_t (* qc_query_has_clause)(GWBUF* stmt, int32_t* has_clause);
 | |
| 
 | |
|     /**
 | |
|      * Reports the database names.
 | |
|      *
 | |
|      * @param stmt   A COM_QUERY or COM_STMT_PREPARE packet.
 | |
|      * @param names  On return, the database names, if
 | |
|      *               @c QC_RESULT_OK is returned.
 | |
|      * @param size   On return, the number of names in @names, if
 | |
|      *               @c QC_RESULT_OK is returned.
 | |
|      *
 | |
|      * @return QC_RESULT_OK, if the parsing was not aborted due to resource
 | |
|      *         exhaustion or equivalent.
 | |
|      */
 | |
|     int32_t (* qc_get_database_names)(GWBUF* stmt, char*** names, int32_t* size);
 | |
| 
 | |
|     /**
 | |
|      * Reports the prepare name.
 | |
|      *
 | |
|      * @param stmt  A COM_QUERY or COM_STMT_PREPARE packet.
 | |
|      * @param name  On return, the name of a prepare statement, if
 | |
|      *              @c QC_RESULT_OK is returned.
 | |
|      *
 | |
|      * @return QC_RESULT_OK, if the parsing was not aborted due to resource
 | |
|      *         exhaustion or equivalent.
 | |
|      */
 | |
|     int32_t (* qc_get_prepare_name)(GWBUF* stmt, char** name);
 | |
| 
 | |
|     /**
 | |
|      * Reports field information.
 | |
|      *
 | |
|      * @param stmt    A COM_QUERY or COM_STMT_PREPARE packet.
 | |
|      * @param infos   On return, array of field infos, if @c QC_RESULT_OK is returned.
 | |
|      * @param n_infos On return, the size of @c infos, if @c QC_RESULT_OK is returned.
 | |
|      *
 | |
|      * @return QC_RESULT_OK, if the parsing was not aborted due to resource
 | |
|      *         exhaustion or equivalent.
 | |
|      */
 | |
|     int32_t (* qc_get_field_info)(GWBUF* stmt, const QC_FIELD_INFO** infos, uint32_t* n_infos);
 | |
| 
 | |
|     /**
 | |
|      * The canonical version of a statement.
 | |
|      *
 | |
|      * @param stmt  A COM_QUERY or COM_STMT_PREPARE packet.
 | |
|      * @param infos   On return, array of function infos, if @c QC_RESULT_OK is returned.
 | |
|      * @param n_infos On return, the size of @c infos, if @c QC_RESULT_OK is returned.
 | |
|      *
 | |
|      * @return QC_RESULT_OK, if the parsing was not aborted due to resource
 | |
|      *         exhaustion or equivalent.
 | |
|      */
 | |
|     int32_t (* qc_get_function_info)(GWBUF* stmt, const QC_FUNCTION_INFO** infos, uint32_t* n_infos);
 | |
| 
 | |
|     /**
 | |
|      * Return the preparable statement of a PREPARE statement.
 | |
|      *
 | |
|      * @param stmt             A COM_QUERY or COM_STMT_PREPARE packet.
 | |
|      * @param preparable_stmt  On return, the preparable statement (provided @c stmt is a
 | |
|      *                         PREPARE statement), if @c QC_RESULT_OK is returned. Otherwise
 | |
|      *                         NULL.
 | |
|      *
 | |
|      * @attention The returned GWBUF is the property of @c stmt and will be deleted when
 | |
|      *            @c stmt is. If the preparable statement need to be retained beyond the
 | |
|      *            lifetime of @c stmt, it must be cloned.
 | |
|      *
 | |
|      * @return QC_RESULT_OK, if the parsing was not aborted due to resource
 | |
|      *         exhaustion or equivalent.
 | |
|      */
 | |
|     int32_t (* qc_get_preparable_stmt)(GWBUF* stmt, GWBUF** preparable_stmt);
 | |
| 
 | |
|     /**
 | |
|      * Set the version of the server. The version may affect how a statement
 | |
|      * is classified. Note that the server version is maintained separately
 | |
|      * for each thread.
 | |
|      *
 | |
|      * @param version  Version encoded as MariaDB encodes the version, i.e.:
 | |
|      *                 version = major * 10000 + minor * 100 + patch
 | |
|      */
 | |
|     void (* qc_set_server_version)(uint64_t version);
 | |
| 
 | |
|     /**
 | |
|      * Get the thread specific version assumed of the server. If the version has
 | |
|      * not been set, all values are 0.
 | |
|      *
 | |
|      * @param version  The version encoded as MariaDB encodes the version, i.e.:
 | |
|      *                 version = major * 10000 + minor * 100 + patch
 | |
|      */
 | |
|     void (* qc_get_server_version)(uint64_t* version);
 | |
| 
 | |
|     /**
 | |
|      * Gets the sql mode of the *calling* thread.
 | |
|      *
 | |
|      * @param sql_mode  The mode.
 | |
|      *
 | |
|      * @return QC_RESULT_OK
 | |
|      */
 | |
|     int32_t (* qc_get_sql_mode)(qc_sql_mode_t* sql_mode);
 | |
| 
 | |
|     /**
 | |
|      * Sets the sql mode for the *calling* thread.
 | |
|      *
 | |
|      * @param sql_mode  The mode.
 | |
|      *
 | |
|      * @return QC_RESULT_OK if @sql_mode is valid, otherwise QC_RESULT_ERROR.
 | |
|      */
 | |
|     int32_t (* qc_set_sql_mode)(qc_sql_mode_t sql_mode);
 | |
| 
 | |
|     /**
 | |
|      * Dups the provided info object. After having been dupped, the info object
 | |
|      * can be stored on another GWBUF.
 | |
|      *
 | |
|      * @param info  The info to be dupped.
 | |
|      *
 | |
|      * @return The same info that was provided as argument.
 | |
|      */
 | |
|     QC_STMT_INFO* (* qc_info_dup)(QC_STMT_INFO* info);
 | |
| 
 | |
|     /**
 | |
|      * Closes a dupped info object. After the info object has been closed, it must
 | |
|      * not be accessed.
 | |
|      *
 | |
|      * @param info  The info to be closed.
 | |
|      */
 | |
|     void (* qc_info_close)(QC_STMT_INFO* info);
 | |
| 
 | |
|     /**
 | |
|      * Gets the options of the *calling* thread.
 | |
|      *
 | |
|      * @return Bit mask of values from qc_option_t.
 | |
|      */
 | |
|     uint32_t (* qc_get_options)();
 | |
| 
 | |
|     /**
 | |
|      * Sets the options for the *calling* thread.
 | |
|      *
 | |
|      * @param options Bits from qc_option_t.
 | |
|      *
 | |
|      * @return QC_RESULT_OK if @c options is valid, otherwise QC_RESULT_ERROR.
 | |
|      */
 | |
|     int32_t (* qc_set_options)(uint32_t options);
 | |
| 
 | |
|     /**
 | |
|      * Get result from info.
 | |
|      *
 | |
|      * @param  The info whose result should be returned.
 | |
|      *
 | |
|      * @return The result of the provided info.
 | |
|      */
 | |
|     QC_STMT_RESULT (* qc_get_result_from_info)(const QC_STMT_INFO* info);
 | |
| 
 | |
|     /**
 | |
|      * Return statement currently being classified.
 | |
|      *
 | |
|      * @param ppStmp  Pointer to pointer that on return will point to the
 | |
|      *                statement being classified.
 | |
|      * @param pLen    Pointer to value that on return will contain the length
 | |
|      *                of the returned string.
 | |
|      *
 | |
|      * @return QC_RESULT_OK if a statement was returned (i.e. a statement is being
 | |
|      *         classified), QC_RESULT_ERROR otherwise.
 | |
|      */
 | |
|     int32_t (* qc_get_current_stmt)(const char** ppStmt, size_t* pLen);
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * QC_CACHE_PROPERTIES specifies the limits of the query classification cache.
 | |
|  */
 | |
| struct QC_CACHE_PROPERTIES
 | |
| {
 | |
|     int64_t max_size;   /** The maximum size of the cache. */
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * QC_CACHE_STATS provides statistics of the cache.
 | |
|  */
 | |
| struct QC_CACHE_STATS
 | |
| {
 | |
|     int64_t size;       /** The current size of the cache. */
 | |
|     int64_t inserts;    /** The number of inserts. */
 | |
|     int64_t hits;       /** The number of hits. */
 | |
|     int64_t misses;     /** The number of misses. */
 | |
|     int64_t evictions;  /** The number of evictions. */
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * Loads and sets up the default query classifier.
 | |
|  *
 | |
|  * This must be called once during the execution of a process. The query
 | |
|  * classifier functions can only be used if this function first and thereafter
 | |
|  * the @c qc_process_init return true.
 | |
|  *
 | |
|  * MaxScale calls this function, so plugins should not do that.
 | |
|  *
 | |
|  * @param cache_properties  If non-NULL, specifies the properties of the QC cache.
 | |
|  * @param sql_mode          The default sql mode.
 | |
|  * @param plugin_name       The name of the plugin from which the query classifier
 | |
|  *                          should be loaded.
 | |
|  * @param plugin_args       The arguments to be provided to the query classifier.
 | |
|  *
 | |
|  * @return True if the query classifier could be loaded and initialized,
 | |
|  *         false otherwise.
 | |
|  *
 | |
|  * @see qc_process_init qc_thread_init
 | |
|  */
 | |
| bool qc_setup(const QC_CACHE_PROPERTIES* cache_properties,
 | |
|               qc_sql_mode_t sql_mode,
 | |
|               const char* plugin_name,
 | |
|               const char* plugin_args);
 | |
| 
 | |
| /**
 | |
|  * Loads and setups the default query classifier, and performs
 | |
|  * process and thread initialization.
 | |
|  *
 | |
|  * This is primary intended for making the setup of stand-alone
 | |
|  * test-programs simpler.
 | |
|  *
 | |
|  * @param cache_properties  If non-NULL, specifies the properties of the QC cache.
 | |
|  * @param sql_mode          The default sql mode.
 | |
|  * @param plugin_name       The name of the plugin from which the query classifier
 | |
|  *                          should be loaded.
 | |
|  * @param plugin_args       The arguments to be provided to the query classifier.
 | |
|  *
 | |
|  * @return True if the query classifier could be loaded and initialized,
 | |
|  *         false otherwise.
 | |
|  *
 | |
|  * @see qc_end.
 | |
|  */
 | |
| bool qc_init(const QC_CACHE_PROPERTIES* cache_properties,
 | |
|              qc_sql_mode_t sql_mode,
 | |
|              const char* plugin_name,
 | |
|              const char* plugin_args);
 | |
| 
 | |
| /**
 | |
|  * Performs thread and process finalization.
 | |
|  *
 | |
|  * This is primary intended for making the tear-down of stand-alone
 | |
|  * test-programs simpler.
 | |
|  */
 | |
| void qc_end();
 | |
| 
 | |
| /**
 | |
|  * Intializes the query classifier.
 | |
|  *
 | |
|  * This function should be called once, provided @c qc_setup returned true,
 | |
|  * before the query classifier functionality is used.
 | |
|  *
 | |
|  * MaxScale calls this functions, so plugins should not do that.
 | |
|  *
 | |
|  * @param kind  What kind of initialization should be performed.
 | |
|  *              Combination of qc_init_kind_t.
 | |
|  *
 | |
|  * @return True, if the process wide initialization could be performed.
 | |
|  *
 | |
|  * @see qc_process_end qc_thread_init
 | |
|  */
 | |
| bool qc_process_init(uint32_t kind);
 | |
| 
 | |
| /**
 | |
|  * Finalizes the query classifier.
 | |
|  *
 | |
|  * A successful call of @c qc_process_init should before program exit be
 | |
|  * followed by a call to this function. MaxScale calls this function, so
 | |
|  * plugins should not do that.
 | |
|  *
 | |
|  * @param kind  What kind of finalization should be performed.
 | |
|  *              Combination of qc_init_kind_t.
 | |
|  *
 | |
|  * @see qc_process_init qc_thread_end
 | |
|  */
 | |
| void qc_process_end(uint32_t kind);
 | |
| 
 | |
| /**
 | |
|  * Loads a particular query classifier.
 | |
|  *
 | |
|  * In general there is no need to use this function, but rely upon qc_init().
 | |
|  * However, if there is a need to use multiple query classifiers concurrently
 | |
|  * then this function provides the means for that. Note that after a query
 | |
|  * classifier has been loaded, it must explicitly be initialized before it
 | |
|  * can be used.
 | |
|  *
 | |
|  * @param plugin_name  The name of the plugin from which the query classifier
 | |
|  *                     should be loaded.
 | |
|  *
 | |
|  * @return A QUERY_CLASSIFIER object if successful, NULL otherwise.
 | |
|  *
 | |
|  * @see qc_unload
 | |
|  */
 | |
| QUERY_CLASSIFIER* qc_load(const char* plugin_name);
 | |
| 
 | |
| /**
 | |
|  * Unloads an explicitly loaded query classifier.
 | |
|  *
 | |
|  * @see qc_load
 | |
|  */
 | |
| void qc_unload(QUERY_CLASSIFIER* classifier);
 | |
| 
 | |
| /**
 | |
|  * Performs thread initialization needed by the query classifier. Should
 | |
|  * be called in every thread.
 | |
|  *
 | |
|  * MaxScale calls this function, so plugins should not do that.
 | |
|  *
 | |
|  * @param kind  What kind of initialization should be performed.
 | |
|  *              Combination of qc_init_kind_t.
 | |
|  *
 | |
|  * @return True if the initialization succeeded, false otherwise.
 | |
|  *
 | |
|  * @see qc_thread_end
 | |
|  */
 | |
| bool qc_thread_init(uint32_t kind);
 | |
| 
 | |
| /**
 | |
|  * Performs thread finalization needed by the query classifier.
 | |
|  * A successful call to @c qc_thread_init should at some point be
 | |
|  * followed by a call to this function.
 | |
|  *
 | |
|  * MaxScale calls this function, so plugins should not do that.
 | |
|  *
 | |
|  * @param kind  What kind of finalization should be performed.
 | |
|  *              Combination of qc_init_kind_t.
 | |
|  *
 | |
|  * @see qc_thread_init
 | |
|  */
 | |
| void qc_thread_end(uint32_t kind);
 | |
| 
 | |
| /**
 | |
|  * Parses the statement in the provided buffer and returns a value specifying
 | |
|  * to what extent the statement could be parsed.
 | |
|  *
 | |
|  * There is no need to call this function explicitly before calling any of
 | |
|  * the other functions; e.g. qc_get_type_mask(). When some particular property of
 | |
|  * a statement is asked for, the statement will be parsed if it has not been
 | |
|  * parsed yet. Also, if the statement in the provided buffer has been parsed
 | |
|  * already then this function will only return the result of that parsing;
 | |
|  * the statement will not be parsed again.
 | |
|  *
 | |
|  * @param stmt     A buffer containing an COM_QUERY or COM_STMT_PREPARE packet.
 | |
|  * @param collect  A bitmask of @c qc_collect_info_t values. Specifies what information
 | |
|  *                 should be collected.
 | |
|  *
 | |
|  *                 Note that this is merely a hint and does not restrict what
 | |
|  *                 information can be queried for. If necessary, the statement
 | |
|  *                 will transparently be reparsed.
 | |
|  *
 | |
|  * @return To what extent the statement could be parsed.
 | |
|  */
 | |
| qc_parse_result_t qc_parse(GWBUF* stmt, uint32_t collect);
 | |
| 
 | |
| /**
 | |
|  * Returns information about affected fields.
 | |
|  *
 | |
|  * @param stmt     A buffer containing a COM_QUERY or COM_STMT_PREPARE packet.
 | |
|  * @param infos    Pointer to pointer that after the call will point to an
 | |
|  *                 array of QC_FIELD_INFO:s.
 | |
|  * @param n_infos  Pointer to size_t variable where the number of items
 | |
|  *                 in @c infos will be returned.
 | |
|  *
 | |
|  * @note The returned array belongs to the GWBUF and remains valid for as
 | |
|  *       long as the GWBUF is valid. If the data is needed for longer than
 | |
|  *       that, it must be copied.
 | |
|  */
 | |
| void qc_get_field_info(GWBUF* stmt, const QC_FIELD_INFO** infos, size_t* n_infos);
 | |
| 
 | |
| /**
 | |
|  * Returns information about function usage.
 | |
|  *
 | |
|  * @param stmt     A buffer containing a COM_QUERY or COM_STMT_PREPARE packet.
 | |
|  * @param infos    Pointer to pointer that after the call will point to an
 | |
|  *                 array of QC_FUNCTION_INFO:s.
 | |
|  * @param n_infos  Pointer to size_t variable where the number of items
 | |
|  *                 in @c infos will be returned.
 | |
|  *
 | |
|  * @note The returned array belongs to the GWBUF and remains valid for as
 | |
|  *       long as the GWBUF is valid. If the data is needed for longer than
 | |
|  *       that, it must be copied.
 | |
|  *
 | |
|  * @note For each function, only the fields that any invocation of it directly
 | |
|  *       accesses will be returned. For instance:
 | |
|  *
 | |
|  *           select length(a), length(concat(b, length(a))) from t
 | |
|  *
 | |
|  *       will for @length return the field @a and for @c concat the field @b.
 | |
|  */
 | |
| void qc_get_function_info(GWBUF* stmt, const QC_FUNCTION_INFO** infos, size_t* n_infos);
 | |
| 
 | |
| /**
 | |
|  * Returns the statement, with literals replaced with question marks.
 | |
|  *
 | |
|  * @param stmt  A buffer containing a COM_QUERY or COM_STMT_PREPARE packet.
 | |
|  *
 | |
|  * @return A statement in its canonical form, or NULL if a memory
 | |
|  *         allocation fails. The string must be freed by the caller.
 | |
|  */
 | |
| char* qc_get_canonical(GWBUF* stmt);
 | |
| 
 | |
| /**
 | |
|  * Returns the name of the created table.
 | |
|  *
 | |
|  * @param stmt  A buffer containing a COM_QUERY or COM_STMT_PREPARE packet.
 | |
|  *
 | |
|  * @return The name of the created table or NULL if the statement
 | |
|  *         does not create a table or a memory allocation failed.
 | |
|  *         The string must be freed by the caller.
 | |
|  */
 | |
| char* qc_get_created_table_name(GWBUF* stmt);
 | |
| 
 | |
| /**
 | |
|  * Returns the databases accessed by the statement. Note that a
 | |
|  * possible default database is not returned.
 | |
|  *
 | |
|  * @param stmt  A buffer containing a COM_QUERY or COM_STMT_PREPARE packet.
 | |
|  * @param size  Pointer to integer where the number of databases
 | |
|  *              is stored.
 | |
|  *
 | |
|  * @return Array of strings or NULL if a memory allocation fails.
 | |
|  *
 | |
|  * @note The returned array and the strings pointed to @b must be freed
 | |
|  *       by the caller.
 | |
|  */
 | |
| char** qc_get_database_names(GWBUF* stmt, int* size);
 | |
| 
 | |
| /**
 | |
|  * Returns the operation of the statement.
 | |
|  *
 | |
|  * @param stmt  A buffer containing a COM_QUERY or COM_STMT_PREPARE packet.
 | |
|  *
 | |
|  * @return The operation of the statement.
 | |
|  */
 | |
| qc_query_op_t qc_get_operation(GWBUF* stmt);
 | |
| 
 | |
| /**
 | |
|  * Returns the name of the prepared statement, if the statement
 | |
|  * is a PREPARE or EXECUTE statement.
 | |
|  *
 | |
|  * @param stmt  A buffer containing a COM_QUERY or COM_STMT_PREPARE packet.
 | |
|  *
 | |
|  * @return The name of the prepared statement, if the statement
 | |
|  *         is a PREPARE or EXECUTE statement; otherwise NULL.
 | |
|  *
 | |
|  * @note The returned string @b must be freed by the caller.
 | |
|  *
 | |
|  * @note Even though a COM_STMT_PREPARE can be given to the query
 | |
|  *       classifier for parsing, this function will in that case
 | |
|  *       return NULL since the id of the statement is provided by
 | |
|  *       the server.
 | |
|  */
 | |
| char* qc_get_prepare_name(GWBUF* stmt);
 | |
| 
 | |
| /**
 | |
|  * Returns the preparable statement of a PREPARE statment. Other query classifier
 | |
|  * functions can then be used on the returned statement to find out information
 | |
|  * about the preparable statement. The returned @c GWBUF should not be used for
 | |
|  * anything else but for obtaining information about the preparable statement.
 | |
|  *
 | |
|  * @param stmt  A buffer containing a COM_QUERY or COM_STMT_PREPARE packet.
 | |
|  *
 | |
|  * @return The preparable statement, if @stmt was a COM_QUERY PREPARE statement,
 | |
|  *         or NULL.
 | |
|  *
 | |
|  * @attention If the packet was a COM_STMT_PREPARE, then this function will
 | |
|  *            return NULL and the actual properties of the query can be obtained
 | |
|  *            by calling any of the qc-functions directly on the GWBUF containing
 | |
|  *            the COM_STMT_PREPARE. However, the type mask will contain the
 | |
|  *            bit @c QUERY_TYPE_PREPARE_STMT.
 | |
|  *
 | |
|  * @attention The returned @c GWBUF is the property of @c stmt and will be
 | |
|  *            deleted along with it.
 | |
|  */
 | |
| GWBUF* qc_get_preparable_stmt(GWBUF* stmt);
 | |
| 
 | |
| /**
 | |
|  * Gets the sql mode of the *calling* thread.
 | |
|  *
 | |
|  * @return The mode.
 | |
|  */
 | |
| qc_sql_mode_t qc_get_sql_mode();
 | |
| 
 | |
| /**
 | |
|  * Returns the tables accessed by the statement.
 | |
|  *
 | |
|  * @param stmt       A buffer containing a COM_QUERY or COM_STMT_PREPARE packet.
 | |
|  * @param tblsize    Pointer to integer where the number of tables is stored.
 | |
|  * @param fullnames  If true, a table names will include the database name
 | |
|  *                   as well (if explicitly referred to in the statement).
 | |
|  *
 | |
|  * @return Array of strings or NULL if a memory allocation fails.
 | |
|  *
 | |
|  * @note The returned array and the strings pointed to @b must be freed
 | |
|  *       by the caller.
 | |
|  */
 | |
| char** qc_get_table_names(GWBUF* stmt, int* size, bool fullnames);
 | |
| 
 | |
| /**
 | |
|  * Free tables returned by qc_get_table_names
 | |
|  *
 | |
|  * @param names List of names
 | |
|  * @param size  Size of @c names
 | |
|  */
 | |
| void qc_free_table_names(char** names, int size);
 | |
| 
 | |
| /**
 | |
|  * Returns a bitmask specifying the type(s) of the statement. The result
 | |
|  * should be tested against specific qc_query_type_t values* using the
 | |
|  * bitwise & operator, never using the == operator.
 | |
|  *
 | |
|  * @param stmt  A buffer containing a COM_QUERY or COM_STMT_PREPARE packet.
 | |
|  *
 | |
|  * @return A bitmask with the type(s) the query.
 | |
|  *
 | |
|  * @see qc_query_is_type
 | |
|  */
 | |
| uint32_t qc_get_type_mask(GWBUF* stmt);
 | |
| 
 | |
| /**
 | |
|  * Returns the type bitmask of transaction related statements.
 | |
|  *
 | |
|  * If the statement starts a transaction, ends a transaction or
 | |
|  * changes the autocommit state, the returned bitmap will be a
 | |
|  * combination of:
 | |
|  *
 | |
|  *    QUERY_TYPE_BEGIN_TRX
 | |
|  *    QUERY_TYPE_COMMIT
 | |
|  *    QUERY_TYPE_ROLLBACK
 | |
|  *    QUERY_TYPE_ENABLE_AUTOCOMMIT
 | |
|  *    QUERY_TYPE_DISABLE_AUTOCOMMIT
 | |
|  *    QUERY_TYPE_READ  (explicitly read only transaction)
 | |
|  *    QUERY_TYPE_WRITE (explicitly read write transaction)
 | |
|  *
 | |
|  * Otherwise the result will be 0.
 | |
|  *
 | |
|  * @param stmt A COM_QUERY or COM_STMT_PREPARE packet.
 | |
|  *
 | |
|  * @return The relevant type bits if the statement is transaction
 | |
|  *         related, otherwise 0.
 | |
|  */
 | |
| uint32_t qc_get_trx_type_mask(GWBUF* stmt);
 | |
| 
 | |
| /**
 | |
|  * Returns whether the statement is a DROP TABLE statement.
 | |
|  *
 | |
|  * @param stmt  A buffer containing a COM_QUERY or COM_STMT_PREPARE packet.
 | |
|  *
 | |
|  * @return True if the statement is a DROP TABLE statement, false otherwise.
 | |
|  *
 | |
|  * @todo This function is far too specific.
 | |
|  */
 | |
| bool qc_is_drop_table_query(GWBUF* stmt);
 | |
| 
 | |
| /**
 | |
|  * Returns the string representation of a query operation.
 | |
|  *
 | |
|  * @param op  A query operation.
 | |
|  *
 | |
|  * @return The corresponding string.
 | |
|  *
 | |
|  * @note The returned string is statically allocated and must *not* be freed.
 | |
|  */
 | |
| const char* qc_op_to_string(qc_query_op_t op);
 | |
| 
 | |
| /**
 | |
|  * Returns whether the typemask contains a particular type.
 | |
|  *
 | |
|  * @param typemask  A bitmask of query types.
 | |
|  * @param type      A particular qc_query_type_t value.
 | |
|  *
 | |
|  * @return True, if the type is in the mask.
 | |
|  */
 | |
| static inline bool qc_query_is_type(uint32_t typemask, qc_query_type_t type)
 | |
| {
 | |
|     return (typemask & (uint32_t)type) == (uint32_t)type;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Returns whether the statement has a WHERE or a USING clause.
 | |
|  *
 | |
|  * @param stmt  A buffer containing a COM_QUERY or COM_STMT_PREPARE packet.
 | |
|  *
 | |
|  * @return True, if the statement has a WHERE or USING clause, false
 | |
|  *         otherwise.
 | |
|  */
 | |
| bool qc_query_has_clause(GWBUF* stmt);
 | |
| 
 | |
| /**
 | |
|  * Sets the sql mode for the *calling* thread.
 | |
|  *
 | |
|  * @param sql_mode  The mode.
 | |
|  */
 | |
| void qc_set_sql_mode(qc_sql_mode_t sql_mode);
 | |
| 
 | |
| /**
 | |
|  * Returns the string representation of a query type.
 | |
|  *
 | |
|  * @param type  A query type (not a bitmask of several).
 | |
|  *
 | |
|  * @return The corresponding string.
 | |
|  *
 | |
|  * @note The returned string is statically allocated and must @b not be freed.
 | |
|  */
 | |
| const char* qc_type_to_string(qc_query_type_t type);
 | |
| 
 | |
| /**
 | |
|  * Returns a string representation of a type bitmask.
 | |
|  *
 | |
|  * @param typemask  A bit mask of query types.
 | |
|  *
 | |
|  * @return The corresponding string or NULL if the allocation fails.
 | |
|  *
 | |
|  * @note The returned string is dynamically allocated and @b must be freed.
 | |
|  */
 | |
| char* qc_typemask_to_string(uint32_t typemask);
 | |
| 
 | |
| /**
 | |
|  * Set the version of the server. The version may affect how a statement
 | |
|  * is classified. Note that the server version is maintained separately
 | |
|  * for each thread.
 | |
|  *
 | |
|  * @param version  Version encoded as MariaDB encodes the version, i.e.:
 | |
|  *                 version = major * 10000 + minor * 100 + patch
 | |
|  */
 | |
| void qc_set_server_version(uint64_t version);
 | |
| 
 | |
| /**
 | |
|  * Get the thread specific version assumed of the server. If the version has
 | |
|  * not been set, all values are 0.
 | |
|  *
 | |
|  * @return The version as MariaDB encodes the version, i.e:
 | |
|  *         version = major * 10000 + minor * 100 + patch
 | |
|  */
 | |
| uint64_t qc_get_server_version();
 | |
| 
 | |
| /**
 | |
|  * Get the cache properties.
 | |
|  *
 | |
|  * @param[out] properties  Cache properties.
 | |
|  */
 | |
| void qc_get_cache_properties(QC_CACHE_PROPERTIES* properties);
 | |
| 
 | |
| /**
 | |
|  * Set the cache properties.
 | |
|  *
 | |
|  * @param properties  Cache properties.
 | |
|  *
 | |
|  * @return True, if the properties could be set, false if at least
 | |
|  *         one property is invalid or if the combination of property
 | |
|  *         values is invalid.
 | |
|  */
 | |
| bool qc_set_cache_properties(const QC_CACHE_PROPERTIES* properties);
 | |
| 
 | |
| /**
 | |
|  * Get cache statistics for the calling thread.
 | |
|  *
 | |
|  * @param stats[out]  Cache statistics.
 | |
|  *
 | |
|  * @return True, if caching is enabled, false otherwise.
 | |
|  */
 | |
| bool qc_get_cache_stats(QC_CACHE_STATS* stats);
 | |
| 
 | |
| /**
 | |
|  * Get cache statistics for the calling thread.
 | |
|  *
 | |
|  * @return An object if caching is enabled, NULL otherwise.
 | |
|  */
 | |
| json_t* qc_get_cache_stats_as_json();
 | |
| 
 | |
| /**
 | |
|  * String represenation for the parse result.
 | |
|  *
 | |
|  * @param result A parsing result.
 | |
|  *
 | |
|  * @return The corresponding string.
 | |
|  */
 | |
| const char* qc_result_to_string(qc_parse_result_t result);
 | |
| 
 | |
| /**
 | |
|  * Gets the options of the *calling* thread.
 | |
|  *
 | |
|  * @return Bit mask of values from qc_option_t.
 | |
|  */
 | |
| uint32_t qc_get_options();
 | |
| 
 | |
| /**
 | |
|  * Sets the options for the *calling* thread.
 | |
|  *
 | |
|  * @param options Bits from qc_option_t.
 | |
|  *
 | |
|  * @return true if the options were valid, false otherwise.
 | |
|  */
 | |
| bool qc_set_options(uint32_t options);
 | |
| 
 | |
| /**
 | |
|  * Public interface to query classifier cache state.
 | |
|  */
 | |
| struct QC_CACHE_ENTRY
 | |
| {
 | |
|     int64_t        hits;
 | |
|     QC_STMT_RESULT result;
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * Obtain query classifier cache information for the @b calling thread.
 | |
|  *
 | |
|  * @param state  Map where information is added.
 | |
|  *
 | |
|  * @note Calling with a non-empty @c state means that a cumulative result
 | |
|  *       will be obtained, that is, the hits of a particular key will
 | |
|  *       be added the hits of that key if it already is in the map.
 | |
|  */
 | |
| void qc_get_cache_state(std::map<std::string, QC_CACHE_ENTRY>& state);
 | |
| 
 | |
| /**
 | |
|  * Return statement currently being classified.
 | |
|  *
 | |
|  * @param ppStmp  Pointer to pointer that on return will point to the
 | |
|  *                statement being classified.
 | |
|  * @param pLen    Pointer to value that on return will contain the length
 | |
|  *                of the returned string.
 | |
|  *
 | |
|  * @return True, if a statement was returned (i.e. a statement is being
 | |
|  *         classified), false otherwise.
 | |
|  *
 | |
|  * @note A string /may/ be returned /only/ when this function is called from
 | |
|  *       a signal handler that is called due to the classifier causing
 | |
|  *       a crash.
 | |
|  */
 | |
| bool qc_get_current_stmt(const char** ppStmt, size_t* pLen);
 | 
