import openai
import random
import json
import re
import os
from typing import List, Dict, Optional, Any

# Make sure OPENAI_API_KEY is set in your environment variables
try:
    client = openai.OpenAI()
except openai.OpenAIError:
    raise ValueError("Please set the OPENAI_API_KEY environment variable.")

from ..prompts_v2 import (
    CONSTRAINT_CHECKER_PROMPT,
    SIMPLE_RESPONSE_PROMPT,
    RECOMMENDATION_RESPONSE_PROMPT,
    PERSONA_GUIDANCE,
    PREFERENCE_INJECTION_TEMPLATE,
    NO_PREFERENCE_INJECTION,
    FINAL_REQUEST_PROMPT
)

class LLMUserV2:
    """A simplied user simulator with focused, single-purpose methods."""
    
    # Mapping from technical attribute names to natural language descriptions
    ATTRIBUTE_NAME_MAPPING = {
        # Distance and location
        "distance_to_park_km": "proximity to the national park (in kms)",
        "location": "location",
        
        # Pricing
        "price": "nightly price",
        "budget": "total budget",
        
        # Quality ratings
        "review_score": "guest review ratings",
        "rating": "guest ratings", 
        "star_rating": "hotel star rating",
        
        # Accommodation amenities (has_ prefix)
        "has_restaurant": "has an on-site restaurant",
        "has_pool": "has swimming pool",
        "has_kitchenette": "has kitchenette",
        "has_gym": "has fitnesss center",
        "has_air_conditioning": "has air conditioning",
        "has_airport_shuttle": "has airport shuttle service",
        "has_spa": "has spa facilities",
        
        # Accommodation amenities (simplified names)
        "pool": "swimming pool",
        "wifi": "WiFi availability",
        "fitness_center": "fitness facilities",
        "parking": "parking availability"
    }
    
    def __init__(self, task_data: Dict[str, Any], model_name: str = "gpt-4o"):
        self.model_name = model_name
        
        # Unpack task data
        self.task_description = task_data.get("user_query", "")
        self.ground_truth = task_data.get("ground_truth", {})
        self.hard_constraints = task_data.get("constraints", [])
        self.utility_function = task_data.get("utility_function", [])
        self.archetype_id = task_data.get("archetype_id", "")
        
        # Load communication styles
        self.communication_styles = self._load_communication_styles()
        
        self.persona = self._generate_persona()
        # Note: conversation_history is now managed by InteractionEnvironment
        
        # Initialize deterministic preference management
        self.preference_revelation = self._initialize_preference_revelation()
        self.preference_reveal_probability = self.persona.get("preference_reveal_probability", 0.8)
        
        # Initialize repetition behavior tracking
        self.repetition_triggered = False
    
    def _load_communication_styles(self) -> Dict[str, str]:
        """Load communication styles mapping from JSON file."""
        try:
            # Load from tasks/ directory
            current_dir = os.path.dirname(__file__)
            project_root = os.path.dirname(os.path.dirname(os.path.dirname(current_dir)))
            styles_path = os.path.join(project_root, "tasks", "communication_styles.json")
            with open(styles_path, 'r') as f:
                return json.load(f)
        except Exception as e:
            print(f"Warning: Could not load communication styles: {e}")
            return {}
    
    def _get_natural_attribute_name(self, attribute: str) -> str:
        """Convert technical attribute name to natural language description."""
        return self.ATTRIBUTE_NAME_MAPPING.get(attribute, attribute.replace('_', ' '))

    def _initialize_preference_revelation(self) -> Dict[str, Any]:
        """Initialize deterministic preference revelation tracking with weight-based ordering."""
        # Extract all soft preferences with their details
        all_preferences = []
        for pref in self.utility_function:
            if isinstance(pref, dict) and "attribute" in pref:
                pref_info = {
                    "attribute": pref["attribute"],
                    "goal": pref.get("goal", "maximize"),  # "maximize" or "minimize"
                    "weight": pref.get("weight", 1.0),
                    "weight_context": self._get_weight_context(pref.get("weight", 1.0))
                }
                all_preferences.append(pref_info)
        
        # Sort preferences by weight (descending) for weight-prioritized revelation
        all_preferences.sort(key=lambda x: x["weight"], reverse=True)
        
        return {
            "to_reveal": all_preferences.copy(),
            "revealed": []
        }
    
    def _get_weight_context(self, weight: float) -> str:
        """Provide contextual description for weight on 0-1 scale."""
        # Weights are on 0-1 scale where 1.0 = highest importance, 0.0 = lowest
        if weight >= 0.8:
            return "very high priority"
        elif weight >= 0.6:
            return "high priority"
        elif weight >= 0.4:
            return "moderate priority" 
        elif weight >= 0.2:
            return "low priority"
        else:
            return "very low priority"

    def _get_revealed_preferences_summary(self) -> str:
        """Generate a formatted summary of currently revealed preferences for system prompt."""
        if not self.preference_revelation["revealed"]:
            return "No soft preferences have been revealed yet."
        
        revealed_prefs = []
        for i, pref in enumerate(self.preference_revelation["revealed"], 1):
            goal = pref.get("goal", "maximize")
            weight = pref.get("weight", 1.0)
            weight_context = pref.get("weight_context", "moderate priority")
            natural_name = self._get_natural_attribute_name(pref['attribute'])
            revealed_prefs.append(f"{i}. {natural_name} (goal: {goal}, weight: {weight}/1.0, {weight_context})")
        
        header = "REVEALED SOFT PREFERENCES (weights range from 0.0 to 1.0, where 1.0 = highest importance):"
        return header + "\n" + "\n".join(revealed_prefs)

    def _get_comparative_preference_context(self, new_preference: Dict[str, Any]) -> str:
        """Generate comparative context for a new preference relative to existing ones."""
        if not self.preference_revelation["revealed"]:
            # First preference - no comparison possible
            return "This is your first revealed preference."
        
        new_weight = new_preference.get("weight", 1.0)
        new_attr = new_preference.get("attribute", "")
        
        # Find preferences with similar, higher, and lower weights
        higher_prefs = []
        lower_prefs = []
        similar_prefs = []
        
        for pref in self.preference_revelation["revealed"]:
            existing_weight = pref.get("weight", 1.0)
            existing_attr = pref.get("attribute", "")
            
            # Skip comparing against itself
            if existing_attr == new_attr:
                continue
                
            existing_natural_name = self._get_natural_attribute_name(existing_attr)
            if abs(existing_weight - new_weight) <= 0.15:  # Similar (within 0.15)
                similar_prefs.append({"attr": existing_natural_name, "weight": existing_weight})
            elif existing_weight > new_weight:
                higher_prefs.append({"attr": existing_natural_name, "weight": existing_weight})
            else:
                lower_prefs.append({"attr": existing_natural_name, "weight": existing_weight})
        
        # Generate comparative language
        comparisons = []
        
        if higher_prefs:
            # Find the most relevant higher priority item
            highest = max(higher_prefs, key=lambda x: x["weight"])
            diff = highest["weight"] - new_weight
            if diff >= 0.3:
                comparisons.append(f"{highest['attr']} ({highest['weight']}) is much more important to you")
            else:
                comparisons.append(f"{highest['attr']} ({highest['weight']}) is somewhat more important to you")
        
        if lower_prefs:
            # Find the most relevant lower priority item  
            lowest = min(lower_prefs, key=lambda x: x["weight"])
            diff = new_weight - lowest["weight"] 
            if diff >= 0.3:
                comparisons.append(f"you care much more about this than {lowest['attr']} ({lowest['weight']})")
            else:
                comparisons.append(f"you care somewhat more about this than {lowest['attr']} ({lowest['weight']})")
        
        if similar_prefs:
            similar = similar_prefs[0]  # Take first similar one
            comparisons.append(f"this is about as important as {similar['attr']} ({similar['weight']})")
        
        return "COMPARATIVE CONTEXT: " + " and ".join(comparisons) if comparisons else "This preference adds to your existing preferences."

    def _generate_persona(self) -> Dict[str, Any]:
        """Generates a multi-dimensional persona with independent behavioral attributes."""
        
        # Revelation Style (Programmatic Control)
        preference_reveal_probability = random.uniform(0.6, 0.9)
        preferences_per_turn = random.randint(1, 3)
        
        # Trust Level + Repetition Behavior (Hybrid Control) 
        trust_level = random.choice(["accepting", "suspicious"])
        repetition_style = random.choice(["comprehensive", "selective"])
        
        # Attention to Detail (Programmatic Control)
        # Distribution: 70% high, 20% medium, 10% low
        attention_level = random.choices(
            ["high", "medium", "low"], 
            weights=[0.7, 0.2, 0.1]
        )[0]
        
        # Communication Style (task archetype-based)
        communication_style = self._get_communication_style_for_archetype()
        
        return {
            # Revelation attributes
            "preference_reveal_probability": preference_reveal_probability,
            "preferences_per_turn": preferences_per_turn,
            
            # Trust and repetition attributes
            "trust_level": trust_level,
            "repetition_style": repetition_style,
            
            # Attention attributes
            "attention_level": attention_level,
            
            # Communication attributes  
            "communication_style": communication_style,
            "archetype_id": self.archetype_id,
            
            # Generate dynamic description based on attributes
            "description": self._generate_persona_description(
                trust_level, attention_level, communication_style
            )
        }

    def _get_communication_style_for_archetype(self) -> str:
        """Get communication style for the current archetype."""
        if self.archetype_id and self.archetype_id in self.communication_styles:
            return self.communication_styles[self.archetype_id]
        else:
            # Fallback to default style
            return "Friendly, helpful communication with moderate detail and polite tone"
    
    def _generate_persona_description(self, trust_level: str, attention_level: str, communication_style: str) -> str:
        """Generate dynamic persona description based on behavioral attributes."""
        base_descriptions = [
            "You are a user looking for travel accommodations.",
        ]
        
        # Add trust-based description
        if trust_level == "suspicious":
            base_descriptions.append("You tend to be cautious and want to double-check recommendations.")
        else:  # accepting
            base_descriptions.append("You generally trust recommendations when they seem reasonable.")
        
        # Add attention-based description
        if attention_level == "high":
            base_descriptions.append("You pay close attention to details and constraints.")
        elif attention_level == "medium":
            base_descriptions.append("You notice most important details but might miss some minor ones.")
        else:  # low
            base_descriptions.append("You focus on the big picture and might overlook some details.")
        
        return " ".join(base_descriptions)

    def _check_all_preferences_revealed(self) -> bool:
        """Check if all preferences have been revealed."""
        return len(self.preference_revelation["to_reveal"]) == 0

    def _should_trigger_repetition_behavior(self) -> bool:
        """Check if repetition behavior should be triggered for suspicious users."""
        return (
            self.persona.get("trust_level") == "suspicious" and
            self._check_all_preferences_revealed() and
            not self.repetition_triggered
        )

    def _generate_repetition_behavioral_instructions(self) -> str:
        """Generate behavioral instructions for repetition behavior when triggered."""
        if not self._should_trigger_repetition_behavior():
            return ""
        
        repetition_style = self.persona.get("repetition_style", "selective")
        
        base_instruction = "IMPORTANT BEHAVIORAL INSTRUCTION: You are feeling cautious about this recommendation and want to double-check. "
        
        if repetition_style == "comprehensive":
            instruction = base_instruction + "Re-iterate ALL of your revealed preferences and ask the agent to confirm this recommendation really addresses each one. Be thorough and mention every preference you've shared."
        else:  # selective  
            instruction = base_instruction + "Re-iterate your TOP 2-3 most important preferences and ask the agent to confirm this is really the best option considering these key priorities. Focus on your highest-weight preferences."
        
        # Mark as triggered after generating instructions
        self.repetition_triggered = True
        
        return instruction

    def _should_reveal_preference(self) -> bool:
        """Deterministic decision on whether to reveal a new preference this turn."""
        return (len(self.preference_revelation["to_reveal"]) > 0 and 
                random.random() < self.preference_reveal_probability)
    
    def _select_preferences_to_reveal(self) -> List[Dict[str, Any]]:
        """Select batch of preferences to reveal based on weight priority and persona limits."""
        if not self.preference_revelation["to_reveal"]:
            return []
        
        # Get persona's preferred batch size
        max_batch_size = self.persona.get("preferences_per_turn", 1)
        
        # Select preferences from highest weight first (already sorted)
        batch_size = min(max_batch_size, len(self.preference_revelation["to_reveal"]))
        preferences_to_reveal = self.preference_revelation["to_reveal"][:batch_size]
        
        # Move selected preferences from to_reveal to revealed
        for preference in preferences_to_reveal:
            self.preference_revelation["to_reveal"].remove(preference)
            self.preference_revelation["revealed"].append(preference)
        
        return preferences_to_reveal

    def _format_multiple_preferences_for_injection(self, preferences: List[Dict[str, Any]]) -> str:
        """Format multiple preferences for injection into system prompt."""
        if not preferences:
            return ""
        
        if len(preferences) == 1:
            # Single preference - use existing format
            pref = preferences[0]
            goal = pref.get('goal', 'maximize')
            weight = pref.get('weight', 1.0)
            weight_context = pref.get('weight_context', 'moderate priority')
            natural_name = self._get_natural_attribute_name(pref['attribute'])
            return f"{natural_name} (goal: {goal}, weight: {weight}/1.0, {weight_context})"
        
        # Multiple preferences - create formatted list
        pref_texts = []
        for i, pref in enumerate(preferences, 1):
            goal = pref.get('goal', 'maximize')
            weight = pref.get('weight', 1.0)
            weight_context = pref.get('weight_context', 'moderate priority')
            natural_name = self._get_natural_attribute_name(pref['attribute'])
            pref_texts.append(f"{i}. {natural_name} (goal: {goal}, weight: {weight}/1.0, {weight_context})")
        
        return "\n".join(pref_texts)

    def _get_batch_comparative_context(self, new_preferences: List[Dict[str, Any]]) -> str:
        """Generate comparative context for a batch of new preferences."""
        if not new_preferences:
            return ""
        
        if len(new_preferences) == 1:
            # Single preference - use existing method
            return self._get_comparative_preference_context(new_preferences[0])
        
        # Multiple preferences - create batch context
        batch_weights = [p.get("weight", 1.0) for p in new_preferences]
        min_weight = min(batch_weights)
        max_weight = max(batch_weights)
        
        if len(self.preference_revelation["revealed"]) == len(new_preferences):
            # These are the first preferences being revealed
            return f"These are your first {len(new_preferences)} revealed preferences (weights: {min_weight:.1f}-{max_weight:.1f})."
        
        # Compare with existing preferences
        existing_weights = [p.get("weight", 1.0) for p in self.preference_revelation["revealed"] if p not in new_preferences]
        if existing_weights:
            avg_existing = sum(existing_weights) / len(existing_weights)
            avg_new = sum(batch_weights) / len(batch_weights)
            
            if avg_new > avg_existing + 0.2:
                return f"These {len(new_preferences)} preferences are more important than your previously revealed ones."
            elif avg_new < avg_existing - 0.2:
                return f"These {len(new_preferences)} preferences are less critical than your previously revealed ones."
            else:
                return f"These {len(new_preferences)} preferences complement your existing preferences."
        
        return f"These are {len(new_preferences)} additional preferences to consider."

    def _format_constraints_natural_language(self) -> str:
        """Convert hard constraints JSON to natural language format with clear emphasis."""
        if not self.hard_constraints:
            return "No specific hard constraints provided."
        
        formatted_constraints = []
        for i, constraint in enumerate(self.hard_constraints, 1):
            if isinstance(constraint, dict):
                # Handle structured constraint objects
                constraint_text = f"🚨 HARD CONSTRAINT {i}: {constraint['attribute'].replace('_', ' ')} "
                if "type" in constraint and "value" in constraint:
                    constraint_text += f"{constraint['type'].replace('_', ' ')}: {constraint['value']}"
                else:
                    constraint_text += str(constraint)
                constraint_text += " (NON-NEGOTIABLE)"
            elif isinstance(constraint, str):
                # Handle simple string constraints
                constraint_text = f"🚨 HARD CONSTRAINT {i}: {constraint} (NON-NEGOTIABLE)"
            else:
                constraint_text = f"🚨 HARD CONSTRAINT {i}: {str(constraint)} (NON-NEGOTIABLE)"
            formatted_constraints.append(constraint_text)
        
        return "\n".join(formatted_constraints)
    
    def _generate_persona_behavioral_guidance(self) -> str:
        """Generate behavioral guidance based on multi-dimensional persona attributes."""
        guidance_parts = ["Be natural and authentic in your responses."]
        
        # Add archetype-specific communication style guidance
        communication_style = self.persona.get("communication_style", "")
        if communication_style and isinstance(communication_style, str):
            guidance_parts.append(f"COMMUNICATION STYLE: {communication_style}")
        
        # Add trust-level guidance
        trust_level = self.persona.get("trust_level", "accepting")
        if trust_level == "suspicious":
            guidance_parts.append("You tend to be cautious and may want to double-check recommendations.")
        else:
            guidance_parts.append("You generally trust reasonable recommendations.")
        
        # Add attention-level guidance
        attention_level = self.persona.get("attention_level", "high")
        if attention_level == "high":
            guidance_parts.append("Pay close attention to all details and constraints.")
        elif attention_level == "medium":
            guidance_parts.append("Notice important details but might miss some minor ones.")
        else:  # low
            guidance_parts.append("Focus on the big picture; some details might slip by.")
        
        # Add revelation style guidance
        preferences_per_turn = self.persona.get("preferences_per_turn", 1)
        if preferences_per_turn > 2:
            guidance_parts.append("You tend to share multiple preferences when you do speak up.")
        elif preferences_per_turn == 1:
            guidance_parts.append("You typically share one preference at a time when revealing them.")
        
        return " ".join(guidance_parts)

    def _check_constraints(self, recommendation_details: str) -> Dict[str, Any]:
        """Simple, focused constraint checking with attention-level based reliability."""
        formatted_constraints = self._format_constraints_natural_language()
        prompt = CONSTRAINT_CHECKER_PROMPT.format(
            hard_constraints=formatted_constraints,
            recommendation_details=recommendation_details
        )
        
        try:
            response = 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)
            
            # Apply attention-level based filtering
            attention_level = self.persona.get("attention_level", "high")
            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(self, tool_calls: Optional[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)

    def get_initial_query(self) -> str:
        """Returns the pre-generated initial query for the task from the JSONL file."""
        return self.task_description

    def generate_final_request(self) -> str:
        """Generates the user's final request for a proposal."""
        try:
            response = client.chat.completions.create(
                model=self.model_name,
                messages=[{"role": "system", "content": FINAL_REQUEST_PROMPT}],
                temperature=0.7,
                max_tokens=100,
            )
            return response.choices[0].message.content.strip()
        except Exception as e:
            # print(f"Error calling OpenAI API for final request: {e}")
            return "Please just give me your best recommendation based on our conversation."
        

    def generate_response(self, agent_dialogue: str, tool_calls: Optional[List[Dict]] = None, is_final_turn_approaching: bool = False, conversation_history: Optional[List[Dict]] = None) -> Dict[str, Any]:
        """Generate user response using focused, targeted prompts."""
        # Use the conversation_history passed from InteractionEnvironment
        if conversation_history is None:
            conversation_history = []

        # Handle final turn nudge
        if is_final_turn_approaching:
            final_nudge = self.generate_final_request()
            return {
                "dialogue": final_nudge,
                "terminating_condition": "continue"
            }

        # Deterministic preference revelation logic
        preferences_to_reveal = []
        if self._should_reveal_preference():
            preferences_to_reveal = self._select_preferences_to_reveal()

        # Generate preference injection text
        if preferences_to_reveal:
            # Format multiple preferences for injection
            preference_text = self._format_multiple_preferences_for_injection(preferences_to_reveal)
            
            # Get comparative context for the batch
            comparative_context = self._get_batch_comparative_context(preferences_to_reveal)
            
            # Create enhanced template - use first attribute for template examples
            first_attribute = preferences_to_reveal[0]['attribute']
            natural_attribute_name = self._get_natural_attribute_name(first_attribute)
            enhanced_template = PREFERENCE_INJECTION_TEMPLATE.replace("[attribute]", natural_attribute_name)
            
            preference_injection = enhanced_template.format(
                preference_to_reveal=preference_text,
                preference_importance=comparative_context
            )
            
            # Enhanced logging for batch revelation
            pref_count = len(preferences_to_reveal)
            weights = [p.get('weight', 1.0) for p in preferences_to_reveal]
            print(f"\n[BATCH PREFERENCE REVELATION] Revealing {pref_count} preferences (weights: {min(weights):.1f}-{max(weights):.1f}) - {comparative_context}")
        else:
            preference_injection = NO_PREFERENCE_INJECTION
            print(f"\n[PREFERENCE DEFERRAL] Using deferral template - will not hallucinate preferences")

        # Check for repetition behavior instructions (only for recommendations)
        repetition_instructions = ""
        if tool_calls:
            repetition_instructions = self._generate_repetition_behavioral_instructions()
            if repetition_instructions:
                print(f"\n[REPETITION BEHAVIOR] Suspicious user triggering repetition behavior: {self.persona.get('repetition_style', 'selective')}")
                print(f"[REPETITION INSTRUCTION] {repetition_instructions[:100]}...")

        # Get persona guidance
        persona_guidance = self._generate_persona_behavioral_guidance()
        
        # Route to appropriate targeted prompt
        if tool_calls:
            return self._handle_recommendation(tool_calls, persona_guidance, preference_injection, repetition_instructions, conversation_history)
        else:
            return self._handle_no_recommendation(persona_guidance, preference_injection, conversation_history)

    def _handle_no_recommendation(self, persona_guidance: str, preference_injection: str, conversation_history: List[Dict]) -> Dict[str, Any]:
        """Handle response when no recommendation is made."""
        revealed_prefs_summary = self._get_revealed_preferences_summary()
        formatted_constraints = self._format_constraints_natural_language()
        system_prompt = SIMPLE_RESPONSE_PROMPT.format(
            persona_description=self.persona['description'],
            task_description=self.task_description,
            hard_constraints=formatted_constraints,
            revealed_preferences=revealed_prefs_summary,
            persona_guidance=persona_guidance,
            preference_injection=preference_injection
        )
        
        return self._call_llm_simple(system_prompt, conversation_history)

    def _handle_recommendation(self, tool_calls: List[Dict], persona_guidance: str, preference_injection: str, repetition_instructions: str, conversation_history: List[Dict]) -> Dict[str, Any]:
        """Handle response to recommendation(s)."""
        # Check constraints using full agent response from conversation history
        full_agent_response = conversation_history[-1]["content"] if conversation_history else ""  # Last agent message
        constraint_check = self._check_constraints(full_agent_response)
        print(f"constraint_check: {constraint_check}")
        
        # Count options and determine which prompt to use
        option_count = self._count_recommended_options(tool_calls)
        revealed_prefs_summary = self._get_revealed_preferences_summary()
        formatted_constraints = self._format_constraints_natural_language()
        
        # Use unified prompt for all recommendations
        option_text = "option" if option_count == 1 else "options"
        
        # Incorporate repetition instructions into persona guidance if present
        enhanced_persona_guidance = persona_guidance
        if repetition_instructions:
            enhanced_persona_guidance = f"{persona_guidance}\n\n{repetition_instructions}"
        
        system_prompt = RECOMMENDATION_RESPONSE_PROMPT.format(
            persona_description=self.persona['description'],
            task_description=self.task_description,
            hard_constraints=formatted_constraints,
            revealed_preferences=revealed_prefs_summary,
            recommendation_summary=f"The agent recommended {option_count} accommodation {option_text}.",
            constraint_check_result=json.dumps(constraint_check),
            persona_guidance=enhanced_persona_guidance,
            preference_injection=preference_injection
        )
        
        return self._call_llm_with_termination(system_prompt, conversation_history)

    def _call_llm_simple(self, system_prompt: str, conversation_history: List[Dict] = None) -> Dict[str, Any]:
        """Call LLM with simple response format using proper message structure."""
        # Build proper message array
        messages = [{"role": "system", "content": system_prompt}]
        
        # Add conversation history as proper messages (if provided)
        if conversation_history:
            messages.extend(conversation_history)
        
        try:
            response = client.chat.completions.create(
                model=self.model_name,
                messages=messages,
                temperature=0.7,
                response_format={"type": "json_object"}
            )
            response_json = json.loads(response.choices[0].message.content)
            dialogue = response_json.get("user_response", "I'm not sure what to say.")
            
            # Ensure terminating_condition is present
            if "terminating_condition" not in response_json:
                response_json["terminating_condition"] = "continue"
            
            return {
                "dialogue": dialogue,
                "terminating_condition": response_json["terminating_condition"]
            }
        except Exception as e:
            print(f"Error in simple LLM call: {e}")
            return {"dialogue": "I'm sorry, I'm not sure how to respond to that.", "terminating_condition": "continue"}

    def _call_llm_with_termination(self, system_prompt: str, conversation_history: List[Dict] = None) -> Dict[str, Any]:
        """Call LLM with termination condition support using proper message structure."""
        # Build proper message array
        messages = [{"role": "system", "content": system_prompt}]
        
        # Add conversation history as proper messages (if provided)
        if conversation_history:
            messages.extend(conversation_history)
        
        try:
            response = client.chat.completions.create(
                model=self.model_name,
                messages=messages,
                temperature=0.7,
                response_format={"type": "json_object"}
            )
            response_json = json.loads(response.choices[0].message.content)
            dialogue = response_json.get("user_response", "I'm not sure what to say.")
            
            # Ensure terminating_condition is present
            if "terminating_condition" not in response_json:
                response_json["terminating_condition"] = "continue"
            
            # Ensure consistent return format with "dialogue" field for compatibility
            return {
                "dialogue": dialogue,
                "terminating_condition": response_json["terminating_condition"],
                "selected_option": response_json.get("selected_option", None)
            }
        except Exception as e:
            print(f"Error in LLM call with termination: {e}")
            return {"dialogue": "I'm sorry, I'm not sure how to respond to that.", "terminating_condition": "continue"} 