
import numpy as np
from typing import Dict, List, Tuple, Optional, Any, Union
from dataclasses import dataclass, field

# Import spec types from skill_conditions (single source of truth)
from ours_utils.skill_conditions import (
    ConditionSpec, EffectSpec, SkillSpec, SKILL_SPECS, get_skill_spec,
    # Condition specs
    GripperOpen, GripperClosed, HoldingObject, NotHolding,
    NearTarget, AtTarget, TargetReachable, NotAtTarget,
    RequiresPrePlace, ObjectGrasped, ObjectReleased, ObjectAtTarget,
    # Effect specs
    MoveToTarget, CloseGripper, OpenGripper, GraspObject, ReleaseObject,
)


@dataclass
class ValidationResult:
    """Result of a condition validation."""
    is_valid: bool
    can_be_validated: bool
    issues: List[str] = field(default_factory=list)
    warnings: List[str] = field(default_factory=list)
    details: Dict[str, Any] = field(default_factory=dict)

def get_gripper_openness(obs) -> Tuple[float, bool]:
    if obs is None:
        return 0.0, False

    value = getattr(obs, "gripper_open", None)
    if value is None:
        return 0.0, False

    return float(value), True


def get_gripper_position(obs) -> Tuple[Optional[np.ndarray], bool]:
    if obs is None:
        return None, False

    pose = getattr(obs, "gripper_pose", None)
    if pose is None or len(pose) < 3:
        return None, False

    return np.array(pose[:3]), True


def get_gripper_quaternion(obs) -> Tuple[Optional[np.ndarray], bool]:
    if obs is None:
        return None, False

    pose = getattr(obs, "gripper_pose", None)
    if pose is None or len(pose) < 7:
        return None, False

    return np.array(pose[3:7]), True


def compute_distance(pos1: Optional[np.ndarray], pos2: Optional[np.ndarray]) -> Tuple[float, bool]:
    if pos1 is None or pos2 is None:
        return float('inf'), False

    try:
        return float(np.linalg.norm(np.array(pos1) - np.array(pos2))), True
    except Exception:
        return float('inf'), False


def check_object_grasped(obs, task=None) -> Tuple[bool, bool, str]:
    grip_open, available = get_gripper_openness(obs)

    if not available:
        return False, False, "gripper_open not available"

    is_grasped = grip_open < 0.5
    return is_grasped, True, "gripper_openness_heuristic"

@dataclass
class ConditionContext:
    """Context for evaluating conditions against projected state."""
    gripper_open: bool
    gripper_position: Optional[np.ndarray]
    held_object: Optional[str]
    target_pos: Optional[np.ndarray]
    target_obj: Optional[str]
    objects: Dict[str, Any]
    flags: Dict[str, bool] = field(default_factory=dict)

    @classmethod
    def from_projected_state(
        cls,
        projected_state: "ProjectedState",
        target_pos: Optional[np.ndarray] = None,
        target_obj: Optional[str] = None,
    ) -> "ConditionContext":
        """Create context from a ProjectedState."""
        return cls(
            gripper_open=projected_state.gripper.is_open,
            gripper_position=projected_state.gripper.position,
            held_object=projected_state.gripper.held_object,
            target_pos=target_pos,
            target_obj=target_obj,
            objects={k: v for k, v in projected_state.objects.items()},
            flags=getattr(projected_state, 'flags', {}),
        )


def validate_condition(
    condition: ConditionSpec,
    ctx: ConditionContext,
    skill_name: str = "",
) -> Tuple[bool, Optional[str], Optional[str]]:
    """
    Validate a single condition against context.

    Args:
        condition: The ConditionSpec to validate
        ctx: Context containing current state
        skill_name: Name of skill (for error messages)

    Returns:
        (success, violation_message, warning_message)
        - success: True if condition is met
        - violation_message: Non-None if hard failure
        - warning_message: Non-None if soft warning
    """
    # Branch on ConditionSpec type (not skill name)

    if isinstance(condition, GripperOpen):
        if not ctx.gripper_open:
            return False, (
                f"{skill_name}: gripper must be open but is closed "
                f"(holding: {ctx.held_object})"
            ), None
        return True, None, None

    elif isinstance(condition, GripperClosed):
        if ctx.gripper_open:
            return False, (
                f"{skill_name}: gripper must be closed but is open"
            ), None
        return True, None, None

    elif isinstance(condition, HoldingObject):
        if ctx.held_object is None:
            # Fallback: if gripper is closed, assume something is held
            # (target_obj tracking may have failed due to argument eval failure)
            if not ctx.gripper_open:
                return True, None, f"{skill_name}: held_object unknown but gripper is closed, assuming holding"
            return False, (
                f"{skill_name}: must be holding an object but gripper is empty"
            ), None
        return True, None, None

    elif isinstance(condition, NotHolding):
        if ctx.held_object is not None:
            # Fallback: if gripper is open, assume not holding
            # (held_object tracking may be stale after release)
            if ctx.gripper_open:
                return True, None, f"{skill_name}: held_object tracked as '{ctx.held_object}' but gripper is open, assuming not holding"
            return False, (
                f"{skill_name}: must not be holding anything but holding '{ctx.held_object}'"
            ), None
        return True, None, None

    elif isinstance(condition, NearTarget):
        if ctx.target_pos is None:
            # Cannot validate without target_pos - pass with warning (not invalid)
            return True, None, f"{skill_name}: target_pos not available, skipping NearTarget check"
        if ctx.gripper_position is not None:
            dist = float(np.linalg.norm(ctx.target_pos - ctx.gripper_position))
            if dist > condition.max_distance:
                return False, (
                    f"{skill_name}: gripper too far from target "
                    f"({dist:.3f}m > {condition.max_distance}m)"
                ), None
        return True, None, None

    elif isinstance(condition, AtTarget):
        if ctx.target_pos is None:
            # Cannot validate without target_pos - pass with warning (not invalid)
            return True, None, f"{skill_name}: target_pos not available, skipping AtTarget check"
        if ctx.gripper_position is not None:
            dist = float(np.linalg.norm(ctx.target_pos - ctx.gripper_position))
            if dist > condition.tolerance:
                return False, (
                    f"{skill_name}: not at target "
                    f"({dist:.3f}m > {condition.tolerance}m tolerance)"
                ), None
        return True, None, None

    elif isinstance(condition, TargetReachable):
        if ctx.target_pos is None:
            return True, None, f"{skill_name}: target_pos not provided for reachability check"
        if ctx.gripper_position is not None:
            dist = float(np.linalg.norm(ctx.target_pos - ctx.gripper_position))
            if dist > condition.max_distance:
                return True, None, (
                    f"{skill_name}: target may be far "
                    f"({dist:.3f}m > {condition.max_distance}m threshold)"
                )
        return True, None, None

    elif isinstance(condition, NotAtTarget):
        if ctx.target_pos is None:
            # Cannot validate without target_pos - pass with warning (not invalid)
            return True, None, f"{skill_name}: target_pos not available, skipping NotAtTarget check"
        if ctx.gripper_position is not None:
            dist = float(np.linalg.norm(ctx.target_pos - ctx.gripper_position))
            if dist < condition.min_distance:
                return False, (
                    f"{skill_name}: already at target "
                    f"(distance {dist:.4f}m < {condition.min_distance}m)"
                ), None
        return True, None, None

    elif isinstance(condition, RequiresPrePlace):
        if condition.enabled:
            # Check flag-based pre-place state
            if not ctx.flags.get("at_preplace", False):
                return True, None, f"{skill_name}: pre-place position not confirmed (soft check)"
        return True, None, None

    elif isinstance(condition, ObjectGrasped):
        if ctx.held_object is None:
            # Fallback: if gripper is closed, assume something is grasped
            # (target_obj tracking may have failed due to argument eval failure)
            if not ctx.gripper_open:
                return True, None, f"{skill_name}: held_object unknown but gripper is closed, assuming grasped"
            return False, (
                f"{skill_name}: object should be grasped but gripper is empty"
            ), None
        return True, None, None

    elif isinstance(condition, ObjectReleased):
        if ctx.held_object is not None:
            # Fallback: if gripper is open, object is released regardless of tracking
            if ctx.gripper_open:
                return True, None, f"{skill_name}: held_object tracked as '{ctx.held_object}' but gripper is open, assuming released"
            return False, (
                f"{skill_name}: object should be released but still holding '{ctx.held_object}'"
            ), None
        return True, None, None

    elif isinstance(condition, ObjectAtTarget):
        if ctx.target_pos is None or ctx.target_obj is None:
            return True, None, None
        if ctx.target_obj in ctx.objects:
            obj_state = ctx.objects[ctx.target_obj]
            obj_pos = getattr(obj_state, 'position', None)
            if obj_pos is not None:
                dist = float(np.linalg.norm(ctx.target_pos - obj_pos))
                if dist > condition.tolerance:
                    return True, None, (
                        f"{skill_name}: object '{ctx.target_obj}' may not be at target "
                        f"({dist:.3f}m > {condition.tolerance}m)"
                    )
        return True, None, None

    # Unknown condition type - pass through
    return True, None, f"Unknown condition type: {type(condition).__name__}"


def validate_conditions_list(
    conditions: List[ConditionSpec],
    ctx: ConditionContext,
    skill_name: str = "",
) -> Dict[str, Any]:
    """
    Validate a list of conditions.

    Returns:
        Dict with success, violations, warnings
    """
    violations = []
    warnings = []

    for cond in conditions:
        success, violation, warning = validate_condition(cond, ctx, skill_name)
        if not success and violation:
            violations.append(violation)
        if warning:
            warnings.append(warning)

    return {
        "success": len(violations) == 0,
        "violations": violations,
        "warnings": warnings,
    }


# =============================================================================
# Spec-Driven Effect Appliers
# =============================================================================

def apply_effect(
    effect: EffectSpec,
    state: "ProjectedState",
    target_pos: Optional[np.ndarray] = None,
    target_obj: Optional[str] = None,
) -> None:
    """
    Apply a single effect to projected state (mutates state).

    Args:
        effect: The EffectSpec to apply
        state: ProjectedState to mutate
        target_pos: Target position (if applicable)
        target_obj: Target object name (if applicable)
    """
    # Branch on EffectSpec type (not skill name)

    if isinstance(effect, MoveToTarget):
        if target_pos is not None:
            state.gripper.position = np.array(target_pos).copy()
            # If holding object, move it too
            if state.gripper.held_object:
                held = state.gripper.held_object
                if held in state.objects:
                    state.objects[held].position = np.array(target_pos).copy()

    elif isinstance(effect, CloseGripper):
        state.gripper.is_open = False

    elif isinstance(effect, OpenGripper):
        state.gripper.is_open = True

    elif isinstance(effect, GraspObject):
        state.gripper.is_open = False
        # Use placeholder if target_obj is unknown (argument eval failed)
        # This ensures held_object tracking works for subsequent checks
        state.gripper.held_object = target_obj if target_obj else "_unknown_grasped_object"
        if target_obj and target_obj in state.objects:
            state.objects[target_obj].is_grasped = True
            state.objects[target_obj].grasped_by = "gripper"

    elif isinstance(effect, ReleaseObject):
        released = state.gripper.held_object
        state.gripper.is_open = True
        state.gripper.held_object = None
        if released and released in state.objects:
            state.objects[released].is_grasped = False
            state.objects[released].grasped_by = None
            # Object stays at release position (gripper position)
            if target_pos is not None:
                state.objects[released].position = np.array(target_pos).copy()


def apply_effects_list(
    effects: List[EffectSpec],
    state: "ProjectedState",
    target_pos: Optional[np.ndarray] = None,
    target_obj: Optional[str] = None,
) -> None:
    """Apply a list of effects to projected state."""
    for effect in effects:
        apply_effect(effect, state, target_pos, target_obj)


# =============================================================================
# Spec-Driven Validation Engine
# =============================================================================

def validate_skill_preconditions_spec(
    skill_name: str,
    projected_state: "ProjectedState",
    target_pos: Optional[np.ndarray] = None,
    target_obj: Optional[str] = None,
) -> Dict[str, Any]:
    """
    Validate preconditions using SkillSpec (spec-driven, no skill if/elif).

    Args:
        skill_name: Name of the skill
        projected_state: Current projected state
        target_pos: Target position
        target_obj: Target object name

    Returns:
        Dict with success, violations, warnings
    """
    spec = get_skill_spec(skill_name)
    if spec is None:
        return {
            "success": True,
            "violations": [],
            "warnings": [f"Unknown skill: {skill_name}, no preconditions defined"],
        }

    ctx = ConditionContext.from_projected_state(projected_state, target_pos, target_obj)
    return validate_conditions_list(spec.preconditions, ctx, skill_name)


def validate_skill_postconditions_spec(
    skill_name: str,
    projected_state: "ProjectedState",
    target_pos: Optional[np.ndarray] = None,
    target_obj: Optional[str] = None,
) -> Dict[str, Any]:
    """
    Validate postconditions using SkillSpec (spec-driven, no skill if/elif).
    """
    spec = get_skill_spec(skill_name)
    if spec is None:
        return {
            "success": True,
            "violations": [],
            "warnings": [f"Unknown skill: {skill_name}, no postconditions defined"],
        }

    ctx = ConditionContext.from_projected_state(projected_state, target_pos, target_obj)
    return validate_conditions_list(spec.postconditions, ctx, skill_name)


def apply_skill_effects_spec(
    skill_name: str,
    state: "ProjectedState",
    target_pos: Optional[np.ndarray] = None,
    target_obj: Optional[str] = None,
) -> None:
    """
    Apply skill effects using SkillSpec (spec-driven, no skill if/elif).
    """
    spec = get_skill_spec(skill_name)
    if spec is None:
        return

    apply_effects_list(spec.effects, state, target_pos, target_obj)


def validate_skill_full_spec(
    skill_name: str,
    projected_state: "ProjectedState",
    target_pos: Optional[np.ndarray] = None,
    target_obj: Optional[str] = None,
) -> Dict[str, Any]:
    """
    Full validation sequence using SkillSpec:
    1. Check preconditions
    2. Apply effects (assume success)
    3. Check postconditions

    Returns combined result.
    """
    spec = get_skill_spec(skill_name)
    if spec is None:
        return {
            "success": True,
            "precondition_success": True,
            "postcondition_success": True,
            "violations": [],
            "precondition_violations": [],
            "postcondition_violations": [],
            "warnings": [f"Unknown skill: {skill_name}"],
            "transition": None,
        }

    # Step 1: Check preconditions
    ctx_pre = ConditionContext.from_projected_state(projected_state, target_pos, target_obj)
    pre_result = validate_conditions_list(spec.preconditions, ctx_pre, skill_name)

    # Step 2: Apply effects (mutates projected_state)
    apply_effects_list(spec.effects, projected_state, target_pos, target_obj)

    # Step 3: Check postconditions
    ctx_post = ConditionContext.from_projected_state(projected_state, target_pos, target_obj)
    post_result = validate_conditions_list(spec.postconditions, ctx_post, skill_name)

    # Combine results
    all_violations = pre_result["violations"] + post_result["violations"]
    all_warnings = pre_result["warnings"] + post_result["warnings"]

    return {
        "success": len(all_violations) == 0,
        "precondition_success": pre_result["success"],
        "postcondition_success": post_result["success"],
        "violations": all_violations,
        "precondition_violations": pre_result["violations"],
        "postcondition_violations": post_result["violations"],
        "warnings": all_warnings,
        "transition": spec.transition,
        "projected_gripper_pos": projected_state.gripper.position.tolist() if projected_state.gripper.position is not None else None,
        "projected_held_object": projected_state.gripper.held_object,
        "projected_gripper_open": projected_state.gripper.is_open,
    }


# =============================================================================
# Condition validation checking (legacy, kept for backward compatibility)
# =============================================================================

def validate_precondition_checkability(skill_name: str) -> ValidationResult:
    name = skill_name.lower()
    result = ValidationResult(is_valid=True, can_be_validated=True)

    if name == "pick":
        # Preconditions: target_pos required, gripper open, gripper near target
        result.details["required_inputs"] = ["target_pos"]
        result.details["observable_state"] = ["gripper_open", "gripper_pose"]
        result.details["conditions"] = [
            "target_pos is not None",
            "gripper_open >= 0.5 (gripper sufficiently open)",
            "distance(gripper, target) <= 0.30m"
        ]
        # All conditions are checkable
        result.can_be_validated = True

    elif name == "place":
        result.details["required_inputs"] = ["target_pos", "held_obj"]
        result.details["observable_state"] = ["gripper_open", "gripper_pose"]
        result.details["conditions"] = [
            "held_obj is not None",
            "target_pos is not None",
            "gripper should be closed (holding object)",
            "distance(gripper, target) <= 0.35m"
        ]

        # Issue: held_obj cannot be reliably validated from observation alone
        result.warnings.append(
            "held_obj validation relies on caller providing correct value; "
            "cannot verify actual grasp state from observation alone"
        )

        # Critical issue in original code: condition logic is inverted
        result.issues.append(
            "CRITICAL: Original condition 'grip_open < 0.2 -> fail' is WRONG. "
            "Place requires gripper to be CLOSED (grip_open < 0.5) to hold object. "
            "Current logic fails when gripper IS closed, which is the correct state for place."
        )
        result.is_valid = False

    elif name == "move":
        result.details["required_inputs"] = ["target_pos"]
        result.details["observable_state"] = ["gripper_pose"]
        result.details["conditions"] = [
            "target_pos is not None",
            "distance(gripper, target) > 0.01m (not already at target)"
        ]
        result.can_be_validated = True

    elif name == "push":
        result.details["required_inputs"] = ["target_pos"]
        result.details["observable_state"] = ["gripper_open", "gripper_pose"]
        result.details["conditions"] = [
            "target_pos is not None",
            "gripper_open >= 0.5 (gripper open for push)",
            "distance(gripper, target) <= 0.35m"
        ]
        result.can_be_validated = True

    elif name in ["ur5_move_to", "ur5_grasp_at", "ur5_release_at", "ur5_align_gripper"]:
        result.warnings.append(f"No preconditions defined for UR5 skill: {name}")
        result.details["conditions"] = []
        result.can_be_validated = False

    elif name == "align_two_axes":
        result.details["required_inputs"] = ["local_axes", "world_axes", "axis_dirs"]
        result.details["observable_state"] = ["gripper_pose (quaternion)"]
        result.details["conditions"] = ["Parameters must be valid axis specifications"]
        result.can_be_validated = True

    else:
        result.warnings.append(f"Unknown skill: {name}")
        result.can_be_validated = False

    return result


def validate_postcondition_checkability(skill_name: str) -> ValidationResult:
    name = skill_name.lower()
    result = ValidationResult(is_valid=True, can_be_validated=True)

    if name == "pick":
        result.details["expected_state"] = [
            "gripper at target position (within 0.04m)",
            "gripper closed (grip_open < 0.5)"
        ]
        result.details["transition"] = "picked"
        result.can_be_validated = True

    elif name == "place":
        result.details["expected_state"] = [
            "gripper at target position (within 0.05m)",
            "gripper open (grip_open >= 0.5)"
        ]
        result.details["transition"] = "released"
        result.can_be_validated = True

    elif name == "move":
        result.details["expected_state"] = [
            "gripper at target position (within 0.03m)"
        ]
        result.can_be_validated = True

    elif name == "push":
        result.details["expected_state"] = [
            "gripper at target position (within 0.06m)",
            "gripper open (grip_open >= 0.5)"
        ]
        result.can_be_validated = True

    elif name in ["ur5_move_to", "ur5_grasp_at", "ur5_release_at", "ur5_align_gripper"]:
        result.warnings.append(f"No postconditions defined for UR5 skill: {name}")
        result.can_be_validated = False

    else:
        result.warnings.append(f"Unknown skill: {name}")
        result.can_be_validated = False

    return result


# =============================================================================
# Enhanced condition checking (with fixes for original bugs)
# =============================================================================

def check_preconditions_enhanced(
    skill_name: str,
    obs,
    target_pos: Optional[np.ndarray] = None,
    held_obj: Any = None,
    strict: bool = False
) -> Dict[str, Any]:
    """
    Enhanced precondition checking with bug fixes and better validation.

    Args:
        skill_name: Name of the skill
        obs: Current observation
        target_pos: Target position (if applicable)
        held_obj: Reference to held object (if applicable)
        strict: If True, return failure for any unvalidatable conditions

    Returns:
        Dict with:
        - success: bool
        - reasons: list of failure reasons
        - warnings: list of warnings
        - details: dict with measured values
    """
    name = skill_name.lower()
    success = True
    reasons = []
    warnings = []
    details = {}

    # Get observable state
    grip_open, grip_available = get_gripper_openness(obs)
    gripper_pos, pos_available = get_gripper_position(obs)

    details["gripper_open"] = grip_open if grip_available else None
    details["gripper_pos"] = gripper_pos.tolist() if pos_available else None
    details["target_pos"] = target_pos.tolist() if target_pos is not None else None

    # Compute distance if possible
    if pos_available and target_pos is not None:
        dist, _ = compute_distance(gripper_pos, target_pos)
        details["distance"] = dist
    else:
        dist = float('inf')
        details["distance"] = None

    if name == "pick":
        # target_pos required
        if target_pos is None:
            success = False
            reasons.append("missing target_pos")

        # Gripper must be open to pick
        if grip_available:
            if grip_open < 0.5:
                success = False
                reasons.append(f"gripper not open enough for pick (gripper_open={grip_open:.3f}, need >= 0.5)")
        elif strict:
            success = False
            reasons.append("cannot verify gripper state")
        else:
            warnings.append("gripper state not available, assuming open")

        # Must be close enough to target
        if target_pos is not None and pos_available:
            if dist > 0.30:
                success = False
                reasons.append(f"gripper too far from target ({dist:.3f}m > 0.30m)")

    elif name == "place":
        # target_pos required
        if target_pos is None:
            success = False
            reasons.append("missing target_pos")

        # held_obj check (limited validation possible)
        if held_obj is None:
            success = False
            reasons.append("no held object specified for place")

        # FIXED: Gripper must be CLOSED when starting place (holding object)
        if grip_available:
            if grip_open >= 0.5:
                success = False
                reasons.append(f"gripper is open but should be closed (holding object) for place (gripper_open={grip_open:.3f})")
        elif strict:
            success = False
            reasons.append("cannot verify gripper state")

        # Distance check
        if target_pos is not None and pos_available:
            if dist > 0.35:
                success = False
                reasons.append(f"gripper too far from place target ({dist:.3f}m > 0.35m)")

    elif name == "move":
        if target_pos is None:
            success = False
            reasons.append("missing target_pos")

        # Already at target check
        if target_pos is not None and pos_available:
            if dist < 0.01:
                success = False
                reasons.append(f"already at target position (distance={dist:.4f}m)")

    elif name == "push":
        if target_pos is None:
            success = False
            reasons.append("missing target_pos")

        # Gripper should be open for push
        if grip_available:
            if grip_open < 0.5:
                success = False
                reasons.append(f"gripper should be open for push (gripper_open={grip_open:.3f})")

        # Distance check
        if target_pos is not None and pos_available:
            if dist > 0.35:
                success = False
                reasons.append(f"gripper too far from push target ({dist:.3f}m > 0.35m)")

    # UR5 skills - add basic checks
    elif name == "ur5_grasp_at":
        if target_pos is None:
            success = False
            reasons.append("missing grasp_pos")

        if grip_available and grip_open < 0.5:
            warnings.append("gripper appears closed before grasp")

    elif name == "ur5_release_at":
        if target_pos is None:
            success = False
            reasons.append("missing place_pos")

        if grip_available and grip_open >= 0.5:
            warnings.append("gripper appears open before release (should be holding)")

    elif name == "ur5_move_to":
        if target_pos is None:
            success = False
            reasons.append("missing target_pos")

    elif name == "ur5_align_gripper":
        # No position-based preconditions
        pass

    elif name == "align_two_axes":
        # Orientation-only skill, minimal preconditions
        pass

    else:
        warnings.append(f"Unknown skill: {name}, no preconditions checked")

    return {
        "success": success,
        "reasons": reasons,
        "warnings": warnings,
        "details": details
    }


def check_postconditions_enhanced(
    skill_name: str,
    before_obs,
    after_obs,
    target_pos: Optional[np.ndarray] = None,
    held_obj: Any = None
) -> Dict[str, Any]:
    """
    Enhanced postcondition checking.

    Args:
        skill_name: Name of the skill
        before_obs: Observation before skill execution
        after_obs: Observation after skill execution
        target_pos: Target position used in skill
        held_obj: Object that was being manipulated

    Returns:
        Dict with success, reasons, warnings, transition, details
    """
    name = skill_name.lower()
    success = True
    reasons = []
    warnings = []
    transition = None
    details = {}

    # Get before/after states
    before_grip, _ = get_gripper_openness(before_obs)
    after_grip, after_grip_available = get_gripper_openness(after_obs)
    after_pos, after_pos_available = get_gripper_position(after_obs)

    details["before_gripper_open"] = before_grip
    details["after_gripper_open"] = after_grip if after_grip_available else None
    details["after_pos"] = after_pos.tolist() if after_pos_available else None

    # Compute distance to target
    if after_pos_available and target_pos is not None:
        dist, _ = compute_distance(after_pos, target_pos)
        details["distance"] = dist
    else:
        dist = float('inf')
        details["distance"] = None

    if name == "pick":
        # Should be at target
        if target_pos is not None and dist > 0.04:
            success = False
            reasons.append(f"gripper not at target after pick ({dist:.3f}m > 0.04m)")

        # Gripper should be closed
        if after_grip_available:
            if after_grip >= 0.5:
                success = False
                reasons.append(f"gripper still open after pick (gripper_open={after_grip:.3f})")
            else:
                transition = "picked"

    elif name == "place":
        if target_pos is None:
            success = False
            reasons.append("missing target_pos for postcondition check")

        # Should be at target
        if target_pos is not None and dist > 0.05:
            success = False
            reasons.append(f"place end position too far from target ({dist:.3f}m > 0.05m)")

        # Gripper should be open
        if after_grip_available:
            if after_grip < 0.5:
                success = False
                reasons.append(f"gripper not opened after place (gripper_open={after_grip:.3f})")
            else:
                transition = "released"

    elif name == "move":
        if target_pos is None:
            success = False
            reasons.append("missing target_pos")

        if target_pos is not None and dist > 0.03:
            success = False
            reasons.append(f"move end position too far from target ({dist:.3f}m > 0.03m)")

    elif name == "push":
        if target_pos is None:
            success = False
            reasons.append("missing target_pos")

        if target_pos is not None and dist > 0.06:
            success = False
            reasons.append(f"push end position too far from target ({dist:.3f}m > 0.06m)")

        if after_grip_available and after_grip < 0.5:
            success = False
            reasons.append("gripper should remain open after push")

    # UR5 skills
    elif name == "ur5_grasp_at":
        if after_grip_available and after_grip >= 0.5:
            success = False
            reasons.append("gripper should be closed after grasp")
        else:
            transition = "picked"

    elif name == "ur5_release_at":
        if after_grip_available and after_grip < 0.5:
            success = False
            reasons.append("gripper should be open after release")
        else:
            transition = "released"

    elif name == "ur5_move_to":
        if target_pos is not None and dist > 0.03:
            success = False
            reasons.append(f"move end position too far from target ({dist:.3f}m)")

    return {
        "success": success,
        "reasons": reasons,
        "warnings": warnings,
        "transition": transition,
        "details": details
    }


# =============================================================================
# Validation report generation
# =============================================================================

def generate_condition_report(skills: Optional[List[str]] = None) -> str:
    if skills is None:
        skills = [
            "pick", "place", "move", "push", "align_two_axes",
            "ur5_move_to", "ur5_grasp_at", "ur5_release_at", "ur5_align_gripper"
        ]

    lines = []
    lines.append("=" * 70)
    lines.append("SKILL CONDITION VALIDATION REPORT")
    lines.append("=" * 70)

    critical_issues = []
    all_warnings = []

    for skill in skills:
        lines.append(f"\n

        # Preconditions
        pre_result = validate_precondition_checkability(skill)
        lines.append("\nPreconditions:")
        lines.append(f"  Can be validated: {pre_result.can_be_validated}")
        lines.append(f"  Logic is valid: {pre_result.is_valid}")

        if pre_result.details.get("conditions"):
            lines.append("  Conditions:")
            for cond in pre_result.details["conditions"]:
                lines.append(f"    - {cond}")

        if pre_result.issues:
            lines.append("  ISSUES:")
            for issue in pre_result.issues:
                lines.append(f"    [!] {issue}")
                critical_issues.append((skill, "precondition", issue))

        if pre_result.warnings:
            lines.append("  Warnings:")
            for warn in pre_result.warnings:
                lines.append(f"    [~] {warn}")
                all_warnings.append((skill, "precondition", warn))

        # Postconditions
        post_result = validate_postcondition_checkability(skill)
        lines.append("\nPostconditions:")
        lines.append(f"  Can be validated: {post_result.can_be_validated}")
        lines.append(f"  Logic is valid: {post_result.is_valid}")

        if post_result.details.get("expected_state"):
            lines.append("  Expected state:")
            for state in post_result.details["expected_state"]:
                lines.append(f"    - {state}")

        if post_result.issues:
            lines.append("  ISSUES:")
            for issue in post_result.issues:
                lines.append(f"    [!] {issue}")
                critical_issues.append((skill, "postcondition", issue))

        if post_result.warnings:
            lines.append("  Warnings:")
            for warn in post_result.warnings:
                lines.append(f"    [~] {warn}")
                all_warnings.append((skill, "postcondition", warn))

    # Summary
    lines.append("\n" + "=" * 70)
    lines.append("SUMMARY")
    lines.append("=" * 70)

    if critical_issues:
        lines.append(f"\nCRITICAL ISSUES ({len(critical_issues)}):")
        for skill, cond_type, issue in critical_issues:
            lines.append(f"  [{skill}:{cond_type}] {issue}")
    else:
        lines.append("\nNo critical issues found.")

    if all_warnings:
        lines.append(f"\nWARNINGS ({len(all_warnings)}):")
        for skill, cond_type, warn in all_warnings:
            lines.append(f"  [{skill}:{cond_type}] {warn}")

    return "\n".join(lines)


# =============================================================================
# Runtime validation wrapper
# =============================================================================

class ConditionValidator:
    """
    Wrapper class for validating conditions during skill execution.

    Example usage:
        validator = ConditionValidator()

        # Before executing pick
        pre_result = validator.check_precondition("pick", obs, target_pos=target)
        if not pre_result["success"]:
            print(f"Precondition failed: {pre_result['reasons']}")

        # Execute skill...

        # After executing pick
        post_result = validator.check_postcondition("pick", before_obs, after_obs, target_pos=target)
        if not post_result["success"]:
            print(f"Postcondition failed: {post_result['reasons']}")
    """

    def __init__(self, strict: bool = False, log_warnings: bool = True):
        self.strict = strict
        self.log_warnings = log_warnings
        self.history: List[Dict] = []

    def check_precondition(
        self,
        skill_name: str,
        obs,
        target_pos: Optional[np.ndarray] = None,
        held_obj: Any = None,
        **kwargs
    ) -> Dict[str, Any]:
        """Check preconditions for a skill."""
        result = check_preconditions_enhanced(
            skill_name, obs, target_pos, held_obj, strict=self.strict
        )

        if self.log_warnings and result["warnings"]:
            for warn in result["warnings"]:
                print(f"[WARN] {skill_name} precondition: {warn}")

        self.history.append({
            "type": "precondition",
            "skill": skill_name,
            "result": result
        })

        return result

    def check_postcondition(
        self,
        skill_name: str,
        before_obs,
        after_obs,
        target_pos: Optional[np.ndarray] = None,
        held_obj: Any = None,
        **kwargs
    ) -> Dict[str, Any]:
        """Check postconditions for a skill."""
        result = check_postconditions_enhanced(
            skill_name, before_obs, after_obs, target_pos, held_obj
        )

        if self.log_warnings and result["warnings"]:
            for warn in result["warnings"]:
                print(f"[WARN] {skill_name} postcondition: {warn}")

        self.history.append({
            "type": "postcondition",
            "skill": skill_name,
            "result": result
        })

        return result

    def get_history(self) -> List[Dict]:
        return self.history

    def clear_history(self):
        self.history = []

    def get_statistics(self) -> Dict[str, Any]:
        if not self.history:
            return {"total": 0}

        total = len(self.history)
        passed = sum(1 for h in self.history if h["result"]["success"])
        failed = total - passed

        by_skill = {}
        for h in self.history:
            skill = h["skill"]
            if skill not in by_skill:
                by_skill[skill] = {"passed": 0, "failed": 0}
            if h["result"]["success"]:
                by_skill[skill]["passed"] += 1
            else:
                by_skill[skill]["failed"] += 1

        return {
            "total": total,
            "passed": passed,
            "failed": failed,
            "pass_rate": passed / total if total > 0 else 0,
            "by_skill": by_skill
        }


# =============================================================================
# Standalone test function
# =============================================================================

def run_validation_tests():
    print(generate_condition_report())

    print("\n" + "=" * 70)
    print("COMPARISON: ORIGINAL vs ENHANCED CONDITIONS")
    print("=" * 70)

    # Demonstrate the place precondition bug
    print("\n
    print("\nScenario: Robot is holding an object (gripper closed, grip_open = 0.1)")
    print("          About to place at target position")

    # Mock observation class
    class MockObs:
        def __init__(self, gripper_open, gripper_pose):
            self.gripper_open = gripper_open
            self.gripper_pose = gripper_pose

    # Create mock observation with closed gripper (holding object)
    mock_obs = MockObs(
        gripper_open=0.1,  # Closed gripper, holding object
        gripper_pose=[0.5, 0.0, 0.3, 0, 0, 0, 1]  # Position near target
    )
    target = np.array([0.5, 0.0, 0.25])

    # Import original check
    try:
        from skill_conditions import check_preconditions as original_check
        original_result = original_check("place", mock_obs, target_pos=target, held_obj="some_object")
        print(f"\nOriginal check_preconditions result:")
        print(f"  Success: {original_result['success']}")
        print(f"  Reasons: {original_result['reasons']}")
    except ImportError:
        print("\nCannot import original check_preconditions for comparison")

    enhanced_result = check_preconditions_enhanced(
        "place", mock_obs, target_pos=target, held_obj="some_object"
    )
    print(f"\nEnhanced check_preconditions result:")
    print(f"  Success: {enhanced_result['success']}")
    print(f"  Reasons: {enhanced_result['reasons']}")
    print(f"  Details: {enhanced_result['details']}")

    print("\nExpected: Both should succeed since gripper IS closed (holding object)")
    print("Original has bug: fails when gripper is closed (grip_open < 0.2)")


# =============================================================================
# Symbolic Effect Table for Projected State Validation
# =============================================================================

@dataclass
class ObjectState:
    """Symbolic state of an object in the scene."""
    name: str
    position: Optional[np.ndarray] = None
    quaternion: Optional[np.ndarray] = None
    is_grasped: bool = False
    grasped_by: Optional[str] = None  # "gripper" or None

    def copy(self) -> "ObjectState":
        return ObjectState(
            name=self.name,
            position=self.position.copy() if self.position is not None else None,
            quaternion=self.quaternion.copy() if self.quaternion is not None else None,
            is_grasped=self.is_grasped,
            grasped_by=self.grasped_by,
        )


@dataclass
class GripperState:
    """Symbolic state of the robot gripper."""
    position: Optional[np.ndarray] = None
    quaternion: Optional[np.ndarray] = None
    is_open: bool = True
    held_object: Optional[str] = None

    def copy(self) -> "GripperState":
        return GripperState(
            position=self.position.copy() if self.position is not None else None,
            quaternion=self.quaternion.copy() if self.quaternion is not None else None,
            is_open=self.is_open,
            held_object=self.held_object,
        )


@dataclass
class ProjectedState:
    """
    Projected world state for symbolic validation.
    Composed of confirmed effects (from observations) + predicted effects (symbolic).
    """
    gripper: GripperState
    objects: Dict[str, ObjectState] = field(default_factory=dict)
    last_confirmed_index: int = -1  # Index of last confirmed (executed) statement

    def copy(self) -> "ProjectedState":
        return ProjectedState(
            gripper=self.gripper.copy(),
            objects={k: v.copy() for k, v in self.objects.items()},
            last_confirmed_index=self.last_confirmed_index,
        )


class SymbolicEffectTable:
    """
    Symbolic effect definitions for primitive skills.
    Used to predict state changes without simulation.

    DEPRECATED: This class is kept for backward compatibility.
    New code should use SKILL_SPECS from skill_conditions.py instead.
    The spec-driven validators (validate_skill_*_spec) use SkillSpec directly.
    """

    # Skill categories
    GRASP_SKILLS = {"pick", "ur5_grasp_at"}
    RELEASE_SKILLS = {"place", "ur5_release_at"}
    MOVE_SKILLS = {"move", "ur5_move_to"}
    PUSH_SKILLS = {"push"}
    ALIGN_SKILLS = {"align_two_axes", "ur5_align_gripper"}
    GRIPPER_SKILLS = {"close_ur5_ee", "open_ur5_ee", "open_gripper", "close_gripper"}

    @staticmethod
    def get_skill_effects(skill_name: str) -> Dict[str, Any]:
        name = skill_name.lower()

        if name in SymbolicEffectTable.GRASP_SKILLS:
            return {
                "type": "grasp",
                "gripper_state_change": {"is_open": False, "held_object": "target"},
                "position_change": "move_to_target",
                "preconditions": {
                    "gripper_open": True,
                    "target_reachable": True,
                    "not_holding": True,
                },
                "postconditions": {
                    "gripper_closed": True,
                    "object_grasped": True,
                    "at_target": True,
                },
            }

        elif name in SymbolicEffectTable.RELEASE_SKILLS:
            return {
                "type": "release",
                "gripper_state_change": {"is_open": True, "held_object": None},
                "position_change": "move_to_target",
                "preconditions": {
                    "holding_object": True,
                    "target_reachable": True,
                },
                "postconditions": {
                    "gripper_open": True,
                    "object_released": True,
                    "object_at_target": True,
                    "at_target": True,
                },
            }

        elif name in SymbolicEffectTable.MOVE_SKILLS:
            return {
                "type": "move",
                "gripper_state_change": None,  # No change
                "position_change": "move_to_target",
                "preconditions": {
                    "target_reachable": True,
                },
                "postconditions": {
                    "at_target": True,
                },
            }

        elif name in SymbolicEffectTable.PUSH_SKILLS:
            return {
                "type": "push",
                "gripper_state_change": None,
                "position_change": "move_to_target",
                "preconditions": {
                    "gripper_open": True,
                    "target_reachable": True,
                },
                "postconditions": {
                    "at_target": True,
                    "object_pushed": True,
                },
            }

        elif name in SymbolicEffectTable.ALIGN_SKILLS:
            return {
                "type": "align",
                "gripper_state_change": None,
                "position_change": None,  # Only orientation changes
                "preconditions": {},
                "postconditions": {
                    "orientation_aligned": True,
                },
            }

        elif name in {"close_ur5_ee", "close_gripper"}:
            return {
                "type": "gripper_close",
                "gripper_state_change": {"is_open": False},
                "position_change": None,
                "preconditions": {
                    "gripper_open": True,
                },
                "postconditions": {
                    "gripper_closed": True,
                },
            }

        elif name in {"open_ur5_ee", "open_gripper"}:
            return {
                "type": "gripper_open",
                "gripper_state_change": {"is_open": True},
                "position_change": None,
                "preconditions": {
                    "gripper_closed": True,
                },
                "postconditions": {
                    "gripper_open": True,
                },
            }

        return {"type": "unknown", "preconditions": {}, "postconditions": {}}


class ProjectedStateTracker:
    """
    Tracks and projects world state for validation of future statements.

    Usage:
        tracker = ProjectedStateTracker()
        tracker.initialize_from_observation(obs, scene_objects)

        # After each executed statement
        tracker.confirm_state(obs, statement_index)

        # To validate future statements
        for stmt in future_statements:
            projected = tracker.project_effect(skill_name, target_pos, target_obj)
            validation = tracker.validate_preconditions(skill_name, projected)
    """

    def __init__(self):
        self.confirmed_state: Optional[ProjectedState] = None
        self.projected_state: Optional[ProjectedState] = None
        self.execution_history: List[Dict[str, Any]] = []

    def initialize_from_observation(
        self,
        obs,
        scene_objects: List[Dict[str, Any]],
    ) -> ProjectedState:
        """
        Initialize state from actual observation and scene info.

        Args:
            obs: RLBench observation
            scene_objects: List of object dicts with name, position, quaternion
        """
        # Initialize gripper state
        gripper_pos, pos_avail = get_gripper_position(obs)
        gripper_quat, quat_avail = get_gripper_quaternion(obs)
        grip_open, grip_avail = get_gripper_openness(obs)

        gripper = GripperState(
            position=gripper_pos if pos_avail else np.zeros(3),
            quaternion=gripper_quat if quat_avail else np.array([0, 0, 0, 1]),
            is_open=grip_open >= 0.5 if grip_avail else True,
            held_object=None,
        )

        # If gripper is closed, try to infer held object
        if not gripper.is_open:
            # Heuristic: find closest object to gripper
            min_dist = float('inf')
            closest_obj = None
            for obj_info in scene_objects:
                if obj_info.get("position") is not None:
                    obj_pos = np.array(obj_info["position"])
                    dist = np.linalg.norm(gripper.position - obj_pos)
                    if dist < min_dist and dist < 0.1:  # Within 10cm
                        min_dist = dist
                        closest_obj = obj_info["name"]
            gripper.held_object = closest_obj

        # Initialize object states
        objects = {}
        for obj_info in scene_objects:
            obj_state = ObjectState(
                name=obj_info["name"],
                position=np.array(obj_info["position"]) if obj_info.get("position") else None,
                quaternion=np.array(obj_info["quaternion"]) if obj_info.get("quaternion") else None,
                is_grasped=obj_info["name"] == gripper.held_object,
                grasped_by="gripper" if obj_info["name"] == gripper.held_object else None,
            )
            objects[obj_info["name"]] = obj_state

        self.confirmed_state = ProjectedState(
            gripper=gripper,
            objects=objects,
            last_confirmed_index=-1,
        )
        self.projected_state = self.confirmed_state.copy()

        return self.confirmed_state

    def confirm_state(
        self,
        obs,
        statement_index: int,
        skill_name: Optional[str] = None,
        target_obj: Optional[str] = None,
    ) -> None:
        """
        Confirm state after a statement has been executed.
        Updates confirmed_state from actual observation.
        """
        if self.confirmed_state is None:
            return

        # Update gripper state from observation
        gripper_pos, pos_avail = get_gripper_position(obs)
        gripper_quat, quat_avail = get_gripper_quaternion(obs)
        grip_open, grip_avail = get_gripper_openness(obs)

        if pos_avail:
            self.confirmed_state.gripper.position = gripper_pos
        if quat_avail:
            self.confirmed_state.gripper.quaternion = gripper_quat
        if grip_avail:
            self.confirmed_state.gripper.is_open = grip_open >= 0.5

        # Update held object based on skill effect
        if skill_name:
            name = skill_name.lower()
            if name in SymbolicEffectTable.GRASP_SKILLS:
                self.confirmed_state.gripper.held_object = target_obj
                self.confirmed_state.gripper.is_open = False
                if target_obj and target_obj in self.confirmed_state.objects:
                    self.confirmed_state.objects[target_obj].is_grasped = True
                    self.confirmed_state.objects[target_obj].grasped_by = "gripper"
            elif name in SymbolicEffectTable.RELEASE_SKILLS:
                released_obj = self.confirmed_state.gripper.held_object
                self.confirmed_state.gripper.held_object = None
                self.confirmed_state.gripper.is_open = True
                if released_obj and released_obj in self.confirmed_state.objects:
                    self.confirmed_state.objects[released_obj].is_grasped = False
                    self.confirmed_state.objects[released_obj].grasped_by = None
                    # Update object position to release position
                    if pos_avail:
                        self.confirmed_state.objects[released_obj].position = gripper_pos.copy()

        # Update held object position if holding something
        if self.confirmed_state.gripper.held_object and pos_avail:
            held = self.confirmed_state.gripper.held_object
            if held in self.confirmed_state.objects:
                self.confirmed_state.objects[held].position = gripper_pos.copy()

        self.confirmed_state.last_confirmed_index = statement_index

        # Record in history
        self.execution_history.append({
            "index": statement_index,
            "skill_name": skill_name,
            "gripper_pos": gripper_pos.tolist() if pos_avail else None,
            "gripper_open": grip_open if grip_avail else None,
            "held_object": self.confirmed_state.gripper.held_object,
        })

        # Reset projected state to confirmed state
        self.projected_state = self.confirmed_state.copy()

    def project_effect(
        self,
        skill_name: str,
        target_pos: Optional[np.ndarray] = None,
        target_obj: Optional[str] = None,
    ) -> ProjectedState:
        """
        Project the effect of executing a skill on current projected state.
        Returns new projected state WITHOUT modifying current state.

        Delegates to spec-driven effect applier (apply_skill_effects_spec).
        """
        if self.projected_state is None:
            raise ValueError("State not initialized")

        new_state = self.projected_state.copy()
        # Delegate to spec-driven effect applier
        apply_skill_effects_spec(skill_name, new_state, target_pos, target_obj)
        return new_state

    def apply_projection(
        self,
        skill_name: str,
        target_pos: Optional[np.ndarray] = None,
        target_obj: Optional[str] = None,
    ) -> None:
        """
        Apply skill effect to projected state (mutates self.projected_state).
        Use this when iterating through future statements sequentially.
        """
        self.projected_state = self.project_effect(skill_name, target_pos, target_obj)

    def reset_projection(self) -> None:
        if self.confirmed_state:
            self.projected_state = self.confirmed_state.copy()

    def validate_preconditions_on_projected(
        self,
        skill_name: str,
        target_pos: Optional[np.ndarray] = None,
        target_obj: Optional[str] = None,
        distance_threshold: float = 0.5,
    ) -> Dict[str, Any]:
        """
        Validate preconditions for a skill against projected state.

        Delegates to spec-driven validator (validate_skill_preconditions_spec).

        Returns:
            Dict with:
            - success: bool
            - violations: list of precondition violations
            - warnings: list of potential issues
        """
        if self.projected_state is None:
            return {"success": False, "violations": ["State not initialized"], "warnings": []}

        # Delegate to spec-driven validator
        result = validate_skill_preconditions_spec(
            skill_name, self.projected_state, target_pos, target_obj
        )

        # Add projected state info to result
        gripper = self.projected_state.gripper
        result["projected_gripper_pos"] = gripper.position.tolist() if gripper.position is not None else None
        result["projected_held_object"] = gripper.held_object
        result["projected_gripper_open"] = gripper.is_open

        return result

    def validate_postconditions_on_projected(
        self,
        skill_name: str,
        target_pos: Optional[np.ndarray] = None,
        target_obj: Optional[str] = None,
        position_threshold: float = 0.1,
    ) -> Dict[str, Any]:
        """
        Validate postconditions for a skill against the CURRENT projected state.

        This should be called AFTER apply_projection() to check if the
        projected state satisfies the expected postconditions.

        Delegates to spec-driven validator (validate_skill_postconditions_spec).

        Returns:
            Dict with:
            - success: bool
            - violations: list of postcondition violations
            - warnings: list of potential issues
        """
        if self.projected_state is None:
            return {"success": False, "violations": ["State not initialized"], "warnings": []}

        # Delegate to spec-driven validator
        result = validate_skill_postconditions_spec(
            skill_name, self.projected_state, target_pos, target_obj
        )

        # Add projected state info to result
        gripper = self.projected_state.gripper
        result["projected_gripper_pos"] = gripper.position.tolist() if gripper.position is not None else None
        result["projected_held_object"] = gripper.held_object
        result["projected_gripper_open"] = gripper.is_open

        return result

    def validate_statement_full(
        self,
        skill_name: str,
        target_pos: Optional[np.ndarray] = None,
        target_obj: Optional[str] = None,
    ) -> Dict[str, Any]:
        """
        Full validation sequence for a statement:
        1. Check preconditions on current projected state
        2. Apply symbolic effect
        3. Check postconditions on updated projected state

        Delegates to spec-driven validator (validate_skill_full_spec).

        Returns combined validation result.
        """
        if self.projected_state is None:
            return {
                "success": False,
                "precondition_success": False,
                "postcondition_success": False,
                "violations": ["State not initialized"],
                "precondition_violations": ["State not initialized"],
                "postcondition_violations": [],
                "warnings": [],
            }

        # Delegate to spec-driven full validator (mutates projected_state)
        return validate_skill_full_spec(
            skill_name, self.projected_state, target_pos, target_obj
        )


# =============================================================================
# Batch Invalid Statement Info
# =============================================================================

@dataclass
class InvalidStatementInfo:
    """Information about an invalid future statement."""
    line_number: int
    statement_code: str
    skill_name: str
    violations: List[str]
    warnings: List[str]
    projected_state_summary: Dict[str, Any]


def create_invalid_statements_payload(
    invalid_statements: List[InvalidStatementInfo]
) -> Dict[str, Any]:
    """
    Create JSON payload for batch invalid statement repair request.

    Args:
        invalid_statements: List of InvalidStatementInfo

    Returns:
        Dict ready to be sent to remote_llm
    """
    statements_data = []
    for stmt_info in invalid_statements:
        statements_data.append({
            "line_number": stmt_info.line_number,
            "statement_code": stmt_info.statement_code,
            "skill_name": stmt_info.skill_name,
            "violations": stmt_info.violations,
            "warnings": stmt_info.warnings,
            "projected_state": stmt_info.projected_state_summary,
        })

    return {
        "invalid_statements": statements_data,
        "count": len(statements_data),
    }


if __name__ == "__main__":
    run_validation_tests()
