@@ -144,6 +144,10 @@ func (r *EtcdClusterReconciler) ensureClusterObjects(
144
144
if err := r .ensureClusterStatefulSet (ctx , cluster ); err != nil {
145
145
return err
146
146
}
147
+ // 3. create or update ClusterIP Service
148
+ if err := r .ensureClusterClientService (ctx , cluster ); err != nil {
149
+ return err
150
+ }
147
151
148
152
return nil
149
153
}
@@ -196,6 +200,52 @@ func (r *EtcdClusterReconciler) ensureClusterService(ctx context.Context, cluste
196
200
return nil
197
201
}
198
202
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
+
199
249
// ensureClusterStateConfigMap creates or updates cluster state configmap.
200
250
func (r * EtcdClusterReconciler ) ensureClusterStateConfigMap (
201
251
ctx context.Context , cluster * etcdaenixiov1alpha1.EtcdCluster , isClusterInitialized bool ) error {
@@ -421,6 +471,10 @@ func (r *EtcdClusterReconciler) getClusterStateConfigMapName(cluster *etcdaenixi
421
471
return cluster .Name + "-cluster-state"
422
472
}
423
473
474
+ func (r * EtcdClusterReconciler ) getClientServiceName (cluster * etcdaenixiov1alpha1.EtcdCluster ) string {
475
+ return cluster .Name + "-client"
476
+ }
477
+
424
478
// updateClusterState patches status condition in cluster using merge by Type
425
479
func (r * EtcdClusterReconciler ) updateClusterState (cluster * etcdaenixiov1alpha1.EtcdCluster , state metav1.Condition ) {
426
480
if initIdx := slices .IndexFunc (cluster .Status .Conditions , func (condition metav1.Condition ) bool {
0 commit comments