aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--src/libutil/expression.c125
-rw-r--r--test/lua/unit/expressions.lua1
2 files changed, 83 insertions, 43 deletions
diff --git a/src/libutil/expression.c b/src/libutil/expression.c
index 838a4ffff..d4559e765 100644
--- a/src/libutil/expression.c
+++ b/src/libutil/expression.c
@@ -86,6 +86,46 @@ rspamd_expr_quark (void)
return g_quark_from_static_string ("rspamd-expression");
}
+static const gchar *
+rspamd_expr_op_to_str (enum rspamd_expression_op op)
+{
+ const gchar *op_str = NULL;
+
+ switch (op) {
+ case OP_AND:
+ op_str = "&";
+ break;
+ case OP_OR:
+ op_str = "|";
+ break;
+ case OP_MULT:
+ op_str = "*";
+ break;
+ case OP_PLUS:
+ op_str = "+";
+ break;
+ case OP_NOT:
+ op_str = "!";
+ break;
+ case OP_GE:
+ op_str = ">=";
+ break;
+ case OP_GT:
+ op_str = ">";
+ break;
+ case OP_LE:
+ op_str = "<=";
+ break;
+ case OP_LT:
+ op_str = "<";
+ break;
+ default:
+ op_str = "???";
+ break;
+ }
+
+ return op_str;
+}
#define G_ARRAY_LAST(ar, type) (&g_array_index((ar), type, (ar)->len - 1))
@@ -96,6 +136,7 @@ rspamd_expr_stack_elt_push (GPtrArray *stack,
g_ptr_array_add (stack, elt);
}
+
static gpointer
rspamd_expr_stack_elt_pop (GPtrArray *stack)
{
@@ -325,8 +366,9 @@ rspamd_expression_destroy (struct rspamd_expression *expr)
}
}
-static void
-rspamd_ast_add_node (GPtrArray *operands, struct rspamd_expression_elt *op)
+static gboolean
+rspamd_ast_add_node (GPtrArray *operands, struct rspamd_expression_elt *op,
+ GError **err)
{
GNode *res, *a1, *a2, *test;
@@ -339,12 +381,27 @@ rspamd_ast_add_node (GPtrArray *operands, struct rspamd_expression_elt *op)
res = g_node_new (op);
a1 = rspamd_expr_stack_elt_pop (operands);
g_node_append (res, a1);
+ if (a1 == NULL) {
+ g_set_error (err, rspamd_expr_quark(), EINVAL, "no operand to "
+ "unary '%s' operation", rspamd_expr_op_to_str (op->p.op));
+ return FALSE;
+ }
}
else {
/* For binary operators we might want to examine chains */
a2 = rspamd_expr_stack_elt_pop (operands);
a1 = rspamd_expr_stack_elt_pop (operands);
+ if (a2 == NULL) {
+ g_set_error (err, rspamd_expr_quark(), EINVAL, "no left operand to "
+ "'%s' operation", rspamd_expr_op_to_str (op->p.op));
+ return FALSE;
+ }
+ if (a1 == NULL) {
+ g_set_error (err, rspamd_expr_quark(), EINVAL, "no right operand to "
+ "'%s' operation", rspamd_expr_op_to_str (op->p.op));
+ return FALSE;
+ }
/* First try with a1 */
test = a1;
test_elt = test->data;
@@ -353,7 +410,7 @@ rspamd_ast_add_node (GPtrArray *operands, struct rspamd_expression_elt *op)
/* Add children */
g_node_append (test, a2);
rspamd_expr_stack_elt_push (operands, a1);
- return;
+ return TRUE;
}
/* Now test a2 */
test = a2;
@@ -363,7 +420,7 @@ rspamd_ast_add_node (GPtrArray *operands, struct rspamd_expression_elt *op)
/* Add children */
g_node_prepend (test, a1);
rspamd_expr_stack_elt_push (operands, a2);
- return;
+ return TRUE;
}
/* No optimizations possible, so create new level */
@@ -374,6 +431,8 @@ rspamd_ast_add_node (GPtrArray *operands, struct rspamd_expression_elt *op)
/* Push back resulting node to the stack */
rspamd_expr_stack_elt_push (operands, res);
+
+ return TRUE;
}
static gboolean
@@ -620,8 +679,10 @@ rspamd_parse_expression (const gchar *line, gsize len,
elt.type = ELT_OP;
elt.p.op = op;
g_array_append_val (e->expressions, elt);
- rspamd_ast_add_node (operand_stack,
- rspamd_expr_dup_elt (pool, &elt));
+ if (!rspamd_ast_add_node (operand_stack,
+ rspamd_expr_dup_elt (pool, &elt), err)) {
+ goto err;
+ }
}
} while (op != OP_OBRACE);
@@ -656,8 +717,10 @@ rspamd_parse_expression (const gchar *line, gsize len,
elt.type = ELT_OP;
elt.p.op = op_stack;
g_array_append_val (e->expressions, elt);
- rspamd_ast_add_node (operand_stack,
- rspamd_expr_dup_elt (pool, &elt));
+ if (!rspamd_ast_add_node (operand_stack,
+ rspamd_expr_dup_elt (pool, &elt), err)) {
+ goto err;
+ }
}
else {
/* Push op_stack back */
@@ -693,8 +756,10 @@ rspamd_parse_expression (const gchar *line, gsize len,
elt.type = ELT_OP;
elt.p.op = op_stack;
g_array_append_val (e->expressions, elt);
- rspamd_ast_add_node (operand_stack,
- rspamd_expr_dup_elt (pool, &elt));
+ if (!rspamd_ast_add_node (operand_stack,
+ rspamd_expr_dup_elt (pool, &elt), err)) {
+ goto err;
+ }
}
else {
g_set_error (err, rspamd_expr_quark(), 600,
@@ -703,7 +768,12 @@ rspamd_parse_expression (const gchar *line, gsize len,
}
}
- g_assert (operand_stack->len == 1);
+ if (operand_stack->len != 1) {
+ g_set_error (err, rspamd_expr_quark(), 601,
+ "Operators mismatch");
+ goto err;
+ }
+
e->ast = rspamd_expr_stack_elt_pop (operand_stack);
g_ptr_array_free (operand_stack, TRUE);
@@ -985,38 +1055,7 @@ rspamd_ast_string_traverse (GNode *n, gpointer d)
rspamd_printf_gstring (res, "%d", elt->p.lim.val);
}
else {
- switch (elt->p.op) {
- case OP_AND:
- op_str = "&";
- break;
- case OP_OR:
- op_str = "|";
- break;
- case OP_MULT:
- op_str = "*";
- break;
- case OP_PLUS:
- op_str = "+";
- break;
- case OP_NOT:
- op_str = "!";
- break;
- case OP_GE:
- op_str = ">=";
- break;
- case OP_GT:
- op_str = ">";
- break;
- case OP_LE:
- op_str = "<=";
- break;
- case OP_LT:
- op_str = "<";
- break;
- default:
- op_str = "???";
- break;
- }
+ op_str = rspamd_expr_op_to_str (elt->p.op);
g_string_append (res, op_str);
if (n->children) {
diff --git a/test/lua/unit/expressions.lua b/test/lua/unit/expressions.lua
index d6deaefd7..1dc92997e 100644
--- a/test/lua/unit/expressions.lua
+++ b/test/lua/unit/expressions.lua
@@ -27,6 +27,7 @@ context("Rspamd expressions", function()
local cases = {
{'A & B | !C', 'C ! A B & |'},
{'A & (B | !C)', 'A B C ! | &'},
+ {'A & B &', nil},
-- Unbalanced braces
{'(((A))', nil},
-- Balanced braces