Skip to content

Commit

Permalink
Fix inadvertent deletion of aggregated ServiceImports on agent restart
Browse files Browse the repository at this point in the history
If the broker co-exists on a managed cluster, on LH agent restart,
the aggregated ServiceImports on the broker are inadvertently deleted
during reconciliation. Reconciliation should only process local
aggregated ServiceImports and should ignore ServiceImports in the
broker namespace. The latter are distinguished by the presence of the
 "multicluster.kubernetes.io/service-name" annotation.

The reconciliation unit test was adjusted to cover this case.

Fixes submariner-io/submariner#3188

Signed-off-by: Tom Pantelis <tompantelis@gmail.com>
  • Loading branch information
tpantelis authored and skitt committed Nov 15, 2024
1 parent 22193b2 commit 04864e9
Show file tree
Hide file tree
Showing 3 changed files with 17 additions and 13 deletions.
13 changes: 8 additions & 5 deletions pkg/agent/controller/controller_suite_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -312,8 +312,8 @@ func newTestDiver() *testDriver {
t.brokerEndpointSliceClient = t.syncerConfig.BrokerClient.Resource(*test.GetGroupVersionResourceFor(t.syncerConfig.RestMapper,
&discovery.EndpointSlice{})).Namespace(test.RemoteNamespace)

t.cluster1.init(t.syncerConfig)
t.cluster2.init(t.syncerConfig)
t.cluster1.init(t.syncerConfig, nil)
t.cluster2.init(t.syncerConfig, nil)

return t
}
Expand All @@ -327,15 +327,18 @@ func (t *testDriver) afterEach() {
close(t.stopCh)
}

func (c *cluster) init(syncerConfig *broker.SyncerConfig) {
func (c *cluster) init(syncerConfig *broker.SyncerConfig, dynClient *dynamicfake.FakeDynamicClient) {
for k, v := range c.service.Labels {
c.serviceEndpointSlices[0].Labels[k] = v
}

c.serviceIP = c.service.Spec.ClusterIP

c.localDynClient = dynamicfake.NewSimpleDynamicClient(syncerConfig.Scheme)
fake.AddBasicReactors(&c.localDynClient.Fake)
c.localDynClient = dynClient
if c.localDynClient == nil {
c.localDynClient = dynamicfake.NewSimpleDynamicClient(syncerConfig.Scheme)
fake.AddBasicReactors(&c.localDynClient.Fake)
}

c.localServiceImportReactor = fake.NewFailingReactorForResource(&c.localDynClient.Fake, "serviceimports")

Expand Down
13 changes: 7 additions & 6 deletions pkg/agent/controller/reconciliation_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -105,6 +105,11 @@ var _ = Describe("Reconciliation", func() {
t.afterEach()
t = newTestDiver()

brokerDynClient := t.syncerConfig.BrokerClient.(*fake.FakeDynamicClient)

// Use the broker client for cluster1 to simulate the broker being on the same cluster.
t.cluster1.init(t.syncerConfig, brokerDynClient)

test.CreateResource(t.cluster1.localServiceImportClient.Namespace(test.LocalNamespace), localServiceImport)
test.CreateResource(t.cluster1.localEndpointSliceClient, localEndpointSlice)
test.CreateResource(t.cluster1.localServiceExportClient, serviceExport)
Expand All @@ -117,12 +122,6 @@ var _ = Describe("Reconciliation", func() {
t.cluster1.start(t, *t.syncerConfig)
t.cluster2.start(t, *t.syncerConfig)

t.awaitNonHeadlessServiceExported(&t.cluster1)

testutil.EnsureNoActionsForResource(&t.cluster1.localDynClient.Fake, "serviceimports", "delete")
testutil.EnsureNoActionsForResource(&t.cluster1.localDynClient.Fake, "endpointslices", "delete")

brokerDynClient := t.syncerConfig.BrokerClient.(*fake.FakeDynamicClient)
testutil.EnsureNoActionsForResource(&brokerDynClient.Fake, "endpointslices", "delete")

// For migration cleanup, it may attempt to delete a local legacy ServiceImport from the broker so ignore it.
Expand All @@ -137,6 +136,8 @@ var _ = Describe("Reconciliation", func() {

return false
}).Should(BeFalse())

t.awaitNonHeadlessServiceExported(&t.cluster1)
})
})

Expand Down
4 changes: 2 additions & 2 deletions pkg/agent/controller/service_import.go
Original file line number Diff line number Diff line change
Expand Up @@ -197,8 +197,8 @@ func (c *ServiceImportController) reconcileLocalAggregatedServiceImports() {
for i := range siList.Items {
si := c.converter.toServiceImport(&siList.Items[i])

if serviceImportSourceName(si) != "" {
// This is not an aggregated ServiceImport.
if serviceImportSourceName(si) != "" || si.Annotations[mcsv1a1.LabelServiceName] != "" {
// This is not a local aggregated ServiceImport.
continue
}

Expand Down

0 comments on commit 04864e9

Please sign in to comment.