Source code for grafanarmadillo.flow

"""Pieces for templating multiple dashboards at once."""
import json
from abc import ABC, abstractmethod
from dataclasses import dataclass, field
from pathlib import Path
from typing import List, Optional, Union

from grafana_client import GrafanaApi

from grafanarmadillo.alerter import Alerter
from grafanarmadillo.dashboarder import Dashboarder
from grafanarmadillo.find import Finder
from grafanarmadillo.templator import Templator
from grafanarmadillo.util import resolve_object_to_filepath


[docs]class Store(ABC): """A destination or source for items."""
[docs] @abstractmethod def read_alert(self, name): """Read an alert from this store."""
[docs] @abstractmethod def read_dashboard(self, name): """Read a dashboard from this store."""
[docs] @abstractmethod def write_alert(self, name, alert): """Write an alert to this store."""
[docs] @abstractmethod def write_dashbaord(self, name, dashboard): """Write an alert to this store."""
[docs]@dataclass class FileStore(Store): """ Store and retrieve Grafana objects in the filesystem. Objects will be stored under the same path as in Grafana. For example, a dashboard titled "MyDashboard" in a folder titled "MyFolder" will appear at `{root}/MyFolder/MyDashboard.json`. """ root: Path @staticmethod def _read(file: Path) -> dict: with file.with_suffix(".json").open(mode="r", encoding="utf-8") as f: return json.load(f) @staticmethod def _write(file: Path, content: dict): file.parent.mkdir(exist_ok=True) with file.with_suffix(".json").open(mode="w", encoding="utf-8") as f: json.dump(content, f)
[docs] def read_alert(self, name): """Read an alert from this store.""" return self._read(resolve_object_to_filepath(self.root, name))
[docs] def read_dashboard(self, name): """Read a dashboard from this store.""" return self._read(resolve_object_to_filepath(self.root, name))
[docs] def write_alert(self, name, alert): """Write an alert to this store.""" return self._write(resolve_object_to_filepath(self.root, name), alert)
[docs] def write_dashbaord(self, name, dashboard): """Write an alert to this store.""" return self._write(resolve_object_to_filepath(self.root, name), dashboard)
[docs]@dataclass class GrafanaStore(Store): """Store and retrieve objects from a Grafana instance.""" gfn: GrafanaApi
[docs] def read_alert(self, name): """Read an alert from this store.""" finder, alerter = Finder(self.gfn), Alerter(self.gfn) alert_info, _ = finder.create_or_get_alert(name) alert, _ = alerter.export_alert(alert_info) return alert_info
[docs] def read_dashboard(self, name): """Read a dashboard from this store.""" finder, dashboarder = Finder(self.gfn), Dashboarder(self.gfn) dashboard_info, _ = finder.create_or_get_dashboard(name) dashboard_content, _ = dashboarder.export_dashboard(dashboard_info) return dashboard_content
[docs] def write_alert(self, name, alert): """Write an alert to this store.""" finder, alerter = Finder(self.gfn), Alerter(self.gfn) alert_info, folder_info = finder.create_or_get_alert(name) alerter.import_alert(alert, folder_info)
[docs] def write_dashbaord(self, name, dashboard): """Write an alert to this store.""" finder, dashboarder = Finder(self.gfn), Dashboarder(self.gfn) dashboard_info, folder = finder.create_or_get_dashboard(name) dashboarder.import_dashboard(dashboard, folder)
[docs]@dataclass class Alert: """Flowable request for an Alert.""" name_obj: str name_tmpl: str templator: Templator
[docs]@dataclass class Dashboard: """Flowable request for a Dashboard.""" name_obj: str name_tmpl: str templator: Templator
Flowable = Union[Alert, Dashboard]
[docs]class FlowException(Exception): """Wrapped Exception of running a Flow.""" def __init__(self, item: Flowable, cause: Optional[BaseException] = None): self.item = item if cause: self.__cause__ = cause
[docs]@dataclass class FlowResult: """Result of running a Flow.""" successes: List[Flowable] failures: List[FlowException]
[docs] def raise_first(self): """Raise the first exception, if present.""" if self.failures: raise self.failures[0]
[docs]@dataclass class Flow: """A collection of templating actions to do.""" store_obj: Store store_tmpl: Store flows: List[Flowable] = field(default=list)
[docs] def append(self, flow: Flowable): """Add a Flowable request to this Flow.""" self.flows.append(flow)
[docs] def obj_to_tmpl(self) -> FlowResult: """Import from the source to the destination.""" return self.run(obj_to_tmpl=True)
[docs] def tmpl_to_obj(self) -> FlowResult: """Export from the destination to the source.""" return self.run(obj_to_tmpl=False)
[docs] def run(self, obj_to_tmpl: bool) -> FlowResult: """Run the flow.""" failures = [] successes = [] for item in self.flows: try: if isinstance(item, Alert): if obj_to_tmpl: obj = self.store_obj.read_alert(item.name_obj) tmpl = item.templator.make_template_from_dashboard(obj) self.store_tmpl.write_alert(item.name_tmpl, tmpl) else: tmpl = self.store_tmpl.read_alert(item.name_tmpl) info = self.store_obj.read_alert(item.name_obj) obj = item.templator.make_dashboard_from_template(info, tmpl) self.store_obj.write_alert(item.name_obj, obj) elif isinstance(item, Dashboard): if obj_to_tmpl: obj = self.store_obj.read_dashboard(item.name_obj) tmpl = item.templator.make_template_from_dashboard(obj) self.store_tmpl.write_dashbaord(item.name_tmpl, tmpl) else: tmpl = self.store_tmpl.read_dashboard(item.name_tmpl) info = self.store_obj.read_dashboard(item.name_obj) obj = item.templator.make_dashboard_from_template(info, tmpl) self.store_obj.write_dashbaord(item.name_obj, obj) else: raise TypeError( f"Invalid flow, expected one of {Alert.__name__}, {Dashboard.__name__}, received {item.__class__.__name__}") successes.append(item) except Exception as e: failures.append(FlowException(item, e)) return FlowResult(successes, failures)