submatview: add a test for handling of NewSnapshotToFollow

Also add some godoc
Rename some vars and functions
Fix a data race in the new cache test for entry closing.
pull/8809/head
Daniel Nephin 2020-10-02 20:04:45 -04:00
parent 534d8b45bb
commit 3fa08beecf
9 changed files with 134 additions and 128 deletions

View File

@ -1,7 +1,7 @@
package cachetype package cachetype
import ( import (
fmt "fmt" "fmt"
"github.com/hashicorp/consul/proto/pbcommon" "github.com/hashicorp/consul/proto/pbcommon"
"github.com/hashicorp/consul/proto/pbservice" "github.com/hashicorp/consul/proto/pbservice"
@ -17,10 +17,9 @@ func newEndOfSnapshotEvent(topic pbsubscribe.Topic, index uint64) *pbsubscribe.E
} }
} }
func newNewSnapshotToFollowEvent(topic pbsubscribe.Topic, index uint64) *pbsubscribe.Event { func newNewSnapshotToFollowEvent(topic pbsubscribe.Topic) *pbsubscribe.Event {
return &pbsubscribe.Event{ return &pbsubscribe.Event{
Topic: topic, Topic: topic,
Index: index,
Payload: &pbsubscribe.Event_NewSnapshotToFollow{NewSnapshotToFollow: true}, Payload: &pbsubscribe.Event_NewSnapshotToFollow{NewSnapshotToFollow: true},
} }
} }
@ -32,12 +31,12 @@ func newNewSnapshotToFollowEvent(topic pbsubscribe.Topic, index uint64) *pbsubsc
// need that. nodeNum should be less than 64k to make the IP address look // need that. nodeNum should be less than 64k to make the IP address look
// realistic. Any other changes can be made on the returned event to avoid // realistic. Any other changes can be made on the returned event to avoid
// adding too many options to callers. // adding too many options to callers.
func newEventServiceHealthRegister(index uint64, nodeNum int, svc string) pbsubscribe.Event { func newEventServiceHealthRegister(index uint64, nodeNum int, svc string) *pbsubscribe.Event {
node := fmt.Sprintf("node%d", nodeNum) node := fmt.Sprintf("node%d", nodeNum)
nodeID := types.NodeID(fmt.Sprintf("11111111-2222-3333-4444-%012d", nodeNum)) nodeID := types.NodeID(fmt.Sprintf("11111111-2222-3333-4444-%012d", nodeNum))
addr := fmt.Sprintf("10.10.%d.%d", nodeNum/256, nodeNum%256) addr := fmt.Sprintf("10.10.%d.%d", nodeNum/256, nodeNum%256)
return pbsubscribe.Event{ return &pbsubscribe.Event{
Topic: pbsubscribe.Topic_ServiceHealth, Topic: pbsubscribe.Topic_ServiceHealth,
Key: svc, Key: svc,
Index: index, Index: index,
@ -114,10 +113,10 @@ func newEventServiceHealthRegister(index uint64, nodeNum int, svc string) pbsubs
// need that. nodeNum should be less than 64k to make the IP address look // need that. nodeNum should be less than 64k to make the IP address look
// realistic. Any other changes can be made on the returned event to avoid // realistic. Any other changes can be made on the returned event to avoid
// adding too many options to callers. // adding too many options to callers.
func newEventServiceHealthDeregister(index uint64, nodeNum int, svc string) pbsubscribe.Event { func newEventServiceHealthDeregister(index uint64, nodeNum int, svc string) *pbsubscribe.Event {
node := fmt.Sprintf("node%d", nodeNum) node := fmt.Sprintf("node%d", nodeNum)
return pbsubscribe.Event{ return &pbsubscribe.Event{
Topic: pbsubscribe.Topic_ServiceHealth, Topic: pbsubscribe.Topic_ServiceHealth,
Key: svc, Key: svc,
Index: index, Index: index,
@ -158,13 +157,13 @@ func newEventServiceHealthDeregister(index uint64, nodeNum int, svc string) pbsu
} }
} }
func newEventBatchWithEvents(first pbsubscribe.Event, evs ...pbsubscribe.Event) pbsubscribe.Event { func newEventBatchWithEvents(first *pbsubscribe.Event, evs ...*pbsubscribe.Event) *pbsubscribe.Event {
events := make([]*pbsubscribe.Event, len(evs)+1) events := make([]*pbsubscribe.Event, len(evs)+1)
events[0] = &first events[0] = first
for i := range evs { for i := range evs {
events[i+1] = &evs[i] events[i+1] = evs[i]
} }
return pbsubscribe.Event{ return &pbsubscribe.Event{
Topic: first.Topic, Topic: first.Topic,
Index: first.Index, Index: first.Index,
Payload: &pbsubscribe.Event_EventBatch{ Payload: &pbsubscribe.Event_EventBatch{

View File

@ -8,12 +8,10 @@ import (
"github.com/hashicorp/go-bexpr" "github.com/hashicorp/go-bexpr"
"github.com/hashicorp/go-hclog" "github.com/hashicorp/go-hclog"
"github.com/hashicorp/consul/agent/submatview"
"github.com/hashicorp/consul/lib/retry"
"github.com/hashicorp/consul/agent/cache" "github.com/hashicorp/consul/agent/cache"
"github.com/hashicorp/consul/agent/structs" "github.com/hashicorp/consul/agent/structs"
"github.com/hashicorp/consul/agent/submatview"
"github.com/hashicorp/consul/lib/retry"
"github.com/hashicorp/consul/proto/pbservice" "github.com/hashicorp/consul/proto/pbservice"
"github.com/hashicorp/consul/proto/pbsubscribe" "github.com/hashicorp/consul/proto/pbsubscribe"
) )
@ -44,8 +42,7 @@ type MaterializerDeps struct {
// Fetch implements cache.Type // Fetch implements cache.Type
func (c *StreamingHealthServices) Fetch(opts cache.FetchOptions, req cache.Request) (cache.FetchResult, error) { func (c *StreamingHealthServices) Fetch(opts cache.FetchOptions, req cache.Request) (cache.FetchResult, error) {
if opts.LastResult != nil && opts.LastResult.State != nil { if opts.LastResult != nil && opts.LastResult.State != nil {
state := opts.LastResult.State.(*streamingHealthState) return opts.LastResult.State.(*streamingHealthState).Fetch(opts)
return state.materializer.Fetch(state.done, opts)
} }
srvReq := req.(*structs.ServiceSpecificRequest) srvReq := req.(*structs.ServiceSpecificRequest)
@ -63,57 +60,64 @@ func (c *StreamingHealthServices) Fetch(opts cache.FetchOptions, req cache.Reque
return req return req
} }
m, err := newMaterializer(c.deps, newReqFn, srvReq.Filter) materializer, err := newMaterializer(c.deps, newReqFn, srvReq.Filter)
if err != nil { if err != nil {
return cache.FetchResult{}, err return cache.FetchResult{}, err
} }
ctx, cancel := context.WithCancel(context.TODO()) ctx, cancel := context.WithCancel(context.TODO())
go m.Run(ctx) go materializer.Run(ctx)
result, err := m.Fetch(ctx.Done(), opts) state := &streamingHealthState{
result.State = &streamingHealthState{ materializer: materializer,
materializer: m,
done: ctx.Done(), done: ctx.Done(),
cancel: cancel, cancel: cancel,
} }
return result, err return state.Fetch(opts)
} }
func newMaterializer( func newMaterializer(
d MaterializerDeps, deps MaterializerDeps,
r func(uint64) pbsubscribe.SubscribeRequest, newRequestFn func(uint64) pbsubscribe.SubscribeRequest,
filter string, filter string,
) (*submatview.Materializer, error) { ) (*submatview.Materializer, error) {
view, err := newHealthViewState(filter) view, err := newHealthView(filter)
if err != nil { if err != nil {
return nil, err return nil, err
} }
return submatview.NewMaterializer(submatview.Deps{ return submatview.NewMaterializer(submatview.Deps{
View: view, View: view,
Client: d.Client, Client: deps.Client,
Logger: d.Logger, Logger: deps.Logger,
Waiter: &retry.Waiter{ Waiter: &retry.Waiter{
MinFailures: 1, MinFailures: 1,
MinWait: 0, MinWait: 0,
MaxWait: 60 * time.Second, MaxWait: 60 * time.Second,
Jitter: retry.NewJitter(100), Jitter: retry.NewJitter(100),
}, },
Request: r, Request: newRequestFn,
}), nil }), nil
} }
// streamingHealthState wraps a Materializer to manage its lifecycle, and to
// add itself to the FetchResult.State.
type streamingHealthState struct { type streamingHealthState struct {
materializer *submatview.Materializer materializer *submatview.Materializer
done <-chan struct{} done <-chan struct{}
cancel func() cancel func()
} }
func (c *streamingHealthState) Close() error { func (s *streamingHealthState) Close() error {
c.cancel() s.cancel()
return nil return nil
} }
func newHealthViewState(filterExpr string) (submatview.View, error) { func (s *streamingHealthState) Fetch(opts cache.FetchOptions) (cache.FetchResult, error) {
result, err := s.materializer.Fetch(s.done, opts)
result.State = s
return result, err
}
func newHealthView(filterExpr string) (*healthView, error) {
s := &healthView{state: make(map[string]structs.CheckServiceNode)} s := &healthView{state: make(map[string]structs.CheckServiceNode)}
// We apply filtering to the raw CheckServiceNodes before we are done mutating // We apply filtering to the raw CheckServiceNodes before we are done mutating
@ -131,8 +135,7 @@ func newHealthViewState(filterExpr string) (submatview.View, error) {
// (IndexedCheckServiceNodes) and update it in place for each event - that // (IndexedCheckServiceNodes) and update it in place for each event - that
// involves re-sorting each time etc. though. // involves re-sorting each time etc. though.
type healthView struct { type healthView struct {
state map[string]structs.CheckServiceNode state map[string]structs.CheckServiceNode
// TODO: test case with filter
filter *bexpr.Filter filter *bexpr.Filter
} }
@ -155,7 +158,6 @@ func (s *healthView) Update(events []*pbsubscribe.Event) error {
delete(s.state, id) delete(s.state, id)
} }
} }
// TODO: replace with a no-op filter instead of a conditional
if s.filter != nil { if s.filter != nil {
filtered, err := s.filter.Execute(s.state) filtered, err := s.filter.Execute(s.state)
if err != nil { if err != nil {
@ -166,16 +168,17 @@ func (s *healthView) Update(events []*pbsubscribe.Event) error {
return nil return nil
} }
// Result implements View // Result returns the structs.IndexedCheckServiceNodes stored by this view.
func (s *healthView) Result(index uint64) (interface{}, error) { func (s *healthView) Result(index uint64) (interface{}, error) {
var result structs.IndexedCheckServiceNodes result := structs.IndexedCheckServiceNodes{
// Avoid a nil slice if there are no results in the view Nodes: make(structs.CheckServiceNodes, 0, len(s.state)),
// TODO: why this ^ QueryMeta: structs.QueryMeta{
result.Nodes = structs.CheckServiceNodes{} Index: index,
},
}
for _, node := range s.state { for _, node := range s.state {
result.Nodes = append(result.Nodes, node) result.Nodes = append(result.Nodes, node)
} }
result.Index = index
return &result, nil return &result, nil
} }

View File

@ -80,10 +80,7 @@ func TestStreamingHealthServices_EmptySnapshot(t *testing.T) {
start := time.Now() start := time.Now()
go func() { go func() {
time.Sleep(200 * time.Millisecond) time.Sleep(200 * time.Millisecond)
client.QueueEvents(newEventServiceHealthRegister(4, 1, "web"))
// Then a service registers
healthEv := newEventServiceHealthRegister(4, 1, "web")
client.QueueEvents(&healthEv)
}() }()
opts.Timeout = time.Second opts.Timeout = time.Second
@ -103,15 +100,9 @@ func TestStreamingHealthServices_EmptySnapshot(t *testing.T) {
opts.LastResult = &result opts.LastResult = &result
}) })
runStep(t, "reconnects and resumes after transient stream error", func(t *testing.T) { runStep(t, "reconnects and resumes after temporary error", func(t *testing.T) {
// Use resetErr just because it's "temporary" this is a stand in for any
// network error that uses that same interface though.
client.QueueErr(tempError("broken pipe")) client.QueueErr(tempError("broken pipe"))
// After the error the view should re-subscribe with same index so will get
// a "resume stream".
client.QueueEvents(newNewSnapshotToFollowEvent(pbsubscribe.Topic_ServiceHealth, opts.MinIndex))
// Next fetch will continue to block until timeout and receive the same // Next fetch will continue to block until timeout and receive the same
// result. // result.
start := time.Now() start := time.Now()
@ -129,8 +120,7 @@ func TestStreamingHealthServices_EmptySnapshot(t *testing.T) {
opts.LastResult = &result opts.LastResult = &result
// But an update should still be noticed due to reconnection // But an update should still be noticed due to reconnection
healthEv := newEventServiceHealthRegister(10, 2, "web") client.QueueEvents(newEventServiceHealthRegister(10, 2, "web"))
client.QueueEvents(&healthEv)
start = time.Now() start = time.Now()
opts.Timeout = time.Second opts.Timeout = time.Second
@ -153,10 +143,6 @@ func TestStreamingHealthServices_EmptySnapshot(t *testing.T) {
go func() { go func() {
time.Sleep(200 * time.Millisecond) time.Sleep(200 * time.Millisecond)
client.QueueErr(errors.New("invalid request")) client.QueueErr(errors.New("invalid request"))
// After the error the view should re-subscribe with same index so will get
// a "resume stream".
client.QueueEvents(newNewSnapshotToFollowEvent(pbsubscribe.Topic_ServiceHealth, opts.MinIndex))
}() }()
// Next fetch should return the error // Next fetch should return the error
@ -182,8 +168,7 @@ func TestStreamingHealthServices_EmptySnapshot(t *testing.T) {
opts.LastResult = &result opts.LastResult = &result
// But an update should still be noticed due to reconnection // But an update should still be noticed due to reconnection
healthEv := newEventServiceHealthRegister(opts.MinIndex+5, 3, "web") client.QueueEvents(newEventServiceHealthRegister(opts.MinIndex+5, 3, "web"))
client.QueueEvents(&healthEv)
opts.Timeout = time.Second opts.Timeout = time.Second
result, err = typ.Fetch(opts, req) result, err = typ.Fetch(opts, req)
@ -238,14 +223,13 @@ func TestStreamingHealthServices_FullSnapshot(t *testing.T) {
}} }}
// Create an initial snapshot of 3 instances on different nodes // Create an initial snapshot of 3 instances on different nodes
makeReg := func(index uint64, nodeNum int) *pbsubscribe.Event { registerServiceWeb := func(index uint64, nodeNum int) *pbsubscribe.Event {
e := newEventServiceHealthRegister(index, nodeNum, "web") return newEventServiceHealthRegister(index, nodeNum, "web")
return &e
} }
client.QueueEvents( client.QueueEvents(
makeReg(5, 1), registerServiceWeb(5, 1),
makeReg(5, 2), registerServiceWeb(5, 2),
makeReg(5, 3), registerServiceWeb(5, 3),
newEndOfSnapshotEvent(pbsubscribe.Topic_ServiceHealth, 5)) newEndOfSnapshotEvent(pbsubscribe.Topic_ServiceHealth, 5))
// This contains the view state so important we share it between calls. // This contains the view state so important we share it between calls.
@ -288,8 +272,7 @@ func TestStreamingHealthServices_FullSnapshot(t *testing.T) {
time.Sleep(200 * time.Millisecond) time.Sleep(200 * time.Millisecond)
// Deregister instance on node1 // Deregister instance on node1
healthEv := newEventServiceHealthDeregister(20, 1, "web") client.QueueEvents(newEventServiceHealthDeregister(20, 1, "web"))
client.QueueEvents(&healthEv)
}() }()
opts.Timeout = time.Second opts.Timeout = time.Second
@ -315,9 +298,9 @@ func TestStreamingHealthServices_FullSnapshot(t *testing.T) {
client.QueueErr(status.Error(codes.Aborted, "reset by server")) client.QueueErr(status.Error(codes.Aborted, "reset by server"))
client.QueueEvents( client.QueueEvents(
makeReg(50, 3), // overlap existing node registerServiceWeb(50, 3), // overlap existing node
makeReg(50, 4), registerServiceWeb(50, 4),
makeReg(50, 5), registerServiceWeb(50, 5),
newEndOfSnapshotEvent(pbsubscribe.Topic_ServiceHealth, 50)) newEndOfSnapshotEvent(pbsubscribe.Topic_ServiceHealth, 50))
// Make another blocking query with THE SAME index. It should immediately // Make another blocking query with THE SAME index. It should immediately
@ -336,6 +319,29 @@ func TestStreamingHealthServices_FullSnapshot(t *testing.T) {
opts.MinIndex = result.Index opts.MinIndex = result.Index
opts.LastResult = &result opts.LastResult = &result
}) })
runStep(t, "reconnects and receives new snapshot when server state has changed", func(t *testing.T) {
client.QueueErr(tempError("temporary connection error"))
client.QueueEvents(
newNewSnapshotToFollowEvent(pbsubscribe.Topic_ServiceHealth),
registerServiceWeb(50, 3), // overlap existing node
registerServiceWeb(50, 4),
registerServiceWeb(50, 5),
newEndOfSnapshotEvent(pbsubscribe.Topic_ServiceHealth, 50))
start := time.Now()
opts.MinIndex = 49
opts.Timeout = time.Second
result, err := typ.Fetch(opts, req)
require.NoError(t, err)
elapsed := time.Since(start)
require.True(t, elapsed < time.Second,
"Fetch should have returned before the timeout")
require.Equal(t, uint64(50), result.Index)
require.Equal(t, []string{"node3", "node4", "node5"}, gatherNodes(result.Value))
})
} }
func TestStreamingHealthServices_EventBatches(t *testing.T) { func TestStreamingHealthServices_EventBatches(t *testing.T) {
@ -351,7 +357,7 @@ func TestStreamingHealthServices_EventBatches(t *testing.T) {
newEventServiceHealthRegister(5, 2, "web"), newEventServiceHealthRegister(5, 2, "web"),
newEventServiceHealthRegister(5, 3, "web")) newEventServiceHealthRegister(5, 3, "web"))
client.QueueEvents( client.QueueEvents(
&batchEv, batchEv,
newEndOfSnapshotEvent(pbsubscribe.Topic_ServiceHealth, 5)) newEndOfSnapshotEvent(pbsubscribe.Topic_ServiceHealth, 5))
// This contains the view state so important we share it between calls. // This contains the view state so important we share it between calls.
@ -394,7 +400,7 @@ func TestStreamingHealthServices_EventBatches(t *testing.T) {
// Register another // Register another
newEventServiceHealthRegister(20, 4, "web"), newEventServiceHealthRegister(20, 4, "web"),
) )
client.QueueEvents(&batchEv) client.QueueEvents(batchEv)
opts.Timeout = time.Second opts.Timeout = time.Second
result, err := typ.Fetch(opts, req) result, err := typ.Fetch(opts, req)
require.NoError(t, err) require.NoError(t, err)
@ -421,7 +427,7 @@ func TestStreamingHealthServices_Filtering(t *testing.T) {
newEventServiceHealthRegister(5, 2, "web"), newEventServiceHealthRegister(5, 2, "web"),
newEventServiceHealthRegister(5, 3, "web")) newEventServiceHealthRegister(5, 3, "web"))
client.QueueEvents( client.QueueEvents(
&batchEv, batchEv,
newEndOfSnapshotEvent(pbsubscribe.Topic_ServiceHealth, 5)) newEndOfSnapshotEvent(pbsubscribe.Topic_ServiceHealth, 5))
// This contains the view state so important we share it between calls. // This contains the view state so important we share it between calls.
@ -451,8 +457,7 @@ func TestStreamingHealthServices_Filtering(t *testing.T) {
require.NoError(t, err) require.NoError(t, err)
require.Equal(t, uint64(5), result.Index) require.Equal(t, uint64(5), result.Index)
require.ElementsMatch(t, []string{"node2"}, require.Equal(t, []string{"node2"}, gatherNodes(result.Value))
gatherNodes(result.Value))
opts.MinIndex = result.Index opts.MinIndex = result.Index
opts.LastResult = &result opts.LastResult = &result
@ -467,14 +472,13 @@ func TestStreamingHealthServices_Filtering(t *testing.T) {
// Register another // Register another
newEventServiceHealthRegister(20, 4, "web"), newEventServiceHealthRegister(20, 4, "web"),
) )
client.QueueEvents(&batchEv) client.QueueEvents(batchEv)
opts.Timeout = time.Second opts.Timeout = time.Second
result, err := typ.Fetch(opts, req) result, err := typ.Fetch(opts, req)
require.NoError(t, err) require.NoError(t, err)
require.Equal(t, uint64(20), result.Index) require.Equal(t, uint64(20), result.Index)
require.ElementsMatch(t, []string{"node2"}, require.Equal(t, []string{"node2"}, gatherNodes(result.Value))
gatherNodes(result.Value))
opts.MinIndex = result.Index opts.MinIndex = result.Index
opts.LastResult = &result opts.LastResult = &result

View File

@ -4,7 +4,6 @@ import (
"context" "context"
"google.golang.org/grpc" "google.golang.org/grpc"
"google.golang.org/grpc/metadata"
"github.com/hashicorp/consul/proto/pbsubscribe" "github.com/hashicorp/consul/proto/pbsubscribe"
) )
@ -12,6 +11,7 @@ import (
// TestStreamingClient is a mock StreamingClient for testing that allows // TestStreamingClient is a mock StreamingClient for testing that allows
// for queueing up custom events to a subscriber. // for queueing up custom events to a subscriber.
type TestStreamingClient struct { type TestStreamingClient struct {
pbsubscribe.StateChangeSubscription_SubscribeClient
events chan eventOrErr events chan eventOrErr
ctx context.Context ctx context.Context
} }
@ -57,15 +57,3 @@ func (t *TestStreamingClient) Recv() (*pbsubscribe.Event, error) {
return nil, t.ctx.Err() return nil, t.ctx.Err()
} }
} }
func (t *TestStreamingClient) Header() (metadata.MD, error) { return nil, nil }
func (t *TestStreamingClient) Trailer() metadata.MD { return nil }
func (t *TestStreamingClient) CloseSend() error { return nil }
func (t *TestStreamingClient) Context() context.Context { return nil }
func (t *TestStreamingClient) SendMsg(m interface{}) error { return nil }
func (t *TestStreamingClient) RecvMsg(m interface{}) error { return nil }

View File

@ -18,15 +18,16 @@ import (
"container/heap" "container/heap"
"context" "context"
"fmt" "fmt"
io "io" "io"
"strconv" "strconv"
"sync" "sync"
"sync/atomic" "sync/atomic"
"time" "time"
"github.com/armon/go-metrics" "github.com/armon/go-metrics"
"github.com/hashicorp/consul/lib"
"golang.org/x/time/rate" "golang.org/x/time/rate"
"github.com/hashicorp/consul/lib"
) )
//go:generate mockery -all -inpkg //go:generate mockery -all -inpkg

View File

@ -10,11 +10,12 @@ import (
"testing" "testing"
"time" "time"
"github.com/hashicorp/consul/sdk/testutil"
"github.com/stretchr/testify/assert" "github.com/stretchr/testify/assert"
"github.com/stretchr/testify/mock" "github.com/stretchr/testify/mock"
"github.com/stretchr/testify/require" "github.com/stretchr/testify/require"
"golang.org/x/time/rate" "golang.org/x/time/rate"
"github.com/hashicorp/consul/sdk/testutil"
) )
// Test a basic Get with no indexes (and therefore no blocking queries). // Test a basic Get with no indexes (and therefore no blocking queries).
@ -841,15 +842,6 @@ func TestCacheGet_expireResetGet(t *testing.T) {
typ.AssertExpectations(t) typ.AssertExpectations(t)
} }
type testCloser struct {
closed bool
}
func (t *testCloser) Close() error {
t.closed = true
return nil
}
// Test that entries with state that satisfies io.Closer get cleaned up // Test that entries with state that satisfies io.Closer get cleaned up
func TestCacheGet_expireClose(t *testing.T) { func TestCacheGet_expireClose(t *testing.T) {
t.Parallel() t.Parallel()
@ -859,6 +851,7 @@ func TestCacheGet_expireClose(t *testing.T) {
typ := &MockType{} typ := &MockType{}
defer typ.AssertExpectations(t) defer typ.AssertExpectations(t)
c := New(Options{}) c := New(Options{})
defer c.Close()
typ.On("RegisterOptions").Return(RegisterOptions{ typ.On("RegisterOptions").Return(RegisterOptions{
SupportsBlocking: true, SupportsBlocking: true,
LastGetTTL: 100 * time.Millisecond, LastGetTTL: 100 * time.Millisecond,
@ -877,13 +870,26 @@ func TestCacheGet_expireClose(t *testing.T) {
require.NoError(err) require.NoError(err)
require.Equal(42, result) require.Equal(42, result)
require.False(meta.Hit) require.False(meta.Hit)
require.False(state.closed) require.False(state.isClosed())
// Sleep for the expiry // Sleep for the expiry
time.Sleep(200 * time.Millisecond) time.Sleep(200 * time.Millisecond)
// state.Close() should have been called // state.Close() should have been called
require.True(state.closed) require.True(state.isClosed())
}
type testCloser struct {
closed uint32
}
func (t *testCloser) Close() error {
atomic.SwapUint32(&t.closed, 1)
return nil
}
func (t *testCloser) isClosed() bool {
return atomic.LoadUint32(&t.closed) == 1
} }
// Test a Get with a request that returns the same cache key across // Test a Get with a request that returns the same cache key across

View File

@ -61,6 +61,7 @@ type SnapshotHandlers map[Topic]SnapshotFunc
// SnapshotFunc builds a snapshot for the subscription request, and appends the // SnapshotFunc builds a snapshot for the subscription request, and appends the
// events to the Snapshot using SnapshotAppender. // events to the Snapshot using SnapshotAppender.
// If err is not nil the SnapshotFunc must return a non-zero index.
type SnapshotFunc func(SubscribeRequest, SnapshotAppender) (index uint64, err error) type SnapshotFunc func(SubscribeRequest, SnapshotAppender) (index uint64, err error)
// SnapshotAppender appends groups of events to create a Snapshot of state. // SnapshotAppender appends groups of events to create a Snapshot of state.

View File

@ -1,8 +1,16 @@
package submatview package submatview
import "github.com/hashicorp/consul/proto/pbsubscribe" import (
"github.com/hashicorp/consul/proto/pbsubscribe"
)
type eventHandler func(events *pbsubscribe.Event) (eventHandler, error) // eventHandler is a function which performs some operation on the received
// events, then returns the eventHandler that should be used for the next set
// of events.
// If eventHandler fails to handle the events it may return an error. If an
// error is returned the next eventHandler will be ignored.
// eventHandler is used to implement a very simple finite-state machine.
type eventHandler func(events *pbsubscribe.Event) (next eventHandler, err error)
func (m *Materializer) initialHandler(index uint64) eventHandler { func (m *Materializer) initialHandler(index uint64) eventHandler {
if index == 0 { if index == 0 {

View File

@ -65,7 +65,6 @@ type Deps struct {
Logger hclog.Logger Logger hclog.Logger
Waiter *retry.Waiter Waiter *retry.Waiter
Request func(index uint64) pbsubscribe.SubscribeRequest Request func(index uint64) pbsubscribe.SubscribeRequest
Stop func()
} }
// StreamClient provides a subscription to state change events. // StreamClient provides a subscription to state change events.
@ -94,37 +93,34 @@ func (m *Materializer) Run(ctx context.Context) {
return return
} }
m.lock.Lock()
// TODO: move this into a func
// If this is a temporary error and it's the first consecutive failure,
// retry to see if we can get a result without erroring back to clients.
// If it's non-temporary or a repeated failure return to clients while we
// retry to get back in a good state.
if _, ok := err.(temporary); !ok || m.retryWaiter.Failures() > 0 {
m.notifyUpdateLocked(err)
}
waitCh := m.retryWaiter.Failed()
failures := m.retryWaiter.Failures() failures := m.retryWaiter.Failures()
m.lock.Unlock() if isNonTemporaryOrConsecutiveFailure(err, failures) {
m.lock.Lock()
m.notifyUpdateLocked(err)
m.lock.Unlock()
}
m.deps.Logger.Error("subscribe call failed", m.deps.Logger.Error("subscribe call failed",
"err", err, "err", err,
"topic", req.Topic, "topic", req.Topic,
"key", req.Key, "key", req.Key,
"failure_count", failures) "failure_count", failures+1)
select { if err := m.retryWaiter.Wait(ctx); err != nil {
case <-ctx.Done():
return return
case <-waitCh:
} }
} }
} }
// temporary is a private interface as used by net and other std lib packages to // isNonTemporaryOrConsecutiveFailure returns true if the error is a temporary error
// show error types represent temporary/recoverable errors. // or if the failures > 0.
type temporary interface { func isNonTemporaryOrConsecutiveFailure(err error, failures int) bool {
Temporary() bool // temporary is an interface used by net and other std lib packages to
// show error types represent temporary/recoverable errors.
_, ok := err.(interface {
Temporary() bool
})
return !ok || failures > 0
} }
// runSubscription opens a new subscribe streaming call to the servers and runs // runSubscription opens a new subscribe streaming call to the servers and runs