"""
Constraint checking and validation utilities for the LLM user simulator.
Handles constraint checking with attention-level based filtering.
"""

import json
import random
import openai
from typing import Dict, Any, List

# Import the prompts we need
try:
    from ...utility_benchmark.user.prompts_v2 import CONSTRAINT_CHECKER_PROMPT
except ImportError:
    # Fallback if import fails
    CONSTRAINT_CHECKER_PROMPT = """
    You are checking if a recommendation violates any hard constraints.
    
    HARD CONSTRAINTS:
    {hard_constraints}
    
    RECOMMENDATION DETAILS:
    {recommendation_details}
    
    Return a JSON object with:
    - "violations": list of constraint violations found
    - "unmentioned_constraints": list of constraints not addressed in the recommendation
    """

from src.tools.utils.text_utils import format_constraints_natural_language


class ConstraintChecker:
    """Handles constraint checking with attention-level based reliability."""
    
    def __init__(self, client, model_name: str = "gpt-4o"):
        self.client = client
        self.model_name = model_name
    
    def check_constraints(self, hard_constraints: List, recommendation_details: str, 
                         attention_level: str = "high", persona_description: str = "A pragmatic person") -> Dict[str, Any]:
        """Simple, focused constraint checking with attention-level based reliability."""
        # Filter out national_park constraints for constraint checking
        filtered_constraints = []
        for constraint in hard_constraints:
            should_exclude = False
            
            if isinstance(constraint, dict):
                # Check if this is a national_park constraint
                attribute = constraint.get('attribute', '')
                if attribute in  ['national_park', 'date']:
                    should_exclude = True
            elif isinstance(constraint, str):
                # Check if string contains national_park
                if 'national_park' in constraint.lower():
                    should_exclude = True
            
            if not should_exclude:
                filtered_constraints.append(constraint)
        
        formatted_constraints = format_constraints_natural_language(filtered_constraints)
        prompt = CONSTRAINT_CHECKER_PROMPT.format(
            hard_constraints=formatted_constraints,
            recommendation_details=recommendation_details,
            persona_description=persona_description
        )
        
        try:
            response = self.client.chat.completions.create(
                model=self.model_name,
                messages=[{"role": "system", "content": prompt}],
                temperature=0.1,  # Low temperature for consistent constraint checking
                response_format={"type": "json_object"}
            )
            
            constraint_result = json.loads(response.choices[0].message.content)
            # print(f"Constraint checker response: {constraint_result}")
            
            # Handle case where no recommendations were made
            if constraint_result.get("no_recommendations", False):
                return {
                    "analysis": [],
                    "violations": [],
                    "constraint_check_natural_user_response": "N/A - agent gathering information"
                }
            
            # Apply attention-level based filtering
            if attention_level == "high":
                catch_rate = 1.0  # 100% - catch all violations
            elif attention_level == "medium":
                catch_rate = 0.8  # 80% - miss 20% of violations  
            else:  # low
                catch_rate = 0.5  # 50% - miss 50% of violations
            
            # Filter violations based on attention level
            if catch_rate < 1.0 and constraint_result.get("violations"):
                filtered_violations = []
                for violation in constraint_result["violations"]:
                    if random.random() < catch_rate:
                        filtered_violations.append(violation)
                    else:
                        print(f"[ATTENTION FILTER] {attention_level.upper()} attention user missed violation: {violation}")
                constraint_result["violations"] = filtered_violations
            
            return constraint_result
        except Exception as e:
            print(f"Error in constraint checking: {e}")
            return {"violations": [], "unmentioned_constraints": [], "error": "Error in constraint checking"}


def count_recommended_options(tool_calls: List[Dict] = None) -> int:
    """Count the number of recommended options."""
    if not tool_calls:
        return 0
    
    package_ids = tool_calls[0].get("arguments", {}).get("package_ids", [])
    return len(package_ids)