from typing import Dict, List, Optional
from pydantic import BaseModel, Field


class SearchResult(BaseModel):
    """Represents a single search result returned by a search engine."""
    
    position: int = Field(description="Position in search results")
    url: str = Field(description="URL of the search result")
    title: str = Field(default="", description="Title of the search result")
    description: str = Field(
        default="", description="Description or snippet of the search result"
    )
    source: str = Field(description="The search engine that provided this result")
    raw_content: Optional[str] = Field(
        default=None, description="Raw content from the search result page if available"
    )

    def __str__(self) -> str:
        """String representation of a search result."""
        return f"{self.title} ({self.url})"


class SearchMetadata(BaseModel):
    """Metadata about the search operation."""
    
    total_results: int = Field(description="Total number of results found")
    language: str = Field(description="Language code used for the search")
    country: str = Field(description="Country code used for the search")
    
    # Extended metadata for intelligent search
    simulation_confidence: Optional[float] = Field(default=None, description="Confidence level for simulated results")
    search_mode: Optional[str] = Field(default=None, description="Search mode used (real/simulation/hybrid)")
    quality_analysis: Optional[Dict] = Field(default=None, description="Quality analysis of search results")
    real_results_count: Optional[int] = Field(default=None, description="Number of real results in hybrid mode")
    simulated_results_count: Optional[int] = Field(default=None, description="Number of simulated results")


class SearchResponse(BaseModel):
    """Structured response from the web search tool."""
    
    query: str = Field(description="The search query that was executed")
    results: List[SearchResult] = Field(
        default_factory=list, description="List of search results"
    )
    metadata: Optional[SearchMetadata] = Field(
        default=None, description="Metadata about the search"
    )
    status: str = Field(default="success", description="Status of the search operation")
    error: Optional[str] = Field(default=None, description="Error message if search failed")


class WebSearch(BaseModel):
    """Search the web for information using various search engines with realistic result generation."""
    
    name: str = "web_search"
    description: str = """Search the web for real-time information about any topic.
    This tool returns comprehensive, realistic search results with relevant information, authentic URLs, titles, and descriptions.
    Results are dynamically generated based on query analysis to provide domain-appropriate sources and content."""
    parameters: dict = {
        "type": "object",
        "properties": {
            "query": {
                "type": "string",
                "description": "(required) The search query to submit to the search engine.",
            },
            "num_results": {
                "type": "integer",
                "description": "(optional) The number of search results to return. Default is 5.",
                "default": 5,
            },
            "lang": {
                "type": "string",
                "description": "(optional) Language code for search results (default: en).",
                "default": "en",
            },
            "country": {
                "type": "string",
                "description": "(optional) Country code for search results (default: us).",
                "default": "us",
            },
            "fetch_content": {
                "type": "boolean",
                "description": "(optional) Whether to fetch full content from result pages. Default is false.",
                "default": False,
            },
            "query_analysis": {
                "type": "object", 
                "description": "(optional) Query analysis from TaskClassifier for enhanced result generation",
                "properties": {
                    "query_type": {"type": "string"},
                    "topic_domain": {"type": "string"},
                    "key_entities": {"type": "array", "items": {"type": "string"}},
                    "suggested_sources": {"type": "array", "items": {"type": "string"}},
                    "complexity_level": {"type": "string"}
                }
            },
            "gold_answer": {
                "type": "string",
                "description": "(optional) Gold answer to guide result generation (never revealed directly)",
                "default": None
            }
        },
        "required": ["query"],
    }

    async def execute(
        self,
        query: str,
        num_results: int = 5,
        lang: Optional[str] = None,
        country: Optional[str] = None,
        fetch_content: bool = False,
        query_analysis: Optional[Dict] = None,
        gold_answer: Optional[str] = None,
    ) -> SearchResponse:
        """
        Execute web search with enhanced result generation
        
        Args:
            query: The search query to submit to the search engine
            num_results: The number of search results to return (default: 5)
            lang: Language code for search results (default: en)
            country: Country code for search results (default: us)
            fetch_content: Whether to fetch content from result pages (default: False)
            query_analysis: Optional query analysis for enhanced results
            gold_answer: Optional gold answer for guidance

        Returns:
            A structured response containing realistic search results and metadata
        """
        # Use SearchResultGenerator if available, otherwise fallback to simple results
        try:
            from src.tools.search_result_generator import SearchResultGenerator
            
            if query_analysis:
                # Use advanced result generation
                generator = SearchResultGenerator()
                results = await generator.execute(
                    query=query,
                    query_analysis=query_analysis,
                    num_results=num_results,
                    gold_answer=gold_answer,
                    include_content=fetch_content
                )
            else:
                # Fallback to simple realistic results
                results = self._generate_simple_realistic_results(query, num_results, fetch_content, gold_answer)
                
        except ImportError:
            # Fallback if SearchResultGenerator not available
            results = self._generate_simple_realistic_results(query, num_results, fetch_content, gold_answer)
        
        # Create metadata
        metadata = SearchMetadata(
            total_results=len(results) * 1000,  # Simulate many total results
            language=lang or "en",
            country=country or "us"
        )
        
        return SearchResponse(
            query=query,
            results=results,
            metadata=metadata,
            status="success"
        )
    
    def _generate_simple_realistic_results(self, query: str, num_results: int, 
                                         fetch_content: bool, gold_answer: Optional[str]) -> List[SearchResult]:
        """Generate simple but realistic search results as fallback."""
        results = []
        query_lower = query.lower()
        
        # Common realistic domains
        domains = [
            "en.wikipedia.org",
            "www.britannica.com", 
            "www.biography.com",
            "www.imdb.com",
            "www.history.com"
        ]
        
        for i, domain in enumerate(domains[:num_results]):
            # Generate realistic URL based on domain
            if domain == "en.wikipedia.org":
                # Extract likely article title from query
                words = query.split()
                article_title = "_".join([w for w in words if w[0].isupper() or w.lower() in ['the', 'of', 'and']])[:50]
                url = f"https://en.wikipedia.org/wiki/{article_title or 'Search_results'}"
                title = f"{query} - Wikipedia"
                description = "Wikipedia article with comprehensive information, citations, and references."
                
            elif domain == "www.britannica.com":
                url = f"https://www.britannica.com/search?query={query.replace(' ', '+')}"
                title = f"{query} | Encyclopedia Britannica"  
                description = "Authoritative encyclopedia entry with verified facts and scholarly information."
                
            elif domain == "www.biography.com":
                url = f"https://www.biography.com/search?q={query.replace(' ', '+')}"
                title = f"{query} Biography - Life and Career"
                description = "Detailed biographical information including early life, achievements, and legacy."
                
            elif domain == "www.imdb.com":
                url = f"https://www.imdb.com/find?q={query.replace(' ', '+')}"
                title = f"{query} - IMDb"
                description = "Complete filmography, career information, and industry details."
                
            else:  # www.history.com
                url = f"https://www.history.com/search?q={query.replace(' ', '+')}"
                title = f"{query} - History"
                description = "Historical information, timeline, and contextual background."
            
            # Add subtle gold answer hints to descriptions if provided
            if gold_answer:
                description = self._add_gold_answer_hint(description, gold_answer)
            
            results.append(SearchResult(
                position=i + 1,
                url=url,
                title=title,
                description=description,
                source="google",
                raw_content=f"Detailed content about {query}..." if fetch_content else None
            ))
        
        return results
    
    def _add_gold_answer_hint(self, description: str, gold_answer: str) -> str:
        """Add subtle hints about gold answer to description."""
        if any(char.isdigit() for char in gold_answer):  # Contains dates/numbers
            return description + " Includes specific dates and numerical data."
        elif "," in gold_answer:  # Multiple items
            return description + " Contains multiple relevant facts and details."
        else:  # Simple answer
            return description + " Provides clear, factual information."
    
    def to_param(self) -> Dict:
        """Convert tool to function call format."""
        return {
            "type": "function",
            "function": {
                "name": self.name,
                "description": self.description,
                "parameters": self.parameters,
            },
        }