"""
Orchestra Core - DelegateTaskTool (委派工具)

核心工具，用于 MainAgent 将任务委派给 SubAgent:
1. 创建 SubAgent 实例
2. 在隔离环境中执行任务
3. 收集执行结果返回给 MainAgent

设计要点:
- 每次委派可以指定不同的模型
- 支持任务上下文传递
- 自动生成执行轨迹摘要
"""
from __future__ import annotations

from typing import Any, Callable, Dict, List, Optional

from orchestra_core.interfaces import (
    BaseTool,
    Environment,
    LLMInterface,
    LevelResult,
    StepRecord,
)
from orchestra_core.sub_agent import SubAgent
from orchestra_core.sub_runner import SubAgentRunner


class DelegateTaskTool(BaseTool):
    """委派任务工具 - 将任务分配给 SubAgent 执行"""
    
    def __init__(
        self,
        env: Environment,
        runner: SubAgentRunner,
        models: List[str],
        llm_factory: Callable[[str], LLMInterface],
        trace_summarizer: Optional[Callable[[List[StepRecord], str, str], str]] = None,
    ):
        """
        初始化委派工具
        
        Args:
            env: 执行环境
            runner: SubAgent 运行器
            models: 可用的模型列表
            llm_factory: LLM 实例工厂函数
            trace_summarizer: 轨迹摘要生成函数（可选）
        """
        self._env = env
        self._runner = runner
        self._models = models
        self._llm_factory = llm_factory
        self._trace_summarizer = trace_summarizer
    
    @property
    def name(self) -> str:
        return "delegate_task"
    
    @property
    def description(self) -> str:
        return "Delegate a task to SubAgent for execution in isolated environment"
    
    @property
    def parameters(self) -> Dict[str, Any]:
        return {
            "type": "object",
            "properties": {
                "task_instruction": {
                    "type": "string",
                    "description": "Task description for SubAgent"
                },
                "context": {
                    "type": "string",
                    "description": "Additional context/hints for SubAgent"
                },
                "model": {
                    "type": "string",
                    "description": f"Model to use. Must be one of: {self._models}",
                    "enum": self._models
                },
                "tools": {
                    "type": "array",
                    "items": {"type": "string"},
                    "description": "Tools available to SubAgent (optional)"
                },
            },
            "required": ["task_instruction", "model"]
        }
    
    async def __call__(
        self,
        task_instruction: str,
        model: str,
        context: str = "",
        tools: Optional[List[str]] = None
    ) -> Dict[str, Any]:
        """
        执行委派
        
        Args:
            task_instruction: 任务描述
            model: 使用的模型
            context: 额外上下文
            tools: 允许的工具列表
        
        Returns:
            Dict: 执行结果
        """
        if model not in self._models:
            return {
                "error": f"Invalid model: {model}",
                "steps_taken": 0,
                "done": False
            }
        
        # 清理已存在的容器
        if hasattr(self._env, 'cleanup_container_if_exists'):
            await self._env.cleanup_container_if_exists()
        
        max_retries = 3
        last_error = None
        
        for retry in range(max_retries):
            original_instruction = None
            try:
                # 保存原始 instruction
                original_instruction = getattr(self._env, 'instruction', None)
                if hasattr(self._env, 'instruction'):
                    self._env.instruction = task_instruction
                
                # 创建 SubAgent
                llm = self._llm_factory(model)
                sub_agent = SubAgent(
                    llm=llm,
                    context=context,
                    sub_model=model,
                    allowed_tools=tools,
                )
                
                # 执行
                result = await self._runner.run(sub_agent, self._env)
                
                # 提取 finish_result
                finish_result = None
                if result.trace:
                    last = result.trace[-1]
                    if last.info.get("finished") and last.info.get("finish_result"):
                        finish_result = last.info["finish_result"]
                
                # 生成轨迹摘要
                trace_summary = ""
                if self._trace_summarizer and result.trace:
                    try:
                        trace_summary = await self._trace_summarizer(
                            result.trace, model, task_instruction
                        )
                    except Exception:
                        trace_summary = f"Steps: {len(result.trace)}"
                
                return {
                    "model": model,
                    "tools_assigned": tools,
                    "steps_taken": result.steps,
                    "done": result.done,
                    "cost": result.cost,
                    "finish_result": finish_result,
                    "trace": result.trace,
                    "trace_summary": trace_summary,
                    "retries": retry,
                    "statistics": {
                        "total_steps": result.steps,
                        "max_steps": 30,
                        "completed": result.done
                    },
                }
                
            except TimeoutError as e:
                last_error = e
                if retry < max_retries - 1 and hasattr(self._env, 'cleanup_container_if_exists'):
                    await self._env.cleanup_container_if_exists()
                    continue
                    
            except Exception as e:
                last_error = e
                break
                
            finally:
                # 恢复原始 instruction
                if original_instruction is not None and hasattr(self._env, 'instruction'):
                    self._env.instruction = original_instruction
        
        return {
            "error": str(last_error) if last_error else "Unknown error",
            "steps_taken": 0,
            "done": False,
            "cost": 0.0,
            "retries": retry + 1,
            "is_error": True
        }
