Add validation of limit value in querying flavors and poolss

Now queues and subscriptions have a constraint that
limit arg in API must be bigger than 0.

But flavor and pool have missed it, so added this validation.

Change-Id: I9cc5146b7149adb0f7592eea15ff21e7fb3549f2
Closes-Bug: #1590182
This commit is contained in:
wanghao 2017-01-20 11:25:04 +08:00
parent f8e0d803cc
commit 8477f95d59
6 changed files with 70 additions and 4 deletions

View File

@ -325,6 +325,14 @@ class TestFlavorsMongoDB(base.V2Base):
path, capabilities = expected[4][:2] path, capabilities = expected[4][:2]
self._flavor_expect(flavor_list[0], path, self.doc['pool_group']) self._flavor_expect(flavor_list[0], path, self.doc['pool_group'])
def test_listing_error_with_invalid_limit(self):
self.simulate_delete(self.flavor_path)
query = 'limit={0}&detailed={1}'.format(0, True)
with flavors(self, 10, self.doc['pool_group']):
self.simulate_get(self.url_prefix + '/flavors', query_string=query)
self.assertEqual(falcon.HTTP_400, self.srmock.status)
def test_queue_create_works(self): def test_queue_create_works(self):
metadata = {'_flavor': self.flavor} metadata = {'_flavor': self.flavor}
self.simulate_put(self.queue_path, body=jsonutils.dumps(metadata)) self.simulate_put(self.queue_path, body=jsonutils.dumps(metadata))

View File

@ -361,3 +361,11 @@ class TestPoolsMongoDB(base.V2Base):
self.assertEqual(6, len(pool_list)) self.assertEqual(6, len(pool_list))
path, weight = expected[4][:2] path, weight = expected[4][:2]
self._pool_expect(pool_list[0], path, weight, self.doc['uri']) self._pool_expect(pool_list[0], path, weight, self.doc['uri'])
def test_listing_error_with_invalid_limit(self):
self.simulate_delete(self.pool)
query = 'limit={0}&detailed={1}'.format(0, True)
with pools(self, 10, self.doc['uri'], 'my-group'):
self.simulate_get(self.url_prefix + '/pools', query_string=query)
self.assertEqual(falcon.HTTP_400, self.srmock.status)

View File

@ -79,6 +79,12 @@ _TRANSPORT_LIMITS_OPTIONS = (
cfg.ListOpt('subscriber_types', default=['http', 'https', 'mailto', cfg.ListOpt('subscriber_types', default=['http', 'https', 'mailto',
'trust+http', 'trust+https'], 'trust+http', 'trust+https'],
help='Defines supported subscriber types.'), help='Defines supported subscriber types.'),
cfg.IntOpt('max_flavors_per_page', default=20,
help='Defines the maximum number of flavors per page.'),
cfg.IntOpt('max_pools_per_page', default=20,
help='Defines the maximum number of pools per page.'),
) )
_TRANSPORT_LIMITS_GROUP = 'transport' _TRANSPORT_LIMITS_GROUP = 'transport'
@ -585,3 +591,29 @@ class Validator(object):
:param limit_conf_name: configuration name :param limit_conf_name: configuration name
""" """
return self._limits_conf[limit_conf_name] return self._limits_conf[limit_conf_name]
def flavor_listing(self, limit=None, **kwargs):
"""Restrictions involving a list of pools.
:param limit: The expected number of flavors in the list
:param kwargs: Ignored arguments passed to storage API
:raises: ValidationFailed if the limit is exceeded
"""
uplimit = self._limits_conf.max_flavors_per_page
if limit is not None and not (0 < limit <= uplimit):
msg = _(u'Limit must be at least 1 and no greater than {0}.')
raise ValidationFailed(msg, self._limits_conf.max_flavors_per_page)
def pool_listing(self, limit=None, **kwargs):
"""Restrictions involving a list of pools.
:param limit: The expected number of flavors in the list
:param kwargs: Ignored arguments passed to storage API
:raises: ValidationFailed if the limit is exceeded
"""
uplimit = self._limits_conf.max_pools_per_page
if limit is not None and not (0 < limit <= uplimit):
msg = _(u'Limit must be at least 1 and no greater than {0}.')
raise ValidationFailed(msg, self._limits_conf.max_pools_per_page)

View File

@ -133,14 +133,16 @@ def private_endpoints(driver, conf):
if conf.pooling: if conf.pooling:
pools_controller = driver._control.pools_controller pools_controller = driver._control.pools_controller
flavors_controller = driver._control.flavors_controller flavors_controller = driver._control.flavors_controller
validate = driver._validate
catalogue.extend([ catalogue.extend([
('/pools', ('/pools',
pools.Listing(pools_controller)), pools.Listing(pools_controller, validate)),
('/pools/{pool}', ('/pools/{pool}',
pools.Resource(pools_controller)), pools.Resource(pools_controller)),
('/flavors', ('/flavors',
flavors.Listing(flavors_controller, pools_controller)), flavors.Listing(flavors_controller, pools_controller,
validate)),
('/flavors/{flavor}', ('/flavors/{flavor}',
flavors.Resource(flavors_controller, pools_controller)), flavors.Resource(flavors_controller, pools_controller)),
]) ])

View File

@ -25,6 +25,7 @@ from zaqar.i18n import _
from zaqar.storage import errors from zaqar.storage import errors
from zaqar.transport import acl from zaqar.transport import acl
from zaqar.transport import utils as transport_utils from zaqar.transport import utils as transport_utils
from zaqar.transport import validation
from zaqar.transport.wsgi import errors as wsgi_errors from zaqar.transport.wsgi import errors as wsgi_errors
from zaqar.transport.wsgi import utils as wsgi_utils from zaqar.transport.wsgi import utils as wsgi_utils
@ -37,9 +38,10 @@ class Listing(object):
:param flavors_controller: means to interact with storage :param flavors_controller: means to interact with storage
""" """
def __init__(self, flavors_controller, pools_controller): def __init__(self, flavors_controller, pools_controller, validate):
self._ctrl = flavors_controller self._ctrl = flavors_controller
self._pools_ctrl = pools_controller self._pools_ctrl = pools_controller
self._validate = validate
@decorators.TransportLog("Flavors collection") @decorators.TransportLog("Flavors collection")
@acl.enforce("flavors:get_all") @acl.enforce("flavors:get_all")
@ -69,6 +71,12 @@ class Listing(object):
request.get_param_as_int('limit', store=store) request.get_param_as_int('limit', store=store)
detailed = request.get_param_as_bool('detailed') detailed = request.get_param_as_bool('detailed')
try:
self._validate.flavor_listing(**store)
except validation.ValidationFailed as ex:
LOG.debug(ex)
raise wsgi_errors.HTTPBadRequestAPI(six.text_type(ex))
cursor = self._ctrl.list(project=project_id, **store) cursor = self._ctrl.list(project=project_id, **store)
flavors = list(next(cursor)) flavors = list(next(cursor))

View File

@ -49,6 +49,7 @@ from zaqar.storage import errors
from zaqar.storage import utils as storage_utils from zaqar.storage import utils as storage_utils
from zaqar.transport import acl from zaqar.transport import acl
from zaqar.transport import utils as transport_utils from zaqar.transport import utils as transport_utils
from zaqar.transport import validation
from zaqar.transport.wsgi import errors as wsgi_errors from zaqar.transport.wsgi import errors as wsgi_errors
from zaqar.transport.wsgi import utils as wsgi_utils from zaqar.transport.wsgi import utils as wsgi_utils
@ -61,8 +62,9 @@ class Listing(object):
:param pools_controller: means to interact with storage :param pools_controller: means to interact with storage
""" """
def __init__(self, pools_controller): def __init__(self, pools_controller, validate):
self._ctrl = pools_controller self._ctrl = pools_controller
self._validate = validate
@decorators.TransportLog("Pools collection") @decorators.TransportLog("Pools collection")
@acl.enforce("pools:get_all") @acl.enforce("pools:get_all")
@ -91,6 +93,12 @@ class Listing(object):
request.get_param_as_int('limit', store=store) request.get_param_as_int('limit', store=store)
request.get_param_as_bool('detailed', store=store) request.get_param_as_bool('detailed', store=store)
try:
self._validate.pool_listing(**store)
except validation.ValidationFailed as ex:
LOG.debug(ex)
raise wsgi_errors.HTTPBadRequestAPI(six.text_type(ex))
cursor = self._ctrl.list(**store) cursor = self._ctrl.list(**store)
pools = list(next(cursor)) pools = list(next(cursor))