from rayuela.fsa.fsa import FSA
from rayuela.base.semiring import Boolean as RayuelaBoolean
from rayuela.base.symbol import (
    Sym as RayuelaSym,
    ε as RayuelaEPSILON
)
from rayuela.base.state import State as RayuelaState
from rayuela.cfg.nonterminal import S, NT
from rayuela.cfg.cfg import CFG
from rayuela.pda.pda import TopDownPDA
from rayuela.base.symbol import ε

from recognizers.automata.automaton import State, Symbol
from recognizers.automata.reserved import ReservedSymbol
from recognizers.automata.finite_automaton import (
    FiniteAutomatonContainer,
    FiniteAutomatonTransition
)
from recognizers.automata.pushdown_automaton import (
    PushdownAutomatonContainer,
    PushdownAutomatonTransition,
    StackSymbol
)
from recognizers.grammars.grammar import Production, Nonterminal
from recognizers.grammars.context_free_grammar import ContextFreeGrammarContainer

def from_rayuela_fsa(A: FSA) -> tuple[FiniteAutomatonContainer, list[str]]:
    sorted_symbols = sorted(A.Sigma, key=lambda x: x.value)
    sorted_states = sorted(A.Q, key=lambda x: x.idx)
    symbol_to_int = { a.value: Symbol(i) for i, a in enumerate(sorted_symbols) }
    state_to_int = { q.idx: State(i) for i, q in enumerate(sorted_states) }
    (q0, _), = A.I
    M = FiniteAutomatonContainer(
        num_states=len(state_to_int),
        alphabet_size=len(symbol_to_int),
        initial_state=state_to_int[q0.idx]
    )
    for q in sorted_states:
        for a, t, _ in sorted(A.arcs(q), key=lambda x: (x[0].value, x[1].idx)):
            M.add_transition(FiniteAutomatonTransition(
                state_to_int[q.idx],
                symbol_to_int[a.value],
                state_to_int[t.idx]
            ))
    for q, _ in sorted(A.F, key=lambda x: x[0].idx):
        M.add_accept_state(state_to_int[q.idx])
    alphabet = [str(a.value) for a in sorted_symbols]
    if len(set(alphabet)) != len(alphabet):
        raise ValueError('alphabet strings are not unique')
    return M, alphabet

def from_rayuela_cfg(cfg: CFG) -> 'ContextFreeGrammarContainer':
    alphabet_minus_epsilon = cfg.Sigma.difference({RayuelaEPSILON})
    nonterminals_to_int = { A : i for i, A in enumerate(sorted(cfg.V, key=str)) }
    alphabet_size = len(alphabet_minus_epsilon)
    num_nonterminals = len(cfg.V)
    start_symbol = Nonterminal(0)
    G = ContextFreeGrammarContainer(
        alphabet_size=alphabet_size,
        num_nonterminals=num_nonterminals,
        start_symbol=start_symbol
    )
    alphabet_to_int = {key: i for i, key in enumerate(alphabet_minus_epsilon)}
    for p, _ in cfg.P:
        X = p.head
        right_hand_side = []
        for alpha in p.body:
            if isinstance(alpha, RayuelaSym):
                processed_alpha = Nonterminal(alphabet_to_int[alpha]) if alpha != RayuelaEPSILON else ReservedSymbol.EPSILON
                right_hand_side.append(processed_alpha)
            elif isinstance(alpha, NT):
                processed_alpha = Nonterminal(nonterminals_to_int[alpha]) if alpha != cfg.S else start_symbol
                right_hand_side.append(processed_alpha)
        G.add_production(Production(
            left_hand_side=Nonterminal(nonterminals_to_int[X]),
            right_hand_side=tuple(right_hand_side)
        ))

    return G

def from_rayuela_pda(pda: TopDownPDA) -> PushdownAutomatonContainer:
    alphabet_size = len(pda.Sigma) - 1
    stack_alphabet_size = len(pda.Gamma)
    num_states = len(pda.Q)
    initial_state = pda.I

    #initial_state = initial_state.idx
    initial_stack_symbol = 0
    accept_state = pda.F
    #accept_state = accept_state.idx

    M = PushdownAutomatonContainer(
        num_states=num_states,
        alphabet_size=alphabet_size,
        stack_alphabet_size=stack_alphabet_size,
        initial_state=initial_state,
        initial_stack_symbol=initial_stack_symbol,
        accept_state=accept_state
    )

    for (p, a, r, head, body), _ in pda.arcs:
        symbol = int(a.value) if a != ε else ReservedSymbol.EPSILON
        popped_symbols = tuple([StackSymbol(i.X) if (i != S) else StackSymbol(0) for i in body])

        if len(popped_symbols) == 1:
            popped_symbols = popped_symbols[0]
        pushed_symbols = tuple(reversed([StackSymbol(i.X) if (i.X != "S") else StackSymbol(0) for i in head]))

        M.add_transition(PushdownAutomatonTransition(
            state_from=p,#.idx,
            state_to=r,#.idx,
            symbol=symbol,
            popped_symbol=popped_symbols,
            pushed_symbols=pushed_symbols
        ))

    return M

def to_rayuela_pda(automaton: PushdownAutomatonContainer) -> TopDownPDA:
    pda = TopDownPDA(R=RayuelaBoolean, _S=RayuelaSym(automaton.initial_stack_symbol))
    pda.set_I(RayuelaState(automaton.initial_state))
    pda.set_F(RayuelaState(automaton.accept_state))

    for t, _ in automaton.transition_weights:
        p = t.state_from
        q = t.state_to
        a = t.symbol
        X = t.popped_symbol
        gamma = t.pushed_symbols
        pda.add(RayuelaBoolean.one,
                RayuelaState(p),
                RayuelaEPSILON if a == ReservedSymbol.EPSILON else RayuelaSym(a),
                RayuelaState(q),
                tuple(reversed([RayuelaSym(g) for g in gamma])),
                (RayuelaSym(X),)
                )
    return pda

def to_rayuela_cfg(grammar: ContextFreeGrammarContainer) -> CFG:
    cfg = CFG(R=RayuelaBoolean, _S=NT(grammar.start_symbol))

    for p, _ in grammar.production_weights:
        X = p.left_hand_side
        rhs = p.right_hand_side
        match rhs:
            case (a,):
                cfg.add(RayuelaBoolean.one, NT(X), RayuelaSym(a))
            case (Y, Z):
                cfg.add(RayuelaBoolean.one, NT(X), NT(Y), NT(Z))
    return cfg

def rayuela_cfg_to_rayuela_pda(cfg: CFG) -> TopDownPDA:
    from rayuela.pda.pda import TopDownPDA

    pda = TopDownPDA(R=cfg.R)

    pda.set_I(State(0))
    pda.set_F(State(0))

    for p, w in cfg.P:
        # A → x
        if len(p.body) == 1 and p.body[0] in cfg.Sigma.union({ε}) :
            pda.add(w, State(0), p.body[0], State(0), (), (p.head,))
        # A → B C ...
        elif all([X in cfg.V for X in p.body]):
            pda.add(w, State(0), ε, State(0), p.body, (p.head,))
        else:
            raise AssertionError(
                "Grammar mixes terminals and non-terminals in a production"
            )
    return pda
