
This patch is adding a redfish driver based on the sushy library. This is just a basic driver that currently supports: * Power: Hard power on/off/reboot, soft power off/reboot * Management: Setting the boot device (PXE, disk, cd-rom and bios) and its frequency (persistent or not) * Management: NMI Injection * SSL authentication Unittest coverage for the redfish modules is now in 100%, let's try to keep it this way (-: Documentation and DevStack updates will be done on subsequent patches. Partial-Bug: #1526477 Change-Id: I14470edff65cd14bb73263ec7310559a8eaa6c84
174 lines
7.5 KiB
Python
174 lines
7.5 KiB
Python
# Copyright 2017 Red Hat, Inc.
|
|
# 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.
|
|
|
|
import mock
|
|
from oslo_utils import importutils
|
|
|
|
from ironic.common import exception
|
|
from ironic.common import states
|
|
from ironic.conductor import task_manager
|
|
from ironic.drivers.modules.redfish import power as redfish_power
|
|
from ironic.drivers.modules.redfish import utils as redfish_utils
|
|
from ironic.tests.unit.conductor import mgr_utils
|
|
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
|
|
|
|
sushy = importutils.try_import('sushy')
|
|
|
|
INFO_DICT = db_utils.get_test_redfish_info()
|
|
|
|
|
|
class MockedSushyError(Exception):
|
|
pass
|
|
|
|
|
|
class RedfishPowerTestCase(db_base.DbTestCase):
|
|
|
|
def setUp(self):
|
|
super(RedfishPowerTestCase, self).setUp()
|
|
self.config(enabled_hardware_types=['redfish'],
|
|
enabled_power_interfaces=['redfish'],
|
|
enabled_management_interfaces=['redfish'])
|
|
mgr_utils.mock_the_extension_manager(
|
|
driver='redfish', namespace='ironic.hardware.types')
|
|
self.node = obj_utils.create_test_node(
|
|
self.context, driver='redfish', driver_info=INFO_DICT)
|
|
|
|
@mock.patch.object(redfish_power, 'sushy', None)
|
|
def test_loading_error(self):
|
|
self.assertRaisesRegex(
|
|
exception.DriverLoadError,
|
|
'Unable to import the sushy library',
|
|
redfish_power.RedfishPower)
|
|
|
|
def test_get_properties(self):
|
|
with task_manager.acquire(self.context, self.node.uuid,
|
|
shared=True) as task:
|
|
properties = task.driver.get_properties()
|
|
for prop in redfish_utils.COMMON_PROPERTIES:
|
|
self.assertIn(prop, properties)
|
|
|
|
@mock.patch.object(redfish_utils, 'parse_driver_info', autospec=True)
|
|
def test_validate(self, mock_parse_driver_info):
|
|
with task_manager.acquire(self.context, self.node.uuid,
|
|
shared=True) as task:
|
|
task.driver.power.validate(task)
|
|
mock_parse_driver_info.assert_called_once_with(task.node)
|
|
|
|
@mock.patch.object(redfish_utils, 'get_system', autospec=True)
|
|
def test_get_power_state(self, mock_get_system):
|
|
with task_manager.acquire(self.context, self.node.uuid,
|
|
shared=True) as task:
|
|
expected_values = [
|
|
(sushy.SYSTEM_POWER_STATE_ON, states.POWER_ON),
|
|
(sushy.SYSTEM_POWER_STATE_POWERING_ON, states.POWER_ON),
|
|
(sushy.SYSTEM_POWER_STATE_OFF, states.POWER_OFF),
|
|
(sushy.SYSTEM_POWER_STATE_POWERING_OFF, states.POWER_OFF)
|
|
]
|
|
for current, expected in expected_values:
|
|
mock_get_system.return_value = mock.Mock(power_state=current)
|
|
self.assertEqual(expected,
|
|
task.driver.power.get_power_state(task))
|
|
mock_get_system.assert_called_once_with(task.node)
|
|
mock_get_system.reset_mock()
|
|
|
|
@mock.patch.object(redfish_utils, 'get_system', autospec=True)
|
|
def test_set_power_state(self, mock_get_system):
|
|
with task_manager.acquire(self.context, self.node.uuid,
|
|
shared=False) as task:
|
|
expected_values = [
|
|
(states.POWER_ON, sushy.RESET_ON),
|
|
(states.POWER_OFF, sushy.RESET_FORCE_OFF),
|
|
(states.REBOOT, sushy.RESET_FORCE_RESTART),
|
|
(states.SOFT_REBOOT, sushy.RESET_GRACEFUL_RESTART),
|
|
(states.SOFT_POWER_OFF, sushy.RESET_GRACEFUL_SHUTDOWN)
|
|
]
|
|
|
|
fake_system = mock_get_system.return_value
|
|
for target, expected in expected_values:
|
|
task.driver.power.set_power_state(task, target)
|
|
|
|
# Asserts
|
|
fake_system.reset_system.assert_called_once_with(expected)
|
|
mock_get_system.assert_called_once_with(task.node)
|
|
|
|
# Reset mocks
|
|
fake_system.reset_system.reset_mock()
|
|
mock_get_system.reset_mock()
|
|
|
|
@mock.patch('ironic.drivers.modules.redfish.power.sushy')
|
|
@mock.patch.object(redfish_utils, 'get_system', autospec=True)
|
|
def test_set_power_state_fail(self, mock_get_system, mock_sushy):
|
|
fake_system = mock_get_system.return_value
|
|
mock_sushy.exceptions.SushyError = MockedSushyError
|
|
fake_system.reset_system.side_effect = MockedSushyError()
|
|
|
|
with task_manager.acquire(self.context, self.node.uuid,
|
|
shared=False) as task:
|
|
self.assertRaisesRegex(
|
|
exception.RedfishError, 'Redfish set power state',
|
|
task.driver.power.set_power_state, task, states.POWER_ON)
|
|
fake_system.reset_system.assert_called_once_with(
|
|
sushy.RESET_ON)
|
|
mock_get_system.assert_called_once_with(task.node)
|
|
|
|
@mock.patch.object(redfish_utils, 'get_system', autospec=True)
|
|
def test_reboot(self, mock_get_system):
|
|
fake_system = mock_get_system.return_value
|
|
with task_manager.acquire(self.context, self.node.uuid,
|
|
shared=False) as task:
|
|
expected_values = [
|
|
(sushy.SYSTEM_POWER_STATE_ON, sushy.RESET_FORCE_RESTART),
|
|
(sushy.SYSTEM_POWER_STATE_OFF, sushy.RESET_ON)
|
|
]
|
|
|
|
for current, expected in expected_values:
|
|
fake_system.power_state = current
|
|
task.driver.power.reboot(task)
|
|
|
|
# Asserts
|
|
fake_system.reset_system.assert_called_once_with(expected)
|
|
mock_get_system.assert_called_once_with(task.node)
|
|
|
|
# Reset mocks
|
|
fake_system.reset_system.reset_mock()
|
|
mock_get_system.reset_mock()
|
|
|
|
@mock.patch('ironic.drivers.modules.redfish.power.sushy')
|
|
@mock.patch.object(redfish_utils, 'get_system', autospec=True)
|
|
def test_reboot_fail(self, mock_get_system, mock_sushy):
|
|
fake_system = mock_get_system.return_value
|
|
mock_sushy.exceptions.SushyError = MockedSushyError
|
|
fake_system.reset_system.side_effect = MockedSushyError()
|
|
|
|
with task_manager.acquire(self.context, self.node.uuid,
|
|
shared=False) as task:
|
|
fake_system.power_state = sushy.SYSTEM_POWER_STATE_ON
|
|
self.assertRaisesRegex(
|
|
exception.RedfishError, 'Redfish reboot failed',
|
|
task.driver.power.reboot, task)
|
|
fake_system.reset_system.assert_called_once_with(
|
|
sushy.RESET_FORCE_RESTART)
|
|
mock_get_system.assert_called_once_with(task.node)
|
|
|
|
def test_get_supported_power_states(self):
|
|
with task_manager.acquire(self.context, self.node.uuid,
|
|
shared=True) as task:
|
|
supported_power_states = (
|
|
task.driver.power.get_supported_power_states(task))
|
|
self.assertEqual(list(redfish_power.SET_POWER_STATE_MAP),
|
|
supported_power_states)
|