1 Star 0 Fork 0

zhangjungang/beats

加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
文件
克隆/下载
watcher.go 1.63 KB
一键复制 编辑 原始数据 按行查看 历史
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
}
Loading...
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化
1
https://gitee.com/zhangjungang/beats.git
git@gitee.com:zhangjungang/beats.git
zhangjungang
beats
beats
v6.1.2

搜索帮助