import os
import json
from typing import Dict, List, Any, Optional
from tavily import TavilyClient
from .base import Tool
from ..logging import logger


class WebSearchTavilyTool(Tool):
    NAME = "web_search"
    DESCRIPTION = "Search the web for information using Tavily API"
    
    def __init__(self):
        super().__init__()
        
        # Get API key from environment variable or keys.cfg
        api_key = os.environ.get("TAVILY_API_KEY")
        if not api_key:
            # Try to read from keys.cfg file
            try:
                keys_path = os.path.join(os.path.dirname(os.path.dirname(os.path.dirname(__file__))), "keys.cfg")
                if os.path.exists(keys_path):
                    with open(keys_path, 'r') as f:
                        for line in f:
                            if line.startswith("TAVILY="):
                                api_key = line.strip().split("=", 1)[1]
                                break
            except Exception as e:
                logger.debug_message(f"Failed to read keys.cfg: {str(e)}")
        
        if not api_key:
            # Use the provided API key as fallback
            api_key = "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"
        
        self.client = TavilyClient(api_key=api_key)
        
        # Tavily pricing (approximate)
        self.cost_per_search = 0.001  # $0.001 per search

    def call(self, query=None, max_results=3, search_type="general", include_images=False):
        if query is None:
            return {"error": "No search query provided"}

        # Limit max_results to prevent abuse
        max_results = min(max(int(max_results), 1), 5)
        
        try:
            print(f"Searching with Tavily for: {query}")
            
            # Search using Tavily
            response = self.client.search(
                query=query,
                search_depth="basic",
                max_results=max_results,
                include_domains=[],
                exclude_domains=[],
                include_answer=False,
                include_raw_content=False,
                include_images=include_images
            )
            
            # Calculate cost
            total_cost = self.cost_per_search
            
            # Format results
            formatted_results = []
            if 'results' in response:
                for result in response['results']:
                    formatted_results.append({
                        'title': result.get('title', 'No title'),
                        'url': result.get('url', ''),
                        'snippet': result.get('content', 'No description available'),
                        'type': 'search_result'
                    })
            
            print(f"Tavily search cost: ${total_cost:.6f}")
            
            return {
                "query": query,
                "total_results": len(formatted_results),
                "results": formatted_results,
                "search_type": search_type,
                "include_images": include_images,
                "search_engine": "tavily",
                "search_cost": total_cost
            }
            
        except Exception as e:
            logger.debug_message(f"Tavily search error: {str(e)}")
            return {
                "query": query,
                "total_results": 1,
                "results": [{
                    'title': f"Search Error for: {query}",
                    'url': "Tavily Search",
                    'snippet': f"Tavily search failed: {str(e)}. Please try again or use a different search approach.",
                    'type': 'search_result'
                }],
                "search_type": search_type,
                "include_images": include_images,
                "search_engine": "tavily",
                "search_cost": 0.0
            }

    def print_tool_call(self, tool_call):
        args = tool_call.parsed_arguments
        query = args.get('query', 'unknown query') if args else 'unknown query'
        logger.assistant_action(f"**{self.NAME}**: searching the web for '{query}' using Tavily search")

    def print_result(self, tool_result):
        if "error" in tool_result.result:
            logger.print(f"[bold]{self.NAME}[/bold]: [red]{tool_result.result['error']}[/red]", markup=True)
        else:
            result = tool_result.result
            total_results = result.get('total_results', 0)
            query = result.get('query', 'unknown')
            results = result.get('results', [])
            
            formatted_results = self._format_results(results)
            logger.observation_message(
                f"**{self.NAME}**: Found {total_results} results for '{query}' using Tavily search\n\n" +
                (formatted_results if formatted_results else "No results found.")
            )

    def _format_results(self, results):
        if not results or not isinstance(results, list):
            return "No results found."
        
        formatted = ""
        for i, result in enumerate(results, 1):
            if not isinstance(result, dict):
                continue
                
            title = result.get('title', 'No title')
            url = result.get('url', '')
            snippet = result.get('snippet', 'No description available')
            
            # Clean up title (remove markdown formatting)
            clean_title = title.replace('**', '').replace('###', '').replace('##', '').strip()
            
            formatted += f"{i}. **{clean_title}**\n"
            if url and url != "Tavily Search":
                formatted += f"   URL: {url}\n"
            
            # Truncate snippet if too long
            clean_snippet = snippet.strip()
            if len(clean_snippet) > 200:
                clean_snippet = clean_snippet[:200] + "..."
            formatted += f"   {clean_snippet}\n\n"
        
        return formatted if formatted else "No results found."
