Move instance power state check to _detach_with_retry

Based on comments in I7f2b6330decb92e2838aa7cee47fb228f00f47da the power
state check now can be moved to be in a common place, in
_detach_with_retry, instead of being at multiple places on the caller
side. So this patch moves the that logic.

Change-Id: I7ef2a2e29704c365241f70347b53cbb7a13c458d
(cherry picked from commit 257b4d83d9)
This commit is contained in:
Balazs Gibizer 2021-03-05 14:37:08 +01:00
parent 14596ca30f
commit ebf1ceb7d6
2 changed files with 73 additions and 82 deletions

View File

@ -9321,7 +9321,7 @@ class LibvirtConnTestCase(test.NoDBTestCase,
# Second time don't return anything about disk vdc so it looks removed
return_list = [
mock_xml_with_disk, # presistent domain
mock_xml_with_disk, # persistent domain
mock_xml_with_disk, # live domain
mock_xml_without_disk, # persistent gone
mock_xml_without_disk # live gone
@ -9467,7 +9467,6 @@ class LibvirtConnTestCase(test.NoDBTestCase,
# see assert below
mock.ANY,
device_name='vdc',
live=True
),
mock.call.detach_encryptor(**encryption),
mock.call.disconnect_volume(connection_info, instance)])
@ -21064,6 +21063,7 @@ class TraitsComparisonMixin(object):
self.assertEqual(exp, actual)
@ddt.ddt
class LibvirtDriverTestCase(test.NoDBTestCase, TraitsComparisonMixin):
"""Test for nova.virt.libvirt.libvirt_driver.LibvirtDriver."""
def setUp(self):
@ -22960,6 +22960,7 @@ class LibvirtDriverTestCase(test.NoDBTestCase, TraitsComparisonMixin):
expected_cfg, # detach_interface() itself gets the config
expected_cfg, # _detach_with_retry: persistent config
None, # _detach_with_retry: no device in live config
None, # _detach_with_retry: persistent config gone as detached
]
else:
if state in (power_state.RUNNING, power_state.PAUSED):
@ -22986,13 +22987,12 @@ class LibvirtDriverTestCase(test.NoDBTestCase, TraitsComparisonMixin):
mock.patch.object(guest, 'get_interface_by_cfg',
side_effect=get_interface_calls),
mock.patch.object(domain, 'detachDeviceFlags'),
mock.patch('nova.virt.libvirt.driver.LOG.warning'),
mock.patch.object(self.drvr.vif_driver, 'unplug'),
mock.patch.object(instance, 'get_network_info')
) as (
mock_get_guest, mock_get_config,
mock_get_interface, mock_detach_device_flags,
mock_warning, mock_unplug, mock_get_network_info
mock_get_interface, mock_detach_device_flags, mock_unplug,
mock_get_network_info
):
# run the detach method
self.drvr.detach_interface(self.context, instance, network_info[0])
@ -23003,14 +23003,16 @@ class LibvirtDriverTestCase(test.NoDBTestCase, TraitsComparisonMixin):
instance, network_info[0], test.MatchType(objects.ImageMeta),
test.MatchType(objects.Flavor), CONF.libvirt.virt_type)
if device_not_found:
mock_detach_device_flags.assert_not_called()
self.assertTrue(mock_warning.called)
mock_detach_device_flags.assert_called_once_with(
expected_cfg.to_xml(),
flags=fakelibvirt.VIR_DOMAIN_AFFECT_CONFIG)
mock_get_interface.assert_has_calls(
[
mock.call(expected_cfg),
mock.call(expected_cfg, from_persistent_config=True),
mock.call(expected_cfg),
])
[
mock.call(expected_cfg),
mock.call(expected_cfg, from_persistent_config=True),
mock.call(expected_cfg),
mock.call(expected_cfg, from_persistent_config=True),
])
else:
mock_get_network_info.assert_called_once_with()
if state in (power_state.RUNNING, power_state.PAUSED):
@ -23210,7 +23212,6 @@ class LibvirtDriverTestCase(test.NoDBTestCase, TraitsComparisonMixin):
self.drvr.vif_driver, 'get_config', return_value=cfg),
mock.patch.object(
guest, 'get_interface_by_cfg', return_value=interface),
mock.patch.object(guest, 'get_power_state'),
mock.patch.object(
instance, 'get_network_info', return_value=network_info),
mock.patch.object(
@ -23220,9 +23221,8 @@ class LibvirtDriverTestCase(test.NoDBTestCase, TraitsComparisonMixin):
mock.patch.object(guest, 'set_metadata')
) as (
mock_get_guest, mock_get_config, mock_get_interface_by_cfg,
mock_get_power_state, mock_get_network_info,
mock_detach_with_retry, mock_get_guest_config_meta,
mock_set_metadata
mock_get_network_info, mock_detach_with_retry,
mock_get_guest_config_meta, mock_set_metadata
):
self.drvr.detach_interface(self.context, instance, vif)
mock_get_guest.assert_called_once_with(instance)
@ -23230,10 +23230,8 @@ class LibvirtDriverTestCase(test.NoDBTestCase, TraitsComparisonMixin):
instance, vif, test.MatchType(objects.ImageMeta),
test.MatchType(objects.Flavor), CONF.libvirt.virt_type)
mock_get_interface_by_cfg.assert_called_once_with(cfg)
mock_get_power_state.assert_called_once_with(self.drvr._host)
mock_detach_with_retry.assert_called_once_with(
guest, instance.uuid, mock.ANY, live=False,
device_name=None)
guest, instance.uuid, mock.ANY, device_name=None)
mock_get_network_info.assert_called_once_with()
mock_get_guest_config_meta.assert_called_once_with(
instance, network_info[1:])
@ -23243,6 +23241,7 @@ class LibvirtDriverTestCase(test.NoDBTestCase, TraitsComparisonMixin):
"""Test detach only from the persistent domain"""
drvr = libvirt_driver.LibvirtDriver(fake.FakeVirtAPI(), False)
mock_guest = mock.Mock(spec=libvirt_guest.Guest)
mock_guest.get_power_state.return_value = power_state.SHUTDOWN
mock_guest.has_persistent_configuration.return_value = True
mock_dev = mock.Mock(spec=vconfig.LibvirtConfigGuestDisk)
@ -23263,7 +23262,6 @@ class LibvirtDriverTestCase(test.NoDBTestCase, TraitsComparisonMixin):
uuids.instance_uuid,
mock_get_device_conf_func,
device_name='vdb',
live=False,
)
mock_guest.has_persistent_configuration.assert_called_once_with()
@ -23279,10 +23277,12 @@ class LibvirtDriverTestCase(test.NoDBTestCase, TraitsComparisonMixin):
# check that the internal event handling is cleaned up
self.assertEqual(set(), drvr._device_event_handler._waiters)
def test__detach_with_retry_live_success(self):
@ddt.data(power_state.RUNNING, power_state.PAUSED)
def test__detach_with_retry_live_success(self, state):
"""Test detach only from the live domain"""
drvr = libvirt_driver.LibvirtDriver(fake.FakeVirtAPI(), False)
mock_guest = mock.Mock(spec=libvirt_guest.Guest)
mock_guest.get_power_state.return_value = state
mock_guest.has_persistent_configuration.return_value = False
mock_dev = mock.Mock(spec=vconfig.LibvirtConfigGuestDisk)
@ -23308,7 +23308,6 @@ class LibvirtDriverTestCase(test.NoDBTestCase, TraitsComparisonMixin):
uuids.instance_uuid,
mock_get_device_conf_func,
device_name='vdb',
live=True,
)
mock_guest.has_persistent_configuration.assert_called_once_with()
@ -23324,10 +23323,12 @@ class LibvirtDriverTestCase(test.NoDBTestCase, TraitsComparisonMixin):
# check that the internal event handling is cleaned up
self.assertEqual(set(), drvr._device_event_handler._waiters)
def test__detach_with_retry_persistent_and_live_success(self):
@ddt.data(power_state.RUNNING, power_state.PAUSED)
def test__detach_with_retry_persistent_and_live_success(self, state):
"""Test detach both from the persistent and from the live domains"""
drvr = libvirt_driver.LibvirtDriver(fake.FakeVirtAPI(), False)
mock_guest = mock.Mock(spec=libvirt_guest.Guest)
mock_guest.get_power_state.return_value = state
mock_guest.has_persistent_configuration.return_value = True
mock_dev = mock.Mock(spec=vconfig.LibvirtConfigGuestDisk)
@ -23362,7 +23363,6 @@ class LibvirtDriverTestCase(test.NoDBTestCase, TraitsComparisonMixin):
uuids.instance_uuid,
mock_get_device_conf_func,
device_name='vdb',
live=True,
)
mock_guest.has_persistent_configuration.assert_called_once_with()
@ -23383,11 +23383,12 @@ class LibvirtDriverTestCase(test.NoDBTestCase, TraitsComparisonMixin):
self.assertEqual(set(), drvr._device_event_handler._waiters)
def test__detach_with_retry_persistent_but_guest_is_not(self):
"""Test that a pesistent only detach is requested but the guest is
"""Test that a persistent only detach is requested but the guest is
not persistent.
"""
drvr = libvirt_driver.LibvirtDriver(fake.FakeVirtAPI(), False)
mock_guest = mock.Mock(spec=libvirt_guest.Guest)
mock_guest.get_power_state.return_value = power_state.SHUTDOWN
mock_guest.has_persistent_configuration.return_value = False
# we expect that no device query or actual detach happens.
@ -23399,17 +23400,17 @@ class LibvirtDriverTestCase(test.NoDBTestCase, TraitsComparisonMixin):
uuids.instance_uuid,
mock_get_device_conf_func,
device_name='vdb',
live=False,
)
mock_guest.has_persistent_configuration.assert_called_once_with()
def test__detach_with_retry_persistent_dev_not_found(self):
"""Tests that expection is raised if a persistent detach is requested,
"""Tests that exception is raised if a persistent detach is requested,
the domain is persistent but the device is not found.
"""
drvr = libvirt_driver.LibvirtDriver(fake.FakeVirtAPI(), False)
mock_guest = mock.Mock(spec=libvirt_guest.Guest)
mock_guest.get_power_state.return_value = power_state.SHUTDOWN
mock_guest.has_persistent_configuration.return_value = True
# we simulate that the device does not exists in the domain even
@ -23423,19 +23424,20 @@ class LibvirtDriverTestCase(test.NoDBTestCase, TraitsComparisonMixin):
uuids.instance_uuid,
mock_get_device_conf_func,
device_name='vdb',
live=False,
)
mock_get_device_conf_func.assert_called_once_with(
from_persistent_config=True)
mock_guest.detach_device.assert_not_called()
def test__detach_with_retry_live_dev_not_found(self):
"""Tests that expection is raised if a live detach is requested,
@ddt.data(power_state.RUNNING, power_state.PAUSED)
def test__detach_with_retry_live_dev_not_found(self, state):
"""Tests that exception is raised if a live detach is requested,
but the device is not found in the live domain.
"""
drvr = libvirt_driver.LibvirtDriver(fake.FakeVirtAPI(), False)
mock_guest = mock.Mock(spec=libvirt_guest.Guest)
mock_guest.get_power_state.return_value = state
mock_guest.has_persistent_configuration.return_value = False
# we simulate that the device does not exists in the domain even
@ -23449,17 +23451,18 @@ class LibvirtDriverTestCase(test.NoDBTestCase, TraitsComparisonMixin):
uuids.instance_uuid,
mock_get_device_conf_func,
device_name='vdb',
live=True,
)
mock_get_device_conf_func.assert_called_once_with()
mock_guest.detach_device.assert_not_called()
def test__detach_with_retry_async_fail(self):
@ddt.data(power_state.RUNNING, power_state.PAUSED)
def test__detach_with_retry_async_fail(self, state):
"""Test that libvirt sends error event during detach"""
drvr = libvirt_driver.LibvirtDriver(fake.FakeVirtAPI(), False)
mock_guest = mock.Mock(spec=libvirt_guest.Guest)
# for simplycity do a live only detach
mock_guest.get_power_state.return_value = state
# for simplicity do a live only detach
mock_guest.has_persistent_configuration.return_value = False
mock_dev = mock.Mock(spec=vconfig.LibvirtConfigGuestDisk)
@ -23480,7 +23483,6 @@ class LibvirtDriverTestCase(test.NoDBTestCase, TraitsComparisonMixin):
uuids.instance_uuid,
mock_get_device_conf_func,
device_name='vdb',
live=True,
)
mock_guest.has_persistent_configuration.assert_called_once_with()
@ -23491,13 +23493,17 @@ class LibvirtDriverTestCase(test.NoDBTestCase, TraitsComparisonMixin):
self.assertEqual(set(), drvr._device_event_handler._waiters)
@mock.patch('threading.Event.wait')
def test__detach_with_retry_timeout_retry_succeeds(self, mock_event_wait):
@ddt.data(power_state.RUNNING, power_state.PAUSED)
def test__detach_with_retry_timeout_retry_succeeds(
self, state, mock_event_wait
):
"""Test that that a live detach times out while waiting for the libvirt
event but then the retry succeeds.
"""
drvr = libvirt_driver.LibvirtDriver(fake.FakeVirtAPI(), False)
mock_guest = mock.Mock(spec=libvirt_guest.Guest)
# for simplycity do a live only detach
mock_guest.get_power_state.return_value = state
# for simplicity do a live only detach
mock_guest.has_persistent_configuration.return_value = False
mock_dev = mock.Mock(spec=vconfig.LibvirtConfigGuestDisk)
@ -23528,7 +23534,6 @@ class LibvirtDriverTestCase(test.NoDBTestCase, TraitsComparisonMixin):
uuids.instance_uuid,
mock_get_device_conf_func,
device_name='vdb',
live=True,
)
mock_guest.has_persistent_configuration.assert_called_once_with()
@ -23554,6 +23559,7 @@ class LibvirtDriverTestCase(test.NoDBTestCase, TraitsComparisonMixin):
"""
drvr = libvirt_driver.LibvirtDriver(fake.FakeVirtAPI(), False)
mock_guest = mock.Mock(spec=libvirt_guest.Guest)
mock_guest.get_power_state.return_value = power_state.RUNNING
# for simplicity do a live only detach
mock_guest.has_persistent_configuration.return_value = False
@ -23607,7 +23613,6 @@ class LibvirtDriverTestCase(test.NoDBTestCase, TraitsComparisonMixin):
uuids.instance_uuid,
mock_get_device_conf_func,
device_name='vdb',
live=True,
)
mock_guest.has_persistent_configuration.assert_called_once_with()
@ -23623,8 +23628,9 @@ class LibvirtDriverTestCase(test.NoDBTestCase, TraitsComparisonMixin):
self.assertEqual(set(), drvr._device_event_handler._waiters)
@mock.patch('threading.Event.wait')
@ddt.data(power_state.RUNNING, power_state.PAUSED)
def test__detach_with_retry_timeout_run_out_of_retries(
self, mock_event_wait
self, state, mock_event_wait
):
"""Test that that a live detach times out while waiting for the libvirt
event at every attempt so the driver runs out of retry attempts.
@ -23634,7 +23640,8 @@ class LibvirtDriverTestCase(test.NoDBTestCase, TraitsComparisonMixin):
drvr = libvirt_driver.LibvirtDriver(fake.FakeVirtAPI(), False)
mock_guest = mock.Mock(spec=libvirt_guest.Guest)
# for simplycity do a live only detach
mock_guest.get_power_state.return_value = state
# for simplicity do a live only detach
mock_guest.has_persistent_configuration.return_value = False
mock_dev = mock.Mock(spec=vconfig.LibvirtConfigGuestDisk)
@ -23655,7 +23662,6 @@ class LibvirtDriverTestCase(test.NoDBTestCase, TraitsComparisonMixin):
uuids.instance_uuid,
mock_get_device_conf_func,
device_name='vdb',
live=True,
)
mock_guest.has_persistent_configuration.assert_called_once_with()
@ -23671,8 +23677,9 @@ class LibvirtDriverTestCase(test.NoDBTestCase, TraitsComparisonMixin):
# check that the internal event handling is cleaned up
self.assertEqual(set(), drvr._device_event_handler._waiters)
@ddt.data(power_state.RUNNING, power_state.PAUSED)
def test__detach_with_retry_libvirt_reports_not_found_live_detach_done(
self
self, state
):
"""Test that libvirt reports that the device is not found in the
persistent domain but as live detach is also requested the original
@ -23680,6 +23687,7 @@ class LibvirtDriverTestCase(test.NoDBTestCase, TraitsComparisonMixin):
"""
drvr = libvirt_driver.LibvirtDriver(fake.FakeVirtAPI(), False)
mock_guest = mock.Mock(spec=libvirt_guest.Guest)
mock_guest.get_power_state.return_value = state
mock_guest.has_persistent_configuration.return_value = True
mock_dev = mock.Mock(spec=vconfig.LibvirtConfigGuestDisk)
@ -23723,7 +23731,6 @@ class LibvirtDriverTestCase(test.NoDBTestCase, TraitsComparisonMixin):
uuids.instance_uuid,
mock_get_device_conf_func,
device_name='vdb',
live=True,
)
mock_guest.has_persistent_configuration.assert_called_once_with()
@ -23749,7 +23756,7 @@ class LibvirtDriverTestCase(test.NoDBTestCase, TraitsComparisonMixin):
"""
drvr = libvirt_driver.LibvirtDriver(fake.FakeVirtAPI(), False)
mock_guest = mock.Mock(spec=libvirt_guest.Guest)
mock_guest.has_persistent_configuration.return_value = True
mock_guest.get_power_state.return_value = power_state.SHUTDOWN
mock_dev = mock.Mock(spec=vconfig.LibvirtConfigGuestDisk)
mock_dev.alias = 'virtio-disk1'
@ -23776,7 +23783,6 @@ class LibvirtDriverTestCase(test.NoDBTestCase, TraitsComparisonMixin):
uuids.instance_uuid,
mock_get_device_conf_func,
device_name='vdb',
live=False,
)
mock_guest.has_persistent_configuration.assert_called_once_with()
@ -23785,26 +23791,21 @@ class LibvirtDriverTestCase(test.NoDBTestCase, TraitsComparisonMixin):
mock_guest.detach_device.assert_called_once_with(
mock_dev, persistent=True, live=False)
def test__detach_with_retry_other_sync_libvirt_error(self):
@ddt.data(power_state.RUNNING, power_state.PAUSED)
def test__detach_with_retry_other_sync_libvirt_error(self, state):
"""Test that libvirt reports non device related error during detach
from persistent config which causes that the driver gives up
regardless of the fact that live detach also requested.
regardless of the fact that the instance has a live domain.
"""
drvr = libvirt_driver.LibvirtDriver(fake.FakeVirtAPI(), False)
mock_guest = mock.Mock(spec=libvirt_guest.Guest)
mock_guest.get_power_state.return_value = state
mock_guest.has_persistent_configuration.return_value = True
mock_dev = mock.Mock(spec=vconfig.LibvirtConfigGuestDisk)
mock_dev.alias = 'virtio-disk1'
mock_get_device_conf_func = mock.Mock(
# the first call is to get the device from the persistent domain
# the second all is to get the device from the live domain
side_effect=[
mock_dev,
mock_dev,
]
)
mock_get_device_conf_func = mock.Mock(return_value=mock_dev)
mock_guest.detach_device.side_effect = fakelibvirt.make_libvirtError(
fakelibvirt.libvirtError,
msg='error',
@ -23817,17 +23818,17 @@ class LibvirtDriverTestCase(test.NoDBTestCase, TraitsComparisonMixin):
uuids.instance_uuid,
mock_get_device_conf_func,
device_name='vdb',
live=True,
)
mock_guest.has_persistent_configuration.assert_called_once_with()
mock_get_device_conf_func.assert_has_calls(
[
mock.call(from_persistent_config=True),
mock.call(),
])
mock.call()
]
)
# NOTE(gibi): the drive does not attempt a live detach even though
# it was requested
# the instance has a live domain
mock_guest.detach_device.assert_called_once_with(
mock_dev, persistent=True, live=False)

View File

@ -2228,12 +2228,14 @@ class LibvirtDriver(driver.ComputeDriver):
# that is only available since python 3.8
get_device_conf_func: ty.Callable,
device_name: str,
live: bool,
) -> None:
"""Detaches a device from the guest
If live detach is requested then this call will wait for the libvirt
event signalling the end of the detach process.
If the guest is a running state then the detach is performed on both
the persistent and live domains.
In case of live detach this call will wait for the libvirt event
signalling the end of the detach process.
If the live detach times out then it will retry the detach. Detach from
the persistent config is not retried as it is:
@ -2251,10 +2253,6 @@ class LibvirtDriver(driver.ComputeDriver):
:param device_name: This is the name of the device used solely for
error messages. Note that it is not the same as the device alias
used by libvirt to identify the device.
:param live: bool to indicate whether it affects the guest in running
state. If live is True then the device is detached from both the
persistent and the live config. If live is False the device only
detached from the persistent config.
:raises exception.DeviceNotFound: if the device does not exist in the
domain even before we try to detach or if libvirt reported that the
device is missing from the domain synchronously.
@ -2264,6 +2262,9 @@ class LibvirtDriver(driver.ComputeDriver):
:raises libvirt.libvirtError: for any other errors reported by libvirt
synchronously.
"""
state = guest.get_power_state(self._host)
live = state in (power_state.RUNNING, power_state.PAUSED)
persistent = guest.has_persistent_configuration()
if not persistent and not live:
@ -2278,13 +2279,8 @@ class LibvirtDriver(driver.ComputeDriver):
if live:
live_dev = get_device_conf_func()
if live and live_dev is None:
# caller requested a live detach but device is not present
raise exception.DeviceNotFound(device=device_name)
if not live and persistent_dev is None:
# caller requested detach from the persistent domain but device is
# not present
# didn't find the device in either domain
if persistent_dev is None and live_dev is None:
raise exception.DeviceNotFound(device=device_name)
if persistent_dev:
@ -2293,19 +2289,19 @@ class LibvirtDriver(driver.ComputeDriver):
guest, instance_uuid, persistent_dev, get_device_conf_func,
device_name)
except exception.DeviceNotFound:
if live:
if live_dev:
# ignore the error so that we can do the live detach
LOG.warning(
'Libvirt reported sync error while detaching '
'device %s from instance %s from the persistent '
'domain config. Ignoring the error to proceed with '
'live detach as that was also requested.',
'live detach as the device exists in the live domain.',
device_name, instance_uuid)
else:
# if only persistent detach was requested then give up
raise
if live and live_dev:
if live_dev:
self._detach_from_live_with_retry(
guest, instance_uuid, live_dev, get_device_conf_func,
device_name)
@ -2541,8 +2537,6 @@ class LibvirtDriver(driver.ComputeDriver):
try:
guest = self._host.get_guest(instance)
state = guest.get_power_state(self._host)
live = state in (power_state.RUNNING, power_state.PAUSED)
# NOTE(lyarwood): The volume must be detached from the VM before
# detaching any attached encryptors or disconnecting the underlying
# volume in _disconnect_volume. Otherwise, the encryptor or volume
@ -2553,7 +2547,6 @@ class LibvirtDriver(driver.ComputeDriver):
instance.uuid,
get_dev,
device_name=disk_dev,
live=live
)
except exception.InstanceNotFound:
# NOTE(zhaoqin): If the instance does not exist, _lookup_by_name()
@ -2759,15 +2752,12 @@ class LibvirtDriver(driver.ComputeDriver):
{'mac': mac}, instance=instance)
return
state = guest.get_power_state(self._host)
live = state in (power_state.RUNNING, power_state.PAUSED)
get_dev = functools.partial(guest.get_interface_by_cfg, cfg)
self._detach_with_retry(
guest,
instance.uuid,
get_dev,
device_name=self.vif_driver.get_vif_devname(vif),
live=live,
)
except exception.DeviceDetachFailed:
# We failed to detach the device even with the retry loop, so let's