Skip to content
This repository was archived by the owner on Sep 1, 2024. It is now read-only.

Commit a694eb7

Browse files
committed
feat(metric): Add a replication monitoring metric
This commit adds a new metric to monitor replication status. A new flag as been added when launching the exporter. It changes the way metrics are registered in the prometheus library to allow more flexibility on those (bring your own metric parser). Signed-off-by: Julien Godin <[email protected]>
1 parent c1d1053 commit a694eb7

File tree

2 files changed

+123
-34
lines changed

2 files changed

+123
-34
lines changed

cmd/openldap_exporter/main.go

+18-11
Original file line numberDiff line numberDiff line change
@@ -16,16 +16,17 @@ import (
1616
)
1717

1818
const (
19-
promAddr = "promAddr"
20-
ldapNet = "ldapNet"
21-
ldapAddr = "ldapAddr"
22-
ldapUser = "ldapUser"
23-
ldapPass = "ldapPass"
24-
interval = "interval"
25-
metrics = "metrPath"
26-
jsonLog = "jsonLog"
27-
webCfgFile = "webCfgFile"
28-
config = "config"
19+
promAddr = "promAddr"
20+
ldapNet = "ldapNet"
21+
ldapAddr = "ldapAddr"
22+
ldapUser = "ldapUser"
23+
ldapPass = "ldapPass"
24+
interval = "interval"
25+
metrics = "metrPath"
26+
jsonLog = "jsonLog"
27+
webCfgFile = "webCfgFile"
28+
config = "config"
29+
replicationObject = "replicationObject"
2930
)
3031

3132
func main() {
@@ -51,7 +52,7 @@ func main() {
5152
altsrc.NewStringFlag(&cli.StringFlag{
5253
Name: ldapAddr,
5354
Value: "localhost:389",
54-
Usage: "Address of OpenLDAP server",
55+
Usage: "Address and port of OpenLDAP server",
5556
EnvVars: []string{"LDAP_ADDR"},
5657
}),
5758
altsrc.NewStringFlag(&cli.StringFlag{
@@ -64,6 +65,7 @@ func main() {
6465
Usage: "OpenLDAP bind password (optional)",
6566
EnvVars: []string{"LDAP_PASS"},
6667
}),
68+
6769
altsrc.NewDurationFlag(&cli.DurationFlag{
6870
Name: interval,
6971
Value: 30 * time.Second,
@@ -81,6 +83,10 @@ func main() {
8183
Usage: "Output logs in JSON format",
8284
EnvVars: []string{"JSON_LOG"},
8385
}),
86+
altsrc.NewStringSliceFlag(&cli.StringSliceFlag{
87+
Name: replicationObject,
88+
Usage: "Object to watch replication uppon",
89+
}),
8490
&cli.StringFlag{
8591
Name: config,
8692
Usage: "Optional configuration from a `YAML_FILE`",
@@ -131,6 +137,7 @@ func runMain(c *cli.Context) error {
131137
User: c.String(ldapUser),
132138
Pass: c.String(ldapPass),
133139
Tick: c.Duration(interval),
140+
Sync: c.StringSlice(replicationObject),
134141
}
135142

136143
ctx, cancel := context.WithCancel(context.Background())

scraper.go

+105-23
Original file line numberDiff line numberDiff line change
@@ -24,13 +24,17 @@ const (
2424

2525
monitorOperation = "monitorOperation"
2626
monitorOpCompleted = "monitorOpCompleted"
27+
28+
monitorReplicationFilter = "contextcsn"
29+
monitorReplication = "monitorReplication"
2730
)
2831

2932
type query struct {
3033
baseDN string
3134
searchFilter string
3235
searchAttr string
3336
metric *prometheus.GaugeVec
37+
setData func([]*ldap.Entry, *query) error
3438
}
3539

3640
var (
@@ -66,24 +70,50 @@ var (
6670
},
6771
[]string{"result"},
6872
)
73+
monitorReplicationGauge = prometheus.NewGaugeVec(
74+
prometheus.GaugeOpts{
75+
Subsystem: "openldap",
76+
Name: "monitor_replication",
77+
Help: help(baseDN, monitorReplication),
78+
},
79+
[]string{"id", "type"},
80+
)
6981
queries = []*query{
7082
{
7183
baseDN: baseDN,
7284
searchFilter: objectClass(monitoredObject),
7385
searchAttr: monitoredInfo,
7486
metric: monitoredObjectGauge,
75-
},
76-
{
87+
setData: func(entries []*ldap.Entry, q *query) error {
88+
return setValue(entries, q)
89+
90+
},
91+
}, {
7792
baseDN: baseDN,
7893
searchFilter: objectClass(monitorCounterObject),
7994
searchAttr: monitorCounter,
8095
metric: monitorCounterObjectGauge,
96+
setData: func(entries []*ldap.Entry, q *query) error {
97+
return setValue(entries, q)
98+
},
99+
},
100+
{
101+
baseDN: opsBaseDN,
102+
searchFilter: objectClass(monitorOperation),
103+
searchAttr: monitorOpCompleted,
104+
metric: monitorOperationGauge,
105+
setData: func(entries []*ldap.Entry, q *query) error {
106+
return setValue(entries, q)
107+
},
81108
},
82109
{
83110
baseDN: opsBaseDN,
84111
searchFilter: objectClass(monitorOperation),
85112
searchAttr: monitorOpCompleted,
86113
metric: monitorOperationGauge,
114+
setData: func(entries []*ldap.Entry, q *query) error {
115+
return setValue(entries, q)
116+
},
87117
},
88118
}
89119
)
@@ -93,6 +123,7 @@ func init() {
93123
monitoredObjectGauge,
94124
monitorCounterObjectGauge,
95125
monitorOperationGauge,
126+
monitorReplicationGauge,
96127
scrapeCounter,
97128
)
98129
}
@@ -105,17 +136,81 @@ func objectClass(name string) string {
105136
return fmt.Sprintf("(objectClass=%v)", name)
106137
}
107138

139+
func setValue(entries []*ldap.Entry, q *query) error {
140+
for _, entry := range entries {
141+
val := entry.GetAttributeValue(q.searchAttr)
142+
if val == "" {
143+
// not every entry will have this attribute
144+
continue
145+
}
146+
num, err := strconv.ParseFloat(val, 64)
147+
if err != nil {
148+
// some of these attributes are not numbers
149+
continue
150+
}
151+
q.metric.WithLabelValues(entry.DN).Set(num)
152+
}
153+
return nil
154+
155+
}
156+
108157
type Scraper struct {
109-
Net string
110-
Addr string
111-
User string
112-
Pass string
113-
Tick time.Duration
114-
log log.FieldLogger
158+
Net string
159+
Addr string
160+
User string
161+
Pass string
162+
Tick time.Duration
163+
LdapSync []string
164+
log log.FieldLogger
165+
Sync []string
166+
}
167+
168+
func (s *Scraper) addReplicationQueries() error {
169+
if len(s.Sync) != 0 {
170+
for _, q := range s.Sync {
171+
queries = append(queries,
172+
&query{
173+
baseDN: q,
174+
searchFilter: objectClass("*"),
175+
searchAttr: "contextCSN",
176+
metric: monitorReplicationGauge,
177+
setData: func(entries []*ldap.Entry, q *query) error {
178+
for _, entry := range entries {
179+
val := entry.GetAttributeValue(q.searchAttr)
180+
if val == "" {
181+
// not every entry will have this attribute
182+
continue
183+
}
184+
valueBuffer := strings.Split(val, "#")
185+
gt, err := time.Parse("20060102150405.999999Z", valueBuffer[0])
186+
if err != nil {
187+
return err
188+
}
189+
count, err := strconv.ParseFloat(valueBuffer[1], 64)
190+
if err != nil {
191+
return err
192+
}
193+
sid := valueBuffer[2]
194+
mod, err := strconv.ParseFloat(valueBuffer[3], 64)
195+
if err != nil {
196+
return err
197+
}
198+
q.metric.WithLabelValues(sid, "gt").Set(float64(gt.Unix()))
199+
q.metric.WithLabelValues(sid, "count").Set(count)
200+
q.metric.WithLabelValues(sid, "mod").Set(mod)
201+
}
202+
return nil
203+
},
204+
},
205+
)
206+
}
207+
}
208+
return nil
115209
}
116210

117211
func (s *Scraper) Start(ctx context.Context) error {
118212
s.log = log.WithField("component", "scraper")
213+
s.addReplicationQueries()
119214
address := fmt.Sprintf("%s://%s", s.Net, s.Addr)
120215
s.log.WithField("addr", address).Info("starting monitor loop")
121216
ticker := time.NewTicker(s.Tick)
@@ -167,24 +262,11 @@ func (s *Scraper) scrape() bool {
167262
func scrapeQuery(conn *ldap.Conn, q *query) error {
168263
req := ldap.NewSearchRequest(
169264
q.baseDN, ldap.ScopeWholeSubtree, ldap.NeverDerefAliases, 0, 0, false,
170-
q.searchFilter, []string{"dn", q.searchAttr}, nil,
265+
q.searchFilter, []string{q.searchAttr}, nil,
171266
)
172267
sr, err := conn.Search(req)
173268
if err != nil {
174269
return err
175270
}
176-
for _, entry := range sr.Entries {
177-
val := entry.GetAttributeValue(q.searchAttr)
178-
if val == "" {
179-
// not every entry will have this attribute
180-
continue
181-
}
182-
num, err := strconv.ParseFloat(val, 64)
183-
if err != nil {
184-
// some of these attributes are not numbers
185-
continue
186-
}
187-
q.metric.WithLabelValues(entry.DN).Set(num)
188-
}
189-
return nil
271+
return q.setData(sr.Entries, q)
190272
}

0 commit comments

Comments
 (0)