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

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


class DotDepthAutomaton:
    """A simple automaton class for building dot-depth hierarchy 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 complement(self, full_alphabet: Optional[Set[str]] = None) -> 'DotDepthAutomaton':
        """
        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
        # We'll delegate this to the determinize function
        deterministic_automaton = determinize_dot_depth_automaton(self)
        
        # Now create a complete DFA (add a sink state for missing transitions)
        complete_automaton = DotDepthAutomaton()
        
        # 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 = DotDepthAutomaton()
        
        # 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: 'DotDepthAutomaton') -> 'DotDepthAutomaton':
        """
        Return the union of this automaton with another.
        Note: This may create a non-deterministic automaton.
        """
        result = DotDepthAutomaton()
        
        # 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: 'DotDepthAutomaton') -> 'DotDepthAutomaton':
        """
        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 = determinize_dot_depth_automaton(self)
        other_det = determinize_dot_depth_automaton(other)
        
        result = DotDepthAutomaton()
        
        # 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: 'DotDepthAutomaton') -> 'DotDepthAutomaton':
        """
        Return the concatenation of this automaton with another.
        Note: This may create a non-deterministic automaton.
        """
        result = DotDepthAutomaton()
        
        # 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 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


def trim_dot_depth_automaton(automaton: DotDepthAutomaton) -> DotDepthAutomaton:
    """
    Trim a DotDepthAutomaton by removing unreachable and dead states.
    
    Args:
        automaton: The automaton to trim
        
    Returns:
        A new automaton with only useful states
    """
    if not automaton.states or automaton.initial_state is None:
        return automaton  # Nothing to trim
    
    # Step 1: Find all states reachable from the initial state
    reachable_states = set()
    frontier = {automaton.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 automaton.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(automaton.final_states)
    changed = True
    
    while changed:
        changed = False
        for (from_state, symbol), to_state in automaton.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 = DotDepthAutomaton()
    
    # 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 automaton.initial_state in useful_states:
        result.set_initial(state_map[automaton.initial_state])
    else:
        # If the initial state isn't useful, the language is empty
        return DotDepthAutomaton()
    
    # Copy useful transitions
    for (from_state, symbol), to_state in automaton.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 automaton.final_states:
        if final in useful_states:
            result.add_final(state_map[final])
    
    return result


def minimize_dot_depth_automaton(automaton: DotDepthAutomaton) -> DotDepthAutomaton:
    """
    Minimize a DotDepthAutomaton using Hopcroft's algorithm.
    
    Args:
        automaton: The automaton to minimize
        
    Returns:
        A minimal equivalent automaton
    """
    # First trim the automaton
    automaton = trim_dot_depth_automaton(automaton)
    
    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)
        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 = DotDepthAutomaton()
    
    # 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_dot_depth_automaton(automaton: DotDepthAutomaton) -> DotDepthAutomaton:
    """
    Determinize a possibly non-deterministic automaton using the subset construction.
    
    Args:
        automaton: The automaton to determinize
        
    Returns:
        A deterministic automaton accepting the same language
    """
    if not automaton.states or automaton.initial_state is None:
        return automaton
    
    result = DotDepthAutomaton()
    
    # Create state sets (Each state in the new DFA represents a set of states in the NFA)
    initial_state_set = frozenset([automaton.initial_state])
    state_sets = {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 automaton.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 automaton.transitions:
                    next_states.add(automaton.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 automaton.final_states for state in current_state_set):
            result.add_final(current_new_state)
    
    return result


def optimize_dot_depth_automaton(automaton: DotDepthAutomaton) -> DotDepthAutomaton:
    """Optimize a DotDepthAutomaton by determinizing, trimming, and minimizing it."""
    # print(f"Starting optimization with an automaton of {len(automaton.states)} states and {len(automaton.final_states)} accepting states")
    
    if automaton.initial_state is None:
        # print("  Warning: Automaton has no initial state. Returning empty automaton.")
        return DotDepthAutomaton()
        
    # First determinize to ensure the automaton is deterministic
    # print("  Step 1: Determinizing automaton...")
    deterministic = determinize_dot_depth_automaton(automaton)
    # 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 DotDepthAutomaton()
        
    # Then trim to remove unreachable states
    # print("  Step 2: Trimming unreachable states...")
    trimmed = trim_dot_depth_automaton(deterministic)
    # 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 DotDepthAutomaton()
        
    # Finally minimize to reduce the number of states further
    # print("  Step 3: Minimizing automaton...")
    minimized = minimize_dot_depth_automaton(trimmed)
    # 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 DotDepthAutomaton()
    
    # print(f"Optimization complete: {len(automaton.states)} → {len(minimized.states)} states")
    return minimized
    

def create_atomic_automaton(symbol: str, alphabet: Set[str]) -> DotDepthAutomaton:
    """Create an automaton that accepts only the given symbol."""
    automaton = DotDepthAutomaton()
    
    # 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


def create_empty_string_automaton(alphabet: Set[str]) -> DotDepthAutomaton:
    """Create an automaton that accepts only the empty string."""
    automaton = DotDepthAutomaton()
    
    # 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


def M(automata: List[DotDepthAutomaton], max_concat_ops: int, generator: random.Random) -> List[DotDepthAutomaton]:
    """
    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 

    Returns:
        A list of automata after applying concatenation
    """
    result = []
    for i in range(max_concat_ops):
        idx_1 = generator.randint(0, len(automata) - 1)
        idx_2 = generator.randint(0, len(automata) - 1)
        
        r = automata[idx_1].concatenate(automata[idx_2])

        r = optimize_dot_depth_automaton(r)

        result.append(r)

    return result


def B(automata: List[DotDepthAutomaton], max_bool_ops: int, generator: random.Random, full_alphabet: Optional[Set[str]] = None) -> List[DotDepthAutomaton]:
    """
    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'])
        
        if op == 'union':
            r = automata[idx_1].union(automata[idx_2])
        elif op == 'intersection':
            r = automata[idx_1].intersect(automata[idx_2])
        elif op == 'complement': 
            # Use the full alphabet for complementation
            r = automata[idx_1].complement(full_alphabet)
        
        r = optimize_dot_depth_automaton(r)

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

        result.append(r)

    return result


def sample_dot_depth(
    mean_alphabet_size: float,
    mean_depth: float,
    max_bool_ops: int, 
    max_concat_ops: int, 
    accept_probability: float = 0.4,
    generator: random.Random = None
) -> tuple[FiniteAutomatonContainer, int]:
    """
    Sample a DFA that recognizes a star-free language of dot-depth sampled from a negative binomial.
    
    Args:
        mean_alphabet_size: Mean size of the alphabet
        mean_depth: Mean dot-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 of the sampled dot-depth
    """
    if generator is None:
        generator = random.Random()
    
    # 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))
    # max_bool_ops and max_concat_ops are now direct inputs, no longer sampled
    
    # print(f"Sampled parameters: alphabet_size={alphabet_size}, depth={depth}, "
        #   f"max_bool_ops={max_bool_ops}, max_concat_ops={max_concat_ops}")
    
    # Create alphabet
    alphabet = set(str(i) for i in range(alphabet_size))
    
    # Start with atomic automata (depth 0)
    depth_0_automata: List[DotDepthAutomaton] = []
    for symbol in alphabet:
        depth_0_automata.append(create_atomic_automaton(symbol, alphabet))
    
    # Add the empty string automaton
    depth_0_automata.append(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):

        # 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)
        
        # 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 dot-depth {depth} with {len(final_automaton.states)} states and "
    #         f"{len(final_automaton.final_states)} accepting states")
    
    # Optimize the automaton
    final_automaton = optimize_dot_depth_automaton(final_automaton)
    # print(f"After optimization: {len(final_automaton.states)} states, {len(final_automaton.final_states)} accepting states")

    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, depth


def main() -> None:
    """Main function for command-line execution."""
    parser = argparse.ArgumentParser(description='Sample a DFA recognizing a star-free 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 dot-depth level (default: 4)')
    parser.add_argument('--max_bool_ops', type=int, default=4,  # Changed from --mean_bool_ops, type float
                       help='Number of boolean operations per level (default: 4)')
    parser.add_argument('--max_concat_ops', type=int, default=4,  # Changed from --mean_concat_ops, type float
                       help='Number of concatenation operations per level (default: 4)')
    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 star-free language with mean alphabet size {args.mean_alphabet_size}, "
    #       f"mean depth {args.mean_depth}")
    # print(f"Using {args.max_bool_ops} boolean operations and {args.max_concat_ops} concatenations per level") # Updated print
    # print(f"Accept probability: {args.accept_probability}")
    # print(f"Random seed: {args.seed}")
    
    automaton, _ = sample_dot_depth(
        mean_alphabet_size=args.mean_alphabet_size,
        mean_depth=args.mean_depth,
        max_bool_ops=args.max_bool_ops, # Changed from mean_bool_ops
        max_concat_ops=args.max_concat_ops, # Changed from mean_concat_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()