"""
Base classes for LLM and Embedding providers.

This module defines abstract base classes that all providers must implement,
ensuring consistent interfaces across different service providers.
"""
from abc import ABC, abstractmethod
from typing import List, Dict, Any, Optional
from dataclasses import dataclass, field, asdict
import json
import re

@dataclass
class LLMOutput:
    response_txt: str
    reasoning_content: str
    created: str
    model: str
    request_id: str
    prompt_tokens: Optional[int] = None
    completion_tokens: Optional[int] = None
    total_tokens: Optional[int] = None
    
    @classmethod
    def extract_first_block_from_response(cls, text: str, language_types: list[str]) -> str:
        """
        Extract the first block from the generated text.

        Args:
            text: The generated text.
            language_types: The language types to extract.
        Returns:
            str: The first block.
        """
        
        matches = re.search(r"```(.*?)```", text, flags=re.DOTALL)
        first_block = text.strip()
        if matches:
            first_block = matches.group(1).strip()
            for language_type in language_types:
                if first_block.startswith(language_type):
                    first_block = first_block[len(language_type):].strip()
                    break
        if not first_block:
            raise ValueError("No first block found in the response")
        
        if "json" in language_types:
            return json.loads(first_block)
        else: 
            return first_block
    
    @classmethod
    def from_response(cls, response) -> 'LLMOutput':
        model = getattr(response, "model", None)
        request_id = getattr(response, "id", None)
        created = getattr(response, "created", None)
        reasoning_content = None
        content = None
        message = response.choices[0].message
        if message and hasattr(message, "reasoning_content") and message.reasoning_content is not None:
            reasoning_content = message.reasoning_content
        if message and hasattr(message, "content") and message.content:
            content = message.content
        if not content:
            raise ValueError(
                f"LLM response content is empty. "
                f"model={model}, request_id={request_id}"
            )
        
        prompt_tokens = response.usage.prompt_tokens if response.usage else None
        completion_tokens = response.usage.completion_tokens if response.usage else None
        total_tokens = response.usage.total_tokens if response.usage else None
        
        return cls(
            model=model,
            request_id=request_id,
            created=created,
            response_txt=content,
            reasoning_content = reasoning_content,
            prompt_tokens=prompt_tokens,
            completion_tokens=completion_tokens,
            total_tokens=total_tokens
        )


class BaseLLM(ABC):
    """
    Abstract base class for Large Language Model providers.
    
    All LLM providers must implement the methods defined in this class to ensure
    consistent behavior across different models and services.
    """
    
    def __init__(self, **kwargs: Any) -> None:
        """Initialize the LLM provider with configuration parameters."""
        pass
    
    @abstractmethod
    def generate(self, messages: List[Dict[str, str]], **kwargs: Any) -> str:
        """
        Generate a response from the LLM given input messages.
        
        Args:
            messages: List of message dictionaries with 'role' and 'content' keys
            **kwargs: Additional generation parameters (temperature, max_tokens, etc.)
            
        Returns:
            Generated response string
            
        Raises:
            Exception: If generation fails
        """
    
    @abstractmethod
    def generate_single(self, message: str, **kwargs: Any) -> str:
        """
        Generate a response from the LLM given a single message.
        
        Args:
            message: Single message string
            **kwargs: Additional generation parameters (temperature, max_tokens, etc.)
            
        Returns:
            Generated response string
        """
    


class BaseEmbedder(ABC):
    """
    Abstract base class for text embedding providers.
    
    Embedding providers are used to convert text into vector representations
    for similarity search and retrieval operations.
    """
    
    def __init__(self, **kwargs: Any) -> None:
        """Initialize the embedding provider with configuration parameters."""
        pass
    
    @abstractmethod
    def embed(self, texts: List[str]) -> List[List[float]]:
        """
        Generate embeddings for a list of texts.
        
        Args:
            texts: List of text strings to embed
            
        Returns:
            List of embedding vectors (one per input text)
            
        Raises:
            Exception: If embedding generation fails
        """
        pass
    
    def embed_single(self, text: str) -> List[float]:
        """
        Generate embedding for a single text string.
        
        Args:
            text: Text string to embed
            
        Returns:
            Embedding vector for the input text
        """
        return self.embed([text])[0]
