mirror of
https://github.com/oauth2-proxy/oauth2-proxy.git
synced 2024-11-24 08:52:25 +02:00
7fe6384f38
* initial commit: add groups to azure Signed-off-by: andrewphamade@gmail.com <andrewphamade@gmail.com> * fix deprecations and linting errors Signed-off-by: Andrew Hamade <andrewphamade@gmail.com> * remove groups testing from azure provider Signed-off-by: Andrew Hamade <andrewphamade@gmail.com> * fix test error Signed-off-by: Andrew Hamade <andrewphamade@gmail.com> * verify-generate Signed-off-by: Andrew Hamade <andrewphamade@gmail.com> Signed-off-by: andrewphamade@gmail.com <andrewphamade@gmail.com> Signed-off-by: Andrew Hamade <andrewphamade@gmail.com>
82 lines
2.3 KiB
Go
82 lines
2.3 KiB
Go
package watcher
|
|
|
|
import (
|
|
"fmt"
|
|
"os"
|
|
"path/filepath"
|
|
"time"
|
|
|
|
"github.com/fsnotify/fsnotify"
|
|
|
|
"github.com/oauth2-proxy/oauth2-proxy/v7/pkg/logger"
|
|
)
|
|
|
|
// WatchFileForUpdates performs an action every time a file on disk is updated
|
|
func WatchFileForUpdates(filename string, done <-chan bool, action func()) error {
|
|
filename = filepath.Clean(filename)
|
|
watcher, err := fsnotify.NewWatcher()
|
|
if err != nil {
|
|
return fmt.Errorf("failed to create watcher for '%s': %s", filename, err)
|
|
}
|
|
|
|
go func() {
|
|
defer watcher.Close()
|
|
|
|
for {
|
|
select {
|
|
case <-done:
|
|
logger.Printf("shutting down watcher for: %s", filename)
|
|
return
|
|
case event := <-watcher.Events:
|
|
filterEvent(watcher, event, filename, action)
|
|
case err = <-watcher.Errors:
|
|
logger.Errorf("error watching '%s': %s", filename, err)
|
|
}
|
|
}
|
|
}()
|
|
if err := watcher.Add(filename); err != nil {
|
|
return fmt.Errorf("failed to add '%s' to watcher: %v", filename, err)
|
|
}
|
|
logger.Printf("watching '%s' for updates", filename)
|
|
|
|
return nil
|
|
}
|
|
|
|
// Filter file operations based on the events sent by the watcher.
|
|
// Execute the action() function when the following conditions are met:
|
|
// - the real path of the file was changed (Kubernetes ConfigMap/Secret)
|
|
// - the file is modified or created
|
|
func filterEvent(watcher *fsnotify.Watcher, event fsnotify.Event, filename string, action func()) {
|
|
switch filepath.Clean(event.Name) == filename {
|
|
// In Kubernetes the file path is a symlink, so we should take action
|
|
// when the ConfigMap/Secret is replaced.
|
|
case event.Op&fsnotify.Remove != 0:
|
|
logger.Printf("watching interrupted on event: %s", event)
|
|
WaitForReplacement(filename, event.Op, watcher)
|
|
action()
|
|
case event.Op&(fsnotify.Create|fsnotify.Write) != 0:
|
|
logger.Printf("reloading after event: %s", event)
|
|
action()
|
|
}
|
|
}
|
|
|
|
// WaitForReplacement waits for a file to exist on disk and then starts a watch
|
|
// for the file
|
|
func WaitForReplacement(filename string, op fsnotify.Op, watcher *fsnotify.Watcher) {
|
|
const sleepInterval = 50 * time.Millisecond
|
|
|
|
// Avoid a race when fsnofity.Remove is preceded by fsnotify.Chmod.
|
|
if op&fsnotify.Chmod != 0 {
|
|
time.Sleep(sleepInterval)
|
|
}
|
|
for {
|
|
if _, err := os.Stat(filename); err == nil {
|
|
if err := watcher.Add(filename); err == nil {
|
|
logger.Printf("watching resumed for '%s'", filename)
|
|
return
|
|
}
|
|
}
|
|
time.Sleep(sleepInterval)
|
|
}
|
|
}
|