164 lines
6.2 KiB
Python
164 lines
6.2 KiB
Python
#
|
|
# 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.
|
|
|
|
"""
|
|
DRAC Power Driver using the Base Server Profile
|
|
"""
|
|
|
|
from ironic.common import exception
|
|
from ironic.common import i18n
|
|
from ironic.common import states
|
|
from ironic.drivers import base
|
|
from ironic.drivers.modules.drac import common as drac_common
|
|
from ironic.drivers.modules.drac import resource_uris
|
|
from ironic.openstack.common import excutils
|
|
from ironic.openstack.common import importutils
|
|
from ironic.openstack.common import log as logging
|
|
|
|
pywsman = importutils.try_import('pywsman')
|
|
|
|
LOG = logging.getLogger(__name__)
|
|
|
|
_LE = i18n._LE
|
|
|
|
POWER_STATES = {
|
|
'2': states.POWER_ON,
|
|
'3': states.POWER_OFF,
|
|
'11': states.REBOOT,
|
|
}
|
|
|
|
REVERSE_POWER_STATES = dict((v, k) for (k, v) in POWER_STATES.items())
|
|
|
|
|
|
def _get_power_state(node):
|
|
"""Returns the current power state of the node
|
|
|
|
:param node: The node.
|
|
:returns: power state, one of :mod: `ironic.common.states`.
|
|
:raises: DracClientError if the client received unexpected response.
|
|
:raises: InvalidParameterValue if required DRAC credentials are missing.
|
|
"""
|
|
|
|
client = drac_common.get_wsman_client(node)
|
|
options = pywsman.ClientOptions()
|
|
filter = pywsman.Filter()
|
|
filter_dialect = 'http://schemas.dmtf.org/wbem/cql/1/dsp0202.pdf'
|
|
filter_query = ('select EnabledState,ElementName from CIM_ComputerSystem '
|
|
'where Name="srv:system"')
|
|
filter.simple(filter_dialect, filter_query)
|
|
|
|
try:
|
|
docs = client.wsman_enumerate(resource_uris.DCIM_ComputerSystem,
|
|
options, filter)
|
|
except exception.DracClientError as exc:
|
|
with excutils.save_and_reraise_exception():
|
|
LOG.error(_LE('DRAC driver failed to get power state for node '
|
|
'%(node_uuid)s. Reason: %(error)s.'),
|
|
{'node_uuid': node.uuid, 'error': exc})
|
|
|
|
doc = docs[0]
|
|
enabled_state = str(doc.find(resource_uris.DCIM_ComputerSystem,
|
|
'EnabledState'))
|
|
|
|
return POWER_STATES[enabled_state]
|
|
|
|
|
|
def _set_power_state(node, target_state):
|
|
"""Turns the server power on/off or do a reboot.
|
|
|
|
:param node: an ironic node object.
|
|
:param target_state: target state of the node.
|
|
:raises: DracClientError if the client received unexpected response.
|
|
:raises: InvalidParameterValue if an invalid power state was specified.
|
|
"""
|
|
|
|
client = drac_common.get_wsman_client(node)
|
|
options = pywsman.ClientOptions()
|
|
options.add_selector('CreationClassName', 'DCIM_ComputerSystem')
|
|
options.add_selector('Name', 'srv:system')
|
|
options.add_property('RequestedState', REVERSE_POWER_STATES[target_state])
|
|
|
|
try:
|
|
root = client.wsman_invoke(resource_uris.DCIM_ComputerSystem, options,
|
|
'RequestStateChange')
|
|
except exception.DracClientError as exc:
|
|
with excutils.save_and_reraise_exception():
|
|
LOG.error(_LE('DRAC driver failed to set power state for node '
|
|
'%(node_uuid)s to %(target_power_state)s. '
|
|
'Reason: %(error)s.'),
|
|
{'node_uuid': node.uuid,
|
|
'target_power_state': target_state,
|
|
'error': exc})
|
|
|
|
return_value = str(root.find(resource_uris.DCIM_ComputerSystem,
|
|
'ReturnValue'))
|
|
|
|
if return_value != '0':
|
|
message = str(root.find(resource_uris.DCIM_ComputerSystem, 'Message'))
|
|
LOG.error(_LE('DRAC driver failed to set power state for node '
|
|
'%(node_uuid)s to %(target_power_state)s. '
|
|
'Reason: %(error)s.'),
|
|
{'node_uuid': node.uuid,
|
|
'target_power_state': target_state,
|
|
'error': message})
|
|
raise exception.DracOperationError(operation='set_power_state',
|
|
error=message)
|
|
|
|
|
|
class DracPower(base.PowerInterface):
|
|
"""Interface for power-related actions."""
|
|
|
|
def get_properties(self):
|
|
return drac_common.COMMON_PROPERTIES
|
|
|
|
def validate(self, task):
|
|
"""Validate the driver-specific Node power info.
|
|
|
|
This method validates whether the 'driver_info' property of the
|
|
supplied node contains the required information for this driver to
|
|
manage the power state of the node.
|
|
|
|
:param task: a TaskManager instance containing the node to act on.
|
|
:raises: InvalidParameterValue if required driver_info attribute
|
|
is missing or invalid on the node.
|
|
"""
|
|
return drac_common.parse_driver_info(task.node)
|
|
|
|
def get_power_state(self, task):
|
|
"""Return the 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: DracClientError if the client received unexpected response.
|
|
"""
|
|
return _get_power_state(task.node)
|
|
|
|
def set_power_state(self, task, power_state):
|
|
"""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`.
|
|
:raises: DracClientError if the client received unexpected response.
|
|
:raises: DracOperationError if failed to set the power state.
|
|
"""
|
|
return _set_power_state(task.node, power_state)
|
|
|
|
def reboot(self, task):
|
|
"""Perform a hard reboot of the task's node.
|
|
|
|
:param task: a TaskManager instance containing the node to act on.
|
|
:raises: DracClientError if the client received unexpected response.
|
|
:raises: DracOperationError if failed to set the power state.
|
|
"""
|
|
return _set_power_state(task.node, states.REBOOT)
|