mirror of
https://github.com/prometheus/statsd_exporter.git
synced 2024-11-26 09:11:01 +00:00
Implement config watching and reloading.
This commit is contained in:
parent
fe0359ec09
commit
eff54bdcf9
5 changed files with 97 additions and 28 deletions
11
bridge.go
11
bridge.go
|
@ -298,14 +298,3 @@ func (l *StatsDListener) handlePacket(packet []byte, e chan<- Events) {
|
||||||
}
|
}
|
||||||
e <- events
|
e <- events
|
||||||
}
|
}
|
||||||
|
|
||||||
var (
|
|
||||||
eventStats = prometheus.NewCounter()
|
|
||||||
networkStats = prometheus.NewCounter()
|
|
||||||
)
|
|
||||||
|
|
||||||
func init() {
|
|
||||||
prometheus.Register("statsd_bridge_events_total", "The total number of StatsD events seen.", prometheus.NilLabels, eventStats)
|
|
||||||
prometheus.Register("statsd_bridge_packets_total", "The total number of StatsD packets seen.", prometheus.NilLabels, networkStats)
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
43
main.go
43
main.go
|
@ -14,6 +14,7 @@ import (
|
||||||
"strconv"
|
"strconv"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/howeyc/fsnotify"
|
||||||
"github.com/prometheus/client_golang/prometheus"
|
"github.com/prometheus/client_golang/prometheus"
|
||||||
"github.com/prometheus/client_golang/prometheus/exp"
|
"github.com/prometheus/client_golang/prometheus/exp"
|
||||||
)
|
)
|
||||||
|
@ -21,7 +22,7 @@ import (
|
||||||
var (
|
var (
|
||||||
listeningAddress = flag.String("listeningAddress", ":8080", "The address on which to expose generated Prometheus metrics.")
|
listeningAddress = flag.String("listeningAddress", ":8080", "The address on which to expose generated Prometheus metrics.")
|
||||||
statsdListeningAddress = flag.String("statsdListeningAddress", ":9125", "The UDP address on which to receive statsd metric lines.")
|
statsdListeningAddress = flag.String("statsdListeningAddress", ":9125", "The UDP address on which to receive statsd metric lines.")
|
||||||
mappingConfig = flag.String("mappingConfig", "mapping.conf", "Metric mapping configuration file name.")
|
mappingConfig = flag.String("mappingConfig", "", "Metric mapping configuration file name.")
|
||||||
summaryFlushInterval = flag.Duration("summaryFlushInterval", 15*time.Minute, "How frequently to reset all summary metrics.")
|
summaryFlushInterval = flag.Duration("summaryFlushInterval", 15*time.Minute, "How frequently to reset all summary metrics.")
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -56,6 +57,39 @@ func udpAddrFromString(addr string) *net.UDPAddr {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func watchConfig(fileName string, mapper *metricMapper) {
|
||||||
|
watcher, err := fsnotify.NewWatcher()
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
err = watcher.WatchFlags(fileName, fsnotify.FSN_MODIFY)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case ev := <-watcher.Event:
|
||||||
|
log.Printf("Config file changed (%s), attempting reload", ev)
|
||||||
|
err = mapper.initFromFile(fileName)
|
||||||
|
if err != nil {
|
||||||
|
log.Println("Error reloading config:", err)
|
||||||
|
configLoads.Increment(map[string]string{"outcome": "failure"})
|
||||||
|
} else {
|
||||||
|
log.Println("Config reloaded successfully")
|
||||||
|
configLoads.Increment(map[string]string{"outcome": "success"})
|
||||||
|
}
|
||||||
|
// Re-add the file watcher since it can get lost on some changes. E.g.
|
||||||
|
// saving a file with vim results in a RENAME-MODIFY-DELETE event
|
||||||
|
// sequence, after which the newly written file is no longer watched.
|
||||||
|
err = watcher.WatchFlags(fileName, fsnotify.FSN_MODIFY)
|
||||||
|
case err := <-watcher.Error:
|
||||||
|
log.Println("Error watching config:", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
flag.Parse()
|
flag.Parse()
|
||||||
|
|
||||||
|
@ -76,14 +110,15 @@ func main() {
|
||||||
l := &StatsDListener{conn: conn}
|
l := &StatsDListener{conn: conn}
|
||||||
go l.Listen(events)
|
go l.Listen(events)
|
||||||
|
|
||||||
mapper := metricMapper{}
|
mapper := &metricMapper{}
|
||||||
if mappingConfig != nil {
|
if *mappingConfig != "" {
|
||||||
err := mapper.initFromFile(*mappingConfig)
|
err := mapper.initFromFile(*mappingConfig)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal("Error loading config:", err)
|
log.Fatal("Error loading config:", err)
|
||||||
}
|
}
|
||||||
|
go watchConfig(*mappingConfig, mapper)
|
||||||
}
|
}
|
||||||
bridge := NewBridge(&mapper)
|
bridge := NewBridge(mapper)
|
||||||
go func() {
|
go func() {
|
||||||
for _ = range time.Tick(*summaryFlushInterval) {
|
for _ = range time.Tick(*summaryFlushInterval) {
|
||||||
bridge.Summaries.Flush()
|
bridge.Summaries.Flush()
|
||||||
|
|
43
mapper.go
43
mapper.go
|
@ -11,6 +11,9 @@ import (
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"regexp"
|
"regexp"
|
||||||
"strings"
|
"strings"
|
||||||
|
"sync"
|
||||||
|
|
||||||
|
"github.com/prometheus/client_golang/prometheus"
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
|
@ -26,6 +29,7 @@ type metricMapping struct {
|
||||||
|
|
||||||
type metricMapper struct {
|
type metricMapper struct {
|
||||||
mappings []metricMapping
|
mappings []metricMapping
|
||||||
|
mutex sync.Mutex
|
||||||
}
|
}
|
||||||
|
|
||||||
type configLoadStates int
|
type configLoadStates int
|
||||||
|
@ -35,11 +39,12 @@ const (
|
||||||
METRIC_DEFINITION
|
METRIC_DEFINITION
|
||||||
)
|
)
|
||||||
|
|
||||||
func (l *metricMapper) initFromString(fileContents string) error {
|
func (m *metricMapper) initFromString(fileContents string) error {
|
||||||
lines := strings.Split(fileContents, "\n")
|
lines := strings.Split(fileContents, "\n")
|
||||||
state := SEARCHING
|
state := SEARCHING
|
||||||
|
|
||||||
mapping := metricMapping{labels: map[string]string{}}
|
parsedMappings := []metricMapping{}
|
||||||
|
currentMapping := metricMapping{labels: map[string]string{}}
|
||||||
for i, line := range lines {
|
for i, line := range lines {
|
||||||
line := strings.TrimSpace(line)
|
line := strings.TrimSpace(line)
|
||||||
|
|
||||||
|
@ -51,24 +56,28 @@ func (l *metricMapper) initFromString(fileContents string) error {
|
||||||
if !metricLineRE.MatchString(line) {
|
if !metricLineRE.MatchString(line) {
|
||||||
return fmt.Errorf("Line %d: expected metric match line, got: %s", i, line)
|
return fmt.Errorf("Line %d: expected metric match line, got: %s", i, line)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Translate the glob-style metric match line into a proper regex that we
|
||||||
|
// can use to match metrics later on.
|
||||||
metricRe := strings.Replace(line, ".", "\\.", -1)
|
metricRe := strings.Replace(line, ".", "\\.", -1)
|
||||||
metricRe = strings.Replace(metricRe, "*", "([^.]+)", -1)
|
metricRe = strings.Replace(metricRe, "*", "([^.]+)", -1)
|
||||||
mapping.regex = regexp.MustCompile("^" + metricRe + "$")
|
currentMapping.regex = regexp.MustCompile("^" + metricRe + "$")
|
||||||
|
|
||||||
state = METRIC_DEFINITION
|
state = METRIC_DEFINITION
|
||||||
|
|
||||||
case METRIC_DEFINITION:
|
case METRIC_DEFINITION:
|
||||||
if line == "" {
|
if line == "" {
|
||||||
if len(mapping.labels) == 0 {
|
if len(currentMapping.labels) == 0 {
|
||||||
return fmt.Errorf("Line %d: metric mapping didn't set any labels", i)
|
return fmt.Errorf("Line %d: metric mapping didn't set any labels", i)
|
||||||
}
|
}
|
||||||
if _, ok := mapping.labels["name"]; !ok {
|
if _, ok := currentMapping.labels["name"]; !ok {
|
||||||
return fmt.Errorf("Line %d: metric mapping didn't set a metric name", i)
|
return fmt.Errorf("Line %d: metric mapping didn't set a metric name", i)
|
||||||
}
|
}
|
||||||
|
|
||||||
l.mappings = append(l.mappings, mapping)
|
parsedMappings = append(parsedMappings, currentMapping)
|
||||||
|
|
||||||
state = SEARCHING
|
state = SEARCHING
|
||||||
mapping = metricMapping{labels: map[string]string{}}
|
currentMapping = metricMapping{labels: map[string]string{}}
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -76,24 +85,34 @@ func (l *metricMapper) initFromString(fileContents string) error {
|
||||||
if len(matches) != 3 {
|
if len(matches) != 3 {
|
||||||
return fmt.Errorf("Line %d: expected label mapping line, got: %s", i, line)
|
return fmt.Errorf("Line %d: expected label mapping line, got: %s", i, line)
|
||||||
}
|
}
|
||||||
mapping.labels[matches[1]] = matches[2]
|
currentMapping.labels[matches[1]] = matches[2]
|
||||||
default:
|
default:
|
||||||
panic("illegal state")
|
panic("illegal state")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
m.mutex.Lock()
|
||||||
|
defer m.mutex.Unlock()
|
||||||
|
m.mappings = parsedMappings
|
||||||
|
|
||||||
|
mappingsCount.Set(prometheus.NilLabels, float64(len(parsedMappings)))
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (l *metricMapper) initFromFile(fileName string) error {
|
func (m *metricMapper) initFromFile(fileName string) error {
|
||||||
mappingStr, err := ioutil.ReadFile(fileName)
|
mappingStr, err := ioutil.ReadFile(fileName)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
return l.initFromString(string(mappingStr))
|
return m.initFromString(string(mappingStr))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (l *metricMapper) getMapping(statsdMetric string) (labels map[string]string, present bool) {
|
func (m *metricMapper) getMapping(statsdMetric string) (labels map[string]string, present bool) {
|
||||||
for _, mapping := range l.mappings {
|
m.mutex.Lock()
|
||||||
|
defer m.mutex.Unlock()
|
||||||
|
|
||||||
|
for _, mapping := range m.mappings {
|
||||||
matches := mapping.regex.FindStringSubmatchIndex(statsdMetric)
|
matches := mapping.regex.FindStringSubmatchIndex(statsdMetric)
|
||||||
if len(matches) == 0 {
|
if len(matches) == 0 {
|
||||||
continue
|
continue
|
||||||
|
|
|
@ -71,8 +71,8 @@ func TestMetricMapper(t *testing.T) {
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
|
mapper := metricMapper{}
|
||||||
for i, scenario := range scenarios {
|
for i, scenario := range scenarios {
|
||||||
mapper := metricMapper{}
|
|
||||||
err := mapper.initFromString(scenario.config)
|
err := mapper.initFromString(scenario.config)
|
||||||
if err != nil && !scenario.configBad {
|
if err != nil && !scenario.configBad {
|
||||||
t.Fatalf("%d. Config load error: %s", i, err)
|
t.Fatalf("%d. Config load error: %s", i, err)
|
||||||
|
|
26
telemetry.go
Normal file
26
telemetry.go
Normal file
|
@ -0,0 +1,26 @@
|
||||||
|
// Copyright (c) 2013, Prometheus Team
|
||||||
|
// All rights reserved.
|
||||||
|
//
|
||||||
|
// Use of this source code is governed by a BSD-style
|
||||||
|
// license that can be found in the LICENSE file.
|
||||||
|
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/prometheus/client_golang/prometheus"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
eventStats = prometheus.NewCounter()
|
||||||
|
networkStats = prometheus.NewCounter()
|
||||||
|
configLoads = prometheus.NewCounter()
|
||||||
|
mappingsCount = prometheus.NewGauge()
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
prometheus.Register("statsd_bridge_events_count", "The total number of StatsD events seen.", prometheus.NilLabels, eventStats)
|
||||||
|
prometheus.Register("statsd_bridge_packets_count", "The total number of StatsD packets seen.", prometheus.NilLabels, networkStats)
|
||||||
|
prometheus.Register("statsd_bridge_config_reloads_count", "The number of configuration reloads.", prometheus.NilLabels, configLoads)
|
||||||
|
prometheus.Register("statsd_bridge_loaded_mappings_count", "The number of configured metric mappings.", prometheus.NilLabels, mappingsCount)
|
||||||
|
|
||||||
|
}
|
Loading…
Reference in a new issue