d388c8a67b
python 3.4 does not have a self.message, so we need to set one up in our own base class so the conversion to string work. Change-Id: I5d963125ccb5cc29fa0563313d229046a81ca3d0
308 lines
9.4 KiB
Python
308 lines
9.4 KiB
Python
# Copyright (c) 2014 VMware, 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.
|
|
|
|
"""
|
|
Exception definitions.
|
|
"""
|
|
|
|
import logging
|
|
|
|
import six
|
|
|
|
from oslo_vmware._i18n import _, _LE, _LW
|
|
|
|
LOG = logging.getLogger(__name__)
|
|
|
|
ALREADY_EXISTS = 'AlreadyExists'
|
|
CANNOT_DELETE_FILE = 'CannotDeleteFile'
|
|
DUPLICATE_NAME = 'DuplicateName'
|
|
FILE_ALREADY_EXISTS = 'FileAlreadyExists'
|
|
FILE_FAULT = 'FileFault'
|
|
FILE_LOCKED = 'FileLocked'
|
|
FILE_NOT_FOUND = 'FileNotFound'
|
|
INVALID_POWER_STATE = 'InvalidPowerState'
|
|
INVALID_PROPERTY = 'InvalidProperty'
|
|
NO_DISK_SPACE = 'NoDiskSpace'
|
|
NO_PERMISSION = 'NoPermission'
|
|
NOT_AUTHENTICATED = 'NotAuthenticated'
|
|
SECURITY_ERROR = "SecurityError"
|
|
MANAGED_OBJECT_NOT_FOUND = 'ManagedObjectNotFound'
|
|
TASK_IN_PROGRESS = 'TaskInProgress'
|
|
TOOLS_UNAVAILABLE = 'ToolsUnavailable'
|
|
|
|
|
|
class VMwareDriverException(Exception):
|
|
"""Base oslo.vmware 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.
|
|
|
|
"""
|
|
msg_fmt = _("An unknown exception occurred.")
|
|
|
|
if six.PY2:
|
|
__str__ = lambda self: six.text_type(self).encode('utf8')
|
|
__unicode__ = lambda self: self.description
|
|
else:
|
|
__str__ = lambda self: self.description
|
|
|
|
def __init__(self, message=None, details=None, **kwargs):
|
|
|
|
if message is not None and isinstance(message, list):
|
|
# we need this to protect against developers using
|
|
# this method like VimFaultException
|
|
raise ValueError(_("exception message must not be a list"))
|
|
|
|
if details is not None and not isinstance(details, dict):
|
|
raise ValueError(_("details must be a dict"))
|
|
|
|
self.kwargs = kwargs
|
|
self.details = details
|
|
self.cause = None
|
|
|
|
if not message:
|
|
try:
|
|
message = self.msg_fmt % kwargs
|
|
except Exception:
|
|
# 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})
|
|
# at least get the core message out if something happened
|
|
message = self.msg_fmt
|
|
|
|
if six.PY3:
|
|
self.message = message
|
|
super(VMwareDriverException, self).__init__(message)
|
|
|
|
@property
|
|
def msg(self):
|
|
return self.message
|
|
|
|
@property
|
|
def description(self):
|
|
# NOTE(jecarey): self.msg and self.cause may be i18n objects
|
|
# that do not support str or concatenation, but can be used
|
|
# as replacement text.
|
|
descr = six.text_type(self.msg)
|
|
if self.cause:
|
|
descr += '\nCause: ' + six.text_type(self.cause)
|
|
return descr
|
|
|
|
|
|
class VimException(VMwareDriverException):
|
|
"""The base exception class for all VIM related exceptions."""
|
|
|
|
def __init__(self, message=None, cause=None, details=None, **kwargs):
|
|
super(VimException, self).__init__(message, details, **kwargs)
|
|
self.cause = cause
|
|
|
|
|
|
class VimSessionOverLoadException(VMwareDriverException):
|
|
"""Thrown when there is an API call overload at the VMware server."""
|
|
|
|
def __init__(self, message, cause=None):
|
|
super(VimSessionOverLoadException, self).__init__(message)
|
|
self.cause = cause
|
|
|
|
|
|
class VimConnectionException(VMwareDriverException):
|
|
"""Thrown when there is a connection problem."""
|
|
|
|
def __init__(self, message, cause=None):
|
|
super(VimConnectionException, self).__init__(message)
|
|
self.cause = cause
|
|
|
|
|
|
class VimAttributeException(VMwareDriverException):
|
|
"""Thrown when a particular attribute cannot be found."""
|
|
|
|
def __init__(self, message, cause=None):
|
|
super(VimAttributeException, self).__init__(message)
|
|
self.cause = cause
|
|
|
|
|
|
class VimFaultException(VimException):
|
|
"""Exception thrown when there are unrecognized VIM faults."""
|
|
|
|
def __init__(self, fault_list, message, cause=None, details=None):
|
|
super(VimFaultException, self).__init__(message, cause, details)
|
|
if not isinstance(fault_list, list):
|
|
raise ValueError(_("fault_list must be a list"))
|
|
self.fault_list = fault_list
|
|
|
|
@property
|
|
def description(self):
|
|
descr = VimException.description.fget(self)
|
|
if self.fault_list:
|
|
# fault_list doesn't contain non-ASCII chars, we can use str()
|
|
descr += '\nFaults: ' + str(self.fault_list)
|
|
if self.details:
|
|
# details may contain non-ASCII values
|
|
details = '{%s}' % ', '.join(["'%s': '%s'" % (k, v) for k, v in
|
|
six.iteritems(self.details)])
|
|
descr += '\nDetails: ' + details
|
|
return descr
|
|
|
|
|
|
class ImageTransferException(VMwareDriverException):
|
|
"""Thrown when there is an error during image transfer."""
|
|
|
|
def __init__(self, message, cause=None):
|
|
super(ImageTransferException, self).__init__(message)
|
|
self.cause = cause
|
|
|
|
|
|
def _print_deprecation_warning(clazz):
|
|
LOG.warn(_LW("Exception %s is deprecated, it will be removed in the "
|
|
"next release."), clazz.__name__)
|
|
|
|
|
|
class VMwareDriverConfigurationException(VMwareDriverException):
|
|
"""Base class for all configuration exceptions.
|
|
"""
|
|
msg_fmt = _("VMware Driver configuration fault.")
|
|
|
|
def __init__(self, message=None, details=None, **kwargs):
|
|
super(VMwareDriverConfigurationException, self).__init__(
|
|
message, details, **kwargs)
|
|
_print_deprecation_warning(self.__class__)
|
|
|
|
|
|
class UseLinkedCloneConfigurationFault(VMwareDriverConfigurationException):
|
|
msg_fmt = _("No default value for use_linked_clone found.")
|
|
|
|
|
|
class MissingParameter(VMwareDriverException):
|
|
msg_fmt = _("Missing parameter : %(param)s")
|
|
|
|
def __init__(self, message=None, details=None, **kwargs):
|
|
super(MissingParameter, self).__init__(message, details, **kwargs)
|
|
_print_deprecation_warning(self.__class__)
|
|
|
|
|
|
class AlreadyExistsException(VimException):
|
|
msg_fmt = _("Resource already exists.")
|
|
code = 409
|
|
|
|
|
|
class CannotDeleteFileException(VimException):
|
|
msg_fmt = _("Cannot delete file.")
|
|
code = 403
|
|
|
|
|
|
class FileAlreadyExistsException(VimException):
|
|
msg_fmt = _("File already exists.")
|
|
code = 409
|
|
|
|
|
|
class FileFaultException(VimException):
|
|
msg_fmt = _("File fault.")
|
|
code = 409
|
|
|
|
|
|
class FileLockedException(VimException):
|
|
msg_fmt = _("File locked.")
|
|
code = 403
|
|
|
|
|
|
class FileNotFoundException(VimException):
|
|
msg_fmt = _("File not found.")
|
|
code = 404
|
|
|
|
|
|
class InvalidPowerStateException(VimException):
|
|
msg_fmt = _("Invalid power state.")
|
|
code = 409
|
|
|
|
|
|
class InvalidPropertyException(VimException):
|
|
msg_fmt = _("Invalid property.")
|
|
code = 400
|
|
|
|
|
|
class NoPermissionException(VimException):
|
|
msg_fmt = _("No Permission.")
|
|
code = 403
|
|
|
|
|
|
class NotAuthenticatedException(VimException):
|
|
msg_fmt = _("Not Authenticated.")
|
|
code = 403
|
|
|
|
|
|
class TaskInProgress(VimException):
|
|
msg_fmt = _("Entity has another operation in process.")
|
|
|
|
|
|
class DuplicateName(VimException):
|
|
msg_fmt = _("Duplicate name.")
|
|
|
|
|
|
class NoDiskSpaceException(VimException):
|
|
msg_fmt = _("Insufficient disk space.")
|
|
|
|
|
|
class ToolsUnavailableException(VimException):
|
|
msg_fmt = _("VMware Tools is not running.")
|
|
|
|
|
|
class ManagedObjectNotFoundException(VimException):
|
|
msg_fmt = _("Managed object not found.")
|
|
code = 404
|
|
|
|
|
|
# Populate the fault registry with the exceptions that have
|
|
# special treatment.
|
|
_fault_classes_registry = {
|
|
ALREADY_EXISTS: AlreadyExistsException,
|
|
CANNOT_DELETE_FILE: CannotDeleteFileException,
|
|
DUPLICATE_NAME: DuplicateName,
|
|
FILE_ALREADY_EXISTS: FileAlreadyExistsException,
|
|
FILE_FAULT: FileFaultException,
|
|
FILE_LOCKED: FileLockedException,
|
|
FILE_NOT_FOUND: FileNotFoundException,
|
|
INVALID_POWER_STATE: InvalidPowerStateException,
|
|
INVALID_PROPERTY: InvalidPropertyException,
|
|
MANAGED_OBJECT_NOT_FOUND: ManagedObjectNotFoundException,
|
|
NO_DISK_SPACE: NoDiskSpaceException,
|
|
NO_PERMISSION: NoPermissionException,
|
|
NOT_AUTHENTICATED: NotAuthenticatedException,
|
|
TASK_IN_PROGRESS: TaskInProgress,
|
|
TOOLS_UNAVAILABLE: ToolsUnavailableException,
|
|
}
|
|
|
|
|
|
def get_fault_class(name):
|
|
"""Get a named subclass of VimException."""
|
|
name = str(name)
|
|
fault_class = _fault_classes_registry.get(name)
|
|
if not fault_class:
|
|
LOG.debug('Fault %s not matched.', name)
|
|
return fault_class
|
|
|
|
|
|
def register_fault_class(name, exception):
|
|
fault_class = _fault_classes_registry.get(name)
|
|
if not issubclass(exception, VimException):
|
|
raise TypeError(_("exception should be a subclass of "
|
|
"VimException"))
|
|
if fault_class:
|
|
LOG.debug('Overriding exception for %s', name)
|
|
_fault_classes_registry[name] = exception
|