testlog.go 4.83 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
// Copyright 2019 The go-ethereum Authors
// This file is part of the go-ethereum library.
//
// The go-ethereum library is free software: you can redistribute it and/or modify
// it under the terms of the GNU Lesser General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// The go-ethereum library is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public License
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.

// Package testlog provides a log handler for unit tests.
package testlog

import (
21
	"os"
22 23 24
	"runtime"
	"strconv"
	"strings"
25 26 27 28 29
	"sync"

	"github.com/ethereum/go-ethereum/log"
)

30 31 32 33 34 35 36 37
var useColorInTestLog bool = true

func init() {
	if os.Getenv("OP_TESTLOG_DISABLE_COLOR") == "true" {
		useColorInTestLog = false
	}
}

38 39 40 41 42 43 44
// Testing interface to log to. Some functions are marked as Helper function to log the call site accurately.
// Standard Go testing.TB implements this, as well as Hive and other Go-like test frameworks.
type Testing interface {
	Logf(format string, args ...any)
	Helper()
}

45
// Handler returns a log handler which logs to the unit test log of t.
46
func Handler(t Testing, level log.Lvl) log.Handler {
47 48 49 50
	return log.LvlFilterHandler(level, &handler{t, log.TerminalFormat(false)})
}

type handler struct {
51
	t   Testing
52 53 54 55 56 57 58 59 60 61 62 63 64
	fmt log.Format
}

func (h *handler) Log(r *log.Record) error {
	h.t.Logf("%s", h.fmt.Format(r))
	return nil
}

// logger implements log.Logger such that all output goes to the unit test log via
// t.Logf(). All methods in between logger.Trace, logger.Debug, etc. are marked as test
// helpers, so the file and line number in unit test output correspond to the call site
// which emitted the log message.
type logger struct {
65
	t  Testing
66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81
	l  log.Logger
	mu *sync.Mutex
	h  *bufHandler
}

type bufHandler struct {
	buf []*log.Record
	fmt log.Format
}

func (h *bufHandler) Log(r *log.Record) error {
	h.buf = append(h.buf, r)
	return nil
}

// Logger returns a logger which logs to the unit test log of t.
82
func Logger(t Testing, level log.Lvl) log.Logger {
83 84 85 86
	l := &logger{
		t:  t,
		l:  log.New(),
		mu: new(sync.Mutex),
87
		h:  &bufHandler{fmt: log.TerminalFormat(useColorInTestLog)},
88 89 90 91 92
	}
	l.l.SetHandler(log.LvlFilterHandler(level, l.h))
	return l
}

93
func (l *logger) Trace(msg string, ctx ...any) {
94 95 96 97 98 99 100
	l.t.Helper()
	l.mu.Lock()
	defer l.mu.Unlock()
	l.l.Trace(msg, ctx...)
	l.flush()
}

101
func (l *logger) Debug(msg string, ctx ...any) {
102 103 104 105 106 107 108
	l.t.Helper()
	l.mu.Lock()
	defer l.mu.Unlock()
	l.l.Debug(msg, ctx...)
	l.flush()
}

109
func (l *logger) Info(msg string, ctx ...any) {
110 111 112 113 114 115 116
	l.t.Helper()
	l.mu.Lock()
	defer l.mu.Unlock()
	l.l.Info(msg, ctx...)
	l.flush()
}

117
func (l *logger) Warn(msg string, ctx ...any) {
118 119 120 121 122 123 124
	l.t.Helper()
	l.mu.Lock()
	defer l.mu.Unlock()
	l.l.Warn(msg, ctx...)
	l.flush()
}

125
func (l *logger) Error(msg string, ctx ...any) {
126 127 128 129 130 131 132
	l.t.Helper()
	l.mu.Lock()
	defer l.mu.Unlock()
	l.l.Error(msg, ctx...)
	l.flush()
}

133
func (l *logger) Crit(msg string, ctx ...any) {
134 135 136 137 138 139 140
	l.t.Helper()
	l.mu.Lock()
	defer l.mu.Unlock()
	l.l.Crit(msg, ctx...)
	l.flush()
}

141
func (l *logger) New(ctx ...any) log.Logger {
142 143 144 145 146 147 148 149 150 151 152 153 154 155
	return &logger{l.t, l.l.New(ctx...), l.mu, l.h}
}

func (l *logger) GetHandler() log.Handler {
	return l.l.GetHandler()
}

func (l *logger) SetHandler(h log.Handler) {
	l.l.SetHandler(h)
}

// flush writes all buffered messages and clears the buffer.
func (l *logger) flush() {
	l.t.Helper()
156 157
	// 2 frame skip for flush() + public logger fn
	decorationLen := estimateInfoLen(2)
158
	padding := 0
159 160 161
	padLength := 30
	if decorationLen <= padLength {
		padding = padLength - decorationLen
162
	}
163
	for _, r := range l.h.buf {
164
		l.t.Logf("%*s%s", padding, "", l.h.fmt.Format(r))
165 166 167
	}
	l.h.buf = nil
}
168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195

// The Go testing lib uses the runtime package to get info about the calling site, and then decorates the line.
// We can't disable this decoration, but we can adjust the contents to align by padding after the info.
// To pad the right amount, we estimate how long the info is.
func estimateInfoLen(frameSkip int) int {
	var pc [50]uintptr
	// Skip two extra frames to account for this function
	// and runtime.Callers itself.
	n := runtime.Callers(frameSkip+2, pc[:])
	if n == 0 {
		return 8
	}
	frames := runtime.CallersFrames(pc[:n])
	frame, _ := frames.Next()
	file := frame.File
	line := frame.Line
	if file != "" {
		// Truncate file name at last file name separator.
		if index := strings.LastIndex(file, "/"); index >= 0 {
			file = file[index+1:]
		} else if index = strings.LastIndex(file, "\\"); index >= 0 {
			file = file[index+1:]
		}
		return 4 + len(file) + 1 + len(strconv.FormatInt(int64(line), 10))
	} else {
		return 8
	}
}