forked from sapcc/mosquitto-exporter
-
Notifications
You must be signed in to change notification settings - Fork 0
/
main.go
292 lines (269 loc) · 9.51 KB
/
main.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
package main
import (
"log"
"net/http"
"os"
"regexp"
"strconv"
"strings"
"time"
"crypto/tls"
"github.com/codegangsta/cli"
mqtt "github.com/eclipse/paho.mqtt.golang"
"github.com/prometheus/client_golang/prometheus"
)
const (
appName = "Mosquitto exporter"
)
var (
ignoreKeyMetrics = map[string]string{
"$SYS/broker/timestamp": "The timestamp at which this particular build of the broker was made. Static.",
"$SYS/broker/version": "The version of the broker. Static.",
"$SYS/broker/clients/active": "deprecated in favour of $SYS/broker/clients/connected",
"$SYS/broker/clients/inactive": "deprecated in favour of $SYS/broker/clients/disconnected",
}
counterKeyMetrics = map[string]string{
"$SYS/broker/bytes/received": "The total number of bytes received since the broker started.",
"$SYS/broker/bytes/sent": "The total number of bytes sent since the broker started.",
"$SYS/broker/messages/received": "The total number of messages of any type received since the broker started.",
"$SYS/broker/messages/sent": "The total number of messages of any type sent since the broker started.",
"$SYS/broker/publish/bytes/received": "The total number of PUBLISH bytes received since the broker started.",
"$SYS/broker/publish/bytes/sent": "The total number of PUBLISH bytes sent since the broker started.",
"$SYS/broker/publish/messages/received": "The total number of PUBLISH messages received since the broker started.",
"$SYS/broker/publish/messages/sent": "The total number of PUBLISH messages sent since the broker started.",
"$SYS/broker/publish/messages/dropped": "The total number of PUBLISH messages that have been dropped due to inflight/queuing limits.",
"$SYS/broker/uptime": "The total number of seconds since the broker started.",
"$SYS/broker/clients/maximum": "The maximum number of clients connected simultaneously since the broker started",
"$SYS/broker/clients/total": "The total number of clients connected since the broker started.",
}
counterMetrics = map[string]*MosquittoCounter{}
gaugeMetrics = map[string]prometheus.Gauge{}
)
func main() {
app := cli.NewApp()
app.Name = appName
app.Version = versionString()
app.Authors = []cli.Author{
{
Name: "Arturo Reuschenbach Puncernau",
Email: "[email protected]",
},
{
Name: "Fabian Ruff",
Email: "[email protected]",
},
}
app.Usage = "Prometheus exporter for broker metrics"
app.Action = runServer
app.Flags = []cli.Flag{
cli.StringFlag{
Name: "endpoint,e",
Usage: "Endpoint for the Mosquitto message broker",
EnvVar: "BROKER_ENDPOINT",
Value: "tcp://127.0.0.1:1883",
},
cli.StringFlag{
Name: "bind-address,b",
Usage: "Listen address for metrics HTTP endpoint",
Value: "0.0.0.0:9234",
EnvVar: "BIND_ADDRESS",
},
cli.StringFlag{
Name: "user,u",
Usage: "Username for the Mosquitto message broker",
Value: "",
EnvVar: "MQTT_USER",
},
cli.StringFlag{
Name: "pass,p",
Usage: "Password for the User on the Mosquitto message broker",
Value: "",
EnvVar: "MQTT_PASS",
},
cli.StringFlag{
Name: "cert,c",
Usage: "Location of a TLS certificate .pem file for the Mosquitto message broker",
Value: "",
EnvVar: "MQTT_CERT",
},
cli.StringFlag{
Name: "key,k",
Usage: "Location of a TLS private key .pem file for the Mosquitto message broker",
Value: "",
EnvVar: "MQTT_KEY",
},
cli.StringFlag{
Name: "client-id,i",
Usage: "Client id to be used to connect to the Mosquitto message broker",
Value: "",
EnvVar: "MQTT_CLIENT_ID",
},
}
app.Run(os.Args)
}
func runServer(c *cli.Context) {
log.Printf("Starting mosquitto_broker %s", versionString())
opts := mqtt.NewClientOptions()
opts.SetCleanSession(true)
opts.AddBroker(c.String("endpoint"))
opts.SetPingTimeout(120 * time.Second)
opts.SetConnectTimeout(120 * time.Second)
opts.SetKeepAlive(120 * time.Second)
opts.SetMaxReconnectInterval(120 * time.Second)
// opts.SetConnectRetryInterval(60 * time.Second)
if c.String("client-id") != "" {
opts.SetClientID(c.String("client-id"))
}
// if you have a username you'll need a password with it
if c.String("user") != "" {
opts.SetUsername(c.String("user"))
if c.String("pass") != "" {
opts.SetPassword(c.String("pass"))
}
}
// if you have a client certificate you want a key aswell
if c.String("cert") != "" && c.String("key") != "" {
keyPair, err := tls.LoadX509KeyPair(c.String("cert"), c.String("key"))
if err != nil {
log.Printf("Failed to load certificate/keypair: %s", err)
}
tlsConfig := &tls.Config{
Certificates: []tls.Certificate{keyPair},
InsecureSkipVerify: true,
ClientAuth: tls.NoClientCert,
}
opts.SetTLSConfig(tlsConfig)
if !strings.HasPrefix(c.String("endpoint"), "ssl://") &&
!strings.HasPrefix(c.String("endpoint"), "tls://") {
log.Println("Warning: To use TLS the endpoint URL will have to begin with 'ssl://' or 'tls://'")
}
} else if (c.String("cert") != "" && c.String("key") == "") ||
(c.String("cert") == "" && c.String("key") != "") {
log.Println("Warning: For TLS to work both certificate and private key are needed. Skipping TLS.")
}
opts.OnConnect = func(client mqtt.Client) {
log.Printf("Connected to %s", c.String("endpoint"))
// subscribe on every (re)connect
// Subscribe to all "$SYS/broker/#" topics
token := client.Subscribe("$SYS/broker/#", 0, func(_ mqtt.Client, msg mqtt.Message) {
processUpdate(msg.Topic(), string(msg.Payload()))
})
if !token.WaitTimeout(60 * time.Second) {
log.Println("Error: Timeout subscribing to topic $SYS/broker/#")
}
if err := token.Error(); err != nil {
log.Printf("Failed to subscribe to topic $SYS/broker/#: %s", err)
}
// unsubscribe on every (re)connect
// Unsubscribe from "$SYS/broker/connection/+/state" topics to prevent auth plugin overloading because of
// big number of mosquitto clients. Auth plugin checks ACL endpoint for each "$SYS/broker/connection/+/state"
// topic when it wants to sent message to exporter
// token = client.Unsubscribe("$SYS/broker/connection/+/state")
// if !token.WaitTimeout(10 * time.Second) {
// log.Println("Error: Timeout unsubscribing to topic $SYS/broker/connection/+/state")
// }
// if err := token.Error(); err != nil {
// log.Printf("Failed to unsubscribe to topic $SYS/broker/connection/+/state: %s", err)
// }
}
opts.OnConnectionLost = func(client mqtt.Client, err error) {
log.Printf("Error: Connection to %s lost: %s", c.String("endpoint"), err)
}
client := mqtt.NewClient(opts)
// try to connect forever
for {
token := client.Connect()
if token.WaitTimeout(5 * time.Second) {
if token.Error() == nil {
break
}
log.Printf("Error: Failed to connect to broker: %s", token.Error())
} else {
log.Printf("Timeout connecting to endpoint %s", c.String("endpoint"))
}
time.Sleep(5 * time.Second)
}
// init the router and server
http.Handle("/metrics", prometheus.Handler())
http.HandleFunc("/", serveVersion)
log.Printf("Listening on %s...", c.GlobalString("bind-address"))
err := http.ListenAndServe(c.GlobalString("bind-address"), nil)
fatalfOnError(err, "Failed to bind on %s: ", c.GlobalString("bind-address"))
}
// $SYS/broker/bytes/received
func processUpdate(topic, payload string) {
//log.Printf("Got broker update with topic %s and data %s", topic, payload)
if _, ok := ignoreKeyMetrics[topic]; !ok {
if _, ok := counterKeyMetrics[topic]; ok {
// log.Printf("Processing counter metric %s with data %s", topic, payload)
processCounterMetric(topic, payload)
} else {
//log.Printf("Processing gauge metric %s with data %s", topic, payload)
processGaugeMetric(topic, payload)
}
}
}
func processCounterMetric(topic, payload string) {
if counterMetrics[topic] != nil {
value := parseValue(payload)
counterMetrics[topic].Set(value)
} else {
// create a mosquitto counter pointer
mCounter := NewMosquittoCounter(prometheus.NewDesc(
parseTopic(topic),
topic,
[]string{},
prometheus.Labels{},
))
// save it
counterMetrics[topic] = mCounter
// register the metric
prometheus.MustRegister(mCounter)
// add the first value
value := parseValue(payload)
counterMetrics[topic].Set(value)
}
}
func processGaugeMetric(topic, payload string) {
if gaugeMetrics[topic] != nil {
value := parseValue(payload)
gaugeMetrics[topic].Set(value)
} else {
gaugeMetrics[topic] = prometheus.NewGauge(prometheus.GaugeOpts{
Name: parseTopic(topic),
Help: topic,
})
// register the metric
prometheus.MustRegister(gaugeMetrics[topic])
// add the first value
value := parseValue(payload)
gaugeMetrics[topic].Set(value)
}
}
func parseTopic(topic string) string {
name := strings.Replace(topic, "$SYS/", "", 1)
name = strings.Replace(name, "/", "_", -1)
name = strings.Replace(name, " ", "_", -1)
name = strings.Replace(name, "-", "_", -1)
name = strings.Replace(name, ".", "_", -1)
return name
}
func parseValue(payload string) float64 {
// fmt.Printf("Payload %s \n", payload)
var validValue = regexp.MustCompile(`-?\d{1,}[.]\d{1,}|\d{1,}`)
// get the first value of the string
strArray := validValue.FindAllString(payload, 1)
if len(strArray) > 0 {
// parse to float
value, err := strconv.ParseFloat(strArray[0], 64)
if err == nil {
return value
}
}
return 0
}
func fatalfOnError(err error, msg string, args ...interface{}) {
if err != nil {
log.Fatalf(msg, args...)
}
}