mirror of
https://github.com/astaxie/beego.git
synced 2024-11-16 22:40:54 +00:00
351 lines
8.4 KiB
Go
351 lines
8.4 KiB
Go
// 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"
|
|
"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() Logger
|
|
|
|
// Logger defines the behavior of a log provider.
|
|
type Logger 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]Logger
|
|
}
|
|
|
|
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]Logger)
|
|
return bl
|
|
}
|
|
|
|
// Async set the log to asynchronous and start the goroutine
|
|
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
|
|
}
|
|
|
|
// DelLogger 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
|
|
}
|
|
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
|
|
}
|
|
|
|
// 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() {
|
|
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)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// 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.writerMsg(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.writerMsg(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.writerMsg(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.writerMsg(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.writerMsg(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.writerMsg(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.writerMsg(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.writerMsg(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.writerMsg(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.writerMsg(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.writerMsg(LevelDebug, msg)
|
|
}
|
|
|
|
// Flush flush all chan data.
|
|
func (bl *BeeLogger) Flush() {
|
|
for _, l := range bl.outputs {
|
|
l.Flush()
|
|
}
|
|
}
|
|
|
|
// Close 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()
|
|
}
|
|
}
|