statping/handlers/incident.go

223 lines
5.4 KiB
Go

package handlers
import (
"fmt"
"github.com/gorilla/mux"
"github.com/razorpay/statping/types/errors"
"github.com/razorpay/statping/types/incidents"
"github.com/razorpay/statping/types/services"
"github.com/razorpay/statping/utils"
"net/http"
"sort"
"time"
)
func findIncident(r *http.Request) (*incidents.Incident, int64, error) {
vars := mux.Vars(r)
if utils.NotNumber(vars["id"]) {
return nil, 0, errors.NotNumber
}
id := utils.ToInt(vars["id"])
if id == 0 {
return nil, id, errors.IDMissing
}
checkin, err := incidents.Find(id)
if err != nil {
return nil, id, errors.Missing(&incidents.Incident{}, id)
}
return checkin, id, nil
}
func apiServiceIncidentsHandler(w http.ResponseWriter, r *http.Request) {
service, err := findService(r)
if err != nil {
sendErrorJson(err, w, r)
return
}
returnJson(service.Incidents, w, r)
}
func apiServiceActiveIncidentsHandler(w http.ResponseWriter, r *http.Request) {
service, err := findService(r)
if err != nil {
sendErrorJson(err, w, r)
return
}
visibleIncidents := getVisibleIncidentsOfService(service)
returnJson(visibleIncidents, w, r)
}
func apiSubServiceActiveIncidentsHandler(w http.ResponseWriter, r *http.Request) {
service, err := findService(r)
if err != nil {
sendErrorJson(err, w, r)
return
}
subService, err := findPublicSubService(r, service)
if err != nil {
sendErrorJson(err, w, r)
return
}
visibleIncidents := getVisibleIncidentsOfService(subService)
returnJson(visibleIncidents, w, r)
}
func getVisibleIncidentsOfService(service *services.Service) []incidents.Incident {
var visibleIncidents []incidents.Incident
var visibleIncidentIds []int64
for _, incident := range service.Incidents {
if hasZeroUpdates(incident.Updates) {
visibleIncidents = append(visibleIncidents, *incident)
visibleIncidentIds = append(visibleIncidentIds, incident.Id)
} else if checkResolvedVisibility(incident.Updates) {
incidentVar := *incident
sortUpdates(incidentVar.Updates)
visibleIncidents = append(visibleIncidents, incidentVar)
visibleIncidentIds = append(visibleIncidentIds, incident.Id)
}
}
log.Info(fmt.Sprintf("Visible Incident Id's for the Service %v : %v", service.Name, visibleIncidentIds))
return visibleIncidents
}
func sortUpdates(updates []*incidents.IncidentUpdate) {
sort.Slice(updates, func(i, j int) bool {
return updates[i].CreatedAt.Unix() >= updates[j].CreatedAt.Unix()
})
}
func hasZeroUpdates(Updates []*incidents.IncidentUpdate) bool {
if len(Updates) == 0 {
return true
}
return false
}
// NOT [(last update is resolved) AND (last update was more than 2 hrs ago)]
func checkResolvedVisibility(incidentUpdates []*incidents.IncidentUpdate) bool {
if !(incidentUpdates[len(incidentUpdates)-1].Type == resolved &&
getTimeDiff(incidentUpdates[len(incidentUpdates)-1]) > incidentsTimeoutInMinutes) {
return true
}
return false
}
func getTimeDiff(update *incidents.IncidentUpdate) float64 {
return time.Now().Sub(update.CreatedAt).Minutes()
}
func apiIncidentUpdatesHandler(w http.ResponseWriter, r *http.Request) {
incid, _, err := findIncident(r)
if err != nil {
sendErrorJson(err, w, r)
return
}
returnJson(incid.Updates, w, r)
}
func apiCreateIncidentUpdateHandler(w http.ResponseWriter, r *http.Request) {
incid, _, err := findIncident(r)
if err != nil {
sendErrorJson(err, w, r)
return
}
var update *incidents.IncidentUpdate
if err := DecodeJSON(r, &update); err != nil {
sendErrorJson(err, w, r)
return
}
update.IncidentId = incid.Id
if err := update.Create(); err != nil {
sendErrorJson(err, w, r)
return
}
sendJsonAction(update, "create", w, r)
}
func apiCreateIncidentHandler(w http.ResponseWriter, r *http.Request) {
service, err := findService(r)
if err != nil {
sendErrorJson(err, w, r)
return
}
var incident *incidents.Incident
if err := DecodeJSON(r, &incident); err != nil {
sendErrorJson(err, w, r)
return
}
incident.ServiceId = service.Id
if err := incident.Create(); err != nil {
sendErrorJson(err, w, r)
return
}
sendJsonAction(incident, "create", w, r)
}
func apiUpdateIncidentHandler(w http.ResponseWriter, r *http.Request) {
incident, _, err := findIncident(r)
if err != nil {
sendErrorJson(err, w, r)
return
}
var updatedIncident *incidents.Incident
if err := DecodeJSON(r, &updatedIncident); err != nil {
sendErrorJson(err, w, r)
return
}
incident.Title = updatedIncident.Title
incident.Description = updatedIncident.Description
if err := incident.Update(); err != nil {
sendErrorJson(err, w, r)
return
}
sendJsonAction(incident, "update", w, r)
}
func apiIncidentUpdateHandler(w http.ResponseWriter, r *http.Request) {
incident, _, err := findIncident(r)
if err != nil {
sendErrorJson(err, w, r)
return
}
if err := DecodeJSON(r, &incident); err != nil {
sendErrorJson(err, w, r)
return
}
sendJsonAction(incident.Updates, "update", w, r)
}
func apiDeleteIncidentHandler(w http.ResponseWriter, r *http.Request) {
incident, _, err := findIncident(r)
if err != nil {
sendErrorJson(err, w, r)
return
}
if err := incident.Delete(); err != nil {
sendErrorJson(err, w, r)
return
}
sendJsonAction(incident, "delete", w, r)
}
func apiDeleteIncidentUpdateHandler(w http.ResponseWriter, r *http.Request) {
vars := mux.Vars(r)
update, err := incidents.FindUpdate(utils.ToInt(vars["uid"]))
if err != nil {
sendErrorJson(err, w, r)
return
}
if err := update.Delete(); err != nil {
sendErrorJson(err, w, r)
return
}
sendJsonAction(update, "delete", w, r)
}