From 5c7571f952c5361dae27abb8c96be6ee1f760668 Mon Sep 17 00:00:00 2001
From: Benedikt Ziemons
Date: Sun, 27 Sep 2020 17:55:20 +0200
Subject: [PATCH] Add better output for prometheus alerts
and add testrequest JSON file.
---
README.md | 3 +-
testrequest.json => testrequest_gitlab.json | 0
testrequest_prometheus.json | 37 ++++++
wmn.py | 134 +++++++++++++-------
4 files changed, 123 insertions(+), 51 deletions(-)
rename testrequest.json => testrequest_gitlab.json (100%)
create mode 100644 testrequest_prometheus.json
diff --git a/README.md b/README.md
index 7db8943..342234f 100644
--- a/README.md
+++ b/README.md
@@ -7,9 +7,8 @@ behind a reverse-proxy with HTTPS.
# Testing the Hook locally
- Start the webserver locally by `env FLASK_APP=wmn.py flask run`
- Or have your IDE do it for you
-- Send a POST request using curl `curl -i -X POST "localhost:5000/matrix?channel=%21yhEUnvhAZZFKRStdXb%3Amatrix.org" -H "X-Gitlab-Event: Push Hook" -H "X-Gitlab-Token: ..." -H "Content-Type: application/json" --data-binary @./testrequest.json`
+- Send a POST request using curl `curl -i -X POST "localhost:5000/matrix?channel=%21yhEUnvhAZZFKRStdXb%3Amatrix.org" -H "X-Gitlab-Event: Push Hook" -H "X-Gitlab-Token: ..." -H "Content-Type: application/json" --data-binary @./testrequest_gitlab.json`
- The part after `channel=` is the room ID which can retrieved from Matrix channels you are part of
- `%21` escapes ! in URI
- `%3A` escapes : in URI
- The `X-Gitlab-Token` must correspond to the one provided in `config.yaml`
-
diff --git a/testrequest.json b/testrequest_gitlab.json
similarity index 100%
rename from testrequest.json
rename to testrequest_gitlab.json
diff --git a/testrequest_prometheus.json b/testrequest_prometheus.json
new file mode 100644
index 0000000..503dc2b
--- /dev/null
+++ b/testrequest_prometheus.json
@@ -0,0 +1,37 @@
+{
+ "receiver": "webhook",
+ "status": "firing",
+ "alerts": [
+ {
+ "status": "firing",
+ "labels": {
+ "alertname": "Test",
+ "dc": "eu-west-1",
+ "instance": "prometheus.example.com:9090",
+ "job": "prometheus24"
+ },
+ "annotations": {
+ "description": "some description"
+ },
+ "startsAt": "2018-08-03T09:52:26.739266876+02:00",
+ "endsAt": "0001-01-01T00:00:00Z",
+ "generatorURL": "http://prometheus.example.com:9090/graph?g0.expr=go_memstats_alloc_bytes+%3E+0\u0026g0.tab=1"
+ }
+ ],
+ "groupLabels": {
+ "alertname": "Test",
+ "job": "prometheus24"
+ },
+ "commonLabels": {
+ "alertname": "Test",
+ "dc": "eu-west-1",
+ "instance": "prometheus.example.com:9090",
+ "job": "prometheus24"
+ },
+ "commonAnnotations": {
+ "description": "some description"
+ },
+ "externalURL": "http://prometheus.example.com:9093",
+ "version": "4",
+ "groupKey": "{}:{alertname=\"Test\", job=\"prometheus24\"}"
+}
\ No newline at end of file
diff --git a/wmn.py b/wmn.py
index d140071..6b2c1bb 100644
--- a/wmn.py
+++ b/wmn.py
@@ -10,7 +10,8 @@ from flask import Flask, request, abort
from matrix_client.client import MatrixClient
from matrix_client.errors import MatrixRequestError
-application = Flask(__name__)
+app = Flask(__name__)
+application = app
# Not going to care for specifics like the underscore.
# Generally match room alias or id [!#]anything:example.com with unicode support.
@@ -63,6 +64,10 @@ def get_msg_type():
abort(400)
+def color_format_html(color_hex: str, text: str):
+ return f'{text}'
+
+
def iter_first_line(string: str):
return iter(map(str.rstrip, string.lstrip().splitlines(keepends=False)))
@@ -203,6 +208,9 @@ def process_prometheus_request():
msgtype = get_msg_type()
room = get_a_room()
+ if not request.json:
+ abort(400)
+
# written for version 4 of the alertmanager webhook JSON
# https://prometheus.io/docs/alerting/configuration/#webhook_config
@@ -210,10 +218,13 @@ def process_prometheus_request():
_status_colors = {"resolved": "34A91D", "firing": "EF2929"}
if text is None:
text = status
- if status in _status_colors:
- return f'{text}'
- else:
- return text
+ return color_format_html(_status_colors.get(status, "FFFFFF"), text)
+
+ def color_severity_html(severity: str, text: typing.Optional[str] = None):
+ _severity_colors = {"warning": "EFAC29", "critical": "EF2929"}
+ if text is None:
+ text = severity
+ return color_format_html(_severity_colors.get(severity, "FFFFFF"), text)
def parse_promtime(date_string):
match = promtime_to_isotime_pattern.match(date_string)
@@ -225,63 +236,88 @@ def process_prometheus_request():
grps[-1] = '+00:00'
return datetime.fromisoformat(''.join(grps))
+ def alert_title(status: str, alertname: str, generator_url: str):
+ if alertname:
+ alertname = " alert " + alertname
+
+ if status:
+ status_msg = status.upper() if status == "firing" else status.title()
+ title = status_msg + alertname
+ html_title = color_status_html(status, title)
+ elif alertname:
+ title = alertname
+ html_title = title
+ else:
+ title = ""
+ html_title = title
+
+ if title:
+ title = f"*{title}*"
+ if generator_url:
+ title = f"{title} {generator_url}"
+
+ if html_title:
+ html_title = f"{html_title}"
+ if generator_url:
+ html_title = f'{html_title}'
+
+ return title, html_title
+
def extract_alert_message(alert: typing.Dict[str, typing.Any]) -> typing.Tuple[str, str]:
"""Takes the alert object and returns (text, html) as a string tuple."""
- alert_status = alert.get("status", "None")
- alert_labels = str(alert.get("labels", None))
- alert_annotations = str(alert.get("annotations", None))
- alert_start = alert.get("startsAt", None)
- alert_end = alert.get("endsAt", None)
+ labels = alert.get("labels", {})
+ severity = labels.get("severity", "")
+ annotations = alert.get("annotations", {})
+ description = annotations.get("description", "")
+ if not description:
+ description = annotations.get("summary", "")
+
alert_daterange = []
- if alert_start is not None and alert_end != '0001-01-01T00:00:00Z':
- alert_start = parse_promtime(alert_start).strftime("%d. %b %y %H:%M %Z").rstrip()
- alert_daterange.append(f'Started at {alert_start}')
- if alert_end is not None and alert_end != '0001-01-01T00:00:00Z':
- alert_end = parse_promtime(alert_end).strftime("%d. %b %y %H:%M %Z").rstrip()
- alert_daterange.append(f'Ended at {alert_end}')
- alert_daterange = "" if len(alert_daterange) == 0 else f'({", ".join(alert_daterange)})'
- alert_generator_url = alert.get("generatorURL", "None")
+ if "startsAt" in alert and alert["startsAt"] != '0001-01-01T00:00:00Z':
+ alert_start = parse_promtime(alert["startsAt"]).strftime("%d. %b %y %H:%M %Z").rstrip()
+ alert_daterange.append(f'started at {alert_start}')
+ if "endsAt" in alert and alert["endsAt"] != '0001-01-01T00:00:00Z':
+ alert_end = parse_promtime(alert["endsAt"]).strftime("%d. %b %y %H:%M %Z").rstrip()
+ alert_daterange.append(f'ended at {alert_end}')
+ alert_daterange = ", ".join(alert_daterange)
- return (
- f'[{alert_status}] Labels: {alert_labels}, Annotations: {alert_annotations} - {alert_daterange} | Generator: {alert_generator_url}',
- f'{color_status_html(alert_status)} Labels: {alert_labels}, Annotations: {alert_annotations} - {alert_daterange} | Generator: {alert_generator_url}',
+ title, html_title = alert_title(
+ status=alert.get("status", ""),
+ alertname=labels.get("alertname", ""),
+ generator_url=alert.get("generatorURL", "")
)
+ if severity:
+ html_severity = f"Severity: {color_severity_html(severity)}"
+ severity = severity.upper() if severity == 'critical' else severity.title()
+ severity = f"Severity: {severity}"
+ else:
+ html_severity = ""
- def extract_prometheus_message() -> typing.Tuple[str, str]:
- """Dissects the request's JSON and returns (text, html) as a string tuple."""
-
- group_key = request.json.get("groupKey", "None")
- status = request.json.get("status", "None")
- receiver = request.json.get("receiver", "None")
- group_labels = str(request.json.get("groupLabels", None))
- common_labels = str(request.json.get("commonLabels", None))
- common_annotations = str(request.json.get("commonAnnotations", None))
- ext_url = request.json.get("externalURL", "None")
- alerts = request.json.get("alerts", []) # type: typing.List[typing.Dict[str, typing.Any]]
-
- text_alerts, html_alerts = zip(*map(extract_alert_message, alerts))
- text_alerts = "\n" + "\n".join((f"- {msg}" for msg in text_alerts))
- html_alerts = "
\n\n" + "\n".join((f" - {msg}
" for msg in html_alerts)) + "\n
"
-
+ html_parts = [html_title, html_severity, description, alert_daterange]
+ html_message = "
\n".join(filter(bool, html_parts))
+ html_message = f"
{html_message}
" if html_message else ""
return (
- f'*{status.title()} alert for group {group_key}*\n Receiver: {receiver}\n Labels: {group_labels} | {common_labels}\n Annotations: {common_annotations}\n External URL: {ext_url}\nAlerts:{text_alerts}',
- f'{color_status_html(status, f"{status.title()} alert for group {group_key}")}
\n Receiver: {receiver}
\n Labels: {group_labels} | {common_labels}
\n Annotations: {common_annotations}
\n External URL: {ext_url}
\nAlerts:{html_alerts}',
+ " \n".join(filter(bool, [title, severity, description, alert_daterange])),
+ html_message
)
- try:
- html, body = extract_prometheus_message()
- except (LookupError, ValueError, TypeError):
- print("Error parsing JSON and forming message:", file=sys.stderr)
- traceback.print_exc()
- print(file=sys.stderr, flush=True)
- return "Error parsing JSON and forming message", 500
-
try:
client = MatrixClient(cfg["matrix"]["server"])
client.login(username=cfg["matrix"]["username"], password=cfg["matrix"]["password"])
room = client.join_room(room_id_or_alias=room)
- room.send_html(html=html, body=body, msgtype=msgtype)
+ try:
+ for body, html in map(extract_alert_message, request.json.get("alerts", [])):
+ if html and body:
+ room.send_html(html=html, body=body, msgtype=msgtype)
+ elif body:
+ room.send_text(body)
+ except (LookupError, ValueError, TypeError):
+ room.send_text("Error parsing data in prometheus request")
+ print("Error parsing JSON and forming message:", file=sys.stderr)
+ traceback.print_exc()
+ print(file=sys.stderr, flush=True)
+ return "Error parsing JSON and forming message", 500
except MatrixRequestError as e:
return matrix_error(e)
@@ -289,7 +325,7 @@ def process_prometheus_request():
return "", 204
-@application.route('/matrix', methods=("POST",))
+@app.route('/matrix', methods=("POST",))
def notify():
if 'X-Gitlab-Token' in request.headers:
return process_gitlab_request()