stateless-uptime-kuma/stateless_uptime_kuma/uptime_kuma.py

301 lines
8.9 KiB
Python

"""
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)})"