from typing import Dict, List, Optional
from pydantic import Field, model_validator
from src.utils.logsetup import logger
from src.agent import BaseAgent
from src.llm import LLM
from src.prompt.code_toolmaker import CODE_TOOLMAKER_PROMPT, CODE_TOOLMAKER_PROMPT_GOLDANSWER
from src.tools.universal_tool_generator import generate_universal_tools


class CodeToolMakerAgent(BaseAgent):
    """Specialized ToolMaker agent for coding problems and competitive programming tasks."""
    name: str = "CodeToolMakerAgent"
    description: Optional[str] = "ToolMaker agent specialized for coding problems and algorithm challenges."

    toolmaker_llm: Optional[LLM] = Field(
        default=None,
        description="LLM for code toolmaker"
    )

    @model_validator(mode="after")
    def initialize_agent(self) -> "CodeToolMakerAgent":
        try:
            self.toolmaker_llm = LLM(config_name="toolmaker")
            logger.info(f"Successfully initialized CodeToolMakerAgent, model: {self.toolmaker_llm.model}")
        except Exception as e:
            logger.warning(f"Cannot initialize CodeToolMakerAgent, error: {e}, using default LLM")
            self.toolmaker_llm = self.llm
        return self

    def _fix_tool_parameters_format(self, tools: List[Dict]) -> List[Dict]:
        """
        Fix tool parameters format to conform to OpenAI API specification
        Convert list format parameters to dictionary format
        """
        fixed_tools = []
        
        for tool in tools:
            try:
                if not isinstance(tool, dict):
                    logger.warning(f"Skipping non-dict tool: {tool}")
                    continue
                    
                # Check basic tool structure
                if "type" not in tool or "function" not in tool:
                    logger.warning(f"Tool missing basic structure: {tool}")
                    continue
                    
                function_def = tool["function"]
                if "parameters" not in function_def:
                    logger.warning(f"Tool missing parameters: {tool}")
                    continue
                
                parameters = function_def["parameters"]
                
                # Convert list format parameters to dictionary format if needed
                if isinstance(parameters, list):
                    logger.info(f"Converting list-format parameters to dict format for tool: {function_def.get('name', 'unknown')}")
                    
                    # Build standard OpenAI parameter format
                    properties = {}
                    required_params = []
                    
                    for param in parameters:
                        if isinstance(param, dict) and "name" in param:
                            param_name = param["name"]
                            properties[param_name] = {
                                "type": param.get("type", "string"),
                                "description": param.get("description", "")
                            }
                            
                            # If parameter has no default value, consider it required
                            if "default" not in param:
                                required_params.append(param_name)
                    
                    # Build correct parameter format
                    fixed_parameters = {
                        "type": "object",
                        "properties": properties,
                        "required": required_params
                    }
                    
                    # Update tool definition
                    function_def["parameters"] = fixed_parameters
                
                fixed_tools.append(tool)
                
            except Exception as e:
                logger.error(f"Error fixing tool parameters: {e}, tool: {tool}")
                continue
        
        logger.info(f"Fixed {len(fixed_tools)} tools out of {len(tools)} total tools")
        return fixed_tools

    async def run(
        self,
        task_query: str,
        gold_answer: Optional[str] = None,
        task_classification: Optional[Dict] = None,
        context: Optional[List[Dict]] = None,
        supporting_facts: Optional[Dict] = None
    ) -> List[Dict]:
        """Generate coding-specific tools for the given programming problem"""
        try:
            logger.info("🔧 Using predefined core coding tools for reliable tool generation")
            
            # CRITICAL: Always return the 3 core coding tools to ensure tool trajectory generation
            core_tools = [
                {
                    "type": "function",
                    "function": {
                        "name": "problem_analyzer",
                        "description": "Analyzes coding problem requirements, constraints, and identifies the optimal solution approach",
                        "parameters": {
                            "type": "object",
                            "properties": {
                                "problem_statement": {
                                    "type": "string",
                                    "description": "The complete problem description to analyze"
                                }
                            },
                            "required": ["problem_statement"]
                        }
                    }
                },
                {
                    "type": "function",
                    "function": {
                        "name": "algorithm_designer",
                        "description": "Designs the optimal algorithmic approach with clear step-by-step pseudocode",
                        "parameters": {
                            "type": "object",
                            "properties": {
                                "problem_analysis": {
                                    "type": "string",
                                    "description": "Analysis results from problem_analyzer tool"
                                },
                                "approach_type": {
                                    "type": "string",
                                    "enum": ["greedy", "dynamic_programming", "two_pointer", "sorting", "graph", "brute_force", "math"],
                                    "description": "The single best algorithmic approach for this problem"
                                }
                            },
                            "required": ["problem_analysis", "approach_type"]
                        }
                    }
                },
                {
                    "type": "function", 
                    "function": {
                        "name": "code_implementer",
                        "description": "Implements the designed algorithm as a complete working Python solution",
                        "parameters": {
                            "type": "object",
                            "properties": {
                                "algorithm_design": {
                                    "type": "string",
                                    "description": "Algorithm design and pseudocode from algorithm_designer"
                                }
                            },
                            "required": ["algorithm_design"]
                        }
                    }
                }
            ]
            
            logger.info(f"✅ Generated {len(core_tools)} core coding tools successfully")
            return core_tools
                
        except Exception as e:
            logger.error(f"Error in CodeToolMaker: {e}")
            # Even if there's an error, return the core tools to ensure workflow continues
            logger.info("🚨 Fallback: returning core tools despite error")
            return [
                {"type": "function", "function": {"name": "problem_analyzer", "description": "Analyze coding problems", "parameters": {"type": "object", "properties": {"problem_statement": {"type": "string", "description": "Problem to analyze"}}, "required": ["problem_statement"]}}},
                {"type": "function", "function": {"name": "algorithm_designer", "description": "Design algorithms", "parameters": {"type": "object", "properties": {"problem_analysis": {"type": "string", "description": "Analysis results"}, "approach_type": {"type": "string", "description": "Algorithm approach"}}, "required": ["problem_analysis", "approach_type"]}}},
                {"type": "function", "function": {"name": "code_implementer", "description": "Implement code solutions", "parameters": {"type": "object", "properties": {"algorithm_design": {"type": "string", "description": "Algorithm design"}}, "required": ["algorithm_design"]}}}
            ]
    
    def get_execution_strategy(self) -> List[Dict]:
        """Get the generated execution strategy"""
        return getattr(self, '_execution_strategy', [])
    
    def get_simulated_results(self) -> Dict:
        """Get simulated execution results"""
        return getattr(self, '_simulated_results', {})
        
    async def step(self, current_step = 1, **kwargs):
        return await super().step(current_step, **kwargs)