forked from arthurkiller/rollingwriter
-
Notifications
You must be signed in to change notification settings - Fork 0
/
rollingwriter.go
257 lines (228 loc) · 7.02 KB
/
rollingwriter.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
package rollingwriter
import (
"errors"
"io"
"os"
"path"
"path/filepath"
"time"
)
// RollingPolicies giveout 3 policy for rolling.
const (
WithoutRolling = iota
TimeRolling
VolumeRolling
)
var (
// BufferSize defined the buffer size, by default 1 KB buffer will be allocated
BufferSize = 1024
// QueueSize defined the queue size for asynchronize write
QueueSize = 1024
// Precision defined the precision about the reopen operation condition
// check duration within second
Precision = 1
// DefaultFileMode set the default open mode rw-r--r-- by default
DefaultFileMode = os.FileMode(0644)
// DefaultFileFlag set the default file flag
DefaultFileFlag = os.O_RDWR | os.O_CREATE | os.O_APPEND
// ErrInternal defined the internal error
ErrInternal = errors.New("error internal")
// ErrClosed defined write while ctx close
ErrClosed = errors.New("error write on close")
// ErrInvalidArgument defined the invalid argument
ErrInvalidArgument = errors.New("error argument invalid")
// ErrQueueFull defined the queue full
ErrQueueFull = errors.New("async log queue full")
)
// Manager used to trigger rolling event.
type Manager interface {
// Fire will return a string channel
// while the rolling event occoured, new file name will generate
Fire() chan string
// Close the Manager
Close()
}
// RollingWriter implement the io writer
type RollingWriter interface {
io.Writer
Close() error
}
// LogFileFormatter log file format function
type LogFileFormatter func(time.Time) string
// Config give out the config for manager
type Config struct {
// LogPath defined the full path of log file directory.
// there comes out 2 different log file:
//
// 1. the current log
// log file path is located here:
// [LogPath]/[FileName].[FileExtension]
//
// 2. the tuncated log file
// the tuncated log file is backup here:
// [LogPath]/[FileName].[FileExtension].[TimeTag]
// if compressed true
// [LogPath]/[FileName].[FileExtension].gz.[TimeTag]
//
// NOTICE: blank field will be ignored
// By default we using '-' as separator, you can set it yourself
TimeTagFormat string `json:"time_tag_format"`
LogPath string `json:"log_path"`
FileName string `json:"file_name"`
// FileExtension defines the log file extension. By default, it's 'log'
FileExtension string `json:"file_extension"`
// FileFormatter log file path formatter for the file start write
// By default, append '.gz' suffix when Compress is true
FileFormatter LogFileFormatter `json:"-"`
// MaxRemain will auto clear the roling file list, set 0 will disable auto clean
MaxRemain int `json:"max_remain"`
// RollingPolicy give out the rolling policy
// We got 3 policies(actually, 2):
//
// 1. WithoutRolling: no rolling will happen
// 2. TimeRolling: rolling by time
// 3. VolumeRolling: rolling by file size
RollingPolicy int `json:"rolling_ploicy"`
RollingTimePattern string `json:"rolling_time_pattern"`
RollingVolumeSize string `json:"rolling_volume_size"`
// WriterMode in 4 modes below
// 1. none 2. lock
// 3. async 4. buffer
WriterMode string `json:"writer_mode"`
// BufferWriterThershould in Byte
BufferWriterThershould int `json:"buffer_thershould"`
// Compress will compress log file with gzip
Compress bool `json:"compress"`
// FilterEmptyBackup will not backup empty file if you set it true
FilterEmptyBackup bool `json:"filter_empty_backup"`
}
func (c *Config) fileFormat(start time.Time) (filename string) {
if c.FileFormatter != nil {
filename = c.FileFormatter(start)
if c.Compress && filepath.Ext(filename) != ".gz" {
filename += ".gz"
}
} else {
// [path-to-log]/filename.[FileExtension].2007010215041517
timeTag := start.Format(c.TimeTagFormat)
if c.Compress {
filename = path.Join(c.LogPath, c.FileName+"."+c.FileExtension+".gz."+timeTag)
} else {
filename = path.Join(c.LogPath, c.FileName+"."+c.FileExtension+"."+timeTag)
}
}
return
}
// NewDefaultConfig return the default config
func NewDefaultConfig() Config {
return Config{
LogPath: "./log",
TimeTagFormat: "200601021504",
FileName: "log",
FileExtension: "log",
MaxRemain: -1, // disable auto delete
RollingPolicy: 1, // TimeRotate by default
RollingTimePattern: "0 0 0 * * *", // Rolling at 00:00 AM everyday
RollingVolumeSize: "1G",
WriterMode: "lock",
BufferWriterThershould: 64,
Compress: false,
}
}
// LogFilePath return the absolute path on log file
func LogFilePath(c *Config) (filepath string) {
filepath = path.Join(c.LogPath, c.FileName) + "." + c.FileExtension
return
}
// Option defined config option
type Option func(*Config)
// WithTimeTagFormat set the TimeTag format string
func WithTimeTagFormat(format string) Option {
return func(p *Config) {
p.TimeTagFormat = format
}
}
// WithLogPath set the log dir and auto create dir tree
// if the dir/path is not exist
func WithLogPath(path string) Option {
return func(p *Config) {
p.LogPath = path
}
}
// WithFileName set the log file name
func WithFileName(name string) Option {
return func(p *Config) {
p.FileName = name
}
}
// WithFileExtension set the log file extension
func WithFileExtension(ext string) Option {
return func(p *Config) {
p.FileExtension = ext
}
}
// WithFileFormatter set the log file formatter
func WithFileFormatter(formatter LogFileFormatter) Option {
return func(p *Config) {
p.FileFormatter = formatter
}
}
// WithAsynchronous enable the asynchronous write for writer
func WithAsynchronous() Option {
return func(p *Config) {
p.WriterMode = "async"
}
}
// WithLock will enable the lock in writer
// Writer will call write with the Lock to guarantee the parallel safe
func WithLock() Option {
return func(p *Config) {
p.WriterMode = "lock"
}
}
// WithBuffer will enable the buffer writer mode
func WithBuffer() Option {
return func(p *Config) {
p.WriterMode = "buffer"
}
}
// WithBufferThershould set buffer write thershould
func WithBufferThershould(n int) Option {
return func(p *Config) {
p.BufferWriterThershould = n
}
}
// WithCompress will auto compress the tuncated log file with gzip
func WithCompress() Option {
return func(p *Config) {
p.Compress = true
}
}
// WithMaxRemain enable the auto deletion for old file when exceed the given max value
// Bydefault -1 will disable the auto deletion
func WithMaxRemain(max int) Option {
return func(p *Config) {
p.MaxRemain = max
}
}
// WithoutRolling set no rolling policy
func WithoutRollingPolicy() Option {
return func(p *Config) {
p.RollingPolicy = WithoutRolling
}
}
// WithRollingTimePattern set the time rolling policy time pattern obey the Corn table style
// visit http://crontab.org/ for details
func WithRollingTimePattern(pattern string) Option {
return func(p *Config) {
p.RollingPolicy = TimeRolling
p.RollingTimePattern = pattern
}
}
// WithRollingVolumeSize set the rolling file truncation threshold size
func WithRollingVolumeSize(size string) Option {
return func(p *Config) {
p.RollingPolicy = VolumeRolling
p.RollingVolumeSize = size
}
}