
import json
import os
from datetime import datetime
from typing import Any

# Ensure numpy scalars can be serialized
try:
    import numpy as np  # type: ignore
except ImportError:  # numpy might not be installed in minimal environments
    np = None  # type: ignore


class AgentDebugLogger:
    """Enhanced logger for debugging agent execution"""
    
    def __init__(self, log_dir="logs/agent_debug"):
        self.log_dir = log_dir
        os.makedirs(log_dir, exist_ok=True)
        self.session_id = datetime.now().strftime('%Y%m%d_%H%M%S')
    
    def log_tool_result(self, agent_name, task_id, tool_name, params, result):
        """Log tool execution results"""
        log_entry = {
            "timestamp": datetime.now().isoformat(),
            "agent": agent_name,
            "task_id": task_id,
            "tool": tool_name,
            "params": params,
            "result": self._serialize_result(result),
            "success": result.get("success", False) if isinstance(result, dict) else True
        }
        
        # Write to agent-specific log file
        filename = f"{self.log_dir}/{agent_name}_{self.session_id}.jsonl"
        with open(filename, 'a') as f:
            f.write(json.dumps(log_entry) + '\n')
        
        # Also log to console in debug mode
        print(f"\n[DEBUG] {agent_name} - {task_id} - {tool_name}")
        print(f"  Params: {json.dumps(params, indent=2)}")
        print(f"  Result: {json.dumps(self._serialize_result(result), indent=2)}")
    
    def log_agent_state(self, agent_name, state_type, data):
        """Log agent state changes"""
        log_entry = {
            "timestamp": datetime.now().isoformat(),
            "agent": agent_name,
            "state_type": state_type,
            "data": self._serialize_result(data)
        }
        
        filename = f"{self.log_dir}/{agent_name}_{self.session_id}_state.jsonl"
        with open(filename, 'a') as f:
            f.write(json.dumps(log_entry) + '\n')
    
    def _serialize_result(self, obj: Any):
        """Recursively convert results into JSON-serializable structures.

        Handles common non-serializable types such as numpy scalar values by
        converting them to their native Python counterparts. This avoids JSON
        dumping errors like "Object of type float32 is not JSON serializable"
        without touching the core tool implementations.
        """
        # Dict → recurse on values
        if isinstance(obj, dict):
            return {k: self._serialize_result(v) for k, v in obj.items()}

        # List / Tuple → recurse on items
        if isinstance(obj, list):
            return [self._serialize_result(item) for item in obj]
        if isinstance(obj, tuple):
            return [self._serialize_result(item) for item in obj]

        # numpy scalar types → convert to python scalar
        if np is not None and isinstance(obj, (np.generic,)):
            return obj.item()

        # Fallback: if object has __dict__ represent as string
        if hasattr(obj, "__dict__"):
            return str(obj)

        # Primitive types or already serializable
            return obj
    
    def save_final_results(self, orchestrator_result):
        """Save final orchestrator results"""
        filename = f"{self.log_dir}/final_results_{self.session_id}.json"
        
        results = {
            "timestamp": datetime.now().isoformat(),
            "query": orchestrator_result.query,
            "execution_time": orchestrator_result.execution_time,
            "activated_agents": orchestrator_result.activated_agents,
            "agent_results": {},
            "synthesis_result": None
        }
        
        # Extract agent results
        for agent_name, analysis in orchestrator_result.agent_results.items():
            if analysis:
                results["agent_results"][agent_name] = {
                    "findings": analysis.findings if hasattr(analysis, 'findings') else [],
                    "confidence_level": analysis.confidence_level if hasattr(analysis, 'confidence_level') else "unknown",
                    "needs_human_review": analysis.needs_human_review if hasattr(analysis, 'needs_human_review') else False
                }
        
        # Extract synthesis
        if orchestrator_result.synthesis_result:
            results["synthesis_result"] = {
                "answer": orchestrator_result.synthesis_result.answer if hasattr(orchestrator_result.synthesis_result, 'answer') else "",
                "confidence": orchestrator_result.synthesis_result.confidence if hasattr(orchestrator_result.synthesis_result, 'confidence') else 0
            }
        
        with open(filename, 'w') as f:
            json.dump(results, f, indent=2)
        
        print(f"\n[DEBUG] Final results saved to: {filename}")
        return filename


# Global logger instance
debug_logger = AgentDebugLogger()
