Merge "Add localboot support for iscsi_irmc driver"

This commit is contained in:
Jenkins 2015-07-07 22:25:52 +00:00 committed by Gerrit Code Review
commit 8e3d876b39
2 changed files with 128 additions and 21 deletions

View File

@ -756,15 +756,49 @@ class VendorPassthru(base.VendorInterface):
:raises: InvalidParameterValue, if any of the parameters have invalid
value.
"""
iscsi_deploy.get_deploy_info(task.node, **kwargs)
if method == 'pass_deploy_info':
iscsi_deploy.get_deploy_info(task.node, **kwargs)
elif method == 'pass_bootloader_install_info':
iscsi_deploy.validate_pass_bootloader_info_input(task, kwargs)
@base.passthru(['POST'])
@task_manager.require_exclusive_lock
def pass_bootloader_install_info(self, task, **kwargs):
"""Accepts the results of bootloader installation.
This method acts as a vendor passthru and accepts the result of
bootloader installation. If the bootloader installation was
successful, then it notifies the baremetal to proceed to reboot
and makes the instance active. If bootloader installation failed,
then it sets provisioning as failed and powers off the node.
:param task: A TaskManager object.
:param kwargs: The arguments sent with vendor passthru. The expected
kwargs are::
'key': The deploy key for authorization
'status': 'SUCCEEDED' or 'FAILED'
'error': The error message if status == 'FAILED'
'address': The IP address of the ramdisk
"""
task.process_event('resume')
iscsi_deploy.validate_bootloader_install_status(task, kwargs)
iscsi_deploy.finish_deploy(task, kwargs['address'])
@base.passthru(['POST'])
@task_manager.require_exclusive_lock
def pass_deploy_info(self, task, **kwargs):
"""Continues the iSCSI deployment from where ramdisk left off.
Continues the iSCSI deployment from the conductor node, finds the
boot ISO to boot the node, and sets the node to boot from boot ISO.
This method continues the iSCSI deployment from the conductor node
and writes the deploy image to the bare metal's disk. After that,
it does the following depending on boot_option for deploy:
- If the boot_option requested for this deploy is 'local', then it
sets the node to boot from disk (ramdisk installs the boot loader
present within the image to the bare metal's disk).
- If the boot_option requested is 'netboot' or no boot_option is
requested, it finds/creates the boot ISO to boot the instance
image, attaches the boot ISO to the bare metal and then sets
the node to boot from CDROM.
:param task: a TaskManager instance containing the node to act on.
:param kwargs: kwargs containing parameters for iSCSI deployment.
@ -778,20 +812,29 @@ class VendorPassthru(base.VendorInterface):
try:
_cleanup_vmedia_boot(task)
_prepare_boot_iso(task, root_uuid)
setup_vmedia_for_boot(task,
node.driver_internal_info['irmc_boot_iso'])
manager_utils.node_set_boot_device(task, boot_devices.CDROM)
if iscsi_deploy.get_boot_option(node) == "local":
manager_utils.node_set_boot_device(task, boot_devices.DISK,
persistent=True)
address = kwargs.get('address')
deploy_utils.notify_ramdisk_to_proceed(address)
# Ask the ramdisk to install bootloader and
# wait for the call-back through the vendor passthru
# 'pass_bootloader_install_info'.
deploy_utils.notify_ramdisk_to_proceed(kwargs['address'])
task.process_event('wait')
return
LOG.info(_LI('Deployment to node %s done'), node.uuid)
else:
_prepare_boot_iso(task, root_uuid)
setup_vmedia_for_boot(
task, node.driver_internal_info['irmc_boot_iso'])
manager_utils.node_set_boot_device(task, boot_devices.CDROM,
persistent=True)
task.process_event('done')
except Exception as e:
LOG.exception(_LE('Deploy failed for instance %(instance)s. '
'Error: %(error)s'),
{'instance': node.instance_uuid, 'error': e})
msg = _('Failed to continue iSCSI deployment.')
deploy_utils.set_failed_state(task, msg)
else:
iscsi_deploy.finish_deploy(task, kwargs.get('address'))

View File

@ -1000,13 +1000,41 @@ class VendorPassthruTestCase(db_base.DbTestCase):
CONF.irmc.remote_image_user_password = 'admin0'
CONF.irmc.remote_image_user_domain = 'local'
@mock.patch.object(iscsi_deploy, 'get_deploy_info')
def test_validate(self, get_deploy_info_mock):
@mock.patch.object(iscsi_deploy, 'get_deploy_info', spec_set=True,
autospec=True)
def test_validate_pass_deploy_info(self, get_deploy_info_mock):
with task_manager.acquire(self.context, self.node.uuid,
shared=False) as task:
task.driver.vendor.validate(task, method=None, a=1)
task.driver.vendor.validate(task, method='pass_deploy_info', a=1)
get_deploy_info_mock.assert_called_once_with(task.node, a=1)
@mock.patch.object(iscsi_deploy, 'validate_pass_bootloader_info_input',
spec_set=True, autospec=True)
def test_validate_pass_bootloader_install_info(self, validate_mock):
with task_manager.acquire(self.context, self.node.uuid,
shared=True) as task:
kwargs = {'address': '1.2.3.4', 'key': 'fake-key',
'status': 'SUCCEEDED', 'error': ''}
task.driver.vendor.validate(
task, method='pass_bootloader_install_info', **kwargs)
validate_mock.assert_called_once_with(task, kwargs)
@mock.patch.object(iscsi_deploy, 'validate_bootloader_install_status',
spec_set=True, autospec=True)
@mock.patch.object(iscsi_deploy, 'finish_deploy', spec_set=True,
autospec=True)
def test_pass_bootloader_install_info(self, finish_deploy_mock,
validate_input_mock):
kwargs = {'method': 'pass_deploy_info', 'address': '123456'}
self.node.provision_state = states.DEPLOYWAIT
self.node.target_provision_state = states.ACTIVE
self.node.save()
with task_manager.acquire(self.context, self.node.uuid,
shared=False) as task:
task.driver.vendor.pass_bootloader_install_info(task, **kwargs)
finish_deploy_mock.assert_called_once_with(task, '123456')
validate_input_mock.assert_called_once_with(task, kwargs)
@mock.patch.object(deploy_utils, 'set_failed_state', spec_set=True,
autospec=True)
@mock.patch.object(deploy_utils, 'notify_ramdisk_to_proceed',
@ -1041,15 +1069,14 @@ class VendorPassthruTestCase(db_base.DbTestCase):
task.driver.vendor.pass_deploy_info(task, **kwargs)
_cleanup_vmedia_boot_mock.assert_called_once_with(task)
continue_deploy_mock.assert_called_once_with(
task, method='pass_deploy_info', address='123456')
continue_deploy_mock.assert_called_once_with(task, **kwargs)
_prepare_boot_iso_mock.assert_called_once_with(
task, 'root_uuid')
setup_vmedia_for_boot_mock.assert_called_once_with(
task, 'irmc_boot.iso')
node_set_boot_device_mock.assert_called_once_with(
task, boot_devices.CDROM)
task, boot_devices.CDROM, persistent=True)
notify_ramdisk_to_proceed_mock.assert_called_once_with(
'123456')
@ -1101,7 +1128,7 @@ class VendorPassthruTestCase(db_base.DbTestCase):
self.assertFalse(notify_ramdisk_to_proceed_mock.called)
self.assertFalse(set_failed_state_mock.called)
@mock.patch.object(deploy_utils, 'set_failed_state', spec_set=True,
@mock.patch.object(manager_utils, 'node_power_action', spec_set=True,
autospec=True)
@mock.patch.object(deploy_utils, 'notify_ramdisk_to_proceed',
spec_set=True, autospec=True)
@ -1123,7 +1150,7 @@ class VendorPassthruTestCase(db_base.DbTestCase):
setup_vmedia_for_boot_mock,
node_set_boot_device_mock,
notify_ramdisk_to_proceed_mock,
set_failed_state_mock):
node_power_mock):
kwargs = {'method': 'pass_deploy_info', 'address': '123456'}
continue_deploy_mock.return_value = {'root uuid': 'root_uuid'}
_prepare_boot_iso_mock.side_effect = Exception("fake error")
@ -1137,12 +1164,49 @@ class VendorPassthruTestCase(db_base.DbTestCase):
continue_deploy_mock.assert_called_once_with(
task, method='pass_deploy_info', address='123456')
_cleanup_vmedia_boot_mock.assert_called_once_with(task)
_prepare_boot_iso_mock.assert_called_once_with(
task, 'root_uuid')
self.assertEqual(states.DEPLOYFAIL, task.node.provision_state)
self.assertEqual(states.ACTIVE, task.node.target_provision_state)
self.assertFalse(setup_vmedia_for_boot_mock.called)
self.assertFalse(node_set_boot_device_mock.called)
self.assertFalse(notify_ramdisk_to_proceed_mock.called)
node_power_mock.assert_called_once_with(task, states.POWER_OFF)
msg = _('Failed to continue iSCSI deployment.')
set_failed_state_mock.assert_called_once_with(task, msg)
@mock.patch.object(deploy_utils, 'notify_ramdisk_to_proceed',
spec_set=True, autospec=True)
@mock.patch.object(manager_utils, 'node_set_boot_device', spec_set=True,
autospec=True)
@mock.patch.object(iscsi_deploy, 'continue_deploy', spec_set=True,
autospec=True)
@mock.patch.object(irmc_deploy, '_cleanup_vmedia_boot', spec_set=True,
autospec=True)
def test_pass_deploy_info_localboot(self,
_cleanup_vmedia_boot_mock,
continue_deploy_mock,
set_boot_device_mock,
notify_ramdisk_to_proceed_mock):
kwargs = {'method': 'pass_deploy_info', 'address': '123456'}
continue_deploy_mock.return_value = {'root uuid': '<some-uuid>'}
self.node.provision_state = states.DEPLOYWAIT
self.node.target_provision_state = states.ACTIVE
self.node.instance_info = {'capabilities': '{"boot_option": "local"}'}
self.node.save()
with task_manager.acquire(self.context, self.node.uuid,
shared=False) as task:
vendor = task.driver.vendor
vendor.pass_deploy_info(task, **kwargs)
_cleanup_vmedia_boot_mock.assert_called_once_with(task)
continue_deploy_mock.assert_called_once_with(task, **kwargs)
set_boot_device_mock.assert_called_once_with(task,
boot_devices.DISK,
persistent=True)
notify_ramdisk_to_proceed_mock.assert_called_once_with('123456')
self.assertEqual(states.DEPLOYWAIT, task.node.provision_state)
self.assertEqual(states.ACTIVE, task.node.target_provision_state)