"""
定义投票、问题、子问题、分解、API响应等类
"""

import copy
from enum import Enum

from pydantic import BaseModel, Field

import logger
from collections import OrderedDict
from typing import Any, List, Literal, Optional, Union, Dict
import warnings


class Role(str, Enum):
    """Message role options"""

    SYSTEM = "system"
    USER = "user"
    ASSISTANT = "assistant"
    TOOL = "tool"


ROLE_VALUES = tuple(role.value for role in Role)
ROLE_TYPE = Literal[ROLE_VALUES]


class ToolChoice(str, Enum):
    """Tool choice options"""

    NONE = "none"
    AUTO = "auto"
    REQUIRED = "required"


TOOL_CHOICE_VALUES = tuple(choice.value for choice in ToolChoice)
TOOL_CHOICE_TYPE = Literal[TOOL_CHOICE_VALUES]


class Function(BaseModel):
    name: str
    arguments: str


class ToolCall(BaseModel):
    """Represents a tool/function call in a message"""

    id: str
    type: str = "function"
    function: Function


class Context(BaseModel):
    """任务执行链路中的轻量上下文，供 DSagent/Planner/Actor 共享关键信息。"""

    task_id: Optional[str] = None
    store: Dict[str, Any] = Field(default_factory=dict)

    # 写入/更新
    def set(self, key: str, value: Any) -> None:
        self.store[key] = value

    # 读取
    def get(self, key: str, default: Any = None) -> Any:
        return self.store.get(key, default)
    @classmethod
    def from_dict(cls, data: Dict[str, Any]) -> "Context":
        context = cls()
        context.store.update(data)
        return context
    
    # 字符串化导出
    def dump_all_as_str(self) -> str:
        lines = [f"{k}: {v}" for k, v in self.store.items()]
        return "\n".join(lines)

    def dump_key_as_str(self, key: str) -> str:
        if key not in self.store:
            return ""
        return f"{key}: {self.store[key]}"

    def dump_recent_as_str(self, n: int) -> str:
        items = list(self.store.items())[-n:]
        lines = [f"{k}: {v}" for k, v in items]
        return "\n".join(lines)

    def to_dict(self) -> Dict[str, Any]:
        return {"task_id": self.task_id, "context": dict(self.store)}
class Message(BaseModel):
    """Represents a chat message in the conversation"""

    role: ROLE_TYPE = Field(...)  # type: ignore
    content: Optional[str] = Field(default=None)
    tool_calls: Optional[List[ToolCall]] = Field(default=None)
    name: Optional[str] = Field(default=None)
    tool_call_id: Optional[str] = Field(default=None)

    def __add__(self, other) -> List["Message"]:
        """支持 Message + list 或 Message + Message 的操作"""
        if isinstance(other, list):
            return [self] + other
        elif isinstance(other, Message):
            return [self, other]
        else:
            raise TypeError(
                f"unsupported operand type(s) for +: '{type(self).__name__}' and '{type(other).__name__}'"
            )

    def __radd__(self, other) -> List["Message"]:
        """支持 list + Message 的操作"""
        if isinstance(other, list):
            return other + [self]
        else:
            raise TypeError(
                f"unsupported operand type(s) for +: '{type(other).__name__}' and '{type(self).__name__}'"
            )

    def to_dict(self) -> dict:
        """Convert message to dictionary format"""
        message = {"role": self.role}
        if self.content is not None:
            message["content"] = self.content
        if self.tool_calls is not None:
            message["tool_calls"] = [
                tool_call.model_dump() for tool_call in self.tool_calls
            ]
        if self.name is not None:
            message["name"] = self.name
        if self.tool_call_id is not None:
            message["tool_call_id"] = self.tool_call_id
        return message

    @classmethod
    def user_message(cls, content: str) -> "Message":
        """Create a user message"""
        return cls(role=Role.USER, content=content)

    @classmethod
    def system_message(cls, content: str) -> "Message":
        """Create a system message"""
        return cls(role=Role.SYSTEM, content=content)

    @classmethod
    def assistant_message(cls, content: Optional[str] = None) -> "Message":
        """Create an assistant message"""
        return cls(role=Role.ASSISTANT, content=content)

    @classmethod
    def tool_message(cls, content: str, name, tool_call_id: str) -> "Message":
        """Create a tool message"""
        return cls(
            role=Role.TOOL,
            content=content,
            name=name,
            tool_call_id=tool_call_id,
        )

    @classmethod
    def from_tool_calls(
        cls,
        tool_calls: List[Any],
        content: Union[str, List[str]] = "",
        **kwargs,
    ):
        """Create ToolCallsMessage from raw tool calls.

        Args:
            tool_calls: Raw tool calls from LLM
            content: Optional message content
            base64_image: Optional base64 encoded image
        """
        formatted_calls = [
            {"id": call.id, "function": call.function.model_dump(), "type": "function"}
            for call in tool_calls
        ]
        return cls(
            role=Role.ASSISTANT,
            content=content,
            tool_calls=formatted_calls,
            **kwargs,
        )


class Memory(BaseModel):
    messages: List[Message] = Field(default_factory=list)
    max_messages: int = Field(default=100)

    def add_message(self, message: Message) -> None:
        """Add a message to memory"""
        self.messages.append(message)
        # Optional: Implement message limit
        if len(self.messages) > self.max_messages:
            self.messages = self.messages[-self.max_messages :]

    def add_messages(self, messages: List[Message]) -> None:
        """Add multiple messages to memory"""
        self.messages.extend(messages)

    def clear(self) -> None:
        """Clear all messages"""
        self.messages.clear()

    def get_recent_messages(self, n: int) -> List[Message]:
        """Get n most recent messages"""
        return self.messages[-n:]

    def to_dict_list(self) -> List[dict]:
        """Convert messages to list of dicts"""
        return [msg.to_dict() for msg in self.messages]

    def get_messages(
        self,
        n: Optional[int] = None,
        roles: Optional[List[str]] = None,
        tool_names: Optional[List[str]] = None,
    ) -> List[Message]:
        """
        过滤消息，可按：
        - 最近 n 条
        - 消息角色（USER / ASSISTANT / TOOL / SYSTEM）
        - 工具名（匹配 tool_calls 或 tool message 的 name）
        """

        msgs = self.messages

        # 角色过滤
        if roles:
            roles = set(roles)
            msgs = [m for m in msgs if m.role in roles]

        # 工具名过滤
        if tool_names:
            names = set(tool_names)
            filtered = []
            for m in msgs:
                # 1) TOOL 类型（tool_message）
                if m.role == Role.TOOL and m.name in names:
                    filtered.append(m)
                    continue

                # 2) ASSISTANT 调用工具（tool_calls）
                if m.tool_calls:
                    for call in m.tool_calls:
                        func_name = call["function"]["name"]
                        if func_name in names:
                            filtered.append(m)
                            break

            msgs = filtered[-n:] if n is not None and n > 0 else filtered

        return msgs

class ModuleConfig:
    def __init__(
        self,
        workspace_path: str = "./workspace",
        need_further_split: bool = False,
        need_clarification: bool = False,
        enable_knowledge_retrieval: bool = False,
        enable_kaggle_retrieval: bool = False,
        replay_failure: bool = False,
        refine_plan: bool = False,
        extract_experience: bool = False,
        enable_human_review: bool = False,
        web_datadownload_tool: bool = False,
        enable_summary: bool = False,
        enable_result_review: bool = False,
        summary_type: Literal["report", "answer"] = "report",
        save_history: bool = True
    ):
        self.workspace_path = workspace_path
        self.need_further_split = need_further_split
        self.need_clarification = need_clarification
        self.enable_knowledge_retrieval = enable_knowledge_retrieval
        self.enable_kaggle_retrieval = enable_kaggle_retrieval
        self.replay_failure = replay_failure
        self.refine_plan = refine_plan
        self.extract_experience = extract_experience
        self.enable_human_review = enable_human_review
        self.web_datadownload_tool = web_datadownload_tool
        self.enable_summary = enable_summary
        self.enable_result_review = enable_result_review
        self.summary_type = summary_type
        self.save_history = save_history
    def to_dict(self):
        """将配置转换为字典"""
        return {
            "workspace_path": self.workspace_path,
            "need_further_split": self.need_further_split,
            "need_clarification": self.need_clarification,
            "enable_knowledge_retrieval": self.enable_knowledge_retrieval,
            "enable_kaggle_retrieval": self.enable_kaggle_retrieval,
            "replay_failure": self.replay_failure,
            "refine_plan": self.refine_plan,
            "extract_experience": self.extract_experience,
            "enable_human_review": self.enable_human_review,
            "web_datadownload_tool": self.web_datadownload_tool,
            "enable_summary": self.enable_summary,
            "enable_result_review": self.enable_result_review,
            "summary_type": self.summary_type,
            "save_history": self.save_history,
        }

    @classmethod
    def from_dict(cls, config_dict):
        """从字典创建配置对象"""
        return cls(**config_dict)

    def __repr__(self):
        return f"ModuleConfig({self.to_dict()})"


class ApiConfig:
    """API 配置对象"""

    def __init__(
        self,
        config_name: str,
        type: str,
        model: str,
        base_url: Optional[str] = None,
        api_key_env: Optional[str] = None,
        temperature: Optional[float] = 0,
        stream: Optional[bool] = False,
        enable_thinking: Optional[bool] = False,
    ):
        self.config_name = config_name
        self.type = type
        self.model = model
        self.base_url = base_url
        self.api_key_env = api_key_env
        self.temperature = temperature
        self.stream = stream
        self.enable_thinking = enable_thinking

    def __repr__(self):
        return (
            f"ApiConfig(config_name='{self.config_name}', type='{self.type}', "
            f"model='{self.model}', base_url='{self.base_url}', api_key_env='{self.api_key_env}')"
        )

    @classmethod
    def from_dict(cls, data):
        return cls(
            config_name=data["config_name"],
            type=data["type"],
            model=data["model"],
            base_url=data.get("base_url"),
            api_key_env=data.get("api_key_env"),
            temperature=data.get("temperature", 0),
            stream=data.get("stream", False),
        )



class AttemptDetail(BaseModel):
    """用于存储单次尝试的代码和结果"""
    code: str
    result: str


class TaskResult(BaseModel):
    """表示任务的最终结果，记录了每一次的尝试详情"""
    attempts: Dict[int, AttemptDetail] = Field(default_factory=dict)  # 使用 Field(default_factory=dict)
    is_success: bool = False
    new_knowledge: Dict[str, Any] = Field(default_factory=dict) # 任务执行过程中获得的新知识
    suggested_next_steps: List[str] = Field(default_factory=list) # Actor 建议的后续步骤
    should_retry: bool = False # 是否建议重试


# ====== 专家反馈数据结构 ======
class FeedbackSource(str, Enum):
    """反馈来源类型"""
    SUCCESS_CASE = "success_case"  # 成功案例库
    EXTERNAL_KNOWLEDGE = "external_knowledge"  # 外部知识库
    HUMAN_INTERACTION = "human_interaction"  # 人机交互
    SYSTEM_GENERATED = "system_generated"  # 系统自动生成

class ExpertFeedbackItem(BaseModel):
    """单条专家经验反馈。
    - task_type: 适用的任务类型（如 父任务类型/子任务类型 名称），用于检索匹配
    - condition: 可选的上下文条件（如数据规模、是否类别不平衡、是否时间序列等）
    - recommendation: 明确的推荐做法/模型/方法
    - anti_pattern: 可选的反面教材/坑位
    - rationale: 简要原因或适用场景说明
    - score: 可选的置信度/有效性评分 [0,1]
    - source: 反馈来源类型
    - source_id: 来源标识（如案例ID、文档ID、交互ID等）
    - created_by: 提交者（专家/用户名/系统）
    - created_at: 创建时间字符串
    - tags: 标签列表，便于分类和检索
    """
    task_type: str
    condition: Optional[Dict[str, Any]] = None
    recommendation: str
    anti_pattern: Optional[str] = None
    rationale: Optional[str] = None
    score: Optional[float] = None
    source: FeedbackSource = FeedbackSource.SYSTEM_GENERATED
    source_id: Optional[str] = None
    created_by: Optional[str] = None
    created_at: Optional[str] = None
    tags: List[str] = Field(default_factory=list)


class ExpertFeedbackPack(BaseModel):
    """一组专家经验反馈，用于持久化存储。"""
    items: List[ExpertFeedbackItem] = Field(default_factory=list)



class Subtask(BaseModel):
    task_id: str
    dependent_task_ids: List[str] = []
    instruction: str
    task_type: str = "other"
    task_result: Optional[TaskResult] = Field(default_factory=TaskResult)
    is_finished: bool = False
    # 如果 is_decomposed 为 True，则该任务本身不直接执行，而是执行其 subtasks
    is_decomposed: bool = False
    subtasks: List["Subtask"] = Field(default_factory=list)
    parent_task_id: Optional[str] = None  # 新增：父任务 ID

    def completed(self) -> bool:
        """Check if the task is completed"""
        return self.is_finished or (self.task_result and self.task_result.is_success)
    
    def to_simple_dict(self):
        """只返回核心字段"""
        return {
            "task_id": self.task_id,
            "dependent_task_ids": self.dependent_task_ids,
            "instruction": self.instruction,
            "task_type": self.task_type,
        }


    def to_full_dict(self) -> Dict:
        """
        递归地将整个任务树序列化为字典，方便保存和传输。
        """
        return {
            **self.to_simple_dict(),
            "is_finished": self.is_finished,
            "task_result": self.task_result.model_dump() if self.task_result else None,
            "is_decomposed": self.is_decomposed,
            # 递归调用 to_full_dict，构建完整的层级结构
            "subtasks": [s.to_full_dict() for s in self.subtasks], 
            "parent_task_id": self.parent_task_id,
        }
    
    def draw_table(self):
        """
        以表格形式打印当前任务的直接子任务信息。
        """
        # 将 self.child_tasks 重命名为 self.subtasks
        if self.subtasks:
            try:
                from texttable import Texttable
                table = Texttable()
                table.set_deco(Texttable.HEADER)
                table.set_cols_align(["c", "l", "l"]) # 任务类型和指令左对齐
                table.set_cols_width([8, 20, 60]) # 调整宽度
                table.add_row(["任务ID", "任务类型", "指令"])
                for task in self.subtasks:
                    table.add_row(
                        [
                            task.task_id,
                            task.task_type,
                            task.instruction,
                        ]
                    )
                # 假设 logger 是一个已配置好的日志记录器
                logger.special(table.draw())
            except ImportError:
                print("请安装 `pip install texttable` 来使用表格绘制功能。")
    
    def get_last_attempt(self) -> Optional[AttemptDetail]:
            if self.task_result and self.task_result.attempts:
                last_attempt_num = max(self.task_result.attempts.keys())
                return self.task_result.attempts[last_attempt_num]
            return None

    def get_last_subtask(self) -> Optional["Subtask"]:
        if self.subtasks:
            return self.subtasks[-1].get_last_subtask()
        else:
            return self
    
    @classmethod
    def from_dict(cls, data: Dict) -> "Subtask":
        """
        从字典递归地创建任务及其所有子任务。
        """
        # 递归地为 'subtasks' 列表中的每个字典创建 Subtask 实例
        subtasks_data = data.get("subtasks", [])
        parent_task_id = data.get("parent_task_id", None)
        subtasks = [cls.from_dict(sub_data) for sub_data in subtasks_data]
        
        return cls(
            task_id=data.get("task_id", ""),
            dependent_task_ids=data.get("dependent_task_ids", []),
            instruction=data.get("instruction", ""),
            task_type=data.get("task_type", "other"),
            is_finished=data.get("is_finished", False),
            is_decomposed=data.get("is_decomposed", False),
            subtasks=subtasks,
            parent_task_id=parent_task_id,
        )
Subtask.model_rebuild()

class Plan(BaseModel):
    raw_question: str
    subtasks: List[Subtask]
    current_task: Optional[Subtask] = None
    # 新增：澄清后的需求与澄清详情（可选）
        

    def __repr__(self):
        return f"Plan(RawQuestion={self.raw_question}, Subtasks={self.subtasks})"

    def completed(self) -> bool:
        """Check if all top-level tasks are completed"""
        if not self.subtasks:
            return True
        return all(st.completed() for st in self.subtasks)

    def process_task_result(self, attempts_data: Dict[int, AttemptDetail]):
        #todo:可添加对结果的评价，也可设计重新执行，并给出用户的建议   (result_review)
        self.current_task.task_result.attempts.update(attempts_data)
    
    def update_plan(self,taskResult:TaskResult):
        # todo:初始生成任务分解树及更新结果时均可调用
        self.current_task.task_result=taskResult
    
    
    def _hierarchical_sort(self) -> List[Subtask]:
        """  
        【已弃用】请改用 `new_function()`。
        
        该函数将在 v2.0 版本中移除。
        """
        warnings.warn(
            "old_function() is deprecated and will be removed in version 2.0. "
            "Use new_function() instead.",
            DeprecationWarning,
            stacklevel=2  # 让警告指向调用者，而不是本函数内部
        )
        # """
        # 新增: 分层排序方法，取代原有的 _topological_sort。
        # 1. 按父任务ID排序 (t1, t2, t3...)
        # 2. 在每个父任务内部，按子任务ID排序 (t1.1, t1.2...)
        # """
        all_tasks = self.get_all_tasks_flat()
        
        # 定义一个排序键函数
        def sort_key(task: Subtask):
            parts = task.task_id.strip('t').split('.')
            return [int(p) for p in parts]
        return sorted(all_tasks, key=sort_key)
    

    def get_all_tasks_flat(self) -> List[Subtask]:
        """新增: 递归获取整个任务树的扁平化列表。"""
        all_tasks = []
        def _recurse(tasks: List[Subtask]):
            for task in tasks:
                all_tasks.append(task)
                if task.subtasks:
                    _recurse(task.subtasks)
        _recurse(self.subtasks)
        return all_tasks
    
    def get_task_by_id(self, task_id: str) -> Optional[Subtask]:
        """修改: 递归查找整个任务树，根据 ID 获取任务。"""
        def _find_recursive(tasks: List[Subtask]):
            for task in tasks:
                if task.task_id == task_id:
                    return task
                if task.subtasks:
                    found = _find_recursive(task.subtasks)
                    if found:
                        return found
            return None
        return _find_recursive(self.subtasks)
    
    def get_current_level_tasks(self) -> List[Subtask]:
        """获取与 current_task 同层级的所有任务。"""
        if not self.current_task:
            return []
        
        if self.current_task.parent_task_id:
            parent = self.get_task_by_id(self.current_task.parent_task_id)
            if parent:
                return parent.subtasks
        
        # 如果没有父任务 ID，说明在顶层
        return self.subtasks

    def get_current_level_unfinished_tasks(self) -> List[Subtask]:
        """获取与 current_task 同层级的所有未完成任务。"""
        tasks = self.get_current_level_tasks()
        return [t for t in tasks if not t.completed()]
    
    def get_current_level_finished_tasks(self) -> List[Subtask]:
        """获取与 current_task 同层级的所有已完成任务。"""
        tasks = self.get_current_level_tasks()
        return [t for t in tasks if t.completed()]
    
    def get_plan_status(self, exclude: List[str] = []) -> str:
        import utils
        from prompt.task_types import TaskType
        from prompt.planner import PLAN_STATUS_PROMPT

        # 裁剪与截断配置（可按需调整）
        MAX_FINISHED_TASKS = 3
        MAX_CODE_PER_TASK = 800
        MAX_RESULT_SNIPPET = 800
        MAX_CURRENT_ATTEMPTS = 1

        def truncate_middle(s: str, max_len: int) -> str:
            s = s or ""
            if len(s) <= max_len:
                return s
            head_len = 400
            tail_len = max_len - head_len
            head = s[:head_len].rstrip()
            tail = s[-tail_len:].lstrip()
            return f"{head}\n\n...（已截断）...\n\n{tail}"

        finished_parent_tasks_desc = []
        finished_parent_tasks = self.get_finished_parent_tasks()
        if len(finished_parent_tasks) > MAX_FINISHED_TASKS:
            finished_parent_tasks = finished_parent_tasks[-MAX_FINISHED_TASKS :]
        finished_parent_tasks_desc = []
        
        # 1. 准备列表
        parent_tasks_to_show = self.get_finished_parent_tasks()
        if len(parent_tasks_to_show) > MAX_FINISHED_TASKS:
            parent_tasks_to_show = parent_tasks_to_show[-MAX_FINISHED_TASKS:]
            
        sibling_tasks_to_show = []
        if self.current_task and self.current_task.parent_task_id:
            sibling_tasks_to_show = self.get_current_level_finished_tasks() or []
            if len(sibling_tasks_to_show) > MAX_FINISHED_TASKS:
                sibling_tasks_to_show = sibling_tasks_to_show[-MAX_FINISHED_TASKS:]

        # 2. 确定哪个任务是“上一个任务”（Context中最后一个展示的任务），仅保留它的完整代码
        last_shown_task_id = None
        if sibling_tasks_to_show:
            last_shown_task_id = sibling_tasks_to_show[-1].task_id
        elif parent_tasks_to_show:
            last_shown_task_id = parent_tasks_to_show[-1].task_id

        # 3. 处理父任务历史
        for task in parent_tasks_to_show:
            finished_parent_tasks_desc.append(f"===任务 {task.task_id} 指令==={task.instruction}")
            last_subtask = task.get_last_subtask()
            last_attempt= last_subtask.get_last_attempt()
            if not last_attempt:
                continue
            raw_code = last_attempt.code or ""
            cleaned = utils.remove_comments(raw_code).strip()
            cleaned = utils.remove_imports_and_empty_lines(cleaned).strip()
            
            is_last_shown = (task.task_id == last_shown_task_id)
            
            if cleaned:
                if is_last_shown:
                    code_snippet = cleaned
                else:
                    code_snippet = truncate_middle(cleaned, MAX_CODE_PER_TASK)
                finished_parent_tasks_desc.append(f"---任务{last_subtask.task_id}代码如下(已移除导入语句和空行)---{code_snippet}")
            raw_result = (last_attempt.result or "").strip()
            if raw_result:
                result_snip = truncate_middle(raw_result, MAX_RESULT_SNIPPET)
                finished_parent_tasks_desc.append(f"---任务{last_subtask.task_id}结果如下---{result_snip}")

        # 4. 处理同级子任务历史
        if self.current_task and self.current_task.parent_task_id:
            parent_id=self.current_task.parent_task_id 
            finished_parent_tasks_desc.append(f"===当前父任务 {parent_id} 下已完成的子任务===")
            for task in sibling_tasks_to_show:  
                finished_parent_tasks_desc.append(f"===任务 {task.task_id} 指令==={task.instruction}")
                last_attempt= task.get_last_attempt()
                if not last_attempt:
                    continue
                raw_code = last_attempt.code or ""
                cleaned = utils.remove_comments(raw_code).strip()
                cleaned = utils.remove_imports_and_empty_lines(cleaned).strip()
                
                is_last_shown = (task.task_id == last_shown_task_id)

                if cleaned:
                    if is_last_shown:
                        code_snippet = cleaned
                    else:
                        code_snippet = truncate_middle(cleaned, MAX_CODE_PER_TASK)
                    finished_parent_tasks_desc.append(f"---代码---{code_snippet}")
                raw_result = (last_attempt.result or "").strip()
                if raw_result:
                    result_snip = truncate_middle(raw_result, MAX_RESULT_SNIPPET)
                    finished_parent_tasks_desc.append(f"---结果（摘要）---{result_snip}")
        finished_tasks_desc ="\n".join(finished_parent_tasks_desc)

        # 当前任务：保留最近若干次尝试并做截断
        current_task_code = ""
        current_task_result = ""
        if self.current_task and self.current_task.task_result:
            attempts_items = sorted(self.current_task.task_result.attempts.items())
            if attempts_items:
                attempts_items = attempts_items[-MAX_CURRENT_ATTEMPTS :]
                attempt_code_parts = []
                attempt_result_parts = []
                for attempt_num, attempt_detail in attempts_items:
                    code = attempt_detail.code or ""
                    result = attempt_detail.result or ""
                    code = utils.remove_comments(code).strip()
                    code = utils.remove_imports_and_empty_lines(code).strip()
                    code_snip = truncate_middle(code, MAX_CODE_PER_TASK)
                    result_snip = truncate_middle(result, MAX_RESULT_SNIPPET)
                    attempt_code_parts.append(f"第{attempt_num}次尝试代码如下---{code_snip}")
                    attempt_result_parts.append(f"第{attempt_num}次尝试结果如下---{result_snip}")

                current_task_code = "\n\n".join(attempt_code_parts)
                current_task_result = "\n\n".join(attempt_result_parts)

        # 组合 Prompt
        if self.current_task:
            task_type_name = self.current_task.task_type
            current_task_instruction = self.current_task.instruction
        else:
            task_type_name = "other"
            current_task_instruction = "N/A"

        task_type = TaskType.find_type(task_type_name)
        guidance = task_type.guidance if task_type else ""

        prompt = PLAN_STATUS_PROMPT.format(
            finished_tasks_desc=finished_tasks_desc,
            current_task=current_task_instruction,
            current_task_code=current_task_code if "code" not in exclude else "",
            current_task_result=current_task_result if "task_result" not in exclude else "",
            guidance=guidance,
        )

        return prompt
    

    def get_finished_parent_tasks(self) -> List[Subtask]:
        """获取所有已完成的父任务。"""
        finished_parent_tasks = []
        for task in self.subtasks:
            if (task.is_finished):
                finished_parent_tasks.append(task)
            elif task.is_decomposed and task.subtasks:
                if task.get_last_subtask().is_finished:
                    finished_parent_tasks.append(task)
        return finished_parent_tasks
    
    def get_finished_tasks(self) -> List[Subtask]:
        """修改: 递归获取所有已完成的任务。"""
        # 利用 get_all_tasks_flat() 方法简化实现
        return [task for task in self.get_all_tasks_flat() if task.is_finished and not task.is_decomposed]
    
    
    def _topological_sort(self):
        """
        仅对父任务排序。
        """
        tasks = self.subtasks
        
        task_map = {task.task_id: task for task in tasks}
        
        dependencies = {task.task_id: set(task.dependent_task_ids) for task in tasks}
        sorted_order = []
        visited = set()
        recursion_stack = set()
        
        all_task_ids = sorted(list(task_map.keys()))

        def visit(task_id):
            # 检查是否存在循环依赖
            if task_id in recursion_stack:
                raise Exception(f"父任务的依赖关系中存在循环: {task_id}")
            if task_id in visited:
                return

            visited.add(task_id)
            recursion_stack.add(task_id)

            # 确保 task_id 在父任务的依赖图中
            if task_id in task_map:
                # 遍历其依赖的其他父任务
                for dependent_id in dependencies.get(task_id, []):
                    visit(dependent_id)
            
            recursion_stack.remove(task_id)
            
            if task_id in task_map:
                sorted_order.append(task_map[task_id])

        for task_id in all_task_ids:
            if task_id not in visited:
                visit(task_id)
        
        return sorted_order
    
    
    @classmethod
    def from_dict(cls, data):
        # 如果直接传的是子任务列表
        if isinstance(data, list):
            subtasks = [Subtask.from_dict(subtask) for subtask in data]
            return cls(raw_question="", subtasks=subtasks)

        # 如果是包含 raw_question + subtasks 的字典
        if isinstance(data, dict) and "subtasks" in data:
            subtasks = [Subtask.from_dict(subtask) for subtask in data.get("subtasks", [])]
            return cls(
                raw_question=data.get("raw_question", ""),
                subtasks=subtasks,
            )

    def to_simple_dict(self):
        return {
            "raw_question": self.raw_question,
            "subtasks": [subtask.to_simple_dict() for subtask in self.subtasks],
        }


    def clone(self):
         return copy.deepcopy(self)

    def draw_table(self):
        """以表格形式打印任务分解"""
        from texttable import Texttable

        table = Texttable()
        table.set_deco(Texttable.HEADER)
        table.set_cols_align(["c", "c", "c", "c"])
        table.set_cols_width([5, 20, 50, 20])
        table.add_row(["ID", "依赖任务", "说明", "任务类型"])
        for task in self.subtasks:
            table.add_row(
                [
                    task.task_id,
                    str(task.dependent_task_ids),
                    task.instruction,
                    task.task_type,
                ]
            )
        logger.special("【任务分解表格】\n" ,
                       table.draw(), )

