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


class TowerOfHanoiState:
    def __init__(self, num_disks: int, num_pegs: int, start_peg: int, target_peg: int):
        """
        Args:
            num_disks: number of disks
            num_pegs: number of pegs
            start_peg: starting peg number (1-indexed)
            target_peg: target peg number (1-indexed)
        """
        self.num_disks = num_disks
        self.num_pegs = num_pegs
        self.start_peg = start_peg
        self.target_peg = target_peg
        
        # generate disk names (A, B, C, ..., A is the smallest and the larger the alphabet, the larger the disk)
        # A = smallest disk (size 1), B = size 2, ..., last alphabet = largest disk
        self.disk_names = [chr(ord('A') + i) for i in range(num_disks)]
        
        # peg-wise disk state (stored from top to bottom)
        # e.g. ['E', 'D', 'C', 'B', 'A'] is on top of E (largest), bottom of A (smallest)
        self.pegs: Dict[int, List[str]] = {i: [] for i in range(1, num_pegs + 1)}
        

        for i in range(num_disks - 1, -1, -1):  # reverse order (largest to smallest)
            disk_name = chr(ord('A') + i)  # from last alphabet to A
            self.pegs[start_peg].append(disk_name)
        
        # disk size mapping (alphabet -> size, A is smallest and the larger the alphabet, the larger the disk)
        # A = 1 (smallest), B = 2, ..., last alphabet = num_disks (largest)
        self.disk_sizes = {chr(ord('A') + i): i + 1 for i in range(num_disks)}
    
    def copy(self) -> "TowerOfHanoiState":
        """return a copy of the state"""
        new_state = TowerOfHanoiState(self.num_disks, self.num_pegs, self.start_peg, self.target_peg)
        new_state.pegs = {peg: disks.copy() for peg, disks in self.pegs.items()}
        return new_state
    
    def move_disk(self, disk_name: str, from_peg: int, to_peg: int) -> bool:
        """
        move a disk from one peg to another.
        
        Args:
            disk_name: name of the disk to move (e.g. 'disk1')
            from_peg: starting peg number
            to_peg: target peg number
            
        Returns:
            True if the move is successful, False otherwise
        """
        if from_peg not in self.pegs or to_peg not in self.pegs:
            return False
        
        if from_peg == to_peg:
            return False
        
        if len(self.pegs[from_peg]) == 0:
            return False
        
        # the disk to move (the top disk)
        top_disk = self.pegs[from_peg][-1]
        
        # check if the specified disk_name matches the actual top disk (case-insensitive)
        if disk_name.upper() != top_disk.upper():
            return False
        
        # normalize disk_name to uppercase
        disk_name = disk_name.upper()
        
        # if the target peg has a disk and the disk to move is larger than the top disk on the target peg, the move is not allowed
        if len(self.pegs[to_peg]) > 0:
            top_disk_on_target = self.pegs[to_peg][-1]
            if self.disk_sizes[disk_name] > self.disk_sizes[top_disk_on_target]:
                return False
        
        # execute the move (disk_name is already normalized to uppercase)
        self.pegs[from_peg].pop()
        self.pegs[to_peg].append(disk_name)
        return True
    
    def is_complete(self) -> bool:
        """check if the target state is reached"""
        # the target peg should have all the disks
        return len(self.pegs[self.target_peg]) == self.num_disks
    
    def to_ascii(self) -> str:
        """convert to ASCII representation"""
        lines = []
        max_height = max(len(disks) for disks in self.pegs.values())
        max_height = max(max_height, 1)
        
        # print each level
        for level in range(max_height - 1, -1, -1):
            line_parts = []
            for peg_num in sorted(self.pegs.keys()):
                if level < len(self.pegs[peg_num]):
                    disk_name = self.pegs[peg_num][level]
                    disk_size = self.disk_sizes[disk_name]
                    disk_str = "=" * disk_size + f"{disk_name}" + "=" * disk_size
                    line_parts.append(f"{disk_str:^20}")
                else:
                    line_parts.append("|".center(20))
            lines.append(" ".join(line_parts))
        
        # peg numbers
        peg_labels = " ".join([f"Peg {i}".center(20) for i in sorted(self.pegs.keys())])
        lines.append(peg_labels)
        return "\n".join(lines)
    
    def to_string(self) -> str:
        """simple string representation: peg 1: E,D,C,B,A (top to bottom)"""
        parts = []
        for peg_num in sorted(self.pegs.keys()):
            if self.pegs[peg_num]:
                # display from top to bottom (top to bottom)
                # pegs are stored from top to bottom, so use as is
                disks_str = ",".join(self.pegs[peg_num])
                parts.append(f"peg {peg_num}: {disks_str}")
            else:
                parts.append(f"peg {peg_num}:")
        return "\n".join(parts)
    
    def __eq__(self, other):
        """compare states"""
        if not isinstance(other, TowerOfHanoiState):
            return False
        return self.pegs == other.pegs
    
    def __hash__(self):
        """make hashable"""
        return hash(tuple(
            (peg, tuple(disks)) for peg, disks in sorted(self.pegs.items())
        ))


class TowerOfHanoiAction:
    """
    Class representing a Tower of Hanoi action.
    """
    
    def __init__(self, disk_name: str, from_peg: int, to_peg: int):
        """
        Args:
            disk_name: disk name (e.g. 'disk1')
            from_peg: starting peg number
            to_peg: target peg number
        """
        # normalize disk_name to uppercase (A, B, C, ...)
        self.disk_name = disk_name.upper()
        self.from_peg = int(from_peg)
        self.to_peg = int(to_peg)
        
        # check if the disk_name is a single letter (A, B, C, ...)
        if not (len(self.disk_name) == 1 and self.disk_name.isalpha()):
            raise ValueError(f"Invalid disk_name: {disk_name}. Should be a single letter (A, B, C, etc.)")
        if self.from_peg < 1:
            raise ValueError(f"Invalid from_peg: {from_peg}. Should be >= 1")
        if self.to_peg < 1:
            raise ValueError(f"Invalid to_peg: {to_peg}. Should be >= 1")
    
    @staticmethod
    def from_serialized(action_str: str) -> "TowerOfHanoiAction":
        """
        parse the action from a string.
        format: move('A', 'from_peg', 'to_peg')
        or: move("A", "from_peg", "to_peg")
        or: move(A, from_peg, to_peg)
        
        Args:
            action_str: action string
            
        Returns:
            TowerOfHanoiAction object
        """
        action_str = action_str.strip()
        
        # move('A', 'from_peg', 'to_peg') format
        # or move("A", "from_peg", "to_peg") format
        # or move(A, from_peg, to_peg) format
        # more flexible pattern: quotes may or may not be present, spaces may or may not be present
        pattern = r"move\s*\(\s*['\"]?([A-Za-z])['\"]?\s*,\s*['\"]?(\d+)['\"]?\s*,\s*['\"]?(\d+)['\"]?\s*\)"
        match = re.match(pattern, action_str, re.IGNORECASE)
        if match:
            disk_name, from_peg, to_peg = match.groups()
            # check if the disk_name is a single letter (A, B, C, ...)
            if not disk_name.isalpha():
                raise ValueError(f"Invalid disk_name: {disk_name}. Should be a single letter (A, B, C, etc.)")
            return TowerOfHanoiAction(disk_name, int(from_peg), int(to_peg))
        
        raise ValueError(f"Invalid action format: {action_str}. Expected: move('A', 'from_peg', 'to_peg')")
    
    def to_string(self) -> str:
        """convert to string"""
        return f"move('{self.disk_name}', {self.from_peg}, {self.to_peg})"
    
    def __repr__(self):
        return f"TowerOfHanoiAction(disk_name={self.disk_name}, from_peg={self.from_peg}, to_peg={self.to_peg})"
    
    def __eq__(self, other):
        if not isinstance(other, TowerOfHanoiAction):
            return False
        return (self.disk_name == other.disk_name and 
                self.from_peg == other.from_peg and 
                self.to_peg == other.to_peg)

