Files
watchx/watchx.go
T
2026-05-04 00:17:15 +08:00

215 lines
4.4 KiB
Go

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 {}
}