log.go 6.3 KB

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