mirror of https://github.com/hashicorp/consul
67 lines
1.4 KiB
Go
67 lines
1.4 KiB
Go
|
package logdrop
|
||
|
|
||
|
import (
|
||
|
"context"
|
||
|
"github.com/hashicorp/go-hclog"
|
||
|
)
|
||
|
|
||
|
// SinkAdapter mimic the interface from hclog.SinkAdapter
|
||
|
//
|
||
|
//go:generate mockery --name SinkAdapter --inpackage
|
||
|
type SinkAdapter interface {
|
||
|
Accept(name string, level hclog.Level, msg string, args ...interface{})
|
||
|
}
|
||
|
|
||
|
type Log struct {
|
||
|
n string
|
||
|
s string
|
||
|
i []interface{}
|
||
|
l hclog.Level
|
||
|
}
|
||
|
|
||
|
type logDropSink struct {
|
||
|
sink SinkAdapter
|
||
|
logCh chan Log
|
||
|
name string
|
||
|
dropFn func(l Log)
|
||
|
}
|
||
|
|
||
|
// Accept consume a log and push it into a channel,
|
||
|
// if the channel is filled it will call dropFn
|
||
|
func (r *logDropSink) Accept(name string, level hclog.Level, msg string, args ...interface{}) {
|
||
|
r.pushLog(Log{n: name, l: level, s: msg, i: args})
|
||
|
}
|
||
|
|
||
|
func (r *logDropSink) pushLog(l Log) {
|
||
|
select {
|
||
|
case r.logCh <- l:
|
||
|
default:
|
||
|
r.dropFn(l)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func (r *logDropSink) logConsumer(ctx context.Context) {
|
||
|
for {
|
||
|
select {
|
||
|
case l := <-r.logCh:
|
||
|
r.sink.Accept(l.n, l.l, l.s, l.i)
|
||
|
case <-ctx.Done():
|
||
|
return
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// NewLogDropSink create a log SinkAdapter that wrap another SinkAdapter
|
||
|
// It also create a go routine for consuming logs, the given context need to be canceled
|
||
|
// to properly deallocate the SinkAdapter.
|
||
|
func NewLogDropSink(ctx context.Context, name string, depth int, sink SinkAdapter, dropFn func(l Log)) hclog.SinkAdapter {
|
||
|
r := &logDropSink{
|
||
|
sink: sink,
|
||
|
logCh: make(chan Log, depth),
|
||
|
name: name,
|
||
|
dropFn: dropFn,
|
||
|
}
|
||
|
go r.logConsumer(ctx)
|
||
|
return r
|
||
|
}
|