ZFS Collector: Add zfetchstats functionality

Signed-Off-By: Joe Handzik <joseph.t.handzik@hpe.com>
This commit is contained in:
Joe Handzik 2017-01-23 10:39:32 -06:00
parent 2a0c80cbe2
commit a3125ab4d9
5 changed files with 112 additions and 11 deletions

View file

@ -2350,6 +2350,39 @@ node_zfsArc_prefetch_metadata_misses 16071
# HELP node_zfsArc_size kstat.zfs.misc.arcstats.size # HELP node_zfsArc_size kstat.zfs.misc.arcstats.size
# TYPE node_zfsArc_size untyped # TYPE node_zfsArc_size untyped
node_zfsArc_size 1.603939792e+09 node_zfsArc_size 1.603939792e+09
# HELP node_zfsFetch_bogus_streams kstat.zfs.misc.zfetchstats.bogus_streams
# TYPE node_zfsFetch_bogus_streams untyped
node_zfsFetch_bogus_streams 0
# HELP node_zfsFetch_colinear_hits kstat.zfs.misc.zfetchstats.colinear_hits
# TYPE node_zfsFetch_colinear_hits untyped
node_zfsFetch_colinear_hits 0
# HELP node_zfsFetch_colinear_misses kstat.zfs.misc.zfetchstats.colinear_misses
# TYPE node_zfsFetch_colinear_misses untyped
node_zfsFetch_colinear_misses 11
# HELP node_zfsFetch_hits kstat.zfs.misc.zfetchstats.hits
# TYPE node_zfsFetch_hits untyped
node_zfsFetch_hits 7.067992e+06
# HELP node_zfsFetch_misses kstat.zfs.misc.zfetchstats.misses
# TYPE node_zfsFetch_misses untyped
node_zfsFetch_misses 11
# HELP node_zfsFetch_reclaim_failures kstat.zfs.misc.zfetchstats.reclaim_failures
# TYPE node_zfsFetch_reclaim_failures untyped
node_zfsFetch_reclaim_failures 11
# HELP node_zfsFetch_reclaim_successes kstat.zfs.misc.zfetchstats.reclaim_successes
# TYPE node_zfsFetch_reclaim_successes untyped
node_zfsFetch_reclaim_successes 0
# HELP node_zfsFetch_streams_noresets kstat.zfs.misc.zfetchstats.streams_noresets
# TYPE node_zfsFetch_streams_noresets untyped
node_zfsFetch_streams_noresets 2
# HELP node_zfsFetch_streams_resets kstat.zfs.misc.zfetchstats.streams_resets
# TYPE node_zfsFetch_streams_resets untyped
node_zfsFetch_streams_resets 0
# HELP node_zfsFetch_stride_hits kstat.zfs.misc.zfetchstats.stride_hits
# TYPE node_zfsFetch_stride_hits untyped
node_zfsFetch_stride_hits 7.06799e+06
# HELP node_zfsFetch_stride_misses kstat.zfs.misc.zfetchstats.stride_misses
# TYPE node_zfsFetch_stride_misses untyped
node_zfsFetch_stride_misses 0
# HELP process_cpu_seconds_total Total user and system CPU time spent in seconds. # HELP process_cpu_seconds_total Total user and system CPU time spent in seconds.
# TYPE process_cpu_seconds_total counter # TYPE process_cpu_seconds_total counter
# HELP process_max_fds Maximum number of open file descriptors. # HELP process_max_fds Maximum number of open file descriptors.

View file

@ -0,0 +1,13 @@
4 1 0x01 11 528 8010434610 345692669858836
name type data
hits 4 7067992
misses 4 11
colinear_hits 4 0
colinear_misses 4 11
stride_hits 4 7067990
stride_misses 4 0
reclaim_successes 4 0
reclaim_failures 4 11
streams_resets 4 0
streams_noresets 4 2
bogus_streams 4 0

View file

@ -35,6 +35,7 @@ type zfsSubsystemName string
const ( const (
arc = zfsSubsystemName("zfsArc") arc = zfsSubsystemName("zfsArc")
zfetch = zfsSubsystemName("zfsFetch")
zpoolSubsystem = zfsSubsystemName("zfsPool") zpoolSubsystem = zfsSubsystemName("zfsPool")
) )
@ -71,6 +72,10 @@ func (c *zfsCollector) Update(ch chan<- prometheus.Metric) (err error) {
return err return err
} }
// Zfetchstats
err = c.updateZfetchstats(ch)
if err != nil { return err }
// Pool stats // Pool stats
return c.updatePoolStats(ch) return c.updatePoolStats(ch)
} }

View file

@ -15,10 +15,10 @@ package collector
import ( import (
"bufio" "bufio"
"errors"
"fmt" "fmt"
"io" "io"
"os" "os"
"path/filepath"
"strconv" "strconv"
"strings" "strings"
@ -27,31 +27,45 @@ import (
) )
const ( const (
zfsArcstatsProcpath = "spl/kstat/zfs/arcstats" zfsProcpathBase = "spl/kstat/zfs/"
zfsArcstatsExt = "arcstats"
zfsFetchstatsExt = "zfetchstats"
) )
func (c *zfsCollector) openArcstatsFile() (file *os.File, err error) { func (c *zfsCollector) openProcFile(path string) (file *os.File, err error) {
file, err = os.Open(procFilePath(zfsArcstatsProcpath)) file, err = os.Open(procFilePath(path))
if err != nil { if err != nil {
log.Debugf("Cannot open %q for reading. Is the kernel module loaded?", procFilePath(zfsArcstatsProcpath)) log.Debugf("Cannot open %q for reading. Is the kernel module loaded?", procFilePath(path))
err = zfsNotAvailableError err = zfsNotAvailableError
} }
return return
} }
func (c *zfsCollector) updateArcstats(ch chan<- prometheus.Metric) (err error) { func (c *zfsCollector) updateArcstats(ch chan<- prometheus.Metric) (err error) {
file, err := c.openArcstatsFile() file, err := c.openProcFile(filepath.Join(zfsProcpathBase, zfsArcstatsExt))
if err != nil { if err != nil {
return err return err
} }
defer file.Close() defer file.Close()
return c.parseArcstatsProcfsFile(file, func(s zfsSysctl, v zfsMetricValue) { return c.parseProcfsFile(file, zfsArcstatsExt, func(s zfsSysctl, v zfsMetricValue) {
ch <- c.constSysctlMetric(arc, s, v) ch <- c.constSysctlMetric(arc, s, v)
}) })
} }
func (c *zfsCollector) parseArcstatsProcfsFile(reader io.Reader, handler func(zfsSysctl, zfsMetricValue)) (err error) { func (c *zfsCollector) updateZfetchstats(ch chan<- prometheus.Metric) (err error) {
file, err := c.openProcFile(filepath.Join(zfsProcpathBase, zfsFetchstatsExt))
if err != nil {
return err
}
defer file.Close()
return c.parseProcfsFile(file, zfsFetchstatsExt, func(s zfsSysctl, v zfsMetricValue) {
ch <- c.constSysctlMetric(zfetch, s, v)
})
}
func (c *zfsCollector) parseProcfsFile(reader io.Reader, fmt_ext string, handler func(zfsSysctl, zfsMetricValue)) (err error) {
scanner := bufio.NewScanner(reader) scanner := bufio.NewScanner(reader)
parseLine := false parseLine := false
@ -69,7 +83,7 @@ func (c *zfsCollector) parseArcstatsProcfsFile(reader io.Reader, handler func(zf
continue continue
} }
key := fmt.Sprintf("kstat.zfs.misc.arcstats.%s", parts[0]) key := fmt.Sprintf("kstat.zfs.misc.%s.%s", fmt_ext, parts[0])
value, err := strconv.Atoi(parts[2]) value, err := strconv.Atoi(parts[2])
if err != nil { if err != nil {
@ -79,7 +93,7 @@ func (c *zfsCollector) parseArcstatsProcfsFile(reader io.Reader, handler func(zf
} }
if !parseLine { if !parseLine {
return errors.New("did not parse a single arcstat metric") return fmt.Errorf("did not parse a single %q metric", fmt_ext)
} }
return scanner.Err() return scanner.Err()

View file

@ -31,7 +31,7 @@ func TestArcstatsParsing(t *testing.T) {
} }
handlerCalled := false handlerCalled := false
err = c.parseArcstatsProcfsFile(arcstatsFile, func(s zfsSysctl, v zfsMetricValue) { err = c.parseProcfsFile(arcstatsFile, "arcstats", func(s zfsSysctl, v zfsMetricValue) {
if s != zfsSysctl("kstat.zfs.misc.arcstats.hits") { if s != zfsSysctl("kstat.zfs.misc.arcstats.hits") {
return return
@ -53,3 +53,39 @@ func TestArcstatsParsing(t *testing.T) {
t.Fatal("Arcstats parsing handler was not called for some expected sysctls") t.Fatal("Arcstats parsing handler was not called for some expected sysctls")
} }
} }
func TestZfetchstatsParsing(t *testing.T) {
zfetchstatsFile, err := os.Open("fixtures/proc/spl/kstat/zfs/zfetchstats")
if err != nil {
t.Fatal(err)
}
defer zfetchstatsFile.Close()
c := zfsCollector{}
if err != nil {
t.Fatal(err)
}
handlerCalled := false
err = c.parseProcfsFile(zfetchstatsFile, "zfetchstats", func(s zfsSysctl, v zfsMetricValue) {
if s != zfsSysctl("kstat.zfs.misc.zfetchstats.hits") {
return
}
handlerCalled = true
if v != zfsMetricValue(7067992) {
t.Fatalf("Incorrect value parsed from procfs data")
}
})
if err != nil {
t.Fatal(err)
}
if !handlerCalled {
t.Fatal("Zfetchstats parsing handler was not called for some expected sysctls")
}
}