informer_manager_test.go 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207
  1. /*
  2. Copyright © 2025 ESO Maintainer Team
  3. Licensed under the Apache License, Version 2.0 (the "License");
  4. you may not use this file except in compliance with the License.
  5. You may obtain a copy of the License at
  6. https://www.apache.org/licenses/LICENSE-2.0
  7. Unless required by applicable law or agreed to in writing, software
  8. distributed under the License is distributed on an "AS IS" BASIS,
  9. WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  10. See the License for the specific language governing permissions and
  11. limitations under the License.
  12. */
  13. package externalsecret
  14. import (
  15. "context"
  16. "fmt"
  17. "testing"
  18. "time"
  19. "github.com/stretchr/testify/assert"
  20. "github.com/stretchr/testify/require"
  21. "k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
  22. "k8s.io/apimachinery/pkg/runtime/schema"
  23. "k8s.io/apimachinery/pkg/types"
  24. toolscache "k8s.io/client-go/tools/cache"
  25. "k8s.io/client-go/util/workqueue"
  26. ctrl "sigs.k8s.io/controller-runtime"
  27. runtimecache "sigs.k8s.io/controller-runtime/pkg/cache"
  28. "sigs.k8s.io/controller-runtime/pkg/client"
  29. )
  30. type fakeInformer struct{}
  31. func (f *fakeInformer) AddEventHandler(handler toolscache.ResourceEventHandler) (toolscache.ResourceEventHandlerRegistration, error) {
  32. return nil, nil
  33. }
  34. func (f *fakeInformer) AddEventHandlerWithResyncPeriod(handler toolscache.ResourceEventHandler, _ time.Duration) (toolscache.ResourceEventHandlerRegistration, error) {
  35. return nil, nil
  36. }
  37. func (f *fakeInformer) AddEventHandlerWithOptions(handler toolscache.ResourceEventHandler, _ toolscache.HandlerOptions) (toolscache.ResourceEventHandlerRegistration, error) {
  38. return nil, nil
  39. }
  40. func (f *fakeInformer) RemoveEventHandler(_ toolscache.ResourceEventHandlerRegistration) error {
  41. return nil
  42. }
  43. func (f *fakeInformer) AddIndexers(indexers toolscache.Indexers) error {
  44. return nil
  45. }
  46. func (f *fakeInformer) HasSynced() bool {
  47. return true
  48. }
  49. func (f *fakeInformer) IsStopped() bool {
  50. return false
  51. }
  52. type fakeCache struct {
  53. runtimecache.Cache
  54. getInformerCalled bool
  55. getInformerObj client.Object
  56. getInformerErr error
  57. }
  58. func (f *fakeCache) GetInformer(ctx context.Context, obj client.Object, opts ...runtimecache.InformerGetOption) (runtimecache.Informer, error) {
  59. f.getInformerCalled = true
  60. f.getInformerObj = obj
  61. if f.getInformerErr != nil {
  62. return nil, f.getInformerErr
  63. }
  64. return &fakeInformer{}, nil
  65. }
  66. func TestEnsureInformer_UsesUnstructured(t *testing.T) {
  67. fc := &fakeCache{}
  68. log := ctrl.Log.WithName("test")
  69. m := &DefaultInformerManager{
  70. managerContext: context.Background(),
  71. cache: fc,
  72. log: log,
  73. informers: make(map[string]*informerEntry),
  74. queue: workqueue.NewTypedRateLimitingQueue(workqueue.DefaultTypedControllerRateLimiter[ctrl.Request]()),
  75. }
  76. gvk := schema.GroupVersionKind{
  77. Group: "monitoring.example.io",
  78. Version: "v1alpha1",
  79. Kind: "CustomNotifier",
  80. }
  81. es := types.NamespacedName{Name: "test-es", Namespace: "default"}
  82. created, err := m.EnsureInformer(context.Background(), gvk, es)
  83. require.NoError(t, err)
  84. assert.True(t, created)
  85. assert.True(t, fc.getInformerCalled, "GetInformer should be called")
  86. obj, ok := fc.getInformerObj.(*unstructured.Unstructured)
  87. require.True(t, ok, "GetInformer should be called with *unstructured.Unstructured")
  88. assert.Equal(t, gvk, obj.GroupVersionKind())
  89. }
  90. func TestEnsureInformer_DeduplicatesExistingGVK(t *testing.T) {
  91. fc := &fakeCache{}
  92. log := ctrl.Log.WithName("test")
  93. m := &DefaultInformerManager{
  94. managerContext: context.Background(),
  95. cache: fc,
  96. log: log,
  97. informers: make(map[string]*informerEntry),
  98. queue: workqueue.NewTypedRateLimitingQueue(workqueue.DefaultTypedControllerRateLimiter[ctrl.Request]()),
  99. }
  100. gvk := schema.GroupVersionKind{Group: "example.io", Version: "v1", Kind: "Foo"}
  101. es1 := types.NamespacedName{Name: "es-1", Namespace: "default"}
  102. es2 := types.NamespacedName{Name: "es-2", Namespace: "default"}
  103. created, err := m.EnsureInformer(context.Background(), gvk, es1)
  104. require.NoError(t, err)
  105. assert.True(t, created)
  106. fc.getInformerCalled = false
  107. created, err = m.EnsureInformer(context.Background(), gvk, es2)
  108. require.NoError(t, err)
  109. assert.False(t, created)
  110. assert.False(t, fc.getInformerCalled, "GetInformer should not be called again for same GVK")
  111. entry := m.informers[gvk.String()]
  112. assert.Len(t, entry.externalSecrets, 2)
  113. }
  114. func TestEnsureInformer_ErrorWhenQueueNotSet(t *testing.T) {
  115. fc := &fakeCache{}
  116. log := ctrl.Log.WithName("test")
  117. m := &DefaultInformerManager{
  118. managerContext: context.Background(),
  119. cache: fc,
  120. log: log,
  121. informers: make(map[string]*informerEntry),
  122. }
  123. gvk := schema.GroupVersionKind{Group: "example.io", Version: "v1", Kind: "Foo"}
  124. es := types.NamespacedName{Name: "es-1", Namespace: "default"}
  125. _, err := m.EnsureInformer(context.Background(), gvk, es)
  126. assert.Error(t, err)
  127. assert.Contains(t, err.Error(), "queue not initialized")
  128. }
  129. func TestEnsureInformer_PropagatesCacheError(t *testing.T) {
  130. fc := &fakeCache{
  131. getInformerErr: fmt.Errorf("CRD not found"),
  132. }
  133. log := ctrl.Log.WithName("test")
  134. m := &DefaultInformerManager{
  135. managerContext: context.Background(),
  136. cache: fc,
  137. log: log,
  138. informers: make(map[string]*informerEntry),
  139. queue: workqueue.NewTypedRateLimitingQueue(workqueue.DefaultTypedControllerRateLimiter[ctrl.Request]()),
  140. }
  141. gvk := schema.GroupVersionKind{Group: "example.io", Version: "v1", Kind: "Foo"}
  142. es := types.NamespacedName{Name: "es-1", Namespace: "default"}
  143. _, err := m.EnsureInformer(context.Background(), gvk, es)
  144. assert.Error(t, err)
  145. assert.Contains(t, err.Error(), "CRD not found")
  146. }
  147. func TestReleaseInformer_RemovesES(t *testing.T) {
  148. fc := &fakeCache{}
  149. log := ctrl.Log.WithName("test")
  150. m := &DefaultInformerManager{
  151. managerContext: context.Background(),
  152. cache: fc,
  153. log: log,
  154. informers: make(map[string]*informerEntry),
  155. queue: workqueue.NewTypedRateLimitingQueue(workqueue.DefaultTypedControllerRateLimiter[ctrl.Request]()),
  156. }
  157. gvk := schema.GroupVersionKind{Group: "example.io", Version: "v1", Kind: "Foo"}
  158. es1 := types.NamespacedName{Name: "es-1", Namespace: "default"}
  159. es2 := types.NamespacedName{Name: "es-2", Namespace: "default"}
  160. _, err := m.EnsureInformer(context.Background(), gvk, es1)
  161. require.NoError(t, err)
  162. _, err = m.EnsureInformer(context.Background(), gvk, es2)
  163. require.NoError(t, err)
  164. err = m.ReleaseInformer(context.Background(), gvk, es1)
  165. require.NoError(t, err)
  166. assert.True(t, m.IsManaged(gvk))
  167. entry := m.informers[gvk.String()]
  168. assert.Len(t, entry.externalSecrets, 1)
  169. }