"""
Configuration classes for OfficeArena agents.

This module defines base and derived configuration classes for different agent types,
providing type safety and validation for agent initialization.
"""

from abc import ABC, abstractmethod
from dataclasses import dataclass, field
from typing import Any, Dict, Optional
import os


@dataclass
class AgentConfig(ABC):
    """
    Base configuration class for all OfficeArena agents.

    This class provides common configuration parameters and validation
    that all agents should support.
    """

    # Display configuration
    display_width: int = 1024
    display_height: int = 768

    # Common model parameters
    temperature: float = 0.0
    top_p: float = 1.0

    # Timeout and retry settings
    timeout_seconds: int = 60
    max_retries: int = 3

    # Additional configuration
    extra_config: Dict[str, Any] = field(default_factory=dict)

    @property
    def display_size(self) -> Dict[str, int]:
        """Get display size as dictionary for compatibility."""
        return {"width": self.display_width, "height": self.display_height}

    @abstractmethod
    def validate(self) -> None:
        """Validate the configuration and raise ValueError if invalid."""
        pass

    @abstractmethod
    def to_dict(self) -> Dict[str, Any]:
        """Convert configuration to dictionary format for agent initialization."""
        pass


@dataclass
class CUAConfig(AgentConfig):
    """
    Configuration for Computer Use Agent (CUA) models.

    Supports both Azure OpenAI and standard OpenAI endpoints.
    """

    # Model configuration
    model_name: str = "computer-use-preview"

    # API configuration
    api_key: Optional[str] = None
    base_url: Optional[str] = None
    endpoint: Optional[str] = None  # "azure" for Azure, None for standard OpenAI
    api_version: Optional[str] = None  # Required for Azure

    # CUA-specific settings
    environment: str = "browser"  # "browser" or "desktop"
    truncation: str = "auto"

    def __post_init__(self):
        """Auto-populate from environment variables if not provided."""
        if self.api_key is None:
            self.api_key = os.getenv("CUA_API_KEY") or os.getenv("OPENAI_API_KEY")

        if self.base_url is None:
            self.base_url = os.getenv("CUA_BASE_URL") or os.getenv("OPENAI_BASE_URL")

        if self.endpoint is None:
            self.endpoint = os.getenv("CUA_ENDPOINT")

        if self.api_version is None:
            self.api_version = os.getenv("CUA_API_VERSION", "2025-04-01-preview")

    def validate(self) -> None:
        """Validate CUA configuration."""
        if not self.model_name:
            raise ValueError("model_name is required for CUA agent")

        if self.endpoint == "azure":
            if not self.base_url:
                raise ValueError("base_url is required for Azure endpoint")
            if not self.api_version:
                raise ValueError("api_version is required for Azure endpoint")
        else:
            if not self.api_key:
                raise ValueError("api_key is required for non-Azure endpoints")

        if self.display_width <= 0 or self.display_height <= 0:
            raise ValueError("Display dimensions must be positive")

        if not (0.0 <= self.temperature <= 2.0):
            raise ValueError("Temperature must be between 0.0 and 2.0")

        if not (0.0 <= self.top_p <= 1.0):
            raise ValueError("top_p must be between 0.0 and 1.0")

    def to_dict(self) -> Dict[str, Any]:
        """Convert to dictionary format for CUA agent initialization."""
        return {
            "model_name": self.model_name,
            "api_key": self.api_key,
            "base_url": self.base_url,
            "endpoint": self.endpoint,
            "api_version": self.api_version,
            "display_size": self.display_size,
            "temperature": self.temperature,
            "top_p": self.top_p,
            "environment": self.environment,
            "truncation": self.truncation,
            **self.extra_config,
        }


@dataclass
class UITARSConfig(AgentConfig):
    """
    Configuration for UI-TARS agent.

    UI-TARS uses HTTP API endpoints for model inference.
    """

    # Model configuration
    model_name: str = "uitars-v1"

    # API configuration
    endpoint_url: str = ""
    token: Optional[str] = None

    # UI-TARS specific settings
    scroll_step: int = 10
    wait_duration: int = 5  # seconds

    def __post_init__(self):
        """Auto-populate from environment variables if not provided."""
        if not self.endpoint_url:
            self.endpoint_url = os.getenv("UITARS_ENDPOINT_URL", "")

        if self.token is None:
            self.token = os.getenv("UITARS_TOKEN")

    def validate(self) -> None:
        """Validate UI-TARS configuration."""
        if not self.endpoint_url:
            raise ValueError("endpoint_url is required for UI-TARS agent")

        if not self.token:
            raise ValueError("token is required for UI-TARS agent")

        if self.display_width <= 0 or self.display_height <= 0:
            raise ValueError("Display dimensions must be positive")

        if self.scroll_step <= 0:
            raise ValueError("scroll_step must be positive")

        if self.wait_duration <= 0:
            raise ValueError("wait_duration must be positive")

    def to_dict(self) -> Dict[str, Any]:
        """Convert to dictionary format for UI-TARS agent initialization."""
        return {
            "model_name": self.model_name,
            "endpoint_url": self.endpoint_url,
            "token": self.token,
            "display_size": self.display_size,
            "scroll_step": self.scroll_step,
            "wait_duration": self.wait_duration,
            **self.extra_config,
        }


@dataclass
class ClaudeConfig(AgentConfig):
    """
    Configuration for Claude agent using Anthropic's API.

    Supports both direct Anthropic API and litellm for unified interface.
    """

    # Model configuration
    model_name: str = "claude-3-5-sonnet-20241022"

    # API configuration
    api_key: Optional[str] = None
    base_url: Optional[str] = None

    # Claude-specific settings
    max_tokens: int = 4096
    use_litellm: bool = True

    # Computer use settings
    computer_use_enabled: bool = True

    def __post_init__(self):
        """Auto-populate from environment variables if not provided."""
        if self.api_key is None:
            self.api_key = os.getenv("ANTHROPIC_API_KEY") or os.getenv("CLAUDE_API_KEY")

        if self.base_url is None:
            self.base_url = os.getenv("ANTHROPIC_BASE_URL") or os.getenv(
                "CLAUDE_BASE_URL"
            )

    def validate(self) -> None:
        """Validate Claude configuration."""
        if not self.model_name:
            raise ValueError("model_name is required for Claude agent")

        if not self.api_key:
            raise ValueError("api_key is required for Claude agent")

        if self.display_width <= 0 or self.display_height <= 0:
            raise ValueError("Display dimensions must be positive")

        if not (0.0 <= self.temperature <= 1.0):
            raise ValueError("Temperature must be between 0.0 and 1.0")

        if not (0.0 <= self.top_p <= 1.0):
            raise ValueError("top_p must be between 0.0 and 1.0")

        if self.max_tokens <= 0:
            raise ValueError("max_tokens must be positive")

    def to_dict(self) -> Dict[str, Any]:
        """Convert to dictionary format for Claude agent initialization."""
        return {
            "model_name": self.model_name,
            "api_key": self.api_key,
            "base_url": self.base_url,
            "display_size": self.display_size,
            "temperature": self.temperature,
            "top_p": self.top_p,
            "max_tokens": self.max_tokens,
            "use_litellm": self.use_litellm,
            "computer_use_enabled": self.computer_use_enabled,
            **self.extra_config,
        }


# Factory function for creating configurations
def create_agent_config(agent_type: str, **kwargs) -> AgentConfig:
    """
    Factory function to create agent configurations.

    Args:
        agent_type: Type of agent ("cua", "uitars", "claude")
        **kwargs: Configuration parameters

    Returns:
        Configured agent config instance

    Raises:
        ValueError: If agent_type is not supported
    """
    agent_type = agent_type.lower()

    if agent_type == "cua":
        return CUAConfig(**kwargs)
    elif agent_type == "uitars":
        return UITARSConfig(**kwargs)
    elif agent_type == "claude":
        return ClaudeConfig(**kwargs)
    else:
        raise ValueError(
            f"Unsupported agent type: {agent_type}. Supported types: cua, uitars, claude"
        )


# Utility functions for configuration validation
def validate_all_configs(*configs: AgentConfig) -> None:
    """Validate multiple configurations at once."""
    for config in configs:
        config.validate()


def merge_config_dict(
    config: AgentConfig, override_dict: Dict[str, Any]
) -> Dict[str, Any]:
    """
    Merge a configuration with override dictionary.

    Args:
        config: Base configuration
        override_dict: Dictionary of values to override

    Returns:
        Merged configuration dictionary
    """
    merged = config.to_dict()
    merged.update(override_dict)
    return merged
