""" Classes to make the needed operations to reach the specified state. """ import logging import sys from requests.api import delete logger = logging.getLogger(__name__) class Manager: def __init__(self, api, target_tree={}, prune_unused=False): self.api = api self.prune_unused = prune_unused self.target_tree = target_tree self.to_prune = [] def process(self): self.sync_tags() self.sync_notifications() self.sync_monitors() self.sync_status_pages() self.sync_settings() self.prune() self.save() def save(self): for v in self.target_tree.values(): for i in v: i.save() # this method should be safe to be called in whatever order def prune(self): for e in self.to_prune: e.prune() def sync_monitors(self): logger.debug(f"Syncing monitors...") old = self.api.get_monitors() new = self.target_tree.get("monitors", []) self.sync(new, old, "name", Monitor) def sync_notifications(self): logger.debug(f"Syncing notifications...") old = self.api.get_notifications() new = self.target_tree.get("notifications", []) self.sync(new, old, "name", Notification) def sync_tags(self): logger.debug(f"Syncing tags...") old = self.api.get_tags() new = self.target_tree.get("tags", []) self.sync(new, old, "name", Tag) def sync_status_pages(self): logger.debug(f"Syncing status pages...") old = self.api.get_status_pages() new = self.target_tree.get("status_pages", []) self.sync(new, old, "slug", StatusPage) def sync_settings(self): if "settings" not in self.target_tree: return old = self.api.get_settings() new = self.target_tree["settings"][0] # We need all of the arguments; settings are saved all at once. # We take any settings that are not explicitly set back from the server. for k, v in old.items(): if k not in new.kwargs: new.kwargs[k] = v def sync(self, new, old, pk, item_t): indexed_old = {elem[pk]: elem for elem in old} indexed_new = {elem.name: elem for elem in new} for k in new: if k.name in indexed_old: k.id = indexed_old[k.name]["id"] logger.debug(f"Synced item named {k}") if k.old_name is not None: logger.warn(f"Found unused oldName for {k}") elif k.old_name in indexed_old: k.id = indexed_old[k.old_name]["id"] logger.info(f"Found renamed item {k.old_name} -> {k}") else: k.id = None # Useless logger.debug(f"Creating key {k}") for k in indexed_old: if k not in indexed_new: if item_t == StatusPage: self.to_prune.append(item_t(api=None, name=indexed_old[k]["slug"], id=indexed_old[k]["id"])) else: self.to_prune.append(item_t(api=self.api, name=indexed_old[k]["name"], id=indexed_old[k]["id"])) class Item: def __init__(self, api, name, id, old_name=None): self.api = api self.name = name self.id = id self.old_name = old_name self.saved = False def save(self): raise NotImplementedError() def prune(self): print(f"Would prune: {self}") def __setattr__(self, name, value): if name != "saved": self.saved = False object.__setattr__(self, name, value) def __str__(self): return self.name class Monitor(Item): def __init__( self, api, name, id=None, old_name=None, tags=[], notifications=[], **kwargs ): super().__init__(api, name, id, old_name) self.kwargs = kwargs self.tags = tags self.notifications = notifications self.saved = False def get_status_codes(self): if "accepted_statuscodes" not in self.kwargs: return set(range(200, 300)) accepted_statuscodes = set() for codes in self.kwargs["accepted_statuscodes"]: if "-" in codes: c = codes.split("-") if len(c) != 2: logger.error( f"Fatal: status codes for {self} must be in the format 'number' or 'number-number'" ) sys.exit(1) a, b = int(c[0]), int(c[1]) accepted_statuscodes.update(range(a, b)) else: accepted_statuscodes.add(int(codes)) return accepted_statuscodes def save(self): if self.saved: return logger.debug(f"Saving {repr(self)}") for t, _ in self.tags: t.save() for n in self.notifications: n.save() if self.id is None: rslt = self.api.add_monitor( name=self.name, notificationIDList=[i.id for i in self.notifications], **self.kwargs, ) self.id = rslt["monitorID"] for t, value in self.tags: self.api.add_monitor_tag(tag_id=t.id, monitor_id=self.id, value=value) else: rslt = self.api.edit_monitor( self.id, name=self.name, notificationIDList=[i.id for i in self.notifications], **self.kwargs, ) current_tags = set( (i["tag_id"], i["value"]) for i in self.api.get_monitor(self.id)["tags"] ) for t, v in self.tags: if v is None: v = "" if (t.id, v) not in current_tags: logger.debug(f"Adding tag: '{t}, {v}'") self.api.add_monitor_tag(tag_id=t.id, monitor_id=self.id, value=v) self.saved = True def prune(self): logger.debug(f"Deleting monitor {self.name}") self.api.delete_monitor(self.id) def __repr__(self): return f"Monitor({str(self)})" class Tag(Item): def __init__(self, api, name, id=None, old_name=None, color="#000000"): super().__init__(api, name, id, old_name) self.tag = name self.color = color def save(self): if self.saved: return logger.debug(f"Saving {repr(self)}") if self.id is None: rslt = self.api.add_tag( name=self.tag, color=self.color, ) self.id = rslt["id"] else: self.api.edit_tag( id_=self.id, name=self.tag, color=self.color, ) self.saved = True def prune(self): pass def __repr__(self): return f"Tag({str(self)})" class Notification(Item): def __init__(self, api, name, id=None, old_name=None, **kwargs): super().__init__(api, name, id, old_name) self.kwargs = kwargs def save(self): if self.saved: return logger.debug(f"Saving {repr(self)}") if self.id is None: rslt = self.api.add_notification( name=self.name, **self.kwargs, ) self.id = rslt["id"] else: self.api.edit_notification( id_=self.id, name=self.name, **self.kwargs, ) self.saved = True def __repr__(self): return f"Notification({str(self)})" class StatusPage(Item): def __init__(self, api, name, id=None, old_name=None, **kwargs): super().__init__(api, name, id, old_name) self.kwargs = kwargs def save(self): if self.saved: return logger.debug(f"Saving {repr(self)}") if self.id is None: self.api.add_status_page( slug=self.name, title=self.kwargs["title"], ) rslt = self.api.get_status_page(self.name) self.id = rslt["id"] kwargs = self.kwargs for group in kwargs.get("publicGroupList", []): if "monitorList" in group: monitorList = [] for monitor in group["monitorList"]: monitorList.append({"id": monitor.id}) group["monitorList"] = monitorList self.api.save_status_page( slug=self.name, **kwargs, ) self.saved = True def __repr__(self): return f"StatusPage({str(self)})" class Settings(Item): def __init__(self, api, **kwargs): super().__init__(api, "settings", None, None) self.kwargs = kwargs def save(self): if self.saved: return logger.debug(f"Saving {repr(self)}") self.api.set_settings(**self.kwargs) self.saved = True def __repr__(self): return f"Settings({str(self.kwargs)})"