import openai
import random
import json
import re
import sqlite3
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.")

# Import database access
from tools.accommodations.apis import Accommodations

from .prompts import (
    USER_SIMULATOR_PROMPT, 
    USER_SIMULATOR_PROMPT_NO_RECOMMENDATION,
    INITIAL_QUERY_PROMPT,
    FINAL_REQUEST_PROMPT
)

class LLMUser:
    """A user simulator that generates feedback using an LLM based on a task and ground truth."""
    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.persona = self._generate_persona()
        self.conversation_history = []

    def _get_id_from_name_and_city(self, hotel_name: str, city: str) -> Optional[str]:
        """Fetches an accommodation's ID from the database by its name and city."""
        if not all([hotel_name, city]):
            return None
        try:
            accommodations_tool = Accommodations()
            conn = sqlite3.connect(accommodations_tool.db_path)
            cursor = conn.cursor()
            
            query = "SELECT id FROM accommodations WHERE name = ? AND city = ?"
            result = cursor.execute(query, (hotel_name, city)).fetchone()
            conn.close()
            
            return result[0] if result else None
        except Exception as e:
            print(f"[USER] Error fetching accommodation ID for {hotel_name} in {city}: {e}")
            return None

    def _get_accommodation_details(self, accommodation_id: str) -> Optional[Dict[str, Any]]:
        """Fetches full accommodation details from the database by ID."""
        try:
            accommodations_tool = Accommodations()

            # Direct database query for the specific accommodation
            conn = sqlite3.connect(accommodations_tool.db_path)
            conn.row_factory = sqlite3.Row
            cursor = conn.cursor()
            
            query = "SELECT * FROM accommodations WHERE id = ?"
            result = cursor.execute(query, (accommodation_id,)).fetchone()
            conn.close()
            
            if result:
                # Convert the sqlite3.Row object to a dictionary before printing
                result_dict = dict(result)
                
                # Standardize the accommodation_id field to match the format used in ground truth
                result_dict['accommodation_id'] = accommodation_id
                return result_dict
            
            return None
        except Exception as e:
            print(f"[USER] Error fetching accommodation details for {accommodation_id}: {e}")
            return None

    def _generate_persona(self) -> Dict[str, Any]:
        """Generates a random persona for the user."""
        personas = [
            {"name": "Pragmatic", "description": "You are straightforward and care about getting the best result efficiently."},
            {"name": "Friendly", "description": "You are friendly, polite, and conversational."},
            {"name": "Indecisive", "description": "You are a bit hesitant and like to hear a few options before deciding."},
            {"name": "Demanding", "description": "You have high standards and expect the agent to find the perfect option."},
        ]
        return random.choice(personas)

    def get_initial_query(self) -> str:
        """Returns the pre-generated initial query for the task."""
        # The 'user_query' from the task file is now used directly.
        emphasized_query = self.task_description + " Please find options that strictly follow what I have asked for."
        return emphasized_query

    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 _format_conversation_history(self, conversation_history: List[Dict]) -> str:
        """Formats the conversation history into a readable string for the prompt."""
        formatted_history = []
        for msg in conversation_history:
            role = msg.get('role', 'unknown').lower()
            content = msg.get('content', '')
            
            # The environment now passes clean dialogue, so we don't need to parse JSON here.
            if role == "user":
                formatted_history.append(f"You: {content}")
            elif role == "assistant":
                formatted_history.append(f"Agent: {content}")
        
        return "\n".join(formatted_history)

    def _prepare_briefing(self, agent_recommendation_ids: List[str]) -> Dict[str, Any]:
        """Prepares the 'conscience' briefing for the LLM by analyzing a list of recommendations."""
        briefing = {}
        optimal_choice_id = self.ground_truth.get("optimal_choice", {}).get("accommodation_id")

        # Scenario 1: The agent recommended the single best option.
        if optimal_choice_id in agent_recommendation_ids:
            optimal_details = self._get_accommodation_details(optimal_choice_id)
            optimal_name = optimal_details.get('name', optimal_choice_id)
            briefing["agent_choice_utility"] = round(self.ground_truth.get("utility", 0), 2)
            briefing["optimal_choice_utility"] = briefing["agent_choice_utility"]
            briefing["guidance_hint"] = f"'{optimal_name}' is the best possible option that maximizes your utility. Please make sure all your preferences are met before accepting."
            return briefing

        # Scenario 2: The agent recommended one or more suboptimal options.
        individual_feedback = []
        for rec_id in agent_recommendation_ids:
            agent_choice_utility = self.ground_truth.get("utilities_by_id", {}).get(rec_id)
            agent_details = self._get_accommodation_details(rec_id)
            
            if agent_choice_utility is None or agent_details is None:
                feedback = f"- One of the recommendations is not a valid option and should be pointed out as a mistake."
            else:
                agent_name = agent_details.get('name', rec_id)
                optimal_details = self._get_accommodation_details(optimal_choice_id)
                
                if optimal_details:
                    # Generate specific, comparative feedback for this one option
                    specifics = self._generate_specific_guidance(agent_details, optimal_details)
                    feedback = f"- For '{agent_name}' (Utility: {round(agent_choice_utility, 2)}): {specifics}"
                else:
                    feedback = f"- For '{agent_name}' (Utility: {round(agent_choice_utility, 2)}): The suggestion is okay, but you feel there might be better options."
            
            individual_feedback.append(feedback)

        briefing["agent_choice_utility"] = "Multiple options presented"
        briefing["optimal_choice_utility"] = round(self.ground_truth.get("utility", 0), 2)
        briefing["guidance_hint"] = ("The agent presented several options, but none are the best possible choice. "
                                    "Synthesize the following points into your response to guide the agent:\n" 
                                    + "\n".join(individual_feedback))
        
        print(f"\n[USER] Conscience Briefing prepared:\n{json.dumps(briefing, indent=2)}")
        return briefing

    def _generate_specific_guidance(self, agent_details: Dict, optimal_details: Dict) -> str:
        """Generates elegant, specific guidance by analyzing trade-offs between preferences."""
        suboptimal_prefs = []
        redeeming_prefs = []

        # Always use the full utility function, sorted by importance, for guidance.
        # This ensures the "conscience" is complete, regardless of what has been revealed.
        full_sorted_preferences = sorted(self.utility_function, key=lambda x: x['weight'], reverse=True)

        # Compare the agent's recommendation to the optimal choice for each preference
        for pref in full_sorted_preferences:
            attribute = pref["attribute"]
            goal = pref["goal"]
            
            agent_value = agent_details.get(attribute)
            optimal_value = optimal_details.get(attribute)
            
            if agent_value is None or optimal_value is None:
                continue

            if goal == "maximize":
                if agent_value < optimal_value:
                    suboptimal_prefs.append(pref)
                elif agent_value > optimal_value:
                    redeeming_prefs.append(pref)
            elif goal == "minimize":
                if agent_value > optimal_value:
                    suboptimal_prefs.append(pref)
                elif agent_value < optimal_value:
                    redeeming_prefs.append(pref)

        # If the agent's choice isn't suboptimal on any preference, it's great.
        if not suboptimal_prefs:
            return "The agent's suggestion seems to align perfectly with your preferences. You can accept it."

        # Identify the most important aspect the user is missing out on.
        primary_suboptimal = suboptimal_prefs[0] 
        suboptimal_attr_name = primary_suboptimal["attribute"].replace("_", " ")

        # If there's a redeeming quality, frame the feedback as a trade-off.
        if redeeming_prefs:
            primary_redeeming = redeeming_prefs[0]
            redeeming_attr_name = primary_redeeming["attribute"].replace("_", " ")
            
            return (f"The {redeeming_attr_name} is very good on this one, but for you, {suboptimal_attr_name} is more important. "
                    f"You should ask if there's an option with a better {suboptimal_attr_name}, even if it impacts the {redeeming_attr_name} a bit.")

        # If there's no redeeming quality, the agent's choice is simply dominated.
        else:
            other_important_attr = full_sorted_preferences[1]['attribute'].replace("_", " ") if len(full_sorted_preferences) > 1 else "price"
            return (f"This option is good, but you feel it could be better on the most important factor: {suboptimal_attr_name}. "
                    f"You should ask if there are any other choices that have a better {suboptimal_attr_name} while keeping the {other_important_attr} similar.")

    def generate_response(self, agent_dialogue: str, tool_calls: Optional[List[Dict]] = None, is_final_turn_approaching: bool = False) -> Dict[str, Any]:
        """Generates a user response based on the agent's last message and any recommendations."""
        self.conversation_history.append({"role": "assistant", "content": agent_dialogue})

        # Handle case where the final turn is approaching and we need to encourage a recommendation
        if is_final_turn_approaching and not tool_calls:
            # Use the generate_final_request method to encourage the agent to make a recommendation
            final_nudge = self.generate_final_request()
            dialogue = final_nudge
            
            return {
                "dialogue": dialogue,
                "terminating_condition": "continue"
            }

        if tool_calls:
            # Handle recommendations, which now come as a list of {'hotel_name': str, 'city': str}
            recommendations = tool_calls[0].get("arguments", {}).get("recommendations", [])
            
            if not recommendations:
                 briefing = {"guidance_hint": "The agent tried to make a recommendation but failed to provide the required hotel name and city. You should point this out and ask it to try again."}
            else:
                # Convert name/city pairs to IDs for the "conscience" to process
                recommendation_ids = [self._get_id_from_name_and_city(rec.get('hotel_name'), rec.get('city')) for rec in recommendations]
                print(f"[USER] Recommendation IDs: {recommendation_ids}")
                briefing = self._prepare_briefing(recommendation_ids)
                print(f"[USER] briefing: {briefing}")
            
            prompt = USER_SIMULATOR_PROMPT.format(
                persona_description=self.persona['name'],
                task_description=self.task_description,
                hard_constraints=json.dumps(self.hard_constraints, indent=2),
                utility_function=json.dumps(self.utility_function, indent=2),
                conversation_history=self._format_conversation_history(self.conversation_history),
                agent_choice_utility=briefing.get("agent_choice_utility", "N/A"),
                optimal_choice_utility=briefing.get("optimal_choice_utility", "N/A"),
                guidance_hint=briefing.get("guidance_hint", "No hint available.")
            )
        else:
            prompt = USER_SIMULATOR_PROMPT_NO_RECOMMENDATION.format(
                persona_description=self.persona,
                task_description=self.task_description,
                hard_constraints=json.dumps(self.hard_constraints, indent=2),
                utility_function=json.dumps(self.utility_function, indent=2),
                conversation_history=self._format_conversation_history(self.conversation_history)
            )

        try:
            response = client.chat.completions.create(
                model=self.model_name,
                messages=[{"role": "system", "content": prompt}],
                temperature=0.7,
                response_format={"type": "json_object"}
            )
            response_content = response.choices[0].message.content
            response_json = json.loads(response_content)
            
            dialogue = response_json.get("dialogue", "I'm not sure what to say.")

            # For debugging: print the JSON the LLM produced
            print(f"\n[USER] Internal Thoughts:\n{response_content}")

            self.conversation_history.append({"role": "user", "content": dialogue})
            
            # Ensure terminating_condition is always present for safe access in the environment
            if "terminating_condition" not in response_json:
                response_json["terminating_condition"] = "continue"
            return response_json

        except Exception as e:
            print(f"Error calling OpenAI API for feedback: {e}")
            # Fallback response
            return {"dialogue": "I'm sorry, I'm not sure how to respond to that.", "terminating_condition": "continue"} 