Skip to content

Commit 4229b86

Browse files
committed
NodeRef controller
Signed-off-by: Vince Prignano <[email protected]>
1 parent 5f71f0b commit 4229b86

11 files changed

+560
-7
lines changed

config/rbac/rbac_role.yaml

+10-4
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,15 @@ rules:
1616
- update
1717
- patch
1818
- delete
19+
- apiGroups:
20+
- ""
21+
resources:
22+
- events
23+
verbs:
24+
- get
25+
- list
26+
- watch
27+
- create
1928
- apiGroups:
2029
- cluster.k8s.io
2130
resources:
@@ -94,9 +103,6 @@ rules:
94103
- apiGroups:
95104
- ""
96105
resources:
97-
- events
106+
- secrets
98107
verbs:
99108
- get
100-
- list
101-
- watch
102-
- create

pkg/controller/BUILD.bazel

+2
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@ go_library(
66
"add_machinedeployment.go",
77
"add_machineset.go",
88
"add_node.go",
9+
"add_noderef.go",
910
"controller.go",
1011
],
1112
importpath = "sigs.k8s.io/cluster-api/pkg/controller",
@@ -14,6 +15,7 @@ go_library(
1415
"//pkg/controller/machinedeployment:go_default_library",
1516
"//pkg/controller/machineset:go_default_library",
1617
"//pkg/controller/node:go_default_library",
18+
"//pkg/controller/noderef:go_default_library",
1719
"//vendor/sigs.k8s.io/controller-runtime/pkg/manager:go_default_library",
1820
],
1921
)

pkg/controller/add_noderef.go

+26
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,26 @@
1+
/*
2+
Copyright 2019 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 controller
18+
19+
import (
20+
"sigs.k8s.io/cluster-api/pkg/controller/noderef"
21+
)
22+
23+
func init() {
24+
// AddToManagerFuncs is a list of functions to create controllers and add them to a manager.
25+
AddToManagerFuncs = append(AddToManagerFuncs, noderef.Add)
26+
}

pkg/controller/noderef/BUILD.bazel

+50
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,50 @@
1+
load("@io_bazel_rules_go//go:def.bzl", "go_library", "go_test")
2+
3+
go_library(
4+
name = "go_default_library",
5+
srcs = ["noderef_controller.go"],
6+
importpath = "sigs.k8s.io/cluster-api/pkg/controller/noderef",
7+
visibility = ["//visibility:public"],
8+
deps = [
9+
"//pkg/apis/cluster/v1alpha1:go_default_library",
10+
"//pkg/controller/noderefutil:go_default_library",
11+
"//pkg/controller/remote:go_default_library",
12+
"//vendor/github.com/pkg/errors:go_default_library",
13+
"//vendor/k8s.io/api/core/v1:go_default_library",
14+
"//vendor/k8s.io/apimachinery/pkg/api/errors:go_default_library",
15+
"//vendor/k8s.io/apimachinery/pkg/apis/meta/v1:go_default_library",
16+
"//vendor/k8s.io/apimachinery/pkg/runtime:go_default_library",
17+
"//vendor/k8s.io/client-go/kubernetes/typed/core/v1:go_default_library",
18+
"//vendor/k8s.io/klog:go_default_library",
19+
"//vendor/sigs.k8s.io/controller-runtime/pkg/client:go_default_library",
20+
"//vendor/sigs.k8s.io/controller-runtime/pkg/controller:go_default_library",
21+
"//vendor/sigs.k8s.io/controller-runtime/pkg/handler:go_default_library",
22+
"//vendor/sigs.k8s.io/controller-runtime/pkg/manager:go_default_library",
23+
"//vendor/sigs.k8s.io/controller-runtime/pkg/reconcile:go_default_library",
24+
"//vendor/sigs.k8s.io/controller-runtime/pkg/source:go_default_library",
25+
],
26+
)
27+
28+
go_test(
29+
name = "go_default_test",
30+
srcs = [
31+
"noderef_controller_suite_test.go",
32+
"noderef_controller_test.go",
33+
],
34+
embed = [":go_default_library"],
35+
deps = [
36+
"//pkg/apis:go_default_library",
37+
"//pkg/apis/cluster/v1alpha1:go_default_library",
38+
"//vendor/github.com/onsi/gomega:go_default_library",
39+
"//vendor/golang.org/x/net/context:go_default_library",
40+
"//vendor/k8s.io/apimachinery/pkg/api/errors:go_default_library",
41+
"//vendor/k8s.io/apimachinery/pkg/apis/meta/v1:go_default_library",
42+
"//vendor/k8s.io/apimachinery/pkg/types:go_default_library",
43+
"//vendor/k8s.io/client-go/kubernetes/scheme:go_default_library",
44+
"//vendor/k8s.io/client-go/rest:go_default_library",
45+
"//vendor/sigs.k8s.io/controller-runtime/pkg/client:go_default_library",
46+
"//vendor/sigs.k8s.io/controller-runtime/pkg/envtest:go_default_library",
47+
"//vendor/sigs.k8s.io/controller-runtime/pkg/manager:go_default_library",
48+
"//vendor/sigs.k8s.io/controller-runtime/pkg/reconcile:go_default_library",
49+
],
50+
)
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,195 @@
1+
/*
2+
Copyright 2019 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 noderef
18+
19+
import (
20+
"context"
21+
"time"
22+
23+
"github.com/pkg/errors"
24+
apicorev1 "k8s.io/api/core/v1"
25+
apierrors "k8s.io/apimachinery/pkg/api/errors"
26+
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
27+
"k8s.io/apimachinery/pkg/runtime"
28+
corev1 "k8s.io/client-go/kubernetes/typed/core/v1"
29+
"k8s.io/klog"
30+
"sigs.k8s.io/cluster-api/pkg/apis/cluster/v1alpha1"
31+
"sigs.k8s.io/cluster-api/pkg/controller/noderefutil"
32+
"sigs.k8s.io/cluster-api/pkg/controller/remote"
33+
"sigs.k8s.io/controller-runtime/pkg/client"
34+
"sigs.k8s.io/controller-runtime/pkg/controller"
35+
"sigs.k8s.io/controller-runtime/pkg/handler"
36+
"sigs.k8s.io/controller-runtime/pkg/manager"
37+
"sigs.k8s.io/controller-runtime/pkg/reconcile"
38+
"sigs.k8s.io/controller-runtime/pkg/source"
39+
)
40+
41+
var (
42+
ErrNodeNotFound = errors.New("cannot find node with maching ProviderID")
43+
)
44+
45+
// Add creates a new NodeRef Controller and adds it to the Manager with default RBAC. The Manager will set fields on the Controller
46+
// and Start it when the Manager is Started.
47+
func Add(mgr manager.Manager) error {
48+
return add(mgr, newReconciler(mgr))
49+
}
50+
51+
// newReconciler returns a new reconcile.Reconciler
52+
func newReconciler(mgr manager.Manager) reconcile.Reconciler {
53+
return &ReconcileNodeRef{Client: mgr.GetClient(), scheme: mgr.GetScheme()}
54+
}
55+
56+
// add adds a new Controller to mgr with r as the reconcile.Reconciler
57+
func add(mgr manager.Manager, r reconcile.Reconciler) error {
58+
// Create a new controller
59+
c, err := controller.New("noderef-controller", mgr, controller.Options{Reconciler: r})
60+
if err != nil {
61+
return err
62+
}
63+
64+
// Watch for changes to Machines.
65+
return c.Watch(&source.Kind{Type: &v1alpha1.Machine{}}, &handler.EnqueueRequestForObject{})
66+
}
67+
68+
var _ reconcile.Reconciler = &ReconcileNodeRef{}
69+
70+
// ReconcileNodeRef reconciles a Machine object.
71+
type ReconcileNodeRef struct {
72+
client.Client
73+
scheme *runtime.Scheme
74+
}
75+
76+
// Reconcile watches Machines.
77+
// +kubebuilder:rbac:groups=,resources=secrets,verbs=get
78+
func (r *ReconcileNodeRef) Reconcile(request reconcile.Request) (reconcile.Result, error) {
79+
// Fetch the NodeRef instance
80+
machine := &v1alpha1.Machine{}
81+
err := r.Get(context.TODO(), request.NamespacedName, machine)
82+
if err != nil {
83+
if apierrors.IsNotFound(err) {
84+
return reconcile.Result{}, nil
85+
}
86+
return reconcile.Result{}, err
87+
}
88+
89+
// Check that the Machine hasn't been deleted or in the process.
90+
if !machine.DeletionTimestamp.IsZero() {
91+
return reconcile.Result{}, nil
92+
}
93+
94+
// Check that the Machine doesn't already have a NodeRef.
95+
if machine.Status.NodeRef != nil {
96+
return reconcile.Result{}, nil
97+
}
98+
99+
// Check that the Machine has a cluster label.
100+
if machine.Labels[v1alpha1.MachineClusterLabelName] == "" {
101+
klog.Warningf("Machine %q in namespace %q doesn't specify %q label, cannot set nodeRef", machine.Name, machine.Namespace, v1alpha1.MachineClusterLabelName)
102+
return reconcile.Result{}, nil
103+
}
104+
105+
// Check that the Machine has a valid ProviderID.
106+
if machine.Spec.ProviderID == nil || *machine.Spec.ProviderID == "" {
107+
klog.Warningf("Machine %q in namespace %q doesn't have a valid ProviderID, cannot set nodeRef", machine.Name, machine.Namespace)
108+
return reconcile.Result{}, nil
109+
}
110+
111+
providerID, err := noderefutil.NewProviderID(*machine.Spec.ProviderID)
112+
if err != nil {
113+
return reconcile.Result{}, err
114+
}
115+
116+
cluster, err := r.getCluster(context.TODO(), machine)
117+
if err != nil {
118+
return reconcile.Result{}, err
119+
}
120+
121+
clusterClient, err := remote.NewClusterClient(r.Client, cluster)
122+
if err != nil {
123+
return reconcile.Result{}, err
124+
}
125+
126+
corev1Client, err := clusterClient.CoreV1()
127+
if err != nil {
128+
return reconcile.Result{}, err
129+
}
130+
131+
// Get the Node reference.
132+
nodeRef, err := r.getNodeReference(corev1Client, providerID)
133+
if err != nil {
134+
if err == ErrNodeNotFound {
135+
return reconcile.Result{RequeueAfter: 10 * time.Second}, nil
136+
}
137+
return reconcile.Result{}, err
138+
}
139+
140+
// Update Machine.
141+
machine.Status.NodeRef = nodeRef
142+
if err := r.Client.Update(context.TODO(), machine); err != nil {
143+
return reconcile.Result{}, err
144+
}
145+
146+
return reconcile.Result{}, nil
147+
}
148+
149+
func (r *ReconcileNodeRef) getCluster(ctx context.Context, machine *v1alpha1.Machine) (*v1alpha1.Cluster, error) {
150+
cluster := &v1alpha1.Cluster{}
151+
key := client.ObjectKey{
152+
Namespace: machine.Namespace,
153+
Name: machine.Labels[v1alpha1.MachineClusterLabelName],
154+
}
155+
156+
if err := r.Client.Get(ctx, key, cluster); err != nil {
157+
return nil, err
158+
}
159+
160+
return cluster, nil
161+
}
162+
163+
func (r *ReconcileNodeRef) getNodeReference(client corev1.NodesGetter, providerID *noderefutil.ProviderID) (*apicorev1.ObjectReference, error) {
164+
listOpt := metav1.ListOptions{}
165+
166+
for {
167+
nodeList, err := client.Nodes().List(listOpt)
168+
if err != nil {
169+
return nil, err
170+
}
171+
172+
for _, node := range nodeList.Items {
173+
nodeProviderID, err := noderefutil.NewProviderID(node.Spec.ProviderID)
174+
if err != nil {
175+
continue
176+
}
177+
178+
if providerID.Equals(nodeProviderID) {
179+
return &apicorev1.ObjectReference{
180+
Kind: node.Kind,
181+
APIVersion: node.APIVersion,
182+
Name: node.Name,
183+
UID: node.UID,
184+
}, nil
185+
}
186+
}
187+
188+
listOpt.Continue = nodeList.Continue
189+
if listOpt.Continue == "" {
190+
break
191+
}
192+
}
193+
194+
return nil, ErrNodeNotFound
195+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,75 @@
1+
/*
2+
Copyright 2019 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 noderef
18+
19+
import (
20+
stdlog "log"
21+
"os"
22+
"path/filepath"
23+
"sync"
24+
"testing"
25+
26+
"github.com/onsi/gomega"
27+
"k8s.io/client-go/kubernetes/scheme"
28+
"k8s.io/client-go/rest"
29+
"sigs.k8s.io/cluster-api/pkg/apis"
30+
"sigs.k8s.io/controller-runtime/pkg/envtest"
31+
"sigs.k8s.io/controller-runtime/pkg/manager"
32+
"sigs.k8s.io/controller-runtime/pkg/reconcile"
33+
)
34+
35+
var cfg *rest.Config
36+
37+
func TestMain(m *testing.M) {
38+
t := &envtest.Environment{
39+
CRDDirectoryPaths: []string{filepath.Join("..", "..", "..", "config", "crds")},
40+
}
41+
apis.AddToScheme(scheme.Scheme)
42+
43+
var err error
44+
if cfg, err = t.Start(); err != nil {
45+
stdlog.Fatal(err)
46+
}
47+
48+
code := m.Run()
49+
t.Stop()
50+
os.Exit(code)
51+
}
52+
53+
// SetupTestReconcile returns a reconcile.Reconcile implementation that delegates to inner and
54+
// writes the request to requests after Reconcile is finished.
55+
func SetupTestReconcile(inner reconcile.Reconciler) (reconcile.Reconciler, chan reconcile.Request) {
56+
requests := make(chan reconcile.Request)
57+
fn := reconcile.Func(func(req reconcile.Request) (reconcile.Result, error) {
58+
result, err := inner.Reconcile(req)
59+
requests <- req
60+
return result, err
61+
})
62+
return fn, requests
63+
}
64+
65+
// StartTestManager adds recFn
66+
func StartTestManager(mgr manager.Manager, g *gomega.GomegaWithT) (chan struct{}, *sync.WaitGroup) {
67+
stop := make(chan struct{})
68+
wg := &sync.WaitGroup{}
69+
wg.Add(1)
70+
go func() {
71+
defer wg.Done()
72+
g.Expect(mgr.Start(stop)).NotTo(gomega.HaveOccurred())
73+
}()
74+
return stop, wg
75+
}

0 commit comments

Comments
 (0)