from datetime import datetime from typing import Any, Dict, List, Optional, Callable, Tuple, Text from six.moves import zip from django.utils.timezone import utc as timezone_utc from django.utils.translation import ugettext as _ from django.http import HttpRequest, HttpResponse from zerver.decorator import api_key_only_webhook_view, REQ, has_request_variables from zerver.lib.response import json_success, json_error from zerver.lib.actions import check_send_stream_message from zerver.models import UserProfile import ujson ALERT_CLEAR = 'clear' ALERT_VIOLATION = 'violations' SNAPSHOT = 'image_url' class LibratoWebhookParser(object): ALERT_URL_TEMPLATE = "https://metrics.librato.com/alerts#/{alert_id}" def __init__(self, payload, attachments): # type: (Dict[str, Any], List[Dict[str, Any]]) -> None self.payload = payload self.attachments = attachments def generate_alert_url(self, alert_id): # type: (int) -> Text return self.ALERT_URL_TEMPLATE.format(alert_id=alert_id) def parse_alert(self): # type: () -> Tuple[int, Text, Text, Text] alert = self.payload['alert'] alert_id = alert['id'] return alert_id, alert['name'], self.generate_alert_url(alert_id), alert['runbook_url'] def parse_condition(self, condition): # type: (Dict[str, Any]) -> Tuple[Text, Text, Text, Text] summary_function = condition['summary_function'] threshold = condition.get('threshold', '') condition_type = condition['type'] duration = condition.get('duration', '') return summary_function, threshold, condition_type, duration def parse_violation(self, violation): # type: (Dict[str, Any]) -> Tuple[Text, Text] metric_name = violation['metric'] recorded_at = datetime.fromtimestamp((violation['recorded_at']), tz=timezone_utc).strftime('%Y-%m-%d %H:%M:%S') return metric_name, recorded_at def parse_conditions(self): # type: () -> List[Dict[str, Any]] conditions = self.payload['conditions'] return conditions def parse_violations(self): # type: () -> List[Dict[str, Any]] violations = self.payload['violations']['test-source'] return violations def parse_snapshot(self, snapshot): # type: (Dict[str, Any]) -> Tuple[Text, Text, Text] author_name, image_url, title = snapshot['author_name'], snapshot['image_url'], snapshot['title'] return author_name, image_url, title class LibratoWebhookHandler(LibratoWebhookParser): def __init__(self, payload, attachments): # type: (Dict[str, Any], List[Dict[str, Any]]) -> None super(LibratoWebhookHandler, self).__init__(payload, attachments) self.payload_available_types = { ALERT_CLEAR: self.handle_alert_clear_message, ALERT_VIOLATION: self.handle_alert_violation_message } self.attachments_available_types = { SNAPSHOT: self.handle_snapshots } def find_handle_method(self): # type: () -> Callable for available_type in self.payload_available_types: if self.payload.get(available_type): return self.payload_available_types[available_type] for available_type in self.attachments_available_types: if self.attachments[0].get(available_type): return self.attachments_available_types[available_type] raise Exception("Unexcepted message type") def handle(self): # type: () -> Text return self.find_handle_method()() def generate_topic(self): # type: () -> Text if self.attachments: return "Snapshots" topic_template = "Alert {alert_name}" alert_id, alert_name, alert_url, alert_runbook_url = self.parse_alert() return topic_template.format(alert_name=alert_name) def handle_alert_clear_message(self): # type: () -> Text alert_clear_template = "Alert [alert_name]({alert_url}) has cleared at {trigger_time} UTC!" trigger_time = datetime.fromtimestamp((self.payload['trigger_time']), tz=timezone_utc).strftime('%Y-%m-%d %H:%M:%S') alert_id, alert_name, alert_url, alert_runbook_url = self.parse_alert() content = alert_clear_template.format(alert_name=alert_name, alert_url=alert_url, trigger_time=trigger_time) return content def handle_snapshots(self): # type: () -> Text content = u'' for attachment in self.attachments: content += self.handle_snapshot(attachment) return content def handle_snapshot(self, snapshot): # type: (Dict[str, Any]) -> Text snapshot_template = u"**{author_name}** sent a [snapshot]({image_url}) of [metric]({title})" author_name, image_url, title = self.parse_snapshot(snapshot) content = snapshot_template.format(author_name=author_name, image_url=image_url, title=title) return content def handle_alert_violation_message(self): # type: () -> Text alert_violation_template = u"Alert [alert_name]({alert_url}) has triggered! " alert_id, alert_name, alert_url, alert_runbook_url = self.parse_alert() content = alert_violation_template.format(alert_name=alert_name, alert_url=alert_url) if alert_runbook_url: alert_runbook_template = u"[Reaction steps]({alert_runbook_url})" content += alert_runbook_template.format(alert_runbook_url=alert_runbook_url) content += self.generate_conditions_and_violations() return content def generate_conditions_and_violations(self): # type: () -> Text conditions = self.parse_conditions() violations = self.parse_violations() content = u"" for condition, violation in zip(conditions, violations): content += self.generate_violated_metric_condition(violation, condition) return content def generate_violated_metric_condition(self, violation, condition): # type: (Dict[str, Any], Dict[str, Any]) -> Text summary_function, threshold, condition_type, duration = self.parse_condition(condition) metric_name, recorded_at = self.parse_violation(violation) metric_condition_template = u"\n>Metric `{metric_name}`, {summary_function} was {condition_type} {threshold}" content = metric_condition_template.format( metric_name=metric_name, summary_function=summary_function, condition_type=condition_type, threshold=threshold) if duration: content += u" by {duration}s".format(duration=duration) content += u", recorded at {recorded_at} UTC".format(recorded_at=recorded_at) return content @api_key_only_webhook_view('Librato') @has_request_variables def api_librato_webhook(request, user_profile, payload=REQ(converter=ujson.loads, default={}), stream=REQ(default='librato'), topic=REQ(default=None)): # type: (HttpRequest, UserProfile, Dict[str, Any], Text, Text) -> HttpResponse try: attachments = ujson.loads(request.body).get('attachments', []) except ValueError: attachments = [] if not attachments and not payload: return json_error(_("Malformed JSON input")) message_handler = LibratoWebhookHandler(payload, attachments) if not topic: topic = message_handler.generate_topic() try: content = message_handler.handle() except Exception as e: return json_error(_(str(e))) check_send_stream_message(user_profile, request.client, stream, topic, content) return json_success()