"""
Abstraction agent for knowledge update and strategy refinement
"""

import json
from typing import Dict
from .base_agent import BaseAgent


class AbstractionAgent(BaseAgent):
    """Agent responsible for abstracting knowledge and strategies from examples"""
    
    def execute(self, task_description: str, latest_knowledge: str = "", 
                latest_example: str = "") -> Dict:
        """
        Perform knowledge abstraction from examples and feedback
        
        Args:
            task_description: Description of the task to be performed
            latest_knowledge: Latest knowledge entry (can be empty for first round)
            latest_example: Latest example entry (can be empty)
            
        Returns:
            Knowledge dictionary with strategies, principles, and need_change flag
        """
        self.log_info("Starting knowledge abstraction")
        
        # Format user message for abstraction step (same as meta_reasoning_agent)
        user_message = self.format_user_message(
            task_description=task_description,
            sample_question=latest_example,
            latest_knowledge=self._clean_knowledge_for_input(latest_knowledge)
        )
        
        # Call LLM for abstraction
        response = self.call_llm(user_message)
        
        # Parse JSON response (same logic as meta_reasoning_agent)
        try:
            abstraction_result = json.loads(response.strip())
            
            # Safety check: if no existing knowledge but LLM says need_change=False, override it
            if not latest_knowledge and not abstraction_result.get('need_change', True):
                self.log_info("Overriding need_change=False because no existing knowledge provided")
                abstraction_result['need_change'] = True
                abstraction_result['change_rationale'] = "First iteration - generating initial knowledge"
            
            # Check if LLM decided no changes are needed
            if not abstraction_result.get('need_change', True):
                self.log_info("LLM determined no changes needed, returning LLM response")
                # Return the actual LLM response, not the latest knowledge
                return abstraction_result
            
            self.log_info("Abstraction step completed successfully with changes")
            return abstraction_result
            
        except json.JSONDecodeError as e:
            self.log_info(f"Failed to parse abstraction JSON: {e}")
            # Fallback result matching the expected output format
            return {"need_change": False}
    
    def _clean_knowledge_for_input(self, latest_knowledge: str) -> str:
        """Remove need_change and other output-only fields from knowledge before sending to LLM"""
        if not latest_knowledge:
            return latest_knowledge
        
        try:
            # Try to parse as JSON
            if isinstance(latest_knowledge, str):
                knowledge_dict = json.loads(latest_knowledge)
            else:
                knowledge_dict = latest_knowledge
            
            # Remove output-only fields
            cleaned_dict = {k: v for k, v in knowledge_dict.items() 
                          if k not in ['need_change', 'change_rationale']}
            
            # Convert back to string
            return json.dumps(cleaned_dict, ensure_ascii=False)
            
        except (json.JSONDecodeError, TypeError):
            # If not JSON, return as is
            return latest_knowledge
