123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422 |
- // 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.
- // Package logs provide a general log interface
- // 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"
- "os"
- "path"
- "runtime"
- "strconv"
- "sync"
- "time"
- )
- // 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() Logger
- // Logger defines the behavior of a log provider.
- type Logger interface {
- Init(config string) error
- WriteMsg(when time.Time, 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
- msgChan chan *logMsg
- signalChan chan string
- wg sync.WaitGroup
- outputs []*nameLogger
- }
- type nameLogger struct {
- Logger
- name string
- }
- type logMsg struct {
- level int
- msg string
- when time.Time
- }
- var logMsgPool *sync.Pool
- // NewLogger returns a new BeeLogger.
- // channelLen means the number of messages in chan(used where asynchronous is true).
- // 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.msgChan = make(chan *logMsg, channelLen)
- bl.signalChan = make(chan string, 1)
- return bl
- }
- // Async set the log to asynchronous and start the goroutine
- func (bl *BeeLogger) Async() *BeeLogger {
- bl.asynchronous = true
- logMsgPool = &sync.Pool{
- New: func() interface{} {
- return &logMsg{}
- },
- }
- bl.wg.Add(1)
- 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()
- for _, l := range bl.outputs {
- if l.name == adapterName {
- return fmt.Errorf("logs: duplicate adaptername %q (you have set this logger before)", adapterName)
- }
- }
- log, ok := adapters[adapterName]
- if !ok {
- return fmt.Errorf("logs: unknown adaptername %q (forgotten Register?)", adapterName)
- }
- lg := log()
- err := lg.Init(config)
- if err != nil {
- fmt.Fprintln(os.Stderr, "logs.BeeLogger.SetLogger: "+err.Error())
- return err
- }
- bl.outputs = append(bl.outputs, &nameLogger{name: adapterName, Logger: lg})
- return nil
- }
- // DelLogger remove a logger adapter in BeeLogger.
- func (bl *BeeLogger) DelLogger(adapterName string) error {
- bl.lock.Lock()
- defer bl.lock.Unlock()
- outputs := []*nameLogger{}
- for _, lg := range bl.outputs {
- if lg.name == adapterName {
- lg.Destroy()
- } else {
- outputs = append(outputs, lg)
- }
- }
- if len(outputs) == len(bl.outputs) {
- return fmt.Errorf("logs: unknown adaptername %q (forgotten Register?)", adapterName)
- }
- bl.outputs = outputs
- return nil
- }
- func (bl *BeeLogger) writeToLoggers(when time.Time, msg string, level int) {
- for _, l := range bl.outputs {
- err := l.WriteMsg(when, msg, level)
- if err != nil {
- fmt.Fprintf(os.Stderr, "unable to WriteMsg to adapter:%v,error:%v\n", l.name, err)
- }
- }
- }
- func (bl *BeeLogger) writeMsg(logLevel int, msg string) error {
- when := time.Now()
- if bl.enableFuncCallDepth {
- _, file, line, ok := runtime.Caller(bl.loggerFuncCallDepth)
- if !ok {
- file = "???"
- line = 0
- }
- _, filename := path.Split(file)
- msg = "[" + filename + ":" + strconv.FormatInt(int64(line), 10) + "]" + msg
- }
- if bl.asynchronous {
- lm := logMsgPool.Get().(*logMsg)
- lm.level = logLevel
- lm.msg = msg
- lm.when = when
- bl.msgChan <- lm
- } else {
- bl.writeToLoggers(when, msg, logLevel)
- }
- return nil
- }
- // SetLevel 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
- }
- // SetLogFuncCallDepth set log funcCallDepth
- func (bl *BeeLogger) SetLogFuncCallDepth(d int) {
- bl.loggerFuncCallDepth = d
- }
- // GetLogFuncCallDepth return log funcCallDepth for wrapper
- func (bl *BeeLogger) GetLogFuncCallDepth() int {
- return bl.loggerFuncCallDepth
- }
- // EnableFuncCallDepth 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() {
- gameOver := false
- for {
- select {
- case bm := <-bl.msgChan:
- bl.writeToLoggers(bm.when, bm.msg, bm.level)
- logMsgPool.Put(bm)
- case sg := <-bl.signalChan:
- // Now should only send "flush" or "close" to bl.signalChan
- bl.flush()
- if sg == "close" {
- for _, l := range bl.outputs {
- l.Destroy()
- }
- bl.outputs = nil
- gameOver = true
- }
- bl.wg.Done()
- }
- if gameOver {
- break
- }
- }
- }
- // Emergency Log EMERGENCY level message.
- func (bl *BeeLogger) Emergency(format string, v ...interface{}) {
- if LevelEmergency > bl.level {
- return
- }
- msg := fmt.Sprintf("[M] "+format, v...)
- bl.writeMsg(LevelEmergency, msg)
- }
- // Alert Log ALERT level message.
- func (bl *BeeLogger) Alert(format string, v ...interface{}) {
- if LevelAlert > bl.level {
- return
- }
- msg := fmt.Sprintf("[A] "+format, v...)
- bl.writeMsg(LevelAlert, msg)
- }
- // Critical Log CRITICAL level message.
- func (bl *BeeLogger) Critical(format string, v ...interface{}) {
- if LevelCritical > bl.level {
- return
- }
- msg := fmt.Sprintf("[C] "+format, v...)
- bl.writeMsg(LevelCritical, msg)
- }
- // Error Log ERROR level message.
- func (bl *BeeLogger) Error(format string, v ...interface{}) {
- if LevelError > bl.level {
- return
- }
- msg := fmt.Sprintf("[E] "+format, v...)
- bl.writeMsg(LevelError, msg)
- }
- // Warning Log WARNING level message.
- func (bl *BeeLogger) Warning(format string, v ...interface{}) {
- if LevelWarning > bl.level {
- return
- }
- msg := fmt.Sprintf("[W] "+format, v...)
- bl.writeMsg(LevelWarning, msg)
- }
- // Notice Log NOTICE level message.
- func (bl *BeeLogger) Notice(format string, v ...interface{}) {
- if LevelNotice > bl.level {
- return
- }
- msg := fmt.Sprintf("[N] "+format, v...)
- bl.writeMsg(LevelNotice, msg)
- }
- // Informational Log INFORMATIONAL level message.
- func (bl *BeeLogger) Informational(format string, v ...interface{}) {
- if LevelInformational > bl.level {
- return
- }
- msg := fmt.Sprintf("[I] "+format, v...)
- bl.writeMsg(LevelInformational, msg)
- }
- // Debug Log DEBUG level message.
- func (bl *BeeLogger) Debug(format string, v ...interface{}) {
- if LevelDebug > bl.level {
- return
- }
- msg := fmt.Sprintf("[D] "+format, v...)
- bl.writeMsg(LevelDebug, msg)
- }
- // Warn 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.writeMsg(LevelWarning, msg)
- }
- // Info 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.writeMsg(LevelInformational, msg)
- }
- // Trace 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.writeMsg(LevelDebug, msg)
- }
- // Flush flush all chan data.
- func (bl *BeeLogger) Flush() {
- if bl.asynchronous {
- bl.signalChan <- "flush"
- bl.wg.Wait()
- bl.wg.Add(1)
- return
- }
- bl.flush()
- }
- // Close close logger, flush all chan data and destroy all adapters in BeeLogger.
- func (bl *BeeLogger) Close() {
- if bl.asynchronous {
- bl.signalChan <- "close"
- bl.wg.Wait()
- } else {
- bl.flush()
- for _, l := range bl.outputs {
- l.Destroy()
- }
- bl.outputs = nil
- }
- close(bl.msgChan)
- close(bl.signalChan)
- }
- // Reset close all outputs, and set bl.outputs to nil
- func (bl *BeeLogger) Reset() {
- bl.Flush()
- for _, l := range bl.outputs {
- l.Destroy()
- }
- bl.outputs = nil
- }
- func (bl *BeeLogger) flush() {
- for {
- if len(bl.msgChan) > 0 {
- bm := <-bl.msgChan
- bl.writeToLoggers(bm.when, bm.msg, bm.level)
- logMsgPool.Put(bm)
- continue
- }
- break
- }
- for _, l := range bl.outputs {
- l.Flush()
- }
- }
|