Merge "Propagate AttributeErrors when lazily loading plugins"
This commit is contained in:
commit
e3a6fc27b0
@ -22,8 +22,10 @@ import sys
|
|||||||
|
|
||||||
from oslo_utils import strutils
|
from oslo_utils import strutils
|
||||||
import requests
|
import requests
|
||||||
|
import six
|
||||||
|
|
||||||
from openstackclient.api import auth
|
from openstackclient.api import auth
|
||||||
|
from openstackclient.common import exceptions
|
||||||
from openstackclient.common import session as osc_session
|
from openstackclient.common import session as osc_session
|
||||||
from openstackclient.identity import client as identity_client
|
from openstackclient.identity import client as identity_client
|
||||||
|
|
||||||
@ -45,7 +47,13 @@ class ClientCache(object):
|
|||||||
def __get__(self, instance, owner):
|
def __get__(self, instance, owner):
|
||||||
# Tell the ClientManager to login to keystone
|
# Tell the ClientManager to login to keystone
|
||||||
if self._handle is None:
|
if self._handle is None:
|
||||||
|
try:
|
||||||
self._handle = self.factory(instance)
|
self._handle = self.factory(instance)
|
||||||
|
except AttributeError as err:
|
||||||
|
# Make sure the failure propagates. Otherwise, the plugin just
|
||||||
|
# quietly isn't there.
|
||||||
|
new_err = exceptions.PluginAttributeError(err)
|
||||||
|
six.reraise(new_err.__class__, new_err, sys.exc_info()[2])
|
||||||
return self._handle
|
return self._handle
|
||||||
|
|
||||||
|
|
||||||
|
@ -24,6 +24,13 @@ class AuthorizationFailure(Exception):
|
|||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class PluginAttributeError(Exception):
|
||||||
|
"""A plugin threw an AttributeError while being lazily loaded."""
|
||||||
|
# This *must not* inherit from AttributeError;
|
||||||
|
# that would defeat the whole purpose.
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
class NoTokenLookupException(Exception):
|
class NoTokenLookupException(Exception):
|
||||||
"""This does not support looking up endpoints from an existing token."""
|
"""This does not support looking up endpoints from an existing token."""
|
||||||
pass
|
pass
|
||||||
|
@ -41,6 +41,7 @@ auth.get_options_list()
|
|||||||
|
|
||||||
class Container(object):
|
class Container(object):
|
||||||
attr = clientmanager.ClientCache(lambda x: object())
|
attr = clientmanager.ClientCache(lambda x: object())
|
||||||
|
buggy_attr = clientmanager.ClientCache(lambda x: x.foo)
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
pass
|
pass
|
||||||
@ -72,6 +73,13 @@ class TestClientCache(utils.TestCase):
|
|||||||
c = Container()
|
c = Container()
|
||||||
self.assertEqual(c.attr, c.attr)
|
self.assertEqual(c.attr, c.attr)
|
||||||
|
|
||||||
|
def test_attribute_error_propagates(self):
|
||||||
|
c = Container()
|
||||||
|
err = self.assertRaises(exc.PluginAttributeError,
|
||||||
|
getattr, c, 'buggy_attr')
|
||||||
|
self.assertNotIsInstance(err, AttributeError)
|
||||||
|
self.assertEqual("'Container' object has no attribute 'foo'", str(err))
|
||||||
|
|
||||||
|
|
||||||
class TestClientManager(utils.TestCase):
|
class TestClientManager(utils.TestCase):
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user