from recognizers.automata.finite_automaton import FiniteAutomatonContainer
from recognizers.grammars.cfg_finite_check import scc

def dfa_to_graph(dfa: FiniteAutomatonContainer) -> tuple[dict, bool]:
    """
    Converts a DFA into a reachability  graph.
    Returns also whether the graph contains self-loops.
    """
    graph = {}
    self_loops = False
    for t in dfa.transitions():
        left = t.state_from
        right = t.state_to
        if left not in graph:
            graph[left] = set()
        graph[left].add(right)
        if left == right:
            self_loops = True
    return graph, self_loops

def is_finite(dfa: FiniteAutomatonContainer) -> bool:
    graph, self_loops = dfa_to_graph(dfa)
    if self_loops:
        return False
    components = scc(graph)
    for comp in components:
        if len(comp) > 1:
            return False
    return True