From cd7fd0485cf69e638cb036778006b231d285ed5b Mon Sep 17 00:00:00 2001 From: MarvinJWendt Date: Thu, 12 Sep 2024 20:02:10 +0200 Subject: [PATCH 1/2] feat!: performance improvements and better thread safety --- doc.go | 7 ++-- event.go | 100 ++++++++++++++++++++++++++++++------------------------- 2 files changed, 59 insertions(+), 48 deletions(-) diff --git a/doc.go b/doc.go index 570d61c..7aa9414 100644 --- a/doc.go +++ b/doc.go @@ -1,4 +1,5 @@ -/* -Package event provides a generic event system for Go. -*/ +// Package event provides a generic and thread-safe event system for Go. +// It allows multiple listeners to subscribe to events carrying data of any type. +// Listeners can be added and notified when events are triggered, and the event +// can be closed to prevent further operations. package event diff --git a/event.go b/event.go index bb12ccf..72e2382 100644 --- a/event.go +++ b/event.go @@ -1,74 +1,84 @@ package event -import "sync" +import ( + "errors" + "sync" +) -// Event represents an event system that can handle multiple listeners. +// ErrEventClosed is returned when an operation is attempted on a closed event. +var ErrEventClosed = errors.New("event is closed") + +// Event represents a generic, thread-safe event system that can handle multiple listeners. +// The type parameter T specifies the type of data that the event carries when triggered. type Event[T any] struct { - listeners []chan T - mu sync.Mutex + listeners []func(T) + mu sync.RWMutex closed bool } -// New creates a new event. +// New creates and returns a new Event instance for the specified type T. func New[T any]() *Event[T] { - // Create a new event - return &Event[T]{ - listeners: []chan T{}, - } + return &Event[T]{} } -// Trigger triggers the event and notifies all listeners. -func (e *Event[T]) Trigger(value T) { - e.mu.Lock() - defer e.mu.Unlock() +// Trigger notifies all registered listeners by invoking their callback functions with the provided value. +// It runs each listener in a separate goroutine and waits for all listeners to complete. +// Returns ErrEventClosed if the event has been closed. +func (e *Event[T]) Trigger(value T) error { + e.mu.RLock() + if e.closed { + e.mu.RUnlock() + return ErrEventClosed + } + + // Copy the listeners to avoid holding the lock during execution. + // This ensures that triggering the event is thread-safe even if listeners are added or removed concurrently. + listeners := make([]func(T), len(e.listeners)) + copy(listeners, e.listeners) + e.mu.RUnlock() + + var wg sync.WaitGroup + for _, listener := range listeners { + wg.Add(1) - for _, listener := range e.listeners { - go func(l chan T) { - if !e.closed { - l <- value - } + go func(f func(T)) { + defer wg.Done() + f(value) }(listener) } + + wg.Wait() + + return nil } -// Listen gets called when the event is triggered. -func (e *Event[T]) Listen(f func(T)) { - // Check if the event is closed - if e.closed { - return - } +// Listen registers a new listener callback function for the event. +// The listener will be invoked with the event's data whenever Trigger is called. +// Returns ErrEventClosed if the event has been closed. +func (e *Event[T]) Listen(f func(T)) error { + e.mu.Lock() + defer e.mu.Unlock() - // Create listener slice if it doesn't exist - if e.listeners == nil { - e.listeners = []chan T{} + if e.closed { + return ErrEventClosed } - // Create a new channel - ch := make(chan T) + e.listeners = append(e.listeners, f) - e.mu.Lock() - e.listeners = append(e.listeners, ch) - e.mu.Unlock() - - go func() { - for v := range ch { - if !e.closed { - f(v) - } - } - }() + return nil } -// Close closes the event and all its listeners. -// After calling this method, the event can't be used anymore and new listeners can't be added. +// Close closes the event system, preventing any new listeners from being added or events from being triggered. +// After calling Close, any subsequent calls to Trigger or Listen will return ErrEventClosed. +// Existing listeners are removed, and resources are cleaned up. func (e *Event[T]) Close() { e.mu.Lock() defer e.mu.Unlock() - for _, listener := range e.listeners { - close(listener) + if e.closed { + return } - e.listeners = nil e.closed = true + e.listeners = nil // Release references to listener functions } From f758c62ddbb1738a2418e0bd18b59faa8954c77f Mon Sep 17 00:00:00 2001 From: MarvinJWendt Date: Thu, 12 Sep 2024 22:37:56 +0200 Subject: [PATCH 2/2] ci: exclude errcheck in test files --- .golangci.yml | 1 + 1 file changed, 1 insertion(+) diff --git a/.golangci.yml b/.golangci.yml index 802fd28..f2444da 100644 --- a/.golangci.yml +++ b/.golangci.yml @@ -192,3 +192,4 @@ issues: - noctx - funlen - dupl + - errcheck