import logging
from typing import Any, Union

from rllm.agents.agent import Action, BaseAgent, Step, Trajectory

logger = logging.getLogger(__name__)


class GameAgent(BaseAgent):
    """
    Factory class for creating game-specific agents based on task_type.
    This class acts as a wrapper that creates the appropriate game agent based on the task_type parameter.
    """
    from rllm.agents.game_agents.sudoku_agent import SudokuAgent
    from rllm.agents.game_agents.sudoku_agent_new import SudokuAgentNew
    from rllm.agents.game_agents.frozenlake_agent import FrozenLakeAgent
    from rllm.agents.game_agents.tetris_agent import TetrisAgent
    from rllm.agents.game_agents.twentyfortyeight_agent import TwentyFortyEightAgent
    from rllm.agents.game_agents.tower_of_hanoi_agent import TowerOfHanoiAgent
    from rllm.agents.game_agents.rush_hour_agent import RushHourAgent

    SUPPORTED_GAMES = {
        "sudoku": SudokuAgent,
        "sudoku_new": SudokuAgentNew,
        "frozenlake": FrozenLakeAgent,
        "tetris": TetrisAgent,
        "twentyfortyeight": TwentyFortyEightAgent,
        "tower_of_hanoi": TowerOfHanoiAgent,
        "rush_hour": RushHourAgent,
    }
    
    def __init__(self, task_type: str = None, **kwargs):
        """
        Initialize GameAgent with a specific game type.
        
        Args:
            task_type (str): The type of game ("sudoku", "frozenlake", etc.)
            **kwargs: Additional arguments to pass to the specific game agent constructor
        """
        if task_type is None:
            raise ValueError("task_type is required")
        else:
            self.task_type = task_type.lower()
        
        if self.task_type not in self.SUPPORTED_GAMES:
            supported_games = ", ".join(self.SUPPORTED_GAMES.keys())
            raise ValueError(f"Unsupported task_type: {self.task_type}. Supported games: {supported_games}")
        
        # Create the actual game agent
        agent_class = self.SUPPORTED_GAMES[self.task_type]
        self._agent = agent_class(**kwargs)
        
        # Initialize base agent properties
        self._trajectory = Trajectory()
        self.step = 0
    
    @classmethod
    def create_agent(cls, task_type: str, **kwargs) -> BaseAgent:
        """
        Create a game-specific agent based on the task_type.
        
        Args:
            task_type (str): The type of game ("sudoku", "frozenlake", etc.)
            **kwargs: Additional arguments to pass to the agent constructor
            
        Returns:
            BaseAgent: An instance of the appropriate game agent
            
        Raises:
            ValueError: If task_type is not supported
        """
        return cls(task_type, **kwargs)
    
    @classmethod
    def get_supported_games(cls) -> list[str]:
        """
        Get a list of supported game types.
        
        Returns:
            list[str]: List of supported game type names
        """
        return list(cls.SUPPORTED_GAMES.keys())
    
    # Delegate all BaseAgent methods to the actual agent
    def update_from_env(self, observation: Any, reward: float, done: bool, info: dict, **kwargs):
        """Delegate to the actual game agent."""
        return self._agent.update_from_env(observation, reward, done, info, **kwargs)
    
    def update_from_model(self, response: str, **kwargs) -> Action:
        """Delegate to the actual game agent."""
        return self._agent.update_from_model(response, **kwargs)
    
    def reset(self):
        """Delegate to the actual game agent."""
        return self._agent.reset()
    
    @property
    def chat_completions(self) -> list[dict[str, str]]:
        """Delegate to the actual game agent."""
        return self._agent.chat_completions
    
    @property
    def trajectory(self) -> Trajectory:
        """Delegate to the actual game agent."""
        return self._agent.trajectory
    
    def get_current_state(self) -> Step | None:
        """Delegate to the actual game agent."""
        return self._agent.get_current_state()