simple-icinga-dashboard/service.py

164 lines
4.8 KiB
Python
Raw Permalink Normal View History

2020-11-21 21:35:43 +00:00
#!/usr/bin/env python3
2023-05-20 18:09:16 +00:00
import logging
import shutil
2023-05-20 18:08:08 +00:00
import sys
2023-05-20 18:09:16 +00:00
from datetime import datetime
from os import environ
2021-04-11 08:24:46 +00:00
2023-05-20 18:09:16 +00:00
import requests
2021-04-11 08:24:46 +00:00
import tomlkit
2023-05-20 18:09:16 +00:00
import urllib3
2021-04-11 08:57:02 +00:00
from mako.template import Template
2023-05-20 18:09:16 +00:00
2020-11-21 21:35:43 +00:00
urllib3.disable_warnings()
CONFIGFILE = environ.get('STATUSPAGE_CONFIG', 'config.toml')
2023-05-20 18:09:16 +00:00
class StatusPage:
def get_api_result(self):
if self.services:
2023-05-20 18:08:08 +00:00
log.debug('services already exist, returning early')
return self.services
2023-05-20 18:09:16 +00:00
headers = {'Accept': 'application/json', 'X-HTTP-Method-Override': 'GET'}
requestbody = {
2023-05-20 18:09:16 +00:00
"attrs": [
"name",
"state",
"last_check_result",
"host_name",
"display_name",
],
"joins": ["host", "host.state", "host.last_check_result", "host.vars"],
"filter": self.config['filters']['services'],
}
r = requests.get(
'{}/v1/objects/services'.format(self.config['icinga2_api']['baseurl']),
headers=headers,
json=requestbody,
2023-05-20 18:09:16 +00:00
auth=(
self.config['icinga2_api']['username'],
self.config['icinga2_api']['password'],
),
verify=False,
)
2023-05-20 18:08:08 +00:00
self.logger.info(f'got http status code {r.status_code}')
self.logger.debug(r.text)
2023-05-20 18:09:16 +00:00
if r.status_code == 200:
self.services = r.json()['results']
else:
r.raise_for_status()
2023-05-20 18:08:08 +00:00
self.logger.info(f'got {len(self.services)} services from api')
return self.services
def prettify(self, text):
2021-04-10 14:05:09 +00:00
for search, replace in self.config.get('prettify', {}).items():
text = text.replace(search, replace)
return text
2021-04-11 08:57:02 +00:00
def get_services_per_host(self):
2021-04-10 15:39:06 +00:00
state_to_design_mapping = [
('success', 'OK'),
('warning', 'WARNING'),
('danger', 'CRITICAL'),
('info', 'UNKNOWN'),
]
2021-04-11 08:57:02 +00:00
result = {}
for service in self.get_api_result():
2023-05-20 18:09:16 +00:00
self.logger.info(
f'now processing {service["attrs"]["host_name"]} "{service["attrs"]["display_name"]}"'
)
2023-05-20 18:08:08 +00:00
self.logger.debug(service)
2021-04-11 08:57:02 +00:00
host = service['joins']['host']['vars']['pretty_name']
if host not in result:
result[host] = {
'hostname': service['attrs']['host_name'],
'services': {},
}
if service['joins']['host']['state'] == 0:
result[host]['host_badge'] = 'success'
result[host]['host_state'] = 'UP'
else:
result[host]['host_badge'] = 'danger'
result[host]['host_state'] = 'DOWN'
self.ragecounter += 10
state = int(service['attrs']['state'])
2021-04-11 08:57:02 +00:00
if state in (1, 2):
self.ragecounter += state
2021-04-10 15:39:06 +00:00
2023-05-20 18:09:16 +00:00
result[host]['services'][
self.prettify(service['attrs']['display_name'])
] = {
2021-04-11 08:57:02 +00:00
'badge': state_to_design_mapping[state][0],
'state': state_to_design_mapping[state][1],
}
2023-05-20 18:08:08 +00:00
self.logger.info(f'ragecounter is now {self.ragecounter}')
2021-04-11 08:57:02 +00:00
return result
2021-04-11 08:57:02 +00:00
def render_html(self, service_details):
2021-04-10 15:39:06 +00:00
if self.ragecounter == 0:
mood = '🆗'
elif self.ragecounter < 10:
mood = '🚨'
else:
mood = '🔥'
2021-04-11 08:57:02 +00:00
2023-05-20 18:08:08 +00:00
self.logger.info('rendering output html')
start = datetime.now()
2023-05-20 18:09:16 +00:00
template = Template(
filename=self.config['output'].get('template', 'template.html')
)
2021-04-11 08:57:02 +00:00
output = template.render(
title=self.config['output'].get('page_title', 'Status Page'),
mood=mood,
hosts=service_details,
)
2023-05-20 18:08:08 +00:00
end = datetime.now()
self.logger.info(f'rendered in {(end-start).total_seconds():.09f}s')
2021-04-11 08:57:02 +00:00
with open(self.config['output']['filename'], 'w') as f:
f.write(output)
def __init__(self):
self.config = tomlkit.loads(open(CONFIGFILE).read())
self.services = {}
2021-04-10 15:39:06 +00:00
self.ragecounter = 0
2020-11-21 21:35:43 +00:00
2023-05-20 18:08:08 +00:00
self.logger = logging.getLogger('StatusPage')
handler = logging.StreamHandler(sys.stdout)
2023-05-20 18:09:16 +00:00
formatter = logging.Formatter(
'%(levelname)s {%(filename)s:%(lineno)d} %(message)s'
)
2023-05-20 18:08:08 +00:00
handler.setFormatter(formatter)
self.logger.addHandler(handler)
self.logger.setLevel(self.config.get('loglevel', 'INFO'))
2023-05-20 18:09:16 +00:00
2020-11-21 21:35:43 +00:00
if __name__ == "__main__":
page = StatusPage()
2021-04-11 08:57:02 +00:00
2023-05-20 18:08:08 +00:00
try:
service_details = page.get_services_per_host()
page.render_html(service_details)
except Exception as e:
shutil.copyfile('error.html', page.config['output']['filename'])
raise e