diff --git a/mocks/pkg/ovs/interface.go b/mocks/pkg/ovs/interface.go index 98cbc0553c9..30921d3efb2 100644 --- a/mocks/pkg/ovs/interface.go +++ b/mocks/pkg/ovs/interface.go @@ -5111,21 +5111,6 @@ func (mr *MockSbClientMockRecorder) DeleteChassisByHost(node any) *gomock.Call { return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "DeleteChassisByHost", reflect.TypeOf((*MockSbClient)(nil).DeleteChassisByHost), node) } -// GetAllChassisByHost mocks base method. -func (m *MockSbClient) GetAllChassisByHost(nodeName string) (*[]ovnsb.Chassis, error) { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "GetAllChassisByHost", nodeName) - ret0, _ := ret[0].(*[]ovnsb.Chassis) - ret1, _ := ret[1].(error) - return ret0, ret1 -} - -// GetAllChassisByHost indicates an expected call of GetAllChassisByHost. -func (mr *MockSbClientMockRecorder) GetAllChassisByHost(nodeName any) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "GetAllChassisByHost", reflect.TypeOf((*MockSbClient)(nil).GetAllChassisByHost), nodeName) -} - // GetChassis mocks base method. func (m *MockSbClient) GetChassis(chassisName string, ignoreNotFound bool) (*ovnsb.Chassis, error) { m.ctrl.T.Helper() @@ -5351,21 +5336,6 @@ func (mr *MockChassisMockRecorder) DeleteChassisByHost(node any) *gomock.Call { return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "DeleteChassisByHost", reflect.TypeOf((*MockChassis)(nil).DeleteChassisByHost), node) } -// GetAllChassisByHost mocks base method. -func (m *MockChassis) GetAllChassisByHost(nodeName string) (*[]ovnsb.Chassis, error) { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "GetAllChassisByHost", nodeName) - ret0, _ := ret[0].(*[]ovnsb.Chassis) - ret1, _ := ret[1].(error) - return ret0, ret1 -} - -// GetAllChassisByHost indicates an expected call of GetAllChassisByHost. -func (mr *MockChassisMockRecorder) GetAllChassisByHost(nodeName any) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "GetAllChassisByHost", reflect.TypeOf((*MockChassis)(nil).GetAllChassisByHost), nodeName) -} - // GetChassis mocks base method. func (m *MockChassis) GetChassis(chassisName string, ignoreNotFound bool) (*ovnsb.Chassis, error) { m.ctrl.T.Helper() diff --git a/pkg/controller/node.go b/pkg/controller/node.go index 343d6731685..9a75d58f2eb 100644 --- a/pkg/controller/node.go +++ b/pkg/controller/node.go @@ -666,19 +666,18 @@ func (c *Controller) checkGatewayReady() error { func (c *Controller) cleanDuplicatedChassis(node *v1.Node) error { // if multi chassis has the same node name, delete all of them - chassises, err := c.OVNSbClient.GetAllChassisByHost(node.Name) - if err != nil { - klog.Errorf("failed to list chassis %v", err) - return err + var err error + if _, err := c.OVNSbClient.GetChassisByHost(node.Name); err == nil { + return nil } - if len(*chassises) > 1 { - klog.Warningf("node %s has multiple chassis", node.Name) + klog.Errorf("failed to get chassis for node %s: %v", node.Name, err) + if errors.Is(err, ovs.ErrOneNodeMultiChassis) { if err := c.OVNSbClient.DeleteChassisByHost(node.Name); err != nil { klog.Errorf("failed to delete chassis for node %s: %v", node.Name, err) return err } } - return nil + return err } func (c *Controller) retryDelDupChassis(attempts, sleep int, f func(node *v1.Node) error, node *v1.Node) (err error) { diff --git a/pkg/ovs/interface.go b/pkg/ovs/interface.go index 67a230b9a16..6f5710ab6c9 100644 --- a/pkg/ovs/interface.go +++ b/pkg/ovs/interface.go @@ -257,7 +257,6 @@ type Common interface { type Chassis interface { DeleteChassis(chassisName string) error DeleteChassisByHost(node string) error - GetAllChassisByHost(nodeName string) (*[]ovnsb.Chassis, error) GetChassisByHost(nodeName string) (*ovnsb.Chassis, error) GetChassis(chassisName string, ignoreNotFound bool) (*ovnsb.Chassis, error) GetKubeOvnChassisses() (*[]ovnsb.Chassis, error) diff --git a/pkg/ovs/ovn-nb-suite_test.go b/pkg/ovs/ovn-nb-suite_test.go index 054d39f8607..01e5b495d31 100644 --- a/pkg/ovs/ovn-nb-suite_test.go +++ b/pkg/ovs/ovn-nb-suite_test.go @@ -31,6 +31,7 @@ type OvnClientTestSuite struct { ovnSBClient *OVNSbClient failedOvnNBClient *OVNNbClient + failedOvnSBClient *OVNSbClient ovnLegacyClient *LegacyClient ovsSocket string @@ -40,6 +41,10 @@ func emptyNbDatabaseModel() (model.ClientDBModel, error) { return model.NewClientDBModel("OVN_Northbound", map[string]model.Model{}) } +func emptySbDatabaseModel() (model.ClientDBModel, error) { + return model.NewClientDBModel("OVN_Southbound", map[string]model.Model{}) +} + func (suite *OvnClientTestSuite) SetupSuite() { fmt.Println("set up ovn client test suite") // setup ovn nb client schema @@ -73,6 +78,21 @@ func (suite *OvnClientTestSuite) SetupSuite() { // setup ovn sb client sbClientSchema := ovnsb.Schema() + + // setup failed case ovn sb client + emptySbDBModel, err := emptySbDatabaseModel() + require.NoError(suite.T(), err) + + fakeSBServer, sbSock1 := newOVSDBServer(suite.T(), "fake-sb", emptySbDBModel, sbClientSchema) + sbEndpoint1 := fmt.Sprintf("unix:%s", sbSock1) + require.FileExists(suite.T(), sbSock1) + failedOvnSBClient, err := newOvnSbClient(suite.T(), sbEndpoint1, 10) + require.NoError(suite.T(), err) + suite.failedOvnSBClient = failedOvnSBClient + // close the server to simulate the failed case + fakeSBServer.Close() + require.NoFileExists(suite.T(), sbSock1) + sbClientDBModel, err := ovnsb.FullDatabaseModel() require.NoError(suite.T(), err) @@ -1024,10 +1044,6 @@ func (suite *OvnClientTestSuite) Test_ListChassis() { suite.testListChassis() } -func (suite *OvnClientTestSuite) Test_GetAllChassisByHost() { - suite.testGetAllChassisByHost() -} - func (suite *OvnClientTestSuite) Test_GetChassisByHost() { suite.testGetChassisByHost() } diff --git a/pkg/ovs/ovn-sb-chassis.go b/pkg/ovs/ovn-sb-chassis.go index 137d1474781..05bcdf2891e 100644 --- a/pkg/ovs/ovn-sb-chassis.go +++ b/pkg/ovs/ovn-sb-chassis.go @@ -12,6 +12,8 @@ import ( "github.com/kubeovn/kube-ovn/pkg/util" ) +var ErrOneNodeMultiChassis = errors.New("OneNodeMultiChassis") + func (c *OVNSbClient) UpdateChassis(chassis *ovnsb.Chassis, fields ...interface{}) error { op, err := c.ovsDbClient.Where(chassis).Update(chassis, fields...) if err != nil { @@ -83,30 +85,6 @@ func (c *OVNSbClient) ListChassis() (*[]ovnsb.Chassis, error) { return &css, nil } -func (c *OVNSbClient) GetAllChassisByHost(nodeName string) (*[]ovnsb.Chassis, error) { - ctx, cancel := context.WithTimeout(context.Background(), c.Timeout) - defer cancel() - - chassisList := make([]ovnsb.Chassis, 0) - if err := c.ovsDbClient.WhereCache(func(chassis *ovnsb.Chassis) bool { - return chassis.Hostname == nodeName - }).List(ctx, &chassisList); err != nil { - klog.Error(err) - return nil, fmt.Errorf("failed to list Chassis with host name=%s: %w", nodeName, err) - } - if len(chassisList) == 0 { - err := fmt.Errorf("failed to get Chassis with with host name=%s", nodeName) - klog.Error(err) - return nil, err - } - if len(chassisList) != 1 { - err := fmt.Errorf("found more than one Chassis with with host name=%s", nodeName) - klog.Error(err) - return nil, err - } - return &chassisList, nil -} - func (c *OVNSbClient) GetChassisByHost(nodeName string) (*ovnsb.Chassis, error) { if nodeName == "" { err := errors.New("failed to get Chassis with empty hostname") @@ -131,7 +109,7 @@ func (c *OVNSbClient) GetChassisByHost(nodeName string) (*ovnsb.Chassis, error) if len(chassisList) != 1 { err := fmt.Errorf("found more than one Chassis with host name=%s", nodeName) klog.Error(err) - return nil, err + return nil, ErrOneNodeMultiChassis } // #nosec G602 diff --git a/pkg/ovs/ovn-sb-chassis_test.go b/pkg/ovs/ovn-sb-chassis_test.go index 013726253d9..87bf6b88be1 100644 --- a/pkg/ovs/ovn-sb-chassis_test.go +++ b/pkg/ovs/ovn-sb-chassis_test.go @@ -112,6 +112,7 @@ func (suite *OvnClientTestSuite) testUpdateChassis() { t.Parallel() sbClient := suite.ovnSBClient + failSBClient := suite.failedOvnSBClient t.Cleanup(func() { err := sbClient.DeleteChassis("chassis-name-3") @@ -140,6 +141,13 @@ func (suite *OvnClientTestSuite) testUpdateChassis() { require.Error(t, err) require.Contains(t, err.Error(), "failed to generate update operations for chassis") }) + + t.Run("test failed client update chassis with valid fields", func(t *testing.T) { + updatedHostname := "updated-host-name" + chassis.Hostname = updatedHostname + err := failSBClient.UpdateChassis(chassis, &chassis.Hostname) + require.NoError(t, err) + }) } func (suite *OvnClientTestSuite) testListChassis() { @@ -147,6 +155,7 @@ func (suite *OvnClientTestSuite) testListChassis() { t.Parallel() sbClient := suite.ovnSBClient + failSBClient := suite.failedOvnSBClient t.Cleanup(func() { err := sbClient.DeleteChassis("chassis-1") @@ -197,104 +206,77 @@ func (suite *OvnClientTestSuite) testListChassis() { require.NotContains(t, names, "chassis-1") require.NotContains(t, names, "chassis-2") }) + + t.Run("test failed client list chassis with no entries", func(t *testing.T) { + _, err := failSBClient.ListChassis() + require.NoError(t, err) + }) } -func (suite *OvnClientTestSuite) testGetAllChassisByHost() { +func (suite *OvnClientTestSuite) testGetChassisByHost() { t := suite.T() t.Parallel() sbClient := suite.ovnSBClient + failSBClient := suite.failedOvnSBClient t.Cleanup(func() { - err := sbClient.DeleteChassis("chassis-3") + err := sbClient.DeleteChassis("chassis-3x") require.NoError(t, err) - err = sbClient.DeleteChassis("chassis-4") + err = sbClient.DeleteChassis("chassis-4x") require.NoError(t, err) - err = sbClient.DeleteChassis("chassis-5") + err = sbClient.DeleteChassis("chassis-5x") + require.NoError(t, err) + err = sbClient.DeleteChassis("chassis-6x") + require.NoError(t, err) + err = sbClient.DeleteChassis("chassis-7x") require.NoError(t, err) }) - chassis1 := newChassis(0, "host-3", "chassis-3", nil, nil, nil, nil, nil) - chassis2 := newChassis(0, "host-4", "chassis-4", nil, nil, nil, nil, nil) - chassis3 := newChassis(0, "host-4", "chassis-5", nil, nil, nil, nil, nil) + chassis3x := newChassis(0, "host-3x", "chassis-3x", nil, nil, nil, nil, nil) + chassis4x := newChassis(0, "host-4x", "chassis-4x", nil, nil, nil, nil, nil) + chassis5x := newChassis(0, "host-5x", "chassis-5x", nil, nil, nil, nil, nil) + chassis6x := newChassis(0, "host-6x", "chassis-6x", nil, nil, nil, nil, nil) + chassis7x := newChassis(0, "host-7x", "chassis-7x", nil, nil, nil, nil, nil) - ops1, err := sbClient.ovsDbClient.Create(chassis1) + ops3x, err := sbClient.ovsDbClient.Create(chassis3x) require.NoError(t, err) - err = sbClient.Transact("chassis-add", ops1) + err = sbClient.Transact("chassis-add", ops3x) require.NoError(t, err) - ops2, err := sbClient.ovsDbClient.Create(chassis2) + ops4x, err := sbClient.ovsDbClient.Create(chassis4x) require.NoError(t, err) - err = sbClient.Transact("chassis-add", ops2) + err = sbClient.Transact("chassis-add", ops4x) require.NoError(t, err) - ops3, err := sbClient.ovsDbClient.Create(chassis3) + ops5x, err := sbClient.ovsDbClient.Create(chassis5x) require.NoError(t, err) - err = sbClient.Transact("chassis-add", ops3) + err = sbClient.Transact("chassis-add", ops5x) require.NoError(t, err) - t.Run("test get all chassis by host with single chassis", func(t *testing.T) { - chassisList, err := sbClient.GetAllChassisByHost("host-3") - require.NoError(t, err) - require.NotNil(t, chassisList) - require.Len(t, *chassisList, 1) - require.Equal(t, "chassis-3", (*chassisList)[0].Name) - }) - - t.Run("test get all chassis by host with multiple chassis", func(t *testing.T) { - chassisList, err := sbClient.GetAllChassisByHost("host-4") - require.Error(t, err) - require.Nil(t, chassisList) - require.Contains(t, err.Error(), "found more than one Chassis") - }) - - t.Run("test get all chassis by non-existent host", func(t *testing.T) { - chassisList, err := sbClient.GetAllChassisByHost("non-existent-host") - require.Error(t, err) - require.Nil(t, chassisList) - require.Contains(t, err.Error(), "failed to get Chassis") - }) - - t.Run("test get all chassis by host with empty hostname", func(t *testing.T) { - chassisList, err := sbClient.GetAllChassisByHost("") - require.Error(t, err) - require.Nil(t, chassisList) - require.Contains(t, err.Error(), "failed to get Chassis") - }) -} - -func (suite *OvnClientTestSuite) testGetChassisByHost() { - t := suite.T() - t.Parallel() - - sbClient := suite.ovnSBClient - - t.Cleanup(func() { - err := sbClient.DeleteChassis("chassis-6") - require.NoError(t, err) - err = sbClient.DeleteChassis("chassis-7") - require.NoError(t, err) - }) - - chassis1 := newChassis(0, "host-6", "chassis-6", nil, nil, nil, nil, nil) - chassis2 := newChassis(0, "host-7", "chassis-7", nil, nil, nil, nil, nil) - - ops1, err := sbClient.ovsDbClient.Create(chassis1) + ops6x, err := sbClient.ovsDbClient.Create(chassis6x) require.NoError(t, err) - err = sbClient.Transact("chassis-add", ops1) + err = sbClient.Transact("chassis-add", ops6x) require.NoError(t, err) - ops2, err := sbClient.ovsDbClient.Create(chassis2) + ops7x, err := sbClient.ovsDbClient.Create(chassis7x) require.NoError(t, err) - err = sbClient.Transact("chassis-add", ops2) + err = sbClient.Transact("chassis-add", ops7x) require.NoError(t, err) + t.Run("test get all chassis by host with single chassis", func(t *testing.T) { + chassis, err := sbClient.GetChassisByHost("host-3x") + require.NoError(t, err) + require.NotNil(t, chassis) + require.Equal(t, "chassis-3x", chassis.Name) + }) + t.Run("test get chassis by host with valid hostname", func(t *testing.T) { - chassis, err := sbClient.GetChassisByHost("host-6") + chassis, err := sbClient.GetChassisByHost("host-6x") require.NoError(t, err) require.NotNil(t, chassis) - require.Equal(t, "chassis-6", chassis.Name) - require.Equal(t, "host-6", chassis.Hostname) + require.Equal(t, "chassis-6x", chassis.Name) + require.Equal(t, "host-6x", chassis.Hostname) }) t.Run("test get chassis by host with non-existent hostname", func(t *testing.T) { @@ -312,20 +294,25 @@ func (suite *OvnClientTestSuite) testGetChassisByHost() { }) t.Run("test get chassis by host with multiple chassis", func(t *testing.T) { - chassis3 := newChassis(0, "host-6", "chassis-8", nil, nil, nil, nil, nil) - ops3, err := sbClient.ovsDbClient.Create(chassis3) + chassis66 := newChassis(0, "host-6x", "chassis-6x-fake", nil, nil, nil, nil, nil) + ops3, err := sbClient.ovsDbClient.Create(chassis66) require.NoError(t, err) err = sbClient.Transact("chassis-add", ops3) require.NoError(t, err) - chassis, err := sbClient.GetChassisByHost("host-6") + chassis, err := sbClient.GetChassisByHost("host-6x") require.Error(t, err) require.Nil(t, chassis) - require.Contains(t, err.Error(), "found more than one Chassis") + require.Contains(t, err.Error(), "OneNodeMultiChassis") - err = sbClient.DeleteChassis("chassis-8") + err = sbClient.DeleteChassis("chassis-6x-fake") require.NoError(t, err) }) + + t.Run("test failed client get chassis by host with non-existent hostname", func(t *testing.T) { + _, err := failSBClient.GetChassisByHost("non-existent-host") + require.Error(t, err) + }) } func (suite *OvnClientTestSuite) testDeleteChassisByHost() { @@ -333,6 +320,7 @@ func (suite *OvnClientTestSuite) testDeleteChassisByHost() { t.Parallel() sbClient := suite.ovnSBClient + failSBClient := suite.failedOvnSBClient t.Cleanup(func() { err := sbClient.DeleteChassis("chassis-node1-1") @@ -374,24 +362,6 @@ func (suite *OvnClientTestSuite) testDeleteChassisByHost() { err = sbClient.Transact("chassis-add", ops4) require.NoError(t, err) - t.Run("test delete chassis by host with multiple chassis", func(t *testing.T) { - err := sbClient.DeleteChassisByHost("node1") - require.NoError(t, err) - - chassisList, err := sbClient.GetAllChassisByHost("node1") - require.ErrorContains(t, err, "failed to get Chassis with with host name") - require.Nil(t, chassisList) - }) - - t.Run("test delete chassis by host with ExternalIDs", func(t *testing.T) { - err := sbClient.DeleteChassisByHost("node2") - require.NoError(t, err) - - chassisList, err := sbClient.GetAllChassisByHost("node2") - require.ErrorContains(t, err, "failed to get Chassis with with host name") - require.Nil(t, chassisList) - }) - t.Run("test delete chassis by non-existent host", func(t *testing.T) { err := sbClient.DeleteChassisByHost("non-existent-node") require.NoError(t, err) @@ -406,6 +376,16 @@ func (suite *OvnClientTestSuite) testDeleteChassisByHost() { err := sbClient.DeleteChassisByHost("node3") require.ErrorContains(t, err, "chassis name is empty") }) + + t.Run("test failed client delete chassis by non-existent host", func(t *testing.T) { + err := failSBClient.DeleteChassisByHost("non-existent-node") + require.NoError(t, err) + }) + + t.Run("test failed client delete chassis by empty chassis name with ExternalIDs", func(t *testing.T) { + err := failSBClient.DeleteChassisByHost("node3") + require.NoError(t, err) + }) } func (suite *OvnClientTestSuite) testUpdateChassisTag() { @@ -413,6 +393,7 @@ func (suite *OvnClientTestSuite) testUpdateChassisTag() { t.Parallel() sbClient := suite.ovnSBClient + failSBClient := suite.failedOvnSBClient t.Cleanup(func() { err := sbClient.DeleteChassis("chassis-update-tag") @@ -472,6 +453,40 @@ func (suite *OvnClientTestSuite) testUpdateChassisTag() { require.Equal(t, "", updatedChassis.ExternalIDs["node"]) require.Equal(t, util.CniTypeName, updatedChassis.ExternalIDs["vendor"]) }) + + t.Run("test failed client update chassis tag with empty node name", func(t *testing.T) { + err := failSBClient.UpdateChassisTag("chassis-update-tag", "") + require.Error(t, err) + }) + + t.Run("test failed client update chassis tag with new node name", func(t *testing.T) { + err := failSBClient.UpdateChassisTag("chassis-update-tag", "new-node-name") + require.Error(t, err) + }) + + t.Run("test failed client update chassis tag with existing ExternalIDs", func(t *testing.T) { + chassis.ExternalIDs = map[string]string{"existing": "value"} + err := failSBClient.UpdateChassis(chassis, &chassis.ExternalIDs) + require.NoError(t, err) + + err = failSBClient.UpdateChassisTag("chassis-update-tag", "another-node-name") + require.Error(t, err) + }) + + t.Run("test failed client update chassis tag with non-existent chassis", func(t *testing.T) { + err := failSBClient.UpdateChassisTag("non-existent-chassis", "node-name") + require.Error(t, err) + }) + + t.Run("test failed client update chassis tag with empty chassis name", func(t *testing.T) { + err := failSBClient.UpdateChassisTag("", "node-name") + require.Error(t, err) + }) + + t.Run("test failed client update chassis tag with empty node name", func(t *testing.T) { + err := failSBClient.UpdateChassisTag("chassis-update-tag", "") + require.Error(t, err) + }) } func (suite *OvnClientTestSuite) testGetKubeOvnChassisses() { @@ -479,27 +494,7 @@ func (suite *OvnClientTestSuite) testGetKubeOvnChassisses() { t.Parallel() sbClient := suite.ovnSBClient - - t.Cleanup(func() { - err := sbClient.DeleteChassis("kube-ovn-chassis-1") - require.NoError(t, err) - err = sbClient.DeleteChassis("kube-ovn-chassis-2") - require.NoError(t, err) - err = sbClient.DeleteChassis("non-kube-ovn-chassis") - require.NoError(t, err) - err = sbClient.DeleteChassis("mixed-chassis") - require.NoError(t, err) - chassisList, err := sbClient.GetKubeOvnChassisses() - require.NoError(t, err) - names := make(map[string]bool) - for _, chassis := range *chassisList { - names[chassis.Name] = true - } - require.False(t, names["kube-ovn-chassis-1"]) - require.False(t, names["kube-ovn-chassis-2"]) - require.False(t, names["non-kube-ovn-chassis"]) - require.False(t, names["mixed-chassis"]) - }) + failSBClient := suite.failedOvnSBClient kubeOvnChassis1 := newChassis(0, "host-1", "kube-ovn-chassis-1", nil, nil, nil, map[string]string{"vendor": util.CniTypeName}, nil) kubeOvnChassis2 := newChassis(0, "host-2", "kube-ovn-chassis-2", nil, nil, nil, map[string]string{"vendor": util.CniTypeName}, nil) @@ -540,4 +535,29 @@ func (suite *OvnClientTestSuite) testGetKubeOvnChassisses() { require.True(t, names["kube-ovn-chassis-2"]) require.False(t, names["non-kube-ovn-chassis"]) require.True(t, names["mixed-chassis"]) + + // failed client test + _, err = failSBClient.GetKubeOvnChassisses() + require.NoError(t, err) + + t.Cleanup(func() { + err := sbClient.DeleteChassis("kube-ovn-chassis-1") + require.NoError(t, err) + err = sbClient.DeleteChassis("kube-ovn-chassis-2") + require.NoError(t, err) + err = sbClient.DeleteChassis("non-kube-ovn-chassis") + require.NoError(t, err) + err = sbClient.DeleteChassis("mixed-chassis") + require.NoError(t, err) + chassisList, err := sbClient.GetKubeOvnChassisses() + require.NoError(t, err) + names := make(map[string]bool) + for _, chassis := range *chassisList { + names[chassis.Name] = true + } + require.False(t, names["kube-ovn-chassis-1"]) + require.False(t, names["kube-ovn-chassis-2"]) + require.False(t, names["non-kube-ovn-chassis"]) + require.False(t, names["mixed-chassis"]) + }) }