from typing import Any, Dict, Union
from logger import VSSMLogger
from time_monitor import time_monitor


class Hypothesis:
    """
    Represents a hypothesis in the version space. '?' represents any value and NormalDistribution represents
    continuous attributes.
    """

    def __init__(self, attributes: Union['Hypothesis', Dict[str, Any]]):
        if isinstance(attributes, Hypothesis):
            self.attributes = attributes.attributes
        else:
            self.attributes = attributes

    @time_monitor
    def is_more_general_than(self, other: 'Hypothesis') -> bool:
        for attr, this_value in self.attributes.items():
            other_d = other if isinstance(other, dict) else other.attributes
            other_value = other_d[attr]
            # If the current value is a wildcard, it's by default more general
            if this_value != '?' and this_value != other_value:
                return False
        return True

    @staticmethod
    @time_monitor
    def generalize(h1: Union['Hypothesis', dict], h2: Union['Hypothesis', dict]) -> 'Hypothesis':
        """
        Minimally generalize a hypothesis to cover the given instance.

        This method computes the minimal generalization of two hypotheses (or a
        hypothesis and an instance). If attribute values differ, they are replaced by a wildcard.

        @param h1: The first hypothesis (or an instance dictionary).
        @param h2: The second hypothesis (or an instance dictionary).
        @return A new, generalized Hypothesis object.
        """
        c1 = h1.attributes if isinstance(h1, Hypothesis) else h1
        c2 = h2.attributes if isinstance(h2, Hypothesis) else h2
        new_attributes = {}
        for attr in c1.keys():
            #  If values are not the same, generalize to a wildcard
            if c1[attr] != c2[attr]:
                new_attributes[attr] = '?'
            else:
                new_attributes[attr] = c1[attr]

        return Hypothesis(new_attributes)

    def __eq__(self, other: object) -> bool:
        if not isinstance(other, Hypothesis): return False
        return self.attributes == other.attributes

    def __hash__(self) -> int:
        return hash(tuple(sorted(self.attributes.items())))

    def __str__(self) -> str:
        return str(self.attributes)

    def __repr__(self) -> str:
        return f"Hypothesis({self.attributes})"

    def __copy__(self) -> 'Hypothesis':
        return Hypothesis(self.attributes)

    def __deepcopy__(self) -> 'Hypothesis':
        return self.__copy__()

    def __json__(self):
        return {
            "attributes": {k: (v.__json__() if hasattr(v, '__json__') else v) for k, v in self.attributes.items()}
        }

class VersionSpace:
    """
    Represents a version space with S and G boundary sets.
    """
    def __init__(self, s_set: list[Hypothesis], g_set: list[Hypothesis], logging_enabled: bool = False):
        self.S = s_set
        self.G = g_set
        self.logger = VSSMLogger(logging_enabled)


    @time_monitor
    def is_consistent(self) -> bool:
        """
        Check if the version space is consistent.

        @return True if consistent, False otherwise
        """
        if not self.S or not self.G:
            self.logger.log("VS is inconsistent because S or G set is empty.", "FAIL")
            return False
        for s in self.S:
            if not any(g.is_more_general_than(s) for g in self.G):
                self.logger.log(f"S-hypothesis {s} is not covered by any G-hypothesis.", "FAIL")
                return False
        self.logger.log("Version space remains consistent.", "SUCCESS")
        return True


    @time_monitor
    def covers(self, instance: Hypothesis) -> bool:
        """
        Check if an instance is covered by this version space. 'Covered' means that some specific set is more general
        than the instance.

        @param instance: A hypothesis to check
        @return True if the hypothesis can be included, False otherwise
        """
        for s in self.S:
            if s.is_more_general_than(instance):
                return True
        return False


    @time_monitor
    def can_include(self, h: Hypothesis) -> bool:
        """
        Check if the hypothesis can be included in this version space.

        @param h: A hypothesis to check
        @return True if the hypothesis can be included, False otherwise
        """
        if not self.G:
            return False

        for g in self.G:
            if g.is_more_general_than(h):
                return True
        return False


    def __str__(self) -> str:
        return f"VersionSpace(S={self.S}, G={self.G})"


    def __repr__(self) -> str:
        return f"VersionSpace(s_set={repr(self.S)}, g_set={repr(self.G)})"
