"""
CodeAutoAgent - Specialized agent for code generation tasks
Uses 3 core tools: problem_analyzer, algorithm_designer, code_implementer
"""
import json
import re
import traceback
from typing import Optional, Dict, List
from pydantic import Field, model_validator
from src.utils.logsetup import logger
from src.agent import BaseAgent
from src.llm import LLM
from src.schema.message import Message, ToolCall, AgentState
from src.tools.problem_analyzer import ProblemAnalyzer
from src.tools.algorithm_designer import AlgorithmDesigner
from src.tools.code_implementer import CodeImplementer


class CodeAutoAgent(BaseAgent):
    """Specialized AutoAgent for code generation tasks using 3 core programming tools."""
    
    name: str = "CodeAutoAgent"
    description: str = "Specialized agent for analyzing, designing, and implementing code solutions"
    
    # Core coding tools
    tools: Optional[List[Dict]] = Field(
        default_factory=lambda: [
            ProblemAnalyzer().to_param(),
            AlgorithmDesigner().to_param(), 
            CodeImplementer().to_param()
        ],
        description="Core coding tools for CodeAutoAgent"
    )
    
    tool_map: Optional[Dict] = Field(
        default_factory=lambda: {
            "problem_analyzer": ProblemAnalyzer(),
            "algorithm_designer": AlgorithmDesigner(),
            "code_implementer": CodeImplementer()
        },
        description="Tool instances map"
    )
    
    system_prompt: str = Field(default="""You are a specialized Python coding agent that solves programming problems systematically.

Your workflow:
1. Analyze the problem using problem_analyzer to understand requirements and constraints
2. Design the optimal algorithm using algorithm_designer with clear steps and complexity analysis  
3. Implement the complete solution using code_implementer with working Python code

CRITICAL REQUIREMENTS:
- Always provide complete, executable Python programs (not just function definitions)
- Include the main function/algorithm implementation
- Add test cases or example executions that demonstrate the solution
- Include print statements or return statements that produce visible output
- Use proper Python syntax and conventions
- Handle edge cases in your Python implementation
- Ensure code is production-ready and well-structured
- The final code must be runnable in a sandbox environment and produce output

EXECUTABLE FORMAT EXAMPLE:
```python
def solution(nums, target):
    # Implementation here
    return result

# Test the solution
if __name__ == "__main__":
    # Test cases with print statements for visible output
    test_cases = [
        ([2, 7, 11, 15], 9),
        ([3, 2, 4], 6)
    ]
    
    for i, (nums, target) in enumerate(test_cases):
        result = solution(nums, target)
        print(f"Test {i+1}: nums={nums}, target={target}, result={result}")
```

Follow the 3-step process and provide detailed reasoning for each step.
End with complete, executable Python code that produces visible output when run.""")
    
    toolactor: Optional[BaseAgent] = None
    gold_answer: Optional[str] = None  # Reference solution for guidance
    current_code_solution: Optional[str] = None  # Track extracted code
    
    @model_validator(mode="after") 
    def initialize_agent(self) -> "CodeAutoAgent":
        """Initialize with code-specific LLM configuration."""
        try:
            # Try to use code-specific config, fallback to default
            self.llm = LLM(config_name="code_autoagent")
            logger.info(f"Successfully initialized CodeAutoAgent, model: {self.llm.model}")
        except Exception as e:
            logger.warning(f"Cannot initialize CodeAutoAgent with code config, using default: {e}")
            self.llm = LLM()
        return self
    
    def set_toolactor(self, toolactor: Optional[BaseAgent]):
        """Set the toolactor for tool simulation."""
        logger.info(f"🤖 Set toolactor: {toolactor}")
        self.toolactor = toolactor
    
    async def step(self, current_step=1, **kwargs):
        """Execute coding workflow step: think -> analyze -> design -> implement."""
        try:
            # Think about the problem and decide next action
            await self.think()
            
            # Check if we have a final code solution
            if self.current_code_solution and self.current_step >= 3:
                logger.info("✅ Code solution found, finishing task")
                self.state = AgentState.FINISHED
                return "Code implementation completed"
            
            # Continue with tool execution if needed
            if self.state != AgentState.FINISHED:
                return await self.act()
                
            return "Step completed"
            
        except Exception as e:
            logger.error(f"Error in step {current_step}: {e}")
            traceback.print_exc()
            return f"Step failed: {e}"
    
    async def think(self):
        """Generate reasoning and decide on tool usage."""
        try:
            # Prepare messages for LLM
            messages = self.memory.messages.copy()
            
            # Call LLM with available tools
            response = await self.llm.ask_tool(
                messages=messages,
                tools=self.tools,
                temperature=0.0
            )
            
            # Parse response and extract tool calls
            self.tool_calls = []
            if hasattr(response, 'tool_calls') and response.tool_calls:
                self.tool_calls = response.tool_calls
                
            # Log the thinking process
            content = getattr(response, 'content', '') or ''
            if content.strip():
                logger.info(f"✨ CodeAutoAgent's coding thoughts: {content}")
                # Create message without tool_calls first
                assistant_msg = Message.assistant_message(content)
                # Manually set tool_calls after creation if they exist
                if self.tool_calls:
                    assistant_msg.tool_calls = self.tool_calls
                self.memory.add_message(assistant_msg)
                
                # Try to extract code from the response
                self._extract_and_store_code(content)
            
            # Log tool calls being prepared
            if self.tool_calls:
                tool_names = [tc.function.name for tc in self.tool_calls]
                tool_args = [tc.function.arguments for tc in self.tool_calls]
                logger.info(f"🧰 Coding tools being prepared: {tool_names}")
                logger.info(f"🔧 Tool arguments: {tool_args}")
                
        except Exception as e:
            logger.error(f"Error in think(): {e}")
            traceback.print_exc()
    
    async def act(self):
        """Execute the planned tool calls."""
        if not self.tool_calls:
            logger.warning("No tool calls to execute")
            return "No actions taken"
        
        results = []
        for idx, tool_call in enumerate(self.tool_calls):
            tool_name = tool_call.function.name
            tool_id = tool_call.id
            
            try:
                result = await self.execute_tool(tool_call)
                
                # Store the tool result
                tool_message = Message.tool_message(result, name=tool_name, tool_call_id=tool_id)
                self.memory.add_message(tool_message)
                
                # Try to extract code from tool result
                self._extract_and_store_code(result)
                
                results.append(f"CODING TOOL {idx + 1} RESULT:\n{result}")
                logger.info(f"CODING TOOL {idx + 1} RESULT:\n{result}")
                
            except Exception as e:
                error_message = f"Error executing coding tool {tool_name}: {str(e)}"
                logger.error(error_message)
                results.append(f"CODING TOOL {idx + 1} ERROR: {error_message}")
                
                # Add error as tool message
                tool_message = Message.tool_message(f"Error: {error_message}", name=tool_name, tool_call_id=tool_id)
                self.memory.add_message(tool_message)
        
        # Clear tool calls after execution
        self.tool_calls = []
        
        return "\n\n".join(results)
    
    def _extract_and_store_code(self, content: str):
        """Extract code from content and store for final output."""
        try:
            # Look for code in various formats
            code_patterns = [
                r'<code>(.*?)</code>',
                r'```python\n(.*?)```', 
                r'```\n(.*?)```',
                r'"code":\s*"([^"]+)"',
                r'def\s+\w+.*?(?=\n\S|\Z)'
            ]
            
            for pattern in code_patterns:
                matches = re.findall(pattern, content, re.DOTALL)
                if matches:
                    # Take the last/longest match
                    code = matches[-1].strip()
                    if len(code) > 20:  # Reasonable code length
                        self.current_code_solution = code
                        logger.info(f"📝 Extracted code solution ({len(code)} characters)")
                        break
                        
        except Exception as e:
            logger.warning(f"Could not extract code: {e}")
    
    async def execute_tool(self, tool_call: ToolCall) -> str:
        """Execute a coding tool call using toolactor simulation."""
        tool_name = tool_call.function.name
        
        # Validate tool is allowed
        allowed_tools = ["problem_analyzer", "algorithm_designer", "code_implementer"]
        if tool_name not in allowed_tools:
            return f"❌ Tool '{tool_name}' not allowed for coding tasks. Use: {', '.join(allowed_tools)}"
        
        if not self.toolactor:
            return f"❌ ToolActor not configured for tool simulation"
        
        try:
            # Use toolactor to simulate the tool execution
            tool_function = self.tool_map.get(tool_name)
            
            # Get current reasoning context
            reasoning = ""
            if self.memory.messages:
                last_msg = self.memory.messages[-1] 
                if last_msg.role == "assistant":
                    reasoning = last_msg.content or ""
            
            # Call toolactor with gold answer as reference
            tool_observation = await self.toolactor.run(
                reasoning=reasoning,
                tool_call=tool_call,
                tool_function=tool_function,
                gold_answer=self.gold_answer
            )
            
            return f"Observed output of coding tool `{tool_name}`:\n{tool_observation}"
            
        except Exception as e:
            logger.error(f"⚠️ Error executing coding tool '{tool_name}': {e}")
            return f"Error: Tool execution failed: {e}"

    async def run(self, request: str = "", **kwargs) -> str:
        """Run the coding agent with forced code output."""
        results = await super().run(request, **kwargs)
        
        # Ensure we have final code output
        if self.current_step >= self.max_steps and self.state != AgentState.FINISHED:
            logger.warning("⚠️ Reached max steps without final code, forcing completion...")
            
            if self.current_code_solution:
                # Add final code message
                final_code_msg = Message.assistant_message(
                    f"Based on my systematic analysis and implementation:\n\n<code>\n{self.current_code_solution}\n</code>"
                )
                self.memory.add_message(final_code_msg)
                self.state = AgentState.FINISHED
                
        return results