-
Notifications
You must be signed in to change notification settings - Fork 1.2k
/
Copy pathcontroller_test.go
132 lines (100 loc) · 3.9 KB
/
controller_test.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
/*
Copyright 2018 The Kubernetes Authors.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package controller_test
import (
"fmt"
rt "runtime"
. "github.com/onsi/ginkgo"
. "github.com/onsi/gomega"
"sigs.k8s.io/controller-runtime/pkg/client"
"sigs.k8s.io/controller-runtime/pkg/controller"
"sigs.k8s.io/controller-runtime/pkg/manager"
"sigs.k8s.io/controller-runtime/pkg/reconcile"
"sigs.k8s.io/controller-runtime/pkg/runtime/inject"
)
var _ = Describe("controller.Controller", func() {
var stop chan struct{}
rec := reconcile.Func(func(reconcile.Request) (reconcile.Result, error) {
return reconcile.Result{}, nil
})
BeforeEach(func() {
stop = make(chan struct{})
})
AfterEach(func() {
close(stop)
})
Describe("New", func() {
It("should return an error if Name is not Specified", func(done Done) {
m, err := manager.New(cfg, manager.Options{})
Expect(err).NotTo(HaveOccurred())
c, err := controller.New("", m, controller.Options{Reconciler: rec})
Expect(c).To(BeNil())
Expect(err.Error()).To(ContainSubstring("must specify Name for Controller"))
close(done)
})
It("should return an error if Reconciler is not Specified", func(done Done) {
m, err := manager.New(cfg, manager.Options{})
Expect(err).NotTo(HaveOccurred())
c, err := controller.New("foo", m, controller.Options{})
Expect(c).To(BeNil())
Expect(err.Error()).To(ContainSubstring("must specify Reconciler"))
close(done)
})
It("NewController should return an error if injecting Reconciler fails", func(done Done) {
m, err := manager.New(cfg, manager.Options{})
Expect(err).NotTo(HaveOccurred())
c, err := controller.New("foo", m, controller.Options{Reconciler: &failRec{}})
Expect(c).To(BeNil())
Expect(err).To(HaveOccurred())
Expect(err.Error()).To(ContainSubstring("expected error"))
close(done)
})
It("should not return an error if two controllers are registered with different names", func(done Done) {
m, err := manager.New(cfg, manager.Options{})
Expect(err).NotTo(HaveOccurred())
c1, err := controller.New("c1", m, controller.Options{Reconciler: rec})
Expect(err).NotTo(HaveOccurred())
Expect(c1).ToNot(BeNil())
c2, err := controller.New("c2", m, controller.Options{Reconciler: rec})
Expect(err).NotTo(HaveOccurred())
Expect(c2).ToNot(BeNil())
close(done)
})
// This test has been marked as pending because it has been causing lots of flakes in CI.
// It should be rewritten at some point later in the future.
XIt("should not leak goroutines when stop", func(done Done) {
// TODO(directxman12): After closing the proper leaks on watch this must be reduced to 0
// The leaks currently come from the event-related code (as in corev1.Event).
threshold := 3
m, err := manager.New(cfg, manager.Options{})
Expect(err).NotTo(HaveOccurred())
_, err = controller.New("new-controller", m, controller.Options{Reconciler: rec})
Expect(err).NotTo(HaveOccurred())
startGoroutines := rt.NumGoroutine()
s := make(chan struct{})
close(s)
Expect(m.Start(s)).NotTo(HaveOccurred())
Expect(rt.NumGoroutine() - startGoroutines).To(BeNumerically("<=", threshold))
close(done)
})
})
})
var _ reconcile.Reconciler = &failRec{}
var _ inject.Client = &failRec{}
type failRec struct{}
func (*failRec) Reconcile(reconcile.Request) (reconcile.Result, error) {
return reconcile.Result{}, nil
}
func (*failRec) InjectClient(client.Client) error {
return fmt.Errorf("expected error")
}