tendermint/log/filter.go

159 lines
4.8 KiB
Go
Raw Normal View History

2017-05-05 04:06:56 -07:00
package log
2017-05-12 13:47:02 -07:00
import "fmt"
2017-09-10 18:45:20 -07:00
type level byte
const (
levelDebug level = 1 << iota
levelInfo
levelError
)
type filter struct {
next Logger
allowed level // XOR'd levels for default case
allowedKeyvals map[keyval]level // When key-value match, use this level
}
type keyval struct {
key interface{}
value interface{}
}
2017-05-05 04:06:56 -07:00
// NewFilter wraps next and implements filtering. See the commentary on the
// Option functions for a detailed description of how to configure levels. If
// no options are provided, all leveled log events created with Debug, Info or
// Error helper methods are squelched.
func NewFilter(next Logger, options ...Option) Logger {
l := &filter{
2017-05-15 11:08:02 -07:00
next: next,
allowedKeyvals: make(map[keyval]level),
2017-05-05 04:06:56 -07:00
}
for _, option := range options {
option(l)
}
return l
}
func (l *filter) Info(msg string, keyvals ...interface{}) {
2017-05-05 04:06:56 -07:00
levelAllowed := l.allowed&levelInfo != 0
if !levelAllowed {
return
2017-05-05 04:06:56 -07:00
}
l.next.Info(msg, keyvals...)
2017-05-05 04:06:56 -07:00
}
func (l *filter) Debug(msg string, keyvals ...interface{}) {
2017-05-05 04:06:56 -07:00
levelAllowed := l.allowed&levelDebug != 0
if !levelAllowed {
return
2017-05-05 04:06:56 -07:00
}
l.next.Debug(msg, keyvals...)
2017-05-05 04:06:56 -07:00
}
func (l *filter) Error(msg string, keyvals ...interface{}) {
2017-05-05 04:06:56 -07:00
levelAllowed := l.allowed&levelError != 0
if !levelAllowed {
return
2017-05-05 04:06:56 -07:00
}
l.next.Error(msg, keyvals...)
2017-05-05 04:06:56 -07:00
}
2017-05-15 11:08:02 -07:00
// With implements Logger by constructing a new filter with a keyvals appended
// to the logger.
//
// If custom level was set for a keyval pair using one of the
// Allow*With methods, it is used as the logger's level.
//
// Examples:
// logger = log.NewFilter(logger, log.AllowError(), log.AllowInfoWith("module", "crypto"))
// logger.With("module", "crypto").Info("Hello") # produces "I... Hello module=crypto"
//
// logger = log.NewFilter(logger, log.AllowError(), log.AllowInfoWith("module", "crypto"), log.AllowNoneWith("user", "Sam"))
// logger.With("module", "crypto", "user", "Sam").Info("Hello") # returns nil
//
// logger = log.NewFilter(logger, log.AllowError(), log.AllowInfoWith("module", "crypto"), log.AllowNoneWith("user", "Sam"))
// logger.With("user", "Sam").With("module", "crypto").Info("Hello") # produces "I... Hello module=crypto user=Sam"
2017-05-05 04:06:56 -07:00
func (l *filter) With(keyvals ...interface{}) Logger {
2017-05-15 11:08:02 -07:00
for i := len(keyvals) - 2; i >= 0; i -= 2 {
for kv, allowed := range l.allowedKeyvals {
if keyvals[i] == kv.key && keyvals[i+1] == kv.value {
return &filter{next: l.next.With(keyvals...), allowed: allowed, allowedKeyvals: l.allowedKeyvals}
2017-05-15 11:08:02 -07:00
}
}
}
return &filter{next: l.next.With(keyvals...), allowed: l.allowed, allowedKeyvals: l.allowedKeyvals}
2017-05-05 04:06:56 -07:00
}
2017-09-10 18:45:20 -07:00
//--------------------------------------------------------------------------------
2017-05-05 04:06:56 -07:00
// Option sets a parameter for the filter.
type Option func(*filter)
2017-09-10 18:45:20 -07:00
// AllowLevel returns an option for the given level or error if no option exist
// for such level.
func AllowLevel(lvl string) (Option, error) {
switch lvl {
case "debug":
return AllowDebug(), nil
case "info":
return AllowInfo(), nil
case "error":
return AllowError(), nil
case "none":
return AllowNone(), nil
default:
return nil, fmt.Errorf("Expected either \"info\", \"debug\", \"error\" or \"none\" level, given %s", lvl)
}
}
2017-05-05 04:06:56 -07:00
// AllowAll is an alias for AllowDebug.
func AllowAll() Option {
return AllowDebug()
}
2017-05-05 09:33:15 -07:00
// AllowDebug allows error, info and debug level log events to pass.
2017-05-05 04:06:56 -07:00
func AllowDebug() Option {
return allowed(levelError | levelInfo | levelDebug)
}
2017-05-05 09:33:15 -07:00
// AllowInfo allows error and info level log events to pass.
2017-05-05 04:06:56 -07:00
func AllowInfo() Option {
return allowed(levelError | levelInfo)
}
// AllowError allows only error level log events to pass.
func AllowError() Option {
return allowed(levelError)
}
// AllowNone allows no leveled log events to pass.
func AllowNone() Option {
return allowed(0)
}
func allowed(allowed level) Option {
return func(l *filter) { l.allowed = allowed }
}
2017-05-15 11:08:02 -07:00
// AllowDebugWith allows error, info and debug level log events to pass for a specific key value pair.
func AllowDebugWith(key interface{}, value interface{}) Option {
return func(l *filter) { l.allowedKeyvals[keyval{key, value}] = levelError | levelInfo | levelDebug }
}
// AllowInfoWith allows error and info level log events to pass for a specific key value pair.
func AllowInfoWith(key interface{}, value interface{}) Option {
return func(l *filter) { l.allowedKeyvals[keyval{key, value}] = levelError | levelInfo }
}
// AllowErrorWith allows only error level log events to pass for a specific key value pair.
func AllowErrorWith(key interface{}, value interface{}) Option {
return func(l *filter) { l.allowedKeyvals[keyval{key, value}] = levelError }
}
// AllowNoneWith allows no leveled log events to pass for a specific key value pair.
func AllowNoneWith(key interface{}, value interface{}) Option {
return func(l *filter) { l.allowedKeyvals[keyval{key, value}] = 0 }
}