0
0
Fork 0
mirror of https://github.com/alerta/alerta.git synced 2025-01-27 02:28:49 +00:00
alerta_alerta/alerta/webhooks/pagerduty.py
Nick Satterly 17ef9f0ceb
Support webhooks with a subpath (#1150)
* Support webhooks with a subpath

* Add tests for webhooks with subpath
2020-02-14 23:00:59 +01:00

99 lines
3.7 KiB
Python

from typing import Any, Dict, Tuple
from flask import g, jsonify
from alerta.exceptions import ApiError
from alerta.models.alert import Alert
from . import WebhookBase
JSON = Dict[str, Any]
def parse_pagerduty(message: JSON) -> Tuple[str, str, str]:
try:
incident_key = message['data']['incident']['incident_key']
incident_number = message['data']['incident']['incident_number']
html_url = message['data']['incident']['html_url']
incident_url = '<a href="{}">#{}</a>'.format(html_url, incident_number)
if message['type'] == 'incident.trigger':
status = 'open'
user = message['data']['incident']['assigned_to_user']['name']
text = 'Incident {} assigned to {}'.format(incident_url, user)
elif message['type'] == 'incident.acknowledge':
status = 'ack'
user = message['data']['incident']['assigned_to_user']['name']
text = 'Incident {} acknowledged by {}'.format(incident_url, user)
elif message['type'] == 'incident.unacknowledge':
status = 'open'
text = 'Incident %s unacknowledged due to timeout' % incident_url
elif message['type'] == 'incident.resolve':
status = 'closed'
if message['data']['incident']['resolved_by_user']:
user = message['data']['incident']['resolved_by_user']['name']
else:
user = 'n/a'
text = 'Incident {} resolved by {}'.format(incident_url, user)
elif message['type'] == 'incident.assign':
status = 'assign'
user = message['data']['incident']['assigned_to_user']['name']
text = 'Incident {} manually assigned to {}'.format(incident_url, user)
elif message['type'] == 'incident.escalate':
status = 'open'
user = message['data']['incident']['assigned_to_user']['name']
text = 'Incident {} escalated to {}'.format(incident_url, user)
elif message['type'] == 'incident.delegate':
status = 'open'
user = message['data']['incident']['assigned_to_user']['name']
text = 'Incident {} reassigned due to escalation to {}'.format(incident_url, user)
else:
status = 'unknown'
text = message['type']
except Exception as e:
raise ValueError(e)
return incident_key, status, text
class PagerDutyWebhook(WebhookBase):
"""
PagerDuty incident webhook
See https://v2.developer.pagerduty.com/docs/webhooks-v2-overview
"""
def incoming(self, path, query_string, payload):
updated = False
if payload and 'messages' in payload:
for message in payload['messages']:
try:
incident_key, status, text = parse_pagerduty(message)
except ValueError as e:
raise ApiError(str(e), 400)
if not incident_key:
raise ApiError('no incident key in PagerDuty data payload', 400)
customers = g.get('customers', None)
try:
alert = Alert.find_by_id(id=incident_key, customers=customers)
except Exception as e:
raise ApiError(str(e), 500)
if not alert:
raise ApiError('not found', 404)
try:
updated = alert.set_status(status, text)
except Exception as e:
raise ApiError(str(e), 500)
if updated:
return jsonify(status='ok')
else:
raise ApiError('update PagerDuty incident status failed', 500)
else:
raise ApiError('no messages in PagerDuty data payload', 400)