commit
This commit is contained in:
@@ -0,0 +1,7 @@
|
||||
module code.yun.ink/pkg/watchx
|
||||
|
||||
go 1.20
|
||||
|
||||
require github.com/fsnotify/fsnotify v1.7.0
|
||||
|
||||
require golang.org/x/sys v0.4.0 // indirect
|
||||
@@ -0,0 +1,4 @@
|
||||
github.com/fsnotify/fsnotify v1.7.0 h1:8JEhPFa5W2WU7YfeZzPNqzMP6Lwt7L2715Ggo0nosvA=
|
||||
github.com/fsnotify/fsnotify v1.7.0/go.mod h1:40Bi/Hjc2AVfZrqy+aj+yEI+/bRxZnMJyTJwOpGvigM=
|
||||
golang.org/x/sys v0.4.0 h1:Zr2JFtRQNX3BCZ8YtxRE9hNJYC8J6I1MVbMg6owUp18=
|
||||
golang.org/x/sys v0.4.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||
@@ -0,0 +1,214 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"context"
|
||||
"fmt"
|
||||
"sync"
|
||||
|
||||
// "log"
|
||||
"os"
|
||||
"path/filepath"
|
||||
|
||||
"github.com/fsnotify/fsnotify"
|
||||
)
|
||||
|
||||
type Watch struct {
|
||||
ctx context.Context
|
||||
watch *fsnotify.Watcher
|
||||
mutex sync.Mutex // 保护watch状态的互斥锁
|
||||
data chan Action // 文件列表
|
||||
}
|
||||
|
||||
type Action struct {
|
||||
Path string // 文件路径
|
||||
Action string //
|
||||
}
|
||||
|
||||
func NewWatch(ctx context.Context) (*Watch, error) {
|
||||
watch, err := fsnotify.NewWatcher()
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
return &Watch{
|
||||
ctx: ctx,
|
||||
watch: watch,
|
||||
data: make(chan Action), // 初始化文件列表
|
||||
}, nil
|
||||
}
|
||||
|
||||
// 添加路径到监控中候,需要加锁
|
||||
func (w *Watch) addPath(path string, isDir bool) error {
|
||||
w.mutex.Lock()
|
||||
defer w.mutex.Unlock()
|
||||
if isDir {
|
||||
fmt.Println("文件夹:", path)
|
||||
} else {
|
||||
fmt.Println("文件:", path)
|
||||
}
|
||||
|
||||
err := w.watch.Add(path)
|
||||
if err != nil {
|
||||
fmt.Printf("Error adding path %q: %v\n", path, err)
|
||||
return err
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
// 处理文件系统事件
|
||||
func (w *Watch) handleEvent(ev fsnotify.Event) error {
|
||||
fmt.Println("event:", ev)
|
||||
|
||||
if ev.Op&fsnotify.Create == fsnotify.Create {
|
||||
fmt.Println("创建文件:", ev.Name)
|
||||
w.addPath(ev.Name, false)
|
||||
w.data <- ev.Name
|
||||
}
|
||||
|
||||
if ev.Op&fsnotify.Write == fsnotify.Write {
|
||||
fmt.Println("写入文件:", ev.Name)
|
||||
}
|
||||
|
||||
if ev.Op&fsnotify.Remove == fsnotify.Remove {
|
||||
fmt.Println("删除文件:", ev.Name)
|
||||
w.watch.Remove(ev.Name)
|
||||
}
|
||||
|
||||
if ev.Op&fsnotify.Rename == fsnotify.Rename {
|
||||
fmt.Println("重命名:", ev.Name)
|
||||
}
|
||||
|
||||
if ev.Op&fsnotify.Chmod == fsnotify.Chmod {
|
||||
fmt.Println("修改权限:", ev.Name)
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func (w *Watch) watchDir(dir string) error {
|
||||
err := filepath.Walk(dir, func(path string, info os.FileInfo, err error) error {
|
||||
if err != nil {
|
||||
fmt.Printf("Error accessing path %q: %v\n", path, err)
|
||||
return err
|
||||
}
|
||||
if info.IsDir() {
|
||||
return w.addPath(path, true)
|
||||
} else {
|
||||
return w.addPath(path, false)
|
||||
}
|
||||
})
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
go func() {
|
||||
defer w.watch.Close() // 确保Watcher在退出时被关闭
|
||||
|
||||
for {
|
||||
select {
|
||||
case ev := <-w.watch.Events:
|
||||
w.handleEvent(ev)
|
||||
case err := <-w.watch.Errors:
|
||||
fmt.Printf("error: %v\n", err)
|
||||
case <-w.ctx.Done(): // 接收到停止信号,退出goroutine
|
||||
return
|
||||
}
|
||||
}
|
||||
}()
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
// 监控目录
|
||||
func (w *Watch) watchDir2(dir string) error {
|
||||
// 通过Walk来遍历目录下的所有子目录
|
||||
filepath.Walk(dir, func(path string, info os.FileInfo, err error) error {
|
||||
if err != nil {
|
||||
fmt.Printf("Error accessing path %q: %v\n", path, err)
|
||||
return err
|
||||
}
|
||||
// 这里判断是否为目录,只需监控目录即可
|
||||
// 目录下的文件也在监控范围内,不需要我们一个一个加
|
||||
if info.IsDir() {
|
||||
path, err := filepath.Abs(path)
|
||||
if err != nil {
|
||||
panic(err)
|
||||
}
|
||||
err = w.watch.Add(path)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
fmt.Println("文件夹:", path)
|
||||
} else {
|
||||
err = w.watch.Add(path)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
fmt.Println("文件:", path)
|
||||
}
|
||||
return nil
|
||||
})
|
||||
go func() {
|
||||
for {
|
||||
select {
|
||||
case ev := <-w.watch.Events:
|
||||
{
|
||||
|
||||
fmt.Printf("ev %+v \n", ev)
|
||||
|
||||
if ev.Op&fsnotify.Create == fsnotify.Create {
|
||||
fmt.Println("创建文件:", ev.Name)
|
||||
// 这里获取新创建文件的信息,如果是目录,则加入监控中
|
||||
fi, err := os.Stat(ev.Name)
|
||||
if err == nil && fi.IsDir() {
|
||||
w.watch.Add(ev.Name)
|
||||
fmt.Println("添加监控:", ev.Name)
|
||||
}
|
||||
}
|
||||
if ev.Op&fsnotify.Write == fsnotify.Write {
|
||||
// 修改文件
|
||||
fmt.Println("写入文件:", ev.Name)
|
||||
}
|
||||
if ev.Op&fsnotify.Remove == fsnotify.Remove {
|
||||
fmt.Println("删除文件:", ev.Name)
|
||||
// 如果删除文件是目录,则移除监控
|
||||
// 注意这里无法使用os.Stat来判断是否是目录了
|
||||
// 因为重命名后,go已经无法找到原文件来获取信息了
|
||||
// 所以这里就简单粗暴的直接remove好了
|
||||
w.watch.Remove(ev.Name)
|
||||
}
|
||||
if ev.Op&fsnotify.Rename == fsnotify.Rename {
|
||||
fmt.Println("重命名:", ev.Name)
|
||||
}
|
||||
if ev.Op&fsnotify.Chmod == fsnotify.Chmod {
|
||||
fmt.Println("修改权限:", ev.Name)
|
||||
}
|
||||
}
|
||||
case err := <-w.watch.Errors:
|
||||
{
|
||||
fmt.Println("error:", err)
|
||||
return
|
||||
}
|
||||
}
|
||||
}
|
||||
}()
|
||||
}
|
||||
|
||||
func main() {
|
||||
|
||||
w, err := NewWatch()
|
||||
if err != nil {
|
||||
fmt.Println(err)
|
||||
return
|
||||
}
|
||||
|
||||
dirToWatch := "/log"
|
||||
err = w.watchDir(dirToWatch)
|
||||
if err != nil {
|
||||
fmt.Println(err)
|
||||
return
|
||||
}
|
||||
|
||||
// 创建一个监控对象
|
||||
select {}
|
||||
}
|
||||
Reference in New Issue
Block a user