import unittest
import random

from recognizers.automata.podfa_check import is_podfa
from recognizers.automata.finite_automaton import (
    FiniteAutomatonContainer,
    FiniteAutomatonTransition
)
from recognizers.automata.automaton import (
    Transition,
    State,
    Symbol
)
from recognizers.language_sampling.sample_podfa import sample_podfa


def test_dfa1() -> FiniteAutomatonContainer:
    q0, q1 = map(State, range(2))
    a, b = map(Symbol, range(2))
    M = FiniteAutomatonContainer(
        num_states=2,
        alphabet_size=2,
        initial_state=q0
    )
    M.add_transition(FiniteAutomatonTransition(q0, a, q1))
    M.add_transition(FiniteAutomatonTransition(q1, b, q0))
    M.add_accept_state(q0)
    return M

def test_dfa2() -> FiniteAutomatonContainer:
    q0, q1, q2 = map(State, range(3))
    a, b = map(Symbol, range(2))
    M = FiniteAutomatonContainer(
        num_states=3,
        alphabet_size=2,
        initial_state=q0
    )
    M.add_transition(FiniteAutomatonTransition(q0, a, q1))
    M.add_transition(FiniteAutomatonTransition(q0, a, q2))
    M.add_transition(FiniteAutomatonTransition(q1, b, q1))
    M.add_accept_state(q0)
    return M

def test_dfa3() -> FiniteAutomatonContainer:
    q0, q1, q2, q3 = map(State, range(4))
    a, b = map(Symbol, range(2))
    M = FiniteAutomatonContainer(
        num_states=4,
        alphabet_size=2,
        initial_state=q0
    )
    M.add_transition(FiniteAutomatonTransition(q0, a, q1))
    M.add_transition(FiniteAutomatonTransition(q1, a, q2))
    M.add_transition(FiniteAutomatonTransition(q2, b, q3))
    M.add_transition(FiniteAutomatonTransition(q3, a, q1))
    M.add_accept_state(q0)
    return M

class TestPODFACheckDFA(unittest.TestCase):

    def test_dfa_true(self) -> None:
        M = test_dfa1()
        podfa = is_podfa(M)
        self.assertFalse(podfa)

    def test_dfa_true2(self) -> None:
        M = test_dfa1()
        podfa = is_podfa(M)
        self.assertFalse(podfa)

    def test_podfa(self) -> None:
        M = test_dfa2()
        podfa = is_podfa(M)
        self.assertTrue(podfa)

    def test_podfa_random(self) -> None:
        for i in range(100):
            generator = random.Random(i)
            M = sample_podfa(
                mean_num_states=10,
                mean_alphabet_size=10,
                transition_probability=0.5,
                accept_probability=0.4,
                generator=generator
            )
            podfa = is_podfa(M)
            self.assertTrue(podfa)

if __name__ == '__main__':
    unittest.main()