zulip/zerver/webhooks/github/view.py

777 lines
25 KiB
Python
Raw Normal View History

2017-11-16 00:43:10 +01:00
import re
2016-10-25 14:50:42 +02:00
from functools import partial
from typing import Any, Callable, Dict, Optional
2017-11-16 00:43:10 +01:00
2016-10-25 14:50:42 +02:00
from django.http import HttpRequest, HttpResponse
2017-11-16 00:43:10 +01:00
from zerver.decorator import log_exception_to_webhook_logger, webhook_view
from zerver.lib.exceptions import UnsupportedWebhookEventType
from zerver.lib.request import REQ, has_request_variables
2016-10-25 14:50:42 +02:00
from zerver.lib.response import json_success
from zerver.lib.webhooks.common import (
check_send_webhook_message,
get_http_headers_from_filename,
get_setup_webhook_message,
validate_extract_webhook_http_header,
)
from zerver.lib.webhooks.git import (
CONTENT_MESSAGE_TEMPLATE,
TOPIC_WITH_BRANCH_TEMPLATE,
TOPIC_WITH_PR_OR_ISSUE_INFO_TEMPLATE,
get_commits_comment_action_message,
get_issue_event_message,
get_pull_request_event_message,
get_push_commits_event_message,
get_push_tag_event_message,
get_release_event_message,
)
from zerver.models import UserProfile
fixture_to_headers = get_http_headers_from_filename("HTTP_X_GITHUB_EVENT")
2016-10-25 14:50:42 +02:00
class Helper:
def __init__(
self,
payload: Dict[str, Any],
include_title: bool,
) -> None:
self.payload = payload
self.include_title = include_title
def log_unsupported(self, event: str) -> None:
summary = f"The '{event}' event isn't currently supported by the GitHub webhook"
log_exception_to_webhook_logger(
summary=summary,
unsupported_event=True,
)
def get_opened_or_update_pull_request_body(helper: Helper) -> str:
payload = helper.payload
include_title = helper.include_title
pull_request = payload["pull_request"]
action = payload["action"]
if action == "synchronize":
action = "updated"
2016-10-25 14:50:42 +02:00
assignee = None
if pull_request.get("assignee"):
assignee = pull_request["assignee"]["login"]
description = None
changes = payload.get("changes", {})
if "body" in changes or action == "opened":
description = pull_request["body"]
2016-10-25 14:50:42 +02:00
return get_pull_request_event_message(
get_sender_name(payload),
action,
pull_request["html_url"],
target_branch=pull_request["head"]["ref"],
base_branch=pull_request["base"]["ref"],
message=description,
assignee=assignee,
number=pull_request["number"],
title=pull_request["title"] if include_title else None,
2016-10-25 14:50:42 +02:00
)
def get_assigned_or_unassigned_pull_request_body(helper: Helper) -> str:
payload = helper.payload
include_title = helper.include_title
pull_request = payload["pull_request"]
assignee = pull_request.get("assignee")
if assignee is not None:
assignee = assignee.get("login")
base_message = get_pull_request_event_message(
get_sender_name(payload),
payload["action"],
pull_request["html_url"],
number=pull_request["number"],
title=pull_request["title"] if include_title else None,
)
if assignee is not None:
return f"{base_message[:-1]} to {assignee}."
return base_message
def get_closed_pull_request_body(helper: Helper) -> str:
payload = helper.payload
include_title = helper.include_title
pull_request = payload["pull_request"]
action = "merged" if pull_request["merged"] else "closed without merge"
2016-10-25 14:50:42 +02:00
return get_pull_request_event_message(
get_sender_name(payload),
action,
pull_request["html_url"],
number=pull_request["number"],
title=pull_request["title"] if include_title else None,
2016-10-25 14:50:42 +02:00
)
def get_membership_body(helper: Helper) -> str:
payload = helper.payload
action = payload["action"]
member = payload["member"]
team_name = payload["team"]["name"]
2016-10-25 14:50:42 +02:00
return "{sender} {action} [{username}]({html_url}) {preposition} the {team_name} team.".format(
sender=get_sender_name(payload),
action=action,
username=member["login"],
html_url=member["html_url"],
preposition="from" if action == "removed" else "to",
team_name=team_name,
2016-10-25 14:50:42 +02:00
)
def get_member_body(helper: Helper) -> str:
payload = helper.payload
return "{} {} [{}]({}) to [{}]({}).".format(
2016-10-25 14:50:42 +02:00
get_sender_name(payload),
payload["action"],
payload["member"]["login"],
payload["member"]["html_url"],
2016-10-25 14:50:42 +02:00
get_repository_name(payload),
payload["repository"]["html_url"],
2016-10-25 14:50:42 +02:00
)
def get_issue_body(helper: Helper) -> str:
payload = helper.payload
include_title = helper.include_title
action = payload["action"]
issue = payload["issue"]
assignee = issue["assignee"]
2016-10-25 14:50:42 +02:00
return get_issue_event_message(
get_sender_name(payload),
action,
issue["html_url"],
issue["number"],
issue["body"],
assignee=assignee["login"] if assignee else None,
title=issue["title"] if include_title else None,
2016-10-25 14:50:42 +02:00
)
def get_issue_comment_body(helper: Helper) -> str:
payload = helper.payload
include_title = helper.include_title
action = payload["action"]
comment = payload["comment"]
issue = payload["issue"]
2016-10-25 14:50:42 +02:00
if action == "created":
action = "[commented]"
2016-10-25 14:50:42 +02:00
else:
action = f"{action} a [comment]"
action += "({}) on".format(comment["html_url"])
2016-10-25 14:50:42 +02:00
return get_issue_event_message(
get_sender_name(payload),
action,
issue["html_url"],
issue["number"],
comment["body"],
title=issue["title"] if include_title else None,
2016-10-25 14:50:42 +02:00
)
def get_fork_body(helper: Helper) -> str:
payload = helper.payload
forkee = payload["forkee"]
return "{} forked [{}]({}).".format(
2016-10-25 14:50:42 +02:00
get_sender_name(payload),
forkee["name"],
forkee["html_url"],
2016-10-25 14:50:42 +02:00
)
def get_deployment_body(helper: Helper) -> str:
payload = helper.payload
return f"{get_sender_name(payload)} created new deployment."
2016-10-25 14:50:42 +02:00
def get_change_deployment_status_body(helper: Helper) -> str:
payload = helper.payload
return "Deployment changed status to {}.".format(
payload["deployment_status"]["state"],
2016-10-25 14:50:42 +02:00
)
def get_create_or_delete_body(helper: Helper, action: str) -> str:
payload = helper.payload
ref_type = payload["ref_type"]
return "{} {} {} {}.".format(
2016-10-25 14:50:42 +02:00
get_sender_name(payload),
action,
ref_type,
payload["ref"],
2017-01-24 06:34:26 +01:00
).rstrip()
2016-10-25 14:50:42 +02:00
def get_commit_comment_body(helper: Helper) -> str:
payload = helper.payload
comment = payload["comment"]
comment_url = comment["html_url"]
commit_url = comment_url.split("#", 1)[0]
action = f"[commented]({comment_url})"
2016-10-25 14:50:42 +02:00
return get_commits_comment_action_message(
get_sender_name(payload),
action,
commit_url,
comment.get("commit_id"),
comment["body"],
2016-10-25 14:50:42 +02:00
)
def get_push_tags_body(helper: Helper) -> str:
payload = helper.payload
2016-10-25 14:50:42 +02:00
return get_push_tag_event_message(
get_sender_name(payload),
get_tag_name_from_ref(payload["ref"]),
action="pushed" if payload.get("created") else "removed",
2016-10-25 14:50:42 +02:00
)
def get_push_commits_body(helper: Helper) -> str:
payload = helper.payload
commits_data = [
{
"name": (commit.get("author").get("username") or commit.get("author").get("name")),
"sha": commit["id"],
"url": commit["url"],
"message": commit["message"],
}
for commit in payload["commits"]
]
2016-10-25 14:50:42 +02:00
return get_push_commits_event_message(
get_sender_name(payload),
payload["compare"],
get_branch_name_from_ref(payload["ref"]),
commits_data,
deleted=payload["deleted"],
2016-10-25 14:50:42 +02:00
)
def get_public_body(helper: Helper) -> str:
payload = helper.payload
return "{} made the repository [{}]({}) public.".format(
2016-10-25 14:50:42 +02:00
get_sender_name(payload),
get_repository_full_name(payload),
payload["repository"]["html_url"],
2016-10-25 14:50:42 +02:00
)
def get_wiki_pages_body(helper: Helper) -> str:
payload = helper.payload
wiki_page_info_template = "* {action} [{title}]({url})\n"
wiki_info = ""
for page in payload["pages"]:
2016-10-25 14:50:42 +02:00
wiki_info += wiki_page_info_template.format(
action=page["action"],
title=page["title"],
url=page["html_url"],
2016-10-25 14:50:42 +02:00
)
return f"{get_sender_name(payload)}:\n{wiki_info.rstrip()}"
2016-10-25 14:50:42 +02:00
def get_watch_body(helper: Helper) -> str:
payload = helper.payload
return "{} starred the repository [{}]({}).".format(
2016-10-25 14:50:42 +02:00
get_sender_name(payload),
get_repository_full_name(payload),
payload["repository"]["html_url"],
2016-10-25 14:50:42 +02:00
)
def get_repository_body(helper: Helper) -> str:
payload = helper.payload
return "{} {} the repository [{}]({}).".format(
2016-10-25 14:50:42 +02:00
get_sender_name(payload),
payload.get("action"),
get_repository_full_name(payload),
payload["repository"]["html_url"],
2016-10-25 14:50:42 +02:00
)
def get_add_team_body(helper: Helper) -> str:
payload = helper.payload
return "The repository [{}]({}) was added to team {}.".format(
get_repository_full_name(payload),
payload["repository"]["html_url"],
payload["team"]["name"],
2016-10-25 14:50:42 +02:00
)
def get_team_body(helper: Helper) -> str:
payload = helper.payload
changes = payload["changes"]
if "description" in changes:
actor = payload["sender"]["login"]
new_description = payload["team"]["description"]
return f"**{actor}** changed the team description to:\n```quote\n{new_description}\n```"
if "name" in changes:
original_name = changes["name"]["from"]
new_name = payload["team"]["name"]
return f"Team `{original_name}` was renamed to `{new_name}`."
if "privacy" in changes:
new_visibility = payload["team"]["privacy"]
return f"Team visibility changed to `{new_visibility}`"
missing_keys = "/".join(sorted(list(changes.keys())))
helper.log_unsupported(f"team/edited (changes: {missing_keys})")
# Do our best to give useful info to the customer--at least
# if they know something changed, they can go to GitHub for
# more details. And if it's just spam, you can control that
# from GitHub.
return f"Team has changes to `{missing_keys}` data."
def get_release_body(helper: Helper) -> str:
payload = helper.payload
data = {
"user_name": get_sender_name(payload),
"action": payload["action"],
"tagname": payload["release"]["tag_name"],
# Not every GitHub release has a "name" set; if not there, use the tag name.
"release_name": payload["release"]["name"] or payload["release"]["tag_name"],
"url": payload["release"]["html_url"],
}
return get_release_event_message(**data)
2016-10-25 14:50:42 +02:00
def get_page_build_body(helper: Helper) -> str:
payload = helper.payload
build = payload["build"]
status = build["status"]
actions = {
"null": "has yet to be built",
"building": "is being built",
"errored": "has failed{}",
"built": "has finished building",
}
action = actions.get(status, f"is {status}")
action.format(
CONTENT_MESSAGE_TEMPLATE.format(message=build["error"]["message"]),
)
return "GitHub Pages build, triggered by {}, {}.".format(
payload["build"]["pusher"]["login"],
action,
2016-10-25 14:50:42 +02:00
)
def get_status_body(helper: Helper) -> str:
payload = helper.payload
if payload["target_url"]:
status = "[{}]({})".format(
payload["state"],
payload["target_url"],
2016-10-25 14:50:42 +02:00
)
else:
status = payload["state"]
return "[{}]({}) changed its status to {}.".format(
payload["sha"][:7], # TODO
payload["commit"]["html_url"],
status,
2016-10-25 14:50:42 +02:00
)
def get_locked_or_unlocked_pull_request_body(helper: Helper) -> str:
payload = helper.payload
action = payload["action"]
message = "{sender} has locked [PR #{pr_number}]({pr_url}) as {reason} and limited conversation to collaborators."
if action == "unlocked":
message = "{sender} has unlocked [PR #{pr_number}]({pr_url})."
return message.format(
sender=get_sender_name(payload),
pr_number=payload["pull_request"]["number"],
pr_url=payload["pull_request"]["html_url"],
reason=payload["pull_request"]["active_lock_reason"],
)
def get_pull_request_auto_merge_body(helper: Helper) -> str:
payload = helper.payload
action = payload["action"]
message = "{sender} has enabled auto merge for [PR #{pr_number}]({pr_url})."
if action == "auto_merge_disabled":
message = "{sender} has disabled auto merge for [PR #{pr_number}]({pr_url})."
return message.format(
sender=get_sender_name(payload),
pr_number=payload["pull_request"]["number"],
pr_url=payload["pull_request"]["html_url"],
)
def get_pull_request_ready_for_review_body(helper: Helper) -> str:
payload = helper.payload
message = "**{sender}** has marked [PR #{pr_number}]({pr_url}) as ready for review."
return message.format(
sender=get_sender_name(payload),
pr_number=payload["pull_request"]["number"],
pr_url=payload["pull_request"]["html_url"],
)
def get_pull_request_review_body(helper: Helper) -> str:
payload = helper.payload
include_title = helper.include_title
title = "for #{} {}".format(
payload["pull_request"]["number"],
payload["pull_request"]["title"],
)
2016-10-25 14:50:42 +02:00
return get_pull_request_event_message(
get_sender_name(payload),
"submitted",
payload["review"]["html_url"],
type="PR review",
title=title if include_title else None,
2016-10-25 14:50:42 +02:00
)
def get_pull_request_review_comment_body(helper: Helper) -> str:
payload = helper.payload
include_title = helper.include_title
action = payload["action"]
2016-10-25 14:50:42 +02:00
message = None
if action == "created":
message = payload["comment"]["body"]
2016-10-25 14:50:42 +02:00
title = "on #{} {}".format(
payload["pull_request"]["number"],
payload["pull_request"]["title"],
)
2016-10-25 14:50:42 +02:00
return get_pull_request_event_message(
get_sender_name(payload),
action,
payload["comment"]["html_url"],
2016-10-25 14:50:42 +02:00
message=message,
type="PR review comment",
title=title if include_title else None,
2016-10-25 14:50:42 +02:00
)
def get_pull_request_review_requested_body(helper: Helper) -> str:
payload = helper.payload
include_title = helper.include_title
requested_reviewer = [payload["requested_reviewer"]] if "requested_reviewer" in payload else []
requested_team = [payload["requested_team"]] if "requested_team" in payload else []
sender = get_sender_name(payload)
pr_number = payload["pull_request"]["number"]
pr_url = payload["pull_request"]["html_url"]
message = "**{sender}** requested {reviewers} for a review on [PR #{pr_number}]({pr_url})."
message_with_title = (
"**{sender}** requested {reviewers} for a review on [PR #{pr_number} {title}]({pr_url})."
)
body = message_with_title if include_title else message
all_reviewers = []
for reviewer in requested_reviewer:
all_reviewers.append("[{login}]({html_url})".format(**reviewer))
for team_reviewer in requested_team:
all_reviewers.append("[{name}]({html_url})".format(**team_reviewer))
reviewers = ""
reviewers = all_reviewers[0]
return body.format(
sender=sender,
reviewers=reviewers,
pr_number=pr_number,
pr_url=pr_url,
title=payload["pull_request"]["title"] if include_title else None,
)
def get_check_run_body(helper: Helper) -> str:
payload = helper.payload
template = """
Check [{name}]({html_url}) {status} ({conclusion}). ([{short_hash}]({commit_url}))
""".strip()
kwargs = {
"name": payload["check_run"]["name"],
"html_url": payload["check_run"]["html_url"],
"status": payload["check_run"]["status"],
"short_hash": payload["check_run"]["head_sha"][:7],
"commit_url": "{}/commit/{}".format(
payload["repository"]["html_url"],
payload["check_run"]["head_sha"],
),
"conclusion": payload["check_run"]["conclusion"],
}
return template.format(**kwargs)
def get_star_body(helper: Helper) -> str:
payload = helper.payload
template = "{user} {action} the repository [{repo}]({url})."
return template.format(
user=payload["sender"]["login"],
action="starred" if payload["action"] == "created" else "unstarred",
repo=get_repository_full_name(payload),
url=payload["repository"]["html_url"],
)
def get_ping_body(helper: Helper) -> str:
payload = helper.payload
return get_setup_webhook_message("GitHub", get_sender_name(payload))
def get_repository_name(payload: Dict[str, Any]) -> str:
return payload["repository"]["name"]
2016-10-25 14:50:42 +02:00
def get_repository_full_name(payload: Dict[str, Any]) -> str:
return payload["repository"]["full_name"]
def get_organization_name(payload: Dict[str, Any]) -> str:
return payload["organization"]["login"]
def get_sender_name(payload: Dict[str, Any]) -> str:
return payload["sender"]["login"]
2016-10-25 14:50:42 +02:00
def get_branch_name_from_ref(ref_string: str) -> str:
return re.sub(r"^refs/heads/", "", ref_string)
2016-10-25 14:50:42 +02:00
def get_tag_name_from_ref(ref_string: str) -> str:
return re.sub(r"^refs/tags/", "", ref_string)
2016-10-25 14:50:42 +02:00
def is_commit_push_event(payload: Dict[str, Any]) -> bool:
return bool(re.match(r"^refs/heads/", payload["ref"]))
2016-10-25 14:50:42 +02:00
def get_subject_based_on_type(payload: Dict[str, Any], event: str) -> str:
if "pull_request" in event:
return TOPIC_WITH_PR_OR_ISSUE_INFO_TEMPLATE.format(
2016-10-25 14:50:42 +02:00
repo=get_repository_name(payload),
type="PR",
id=payload["pull_request"]["number"],
title=payload["pull_request"]["title"],
2016-10-25 14:50:42 +02:00
)
elif event.startswith("issue"):
return TOPIC_WITH_PR_OR_ISSUE_INFO_TEMPLATE.format(
2016-10-25 14:50:42 +02:00
repo=get_repository_name(payload),
type="issue",
id=payload["issue"]["number"],
title=payload["issue"]["title"],
2016-10-25 14:50:42 +02:00
)
elif event.startswith("deployment"):
return "{} / Deployment on {}".format(
2016-10-25 14:50:42 +02:00
get_repository_name(payload),
payload["deployment"]["environment"],
2016-10-25 14:50:42 +02:00
)
elif event == "membership":
return "{} organization".format(payload["organization"]["login"])
elif event == "team":
return "team {}".format(payload["team"]["name"])
elif event == "push_commits":
return TOPIC_WITH_BRANCH_TEMPLATE.format(
2016-10-25 14:50:42 +02:00
repo=get_repository_name(payload),
branch=get_branch_name_from_ref(payload["ref"]),
2016-10-25 14:50:42 +02:00
)
elif event == "gollum":
return TOPIC_WITH_BRANCH_TEMPLATE.format(
2016-10-25 14:50:42 +02:00
repo=get_repository_name(payload),
branch="wiki pages",
2016-10-25 14:50:42 +02:00
)
elif event == "ping":
if payload.get("repository") is None:
return get_organization_name(payload)
elif event == "check_run":
return f"{get_repository_name(payload)} / checks"
2016-10-25 14:50:42 +02:00
return get_repository_name(payload)
EVENT_FUNCTION_MAPPER: Dict[str, Callable[[Helper], str]] = {
"commit_comment": get_commit_comment_body,
"closed_pull_request": get_closed_pull_request_body,
"create": partial(get_create_or_delete_body, action="created"),
"check_run": get_check_run_body,
"delete": partial(get_create_or_delete_body, action="deleted"),
"deployment": get_deployment_body,
"deployment_status": get_change_deployment_status_body,
"fork": get_fork_body,
"gollum": get_wiki_pages_body,
"issue_comment": get_issue_comment_body,
"issues": get_issue_body,
"member": get_member_body,
"membership": get_membership_body,
"opened_or_update_pull_request": get_opened_or_update_pull_request_body,
"assigned_or_unassigned_pull_request": get_assigned_or_unassigned_pull_request_body,
"page_build": get_page_build_body,
"ping": get_ping_body,
"public": get_public_body,
"pull_request_ready_for_review": get_pull_request_ready_for_review_body,
"pull_request_review": get_pull_request_review_body,
"pull_request_review_comment": get_pull_request_review_comment_body,
"pull_request_review_requested": get_pull_request_review_requested_body,
"pull_request_auto_merge": get_pull_request_auto_merge_body,
"locked_or_unlocked_pull_request": get_locked_or_unlocked_pull_request_body,
"push_commits": get_push_commits_body,
"push_tags": get_push_tags_body,
"release": get_release_body,
"repository": get_repository_body,
"star": get_star_body,
"status": get_status_body,
"team": get_team_body,
"team_add": get_add_team_body,
"watch": get_watch_body,
2016-10-25 14:50:42 +02:00
}
IGNORED_EVENTS = [
"check_suite",
"label",
"meta",
"milestone",
"organization",
"project_card",
"repository_vulnerability_alert",
]
IGNORED_PULL_REQUEST_ACTIONS = [
"approved",
"converted_to_draft",
"labeled",
"review_request_removed",
"unlabeled",
]
IGNORED_TEAM_ACTIONS = [
# These are actions that are well documented by github
# (https://docs.github.com/en/developers/webhooks-and-events/webhook-events-and-payloads)
# but we ignore them for now, possibly just due to laziness.
# One curious example here is team/added_to_repository, which is
# possibly the same as team_add.
"added_to_repository",
"created",
"deleted",
"removed_from_repository",
]
ALL_EVENT_TYPES = list(EVENT_FUNCTION_MAPPER.keys())
@webhook_view("GitHub", notify_bot_owner_on_invalid_json=True, all_event_types=ALL_EVENT_TYPES)
2016-10-25 14:50:42 +02:00
@has_request_variables
def api_github_webhook(
request: HttpRequest,
user_profile: UserProfile,
payload: Dict[str, Any] = REQ(argument_type="body"),
branches: Optional[str] = REQ(default=None),
user_specified_topic: Optional[str] = REQ("topic", default=None),
) -> HttpResponse:
"""
GitHub sends the event as an HTTP header. We have our
own Zulip-specific concept of an event that often maps
directly to the X_GITHUB_EVENT header's event, but we sometimes
refine it based on the payload.
"""
header_event = validate_extract_webhook_http_header(request, "X_GITHUB_EVENT", "GitHub")
if header_event is None:
raise UnsupportedWebhookEventType("no header provided")
event = get_zulip_event_name(header_event, payload, branches)
if event is None:
# This is nothing to worry about--get_event() returns None
# for events that are valid but not yet handled by us.
# See IGNORED_EVENTS, for example.
return json_success()
subject = get_subject_based_on_type(payload, event)
body_function = EVENT_FUNCTION_MAPPER[event]
helper = Helper(
payload=payload,
include_title=user_specified_topic is not None,
)
body = body_function(helper)
check_send_webhook_message(request, user_profile, subject, body, event)
2016-10-25 14:50:42 +02:00
return json_success()
def get_zulip_event_name(
header_event: str,
payload: Dict[str, Any],
branches: Optional[str],
) -> Optional[str]:
"""
Usually, we return an event name that is a key in EVENT_FUNCTION_MAPPER.
We return None for an event that we know we don't want to handle.
"""
if header_event == "pull_request":
action = payload["action"]
if action in ("opened", "synchronize", "reopened", "edited"):
return "opened_or_update_pull_request"
if action in ("assigned", "unassigned"):
return "assigned_or_unassigned_pull_request"
if action == "closed":
return "closed_pull_request"
if action == "review_requested":
return "pull_request_review_requested"
if action == "ready_for_review":
return "pull_request_ready_for_review"
if action in ("locked", "unlocked"):
return "locked_or_unlocked_pull_request"
if action in ("auto_merge_enabled", "auto_merge_disabled"):
return "pull_request_auto_merge"
if action in IGNORED_PULL_REQUEST_ACTIONS:
return None
elif header_event == "push":
2016-10-25 14:50:42 +02:00
if is_commit_push_event(payload):
if branches is not None:
branch = get_branch_name_from_ref(payload["ref"])
if branches.find(branch) == -1:
return None
2016-10-25 14:50:42 +02:00
return "push_commits"
else:
return "push_tags"
elif header_event == "check_run":
if payload["check_run"]["status"] != "completed":
return None
return header_event
elif header_event == "team":
action = payload["action"]
if action == "edited":
return "team"
if action in IGNORED_TEAM_ACTIONS:
# no need to spam our logs, we just haven't implemented it yet
return None
else:
# this means GH has actually added new actions since September 2020,
# so it's a bit more cause for alarm
raise UnsupportedWebhookEventType(f"unsupported team action {action}")
elif header_event in list(EVENT_FUNCTION_MAPPER.keys()):
return header_event
elif header_event in IGNORED_EVENTS:
return None
complete_event = "{}:{}".format(header_event, payload.get("action", "???")) # nocoverage
raise UnsupportedWebhookEventType(complete_event)