| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350 |
- // Copyright 2014 beego Author. All Rights Reserved.
- //
- // Licensed under the Apache License, Version 2.0 (the "License");
- // you may not use this file except in compliance with the License.
- // You may obtain a copy of the License at
- //
- // http://www.apache.org/licenses/LICENSE-2.0
- //
- // Unless required by applicable law or agreed to in writing, software
- // distributed under the License is distributed on an "AS IS" BASIS,
- // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- // See the License for the specific language governing permissions and
- // limitations under the License.
- // Usage:
- //
- // import "github.com/astaxie/beego/logs"
- //
- // log := NewLogger(10000)
- // log.SetLogger("console", "")
- //
- // > the first params stand for how many channel
- //
- // Use it like this:
- //
- // log.Trace("trace")
- // log.Info("info")
- // log.Warn("warning")
- // log.Debug("debug")
- // log.Critical("critical")
- //
- // more docs http://beego.me/docs/module/logs.md
- package logs
- import (
- "fmt"
- "path"
- "runtime"
- "sync"
- )
- // RFC5424 log message levels.
- const (
- LevelEmergency = iota
- LevelAlert
- LevelCritical
- LevelError
- LevelWarning
- LevelNotice
- LevelInformational
- LevelDebug
- )
- // Legacy loglevel constants to ensure backwards compatibility.
- //
- // Deprecated: will be removed in 1.5.0.
- const (
- LevelInfo = LevelInformational
- LevelTrace = LevelDebug
- LevelWarn = LevelWarning
- )
- type loggerType func() LoggerInterface
- // LoggerInterface defines the behavior of a log provider.
- type LoggerInterface interface {
- Init(config string) error
- WriteMsg(msg string, level int) error
- Destroy()
- Flush()
- }
- var adapters = make(map[string]loggerType)
- // Register makes a log provide available by the provided name.
- // If Register is called twice with the same name or if driver is nil,
- // it panics.
- func Register(name string, log loggerType) {
- if log == nil {
- panic("logs: Register provide is nil")
- }
- if _, dup := adapters[name]; dup {
- panic("logs: Register called twice for provider " + name)
- }
- adapters[name] = log
- }
- // BeeLogger is default logger in beego application.
- // it can contain several providers and log message into all providers.
- type BeeLogger struct {
- lock sync.Mutex
- level int
- enableFuncCallDepth bool
- loggerFuncCallDepth int
- asynchronous bool
- msg chan *logMsg
- outputs map[string]LoggerInterface
- }
- type logMsg struct {
- level int
- msg string
- }
- // NewLogger returns a new BeeLogger.
- // channellen means the number of messages in chan.
- // if the buffering chan is full, logger adapters write to file or other way.
- func NewLogger(channellen int64) *BeeLogger {
- bl := new(BeeLogger)
- bl.level = LevelDebug
- bl.loggerFuncCallDepth = 2
- bl.msg = make(chan *logMsg, channellen)
- bl.outputs = make(map[string]LoggerInterface)
- return bl
- }
- func (bl *BeeLogger) Async() *BeeLogger {
- bl.asynchronous = true
- go bl.startLogger()
- return bl
- }
- // SetLogger provides a given logger adapter into BeeLogger with config string.
- // config need to be correct JSON as string: {"interval":360}.
- func (bl *BeeLogger) SetLogger(adaptername string, config string) error {
- bl.lock.Lock()
- defer bl.lock.Unlock()
- if log, ok := adapters[adaptername]; ok {
- lg := log()
- err := lg.Init(config)
- bl.outputs[adaptername] = lg
- if err != nil {
- fmt.Println("logs.BeeLogger.SetLogger: " + err.Error())
- return err
- }
- } else {
- return fmt.Errorf("logs: unknown adaptername %q (forgotten Register?)", adaptername)
- }
- return nil
- }
- // remove a logger adapter in BeeLogger.
- func (bl *BeeLogger) DelLogger(adaptername string) error {
- bl.lock.Lock()
- defer bl.lock.Unlock()
- if lg, ok := bl.outputs[adaptername]; ok {
- lg.Destroy()
- delete(bl.outputs, adaptername)
- return nil
- } else {
- return fmt.Errorf("logs: unknown adaptername %q (forgotten Register?)", adaptername)
- }
- }
- func (bl *BeeLogger) writerMsg(loglevel int, msg string) error {
- lm := new(logMsg)
- lm.level = loglevel
- if bl.enableFuncCallDepth {
- _, file, line, ok := runtime.Caller(bl.loggerFuncCallDepth)
- if !ok {
- file = "???"
- line = 0
- }
- _, filename := path.Split(file)
- lm.msg = fmt.Sprintf("[%s:%d] %s", filename, line, msg)
- } else {
- lm.msg = msg
- }
- if bl.asynchronous {
- bl.msg <- lm
- } else {
- for name, l := range bl.outputs {
- err := l.WriteMsg(lm.msg, lm.level)
- if err != nil {
- fmt.Println("unable to WriteMsg to adapter:", name, err)
- return err
- }
- }
- }
- return nil
- }
- // Set log message level.
- //
- // If message level (such as LevelDebug) is higher than logger level (such as LevelWarning),
- // log providers will not even be sent the message.
- func (bl *BeeLogger) SetLevel(l int) {
- bl.level = l
- }
- // set log funcCallDepth
- func (bl *BeeLogger) SetLogFuncCallDepth(d int) {
- bl.loggerFuncCallDepth = d
- }
- // get log funcCallDepth for wrapper
- func (bl *BeeLogger) GetLogFuncCallDepth() int {
- return bl.loggerFuncCallDepth
- }
- // enable log funcCallDepth
- func (bl *BeeLogger) EnableFuncCallDepth(b bool) {
- bl.enableFuncCallDepth = b
- }
- // start logger chan reading.
- // when chan is not empty, write logs.
- func (bl *BeeLogger) startLogger() {
- for {
- select {
- case bm := <-bl.msg:
- for _, l := range bl.outputs {
- err := l.WriteMsg(bm.msg, bm.level)
- if err != nil {
- fmt.Println("ERROR, unable to WriteMsg:", err)
- }
- }
- }
- }
- }
- // Log EMERGENCY level message.
- func (bl *BeeLogger) Emergency(format string, v ...interface{}) {
- if LevelEmergency > bl.level {
- return
- }
- msg := fmt.Sprintf("[M] "+format, v...)
- bl.writerMsg(LevelEmergency, msg)
- }
- // Log ALERT level message.
- func (bl *BeeLogger) Alert(format string, v ...interface{}) {
- if LevelAlert > bl.level {
- return
- }
- msg := fmt.Sprintf("[A] "+format, v...)
- bl.writerMsg(LevelAlert, msg)
- }
- // Log CRITICAL level message.
- func (bl *BeeLogger) Critical(format string, v ...interface{}) {
- if LevelCritical > bl.level {
- return
- }
- msg := fmt.Sprintf("[C] "+format, v...)
- bl.writerMsg(LevelCritical, msg)
- }
- // Log ERROR level message.
- func (bl *BeeLogger) Error(format string, v ...interface{}) {
- if LevelError > bl.level {
- return
- }
- msg := fmt.Sprintf("[E] "+format, v...)
- bl.writerMsg(LevelError, msg)
- }
- // Log WARNING level message.
- func (bl *BeeLogger) Warning(format string, v ...interface{}) {
- if LevelWarning > bl.level {
- return
- }
- msg := fmt.Sprintf("[W] "+format, v...)
- bl.writerMsg(LevelWarning, msg)
- }
- // Log NOTICE level message.
- func (bl *BeeLogger) Notice(format string, v ...interface{}) {
- if LevelNotice > bl.level {
- return
- }
- msg := fmt.Sprintf("[N] "+format, v...)
- bl.writerMsg(LevelNotice, msg)
- }
- // Log INFORMATIONAL level message.
- func (bl *BeeLogger) Informational(format string, v ...interface{}) {
- if LevelInformational > bl.level {
- return
- }
- msg := fmt.Sprintf("[I] "+format, v...)
- bl.writerMsg(LevelInformational, msg)
- }
- // Log DEBUG level message.
- func (bl *BeeLogger) Debug(format string, v ...interface{}) {
- if LevelDebug > bl.level {
- return
- }
- msg := fmt.Sprintf("[D] "+format, v...)
- bl.writerMsg(LevelDebug, msg)
- }
- // Log WARN level message.
- // compatibility alias for Warning()
- func (bl *BeeLogger) Warn(format string, v ...interface{}) {
- if LevelWarning > bl.level {
- return
- }
- msg := fmt.Sprintf("[W] "+format, v...)
- bl.writerMsg(LevelWarning, msg)
- }
- // Log INFO level message.
- // compatibility alias for Informational()
- func (bl *BeeLogger) Info(format string, v ...interface{}) {
- if LevelInformational > bl.level {
- return
- }
- msg := fmt.Sprintf("[I] "+format, v...)
- bl.writerMsg(LevelInformational, msg)
- }
- // Log TRACE level message.
- // compatibility alias for Debug()
- func (bl *BeeLogger) Trace(format string, v ...interface{}) {
- if LevelDebug > bl.level {
- return
- }
- msg := fmt.Sprintf("[D] "+format, v...)
- bl.writerMsg(LevelDebug, msg)
- }
- // flush all chan data.
- func (bl *BeeLogger) Flush() {
- for _, l := range bl.outputs {
- l.Flush()
- }
- }
- // close logger, flush all chan data and destroy all adapters in BeeLogger.
- func (bl *BeeLogger) Close() {
- for {
- if len(bl.msg) > 0 {
- bm := <-bl.msg
- for _, l := range bl.outputs {
- err := l.WriteMsg(bm.msg, bm.level)
- if err != nil {
- fmt.Println("ERROR, unable to WriteMsg (while closing logger):", err)
- }
- }
- continue
- }
- break
- }
- for _, l := range bl.outputs {
- l.Flush()
- l.Destroy()
- }
- }
|