2013-05-07 14:40:10 +00:00
package main
import (
2014-02-18 11:35:11 +00:00
"encoding/json"
2013-05-07 14:40:10 +00:00
"flag"
2014-06-04 13:09:33 +00:00
"fmt"
2014-02-18 11:35:11 +00:00
"io/ioutil"
"net/http"
2015-03-05 16:02:17 +00:00
_ "net/http/pprof"
2014-02-18 11:35:11 +00:00
"os"
"os/signal"
2015-05-12 15:04:08 +00:00
"sort"
2014-06-04 11:12:34 +00:00
"strings"
2014-02-18 11:35:11 +00:00
"sync"
"syscall"
"time"
2014-02-07 16:09:39 +00:00
2014-06-04 11:12:34 +00:00
"github.com/golang/glog"
2014-02-18 11:35:11 +00:00
"github.com/prometheus/client_golang/prometheus"
"github.com/prometheus/node_exporter/collector"
2013-05-07 14:40:10 +00:00
)
2014-06-26 17:20:36 +00:00
const subsystem = "exporter"
2013-05-07 14:40:10 +00:00
var (
2015-04-16 03:51:09 +00:00
// set at build time
Version = "0.0.0.dev"
2015-02-08 23:21:57 +00:00
configFile = flag . String ( "config.file" , "" , "Path to config file." )
memProfile = flag . String ( "debug.memprofile-file" , "" , "Write memory profile to this file upon receipt of SIGUSR1." )
listenAddress = flag . String ( "web.listen-address" , ":9100" , "Address on which to expose metrics and web interface." )
metricsPath = flag . String ( "web.telemetry-path" , "/metrics" , "Path under which to expose metrics." )
enabledCollectors = flag . String ( "collectors.enabled" , "attributes,diskstats,filesystem,loadavg,meminfo,stat,textfile,time,netdev,netstat" , "Comma-separated list of collectors to use." )
printCollectors = flag . Bool ( "collectors.print" , false , "If true, print available collectors and exit." )
2014-12-17 17:59:06 +00:00
authUser = flag . String ( "auth.user" , "" , "Username for basic auth." )
authPass = flag . String ( "auth.pass" , "" , "Password for basic auth." )
2014-06-26 17:20:36 +00:00
collectorLabelNames = [ ] string { "collector" , "result" }
scrapeDurations = prometheus . NewSummaryVec (
prometheus . SummaryOpts {
Namespace : collector . Namespace ,
Subsystem : subsystem ,
Name : "scrape_duration_seconds" ,
Help : "node_exporter: Duration of a scrape job." ,
} ,
collectorLabelNames ,
)
2013-05-07 14:40:10 +00:00
)
2014-10-29 14:16:43 +00:00
// Implements Collector.
type NodeCollector struct {
collectors map [ string ] collector . Collector
}
// Implements Collector.
func ( n NodeCollector ) Describe ( ch chan <- * prometheus . Desc ) {
scrapeDurations . Describe ( ch )
}
// Implements Collector.
func ( n NodeCollector ) Collect ( ch chan <- prometheus . Metric ) {
wg := sync . WaitGroup { }
wg . Add ( len ( n . collectors ) )
for name , c := range n . collectors {
go func ( name string , c collector . Collector ) {
Execute ( name , c , ch )
wg . Done ( )
} ( name , c )
}
wg . Wait ( )
scrapeDurations . Collect ( ch )
}
2014-12-17 17:59:06 +00:00
type basicAuthHandler struct {
handler http . HandlerFunc
user string
password string
}
func ( h * basicAuthHandler ) ServeHTTP ( w http . ResponseWriter , r * http . Request ) {
user , password , ok := r . BasicAuth ( )
if ! ok || password != h . password || user != h . user {
w . Header ( ) . Set ( "WWW-Authenticate" , "Basic realm=\"metrics\"" )
http . Error ( w , "Invalid username or password" , http . StatusUnauthorized )
return
}
h . handler ( w , r )
return
}
2014-10-29 14:16:43 +00:00
func Execute ( name string , c collector . Collector , ch chan <- prometheus . Metric ) {
begin := time . Now ( )
err := c . Update ( ch )
duration := time . Since ( begin )
var result string
if err != nil {
glog . Infof ( "ERROR: %s failed after %fs: %s" , name , duration . Seconds ( ) , err )
result = "error"
} else {
glog . Infof ( "OK: %s success after %fs." , name , duration . Seconds ( ) )
result = "success"
}
scrapeDurations . WithLabelValues ( name , result ) . Observe ( duration . Seconds ( ) )
}
func getConfig ( file string ) ( * collector . Config , error ) {
config := & collector . Config { }
glog . Infof ( "Reading config %s" , * configFile )
bytes , err := ioutil . ReadFile ( * configFile )
if err != nil {
return nil , err
}
return config , json . Unmarshal ( bytes , & config )
}
func loadCollectors ( file string ) ( map [ string ] collector . Collector , error ) {
collectors := map [ string ] collector . Collector { }
2014-12-18 11:25:02 +00:00
config := & collector . Config { }
if file != "" {
var err error
config , err = getConfig ( file )
if err != nil {
return nil , fmt . Errorf ( "couldn't read config %s: %s" , file , err )
}
2014-10-29 14:16:43 +00:00
}
for _ , name := range strings . Split ( * enabledCollectors , "," ) {
fn , ok := collector . Factories [ name ]
if ! ok {
2014-11-24 22:15:13 +00:00
return nil , fmt . Errorf ( "collector '%s' not available" , name )
2014-10-29 14:16:43 +00:00
}
c , err := fn ( * config )
if err != nil {
return nil , err
}
collectors [ name ] = c
}
return collectors , nil
}
2013-05-07 14:40:10 +00:00
func main ( ) {
flag . Parse ( )
2015-04-16 03:51:09 +00:00
2014-06-04 13:09:33 +00:00
if * printCollectors {
2015-05-12 15:04:08 +00:00
collectorNames := make ( sort . StringSlice , 0 , len ( collector . Factories ) )
2014-06-04 13:09:33 +00:00
for n , _ := range collector . Factories {
2015-05-12 15:04:08 +00:00
collectorNames = append ( collectorNames , n )
}
collectorNames . Sort ( )
fmt . Printf ( "Available collectors:\n" )
for _ , n := range collectorNames {
2014-06-04 13:09:33 +00:00
fmt . Printf ( " - %s\n" , n )
}
return
}
2014-06-26 17:20:36 +00:00
collectors , err := loadCollectors ( * configFile )
2013-05-07 14:40:10 +00:00
if err != nil {
2014-12-17 18:20:34 +00:00
glog . Fatalf ( "Couldn't load config and collectors: %s" , err )
2013-05-07 14:40:10 +00:00
}
2014-02-18 11:35:11 +00:00
2014-06-04 11:12:34 +00:00
glog . Infof ( "Enabled collectors:" )
for n , _ := range collectors {
glog . Infof ( " - %s" , n )
2014-02-07 16:09:39 +00:00
}
2014-02-18 11:35:11 +00:00
2014-10-29 14:16:43 +00:00
nodeCollector := NodeCollector { collectors : collectors }
prometheus . MustRegister ( nodeCollector )
2014-02-18 11:35:11 +00:00
sigUsr1 := make ( chan os . Signal )
signal . Notify ( sigUsr1 , syscall . SIGUSR1 )
2014-12-17 17:59:06 +00:00
handler := prometheus . Handler ( )
if * authUser != "" || * authPass != "" {
if * authUser == "" || * authPass == "" {
glog . Fatal ( "You need to specify -auth.user and -auth.pass to enable basic auth" )
}
handler = & basicAuthHandler {
handler : prometheus . Handler ( ) . ServeHTTP ,
user : * authUser ,
password : * authPass ,
}
}
2014-02-18 11:35:11 +00:00
2015-03-05 16:02:17 +00:00
http . Handle ( * metricsPath , handler )
http . HandleFunc ( "/" , func ( w http . ResponseWriter , r * http . Request ) {
w . Write ( [ ] byte ( ` < html >
< head > < title > Node Exporter < / title > < / head >
< body >
< h1 > Node Exporter < / h1 >
< p > < a href = "` + *metricsPath + `" > Metrics < / a > < / p >
< / body >
< / html > ` ) )
} )
2015-04-16 03:51:09 +00:00
glog . Infof ( "Starting node_exporter v%s at %s" , Version , * listenAddress )
2015-03-05 16:02:17 +00:00
err = http . ListenAndServe ( * listenAddress , nil )
if err != nil {
glog . Fatal ( err )
}
2014-02-18 11:35:11 +00:00
}