from recognizers.grammars.context_free_grammar import ContextFreeGrammarContainer

def cfg_to_graph(grammar: ContextFreeGrammarContainer) -> tuple[dict, bool]:
    """
    Converts a CFG into a reachability  graph.
    Returns also whether the graph contains self-loops.
    """
    graph = {}
    self_loops = False
    for p in grammar.rules():
        left = p.left
        right = p.right
        if len(right) < 2:
            pass
        if left not in graph:
            graph[left] = set()
        for X in right:
            graph[left].add(X)
            if left == X:
                self_loops = True
    return graph, self_loops

def scc(graph: dict) -> list[tuple]:
    """
    Computes the strongly connected components of a graph. Based on Tarjan (1972)'s algorithm.
    Implementation following closely Alexander Clark's implementation from
    https://github.com/alexc17/testcfg/tree/master
    """
    index_counter = [0]
    stack = []
    lowlinks = {}
    index = {}
    result = []

    def strongconnect(node):
        index[node] = index_counter[0]
        lowlinks[node] = index_counter[0]
        index_counter[0] += 1
        stack.append(node)

        try:
            successors = graph[node]
        except:
            successors = []
        for successor in successors:
            if successor not in lowlinks:
                strongconnect(successor)
                lowlinks[node] = min(lowlinks[node], lowlinks[successor])
            elif successor in stack:
                lowlinks[node] = min(lowlinks[node], index[successor])

        if lowlinks[node] == index[node]:
            connected_component = []

            while True:
                successor = stack.pop()
                connected_component.append(successor)
                if successor == node:
                    break
            component = tuple(connected_component)
            result.append(component)

    for node in graph:
        if node not in lowlinks:
            strongconnect(node)
    return result

def is_infinite(grammar: ContextFreeGrammarContainer) -> bool:
    graph, self_loops = cfg_to_graph(grammar)
    if self_loops:
        return True
    components = scc(graph)
    for comp in components:
        if len(comp) > 1:
            return True
    return False