Skip to content

Commit

Permalink
Set nodeSelector on jobs and allow empty nodeSelector
Browse files Browse the repository at this point in the history
Switch to a pointer for nodeSelector to allow different logic for empty vs unset
  • Loading branch information
olliewalsh committed Nov 20, 2024
1 parent 27b5458 commit fdeb8f1
Show file tree
Hide file tree
Showing 9 changed files with 141 additions and 24 deletions.
2 changes: 1 addition & 1 deletion api/v1beta1/common_types.go
Original file line number Diff line number Diff line change
Expand Up @@ -61,7 +61,7 @@ type HeatServiceTemplate struct {

// +kubebuilder:validation:Optional
// NodeSelector to target subset of worker nodes for running the service
NodeSelector map[string]string `json:"nodeSelector,omitempty"`
NodeSelector *map[string]string `json:"nodeSelector,omitempty"`

// +kubebuilder:validation:Optional
// CustomServiceConfig - customize the service config using this parameter to change service defaults,
Expand Down
2 changes: 1 addition & 1 deletion api/v1beta1/heat_types.go
Original file line number Diff line number Diff line change
Expand Up @@ -119,7 +119,7 @@ type HeatSpecBase struct {

// +kubebuilder:validation:Optional
// NodeSelector to target subset of worker nodes for running the Heat services
NodeSelector map[string]string `json:"nodeSelector,omitempty"`
NodeSelector *map[string]string `json:"nodeSelector,omitempty"`

// +kubebuilder:validation:Optional
// +kubebuilder:default=600
Expand Down
20 changes: 14 additions & 6 deletions api/v1beta1/zz_generated.deepcopy.go

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

21 changes: 12 additions & 9 deletions controllers/heat_controller.go
Original file line number Diff line number Diff line change
Expand Up @@ -835,11 +835,12 @@ func (r *HeatReconciler) apiDeploymentCreateOrUpdate(
},
}

if heatAPISpec.NodeSelector == nil {
heatAPISpec.NodeSelector = instance.Spec.NodeSelector
}

op, err := controllerutil.CreateOrUpdate(ctx, r.Client, deployment, func() error {
deployment.Spec = heatAPISpec
if len(deployment.Spec.NodeSelector) == 0 {
deployment.Spec.NodeSelector = instance.Spec.NodeSelector
}
return controllerutil.SetControllerReference(instance, deployment, r.Scheme)
})

Expand All @@ -858,6 +859,10 @@ func (r *HeatReconciler) cfnapiDeploymentCreateOrUpdate(
ServiceAccount: instance.RbacResourceName(),
}

if heatCfnAPISpec.NodeSelector == nil {
heatCfnAPISpec.NodeSelector = instance.Spec.NodeSelector
}

deployment := &heatv1beta1.HeatCfnAPI{
ObjectMeta: metav1.ObjectMeta{
Name: fmt.Sprintf("%s-cfnapi", instance.Name),
Expand All @@ -867,9 +872,6 @@ func (r *HeatReconciler) cfnapiDeploymentCreateOrUpdate(

op, err := controllerutil.CreateOrUpdate(ctx, r.Client, deployment, func() error {
deployment.Spec = heatCfnAPISpec
if len(deployment.Spec.NodeSelector) == 0 {
deployment.Spec.NodeSelector = instance.Spec.NodeSelector
}
return controllerutil.SetControllerReference(instance, deployment, r.Scheme)
})

Expand All @@ -889,6 +891,10 @@ func (r *HeatReconciler) engineDeploymentCreateOrUpdate(
TLS: instance.Spec.HeatAPI.TLS.Ca,
}

if heatEngineSpec.NodeSelector == nil {
heatEngineSpec.NodeSelector = instance.Spec.NodeSelector
}

deployment := &heatv1beta1.HeatEngine{
ObjectMeta: metav1.ObjectMeta{
Name: fmt.Sprintf("%s-engine", instance.Name),
Expand All @@ -898,9 +904,6 @@ func (r *HeatReconciler) engineDeploymentCreateOrUpdate(

op, err := controllerutil.CreateOrUpdate(ctx, r.Client, deployment, func() error {
deployment.Spec = heatEngineSpec
if len(deployment.Spec.NodeSelector) == 0 {
deployment.Spec.NodeSelector = instance.Spec.NodeSelector
}
return controllerutil.SetControllerReference(instance, deployment, r.Scheme)
})

Expand Down
4 changes: 4 additions & 0 deletions pkg/heat/dbsync.go
Original file line number Diff line number Diff line change
Expand Up @@ -82,5 +82,9 @@ func DBSyncJob(
},
}

if instance.Spec.NodeSelector != nil {
job.Spec.Template.Spec.NodeSelector = *instance.Spec.NodeSelector
}

return job
}
4 changes: 2 additions & 2 deletions pkg/heatapi/deployment.go
Original file line number Diff line number Diff line change
Expand Up @@ -169,8 +169,8 @@ func Deployment(
},
corev1.LabelHostname,
)
if instance.Spec.NodeSelector != nil && len(instance.Spec.NodeSelector) > 0 {
deployment.Spec.Template.Spec.NodeSelector = instance.Spec.NodeSelector
if instance.Spec.NodeSelector != nil {
deployment.Spec.Template.Spec.NodeSelector = *instance.Spec.NodeSelector
}

return deployment, nil
Expand Down
4 changes: 2 additions & 2 deletions pkg/heatcfnapi/deployment.go
Original file line number Diff line number Diff line change
Expand Up @@ -169,8 +169,8 @@ func Deployment(
},
corev1.LabelHostname,
)
if instance.Spec.NodeSelector != nil && len(instance.Spec.NodeSelector) > 0 {
deployment.Spec.Template.Spec.NodeSelector = instance.Spec.NodeSelector
if instance.Spec.NodeSelector != nil {
deployment.Spec.Template.Spec.NodeSelector = *instance.Spec.NodeSelector
}

return deployment, nil
Expand Down
4 changes: 2 additions & 2 deletions pkg/heatengine/deployment.go
Original file line number Diff line number Diff line change
Expand Up @@ -139,8 +139,8 @@ func Deployment(instance *heatv1beta1.HeatEngine, configHash string, labels map[
},
corev1.LabelHostname,
)
if instance.Spec.NodeSelector != nil && len(instance.Spec.NodeSelector) > 0 {
deployment.Spec.Template.Spec.NodeSelector = instance.Spec.NodeSelector
if instance.Spec.NodeSelector != nil {
deployment.Spec.Template.Spec.NodeSelector = *instance.Spec.NodeSelector
}

return deployment
Expand Down
104 changes: 103 additions & 1 deletion tests/functional/heat_controller_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,7 @@ var _ = Describe("Heat controller", func() {
var heatConfigSecretName types.NamespacedName
var memcachedSpec memcachedv1.MemcachedSpec
var keystoneAPI *keystonev1.KeystoneAPI
var heatDbSyncName types.NamespacedName

BeforeEach(func() {

Expand All @@ -66,7 +67,10 @@ var _ = Describe("Heat controller", func() {
Replicas: ptr.To[int32](3),
},
}

heatDbSyncName = types.NamespacedName{
Name: "heat-db-sync",
Namespace: namespace,
}
err := os.Setenv("OPERATOR_TEMPLATES", "../../templates")
Expect(err).NotTo(HaveOccurred())
})
Expand Down Expand Up @@ -542,6 +546,104 @@ var _ = Describe("Heat controller", func() {
})
})

When("heatAPI is created with nodeSelector", func() {
BeforeEach(func() {
spec := GetDefaultHeatSpec()
spec["nodeSelector"] = map[string]interface{}{
"foo": "bar",
}
heatAPI := GetDefaultHeatAPISpec()
spec["heatAPI"] = heatAPI
DeferCleanup(th.DeleteInstance, CreateHeat(heatName, spec))

DeferCleanup(
k8sClient.Delete, ctx, CreateHeatSecret(namespace, SecretName))
DeferCleanup(infra.DeleteMemcached, infra.CreateMemcached(namespace, "memcached", memcachedSpec))
infra.SimulateMemcachedReady(types.NamespacedName{
Name: "memcached",
Namespace: namespace,
})
DeferCleanup(
k8sClient.Delete, ctx, CreateHeatMessageBusSecret(namespace, HeatMessageBusSecretName))
infra.SimulateTransportURLReady(heatTransportURLName)
keystoneAPIName := keystone.CreateKeystoneAPI(namespace)
keystoneAPI = keystone.GetKeystoneAPI(keystoneAPIName)
DeferCleanup(keystone.DeleteKeystoneAPI, keystoneAPIName)
DeferCleanup(
mariadb.DeleteDBService,
mariadb.CreateDBService(
namespace,
GetHeat(heatName).Spec.DatabaseInstance,
corev1.ServiceSpec{
Ports: []corev1.ServicePort{{Port: 3306}},
},
),
)
mariadb.SimulateMariaDBAccountCompleted(types.NamespacedName{Namespace: namespace, Name: GetHeat(heatName).Spec.DatabaseAccount})
mariadb.SimulateMariaDBDatabaseCompleted(types.NamespacedName{Namespace: namespace, Name: heat.DatabaseCRName})
th.SimulateJobSuccess(heatDbSyncName)
// TODO: assert deployment once it's supported in the tests
})

It("sets nodeSelector in resource specs", func() {
Eventually(func(g Gomega) {
g.Expect(th.GetJob(heatDbSyncName).Spec.Template.Spec.NodeSelector).To(Equal(map[string]string{"foo": "bar"}))
}, timeout, interval).Should(Succeed())
})

It("updates nodeSelector in resource specs when changed", func() {
Eventually(func(g Gomega) {
g.Expect(th.GetJob(heatDbSyncName).Spec.Template.Spec.NodeSelector).To(Equal(map[string]string{"foo": "bar"}))
}, timeout, interval).Should(Succeed())

Eventually(func(g Gomega) {
heat := GetHeat(heatName)
newNodeSelector := map[string]string{
"foo2": "bar2",
}
heat.Spec.NodeSelector = &newNodeSelector
g.Expect(k8sClient.Update(ctx, heat)).Should(Succeed())
}, timeout, interval).Should(Succeed())

Eventually(func(g Gomega) {
g.Expect(th.GetJob(heatDbSyncName).Spec.Template.Spec.NodeSelector).To(Equal(map[string]string{"foo2": "bar2"}))
}, timeout, interval).Should(Succeed())
})

It("removes nodeSelector from resource specs when cleared", func() {
Eventually(func(g Gomega) {
g.Expect(th.GetJob(heatDbSyncName).Spec.Template.Spec.NodeSelector).To(Equal(map[string]string{"foo": "bar"}))
}, timeout, interval).Should(Succeed())

Eventually(func(g Gomega) {
heat := GetHeat(heatName)
emptyNodeSelector := map[string]string{}
heat.Spec.NodeSelector = &emptyNodeSelector
g.Expect(k8sClient.Update(ctx, heat)).Should(Succeed())
}, timeout, interval).Should(Succeed())

Eventually(func(g Gomega) {
g.Expect(th.GetJob(heatDbSyncName).Spec.Template.Spec.NodeSelector).To(BeNil())
}, timeout, interval).Should(Succeed())
})

It("removes nodeSelector from resource specs when nilled", func() {
Eventually(func(g Gomega) {
g.Expect(th.GetJob(heatDbSyncName).Spec.Template.Spec.NodeSelector).To(Equal(map[string]string{"foo": "bar"}))
}, timeout, interval).Should(Succeed())

Eventually(func(g Gomega) {
heat := GetHeat(heatName)
heat.Spec.NodeSelector = nil
g.Expect(k8sClient.Update(ctx, heat)).Should(Succeed())
}, timeout, interval).Should(Succeed())

Eventually(func(g Gomega) {
g.Expect(th.GetJob(heatDbSyncName).Spec.Template.Spec.NodeSelector).To(BeNil())
}, timeout, interval).Should(Succeed())
})
})

// Run MariaDBAccount suite tests. these are pre-packaged ginkgo tests
// that exercise standard account create / update patterns that should be
// common to all controllers that ensure MariaDBAccount CRs.
Expand Down

0 comments on commit fdeb8f1

Please sign in to comment.