logger.go 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352
  1. package logrus
  2. import (
  3. "context"
  4. "io"
  5. "os"
  6. "sync"
  7. "sync/atomic"
  8. "time"
  9. )
  10. type Logger struct {
  11. // The logs are `io.Copy`'d to this in a mutex. It's common to set this to a
  12. // file, or leave it default which is `os.Stderr`. You can also set this to
  13. // something more adventurous, such as logging to Kafka.
  14. Out io.Writer
  15. // Hooks for the logger instance. These allow firing events based on logging
  16. // levels and log entries. For example, to send errors to an error tracking
  17. // service, log to StatsD or dump the core on fatal errors.
  18. Hooks LevelHooks
  19. // All log entries pass through the formatter before logged to Out. The
  20. // included formatters are `TextFormatter` and `JSONFormatter` for which
  21. // TextFormatter is the default. In development (when a TTY is attached) it
  22. // logs with colors, but to a file it wouldn't. You can easily implement your
  23. // own that implements the `Formatter` interface, see the `README` or included
  24. // formatters for examples.
  25. Formatter Formatter
  26. // Flag for whether to log caller info (off by default)
  27. ReportCaller bool
  28. // The logging level the logger should log at. This is typically (and defaults
  29. // to) `logrus.Info`, which allows Info(), Warn(), Error() and Fatal() to be
  30. // logged.
  31. Level Level
  32. // Used to sync writing to the log. Locking is enabled by Default
  33. mu MutexWrap
  34. // Reusable empty entry
  35. entryPool sync.Pool
  36. // Function to exit the application, defaults to `os.Exit()`
  37. ExitFunc exitFunc
  38. }
  39. type exitFunc func(int)
  40. type MutexWrap struct {
  41. lock sync.Mutex
  42. disabled bool
  43. }
  44. func (mw *MutexWrap) Lock() {
  45. if !mw.disabled {
  46. mw.lock.Lock()
  47. }
  48. }
  49. func (mw *MutexWrap) Unlock() {
  50. if !mw.disabled {
  51. mw.lock.Unlock()
  52. }
  53. }
  54. func (mw *MutexWrap) Disable() {
  55. mw.disabled = true
  56. }
  57. // Creates a new logger. Configuration should be set by changing `Formatter`,
  58. // `Out` and `Hooks` directly on the default logger instance. You can also just
  59. // instantiate your own:
  60. //
  61. // var log = &logrus.Logger{
  62. // Out: os.Stderr,
  63. // Formatter: new(logrus.JSONFormatter),
  64. // Hooks: make(logrus.LevelHooks),
  65. // Level: logrus.DebugLevel,
  66. // }
  67. //
  68. // It's recommended to make this a global instance called `log`.
  69. func New() *Logger {
  70. return &Logger{
  71. Out: os.Stderr,
  72. Formatter: new(TextFormatter),
  73. Hooks: make(LevelHooks),
  74. Level: InfoLevel,
  75. ExitFunc: os.Exit,
  76. ReportCaller: false,
  77. }
  78. }
  79. func (logger *Logger) newEntry() *Entry {
  80. entry, ok := logger.entryPool.Get().(*Entry)
  81. if ok {
  82. return entry
  83. }
  84. return NewEntry(logger)
  85. }
  86. func (logger *Logger) releaseEntry(entry *Entry) {
  87. entry.Data = map[string]interface{}{}
  88. logger.entryPool.Put(entry)
  89. }
  90. // WithField allocates a new entry and adds a field to it.
  91. // Debug, Print, Info, Warn, Error, Fatal or Panic must be then applied to
  92. // this new returned entry.
  93. // If you want multiple fields, use `WithFields`.
  94. func (logger *Logger) WithField(key string, value interface{}) *Entry {
  95. entry := logger.newEntry()
  96. defer logger.releaseEntry(entry)
  97. return entry.WithField(key, value)
  98. }
  99. // Adds a struct of fields to the log entry. All it does is call `WithField` for
  100. // each `Field`.
  101. func (logger *Logger) WithFields(fields Fields) *Entry {
  102. entry := logger.newEntry()
  103. defer logger.releaseEntry(entry)
  104. return entry.WithFields(fields)
  105. }
  106. // Add an error as single field to the log entry. All it does is call
  107. // `WithError` for the given `error`.
  108. func (logger *Logger) WithError(err error) *Entry {
  109. entry := logger.newEntry()
  110. defer logger.releaseEntry(entry)
  111. return entry.WithError(err)
  112. }
  113. // Add a context to the log entry.
  114. func (logger *Logger) WithContext(ctx context.Context) *Entry {
  115. entry := logger.newEntry()
  116. defer logger.releaseEntry(entry)
  117. return entry.WithContext(ctx)
  118. }
  119. // Overrides the time of the log entry.
  120. func (logger *Logger) WithTime(t time.Time) *Entry {
  121. entry := logger.newEntry()
  122. defer logger.releaseEntry(entry)
  123. return entry.WithTime(t)
  124. }
  125. func (logger *Logger) Logf(level Level, format string, args ...interface{}) {
  126. if logger.IsLevelEnabled(level) {
  127. entry := logger.newEntry()
  128. entry.Logf(level, format, args...)
  129. logger.releaseEntry(entry)
  130. }
  131. }
  132. func (logger *Logger) Tracef(format string, args ...interface{}) {
  133. logger.Logf(TraceLevel, format, args...)
  134. }
  135. func (logger *Logger) Debugf(format string, args ...interface{}) {
  136. logger.Logf(DebugLevel, format, args...)
  137. }
  138. func (logger *Logger) Infof(format string, args ...interface{}) {
  139. logger.Logf(InfoLevel, format, args...)
  140. }
  141. func (logger *Logger) Printf(format string, args ...interface{}) {
  142. entry := logger.newEntry()
  143. entry.Printf(format, args...)
  144. logger.releaseEntry(entry)
  145. }
  146. func (logger *Logger) Warnf(format string, args ...interface{}) {
  147. logger.Logf(WarnLevel, format, args...)
  148. }
  149. func (logger *Logger) Warningf(format string, args ...interface{}) {
  150. logger.Warnf(format, args...)
  151. }
  152. func (logger *Logger) Errorf(format string, args ...interface{}) {
  153. logger.Logf(ErrorLevel, format, args...)
  154. }
  155. func (logger *Logger) Fatalf(format string, args ...interface{}) {
  156. logger.Logf(FatalLevel, format, args...)
  157. logger.Exit(1)
  158. }
  159. func (logger *Logger) Panicf(format string, args ...interface{}) {
  160. logger.Logf(PanicLevel, format, args...)
  161. }
  162. func (logger *Logger) Log(level Level, args ...interface{}) {
  163. if logger.IsLevelEnabled(level) {
  164. entry := logger.newEntry()
  165. entry.Log(level, args...)
  166. logger.releaseEntry(entry)
  167. }
  168. }
  169. func (logger *Logger) Trace(args ...interface{}) {
  170. logger.Log(TraceLevel, args...)
  171. }
  172. func (logger *Logger) Debug(args ...interface{}) {
  173. logger.Log(DebugLevel, args...)
  174. }
  175. func (logger *Logger) Info(args ...interface{}) {
  176. logger.Log(InfoLevel, args...)
  177. }
  178. func (logger *Logger) Print(args ...interface{}) {
  179. entry := logger.newEntry()
  180. entry.Print(args...)
  181. logger.releaseEntry(entry)
  182. }
  183. func (logger *Logger) Warn(args ...interface{}) {
  184. logger.Log(WarnLevel, args...)
  185. }
  186. func (logger *Logger) Warning(args ...interface{}) {
  187. logger.Warn(args...)
  188. }
  189. func (logger *Logger) Error(args ...interface{}) {
  190. logger.Log(ErrorLevel, args...)
  191. }
  192. func (logger *Logger) Fatal(args ...interface{}) {
  193. logger.Log(FatalLevel, args...)
  194. logger.Exit(1)
  195. }
  196. func (logger *Logger) Panic(args ...interface{}) {
  197. logger.Log(PanicLevel, args...)
  198. }
  199. func (logger *Logger) Logln(level Level, args ...interface{}) {
  200. if logger.IsLevelEnabled(level) {
  201. entry := logger.newEntry()
  202. entry.Logln(level, args...)
  203. logger.releaseEntry(entry)
  204. }
  205. }
  206. func (logger *Logger) Traceln(args ...interface{}) {
  207. logger.Logln(TraceLevel, args...)
  208. }
  209. func (logger *Logger) Debugln(args ...interface{}) {
  210. logger.Logln(DebugLevel, args...)
  211. }
  212. func (logger *Logger) Infoln(args ...interface{}) {
  213. logger.Logln(InfoLevel, args...)
  214. }
  215. func (logger *Logger) Println(args ...interface{}) {
  216. entry := logger.newEntry()
  217. entry.Println(args...)
  218. logger.releaseEntry(entry)
  219. }
  220. func (logger *Logger) Warnln(args ...interface{}) {
  221. logger.Logln(WarnLevel, args...)
  222. }
  223. func (logger *Logger) Warningln(args ...interface{}) {
  224. logger.Warnln(args...)
  225. }
  226. func (logger *Logger) Errorln(args ...interface{}) {
  227. logger.Logln(ErrorLevel, args...)
  228. }
  229. func (logger *Logger) Fatalln(args ...interface{}) {
  230. logger.Logln(FatalLevel, args...)
  231. logger.Exit(1)
  232. }
  233. func (logger *Logger) Panicln(args ...interface{}) {
  234. logger.Logln(PanicLevel, args...)
  235. }
  236. func (logger *Logger) Exit(code int) {
  237. runHandlers()
  238. if logger.ExitFunc == nil {
  239. logger.ExitFunc = os.Exit
  240. }
  241. logger.ExitFunc(code)
  242. }
  243. //When file is opened with appending mode, it's safe to
  244. //write concurrently to a file (within 4k message on Linux).
  245. //In these cases user can choose to disable the lock.
  246. func (logger *Logger) SetNoLock() {
  247. logger.mu.Disable()
  248. }
  249. func (logger *Logger) level() Level {
  250. return Level(atomic.LoadUint32((*uint32)(&logger.Level)))
  251. }
  252. // SetLevel sets the logger level.
  253. func (logger *Logger) SetLevel(level Level) {
  254. atomic.StoreUint32((*uint32)(&logger.Level), uint32(level))
  255. }
  256. // GetLevel returns the logger level.
  257. func (logger *Logger) GetLevel() Level {
  258. return logger.level()
  259. }
  260. // AddHook adds a hook to the logger hooks.
  261. func (logger *Logger) AddHook(hook Hook) {
  262. logger.mu.Lock()
  263. defer logger.mu.Unlock()
  264. logger.Hooks.Add(hook)
  265. }
  266. // IsLevelEnabled checks if the log level of the logger is greater than the level param
  267. func (logger *Logger) IsLevelEnabled(level Level) bool {
  268. return logger.level() >= level
  269. }
  270. // SetFormatter sets the logger formatter.
  271. func (logger *Logger) SetFormatter(formatter Formatter) {
  272. logger.mu.Lock()
  273. defer logger.mu.Unlock()
  274. logger.Formatter = formatter
  275. }
  276. // SetOutput sets the logger output.
  277. func (logger *Logger) SetOutput(output io.Writer) {
  278. logger.mu.Lock()
  279. defer logger.mu.Unlock()
  280. logger.Out = output
  281. }
  282. func (logger *Logger) SetReportCaller(reportCaller bool) {
  283. logger.mu.Lock()
  284. defer logger.mu.Unlock()
  285. logger.ReportCaller = reportCaller
  286. }
  287. // ReplaceHooks replaces the logger hooks and returns the old ones
  288. func (logger *Logger) ReplaceHooks(hooks LevelHooks) LevelHooks {
  289. logger.mu.Lock()
  290. oldHooks := logger.Hooks
  291. logger.Hooks = hooks
  292. logger.mu.Unlock()
  293. return oldHooks
  294. }