cbe0dbf32d
The root device configuration code and unit tests come from the python-tripleoclient project. By moving them into a common Mistral action, they can be reused by other clients like the UI. The workflows will also be used for the new 'overcloud node configure' command in the client. Change-Id: I5ba0a3710012c44822dd3b8e69662bbef04d3787 Related-Bug: #1595205
114 lines
3.4 KiB
Python
114 lines
3.4 KiB
Python
# Copyright 2015 Red Hat, Inc.
|
|
# All Rights Reserved.
|
|
#
|
|
# -*- coding: utf-8 -*-
|
|
|
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
|
# not use this file except in compliance with the License. You may obtain
|
|
# a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
# License for the specific language governing permissions and limitations
|
|
# under the License.
|
|
import logging
|
|
import sys
|
|
|
|
import six
|
|
from six import reraise as raise_
|
|
|
|
from tripleo_common.i18n import _
|
|
from tripleo_common.i18n import _LE
|
|
|
|
_FATAL_EXCEPTION_FORMAT_ERRORS = False
|
|
|
|
LOG = logging.getLogger(__name__)
|
|
|
|
|
|
@six.python_2_unicode_compatible
|
|
class TripleoCommonException(Exception):
|
|
"""Base Tripleo-Common Exception.
|
|
|
|
To correctly use this class, inherit from it and define a 'msg_fmt'
|
|
property. That msg_fmt will get printf'd with the keyword arguments
|
|
provided to the constructor.
|
|
"""
|
|
message = _("An unknown exception occurred.")
|
|
|
|
def __init__(self, **kwargs):
|
|
self.kwargs = kwargs
|
|
|
|
try:
|
|
self.message = self.msg_fmt % kwargs
|
|
except KeyError:
|
|
exc_info = sys.exc_info()
|
|
# kwargs doesn't match a variable in the message
|
|
# log the issue and the kwargs
|
|
LOG.exception(_LE('Exception in string format operation'))
|
|
for name, value in six.iteritems(kwargs):
|
|
LOG.error(_LE("%(name)s: %(value)s"),
|
|
{'name': name, 'value': value}) # noqa
|
|
|
|
if _FATAL_EXCEPTION_FORMAT_ERRORS:
|
|
raise_(exc_info[0], exc_info[1], exc_info[2])
|
|
|
|
def __str__(self):
|
|
return self.message
|
|
|
|
def __deepcopy__(self, memo):
|
|
return self.__class__(**self.kwargs)
|
|
|
|
|
|
class StackInUseError(TripleoCommonException):
|
|
msg_fmt = _("Cannot delete a plan that has an associated stack.")
|
|
|
|
|
|
class ContainerAlreadyExistsError(TripleoCommonException):
|
|
msg_fmt = _("A container with the name %(name)s already exists.")
|
|
|
|
|
|
class HeatValidationFailedError(TripleoCommonException):
|
|
msg_fmt = _("The plan failed to validate via the Heat service. %(msg)s")
|
|
|
|
|
|
class InvalidNode(ValueError):
|
|
"""Node data is invalid."""
|
|
|
|
def __init__(self, message, node=None):
|
|
message = 'Invalid node data: %s' % message
|
|
self.node = node
|
|
super(InvalidNode, self).__init__(message)
|
|
|
|
|
|
class Timeout(Exception):
|
|
"""An operation timed out"""
|
|
|
|
def __init__(self, message):
|
|
message = 'An operation timed out: %s' % message
|
|
super(Timeout, self).__init__(message)
|
|
|
|
|
|
class StateTransitionFailed(Exception):
|
|
"""Ironic node state transition failed"""
|
|
|
|
def __init__(self, node, target_state):
|
|
self.node = node
|
|
self.target_state = target_state
|
|
message = (
|
|
"Error transitioning Ironic node %(uuid)s to provision state "
|
|
"%(state)s: %(error)s. Now in state %(actual)s." % {
|
|
'uuid': node.uuid,
|
|
'state': target_state,
|
|
'error': node.last_error,
|
|
'actual': node.provision_state
|
|
}
|
|
)
|
|
super(StateTransitionFailed, self).__init__(message)
|
|
|
|
|
|
class RootDeviceDetectionError(Exception):
|
|
"""Failed to detect the root device"""
|