2005-02-17 [paul] 1.0.1cvs11.4
[claws.git] / src / matcher_parser_parse.y
index c8c4db58586093843b294ceb24d9d90a9506347a..abe5b2ab1b3b927eb61117d02d3f51b4a4242a4f 100644 (file)
+
 %{
+
+/*
+ * Sylpheed -- a GTK+ based, lightweight, and fast e-mail client
+ * Copyright (c) 2001-2002 by Hiroyuki Yamamoto & The Sylpheed Claws Team.
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+ */
+
+#include "defs.h"
+
+#include <glib.h>
+#include <glib/gi18n.h>
+
+#include "utils.h"
 #include "filtering.h"
-#include "scoring.h"
 #include "matcher.h"
 #include "matcher_parser.h"
 #include "matcher_parser_lex.h"
-#include "intl.h"
-#include <glib.h>
+#include "procmsg.h"
+
+#define MAX_COLORLABELS (MSG_CLABEL_7 - MSG_CLABEL_NONE)
 
 static gint error = 0;
 static gint bool_op = 0;
 static gint match_type = 0;
-static gchar * header = NULL;
+static gchar *header = NULL;
+
+static MatcherProp *prop;
+
+static GSList *matchers_list = NULL;
+
+static MatcherList *cond;
+static GSList *action_list = NULL;
+static FilteringAction *action = NULL;
 
-static MatcherProp * prop;
+static FilteringProp *filtering;
 
-static GSList * matchers_list = NULL;
+static GSList **prefs_filtering = NULL;
+static int enable_compatibility = 0;
 
-static MatcherList * cond;
-static gint score = 0;
-static FilteringAction * action = NULL;
+enum {
+        MATCHER_PARSE_FILE,
+        MATCHER_PARSE_NO_EOL,
+        MATCHER_PARSE_CONDITION,
+        MATCHER_PARSE_FILTERING_ACTION,
+};
 
-static FilteringProp *  filtering;
-static ScoringProp * scoring = NULL;
+static int matcher_parse_op = MATCHER_PARSE_FILE;
 
-static GSList ** prefs_scoring = NULL;
-static GSList ** prefs_filtering = NULL;
 
-static int matcher_parser_dialog = 0;
+/* ******************************************************************** */
 
-FilteringProp * matcher_parser_get_filtering(gchar * str)
+void matcher_parser_start_parsing(FILE *f)
 {
-       void * bufstate;
+       matcher_parserrestart(f);
+       matcher_parserparse();
+}
 
+void * matcher_parser_scan_string(const char * str);
+FilteringProp *matcher_parser_get_filtering(gchar *str)
+{
+       void *bufstate;
+
+       /* bad coding to enable the sub-grammar matching
+          in yacc */
        matcher_parserlineno = 1;
-       matcher_parser_dialog = 1;
-       bufstate = matcher_parser_scan_string(str);
-       matcher_parserparse();
-       matcher_parser_dialog = 0;
+       matcher_parse_op = MATCHER_PARSE_NO_EOL;
+       matcher_parserrestart(NULL);
+        matcher_parser_init();
+       bufstate = matcher_parser_scan_string((const char *) str);
+        matcher_parser_switch_to_buffer(bufstate);
+       if (matcher_parserparse() != 0)
+               filtering = NULL;
+       matcher_parse_op = MATCHER_PARSE_FILE;
        matcher_parser_delete_buffer(bufstate);
        return filtering;
 }
 
-ScoringProp * matcher_parser_get_scoring(gchar * str)
+static gboolean check_quote_symetry(gchar *str)
+{
+       const gchar *walk;
+       int ret = 0;
+       
+       if (str == NULL)
+               return TRUE; /* heh, that's symetric */
+       if (*str == '\0')
+               return TRUE;
+       for (walk = str; *walk; walk++) {
+               if (*walk == '\"') {
+                       if (walk == str         /* first char */
+                       || *(walk - 1) != '\\') /* not escaped */
+                               ret ++;
+               }
+       }
+       return !(ret % 2);
+}
+
+MatcherList *matcher_parser_get_cond(gchar *str)
 {
-       void * bufstate;
+       void *bufstate;
 
+       if (!check_quote_symetry(str)) {
+               cond = NULL;
+               return cond;
+       }
+       
+       /* bad coding to enable the sub-grammar matching
+          in yacc */
        matcher_parserlineno = 1;
-       matcher_parser_dialog = 1;
+       matcher_parse_op = MATCHER_PARSE_CONDITION;
+       matcher_parserrestart(NULL);
+        matcher_parser_init();
        bufstate = matcher_parser_scan_string(str);
        matcher_parserparse();
-       matcher_parser_dialog = 0;
+       matcher_parse_op = MATCHER_PARSE_FILE;
        matcher_parser_delete_buffer(bufstate);
-       return scoring;
+       return cond;
 }
 
-MatcherList * matcher_parser_get_cond(gchar * str)
+GSList *matcher_parser_get_action_list(gchar *str)
 {
-       void * bufstate;
+       void *bufstate;
 
+       if (!check_quote_symetry(str)) {
+               action_list = NULL;
+               return action_list;
+       }
+       
+       /* bad coding to enable the sub-grammar matching
+          in yacc */
        matcher_parserlineno = 1;
-       matcher_parser_dialog = 1;
+       matcher_parse_op = MATCHER_PARSE_FILTERING_ACTION;
+       matcher_parserrestart(NULL);
+        matcher_parser_init();
        bufstate = matcher_parser_scan_string(str);
        matcher_parserparse();
-       matcher_parser_dialog = 0;
+       matcher_parse_op = MATCHER_PARSE_FILE;
        matcher_parser_delete_buffer(bufstate);
-       return cond;
+       return action_list;
 }
 
-MatcherProp * matcher_parser_get_prop(gchar * str)
+MatcherProp *matcher_parser_get_prop(gchar *str)
 {
-       MatcherList * list;
-       MatcherProp * prop;
+       MatcherList *list;
+       MatcherProp *prop;
 
        matcher_parserlineno = 1;
        list = matcher_parser_get_cond(str);
@@ -87,22 +175,25 @@ MatcherProp * matcher_parser_get_prop(gchar * str)
 
        g_slist_free(list->matchers);
        g_free(list);
+
+       return prop;
 }
 
-void matcher_parsererror(char * str)
+void matcher_parsererror(char *str)
 {
-       GSList * l;
+       GSList *l;
 
        if (matchers_list) {
-               for(l = matchers_list ; l != NULL ;
-                   l = g_slist_next(l))
+               for (l = matchers_list; l != NULL; l = g_slist_next(l)) {
                        matcherprop_free((MatcherProp *)
                                         l->data);
+                       l->data = NULL;
+               }
                g_slist_free(matchers_list);
                matchers_list = NULL;
        }
-
-       g_warning(_("scoring / filtering parsing: %i: %s\n"),
+       cond = NULL;
+       g_warning("filtering parsing: %i: %s\n",
                  matcher_parserlineno, str);
        error = 1;
 }
@@ -114,7 +205,7 @@ int matcher_parserwrap(void)
 %}
 
 %union {
-       char * str;
+       char *str;
        int value;
 }
 
@@ -131,26 +222,36 @@ int matcher_parserwrap(void)
 %token MATCHER_SCORE_LOWER  MATCHER_HEADER  MATCHER_NOT_HEADER
 %token MATCHER_HEADERS_PART  MATCHER_NOT_HEADERS_PART  MATCHER_MESSAGE
 %token MATCHER_NOT_MESSAGE  MATCHER_BODY_PART  MATCHER_NOT_BODY_PART
-%token MATCHER_EXECUTE  MATCHER_NOT_EXECUTE  MATCHER_MATCHCASE  MATCHER_MATCH
+%token MATCHER_TEST  MATCHER_NOT_TEST  MATCHER_MATCHCASE  MATCHER_MATCH
 %token MATCHER_REGEXPCASE  MATCHER_REGEXP  MATCHER_SCORE  MATCHER_MOVE
+%token MATCHER_ANY_IN_ADDRESSBOOK MATCHER_ALL_IN_ADDRESSBOOK
 %token MATCHER_COPY  MATCHER_DELETE  MATCHER_MARK  MATCHER_UNMARK
+%token MATCHER_LOCK MATCHER_UNLOCK
+%token MATCHER_EXECUTE
 %token MATCHER_MARK_AS_READ  MATCHER_MARK_AS_UNREAD  MATCHER_FORWARD
 %token MATCHER_FORWARD_AS_ATTACHMENT  MATCHER_EOL  MATCHER_STRING  
 %token MATCHER_OR MATCHER_AND  
+%token MATCHER_COLOR MATCHER_SCORE_EQUAL MATCHER_REDIRECT 
+%token MATCHER_SIZE_GREATER MATCHER_SIZE_SMALLER MATCHER_SIZE_EQUAL
+%token MATCHER_LOCKED MATCHER_NOT_LOCKED
+%token MATCHER_PARTIAL MATCHER_NOT_PARTIAL
+%token MATCHER_COLORLABEL MATCHER_NOT_COLORLABEL
+%token MATCHER_IGNORE_THREAD MATCHER_NOT_IGNORE_THREAD
+%token MATCHER_CHANGE_SCORE MATCHER_SET_SCORE
+%token MATCHER_STOP MATCHER_HIDE
 
 %start file
 
 %token <str> MATCHER_STRING
 %token <str> MATCHER_SECTION
-%token <value> MATCHER_INTEGER
+%token <str> MATCHER_INTEGER
 
 %%
 
 file:
 {
-       if (!matcher_parser_dialog) {
-               prefs_scoring = &global_scoring;
-               prefs_filtering = &global_filtering;
+       if (matcher_parse_op == MATCHER_PARSE_FILE) {
+               prefs_filtering = &pre_global_processing;
        }
 }
 file_line_list;
@@ -163,66 +264,118 @@ file_line_list
 
 file_line:
 section_notification
-| instruction
-;
+| 
+{ action_list = NULL; }
+instruction
+| error MATCHER_EOL
+{
+       yyerrok;
+};
 
 section_notification:
 MATCHER_SECTION MATCHER_EOL
 {
-       gchar * folder = $1;
-       FolderItem * item = NULL;
-
-       if (!matcher_parser_dialog) {
-               item = folder_find_item_from_identifier(folder);
-               if (item == NULL) {
-                       prefs_scoring = &global_scoring;
-                       prefs_filtering = &global_scoring;
+       gchar *folder = $1;
+       FolderItem *item = NULL;
+
+       if (matcher_parse_op == MATCHER_PARSE_FILE) {
+                enable_compatibility = 0;
+               if (!strcmp(folder, "global")) {
+                        /* backward compatibility */
+                        enable_compatibility = 1;
+                }
+               else if (!strcmp(folder, "preglobal")) {
+                       prefs_filtering = &pre_global_processing;
+                }
+               else if (!strcmp(folder, "postglobal")) {
+                       prefs_filtering = &post_global_processing;
+                }
+               else if (!strcmp(folder, "filtering")) {
+                        prefs_filtering = &filtering_rules;
                }
-               else {
-                       prefs_scoring = &item->prefs->scoring;
-                       prefs_filtering = &item->prefs->processing;
+                else {
+                       item = folder_find_item_from_identifier(folder);
+                       if (item != NULL) {
+                               prefs_filtering = &item->prefs->processing;
+                       } else {
+                               prefs_filtering = NULL;
+                       }
                }
        }
 }
 ;
 
 instruction:
-condition end_instr_opt
-| MATCHER_EOL
-;
-
-end_instr_opt:
-filtering_or_scoring MATCHER_EOL
-|
+condition filtering MATCHER_EOL
+| condition filtering
+{
+       if (matcher_parse_op == MATCHER_PARSE_NO_EOL)
+               YYACCEPT;
+       else {
+               matcher_parsererror("parse error");
+               YYERROR;
+       }
+}
+| condition
 {
-       if (!matcher_parser_dialog) {
-               yyerror("parse error");
-               return 1;
+       if (matcher_parse_op == MATCHER_PARSE_CONDITION)
+               YYACCEPT;
+       else {
+               matcher_parsererror("parse error");
+               YYERROR;
        }
 }
+| filtering_action_list
+{
+       if (matcher_parse_op == MATCHER_PARSE_FILTERING_ACTION)
+               YYACCEPT;
+       else {
+               matcher_parsererror("parse error");
+               YYERROR;
+       }
+}
+| MATCHER_EOL
 ;
 
-filtering_or_scoring:
-filtering_action
-{
-       filtering = filteringprop_new(cond, action);
+filtering:
+filtering_action_list
+{
+       filtering = filteringprop_new(cond, action_list);
+        
+        if (enable_compatibility) {
+                prefs_filtering = &filtering_rules;
+                if (action_list != NULL) {
+                        FilteringAction * first_action;
+                        
+                        first_action = action_list->data;
+                        
+                        if (first_action->type == MATCHACTION_CHANGE_SCORE)
+                                prefs_filtering = &pre_global_processing;
+                }
+        }
+        
        cond = NULL;
-       action = NULL;
-       if (!matcher_parser_dialog) {
-               * prefs_filtering = g_slist_append(* prefs_filtering,
-                                                  filtering);
+       action_list = NULL;
+        
+       if ((matcher_parse_op == MATCHER_PARSE_FILE) &&
+            (prefs_filtering != NULL)) {
+               *prefs_filtering = g_slist_append(*prefs_filtering,
+                                                 filtering);
                filtering = NULL;
        }
 }
-| scoring_rule
+;
+
+filtering_action_list:
+filtering_action_b filtering_action_list
+| filtering_action_b
+;
+
+filtering_action_b:
+filtering_action
 {
-       scoring = scoringprop_new(cond, score);
-       cond = NULL;
-       score = 0;
-       if (!matcher_parser_dialog) {
-               * prefs_scoring = g_slist_append(* prefs_scoring, scoring);
-               scoring = NULL;
-       }
+        action_list = g_slist_append(action_list, action);
+        action = NULL;
 }
 ;
 
@@ -243,6 +396,14 @@ MATCHER_MATCHCASE
 {
        match_type = MATCHTYPE_REGEXP;
 }
+| MATCHER_ANY_IN_ADDRESSBOOK
+{
+       match_type = MATCHTYPE_ANY_IN_ADDRESSBOOK;
+}
+| MATCHER_ALL_IN_ADDRESSBOOK
+{
+       match_type = MATCHTYPE_ALL_IN_ADDRESSBOOK;
+}
 ;
 
 condition:
@@ -368,10 +529,74 @@ MATCHER_ALL
        criteria = MATCHCRITERIA_NOT_FORWARDED;
        prop = matcherprop_new(criteria, NULL, 0, NULL, 0);
 }
+| MATCHER_LOCKED
+{
+       gint criteria = 0;
+
+       criteria = MATCHCRITERIA_LOCKED;
+       prop = matcherprop_new(criteria, NULL, 0, NULL, 0);
+}
+| MATCHER_NOT_LOCKED
+{
+       gint criteria = 0;
+
+       criteria = MATCHCRITERIA_NOT_LOCKED;
+       prop = matcherprop_new(criteria, NULL, 0, NULL, 0);
+}
+| MATCHER_PARTIAL
+{
+       gint criteria = 0;
+
+       criteria = MATCHCRITERIA_PARTIAL;
+       prop = matcherprop_new(criteria, NULL, 0, NULL, 0);
+}
+| MATCHER_NOT_PARTIAL
+{
+       gint criteria = 0;
+
+       criteria = MATCHCRITERIA_NOT_PARTIAL;
+       prop = matcherprop_new(criteria, NULL, 0, NULL, 0);
+}
+| MATCHER_COLORLABEL MATCHER_INTEGER
+{
+       gint criteria = 0;
+       gint value = 0;
+
+       criteria = MATCHCRITERIA_COLORLABEL;
+       value = strtol($2, NULL, 10);
+       if (value < 0) value = 0;
+       else if (value > MAX_COLORLABELS) value = MAX_COLORLABELS;
+       prop = matcherprop_new(criteria, NULL, 0, NULL, value);
+}
+| MATCHER_NOT_COLORLABEL MATCHER_INTEGER
+{
+       gint criteria = 0;
+       gint value = 0;
+
+       criteria = MATCHCRITERIA_NOT_COLORLABEL;
+       value = strtol($2, NULL, 0);
+       if (value < 0) value = 0;
+       else if (value > MAX_COLORLABELS) value = MAX_COLORLABELS;
+       prop = matcherprop_new(criteria, NULL, 0, NULL, value);
+}
+| MATCHER_IGNORE_THREAD
+{
+       gint criteria = 0;
+
+       criteria = MATCHCRITERIA_IGNORE_THREAD;
+       prop = matcherprop_new(criteria, NULL, 0, NULL, 0);
+}
+| MATCHER_NOT_IGNORE_THREAD
+{
+       gint criteria = 0;
+
+       criteria = MATCHCRITERIA_NOT_IGNORE_THREAD;
+       prop = matcherprop_new(criteria, NULL, 0, NULL, 0);
+}
 | MATCHER_SUBJECT match_type MATCHER_STRING
 {
        gint criteria = 0;
-       gchar * expr = NULL;
+       gchar *expr = NULL;
 
        criteria = MATCHCRITERIA_SUBJECT;
        expr = $3;
@@ -380,7 +605,7 @@ MATCHER_ALL
 | MATCHER_NOT_SUBJECT match_type MATCHER_STRING
 {
        gint criteria = 0;
-       gchar * expr = NULL;
+       gchar *expr = NULL;
 
        criteria = MATCHCRITERIA_NOT_SUBJECT;
        expr = $3;
@@ -389,7 +614,7 @@ MATCHER_ALL
 | MATCHER_FROM match_type MATCHER_STRING
 {
        gint criteria = 0;
-       gchar * expr = NULL;
+       gchar *expr = NULL;
 
        criteria = MATCHCRITERIA_FROM;
        expr = $3;
@@ -398,7 +623,7 @@ MATCHER_ALL
 | MATCHER_NOT_FROM match_type MATCHER_STRING
 {
        gint criteria = 0;
-       gchar * expr = NULL;
+       gchar *expr = NULL;
 
        criteria = MATCHCRITERIA_NOT_FROM;
        expr = $3;
@@ -407,7 +632,7 @@ MATCHER_ALL
 | MATCHER_TO match_type MATCHER_STRING
 {
        gint criteria = 0;
-       gchar * expr = NULL;
+       gchar *expr = NULL;
 
        criteria = MATCHCRITERIA_TO;
        expr = $3;
@@ -416,7 +641,7 @@ MATCHER_ALL
 | MATCHER_NOT_TO match_type MATCHER_STRING
 {
        gint criteria = 0;
-       gchar * expr = NULL;
+       gchar *expr = NULL;
 
        criteria = MATCHCRITERIA_NOT_TO;
        expr = $3;
@@ -425,7 +650,7 @@ MATCHER_ALL
 | MATCHER_CC match_type MATCHER_STRING
 {
        gint criteria = 0;
-       gchar * expr = NULL;
+       gchar *expr = NULL;
 
        criteria = MATCHCRITERIA_CC;
        expr = $3;
@@ -434,7 +659,7 @@ MATCHER_ALL
 | MATCHER_NOT_CC match_type MATCHER_STRING
 {
        gint criteria = 0;
-       gchar * expr = NULL;
+       gchar *expr = NULL;
 
        criteria = MATCHCRITERIA_NOT_CC;
        expr = $3;
@@ -443,7 +668,7 @@ MATCHER_ALL
 | MATCHER_TO_OR_CC match_type MATCHER_STRING
 {
        gint criteria = 0;
-       gchar * expr = NULL;
+       gchar *expr = NULL;
 
        criteria = MATCHCRITERIA_TO_OR_CC;
        expr = $3;
@@ -452,7 +677,7 @@ MATCHER_ALL
 | MATCHER_NOT_TO_AND_NOT_CC match_type MATCHER_STRING
 {
        gint criteria = 0;
-       gchar * expr = NULL;
+       gchar *expr = NULL;
 
        criteria = MATCHCRITERIA_NOT_TO_AND_NOT_CC;
        expr = $3;
@@ -464,7 +689,7 @@ MATCHER_ALL
        gint value = 0;
 
        criteria = MATCHCRITERIA_AGE_GREATER;
-       value = atoi($2);
+       value = strtol($2, NULL, 0);
        prop = matcherprop_new(criteria, NULL, 0, NULL, value);
 }
 | MATCHER_AGE_LOWER MATCHER_INTEGER
@@ -473,13 +698,13 @@ MATCHER_ALL
        gint value = 0;
 
        criteria = MATCHCRITERIA_AGE_LOWER;
-       value = atoi($2);
+       value = strtol($2, NULL, 0);
        prop = matcherprop_new(criteria, NULL, 0, NULL, value);
 }
 | MATCHER_NEWSGROUPS match_type MATCHER_STRING
 {
        gint criteria = 0;
-       gchar * expr = NULL;
+       gchar *expr = NULL;
 
        criteria = MATCHCRITERIA_NEWSGROUPS;
        expr = $3;
@@ -488,7 +713,7 @@ MATCHER_ALL
 | MATCHER_NOT_NEWSGROUPS match_type MATCHER_STRING
 {
        gint criteria = 0;
-       gchar * expr = NULL;
+       gchar *expr = NULL;
 
        criteria = MATCHCRITERIA_NOT_NEWSGROUPS;
        expr = $3;
@@ -497,7 +722,7 @@ MATCHER_ALL
 | MATCHER_INREPLYTO match_type MATCHER_STRING
 {
        gint criteria = 0;
-       gchar * expr = NULL;
+       gchar *expr = NULL;
 
        criteria = MATCHCRITERIA_INREPLYTO;
        expr = $3;
@@ -506,7 +731,7 @@ MATCHER_ALL
 | MATCHER_NOT_INREPLYTO match_type MATCHER_STRING
 {
        gint criteria = 0;
-       gchar * expr = NULL;
+       gchar *expr = NULL;
 
        criteria = MATCHCRITERIA_NOT_INREPLYTO;
        expr = $3;
@@ -515,7 +740,7 @@ MATCHER_ALL
 | MATCHER_REFERENCES match_type MATCHER_STRING
 {
        gint criteria = 0;
-       gchar * expr = NULL;
+       gchar *expr = NULL;
 
        criteria = MATCHCRITERIA_REFERENCES;
        expr = $3;
@@ -524,7 +749,7 @@ MATCHER_ALL
 | MATCHER_NOT_REFERENCES match_type MATCHER_STRING
 {
        gint criteria = 0;
-       gchar * expr = NULL;
+       gchar *expr = NULL;
 
        criteria = MATCHCRITERIA_NOT_REFERENCES;
        expr = $3;
@@ -536,7 +761,7 @@ MATCHER_ALL
        gint value = 0;
 
        criteria = MATCHCRITERIA_SCORE_GREATER;
-       value = atoi($2);
+       value = strtol($2, NULL, 0);
        prop = matcherprop_new(criteria, NULL, 0, NULL, value);
 }
 | MATCHER_SCORE_LOWER MATCHER_INTEGER
@@ -545,7 +770,40 @@ MATCHER_ALL
        gint value = 0;
 
        criteria = MATCHCRITERIA_SCORE_LOWER;
-       value = atoi($2);
+       value = strtol($2, NULL, 0);
+       prop = matcherprop_new(criteria, NULL, 0, NULL, value);
+}
+| MATCHER_SCORE_EQUAL MATCHER_INTEGER
+{
+       gint criteria = 0;
+       gint value = 0;
+
+       criteria = MATCHCRITERIA_SCORE_EQUAL;
+       value = strtol($2, NULL, 0);
+       prop = matcherprop_new(criteria, NULL, 0, NULL, value);
+}
+| MATCHER_SIZE_GREATER MATCHER_INTEGER 
+{
+       gint criteria = 0;
+       gint value    = 0;
+       criteria = MATCHCRITERIA_SIZE_GREATER;
+       value = strtol($2, NULL, 0);
+       prop = matcherprop_new(criteria, NULL, 0, NULL, value);
+}
+| MATCHER_SIZE_SMALLER MATCHER_INTEGER
+{
+       gint criteria = 0;
+       gint value    = 0;
+       criteria = MATCHCRITERIA_SIZE_SMALLER;
+       value = strtol($2, NULL, 0);
+       prop = matcherprop_new(criteria, NULL, 0, NULL, value);
+}
+| MATCHER_SIZE_EQUAL MATCHER_INTEGER
+{
+       gint criteria = 0;
+       gint value    = 0;
+       criteria = MATCHCRITERIA_SIZE_EQUAL;
+       value = strtol($2, NULL, 0);
        prop = matcherprop_new(criteria, NULL, 0, NULL, value);
 }
 | MATCHER_HEADER MATCHER_STRING
@@ -554,7 +812,7 @@ MATCHER_ALL
 } match_type MATCHER_STRING
 {
        gint criteria = 0;
-       gchar * expr = NULL;
+       gchar *expr = NULL;
 
        criteria = MATCHCRITERIA_HEADER;
        expr = $2;
@@ -567,7 +825,7 @@ MATCHER_ALL
 } match_type MATCHER_STRING
 {
        gint criteria = 0;
-       gchar * expr = NULL;
+       gchar *expr = NULL;
 
        criteria = MATCHCRITERIA_NOT_HEADER;
        expr = $2;
@@ -577,7 +835,7 @@ MATCHER_ALL
 | MATCHER_HEADERS_PART match_type MATCHER_STRING
 {
        gint criteria = 0;
-       gchar * expr = NULL;
+       gchar *expr = NULL;
 
        criteria = MATCHCRITERIA_HEADERS_PART;
        expr = $3;
@@ -586,7 +844,7 @@ MATCHER_ALL
 | MATCHER_NOT_HEADERS_PART match_type MATCHER_STRING
 {
        gint criteria = 0;
-       gchar * expr = NULL;
+       gchar *expr = NULL;
 
        criteria = MATCHCRITERIA_NOT_HEADERS_PART;
        expr = $3;
@@ -595,7 +853,7 @@ MATCHER_ALL
 | MATCHER_MESSAGE match_type MATCHER_STRING
 {
        gint criteria = 0;
-       gchar * expr = NULL;
+       gchar *expr = NULL;
 
        criteria = MATCHCRITERIA_MESSAGE;
        expr = $3;
@@ -604,7 +862,7 @@ MATCHER_ALL
 | MATCHER_NOT_MESSAGE match_type MATCHER_STRING
 {
        gint criteria = 0;
-       gchar * expr = NULL;
+       gchar *expr = NULL;
 
        criteria = MATCHCRITERIA_NOT_MESSAGE;
        expr = $3;
@@ -613,7 +871,7 @@ MATCHER_ALL
 | MATCHER_BODY_PART match_type MATCHER_STRING
 {
        gint criteria = 0;
-       gchar * expr = NULL;
+       gchar *expr = NULL;
 
        criteria = MATCHCRITERIA_BODY_PART;
        expr = $3;
@@ -622,36 +880,27 @@ MATCHER_ALL
 | MATCHER_NOT_BODY_PART match_type MATCHER_STRING
 {
        gint criteria = 0;
-       gchar * expr = NULL;
+       gchar *expr = NULL;
 
        criteria = MATCHCRITERIA_NOT_BODY_PART;
        expr = $3;
        prop = matcherprop_new(criteria, NULL, match_type, expr, 0);
 }
-| MATCHER_NOT_MESSAGE match_type MATCHER_STRING
+| MATCHER_TEST MATCHER_STRING
 {
        gint criteria = 0;
-       gchar * expr = NULL;
+       gchar *expr = NULL;
 
-       criteria = MATCHCRITERIA_NOT_MESSAGE;
-       expr = $3;
-       prop = matcherprop_new(criteria, NULL, match_type, expr, 0);
-}
-| MATCHER_EXECUTE MATCHER_STRING
-{
-       gint criteria = 0;
-       gchar * expr = NULL;
-
-       criteria = MATCHCRITERIA_EXECUTE;
+       criteria = MATCHCRITERIA_TEST;
        expr = $2;
        prop = matcherprop_new(criteria, NULL, 0, expr, 0);
 }
-| MATCHER_NOT_EXECUTE MATCHER_STRING
+| MATCHER_NOT_TEST MATCHER_STRING
 {
        gint criteria = 0;
-       gchar * expr = NULL;
+       gchar *expr = NULL;
 
-       criteria = MATCHCRITERIA_NOT_EXECUTE;
+       criteria = MATCHCRITERIA_NOT_TEST;
        expr = $2;
        prop = matcherprop_new(criteria, NULL, 0, expr, 0);
 }
@@ -660,93 +909,156 @@ MATCHER_ALL
 filtering_action:
 MATCHER_EXECUTE MATCHER_STRING
 {
-       gchar * cmd = NULL;
+       gchar *cmd = NULL;
        gint action_type = 0;
 
        action_type = MATCHACTION_EXECUTE;
        cmd = $2;
-       action = filteringaction_new(action_type, 0, cmd);
+       action = filteringaction_new(action_type, 0, cmd, 0, 0);
 }
 | MATCHER_MOVE MATCHER_STRING
 {
-       gchar * destination = NULL;
+       gchar *destination = NULL;
        gint action_type = 0;
 
        action_type = MATCHACTION_MOVE;
        destination = $2;
-       action = filteringaction_new(action_type, 0, destination);
+       action = filteringaction_new(action_type, 0, destination, 0, 0);
 }
 | MATCHER_COPY MATCHER_STRING
 {
-       gchar * destination = NULL;
+       gchar *destination = NULL;
        gint action_type = 0;
 
        action_type = MATCHACTION_COPY;
        destination = $2;
-       action = filteringaction_new(action_type, 0, destination);
+       action = filteringaction_new(action_type, 0, destination, 0, 0);
 }
 | MATCHER_DELETE
 {
        gint action_type = 0;
 
        action_type = MATCHACTION_DELETE;
-       action = filteringaction_new(action_type, 0, NULL);
+       action = filteringaction_new(action_type, 0, NULL, 0, 0);
 }
 | MATCHER_MARK
 {
        gint action_type = 0;
 
        action_type = MATCHACTION_MARK;
-       action = filteringaction_new(action_type, 0, NULL);
+       action = filteringaction_new(action_type, 0, NULL, 0, 0);
 }
 | MATCHER_UNMARK
 {
        gint action_type = 0;
 
        action_type = MATCHACTION_UNMARK;
-       action = filteringaction_new(action_type, 0, NULL);
+       action = filteringaction_new(action_type, 0, NULL, 0, 0);
+}
+| MATCHER_LOCK
+{
+       gint action_type = 0;
+
+       action_type = MATCHACTION_LOCK;
+       action = filteringaction_new(action_type, 0, NULL, 0, 0);
+}
+| MATCHER_UNLOCK
+{
+       gint action_type = 0;
+
+       action_type = MATCHACTION_UNLOCK;
+       action = filteringaction_new(action_type, 0, NULL, 0, 0);
 }
 | MATCHER_MARK_AS_READ
 {
        gint action_type = 0;
 
        action_type = MATCHACTION_MARK_AS_READ;
-       action = filteringaction_new(action_type, 0, NULL);
+       action = filteringaction_new(action_type, 0, NULL, 0, 0);
 }
 | MATCHER_MARK_AS_UNREAD
 {
        gint action_type = 0;
 
        action_type = MATCHACTION_MARK_AS_UNREAD;
-       action = filteringaction_new(action_type, 0, NULL);
+       action = filteringaction_new(action_type, 0, NULL, 0, 0);
 }
 | MATCHER_FORWARD MATCHER_INTEGER MATCHER_STRING
 {
-       gchar * destination = NULL;
+       gchar *destination = NULL;
        gint action_type = 0;
        gint account_id = 0;
 
        action_type = MATCHACTION_FORWARD;
-       account_id = $2;
+       account_id = strtol($2, NULL, 10);
        destination = $3;
-       action = filteringaction_new(action_type, account_id, destination);
+       action = filteringaction_new(action_type,
+            account_id, destination, 0, 0);
 }
 | MATCHER_FORWARD_AS_ATTACHMENT MATCHER_INTEGER MATCHER_STRING
 {
-       gchar * destination = NULL;
+       gchar *destination = NULL;
        gint action_type = 0;
        gint account_id = 0;
 
        action_type = MATCHACTION_FORWARD_AS_ATTACHMENT;
-       account_id = $2;
+       account_id = strtol($2, NULL, 10);
        destination = $3;
-       action = filteringaction_new(action_type, account_id, destination);
+       action = filteringaction_new(action_type,
+            account_id, destination, 0, 0);
 }
-;
+| MATCHER_REDIRECT MATCHER_INTEGER MATCHER_STRING
+{
+       gchar *destination = NULL;
+       gint action_type = 0;
+       gint account_id = 0;
+
+       action_type = MATCHACTION_REDIRECT;
+       account_id = strtol($2, NULL, 10);
+       destination = $3;
+       action = filteringaction_new(action_type,
+            account_id, destination, 0, 0);
+}
+| MATCHER_COLOR MATCHER_INTEGER
+{
+       gint action_type = 0;
+       gint color = 0;
 
-scoring_rule:
-MATCHER_SCORE MATCHER_INTEGER
+       action_type = MATCHACTION_COLOR;
+       color = strtol($2, NULL, 10);
+       action = filteringaction_new(action_type, 0, NULL, color, 0);
+}
+| MATCHER_CHANGE_SCORE MATCHER_INTEGER
+{
+        gint score = 0;
+        
+        score = strtol($2, NULL, 10);
+       action = filteringaction_new(MATCHACTION_CHANGE_SCORE, 0,
+                                    NULL, 0, score);
+}
+/* backward compatibility */
+| MATCHER_SCORE MATCHER_INTEGER
+{
+        gint score = 0;
+        
+        score = strtol($2, NULL, 10);
+       action = filteringaction_new(MATCHACTION_CHANGE_SCORE, 0,
+                                    NULL, 0, score);
+}
+| MATCHER_SET_SCORE MATCHER_INTEGER
+{
+        gint score = 0;
+        
+        score = strtol($2, NULL, 10);
+       action = filteringaction_new(MATCHACTION_SET_SCORE, 0,
+                                    NULL, 0, score);
+}
+| MATCHER_HIDE
+{
+       action = filteringaction_new(MATCHACTION_HIDE, 0, NULL, 0, 0);
+}
+| MATCHER_STOP
 {
-       score = atoi($2);
+       action = filteringaction_new(MATCHACTION_STOP, 0, NULL, 0, 0);
 }
 ;