telegraf/plugins/zfs/zfs.go

138 lines
2.8 KiB
Go
Raw Normal View History

2015-11-03 15:53:09 +00:00
package zfs
import (
2015-12-08 09:50:42 +00:00
"fmt"
2015-11-03 15:53:09 +00:00
"path/filepath"
"strconv"
"strings"
"github.com/influxdb/telegraf/internal"
2015-11-03 15:53:09 +00:00
"github.com/influxdb/telegraf/plugins"
)
type Zfs struct {
KstatPath string
KstatMetrics []string
2015-12-08 09:50:42 +00:00
PoolMetrics bool
2015-11-03 15:53:09 +00:00
}
var sampleConfig = `
# ZFS kstat path
# If not specified, then default is:
# kstatPath = "/proc/spl/kstat/zfs"
#
# By default, telegraf gather all zfs stats
# If not specified, then default is:
# kstatMetrics = ["arcstats", "zfetchstats", "vdev_cache_stats"]
2015-12-08 09:50:42 +00:00
#
# By default, don't gather zpool stats
# poolMetrics = false
2015-11-03 15:53:09 +00:00
`
func (z *Zfs) SampleConfig() string {
return sampleConfig
}
func (z *Zfs) Description() string {
return "Read metrics of ZFS from arcstats, zfetchstats and vdev_cache_stats"
}
2015-12-08 09:50:42 +00:00
func getPoolStats(kstatPath string, poolMetrics bool, acc plugins.Accumulator) (map[string]string, error) {
2015-11-03 15:53:09 +00:00
var pools string
poolsDirs, _ := filepath.Glob(kstatPath + "/*/io")
for _, poolDir := range poolsDirs {
poolDirSplit := strings.Split(poolDir, "/")
pool := poolDirSplit[len(poolDirSplit)-2]
if len(pools) != 0 {
pools += "::"
}
pools += pool
2015-12-08 09:50:42 +00:00
if poolMetrics {
err := readPoolStats(poolDir, pool, acc)
if err != nil {
return nil, err
}
}
}
return map[string]string{"pools": pools}, nil
}
func readPoolStats(poolIoPath string, poolName string, acc plugins.Accumulator) error {
lines, err := internal.ReadLines(poolIoPath)
if err != nil {
return err
}
if len(lines) != 3 {
return err
}
keys := strings.Fields(lines[1])
values := strings.Fields(lines[2])
keyCount := len(keys)
if keyCount != len(values) {
return fmt.Errorf("Key and value count don't match Keys:%v Values:%v", keys, values)
}
tag := map[string]string{"pool": poolName}
for i := 0; i < keyCount; i++ {
value, err := strconv.ParseInt(values[i], 10, 64)
if err != nil {
return err
}
acc.Add(keys[i], value, tag)
2015-11-03 15:53:09 +00:00
}
2015-12-08 09:50:42 +00:00
return nil
2015-11-03 15:53:09 +00:00
}
func (z *Zfs) Gather(acc plugins.Accumulator) error {
kstatMetrics := z.KstatMetrics
if len(kstatMetrics) == 0 {
kstatMetrics = []string{"arcstats", "zfetchstats", "vdev_cache_stats"}
}
kstatPath := z.KstatPath
if len(kstatPath) == 0 {
kstatPath = "/proc/spl/kstat/zfs"
}
2015-12-08 09:50:42 +00:00
tags, err := getPoolStats(kstatPath, z.PoolMetrics, acc)
if err != nil {
return err
}
2015-11-03 15:53:09 +00:00
for _, metric := range kstatMetrics {
lines, err := internal.ReadLines(kstatPath + "/" + metric)
2015-11-03 15:53:09 +00:00
if err != nil {
return err
2015-11-03 15:53:09 +00:00
}
for i, line := range lines {
if i == 0 || i == 1 {
continue
}
if len(line) < 1 {
continue
}
rawData := strings.Split(line, " ")
key := metric + "_" + rawData[0]
rawValue := rawData[len(rawData)-1]
value, _ := strconv.ParseInt(rawValue, 10, 64)
acc.Add(key, value, tags)
}
}
return nil
}
func init() {
plugins.Add("zfs", func() plugins.Plugin {
return &Zfs{}
})
}