Fixed #31340 -- Allowed query expressions in SearchQuery.value and __search lookup.
This commit is contained in:
parent
924c01ba09
commit
3baf92cf82
|
@ -11,7 +11,7 @@ class SearchVectorExact(Lookup):
|
||||||
lookup_name = 'exact'
|
lookup_name = 'exact'
|
||||||
|
|
||||||
def process_rhs(self, qn, connection):
|
def process_rhs(self, qn, connection):
|
||||||
if not hasattr(self.rhs, 'resolve_expression'):
|
if not isinstance(self.rhs, (SearchQuery, CombinedSearchQuery)):
|
||||||
config = getattr(self.lhs, 'config', None)
|
config = getattr(self.lhs, 'config', None)
|
||||||
self.rhs = SearchQuery(self.rhs, config=config)
|
self.rhs = SearchQuery(self.rhs, config=config)
|
||||||
rhs, rhs_params = super().process_rhs(qn, connection)
|
rhs, rhs_params = super().process_rhs(qn, connection)
|
||||||
|
@ -170,7 +170,8 @@ class SearchQuery(SearchQueryCombinable, Func):
|
||||||
self.function = self.SEARCH_TYPES.get(search_type)
|
self.function = self.SEARCH_TYPES.get(search_type)
|
||||||
if self.function is None:
|
if self.function is None:
|
||||||
raise ValueError("Unknown search_type argument '%s'." % search_type)
|
raise ValueError("Unknown search_type argument '%s'." % search_type)
|
||||||
value = Value(value)
|
if not hasattr(value, 'resolve_expression'):
|
||||||
|
value = Value(value)
|
||||||
expressions = (value,)
|
expressions = (value,)
|
||||||
self.config = SearchConfig.from_parameter(config)
|
self.config = SearchConfig.from_parameter(config)
|
||||||
if self.config is not None:
|
if self.config is not None:
|
||||||
|
|
|
@ -35,6 +35,10 @@ query and the vector.
|
||||||
To use the ``search`` lookup, ``'django.contrib.postgres'`` must be in your
|
To use the ``search`` lookup, ``'django.contrib.postgres'`` must be in your
|
||||||
:setting:`INSTALLED_APPS`.
|
:setting:`INSTALLED_APPS`.
|
||||||
|
|
||||||
|
.. versionchanged:: 3.1
|
||||||
|
|
||||||
|
Support for query expressions was added.
|
||||||
|
|
||||||
``SearchVector``
|
``SearchVector``
|
||||||
================
|
================
|
||||||
|
|
||||||
|
@ -108,7 +112,8 @@ See :ref:`postgresql-fts-search-configuration` for an explanation of the
|
||||||
|
|
||||||
.. versionchanged:: 3.1
|
.. versionchanged:: 3.1
|
||||||
|
|
||||||
Support for ``'websearch'`` search type was added.
|
Support for ``'websearch'`` search type and query expressions in
|
||||||
|
``SearchQuery.value`` were added.
|
||||||
|
|
||||||
``SearchRank``
|
``SearchRank``
|
||||||
==============
|
==============
|
||||||
|
|
|
@ -113,9 +113,14 @@ Minor features
|
||||||
* :class:`~django.contrib.postgres.search.SearchQuery` now supports
|
* :class:`~django.contrib.postgres.search.SearchQuery` now supports
|
||||||
``'websearch'`` search type on PostgreSQL 11+.
|
``'websearch'`` search type on PostgreSQL 11+.
|
||||||
|
|
||||||
|
* :class:`SearchQuery.value <django.contrib.postgres.search.SearchQuery>` now
|
||||||
|
supports query expressions.
|
||||||
|
|
||||||
* The new :class:`~django.contrib.postgres.search.SearchHeadline` class allows
|
* The new :class:`~django.contrib.postgres.search.SearchHeadline` class allows
|
||||||
highlighting search results.
|
highlighting search results.
|
||||||
|
|
||||||
|
* :lookup:`search` lookup now supports query expressions.
|
||||||
|
|
||||||
:mod:`django.contrib.redirects`
|
:mod:`django.contrib.redirects`
|
||||||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
|
|
||||||
|
|
|
@ -185,6 +185,17 @@ class Migration(migrations.Migration):
|
||||||
},
|
},
|
||||||
bases=None,
|
bases=None,
|
||||||
),
|
),
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='LineSavedSearch',
|
||||||
|
fields=[
|
||||||
|
('id', models.AutoField(verbose_name='ID', serialize=False, auto_created=True, primary_key=True)),
|
||||||
|
('line', models.ForeignKey('postgres_tests.Line', on_delete=models.CASCADE)),
|
||||||
|
('query', models.CharField(max_length=100)),
|
||||||
|
],
|
||||||
|
options={
|
||||||
|
'required_db_vendor': 'postgresql',
|
||||||
|
},
|
||||||
|
),
|
||||||
migrations.CreateModel(
|
migrations.CreateModel(
|
||||||
name='AggregateTestModel',
|
name='AggregateTestModel',
|
||||||
fields=[
|
fields=[
|
||||||
|
|
|
@ -139,6 +139,11 @@ class Line(PostgreSQLModel):
|
||||||
return self.dialogue or ''
|
return self.dialogue or ''
|
||||||
|
|
||||||
|
|
||||||
|
class LineSavedSearch(PostgreSQLModel):
|
||||||
|
line = models.ForeignKey('Line', models.CASCADE)
|
||||||
|
query = models.CharField(max_length=100)
|
||||||
|
|
||||||
|
|
||||||
class RangesModel(PostgreSQLModel):
|
class RangesModel(PostgreSQLModel):
|
||||||
ints = IntegerRangeField(blank=True, null=True)
|
ints = IntegerRangeField(blank=True, null=True)
|
||||||
bigints = BigIntegerRangeField(blank=True, null=True)
|
bigints = BigIntegerRangeField(blank=True, null=True)
|
||||||
|
|
|
@ -10,7 +10,7 @@ from django.db.models import F
|
||||||
from django.test import modify_settings, skipUnlessDBFeature
|
from django.test import modify_settings, skipUnlessDBFeature
|
||||||
|
|
||||||
from . import PostgreSQLSimpleTestCase, PostgreSQLTestCase
|
from . import PostgreSQLSimpleTestCase, PostgreSQLTestCase
|
||||||
from .models import Character, Line, Scene
|
from .models import Character, Line, LineSavedSearch, Scene
|
||||||
|
|
||||||
try:
|
try:
|
||||||
from django.contrib.postgres.search import (
|
from django.contrib.postgres.search import (
|
||||||
|
@ -110,6 +110,18 @@ class SimpleSearchTest(GrailTestData, PostgreSQLTestCase):
|
||||||
)
|
)
|
||||||
self.assertSequenceEqual(searched, [self.verse2])
|
self.assertSequenceEqual(searched, [self.verse2])
|
||||||
|
|
||||||
|
def test_search_with_F_expression(self):
|
||||||
|
# Non-matching query.
|
||||||
|
LineSavedSearch.objects.create(line=self.verse1, query='hearts')
|
||||||
|
# Matching query.
|
||||||
|
match = LineSavedSearch.objects.create(line=self.verse1, query='elbows')
|
||||||
|
for query_expression in [F('query'), SearchQuery(F('query'))]:
|
||||||
|
with self.subTest(query_expression):
|
||||||
|
searched = LineSavedSearch.objects.filter(
|
||||||
|
line__dialogue__search=query_expression,
|
||||||
|
)
|
||||||
|
self.assertSequenceEqual(searched, [match])
|
||||||
|
|
||||||
|
|
||||||
@modify_settings(INSTALLED_APPS={'append': 'django.contrib.postgres'})
|
@modify_settings(INSTALLED_APPS={'append': 'django.contrib.postgres'})
|
||||||
class SearchVectorFieldTest(GrailTestData, PostgreSQLTestCase):
|
class SearchVectorFieldTest(GrailTestData, PostgreSQLTestCase):
|
||||||
|
|
Loading…
Reference in New Issue