# Webhooks for external integrations. from django.db.models import Q from django.http import HttpRequest, HttpResponse from zerver.decorator import webhook_view from zerver.lib.exceptions import UnsupportedWebhookEventTypeError from zerver.lib.response import json_success from zerver.lib.typed_endpoint import JsonBodyPayload, typed_endpoint from zerver.lib.validator import WildValue, check_int, check_string from zerver.lib.webhooks.common import check_send_webhook_message from zerver.models import Realm, UserProfile IGNORED_EVENTS = [ "DOWNLOAD_CHART", "DELETE_CHART", "UPLOAD_CHART", "PULL_ARTIFACT", "DELETE_ARTIFACT", "SCANNING_FAILED", ] def guess_zulip_user_from_harbor(harbor_username: str, realm: Realm) -> UserProfile | None: try: # Try to find a matching user in Zulip # We search a user's full name, short name, # and beginning of email address user = UserProfile.objects.filter( Q(full_name__iexact=harbor_username) | Q(email__istartswith=harbor_username), is_active=True, realm=realm, ).order_by("id")[0] return user # nocoverage except IndexError: return None def image_id(payload: WildValue) -> str: image_name = payload["event_data"]["repository"]["repo_full_name"].tame(check_string) resource = payload["event_data"]["resources"][0] if "tag" in resource: return image_name + ":" + resource["tag"].tame(check_string) else: return image_name + "@" + resource["digest"].tame(check_string) def handle_push_image_event( payload: WildValue, user_profile: UserProfile, operator_username: str ) -> str: return f"{operator_username} pushed image `{image_id(payload)}`" SCANNING_COMPLETED_TEMPLATE = """ Image scan completed for `{image_id}`. Vulnerabilities by severity: {scan_results} """.strip() def handle_scanning_completed_event( payload: WildValue, user_profile: UserProfile, operator_username: str ) -> str: scan_results = "" scan_overview = payload["event_data"]["resources"][0]["scan_overview"] if "application/vnd.security.vulnerability.report; version=1.1" not in scan_overview: raise UnsupportedWebhookEventTypeError(str(list(scan_overview.keys()))) scan_summaries = scan_overview["application/vnd.security.vulnerability.report; version=1.1"][ "summary" ]["summary"] if len(scan_summaries) > 0: for severity, count in scan_summaries.items(): scan_results += f"* {severity}: **{count.tame(check_int)}**\n" else: scan_results += "None\n" return SCANNING_COMPLETED_TEMPLATE.format( image_id=image_id(payload), scan_results=scan_results, ) EVENT_FUNCTION_MAPPER = { "PUSH_ARTIFACT": handle_push_image_event, "SCANNING_COMPLETED": handle_scanning_completed_event, } ALL_EVENT_TYPES = list(EVENT_FUNCTION_MAPPER.keys()) @webhook_view("Harbor", all_event_types=ALL_EVENT_TYPES) @typed_endpoint def api_harbor_webhook( request: HttpRequest, user_profile: UserProfile, *, payload: JsonBodyPayload[WildValue], ) -> HttpResponse: operator_username = "**{}**".format(payload["operator"].tame(check_string)) if operator_username != "auto": operator_profile = guess_zulip_user_from_harbor(operator_username, user_profile.realm) if operator_profile: operator_username = f"@**{operator_profile.full_name}**" # nocoverage event = payload["type"].tame(check_string) topic_name = payload["event_data"]["repository"]["repo_full_name"].tame(check_string) if event in IGNORED_EVENTS: return json_success(request) content_func = EVENT_FUNCTION_MAPPER.get(event) if content_func is None: raise UnsupportedWebhookEventTypeError(event) content: str = content_func(payload, user_profile, operator_username) check_send_webhook_message( request, user_profile, topic_name, content, event, unquote_url_parameters=True ) return json_success(request)