iris/middleware/accesslog/accesslog_test.go
Gerasimos (Makis) Maropoulos 7431fcc9cf
more tiny improvements
2020-09-13 14:28:05 +03:00

239 lines
4.7 KiB
Go

package accesslog
import (
"bytes"
"fmt"
"io"
"io/ioutil"
"net/http"
"net/http/httptest"
"strings"
"sync"
"sync/atomic"
"testing"
"time"
"github.com/kataras/iris/v12/context"
"github.com/kataras/iris/v12/core/memstore"
)
func TestAccessLogPrint_Simple(t *testing.T) {
const goroutinesN = 42
w := new(bytes.Buffer)
ac := New(w)
ac.Async = true
ac.ResponseBody = true
ac.Clock = TClock(time.Time{})
var (
expected string
expectedLines int
mu sync.Mutex
)
now := time.Now()
for i := 0; i < goroutinesN; i++ {
go func() {
err := ac.Print(
nil,
1*time.Second,
ac.TimeFormat,
200,
"GET",
"/path_value?url_query=url_query_value",
"::1",
"Incoming",
"Outcoming",
0,
0,
memstore.Store{
{Key: "path_param", ValueRaw: "path_param_value"},
},
[]memstore.StringEntry{
{Key: "url_query", Value: "url_query_value"},
},
[]memstore.Entry{
{Key: "custom", ValueRaw: "custom_value"},
})
if err == nil {
mu.Lock()
expected += "0001-01-01 00:00:00|1s|200|GET|/path_value?url_query=url_query_value|::1|path_param=path_param_value url_query=url_query_value custom=custom_value|0 B|0 B|Incoming|Outcoming|\n"
expectedLines++
mu.Unlock()
}
}()
}
// give some time to write at least some messages or all
// (depends on the machine the test is running).
time.Sleep(42 * time.Millisecond)
ac.Close()
end := time.Since(now)
if got := atomic.LoadUint32(&ac.remaining); got > 0 { // test wait.
t.Fatalf("expected remaining: %d but got: %d", 0, got)
}
mu.Lock()
expectedSoFoar := expected
expectedLinesSoFar := expectedLines
mu.Unlock()
if got := w.String(); expectedSoFoar != got {
gotLines := strings.Count(got, "\n")
t.Logf("expected printed result to be[%d]:\n'%s'\n\nbut got[%d]:\n'%s'", expectedLinesSoFar, expectedSoFoar, gotLines, got)
t.Fatalf("expected: %d | got: %d lines", expectedLinesSoFar, gotLines)
} else {
t.Logf("We've got [%d/%d] lines of logs in %s", expectedLinesSoFar, goroutinesN, end.String())
}
}
func TestAccessLogBroker(t *testing.T) {
w := new(bytes.Buffer)
ac := New(w)
ac.Clock = TClock(time.Time{})
broker := ac.Broker()
wg := new(sync.WaitGroup)
n := 4
wg.Add(4)
go func() {
i := 0
ln := broker.NewListener()
for {
select {
case log, ok := <-ln:
if !ok {
if i != n {
for i < n {
wg.Done()
i++
}
}
t.Log("Log Listener Closed: interrupted")
return
}
lat := log.Latency
t.Log(lat.String())
wg.Done()
if expected := time.Duration(i) * time.Second; expected != lat {
panic(fmt.Sprintf("expected latency: %s but got: %s", expected, lat))
}
time.Sleep(1350 * time.Millisecond)
if log.Latency != lat {
panic("expected logger to wait for notifier before release the log")
}
i++
}
}
}()
time.Sleep(time.Second)
printLog := func(lat time.Duration) {
err := ac.Print(
nil,
lat,
"",
0,
"",
"",
"",
"",
"",
0,
0,
nil,
nil,
nil,
)
if err != nil {
t.Fatal(err)
}
}
for i := 0; i < n; i++ {
printLog(time.Duration(i) * time.Second)
}
// wait for all listeners to finish.
wg.Wait()
ac.Close()
}
type noOpFormatter struct{}
func (*noOpFormatter) SetOutput(io.Writer) {}
// Format prints the logs in text/template format.
func (*noOpFormatter) Format(*Log) (bool, error) {
return true, nil
}
// go test -run=^$ -bench=BenchmarkAccessLogAfter -benchmem
func BenchmarkAccessLogAfter(b *testing.B) {
benchmarkAccessLogAfter(b, true, false)
}
func BenchmarkAccessLogAfterPrint(b *testing.B) {
benchmarkAccessLogAfter(b, false, false)
}
func benchmarkAccessLogAfter(b *testing.B, withLogStruct, async bool) {
ac := New(ioutil.Discard)
ac.Clock = TClock(time.Time{})
ac.BytesReceived = false
ac.BytesReceivedBody = false
ac.BytesSent = false
ac.BytesSentBody = false
ac.BodyMinify = false
ac.RequestBody = false
ac.ResponseBody = false
ac.Async = false
ac.IP = false
if withLogStruct {
ac.SetFormatter(new(noOpFormatter)) // just to create the log structure, here we test the log creation time only.
}
ctx := new(context.Context)
req, err := http.NewRequest("GET", "/", nil)
if err != nil {
b.Fatal(err)
}
ctx.ResetRequest(req)
recorder := httptest.NewRecorder()
w := context.AcquireResponseWriter()
w.BeginResponse(recorder)
ctx.ResetResponseWriter(w)
wg := new(sync.WaitGroup)
if async {
wg.Add(b.N)
}
b.ResetTimer()
for n := 0; n < b.N; n++ {
if async {
go func() {
ac.after(ctx, time.Millisecond, "GET", "/")
wg.Done()
}()
} else {
ac.after(ctx, time.Millisecond, "GET", "/")
}
}
b.StopTimer()
if async {
wg.Wait()
}
w.EndResponse()
}