diff --git a/README.md b/README.md index 14af73a..f7bff49 100644 --- a/README.md +++ b/README.md @@ -10,7 +10,7 @@ Has been tested with Nagios, works well for us. Any Pagerduty Notifier using the [Icinga2 config](https://github.com/deathowl/OpenDuty-Icinga2) for openduty integration #Notifications -XMPP, email, SMS, Phone(Thanks Twilio for being awesome!), and Push notifications(thanks Pushover also),and Slack are supported at the moment. +XMPP, email, SMS, Phone(Thanks Twilio for being awesome!), Push notifications(thanks Pushover, Prowl as well!)and Slack, HipChat, Rocket.chat are supported at the moment. #Current status Openduty is in Beta status, it can be considered stable at the moment, however major structural changes can appear anytime (not affecting the API, or the Notifier structure) @@ -79,7 +79,24 @@ python manage.py runserver ``` now, you can start hacking on it. -# After models you've changed your models please run: +# Running as a service with systemd +*OpenDuty can be ran as a service with the help of gunicorn and systemd* +``` +cp -r systemd/gunicorn.service.* /etc/systemd/system/ + +cp -r systemd/celery.service* /etc/systemd/system/ + +// EDIT VARIABLES IN *.service.d/main.conf TO REFLECT YOUR ENV +vi /etc/systemd/system/gunicorn.service.d/main.conf +vi /etc/systemd/system/celery.service.d/main.conf + +systemctl daemon-reload +sudo systemctl start gunicorn +sudo systemctl enable gunicorn + +``` + +# After you've changed your models please run: ``` ./manage.py schemamigration openduty --auto ./manage.py schemamigration notification --auto diff --git a/nginx.conf b/nginx.conf new file mode 100644 index 0000000..560c2b8 --- /dev/null +++ b/nginx.conf @@ -0,0 +1,15 @@ +server { + listen 80; + server_name _; + access_log /var/log/openduty/nginx/access.log; + error_log /var/log/openduty/nginx/error.log; + location / { + proxy_set_header Host $http_host; + proxy_set_header X-Real-IP $remote_addr; + proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; + proxy_set_header X-Forwarded-Proto $scheme; + proxy_pass http://localhost:8080; + } +} + + diff --git a/notification/helper.py b/notification/helper.py index 6fddbed..8c6e5ef 100644 --- a/notification/helper.py +++ b/notification/helper.py @@ -1,9 +1,10 @@ __author__ = 'deathowl' from datetime import datetime, timedelta - +from django.db import transaction from notification.tasks import send_notifications from openduty.escalation_helper import get_escalation_for_service +from openduty.models import Incident from django.utils import timezone from notification.models import ScheduledNotification from django.conf import settings @@ -28,21 +29,41 @@ def notify_user_about_incident(incident, user, delay=None, preparedmsg = None): @staticmethod def generate_notifications_for_incident(incident): now = timezone.make_aware(datetime.now(), timezone.get_current_timezone()) - duty_officers = get_escalation_for_service(incident.service_key) - + duty_officers = [] + # if incident has been escalated, notify according to the escalated service's escalation rule + if hasattr(incident, 'service_to_escalate_to') and incident.service_to_escalate_to is not None: + print "escalation rule in place to " + incident.service_to_escalate_to.name + duty_officers = get_escalation_for_service(incident.service_to_escalate_to) + with transaction.atomic(): + incident.description = "[escalated] " + incident.description + incident.save() + else: + duty_officers = get_escalation_for_service(incident.service_key) current_time = now notifications = [] + group_index = {} + user_method_index = {} for officer_index, duty_officer in enumerate(duty_officers): - escalation_time = incident.service_key.escalate_after * (officer_index + 1) + if incident.event_type == Incident.RESOLVE and not duty_officer.profile.send_resolve_enabled: + print "Skipping notification for %s because type is RESOLVE and user %s has send_resolve_enabled OFF" % (incident.incident_key, duty_officer.username) + continue + index = 0 + if hasattr(duty_officer ,'came_from_group' ): + if not duty_officer.came_from_group in group_index: + group_index[duty_officer.came_from_group] = officer_index + index = group_index[duty_officer.came_from_group] + else: + index = officer_index + escalation_time = incident.service_key.escalate_after * (index + 1) escalate_at = current_time + timedelta(minutes=escalation_time) + user_method_index[duty_officer.username] = 0 methods = duty_officer.notification_methods.order_by('position').all() - method_index = 0 for method in methods: - notification_time = incident.service_key.retry * method_index + incident.service_key.escalate_after * officer_index + notification_time = incident.service_key.retry * user_method_index[duty_officer.username] + incident.service_key.escalate_after * index notify_at = current_time + timedelta(minutes=notification_time) if notify_at < escalate_at: notification = ScheduledNotification() @@ -51,14 +72,14 @@ def generate_notifications_for_incident(incident): notification.notifier = method.method notification.send_at = notify_at uri = settings.BASE_URL + "/incidents/details/" + str(incident.id) - notification.message = "A Service is experiencing a problem: " + incident.incident_key + " " + incident.description + ". Handle at: " + uri + " Details: " + incident.details + notification.message = incident.description + ". Handle at: " + uri + " Details: " + incident.details notifications.append(notification) - print "Notify %s at %s with method: %s" % (duty_officer.username, notify_at, notification.notifier) + print "[%s] Notify %s about %s at %s with method: %s" % (notification.incident.event_type, duty_officer.username, notification.incident.incident_key, notify_at, notification.notifier) else: break - method_index += 1 + user_method_index[duty_officer.username] += 1 # todo: error handling @@ -90,7 +111,8 @@ def generate_notifications_for_user(incident, user, delay=None, preparedmsg = No else: notification.message = preparedmsg notifications.append(notification) - print "Notify %s at %s with method: %s" % (user.username, notify_at, notification.notifier) + if notification.incident: + print "[%s] Notify %s at %s with method: %s" % (notification.incident.event_type, user.username, notify_at, notification.notifier) method_index += 1 # todo: error handling diff --git a/notification/models.py b/notification/models.py index a395970..8168255 100644 --- a/notification/models.py +++ b/notification/models.py @@ -21,9 +21,10 @@ class UserNotificationMethod(models.Model): METHOD_SLACK = 'slack' METHOD_PROWL = 'prowl' METHOD_ROCKET = 'rocket' + METHOD_HIPCHAT = 'hipchat' - methods = [METHOD_XMPP, METHOD_PUSHOVER, METHOD_EMAIL, METHOD_TWILIO_SMS, METHOD_TWILIO_CALL, METHOD_SLACK, METHOD_PROWL, METHOD_ROCKET] + methods = [METHOD_XMPP, METHOD_PUSHOVER, METHOD_EMAIL, METHOD_TWILIO_SMS, METHOD_TWILIO_CALL, METHOD_SLACK, METHOD_PROWL, METHOD_ROCKET, METHOD_HIPCHAT] user = models.ForeignKey(User, related_name='notification_methods') position = models.IntegerField() diff --git a/notification/notifier/email.py b/notification/notifier/email.py index a0a7c66..0a5b33b 100644 --- a/notification/notifier/email.py +++ b/notification/notifier/email.py @@ -10,8 +10,10 @@ def __init__(self, config): self.__config = config def notify(self, notification): - gmail_user = self.__config['user'] - gmail_pwd = self.__config['password'] + host = self.__config['host'] + port = self.__config['port'] + user = self.__config['user'] + password = self.__config['password'] truncate_length = int(self.__config.get('max_subject_length', 100)) FROM = self.__config['user'] TO = [notification.user_to_notify.email] @@ -25,10 +27,12 @@ def notify(self, notification): message = """\From: %s\nTo: %s\nSubject: %s\n\n%s """ % (FROM, ", ".join(TO), SUBJECT, TEXT) try: - server = smtplib.SMTP("smtp.gmail.com", 587) - server.starttls() + server = smtplib.SMTP(host, int(port)) + if self.__config['tls']: + server.starttls() server.ehlo() - server.login(gmail_user, gmail_pwd) + if user and password: + server.login(user, password) server.sendmail(FROM, TO, message) server.close() print 'successfully sent the mail' diff --git a/notification/notifier/hipchat.py b/notification/notifier/hipchat.py new file mode 100644 index 0000000..0f46232 --- /dev/null +++ b/notification/notifier/hipchat.py @@ -0,0 +1,68 @@ +from hypchat import HypChat +import json +import requests + +class HipchatNotifier: + + def __init__(self, config): + self.__config = config + + def start(self): + if not self.__config['token'] or not self.__config['endpoint']: + print "HipChat configuration is missing %s" % self.__config + raise + elif self.__config['endpoint']: + return HypChat(self.__config['token'], endpoint=self.__config['endpoint']) + else: + return HypChat(self.__config['token']) + + def notify(self, notification): + hc = self.start() + try: + description = notification.incident.description + details = notification.incident.details + except : + description = notification.message + details = "" + try: + message = description + " " + details + colour = "yellow" + if "CRITICAL" in message: + colour = "red" + elif "UNKNOWN" in message: + colour = "gray" + if notification.user_to_notify.profile.hipchat_room_name: + print "Notifying HipChat via API v2" + response = hc.get_room(notification.user_to_notify.profile.hipchat_room_name).notification(message, color=colour, notify="True", format="html") + elif notification.user_to_notify.profile.hipchat_room_url: + print "Notifying HipChat via a simple POST" + headers = {"content-type": "application/json"} + hip_msg = '{"color": "' + colour + '", "message": "' + message + '", "notify": true, "message_format": "html"}' + response = requests.post(notification.user_to_notify.profile.hipchat_room_url,headers=headers,data=hip_msg) + print response.content + else: + print "HipChat message send failed" + return + print "HipChat message sent" + except Exception, e: + try: + resp = json.loads(str(e)) + print "Failed to send HipChat message %s " % (e, str(resp['error']['code']) + " " + resp['error']['message']) + raise + except ValueError, e2: + print "Failed to send HipChat message and failed to get it's error %s ; %s" % (e, e2) + raise + + def get_all_rooms(self): + try: + hc = self.start() + if not hc: + return [""] + else: + rooms = hc.rooms().contents() + names = [] + for room in rooms: + names.append(room['name']) + return names + except: + return [""] diff --git a/notification/tasks.py b/notification/tasks.py index 788dc67..7a119f9 100644 --- a/notification/tasks.py +++ b/notification/tasks.py @@ -1,6 +1,5 @@ from __future__ import absolute_import -from notification.notifier.rocket import RocketNotifier from openduty.celery import app from notification.notifier.pushover import PushoverNotifier from notification.notifier.xmpp import XmppNotifier @@ -9,14 +8,15 @@ from notification.notifier.twilio_call import TwilioCallNotifier from notification.notifier.slack import SlackNotifier from notification.notifier.prowl import ProwlNotifier - +from notification.notifier.rocket import RocketNotifier +from notification.notifier.hipchat import HipchatNotifier from notification.models import ScheduledNotification, UserNotificationMethod from django.conf import settings from django.utils import timezone from openduty.models import EventLog -@app.task(ignore_result=True) +@app.task(ignore_result=False) def send_notifications(notification_id): try: notification = ScheduledNotification.objects.get(id = notification_id) @@ -36,6 +36,9 @@ def send_notifications(notification_id): notifier = ProwlNotifier(settings.PROWL_SETTINGS) elif notification.notifier == UserNotificationMethod.METHOD_ROCKET: notifier = RocketNotifier() + elif notification.notifier == UserNotificationMethod.METHOD_HIPCHAT: + notifier = HipchatNotifier(settings.HIPCHAT_SETTINGS) + notifier.notify(notification) # Log successful notification logmessage = EventLog() @@ -44,15 +47,16 @@ def send_notifications(notification_id): logmessage.incident_key = notification.incident logmessage.user = notification.user_to_notify logmessage.action = 'notified' - logmessage.data = "Notification sent to %s about %s service" % (notification.user_to_notify, logmessage.service_key, ) + logmessage.data = "Notification sent to %s about %s service via %s" % (notification.user_to_notify, logmessage.service_key, notification.notifier, ) logmessage.occurred_at = timezone.now() logmessage.save() + return (logmessage.data) if notification.notifier != UserNotificationMethod.METHOD_TWILIO_CALL: # In case of a twilio call, we need the object for TWiml generation notification.delete() except ScheduledNotification.DoesNotExist: pass #Incident was resolved. NOP. - except: + except Exception, e: # Log successful notification logmessage = EventLog() if notification.incident: @@ -60,8 +64,8 @@ def send_notifications(notification_id): logmessage.incident_key = notification.incident logmessage.user = notification.user_to_notify logmessage.action = 'notification_failed' - logmessage.data = "Sending notification failed to %s about %s service" % (notification.user_to_notify, logmessage.service_key, ) + logmessage.data = "Sending notification failed to %s about %s service because %s" % (notification.user_to_notify, logmessage.service_key, e,) logmessage.occurred_at = timezone.now() logmessage.save() + return (logmessage.data) raise - diff --git a/openduty/escalation.py b/openduty/escalation.py index 4f14017..f685adb 100644 --- a/openduty/escalation.py +++ b/openduty/escalation.py @@ -5,6 +5,7 @@ from django.contrib.auth.decorators import login_required from .models import Calendar, User, SchedulePolicy, SchedulePolicyRule +from django.contrib.auth.models import Group from django.http import Http404 from django.views.decorators.http import require_http_methods from django.db import IntegrityError @@ -32,12 +33,16 @@ def new(request): users = User.objects.all() except User.DoesNotExist: users = None + try: + groups = Group.objects.all() + except Group.DoesNotExist: + groups = None try: calendars = Calendar.objects.all() except Calendar.DoesNotExist: calendars = None - return TemplateResponse(request, 'escalation/edit.html', {'calendars': calendars, 'users': users}) + return TemplateResponse(request, 'escalation/edit.html', {'calendars': calendars, 'groups': groups, 'users': users}) @login_required() def edit(request, id): @@ -51,13 +56,17 @@ def edit(request, id): calendars = Calendar.objects.all() except Calendar.DoesNotExist: calendars = None + try: + groups = Group.objects.all() + except Group.DoesNotExist: + groups = None try: users = User.objects.all() except User.DoesNotExist: users = None return TemplateResponse(request, 'escalation/edit.html', {'item': policy, 'elements': elements, - 'calendars': calendars, 'users': users}) + 'calendars': calendars, 'groups': groups, 'users': users}) except Calendar.DoesNotExist: raise Http404 @@ -91,12 +100,16 @@ def save(request): parts = item.split("|") rule.escalate_after = 0 # HACK! rule.position = idx + 1 + rule.schedule = None + rule.user_id = None + rule.group_id = None if parts[0] == "user": rule.user_id = User.objects.get(id=parts[1]) - rule.schedule = None - if parts[0] == "calendar": + elif parts[0] == "calendar": rule.schedule = Calendar.objects.get(id=parts[1]) - rule.user_id = None + elif parts[0] == "group": + rule.group_id = Group.objects.get(id=parts[1]) + try: rule.save() except IntegrityError: @@ -104,3 +117,4 @@ def save(request): return HttpResponseRedirect('/policies/') + diff --git a/openduty/escalation_helper.py b/openduty/escalation_helper.py index 4873bef..fb20b42 100644 --- a/openduty/escalation_helper.py +++ b/openduty/escalation_helper.py @@ -1,22 +1,45 @@ __author__ = 'deathowl' from .models import User, SchedulePolicyRule, Service +from django.contrib.auth.models import Group from datetime import datetime, timedelta from django.utils import timezone from schedule.periods import Day from datetime import timedelta -def get_current_events_users(calendar): + +"""def get_current_events_users(calendar): now = timezone.make_aware(datetime.now(), timezone.get_current_timezone()) result = [] day = Day(calendar.events.all(), now) for o in day.get_occurrences(): if o.start <= now <= o.end: usernames = o.event.title.split(',') + print usernames for username in usernames: result.append(User.objects.get(username=username.strip())) return result +""" +def get_current_events_users(calendar): + now = timezone.make_aware(datetime.now(), timezone.get_current_timezone()) + result = [] + day = Day(calendar.events.all(), now) + for o in day.get_occurrences(): + if o.start <= now <= o.end: + items = o.event.title.split(',') + for item in items: + if Group.objects.filter(name=item.strip()).exists(): + for user in User.objects.filter(groups__name=item.strip()): + user.came_from_group = item.strip() + result.append(user) + else: + result.append(User.objects.get(username=item.strip())) + #tache suivante apres add group calendar + return result + + + def get_events_users_inbetween(calendar, since, until): delta = until - since result = {} @@ -26,14 +49,22 @@ def get_events_users_inbetween(calendar, since, until): day = Day(calendar.events.all(), that_day) for o in day.get_occurrences(): if o.start <= that_day <= o.end: - usernames = o.event.title.split(',') - for username in usernames: - if username not in result.keys(): - user_instance = User.objects.get(username=username.strip()) - result[username] = {"start": o.start, "person": username.strip(), "end": o.end, - "email": user_instance.email} + items = o.event.title.split(',') + for item in items: + username = item + if Group.objects.filter(name=item.strip()) is not None: + for user in User.objects.filter(groups__name=item): + if user not in result.keys(): + result.append(user) + else: + result[username]["end"] = o.end else: - result[username]["end"] = o.end + if item not in result.keys(): + user_instance = User.objects.get(username=item.strip()) + result[username] = {"start": o.start, "person": username.strip(), "end": o.end, + "email": user_instance.email} + else: + result[username]["end"] = o.end return result.values() @@ -45,8 +76,12 @@ def get_escalation_for_service(service): for item in rules: if item.schedule: result += get_current_events_users(item.schedule) - if item.user_id: + elif item.user_id: result.append(item.user_id) + elif item.group_id and Group.objects.filter(name=item.group_id) is not None: + for user in User.objects.filter(groups__name=item.group_id): + user.came_from_group = item.group_id.name + result.append(user) #TODO: This isnt de-deuped, is that right? return result @@ -56,3 +91,4 @@ def services_where_user_is_on_call(user): Q(policy__rules__user_id=user) | Q(policy__rules__schedule__event__title__icontains=user) ) return services + diff --git a/openduty/events.py b/openduty/events.py index baa5300..75688c7 100644 --- a/openduty/events.py +++ b/openduty/events.py @@ -1,4 +1,5 @@ import datetime +from django.contrib.auth.models import Group from django.core.urlresolvers import reverse from django.http import Http404, HttpResponseRedirect from django.shortcuts import get_object_or_404, render_to_response @@ -38,7 +39,9 @@ def create_or_edit_event(request, calendar_slug, event_id=None, next=None, if data: data["title"] = data["oncall"]+","+data["fallback"] form = form_class(data=data or None, instance=instance, initial=initial_data) - users = User.objects.all(); + users = User.objects.all() + groups = Group.objects.all() + #users = Item.groups.all(); if form.is_valid(): event = form.save(commit=False) if instance is None: @@ -57,7 +60,7 @@ def create_or_edit_event(request, calendar_slug, event_id=None, next=None, if instance.end_recurring_period: data["recurr_ymd"] = instance.end_recurring_period.date().isoformat() data["description"] = instance.description - data["rule"] = instance.rule and instance.rule.id or "" + data["rule"] = (instance.rule and instance.rule.id) or "" next = get_next_url(request, next) @@ -66,6 +69,7 @@ def create_or_edit_event(request, calendar_slug, event_id=None, next=None, "calendar": calendar, "next":next, "users":users, + "groups": groups, "form": form, }, context_instance=RequestContext(request)) diff --git a/openduty/incidents.py b/openduty/incidents.py index c3e9917..a0aeb5e 100644 --- a/openduty/incidents.py +++ b/openduty/incidents.py @@ -13,6 +13,7 @@ from .serializers import IncidentSerializer from rest_framework import status from rest_framework.response import Response +from rest_framework.decorators import detail_route, list_route from django.http import HttpResponseRedirect from django.contrib.auth.decorators import login_required from django.template.response import TemplateResponse @@ -24,7 +25,6 @@ from openduty.tasks import unsilence_incident import uuid import base64 - from .tables import IncidentTable from django_tables2_simplefilter import FilteredSingleTableView @@ -50,7 +50,7 @@ def is_relevant(self, incident, new_event_type): # True if not acknowleged or type is resolve return (incident.event_type != Incident.ACKNOWLEDGE or (incident.event_type == Incident.ACKNOWLEDGE and - new_event_type == Incident.RESOLVE)) + new_event_type == Incident.RESOLVE) or (incident.event_type == Incident.ACKNOWLEDGE and new_event_type == Incident.UNACKNOWLEDGE)) # New incident else: # True if this is a trigger action @@ -62,19 +62,33 @@ def create(self, request, *args, **kwargs): serviceToken = ServiceTokens.objects.get(token_id=token) service = serviceToken.service_id except ServiceTokens.DoesNotExist: - return Response({}, status=status.HTTP_404_NOT_FOUND) + return Response({"Service key does not exist"}, status=status.HTTP_404_NOT_FOUND) except Token.DoesNotExist: - return Response({}, status=status.HTTP_403_FORBIDDEN) + return Response({"No service key"}, status=status.HTTP_403_FORBIDDEN) with transaction.atomic(): try: - incident = Incident.objects.get( - incident_key=request.DATA["incident_key"], - service_key=service) - - event_log_message = "%s api key changed %s from %s to %s" % ( - serviceToken.name, incident.incident_key, - incident.event_type, request.DATA['event_type']) + esc = False + incident = Incident.objects.get(incident_key=request.DATA["incident_key"],service_key=service) + print "Received %s for %s on service %s" % (request.DATA['event_type'],request.DATA['incident_key'],serviceToken.name) + #check if type is ACK or resolve and if there's an escalation to a different escalation policy, remove it + if request.DATA['event_type'] == Incident.ACKNOWLEDGE or request.DATA['event_type'] == Incident.RESOLVE: + print "ACK or Resolve, removing specific escalation" + esc = True + incident.service_to_escalate_to = None + incident.save() + # check if incident is resolved and refuse to ACK + if not (incident.event_type == Incident.RESOLVE and request.DATA['event_type'] == Incident.ACKNOWLEDGE): + event_log_message = "%s api key changed %s from %s to %s" % ( + serviceToken.name, incident.incident_key, + incident.event_type, request.DATA['event_type']) + if esc: + event_log_message += ", unescalated" + else: + response = {} + response["status"] = "failure" + response["message"] = "Can\'t ACK a resolved incident!" + return Response(response, status=status.HTTP_400_BAD_REQUEST) except (Incident.DoesNotExist, KeyError): incident = Incident() try: @@ -127,8 +141,10 @@ def create(self, request, *args, **kwargs): service=service).count() > 0 if incident.event_type == Incident.TRIGGER and not servicesilenced: NotificationHelper.notify_incident(incident) - if incident.event_type == "resolve" or incident.event_type == Incident.ACKNOWLEDGE: + if incident.event_type == Incident.RESOLVE or incident.event_type == Incident.ACKNOWLEDGE: ScheduledNotification.remove_all_for_incident(incident) + if incident.event_type == Incident.RESOLVE and service.send_resolve_enabled: + NotificationHelper.notify_incident(incident) headers = self.get_success_headers(request.POST) @@ -136,10 +152,66 @@ def create(self, request, *args, **kwargs): response["status"] = "success" response["message"] = "Event processed" response["incident_key"] = incident.incident_key - return Response( - response, - status=status.HTTP_201_CREATED, - headers=headers) + return Response(response,status=status.HTTP_201_CREATED,headers=headers) + + + #escalate an incident to another service's escalation rule; persists until ACK + @detail_route(methods=['put']) + def escalate(self, request, *args, **kwargs): + #get arguments + try: + token = Token.objects.get(key=request.DATA["service_key"]) + serviceToken = ServiceTokens.objects.get(token_id=token) + service = serviceToken.service_id + except ServiceTokens.DoesNotExist: + return Response({"Service key does not exist"}, status=status.HTTP_404_NOT_FOUND) + except Token.DoesNotExist: + return Response({"No service key"}, status=status.HTTP_403_FORBIDDEN) + + try: + token2 = Token.objects.get(key=request.DATA["service_key_to_escalate_to"]) + serviceToken2 = ServiceTokens.objects.get(token_id=token2) + service2 = serviceToken2.service_id + except ServiceTokens.DoesNotExist: + return Response({"Service to escalate to key does not exist"}, status=status.HTTP_404_NOT_FOUND) + except Token.DoesNotExist: + return Response({"No service to escalate to key"}, status=status.HTTP_403_FORBIDDEN) + + #modify incident + with transaction.atomic(): + try: + # get service_to_escalate to and modify incident object + incident = Incident.objects.get(incident_key=request.DATA["incident_key"],service_key=service) + incident.service_to_escalate_to = service2 + incident.event_type = "escalated" + if request.DATA["incident_details"]: + incident.details = request.DATA["incident_details"] +# incident.description = "[escalated] " + incident.description + incident.save() + + event_log_message = "%s escalated to service escalation policy : %s to %s" % (request.user.username, incident.incident_key, service2.name) + event_log = EventLog() + event_log.user = request.user + event_log.action = "escalate" + event_log.incident_key = incident + event_log.service_key = incident.service_key + event_log.data = event_log_message + event_log.occurred_at = timezone.now() + event_log.save() + + except (Incident.DoesNotExist, KeyError): + return Response({"Incident does not exist"}, status=status.HTTP_404_NOT_FOUND) + except (Service.DoesNotExist, KeyError): + return Response({"Service does not exist"}, status=status.HTTP_400_BAD_REQUEST) + # remove all planned notifs + ScheduledNotification.remove_all_for_incident(incident) + # notify anew, this time notify_incident will detect the service_to_escalate to and notify its escalation rule + NotificationHelper.notify_incident(incident) + + headers = self.get_success_headers(request.POST) + + return Response({"Incident successfully escalated to service " + service2.name + " escalation policy"},status=status.HTTP_200_OK,headers=headers) + class ServicesByMe(FilteredSingleTableView): model = Incident @@ -185,7 +257,10 @@ def _update_type(user, ids, event_type): for incident_id in ids: with transaction.atomic(): incident = Incident.objects.get(id=int(incident_id)) - + unesc = False + if incident.service_to_escalate_to is not None: + incident.service_to_escalate_to = None + unesc = True logmessage = EventLog() logmessage.service_key = incident.service_key logmessage.user = user @@ -195,6 +270,8 @@ def _update_type(user, ids, event_type): incident.incident_key, incident.event_type, event_type) + if unesc: + logmessage.data += ", unescalated" logmessage.occurred_at = timezone.now() incident.event_type = event_type @@ -213,16 +290,27 @@ def update_type(request): event_type = request.POST['event_type'] event_types = ('acknowledge', 'resolve') incident_ids = request.POST.getlist('selection', None) - if not event_type: messages.error(request, 'Invalid event modification!') return HttpResponseRedirect(request.POST['url']) try: if incident_ids: - _update_type(request.user, incident_ids, event_type) + for id in incident_ids: + with transaction.atomic(): + incident = Incident.objects.get(id=id) + if incident.event_type == 'resolve' and event_type == 'acknowledge': + messages.error(request, 'Can\' ACK a resolved incident!') + return HttpResponseRedirect(request.POST['url']) + else: + _update_type(request.user, incident_ids, event_type) else: id = request.POST.get('id') - _update_type(request.user, [id], event_type) + incident = Incident.objects.get(id=id) + if incident.event_type == 'resolve' and event_type == 'acknowledge': + messages.error(request, 'Can\' ACK a resolved incident!') + return HttpResponseRedirect(request.POST['url']) + else: + _update_type(request.user, [id], event_type) except Incident.DoesNotExist: messages.error(request, 'Incident not found') return HttpResponseRedirect(request.POST['url']) @@ -317,4 +405,4 @@ def unsilence(request, incident_id): pass return HttpResponseRedirect(url) except Service.DoesNotExist: - raise Http404 \ No newline at end of file + raise Http404 diff --git a/openduty/models.py b/openduty/models.py index 736bcf4..4201222 100644 --- a/openduty/models.py +++ b/openduty/models.py @@ -7,7 +7,7 @@ from django.db import models from django.utils.translation import ugettext_lazy as _ from django.utils.encoding import python_2_unicode_compatible -from django.contrib.auth.models import User +from django.contrib.auth.models import User, Group from uuidfield import UUIDField from django.core.exceptions import ValidationError from schedule.models import Calendar @@ -72,6 +72,7 @@ class Service(models.Model): policy = models.ForeignKey(SchedulePolicy, blank=True, null=True) escalate_after = models.IntegerField(blank=True, null=True) notifications_disabled = models.BooleanField(default=False) + send_resolve_enabled = models.BooleanField(default=False) class Meta: verbose_name = _('service') @@ -89,6 +90,7 @@ class EventLog(models.Model): Event Log """ ACTIONS = (('acknowledge', 'acknowledge'), + ('unacknowledge', 'unacknowledge'), ('resolve', 'resolve'), ('silence_service', 'silence service'), ('unsilence_service', 'unsilence service'), @@ -103,12 +105,14 @@ class EventLog(models.Model): @property def color(self): colort_dict = {'acknowledge': 'warning', + 'unacknowledge' : 'warning', 'resolve': 'success', 'silence_service': 'active', 'unsilence_service': 'active', 'silence_incident': 'active', 'unsilence_incident': 'active', 'forward': 'info', + 'escalate': 'info', 'trigger': 'trigger', 'notified': 'success', 'notification_failed': 'danger', @@ -137,19 +141,23 @@ class Incident(models.Model): TRIGGER = "trigger" RESOLVE = "resolve" ACKNOWLEDGE = "acknowledge" + UNACKNOWLEDGE = "unacknowledge" + ESCALATE = "escalate" """ Incidents are representations of a malfunction in the system. """ - service_key = models.ForeignKey(Service) + service_key = models.ForeignKey(Service,related_name="incident") incident_key = models.CharField(max_length=200) event_type = models.CharField(max_length=15) description = models.CharField(max_length=200) details = models.TextField() occurred_at = models.DateTimeField() + service_to_escalate_to = models.ForeignKey(Service,related_name="service_to_escalate_to_id",null=True, blank=True, default = None) @property def color(self): colort_dict = {'acknowledge': 'warning', + 'unacknowledge': 'warning', 'resolve': 'success', 'silence_service': 'active', 'silence_incident': 'active', @@ -169,8 +177,8 @@ def __str__(self): def natural_key(self): return (self.service_key, self.incident_key) def clean(self): - if self.event_type not in ['trigger', 'acknowledge', 'resolve']: - raise ValidationError("'%s' is an invalid event type, valid values are 'trigger', 'acknowledge' and 'resolve'" % self.event_type) + if self.event_type not in ['trigger', 'acknowledge','unacknowledge', 'resolve']: + raise ValidationError("'%s' is an invalid event type, valid values are 'trigger', 'acknowledge', 'unacknowledge' and 'resolve'" % self.event_type) @python_2_unicode_compatible class ServiceTokens(models.Model): @@ -199,6 +207,7 @@ class SchedulePolicyRule(models.Model): schedule_policy = models.ForeignKey(SchedulePolicy, related_name='rules') position = models.IntegerField() user_id = models.ForeignKey(User, blank=True, null=True) + group_id = models.ForeignKey(Group, blank=True, null=True) schedule = models.ForeignKey(Calendar, blank=True, null=True) escalate_after = models.IntegerField() @@ -211,7 +220,7 @@ def __str__(self): @classmethod def getRulesForService(cls, service): - return cls.objects.filter(schedule_policy=service.policy) + return cls.objects.filter(schedule_policy=service.policy.id) class UserProfile(models.Model): user = models.OneToOneField('auth.User', related_name='profile') @@ -223,6 +232,9 @@ class UserProfile(models.Model): prowl_application = models.CharField(max_length=256, blank=True) prowl_url = models.CharField(max_length=512, blank=True) rocket_webhook_url = models.CharField(max_length=512, blank=True) + hipchat_room_name = models.CharField(max_length=100) + hipchat_room_url = models.CharField(max_length=100) + send_resolve_enabled = models.BooleanField(default=False) class ServiceSilenced(models.Model): service = models.ForeignKey(Service) diff --git a/openduty/serializers.py b/openduty/serializers.py index d12cedf..bad826a 100644 --- a/openduty/serializers.py +++ b/openduty/serializers.py @@ -37,7 +37,7 @@ class SchedulePolicyRuleSerializer(serializers.HyperlinkedModelSerializer): class Meta: model = SchedulePolicyRule - fields = ('schedule_policy', 'position', 'user_id', 'schedule', 'escalate_after') + fields = ('schedule_policy', 'position', 'user_id', 'group_id', 'schedule', 'escalate_after') class NoneSerializer(serializers.Serializer): diff --git a/openduty/services.py b/openduty/services.py index 7bfb5ae..b666647 100644 --- a/openduty/services.py +++ b/openduty/services.py @@ -80,6 +80,7 @@ def save(request): service.escalate_after = request.POST['escalate_after'] service.retry = request.POST['retry'] service.notifications_disabled = request.POST.get("disable_notification", "off") == "on" + service.send_resolve_enabled = request.POST.get("send_resolve_notification", "off") == "on" if(request.POST['policy']): pol = SchedulePolicy.objects.get(id = request.POST['policy']) else: @@ -175,4 +176,4 @@ def unsilence(request, service_id): pass return HttpResponseRedirect(url) except Service.DoesNotExist: - raise Http404 \ No newline at end of file + raise Http404 diff --git a/openduty/settings.py b/openduty/settings.py index 97e4b78..b269971 100644 --- a/openduty/settings.py +++ b/openduty/settings.py @@ -52,7 +52,8 @@ 'django_tables2', 'django_tables2_simplefilter', 'bootstrap3', - "django_twilio" + "django_twilio", + "debug_toolbar" ) TEMPLATE_CONTEXT_PROCESSORS = ( @@ -120,6 +121,11 @@ } EMAIL_SETTINGS = { + 'user': '', + 'password': '', + 'host': '', + 'port': '', + 'tls': False } TWILIO_SETTINGS = { @@ -131,6 +137,12 @@ PROWL_SETTINGS = { } +HIPCHAT_SETTINGS = { + 'token' : '', + 'endpoint' : '' +} + + CACHES = { 'default': { 'BACKEND': 'django.core.cache.backends.locmem.LocMemCache', diff --git a/openduty/tables.py b/openduty/tables.py index af48ee8..3726040 100644 --- a/openduty/tables.py +++ b/openduty/tables.py @@ -35,7 +35,9 @@ class IncidentTable(tables.Table): values_list=[ (str(x), str(x.id)) for x in Service.objects.all()]), F('event_type', 'Event', values_list=EventLog.ACTIONS), - ) + F('incident_key', 'Incident Key', + values_list= [(i, i) for i in Incident.objects.values_list('incident_key', flat=True).order_by('-occurred_at')[:500] ]) + ) tr_class = tables.Column(visible=False, empty_values=()) def render_tr_class(self, record): diff --git a/openduty/templates/base.html b/openduty/templates/base.html index 2bfdfe7..ba1135a 100644 --- a/openduty/templates/base.html +++ b/openduty/templates/base.html @@ -30,9 +30,9 @@