From a7b678bd4b78baa7260a2485bffc0ccfcd056aab Mon Sep 17 00:00:00 2001 From: Michael Matloka Date: Thu, 20 Jun 2024 16:47:55 +0200 Subject: [PATCH] feat(insights): Support unary minus and plus in formulas (#23119) * feat(insights): Support unary minus and plus in formulas * Fix variable use issue --- posthog/hogql_queries/utils/formula_ast.py | 10 +++++++++- posthog/hogql_queries/utils/test/test_formula_ast.py | 10 ++++++++++ 2 files changed, 19 insertions(+), 1 deletion(-) diff --git a/posthog/hogql_queries/utils/formula_ast.py b/posthog/hogql_queries/utils/formula_ast.py index 922e283362a49..c46f5f8dc1bfc 100644 --- a/posthog/hogql_queries/utils/formula_ast.py +++ b/posthog/hogql_queries/utils/formula_ast.py @@ -49,7 +49,6 @@ def _evaluate(self, node, const_map: dict[str, Any]): left = self._evaluate(node.left, const_map) op = node.op right = self._evaluate(node.right, const_map) - try: return self.op_map[type(op)](left, right) except ZeroDivisionError: @@ -57,6 +56,15 @@ def _evaluate(self, node, const_map: dict[str, Any]): except KeyError: raise ValueError(f"Operator {op.__class__.__name__} not supported") + elif isinstance(node, ast.UnaryOp): + operand = self._evaluate(node.operand, const_map) + unary_op = node.op + if isinstance(unary_op, ast.USub): + return -operand + elif isinstance(unary_op, ast.UAdd): + return operand + raise ValueError(f"Operator {unary_op.__class__.__name__} not supported") + elif isinstance(node, ast.Num): return node.n diff --git a/posthog/hogql_queries/utils/test/test_formula_ast.py b/posthog/hogql_queries/utils/test/test_formula_ast.py index 3f065bf516cbc..10dd4cde6c453 100644 --- a/posthog/hogql_queries/utils/test/test_formula_ast.py +++ b/posthog/hogql_queries/utils/test/test_formula_ast.py @@ -56,3 +56,13 @@ def test_named_values_lower_case(self): formula = self._get_formula_ast() response = formula.call("a+b") self.assertListEqual([2, 4, 6, 8], response) + + def test_unary_minus(self): + formula = self._get_formula_ast() + response = formula.call("-A") + self.assertListEqual([-1, -2, -3, -4], response) + + def test_unary_plus(self): + formula = self._get_formula_ast() + response = formula.call("+A") + self.assertListEqual([1, 2, 3, 4], response)