Fixed #21863 -- supplemented get_lookup() with get_transform()

Also fixed #22124 -- Expanded explanation of exactly what is going on in
as_sql() methods.
This commit is contained in:
Anssi Kääriäinen 2014-03-01 21:21:57 +02:00 committed by Marc Tamlyn
parent a0f2525202
commit 219d928852
6 changed files with 177 additions and 41 deletions

View File

@ -9,11 +9,11 @@ from django.utils.six.moves import xrange
class RegisterLookupMixin(object): class RegisterLookupMixin(object):
def get_lookup(self, lookup_name): def _get_lookup(self, lookup_name):
try: try:
return self.class_lookups[lookup_name] return self.class_lookups[lookup_name]
except KeyError: except KeyError:
# To allow for inheritance, check parent class class lookups. # To allow for inheritance, check parent class' class_lookups.
for parent in inspect.getmro(self.__class__): for parent in inspect.getmro(self.__class__):
if not 'class_lookups' in parent.__dict__: if not 'class_lookups' in parent.__dict__:
continue continue
@ -26,6 +26,18 @@ class RegisterLookupMixin(object):
return self.output_type.get_lookup(lookup_name) return self.output_type.get_lookup(lookup_name)
return None return None
def get_lookup(self, lookup_name):
found = self._get_lookup(lookup_name)
if found is not None and not issubclass(found, Lookup):
return None
return found
def get_transform(self, lookup_name):
found = self._get_lookup(lookup_name)
if found is not None and not issubclass(found, Transform):
return None
return found
@classmethod @classmethod
def register_lookup(cls, lookup): def register_lookup(cls, lookup):
if not 'class_lookups' in cls.__dict__: if not 'class_lookups' in cls.__dict__:

View File

@ -24,6 +24,9 @@ class Col(object):
def get_lookup(self, name): def get_lookup(self, name):
return self.output_type.get_lookup(name) return self.output_type.get_lookup(name)
def get_transform(self, name):
return self.output_type.get_transform(name)
def prepare(self): def prepare(self):
return self return self

View File

@ -1088,24 +1088,21 @@ class Query(object):
lookups = lookups[:] lookups = lookups[:]
while lookups: while lookups:
lookup = lookups[0] lookup = lookups[0]
next = lhs.get_lookup(lookup) if len(lookups) == 1:
final_lookup = lhs.get_lookup(lookup)
if final_lookup:
return final_lookup(lhs, rhs)
# We didn't find a lookup, so we are going to try get_transform
# + get_lookup('exact').
lookups.append('exact')
next = lhs.get_transform(lookup)
if next: if next:
if len(lookups) == 1: lhs = next(lhs, lookups)
# This was the last lookup, so return value lookup.
if issubclass(next, Transform):
lookups.append('exact')
lhs = next(lhs, lookups)
else:
return next(lhs, rhs)
else:
lhs = next(lhs, lookups)
# A field's get_lookup() can return None to opt for backwards
# compatibility path.
elif len(lookups) > 2:
raise FieldError(
"Unsupported lookup for field '%s'" % lhs.output_type.name)
else: else:
return None raise FieldError(
"Unsupported lookup '%s' for %s or join on the field not "
"permitted." %
(lookup, lhs.output_type.__class__.__name__))
lookups = lookups[1:] lookups = lookups[1:]
def build_filter(self, filter_expr, branch_negated=False, current_negated=False, def build_filter(self, filter_expr, branch_negated=False, current_negated=False,

View File

@ -60,6 +60,14 @@ and use ``NotEqual`` to generate the SQL. By convention, these names are always
lowercase strings containing only letters, but the only hard requirement is lowercase strings containing only letters, but the only hard requirement is
that it must not contain the string ``__``. that it must not contain the string ``__``.
We then need to define the ``as_sql`` method. This takes a ``SQLCompiler``
object, called ``qn``, and the active database connection. ``SQLCompiler``
objects are not documented, but the only thing we need to know about them is
that they have a ``compile()`` method which returns a tuple containing a SQL
string, and the parameters to be interpolated into that string. In most cases,
you don't need to use it directly and can pass it on to ``process_lhs()`` and
``process_rhs()``.
A ``Lookup`` works against two values, ``lhs`` and ``rhs``, standing for A ``Lookup`` works against two values, ``lhs`` and ``rhs``, standing for
left-hand side and right-hand side. The left-hand side is usually a field left-hand side and right-hand side. The left-hand side is usually a field
reference, but it can be anything implementing the :ref:`query expression API reference, but it can be anything implementing the :ref:`query expression API
@ -69,11 +77,13 @@ reference to the ``name`` field of the ``Author`` model, and ``'Jack'`` is the
right-hand side. right-hand side.
We call ``process_lhs`` and ``process_rhs`` to convert them into the values we We call ``process_lhs`` and ``process_rhs`` to convert them into the values we
need for SQL. In the above example, ``process_lhs`` returns need for SQL using the ``qn`` object described before. These methods return
``('"author"."name"', [])`` and ``process_rhs`` returns ``('"%s"', ['Jack'])``. tuples containing some SQL and the parameters to be interpolated into that SQL,
In this example there were no parameters for the left hand side, but this would just as we need to return from our ``as_sql`` method. In the above example,
depend on the object we have, so we still need to include them in the ``process_lhs`` returns ``('"author"."name"', [])`` and ``process_rhs`` returns
parameters we return. ``('"%s"', ['Jack'])``. In this example there were no parameters for the left
hand side, but this would depend on the object we have, so we still need to
include them in the parameters we return.
Finally we combine the parts into a SQL expression with ``<>``, and supply all Finally we combine the parts into a SQL expression with ``<>``, and supply all
the parameters for the query. We then return a tuple containing the generated the parameters for the query. We then return a tuple containing the generated
@ -216,6 +226,52 @@ When compiling a query, Django first looks for ``as_%s % connection.vendor``
methods, and then falls back to ``as_sql``. The vendor names for the in-built methods, and then falls back to ``as_sql``. The vendor names for the in-built
backends are ``sqlite``, ``postgresql``, ``oracle`` and ``mysql``. backends are ``sqlite``, ``postgresql``, ``oracle`` and ``mysql``.
How Django determines the lookups and transforms which are used
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
In some cases you may which to dynamically change which ``Transform`` or
``Lookup`` is returned based on the name passed in, rather than fixing it. As
an example, you could have a field which stores coordinates or an arbitrary
dimension, and wish to allow a syntax like ``.filter(coords__x7=4)`` to return
the objects where the 7th coordinate has value 4. In order to do this, you
would override ``get_lookup`` with something like::
class CoordinatesField(Field):
def get_lookup(self, lookup_name):
if lookup_name.startswith('x'):
try:
dimension = int(lookup_name[1:])
except ValueError:
pass
finally:
return get_coordinate_lookup(dimension)
return super(CoordinatesField, self).get_lookup(lookup_name)
You would then define ``get_coordinate_lookup`` appropriately to return a
``Lookup`` subclass which handles the relevant value of ``dimension``.
There is a similarly named method called ``get_transform()``. ``get_lookup()``
should always return a ``Lookup`` subclass, and ``get_transform()`` a
``Transform`` subclass. It is important to remember that ``Transform``
objects can be further filtered on, and ``Lookup`` objects cannot.
When filtering, if there is only one lookup name remaining to be resolved, we
will look for a ``Lookup``. If there are multiple names, it will look for a
``Transform``. In the situation where there is only one name and a ``Lookup``
is not found, we look for a ``Transform`` and then the ``exact`` lookup on that
``Transform``. All call sequences always end with a ``Lookup``. To clarify:
- ``.filter(myfield__mylookup)`` will call ``myfield.get_lookup('mylookup')``.
- ``.filter(myfield__mytransform__mylookup)`` will call
``myfield.get_transform('mytransform')``, and then
``mytransform.get_lookup('mylookup')``.
- ``.filter(myfield__mytransform)`` will first call
``myfield.get_lookup('mytransform')``, which will fail, so it will fall back
to calling ``myfield.get_transform('mytransform')`` and then
``mytransform.get_lookup('exact')``.
Lookups and transforms are registered using the same API - ``register_lookup``.
.. _query-expression: .. _query-expression:
The Query Expression API The Query Expression API
@ -228,21 +284,14 @@ to this API.
.. method:: as_sql(qn, connection) .. method:: as_sql(qn, connection)
Responsible for producing the query string and parameters for the Responsible for producing the query string and parameters for the
expression. The ``qn`` has a ``compile()`` method that can be used to expression. The ``qn`` is a ``SQLCompiler`` object, which has a
compile other expressions. The ``connection`` is the connection used to ``compile()`` method that can be used to compile other expressions. The
execute the query. ``connection`` is the connection used to execute the query.
Calling expression.as_sql() directly is usually incorrect - instead Calling expression.as_sql() directly is usually incorrect - instead
``qn.compile(expression)`` should be used. The ``qn.compile()`` method will ``qn.compile(expression)`` should be used. The ``qn.compile()`` method will
take care of calling vendor-specific methods of the expression. take care of calling vendor-specific methods of the expression.
.. method:: get_lookup(lookup_name)
The ``get_lookup()`` method is used to fetch lookups. By default the
lookup is fetched from the expression's output type in the same way
described in registering and fetching lookup documentation below.
It is possible to override this method to alter that behavior.
.. method:: as_vendorname(qn, connection) .. method:: as_vendorname(qn, connection)
Works like ``as_sql()`` method. When an expression is compiled by Works like ``as_sql()`` method. When an expression is compiled by
@ -251,6 +300,21 @@ to this API.
The vendorname is one of ``postgresql``, ``oracle``, ``sqlite`` or The vendorname is one of ``postgresql``, ``oracle``, ``sqlite`` or
``mysql`` for Django's built-in backends. ``mysql`` for Django's built-in backends.
.. method:: get_lookup(lookup_name)
The ``get_lookup()`` method is used to fetch lookups. By default the
lookup is fetched from the expression's output type in the same way
described in registering and fetching lookup documentation below.
It is possible to override this method to alter that behavior.
.. method:: get_transform(lookup_name)
The ``get_transform()`` method is used when a transform is needed rather
than a lookup, or if a lookup is not found. This is a more complex
situation which is useful when there arbitrary possible lookups for a
field. Generally speaking, you will not need to override ``get_lookup()``
or ``get_transform()``, and can use ``register_lookup()`` instead.
.. attribute:: output_type .. attribute:: output_type
The ``output_type`` attribute is used by the ``get_lookup()`` method to check for The ``output_type`` attribute is used by the ``get_lookup()`` method to check for
@ -325,12 +389,19 @@ The lookup registration API is explained below.
Registers the Lookup or Transform for the class. For example Registers the Lookup or Transform for the class. For example
``DateField.register_lookup(YearExact)`` will register ``YearExact`` for ``DateField.register_lookup(YearExact)`` will register ``YearExact`` for
all ``DateFields`` in the project, but also for fields that are instances all ``DateFields`` in the project, but also for fields that are instances
of a subclass of ``DateField`` (for example ``DateTimeField``). of a subclass of ``DateField`` (for example ``DateTimeField``). You can
register a Lookup or a Transform using the same class method.
.. method:: get_lookup(lookup_name) .. method:: get_lookup(lookup_name)
Django uses ``get_lookup(lookup_name)`` to fetch lookups or transforms. Django uses ``get_lookup(lookup_name)`` to fetch lookups. The
The implementation of ``get_lookup()`` fetches lookups or transforms implementation of ``get_lookup()`` looks for a subclass which is registered
registered for the current class based on their lookup_name attribute. for the current class with the correct ``lookup_name``.
.. method:: get_transform(lookup_name)
Django uses ``get_transform(lookup_name)`` to fetch lookups. The
implementation of ``get_transform()`` looks for a subclass which is registered
for the current class with the correct ``transform_name``.
The lookup registration API is available for ``Transform`` and ``Field`` classes. The lookup registration API is available for ``Transform`` and ``Field`` classes.

View File

@ -3,10 +3,11 @@ from __future__ import unicode_literals
from datetime import date from datetime import date
import unittest import unittest
from django.test import TestCase from django.core.exceptions import FieldError
from .models import Author
from django.db import models from django.db import models
from django.db import connection from django.db import connection
from django.test import TestCase
from .models import Author
class Div3Lookup(models.Lookup): class Div3Lookup(models.Lookup):
@ -289,3 +290,54 @@ class YearLteTests(TestCase):
finally: finally:
YearTransform._unregister_lookup(CustomYearExact) YearTransform._unregister_lookup(CustomYearExact)
YearTransform.register_lookup(YearExact) YearTransform.register_lookup(YearExact)
class TrackCallsYearTransform(YearTransform):
lookup_name = 'year'
call_order = []
def as_sql(self, qn, connection):
lhs_sql, params = qn.compile(self.lhs)
return connection.ops.date_extract_sql('year', lhs_sql), params
@property
def output_type(self):
return models.IntegerField()
def get_lookup(self, lookup_name):
self.call_order.append('lookup')
return super(TrackCallsYearTransform, self).get_lookup(lookup_name)
def get_transform(self, lookup_name):
self.call_order.append('transform')
return super(TrackCallsYearTransform, self).get_transform(lookup_name)
class LookupTransformCallOrderTests(TestCase):
def test_call_order(self):
models.DateField.register_lookup(TrackCallsYearTransform)
try:
# junk lookup - tries lookup, then transform, then fails
with self.assertRaises(FieldError):
Author.objects.filter(birthdate__year__junk=2012)
self.assertEqual(TrackCallsYearTransform.call_order,
['lookup', 'transform'])
TrackCallsYearTransform.call_order = []
# junk transform - tries transform only, then fails
with self.assertRaises(FieldError):
Author.objects.filter(birthdate__year__junk__more_junk=2012)
self.assertEqual(TrackCallsYearTransform.call_order,
['transform'])
TrackCallsYearTransform.call_order = []
# Just getting the year (implied __exact) - lookup only
Author.objects.filter(birthdate__year=2012)
self.assertEqual(TrackCallsYearTransform.call_order,
['lookup'])
TrackCallsYearTransform.call_order = []
# Just getting the year (explicit __exact) - lookup only
Author.objects.filter(birthdate__year__exact=2012)
self.assertEqual(TrackCallsYearTransform.call_order,
['lookup'])
finally:
models.DateField._unregister_lookup(TrackCallsYearTransform)

View File

@ -476,8 +476,9 @@ class LookupTests(TestCase):
Article.objects.filter(headline__starts='Article') Article.objects.filter(headline__starts='Article')
self.fail('FieldError not raised') self.fail('FieldError not raised')
except FieldError as ex: except FieldError as ex:
self.assertEqual(str(ex), "Join on field 'headline' not permitted. " self.assertEqual(
"Did you misspell 'starts' for the lookup type?") str(ex), "Unsupported lookup 'starts' for CharField "
"or join on the field not permitted.")
def test_regex(self): def test_regex(self):
# Create some articles with a bit more interesting headlines for testing field lookups: # Create some articles with a bit more interesting headlines for testing field lookups: