diff options
-rw-r--r-- | .github/workflows/build.yml | 1 | ||||
-rwxr-xr-x | Makefile | 1 | ||||
-rwxr-xr-x | bors.toml | 20 | ||||
-rw-r--r-- | plugins/reload/plugin.go | 34 | ||||
-rw-r--r-- | plugins/reload/tests/reload_plugin_test.go | 2 | ||||
-rw-r--r-- | plugins/reload/tests/watcher_test.go | 37 | ||||
-rw-r--r-- | plugins/reload/watcher.go | 17 |
7 files changed, 76 insertions, 36 deletions
diff --git a/.github/workflows/build.yml b/.github/workflows/build.yml index 83e9741d..29a42af2 100644 --- a/.github/workflows/build.yml +++ b/.github/workflows/build.yml @@ -96,6 +96,7 @@ jobs: go test -v -race -cover -tags=debug -coverprofile=./coverage-ci/static_root.txt -covermode=atomic ./plugins/static go test -v -race -cover -tags=debug -coverprofile=./coverage-ci/headers.txt -covermode=atomic ./plugins/headers/tests go test -v -race -cover -tags=debug -coverprofile=./coverage-ci/checker.txt -covermode=atomic ./plugins/checker/tests + go test -v -race -cover -tags=debug -coverprofile=./coverage-ci/reload.txt -covermode=atomic ./plugins/reload/tests cat ./coverage-ci/*.txt > ./coverage-ci/summary.txt - uses: codecov/codecov-action@v1 # Docs: <https://github.com/codecov/codecov-action> @@ -39,6 +39,7 @@ test: ## Run application tests go test -v -race -cover -tags=debug -covermode=atomic ./plugins/static/tests go test -v -race -cover -tags=debug -covermode=atomic ./plugins/headers/tests go test -v -race -cover -tags=debug -covermode=atomic ./plugins/checker/tests + go test -v -race -cover -tags=debug -covermode=atomic ./plugins/reload/tests lint: ## Run application linters go fmt ./... @@ -1,10 +1,18 @@ status = [ -'Build (Go 1.14, PHP 7.4)', -'Build (Go 1.15, PHP 7.4)', -'Build (Go 1.14, PHP 8.0)', -'Build (Go 1.15, PHP 8.0)', -'Golang-CI (lint)', -'Build docker image', + 'Build (Go 1.14, PHP 7.4, OS ubuntu-latest)', + 'Build (Go 1.14, PHP 7.4, OS windows-latest)', + 'Build (Go 1.14, PHP 7.4, OS macos-latest)', + 'Build (Go 1.15, PHP 7.4, OS ubuntu-latest)', + 'Build (Go 1.15, PHP 7.4, OS windows-latest)', + 'Build (Go 1.15, PHP 7.4, OS macos-latest)', + 'Build (Go 1.14, PHP 8.0, OS ubuntu-latest)', + 'Build (Go 1.14, PHP 8.0, OS windows-latest)', + 'Build (Go 1.14, PHP 8.0, OS macos-latest)', + 'Build (Go 1.15, PHP 8.0, OS ubuntu-latest)', + 'Build (Go 1.15, PHP 8.0, OS windows-latest)', + 'Build (Go 1.15, PHP 8.0, OS macos-latest)', + 'Golang-CI (lint)', + 'Build docker image', ] required_approvals = 1 diff --git a/plugins/reload/plugin.go b/plugins/reload/plugin.go index bad85b59..386c5e39 100644 --- a/plugins/reload/plugin.go +++ b/plugins/reload/plugin.go @@ -56,7 +56,7 @@ func (s *Plugin) Init(cfg config.Configurer, log log.Logger, res resetter.Resett return nil } } - return errors.E(op, errors.Skip, err) + return errors.E(op, errors.Skip) }, Files: make(map[string]os.FileInfo), Ignored: ignored, @@ -64,7 +64,7 @@ func (s *Plugin) Init(cfg config.Configurer, log log.Logger, res resetter.Resett }) } - s.watcher, err = NewWatcher(configs) + s.watcher, err = NewWatcher(configs, s.log) if err != nil { return errors.E(op, err) } @@ -91,15 +91,6 @@ func (s *Plugin) Serve() chan error { // use the same interval ticker := time.NewTicker(s.cfg.Interval) - // drain channel in case of leaved messages - defer func() { - go func() { - for range treshholdc { - - } - }() - }() - go func() { for e := range s.watcher.Event { treshholdc <- struct { @@ -118,14 +109,12 @@ func (s *Plugin) Serve() chan error { case cfg := <-treshholdc: // replace previous value in map by more recent without adding new one updated[cfg.service] = cfg.serviceConfig - // stop ticker - ticker.Stop() // restart // logic is following: - // if we getting a lot of events, we should't restart particular service on each of it (user doing bug move or very fast typing) + // if we getting a lot of events, we shouldn't restart particular service on each of it (user doing batch move or very fast typing) // instead, we are resetting the ticker and wait for Interval time // If there is no more events, we restart service only once - ticker = time.NewTicker(s.cfg.Interval) + ticker.Reset(s.cfg.Interval) case <-ticker.C: if len(updated) > 0 { for name := range updated { @@ -145,18 +134,21 @@ func (s *Plugin) Serve() chan error { } }() - err := s.watcher.StartPolling(s.cfg.Interval) - if err != nil { - errCh <- errors.E(op, err) - return errCh - } + go func() { + err := s.watcher.StartPolling(s.cfg.Interval) + if err != nil { + errCh <- errors.E(op, err) + return + } + }() return errCh } -func (s *Plugin) Stop() { +func (s *Plugin) Stop() error { s.watcher.Stop() s.stopc <- struct{}{} + return nil } func (s *Plugin) Name() string { diff --git a/plugins/reload/tests/reload_plugin_test.go b/plugins/reload/tests/reload_plugin_test.go index 376df9c8..3d50030f 100644 --- a/plugins/reload/tests/reload_plugin_test.go +++ b/plugins/reload/tests/reload_plugin_test.go @@ -51,7 +51,7 @@ func TestReloadInit(t *testing.T) { wg := &sync.WaitGroup{} wg.Add(1) - tt := time.NewTimer(time.Second * 5) + tt := time.NewTimer(time.Second * 10) go func() { defer wg.Done() diff --git a/plugins/reload/tests/watcher_test.go b/plugins/reload/tests/watcher_test.go index 6a4521a4..f1e0d737 100644 --- a/plugins/reload/tests/watcher_test.go +++ b/plugins/reload/tests/watcher_test.go @@ -12,7 +12,9 @@ import ( "time" "github.com/spiral/errors" + logPlugin "github.com/spiral/roadrunner/v2/plugins/logger" "github.com/spiral/roadrunner/v2/plugins/reload" + "go.uber.org/zap" ) var testServiceName = "test" @@ -46,7 +48,10 @@ func Test_Correct_Watcher_Init(t *testing.T) { FilePatterns: nil, } - w, err := reload.NewWatcher([]reload.WatcherConfig{wc}) + logger, _ := zap.NewDevelopment() + lg := logPlugin.NewZapAdapter(logger) + + w, err := reload.NewWatcher([]reload.WatcherConfig{wc}, lg) if err != nil { t.Fatal(err) } @@ -103,7 +108,10 @@ func Test_Get_FileEvent(t *testing.T) { FilePatterns: []string{"aaa", "txt"}, } - w, err := reload.NewWatcher([]reload.WatcherConfig{wc}) + logger, _ := zap.NewDevelopment() + lg := logPlugin.NewZapAdapter(logger) + + w, err := reload.NewWatcher([]reload.WatcherConfig{wc}, lg) if err != nil { t.Fatal(err) } @@ -195,7 +203,10 @@ func Test_FileExtensionFilter(t *testing.T) { FilePatterns: []string{"aaa", "bbb"}, } - w, err := reload.NewWatcher([]reload.WatcherConfig{wc}) + logger, _ := zap.NewDevelopment() + lg := logPlugin.NewZapAdapter(logger) + + w, err := reload.NewWatcher([]reload.WatcherConfig{wc}, lg) if err != nil { t.Fatal(err) } @@ -294,7 +305,10 @@ func Test_Recursive_Support(t *testing.T) { FilePatterns: []string{"aaa", "bbb"}, } - w, err := reload.NewWatcher([]reload.WatcherConfig{wc}) + logger, _ := zap.NewDevelopment() + lg := logPlugin.NewZapAdapter(logger) + + w, err := reload.NewWatcher([]reload.WatcherConfig{wc}, lg) if err != nil { t.Fatal(err) } @@ -351,7 +365,10 @@ func Test_Wrong_Dir(t *testing.T) { FilePatterns: []string{"aaa", "bbb"}, } - _, err := reload.NewWatcher([]reload.WatcherConfig{wc}) + logger, _ := zap.NewDevelopment() + lg := logPlugin.NewZapAdapter(logger) + + _, err := reload.NewWatcher([]reload.WatcherConfig{wc}, lg) if err == nil { t.Fatal(err) } @@ -420,7 +437,10 @@ func Test_Filter_Directory(t *testing.T) { FilePatterns: []string{"aaa", "bbb", "txt"}, } - w, err := reload.NewWatcher([]reload.WatcherConfig{wc}) + logger, _ := zap.NewDevelopment() + lg := logPlugin.NewZapAdapter(logger) + + w, err := reload.NewWatcher([]reload.WatcherConfig{wc}, lg) if err != nil { t.Fatal(err) } @@ -522,7 +542,10 @@ func Test_Copy_Directory(t *testing.T) { FilePatterns: []string{"aaa", "bbb", "txt"}, } - w, err := reload.NewWatcher([]reload.WatcherConfig{wc}) + logger, _ := zap.NewDevelopment() + lg := logPlugin.NewZapAdapter(logger) + + w, err := reload.NewWatcher([]reload.WatcherConfig{wc}, lg) if err != nil { t.Fatal(err) } diff --git a/plugins/reload/watcher.go b/plugins/reload/watcher.go index ba0b8e82..1829839b 100644 --- a/plugins/reload/watcher.go +++ b/plugins/reload/watcher.go @@ -8,6 +8,7 @@ import ( "time" "github.com/spiral/errors" + "github.com/spiral/roadrunner/v2/interfaces/log" ) // SimpleHook is used to filter by simple criteria, CONTAINS @@ -60,17 +61,22 @@ type Watcher struct { // config for each service // need pointer here to assign files watcherConfigs map[string]WatcherConfig + + // logger + log log.Logger } // Options is used to set Watcher Options type Options func(*Watcher) // NewWatcher returns new instance of File Watcher -func NewWatcher(configs []WatcherConfig, options ...Options) (*Watcher, error) { +func NewWatcher(configs []WatcherConfig, log log.Logger, options ...Options) (*Watcher, error) { w := &Watcher{ Event: make(chan Event), mu: &sync.Mutex{}, + log: log, + close: make(chan struct{}), //workingDir: workDir, @@ -320,6 +326,7 @@ func (w *Watcher) pollEvents(serviceName string, files map[string]os.FileInfo) { for pth, info := range w.watcherConfigs[serviceName].Files { if _, found := files[pth]; !found { removes[pth] = info + w.log.Debug("file was removed", "path", pth, "name", info.Name(), "size", info.Size()) } } @@ -332,10 +339,12 @@ func (w *Watcher) pollEvents(serviceName string, files map[string]os.FileInfo) { if !found { // A file was created. creates[pth] = info + w.log.Debug("file was created", "path", pth, "name", info.Name(), "size", info.Size()) continue } if oldInfo.ModTime() != info.ModTime() { w.watcherConfigs[serviceName].Files[pth] = info + w.log.Debug("file was updated", "path", pth, "name", info.Name(), "size", info.Size()) w.Event <- Event{ Path: pth, Info: info, @@ -344,6 +353,7 @@ func (w *Watcher) pollEvents(serviceName string, files map[string]os.FileInfo) { } if oldInfo.Mode() != info.Mode() { w.watcherConfigs[serviceName].Files[pth] = info + w.log.Debug("file was updated", "path", pth, "name", info.Name(), "size", info.Size()) w.Event <- Event{ Path: pth, Info: info, @@ -367,6 +377,7 @@ func (w *Watcher) pollEvents(serviceName string, files map[string]os.FileInfo) { // update with new w.watcherConfigs[serviceName].Files[path2] = info2 + w.log.Debug("file was renamed/moved", "old path", path1, "new path", path2, "name", info2.Name(), "size", info2.Size()) w.Event <- e } } @@ -375,6 +386,8 @@ func (w *Watcher) pollEvents(serviceName string, files map[string]os.FileInfo) { // Send all the remaining create and remove events. for pth, info := range creates { w.watcherConfigs[serviceName].Files[pth] = info + w.log.Debug("file was created", "path", pth, "name", info.Name(), "size", info.Size()) + w.Event <- Event{ Path: pth, Info: info, @@ -383,6 +396,8 @@ func (w *Watcher) pollEvents(serviceName string, files map[string]os.FileInfo) { } for pth, info := range removes { delete(w.watcherConfigs[serviceName].Files, pth) + w.log.Debug("file was removed", "path", pth, "name", info.Name(), "size", info.Size()) + w.Event <- Event{ Path: pth, Info: info, |