logger.go 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338
  1. // Package logger provides logging capabilities.
  2. // It is a wrapper around zerolog for logging and lumberjack for log rotation.
  3. // Logs are written to the specified log file.
  4. // Logging on the console is provided to print initialization info, errors and warnings.
  5. // The package provides a request logger to log the HTTP requests for REST API too.
  6. // The request logger uses chi.middleware.RequestLogger,
  7. // chi.middleware.LogFormatter and chi.middleware.LogEntry to build a structured
  8. // logger using zerolog
  9. package logger
  10. import (
  11. "errors"
  12. "fmt"
  13. "os"
  14. "path/filepath"
  15. "time"
  16. ftpserverlog "github.com/fclairamb/go-log"
  17. "github.com/rs/zerolog"
  18. lumberjack "gopkg.in/natefinch/lumberjack.v2"
  19. )
  20. const (
  21. dateFormat = "2006-01-02T15:04:05.000" // YYYY-MM-DDTHH:MM:SS.ZZZ
  22. )
  23. // LogLevel defines log levels.
  24. type LogLevel uint8
  25. // defines our own log levels, just in case we'll change logger in future
  26. const (
  27. LevelDebug LogLevel = iota
  28. LevelInfo
  29. LevelWarn
  30. LevelError
  31. )
  32. var (
  33. logger zerolog.Logger
  34. consoleLogger zerolog.Logger
  35. rollingLogger *lumberjack.Logger
  36. )
  37. // StdLoggerWrapper is a wrapper for standard logger compatibility
  38. type StdLoggerWrapper struct {
  39. Sender string
  40. }
  41. // Write implements the io.Writer interface. This is useful to set as a writer
  42. // for the standard library log.
  43. func (l *StdLoggerWrapper) Write(p []byte) (n int, err error) {
  44. n = len(p)
  45. if n > 0 && p[n-1] == '\n' {
  46. // Trim CR added by stdlog.
  47. p = p[0 : n-1]
  48. }
  49. Log(LevelError, l.Sender, "", string(p))
  50. return
  51. }
  52. // LeveledLogger is a logger that accepts a message string and a variadic number of key-value pairs
  53. type LeveledLogger struct {
  54. Sender string
  55. additionalKeyVals []interface{}
  56. }
  57. func addKeysAndValues(ev *zerolog.Event, keysAndValues ...interface{}) {
  58. kvLen := len(keysAndValues)
  59. if kvLen%2 != 0 {
  60. extra := keysAndValues[kvLen-1]
  61. keysAndValues = append(keysAndValues[:kvLen-1], "EXTRA_VALUE_AT_END", extra)
  62. }
  63. for i := 0; i < len(keysAndValues); i += 2 {
  64. key, val := keysAndValues[i], keysAndValues[i+1]
  65. if keyStr, ok := key.(string); ok && keyStr != "timestamp" {
  66. ev.Str(keyStr, fmt.Sprintf("%v", val))
  67. }
  68. }
  69. }
  70. // Error logs at error level for the specified sender
  71. func (l *LeveledLogger) Error(msg string, keysAndValues ...interface{}) {
  72. ev := logger.Error()
  73. ev.Timestamp().Str("sender", l.Sender)
  74. if len(l.additionalKeyVals) > 0 {
  75. addKeysAndValues(ev, l.additionalKeyVals...)
  76. }
  77. addKeysAndValues(ev, keysAndValues...)
  78. ev.Msg(msg)
  79. }
  80. // Info logs at info level for the specified sender
  81. func (l *LeveledLogger) Info(msg string, keysAndValues ...interface{}) {
  82. ev := logger.Info()
  83. ev.Timestamp().Str("sender", l.Sender)
  84. if len(l.additionalKeyVals) > 0 {
  85. addKeysAndValues(ev, l.additionalKeyVals...)
  86. }
  87. addKeysAndValues(ev, keysAndValues...)
  88. ev.Msg(msg)
  89. }
  90. // Debug logs at debug level for the specified sender
  91. func (l *LeveledLogger) Debug(msg string, keysAndValues ...interface{}) {
  92. ev := logger.Debug()
  93. ev.Timestamp().Str("sender", l.Sender)
  94. if len(l.additionalKeyVals) > 0 {
  95. addKeysAndValues(ev, l.additionalKeyVals...)
  96. }
  97. addKeysAndValues(ev, keysAndValues...)
  98. ev.Msg(msg)
  99. }
  100. // Warn logs at warn level for the specified sender
  101. func (l *LeveledLogger) Warn(msg string, keysAndValues ...interface{}) {
  102. ev := logger.Warn()
  103. ev.Timestamp().Str("sender", l.Sender)
  104. if len(l.additionalKeyVals) > 0 {
  105. addKeysAndValues(ev, l.additionalKeyVals...)
  106. }
  107. addKeysAndValues(ev, keysAndValues...)
  108. ev.Msg(msg)
  109. }
  110. // With returns a LeveledLogger with additional context specific keyvals
  111. func (l *LeveledLogger) With(keysAndValues ...interface{}) ftpserverlog.Logger {
  112. return &LeveledLogger{
  113. Sender: l.Sender,
  114. additionalKeyVals: append(l.additionalKeyVals, keysAndValues...),
  115. }
  116. }
  117. // GetLogger get the configured logger instance
  118. func GetLogger() *zerolog.Logger {
  119. return &logger
  120. }
  121. // SetLogTime sets logging time related setting
  122. func SetLogTime(utc bool) {
  123. zerolog.TimeFieldFormat = dateFormat
  124. if utc {
  125. zerolog.TimestampFunc = time.Now().UTC
  126. } else {
  127. zerolog.TimestampFunc = time.Now
  128. }
  129. }
  130. // InitLogger configures the logger using the given parameters
  131. func InitLogger(logFilePath string, logMaxSize int, logMaxBackups int, logMaxAge int, logCompress, logUTCTime bool,
  132. level zerolog.Level,
  133. ) {
  134. SetLogTime(logUTCTime)
  135. if isLogFilePathValid(logFilePath) {
  136. logDir := filepath.Dir(logFilePath)
  137. if _, err := os.Stat(logDir); os.IsNotExist(err) {
  138. err = os.MkdirAll(logDir, os.ModePerm)
  139. if err != nil {
  140. fmt.Printf("unable to create log dir %#v: %v", logDir, err)
  141. }
  142. }
  143. rollingLogger = &lumberjack.Logger{
  144. Filename: logFilePath,
  145. MaxSize: logMaxSize,
  146. MaxBackups: logMaxBackups,
  147. MaxAge: logMaxAge,
  148. Compress: logCompress,
  149. LocalTime: !logUTCTime,
  150. }
  151. logger = zerolog.New(rollingLogger)
  152. EnableConsoleLogger(level)
  153. } else {
  154. logger = zerolog.New(&logSyncWrapper{
  155. output: os.Stdout,
  156. })
  157. consoleLogger = zerolog.Nop()
  158. }
  159. logger = logger.Level(level)
  160. }
  161. // InitStdErrLogger configures the logger to write to stderr
  162. func InitStdErrLogger(level zerolog.Level) {
  163. logger = zerolog.New(&logSyncWrapper{
  164. output: os.Stderr,
  165. }).Level(level)
  166. consoleLogger = zerolog.Nop()
  167. }
  168. // DisableLogger disable the main logger.
  169. // ConsoleLogger will not be affected
  170. func DisableLogger() {
  171. logger = zerolog.Nop()
  172. rollingLogger = nil
  173. }
  174. // EnableConsoleLogger enables the console logger
  175. func EnableConsoleLogger(level zerolog.Level) {
  176. consoleOutput := zerolog.ConsoleWriter{
  177. Out: os.Stdout,
  178. TimeFormat: dateFormat,
  179. }
  180. consoleLogger = zerolog.New(consoleOutput).With().Timestamp().Logger().Level(level)
  181. }
  182. // RotateLogFile closes the existing log file and immediately create a new one
  183. func RotateLogFile() error {
  184. if rollingLogger != nil {
  185. return rollingLogger.Rotate()
  186. }
  187. return errors.New("logging to file is disabled")
  188. }
  189. // Log logs at the specified level for the specified sender
  190. func Log(level LogLevel, sender string, connectionID string, format string, v ...interface{}) {
  191. var ev *zerolog.Event
  192. switch level {
  193. case LevelDebug:
  194. ev = logger.Debug()
  195. case LevelInfo:
  196. ev = logger.Info()
  197. case LevelWarn:
  198. ev = logger.Warn()
  199. default:
  200. ev = logger.Error()
  201. }
  202. ev.Timestamp().Str("sender", sender)
  203. if connectionID != "" {
  204. ev.Str("connection_id", connectionID)
  205. }
  206. ev.Msg(fmt.Sprintf(format, v...))
  207. }
  208. // Debug logs at debug level for the specified sender
  209. func Debug(sender string, connectionID string, format string, v ...interface{}) {
  210. Log(LevelDebug, sender, connectionID, format, v...)
  211. }
  212. // Info logs at info level for the specified sender
  213. func Info(sender string, connectionID string, format string, v ...interface{}) {
  214. Log(LevelInfo, sender, connectionID, format, v...)
  215. }
  216. // Warn logs at warn level for the specified sender
  217. func Warn(sender string, connectionID string, format string, v ...interface{}) {
  218. Log(LevelWarn, sender, connectionID, format, v...)
  219. }
  220. // Error logs at error level for the specified sender
  221. func Error(sender string, connectionID string, format string, v ...interface{}) {
  222. Log(LevelError, sender, connectionID, format, v...)
  223. }
  224. // DebugToConsole logs at debug level to stdout
  225. func DebugToConsole(format string, v ...interface{}) {
  226. consoleLogger.Debug().Msg(fmt.Sprintf(format, v...))
  227. }
  228. // InfoToConsole logs at info level to stdout
  229. func InfoToConsole(format string, v ...interface{}) {
  230. consoleLogger.Info().Msg(fmt.Sprintf(format, v...))
  231. }
  232. // WarnToConsole logs at info level to stdout
  233. func WarnToConsole(format string, v ...interface{}) {
  234. consoleLogger.Warn().Msg(fmt.Sprintf(format, v...))
  235. }
  236. // ErrorToConsole logs at error level to stdout
  237. func ErrorToConsole(format string, v ...interface{}) {
  238. consoleLogger.Error().Msg(fmt.Sprintf(format, v...))
  239. }
  240. // TransferLog logs uploads or downloads
  241. func TransferLog(operation, path string, elapsed int64, size int64, user, connectionID, protocol, localAddr,
  242. remoteAddr, ftpMode string,
  243. ) {
  244. ev := logger.Info().
  245. Timestamp().
  246. Str("sender", operation).
  247. Str("local_addr", localAddr).
  248. Str("remote_addr", remoteAddr).
  249. Int64("elapsed_ms", elapsed).
  250. Int64("size_bytes", size).
  251. Str("username", user).
  252. Str("file_path", path).
  253. Str("connection_id", connectionID).
  254. Str("protocol", protocol)
  255. if ftpMode != "" {
  256. ev.Str("ftp_mode", ftpMode)
  257. }
  258. ev.Send()
  259. }
  260. // CommandLog logs an SFTP/SCP/SSH command
  261. func CommandLog(command, path, target, user, fileMode, connectionID, protocol string, uid, gid int, atime, mtime,
  262. sshCommand string, size int64, localAddr, remoteAddr string) {
  263. logger.Info().
  264. Timestamp().
  265. Str("sender", command).
  266. Str("remote_addr", remoteAddr).
  267. Str("username", user).
  268. Str("file_path", path).
  269. Str("target_path", target).
  270. Str("filemode", fileMode).
  271. Int("uid", uid).
  272. Int("gid", gid).
  273. Str("access_time", atime).
  274. Str("modification_time", atime).
  275. Int64("size", size).
  276. Str("ssh_command", sshCommand).
  277. Str("connection_id", connectionID).
  278. Str("protocol", protocol).
  279. Send()
  280. }
  281. // ConnectionFailedLog logs failed attempts to initialize a connection.
  282. // A connection can fail for an authentication error or other errors such as
  283. // a client abort or a time out if the login does not happen in two minutes.
  284. // These logs are useful for better integration with Fail2ban and similar tools.
  285. func ConnectionFailedLog(user, ip, loginType, protocol, errorString string) {
  286. logger.Debug().
  287. Timestamp().
  288. Str("sender", "connection_failed").
  289. Str("client_ip", ip).
  290. Str("username", user).
  291. Str("login_type", loginType).
  292. Str("protocol", protocol).
  293. Str("error", errorString).
  294. Send()
  295. }
  296. func isLogFilePathValid(logFilePath string) bool {
  297. cleanInput := filepath.Clean(logFilePath)
  298. if cleanInput == "." || cleanInput == ".." {
  299. return false
  300. }
  301. return true
  302. }