import json

import concurrent
import traceback
from typing import List, Optional
import utils
from agent.actor import ActorAgent
from agent.critic import CriticAgent
from agent.base import BaseAgent
from pathlib import Path
import os
import requests
from knowledgeBase.knowledge import ExpertFeedbackStore
from prompt.planner import FATHER_PLANNER_PROMPT,CHILD_PLANNER_PROMPT,CLARIFY_PHASE1_PROMPT,CLARIFY_PHASE2_PROMPT,PLAN_REFINE_HUMAN_PROMPT,CHILD_PLAN_REFINE_HUMAN_PROMPT
from schema import Plan, Subtask, TaskResult, ExpertFeedbackItem, FeedbackSource
from utils import parse_res
import logger
from prompt.task_types import ParentTaskType, ChildTaskType, PARENT_CHILD_TASK_MAPPING,PARENT_TASK_TYPES_CAN_BE_DECOMPOSED
from knowledgeBase.case_retriever import retrieve_kaggle_knowledge,format_kaggle_competitions
import random
from itertools import groupby

class PlannerAgent(BaseAgent):
    """负责进行任务分解、更新任务树、总结的Agent"""

    name: str = "PlannerAgent"
    description: str = "负责进行任务分解、更新任务树、总结的Agent"
    plan: Plan = None
    critic: Optional[CriticAgent] = None
    feedback_store: ExpertFeedbackStore = ExpertFeedbackStore()
    run_id: str = ""
    _human_suggestion=""



    def act(self) -> Plan:
        logger.trace(f"【Planner Context Store】: {json.dumps(self.context.store, ensure_ascii=False)}")
        
        # solution = ProblemSolution(self.requirement)
        need_further_split = getattr(utils.module_config, 'need_further_split', False)
        need_clarification = getattr(utils.module_config, 'need_clarification', False)
        enable_knowledge_retrieval = getattr(utils.module_config, 'enable_knowledge_retrieval', False)
        enable_kaggle_retrieval = getattr(utils.module_config, 'enable_kaggle_retrieval', False)
        refine_plan = getattr(utils.module_config, 'refine_plan', False)
        try:
            if need_clarification:
                # 两阶段澄清：Phase1(检索/下载) + Phase2(数据探索)
                clarification = self._clarify_requirement_two_phase()
                if clarification and isinstance(clarification, dict):
                    refined = clarification.get("refined_requirement") or ""
                    required_data_schema = clarification.get("required_data_schema") or {}
                    final_output_requirements = clarification.get("final_output_requirements") or ""
                    if refined:
                        # 在后续规划中使用澄清后的需求
                        self.requirement = refined
                        logger.info("【需求澄清结果】\n" + refined)
                    self.context.set("required_data_schema",required_data_schema)
                    self.context.set("final_output_requirements",final_output_requirements)
            # 2) 正式进行任务分解
            self.plan = self.get_planning(enable_kaggle_retrieval=enable_kaggle_retrieval, enable_knowledge_retrieval=enable_knowledge_retrieval)
            #todo: 确认任务分解树
            if not self.plan.subtasks:
                raise ValueError("任务分解结果为空")
        except Exception as e:
            logger.error(f"【任务分解出错】\n{traceback.format_exc()}")
            logger.info("【尝试修改任务分解Json格式】")
            try:
                self.plan = self.get_planning(enable_kaggle_retrieval=enable_kaggle_retrieval, enable_knowledge_retrieval=enable_knowledge_retrieval)
                if not self.plan.subtasks:
                    raise ValueError("任务分解结果为空")
                plan_confirmed = True
                logger.info("【修改成功】\n")
            except Exception as e:
                logger.error("【修改失败，重新进行任务分解】\n")
                logger.error(f"{traceback.format_exc()}")
        # 任务分解后人工审查
        from utils import human_review
        self._human_suggestion = ""
        suggestion = human_review("任务分解后", {"plan": self.plan.dict()})
        self._human_suggestion = suggestion
        while(self._human_suggestion.lower()!="y"):
            logger.info(f"【人机交互建议·任务分解后】{suggestion}")
            logger.info("【根据人机交互建议，调整任务分解结果】")
            self.plan = self.get_planning(enable_kaggle_retrieval=enable_kaggle_retrieval, enable_knowledge_retrieval=enable_knowledge_retrieval)
            suggestion = human_review("任务分解后", {"plan": self.plan.dict()})
            self._human_suggestion = suggestion
        self.plan.draw_table()
        # 设置原始需求并保存初始快照
        if not self.plan.raw_question:
            # raw_question 保存用户最初的自然语言需求
            self.plan.raw_question = self.requirement
        # 初始化保存目录并保存一次
        self._save_plan_snapshot(reason="initial_plan")

        self.plan.subtasks = self.plan._topological_sort()
        taskResult = None
        decomposition_attempted = set()
        
        # 记录上一次执行的任务ID和连续执行次数，防止死循环
        last_executed_task_id = None
        execution_count = 1

        while not self.plan.completed():
            # 获取所有未完成的父任务
            unfinished_parents = [t for t in self.plan.subtasks if not t.completed()]
            if not unfinished_parents:
                break
                
            current_parent = unfinished_parents[0]
            
            # 1. 自动分解逻辑
            # 如果需要分解，且尚未分解，且属于可分解类型，且本轮尚未尝试过分解
            if (need_further_split 
                and not current_parent.is_decomposed 
                and current_parent.task_type in PARENT_TASK_TYPES_CAN_BE_DECOMPOSED
                and current_parent.task_id not in decomposition_attempted):
                
                logger.info(f"[进入父层任务]开始处理任务 {current_parent.task_id}: {current_parent.task_type}")
                
                # 获取上一个父任务作为上下文
                idx = self.plan.subtasks.index(current_parent)
                previous_parent = self.plan.subtasks[idx-1] if idx > 0 else None
                
                # 标记已尝试分解，避免死循环
                decomposition_attempted.add(current_parent.task_id)
                self.plan.current_task = current_parent

                current_parent.subtasks = self.generate_subtasks(current_parent, previous_parent)
                
                if current_parent.subtasks:
                    current_parent.is_decomposed = True
                    current_parent.task_result = TaskResult(is_success=False) # 标记父节点本身为成功（逻辑上）
                    logger.info(f"【父层任务拆分为 {len(current_parent.subtasks)} 个子任务】原父层任务指令：" + current_parent.instruction)
                    current_parent.draw_table()
                    self._save_plan_snapshot(reason="after_split")
                    # 分解成功后，直接进入下一轮循环，逻辑会自动选中第一个子任务执行
                    continue
            
            # 2. 确定实际执行的任务 (Executable Task)
            executable_task = None
            
            if current_parent.is_decomposed:
                # 寻找未完成的子任务
                unfinished_subs = [st for st in current_parent.subtasks if not st.completed()]
                if not unfinished_subs:
                    # 所有子任务均已完成，却没出发completed？强制标记完成
                    current_parent.is_finished = True
                    continue
                executable_task = unfinished_subs[0]
                logger.info(f"[执行子层任务]正在执行任务 {executable_task.task_id}: {executable_task.task_type} - {executable_task.instruction}")
            else:
                # 不需要拆分或拆分失败，直接执行父任务
                executable_task = current_parent
                logger.info(f"[执行父层任务]正在执行任务 {executable_task.task_id}: {executable_task.task_type} - {executable_task.instruction}")

            # 死循环检测
            if executable_task.task_id == last_executed_task_id:
                execution_count += 1
                logger.warning(f"【循环检测】任务 {executable_task.task_id} 第 {execution_count} 次连续被执行。")
            else:
                last_executed_task_id = executable_task.task_id
                execution_count = 1
            
            if execution_count > 2:
                logger.error(f"【循环保护】任务 {executable_task.task_id} 已连续执行 {execution_count} 次，强制标记为失败。")
                executable_task.task_result = TaskResult(is_success=False)
                executable_task.is_finished = True
                self._save_plan_snapshot(reason="loop_terminated")
                return self.plan

            # 3. 执行任务
            self.plan.current_task = executable_task
            taskResult = self.handle_task(executable_task)
            executable_task.task_result = taskResult
            
            # 兼容性：确保 is_finished 标志被设置，虽然 completed() 会检查 task_result
            if taskResult and taskResult.is_success:
                 executable_task.is_finished = True
            
            self._save_plan_snapshot(reason="after_execution")
            
            # 4. 动态规划与精炼
            # 无论父/子任务，执行完都可进行 Refine
            if refine_plan:
                prev_plan_str = str(self.plan.subtasks)
                logger.info("【开始进行动态规划与精炼计划】")
                self._refine_plan(executable_task)
                # 如果 Plan 被 Refine 修改了（例如移除了任务），下一轮循环会自动适应
            
            # 5. 更新父任务状态
            if current_parent.is_decomposed:
                 current_parent.is_finished = all([st.completed() for st in current_parent.subtasks])

        extract_experience = getattr(utils.module_config, 'extract_experience', False)
        if extract_experience and taskResult and taskResult.is_success:
            case_path = self._results_dir() / "plan.json"
            self.feedback_store.extract_from_success_case(case_path=str(case_path), task_type=None)
            try:
                self._save_case_to_library()
            except Exception as e:
                logger.error(f"【案例保存失败】\n{traceback.format_exc()}")
        return self.plan

    def _save_case_to_library(self):
        """将当前成功的任务作为案例保存到知识库 (LLM 生成案例内容)"""
        try:
            from prompt.knowledge import CASE_GENERATION_SYSTEM_PROMPT, CASE_GENERATION_USER_PROMPT
            from utils import remove_code_recursive

            case_path = Path("knowledgeBase/case_library.jsonl")
            case_path.parent.mkdir(parents=True, exist_ok=True)
            
            # 准备上下文数据
            plan_dict = self.plan.dict()
            # 移除过长的详细代码，但保留部分用于摘要（如果需要详细代码，需调整 prompt 或摘要策略）
            # 这里为了不超出 context window，我们尽量移除所有代码，或者仅保留最后成功的关键步骤代码
            # 实际上，remove_code_recursive 可能会移除所有 'code' 字段，这可能导致解决方案细节缺失
            # 我们可以构造一个专门的 summary dict
            remove_code_recursive(plan_dict, field_name="code")
            # 调用 LLM 生成案例
            messages = [
                {"role": "system", "content": CASE_GENERATION_SYSTEM_PROMPT},
                {"role": "user", "content": CASE_GENERATION_USER_PROMPT.format(record_json=json.dumps(plan_dict, ensure_ascii=False))}
            ]
            
            response = self.llm.ask(messages)
            try:
                case_data = json.loads(parse_res(response))
                
                # 构造最终存储的 case 对象
                # 确保格式符合 knowledgeBase/case_library.jsonl 的要求
                new_case = {
                    "title": case_data.get("title", f"AutoDS Success Case - {self.run_id}"),
                    "description": case_data.get("description", self.requirement),
                    "solutions": case_data.get("solutions", {}),
                    "tags": case_data.get("tags", []) + ["autods", "generated"]
                }
                
                with open(case_path, "a", encoding="utf-8") as f:
                    f.write(json.dumps(new_case, ensure_ascii=False) + "\n")
                
                logger.info(f"【案例库更新】已保存生成案例: {new_case['title']}")

                # 尝试增量更新向量索引
                try:
                    from knowledgeBase.case_retriever import add_new_case_to_index
                    add_new_case_to_index(str(case_path))
                except Exception as e:
                    logger.warning(f"自动更新向量索引失败: {e}")
                
            except Exception as e:
                logger.error(f"【案例库生成解析失败】{e}\nResponse: {response}")

        except Exception as e:
            logger.error(f"【案例库保存流程失败】{e}")


    
    def _refine_subtasks_by_human(self, parent_task: Subtask, current_subtasks: List[Subtask], suggestions: str) -> List[Subtask]:
        # Serialize current subtasks
        subtasks_json = [st.to_simple_dict() for st in current_subtasks]
            
        parent_type_name = parent_task.task_type
        child_task_names = PARENT_CHILD_TASK_MAPPING.get(parent_type_name, [])
        child_task_names.append("其他") 

        child_task_type_desc = "\n".join([
            f"- **{member.value.name}**: {member.value.desc}"
            for member in ChildTaskType
            if member.value.name in child_task_names
        ])

        prompt = CHILD_PLAN_REFINE_HUMAN_PROMPT.format(
            parent_task_desc=f"Task ID: {parent_task.task_id}\nType: {parent_task.task_type}\nInstruction: {parent_task.instruction}",
            current_subtasks=json.dumps(subtasks_json, indent=2, ensure_ascii=False),
            human_suggestion=suggestions,
            child_task_type_desc=child_task_type_desc
        )

        messages = [{"role": "user", "content": prompt}]
        response = self.llm.ask(messages)
        
        try:
            content = parse_res(response)
            try:
                result = json.loads(content)
            except json.JSONDecodeError:
                # 尝试用 ast 修复单引号 JSON
                import ast
                result = ast.literal_eval(content)

            if result.get("split", False):
                new_subtasks = []
                for child in result.get("children", []):
                    new_st = Subtask(
                        task_id=child["task_id"],
                        instruction=child["instruction"],
                        task_type=child["task_type"],
                        dependent_task_ids=child.get("dependent_task_ids", [])
                    )
                    new_subtasks.append(new_st)
                return new_subtasks
            else:
                 if "refined_parent_instruction" in result:
                     parent_task.instruction = result["refined_parent_instruction"]
                 return []
                 
        except Exception as e:
            logger.error(f"Error parsing refined subtasks: {e}")
            return current_subtasks

    def generate_subtasks(self, current_task: Subtask, previous_task: Subtask) -> List[Subtask]:
        """
        current_task: 当前的父层任务
        previous_task: 上一级的父层任务（可为 None）
        作用：
        根据父层任务，自动生成子层任务。
        调用 LLM 进行智能拆分。
        """
        logger.info(f"【开始拆分父层任务】{current_task.task_type}")
        pre_task_results=""
        if previous_task and previous_task.task_result:
            last_attempt = previous_task.get_last_subtask().get_last_attempt()
            if last_attempt:
                pre_task_results = str(last_attempt.result)
        parent_type_name = current_task.task_type
        child_task_names = PARENT_CHILD_TASK_MAPPING.get(parent_type_name, [])
        child_task_names.append("其他")  # 允许生成其他类型的子任务

        child_task_type_desc = "\n".join([
            f"- **{member.value.name}**: {member.value.desc}"
            for member in ChildTaskType
            if member.value.name in child_task_names
        ])
        # 注入与当前父任务类型相关的专家反馈，以影响拆分与指令精炼
        enable_knowledge_retrieval = getattr(utils.module_config, 'enable_knowledge_retrieval', False)
        expert_suggestions=""
        if enable_knowledge_retrieval:
            expert_suggestions = self._render_expert_feedback(current_task.task_type)
        system_prompt = CHILD_PLANNER_PROMPT.format(pre_task_results=pre_task_results,
                                                    child_task_type_desc=child_task_type_desc + ("\n\n# 专家建议：\n" + expert_suggestions if expert_suggestions else ""))
        user_prompt = ""
        if current_task.task_type=="建模与训练层":
            user_prompt+= f"总任务需求：{self.requirement}\n"
        user_prompt+=f"- **父任务类型**: {current_task.task_id}:{current_task.task_type}\n- **父任务指令**: {current_task.instruction}"
        messages = [
            {
                "role": "system",
                "content": system_prompt,
            },
            {
                "role": "user",
                "content": user_prompt,
            },
        ]
        response = self.llm.ask(messages)
        content = parse_res(response)
        try:
            res = json.loads(content)
        except json.JSONDecodeError:
            logger.warning(f"JSON 解析失败，尝试使用 ast.literal_eval 修复: {content[:200]}...")
            try:
                import ast
                # 尝试解析 Python 字典格式（单引号键值对）
                res = ast.literal_eval(content)
            except Exception as e:
                logger.error(f"JSON 和 AST 解析均失败。\n原始响应: {content}\n错误: {e}")
                # 既然解析失败，为了不让程序直接崩掉，尝试再次请求或者返回默认失败结构
                # 这里我们抛出异常让外层捕获并重试（如果有外层重试逻辑的话）
                # 现在的逻辑中，外层有个 try-except 会捕获异常并重试任务分解
                raise e

        if res.get("split") is False:
            # 不需要拆分，直接返回空列表
            current_task.instruction = res.get("refined_parent_instruction", current_task.instruction)
            logger.info("【父层任务不需拆分】父任务指令明确后：" + current_task.instruction)
            subtasks = []
        else:
            # 需要拆分，返回子任务列表
            subtasks_data = res.get("children", [])
            subtasks = []
            for child_data in subtasks_data:
                try:
                    child_task = Subtask.from_dict(child_data)
                    child_task.parent_task_id = current_task.task_id
                    subtasks.append(child_task)
                except Exception as e:
                    logger.error(f"【子任务格式错误，跳过该子任务】\n{traceback.format_exc()}")
                    logger.error(f"{child_data}")
            # for ct in subtasks:
            #     logger.info(f"  - 子任务 {ct.task_id}: {ct.task_type} - {ct.instruction}")
        
        # 增加人机交互调整
        from utils import human_review
        review_context={
            "subtasks": [st.to_simple_dict() for st in subtasks]
        }
        suggestion = human_review("子任务分解后", review_context)
        while suggestion.lower() != "y":
            logger.info(f"【人机交互建议·子任务分解后】{suggestion}")
            subtasks = self._refine_subtasks_by_human(current_task, subtasks, suggestion)
            review_context["subtasks"] = [st.to_simple_dict() for st in subtasks]
            suggestion = human_review("子任务分解后", review_context)
            
        return subtasks
    
    def get_planning(self, enable_knowledge_retrieval: bool = False, enable_kaggle_retrieval: bool = False) -> Plan:
        """
        获得问题的分解结果

        :return: 问题的分解结果
        """
        logger.info(f"【开始获取任务的分解结果】", self.requirement)
        task_type_desc = "\n".join([f"- **{tt.value.name}**: {tt.value.desc}" for tt in ParentTaskType])
        
        # 决定使用哪个 prompt：如果有 Plan 且有人工建议，则使用 Refine Prompt；否则使用 Create Prompt
        # 注意：这里假设如果 _human_suggestion 存在，则 implies 我们是在迭代。
        # 如果 plan 为空（第一次），即使有 suggestion (理论上不应有)，也只能重头生成。
        
        use_refine = False
        if self._human_suggestion and self.plan and self.plan.subtasks:
            use_refine = True
            
        if use_refine:
             system_prompt = PLAN_REFINE_HUMAN_PROMPT.format(
                requirement=self.requirement,
                current_plan=json.dumps([t.to_simple_dict() for t in self.plan.subtasks], ensure_ascii=False, indent=2),
                human_suggestion=self._human_suggestion,
                parent_task_type_desc=task_type_desc
            )
             user_prompt = "请根据人工建议调整上述计划。"
             # 为了保持逻辑简单，refine 模式通常不需要检索工具，纯逻辑修改
             tool_names = [] 
        else:
            # 注入与父任务设计相关的专家建议
            experience_knowledge = ""
            related_cases = ""
            required_data_schema = self.context.get("required_data_schema",{})
            final_output_requirements = self.context.get("final_output_requirements","")
            if enable_knowledge_retrieval:
                experience_knowledge = self._render_expert_feedback("方案建议")
            if enable_kaggle_retrieval:
                related_cases_obj=retrieve_kaggle_knowledge(
                    task_description=self.requirement,
                    top_k=2,  # 检索前2个最相似的竞赛
                    threshold=0.65 # 相似度阈值
                )
                related_cases = format_kaggle_competitions(related_cases_obj)
            system_prompt = FATHER_PLANNER_PROMPT.format(
                experience_knowledge=experience_knowledge,
                related_cases=related_cases,
                parent_task_type_desc=task_type_desc
            )
            user_prompt = "用户需求："+self.requirement
            if self._human_suggestion:
                # 只有 suggestion 但没 plan (或者 plan 被清空)，则附带建议重生成
                user_prompt += f"\n\n人机交互建议：\n{self._human_suggestion}"
            if required_data_schema:
                user_prompt += f"\n\n数据结构信息：\n{json.dumps(required_data_schema, ensure_ascii=False, indent=2)}"
            if final_output_requirements:
                user_prompt += f"\n\n最终输出要求：\n{final_output_requirements}"
            
            tool_names = []
            web_datadownload_tool=getattr(utils.module_config, 'web_datadownload_tool', False)
            if web_datadownload_tool:
                tool_names = ["web_search_summarize"]

        messages = [
            {
                "role": "system",
                "content": system_prompt,
            },
            {
                "role": "user",
                "content": user_prompt,
            },
        ]
        
        try:
            if tool_names:
                response = self.llm.ask_with_tools(
                    messages=messages,
                    tool_names=tool_names,
                    max_loops=2,
                )
            else:
                response = self.llm.ask(messages)
        except Exception:
            response = self.llm.ask(messages)
            
        content = parse_res(response)
        try:
            res = json.loads(content)
        except json.JSONDecodeError:
            logger.warning(f"get_planning 解析 JSON 失败，尝试用 ast 修复: {content[:200]}...")
            try:
                import ast
                res = ast.literal_eval(content)
            except Exception as e:
                logger.error(f"get_planning 解析失败。Content: {content}\nError: {e}")
                raise e
        
        # 如果是 Refine 模式，解析出来的 list 需要转为 Plan 对象
        # Plan.from_dict 期望 {"task_id": ...} 列表 或者是 {"subtasks": [...]}
        # 假设 LLM 返回的是 list
        if isinstance(res, list):
             # 构造一个符合 Plan 结构的 dict
             res = {"subtasks": res}
        
        plan = Plan.from_dict(res)
        return plan

    def _clarify_requirement_two_phase(self) -> dict:
        """两阶段需求澄清：
        Phase1: 可调用 web_search_summarize / kaggle_downloader / data_downloader。
        输出 refined_requirement / knowledge_summary / dataset_paths。
        Phase2: 仅可调用 data_explorer，对 Phase1 的 dataset_paths 做结构探索。
        最终合并输出：{
          refined_requirement,
          knowledge_summary,
          dataset_paths,
          required_data_schema: {...},
          explore_decision,
          notes: []
        }
        """
        # ========== Phase 1 ==========
        try:
            logger.info("【澄清阶段1】开始：检索/数据下载判断")
            system_prompt_1 = CLARIFY_PHASE1_PROMPT
            messages1 = [
                {"role": "system", "content": system_prompt_1},
                {"role": "user", "content": f"用户原始需求：{self.requirement}"},
            ]
            # tool_names = ["kaggle_downloader", "data_downloader"]
            tool_names=[]
            if getattr(utils.module_config, 'web_datadownload_tool', False):
                tool_names = ["kaggle_downloader", "data_downloader","web_search_summarize"]
                
                # tool_names.insert(0, "web_search_summarize")
            resp1 = self.llm.ask_with_tools(
                messages1,
                tool_names=tool_names,
                max_loops=3,
            )
            phase1 = json.loads(parse_res(resp1)) if resp1 else {}
        except Exception:
            phase1 = {}
            logger.error(f"【澄清阶段1异常】\n{traceback.format_exc()}")
        if not isinstance(phase1, dict):
            phase1 = {}
        refined_req = phase1.get("refined_requirement") or self.requirement
        dataset_paths = phase1.get("dataset_paths") or []
        dataset_description = phase1.get("dataset_description") or ""
        final_output_requirements = phase1.get("final_output_requirements") or ""
        notes= phase1.get("notes") or []
        knowledge_summary = phase1.get("knowledge_summary") or ""

        # ========== Phase 2 ==========
        try:
            logger.info("【澄清阶段2】开始：数据探索")
            system_prompt_2 = CLARIFY_PHASE2_PROMPT
            user_payload = {
                "refined_requirement": refined_req,
                "dataset_paths": dataset_paths,
                "dataset_description": dataset_description,
                "notes": notes
            }
            messages2 = [
                {"role": "system", "content": system_prompt_2},
                {"role": "user", "content": json.dumps(user_payload, ensure_ascii=False)},
            ]
            resp2 = self.llm.ask_with_tools(
                messages2,
                tool_names=["data_explorer"],
                max_loops=3,
            )
            phase2 = json.loads(parse_res(resp2)) if resp2 else {}
        except Exception:
            phase2 = {}

        if not isinstance(phase2, dict):
            phase2 = {}

        required_data_schema = phase2.get("required_data_schema") or {}
        explore_decision = phase2.get("explore_decision") or ("explored" if required_data_schema else "skip")
        notes_combined = []
        notes_combined.extend(phase1.get("notes") or [])
        notes_combined.extend(phase2.get("notes") or [])

        result = {
            "refined_requirement": refined_req,
            "knowledge_summary": knowledge_summary,
            "dataset_paths": dataset_paths,
            "required_data_schema": required_data_schema,
            "final_output_requirements": final_output_requirements,
            "explore_decision": explore_decision,
            "notes": notes_combined,
        }
        # 兜底
        if not result.get("refined_requirement"):
            result["refined_requirement"] = self.requirement
        return result

    def _render_expert_feedback(self, task_type: str) -> str:
        """汇总专家反馈为可读文本注入到系统提示中。
        按评分降序排列，相同分数的项内部随机排序，最多取前3条。
        """
        try:
            logger.debug(f"正在汇总专家反馈，任务类型: {task_type}")
            items = self.feedback_store.query(task_type=task_type)
            if not items:
                logger.debug("无相关专家反馈")
                return ""

            # 默认 score 为 0（处理 None）
            def get_score(it):
                return it.score if it.score is not None else 0

            # 先按 score 降序排序
            items.sort(key=get_score, reverse=True)

            # 按 score 分组，在每组内随机打乱
            grouped = []
            for score_val, group in groupby(items, key=get_score):
                group_list = list(group)
                random.shuffle(group_list)  # 同分组内随机
                grouped.extend(group_list)

            # 取前3条
            selected_items = grouped[:3]

            lines = []
            for it in selected_items:
                line = f"- [建议] {it.recommendation}"
                if it.anti_pattern:
                    line += f"\n  [警示] 避免：{it.anti_pattern}"
                if it.rationale:
                    line += f"（原因：{it.rationale}）"
                if it.condition:
                    line += f" [条件: {it.condition}]"
                lines.append(line)

            logger.debug(f"汇总的专家反馈内容:\n{chr(10).join(lines)}")
            return "\n".join(lines)
        except Exception as e:
            logger.error(f"汇总专家反馈失败: {e}")
            return ""


    
    def handle_task(self, task: Subtask):
        """
        处理每个子任务，包含执行、重试和动态规划

        :param task: 子任务
        """
        max_retries = 1
        attempt = 0
        
        while attempt < max_retries:
            attempt += 1
            
            actor = ActorAgent(
                task=task,
                requirement=self.requirement,
                context=self.context,
                critic=self.critic
            )
            taskResult=actor.act(plan=self.plan)
            task.task_result = taskResult
            
            if taskResult.is_success:
                # 任务成功，更新全局上下文中的关键中间结果
                if taskResult.new_knowledge:
                    current_knowledge = self.context.get("key_intermediate_results", {})
                    # 使用 update 方法合并新知识，而不是覆盖
                    current_knowledge.update(taskResult.new_knowledge)
                    self.context.set("key_intermediate_results", current_knowledge)
                    logger.info(f"【知识更新】已更新关键中间结果: {list(taskResult.new_knowledge.keys())}")
                return taskResult
            else:
                replay_failure = getattr(utils.module_config, 'replay_failure', False)
                if replay_failure:
                    # 任务失败，尝试恢复
                    should_retry = self._handle_task_failure(task)
                    if should_retry:
                        logger.info(f"【任务重试】正在重试任务 {task.task_id} (第 {attempt} 次)")
                        continue
                    else:
                        # 无法恢复或已跳过/插入新任务
                        return taskResult
        
        return task.task_result

    # ============ 内部工具：保存计划快照并通知前端 ============
    def _results_dir(self) -> Path:
        rid = self.run_id or os.environ.get("RUN_ID") or ""
        if not rid:
            rid = "no_run"
        p = Path("output")/"results"/rid
        p.mkdir(parents=True, exist_ok=True)
        return p

    def _save_plan_snapshot(self, reason: str = ""):
        try:
            base = self._results_dir()
            with open(base/"plan.json", "w", encoding="utf-8") as f:
                json.dump(self.plan.dict(), f, ensure_ascii=False, indent=2)
            logger.debug(f"【计划已保存】{reason} -> {base/ 'plan.json'}")
            # 通知后端广播文件更新
            self._notify_server_file("plan.json")
        except Exception as e:
            logger.debug(f"【计划保存失败（忽略）】{e}")

    def _notify_server_file(self, filename: str):
        try:
            rid = self.run_id or os.environ.get("RUN_ID")
            if not rid:
                return
            base_url = os.environ.get("SERVER_BASE_URL", "http://localhost:8001")
            url = f"{base_url}/api/runs/{rid}/notify_file"
            requests.post(url, json={"filename": filename}, timeout=1.5)
        except Exception:
            # 忽略通知失败，以免影响主流程
            pass


    def _refine_plan(self, current_task: Subtask):
        """
        根据当前任务的执行结果和 Actor 的建议，动态调整后续计划。
        """
        if not self.critic:
            return
        
        # 1. Determine level and tasks
        refine_level = "子任务层" if current_task.parent_task_id else "父任务层"
        current_level_tasks = self.plan.get_current_level_tasks()
        
        # pending_tasks (uncompleted and not current)
        # Note: current_level_tasks includes current_task.
        completed_tasks = [t for t in current_level_tasks if t.completed() or t.task_id == current_task.task_id]
        pending_tasks = [t for t in current_level_tasks if not t.completed() and t.task_id != current_task.task_id]

        from utils import human_review
        # 动态 plan refine 前人工审查后续任务
        suggestion = human_review("动态 plan refine 前（后续任务）", {"后续task": [t.to_simple_dict() for t in pending_tasks]})
        if suggestion.lower() != "y":
            logger.info(f"【人机交互建议·plan refine】{suggestion}")
            self._human_suggestion = suggestion
        
        if not current_task.task_result or not current_task.task_result.suggested_next_steps:
            return

        logger.info(f"【动态规划】收到 Actor 建议: {current_task.task_result.suggested_next_steps}")
        
        if not pending_tasks:
            logger.info(f"【动态规划】当前层{refine_level}无后续任务，跳过计划更新")
            return

        # 获取最后一次尝试的结果作为 execution_summary
        execution_summary = ""
        if current_task.task_result.attempts:
            last_attempt_idx = max(current_task.task_result.attempts.keys())
            execution_summary = current_task.task_result.attempts[last_attempt_idx].result

        try:
            res = self.critic.refine_plan(
                current_task=current_task,
                execution_summary=execution_summary,
                actor_suggestions=json.dumps(current_task.task_result.suggested_next_steps, ensure_ascii=False),
                new_knowledge=json.dumps(current_task.task_result.new_knowledge, ensure_ascii=False),
                pending_tasks=pending_tasks,
                human_suggestion=getattr(self, '_human_suggestion', ""),
                refine_level=refine_level,
                finished_task_count=len(completed_tasks),
            )
            
            if hasattr(self, '_human_suggestion'):
                self._human_suggestion = ""
            
            if res.get("refined"):
                logger.info(f"【动态规划】计划已更新({refine_level}): {res.get('reason')}")
                new_pending_tasks_data = res.get("pending_tasks", [])
                
                # 重建待执行任务列表
                new_pending_tasks = []
                for t_data in new_pending_tasks_data:
                    try:
                        task_obj = Subtask.from_dict(t_data)
                        if refine_level == "子任务层":
                            task_obj.parent_task_id = current_task.parent_task_id
                        new_pending_tasks.append(task_obj)
                    except:
                        pass

                if current_task.parent_task_id:
                    parent_task= self.plan.get_task_by_id(current_task.parent_task_id)
                    parent_task.subtasks = completed_tasks + new_pending_tasks
                    logger.info(f"【动态规划】更新父任务 {parent_task.task_id} 的子任务列表")
                    parent_task.draw_table()
                else:
                    self.plan.subtasks = completed_tasks + new_pending_tasks
                    logger.info(f"【动态规划】更新顶层任务的子任务列表")
                    self.plan.draw_table()
                self._save_plan_snapshot(reason="plan_refined")
        except Exception as e:
            logger.error(f"【动态规划】更新计划失败: {e}")

    def _handle_task_failure(self, current_task: Subtask) -> bool:
        """
        处理任务失败的情况，尝试重规划。
        返回 True 表示已生成恢复计划（需要重新执行），False 表示无法恢复（任务彻底失败）。
        """
        if not self.critic:
            return False

        from utils import human_review
        # 故障恢复前人工审查
        suggestion = human_review("故障恢复前", {"task": current_task.to_full_dict()})
        if suggestion.lower() != "y":
            logger.info(f"【人机交互建议·故障恢复】{suggestion}")
            self._human_suggestion = suggestion
        logger.warning(f"【故障恢复】任务 {current_task.task_id} 失败，尝试重规划...")
        sub_level = True if current_task.parent_task_id else False
        pending_tasks = [t for t in self.plan.get_current_level_unfinished_tasks() if t.task_id != current_task.task_id]
        
        # 获取最后一次尝试的错误信息
        last_error = "Unknown error"
        if current_task.task_result and current_task.task_result.attempts:
            last_attempt = list(current_task.task_result.attempts.values())[-1]
            last_error = last_attempt.result
            if len(last_error) > 1000:
                last_error = last_error[:500] + "..." + last_error[-500:]  # 截取前500字符跟后500字作为错误信息摘要
            
        try:
            res = self.critic.replan_on_failure(
                failed_task=current_task,
                error_message=last_error,
                attempts=len(current_task.task_result.attempts) if current_task.task_result else 0,
                pending_tasks=pending_tasks,
                human_suggestion=getattr(self, '_human_suggestion', "")
            )
            
            if hasattr(self, '_human_suggestion'):
                self._human_suggestion = ""
            
            action = res.get("action")
            reason = res.get("reason")
            logger.info(f"【故障恢复】策略: {action}, 原因: {reason}")
            
            if action == "RETRY":
                new_task_data = res.get("new_task")
                if new_task_data:
                    current_task.instruction = new_task_data.get("instruction", current_task.instruction)
                    return True 
                    
            elif action == "SKIP":
                new_pending_tasks = []
                adjusted_tasks_data = res.get("adjusted_pending_tasks", [])
                for t_data in adjusted_tasks_data:
                    task_obj = Subtask.from_dict(t_data)
                    if sub_level:
                        task_obj.parent_task_id = current_task.parent_task_id
                new_pending_tasks.append(task_obj)
                
                completed_tasks = [t for t in self.plan.get_current_level_finished_tasks() if t.completed() and t.task_id != current_task.task_id]
                current_task.task_result.is_success = False 
                current_task.is_finished = True
                
                if sub_level:
                    parent_task = self.plan.get_task_by_id(current_task.parent_task_id)
                    parent_task.subtasks = completed_tasks + [current_task] + new_pending_tasks
                    parent_task.draw_table()
                else:
                    self.plan.subtasks = completed_tasks + [current_task] + new_pending_tasks
                    self.plan.draw_table()
                self._save_plan_snapshot(reason="task_skipped")
                return False 
                
            elif action == "INSERT":
                new_task_data = res.get("new_task")
                
                if new_task_data:
                    new_task = Subtask.from_dict(new_task_data)
                    if sub_level:
                        new_task.parent_task_id = current_task.parent_task_id  
                    
                    completed_tasks = [t for t in self.plan.get_current_level_finished_tasks() if t.completed() and t.task_id != current_task.task_id]
                    if sub_level:
                        parent_task = self.plan.get_task_by_id(current_task.parent_task_id)
                        parent_task.subtasks = completed_tasks + [new_task, current_task] + pending_tasks
                        parent_task.draw_table()
                    else:
                        self.plan.subtasks = completed_tasks + [new_task, current_task] + pending_tasks
                        self.plan.draw_table()
                    self._save_plan_snapshot(reason="task_inserted")
                    return False 
                    
        except Exception as e:
            logger.error(f"【故障恢复】执行失败: {e}")
            
        return False
