from typing import List, Tuple
import math
import re
from typing import Optional, Tuple, Dict, Iterable, Sequence

from .Plotter import RealWorldPlotter
DSL_PATTERN = re.compile(r"(\w+)\((.*?)\)")
EPS = 1e-6

from ..utils import latex_to_float
import ast
import operator
import math

BIN_OPS = {
    ast.Add: operator.add,
    ast.Sub: operator.sub,
    ast.Mult: operator.mul,
    ast.Div: operator.truediv,
    ast.Pow: operator.pow,
}

UNARY_OPS = {
    ast.UAdd: operator.pos,
    ast.USub: operator.neg,
}

Point = Tuple[float, float]

def cross(o: Point, a: Point, b: Point) -> float:
    """Cross product (OA × OB). >0 means vector o->a is on the left side of o->b."""
    return (a[0] - o[0]) * (b[1] - o[1]) - (a[1] - o[1]) * (b[0] - o[0])


def convex_hull_labels(points: Dict[str, Point]) -> List[str]:
    """
    Compute the convex hull labels using Andrew's monotone chain algorithm.
    Returns the labels of the hull vertices in counterclockwise order.
    Note: collinear points on edges are removed; only real corner points are kept.
    If all points are collinear, the number of returned points may be less than 3.
    """
    if len(points) == 0:
        return []

    # Sort by (x, y)
    items = sorted(points.items(), key=lambda kv: (kv[1][0], kv[1][1]))
    labels_sorted = [lb for lb, _ in items]

    def build_half(seq: List[str]) -> List[str]:
        half: List[str] = []
        for lb in seq:
            p = points[lb]
            while len(half) >= 2:
                o = points[half[-2]]
                a = points[half[-1]]
                if cross(o, a, p) <= 0:
                    # <= 0: remove collinear and right‑turn points to ensure a strictly convex hull
                    half.pop()
                else:
                    break
            half.append(lb)
        return half

    lower = build_half(labels_sorted)
    upper = build_half(list(reversed(labels_sorted)))

    # Remove duplicated endpoints
    hull = lower[:-1] + upper[:-1]

    return hull


def can_form_convex_polygon(points: Dict[str, Point]) -> bool:
    """
    Check whether these points can form a strictly convex polygon:
    - at least 3 points
    - all points lie on the convex hull (i.e. number of hull vertices == total number of points)
    - no point is allowed to lie strictly on an edge (collinear inner points already removed in
      `convex_hull_labels` by using `<= 0` in the cross‑product test)
    """
    n = len(points)
    if n < 3:
        return False

    hull = convex_hull_labels(points)

    # If hull vertex count < 3: either too few points or all collinear
    if len(hull) < 3:
        return False

    # Only when all points lie on the hull do we have a strictly convex polygon
    if len(hull) != n:
        return False

    return True


def infer_convex_polygon_order(points: Dict[str, Point]) -> List[str]:
    """
    Assuming these points can form a convex polygon,
    return the vertex labels in counterclockwise order.

    Raise ValueError if they cannot form a strictly convex polygon.
    """
    if not can_form_convex_polygon(points):
        raise ValueError("These points cannot form a strictly convex polygon")

    hull = convex_hull_labels(points)
    return hull  # hull is already in CCW order


def on_segment(a: Point, b: Point, p: Point) -> bool:
    """Check whether point p lies on segment ab (including endpoints)."""
    return (min(a[0], b[0]) <= p[0] <= max(a[0], b[0]) and
            min(a[1], b[1]) <= p[1] <= max(a[1], b[1]) and
            cross(a, b, p) == 0)


def segments_intersect(a: Point, b: Point, c: Point, d: Point) -> bool:
    """
    Check whether segments ab and cd intersect.
    - True: they have an intersection point (including endpoint overlap or collinear overlap)
    - False: they do not intersect
    """
    c1 = cross(a, b, c)
    c2 = cross(a, b, d)
    c3 = cross(c, d, a)
    c4 = cross(c, d, b)

    # General proper intersection
    if c1 * c2 < 0 and c3 * c4 < 0:
        return True

    # Special cases: collinear + endpoint lies on the other segment
    if c1 == 0 and on_segment(a, b, c): return True
    if c2 == 0 and on_segment(a, b, d): return True
    if c3 == 0 and on_segment(c, d, a): return True
    if c4 == 0 and on_segment(c, d, b): return True

    return False


def polygon_area(points: List[Point]) -> float:
    """
    Area of a polygon using the shoelace formula.
    `points`: [(x1, y1), ..., (xn, yn)] in edge order.
    """
    n = len(points)
    if n < 3:
        return 0.0

    s1 = s2 = 0.0
    for i in range(n):
        x1, y1 = points[i]
        x2, y2 = points[(i + 1) % n]
        s1 += x1 * y2
        s2 += y1 * x2
    return 0.5 * abs(s1 - s2)


def is_simple_polygon(points: List[Point]) -> bool:
    """
    Check whether a polygon defined by the ordered vertices is a simple polygon:
    - no self‑intersections
    - concave allowed
    - at least 3 vertices
    """
    n = len(points)
    if n < 3:
        return False

    # 1. vertices must be unique
    if len(set(points)) != n:
        return False

    # 2. area must be non‑zero (exclude fully collinear or degenerate polygons)
    if polygon_area(points) == 0:
        return False

    # 3. check that all edge pairs without a common endpoint do not intersect
    # edge i is (points[i], points[(i+1)%n])
    for i in range(n):
        a1 = points[i]
        a2 = points[(i + 1) % n]
        for j in range(i + 1, n):
            b1 = points[j]
            b2 = points[(j + 1) % n]

            # Skip if two edges share endpoints (adjacent or first/last edge)
            if a1 == b1 or a1 == b2 or a2 == b1 or a2 == b2:
                continue

            if segments_intersect(a1, a2, b1, b2):
                return False

    return True

def calculate_area(points: List[Point]) -> float:
    """
    Calculate area of a polygon via the shoelace formula.
    `points`: [(x1, y1), ..., (xn, yn)] in edge order.
    """
    if is_simple_polygon(points):
        return polygon_area(points)
    else:
        return 0.0

def segment_length(a: Point, b: Point) -> float:
    """Length of segment AB."""
    return math.hypot(b[0] - a[0], b[1] - a[1])

def calculate_perimeter(points: List[Point]) -> float:
    """
    Calculate the perimeter of a polygon.
    `points`: [(x1, y1), ..., (xn, yn)] in edge order.
    """
    if is_simple_polygon(points):
        perimeter = 0.0
        for i in range(len(points)):
            perimeter += segment_length(points[i], points[(i + 1) % len(points)])
        return perimeter
    
    else:
        return 0.0

def angle_between_vectors(u: Point, v: Point) -> float:
    """
    Angle between vectors `u` and `v` in radians within [0, π].
    `u`, `v`: (x, y)
    """
    ux, uy = u
    vx, vy = v
    dot = ux * vx + uy * vy
    nu = math.hypot(ux, uy)
    nv = math.hypot(vx, vy)
    if nu == 0 or nv == 0:
        raise ValueError("Zero-length vector in angle computation")

    cos_theta = dot / (nu * nv)
    # Clamp for numerical errors to avoid domain issues for acos
    cos_theta = max(-1.0, min(1.0, cos_theta))
    return math.acos(cos_theta)

def angle_at(A: Point, B: Point, C: Point) -> float:
    """
    Return the size of angle ∠ABC in radians.
    """
    BA = (A[0] - B[0], A[1] - B[1])
    BC = (C[0] - B[0], C[1] - B[1])
    return angle_between_vectors(BA, BC)

def angle_at_deg(A: Point, B: Point, C: Point) -> float:
    """
    Return the size of angle ∠ABC in degrees within [0, 180].
    """
    return math.degrees(angle_at(A, B, C))

def angle_between_lines(A: Point, B: Point, C: Point, D: Point) -> float:
    """
    Compute the angle between line AB and line CD in degrees within [0°, 90°].
    The angle is normalized to [0°, 90°] so that all trigonometric values are non‑negative.
    """
    # Direction vectors of the two lines
    vec1 = (B[0] - A[0], B[1] - A[1])
    vec2 = (D[0] - C[0], D[1] - C[1])
    
    # Angle in radians, in [0, π]
    angle_rad = angle_between_vectors(vec1, vec2)
    
    # Convert to degrees (0°~180°)
    angle_deg = math.degrees(angle_rad)
    
    # Normalize to [0°, 90°]; if >90°, use its supplementary angle
    if angle_deg > 90.0:
        angle_deg = 180.0 - angle_deg
    
    return angle_deg

def length(a, b):
    return math.hypot(b[0]-a[0], b[1]-a[1])

def feq(a, b, eps=EPS):
    """
    Float approximate‑equality check with None safety.
    If either side is None, return False immediately to avoid type errors.
    """
    if a is None or b is None:
        return False
    return abs(a - b) <= eps

def check_on_circle(O, A, B):
    # Check whether A and B lie on the same circle O
    r1 = length(O[0], A)
    r2 = length(O[0], B)
    return feq(r1, O[1]) and feq(r2, O[1])

def central_angle_rad(O, A, B):
    """Return the central angle ∠AOB in radians within [0, π] (the minor central angle)."""
    ax, ay = A[0] - O[0], A[1] - O[1]
    bx, by = B[0] - O[0], B[1] - O[1]
    dot = ax * bx + ay * by
    na = math.hypot(ax, ay)
    nb = math.hypot(bx, by)
    
    cos_theta = dot / (na * nb)
    cos_theta = max(-1.0, min(1.0, cos_theta))  # numerical clamping
    return math.acos(cos_theta)

def central_angle_deg(O, A, B):
    # Central angle in degrees
    if check_on_circle(O, A, B):
        return math.degrees(central_angle_rad(O[0], A, B))
    else:
        return 0.0

def arc_length(O, A, B):
    # Arc length between points A and B on circle O
    if check_on_circle(O, A, B):
        r = O[1]
        theta = central_angle_rad(O[0], A, B)
        return r * theta
    else:
        return 0.0

def sector_area(O, A, B):
    # Area of the sector formed by OA and OB
    if check_on_circle(O, A, B):
        r = O[1]
        theta = central_angle_rad(O[0], A, B)
        return 0.5 * r * r * theta
    else:
        return 0.0

def arc_inscribed_angle(O, A, B):
    # Inscribed angle that intercepts arc AB on circle O
    if check_on_circle(O, A, B):
        return 0.5 * math.degrees(central_angle_rad(O[0], A, B))
    else:
        return 0.0

def circle_area(O):
    """
    Compute the area of a circle.
    O: (center, radius) tuple, where center is (x, y) and radius is r.
    Returns: π * r².
    """
    if O is None or len(O) != 2:
        return 0.0
    r = O[1]
    return math.pi * r * r

def circle_perimeter(O):
    """
    Compute the perimeter (circumference) of a circle.
    O: (center, radius) tuple, where center is (x, y) and radius is r.
    Returns: 2 * π * r.
    """
    if O is None or len(O) != 2:
        return 0.0
    r = O[1]
    return 2 * math.pi * r

def circle_radius(O):
    """
    Return the radius of the circle.
    O: (center, radius) tuple, where center is (x, y) and radius is r.
    Returns: r.
    """
    if O is None or len(O) != 2:
        return 0.0
    return O[1]

def circle_diameter(O):
    """
    Return the diameter of the circle.
    O: (center, radius) tuple, where center is (x, y) and radius is r.
    Returns: 2 * r.
    """
    if O is None or len(O) != 2:
        return 0.0
    r = O[1]
    return 2 * r

def segment_area(O, A, B):
    """
    Compute the area of a circular segment defined by two points A and B on circle O.
    Segment area = sector area − area of triangle OAB.
    O: (center, radius) tuple.
    A, B: two points on the circle.
    Returns: segment area.
    """
    if not check_on_circle(O, A, B):
        return 0.0
    
    # Sector area
    sector = sector_area(O, A, B)
    
    # Area of triangle OAB (via cross product)
    center = O[0]
    OA = (A[0] - center[0], A[1] - center[1])
    OB = (B[0] - center[0], B[1] - center[1])
    # Triangle area is half of the absolute value of the cross product
    triangle_area = 0.5 * abs(OA[0] * OB[1] - OA[1] * OB[0])
    
    # Segment area = sector area − triangle area
    return sector - triangle_area

class NumericalCheck:
    def __init__(self):
        self.plotter = RealWorldPlotter()
    
    def _calculate_circle_from_three_points(
        self, 
        p1: Tuple[int, int],
        p2: Tuple[int, int],
        p3: Tuple[int, int],
    ) -> Tuple[Optional[Tuple[float, float]], Optional[float]]:
        """Compute center and radius of the circle through three points."""
        # Intersection of two perpendicular bisectors is the circle center
        # Perpendicular bisector 1: passes through midpoint of p1p2 and is perpendicular to p1p2
        mid1 = [(p1[0] + p2[0]) / 2, (p1[1] + p2[1]) / 2]
        dir1 = [p2[0] - p1[0], p2[1] - p1[1]]
        # Perpendicular direction
        perp1 = [-dir1[1], dir1[0]]
        
        # Perpendicular bisector 2: passes through midpoint of p2p3 and is perpendicular to p2p3
        mid2 = [(p2[0] + p3[0]) / 2, (p2[1] + p3[1]) / 2]
        dir2 = [p3[0] - p2[0], p3[1] - p2[1]]
        perp2 = [-dir2[1], dir2[0]]
        
        # Solve the intersection of the two lines
        det = perp1[0] * perp2[1] - perp1[1] * perp2[0]
        if abs(det) < 1e-10:
            # Fall back to using the perpendicular bisector of p1p3
            mid2 = [(p1[0] + p3[0]) / 2, (p1[1] + p3[1]) / 2]
            dir2 = [p3[0] - p1[0], p3[1] - p1[1]]
            perp2 = [-dir2[1], dir2[0]]
            det = perp1[0] * perp2[1] - perp1[1] * perp2[0]
        
        if abs(det) < 1e-10:
            # Three points are collinear, no valid circle
            return None, None
        
        # Solve for parameter t
        dx = mid2[0] - mid1[0]
        dy = mid2[1] - mid1[1]
        t = (dx * perp2[1] - dy * perp2[0]) / det
        
        center = (mid1[0] + t * perp1[0], mid1[1] + t * perp1[1])
        radius = math.sqrt((center[0] - p1[0])**2 + (center[1] - p1[1])**2)
        
        return center, radius
        
    def _calculate_circle_info_for_check(
        self,
        points: Dict[str, Tuple[int, int]],
        circles: Iterable[Sequence],
    ):
    
        # TODO: align with plotter
        """
        Compute center and radius for circles while preserving their IDs.
        Return format:
            {
                circle_name: (center_x, center_y), radius),
                ...
            }
        """

        def _is_digit(x):
            if isinstance(x, (int, float)):
                return True
            if isinstance(x, str):
                return x.replace(".", "", 1).isdigit()
            return False

        # Normalize point names to lower case
        pmap = {str(k).lower(): v for k, v in points.items()}

        result = {}

        for circle_def in circles:
            if not circle_def or not isinstance(circle_def, (list, tuple)):
                continue

            circle_name = circle_def[0]   # Keep circle ID, e.g. "C1"
            params = circle_def[1:]

            center = None
            radius = None

            # ---- Format 1: diameter ["A","B","diameter"] ----
            if len(params) == 3 and params[2] == "diameter":
                p1, p2 = params[0].lower(), params[1].lower()
                if p1 in pmap and p2 in pmap:
                    A = pmap[p1]
                    B = pmap[p2]
                    center = ((A[0] + B[0]) / 2, (A[1] + B[1]) / 2)
                    radius = float(math.dist(A, B) / 2)

            # ---- Format 2: circle defined by three points ["A","B","C"] ----
            elif len(params) == 3 and all(isinstance(x, str) for x in params):
                p1, p2, p3 = (p.lower() for p in params)
                if p1 in pmap and p2 in pmap and p3 in pmap:
                    ctuple, rfloat = self._calculate_circle_from_three_points(
                        pmap[p1], pmap[p2], pmap[p3]
                    )
                    if ctuple and rfloat:
                        center = ctuple
                        radius = float(rfloat)

            # ---- Format 3: center + radius ["O", 5] or ["O", "P"] ----
            elif len(params) == 2:
                cname = params[0].lower()
                if cname in pmap:
                    center = pmap[cname]
                    descriptor = params[1]

                    if _is_digit(descriptor):
                        radius = float(descriptor)
                    else:
                        pname = str(descriptor).lower()
                        if pname in pmap:
                            radius = float(math.dist(center, pmap[pname]))

            if center is not None and radius is not None:
                # Align the circle ID access with what `eval_quantity_expr` uses:
                # there we access by `str(cid).lower()`, so convert keys to lower‑case
                # to avoid mismatches like "C1" vs "c1".
                result[str(circle_name).lower()] = (center, radius)
        return result

    def parse_quantity(self, expr):
        """
        Parse a DSL expression and return (op, args).
        Example: angle(ADB) -> ("angle", ["A","D","B"])
                 central_angle(C1, A, B) -> ("central_angle", ["C1","A","B"])
        """
        match = DSL_PATTERN.match(expr.strip())
        if not match:
            raise ValueError(f"Cannot parse DSL expression: {expr}")

        op = match.group(1)
        arg_raw = match.group(2)

        # Strip spaces
        arg_raw = arg_raw.replace(" ", "")

        # Parse arguments
        if "," in arg_raw:
            # Form like C1,A,B
            args = arg_raw.split(",")
        else:
            # Form like ADB
            args = [arg_raw]
        return op, args

    def eval_quantity_expr(self, expr: str, point, circle_info_list) -> float:
        """
        expr: e.g. "length(A,B) + length(C,D)/2"
        point: dict, point_name -> [x, y]
        circle_info_list: dict/list, circle_id -> circle information structure
        """
        # Normalize point names to lower case
        pmap = {str(k).lower(): v for k, v in point.items()}
        
        tree = ast.parse(expr, mode="eval")

        def _eval(node):
            # Numeric constant
            if isinstance(node, ast.Constant):
                return node.value

            # Unary operations: -x, +x
            if isinstance(node, ast.UnaryOp):
                op_type = type(node.op)
                if op_type not in UNARY_OPS:
                    raise ValueError(f"Unsupported unary operator: {op_type}")
                return UNARY_OPS[op_type](_eval(node.operand))

            # Binary operators: +, -, *, /, **
            if isinstance(node, ast.BinOp):
                op_type = type(node.op)
                if op_type not in BIN_OPS:
                    raise ValueError(f"Unsupported binary operator: {op_type}")
                left = _eval(node.left)
                right = _eval(node.right)
                return BIN_OPS[op_type](left, right)

            # Function calls: length(A,B), angle(A,B,C), ...
            if isinstance(node, ast.Call):
                if not isinstance(node.func, ast.Name):
                    raise ValueError("Only simple function names are supported")

                fname = node.func.id
                # Arguments can be:
                #   - point names: A, B, C -> ast.Name
                #   - numeric literals: 1, 2.5 -> Constant
                #   - circle IDs: C1, C2 -> ast.Name / Constant
                args = []
                for arg in node.args:
                    if isinstance(arg, ast.Name):
                        # Use string for point / circle IDs
                        args.append(arg.id)
                    else:
                        args.append(_eval(arg))

                # Map to the underlying geometric / math functions.
                # Note: point names must be converted to lower‑case to match `pmap`.
                if fname == "length":
                    a, b = args
                    return length(pmap[str(a).lower()], pmap[str(b).lower()])
                    
                elif fname == "angle":
                    a, b, c = args
                    return angle_at_deg(pmap[str(a).lower()], pmap[str(b).lower()], pmap[str(c).lower()])

                elif fname == "tan":
                    a, b, c = args
                    return math.tan(angle_at(pmap[str(a).lower()], pmap[str(b).lower()], pmap[str(c).lower()]))

                elif fname == "sin":
                    a, b, c = args
                    return math.sin(angle_at(pmap[str(a).lower()], pmap[str(b).lower()], pmap[str(c).lower()]))

                elif fname == "cos":
                    a, b, c = args
                    return math.cos(angle_at(pmap[str(a).lower()], pmap[str(b).lower()], pmap[str(c).lower()]))

                elif fname == "angle_between_lines":
                    a, b, c, d = args
                    return angle_between_lines(pmap[str(a).lower()], pmap[str(b).lower()], pmap[str(c).lower()], pmap[str(d).lower()])

                elif fname == "tan_between_lines":
                    a, b, c, d = args
                    angle_rad = math.radians(angle_between_lines(pmap[str(a).lower()], pmap[str(b).lower()], pmap[str(c).lower()], pmap[str(d).lower()]))
                    return math.tan(angle_rad)

                elif fname == "sin_between_lines":
                    a, b, c, d = args
                    angle_rad = math.radians(angle_between_lines(pmap[str(a).lower()], pmap[str(b).lower()], pmap[str(c).lower()], pmap[str(d).lower()]))
                    return math.sin(angle_rad)

                elif fname == "cos_between_lines":
                    a, b, c, d = args
                    angle_rad = math.radians(angle_between_lines(pmap[str(a).lower()], pmap[str(b).lower()], pmap[str(c).lower()], pmap[str(d).lower()]))
                    return math.cos(angle_rad)

                elif fname == "area":
                    # area(A,B,C,D,...) polygon area
                    # First check whether points can form a convex polygon; if so, use
                    # the hull order to compute the area.
                    if len(args) < 3:
                        return 0.0
                    
                    # Convert label list into Dict[str, Point] in lower case
                    points_dict = {str(p).lower(): tuple(pmap[str(p).lower()]) for p in args}
                    
                    # Check whether we can form a convex polygon
                    if can_form_convex_polygon(points_dict):
                        # Get CCW hull order
                        hull_order = infer_convex_polygon_order(points_dict)
                        # Reorder the points according to the hull
                        pts = [points_dict[p] for p in hull_order]
                        return polygon_area(pts)
                    else:
                        # If a convex polygon cannot be formed, return 0 (or raise error as needed)
                        return 0.0

                elif fname == "perimeter":
                    pts = [tuple(pmap[str(p).lower()]) for p in args]
                    return calculate_perimeter(pts)

                elif fname == "central_angle":
                    cid, p1, p2 = args
                    return central_angle_deg(circle_info_list[str(cid).lower()],
                                            tuple(pmap[str(p1).lower()]), tuple(pmap[str(p2).lower()]))

                elif fname == "arc_length":
                    cid, p1, p2 = args
                    return arc_length(circle_info_list[str(cid).lower()],
                                    pmap[str(p1).lower()], pmap[str(p2).lower()])

                elif fname == "sector_area":
                    cid, p1, p2 = args
                    return sector_area(circle_info_list[str(cid).lower()],
                                    pmap[str(p1).lower()], pmap[str(p2).lower()])

                elif fname == "arc_inscribed_angle":
                    cid, p1, p2 = args
                    return arc_inscribed_angle(circle_info_list[str(cid).lower()],
                                            pmap[str(p1).lower()], pmap[str(p2).lower()])

                elif fname == "circle_area":
                    cid = args[0]
                    return circle_area(circle_info_list[str(cid).lower()])

                elif fname == "circle_perimeter":
                    cid = args[0]
                    return circle_perimeter(circle_info_list[str(cid).lower()])

                elif fname == "segment_area":
                    cid, p1, p2 = args
                    return segment_area(circle_info_list[str(cid).lower()],
                                        pmap[str(p1).lower()], pmap[str(p2).lower()])

                elif fname == "radius":
                    cid = args[0]
                    return circle_radius(circle_info_list[str(cid).lower()])

                elif fname == "diameter":
                    cid = args[0]
                    return circle_diameter(circle_info_list[str(cid).lower()])
                
                # Support generic math function sqrt for scalar operations in quantities (e.g. sqrt(3))
                elif fname == "sqrt":
                    if len(args) != 1:
                        raise ValueError("sqrt expects exactly one argument")
                    return math.sqrt(float(args[0]))

                else:
                    raise ValueError(f"Unsupported function: {fname}")
            
            # Point / circle ID / constants appearing at top level
            if isinstance(node, ast.Name):
                name = node.id
                # Support math constants: pi / π
                if name.lower() in ("pi", "π"):
                    return math.pi
                # Any other bare identifier is invalid (e.g. single A / C1)
                raise ValueError(f"Unsupported bare identifier: {name}")
            
            raise ValueError(f"Unsupported AST node: {type(node)}")

        return _eval(tree.body)


    def check(self, answer, meta_data):
        """
        Numerical checking:
        - assume the incoming `answer` is already numeric (float / int)
        - do not perform LaTeX / string parsing here in NumericalCheck
        - parsing logic is handled upstream (e.g. in the Pipeline)
        """
        meta_exprs = meta_data["quantities"]
        point = meta_data["points"]
        circle = meta_data["circles"]
        annotations = meta_data.get("annotations", {})
        circle_info_list = self._calculate_circle_info_for_check(point, circle)
        
        # Check predicate (quantities)
        predicate_passed = False
        for meta_expr in meta_exprs:
            num = self.eval_quantity_expr(meta_expr, point, circle_info_list)
            print(num, answer)
            if feq(num, answer):
                predicate_passed = True
                break
        
        if not predicate_passed:
            return False
        
        # Normalize point names to lower case
        pmap = {str(k).lower(): v for k, v in point.items()}
        
        # Check `length_of_line` annotations
        length_annotations = annotations.get("length_of_line", [])
        for length_entry in length_annotations:
            if not isinstance(length_entry, list) or len(length_entry) < 2:
                continue
            segment = length_entry[0]  # [["A", "B"], "5"]
            annotated_value_str = length_entry[1]
            
            if not isinstance(segment, list) or len(segment) < 2:
                continue
            
            point_a = str(segment[0]).lower()
            point_b = str(segment[1]).lower()
            
            if point_a not in pmap or point_b not in pmap:
                continue
            
            # Compute actual length
            actual_length = length(pmap[point_a], pmap[point_b])
            
            # Parse annotated value
            annotated_value = latex_to_float(annotated_value_str)
            if annotated_value is None:
                continue
            
            # Check consistency
            if not feq(actual_length, annotated_value):
                print(f"Length check failed: segment {point_a}{point_b}, actual={actual_length}, annotated={annotated_value}")
                return False
        
        # Check `right_angles` annotations
        right_angle_annotations = annotations.get("right_angles", [])
        for right_angle_entry in right_angle_annotations:
            if not isinstance(right_angle_entry, list) or len(right_angle_entry) < 3:
                continue
            
            point_a = str(right_angle_entry[0]).lower()
            point_b = str(right_angle_entry[1]).lower()  # vertex
            point_c = str(right_angle_entry[2]).lower()
            
            if point_a not in pmap or point_b not in pmap or point_c not in pmap:
                continue
            
            # Compute actual angle in degrees
            actual_angle = angle_at_deg(pmap[point_a], pmap[point_b], pmap[point_c])
            
            # Check if angle is 90° (with tolerance)
            if not feq(actual_angle, 90.0):
                print(f"Right angle check failed: angle {point_a}{point_b}{point_c}, actual={actual_angle} degrees")
                return False
        
        # Check `measure_of_angle` annotations
        angle_annotations = annotations.get("measure_of_angle", [])
        for angle_entry in angle_annotations:
            if not isinstance(angle_entry, list) or len(angle_entry) < 2:
                continue
            
            angle_points = angle_entry[0]  # [["A", "B", "C"], "30"]
            annotated_value_str = angle_entry[1]
            
            if not isinstance(angle_points, list) or len(angle_points) < 3:
                continue
            
            point_a = str(angle_points[0]).lower()
            point_b = str(angle_points[1]).lower()  # vertex
            point_c = str(angle_points[2]).lower()
            
            if point_a not in pmap or point_b not in pmap or point_c not in pmap:
                continue
            
            # Compute actual angle in degrees
            actual_angle = angle_at_deg(pmap[point_a], pmap[point_b], pmap[point_c])
            
            # Parse annotated value
            annotated_value = latex_to_float(annotated_value_str)
            if annotated_value is None:
                continue
            
            # If the annotation string contains "pi", treat it as radians and
            # convert to degrees, so that expressions like 2*pi/3 are supported
            # while internal representation stays in degrees.
            if isinstance(annotated_value, (int, float)) and "pi" in str(annotated_value_str):
                annotated_value = annotated_value * 180.0 / math.pi
            
            # Check consistency
            if not feq(actual_angle, annotated_value):
                print(f"Angle check failed: angle {point_a}{point_b}{point_c}, actual={actual_angle}, annotated={annotated_value}")
                return False
        
        # All checks passed
        return True