2020-01-14 22:06:24 +01:00
|
|
|
import operator
|
2021-12-17 07:03:22 +01:00
|
|
|
from typing import Dict, List
|
2018-09-13 00:45:22 +02:00
|
|
|
|
|
|
|
from django.http import HttpRequest, HttpResponse
|
|
|
|
|
2021-07-16 22:11:10 +02:00
|
|
|
from zerver.decorator import webhook_view
|
2019-02-02 23:53:55 +01:00
|
|
|
from zerver.lib.response import json_success
|
2023-09-27 19:01:31 +02:00
|
|
|
from zerver.lib.typed_endpoint import JsonBodyPayload, typed_endpoint
|
2023-08-12 09:34:31 +02:00
|
|
|
from zerver.lib.validator import WildValue, check_bool, check_int, check_string
|
2020-01-14 22:06:24 +01:00
|
|
|
from zerver.lib.webhooks.common import check_send_webhook_message
|
2018-09-13 00:45:22 +02:00
|
|
|
from zerver.models import UserProfile
|
|
|
|
|
2019-04-10 05:40:00 +02:00
|
|
|
ANSIBLETOWER_DEFAULT_MESSAGE_TEMPLATE = "{friendly_name}: [#{id} {name}]({url}) {status}."
|
2018-09-13 00:45:22 +02:00
|
|
|
|
|
|
|
|
2019-04-10 05:40:00 +02:00
|
|
|
ANSIBLETOWER_JOB_MESSAGE_TEMPLATE = """
|
|
|
|
{friendly_name}: [#{id} {name}]({url}) {status}:
|
|
|
|
{hosts_final_data}
|
|
|
|
""".strip()
|
2018-09-13 00:45:22 +02:00
|
|
|
|
2021-02-12 08:20:45 +01:00
|
|
|
ANSIBLETOWER_JOB_HOST_ROW_TEMPLATE = "* {hostname}: {status}\n"
|
2018-09-13 00:45:22 +02:00
|
|
|
|
2021-02-12 08:19:30 +01:00
|
|
|
|
2021-02-12 08:20:45 +01:00
|
|
|
@webhook_view("AnsibleTower")
|
2023-08-12 09:34:31 +02:00
|
|
|
@typed_endpoint
|
2021-02-12 08:19:30 +01:00
|
|
|
def api_ansibletower_webhook(
|
|
|
|
request: HttpRequest,
|
|
|
|
user_profile: UserProfile,
|
2023-08-12 09:34:31 +02:00
|
|
|
*,
|
2023-09-27 19:01:31 +02:00
|
|
|
payload: JsonBodyPayload[WildValue],
|
2021-02-12 08:19:30 +01:00
|
|
|
) -> HttpResponse:
|
2018-09-13 00:45:22 +02:00
|
|
|
body = get_body(payload)
|
2024-01-17 15:53:30 +01:00
|
|
|
topic_name = payload["name"].tame(check_string)
|
2018-09-13 00:45:22 +02:00
|
|
|
|
2024-01-17 15:53:30 +01:00
|
|
|
check_send_webhook_message(request, user_profile, topic_name, body)
|
2022-01-31 13:44:02 +01:00
|
|
|
return json_success(request)
|
2018-09-13 00:45:22 +02:00
|
|
|
|
2021-02-12 08:19:30 +01:00
|
|
|
|
2021-12-17 07:03:22 +01:00
|
|
|
def extract_friendly_name(payload: WildValue) -> str:
|
|
|
|
tentative_job_name = payload.get("friendly_name", "").tame(check_string)
|
2020-02-01 10:00:06 +01:00
|
|
|
if not tentative_job_name:
|
2021-12-17 07:03:22 +01:00
|
|
|
url = payload["url"].tame(check_string)
|
2020-02-01 10:00:06 +01:00
|
|
|
segments = url.split("/")
|
|
|
|
tentative_job_name = segments[-3]
|
|
|
|
if tentative_job_name == "jobs":
|
|
|
|
tentative_job_name = "Job"
|
|
|
|
return tentative_job_name
|
|
|
|
|
2021-02-12 08:19:30 +01:00
|
|
|
|
2021-12-17 07:03:22 +01:00
|
|
|
def get_body(payload: WildValue) -> str:
|
2020-02-01 10:00:06 +01:00
|
|
|
friendly_name = extract_friendly_name(payload)
|
2021-02-12 08:20:45 +01:00
|
|
|
if friendly_name == "Job":
|
2018-09-13 00:45:22 +02:00
|
|
|
hosts_data = []
|
2021-12-17 07:03:22 +01:00
|
|
|
for host, host_data in payload["hosts"].items():
|
|
|
|
if host_data["failed"].tame(check_bool):
|
2021-02-12 08:20:45 +01:00
|
|
|
hoststatus = "Failed"
|
2021-12-17 07:03:22 +01:00
|
|
|
else:
|
2021-02-12 08:20:45 +01:00
|
|
|
hoststatus = "Success"
|
2021-02-12 08:19:30 +01:00
|
|
|
hosts_data.append(
|
|
|
|
{
|
2021-02-12 08:20:45 +01:00
|
|
|
"hostname": host,
|
|
|
|
"status": hoststatus,
|
2021-02-12 08:19:30 +01:00
|
|
|
}
|
|
|
|
)
|
|
|
|
|
2021-02-12 08:20:45 +01:00
|
|
|
if payload["status"] == "successful":
|
|
|
|
status = "was successful"
|
2018-09-13 00:45:22 +02:00
|
|
|
else:
|
2021-02-12 08:20:45 +01:00
|
|
|
status = "failed"
|
2018-09-13 00:45:22 +02:00
|
|
|
|
|
|
|
return ANSIBLETOWER_JOB_MESSAGE_TEMPLATE.format(
|
2021-12-17 07:03:22 +01:00
|
|
|
name=payload["name"].tame(check_string),
|
2020-02-01 10:00:06 +01:00
|
|
|
friendly_name=friendly_name,
|
2021-12-17 07:03:22 +01:00
|
|
|
id=payload["id"].tame(check_int),
|
|
|
|
url=payload["url"].tame(check_string),
|
2018-09-13 00:45:22 +02:00
|
|
|
status=status,
|
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
|
|
|
hosts_final_data=get_hosts_content(hosts_data),
|
2018-09-13 00:45:22 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
else:
|
2021-12-17 07:03:22 +01:00
|
|
|
if payload["status"].tame(check_string) == "successful":
|
2021-02-12 08:20:45 +01:00
|
|
|
status = "was successful"
|
2018-09-13 00:45:22 +02:00
|
|
|
else:
|
2021-02-12 08:20:45 +01:00
|
|
|
status = "failed"
|
2018-09-13 00:45:22 +02:00
|
|
|
|
|
|
|
data = {
|
2021-12-17 07:03:22 +01:00
|
|
|
"name": payload["name"].tame(check_string),
|
2020-02-01 10:00:06 +01:00
|
|
|
"friendly_name": friendly_name,
|
2021-12-17 07:03:22 +01:00
|
|
|
"id": payload["id"].tame(check_int),
|
|
|
|
"url": payload["url"].tame(check_string),
|
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
|
|
|
"status": status,
|
2018-09-13 00:45:22 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return ANSIBLETOWER_DEFAULT_MESSAGE_TEMPLATE.format(**data)
|
|
|
|
|
2021-02-12 08:19:30 +01:00
|
|
|
|
2021-12-17 07:03:22 +01:00
|
|
|
def get_hosts_content(hosts_data: List[Dict[str, str]]) -> str:
|
2021-02-12 08:20:45 +01:00
|
|
|
hosts_data = sorted(hosts_data, key=operator.itemgetter("hostname"))
|
|
|
|
hosts_content = ""
|
2018-09-13 00:45:22 +02:00
|
|
|
for host in hosts_data:
|
|
|
|
hosts_content += ANSIBLETOWER_JOB_HOST_ROW_TEMPLATE.format(
|
2021-12-17 07:03:22 +01:00
|
|
|
hostname=host["hostname"],
|
|
|
|
status=host["status"],
|
2018-09-13 00:45:22 +02:00
|
|
|
)
|
|
|
|
return hosts_content
|