This repository has been archived by the owner on Sep 20, 2022. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 47
/
gather.go
260 lines (214 loc) · 7.57 KB
/
gather.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
package main
import (
"encoding/json"
"net/http"
"strconv"
"strings"
"time"
"github.com/infinityworks/prometheus-rancher-exporter/measure"
"github.com/prometheus/client_golang/prometheus"
)
// Data is used to store data from all the relevant endpoints in the API
type Data struct {
Data []struct {
HealthState string `json:"healthState"`
Name string `json:"name"`
State string `json:"state"`
System bool `json:"system"`
Scale int `json:"scale"`
HostName string `json:"hostname"`
ID string `json:"id"`
StackID string `json:"stackId"`
EnvID string `json:"environmentId"`
BaseType string `json:"basetype"`
Type string `json:"type"`
AgentState string `json:"agentState"`
Labels map[string]string `json:"labels"`
ClusterID string `json:"clusterId"`
NodeName string `json:"nodeName"`
// HostInfo for hosts
HostInfo *HostInfo `json:"info"`
// LaunchConfig for services
LaunchConfig *LaunchConfig `json:"launchConfig"`
// ComponentStatuses for clusters
ComponentStatuses []*ComponentStatuses `json:"componentStatuses"`
} `json:"data"`
}
type HostInfo struct {
CPUInfo struct {
Count int `json:"count"`
} `json:"cpuInfo"`
MemoryInfo struct {
MemTotal int `json:"memTotal"`
MemFree int `json:"memFree"`
} `json:"memoryInfo"`
DiskInfo struct {
MountPoints map[string]MountPoint `json:"mountPoints"`
} `json:"diskInfo"`
}
type MountPoint struct {
Total int `json:"total"`
Used int `json:"used"`
}
type LaunchConfig struct {
Labels map[string]string `json:"labels"`
}
type ComponentStatuses struct {
Conditions []*Condition `json:"conditions"`
Name string `json:"name"`
}
type Condition struct {
Status string `json:"status"`
}
// processMetrics - Collects the data from the API, returns data object
func (e *Exporter) processMetrics(data *Data, endpoint string, hideSys bool, ch chan<- prometheus.Metric) error {
var filteredLabels map[string]string
// Metrics - range through the data object
for _, x := range data.Data {
// If system services have been ignored, the loop simply skips them
if hideSys && x.System {
continue
}
// Checks the metric is of the expected type
dataType := x.BaseType
if dataType == "" {
dataType = x.Type
}
if !checkMetric(endpoint, dataType) {
continue
}
log.Debugf("Processing metrics for %s", endpoint)
if endpoint == "hosts" {
filteredLabels = e.allowedLabels(x.Labels)
var s = x.HostName
if x.Name != "" {
s = x.Name
}
e.setHostStateMetrics(s, x.State, x.AgentState, filteredLabels)
if x.HostInfo != nil {
e.setHostInfoMetrics(s, x.HostInfo, filteredLabels)
}
} else if endpoint == "stacks" {
// Used to create a map of stackID and stackName
// Later used as a dimension in service metrics
stackRef = storeStackRef(x.ID, x.Name)
e.setStackMetrics(x.Name, x.State, x.HealthState, strconv.FormatBool(x.System))
} else if endpoint == "services" {
// Retrieves the stack Name from the previous values stored.
var stackName = retrieveStackRef(x.StackID)
if stackName == "unknown" {
log.Warnf("Failed to obtain stack_name for %s from the API", x.Name)
}
if x.LaunchConfig != nil && len(x.LaunchConfig.Labels) > 0 {
filteredLabels = e.allowedLabels(x.LaunchConfig.Labels)
}
e.setServiceMetrics(x.Name, stackName, x.State, x.HealthState, x.Scale, filteredLabels)
} else if endpoint == "clusters" {
clusterRef = storeClusterRef(x.ID, x.Name)
e.setClusterMetrics(x.Name, x.State, x.ComponentStatuses)
} else if endpoint == "nodes" {
// Retrieves the cluster Name from the previous values stored.
var clusterName = retrieveClusterRef(x.ClusterID)
if clusterName == "unknown" {
log.Warnf("Failed to obtain cluster_name for %s from the API", x.NodeName)
}
e.setNodeMetrics(x.NodeName, x.State, clusterName)
}
}
return nil
}
// gatherData - Collects the data from thw API, invokes functions to transform that data into metrics
func (e *Exporter) gatherData(rancherURL string, resourceLimit string, accessKey string, secretKey string, endpoint string, ch chan<- prometheus.Metric) (*Data, error) {
// Return the correct URL path
url := setEndpoint(rancherURL, endpoint, resourceLimit)
// Create new data slice from Struct
var data = new(Data)
// Scrape EndPoint for JSON Data
err := getJSON(url, accessKey, secretKey, &data)
if err != nil {
log.Error("Error getting JSON from endpoint ", endpoint)
return nil, err
}
log.Debugf("JSON Fetched for: "+endpoint+": %+v", data)
return data, err
}
func (e *Exporter) allowedLabels(labels map[string]string) map[string]string {
result := make(map[string]string)
for name, val := range labels {
if e.labelsFilter.MatchString(name) {
result[name] = val
}
}
return result
}
// getJSON return json from server, return the formatted JSON
func getJSON(url string, accessKey string, secretKey string, target interface{}) error {
start := time.Now()
// Counter for internal exporter metrics
measure.FunctionCountTotal.With(prometheus.Labels{"pkg": "main", "fnc": "getJSON"}).Inc()
log.Info("Scraping: ", url)
client := &http.Client{}
req, err := http.NewRequest("GET", url, nil)
if err != nil {
log.Error("Error Collecting JSON from API: ", err)
}
req.SetBasicAuth(accessKey, secretKey)
resp, err := client.Do(req)
if err != nil {
log.Error("Error Collecting JSON from API: ", err)
}
if resp.StatusCode != 200 {
log.Error("Error Collecting JSON from API: ", resp.Status)
}
respFormatted := json.NewDecoder(resp.Body).Decode(target)
// Timings recorded as part of internal metrics
elapsed := float64((time.Since(start)) / time.Microsecond)
measure.FunctionDurations.WithLabelValues("main", "getJSON").Observe(elapsed)
// Close the response body, the underlying Transport should then close the connection.
resp.Body.Close()
// return formatted JSON
return respFormatted
}
// setEndpoint - Determines the correct URL endpoint to use, gives us backwards compatibility
func setEndpoint(rancherURL string, component string, resourceLimit string) string {
var endpoint string
endpoint = (rancherURL + "/" + component + "/" + "?limit=" + resourceLimit)
endpoint = strings.Replace(endpoint, "v1", "v2-beta", 1)
return endpoint
}
// storeStackRef stores the stackID and stack name for use as a label elsewhere
func storeStackRef(stackID string, stackName string) map[string]string {
stackRef[stackID] = stackName
return stackRef
}
// retrieveStackRef returns the stack name, when sending the stackID
func retrieveStackRef(stackID string) string {
for key, value := range stackRef {
if stackID == "" {
return "unknown"
} else if stackID == key {
log.Debugf("StackRef - Key is %s, Value is %s StackID is %s", key, value, stackID)
return value
}
}
// returns unknown if no match was found
return "unknown"
}
// storeClusterRef stores the clusterID and cluster name for use as a label elsewhere
func storeClusterRef(clusterID string, clusterName string) map[string]string {
clusterRef[clusterID] = clusterName
return clusterRef
}
// retrieveClusterRef returns the cluster name, when sending the clusterID
func retrieveClusterRef(clusterID string) string {
for key, value := range clusterRef {
if clusterID == "" {
return "unknown"
} else if clusterID == key {
log.Debugf("ClusterRef - Key is %s, Value is %s ClusterID is %s", key, value, clusterID)
return value
}
}
// returns unknown if no match was found
return "unknown"
}