From 38790ba19b12041b6a1aa562f62cd176b76ef827 Mon Sep 17 00:00:00 2001 From: sambaiz Date: Wed, 23 Dec 2020 18:02:05 +0900 Subject: [PATCH 1/2] NewRotatableFileAppenderWithBufferSizeAndFlushInterval --- README.md | 2 +- appender_file_rotatable.go | 20 +++++++++++++++++--- 2 files changed, 18 insertions(+), 4 deletions(-) diff --git a/README.md b/README.md index be4ad1c..5b54a37 100644 --- a/README.md +++ b/README.md @@ -178,7 +178,7 @@ Result: ``` ## 4.3. FileAppender -LogEventをファイルに出力します。内部で4096Byteをデフォルトでバッファし、容量超えた場合にFlushを行います。 +LogEventをファイルに出力します。内部で4096Byteをデフォルトでバッファし、容量超えた場合か一定時間ごとにFlushを行います。 また、バッファの内容はAppenderもしくはLoggerのCloserによってFlushされます。 アプリケーションの終了時やpanicリカバリーのタイミングでcloseを実行し、バッファが確実に出力されるように実装してください。 diff --git a/appender_file_rotatable.go b/appender_file_rotatable.go index 5860095..6e88b7f 100644 --- a/appender_file_rotatable.go +++ b/appender_file_rotatable.go @@ -5,12 +5,16 @@ import ( "os/signal" "sync" "syscall" + "time" ) +var defaultFlushInterval = time.Minute * 5 + // RotatableFileAppender RotatableFileAppender struct type RotatableFileAppender struct { *FileAppender - mu *sync.Mutex + mu *sync.Mutex + ticker *time.Ticker } // NewRotatableFileAppender returns new FileAppender @@ -20,6 +24,11 @@ func NewRotatableFileAppender(fileName string) (asyncFileAppender *RotatableFile // NewRotatableFileAppenderWithBufferSize returns new FileAppender func NewRotatableFileAppenderWithBufferSize(fileName string, bufferSize int) (asyncFileAppender *RotatableFileAppender, err error) { + return NewRotatableFileAppenderWithBufferSizeAndFlushInterval(fileName, bufferSize, defaultFlushInterval) +} + +// NewRotatableFileAppenderWithBufferSize returns new FileAppender +func NewRotatableFileAppenderWithBufferSizeAndFlushInterval(fileName string, bufferSize int, flushInterval time.Duration) (asyncFileAppender *RotatableFileAppender, err error) { fileAppender, err := NewFileAppenderWithBufferSize(fileName, bufferSize) if err != nil { @@ -29,6 +38,7 @@ func NewRotatableFileAppenderWithBufferSize(fileName string, bufferSize int) (as appender := &RotatableFileAppender{ FileAppender: fileAppender, mu: new(sync.Mutex), + ticker: time.NewTicker(flushInterval), } hup := make(chan os.Signal, 1) @@ -36,9 +46,12 @@ func NewRotatableFileAppenderWithBufferSize(fileName string, bufferSize int) (as go func() { for { - <-hup - appender.mu.Lock() + select { + case <-hup: + case <-appender.ticker.C: + } + appender.mu.Lock() appender.FileAppender.Close() newAppender, err := NewFileAppenderWithBufferSize(fileName, bufferSize) @@ -65,5 +78,6 @@ func (appender *RotatableFileAppender) Write(data []byte) (n int, err error) { func (appender *RotatableFileAppender) Close() error { appender.mu.Lock() defer appender.mu.Unlock() + appender.ticker.Stop() return appender.FileAppender.Close() } From dbd69eea19e93550466f51ab010061d7f872912c Mon Sep 17 00:00:00 2001 From: sambaiz Date: Thu, 24 Dec 2020 12:04:12 +0900 Subject: [PATCH 2/2] appender_file flush interval --- appender_file.go | 51 ++++++++++++++++++++++++++------------ appender_file_rotatable.go | 20 +++++---------- 2 files changed, 41 insertions(+), 30 deletions(-) diff --git a/appender_file.go b/appender_file.go index 3442e3d..c477fd0 100644 --- a/appender_file.go +++ b/appender_file.go @@ -1,39 +1,35 @@ package golog import ( + "context" "fmt" "os" "sync" + "time" ) // defaultBufferSize const defaultBufferSize = 4096 +const defaultFlushInterval = time.Minute * 5 + // FileAppender type FileAppender struct { file *os.File bufferedWriter *bufferedWriter mu *sync.Mutex activated bool + ticker *time.Ticker + stopTicker context.CancelFunc } // NewFileAppender returns new FileAppender func NewFileAppender(fileName string) (asyncFileAppender *FileAppender, err error) { - file, err := os.OpenFile(fileName, os.O_WRONLY|os.O_CREATE|os.O_APPEND, 0666) - if err != nil { - return nil, err - } - - return &FileAppender{ - file: file, - bufferedWriter: newBufferedWriter(file, withBufferSize(defaultBufferSize)), - mu: new(sync.Mutex), - activated: true, - }, nil + return NewFileAppenderWithBufferSizeAndFlushInterval(fileName, defaultBufferSize, defaultFlushInterval) } -// NewFileAppender returns new FileAppender -func NewFileAppenderWithBufferSize(fileName string, bufferSize int) (asyncFileAppender *FileAppender, err error) { +// NewFileAppenderWithBufferSizeAndFlushInterval returns new FileAppender +func NewFileAppenderWithBufferSizeAndFlushInterval(fileName string, bufferSize int, flushInterval time.Duration) (asyncFileAppender *FileAppender, err error) { file, err := os.OpenFile(fileName, os.O_WRONLY|os.O_CREATE|os.O_APPEND, 0666) if err != nil { return nil, err @@ -44,12 +40,34 @@ func NewFileAppenderWithBufferSize(fileName string, bufferSize int) (asyncFileAp size = bufferSize } - return &FileAppender{ + ctx, cancel := context.WithCancel(context.Background()) + appender := &FileAppender{ file: file, bufferedWriter: newBufferedWriter(file, withBufferSize(size)), mu: new(sync.Mutex), activated: true, - }, nil + ticker: time.NewTicker(flushInterval), + stopTicker: cancel, + } + + go func() { + for { + select { + case <-appender.ticker.C: + case <-ctx.Done(): + } + appender.mu.Lock() + if !appender.activated { + appender.ticker.Stop() + appender.mu.Unlock() + return + } + appender.bufferedWriter.Flush() + appender.mu.Unlock() + } + }() + + return appender, nil } // Write implements io.Write @@ -67,8 +85,9 @@ func (appender *FileAppender) Write(data []byte) (n int, err error) { func (appender *FileAppender) Close() error { appender.mu.Lock() defer func() { - appender.mu.Unlock() appender.activated = false + appender.stopTicker() + appender.mu.Unlock() }() if appender.activated { diff --git a/appender_file_rotatable.go b/appender_file_rotatable.go index 6e88b7f..46f8126 100644 --- a/appender_file_rotatable.go +++ b/appender_file_rotatable.go @@ -8,13 +8,10 @@ import ( "time" ) -var defaultFlushInterval = time.Minute * 5 - // RotatableFileAppender RotatableFileAppender struct type RotatableFileAppender struct { *FileAppender - mu *sync.Mutex - ticker *time.Ticker + mu *sync.Mutex } // NewRotatableFileAppender returns new FileAppender @@ -27,10 +24,10 @@ func NewRotatableFileAppenderWithBufferSize(fileName string, bufferSize int) (as return NewRotatableFileAppenderWithBufferSizeAndFlushInterval(fileName, bufferSize, defaultFlushInterval) } -// NewRotatableFileAppenderWithBufferSize returns new FileAppender +// NewRotatableFileAppenderWithBufferSizeAndFlushInterval returns new FileAppender func NewRotatableFileAppenderWithBufferSizeAndFlushInterval(fileName string, bufferSize int, flushInterval time.Duration) (asyncFileAppender *RotatableFileAppender, err error) { - fileAppender, err := NewFileAppenderWithBufferSize(fileName, bufferSize) + fileAppender, err := NewFileAppenderWithBufferSizeAndFlushInterval(fileName, bufferSize, flushInterval) if err != nil { return nil, err } @@ -38,7 +35,6 @@ func NewRotatableFileAppenderWithBufferSizeAndFlushInterval(fileName string, buf appender := &RotatableFileAppender{ FileAppender: fileAppender, mu: new(sync.Mutex), - ticker: time.NewTicker(flushInterval), } hup := make(chan os.Signal, 1) @@ -46,15 +42,12 @@ func NewRotatableFileAppenderWithBufferSizeAndFlushInterval(fileName string, buf go func() { for { - select { - case <-hup: - case <-appender.ticker.C: - } - + <-hup appender.mu.Lock() + appender.FileAppender.Close() - newAppender, err := NewFileAppenderWithBufferSize(fileName, bufferSize) + newAppender, err := NewFileAppenderWithBufferSizeAndFlushInterval(fileName, bufferSize, flushInterval) if err != nil { panic(err) } @@ -78,6 +71,5 @@ func (appender *RotatableFileAppender) Write(data []byte) (n int, err error) { func (appender *RotatableFileAppender) Close() error { appender.mu.Lock() defer appender.mu.Unlock() - appender.ticker.Stop() return appender.FileAppender.Close() }