import unittest
import random

from recognizers.grammars.cfg_finite_check import is_infinite
from recognizers.language_sampling.sample_cnf_cfg import add_self_embedding_rules
from recognizers.grammars.context_free_grammar import (
    ContextFreeGrammarContainer,
    Production
)


def test_cfg1() -> ContextFreeGrammarContainer:
    M = ContextFreeGrammarContainer(
        num_variables=3,
        num_terminals=2
    )
    A, B, S = M.variables()
    a, b = M.terminals()
    M.add_production(Production(S, (A, A)))
    M.add_production(Production(A, (A, B)))
    M.add_production(Production(A, (B, B)))
    M.add_production(Production(A, (a, )))
    M.add_production(Production(B, (b, )))
    M.add_production(Production(S, ()))


    return M

def test_cfg2() -> ContextFreeGrammarContainer:
    M = ContextFreeGrammarContainer(
        num_variables=3,
        num_terminals=2
    )
    A, B, S = M.variables()
    a, b = M.terminals()
    M.add_production(Production(S, (A, B)))
    M.add_production(Production(S, (b, )))
    M.add_production(Production(A, (a,)))
    M.add_production(Production(A, ()))
    M.add_production(Production(B, ()))
    M.add_production(Production(B, (b, )))
    M.add_production(Production(S, ()))
    return M

class TestFiniteCheckCFG(unittest.TestCase):

    def test_finite_cfg(self) -> None:
        G = test_cfg2()
        infinite = is_infinite(G)
        self.assertFalse(infinite)

    def test_infinite_cfg(self) -> None:
        G = test_cfg1()
        infinite = is_infinite(G)
        self.assertTrue(infinite)

    def test_infinite_cfg2(self) -> None:
        G = test_cfg1()
        generator = random.Random(123)
        add_self_embedding_rules(G, 3, generator)
        infinite = is_infinite(G)
        self.assertTrue(infinite)

    def test_infinite_cfg3(self) -> None:
        G = test_cfg2()
        generator = random.Random(123)
        add_self_embedding_rules(G, 3, generator)
        infinite = is_infinite(G)
        self.assertTrue(infinite)

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