Merge "Don't query db if criteria longer than col length"
This commit is contained in:
commit
a637ebcbc4
@ -30,6 +30,10 @@ class Hints(object):
|
|||||||
accessed publicly. Also it contains a dict called limit, which will
|
accessed publicly. Also it contains a dict called limit, which will
|
||||||
indicate the amount of data we want to limit our listing to.
|
indicate the amount of data we want to limit our listing to.
|
||||||
|
|
||||||
|
If the filter is discovered to never match, then `cannot_match` can be set
|
||||||
|
to indicate that there will not be any matches and the backend work can be
|
||||||
|
short-circuited.
|
||||||
|
|
||||||
Each filter term consists of:
|
Each filter term consists of:
|
||||||
|
|
||||||
* ``name``: the name of the attribute being matched
|
* ``name``: the name of the attribute being matched
|
||||||
@ -44,6 +48,7 @@ class Hints(object):
|
|||||||
def __init__(self):
|
def __init__(self):
|
||||||
self.limit = None
|
self.limit = None
|
||||||
self.filters = list()
|
self.filters = list()
|
||||||
|
self.cannot_match = False
|
||||||
|
|
||||||
def add_filter(self, name, value, comparator='equals',
|
def add_filter(self, name, value, comparator='equals',
|
||||||
case_sensitive=False):
|
case_sensitive=False):
|
||||||
|
@ -239,6 +239,39 @@ def truncated(f):
|
|||||||
return wrapper
|
return wrapper
|
||||||
|
|
||||||
|
|
||||||
|
class _WontMatch(Exception):
|
||||||
|
"""Raised to indicate that the filter won't match.
|
||||||
|
|
||||||
|
This is raised to short-circuit the computation of the filter as soon as
|
||||||
|
it's discovered that the filter requested isn't going to match anything.
|
||||||
|
|
||||||
|
A filter isn't going to match anything if the value is too long for the
|
||||||
|
field, for example.
|
||||||
|
|
||||||
|
"""
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def check(cls, value, col_attr):
|
||||||
|
"""Check if the value can match given the column attributes.
|
||||||
|
|
||||||
|
Raises this class if the value provided can't match any value in the
|
||||||
|
column in the table given the column's attributes. For example, if the
|
||||||
|
column is a string and the value is longer than the column then it
|
||||||
|
won't match any value in the column in the table.
|
||||||
|
|
||||||
|
"""
|
||||||
|
col = col_attr.property.columns[0]
|
||||||
|
if isinstance(col.type, sql.types.Boolean):
|
||||||
|
# The column is a Boolean, we should have already validated input.
|
||||||
|
return
|
||||||
|
if not col.type.length:
|
||||||
|
# The column doesn't have a length so can't validate anymore.
|
||||||
|
return
|
||||||
|
if len(value) > col.type.length:
|
||||||
|
raise cls()
|
||||||
|
# Otherwise the value could match a value in the column.
|
||||||
|
|
||||||
|
|
||||||
def _filter(model, query, hints):
|
def _filter(model, query, hints):
|
||||||
"""Applies filtering to a query.
|
"""Applies filtering to a query.
|
||||||
|
|
||||||
@ -276,10 +309,13 @@ def _filter(model, query, hints):
|
|||||||
return query
|
return query
|
||||||
|
|
||||||
if filter_['comparator'] == 'contains':
|
if filter_['comparator'] == 'contains':
|
||||||
|
_WontMatch.check(filter_['value'], column_attr)
|
||||||
query_term = column_attr.ilike('%%%s%%' % filter_['value'])
|
query_term = column_attr.ilike('%%%s%%' % filter_['value'])
|
||||||
elif filter_['comparator'] == 'startswith':
|
elif filter_['comparator'] == 'startswith':
|
||||||
|
_WontMatch.check(filter_['value'], column_attr)
|
||||||
query_term = column_attr.ilike('%s%%' % filter_['value'])
|
query_term = column_attr.ilike('%s%%' % filter_['value'])
|
||||||
elif filter_['comparator'] == 'endswith':
|
elif filter_['comparator'] == 'endswith':
|
||||||
|
_WontMatch.check(filter_['value'], column_attr)
|
||||||
query_term = column_attr.ilike('%%%s' % filter_['value'])
|
query_term = column_attr.ilike('%%%s' % filter_['value'])
|
||||||
else:
|
else:
|
||||||
# It's a filter we don't understand, so let the caller
|
# It's a filter we don't understand, so let the caller
|
||||||
@ -299,33 +335,40 @@ def _filter(model, query, hints):
|
|||||||
|
|
||||||
"""
|
"""
|
||||||
key = filter_['name']
|
key = filter_['name']
|
||||||
if isinstance(getattr(model, key).property.columns[0].type,
|
|
||||||
sql.types.Boolean):
|
col = getattr(model, key)
|
||||||
|
if isinstance(col.property.columns[0].type, sql.types.Boolean):
|
||||||
cumulative_filter_dict[key] = (
|
cumulative_filter_dict[key] = (
|
||||||
utils.attr_as_boolean(filter_['value']))
|
utils.attr_as_boolean(filter_['value']))
|
||||||
else:
|
else:
|
||||||
|
_WontMatch.check(filter_['value'], col)
|
||||||
cumulative_filter_dict[key] = filter_['value']
|
cumulative_filter_dict[key] = filter_['value']
|
||||||
|
|
||||||
filter_dict = {}
|
try:
|
||||||
satisfied_filters = []
|
filter_dict = {}
|
||||||
for filter_ in hints.filters:
|
satisfied_filters = []
|
||||||
if filter_['name'] not in model.attributes:
|
for filter_ in hints.filters:
|
||||||
continue
|
if filter_['name'] not in model.attributes:
|
||||||
if filter_['comparator'] == 'equals':
|
continue
|
||||||
exact_filter(model, filter_, filter_dict)
|
if filter_['comparator'] == 'equals':
|
||||||
satisfied_filters.append(filter_)
|
exact_filter(model, filter_, filter_dict)
|
||||||
else:
|
satisfied_filters.append(filter_)
|
||||||
query = inexact_filter(model, query, filter_, satisfied_filters)
|
else:
|
||||||
|
query = inexact_filter(model, query, filter_,
|
||||||
|
satisfied_filters)
|
||||||
|
|
||||||
# Apply any exact filters we built up
|
# Apply any exact filters we built up
|
||||||
if filter_dict:
|
if filter_dict:
|
||||||
query = query.filter_by(**filter_dict)
|
query = query.filter_by(**filter_dict)
|
||||||
|
|
||||||
# Remove satisfied filters, then the caller will know remaining filters
|
# Remove satisfied filters, then the caller will know remaining filters
|
||||||
for filter_ in satisfied_filters:
|
for filter_ in satisfied_filters:
|
||||||
hints.filters.remove(filter_)
|
hints.filters.remove(filter_)
|
||||||
|
|
||||||
return query
|
return query
|
||||||
|
except _WontMatch:
|
||||||
|
hints.cannot_match = True
|
||||||
|
return
|
||||||
|
|
||||||
|
|
||||||
def _limit(query, hints):
|
def _limit(query, hints):
|
||||||
@ -366,6 +409,10 @@ def filter_limit_query(model, query, hints):
|
|||||||
# First try and satisfy any filters
|
# First try and satisfy any filters
|
||||||
query = _filter(model, query, hints)
|
query = _filter(model, query, hints)
|
||||||
|
|
||||||
|
if hints.cannot_match:
|
||||||
|
# Nothing's going to match, so don't bother with the query.
|
||||||
|
return []
|
||||||
|
|
||||||
# NOTE(henry-nash): Any unsatisfied filters will have been left in
|
# NOTE(henry-nash): Any unsatisfied filters will have been left in
|
||||||
# the hints list for the controller to handle. We can only try and
|
# the hints list for the controller to handle. We can only try and
|
||||||
# limit here if all the filters are already satisfied since, if not,
|
# limit here if all the filters are already satisfied since, if not,
|
||||||
|
@ -5804,6 +5804,40 @@ class FilterTests(filtering.FilterTests):
|
|||||||
self._delete_test_data('user', user_list)
|
self._delete_test_data('user', user_list)
|
||||||
self._delete_test_data('group', group_list)
|
self._delete_test_data('group', group_list)
|
||||||
|
|
||||||
|
def _get_user_name_field_size(self):
|
||||||
|
"""Return the size of the user name field for the backend.
|
||||||
|
|
||||||
|
Subclasses can override this method to indicate that the user name
|
||||||
|
field is limited in length. The user name is the field used in the test
|
||||||
|
that validates that a filter value works even if it's longer than a
|
||||||
|
field.
|
||||||
|
|
||||||
|
If the backend doesn't limit the value length then return None.
|
||||||
|
|
||||||
|
"""
|
||||||
|
return None
|
||||||
|
|
||||||
|
def test_filter_value_wider_than_field(self):
|
||||||
|
# If a filter value is given that's larger than the field in the
|
||||||
|
# backend then no values are returned.
|
||||||
|
|
||||||
|
user_name_field_size = self._get_user_name_field_size()
|
||||||
|
|
||||||
|
if user_name_field_size is None:
|
||||||
|
# The backend doesn't limit the size of the user name, so pass this
|
||||||
|
# test.
|
||||||
|
return
|
||||||
|
|
||||||
|
# Create some users just to make sure would return something if the
|
||||||
|
# filter was ignored.
|
||||||
|
self._create_test_data('user', 2)
|
||||||
|
|
||||||
|
hints = driver_hints.Hints()
|
||||||
|
value = 'A' * (user_name_field_size + 1)
|
||||||
|
hints.add_filter('name', value)
|
||||||
|
users = self.identity_api.list_users(hints=hints)
|
||||||
|
self.assertEqual([], users)
|
||||||
|
|
||||||
|
|
||||||
class LimitTests(filtering.FilterTests):
|
class LimitTests(filtering.FilterTests):
|
||||||
ENTITIES = ['user', 'group', 'project']
|
ENTITIES = ['user', 'group', 'project']
|
||||||
|
@ -743,6 +743,9 @@ class SqlTokenCacheInvalidation(SqlTests, test_backend.TokenCacheInvalidation):
|
|||||||
|
|
||||||
class SqlFilterTests(SqlTests, test_backend.FilterTests):
|
class SqlFilterTests(SqlTests, test_backend.FilterTests):
|
||||||
|
|
||||||
|
def _get_user_name_field_size(self):
|
||||||
|
return identity_sql.User.name.type.length
|
||||||
|
|
||||||
def clean_up_entities(self):
|
def clean_up_entities(self):
|
||||||
"""Clean up entity test data from Filter Test Cases."""
|
"""Clean up entity test data from Filter Test Cases."""
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user