"""
Orchestra Core - 配置管理

提供统一的配置加载和管理机制。
"""
from __future__ import annotations

from dataclasses import dataclass, field
from pathlib import Path
from typing import Dict, List, Optional

import yaml


@dataclass
class OrchestraConfig:
    """Orchestra 编排框架配置"""
    
    # 模型配置
    main_model: str  # MainAgent 使用的模型
    sub_models: List[str]  # SubAgent 可用的模型列表
    
    # 执行配置
    max_steps: int = 30  # SubAgent 单次执行最大步数
    max_attempts: int = 10  # MainAgent 最大尝试次数
    max_concurrency: int = 1  # 并发任务数
    
    # 沙箱配置
    sandbox: str = "docker"  # 沙箱类型: docker | e2b | daytona | local
    sandbox_timeout: int = 600  # 沙箱超时时间（秒）
    
    # 输出配置
    result_folder: Path = field(default_factory=lambda: Path("workspace/logs"))
    trajectory_dir: Optional[Path] = None
    csv_summary_path: Optional[Path] = None
    timestamp: Optional[str] = None
    
    # 环境初始化
    env_init: Optional[Dict[str, str]] = None
    
    # 模型名称遮蔽（避免 MainAgent 对特定模型产生偏好）
    mask_model_names: bool = True
    
    @classmethod
    def load(cls, config_path: Path | str) -> "OrchestraConfig":
        """从 YAML 文件加载配置"""
        config_path = Path(config_path)
        with config_path.open("r", encoding="utf-8") as f:
            raw = yaml.safe_load(f) or {}
        
        # 必填字段校验
        main_model = raw.get("main_model")
        if not main_model:
            raise ValueError("main_model is required")
        
        sub_models = raw.get("sub_models")
        if not sub_models or not isinstance(sub_models, list):
            raise ValueError("sub_models must be a non-empty list")
        
        # 路径解析
        result_folder = cls._resolve_path(
            raw.get("result_folder", "workspace/logs"),
            config_path
        )
        
        trajectory_dir = raw.get("trajectory_dir")
        if trajectory_dir:
            trajectory_dir = cls._resolve_path(trajectory_dir, config_path)
        
        csv_path = raw.get("csv_summary_path")
        if csv_path:
            csv_path = cls._resolve_path(csv_path, config_path)
        
        return cls(
            main_model=str(main_model),
            sub_models=[str(m) for m in sub_models],
            max_steps=int(raw.get("max_steps", 30)),
            max_attempts=int(raw.get("max_attempts", 10)),
            max_concurrency=int(raw.get("max_concurrency", 1)),
            sandbox=str(raw.get("sandbox", "docker")),
            sandbox_timeout=int(raw.get("sandbox_timeout", 600)),
            result_folder=result_folder,
            trajectory_dir=trajectory_dir,
            csv_summary_path=csv_path,
            env_init=raw.get("env_init"),
            mask_model_names=raw.get("mask_model_names", True),
        )
    
    @staticmethod
    def _resolve_path(path_str: str, config_path: Path) -> Path:
        """解析相对路径（相对于配置文件或项目根目录）"""
        path = Path(path_str)
        if path.is_absolute():
            return path
        
        # 尝试相对于配置文件
        rel_to_config = config_path.parent / path
        if rel_to_config.exists():
            return rel_to_config.resolve()
        
        # 尝试相对于当前工作目录
        return path.resolve()
    
    def setup_directories(self) -> None:
        """创建必要的输出目录"""
        self.result_folder.mkdir(parents=True, exist_ok=True)
        if self.trajectory_dir:
            self.trajectory_dir.mkdir(parents=True, exist_ok=True)
        if self.csv_summary_path:
            self.csv_summary_path.parent.mkdir(parents=True, exist_ok=True)
