"""
Persona generation and management utilities for the LLM user simulator.
Handles communication styles, behavioral attributes, and persona descriptions.
"""

import json
import os
import random
from typing import Dict, Any


class PersonaManager:
    """Manages persona generation and behavioral guidance."""
    
    def __init__(self, archetype_id: str = ""):
        self.archetype_id = archetype_id
        self.communication_styles = self._load_communication_styles()
    
    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_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) -> 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 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.5, 0.8)
        preferences_per_turn = random.randint(1, 2)
        
        # Trust Level + Repetition Behavior (Hybrid Control)
        # DEPRECATED: Always use 'accepting' to disable repetition behavior
        trust_level = "accepting"  # was: random.choice(["accepting", "suspicious"])
        repetition_style = random.choice(["comprehensive", "selective"])  # kept for compatibility
        
        # 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, 
            )
        }
    
    def generate_behavioral_guidance(self, persona: Dict[str, Any]) -> 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 = 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 = 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 = 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 = 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 apply_reveal_pattern_override(self, persona: Dict[str, Any], reveal_pattern: str) -> Dict[str, Any]:
        """Apply revelation pattern overrides to persona settings.
        
        Args:
            persona: The generated persona dict
            reveal_pattern: One of "default", "upfront", "gradual"
            
        Returns:
            Modified persona dict with revelation settings overridden if needed
        """
        if reveal_pattern == "gradual":
            # Override for minimal revelation rate
            persona["preference_reveal_probability"] = 0.5  # Minimum probability
            persona["preferences_per_turn"] = 1  # One preference at a time
            persona["constraint_reveal_probability"] = 0.5  # Minimum probability  
            persona["constraints_per_turn"] = 1  # One constraint at a time
        elif reveal_pattern == "upfront":
            # Upfront revelation is handled in LLMUserV2 logic, no persona override needed
            pass
        # "default" uses the randomly generated values as-is
        
        return persona


def generate_repetition_behavioral_instructions(persona: Dict[str, Any], repetition_triggered: bool, 
                                               all_progressive_constraints_revealed: bool) -> tuple[str, bool]:
    """Generate behavioral instructions for repetition behavior when triggered.
    
    DEPRECATED: This repetition behavior is currently disabled. All personas are generated
    with trust_level='accepting' to prevent repetition behavior from being triggered.
    """
    if not _should_trigger_repetition_behavior(persona, all_progressive_constraints_revealed, repetition_triggered):
        return "", repetition_triggered
    
    repetition_style = 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 progressive constraints and ask the agent to confirm this recommendation really addresses each one. Be thorough and mention every constraint you've shared."
    else:  # selective  
        instruction = base_instruction + "Re-iterate your TOP 2-3 most important progressive constraints and ask the agent to confirm this is really the best option considering these key priorities. Focus on your highest-weight constraints."
    
    # Mark as triggered after generating instructions
    return instruction, True


def _should_trigger_repetition_behavior(persona: Dict[str, Any], all_progressive_constraints_revealed: bool, 
                                       repetition_triggered: bool) -> bool:
    """Check if repetition behavior should be triggered for suspicious users.
    
    DEPRECATED: This function will always return False since trust_level is always 'accepting'.
    The repetition behavior has been disabled by ensuring no 'suspicious' personas are generated.
    """
    return (
        persona.get("trust_level") == "suspicious" and
        all_progressive_constraints_revealed and
        not repetition_triggered
    )