Adds **params to v3 list_endpoints

This patch makes the following changes:
  - Adds **params to list_endpoints since identity v3 endpoints
    can be filtered by "interface" and "service_id" [0]
  - Enhance v3 endpoint API tests to check that filtering by
    such params works correctly.

[0] https://developer.openstack.org/api-ref/identity/v3/#list-endpoints?expanded=list-endpoints-detail

Change-Id: I133485db6f9a13d71595d43547f0dc9ea6ca0584
This commit is contained in:
Felipe Monteiro 2017-04-29 20:44:19 +01:00
parent 8fde5658a8
commit 63444d61a7
4 changed files with 101 additions and 37 deletions

View File

@ -0,0 +1,6 @@
---
features:
- |
The ``list_endpoints`` method of the v3 ``EndPointsClient`` class now has
an additional ``**params`` argument that enables passing additional
information in the query string of the HTTP request.

View File

@ -29,56 +29,93 @@ class EndPointsTestJSON(base.BaseIdentityV3AdminTest):
def resource_setup(cls): def resource_setup(cls):
super(EndPointsTestJSON, cls).resource_setup() super(EndPointsTestJSON, cls).resource_setup()
cls.service_ids = list() cls.service_ids = list()
s_name = data_utils.rand_name('service')
s_type = data_utils.rand_name('type') # Create endpoints so as to use for LIST and GET test cases
s_description = data_utils.rand_name('description') interfaces = ['public', 'internal']
cls.setup_endpoint_ids = list()
for i in range(2):
cls._create_service()
region = data_utils.rand_name('region')
url = data_utils.rand_url()
endpoint = cls.client.create_endpoint(
service_id=cls.service_ids[i], interface=interfaces[i],
url=url, region=region, enabled=True)['endpoint']
cls.setup_endpoint_ids.append(endpoint['id'])
@classmethod
def _create_service(cls, s_name=None, s_type=None, s_description=None):
if s_name is None:
s_name = data_utils.rand_name('service')
if s_type is None:
s_type = data_utils.rand_name('type')
if s_description is None:
s_description = data_utils.rand_name('description')
service_data = ( service_data = (
cls.services_client.create_service(name=s_name, type=s_type, cls.services_client.create_service(name=s_name, type=s_type,
description=s_description)) description=s_description))
cls.service_id = service_data['service']['id'] service = service_data['service']
cls.service_ids.append(cls.service_id) cls.service_ids.append(service['id'])
# Create endpoints so as to use for LIST and GET test cases return service
cls.setup_endpoints = list()
for _ in range(2):
region = data_utils.rand_name('region')
url = data_utils.rand_url()
interface = 'public'
endpoint = cls.client.create_endpoint(service_id=cls.service_id,
interface=interface,
url=url, region=region,
enabled=True)['endpoint']
cls.setup_endpoints.append(endpoint)
@classmethod @classmethod
def resource_cleanup(cls): def resource_cleanup(cls):
for e in cls.setup_endpoints: for e in cls.setup_endpoint_ids:
cls.client.delete_endpoint(e['id']) cls.client.delete_endpoint(e)
for s in cls.service_ids: for s in cls.service_ids:
cls.services_client.delete_service(s) cls.services_client.delete_service(s)
super(EndPointsTestJSON, cls).resource_cleanup() super(EndPointsTestJSON, cls).resource_cleanup()
@decorators.idempotent_id('c19ecf90-240e-4e23-9966-21cee3f6a618') @decorators.idempotent_id('c19ecf90-240e-4e23-9966-21cee3f6a618')
def test_list_endpoints(self): def test_list_endpoints(self):
# Get a list of endpoints # Get the list of all the endpoints.
fetched_endpoints = self.client.list_endpoints()['endpoints'] fetched_endpoints = self.client.list_endpoints()['endpoints']
# Asserting LIST endpoints fetched_endpoint_ids = [e['id'] for e in fetched_endpoints]
# Check that all the created endpoints are present in
# "fetched_endpoints".
missing_endpoints =\ missing_endpoints =\
[e for e in self.setup_endpoints if e not in fetched_endpoints] [e for e in self.setup_endpoint_ids
self.assertEmpty(missing_endpoints, if e not in fetched_endpoint_ids]
self.assertEqual(0, len(missing_endpoints),
"Failed to find endpoint %s in fetched list" % "Failed to find endpoint %s in fetched list" %
', '.join(str(e) for e in missing_endpoints)) ', '.join(str(e) for e in missing_endpoints))
# Check that filtering endpoints by service_id works.
fetched_endpoints_for_service = self.client.list_endpoints(
service_id=self.service_ids[0])['endpoints']
fetched_endpoints_for_alt_service = self.client.list_endpoints(
service_id=self.service_ids[1])['endpoints']
# Assert that both filters returned the correct result.
self.assertEqual(1, len(fetched_endpoints_for_service))
self.assertEqual(1, len(fetched_endpoints_for_alt_service))
self.assertEqual(set(self.setup_endpoint_ids),
set([fetched_endpoints_for_service[0]['id'],
fetched_endpoints_for_alt_service[0]['id']]))
# Check that filtering endpoints by interface works.
fetched_public_endpoints = self.client.list_endpoints(
interface='public')['endpoints']
fetched_internal_endpoints = self.client.list_endpoints(
interface='internal')['endpoints']
# Check that the expected endpoint_id is present per filter. [0] is
# public and [1] is internal.
self.assertIn(self.setup_endpoint_ids[0],
[e['id'] for e in fetched_public_endpoints])
self.assertIn(self.setup_endpoint_ids[1],
[e['id'] for e in fetched_internal_endpoints])
@decorators.idempotent_id('0e2446d2-c1fd-461b-a729-b9e73e3e3b37') @decorators.idempotent_id('0e2446d2-c1fd-461b-a729-b9e73e3e3b37')
def test_create_list_show_delete_endpoint(self): def test_create_list_show_delete_endpoint(self):
region = data_utils.rand_name('region') region = data_utils.rand_name('region')
url = data_utils.rand_url() url = data_utils.rand_url()
interface = 'public' interface = 'public'
endpoint = self.client.create_endpoint(service_id=self.service_id, endpoint = self.client.create_endpoint(service_id=self.service_ids[0],
interface=interface, interface=interface,
url=url, region=region, url=url, region=region,
enabled=True)['endpoint'] enabled=True)['endpoint']
self.setup_endpoints.append(endpoint) self.setup_endpoint_ids.append(endpoint['id'])
# Asserting Create Endpoint response body # Asserting Create Endpoint response body
self.assertIn('id', endpoint) self.assertIn('id', endpoint)
self.assertEqual(region, endpoint['region']) self.assertEqual(region, endpoint['region'])
@ -93,7 +130,7 @@ class EndPointsTestJSON(base.BaseIdentityV3AdminTest):
fetched_endpoint = ( fetched_endpoint = (
self.client.show_endpoint(endpoint['id'])['endpoint']) self.client.show_endpoint(endpoint['id'])['endpoint'])
# Asserting if the attributes of endpoint are the same # Asserting if the attributes of endpoint are the same
self.assertEqual(self.service_id, fetched_endpoint['service_id']) self.assertEqual(self.service_ids[0], fetched_endpoint['service_id'])
self.assertEqual(interface, fetched_endpoint['interface']) self.assertEqual(interface, fetched_endpoint['interface'])
self.assertEqual(url, fetched_endpoint['url']) self.assertEqual(url, fetched_endpoint['url'])
self.assertEqual(region, fetched_endpoint['region']) self.assertEqual(region, fetched_endpoint['region'])
@ -101,7 +138,7 @@ class EndPointsTestJSON(base.BaseIdentityV3AdminTest):
# Deleting the endpoint created in this method # Deleting the endpoint created in this method
self.client.delete_endpoint(endpoint['id']) self.client.delete_endpoint(endpoint['id'])
self.setup_endpoints.remove(endpoint) self.setup_endpoint_ids.remove(endpoint['id'])
# Checking whether endpoint is deleted successfully # Checking whether endpoint is deleted successfully
fetched_endpoints = self.client.list_endpoints()['endpoints'] fetched_endpoints = self.client.list_endpoints()['endpoints']
@ -117,7 +154,7 @@ class EndPointsTestJSON(base.BaseIdentityV3AdminTest):
url1 = data_utils.rand_url() url1 = data_utils.rand_url()
interface1 = 'public' interface1 = 'public'
endpoint_for_update = ( endpoint_for_update = (
self.client.create_endpoint(service_id=self.service_id, self.client.create_endpoint(service_id=self.service_ids[0],
interface=interface1, interface=interface1,
url=url1, region=region1, url=url1, region=region1,
enabled=True)['endpoint']) enabled=True)['endpoint'])
@ -126,11 +163,8 @@ class EndPointsTestJSON(base.BaseIdentityV3AdminTest):
s_name = data_utils.rand_name('service') s_name = data_utils.rand_name('service')
s_type = data_utils.rand_name('type') s_type = data_utils.rand_name('type')
s_description = data_utils.rand_name('description') s_description = data_utils.rand_name('description')
service2 = ( service2 = self._create_service(s_name=s_name, s_type=s_type,
self.services_client.create_service(name=s_name, type=s_type, s_description=s_description)
description=s_description))
service2 = service2['service']
self.service_ids.append(service2['id'])
# Updating endpoint with new values # Updating endpoint with new values
region2 = data_utils.rand_name('region') region2 = data_utils.rand_name('region')
url2 = data_utils.rand_url() url2 = data_utils.rand_url()

View File

@ -18,6 +18,7 @@ https://developer.openstack.org/api-ref/identity/v3/index.html#service-catalog-a
""" """
from oslo_serialization import jsonutils as json from oslo_serialization import jsonutils as json
from six.moves.urllib import parse as urllib
from tempest.lib.common import rest_client from tempest.lib.common import rest_client
@ -25,9 +26,17 @@ from tempest.lib.common import rest_client
class EndPointsClient(rest_client.RestClient): class EndPointsClient(rest_client.RestClient):
api_version = "v3" api_version = "v3"
def list_endpoints(self): def list_endpoints(self, **params):
"""GET endpoints.""" """List endpoints.
resp, body = self.get('endpoints')
For a full list of available parameters, please refer to the official
API reference:
http://developer.openstack.org/api-ref/identity/v3/#list-endpoints
"""
url = 'endpoints'
if params:
url += '?%s' % urllib.urlencode(params)
resp, body = self.get(url)
self.expected_success(200, resp.status) self.expected_success(200, resp.status)
body = json.loads(body) body = json.loads(body)
return rest_client.ResponseBody(resp, body) return rest_client.ResponseBody(resp, body)

View File

@ -53,6 +53,8 @@ class TestEndpointsClient(base.BaseServiceTest):
] ]
} }
FAKE_SERVICE_ID = "a4dc5060-f757-4662-b658-edd2aefbb41d"
def setUp(self): def setUp(self):
super(TestEndpointsClient, self).setUp() super(TestEndpointsClient, self).setUp()
fake_auth = fake_auth_provider.FakeAuthProvider() fake_auth = fake_auth_provider.FakeAuthProvider()
@ -72,12 +74,15 @@ class TestEndpointsClient(base.BaseServiceTest):
adminurl="https://compute.north.internal.com/v1", adminurl="https://compute.north.internal.com/v1",
internalurl="https://compute.north.internal.com/v1") internalurl="https://compute.north.internal.com/v1")
def _test_list_endpoints(self, bytes_body=False): def _test_list_endpoints(self, bytes_body=False, mock_args='endpoints',
**params):
self.check_service_client_function( self.check_service_client_function(
self.client.list_endpoints, self.client.list_endpoints,
'tempest.lib.common.rest_client.RestClient.get', 'tempest.lib.common.rest_client.RestClient.get',
self.FAKE_LIST_ENDPOINTS, self.FAKE_LIST_ENDPOINTS,
bytes_body) bytes_body,
mock_args=[mock_args],
**params)
def test_create_endpoint_with_str_body(self): def test_create_endpoint_with_str_body(self):
self._test_create_endpoint() self._test_create_endpoint()
@ -91,6 +96,16 @@ class TestEndpointsClient(base.BaseServiceTest):
def test_list_endpoints_with_bytes_body(self): def test_list_endpoints_with_bytes_body(self):
self._test_list_endpoints(bytes_body=True) self._test_list_endpoints(bytes_body=True)
def test_list_endpoints_with_params(self):
# Run the test separately for each param, to avoid assertion error
# resulting from randomized params order.
mock_args = 'endpoints?service_id=%s' % self.FAKE_SERVICE_ID
self._test_list_endpoints(mock_args=mock_args,
service_id=self.FAKE_SERVICE_ID)
mock_args = 'endpoints?interface=public'
self._test_list_endpoints(mock_args=mock_args, interface='public')
def test_delete_endpoint(self): def test_delete_endpoint(self):
self.check_service_client_function( self.check_service_client_function(
self.client.delete_endpoint, self.client.delete_endpoint,