| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277 |
- // Copyright 2014 The Gogs Authors. All rights reserved.
- // Use of this source code is governed by a MIT-style
- // license that can be found in the LICENSE file.
- package log
- import (
- "fmt"
- "os"
- "path/filepath"
- "strings"
- "github.com/go-stack/stack"
- "github.com/grafana/grafana/pkg/util"
- "github.com/inconshreveable/log15"
- isatty "github.com/mattn/go-isatty"
- "gopkg.in/ini.v1"
- )
- var Root log15.Logger
- var loggersToClose []DisposableHandler
- var loggersToReload []ReloadableHandler
- var filters map[string]log15.Lvl
- func init() {
- loggersToClose = make([]DisposableHandler, 0)
- loggersToReload = make([]ReloadableHandler, 0)
- filters = map[string]log15.Lvl{}
- Root = log15.Root()
- Root.SetHandler(log15.DiscardHandler())
- }
- func New(logger string, ctx ...interface{}) Logger {
- params := append([]interface{}{"logger", logger}, ctx...)
- return Root.New(params...)
- }
- func Trace(format string, v ...interface{}) {
- var message string
- if len(v) > 0 {
- message = fmt.Sprintf(format, v...)
- } else {
- message = format
- }
- Root.Debug(message)
- }
- func Debug(format string, v ...interface{}) {
- var message string
- if len(v) > 0 {
- message = fmt.Sprintf(format, v...)
- } else {
- message = format
- }
- Root.Debug(message)
- }
- func Info(format string, v ...interface{}) {
- var message string
- if len(v) > 0 {
- message = fmt.Sprintf(format, v...)
- } else {
- message = format
- }
- Root.Info(message)
- }
- func Warn(format string, v ...interface{}) {
- var message string
- if len(v) > 0 {
- message = fmt.Sprintf(format, v...)
- } else {
- message = format
- }
- Root.Warn(message)
- }
- func Error(skip int, format string, v ...interface{}) {
- Root.Error(fmt.Sprintf(format, v...))
- }
- func Critical(skip int, format string, v ...interface{}) {
- Root.Crit(fmt.Sprintf(format, v...))
- }
- func Fatal(skip int, format string, v ...interface{}) {
- Root.Crit(fmt.Sprintf(format, v...))
- Close()
- os.Exit(1)
- }
- func Close() {
- for _, logger := range loggersToClose {
- logger.Close()
- }
- loggersToClose = make([]DisposableHandler, 0)
- }
- func Reload() {
- for _, logger := range loggersToReload {
- logger.Reload()
- }
- }
- func GetLogLevelFor(name string) Lvl {
- if level, ok := filters[name]; ok {
- switch level {
- case log15.LvlWarn:
- return LvlWarn
- case log15.LvlInfo:
- return LvlInfo
- case log15.LvlError:
- return LvlError
- case log15.LvlCrit:
- return LvlCrit
- default:
- return LvlDebug
- }
- }
- return LvlInfo
- }
- var logLevels = map[string]log15.Lvl{
- "trace": log15.LvlDebug,
- "debug": log15.LvlDebug,
- "info": log15.LvlInfo,
- "warn": log15.LvlWarn,
- "error": log15.LvlError,
- "critical": log15.LvlCrit,
- }
- func getLogLevelFromConfig(key string, defaultName string, cfg *ini.File) (string, log15.Lvl) {
- levelName := cfg.Section(key).Key("level").MustString(defaultName)
- levelName = strings.ToLower(levelName)
- level := getLogLevelFromString(levelName)
- return levelName, level
- }
- func getLogLevelFromString(levelName string) log15.Lvl {
- level, ok := logLevels[levelName]
- if !ok {
- Root.Error("Unknown log level", "level", levelName)
- return log15.LvlError
- }
- return level
- }
- func getFilters(filterStrArray []string) map[string]log15.Lvl {
- filterMap := make(map[string]log15.Lvl)
- for _, filterStr := range filterStrArray {
- parts := strings.Split(filterStr, ":")
- if len(parts) > 1 {
- filterMap[parts[0]] = getLogLevelFromString(parts[1])
- }
- }
- return filterMap
- }
- func getLogFormat(format string) log15.Format {
- switch format {
- case "console":
- if isatty.IsTerminal(os.Stdout.Fd()) {
- return log15.TerminalFormat()
- }
- return log15.LogfmtFormat()
- case "text":
- return log15.LogfmtFormat()
- case "json":
- return log15.JsonFormat()
- default:
- return log15.LogfmtFormat()
- }
- }
- func ReadLoggingConfig(modes []string, logsPath string, cfg *ini.File) {
- Close()
- defaultLevelName, _ := getLogLevelFromConfig("log", "info", cfg)
- defaultFilters := getFilters(util.SplitString(cfg.Section("log").Key("filters").String()))
- handlers := make([]log15.Handler, 0)
- for _, mode := range modes {
- mode = strings.TrimSpace(mode)
- sec, err := cfg.GetSection("log." + mode)
- if err != nil {
- Root.Error("Unknown log mode", "mode", mode)
- }
- // Log level.
- _, level := getLogLevelFromConfig("log."+mode, defaultLevelName, cfg)
- modeFilters := getFilters(util.SplitString(sec.Key("filters").String()))
- format := getLogFormat(sec.Key("format").MustString(""))
- var handler log15.Handler
- // Generate log configuration.
- switch mode {
- case "console":
- handler = log15.StreamHandler(os.Stdout, format)
- case "file":
- fileName := sec.Key("file_name").MustString(filepath.Join(logsPath, "grafana.log"))
- os.MkdirAll(filepath.Dir(fileName), os.ModePerm)
- fileHandler := NewFileWriter()
- fileHandler.Filename = fileName
- fileHandler.Format = format
- fileHandler.Rotate = sec.Key("log_rotate").MustBool(true)
- fileHandler.Maxlines = sec.Key("max_lines").MustInt(1000000)
- fileHandler.Maxsize = 1 << uint(sec.Key("max_size_shift").MustInt(28))
- fileHandler.Daily = sec.Key("daily_rotate").MustBool(true)
- fileHandler.Maxdays = sec.Key("max_days").MustInt64(7)
- fileHandler.Init()
- loggersToClose = append(loggersToClose, fileHandler)
- loggersToReload = append(loggersToReload, fileHandler)
- handler = fileHandler
- case "syslog":
- sysLogHandler := NewSyslog(sec, format)
- loggersToClose = append(loggersToClose, sysLogHandler)
- handler = sysLogHandler
- }
- for key, value := range defaultFilters {
- if _, exist := modeFilters[key]; !exist {
- modeFilters[key] = value
- }
- }
- for key, value := range modeFilters {
- if _, exist := filters[key]; !exist {
- filters[key] = value
- }
- }
- handler = LogFilterHandler(level, modeFilters, handler)
- handlers = append(handlers, handler)
- }
- Root.SetHandler(log15.MultiHandler(handlers...))
- }
- func LogFilterHandler(maxLevel log15.Lvl, filters map[string]log15.Lvl, h log15.Handler) log15.Handler {
- return log15.FilterHandler(func(r *log15.Record) (pass bool) {
- if len(filters) > 0 {
- for i := 0; i < len(r.Ctx); i += 2 {
- key, ok := r.Ctx[i].(string)
- if ok && key == "logger" {
- loggerName, strOk := r.Ctx[i+1].(string)
- if strOk {
- if filterLevel, ok := filters[loggerName]; ok {
- return r.Lvl <= filterLevel
- }
- }
- }
- }
- }
- return r.Lvl <= maxLevel
- }, h)
- }
- func Stack(skip int) string {
- call := stack.Caller(skip)
- s := stack.Trace().TrimBelow(call).TrimRuntime()
- return s.String()
- }
|