Merge "Pagination support for server list API"

This commit is contained in:
Zuul 2017-10-28 16:55:05 +00:00 committed by Gerrit Code Review
commit 8d2d165755
8 changed files with 104 additions and 9 deletions

View File

@ -122,12 +122,48 @@ image_query:
in: query in: query
required: false required: false
type: string type: string
limit:
description: |
Requests a page size of items. Returns a number of items up to a limit value.
Use the ``limit`` parameter to make an initial limited request and use the ID
of the last-seen item from the response as the ``marker`` parameter value in a
subsequent limited request.
in: query
required: false
type: integer
marker:
description: |
The ID of the last-seen item. Use the ``limit`` parameter to make an initial limited
request and use the ID of the last-seen item from the response as the ``marker``
parameter value in a subsequent limited request.
in: query
required: false
type: string
server_name_query: server_name_query:
description: | description: |
Filters the server list by name. Users can filter by prefix of server's name. Filters the server list by name. Users can filter by prefix of server's name.
in: query in: query
required: false required: false
type: string type: string
sort_dir:
description: |
Sort direction. A valid value is ``asc`` (ascending) or ``desc`` (descending).
Default is ``asc``. You can specify multiple pairs of sort key and sort direction
query parameters. If you omit the sort direction in a pair, the API uses the natural
sorting direction of the direction of the server ``sort_key`` attribute.
in: query
required: false
type: string
sort_key:
description: |
Sorts the response by the this attribute value.
Default is ``id``. You can specify multiple pairs of sort key and
sort direction query parameters. If you omit the sort direction in
a pair, the API uses the natural sorting direction of the server
attribute that is provided as the ``sort_key``.
in: query
required: false
type: string
status_query: status_query:
description: | description: |
Filters the server list by the server's status. Filters the server list by the server's status.

View File

@ -157,6 +157,10 @@ Request
- flavor_name: flavor_name_query - flavor_name: flavor_name_query
- image_uuid: image_query - image_uuid: image_query
- ip: fixed_ip_query - ip: fixed_ip_query
- limit: limit
- marker: marker
- sort_key: sort_key
- sort_dir: sort_dir
- all_tenants: all_tenants - all_tenants: all_tenants
- fields: fields - fields: fields
@ -202,6 +206,10 @@ Request
- flavor_name: flavor_name_query - flavor_name: flavor_name_query
- image_uuid: image_query - image_uuid: image_query
- ip: fixed_ip_query - ip: fixed_ip_query
- limit: limit
- marker: marker
- sort_key: sort_key
- sort_dir: sort_dir
- all_tenants: all_tenants - all_tenants: all_tenants

View File

@ -568,12 +568,20 @@ class ServerController(ServerControllerBase):
def _get_server_collection(self, name=None, status=None, def _get_server_collection(self, name=None, status=None,
flavor_uuid=None, flavor_name=None, flavor_uuid=None, flavor_name=None,
image_uuid=None, ip=None, image_uuid=None, ip=None,
limit=None, marker=None,
sort_key=None, sort_dir=None,
all_tenants=None, fields=None): all_tenants=None, fields=None):
context = pecan.request.context context = pecan.request.context
project_only = True project_only = True
if context.is_admin and all_tenants: if context.is_admin and all_tenants:
project_only = False project_only = False
limit = api_utils.validate_limit(limit)
marker_obj = None
if marker:
marker_obj = objects.Server.get(pecan.request.context, marker)
filters = {} filters = {}
if name: if name:
filters['name'] = name filters['name'] = name
@ -587,6 +595,8 @@ class ServerController(ServerControllerBase):
filters['image_uuid'] = image_uuid filters['image_uuid'] = image_uuid
servers = objects.Server.list(pecan.request.context, servers = objects.Server.list(pecan.request.context,
limit, marker_obj,
sort_key=sort_key, sort_dir=sort_dir,
project_only=project_only, project_only=project_only,
filters=filters) filters=filters)
if ip: if ip:
@ -617,10 +627,12 @@ class ServerController(ServerControllerBase):
@expose.expose(ServerCollection, wtypes.text, wtypes.text, @expose.expose(ServerCollection, wtypes.text, wtypes.text,
types.uuid, wtypes.text, types.uuid, wtypes.text, types.uuid, wtypes.text, types.uuid, wtypes.text,
types.listtype, types.boolean) int, types.uuid, types.listtype, wtypes.text,
wtypes.text, types.boolean)
def get_all(self, name=None, status=None, def get_all(self, name=None, status=None,
flavor_uuid=None, flavor_name=None, image_uuid=None, ip=None, flavor_uuid=None, flavor_name=None, image_uuid=None, ip=None,
fields=None, all_tenants=None): limit=None, marker=None, fields=None, sort_key='id',
sort_dir='asc', all_tenants=None):
"""Retrieve a list of server. """Retrieve a list of server.
:param fields: Optional, a list with a specified set of fields :param fields: Optional, a list with a specified set of fields
@ -635,6 +647,8 @@ class ServerController(ServerControllerBase):
return self._get_server_collection(name, status, return self._get_server_collection(name, status,
flavor_uuid, flavor_name, flavor_uuid, flavor_name,
image_uuid, ip, image_uuid, ip,
limit, marker,
sort_key, sort_dir,
all_tenants=all_tenants, all_tenants=all_tenants,
fields=fields) fields=fields)
@ -654,9 +668,11 @@ class ServerController(ServerControllerBase):
@expose.expose(ServerCollection, wtypes.text, wtypes.text, @expose.expose(ServerCollection, wtypes.text, wtypes.text,
types.uuid, wtypes.text, types.uuid, wtypes.text, types.uuid, wtypes.text, types.uuid, wtypes.text,
int, types.uuid, wtypes.text, wtypes.text,
types.boolean) types.boolean)
def detail(self, name=None, status=None, def detail(self, name=None, status=None,
flavor_uuid=None, flavor_name=None, image_uuid=None, ip=None, flavor_uuid=None, flavor_name=None, image_uuid=None, ip=None,
limit=None, marker=None, sort_key='id', sort_dir='asc',
all_tenants=None): all_tenants=None):
"""Retrieve detail of a list of servers.""" """Retrieve detail of a list of servers."""
# /detail should only work against collections # /detail should only work against collections
@ -668,6 +684,8 @@ class ServerController(ServerControllerBase):
return self._get_server_collection(name, status, return self._get_server_collection(name, status,
flavor_uuid, flavor_name, flavor_uuid, flavor_name,
image_uuid, ip, image_uuid, ip,
limit, marker,
sort_key, sort_dir,
all_tenants=all_tenants) all_tenants=all_tenants)
@policy.authorize_wsgi("mogan:server", "create", False) @policy.authorize_wsgi("mogan:server", "create", False)

View File

@ -31,7 +31,6 @@ JSONPATCH_EXCEPTIONS = (jsonpatch.JsonPatchException,
def validate_limit(limit): def validate_limit(limit):
if limit is None: if limit is None:
return CONF.api.max_limit return CONF.api.max_limit
if limit <= 0: if limit <= 0:
raise wsme.exc.ClientSideError(_("Limit must be positive")) raise wsme.exc.ClientSideError(_("Limit must be positive"))

View File

@ -72,7 +72,8 @@ class Connection(object):
"""Get server by name.""" """Get server by name."""
@abc.abstractmethod @abc.abstractmethod
def server_get_all(self, context, project_only, filters=None): def server_get_all(self, context, project_only, limit=None, marker=None,
sort_key=None, sort_dir=None, filters=None):
"""Get all servers.""" """Get all servers."""
@abc.abstractmethod @abc.abstractmethod

View File

@ -84,6 +84,24 @@ def model_query(context, model, *args, **kwargs):
return query return query
def _paginate_query(context, model, limit=None, marker=None, sort_key=None,
sort_dir=None, query=None):
if not query:
query = model_query(context, model)
sort_keys = ['id']
if sort_key and sort_key not in sort_keys:
sort_keys.insert(0, sort_key)
try:
query = sqlalchemyutils.paginate_query(query, model, limit, sort_keys,
marker=marker,
sort_dir=sort_dir)
except db_exc.InvalidSortKey:
raise exception.InvalidParameterValue(
_('The sort_key value "%(key)s" is an invalid field for sorting')
% {'key': sort_key})
return query.all()
def add_identity_filter(query, value): def add_identity_filter(query, value):
"""Adds an identity filter to a query. """Adds an identity filter to a query.
@ -254,11 +272,13 @@ class Connection(api.Connection):
except NoResultFound: except NoResultFound:
raise exception.ServerNotFound(server=server_id) raise exception.ServerNotFound(server=server_id)
def server_get_all(self, context, project_only, filters=None): def server_get_all(self, context, project_only, limit=None, marker=None,
sort_key=None, sort_dir=None, filters=None):
query = model_query(context, models.Server, query = model_query(context, models.Server,
project_only=project_only) project_only=project_only)
query = self._add_servers_filters(context, query, filters) query = self._add_servers_filters(context, query, filters)
return query.all() return _paginate_query(context, models.Server, limit, marker,
sort_key, sort_dir, query)
@oslo_db_api.retry_on_deadlock @oslo_db_api.retry_on_deadlock
def server_destroy(self, context, server_id): def server_destroy(self, context, server_id):

View File

@ -133,10 +133,16 @@ class Server(base.MoganObject, object_base.VersionedObjectDictCompat):
return data return data
@classmethod @classmethod
def list(cls, context, project_only=False, filters=None): def list(cls, context, limit=None, marker=None, sort_key=None,
sort_dir=None, project_only=False,
filters=None):
"""Return a list of Server objects.""" """Return a list of Server objects."""
db_servers = cls.dbapi.server_get_all(context, db_servers = cls.dbapi.server_get_all(context,
project_only=project_only, project_only=project_only,
limit=limit,
marker=marker,
sort_key=sort_key,
sort_dir=sort_dir,
filters=filters) filters=filters)
return Server._from_db_object_list(db_servers, cls, context) return Server._from_db_object_list(db_servers, cls, context)

View File

@ -47,9 +47,16 @@ class TestServerObject(base.DbTestCase):
mock_server_get_all.return_value = [self.fake_server] mock_server_get_all.return_value = [self.fake_server]
project_only = False project_only = False
filters = None filters = None
servers = objects.Server.list(self.context, project_only, filters) marker = None
limit = None
sort_key = None
sort_dir = None
servers = objects.Server.list(self.context, limit, marker,
sort_key, sort_dir,
project_only, filters)
mock_server_get_all.assert_called_once_with( mock_server_get_all.assert_called_once_with(
self.context, project_only, filters) self.context, project_only, limit, marker, sort_key, sort_dir,
filters)
self.assertIsInstance(servers[0], objects.Server) self.assertIsInstance(servers[0], objects.Server)
self.assertEqual(self.context, servers[0]._context) self.assertEqual(self.context, servers[0]._context)