import math import pytest from collections import OrderedDict from unittest.mock import patch, MagicMock mock_nodes = MagicMock() mock_nodes.MAX_RESOLUTION = 16384 mock_server = MagicMock() with patch.dict("sys.modules", {"nodes": mock_nodes, "server": mock_server}): from comfy_extras.nodes_math import MathExpressionNode class TestMathExpressionExecute: @staticmethod def _exec(expression: str, **kwargs) -> object: values = OrderedDict(kwargs) return MathExpressionNode.execute(expression, values) def test_addition(self): result = self._exec("a + b", a=3, b=4) assert result[0] == 7.0 assert result[1] == 7 def test_subtraction(self): result = self._exec("a - b", a=10, b=3) assert result[0] == 7.0 assert result[1] == 7 def test_multiplication(self): result = self._exec("a * b", a=3, b=5) assert result[0] == 15.0 assert result[1] == 15 def test_division(self): result = self._exec("a / b", a=10, b=4) assert result[0] == 2.5 assert result[1] == 2 def test_single_input(self): result = self._exec("a * 2", a=5) assert result[0] == 10.0 assert result[1] == 10 def test_three_inputs(self): result = self._exec("a + b + c", a=1, b=2, c=3) assert result[0] == 6.0 assert result[1] == 6 def test_float_inputs(self): result = self._exec("a + b", a=1.5, b=2.5) assert result[0] == 4.0 assert result[1] == 4 def test_mixed_int_float_inputs(self): result = self._exec("a * b", a=1024, b=1.5) assert result[0] == 1536.0 assert result[1] == 1536 def test_mixed_resolution_scale(self): result = self._exec("a * b", a=512, b=0.75) assert result[0] == 384.0 assert result[1] == 384 def test_sum_values_array(self): result = self._exec("sum(values)", a=1, b=2, c=3) assert result[0] == 6.0 def test_sum_variadic(self): result = self._exec("sum(a, b, c)", a=1, b=2, c=3) assert result[0] == 6.0 def test_min_values(self): result = self._exec("min(values)", a=5, b=2, c=8) assert result[0] == 2.0 def test_max_values(self): result = self._exec("max(values)", a=5, b=2, c=8) assert result[0] == 8.0 def test_abs_function(self): result = self._exec("abs(a)", a=-7) assert result[0] == 7.0 assert result[1] == 7 def test_sqrt(self): result = self._exec("sqrt(a)", a=16) assert result[0] == 4.0 assert result[1] == 4 def test_ceil(self): result = self._exec("ceil(a)", a=2.3) assert result[0] == 3.0 assert result[1] == 3 def test_floor(self): result = self._exec("floor(a)", a=2.7) assert result[0] == 2.0 assert result[1] == 2 def test_sin(self): result = self._exec("sin(a)", a=0) assert result[0] == 0.0 def test_log10(self): result = self._exec("log10(a)", a=100) assert result[0] == 2.0 assert result[1] == 2 def test_float_output_type(self): result = self._exec("a + b", a=1, b=2) assert isinstance(result[0], float) def test_int_output_type(self): result = self._exec("a + b", a=1, b=2) assert isinstance(result[1], int) def test_non_numeric_result_raises(self): with pytest.raises(ValueError, match="must evaluate to a numeric result"): self._exec("'hello'", a=42) def test_undefined_function_raises(self): with pytest.raises(Exception, match="not defined"): self._exec("str(a)", a=42) def test_boolean_result_raises(self): with pytest.raises(ValueError, match="got bool"): self._exec("a > b", a=5, b=3) def test_empty_expression_raises(self): with pytest.raises(ValueError, match="Expression cannot be empty"): self._exec("", a=1) def test_whitespace_only_expression_raises(self): with pytest.raises(ValueError, match="Expression cannot be empty"): self._exec(" ", a=1) # --- Missing function coverage (round, pow, log, log2, cos, tan) --- def test_round(self): result = self._exec("round(a)", a=2.7) assert result[0] == 3.0 assert result[1] == 3 def test_round_with_ndigits(self): result = self._exec("round(a, 2)", a=3.14159) assert result[0] == pytest.approx(3.14) def test_pow(self): result = self._exec("pow(a, b)", a=2, b=10) assert result[0] == 1024.0 assert result[1] == 1024 def test_log(self): result = self._exec("log(a)", a=math.e) assert result[0] == pytest.approx(1.0) def test_log2(self): result = self._exec("log2(a)", a=8) assert result[0] == pytest.approx(3.0) def test_cos(self): result = self._exec("cos(a)", a=0) assert result[0] == 1.0 def test_tan(self): result = self._exec("tan(a)", a=0) assert result[0] == 0.0 # --- int/float converter functions --- def test_int_converter(self): result = self._exec("int(a / b)", a=7, b=2) assert result[1] == 3 def test_float_converter(self): result = self._exec("float(a)", a=5) assert result[0] == 5.0 # --- Error path tests --- def test_division_by_zero_raises(self): with pytest.raises(ZeroDivisionError): self._exec("a / b", a=1, b=0) def test_sqrt_negative_raises(self): with pytest.raises(ValueError, match="math domain error"): self._exec("sqrt(a)", a=-1) def test_overflow_inf_raises(self): with pytest.raises(ValueError, match="non-finite result"): self._exec("a * b", a=1e308, b=10) def test_pow_huge_exponent_raises(self): with pytest.raises(ValueError, match="Exponent .* exceeds maximum"): self._exec("pow(a, b)", a=10, b=10000000)