1
1
package main
2
2
3
3
import (
4
- "bytes"
5
4
"encoding/json"
6
5
"fmt"
7
- "io"
8
6
"net/http"
9
7
"strconv"
10
8
"strings"
35
33
"node_stats" : "/api/v3/nodes/%s/stats/" ,
36
34
"nodes" : "/api/v3/nodes/%s" ,
37
35
}
36
+
37
+ //GitTag stands for a git tag, populated at build time
38
+ GitTag string
39
+ //GitCommit stands for a git commit hash populated at build time
40
+ GitCommit string
38
41
)
39
42
40
43
type metric struct {
@@ -49,11 +52,11 @@ type emqResponse struct {
49
52
Data map [string ]interface {} `json:"data,omitempty"` //api v3 json key
50
53
}
51
54
52
- // Exporter collects EMQ stats from the given URI and exports them using
55
+ // Exporter collects EMQ stats from the given host and exports them using
53
56
// the prometheus metrics package.
54
57
type Exporter struct {
55
- URI string
56
- client http.Client
58
+ Host string
59
+ client * http.Client
57
60
username , password , node string
58
61
up prometheus.Gauge
59
62
totalScrapes prometheus.Counter
@@ -63,15 +66,15 @@ type Exporter struct {
63
66
}
64
67
65
68
// NewExporter returns an initialized Exporter.
66
- func NewExporter (uri , username , password , node string , timeout time.Duration , apiVersion string ) ( * Exporter , error ) {
69
+ func NewExporter (uri , username , password , node , apiVersion string , timeout time.Duration ) * Exporter {
67
70
68
71
return & Exporter {
69
- URI : uri ,
72
+ Host : uri ,
70
73
username : username ,
71
74
password : password ,
72
75
node : node ,
73
76
apiVersion : apiVersion ,
74
- client : http.Client {
77
+ client : & http.Client {
75
78
Timeout : timeout ,
76
79
},
77
80
up : prometheus .NewGauge (prometheus.GaugeOpts {
@@ -84,7 +87,7 @@ func NewExporter(uri, username, password, node string, timeout time.Duration, ap
84
87
Name : "exporter_total_scrapes" ,
85
88
Help : "Current total scrapes." ,
86
89
}),
87
- }, nil
90
+ }
88
91
89
92
}
90
93
@@ -138,19 +141,19 @@ func (e *Exporter) scrape() error {
138
141
139
142
for name , path := range targets {
140
143
141
- resp , err := e .fetch (path )
144
+ res , err := e .fetch (path )
142
145
if err != nil {
143
146
return err
144
147
}
145
148
146
- if resp .Code != 0 {
149
+ if res .Code != 0 {
147
150
return fmt .Errorf ("Received code != 0" )
148
151
}
149
152
150
153
if e .apiVersion == "v2" {
151
- data = resp .Result
154
+ data = res .Result
152
155
} else {
153
- data = resp .Data
156
+ data = res .Data
154
157
}
155
158
156
159
for k , v := range data {
@@ -192,44 +195,45 @@ func (e *Exporter) addMetric(fqName, help string, value float64, labels []string
192
195
}
193
196
194
197
//get the response from the provided target url
195
- func (e * Exporter ) fetch (target string ) (emqResponse , error ) {
196
- var dat emqResponse
198
+ func (e * Exporter ) fetch (target string ) (* emqResponse , error ) {
199
+
200
+ dat := & emqResponse {}
197
201
198
- u := e .URI + fmt .Sprintf (target , e .node )
202
+ u := e .Host + fmt .Sprintf (target , e .node )
199
203
200
- req , err := http .NewRequest ("GET" , u , nil )
204
+ log .Debugln ("fetching from" , u )
205
+
206
+ req , err := http .NewRequest (http .MethodGet , u , nil )
201
207
if err != nil {
202
- return dat , fmt .Errorf ("Failed to get metrics from %s " , u )
208
+ return dat , fmt .Errorf ("Failed to create http request: %v " , err )
203
209
}
204
210
205
211
req .SetBasicAuth (e .username , e .password )
212
+ req .Header .Set ("Accept" , "application/json" )
213
+ req .Header .Set ("User-Agent" , "emp_exporter/" + GitTag )
214
+
206
215
res , err := e .client .Do (req )
207
216
if err != nil {
208
- return dat , fmt .Errorf ("Failed to get metrics from %s" , u )
217
+ return dat , fmt .Errorf ("Failed to get metrics from %s: %v " , u , err )
209
218
}
210
219
defer res .Body .Close ()
211
220
212
221
if res .StatusCode != http .StatusOK {
213
- return dat , fmt .Errorf ("Failed to get metrics from %s" , u )
222
+ return dat , fmt .Errorf ("Received status code not ok %s" , u )
214
223
}
215
224
216
- if err := json .Unmarshal ( streamToByte ( res .Body ), & dat ); err != nil {
217
- return dat , fmt .Errorf ("Failed to unmarshal json" )
225
+ if err := json .NewDecoder ( res .Body ). Decode ( dat ); err != nil {
226
+ return dat , fmt .Errorf ("Error in json decoder %v" , err )
218
227
}
219
228
220
- return dat , nil
221
- }
229
+ //Print the returned response data for debuging
230
+ log . Debugf ( "%#v" , * dat )
222
231
223
- //Convert a stream into byte array
224
- func streamToByte (stream io.Reader ) []byte {
225
- buf := new (bytes.Buffer )
226
- buf .ReadFrom (stream )
227
- return buf .Bytes ()
232
+ return dat , nil
228
233
}
229
234
230
235
//Try to parse value from string to float64, return error on failure
231
236
func parseString (s string ) (float64 , error ) {
232
-
233
237
v , err := strconv .ParseFloat (s , 64 )
234
238
235
239
if err != nil {
@@ -245,19 +249,11 @@ func parseString(s string) (float64, error) {
245
249
return v , nil
246
250
}
247
251
248
- var (
249
- // GitTag stands for a git tag, populated at build time
250
- GitTag string
251
- // GitCommit stands for a git commit hash populated at build time
252
- GitCommit string
253
- )
254
-
255
252
func main () {
256
-
257
253
var (
258
254
listenAddress = kingpin .Flag ("web.listen-address" , "Address to listen on for web interface and telemetry." ).Default (":9505" ).String ()
259
255
metricsPath = kingpin .Flag ("web.telemetry-path" , "Path under which to expose metrics." ).Default ("/metrics" ).String ()
260
- emqURI = kingpin .Flag ("emq.uri" , "HTTP API address of the EMQ node." ).Default ("http://127.0.0.1:8080 " ).String ()
256
+ emqURI = kingpin .Flag ("emq.uri" , "HTTP API address of the EMQ node." ).Default ("http://127.0.0.1:18083 " ).String ()
261
257
emqUsername = kingpin .Flag ("emq.username" , "EMQ username (or use $EMQ_USERNAME env var)" ).Default ("admin" ).Envar ("EMQ_USERNAME" ).String ()
262
258
emqPassword = kingpin .Flag ("emq.password" , "EMQ password (or use $EMQ_PASSWORD env var)" ).Default ("public" ).Envar ("EMQ_PASSWORD" ).String ()
263
259
emqNodeName = kingpin .
Flag (
"emq.node" ,
"Node name of the emq node to scrape." ).
Default (
"[email protected] " ).
String ()
@@ -272,12 +268,9 @@ func main() {
272
268
kingpin .Parse ()
273
269
274
270
log .Infoln ("Starting emq_exporter" )
275
- log .Infoln ( fmt . Sprintf ( "Version %s (git-%s)" , GitTag , GitCommit ) )
271
+ log .Infof ( "Version %s (git-%s)\n " , GitTag , GitCommit )
276
272
277
- exporter , err := NewExporter (* emqURI , * emqUsername , * emqPassword , * emqNodeName , * emqTimeout , * emqAPIVersion )
278
- if err != nil {
279
- log .Fatal (err )
280
- }
273
+ exporter := NewExporter (* emqURI , * emqUsername , * emqPassword , * emqNodeName , * emqAPIVersion , * emqTimeout )
281
274
282
275
prometheus .MustRegister (exporter )
283
276
0 commit comments