package async_test import ( "context" "sync" "sync/atomic" "testing" "time" "github.com/prysmaticlabs/prysm/v5/async" "github.com/prysmaticlabs/prysm/v5/testing/assert" "github.com/prysmaticlabs/prysm/v5/testing/require" "github.com/prysmaticlabs/prysm/v5/testing/util" ) func TestDebounce_NoEvents(t *testing.T) { eventsChan := make(chan interface{}, 100) ctx, cancel := context.WithCancel(context.Background()) interval := time.Second timesHandled := int32(0) wg := &sync.WaitGroup{} wg.Add(1) go func() { time.AfterFunc(interval, func() { cancel() }) }() go func() { async.Debounce(ctx, interval, eventsChan, func(event interface{}) { atomic.AddInt32(×Handled, 1) }) wg.Done() }() if util.WaitTimeout(wg, interval*2) { t.Fatalf("Test should have exited by now, timed out") } assert.Equal(t, int32(0), atomic.LoadInt32(×Handled), "Wrong number of handled calls") } func TestDebounce_CtxClosing(t *testing.T) { eventsChan := make(chan interface{}, 100) ctx, cancel := context.WithCancel(context.Background()) interval := time.Second timesHandled := int32(0) wg := &sync.WaitGroup{} wg.Add(1) go func() { ticker := time.NewTicker(time.Millisecond * 100) defer ticker.Stop() for { select { case <-ctx.Done(): return case <-ticker.C: eventsChan <- struct{}{} } } }() go func() { time.AfterFunc(interval, func() { cancel() }) }() go func() { async.Debounce(ctx, interval, eventsChan, func(event interface{}) { atomic.AddInt32(×Handled, 1) }) wg.Done() }() if util.WaitTimeout(wg, interval*2) { t.Fatalf("Test should have exited by now, timed out") } assert.Equal(t, int32(0), atomic.LoadInt32(×Handled), "Wrong number of handled calls") } func TestDebounce_SingleHandlerInvocation(t *testing.T) { eventsChan := make(chan interface{}, 100) ctx, cancel := context.WithCancel(context.Background()) interval := time.Second timesHandled := int32(0) go async.Debounce(ctx, interval, eventsChan, func(event interface{}) { atomic.AddInt32(×Handled, 1) }) for i := 0; i < 100; i++ { eventsChan <- struct{}{} } // We should expect 100 rapid fire changes to only have caused // 1 handler to trigger after the debouncing period. time.Sleep(interval * 2) assert.Equal(t, int32(1), atomic.LoadInt32(×Handled), "Wrong number of handled calls") cancel() } func TestDebounce_MultipleHandlerInvocation(t *testing.T) { eventsChan := make(chan interface{}, 100) ctx, cancel := context.WithCancel(context.Background()) interval := time.Second timesHandled := int32(0) go async.Debounce(ctx, interval, eventsChan, func(event interface{}) { atomic.AddInt32(×Handled, 1) }) for i := 0; i < 100; i++ { eventsChan <- struct{}{} } require.Equal(t, int32(0), atomic.LoadInt32(×Handled), "Events must prevent from handler execution") // By this time the first event should be triggered. time.Sleep(2 * time.Second) assert.Equal(t, int32(1), atomic.LoadInt32(×Handled), "Wrong number of handled calls") // Second event. eventsChan <- struct{}{} time.Sleep(2 * time.Second) assert.Equal(t, int32(2), atomic.LoadInt32(×Handled), "Wrong number of handled calls") cancel() }