123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657 |
- // Copyright 2012 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"
- "log"
- "os"
- "path"
- "runtime"
- "strconv"
- "strings"
- "sync"
- "time"
- )
- // RFC5424 log message levels.
- const (
- LevelEmergency = iota
- LevelAlert
- LevelCritical
- LevelError
- LevelWarning
- LevelNotice
- LevelInformational
- LevelDebug
- LevelTrace
- )
- // levelLogLogger is defined to implement log.Logger
- // the real log level will be LevelEmergency
- const levelLoggerImpl = -1
- // Name for adapter with beego official support
- const (
- AdapterConsole = "console"
- AdapterFile = "file"
- AdapterMultiFile = "multifile"
- AdapterMail = "smtp"
- AdapterConn = "conn"
- AdapterEs = "es"
- AdapterJianLiao = "jianliao"
- AdapterSlack = "slack"
- AdapterAliLS = "alils"
- )
- // Legacy log level constants to ensure backwards compatibility.
- const (
- LevelInfo = LevelInformational
- LevelWarn = LevelWarning
- )
- type newLoggerFunc 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]newLoggerFunc)
- var levelPrefix = [LevelTrace + 1]string{"[M] ", "[A] ", "[C] ", "[E] ", "[W] ", "[N] ", "[I] ", "[D] ", "[T] "}
- // 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 newLoggerFunc) {
- 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
- init bool
- enableFuncCallDepth bool
- loggerFuncCallDepth int
- asynchronous bool
- msgChanLen int64
- msgChan chan *logMsg
- signalChan chan string
- wg sync.WaitGroup
- outputs []*nameLogger
- }
- const defaultAsyncMsgLen = 1e3
- 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(channelLens ...int64) *BeeLogger {
- bl := new(BeeLogger)
- bl.level = LevelDebug
- bl.loggerFuncCallDepth = 2
- bl.msgChanLen = append(channelLens, 0)[0]
- if bl.msgChanLen <= 0 {
- bl.msgChanLen = defaultAsyncMsgLen
- }
- bl.signalChan = make(chan string, 1)
- bl.setLogger(AdapterConsole)
- return bl
- }
- // Async set the log to asynchronous and start the goroutine
- func (bl *BeeLogger) Async(msgLen ...int64) *BeeLogger {
- bl.lock.Lock()
- defer bl.lock.Unlock()
- if bl.asynchronous {
- return bl
- }
- bl.asynchronous = true
- if len(msgLen) > 0 && msgLen[0] > 0 {
- bl.msgChanLen = msgLen[0]
- }
- bl.msgChan = make(chan *logMsg, bl.msgChanLen)
- 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, configs ...string) error {
- config := append(configs, "{}")[0]
- 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
- }
- // 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, configs ...string) error {
- bl.lock.Lock()
- defer bl.lock.Unlock()
- if !bl.init {
- bl.outputs = []*nameLogger{}
- bl.init = true
- }
- return bl.setLogger(adapterName, configs...)
- }
- // 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) Write(p []byte) (n int, err error) {
- if len(p) == 0 {
- return 0, nil
- }
- // writeMsg will always add a '\n' character
- if p[len(p)-1] == '\n' {
- p = p[0 : len(p)-1]
- }
- // set levelLoggerImpl to ensure all log message will be write out
- err = bl.writeMsg(levelLoggerImpl, string(p))
- if err == nil {
- return len(p), err
- }
- return 0, err
- }
- func (bl *BeeLogger) writeMsg(logLevel int, msg string, v ...interface{}) error {
- if !bl.init {
- bl.lock.Lock()
- bl.setLogger(AdapterConsole)
- bl.lock.Unlock()
- }
- if len(v) > 0 {
- msg = fmt.Sprintf(msg, v...)
- }
- when := time.Now()
- if bl.enableFuncCallDepth {
- _, file, line, ok := runtime.Caller(bl.loggerFuncCallDepth)
- if !ok {
- file = "???"
- line = 0
- } else {
- if strings.Contains(file, "<autogenerated>") {
- _, file, line, ok = runtime.Caller(bl.loggerFuncCallDepth + 1)
- if !ok {
- file = "???"
- line = 0
- }
- }
- }
- _, filename := path.Split(file)
- msg = "[" + filename + ":" + strconv.FormatInt(int64(line), 10) + "] " + msg
- }
- //set level info in front of filename info
- if logLevel == levelLoggerImpl {
- // set to emergency to ensure all log will be print out correctly
- logLevel = LevelEmergency
- } else {
- msg = levelPrefix[logLevel] + 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
- }
- bl.writeMsg(LevelEmergency, format, v...)
- }
- // Alert Log ALERT level message.
- func (bl *BeeLogger) Alert(format string, v ...interface{}) {
- if LevelAlert > bl.level {
- return
- }
- bl.writeMsg(LevelAlert, format, v...)
- }
- // Critical Log CRITICAL level message.
- func (bl *BeeLogger) Critical(format string, v ...interface{}) {
- if LevelCritical > bl.level {
- return
- }
- bl.writeMsg(LevelCritical, format, v...)
- }
- // Error Log ERROR level message.
- func (bl *BeeLogger) Error(format string, v ...interface{}) {
- if LevelError > bl.level {
- return
- }
- bl.writeMsg(LevelError, format, v...)
- }
- // Warning Log WARNING level message.
- func (bl *BeeLogger) Warning(format string, v ...interface{}) {
- if LevelWarn > bl.level {
- return
- }
- bl.writeMsg(LevelWarn, format, v...)
- }
- // Notice Log NOTICE level message.
- func (bl *BeeLogger) Notice(format string, v ...interface{}) {
- if LevelNotice > bl.level {
- return
- }
- bl.writeMsg(LevelNotice, format, v...)
- }
- // Informational Log INFORMATIONAL level message.
- func (bl *BeeLogger) Informational(format string, v ...interface{}) {
- if LevelInfo > bl.level {
- return
- }
- bl.writeMsg(LevelInfo, format, v...)
- }
- // Debug Log DEBUG level message.
- func (bl *BeeLogger) Debug(format string, v ...interface{}) {
- if LevelDebug > bl.level {
- return
- }
- bl.writeMsg(LevelDebug, format, v...)
- }
- // Warn Log WARN level message.
- // compatibility alias for Warning()
- func (bl *BeeLogger) Warn(format string, v ...interface{}) {
- if LevelWarn > bl.level {
- return
- }
- bl.writeMsg(LevelWarn, format, v...)
- }
- // Info Log INFO level message.
- // compatibility alias for Informational()
- func (bl *BeeLogger) Info(format string, v ...interface{}) {
- if LevelInfo > bl.level {
- return
- }
- bl.writeMsg(LevelInfo, format, v...)
- }
- // Trace Log TRACE level message.
- // compatibility alias for Debug()
- func (bl *BeeLogger) Trace(format string, v ...interface{}) {
- if LevelTrace > bl.level {
- return
- }
- bl.writeMsg(LevelTrace, format, v...)
- }
- // 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()
- close(bl.msgChan)
- } else {
- bl.flush()
- for _, l := range bl.outputs {
- l.Destroy()
- }
- bl.outputs = nil
- }
- 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() {
- if bl.asynchronous {
- 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()
- }
- }
- // beeLogger references the used application logger.
- var beeLogger *BeeLogger = NewLogger()
- // GetLogger returns the default BeeLogger
- func GetBeeLogger() *BeeLogger {
- return beeLogger
- }
- var beeLoggerMap = struct {
- sync.RWMutex
- logs map[string]*log.Logger
- }{
- logs: map[string]*log.Logger{},
- }
- // GetLogger returns the default BeeLogger
- func GetLogger(prefixes ...string) *log.Logger {
- prefix := append(prefixes, "")[0]
- if prefix != "" {
- prefix = fmt.Sprintf(`[%s] `, strings.ToUpper(prefix))
- }
- beeLoggerMap.RLock()
- l, ok := beeLoggerMap.logs[prefix]
- if ok {
- beeLoggerMap.RUnlock()
- return l
- }
- beeLoggerMap.RUnlock()
- beeLoggerMap.Lock()
- defer beeLoggerMap.Unlock()
- l, ok = beeLoggerMap.logs[prefix]
- if !ok {
- l = log.New(beeLogger, prefix, 0)
- beeLoggerMap.logs[prefix] = l
- }
- return l
- }
- // Reset will remove all the adapter
- func Reset() {
- beeLogger.Reset()
- }
- func Async(msgLen ...int64) *BeeLogger {
- return beeLogger.Async(msgLen...)
- }
- // SetLevel sets the global log level used by the simple logger.
- func SetLevel(l int) {
- beeLogger.SetLevel(l)
- }
- // EnableFuncCallDepth enable log funcCallDepth
- func EnableFuncCallDepth(b bool) {
- beeLogger.enableFuncCallDepth = b
- }
- // SetLogFuncCall set the CallDepth, default is 4
- func SetLogFuncCall(b bool) {
- beeLogger.EnableFuncCallDepth(b)
- beeLogger.SetLogFuncCallDepth(4)
- }
- // SetLogFuncCallDepth set log funcCallDepth
- func SetLogFuncCallDepth(d int) {
- beeLogger.loggerFuncCallDepth = d
- }
- // SetLogger sets a new logger.
- func SetLogger(adapter string, config ...string) error {
- err := beeLogger.SetLogger(adapter, config...)
- if err != nil {
- return err
- }
- return nil
- }
- // Emergency logs a message at emergency level.
- func Emergency(f interface{}, v ...interface{}) {
- beeLogger.Emergency(formatLog(f, v...))
- }
- // Alert logs a message at alert level.
- func Alert(f interface{}, v ...interface{}) {
- beeLogger.Alert(formatLog(f, v...))
- }
- // Critical logs a message at critical level.
- func Critical(f interface{}, v ...interface{}) {
- beeLogger.Critical(formatLog(f, v...))
- }
- // Error logs a message at error level.
- func Error(f interface{}, v ...interface{}) {
- beeLogger.Error(formatLog(f, v...))
- }
- // Warning logs a message at warning level.
- func Warning(f interface{}, v ...interface{}) {
- beeLogger.Warn(formatLog(f, v...))
- }
- // Warn compatibility alias for Warning()
- func Warn(f interface{}, v ...interface{}) {
- beeLogger.Warn(formatLog(f, v...))
- }
- // Notice logs a message at notice level.
- func Notice(f interface{}, v ...interface{}) {
- beeLogger.Notice(formatLog(f, v...))
- }
- // Informational logs a message at info level.
- func Informational(f interface{}, v ...interface{}) {
- beeLogger.Info(formatLog(f, v...))
- }
- // Info compatibility alias for Warning()
- func Info(f interface{}, v ...interface{}) {
- beeLogger.Info(formatLog(f, v...))
- }
- // Debug logs a message at debug level.
- func Debug(f interface{}, v ...interface{}) {
- beeLogger.Debug(formatLog(f, v...))
- }
- // Trace logs a message at trace level.
- // compatibility alias for Warning()
- func Trace(f interface{}, v ...interface{}) {
- beeLogger.Trace(formatLog(f, v...))
- }
- func formatLog(f interface{}, v ...interface{}) string {
- var msg string
- switch f.(type) {
- case string:
- msg = f.(string)
- if len(v) == 0 {
- return msg
- }
- if strings.Contains(msg, "%") && !strings.Contains(msg, "%%") {
- //format string
- } else {
- //do not contain format char
- msg += strings.Repeat(" %v", len(v))
- }
- default:
- msg = fmt.Sprint(f)
- if len(v) == 0 {
- return msg
- }
- msg += strings.Repeat(" %v", len(v))
- }
- return fmt.Sprintf(msg, v...)
- }
|