from recognizers.automata.automaton import Symbol, State
from recognizers.automata.finite_automaton import (
    FiniteAutomatonContainer,
    FiniteAutomatonTransition
)
from recognizers.automata.trim_dfa import trim_dfa

def result(
    num_states,
    alphabet_size,
    transitions,
    initial_state,
    accept_states
):
    M = FiniteAutomatonContainer(
        num_states=num_states,
        alphabet_size=alphabet_size,
        initial_state=State(initial_state)
    )
    for q, a, r in transitions:
        M.add_transition(FiniteAutomatonTransition(State(q), Symbol(a), State(r)))
    for f in accept_states:
        M.add_accept_state(State(f))
    M = trim_dfa(M)
    return (
        M.num_states(),
        M.alphabet_size(),
        {(t.state_from, t.symbol, t.state_to) for t in M.transitions()},
        M.initial_state(),
        {q for q in M.states() if M.is_accept_state(q)}
    )

def test_reachable_from_start():
    assert result(
        6,
        1,
        [
            (0, 0, 1),
            (1, 0, 4),
            (4, 0, 0),
            (2, 0, 3),
            (3, 0, 5),
            (5, 0, 2)
        ],
        0,
        [1, 5]
    ) == (
        3,
        1,
        {
            (0, 0, 1),
            (1, 0, 2),
            (2, 0, 0)
        },
        0,
        {1}
    )

def test_reachable_from_accept():
    assert result(
        8,
        1,
        [
            (0, 0, 1),
            (1, 0, 3),
            (3, 0, 4),
            (4, 0, 1),
            (0, 0, 2),
            (2, 0, 5),
            (5, 0, 4),
            (2, 0, 6),
            (6, 0, 7),
            (5, 0, 7)
        ],
        0,
        [7]
    ) == (
        5,
        1,
        {
            (0, 0, 1),
            (1, 0, 2),
            (1, 0, 3),
            (2, 0, 4),
            (3, 0, 4)
        },
        0,
        {4}
    )

def test_reachable_from_multiple_accept():
    assert result(
        5,
        1,
        [
            (3, 0, 4),
            (3, 0, 2),
            (0, 0, 2),
            (0, 0, 1)
        ],
        3,
        [1, 2, 4]
    ) == (
        3,
        1,
        {
            (1, 0, 2),
            (1, 0, 0)
        },
        1,
        {0, 2}
    )

def test_reject_after_accept():
    assert result(
        3,
        1,
        [
            (0, 0, 1),
            (1, 0, 2)
        ],
        0,
        [1]
    ) == (
        2,
        1,
        {
            (0, 0, 1)
        },
        0,
        {1}
    )

def test_extra_states():
    assert result(
        3,
        1,
        [],
        0,
        []
    ) == (
        1,
        1,
        set(),
        0,
        set()
    )

def test_empty_cycle():
    assert result(
        3,
        1,
        [
            (0, 0, 1),
            (1, 0, 2),
            (2, 0, 0)
        ],
        0,
        []
    ) == (
        1,
        1,
        set(),
        0,
        set()
    )

def test_empty_loop():
    assert result(
        1,
        1,
        [
            (0, 0, 0)
        ],
        0,
        []
    ) == (
        1,
        1,
        set(),
        0,
        set()
    )

def test_empty_cycle_and_loop():
    assert result(
        3,
        1,
        [
            (0, 0, 1),
            (1, 0, 2),
            (2, 0, 0),
            (0, 0, 0)
        ],
        0,
        []
    ) == (
        1,
        1,
        set(),
        0,
        set()
    )

def test_empty_one_state():
    assert result(
        1,
        1,
        [],
        0,
        []
    ) == (
        1,
        1,
        set(),
        0,
        set()
    )

def test_accept_one_state():
    assert result(
        1,
        1,
        [],
        0,
        [0]
    ) == (
        1,
        1,
        set(),
        0,
        {0}
    )

def test_remove_symbol_1():
    assert result(
        3,
        2,
        [
            (0, 0, 1),
            (1, 1, 2)
        ],
        0,
        [1]
    ) == (
        2,
        1,
        {
            (0, 0, 1)
        },
        0,
        {1}
    )

def test_remove_symbol_0():
    assert result(
        3,
        2,
        [
            (0, 1, 1),
            (1, 0, 2)
        ],
        0,
        [1]
    ) == (
        2,
        1,
        {
            (0, 0, 1)
        },
        0,
        {1}
    )

def test_remap_symbol_and_state():
    assert result(
        3,
        2,
        [
            (0, 1, 2),
            (2, 0, 1)
        ],
        0,
        [2]
    ) == (
        2,
        1,
        {
            (0, 0, 1)
        },
        0,
        {1}
    )
