2022-06-08 00:13:42 +02:00
|
|
|
from django.core.exceptions import ValidationError
|
2018-06-04 22:34:26 +02:00
|
|
|
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
|
|
|
|
2022-04-14 23:50:10 +02:00
|
|
|
from zerver.actions.message_send import send_rate_limited_pm_notification_to_bot_owner
|
2021-07-16 22:11:10 +02:00
|
|
|
from zerver.decorator import webhook_view
|
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
|
2022-06-08 00:13:42 +02:00
|
|
|
from zerver.lib.validator import WildValue, check_string, to_wild_value
|
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,
|
2022-06-08 00:13:42 +02:00
|
|
|
payload: WildValue = REQ(argument_type="body", converter=to_wild_value),
|
2021-02-12 08:19:30 +01:00
|
|
|
) -> HttpResponse:
|
2018-12-18 22:14:44 +01:00
|
|
|
try:
|
|
|
|
body = get_body_for_http_request(payload)
|
2023-07-12 13:37:08 +02:00
|
|
|
topic = get_topic_for_http_request(payload)
|
2022-06-08 00:13:42 +02:00
|
|
|
except ValidationError:
|
2018-12-18 22:14:44 +01:00
|
|
|
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
|
|
|
|
2023-07-12 13:37:08 +02:00
|
|
|
check_send_webhook_message(request, user_profile, topic, body)
|
2022-01-31 13:44:02 +01:00
|
|
|
return json_success(request)
|
2018-06-04 22:34:26 +02:00
|
|
|
|
2021-02-12 08:19:30 +01:00
|
|
|
|
2023-07-12 13:37:08 +02:00
|
|
|
def get_topic_for_http_request(payload: WildValue) -> str:
|
2022-06-08 00:13:42 +02:00
|
|
|
return ZABBIX_TOPIC_TEMPLATE.format(hostname=payload["hostname"].tame(check_string))
|
2018-06-04 22:34:26 +02:00
|
|
|
|
2021-02-12 08:19:30 +01:00
|
|
|
|
2022-06-08 00:13:42 +02:00
|
|
|
def get_body_for_http_request(payload: WildValue) -> str:
|
|
|
|
hostname = payload["hostname"].tame(check_string)
|
|
|
|
severity = payload["severity"].tame(check_string)
|
|
|
|
status = payload["status"].tame(check_string)
|
|
|
|
item = payload["item"].tame(check_string)
|
|
|
|
trigger = payload["trigger"].tame(check_string)
|
|
|
|
link = payload["link"].tame(check_string)
|
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)
|