forked from elastic/beats
-
Notifications
You must be signed in to change notification settings - Fork 0
/
watcher.go
105 lines (87 loc) · 1.63 KB
/
watcher.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
package watcher
import (
"hash/fnv"
"io/ioutil"
"os"
"time"
"github.com/elastic/beats/libbeat/logp"
)
type Watch interface {
Stop()
}
type filePoller struct {
done chan struct{}
}
type fileChangeTester struct {
path string
sz int
hash uint64
stat os.FileInfo
}
func NewFilePoller(
path string,
poll time.Duration,
do func([]byte),
) (Watch, error) {
fw := &filePoller{
done: make(chan struct{}),
}
tester := &fileChangeTester{path: path, sz: -1}
if content, changed := tester.check(); changed {
do(content)
}
go func() {
ticker := time.NewTicker(poll)
defer ticker.Stop()
for {
if content, changed := tester.check(); changed {
do(content)
}
select {
case <-fw.done:
return
case <-ticker.C:
}
}
}()
return fw, nil
}
func (f *filePoller) Stop() {
close(f.done)
}
func (w *fileChangeTester) check() ([]byte, bool) {
f, err := os.Open(w.path)
if err != nil {
logp.Info("Failed to load file: %v", err)
return nil, false
}
defer f.Close()
stat, err := f.Stat()
if err != nil {
logp.Info("Reading file '%v' stat failed with: %v", w.path, err)
return nil, false
}
if w.stat != nil {
if stat.Size() == w.stat.Size() && !stat.ModTime().After(w.stat.ModTime()) {
return nil, false
}
}
content, err := ioutil.ReadAll(f)
if err != nil {
logp.Info("Reading file '%v' failed with: %v", w.path, err)
return nil, false
}
var hash uint64
if len(content) != 0 || w.sz == 0 {
hasher := fnv.New64a()
hasher.Write(content)
hash = hasher.Sum64()
if w.hash == hash {
return nil, false
}
}
w.hash = hash
w.stat = stat
w.sz = len(content)
return content, true
}