From 105e5aede58f921399ba32a76d7e83257d40107d Mon Sep 17 00:00:00 2001 From: Bogdan Drutu Date: Mon, 20 Feb 2023 14:05:21 -0800 Subject: [PATCH] confmap: clear list of old closers (#7222) Fixes https://github.com/open-telemetry/opentelemetry-collector/issues/7215 Signed-off-by: Bogdan Drutu --- .chloggen/removeoldclosers.yaml | 11 +++++++++++ confmap/resolver.go | 1 + confmap/resolver_test.go | 34 ++++++++++++++++++++++----------- 3 files changed, 35 insertions(+), 11 deletions(-) create mode 100755 .chloggen/removeoldclosers.yaml diff --git a/.chloggen/removeoldclosers.yaml b/.chloggen/removeoldclosers.yaml new file mode 100755 index 00000000000..465138c94c3 --- /dev/null +++ b/.chloggen/removeoldclosers.yaml @@ -0,0 +1,11 @@ +# One of 'breaking', 'deprecation', 'new_component', 'enhancement', 'bug_fix' +change_type: bug_fix + +# The name of the component, or a single word describing the area of concern, (e.g. otlpreceiver) +component: confmap + +# A brief description of the change. Surround your text with quotes ("") if it needs to start with a backtick (`). +note: Clear list of old already closed closers. + +# One or more tracking issues or pull requests related to the change +issues: [7215] diff --git a/confmap/resolver.go b/confmap/resolver.go index cb4360c561e..ba9a33efa2a 100644 --- a/confmap/resolver.go +++ b/confmap/resolver.go @@ -223,6 +223,7 @@ func (mr *Resolver) closeIfNeeded(ctx context.Context) error { for _, ret := range mr.closers { err = multierr.Append(err, ret(ctx)) } + mr.closers = nil return err } diff --git a/confmap/resolver_test.go b/confmap/resolver_test.go index 5e9b0502a78..6280560453d 100644 --- a/confmap/resolver_test.go +++ b/confmap/resolver_test.go @@ -23,15 +23,16 @@ import ( "github.com/stretchr/testify/assert" "github.com/stretchr/testify/require" + "go.uber.org/atomic" ) type mockProvider struct { - scheme string - retM any - errR error - errS error - errW error - errC error + scheme string + retM any + errR error + errS error + errW error + closeFunc func(ctx context.Context) error } func (m *mockProvider) Retrieve(_ context.Context, _ string, watcher WatcherFunc) (*Retrieved, error) { @@ -43,7 +44,7 @@ func (m *mockProvider) Retrieve(_ context.Context, _ string, watcher WatcherFunc } watcher(&ChangeEvent{Error: m.errW}) - return NewRetrieved(m.retM, WithRetrievedClose(func(ctx context.Context) error { return m.errC })) + return NewRetrieved(m.retM, WithRetrievedClose(m.closeFunc)) } func (m *mockProvider) Scheme() string { @@ -157,7 +158,7 @@ func TestResolverErrors(t *testing.T) { locations: []string{"mock:", "err:"}, providers: []Provider{ &mockProvider{}, - &mockProvider{scheme: "err", retM: map[string]any{}, errC: errors.New("close_err")}, + &mockProvider{scheme: "err", retM: map[string]any{}, closeFunc: func(ctx context.Context) error { return errors.New("close_err") }}, }, expectCloseErr: true, }, @@ -269,13 +270,18 @@ func TestBackwardsCompatibilityForFilePath(t *testing.T) { } func TestResolver(t *testing.T) { + numCalls := atomic.NewInt32(0) resolver, err := NewResolver(ResolverSettings{ - URIs: []string{"mock:"}, - Providers: makeMapProvidersMap(&mockProvider{retM: newConfFromFile(t, filepath.Join("testdata", "config.yaml"))}), + URIs: []string{"mock:"}, + Providers: makeMapProvidersMap(&mockProvider{retM: map[string]any{}, closeFunc: func(ctx context.Context) error { + numCalls.Add(1) + return nil + }}), Converters: nil}) require.NoError(t, err) _, errN := resolver.Resolve(context.Background()) assert.NoError(t, errN) + assert.Equal(t, int32(0), numCalls.Load()) errW := <-resolver.Watch() assert.NoError(t, errW) @@ -284,18 +290,24 @@ func TestResolver(t *testing.T) { _, errN = resolver.Resolve(context.Background()) assert.NoError(t, errN) + assert.Equal(t, int32(1), numCalls.Load()) errW = <-resolver.Watch() assert.NoError(t, errW) + _, errN = resolver.Resolve(context.Background()) + assert.NoError(t, errN) + assert.Equal(t, int32(2), numCalls.Load()) + errC := resolver.Shutdown(context.Background()) assert.NoError(t, errC) + assert.Equal(t, int32(3), numCalls.Load()) } func TestResolverNewLinesInOpaqueValue(t *testing.T) { _, err := NewResolver(ResolverSettings{ URIs: []string{"mock:receivers:\n nop:\n"}, - Providers: makeMapProvidersMap(&mockProvider{retM: newConfFromFile(t, filepath.Join("testdata", "config.yaml"))}), + Providers: makeMapProvidersMap(&mockProvider{retM: map[string]any{}}), Converters: nil}) assert.NoError(t, err) }