diff options
author | Vsevolod Stakhov <vsevolod@rambler-co.ru> | 2011-06-06 15:51:52 +0400 |
---|---|---|
committer | Vsevolod Stakhov <vsevolod@rambler-co.ru> | 2011-06-06 15:51:52 +0400 |
commit | 87143d06278ea1a3bda810d606b9e86b7bd1ba3e (patch) | |
tree | 9b6b1b6612410e176165ad921717edcdd82ca398 | |
parent | 92de380c2c5e8ce7073ce979df4e5c7868e52bb6 (diff) | |
download | rspamd-87143d06278ea1a3bda810d606b9e86b7bd1ba3e.tar.gz rspamd-87143d06278ea1a3bda810d606b9e86b7bd1ba3e.zip |
* Add more convinient operations into expressions ('and', 'or', 'not')
-rw-r--r-- | src/expressions.c | 106 |
1 files changed, 87 insertions, 19 deletions
diff --git a/src/expressions.c b/src/expressions.c index 8c90160d2..e590ad630 100644 --- a/src/expressions.c +++ b/src/expressions.c @@ -224,18 +224,74 @@ logic_priority (gchar a) * Return TRUE if symbol is operation symbol */ static gboolean -is_operation_symbol (gchar a) +is_operation_symbol (gchar *a) { - switch (a) { + switch (*a) { case '!': case '&': case '|': case '(': case ')': return TRUE; - default: - return FALSE; + case 'O': + case 'o': + if (g_ascii_strncasecmp (a, "or", sizeof ("or") - 1) == 0&& g_ascii_isspace (a[2])) { + return TRUE; + } + break; + case 'A': + case 'a': + if (g_ascii_strncasecmp (a, "and", sizeof ("and") - 1) == 0&& g_ascii_isspace (a[3])) { + return TRUE; + } + break; + case 'N': + case 'n': + if (g_ascii_strncasecmp (a, "not", sizeof ("not") - 1) == 0 && g_ascii_isspace (a[3])) { + return TRUE; + } + break; + } + + return FALSE; +} + +/* Return character representation of operation */ +static gchar +op_to_char (gchar *a, gchar **next) +{ + switch (*a) { + case '!': + case '&': + case '|': + case '(': + case ')': + *next = a + 1; + return *a; + case 'O': + case 'o': + if (g_ascii_strncasecmp (a, "or", sizeof ("or") - 1) == 0) { + *next = a + sizeof ("or") - 1; + return '|'; + } + break; + case 'A': + case 'a': + if (g_ascii_strncasecmp (a, "and", sizeof ("and") - 1) == 0) { + *next = a + sizeof ("and") - 1; + return '&'; + } + break; + case 'N': + case 'n': + if (g_ascii_strncasecmp (a, "not", sizeof ("not") - 1) == 0) { + *next = a + sizeof ("not") - 1; + return '!'; + } + break; } + + return '\0'; } /* @@ -300,7 +356,7 @@ maybe_parse_expression (memory_pool_t * pool, gchar *line) gchar *p = line; while (*p) { - if (is_operation_symbol (*p)) { + if (is_operation_symbol (p)) { return parse_expression (pool, line); } p++; @@ -326,7 +382,7 @@ parse_expression (memory_pool_t * pool, gchar *line) struct expression_function *func = NULL; struct expression *arg; GQueue *function_stack; - gchar *p, *c, *str, op, *copy; + gchar *p, *c, *str, op, newop, *copy, *next; gboolean in_regexp = FALSE; gint brackets = 0; @@ -353,7 +409,7 @@ parse_expression (memory_pool_t * pool, gchar *line) switch (state) { case SKIP_SPACES: if (!g_ascii_isspace (*p)) { - if (is_operation_symbol (*p)) { + if (is_operation_symbol (p)) { state = READ_OPERATOR; } else if (*p == '/') { @@ -391,23 +447,35 @@ parse_expression (memory_pool_t * pool, gchar *line) } else { if (stack == NULL) { - stack = push_expression_stack (pool, stack, *p); + newop = op_to_char (p, &next); + if (newop != '\0') { + stack = push_expression_stack (pool, stack, newop); + p = next; + state = SKIP_SPACES; + continue; + } } /* Check priority of logic operation */ else { - if (logic_priority (stack->op) < logic_priority (*p)) { - stack = push_expression_stack (pool, stack, *p); - } - else { - /* Pop all operations that have higher priority than this one */ - while ((stack != NULL) && (logic_priority (stack->op) >= logic_priority (*p))) { - op = delete_expression_stack (&stack); - if (op != '(') { - insert_expression (pool, &expr, EXPR_OPERATION, op, NULL, copy); + newop = op_to_char (p, &next); + if (newop != '\0') { + if (logic_priority (stack->op) < logic_priority (newop)) { + stack = push_expression_stack (pool, stack, newop); + } + else { + /* Pop all operations that have higher priority than this one */ + while ((stack != NULL) && (logic_priority (stack->op) >= logic_priority (newop))) { + op = delete_expression_stack (&stack); + if (op != '(') { + insert_expression (pool, &expr, EXPR_OPERATION, op, NULL, copy); + } } + stack = push_expression_stack (pool, stack, newop); } - stack = push_expression_stack (pool, stack, *p); } + p = next; + state = SKIP_SPACES; + continue; } } p++; @@ -466,7 +534,7 @@ parse_expression (memory_pool_t * pool, gchar *line) insert_expression (pool, &expr, EXPR_FUNCTION, 0, func, copy); c = ++p; } - else if (is_operation_symbol (*p)) { + else if (is_operation_symbol (p)) { /* In fact it is not function, but symbol */ if (c != p) { str = memory_pool_alloc (pool, p - c + 1); |