logger.go 11 KB

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