Add check on get_endpoint returning None
Currently we expect keystoneauth to raise on missing endpoint, but in many cases it returns None from get_endpoint instead. This causes two kinds of problems: * Raising Python exceptions in the code that is not ready to handle None (e.g. get_ironic_api_url in deploy_utils). * Passing endpoint_override=None to client constructors, thus effectively ignoring provided adapter configuration options. This change turns None into a proper error. Change-Id: I8dad2e8ef5ff39167995dfe8266a48e80a29788b
This commit is contained in:
parent
f52b386b4c
commit
b2ee05a0b1
@ -55,11 +55,10 @@ def get_client(context):
|
|||||||
if CONF.cinder.url and not CONF.cinder.endpoint_override:
|
if CONF.cinder.url and not CONF.cinder.endpoint_override:
|
||||||
adapter_opts['endpoint_override'] = CONF.cinder.url
|
adapter_opts['endpoint_override'] = CONF.cinder.url
|
||||||
|
|
||||||
adapter = keystone.get_adapter('cinder', session=session,
|
|
||||||
auth=service_auth, **adapter_opts)
|
|
||||||
# TODO(pas-ha) use versioned endpoint data to select required
|
# TODO(pas-ha) use versioned endpoint data to select required
|
||||||
# cinder api version
|
# cinder api version
|
||||||
cinder_url = adapter.get_endpoint()
|
cinder_url = keystone.get_endpoint('cinder', session=session,
|
||||||
|
auth=service_auth, **adapter_opts)
|
||||||
# TODO(pas-ha) investigate possibility of passing a user context here,
|
# TODO(pas-ha) investigate possibility of passing a user context here,
|
||||||
# similar to what neutron/glance-related code does
|
# similar to what neutron/glance-related code does
|
||||||
# NOTE(pas-ha) cinderclient has both 'connect_retries' (passed to
|
# NOTE(pas-ha) cinderclient has both 'connect_retries' (passed to
|
||||||
|
@ -74,9 +74,9 @@ def check_image_service(func):
|
|||||||
# so we can pass session and auth separately, makes things easier
|
# so we can pass session and auth separately, makes things easier
|
||||||
service_auth = keystone.get_auth('glance')
|
service_auth = keystone.get_auth('glance')
|
||||||
|
|
||||||
adapter = keystone.get_adapter('glance', session=_GLANCE_SESSION,
|
self.endpoint = keystone.get_endpoint('glance',
|
||||||
|
session=_GLANCE_SESSION,
|
||||||
auth=service_auth)
|
auth=service_auth)
|
||||||
self.endpoint = adapter.get_endpoint()
|
|
||||||
|
|
||||||
user_auth = None
|
user_auth = None
|
||||||
# NOTE(pas-ha) our ContextHook removes context.auth_token in noauth
|
# NOTE(pas-ha) our ContextHook removes context.auth_token in noauth
|
||||||
@ -296,14 +296,14 @@ class GlanceImageService(object):
|
|||||||
endpoint_url = CONF.glance.swift_endpoint_url
|
endpoint_url = CONF.glance.swift_endpoint_url
|
||||||
if not endpoint_url:
|
if not endpoint_url:
|
||||||
swift_session = swift.get_swift_session()
|
swift_session = swift.get_swift_session()
|
||||||
adapter = keystone.get_adapter('swift', session=swift_session)
|
try:
|
||||||
endpoint_url = adapter.get_endpoint()
|
endpoint_url = keystone.get_endpoint('swift',
|
||||||
|
session=swift_session)
|
||||||
if not endpoint_url:
|
except exception.CatalogNotFound:
|
||||||
raise exception.MissingParameterValue(_(
|
raise exception.MissingParameterValue(_(
|
||||||
'Swift temporary URLs require a Swift endpoint URL, but it '
|
'Swift temporary URLs require a Swift endpoint URL, '
|
||||||
'was not found in the service catalog. '
|
'but it was not found in the service catalog. You must '
|
||||||
'You must provide "swift_endpoint_url" as a config option.'))
|
'provide "swift_endpoint_url" as a config option.'))
|
||||||
|
|
||||||
# Strip /v1/AUTH_%(tenant_id)s, if present
|
# Strip /v1/AUTH_%(tenant_id)s, if present
|
||||||
endpoint_url = re.sub('/v1/AUTH_[^/]+/?$', '', endpoint_url)
|
endpoint_url = re.sub('/v1/AUTH_[^/]+/?$', '', endpoint_url)
|
||||||
|
@ -103,6 +103,25 @@ def get_adapter(group, **adapter_kwargs):
|
|||||||
**adapter_kwargs)
|
**adapter_kwargs)
|
||||||
|
|
||||||
|
|
||||||
|
def get_endpoint(group, **adapter_kwargs):
|
||||||
|
"""Get an endpoint from an adapter.
|
||||||
|
|
||||||
|
The adapter_kwargs will be passed directly to keystoneauth1 Adapter
|
||||||
|
and will override the values loaded from config.
|
||||||
|
Consult keystoneauth1 docs for available adapter options.
|
||||||
|
|
||||||
|
:param group: name of the config section to load adapter options from
|
||||||
|
:raises: CatalogNotFound if the endpoint is not found
|
||||||
|
"""
|
||||||
|
result = get_adapter(group, **adapter_kwargs).get_endpoint()
|
||||||
|
if not result:
|
||||||
|
service_type = adapter_kwargs.get('service_type', 'baremetal')
|
||||||
|
endpoint_type = adapter_kwargs.get('endpoint_type', 'internal')
|
||||||
|
raise exception.CatalogNotFound(
|
||||||
|
service_type=service_type, endpoint_type=endpoint_type)
|
||||||
|
return result
|
||||||
|
|
||||||
|
|
||||||
def get_service_auth(context, endpoint, service_auth):
|
def get_service_auth(context, endpoint, service_auth):
|
||||||
"""Create auth plugin wrapping both user and service auth.
|
"""Create auth plugin wrapping both user and service auth.
|
||||||
|
|
||||||
|
@ -76,9 +76,8 @@ def get_client(token=None, context=None):
|
|||||||
else:
|
else:
|
||||||
if CONF.neutron.url and not CONF.neutron.endpoint_override:
|
if CONF.neutron.url and not CONF.neutron.endpoint_override:
|
||||||
adapter_params['endpoint_override'] = CONF.neutron.url
|
adapter_params['endpoint_override'] = CONF.neutron.url
|
||||||
adapter = keystone.get_adapter('neutron', session=session,
|
endpoint = keystone.get_endpoint('neutron', session=session,
|
||||||
auth=service_auth, **adapter_params)
|
auth=service_auth, **adapter_params)
|
||||||
endpoint = adapter.get_endpoint()
|
|
||||||
|
|
||||||
user_auth = None
|
user_auth = None
|
||||||
if CONF.neutron.auth_type != 'none' and context.auth_token:
|
if CONF.neutron.auth_type != 'none' and context.auth_token:
|
||||||
|
@ -63,8 +63,8 @@ class SwiftAPI(object):
|
|||||||
# TODO(pas-ha) pass the context here and use token from context
|
# TODO(pas-ha) pass the context here and use token from context
|
||||||
# with service auth
|
# with service auth
|
||||||
params['session'] = session = get_swift_session()
|
params['session'] = session = get_swift_session()
|
||||||
adapter = keystone.get_adapter('swift', session=session)
|
endpoint = keystone.get_endpoint('swift', session=session)
|
||||||
params['os_options'] = {'object_storage_url': adapter.get_endpoint()}
|
params['os_options'] = {'object_storage_url': endpoint}
|
||||||
# deconstruct back session-related options
|
# deconstruct back session-related options
|
||||||
params['timeout'] = session.timeout
|
params['timeout'] = session.timeout
|
||||||
if session.verify is False:
|
if session.verify is False:
|
||||||
|
@ -108,9 +108,8 @@ def get_ironic_api_url():
|
|||||||
# NOTE(pas-ha) if both set, the new options win
|
# NOTE(pas-ha) if both set, the new options win
|
||||||
if CONF.conductor.api_url and not CONF.service_catalog.endpoint_override:
|
if CONF.conductor.api_url and not CONF.service_catalog.endpoint_override:
|
||||||
adapter_opts['endpoint_override'] = CONF.conductor.api_url
|
adapter_opts['endpoint_override'] = CONF.conductor.api_url
|
||||||
adapter = keystone.get_adapter('service_catalog', **adapter_opts)
|
|
||||||
try:
|
try:
|
||||||
ironic_api = adapter.get_endpoint()
|
ironic_api = keystone.get_endpoint('service_catalog', **adapter_opts)
|
||||||
except (exception.KeystoneFailure,
|
except (exception.KeystoneFailure,
|
||||||
exception.CatalogNotFound,
|
exception.CatalogNotFound,
|
||||||
exception.KeystoneUnauthorized) as e:
|
exception.KeystoneUnauthorized) as e:
|
||||||
|
@ -56,9 +56,8 @@ def _get_client(context):
|
|||||||
adapter_params = {}
|
adapter_params = {}
|
||||||
if CONF.inspector.service_url and not CONF.inspector.endpoint_override:
|
if CONF.inspector.service_url and not CONF.inspector.endpoint_override:
|
||||||
adapter_params['endpoint_override'] = CONF.inspector.service_url
|
adapter_params['endpoint_override'] = CONF.inspector.service_url
|
||||||
adapter = keystone.get_adapter('inspector', session=session,
|
inspector_url = keystone.get_endpoint('inspector', session=session,
|
||||||
**adapter_params)
|
**adapter_params)
|
||||||
inspector_url = adapter.get_endpoint()
|
|
||||||
# TODO(pas-ha) investigate possibility of passing user context here,
|
# TODO(pas-ha) investigate possibility of passing user context here,
|
||||||
# similar to what neutron/glance-related code does
|
# similar to what neutron/glance-related code does
|
||||||
# NOTE(pas-ha) ironic-inspector-client has no Adaper-based
|
# NOTE(pas-ha) ironic-inspector-client has no Adaper-based
|
||||||
|
@ -1315,6 +1315,17 @@ class OtherFunctionTestCase(db_base.DbTestCase):
|
|||||||
self.assertRaises(exception.InvalidParameterValue,
|
self.assertRaises(exception.InvalidParameterValue,
|
||||||
utils.get_ironic_api_url)
|
utils.get_ironic_api_url)
|
||||||
|
|
||||||
|
@mock.patch('ironic.common.keystone.get_auth')
|
||||||
|
@mock.patch.object(utils, '_get_ironic_session')
|
||||||
|
@mock.patch('ironic.common.keystone.get_adapter')
|
||||||
|
def test_get_ironic_api_url_none(self, mock_ka, mock_ks, mock_auth):
|
||||||
|
mock_sess = mock.Mock()
|
||||||
|
mock_ks.return_value = mock_sess
|
||||||
|
mock_ka.return_value.get_endpoint.return_value = None
|
||||||
|
self.config(api_url=None, group='conductor')
|
||||||
|
self.assertRaises(exception.InvalidParameterValue,
|
||||||
|
utils.get_ironic_api_url)
|
||||||
|
|
||||||
|
|
||||||
class GetSingleNicTestCase(db_base.DbTestCase):
|
class GetSingleNicTestCase(db_base.DbTestCase):
|
||||||
|
|
||||||
|
8
releasenotes/notes/api-none-cdb95e58b69a5c50.yaml
Normal file
8
releasenotes/notes/api-none-cdb95e58b69a5c50.yaml
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
---
|
||||||
|
fixes:
|
||||||
|
- |
|
||||||
|
Fixes a confusing ``AttributeError`` if an adapter returns ``None`` for
|
||||||
|
the bare metal API.
|
||||||
|
- |
|
||||||
|
Prevents the adapter configuration options from getting ignored if
|
||||||
|
a matching endpoint cannot be found. An error is now raised.
|
Loading…
Reference in New Issue
Block a user