2018-05-21 16:36:45 +02:00
|
|
|
// Copyright 2018 Google LLC
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// https://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"io"
|
|
|
|
"io/ioutil"
|
|
|
|
"net/http"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
2020-03-05 12:58:39 +01:00
|
|
|
|
2021-04-07 03:20:52 +02:00
|
|
|
"github.com/google/alertmanager-irc-relay/logging"
|
2020-03-05 12:58:39 +01:00
|
|
|
"github.com/gorilla/mux"
|
|
|
|
promtmpl "github.com/prometheus/alertmanager/template"
|
|
|
|
"github.com/prometheus/client_golang/prometheus"
|
|
|
|
"github.com/prometheus/client_golang/prometheus/promauto"
|
|
|
|
"github.com/prometheus/client_golang/prometheus/promhttp"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
handledAlertGroups = promauto.NewCounterVec(prometheus.CounterOpts{
|
|
|
|
Name: "webhook_handled_alert_groups",
|
|
|
|
Help: "Number of alert groups received"},
|
|
|
|
[]string{"ircchannel"},
|
|
|
|
)
|
|
|
|
handledAlerts = promauto.NewCounterVec(prometheus.CounterOpts{
|
|
|
|
Name: "webhook_handled_alerts",
|
|
|
|
Help: "Number of single alert messages relayed"},
|
|
|
|
[]string{"ircchannel"},
|
|
|
|
)
|
|
|
|
alertHandlingErrors = promauto.NewCounterVec(prometheus.CounterOpts{
|
|
|
|
Name: "webhook_alert_handling_errors",
|
|
|
|
Help: "Errors while processing webhook requests"},
|
|
|
|
[]string{"ircchannel", "error"},
|
|
|
|
)
|
2018-05-21 16:36:45 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
type HTTPListener func(string, http.Handler) error
|
|
|
|
|
|
|
|
type HTTPServer struct {
|
2021-02-20 00:41:10 +01:00
|
|
|
Addr string
|
|
|
|
Port int
|
|
|
|
formatter *Formatter
|
|
|
|
AlertMsgs chan AlertMsg
|
|
|
|
httpListener HTTPListener
|
2018-05-21 16:36:45 +02:00
|
|
|
}
|
|
|
|
|
2020-01-25 17:42:59 +01:00
|
|
|
func NewHTTPServer(config *Config, alertMsgs chan AlertMsg) (
|
2018-05-21 16:36:45 +02:00
|
|
|
*HTTPServer, error) {
|
2020-01-25 17:42:59 +01:00
|
|
|
return NewHTTPServerForTesting(config, alertMsgs, http.ListenAndServe)
|
2018-05-21 16:36:45 +02:00
|
|
|
}
|
|
|
|
|
2020-01-25 17:42:59 +01:00
|
|
|
func NewHTTPServerForTesting(config *Config, alertMsgs chan AlertMsg,
|
2018-05-21 16:36:45 +02:00
|
|
|
httpListener HTTPListener) (*HTTPServer, error) {
|
2020-10-28 17:54:20 +01:00
|
|
|
formatter, err := NewFormatter(config)
|
2018-05-21 16:36:45 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
server := &HTTPServer{
|
2021-02-20 00:41:10 +01:00
|
|
|
Addr: config.HTTPHost,
|
|
|
|
Port: config.HTTPPort,
|
|
|
|
formatter: formatter,
|
|
|
|
AlertMsgs: alertMsgs,
|
|
|
|
httpListener: httpListener,
|
2018-05-21 16:36:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return server, nil
|
|
|
|
}
|
|
|
|
|
2021-03-21 19:48:11 +01:00
|
|
|
func (s *HTTPServer) RelayAlert(w http.ResponseWriter, r *http.Request) {
|
2018-05-21 16:36:45 +02:00
|
|
|
vars := mux.Vars(r)
|
|
|
|
ircChannel := "#" + vars["IRCChannel"]
|
|
|
|
|
|
|
|
body, err := ioutil.ReadAll(io.LimitReader(r.Body, 1024*1024*1024))
|
|
|
|
if err != nil {
|
2021-04-07 03:20:52 +02:00
|
|
|
logging.Error("Could not get body: %s", err)
|
2020-03-05 12:58:39 +01:00
|
|
|
alertHandlingErrors.WithLabelValues(ircChannel, "read_body").Inc()
|
2018-05-21 16:36:45 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
var alertMessage = promtmpl.Data{}
|
|
|
|
if err := json.Unmarshal(body, &alertMessage); err != nil {
|
2021-04-07 03:20:52 +02:00
|
|
|
logging.Error("Could not decode request body (%s): %s", err, body)
|
2020-03-05 12:58:39 +01:00
|
|
|
alertHandlingErrors.WithLabelValues(ircChannel, "decode_body").Inc()
|
2018-05-21 16:36:45 +02:00
|
|
|
w.Header().Set("Content-Type", "application/json; charset=UTF-8")
|
|
|
|
w.WriteHeader(422) // Unprocessable entity
|
|
|
|
if err := json.NewEncoder(w).Encode(err); err != nil {
|
2021-04-07 03:20:52 +02:00
|
|
|
logging.Error("Could not write decoding error: %s", err)
|
2018-05-21 16:36:45 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
2020-03-05 12:58:39 +01:00
|
|
|
handledAlertGroups.WithLabelValues(ircChannel).Inc()
|
2021-03-21 19:48:11 +01:00
|
|
|
for _, alertMsg := range s.formatter.GetMsgsFromAlertMessage(
|
2018-05-21 16:36:45 +02:00
|
|
|
ircChannel, &alertMessage) {
|
|
|
|
select {
|
2021-03-21 19:48:11 +01:00
|
|
|
case s.AlertMsgs <- alertMsg:
|
2020-03-05 12:58:39 +01:00
|
|
|
handledAlerts.WithLabelValues(ircChannel).Inc()
|
2018-05-21 16:36:45 +02:00
|
|
|
default:
|
2021-04-07 03:20:52 +02:00
|
|
|
logging.Error("Could not send this alert to the IRC routine: %s",
|
2020-01-25 17:42:59 +01:00
|
|
|
alertMsg)
|
2020-03-05 12:58:39 +01:00
|
|
|
alertHandlingErrors.WithLabelValues(ircChannel, "internal_comm_channel_full").Inc()
|
2018-05-21 16:36:45 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-21 19:48:11 +01:00
|
|
|
func (s *HTTPServer) Run() {
|
2018-05-21 16:36:45 +02:00
|
|
|
router := mux.NewRouter().StrictSlash(true)
|
|
|
|
|
2020-03-05 12:58:39 +01:00
|
|
|
router.Path("/metrics").Handler(promhttp.Handler())
|
|
|
|
|
2018-05-21 16:36:45 +02:00
|
|
|
handler := http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
2021-03-21 19:48:11 +01:00
|
|
|
s.RelayAlert(w, r)
|
2018-05-21 16:36:45 +02:00
|
|
|
})
|
|
|
|
router.Path("/{IRCChannel}").Handler(handler).Methods("POST")
|
|
|
|
|
|
|
|
listenAddr := strings.Join(
|
2021-03-21 19:48:11 +01:00
|
|
|
[]string{s.Addr, strconv.Itoa(s.Port)}, ":")
|
2021-04-07 03:20:52 +02:00
|
|
|
logging.Info("Starting HTTP server")
|
2021-03-21 19:48:11 +01:00
|
|
|
if err := s.httpListener(listenAddr, router); err != nil {
|
2021-04-07 03:20:52 +02:00
|
|
|
logging.Error("Could not start http server: %s", err)
|
2018-05-21 16:36:45 +02:00
|
|
|
}
|
|
|
|
}
|