2015-01-21 18:04:11 +02:00
|
|
|
package logger
|
|
|
|
|
|
|
|
import (
|
2015-03-21 11:20:47 +02:00
|
|
|
"fmt"
|
2015-01-21 18:04:11 +02:00
|
|
|
"sync"
|
|
|
|
)
|
|
|
|
|
2015-03-21 11:20:47 +02:00
|
|
|
type stdMsg struct {
|
2015-01-21 18:26:54 +02:00
|
|
|
level LogLevel
|
|
|
|
msg string
|
|
|
|
}
|
|
|
|
|
2015-03-21 11:20:47 +02:00
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2015-01-21 18:04:11 +02:00
|
|
|
var (
|
2015-03-21 11:20:47 +02:00
|
|
|
logMessageC = make(chan LogMsg)
|
2015-01-21 18:04:11 +02:00
|
|
|
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
|
2015-03-21 11:20:47 +02:00
|
|
|
systemIn []chan LogMsg
|
2015-01-21 18:04:11 +02:00
|
|
|
systemWG sync.WaitGroup
|
|
|
|
)
|
|
|
|
bootSystem := func(sys LogSystem) {
|
2015-03-21 11:20:47 +02:00
|
|
|
in := make(chan LogMsg, sysBufferSize)
|
2015-01-21 18:04:11 +02:00
|
|
|
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)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-21 11:20:47 +02:00
|
|
|
func sysLoop(sys LogSystem, in <-chan LogMsg, wg *sync.WaitGroup) {
|
2015-01-21 18:04:11 +02:00
|
|
|
for msg := range in {
|
2015-03-21 11:20:47 +02:00
|
|
|
sys.LogPrint(msg)
|
2015-01-21 18:04:11 +02:00
|
|
|
}
|
|
|
|
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
|
|
|
|
}
|