From a99a4103bd657d7c3f92fc86ee37a21583554eb1 Mon Sep 17 00:00:00 2001 From: Daniel Nephin Date: Wed, 15 Jul 2020 17:03:39 -0400 Subject: [PATCH] stream: fix overallocation in filter And add tests --- agent/consul/stream/subscription.go | 30 ++++++++-------- agent/consul/stream/subscription_test.go | 46 ++++++++++++++++++++++++ 2 files changed, 61 insertions(+), 15 deletions(-) diff --git a/agent/consul/stream/subscription.go b/agent/consul/stream/subscription.go index c2177468ff..e4a24cc1eb 100644 --- a/agent/consul/stream/subscription.go +++ b/agent/consul/stream/subscription.go @@ -82,7 +82,7 @@ func (s *Subscription) Next(ctx context.Context) ([]Event, error) { } s.currentItem = next - events := s.filter(next.Events) + events := filter(s.req.Key, next.Events) if len(events) == 0 { continue } @@ -90,34 +90,34 @@ func (s *Subscription) Next(ctx context.Context) ([]Event, error) { } } -// TODO: test cases for this method -func (s *Subscription) filter(events []Event) []Event { - if s.req.Key == "" || len(events) == 0 { +// filter events to only those that match the key exactly. +func filter(key string, events []Event) []Event { + if key == "" || len(events) == 0 { return events } - allMatch := true + var count int for _, e := range events { - if s.req.Key != e.Key { - allMatch = false - break + if key == e.Key { + count++ } } // Only allocate a new slice if some events need to be filtered out - if allMatch { + switch count { + case 0: + return nil + case len(events): return events } - // FIXME: this will over-allocate. We could get a count from the previous range - // over events. - events = make([]Event, 0, len(events)) + result := make([]Event, 0, count) for _, e := range events { - if s.req.Key == e.Key { - events = append(events, e) + if key == e.Key { + result = append(result, e) } } - return events + return result } // Close the subscription. Subscribers will receive an error when they call Next, diff --git a/agent/consul/stream/subscription_test.go b/agent/consul/stream/subscription_test.go index 36a60dc482..84e941a3bf 100644 --- a/agent/consul/stream/subscription_test.go +++ b/agent/consul/stream/subscription_test.go @@ -148,3 +148,49 @@ func publishTestEvent(index uint64, b *eventBuffer, key string) { } b.Append([]Event{e}) } + +func TestFilter_NoKey(t *testing.T) { + events := make([]Event, 0, 5) + events = append(events, Event{Key: "One"}, Event{Key: "Two"}) + + actual := filter("", events) + require.Equal(t, events, actual) + + // test that a new array was not allocated + require.Equal(t, cap(actual), 5) +} + +func TestFilter_WithKey_AllEventsMatch(t *testing.T) { + events := make([]Event, 0, 5) + events = append(events, Event{Key: "Same"}, Event{Key: "Same"}) + + actual := filter("Same", events) + require.Equal(t, events, actual) + + // test that a new array was not allocated + require.Equal(t, cap(actual), 5) +} + +func TestFilter_WithKey_SomeEventsMatch(t *testing.T) { + events := make([]Event, 0, 5) + events = append(events, Event{Key: "Same"}, Event{Key: "Other"}, Event{Key: "Same"}) + + actual := filter("Same", events) + expected := []Event{{Key: "Same"}, {Key: "Same"}} + require.Equal(t, expected, actual) + + // test that a new array was allocated with the correct size + require.Equal(t, cap(actual), 2) +} + +func TestFilter_WithKey_NoEventsMatch(t *testing.T) { + events := make([]Event, 0, 5) + events = append(events, Event{Key: "Same"}, Event{Key: "Same"}) + + actual := filter("Other", events) + var expected []Event + require.Equal(t, expected, actual) + + // test that no array was allocated + require.Equal(t, cap(actual), 0) +}