2018-06-04 22:34:26 +02:00
|
|
|
from typing import Any, Dict
|
|
|
|
|
|
|
|
from django.http import HttpRequest, HttpResponse
|
2021-04-16 00:57:30 +02:00
|
|
|
from django.utils.translation import gettext as _
|
2018-06-04 22:34:26 +02:00
|
|
|
|
2021-07-16 22:11:10 +02:00
|
|
|
from zerver.decorator import webhook_view
|
2018-12-18 22:14:44 +01:00
|
|
|
from zerver.lib.actions import send_rate_limited_pm_notification_to_bot_owner
|
2021-06-30 18:35:50 +02:00
|
|
|
from zerver.lib.exceptions import JsonableError
|
2021-07-16 22:11:10 +02:00
|
|
|
from zerver.lib.request import REQ, has_request_variables
|
2021-06-30 18:35:50 +02:00
|
|
|
from zerver.lib.response import json_success
|
2018-12-18 22:14:44 +01:00
|
|
|
from zerver.lib.send_email import FromAddress
|
2020-01-14 22:06:24 +01:00
|
|
|
from zerver.lib.webhooks.common import check_send_webhook_message
|
2018-06-04 22:34:26 +02:00
|
|
|
from zerver.models import UserProfile
|
|
|
|
|
2018-12-18 22:14:44 +01:00
|
|
|
MISCONFIGURED_PAYLOAD_ERROR_MESSAGE = """
|
|
|
|
Hi there! Your bot {bot_name} just received a Zabbix payload that is missing
|
|
|
|
some data that Zulip requires. This usually indicates a configuration issue
|
|
|
|
in your Zabbix webhook settings. Please make sure that you set the
|
2021-03-15 01:58:53 +01:00
|
|
|
script correctly and provide all the required parameters
|
2018-12-18 22:14:44 +01:00
|
|
|
when configuring the Zabbix webhook. Contact {support_email} if you
|
|
|
|
need further help!
|
|
|
|
"""
|
|
|
|
|
2021-02-12 08:20:45 +01:00
|
|
|
ZABBIX_TOPIC_TEMPLATE = "{hostname}"
|
2019-04-17 01:53:42 +02:00
|
|
|
ZABBIX_MESSAGE_TEMPLATE = """
|
|
|
|
{status} ({severity}) alert on [{hostname}]({link}):
|
|
|
|
* {trigger}
|
|
|
|
* {item}
|
|
|
|
""".strip()
|
2018-06-04 22:34:26 +02:00
|
|
|
|
2021-02-12 08:19:30 +01:00
|
|
|
|
2021-02-12 08:20:45 +01:00
|
|
|
@webhook_view("Zabbix")
|
2018-06-04 22:34:26 +02:00
|
|
|
@has_request_variables
|
2021-02-12 08:19:30 +01:00
|
|
|
def api_zabbix_webhook(
|
|
|
|
request: HttpRequest,
|
|
|
|
user_profile: UserProfile,
|
2021-02-12 08:20:45 +01:00
|
|
|
payload: Dict[str, Any] = REQ(argument_type="body"),
|
2021-02-12 08:19:30 +01:00
|
|
|
) -> HttpResponse:
|
2018-06-04 22:34:26 +02:00
|
|
|
|
2018-12-18 22:14:44 +01:00
|
|
|
try:
|
|
|
|
body = get_body_for_http_request(payload)
|
|
|
|
subject = get_subject_for_http_request(payload)
|
|
|
|
except KeyError:
|
|
|
|
message = MISCONFIGURED_PAYLOAD_ERROR_MESSAGE.format(
|
|
|
|
bot_name=user_profile.full_name,
|
|
|
|
support_email=FromAddress.SUPPORT,
|
|
|
|
).strip()
|
2021-02-12 08:19:30 +01:00
|
|
|
send_rate_limited_pm_notification_to_bot_owner(user_profile, user_profile.realm, message)
|
2018-12-18 22:14:44 +01:00
|
|
|
|
2021-06-30 18:35:50 +02:00
|
|
|
raise JsonableError(_("Invalid payload"))
|
2018-06-04 22:34:26 +02:00
|
|
|
|
|
|
|
check_send_webhook_message(request, user_profile, subject, body)
|
|
|
|
return json_success()
|
|
|
|
|
2021-02-12 08:19:30 +01:00
|
|
|
|
2018-06-04 22:34:26 +02:00
|
|
|
def get_subject_for_http_request(payload: Dict[str, Any]) -> str:
|
2021-02-12 08:20:45 +01:00
|
|
|
return ZABBIX_TOPIC_TEMPLATE.format(hostname=payload["hostname"])
|
2018-06-04 22:34:26 +02:00
|
|
|
|
2021-02-12 08:19:30 +01:00
|
|
|
|
2018-06-04 22:34:26 +02:00
|
|
|
def get_body_for_http_request(payload: Dict[str, Any]) -> str:
|
2021-02-12 08:20:45 +01:00
|
|
|
hostname = payload["hostname"]
|
|
|
|
severity = payload["severity"]
|
|
|
|
status = payload["status"]
|
|
|
|
item = payload["item"]
|
|
|
|
trigger = payload["trigger"]
|
|
|
|
link = payload["link"]
|
2018-06-04 22:34:26 +02:00
|
|
|
|
|
|
|
data = {
|
|
|
|
"hostname": hostname,
|
|
|
|
"severity": severity,
|
|
|
|
"status": status,
|
|
|
|
"item": item,
|
|
|
|
"trigger": trigger,
|
python: Use trailing commas consistently.
Automatically generated by the following script, based on the output
of lint with flake8-comma:
import re
import sys
last_filename = None
last_row = None
lines = []
for msg in sys.stdin:
m = re.match(
r"\x1b\[35mflake8 \|\x1b\[0m \x1b\[1;31m(.+):(\d+):(\d+): (\w+)", msg
)
if m:
filename, row_str, col_str, err = m.groups()
row, col = int(row_str), int(col_str)
if filename == last_filename:
assert last_row != row
else:
if last_filename is not None:
with open(last_filename, "w") as f:
f.writelines(lines)
with open(filename) as f:
lines = f.readlines()
last_filename = filename
last_row = row
line = lines[row - 1]
if err in ["C812", "C815"]:
lines[row - 1] = line[: col - 1] + "," + line[col - 1 :]
elif err in ["C819"]:
assert line[col - 2] == ","
lines[row - 1] = line[: col - 2] + line[col - 1 :].lstrip(" ")
if last_filename is not None:
with open(last_filename, "w") as f:
f.writelines(lines)
Signed-off-by: Anders Kaseorg <anders@zulipchat.com>
2020-04-10 05:23:40 +02:00
|
|
|
"link": link,
|
2018-06-04 22:34:26 +02:00
|
|
|
}
|
|
|
|
return ZABBIX_MESSAGE_TEMPLATE.format(**data)
|