import argparse
import random
from collections import defaultdict
from typing import Dict, FrozenSet, List, Optional, Set

from recognizers.automata.finite_automaton import (
    FiniteAutomatonContainer,
    FiniteAutomatonTransition,
    State,
)
from recognizers.random_utils import (
    sample_from_negative_binomial,
)


class Automaton:
    """A simple automaton class for building finite state automata."""
    
    def __init__(self) -> None:
        self.states: Set[int] = set()
        self.alphabet: Set[str] = set()
        self.transitions: dict[tuple[int, str], int] = {}
        self.initial_state: Optional[int] = None
        self.final_states: Set[int] = set()
        self.next_state_id: int = 0
    
    def add_state(self) -> int:
        """Add a new state and return its ID."""
        state_id = self.next_state_id
        self.states.add(state_id)
        self.next_state_id += 1
        return state_id
    
    def set_initial(self, state: int) -> None:
        """Set the initial state."""
        self.initial_state = state
    
    def add_final(self, state: int) -> None:
        """Add a final state."""
        self.final_states.add(state)
    
    def add_transition(self, from_state: int, symbol: str, to_state: int) -> None:
        """Add a transition."""
        self.states.add(from_state)
        self.states.add(to_state)
        self.alphabet.add(symbol)
        self.transitions[(from_state, symbol)] = to_state
    
    def trim(self) -> 'Automaton':
        """
        Trim this automaton by removing unreachable and dead states.
        
        Returns:
            A new automaton with only useful states
        """
        if not self.states or self.initial_state is None:
            return self.__class__()  # Return empty automaton
        
        # Step 1: Find all states reachable from the initial state
        reachable_states = set()
        frontier = {self.initial_state}
        
        while frontier:
            current = frontier.pop()
            reachable_states.add(current)
            
            # Find all states reachable in one step
            for (from_state, symbol), to_state in self.transitions.items():
                if from_state == current and to_state not in reachable_states:
                    frontier.add(to_state)
        
        # Step 2: Find all states that can reach a final state
        can_reach_final = set(self.final_states)
        changed = True
        
        while changed:
            changed = False
            for (from_state, symbol), to_state in self.transitions.items():
                if to_state in can_reach_final and from_state not in can_reach_final:
                    can_reach_final.add(from_state)
                    changed = True
        
        # Step 3: Keep only states that are both reachable and can reach a final state
        useful_states = reachable_states.intersection(can_reach_final)
        
        # Create a new automaton with only the useful states
        result = self.__class__()
        
        # Map old states to new states
        state_map = {}
        for i, state in enumerate(sorted(useful_states)):
            state_map[state] = i
            result.add_state()
        
        # Set initial state
        if self.initial_state in useful_states:
            result.set_initial(state_map[self.initial_state])
        else:
            # If the initial state isn't useful, the language is empty
            return self.__class__()
        
        # Copy useful transitions
        for (from_state, symbol), to_state in self.transitions.items():
            if from_state in useful_states and to_state in useful_states:
                result.add_transition(state_map[from_state], symbol, state_map[to_state])
        
        # Copy final states
        for final in self.final_states:
            if final in useful_states:
                result.add_final(state_map[final])
        
        return result
    
    def minimize(self) -> 'Automaton':
        """
        Minimize this automaton using Hopcroft's algorithm.
        
        Returns:
            A minimal equivalent automaton
        """
        # First trim the automaton
        automaton = self.trim()
        
        if not automaton.states:
            return automaton  # Empty automaton
        
        # Create initial partition: accepting and non-accepting states
        accepting = sorted([s for s in automaton.states if s in automaton.final_states])
        non_accepting = sorted([s for s in automaton.states if s not in automaton.final_states])
        partitions = [accepting, non_accepting]
        partitions = [p for p in partitions if p]  # Remove empty partitions
        
        # Function to find which partition a state belongs to
        def find_partition(state, partitions):
            for i, partition in enumerate(partitions):
                if state in partition:
                    return i
            return -1
        
        # Refine partitions until no more refinement is possible
        workset = list(range(len(partitions)))
        while workset:
            partition_idx = workset.pop(0)
            if partition_idx >= len(partitions):
                continue
                
            partition = partitions[partition_idx]
            
            for symbol in automaton.alphabet:
                # For each symbol, we check if states in the same partition 
                # transition to states in different partitions
                new_partitions = defaultdict(list)
                
                for state in partition:
                    # Find destination partition for this state and symbol
                    dest_state = automaton.transitions.get((state, symbol))
                    if dest_state is None:
                        destination_partition = -1
                    else:
                        destination_partition = find_partition(dest_state, partitions)
                    new_partitions[destination_partition].append(state)
                
                # If we have more than one new partition, we need to refine
                if len(new_partitions) > 1:
                    # Remove the original partition
                    partitions.pop(partition_idx)
                    
                    # Add the new partitions
                    new_partition_indices = []
                    for new_partition in new_partitions.values():
                        partitions.append(sorted(new_partition))
                        new_partition_indices.append(len(partitions) - 1)
                    
                    # Update workset
                    if partition_idx in workset:
                        workset.remove(partition_idx)
                    workset.extend(new_partition_indices)
                    break
        
        # Create a new automaton with the minimal states
        result = self.__class__()
        
        # Map old states to new states
        state_map = {}
        for i, partition in enumerate(partitions):
            for state in partition:
                state_map[state] = i
            result.add_state()
        
        # Set initial state
        result.set_initial(state_map[automaton.initial_state])
        
        # Add transitions
        for (from_state, symbol), to_state in automaton.transitions.items():
            result.add_transition(state_map[from_state], symbol, state_map[to_state])
        
        # Add final states
        for final in automaton.final_states:
            result.add_final(state_map[final])
        
        return result
    
    def determinize(self) -> 'Automaton':
        """
        Determinize this possibly non-deterministic automaton using the subset construction.
        
        Returns:
            A deterministic automaton accepting the same language
        """
        if not self.states or self.initial_state is None:
            return self.__class__()
        
        result = self.__class__()
        
        # Create state sets (Each state in the new DFA represents a set of states in the NFA)
        initial_state_set = frozenset([self.initial_state])
        state_sets: Dict[FrozenSet[int], int] = {initial_state_set: result.add_state()}
        frontier = [initial_state_set]
        
        # Set the initial state
        result.set_initial(state_sets[initial_state_set])
        
        # Process all state sets
        while frontier:
            current_state_set = frontier.pop(0)
            current_new_state = state_sets[current_state_set]
            
            # For each symbol, find the next state set
            for symbol in self.alphabet:
                next_states = set()
                
                # For each state in the current set, find all transitions on this symbol
                for state in current_state_set:
                    transition_key = (state, symbol)
                    if transition_key in self.transitions:
                        next_states.add(self.transitions[transition_key])
                
                # Skip if there are no transitions
                if not next_states:
                    continue
                
                # Convert to frozenset for dictionary key
                next_state_set = frozenset(next_states)
                
                # Create a new state if we haven't seen this set before
                if next_state_set not in state_sets:
                    state_sets[next_state_set] = result.add_state()
                    frontier.append(next_state_set)
                
                # Add the transition
                result.add_transition(current_new_state, symbol, state_sets[next_state_set])
            
            # Check if this state set contains any accepting states
            if any(state in self.final_states for state in current_state_set):
                result.add_final(current_new_state)
        
        return result
    
    def optimize(self) -> 'Automaton':
        """Optimize this automaton by trimming, determinizing, and minimizing it."""
        print(f"Starting optimization with an automaton of {len(self.states)} states and {len(self.final_states)} accepting states")
        
        if self.initial_state is None:
            print("  Warning: Automaton has no initial state. Returning empty automaton.")
            return self.__class__()
        
        # First trim to remove unreachable and dead states (more efficient before determinization)
        print("  Step 1: Trimming unreachable and dead states...")
        trimmed = self.trim()
        print(f"  After trimming: {len(trimmed.states)} states, {len(trimmed.final_states)} accepting states")
        
        if trimmed.initial_state is None:
            print("  Warning: Trimmed automaton has no initial state. Returning empty automaton.")
            return self.__class__()
            
        # Then determinize to ensure the automaton is deterministic
        print("  Step 2: Determinizing automaton...")
        deterministic = trimmed.determinize()
        print(f"  After determinization: {len(deterministic.states)} states, {len(deterministic.final_states)} accepting states")
        
        if deterministic.initial_state is None:
            print("  Warning: Determinized automaton has no initial state. Returning empty automaton.")
            return self.__class__()
            
        # Finally minimize to reduce the number of states further
        print("  Step 3: Minimizing automaton...")
        minimized = deterministic.minimize()
        print(f"  After minimization: {len(minimized.states)} states, {len(minimized.final_states)} accepting states")
        
        if minimized.initial_state is None:
            print("  Warning: Minimized automaton has no initial state. Returning empty automaton.")
            return self.__class__()
        
        print(f"Optimization complete: {len(self.states)} → {len(minimized.states)} states")
        return minimized
    
    def complement(self, full_alphabet: Optional[Set[str]] = None) -> 'Automaton':
        """
        Return the complement of this automaton.
        Note: The automaton must be deterministic for this to work correctly.
        
        Args:
            full_alphabet: Optional complete alphabet to use for complementation.
                           If not provided, uses the automaton's own alphabet.
        """
        # First ensure the automaton is deterministic
        deterministic_automaton = self.determinize()
        
        # Now create a complete DFA (add a sink state for missing transitions)
        complete_automaton = self.__class__()
        
        # Copy all states
        for _ in deterministic_automaton.states:
            complete_automaton.add_state()
        
        # Add a sink state if needed
        sink_state = None
        
        # Copy the initial state
        complete_automaton.set_initial(deterministic_automaton.initial_state)
        
        # Use the provided full alphabet or default to the automaton's alphabet
        alphabet_to_use = full_alphabet if full_alphabet is not None else deterministic_automaton.alphabet
        
        # Copy the alphabet
        complete_automaton.alphabet = alphabet_to_use.copy()
        
        # First, copy all existing transitions
        for (from_state, symbol), to_state in deterministic_automaton.transitions.items():
            complete_automaton.add_transition(from_state, symbol, to_state)
        
        # Then add missing transitions to the sink state
        for state in deterministic_automaton.states:
            for symbol in alphabet_to_use:
                if (state, symbol) not in deterministic_automaton.transitions:
                    # This is a missing transition - add a transition to the sink state
                    if sink_state is None:
                        sink_state = complete_automaton.add_state()
                    complete_automaton.add_transition(state, symbol, sink_state)
        
        # If we created a sink state, add self-loops for all symbols
        if sink_state is not None:
            for symbol in alphabet_to_use:
                complete_automaton.add_transition(sink_state, symbol, sink_state)
        
        # Finally, flip the accepting states
        result = self.__class__()
        
        # Copy states and transitions
        for _ in complete_automaton.states:
            result.add_state()
        
        result.initial_state = complete_automaton.initial_state
        result.alphabet = complete_automaton.alphabet.copy()
        result.transitions = complete_automaton.transitions.copy()
        result.next_state_id = complete_automaton.next_state_id
        
        # Flip accepting states
        result.final_states = complete_automaton.states - complete_automaton.final_states
        
        return result
    
    def union(self, other: 'Automaton') -> 'Automaton':
        """
        Return the union of this automaton with another.
        Note: This may create a non-deterministic automaton.
        """
        result = self.__class__()
        
        # Create a new initial state
        new_initial = result.add_state()
        result.set_initial(new_initial)
        
        # Copy states with offset
        offset1 = result.next_state_id
        for state in self.states:
            result.states.add(state + offset1)
        result.next_state_id += len(self.states)
        
        offset2 = result.next_state_id
        for state in other.states:
            result.states.add(state + offset2)
        result.next_state_id += len(other.states)
        
        # Add epsilon transitions from new initial state
        # Since we can't use epsilon transitions directly, 
        # we'll copy transitions from initial states
        if self.initial_state is not None:
            for symbol in self.alphabet:
                if (self.initial_state, symbol) in self.transitions:
                    to_state = self.transitions[(self.initial_state, symbol)]
                    result.add_transition(new_initial, symbol, to_state + offset1)
        
        if other.initial_state is not None:
            for symbol in other.alphabet:
                if (other.initial_state, symbol) in other.transitions:
                    to_state = other.transitions[(other.initial_state, symbol)]
                    result.add_transition(new_initial, symbol, to_state + offset2)
        
        # Copy transitions with offsets
        for (from_state, symbol), to_state in self.transitions.items():
            if from_state != self.initial_state:  # Initial state already handled
                result.add_transition(from_state + offset1, symbol, to_state + offset1)
        
        for (from_state, symbol), to_state in other.transitions.items():
            if from_state != other.initial_state:  # Initial state already handled
                result.add_transition(from_state + offset2, symbol, to_state + offset2)
        
        # Combine alphabets
        result.alphabet = self.alphabet.union(other.alphabet)
        
        # Set final states
        for final in self.final_states:
            result.add_final(final + offset1)
        for final in other.final_states:
            result.add_final(final + offset2)
        
        return result
    
    def intersect(self, other: 'Automaton') -> 'Automaton':
        """
        Return the intersection of this automaton with another.
        Uses the product construction for DFAs.
        Note: Both automata should be deterministic for this to work correctly.
        """
        # First ensure both automata are deterministic
        self_det = self.determinize()
        other_det = other.determinize()
        
        result = self.__class__()
        
        # Create the product state space
        state_pairs = {}  # Maps pairs of states to new state IDs
        
        # Start with the initial states
        if self_det.initial_state is None or other_det.initial_state is None:
            # If either automaton is empty, the intersection is empty
            return result
            
        initial_pair = (self_det.initial_state, other_det.initial_state)
        initial_state = result.add_state()
        state_pairs[initial_pair] = initial_state
        result.set_initial(initial_state)
        
        # Build the intersection using product construction
        frontier = [initial_pair]
        
        while frontier:
            current_pair = frontier.pop(0)
            current_self_state, current_other_state = current_pair
            current_new_state = state_pairs[current_pair]
            
            # Process transitions for symbols in both alphabets
            for symbol in self_det.alphabet.intersection(other_det.alphabet):
                # Find transitions in both automata
                self_transition = self_det.transitions.get((current_self_state, symbol))
                other_transition = other_det.transitions.get((current_other_state, symbol))
                
                # Only add a transition if both automata have transitions on this symbol
                if self_transition is not None and other_transition is not None:
                    next_pair = (self_transition, other_transition)
                    
                    # Create a new state if we haven't seen this pair before
                    if next_pair not in state_pairs:
                        new_state = result.add_state()
                        state_pairs[next_pair] = new_state
                        frontier.append(next_pair)
                    
                    # Add the transition in the result automaton
                    result.add_transition(current_new_state, symbol, state_pairs[next_pair])
        
        # Set final states - a state is accepting if both original states are accepting
        for (self_state, other_state), new_state in state_pairs.items():
            if self_state in self_det.final_states and other_state in other_det.final_states:
                result.add_final(new_state)
        
        return result
    
    def concatenate(self, other: 'Automaton') -> 'Automaton':
        """
        Return the concatenation of this automaton with another.
        Note: This may create a non-deterministic automaton.
        """
        result = self.__class__()
        
        # Copy states with offset
        offset1 = 0
        for state in self.states:
            result.states.add(state + offset1)
        result.next_state_id = len(self.states)
        
        offset2 = result.next_state_id
        for state in other.states:
            result.states.add(state + offset2)
        result.next_state_id += len(other.states)
        
        # Set initial state
        if self.initial_state is not None:
            result.set_initial(self.initial_state + offset1)
        
        # Copy transitions
        for (from_state, symbol), to_state in self.transitions.items():
            result.add_transition(from_state + offset1, symbol, to_state + offset1)
        
        for (from_state, symbol), to_state in other.transitions.items():
            result.add_transition(from_state + offset2, symbol, to_state + offset2)
        
        # Connect final states of first automaton to initial state of second
        if other.initial_state is not None:
            for final in self.final_states:
                for symbol in other.alphabet:
                    if (other.initial_state, symbol) in other.transitions:
                        to_state = other.transitions[(other.initial_state, symbol)]
                        result.add_transition(final + offset1, symbol, to_state + offset2)
        
        # Set final states to be only those from the second automaton
        for final in other.final_states:
            result.add_final(final + offset2)
        
        return result
    
    def kleene_star(self) -> 'Automaton':
        """
        Apply the Kleene star operation to this automaton.
        This creates an automaton that accepts zero or more repetitions of strings in the original language.
        """
        result = self.__class__()
        
        # Create a new initial state that is also accepting (to accept empty string)
        new_initial = result.add_state()
        result.set_initial(new_initial)
        result.add_final(new_initial)  # New initial state is accepting for empty string
        
        # Copy states with offset
        offset = result.next_state_id
        for state in self.states:
            result.states.add(state + offset)
        result.next_state_id += len(self.states)
        
        # Copy transitions with offset
        for (from_state, symbol), to_state in self.transitions.items():
            result.add_transition(from_state + offset, symbol, to_state + offset)
        
        # Copy alphabet
        result.alphabet = self.alphabet.copy()
        
        # Add epsilon transitions from new initial state to original initial state (if it exists)
        if self.initial_state is not None:
            # Since we can't use epsilon transitions directly,
            # copy transitions from the original initial state
            for symbol in self.alphabet:
                if (self.initial_state, symbol) in self.transitions:
                    to_state = self.transitions[(self.initial_state, symbol)]
                    result.add_transition(new_initial, symbol, to_state + offset)
        
        # Add epsilon transitions from final states back to initial state
        for final in self.final_states:
            for symbol in self.alphabet:
                if (self.initial_state, symbol) in self.transitions:
                    to_state = self.transitions[(self.initial_state, symbol)]
                    result.add_transition(final + offset, symbol, to_state + offset)
        
        # Copy original final states
        for final in self.final_states:
            result.add_final(final + offset)
        
        return result
    
    def to_finite_automaton_container(self) -> FiniteAutomatonContainer:
        """Convert to FiniteAutomatonContainer."""
        # Map symbols to integers
        alphabet_map = {sym: idx for idx, sym in enumerate(sorted(self.alphabet))}
        
        fac = FiniteAutomatonContainer(
            num_states=len(self.states),
            alphabet_size=len(self.alphabet),
            initial_state=State(self.initial_state if self.initial_state is not None else 0)
        )
        
        # Add transitions
        for (from_state, symbol), to_state in self.transitions.items():
            fac.add_transition(FiniteAutomatonTransition(
                State(from_state),
                alphabet_map[symbol],
                State(to_state)
            ))
        
        # Set final states
        for final in self.final_states:
            fac.add_accept_state(State(final))
        
        return fac
    
    @classmethod
    def create_atomic_automaton(cls, symbol: str, alphabet: Set[str]) -> 'Automaton':
        """Create an automaton that accepts only the given symbol."""
        automaton = cls()
        
        # State 0: initial
        # State 1: after reading the symbol (accepting)
        # State 2: sink state
        automaton.add_state()  # 0
        automaton.add_state()  # 1
        automaton.add_state()  # 2
        
        automaton.set_initial(0)
        automaton.add_final(1)
        
        # Transition for the specific symbol
        automaton.add_transition(0, symbol, 1)
        
        # All other symbols go to the sink state
        for a in alphabet:
            if a != symbol:
                automaton.add_transition(0, a, 2)
        
        # From accepting state, all symbols go to sink
        for a in alphabet:
            automaton.add_transition(1, a, 2)
            automaton.add_transition(2, a, 2)
        
        return automaton
    
    @classmethod
    def create_empty_string_automaton(cls, alphabet: Set[str]) -> 'Automaton':
        """Create an automaton that accepts only the empty string."""
        automaton = cls()
        
        # State 0: initial and accepting
        # State 1: sink state
        automaton.add_state()  # 0
        automaton.add_state()  # 1
        
        automaton.set_initial(0)
        automaton.add_final(0)
        
        # All symbols go to the sink state
        for a in alphabet:
            automaton.add_transition(0, a, 1)
            automaton.add_transition(1, a, 1)
        
        return automaton


# Helper functions for constructing and sampling automata
def M(automata: List[Automaton], max_concat_ops: int, generator: random.Random) -> List[Automaton]:
    """
    Apply concatenation operations to a list of automata.

    Args:
        automata: List of automata to concatenate
        max_concat_ops: Maximum number of concatenation operations to apply 
        generator: Random number generator

    Returns:
        A list of automata after applying concatenation
    """
    print(f"\nPerforming {max_concat_ops} concatenation operations:")
    result = []
    for i in range(max_concat_ops):
        idx_1 = generator.randint(0, len(automata) - 1)
        idx_2 = generator.randint(0, len(automata) - 1)
        
        print(f"  Operation {i+1}: Concatenating automata {idx_1} and {idx_2}")
        r = automata[idx_1].concatenate(automata[idx_2])
        r = r.optimize()
        
        if r.initial_state is None:
            print("  Warning: Resulting automaton has no initial state. Skipping.")
            continue
            
        result.append(r)
        print(f"  Result: Automaton with {len(r.states)} states and {len(r.final_states)} accepting states")

    return result


def B(automata: List[Automaton], max_bool_ops: int, generator: random.Random, full_alphabet: Optional[Set[str]] = None) -> List[Automaton]:
    """
    Apply boolean operations to a list of automata.

    Args:
        automata: List of automata to apply boolean operations
        max_bool_ops: Maximum number of boolean operations to apply
        generator: Random number generator
        full_alphabet: Complete alphabet to use for all operations (especially complement)

    Returns:
        A list of automata after applying boolean operations
    """
    print(f"\nPerforming {max_bool_ops} boolean operations:")
    
    # If full_alphabet is not provided, compute it from all automata
    if full_alphabet is None:
        full_alphabet = set()
        for a in automata:
            full_alphabet.update(a.alphabet)
            
    result = []
    while len(result) < max_bool_ops:
        idx_1 = generator.randint(0, len(automata) - 1)
        idx_2 = generator.randint(0, len(automata) - 1)

        op = generator.choice(['union', 'intersection', 'complement'])
        
        print(f"  Operation {len(result)+1}: ", end="")
        if op == 'union':
            print(f"Union of automata {idx_1} and {idx_2}")
            r = automata[idx_1].union(automata[idx_2])
        elif op == 'intersection':
            print(f"Intersection of automata {idx_1} and {idx_2}")
            r = automata[idx_1].intersect(automata[idx_2])
        elif op == 'complement': 
            print(f"Complement of automaton {idx_1}")
            # Use the full alphabet for complementation
            r = automata[idx_1].complement(full_alphabet)
        
        r = r.optimize()

        if r.initial_state is None:
            print("  Warning: Resulting automaton has no initial state. Skipping.")
            continue

        result.append(r)
        print(f"  Result: Automaton with {len(r.states)} states and {len(r.final_states)} accepting states")

    return result


def K(automata: List[Automaton], max_kleene_ops: int, generator: random.Random) -> List[Automaton]:
    """
    Apply Kleene star operations to automata.
    
    Args:
        automata: List of automata to apply Kleene star to
        max_kleene_ops: Maximum number of Kleene star operations to apply
        generator: Random number generator
        
    Returns:
        A list of automata after applying Kleene star operations
    """
    print(f"\nPerforming {max_kleene_ops} Kleene star operations:")
    
    result = []
    while len(result) < max_kleene_ops:
        idx = generator.randint(0, len(automata) - 1)
        
        print(f"  Operation {len(result)+1}: Kleene star of automaton {idx}")
        r = automata[idx].kleene_star()
        r = r.optimize()
        
        if r.initial_state is None:
            print("  Warning: Resulting automaton has no initial state. Skipping.")
            continue
            
        result.append(r)
        print(f"  Result: Automaton with {len(r.states)} states and {len(r.final_states)} accepting states")
        
    return result


def sample_fsa(
    mean_alphabet_size: float,
    mean_depth: float,
    max_bool_ops: int, 
    max_concat_ops: int,
    max_kleene_ops: int,
    accept_probability: float = 0.4,
    generator: random.Random = None
) -> FiniteAutomatonContainer:
    """
    Sample a general finite state automaton with boolean operations, concatenation, and Kleene star.
    
    Args:
        mean_alphabet_size: Mean size of the alphabet
        mean_depth: Mean depth level
        max_bool_ops: Fixed number of boolean operations at each level
        max_concat_ops: Fixed number of concatenation operations at each level
        max_kleene_ops: Fixed number of Kleene star operations at each level
        accept_probability: Probability of each state being accepting (default: 0.4)
        generator: Random number generator
    
    Returns:
        A DFA that recognizes a regular language
    """
    if generator is None:
        generator = random.Random()

    star_free = max_kleene_ops == 0
    
    # Sample actual parameters from negative binomial distributions
    alphabet_size = max(2, sample_from_negative_binomial(mean_alphabet_size, 1, generator))
    depth = max(1, sample_from_negative_binomial(mean_depth, 1, generator))
    
    print(f"Sampled parameters: alphabet_size={alphabet_size}, depth={depth}, "
          f"max_bool_ops={max_bool_ops}, max_concat_ops={max_concat_ops}, max_kleene_ops={max_kleene_ops}")
    
    # Create alphabet
    alphabet = set(str(i) for i in range(alphabet_size))
    
    # Start with atomic automata (depth 0)
    depth_0_automata: List[Automaton] = []
    for symbol in alphabet:
        depth_0_automata.append(Automaton.create_atomic_automaton(symbol, alphabet))
    
    # Add the empty string automaton
    depth_0_automata.append(Automaton.create_empty_string_automaton(alphabet))
    
    # Track automata by depth level
    automata_by_level = [depth_0_automata]
    
    # Build hierarchy level by level
    for level in range(1, depth):
        print(f"\nBuilding level {level}:")
        
        # Perform concatenations
        current_level_automata = M(automata_by_level[level - 1], max_concat_ops, generator)
        
        # Perform boolean operations
        current_level_automata = B(current_level_automata, max_bool_ops, generator, alphabet)
        
        # Perform Kleene star operations
        if not star_free:
            current_level_automata = K(current_level_automata, max_kleene_ops, generator)
        
        # Store automata for this level
        automata_by_level.append(current_level_automata)
    
    # Select the automaton of the desired depth with the most states
    automata_states = [len(a.states) for a in automata_by_level[depth - 1]]
    chosen_idx = automata_states.index(max(automata_states))
    final_automaton = automata_by_level[depth - 1][chosen_idx]
    print(f"\nSelected automaton of depth {depth} with {len(final_automaton.states)} states and "
          f"{len(final_automaton.final_states)} accepting states")
    
    # Optimize the automaton
    final_automaton = final_automaton.optimize()
    print(f"After optimization: {len(final_automaton.states)} states, {len(final_automaton.final_states)} accepting states")

    # Adjust accepting states to match the target probability
    non_accepting_states = [s for s in final_automaton.states if s not in final_automaton.final_states]
    num_to_add = max(0, int(len(final_automaton.states) * accept_probability) - len(final_automaton.final_states))
    # Ensure we don't try to sample more states than available
    num_to_sample = min(num_to_add, len(non_accepting_states))
    if num_to_sample > 0:
        final_automaton.final_states.update(generator.sample(non_accepting_states, num_to_sample))
    print(f"After adding accepting states: {len(final_automaton.states)} states, {len(final_automaton.final_states)} accepting states "
          f"({len(final_automaton.final_states)/len(final_automaton.states) if len(final_automaton.states) > 0 else 0:.1%} of all states)")

    # Convert to FiniteAutomatonContainer
    result = final_automaton.to_finite_automaton_container()
    
    return result


def sample_star_free(
    mean_alphabet_size: float,
    mean_depth: float,
    max_bool_ops: int, 
    max_concat_ops: int, 
    accept_probability: float = 0.4,
    generator: random.Random = None
) -> FiniteAutomatonContainer:
    """
    Sample a DFA that recognizes a star-free language.
    This uses sample_fsa with max_kleene_ops=0.
    
    Args:
        mean_alphabet_size: Mean size of the alphabet
        mean_depth: Mean depth level
        max_bool_ops: Fixed number of boolean operations at each level
        max_concat_ops: Fixed number of concatenation operations at each level
        accept_probability: Probability of each state being accepting (default: 0.4)
        generator: Random number generator
    
    Returns:
        A DFA that recognizes a star-free language
    """
    return sample_fsa(
        mean_alphabet_size=mean_alphabet_size,
        mean_depth=mean_depth,
        max_bool_ops=max_bool_ops,
        max_concat_ops=max_concat_ops,
        max_kleene_ops=0,  # No Kleene operations for star-free languages
        accept_probability=accept_probability,
        generator=generator
    )


def main() -> None:
    """Main function for command-line execution."""
    parser = argparse.ArgumentParser(description='Sample a DFA recognizing a regular language')
    parser.add_argument('--mean_alphabet_size', type=float, default=10, 
                       help='Mean size of the alphabet (default: 10)')
    parser.add_argument('--mean_depth', type=float, default=4, 
                       help='Mean depth level (default: 4)')
    parser.add_argument('--max_bool_ops', type=int, default=4,
                       help='Number of boolean operations per level (default: 4)')
    parser.add_argument('--max_concat_ops', type=int, default=4,
                       help='Number of concatenation operations per level (default: 4)')
    parser.add_argument('--max_kleene_ops', type=int, default=2,
                       help='Number of Kleene star operations per level (default: 2)')
    parser.add_argument('--accept_probability', type=float, default=0.4, 
                       help='Probability of each state being accepting (default: 0.4)')
    parser.add_argument('--seed', type=int, default=42, help='Random seed')
    parser.add_argument('--verbose', '-v', action='store_true', help='Show more detailed information')
    
    args = parser.parse_args()
    
    generator = random.Random(args.seed)
    
    print(f"Sampling a regular language with mean alphabet size {args.mean_alphabet_size}, "
          f"mean depth {args.mean_depth}")
    print(f"Using {args.max_bool_ops} boolean operations, {args.max_concat_ops} concatenations, "
          f"and {args.max_kleene_ops} Kleene star operations per level")
    print(f"Accept probability: {args.accept_probability}")
    print(f"Random seed: {args.seed}")
    
    automaton = sample_fsa(
        mean_alphabet_size=args.mean_alphabet_size,
        mean_depth=args.mean_depth,
        max_bool_ops=args.max_bool_ops,
        max_concat_ops=args.max_concat_ops,
        max_kleene_ops=args.max_kleene_ops,
        accept_probability=args.accept_probability,
        generator=generator
    )
    
    # Print basic information about the sampled automaton
    print(f"\n==== GENERATED AUTOMATON SUMMARY ====")
    print(f"States: {len(list(automaton.states()))}")
    print(f"Alphabet size: {len(list(automaton.alphabet()))}")
    print(f"Transitions: {len(list(automaton.transitions()))}")
    print(f"Accept states: {sum(1 for s in automaton.states() if automaton.is_accept_state(s))}")
    
    if args.verbose:
        print("\nTransitions:")
        for t in automaton.transitions():
            dest_type = "ACCEPT" if automaton.is_accept_state(t.state_to) else "REJECT"
            src_type = "INITIAL" if t.state_from == automaton.initial_state() else (
                "ACCEPT" if automaton.is_accept_state(t.state_from) else "REGULAR")
            print(f"  {t.state_from}({src_type}) --{t.symbol}--> {t.state_to}({dest_type})")


if __name__ == "__main__":
    main()