mirror of
				https://github.com/go-gitea/gitea.git
				synced 2025-10-31 02:46:04 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			180 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			180 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2023 The Gitea Authors. All rights reserved.
 | |
| // SPDX-License-Identifier: MIT
 | |
| 
 | |
| package log
 | |
| 
 | |
| import (
 | |
| 	"regexp"
 | |
| 	"sync"
 | |
| 	"testing"
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/stretchr/testify/assert"
 | |
| )
 | |
| 
 | |
| type dummyWriter struct {
 | |
| 	*EventWriterBaseImpl
 | |
| 
 | |
| 	delay time.Duration
 | |
| 
 | |
| 	mu   sync.Mutex
 | |
| 	logs []string
 | |
| }
 | |
| 
 | |
| func (d *dummyWriter) Write(p []byte) (n int, err error) {
 | |
| 	if d.delay > 0 {
 | |
| 		time.Sleep(d.delay)
 | |
| 	}
 | |
| 	d.mu.Lock()
 | |
| 	defer d.mu.Unlock()
 | |
| 	d.logs = append(d.logs, string(p))
 | |
| 	return len(p), nil
 | |
| }
 | |
| 
 | |
| func (d *dummyWriter) Close() error {
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (d *dummyWriter) FetchLogs() []string {
 | |
| 	d.mu.Lock()
 | |
| 	defer d.mu.Unlock()
 | |
| 	logs := d.logs
 | |
| 	d.logs = nil
 | |
| 	return logs
 | |
| }
 | |
| 
 | |
| func newDummyWriter(name string, level Level, delay time.Duration) *dummyWriter {
 | |
| 	w := &dummyWriter{
 | |
| 		EventWriterBaseImpl: NewEventWriterBase(name, "dummy", WriterMode{Level: level, Flags: FlagsFromBits(0)}),
 | |
| 	}
 | |
| 	w.delay = delay
 | |
| 	w.Base().OutputWriteCloser = w
 | |
| 	return w
 | |
| }
 | |
| 
 | |
| func TestLogger(t *testing.T) {
 | |
| 	logger := NewLoggerWithWriters(t.Context(), "test")
 | |
| 
 | |
| 	dump := logger.DumpWriters()
 | |
| 	assert.Empty(t, dump)
 | |
| 	assert.Equal(t, NONE, logger.GetLevel())
 | |
| 	assert.False(t, logger.IsEnabled())
 | |
| 
 | |
| 	w1 := newDummyWriter("dummy-1", DEBUG, 0)
 | |
| 	logger.AddWriters(w1)
 | |
| 	assert.Equal(t, DEBUG, logger.GetLevel())
 | |
| 
 | |
| 	w2 := newDummyWriter("dummy-2", WARN, 200*time.Millisecond)
 | |
| 	logger.AddWriters(w2)
 | |
| 	assert.Equal(t, DEBUG, logger.GetLevel())
 | |
| 
 | |
| 	dump = logger.DumpWriters()
 | |
| 	assert.Len(t, dump, 2)
 | |
| 
 | |
| 	logger.Trace("trace-level") // this level is not logged
 | |
| 	logger.Debug("debug-level")
 | |
| 	logger.Error("error-level")
 | |
| 
 | |
| 	// w2 is slow, so only w1 has logs
 | |
| 	time.Sleep(100 * time.Millisecond)
 | |
| 	assert.Equal(t, []string{"debug-level\n", "error-level\n"}, w1.FetchLogs())
 | |
| 	assert.Empty(t, w2.FetchLogs())
 | |
| 
 | |
| 	logger.Close()
 | |
| 
 | |
| 	// after Close, all logs are flushed
 | |
| 	assert.Empty(t, w1.FetchLogs())
 | |
| 	assert.Equal(t, []string{"error-level\n"}, w2.FetchLogs())
 | |
| }
 | |
| 
 | |
| func TestLoggerPause(t *testing.T) {
 | |
| 	logger := NewLoggerWithWriters(t.Context(), "test")
 | |
| 
 | |
| 	w1 := newDummyWriter("dummy-1", DEBUG, 0)
 | |
| 	logger.AddWriters(w1)
 | |
| 
 | |
| 	GetManager().PauseAll()
 | |
| 	time.Sleep(50 * time.Millisecond)
 | |
| 
 | |
| 	logger.Info("info-level")
 | |
| 	time.Sleep(100 * time.Millisecond)
 | |
| 	assert.Empty(t, w1.FetchLogs())
 | |
| 
 | |
| 	GetManager().ResumeAll()
 | |
| 
 | |
| 	time.Sleep(100 * time.Millisecond)
 | |
| 	assert.Equal(t, []string{"info-level\n"}, w1.FetchLogs())
 | |
| 
 | |
| 	logger.Close()
 | |
| }
 | |
| 
 | |
| type testLogString struct {
 | |
| 	Field string
 | |
| }
 | |
| 
 | |
| func (t testLogString) LogString() string {
 | |
| 	return "log-string"
 | |
| }
 | |
| 
 | |
| type testLogStringPtrReceiver struct {
 | |
| 	Field string
 | |
| }
 | |
| 
 | |
| func (t *testLogStringPtrReceiver) LogString() string {
 | |
| 	return "log-string-ptr-receiver"
 | |
| }
 | |
| 
 | |
| func genericFunc[T any](logger Logger, v T) {
 | |
| 	logger.Info("from genericFunc: %v", v)
 | |
| }
 | |
| 
 | |
| func TestLoggerOutput(t *testing.T) {
 | |
| 	t.Run("LogString", func(t *testing.T) {
 | |
| 		logger := NewLoggerWithWriters(t.Context(), "test")
 | |
| 		w1 := newDummyWriter("dummy-1", DEBUG, 0)
 | |
| 		w1.Mode.Colorize = true
 | |
| 		logger.AddWriters(w1)
 | |
| 		logger.Info("%s %s %#v %v", testLogString{}, &testLogString{}, testLogString{Field: "detail"}, NewColoredValue(testLogString{}, FgRed))
 | |
| 		logger.Info("%s %s %#v %v", testLogStringPtrReceiver{}, &testLogStringPtrReceiver{}, testLogStringPtrReceiver{Field: "detail"}, NewColoredValue(testLogStringPtrReceiver{}, FgRed))
 | |
| 		logger.Close()
 | |
| 
 | |
| 		assert.Equal(t, []string{
 | |
| 			"log-string log-string log.testLogString{Field:\"detail\"} \x1b[31mlog-string\x1b[0m\n",
 | |
| 			"log-string-ptr-receiver log-string-ptr-receiver &log.testLogStringPtrReceiver{Field:\"detail\"} \x1b[31mlog-string-ptr-receiver\x1b[0m\n",
 | |
| 		}, w1.FetchLogs())
 | |
| 	})
 | |
| 
 | |
| 	t.Run("Caller", func(t *testing.T) {
 | |
| 		logger := NewLoggerWithWriters(t.Context(), "test")
 | |
| 		w1 := newDummyWriter("dummy-1", DEBUG, 0)
 | |
| 		w1.EventWriterBaseImpl.Mode.Flags.flags = Lmedfile | Lshortfuncname
 | |
| 		logger.AddWriters(w1)
 | |
| 		anonymousFunc := func(logger Logger) {
 | |
| 			logger.Info("from anonymousFunc")
 | |
| 		}
 | |
| 		genericFunc(logger, "123")
 | |
| 		anonymousFunc(logger)
 | |
| 		logger.Close()
 | |
| 		logs := w1.FetchLogs()
 | |
| 		assert.Len(t, logs, 2)
 | |
| 		assert.Regexp(t, `modules/log/logger_test.go:\w+:`+regexp.QuoteMeta(`genericFunc() from genericFunc: 123`), logs[0])
 | |
| 		assert.Regexp(t, `modules/log/logger_test.go:\w+:`+regexp.QuoteMeta(`TestLoggerOutput.2.1() from anonymousFunc`), logs[1])
 | |
| 	})
 | |
| }
 | |
| 
 | |
| func TestLoggerExpressionFilter(t *testing.T) {
 | |
| 	logger := NewLoggerWithWriters(t.Context(), "test")
 | |
| 
 | |
| 	w1 := newDummyWriter("dummy-1", DEBUG, 0)
 | |
| 	w1.Mode.Expression = "foo.*"
 | |
| 	logger.AddWriters(w1)
 | |
| 
 | |
| 	logger.Info("foo")
 | |
| 	logger.Info("bar")
 | |
| 	logger.Info("foo bar")
 | |
| 	logger.SendLogEvent(&Event{Level: INFO, Filename: "foo.go", MsgSimpleText: "by filename"})
 | |
| 	logger.Close()
 | |
| 
 | |
| 	assert.Equal(t, []string{"foo\n", "foo bar\n", "by filename\n"}, w1.FetchLogs())
 | |
| }
 |