2016-03-13 15:28:44 +01:00
|
|
|
# Webhooks for external integrations.
|
|
|
|
from __future__ import absolute_import
|
2016-06-06 00:33:59 +02:00
|
|
|
|
2016-03-13 15:28:44 +01:00
|
|
|
from zerver.lib.actions import check_send_message
|
|
|
|
from zerver.lib.response import json_success
|
|
|
|
from zerver.decorator import REQ, has_request_variables, api_key_only_webhook_view
|
2016-06-06 00:33:59 +02:00
|
|
|
from zerver.models import Client, UserProfile
|
|
|
|
|
|
|
|
from django.http import HttpRequest, HttpResponse
|
2016-03-13 15:28:44 +01:00
|
|
|
|
|
|
|
import pprint
|
|
|
|
import ujson
|
2016-06-06 00:33:59 +02:00
|
|
|
import six
|
|
|
|
from typing import Dict, Any, Iterable, Optional
|
2016-03-13 15:28:44 +01:00
|
|
|
|
|
|
|
|
|
|
|
PAGER_DUTY_EVENT_NAMES = {
|
|
|
|
'incident.trigger': 'triggered',
|
|
|
|
'incident.acknowledge': 'acknowledged',
|
|
|
|
'incident.unacknowledge': 'unacknowledged',
|
|
|
|
'incident.resolve': 'resolved',
|
|
|
|
'incident.assign': 'assigned',
|
|
|
|
'incident.escalate': 'escalated',
|
|
|
|
'incident.delegate': 'delineated',
|
|
|
|
}
|
|
|
|
|
|
|
|
def build_pagerduty_formatdict(message):
|
2016-06-06 00:33:59 +02:00
|
|
|
# type: (Dict[str, Any]) -> Dict[str, Any]
|
2016-03-13 15:28:44 +01:00
|
|
|
# Normalize the message dict, after this all keys will exist. I would
|
|
|
|
# rather some strange looking messages than dropping pages.
|
|
|
|
|
2016-04-07 08:41:45 +02:00
|
|
|
format_dict = {} # type: Dict[str, Any]
|
2016-03-13 15:28:44 +01:00
|
|
|
format_dict['action'] = PAGER_DUTY_EVENT_NAMES[message['type']]
|
|
|
|
|
|
|
|
format_dict['incident_id'] = message['data']['incident']['id']
|
|
|
|
format_dict['incident_num'] = message['data']['incident']['incident_number']
|
|
|
|
format_dict['incident_url'] = message['data']['incident']['html_url']
|
|
|
|
|
|
|
|
format_dict['service_name'] = message['data']['incident']['service']['name']
|
|
|
|
format_dict['service_url'] = message['data']['incident']['service']['html_url']
|
|
|
|
|
|
|
|
# This key can be missing on null
|
|
|
|
if message['data']['incident'].get('assigned_to_user', None):
|
|
|
|
format_dict['assigned_to_email'] = message['data']['incident']['assigned_to_user']['email']
|
|
|
|
format_dict['assigned_to_username'] = message['data']['incident']['assigned_to_user']['email'].split('@')[0]
|
|
|
|
format_dict['assigned_to_url'] = message['data']['incident']['assigned_to_user']['html_url']
|
|
|
|
else:
|
|
|
|
format_dict['assigned_to_email'] = 'nobody'
|
|
|
|
format_dict['assigned_to_username'] = 'nobody'
|
|
|
|
format_dict['assigned_to_url'] = ''
|
|
|
|
|
|
|
|
# This key can be missing on null
|
|
|
|
if message['data']['incident'].get('resolved_by_user', None):
|
|
|
|
format_dict['resolved_by_email'] = message['data']['incident']['resolved_by_user']['email']
|
|
|
|
format_dict['resolved_by_username'] = message['data']['incident']['resolved_by_user']['email'].split('@')[0]
|
|
|
|
format_dict['resolved_by_url'] = message['data']['incident']['resolved_by_user']['html_url']
|
|
|
|
else:
|
|
|
|
format_dict['resolved_by_email'] = 'nobody'
|
|
|
|
format_dict['resolved_by_username'] = 'nobody'
|
|
|
|
format_dict['resolved_by_url'] = ''
|
|
|
|
|
|
|
|
trigger_message = []
|
|
|
|
trigger_subject = message['data']['incident']['trigger_summary_data'].get('subject', '')
|
|
|
|
if trigger_subject:
|
|
|
|
trigger_message.append(trigger_subject)
|
|
|
|
trigger_description = message['data']['incident']['trigger_summary_data'].get('description', '')
|
|
|
|
if trigger_description:
|
|
|
|
trigger_message.append(trigger_description)
|
|
|
|
format_dict['trigger_message'] = u'\n'.join(trigger_message)
|
|
|
|
return format_dict
|
|
|
|
|
|
|
|
|
2016-05-12 22:49:36 +02:00
|
|
|
def send_raw_pagerduty_json(user_profile, client, stream, message, topic):
|
2016-06-06 00:33:59 +02:00
|
|
|
# type: (UserProfile, Client, six.text_type, Dict[str, Any], six.text_type) -> None
|
2016-03-13 15:28:44 +01:00
|
|
|
subject = topic or 'pagerduty'
|
|
|
|
body = (
|
|
|
|
u'Unknown pagerduty message\n'
|
|
|
|
u'``` py\n'
|
|
|
|
u'%s\n'
|
|
|
|
u'```') % (pprint.pformat(message),)
|
2016-05-12 22:49:36 +02:00
|
|
|
check_send_message(user_profile, client, 'stream',
|
2016-03-13 15:28:44 +01:00
|
|
|
[stream], subject, body)
|
|
|
|
|
|
|
|
|
2016-05-12 22:49:36 +02:00
|
|
|
def send_formated_pagerduty(user_profile, client, stream, message_type, format_dict, topic):
|
2016-06-06 00:33:59 +02:00
|
|
|
# type: (UserProfile, Client, six.text_type, six.text_type, Dict[str, Any], six.text_type) -> None
|
2016-03-13 15:28:44 +01:00
|
|
|
if message_type in ('incident.trigger', 'incident.unacknowledge'):
|
|
|
|
template = (u':imp: Incident '
|
|
|
|
u'[{incident_num}]({incident_url}) {action} by '
|
|
|
|
u'[{service_name}]({service_url}) and assigned to '
|
|
|
|
u'[{assigned_to_username}@]({assigned_to_url})\n\n>{trigger_message}')
|
|
|
|
|
|
|
|
elif message_type == 'incident.resolve' and format_dict['resolved_by_url']:
|
|
|
|
template = (u':grinning: Incident '
|
|
|
|
u'[{incident_num}]({incident_url}) resolved by '
|
|
|
|
u'[{resolved_by_username}@]({resolved_by_url})\n\n>{trigger_message}')
|
|
|
|
elif message_type == 'incident.resolve' and not format_dict['resolved_by_url']:
|
|
|
|
template = (u':grinning: Incident '
|
|
|
|
u'[{incident_num}]({incident_url}) resolved\n\n>{trigger_message}')
|
|
|
|
else:
|
|
|
|
template = (u':no_good: Incident [{incident_num}]({incident_url}) '
|
|
|
|
u'{action} by [{assigned_to_username}@]({assigned_to_url})\n\n>{trigger_message}')
|
|
|
|
|
|
|
|
subject = topic or u'incident {incident_num}'.format(**format_dict)
|
|
|
|
body = template.format(**format_dict)
|
|
|
|
|
2016-05-12 22:49:36 +02:00
|
|
|
check_send_message(user_profile, client, 'stream',
|
2016-03-13 15:28:44 +01:00
|
|
|
[stream], subject, body)
|
|
|
|
|
|
|
|
|
2016-05-12 22:49:36 +02:00
|
|
|
@api_key_only_webhook_view('PagerDuty')
|
2016-03-13 15:28:44 +01:00
|
|
|
@has_request_variables
|
2016-05-12 22:49:36 +02:00
|
|
|
def api_pagerduty_webhook(request, user_profile, client, payload=REQ(argument_type='body'),
|
2016-05-06 21:19:34 +02:00
|
|
|
stream=REQ(default='pagerduty'), topic=REQ(default=None)):
|
2016-06-06 00:33:59 +02:00
|
|
|
# type: (HttpRequest, UserProfile, Client, Dict[str, Iterable[Dict[str, Any]]], six.text_type, Optional[six.text_type]) -> HttpResponse
|
2016-03-13 15:28:44 +01:00
|
|
|
for message in payload['messages']:
|
|
|
|
message_type = message['type']
|
|
|
|
|
|
|
|
if message_type not in PAGER_DUTY_EVENT_NAMES:
|
2016-05-12 22:49:36 +02:00
|
|
|
send_raw_pagerduty_json(user_profile, client, stream, message, topic)
|
2016-03-13 15:28:44 +01:00
|
|
|
|
|
|
|
try:
|
|
|
|
format_dict = build_pagerduty_formatdict(message)
|
|
|
|
except:
|
2016-05-12 22:49:36 +02:00
|
|
|
send_raw_pagerduty_json(user_profile, client, stream, message, topic)
|
2016-03-13 15:28:44 +01:00
|
|
|
else:
|
2016-05-12 22:49:36 +02:00
|
|
|
send_formated_pagerduty(user_profile, client, stream, message_type, format_dict, topic)
|
2016-03-13 15:28:44 +01:00
|
|
|
|
|
|
|
return json_success()
|