From a0f64785a5495d25f177d4b7caba1abfd8776625 Mon Sep 17 00:00:00 2001 From: Sotirios Mantziaris Date: Sun, 20 Oct 2024 22:14:10 +0300 Subject: [PATCH 1/9] Logging refactoring --- observability/integration_test.go | 2 +- observability/log/log.go | 34 ++++++++++++++++++ observability/observability.go | 16 +++++++-- options.go | 4 +-- options_test.go | 25 +++++++++----- service.go | 57 ++++++++++--------------------- service_test.go | 19 ----------- 7 files changed, 84 insertions(+), 73 deletions(-) diff --git a/observability/integration_test.go b/observability/integration_test.go index ad39de8c91..1d7bdb3215 100644 --- a/observability/integration_test.go +++ b/observability/integration_test.go @@ -13,7 +13,7 @@ func TestSetup(t *testing.T) { t.Setenv("OTEL_EXPORTER_OTLP_INSECURE", "true") ctx := context.Background() - got, err := Setup(ctx, "test", "1.2.3") + got, err := Setup(ctx, Config{}) require.NoError(t, err) require.NoError(t, got.Shutdown(ctx)) diff --git a/observability/log/log.go b/observability/log/log.go index a388dd60d9..96a5347d2f 100644 --- a/observability/log/log.go +++ b/observability/log/log.go @@ -4,10 +4,44 @@ package log import ( "context" "log/slog" + "os" ) +// Config represents the configuration for setting up the logger. +type Config struct { + Attributes []slog.Attr + IsJSON bool + Level string +} + type ctxKey struct{} +func Setup(cfg *Config) { + ho := &slog.HandlerOptions{ + AddSource: true, + Level: level(cfg.Level), + } + + var hnd slog.Handler + + if cfg.IsJSON { + hnd = slog.NewJSONHandler(os.Stderr, ho) + } else { + hnd = slog.NewTextHandler(os.Stderr, ho) + } + + slog.New(hnd.WithAttrs(cfg.Attributes)) +} + +func level(lvl string) slog.Level { + lv := slog.LevelVar{} + if err := lv.UnmarshalText([]byte(lvl)); err != nil { + return slog.LevelInfo + } + + return lv.Level() +} + // FromContext returns the logger, if it exists in the context, or nil. func FromContext(ctx context.Context) *slog.Logger { if l, ok := ctx.Value(ctxKey{}).(*slog.Logger); ok { diff --git a/observability/observability.go b/observability/observability.go index dc86448506..a755991cb5 100644 --- a/observability/observability.go +++ b/observability/observability.go @@ -7,6 +7,7 @@ import ( "log/slog" "github.com/beatlabs/patron/observability/log" + patronlog "github.com/beatlabs/patron/observability/log" patronmetric "github.com/beatlabs/patron/observability/metric" patrontrace "github.com/beatlabs/patron/observability/trace" "go.opentelemetry.io/otel" @@ -49,11 +50,20 @@ func StatusAttribute(err error) attribute.KeyValue { return SucceededAttribute } +// Config represents the configuration for setting up traces, metrics and logs. +type Config struct { + Name string + Version string + LogConfig log.Config +} + // Setup initializes OpenTelemetry's traces and metrics. // It creates a resource with the given name and version, sets up the metric and trace providers, // and returns a Provider containing the initialized providers. -func Setup(ctx context.Context, name, version string) (*Provider, error) { - res, err := createResource(name, version) +func Setup(ctx context.Context, cfg Config) (*Provider, error) { + patronlog.Setup(&cfg.LogConfig) + + res, err := createResource(cfg.Name, cfg.Version) if err != nil { return nil, err } @@ -64,7 +74,7 @@ func Setup(ctx context.Context, name, version string) (*Provider, error) { if err != nil { return nil, err } - traceProvider, err := patrontrace.SetupGRPC(ctx, name, res) + traceProvider, err := patrontrace.SetupGRPC(ctx, cfg.Name, res) if err != nil { return nil, err } diff --git a/options.go b/options.go index 36f591988d..ebf43f3e5e 100644 --- a/options.go +++ b/options.go @@ -34,7 +34,7 @@ func WithLogFields(attrs ...slog.Attr) OptionFunc { continue } - svc.logConfig.attrs = append(svc.logConfig.attrs, attr) + svc.observabilityCfg.LogConfig.Attributes = append(svc.observabilityCfg.LogConfig.Attributes, attr) } return nil @@ -44,7 +44,7 @@ func WithLogFields(attrs ...slog.Attr) OptionFunc { // WithJSONLogger to use Go's slog package. func WithJSONLogger() OptionFunc { return func(svc *Service) error { - svc.logConfig.json = true + svc.observabilityCfg.LogConfig.IsJSON = true return nil } } diff --git a/options_test.go b/options_test.go index b1d250a242..6741cd79b9 100644 --- a/options_test.go +++ b/options_test.go @@ -5,39 +5,46 @@ import ( "log/slog" "testing" + "github.com/beatlabs/patron/observability" + "github.com/beatlabs/patron/observability/log" "github.com/stretchr/testify/assert" "github.com/stretchr/testify/require" ) func TestLogFields(t *testing.T) { - defaultAttrs := defaultLogAttrs("test", "1.0") attrs := []slog.Attr{slog.String("key", "value")} - attrs1 := defaultLogAttrs("name1", "version1") + attrs1 := []slog.Attr{slog.String("name1", "version1")} + + expectedSuccess := observability.Config{LogConfig: log.Config{ + Attributes: attrs, + }} + expectedNoOverwrite := observability.Config{LogConfig: log.Config{ + Attributes: []slog.Attr{slog.String("name1", "version2")}, + }} + type args struct { fields []slog.Attr } tests := map[string]struct { args args - want logConfig + want observability.Config expectedErr string }{ "empty attributes": {args: args{fields: nil}, expectedErr: "attributes are empty"}, - "success": {args: args{fields: attrs}, want: logConfig{attrs: append(defaultAttrs, attrs...)}}, - "no overwrite": {args: args{fields: attrs1}, want: logConfig{attrs: defaultAttrs}}, + "success": {args: args{fields: attrs}, want: expectedSuccess}, + "no overwrite": {args: args{fields: attrs1}, want: expectedNoOverwrite}, } for name, tt := range tests { t.Run(name, func(t *testing.T) { svc := &Service{ - logConfig: logConfig{ - attrs: defaultAttrs, - }, + observabilityCfg: observability.Config{}, } err := WithLogFields(tt.args.fields...)(svc) if tt.expectedErr == "" { require.NoError(t, err) - assert.Equal(t, tt.want, svc.logConfig) + assert.Equal(t, tt.want, svc.observabilityCfg) } else { require.EqualError(t, err, tt.expectedErr) } diff --git a/service.go b/service.go index 48c64755ae..b7bc5860e1 100644 --- a/service.go +++ b/service.go @@ -32,7 +32,7 @@ type Service struct { version string termSig chan os.Signal sighupHandler func() - logConfig logConfig + observabilityCfg observability.Config observabilityProvider *observability.Provider } @@ -46,7 +46,10 @@ func New(name, version string, options ...OptionFunc) (*Service, error) { var err error ctx := context.Background() - observabilityProvider, err := observability.Setup(ctx, name, version) + + cfg := observabilityConfig(name, version) + + observabilityProvider, err := observability.Setup(ctx, cfg) if err != nil { return nil, err } @@ -58,10 +61,7 @@ func New(name, version string, options ...OptionFunc) (*Service, error) { sighupHandler: func() { slog.Debug("sighup received: nothing setup") }, - logConfig: logConfig{ - attrs: defaultLogAttrs(name, version), - json: false, - }, + observabilityCfg: cfg, observabilityProvider: observabilityProvider, } @@ -78,7 +78,6 @@ func New(name, version string, options ...OptionFunc) (*Service, error) { return nil, errors.Join(optionErrors...) } - setupLogging(s.logConfig) s.setupOSSignal() return s, nil @@ -149,51 +148,31 @@ func (s *Service) waitTermination(chErr <-chan error) error { } } -type logConfig struct { - attrs []slog.Attr - json bool -} - -func getLogLevel() slog.Level { +func observabilityConfig(name, version string) observability.Config { + var lvl string lvl, ok := os.LookupEnv("PATRON_LOG_LEVEL") if !ok { - return slog.LevelInfo - } - - lv := slog.LevelVar{} - if err := lv.UnmarshalText([]byte(lvl)); err != nil { - return slog.LevelInfo + lvl = "info" } - return lv.Level() -} - -func defaultLogAttrs(name, version string) []slog.Attr { hostname, err := os.Hostname() if err != nil { hostname = host } - return []slog.Attr{ + attrs := []slog.Attr{ slog.String(srv, name), slog.String(ver, version), slog.String(host, hostname), } -} - -func setupLogging(lc logConfig) { - ho := &slog.HandlerOptions{ - AddSource: true, - Level: getLogLevel(), - } - var hnd slog.Handler - - if lc.json { - hnd = slog.NewJSONHandler(os.Stderr, ho) - } else { - hnd = slog.NewTextHandler(os.Stderr, ho) + return observability.Config{ + Name: name, + Version: version, + LogConfig: log.Config{ + Attributes: attrs, + IsJSON: false, + Level: lvl, + }, } - - slog.New(hnd.WithAttrs(lc.attrs)) } diff --git a/service_test.go b/service_test.go index 5ade10161d..d99510b4a4 100644 --- a/service_test.go +++ b/service_test.go @@ -58,22 +58,3 @@ func TestNew(t *testing.T) { }) } } - -func Test_getLogLevel(t *testing.T) { - tests := map[string]struct { - lvl string - want slog.Level - }{ - "debug": {lvl: "debug", want: slog.LevelDebug}, - "info": {lvl: "info", want: slog.LevelInfo}, - "warn": {lvl: "warn", want: slog.LevelWarn}, - "error": {lvl: "error", want: slog.LevelError}, - "invalid level": {lvl: "invalid", want: slog.LevelInfo}, - } - for name, tt := range tests { - t.Run(name, func(t *testing.T) { - t.Setenv("PATRON_LOG_LEVEL", tt.lvl) - assert.Equal(t, tt.want, getLogLevel()) - }) - } -} From 2c82f611c8489e9a51dd1fd552054a6900aeeb91 Mon Sep 17 00:00:00 2001 From: Sotirios Mantziaris Date: Sun, 20 Oct 2024 22:21:20 +0300 Subject: [PATCH 2/9] Logging refactoring --- observability/observability.go | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/observability/observability.go b/observability/observability.go index a755991cb5..4e316febfd 100644 --- a/observability/observability.go +++ b/observability/observability.go @@ -7,7 +7,6 @@ import ( "log/slog" "github.com/beatlabs/patron/observability/log" - patronlog "github.com/beatlabs/patron/observability/log" patronmetric "github.com/beatlabs/patron/observability/metric" patrontrace "github.com/beatlabs/patron/observability/trace" "go.opentelemetry.io/otel" @@ -61,7 +60,7 @@ type Config struct { // It creates a resource with the given name and version, sets up the metric and trace providers, // and returns a Provider containing the initialized providers. func Setup(ctx context.Context, cfg Config) (*Provider, error) { - patronlog.Setup(&cfg.LogConfig) + log.Setup(&cfg.LogConfig) res, err := createResource(cfg.Name, cfg.Version) if err != nil { From 1578b27737876159d5e490ae3fd6adf5eedfe72a Mon Sep 17 00:00:00 2001 From: Sotirios Mantziaris Date: Sun, 20 Oct 2024 22:52:18 +0300 Subject: [PATCH 3/9] Refactor log_test.go and add unit tests for Setup function --- observability/log/log_test.go | 22 ++++++++++++++++++++++ 1 file changed, 22 insertions(+) diff --git a/observability/log/log_test.go b/observability/log/log_test.go index 76d4a0d153..c2544bddce 100644 --- a/observability/log/log_test.go +++ b/observability/log/log_test.go @@ -9,6 +9,28 @@ import ( "github.com/stretchr/testify/assert" ) +func TestSetup(t *testing.T) { + t.Run("JSON", func(t *testing.T) { + cfg := &Config{ + Attributes: []slog.Attr{}, + IsJSON: true, + Level: "debug", + } + Setup(cfg) + assert.NotNil(t, slog.Default()) + }) + + t.Run("Text", func(t *testing.T) { + cfg := &Config{ + Attributes: []slog.Attr{}, + IsJSON: false, + Level: "debug", + } + Setup(cfg) + assert.NotNil(t, slog.Default()) + }) +} + func TestContext(t *testing.T) { l := slog.Default() From d86e1d56d7a437f1d4d71e85eb56499a7a6a01b7 Mon Sep 17 00:00:00 2001 From: Sotirios Mantziaris Date: Thu, 24 Oct 2024 22:19:54 +0300 Subject: [PATCH 4/9] Logging refactoring --- observability/log/log.go | 16 +++++++++++++++- observability/log/log_test.go | 29 +++++++++++++---------------- 2 files changed, 28 insertions(+), 17 deletions(-) diff --git a/observability/log/log.go b/observability/log/log.go index 96a5347d2f..84daca2336 100644 --- a/observability/log/log.go +++ b/observability/log/log.go @@ -16,7 +16,21 @@ type Config struct { type ctxKey struct{} +var logCfg *Config + +// Setup sets up the logger with the given configuration. func Setup(cfg *Config) { + logCfg = cfg + setDefaultLogger(cfg) +} + +// SetLevel sets the logger level. +func SetLevel(lvl string) { + logCfg.Level = lvl + setDefaultLogger(logCfg) +} + +func setDefaultLogger(cfg *Config) { ho := &slog.HandlerOptions{ AddSource: true, Level: level(cfg.Level), @@ -30,7 +44,7 @@ func Setup(cfg *Config) { hnd = slog.NewTextHandler(os.Stderr, ho) } - slog.New(hnd.WithAttrs(cfg.Attributes)) + slog.SetDefault(slog.New(hnd.WithAttrs(cfg.Attributes))) } func level(lvl string) slog.Level { diff --git a/observability/log/log_test.go b/observability/log/log_test.go index c2544bddce..0fc1089b02 100644 --- a/observability/log/log_test.go +++ b/observability/log/log_test.go @@ -45,22 +45,19 @@ func TestContext(t *testing.T) { }) } -func TestEnabled(t *testing.T) { - type args struct { - l slog.Level - } - tests := map[string]struct { - args args - want bool - }{ - "Disabled": {args{slog.LevelDebug}, false}, - "Enabled": {args{slog.LevelInfo}, true}, - } - for name, tt := range tests { - t.Run(name, func(t *testing.T) { - assert.Equal(t, tt.want, Enabled(tt.args.l)) - }) - } +func TestSetLevelAndCheckEnable(t *testing.T) { + Setup(&Config{ + Attributes: []slog.Attr{}, + IsJSON: true, + Level: "info", + }) + + assert.True(t, Enabled(slog.LevelInfo)) + assert.False(t, Enabled(slog.LevelDebug)) + + SetLevel("debug") + + assert.True(t, Enabled(slog.LevelDebug)) } func TestErrorAttr(t *testing.T) { From 8d126f92a8a5d7de5c234e914de2a12a8deaed5a Mon Sep 17 00:00:00 2001 From: Sotirios Mantziaris Date: Fri, 25 Oct 2024 23:37:42 +0300 Subject: [PATCH 5/9] Logging refactoring --- component/http/observability.go | 23 ++++++++++++++ component/http/observability_test.go | 46 ++++++++++++++++++++++++++-- observability/log/log.go | 24 +++++++++------ observability/log/log_test.go | 6 ++-- observability/observability.go | 5 ++- 5 files changed, 88 insertions(+), 16 deletions(-) diff --git a/component/http/observability.go b/component/http/observability.go index 1d30c05cc0..16da648ee3 100644 --- a/component/http/observability.go +++ b/component/http/observability.go @@ -5,6 +5,8 @@ import ( "fmt" "net/http" "net/http/pprof" + + "github.com/beatlabs/patron/observability/log" ) func ProfilingRoutes(enableExpVar bool) []*Route { @@ -47,3 +49,24 @@ func expVars(w http.ResponseWriter, _ *http.Request) { }) _, _ = fmt.Fprintf(w, "\n}\n") } + +// LoggingRoutes returns a routes relates to logs. +func LoggingRoutes() []*Route { + handler := func(w http.ResponseWriter, r *http.Request) { + lvl := r.PathValue("level") + if lvl == "" { + http.Error(w, "missing log level", http.StatusBadRequest) + return + } + + err := log.SetLevel(lvl) + if err != nil { + http.Error(w, err.Error(), http.StatusBadRequest) + return + } + w.WriteHeader(http.StatusOK) + } + + route, _ := NewRoute("POST /debug/log/{level}", handler) + return []*Route{route} +} diff --git a/component/http/observability_test.go b/component/http/observability_test.go index 0fd297403e..5825ca1105 100644 --- a/component/http/observability_test.go +++ b/component/http/observability_test.go @@ -7,6 +7,7 @@ import ( "net/http/httptest" "testing" + "github.com/beatlabs/patron/observability/log" "github.com/stretchr/testify/assert" "github.com/stretchr/testify/require" ) @@ -18,7 +19,7 @@ type profilingTestCase struct { func TestProfilingRoutes(t *testing.T) { t.Run("without vars", func(t *testing.T) { - server := createProfilingServer(false) + server := createServer(false) defer server.Close() for name, tt := range createProfilingTestCases(false) { @@ -34,7 +35,7 @@ func TestProfilingRoutes(t *testing.T) { }) t.Run("with vars", func(t *testing.T) { - server := createProfilingServer(true) + server := createServer(true) defer server.Close() for name, tt := range createProfilingTestCases(true) { @@ -50,11 +51,14 @@ func TestProfilingRoutes(t *testing.T) { }) } -func createProfilingServer(enableExpVar bool) *httptest.Server { +func createServer(enableExpVar bool) *httptest.Server { mux := http.NewServeMux() for _, route := range ProfilingRoutes(enableExpVar) { mux.HandleFunc(route.path, route.handler) } + for _, route := range LoggingRoutes() { + mux.HandleFunc(route.path, route.handler) + } return httptest.NewServer(mux) } @@ -80,3 +84,39 @@ func createProfilingTestCases(enableExpVar bool) map[string]profilingTestCase { "vars": {"/debug/vars/", expVarWant}, } } + +func TestLoggingRoutes(t *testing.T) { + log.Setup(&log.Config{ + IsJSON: true, + Level: "info", + }) + server := createServer(true) + defer server.Close() + + t.Run("change log level to debug", func(t *testing.T) { + req, err := http.NewRequestWithContext(context.Background(), http.MethodPost, fmt.Sprintf("%s/debug/log/debug", server.URL), nil) + require.NoError(t, err) + resp, err := http.DefaultClient.Do(req) + require.NoError(t, err) + assert.Equal(t, http.StatusOK, resp.StatusCode) + require.NoError(t, resp.Body.Close()) + }) + + t.Run("wrong log level", func(t *testing.T) { + req, err := http.NewRequestWithContext(context.Background(), http.MethodPost, fmt.Sprintf("%s/debug/log/xxx", server.URL), nil) + require.NoError(t, err) + resp, err := http.DefaultClient.Do(req) + require.NoError(t, err) + assert.Equal(t, http.StatusBadRequest, resp.StatusCode) + require.NoError(t, resp.Body.Close()) + }) + + t.Run("empty log level", func(t *testing.T) { + req, err := http.NewRequestWithContext(context.Background(), http.MethodPost, fmt.Sprintf("%s/debug/log/", server.URL), nil) + require.NoError(t, err) + resp, err := http.DefaultClient.Do(req) + require.NoError(t, err) + assert.Equal(t, http.StatusNotFound, resp.StatusCode) + require.NoError(t, resp.Body.Close()) + }) +} diff --git a/observability/log/log.go b/observability/log/log.go index 84daca2336..76f0e51495 100644 --- a/observability/log/log.go +++ b/observability/log/log.go @@ -19,21 +19,26 @@ type ctxKey struct{} var logCfg *Config // Setup sets up the logger with the given configuration. -func Setup(cfg *Config) { +func Setup(cfg *Config) error { logCfg = cfg - setDefaultLogger(cfg) + return setDefaultLogger(cfg) } // SetLevel sets the logger level. -func SetLevel(lvl string) { +func SetLevel(lvl string) error { logCfg.Level = lvl - setDefaultLogger(logCfg) + return setDefaultLogger(logCfg) } -func setDefaultLogger(cfg *Config) { +func setDefaultLogger(cfg *Config) error { + lvl, err := level(cfg.Level) + if err != nil { + return err + } + ho := &slog.HandlerOptions{ AddSource: true, - Level: level(cfg.Level), + Level: lvl, } var hnd slog.Handler @@ -45,15 +50,16 @@ func setDefaultLogger(cfg *Config) { } slog.SetDefault(slog.New(hnd.WithAttrs(cfg.Attributes))) + return nil } -func level(lvl string) slog.Level { +func level(lvl string) (slog.Level, error) { lv := slog.LevelVar{} if err := lv.UnmarshalText([]byte(lvl)); err != nil { - return slog.LevelInfo + return slog.LevelInfo, err } - return lv.Level() + return lv.Level(), nil } // FromContext returns the logger, if it exists in the context, or nil. diff --git a/observability/log/log_test.go b/observability/log/log_test.go index 0fc1089b02..a42e32b627 100644 --- a/observability/log/log_test.go +++ b/observability/log/log_test.go @@ -16,7 +16,7 @@ func TestSetup(t *testing.T) { IsJSON: true, Level: "debug", } - Setup(cfg) + assert.NoError(t, Setup(cfg)) assert.NotNil(t, slog.Default()) }) @@ -26,7 +26,7 @@ func TestSetup(t *testing.T) { IsJSON: false, Level: "debug", } - Setup(cfg) + assert.NoError(t, Setup(cfg)) assert.NotNil(t, slog.Default()) }) } @@ -55,7 +55,7 @@ func TestSetLevelAndCheckEnable(t *testing.T) { assert.True(t, Enabled(slog.LevelInfo)) assert.False(t, Enabled(slog.LevelDebug)) - SetLevel("debug") + assert.NoError(t, SetLevel("debug")) assert.True(t, Enabled(slog.LevelDebug)) } diff --git a/observability/observability.go b/observability/observability.go index 4e316febfd..436329988a 100644 --- a/observability/observability.go +++ b/observability/observability.go @@ -60,7 +60,10 @@ type Config struct { // It creates a resource with the given name and version, sets up the metric and trace providers, // and returns a Provider containing the initialized providers. func Setup(ctx context.Context, cfg Config) (*Provider, error) { - log.Setup(&cfg.LogConfig) + err := log.Setup(&cfg.LogConfig) + if err != nil { + return nil, err + } res, err := createResource(cfg.Name, cfg.Version) if err != nil { From c24c0710e6913c0158e98253a9cc5455548e5fa3 Mon Sep 17 00:00:00 2001 From: Sotirios Mantziaris Date: Fri, 25 Oct 2024 23:46:01 +0300 Subject: [PATCH 6/9] Logging refactoring --- component/http/observability_test.go | 10 +++++----- observability/log/log_test.go | 11 ++++++----- 2 files changed, 11 insertions(+), 10 deletions(-) diff --git a/component/http/observability_test.go b/component/http/observability_test.go index 5825ca1105..c4daa09f0a 100644 --- a/component/http/observability_test.go +++ b/component/http/observability_test.go @@ -86,15 +86,15 @@ func createProfilingTestCases(enableExpVar bool) map[string]profilingTestCase { } func TestLoggingRoutes(t *testing.T) { - log.Setup(&log.Config{ + require.NoError(t, log.Setup(&log.Config{ IsJSON: true, Level: "info", - }) + })) server := createServer(true) defer server.Close() t.Run("change log level to debug", func(t *testing.T) { - req, err := http.NewRequestWithContext(context.Background(), http.MethodPost, fmt.Sprintf("%s/debug/log/debug", server.URL), nil) + req, err := http.NewRequestWithContext(context.Background(), http.MethodPost, server.URL+"/debug/log/debug", nil) require.NoError(t, err) resp, err := http.DefaultClient.Do(req) require.NoError(t, err) @@ -103,7 +103,7 @@ func TestLoggingRoutes(t *testing.T) { }) t.Run("wrong log level", func(t *testing.T) { - req, err := http.NewRequestWithContext(context.Background(), http.MethodPost, fmt.Sprintf("%s/debug/log/xxx", server.URL), nil) + req, err := http.NewRequestWithContext(context.Background(), http.MethodPost, server.URL+"/debug/log/xxx", nil) require.NoError(t, err) resp, err := http.DefaultClient.Do(req) require.NoError(t, err) @@ -112,7 +112,7 @@ func TestLoggingRoutes(t *testing.T) { }) t.Run("empty log level", func(t *testing.T) { - req, err := http.NewRequestWithContext(context.Background(), http.MethodPost, fmt.Sprintf("%s/debug/log/", server.URL), nil) + req, err := http.NewRequestWithContext(context.Background(), http.MethodPost, server.URL+"/debug/log/", nil) require.NoError(t, err) resp, err := http.DefaultClient.Do(req) require.NoError(t, err) diff --git a/observability/log/log_test.go b/observability/log/log_test.go index a42e32b627..3b0ed5d53e 100644 --- a/observability/log/log_test.go +++ b/observability/log/log_test.go @@ -7,6 +7,7 @@ import ( "testing" "github.com/stretchr/testify/assert" + "github.com/stretchr/testify/require" ) func TestSetup(t *testing.T) { @@ -16,7 +17,7 @@ func TestSetup(t *testing.T) { IsJSON: true, Level: "debug", } - assert.NoError(t, Setup(cfg)) + require.NoError(t, Setup(cfg)) assert.NotNil(t, slog.Default()) }) @@ -26,7 +27,7 @@ func TestSetup(t *testing.T) { IsJSON: false, Level: "debug", } - assert.NoError(t, Setup(cfg)) + require.NoError(t, Setup(cfg)) assert.NotNil(t, slog.Default()) }) } @@ -46,16 +47,16 @@ func TestContext(t *testing.T) { } func TestSetLevelAndCheckEnable(t *testing.T) { - Setup(&Config{ + require.NoError(t, Setup(&Config{ Attributes: []slog.Attr{}, IsJSON: true, Level: "info", - }) + })) assert.True(t, Enabled(slog.LevelInfo)) assert.False(t, Enabled(slog.LevelDebug)) - assert.NoError(t, SetLevel("debug")) + require.NoError(t, SetLevel("debug")) assert.True(t, Enabled(slog.LevelDebug)) } From bebbebaa4e1e09e527e021c78999ebf5682c87e0 Mon Sep 17 00:00:00 2001 From: Sotirios Mantziaris Date: Sat, 26 Oct 2024 00:17:16 +0300 Subject: [PATCH 7/9] Logging refactoring --- component/amqp/message_test.go | 4 +++- component/kafka/component_test.go | 6 ++++-- component/sqs/component_test.go | 4 +++- component/sqs/metric.go | 4 ++-- examples/service/http.go | 4 ++-- 5 files changed, 14 insertions(+), 8 deletions(-) diff --git a/component/amqp/message_test.go b/component/amqp/message_test.go index e69a63c4be..cf2cedef93 100644 --- a/component/amqp/message_test.go +++ b/component/amqp/message_test.go @@ -27,7 +27,9 @@ var ( ) func TestMain(m *testing.M) { - os.Setenv("OTEL_BSP_SCHEDULE_DELAY", "100") + if err := os.Setenv("OTEL_BSP_SCHEDULE_DELAY", "100"); err != nil { + panic(err) + } tracePublisher = patrontrace.Setup("test", nil, traceExporter) diff --git a/component/kafka/component_test.go b/component/kafka/component_test.go index 00af1330f8..a5cf9cdca6 100644 --- a/component/kafka/component_test.go +++ b/component/kafka/component_test.go @@ -26,7 +26,9 @@ var ( ) func TestMain(m *testing.M) { - os.Setenv("OTEL_BSP_SCHEDULE_DELAY", "100") + if err := os.Setenv("OTEL_BSP_SCHEDULE_DELAY", "100"); err != nil { + panic(err) + } tracePublisher = patrontrace.Setup("test", nil, traceExporter) os.Exit(m.Run()) @@ -39,7 +41,7 @@ func TestNew(t *testing.T) { // consumer will commit every batch in a blocking operation saramaCfg.Consumer.Offsets.AutoCommit.Enable = false saramaCfg.Consumer.Offsets.Initial = sarama.OffsetOldest - saramaCfg.Consumer.Group.Rebalance.Strategy = sarama.BalanceStrategySticky + saramaCfg.Consumer.Group.Rebalance.GroupStrategies = append(saramaCfg.Consumer.Group.Rebalance.GroupStrategies, sarama.NewBalanceStrategySticky()) saramaCfg.Net.DialTimeout = 15 * time.Second saramaCfg.Version = sarama.V2_6_0_0 diff --git a/component/sqs/component_test.go b/component/sqs/component_test.go index 7a7eedb2be..cc4642cd99 100644 --- a/component/sqs/component_test.go +++ b/component/sqs/component_test.go @@ -22,7 +22,9 @@ var ( ) func TestMain(m *testing.M) { - os.Setenv("OTEL_BSP_SCHEDULE_DELAY", "100") + if err := os.Setenv("OTEL_BSP_SCHEDULE_DELAY", "100"); err != nil { + panic(err) + } tracePublisher = patrontrace.Setup("test", nil, traceExporter) diff --git a/component/sqs/metric.go b/component/sqs/metric.go index 138a83e199..b284a8efa9 100644 --- a/component/sqs/metric.go +++ b/component/sqs/metric.go @@ -30,8 +30,8 @@ func init() { messageQueueSizeGauge = patronmetric.Float64Gauge(packageName, "sqs.queue.size", "SQS message queue size.", "1") } -func observerMessageAge(ctx context.Context, queue string, attributes map[string]string) { - attribute, ok := attributes[sqsAttributeSentTimestamp] +func observerMessageAge(ctx context.Context, queue string, attrs map[string]string) { + attribute, ok := attrs[sqsAttributeSentTimestamp] if !ok || len(strings.TrimSpace(attribute)) == 0 { return } diff --git a/examples/service/http.go b/examples/service/http.go index 04db86fc34..97b82d9372 100644 --- a/examples/service/http.go +++ b/examples/service/http.go @@ -32,10 +32,10 @@ func createHttpRouter() (patron.Component, error) { return nil, fmt.Errorf("failed to create routes: %w", err) } - router, err := router.New(router.WithRoutes(rr...)) + rt, err := router.New(router.WithRoutes(rr...)) if err != nil { return nil, fmt.Errorf("failed to create http router: %w", err) } - return patronhttp.New(router) + return patronhttp.New(rt) } From 58223443d0503e1f3d5003798bc81032b3c015c2 Mon Sep 17 00:00:00 2001 From: Sotirios Mantziaris Date: Sat, 26 Oct 2024 08:40:03 +0300 Subject: [PATCH 8/9] Logging refactoring --- service.go | 2 ++ 1 file changed, 2 insertions(+) diff --git a/service.go b/service.go index b7bc5860e1..57b1a76ece 100644 --- a/service.go +++ b/service.go @@ -36,6 +36,7 @@ type Service struct { observabilityProvider *observability.Provider } +// New creates a new Service instance. func New(name, version string, options ...OptionFunc) (*Service, error) { if name == "" { return nil, errors.New("name is required") @@ -83,6 +84,7 @@ func New(name, version string, options ...OptionFunc) (*Service, error) { return s, nil } +// Run starts the service with the provided components. func (s *Service) Run(ctx context.Context, components ...Component) error { if len(components) == 0 || components[0] == nil { return errors.New("components are empty or nil") From 2700c5506eed773db3a9adbfa24348726843b149 Mon Sep 17 00:00:00 2001 From: Sotirios Mantziaris Date: Sat, 26 Oct 2024 08:49:06 +0300 Subject: [PATCH 9/9] Logging refactoring --- observability/integration_test.go | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/observability/integration_test.go b/observability/integration_test.go index 1d7bdb3215..d36278718d 100644 --- a/observability/integration_test.go +++ b/observability/integration_test.go @@ -6,6 +6,7 @@ import ( "context" "testing" + "github.com/beatlabs/patron/observability/log" "github.com/stretchr/testify/require" ) @@ -13,7 +14,11 @@ func TestSetup(t *testing.T) { t.Setenv("OTEL_EXPORTER_OTLP_INSECURE", "true") ctx := context.Background() - got, err := Setup(ctx, Config{}) + got, err := Setup(ctx, Config{ + LogConfig: log.Config{ + Level: "debug", + }, + }) require.NoError(t, err) require.NoError(t, got.Shutdown(ctx))