Skip to content

Commit e2f8bb2

Browse files
authored
Merge pull request #254 from Ithrael/feat-aliveness
new feat:aliveness check
2 parents dd5ad50 + 7897a59 commit e2f8bb2

File tree

4 files changed

+119
-1
lines changed

4 files changed

+119
-1
lines changed

config.example.json

+3-1
Original file line numberDiff line numberDiff line change
@@ -15,11 +15,13 @@
1515
"skip_vhost": "^$",
1616
"include_vhost": ".*",
1717
"rabbit_capabilities": "no_sort,bert",
18+
"aliveness_vhost": "/",
1819
"enabled_exporters": [
1920
"exchange",
2021
"node",
2122
"overview",
22-
"queue"
23+
"queue",
24+
"aliveness"
2325
],
2426
"timeout": 30,
2527
"max_queues": 0

config.go

+6
Original file line numberDiff line numberDiff line change
@@ -30,6 +30,7 @@ var (
3030
SkipVHost: regexp.MustCompile("^$"),
3131
IncludeVHost: regexp.MustCompile(".*"),
3232
RabbitCapabilities: parseCapabilities("no_sort,bert"),
33+
AlivenessVhost: "/",
3334
EnabledExporters: []string{"exchange", "node", "overview", "queue"},
3435
Timeout: 30,
3536
MaxQueues: 0,
@@ -58,6 +59,7 @@ type rabbitExporterConfig struct {
5859
IncludeVHostString string `json:"include_vhost"`
5960
RabbitCapabilitiesString string `json:"rabbit_capabilities"`
6061
RabbitCapabilities rabbitCapabilitySet `json:"-"`
62+
AlivenessVhost string `json:"aliveness_vhost"`
6163
EnabledExporters []string `json:"enabled_exporters"`
6264
Timeout int `json:"timeout"`
6365
MaxQueues int `json:"max_queues"`
@@ -195,6 +197,10 @@ func initConfig() {
195197
config.EnabledExporters = strings.Split(enabledExporters, ",")
196198
}
197199

200+
if alivenessVhost := os.Getenv("ALIVENESS_VHOST"); alivenessVhost != "" {
201+
config.AlivenessVhost = alivenessVhost
202+
}
203+
198204
if timeout := os.Getenv("RABBIT_TIMEOUT"); timeout != "" {
199205
t, err := strconv.Atoi(timeout)
200206
if err != nil {

exporter_aliveness.go

+104
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,104 @@
1+
package main
2+
3+
import (
4+
"context"
5+
6+
"github.com/prometheus/client_golang/prometheus"
7+
log "github.com/sirupsen/logrus"
8+
)
9+
10+
func init() {
11+
RegisterExporter("aliveness", newExporterAliveness)
12+
}
13+
14+
var (
15+
alivenessLabels = []string{"vhost"}
16+
17+
alivenessGaugeVec = map[string]*prometheus.GaugeVec{
18+
"vhost.aliveness": newGaugeVec("aliveness_test", "vhost aliveness test", alivenessLabels),
19+
}
20+
21+
rabbitmqAlivenessMetric = prometheus.NewGaugeVec(
22+
prometheus.GaugeOpts{
23+
Name: "rabbitmq_aliveness_info",
24+
Help: "A metric with value 1 status:ok else 0 labeled by aliveness test status, error, reason",
25+
},
26+
[]string{"status", "error", "reason"},
27+
)
28+
)
29+
30+
type exporterAliveness struct {
31+
alivenessMetrics map[string]*prometheus.GaugeVec
32+
alivenessInfo AlivenessInfo
33+
}
34+
35+
type AlivenessInfo struct {
36+
Status string
37+
Error string
38+
Reason string
39+
}
40+
41+
func newExporterAliveness() Exporter {
42+
alivenessGaugeVecActual := alivenessGaugeVec
43+
44+
if len(config.ExcludeMetrics) > 0 {
45+
for _, metric := range config.ExcludeMetrics {
46+
if alivenessGaugeVecActual[metric] != nil {
47+
delete(alivenessGaugeVecActual, metric)
48+
}
49+
}
50+
}
51+
52+
return &exporterAliveness{
53+
alivenessMetrics: alivenessGaugeVecActual,
54+
alivenessInfo: AlivenessInfo{},
55+
}
56+
}
57+
58+
func (e *exporterAliveness) Collect(ctx context.Context, ch chan<- prometheus.Metric) error {
59+
body, contentType, err := apiRequest(config, "aliveness-test")
60+
if err != nil {
61+
return err
62+
}
63+
64+
reply, err := MakeReply(contentType, body)
65+
if err != nil {
66+
return err
67+
}
68+
69+
rabbitMqAlivenessData := reply.MakeMap()
70+
71+
e.alivenessInfo.Status, _ = reply.GetString("status")
72+
e.alivenessInfo.Error, _ = reply.GetString("error")
73+
e.alivenessInfo.Reason, _ = reply.GetString("reason")
74+
75+
rabbitmqAlivenessMetric.Reset()
76+
var flag float64 = 0
77+
if e.alivenessInfo.Status == "ok" {
78+
flag = 1
79+
}
80+
rabbitmqAlivenessMetric.WithLabelValues(e.alivenessInfo.Status, e.alivenessInfo.Error, e.alivenessInfo.Reason).Set(flag)
81+
82+
log.WithField("alivenesswData", rabbitMqAlivenessData).Debug("Aliveness data")
83+
for key, gauge := range e.alivenessMetrics {
84+
if value, ok := rabbitMqAlivenessData[key]; ok {
85+
log.WithFields(log.Fields{"key": key, "value": value}).Debug("Set aliveness metric for key")
86+
gauge.WithLabelValues(e.alivenessInfo.Status).Set(value)
87+
}
88+
}
89+
90+
if ch != nil {
91+
rabbitmqAlivenessMetric.Collect(ch)
92+
for _, gauge := range e.alivenessMetrics {
93+
gauge.Collect(ch)
94+
}
95+
}
96+
return nil
97+
}
98+
99+
func (e exporterAliveness) Describe(ch chan<- *prometheus.Desc) {
100+
rabbitmqVersionMetric.Describe(ch)
101+
for _, gauge := range e.alivenessMetrics {
102+
gauge.Describe(ch)
103+
}
104+
}

rabbitClient.go

+6
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@ import (
66
"errors"
77
"io/ioutil"
88
"net/http"
9+
"net/url"
910
"os"
1011
"time"
1112

@@ -67,6 +68,11 @@ func apiRequest(config rabbitExporterConfig, endpoint string) ([]byte, string, e
6768
args = "?sort="
6869
}
6970

71+
if endpoint == "aliveness-test" {
72+
escapeAlivenessVhost := url.QueryEscape(config.AlivenessVhost)
73+
args = "/" + escapeAlivenessVhost
74+
}
75+
7076
req, err := http.NewRequest("GET", config.RabbitURL+"/api/"+endpoint+args, nil)
7177
if err != nil {
7278
log.WithFields(log.Fields{"error": err, "host": config.RabbitURL}).Error("Error while constructing rabbitHost request")

0 commit comments

Comments
 (0)