2014-07-26 00:24:27 -04:00
// Copyright 2014 The Gogs Authors. All rights reserved.
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
package log
import (
2019-04-02 08:48:31 +01:00
"bufio"
"compress/gzip"
2014-07-26 00:24:27 -04:00
"encoding/json"
"errors"
"fmt"
"os"
"path/filepath"
"strings"
"sync"
"time"
)
2019-04-02 08:48:31 +01:00
// FileLogger implements LoggerProvider.
2014-07-26 00:24:27 -04:00
// It writes messages by lines limit, file size limit, or time frequency.
2019-04-02 08:48:31 +01:00
type FileLogger struct {
2019-04-07 01:25:14 +01:00
WriterLogger
2014-07-26 00:24:27 -04:00
mw * MuxWriter
// The opened file
Filename string ` json:"filename" `
// Rotate at size
2016-11-26 19:53:29 +08:00
Maxsize int ` json:"maxsize" `
maxsizeCursize int
2014-07-26 00:24:27 -04:00
// Rotate daily
2016-11-26 19:53:29 +08:00
Daily bool ` json:"daily" `
Maxdays int64 ` json:"maxdays" `
dailyOpenDate int
2014-07-26 00:24:27 -04:00
Rotate bool ` json:"rotate" `
2019-04-02 08:48:31 +01:00
Compress bool ` json:"compress" `
CompressionLevel int ` json:"compressionLevel" `
2014-07-26 00:24:27 -04:00
2019-04-02 08:48:31 +01:00
startLock sync . Mutex // Only one log can write to the file
2014-07-26 00:24:27 -04:00
}
2016-11-26 19:53:29 +08:00
// MuxWriter an *os.File writer with locker.
2014-07-26 00:24:27 -04:00
type MuxWriter struct {
2019-04-02 08:48:31 +01:00
mu sync . Mutex
fd * os . File
owner * FileLogger
2014-07-26 00:24:27 -04:00
}
2016-11-26 19:53:29 +08:00
// Write writes to os.File.
2019-04-02 08:48:31 +01:00
func ( mw * MuxWriter ) Write ( b [ ] byte ) ( int , error ) {
mw . mu . Lock ( )
defer mw . mu . Unlock ( )
mw . owner . docheck ( len ( b ) )
return mw . fd . Write ( b )
}
// Close the internal writer
func ( mw * MuxWriter ) Close ( ) error {
return mw . fd . Close ( )
2014-07-26 00:24:27 -04:00
}
2016-11-26 19:53:29 +08:00
// SetFd sets os.File in writer.
2019-04-02 08:48:31 +01:00
func ( mw * MuxWriter ) SetFd ( fd * os . File ) {
if mw . fd != nil {
mw . fd . Close ( )
2014-07-26 00:24:27 -04:00
}
2019-04-02 08:48:31 +01:00
mw . fd = fd
}
// NewFileLogger create a FileLogger returning as LoggerProvider.
func NewFileLogger ( ) LoggerProvider {
log := & FileLogger {
Filename : "" ,
Maxsize : 1 << 28 , //256 MB
Daily : true ,
Maxdays : 7 ,
Rotate : true ,
Compress : true ,
CompressionLevel : gzip . DefaultCompression ,
2014-07-26 00:24:27 -04:00
}
2019-04-02 08:48:31 +01:00
log . Level = TRACE
2014-07-26 00:24:27 -04:00
// use MuxWriter instead direct use os.File for lock write when rotate
2019-04-02 08:48:31 +01:00
log . mw = new ( MuxWriter )
log . mw . owner = log
return log
2014-07-26 00:24:27 -04:00
}
// Init file logger with json config.
// config like:
// {
// "filename":"log/gogs.log",
// "maxsize":1<<30,
// "daily":true,
// "maxdays":15,
// "rotate":true
// }
2019-04-02 08:48:31 +01:00
func ( log * FileLogger ) Init ( config string ) error {
if err := json . Unmarshal ( [ ] byte ( config ) , log ) ; err != nil {
2014-07-26 00:24:27 -04:00
return err
}
2019-04-02 08:48:31 +01:00
if len ( log . Filename ) == 0 {
2014-07-26 00:24:27 -04:00
return errors . New ( "config must have filename" )
}
2019-04-02 08:48:31 +01:00
// set MuxWriter as Logger's io.Writer
2019-04-07 01:25:14 +01:00
log . NewWriterLogger ( log . mw )
2019-04-02 08:48:31 +01:00
return log . StartLogger ( )
2014-07-26 00:24:27 -04:00
}
2016-11-26 19:53:29 +08:00
// StartLogger start file logger. create log file and set to locker-inside file writer.
2019-04-02 08:48:31 +01:00
func ( log * FileLogger ) StartLogger ( ) error {
fd , err := log . createLogFile ( )
2014-07-26 00:24:27 -04:00
if err != nil {
return err
}
2019-04-02 08:48:31 +01:00
log . mw . SetFd ( fd )
return log . initFd ( )
2014-07-26 00:24:27 -04:00
}
2019-04-02 08:48:31 +01:00
func ( log * FileLogger ) docheck ( size int ) {
log . startLock . Lock ( )
defer log . startLock . Unlock ( )
if log . Rotate && ( ( log . Maxsize > 0 && log . maxsizeCursize >= log . Maxsize ) ||
( log . Daily && time . Now ( ) . Day ( ) != log . dailyOpenDate ) ) {
if err := log . DoRotate ( ) ; err != nil {
fmt . Fprintf ( os . Stderr , "FileLogger(%q): %s\n" , log . Filename , err )
2014-07-26 00:24:27 -04:00
return
}
}
2019-04-02 08:48:31 +01:00
log . maxsizeCursize += size
2014-07-26 00:24:27 -04:00
}
2019-04-02 08:48:31 +01:00
func ( log * FileLogger ) createLogFile ( ) ( * os . File , error ) {
2014-07-26 00:24:27 -04:00
// Open the log file
2019-04-02 08:48:31 +01:00
return os . OpenFile ( log . Filename , os . O_WRONLY | os . O_APPEND | os . O_CREATE , 0660 )
2014-07-26 00:24:27 -04:00
}
2019-04-02 08:48:31 +01:00
func ( log * FileLogger ) initFd ( ) error {
fd := log . mw . fd
2014-07-26 00:24:27 -04:00
finfo , err := fd . Stat ( )
if err != nil {
2017-01-29 12:13:57 -08:00
return fmt . Errorf ( "get stat: %v" , err )
2014-07-26 00:24:27 -04:00
}
2019-04-02 08:48:31 +01:00
log . maxsizeCursize = int ( finfo . Size ( ) )
log . dailyOpenDate = time . Now ( ) . Day ( )
2014-07-26 00:24:27 -04:00
return nil
}
// DoRotate means it need to write file in new file.
// new file name like xx.log.2013-01-01.2
2019-04-02 08:48:31 +01:00
func ( log * FileLogger ) DoRotate ( ) error {
_ , err := os . Lstat ( log . Filename )
2014-07-26 00:24:27 -04:00
if err == nil { // file exists
// Find the next available number
num := 1
fname := ""
for ; err == nil && num <= 999 ; num ++ {
2019-04-02 08:48:31 +01:00
fname = log . Filename + fmt . Sprintf ( ".%s.%03d" , time . Now ( ) . Format ( "2006-01-02" ) , num )
2014-07-26 00:24:27 -04:00
_ , err = os . Lstat ( fname )
2019-04-02 08:48:31 +01:00
if log . Compress && err != nil {
_ , err = os . Lstat ( fname + ".gz" )
}
2014-07-26 00:24:27 -04:00
}
// return error if the last file checked still existed
if err == nil {
2019-04-02 08:48:31 +01:00
return fmt . Errorf ( "rotate: cannot find free log number to rename %s" , log . Filename )
2014-07-26 00:24:27 -04:00
}
2019-04-02 08:48:31 +01:00
fd := log . mw . fd
2014-07-26 00:24:27 -04:00
fd . Close ( )
// close fd before rename
// Rename the file to its newfound home
2019-04-02 08:48:31 +01:00
if err = os . Rename ( log . Filename , fname ) ; err != nil {
2017-01-29 12:13:57 -08:00
return fmt . Errorf ( "Rotate: %v" , err )
2014-07-26 00:24:27 -04:00
}
2019-04-02 08:48:31 +01:00
if log . Compress {
go compressOldLogFile ( fname , log . CompressionLevel )
}
2014-07-26 00:24:27 -04:00
// re-start logger
2019-04-02 08:48:31 +01:00
if err = log . StartLogger ( ) ; err != nil {
2017-01-29 12:13:57 -08:00
return fmt . Errorf ( "Rotate StartLogger: %v" , err )
2014-07-26 00:24:27 -04:00
}
2019-04-02 08:48:31 +01:00
go log . deleteOldLog ( )
2014-07-26 00:24:27 -04:00
}
return nil
}
2019-04-02 08:48:31 +01:00
func compressOldLogFile ( fname string , compressionLevel int ) error {
reader , err := os . Open ( fname )
if err != nil {
return err
}
defer reader . Close ( )
buffer := bufio . NewReader ( reader )
fw , err := os . OpenFile ( fname + ".gz" , os . O_WRONLY | os . O_CREATE , 0660 )
if err != nil {
return err
}
defer fw . Close ( )
zw , err := gzip . NewWriterLevel ( fw , compressionLevel )
if err != nil {
return err
}
defer zw . Close ( )
_ , err = buffer . WriteTo ( zw )
if err != nil {
zw . Close ( )
fw . Close ( )
os . Remove ( fname + ".gz" )
return err
}
reader . Close ( )
return os . Remove ( fname )
}
func ( log * FileLogger ) deleteOldLog ( ) {
dir := filepath . Dir ( log . Filename )
2019-06-12 21:41:28 +02:00
_ = filepath . Walk ( dir , func ( path string , info os . FileInfo , err error ) ( returnErr error ) {
2014-08-27 16:39:36 +08:00
defer func ( ) {
if r := recover ( ) ; r != nil {
returnErr = fmt . Errorf ( "Unable to delete old log '%s', error: %+v" , path , r )
}
} ( )
2019-04-02 08:48:31 +01:00
if ! info . IsDir ( ) && info . ModTime ( ) . Unix ( ) < ( time . Now ( ) . Unix ( ) - 60 * 60 * 24 * log . Maxdays ) {
if strings . HasPrefix ( filepath . Base ( path ) , filepath . Base ( log . Filename ) ) {
2016-12-01 00:56:15 +01:00
if err := os . Remove ( path ) ; err != nil {
2017-01-29 12:13:57 -08:00
returnErr = fmt . Errorf ( "Failed to remove %s: %v" , path , err )
2016-12-01 00:56:15 +01:00
}
2014-07-26 00:24:27 -04:00
}
}
2014-08-27 16:39:36 +08:00
return returnErr
2014-07-26 00:24:27 -04:00
} )
}
2016-11-26 19:53:29 +08:00
// Flush flush file logger.
2014-07-26 00:24:27 -04:00
// there are no buffering messages in file logger in memory.
// flush file means sync file from disk.
2019-04-02 08:48:31 +01:00
func ( log * FileLogger ) Flush ( ) {
2019-06-12 21:41:28 +02:00
_ = log . mw . fd . Sync ( )
2019-04-02 08:48:31 +01:00
}
// GetName returns the default name for this implementation
func ( log * FileLogger ) GetName ( ) string {
return "file"
2014-07-26 00:24:27 -04:00
}
func init ( ) {
2019-04-02 08:48:31 +01:00
Register ( "file" , NewFileLogger )
2014-07-26 00:24:27 -04:00
}