"""
Preference management utilities for the LLM user simulator.
Handles preference revelation, formatting, and comparison logic.
"""

import random
from typing import List, Dict, Any
from src.tools.utils.text_utils import get_natural_attribute_name, get_weight_context


class PreferenceManager:
    """Manages preference revelation and formatting for the user simulator."""
    
    def __init__(self, utility_function: List[Dict[str, Any]]):
        self.utility_function = utility_function
        self.preference_revelation = self._initialize_preference_revelation()
    
    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": 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_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 = 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 = 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 check_all_preferences_revealed(self) -> bool:
        """Check if all preferences have been revealed."""
        return len(self.preference_revelation["to_reveal"]) == 0
    
    def should_reveal_preference(self, preference_reveal_probability: float) -> bool:
        """Deterministic decision on whether to reveal a new preference this turn."""
        return (len(self.preference_revelation["to_reveal"]) > 0 and 
                random.random() < preference_reveal_probability)
    
    def select_preferences_to_reveal(self, max_batch_size: int) -> 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 []
        
        # 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 = 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 = get_natural_attribute_name(pref['attribute'])
            pref_texts.append(f"{i}. {natural_name} (goal: {goal}, relative importance: {weight} out of 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."