2020-09-28 18:52:31 -04:00
|
|
|
package subscribe
|
|
|
|
|
|
|
|
import (
|
|
|
|
"sync"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/hashicorp/go-uuid"
|
|
|
|
|
|
|
|
"github.com/hashicorp/consul/agent/consul/stream"
|
|
|
|
"github.com/hashicorp/consul/proto/pbsubscribe"
|
|
|
|
)
|
|
|
|
|
|
|
|
// streamID is used in logs as a unique identifier for a subscription. The value
|
|
|
|
// is created lazily on the first call to String() so that we do not call it
|
|
|
|
// if trace logging is disabled.
|
|
|
|
// If a random UUID can not be created, defaults to the current time formatted
|
|
|
|
// as RFC3339Nano.
|
|
|
|
//
|
|
|
|
// TODO(banks) it might be nice one day to replace this with OpenTracing ID
|
|
|
|
// if one is set etc. but probably pointless until we support that properly
|
|
|
|
// in other places so it's actually propagated properly. For now this just
|
|
|
|
// makes lifetime of a stream more traceable in our regular server logs for
|
|
|
|
// debugging/dev.
|
|
|
|
type streamID struct {
|
|
|
|
once sync.Once
|
|
|
|
id string
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *streamID) String() string {
|
|
|
|
s.once.Do(func() {
|
|
|
|
var err error
|
|
|
|
s.id, err = uuid.GenerateUUID()
|
|
|
|
if err != nil {
|
|
|
|
s.id = time.Now().Format(time.RFC3339Nano)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
return s.id
|
|
|
|
}
|
|
|
|
|
2020-11-05 14:19:10 -05:00
|
|
|
func newLoggerForRequest(l Logger, req *pbsubscribe.SubscribeRequest) Logger {
|
|
|
|
return l.With(
|
2020-09-28 18:52:31 -04:00
|
|
|
"topic", req.Topic.String(),
|
|
|
|
"dc", req.Datacenter,
|
|
|
|
"key", req.Key,
|
2020-11-05 14:19:10 -05:00
|
|
|
"namespace", req.Namespace,
|
2020-09-28 18:52:31 -04:00
|
|
|
"index", req.Index,
|
|
|
|
"stream_id", &streamID{})
|
|
|
|
}
|
|
|
|
|
|
|
|
type eventLogger struct {
|
|
|
|
logger Logger
|
|
|
|
snapshotDone bool
|
|
|
|
count uint64
|
|
|
|
}
|
|
|
|
|
2020-10-05 12:38:38 -04:00
|
|
|
func (l *eventLogger) Trace(e stream.Event) {
|
2020-09-28 18:52:31 -04:00
|
|
|
switch {
|
2020-10-05 12:38:38 -04:00
|
|
|
case e.IsEndOfSnapshot():
|
2020-09-28 18:52:31 -04:00
|
|
|
l.snapshotDone = true
|
2020-10-05 12:38:38 -04:00
|
|
|
l.logger.Trace("snapshot complete", "index", e.Index, "sent", l.count)
|
2020-11-16 14:20:07 -05:00
|
|
|
return
|
2020-10-05 12:38:38 -04:00
|
|
|
case e.IsNewSnapshotToFollow():
|
|
|
|
l.logger.Trace("starting new snapshot", "sent", l.count)
|
2020-10-02 13:55:41 -04:00
|
|
|
return
|
2020-09-28 18:52:31 -04:00
|
|
|
}
|
|
|
|
|
2020-11-16 14:20:07 -05:00
|
|
|
size := 1
|
|
|
|
if l, ok := e.Payload.(length); ok {
|
|
|
|
size = l.Len()
|
|
|
|
}
|
|
|
|
l.logger.Trace("sending events", "index", e.Index, "sent", l.count, "batch_size", size)
|
|
|
|
l.count += uint64(size)
|
|
|
|
}
|
|
|
|
|
|
|
|
type length interface {
|
|
|
|
Len() int
|
2020-09-28 18:52:31 -04:00
|
|
|
}
|