"""
Refactored LLM User Simulator V2 - Simplified and modularized version.
Uses utility classes for better separation of concerns and maintainability.
"""

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

# 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 (
    RECOMMENDATION_RESPONSE_PROMPT,
    FINAL_REQUEST_PROMPT
)

# Import utility classes
from src.tools.utils.persona_utils import PersonaManager, generate_repetition_behavioral_instructions
from src.tools.utils.constraint_utils import ConstraintChecker
from src.tools.utils.text_utils import get_natural_attribute_name


class LLMCommunicator:
    """Handles LLM communication with unified interface."""
    
    def __init__(self, client, model_name: str = "gpt-4o"):
        self.client = client
        self.model_name = model_name
    
    def call_llm(self, system_prompt: str, conversation_history: List[Dict] = None, 
                 return_termination: bool = False) -> Dict[str, Any]:
        """Unified LLM calling method with optional termination condition support."""
        # 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 = self.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", "Errrr, I'm not sure what to say.")
            # print(f"Response Json: {response_json}")
            # print(f"Response Analysis: {analysis}")
            
            # Ensure terminating_condition is present
            if "terminating_condition" not in response_json:
                response_json["terminating_condition"] = "continue"
            
            result = {
                "dialogue": dialogue,
                "terminating_condition": response_json["terminating_condition"]
            }
            
            # Add selected_option if present and requested
            if return_termination and "selected_option" in response_json:
                result["selected_option"] = response_json["selected_option"]
            
            return result
        except Exception as e:
            print(f"Error in LLM call: {e}")
            return {
                "dialogue": "I'm sorry, I'm not sure how to respond to that.", 
                "terminating_condition": "continue"
            }
    
    def generate_final_request(self) -> str:
        """Generates the user's final request for a proposal."""
        return "I need to go, please give me your best recommendation based on all our conversations so far."


class LLMUserV2:
    """A simplified user simulator with focused, single-purpose methods using utility classes."""
    
    def __init__(self, task_data: Dict[str, Any], model_name: str = "gpt-4o", disable_constraint_checking: bool = False, reveal_pattern: str = "default"):
        self.model_name = model_name
        self.disable_constraint_checking = disable_constraint_checking
        self.reveal_pattern = reveal_pattern
        
        # Unpack new task data structure
        self.task_description = task_data.get("user_query", "")
        self.ground_truth = task_data.get("ground_truth", {})
        self.archetype_id = task_data.get("archetype_id", "")
        
        # New constraint structure
        self.initial_constraints = task_data.get("initial_constraints", [])
        self.progressive_constraints = task_data.get("progressive_constraints", [])
        self.progressive_constraints_revealed = []  # Track which progressive constraints have been revealed
        
        # New utility objective structure
        self.utility_type = task_data.get("utility_type", "")
        self.utility_objective = task_data.get("utility_objective", {})
        self.utility_stated = False  # Track if utility objective has been stated
        
        # Initialize utility classes
        self.persona_manager = PersonaManager(self.archetype_id)
        self.constraint_checker = ConstraintChecker(client, model_name)
        self.llm_communicator = LLMCommunicator(client, model_name)
        
        # Generate persona using utility class, but override communication style with random selection
        self.persona = self.persona_manager.generate_persona()
        
        # Override persona revelation settings based on reveal_pattern
        self.persona = self.persona_manager.apply_reveal_pattern_override(self.persona, reveal_pattern)
        
        # Override communication style with random selection from communication_styles.json
        self.persona['communication_style'] = self._get_random_communication_style()
        
        # Initialize repetition behavior tracking
        self.repetition_triggered = False
        
        # Initialize upfront revelation tracking
        self.upfront_revelation_done = False
        
        # Note: Termination logic now handled by InteractionEnvironment
    
    def _get_random_communication_style(self) -> str:
        """Randomly select a communication style from communication_styles.json."""
        import os
        import json
        
        # Get the path to communication_styles.json
        current_dir = os.path.dirname(os.path.abspath(__file__))
        styles_path = os.path.join(current_dir, "..", "..", "..", "tasks", "communication_styles.json")
        
        try:
            with open(styles_path, 'r') as f:
                styles_list = json.load(f)
            
            # Randomly select one style from the list
            return random.choice(styles_list)
        except (FileNotFoundError, json.JSONDecodeError) as e:
            print(f"Warning: Could not load communication styles from {styles_path}: {e}")
            # Fallback to a default style
            return "You are polite and friendly in your communication style."
    
    
    # def _translate_max_occupancy_constraints(self, constraints: List[Dict[str, Any]]) -> List[Dict[str, Any]]:
    #     """Translate max_occupancy 'equals' constraints to 'greater_than_or_equal' for user simulator."""
    #     translated_constraints = []
    #     for constraint in constraints:
    #         constraint_copy = constraint.copy()
    #         if constraint_copy.get("attribute") == "max_occupancy" and constraint_copy.get("type") == "equals":
    #             constraint_copy["type"] = "greater_than_or_equal"
    #         translated_constraints.append(constraint_copy)
    #     return translated_constraints
    
    def get_all_current_constraints(self) -> List[Dict[str, Any]]:
        """Get all currently active constraints (initial + revealed progressive)."""
        all_constraints = self.initial_constraints + self.progressive_constraints_revealed
        return all_constraints
    
    def get_unrevealed_progressive_constraints(self) -> List[Dict[str, Any]]:
        """Get progressive constraints that haven't been revealed yet."""
        revealed_attrs = {c["attribute"] for c in self.progressive_constraints_revealed}
        return [c for c in self.progressive_constraints if c["attribute"] not in revealed_attrs]
    
    def check_all_progressive_constraints_revealed(self) -> bool:
        """Check if all progressive constraints have been revealed."""
        return len(self.get_unrevealed_progressive_constraints()) == 0
    
    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."""
        return self.llm_communicator.generate_final_request()
    
    def generate_response(self, agent_dialogue: str, formal_recommendation: Optional[Dict] = None, 
                         is_final_turn_approaching: bool = False, 
                         conversation_history: Optional[List[Dict]] = None,
                         save_human_eval: bool = False) -> Dict[str, Any]:
        """Generate user response using focused, targeted prompts."""
        # Use the conversation_history passed from InteractionEnvironment
        if conversation_history is None:
            conversation_history = []
        

        # Check termination conditions
        termination_result = self._check_termination_conditions(formal_recommendation, conversation_history)
        # print(f"Termination result: {termination_result}")
        if termination_result:
            return termination_result
        
        # Handle final turn nudge
        if is_final_turn_approaching:
            final_nudge = self.generate_final_request()
            return {
                "dialogue": final_nudge,
                "terminating_condition": "continue"
            }
        
        # Check for repetition behavior instructions (now applies to progressive constraints)
        repetition_instructions = self._handle_repetition_behavior(formal_recommendation)
        # print(f"➡️ Repetition instructions: {repetition_instructions}")
        
        # Handle utility objective statement (early in conversation)
        utility_injection = self._handle_utility_objective_statement()
        
        # Handle progressive constraint revelation
        progressive_constraint_injection = self._handle_progressive_constraint_revelation()

        # Get persona guidance
        persona_guidance = self.persona_manager.generate_behavioral_guidance(self.persona)
        
        # Use unified response method for all scenarios
        return self._handle_unified_response(
            formal_recommendation, persona_guidance, repetition_instructions, conversation_history, 
            utility_injection, progressive_constraint_injection, save_human_eval
        )
    
    
    def _handle_repetition_behavior(self, formal_recommendation: Optional[Dict] = None) -> str:
        """Handle repetition behavior instructions for suspicious users."""
        # if not tool_calls:
        #     return ""
        
        all_progressive_constraints_revealed = self.check_all_progressive_constraints_revealed()
        repetition_instructions, self.repetition_triggered = generate_repetition_behavioral_instructions(
            self.persona, self.repetition_triggered, all_progressive_constraints_revealed
        )
        
        # 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]}...")
        
        return repetition_instructions
    
    def _handle_utility_objective_statement(self) -> str:
        """Handle utility objective statement in early conversation turns."""
        if self.utility_stated:
            return ""
        
        # Mark as stated to avoid repetition
        self.utility_stated = True
        
        if self.utility_type == "continuous_metric":
            attribute = self.utility_objective.get("attribute", "")
            goal = self.utility_objective.get("goal", "")
            if goal == "minimize":
                return f"You should naturally mention that you want to {goal} {attribute}. For example, say something like 'I want to find the most affordable option' or 'I'm looking to minimize costs'."
            else:
                return f"You should naturally mention that you want to {goal} {attribute}. For example, say something like 'I want to find the highest-rated option' or 'I'm looking for places with the best ratings'."
        
        elif self.utility_type == "maximal_attribute_satisfaction":
            target_attrs = self.utility_objective.get("target_attributes", [])
            if target_attrs:
                # Create a natural list of desired attributes
                attr_list = []
                for attr in target_attrs:
                    attr_name = get_natural_attribute_name(attr["attribute"])
                    if attr["operator"] == "equals" and attr["value"] is True:
                        if attr["attribute"].startswith("has_"):
                            attr_list.append(attr_name)  # Already formatted as "has X"
                        else:
                            attr_list.append(f"{attr_name}")
                    elif attr["operator"] == "equals" and attr["value"] is False:
                        if attr["attribute"].startswith("has_"):
                            attr_list.append(f"does not {attr_name.replace('has ', '')}")
                        else:
                            attr_list.append(f"not {attr_name}")
                    elif attr['operator'] == 'equals':
                        attr_list.append(f"{attr_name} is {attr['value']}")
                    elif attr["operator"] == "greater_than_or_equal":
                        attr_list.append(f"{attr_name} of at least {attr['value']}")
                    elif attr["operator"] == "less_than_or_equal":
                        if attr["attribute"] == "price":
                            attr_list.append(f"{attr_name} of at most ${attr['value']}")
                        else:
                            attr_list.append(f"{attr_name} of at most {attr['value']}")
                    elif attr["operator"] == "in":
                        if isinstance(attr["value"], list) and len(attr["value"]) > 1:
                            last_item = attr["value"][-1]
                            value_text = ", ".join(attr["value"][:-1]) + f", or {last_item}"
                            attr_list.append(f"{attr_name} should be one of: {value_text}")
                        elif isinstance(attr["value"], list) and len(attr["value"]) == 1:
                            first_item = attr["value"][0]
                            attr_list.append(f"{attr_name} should be {first_item}")
                        else:
                            attr_list.append(f"{attr_name} should be {attr['value']}")
                    elif attr["operator"] == "contains":
                        attr_list.append(f"room should include {attr['value']}")
                    else:
                        attr_list.append(attr_name)
                
                attr_text = ", ".join(attr_list)
                return f"You should naturally mention that you want places that meet as many of these criteria as possible (not necessarily all of them), such as: {attr_text}. These are wish-list items, not requirements - the goal is optimization. Say something like 'I'm looking for places that have as many of these features as possible: {attr_text}. I understand that no place might have everything, but I want to maximize how many of these I can get' or 'I want to find the option that checks off the most boxes from my wishlist: {attr_text}. These are nice-to-haves, not deal-breakers'."
        
        return ""
    
    def _handle_progressive_constraint_revelation(self) -> str:
        """Handle progressive constraint revelation logic."""
        if not self.progressive_constraints:
            return ""
        
        # Handle upfront revelation mode
        if self.reveal_pattern == "upfront":
            if not self.upfront_revelation_done:
                # Reveal ALL progressive constraints in the first turn
                self.upfront_revelation_done = True
                to_reveal = self.progressive_constraints.copy()
                
                # Add all to revealed list
                for constraint in to_reveal:
                    self.progressive_constraints_revealed.append(constraint)
                
                # Update hard_constraints for constraint checker
                self.hard_constraints = self.get_all_current_constraints()
                
                # Generate natural constraint text for all constraints
                constraint_texts = []
                for constraint in to_reveal:
                    attr = get_natural_attribute_name(constraint["attribute"])
                    constraint_type = constraint["type"]
                    value = constraint["value"]
                    
                    # Translate max_occupancy "equals" to "greater_than_or_equal" for user simulator
                    if constraint["attribute"] == "max_occupancy" and constraint_type == "equals":
                        constraint_type = "greater_than_or_equal"
                    
                    if constraint_type == "equals":
                        if isinstance(value, bool):
                            if value:
                                constraint_texts.append(f"the place should have {attr}")
                            else:
                                constraint_texts.append(f"the place should not have {attr}")
                        else:
                            constraint_texts.append(f"{attr} should be {value}")
                    elif constraint_type == "greater_than_or_equal":
                        constraint_texts.append(f"{attr} should be at least {value}")
                    elif constraint_type == "less_than_or_equal":
                        if constraint["attribute"] == "price":
                            constraint_texts.append(f"nightly room price should be at most ${value}")
                        else:
                            constraint_texts.append(f"{attr} should be at most {value}")
                    else:
                        constraint_texts.append(f"{attr} constraint: {constraint_type} {value}")
                
                constraint_text = ", ".join(constraint_texts)
                return f"You MUST mention all these additional hard constraints upfront (this is mandatory): {constraint_text}. These are non-negotiable requirements, not wish-list items. Say something like 'Oh, and I should also mention another hard requirement: {constraint_text}. This is non-negotiable.'"
            else:
                # Already revealed all constraints upfront
                return ""
        
        # Standard progressive revelation mode
        # Get unrevealed progressive constraints
        unrevealed = self.get_unrevealed_progressive_constraints()
        if not unrevealed:
            return ""
        
        # Use persona-based probability for revelation (fallback to preference settings if needed)
        reveal_probability = self.persona.get("constraint_reveal_probability", 
                                             self.persona.get("preference_reveal_probability", 0.8))
        constraints_per_turn = self.persona.get("constraints_per_turn", 
                                               self.persona.get("preferences_per_turn", 1))
        
        # Simple probability-based revelation
        if random.random() < reveal_probability:
            # Reveal up to constraints_per_turn constraints
            to_reveal = unrevealed[:constraints_per_turn]
            
            # Add to revealed list
            for constraint in to_reveal:
                self.progressive_constraints_revealed.append(constraint)
                
            # Update hard_constraints for constraint checker
            self.hard_constraints = self.get_all_current_constraints()
            
            # Generate natural constraint text  
            constraint_texts = []
            for constraint in to_reveal:
                attr = get_natural_attribute_name(constraint["attribute"])
                constraint_type = constraint["type"]
                value = constraint["value"]
                
                if constraint_type == "equals":
                    if isinstance(value, bool):
                        if value:
                            constraint_texts.append(f"the place should have {attr}")
                        else:
                            constraint_texts.append(f"the place should not have {attr}")
                    else:
                        constraint_texts.append(f"{attr} should be {value}")
                elif constraint_type == "greater_than_or_equal":
                    constraint_texts.append(f"{attr} should be at least {value}")
                elif constraint_type == "less_than_or_equal":
                    if constraint["attribute"] == "price":
                        constraint_texts.append(f"nightly room price should be at most ${value}")
                    else:
                        constraint_texts.append(f"{attr} should be at most {value}")
                else:
                    constraint_texts.append(f"{attr} constraint: {constraint_type} {value}")
            
            constraint_text = ", ".join(constraint_texts)
            return f"You MUST mention this new hard constraint (this is mandatory): {constraint_text}. This is a non-negotiable requirement, not a wish-list item. Say something like 'Actually, I forgot to mention that {constraint_text}. This is non-negotiable and important to me.'"
        
        return ""
    
    def _format_utility_objective_for_prompt(self) -> str:
        """Format utility objective for display in prompts."""
        if self.utility_type == "continuous_metric":
            attribute = self.utility_objective.get("attribute", "")
            goal = self.utility_objective.get("goal", "")
            return f"Your goal is to {goal} {get_natural_attribute_name(attribute)}"
        elif self.utility_type == "maximal_attribute_satisfaction":
            target_attrs = self.utility_objective.get("target_attributes", [])
            if target_attrs:
                attr_list = []
                for attr in target_attrs:
                    attr_name = get_natural_attribute_name(attr["attribute"])
                    if attr["operator"] == "equals" and attr["value"] is True:
                        if attr["attribute"].startswith("has_"):
                            attr_list.append(attr_name)  # Already formatted as "has X"
                        else:
                            attr_list.append(f"{attr_name}")
                    elif attr["operator"] == "equals" and attr["value"] is False:
                        if attr["attribute"].startswith("has_"):
                            attr_list.append(f"does not {attr_name.replace('has ', '')}")
                        else:
                            attr_list.append(f"not {attr_name}")
                    elif attr["operator"] == "greater_than_or_equal":
                        attr_list.append(f"{attr_name} of at least {attr['value']}")
                    elif attr["operator"] == "less_than_or_equal":
                        if attr["attribute"] == "price":
                            attr_list.append(f"{attr_name} of at most ${attr['value']}")
                        else:
                            attr_list.append(f"{attr_name} of at most {attr['value']}")
                    elif attr["operator"] == "in":
                        if isinstance(attr["value"], list) and len(attr["value"]) > 1:
                            last_item = attr["value"][-1]
                            value_text = ", ".join(attr["value"][:-1]) + f", or {last_item}"
                            attr_list.append(f"{attr_name} should be one of: {value_text}")
                        elif isinstance(attr["value"], list) and len(attr["value"]) == 1:
                            first_item = attr["value"][0]
                            attr_list.append(f"{attr_name} should be {first_item}")
                        else:
                            attr_list.append(f"{attr_name} should be {attr['value']}")
                    elif attr["operator"] == "contains":
                        attr_list.append(f"room should include {attr['value']}")
                    else:
                        attr_list.append(attr_name)
                
                attr_text = ", ".join(attr_list)
                return f"You want to find a place that has as many of the following nice-to-have features as possible (not necessarily all of them): {attr_text}. These are optimization targets/wish-list items, not hard requirements."
        return ""
    
    def _handle_unified_response(self, formal_recommendation: Optional[Dict], persona_guidance: str, 
                                repetition_instructions: str, conversation_history: List[Dict], 
                                utility_injection: str = "", progressive_constraint_injection: str = "",
                                save_human_eval: bool = False) -> Dict[str, Any]:
        """Handle unified response - works for both recommendations and information gathering."""
        
        # Get formatted constraints and utility objective
        from src.tools.utils.text_utils import format_constraints_natural_language
        current_constraints = self.get_all_current_constraints()
        formatted_constraints = format_constraints_natural_language(current_constraints)
        utility_objective_text = self._format_utility_objective_for_prompt()
        
        # Get full agent response from conversation history
        full_agent_response = conversation_history[-1]["content"] if conversation_history else ""
        
        # Prepare separate instruction fields (use "NONE" if empty)
        utility_instruction = utility_injection if utility_injection else "NONE"
        constraint_instruction = progressive_constraint_injection if progressive_constraint_injection else "NONE"
        repetition_instruction = repetition_instructions if repetition_instructions else "NONE"
        
        # Prepare human eval data components
        constraint_check_natural_response = "N/A - agent gathering information"
        
        # Route to appropriate prompt based on constraint checking mode
        if self.disable_constraint_checking:
            # Use low-attention user prompt that misses constraint violations
            from .prompts_v2 import LOW_ATTENTION_USER_PROMPT
            system_prompt = LOW_ATTENTION_USER_PROMPT.format(
                persona_description=f"{self.persona.get('communication_style', 'Friendly, helpful communication')}",
                hard_constraints=formatted_constraints,
                utility_objective=utility_objective_text,
                recommendation_details=full_agent_response,
                utility_instruction=utility_instruction,
                constraint_instruction=constraint_instruction,
                repetition_instruction=repetition_instruction
            )
            # For low attention mode, constraint check is always "looks good"
            constraint_check_natural_response = "The recommendations look good to me!"
        else:
            # Use normal constraint-checking prompt 
            constraint_check = self._get_constraint_check_result(
                full_agent_response, self.persona.get("attention_level", "high"), 
                self.persona.get("description", "A pragmatic person")
            )
            
            constraint_check_natural_response = constraint_check.get("constraint_check_natural_user_response", "The recommendations look fine.")
            
            system_prompt = RECOMMENDATION_RESPONSE_PROMPT.format(
                persona_description=f"{self.persona.get('communication_style', 'Friendly, helpful communication')}",
                hard_constraints=formatted_constraints,
                utility_objective=utility_objective_text,
                constraint_check_natural_response=constraint_check_natural_response,
                utility_instruction=utility_instruction,
                constraint_instruction=constraint_instruction,
                repetition_instruction=repetition_instruction
            )
        
        # Print the user simulator prompt for debugging
        # print("\n" + "="*80)
        # print("USER SIMULATOR PROMPT:")
        # print("="*80)
        # print(system_prompt)
        # print("="*80)
        
        result = self.llm_communicator.call_llm(system_prompt, conversation_history, return_termination=True)
        
        # Add human eval data to result (only if requested)
        if save_human_eval:
            result["human_eval_data"] = {
                "constraint_check_natural_response": constraint_check_natural_response,
                "utility_instruction": utility_instruction,
                "constraint_instruction": constraint_instruction,
                "repetition_instruction": repetition_instruction
            }
        
        return result
    
    def _get_constraint_check_result(self, full_agent_response: str, attention_level: str, persona_description: str) -> Dict[str, Any]:
        """Get constraint check result - either real checking or dummy results for low-attention mode."""
        if self.disable_constraint_checking:
            # Return dummy constraint check results with no violations (for termination logic)
            return {
                "analysis": [],
                "violations": [],
                "constraint_check_natural_user_response": "The recommendations look fine to me!"
            }
        else:
            # Perform actual constraint checking using all current constraints
            current_constraints = self.get_all_current_constraints()
            return self.constraint_checker.check_constraints(
                current_constraints, full_agent_response, attention_level, persona_description
            )
    
    
    def _check_termination_conditions(self, 
                                      formal_recommendation: Optional[Dict] = None,
                                      conversation_history: Optional[List[Dict]] = None) -> Optional[Dict[str, Any]]:
        """Check termination conditions and return response if termination should occur."""
        # Only check natural termination if all progressive constraints have been revealed
        if not self.check_all_progressive_constraints_revealed():
            # print("➡️ Termination check: Not all progressive constraints have been revealed")
            return None
        
        # Check natural termination 
        if conversation_history is not None and len(conversation_history) > 0:
            # Check for constraint violations first using full agent response
            full_agent_response = conversation_history[-1]["content"] if conversation_history else ""
            constraint_check = self._get_constraint_check_result(
                full_agent_response, self.persona.get("attention_level", "high"),
                self.persona.get("description", "A pragmatic person")
            )
            # print(f"Constraint check in termination condition: {constraint_check}")
            violations = constraint_check.get("violations", None)
            if violations is None:
                return None
            elif formal_recommendation is not None and formal_recommendation.get("package_ids") and len(violations) == 0:
                # ready to terminate - agent made formal recommendation with no violations
                # if self.disable_constraint_checking:
                #     print(f"[LOW-ATTENTION TERMINATION] User accepting recommendation without careful constraint checking")
                termination_response = "Thank you!"
                return {
                    "dialogue": termination_response,
                    "terminating_condition": "###STOP###"
                }
            elif (formal_recommendation is None or not formal_recommendation.get("package_ids")) and len(violations) == 0:
                termination_response = "Please formally recommend me some options, thank you!"
                return {
                    "dialogue": termination_response,
                    "terminating_condition": "continue"
                }
            else:
                # violations, not ready to terminate
                termination_response = constraint_check.get("constraint_check_natural_user_response", "Could you please go search again and make sure you have found the best options for me?")
                return {
                    "dialogue": termination_response,
                    "terminating_condition": "continue"
                }
            
        elif conversation_history is None:
            # print("Termination check: No conversation history provided")
            pass
        # Empty conversation_history list is valid (early in conversation) - no logging needed
        
        return None
    
    def _check_natural_termination(self, formal_recommendation: Optional[Dict] = None) -> Optional[Dict[str, Any]]:
        if formal_recommendation and formal_recommendation.get("package_ids"):
            # print(f"formal_recommendation: {formal_recommendation}")
            return {
                        "dialogue": "Thank you!",
                        "terminating_condition": "###STOP###"
                    } 
        else:
            return None
        
    