garm/database/watcher/consumer.go
Gabriel Adrian Samfira b51683f1ae Add some tests
Signed-off-by: Gabriel Adrian Samfira <gsamfira@cloudbasesolutions.com>
2024-06-17 19:42:50 +00:00

82 lines
1.4 KiB
Go

package watcher
import (
"context"
"log/slog"
"sync"
"time"
"github.com/cloudbase/garm/database/common"
)
type consumer struct {
messages chan common.ChangePayload
filters []common.PayloadFilterFunc
id string
mux sync.Mutex
closed bool
quit chan struct{}
ctx context.Context
}
func (w *consumer) SetFilters(filters ...common.PayloadFilterFunc) {
w.mux.Lock()
defer w.mux.Unlock()
w.filters = filters
}
func (w *consumer) Watch() <-chan common.ChangePayload {
return w.messages
}
func (w *consumer) Close() {
w.mux.Lock()
defer w.mux.Unlock()
if w.closed {
return
}
close(w.messages)
close(w.quit)
w.closed = true
}
func (w *consumer) IsClosed() bool {
w.mux.Lock()
defer w.mux.Unlock()
return w.closed
}
func (w *consumer) Send(payload common.ChangePayload) {
w.mux.Lock()
defer w.mux.Unlock()
if w.closed {
return
}
if len(w.filters) > 0 {
shouldSend := true
for _, filter := range w.filters {
if !filter(payload) {
shouldSend = false
break
}
}
if !shouldSend {
return
}
}
slog.DebugContext(w.ctx, "sending payload")
select {
case <-w.quit:
slog.DebugContext(w.ctx, "consumer is closed")
case <-w.ctx.Done():
slog.DebugContext(w.ctx, "consumer is closed")
case <-time.After(1 * time.Second):
slog.DebugContext(w.ctx, "timeout trying to send payload", "payload", payload)
case w.messages <- payload:
}
}