2020-09-07 15:31:43 +02:00
|
|
|
package accesslog
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2020-09-10 18:41:30 +02:00
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"io/ioutil"
|
|
|
|
"net/http"
|
|
|
|
"net/http/httptest"
|
2020-09-13 01:56:22 +02:00
|
|
|
"strings"
|
2020-09-07 15:31:43 +02:00
|
|
|
"sync"
|
2020-09-13 01:56:22 +02:00
|
|
|
"sync/atomic"
|
2020-09-07 15:31:43 +02:00
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/kataras/iris/v12/context"
|
|
|
|
"github.com/kataras/iris/v12/core/memstore"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestAccessLogPrint_Simple(t *testing.T) {
|
2020-09-13 13:28:05 +02:00
|
|
|
const goroutinesN = 42
|
2020-09-07 15:31:43 +02:00
|
|
|
|
|
|
|
w := new(bytes.Buffer)
|
2020-09-07 17:04:35 +02:00
|
|
|
ac := New(w)
|
2020-09-13 01:56:22 +02:00
|
|
|
ac.Async = true
|
|
|
|
ac.ResponseBody = true
|
2020-09-07 15:31:43 +02:00
|
|
|
ac.Clock = TClock(time.Time{})
|
|
|
|
|
2020-09-13 03:20:00 +02:00
|
|
|
var (
|
|
|
|
expected string
|
|
|
|
expectedLines int
|
|
|
|
mu sync.Mutex
|
|
|
|
)
|
2020-09-07 15:31:43 +02:00
|
|
|
|
2020-09-13 13:28:05 +02:00
|
|
|
now := time.Now()
|
2020-09-13 03:20:00 +02:00
|
|
|
for i := 0; i < goroutinesN; i++ {
|
2020-09-07 15:31:43 +02:00
|
|
|
go func() {
|
2020-09-13 13:28:05 +02:00
|
|
|
err := ac.Print(
|
2020-09-07 15:31:43 +02:00
|
|
|
nil,
|
|
|
|
1*time.Second,
|
|
|
|
ac.TimeFormat,
|
|
|
|
200,
|
|
|
|
"GET",
|
|
|
|
"/path_value?url_query=url_query_value",
|
2020-09-11 08:38:55 +02:00
|
|
|
"::1",
|
2020-09-07 15:31:43 +02:00
|
|
|
"Incoming",
|
|
|
|
"Outcoming",
|
2020-09-08 12:44:50 +02:00
|
|
|
0,
|
|
|
|
0,
|
2020-09-12 11:34:59 +02:00
|
|
|
memstore.Store{
|
|
|
|
{Key: "path_param", ValueRaw: "path_param_value"},
|
|
|
|
},
|
|
|
|
[]memstore.StringEntry{
|
2020-09-07 15:31:43 +02:00
|
|
|
{Key: "url_query", Value: "url_query_value"},
|
2020-09-12 11:34:59 +02:00
|
|
|
},
|
|
|
|
[]memstore.Entry{
|
2020-09-07 15:31:43 +02:00
|
|
|
{Key: "custom", ValueRaw: "custom_value"},
|
|
|
|
})
|
2020-09-13 13:28:05 +02:00
|
|
|
|
|
|
|
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()
|
|
|
|
}
|
2020-09-07 15:31:43 +02:00
|
|
|
}()
|
2020-09-13 01:56:22 +02:00
|
|
|
|
2020-09-07 15:31:43 +02:00
|
|
|
}
|
|
|
|
|
2020-09-13 13:28:05 +02:00
|
|
|
// 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)
|
2020-09-13 01:56:22 +02:00
|
|
|
|
|
|
|
if got := atomic.LoadUint32(&ac.remaining); got > 0 { // test wait.
|
|
|
|
t.Fatalf("expected remaining: %d but got: %d", 0, got)
|
|
|
|
}
|
2020-09-07 15:31:43 +02:00
|
|
|
|
2020-09-13 13:28:05 +02:00
|
|
|
mu.Lock()
|
|
|
|
expectedSoFoar := expected
|
|
|
|
expectedLinesSoFar := expectedLines
|
|
|
|
mu.Unlock()
|
|
|
|
|
|
|
|
if got := w.String(); expectedSoFoar != got {
|
2020-09-13 01:56:22 +02:00
|
|
|
gotLines := strings.Count(got, "\n")
|
2020-09-13 13:28:05 +02:00
|
|
|
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())
|
2020-09-07 15:31:43 +02:00
|
|
|
}
|
|
|
|
}
|
2020-09-10 18:41:30 +02:00
|
|
|
|
|
|
|
func TestAccessLogBroker(t *testing.T) {
|
|
|
|
w := new(bytes.Buffer)
|
|
|
|
ac := New(w)
|
2020-09-13 01:56:22 +02:00
|
|
|
|
2020-09-10 18:41:30 +02:00
|
|
|
ac.Clock = TClock(time.Time{})
|
|
|
|
broker := ac.Broker()
|
|
|
|
|
|
|
|
wg := new(sync.WaitGroup)
|
|
|
|
n := 4
|
2020-09-14 16:35:17 +02:00
|
|
|
wg.Add(n)
|
2020-09-10 18:41:30 +02:00
|
|
|
go func() {
|
|
|
|
i := 0
|
|
|
|
ln := broker.NewListener()
|
2020-09-13 13:28:05 +02:00
|
|
|
|
2021-01-09 04:41:20 +01:00
|
|
|
for log := range ln {
|
|
|
|
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++
|
|
|
|
}
|
2020-09-13 13:28:05 +02:00
|
|
|
|
2021-01-09 04:41:20 +01:00
|
|
|
if i != n {
|
|
|
|
for i < n {
|
2020-09-10 18:41:30 +02:00
|
|
|
wg.Done()
|
2020-09-13 13:28:05 +02:00
|
|
|
i++
|
2020-09-10 18:41:30 +02:00
|
|
|
}
|
|
|
|
}
|
2021-01-09 04:41:20 +01:00
|
|
|
|
|
|
|
t.Log("Log Listener Closed: interrupted")
|
2020-09-10 18:41:30 +02:00
|
|
|
}()
|
|
|
|
|
|
|
|
time.Sleep(time.Second)
|
|
|
|
|
|
|
|
printLog := func(lat time.Duration) {
|
|
|
|
err := ac.Print(
|
|
|
|
nil,
|
|
|
|
lat,
|
|
|
|
"",
|
|
|
|
0,
|
|
|
|
"",
|
|
|
|
"",
|
|
|
|
"",
|
|
|
|
"",
|
2020-09-11 08:38:55 +02:00
|
|
|
"",
|
2020-09-10 18:41:30 +02:00
|
|
|
0,
|
|
|
|
0,
|
2020-09-12 11:34:59 +02:00
|
|
|
nil,
|
2020-09-10 18:41:30 +02:00
|
|
|
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()
|
|
|
|
|
2020-09-13 01:56:22 +02:00
|
|
|
ac.Close()
|
2020-09-10 18:41:30 +02:00
|
|
|
}
|
|
|
|
|
2020-09-13 16:01:57 +02:00
|
|
|
func TestAccessLogBlank(t *testing.T) {
|
|
|
|
w := new(bytes.Buffer)
|
|
|
|
ac := New(w)
|
|
|
|
clockTime, _ := time.Parse(defaultTimeFormat, "1993-01-01 05:00:00")
|
|
|
|
ac.Clock = TClock(clockTime)
|
|
|
|
ac.Blank = []byte("<no value>")
|
|
|
|
|
|
|
|
ac.Print(
|
|
|
|
nil,
|
|
|
|
time.Second,
|
|
|
|
defaultTimeFormat,
|
|
|
|
200,
|
|
|
|
"GET",
|
|
|
|
"/",
|
|
|
|
"127.0.0.1",
|
|
|
|
"",
|
|
|
|
"",
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
nil,
|
|
|
|
nil,
|
|
|
|
nil,
|
|
|
|
)
|
|
|
|
|
|
|
|
ac.Close()
|
|
|
|
// the request and bodies length are enabled by-default, zero bytes length
|
|
|
|
// are written with 0 B and this cannot changed, so the request field
|
|
|
|
// should be written as "<no value>".
|
|
|
|
expected := "1993-01-01 05:00:00|1s|200|GET|/|127.0.0.1|0 B|0 B|<no value>|\n"
|
|
|
|
if got := w.String(); expected != got {
|
|
|
|
t.Fatalf("expected:\n'%s'\n\nbut got:\n'%s'", expected, got)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-09-14 16:35:17 +02:00
|
|
|
type slowClose struct{ *bytes.Buffer }
|
|
|
|
|
|
|
|
func (c *slowClose) Close() error {
|
|
|
|
time.Sleep(1 * time.Second)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestAccessLogSetOutput(t *testing.T) {
|
|
|
|
var (
|
|
|
|
w1 = &bytes.Buffer{}
|
|
|
|
w2 = &bytes.Buffer{}
|
|
|
|
w3 = &slowClose{&bytes.Buffer{}}
|
|
|
|
w4 = &bytes.Buffer{}
|
|
|
|
)
|
|
|
|
|
|
|
|
ac := New(w1)
|
|
|
|
ac.Clock = TClock(time.Time{})
|
|
|
|
|
|
|
|
n := 40
|
|
|
|
expected := strings.Repeat("0001-01-01 00:00:00|1s|200|GET|/|127.0.0.1|0 B|0 B||\n", n)
|
|
|
|
|
|
|
|
printLog := func() {
|
|
|
|
err := ac.Print(
|
|
|
|
nil,
|
|
|
|
time.Second,
|
|
|
|
defaultTimeFormat,
|
|
|
|
200,
|
|
|
|
"GET",
|
|
|
|
"/",
|
|
|
|
"127.0.0.1",
|
|
|
|
"",
|
|
|
|
"",
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
nil,
|
|
|
|
nil,
|
|
|
|
nil,
|
|
|
|
)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
testSetOutput := func(name string, w io.Writer, withSlowClose bool) {
|
|
|
|
wg := new(sync.WaitGroup)
|
|
|
|
wg.Add(n / 4)
|
|
|
|
for i := 0; i < n/4; i++ {
|
|
|
|
go func(i int) {
|
|
|
|
defer wg.Done()
|
|
|
|
|
|
|
|
if i%2 == 0 {
|
|
|
|
time.Sleep(10 * time.Millisecond)
|
|
|
|
}
|
|
|
|
|
2021-01-09 04:41:20 +01:00
|
|
|
if i == 5 {
|
|
|
|
if w != nil {
|
|
|
|
now := time.Now()
|
|
|
|
ac.SetOutput(w)
|
|
|
|
if withSlowClose {
|
|
|
|
end := time.Since(now)
|
|
|
|
if end < time.Second {
|
|
|
|
panic(fmt.Sprintf("[%s] [%d]: SetOutput should wait for previous Close. Expected to return a bit after %s but %s", name, i, time.Second, end))
|
|
|
|
}
|
2020-09-14 16:35:17 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
printLog()
|
|
|
|
}(i)
|
|
|
|
}
|
|
|
|
|
|
|
|
// wait to finish.
|
|
|
|
wg.Wait()
|
|
|
|
}
|
|
|
|
|
|
|
|
go testSetOutput("w1", nil, false) // write at least one line and then
|
|
|
|
time.Sleep(100 * time.Millisecond) // concurrently
|
|
|
|
testSetOutput("w2", w2, false) // change the writer
|
|
|
|
testSetOutput("w3", w3, false)
|
|
|
|
testSetOutput("w4", w4, true)
|
|
|
|
|
|
|
|
gotAll := w1.String() + w2.String() + w3.String() + w4.String()
|
|
|
|
|
|
|
|
// test if all content written and we have no loses.
|
|
|
|
if expected != gotAll {
|
|
|
|
t.Fatalf("expected total written result to be:\n'%s'\n\nbut got:\n'%s'", expected, gotAll)
|
|
|
|
}
|
|
|
|
|
|
|
|
// now, check if all have contents, they should because we wait between them,
|
|
|
|
// contents spread.
|
|
|
|
checkLines := func(name, s string, minimumLines int) {
|
|
|
|
if got := strings.Count(s, "\n"); got < minimumLines {
|
|
|
|
t.Logf("[%s] expected minimum lines of: %d but got %d", name, minimumLines, got)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
checkLines("w1", w1.String(), 1)
|
|
|
|
checkLines("w2", w2.String(), 5)
|
|
|
|
checkLines("w3", w3.String(), 5)
|
|
|
|
checkLines("w4", w4.String(), 5)
|
|
|
|
|
|
|
|
if err := ac.Close(); err != nil {
|
|
|
|
t.Fatalf("On close: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-09-10 18:41:30 +02:00
|
|
|
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) {
|
2020-09-11 22:22:18 +02:00
|
|
|
benchmarkAccessLogAfter(b, true, false)
|
|
|
|
}
|
|
|
|
|
|
|
|
func BenchmarkAccessLogAfterPrint(b *testing.B) {
|
|
|
|
benchmarkAccessLogAfter(b, false, false)
|
|
|
|
}
|
|
|
|
|
|
|
|
func benchmarkAccessLogAfter(b *testing.B, withLogStruct, async bool) {
|
2020-09-10 18:41:30 +02:00
|
|
|
ac := New(ioutil.Discard)
|
|
|
|
ac.Clock = TClock(time.Time{})
|
|
|
|
ac.BytesReceived = false
|
2020-09-13 01:56:22 +02:00
|
|
|
ac.BytesReceivedBody = false
|
2020-09-10 18:41:30 +02:00
|
|
|
ac.BytesSent = false
|
2020-09-13 01:56:22 +02:00
|
|
|
ac.BytesSentBody = false
|
2020-09-10 18:41:30 +02:00
|
|
|
ac.BodyMinify = false
|
|
|
|
ac.RequestBody = false
|
|
|
|
ac.ResponseBody = false
|
|
|
|
ac.Async = false
|
2020-09-11 08:38:55 +02:00
|
|
|
ac.IP = false
|
2020-09-11 22:22:18 +02:00
|
|
|
if withLogStruct {
|
|
|
|
ac.SetFormatter(new(noOpFormatter)) // just to create the log structure, here we test the log creation time only.
|
|
|
|
}
|
2020-09-10 18:41:30 +02:00
|
|
|
|
|
|
|
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)
|
|
|
|
|
2020-09-11 22:22:18 +02:00
|
|
|
wg := new(sync.WaitGroup)
|
|
|
|
if async {
|
|
|
|
wg.Add(b.N)
|
|
|
|
}
|
|
|
|
|
2020-09-10 18:41:30 +02:00
|
|
|
b.ResetTimer()
|
|
|
|
for n := 0; n < b.N; n++ {
|
2020-09-11 22:22:18 +02:00
|
|
|
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()
|
2020-09-10 18:41:30 +02:00
|
|
|
}
|
|
|
|
w.EndResponse()
|
|
|
|
}
|