2023-01-30 11:58:55 +01:00
|
|
|
/*
|
|
|
|
Copyright 2020 Docker Compose CLI authors
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2018-08-16 20:53:47 +02:00
|
|
|
package watch
|
|
|
|
|
2019-07-11 23:54:50 +02:00
|
|
|
import (
|
|
|
|
"expvar"
|
2019-07-13 20:13:24 +02:00
|
|
|
"fmt"
|
2020-07-28 00:59:02 +02:00
|
|
|
"os"
|
2019-07-13 20:13:24 +02:00
|
|
|
"path/filepath"
|
2020-07-28 00:59:02 +02:00
|
|
|
"runtime"
|
|
|
|
"strconv"
|
2023-02-01 09:18:43 +01:00
|
|
|
|
|
|
|
"github.com/pkg/errors"
|
|
|
|
"github.com/tilt-dev/fsnotify"
|
2019-07-11 23:54:50 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
numberOfWatches = expvar.NewInt("watch.naive.numberOfWatches")
|
|
|
|
)
|
2019-07-11 17:40:40 +02:00
|
|
|
|
2018-08-22 21:33:06 +02:00
|
|
|
type FileEvent struct {
|
2019-07-13 20:13:24 +02:00
|
|
|
path string
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewFileEvent(p string) FileEvent {
|
|
|
|
if !filepath.IsAbs(p) {
|
|
|
|
panic(fmt.Sprintf("NewFileEvent only accepts absolute paths. Actual: %s", p))
|
|
|
|
}
|
|
|
|
return FileEvent{path: p}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e FileEvent) Path() string {
|
|
|
|
return e.path
|
2018-08-22 21:33:06 +02:00
|
|
|
}
|
2018-08-16 20:53:47 +02:00
|
|
|
|
|
|
|
type Notify interface {
|
2019-07-11 17:40:40 +02:00
|
|
|
// Start watching the paths set at init time
|
|
|
|
Start() error
|
|
|
|
|
|
|
|
// Stop watching and close all channels
|
2018-08-16 20:53:47 +02:00
|
|
|
Close() error
|
2019-07-11 17:40:40 +02:00
|
|
|
|
|
|
|
// A channel to read off incoming file changes
|
2018-08-22 21:33:06 +02:00
|
|
|
Events() chan FileEvent
|
2019-07-11 17:40:40 +02:00
|
|
|
|
|
|
|
// A channel to read off show-stopping errors
|
2018-08-16 20:53:47 +02:00
|
|
|
Errors() chan error
|
|
|
|
}
|
2019-07-11 17:40:40 +02:00
|
|
|
|
|
|
|
// When we specify directories to watch, we often want to
|
|
|
|
// ignore some subset of the files under those directories.
|
|
|
|
//
|
|
|
|
// For example:
|
|
|
|
// - Watch /src/repo, but ignore /src/repo/.git
|
|
|
|
// - Watch /src/repo, but ignore everything in /src/repo/bazel-bin except /src/repo/bazel-bin/app-binary
|
|
|
|
//
|
2023-06-25 13:52:03 +02:00
|
|
|
// The PathMatcher interface helps us manage these ignores.
|
2019-07-11 17:40:40 +02:00
|
|
|
type PathMatcher interface {
|
|
|
|
Matches(file string) (bool, error)
|
2019-07-16 22:23:05 +02:00
|
|
|
|
|
|
|
// If this matches the entire dir, we can often optimize filetree walks a bit.
|
|
|
|
MatchesEntireDir(file string) (bool, error)
|
2019-07-11 17:40:40 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
type EmptyMatcher struct {
|
|
|
|
}
|
|
|
|
|
2019-07-16 22:23:05 +02:00
|
|
|
func (EmptyMatcher) Matches(f string) (bool, error) { return false, nil }
|
|
|
|
func (EmptyMatcher) MatchesEntireDir(f string) (bool, error) { return false, nil }
|
2019-07-11 17:40:40 +02:00
|
|
|
|
|
|
|
var _ PathMatcher = EmptyMatcher{}
|
|
|
|
|
2023-01-30 11:58:55 +01:00
|
|
|
func NewWatcher(paths []string, ignore PathMatcher) (Notify, error) {
|
|
|
|
return newWatcher(paths, ignore)
|
2019-07-11 17:40:40 +02:00
|
|
|
}
|
2020-07-28 00:59:02 +02:00
|
|
|
|
2023-02-01 09:18:43 +01:00
|
|
|
const WindowsBufferSizeEnvVar = "COMPOSE_WATCH_WINDOWS_BUFFER_SIZE"
|
2020-07-28 00:59:02 +02:00
|
|
|
|
|
|
|
const defaultBufferSize int = 65536
|
|
|
|
|
|
|
|
func DesiredWindowsBufferSize() int {
|
|
|
|
envVar := os.Getenv(WindowsBufferSizeEnvVar)
|
|
|
|
if envVar != "" {
|
|
|
|
size, err := strconv.Atoi(envVar)
|
2020-07-28 01:36:15 +02:00
|
|
|
if err == nil {
|
2020-07-28 00:59:02 +02:00
|
|
|
return size
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return defaultBufferSize
|
|
|
|
}
|
|
|
|
|
|
|
|
func IsWindowsShortReadError(err error) bool {
|
2023-02-01 09:18:43 +01:00
|
|
|
return runtime.GOOS == "windows" && !errors.Is(err, fsnotify.ErrEventOverflow)
|
2020-07-28 00:59:02 +02:00
|
|
|
}
|
2023-02-24 17:10:15 +01:00
|
|
|
|
|
|
|
type CompositePathMatcher struct {
|
|
|
|
Matchers []PathMatcher
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewCompositeMatcher(matchers ...PathMatcher) PathMatcher {
|
|
|
|
if len(matchers) == 0 {
|
|
|
|
return EmptyMatcher{}
|
|
|
|
}
|
|
|
|
return CompositePathMatcher{Matchers: matchers}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c CompositePathMatcher) Matches(f string) (bool, error) {
|
|
|
|
for _, t := range c.Matchers {
|
|
|
|
ret, err := t.Matches(f)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
if ret {
|
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c CompositePathMatcher) MatchesEntireDir(f string) (bool, error) {
|
|
|
|
for _, t := range c.Matchers {
|
|
|
|
matches, err := t.MatchesEntireDir(f)
|
|
|
|
if matches || err != nil {
|
|
|
|
return matches, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
var _ PathMatcher = CompositePathMatcher{}
|