Merge "Always cleanup stale devices on DHCP setup failure"
This commit is contained in:
commit
51b34f0729
@ -1291,14 +1291,22 @@ class DeviceManager(object):
|
|||||||
network.ports.append(port)
|
network.ports.append(port)
|
||||||
|
|
||||||
def _cleanup_stale_devices(self, network, dhcp_port):
|
def _cleanup_stale_devices(self, network, dhcp_port):
|
||||||
|
"""Unplug any devices found in the namespace except for dhcp_port."""
|
||||||
LOG.debug("Cleaning stale devices for network %s", network.id)
|
LOG.debug("Cleaning stale devices for network %s", network.id)
|
||||||
dev_name = self.driver.get_device_name(dhcp_port)
|
skip_dev_name = (self.driver.get_device_name(dhcp_port)
|
||||||
|
if dhcp_port else None)
|
||||||
ns_ip = ip_lib.IPWrapper(namespace=network.namespace)
|
ns_ip = ip_lib.IPWrapper(namespace=network.namespace)
|
||||||
|
if not ns_ip.netns.exists(network.namespace):
|
||||||
|
return
|
||||||
for d in ns_ip.get_devices(exclude_loopback=True):
|
for d in ns_ip.get_devices(exclude_loopback=True):
|
||||||
# delete all devices except current active DHCP port device
|
# delete all devices except current active DHCP port device
|
||||||
if d.name != dev_name:
|
if d.name != skip_dev_name:
|
||||||
LOG.debug("Found stale device %s, deleting", d.name)
|
LOG.debug("Found stale device %s, deleting", d.name)
|
||||||
self.unplug(d.name, network)
|
try:
|
||||||
|
self.unplug(d.name, network)
|
||||||
|
except Exception:
|
||||||
|
LOG.exception(_LE("Exception during stale "
|
||||||
|
"dhcp device cleanup"))
|
||||||
|
|
||||||
def plug(self, network, port, interface_name):
|
def plug(self, network, port, interface_name):
|
||||||
"""Plug device settings for the network's DHCP on this host."""
|
"""Plug device settings for the network's DHCP on this host."""
|
||||||
@ -1311,7 +1319,13 @@ class DeviceManager(object):
|
|||||||
|
|
||||||
def setup(self, network):
|
def setup(self, network):
|
||||||
"""Create and initialize a device for network's DHCP on this host."""
|
"""Create and initialize a device for network's DHCP on this host."""
|
||||||
port = self.setup_dhcp_port(network)
|
try:
|
||||||
|
port = self.setup_dhcp_port(network)
|
||||||
|
except Exception:
|
||||||
|
with excutils.save_and_reraise_exception():
|
||||||
|
# clear everything out so we don't leave dangling interfaces
|
||||||
|
# if setup never succeeds in the future.
|
||||||
|
self._cleanup_stale_devices(network, dhcp_port=None)
|
||||||
self._update_dhcp_port(network, port)
|
self._update_dhcp_port(network, port)
|
||||||
interface_name = self.get_interface_name(network, port)
|
interface_name = self.get_interface_name(network, port)
|
||||||
|
|
||||||
@ -1355,12 +1369,7 @@ class DeviceManager(object):
|
|||||||
namespace=network.namespace)
|
namespace=network.namespace)
|
||||||
|
|
||||||
self._set_default_route(network, interface_name)
|
self._set_default_route(network, interface_name)
|
||||||
try:
|
self._cleanup_stale_devices(network, port)
|
||||||
self._cleanup_stale_devices(network, port)
|
|
||||||
except Exception:
|
|
||||||
# catch everything as we don't want to fail because of
|
|
||||||
# cleanup step
|
|
||||||
LOG.error(_LE("Exception during stale dhcp device cleanup"))
|
|
||||||
|
|
||||||
return interface_name
|
return interface_name
|
||||||
|
|
||||||
|
@ -1441,6 +1441,20 @@ class TestDeviceManager(base.BaseTestCase):
|
|||||||
expected = [mock.call.add_rule('POSTROUTING', rule)]
|
expected = [mock.call.add_rule('POSTROUTING', rule)]
|
||||||
self.mangle_inst.assert_has_calls(expected)
|
self.mangle_inst.assert_has_calls(expected)
|
||||||
|
|
||||||
|
def test_setup_dhcp_port_doesnt_orphan_devices(self):
|
||||||
|
with mock.patch.object(dhcp.ip_lib, 'IPDevice') as mock_IPDevice:
|
||||||
|
plugin = mock.Mock()
|
||||||
|
device = mock.Mock()
|
||||||
|
mock_IPDevice.return_value = device
|
||||||
|
device.route.get_gateway.return_value = None
|
||||||
|
net = copy.deepcopy(fake_network)
|
||||||
|
plugin.create_dhcp_port.side_effect = exceptions.Conflict()
|
||||||
|
dh = dhcp.DeviceManager(cfg.CONF, plugin)
|
||||||
|
clean = mock.patch.object(dh, '_cleanup_stale_devices').start()
|
||||||
|
with testtools.ExpectedException(exceptions.Conflict):
|
||||||
|
dh.setup(net)
|
||||||
|
clean.assert_called_once_with(net, dhcp_port=None)
|
||||||
|
|
||||||
def test_setup_create_dhcp_port(self):
|
def test_setup_create_dhcp_port(self):
|
||||||
with mock.patch.object(dhcp.ip_lib, 'IPDevice') as mock_IPDevice:
|
with mock.patch.object(dhcp.ip_lib, 'IPDevice') as mock_IPDevice:
|
||||||
plugin = mock.Mock()
|
plugin = mock.Mock()
|
||||||
|
Loading…
x
Reference in New Issue
Block a user