Merge "Remove ibmc hardware type"
This commit is contained in:
commit
941f2e36fa
@ -842,26 +842,11 @@ function is_deployed_by_irmc {
|
|||||||
return 1
|
return 1
|
||||||
}
|
}
|
||||||
|
|
||||||
function is_deployed_by_ibmc {
|
|
||||||
[[ "$IRONIC_DEPLOY_DRIVER" == ibmc ]] && return 0
|
|
||||||
return 1
|
|
||||||
}
|
|
||||||
|
|
||||||
function is_drac_enabled {
|
function is_drac_enabled {
|
||||||
[[ -z "${IRONIC_ENABLED_HARDWARE_TYPES%%*idrac*}" ]] && return 0
|
[[ -z "${IRONIC_ENABLED_HARDWARE_TYPES%%*idrac*}" ]] && return 0
|
||||||
return 1
|
return 1
|
||||||
}
|
}
|
||||||
|
|
||||||
function is_ibmc_enabled {
|
|
||||||
[[ -z "${IRONIC_ENABLED_HARDWARE_TYPES%%*ibmc*}" ]] && return 0
|
|
||||||
return 1
|
|
||||||
}
|
|
||||||
|
|
||||||
function is_irmc_enabled {
|
|
||||||
[[ -z "${IRONIC_ENABLED_HARDWARE_TYPES%%*irmc*}" ]] && return 0
|
|
||||||
return 1
|
|
||||||
}
|
|
||||||
|
|
||||||
function is_ansible_deploy_enabled {
|
function is_ansible_deploy_enabled {
|
||||||
[[ -z "${IRONIC_ENABLED_DEPLOY_INTERFACES%%*ansible*}" ]] && return 0
|
[[ -z "${IRONIC_ENABLED_DEPLOY_INTERFACES%%*ansible*}" ]] && return 0
|
||||||
return 1
|
return 1
|
||||||
@ -1183,10 +1168,6 @@ function install_ironic {
|
|||||||
pip_install python-dracclient
|
pip_install python-dracclient
|
||||||
fi
|
fi
|
||||||
|
|
||||||
if is_ibmc_enabled; then
|
|
||||||
pip_install python-ibmcclient
|
|
||||||
fi
|
|
||||||
|
|
||||||
if is_irmc_enabled; then
|
if is_irmc_enabled; then
|
||||||
pip_install python-scciclient pysnmp
|
pip_install python-scciclient pysnmp
|
||||||
fi
|
fi
|
||||||
@ -2598,11 +2579,6 @@ function enroll_nodes {
|
|||||||
if [[ -n "$IRONIC_DEPLOY_ISO_ID" ]]; then
|
if [[ -n "$IRONIC_DEPLOY_ISO_ID" ]]; then
|
||||||
node_options+=" --driver-info deploy_iso=$IRONIC_DEPLOY_ISO_ID"
|
node_options+=" --driver-info deploy_iso=$IRONIC_DEPLOY_ISO_ID"
|
||||||
fi
|
fi
|
||||||
elif is_deployed_by_ibmc; then
|
|
||||||
node_options+=" --driver-info ibmc_address=$bmc_address \
|
|
||||||
--driver-info ibmc_username=$bmc_username \
|
|
||||||
--driver-info ibmc_password=$bmc_passwd \
|
|
||||||
--driver-info ibmc_verify_ca=False"
|
|
||||||
fi
|
fi
|
||||||
|
|
||||||
interface_info="${mac_address}"
|
interface_info="${mac_address}"
|
||||||
|
@ -17,7 +17,6 @@ Hardware Types
|
|||||||
.. toctree::
|
.. toctree::
|
||||||
:maxdepth: 1
|
:maxdepth: 1
|
||||||
|
|
||||||
drivers/ibmc
|
|
||||||
drivers/idrac
|
drivers/idrac
|
||||||
drivers/ilo
|
drivers/ilo
|
||||||
drivers/intel-ipmi
|
drivers/intel-ipmi
|
||||||
|
@ -1,323 +0,0 @@
|
|||||||
===============
|
|
||||||
iBMC driver
|
|
||||||
===============
|
|
||||||
|
|
||||||
Overview
|
|
||||||
========
|
|
||||||
|
|
||||||
.. warning::
|
|
||||||
The ``ibmc`` driver has been deprecated and is anticipated to be removed
|
|
||||||
from Ironic at some point during or after the 2024.2 development cycle.
|
|
||||||
The anticipated forward management path is to migrate to the ``redfish``
|
|
||||||
hardware type.
|
|
||||||
|
|
||||||
The ``ibmc`` driver is targeted for Huawei V5 series rack server such as
|
|
||||||
2288H V5, CH121 V5. The iBMC hardware type enables the user to take advantage
|
|
||||||
of features of `Huawei iBMC`_ to control Huawei server.
|
|
||||||
|
|
||||||
The ``ibmc`` hardware type supports the following Ironic interfaces:
|
|
||||||
|
|
||||||
* Management Interface: Boot device management
|
|
||||||
* Power Interface: Power management
|
|
||||||
* `RAID Interface`_: RAID controller and disk management
|
|
||||||
* `Vendor Interface`_: ibmc passthru interfaces
|
|
||||||
|
|
||||||
Prerequisites
|
|
||||||
=============
|
|
||||||
|
|
||||||
The `HUAWEI iBMC Client library`_ should be installed on the ironic conductor
|
|
||||||
node(s).
|
|
||||||
|
|
||||||
For example, it can be installed with ``pip``::
|
|
||||||
|
|
||||||
sudo pip install python-ibmcclient
|
|
||||||
|
|
||||||
Enabling the iBMC driver
|
|
||||||
============================
|
|
||||||
|
|
||||||
#. Add ``ibmc`` to the list of ``enabled_hardware_types``,
|
|
||||||
``enabled_power_interfaces``, ``enabled_vendor_interfaces``
|
|
||||||
and ``enabled_management_interfaces`` in ``/etc/ironic/ironic.conf``. For example::
|
|
||||||
|
|
||||||
[DEFAULT]
|
|
||||||
...
|
|
||||||
enabled_hardware_types = ibmc
|
|
||||||
enabled_power_interfaces = ibmc
|
|
||||||
enabled_management_interfaces = ibmc
|
|
||||||
enabled_raid_interfaces = ibmc
|
|
||||||
enabled_vendor_interfaces = ibmc
|
|
||||||
|
|
||||||
#. Restart the ironic conductor service::
|
|
||||||
|
|
||||||
sudo service ironic-conductor restart
|
|
||||||
|
|
||||||
# Or, for RDO:
|
|
||||||
sudo systemctl restart openstack-ironic-conductor
|
|
||||||
|
|
||||||
Registering a node with the iBMC driver
|
|
||||||
===========================================
|
|
||||||
|
|
||||||
Nodes configured to use the driver should have the ``driver`` property
|
|
||||||
set to ``ibmc``.
|
|
||||||
|
|
||||||
The following properties are specified in the node's ``driver_info``
|
|
||||||
field:
|
|
||||||
|
|
||||||
- ``ibmc_address``:
|
|
||||||
|
|
||||||
The URL address to the ibmc controller. It must
|
|
||||||
include the authority portion of the URL, and can
|
|
||||||
optionally include the scheme. If the scheme is
|
|
||||||
missing, https is assumed.
|
|
||||||
For example: https://ibmc.example.com. This is required.
|
|
||||||
|
|
||||||
- ``ibmc_username``:
|
|
||||||
|
|
||||||
User account with admin/server-profile access
|
|
||||||
privilege. This is required.
|
|
||||||
|
|
||||||
- ``ibmc_password``:
|
|
||||||
|
|
||||||
User account password. This is required.
|
|
||||||
|
|
||||||
- ``ibmc_verify_ca``:
|
|
||||||
|
|
||||||
If ibmc_address has the **https** scheme, the
|
|
||||||
driver will use a secure (TLS_) connection when
|
|
||||||
talking to the ibmc controller. By default
|
|
||||||
(if this is set to True), the driver will try to
|
|
||||||
verify the host certificates. This can be set to
|
|
||||||
the path of a certificate file or directory with
|
|
||||||
trusted certificates that the driver will use for
|
|
||||||
verification. To disable verifying TLS_, set this
|
|
||||||
to False. This is optional.
|
|
||||||
|
|
||||||
The ``baremetal node create`` command can be used to enroll
|
|
||||||
a node with the ``ibmc`` driver. For example:
|
|
||||||
|
|
||||||
.. code-block:: bash
|
|
||||||
|
|
||||||
baremetal node create --driver ibmc
|
|
||||||
--driver-info ibmc_address=https://example.com \
|
|
||||||
--driver-info ibmc_username=admin \
|
|
||||||
--driver-info ibmc_password=password
|
|
||||||
|
|
||||||
For more information about enrolling nodes see :ref:`enrollment`
|
|
||||||
in the install guide.
|
|
||||||
|
|
||||||
RAID Interface
|
|
||||||
==============
|
|
||||||
|
|
||||||
Currently, only RAID controller which supports OOB management can be managed.
|
|
||||||
|
|
||||||
See :doc:`/admin/raid` for more information on Ironic RAID support.
|
|
||||||
|
|
||||||
The following properties are supported by the iBMC raid interface
|
|
||||||
implementation, ``ibmc``:
|
|
||||||
|
|
||||||
Mandatory properties
|
|
||||||
--------------------
|
|
||||||
|
|
||||||
* ``size_gb``: Size in gigabytes (integer) for the logical disk. Use ``MAX`` as
|
|
||||||
``size_gb`` if this logical disk is supposed to use the rest of the space
|
|
||||||
available.
|
|
||||||
* ``raid_level``: RAID level for the logical disk. Valid values are
|
|
||||||
``JBOD``, ``0``, ``1``, ``5``, ``6``, ``1+0``, ``5+0`` and ``6+0``. And it
|
|
||||||
is possible that some RAID controllers can only support a subset RAID
|
|
||||||
levels.
|
|
||||||
|
|
||||||
.. NOTE::
|
|
||||||
RAID level ``2`` is not supported by ``iBMC`` driver.
|
|
||||||
|
|
||||||
Optional properties
|
|
||||||
-------------------
|
|
||||||
|
|
||||||
* ``is_root_volume``: Optional. Specifies whether this disk is a root volume.
|
|
||||||
By default, this is ``False``.
|
|
||||||
* ``volume_name``: Optional. Name of the volume to be created. If this is not
|
|
||||||
specified, it will be N/A.
|
|
||||||
|
|
||||||
Backing physical disk hints
|
|
||||||
---------------------------
|
|
||||||
|
|
||||||
See :doc:`/admin/raid` for more information on backing disk hints.
|
|
||||||
|
|
||||||
These are machine-independent properties. The hints are specified for each
|
|
||||||
logical disk to help Ironic find the desired disks for RAID configuration.
|
|
||||||
|
|
||||||
* ``share_physical_disks``
|
|
||||||
* ``disk_type``
|
|
||||||
* ``interface_type``
|
|
||||||
* ``number_of_physical_disks``
|
|
||||||
|
|
||||||
Backing physical disks
|
|
||||||
----------------------
|
|
||||||
|
|
||||||
These are HUAWEI RAID controller dependent properties:
|
|
||||||
|
|
||||||
* ``controller``: Optional. Supported values are: RAID storage id,
|
|
||||||
RAID storage name or RAID controller name. If a bare metal server have more
|
|
||||||
than one controller, this is mandatory. Typical values would look like:
|
|
||||||
|
|
||||||
* RAID Storage Id: ``RAIDStorage0``
|
|
||||||
* RAID Storage Name: ``RAIDStorage0``
|
|
||||||
* RAID Controller Name: ``RAID Card1 Controller``.
|
|
||||||
|
|
||||||
* ``physical_disks``: Optional. Supported values are: disk-id, disk-name or
|
|
||||||
disk serial number. Typical values for hdd disk would look like:
|
|
||||||
|
|
||||||
* Disk Id: ``HDDPlaneDisk0``
|
|
||||||
* Disk Name: ``Disk0``.
|
|
||||||
* Disk SerialNumber: ``38DGK77LF77D``
|
|
||||||
|
|
||||||
Delete RAID configuration
|
|
||||||
-------------------------
|
|
||||||
|
|
||||||
For ``delete_configuration`` step, ``ibmc`` will do:
|
|
||||||
|
|
||||||
* delete all logical disks
|
|
||||||
* delete all hot-spare disks
|
|
||||||
|
|
||||||
Logical disks creation priority
|
|
||||||
-------------------------------
|
|
||||||
|
|
||||||
Logical Disks creation priority based on three properties:
|
|
||||||
|
|
||||||
* ``share_physical_disks``
|
|
||||||
* ``physical_disks``
|
|
||||||
* ``size_gb``
|
|
||||||
|
|
||||||
The logical disks creation priority strictly follow the table below, if
|
|
||||||
multiple logical disks have the same priority, then they will be created with
|
|
||||||
the same order in ``logical_disks`` array.
|
|
||||||
|
|
||||||
==================== ========================== =========
|
|
||||||
Share physical disks Specified Physical Disks Size
|
|
||||||
==================== ========================== =========
|
|
||||||
no yes int|max
|
|
||||||
no no int
|
|
||||||
yes yes int
|
|
||||||
yes yes max
|
|
||||||
yes no int
|
|
||||||
yes no max
|
|
||||||
no no max
|
|
||||||
==================== ========================== =========
|
|
||||||
|
|
||||||
Physical disks choice strategy
|
|
||||||
------------------------------
|
|
||||||
|
|
||||||
.. note::
|
|
||||||
physical-disk-group: a group of physical disks which have been used by some
|
|
||||||
logical-disks with same RAID level.
|
|
||||||
|
|
||||||
|
|
||||||
* If no ``physical_disks`` are specified, the "waste least" strategy will be
|
|
||||||
used to choose the physical disks.
|
|
||||||
|
|
||||||
* waste least disk capacity: when using disks with different capacity, it
|
|
||||||
will cause a waste of disk capacity. This is to avoid with highest
|
|
||||||
priority.
|
|
||||||
* using least total disk capacity: for example, we can create 400G RAID 5
|
|
||||||
with both 5 100G-disks and 3 200G-disks. 5 100G disks is a better
|
|
||||||
strategy because it uses a 500G capacity totally. While 3 200G-disks
|
|
||||||
are 600G totally.
|
|
||||||
* using least disk count: finally, if waste capacity and total disk
|
|
||||||
capacity are both the same (it rarely happens?), we will choose the one
|
|
||||||
with the minimum number of disks.
|
|
||||||
|
|
||||||
* when ``share_physical_disks`` option is present, ``ibmc`` driver will
|
|
||||||
create logical disk upon existing physical-disk-group list first. Only
|
|
||||||
when no existing physical-disk-group matches, then it chooses unused
|
|
||||||
physical disks with same strategy described above. When multiple exists
|
|
||||||
physical-disk-groups matches, it will use "waste least" strategy too,
|
|
||||||
the bigger capacity left the better. For example, to create a logical disk
|
|
||||||
shown below on a ``ibmc`` server which has two RAID5 logical disks already.
|
|
||||||
And the shareable capacity of this two logical-disks are 500G and 300G,
|
|
||||||
then ``ibmc`` driver will choose the second one.
|
|
||||||
|
|
||||||
.. code-block:: json
|
|
||||||
|
|
||||||
{
|
|
||||||
"logical_disks": [
|
|
||||||
{
|
|
||||||
"controller": "RAID Card1 Controller",
|
|
||||||
"raid_level": "5",
|
|
||||||
"size_gb": 100,
|
|
||||||
"share_physical_disks": true
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
|
|
||||||
And the ``ibmc`` server has two RAID5 logical disks already.
|
|
||||||
|
|
||||||
* When ``size_gb`` is set to ``MAX``, ``ibmc`` driver will auto work through
|
|
||||||
all possible cases and choose the "best" solution which has the biggest
|
|
||||||
capacity and use least capacity. For example: to create a RAID 5+0 logical
|
|
||||||
disk with MAX size in a server has 9 200G-disks, it will finally choose
|
|
||||||
"8 disks + span-number 2" but not "9 disks + span-number 3". Although they
|
|
||||||
both have 1200G capacity totally, but the former uses only 8 disks and the
|
|
||||||
latter uses 9 disks. If you want to choose the latter solution, you can
|
|
||||||
specified the disk count to use by adding ``number_of_physical_disks``
|
|
||||||
option.
|
|
||||||
|
|
||||||
.. code-block:: json
|
|
||||||
|
|
||||||
{
|
|
||||||
"logical_disks": [
|
|
||||||
{
|
|
||||||
"controller": "RAID Card1 Controller",
|
|
||||||
"raid_level": "5+0",
|
|
||||||
"size_gb": "MAX"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
Examples
|
|
||||||
--------
|
|
||||||
|
|
||||||
In a typical scenario we may want to create:
|
|
||||||
* RAID 5, 500G, root OS volume with 3 disks
|
|
||||||
* RAID 5, rest available space, data volume with rest disks
|
|
||||||
|
|
||||||
.. code-block:: json
|
|
||||||
|
|
||||||
{
|
|
||||||
"logical_disks": [
|
|
||||||
{
|
|
||||||
"volume_name": "os_volume",
|
|
||||||
"controller": "RAID Card1 Controller",
|
|
||||||
"is_root_volume": "True",
|
|
||||||
"physical_disks": [
|
|
||||||
"Disk0",
|
|
||||||
"Disk1",
|
|
||||||
"Disk2"
|
|
||||||
],
|
|
||||||
"raid_level": "5",
|
|
||||||
"size_gb": "500"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"volume_name": "data_volume",
|
|
||||||
"controller": "RAID Card1 Controller",
|
|
||||||
"raid_level": "5",
|
|
||||||
"size_gb": "MAX"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
|
|
||||||
Vendor Interface
|
|
||||||
=========================================
|
|
||||||
|
|
||||||
The ``ibmc`` hardware type provides vendor passthru interfaces shown below:
|
|
||||||
|
|
||||||
|
|
||||||
======================== ============ ======================================
|
|
||||||
Method Name HTTP Method Description
|
|
||||||
======================== ============ ======================================
|
|
||||||
boot_up_seq GET Query boot up sequence
|
|
||||||
get_raid_controller_list GET Query RAID controller summary info
|
|
||||||
======================== ============ ======================================
|
|
||||||
|
|
||||||
.. _Huawei iBMC: https://e.huawei.com/en/products/computing/kunpeng/accessories/ibmc
|
|
||||||
.. _TLS: https://en.wikipedia.org/wiki/Transport_Layer_Security
|
|
||||||
.. _HUAWEI iBMC Client library: https://pypi.org/project/python-ibmcclient/
|
|
@ -14,8 +14,5 @@ python-dracclient>=5.1.0,<9.0.0
|
|||||||
# Ansible-deploy interface
|
# Ansible-deploy interface
|
||||||
ansible>=2.7
|
ansible>=2.7
|
||||||
|
|
||||||
# HUAWEI iBMC hardware type uses the python-ibmcclient library
|
|
||||||
python-ibmcclient>=0.2.2,<0.3.0
|
|
||||||
|
|
||||||
# Dell EMC iDRAC sushy OEM extension
|
# Dell EMC iDRAC sushy OEM extension
|
||||||
sushy-oem-idrac>=5.0.0,<6.0.0
|
sushy-oem-idrac>=5.0.0,<6.0.0
|
||||||
|
@ -720,14 +720,6 @@ class InvalidKickstartFile(Invalid):
|
|||||||
_msg_fmt = _("The kickstart file is not valid.")
|
_msg_fmt = _("The kickstart file is not valid.")
|
||||||
|
|
||||||
|
|
||||||
class IBMCError(DriverOperationError):
|
|
||||||
_msg_fmt = _("IBMC exception occurred on node %(node)s. Error: %(error)s")
|
|
||||||
|
|
||||||
|
|
||||||
class IBMCConnectionError(IBMCError):
|
|
||||||
_msg_fmt = _("IBMC connection failed for node %(node)s: %(error)s")
|
|
||||||
|
|
||||||
|
|
||||||
class ClientSideError(RuntimeError):
|
class ClientSideError(RuntimeError):
|
||||||
def __init__(self, msg=None, status_code=400, faultcode='Client'):
|
def __init__(self, msg=None, status_code=400, faultcode='Client'):
|
||||||
self.msg = msg
|
self.msg = msg
|
||||||
|
@ -32,7 +32,6 @@ from ironic.conf import drac
|
|||||||
from ironic.conf import fake
|
from ironic.conf import fake
|
||||||
from ironic.conf import glance
|
from ironic.conf import glance
|
||||||
from ironic.conf import healthcheck
|
from ironic.conf import healthcheck
|
||||||
from ironic.conf import ibmc
|
|
||||||
from ironic.conf import ilo
|
from ironic.conf import ilo
|
||||||
from ironic.conf import inspector
|
from ironic.conf import inspector
|
||||||
from ironic.conf import inventory
|
from ironic.conf import inventory
|
||||||
@ -69,7 +68,6 @@ dnsmasq.register_opts(CONF)
|
|||||||
fake.register_opts(CONF)
|
fake.register_opts(CONF)
|
||||||
glance.register_opts(CONF)
|
glance.register_opts(CONF)
|
||||||
healthcheck.register_opts(CONF)
|
healthcheck.register_opts(CONF)
|
||||||
ibmc.register_opts(CONF)
|
|
||||||
ilo.register_opts(CONF)
|
ilo.register_opts(CONF)
|
||||||
inspector.register_opts(CONF)
|
inspector.register_opts(CONF)
|
||||||
inventory.register_opts(CONF)
|
inventory.register_opts(CONF)
|
||||||
|
@ -1,35 +0,0 @@
|
|||||||
#
|
|
||||||
# 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.
|
|
||||||
|
|
||||||
# Version 1.0.0
|
|
||||||
|
|
||||||
from oslo_config import cfg
|
|
||||||
|
|
||||||
from ironic.common.i18n import _
|
|
||||||
|
|
||||||
opts = [
|
|
||||||
cfg.IntOpt('connection_attempts',
|
|
||||||
min=1,
|
|
||||||
default=5,
|
|
||||||
help=_('Maximum number of attempts to try to connect '
|
|
||||||
'to iBMC')),
|
|
||||||
cfg.IntOpt('connection_retry_interval',
|
|
||||||
min=1,
|
|
||||||
default=4,
|
|
||||||
help=_('Number of seconds to wait between attempts to '
|
|
||||||
'connect to iBMC'))
|
|
||||||
]
|
|
||||||
|
|
||||||
|
|
||||||
def register_opts(conf):
|
|
||||||
conf.register_opts(opts, group='ibmc')
|
|
@ -1,47 +0,0 @@
|
|||||||
#
|
|
||||||
# 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.
|
|
||||||
"""
|
|
||||||
iBMC Driver for managing HUAWEI V5 series rack servers such as 2288H V5,
|
|
||||||
CH121 V5.
|
|
||||||
"""
|
|
||||||
|
|
||||||
from ironic.drivers import generic
|
|
||||||
from ironic.drivers.modules.ibmc import management as ibmc_mgmt
|
|
||||||
from ironic.drivers.modules.ibmc import power as ibmc_power
|
|
||||||
from ironic.drivers.modules.ibmc import raid as ibmc_raid
|
|
||||||
from ironic.drivers.modules.ibmc import vendor as ibmc_vendor
|
|
||||||
from ironic.drivers.modules import noop
|
|
||||||
|
|
||||||
|
|
||||||
class IBMCHardware(generic.GenericHardware):
|
|
||||||
"""Huawei iBMC hardware type."""
|
|
||||||
|
|
||||||
@property
|
|
||||||
def supported_management_interfaces(self):
|
|
||||||
"""List of supported management interfaces."""
|
|
||||||
return [ibmc_mgmt.IBMCManagement]
|
|
||||||
|
|
||||||
@property
|
|
||||||
def supported_power_interfaces(self):
|
|
||||||
"""List of supported power interfaces."""
|
|
||||||
return [ibmc_power.IBMCPower]
|
|
||||||
|
|
||||||
@property
|
|
||||||
def supported_vendor_interfaces(self):
|
|
||||||
"""List of supported vendor interfaces."""
|
|
||||||
return [ibmc_vendor.IBMCVendor, noop.NoVendor]
|
|
||||||
|
|
||||||
@property
|
|
||||||
def supported_raid_interfaces(self):
|
|
||||||
"""List of supported raid interfaces."""
|
|
||||||
return [ibmc_raid.IbmcRAID, noop.NoRAID]
|
|
@ -1,241 +0,0 @@
|
|||||||
# Copyright 2019 HUAWEI, Inc. All Rights Reserved.
|
|
||||||
# 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.
|
|
||||||
"""
|
|
||||||
iBMC Management Interface
|
|
||||||
"""
|
|
||||||
|
|
||||||
from oslo_log import log
|
|
||||||
from oslo_utils import importutils
|
|
||||||
|
|
||||||
from ironic.common import exception
|
|
||||||
from ironic.common.i18n import _
|
|
||||||
from ironic.conductor import task_manager
|
|
||||||
from ironic.drivers import base
|
|
||||||
from ironic.drivers.modules.ibmc import mappings
|
|
||||||
from ironic.drivers.modules.ibmc import utils
|
|
||||||
|
|
||||||
constants = importutils.try_import('ibmc_client.constants')
|
|
||||||
ibmc_client = importutils.try_import('ibmc_client')
|
|
||||||
|
|
||||||
LOG = log.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
class IBMCManagement(base.ManagementInterface):
|
|
||||||
|
|
||||||
# NOTE(TheJulia): Deprecating November 2023 in favor of Redfish
|
|
||||||
# and due to a lack of active driver maintenance.
|
|
||||||
supported = False
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
"""Initialize the iBMC management interface
|
|
||||||
|
|
||||||
:raises: DriverLoadError if the driver can't be loaded due to
|
|
||||||
missing dependencies
|
|
||||||
"""
|
|
||||||
super(IBMCManagement, self).__init__()
|
|
||||||
if not ibmc_client:
|
|
||||||
raise exception.DriverLoadError(
|
|
||||||
driver='ibmc',
|
|
||||||
reason=_('Unable to import the python-ibmcclient library'))
|
|
||||||
|
|
||||||
def get_properties(self):
|
|
||||||
"""Return the properties of the interface.
|
|
||||||
|
|
||||||
:returns: dictionary of <property name>:<property description> entries.
|
|
||||||
"""
|
|
||||||
return utils.COMMON_PROPERTIES.copy()
|
|
||||||
|
|
||||||
def validate(self, task):
|
|
||||||
"""Validates the driver information needed by the iBMC driver.
|
|
||||||
|
|
||||||
:param task: A TaskManager instance containing the node to act on.
|
|
||||||
:raises: InvalidParameterValue on malformed parameter(s)
|
|
||||||
:raises: MissingParameterValue on missing parameter(s)
|
|
||||||
"""
|
|
||||||
utils.parse_driver_info(task.node)
|
|
||||||
|
|
||||||
@utils.handle_ibmc_exception('get iBMC supported boot devices')
|
|
||||||
def get_supported_boot_devices(self, task):
|
|
||||||
"""Get a list of the supported boot devices.
|
|
||||||
|
|
||||||
:param task: a task from TaskManager.
|
|
||||||
:raises: InvalidParameterValue on malformed parameter(s)
|
|
||||||
:raises: MissingParameterValue on missing parameter(s)
|
|
||||||
:raises: IBMCConnectionError when it fails to connect to iBMC
|
|
||||||
:raises: IBMCError when iBMC responses an error information
|
|
||||||
:returns: A list with the supported boot devices defined
|
|
||||||
in :mod:`ironic.common.boot_devices`.
|
|
||||||
"""
|
|
||||||
ibmc = utils.parse_driver_info(task.node)
|
|
||||||
with ibmc_client.connect(**ibmc) as conn:
|
|
||||||
system = conn.system.get()
|
|
||||||
boot_source_override = system.boot_source_override
|
|
||||||
return list(map(mappings.GET_BOOT_DEVICE_MAP.get,
|
|
||||||
boot_source_override.supported_boot_devices))
|
|
||||||
|
|
||||||
@task_manager.require_exclusive_lock
|
|
||||||
@utils.handle_ibmc_exception('set iBMC boot device')
|
|
||||||
def set_boot_device(self, task, device, persistent=False):
|
|
||||||
"""Set the boot device for a node.
|
|
||||||
|
|
||||||
:param task: A task from TaskManager.
|
|
||||||
:param device: The boot device, one of
|
|
||||||
:mod:`ironic.common.boot_device`.
|
|
||||||
:param persistent: Boolean value. True if the boot device will
|
|
||||||
persist to all future boots, False if not.
|
|
||||||
Default: False.
|
|
||||||
:raises: InvalidParameterValue on malformed parameter(s)
|
|
||||||
:raises: MissingParameterValue on missing parameter(s)
|
|
||||||
:raises: IBMCConnectionError when it fails to connect to iBMC
|
|
||||||
:raises: IBMCError when iBMC responses an error information
|
|
||||||
"""
|
|
||||||
ibmc = utils.parse_driver_info(task.node)
|
|
||||||
with ibmc_client.connect(**ibmc) as conn:
|
|
||||||
boot_device = mappings.SET_BOOT_DEVICE_MAP[device]
|
|
||||||
enabled = mappings.SET_BOOT_DEVICE_PERSISTENT_MAP[persistent]
|
|
||||||
conn.system.set_boot_source(boot_device, enabled=enabled)
|
|
||||||
|
|
||||||
@utils.handle_ibmc_exception('get iBMC boot device')
|
|
||||||
def get_boot_device(self, task):
|
|
||||||
"""Get the current boot device for a node.
|
|
||||||
|
|
||||||
:param task: A task from TaskManager.
|
|
||||||
:raises: InvalidParameterValue on malformed parameter(s)
|
|
||||||
:raises: MissingParameterValue on missing parameter(s)
|
|
||||||
:raises: IBMCConnectionError when it fails to connect to iBMC
|
|
||||||
:raises: IBMCError when iBMC responses an error information
|
|
||||||
:returns: a dictionary containing:
|
|
||||||
|
|
||||||
:boot_device:
|
|
||||||
the boot device, one of :mod:`ironic.common.boot_devices` or
|
|
||||||
None if it is unknown.
|
|
||||||
:persistent:
|
|
||||||
Boolean value or None, True if the boot device persists,
|
|
||||||
False otherwise. None if it's disabled.
|
|
||||||
|
|
||||||
"""
|
|
||||||
ibmc = utils.parse_driver_info(task.node)
|
|
||||||
with ibmc_client.connect(**ibmc) as conn:
|
|
||||||
system = conn.system.get()
|
|
||||||
boot_source_override = system.boot_source_override
|
|
||||||
boot_device = boot_source_override.target
|
|
||||||
enabled = boot_source_override.enabled
|
|
||||||
return {
|
|
||||||
'boot_device': mappings.GET_BOOT_DEVICE_MAP.get(boot_device),
|
|
||||||
'persistent':
|
|
||||||
mappings.GET_BOOT_DEVICE_PERSISTENT_MAP.get(enabled)
|
|
||||||
}
|
|
||||||
|
|
||||||
def get_supported_boot_modes(self, task):
|
|
||||||
"""Get a list of the supported boot modes.
|
|
||||||
|
|
||||||
:param task: A task from TaskManager.
|
|
||||||
:returns: A list with the supported boot modes defined
|
|
||||||
in :mod:`ironic.common.boot_modes`. If boot
|
|
||||||
mode support can't be determined, empty list
|
|
||||||
is returned.
|
|
||||||
"""
|
|
||||||
return list(mappings.SET_BOOT_MODE_MAP)
|
|
||||||
|
|
||||||
@task_manager.require_exclusive_lock
|
|
||||||
@utils.handle_ibmc_exception('set iBMC boot mode')
|
|
||||||
def set_boot_mode(self, task, mode):
|
|
||||||
"""Set the boot mode for a node.
|
|
||||||
|
|
||||||
Set the boot mode to use on next reboot of the node.
|
|
||||||
|
|
||||||
:param task: A task from TaskManager.
|
|
||||||
:param mode: The boot mode, one of
|
|
||||||
:mod:`ironic.common.boot_modes`.
|
|
||||||
:raises: InvalidParameterValue on malformed parameter(s)
|
|
||||||
:raises: MissingParameterValue on missing parameter(s)
|
|
||||||
:raises: IBMCConnectionError when it fails to connect to iBMC
|
|
||||||
:raises: IBMCError when iBMC responses an error information
|
|
||||||
"""
|
|
||||||
ibmc = utils.parse_driver_info(task.node)
|
|
||||||
with ibmc_client.connect(**ibmc) as conn:
|
|
||||||
system = conn.system.get()
|
|
||||||
boot_source_override = system.boot_source_override
|
|
||||||
boot_device = boot_source_override.target
|
|
||||||
boot_override = boot_source_override.enabled
|
|
||||||
|
|
||||||
# Copied from redfish driver
|
|
||||||
# TODO(Qianbiao.NG) what if boot device is "NONE"?
|
|
||||||
if not boot_device:
|
|
||||||
error_msg = (_('Cannot change boot mode on node %(node)s '
|
|
||||||
'because its boot device is not set.') %
|
|
||||||
{'node': task.node.uuid})
|
|
||||||
LOG.error(error_msg)
|
|
||||||
raise exception.IBMCError(error_msg)
|
|
||||||
|
|
||||||
# TODO(Qianbiao.NG) what if boot override is "disabled"?
|
|
||||||
if not boot_override:
|
|
||||||
i18n = _('Cannot change boot mode on node %(node)s '
|
|
||||||
'because its boot source override is not set.')
|
|
||||||
error_msg = i18n % {'node': task.node.uuid}
|
|
||||||
LOG.error(error_msg)
|
|
||||||
raise exception.IBMCError(error_msg)
|
|
||||||
|
|
||||||
boot_mode = mappings.SET_BOOT_MODE_MAP[mode]
|
|
||||||
conn.system.set_boot_source(boot_device,
|
|
||||||
enabled=boot_override,
|
|
||||||
mode=boot_mode)
|
|
||||||
|
|
||||||
@utils.handle_ibmc_exception('get iBMC boot mode')
|
|
||||||
def get_boot_mode(self, task):
|
|
||||||
"""Get the current boot mode for a node.
|
|
||||||
|
|
||||||
Provides the current boot mode of the node.
|
|
||||||
|
|
||||||
:param task: A task from TaskManager.
|
|
||||||
:raises: InvalidParameterValue on malformed parameter(s)
|
|
||||||
:raises: MissingParameterValue on missing parameter(s)
|
|
||||||
:raises: IBMCConnectionError when it fails to connect to iBMC
|
|
||||||
:raises: IBMCError when iBMC responses an error information
|
|
||||||
:returns: The boot mode, one of :mod:`ironic.common.boot_mode` or
|
|
||||||
None if it is unknown.
|
|
||||||
"""
|
|
||||||
ibmc = utils.parse_driver_info(task.node)
|
|
||||||
with ibmc_client.connect(**ibmc) as conn:
|
|
||||||
system = conn.system.get()
|
|
||||||
boot_source_override = system.boot_source_override
|
|
||||||
boot_mode = boot_source_override.mode
|
|
||||||
return mappings.GET_BOOT_MODE_MAP.get(boot_mode)
|
|
||||||
|
|
||||||
def get_sensors_data(self, task):
|
|
||||||
"""Get sensors data.
|
|
||||||
|
|
||||||
Not implemented for this driver.
|
|
||||||
|
|
||||||
:raises: NotImplementedError
|
|
||||||
"""
|
|
||||||
raise NotImplementedError()
|
|
||||||
|
|
||||||
@task_manager.require_exclusive_lock
|
|
||||||
@utils.handle_ibmc_exception('inject iBMC NMI')
|
|
||||||
def inject_nmi(self, task):
|
|
||||||
"""Inject NMI, Non Maskable Interrupt.
|
|
||||||
|
|
||||||
Inject NMI (Non Maskable Interrupt) for a node immediately.
|
|
||||||
|
|
||||||
:param task: A TaskManager instance containing the node to act on.
|
|
||||||
:raises: InvalidParameterValue on malformed parameter(s)
|
|
||||||
:raises: MissingParameterValue on missing parameter(s)
|
|
||||||
:raises: IBMCConnectionError when it fails to connect to iBMC
|
|
||||||
:raises: IBMCError when iBMC responses an error information
|
|
||||||
"""
|
|
||||||
ibmc = utils.parse_driver_info(task.node)
|
|
||||||
with ibmc_client.connect(**ibmc) as conn:
|
|
||||||
conn.system.reset(constants.RESET_NMI)
|
|
@ -1,70 +0,0 @@
|
|||||||
#
|
|
||||||
# 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.
|
|
||||||
"""
|
|
||||||
iBMC and Ironic constants mapping
|
|
||||||
"""
|
|
||||||
|
|
||||||
from oslo_utils import importutils
|
|
||||||
|
|
||||||
from ironic.common import boot_devices
|
|
||||||
from ironic.common import boot_modes
|
|
||||||
from ironic.common import states
|
|
||||||
from ironic.drivers.modules.ibmc import utils
|
|
||||||
|
|
||||||
constants = importutils.try_import('ibmc_client.constants')
|
|
||||||
|
|
||||||
if constants:
|
|
||||||
# Set power state mapping
|
|
||||||
SET_POWER_STATE_MAP = {
|
|
||||||
states.POWER_ON: constants.RESET_ON,
|
|
||||||
states.POWER_OFF: constants.RESET_FORCE_OFF,
|
|
||||||
states.REBOOT: constants.RESET_FORCE_RESTART,
|
|
||||||
states.SOFT_REBOOT: constants.RESET_FORCE_POWER_CYCLE,
|
|
||||||
states.SOFT_POWER_OFF: constants.RESET_GRACEFUL_SHUTDOWN,
|
|
||||||
}
|
|
||||||
|
|
||||||
# Get power state mapping
|
|
||||||
GET_POWER_STATE_MAP = {
|
|
||||||
constants.SYSTEM_POWER_STATE_ON: states.POWER_ON,
|
|
||||||
constants.SYSTEM_POWER_STATE_OFF: states.POWER_OFF,
|
|
||||||
}
|
|
||||||
|
|
||||||
# Boot device mapping
|
|
||||||
GET_BOOT_DEVICE_MAP = {
|
|
||||||
constants.BOOT_SOURCE_TARGET_NONE: 'none',
|
|
||||||
constants.BOOT_SOURCE_TARGET_PXE: boot_devices.PXE,
|
|
||||||
constants.BOOT_SOURCE_TARGET_FLOPPY: 'floppy',
|
|
||||||
constants.BOOT_SOURCE_TARGET_CD: boot_devices.CDROM,
|
|
||||||
constants.BOOT_SOURCE_TARGET_HDD: boot_devices.DISK,
|
|
||||||
constants.BOOT_SOURCE_TARGET_BIOS_SETUP: boot_devices.BIOS,
|
|
||||||
}
|
|
||||||
|
|
||||||
SET_BOOT_DEVICE_MAP = utils.revert_dictionary(GET_BOOT_DEVICE_MAP)
|
|
||||||
|
|
||||||
# Boot mode mapping
|
|
||||||
GET_BOOT_MODE_MAP = {
|
|
||||||
constants.BOOT_SOURCE_MODE_BIOS: boot_modes.LEGACY_BIOS,
|
|
||||||
constants.BOOT_SOURCE_MODE_UEFI: boot_modes.UEFI,
|
|
||||||
}
|
|
||||||
|
|
||||||
SET_BOOT_MODE_MAP = utils.revert_dictionary(GET_BOOT_MODE_MAP)
|
|
||||||
|
|
||||||
# Boot device persistent mapping
|
|
||||||
GET_BOOT_DEVICE_PERSISTENT_MAP = {
|
|
||||||
constants.BOOT_SOURCE_ENABLED_ONCE: False,
|
|
||||||
constants.BOOT_SOURCE_ENABLED_CONTINUOUS: True,
|
|
||||||
constants.BOOT_SOURCE_ENABLED_DISABLED: None,
|
|
||||||
}
|
|
||||||
|
|
||||||
SET_BOOT_DEVICE_PERSISTENT_MAP = utils.revert_dictionary(
|
|
||||||
GET_BOOT_DEVICE_PERSISTENT_MAP)
|
|
@ -1,149 +0,0 @@
|
|||||||
#
|
|
||||||
# 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.
|
|
||||||
"""
|
|
||||||
iBMC Power Interface
|
|
||||||
"""
|
|
||||||
|
|
||||||
from oslo_log import log
|
|
||||||
from oslo_utils import importutils
|
|
||||||
|
|
||||||
from ironic.common import exception
|
|
||||||
from ironic.common.i18n import _
|
|
||||||
from ironic.common import states
|
|
||||||
from ironic.conductor import task_manager
|
|
||||||
from ironic.conductor import utils as cond_utils
|
|
||||||
from ironic.drivers import base
|
|
||||||
from ironic.drivers.modules.ibmc import mappings
|
|
||||||
from ironic.drivers.modules.ibmc import utils
|
|
||||||
|
|
||||||
constants = importutils.try_import('ibmc_client.constants')
|
|
||||||
ibmc_client = importutils.try_import('ibmc_client')
|
|
||||||
|
|
||||||
LOG = log.getLogger(__name__)
|
|
||||||
|
|
||||||
EXPECT_POWER_STATE_MAP = {
|
|
||||||
states.REBOOT: states.POWER_ON,
|
|
||||||
states.SOFT_REBOOT: states.POWER_ON,
|
|
||||||
states.SOFT_POWER_OFF: states.POWER_OFF,
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
class IBMCPower(base.PowerInterface):
|
|
||||||
|
|
||||||
# NOTE(TheJulia): Deprecating November 2023 in favor of Redfish
|
|
||||||
# and due to a lack of active driver maintenance.
|
|
||||||
supported = False
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
"""Initialize the iBMC power interface.
|
|
||||||
|
|
||||||
:raises: DriverLoadError if the driver can't be loaded due to
|
|
||||||
missing dependencies
|
|
||||||
"""
|
|
||||||
super(IBMCPower, self).__init__()
|
|
||||||
if not ibmc_client:
|
|
||||||
raise exception.DriverLoadError(
|
|
||||||
driver='ibmc',
|
|
||||||
reason=_('Unable to import the python-ibmcclient library'))
|
|
||||||
|
|
||||||
def get_properties(self):
|
|
||||||
"""Return the properties of the interface.
|
|
||||||
|
|
||||||
:returns: dictionary of <property name>:<property description> entries.
|
|
||||||
"""
|
|
||||||
return utils.COMMON_PROPERTIES.copy()
|
|
||||||
|
|
||||||
def validate(self, task):
|
|
||||||
"""Validates the driver information needed by the iBMC driver.
|
|
||||||
|
|
||||||
:param task: A TaskManager instance containing the node to act on.
|
|
||||||
:raises: InvalidParameterValue on malformed parameter(s)
|
|
||||||
:raises: MissingParameterValue on missing parameter(s)
|
|
||||||
"""
|
|
||||||
utils.parse_driver_info(task.node)
|
|
||||||
|
|
||||||
@utils.handle_ibmc_exception('get iBMC power state')
|
|
||||||
def get_power_state(self, task):
|
|
||||||
"""Get the current power state of the task's node.
|
|
||||||
|
|
||||||
:param task: A TaskManager instance containing the node to act on.
|
|
||||||
:returns: A power state. One of :mod:`ironic.common.states`.
|
|
||||||
:raises: InvalidParameterValue on malformed parameter(s)
|
|
||||||
:raises: MissingParameterValue on missing parameter(s)
|
|
||||||
:raises: IBMCConnectionError when it fails to connect to iBMC
|
|
||||||
:raises: IBMCError when iBMC responses an error information
|
|
||||||
"""
|
|
||||||
ibmc = utils.parse_driver_info(task.node)
|
|
||||||
with ibmc_client.connect(**ibmc) as conn:
|
|
||||||
system = conn.system.get()
|
|
||||||
return mappings.GET_POWER_STATE_MAP.get(system.power_state)
|
|
||||||
|
|
||||||
@task_manager.require_exclusive_lock
|
|
||||||
@utils.handle_ibmc_exception('set iBMC power state')
|
|
||||||
def set_power_state(self, task, power_state, timeout=None):
|
|
||||||
"""Set the power state of the task's node.
|
|
||||||
|
|
||||||
:param task: A TaskManager instance containing the node to act on.
|
|
||||||
:param power_state: Any power state from :mod:`ironic.common.states`.
|
|
||||||
:param timeout: Time to wait for the node to reach the requested state.
|
|
||||||
:raises: InvalidParameterValue on malformed parameter(s)
|
|
||||||
:raises: MissingParameterValue if a required parameter is missing.
|
|
||||||
:raises: IBMCConnectionError when it fails to connect to iBMC
|
|
||||||
:raises: IBMCError when iBMC responses an error information
|
|
||||||
"""
|
|
||||||
ibmc = utils.parse_driver_info(task.node)
|
|
||||||
with ibmc_client.connect(**ibmc) as conn:
|
|
||||||
reset_type = mappings.SET_POWER_STATE_MAP.get(power_state)
|
|
||||||
conn.system.reset(reset_type)
|
|
||||||
|
|
||||||
target_state = EXPECT_POWER_STATE_MAP.get(power_state, power_state)
|
|
||||||
cond_utils.node_wait_for_power_state(task, target_state,
|
|
||||||
timeout=timeout)
|
|
||||||
|
|
||||||
@task_manager.require_exclusive_lock
|
|
||||||
@utils.handle_ibmc_exception('reboot iBMC')
|
|
||||||
def reboot(self, task, timeout=None):
|
|
||||||
"""Perform a hard reboot of the task's node.
|
|
||||||
|
|
||||||
:param task: A TaskManager instance containing the node to act on.
|
|
||||||
:param timeout: Time to wait for the node to become powered on.
|
|
||||||
:raises: InvalidParameterValue on malformed parameter(s)
|
|
||||||
:raises: MissingParameterValue if a required parameter is missing.
|
|
||||||
:raises: IBMCConnectionError when it fails to connect to iBMC
|
|
||||||
:raises: IBMCError when iBMC responses an error information
|
|
||||||
"""
|
|
||||||
ibmc = utils.parse_driver_info(task.node)
|
|
||||||
with ibmc_client.connect(**ibmc) as conn:
|
|
||||||
system = conn.system.get()
|
|
||||||
current_power_state = (
|
|
||||||
mappings.GET_POWER_STATE_MAP.get(system.power_state)
|
|
||||||
)
|
|
||||||
if current_power_state == states.POWER_ON:
|
|
||||||
conn.system.reset(
|
|
||||||
mappings.SET_POWER_STATE_MAP.get(states.REBOOT))
|
|
||||||
else:
|
|
||||||
conn.system.reset(
|
|
||||||
mappings.SET_POWER_STATE_MAP.get(states.POWER_ON))
|
|
||||||
|
|
||||||
cond_utils.node_wait_for_power_state(task, states.POWER_ON,
|
|
||||||
timeout=timeout)
|
|
||||||
|
|
||||||
def get_supported_power_states(self, task):
|
|
||||||
"""Get a list of the supported power states.
|
|
||||||
|
|
||||||
:param task: A TaskManager instance containing the node to act on.
|
|
||||||
Not used by this driver at the moment.
|
|
||||||
:returns: A list with the supported power states defined
|
|
||||||
in :mod:`ironic.common.states`.
|
|
||||||
"""
|
|
||||||
return list(mappings.SET_POWER_STATE_MAP)
|
|
@ -1,203 +0,0 @@
|
|||||||
# 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.
|
|
||||||
|
|
||||||
"""
|
|
||||||
iBMC RAID configuration specific methods
|
|
||||||
"""
|
|
||||||
|
|
||||||
from ironic_lib import metrics_utils
|
|
||||||
from oslo_log import log as logging
|
|
||||||
from oslo_utils import importutils
|
|
||||||
|
|
||||||
from ironic.common.i18n import _
|
|
||||||
from ironic.common import raid
|
|
||||||
from ironic import conf
|
|
||||||
from ironic.drivers import base
|
|
||||||
from ironic.drivers.modules.ibmc import utils
|
|
||||||
|
|
||||||
constants = importutils.try_import('ibmc_client.constants')
|
|
||||||
ibmc_client = importutils.try_import('ibmc_client')
|
|
||||||
ibmc_error = importutils.try_import('ibmc_client.exceptions')
|
|
||||||
|
|
||||||
CONF = conf.CONF
|
|
||||||
LOG = logging.getLogger(__name__)
|
|
||||||
METRICS = metrics_utils.get_metrics_logger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
class IbmcRAID(base.RAIDInterface):
|
|
||||||
"""Implementation of RAIDInterface for iBMC."""
|
|
||||||
|
|
||||||
# NOTE(TheJulia): Deprecating November 2023 in favor of Redfish
|
|
||||||
# and due to a lack of active driver maintenance.
|
|
||||||
supported = False
|
|
||||||
|
|
||||||
RAID_APPLY_CONFIGURATION_ARGSINFO = {
|
|
||||||
"raid_config": {
|
|
||||||
"description": "The RAID configuration to apply.",
|
|
||||||
"required": True,
|
|
||||||
},
|
|
||||||
"create_root_volume": {
|
|
||||||
"description": (
|
|
||||||
"Setting this to 'False' indicates not to create root "
|
|
||||||
"volume that is specified in 'raid_config'. Default "
|
|
||||||
"value is 'True'."
|
|
||||||
),
|
|
||||||
"required": False,
|
|
||||||
},
|
|
||||||
"create_nonroot_volumes": {
|
|
||||||
"description": (
|
|
||||||
"Setting this to 'False' indicates not to create "
|
|
||||||
"non-root volumes (all except the root volume) in "
|
|
||||||
"'raid_config'. Default value is 'True'."
|
|
||||||
),
|
|
||||||
"required": False,
|
|
||||||
},
|
|
||||||
"delete_existing": {
|
|
||||||
"description": (
|
|
||||||
"Setting this to 'True' indicates to delete existing RAID "
|
|
||||||
"configuration prior to creating the new configuration. "
|
|
||||||
"Default value is 'True'."
|
|
||||||
),
|
|
||||||
"required": False,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
def get_properties(self):
|
|
||||||
"""Return the properties of the interface.
|
|
||||||
|
|
||||||
:returns: dictionary of <property name>:<property description> entries.
|
|
||||||
"""
|
|
||||||
return utils.COMMON_PROPERTIES.copy()
|
|
||||||
|
|
||||||
@utils.handle_ibmc_exception('delete iBMC RAID configuration')
|
|
||||||
def _delete_raid_configuration(self, task):
|
|
||||||
"""Delete the RAID configuration through `python-ibmcclient` lib.
|
|
||||||
|
|
||||||
:param task: a TaskManager instance containing the node to act on.
|
|
||||||
"""
|
|
||||||
ibmc = utils.parse_driver_info(task.node)
|
|
||||||
with ibmc_client.connect(**ibmc) as conn:
|
|
||||||
# NOTE(qianbiao.ng): To reduce review workload, we should keep all
|
|
||||||
# delete logic in python-ibmcclient. And delete raid configuration
|
|
||||||
# logic should be synchronized. if async required, do it in
|
|
||||||
# python-ibmcclient.
|
|
||||||
conn.system.storage.delete_all_raid_configuration()
|
|
||||||
|
|
||||||
@utils.handle_ibmc_exception('create iBMC RAID configuration')
|
|
||||||
def _create_raid_configuration(self, task, logical_disks):
|
|
||||||
"""Create the RAID configuration through `python-ibmcclient` lib.
|
|
||||||
|
|
||||||
:param task: a TaskManager instance containing the node to act on.
|
|
||||||
:param logical_disks: a list of JSON dictionaries which represents
|
|
||||||
the logical disks to be created. The JSON dictionary should match
|
|
||||||
the (ironic.drivers.raid_config_schema.json) scheme.
|
|
||||||
"""
|
|
||||||
ibmc = utils.parse_driver_info(task.node)
|
|
||||||
with ibmc_client.connect(**ibmc) as conn:
|
|
||||||
# NOTE(qianbiao.ng): To reduce review workload, we should keep all
|
|
||||||
# apply logic in python-ibmcclient. And apply raid configuration
|
|
||||||
# logic should be synchronized. if async required, do it in
|
|
||||||
# python-ibmcclient.
|
|
||||||
conn.system.storage.apply_raid_configuration(logical_disks)
|
|
||||||
|
|
||||||
@base.deploy_step(priority=0,
|
|
||||||
argsinfo=RAID_APPLY_CONFIGURATION_ARGSINFO)
|
|
||||||
def apply_configuration(self, task, raid_config, create_root_volume=True,
|
|
||||||
create_nonroot_volumes=False):
|
|
||||||
return super(IbmcRAID, self).apply_configuration(
|
|
||||||
task, raid_config, create_root_volume=create_root_volume,
|
|
||||||
create_nonroot_volumes=create_nonroot_volumes)
|
|
||||||
|
|
||||||
@METRICS.timer('IbmcRAID.create_configuration')
|
|
||||||
@base.clean_step(priority=0, abortable=False, argsinfo={
|
|
||||||
'create_root_volume': {
|
|
||||||
'description': ('This specifies whether to create the root '
|
|
||||||
'volume. Defaults to `True`.'),
|
|
||||||
'required': False
|
|
||||||
},
|
|
||||||
'create_nonroot_volumes': {
|
|
||||||
'description': ('This specifies whether to create the non-root '
|
|
||||||
'volumes. Defaults to `True`.'),
|
|
||||||
'required': False
|
|
||||||
},
|
|
||||||
"delete_existing": {
|
|
||||||
"description": ("Setting this to 'True' indicates to delete "
|
|
||||||
"existing RAID configuration prior to creating "
|
|
||||||
"the new configuration. "
|
|
||||||
"Default value is 'False'."),
|
|
||||||
"required": False,
|
|
||||||
}
|
|
||||||
})
|
|
||||||
def create_configuration(self, task, create_root_volume=True,
|
|
||||||
create_nonroot_volumes=True,
|
|
||||||
delete_existing=False):
|
|
||||||
"""Create a RAID configuration.
|
|
||||||
|
|
||||||
This method creates a RAID configuration on the given node.
|
|
||||||
|
|
||||||
:param task: a TaskManager instance.
|
|
||||||
:param create_root_volume: If True, a root volume is created
|
|
||||||
during RAID configuration. Otherwise, no root volume is
|
|
||||||
created. Default is True.
|
|
||||||
:param create_nonroot_volumes: If True, non-root volumes are
|
|
||||||
created. If False, no non-root volumes are created. Default
|
|
||||||
is True.
|
|
||||||
:param delete_existing: Setting this to True indicates to delete RAID
|
|
||||||
configuration prior to creating the new configuration. Default is
|
|
||||||
False.
|
|
||||||
:raises: MissingParameterValue, if node.target_raid_config is missing
|
|
||||||
or empty after skipping root volume and/or non-root volumes.
|
|
||||||
:raises: IBMCError, on failure to execute step.
|
|
||||||
"""
|
|
||||||
node = task.node
|
|
||||||
raid_config = raid.filter_target_raid_config(
|
|
||||||
node, create_root_volume=create_root_volume,
|
|
||||||
create_nonroot_volumes=create_nonroot_volumes)
|
|
||||||
LOG.info(_("Invoke RAID create_configuration step for node %s(uuid). "
|
|
||||||
"Current provision state is: %(status)s. "
|
|
||||||
"Target RAID configuration is: %(config)s."),
|
|
||||||
{'uuid': node.uuid, 'status': node.provision_state,
|
|
||||||
'target': raid_config})
|
|
||||||
|
|
||||||
# cache current raid config to node's driver_internal_info
|
|
||||||
node.driver_internal_info['raid_config'] = raid_config
|
|
||||||
node.save()
|
|
||||||
|
|
||||||
# delete exist volumes if necessary
|
|
||||||
if delete_existing:
|
|
||||||
self._delete_raid_configuration(task)
|
|
||||||
|
|
||||||
# create raid configuration
|
|
||||||
logical_disks = raid_config.get('logical_disks', [])
|
|
||||||
self._create_raid_configuration(task, logical_disks)
|
|
||||||
LOG.info(_("Succeed to create raid configuration on node %s."),
|
|
||||||
task.node.uuid)
|
|
||||||
|
|
||||||
@METRICS.timer('IbmcRAID.delete_configuration')
|
|
||||||
@base.clean_step(priority=0, abortable=False)
|
|
||||||
@base.deploy_step(priority=0)
|
|
||||||
def delete_configuration(self, task):
|
|
||||||
"""Delete the RAID configuration.
|
|
||||||
|
|
||||||
:param task: a TaskManager instance containing the node to act on.
|
|
||||||
:returns: states.CLEANWAIT if cleaning operation in progress
|
|
||||||
asynchronously or states.DEPLOYWAIT if deploy operation in
|
|
||||||
progress synchronously or None if it is completed.
|
|
||||||
:raises: IBMCError, on failure to execute step.
|
|
||||||
"""
|
|
||||||
node = task.node
|
|
||||||
LOG.info("Invoke RAID delete_configuration step for node %s(uuid). "
|
|
||||||
"Current provision state is: %(status)s. ",
|
|
||||||
{'uuid': node.uuid, 'status': node.provision_state})
|
|
||||||
self._delete_raid_configuration(task)
|
|
||||||
LOG.info(_("Succeed to delete raid configuration on node %s."),
|
|
||||||
task.node.uuid)
|
|
@ -1,172 +0,0 @@
|
|||||||
#
|
|
||||||
# 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.
|
|
||||||
"""
|
|
||||||
iBMC Driver common utils
|
|
||||||
"""
|
|
||||||
|
|
||||||
import functools
|
|
||||||
import os
|
|
||||||
|
|
||||||
from oslo_log import log
|
|
||||||
from oslo_utils import importutils
|
|
||||||
from oslo_utils import netutils
|
|
||||||
from oslo_utils import strutils
|
|
||||||
import tenacity
|
|
||||||
|
|
||||||
from ironic.common import exception
|
|
||||||
from ironic.common.i18n import _
|
|
||||||
from ironic.conductor import task_manager
|
|
||||||
from ironic.conf import CONF
|
|
||||||
|
|
||||||
ibmc_client = importutils.try_import('ibmcclient')
|
|
||||||
ibmc_error = importutils.try_import('ibmc_client.exceptions')
|
|
||||||
|
|
||||||
LOG = log.getLogger(__name__)
|
|
||||||
|
|
||||||
REQUIRED_PROPERTIES = {
|
|
||||||
'ibmc_address': _('The URL address to the iBMC controller. It must '
|
|
||||||
'include the authority portion of the URL. '
|
|
||||||
'If the scheme is missing, https is assumed. '
|
|
||||||
'For example: https://mgmt.vendor.com. Required.'),
|
|
||||||
'ibmc_username': _('User account with admin/server-profile access '
|
|
||||||
'privilege. Required.'),
|
|
||||||
'ibmc_password': _('User account password. Required.'),
|
|
||||||
}
|
|
||||||
|
|
||||||
OPTIONAL_PROPERTIES = {
|
|
||||||
'ibmc_verify_ca': _('Either a Boolean value, a path to a CA_BUNDLE '
|
|
||||||
'file or directory with certificates of trusted '
|
|
||||||
'CAs. If set to True the driver will verify the '
|
|
||||||
'host certificates; if False the driver will '
|
|
||||||
'ignore verifying the SSL certificate. If it\'s '
|
|
||||||
'a path the driver will use the specified '
|
|
||||||
'certificate or one of the certificates in the '
|
|
||||||
'directory. Defaults to True. Optional.'),
|
|
||||||
}
|
|
||||||
|
|
||||||
COMMON_PROPERTIES = REQUIRED_PROPERTIES.copy()
|
|
||||||
COMMON_PROPERTIES.update(OPTIONAL_PROPERTIES)
|
|
||||||
|
|
||||||
|
|
||||||
def parse_driver_info(node):
|
|
||||||
"""Parse the information required for Ironic to connect to iBMC.
|
|
||||||
|
|
||||||
:param node: an Ironic node object
|
|
||||||
:returns: dictionary of parameters
|
|
||||||
:raises: InvalidParameterValue on malformed parameter(s)
|
|
||||||
:raises: MissingParameterValue on missing parameter(s)
|
|
||||||
"""
|
|
||||||
driver_info = node.driver_info or {}
|
|
||||||
missing_info = [key for key in REQUIRED_PROPERTIES
|
|
||||||
if not driver_info.get(key)]
|
|
||||||
if missing_info:
|
|
||||||
raise exception.MissingParameterValue(_(
|
|
||||||
'Missing the following iBMC properties in node '
|
|
||||||
'%(node)s driver_info: %(info)s') % {'node': node.uuid,
|
|
||||||
'info': missing_info})
|
|
||||||
|
|
||||||
# Validate the iBMC address
|
|
||||||
address = driver_info['ibmc_address']
|
|
||||||
if '://' not in address:
|
|
||||||
address = 'https://%s' % address
|
|
||||||
|
|
||||||
parsed = netutils.urlsplit(address)
|
|
||||||
if not parsed.netloc:
|
|
||||||
raise exception.InvalidParameterValue(
|
|
||||||
_('Invalid iBMC address %(address)s set in '
|
|
||||||
'driver_info/ibmc_address on node %(node)s') %
|
|
||||||
{'address': address, 'node': node.uuid})
|
|
||||||
|
|
||||||
# Check if verify_ca is a Boolean or a file/directory in the file-system
|
|
||||||
verify_ca = driver_info.get('ibmc_verify_ca', True)
|
|
||||||
if isinstance(verify_ca, str):
|
|
||||||
if not os.path.exists(verify_ca):
|
|
||||||
try:
|
|
||||||
verify_ca = strutils.bool_from_string(verify_ca, strict=True)
|
|
||||||
except ValueError:
|
|
||||||
raise exception.InvalidParameterValue(
|
|
||||||
_('Invalid value type set in driver_info/'
|
|
||||||
'ibmc_verify_ca on node %(node)s. '
|
|
||||||
'The value should be a Boolean or the path '
|
|
||||||
'to a file/directory, not "%(value)s"'
|
|
||||||
) % {'value': verify_ca, 'node': node.uuid})
|
|
||||||
elif not isinstance(verify_ca, bool):
|
|
||||||
raise exception.InvalidParameterValue(
|
|
||||||
_('Invalid value type set in driver_info/ibmc_verify_ca '
|
|
||||||
'on node %(node)s. The value should be a Boolean or the path '
|
|
||||||
'to a file/directory, not "%(value)s"') % {'value': verify_ca,
|
|
||||||
'node': node.uuid})
|
|
||||||
return {'address': address,
|
|
||||||
'username': driver_info.get('ibmc_username'),
|
|
||||||
'password': driver_info.get('ibmc_password'),
|
|
||||||
'verify_ca': verify_ca}
|
|
||||||
|
|
||||||
|
|
||||||
def revert_dictionary(d):
|
|
||||||
return {v: k for k, v in d.items()}
|
|
||||||
|
|
||||||
|
|
||||||
def handle_ibmc_exception(action):
|
|
||||||
"""Decorator to handle iBMC client exception.
|
|
||||||
|
|
||||||
Decorated functions must take a :class:`TaskManager` as the first
|
|
||||||
parameter.
|
|
||||||
"""
|
|
||||||
|
|
||||||
def decorator(f):
|
|
||||||
|
|
||||||
def should_retry(e):
|
|
||||||
connect_error = isinstance(e, exception.IBMCConnectionError)
|
|
||||||
if connect_error:
|
|
||||||
LOG.info(_('Failed to connect to iBMC, will retry now. '
|
|
||||||
'Max retry times is %(retry_times)d.'),
|
|
||||||
{'retry_times': CONF.ibmc.connection_attempts})
|
|
||||||
return connect_error
|
|
||||||
|
|
||||||
@tenacity.retry(
|
|
||||||
retry=tenacity.retry_if_exception(should_retry),
|
|
||||||
stop=tenacity.stop_after_attempt(CONF.ibmc.connection_attempts),
|
|
||||||
wait=tenacity.wait_fixed(CONF.ibmc.connection_retry_interval),
|
|
||||||
reraise=True)
|
|
||||||
@functools.wraps(f)
|
|
||||||
def wrapper(*args, **kwargs):
|
|
||||||
# NOTE(dtantsur): this code could be written simpler, but then unit
|
|
||||||
# testing decorated functions is pretty hard, as we usually pass a
|
|
||||||
# Mock object instead of TaskManager there.
|
|
||||||
if len(args) > 1:
|
|
||||||
is_task_mgr = isinstance(args[1], task_manager.TaskManager)
|
|
||||||
task = args[1] if is_task_mgr else args[0]
|
|
||||||
else:
|
|
||||||
task = args[0]
|
|
||||||
|
|
||||||
node = task.node
|
|
||||||
|
|
||||||
try:
|
|
||||||
return f(*args, **kwargs)
|
|
||||||
except ibmc_error.IBMCConnectionError as e:
|
|
||||||
error = (_('Failed to connect to iBMC for node %(node)s, '
|
|
||||||
'Error: %(error)s')
|
|
||||||
% {'node': node.uuid, 'error': e})
|
|
||||||
LOG.error(error)
|
|
||||||
raise exception.IBMCConnectionError(node=node.uuid,
|
|
||||||
error=error)
|
|
||||||
except ibmc_error.IBMCClientError as e:
|
|
||||||
error = (_('Failed to %(action)s for node %(node)s, '
|
|
||||||
'Error %(error)s')
|
|
||||||
% {'node': node.uuid, 'action': action, 'error': e})
|
|
||||||
LOG.error(error)
|
|
||||||
raise exception.IBMCError(node=node.uuid, error=error)
|
|
||||||
|
|
||||||
return wrapper
|
|
||||||
|
|
||||||
return decorator
|
|
@ -1,112 +0,0 @@
|
|||||||
#
|
|
||||||
# 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.
|
|
||||||
"""
|
|
||||||
iBMC Vendor Interface
|
|
||||||
"""
|
|
||||||
|
|
||||||
from oslo_log import log
|
|
||||||
from oslo_utils import importutils
|
|
||||||
|
|
||||||
from ironic.common import exception
|
|
||||||
from ironic.common.i18n import _
|
|
||||||
from ironic.drivers import base
|
|
||||||
from ironic.drivers.modules.ibmc import utils
|
|
||||||
|
|
||||||
ibmc_client = importutils.try_import('ibmc_client')
|
|
||||||
|
|
||||||
LOG = log.getLogger(__name__)
|
|
||||||
|
|
||||||
|
|
||||||
class IBMCVendor(base.VendorInterface):
|
|
||||||
|
|
||||||
# NOTE(TheJulia): Deprecating November 2023 in favor of Redfish
|
|
||||||
# and due to a lack of active driver maintenance.
|
|
||||||
supported = False
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
"""Initialize the iBMC vendor interface.
|
|
||||||
|
|
||||||
:raises: DriverLoadError if the driver can't be loaded due to
|
|
||||||
missing dependencies
|
|
||||||
"""
|
|
||||||
super(IBMCVendor, self).__init__()
|
|
||||||
if not ibmc_client:
|
|
||||||
raise exception.DriverLoadError(
|
|
||||||
driver='ibmc',
|
|
||||||
reason=_('Unable to import the python-ibmcclient library'))
|
|
||||||
|
|
||||||
def validate(self, task, method=None, **kwargs):
|
|
||||||
"""Validate vendor-specific actions.
|
|
||||||
|
|
||||||
If invalid, raises an exception; otherwise returns None.
|
|
||||||
|
|
||||||
:param task: A task from TaskManager.
|
|
||||||
:param method: Method to be validated
|
|
||||||
:param kwargs: Info for action.
|
|
||||||
:raises: UnsupportedDriverExtension if 'method' can not be mapped to
|
|
||||||
the supported interfaces.
|
|
||||||
:raises: InvalidParameterValue if kwargs does not contain 'method'.
|
|
||||||
:raises: MissingParameterValue
|
|
||||||
"""
|
|
||||||
utils.parse_driver_info(task.node)
|
|
||||||
|
|
||||||
def get_properties(self):
|
|
||||||
"""Return the properties of the interface.
|
|
||||||
|
|
||||||
:returns: dictionary of <property name>:<property description> entries.
|
|
||||||
"""
|
|
||||||
return utils.COMMON_PROPERTIES.copy()
|
|
||||||
|
|
||||||
@base.passthru(['GET'], async_call=False,
|
|
||||||
description=_('Returns a dictionary, '
|
|
||||||
'containing node boot up sequence, '
|
|
||||||
'in ascending order'))
|
|
||||||
@utils.handle_ibmc_exception('get iBMC boot up sequence')
|
|
||||||
def boot_up_seq(self, task, **kwargs):
|
|
||||||
"""List boot type order of the node.
|
|
||||||
|
|
||||||
:param task: A TaskManager instance containing the node to act on.
|
|
||||||
:param kwargs: Not used.
|
|
||||||
:raises: InvalidParameterValue if kwargs does not contain 'method'.
|
|
||||||
:raises: MissingParameterValue
|
|
||||||
:raises: IBMCConnectionError when it fails to connect to iBMC
|
|
||||||
:raises: IBMCError when iBMC responses an error information
|
|
||||||
:returns: A dictionary, containing node boot up sequence,
|
|
||||||
in ascending order.
|
|
||||||
"""
|
|
||||||
driver_info = utils.parse_driver_info(task.node)
|
|
||||||
with ibmc_client.connect(**driver_info) as conn:
|
|
||||||
system = conn.system.get()
|
|
||||||
boot_sequence = system.boot_sequence
|
|
||||||
return {'boot_up_sequence': boot_sequence}
|
|
||||||
|
|
||||||
@base.passthru(['GET'], async_call=False,
|
|
||||||
description=_('Returns a list of dictionary, every '
|
|
||||||
'dictionary represents a RAID controller '
|
|
||||||
'summary info'))
|
|
||||||
@utils.handle_ibmc_exception('get iBMC RAID controller summary')
|
|
||||||
def get_raid_controller_list(self, task, **kwargs):
|
|
||||||
"""List RAID controllers summary info of the node.
|
|
||||||
|
|
||||||
:param task: A TaskManager instance containing the node to act on.
|
|
||||||
:param kwargs: Not used.
|
|
||||||
:raises: IBMCConnectionError when it fails to connect to iBMC
|
|
||||||
:raises: IBMCError when iBMC responses an error information
|
|
||||||
:returns: A list of dictionaries, every dictionary represents a RAID
|
|
||||||
controller summary of node.
|
|
||||||
"""
|
|
||||||
driver_info = utils.parse_driver_info(task.node)
|
|
||||||
with ibmc_client.connect(**driver_info) as conn:
|
|
||||||
controllers = conn.system.storage.list()
|
|
||||||
summaries = [ctrl.summary() for ctrl in controllers]
|
|
||||||
return summaries
|
|
@ -679,15 +679,6 @@ def create_test_deploy_template(**kw):
|
|||||||
return dbapi.create_deploy_template(template)
|
return dbapi.create_deploy_template(template)
|
||||||
|
|
||||||
|
|
||||||
def get_test_ibmc_info():
|
|
||||||
return {
|
|
||||||
"ibmc_address": "https://example.com",
|
|
||||||
"ibmc_username": "username",
|
|
||||||
"ibmc_password": "password",
|
|
||||||
"verify_ca": False,
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
def get_test_history(**kw):
|
def get_test_history(**kw):
|
||||||
return {
|
return {
|
||||||
'id': kw.get('id', 345),
|
'id': kw.get('id', 345),
|
||||||
|
@ -1,43 +0,0 @@
|
|||||||
#
|
|
||||||
# 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 base class for iBMC Driver."""
|
|
||||||
|
|
||||||
from unittest import mock
|
|
||||||
|
|
||||||
from ironic.drivers.modules.ibmc import 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
|
|
||||||
|
|
||||||
|
|
||||||
class IBMCTestCase(db_base.DbTestCase):
|
|
||||||
|
|
||||||
def setUp(self):
|
|
||||||
super(IBMCTestCase, self).setUp()
|
|
||||||
self.driver_info = db_utils.get_test_ibmc_info()
|
|
||||||
self.config(enabled_hardware_types=['ibmc'],
|
|
||||||
enabled_power_interfaces=['ibmc'],
|
|
||||||
enabled_management_interfaces=['ibmc'],
|
|
||||||
enabled_vendor_interfaces=['ibmc'],
|
|
||||||
enabled_raid_interfaces=['ibmc'])
|
|
||||||
self.node = obj_utils.create_test_node(
|
|
||||||
self.context, driver='ibmc', driver_info=self.driver_info)
|
|
||||||
self.ibmc = utils.parse_driver_info(self.node)
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def mock_ibmc_conn(ibmc_client_connect):
|
|
||||||
conn = mock.Mock(system=mock.PropertyMock())
|
|
||||||
conn.__enter__ = mock.Mock(return_value=conn)
|
|
||||||
conn.__exit__ = mock.Mock(return_value=None)
|
|
||||||
ibmc_client_connect.return_value = conn
|
|
||||||
return conn
|
|
@ -1,294 +0,0 @@
|
|||||||
#
|
|
||||||
# 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 iBMC Management interface."""
|
|
||||||
|
|
||||||
import itertools
|
|
||||||
from unittest import mock
|
|
||||||
|
|
||||||
from oslo_utils import importutils
|
|
||||||
|
|
||||||
from ironic.common import boot_devices
|
|
||||||
from ironic.common import boot_modes
|
|
||||||
from ironic.common import exception
|
|
||||||
from ironic.conductor import task_manager
|
|
||||||
from ironic.drivers.modules.ibmc import mappings
|
|
||||||
from ironic.drivers.modules.ibmc import utils
|
|
||||||
from ironic.tests.unit.drivers.modules.ibmc import base
|
|
||||||
|
|
||||||
constants = importutils.try_import('ibmc_client.constants')
|
|
||||||
ibmc_client = importutils.try_import('ibmc_client')
|
|
||||||
ibmc_error = importutils.try_import('ibmc_client.exceptions')
|
|
||||||
|
|
||||||
|
|
||||||
class IBMCManagementTestCase(base.IBMCTestCase):
|
|
||||||
|
|
||||||
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 utils.COMMON_PROPERTIES:
|
|
||||||
self.assertIn(prop, properties)
|
|
||||||
|
|
||||||
@mock.patch.object(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.management.validate(task)
|
|
||||||
mock_parse_driver_info.assert_called_once_with(task.node)
|
|
||||||
|
|
||||||
def test_get_supported_boot_devices(self):
|
|
||||||
if not mock._is_instance_mock(ibmc_client):
|
|
||||||
mock.patch.object(ibmc_client, 'connect', autospec=True).start()
|
|
||||||
connect_ibmc = ibmc_client.connect
|
|
||||||
conn = self.mock_ibmc_conn(connect_ibmc)
|
|
||||||
# mock return value
|
|
||||||
_supported_boot_devices = list(mappings.GET_BOOT_DEVICE_MAP)
|
|
||||||
conn.system.get.return_value = mock.Mock(
|
|
||||||
boot_source_override=mock.Mock(
|
|
||||||
supported_boot_devices=_supported_boot_devices
|
|
||||||
)
|
|
||||||
)
|
|
||||||
with task_manager.acquire(self.context, self.node.uuid,
|
|
||||||
shared=True) as task:
|
|
||||||
supported_boot_devices = (
|
|
||||||
task.driver.management.get_supported_boot_devices(task))
|
|
||||||
connect_ibmc.assert_called_with(**self.ibmc)
|
|
||||||
expect = sorted(list(mappings.GET_BOOT_DEVICE_MAP.values()))
|
|
||||||
self.assertEqual(expect, sorted(supported_boot_devices))
|
|
||||||
|
|
||||||
def test_set_boot_device(self):
|
|
||||||
if not mock._is_instance_mock(ibmc_client):
|
|
||||||
mock.patch.object(ibmc_client, 'connect', autospec=True).start()
|
|
||||||
connect_ibmc = ibmc_client.connect
|
|
||||||
conn = self.mock_ibmc_conn(connect_ibmc)
|
|
||||||
# mock return value
|
|
||||||
conn.system.set_boot_source.return_value = None
|
|
||||||
with task_manager.acquire(self.context, self.node.uuid,
|
|
||||||
shared=False) as task:
|
|
||||||
device_mapping = [
|
|
||||||
(boot_devices.PXE, constants.BOOT_SOURCE_TARGET_PXE),
|
|
||||||
(boot_devices.DISK, constants.BOOT_SOURCE_TARGET_HDD),
|
|
||||||
(boot_devices.CDROM, constants.BOOT_SOURCE_TARGET_CD),
|
|
||||||
(boot_devices.BIOS,
|
|
||||||
constants.BOOT_SOURCE_TARGET_BIOS_SETUP),
|
|
||||||
('floppy', constants.BOOT_SOURCE_TARGET_FLOPPY),
|
|
||||||
]
|
|
||||||
|
|
||||||
persistent_mapping = [
|
|
||||||
(True, constants.BOOT_SOURCE_ENABLED_CONTINUOUS),
|
|
||||||
(False, constants.BOOT_SOURCE_ENABLED_ONCE)
|
|
||||||
]
|
|
||||||
|
|
||||||
data_source = list(itertools.product(device_mapping,
|
|
||||||
persistent_mapping))
|
|
||||||
for (device, persistent) in data_source:
|
|
||||||
task.driver.management.set_boot_device(
|
|
||||||
task, device[0], persistent=persistent[0])
|
|
||||||
connect_ibmc.assert_called_with(**self.ibmc)
|
|
||||||
conn.system.set_boot_source.assert_called_once_with(
|
|
||||||
device[1],
|
|
||||||
enabled=persistent[1])
|
|
||||||
# Reset mocks
|
|
||||||
connect_ibmc.reset_mock()
|
|
||||||
conn.system.set_boot_source.reset_mock()
|
|
||||||
|
|
||||||
def test_set_boot_device_fail(self):
|
|
||||||
if not mock._is_instance_mock(ibmc_client):
|
|
||||||
mock.patch.object(ibmc_client, 'connect', autospec=True).start()
|
|
||||||
connect_ibmc = ibmc_client.connect
|
|
||||||
conn = self.mock_ibmc_conn(connect_ibmc)
|
|
||||||
# mock return value
|
|
||||||
conn.system.set_boot_source.side_effect = (
|
|
||||||
ibmc_error.IBMCClientError
|
|
||||||
)
|
|
||||||
|
|
||||||
with task_manager.acquire(self.context, self.node.uuid,
|
|
||||||
shared=False) as task:
|
|
||||||
self.assertRaisesRegex(
|
|
||||||
exception.IBMCError, 'set iBMC boot device',
|
|
||||||
task.driver.management.set_boot_device, task,
|
|
||||||
boot_devices.PXE)
|
|
||||||
connect_ibmc.assert_called_with(**self.ibmc)
|
|
||||||
conn.system.set_boot_source.assert_called_once_with(
|
|
||||||
constants.BOOT_SOURCE_TARGET_PXE,
|
|
||||||
enabled=constants.BOOT_SOURCE_ENABLED_ONCE)
|
|
||||||
|
|
||||||
def test_get_boot_device(self):
|
|
||||||
if not mock._is_instance_mock(ibmc_client):
|
|
||||||
mock.patch.object(ibmc_client, 'connect', autospec=True).start()
|
|
||||||
connect_ibmc = ibmc_client.connect
|
|
||||||
conn = self.mock_ibmc_conn(connect_ibmc)
|
|
||||||
# mock return value
|
|
||||||
conn.system.get.return_value = mock.Mock(
|
|
||||||
boot_source_override=mock.Mock(
|
|
||||||
target=constants.BOOT_SOURCE_TARGET_PXE,
|
|
||||||
enabled=constants.BOOT_SOURCE_ENABLED_CONTINUOUS
|
|
||||||
)
|
|
||||||
)
|
|
||||||
with task_manager.acquire(self.context, self.node.uuid,
|
|
||||||
shared=True) as task:
|
|
||||||
result_boot_device = task.driver.management.get_boot_device(task)
|
|
||||||
conn.system.get.assert_called_once()
|
|
||||||
connect_ibmc.assert_called_once_with(**self.ibmc)
|
|
||||||
expected = {'boot_device': boot_devices.PXE,
|
|
||||||
'persistent': True}
|
|
||||||
self.assertEqual(expected, result_boot_device)
|
|
||||||
|
|
||||||
def test_get_supported_boot_modes(self):
|
|
||||||
with task_manager.acquire(self.context, self.node.uuid,
|
|
||||||
shared=True) as task:
|
|
||||||
supported_boot_modes = (
|
|
||||||
task.driver.management.get_supported_boot_modes(task))
|
|
||||||
self.assertEqual(list(mappings.SET_BOOT_MODE_MAP),
|
|
||||||
supported_boot_modes)
|
|
||||||
|
|
||||||
def test_set_boot_mode(self):
|
|
||||||
if not mock._is_instance_mock(ibmc_client):
|
|
||||||
mock.patch.object(ibmc_client, 'connect', autospec=True).start()
|
|
||||||
connect_ibmc = ibmc_client.connect
|
|
||||||
conn = self.mock_ibmc_conn(connect_ibmc)
|
|
||||||
# mock system boot source override return value
|
|
||||||
conn.system.get.return_value = mock.Mock(
|
|
||||||
boot_source_override=mock.Mock(
|
|
||||||
target=constants.BOOT_SOURCE_TARGET_PXE,
|
|
||||||
enabled=constants.BOOT_SOURCE_ENABLED_CONTINUOUS
|
|
||||||
)
|
|
||||||
)
|
|
||||||
conn.system.set_boot_source.return_value = None
|
|
||||||
|
|
||||||
with task_manager.acquire(self.context, self.node.uuid,
|
|
||||||
shared=False) as task:
|
|
||||||
expected_values = [
|
|
||||||
(boot_modes.LEGACY_BIOS, constants.BOOT_SOURCE_MODE_BIOS),
|
|
||||||
(boot_modes.UEFI, constants.BOOT_SOURCE_MODE_UEFI)
|
|
||||||
]
|
|
||||||
|
|
||||||
for ironic_boot_mode, ibmc_boot_mode in expected_values:
|
|
||||||
task.driver.management.set_boot_mode(task,
|
|
||||||
mode=ironic_boot_mode)
|
|
||||||
|
|
||||||
conn.system.get.assert_called_once()
|
|
||||||
connect_ibmc.assert_called_with(**self.ibmc)
|
|
||||||
|
|
||||||
conn.system.set_boot_source.assert_called_once_with(
|
|
||||||
constants.BOOT_SOURCE_TARGET_PXE,
|
|
||||||
enabled=constants.BOOT_SOURCE_ENABLED_CONTINUOUS,
|
|
||||||
mode=ibmc_boot_mode)
|
|
||||||
|
|
||||||
# Reset
|
|
||||||
connect_ibmc.reset_mock()
|
|
||||||
conn.system.set_boot_source.reset_mock()
|
|
||||||
conn.system.get.reset_mock()
|
|
||||||
|
|
||||||
def test_set_boot_mode_fail(self):
|
|
||||||
if not mock._is_instance_mock(ibmc_client):
|
|
||||||
mock.patch.object(ibmc_client, 'connect', autospec=True).start()
|
|
||||||
connect_ibmc = ibmc_client.connect
|
|
||||||
conn = self.mock_ibmc_conn(connect_ibmc)
|
|
||||||
# mock system boot source override return value
|
|
||||||
conn.system.get.return_value = mock.Mock(
|
|
||||||
boot_source_override=mock.Mock(
|
|
||||||
target=constants.BOOT_SOURCE_TARGET_PXE,
|
|
||||||
enabled=constants.BOOT_SOURCE_ENABLED_CONTINUOUS
|
|
||||||
)
|
|
||||||
)
|
|
||||||
conn.system.set_boot_source.side_effect = (
|
|
||||||
ibmc_error.IBMCClientError
|
|
||||||
)
|
|
||||||
|
|
||||||
with task_manager.acquire(self.context, self.node.uuid,
|
|
||||||
shared=False) as task:
|
|
||||||
expected_values = [
|
|
||||||
(boot_modes.LEGACY_BIOS, constants.BOOT_SOURCE_MODE_BIOS),
|
|
||||||
(boot_modes.UEFI, constants.BOOT_SOURCE_MODE_UEFI)
|
|
||||||
]
|
|
||||||
|
|
||||||
for ironic_boot_mode, ibmc_boot_mode in expected_values:
|
|
||||||
self.assertRaisesRegex(
|
|
||||||
exception.IBMCError, 'set iBMC boot mode',
|
|
||||||
task.driver.management.set_boot_mode, task,
|
|
||||||
ironic_boot_mode)
|
|
||||||
|
|
||||||
conn.system.set_boot_source.assert_called_once_with(
|
|
||||||
constants.BOOT_SOURCE_TARGET_PXE,
|
|
||||||
enabled=constants.BOOT_SOURCE_ENABLED_CONTINUOUS,
|
|
||||||
mode=ibmc_boot_mode)
|
|
||||||
|
|
||||||
conn.system.get.assert_called_once()
|
|
||||||
connect_ibmc.assert_called_with(**self.ibmc)
|
|
||||||
|
|
||||||
# Reset
|
|
||||||
connect_ibmc.reset_mock()
|
|
||||||
conn.system.set_boot_source.reset_mock()
|
|
||||||
conn.system.get.reset_mock()
|
|
||||||
|
|
||||||
def test_get_boot_mode(self):
|
|
||||||
if not mock._is_instance_mock(ibmc_client):
|
|
||||||
mock.patch.object(ibmc_client, 'connect', autospec=True).start()
|
|
||||||
connect_ibmc = ibmc_client.connect
|
|
||||||
conn = self.mock_ibmc_conn(connect_ibmc)
|
|
||||||
# mock system boot source override return value
|
|
||||||
conn.system.get.return_value = mock.Mock(
|
|
||||||
boot_source_override=mock.Mock(
|
|
||||||
target=constants.BOOT_SOURCE_TARGET_PXE,
|
|
||||||
enabled=constants.BOOT_SOURCE_ENABLED_CONTINUOUS,
|
|
||||||
mode=constants.BOOT_SOURCE_MODE_BIOS,
|
|
||||||
)
|
|
||||||
)
|
|
||||||
with task_manager.acquire(self.context, self.node.uuid,
|
|
||||||
shared=True) as task:
|
|
||||||
response = task.driver.management.get_boot_mode(task)
|
|
||||||
|
|
||||||
conn.system.get.assert_called_once()
|
|
||||||
connect_ibmc.assert_called_with(**self.ibmc)
|
|
||||||
|
|
||||||
expected = boot_modes.LEGACY_BIOS
|
|
||||||
self.assertEqual(expected, response)
|
|
||||||
|
|
||||||
def test_get_sensors_data(self):
|
|
||||||
with task_manager.acquire(self.context, self.node.uuid,
|
|
||||||
shared=True) as task:
|
|
||||||
self.assertRaises(NotImplementedError,
|
|
||||||
task.driver.management.get_sensors_data, task)
|
|
||||||
|
|
||||||
def test_inject_nmi(self):
|
|
||||||
if not mock._is_instance_mock(ibmc_client):
|
|
||||||
mock.patch.object(ibmc_client, 'connect', autospec=True).start()
|
|
||||||
connect_ibmc = ibmc_client.connect
|
|
||||||
conn = self.mock_ibmc_conn(connect_ibmc)
|
|
||||||
# mock system boot source override return value
|
|
||||||
conn.system.reset.return_value = None
|
|
||||||
with task_manager.acquire(self.context, self.node.uuid,
|
|
||||||
shared=False) as task:
|
|
||||||
task.driver.management.inject_nmi(task)
|
|
||||||
|
|
||||||
connect_ibmc.assert_called_with(**self.ibmc)
|
|
||||||
conn.system.reset.assert_called_once_with(constants.RESET_NMI)
|
|
||||||
|
|
||||||
def test_inject_nmi_fail(self):
|
|
||||||
if not mock._is_instance_mock(ibmc_client):
|
|
||||||
mock.patch.object(ibmc_client, 'connect', autospec=True).start()
|
|
||||||
connect_ibmc = ibmc_client.connect
|
|
||||||
conn = self.mock_ibmc_conn(connect_ibmc)
|
|
||||||
# mock system boot source override return value
|
|
||||||
conn.system.reset.side_effect = (
|
|
||||||
ibmc_error.IBMCClientError
|
|
||||||
)
|
|
||||||
with task_manager.acquire(self.context, self.node.uuid,
|
|
||||||
shared=False) as task:
|
|
||||||
self.assertRaisesRegex(
|
|
||||||
exception.IBMCError, 'inject iBMC NMI',
|
|
||||||
task.driver.management.inject_nmi, task)
|
|
||||||
|
|
||||||
connect_ibmc.assert_called_with(**self.ibmc)
|
|
||||||
conn.system.reset.assert_called_once_with(constants.RESET_NMI)
|
|
@ -1,288 +0,0 @@
|
|||||||
#
|
|
||||||
# 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 iBMC Power interface."""
|
|
||||||
|
|
||||||
from unittest 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.ibmc import mappings
|
|
||||||
from ironic.drivers.modules.ibmc import utils
|
|
||||||
from ironic.tests.unit.drivers.modules.ibmc import base
|
|
||||||
|
|
||||||
constants = importutils.try_import('ibmc_client.constants')
|
|
||||||
ibmc_client = importutils.try_import('ibmc_client')
|
|
||||||
ibmc_error = importutils.try_import('ibmc_client.exceptions')
|
|
||||||
|
|
||||||
|
|
||||||
@mock.patch('oslo_utils.eventletutils.EventletEvent.wait',
|
|
||||||
lambda *args, **kwargs: None)
|
|
||||||
class IBMCPowerTestCase(base.IBMCTestCase):
|
|
||||||
|
|
||||||
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 utils.COMMON_PROPERTIES:
|
|
||||||
self.assertIn(prop, properties)
|
|
||||||
|
|
||||||
@mock.patch.object(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(ibmc_client, 'connect', spec=object)
|
|
||||||
def test_get_power_state(self, connect_ibmc):
|
|
||||||
conn = self.mock_ibmc_conn(connect_ibmc)
|
|
||||||
with task_manager.acquire(self.context, self.node.uuid,
|
|
||||||
shared=True) as task:
|
|
||||||
expected_values = mappings.GET_POWER_STATE_MAP
|
|
||||||
for current, expected in expected_values.items():
|
|
||||||
# Mock
|
|
||||||
conn.system.get.return_value = mock.Mock(
|
|
||||||
power_state=current
|
|
||||||
)
|
|
||||||
|
|
||||||
# Asserts
|
|
||||||
self.assertEqual(expected,
|
|
||||||
task.driver.power.get_power_state(task))
|
|
||||||
|
|
||||||
conn.system.get.assert_called_once()
|
|
||||||
connect_ibmc.assert_called_once_with(**self.ibmc)
|
|
||||||
|
|
||||||
# Reset Mock
|
|
||||||
conn.system.get.reset_mock()
|
|
||||||
connect_ibmc.reset_mock()
|
|
||||||
|
|
||||||
@mock.patch.object(ibmc_client, 'connect', spec=object)
|
|
||||||
def test_set_power_state(self, connect_ibmc):
|
|
||||||
conn = self.mock_ibmc_conn(connect_ibmc)
|
|
||||||
with task_manager.acquire(self.context, self.node.uuid,
|
|
||||||
shared=False) as task:
|
|
||||||
state_mapping = mappings.SET_POWER_STATE_MAP
|
|
||||||
for (expect_state, reset_type) in state_mapping.items():
|
|
||||||
if expect_state in (states.POWER_OFF, states.SOFT_POWER_OFF):
|
|
||||||
final = constants.SYSTEM_POWER_STATE_OFF
|
|
||||||
transient = constants.SYSTEM_POWER_STATE_ON
|
|
||||||
else:
|
|
||||||
final = constants.SYSTEM_POWER_STATE_ON
|
|
||||||
transient = constants.SYSTEM_POWER_STATE_OFF
|
|
||||||
|
|
||||||
# Mocks
|
|
||||||
mock_system_get_results = (
|
|
||||||
[mock.Mock(power_state=transient)] * 3
|
|
||||||
+ [mock.Mock(power_state=final)])
|
|
||||||
conn.system.get.side_effect = mock_system_get_results
|
|
||||||
|
|
||||||
task.driver.power.set_power_state(task, expect_state)
|
|
||||||
|
|
||||||
# Asserts
|
|
||||||
connect_ibmc.assert_called_with(**self.ibmc)
|
|
||||||
conn.system.reset.assert_called_once_with(reset_type)
|
|
||||||
self.assertEqual(4, conn.system.get.call_count)
|
|
||||||
|
|
||||||
# Reset Mocks
|
|
||||||
# TODO(Qianbiao.NG) why reset_mock does not reset call_count
|
|
||||||
connect_ibmc.reset_mock()
|
|
||||||
conn.system.get.reset_mock()
|
|
||||||
conn.system.reset.reset_mock()
|
|
||||||
|
|
||||||
@mock.patch.object(ibmc_client, 'connect', spec=object)
|
|
||||||
def test_set_power_state_not_reached(self, connect_ibmc):
|
|
||||||
conn = self.mock_ibmc_conn(connect_ibmc)
|
|
||||||
with task_manager.acquire(self.context, self.node.uuid,
|
|
||||||
shared=False) as task:
|
|
||||||
self.config(power_state_change_timeout=2, group='conductor')
|
|
||||||
|
|
||||||
state_mapping = mappings.SET_POWER_STATE_MAP
|
|
||||||
for (expect_state, reset_type) in state_mapping.items():
|
|
||||||
if expect_state in (states.POWER_OFF, states.SOFT_POWER_OFF):
|
|
||||||
final = constants.SYSTEM_POWER_STATE_OFF
|
|
||||||
transient = constants.SYSTEM_POWER_STATE_ON
|
|
||||||
else:
|
|
||||||
final = constants.SYSTEM_POWER_STATE_ON
|
|
||||||
transient = constants.SYSTEM_POWER_STATE_OFF
|
|
||||||
|
|
||||||
# Mocks
|
|
||||||
mock_system_get_results = (
|
|
||||||
[mock.Mock(power_state=transient)] * 5
|
|
||||||
+ [mock.Mock(power_state=final)])
|
|
||||||
conn.system.get.side_effect = mock_system_get_results
|
|
||||||
|
|
||||||
self.assertRaises(exception.PowerStateFailure,
|
|
||||||
task.driver.power.set_power_state,
|
|
||||||
task, expect_state)
|
|
||||||
|
|
||||||
# Asserts
|
|
||||||
connect_ibmc.assert_called_with(**self.ibmc)
|
|
||||||
conn.system.reset.assert_called_once_with(reset_type)
|
|
||||||
|
|
||||||
# Reset Mocks
|
|
||||||
connect_ibmc.reset_mock()
|
|
||||||
conn.system.get.reset_mock()
|
|
||||||
conn.system.reset.reset_mock()
|
|
||||||
|
|
||||||
@mock.patch.object(ibmc_client, 'connect', spec=object)
|
|
||||||
def test_set_power_state_fail(self, connect_ibmc):
|
|
||||||
conn = self.mock_ibmc_conn(connect_ibmc)
|
|
||||||
|
|
||||||
# Mocks
|
|
||||||
conn.system.reset.side_effect = (
|
|
||||||
ibmc_error.IBMCClientError
|
|
||||||
)
|
|
||||||
|
|
||||||
with task_manager.acquire(self.context, self.node.uuid,
|
|
||||||
shared=False) as task:
|
|
||||||
# Asserts
|
|
||||||
self.assertRaisesRegex(
|
|
||||||
exception.IBMCError, 'set iBMC power state',
|
|
||||||
task.driver.power.set_power_state, task, states.POWER_ON)
|
|
||||||
connect_ibmc.assert_called_with(**self.ibmc)
|
|
||||||
conn.system.reset.assert_called_once_with(constants.RESET_ON)
|
|
||||||
|
|
||||||
@mock.patch.object(ibmc_client, 'connect', spec=object)
|
|
||||||
def test_set_power_state_timeout(self, connect_ibmc):
|
|
||||||
conn = self.mock_ibmc_conn(connect_ibmc)
|
|
||||||
with task_manager.acquire(self.context, self.node.uuid,
|
|
||||||
shared=False) as task:
|
|
||||||
self.config(power_state_change_timeout=2, group='conductor')
|
|
||||||
|
|
||||||
# Mocks
|
|
||||||
conn.system.get.side_effect = (
|
|
||||||
[mock.Mock(power_state=constants.SYSTEM_POWER_STATE_OFF)] * 3
|
|
||||||
)
|
|
||||||
|
|
||||||
# Asserts
|
|
||||||
self.assertRaisesRegex(
|
|
||||||
exception.PowerStateFailure,
|
|
||||||
'Failed to set node power state to power on',
|
|
||||||
task.driver.power.set_power_state, task, states.POWER_ON)
|
|
||||||
|
|
||||||
connect_ibmc.assert_called_with(**self.ibmc)
|
|
||||||
conn.system.reset.assert_called_once_with(constants.RESET_ON)
|
|
||||||
|
|
||||||
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(sorted(list(mappings.SET_POWER_STATE_MAP)),
|
|
||||||
sorted(supported_power_states))
|
|
||||||
|
|
||||||
|
|
||||||
@mock.patch('oslo_utils.eventletutils.EventletEvent.wait',
|
|
||||||
lambda *args, **kwargs: None)
|
|
||||||
class IBMCPowerRebootTestCase(base.IBMCTestCase):
|
|
||||||
|
|
||||||
@mock.patch.object(ibmc_client, 'connect', spec=object)
|
|
||||||
def test_reboot(self, connect_ibmc):
|
|
||||||
conn = self.mock_ibmc_conn(connect_ibmc)
|
|
||||||
expected_values = [
|
|
||||||
(constants.SYSTEM_POWER_STATE_OFF, constants.RESET_ON),
|
|
||||||
(constants.SYSTEM_POWER_STATE_ON,
|
|
||||||
constants.RESET_FORCE_RESTART)
|
|
||||||
]
|
|
||||||
|
|
||||||
# for (expect_state, reset_type) in state_mapping.items():
|
|
||||||
for current, reset_type in expected_values:
|
|
||||||
mock_system_get_results = [
|
|
||||||
# Initial state
|
|
||||||
mock.Mock(power_state=current),
|
|
||||||
# Transient state - powering off
|
|
||||||
mock.Mock(power_state=constants.SYSTEM_POWER_STATE_OFF),
|
|
||||||
# Final state - down powering off
|
|
||||||
mock.Mock(power_state=constants.SYSTEM_POWER_STATE_ON)
|
|
||||||
]
|
|
||||||
conn.system.get.side_effect = mock_system_get_results
|
|
||||||
with task_manager.acquire(self.context, self.node.uuid,
|
|
||||||
shared=False) as task:
|
|
||||||
task.driver.power.reboot(task)
|
|
||||||
|
|
||||||
# Asserts
|
|
||||||
connect_ibmc.assert_called_with(**self.ibmc)
|
|
||||||
conn.system.reset.assert_called_once_with(reset_type)
|
|
||||||
|
|
||||||
# Reset Mocks
|
|
||||||
connect_ibmc.reset_mock()
|
|
||||||
conn.system.get.reset_mock()
|
|
||||||
conn.system.reset.reset_mock()
|
|
||||||
|
|
||||||
@mock.patch.object(ibmc_client, 'connect', spec=object)
|
|
||||||
def test_reboot_not_reached(self, connect_ibmc):
|
|
||||||
conn = self.mock_ibmc_conn(connect_ibmc)
|
|
||||||
with task_manager.acquire(self.context, self.node.uuid,
|
|
||||||
shared=False) as task:
|
|
||||||
|
|
||||||
# Mocks
|
|
||||||
conn.system.get.return_value = mock.Mock(
|
|
||||||
power_state=constants.SYSTEM_POWER_STATE_OFF)
|
|
||||||
self.assertRaisesRegex(
|
|
||||||
exception.PowerStateFailure,
|
|
||||||
'Failed to set node power state to power on',
|
|
||||||
task.driver.power.reboot, task)
|
|
||||||
|
|
||||||
# Asserts
|
|
||||||
connect_ibmc.assert_called_with(**self.ibmc)
|
|
||||||
conn.system.reset.assert_called_once_with(constants.RESET_ON)
|
|
||||||
|
|
||||||
@mock.patch.object(ibmc_client, 'connect', spec=object)
|
|
||||||
def test_reboot_fail(self, connect_ibmc):
|
|
||||||
conn = self.mock_ibmc_conn(connect_ibmc)
|
|
||||||
|
|
||||||
# Mocks
|
|
||||||
conn.system.reset.side_effect = (
|
|
||||||
ibmc_error.IBMCClientError
|
|
||||||
)
|
|
||||||
conn.system.get.return_value = mock.Mock(
|
|
||||||
power_state=constants.SYSTEM_POWER_STATE_ON
|
|
||||||
)
|
|
||||||
|
|
||||||
with task_manager.acquire(self.context, self.node.uuid,
|
|
||||||
shared=False) as task:
|
|
||||||
# Asserts
|
|
||||||
self.assertRaisesRegex(
|
|
||||||
exception.IBMCError, 'reboot iBMC',
|
|
||||||
task.driver.power.reboot, task)
|
|
||||||
connect_ibmc.assert_called_with(**self.ibmc)
|
|
||||||
conn.system.get.assert_called_once()
|
|
||||||
conn.system.reset.assert_called_once_with(
|
|
||||||
constants.RESET_FORCE_RESTART)
|
|
||||||
|
|
||||||
@mock.patch.object(ibmc_client, 'connect', spec=object)
|
|
||||||
def test_reboot_timeout(self, connect_ibmc):
|
|
||||||
conn = self.mock_ibmc_conn(connect_ibmc)
|
|
||||||
|
|
||||||
# Mocks
|
|
||||||
conn.system.get.side_effect = [mock.Mock(
|
|
||||||
power_state=constants.SYSTEM_POWER_STATE_OFF
|
|
||||||
)] * 5
|
|
||||||
|
|
||||||
with task_manager.acquire(self.context, self.node.uuid,
|
|
||||||
shared=False) as task:
|
|
||||||
self.config(power_state_change_timeout=2, group='conductor')
|
|
||||||
|
|
||||||
# Asserts
|
|
||||||
self.assertRaisesRegex(
|
|
||||||
exception.PowerStateFailure,
|
|
||||||
'Failed to set node power state to power on',
|
|
||||||
task.driver.power.reboot, task)
|
|
||||||
|
|
||||||
# Asserts
|
|
||||||
connect_ibmc.assert_called_with(**self.ibmc)
|
|
||||||
conn.system.reset.assert_called_once_with(
|
|
||||||
constants.RESET_ON)
|
|
@ -1,167 +0,0 @@
|
|||||||
# 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 iBMC RAID interface."""
|
|
||||||
|
|
||||||
from unittest import mock
|
|
||||||
|
|
||||||
from oslo_utils import importutils
|
|
||||||
|
|
||||||
from ironic.common import exception
|
|
||||||
from ironic.conductor import task_manager
|
|
||||||
from ironic.drivers.modules.ilo import raid as ilo_raid
|
|
||||||
from ironic.tests.unit.db import utils as db_utils
|
|
||||||
from ironic.tests.unit.drivers.modules.ibmc import base
|
|
||||||
|
|
||||||
constants = importutils.try_import('ibmc_client.constants')
|
|
||||||
ibmc_client = importutils.try_import('ibmc_client')
|
|
||||||
ibmc_error = importutils.try_import('ibmc_client.exceptions')
|
|
||||||
|
|
||||||
INFO_DICT = db_utils.get_test_ilo_info()
|
|
||||||
|
|
||||||
|
|
||||||
class IbmcRAIDTestCase(base.IBMCTestCase):
|
|
||||||
|
|
||||||
def setUp(self):
|
|
||||||
super(IbmcRAIDTestCase, self).setUp()
|
|
||||||
self.driver = mock.Mock(raid=ilo_raid.Ilo5RAID())
|
|
||||||
self.target_raid_config = {
|
|
||||||
"logical_disks": [
|
|
||||||
{
|
|
||||||
'size_gb': 200,
|
|
||||||
'raid_level': 0,
|
|
||||||
'is_root_volume': True
|
|
||||||
},
|
|
||||||
{
|
|
||||||
'size_gb': 'MAX',
|
|
||||||
'raid_level': 5
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
self.node.target_raid_config = self.target_raid_config
|
|
||||||
self.node.save()
|
|
||||||
|
|
||||||
@mock.patch.object(ibmc_client, 'connect', spec=object)
|
|
||||||
def test_sync_create_configuration_without_delete(self, connect_ibmc):
|
|
||||||
conn = self.mock_ibmc_conn(connect_ibmc)
|
|
||||||
conn.system.storage.apply_raid_configuration.return_value = None
|
|
||||||
|
|
||||||
with task_manager.acquire(self.context, self.node.uuid) as task:
|
|
||||||
result = task.driver.raid.create_configuration(
|
|
||||||
task, create_root_volume=True, create_nonroot_volumes=True,
|
|
||||||
delete_existing=False)
|
|
||||||
self.assertIsNone(result, "synchronous create raid configuration "
|
|
||||||
"should return None")
|
|
||||||
|
|
||||||
conn.system.storage.apply_raid_configuration.assert_called_once_with(
|
|
||||||
self.node.target_raid_config.get('logical_disks')
|
|
||||||
)
|
|
||||||
|
|
||||||
@mock.patch.object(ibmc_client, 'connect', spec=object)
|
|
||||||
def test_sync_create_configuration_with_delete(self, connect_ibmc):
|
|
||||||
conn = self.mock_ibmc_conn(connect_ibmc)
|
|
||||||
conn.system.storage.delete_all_raid_configuration.return_value = None
|
|
||||||
conn.system.storage.apply_raid_configuration.return_value = None
|
|
||||||
|
|
||||||
with task_manager.acquire(self.context, self.node.uuid) as task:
|
|
||||||
result = task.driver.raid.create_configuration(
|
|
||||||
task, create_root_volume=True, create_nonroot_volumes=True,
|
|
||||||
delete_existing=True)
|
|
||||||
self.assertIsNone(result, "synchronous create raid configuration "
|
|
||||||
"should return None")
|
|
||||||
|
|
||||||
conn.system.storage.delete_all_raid_configuration.assert_called_once()
|
|
||||||
conn.system.storage.apply_raid_configuration.assert_called_once_with(
|
|
||||||
self.node.target_raid_config.get('logical_disks')
|
|
||||||
)
|
|
||||||
|
|
||||||
@mock.patch.object(ibmc_client, 'connect', spec=object)
|
|
||||||
def test_sync_create_configuration_without_nonroot(self, connect_ibmc):
|
|
||||||
conn = self.mock_ibmc_conn(connect_ibmc)
|
|
||||||
conn.system.storage.delete_all_raid_configuration.return_value = None
|
|
||||||
conn.system.storage.apply_raid_configuration.return_value = None
|
|
||||||
|
|
||||||
with task_manager.acquire(self.context, self.node.uuid) as task:
|
|
||||||
result = task.driver.raid.create_configuration(
|
|
||||||
task, create_root_volume=True, create_nonroot_volumes=False,
|
|
||||||
delete_existing=True)
|
|
||||||
self.assertIsNone(result, "synchronous create raid configuration "
|
|
||||||
"should return None")
|
|
||||||
|
|
||||||
conn.system.storage.delete_all_raid_configuration.assert_called_once()
|
|
||||||
conn.system.storage.apply_raid_configuration.assert_called_once_with(
|
|
||||||
[{'size_gb': 200, 'raid_level': 0, 'is_root_volume': True}]
|
|
||||||
)
|
|
||||||
|
|
||||||
@mock.patch.object(ibmc_client, 'connect', spec=object)
|
|
||||||
def test_sync_create_configuration_without_root(self, connect_ibmc):
|
|
||||||
conn = self.mock_ibmc_conn(connect_ibmc)
|
|
||||||
conn.system.storage.delete_all_raid_configuration.return_value = None
|
|
||||||
conn.system.storage.apply_raid_configuration.return_value = None
|
|
||||||
|
|
||||||
with task_manager.acquire(self.context, self.node.uuid) as task:
|
|
||||||
result = task.driver.raid.create_configuration(
|
|
||||||
task, create_root_volume=False, create_nonroot_volumes=True,
|
|
||||||
delete_existing=True)
|
|
||||||
self.assertIsNone(result, "synchronous create raid configuration "
|
|
||||||
"should return None")
|
|
||||||
|
|
||||||
conn.system.storage.delete_all_raid_configuration.assert_called_once()
|
|
||||||
conn.system.storage.apply_raid_configuration.assert_called_once_with(
|
|
||||||
[{'size_gb': 'MAX', 'raid_level': 5}]
|
|
||||||
)
|
|
||||||
|
|
||||||
@mock.patch.object(ibmc_client, 'connect', spec=object)
|
|
||||||
def test_sync_create_configuration_failed(self, connect_ibmc):
|
|
||||||
conn = self.mock_ibmc_conn(connect_ibmc)
|
|
||||||
conn.system.storage.delete_all_raid_configuration.return_value = None
|
|
||||||
conn.system.storage.apply_raid_configuration.side_effect = (
|
|
||||||
ibmc_error.IBMCClientError
|
|
||||||
)
|
|
||||||
|
|
||||||
with task_manager.acquire(self.context, self.node.uuid) as task:
|
|
||||||
self.assertRaisesRegex(
|
|
||||||
exception.IBMCError, 'create iBMC RAID configuration',
|
|
||||||
task.driver.raid.create_configuration, task,
|
|
||||||
create_root_volume=True, create_nonroot_volumes=True,
|
|
||||||
delete_existing=True)
|
|
||||||
|
|
||||||
conn.system.storage.delete_all_raid_configuration.assert_called_once()
|
|
||||||
conn.system.storage.apply_raid_configuration.assert_called_once_with(
|
|
||||||
self.node.target_raid_config.get('logical_disks')
|
|
||||||
)
|
|
||||||
|
|
||||||
@mock.patch.object(ibmc_client, 'connect', spec=object)
|
|
||||||
def test_sync_delete_configuration_success(self, connect_ibmc):
|
|
||||||
conn = self.mock_ibmc_conn(connect_ibmc)
|
|
||||||
conn.system.storage.delete_all_raid_configuration.return_value = None
|
|
||||||
|
|
||||||
with task_manager.acquire(self.context, self.node.uuid) as task:
|
|
||||||
result = task.driver.raid.delete_configuration(task)
|
|
||||||
self.assertIsNone(result, "synchronous delete raid configuration "
|
|
||||||
"should return None")
|
|
||||||
|
|
||||||
conn.system.storage.delete_all_raid_configuration.assert_called_once()
|
|
||||||
|
|
||||||
@mock.patch.object(ibmc_client, 'connect', spec=object)
|
|
||||||
def test_sync_delete_configuration_failed(self, connect_ibmc):
|
|
||||||
conn = self.mock_ibmc_conn(connect_ibmc)
|
|
||||||
conn.system.storage.delete_all_raid_configuration.side_effect = (
|
|
||||||
ibmc_error.IBMCClientError
|
|
||||||
)
|
|
||||||
|
|
||||||
with task_manager.acquire(self.context, self.node.uuid) as task:
|
|
||||||
self.assertRaisesRegex(
|
|
||||||
exception.IBMCError, 'delete iBMC RAID configuration',
|
|
||||||
task.driver.raid.delete_configuration, task)
|
|
||||||
|
|
||||||
conn.system.storage.delete_all_raid_configuration.assert_called_once()
|
|
@ -1,173 +0,0 @@
|
|||||||
#
|
|
||||||
# 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 iBMC Driver common utils."""
|
|
||||||
|
|
||||||
import copy
|
|
||||||
import os
|
|
||||||
from unittest import mock
|
|
||||||
|
|
||||||
from oslo_utils import importutils
|
|
||||||
|
|
||||||
from ironic.common import exception
|
|
||||||
from ironic.conductor import task_manager
|
|
||||||
from ironic.drivers.modules.ibmc import utils
|
|
||||||
from ironic.tests.unit.drivers.modules.ibmc import base
|
|
||||||
|
|
||||||
constants = importutils.try_import('ibmc_client.constants')
|
|
||||||
ibmc_client = importutils.try_import('ibmc_client')
|
|
||||||
ibmc_error = importutils.try_import('ibmc_client.exceptions')
|
|
||||||
|
|
||||||
|
|
||||||
class IBMCUtilsTestCase(base.IBMCTestCase):
|
|
||||||
|
|
||||||
def setUp(self):
|
|
||||||
super(IBMCUtilsTestCase, self).setUp()
|
|
||||||
# Redfish specific configurations
|
|
||||||
self.config(connection_attempts=2, group='ibmc')
|
|
||||||
self.parsed_driver_info = {
|
|
||||||
'address': 'https://example.com',
|
|
||||||
'username': 'username',
|
|
||||||
'password': 'password',
|
|
||||||
'verify_ca': True,
|
|
||||||
}
|
|
||||||
|
|
||||||
def test_parse_driver_info(self):
|
|
||||||
response = utils.parse_driver_info(self.node)
|
|
||||||
self.assertEqual(self.parsed_driver_info, response)
|
|
||||||
|
|
||||||
def test_parse_driver_info_default_scheme(self):
|
|
||||||
self.node.driver_info['ibmc_address'] = 'example.com'
|
|
||||||
response = utils.parse_driver_info(self.node)
|
|
||||||
self.assertEqual(self.parsed_driver_info, response)
|
|
||||||
|
|
||||||
def test_parse_driver_info_default_scheme_with_port(self):
|
|
||||||
self.node.driver_info['ibmc_address'] = 'example.com:42'
|
|
||||||
self.parsed_driver_info['address'] = 'https://example.com:42'
|
|
||||||
response = utils.parse_driver_info(self.node)
|
|
||||||
self.assertEqual(self.parsed_driver_info, response)
|
|
||||||
|
|
||||||
def test_parse_driver_info_missing_info(self):
|
|
||||||
for prop in utils.REQUIRED_PROPERTIES:
|
|
||||||
self.node.driver_info = self.driver_info.copy()
|
|
||||||
self.node.driver_info.pop(prop)
|
|
||||||
self.assertRaises(exception.MissingParameterValue,
|
|
||||||
utils.parse_driver_info, self.node)
|
|
||||||
|
|
||||||
def test_parse_driver_info_invalid_address(self):
|
|
||||||
for value in ['/banana!', '#location', '?search=hello']:
|
|
||||||
self.node.driver_info['ibmc_address'] = value
|
|
||||||
self.assertRaisesRegex(exception.InvalidParameterValue,
|
|
||||||
'Invalid iBMC address',
|
|
||||||
utils.parse_driver_info, self.node)
|
|
||||||
|
|
||||||
@mock.patch.object(os.path, 'exists', autospec=True)
|
|
||||||
def test_parse_driver_info_path_verify_ca(self,
|
|
||||||
mock_isdir):
|
|
||||||
mock_isdir.return_value = True
|
|
||||||
fake_path = '/path/to/a/valid/CA'
|
|
||||||
self.node.driver_info['ibmc_verify_ca'] = fake_path
|
|
||||||
self.parsed_driver_info['verify_ca'] = fake_path
|
|
||||||
|
|
||||||
response = utils.parse_driver_info(self.node)
|
|
||||||
self.assertEqual(self.parsed_driver_info, response)
|
|
||||||
mock_isdir.assert_called_once_with(fake_path)
|
|
||||||
|
|
||||||
@mock.patch.object(os.path, 'exists', autospec=True)
|
|
||||||
def test_parse_driver_info_valid_capath(self, mock_isfile):
|
|
||||||
mock_isfile.return_value = True
|
|
||||||
fake_path = '/path/to/a/valid/CA.pem'
|
|
||||||
self.node.driver_info['ibmc_verify_ca'] = fake_path
|
|
||||||
self.parsed_driver_info['verify_ca'] = fake_path
|
|
||||||
|
|
||||||
response = utils.parse_driver_info(self.node)
|
|
||||||
self.assertEqual(self.parsed_driver_info, response)
|
|
||||||
mock_isfile.assert_called_once_with(fake_path)
|
|
||||||
|
|
||||||
def test_parse_driver_info_invalid_value_verify_ca(self):
|
|
||||||
# Integers are not supported
|
|
||||||
self.node.driver_info['ibmc_verify_ca'] = 123456
|
|
||||||
self.assertRaisesRegex(exception.InvalidParameterValue,
|
|
||||||
'Invalid value type',
|
|
||||||
utils.parse_driver_info, self.node)
|
|
||||||
|
|
||||||
def test_parse_driver_info_valid_string_value_verify_ca(self):
|
|
||||||
for value in ('0', 'f', 'false', 'off', 'n', 'no'):
|
|
||||||
self.node.driver_info['ibmc_verify_ca'] = value
|
|
||||||
response = utils.parse_driver_info(self.node)
|
|
||||||
parsed_driver_info = copy.deepcopy(self.parsed_driver_info)
|
|
||||||
parsed_driver_info['verify_ca'] = False
|
|
||||||
self.assertEqual(parsed_driver_info, response)
|
|
||||||
|
|
||||||
for value in ('1', 't', 'true', 'on', 'y', 'yes'):
|
|
||||||
self.node.driver_info['ibmc_verify_ca'] = value
|
|
||||||
response = utils.parse_driver_info(self.node)
|
|
||||||
self.assertEqual(self.parsed_driver_info, response)
|
|
||||||
|
|
||||||
def test_parse_driver_info_invalid_string_value_verify_ca(self):
|
|
||||||
for value in ('xyz', '*', '!123', '123'):
|
|
||||||
self.node.driver_info['ibmc_verify_ca'] = value
|
|
||||||
self.assertRaisesRegex(exception.InvalidParameterValue,
|
|
||||||
'The value should be a Boolean',
|
|
||||||
utils.parse_driver_info, self.node)
|
|
||||||
|
|
||||||
def test_revert_dictionary(self):
|
|
||||||
data = {
|
|
||||||
"key1": "value1",
|
|
||||||
"key2": "value2"
|
|
||||||
}
|
|
||||||
|
|
||||||
revert = utils.revert_dictionary(data)
|
|
||||||
self.assertEqual({
|
|
||||||
"value1": "key1",
|
|
||||||
"value2": "key2"
|
|
||||||
}, revert)
|
|
||||||
|
|
||||||
@mock.patch('time.sleep', autospec=True)
|
|
||||||
@mock.patch.object(ibmc_client, 'connect', spec=object)
|
|
||||||
def test_handle_ibmc_exception_retry(self, connect_ibmc, mock_sleep):
|
|
||||||
|
|
||||||
@utils.handle_ibmc_exception('get IBMC system')
|
|
||||||
def get_ibmc_system(_task):
|
|
||||||
driver_info = utils.parse_driver_info(_task.node)
|
|
||||||
with ibmc_client.connect(**driver_info) as _conn:
|
|
||||||
return _conn.system.get()
|
|
||||||
|
|
||||||
conn = self.mock_ibmc_conn(connect_ibmc)
|
|
||||||
# Mocks
|
|
||||||
conn.system.get.side_effect = [
|
|
||||||
ibmc_error.IBMCConnectionError(url=self.ibmc['address'],
|
|
||||||
error='Failed to connect to host'),
|
|
||||||
mock.PropertyMock(
|
|
||||||
boot_source_override=mock.PropertyMock(
|
|
||||||
target=constants.BOOT_SOURCE_TARGET_PXE,
|
|
||||||
enabled=constants.BOOT_SOURCE_ENABLED_CONTINUOUS
|
|
||||||
)
|
|
||||||
)
|
|
||||||
]
|
|
||||||
|
|
||||||
with task_manager.acquire(self.context, self.node.uuid,
|
|
||||||
shared=True) as task:
|
|
||||||
system = get_ibmc_system(task)
|
|
||||||
|
|
||||||
# Asserts
|
|
||||||
self.assertEqual(constants.BOOT_SOURCE_TARGET_PXE,
|
|
||||||
system.boot_source_override.target)
|
|
||||||
self.assertEqual(constants.BOOT_SOURCE_ENABLED_CONTINUOUS,
|
|
||||||
system.boot_source_override.enabled)
|
|
||||||
|
|
||||||
# 1 failed, 1 succeed
|
|
||||||
connect_ibmc.assert_called_with(**self.ibmc)
|
|
||||||
self.assertEqual(2, connect_ibmc.call_count)
|
|
||||||
|
|
||||||
# 1 failed, 1 succeed
|
|
||||||
self.assertEqual(2, conn.system.get.call_count)
|
|
@ -1,79 +0,0 @@
|
|||||||
#
|
|
||||||
# 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 iBMC vendor interface."""
|
|
||||||
|
|
||||||
from unittest import mock
|
|
||||||
|
|
||||||
from oslo_utils import importutils
|
|
||||||
|
|
||||||
from ironic.conductor import task_manager
|
|
||||||
from ironic.drivers.modules.ibmc import utils
|
|
||||||
from ironic.tests.unit.drivers.modules.ibmc import base
|
|
||||||
|
|
||||||
ibmc_client = importutils.try_import('ibmc_client')
|
|
||||||
|
|
||||||
|
|
||||||
@mock.patch('oslo_utils.eventletutils.EventletEvent.wait',
|
|
||||||
lambda *args, **kwargs: None)
|
|
||||||
class IBMCVendorTestCase(base.IBMCTestCase):
|
|
||||||
|
|
||||||
def setUp(self):
|
|
||||||
super(IBMCVendorTestCase, self).setUp()
|
|
||||||
|
|
||||||
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 utils.COMMON_PROPERTIES:
|
|
||||||
self.assertIn(prop, properties)
|
|
||||||
|
|
||||||
@mock.patch.object(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(ibmc_client, 'connect', spec=object)
|
|
||||||
def test_list_boot_type_order(self, connect_ibmc):
|
|
||||||
# Mocks
|
|
||||||
conn = self.mock_ibmc_conn(connect_ibmc)
|
|
||||||
boot_up_seq = ['Pxe', 'Hdd', 'Others', 'Cd']
|
|
||||||
conn.system.get.return_value = mock.Mock(
|
|
||||||
boot_sequence=['Pxe', 'Hdd', 'Others', 'Cd']
|
|
||||||
)
|
|
||||||
|
|
||||||
expected = {'boot_up_sequence': boot_up_seq}
|
|
||||||
with task_manager.acquire(self.context, self.node.uuid,
|
|
||||||
shared=True) as task:
|
|
||||||
seq = task.driver.vendor.boot_up_seq(task)
|
|
||||||
conn.system.get.assert_called_once_with()
|
|
||||||
connect_ibmc.assert_called_once_with(**self.ibmc)
|
|
||||||
self.assertEqual(expected, seq)
|
|
||||||
|
|
||||||
@mock.patch.object(ibmc_client, 'connect', spec=object)
|
|
||||||
def test_list_raid_controller(self, connect_ibmc):
|
|
||||||
# Mocks
|
|
||||||
conn = self.mock_ibmc_conn(connect_ibmc)
|
|
||||||
|
|
||||||
ctrl = mock.Mock()
|
|
||||||
summary = ctrl.summary.return_value
|
|
||||||
conn.system.storage.list.return_value = [ctrl]
|
|
||||||
|
|
||||||
with task_manager.acquire(self.context, self.node.uuid,
|
|
||||||
shared=True) as task:
|
|
||||||
summries = task.driver.vendor.get_raid_controller_list(task)
|
|
||||||
ctrl.summary.assert_called_once_with()
|
|
||||||
conn.system.storage.list.assert_called_once_with()
|
|
||||||
connect_ibmc.assert_called_once_with(**self.ibmc)
|
|
||||||
self.assertEqual([summary], summries)
|
|
@ -10,7 +10,7 @@
|
|||||||
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
# License for the specific language governing permissions and limitations
|
# License for the specific language governing permissions and limitations
|
||||||
# under the License.
|
# under the License.
|
||||||
"""Test base class for iBMC Driver."""
|
"""Test base class for intel ipmi Driver."""
|
||||||
|
|
||||||
from ironic.tests.unit.db import base as db_base
|
from ironic.tests.unit.db import base as db_base
|
||||||
from ironic.tests.unit.db import utils as db_utils
|
from ironic.tests.unit.db import utils as db_utils
|
||||||
|
@ -1,52 +0,0 @@
|
|||||||
#
|
|
||||||
# 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.
|
|
||||||
|
|
||||||
# Version 1.0.0
|
|
||||||
|
|
||||||
from ironic.conductor import task_manager
|
|
||||||
from ironic.drivers.modules import agent
|
|
||||||
from ironic.drivers.modules.ibmc import management as ibmc_mgmt
|
|
||||||
from ironic.drivers.modules.ibmc import power as ibmc_power
|
|
||||||
from ironic.drivers.modules.ibmc import raid as ibmc_raid
|
|
||||||
from ironic.drivers.modules.ibmc import vendor as ibmc_vendor
|
|
||||||
from ironic.drivers.modules import inspector
|
|
||||||
from ironic.drivers.modules import noop
|
|
||||||
from ironic.drivers.modules import pxe
|
|
||||||
from ironic.tests.unit.db import base as db_base
|
|
||||||
from ironic.tests.unit.objects import utils as obj_utils
|
|
||||||
|
|
||||||
|
|
||||||
class IBMCHardwareTestCase(db_base.DbTestCase):
|
|
||||||
|
|
||||||
def setUp(self):
|
|
||||||
super(IBMCHardwareTestCase, self).setUp()
|
|
||||||
self.config(enabled_hardware_types=['ibmc'],
|
|
||||||
enabled_power_interfaces=['ibmc'],
|
|
||||||
enabled_management_interfaces=['ibmc'],
|
|
||||||
enabled_vendor_interfaces=['ibmc'],
|
|
||||||
enabled_raid_interfaces=['ibmc'],
|
|
||||||
enabled_inspect_interfaces=['inspector', 'no-inspect'])
|
|
||||||
|
|
||||||
def test_default_interfaces(self):
|
|
||||||
node = obj_utils.create_test_node(self.context, driver='ibmc')
|
|
||||||
with task_manager.acquire(self.context, node.id) as task:
|
|
||||||
self.assertIsInstance(task.driver.management,
|
|
||||||
ibmc_mgmt.IBMCManagement)
|
|
||||||
self.assertIsInstance(task.driver.power,
|
|
||||||
ibmc_power.IBMCPower)
|
|
||||||
self.assertIsInstance(task.driver.boot, pxe.PXEBoot)
|
|
||||||
self.assertIsInstance(task.driver.console, noop.NoConsole)
|
|
||||||
self.assertIsInstance(task.driver.deploy, agent.AgentDeploy)
|
|
||||||
self.assertIsInstance(task.driver.raid, ibmc_raid.IbmcRAID)
|
|
||||||
self.assertIsInstance(task.driver.vendor, ibmc_vendor.IBMCVendor)
|
|
||||||
self.assertIsInstance(task.driver.inspect, inspector.Inspector)
|
|
@ -121,10 +121,3 @@ SCCICLIENT_VIOM_CONF_SPEC = (
|
|||||||
REDFISH_SPEC = (
|
REDFISH_SPEC = (
|
||||||
'redfish',
|
'redfish',
|
||||||
)
|
)
|
||||||
|
|
||||||
# python-ibmcclient
|
|
||||||
IBMCCLIENT_SPEC = (
|
|
||||||
'connect',
|
|
||||||
'exceptions',
|
|
||||||
'constants',
|
|
||||||
)
|
|
||||||
|
@ -27,7 +27,6 @@ Current list of mocked libraries:
|
|||||||
- pysnmp
|
- pysnmp
|
||||||
- scciclient
|
- scciclient
|
||||||
- python-dracclient
|
- python-dracclient
|
||||||
- python-ibmcclient
|
|
||||||
- sushy_oem_idrac
|
- sushy_oem_idrac
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@ -194,44 +193,3 @@ class MockKwargsException(Exception):
|
|||||||
def __init__(self, *args, **kwargs):
|
def __init__(self, *args, **kwargs):
|
||||||
super(MockKwargsException, self).__init__(*args)
|
super(MockKwargsException, self).__init__(*args)
|
||||||
self.kwargs = kwargs
|
self.kwargs = kwargs
|
||||||
|
|
||||||
|
|
||||||
# python-ibmcclient mocks for HUAWEI rack server driver
|
|
||||||
ibmc_client = importutils.try_import('ibmc_client')
|
|
||||||
if not ibmc_client:
|
|
||||||
ibmc_client = mock.MagicMock(spec_set=mock_specs.IBMCCLIENT_SPEC)
|
|
||||||
sys.modules['ibmc_client'] = ibmc_client
|
|
||||||
|
|
||||||
# Mock iBMC client exceptions
|
|
||||||
exceptions = mock.MagicMock()
|
|
||||||
exceptions.IBMCConnectionError = (
|
|
||||||
type('IBMCConnectionError', (MockKwargsException,), {}))
|
|
||||||
exceptions.IBMCClientError = (
|
|
||||||
type('IBMCClientError', (MockKwargsException,), {}))
|
|
||||||
sys.modules['ibmc_client.exceptions'] = exceptions
|
|
||||||
|
|
||||||
# Mock iIBMC client constants
|
|
||||||
constants = mock.MagicMock(
|
|
||||||
SYSTEM_POWER_STATE_ON='On',
|
|
||||||
SYSTEM_POWER_STATE_OFF='Off',
|
|
||||||
BOOT_SOURCE_TARGET_NONE='None',
|
|
||||||
BOOT_SOURCE_TARGET_PXE='Pxe',
|
|
||||||
BOOT_SOURCE_TARGET_FLOPPY='Floppy',
|
|
||||||
BOOT_SOURCE_TARGET_CD='Cd',
|
|
||||||
BOOT_SOURCE_TARGET_HDD='Hdd',
|
|
||||||
BOOT_SOURCE_TARGET_BIOS_SETUP='BiosSetup',
|
|
||||||
BOOT_SOURCE_MODE_BIOS='Legacy',
|
|
||||||
BOOT_SOURCE_MODE_UEFI='UEFI',
|
|
||||||
BOOT_SOURCE_ENABLED_ONCE='Once',
|
|
||||||
BOOT_SOURCE_ENABLED_CONTINUOUS='Continuous',
|
|
||||||
BOOT_SOURCE_ENABLED_DISABLED='Disabled',
|
|
||||||
RESET_NMI='Nmi',
|
|
||||||
RESET_ON='On',
|
|
||||||
RESET_FORCE_OFF='ForceOff',
|
|
||||||
RESET_GRACEFUL_SHUTDOWN='GracefulShutdown',
|
|
||||||
RESET_FORCE_RESTART='ForceRestart',
|
|
||||||
RESET_FORCE_POWER_CYCLE='ForcePowerCycle')
|
|
||||||
sys.modules['ibmc_client.constants'] = constants
|
|
||||||
|
|
||||||
if 'ironic.drivers.modules.ibmc' in sys.modules:
|
|
||||||
importlib.reload(sys.modules['ironic.drivers.modules.ibmc'])
|
|
||||||
|
@ -0,0 +1,4 @@
|
|||||||
|
---
|
||||||
|
upgrade:
|
||||||
|
- |
|
||||||
|
The deprecated ``ibmc`` hardware type has been removed from Ironic.
|
@ -119,7 +119,6 @@ ironic.hardware.interfaces.inspect =
|
|||||||
|
|
||||||
ironic.hardware.interfaces.management =
|
ironic.hardware.interfaces.management =
|
||||||
fake = ironic.drivers.modules.fake:FakeManagement
|
fake = ironic.drivers.modules.fake:FakeManagement
|
||||||
ibmc = ironic.drivers.modules.ibmc.management:IBMCManagement
|
|
||||||
idrac = ironic.drivers.modules.drac.management:DracManagement
|
idrac = ironic.drivers.modules.drac.management:DracManagement
|
||||||
idrac-redfish = ironic.drivers.modules.drac.management:DracRedfishManagement
|
idrac-redfish = ironic.drivers.modules.drac.management:DracRedfishManagement
|
||||||
idrac-wsman = ironic.drivers.modules.drac.management:DracWSManManagement
|
idrac-wsman = ironic.drivers.modules.drac.management:DracWSManManagement
|
||||||
@ -139,7 +138,6 @@ ironic.hardware.interfaces.network =
|
|||||||
ironic.hardware.interfaces.power =
|
ironic.hardware.interfaces.power =
|
||||||
agent = ironic.drivers.modules.agent_power:AgentPower
|
agent = ironic.drivers.modules.agent_power:AgentPower
|
||||||
fake = ironic.drivers.modules.fake:FakePower
|
fake = ironic.drivers.modules.fake:FakePower
|
||||||
ibmc = ironic.drivers.modules.ibmc.power:IBMCPower
|
|
||||||
idrac = ironic.drivers.modules.drac.power:DracPower
|
idrac = ironic.drivers.modules.drac.power:DracPower
|
||||||
idrac-redfish = ironic.drivers.modules.drac.power:DracRedfishPower
|
idrac-redfish = ironic.drivers.modules.drac.power:DracRedfishPower
|
||||||
idrac-wsman = ironic.drivers.modules.drac.power:DracWSManPower
|
idrac-wsman = ironic.drivers.modules.drac.power:DracWSManPower
|
||||||
@ -152,7 +150,6 @@ ironic.hardware.interfaces.power =
|
|||||||
ironic.hardware.interfaces.raid =
|
ironic.hardware.interfaces.raid =
|
||||||
agent = ironic.drivers.modules.agent:AgentRAID
|
agent = ironic.drivers.modules.agent:AgentRAID
|
||||||
fake = ironic.drivers.modules.fake:FakeRAID
|
fake = ironic.drivers.modules.fake:FakeRAID
|
||||||
ibmc = ironic.drivers.modules.ibmc.raid:IbmcRAID
|
|
||||||
idrac = ironic.drivers.modules.drac.raid:DracRAID
|
idrac = ironic.drivers.modules.drac.raid:DracRAID
|
||||||
idrac-redfish = ironic.drivers.modules.drac.raid:DracRedfishRAID
|
idrac-redfish = ironic.drivers.modules.drac.raid:DracRedfishRAID
|
||||||
idrac-wsman = ironic.drivers.modules.drac.raid:DracWSManRAID
|
idrac-wsman = ironic.drivers.modules.drac.raid:DracWSManRAID
|
||||||
@ -174,7 +171,6 @@ ironic.hardware.interfaces.storage =
|
|||||||
|
|
||||||
ironic.hardware.interfaces.vendor =
|
ironic.hardware.interfaces.vendor =
|
||||||
fake = ironic.drivers.modules.fake:FakeVendorB
|
fake = ironic.drivers.modules.fake:FakeVendorB
|
||||||
ibmc = ironic.drivers.modules.ibmc.vendor:IBMCVendor
|
|
||||||
idrac = ironic.drivers.modules.drac.vendor_passthru:DracVendorPassthru
|
idrac = ironic.drivers.modules.drac.vendor_passthru:DracVendorPassthru
|
||||||
idrac-wsman = ironic.drivers.modules.drac.vendor_passthru:DracWSManVendorPassthru
|
idrac-wsman = ironic.drivers.modules.drac.vendor_passthru:DracWSManVendorPassthru
|
||||||
idrac-redfish = ironic.drivers.modules.drac.vendor_passthru:DracRedfishVendorPassthru
|
idrac-redfish = ironic.drivers.modules.drac.vendor_passthru:DracRedfishVendorPassthru
|
||||||
@ -186,7 +182,6 @@ ironic.hardware.interfaces.vendor =
|
|||||||
|
|
||||||
ironic.hardware.types =
|
ironic.hardware.types =
|
||||||
fake-hardware = ironic.drivers.fake_hardware:FakeHardware
|
fake-hardware = ironic.drivers.fake_hardware:FakeHardware
|
||||||
ibmc = ironic.drivers.ibmc:IBMCHardware
|
|
||||||
idrac = ironic.drivers.drac:IDRACHardware
|
idrac = ironic.drivers.drac:IDRACHardware
|
||||||
ilo = ironic.drivers.ilo:IloHardware
|
ilo = ironic.drivers.ilo:IloHardware
|
||||||
ilo5 = ironic.drivers.ilo:Ilo5Hardware
|
ilo5 = ironic.drivers.ilo:Ilo5Hardware
|
||||||
|
Loading…
x
Reference in New Issue
Block a user