2018-08-16 20:53:47 +02:00
|
|
|
package watch
|
|
|
|
|
|
|
|
import (
|
2019-06-08 00:03:02 +02:00
|
|
|
"bytes"
|
|
|
|
"context"
|
2018-08-16 20:53:47 +02:00
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
2019-05-14 21:05:03 +02:00
|
|
|
"runtime"
|
2018-08-16 20:53:47 +02:00
|
|
|
"strings"
|
|
|
|
"testing"
|
|
|
|
"time"
|
2018-12-20 00:56:40 +01:00
|
|
|
|
2019-06-08 00:03:02 +02:00
|
|
|
"github.com/stretchr/testify/assert"
|
2019-06-24 17:28:29 +02:00
|
|
|
|
2019-07-11 23:54:50 +02:00
|
|
|
"github.com/windmilleng/tilt/internal/dockerignore"
|
2019-06-08 00:03:02 +02:00
|
|
|
"github.com/windmilleng/tilt/internal/logger"
|
2018-12-20 00:56:40 +01:00
|
|
|
"github.com/windmilleng/tilt/internal/testutils/tempdir"
|
2018-08-16 20:53:47 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
// Each implementation of the notify interface should have the same basic
|
|
|
|
// behavior.
|
|
|
|
|
|
|
|
func TestNoEvents(t *testing.T) {
|
|
|
|
f := newNotifyFixture(t)
|
|
|
|
defer f.tearDown()
|
|
|
|
f.assertEvents()
|
|
|
|
}
|
|
|
|
|
2019-07-16 00:57:15 +02:00
|
|
|
func TestNoWatches(t *testing.T) {
|
|
|
|
f := newNotifyFixture(t)
|
|
|
|
defer f.tearDown()
|
|
|
|
f.paths = nil
|
|
|
|
f.rebuildWatcher()
|
|
|
|
f.assertEvents()
|
|
|
|
}
|
|
|
|
|
2018-08-16 20:53:47 +02:00
|
|
|
func TestEventOrdering(t *testing.T) {
|
|
|
|
f := newNotifyFixture(t)
|
|
|
|
defer f.tearDown()
|
|
|
|
|
|
|
|
count := 8
|
|
|
|
dirs := make([]string, count)
|
|
|
|
for i, _ := range dirs {
|
2018-12-20 00:56:40 +01:00
|
|
|
dir := f.TempDir("watched")
|
|
|
|
dirs[i] = dir
|
2019-07-11 17:40:40 +02:00
|
|
|
f.watch(dir)
|
2018-08-16 20:53:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
f.fsync()
|
|
|
|
f.events = nil
|
|
|
|
|
2018-08-22 21:33:06 +02:00
|
|
|
var expected []string
|
2018-08-16 20:53:47 +02:00
|
|
|
for i, dir := range dirs {
|
|
|
|
base := fmt.Sprintf("%d.txt", i)
|
|
|
|
p := filepath.Join(dir, base)
|
|
|
|
err := ioutil.WriteFile(p, []byte(base), os.FileMode(0777))
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2018-08-22 21:33:06 +02:00
|
|
|
expected = append(expected, filepath.Join(dir, base))
|
2018-08-16 20:53:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
f.assertEvents(expected...)
|
|
|
|
}
|
|
|
|
|
2019-06-08 00:03:02 +02:00
|
|
|
// Simulate a git branch switch that creates a bunch
|
|
|
|
// of directories, creates files in them, then deletes
|
|
|
|
// them all quickly. Make sure there are no errors.
|
|
|
|
func TestGitBranchSwitch(t *testing.T) {
|
|
|
|
f := newNotifyFixture(t)
|
|
|
|
defer f.tearDown()
|
|
|
|
|
|
|
|
count := 10
|
|
|
|
dirs := make([]string, count)
|
|
|
|
for i, _ := range dirs {
|
|
|
|
dir := f.TempDir("watched")
|
|
|
|
dirs[i] = dir
|
2019-07-11 17:40:40 +02:00
|
|
|
f.watch(dir)
|
2019-06-08 00:03:02 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
f.fsync()
|
|
|
|
f.events = nil
|
|
|
|
|
|
|
|
// consume all the events in the background
|
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
|
|
done := f.consumeEventsInBackground(ctx)
|
|
|
|
|
|
|
|
for i, dir := range dirs {
|
|
|
|
for j := 0; j < count; j++ {
|
|
|
|
base := fmt.Sprintf("x/y/dir-%d/x.txt", j)
|
|
|
|
p := filepath.Join(dir, base)
|
|
|
|
f.WriteFile(p, "contents")
|
|
|
|
}
|
|
|
|
|
|
|
|
if i != 0 {
|
|
|
|
os.RemoveAll(dir)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cancel()
|
|
|
|
err := <-done
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
f.fsync()
|
|
|
|
f.events = nil
|
|
|
|
|
|
|
|
// Make sure the watch on the first dir still works.
|
|
|
|
dir := dirs[0]
|
|
|
|
path := filepath.Join(dir, "change")
|
|
|
|
|
|
|
|
f.WriteFile(path, "hello\n")
|
|
|
|
f.fsync()
|
|
|
|
|
|
|
|
f.assertEvents(path)
|
|
|
|
|
|
|
|
// Make sure there are no errors in the out stream
|
|
|
|
assert.Equal(t, "", f.out.String())
|
|
|
|
}
|
|
|
|
|
2018-08-16 20:53:47 +02:00
|
|
|
func TestWatchesAreRecursive(t *testing.T) {
|
|
|
|
f := newNotifyFixture(t)
|
|
|
|
defer f.tearDown()
|
|
|
|
|
2018-12-20 00:56:40 +01:00
|
|
|
root := f.TempDir("root")
|
2018-08-16 20:53:47 +02:00
|
|
|
|
|
|
|
// add a sub directory
|
2018-12-20 00:56:40 +01:00
|
|
|
subPath := filepath.Join(root, "sub")
|
|
|
|
f.MkdirAll(subPath)
|
2018-08-16 20:53:47 +02:00
|
|
|
|
|
|
|
// watch parent
|
2019-07-11 17:40:40 +02:00
|
|
|
f.watch(root)
|
2018-08-16 20:53:47 +02:00
|
|
|
|
|
|
|
f.fsync()
|
|
|
|
f.events = nil
|
|
|
|
// change sub directory
|
|
|
|
changeFilePath := filepath.Join(subPath, "change")
|
2019-07-11 17:40:40 +02:00
|
|
|
_, err := os.OpenFile(changeFilePath, os.O_RDONLY|os.O_CREATE, 0666)
|
2018-08-16 20:53:47 +02:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2018-08-22 21:33:06 +02:00
|
|
|
f.assertEvents(changeFilePath)
|
2018-08-16 20:53:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestNewDirectoriesAreRecursivelyWatched(t *testing.T) {
|
|
|
|
f := newNotifyFixture(t)
|
|
|
|
defer f.tearDown()
|
|
|
|
|
2018-12-20 00:56:40 +01:00
|
|
|
root := f.TempDir("root")
|
2018-08-16 20:53:47 +02:00
|
|
|
|
|
|
|
// watch parent
|
2019-07-11 17:40:40 +02:00
|
|
|
f.watch(root)
|
2018-08-16 20:53:47 +02:00
|
|
|
f.fsync()
|
|
|
|
f.events = nil
|
2018-12-20 00:56:40 +01:00
|
|
|
|
2018-08-16 20:53:47 +02:00
|
|
|
// add a sub directory
|
2018-12-20 00:56:40 +01:00
|
|
|
subPath := filepath.Join(root, "sub")
|
|
|
|
f.MkdirAll(subPath)
|
|
|
|
|
2018-08-16 20:53:47 +02:00
|
|
|
// change something inside sub directory
|
|
|
|
changeFilePath := filepath.Join(subPath, "change")
|
2019-07-11 17:40:40 +02:00
|
|
|
_, err := os.OpenFile(changeFilePath, os.O_RDONLY|os.O_CREATE, 0666)
|
2018-08-16 20:53:47 +02:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2018-08-22 21:33:06 +02:00
|
|
|
f.assertEvents(subPath, changeFilePath)
|
2018-08-16 20:53:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestWatchNonExistentPath(t *testing.T) {
|
|
|
|
f := newNotifyFixture(t)
|
|
|
|
defer f.tearDown()
|
|
|
|
|
2018-12-20 00:56:40 +01:00
|
|
|
root := f.TempDir("root")
|
|
|
|
path := filepath.Join(root, "change")
|
2018-08-16 20:53:47 +02:00
|
|
|
|
2019-07-11 17:40:40 +02:00
|
|
|
f.watch(path)
|
2018-08-23 19:24:47 +02:00
|
|
|
f.fsync()
|
|
|
|
|
2018-12-20 00:56:40 +01:00
|
|
|
d1 := "hello\ngo\n"
|
|
|
|
f.WriteFile(path, d1)
|
2018-08-22 21:33:06 +02:00
|
|
|
f.assertEvents(path)
|
2018-08-16 20:53:47 +02:00
|
|
|
}
|
|
|
|
|
2019-03-22 21:17:12 +01:00
|
|
|
func TestWatchNonExistentPathDoesNotFireSiblingEvent(t *testing.T) {
|
|
|
|
f := newNotifyFixture(t)
|
|
|
|
defer f.tearDown()
|
|
|
|
|
|
|
|
root := f.TempDir("root")
|
|
|
|
watchedFile := filepath.Join(root, "a.txt")
|
|
|
|
unwatchedSibling := filepath.Join(root, "b.txt")
|
|
|
|
|
2019-07-11 17:40:40 +02:00
|
|
|
f.watch(watchedFile)
|
2019-03-22 21:17:12 +01:00
|
|
|
f.fsync()
|
|
|
|
|
|
|
|
d1 := "hello\ngo\n"
|
|
|
|
f.WriteFile(unwatchedSibling, d1)
|
|
|
|
f.assertEvents()
|
|
|
|
}
|
|
|
|
|
2018-08-16 20:53:47 +02:00
|
|
|
func TestRemove(t *testing.T) {
|
|
|
|
f := newNotifyFixture(t)
|
|
|
|
defer f.tearDown()
|
|
|
|
|
2018-12-20 00:56:40 +01:00
|
|
|
root := f.TempDir("root")
|
|
|
|
path := filepath.Join(root, "change")
|
2018-08-16 20:53:47 +02:00
|
|
|
|
2018-12-20 00:56:40 +01:00
|
|
|
d1 := "hello\ngo\n"
|
|
|
|
f.WriteFile(path, d1)
|
2018-08-16 20:53:47 +02:00
|
|
|
|
2019-07-11 17:40:40 +02:00
|
|
|
f.watch(path)
|
2018-08-16 20:53:47 +02:00
|
|
|
f.fsync()
|
|
|
|
f.events = nil
|
2019-07-11 17:40:40 +02:00
|
|
|
err := os.Remove(path)
|
2018-08-16 20:53:47 +02:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2018-08-22 21:33:06 +02:00
|
|
|
f.assertEvents(path)
|
2018-08-16 20:53:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestRemoveAndAddBack(t *testing.T) {
|
|
|
|
f := newNotifyFixture(t)
|
|
|
|
defer f.tearDown()
|
|
|
|
|
2019-07-11 17:40:40 +02:00
|
|
|
path := filepath.Join(f.paths[0], "change")
|
2018-08-16 20:53:47 +02:00
|
|
|
|
|
|
|
d1 := []byte("hello\ngo\n")
|
2018-08-22 21:48:33 +02:00
|
|
|
err := ioutil.WriteFile(path, d1, 0644)
|
2018-08-16 20:53:47 +02:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2019-07-11 17:40:40 +02:00
|
|
|
f.watch(path)
|
2018-08-22 21:48:33 +02:00
|
|
|
f.assertEvents(path)
|
|
|
|
|
2018-08-16 20:53:47 +02:00
|
|
|
err = os.Remove(path)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2018-08-22 21:33:06 +02:00
|
|
|
f.assertEvents(path)
|
2018-08-16 20:53:47 +02:00
|
|
|
f.events = nil
|
|
|
|
|
|
|
|
err = ioutil.WriteFile(path, d1, 0644)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2018-08-22 21:33:06 +02:00
|
|
|
f.assertEvents(path)
|
2018-08-16 20:53:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestSingleFile(t *testing.T) {
|
|
|
|
f := newNotifyFixture(t)
|
|
|
|
defer f.tearDown()
|
|
|
|
|
2018-12-20 00:56:40 +01:00
|
|
|
root := f.TempDir("root")
|
|
|
|
path := filepath.Join(root, "change")
|
2018-08-16 20:53:47 +02:00
|
|
|
|
2018-12-20 00:56:40 +01:00
|
|
|
d1 := "hello\ngo\n"
|
|
|
|
f.WriteFile(path, d1)
|
2018-08-16 20:53:47 +02:00
|
|
|
|
2019-07-11 17:40:40 +02:00
|
|
|
f.watch(path)
|
2018-08-23 17:56:00 +02:00
|
|
|
f.fsync()
|
2018-08-16 20:53:47 +02:00
|
|
|
|
|
|
|
d2 := []byte("hello\nworld\n")
|
2019-07-11 17:40:40 +02:00
|
|
|
err := ioutil.WriteFile(path, d2, 0644)
|
2018-08-16 20:53:47 +02:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2018-08-22 21:33:06 +02:00
|
|
|
f.assertEvents(path)
|
2018-08-16 20:53:47 +02:00
|
|
|
}
|
|
|
|
|
2018-08-22 21:59:46 +02:00
|
|
|
func TestWriteBrokenLink(t *testing.T) {
|
|
|
|
f := newNotifyFixture(t)
|
|
|
|
defer f.tearDown()
|
|
|
|
|
2019-07-11 17:40:40 +02:00
|
|
|
link := filepath.Join(f.paths[0], "brokenLink")
|
|
|
|
missingFile := filepath.Join(f.paths[0], "missingFile")
|
2018-08-22 21:59:46 +02:00
|
|
|
err := os.Symlink(missingFile, link)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
f.assertEvents(link)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestWriteGoodLink(t *testing.T) {
|
|
|
|
f := newNotifyFixture(t)
|
|
|
|
defer f.tearDown()
|
|
|
|
|
2019-07-11 17:40:40 +02:00
|
|
|
goodFile := filepath.Join(f.paths[0], "goodFile")
|
2018-08-22 21:59:46 +02:00
|
|
|
err := ioutil.WriteFile(goodFile, []byte("hello"), 0644)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2019-07-11 17:40:40 +02:00
|
|
|
link := filepath.Join(f.paths[0], "goodFileSymlink")
|
2018-08-22 21:59:46 +02:00
|
|
|
err = os.Symlink(goodFile, link)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
f.assertEvents(goodFile, link)
|
|
|
|
}
|
|
|
|
|
2018-09-14 23:13:36 +02:00
|
|
|
func TestWatchBrokenLink(t *testing.T) {
|
|
|
|
f := newNotifyFixture(t)
|
|
|
|
defer f.tearDown()
|
|
|
|
|
|
|
|
newRoot, err := NewDir(t.Name())
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer newRoot.TearDown()
|
|
|
|
|
|
|
|
link := filepath.Join(newRoot.Path(), "brokenLink")
|
|
|
|
missingFile := filepath.Join(newRoot.Path(), "missingFile")
|
|
|
|
err = os.Symlink(missingFile, link)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
2019-07-11 17:40:40 +02:00
|
|
|
f.watch(newRoot.Path())
|
2018-09-14 23:13:36 +02:00
|
|
|
os.Remove(link)
|
|
|
|
f.assertEvents(link)
|
|
|
|
}
|
|
|
|
|
2018-10-18 16:44:07 +02:00
|
|
|
func TestMoveAndReplace(t *testing.T) {
|
|
|
|
f := newNotifyFixture(t)
|
|
|
|
defer f.tearDown()
|
|
|
|
|
2018-12-20 00:56:40 +01:00
|
|
|
root := f.TempDir("root")
|
|
|
|
file := filepath.Join(root, "myfile")
|
|
|
|
f.WriteFile(file, "hello")
|
2018-10-18 16:44:07 +02:00
|
|
|
|
2019-07-11 17:40:40 +02:00
|
|
|
f.watch(file)
|
2018-12-20 00:56:40 +01:00
|
|
|
tmpFile := filepath.Join(root, ".myfile.swp")
|
|
|
|
f.WriteFile(tmpFile, "world")
|
2018-10-18 16:44:07 +02:00
|
|
|
|
2019-07-11 17:40:40 +02:00
|
|
|
err := os.Rename(tmpFile, file)
|
2018-10-18 16:44:07 +02:00
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
f.assertEvents(file)
|
|
|
|
}
|
|
|
|
|
2018-12-20 18:20:34 +01:00
|
|
|
func TestWatchBothDirAndFile(t *testing.T) {
|
|
|
|
f := newNotifyFixture(t)
|
|
|
|
defer f.tearDown()
|
|
|
|
|
|
|
|
dir := f.JoinPath("foo")
|
|
|
|
fileA := f.JoinPath("foo", "a")
|
|
|
|
fileB := f.JoinPath("foo", "b")
|
|
|
|
f.WriteFile(fileA, "a")
|
|
|
|
f.WriteFile(fileB, "b")
|
|
|
|
|
|
|
|
f.watch(fileA)
|
|
|
|
f.watch(dir)
|
|
|
|
f.fsync()
|
|
|
|
f.events = nil
|
|
|
|
|
|
|
|
f.WriteFile(fileB, "b-new")
|
|
|
|
f.assertEvents(fileB)
|
|
|
|
}
|
|
|
|
|
2019-05-14 21:05:03 +02:00
|
|
|
func TestWatchNonexistentFileInNonexistentDirectoryCreatedSimultaneously(t *testing.T) {
|
2019-01-16 20:57:50 +01:00
|
|
|
f := newNotifyFixture(t)
|
|
|
|
defer f.tearDown()
|
|
|
|
|
|
|
|
root := f.JoinPath("root")
|
|
|
|
err := os.Mkdir(root, 0777)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
file := f.JoinPath("root", "parent", "a")
|
|
|
|
|
|
|
|
f.watch(file)
|
|
|
|
f.fsync()
|
|
|
|
f.events = nil
|
|
|
|
f.WriteFile(file, "hello")
|
2019-03-22 21:17:12 +01:00
|
|
|
f.assertEvents(file)
|
2019-01-16 20:57:50 +01:00
|
|
|
}
|
|
|
|
|
2019-05-14 21:05:03 +02:00
|
|
|
func TestWatchNonexistentDirectory(t *testing.T) {
|
|
|
|
f := newNotifyFixture(t)
|
|
|
|
defer f.tearDown()
|
|
|
|
|
|
|
|
root := f.JoinPath("root")
|
|
|
|
err := os.Mkdir(root, 0777)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
parent := f.JoinPath("parent")
|
|
|
|
file := f.JoinPath("parent", "a")
|
|
|
|
|
|
|
|
f.watch(parent)
|
|
|
|
f.fsync()
|
|
|
|
f.events = nil
|
|
|
|
|
|
|
|
err = os.Mkdir(parent, 0777)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if runtime.GOOS == "darwin" {
|
|
|
|
// for directories that were the root of an Add, we don't report creation, cf. watcher_darwin.go
|
|
|
|
f.assertEvents()
|
|
|
|
} else {
|
|
|
|
f.assertEvents(parent)
|
|
|
|
}
|
|
|
|
f.WriteFile(file, "hello")
|
|
|
|
|
|
|
|
if runtime.GOOS == "darwin" {
|
|
|
|
// mac doesn't return the dir change as part of file creation
|
|
|
|
f.assertEvents(file)
|
|
|
|
} else {
|
|
|
|
f.assertEvents(parent, file)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-11 23:54:50 +02:00
|
|
|
func TestWatchNonexistentFileInNonexistentDirectory(t *testing.T) {
|
|
|
|
f := newNotifyFixture(t)
|
|
|
|
defer f.tearDown()
|
|
|
|
|
|
|
|
root := f.JoinPath("root")
|
|
|
|
err := os.Mkdir(root, 0777)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
parent := f.JoinPath("parent")
|
|
|
|
file := f.JoinPath("parent", "a")
|
|
|
|
|
|
|
|
f.watch(file)
|
|
|
|
f.assertEvents()
|
2019-05-14 21:05:03 +02:00
|
|
|
|
2019-07-11 23:54:50 +02:00
|
|
|
err = os.Mkdir(parent, 0777)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
2019-05-14 21:05:03 +02:00
|
|
|
|
2019-07-11 23:54:50 +02:00
|
|
|
f.assertEvents()
|
|
|
|
f.WriteFile(file, "hello")
|
|
|
|
f.assertEvents(file)
|
|
|
|
}
|
2019-05-14 21:05:03 +02:00
|
|
|
|
2019-07-11 23:54:50 +02:00
|
|
|
func TestWatchCountInnerFile(t *testing.T) {
|
|
|
|
f := newNotifyFixture(t)
|
|
|
|
defer f.tearDown()
|
2019-05-14 21:05:03 +02:00
|
|
|
|
2019-07-11 23:54:50 +02:00
|
|
|
root := f.paths[0]
|
|
|
|
a := f.JoinPath(root, "a")
|
|
|
|
b := f.JoinPath(a, "b")
|
|
|
|
file := f.JoinPath(b, "bigFile")
|
|
|
|
f.WriteFile(file, "hello")
|
|
|
|
f.assertEvents(a, b, file)
|
|
|
|
|
|
|
|
expectedWatches := 3
|
|
|
|
if runtime.GOOS == "darwin" {
|
|
|
|
expectedWatches = 1
|
|
|
|
}
|
|
|
|
assert.Equal(t, expectedWatches, int(numberOfWatches.Value()))
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestWatchCountInnerFileWithIgnore(t *testing.T) {
|
|
|
|
f := newNotifyFixture(t)
|
|
|
|
defer f.tearDown()
|
|
|
|
|
|
|
|
root := f.paths[0]
|
|
|
|
ignore, _ := dockerignore.NewDockerPatternMatcher(root, []string{
|
|
|
|
"a",
|
|
|
|
"!a/b",
|
|
|
|
})
|
|
|
|
f.setIgnore(ignore)
|
|
|
|
|
|
|
|
a := f.JoinPath(root, "a")
|
|
|
|
b := f.JoinPath(a, "b")
|
|
|
|
file := f.JoinPath(b, "bigFile")
|
|
|
|
f.WriteFile(file, "hello")
|
|
|
|
f.assertEvents(b, file)
|
|
|
|
|
|
|
|
expectedWatches := 3
|
|
|
|
if runtime.GOOS == "darwin" {
|
|
|
|
expectedWatches = 1
|
|
|
|
}
|
|
|
|
assert.Equal(t, expectedWatches, int(numberOfWatches.Value()))
|
|
|
|
}
|
|
|
|
|
2019-07-16 22:12:08 +02:00
|
|
|
func TestIgnoreCreatedDir(t *testing.T) {
|
2019-07-11 23:54:50 +02:00
|
|
|
f := newNotifyFixture(t)
|
|
|
|
defer f.tearDown()
|
|
|
|
|
|
|
|
root := f.paths[0]
|
|
|
|
ignore, _ := dockerignore.NewDockerPatternMatcher(root, []string{"a/b"})
|
|
|
|
f.setIgnore(ignore)
|
|
|
|
|
|
|
|
a := f.JoinPath(root, "a")
|
|
|
|
b := f.JoinPath(a, "b")
|
|
|
|
file := f.JoinPath(b, "bigFile")
|
|
|
|
f.WriteFile(file, "hello")
|
|
|
|
f.assertEvents(a)
|
|
|
|
|
2019-07-16 22:12:08 +02:00
|
|
|
expectedWatches := 2
|
2019-07-11 23:54:50 +02:00
|
|
|
if runtime.GOOS == "darwin" {
|
|
|
|
expectedWatches = 1
|
|
|
|
}
|
|
|
|
assert.Equal(t, expectedWatches, int(numberOfWatches.Value()))
|
|
|
|
}
|
2019-05-14 21:05:03 +02:00
|
|
|
|
2019-07-16 22:12:08 +02:00
|
|
|
func TestIgnoreInitialDir(t *testing.T) {
|
|
|
|
f := newNotifyFixture(t)
|
|
|
|
defer f.tearDown()
|
|
|
|
|
|
|
|
root := f.TempDir("root")
|
|
|
|
ignore, _ := dockerignore.NewDockerPatternMatcher(root, []string{"a/b"})
|
|
|
|
f.setIgnore(ignore)
|
|
|
|
|
|
|
|
a := f.JoinPath(root, "a")
|
|
|
|
b := f.JoinPath(a, "b")
|
|
|
|
file := f.JoinPath(b, "bigFile")
|
|
|
|
f.WriteFile(file, "hello")
|
|
|
|
f.watch(root)
|
|
|
|
|
|
|
|
f.assertEvents()
|
|
|
|
|
|
|
|
expectedWatches := 3
|
|
|
|
if runtime.GOOS == "darwin" {
|
|
|
|
expectedWatches = 2
|
|
|
|
}
|
|
|
|
assert.Equal(t, expectedWatches, int(numberOfWatches.Value()))
|
|
|
|
}
|
|
|
|
|
2018-08-16 20:53:47 +02:00
|
|
|
type notifyFixture struct {
|
2019-06-08 00:03:02 +02:00
|
|
|
out *bytes.Buffer
|
2018-12-20 00:56:40 +01:00
|
|
|
*tempdir.TempDirFixture
|
2019-07-11 17:40:40 +02:00
|
|
|
notify Notify
|
2019-07-11 23:54:50 +02:00
|
|
|
ignore PathMatcher
|
2019-07-11 17:40:40 +02:00
|
|
|
paths []string
|
|
|
|
events []FileEvent
|
2018-08-16 20:53:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func newNotifyFixture(t *testing.T) *notifyFixture {
|
2019-06-08 00:03:02 +02:00
|
|
|
out := bytes.NewBuffer(nil)
|
2019-07-11 17:40:40 +02:00
|
|
|
nf := ¬ifyFixture{
|
|
|
|
TempDirFixture: tempdir.NewTempDirFixture(t),
|
|
|
|
paths: []string{},
|
2019-07-11 23:54:50 +02:00
|
|
|
ignore: EmptyMatcher{},
|
2019-06-08 00:03:02 +02:00
|
|
|
out: out,
|
2018-08-16 20:53:47 +02:00
|
|
|
}
|
2019-07-11 17:40:40 +02:00
|
|
|
nf.watch(nf.TempDir("watched"))
|
|
|
|
return nf
|
2018-08-16 20:53:47 +02:00
|
|
|
}
|
|
|
|
|
2019-07-11 23:54:50 +02:00
|
|
|
func (f *notifyFixture) setIgnore(ignore PathMatcher) {
|
|
|
|
f.ignore = ignore
|
|
|
|
f.rebuildWatcher()
|
|
|
|
}
|
|
|
|
|
2018-12-20 18:20:34 +01:00
|
|
|
func (f *notifyFixture) watch(path string) {
|
2019-07-11 17:40:40 +02:00
|
|
|
f.paths = append(f.paths, path)
|
2019-07-11 23:54:50 +02:00
|
|
|
f.rebuildWatcher()
|
|
|
|
}
|
2019-07-11 17:40:40 +02:00
|
|
|
|
2019-07-11 23:54:50 +02:00
|
|
|
func (f *notifyFixture) rebuildWatcher() {
|
2019-07-11 17:40:40 +02:00
|
|
|
// sync any outstanding events and close the old watcher
|
|
|
|
if f.notify != nil {
|
|
|
|
f.fsync()
|
|
|
|
f.closeWatcher()
|
|
|
|
}
|
|
|
|
|
|
|
|
// create a new watcher
|
2019-07-11 23:54:50 +02:00
|
|
|
notify, err := NewWatcher(f.paths, f.ignore, logger.NewLogger(logger.DebugLvl, f.out))
|
2018-12-20 18:20:34 +01:00
|
|
|
if err != nil {
|
2019-07-11 17:40:40 +02:00
|
|
|
f.T().Fatal(err)
|
|
|
|
}
|
|
|
|
f.notify = notify
|
|
|
|
err = f.notify.Start()
|
|
|
|
if err != nil {
|
|
|
|
f.T().Fatal(err)
|
2018-12-20 18:20:34 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-22 21:33:06 +02:00
|
|
|
func (f *notifyFixture) assertEvents(expected ...string) {
|
|
|
|
f.fsync()
|
2018-08-16 20:53:47 +02:00
|
|
|
|
|
|
|
if len(f.events) != len(expected) {
|
2018-12-20 00:56:40 +01:00
|
|
|
f.T().Fatalf("Got %d events (expected %d): %v %v", len(f.events), len(expected), f.events, expected)
|
2018-08-16 20:53:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
for i, actual := range f.events {
|
2018-08-22 21:33:06 +02:00
|
|
|
e := FileEvent{expected[i]}
|
|
|
|
if actual != e {
|
2018-12-20 00:56:40 +01:00
|
|
|
f.T().Fatalf("Got event %v (expected %v)", actual, e)
|
2018-08-16 20:53:47 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-08 00:03:02 +02:00
|
|
|
func (f *notifyFixture) consumeEventsInBackground(ctx context.Context) chan error {
|
|
|
|
done := make(chan error)
|
|
|
|
go func() {
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-ctx.Done():
|
|
|
|
close(done)
|
|
|
|
return
|
|
|
|
case err := <-f.notify.Errors():
|
|
|
|
done <- err
|
|
|
|
close(done)
|
|
|
|
return
|
|
|
|
case <-f.notify.Events():
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
return done
|
|
|
|
}
|
|
|
|
|
2018-08-16 20:53:47 +02:00
|
|
|
func (f *notifyFixture) fsync() {
|
2019-07-16 00:57:15 +02:00
|
|
|
if len(f.paths) == 0 {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2018-08-16 20:53:47 +02:00
|
|
|
syncPathBase := fmt.Sprintf("sync-%d.txt", time.Now().UnixNano())
|
2019-07-11 17:40:40 +02:00
|
|
|
syncPath := filepath.Join(f.paths[0], syncPathBase)
|
|
|
|
anySyncPath := filepath.Join(f.paths[0], "sync-")
|
2018-08-16 20:53:47 +02:00
|
|
|
timeout := time.After(time.Second)
|
|
|
|
|
2018-12-20 00:56:40 +01:00
|
|
|
f.WriteFile(syncPath, fmt.Sprintf("%s", time.Now()))
|
2018-08-16 20:53:47 +02:00
|
|
|
|
|
|
|
F:
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case err := <-f.notify.Errors():
|
2018-12-20 00:56:40 +01:00
|
|
|
f.T().Fatal(err)
|
2018-08-16 20:53:47 +02:00
|
|
|
|
|
|
|
case event := <-f.notify.Events():
|
2019-07-13 20:13:24 +02:00
|
|
|
if strings.Contains(event.Path(), syncPath) {
|
2018-08-16 20:53:47 +02:00
|
|
|
break F
|
|
|
|
}
|
2019-07-13 20:13:24 +02:00
|
|
|
if strings.Contains(event.Path(), anySyncPath) {
|
2018-08-16 20:53:47 +02:00
|
|
|
continue
|
|
|
|
}
|
2018-08-22 21:33:06 +02:00
|
|
|
|
|
|
|
// Don't bother tracking duplicate changes to the same path
|
|
|
|
// for testing.
|
2019-07-13 20:13:24 +02:00
|
|
|
if len(f.events) > 0 && f.events[len(f.events)-1].Path() == event.Path() {
|
2018-08-22 21:33:06 +02:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2018-08-16 20:53:47 +02:00
|
|
|
f.events = append(f.events, event)
|
|
|
|
|
|
|
|
case <-timeout:
|
2018-12-20 00:56:40 +01:00
|
|
|
f.T().Fatalf("fsync: timeout")
|
2018-08-16 20:53:47 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-11 17:40:40 +02:00
|
|
|
func (f *notifyFixture) closeWatcher() {
|
|
|
|
notify := f.notify
|
|
|
|
err := notify.Close()
|
2018-08-16 20:53:47 +02:00
|
|
|
if err != nil {
|
2018-12-20 00:56:40 +01:00
|
|
|
f.T().Fatal(err)
|
2018-08-16 20:53:47 +02:00
|
|
|
}
|
2018-12-20 00:56:40 +01:00
|
|
|
|
2019-05-14 21:05:03 +02:00
|
|
|
// drain channels from watcher
|
|
|
|
go func() {
|
2019-07-11 17:40:40 +02:00
|
|
|
for _ = range notify.Events() {
|
2019-05-14 21:05:03 +02:00
|
|
|
}
|
|
|
|
}()
|
|
|
|
go func() {
|
2019-07-11 17:40:40 +02:00
|
|
|
for _ = range notify.Errors() {
|
2019-05-14 21:05:03 +02:00
|
|
|
}
|
|
|
|
}()
|
2019-07-11 17:40:40 +02:00
|
|
|
}
|
2019-05-14 21:05:03 +02:00
|
|
|
|
2019-07-11 17:40:40 +02:00
|
|
|
func (f *notifyFixture) tearDown() {
|
|
|
|
f.closeWatcher()
|
2018-12-20 00:56:40 +01:00
|
|
|
f.TempDirFixture.TearDown()
|
2019-07-11 23:54:50 +02:00
|
|
|
numberOfWatches.Set(0)
|
2018-08-16 20:53:47 +02:00
|
|
|
}
|