# postponed evaluation of annotations (use a class name as type hint inside of the same class)
from __future__ import annotations


class MetaNode:
    # TODO move disjoint-set functionality to a separate class
    # https://en.wikipedia.org/wiki/Disjoint-set_data_structure
    name: int
    counting_sort_index: int  # index for counting sort of edges
    disjoint_set_parent: MetaNode
    disjoint_set_rank: int  # an upper bound of the sub-tree below this node

    def __init__(self, name: int):
        self.name = name
        self.disjoint_set_parent = self

    def is_root(self) -> bool:
        """
        Returns true if this node is the canonical representation for the set that contains it.
        """
        return self.disjoint_set_parent == self

    def find_root_safe(self) -> MetaNode:
        """
        Find operation for disjoint-set.
        Does not use path compression, which means the function has no side effects.
        See: https://en.wikipedia.org/wiki/Disjoint-set_data_structure#Finding_set_representatives

        Returns the meta node that is the canonical representation for the set that contains this node.
        """
        # TODO use while loop?
        if self.is_root():
            return self
        else:
            return self.disjoint_set_parent.find_root_safe()

    def find_root(self) -> MetaNode:
        """
        Find operation for disjoint-set.
        Uses path compression, i.e. might update the parent pointers of this meta node and its ancestors.
        See: https://en.wikipedia.org/wiki/Disjoint-set_data_structure#Finding_set_representatives

        Returns the meta node that is the canonical representation for the set that contains this node.
        """
        # TODO use while loop?
        if self.is_root():
            return self
        else:
            root = self.disjoint_set_parent.find_root()
            self.disjoint_set_parent = root
            return root

    def union(self, other: MetaNode) -> bool:
        """
        Union operation for disjoint-set.
        Ranks are updated accordingly.
        See: https://en.wikipedia.org/wiki/Disjoint-set_data_structure#Union_by_rank

        Returns True iff a change was made, i.e. iff `self` and `other` were not already in the same set.
        """
        root_self = self.find_root()
        root_other = other.find_root()

        if root_self == root_other:
            # already in the same set => don't need to do anything
            return False

        # update the node with the lower rank
        if root_self.disjoint_set_rank > other.disjoint_set_rank:
            root_other.disjoint_set_parent = root_self
        else:
            root_self.disjoint_set_parent = root_other
            if root_self.disjoint_set_rank == root_other.disjoint_set_rank:
                root_other.disjoint_set_rank += 1

        return True

    def reset_root(self):
        """
        Initializes the parent pointer and rank for disjoint-set.
        If this is called on all nodes, all nodes will be alone in their sets.
        """
        self.disjoint_set_parent = self
        self.disjoint_set_rank = 1  # TODO use 0 instead of 1
