-
Notifications
You must be signed in to change notification settings - Fork 14
/
Copy pathlogging.go
718 lines (598 loc) · 21.9 KB
/
logging.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
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
//Package logging for Go takes a slightly different approach based on our experience with real world logging.
//
//A default logger provides an easy path to logging.
//Named loggers provide a way to set levels by package.
//Tags provide a way to set levels across concepts, perpendicular to logger names.
//Default log levels and default tag levels are independent of the default logger.
//All loggers share the same appenders - but appenders can be associated with a level which is unrelated to tags.
//Each logger has an optional buffer, that will be flushd whenever its level/tags change.
//This buffer contains un-passed messages. So that it is possible to configure the system to capture messages and replay them later.
//Replayed messages are tagged and have a double time stamp.
//A default appender is initialized to send log messages to stderr.
package logging
import (
"container/ring"
"fmt"
"runtime"
"sync"
"sync/atomic"
"time"
)
//Logger is the interface for the objects that are the target of logging messages. Logging methods
//imply a level. For example, Info() implies a level of LogLevel.INFO.
type Logger interface {
ErrorWithTagsf(tags []string, fmt string, args ...interface{})
ErrorWithTags(tags []string, args ...interface{})
Errorf(fmt string, args ...interface{})
Error(args ...interface{})
WarnWithTagsf(tags []string, fmt string, args ...interface{})
WarnWithTags(tags []string, args ...interface{})
Warnf(fmt string, args ...interface{})
Warn(args ...interface{})
InfoWithTagsf(tags []string, fmt string, args ...interface{})
InfoWithTags(tags []string, args ...interface{})
Infof(fmt string, args ...interface{})
Info(args ...interface{})
DebugWithTagsf(tags []string, fmt string, args ...interface{})
DebugWithTags(tags []string, args ...interface{})
Debugf(fmt string, args ...interface{})
Debug(args ...interface{})
VerboseWithTagsf(tags []string, fmt string, args ...interface{})
Verbosef(fmt string, args ...interface{})
SetLogLevel(l LogLevel)
SetTagLevel(tag string, l LogLevel)
CheckLevel(l LogLevel, tags []string) bool
SetBufferLength(length int)
}
const (
stopped = iota
paused
running
)
//logMutex is a global lock for protecting all global state in package
var logMutex = new(sync.RWMutex)
//defaultLogger is provided for most logging situations
var defaultLogger *LoggerImpl
//The default format is used to determine how appenders without a custom format log their messages
var defaultFormatter = GetFormatter(FULL)
//Loggers share the appenders
var appenders = make([]LogAppender, 0)
//The package maintains a map of named loggers
var loggers = make(map[string]*LoggerImpl)
var incomingChannel = make(chan *LogRecord, 2048)
var stateChannel = make(chan int, 0)
var waiter = new(sync.WaitGroup)
var logged uint64
var processed uint64
var logErrors chan<- error
var enableVerbose int32
func init() {
defaultLogger = new(LoggerImpl)
defaultLogger.name = "_default"
defaultLogger.level = INFO
defaultLogger.SetBufferLength(0)
AddAppender(NewStdErrAppender())
AdaptStandardLogging(INFO, nil)
go processIncoming()
}
//LogRecord is the type used in the logging buffer
type LogRecord struct {
//Time is the time that the log record is being appended, can be
//different from Original if the record was buffered
Time time.Time
//Original is the original time for the log record
Original time.Time
//Level is the level the record was logged at
Level LogLevel
//Tags are the custom tags assigned to the record when it was logged
Tags []string
//Message is the actual log message
Message string
//Logger is the logger associated with this log record, if any
Logger *LoggerImpl
}
//LoggerImpl stores the data for a logger.
//A Logger maintains its own level, tag levels and buffer. Each logger is named.
type LoggerImpl struct {
name string
level LogLevel
tagLevels map[string]LogLevel
buffer *ring.Ring
}
//PauseLogging stops all logging from being processed.
//Pause will not wait for all log messages to be processed
func PauseLogging() {
stateChannel <- paused
}
//RestartLogging starts messages logging again
func RestartLogging() {
stateChannel <- running
}
//StopLogging can only be called once, and completely stops the logging
//process
func StopLogging() {
stateChannel <- stopped
waiter.Wait()
}
func processIncoming() {
loop:
for {
select {
case record := <-incomingChannel:
processLogRecord(record)
case newState := <-stateChannel:
switch newState {
case stopped:
waiter.Done()
break loop
case paused: //run a sub-loop looking for a state change
subloop:
for {
select {
case state := <-stateChannel:
switch state {
case stopped:
waiter.Done()
break loop
case running:
break subloop
default:
continue
}
}
}
}
}
}
}
//WaitForIncoming should be used in tests or system shutdowns to make sure
//that all of the log messages pushed into the logging channel are processed
//and appended appropriately.
func WaitForIncoming() {
runtime.Gosched() //start by giving the other go routines a chance to run
for {
if atomic.LoadUint64(&processed) != atomic.LoadUint64(&logged) {
time.Sleep(2 * time.Millisecond)
} else {
return
}
}
}
//CaptureLoggingErrors allows the logging user to provide a channel
//for capturing logging errors. Any error during the logging process, like an
//appender failing will be sent to this channel.
//By default there is no error channel.
//Logging will not block when writting to the error channel so make sure the
//channel is big enough to capture errors
func CaptureLoggingErrors(errs chan<- error) {
logMutex.Lock()
logErrors = errs
logMutex.Unlock()
}
//DefaultLogger returns a logger that can be used when a named logger isn't required
func DefaultLogger() Logger {
return defaultLogger
}
//GetLogger returns a named logger, creating it if necessary. The logger will have the default settings.
//By default the logger will use the DefaultLoggers level and tag levels
func GetLogger(name string) Logger {
logMutex.RLock()
logger := loggers[name]
logMutex.RUnlock()
if logger == nil {
logger = new(LoggerImpl)
logger.name = name
logger.level = DEFAULT
logger.SetBufferLength(defaultLogger.buffer.Len())
logMutex.Lock()
loggers[name] = logger
logMutex.Unlock()
}
return logger
}
//EnableVerboseLogging by default verbose logging is ignored, use this
//method to allow verbose logging
func EnableVerboseLogging() {
atomic.StoreInt32(&enableVerbose, 1)
}
//DisableVerboseLogging by default verbose logging is ignored, use this
//method to turn off verbose logging if you have enabled it
func DisableVerboseLogging() {
atomic.StoreInt32(&enableVerbose, 0)
}
//SetDefaultLogLevel sets the default loggers log level, flushes all buffers in case messages are cleared for logging
func SetDefaultLogLevel(l LogLevel) {
defaultLogger.SetLogLevel(l)
}
//SetDefaultTagLogLevel sets the default loggers level for the specified tag, flushes all buffers in case messages are cleared for logging..
func SetDefaultTagLogLevel(tag string, l LogLevel) {
defaultLogger.SetTagLevel(tag, l)
}
//SetDefaultFormatter sets the default formatter used by appenders that don't have their own
func SetDefaultFormatter(formatter LogFormatter) {
logMutex.Lock()
defaultFormatter = formatter
logMutex.Unlock()
}
//SetDefaultBufferLength sets the buffer length for the default logger, new loggers will use this length.
//Existing loggers with buffers are not affected, those with buffers are not effected.
func SetDefaultBufferLength(length int) {
logMutex.Lock()
defaultLogger.setBufferLengthImpl(length)
for _, val := range loggers {
if val.buffer == nil {
val.setBufferLengthImpl(length)
}
}
logMutex.Unlock()
}
//AddAppender adds a new global appender for use by all loggers. Levels can be used to restrict logging to specific appenders.
func AddAppender(appender LogAppender) {
logMutex.Lock()
appenders = append(appenders, appender)
logMutex.Unlock()
}
//ClearAppenders removes all of the global appenders, mainly used during configuration.
//Will pause and restart logging
func ClearAppenders() {
PauseLogging()
logMutex.Lock()
for _, appender := range appenders {
if app, ok := appender.(ClosableAppender); ok {
app.Close()
}
}
appenders = make([]LogAppender, 0)
logMutex.Unlock()
RestartLogging()
}
//ClearLoggers is provided so that an application can
//completely reset its logging configuration, for example
//on a SIGHUP
func ClearLoggers() {
PauseLogging()
logMutex.Lock()
loggers = make(map[string]*LoggerImpl)
logMutex.Unlock()
RestartLogging()
}
/*
AddTag creates a new array and adds a string to it. This insures that no
slices are shared for tags.
*/
func AddTag(tags []string, newTag string) []string {
newTags := make([]string, 0, len(tags)+1)
newTags = append(newTags, tags...)
newTags = append(newTags, newTag)
return newTags
}
//SetLogLevel sets the level of messages allowed for a logger. This level can be
//overriden for specific tags using SetTagLevel. Changing the level for a Logger
//flushes its buffer in case messages are now free to be logged. This means that
//buffered messages might be printed out of order, but will be formatted to indicate this.
func (logger *LoggerImpl) SetLogLevel(l LogLevel) {
logMutex.Lock()
logger.level = l
wait := new(sync.WaitGroup)
if logger == defaultLogger {
flushAllLoggers(wait)
} else {
wait.Add(1)
logger.flushBuffer(wait)
}
logMutex.Unlock()
wait.Wait()
}
//SetTagLevel assigns a log level to a specific tag. This level can override the general
//level for a logger allowing specific log messages to slip through and be appended to the logs
func (logger *LoggerImpl) SetTagLevel(tag string, l LogLevel) {
logMutex.Lock()
if logger.tagLevels == nil {
logger.tagLevels = make(map[string]LogLevel)
}
logger.tagLevels[tag] = l
wait := new(sync.WaitGroup)
if logger == defaultLogger {
flushAllLoggers(wait)
} else {
wait.Add(1)
logger.flushBuffer(wait)
}
logMutex.Unlock()
wait.Wait()
}
//SetBufferLength clears the buffer and creates a new one of the specified length.
func (logger *LoggerImpl) SetBufferLength(length int) {
logMutex.Lock()
logger.setBufferLengthImpl(length)
logMutex.Unlock()
}
//expects the lock
func (logger *LoggerImpl) setBufferLengthImpl(length int) {
if length == 0 {
logger.buffer = nil
} else if length != logger.buffer.Len() {
logger.buffer = ring.New(length)
}
}
//NewLogRecord creates a log record object
func NewLogRecord(logger *LoggerImpl, level LogLevel, tags []string, message string, time time.Time, original time.Time) *LogRecord {
record := new(LogRecord)
record.Logger = logger
record.Level = level
record.Tags = tags
record.Message = message
record.Time = time
record.Original = original
return record
}
//should be called inside the logging lock,
//puts the error on the logging error channel if one is set
func logError(err error) {
if err != nil && logErrors != nil {
select {
case logErrors <- err:
//write the error
default:
//don't write or block
}
}
}
/* Check the tags for this logger, or the defaults, if any pass, then we pass */
/* Should be called inside the logging lock */
func (logger *LoggerImpl) checkTagLevel(l LogLevel, tags []string) bool {
for _, tag := range tags {
if logger.tagLevels != nil {
if tagLevel, ok := logger.tagLevels[tag]; ok && tagLevel <= l {
return true
}
}
if logger != defaultLogger && defaultLogger.tagLevels != nil {
if tagLevel, ok := defaultLogger.tagLevels[tag]; ok && tagLevel <= l {
return true
}
}
}
return false
}
//CheckLevel tests the default logger for its permissions
func CheckLevel(l LogLevel, tags []string) bool {
return defaultLogger.CheckLevel(l, tags)
}
//CheckLevel checks tags, then check the level on this , or the default level
func (logger *LoggerImpl) CheckLevel(l LogLevel, tags []string) bool {
logMutex.RLock()
defer logMutex.RUnlock()
return logger.checkLevelWithTags(l, tags)
}
//requires the lock be acquired
func (logger *LoggerImpl) checkLevelWithTags(l LogLevel, tags []string) bool {
if (logger.tagLevels != nil || defaultLogger.tagLevels != nil) && tags != nil {
matchTag := logger.checkTagLevel(l, tags)
if matchTag {
return true //otherwise check the general level
}
}
if logger.level != DEFAULT {
return logger.level <= l
}
return defaultLogger.level <= l
}
//flushAllLoggers expects the logging lock to be held by the caller
func flushAllLoggers(wait *sync.WaitGroup) {
wait.Add(len(loggers) + 1)
for _, val := range loggers {
val.flushBuffer(wait)
}
defaultLogger.flushBuffer(wait)
}
//should be called witin the lock
func logToAppenders(record *LogRecord) {
for _, appender := range appenders {
err := appender.Log(record)
logError(err)
}
}
func processLogRecord(record *LogRecord) {
logMutex.RLock()
defer logMutex.RUnlock()
logger := record.Logger
passed := logger.checkLevelWithTags(record.Level, record.Tags)
if passed {
logToAppenders(record)
} else if logger.buffer != nil && record.Level > VERBOSE {
logger.buffer.Next().Value = record
logger.buffer = logger.buffer.Next()
}
atomic.AddUint64(&processed, 1)
}
//flushBuffer expects the logging lock to be held, and does not take the lock
//should call done on the wait group when the buffer is flushed
//does not 1 to the waitgroup
func (logger *LoggerImpl) flushBuffer(wait *sync.WaitGroup) {
if logger.buffer != nil {
now := time.Now()
oldBuffer := logger.buffer
logger.buffer = ring.New(oldBuffer.Len())
go func() {
oldBuffer.Do(func(x interface{}) {
if x == nil {
return
}
record := x.(*LogRecord)
record.Time = now
atomic.AddUint64(&logged, 1)
incomingChannel <- record
})
wait.Done()
}()
} else {
wait.Done()
}
}
func (logger *LoggerImpl) logwithformat(level LogLevel, tags []string, format string, args ...interface{}) {
if level == VERBOSE && atomic.LoadInt32(&enableVerbose) != 1 {
return
}
now := time.Now()
msg := ""
if format == "" {
msg = fmt.Sprint(args...)
} else {
msg = fmt.Sprintf(format, args...)
}
logRecord := NewLogRecord(logger, level, tags, msg, now, now)
atomic.AddUint64(&logged, 1)
incomingChannel <- logRecord
}
func (logger *LoggerImpl) log(level LogLevel, tags []string, args ...interface{}) {
logger.logwithformat(level, tags, "", args...)
}
//ErrorWithTagsf logs an ERROR level message with the provided tags and formatted string.
func (logger *LoggerImpl) ErrorWithTagsf(tags []string, fmt string, args ...interface{}) {
logger.logwithformat(ERROR, tags, fmt, args...)
}
//ErrorWithTags logs an ERROR level message with the provided tags and provided arguments joined into a string.
func (logger *LoggerImpl) ErrorWithTags(tags []string, args ...interface{}) {
logger.log(ERROR, tags, args...)
}
//Errorf logs an ERROR level message with the no tags and formatted string.
func (logger *LoggerImpl) Errorf(fmt string, args ...interface{}) {
logger.logwithformat(ERROR, nil, fmt, args...)
}
//Error logs an ERROR level message with no tags and provided arguments joined into a string.
func (logger *LoggerImpl) Error(args ...interface{}) {
logger.log(ERROR, nil, args...)
}
//WarnWithTagsf logs an WARN level message with the provided tags and formatted string.
func (logger *LoggerImpl) WarnWithTagsf(tags []string, fmt string, args ...interface{}) {
logger.logwithformat(WARN, tags, fmt, args...)
}
//WarnWithTags logs an WARN level message with the provided tags and provided arguments joined into a string.
func (logger *LoggerImpl) WarnWithTags(tags []string, args ...interface{}) {
logger.log(WARN, tags, args...)
}
//Warnf logs an WARN level message with the no tags and formatted string.
func (logger *LoggerImpl) Warnf(fmt string, args ...interface{}) {
logger.logwithformat(WARN, nil, fmt, args...)
}
//Warn logs an WARN level message with no tags and provided arguments joined into a string.
func (logger *LoggerImpl) Warn(args ...interface{}) {
logger.log(WARN, nil, args...)
}
//InfoWithTagsf logs an INFO level message with the provided tags and formatted string.
func (logger *LoggerImpl) InfoWithTagsf(tags []string, fmt string, args ...interface{}) {
logger.logwithformat(INFO, tags, fmt, args...)
}
//InfoWithTags logs an INFO level message with the provided tags and provided arguments joined into a string.
func (logger *LoggerImpl) InfoWithTags(tags []string, args ...interface{}) {
logger.log(INFO, tags, args...)
}
//Infof logs an INFO level message with the no tags and formatted string.
func (logger *LoggerImpl) Infof(fmt string, args ...interface{}) {
logger.logwithformat(INFO, nil, fmt, args...)
}
//Info logs an INFO level message with no tags and provided arguments joined into a string.
func (logger *LoggerImpl) Info(args ...interface{}) {
logger.log(INFO, nil, args...)
}
//DebugWithTagsf logs an DEBUG level message with the provided tags and formatted string.
func (logger *LoggerImpl) DebugWithTagsf(tags []string, fmt string, args ...interface{}) {
logger.logwithformat(DEBUG, tags, fmt, args...)
}
//DebugWithTags logs an DEBUG level message with the provided tags and provided arguments joined into a string.
func (logger *LoggerImpl) DebugWithTags(tags []string, args ...interface{}) {
logger.log(DEBUG, tags, args...)
}
//Debugf logs an DEBUG level message with the no tags and formatted string.
func (logger *LoggerImpl) Debugf(fmt string, args ...interface{}) {
logger.logwithformat(DEBUG, nil, fmt, args...)
}
//Debug logs an DEBUG level message with no tags and provided arguments joined into a string.
func (logger *LoggerImpl) Debug(args ...interface{}) {
logger.log(DEBUG, nil, args...)
}
//VerboseWithTagsf logs an VERBOSE level message with the provided tags and formatted string.
//Verbose messages are not buffered
func (logger *LoggerImpl) VerboseWithTagsf(tags []string, fmt string, args ...interface{}) {
logger.logwithformat(VERBOSE, tags, fmt, args...)
}
//Verbosef logs an VERBOSE level message with the no tags and formatted string.
//Verbose messages are not buffered
func (logger *LoggerImpl) Verbosef(fmt string, args ...interface{}) {
logger.logwithformat(VERBOSE, nil, fmt, args...)
}
//ErrorWithTagsf logs an ERROR level message with the provided tags and formatted string. Uses the default logger.
func ErrorWithTagsf(tags []string, fmt string, args ...interface{}) {
defaultLogger.logwithformat(ERROR, tags, fmt, args...)
}
//ErrorWithTags logs an ERROR level message with the provided tags and provided arguments joined into a string. Uses the default logger.
func ErrorWithTags(tags []string, args ...interface{}) {
defaultLogger.log(ERROR, tags, args...)
}
//Errorf logs an ERROR level message with the no tags and formatted string. Uses the default logger.
func Errorf(fmt string, args ...interface{}) {
defaultLogger.logwithformat(ERROR, nil, fmt, args...)
}
//Error logs an ERROR level message with no tags and provided arguments joined into a string. Uses the default logger.
func Error(args ...interface{}) {
defaultLogger.log(ERROR, nil, args...)
}
//WarnWithTagsf logs an WARN level message with the provided tags and formatted string. Uses the default logger.
func WarnWithTagsf(tags []string, fmt string, args ...interface{}) {
defaultLogger.logwithformat(WARN, tags, fmt, args...)
}
//WarnWithTags logs an WARN level message with the provided tags and provided arguments joined into a string. Uses the default logger.
func WarnWithTags(tags []string, args ...interface{}) {
defaultLogger.log(WARN, tags, args...)
}
//Warnf logs an WARN level message with the no tags and formatted string. Uses the default logger.
func Warnf(fmt string, args ...interface{}) {
defaultLogger.logwithformat(WARN, nil, fmt, args...)
}
//Warn logs an WARN level message with no tags and provided arguments joined into a string. Uses the default logger.
func Warn(args ...interface{}) {
defaultLogger.log(WARN, nil, args...)
}
//InfoWithTagsf logs an INFO level message with the provided tags and formatted string. Uses the default logger.
func InfoWithTagsf(tags []string, fmt string, args ...interface{}) {
defaultLogger.logwithformat(INFO, tags, fmt, args...)
}
//InfoWithTags logs an INFO level message with the provided tags and provided arguments joined into a string. Uses the default logger.
func InfoWithTags(tags []string, args ...interface{}) {
defaultLogger.log(INFO, tags, args...)
}
//Infof logs an INFO level message with the no tags and formatted string. Uses the default logger.
func Infof(fmt string, args ...interface{}) {
defaultLogger.logwithformat(INFO, nil, fmt, args...)
}
//Info logs an INFO level message with no tags and provided arguments joined into a string. Uses the default logger.
func Info(args ...interface{}) {
defaultLogger.log(INFO, nil, args...)
}
//DebugWithTagsf logs an DEBUG level message with the provided tags and formatted string. Uses the default logger.
func DebugWithTagsf(tags []string, fmt string, args ...interface{}) {
defaultLogger.logwithformat(DEBUG, tags, fmt, args...)
}
//DebugWithTags logs an DEBUG level message with the provided tags and provided arguments joined into a string. Uses the default logger.
func DebugWithTags(tags []string, args ...interface{}) {
defaultLogger.log(DEBUG, tags, args...)
}
//Debugf logs an DEBUG level message with the no tags and formatted string. Uses the default logger.
func Debugf(fmt string, args ...interface{}) {
defaultLogger.logwithformat(DEBUG, nil, fmt, args...)
}
//Debug logs an DEBUG level message with no tags and provided arguments joined into a string. Uses the default logger.
func Debug(args ...interface{}) {
defaultLogger.log(DEBUG, nil, args...)
}
//VerboseWithTagsf logs an VERBOSE level message with the provided tags and formatted string. Uses the default logger.
//Verbose messages are not buffered
func VerboseWithTagsf(tags []string, fmt string, args ...interface{}) {
defaultLogger.logwithformat(VERBOSE, tags, fmt, args...)
}
//Verbosef logs an VERBOSE level message with the no tags and formatted string. Uses the default logger.
//Verbose messages are not buffered
func Verbosef(fmt string, args ...interface{}) {
defaultLogger.logwithformat(VERBOSE, nil, fmt, args...)
}