2013-07-02 16:16:39 +00:00
|
|
|
// 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 (
|
|
|
|
"flag"
|
|
|
|
"log"
|
|
|
|
"net"
|
|
|
|
"net/http"
|
|
|
|
"strconv"
|
|
|
|
|
2013-07-08 18:37:12 +00:00
|
|
|
"github.com/howeyc/fsnotify"
|
2013-07-02 16:16:39 +00:00
|
|
|
"github.com/prometheus/client_golang/prometheus"
|
|
|
|
)
|
|
|
|
|
2013-07-02 18:10:01 +00:00
|
|
|
var (
|
|
|
|
listeningAddress = flag.String("listeningAddress", ":8080", "The address on which to expose generated Prometheus metrics.")
|
2013-07-05 21:53:15 +00:00
|
|
|
statsdListeningAddress = flag.String("statsdListeningAddress", ":9125", "The UDP address on which to receive statsd metric lines.")
|
2013-07-08 18:37:12 +00:00
|
|
|
mappingConfig = flag.String("mappingConfig", "", "Metric mapping configuration file name.")
|
2013-07-02 18:10:01 +00:00
|
|
|
)
|
2013-07-02 16:16:39 +00:00
|
|
|
|
|
|
|
func serveHTTP() {
|
2014-06-26 13:56:21 +00:00
|
|
|
http.Handle("/metrics", prometheus.Handler())
|
|
|
|
http.ListenAndServe(*listeningAddress, nil)
|
2013-07-02 16:16:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func udpAddrFromString(addr string) *net.UDPAddr {
|
|
|
|
host, portStr, err := net.SplitHostPort(*statsdListeningAddress)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal("Bad StatsD listening address", *statsdListeningAddress)
|
|
|
|
}
|
|
|
|
|
|
|
|
if host == "" {
|
|
|
|
host = "0.0.0.0"
|
|
|
|
}
|
|
|
|
ip, err := net.ResolveIPAddr("ip", host)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatalf("Unable to resolve %s: %s", host, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
port, err := strconv.Atoi(portStr)
|
|
|
|
if err != nil || port < 0 || port > 65535 {
|
|
|
|
log.Fatal("Bad port %s: %s", portStr, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return &net.UDPAddr{
|
|
|
|
IP: ip.IP,
|
|
|
|
Port: port,
|
|
|
|
Zone: ip.Zone,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-07-08 18:37:12 +00:00
|
|
|
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)
|
2014-06-26 13:56:21 +00:00
|
|
|
configLoads.WithLabelValues("failure").Inc()
|
2013-07-08 18:37:12 +00:00
|
|
|
} else {
|
|
|
|
log.Println("Config reloaded successfully")
|
2014-06-26 13:56:21 +00:00
|
|
|
configLoads.WithLabelValues("success").Inc()
|
2013-07-08 18:37:12 +00:00
|
|
|
}
|
|
|
|
// 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)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-07-02 16:16:39 +00:00
|
|
|
func main() {
|
|
|
|
flag.Parse()
|
|
|
|
|
2013-07-03 13:25:52 +00:00
|
|
|
log.Println("Starting StatsD -> Prometheus Bridge...")
|
|
|
|
log.Println("Accepting StatsD Traffic on", *statsdListeningAddress)
|
|
|
|
log.Println("Accepting Prometheus Requests on", *listeningAddress)
|
|
|
|
|
2013-07-02 16:16:39 +00:00
|
|
|
go serveHTTP()
|
|
|
|
|
|
|
|
events := make(chan Events, 1024)
|
|
|
|
defer close(events)
|
|
|
|
|
|
|
|
listenAddr := udpAddrFromString(*statsdListeningAddress)
|
|
|
|
conn, err := net.ListenUDP("udp", listenAddr)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
l := &StatsDListener{conn: conn}
|
|
|
|
go l.Listen(events)
|
|
|
|
|
2013-07-08 18:37:12 +00:00
|
|
|
mapper := &metricMapper{}
|
|
|
|
if *mappingConfig != "" {
|
2013-07-05 17:31:53 +00:00
|
|
|
err := mapper.initFromFile(*mappingConfig)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal("Error loading config:", err)
|
|
|
|
}
|
2013-07-08 18:37:12 +00:00
|
|
|
go watchConfig(*mappingConfig, mapper)
|
2013-07-05 17:31:53 +00:00
|
|
|
}
|
2013-07-08 18:37:12 +00:00
|
|
|
bridge := NewBridge(mapper)
|
2013-07-02 16:16:39 +00:00
|
|
|
bridge.Listen(events)
|
|
|
|
}
|