from __future__ import annotations

from dataclasses import dataclass
from typing import ClassVar, Dict, Final

from tgm.exceptions import EventOrderedConversionError


@dataclass(frozen=True, slots=True)
class TimeDeltaDG:
    """Represents the time granularity for a temporal index in a dynamic graph.

    This class is used to define the resolution at which events or interactions
    are indexed in a dynamic/temporal graph. It supports both standard temporal
    units (e.g., seconds, minutes, days) and a special event-ordered unit for strictly
    sequential indices.

    Args:
        unit (str): The time unit, e.g., 's', 'm', 'h', 'D', or 'r' for event-ordered.
        value (int, optional): Multiplier for the unit. Must be a positive integer.

    Raises:
        ValueError: If `value` is not a positive integer.
        ValueError: If `unit` is event-ordered and `value` != 1.
        ValueError: If `unit` is not recognized among allowed temporal units.

    Note:
        For event-ordered units ('r'), only value = 1 is permitted.
    """

    unit: str
    value: int = 1

    _EVENT_ORDERED: ClassVar[str] = 'r'
    _UNIT_TO_NANOS: ClassVar[Dict[str, int]] = {
        'Y': 1000 * 1000 * 1000 * 60 * 60 * 24 * 365,
        'M': 1000 * 1000 * 1000 * 60 * 60 * 24 * 30,
        'W': 1000 * 1000 * 1000 * 60 * 60 * 24 * 7,
        'D': 1000 * 1000 * 1000 * 60 * 60 * 24,
        'h': 1000 * 1000 * 1000 * 60 * 60,
        'm': 1000 * 1000 * 1000 * 60,
        's': 1000 * 1000 * 1000,
        'ms': 1000 * 1000,
        'us': 1000,
        'ns': 1,
    }

    def __post_init__(self) -> None:
        if not isinstance(self.value, int) or self.value <= 0:
            raise ValueError(f'Value must be a positive integer, got: {self.value}')
        if self.is_event_ordered and self.value != 1:
            raise ValueError(f'Only value=1 is supported for event-ordered TimeDeltaDG')
        if self.is_time_ordered and self.unit not in TimeDeltaDG._UNIT_TO_NANOS:
            raise ValueError(
                f'Unknown unit: {self.unit}, expected one of {[TimeDeltaDG._EVENT_ORDERED] + list(TimeDeltaDG._UNIT_TO_NANOS.keys())}'
            )

    @property
    def is_event_ordered(self) -> bool:
        """Return True if this is the special event-ordered unit ('r')."""
        return self.unit == TimeDeltaDG._EVENT_ORDERED

    @property
    def is_time_ordered(self) -> bool:
        """Return True if this is not special event-ordered unit ('r')."""
        return not self.is_event_ordered

    def is_coarser_than(self, other: str | TimeDeltaDG) -> bool:
        """Return True if this granularity is strictly coarser than `other`.

        Args:
            other (str | TimeDeltaDG): The time delta to compare against.

        Raises:
            EventOrderedConversionError: If either self or `other` is event-ordered.
        """
        return self.convert(other) > 1

    def convert(self, time_delta: str | TimeDeltaDG) -> float:
        """Convert this granularity into the scale of another time delta.

        Args:
            time_delta (str | TimeDeltaDG): Target time delta to convert into.

        Returns:
            float: Ratio of self to target granularity.

        Raises:
            EventOrderedConversionError: If either self or target granularity is event-ordered.
        """
        if isinstance(time_delta, str):
            time_delta = TimeDeltaDG(time_delta)
        if self.is_event_ordered or time_delta.is_event_ordered:
            raise EventOrderedConversionError(
                'Cannot compare granularity for event-ordered TimeDeltaDG'
            )
        return self._convert(time_delta)

    def _convert(self, other: TimeDeltaDG) -> float:
        self_nanos = TimeDeltaDG._UNIT_TO_NANOS[self.unit]
        other_nanos = TimeDeltaDG._UNIT_TO_NANOS[other.unit]

        invert_unit = False
        if self_nanos > other_nanos:
            # The unit ratio is safe to integer divide without precision error
            unit_ratio = self_nanos // other_nanos
        else:
            invert_unit = True
            unit_ratio = other_nanos // self_nanos

        value_ratio = self.value / other.value
        return value_ratio / unit_ratio if invert_unit else value_ratio * unit_ratio


TGB_TIME_DELTAS: Final[Dict[str, TimeDeltaDG]] = {
    'tgbl-wiki': TimeDeltaDG('s'),
    'tgbl-subreddit': TimeDeltaDG('s'),
    'tgbl-lastfm': TimeDeltaDG('s'),
    'tgbl-review': TimeDeltaDG('s'),
    'tgbl-coin': TimeDeltaDG('s'),
    'tgbl-flight': TimeDeltaDG('s'),
    'tgbl-comment': TimeDeltaDG('s'),
    'tgbn-trade': TimeDeltaDG('Y'),
    'tgbn-genre': TimeDeltaDG('s'),
    'tgbn-reddit': TimeDeltaDG('s'),
    'tgbn-token': TimeDeltaDG('s'),
}
