2019-11-12 22:19:20 +11:00
|
|
|
package gui
|
|
|
|
|
|
|
|
import (
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
|
|
|
|
"github.com/fsnotify/fsnotify"
|
2020-09-29 20:28:39 +10:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/commands/models"
|
2022-01-28 20:44:36 +11:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/gui/types"
|
2020-10-07 21:19:38 +11:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/utils"
|
2020-01-08 21:02:01 +11:00
|
|
|
"github.com/sirupsen/logrus"
|
2019-11-12 22:19:20 +11:00
|
|
|
)
|
|
|
|
|
2020-01-08 21:02:01 +11:00
|
|
|
// macs for some bizarre reason cap the number of watchable files to 256.
|
2021-09-01 22:51:24 +02:00
|
|
|
// there's no obvious platform agnostic way to check the situation of the user's
|
2020-01-08 21:02:01 +11:00
|
|
|
// computer so we're just arbitrarily capping at 200. This isn't so bad because
|
|
|
|
// file watching is only really an added bonus for faster refreshing.
|
2020-01-08 21:41:39 +11:00
|
|
|
const MAX_WATCHED_FILES = 50
|
2020-01-08 21:02:01 +11:00
|
|
|
|
2022-12-30 23:24:24 +11:00
|
|
|
var _ types.IFileWatcher = new(fileWatcher)
|
|
|
|
|
2020-01-08 21:02:01 +11:00
|
|
|
type fileWatcher struct {
|
|
|
|
Watcher *fsnotify.Watcher
|
|
|
|
WatchedFilenames []string
|
|
|
|
Log *logrus.Entry
|
2020-01-12 14:43:43 +11:00
|
|
|
Disabled bool
|
2020-01-08 21:02:01 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
func NewFileWatcher(log *logrus.Entry) *fileWatcher {
|
2020-03-09 11:34:10 +11:00
|
|
|
// TODO: get this going again, and ensure we don't see any crashes from it
|
2020-01-12 14:43:43 +11:00
|
|
|
return &fileWatcher{
|
|
|
|
Disabled: true,
|
|
|
|
}
|
2020-01-08 21:02:01 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
func (w *fileWatcher) watchingFilename(filename string) bool {
|
|
|
|
for _, watchedFilename := range w.WatchedFilenames {
|
|
|
|
if watchedFilename == filename {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *fileWatcher) popOldestFilename() {
|
|
|
|
// shift the last off the array to make way for this one
|
|
|
|
oldestFilename := w.WatchedFilenames[0]
|
|
|
|
w.WatchedFilenames = w.WatchedFilenames[1:]
|
|
|
|
if err := w.Watcher.Remove(oldestFilename); err != nil {
|
|
|
|
// swallowing errors here because it doesn't really matter if we can't unwatch a file
|
2020-09-26 10:23:10 +10:00
|
|
|
w.Log.Error(err)
|
2020-01-08 21:02:01 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *fileWatcher) watchFilename(filename string) {
|
|
|
|
if err := w.Watcher.Add(filename); err != nil {
|
|
|
|
// swallowing errors here because it doesn't really matter if we can't watch a file
|
2020-09-26 10:23:10 +10:00
|
|
|
w.Log.Error(err)
|
2020-01-08 21:02:01 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
// assume we're watching it now to be safe
|
|
|
|
w.WatchedFilenames = append(w.WatchedFilenames, filename)
|
|
|
|
}
|
|
|
|
|
2022-12-30 23:24:24 +11:00
|
|
|
func (w *fileWatcher) AddFilesToFileWatcher(files []*models.File) error {
|
2020-01-12 14:43:43 +11:00
|
|
|
if w.Disabled {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-01-08 21:48:40 +11:00
|
|
|
if len(files) == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-01-08 21:02:01 +11:00
|
|
|
// watch the files for changes
|
|
|
|
dirName, err := os.Getwd()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-01-08 21:48:40 +11:00
|
|
|
for _, file := range files[0:min(MAX_WATCHED_FILES, len(files))] {
|
2020-01-08 21:57:39 +11:00
|
|
|
if file.Deleted {
|
|
|
|
continue
|
|
|
|
}
|
2020-01-08 21:02:01 +11:00
|
|
|
filename := filepath.Join(dirName, file.Name)
|
|
|
|
if w.watchingFilename(filename) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if len(w.WatchedFilenames) > MAX_WATCHED_FILES {
|
|
|
|
w.popOldestFilename()
|
|
|
|
}
|
|
|
|
|
|
|
|
w.watchFilename(filename)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-01-08 21:48:40 +11:00
|
|
|
func min(a int, b int) int {
|
|
|
|
if a < b {
|
|
|
|
return a
|
|
|
|
}
|
|
|
|
return b
|
|
|
|
}
|
|
|
|
|
2019-11-12 22:19:20 +11:00
|
|
|
// NOTE: given that we often edit files ourselves, this may make us end up refreshing files too often
|
|
|
|
// TODO: consider watching the whole directory recursively (could be more expensive)
|
2022-12-30 23:24:24 +11:00
|
|
|
func (gui *Gui) WatchFilesForChanges() {
|
2020-01-08 21:02:01 +11:00
|
|
|
gui.fileWatcher = NewFileWatcher(gui.Log)
|
2020-01-12 14:43:43 +11:00
|
|
|
if gui.fileWatcher.Disabled {
|
2019-11-12 22:19:20 +11:00
|
|
|
return
|
|
|
|
}
|
2020-10-07 21:19:38 +11:00
|
|
|
go utils.Safe(func() {
|
2019-11-12 22:19:20 +11:00
|
|
|
for {
|
|
|
|
select {
|
|
|
|
// watch for events
|
2020-01-08 21:02:01 +11:00
|
|
|
case event := <-gui.fileWatcher.Watcher.Events:
|
2019-11-12 22:19:20 +11:00
|
|
|
if event.Op == fsnotify.Chmod {
|
|
|
|
// for some reason we pick up chmod events when they don't actually happen
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
// only refresh if we're not already
|
2022-12-30 23:24:24 +11:00
|
|
|
if !gui.IsRefreshingFiles {
|
2023-07-03 14:16:43 +10:00
|
|
|
gui.c.OnUIThread(func() error {
|
|
|
|
// TODO: find out if refresh needs to be run on the UI thread
|
|
|
|
return gui.c.Refresh(types.RefreshOptions{Mode: types.ASYNC, Scope: []types.RefreshableView{types.FILES}})
|
|
|
|
})
|
2019-11-12 22:19:20 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
// watch for errors
|
2020-01-08 21:02:01 +11:00
|
|
|
case err := <-gui.fileWatcher.Watcher.Errors:
|
2019-11-12 22:19:20 +11:00
|
|
|
if err != nil {
|
2022-01-16 14:46:53 +11:00
|
|
|
gui.c.Log.Error(err)
|
2019-11-12 22:19:20 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-10-07 21:19:38 +11:00
|
|
|
})
|
2019-11-12 22:19:20 +11:00
|
|
|
}
|