mirror of
https://git.mirrors.martin98.com/https://github.com/ceph/ceph-csi.git
synced 2025-08-18 17:35:54 +08:00

Several packages are only used while running the e2e suite. These packages are less important to update, as the they can not influence the final executable that is part of the Ceph-CSI container-image. By moving these dependencies out of the main Ceph-CSI go.mod, it is easier to identify if a reported CVE affects Ceph-CSI, or only the testing (like most of the Kubernetes CVEs). Signed-off-by: Niels de Vos <ndevos@ibm.com>
217 lines
5.7 KiB
Go
217 lines
5.7 KiB
Go
/*
|
|
Copyright 2017 The Kubernetes 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.
|
|
*/
|
|
|
|
package filesystem
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"time"
|
|
|
|
"github.com/fsnotify/fsnotify"
|
|
)
|
|
|
|
// FSWatcher is a callback-based filesystem watcher abstraction for fsnotify.
|
|
type FSWatcher interface {
|
|
// Initializes the watcher with the given watch handlers.
|
|
// Called before all other methods.
|
|
Init(FSEventHandler, FSErrorHandler) error
|
|
|
|
// Starts listening for events and errors.
|
|
// When an event or error occurs, the corresponding handler is called.
|
|
Run()
|
|
|
|
// Add a filesystem path to watch
|
|
AddWatch(path string) error
|
|
}
|
|
|
|
// FSEventHandler is called when a fsnotify event occurs.
|
|
type FSEventHandler func(event fsnotify.Event)
|
|
|
|
// FSErrorHandler is called when a fsnotify error occurs.
|
|
type FSErrorHandler func(err error)
|
|
|
|
type fsnotifyWatcher struct {
|
|
watcher *fsnotify.Watcher
|
|
eventHandler FSEventHandler
|
|
errorHandler FSErrorHandler
|
|
}
|
|
|
|
var _ FSWatcher = &fsnotifyWatcher{}
|
|
|
|
// NewFsnotifyWatcher returns an implementation of FSWatcher that continuously listens for
|
|
// fsnotify events and calls the event handler as soon as an event is received.
|
|
func NewFsnotifyWatcher() FSWatcher {
|
|
return &fsnotifyWatcher{}
|
|
}
|
|
|
|
func (w *fsnotifyWatcher) AddWatch(path string) error {
|
|
return w.watcher.Add(path)
|
|
}
|
|
|
|
func (w *fsnotifyWatcher) Init(eventHandler FSEventHandler, errorHandler FSErrorHandler) error {
|
|
var err error
|
|
w.watcher, err = fsnotify.NewWatcher()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
w.eventHandler = eventHandler
|
|
w.errorHandler = errorHandler
|
|
return nil
|
|
}
|
|
|
|
func (w *fsnotifyWatcher) Run() {
|
|
go func() {
|
|
defer w.watcher.Close()
|
|
for {
|
|
select {
|
|
case event := <-w.watcher.Events:
|
|
if w.eventHandler != nil {
|
|
w.eventHandler(event)
|
|
}
|
|
case err := <-w.watcher.Errors:
|
|
if w.errorHandler != nil {
|
|
w.errorHandler(err)
|
|
}
|
|
}
|
|
}
|
|
}()
|
|
}
|
|
|
|
type watchAddRemover interface {
|
|
Add(path string) error
|
|
Remove(path string) error
|
|
}
|
|
type noopWatcher struct{}
|
|
|
|
func (noopWatcher) Add(path string) error { return nil }
|
|
func (noopWatcher) Remove(path string) error { return nil }
|
|
|
|
// WatchUntil watches the specified path for changes and blocks until ctx is canceled.
|
|
// eventHandler() must be non-nil, and pollInterval must be greater than 0.
|
|
// eventHandler() is invoked whenever a change event is observed or pollInterval elapses.
|
|
// errorHandler() is invoked (if non-nil) whenever an error occurs initializing or watching the specified path.
|
|
//
|
|
// If path is a directory, only the directory and immediate children are watched.
|
|
//
|
|
// If path does not exist or cannot be watched, an error is passed to errorHandler() and eventHandler() is called at pollInterval.
|
|
//
|
|
// Multiple observed events may collapse to a single invocation of eventHandler().
|
|
//
|
|
// eventHandler() is invoked immediately after successful initialization of the filesystem watch,
|
|
// in case the path changed concurrent with calling WatchUntil().
|
|
func WatchUntil(ctx context.Context, pollInterval time.Duration, path string, eventHandler func(), errorHandler func(err error)) {
|
|
if pollInterval <= 0 {
|
|
panic(fmt.Errorf("pollInterval must be > 0"))
|
|
}
|
|
if eventHandler == nil {
|
|
panic(fmt.Errorf("eventHandler must be non-nil"))
|
|
}
|
|
if errorHandler == nil {
|
|
errorHandler = func(err error) {}
|
|
}
|
|
|
|
// Initialize watcher, fall back to no-op
|
|
var (
|
|
eventsCh chan fsnotify.Event
|
|
errorCh chan error
|
|
watcher watchAddRemover
|
|
)
|
|
if w, err := fsnotify.NewWatcher(); err != nil {
|
|
errorHandler(fmt.Errorf("error creating file watcher, falling back to poll at interval %s: %w", pollInterval, err))
|
|
watcher = noopWatcher{}
|
|
} else {
|
|
watcher = w
|
|
eventsCh = w.Events
|
|
errorCh = w.Errors
|
|
defer func() {
|
|
_ = w.Close()
|
|
}()
|
|
}
|
|
|
|
// Initialize background poll
|
|
t := time.NewTicker(pollInterval)
|
|
defer t.Stop()
|
|
|
|
attemptPeriodicRewatch := false
|
|
|
|
// Start watching the path
|
|
if err := watcher.Add(path); err != nil {
|
|
errorHandler(err)
|
|
attemptPeriodicRewatch = true
|
|
} else {
|
|
// Invoke handle() at least once after successfully registering the listener,
|
|
// in case the file changed concurrent with calling WatchUntil.
|
|
eventHandler()
|
|
}
|
|
|
|
for {
|
|
select {
|
|
case <-ctx.Done():
|
|
return
|
|
|
|
case <-t.C:
|
|
// Prioritize exiting if context is canceled
|
|
if ctx.Err() != nil {
|
|
return
|
|
}
|
|
|
|
// Try to re-establish the watcher if we previously got a watch error
|
|
if attemptPeriodicRewatch {
|
|
_ = watcher.Remove(path)
|
|
if err := watcher.Add(path); err != nil {
|
|
errorHandler(err)
|
|
} else {
|
|
attemptPeriodicRewatch = false
|
|
}
|
|
}
|
|
|
|
// Handle
|
|
eventHandler()
|
|
|
|
case e := <-eventsCh:
|
|
// Prioritize exiting if context is canceled
|
|
if ctx.Err() != nil {
|
|
return
|
|
}
|
|
|
|
// Try to re-establish the watcher for events which dropped the existing watch
|
|
if e.Name == path && (e.Has(fsnotify.Remove) || e.Has(fsnotify.Rename)) {
|
|
_ = watcher.Remove(path)
|
|
if err := watcher.Add(path); err != nil {
|
|
errorHandler(err)
|
|
attemptPeriodicRewatch = true
|
|
}
|
|
}
|
|
|
|
// Handle
|
|
eventHandler()
|
|
|
|
case err := <-errorCh:
|
|
// Prioritize exiting if context is canceled
|
|
if ctx.Err() != nil {
|
|
return
|
|
}
|
|
|
|
// If the error occurs in response to calling watcher.Add, re-adding here could hot-loop.
|
|
// The periodic poll will attempt to re-establish the watch.
|
|
errorHandler(err)
|
|
attemptPeriodicRewatch = true
|
|
}
|
|
}
|
|
}
|