Skip to content

Commit 6890503

Browse files
committed
Add simple operator package
1 parent eb6b5de commit 6890503

File tree

6 files changed

+543
-0
lines changed

6 files changed

+543
-0
lines changed
Lines changed: 159 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,159 @@
1+
/*
2+
Copyright 2018 The Kubernetes Authors.
3+
4+
Licensed under the Apache License, Version 2.0 (the "License");
5+
you may not use this file except in compliance with the License.
6+
You may obtain a copy of the License at
7+
8+
http://www.apache.org/licenses/LICENSE-2.0
9+
10+
Unless required by applicable law or agreed to in writing, software
11+
distributed under the License is distributed on an "AS IS" BASIS,
12+
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13+
See the License for the specific language governing permissions and
14+
limitations under the License.
15+
*/
16+
17+
package application
18+
19+
import (
20+
"fmt"
21+
"strings"
22+
23+
"k8s.io/apimachinery/pkg/runtime"
24+
"k8s.io/client-go/rest"
25+
"sigs.k8s.io/controller-runtime/pkg/client/apiutil"
26+
"sigs.k8s.io/controller-runtime/pkg/client/config"
27+
"sigs.k8s.io/controller-runtime/pkg/controller"
28+
"sigs.k8s.io/controller-runtime/pkg/handler"
29+
"sigs.k8s.io/controller-runtime/pkg/manager"
30+
"sigs.k8s.io/controller-runtime/pkg/predicate"
31+
"sigs.k8s.io/controller-runtime/pkg/reconcile"
32+
"sigs.k8s.io/controller-runtime/pkg/source"
33+
)
34+
35+
// application is a simple Controller for a single API type. It will create a Manager for itself
36+
// if one is not provided.
37+
type application struct {
38+
mgr manager.Manager
39+
ctrl controller.Controller
40+
}
41+
42+
// Supporting mocking out functions for testing
43+
var getConfig = config.GetConfig
44+
var newController = controller.New
45+
var newManager = manager.New
46+
var getGvk = apiutil.GVKForObject
47+
48+
// Builder builds an Application Controller (e.g. Operator) and returns a manager.Manager to start it.
49+
type Builder struct {
50+
apiType runtime.Object
51+
mgr manager.Manager
52+
predicates []predicate.Predicate
53+
managedObjects []runtime.Object
54+
config *rest.Config
55+
reconcile reconcile.Reconciler
56+
}
57+
58+
// BuilderFor returns a new Builder for and BuilderFor
59+
func BuilderFor(apiType runtime.Object) *Builder {
60+
return &Builder{apiType: apiType}
61+
}
62+
63+
// Owns configures the Application Controller to respond to create / delete / update events for objects it managedObjects
64+
// - e.g. creates. apiType is an empty instance of an object matching the managed object type.
65+
func (b *Builder) Owns(apiType runtime.Object) *Builder {
66+
b.managedObjects = append(b.managedObjects, apiType)
67+
return b
68+
}
69+
70+
// WithConfig sets the Config to use for configuring clients. Defaults to the in-cluster config or to ~/.kube/config.
71+
func (b *Builder) WithConfig(config *rest.Config) *Builder {
72+
b.config = config
73+
return b
74+
}
75+
76+
// WithManager sets the Manager to use for registering the Controller. Defaults to a new manager.Manager.
77+
func (b *Builder) WithManager(m manager.Manager) *Builder {
78+
b.mgr = m
79+
return b
80+
}
81+
82+
// WithEventFilter sets the event filters, to filter which create/update/delete/generic events eventually
83+
// trigger reconciliations. For example, filtering on whether the resource version has changed.
84+
// Defaults to the empty list.
85+
func (b *Builder) WithEventFilter(p predicate.Predicate) *Builder {
86+
b.predicates = append(b.predicates, p)
87+
return b
88+
}
89+
90+
// WithReconciler sets the Reconcile called in response to create / update / delete events for the
91+
// BuilderFor type or Created object types.
92+
func (b *Builder) WithReconciler(r reconcile.Reconciler) *Builder {
93+
b.reconcile = r
94+
return b
95+
}
96+
97+
func (b *Builder) doConfig() error {
98+
if b.config != nil {
99+
return nil
100+
}
101+
var err error
102+
b.config, err = getConfig()
103+
return err
104+
}
105+
106+
func (b *Builder) doManager() error {
107+
if b.mgr != nil {
108+
return nil
109+
}
110+
var err error
111+
b.mgr, err = newManager(b.config, manager.Options{})
112+
return err
113+
}
114+
115+
// Build builds the Application Controller and returns the Manager used to start it.
116+
func (b *Builder) Build() (manager.Manager, error) {
117+
a := &application{}
118+
119+
if b.reconcile == nil {
120+
return nil, fmt.Errorf("must call WithReconciler to set Reconciler")
121+
}
122+
if err := b.doConfig(); err != nil {
123+
return nil, err
124+
}
125+
if err := b.doManager(); err != nil {
126+
return nil, err
127+
}
128+
129+
a.mgr = b.mgr
130+
131+
gvk, err := getGvk(b.apiType, a.mgr.GetScheme())
132+
if err != nil {
133+
return nil, err
134+
}
135+
136+
// Create the controller
137+
name := fmt.Sprintf("%s-application", strings.ToLower(gvk.Kind))
138+
a.ctrl, err = newController(name, a.mgr, controller.Options{Reconciler: b.reconcile})
139+
if err != nil {
140+
return nil, err
141+
}
142+
143+
// Reconcile type
144+
err = a.ctrl.Watch(&source.Kind{Type: b.apiType}, &handler.EnqueueRequestForObject{}, b.predicates...)
145+
if err != nil {
146+
return nil, err
147+
}
148+
149+
// Watch the managed types
150+
for _, t := range b.managedObjects {
151+
err = a.ctrl.Watch(&source.Kind{Type: t}, &handler.EnqueueRequestForOwner{
152+
OwnerType: b.apiType, IsController: true}, b.predicates...)
153+
if err != nil {
154+
return nil, err
155+
}
156+
}
157+
158+
return a.mgr, nil
159+
}
Lines changed: 35 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
package application
2+
3+
import (
4+
"testing"
5+
6+
. "github.com/onsi/ginkgo"
7+
. "github.com/onsi/gomega"
8+
"k8s.io/client-go/rest"
9+
"sigs.k8s.io/controller-runtime/pkg/envtest"
10+
logf "sigs.k8s.io/controller-runtime/pkg/runtime/log"
11+
)
12+
13+
func TestSource(t *testing.T) {
14+
RegisterFailHandler(Fail)
15+
RunSpecsWithDefaultAndCustomReporters(t, "application Suite", []Reporter{envtest.NewlineReporter{}})
16+
}
17+
18+
var testenv *envtest.Environment
19+
var cfg *rest.Config
20+
21+
var _ = BeforeSuite(func(done Done) {
22+
logf.SetLogger(logf.ZapLogger(false))
23+
24+
testenv = &envtest.Environment{}
25+
26+
var err error
27+
cfg, err = testenv.Start()
28+
Expect(err).NotTo(HaveOccurred())
29+
30+
close(done)
31+
}, 60)
32+
33+
var _ = AfterSuite(func() {
34+
testenv.Stop()
35+
})

0 commit comments

Comments
 (0)