// Copyright 2015 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 logger import ( "fmt" "sync" ) type stdMsg struct { level LogLevel msg string } type jsonMsg []byte func (m jsonMsg) Level() LogLevel { return 0 } func (m jsonMsg) String() string { return string(m) } type LogMsg interface { Level() LogLevel fmt.Stringer } func (m stdMsg) Level() LogLevel { return m.level } func (m stdMsg) String() string { return m.msg } var ( logMessageC = make(chan LogMsg) addSystemC = make(chan LogSystem) flushC = make(chan chan struct{}) resetC = make(chan chan struct{}) ) func init() { go dispatchLoop() } // each system can buffer this many messages before // blocking incoming log messages. const sysBufferSize = 500 func dispatchLoop() { var ( systems []LogSystem systemIn []chan LogMsg systemWG sync.WaitGroup ) bootSystem := func(sys LogSystem) { in := make(chan LogMsg, sysBufferSize) systemIn = append(systemIn, in) systemWG.Add(1) go sysLoop(sys, in, &systemWG) } for { select { case msg := <-logMessageC: for _, c := range systemIn { c <- msg } case sys := <-addSystemC: systems = append(systems, sys) bootSystem(sys) case waiter := <-resetC: // reset means terminate all systems for _, c := range systemIn { close(c) } systems = nil systemIn = nil systemWG.Wait() close(waiter) case waiter := <-flushC: // flush means reboot all systems for _, c := range systemIn { close(c) } systemIn = nil systemWG.Wait() for _, sys := range systems { bootSystem(sys) } close(waiter) } } } func sysLoop(sys LogSystem, in <-chan LogMsg, wg *sync.WaitGroup) { for msg := range in { sys.LogPrint(msg) } wg.Done() } // Reset removes all active log systems. // It blocks until all current messages have been delivered. func Reset() { waiter := make(chan struct{}) resetC <- waiter <-waiter } // Flush waits until all current log messages have been dispatched to // the active log systems. func Flush() { waiter := make(chan struct{}) flushC <- waiter <-waiter } // AddLogSystem starts printing messages to the given LogSystem. func AddLogSystem(sys LogSystem) { addSystemC <- sys }