"""LLM API Client for interacting with various Large Language Model providers."""

import os
from typing import Optional, Union, List

# Import modules for mocking in tests
try:
    import openai
except ImportError:
    openai = None

try:
    import google.generativeai as genai
except ImportError:
    genai = None

try:
    import anthropic
except ImportError:
    anthropic = None

try:
    import requests
except ImportError:
    requests = None


class LLMAPIClient:
    """Client for interacting with various LLM providers through their APIs."""
    
    def __init__(self):
        """Initialize the LLM API client."""
        pass
    
    def _get_api_key(self, provider_name: str, env_var_name: str) -> str:
        """Get API key from environment variables.
        
        Args:
            provider_name: Human-readable name of the provider (e.g., "OpenAI", "Gemini")
            env_var_name: Environment variable name for the API key
            
        Returns:
            API key as a string
            
        Raises:
            ValueError: If the API key is not found in environment variables
        """
        api_key = os.getenv(env_var_name)
        if not api_key:
            raise ValueError(f"Missing {provider_name} API key. Please set the {env_var_name} environment variable.")
        return api_key
    
    def call_llm_model(self, prompt: str, model_name: str, temperature: float = 0.7, **kwargs) -> str:
        """Call LLM API to generate text using the appropriate provider based on model name.
        
        Args:
            prompt: The input prompt for text generation
            model_name: Name of the LLM model to use (e.g., 'gpt-4o', 'claude-sonnet-4-20250514', 'deepseek-reasoner', 'o1')
            temperature: Sampling temperature (0.0 to 2.0, default: 0.7)
            **kwargs: Additional provider-specific parameters
            
        Returns:
            Generated solution text from the LLM
            
        Raises:
            ValueError: If API keys are missing or parameters are invalid
            Exception: For any other API call errors
        """
        if not prompt or not prompt.strip():
            raise ValueError("prompt must be a non-empty string")
        
        if not model_name or not model_name.strip():
            raise ValueError("model_name must be a non-empty string")
        
        if not isinstance(temperature, (int, float)) or temperature < 0.0 or temperature > 2.0:
            raise ValueError("temperature must be a float between 0.0 and 2.0")
        
        # Route to appropriate provider based on model name prefix
        if model_name.startswith('claude'):
            return self._call_claude(prompt, model_name, temperature, **kwargs)
        elif model_name.startswith('deepseek'):
            return self._call_deepseek(prompt, model_name, temperature, **kwargs)
        else:
            # Default to OpenAI for all other models (including 'o1', 'gpt-4o', etc.)
            return self._call_openai(prompt, model_name, temperature, **kwargs)
    
    def _call_openai(self, prompt: str, model_name: str, temperature: float = 0.7, **kwargs) -> str:
        """Call OpenAI API to generate text.
        
        Args:
            prompt: The input prompt for text generation
            model_name: Name of the OpenAI model to use
            temperature: Sampling temperature (0.0 to 2.0, default: 0.7)
            **kwargs: Additional parameters (ignored for OpenAI)
            
        Returns:
            Generated solution text from the LLM
            
        Raises:
            ValueError: If OPENAI_API_KEY is missing or parameters are invalid
            Exception: For any other API call errors
        """
        # Get API key
        api_key = self._get_api_key("OpenAI", "OPENAI_API_KEY")
        
        try:
            # Check if OpenAI client is available
            if openai is None:
                raise ImportError("OpenAI client library not installed. Please install 'openai' package.")
            
            # Initialize client
            client = openai.OpenAI(api_key=api_key)
            
            # Make API call - handle models that don't support temperature
            if model_name == "o1" or model_name.startswith("gpt-5"):
                # o1 and gpt-5 models don't support temperature parameter
                response = client.chat.completions.create(
                    model=model_name,
                    messages=[{"role": "user", "content": prompt}]
                )
            else:
                response = client.chat.completions.create(
                    model=model_name,
                    messages=[{"role": "user", "content": prompt}],
                    temperature=temperature
                )
            
            # Extract and return solution text
            solution_text = response.choices[0].message.content
            if not solution_text:
                raise RuntimeError("OpenAI API returned empty response")
            
            return solution_text
            
        except ImportError:
            raise ImportError("OpenAI client library not installed. Please install 'openai' package.")
        except RuntimeError:
            raise  # Re-raise RuntimeError directly
        except Exception as e:
            raise Exception(f"OpenAI API call failed: {e}")
    
    def call_gemini(self, prompt: str, model_name: str, temperature: float = 0.7) -> str:
        """Call Google Gemini API to generate text.
        
        Args:
            prompt: The input prompt for text generation
            model_name: Name of the Gemini model to use
            temperature: Sampling temperature (0.0 to 2.0, default: 0.7)
            
        Returns:
            Generated solution text from the LLM
            
        Raises:
            ValueError: If GEMINI_API_KEY is missing or parameters are invalid
            Exception: For any other API call errors
        """
        if not prompt or not prompt.strip():
            raise ValueError("prompt must be a non-empty string")
        
        if not model_name or not model_name.strip():
            raise ValueError("model_name must be a non-empty string")
        
        if not isinstance(temperature, (int, float)) or temperature < 0.0 or temperature > 2.0:
            raise ValueError("temperature must be a float between 0.0 and 2.0")
        
        # Get API key
        api_key = self._get_api_key("Gemini", "GEMINI_API_KEY")
        
        try:
            # Check if Google Generative AI client is available
            if genai is None:
                raise ImportError("Google Generative AI client library not installed. Please install 'google-generativeai' package.")
            
            # Configure API key
            genai.configure(api_key=api_key)
            
            # Get model
            model = genai.GenerativeModel(model_name)
            
            # Generate content
            response = model.generate_content(prompt)
            
            # Extract and return solution text
            solution_text = response.text
            if not solution_text:
                raise RuntimeError("Gemini API returned empty response")
            
            return solution_text
            
        except ImportError:
            raise ImportError("Google Generative AI client library not installed. Please install 'google-generativeai' package.")
        except RuntimeError:
            raise  # Re-raise RuntimeError directly
        except Exception as e:
            raise Exception(f"Gemini API call failed: {e}")
    
    def _call_claude(self, prompt: str, model_name: str, temperature: float = 0.7, **kwargs) -> str:
        """Call Anthropic Claude API to generate text.
        
        Args:
            prompt: The input prompt for text generation
            model_name: Name of the Claude model to use
            temperature: Sampling temperature (0.0 to 2.0, default: 0.7)
            **kwargs: Additional parameters (max_tokens can be overridden)
            
        Returns:
            Generated solution text from the LLM
            
        Raises:
            ValueError: If CLAUDE_API_KEY is missing or parameters are invalid
            Exception: For any other API call errors
        """
        # Get API key
        api_key = self._get_api_key("Claude", "CLAUDE_API_KEY")
        
        try:
            # Check if Anthropic client is available
            if anthropic is None:
                raise ImportError("Anthropic client library not installed. Please install 'anthropic' package.")
            
            # Initialize client
            client = anthropic.Anthropic(api_key=api_key)
            
            # Make API call - Claude requires max_tokens
            # Increased default max_tokens to handle longer responses (like generating 50 solutions)
            # Claude 3.5 Sonnet supports up to 8192 tokens, but we can go higher for newer models
            max_tokens = kwargs.get('max_tokens', 8192)
            response = client.messages.create(
                model=model_name,
                max_tokens=max_tokens,
                temperature=temperature,
                messages=[{"role": "user", "content": prompt}]
            )
            
            # Extract and return solution text
            solution_text = response.content[0].text
            if not solution_text:
                raise RuntimeError("Claude API returned empty response")
            
            return solution_text
            
        except ImportError:
            raise ImportError("Anthropic client library not installed. Please install 'anthropic' package.")
        except RuntimeError:
            raise  # Re-raise RuntimeError directly
        except Exception as e:
            raise Exception(f"Claude API call failed: {e}")
    
    def _call_deepseek(self, prompt: str, model_name: str, temperature: float = 0.7, **kwargs) -> str:
        """Call DeepSeek API to generate text.
        
        Args:
            prompt: The input prompt for text generation
            model_name: Name of the DeepSeek model to use
            temperature: Sampling temperature (0.0 to 2.0, default: 0.7)
            **kwargs: Additional parameters (ignored for DeepSeek)
            
        Returns:
            Generated solution text from the LLM
            
        Raises:
            ValueError: If DEEPSEEK_API_KEY is missing or parameters are invalid
            Exception: For any other API call errors
        """
        # Get API key
        api_key = self._get_api_key("DeepSeek", "DEEPSEEK_API_KEY")
        
        try:
            # Check if requests library is available
            if requests is None:
                raise ImportError("Requests library not installed. Please install 'requests' package.")
            
            # DeepSeek API endpoint
            url = "https://api.deepseek.com/v1/chat/completions"
            
            # Headers
            headers = {
                "Authorization": f"Bearer {api_key}",
                "Content-Type": "application/json"
            }
            
            # Request payload
            payload = {
                "model": model_name,
                "messages": [{"role": "user", "content": prompt}],
                "temperature": temperature
            }
            
            # Make API call
            response = requests.post(url, headers=headers, json=payload)
            response.raise_for_status()
            
            # Parse response
            response_data = response.json()
            solution_text = response_data["choices"][0]["message"]["content"]
            
            if not solution_text:
                raise RuntimeError("DeepSeek API returned empty response")
            
            return solution_text
            
        except ImportError:
            raise ImportError("Requests library not installed. Please install 'requests' package.")
        except RuntimeError:
            raise  # Re-raise RuntimeError directly
        except Exception as e:
            raise Exception(f"DeepSeek API call failed: {e}")
    
    def embed_content(self, model_name: str, contents: Union[str, List[str]]) -> List[List[float]]:
        """Generate embeddings for text using the Gemini embedding model.
        
        Args:
            model_name: Name of the embedding model to use (e.g., "gemini-embedding-001")
            contents: Text content(s) to embed. Can be a single string or list of strings.
            
        Returns:
            List of embeddings, where each embedding is a list of floats
            
        Raises:
            ValueError: If GEMINI_API_KEY is missing or parameters are invalid
            ImportError: If google-generativeai library is not installed
            Exception: For any other API call errors
        """
        if not model_name or not model_name.strip():
            raise ValueError("model_name must be a non-empty string")
        
        if not contents:
            raise ValueError("contents must not be empty")
        
        # Convert single string to list for consistent processing
        if isinstance(contents, str):
            contents = [contents]
        
        # Get API key
        api_key = self._get_api_key("Gemini", "GEMINI_API_KEY")
        
        try:
            # Check if Google Generative AI client is available
            if genai is None:
                raise ImportError("Google Generative AI client library not installed. Please install 'google-generativeai' package.")
            
            # Configure API key
            genai.configure(api_key=api_key)
            
            # Generate embeddings
            embeddings = []
            for content in contents:
                if not content.strip():
                    raise ValueError("Content cannot be empty")
                
                # Use the embedding model
                result = genai.embed_content(
                    model=model_name,
                    content=content,
                    task_type="RETRIEVAL_DOCUMENT"
                )
                
                # Extract embedding values
                embedding_values = result['embedding']
                if not embedding_values:
                    raise RuntimeError(f"Gemini embedding API returned empty response for content: {content[:100]}...")
                
                embeddings.append(embedding_values)
            
            return embeddings
            
        except ImportError:
            raise ImportError("Google Generative AI client library not installed. Please install 'google-generativeai' package.")
        except RuntimeError:
            raise  # Re-raise RuntimeError directly
        except Exception as e:
            raise Exception(f"Gemini embedding API call failed: {e}")
