Fix TypeError: __str__ returned non-string (type ImageRefValidationFailed)

This change contains two fixes for the same issue:
* Do not pass an instance of ImageRefValidationFailed as a message
  to an exception constructor.
* Make sure that IronicException.__str__ always returns an str,
  even when a non-string is passed as the first argument to __init__.

Change-Id: I96edb28955e64915e9d6a481634857fd27690555
Story: #2003682
Task: #26206
This commit is contained in:
Dmitry Tantsur 2019-03-04 13:07:54 +01:00
parent 3f6d4c6a78
commit 245af384ff
5 changed files with 22 additions and 9 deletions
ironic
common
drivers/modules
tests/unit
releasenotes/notes

@ -128,10 +128,12 @@ class IronicException(Exception):
def __str__(self): def __str__(self):
"""Encode to utf-8 then wsme api can consume it as well.""" """Encode to utf-8 then wsme api can consume it as well."""
if not six.PY3: value = self.__unicode__()
return six.text_type(self.args[0]).encode('utf-8') if six.PY3:
# On Python 3 unicode is the same as str
return self.args[0] return value
else:
return value.encode('utf-8')
def __unicode__(self): def __unicode__(self):
"""Return a unicode representation of the exception message.""" """Return a unicode representation of the exception message."""

@ -829,7 +829,7 @@ def validate_image_properties(ctx, deploy_info, properties):
raise exception.InvalidParameterValue(_( raise exception.InvalidParameterValue(_(
"Image %s can not be found.") % image_href) "Image %s can not be found.") % image_href)
except exception.ImageRefValidationFailed as e: except exception.ImageRefValidationFailed as e:
raise exception.InvalidParameterValue(e) raise exception.InvalidParameterValue(err=e)
missing_props = [] missing_props = []
for prop in properties: for prop in properties:

@ -32,7 +32,7 @@ class TestException(exception.IronicException):
class TestIronicException(base.TestCase): class TestIronicException(base.TestCase):
def test___init__(self): def test___str__encoding(self):
expected = b'\xc3\xa9\xe0\xaf\xb2\xe0\xbe\x84' expected = b'\xc3\xa9\xe0\xaf\xb2\xe0\xbe\x84'
if six.PY3: if six.PY3:
expected = expected.decode('utf-8') expected = expected.decode('utf-8')
@ -40,6 +40,11 @@ class TestIronicException(base.TestCase):
exc = exception.IronicException(message) exc = exception.IronicException(message)
self.assertEqual(expected, exc.__str__()) self.assertEqual(expected, exc.__str__())
def test___str__non_string(self):
exc = exception.IronicException(42)
self.assertEqual("42", exc.__str__())
self.assertEqual(u"42", exc.__unicode__())
@mock.patch.object(exception.LOG, 'error', autospec=True) @mock.patch.object(exception.LOG, 'error', autospec=True)
def test___init___invalid_kwarg(self, log_mock): def test___init___invalid_kwarg(self, log_mock):
self.config(fatal_exception_format_errors=False) self.config(fatal_exception_format_errors=False)

@ -1955,7 +1955,8 @@ class ValidateImagePropertiesTestCase(db_base.DbTestCase):
driver_internal_info=DRV_INTERNAL_INFO_DICT, driver_internal_info=DRV_INTERNAL_INFO_DICT,
) )
inst_info = utils.get_image_instance_info(node) inst_info = utils.get_image_instance_info(node)
self.assertRaises(exception.InvalidParameterValue, self.assertRaisesRegex(exception.InvalidParameterValue,
'HTTPError',
utils.validate_image_properties, self.context, utils.validate_image_properties, self.context,
inst_info, ['kernel', 'ramdisk']) inst_info, ['kernel', 'ramdisk'])

@ -0,0 +1,5 @@
---
fixes:
- |
Returns the correct error message on providing an invalid reference to
``image_source``. Previously an internal error was raised.