from typing import List, Dict, Any, Optional
from agent.base import BaseAgent
from prompt.critic import (
    ACTOR_FEEDBACK_PROMPT,
    RESULT_REVIEW_PROMPT,
    REFINE_PLAN_PROMPT,
    REPLAN_ON_FAILURE_PROMPT
)
import json
import re
import ast
from schema import Subtask, Context
import logger
import utils
from utils import parse_res
from prompt.task_types import get_all_task_types_in_current_level

class CriticAgent(BaseAgent):
    name: str = "CriticAgent"
    description: str = (
        "负责审查任务执行结果、分析失败原因、优化后续计划的 Agent"
    )
        
    def act(self, *args, **kwargs):
        return super().act(*args, **kwargs)

    def review_result(self, task: Subtask, execution_result: str) -> Dict[str, Any]:
        """
        Review the execution result of a task and provide feedback or suggestions.
        Migrated from ActorAgent.
        """
        result_preview = execution_result
        if len(execution_result) > 1600:
            result_preview = execution_result[:800] + "\n...\n" + execution_result[-800:]

        prompt = RESULT_REVIEW_PROMPT.format(
            instruction=task.instruction,
            result=result_preview,
        )
        
        response = self.llm.ask(messages=[{"role": "user", "content": prompt}])
        response_text = parse_res(response)

        try:
            return self._extract_json(response_text)
        except Exception as e:
            logger.error(f"Failed to parse review result: {e}")
            return {
                "success": "True" in response_text or "成功" in response_text,
                "reason": response_text
            }

    def analyze_result(self, task: Subtask,  execution_result: str) -> Dict[str, Any]:
        """
        Analyze the execution result and extract useful knowledge for future tasks.
        Migrated from ActorAgent.
        """
        result_preview = execution_result
        if len(execution_result) > 1200:
            result_preview = execution_result[:600] + "\n...\n" + execution_result[-600:]

        prompt = ACTOR_FEEDBACK_PROMPT.format(
            instruction=task.instruction,
            result=result_preview,
        )
        
        response = self.llm.ask(messages=[{"role": "user", "content": prompt}])
        response_text = parse_res(response)

        try:
            return self._extract_json(response_text)
        except Exception as e:
            logger.warning(f"【结果分析失败】{e}")
            return {}

    def refine_plan(
        self, 
        current_task: Subtask, 
        execution_summary: str, 
        actor_suggestions: str, 
        new_knowledge: str, 
        pending_tasks: List[Subtask],
        human_suggestion: str = "",
        refine_level: str = "子任务层",
        finished_task_count: int = 0,
    ) -> Dict[str, Any]:
        """
        Refine the remaining plan based on the execution result and feedback.
        Migrated from PlannerAgent.
        """
        pending_tasks_desc = json.dumps([t.to_simple_dict() for t in pending_tasks], ensure_ascii=False, indent=2)
        
        prompt = REFINE_PLAN_PROMPT.format(
            current_task_desc=f"{current_task.task_id}: {current_task.instruction}",
            execution_summary=execution_summary,
            actor_suggestions=actor_suggestions,
            new_knowledge=new_knowledge,
            pending_tasks_desc=pending_tasks_desc,
            task_types=get_all_task_types_in_current_level(task_type=current_task.task_type,is_parent=(refine_level=="父任务层")),
            refine_level=refine_level,
        )

        if human_suggestion:
            prompt =prompt + f"【人类建议】{human_suggestion}\n" 
        
        response = self.llm.ask(messages=[{"role": "user", "content": prompt}])
        response_text = parse_res(response)

        try:
            return self._extract_json(response_text)
        except Exception as e:
            logger.error(f"Failed to parse refined plan: {e}")
            return {"refined": False, "reason": f"Parsing error: {str(e)}", "pending_tasks": []}

    def replan_on_failure(
        self, 
        failed_task: Subtask, 
        error_message: str, 
        attempts: int, 
        pending_tasks: List[Subtask],
        human_suggestion: str = "",
    ) -> Dict[str, Any]:
        """
        Formulate a recovery plan when a task fails.
        Migrated from PlannerAgent.
        """
        pending_tasks_desc = json.dumps([t.to_simple_dict() for t in pending_tasks], ensure_ascii=False, indent=2)
        if failed_task.parent_task_id:
            is_parent = False
        else:
            is_parent = True
        prompt = REPLAN_ON_FAILURE_PROMPT.format(
            failed_task_desc=f"{failed_task.task_id}: {failed_task.instruction}",
            error_message=error_message,
            attempts=attempts,
            pending_tasks_desc=pending_tasks_desc,
            task_types=get_all_task_types_in_current_level(task_type=failed_task.task_type,is_parent=is_parent)
        )

        if human_suggestion:
            prompt = f"【人类建议】{human_suggestion}\n" + prompt
        
        response = self.llm.ask(messages=[{"role": "system", "content": prompt}])
        response_text = parse_res(response)

        try:
            return self._extract_json(response_text)
        except Exception as e:
            logger.error(f"Failed to parse failure recovery plan: {e}")
            return {
                "action": "RETRY", 
                "reason": f"Parsing error: {str(e)}", 
                "new_task": {
                    "task_id": failed_task.task_id,
                    "task_type": failed_task.task_type,
                    "instruction": failed_task.instruction,
                    "dependent_task_ids": failed_task.dependent_task_ids
                }
            }

    def _extract_json(self, text: str) -> Any:
        # 1. Try to find JSON block in markdown ```json ... ```
        json_block_match = re.search(r"```json\s*(\{.*?\})\s*```", text, re.DOTALL)
        if json_block_match:
            try:
                return json.loads(json_block_match.group(1))
            except:
                pass

        # 2. Try to find the first { and last }
        json_match = re.search(r"(\{.*\})", text, re.DOTALL)
        if json_match:
            json_str = json_match.group(1)
            try:
                return json.loads(json_str)
            except:
                # 3. If json.loads fails, try ast.literal_eval (handles single quotes, Python dicts)
                try:
                    return ast.literal_eval(json_str)
                except:
                    pass

        # 4. Fallback: try to parse the whole text as json
        return json.loads(text)
