Merge "Pagination support for server list API"
This commit is contained in:
commit
8d2d165755
@ -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.
|
||||||
|
@ -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
|
||||||
|
|
||||||
|
|
||||||
|
@ -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)
|
||||||
|
@ -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"))
|
||||||
|
|
||||||
|
@ -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
|
||||||
|
@ -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):
|
||||||
|
@ -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)
|
||||||
|
|
||||||
|
@ -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)
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user