log.go 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270
  1. // Copyright 2014 The Gogs Authors. All rights reserved.
  2. // Use of this source code is governed by a MIT-style
  3. // license that can be found in the LICENSE file.
  4. package log
  5. import (
  6. "fmt"
  7. "os"
  8. "path/filepath"
  9. "strings"
  10. "github.com/go-stack/stack"
  11. "github.com/grafana/grafana/pkg/util"
  12. "github.com/inconshreveable/log15"
  13. isatty "github.com/mattn/go-isatty"
  14. "gopkg.in/ini.v1"
  15. )
  16. var Root log15.Logger
  17. var loggersToClose []DisposableHandler
  18. var loggersToReload []ReloadableHandler
  19. var filters map[string]log15.Lvl
  20. func init() {
  21. loggersToClose = make([]DisposableHandler, 0)
  22. loggersToReload = make([]ReloadableHandler, 0)
  23. Root = log15.Root()
  24. Root.SetHandler(log15.DiscardHandler())
  25. }
  26. func New(logger string, ctx ...interface{}) Logger {
  27. params := append([]interface{}{"logger", logger}, ctx...)
  28. return Root.New(params...)
  29. }
  30. func Trace(format string, v ...interface{}) {
  31. var message string
  32. if len(v) > 0 {
  33. message = fmt.Sprintf(format, v...)
  34. } else {
  35. message = format
  36. }
  37. Root.Debug(message)
  38. }
  39. func Debug(format string, v ...interface{}) {
  40. var message string
  41. if len(v) > 0 {
  42. message = fmt.Sprintf(format, v...)
  43. } else {
  44. message = format
  45. }
  46. Root.Debug(message)
  47. }
  48. func Info(format string, v ...interface{}) {
  49. var message string
  50. if len(v) > 0 {
  51. message = fmt.Sprintf(format, v...)
  52. } else {
  53. message = format
  54. }
  55. Root.Info(message)
  56. }
  57. func Warn(format string, v ...interface{}) {
  58. var message string
  59. if len(v) > 0 {
  60. message = fmt.Sprintf(format, v...)
  61. } else {
  62. message = format
  63. }
  64. Root.Warn(message)
  65. }
  66. func Error(skip int, format string, v ...interface{}) {
  67. Root.Error(fmt.Sprintf(format, v...))
  68. }
  69. func Critical(skip int, format string, v ...interface{}) {
  70. Root.Crit(fmt.Sprintf(format, v...))
  71. }
  72. func Fatal(skip int, format string, v ...interface{}) {
  73. Root.Crit(fmt.Sprintf(format, v...))
  74. Close()
  75. os.Exit(1)
  76. }
  77. func Close() {
  78. for _, logger := range loggersToClose {
  79. logger.Close()
  80. }
  81. loggersToClose = make([]DisposableHandler, 0)
  82. }
  83. func Reload() {
  84. for _, logger := range loggersToReload {
  85. logger.Reload()
  86. }
  87. }
  88. func GetLogLevelFor(name string) Lvl {
  89. if level, ok := filters[name]; ok {
  90. switch level {
  91. case log15.LvlWarn:
  92. return LvlWarn
  93. case log15.LvlInfo:
  94. return LvlInfo
  95. case log15.LvlError:
  96. return LvlError
  97. case log15.LvlCrit:
  98. return LvlCrit
  99. default:
  100. return LvlDebug
  101. }
  102. }
  103. return LvlInfo
  104. }
  105. var logLevels = map[string]log15.Lvl{
  106. "trace": log15.LvlDebug,
  107. "debug": log15.LvlDebug,
  108. "info": log15.LvlInfo,
  109. "warn": log15.LvlWarn,
  110. "error": log15.LvlError,
  111. "critical": log15.LvlCrit,
  112. }
  113. func getLogLevelFromConfig(key string, defaultName string, cfg *ini.File) (string, log15.Lvl) {
  114. levelName := cfg.Section(key).Key("level").MustString(defaultName)
  115. levelName = strings.ToLower(levelName)
  116. level := getLogLevelFromString(levelName)
  117. return levelName, level
  118. }
  119. func getLogLevelFromString(levelName string) log15.Lvl {
  120. level, ok := logLevels[levelName]
  121. if !ok {
  122. Root.Error("Unknown log level", "level", levelName)
  123. return log15.LvlError
  124. }
  125. return level
  126. }
  127. func getFilters(filterStrArray []string) map[string]log15.Lvl {
  128. filterMap := make(map[string]log15.Lvl)
  129. for _, filterStr := range filterStrArray {
  130. parts := strings.Split(filterStr, ":")
  131. if len(parts) > 1 {
  132. filterMap[parts[0]] = getLogLevelFromString(parts[1])
  133. }
  134. }
  135. return filterMap
  136. }
  137. func getLogFormat(format string) log15.Format {
  138. switch format {
  139. case "console":
  140. if isatty.IsTerminal(os.Stdout.Fd()) {
  141. return log15.TerminalFormat()
  142. }
  143. return log15.LogfmtFormat()
  144. case "text":
  145. return log15.LogfmtFormat()
  146. case "json":
  147. return log15.JsonFormat()
  148. default:
  149. return log15.LogfmtFormat()
  150. }
  151. }
  152. func ReadLoggingConfig(modes []string, logsPath string, cfg *ini.File) {
  153. Close()
  154. defaultLevelName, _ := getLogLevelFromConfig("log", "info", cfg)
  155. defaultFilters := getFilters(util.SplitString(cfg.Section("log").Key("filters").String()))
  156. handlers := make([]log15.Handler, 0)
  157. for _, mode := range modes {
  158. mode = strings.TrimSpace(mode)
  159. sec, err := cfg.GetSection("log." + mode)
  160. if err != nil {
  161. Root.Error("Unknown log mode", "mode", mode)
  162. }
  163. // Log level.
  164. _, level := getLogLevelFromConfig("log."+mode, defaultLevelName, cfg)
  165. filters := getFilters(util.SplitString(sec.Key("filters").String()))
  166. format := getLogFormat(sec.Key("format").MustString(""))
  167. var handler log15.Handler
  168. // Generate log configuration.
  169. switch mode {
  170. case "console":
  171. handler = log15.StreamHandler(os.Stdout, format)
  172. case "file":
  173. fileName := sec.Key("file_name").MustString(filepath.Join(logsPath, "grafana.log"))
  174. os.MkdirAll(filepath.Dir(fileName), os.ModePerm)
  175. fileHandler := NewFileWriter()
  176. fileHandler.Filename = fileName
  177. fileHandler.Format = format
  178. fileHandler.Rotate = sec.Key("log_rotate").MustBool(true)
  179. fileHandler.Maxlines = sec.Key("max_lines").MustInt(1000000)
  180. fileHandler.Maxsize = 1 << uint(sec.Key("max_size_shift").MustInt(28))
  181. fileHandler.Daily = sec.Key("daily_rotate").MustBool(true)
  182. fileHandler.Maxdays = sec.Key("max_days").MustInt64(7)
  183. fileHandler.Init()
  184. loggersToClose = append(loggersToClose, fileHandler)
  185. loggersToReload = append(loggersToReload, fileHandler)
  186. handler = fileHandler
  187. case "syslog":
  188. sysLogHandler := NewSyslog(sec, format)
  189. loggersToClose = append(loggersToClose, sysLogHandler)
  190. handler = sysLogHandler
  191. }
  192. for key, value := range defaultFilters {
  193. if _, exist := filters[key]; !exist {
  194. filters[key] = value
  195. }
  196. }
  197. handler = LogFilterHandler(level, filters, handler)
  198. handlers = append(handlers, handler)
  199. }
  200. Root.SetHandler(log15.MultiHandler(handlers...))
  201. }
  202. func LogFilterHandler(maxLevel log15.Lvl, filters map[string]log15.Lvl, h log15.Handler) log15.Handler {
  203. return log15.FilterHandler(func(r *log15.Record) (pass bool) {
  204. if len(filters) > 0 {
  205. for i := 0; i < len(r.Ctx); i += 2 {
  206. key, ok := r.Ctx[i].(string)
  207. if ok && key == "logger" {
  208. loggerName, strOk := r.Ctx[i+1].(string)
  209. if strOk {
  210. if filterLevel, ok := filters[loggerName]; ok {
  211. return r.Lvl <= filterLevel
  212. }
  213. }
  214. }
  215. }
  216. }
  217. return r.Lvl <= maxLevel
  218. }, h)
  219. }
  220. func Stack(skip int) string {
  221. call := stack.Caller(skip)
  222. s := stack.Trace().TrimBelow(call).TrimRuntime()
  223. return s.String()
  224. }