
import ast
import threading
import time
from typing import Any, Callable, Dict, List, Optional, Tuple

import numpy as np

from ours_utils import (
    SkillCallInfo,
    extract_all_skill_calls,
    get_statement_code,
    create_projected_state,
    projected_state_to_scene_info,
    apply_skill_effect,
    basic_validate_skill_call,
    extract_target_object_from_call,
)

# =============================================================================
# MockEnv for simulation-based validation
# =============================================================================

class MockEnv:

    def __init__(self, captured_scene: Dict[str, Any]):
        self._scene = captured_scene
        self._objects = {
            obj["name"]: obj
            for obj in captured_scene.get("objects", [])
        }
        self._gripper = captured_scene.get("gripper", {})

    def get_obj_pos(self, obj_name: str) -> np.ndarray:
        if obj_name in self._objects:
            pos = self._objects[obj_name].get("position")
            if pos is not None:
                return np.array(pos)
        raise ValueError(f"Object '{obj_name}' not found in scene")

    def get_obj_bbox(self, obj_name: str) -> np.ndarray:
        if obj_name in self._objects:
            bbox = self._objects[obj_name].get("bbox")
            if bbox is not None:
                return np.array(bbox)
        raise ValueError(f"Object '{obj_name}' bbox not found")

    def is_obj_visible(self, obj_name: str) -> bool:
        if obj_name in self._objects:
            return self._objects[obj_name].get("visible", True)
        return False

    def get_gripper_pos(self) -> Optional[np.ndarray]:
        pos = self._gripper.get("position")
        return np.array(pos) if pos is not None else None

# =============================================================================
# Release pattern detection and collision checking
# =============================================================================

# Skills that indicate a "release" action when following move_to_position
RELEASE_SKILLS = {
    "open_gripper", "open_robotiq85",
    "deactivate_vacuum", "detach_vacuum_handle",
}

MOVE_SKILLS = {"move_to_position", "move_gripper_to"}

# Graspable object patterns - objects that can be picked up and placed
# Pattern: {color}_{shape}
GRASPABLE_SHAPES = {
    # Basic shapes
    "cube", "ball", "cylinder", "block", "sphere",
    # Chess pieces
    "pawn", "rook", "knight", "bishop", "queen", "king",
}
GRASPABLE_COLORS = {"red", "blue", "yellow", "green", "purple", "orange", "cyan", "pink", "white", "black"}

# Objects that are never graspable (containers, fixtures, etc.)
NON_GRASPABLE_OBJECTS = {
    "floor", "gripper", "table", "workspace",
    "toy_box", "tray", "box_full", "box_body", "box_lid",
    "hinge_box", "prismatic_drawer", "drawer",
}

def is_graspable_object(obj_name: str) -> bool:
    if not obj_name:
        return False

    obj_lower = obj_name.lower()

    # Exclude known non-graspable objects
    if obj_lower in NON_GRASPABLE_OBJECTS:
        return False

    # Exclude collision/wall objects
    if "collision" in obj_lower or "wall" in obj_lower:
        return False

    # Check if matches {color}_{shape} pattern
    parts = obj_lower.split("_")
    if len(parts) >= 2:
        # Check if any part is a graspable shape
        for part in parts:
            if part in GRASPABLE_SHAPES:
                return True

    return False

def is_release_pattern(current_skill: str, next_skill: Optional[str]) -> bool:
    if next_skill is None:
        return False
    return current_skill.lower() in MOVE_SKILLS and next_skill.lower() in RELEASE_SKILLS

def check_collision_and_find_alternative(
    target_pos: np.ndarray,
    scene_info: Dict[str, Any],
    held_object: Optional[str],
    collision_threshold: float = 0.02,  # cube size ~0.03m, so 0.02 is safe margin
    search_radius: float = 0.05,
    grid_steps: int = 7,
    target_surface_bbox: Optional[List[List[float]]] = None,  # Surface constraint
) -> Tuple[bool, Optional[str], Optional[List[float]]]:
    # Debug: Print scene_info summary
    print(f"[BG_VALIDATION] === Scene Info Debug ===")
    print(f"[BG_VALIDATION]   held_object: {held_object}")
    print(f"[BG_VALIDATION]   target_pos: [{target_pos[0]:.3f}, {target_pos[1]:.3f}, {target_pos[2]:.3f}]")
    print(f"[BG_VALIDATION]   collision_threshold: {collision_threshold}, search_radius: {search_radius}")

    # Collect graspable object positions
    occupied = []

    for obj in scene_info.get("objects", []):
        obj_name = obj.get("name", "")
        obj_pos = obj.get("position", [0, 0, 0])
        is_grasped = obj.get("is_grasped", False)
        graspable = is_graspable_object(obj_name)

        obj_bbox = obj.get("bbox", [[0,0,0], [0,0,0]])
        print(f"[BG_VALIDATION]   {obj_name}: pos=[{obj_pos[0]:.3f}, {obj_pos[1]:.3f}, {obj_pos[2]:.3f}], "
              f"bbox_xy=[{obj_bbox[0][0]:.3f},{obj_bbox[0][1]:.3f}]~[{obj_bbox[1][0]:.3f},{obj_bbox[1][1]:.3f}], "
              f"grasped={is_grasped}, graspable={graspable}")

        # Skip non-graspable objects
        if not graspable:
            continue
        # Skip the object being held
        if obj_name == held_object:
            continue
        # Skip already grasped objects
        if is_grasped:
            continue

        # Store bbox for XY overlap check: (obj_name, bbox_min_xy, bbox_max_xy)
        bbox_min_xy = np.array([obj_bbox[0][0], obj_bbox[0][1]])
        bbox_max_xy = np.array([obj_bbox[1][0], obj_bbox[1][1]])
        occupied.append((obj_name, bbox_min_xy, bbox_max_xy))

    print(f"[BG_VALIDATION] === End Scene Info ===")
    print(f"[BG_VALIDATION] Objects to check collision against: {[o[0] for o in occupied]}")

    if not occupied:
        print(f"[BG_VALIDATION] No graspable objects to check - no collision")
        return False, None, None

    # Get held object's half-size for bbox calculation
    # Try to find held_object's bbox in scene_info, otherwise use default
    held_half_size = collision_threshold
    if held_object:
        for obj in scene_info.get("objects", []):
            if obj.get("name") == held_object:
                obj_bbox = obj.get("bbox", [[0,0,0], [0,0,0]])
                # Calculate half-size from bbox
                held_half_size = max(
                    (obj_bbox[1][0] - obj_bbox[0][0]) / 2,
                    (obj_bbox[1][1] - obj_bbox[0][1]) / 2
                )
                print(f"[BG_VALIDATION] Held object '{held_object}' half_size: {held_half_size:.3f}")
                break

    # Check collision at target_pos using XY bbox overlap (ignoring Z)
    # Rationale: When releasing an object, gravity will cause it to fall.
    # So if XY bboxes overlap, there WILL be collision regardless of release height.
    target_xy = target_pos[:2]

    # Create bbox for the object being placed (centered at target_pos)
    margin = 0.005
    target_min_xy = target_xy - held_half_size - margin
    target_max_xy = target_xy + held_half_size + margin

    print(f"[BG_VALIDATION] Target bbox_xy: [{target_min_xy[0]:.3f},{target_min_xy[1]:.3f}]~[{target_max_xy[0]:.3f},{target_max_xy[1]:.3f}]")

    colliding_obj = None

    for obj_name, obj_min_xy, obj_max_xy in occupied:
        # Check XY bbox overlap (AABB collision)
        # Two boxes overlap if: min1 < max2 AND max1 > min2 (for both axes)
        x_overlap = target_min_xy[0] < obj_max_xy[0] and target_max_xy[0] > obj_min_xy[0]
        y_overlap = target_min_xy[1] < obj_max_xy[1] and target_max_xy[1] > obj_min_xy[1]

        print(f"[BG_VALIDATION] Checking {obj_name}: "
              f"x_overlap={x_overlap} (target[{target_min_xy[0]:.3f},{target_max_xy[0]:.3f}] vs obj[{obj_min_xy[0]:.3f},{obj_max_xy[0]:.3f}]), "
              f"y_overlap={y_overlap} (target[{target_min_xy[1]:.3f},{target_max_xy[1]:.3f}] vs obj[{obj_min_xy[1]:.3f},{obj_max_xy[1]:.3f}])")

        if x_overlap and y_overlap:
            colliding_obj = obj_name
            print(f"[BG_VALIDATION] >>> COLLISION with {obj_name}! (XY bbox overlap - object will fall onto it)")
            break

    if not colliding_obj:
        print(f"[BG_VALIDATION] No collision detected - position is clear")
        return False, None, None

    # Collision detected - find alternative position
    # If target_surface_bbox is provided, search within that surface
    # Otherwise, search within search_radius around target_pos

    if target_surface_bbox is not None:
        # Search within surface bbox (e.g., board, box)
        surface_min_xy = np.array([target_surface_bbox[0][0], target_surface_bbox[0][1]])
        surface_max_xy = np.array([target_surface_bbox[1][0], target_surface_bbox[1][1]])
        print(f"[BG_VALIDATION] Searching within surface bbox: [{surface_min_xy[0]:.3f},{surface_min_xy[1]:.3f}]~[{surface_max_xy[0]:.3f},{surface_max_xy[1]:.3f}]")
    else:
        # Search within radius around target_pos
        surface_min_xy = target_xy - search_radius
        surface_max_xy = target_xy + search_radius
        print(f"[BG_VALIDATION] Searching within radius={search_radius} around target")

    best_pos = None
    best_dist_to_target = float('inf')

    for i in range(grid_steps):
        for j in range(grid_steps):
            x = surface_min_xy[0] + (surface_max_xy[0] - surface_min_xy[0]) * (i + 0.5) / grid_steps
            y = surface_min_xy[1] + (surface_max_xy[1] - surface_min_xy[1]) * (j + 0.5) / grid_steps
            candidate = np.array([x, y])

            # Check if candidate position has bbox overlap with any object
            cand_min_xy = candidate - held_half_size - margin
            cand_max_xy = candidate + held_half_size + margin

            has_overlap = False

            for _, obj_min_xy, obj_max_xy in occupied:
                # Check XY bbox overlap
                x_overlap = cand_min_xy[0] < obj_max_xy[0] and cand_max_xy[0] > obj_min_xy[0]
                y_overlap = cand_min_xy[1] < obj_max_xy[1] and cand_max_xy[1] > obj_min_xy[1]

                if x_overlap and y_overlap:
                    has_overlap = True
                    break

            # Find the closest clear position to original target
            if not has_overlap:
                dist_to_target = np.linalg.norm(candidate - target_xy)
                if dist_to_target < best_dist_to_target:
                    best_dist_to_target = dist_to_target
                    best_pos = candidate

    if best_pos is None:
        # Fallback: move slightly away from colliding object
        for obj_name, obj_min_xy, obj_max_xy in occupied:
            if obj_name == colliding_obj:
                # Move in Y direction, just past the object
                offset = held_half_size + margin + 0.01
                fallback_y = obj_max_xy[1] + offset
                # Clamp to surface if constrained
                if target_surface_bbox is not None:
                    fallback_y = min(fallback_y, surface_max_xy[1] - held_half_size)
                best_pos = np.array([target_xy[0], fallback_y])
                break
        if best_pos is None:
            best_pos = target_xy + np.array([0, search_radius])
        print(f"[BG_VALIDATION] Warning: No clear position in grid, using fallback")

    alternative = [float(best_pos[0]), float(best_pos[1]), float(target_pos[2])]

    print(f"[BG_VALIDATION] Found alternative position: [{alternative[0]:.3f}, {alternative[1]:.3f}, {alternative[2]:.3f}] "
          f"(dist_from_original={best_dist_to_target:.3f})")

    msg = (
        f"Release position [{target_pos[0]:.3f}, {target_pos[1]:.3f}, {target_pos[2]:.3f}] "
        f"has XY bbox overlap with '{colliding_obj}' (object will fall onto it). "
        f"Use alternative position: [{alternative[0]:.3f}, {alternative[1]:.3f}, {alternative[2]:.3f}]"
    )

    return True, msg, alternative

# Keep old function name as alias for compatibility
def check_position_collision(
    target_pos: np.ndarray,
    scene_info: Dict[str, Any],
    held_object: Optional[str],
    collision_threshold: float = 0.02,
) -> Optional[str]:
    has_collision, msg, _ = check_collision_and_find_alternative(
        target_pos, scene_info, held_object, collision_threshold
    )
    return msg if has_collision else None

# =============================================================================
# Simulation-based validation
# =============================================================================

def simulate_future_code_and_extract_positions(
    future_code: str,
    func_name: str,
    captured_scene: Dict[str, Any],
    subtask_kwargs: Optional[Dict[str, Any]] = None,
) -> List[Dict[str, Any]]:
    results = []

    try:
        mock_env = MockEnv(captured_scene)

        # Track move_to_position calls and their positions
        recorded_calls: List[Dict[str, Any]] = []

        # Create mock skill functions that record calls
        def mock_move_to_position(env, pos, **kwargs):
            pos_array = np.array(pos) if not isinstance(pos, np.ndarray) else pos
            recorded_calls.append({
                "skill_name": "move_to_position",
                "target_pos": pos_array.copy(),
            })

        def mock_move_gripper_to(env, obj_name, **kwargs):
            try:
                pos = env.get_obj_pos(obj_name)
                recorded_calls.append({
                    "skill_name": "move_gripper_to",
                    "target_pos": pos.copy(),
                    "target_obj": obj_name,
                })
            except:
                recorded_calls.append({
                    "skill_name": "move_gripper_to",
                    "target_pos": None,
                    "target_obj": obj_name,
                })
            return True

        def mock_noop(*args, **kwargs):
            pass

        def mock_noop_return_true(*args, **kwargs):
            return True

        # Record open/release skills
        def make_release_recorder(skill_name):
            def recorder(*args, **kwargs):
                recorded_calls.append({
                    "skill_name": skill_name,
                    "target_pos": None,
                })
            return recorder

        # Build execution namespace
        exec_globals = {
            "env": mock_env,
            "np": np,
            # Mock env methods
            "get_obj_pos": mock_env.get_obj_pos,
            "get_obj_bbox": mock_env.get_obj_bbox,
            "is_obj_visible": mock_env.is_obj_visible,
            # Mock skill functions
            "move_to_position": mock_move_to_position,
            "move_gripper_to": mock_move_gripper_to,
            "move_parallel": mock_noop,
            "rotate_gripper": mock_noop,
            # Gripper skills
            "close_gripper": mock_noop,
            "open_gripper": make_release_recorder("open_gripper"),
            "close_robotiq85": mock_noop,
            "open_robotiq85": make_release_recorder("open_robotiq85"),
            "activate_vacuum": mock_noop,
            "deactivate_vacuum": make_release_recorder("deactivate_vacuum"),
            # Other common functions
            "print": lambda *args, **kwargs: None,
        }

        exec_locals = {}

        # Execute the code to define the function
        exec(future_code, exec_globals, exec_locals)

        # Get the function and call it with subtask arguments
        if func_name in exec_locals:
            func = exec_locals[func_name]
            try:
                kwargs = subtask_kwargs or {}

                # Build positional args from common parameter names
                # Order: obj_name, target_name (matching typical subtask signatures)
                pos_args = []
                if "obj_name" in kwargs and kwargs["obj_name"] is not None:
                    pos_args.append(kwargs["obj_name"])
                if "target_name" in kwargs and kwargs["target_name"] is not None:
                    pos_args.append(kwargs["target_name"])

                if pos_args:
                    args_str = ", ".join(f"'{a}'" for a in pos_args)
                    print(f"[BG_VALIDATION] Simulating {func_name}({args_str})")
                    func(mock_env, *pos_args)
                else:
                    # Fallback: try to call with first object in scene
                    obj_names = list(mock_env._objects.keys())
                    if obj_names:
                        print(f"[BG_VALIDATION] Simulating {func_name}('{obj_names[0]}') [fallback]")
                        func(mock_env, obj_names[0])
            except Exception as e:
                print(f"[BG_VALIDATION] Error calling simulated function: {e}")

        results = recorded_calls

        # Log simulation results
        if results:
            print(f"[BG_VALIDATION] Simulation recorded {len(results)} skill calls:")
            for i, call in enumerate(results):
                pos_str = f"pos={call['target_pos']}" if call.get('target_pos') is not None else "pos=None"
                print(f"  [{i}] {call['skill_name']} {pos_str}")
        else:
            print(f"[BG_VALIDATION] Simulation recorded no skill calls")

    except Exception as e:
        print(f"[BG_VALIDATION] Simulation error: {e}")

    return results

def validate_release_positions(
    skill_calls: List[Dict[str, Any]],
    scene_info: Dict[str, Any],
    held_object: Optional[str],
    subtask_kwargs: Optional[Dict[str, Any]] = None,
) -> List[Dict[str, Any]]:
    violations = []
    release_patterns_found = 0

    # Extract target_surface_bbox from subtask_kwargs if target_name is non-graspable
    target_surface_bbox = None
    if subtask_kwargs:
        target_name = subtask_kwargs.get("target_name")
        if target_name and not is_graspable_object(target_name):
            # Find target's bbox in scene
            for obj in scene_info.get("objects", []):
                if obj.get("name") == target_name:
                    target_surface_bbox = obj.get("bbox")
                    print(f"[BG_VALIDATION] Using surface constraint from '{target_name}': {target_surface_bbox}")
                    break

    for i, call in enumerate(skill_calls):
        skill_name = call.get("skill_name", "")
        target_pos = call.get("target_pos")

        if target_pos is None:
            continue

        # Check if this is part of a release pattern
        next_skill = skill_calls[i + 1].get("skill_name") if i + 1 < len(skill_calls) else None

        if is_release_pattern(skill_name, next_skill):
            release_patterns_found += 1
            print(f"[BG_VALIDATION] Found release pattern: {skill_name} -> {next_skill}")
            print(f"[BG_VALIDATION]   target_pos={target_pos.tolist()}, held_object={held_object}")

            has_collision, collision_msg, alternative_pos = check_collision_and_find_alternative(
                target_pos,
                scene_info,
                held_object,
                target_surface_bbox=target_surface_bbox,
            )
            if has_collision:
                print(f"[BG_VALIDATION]   COLLISION DETECTED: {collision_msg}")
                violations.append({
                    "skill_name": skill_name,
                    "target_pos": target_pos.tolist(),
                    "violation": collision_msg,
                    "next_skill": next_skill,
                    "alternative_pos": alternative_pos,
                })
            else:
                print(f"[BG_VALIDATION]   No collision - position is clear")

    if release_patterns_found == 0:
        print(f"[BG_VALIDATION] No release patterns found in skill calls")
    else:
        print(f"[BG_VALIDATION] Checked {release_patterns_found} release patterns, found {len(violations)} collisions")

    return violations

class BackgroundValidator:

    def __init__(
        self,
        batch_repair_func: Optional[Callable] = None,
        log_llm_call_func: Optional[Callable] = None,
    ):
        self.batch_repair_func = batch_repair_func
        self.log_llm_call_func = log_llm_call_func

        # Thread synchronization
        self._lock = threading.Lock()
        self._pending_repair_code: Optional[str] = None
        self._thread: Optional[threading.Thread] = None
        self._last_response_num_tokens: Optional[int] = None

    def start_validation(
        self,
        current_code: str,
        subtask_func_name: str,
        executed_stmt_texts: List[str],
        pre_captured_scene: Dict[str, Any],
        subtask_kwargs: Optional[Dict[str, Any]] = None,
    ) -> None:
        self._thread = threading.Thread(
            target=self._validate_future_statements,
            args=(
                current_code,
                subtask_func_name,
                executed_stmt_texts,
                pre_captured_scene,
                subtask_kwargs,
            ),
            daemon=True,
        )
        self._thread.start()

    def wait_for_completion(self, timeout: float = 1.0) -> None:
        if self._thread is not None and self._thread.is_alive():
            self._thread.join(timeout=timeout)

    def get_pending_repair(self) -> Optional[str]:
        with self._lock:
            code = self._pending_repair_code
            self._pending_repair_code = None
            return code

    def has_pending_repair(self) -> bool:
        with self._lock:
            return self._pending_repair_code is not None

    def _validate_future_statements(
        self,
        current_code: str,
        subtask_func_name: str,
        executed_stmt_texts: List[str],
        pre_captured_scene: Dict[str, Any],
        subtask_kwargs: Optional[Dict[str, Any]] = None,
    ) -> None:
        try:
            # Try to import validate_skill_preconditions if available
            try:
                from ours_utils.validate_conditions import validate_skill_preconditions
                has_validator = True
            except ImportError:
                has_validator = False

            invalid_statements: List[Dict[str, Any]] = []

            # Extract ALL skill calls including those in compound statements
            skill_calls, src_lines = extract_all_skill_calls(current_code, subtask_func_name)

            if not skill_calls:
                return

            # Create a mutable copy of scene state for projection
            projected_state = create_projected_state(pre_captured_scene)

            # ================================================================
            # Phase 1: AST-based validation (existing logic)
            # ================================================================
            for skill_info in skill_calls:
                skill_name = skill_info.skill_name
                call_node = skill_info.call_node
                stmt = skill_info.stmt

                stmt_code = get_statement_code(stmt, src_lines)

                # Convert projected state back to scene_info format for validation
                current_scene = projected_state_to_scene_info(projected_state, pre_captured_scene)

                # Validate preconditions
                if has_validator:
                    # Extract target object from call node
                    target_obj = extract_target_object_from_call(call_node)

                    validation_result = validate_skill_preconditions(
                        skill_name=skill_name,
                        scene_info=current_scene,
                        call_node=call_node,
                        target_obj=target_obj,
                    )

                    if not validation_result.get("valid", True):
                        invalid_statements.append({
                            "line_number": skill_info.lineno,
                            "statement_code": stmt_code,
                            "skill_name": skill_name,
                            "violations": validation_result.get("violations", []),
                            "warnings": validation_result.get("warnings", []),
                            "projected_state": current_scene,
                            "in_conditional": skill_info.in_conditional,
                            "in_loop": skill_info.in_loop,
                            "parent_context": skill_info.parent_context,
                        })
                else:
                    # Basic validation without full validator
                    violations = basic_validate_skill_call(
                        skill_name, call_node, current_scene
                    )
                    if violations:
                        invalid_statements.append({
                            "line_number": skill_info.lineno,
                            "statement_code": stmt_code,
                            "skill_name": skill_name,
                            "violations": violations,
                            "warnings": [],
                            "projected_state": current_scene,
                            "in_conditional": skill_info.in_conditional,
                            "in_loop": skill_info.in_loop,
                            "parent_context": skill_info.parent_context,
                        })

                # Apply skill effect to projected state for next iteration
                apply_skill_effect(projected_state, skill_name, call_node)

            # ================================================================
            # Phase 2: Simulation-based validation for release patterns
            # ================================================================
            try:
                # Get currently held object from scene
                gripper_info = pre_captured_scene.get("gripper", {})
                held_object = gripper_info.get("held_object")

                # Simulate code execution to get actual position values
                simulated_calls = simulate_future_code_and_extract_positions(
                    current_code,
                    subtask_func_name,
                    pre_captured_scene,
                    subtask_kwargs,
                )

                if simulated_calls:
                    # Check for release position collisions
                    release_violations = validate_release_positions(
                        simulated_calls,
                        pre_captured_scene,
                        held_object,
                        subtask_kwargs=subtask_kwargs,
                    )

                    for viol in release_violations:
                        print(f"[BG_VALIDATION] Release position collision detected: {viol['violation']}")
                        # Find the corresponding statement in skill_calls
                        for skill_info in skill_calls:
                            if skill_info.skill_name == viol["skill_name"]:
                                stmt_code = get_statement_code(skill_info.stmt, src_lines)
                                invalid_statements.append({
                                    "line_number": skill_info.lineno,
                                    "statement_code": stmt_code,
                                    "skill_name": viol["skill_name"],
                                    "violations": [viol["violation"]],
                                    "warnings": [],
                                    "projected_state": pre_captured_scene,
                                    "in_conditional": skill_info.in_conditional,
                                    "in_loop": skill_info.in_loop,
                                    "parent_context": skill_info.parent_context,
                                    "collision_info": {
                                        "target_pos": viol["target_pos"],
                                        "next_skill": viol["next_skill"],
                                    },
                                })
                                break

            except Exception as sim_error:
                print(f"[BG_VALIDATION] Simulation-based validation error: {sim_error}")

            # ================================================================
            # Phase 3: Request repair if violations found
            # ================================================================
            if invalid_statements and self.batch_repair_func is not None:
                print(f"[BG_VALIDATION] Found {len(invalid_statements)} invalid statements, requesting repair...")
                for inv in invalid_statements:
                    ctx = f" (in {inv['parent_context']})" if inv.get('parent_context') else ""
                    print(f"  - Line {inv['line_number']}: {inv['skill_name']}{ctx}")
                    for v in inv.get("violations", []):
                        print(f"    Violation: {v}")

                repair_start = time.time()
                new_code = self.batch_repair_func(
                    current_code=current_code,
                    executed_stmt_texts=executed_stmt_texts,
                    invalid_statements=invalid_statements,
                    scene=pre_captured_scene,
                    subtask_name=subtask_func_name,
                )
                repair_duration = time.time() - repair_start

                if new_code:
                    with self._lock:
                        self._pending_repair_code = new_code
                    print(f"[BG_VALIDATION] Batch repair completed in {repair_duration:.2f}s")
                    if self.log_llm_call_func:
                        self.log_llm_call_func(
                            "bg_batch_repair",
                            subtask_func_name,
                            repair_duration,
                            self._last_response_num_tokens,
                        )
                else:
                    print(f"[BG_VALIDATION] Batch repair returned no code change")

        except Exception as e:
            print(f"[BG_VALIDATION] Background validation error: {e}")

def create_background_validator(
    batch_repair_func: Optional[Callable] = None,
    log_llm_call_func: Optional[Callable] = None,
) -> BackgroundValidator:
    return BackgroundValidator(
        batch_repair_func=batch_repair_func,
        log_llm_call_func=log_llm_call_func,
    )
