logger.go 10 KB

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