package logger import ( "os" "os/signal" "sync" "syscall" "testing" "github.com/golang/mock/gomock" endure "github.com/spiral/endure/pkg/container" "github.com/spiral/roadrunner/v2/plugins/config" "github.com/spiral/roadrunner/v2/plugins/http" "github.com/spiral/roadrunner/v2/plugins/logger" "github.com/spiral/roadrunner/v2/plugins/rpc" "github.com/spiral/roadrunner/v2/plugins/server" "github.com/spiral/roadrunner/v2/tests/mocks" "github.com/stretchr/testify/assert" ) func TestLogger(t *testing.T) { container, err := endure.NewContainer(nil, endure.RetryOnFail(true), endure.SetLogLevel(endure.ErrorLevel)) if err != nil { t.Fatal(err) } // config plugin vp := &config.Viper{} vp.Path = "configs/.rr.yaml" vp.Prefix = "rr" err = container.RegisterAll( vp, &Plugin{}, &logger.ZapLogger{}, ) assert.NoError(t, err) err = container.Init() if err != nil { t.Fatal(err) } errCh, err := container.Serve() if err != nil { t.Fatal(err) } // stop by CTRL+C c := make(chan os.Signal, 1) signal.Notify(c, os.Interrupt) stopCh := make(chan struct{}, 1) wg := &sync.WaitGroup{} wg.Add(1) go func() { defer wg.Done() for { select { case e := <-errCh: assert.NoError(t, e.Error) assert.NoError(t, container.Stop()) return case <-c: err = container.Stop() assert.NoError(t, err) return case <-stopCh: assert.NoError(t, container.Stop()) return } } }() stopCh <- struct{}{} wg.Wait() } func TestLoggerRawErr(t *testing.T) { cont, err := endure.NewContainer(nil, endure.SetLogLevel(endure.ErrorLevel)) assert.NoError(t, err) // config plugin cfg := &config.Viper{} cfg.Path = "configs/.rr-raw-mode.yaml" cfg.Prefix = "rr" controller := gomock.NewController(t) mockLogger := mocks.NewMockLogger(controller) mockLogger.EXPECT().Debug("worker destructed", "pid", gomock.Any()).MinTimes(1) mockLogger.EXPECT().Info("{\"field\": \"value\"}").MinTimes(1) mockLogger.EXPECT().Debug("worker constructed", "pid", gomock.Any()).MinTimes(1) err = cont.RegisterAll( cfg, mockLogger, &server.Plugin{}, &http.Plugin{}, ) assert.NoError(t, err) err = cont.Init() if err != nil { t.Fatal(err) } ch, err := cont.Serve() assert.NoError(t, err) sig := make(chan os.Signal, 1) signal.Notify(sig, os.Interrupt, syscall.SIGINT, syscall.SIGTERM) wg := &sync.WaitGroup{} wg.Add(1) stopCh := make(chan struct{}, 1) go func() { defer wg.Done() for { select { case e := <-ch: assert.Fail(t, "error", e.Error.Error()) err = cont.Stop() if err != nil { assert.FailNow(t, "error", err.Error()) } case <-sig: err = cont.Stop() if err != nil { assert.FailNow(t, "error", err.Error()) } return case <-stopCh: // timeout err = cont.Stop() if err != nil { assert.FailNow(t, "error", err.Error()) } return } } }() stopCh <- struct{}{} wg.Wait() } func TestLoggerNoConfig(t *testing.T) { container, err := endure.NewContainer(nil, endure.RetryOnFail(true), endure.SetLogLevel(endure.ErrorLevel)) if err != nil { t.Fatal(err) } // config plugin vp := &config.Viper{} vp.Path = "configs/.rr-no-logger.yaml" vp.Prefix = "rr" err = container.RegisterAll( vp, &Plugin{}, &logger.ZapLogger{}, ) assert.NoError(t, err) err = container.Init() if err != nil { t.Fatal(err) } errCh, err := container.Serve() if err != nil { t.Fatal(err) } // stop by CTRL+C c := make(chan os.Signal, 1) signal.Notify(c, os.Interrupt) stopCh := make(chan struct{}, 1) wg := &sync.WaitGroup{} wg.Add(1) go func() { defer wg.Done() for { select { case e := <-errCh: assert.NoError(t, e.Error) assert.NoError(t, container.Stop()) return case <-c: err = container.Stop() assert.NoError(t, err) return case <-stopCh: assert.NoError(t, container.Stop()) return } } }() stopCh <- struct{}{} wg.Wait() } // Should no panic func TestLoggerNoConfig2(t *testing.T) { container, err := endure.NewContainer(nil, endure.RetryOnFail(true), endure.SetLogLevel(endure.ErrorLevel)) if err != nil { t.Fatal(err) } // config plugin vp := &config.Viper{} vp.Path = "configs/.rr-no-logger2.yaml" vp.Prefix = "rr" err = container.RegisterAll( vp, &rpc.Plugin{}, &logger.ZapLogger{}, &http.Plugin{}, &server.Plugin{}, ) assert.NoError(t, err) err = container.Init() if err != nil { t.Fatal(err) } errCh, err := container.Serve() if err != nil { t.Fatal(err) } // stop by CTRL+C c := make(chan os.Signal, 1) signal.Notify(c, os.Interrupt) stopCh := make(chan struct{}, 1) wg := &sync.WaitGroup{} wg.Add(1) go func() { defer wg.Done() for { select { case e := <-errCh: assert.NoError(t, e.Error) assert.NoError(t, container.Stop()) return case <-c: err = container.Stop() assert.NoError(t, err) return case <-stopCh: assert.NoError(t, container.Stop()) return } } }() stopCh <- struct{}{} wg.Wait() }