Fix amphorav1 member deletion bug

When the worker calculates the delta for plugging/unplugging subnets,
it need to check the provisioning_status of the members. But in
amphorav1, this status comes from the LB DB object that is passed to
taskflow. If the provisioning_status is modified by another task
(MarkMemberPendingDeleteInDb), the LB object is not updated.
We need to reload it from the DB to get the current status of the
members.

Note: this patch is for stable branches only as the amphorav1 driver was
      removed during the Bobcat cycle
Closes-Bug: #2027967
Change-Id: Iea0ffee1a7307b0852c5a9faf31739c96e76a7ee
This commit is contained in:
Gregory Thiemonge 2023-07-17 11:02:08 +02:00
parent 3ac52a0b2b
commit 8b196bb3bb
3 changed files with 27 additions and 3 deletions

View File

@ -70,6 +70,11 @@ class CalculateAmphoraDelta(BaseNetworkTask):
else: else:
management_nets = CONF.controller_worker.amp_boot_network_list management_nets = CONF.controller_worker.amp_boot_network_list
# Reload the load balancer, the provisioning status of the members may
# have been updated by a previous task
loadbalancer = self.lb_repo.get(
db_apis.get_session(), id=loadbalancer.id)
desired_subnet_to_net_map = {} desired_subnet_to_net_map = {}
for mgmt_net_id in management_nets: for mgmt_net_id in management_nets:
for subnet_id in self.network_driver.get_network( for subnet_id in self.network_driver.get_network(

View File

@ -65,6 +65,7 @@ AMPS_DATA = [o_data_models.Amphora(id=t_constants.MOCK_AMP_ID1,
vrrp_ip=t_constants.MOCK_VRRP_IP3) vrrp_ip=t_constants.MOCK_VRRP_IP3)
] ]
UPDATE_DICT = {constants.TOPOLOGY: None} UPDATE_DICT = {constants.TOPOLOGY: None}
_session_mock = mock.MagicMock()
class TestException(Exception): class TestException(Exception):
@ -98,7 +99,10 @@ class TestNetworkTasks(base.TestCase):
conf.config(group="networking", max_retries=1) conf.config(group="networking", max_retries=1)
super().setUp() super().setUp()
def test_calculate_amphora_delta(self, mock_get_net_driver): @mock.patch('octavia.db.repositories.LoadBalancerRepository.get')
@mock.patch('octavia.db.api.get_session', return_value=_session_mock)
def test_calculate_amphora_delta(self, mock_get_session, mock_lb_repo_get,
mock_get_net_driver):
VRRP_PORT_ID = uuidutils.generate_uuid() VRRP_PORT_ID = uuidutils.generate_uuid()
VIP_NETWORK_ID = uuidutils.generate_uuid() VIP_NETWORK_ID = uuidutils.generate_uuid()
VIP_SUBNET_ID = uuidutils.generate_uuid() VIP_SUBNET_ID = uuidutils.generate_uuid()
@ -155,6 +159,7 @@ class TestNetworkTasks(base.TestCase):
id=mock.Mock(), id=mock.Mock(),
network_id=DELETE_NETWORK_ID) network_id=DELETE_NETWORK_ID)
mock_lb_repo_get.return_value = lb_mock
mock_driver.get_port.return_value = vrrp_port mock_driver.get_port.return_value = vrrp_port
mock_driver.get_subnet.return_value = member_subnet mock_driver.get_subnet.return_value = member_subnet
mock_driver.get_network.return_value = mgmt_net mock_driver.get_network.return_value = mgmt_net
@ -177,8 +182,12 @@ class TestNetworkTasks(base.TestCase):
mock_driver.get_subnet.assert_called_once_with(MEMBER_SUBNET_ID) mock_driver.get_subnet.assert_called_once_with(MEMBER_SUBNET_ID)
mock_driver.get_plugged_networks.assert_called_once_with(COMPUTE_ID) mock_driver.get_plugged_networks.assert_called_once_with(COMPUTE_ID)
def test_calculate_delta(self, mock_get_net_driver): @mock.patch('octavia.db.repositories.LoadBalancerRepository.get')
@mock.patch('octavia.db.api.get_session', return_value=_session_mock)
def test_calculate_delta(self, mock_get_session, mock_get_lb,
mock_get_net_driver):
mock_driver = mock.MagicMock() mock_driver = mock.MagicMock()
mock_get_lb.return_value = self.load_balancer_mock
mock_get_net_driver.return_value = mock_driver mock_get_net_driver.return_value = mock_driver
empty_deltas = {self.amphora_mock.id: data_models.Delta( empty_deltas = {self.amphora_mock.id: data_models.Delta(
amphora_id=self.amphora_mock.id, amphora_id=self.amphora_mock.id,
@ -572,7 +581,11 @@ class TestNetworkTasks(base.TestCase):
self.assertEqual({self.amphora_mock.id: ndm}, self.assertEqual({self.amphora_mock.id: ndm},
calc_delta.execute(self.load_balancer_mock, {})) calc_delta.execute(self.load_balancer_mock, {}))
def test_calculate_delta_ipv6_ipv4_subnets(self, mock_get_net_driver): @mock.patch('octavia.db.repositories.LoadBalancerRepository.get')
@mock.patch('octavia.db.api.get_session', return_value=_session_mock)
def test_calculate_delta_ipv6_ipv4_subnets(self, mock_get_session,
mock_get_lb,
mock_get_net_driver):
mock_driver = mock.MagicMock() mock_driver = mock.MagicMock()
mock_get_net_driver.return_value = mock_driver mock_get_net_driver.return_value = mock_driver

View File

@ -0,0 +1,6 @@
---
fixes:
- |
Fixed a bug in amphorav1, the subnet of a member that was being deleted was
not immediately unplugged from the amphora, but only during the next update
of the members.