459 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			459 lines
		
	
	
		
			12 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: 2025-03-08
 | |
|  *
 | |
|  * 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.
 | |
|  */
 | |
| 
 | |
| /**
 | |
|  * @file maxinfo_parse.c - Parse the limited set of SQL that the MaxScale
 | |
|  * information schema can use
 | |
|  *
 | |
|  * @verbatim
 | |
|  * Revision History
 | |
|  *
 | |
|  * Date     Who           Description
 | |
|  * 16/02/15 Mark Riddoch  Initial implementation
 | |
|  *
 | |
|  * @endverbatim
 | |
|  */
 | |
| 
 | |
| #include "maxinfo.hh"
 | |
| 
 | |
| #include <ctype.h>
 | |
| #include <stdio.h>
 | |
| #include <stdlib.h>
 | |
| #include <string.h>
 | |
| #include <maxbase/alloc.h>
 | |
| #include <maxscale/service.hh>
 | |
| #include <maxscale/session.hh>
 | |
| #include <maxscale/router.hh>
 | |
| #include <maxscale/modinfo.h>
 | |
| #include <maxscale/modutil.hh>
 | |
| #include <maxbase/atomic.h>
 | |
| #include <maxscale/dcb.hh>
 | |
| #include <maxscale/poll.hh>
 | |
| 
 | |
| static MAXINFO_TREE* make_tree_node(MAXINFO_OPERATOR, char*, MAXINFO_TREE*, MAXINFO_TREE*);
 | |
| void                 maxinfo_free_tree(MAXINFO_TREE*);  // This function is needed by maxinfo.c
 | |
| static char*         fetch_token(char*, int*, char**);
 | |
| static MAXINFO_TREE* parse_column_list(char** sql);
 | |
| static MAXINFO_TREE* parse_table_name(char** sql);
 | |
| MAXINFO_TREE*        maxinfo_parse_literals(MAXINFO_TREE* tree,
 | |
|                                             int min_args,
 | |
|                                             char* ptr,
 | |
|                                             PARSE_ERROR* parse_error);
 | |
| 
 | |
| /**
 | |
|  * Parse a SQL subset for the maxinfo plugin and return a parse tree
 | |
|  *
 | |
|  * @param sql       The SQL query
 | |
|  * @return  Parse tree or NULL on error
 | |
|  */
 | |
| MAXINFO_TREE* maxinfo_parse(char* sql, PARSE_ERROR* parse_error)
 | |
| {
 | |
|     int token;
 | |
|     char* ptr, * text;
 | |
|     MAXINFO_TREE* tree = NULL;
 | |
|     MAXINFO_TREE* col, * table;
 | |
| 
 | |
|     *parse_error = PARSE_NOERROR;
 | |
|     while ((ptr = fetch_token(sql, &token, &text)) != NULL)
 | |
|     {
 | |
|         switch (token)
 | |
|         {
 | |
|         case LT_SHOW:
 | |
|             MXS_FREE(text);     // not needed
 | |
|             ptr = fetch_token(ptr, &token, &text);
 | |
|             if (ptr == NULL || token != LT_STRING)
 | |
|             {
 | |
|                 // Expected show "name"
 | |
|                 *parse_error = PARSE_MALFORMED_SHOW;
 | |
|                 return NULL;
 | |
|             }
 | |
|             tree = make_tree_node(MAXOP_SHOW, text, NULL, NULL);
 | |
|             if ((ptr = fetch_token(ptr, &token, &text)) == NULL)
 | |
|             {
 | |
|                 return tree;
 | |
|             }
 | |
|             else if (token == LT_LIKE)
 | |
|             {
 | |
|                 if ((ptr = fetch_token(ptr, &token, &text)) != NULL)
 | |
|                 {
 | |
|                     tree->right = make_tree_node(MAXOP_LIKE,
 | |
|                                                  text,
 | |
|                                                  NULL,
 | |
|                                                  NULL);
 | |
|                     return tree;
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     // Expected expression
 | |
|                     *parse_error = PARSE_EXPECTED_LIKE;
 | |
|                     maxinfo_free_tree(tree);
 | |
|                     return NULL;
 | |
|                 }
 | |
|             }
 | |
|             // Malformed show
 | |
|             MXS_FREE(text);
 | |
|             maxinfo_free_tree(tree);
 | |
|             *parse_error = PARSE_MALFORMED_SHOW;
 | |
|             return NULL;
 | |
| 
 | |
| #if 0
 | |
|         case LT_SELECT:
 | |
|             MXS_FREE(text);     // not needed
 | |
|             col = parse_column_list(&ptr);
 | |
|             table = parse_table_name(&ptr);
 | |
|             return make_tree_node(MAXOP_SELECT, NULL, col, table);
 | |
| 
 | |
| #endif
 | |
|         case LT_FLUSH:
 | |
|             MXS_FREE(text);     // not needed
 | |
|             ptr = fetch_token(ptr, &token, &text);
 | |
|             return make_tree_node(MAXOP_FLUSH, text, NULL, NULL);
 | |
| 
 | |
|         case LT_SHUTDOWN:
 | |
|             MXS_FREE(text);
 | |
|             ptr = fetch_token(ptr, &token, &text);
 | |
|             tree = make_tree_node(MAXOP_SHUTDOWN, text, NULL, NULL);
 | |
| 
 | |
|             if ((ptr = fetch_token(ptr, &token, &text)) == NULL)
 | |
|             {
 | |
|                 /** Possibly SHUTDOWN MAXSCALE */
 | |
|                 return tree;
 | |
|             }
 | |
|             tree->right = make_tree_node(MAXOP_LITERAL, text, NULL, NULL);
 | |
| 
 | |
|             if ((ptr = fetch_token(ptr, &token, &text)) != NULL)
 | |
|             {
 | |
|                 /** Unknown token after SHUTDOWN MONITOR|SERVICE */
 | |
|                 *parse_error = PARSE_SYNTAX_ERROR;
 | |
|                 maxinfo_free_tree(tree);
 | |
|                 return NULL;
 | |
|             }
 | |
|             return tree;
 | |
| 
 | |
|         case LT_RESTART:
 | |
|             MXS_FREE(text);
 | |
|             ptr = fetch_token(ptr, &token, &text);
 | |
|             tree = make_tree_node(MAXOP_RESTART, text, NULL, NULL);
 | |
| 
 | |
|             if ((ptr = fetch_token(ptr, &token, &text)) == NULL)
 | |
|             {
 | |
|                 /** Missing token for RESTART MONITOR|SERVICE */
 | |
|                 *parse_error = PARSE_SYNTAX_ERROR;
 | |
|                 maxinfo_free_tree(tree);
 | |
|                 return NULL;
 | |
|             }
 | |
|             tree->right = make_tree_node(MAXOP_LITERAL, text, NULL, NULL);
 | |
| 
 | |
|             if ((ptr = fetch_token(ptr, &token, &text)) != NULL)
 | |
|             {
 | |
|                 /** Unknown token after RESTART MONITOR|SERVICE */
 | |
|                 *parse_error = PARSE_SYNTAX_ERROR;
 | |
|                 MXS_FREE(text);
 | |
|                 maxinfo_free_tree(tree);
 | |
|                 return NULL;
 | |
|             }
 | |
|             return tree;
 | |
| 
 | |
|         case LT_SET:
 | |
|             MXS_FREE(text);     // not needed
 | |
|             ptr = fetch_token(ptr, &token, &text);
 | |
|             tree = make_tree_node(MAXOP_SET, text, NULL, NULL);
 | |
|             return maxinfo_parse_literals(tree, 2, ptr, parse_error);
 | |
| 
 | |
|         case LT_CLEAR:
 | |
|             MXS_FREE(text);     // not needed
 | |
|             ptr = fetch_token(ptr, &token, &text);
 | |
|             tree = make_tree_node(MAXOP_CLEAR, text, NULL, NULL);
 | |
|             return maxinfo_parse_literals(tree, 2, ptr, parse_error);
 | |
|             break;
 | |
| 
 | |
|         default:
 | |
|             *parse_error = PARSE_SYNTAX_ERROR;
 | |
|             return NULL;
 | |
|         }
 | |
|     }
 | |
|     *parse_error = PARSE_SYNTAX_ERROR;
 | |
|     return NULL;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Parse a column list, may be a * or a valid list of string name
 | |
|  * separated by a comma
 | |
|  *
 | |
|  * @param sql   Pointer to pointer to column list updated to point to the table name
 | |
|  * @return  A tree of column names
 | |
|  */
 | |
| static MAXINFO_TREE* parse_column_list(char** ptr)
 | |
| {
 | |
|     int token, lookahead;
 | |
|     char* text, * text2;
 | |
|     MAXINFO_TREE* tree = NULL;
 | |
|     MAXINFO_TREE* rval = NULL;
 | |
|     *ptr = fetch_token(*ptr, &token, &text);
 | |
|     *ptr = fetch_token(*ptr, &lookahead, &text2);
 | |
|     switch (token)
 | |
|     {
 | |
|     case LT_STRING:
 | |
|         switch (lookahead)
 | |
|         {
 | |
|         case LT_COMMA:
 | |
|             rval = make_tree_node(MAXOP_COLUMNS,
 | |
|                                   text,
 | |
|                                   NULL,
 | |
|                                   parse_column_list(ptr));
 | |
|             break;
 | |
| 
 | |
|         case LT_FROM:
 | |
|             rval = make_tree_node(MAXOP_COLUMNS,
 | |
|                                   text,
 | |
|                                   NULL,
 | |
|                                   NULL);
 | |
|             break;
 | |
| 
 | |
|         default:
 | |
|             break;
 | |
|         }
 | |
|         break;
 | |
| 
 | |
|     case LT_STAR:
 | |
|         if (lookahead != LT_FROM)
 | |
|         {
 | |
|             rval = make_tree_node(MAXOP_ALL_COLUMNS,
 | |
|                                   NULL,
 | |
|                                   NULL,
 | |
|                                   NULL);
 | |
|         }
 | |
|         break;
 | |
| 
 | |
|     default:
 | |
|         break;
 | |
|     }
 | |
|     MXS_FREE(text);
 | |
|     MXS_FREE(text2);
 | |
|     return rval;
 | |
| }
 | |
| 
 | |
| 
 | |
| /**
 | |
|  * Parse a table name
 | |
|  *
 | |
|  * @param sql   Pointer to pointer to column list updated to point to the table name
 | |
|  * @return  A tree of table names
 | |
|  */
 | |
| static MAXINFO_TREE* parse_table_name(char** ptr)
 | |
| {
 | |
|     int token;
 | |
|     char* text;
 | |
|     MAXINFO_TREE* tree = NULL;
 | |
| 
 | |
|     *ptr = fetch_token(*ptr, &token, &text);
 | |
|     if (token == LT_STRING)
 | |
|     {
 | |
|         return make_tree_node(MAXOP_TABLE, text, NULL, NULL);
 | |
|     }
 | |
|     MXS_FREE(text);
 | |
|     return NULL;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Allocate and populate a parse tree node
 | |
|  *
 | |
|  * @param op    The node operator
 | |
|  * @param value The node value
 | |
|  * @param left  The left branch of the parse tree
 | |
|  * @param right The right branch of the parse tree
 | |
|  * @return The new parse tree node
 | |
|  */
 | |
| static MAXINFO_TREE* make_tree_node(MAXINFO_OPERATOR op, char* value, MAXINFO_TREE* left, MAXINFO_TREE* right)
 | |
| {
 | |
|     MAXINFO_TREE* node;
 | |
| 
 | |
|     if ((node = (MAXINFO_TREE*)MXS_MALLOC(sizeof(MAXINFO_TREE))) == NULL)
 | |
|     {
 | |
|         return NULL;
 | |
|     }
 | |
|     node->op = op;
 | |
|     node->value = value;
 | |
|     node->left = left;
 | |
|     node->right = right;
 | |
| 
 | |
|     return node;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Recursively free the storage associated with a parse tree
 | |
|  *
 | |
|  * @param tree  The parse tree to free
 | |
|  */
 | |
| void maxinfo_free_tree(MAXINFO_TREE* tree)
 | |
| {
 | |
|     if (tree->left)
 | |
|     {
 | |
|         maxinfo_free_tree(tree->left);
 | |
|     }
 | |
|     if (tree->right)
 | |
|     {
 | |
|         maxinfo_free_tree(tree->right);
 | |
|     }
 | |
|     if (tree->value)
 | |
|     {
 | |
|         MXS_FREE(tree->value);
 | |
|     }
 | |
|     MXS_FREE(tree);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * The set of keywords known to the tokeniser
 | |
|  */
 | |
| static struct
 | |
| {
 | |
|     const char* text;
 | |
|     int         token;
 | |
| } keywords[] =
 | |
| {
 | |
|     {"show",     LT_SHOW    },
 | |
|     {"select",   LT_SELECT  },
 | |
|     {"from",     LT_FROM    },
 | |
|     {"like",     LT_LIKE    },
 | |
|     {"=",        LT_EQUAL   },
 | |
|     {",",        LT_COMMA   },
 | |
|     {"*",        LT_STAR    },
 | |
|     {"flush",    LT_FLUSH   },
 | |
|     {"set",      LT_SET     },
 | |
|     {"clear",    LT_CLEAR   },
 | |
|     {"shutdown", LT_SHUTDOWN},
 | |
|     {"restart",  LT_RESTART },
 | |
|     {NULL,       0          }
 | |
| };
 | |
| 
 | |
| /**
 | |
|  * Limited SQL tokeniser. Understands a limited set of key words and
 | |
|  * quoted strings.
 | |
|  *
 | |
|  * @param sql   The SQL to tokenise
 | |
|  * @param token The returned token
 | |
|  * @param text  The matching text
 | |
|  * @return  The next position to tokenise from
 | |
|  */
 | |
| static char* fetch_token(char* sql, int* token, char** text)
 | |
| {
 | |
|     char* s1, * s2, quote = '\0';
 | |
|     int i;
 | |
| 
 | |
|     s1 = sql;
 | |
|     while (*s1 && isspace(*s1))
 | |
|     {
 | |
|         s1++;
 | |
|     }
 | |
|     if (quote == '\0' && (*s1 == '\'' || *s1 == '\"'))
 | |
|     {
 | |
|         quote = *s1++;
 | |
|     }
 | |
|     if (*s1 == '/' && *(s1 + 1) == '*')
 | |
|     {
 | |
|         s1 += 2;
 | |
|         // Skip the comment
 | |
|         do
 | |
|         {
 | |
|             while (*s1 && *s1 != '*')
 | |
|             {
 | |
|                 s1++;
 | |
|             }
 | |
|         }
 | |
|         while (*(s1 + 1) && *(s1 + 1) != '/');
 | |
|         s1 += 2;
 | |
|         while (*s1 && isspace(*s1))
 | |
|         {
 | |
|             s1++;
 | |
|         }
 | |
|         if (quote == '\0' && (*s1 == '\'' || *s1 == '\"'))
 | |
|         {
 | |
|             quote = *s1++;
 | |
|         }
 | |
|     }
 | |
|     s2 = s1;
 | |
|     while (*s2)
 | |
|     {
 | |
|         if (quote == '\0' && (isspace(*s2)
 | |
|                               || *s2 == ',' || *s2 == '='))
 | |
|         {
 | |
|             break;
 | |
|         }
 | |
|         else if (quote == *s2)
 | |
|         {
 | |
|             break;
 | |
|         }
 | |
|         s2++;
 | |
|     }
 | |
| 
 | |
|     if (*s1 == '@' && *(s1 + 1) == '@')
 | |
|     {
 | |
|         *text = strndup(s1 + 2, (s2 - s1) - 2);
 | |
|         *token = LT_VARIABLE;
 | |
|         return s2;
 | |
|     }
 | |
| 
 | |
|     if (s1 == s2)
 | |
|     {
 | |
|         *text = NULL;
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     *text = strndup(s1, s2 - s1);
 | |
|     for (i = 0; keywords[i].text; i++)
 | |
|     {
 | |
|         if (strcasecmp(keywords[i].text, *text) == 0)
 | |
|         {
 | |
|             *token = keywords[i].token;
 | |
|             return s2;
 | |
|         }
 | |
|     }
 | |
|     *token = LT_STRING;
 | |
|     return s2;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Parse the remaining arguments as literals.
 | |
|  * @param tree Previous head of the parse tree
 | |
|  * @param min_args Minimum required number of arguments
 | |
|  * @param ptr Pointer to client command
 | |
|  * @param parse_error Pointer to parsing error to fill
 | |
|  * @return Parsed tree or NULL if parsing failed
 | |
|  */
 | |
| MAXINFO_TREE* maxinfo_parse_literals(MAXINFO_TREE* tree,
 | |
|                                      int min_args,
 | |
|                                      char* ptr,
 | |
|                                      PARSE_ERROR* parse_error)
 | |
| {
 | |
|     int token;
 | |
|     MAXINFO_TREE* node = tree;
 | |
|     char* text;
 | |
|     for (int i = 0; i < min_args; i++)
 | |
|     {
 | |
|         if ((ptr = fetch_token(ptr, &token, &text)) == NULL
 | |
|             || (node->right = make_tree_node(MAXOP_LITERAL, text, NULL, NULL)) == NULL)
 | |
|         {
 | |
|             *parse_error = PARSE_SYNTAX_ERROR;
 | |
|             maxinfo_free_tree(tree);
 | |
|             if (ptr)
 | |
|             {
 | |
|                 MXS_FREE(text);
 | |
|             }
 | |
|             return NULL;
 | |
|         }
 | |
|         node = node->right;
 | |
|     }
 | |
| 
 | |
|     return tree;
 | |
| }
 | 
