package main import ( "log" "sync" ) type SyncFilter[MsgType any] interface { Then(what SyncFilter[MsgType]) SyncFilter[MsgType] Call(msg MsgType) } type SyncFilterImpl[MsgType any] struct { next SyncFilter[MsgType] } func (impl *SyncFilterImpl[MsgType]) Then(what SyncFilter[MsgType]) SyncFilter[MsgType] { impl.next = what return what } func (impl *SyncFilterImpl[MsgType]) CallNext(msg MsgType) { if impl.next != nil { impl.next.Call(msg) } } type CollectFromChannel struct{ SyncFilterImpl[GenericMessage] } func (collect *CollectFromChannel) Call(msg GenericMessage) { collect.CallNext(msg) } func (collect CollectFromChannel) Collect(events <-chan GenericMessage, wg *sync.WaitGroup) { wg.Add(1) go func() { defer wg.Done() for e := range events { collect.Call(e) } }() } type ThrottleSync struct { SyncFilterImpl[GenericMessage] events chan GenericMessage } func MakeThrottleSync(sleeper Sleeper, logger *log.Logger, wg *sync.WaitGroup) *ThrottleSync { events := make(chan GenericMessage) throttle := ThrottleSync{SyncFilterImpl[GenericMessage]{}, events} wg.Add(1) go func() { timeoutEvents := make(chan interface{}) var currentEvent *GenericMessage = nil loop: for { select { case ev, ok := <-events: if !ok { break loop } if currentEvent == nil { logger.Print("Waiting for more messages to arrive before sending...") sleeper.Sleep(timeoutEvents) } currentEvent = appendToGenericMessage(currentEvent, &ev) case <-timeoutEvents: logger.Print("Time's up, sending all messages we've got for now.") throttle.CallNext(*currentEvent) currentEvent = nil } } // If anything is left to be sent, send it now if currentEvent != nil { throttle.CallNext(*currentEvent) } wg.Done() logger.Print("Throttling goroutine finishing") }() return &throttle } func (throttle *ThrottleSync) Close() { close(throttle.events) } func (throttle *ThrottleSync) Call(msg GenericMessage) { throttle.events <- msg }