add global callback

This commit is contained in:
UlricQin 2022-02-10 17:32:06 +08:00
parent 3ccc09674e
commit b32a7b3a9e
3 changed files with 88 additions and 1 deletions

View File

@ -64,6 +64,14 @@ Enable = false
# complete redis key: ${ChannelPrefix} + ${Cluster} # complete redis key: ${ChannelPrefix} + ${Cluster}
ChannelPrefix = "/alerts/" ChannelPrefix = "/alerts/"
[Alerting.GlobalCallback]
Enable = false
Url = "http://a.com/n9e/callback"
BasicAuthUser = ""
BasicAuthPass = ""
Timeout = "5s"
Headers = ["Content-Type", "application/json", "X-From", "N9E"]
[NoData] [NoData]
Metric = "target_up" Metric = "target_up"
# unit: second # unit: second

View File

@ -6,6 +6,7 @@ import (
"os" "os"
"strings" "strings"
"sync" "sync"
"time"
"github.com/gin-gonic/gin" "github.com/gin-gonic/gin"
"github.com/koding/multiconfig" "github.com/koding/multiconfig"
@ -76,9 +77,21 @@ func MustLoad(fpaths ...string) {
} }
C.Heartbeat.Endpoint = fmt.Sprintf("%s:%d", C.Heartbeat.IP, C.HTTP.Port) C.Heartbeat.Endpoint = fmt.Sprintf("%s:%d", C.Heartbeat.IP, C.HTTP.Port)
C.Alerting.RedisPub.ChannelKey = C.Alerting.RedisPub.ChannelPrefix + C.ClusterName C.Alerting.RedisPub.ChannelKey = C.Alerting.RedisPub.ChannelPrefix + C.ClusterName
if C.Alerting.GlobalCallback.Enable {
if C.Alerting.GlobalCallback.Timeout == "" {
C.Alerting.GlobalCallback.TimeoutDuration = time.Second * 5
} else {
dur, err := time.ParseDuration(C.Alerting.GlobalCallback.Timeout)
if err != nil {
fmt.Println("failed to parse Alerting.GlobalCallback.Timeout")
os.Exit(1)
}
C.Alerting.GlobalCallback.TimeoutDuration = dur
}
}
fmt.Println("heartbeat.ip:", C.Heartbeat.IP) fmt.Println("heartbeat.ip:", C.Heartbeat.IP)
fmt.Printf("heartbeat.interval: %dms\n", C.Heartbeat.Interval) fmt.Printf("heartbeat.interval: %dms\n", C.Heartbeat.Interval)
}) })
@ -115,6 +128,7 @@ type Alerting struct {
NotifyConcurrency int NotifyConcurrency int
TemplatesDir string TemplatesDir string
RedisPub RedisPub RedisPub RedisPub
GlobalCallback GlobalCallback
} }
type RedisPub struct { type RedisPub struct {
@ -123,6 +137,16 @@ type RedisPub struct {
ChannelKey string ChannelKey string
} }
type GlobalCallback struct {
Enable bool
Url string
BasicAuthUser string
BasicAuthPass string
Timeout string
TimeoutDuration time.Duration
Headers []string
}
type NoData struct { type NoData struct {
Metric string Metric string
Interval int64 Interval int64

View File

@ -5,6 +5,8 @@ import (
"context" "context"
"encoding/json" "encoding/json"
"html/template" "html/template"
"io/ioutil"
"net/http"
"os/exec" "os/exec"
"path" "path"
"strings" "strings"
@ -119,6 +121,10 @@ func notify(event *models.AlertCurEvent) {
} }
} }
if config.C.Alerting.GlobalCallback.Enable {
DoGlobalCallback(event)
}
// no notify.py? do nothing // no notify.py? do nothing
if config.C.Alerting.NotifyScriptPath == "" { if config.C.Alerting.NotifyScriptPath == "" {
return return
@ -138,6 +144,55 @@ func notify(event *models.AlertCurEvent) {
} }
} }
func DoGlobalCallback(event *models.AlertCurEvent) {
conf := config.C.Alerting.GlobalCallback
if conf.Url == "" {
return
}
bs, err := json.Marshal(event)
if err != nil {
return
}
bf := bytes.NewBuffer(bs)
req, err := http.NewRequest("POST", conf.Url, bf)
if err != nil {
logger.Warning("DoGlobalCallback failed to new request", err)
return
}
if conf.BasicAuthUser != "" && conf.BasicAuthPass != "" {
req.SetBasicAuth(conf.BasicAuthUser, conf.BasicAuthPass)
}
if len(conf.Headers) > 0 && len(conf.Headers)%2 == 0 {
for i := 0; i < len(conf.Headers); i += 2 {
req.Header.Set(conf.Headers[i], conf.Headers[i+1])
}
}
client := http.Client{
Timeout: conf.TimeoutDuration,
}
var resp *http.Response
resp, err = client.Do(req)
if err != nil {
logger.Warning("DoGlobalCallback failed to call url", err)
return
}
var body []byte
if resp.Body != nil {
defer resp.Body.Close()
body, err = ioutil.ReadAll(resp.Body)
}
logger.Debugf("DoGlobalCallback done, url: %s, response code: %d, body: %s", conf.Url, resp.StatusCode, string(body))
}
func handleSubscribes(event models.AlertCurEvent, subs []*models.AlertSubscribe) { func handleSubscribes(event models.AlertCurEvent, subs []*models.AlertSubscribe) {
for i := 0; i < len(subs); i++ { for i := 0; i < len(subs); i++ {
handleSubscribe(event, subs[i]) handleSubscribe(event, subs[i])