from typing import List, Optional, Tuple, Dict, Set
from enum import Enum
import re


class Direction(Enum):
    """Queried Direction"""
    UP = "up"
    DOWN = "down"
    LEFT = "left"
    RIGHT = "right"


class RushHourState:
    """
    Rush Hour Game State
    The board is represented as a 2D grid, and each vehicle is represented by a character.
    """
    
    def __init__(self, board_config: str, rows: int = 6, cols: int = 6):
        """
        Args:
            board_config: Board configuration string (e.g. 'oxCCDDooxoMoIoAAMoIoKLFFJoKLooJGGHHH')
            rows: Number of rows
            cols: Number of columns
        """
        self.rows = rows
        self.cols = cols
        self.board_config = board_config
        
        self.board = [['.' for _ in range(cols)] for _ in range(rows)]
        
        # Store vehicle information: {vehicle character: [(row, col), ...]}
        self.vehicles: Dict[str, List[Tuple[int, int]]] = {}
        
        # Identify the red car (target vehicle)
        self.target_car = None
        
        # Parse the board configuration
        self._parse_board_config(board_config)
    
    def _parse_board_config(self, config: str):
        """Parse the board configuration string to create the board and vehicle information"""
        # Read the config by rows (row-major order)
        idx = 0
        for row in range(self.rows):
            for col in range(self.cols):
                if idx < len(config):
                    char = config[idx]
                    if char == 'x':
                        self.board[row][col] = 'x'  # Wall
                    elif char == 'o':
                        self.board[row][col] = '.'  # Empty space
                    elif char.isalpha():
                        self.board[row][col] = char
                        # Store vehicle information
                        if char not in self.vehicles:
                            self.vehicles[char] = []
                        self.vehicles[char].append((row, col))
                        # Identify the red car (AA - vehicles with the same character)
                        if char == 'A' and self.target_car is None:
                            self.target_car = 'A'
                    else:
                        # Unknown characters are treated as empty spaces
                        self.board[row][col] = '.'
                    idx += 1
                else:
                    # If the config is shorter, the remaining are empty spaces
                    self.board[row][col] = '.'
        
        # Determine the vehicle direction (horizontal/vertical)
        self.vehicle_orientations = {}
        for vehicle, positions in self.vehicles.items():
            if len(positions) > 1:
                # Determine the direction by comparing the first two positions
                r1, c1 = positions[0]
                r2, c2 = positions[1]
                if r1 == r2:
                    self.vehicle_orientations[vehicle] = 'horizontal'
                else:
                    self.vehicle_orientations[vehicle] = 'vertical'
            else:
                # A single cell vehicle is assumed to be horizontal
                self.vehicle_orientations[vehicle] = 'horizontal'
    
    def copy(self) -> "RushHourState":
        """Return a copy of the state"""
        new_state = RushHourState(self.board_config, self.rows, self.cols)
        new_state.board = [row.copy() for row in self.board]
        new_state.vehicles = {v: pos.copy() for v, pos in self.vehicles.items()}
        new_state.vehicle_orientations = self.vehicle_orientations.copy()
        new_state.target_car = self.target_car
        return new_state
    
    def move_vehicle(self, vehicle: str, direction: str, num_moves: int = 1) -> bool:
        """
        Move a vehicle.
        
        Args:
            vehicle: Vehicle character (e.g. 'A', 'B', 'C')
            direction: Direction of movement ('up', 'down', 'left', 'right')
            num_moves: Number of cells to move (default: 1)
            
        Returns:
            Success of the move
        """
        if vehicle not in self.vehicles:
            return False
        
        if num_moves < 1:
            return False
        
        # Normalize the direction
        direction = direction.strip().lower()
        
        # Check if the direction is valid
        valid_directions = ['up', 'down', 'left', 'right']
        if direction not in valid_directions:
            return False
        
        # Check the vehicle direction
        orientation = self.vehicle_orientations.get(vehicle, 'horizontal')
        
        # Determine the movement direction
        if orientation == 'horizontal':
            # Horizontal vehicle: only left and right movement is possible
            if direction in ['up', 'down']:
                return False
            if direction == 'right':
                delta_row, delta_col = 0, 1
            else:  # left
                delta_row, delta_col = 0, -1
        else:
            # Vertical vehicle: only up and down movement is possible
            if direction in ['left', 'right']:
                return False
            if direction == 'down':
                delta_row, delta_col = 1, 0
            else:  # up
                delta_row, delta_col = -1, 0
        
        # Multiple cell movement: check if each cell is movable
        for move_step in range(1, num_moves + 1):
            step_delta_row = delta_row * move_step
            step_delta_col = delta_col * move_step
            if not self._can_move(vehicle, step_delta_row, step_delta_col):
                return False
        
        # If all cells are movable, move the vehicle
        final_delta_row = delta_row * num_moves
        final_delta_col = delta_col * num_moves
        
        # Update the vehicle positions
        new_positions = []
        for row, col in self.vehicles[vehicle]:
            new_row = row + final_delta_row
            new_col = col + final_delta_col
            new_positions.append((new_row, new_col))
        
        # Update the board
        for row, col in self.vehicles[vehicle]:
            self.board[row][col] = '.'
        
        # Place the vehicle at the new positions
        for row, col in new_positions:
            self.board[row][col] = vehicle
        
        # Update the vehicle information
        self.vehicles[vehicle] = new_positions
        
        return True
    
    def _can_move(self, vehicle: str, delta_row: int, delta_col: int) -> bool:
        """Check if the vehicle can move"""
        if vehicle not in self.vehicles:
            return False
        
        # Check the positions to move
        for row, col in self.vehicles[vehicle]:
            new_row = row + delta_row
            new_col = col + delta_col
            
            # Check the board range
            if new_row < 0 or new_row >= self.rows or new_col < 0 or new_col >= self.cols:
                return False
            
            # Check the wall
            if self.board[new_row][new_col] == 'x':
                return False
            
            # Check the other vehicles (excluding the own vehicle)
            cell_content = self.board[new_row][new_col]
            if cell_content != '.' and cell_content != vehicle:
                return False
        
        return True
    
    def is_complete(self) -> bool:
        """Check if the target vehicle (red car) has reached the right exit"""
        if self.target_car is None or self.target_car not in self.vehicles:
            return False
        
        # Check the positions of the red car
        positions = self.vehicles[self.target_car]
        
        # Check if the red car is in the rightmost column
        for row, col in positions:
            if col == self.cols - 1:  # Rightmost column
                return True
        
        return False
    
    def to_ascii(self) -> str:
        """Convert to ASCII representation"""
        lines = []
        for row in self.board:
            lines.append(''.join(row))
        return '\n'.join(lines)
    
    def to_string(self) -> str:
        """Simple string representation"""
        return self.to_ascii()
    
    def __eq__(self, other):
        """Compare states"""
        if not isinstance(other, RushHourState):
            return False
        return self.board == other.board
    
    def __hash__(self):
        """Make hashable"""
        return hash(tuple(tuple(row) for row in self.board))


class RushHourAction:
    """
    Rush Hour Action
    """
    
    def __init__(self, vehicle: str, direction: str, num_moves: int = 1):
        """
        Args:
            vehicle: Vehicle character (e.g. 'A', 'B', 'C', or 'AA' which will be normalized to 'A')
            direction: Direction of movement ('up', 'down', 'left', 'right')
            num_moves: Number of cells to move (default: 1)
        """
        # Normalize vehicle name: 'AA' -> 'A' (since board uses single 'A' for red car)
        vehicle = vehicle.upper()
        if vehicle == 'AA':
            vehicle = 'A'
        self.vehicle = vehicle
        self.direction = direction.strip().lower()
        self.num_moves = int(num_moves) if num_moves is not None else 1
    
    @staticmethod
    def from_serialized(action_str: str) -> "RushHourAction":
        """
        Parse the action string.
        Format: move('A', 'right') or move('A', 'right', 2) or move("A", "left", 3)
        
        Args:
            action_str: Action string
            
        Returns:
            RushHourAction object
        """
        action_str = action_str.strip()
        
        # move('vehicle', 'direction', num_moves) or move(vehicle, direction, num_moves) format
        # Allow multiple characters for vehicle name (e.g., 'AA', 'aa')
        # Support both quoted and unquoted formats
        pattern = r"move\s*\(\s*['\"]?([A-Za-z]+)['\"]?\s*,\s*['\"]?([A-Za-z]+)['\"]?\s*(?:,\s*(\d+))?\s*\)"
        match = re.match(pattern, action_str, re.IGNORECASE)
        if match:
            vehicle, direction, num_moves_str = match.groups()
            # Normalize vehicle name: 'AA' or 'aa' -> 'A' (since board uses single 'A' for red car)
            vehicle = vehicle.upper()
            if vehicle == 'AA':
                vehicle = 'A'
            direction = direction.strip().lower()
            # Check if the direction is valid
            valid_directions = ['up', 'down', 'left', 'right']
            if direction not in valid_directions:
                raise ValueError(f"Invalid direction: {direction}. Must be one of: {valid_directions}")
            num_moves = int(num_moves_str) if num_moves_str else 1
            return RushHourAction(vehicle, direction, num_moves)
        
        raise ValueError(f"Invalid action format: {action_str}. Expected: move('A', 'right') or move('A', 'right', 2)")
    
    def to_string(self) -> str:
        """Convert the action to a string"""
        if self.num_moves == 1:
            return f"move('{self.vehicle}', '{self.direction}')"
        else:
            return f"move('{self.vehicle}', '{self.direction}', {self.num_moves})"
    
    def __repr__(self):
        return f"RushHourAction(vehicle={self.vehicle}, direction={self.direction}, num_moves={self.num_moves})"
    
    def __eq__(self, other):
        if not isinstance(other, RushHourAction):
            return False
        return (self.vehicle == other.vehicle and 
                self.direction == other.direction and 
                self.num_moves == other.num_moves)

