"""
Prompt Configuration Loader

This module loads and manages prompts from the centralized YAML configuration file.
"""

import os
import yaml
from pathlib import Path
from typing import Dict, Any, Optional
from decouple import config


class PromptConfig:
    """Manages loading and accessing prompt templates from YAML configuration."""
    
    def __init__(self, config_path: Optional[Path] = None):
        """
        Initialize the prompt configuration.
        
        Args:
            config_path: Path to the prompts.yaml file. If None, uses default location.
        """
        if config_path is None:
            # Get the directory of this file
            config_dir = Path(__file__).parent
            config_path = config_dir / "prompts.yaml"
        
        self.config_path = config_path
        self._config = self._load_config()
    
    def _load_config(self) -> Dict[str, Any]:
        """Load the YAML configuration file."""
        try:
            with open(self.config_path, 'r', encoding='utf-8') as f:
                return yaml.safe_load(f)
        except Exception as e:
            print(f"Warning: Could not load prompt config from {self.config_path}: {e}")
            # Return a minimal default config if loading fails
            return self._get_default_config()
    
    def _get_default_config(self) -> Dict[str, Any]:
        """Provide default configuration as fallback."""
        return {
            'system_prompt': 'You are an expert mathematical problem solver.',
            'submit_responses': {
                'stage_success': 'Answer submitted successfully.',
                'final_success': 'Evaluation complete.',
                'submission_error': 'Submission error: {error}'
            },
            'subquestion_transition': {
                'template': 'Please solve the following subquestion: {subquestion_text}'
            }
        }
    
    def get_system_prompt(self, main_token_limit: int = 300000, subquestion_token_limit: int = 100000) -> str:
        """
        Get the system prompt for the mathematical assistant with dynamic token limits.
        
        Args:
            main_token_limit: Token limit for main question
            subquestion_token_limit: Token limit for subquestions
        
        Returns:
            Formatted system prompt with token limits
        """
        template = self._config.get('system_prompt', 'You are an expert mathematical problem solver.')
        try:
            return template.format(
                main_question_token_limit=main_token_limit,
                subquestion_token_limit=subquestion_token_limit
            )
        except KeyError as e:
            print(f"Warning: Missing template variable {e} in system prompt")
            return template
    
    def get_submit_response(self, response_type: str, **kwargs) -> str:
        """
        Get a formatted submit response message.
        
        Args:
            response_type: One of 'stage_success', 'final_success', 'submission_error'
            **kwargs: Variables to format into the template
        
        Returns:
            Formatted response message
        """
        responses = self._config.get('submit_responses', {})
        template = responses.get(response_type, '')
        
        # Handle stage name display formatting
        if 'stage_name' in kwargs and 'stage_display_names' in self._config:
            stage_key = kwargs.get('stage_name', 'main_question')
            display_names = self._config['stage_display_names']
            kwargs['stage_name'] = display_names.get(stage_key, stage_key.replace('_', ' ').title())
        
        try:
            return template.format(**kwargs)
        except KeyError as e:
            print(f"Warning: Missing template variable {e} in {response_type}")
            return template
    
    def get_subquestion_transition(self, subquestion_order: str, subquestion_text: str) -> str:
        """
        Get the formatted subquestion transition prompt.
        
        Args:
            subquestion_order: The order letter (e.g., 'a', 'b', 'c')
            subquestion_text: The text of the subquestion
        
        Returns:
            Formatted transition prompt
        """
        template = self._config.get('subquestion_transition', {}).get('template', '')
        return template.format(
            subquestion_order=subquestion_order.upper(),
            subquestion_text=subquestion_text
        )
    
    def get_answer_delimiters(self, stage: str) -> tuple[str, str]:
        """
        Get the answer delimiter strings for a given stage.
        
        Args:
            stage: The evaluation stage (e.g., 'main_question', 'subquestion_a')
        
        Returns:
            Tuple of (start_delimiter, end_delimiter)
        """
        import hashlib
        
        delim_config = self._config.get('answer_delimiters', {})
        prefix = delim_config.get('prefix', '<<<ANSWER_START_')
        suffix = delim_config.get('suffix', '>>>')
        end_prefix = delim_config.get('end_prefix', '<<<ANSWER_END_')
        end_suffix = delim_config.get('end_suffix', '>>>')
        
        # Generate hash for stage
        stage_hash = hashlib.md5(stage.encode()).hexdigest()[:8]
        
        start_delimiter = f"{prefix}{stage_hash}{suffix}"
        end_delimiter = f"{end_prefix}{stage_hash}{end_suffix}"
        
        return start_delimiter, end_delimiter

    def get_tool_description(self, tool_name: str) -> Dict[str, Any]:
        """
        Get the tool description configuration for a specific tool.
        
        Args:
            tool_name: Name of the tool (e.g., 'python', 'bash', 'sage_computation', 'submit')
        
        Returns:
            Dictionary containing tool name, description, and parameters
        """
        tools = self._config.get('tool_descriptions', {})
        tool_config = tools.get(tool_name, {})
        
        # Provide defaults if tool not found
        if not tool_config:
            return {
                'name': tool_name,
                'description': f'Tool {tool_name}',
                'parameters': {}
            }
        
        return tool_config
    
    def get_continue_prompt(self, current_tokens: int = None, token_limit: int = None, include_emulation_reminder: bool = False) -> str:
        """
        Get the continue prompt for when the model needs to continue working.

        Args:
            current_tokens: Current token usage (optional)
            token_limit: Token limit for current stage (optional)
            include_emulation_reminder: Whether to append tool emulation reminder (default: False)

        Returns:
            The continue prompt string, with token info if provided and optional emulation reminder
        """
        if current_tokens is not None and token_limit is not None:
            template = self._config.get('continue_prompt_with_tokens',
                'Please continue working on the current question. Token usage: {current_tokens:,} of {token_limit:,} tokens used.')
            try:
                base_prompt = template.format(current_tokens=current_tokens, token_limit=token_limit)
            except KeyError as e:
                print(f"Warning: Missing template variable {e} in continue_prompt_with_tokens")
                base_prompt = template
        else:
            base_prompt = self._config.get('continue_prompt',
                'Please continue working on the current question. To formally register your answer, use the `submit` tool.')

        # Append tool emulation reminder if requested
        if include_emulation_reminder:
            emulation_reminder = self.get_tool_emulation_reminder('submit_syntax')
            base_prompt += emulation_reminder

        return base_prompt

    def get_tool_emulation_reminder(self, reminder_type: str = 'submit_syntax') -> str:
        """
        Get tool emulation reminders for models without native tool calling support.

        Args:
            reminder_type: Type of reminder (default: 'submit_syntax')

        Returns:
            The reminder string
        """
        reminders = self._config.get('tool_emulation_reminders', {})
        return reminders.get(reminder_type, '')
    
    def get_evaluation_transition_prompt(self, transition_type: str, **kwargs) -> str:
        """
        Get evaluation transition prompts for stage transitions.
        
        Args:
            transition_type: Type of transition ('main_to_subquestion' or 'subquestion_to_subquestion')
            **kwargs: Variables to format into the template
        
        Returns:
            Formatted transition prompt
        """
        transitions = self._config.get('evaluation_transitions', {})
        template = transitions.get(transition_type, '')
        
        try:
            return template.format(**kwargs)
        except KeyError as e:
            print(f"Warning: Missing template variable {e} in {transition_type}")
            return template
    
    def get_final_round_prompt(self) -> str:
        """
        Get the final round prompt for when token limit is exceeded.

        Returns:
            The final round prompt string
        """
        return self._config.get('final_round_prompt',
            '⚠️ TOKEN LIMIT REACHED - You must submit your answer NOW.')

    def get_malformed_submit_confirmation(self, confirmation_type: str, answer_preview: str) -> str:
        """
        Get the malformed submit confirmation prompt.

        Args:
            confirmation_type: Type of malformation ('xml_malformed_json', 'raw_json_no_xml',
                             'incomplete_xml', 'tool_call_malformed_json')
            answer_preview: The extracted answer to show to the model

        Returns:
            Formatted confirmation prompt
        """
        confirmations = self._config.get('malformed_submit_confirmations', {})
        template = confirmations.get(confirmation_type, '')

        if not template:
            # Fallback if configuration is missing
            return f"""The system has detected a malformed submit tool call. The following string was extracted:

{answer_preview}

The above string will be recorded as your submission unless your answer to this prompt contains the string ABORT_SUBMISSION. If you wish to proceed with the submission, please reply with a short acknowledgement."""

        try:
            return template.format(answer_preview=answer_preview)
        except KeyError as e:
            print(f"Warning: Missing template variable {e} in malformed_submit_confirmations.{confirmation_type}")
            return template

    def get_token_limits(self) -> Dict[str, int]:
        """
        Get token limits from environment variables.
        
        Returns:
            Dictionary with token limit configuration
        """
        return {
            'main_question': config('MAIN_QUESTION_TOKEN_LIMIT', default=300000, cast=int),
            'subquestion': config('SUBQUESTION_TOKEN_LIMIT', default=100000, cast=int),
            'max_per_call': config('MAX_TOKENS_PER_CALL', default=100000, cast=int)
        }


# Global instance for easy access
_prompt_config: Optional[PromptConfig] = None


def get_prompt_config() -> PromptConfig:
    """Get the global PromptConfig instance."""
    global _prompt_config
    if _prompt_config is None:
        _prompt_config = PromptConfig()
    return _prompt_config