Skip to content

Commit

Permalink
fix: race-condition problem on testhelper (#5) (#8)
Browse files Browse the repository at this point in the history
  • Loading branch information
karupanerura authored Feb 28, 2023
1 parent 33dd87f commit f037fda
Show file tree
Hide file tree
Showing 6 changed files with 73 additions and 27 deletions.
23 changes: 23 additions & 0 deletions internal/config/config.go
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
package config

import (
"context"
"io"
"os"

Expand Down Expand Up @@ -28,3 +29,25 @@ func init() {
ApplicationLogOut = os.Stdout
AccessLogOut = os.Stderr
}

type ContextConfig struct {
// ApplicationLogOut is io.Writer object for application log
ApplicationLogOut io.Writer
// AccessLogOut is io.Writer object for access log
AccessLogOut io.Writer
}

type contextKey struct{}

var contextConfigKey = &contextKey{}

// GetContextConfig sets the ContextConfig instance to context
func SetContextConfig(ctx context.Context, cs *ContextConfig) context.Context {
return context.WithValue(ctx, contextConfigKey, cs)
}

// GetContextConfig gets the ContextSeverity instance from context
func GetContextConfig(ctx context.Context) *ContextConfig {
v, _ := ctx.Value(contextConfigKey).(*ContextConfig)
return v
}
14 changes: 12 additions & 2 deletions internal/logger/logger.go
Original file line number Diff line number Diff line change
Expand Up @@ -53,7 +53,12 @@ func WriteApplicationLog(ctx context.Context, s severity.Severity, format string
TraceSampled: sc.TraceSampled,
}

if err := json.NewEncoder(config.ApplicationLogOut).Encode(ety); err != nil {
w := config.ApplicationLogOut
if cc := config.GetContextConfig(ctx); cc != nil && cc.ApplicationLogOut != nil {
w = cc.ApplicationLogOut
}

if err := json.NewEncoder(w).Encode(ety); err != nil {
fmt.Printf("failed to write log: %v", err)
}
}
Expand Down Expand Up @@ -86,7 +91,12 @@ func WriteAccessLog(ctx context.Context, req types.HTTPRequest) {
HTTPRequest: req,
}

if err := json.NewEncoder(config.AccessLogOut).Encode(ety); err != nil {
w := config.AccessLogOut
if cc := config.GetContextConfig(ctx); cc != nil && cc.AccessLogOut != nil {
w = cc.AccessLogOut
}

if err := json.NewEncoder(w).Encode(ety); err != nil {
fmt.Printf("failed to write log: %v", err)
}
}
5 changes: 3 additions & 2 deletions internal/logger/logger_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -48,7 +48,7 @@ func TestLoggerWriteApplicationLog(t *testing.T) {
}()

t.Run("Tests WriteApplicationLog function", func(t *testing.T) {
got := testhelper.ExtractApplicationLogOut(t, func() {
got := testhelper.ExtractApplicationLogOut(t, ctx, func(ctx context.Context) {
// tests the function
logger.WriteApplicationLog(ctx, severity.Info, "writes %s log", "info")
})
Expand Down Expand Up @@ -99,7 +99,7 @@ func TestLoggerWriteAccessLog(t *testing.T) {

t.Run("Tests WriteAccessLog function", func(t *testing.T) {

got := testhelper.ExtractAccessLogOut(t, func() {
got := testhelper.ExtractAccessLogOut(t, ctx, func(ctx context.Context) {
// Tests the function
httpReq := httptest.NewRequest(http.MethodGet, "/test1", nil)
req := types.MakeHTTPRequest(*httpReq, 200, 333, time.Duration(100))
Expand All @@ -112,4 +112,5 @@ func TestLoggerWriteAccessLog(t *testing.T) {
t.Errorf("failed log info test: %v", diff)
}
})

}
2 changes: 1 addition & 1 deletion logz_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -130,7 +130,7 @@ func TestLogzWriteLog(t *testing.T) {

sc := spancontext.Extract(ctx)

got := testhelper.ExtractApplicationLogOut(t, func() {
got := testhelper.ExtractApplicationLogOut(t, ctx, func(ctx context.Context) {
logz.Infof(ctx, "writes %s log", "info")
})

Expand Down
17 changes: 9 additions & 8 deletions middleware/nethttp_test.go
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
package middleware_test

import (
"context"
"fmt"
"net/http"
"net/http/httptest"
Expand Down Expand Up @@ -65,13 +66,13 @@ func TestNetHTTPMaxSeverity(t *testing.T) {
mid := middleware.NetHTTP("test/component")(mux)
rec := httptest.NewRecorder()

got := testhelper.ExtractAccessLogOut(t, func() {
got := testhelper.ExtractAccessLogOut(t, context.Background(), func(ctx context.Context) {
req1 := httptest.NewRequest(http.MethodGet, "/test1", nil)
mid.ServeHTTP(rec, req1)
mid.ServeHTTP(rec, req1.WithContext(ctx))
req2 := httptest.NewRequest(http.MethodGet, "/test2", nil)
mid.ServeHTTP(rec, req2)
mid.ServeHTTP(rec, req2.WithContext(ctx))
req3 := httptest.NewRequest(http.MethodGet, "/test3", nil)
mid.ServeHTTP(rec, req3)
mid.ServeHTTP(rec, req3.WithContext(ctx))
})

if !strings.Contains(got, `"severity":"ERROR"`) {
Expand Down Expand Up @@ -107,13 +108,13 @@ func TestNetHTTPMaxSeverityNoLog(t *testing.T) {
mid := middleware.NetHTTP("test/component")(mux)
rec := httptest.NewRecorder()

got := testhelper.ExtractAccessLogOut(t, func() {
got := testhelper.ExtractAccessLogOut(t, context.Background(), func(ctx context.Context) {
req1 := httptest.NewRequest(http.MethodGet, "/test1", nil)
mid.ServeHTTP(rec, req1)
mid.ServeHTTP(rec, req1.WithContext(ctx))
req2 := httptest.NewRequest(http.MethodGet, "/test2", nil)
mid.ServeHTTP(rec, req2)
mid.ServeHTTP(rec, req2.WithContext(ctx))
req3 := httptest.NewRequest(http.MethodGet, "/test3", nil)
mid.ServeHTTP(rec, req3)
mid.ServeHTTP(rec, req3.WithContext(ctx))
})

if !strings.Contains(got, `"severity":"ERROR"`) {
Expand Down
39 changes: 25 additions & 14 deletions testhelper/testhelper.go
Original file line number Diff line number Diff line change
Expand Up @@ -2,39 +2,50 @@ package testhelper

import (
"bytes"
"os"
"context"
"io"
"strings"
"testing"

"github.com/glassonion1/logz/internal/config"
)

// ExtractStdout extracts string from stdout
func ExtractApplicationLogOut(t *testing.T, fnc func()) string {
func ExtractApplicationLogOut(t *testing.T, ctx context.Context, fnc func(ctx context.Context)) string {
t.Helper()

var buf bytes.Buffer
config.ApplicationLogOut = &buf
defer func() {
config.ApplicationLogOut = os.Stdout
}()

fnc()
if cc := config.GetContextConfig(ctx); cc != nil {
cc.ApplicationLogOut = &buf
} else {
ctx = config.SetContextConfig(ctx, &config.ContextConfig{ApplicationLogOut: &buf})
}
fnc(ctx)

return strings.TrimRight(buf.String(), "\n")
}

// ExtractStdout extracts string from stderr
func ExtractAccessLogOut(t *testing.T, fnc func()) string {
func ExtractAccessLogOut(t *testing.T, ctx context.Context, fnc func(ctx context.Context)) string {
t.Helper()

var buf bytes.Buffer
config.AccessLogOut = &buf
defer func() {
config.AccessLogOut = os.Stdout
}()
if cc := config.GetContextConfig(ctx); cc != nil {
cc.AccessLogOut = &buf
} else {
ctx = config.SetContextConfig(ctx, &config.ContextConfig{AccessLogOut: &buf})
}

fnc()
fnc(ctx)

return strings.TrimRight(buf.String(), "\n")
}

// OverrideLogOutContext override log I/O in the context
func OverrideLogOutContext(t *testing.T, ctx context.Context, appLogOut, accessLogOut io.Writer) context.Context {
t.Helper()
return config.SetContextConfig(ctx, &config.ContextConfig{
ApplicationLogOut: appLogOut,
AccessLogOut: accessLogOut,
})
}

0 comments on commit f037fda

Please sign in to comment.