from typing import Dict, List, Optional, Any
import importlib
import os
import logging
from pathlib import Path

from .bash import Bash
from .python_execute import PythonExecute
from .web_search import WebSearch
from .simple_intelligent_search import SimpleIntelligentWebSearch
from .answer_summarizer import AnswerSummarizer
from src.config import config

logger = logging.getLogger(__name__)


class ToolManager:
    """Tool manager for handling real tools and intelligent tools with automatic fallback strategy"""
    
    def __init__(self):
        self.real_tools = {}
        self.intelligent_tools = {}
        self.search_config = config.search
        self.load_tools()
    
    def load_tools(self):
        """Load all tools, including intelligent search tools"""
        # Load basic real tools
        self.real_tools = {
            "bash": Bash(),
            "python_execute": PythonExecute(),
            "web_search": WebSearch(),
            "answer_summarizer": AnswerSummarizer(),
        }
        
        # Load intelligent tools (with automatic fallback functionality)
        self.intelligent_tools = {
            "web_search": SimpleIntelligentWebSearch(),
        }
        
        logger.info(f"Loaded {len(self.real_tools)} real tools and {len(self.intelligent_tools)} intelligent tools")
    
    def get_optimal_tool(self, tool_name: str, query: str = "", query_analysis: Optional[Dict] = None):
        """Get optimal tool instance (automatically choose real or intelligent version)"""
        if tool_name == "web_search":
            # Automatically decide which search tool to use
            search_mode = self._get_search_mode(query, query_analysis)
            
            if search_mode == "simulation":
                logger.debug("Auto-selected intelligent search (simulation mode)")
                return self.intelligent_tools["web_search"]
            elif search_mode == "real":
                logger.debug("Auto-selected real search")
                return self.real_tools["web_search"]
            else:  # intelligent/auto mode
                logger.debug("Auto-selected intelligent search (auto mode)")
                return self.intelligent_tools["web_search"]
        
        # For other tools, return real tools
        return self.real_tools.get(tool_name)
    
    def _get_search_mode(self, query: str, query_analysis: Optional[Dict] = None) -> str:
        """Automatically determine search mode"""
        
        if self.search_config.search_mode == "simulation":
            return "simulation"
        elif self.search_config.search_mode == "real":
            return "real"
        else:  # auto mode
            # Simple automatic detection logic
            query_lower = query.lower()
            
            # Use simulation for entertainment-related queries
            if self.search_config.entertainment_auto_simulation and any(
                term in query_lower for term in ["tv", "show", "celebrity", "network", "entertainment"]
            ):
                return "simulation"
            
            # Default to intelligent mode (tries real search, falls back on failure)
            return "intelligent"
    
    def get_tool_definitions(self, include_intelligent: bool = True) -> List[Dict]:
        """Get tool definitions (prioritize intelligent versions)"""
        definitions = []
        processed_tools = set()
        
        # Prioritize adding intelligent tools
        if include_intelligent:
            for tool_name, tool in self.intelligent_tools.items():
                definitions.append(tool.to_param())
                processed_tools.add(tool_name)
        
        # Add other real tools
        for tool_name, tool in self.real_tools.items():
            if tool_name not in processed_tools:
                definitions.append(tool.to_param())
        
        return definitions
    
    async def execute_tool(self, tool_name: str, query: str = "", query_analysis: Optional[Dict] = None, **kwargs) -> Any:
        """Execute tool (automatically select optimal version)"""
        tool = self.get_optimal_tool(tool_name, query, query_analysis)
        
        if tool is None:
            raise ValueError(f"Tool '{tool_name}' not found")
        
        try:
            result = await tool.execute(query=query, query_analysis=query_analysis, **kwargs)
            return result
            
        except Exception as e:
            logger.warning(f"Tool {tool_name} failed: {e}")
            raise
    
    # Maintain backward compatibility methods
    def get_real_tool_definitions(self) -> List[Dict]:
        """Get all real tool definitions (backward compatibility)"""
        return self.get_tool_definitions(include_intelligent=False)
    
    def has_real_tool(self, tool_name: str) -> bool:
        """Check if a tool with the specified name exists"""
        return tool_name in self.real_tools or tool_name in self.intelligent_tools
    
    def get_real_tool(self, tool_name: str):
        """Get tool instance (backward compatibility)"""
        return self.get_optimal_tool(tool_name)
    
    async def execute_real_tool(self, tool_name: str, **kwargs) -> Any:
        """Execute tool (backward compatibility)"""
        return await self.execute_tool(tool_name, **kwargs)
    
    def get_tool_names(self) -> List[str]:
        """Get all tool names"""
        all_tools = set(self.real_tools.keys())
        all_tools.update(self.intelligent_tools.keys())
        return list(all_tools)
    
    def filter_needed_tools(self, requested_tools: List[str]) -> tuple[List[str], List[str]]:
        """
        Filter requested tools, separating existing tools and tools that need to be generated for simulation

        Returns:
            tuple: (existing_tools, tools_to_generate)
        """
        existing_tools = []
        tools_to_generate = []
        
        for tool_name in requested_tools:
            if self.has_real_tool(tool_name):
                existing_tools.append(tool_name)
            else:
                tools_to_generate.append(tool_name)
        
        return existing_tools, tools_to_generate


# Global tool manager instance
tool_manager = ToolManager()