Add support for generalized filtering on list APIs
This patch adds generalized filtering support for these list APIs: 1. list volume 2. list backup 3. list snapshot 4. list group 5. list group-snapshot 6. list attachment 7. list message 8. get pools DocImpact APIImpact Co-Authored-By: TommyLike <tommylikehu@gmail.com> Change-Id: Icee6c22621489f93614f4adf071329d8d2115637 Partial: blueprint generalized-filtering-for-cinder-list-resource
This commit is contained in:
parent
e726321d97
commit
ff3d41b15a
@ -14,6 +14,7 @@
|
|||||||
# under the License.
|
# under the License.
|
||||||
|
|
||||||
|
|
||||||
|
import json
|
||||||
import os
|
import os
|
||||||
import re
|
import re
|
||||||
|
|
||||||
@ -39,10 +40,16 @@ api_common_opts = [
|
|||||||
help='Base URL that will be presented to users in links '
|
help='Base URL that will be presented to users in links '
|
||||||
'to the OpenStack Volume API',
|
'to the OpenStack Volume API',
|
||||||
deprecated_name='osapi_compute_link_prefix'),
|
deprecated_name='osapi_compute_link_prefix'),
|
||||||
|
cfg.StrOpt('resource_query_filters_file',
|
||||||
|
default='/etc/cinder/resource_filters.json',
|
||||||
|
help="Json file indicating user visible filter "
|
||||||
|
"parameters for list queries.",
|
||||||
|
deprecated_name='query_volume_filters'),
|
||||||
cfg.ListOpt('query_volume_filters',
|
cfg.ListOpt('query_volume_filters',
|
||||||
default=['name', 'status', 'metadata',
|
default=['name', 'status', 'metadata',
|
||||||
'availability_zone',
|
'availability_zone',
|
||||||
'bootable', 'group_id'],
|
'bootable', 'group_id'],
|
||||||
|
deprecated_for_removal=True,
|
||||||
help="Volume filter options which "
|
help="Volume filter options which "
|
||||||
"non-admin user could use to "
|
"non-admin user could use to "
|
||||||
"query volumes. Default values "
|
"query volumes. Default values "
|
||||||
@ -55,6 +62,8 @@ CONF = cfg.CONF
|
|||||||
CONF.register_opts(api_common_opts)
|
CONF.register_opts(api_common_opts)
|
||||||
|
|
||||||
LOG = logging.getLogger(__name__)
|
LOG = logging.getLogger(__name__)
|
||||||
|
_FILTERS_COLLECTION = None
|
||||||
|
FILTERING_VERSION = '3.30'
|
||||||
|
|
||||||
|
|
||||||
METADATA_TYPES = enum.Enum('METADATA_TYPES', 'user image')
|
METADATA_TYPES = enum.Enum('METADATA_TYPES', 'user image')
|
||||||
@ -399,3 +408,71 @@ def get_cluster_host(req, params, cluster_version=None):
|
|||||||
if bool(cluster_name) == bool(host):
|
if bool(cluster_name) == bool(host):
|
||||||
raise exception.InvalidInput(reason=msg)
|
raise exception.InvalidInput(reason=msg)
|
||||||
return cluster_name, host
|
return cluster_name, host
|
||||||
|
|
||||||
|
|
||||||
|
def _initialize_filters():
|
||||||
|
global _FILTERS_COLLECTION
|
||||||
|
if not _FILTERS_COLLECTION:
|
||||||
|
with open(CONF.resource_query_filters_file, 'r') as filters_file:
|
||||||
|
_FILTERS_COLLECTION = json.load(filters_file)
|
||||||
|
|
||||||
|
|
||||||
|
def get_enabled_resource_filters(resource=None):
|
||||||
|
"""Get list of configured/allowed filters for the specified resource.
|
||||||
|
|
||||||
|
This method checks resource_query_filters_file and returns dictionary
|
||||||
|
which contains the specified resource and its allowed filters:
|
||||||
|
|
||||||
|
.. code-block:: json
|
||||||
|
|
||||||
|
{
|
||||||
|
"resource": ['filter1', 'filter2', 'filter3']
|
||||||
|
}
|
||||||
|
|
||||||
|
if resource is not specified, all of the configuration will be returned,
|
||||||
|
and if the resource is not found, empty dict will be returned.
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
_initialize_filters()
|
||||||
|
if not resource:
|
||||||
|
return _FILTERS_COLLECTION
|
||||||
|
else:
|
||||||
|
return {resource: _FILTERS_COLLECTION[resource]}
|
||||||
|
except Exception:
|
||||||
|
LOG.debug("Failed to collect resource %s's filters.", resource)
|
||||||
|
return {}
|
||||||
|
|
||||||
|
|
||||||
|
def update_general_filters(context, filters, resource):
|
||||||
|
if context.is_admin:
|
||||||
|
# Allow all options
|
||||||
|
return
|
||||||
|
# Check the configured filters against those passed in resource
|
||||||
|
configured_filters = get_enabled_resource_filters(resource)
|
||||||
|
if configured_filters:
|
||||||
|
configured_filters = configured_filters[resource]
|
||||||
|
else:
|
||||||
|
configured_filters = []
|
||||||
|
invalid_filters = []
|
||||||
|
for key in filters.copy().keys():
|
||||||
|
if key not in configured_filters:
|
||||||
|
filters.pop(key)
|
||||||
|
invalid_filters.append(key)
|
||||||
|
if invalid_filters:
|
||||||
|
raise webob.exc.HTTPBadRequest(
|
||||||
|
explanation=_('Invalid filters %s are found in query '
|
||||||
|
'options.') % ','.join(invalid_filters))
|
||||||
|
|
||||||
|
|
||||||
|
def process_general_filtering(resource):
|
||||||
|
def wrapper(process_non_general_filtering):
|
||||||
|
def _decorator(*args, **kwargs):
|
||||||
|
req_version = kwargs.get('req_version')
|
||||||
|
filters = kwargs.get('filters')
|
||||||
|
context = kwargs.get('context')
|
||||||
|
if req_version.matches(FILTERING_VERSION):
|
||||||
|
update_general_filters(context, filters, resource)
|
||||||
|
else:
|
||||||
|
process_non_general_filtering(*args, **kwargs)
|
||||||
|
return _decorator
|
||||||
|
return wrapper
|
||||||
|
@ -83,16 +83,23 @@ class BackupsController(wsgi.Controller):
|
|||||||
"""Return volume search options allowed by non-admin."""
|
"""Return volume search options allowed by non-admin."""
|
||||||
return ('name', 'status', 'volume_id')
|
return ('name', 'status', 'volume_id')
|
||||||
|
|
||||||
|
@common.process_general_filtering('backup')
|
||||||
|
def _process_backup_filtering(self, context=None, filters=None,
|
||||||
|
req_version=None):
|
||||||
|
utils.remove_invalid_filter_options(context,
|
||||||
|
filters,
|
||||||
|
self._get_backup_filter_options())
|
||||||
|
|
||||||
def _get_backups(self, req, is_detail):
|
def _get_backups(self, req, is_detail):
|
||||||
"""Returns a list of backups, transformed through view builder."""
|
"""Returns a list of backups, transformed through view builder."""
|
||||||
context = req.environ['cinder.context']
|
context = req.environ['cinder.context']
|
||||||
filters = req.params.copy()
|
filters = req.params.copy()
|
||||||
|
req_version = req.api_version_request
|
||||||
marker, limit, offset = common.get_pagination_params(filters)
|
marker, limit, offset = common.get_pagination_params(filters)
|
||||||
sort_keys, sort_dirs = common.get_sort_params(filters)
|
sort_keys, sort_dirs = common.get_sort_params(filters)
|
||||||
|
|
||||||
utils.remove_invalid_filter_options(context,
|
self._process_backup_filtering(context=context, filters=filters,
|
||||||
filters,
|
req_version=req_version)
|
||||||
self._get_backup_filter_options())
|
|
||||||
|
|
||||||
if 'name' in filters:
|
if 'name' in filters:
|
||||||
filters['display_name'] = filters.pop('name')
|
filters['display_name'] = filters.pop('name')
|
||||||
|
@ -14,6 +14,7 @@
|
|||||||
|
|
||||||
"""The Scheduler Stats extension"""
|
"""The Scheduler Stats extension"""
|
||||||
|
|
||||||
|
from cinder.api import common
|
||||||
from cinder.api import extensions
|
from cinder.api import extensions
|
||||||
from cinder.api.openstack import wsgi
|
from cinder.api.openstack import wsgi
|
||||||
from cinder.api.views import scheduler_stats as scheduler_stats_view
|
from cinder.api.views import scheduler_stats as scheduler_stats_view
|
||||||
@ -37,6 +38,12 @@ class SchedulerStatsController(wsgi.Controller):
|
|||||||
self.scheduler_api = rpcapi.SchedulerAPI()
|
self.scheduler_api = rpcapi.SchedulerAPI()
|
||||||
super(SchedulerStatsController, self).__init__()
|
super(SchedulerStatsController, self).__init__()
|
||||||
|
|
||||||
|
@common.process_general_filtering('pool')
|
||||||
|
def _process_pool_filtering(self, context=None, filters=None,
|
||||||
|
req_version=None):
|
||||||
|
if not req_version.matches(GET_POOL_NAME_FILTER_MICRO_VERSION):
|
||||||
|
filters.clear()
|
||||||
|
|
||||||
def get_pools(self, req):
|
def get_pools(self, req):
|
||||||
"""List all active pools in scheduler."""
|
"""List all active pools in scheduler."""
|
||||||
context = req.environ['cinder.context']
|
context = req.environ['cinder.context']
|
||||||
@ -45,13 +52,14 @@ class SchedulerStatsController(wsgi.Controller):
|
|||||||
detail = utils.get_bool_param('detail', req.params)
|
detail = utils.get_bool_param('detail', req.params)
|
||||||
|
|
||||||
req_version = req.api_version_request
|
req_version = req.api_version_request
|
||||||
|
filters = req.params.copy()
|
||||||
|
filters.pop('detail', None)
|
||||||
|
|
||||||
if req_version.matches(GET_POOL_NAME_FILTER_MICRO_VERSION):
|
self._process_pool_filtering(context=context,
|
||||||
filters = req.params.copy()
|
filters=filters,
|
||||||
filters.pop('detail', None)
|
req_version=req_version)
|
||||||
pools = self.scheduler_api.get_pools(context, filters=filters)
|
|
||||||
else:
|
pools = self.scheduler_api.get_pools(context, filters=filters)
|
||||||
pools = self.scheduler_api.get_pools(context, filters=None)
|
|
||||||
|
|
||||||
return self._view_builder.pools(req, pools, detail)
|
return self._view_builder.pools(req, pools, detail)
|
||||||
|
|
||||||
|
@ -80,6 +80,8 @@ REST_API_VERSION_HISTORY = """
|
|||||||
* 3.27 - Add attachment API
|
* 3.27 - Add attachment API
|
||||||
* 3.28 - Add filters support to get_pools
|
* 3.28 - Add filters support to get_pools
|
||||||
* 3.29 - Add filter, sorter and pagination support in group snapshot.
|
* 3.29 - Add filter, sorter and pagination support in group snapshot.
|
||||||
|
* 3.30 - Add support for configure resource query filters, also add
|
||||||
|
``filters`` API to retrieve configured resource filters.
|
||||||
|
|
||||||
"""
|
"""
|
||||||
|
|
||||||
@ -88,7 +90,7 @@ REST_API_VERSION_HISTORY = """
|
|||||||
# minimum version of the API supported.
|
# minimum version of the API supported.
|
||||||
# Explicitly using /v1 or /v2 enpoints will still work
|
# Explicitly using /v1 or /v2 enpoints will still work
|
||||||
_MIN_API_VERSION = "3.0"
|
_MIN_API_VERSION = "3.0"
|
||||||
_MAX_API_VERSION = "3.29"
|
_MAX_API_VERSION = "3.30"
|
||||||
_LEGACY_API_VERSION1 = "1.0"
|
_LEGACY_API_VERSION1 = "1.0"
|
||||||
_LEGACY_API_VERSION2 = "2.0"
|
_LEGACY_API_VERSION2 = "2.0"
|
||||||
|
|
||||||
|
@ -292,3 +292,9 @@ user documentation.
|
|||||||
3.29
|
3.29
|
||||||
----
|
----
|
||||||
Add filter, sorter and pagination support in group snapshot.
|
Add filter, sorter and pagination support in group snapshot.
|
||||||
|
|
||||||
|
3.30
|
||||||
|
----
|
||||||
|
Add support for configure resource query filters, also add ``filters`` API
|
||||||
|
to retrieve configured resource filters.
|
||||||
|
|
||||||
|
@ -61,17 +61,25 @@ class AttachmentsController(wsgi.Controller):
|
|||||||
attachments = self._items(req)
|
attachments = self._items(req)
|
||||||
return attachment_views.ViewBuilder.list(attachments, detail=True)
|
return attachment_views.ViewBuilder.list(attachments, detail=True)
|
||||||
|
|
||||||
|
@common.process_general_filtering('attachment')
|
||||||
|
def _process_attachment_filtering(self, context=None, filters=None,
|
||||||
|
req_version=None):
|
||||||
|
utils.remove_invalid_filter_options(context, filters,
|
||||||
|
self.allowed_filters)
|
||||||
|
|
||||||
def _items(self, req):
|
def _items(self, req):
|
||||||
"""Return a list of attachments, transformed through view builder."""
|
"""Return a list of attachments, transformed through view builder."""
|
||||||
context = req.environ['cinder.context']
|
context = req.environ['cinder.context']
|
||||||
|
req_version = req.api_version_request
|
||||||
|
|
||||||
# Pop out non search_opts and create local variables
|
# Pop out non search_opts and create local variables
|
||||||
search_opts = req.GET.copy()
|
search_opts = req.GET.copy()
|
||||||
sort_keys, sort_dirs = common.get_sort_params(search_opts)
|
sort_keys, sort_dirs = common.get_sort_params(search_opts)
|
||||||
marker, limit, offset = common.get_pagination_params(search_opts)
|
marker, limit, offset = common.get_pagination_params(search_opts)
|
||||||
|
|
||||||
utils.remove_invalid_filter_options(context, search_opts,
|
self._process_attachment_filtering(context=context,
|
||||||
self.allowed_filters)
|
filters=search_opts,
|
||||||
|
req_version=req_version)
|
||||||
if search_opts.get('instance_id', None):
|
if search_opts.get('instance_id', None):
|
||||||
search_opts['instance_uuid'] = search_opts.pop('instance_id', None)
|
search_opts['instance_uuid'] = search_opts.pop('instance_id', None)
|
||||||
if context.is_admin and 'all_tenants' in search_opts:
|
if context.is_admin and 'all_tenants' in search_opts:
|
||||||
|
@ -113,6 +113,10 @@ class GroupSnapshotsController(wsgi.Controller):
|
|||||||
filters = req.params.copy()
|
filters = req.params.copy()
|
||||||
marker, limit, offset = common.get_pagination_params(filters)
|
marker, limit, offset = common.get_pagination_params(filters)
|
||||||
sort_keys, sort_dirs = common.get_sort_params(filters)
|
sort_keys, sort_dirs = common.get_sort_params(filters)
|
||||||
|
|
||||||
|
if req.api_version_request.matches(common.FILTERING_VERSION):
|
||||||
|
common.update_general_filters(context, filters, 'group_snapshot')
|
||||||
|
|
||||||
group_snapshots = self.group_snapshot_api.get_all_group_snapshots(
|
group_snapshots = self.group_snapshot_api.get_all_group_snapshots(
|
||||||
context, filters=filters, marker=marker, limit=limit,
|
context, filters=filters, marker=marker, limit=limit,
|
||||||
offset=offset, sort_keys=sort_keys, sort_dirs=sort_dirs)
|
offset=offset, sort_keys=sort_keys, sort_dirs=sort_dirs)
|
||||||
|
@ -167,6 +167,8 @@ class GroupsController(wsgi.Controller):
|
|||||||
sort_keys, sort_dirs = common.get_sort_params(filters)
|
sort_keys, sort_dirs = common.get_sort_params(filters)
|
||||||
|
|
||||||
filters.pop('list_volume', None)
|
filters.pop('list_volume', None)
|
||||||
|
if req.api_version_request.matches(common.FILTERING_VERSION):
|
||||||
|
common.update_general_filters(context, filters, 'group')
|
||||||
|
|
||||||
groups = self.group_api.get_all(
|
groups = self.group_api.get_all(
|
||||||
context, filters=filters, marker=marker, limit=limit,
|
context, filters=filters, marker=marker, limit=limit,
|
||||||
|
@ -93,6 +93,9 @@ class MessagesController(wsgi.Controller):
|
|||||||
marker, limit, offset = common.get_pagination_params(filters)
|
marker, limit, offset = common.get_pagination_params(filters)
|
||||||
sort_keys, sort_dirs = common.get_sort_params(filters)
|
sort_keys, sort_dirs = common.get_sort_params(filters)
|
||||||
|
|
||||||
|
if req.api_version_request.matches(common.FILTERING_VERSION):
|
||||||
|
common.update_general_filters(context, filters, 'message')
|
||||||
|
|
||||||
messages = self.message_api.get_all(context, filters=filters,
|
messages = self.message_api.get_all(context, filters=filters,
|
||||||
marker=marker, limit=limit,
|
marker=marker, limit=limit,
|
||||||
offset=offset,
|
offset=offset,
|
||||||
|
@ -51,35 +51,37 @@ class SnapshotsController(snapshots_v2.SnapshotsController):
|
|||||||
LOG.debug('Could not evaluate value %s, assuming string',
|
LOG.debug('Could not evaluate value %s, assuming string',
|
||||||
search_opts['metadata'])
|
search_opts['metadata'])
|
||||||
|
|
||||||
def _process_filters(self, req, context, search_opts):
|
@common.process_general_filtering('snapshot')
|
||||||
|
def _process_snapshot_filtering(self, context=None, filters=None,
|
||||||
|
req_version=None):
|
||||||
"""Formats allowed filters"""
|
"""Formats allowed filters"""
|
||||||
|
|
||||||
req_version = req.api_version_request
|
|
||||||
# if the max version is less than or same as 3.21
|
# if the max version is less than or same as 3.21
|
||||||
# metadata based filtering is not supported
|
# metadata based filtering is not supported
|
||||||
if req_version.matches(None, "3.21"):
|
if req_version.matches(None, "3.21"):
|
||||||
search_opts.pop('metadata', None)
|
filters.pop('metadata', None)
|
||||||
|
|
||||||
# Filter out invalid options
|
# Filter out invalid options
|
||||||
allowed_search_options = self._get_snapshot_filter_options()
|
allowed_search_options = self._get_snapshot_filter_options()
|
||||||
|
|
||||||
utils.remove_invalid_filter_options(context, search_opts,
|
utils.remove_invalid_filter_options(context, filters,
|
||||||
allowed_search_options)
|
allowed_search_options)
|
||||||
|
|
||||||
# process snapshot filters to appropriate formats if required
|
|
||||||
self._format_snapshot_filter_options(search_opts)
|
|
||||||
|
|
||||||
def _items(self, req, is_detail=True):
|
def _items(self, req, is_detail=True):
|
||||||
"""Returns a list of snapshots, transformed through view builder."""
|
"""Returns a list of snapshots, transformed through view builder."""
|
||||||
context = req.environ['cinder.context']
|
context = req.environ['cinder.context']
|
||||||
|
req_version = req.api_version_request
|
||||||
# Pop out non search_opts and create local variables
|
# Pop out non search_opts and create local variables
|
||||||
search_opts = req.GET.copy()
|
search_opts = req.GET.copy()
|
||||||
sort_keys, sort_dirs = common.get_sort_params(search_opts)
|
sort_keys, sort_dirs = common.get_sort_params(search_opts)
|
||||||
marker, limit, offset = common.get_pagination_params(search_opts)
|
marker, limit, offset = common.get_pagination_params(search_opts)
|
||||||
|
|
||||||
# process filters
|
# process filters
|
||||||
self._process_filters(req, context, search_opts)
|
self._process_snapshot_filtering(context=context,
|
||||||
|
filters=search_opts,
|
||||||
|
req_version=req_version)
|
||||||
|
# process snapshot filters to appropriate formats if required
|
||||||
|
self._format_snapshot_filter_options(search_opts)
|
||||||
|
|
||||||
# NOTE(thingee): v3 API allows name instead of display_name
|
# NOTE(thingee): v3 API allows name instead of display_name
|
||||||
if 'name' in search_opts:
|
if 'name' in search_opts:
|
||||||
|
@ -83,6 +83,19 @@ class VolumeController(volumes_v2.VolumeController):
|
|||||||
|
|
||||||
return webob.Response(status_int=202)
|
return webob.Response(status_int=202)
|
||||||
|
|
||||||
|
@common.process_general_filtering('volume')
|
||||||
|
def _process_volume_filtering(self, context=None, filters=None,
|
||||||
|
req_version=None):
|
||||||
|
if req_version.matches(None, "3.3"):
|
||||||
|
filters.pop('glance_metadata', None)
|
||||||
|
|
||||||
|
if req_version.matches(None, "3.9"):
|
||||||
|
filters.pop('group_id', None)
|
||||||
|
|
||||||
|
utils.remove_invalid_filter_options(
|
||||||
|
context, filters,
|
||||||
|
self._get_volume_filter_options())
|
||||||
|
|
||||||
def _get_volumes(self, req, is_detail):
|
def _get_volumes(self, req, is_detail):
|
||||||
"""Returns a list of volumes, transformed through view builder."""
|
"""Returns a list of volumes, transformed through view builder."""
|
||||||
|
|
||||||
@ -94,14 +107,9 @@ class VolumeController(volumes_v2.VolumeController):
|
|||||||
sort_keys, sort_dirs = common.get_sort_params(params)
|
sort_keys, sort_dirs = common.get_sort_params(params)
|
||||||
filters = params
|
filters = params
|
||||||
|
|
||||||
if req_version.matches(None, "3.3"):
|
self._process_volume_filtering(context=context, filters=filters,
|
||||||
filters.pop('glance_metadata', None)
|
req_version=req_version)
|
||||||
|
|
||||||
if req_version.matches(None, "3.9"):
|
|
||||||
filters.pop('group_id', None)
|
|
||||||
|
|
||||||
utils.remove_invalid_filter_options(context, filters,
|
|
||||||
self._get_volume_filter_options())
|
|
||||||
# NOTE(thingee): v2 API allows name instead of display_name
|
# NOTE(thingee): v2 API allows name instead of display_name
|
||||||
if 'name' in sort_keys:
|
if 'name' in sort_keys:
|
||||||
sort_keys[sort_keys.index('name')] = 'display_name'
|
sort_keys[sort_keys.index('name')] = 'display_name'
|
||||||
|
@ -358,6 +358,61 @@ class TestCollectionLinks(test.TestCase):
|
|||||||
should_link_exist)
|
should_link_exist)
|
||||||
|
|
||||||
|
|
||||||
|
@ddt.ddt
|
||||||
|
class GeneralFiltersTest(test.TestCase):
|
||||||
|
|
||||||
|
@ddt.data({'filters': {'volume': ['key1', 'key2']},
|
||||||
|
'resource': 'volume',
|
||||||
|
'expected': {'volume': ['key1', 'key2']}},
|
||||||
|
{'filters': {'volume': ['key1', 'key2']},
|
||||||
|
'resource': 'snapshot',
|
||||||
|
'expected': {}},
|
||||||
|
{'filters': {'volume': ['key1', 'key2']},
|
||||||
|
'resource': None,
|
||||||
|
'expected': {'volume': ['key1', 'key2']}})
|
||||||
|
@ddt.unpack
|
||||||
|
def test_get_enabled_resource_filters(self, filters, resource, expected):
|
||||||
|
common._FILTERS_COLLECTION = filters
|
||||||
|
result = common.get_enabled_resource_filters(resource)
|
||||||
|
self.assertEqual(expected, result)
|
||||||
|
|
||||||
|
@ddt.data({'filters': {'key1': 'value1'},
|
||||||
|
'is_admin': False,
|
||||||
|
'result': {'fake_resource': ['key1']},
|
||||||
|
'expected': {'key1': 'value1'}},
|
||||||
|
{'filters': {'key1': 'value1', 'key2': 'value2'},
|
||||||
|
'is_admin': False,
|
||||||
|
'result': {'fake_resource': ['key1']},
|
||||||
|
'expected': None},
|
||||||
|
{'filters': {'key1': 'value1',
|
||||||
|
'all_tenants': 'value2',
|
||||||
|
'key3': 'value3'},
|
||||||
|
'is_admin': True,
|
||||||
|
'result': {'fake_resource': []},
|
||||||
|
'expected': {'key1': 'value1',
|
||||||
|
'all_tenants': 'value2',
|
||||||
|
'key3': 'value3'}})
|
||||||
|
@ddt.unpack
|
||||||
|
@mock.patch('cinder.api.common.get_enabled_resource_filters')
|
||||||
|
def test_update_general_filters(self, mock_get, filters,
|
||||||
|
is_admin, result, expected):
|
||||||
|
class FakeContext(object):
|
||||||
|
def __init__(self, admin):
|
||||||
|
self.is_admin = admin
|
||||||
|
|
||||||
|
fake_context = FakeContext(is_admin)
|
||||||
|
mock_get.return_value = result
|
||||||
|
if expected:
|
||||||
|
common.update_general_filters(fake_context,
|
||||||
|
filters, 'fake_resource')
|
||||||
|
self.assertEqual(expected, filters)
|
||||||
|
else:
|
||||||
|
self.assertRaises(
|
||||||
|
webob.exc.HTTPBadRequest,
|
||||||
|
common.update_general_filters, fake_context,
|
||||||
|
filters, 'fake_resource')
|
||||||
|
|
||||||
|
|
||||||
@ddt.ddt
|
@ddt.ddt
|
||||||
class LinkPrefixTest(test.TestCase):
|
class LinkPrefixTest(test.TestCase):
|
||||||
|
|
||||||
|
@ -139,6 +139,19 @@ class AttachmentsAPITestCase(test.TestCase):
|
|||||||
self.controller.delete, req,
|
self.controller.delete, req,
|
||||||
self.attachment1.id)
|
self.attachment1.id)
|
||||||
|
|
||||||
|
@ddt.data('3.29', '3.30')
|
||||||
|
@mock.patch('cinder.api.common.update_general_filters')
|
||||||
|
def test_attachment_list_with_general_filter(self, version, mock_update):
|
||||||
|
url = '/v3/%s/attachments' % fake.PROJECT_ID
|
||||||
|
req = fakes.HTTPRequest.blank(url,
|
||||||
|
version=version,
|
||||||
|
use_admin_context=False)
|
||||||
|
self.controller.index(req)
|
||||||
|
|
||||||
|
if version == '3.30':
|
||||||
|
mock_update.assert_called_once_with(req.environ['cinder.context'],
|
||||||
|
mock.ANY, 'attachment')
|
||||||
|
|
||||||
@ddt.data('reserved', 'attached')
|
@ddt.data('reserved', 'attached')
|
||||||
@mock.patch.object(volume_rpcapi.VolumeAPI, 'attachment_delete')
|
@mock.patch.object(volume_rpcapi.VolumeAPI, 'attachment_delete')
|
||||||
def test_delete_attachment(self, status, mock_delete):
|
def test_delete_attachment(self, status, mock_delete):
|
||||||
|
@ -15,6 +15,8 @@
|
|||||||
|
|
||||||
"""The backups V3 api."""
|
"""The backups V3 api."""
|
||||||
|
|
||||||
|
import ddt
|
||||||
|
import mock
|
||||||
import webob
|
import webob
|
||||||
|
|
||||||
from cinder.api.openstack import api_version_request as api_version
|
from cinder.api.openstack import api_version_request as api_version
|
||||||
@ -29,6 +31,7 @@ from cinder.tests.unit import fake_constants as fake
|
|||||||
from cinder.tests.unit import utils as test_utils
|
from cinder.tests.unit import utils as test_utils
|
||||||
|
|
||||||
|
|
||||||
|
@ddt.ddt
|
||||||
class BackupsControllerAPITestCase(test.TestCase):
|
class BackupsControllerAPITestCase(test.TestCase):
|
||||||
"""Test cases for backups API."""
|
"""Test cases for backups API."""
|
||||||
|
|
||||||
@ -82,6 +85,19 @@ class BackupsControllerAPITestCase(test.TestCase):
|
|||||||
self.controller.update,
|
self.controller.update,
|
||||||
req, fake.BACKUP_ID, body)
|
req, fake.BACKUP_ID, body)
|
||||||
|
|
||||||
|
@ddt.data('3.29', '3.30')
|
||||||
|
@mock.patch('cinder.api.common.update_general_filters')
|
||||||
|
def test_backup_list_with_general_filter(self, version, mock_update):
|
||||||
|
url = '/v3/%s/backups' % fake.PROJECT_ID
|
||||||
|
req = fakes.HTTPRequest.blank(url,
|
||||||
|
version=version,
|
||||||
|
use_admin_context=False)
|
||||||
|
self.controller.index(req)
|
||||||
|
|
||||||
|
if version == '3.30':
|
||||||
|
mock_update.assert_called_once_with(req.environ['cinder.context'],
|
||||||
|
mock.ANY, 'backup')
|
||||||
|
|
||||||
def test_backup_update(self):
|
def test_backup_update(self):
|
||||||
backup = test_utils.create_backup(
|
backup = test_utils.create_backup(
|
||||||
self.ctxt,
|
self.ctxt,
|
||||||
|
@ -183,6 +183,20 @@ class GroupSnapshotsAPITestCase(test.TestCase):
|
|||||||
res_dict['group_snapshots'][0].keys())
|
res_dict['group_snapshots'][0].keys())
|
||||||
group_snapshot.destroy()
|
group_snapshot.destroy()
|
||||||
|
|
||||||
|
@ddt.data('3.29', '3.30')
|
||||||
|
@mock.patch('cinder.api.common.update_general_filters')
|
||||||
|
def test_group_snapshot_list_with_general_filter(self,
|
||||||
|
version, mock_update):
|
||||||
|
url = '/v3/%s/group_snapshots' % fake.PROJECT_ID
|
||||||
|
req = fakes.HTTPRequest.blank(url,
|
||||||
|
version=version,
|
||||||
|
use_admin_context=False)
|
||||||
|
self.controller.index(req)
|
||||||
|
|
||||||
|
if version == '3.30':
|
||||||
|
mock_update.assert_called_once_with(req.environ['cinder.context'],
|
||||||
|
mock.ANY, 'group_snapshot')
|
||||||
|
|
||||||
@ddt.data(False, True)
|
@ddt.data(False, True)
|
||||||
def test_list_group_snapshot_with_filter(self, is_detail):
|
def test_list_group_snapshot_with_filter(self, is_detail):
|
||||||
url = ('/v3/%s/group_snapshots?'
|
url = ('/v3/%s/group_snapshots?'
|
||||||
|
@ -239,6 +239,19 @@ class GroupsAPITestCase(test.TestCase):
|
|||||||
self.assertRaises(exception.GroupNotFound, self.controller.show,
|
self.assertRaises(exception.GroupNotFound, self.controller.show,
|
||||||
req, fake.WILL_NOT_BE_FOUND_ID)
|
req, fake.WILL_NOT_BE_FOUND_ID)
|
||||||
|
|
||||||
|
@ddt.data('3.29', '3.30')
|
||||||
|
@mock.patch('cinder.api.common.update_general_filters')
|
||||||
|
def test_group_list_with_general_filter(self, version, mock_update):
|
||||||
|
url = '/v3/%s/groups' % fake.PROJECT_ID
|
||||||
|
req = fakes.HTTPRequest.blank(url,
|
||||||
|
version=version,
|
||||||
|
use_admin_context=False)
|
||||||
|
self.controller.index(req)
|
||||||
|
|
||||||
|
if version == '3.30':
|
||||||
|
mock_update.assert_called_once_with(req.environ['cinder.context'],
|
||||||
|
mock.ANY, 'group')
|
||||||
|
|
||||||
def test_list_groups_json(self):
|
def test_list_groups_json(self):
|
||||||
self.group2.group_type_id = fake.GROUP_TYPE2_ID
|
self.group2.group_type_id = fake.GROUP_TYPE2_ID
|
||||||
# TODO(geguileo): One `volume_type_ids` gets sorted out make proper
|
# TODO(geguileo): One `volume_type_ids` gets sorted out make proper
|
||||||
|
@ -10,6 +10,8 @@
|
|||||||
# License for the specific language governing permissions and limitations
|
# License for the specific language governing permissions and limitations
|
||||||
# under the License.
|
# under the License.
|
||||||
|
|
||||||
|
import ddt
|
||||||
|
import mock
|
||||||
from six.moves import http_client
|
from six.moves import http_client
|
||||||
|
|
||||||
from cinder.api import extensions
|
from cinder.api import extensions
|
||||||
@ -26,6 +28,7 @@ from cinder.tests.unit.api.v3 import fakes as v3_fakes
|
|||||||
NS = '{http://docs.openstack.org/api/openstack-block-storage/3.0/content}'
|
NS = '{http://docs.openstack.org/api/openstack-block-storage/3.0/content}'
|
||||||
|
|
||||||
|
|
||||||
|
@ddt.ddt
|
||||||
class MessageApiTest(test.TestCase):
|
class MessageApiTest(test.TestCase):
|
||||||
def setUp(self):
|
def setUp(self):
|
||||||
super(MessageApiTest, self).setUp()
|
super(MessageApiTest, self).setUp()
|
||||||
@ -120,6 +123,19 @@ class MessageApiTest(test.TestCase):
|
|||||||
self.assertRaises(exception.MessageNotFound, self.controller.delete,
|
self.assertRaises(exception.MessageNotFound, self.controller.delete,
|
||||||
req, fakes.FAKE_UUID)
|
req, fakes.FAKE_UUID)
|
||||||
|
|
||||||
|
@ddt.data('3.29', '3.30')
|
||||||
|
@mock.patch('cinder.api.common.update_general_filters')
|
||||||
|
def test_message_list_with_general_filter(self, version, mock_update):
|
||||||
|
url = '/v3/%s/messages' % fakes.FAKE_UUID
|
||||||
|
req = fakes.HTTPRequest.blank(url,
|
||||||
|
version=version,
|
||||||
|
use_admin_context=False)
|
||||||
|
self.controller.index(req)
|
||||||
|
|
||||||
|
if version == '3.30':
|
||||||
|
mock_update.assert_called_once_with(req.environ['cinder.context'],
|
||||||
|
mock.ANY, 'message')
|
||||||
|
|
||||||
def test_index(self):
|
def test_index(self):
|
||||||
self.mock_object(message_api.API, 'get_all',
|
self.mock_object(message_api.API, 'get_all',
|
||||||
return_value=[v3_fakes.fake_message(fakes.FAKE_UUID)])
|
return_value=[v3_fakes.fake_message(fakes.FAKE_UUID)])
|
||||||
|
@ -179,6 +179,19 @@ class SnapshotApiTest(test.TestCase):
|
|||||||
self.assertDictEqual({"key1": "val1", "key11": "val11"}, res_dict[
|
self.assertDictEqual({"key1": "val1", "key11": "val11"}, res_dict[
|
||||||
'snapshots'][0]['metadata'])
|
'snapshots'][0]['metadata'])
|
||||||
|
|
||||||
|
@ddt.data('3.29', '3.30')
|
||||||
|
@mock.patch('cinder.api.common.update_general_filters')
|
||||||
|
def test_snapshot_list_with_general_filter(self, version, mock_update):
|
||||||
|
url = '/v3/%s/snapshots' % fake.PROJECT_ID
|
||||||
|
req = fakes.HTTPRequest.blank(url,
|
||||||
|
version=version,
|
||||||
|
use_admin_context=False)
|
||||||
|
self.controller.index(req)
|
||||||
|
|
||||||
|
if version == '3.30':
|
||||||
|
mock_update.assert_called_once_with(req.environ['cinder.context'],
|
||||||
|
mock.ANY, 'snapshot')
|
||||||
|
|
||||||
def test_snapshot_list_with_metadata_unsupported_microversion(self):
|
def test_snapshot_list_with_metadata_unsupported_microversion(self):
|
||||||
# Create snapshot with metadata key1: value1
|
# Create snapshot with metadata key1: value1
|
||||||
metadata = {"key1": "val1"}
|
metadata = {"key1": "val1"}
|
||||||
|
@ -30,6 +30,7 @@ from cinder.tests.unit.api import fakes
|
|||||||
from cinder.tests.unit.api.v2 import fakes as v2_fakes
|
from cinder.tests.unit.api.v2 import fakes as v2_fakes
|
||||||
from cinder.tests.unit.api.v2 import test_volumes as v2_test_volumes
|
from cinder.tests.unit.api.v2 import test_volumes as v2_test_volumes
|
||||||
from cinder.tests.unit import fake_constants as fake
|
from cinder.tests.unit import fake_constants as fake
|
||||||
|
from cinder import utils
|
||||||
from cinder.volume import api as volume_api
|
from cinder.volume import api as volume_api
|
||||||
from cinder.volume import api as vol_get
|
from cinder.volume import api as vol_get
|
||||||
|
|
||||||
@ -73,7 +74,7 @@ class VolumeApiTest(test.TestCase):
|
|||||||
req.content_type = 'application/json'
|
req.content_type = 'application/json'
|
||||||
req.headers = {version_header_name: 'volume 3.2'}
|
req.headers = {version_header_name: 'volume 3.2'}
|
||||||
req.environ['cinder.context'].is_admin = True
|
req.environ['cinder.context'].is_admin = True
|
||||||
req.api_version_request = api_version.max_api_version()
|
req.api_version_request = api_version.APIVersionRequest('3.29')
|
||||||
|
|
||||||
self.override_config('query_volume_filters', 'bootable')
|
self.override_config('query_volume_filters', 'bootable')
|
||||||
self.controller.index(req)
|
self.controller.index(req)
|
||||||
@ -388,6 +389,17 @@ class VolumeApiTest(test.TestCase):
|
|||||||
self.assertRaises(webob.exc.HTTPBadRequest, self.controller.create,
|
self.assertRaises(webob.exc.HTTPBadRequest, self.controller.create,
|
||||||
req, body)
|
req, body)
|
||||||
|
|
||||||
|
@ddt.data('3.29', '3.30')
|
||||||
|
@mock.patch.object(volume_api.API, 'check_volume_filters', mock.Mock())
|
||||||
|
@mock.patch.object(utils, 'add_visible_admin_metadata', mock.Mock())
|
||||||
|
@mock.patch('cinder.api.common.update_general_filters')
|
||||||
|
def test_list_volume_with_general_filter(self, version, mock_update):
|
||||||
|
req = fakes.HTTPRequest.blank('/v3/volumes', version=version)
|
||||||
|
self.controller.index(req)
|
||||||
|
if version == '3.30':
|
||||||
|
mock_update.assert_called_once_with(req.environ['cinder.context'],
|
||||||
|
mock.ANY, 'volume')
|
||||||
|
|
||||||
@ddt.data({'admin': True, 'version': '3.21'},
|
@ddt.data({'admin': True, 'version': '3.21'},
|
||||||
{'admin': False, 'version': '3.21'},
|
{'admin': False, 'version': '3.21'},
|
||||||
{'admin': True, 'version': '3.20'},
|
{'admin': True, 'version': '3.20'},
|
||||||
|
@ -134,7 +134,7 @@ class MessageApiTest(test.TestCase):
|
|||||||
req.method = 'GET'
|
req.method = 'GET'
|
||||||
req.content_type = 'application/json'
|
req.content_type = 'application/json'
|
||||||
req.headers = {version_header_name: 'volume 3.5'}
|
req.headers = {version_header_name: 'volume 3.5'}
|
||||||
req.api_version_request = api_version.max_api_version()
|
req.api_version_request = api_version.APIVersionRequest('3.29')
|
||||||
req.environ['cinder.context'].is_admin = True
|
req.environ['cinder.context'].is_admin = True
|
||||||
|
|
||||||
res = self.controller.index(req)
|
res = self.controller.index(req)
|
||||||
@ -145,7 +145,7 @@ class MessageApiTest(test.TestCase):
|
|||||||
req.method = 'GET'
|
req.method = 'GET'
|
||||||
req.content_type = 'application/json'
|
req.content_type = 'application/json'
|
||||||
req.headers = {version_header_name: 'volume 3.5'}
|
req.headers = {version_header_name: 'volume 3.5'}
|
||||||
req.api_version_request = api_version.max_api_version()
|
req.api_version_request = api_version.APIVersionRequest('3.29')
|
||||||
req.environ['cinder.context'].is_admin = True
|
req.environ['cinder.context'].is_admin = True
|
||||||
|
|
||||||
res = self.controller.index(req)
|
res = self.controller.index(req)
|
||||||
@ -248,7 +248,7 @@ class MessageApiTest(test.TestCase):
|
|||||||
req.method = 'GET'
|
req.method = 'GET'
|
||||||
req.content_type = 'application/json'
|
req.content_type = 'application/json'
|
||||||
req.headers = {version_header_name: 'volume 3.5'}
|
req.headers = {version_header_name: 'volume 3.5'}
|
||||||
req.api_version_request = api_version.max_api_version()
|
req.api_version_request = api_version.APIVersionRequest('3.29')
|
||||||
req.environ['cinder.context'].is_admin = True
|
req.environ['cinder.context'].is_admin = True
|
||||||
|
|
||||||
res = self.controller.index(req)
|
res = self.controller.index(req)
|
||||||
|
65
doc/source/man/generalized_filters.rst
Normal file
65
doc/source/man/generalized_filters.rst
Normal file
@ -0,0 +1,65 @@
|
|||||||
|
Generalized filters
|
||||||
|
===================
|
||||||
|
|
||||||
|
Background
|
||||||
|
----------
|
||||||
|
|
||||||
|
Cinder introduced generalized resource filters since Pike, it has the
|
||||||
|
same purpose as ``query_volume_filters`` option, but it's more convenient
|
||||||
|
and can be applied to more cinder resources, administrator can control the
|
||||||
|
allowed filter keys for **non-admin** user by editing the filter
|
||||||
|
configuration file. Also since this feature, cinder will raise
|
||||||
|
``400 BadRequest`` if any invalid query filter is specified.
|
||||||
|
|
||||||
|
How do I configure the filter keys?
|
||||||
|
-----------------------------------
|
||||||
|
|
||||||
|
``resource_query_filters_file`` is introduced to cinder to represent the
|
||||||
|
filter config file path, and the config file accepts the valid filter keys
|
||||||
|
for **non-admin** user with json format:
|
||||||
|
|
||||||
|
.. code-block:: json
|
||||||
|
|
||||||
|
{
|
||||||
|
"volume": ["name", "status", "metadata"]
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
the key ``volume`` (singular) here stands for the resource you want to apply and the value
|
||||||
|
accepts an list which contains the allowed filters collection, once the configuration
|
||||||
|
file is changed and API service is restarted, cinder will only recognize this filter
|
||||||
|
keys (not allowed filter keys will be dropped before filtering), **NOTE**: the default
|
||||||
|
configuration file will include all the filters we already enabled.
|
||||||
|
|
||||||
|
Which filter keys are supported?
|
||||||
|
--------------------------------
|
||||||
|
|
||||||
|
Not all the attributes are supported at present, so we add this table below to
|
||||||
|
indicate which filter keys are valid and can be used in the configuration:
|
||||||
|
|
||||||
|
+----------------+-------------------------------------------------------------------------+
|
||||||
|
| API | Valid filter keys |
|
||||||
|
+================+=========================================================================+
|
||||||
|
| | id, group_id, name, status, bootable, migration_status, metadata, host, |
|
||||||
|
| list volume | image_metadata, availability_zone, user_id, volume_type_id, project_id, |
|
||||||
|
| | size, description, replication_status, multiattach |
|
||||||
|
+----------------+-------------------------------------------------------------------------+
|
||||||
|
| | id, volume_id, user_id, project_id, status, volume_size, name, |
|
||||||
|
| list snapshot | description, volume_type_id, group_snapshot_id, metadata |
|
||||||
|
+----------------+-------------------------------------------------------------------------+
|
||||||
|
| | id, name, status, container, availability_zone, description, |
|
||||||
|
| list backup | volume_id, is_incremental, size, host, parent_id |
|
||||||
|
+----------------+-------------------------------------------------------------------------+
|
||||||
|
| | id, user_id, status, availability_zone, group_type, name, description, |
|
||||||
|
| list group | host |
|
||||||
|
+----------------+-------------------------------------------------------------------------+
|
||||||
|
| list g-snapshot| id, name, description, group_id, group_type_id, status |
|
||||||
|
+----------------+-------------------------------------------------------------------------+
|
||||||
|
| | id, volume_id, instance_id, attach_status, attach_mode, |
|
||||||
|
| list attachment| connection_info, mountpoint, attached_host |
|
||||||
|
+----------------+-------------------------------------------------------------------------+
|
||||||
|
| | id, event_id, resource_uuid, resource_type, request_id, message_level, |
|
||||||
|
| list message | project_id |
|
||||||
|
+----------------+-------------------------------------------------------------------------+
|
||||||
|
| get pools | name |
|
||||||
|
+----------------+-------------------------------------------------------------------------+
|
12
etc/cinder/resource_filters.json
Normal file
12
etc/cinder/resource_filters.json
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
{
|
||||||
|
"volume": ["name", "status", "image_metadata",
|
||||||
|
"bootable", "migration_status"],
|
||||||
|
"backup": ["name", "status", "volume_id"],
|
||||||
|
"snapshot": ["name", "status", "volume_id"],
|
||||||
|
"group": [],
|
||||||
|
"group_snapshot": ["status", "group_id"],
|
||||||
|
"attachment": ["volume_id"],
|
||||||
|
"message": ["resource_uuid", "resource_type", "event_id",
|
||||||
|
"request_id", "message_level"],
|
||||||
|
"pool": ["name"]
|
||||||
|
}
|
@ -0,0 +1,6 @@
|
|||||||
|
---
|
||||||
|
features:
|
||||||
|
- Added generalized resource filter support in
|
||||||
|
``list volume``, ``list backup``, ``list snapshot``,
|
||||||
|
``list group``, ``list group-snapshot``, ``list attachment``,
|
||||||
|
``list message`` and ``list pools`` APIs.
|
Loading…
x
Reference in New Issue
Block a user