Skip to content

Commit aecadfa

Browse files
committed
Create ClusterIP Service for clients access to cluster
1 parent 921377a commit aecadfa

File tree

2 files changed

+64
-1
lines changed

2 files changed

+64
-1
lines changed

internal/controller/etcdcluster_controller.go

+54
Original file line numberDiff line numberDiff line change
@@ -144,6 +144,10 @@ func (r *EtcdClusterReconciler) ensureClusterObjects(
144144
if err := r.ensureClusterStatefulSet(ctx, cluster); err != nil {
145145
return err
146146
}
147+
// 3. create or update ClusterIP Service
148+
if err := r.ensureClusterClientService(ctx, cluster); err != nil {
149+
return err
150+
}
147151

148152
return nil
149153
}
@@ -196,6 +200,52 @@ func (r *EtcdClusterReconciler) ensureClusterService(ctx context.Context, cluste
196200
return nil
197201
}
198202

203+
func (r *EtcdClusterReconciler) ensureClusterClientService(ctx context.Context, cluster *etcdaenixiov1alpha1.EtcdCluster) error {
204+
svc := &corev1.Service{}
205+
err := r.Get(ctx, client.ObjectKey{
206+
Namespace: cluster.Namespace,
207+
Name: r.getClientServiceName(cluster),
208+
}, svc)
209+
// Service exists, skip creation
210+
if err == nil {
211+
return nil
212+
}
213+
if !errors.IsNotFound(err) {
214+
return fmt.Errorf("cannot get cluster client service: %w", err)
215+
}
216+
217+
svc = &corev1.Service{
218+
ObjectMeta: metav1.ObjectMeta{
219+
Name: r.getClientServiceName(cluster),
220+
Namespace: cluster.Namespace,
221+
Labels: map[string]string{
222+
"app.kubernetes.io/name": "etcd",
223+
"app.kubernetes.io/instance": cluster.Name,
224+
"app.kubernetes.io/managed-by": "etcd-operator",
225+
},
226+
},
227+
Spec: corev1.ServiceSpec{
228+
Ports: []corev1.ServicePort{
229+
{Name: "client", TargetPort: intstr.FromInt32(2379), Port: 2379, Protocol: corev1.ProtocolTCP},
230+
},
231+
Type: corev1.ServiceTypeClusterIP,
232+
Selector: map[string]string{
233+
"app.kubernetes.io/name": "etcd",
234+
"app.kubernetes.io/instance": cluster.Name,
235+
"app.kubernetes.io/managed-by": "etcd-operator",
236+
},
237+
PublishNotReadyAddresses: false,
238+
},
239+
}
240+
if err = ctrl.SetControllerReference(cluster, svc, r.Scheme); err != nil {
241+
return fmt.Errorf("cannot set controller reference: %w", err)
242+
}
243+
if err = r.Create(ctx, svc); err != nil {
244+
return fmt.Errorf("cannot create cluster client service: %w", err)
245+
}
246+
return nil
247+
}
248+
199249
// ensureClusterStateConfigMap creates or updates cluster state configmap.
200250
func (r *EtcdClusterReconciler) ensureClusterStateConfigMap(
201251
ctx context.Context, cluster *etcdaenixiov1alpha1.EtcdCluster, isClusterInitialized bool) error {
@@ -421,6 +471,10 @@ func (r *EtcdClusterReconciler) getClusterStateConfigMapName(cluster *etcdaenixi
421471
return cluster.Name + "-cluster-state"
422472
}
423473

474+
func (r *EtcdClusterReconciler) getClientServiceName(cluster *etcdaenixiov1alpha1.EtcdCluster) string {
475+
return cluster.Name + "-client"
476+
}
477+
424478
// updateClusterState patches status condition in cluster using merge by Type
425479
func (r *EtcdClusterReconciler) updateClusterState(cluster *etcdaenixiov1alpha1.EtcdCluster, state metav1.Condition) {
426480
if initIdx := slices.IndexFunc(cluster.Status.Conditions, func(condition metav1.Condition) bool {

internal/controller/etcdcluster_controller_test.go

+10-1
Original file line numberDiff line numberDiff line change
@@ -110,12 +110,21 @@ var _ = Describe("EtcdCluster Controller", func() {
110110
// check that Service is created
111111
svc := &v1.Service{}
112112
err = k8sClient.Get(ctx, typeNamespacedName, svc)
113-
Expect(err).NotTo(HaveOccurred(), "cluster headless Service state should exist")
113+
Expect(err).NotTo(HaveOccurred(), "cluster headless Service should exist")
114114
Expect(svc.Spec.ClusterIP).To(Equal("None"), "cluster Service should be headless")
115115
// check that StatefulSet is created
116116
sts := &appsv1.StatefulSet{}
117117
err = k8sClient.Get(ctx, typeNamespacedName, sts)
118118
Expect(err).NotTo(HaveOccurred(), "cluster statefulset should exist")
119+
// check that Service is created
120+
svc = &v1.Service{}
121+
clientSvcName := types.NamespacedName{
122+
Namespace: typeNamespacedName.Namespace,
123+
Name: controllerReconciler.getClientServiceName(etcdcluster),
124+
}
125+
err = k8sClient.Get(ctx, clientSvcName, svc)
126+
Expect(err).NotTo(HaveOccurred(), "cluster client Service should exist")
127+
Expect(svc.Spec.ClusterIP).NotTo(Equal("None"), "cluster client Service should NOT be headless")
119128
})
120129

121130
It("should successfully reconcile the resource twice and mark as ready", func() {

0 commit comments

Comments
 (0)