aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--src/libutil/expression.c122
-rw-r--r--test/lua/unit/expressions.lua2
2 files changed, 68 insertions, 56 deletions
diff --git a/src/libutil/expression.c b/src/libutil/expression.c
index 6d4b33b5b..70da0deec 100644
--- a/src/libutil/expression.c
+++ b/src/libutil/expression.c
@@ -325,6 +325,67 @@ rspamd_expr_is_operation_symbol (gchar a)
return FALSE;
}
+static gboolean
+rspamd_expr_is_operation (struct rspamd_expression *e,
+ const gchar *p, const gchar *end, rspamd_regexp_t *num_re)
+{
+ if (rspamd_expr_is_operation_symbol (*p)) {
+ if (p + 1 < end) {
+ gchar t = *(p + 1);
+
+ if (t == ':') {
+ /* Special case, treat it as an atom */
+ }
+ else if (*p == '/') {
+ /* Lookahead for division operation to distinguish from regexp */
+ const gchar *track = p + 1;
+
+ /* Skip spaces */
+ while (track < end && g_ascii_isspace (*track)) {
+ track++;
+ }
+
+ /* Check for a number */
+ if (rspamd_regexp_search (num_re,
+ track,
+ end - track,
+ NULL,
+ NULL,
+ FALSE,
+ NULL)) {
+ msg_debug_expression ("found divide operation");
+ return TRUE;
+ }
+
+ msg_debug_expression ("false divide operation");
+ /* Fallback to PARSE_ATOM state */
+ }
+ else if (*p == '-') {
+ /* - is used in composites, so we need to distinguish - from
+ * 1) unary minus of a limit!
+ * 2) -BLAH in composites
+ * Decision is simple: require a space after binary `-` op
+ */
+ if (g_ascii_isspace (t)) {
+ return TRUE;
+ }
+ /* Fallback to PARSE_ATOM state */
+ msg_debug_expression ("false minus operation");
+ }
+ else {
+ /* Generic operation */
+ return TRUE;
+ }
+ }
+ else {
+ /* Last op */
+ return TRUE;
+ }
+ }
+
+ return FALSE;
+}
+
/* Return character representation of operation */
static enum rspamd_expression_op
rspamd_expr_str_to_op (const gchar *a, const gchar *end, const gchar **next)
@@ -784,62 +845,10 @@ rspamd_parse_expression (const gchar *line, gsize len,
state = SKIP_SPACES;
continue;
}
- else if (rspamd_expr_is_operation_symbol (*p)) {
+ else if (rspamd_expr_is_operation (e, p, end, num_re)) {
/* Lookahead */
- if (p + 1 < end) {
- gchar t = *(p + 1);
-
- if (t == ':') {
- /* Special case, treat it as an atom */
- }
- else if (*p == '/') {
- /* Lookahead for division operation to distinguish from regexp */
- const gchar *track = p + 1;
-
- /* Skip spaces */
- while (track < end && g_ascii_isspace (*track)) {
- track++;
- }
-
- /* Check for a number */
- if (rspamd_regexp_search (num_re,
- track,
- end - track,
- NULL,
- NULL,
- FALSE,
- NULL)) {
- state = PARSE_OP;
- msg_debug_expression ("found divide operation");
- continue;
- }
-
- msg_debug_expression ("false divide operation");
- /* Fallback to PARSE_ATOM state */
- }
- else if (*p == '-') {
- /* - is used in composites, so we need to distinguish - from
- * 1) unary minus of a limit!
- * 2) -BLAH in composites
- * Decision is simple: require a space after binary `-` op
- */
- if (g_ascii_isspace (t)) {
- state = PARSE_OP;
- continue;
- }
- /* Fallback to PARSE_ATOM state */
- msg_debug_expression ("false minus operation");
- }
- else {
- /* Generic operation */
- state = PARSE_OP;
- continue;
- }
- }
- else {
- state = PARSE_OP;
- continue;
- }
+ state = PARSE_OP;
+ continue;
}
/*
@@ -1078,7 +1087,8 @@ rspamd_parse_expression (const gchar *line, gsize len,
if (g_ascii_isspace (*p)) {
p ++;
}
- else if (rspamd_expr_is_operation_symbol (*p)) {
+ if (rspamd_expr_is_operation (e, p, end, num_re)) {
+ /* Lookahead */
state = PARSE_OP;
}
else {
diff --git a/test/lua/unit/expressions.lua b/test/lua/unit/expressions.lua
index c3a771c48..f4f4d2b58 100644
--- a/test/lua/unit/expressions.lua
+++ b/test/lua/unit/expressions.lua
@@ -60,6 +60,8 @@ context("Rspamd expressions", function()
{'(A) & (B) & ((C) | (D) | (E) | (F))', '(A) (B) (C) (D) (E) (F) |(4) &(3)' },
-- Extra space
{'A & B | ! C', '(C) ! (A) (B) & |'},
+ -- False minus
+ {'A + B + -C', '(A) (B) (-C) +(3)'},
}
for _,c in ipairs(cases) do
test("Expression creation function: " .. c[1], function()