Source code for grafanarmadillo.templator

"""Make and fill templates for dashboards."""
from typing import Callable, Dict, NewType

from grafanarmadillo.types import (
	DashboardContent,
	DashboardPanel,
	DashboardSearchResult,
)
from grafanarmadillo.util import (
	map_json_strings,
	project_dashboard_identity,
	project_dict,
)


DashboardTransformer = Callable[[DashboardContent], DashboardContent]


[docs]def nop(d: DashboardContent) -> DashboardContent: """Pass template through.""" return d
[docs]def findreplace(context: Dict[str, str]) -> DashboardTransformer: """Make DashboardTransformer to make replacements in strings in dashboards.""" def replace_strings(s: str): out = s for k, v in context.items(): out = out.replace(k, v) return out def _findreplace(d: DashboardContent) -> DashboardContent: return map_json_strings(replace_strings, d) return _findreplace
[docs]def combine_transformers(*transformers: DashboardTransformer) -> DashboardTransformer: """Chain transformers together into one big transformer.""" def _chained(d: DashboardContent): out = d for t in transformers: out = t(out) return out return _chained
[docs]def panel_transformer(f: Callable[[DashboardPanel], DashboardPanel]) -> DashboardTransformer: """ Make DashboardTransformer which processes all panels in a dashboard. Will omit a dashboard if function returns None """ def _panel_transformer(d: DashboardContent) -> DashboardContent: out = d.copy() new_panels = list(filter(lambda p: bool(p), (f(p) for p in d["panels"]))) out["panels"] = new_panels return DashboardContent(out) return _panel_transformer
[docs]class Templator(object): """Collection of methods for filling and making templates.""" def __init__( self, make_template: DashboardTransformer = nop, fill_template: DashboardTransformer = nop, ) -> None: super().__init__() self.make_template = make_template self.fill_template = fill_template
[docs] def make_template_from_dashboard( self, dashboard: DashboardContent ) -> DashboardContent: """Convert a dashboard into a one ready for templating.""" new = dashboard.copy() new = project_dict( new, {"id", "uid"}, inverse=True ) # we don't erase the title so that we can template it later return self.make_template(DashboardContent(new))
[docs] def make_dashboard_from_template( self, dashboard_info: DashboardSearchResult, template: DashboardContent ) -> DashboardContent: """Inflate a template.""" new = template.copy() new.update(project_dashboard_identity(dashboard_info)) return self.fill_template(DashboardContent(new))
EnvMapping = NewType("EnvMapping", Dict[str, Dict[str, str]]) TOK_AUTO_MAPPING = "$auto"
[docs]def make_mapping_templator(mapping: EnvMapping, env_grafana: str, env_template: str) -> Templator: """Assemble the templator from the environment mapping.""" mapping_grafana = mapping[env_grafana] if env_template == TOK_AUTO_MAPPING: mapping_template = {k: "${%s}" % k for k in mapping_grafana.keys()} else: mapping_template = mapping[env_template] # if some keys in the src mapping are not in the dst mapping missing = mapping_grafana.keys() - mapping_template.keys() if missing: raise ValueError(f"Some keys in the source mapping are not present in the destination mapping. {missing=}") grafana_to_template = {v: mapping_template[k] for k, v in mapping_grafana.items()} template_to_grafana = {v: k for k, v in grafana_to_template.items()} return Templator(make_template=findreplace(grafana_to_template), fill_template=findreplace(template_to_grafana))