2014-06-05 12:44:44 -07:00
// +build !nofilesystem
package collector
import (
"bufio"
"flag"
"fmt"
"os"
"regexp"
"strings"
"syscall"
2014-06-26 10:20:36 -07:00
"github.com/golang/glog"
2014-06-05 12:44:44 -07:00
"github.com/prometheus/client_golang/prometheus"
)
const (
2014-06-26 10:20:36 -07:00
procMounts = "/proc/mounts"
filesystemSubsystem = "filesystem"
2014-06-05 12:44:44 -07:00
)
var (
2015-02-18 09:15:57 -08:00
ignoredMountPoints = flag . String ( "collector.filesystem.ignored-mount-points" , "^/(sys|proc|dev)($|/)" , "Regexp of mount points to ignore for filesystem collector." )
2014-06-05 12:44:44 -07:00
)
type filesystemCollector struct {
2015-05-20 11:04:49 -07:00
2014-06-05 12:44:44 -07:00
ignoredMountPointsPattern * regexp . Regexp
2014-11-24 18:00:17 -08:00
size , free , avail , files , filesFree * prometheus . GaugeVec
2014-06-05 12:44:44 -07:00
}
func init ( ) {
Factories [ "filesystem" ] = NewFilesystemCollector
}
2015-05-20 11:04:49 -07:00
// Takes a prometheus registry and returns a new Collector exposing
2014-06-05 12:44:44 -07:00
// network device filesystems.
2015-05-20 11:04:49 -07:00
func NewFilesystemCollector ( ) ( Collector , error ) {
2014-11-24 18:00:17 -08:00
var filesystemLabelNames = [ ] string { "filesystem" }
return & filesystemCollector {
2015-05-20 11:04:49 -07:00
2014-06-05 12:44:44 -07:00
ignoredMountPointsPattern : regexp . MustCompile ( * ignoredMountPoints ) ,
2014-11-24 18:00:17 -08:00
size : prometheus . NewGaugeVec (
prometheus . GaugeOpts {
Namespace : Namespace ,
Subsystem : filesystemSubsystem ,
Name : "size" ,
Help : "Filesystem size in bytes." ,
} ,
filesystemLabelNames ,
) ,
free : prometheus . NewGaugeVec (
prometheus . GaugeOpts {
Namespace : Namespace ,
Subsystem : filesystemSubsystem ,
Name : "free" ,
Help : "Filesystem free space in bytes." ,
} ,
filesystemLabelNames ,
) ,
avail : prometheus . NewGaugeVec (
prometheus . GaugeOpts {
Namespace : Namespace ,
Subsystem : filesystemSubsystem ,
Name : "avail" ,
Help : "Filesystem space available to non-root users in bytes." ,
} ,
filesystemLabelNames ,
) ,
files : prometheus . NewGaugeVec (
prometheus . GaugeOpts {
Namespace : Namespace ,
Subsystem : filesystemSubsystem ,
Name : "files" ,
Help : "Filesystem total file nodes." ,
} ,
filesystemLabelNames ,
) ,
filesFree : prometheus . NewGaugeVec (
prometheus . GaugeOpts {
Namespace : Namespace ,
Subsystem : filesystemSubsystem ,
Name : "files_free" ,
Help : "Filesystem total free file nodes." ,
} ,
filesystemLabelNames ,
) ,
} , nil
2014-06-05 12:44:44 -07:00
}
// Expose filesystem fullness.
2014-10-29 07:16:43 -07:00
func ( c * filesystemCollector ) Update ( ch chan <- prometheus . Metric ) ( err error ) {
2014-06-05 12:44:44 -07:00
mps , err := mountPoints ( )
if err != nil {
2014-10-29 07:16:43 -07:00
return err
2014-06-05 12:44:44 -07:00
}
for _ , mp := range mps {
if c . ignoredMountPointsPattern . MatchString ( mp ) {
glog . V ( 1 ) . Infof ( "Ignoring mount point: %s" , mp )
continue
}
buf := new ( syscall . Statfs_t )
err := syscall . Statfs ( mp , buf )
if err != nil {
2014-10-29 07:16:43 -07:00
return fmt . Errorf ( "Statfs on %s returned %s" , mp , err )
2014-06-05 12:44:44 -07:00
}
2014-11-24 18:00:17 -08:00
c . size . WithLabelValues ( mp ) . Set ( float64 ( buf . Blocks ) * float64 ( buf . Bsize ) )
c . free . WithLabelValues ( mp ) . Set ( float64 ( buf . Bfree ) * float64 ( buf . Bsize ) )
c . avail . WithLabelValues ( mp ) . Set ( float64 ( buf . Bavail ) * float64 ( buf . Bsize ) )
c . files . WithLabelValues ( mp ) . Set ( float64 ( buf . Files ) )
c . filesFree . WithLabelValues ( mp ) . Set ( float64 ( buf . Ffree ) )
2014-06-05 12:44:44 -07:00
}
2014-11-24 18:00:17 -08:00
c . size . Collect ( ch )
c . free . Collect ( ch )
c . avail . Collect ( ch )
c . files . Collect ( ch )
c . filesFree . Collect ( ch )
2014-10-29 07:16:43 -07:00
return err
2014-06-05 12:44:44 -07:00
}
func mountPoints ( ) ( [ ] string , error ) {
file , err := os . Open ( procMounts )
if err != nil {
return nil , err
}
defer file . Close ( )
mountPoints := [ ] string { }
scanner := bufio . NewScanner ( file )
for scanner . Scan ( ) {
parts := strings . Fields ( scanner . Text ( ) )
mountPoints = append ( mountPoints , parts [ 1 ] )
}
return mountPoints , nil
}