import itertools

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

class StateSet:

    def __init__(self):
        super().__init__()
        self.states = {}

    def __getitem__(self, key):
        if key in self.states:
            return self.states[key]
        else:
            result = self.states[key] = len(self.states)
            return result

    def __len__(self):
        return len(self.states)

    def items(self):
        return self.states.items()

def construct_dfa(num_keys: int, num_values: int):
    key_symbols = list(map(Symbol, range(num_keys)))
    value_symbols = list(map(Symbol, range(num_keys, num_keys + num_values)))
    separator = Symbol(num_keys + num_values)
    alphabet = [
        *(f'k{i}' for i in range(num_keys)),
        *(f'v{i}' for i in range(num_values)),
        '#'
    ]
    state_list = []
    state_dicts = []
    for values in itertools.product([None] + list(value_symbols), repeat=num_keys):
        key_value_dict = {}
        for key, value in zip(key_symbols, values, strict=True):
            if value is not None:
                key_value_dict[key] = value
        state_dicts.append(key_value_dict)
        D = frozenset(key_value_dict.items())
        state_list.append(('set', D))
        for k in key_symbols:
            state_list.append(('set', D, k))
        state_list.append(('query', D))
        for k in key_symbols:
            state_list.append(('query', D, k))
    states = {q : i for i, q in enumerate(state_list)}
    M = FiniteAutomatonContainer(
        num_states=len(states),
        alphabet_size=len(alphabet),
        initial_state=states[('set', frozenset())]
    )
    for state_dict in state_dicts:
        D = frozenset(state_dict.items())
        for k in key_symbols:
            if k not in state_dict:
                M.add_transition(FiniteAutomatonTransition(states[('set', D)], k, states[('set', D, k)]))
                for v in value_symbols:
                    new_state_dict = state_dict | {k : v}
                    new_D = frozenset(new_state_dict.items())
                    M.add_transition(FiniteAutomatonTransition(states[('set', D, k)], v, states[('set', new_D)]))
        M.add_transition(FiniteAutomatonTransition(states[('set', D)], separator, states[('query', D)]))
        for k in key_symbols:
            if k in state_dict:
                M.add_transition(FiniteAutomatonTransition(states[('query', D)], k, states[('query', D, k)]))
                M.add_transition(FiniteAutomatonTransition(states[('query', D, k)], state_dict[k], states[('query', D)]))
        M.add_accept_state(states[('query', D)])
    return M, alphabet
