loggers_test.go 4.6 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
// Copyright 2014 The go-ethereum Authors
// This file is part of go-ethereum.
//
// go-ethereum 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.
//
// go-ethereum 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 go-ethereum.  If not, see <http://www.gnu.org/licenses/>.

obscuren's avatar
obscuren committed
17
package logger
zelig's avatar
zelig committed
18 19

import (
zelig's avatar
zelig committed
20
	"io/ioutil"
21
	"math/rand"
zelig's avatar
zelig committed
22
	"os"
23
	"sync"
zelig's avatar
zelig committed
24
	"testing"
25
	"time"
zelig's avatar
zelig committed
26 27 28
)

type TestLogSystem struct {
29 30
	mutex  sync.Mutex
	output string
zelig's avatar
zelig committed
31
	level  LogLevel
zelig's avatar
zelig committed
32 33
}

34
func (ls *TestLogSystem) LogPrint(msg LogMsg) {
35
	ls.mutex.Lock()
36 37 38
	if ls.level >= msg.Level() {
		ls.output += msg.String()
	}
39
	ls.mutex.Unlock()
zelig's avatar
zelig committed
40 41
}

42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60
func (ls *TestLogSystem) SetLogLevel(i LogLevel) {
	ls.mutex.Lock()
	ls.level = i
	ls.mutex.Unlock()
}

func (ls *TestLogSystem) GetLogLevel() LogLevel {
	ls.mutex.Lock()
	defer ls.mutex.Unlock()
	return ls.level
}

func (ls *TestLogSystem) CheckOutput(t *testing.T, expected string) {
	ls.mutex.Lock()
	output := ls.output
	ls.mutex.Unlock()
	if output != expected {
		t.Errorf("log output mismatch:\n   got: %q\n  want: %q\n", output, expected)
	}
zelig's avatar
zelig committed
61 62
}

63 64 65 66 67
type blockedLogSystem struct {
	LogSystem
	unblock chan struct{}
}

68
func (ls blockedLogSystem) LogPrint(msg LogMsg) {
69
	<-ls.unblock
70
	ls.LogSystem.LogPrint(msg)
zelig's avatar
zelig committed
71 72
}

zelig's avatar
zelig committed
73
func TestLoggerFlush(t *testing.T) {
74 75
	Reset()

zelig's avatar
zelig committed
76
	logger := NewLogger("TEST")
77 78
	ls := blockedLogSystem{&TestLogSystem{level: WarnLevel}, make(chan struct{})}
	AddLogSystem(ls)
zelig's avatar
zelig committed
79
	for i := 0; i < 5; i++ {
80
		// these writes shouldn't hang even though ls is blocked
zelig's avatar
zelig committed
81 82
		logger.Errorf(".")
	}
83 84 85 86 87 88

	beforeFlush := time.Now()
	time.AfterFunc(80*time.Millisecond, func() { close(ls.unblock) })
	Flush() // this should hang for approx. 80ms
	if blockd := time.Now().Sub(beforeFlush); blockd < 80*time.Millisecond {
		t.Errorf("Flush didn't block long enough, blocked for %v, should've been >= 80ms", blockd)
zelig's avatar
zelig committed
89
	}
90 91

	ls.LogSystem.(*TestLogSystem).CheckOutput(t, "[TEST] .[TEST] .[TEST] .[TEST] .[TEST] .")
zelig's avatar
zelig committed
92 93
}

zelig's avatar
zelig committed
94
func TestLoggerPrintln(t *testing.T) {
95 96
	Reset()

zelig's avatar
zelig committed
97 98 99 100 101 102 103 104
	logger := NewLogger("TEST")
	testLogSystem := &TestLogSystem{level: WarnLevel}
	AddLogSystem(testLogSystem)
	logger.Errorln("error")
	logger.Warnln("warn")
	logger.Infoln("info")
	logger.Debugln("debug")
	Flush()
105 106

	testLogSystem.CheckOutput(t, "[TEST] error\n[TEST] warn\n")
zelig's avatar
zelig committed
107 108 109
}

func TestLoggerPrintf(t *testing.T) {
110 111
	Reset()

zelig's avatar
zelig committed
112 113 114
	logger := NewLogger("TEST")
	testLogSystem := &TestLogSystem{level: WarnLevel}
	AddLogSystem(testLogSystem)
115
	logger.Errorf("error to %v\n", []int{1, 2, 3})
116
	logger.Warnf("warn %%d %d", 5)
zelig's avatar
zelig committed
117 118 119
	logger.Infof("info")
	logger.Debugf("debug")
	Flush()
120
	testLogSystem.CheckOutput(t, "[TEST] error to [1 2 3]\n[TEST] warn %d 5")
zelig's avatar
zelig committed
121 122 123
}

func TestMultipleLogSystems(t *testing.T) {
124 125
	Reset()

zelig's avatar
zelig committed
126 127 128 129 130 131 132 133
	logger := NewLogger("TEST")
	testLogSystem0 := &TestLogSystem{level: ErrorLevel}
	testLogSystem1 := &TestLogSystem{level: WarnLevel}
	AddLogSystem(testLogSystem0)
	AddLogSystem(testLogSystem1)
	logger.Errorln("error")
	logger.Warnln("warn")
	Flush()
134 135 136

	testLogSystem0.CheckOutput(t, "[TEST] error\n")
	testLogSystem1.CheckOutput(t, "[TEST] error\n[TEST] warn\n")
zelig's avatar
zelig committed
137 138 139
}

func TestFileLogSystem(t *testing.T) {
140 141
	Reset()

zelig's avatar
zelig committed
142 143 144 145 146 147 148 149 150 151 152
	logger := NewLogger("TEST")
	filename := "test.log"
	file, _ := os.OpenFile(filename, os.O_RDWR|os.O_CREATE, os.ModePerm)
	testLogSystem := NewStdLogSystem(file, 0, WarnLevel)
	AddLogSystem(testLogSystem)
	logger.Errorf("error to %s\n", filename)
	logger.Warnln("warn")
	Flush()
	contents, _ := ioutil.ReadFile(filename)
	output := string(contents)
	if output != "[TEST] error to test.log\n[TEST] warn\n" {
zelig's avatar
zelig committed
153
		t.Error("Expected contents of file 'test.log': '[TEST] error to test.log\\n[TEST] warn\\n', got ", output)
zelig's avatar
zelig committed
154 155 156
	} else {
		os.Remove(filename)
	}
zelig's avatar
zelig committed
157 158 159
}

func TestNoLogSystem(t *testing.T) {
160 161
	Reset()

zelig's avatar
zelig committed
162 163 164
	logger := NewLogger("TEST")
	logger.Warnln("warn")
	Flush()
zelig's avatar
zelig committed
165
}
166 167 168 169 170 171 172 173 174 175 176 177

func TestConcurrentAddSystem(t *testing.T) {
	rand.Seed(time.Now().Unix())
	Reset()

	logger := NewLogger("TEST")
	stop := make(chan struct{})
	writer := func() {
		select {
		case <-stop:
			return
		default:
178
			logger.Infoln("foo")
179 180 181 182 183 184 185 186 187 188
			Flush()
		}
	}

	go writer()
	go writer()

	stopTime := time.Now().Add(100 * time.Millisecond)
	for time.Now().Before(stopTime) {
		time.Sleep(time.Duration(rand.Intn(20)) * time.Millisecond)
189
		AddLogSystem(NewStdLogSystem(ioutil.Discard, 0, InfoLevel))
190 191 192
	}
	close(stop)
}