"""
LLM Client for React* agent.

Provides a clean interface for LLM interactions,
independent of any specific framework.
"""

from typing import List, Any, Optional, Tuple
from pydantic import BaseModel
import json


class LLMClient:
    """
    Client for interacting with LLMs.
    
    This class wraps LLM API calls and provides structured output parsing.
    It's designed to be independent of any specific LLM provider.
    """
    
    def __init__(self, model_name: str, api_key: Optional[str] = None, **kwargs):
        """
        Initialize LLM client.
        
        Args:
            model_name: Name of the model to use
            api_key: API key for the LLM provider
            **kwargs: Additional provider-specific parameters
        """
        self.model_name = model_name
        self.api_key = api_key
        self.config = kwargs
        self._client = None
    
    def predict(
        self,
        prompt: str,
        output_format: Optional[type[BaseModel]] = None,
        **kwargs
    ) -> Tuple[Any, Any]:
        """
        Make a prediction using text-only prompt.
        
        Args:
            prompt: Text prompt
            output_format: Pydantic model for structured output
            **kwargs: Additional parameters
            
        Returns:
            Tuple of (parsed_output, raw_response)
        """
        raise NotImplementedError("Subclasses must implement predict()")
    
    def predict_mm(
        self,
        prompt: str,
        images: List[Any],
        output_format: Optional[type[BaseModel]] = None,
        **kwargs
    ) -> Tuple[Any, Any]:
        """
        Make a prediction using multimodal prompt (text + images).
        
        Args:
            prompt: Text prompt
            images: List of images (PIL Image objects or paths)
            output_format: Pydantic model for structured output
            **kwargs: Additional parameters
            
        Returns:
            Tuple of (parsed_output, raw_response)
        """
        raise NotImplementedError("Subclasses must implement predict_mm()")
    
    def parse_structured_output(
        self,
        response_text: str,
        output_format: type[BaseModel]
    ) -> Any:
        """
        Parse structured output from response.
        
        Args:
            response_text: Raw response text
            output_format: Pydantic model to parse into
            
        Returns:
            Parsed output as Pydantic model instance
        """
        try:
            # Try to extract JSON from response
            start_idx = response_text.find('{')
            end_idx = response_text.rfind('}')
            
            if start_idx != -1 and end_idx != -1:
                json_str = response_text[start_idx:end_idx+1]
                data = json.loads(json_str)
                return output_format(**data)
            else:
                raise ValueError("No JSON found in response")
        except Exception as e:
            raise ValueError(f"Failed to parse structured output: {e}")


class OpenAILLMClient(LLMClient):
    """
    OpenAI-specific LLM client implementation.
    
    This can be extended or replaced with other providers.
    """
    
    def __init__(self, model_name: str, api_key: Optional[str] = None, **kwargs):
        """Initialize OpenAI client."""
        super().__init__(model_name, api_key, **kwargs)
        # Initialize OpenAI client here if needed
        # self._client = OpenAI(api_key=api_key)
    
    def predict(
        self,
        prompt: str,
        output_format: Optional[type[BaseModel]] = None,
        **kwargs
    ) -> Tuple[Any, Any]:
        """
        Make a text-only prediction.
        
        This is a placeholder that should be implemented with actual API calls.
        """
        # Placeholder implementation
        # In real usage, this would call OpenAI API
        raise NotImplementedError("OpenAI client needs to be connected to actual API")
    
    def predict_mm(
        self,
        prompt: str,
        images: List[Any],
        output_format: Optional[type[BaseModel]] = None,
        **kwargs
    ) -> Tuple[Any, Any]:
        """
        Make a multimodal prediction.
        
        This is a placeholder that should be implemented with actual API calls.
        """
        # Placeholder implementation
        # In real usage, this would call OpenAI Vision API
        raise NotImplementedError("OpenAI MM client needs to be connected to actual API")


def create_llm_client(
    model_name: str,
    provider: str = "openai",
    **kwargs
) -> LLMClient:
    """
    Factory function to create LLM clients.
    
    Args:
        model_name: Name of the model
        provider: LLM provider ('openai', 'anthropic', etc.)
        **kwargs: Additional parameters
        
    Returns:
        LLMClient instance
    """
    if provider.lower() == "openai":
        return OpenAILLMClient(model_name, **kwargs)
    else:
        raise ValueError(f"Unsupported provider: {provider}")

