ffca0e5fb6
And remove corresponding H210 filters. Change-Id: I074eac9088f4739f540b57d07fc487cb87194599
1171 lines
56 KiB
Python
1171 lines
56 KiB
Python
# Copyright 2013 Hewlett-Packard Development Company, L.P.
|
|
# All Rights Reserved.
|
|
#
|
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
|
# not use this file except in compliance with the License. You may obtain
|
|
# a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
# License for the specific language governing permissions and limitations
|
|
# under the License.
|
|
|
|
"""Test class for PXE driver."""
|
|
|
|
import os
|
|
import tempfile
|
|
from unittest import mock
|
|
|
|
from oslo_config import cfg
|
|
from oslo_serialization import jsonutils as json
|
|
from oslo_utils import timeutils
|
|
from oslo_utils import uuidutils
|
|
|
|
from ironic.common import boot_devices
|
|
from ironic.common import boot_modes
|
|
from ironic.common import dhcp_factory
|
|
from ironic.common import exception
|
|
from ironic.common.glance_service import image_service
|
|
from ironic.common import pxe_utils
|
|
from ironic.common import states
|
|
from ironic.conductor import task_manager
|
|
from ironic.conductor import utils as manager_utils
|
|
from ironic.drivers import base as drivers_base
|
|
from ironic.drivers.modules import agent_base
|
|
from ironic.drivers.modules import deploy_utils
|
|
from ironic.drivers.modules import fake
|
|
from ironic.drivers.modules import ipxe
|
|
from ironic.drivers.modules import pxe
|
|
from ironic.drivers.modules import pxe_base
|
|
from ironic.drivers.modules.storage import noop as noop_storage
|
|
from ironic.tests.unit.db import base as db_base
|
|
from ironic.tests.unit.db import utils as db_utils
|
|
from ironic.tests.unit.objects import utils as obj_utils
|
|
|
|
CONF = cfg.CONF
|
|
|
|
INST_INFO_DICT = db_utils.get_test_pxe_instance_info()
|
|
DRV_INFO_DICT = db_utils.get_test_pxe_driver_info()
|
|
DRV_INTERNAL_INFO_DICT = db_utils.get_test_pxe_driver_internal_info()
|
|
|
|
|
|
# NOTE(TheJulia): Mark pxe interface loading as None in order
|
|
# to prent false counts for individual method tests.
|
|
@mock.patch.object(ipxe.iPXEBoot, '__init__', lambda self: None)
|
|
@mock.patch.object(pxe.PXEBoot, '__init__', lambda self: None)
|
|
class PXEBootTestCase(db_base.DbTestCase):
|
|
|
|
driver = 'fake-hardware'
|
|
boot_interface = 'pxe'
|
|
driver_info = DRV_INFO_DICT
|
|
driver_internal_info = DRV_INTERNAL_INFO_DICT
|
|
|
|
def setUp(self):
|
|
super(PXEBootTestCase, self).setUp()
|
|
self.context.auth_token = 'fake'
|
|
self.config_temp_dir('tftp_root', group='pxe')
|
|
self.config_temp_dir('images_path', group='pxe')
|
|
self.config_temp_dir('http_root', group='deploy')
|
|
instance_info = INST_INFO_DICT
|
|
instance_info['deploy_key'] = 'fake-56789'
|
|
|
|
self.config(enabled_boot_interfaces=[self.boot_interface,
|
|
'ipxe', 'fake'])
|
|
self.node = obj_utils.create_test_node(
|
|
self.context,
|
|
driver=self.driver,
|
|
boot_interface=self.boot_interface,
|
|
# Avoid fake properties in get_properties() output
|
|
vendor_interface='no-vendor',
|
|
instance_info=instance_info,
|
|
driver_info=self.driver_info,
|
|
driver_internal_info=self.driver_internal_info)
|
|
self.port = obj_utils.create_test_port(self.context,
|
|
node_id=self.node.id)
|
|
self.config(my_ipv6='2001:db8::1')
|
|
|
|
def test_get_properties(self):
|
|
expected = pxe_base.COMMON_PROPERTIES
|
|
expected.update(agent_base.VENDOR_PROPERTIES)
|
|
with task_manager.acquire(self.context, self.node.uuid,
|
|
shared=True) as task:
|
|
self.assertEqual(expected, task.driver.get_properties())
|
|
|
|
@mock.patch.object(image_service.GlanceImageService, 'show', autospec=True)
|
|
def test_validate_good(self, mock_glance):
|
|
mock_glance.return_value = {'properties': {'kernel_id': 'fake-kernel',
|
|
'ramdisk_id': 'fake-initr'}}
|
|
with task_manager.acquire(self.context, self.node.uuid,
|
|
shared=True) as task:
|
|
task.driver.boot.validate(task)
|
|
|
|
@mock.patch.object(image_service.GlanceImageService, 'show', autospec=True)
|
|
def test_validate_good_whole_disk_image(self, mock_glance):
|
|
with task_manager.acquire(self.context, self.node.uuid,
|
|
shared=True) as task:
|
|
task.node.driver_internal_info['is_whole_disk_image'] = True
|
|
task.driver.boot.validate(task)
|
|
|
|
@mock.patch.object(image_service.GlanceImageService, 'show', autospec=True)
|
|
@mock.patch.object(noop_storage.NoopStorage, 'should_write_image',
|
|
autospec=True)
|
|
def test_validate_skip_check_write_image_false(self, mock_write,
|
|
mock_glance):
|
|
mock_write.return_value = False
|
|
with task_manager.acquire(self.context, self.node.uuid,
|
|
shared=True) as task:
|
|
task.driver.boot.validate(task)
|
|
self.assertFalse(mock_glance.called)
|
|
|
|
def test_validate_fail_missing_deploy_kernel(self):
|
|
with task_manager.acquire(self.context, self.node.uuid,
|
|
shared=True) as task:
|
|
del task.node.driver_info['deploy_kernel']
|
|
self.assertRaises(exception.MissingParameterValue,
|
|
task.driver.boot.validate, task)
|
|
|
|
def test_validate_fail_missing_deploy_ramdisk(self):
|
|
with task_manager.acquire(self.context, self.node.uuid,
|
|
shared=True) as task:
|
|
del task.node.driver_info['deploy_ramdisk']
|
|
self.assertRaises(exception.MissingParameterValue,
|
|
task.driver.boot.validate, task)
|
|
|
|
def test_validate_fail_missing_image_source(self):
|
|
info = dict(INST_INFO_DICT)
|
|
del info['image_source']
|
|
self.node.instance_info = json.dumps(info)
|
|
with task_manager.acquire(self.context, self.node.uuid,
|
|
shared=True) as task:
|
|
task.node['instance_info'] = json.dumps(info)
|
|
self.assertRaises(exception.MissingParameterValue,
|
|
task.driver.boot.validate, task)
|
|
|
|
def test_validate_fail_no_port(self):
|
|
new_node = obj_utils.create_test_node(
|
|
self.context,
|
|
uuid='aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee',
|
|
driver=self.driver, boot_interface=self.boot_interface,
|
|
instance_info=INST_INFO_DICT, driver_info=DRV_INFO_DICT)
|
|
with task_manager.acquire(self.context, new_node.uuid,
|
|
shared=True) as task:
|
|
self.assertRaises(exception.MissingParameterValue,
|
|
task.driver.boot.validate, task)
|
|
|
|
def test_validate_fail_trusted_boot_with_secure_boot(self):
|
|
instance_info = {"boot_option": "netboot",
|
|
"secure_boot": "true",
|
|
"trusted_boot": "true"}
|
|
properties = {'capabilities': 'trusted_boot:true'}
|
|
with task_manager.acquire(self.context, self.node.uuid,
|
|
shared=True) as task:
|
|
task.node.instance_info['capabilities'] = instance_info
|
|
task.node.properties = properties
|
|
task.node.driver_internal_info['is_whole_disk_image'] = False
|
|
self.assertRaises(exception.InvalidParameterValue,
|
|
task.driver.boot.validate, task)
|
|
|
|
def test_validate_fail_invalid_trusted_boot_value(self):
|
|
properties = {'capabilities': 'trusted_boot:value'}
|
|
instance_info = {"trusted_boot": "value"}
|
|
with task_manager.acquire(self.context, self.node.uuid,
|
|
shared=True) as task:
|
|
task.node.properties = properties
|
|
task.node.instance_info['capabilities'] = instance_info
|
|
self.assertRaises(exception.InvalidParameterValue,
|
|
task.driver.boot.validate, task)
|
|
|
|
@mock.patch.object(image_service.GlanceImageService, 'show', autospec=True)
|
|
def test_validate_fail_no_image_kernel_ramdisk_props(self, mock_glance):
|
|
instance_info = {"boot_option": "netboot"}
|
|
mock_glance.return_value = {'properties': {}}
|
|
with task_manager.acquire(self.context, self.node.uuid,
|
|
shared=True) as task:
|
|
task.node.instance_info['capabilities'] = instance_info
|
|
self.assertRaises(exception.MissingParameterValue,
|
|
task.driver.boot.validate,
|
|
task)
|
|
|
|
@mock.patch.object(image_service.GlanceImageService, 'show', autospec=True)
|
|
def test_validate_fail_glance_image_doesnt_exists(self, mock_glance):
|
|
mock_glance.side_effect = exception.ImageNotFound('not found')
|
|
with task_manager.acquire(self.context, self.node.uuid,
|
|
shared=True) as task:
|
|
self.assertRaises(exception.InvalidParameterValue,
|
|
task.driver.boot.validate, task)
|
|
|
|
@mock.patch.object(image_service.GlanceImageService, 'show', autospec=True)
|
|
def test_validate_fail_glance_conn_problem(self, mock_glance):
|
|
exceptions = (exception.GlanceConnectionFailed('connection fail'),
|
|
exception.ImageNotAuthorized('not authorized'),
|
|
exception.Invalid('invalid'))
|
|
mock_glance.side_effect = exceptions
|
|
for exc in exceptions:
|
|
with task_manager.acquire(self.context, self.node.uuid,
|
|
shared=True) as task:
|
|
self.assertRaises(exception.InvalidParameterValue,
|
|
task.driver.boot.validate, task)
|
|
|
|
def test_validate_inspection(self):
|
|
with task_manager.acquire(self.context, self.node.uuid) as task:
|
|
task.driver.boot.validate_inspection(task)
|
|
|
|
def test_validate_inspection_no_inspection_ramdisk(self):
|
|
driver_info = self.node.driver_info
|
|
del driver_info['deploy_ramdisk']
|
|
self.node.driver_info = driver_info
|
|
self.node.save()
|
|
with task_manager.acquire(self.context, self.node.uuid) as task:
|
|
self.assertRaises(exception.UnsupportedDriverExtension,
|
|
task.driver.boot.validate_inspection, task)
|
|
|
|
@mock.patch.object(manager_utils, 'node_get_boot_mode', autospec=True)
|
|
@mock.patch.object(manager_utils, 'node_set_boot_device', autospec=True)
|
|
@mock.patch.object(dhcp_factory, 'DHCPFactory', autospec=True)
|
|
@mock.patch.object(pxe_utils, 'get_instance_image_info', autospec=True)
|
|
@mock.patch.object(pxe_utils, 'get_image_info', autospec=True)
|
|
@mock.patch.object(pxe_utils, 'cache_ramdisk_kernel', autospec=True)
|
|
@mock.patch.object(pxe_utils, 'build_pxe_config_options', autospec=True)
|
|
@mock.patch.object(pxe_utils, 'create_pxe_config', autospec=True)
|
|
def _test_prepare_ramdisk(self, mock_pxe_config,
|
|
mock_build_pxe, mock_cache_r_k,
|
|
mock_deploy_img_info,
|
|
mock_instance_img_info,
|
|
dhcp_factory_mock,
|
|
set_boot_device_mock,
|
|
get_boot_mode_mock,
|
|
uefi=False,
|
|
cleaning=False,
|
|
ipxe_use_swift=False,
|
|
whole_disk_image=False,
|
|
mode='deploy',
|
|
node_boot_mode=None,
|
|
persistent=False):
|
|
mock_build_pxe.return_value = {}
|
|
kernel_label = '%s_kernel' % mode
|
|
ramdisk_label = '%s_ramdisk' % mode
|
|
mock_deploy_img_info.return_value = {kernel_label: 'a',
|
|
ramdisk_label: 'r'}
|
|
if whole_disk_image:
|
|
mock_instance_img_info.return_value = {}
|
|
else:
|
|
mock_instance_img_info.return_value = {'kernel': 'b'}
|
|
mock_pxe_config.return_value = None
|
|
mock_cache_r_k.return_value = None
|
|
provider_mock = mock.MagicMock()
|
|
dhcp_factory_mock.return_value = provider_mock
|
|
get_boot_mode_mock.return_value = node_boot_mode
|
|
driver_internal_info = self.node.driver_internal_info
|
|
driver_internal_info['is_whole_disk_image'] = whole_disk_image
|
|
self.node.driver_internal_info = driver_internal_info
|
|
if mode == 'rescue':
|
|
mock_deploy_img_info.return_value = {
|
|
'rescue_kernel': 'a',
|
|
'rescue_ramdisk': 'r'}
|
|
self.node.save()
|
|
with task_manager.acquire(self.context, self.node.uuid) as task:
|
|
dhcp_opts = pxe_utils.dhcp_options_for_instance(
|
|
task, ipxe_enabled=False)
|
|
dhcp_opts += pxe_utils.dhcp_options_for_instance(
|
|
task, ipxe_enabled=False, ip_version=6)
|
|
task.driver.boot.prepare_ramdisk(task, {'foo': 'bar'})
|
|
mock_deploy_img_info.assert_called_once_with(task.node,
|
|
mode=mode,
|
|
ipxe_enabled=False)
|
|
provider_mock.update_dhcp.assert_called_once_with(task, dhcp_opts)
|
|
if self.node.provision_state == states.DEPLOYING:
|
|
get_boot_mode_mock.assert_called_once_with(task)
|
|
set_boot_device_mock.assert_called_once_with(task,
|
|
boot_devices.PXE,
|
|
persistent=persistent)
|
|
if ipxe_use_swift:
|
|
if whole_disk_image:
|
|
self.assertFalse(mock_cache_r_k.called)
|
|
else:
|
|
mock_cache_r_k.assert_called_once_with(
|
|
task,
|
|
{'kernel': 'b'},
|
|
ipxe_enabled=False)
|
|
mock_instance_img_info.assert_called_once_with(
|
|
task, ipxe_enabled=False)
|
|
elif not cleaning and mode == 'deploy':
|
|
mock_cache_r_k.assert_called_once_with(
|
|
task,
|
|
{'deploy_kernel': 'a', 'deploy_ramdisk': 'r',
|
|
'kernel': 'b'},
|
|
ipxe_enabled=False)
|
|
mock_instance_img_info.assert_called_once_with(
|
|
task, ipxe_enabled=False)
|
|
elif mode == 'deploy':
|
|
mock_cache_r_k.assert_called_once_with(
|
|
task,
|
|
{'deploy_kernel': 'a', 'deploy_ramdisk': 'r'},
|
|
ipxe_enabled=False)
|
|
elif mode == 'rescue':
|
|
mock_cache_r_k.assert_called_once_with(
|
|
task,
|
|
{'rescue_kernel': 'a', 'rescue_ramdisk': 'r'},
|
|
ipxe_enabled=False)
|
|
if uefi:
|
|
mock_pxe_config.assert_called_once_with(
|
|
task, {}, CONF.pxe.uefi_pxe_config_template,
|
|
ipxe_enabled=False)
|
|
else:
|
|
mock_pxe_config.assert_called_once_with(
|
|
task, {}, CONF.pxe.pxe_config_template,
|
|
ipxe_enabled=False)
|
|
|
|
def test_prepare_ramdisk(self):
|
|
self.node.provision_state = states.DEPLOYING
|
|
self.node.save()
|
|
self._test_prepare_ramdisk()
|
|
|
|
def test_prepare_ramdisk_force_persistent_boot_device_true(self):
|
|
self.node.provision_state = states.DEPLOYING
|
|
driver_info = self.node.driver_info
|
|
driver_info['force_persistent_boot_device'] = 'True'
|
|
self.node.driver_info = driver_info
|
|
self.node.save()
|
|
self._test_prepare_ramdisk(persistent=True)
|
|
|
|
def test_prepare_ramdisk_force_persistent_boot_device_bool_true(self):
|
|
self.node.provision_state = states.DEPLOYING
|
|
driver_info = self.node.driver_info
|
|
driver_info['force_persistent_boot_device'] = True
|
|
self.node.driver_info = driver_info
|
|
self.node.save()
|
|
self._test_prepare_ramdisk(persistent=True)
|
|
|
|
def test_prepare_ramdisk_force_persistent_boot_device_sloppy_true(self):
|
|
for value in ['true', 't', '1', 'on', 'y', 'YES']:
|
|
self.node.provision_state = states.DEPLOYING
|
|
driver_info = self.node.driver_info
|
|
driver_info['force_persistent_boot_device'] = value
|
|
self.node.driver_info = driver_info
|
|
self.node.save()
|
|
self._test_prepare_ramdisk(persistent=True)
|
|
|
|
def test_prepare_ramdisk_force_persistent_boot_device_false(self):
|
|
self.node.provision_state = states.DEPLOYING
|
|
driver_info = self.node.driver_info
|
|
driver_info['force_persistent_boot_device'] = 'False'
|
|
self.node.driver_info = driver_info
|
|
self.node.save()
|
|
self._test_prepare_ramdisk()
|
|
|
|
def test_prepare_ramdisk_force_persistent_boot_device_bool_false(self):
|
|
self.node.provision_state = states.DEPLOYING
|
|
driver_info = self.node.driver_info
|
|
driver_info['force_persistent_boot_device'] = False
|
|
self.node.driver_info = driver_info
|
|
self.node.save()
|
|
self._test_prepare_ramdisk(persistent=False)
|
|
|
|
def test_prepare_ramdisk_force_persistent_boot_device_sloppy_false(self):
|
|
for value in ['false', 'f', '0', 'off', 'n', 'NO', 'yxz']:
|
|
self.node.provision_state = states.DEPLOYING
|
|
driver_info = self.node.driver_info
|
|
driver_info['force_persistent_boot_device'] = value
|
|
self.node.driver_info = driver_info
|
|
self.node.save()
|
|
self._test_prepare_ramdisk()
|
|
|
|
def test_prepare_ramdisk_force_persistent_boot_device_default(self):
|
|
self.node.provision_state = states.DEPLOYING
|
|
driver_info = self.node.driver_info
|
|
driver_info['force_persistent_boot_device'] = 'Default'
|
|
self.node.driver_info = driver_info
|
|
self.node.save()
|
|
self._test_prepare_ramdisk(persistent=False)
|
|
|
|
def test_prepare_ramdisk_force_persistent_boot_device_always(self):
|
|
self.node.provision_state = states.DEPLOYING
|
|
driver_info = self.node.driver_info
|
|
driver_info['force_persistent_boot_device'] = 'Always'
|
|
self.node.driver_info = driver_info
|
|
self.node.save()
|
|
self._test_prepare_ramdisk(persistent=True)
|
|
|
|
def test_prepare_ramdisk_force_persistent_boot_device_never(self):
|
|
self.node.provision_state = states.DEPLOYING
|
|
driver_info = self.node.driver_info
|
|
driver_info['force_persistent_boot_device'] = 'Never'
|
|
self.node.driver_info = driver_info
|
|
self.node.save()
|
|
self._test_prepare_ramdisk(persistent=False)
|
|
|
|
def test_prepare_ramdisk_rescue(self):
|
|
self.node.provision_state = states.RESCUING
|
|
self.node.save()
|
|
self._test_prepare_ramdisk(mode='rescue')
|
|
|
|
def test_prepare_ramdisk_uefi(self):
|
|
self.node.provision_state = states.DEPLOYING
|
|
self.node.save()
|
|
properties = self.node.properties
|
|
properties['capabilities'] = 'boot_mode:uefi'
|
|
self.node.properties = properties
|
|
self.node.save()
|
|
self._test_prepare_ramdisk(uefi=True)
|
|
|
|
def test_prepare_ramdisk_cleaning(self):
|
|
self.node.provision_state = states.CLEANING
|
|
self.node.save()
|
|
self._test_prepare_ramdisk(cleaning=True)
|
|
|
|
@mock.patch.object(manager_utils, 'node_set_boot_mode', autospec=True)
|
|
def test_prepare_ramdisk_set_boot_mode_on_bm(
|
|
self, set_boot_mode_mock):
|
|
self.node.provision_state = states.DEPLOYING
|
|
properties = self.node.properties
|
|
properties['capabilities'] = 'boot_mode:uefi'
|
|
self.node.properties = properties
|
|
self.node.save()
|
|
self._test_prepare_ramdisk(uefi=True)
|
|
set_boot_mode_mock.assert_called_once_with(mock.ANY, boot_modes.UEFI)
|
|
|
|
@mock.patch.object(manager_utils, 'node_set_boot_mode', autospec=True)
|
|
def test_prepare_ramdisk_set_boot_mode_on_ironic(
|
|
self, set_boot_mode_mock):
|
|
self.node.provision_state = states.DEPLOYING
|
|
self.node.save()
|
|
self._test_prepare_ramdisk(node_boot_mode=boot_modes.LEGACY_BIOS)
|
|
|
|
with task_manager.acquire(self.context, self.node.uuid) as task:
|
|
driver_internal_info = task.node.driver_internal_info
|
|
self.assertIn('deploy_boot_mode', driver_internal_info)
|
|
self.assertEqual(boot_modes.LEGACY_BIOS,
|
|
driver_internal_info['deploy_boot_mode'])
|
|
self.assertEqual(set_boot_mode_mock.call_count, 0)
|
|
|
|
@mock.patch.object(manager_utils, 'node_set_boot_mode', autospec=True)
|
|
def test_prepare_ramdisk_set_default_boot_mode_on_ironic_bios(
|
|
self, set_boot_mode_mock):
|
|
self.node.provision_state = states.DEPLOYING
|
|
self.node.save()
|
|
|
|
self.config(default_boot_mode=boot_modes.LEGACY_BIOS, group='deploy')
|
|
|
|
self._test_prepare_ramdisk()
|
|
|
|
with task_manager.acquire(self.context, self.node.uuid) as task:
|
|
driver_internal_info = task.node.driver_internal_info
|
|
self.assertIn('deploy_boot_mode', driver_internal_info)
|
|
self.assertEqual(boot_modes.LEGACY_BIOS,
|
|
driver_internal_info['deploy_boot_mode'])
|
|
self.assertEqual(set_boot_mode_mock.call_count, 1)
|
|
|
|
@mock.patch.object(manager_utils, 'node_set_boot_mode', autospec=True)
|
|
def test_prepare_ramdisk_set_default_boot_mode_on_ironic_uefi(
|
|
self, set_boot_mode_mock):
|
|
self.node.provision_state = states.DEPLOYING
|
|
self.node.save()
|
|
|
|
self.config(default_boot_mode=boot_modes.UEFI, group='deploy')
|
|
|
|
self._test_prepare_ramdisk(uefi=True)
|
|
|
|
with task_manager.acquire(self.context, self.node.uuid) as task:
|
|
driver_internal_info = task.node.driver_internal_info
|
|
self.assertIn('deploy_boot_mode', driver_internal_info)
|
|
self.assertEqual(boot_modes.UEFI,
|
|
driver_internal_info['deploy_boot_mode'])
|
|
self.assertEqual(set_boot_mode_mock.call_count, 1)
|
|
|
|
@mock.patch.object(manager_utils, 'node_set_boot_mode', autospec=True)
|
|
def test_prepare_ramdisk_conflicting_boot_modes(
|
|
self, set_boot_mode_mock):
|
|
self.node.provision_state = states.DEPLOYING
|
|
properties = self.node.properties
|
|
properties['capabilities'] = 'boot_mode:uefi'
|
|
self.node.properties = properties
|
|
self.node.save()
|
|
self._test_prepare_ramdisk(uefi=True,
|
|
node_boot_mode=boot_modes.LEGACY_BIOS)
|
|
set_boot_mode_mock.assert_called_once_with(mock.ANY, boot_modes.UEFI)
|
|
|
|
@mock.patch.object(manager_utils, 'node_set_boot_mode', autospec=True)
|
|
def test_prepare_ramdisk_conflicting_boot_modes_set_unsupported(
|
|
self, set_boot_mode_mock):
|
|
self.node.provision_state = states.DEPLOYING
|
|
properties = self.node.properties
|
|
properties['capabilities'] = 'boot_mode:uefi'
|
|
self.node.properties = properties
|
|
self.node.save()
|
|
set_boot_mode_mock.side_effect = exception.UnsupportedDriverExtension(
|
|
extension='management', driver='test-driver'
|
|
)
|
|
self.assertRaises(exception.UnsupportedDriverExtension,
|
|
self._test_prepare_ramdisk,
|
|
uefi=True, node_boot_mode=boot_modes.LEGACY_BIOS)
|
|
|
|
@mock.patch.object(manager_utils, 'node_set_boot_mode', autospec=True)
|
|
def test_prepare_ramdisk_set_boot_mode_not_called(
|
|
self, set_boot_mode_mock):
|
|
self.node.provision_state = states.DEPLOYING
|
|
self.node.save()
|
|
properties = self.node.properties
|
|
properties['capabilities'] = 'boot_mode:uefi'
|
|
self.node.properties = properties
|
|
self.node.save()
|
|
self._test_prepare_ramdisk(uefi=True, node_boot_mode=boot_modes.UEFI)
|
|
self.assertEqual(set_boot_mode_mock.call_count, 0)
|
|
|
|
@mock.patch.object(pxe_utils, 'clean_up_pxe_env', autospec=True)
|
|
@mock.patch.object(pxe_utils, 'get_image_info', autospec=True)
|
|
def _test_clean_up_ramdisk(self, get_image_info_mock,
|
|
clean_up_pxe_env_mock, mode='deploy'):
|
|
with task_manager.acquire(self.context, self.node.uuid) as task:
|
|
kernel_label = '%s_kernel' % mode
|
|
ramdisk_label = '%s_ramdisk' % mode
|
|
image_info = {kernel_label: ['', '/path/to/' + kernel_label],
|
|
ramdisk_label: ['', '/path/to/' + ramdisk_label]}
|
|
get_image_info_mock.return_value = image_info
|
|
task.driver.boot.clean_up_ramdisk(task)
|
|
clean_up_pxe_env_mock.assert_called_once_with(
|
|
task, image_info, ipxe_enabled=False)
|
|
get_image_info_mock.assert_called_once_with(
|
|
task.node, mode=mode, ipxe_enabled=False)
|
|
|
|
def test_clean_up_ramdisk(self):
|
|
self.node.provision_state = states.DEPLOYING
|
|
self.node.save()
|
|
self._test_clean_up_ramdisk()
|
|
|
|
def test_clean_up_ramdisk_rescue(self):
|
|
self.node.provision_state = states.RESCUING
|
|
self.node.save()
|
|
self._test_clean_up_ramdisk(mode='rescue')
|
|
|
|
@mock.patch.object(manager_utils, 'node_set_boot_device', autospec=True)
|
|
@mock.patch.object(deploy_utils, 'switch_pxe_config', autospec=True)
|
|
@mock.patch.object(dhcp_factory, 'DHCPFactory', autospec=True)
|
|
@mock.patch.object(pxe_utils, 'cache_ramdisk_kernel', autospec=True)
|
|
@mock.patch.object(pxe_utils, 'get_instance_image_info', autospec=True)
|
|
def test_prepare_instance_netboot(
|
|
self, get_image_info_mock, cache_mock,
|
|
dhcp_factory_mock, switch_pxe_config_mock,
|
|
set_boot_device_mock):
|
|
provider_mock = mock.MagicMock()
|
|
dhcp_factory_mock.return_value = provider_mock
|
|
image_info = {'kernel': ('', '/path/to/kernel'),
|
|
'ramdisk': ('', '/path/to/ramdisk')}
|
|
get_image_info_mock.return_value = image_info
|
|
with task_manager.acquire(self.context, self.node.uuid) as task:
|
|
dhcp_opts = pxe_utils.dhcp_options_for_instance(
|
|
task, ipxe_enabled=False, ip_version=4)
|
|
dhcp_opts += pxe_utils.dhcp_options_for_instance(
|
|
task, ipxe_enabled=False, ip_version=6)
|
|
pxe_config_path = pxe_utils.get_pxe_config_file_path(
|
|
task.node.uuid)
|
|
task.node.properties['capabilities'] = 'boot_mode:bios'
|
|
task.node.driver_internal_info['root_uuid_or_disk_id'] = (
|
|
"30212642-09d3-467f-8e09-21685826ab50")
|
|
task.node.driver_internal_info['is_whole_disk_image'] = False
|
|
task.node.instance_info = {
|
|
'capabilities': {'boot_option': 'netboot'}}
|
|
task.driver.boot.prepare_instance(task)
|
|
|
|
get_image_info_mock.assert_called_once_with(
|
|
task, ipxe_enabled=False)
|
|
cache_mock.assert_called_once_with(
|
|
task, image_info, ipxe_enabled=False)
|
|
provider_mock.update_dhcp.assert_called_once_with(task, dhcp_opts)
|
|
switch_pxe_config_mock.assert_called_once_with(
|
|
pxe_config_path, "30212642-09d3-467f-8e09-21685826ab50",
|
|
'bios', False, False, False, False, ipxe_enabled=False)
|
|
set_boot_device_mock.assert_called_once_with(task,
|
|
boot_devices.PXE,
|
|
persistent=True)
|
|
|
|
@mock.patch('os.path.isfile', return_value=False, autospec=True)
|
|
@mock.patch.object(pxe_utils, 'create_pxe_config', autospec=True)
|
|
@mock.patch.object(manager_utils, 'node_set_boot_device', autospec=True)
|
|
@mock.patch.object(deploy_utils, 'switch_pxe_config', autospec=True)
|
|
@mock.patch.object(dhcp_factory, 'DHCPFactory', autospec=True)
|
|
@mock.patch.object(pxe_utils, 'cache_ramdisk_kernel', autospec=True)
|
|
@mock.patch.object(pxe_utils, 'get_instance_image_info', autospec=True)
|
|
def test_prepare_instance_netboot_active(
|
|
self, get_image_info_mock, cache_mock,
|
|
dhcp_factory_mock, switch_pxe_config_mock,
|
|
set_boot_device_mock, create_pxe_config_mock, isfile_mock):
|
|
provider_mock = mock.MagicMock()
|
|
dhcp_factory_mock.return_value = provider_mock
|
|
image_info = {'kernel': ('', '/path/to/kernel'),
|
|
'ramdisk': ('', '/path/to/ramdisk')}
|
|
instance_info = {"boot_option": "netboot"}
|
|
get_image_info_mock.return_value = image_info
|
|
self.node.provision_state = states.ACTIVE
|
|
self.node.save()
|
|
with task_manager.acquire(self.context, self.node.uuid) as task:
|
|
dhcp_opts = pxe_utils.dhcp_options_for_instance(
|
|
task, ipxe_enabled=False)
|
|
dhcp_opts += pxe_utils.dhcp_options_for_instance(
|
|
task, ipxe_enabled=False, ip_version=6)
|
|
pxe_config_path = pxe_utils.get_pxe_config_file_path(
|
|
task.node.uuid)
|
|
task.node.properties['capabilities'] = 'boot_mode:bios'
|
|
task.node.driver_internal_info['root_uuid_or_disk_id'] = (
|
|
"30212642-09d3-467f-8e09-21685826ab50")
|
|
task.node.driver_internal_info['is_whole_disk_image'] = False
|
|
task.node.instance_info['capabilities'] = instance_info
|
|
task.driver.boot.prepare_instance(task)
|
|
|
|
get_image_info_mock.assert_called_once_with(
|
|
task, ipxe_enabled=False)
|
|
cache_mock.assert_called_once_with(
|
|
task, image_info, ipxe_enabled=False)
|
|
provider_mock.update_dhcp.assert_called_once_with(task, dhcp_opts)
|
|
create_pxe_config_mock.assert_called_once_with(
|
|
task, mock.ANY, CONF.pxe.pxe_config_template,
|
|
ipxe_enabled=False)
|
|
switch_pxe_config_mock.assert_called_once_with(
|
|
pxe_config_path, "30212642-09d3-467f-8e09-21685826ab50",
|
|
'bios', False, False, False, False, ipxe_enabled=False)
|
|
self.assertFalse(set_boot_device_mock.called)
|
|
|
|
@mock.patch.object(manager_utils, 'node_set_boot_device', autospec=True)
|
|
@mock.patch.object(deploy_utils, 'switch_pxe_config', autospec=True)
|
|
@mock.patch.object(dhcp_factory, 'DHCPFactory', autospec=True)
|
|
@mock.patch.object(pxe_utils, 'cache_ramdisk_kernel', autospec=True)
|
|
@mock.patch.object(pxe_utils, 'get_instance_image_info', autospec=True)
|
|
def test_prepare_instance_netboot_missing_root_uuid(
|
|
self, get_image_info_mock, cache_mock,
|
|
dhcp_factory_mock, switch_pxe_config_mock,
|
|
set_boot_device_mock):
|
|
provider_mock = mock.MagicMock()
|
|
dhcp_factory_mock.return_value = provider_mock
|
|
image_info = {'kernel': ('', '/path/to/kernel'),
|
|
'ramdisk': ('', '/path/to/ramdisk')}
|
|
instance_info = {"boot_option": "netboot"}
|
|
get_image_info_mock.return_value = image_info
|
|
with task_manager.acquire(self.context, self.node.uuid) as task:
|
|
dhcp_opts = pxe_utils.dhcp_options_for_instance(
|
|
task, ipxe_enabled=False)
|
|
dhcp_opts += pxe_utils.dhcp_options_for_instance(
|
|
task, ipxe_enabled=False, ip_version=6)
|
|
task.node.properties['capabilities'] = 'boot_mode:bios'
|
|
task.node.instance_info['capabilities'] = instance_info
|
|
task.node.driver_internal_info['is_whole_disk_image'] = False
|
|
|
|
task.driver.boot.prepare_instance(task)
|
|
|
|
get_image_info_mock.assert_called_once_with(task,
|
|
ipxe_enabled=False)
|
|
cache_mock.assert_called_once_with(
|
|
task, image_info, ipxe_enabled=False)
|
|
provider_mock.update_dhcp.assert_called_once_with(task, dhcp_opts)
|
|
self.assertFalse(switch_pxe_config_mock.called)
|
|
self.assertFalse(set_boot_device_mock.called)
|
|
|
|
@mock.patch.object(pxe_base.LOG, 'warning', autospec=True)
|
|
@mock.patch.object(pxe_utils, 'clean_up_pxe_config', autospec=True)
|
|
@mock.patch.object(manager_utils, 'node_set_boot_device', autospec=True)
|
|
@mock.patch.object(dhcp_factory, 'DHCPFactory', autospec=True)
|
|
@mock.patch.object(pxe_utils, 'cache_ramdisk_kernel', autospec=True)
|
|
@mock.patch.object(pxe_utils, 'get_instance_image_info', autospec=True)
|
|
def test_prepare_instance_whole_disk_image_missing_root_uuid(
|
|
self, get_image_info_mock, cache_mock,
|
|
dhcp_factory_mock, set_boot_device_mock,
|
|
clean_up_pxe_mock, log_mock):
|
|
provider_mock = mock.MagicMock()
|
|
dhcp_factory_mock.return_value = provider_mock
|
|
get_image_info_mock.return_value = {}
|
|
instance_info = {"boot_option": "netboot"}
|
|
with task_manager.acquire(self.context, self.node.uuid) as task:
|
|
dhcp_opts = pxe_utils.dhcp_options_for_instance(
|
|
task, ipxe_enabled=False)
|
|
dhcp_opts += pxe_utils.dhcp_options_for_instance(
|
|
task, ipxe_enabled=False, ip_version=6)
|
|
task.node.properties['capabilities'] = 'boot_mode:bios'
|
|
task.node.instance_info['capabilities'] = instance_info
|
|
task.node.driver_internal_info['is_whole_disk_image'] = True
|
|
task.driver.boot.prepare_instance(task)
|
|
get_image_info_mock.assert_called_once_with(task,
|
|
ipxe_enabled=False)
|
|
cache_mock.assert_called_once_with(
|
|
task, {}, ipxe_enabled=False)
|
|
provider_mock.update_dhcp.assert_called_once_with(task, dhcp_opts)
|
|
self.assertTrue(log_mock.called)
|
|
clean_up_pxe_mock.assert_called_once_with(
|
|
task, ipxe_enabled=False)
|
|
set_boot_device_mock.assert_called_once_with(
|
|
task, boot_devices.DISK, persistent=True)
|
|
|
|
@mock.patch.object(manager_utils, 'node_set_boot_device', autospec=True)
|
|
@mock.patch.object(pxe_utils, 'clean_up_pxe_config', autospec=True)
|
|
def test_prepare_instance_localboot(self, clean_up_pxe_config_mock,
|
|
set_boot_device_mock):
|
|
with task_manager.acquire(self.context, self.node.uuid) as task:
|
|
instance_info = task.node.instance_info
|
|
instance_info['capabilities'] = {'boot_option': 'local'}
|
|
task.node.instance_info = instance_info
|
|
task.node.save()
|
|
task.driver.boot.prepare_instance(task)
|
|
clean_up_pxe_config_mock.assert_called_once_with(
|
|
task, ipxe_enabled=False)
|
|
set_boot_device_mock.assert_called_once_with(task,
|
|
boot_devices.DISK,
|
|
persistent=True)
|
|
|
|
@mock.patch.object(manager_utils, 'node_set_boot_device', autospec=True)
|
|
@mock.patch.object(pxe_utils, 'clean_up_pxe_config', autospec=True)
|
|
def test_prepare_instance_localboot_active(self, clean_up_pxe_config_mock,
|
|
set_boot_device_mock):
|
|
self.node.provision_state = states.ACTIVE
|
|
self.node.save()
|
|
with task_manager.acquire(self.context, self.node.uuid) as task:
|
|
instance_info = task.node.instance_info
|
|
instance_info['capabilities'] = {'boot_option': 'local'}
|
|
task.node.instance_info = instance_info
|
|
task.node.save()
|
|
task.driver.boot.prepare_instance(task)
|
|
clean_up_pxe_config_mock.assert_called_once_with(
|
|
task, ipxe_enabled=False)
|
|
self.assertFalse(set_boot_device_mock.called)
|
|
|
|
@mock.patch.object(manager_utils, 'node_set_boot_device', autospec=True)
|
|
@mock.patch.object(deploy_utils, 'switch_pxe_config', autospec=True)
|
|
@mock.patch.object(pxe_utils, 'create_pxe_config', autospec=True)
|
|
@mock.patch.object(dhcp_factory, 'DHCPFactory', autospec=True)
|
|
@mock.patch.object(pxe_utils, 'cache_ramdisk_kernel', autospec=True)
|
|
@mock.patch.object(pxe_utils, 'get_instance_image_info', autospec=True)
|
|
def _test_prepare_instance_ramdisk(
|
|
self, get_image_info_mock, cache_mock,
|
|
dhcp_factory_mock, create_pxe_config_mock,
|
|
switch_pxe_config_mock,
|
|
set_boot_device_mock, config_file_exits=False):
|
|
image_info = {'kernel': ['', '/path/to/kernel'],
|
|
'ramdisk': ['', '/path/to/ramdisk']}
|
|
get_image_info_mock.return_value = image_info
|
|
provider_mock = mock.MagicMock()
|
|
dhcp_factory_mock.return_value = provider_mock
|
|
self.node.provision_state = states.DEPLOYING
|
|
get_image_info_mock.return_value = image_info
|
|
with task_manager.acquire(self.context, self.node.uuid) as task:
|
|
instance_info = task.node.instance_info
|
|
instance_info['capabilities'] = {'boot_option': 'ramdisk'}
|
|
task.node.instance_info = instance_info
|
|
task.node.save()
|
|
dhcp_opts = pxe_utils.dhcp_options_for_instance(
|
|
task, ipxe_enabled=False)
|
|
dhcp_opts += pxe_utils.dhcp_options_for_instance(
|
|
task, ipxe_enabled=False, ip_version=6)
|
|
pxe_config_path = pxe_utils.get_pxe_config_file_path(
|
|
task.node.uuid)
|
|
task.driver.boot.prepare_instance(task)
|
|
|
|
get_image_info_mock.assert_called_once_with(task,
|
|
ipxe_enabled=False)
|
|
cache_mock.assert_called_once_with(
|
|
task, image_info, False)
|
|
provider_mock.update_dhcp.assert_called_once_with(task, dhcp_opts)
|
|
if config_file_exits:
|
|
self.assertFalse(create_pxe_config_mock.called)
|
|
else:
|
|
create_pxe_config_mock.assert_called_once_with(
|
|
task, mock.ANY, CONF.pxe.pxe_config_template,
|
|
ipxe_enabled=False)
|
|
switch_pxe_config_mock.assert_called_once_with(
|
|
pxe_config_path, None,
|
|
'bios', False, ipxe_enabled=False, iscsi_boot=False,
|
|
ramdisk_boot=True)
|
|
set_boot_device_mock.assert_called_once_with(task,
|
|
boot_devices.PXE,
|
|
persistent=True)
|
|
|
|
@mock.patch.object(os.path, 'isfile', lambda path: True)
|
|
def test_prepare_instance_ramdisk_pxe_conf_missing(self):
|
|
self._test_prepare_instance_ramdisk(config_file_exits=True)
|
|
|
|
@mock.patch.object(os.path, 'isfile', lambda path: False)
|
|
def test_prepare_instance_ramdisk_pxe_conf_exists(self):
|
|
self._test_prepare_instance_ramdisk(config_file_exits=False)
|
|
|
|
@mock.patch.object(pxe_utils, 'clean_up_pxe_env', autospec=True)
|
|
@mock.patch.object(pxe_utils, 'get_instance_image_info', autospec=True)
|
|
def test_clean_up_instance(self, get_image_info_mock,
|
|
clean_up_pxe_env_mock):
|
|
with task_manager.acquire(self.context, self.node.uuid) as task:
|
|
image_info = {'kernel': ['', '/path/to/kernel'],
|
|
'ramdisk': ['', '/path/to/ramdisk']}
|
|
get_image_info_mock.return_value = image_info
|
|
task.driver.boot.clean_up_instance(task)
|
|
clean_up_pxe_env_mock.assert_called_once_with(task, image_info,
|
|
ipxe_enabled=False)
|
|
get_image_info_mock.assert_called_once_with(task,
|
|
ipxe_enabled=False)
|
|
|
|
|
|
class PXERamdiskDeployTestCase(db_base.DbTestCase):
|
|
|
|
def setUp(self):
|
|
super(PXERamdiskDeployTestCase, self).setUp()
|
|
self.temp_dir = tempfile.mkdtemp()
|
|
self.config(tftp_root=self.temp_dir, group='pxe')
|
|
self.temp_dir = tempfile.mkdtemp()
|
|
self.config(images_path=self.temp_dir, group='pxe')
|
|
self.config(enabled_deploy_interfaces=['ramdisk'])
|
|
self.config(enabled_boot_interfaces=['pxe'])
|
|
for iface in drivers_base.ALL_INTERFACES:
|
|
impl = 'fake'
|
|
if iface == 'network':
|
|
impl = 'noop'
|
|
if iface == 'deploy':
|
|
impl = 'ramdisk'
|
|
if iface == 'boot':
|
|
impl = 'pxe'
|
|
config_kwarg = {'enabled_%s_interfaces' % iface: [impl],
|
|
'default_%s_interface' % iface: impl}
|
|
self.config(**config_kwarg)
|
|
self.config(enabled_hardware_types=['fake-hardware'])
|
|
instance_info = INST_INFO_DICT
|
|
self.node = obj_utils.create_test_node(
|
|
self.context,
|
|
driver='fake-hardware',
|
|
instance_info=instance_info,
|
|
driver_info=DRV_INFO_DICT,
|
|
driver_internal_info=DRV_INTERNAL_INFO_DICT)
|
|
self.port = obj_utils.create_test_port(self.context,
|
|
node_id=self.node.id)
|
|
|
|
@mock.patch.object(manager_utils, 'node_set_boot_device', autospec=True)
|
|
@mock.patch.object(deploy_utils, 'switch_pxe_config', autospec=True)
|
|
@mock.patch.object(dhcp_factory, 'DHCPFactory', autospec=True)
|
|
@mock.patch.object(pxe_utils, 'cache_ramdisk_kernel', autospec=True)
|
|
@mock.patch.object(pxe_utils, 'get_instance_image_info', autospec=True)
|
|
def test_prepare_instance_ramdisk(
|
|
self, get_image_info_mock, cache_mock,
|
|
dhcp_factory_mock, switch_pxe_config_mock,
|
|
set_boot_device_mock):
|
|
provider_mock = mock.MagicMock()
|
|
dhcp_factory_mock.return_value = provider_mock
|
|
self.node.provision_state = states.DEPLOYING
|
|
image_info = {'kernel': ('', '/path/to/kernel'),
|
|
'ramdisk': ('', '/path/to/ramdisk')}
|
|
get_image_info_mock.return_value = image_info
|
|
with task_manager.acquire(self.context, self.node.uuid) as task:
|
|
dhcp_opts = pxe_utils.dhcp_options_for_instance(
|
|
task, ipxe_enabled=False)
|
|
dhcp_opts += pxe_utils.dhcp_options_for_instance(
|
|
task, ipxe_enabled=False, ip_version=6)
|
|
pxe_config_path = pxe_utils.get_pxe_config_file_path(
|
|
task.node.uuid)
|
|
task.node.properties['capabilities'] = 'boot_option:netboot'
|
|
task.node.driver_internal_info['is_whole_disk_image'] = False
|
|
task.driver.deploy.prepare(task)
|
|
task.driver.deploy.deploy(task)
|
|
|
|
get_image_info_mock.assert_called_once_with(task,
|
|
ipxe_enabled=False)
|
|
cache_mock.assert_called_once_with(
|
|
task, image_info, ipxe_enabled=False)
|
|
provider_mock.update_dhcp.assert_called_once_with(task, dhcp_opts)
|
|
switch_pxe_config_mock.assert_called_once_with(
|
|
pxe_config_path, None,
|
|
'bios', False, ipxe_enabled=False, iscsi_boot=False,
|
|
ramdisk_boot=True)
|
|
set_boot_device_mock.assert_called_once_with(task,
|
|
boot_devices.PXE,
|
|
persistent=True)
|
|
|
|
@mock.patch.object(pxe.LOG, 'warning', autospec=True)
|
|
@mock.patch.object(deploy_utils, 'switch_pxe_config', autospec=True)
|
|
@mock.patch.object(dhcp_factory, 'DHCPFactory', autospec=True)
|
|
@mock.patch.object(pxe_utils, 'cache_ramdisk_kernel', autospec=True)
|
|
@mock.patch.object(pxe_utils, 'get_instance_image_info', autospec=True)
|
|
def test_deploy(self, mock_image_info, mock_cache,
|
|
mock_dhcp_factory, mock_switch_config, mock_warning):
|
|
image_info = {'kernel': ('', '/path/to/kernel'),
|
|
'ramdisk': ('', '/path/to/ramdisk')}
|
|
mock_image_info.return_value = image_info
|
|
i_info = self.node.instance_info
|
|
i_info.update({'capabilities': {'boot_option': 'ramdisk'}})
|
|
self.node.instance_info = i_info
|
|
self.node.save()
|
|
with task_manager.acquire(self.context, self.node.uuid) as task:
|
|
self.assertIsNone(task.driver.deploy.deploy(task))
|
|
mock_image_info.assert_called_once_with(task, ipxe_enabled=False)
|
|
mock_cache.assert_called_once_with(
|
|
task, image_info, ipxe_enabled=False)
|
|
self.assertFalse(mock_warning.called)
|
|
i_info['configdrive'] = 'meow'
|
|
self.node.instance_info = i_info
|
|
self.node.save()
|
|
mock_warning.reset_mock()
|
|
with task_manager.acquire(self.context, self.node.uuid) as task:
|
|
self.assertIsNone(task.driver.deploy.deploy(task))
|
|
self.assertTrue(mock_warning.called)
|
|
|
|
@mock.patch.object(pxe.PXEBoot, 'prepare_instance', autospec=True)
|
|
def test_prepare(self, mock_prepare_instance):
|
|
node = self.node
|
|
node.provision_state = states.DEPLOYING
|
|
node.instance_info = {}
|
|
node.save()
|
|
with task_manager.acquire(self.context, node.uuid) as task:
|
|
task.driver.deploy.prepare(task)
|
|
self.assertFalse(mock_prepare_instance.called)
|
|
self.assertEqual({'boot_option': 'ramdisk'},
|
|
task.node.instance_info['capabilities'])
|
|
|
|
@mock.patch.object(pxe.PXEBoot, 'prepare_instance', autospec=True)
|
|
def test_prepare_active(self, mock_prepare_instance):
|
|
node = self.node
|
|
node.provision_state = states.ACTIVE
|
|
node.save()
|
|
with task_manager.acquire(self.context, node.uuid) as task:
|
|
task.driver.deploy.prepare(task)
|
|
mock_prepare_instance.assert_called_once_with(mock.ANY, task)
|
|
|
|
@mock.patch.object(pxe.PXEBoot, 'prepare_instance', autospec=True)
|
|
def test_prepare_unrescuing(self, mock_prepare_instance):
|
|
node = self.node
|
|
node.provision_state = states.UNRESCUING
|
|
node.save()
|
|
with task_manager.acquire(self.context, node.uuid) as task:
|
|
task.driver.deploy.prepare(task)
|
|
mock_prepare_instance.assert_called_once_with(mock.ANY, task)
|
|
|
|
@mock.patch.object(pxe.LOG, 'warning', autospec=True)
|
|
@mock.patch.object(pxe.PXEBoot, 'prepare_instance', autospec=True)
|
|
def test_prepare_fixes_and_logs_boot_option_warning(
|
|
self, mock_prepare_instance, mock_warning):
|
|
node = self.node
|
|
node.properties['capabilities'] = 'boot_option:ramdisk'
|
|
node.provision_state = states.DEPLOYING
|
|
node.instance_info = {}
|
|
node.save()
|
|
with task_manager.acquire(self.context, node.uuid) as task:
|
|
task.driver.deploy.prepare(task)
|
|
self.assertFalse(mock_prepare_instance.called)
|
|
self.assertEqual({'boot_option': 'ramdisk'},
|
|
task.node.instance_info['capabilities'])
|
|
self.assertTrue(mock_warning.called)
|
|
|
|
@mock.patch.object(deploy_utils, 'validate_image_properties',
|
|
autospec=True)
|
|
def test_validate(self, mock_validate_img):
|
|
node = self.node
|
|
node.properties['capabilities'] = 'boot_option:netboot'
|
|
node.save()
|
|
with task_manager.acquire(self.context, node.uuid) as task:
|
|
task.driver.deploy.validate(task)
|
|
self.assertTrue(mock_validate_img.called)
|
|
|
|
@mock.patch.object(fake.FakeBoot, 'validate', autospec=True)
|
|
@mock.patch.object(deploy_utils, 'validate_image_properties',
|
|
autospec=True)
|
|
def test_validate_interface_mismatch(self, mock_validate_image,
|
|
mock_boot_validate):
|
|
node = self.node
|
|
node.boot_interface = 'fake'
|
|
node.save()
|
|
self.config(enabled_boot_interfaces=['fake'],
|
|
default_boot_interface='fake')
|
|
with task_manager.acquire(self.context, node.uuid) as task:
|
|
error = self.assertRaises(exception.InvalidParameterValue,
|
|
task.driver.deploy.validate, task)
|
|
error_message = ('Invalid configuration: The boot interface must '
|
|
'have the `ramdisk_boot` capability. You are '
|
|
'using an incompatible boot interface.')
|
|
self.assertEqual(error_message, str(error))
|
|
self.assertFalse(mock_boot_validate.called)
|
|
self.assertFalse(mock_validate_image.called)
|
|
|
|
@mock.patch.object(pxe.PXEBoot, 'validate', autospec=True)
|
|
def test_validate_calls_boot_validate(self, mock_validate):
|
|
with task_manager.acquire(self.context, self.node.uuid) as task:
|
|
task.driver.deploy.validate(task)
|
|
mock_validate.assert_called_once_with(mock.ANY, task)
|
|
|
|
@mock.patch.object(manager_utils, 'restore_power_state_if_needed',
|
|
autospec=True)
|
|
@mock.patch.object(manager_utils, 'power_on_node_if_needed',
|
|
autospec=True)
|
|
@mock.patch.object(pxe.LOG, 'warning', autospec=True)
|
|
@mock.patch.object(deploy_utils, 'switch_pxe_config', autospec=True)
|
|
@mock.patch.object(dhcp_factory, 'DHCPFactory', autospec=True)
|
|
@mock.patch.object(pxe_utils, 'cache_ramdisk_kernel', autospec=True)
|
|
@mock.patch.object(pxe_utils, 'get_instance_image_info', autospec=True)
|
|
def test_deploy_with_smartnic_port(
|
|
self, mock_image_info, mock_cache,
|
|
mock_dhcp_factory, mock_switch_config, mock_warning,
|
|
power_on_node_if_needed_mock, restore_power_state_mock):
|
|
image_info = {'kernel': ('', '/path/to/kernel'),
|
|
'ramdisk': ('', '/path/to/ramdisk')}
|
|
mock_image_info.return_value = image_info
|
|
i_info = self.node.instance_info
|
|
i_info.update({'capabilities': {'boot_option': 'ramdisk'}})
|
|
self.node.instance_info = i_info
|
|
self.node.save()
|
|
with task_manager.acquire(self.context, self.node.uuid) as task:
|
|
power_on_node_if_needed_mock.return_value = states.POWER_OFF
|
|
self.assertIsNone(task.driver.deploy.deploy(task))
|
|
mock_image_info.assert_called_once_with(task, ipxe_enabled=False)
|
|
mock_cache.assert_called_once_with(
|
|
task, image_info, ipxe_enabled=False)
|
|
self.assertFalse(mock_warning.called)
|
|
power_on_node_if_needed_mock.assert_called_once_with(task)
|
|
restore_power_state_mock.assert_called_once_with(
|
|
task, states.POWER_OFF)
|
|
i_info['configdrive'] = 'meow'
|
|
self.node.instance_info = i_info
|
|
self.node.save()
|
|
mock_warning.reset_mock()
|
|
with task_manager.acquire(self.context, self.node.uuid) as task:
|
|
self.assertIsNone(task.driver.deploy.deploy(task))
|
|
self.assertTrue(mock_warning.called)
|
|
|
|
|
|
class PXEValidateRescueTestCase(db_base.DbTestCase):
|
|
|
|
def setUp(self):
|
|
super(PXEValidateRescueTestCase, self).setUp()
|
|
for iface in drivers_base.ALL_INTERFACES:
|
|
impl = 'fake'
|
|
if iface == 'network':
|
|
impl = 'flat'
|
|
if iface == 'rescue':
|
|
impl = 'agent'
|
|
if iface == 'boot':
|
|
impl = 'pxe'
|
|
config_kwarg = {'enabled_%s_interfaces' % iface: [impl],
|
|
'default_%s_interface' % iface: impl}
|
|
self.config(**config_kwarg)
|
|
self.config(enabled_hardware_types=['fake-hardware'])
|
|
driver_info = DRV_INFO_DICT
|
|
driver_info.update({'rescue_ramdisk': 'my_ramdisk',
|
|
'rescue_kernel': 'my_kernel'})
|
|
instance_info = INST_INFO_DICT
|
|
instance_info.update({'rescue_password': 'password'})
|
|
n = {
|
|
'driver': 'fake-hardware',
|
|
'instance_info': instance_info,
|
|
'driver_info': driver_info,
|
|
'driver_internal_info': DRV_INTERNAL_INFO_DICT,
|
|
}
|
|
self.node = obj_utils.create_test_node(self.context, **n)
|
|
|
|
def test_validate_rescue(self):
|
|
with task_manager.acquire(self.context, self.node.uuid) as task:
|
|
task.driver.boot.validate_rescue(task)
|
|
|
|
def test_validate_rescue_no_rescue_ramdisk(self):
|
|
driver_info = self.node.driver_info
|
|
del driver_info['rescue_ramdisk']
|
|
self.node.driver_info = driver_info
|
|
self.node.save()
|
|
with task_manager.acquire(self.context, self.node.uuid) as task:
|
|
self.assertRaisesRegex(exception.MissingParameterValue,
|
|
'Missing.*rescue_ramdisk',
|
|
task.driver.boot.validate_rescue, task)
|
|
|
|
def test_validate_rescue_fails_no_rescue_kernel(self):
|
|
driver_info = self.node.driver_info
|
|
del driver_info['rescue_kernel']
|
|
self.node.driver_info = driver_info
|
|
self.node.save()
|
|
with task_manager.acquire(self.context, self.node.uuid) as task:
|
|
self.assertRaisesRegex(exception.MissingParameterValue,
|
|
'Missing.*rescue_kernel',
|
|
task.driver.boot.validate_rescue, task)
|
|
|
|
|
|
@mock.patch.object(ipxe.iPXEBoot, '__init__', lambda self: None)
|
|
@mock.patch.object(pxe.PXEBoot, '__init__', lambda self: None)
|
|
@mock.patch.object(manager_utils, 'node_set_boot_device', autospec=True)
|
|
@mock.patch.object(manager_utils, 'node_power_action', autospec=True)
|
|
class PXEBootRetryTestCase(db_base.DbTestCase):
|
|
|
|
boot_interface = 'pxe'
|
|
|
|
def setUp(self):
|
|
super(PXEBootRetryTestCase, self).setUp()
|
|
self.config(enabled_boot_interfaces=['pxe', 'ipxe', 'fake'])
|
|
self.config(boot_retry_timeout=300, group='pxe')
|
|
self.node = obj_utils.create_test_node(
|
|
self.context,
|
|
driver='fake-hardware',
|
|
boot_interface=self.boot_interface,
|
|
provision_state=states.DEPLOYWAIT)
|
|
|
|
@mock.patch.object(pxe.PXEBoot, '_check_boot_status', autospec=True)
|
|
def test_check_boot_timeouts(self, mock_check_status, mock_power,
|
|
mock_boot_dev):
|
|
def _side_effect(iface, task):
|
|
self.assertEqual(self.node.uuid, task.node.uuid)
|
|
|
|
mock_check_status.side_effect = _side_effect
|
|
manager = mock.Mock(spec=['iter_nodes'])
|
|
manager.iter_nodes.return_value = [
|
|
(uuidutils.generate_uuid(), 'fake-hardware', ''),
|
|
(self.node.uuid, self.node.driver, self.node.conductor_group)
|
|
]
|
|
iface = pxe.PXEBoot()
|
|
iface._check_boot_timeouts(manager, self.context)
|
|
mock_check_status.assert_called_once_with(iface, mock.ANY)
|
|
|
|
def test_check_boot_status_another_boot_interface(self, mock_power,
|
|
mock_boot_dev):
|
|
with task_manager.acquire(self.context, self.node.uuid,
|
|
shared=True) as task:
|
|
task.driver.boot = fake.FakeBoot()
|
|
pxe.PXEBoot()._check_boot_status(task)
|
|
self.assertTrue(task.shared)
|
|
self.assertFalse(mock_power.called)
|
|
self.assertFalse(mock_boot_dev.called)
|
|
|
|
def test_check_boot_status_recent_power_change(self, mock_power,
|
|
mock_boot_dev):
|
|
for field in ('agent_last_heartbeat', 'last_power_state_change'):
|
|
with task_manager.acquire(self.context, self.node.uuid,
|
|
shared=True) as task:
|
|
task.node.driver_internal_info = {
|
|
field: str(timeutils.utcnow().isoformat())
|
|
}
|
|
task.driver.boot._check_boot_status(task)
|
|
self.assertTrue(task.shared)
|
|
self.assertFalse(mock_power.called)
|
|
self.assertFalse(mock_boot_dev.called)
|
|
|
|
def test_check_boot_status_maintenance(self, mock_power, mock_boot_dev):
|
|
self.node.maintenance = True
|
|
self.node.save()
|
|
with task_manager.acquire(self.context, self.node.uuid,
|
|
shared=True) as task:
|
|
task.driver.boot._check_boot_status(task)
|
|
self.assertFalse(task.shared)
|
|
self.assertFalse(mock_power.called)
|
|
self.assertFalse(mock_boot_dev.called)
|
|
|
|
def test_check_boot_status_wrong_state(self, mock_power, mock_boot_dev):
|
|
self.node.provision_state = states.DEPLOYING
|
|
self.node.save()
|
|
with task_manager.acquire(self.context, self.node.uuid,
|
|
shared=True) as task:
|
|
task.driver.boot._check_boot_status(task)
|
|
self.assertFalse(task.shared)
|
|
self.assertFalse(mock_power.called)
|
|
self.assertFalse(mock_boot_dev.called)
|
|
|
|
def test_check_boot_status_retry(self, mock_power, mock_boot_dev):
|
|
with task_manager.acquire(self.context, self.node.uuid,
|
|
shared=True) as task:
|
|
task.driver.boot._check_boot_status(task)
|
|
self.assertFalse(task.shared)
|
|
mock_power.assert_has_calls([
|
|
mock.call(task, states.POWER_OFF),
|
|
mock.call(task, states.POWER_ON)
|
|
])
|
|
mock_boot_dev.assert_called_once_with(task, 'pxe',
|
|
persistent=False)
|
|
|
|
|
|
class iPXEBootRetryTestCase(PXEBootRetryTestCase):
|
|
|
|
boot_interface = 'ipxe'
|