import dataclasses
from collections.abc import Iterable
from typing import Any, Generic, NewType, TypeVar

from recognizers.automata.semiring import Semiring
from recognizers.automata.automaton import Symbol

Variable = NewType('Variable', int)
TerminalOrVariable = Symbol | Variable
Weight = TypeVar('Weight')

# aliases
Nonterminal = Variable
TerminalOrNonterminal = TerminalOrVariable

@dataclasses.dataclass(frozen=True)
class Rule:
    left: Variable
    right: tuple[TerminalOrVariable, ...]

    # aliases
    @property
    def left_hand_side(self):
        return self.left

    @property
    def right_hand_side(self):
        return self.right

Production = Rule

class Grammar:

    def variables(self) -> Iterable[Variable]:
        raise NotImplementedError

    def nonterminals(self):
        # alias
        return self.variables()

    def num_variables(self) -> int:
        raise NotImplementedError

    def num_nonterminals(self):
        # alias
        return self.num_variables()

    def variable_index(self, variable: Variable) -> int:
        raise NotImplementedError

    def terminals(self) -> Iterable[Symbol]:
        raise NotImplementedError

    def num_terminals(self) -> int:
        raise NotImplementedError

    def alphabet_size(self) -> int:
        # alias
        return self.num_terminals()

    def rules(self) -> Iterable[Rule]:
        raise NotImplementedError

    def productions(self):
        # alias
        return self.rules()

    def start_variable(self) -> Variable:
        raise NotImplementedError

    def start_symbol(self):
        # alias
        return self.start_variable()

    def is_variable(self, X: TerminalOrVariable) -> bool:
        raise NotImplementedError

    def is_terminal(self, X: TerminalOrVariable) -> bool:
        raise NotImplementedError

class GrammarContainerBase(Grammar):

    _num_terminals: int
    _num_terminals_and_variables: int
    _start_variable: Variable
    _rules: dict[Rule, Any]

    def __init__(self,
        *,
        num_variables: int,
        num_terminals: int
    ):
        super().__init__()
        if num_variables < 1:
            raise ValueError
        self._num_terminals = num_terminals
        self._num_terminals_and_variables = num_terminals + num_variables
        self._start_variable = Nonterminal(self._num_terminals)
        self._rules = {}

    def variables(self) -> Iterable[Variable]:
        return range(self._num_terminals, self._num_terminals_and_variables) # type: ignore

    def num_variables(self) -> int:
        return self._num_terminals_and_variables - self._num_terminals

    def variable_index(self, variable: Variable) -> int:
        return variable - self._num_terminals

    def new_variable(self) -> Variable:
        nonterminal = self._num_nonterminals
        self._num_nonterminals += 1
        return Nonterminal(nonterminal)

    def new_nonterminal(self):
        # alias
        return self.new_variable()

    def terminals(self) -> Iterable[Symbol]:
        return range(self._num_terminals) # type: ignore

    def num_terminals(self) -> int:
        return self._num_terminals

    def rules(self) -> Iterable[Rule]:
        return self._rules.keys()

    def start_variable(self) -> Variable:
        return self._start_variable

    def set_start_variable(self, start_variable: Variable) -> None:
        self._start_variable = start_variable

    def is_variable(self, X: TerminalOrVariable) -> bool:
        return X >= self._num_terminals

    def is_terminal(self, X: TerminalOrVariable) -> bool:
        return X < self._num_terminals

class GrammarContainer(GrammarContainerBase):

    _rules: dict[Rule, None]

    def add_rule(self, rule: Rule) -> None:
        self._rules[rule] = None

    def add_production(self, production):
        # alias
        self.add_rule(production)

class WeightedGrammar(Grammar, Generic[Weight]):

    def semiring(self) -> Semiring[Weight]:
        raise NotImplementedError

    def rule_weights(self) -> Iterable[tuple[Rule, Weight]]:
        raise NotImplementedError

    def production_weights(self):
        # alias
        return self.rule_weights()

class WeightedGrammarContainer(WeightedGrammar[Weight], GrammarContainerBase):

    _rules: dict[Rule, Weight]
    _semiring: Semiring[Weight]

    def __init__(self,
        *,
        num_variables: int,
        num_terminals: int,
        semiring: Semiring[Weight]
    ):
        super().__init__(
            num_variables=num_variables,
            num_terminals=num_terminals
        )
        self._semiring = semiring

    def semiring(self) -> Semiring[Weight]:
        return self._semiring

    def rule_weights(self) -> Iterable[tuple[Production, Weight]]:
        return self._rules.items()

    def set_rule_weight(self, rule: Rule, weight: Weight) -> None:
        self._rules[rule] = weight

    def set_production_weight(self, production, weight):
        # alias
        self.set_rule_weight(production, weight)
