| 
									
										
										
										
											2019-07-22 12:17:27 +03:00
										 |  |  | // Copyright 2019 The go-ethereum Authors | 
					
						
							| 
									
										
										
										
											2019-04-30 13:12:11 +02:00
										 |  |  | // 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 ( | 
					
						
							| 
									
										
										
										
											2020-01-08 16:11:51 +01:00
										 |  |  | 	"sync" | 
					
						
							| 
									
										
										
										
											2019-04-30 13:12:11 +02:00
										 |  |  | 	"testing" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/ethereum/go-ethereum/log" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Handler returns a log handler which logs to the unit test log of t. | 
					
						
							|  |  |  | func Handler(t *testing.T, level log.Lvl) log.Handler { | 
					
						
							|  |  |  | 	return log.LvlFilterHandler(level, &handler{t, log.TerminalFormat(false)}) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type handler struct { | 
					
						
							|  |  |  | 	t   *testing.T | 
					
						
							|  |  |  | 	fmt log.Format | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (h *handler) Log(r *log.Record) error { | 
					
						
							|  |  |  | 	h.t.Logf("%s", h.fmt.Format(r)) | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2020-01-08 16:11:51 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | // 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 { | 
					
						
							|  |  |  | 	t  *testing.T | 
					
						
							|  |  |  | 	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. | 
					
						
							|  |  |  | func Logger(t *testing.T, level log.Lvl) log.Logger { | 
					
						
							|  |  |  | 	l := &logger{ | 
					
						
							|  |  |  | 		t:  t, | 
					
						
							|  |  |  | 		l:  log.New(), | 
					
						
							|  |  |  | 		mu: new(sync.Mutex), | 
					
						
							|  |  |  | 		h:  &bufHandler{fmt: log.TerminalFormat(false)}, | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	l.l.SetHandler(log.LvlFilterHandler(level, l.h)) | 
					
						
							|  |  |  | 	return l | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (l *logger) Trace(msg string, ctx ...interface{}) { | 
					
						
							|  |  |  | 	l.t.Helper() | 
					
						
							|  |  |  | 	l.mu.Lock() | 
					
						
							|  |  |  | 	defer l.mu.Unlock() | 
					
						
							|  |  |  | 	l.l.Trace(msg, ctx...) | 
					
						
							|  |  |  | 	l.flush() | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (l *logger) Debug(msg string, ctx ...interface{}) { | 
					
						
							|  |  |  | 	l.t.Helper() | 
					
						
							|  |  |  | 	l.mu.Lock() | 
					
						
							|  |  |  | 	defer l.mu.Unlock() | 
					
						
							|  |  |  | 	l.l.Debug(msg, ctx...) | 
					
						
							|  |  |  | 	l.flush() | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (l *logger) Info(msg string, ctx ...interface{}) { | 
					
						
							|  |  |  | 	l.t.Helper() | 
					
						
							|  |  |  | 	l.mu.Lock() | 
					
						
							|  |  |  | 	defer l.mu.Unlock() | 
					
						
							|  |  |  | 	l.l.Info(msg, ctx...) | 
					
						
							|  |  |  | 	l.flush() | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (l *logger) Warn(msg string, ctx ...interface{}) { | 
					
						
							|  |  |  | 	l.t.Helper() | 
					
						
							|  |  |  | 	l.mu.Lock() | 
					
						
							|  |  |  | 	defer l.mu.Unlock() | 
					
						
							|  |  |  | 	l.l.Warn(msg, ctx...) | 
					
						
							|  |  |  | 	l.flush() | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (l *logger) Error(msg string, ctx ...interface{}) { | 
					
						
							|  |  |  | 	l.t.Helper() | 
					
						
							|  |  |  | 	l.mu.Lock() | 
					
						
							|  |  |  | 	defer l.mu.Unlock() | 
					
						
							|  |  |  | 	l.l.Error(msg, ctx...) | 
					
						
							|  |  |  | 	l.flush() | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (l *logger) Crit(msg string, ctx ...interface{}) { | 
					
						
							|  |  |  | 	l.t.Helper() | 
					
						
							|  |  |  | 	l.mu.Lock() | 
					
						
							|  |  |  | 	defer l.mu.Unlock() | 
					
						
							|  |  |  | 	l.l.Crit(msg, ctx...) | 
					
						
							|  |  |  | 	l.flush() | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (l *logger) New(ctx ...interface{}) log.Logger { | 
					
						
							|  |  |  | 	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() | 
					
						
							|  |  |  | 	for _, r := range l.h.buf { | 
					
						
							|  |  |  | 		l.t.Logf("%s", l.h.fmt.Format(r)) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	l.h.buf = nil | 
					
						
							|  |  |  | } |