/
state_actuator.go
200 lines (165 loc) · 7.39 KB
/
state_actuator.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
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
// Copyright (c) 2020 SAP SE or an SAP affiliate company. All rights reserved. This file is licensed under the Apache Software License, v. 2 except as noted otherwise in the LICENSE file
//
// 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 worker
import (
"context"
"encoding/json"
"fmt"
"sort"
machinev1alpha1 "github.com/gardener/machine-controller-manager/pkg/apis/machine/v1alpha1"
"github.com/go-logr/logr"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/runtime"
"sigs.k8s.io/controller-runtime/pkg/client"
workerhelper "github.com/gardener/gardener/extensions/pkg/controller/worker/helper"
extensionsv1alpha1 "github.com/gardener/gardener/pkg/apis/extensions/v1alpha1"
)
type genericStateActuator struct {
logger logr.Logger
client client.Client
}
// NewStateActuator creates a new Actuator that reconciles Worker's State subresource
// It provides a default implementation that allows easier integration of providers.
func NewStateActuator(logger logr.Logger) StateActuator {
return &genericStateActuator{logger: logger.WithName("worker-state-actuator")}
}
func (a *genericStateActuator) InjectClient(client client.Client) error {
a.client = client
return nil
}
// Reconcile update the Worker state with the latest.
func (a *genericStateActuator) Reconcile(ctx context.Context, worker *extensionsv1alpha1.Worker) error {
copyOfWorker := worker.DeepCopy()
if err := a.updateWorkerState(ctx, copyOfWorker); err != nil {
return fmt.Errorf("failed to update the state in worker status: %w", err)
}
return nil
}
func (a *genericStateActuator) updateWorkerState(ctx context.Context, worker *extensionsv1alpha1.Worker) error {
state, err := a.getWorkerState(ctx, worker.Namespace)
if err != nil {
return err
}
rawState, err := json.Marshal(state)
if err != nil {
return err
}
worker.Status.State = &runtime.RawExtension{Raw: rawState}
return a.client.Status().Update(ctx, worker)
}
func (a *genericStateActuator) getWorkerState(ctx context.Context, namespace string) (*State, error) {
existingMachineDeployments := &machinev1alpha1.MachineDeploymentList{}
if err := a.client.List(ctx, existingMachineDeployments, client.InNamespace(namespace)); err != nil {
return nil, err
}
machineSets, err := a.getExistingMachineSetsMap(ctx, namespace)
if err != nil {
return nil, err
}
machines, err := a.getExistingMachinesMap(ctx, namespace)
if err != nil {
return nil, err
}
workerState := &State{
MachineDeployments: make(map[string]*MachineDeploymentState),
}
for _, deployment := range existingMachineDeployments.Items {
machineDeploymentState := &MachineDeploymentState{}
machineDeploymentState.Replicas = deployment.Spec.Replicas
machineDeploymentMachineSets, ok := machineSets[deployment.Name]
if !ok {
continue
}
addMachineSetToMachineDeploymentState(machineDeploymentMachineSets, machineDeploymentState)
for _, machineSet := range machineDeploymentMachineSets {
currentMachines := append(machines[machineSet.Name], machines[deployment.Name]...)
if len(currentMachines) <= 0 {
continue
}
for index := range currentMachines {
addMachineToMachineDeploymentState(¤tMachines[index], machineDeploymentState)
}
}
workerState.MachineDeployments[deployment.Name] = machineDeploymentState
}
return workerState, nil
}
// getExistingMachineSetsMap returns a map of existing MachineSets as values and their owners as keys
func (a *genericStateActuator) getExistingMachineSetsMap(ctx context.Context, namespace string) (map[string][]machinev1alpha1.MachineSet, error) {
existingMachineSets := &machinev1alpha1.MachineSetList{}
if err := a.client.List(ctx, existingMachineSets, client.InNamespace(namespace)); err != nil {
return nil, err
}
// When we read from the cache we get unsorted results, hence, we sort to prevent unnecessary state updates from happening.
sort.Slice(existingMachineSets.Items, func(i, j int) bool { return existingMachineSets.Items[i].Name < existingMachineSets.Items[j].Name })
return workerhelper.BuildOwnerToMachineSetsMap(existingMachineSets.Items), nil
}
// getExistingMachinesMap returns a map of the existing Machines as values and the name of their owner
// no matter of being machineSet or MachineDeployment. If a Machine has a ownerReference the key(owner)
// will be the MachineSet if not the key will be the name of the MachineDeployment which is stored as
// a label. We assume that there is no MachineDeployment and MachineSet with the same names.
func (a *genericStateActuator) getExistingMachinesMap(ctx context.Context, namespace string) (map[string][]machinev1alpha1.Machine, error) {
existingMachines := &machinev1alpha1.MachineList{}
if err := a.client.List(ctx, existingMachines, client.InNamespace(namespace)); err != nil {
return nil, err
}
// We temporarily filter out machines without provider ID or node status (VMs which got created but not yet joined the cluster)
// to prevent unnecessarily persisting them in the Worker state.
// TODO: Remove this again once machine-controller-manager supports backing off creation/deletion of failed machines, see
// https://github.com/gardener/machine-controller-manager/issues/483.
var filteredMachines []machinev1alpha1.Machine
for _, machine := range existingMachines.Items {
if machine.Spec.ProviderID != "" || machine.Status.Node != "" {
filteredMachines = append(filteredMachines, machine)
}
}
// When we read from the cache we get unsorted results, hence, we sort to prevent unnecessary state updates from happening.
sort.Slice(filteredMachines, func(i, j int) bool { return filteredMachines[i].Name < filteredMachines[j].Name })
return workerhelper.BuildOwnerToMachinesMap(filteredMachines), nil
}
func addMachineSetToMachineDeploymentState(machineSets []machinev1alpha1.MachineSet, machineDeploymentState *MachineDeploymentState) {
if len(machineSets) < 1 || machineDeploymentState == nil {
return
}
// remove redundant data from the machine set
for index := range machineSets {
machineSet := &machineSets[index]
machineSet.ObjectMeta = metav1.ObjectMeta{
Name: machineSet.Name,
Namespace: machineSet.Namespace,
Annotations: machineSet.Annotations,
Labels: machineSet.Labels,
}
machineSet.OwnerReferences = nil
machineSet.Status = machinev1alpha1.MachineSetStatus{}
}
machineDeploymentState.MachineSets = machineSets
}
func addMachineToMachineDeploymentState(machine *machinev1alpha1.Machine, machineDeploymentState *MachineDeploymentState) {
if machine == nil || machineDeploymentState == nil {
return
}
// remove redundant data from the machine
machine.ObjectMeta = metav1.ObjectMeta{
Name: machine.Name,
Namespace: machine.Namespace,
Annotations: machine.Annotations,
Labels: machine.Labels,
}
machine.OwnerReferences = nil
machine.Status = machinev1alpha1.MachineStatus{
Node: machine.Status.Node,
}
machineDeploymentState.Machines = append(machineDeploymentState.Machines, *machine)
}