compose/pkg/watch/watcher_darwin.go

143 lines
3.2 KiB
Go
Raw Normal View History

2018-08-16 20:53:47 +02:00
package watch
import (
"path/filepath"
"sync"
"time"
"github.com/windmilleng/tilt/internal/logger"
"github.com/windmilleng/tilt/internal/ospath"
2018-08-16 20:53:47 +02:00
"github.com/windmilleng/fsevents"
)
// A file watcher optimized for Darwin.
// Uses FSEvents to avoid the terrible perf characteristics of kqueue.
2018-08-16 20:53:47 +02:00
type darwinNotify struct {
stream *fsevents.EventStream
events chan FileEvent
2018-08-16 20:53:47 +02:00
errors chan error
stop chan struct{}
// TODO(nick): This mutex is needed for the case where we add paths after we
// start watching. But because fsevents supports recursive watches, we don't
// actually need this feature. We should change the api contract of wmNotify
// so that, for recursive watches, we can guarantee that the path list doesn't
// change.
sm *sync.Mutex
pathsWereWatching map[string]interface{}
sawAnyHistoryDone bool
2018-08-16 20:53:47 +02:00
}
func (d *darwinNotify) loop() {
for {
select {
case <-d.stop:
return
case events, ok := <-d.stream.Events:
if !ok {
return
}
for _, e := range events {
e.Path = filepath.Join("/", e.Path)
if e.Flags&fsevents.HistoryDone == fsevents.HistoryDone {
d.sm.Lock()
d.sawAnyHistoryDone = true
d.sm.Unlock()
continue
}
// We wait until we've seen the HistoryDone event for this watcher before processing any events
// so that we skip all of the "spurious" events that precede it.
if !d.sawAnyHistoryDone {
continue
}
_, isPathWereWatching := d.pathsWereWatching[e.Path]
if e.Flags&fsevents.ItemIsDir == fsevents.ItemIsDir && e.Flags&fsevents.ItemCreated == fsevents.ItemCreated && isPathWereWatching {
// This is the first create for the path that we're watching. We always get exactly one of these
// even after we get the HistoryDone event. Skip it.
continue
2018-08-16 20:53:47 +02:00
}
d.events <- FileEvent{
Path: e.Path,
2018-08-16 20:53:47 +02:00
}
}
}
}
}
func (d *darwinNotify) Add(name string) error {
d.sm.Lock()
defer d.sm.Unlock()
es := d.stream
// Check if this is a subdirectory of any of the paths
// we're already watching.
for _, parent := range es.Paths {
if ospath.IsChild(parent, name) {
2018-08-16 20:53:47 +02:00
return nil
}
}
es.Paths = append(es.Paths, name)
if d.pathsWereWatching == nil {
d.pathsWereWatching = make(map[string]interface{})
}
d.pathsWereWatching[name] = struct{}{}
2018-08-16 20:53:47 +02:00
if len(es.Paths) == 1 {
es.Start()
go d.loop()
2018-08-16 20:53:47 +02:00
} else {
es.Restart()
}
return nil
}
func (d *darwinNotify) Close() error {
d.sm.Lock()
defer d.sm.Unlock()
d.stream.Stop()
close(d.errors)
close(d.stop)
return nil
}
func (d *darwinNotify) Events() chan FileEvent {
2018-08-16 20:53:47 +02:00
return d.events
}
func (d *darwinNotify) Errors() chan error {
return d.errors
}
func NewWatcher(l logger.Logger) (Notify, error) {
2018-08-16 20:53:47 +02:00
dw := &darwinNotify{
stream: &fsevents.EventStream{
Latency: 1 * time.Millisecond,
Flags: fsevents.FileEvents,
// NOTE(dmiller): this corresponds to the `sinceWhen` parameter in FSEventStreamCreate
// https://developer.apple.com/documentation/coreservices/1443980-fseventstreamcreate
EventID: fsevents.LatestEventID(),
2018-08-16 20:53:47 +02:00
},
sm: &sync.Mutex{},
events: make(chan FileEvent),
2018-08-16 20:53:47 +02:00
errors: make(chan error),
stop: make(chan struct{}),
}
return dw, nil
}
var _ Notify = &darwinNotify{}