# Webhooks for external integrations. from django.http import HttpRequest, HttpResponse from django.utils.translation import gettext as _ from zerver.decorator import webhook_view from zerver.lib.exceptions import JsonableError 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_list, check_none_or, check_string, check_string_fixed_length, check_union, ) from zerver.lib.webhooks.common import check_send_webhook_message, unix_milliseconds_to_timestamp from zerver.models import UserProfile # Newrelic planned to upgrade Alert Notification Channels to Workflows and Destinations # https://discuss.newrelic.com/t/plan-to-upgrade-alert-notification-channels-to-workflows-and-destinations/188205 # This view will handle both old and new format but will keep it easy to delete the old code # once it is EOLed by the end of June, 2023 # Once old is EOLed, delete the OPEN_TEMPLATE OPEN_TEMPLATE = """ [Incident]({incident_url}) **opened** for condition: **{condition_name}** at ``` quote {details} ``` """.strip() ACTIVE_TEMPLATE = """ [Incident]({incident_url}) **active** for condition: **{condition_name}** at ``` quote {details} ``` """.strip() DEFAULT_TEMPLATE = ( """[Incident]({incident_url}) **{status}** {owner}for condition: **{condition_name}**""".strip() ) TOPIC_TEMPLATE = """{policy_name} ({incident_id})""".strip() # Once old is EOLed, delete old and keep new OLD_EVENT_TYPES = ["closed", "acknowledged", "open"] NEW_EVENT_TYPES = ["created", "activated", "acknowledged", "closed"] ALL_EVENT_TYPES = list(set(OLD_EVENT_TYPES).union(set(NEW_EVENT_TYPES))) @webhook_view("NewRelic", all_event_types=ALL_EVENT_TYPES) @typed_endpoint def api_newrelic_webhook( request: HttpRequest, user_profile: UserProfile, *, payload: JsonBodyPayload[WildValue], ) -> HttpResponse: # Handle old format # Once old is EOLed, delete if block and keep else block if not payload.get("id").tame(check_none_or(check_string_fixed_length(36))): info = { "condition_name": payload.get("condition_name", "Unknown condition").tame(check_string), "details": payload.get("details", "No details.").tame(check_string), "incident_url": payload.get("incident_url", "https://alerts.newrelic.com").tame( check_string ), "incident_acknowledge_url": payload.get( "incident_acknowledge_url", "https://alerts.newrelic.com" ).tame(check_string), "status": payload.get("current_state", "None").tame(check_string), "iso_timestamp": "", "owner": payload.get("owner", "").tame(check_string), } unix_time = payload.get("timestamp").tame( check_none_or(check_union([check_string, check_int])) ) if unix_time is None: raise JsonableError(_("The newrelic webhook requires timestamp in milliseconds")) info["iso_timestamp"] = str(unix_milliseconds_to_timestamp(unix_time, "newrelic")) # Add formatting to the owner field if owner is present if info["owner"] != "": info["owner"] = "by **{}** ".format(info["owner"]) # These are the three promised current_state values if info["status"].lower() == "open": content = OPEN_TEMPLATE.format(**info) elif info["status"].lower() == "acknowledged": content = DEFAULT_TEMPLATE.format(**info) elif info["status"].lower() == "closed": content = DEFAULT_TEMPLATE.format(**info) else: raise JsonableError( _("The newrelic webhook requires current_state be in [open|acknowledged|closed]") ) topic_info = { "policy_name": payload.get("policy_name", "Unknown Policy").tame(check_string), "incident_id": payload.get("incident_id", "Unknown ID").tame( check_union([check_string, check_int]) ), } topic = TOPIC_TEMPLATE.format(**topic_info) check_send_webhook_message(request, user_profile, topic, content, info["status"]) return json_success(request) # Handle new format else: info = { "condition_name": payload.get("condition_name", "Unknown condition").tame(check_string), "details": payload.get("details", "No details.").tame(check_string), "incident_url": payload.get("issueUrl", "https://alerts.newrelic.com").tame( check_string ), "incident_acknowledge_url": payload.get( "incident_acknowledge_url", "https://alerts.newrelic.com" ).tame(check_string), "status": payload.get("state", "None").tame(check_string), "iso_timestamp": "", "owner": payload.get("owner", "").tame(check_string), } unix_time = payload.get("createdAt").tame( check_none_or(check_union([check_string, check_int])) ) if unix_time is None: raise JsonableError(_("The newrelic webhook requires timestamp in milliseconds")) info["iso_timestamp"] = str(unix_milliseconds_to_timestamp(unix_time, "newrelic")) # Add formatting to the owner field if owner is present if info["owner"] != "": info["owner"] = "by **{}** ".format(info["owner"]) # These are the three promised state values if info["status"].lower() == "activated": content = ACTIVE_TEMPLATE.format(**info) elif info["status"].lower() == "acknowledged": content = DEFAULT_TEMPLATE.format(**info) elif info["status"].lower() == "closed": content = DEFAULT_TEMPLATE.format(**info) elif info["status"].lower() == "created": content = DEFAULT_TEMPLATE.format(**info) else: raise JsonableError( _( "The newrelic webhook requires state be in [created|activated|acknowledged|closed]" ) ) policy_names_list = payload.get("alertPolicyNames", []).tame(check_list(check_string)) if policy_names_list: policy_names_str = ",".join(policy_names_list) else: policy_names_str = "Unknown Policy" topic_info = { "policy_name": policy_names_str, "incident_id": payload.get("id", "Unknown ID").tame(check_string), } topic = TOPIC_TEMPLATE.format(**topic_info) check_send_webhook_message(request, user_profile, topic, content, info["status"]) return json_success(request)