224 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			Go
		
	
	
			
		
		
	
	
			224 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			Go
		
	
	
package notifications
 | 
						|
 | 
						|
import (
 | 
						|
	"fmt"
 | 
						|
	"math/rand"
 | 
						|
	"sync"
 | 
						|
	"time"
 | 
						|
 | 
						|
	"github.com/Sirupsen/logrus"
 | 
						|
 | 
						|
	"testing"
 | 
						|
)
 | 
						|
 | 
						|
func TestBroadcaster(t *testing.T) {
 | 
						|
	const nEvents = 1000
 | 
						|
	var sinks []Sink
 | 
						|
 | 
						|
	for i := 0; i < 10; i++ {
 | 
						|
		sinks = append(sinks, &testSink{})
 | 
						|
	}
 | 
						|
 | 
						|
	b := NewBroadcaster(sinks...)
 | 
						|
 | 
						|
	var block []Event
 | 
						|
	var wg sync.WaitGroup
 | 
						|
	for i := 1; i <= nEvents; i++ {
 | 
						|
		block = append(block, createTestEvent("push", "library/test", "blob"))
 | 
						|
 | 
						|
		if i%10 == 0 && i > 0 {
 | 
						|
			wg.Add(1)
 | 
						|
			go func(block ...Event) {
 | 
						|
				if err := b.Write(block...); err != nil {
 | 
						|
					t.Fatalf("error writing block of length %d: %v", len(block), err)
 | 
						|
				}
 | 
						|
				wg.Done()
 | 
						|
			}(block...)
 | 
						|
 | 
						|
			block = nil
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	wg.Wait() // Wait until writes complete
 | 
						|
	checkClose(t, b)
 | 
						|
 | 
						|
	// Iterate through the sinks and check that they all have the expected length.
 | 
						|
	for _, sink := range sinks {
 | 
						|
		ts := sink.(*testSink)
 | 
						|
		ts.mu.Lock()
 | 
						|
		defer ts.mu.Unlock()
 | 
						|
 | 
						|
		if len(ts.events) != nEvents {
 | 
						|
			t.Fatalf("not all events ended up in testsink: len(testSink) == %d, not %d", len(ts.events), nEvents)
 | 
						|
		}
 | 
						|
 | 
						|
		if !ts.closed {
 | 
						|
			t.Fatalf("sink should have been closed")
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
}
 | 
						|
 | 
						|
func TestEventQueue(t *testing.T) {
 | 
						|
	const nevents = 1000
 | 
						|
	var ts testSink
 | 
						|
	metrics := newSafeMetrics()
 | 
						|
	eq := newEventQueue(
 | 
						|
		// delayed sync simulates destination slower than channel comms
 | 
						|
		&delayedSink{
 | 
						|
			Sink:  &ts,
 | 
						|
			delay: time.Millisecond * 1,
 | 
						|
		}, metrics.eventQueueListener())
 | 
						|
 | 
						|
	var wg sync.WaitGroup
 | 
						|
	var block []Event
 | 
						|
	for i := 1; i <= nevents; i++ {
 | 
						|
		block = append(block, createTestEvent("push", "library/test", "blob"))
 | 
						|
		if i%10 == 0 && i > 0 {
 | 
						|
			wg.Add(1)
 | 
						|
			go func(block ...Event) {
 | 
						|
				if err := eq.Write(block...); err != nil {
 | 
						|
					t.Fatalf("error writing event block: %v", err)
 | 
						|
				}
 | 
						|
				wg.Done()
 | 
						|
			}(block...)
 | 
						|
 | 
						|
			block = nil
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	wg.Wait()
 | 
						|
	checkClose(t, eq)
 | 
						|
 | 
						|
	ts.mu.Lock()
 | 
						|
	defer ts.mu.Unlock()
 | 
						|
	metrics.Lock()
 | 
						|
	defer metrics.Unlock()
 | 
						|
 | 
						|
	if len(ts.events) != nevents {
 | 
						|
		t.Fatalf("events did not make it to the sink: %d != %d", len(ts.events), 1000)
 | 
						|
	}
 | 
						|
 | 
						|
	if !ts.closed {
 | 
						|
		t.Fatalf("sink should have been closed")
 | 
						|
	}
 | 
						|
 | 
						|
	if metrics.Events != nevents {
 | 
						|
		t.Fatalf("unexpected ingress count: %d != %d", metrics.Events, nevents)
 | 
						|
	}
 | 
						|
 | 
						|
	if metrics.Pending != 0 {
 | 
						|
		t.Fatalf("unexpected egress count: %d != %d", metrics.Pending, 0)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func TestRetryingSink(t *testing.T) {
 | 
						|
 | 
						|
	// Make a sync that fails most of the time, ensuring that all the events
 | 
						|
	// make it through.
 | 
						|
	var ts testSink
 | 
						|
	flaky := &flakySink{
 | 
						|
		rate: 1.0, // start out always failing.
 | 
						|
		Sink: &ts,
 | 
						|
	}
 | 
						|
	s := newRetryingSink(flaky, 3, 10*time.Millisecond)
 | 
						|
 | 
						|
	var wg sync.WaitGroup
 | 
						|
	var block []Event
 | 
						|
	for i := 1; i <= 100; i++ {
 | 
						|
		block = append(block, createTestEvent("push", "library/test", "blob"))
 | 
						|
 | 
						|
		// Above 50, set the failure rate lower
 | 
						|
		if i > 50 {
 | 
						|
			s.mu.Lock()
 | 
						|
			flaky.rate = 0.90
 | 
						|
			s.mu.Unlock()
 | 
						|
		}
 | 
						|
 | 
						|
		if i%10 == 0 && i > 0 {
 | 
						|
			wg.Add(1)
 | 
						|
			go func(block ...Event) {
 | 
						|
				defer wg.Done()
 | 
						|
				if err := s.Write(block...); err != nil {
 | 
						|
					t.Fatalf("error writing event block: %v", err)
 | 
						|
				}
 | 
						|
			}(block...)
 | 
						|
 | 
						|
			block = nil
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	wg.Wait()
 | 
						|
	checkClose(t, s)
 | 
						|
 | 
						|
	ts.mu.Lock()
 | 
						|
	defer ts.mu.Unlock()
 | 
						|
 | 
						|
	if len(ts.events) != 100 {
 | 
						|
		t.Fatalf("events not propagated: %d != %d", len(ts.events), 100)
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
type testSink struct {
 | 
						|
	events []Event
 | 
						|
	mu     sync.Mutex
 | 
						|
	closed bool
 | 
						|
}
 | 
						|
 | 
						|
func (ts *testSink) Write(events ...Event) error {
 | 
						|
	ts.mu.Lock()
 | 
						|
	defer ts.mu.Unlock()
 | 
						|
	ts.events = append(ts.events, events...)
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
func (ts *testSink) Close() error {
 | 
						|
	ts.mu.Lock()
 | 
						|
	defer ts.mu.Unlock()
 | 
						|
	ts.closed = true
 | 
						|
 | 
						|
	logrus.Infof("closing testSink")
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
type delayedSink struct {
 | 
						|
	Sink
 | 
						|
	delay time.Duration
 | 
						|
}
 | 
						|
 | 
						|
func (ds *delayedSink) Write(events ...Event) error {
 | 
						|
	time.Sleep(ds.delay)
 | 
						|
	return ds.Sink.Write(events...)
 | 
						|
}
 | 
						|
 | 
						|
type flakySink struct {
 | 
						|
	Sink
 | 
						|
	rate float64
 | 
						|
}
 | 
						|
 | 
						|
func (fs *flakySink) Write(events ...Event) error {
 | 
						|
	if rand.Float64() < fs.rate {
 | 
						|
		return fmt.Errorf("error writing %d events", len(events))
 | 
						|
	}
 | 
						|
 | 
						|
	return fs.Sink.Write(events...)
 | 
						|
}
 | 
						|
 | 
						|
func checkClose(t *testing.T, sink Sink) {
 | 
						|
	if err := sink.Close(); err != nil {
 | 
						|
		t.Fatalf("unexpected error closing: %v", err)
 | 
						|
	}
 | 
						|
 | 
						|
	// second close should not crash but should return an error.
 | 
						|
	if err := sink.Close(); err == nil {
 | 
						|
		t.Fatalf("no error on double close")
 | 
						|
	}
 | 
						|
 | 
						|
	// Write after closed should be an error
 | 
						|
	if err := sink.Write([]Event{}...); err == nil {
 | 
						|
		t.Fatalf("write after closed did not have an error")
 | 
						|
	} else if err != ErrSinkClosed {
 | 
						|
		t.Fatalf("error should be ErrSinkClosed")
 | 
						|
	}
 | 
						|
}
 |