"""
Database-backed Configuration for Mathematical AI Evaluation
Updated: August 18, 2025

This module provides model configuration from the database for all evaluation components.
Uses database-driven configuration for models, companies, and API keys.
"""

import os
from pathlib import Path
from typing import Dict, Any, List, Optional, NamedTuple
from dataclasses import dataclass


class ModelConfig(NamedTuple):
    """Configuration for an individual model loaded from database."""
    model_name: str  # Inspect model identifier (e.g., "anthropic/claude-opus-4-1-20250805")
    display_name: str  # Human-readable name
    company: str  # Company identifier  
    framework_type: str  # "inspect" or "cli"
    api_key_env: str  # Environment variable name for API key
    tier: int  # Model tier for evaluation priority
    model_args: Dict[str, Any]  # Model-specific arguments for client initialization
    reasoning_args: Dict[str, Any]  # Reasoning parameters for eval() function


class DatabaseFrameworkConfig:
    """Database-backed configuration class for Inspect framework integration."""
    
    # Base paths (unchanged from original)
    BASE_DIR = Path(__file__).parent.parent.parent
    LOGS_DIR = BASE_DIR / "inspect_logs"
    EVAL_DIR = BASE_DIR / "web" / "model_evaluation" / "inspect_evaluations"
    
    # Docker configuration - using local built image (unchanged from original)
    DOCKER_IMAGE = "proofbench-math:latest"  # Local image built from Dockerfile
    MAX_CONTAINERS = 4  # Maximum concurrent Docker containers
    
    # Evaluation settings (unchanged from original)
    MAX_RETRIES = 3
    RETRY_DELAY = 30  # seconds
    
    @property
    def MODELS(self):
        """Compatibility property to access models like the original config."""
        return self.get_all_models()
    
    @classmethod
    def get_model_config(cls, model_key: str) -> Optional[ModelConfig]:
        """
        Get model configuration from database (synchronous version).

        Args:
            model_key: Model identifier - can be:
                - model ID as string (e.g., "42") - UNIQUE and recommended
                - company/model_name format (e.g., "anthropic/claude-opus-4-1-20250805") - deprecated (not unique with CLI variants)
                - model_name alone (e.g., "claude-opus-4-1-20250805") - deprecated (not unique)
                - display_name (e.g., "Claude Opus 4.1") - deprecated (not unique)

        Returns:
            ModelConfig object or None if not found

        Note: Only model ID is guaranteed to be unique. Other lookups kept for backward compatibility.
        """
        from .models import Model, ModelArg, ReasoningArg

        try:
            model = None

            # Try model ID first (primary key - guaranteed unique)
            try:
                model_id = int(model_key)
                try:
                    model = Model.objects.get(id=model_id, is_active=True)
                except Model.DoesNotExist:
                    pass
            except (ValueError, TypeError):
                # Not an integer, try other lookup methods
                pass

            # If not found by ID, try company/model_name format
            if not model and '/' in model_key:
                company_name, model_name = model_key.split('/', 1)
                try:
                    model = Model.objects.get(
                        company__company_name=company_name,
                        model_name=model_name,
                        is_active=True
                    )
                except Model.DoesNotExist:
                    pass
                except Model.MultipleObjectsReturned:
                    # Multiple models with same company/model_name (inspect vs CLI)
                    import logging
                    logger = logging.getLogger(__name__)
                    logger.error(f"Multiple models found with company/model_name '{model_key}'. Use model ID instead.")
                    return None

            # If not found, try model_name alone
            if not model:
                try:
                    model = Model.objects.get(model_name=model_key, is_active=True)
                except Model.MultipleObjectsReturned:
                    # Multiple models with same name
                    import logging
                    logger = logging.getLogger(__name__)
                    logger.error(f"Multiple models found with name '{model_key}'. Use model ID instead.")
                    return None
                except Model.DoesNotExist:
                    pass

            # Finally, try display_name
            if not model:
                try:
                    model = Model.objects.get(display_name=model_key, is_active=True)
                except Model.MultipleObjectsReturned:
                    # Multiple models with same display_name
                    import logging
                    logger = logging.getLogger(__name__)
                    logger.error(f"Multiple models found with display_name '{model_key}'. Use model ID instead.")
                    return None
                except Model.DoesNotExist:
                    return None

            if not model:
                return None
            
            # Get model args
            model_args = {}
            for arg in model.model_args.all():
                model_args[arg.arg_name] = arg.get_value()
            
            # Get reasoning args
            reasoning_args = {}
            for arg in model.reasoning_args.all():
                reasoning_args[arg.arg_name] = arg.get_value()
            
            return ModelConfig(
                model_name=model.get_full_model_name(),  # e.g., "anthropic/claude-opus-4-1-20250805"
                display_name=model.display_name or model.model_name,
                company=model.company.company_name if model.company else "unknown",
                framework_type=model.framework_type or "inspect",
                api_key_env=model.company.api_key if model.company else "",
                tier=model.tier.tier_number if model.tier else 1,
                model_args=model_args,
                reasoning_args=reasoning_args
            )
            
        except Exception as e:
            import logging
            logger = logging.getLogger(__name__)
            logger.error(f"Error getting model config for '{model_key}': {str(e)}")
            return None

    @classmethod
    def get_all_models(cls) -> Dict[str, ModelConfig]:
        """
        Get all model configurations from database (synchronous version).
        
        Returns:
            Dictionary of model_key -> ModelConfig where keys include:
            - company/model_name (primary, guaranteed unique)
            - model_name (if unique)
            - display_name (if unique)
        """
        from .models import Model
        
        models_dict = {}
        
        # Track which model_names and display_names are unique
        model_name_counts = {}
        display_name_counts = {}
        
        active_models = list(Model.objects.filter(is_active=True).select_related('company'))
        
        # Count occurrences
        for model in active_models:
            if model.model_name:
                model_name_counts[model.model_name] = model_name_counts.get(model.model_name, 0) + 1
            if model.display_name:
                display_name_counts[model.display_name] = display_name_counts.get(model.display_name, 0) + 1
        
        # Build dictionary
        for model in active_models:
            config = cls.get_model_config(model.get_full_model_name())
            if config:
                # Always use full company/model_name as primary key (guaranteed unique)
                full_name = model.get_full_model_name()
                models_dict[full_name] = config
                
                # Add model_name as key only if it's unique
                if model.model_name and model_name_counts.get(model.model_name, 0) == 1:
                    models_dict[model.model_name] = config
                
                # Add display_name as key only if it's unique
                if model.display_name and display_name_counts.get(model.display_name, 0) == 1:
                    models_dict[model.display_name] = config
        
        return models_dict

    @classmethod
    def get_models_by_tier(cls, tier: int) -> List[ModelConfig]:
        """
        Get all models for a specific tier (synchronous version).
        
        Args:
            tier: Tier number (1, 2, 3, etc.)
            
        Returns:
            List of ModelConfig objects
        """
        from .models import Model
        
        models = []
        for model in Model.objects.filter(tier__tier_number=tier, is_active=True):
            config = cls.get_model_config(model.model_name)
            if config:
                models.append(config)
        
        return models

    @classmethod
    def setup_directories(cls):
        """Setup required directories for evaluation system."""
        # Create logs directory
        cls.LOGS_DIR.mkdir(parents=True, exist_ok=True)
        
        # Create eval directory
        cls.EVAL_DIR.mkdir(parents=True, exist_ok=True)
        
        print(f"✅ Directories setup complete:")
        print(f"   Logs: {cls.LOGS_DIR}")
        print(f"   Evaluations: {cls.EVAL_DIR}")


def validate_environment():
    """Validate that required environment variables are set (synchronous version)."""
    from .models import Company
    
    missing_keys = []
    
    for company in Company.objects.all():
        if company.api_key:
            if not os.getenv(company.api_key):
                missing_keys.append(company.api_key)
    
    if missing_keys:
        print("⚠️  Missing environment variables:")
        for key in missing_keys:
            print(f"   - {key}")
        return False
    
    return True


def get_proofbench_sandbox():
    """
    Get sandbox configuration for ProofBench mathematical evaluations.
    
    According to Inspect AI documentation, you can explicitly specify a compose file
    using tuple syntax: ("docker", "compose-file.yaml")
    
    Returns:
        Tuple specifying docker sandbox with our custom compose.yaml file
    """
    import os
    from pathlib import Path
    
    # Get the absolute path to the compose.yaml file
    current_dir = Path(__file__).parent
    compose_file = current_dir / "compose.yaml"
    
    return ("docker", str(compose_file))