Code

plugin: Make sdb_plugin_info_t public.
[sysdb.git] / src / frontend / grammar.y
index 50bb00e7f632fa35abb283ca7025b0c02a80a908..34a9c6b6474502d9fc059df45a6ed93a79890567 100644 (file)
@@ -32,6 +32,7 @@
 #include "frontend/grammar.h"
 
 #include "core/store.h"
+#include "core/store-private.h"
 
 #include "utils/error.h"
 #include "utils/llist.h"
@@ -62,23 +63,43 @@ sdb_fe_yyerror(YYLTYPE *lval, sdb_fe_yyscan_t scanner, const char *msg);
 %locations
 %error-verbose
 %expect 0
-%name-prefix="sdb_fe_yy"
+%name-prefix "sdb_fe_yy"
 
 %union {
+       const char *sstr; /* static string */
        char *str;
 
+       sdb_data_t data;
+
        sdb_llist_t     *list;
        sdb_conn_node_t *node;
+
+       sdb_store_matcher_t *m;
 }
 
 %start statements
 
 %token SCANNER_ERROR
 
-%token WHERE
+%token AND OR NOT WHERE
+%token CMP_EQUAL CMP_NEQUAL CMP_REGEX CMP_NREGEX
+%token CMP_LT CMP_LE CMP_GE CMP_GT
+
+%token FETCH LIST LOOKUP
 
 %token <str> IDENTIFIER STRING
-%token <node> FETCH LIST LOOKUP
+
+%token <data> INTEGER FLOAT
+
+/* Precedence (lowest first): */
+%left OR
+%left AND
+%right NOT
+%left CMP_EQUAL CMP_NEQUAL
+%left CMP_LT CMP_LE CMP_GE CMP_GT
+%left CMP_REGEX CMP_NREGEX
+%left '(' ')'
+%left '.'
 
 %type <list> statements
 %type <node> statement
@@ -87,6 +108,16 @@ sdb_fe_yyerror(YYLTYPE *lval, sdb_fe_yyscan_t scanner, const char *msg);
        lookup_statement
        expression
 
+%type <m> matcher
+       compare_matcher
+
+%type <sstr> op
+
+%type <data> data
+
+%destructor { free($$); } <str>
+%destructor { sdb_object_deref(SDB_OBJ($$)); } <node> <m>
+
 %%
 
 statements:
@@ -97,6 +128,7 @@ statements:
                                sdb_fe_yyerror(&yylloc, scanner,
                                                YY_("syntax error, unexpected statement, "
                                                        "expecting expression"));
+                               sdb_object_deref(SDB_OBJ($3));
                                YYABORT;
                        }
 
@@ -113,6 +145,7 @@ statements:
                                sdb_fe_yyerror(&yylloc, scanner,
                                                YY_("syntax error, unexpected statement, "
                                                        "expecting expression"));
+                               sdb_object_deref(SDB_OBJ($1));
                                YYABORT;
                        }
 
@@ -129,6 +162,7 @@ statements:
                                sdb_fe_yyerror(&yylloc, scanner,
                                                YY_("syntax error, unexpected expression, "
                                                        "expecting statement"));
+                               sdb_object_deref(SDB_OBJ($1));
                                YYABORT;
                        }
 
@@ -164,8 +198,7 @@ fetch_statement:
                                                conn_fetch_t, conn_fetch_destroy));
                        CONN_FETCH($$)->name = strdup($2);
                        $$->cmd = CONNECTION_FETCH;
-                       free($2);
-                       $2 = NULL;
+                       free($2); $2 = NULL;
                }
        ;
 
@@ -197,6 +230,8 @@ lookup_statement:
                                snprintf(errmsg, sizeof(errmsg),
                                                YY_("unknown table %s"), $2);
                                sdb_fe_yyerror(&yylloc, scanner, errmsg);
+                               free($2); $2 = NULL;
+                               sdb_object_deref(SDB_OBJ($4));
                                YYABORT;
                        }
 
@@ -204,26 +239,100 @@ lookup_statement:
                                                conn_lookup_t, conn_lookup_destroy));
                        CONN_LOOKUP($$)->matcher = CONN_MATCHER($4);
                        $$->cmd = CONNECTION_LOOKUP;
-                       free($2);
-                       $2 = NULL;
+                       free($2); $2 = NULL;
                }
        ;
 
 expression:
-       STRING
+       matcher
                {
-                       $$ = SDB_CONN_NODE(sdb_object_create_T(/* name = */ NULL,
-                                               conn_node_matcher_t));
+                       if (! $1) {
+                               /* TODO: improve error reporting */
+                               sdb_fe_yyerror(&yylloc, scanner,
+                                               YY_("syntax error, invalid expression"));
+                               YYABORT;
+                       }
+
+                       $$ = SDB_CONN_NODE(sdb_object_create_dT(/* name = */ NULL,
+                                               conn_node_matcher_t, conn_matcher_destroy));
                        $$->cmd = CONNECTION_EXPR;
-                       /* XXX: this is just a placeholder for now */
-                       CONN_MATCHER($$)->matcher = sdb_store_host_matcher($1,
-                                       /* name_re = */ NULL, /* service = */ NULL,
-                                       /* attr = */ NULL);
-                       free($1);
-                       $1 = NULL;
+                       CONN_MATCHER($$)->matcher = $1;
+               }
+       ;
+
+matcher:
+       '(' matcher ')'
+               {
+                       $$ = $2;
+               }
+       |
+       matcher AND matcher
+               {
+                       $$ = sdb_store_con_matcher($1, $3);
+                       sdb_object_deref(SDB_OBJ($1));
+                       sdb_object_deref(SDB_OBJ($3));
+               }
+       |
+       matcher OR matcher
+               {
+                       $$ = sdb_store_dis_matcher($1, $3);
+                       sdb_object_deref(SDB_OBJ($1));
+                       sdb_object_deref(SDB_OBJ($3));
+               }
+       |
+       NOT matcher
+               {
+                       $$ = sdb_store_inv_matcher($2);
+                       sdb_object_deref(SDB_OBJ($2));
+               }
+       |
+       compare_matcher
+               {
+                       $$ = $1;
+               }
+       ;
+
+/*
+ * <object_type>.<object_attr> <op> <value>
+ *
+ * Parse matchers comparing object attributes with a value.
+ */
+compare_matcher:
+       IDENTIFIER '.' IDENTIFIER op data
+               {
+                       $$ = sdb_store_matcher_parse_cmp($1, $3, $4, &$5);
+                       free($1); $1 = NULL;
+                       free($3); $3 = NULL;
+                       sdb_data_free_datum(&$5);
                }
        ;
 
+op:
+       CMP_EQUAL { $$ = "="; }
+       |
+       CMP_NEQUAL { $$ = "!="; }
+       |
+       CMP_REGEX { $$ = "=~"; }
+       |
+       CMP_NREGEX { $$ = "!~"; }
+       |
+       CMP_LT { $$ = "<"; }
+       |
+       CMP_LE { $$ = "<="; }
+       |
+       CMP_GE { $$ = ">="; }
+       |
+       CMP_GT { $$ = ">"; }
+       ;
+
+data:
+       STRING { $$.type = SDB_TYPE_STRING; $$.data.string = $1; }
+       |
+       INTEGER { $$ = $1; }
+       |
+       FLOAT { $$ = $1; }
+       ;
+
 %%
 
 void