import unittest
import sys
import os
import math

# Add the parent directory to the sys.path to allow imports from the project root
sys.path.insert(0, os.path.abspath(os.path.join(os.path.dirname(__file__), '..')))

from olym_gen.generator.answer_compare_generator import AnswerPairGenerator, ExtractAnswerMixin

import logging
from olym_gen.utils.utils import get_logger
logger = get_logger()

class ConcreteExtractAnswerMixin(ExtractAnswerMixin):
    pass

class LoggerCaptureHandler(unittest.TestCase):
    """A simple logger handler to capture log messages for testing."""
    
    def setUp(self):
        self.log_capture = []
        # remove all existing handler
        for handler in logger.handlers[:]:
            logger.removeHandler(handler)
            handler.close()
        handler = logging.Handler()
        handler.emit = lambda record: self.log_capture.append(record.getMessage())
        logger.addHandler(handler)
        # aviod logger propagate to root logger
        logger.propagate = False
        
    def tearDown(self):
        # 移除所有 handler，防止影响其他测试
        for handler in logger.handlers[:]:
            logger.removeHandler(handler)
            handler.close()
        logger.propagate = True

    def get_log_messages(self):
        return self.log_capture

    def clear_log_messages(self):
        self.log_capture.clear()

class TestAreCommutativelyEqual(LoggerCaptureHandler):

    def setUp(self):
        """Set up a new AnswerPairGenerator for each test."""
        # The provider and model are not used in these tests, so they can be placeholders.
        self.generator = AnswerPairGenerator(provider="dummy", model="placeholder")
        super().setUp()

    def test_simple_addition_without_spaces(self):
        """Test commutativity of simple addition without spaces."""
        self.assertTrue(self.generator._are_commutatively_equal("a+b", "b+a"))
        self.assertTrue(self.generator._are_commutatively_equal("x+y", "y+x"))
        self.assertTrue(self.generator._are_commutatively_equal("1+2", "2+1"))
        self.assertTrue(self.generator._are_commutatively_equal("a+b+c", "c+b+a"))
        self.assertTrue(self.generator._are_commutatively_equal("a+b+c+d", "d+c+b+a"))
    
    def test_simple_addition(self):
        """Test commutativity of simple addition."""
        self.assertTrue(self.generator._are_commutatively_equal("a + b", "b + a"))
        self.assertTrue(self.generator._are_commutatively_equal("x + y", "y + x"))
        self.assertTrue(self.generator._are_commutatively_equal("x + 1", "1 + x"))
        self.assertTrue(self.generator._are_commutatively_equal("a + b + c", "c + b + a"))

    def test_simple_multiplication(self):
        """Test commutativity of simple multiplication."""
        self.assertTrue(self.generator._are_commutatively_equal("a * b", "b * a"))
        self.assertTrue(self.generator._are_commutatively_equal("2 * x", "x * 2"))
        self.assertTrue(self.generator._are_commutatively_equal("a * b * c", "c * b * a"))

    def test_mixed_operations(self):
        """Test commutativity of mixed addition and multiplication."""
        self.assertTrue(self.generator._are_commutatively_equal("a + b * c", "c * b + a"))
        self.assertTrue(self.generator._are_commutatively_equal("x * y + z", "z + y * x"))
        self.assertTrue(self.generator._are_commutatively_equal("a * b + c * d", "d * c + b * a"))
        self.assertTrue(self.generator._are_commutatively_equal("a - b * c", "a - c * b"))
        self.assertTrue(self.generator._are_commutatively_equal("-b * c + a", "a - c * b"))
        self.assertTrue(self.generator._are_commutatively_equal("a + b * c + d", "d + a + c * b"))

    def test_subtraction_and_mixed_terms(self):
        """Test more complex cases involving subtraction and mixed terms."""
        # Note: The function checks for commutative equality, not full mathematical equivalence.
        # 'a - b' vs '-b + a' is a key test case for parsing terms with signs.
        self.assertTrue(self.generator._are_commutatively_equal("a - b", "-b + a"))
        self.assertTrue(self.generator._are_commutatively_equal("a * b - c", "-c + b * a"))
        self.assertTrue(self.generator._are_commutatively_equal("a - b - c", "-c - b + a"))
        self.assertTrue(self.generator._are_commutatively_equal("x * y - z * w + k", "k + y * x - w * z"))
        self.assertTrue(self.generator._are_commutatively_equal("2 * x - 3 * y", "-3 * y + 2 * x"))

    def test_non_commutative_cases(self):
        """Test expressions that are not commutatively equal."""
        self.assertFalse(self.generator._are_commutatively_equal("a - b", "b - a"))
        self.assertFalse(self.generator._are_commutatively_equal("a / b", "b / a"))
        self.assertFalse(self.generator._are_commutatively_equal("a + b", "a * b"))
        self.assertFalse(self.generator._are_commutatively_equal("a + b * c", "a * b + c"))
        # This case is correctly False because 'a - b' is not commutatively equal to '-b - a' (-a-b)
        self.assertFalse(self.generator._are_commutatively_equal("a - b", "-b - a"))

    def test_expressions_with_functions_or_parentheses(self):
        """Test that complex expressions are not considered equal by this simple check."""
        self.assertFalse(self.generator._are_commutatively_equal("sin(x) + cos(y)", "cos(y) + sin(x)"))
        self.assertFalse(self.generator._are_commutatively_equal("a * (b + c)", "(b + c) * a"))
        self.assertFalse(self.generator._are_commutatively_equal("a * (b + c)", "a * b + a * c"))

    def test_identical_expressions(self):
        """Test that identical expressions return False as no commutation is performed."""
        self.assertTrue(self.generator._are_commutatively_equal("a + b", "a + b"))
        self.assertTrue(self.generator._are_commutatively_equal("a * b", "a * b"))

    def test_different_expressions(self):
        """Test completely different expressions."""
        self.assertFalse(self.generator._are_commutatively_equal("a + b", "x + y"))
        self.assertFalse(self.generator._are_commutatively_equal("a * b", "x * y"))
        self.assertFalse(self.generator._are_commutatively_equal("a + b * c", "x + y * z"))
        
    def test_with_illegal_math_expr(self):
        """Test expressions that are not valid mathematical expressions."""
        self.assertFalse(self.generator._are_commutatively_equal("a-b", "a--b"))
        self.assertTrue(self.generator._are_commutatively_equal("a-b", "a+-b"))
        self.assertFalse(self.generator._are_commutatively_equal("a*b", "a/-+ds"))
        self.assertFalse(self.generator._are_commutatively_equal("a+b", "*bce"))
        
    def test_very_different_expressions(self):
        self.assertFalse(self.generator._are_commutatively_equal("(a+1)/4", '1'))
        self.assertFalse(self.generator._are_commutatively_equal("100", "10"))
        self.assertFalse(self.generator._are_commutatively_equal("a+b", "a*c"))
        self.assertFalse(self.generator._are_commutatively_equal("10", "arccos(30)"))
        self.assertFalse(self.generator._are_commutatively_equal("a-b+db-18", "-d+-+-23"))
        self.assertFalse(self.generator._are_commutatively_equal("x^2+2x+1", "a random string"))
        self.assertFalse(self.generator._are_commutatively_equal("1-x", "y-z^3"))
        self.assertFalse(self.generator._are_commutatively_equal("a/b", "c*d-e"))
        self.assertFalse(self.generator._are_commutatively_equal("12345", "abcde"))
        self.assertFalse(self.generator._are_commutatively_equal("a+b-c", "xyz"))

class TestAreNumericallyEqual(LoggerCaptureHandler):
    def setUp(self):
        """Set up a new AnswerPairGenerator for each test."""
        self.generator = AnswerPairGenerator(provider="dummy", model="placeholder")
        super().setUp()

    def evaluate_and_check(self, expr: str) -> float:
        """Helper function to evaluate expression and check for None."""
        result = self.generator._evaluate_safe_expression(expr)
        self.assertIsNotNone(result, f"Evaluation of '{expr}' returned None unexpectedly.")
        # We need to cast result to float as assertIsNotNone doesn't narrow the type for static checkers.
        return float(result) # type: ignore

    def test_evaluate_safe_expression(self):
        # Test basic arithmetic
        self.assertAlmostEqual(self.evaluate_and_check("2 + 3"), 5.0)
        self.assertAlmostEqual(self.evaluate_and_check("10 - 4"), 6.0)
        self.assertAlmostEqual(self.evaluate_and_check("5 * 6"), 30.0)
        self.assertAlmostEqual(self.evaluate_and_check("20 / 4"), 5.0)
        self.assertAlmostEqual(self.evaluate_and_check("2 * (3 + 4)"), 14.0)
        self.assertAlmostEqual(self.evaluate_and_check("(10 - 2) / 2"), 4.0)
        
        # Test with constants
        self.assertAlmostEqual(self.evaluate_and_check("π"), math.pi)
        self.assertAlmostEqual(self.evaluate_and_check("2 * π"), 2 * math.pi)
        self.assertAlmostEqual(self.evaluate_and_check("e"), math.e)

        # Test with decimals
        self.assertAlmostEqual(self.evaluate_and_check("1.5 + 2.5"), 4.0)
        self.assertAlmostEqual(self.evaluate_and_check("0.5 * 3"), 1.5)

        # Test invalid/unsafe expressions
        self.assertIsNone(self.generator._evaluate_safe_expression("2^3")) # unsupported operator
        self.assertIsNone(self.generator._evaluate_safe_expression("sqrt(4)")) # unsupported function
        self.assertIsNone(self.generator._evaluate_safe_expression("import os")) # unsafe
        self.assertIsNone(self.generator._evaluate_safe_expression("a + b")) # contains variables
        self.assertIsNone(self.generator._evaluate_safe_expression("__import__('os').system('ls')"))

    def test_are_numerically_equal(self):
        # Test equal expressions
        self.assertTrue(self.generator._are_numerically_equal("2 + 3", "5"))
        self.assertTrue(self.generator._are_numerically_equal("10 / 2", "5.0"))
        self.assertTrue(self.generator._are_numerically_equal("2 * 3", "6"))
        self.assertTrue(self.generator._are_numerically_equal("2 * (3 + 1)", "8"))
        self.assertTrue(self.generator._are_numerically_equal("π", "3.141592653589793"))
        self.assertTrue(self.generator._are_numerically_equal("2*π", "6.283185307179586"))

        # Test unequal expressions
        self.assertFalse(self.generator._are_numerically_equal("2 + 3", "6"))
        self.assertFalse(self.generator._are_numerically_equal("10 / 2", "4"))

        # Test non-numerical expressions
        self.assertFalse(self.generator._are_numerically_equal("a+b", "c"))
        self.assertFalse(self.generator._are_numerically_equal("2*x", "2x"))
        self.assertFalse(self.generator._are_numerically_equal("sin(x)", "0"))

        # Test with slight precision differences
        self.assertTrue(self.generator._are_numerically_equal("1/3", "0.3333333333"))
        self.assertTrue(self.generator._are_numerically_equal("2*π", "6.283185307179587"))

        # Test with unsafe or illegal expressions
        self.assertFalse(self.generator._are_numerically_equal("import os", "0")) # Unsafe
        self.assertFalse(self.generator._are_numerically_equal("sqrt(4)", "2")) # Unsupported function
        self.assertFalse(self.generator._are_numerically_equal("2^3", "8")) # Unsupported operator
        self.assertFalse(self.generator._are_numerically_equal("1/0", "0")) # Division by zero
        self.assertFalse(self.generator._are_numerically_equal("a+b", "5")) # Contains variables

        # Test one valid, one invalid expression
        self.assertFalse(self.generator._are_numerically_equal("5", "1/0"))
        self.assertFalse(self.generator._are_numerically_equal("2+3", "a+b"))
        self.assertFalse(self.generator._are_numerically_equal("import os", "5"))

        # Test both expressions being invalid
        self.assertFalse(self.generator._are_numerically_equal("1/0", "a+b"))
        self.assertFalse(self.generator._are_numerically_equal("sqrt(4)", "2^3"))
        self.assertFalse(self.generator._are_numerically_equal("import os", "__import__('shutil')"))

class TestRemoveOuterBraces(LoggerCaptureHandler):
    def setUp(self):
        class Dummy(ExtractAnswerMixin):
            pass
        self.remover = Dummy()
        super().setUp()

    def test_simple_braces(self):
        self.assertEqual(self.remover._remove_outer_braces("(a+b)"), "a+b")
        self.assertEqual(self.remover._remove_outer_braces("[a+b]"), "a+b")
        self.assertEqual(self.remover._remove_outer_braces("{a+b}"), "a+b")

    def test_escaped_braces(self):
        self.assertEqual(self.remover._remove_outer_braces("\\(a+b\\)"), "a+b")
        self.assertEqual(self.remover._remove_outer_braces("\\[a+b\\]"), "a+b")
        self.assertEqual(self.remover._remove_outer_braces("\\{a+b\\}"), "a+b")

    def test_no_outer_braces(self):
        self.assertEqual(self.remover._remove_outer_braces("a+b"), "a+b")
        self.assertEqual(self.remover._remove_outer_braces(" (a+b) "), " (a+b) ")

    def test_mismatched_outer_braces(self):
        self.assertEqual(self.remover._remove_outer_braces("(a+b]"), "(a+b]")
        self.assertEqual(self.remover._remove_outer_braces("{a+b)"), "{a+b)")

    def test_unbalanced_internal_braces(self):
        self.assertEqual(self.remover._remove_outer_braces("(a+{b)"), "a+{b")
        self.assertEqual(self.remover._remove_outer_braces("([a+b)]"), "([a+b)]")

    def test_nested_braces(self):
        self.assertEqual(self.remover._remove_outer_braces("((a+b))"), "(a+b)")
        self.assertEqual(self.remover._remove_outer_braces("[[a+b]]"), "[a+b]")
        self.assertEqual(self.remover._remove_outer_braces("{{a+b}}"), "{a+b}")
        self.assertEqual(self.remover._remove_outer_braces("\\(\\(a+b\\)\\)"), "\\(a+b\\)")

    def test_mixed_nested_braces(self):
        self.assertEqual(self.remover._remove_outer_braces("([a+b])"), "[a+b]")
        self.assertEqual(self.remover._remove_outer_braces("{[a+b]}"), "[a+b]")

    def test_braces_not_enclosing_whole_string(self):
        self.assertEqual(self.remover._remove_outer_braces("(a)+(b)"), "(a)+(b)")
        self.assertEqual(self.remover._remove_outer_braces("{a}+{b}"), "{a}+{b}")
        self.assertEqual(self.remover._remove_outer_braces("(a+b) - (c+d)"), "(a+b) - (c+d)")

    def test_empty_content(self):
        self.assertEqual(self.remover._remove_outer_braces("()"), "")
        self.assertEqual(self.remover._remove_outer_braces("[]"), "")
        self.assertEqual(self.remover._remove_outer_braces("{}"), "")
        self.assertEqual(self.remover._remove_outer_braces("\\(\\)"), "")

    def test_single_character_content(self):
        self.assertEqual(self.remover._remove_outer_braces("(a)"), "a")
        self.assertEqual(self.remover._remove_outer_braces("[b]"), "b")
        self.assertEqual(self.remover._remove_outer_braces("{c}"), "c")

    def test_string_with_only_braces(self):
        self.assertEqual(self.remover._remove_outer_braces("()"), "")
        self.assertEqual(self.remover._remove_outer_braces("[]"), "")
        self.assertEqual(self.remover._remove_outer_braces("{}"), "")

    def test_complex_expressions(self):
        self.assertEqual(self.remover._remove_outer_braces("((a*b)/(c+d))"), "(a*b)/(c+d)")
        self.assertEqual(self.remover._remove_outer_braces("{\\frac{a}{b}}"), "\\frac{a}{b}")

    def test_no_braces_at_all(self):
        self.assertEqual(self.remover._remove_outer_braces("just a string"), "just a string")

    def test_incomplete_braces(self):
        self.assertEqual(self.remover._remove_outer_braces("(a+b"), "(a+b")
        self.assertEqual(self.remover._remove_outer_braces("a+b)"), "a+b)")
        self.assertEqual(self.remover._remove_outer_braces("{a+b"), "{a+b")
        self.assertEqual(self.remover._remove_outer_braces("a+b}"), "a+b}")

    def test_escaped_braces_inside(self):
        self.assertEqual(self.remover._remove_outer_braces("(\\sqrt\\{2\\})"), "\\sqrt\\{2\\}")
        self.assertEqual(self.remover._remove_outer_braces("a \\( b \\) c"), "a \\( b \\) c")

class TestNormalizeLatexAnswer(LoggerCaptureHandler):
    def setUp(self):
        class Dummy(ExtractAnswerMixin):
            pass
        self.normalizer = Dummy()
        super().setUp()

    def test_empty_and_whitespace(self):
        self.assertEqual(self.normalizer.normalize_latex_answer(""), "")
        self.assertEqual(self.normalizer.normalize_latex_answer("   "), "")
        self.assertEqual(self.normalizer.normalize_latex_answer("  hello world  "), "helloworld")
        self.assertEqual(self.normalizer.normalize_latex_answer("hello   world"), "helloworld")

    def test_operator_spacing(self):
        self.assertEqual(self.normalizer.normalize_latex_answer("a + b"), "a+b")
        self.assertEqual(self.normalizer.normalize_latex_answer("a-b"), "a-b")
        self.assertEqual(self.normalizer.normalize_latex_answer("a = b"), "a=b")
        self.assertEqual(self.normalizer.normalize_latex_answer("a < b > c"), "a<b>c")
        self.assertEqual(self.normalizer.normalize_latex_answer(" ( a ) "), "a")
        self.assertEqual(self.normalizer.normalize_latex_answer(" [ b ] "), "b")
        self.assertEqual(self.normalizer.normalize_latex_answer(" { c } "), "c")
        self.assertEqual(self.normalizer.normalize_latex_answer("a, b; c"), "a,b;c")

    def test_latex_fractions(self):
        self.assertEqual(self.normalizer.normalize_latex_answer("\\frac{1}{2}"), "(1)/(2)")
        self.assertEqual(self.normalizer.normalize_latex_answer("\\frac{x+1}{y-1}"), "(x+1)/(y-1)")
        self.assertEqual(self.normalizer.normalize_latex_answer(" \\frac { a } { b } "), "(a)/(b)")
        self.assertEqual(self.normalizer.normalize_latex_answer("\\frac{a+b}{(c+d)f}"), "(a+b)/((c+d)f)")
        
    def test_complex_fractions(self):
        """
        If the fraction is complex, directly return it as is.
        """
        self.assertEqual(self.normalizer.normalize_latex_answer("\\frac{\\frac{1}{2}}{\\frac{3}{4}}"), "((1)/(2))/((3)/(4))")
        self.assertEqual(self.normalizer.normalize_latex_answer("\\frac{\\sqrt{x}}{y}"), "(\\sqrt{x})/(y)")
        self.assertEqual(self.normalizer.normalize_latex_answer("\\frac{\\sqrt{16}}{4}"), "(\\sqrt{16})/(4)")
        self.assertEqual(self.normalizer.normalize_latex_answer("\\frac{1}{2+\\frac{1}{3}}"), "(1)/(2+(1)/(3))")
        

    def test_latex_functions_and_constants(self):
        self.assertEqual(self.normalizer.normalize_latex_answer("\\pi"), "π")
        self.assertEqual(self.normalizer.normalize_latex_answer("\\alpha"), "α")
        self.assertEqual(self.normalizer.normalize_latex_answer("\\beta"), "β")
        self.assertEqual(self.normalizer.normalize_latex_answer("\\gamma"), "γ")
        self.assertEqual(self.normalizer.normalize_latex_answer("4\\pi"), "4π")
        self.assertEqual(self.normalizer.normalize_latex_answer("4\\pi r"), "4πr")
        self.assertEqual(self.normalizer.normalize_latex_answer("\\pm"), "±")
        self.assertEqual(self.normalizer.normalize_latex_answer("\\infty"), "∞")
        self.assertEqual(self.normalizer.normalize_latex_answer("a \\ne b"), "a!=b")
        self.assertEqual(self.normalizer.normalize_latex_answer("a \\neq b"), "a!=b")
        self.assertEqual(self.normalizer.normalize_latex_answer("a \\ge b"), "a>=b")
        self.assertEqual(self.normalizer.normalize_latex_answer("a \\geq b"), "a>=b")
        self.assertEqual(self.normalizer.normalize_latex_answer("a \\le b"), "a<=b")
        self.assertEqual(self.normalizer.normalize_latex_answer("a \\leq b"), "a<=b")
        self.assertEqual(self.normalizer.normalize_latex_answer("a \\approx b"), "a≈b")
        self.assertEqual(self.normalizer.normalize_latex_answer("a \\dots b"), "a...b")

    def test_latex_symbols(self):
        self.assertEqual(self.normalizer.normalize_latex_answer("a \\cdot b"), "a*b")
        self.assertEqual(self.normalizer.normalize_latex_answer("a \\times b"), "a*b")
        self.assertEqual(self.normalizer.normalize_latex_answer("a \\div b"), "a/b")

    def test_brace_removal(self):
        self.assertEqual(self.normalizer.normalize_latex_answer("{x}"), "x")
        self.assertEqual(self.normalizer.normalize_latex_answer("{1}"), "1")
        self.assertEqual(self.normalizer.normalize_latex_answer("a + {b}"), "a+b")
        self.assertEqual(self.normalizer.normalize_latex_answer("f({x})"), "f(x)")
        self.assertEqual(self.normalizer.normalize_latex_answer("sin{x}"), "sin{x}")
        self.assertEqual(self.normalizer.normalize_latex_answer("a{b}c"), "a{b}c")
        self.assertEqual(self.normalizer.normalize_latex_answer("a*{b}"), "a*b")
        self.assertEqual(self.normalizer.normalize_latex_answer("{a}+{b}"), "a+b")
        self.assertEqual(self.normalizer.normalize_latex_answer("{-1}"), "-1")
        self.assertEqual(self.normalizer.normalize_latex_answer("{-a}"), "-a")
        self.assertEqual(self.normalizer.normalize_latex_answer("a{10}"), "a{10}")
        self.assertEqual(self.normalizer.normalize_latex_answer("a{bc}"), "a{bc}") # as designed
        self.assertEqual(self.normalizer.normalize_latex_answer("a{{{b}}}c{d}"), "a{b}c{d}")
        self.assertEqual(self.normalizer.normalize_latex_answer("a{{a+b}{c}} + d"), "a{{a+b}{c}}+d")
        self.assertEqual(self.normalizer.normalize_latex_answer("{{-1}}"), "-1")
        self.assertEqual(self.normalizer.normalize_latex_answer("a+{{b}}"), "a+b")
        self.assertEqual(self.normalizer.normalize_latex_answer("a+{b+c}"), "a+{b+c}") # if the content is not a single variable, we don't remove the outer braces

    def test_number_normalization(self):
        self.assertEqual(self.normalizer.normalize_latex_answer("0"), "0")
        self.assertEqual(self.normalizer.normalize_latex_answer("1"), "1")
        self.assertEqual(self.normalizer.normalize_latex_answer("a."), "a.")
        self.assertEqual(self.normalizer.normalize_latex_answer("a.b"), "a.b")
        self.assertEqual(self.normalizer.normalize_latex_answer("01"), "1")
        self.assertEqual(self.normalizer.normalize_latex_answer("007"), "7")
        self.assertEqual(self.normalizer.normalize_latex_answer("2.0"), "2")
        self.assertEqual(self.normalizer.normalize_latex_answer("3.000"), "3")
        self.assertEqual(self.normalizer.normalize_latex_answer("0.5"), ".5")
        self.assertEqual(self.normalizer.normalize_latex_answer("a+01"), "a+1")
        self.assertEqual(self.normalizer.normalize_latex_answer("1.200"), "1.2")
        self.assertEqual(self.normalizer.normalize_latex_answer("1.02"), "1.02")
        self.assertEqual(self.normalizer.normalize_latex_answer("1.020"), "1.02")
        self.assertEqual(self.normalizer.normalize_latex_answer("0.05"), ".05")
        self.assertEqual(self.normalizer.normalize_latex_answer("0.050"), ".05")
        self.assertEqual(self.normalizer.normalize_latex_answer("12.0"), "12")
        self.assertEqual(self.normalizer.normalize_latex_answer("12."), "12")
        self.assertEqual(self.normalizer.normalize_latex_answer("0.123456789+1345"), ".123456789+1345")
        self.assertEqual(self.normalizer.normalize_latex_answer("a+0.12345678900000+00001345.0000+b+c+00001"), "a+.123456789+1345+b+c+1")
        self.assertEqual(self.normalizer.normalize_latex_answer("a.b.c.d", ), "a.b.c.d")  # no normalization for dots in variable names
        self.assertEqual(self.normalizer.normalize_latex_answer("a3.1b.2c.3"), "a3.1b.2c.3")  # no normalization for dots in variable names
        self.assertEqual(self.normalizer.normalize_latex_answer("123,000,123,000.000,123,000"), "123000123000.000123")  # no normalization for dots in variable names

    def test_outer_brace_removal(self):
        self.assertEqual(self.normalizer.normalize_latex_answer("{a+b}"), "a+b")
        self.assertEqual(self.normalizer.normalize_latex_answer("(a+b)"), "a+b")
        self.assertEqual(self.normalizer.normalize_latex_answer("[a+b]"), "a+b")
        self.assertEqual(self.normalizer.normalize_latex_answer("\\(a+b\\)"), "a+b")
        self.assertEqual(self.normalizer.normalize_latex_answer("\\[a+b\\]"), "a+b")
        self.assertEqual(self.normalizer.normalize_latex_answer("\\{a+b\\}"), "a+b")
        self.assertEqual(self.normalizer.normalize_latex_answer("{{a+b}}"), "a+b") # only one level
        self.assertEqual(self.normalizer.normalize_latex_answer(" ( (a+b) ) "), "a+b")

    def test_mixed_spacing(self):
        self.assertEqual(self.normalizer.normalize_latex_answer("2 π r"), "2πr")
        self.assertEqual(self.normalizer.normalize_latex_answer("2 * π * r"), "2*π*r")
        self.assertEqual(self.normalizer.normalize_latex_answer("π r"), "πr")

    def test_unicode_normalization(self):
        # example with full-width characters
        self.assertEqual(self.normalizer.normalize_latex_answer("ａ＋ｂ"), "a+b")
        self.assertEqual(self.normalizer.normalize_latex_answer("１２３"), "123")

    def test_complex_cases(self):
        self.assertEqual(self.normalizer.normalize_latex_answer(" \\frac { -b \\pm \\sqrt { b^2 - 4ac } } { 2a } "), "(-b±\\sqrt{b^2-4ac})/(2a)")
        self.assertEqual(self.normalizer.normalize_latex_answer(" { ( 1 + 2.0 ) } "), "1+2")
        self.assertEqual(self.normalizer.normalize_latex_answer("\\log_{2}{8}"), "\\log_2{8}")

class TestSolveFrac(LoggerCaptureHandler):
    def setUp(self):
        """Set up an instance of the class containing solve_frac."""
        self.extractor = ExtractAnswerMixin()
        super().setUp()

    def test_simple_fraction(self):
        """Test a simple fraction."""
        self.assertEqual(self.extractor.solve_frac(r'\frac{1}{2}'), '(1)/(2)')

    def test_fraction_with_variables(self):
        """Test a fraction with variables."""
        self.assertEqual(self.extractor.solve_frac(r'\frac{a}{b}'), '(a)/(b)')

    def test_fraction_with_expressions(self):
        """Test a fraction with expressions in numerator and denominator."""
        self.assertEqual(self.extractor.solve_frac(r'\frac{x+y}{z-w}'), '(x+y)/(z-w)')

    def test_nested_fraction_in_numerator(self):
        """Test a nested fraction in the numerator."""
        self.assertEqual(self.extractor.solve_frac(r'\frac{\frac{1}{2}}{3}'), '((1)/(2))/(3)')

    def test_nested_fraction_in_denominator(self):
        """Test a nested fraction in the denominator."""
        self.assertEqual(self.extractor.solve_frac(r'\frac{1}{\frac{2}{3}}'), '(1)/((2)/(3))')
        
    def test_nested_fraction_both_sides(self):
        """Test a fraction with nested fractions on both sides."""
        self.assertEqual(self.extractor.solve_frac(r'\frac{\frac{1}{2}}{\frac{3}{4}}'), '((1)/(2))/((3)/(4))')

    def test_multiple_fractions(self):
        """Test a string with multiple fractions."""
        self.assertEqual(self.extractor.solve_frac(r'\frac{1}{2} + \frac{3}{4}'), '(1)/(2) + (3)/(4)')

    def test_fraction_with_extra_braces(self):
        """Test a fraction with extra, non-functional braces."""
        self.assertEqual(self.extractor.solve_frac(r'\frac{a+{b}}{c}'), '(a+b)/(c)')

    def test_complex_nested_fraction(self):
        """Test a more complex nested fraction scenario."""
        self.assertEqual(self.extractor.solve_frac(r'\frac{a+{\frac{b}{c}}}{d}'), '(a+{(b)/(c)})/(d)')

    def test_no_fraction(self):
        """Test a string with no fraction."""
        self.assertEqual(self.extractor.solve_frac('a+b'), 'a+b')

    def test_malformed_fraction_incomplete(self):
        """Test a malformed (incomplete) fraction string."""
        self.assertEqual(self.extractor.solve_frac(r'\frac{1'), r'\frac{1')
        self.assertEqual(self.extractor.solve_frac(r'\frac{1}{'), r'\frac{1}{')
        self.assertEqual(self.extractor.solve_frac(r'\frac{1}{2'), r'\frac{1}{2')
        self.assertEqual(self.extractor.solve_frac(r'\frac(1){2'), r'\frac(1){2')
        self.assertEqual(self.get_log_messages()[0], f"Found non-whitespace character before opening brace in \\frac: `(` at position 5")
        self.clear_log_messages()
        
        
    def test_whitespace_around_fraction(self):
        """Test a fraction with whitespace around it."""
        self.assertEqual(self.extractor.solve_frac(r'\frac{ 1 }{ 2 }'), '(1)/(2)')
        # allow whitespace between the command and the braces
        self.assertEqual(self.extractor.solve_frac(r'\frac { 1 }{ 2 }'), '(1)/(2)')
        self.assertEqual(self.extractor.solve_frac(r'\frac{1}   { 2 }'), '(1)/(2)')

    def test_malformed_fraction_unmatched_braces(self):
        """Test a fraction with unmatched braces."""
        self.assertEqual(self.extractor.solve_frac(r'\frac{{1}}{{2}'), r'\frac{{1}}{{2}')
        
    def test_empty_fraction(self):
        """Test an empty fraction. This function should return the input as is."""
        self.assertEqual(self.extractor.solve_frac(r'\frac{}{2}'), r'\frac{}{2}')
        self.assertEqual(self.extractor.solve_frac(r'\frac{1}{ }'), r'\frac{1}{ }')
        self.assertEqual(self.extractor.solve_frac(r'\frac{     }{ }'), r'\frac{     }{ }')

    def test_malformed_fraction_unmatched_internal_braces(self):
        """Test a fraction with unmatched internal braces that should cause it to be skipped."""
        self.assertEqual(self.extractor.solve_frac(r'\frac{a{b}{c}'), r'\frac{a{b}{c}')

    def test_deeply_nested_fractions(self):
        """Test deeply nested fractions."""
        self.assertEqual(self.extractor.solve_frac(r'\frac{\frac{\frac{1}{2}}{3}}{4}'), '(((1)/(2))/(3))/(4)')

    def test_side_by_side_nested_fractions(self):
        """Test multiple nested fractions side-by-side."""
        self.assertEqual(self.extractor.solve_frac(r'\frac{1}{\frac{2}{3}} - \frac{\frac{4}{5}}{6}'), '(1)/((2)/(3)) - ((4)/(5))/(6)')

if __name__ == '__main__':
    unittest.main()
