telegraf/plugins/inputs/filestat/filestat.go

136 lines
2.8 KiB
Go
Raw Normal View History

2016-04-21 01:51:25 +00:00
package filestat
import (
"crypto/md5"
"fmt"
"io"
"os"
"github.com/influxdata/telegraf"
"github.com/influxdata/telegraf/internal/globpath"
2016-04-21 01:51:25 +00:00
"github.com/influxdata/telegraf/plugins/inputs"
)
const sampleConfig = `
## Files to gather stats about.
## These accept standard unix glob matching rules, but with the addition of
2016-04-22 21:47:26 +00:00
## ** as a "super asterisk". ie:
## "/var/log/**.log" -> recursively find all .log files in /var/log
## "/var/log/*/*.log" -> find all .log files with a parent dir in /var/log
## "/var/log/apache.log" -> just tail the apache log file
##
## See https://github.com/gobwas/glob for more examples
##
files = ["/var/log/**.log"]
2016-04-21 01:51:25 +00:00
## If true, read the entire file and calculate an md5 checksum.
md5 = false
`
type FileStat struct {
Md5 bool
Files []string
Log telegraf.Logger
// maps full file paths to globmatch obj
globs map[string]*globpath.GlobPath
2016-04-21 01:51:25 +00:00
}
func NewFileStat() *FileStat {
return &FileStat{
globs: make(map[string]*globpath.GlobPath),
2016-04-21 01:51:25 +00:00
}
}
func (*FileStat) Description() string {
2016-04-21 01:51:25 +00:00
return "Read stats about given file(s)"
}
func (*FileStat) SampleConfig() string { return sampleConfig }
2016-04-21 01:51:25 +00:00
func (f *FileStat) Gather(acc telegraf.Accumulator) error {
var err error
for _, filepath := range f.Files {
// Get the compiled glob object for this filepath
g, ok := f.globs[filepath]
if !ok {
if g, err = globpath.Compile(filepath); err != nil {
2017-04-24 18:13:26 +00:00
acc.AddError(err)
2016-04-21 01:51:25 +00:00
continue
}
f.globs[filepath] = g
}
files := g.Match()
if len(files) == 0 {
acc.AddFields("filestat",
map[string]interface{}{
"exists": int64(0),
},
map[string]string{
"file": filepath,
})
continue
}
2018-12-18 22:23:25 +00:00
for _, fileName := range files {
2016-04-21 01:51:25 +00:00
tags := map[string]string{
"file": fileName,
2016-04-21 01:51:25 +00:00
}
fields := map[string]interface{}{
"exists": int64(1),
}
2018-12-18 22:23:25 +00:00
fileInfo, err := os.Stat(fileName)
if os.IsNotExist(err) {
fields["exists"] = int64(0)
}
if fileInfo == nil {
f.Log.Errorf("Unable to get info for file %q, possible permissions issue",
fileName)
} else {
fields["size_bytes"] = fileInfo.Size()
fields["modification_time"] = fileInfo.ModTime().UnixNano()
2016-04-21 01:51:25 +00:00
}
if f.Md5 {
md5, err := getMd5(fileName)
2016-04-21 01:51:25 +00:00
if err != nil {
2017-04-24 18:13:26 +00:00
acc.AddError(err)
2016-04-21 01:51:25 +00:00
} else {
fields["md5_sum"] = md5
}
}
acc.AddFields("filestat", fields, tags)
}
}
return nil
}
// Read given file and calculate an md5 hash.
func getMd5(file string) (string, error) {
of, err := os.Open(file)
if err != nil {
return "", err
}
defer of.Close()
hash := md5.New()
_, err = io.Copy(hash, of)
if err != nil {
// fatal error
return "", err
}
return fmt.Sprintf("%x", hash.Sum(nil)), nil
}
func init() {
inputs.Add("filestat", func() telegraf.Input {
return NewFileStat()
})
}