Integrate BACKBEAT SDK and resolve KACHING license validation
Major integrations and fixes: - Added BACKBEAT SDK integration for P2P operation timing - Implemented beat-aware status tracking for distributed operations - Added Docker secrets support for secure license management - Resolved KACHING license validation via HTTPS/TLS - Updated docker-compose configuration for clean stack deployment - Disabled rollback policies to prevent deployment failures - Added license credential storage (CHORUS-DEV-MULTI-001) Technical improvements: - BACKBEAT P2P operation tracking with phase management - Enhanced configuration system with file-based secrets - Improved error handling for license validation - Clean separation of KACHING and CHORUS deployment stacks 🤖 Generated with [Claude Code](https://claude.ai/code) Co-Authored-By: Claude <noreply@anthropic.com>
This commit is contained in:
418
vendor/github.com/libp2p/go-libp2p/p2p/host/eventbus/basic.go
generated
vendored
Normal file
418
vendor/github.com/libp2p/go-libp2p/p2p/host/eventbus/basic.go
generated
vendored
Normal file
@@ -0,0 +1,418 @@
|
||||
package eventbus
|
||||
|
||||
import (
|
||||
"errors"
|
||||
"fmt"
|
||||
"reflect"
|
||||
"sync"
|
||||
"sync/atomic"
|
||||
|
||||
"github.com/libp2p/go-libp2p/core/event"
|
||||
)
|
||||
|
||||
// /////////////////////
|
||||
// BUS
|
||||
|
||||
// basicBus is a type-based event delivery system
|
||||
type basicBus struct {
|
||||
lk sync.RWMutex
|
||||
nodes map[reflect.Type]*node
|
||||
wildcard *wildcardNode
|
||||
metricsTracer MetricsTracer
|
||||
}
|
||||
|
||||
var _ event.Bus = (*basicBus)(nil)
|
||||
|
||||
type emitter struct {
|
||||
n *node
|
||||
w *wildcardNode
|
||||
typ reflect.Type
|
||||
closed atomic.Bool
|
||||
dropper func(reflect.Type)
|
||||
metricsTracer MetricsTracer
|
||||
}
|
||||
|
||||
func (e *emitter) Emit(evt interface{}) error {
|
||||
if e.closed.Load() {
|
||||
return fmt.Errorf("emitter is closed")
|
||||
}
|
||||
|
||||
e.n.emit(evt)
|
||||
e.w.emit(evt)
|
||||
|
||||
if e.metricsTracer != nil {
|
||||
e.metricsTracer.EventEmitted(e.typ)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (e *emitter) Close() error {
|
||||
if !e.closed.CompareAndSwap(false, true) {
|
||||
return fmt.Errorf("closed an emitter more than once")
|
||||
}
|
||||
if e.n.nEmitters.Add(-1) == 0 {
|
||||
e.dropper(e.typ)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func NewBus(opts ...Option) event.Bus {
|
||||
bus := &basicBus{
|
||||
nodes: map[reflect.Type]*node{},
|
||||
wildcard: &wildcardNode{},
|
||||
}
|
||||
for _, opt := range opts {
|
||||
opt(bus)
|
||||
}
|
||||
return bus
|
||||
}
|
||||
|
||||
func (b *basicBus) withNode(typ reflect.Type, cb func(*node), async func(*node)) {
|
||||
b.lk.Lock()
|
||||
|
||||
n, ok := b.nodes[typ]
|
||||
if !ok {
|
||||
n = newNode(typ, b.metricsTracer)
|
||||
b.nodes[typ] = n
|
||||
}
|
||||
|
||||
n.lk.Lock()
|
||||
b.lk.Unlock()
|
||||
|
||||
cb(n)
|
||||
|
||||
if async == nil {
|
||||
n.lk.Unlock()
|
||||
} else {
|
||||
go func() {
|
||||
defer n.lk.Unlock()
|
||||
async(n)
|
||||
}()
|
||||
}
|
||||
}
|
||||
|
||||
func (b *basicBus) tryDropNode(typ reflect.Type) {
|
||||
b.lk.Lock()
|
||||
n, ok := b.nodes[typ]
|
||||
if !ok { // already dropped
|
||||
b.lk.Unlock()
|
||||
return
|
||||
}
|
||||
|
||||
n.lk.Lock()
|
||||
if n.nEmitters.Load() > 0 || len(n.sinks) > 0 {
|
||||
n.lk.Unlock()
|
||||
b.lk.Unlock()
|
||||
return // still in use
|
||||
}
|
||||
n.lk.Unlock()
|
||||
|
||||
delete(b.nodes, typ)
|
||||
b.lk.Unlock()
|
||||
}
|
||||
|
||||
type wildcardSub struct {
|
||||
ch chan interface{}
|
||||
w *wildcardNode
|
||||
metricsTracer MetricsTracer
|
||||
name string
|
||||
}
|
||||
|
||||
func (w *wildcardSub) Out() <-chan interface{} {
|
||||
return w.ch
|
||||
}
|
||||
|
||||
func (w *wildcardSub) Close() error {
|
||||
w.w.removeSink(w.ch)
|
||||
if w.metricsTracer != nil {
|
||||
w.metricsTracer.RemoveSubscriber(reflect.TypeOf(event.WildcardSubscription))
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
func (w *wildcardSub) Name() string {
|
||||
return w.name
|
||||
}
|
||||
|
||||
type namedSink struct {
|
||||
name string
|
||||
ch chan interface{}
|
||||
}
|
||||
|
||||
type sub struct {
|
||||
ch chan interface{}
|
||||
nodes []*node
|
||||
dropper func(reflect.Type)
|
||||
metricsTracer MetricsTracer
|
||||
name string
|
||||
}
|
||||
|
||||
func (s *sub) Name() string {
|
||||
return s.name
|
||||
}
|
||||
|
||||
func (s *sub) Out() <-chan interface{} {
|
||||
return s.ch
|
||||
}
|
||||
|
||||
func (s *sub) Close() error {
|
||||
go func() {
|
||||
// drain the event channel, will return when closed and drained.
|
||||
// this is necessary to unblock publishes to this channel.
|
||||
for range s.ch {
|
||||
}
|
||||
}()
|
||||
|
||||
for _, n := range s.nodes {
|
||||
n.lk.Lock()
|
||||
|
||||
for i := 0; i < len(n.sinks); i++ {
|
||||
if n.sinks[i].ch == s.ch {
|
||||
n.sinks[i], n.sinks[len(n.sinks)-1] = n.sinks[len(n.sinks)-1], nil
|
||||
n.sinks = n.sinks[:len(n.sinks)-1]
|
||||
|
||||
if s.metricsTracer != nil {
|
||||
s.metricsTracer.RemoveSubscriber(n.typ)
|
||||
}
|
||||
break
|
||||
}
|
||||
}
|
||||
|
||||
tryDrop := len(n.sinks) == 0 && n.nEmitters.Load() == 0
|
||||
|
||||
n.lk.Unlock()
|
||||
|
||||
if tryDrop {
|
||||
s.dropper(n.typ)
|
||||
}
|
||||
}
|
||||
close(s.ch)
|
||||
return nil
|
||||
}
|
||||
|
||||
var _ event.Subscription = (*sub)(nil)
|
||||
|
||||
// Subscribe creates new subscription. Failing to drain the channel will cause
|
||||
// publishers to get blocked. CancelFunc is guaranteed to return after last send
|
||||
// to the channel
|
||||
func (b *basicBus) Subscribe(evtTypes interface{}, opts ...event.SubscriptionOpt) (_ event.Subscription, err error) {
|
||||
settings := newSubSettings()
|
||||
for _, opt := range opts {
|
||||
if err := opt(&settings); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
}
|
||||
|
||||
if evtTypes == event.WildcardSubscription {
|
||||
out := &wildcardSub{
|
||||
ch: make(chan interface{}, settings.buffer),
|
||||
w: b.wildcard,
|
||||
metricsTracer: b.metricsTracer,
|
||||
name: settings.name,
|
||||
}
|
||||
b.wildcard.addSink(&namedSink{ch: out.ch, name: out.name})
|
||||
return out, nil
|
||||
}
|
||||
|
||||
types, ok := evtTypes.([]interface{})
|
||||
if !ok {
|
||||
types = []interface{}{evtTypes}
|
||||
}
|
||||
|
||||
if len(types) > 1 {
|
||||
for _, t := range types {
|
||||
if t == event.WildcardSubscription {
|
||||
return nil, fmt.Errorf("wildcard subscriptions must be started separately")
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
out := &sub{
|
||||
ch: make(chan interface{}, settings.buffer),
|
||||
nodes: make([]*node, len(types)),
|
||||
|
||||
dropper: b.tryDropNode,
|
||||
metricsTracer: b.metricsTracer,
|
||||
name: settings.name,
|
||||
}
|
||||
|
||||
for _, etyp := range types {
|
||||
if reflect.TypeOf(etyp).Kind() != reflect.Ptr {
|
||||
return nil, errors.New("subscribe called with non-pointer type")
|
||||
}
|
||||
}
|
||||
|
||||
for i, etyp := range types {
|
||||
typ := reflect.TypeOf(etyp)
|
||||
|
||||
b.withNode(typ.Elem(), func(n *node) {
|
||||
n.sinks = append(n.sinks, &namedSink{ch: out.ch, name: out.name})
|
||||
out.nodes[i] = n
|
||||
if b.metricsTracer != nil {
|
||||
b.metricsTracer.AddSubscriber(typ.Elem())
|
||||
}
|
||||
}, func(n *node) {
|
||||
if n.keepLast {
|
||||
l := n.last
|
||||
if l == nil {
|
||||
return
|
||||
}
|
||||
out.ch <- l
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
return out, nil
|
||||
}
|
||||
|
||||
// Emitter creates new emitter
|
||||
//
|
||||
// eventType accepts typed nil pointers, and uses the type information to
|
||||
// select output type
|
||||
//
|
||||
// Example:
|
||||
// emit, err := eventbus.Emitter(new(EventT))
|
||||
// defer emit.Close() // MUST call this after being done with the emitter
|
||||
//
|
||||
// emit(EventT{})
|
||||
func (b *basicBus) Emitter(evtType interface{}, opts ...event.EmitterOpt) (e event.Emitter, err error) {
|
||||
if evtType == event.WildcardSubscription {
|
||||
return nil, fmt.Errorf("illegal emitter for wildcard subscription")
|
||||
}
|
||||
|
||||
var settings emitterSettings
|
||||
for _, opt := range opts {
|
||||
if err := opt(&settings); err != nil {
|
||||
return nil, err
|
||||
}
|
||||
}
|
||||
|
||||
typ := reflect.TypeOf(evtType)
|
||||
if typ.Kind() != reflect.Ptr {
|
||||
return nil, errors.New("emitter called with non-pointer type")
|
||||
}
|
||||
typ = typ.Elem()
|
||||
|
||||
b.withNode(typ, func(n *node) {
|
||||
n.nEmitters.Add(1)
|
||||
n.keepLast = n.keepLast || settings.makeStateful
|
||||
e = &emitter{n: n, typ: typ, dropper: b.tryDropNode, w: b.wildcard, metricsTracer: b.metricsTracer}
|
||||
}, nil)
|
||||
return
|
||||
}
|
||||
|
||||
// GetAllEventTypes returns all the event types that this bus has emitters
|
||||
// or subscribers for.
|
||||
func (b *basicBus) GetAllEventTypes() []reflect.Type {
|
||||
b.lk.RLock()
|
||||
defer b.lk.RUnlock()
|
||||
|
||||
types := make([]reflect.Type, 0, len(b.nodes))
|
||||
for t := range b.nodes {
|
||||
types = append(types, t)
|
||||
}
|
||||
return types
|
||||
}
|
||||
|
||||
// /////////////////////
|
||||
// NODE
|
||||
|
||||
type wildcardNode struct {
|
||||
sync.RWMutex
|
||||
nSinks atomic.Int32
|
||||
sinks []*namedSink
|
||||
metricsTracer MetricsTracer
|
||||
}
|
||||
|
||||
func (n *wildcardNode) addSink(sink *namedSink) {
|
||||
n.nSinks.Add(1) // ok to do outside the lock
|
||||
n.Lock()
|
||||
n.sinks = append(n.sinks, sink)
|
||||
n.Unlock()
|
||||
|
||||
if n.metricsTracer != nil {
|
||||
n.metricsTracer.AddSubscriber(reflect.TypeOf(event.WildcardSubscription))
|
||||
}
|
||||
}
|
||||
|
||||
func (n *wildcardNode) removeSink(ch chan interface{}) {
|
||||
n.nSinks.Add(-1) // ok to do outside the lock
|
||||
n.Lock()
|
||||
for i := 0; i < len(n.sinks); i++ {
|
||||
if n.sinks[i].ch == ch {
|
||||
n.sinks[i], n.sinks[len(n.sinks)-1] = n.sinks[len(n.sinks)-1], nil
|
||||
n.sinks = n.sinks[:len(n.sinks)-1]
|
||||
break
|
||||
}
|
||||
}
|
||||
n.Unlock()
|
||||
}
|
||||
|
||||
func (n *wildcardNode) emit(evt interface{}) {
|
||||
if n.nSinks.Load() == 0 {
|
||||
return
|
||||
}
|
||||
|
||||
n.RLock()
|
||||
for _, sink := range n.sinks {
|
||||
|
||||
// Sending metrics before sending on channel allows us to
|
||||
// record channel full events before blocking
|
||||
sendSubscriberMetrics(n.metricsTracer, sink)
|
||||
|
||||
sink.ch <- evt
|
||||
}
|
||||
n.RUnlock()
|
||||
}
|
||||
|
||||
type node struct {
|
||||
// Note: make sure to NEVER lock basicBus.lk when this lock is held
|
||||
lk sync.Mutex
|
||||
|
||||
typ reflect.Type
|
||||
|
||||
// emitter ref count
|
||||
nEmitters atomic.Int32
|
||||
|
||||
keepLast bool
|
||||
last interface{}
|
||||
|
||||
sinks []*namedSink
|
||||
metricsTracer MetricsTracer
|
||||
}
|
||||
|
||||
func newNode(typ reflect.Type, metricsTracer MetricsTracer) *node {
|
||||
return &node{
|
||||
typ: typ,
|
||||
metricsTracer: metricsTracer,
|
||||
}
|
||||
}
|
||||
|
||||
func (n *node) emit(evt interface{}) {
|
||||
typ := reflect.TypeOf(evt)
|
||||
if typ != n.typ {
|
||||
panic(fmt.Sprintf("Emit called with wrong type. expected: %s, got: %s", n.typ, typ))
|
||||
}
|
||||
|
||||
n.lk.Lock()
|
||||
if n.keepLast {
|
||||
n.last = evt
|
||||
}
|
||||
|
||||
for _, sink := range n.sinks {
|
||||
|
||||
// Sending metrics before sending on channel allows us to
|
||||
// record channel full events before blocking
|
||||
sendSubscriberMetrics(n.metricsTracer, sink)
|
||||
sink.ch <- evt
|
||||
}
|
||||
n.lk.Unlock()
|
||||
}
|
||||
|
||||
func sendSubscriberMetrics(metricsTracer MetricsTracer, sink *namedSink) {
|
||||
if metricsTracer != nil {
|
||||
metricsTracer.SubscriberQueueLength(sink.name, len(sink.ch)+1)
|
||||
metricsTracer.SubscriberQueueFull(sink.name, len(sink.ch)+1 >= cap(sink.ch))
|
||||
metricsTracer.SubscriberEventQueued(sink.name)
|
||||
}
|
||||
}
|
||||
164
vendor/github.com/libp2p/go-libp2p/p2p/host/eventbus/basic_metrics.go
generated
vendored
Normal file
164
vendor/github.com/libp2p/go-libp2p/p2p/host/eventbus/basic_metrics.go
generated
vendored
Normal file
@@ -0,0 +1,164 @@
|
||||
package eventbus
|
||||
|
||||
import (
|
||||
"reflect"
|
||||
"strings"
|
||||
|
||||
"github.com/libp2p/go-libp2p/p2p/metricshelper"
|
||||
|
||||
"github.com/prometheus/client_golang/prometheus"
|
||||
)
|
||||
|
||||
const metricNamespace = "libp2p_eventbus"
|
||||
|
||||
var (
|
||||
eventsEmitted = prometheus.NewCounterVec(
|
||||
prometheus.CounterOpts{
|
||||
Namespace: metricNamespace,
|
||||
Name: "events_emitted_total",
|
||||
Help: "Events Emitted",
|
||||
},
|
||||
[]string{"event"},
|
||||
)
|
||||
totalSubscribers = prometheus.NewGaugeVec(
|
||||
prometheus.GaugeOpts{
|
||||
Namespace: metricNamespace,
|
||||
Name: "subscribers_total",
|
||||
Help: "Number of subscribers for an event type",
|
||||
},
|
||||
[]string{"event"},
|
||||
)
|
||||
subscriberQueueLength = prometheus.NewGaugeVec(
|
||||
prometheus.GaugeOpts{
|
||||
Namespace: metricNamespace,
|
||||
Name: "subscriber_queue_length",
|
||||
Help: "Subscriber queue length",
|
||||
},
|
||||
[]string{"subscriber_name"},
|
||||
)
|
||||
subscriberQueueFull = prometheus.NewGaugeVec(
|
||||
prometheus.GaugeOpts{
|
||||
Namespace: metricNamespace,
|
||||
Name: "subscriber_queue_full",
|
||||
Help: "Subscriber Queue completely full",
|
||||
},
|
||||
[]string{"subscriber_name"},
|
||||
)
|
||||
subscriberEventQueued = prometheus.NewCounterVec(
|
||||
prometheus.CounterOpts{
|
||||
Namespace: metricNamespace,
|
||||
Name: "subscriber_event_queued",
|
||||
Help: "Event Queued for subscriber",
|
||||
},
|
||||
[]string{"subscriber_name"},
|
||||
)
|
||||
collectors = []prometheus.Collector{
|
||||
eventsEmitted,
|
||||
totalSubscribers,
|
||||
subscriberQueueLength,
|
||||
subscriberQueueFull,
|
||||
subscriberEventQueued,
|
||||
}
|
||||
)
|
||||
|
||||
// MetricsTracer tracks metrics for the eventbus subsystem
|
||||
type MetricsTracer interface {
|
||||
|
||||
// EventEmitted counts the total number of events grouped by event type
|
||||
EventEmitted(typ reflect.Type)
|
||||
|
||||
// AddSubscriber adds a subscriber for the event type
|
||||
AddSubscriber(typ reflect.Type)
|
||||
|
||||
// RemoveSubscriber removes a subscriber for the event type
|
||||
RemoveSubscriber(typ reflect.Type)
|
||||
|
||||
// SubscriberQueueLength is the length of the subscribers channel
|
||||
SubscriberQueueLength(name string, n int)
|
||||
|
||||
// SubscriberQueueFull tracks whether a subscribers channel if full
|
||||
SubscriberQueueFull(name string, isFull bool)
|
||||
|
||||
// SubscriberEventQueued counts the total number of events grouped by subscriber
|
||||
SubscriberEventQueued(name string)
|
||||
}
|
||||
|
||||
type metricsTracer struct{}
|
||||
|
||||
var _ MetricsTracer = &metricsTracer{}
|
||||
|
||||
type metricsTracerSetting struct {
|
||||
reg prometheus.Registerer
|
||||
}
|
||||
|
||||
type MetricsTracerOption func(*metricsTracerSetting)
|
||||
|
||||
func WithRegisterer(reg prometheus.Registerer) MetricsTracerOption {
|
||||
return func(s *metricsTracerSetting) {
|
||||
if reg != nil {
|
||||
s.reg = reg
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func NewMetricsTracer(opts ...MetricsTracerOption) MetricsTracer {
|
||||
setting := &metricsTracerSetting{reg: prometheus.DefaultRegisterer}
|
||||
for _, opt := range opts {
|
||||
opt(setting)
|
||||
}
|
||||
metricshelper.RegisterCollectors(setting.reg, collectors...)
|
||||
return &metricsTracer{}
|
||||
}
|
||||
|
||||
func (m *metricsTracer) EventEmitted(typ reflect.Type) {
|
||||
tags := metricshelper.GetStringSlice()
|
||||
defer metricshelper.PutStringSlice(tags)
|
||||
|
||||
*tags = append(*tags, strings.TrimPrefix(typ.String(), "event."))
|
||||
eventsEmitted.WithLabelValues(*tags...).Inc()
|
||||
}
|
||||
|
||||
func (m *metricsTracer) AddSubscriber(typ reflect.Type) {
|
||||
tags := metricshelper.GetStringSlice()
|
||||
defer metricshelper.PutStringSlice(tags)
|
||||
|
||||
*tags = append(*tags, strings.TrimPrefix(typ.String(), "event."))
|
||||
totalSubscribers.WithLabelValues(*tags...).Inc()
|
||||
}
|
||||
|
||||
func (m *metricsTracer) RemoveSubscriber(typ reflect.Type) {
|
||||
tags := metricshelper.GetStringSlice()
|
||||
defer metricshelper.PutStringSlice(tags)
|
||||
|
||||
*tags = append(*tags, strings.TrimPrefix(typ.String(), "event."))
|
||||
totalSubscribers.WithLabelValues(*tags...).Dec()
|
||||
}
|
||||
|
||||
func (m *metricsTracer) SubscriberQueueLength(name string, n int) {
|
||||
tags := metricshelper.GetStringSlice()
|
||||
defer metricshelper.PutStringSlice(tags)
|
||||
|
||||
*tags = append(*tags, name)
|
||||
subscriberQueueLength.WithLabelValues(*tags...).Set(float64(n))
|
||||
}
|
||||
|
||||
func (m *metricsTracer) SubscriberQueueFull(name string, isFull bool) {
|
||||
tags := metricshelper.GetStringSlice()
|
||||
defer metricshelper.PutStringSlice(tags)
|
||||
|
||||
*tags = append(*tags, name)
|
||||
observer := subscriberQueueFull.WithLabelValues(*tags...)
|
||||
if isFull {
|
||||
observer.Set(1)
|
||||
} else {
|
||||
observer.Set(0)
|
||||
}
|
||||
}
|
||||
|
||||
func (m *metricsTracer) SubscriberEventQueued(name string) {
|
||||
tags := metricshelper.GetStringSlice()
|
||||
defer metricshelper.PutStringSlice(tags)
|
||||
|
||||
*tags = append(*tags, name)
|
||||
subscriberEventQueued.WithLabelValues(*tags...).Inc()
|
||||
}
|
||||
79
vendor/github.com/libp2p/go-libp2p/p2p/host/eventbus/opts.go
generated
vendored
Normal file
79
vendor/github.com/libp2p/go-libp2p/p2p/host/eventbus/opts.go
generated
vendored
Normal file
@@ -0,0 +1,79 @@
|
||||
package eventbus
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"runtime"
|
||||
"strings"
|
||||
"sync/atomic"
|
||||
)
|
||||
|
||||
type subSettings struct {
|
||||
buffer int
|
||||
name string
|
||||
}
|
||||
|
||||
var subCnt atomic.Int64
|
||||
|
||||
var subSettingsDefault = subSettings{
|
||||
buffer: 16,
|
||||
}
|
||||
|
||||
// newSubSettings returns the settings for a new subscriber
|
||||
// The default naming strategy is sub-<fileName>-L<lineNum>
|
||||
func newSubSettings() subSettings {
|
||||
settings := subSettingsDefault
|
||||
_, file, line, ok := runtime.Caller(2) // skip=1 is eventbus.Subscriber
|
||||
if ok {
|
||||
file = strings.TrimPrefix(file, "github.com/")
|
||||
// remove the version number from the path, for example
|
||||
// go-libp2p-package@v0.x.y-some-hash-123/file.go will be shortened go go-libp2p-package/file.go
|
||||
if idx1 := strings.Index(file, "@"); idx1 != -1 {
|
||||
if idx2 := strings.Index(file[idx1:], "/"); idx2 != -1 {
|
||||
file = file[:idx1] + file[idx1+idx2:]
|
||||
}
|
||||
}
|
||||
settings.name = fmt.Sprintf("%s-L%d", file, line)
|
||||
} else {
|
||||
settings.name = fmt.Sprintf("subscriber-%d", subCnt.Add(1))
|
||||
}
|
||||
return settings
|
||||
}
|
||||
|
||||
func BufSize(n int) func(interface{}) error {
|
||||
return func(s interface{}) error {
|
||||
s.(*subSettings).buffer = n
|
||||
return nil
|
||||
}
|
||||
}
|
||||
|
||||
func Name(name string) func(interface{}) error {
|
||||
return func(s interface{}) error {
|
||||
s.(*subSettings).name = name
|
||||
return nil
|
||||
}
|
||||
}
|
||||
|
||||
type emitterSettings struct {
|
||||
makeStateful bool
|
||||
}
|
||||
|
||||
// Stateful is an Emitter option which makes the eventbus channel
|
||||
// 'remember' last event sent, and when a new subscriber joins the
|
||||
// bus, the remembered event is immediately sent to the subscription
|
||||
// channel.
|
||||
//
|
||||
// This allows to provide state tracking for dynamic systems, and/or
|
||||
// allows new subscribers to verify that there are Emitters on the channel
|
||||
func Stateful(s interface{}) error {
|
||||
s.(*emitterSettings).makeStateful = true
|
||||
return nil
|
||||
}
|
||||
|
||||
type Option func(*basicBus)
|
||||
|
||||
func WithMetricsTracer(metricsTracer MetricsTracer) Option {
|
||||
return func(bus *basicBus) {
|
||||
bus.metricsTracer = metricsTracer
|
||||
bus.wildcard.metricsTracer = metricsTracer
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user