83 lines
1.8 KiB
Go
83 lines
1.8 KiB
Go
package main
|
|
|
|
import (
|
|
"github.com/fsnotify/fsnotify"
|
|
"log"
|
|
"log/slog"
|
|
"os"
|
|
"path/filepath"
|
|
"time"
|
|
)
|
|
|
|
var ignoredDirs = []string{".git", ".idea", "node_modules", "vendor"}
|
|
|
|
func startWatcher(cb func(file []*fsnotify.Event)) {
|
|
events := make([]*fsnotify.Event, 0)
|
|
debouncer := NewDebouncer(100 * time.Millisecond)
|
|
|
|
defer func() {
|
|
if r := recover(); r != nil {
|
|
slog.Debug("Recovered from fatal error:", slog.String("error", r.(error).Error()))
|
|
}
|
|
}()
|
|
// Create new watcher.
|
|
watcher, err := fsnotify.NewWatcher()
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
defer watcher.Close()
|
|
// Start listening for events.
|
|
go func() {
|
|
for {
|
|
select {
|
|
case event, ok := <-watcher.Events:
|
|
if !ok {
|
|
return
|
|
}
|
|
if event.Has(fsnotify.Write) || event.Has(fsnotify.Remove) || event.Has(fsnotify.Rename) {
|
|
events = append(events, &event)
|
|
debouncer.Do(func() {
|
|
cb(events)
|
|
events = make([]*fsnotify.Event, 0)
|
|
})
|
|
}
|
|
case err, ok := <-watcher.Errors:
|
|
if !ok {
|
|
return
|
|
}
|
|
slog.Error("error:", err.Error())
|
|
}
|
|
}
|
|
}()
|
|
|
|
rootDir := "."
|
|
// Walk through the root directory and add all subdirectories to the watcher
|
|
err = filepath.Walk(rootDir, func(path string, info os.FileInfo, err error) error {
|
|
if err != nil {
|
|
return err
|
|
}
|
|
// Ignore directories in the ignoredDirs list
|
|
for _, ignoredDir := range ignoredDirs {
|
|
if ignoredDir == info.Name() {
|
|
return filepath.SkipDir
|
|
}
|
|
}
|
|
// Only watch directories
|
|
if info.IsDir() {
|
|
err = watcher.Add(path)
|
|
if err != nil {
|
|
slog.Error("Error adding directory to watcher:", err)
|
|
} else {
|
|
slog.Debug("Watching directory:", slog.String("path", path))
|
|
}
|
|
}
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
|
|
done := RegisterSignals()
|
|
<-done
|
|
println("process exited")
|
|
}
|