from __future__ import annotations
import numpy as np
from typing import Optional, Tuple
class Alarm:
    def __init__(self,device:int,type:int,start:int) -> None:
        pass
    #     self.device = device #needed?
        self.start = start
        self.type = type

    #     self.explained = False # None is base residual encoding 
    #     self.rest_explained = False
        
    #     self.explained_by:Optional[Tuple[int,int]] = None
    #     self.reset_explained_by:Optional[Tuple[int,int]] = None

    #     self.explained_by_device:Optional[int] = None
    #     self.reset_explained_by_device:Optional[int] = None

    #     self.cost = np.NaN #TODO initially base cost then depending on explanation cost
    #     self.reset_cost = np.NaN

    #     self.delay = None
    #     self.reset_delay = None

    #     # do not use for length computation, values only get added never removed 
    #     self.__used_as_cause_for = set() #key (effect,device) i.e. triggers effect on device
    
    # def __repr__(self) -> str:
    #     return f"Alarm(device={self.device},type={self.type},start={self.start},explained_by={self.explained_by})"
    
    # def __lt__(self, other:Alarm) -> bool:
    #     return self.start < other.start
    
    # def __gt__(self, other:Alarm) -> bool:
    #     return self.start > other.start
    
    # def __eq__(self, other:Alarm) -> bool:
    #     return self.start == other.start
    
    # def __le__(self, other:Alarm) -> bool:
    #     return self.start <= other.start
    
    # def __ge__(self, other:Alarm) -> bool:
    #     return self.start >= other.start
    
    # def set_explained(self, explained_by:Tuple[int,int], cost:float, delay:int, device:int) -> None:
    #     self.explained = True
    #     self.explained_by = explained_by
    #     self.cost = cost
    #     self.delay = delay
    #     self.explained_by_device = device
    
    # def lock_in_explained(self) -> None:
    #     self.rest_explained = self.explained
    #     self.reset_explained_by = self.explained_by
    #     self.reset_cost = self.cost
    #     self.reset_delay = self.delay
    #     self.reset_explained_by_device = self.explained_by_device

    # def reset_explained(self) -> None:
    #     self.explained = self.rest_explained
    #     self.explained_by = self.reset_explained_by
    #     self.cost = self.reset_cost
    #     self.delay = self.reset_delay
    #     self.explained_by_device = self.reset_explained_by_device
    
    # def set_cost(self, cost:float) -> None: # use after refitting distribution
    #     self.cost = cost
    
    # def init_cost(self, base_cost:float) -> None: # also sets reset cost, only use for init #TODO make part of __inti__ ? 
    #     self.cost = base_cost
    #     self.reset_cost = base_cost
    
    # def add_used_as_cause_for(self, effect:int, device:int) -> None:
    #     self.__used_as_cause_for.add((effect,device))
    
    # def check_available(self, effect:int, device:int) -> bool:
    #     return (effect,device) not in self.__used_as_cause_for