Fixed #17813 -- Added a .earliest() method to QuerySet
Thanks a lot to everybody participating in developing this feature. The patch was developed by multiple people, at least Trac aliases tonnzor, jimmysong, Fandekasp and slurms. Stylistic changes added by committer.
This commit is contained in:
parent
37718eb50b
commit
fe54377dae
|
@ -172,6 +172,9 @@ class Manager(object):
|
||||||
def iterator(self, *args, **kwargs):
|
def iterator(self, *args, **kwargs):
|
||||||
return self.get_query_set().iterator(*args, **kwargs)
|
return self.get_query_set().iterator(*args, **kwargs)
|
||||||
|
|
||||||
|
def earliest(self, *args, **kwargs):
|
||||||
|
return self.get_query_set().earliest(*args, **kwargs)
|
||||||
|
|
||||||
def latest(self, *args, **kwargs):
|
def latest(self, *args, **kwargs):
|
||||||
return self.get_query_set().latest(*args, **kwargs)
|
return self.get_query_set().latest(*args, **kwargs)
|
||||||
|
|
||||||
|
|
|
@ -29,6 +29,7 @@ REPR_OUTPUT_SIZE = 20
|
||||||
# Pull into this namespace for backwards compatibility.
|
# Pull into this namespace for backwards compatibility.
|
||||||
EmptyResultSet = sql.EmptyResultSet
|
EmptyResultSet = sql.EmptyResultSet
|
||||||
|
|
||||||
|
|
||||||
class QuerySet(object):
|
class QuerySet(object):
|
||||||
"""
|
"""
|
||||||
Represents a lazy database lookup for a set of objects.
|
Represents a lazy database lookup for a set of objects.
|
||||||
|
@ -487,21 +488,28 @@ class QuerySet(object):
|
||||||
# Re-raise the IntegrityError with its original traceback.
|
# Re-raise the IntegrityError with its original traceback.
|
||||||
six.reraise(*exc_info)
|
six.reraise(*exc_info)
|
||||||
|
|
||||||
def latest(self, field_name=None):
|
def _earliest_or_latest(self, field_name=None, direction="-"):
|
||||||
"""
|
"""
|
||||||
Returns the latest object, according to the model's 'get_latest_by'
|
Returns the latest object, according to the model's
|
||||||
option or optional given field_name.
|
'get_latest_by' option or optional given field_name.
|
||||||
"""
|
"""
|
||||||
latest_by = field_name or self.model._meta.get_latest_by
|
order_by = field_name or getattr(self.model._meta, 'get_latest_by')
|
||||||
assert bool(latest_by), "latest() requires either a field_name parameter or 'get_latest_by' in the model"
|
assert bool(order_by), "earliest() and latest() require either a "\
|
||||||
|
"field_name parameter or 'get_latest_by' in the model"
|
||||||
assert self.query.can_filter(), \
|
assert self.query.can_filter(), \
|
||||||
"Cannot change a query once a slice has been taken."
|
"Cannot change a query once a slice has been taken."
|
||||||
obj = self._clone()
|
obj = self._clone()
|
||||||
obj.query.set_limits(high=1)
|
obj.query.set_limits(high=1)
|
||||||
obj.query.clear_ordering()
|
obj.query.clear_ordering()
|
||||||
obj.query.add_ordering('-%s' % latest_by)
|
obj.query.add_ordering('%s%s' % (direction, order_by))
|
||||||
return obj.get()
|
return obj.get()
|
||||||
|
|
||||||
|
def earliest(self, field_name=None):
|
||||||
|
return self._earliest_or_latest(field_name=field_name, direction="")
|
||||||
|
|
||||||
|
def latest(self, field_name=None):
|
||||||
|
return self._earliest_or_latest(field_name=field_name, direction="-")
|
||||||
|
|
||||||
def in_bulk(self, id_list):
|
def in_bulk(self, id_list):
|
||||||
"""
|
"""
|
||||||
Returns a dictionary mapping each of the given IDs to the object with
|
Returns a dictionary mapping each of the given IDs to the object with
|
||||||
|
|
|
@ -86,7 +86,8 @@ Django quotes column and table names behind the scenes.
|
||||||
The name of an orderable field in the model, typically a :class:`DateField`,
|
The name of an orderable field in the model, typically a :class:`DateField`,
|
||||||
:class:`DateTimeField`, or :class:`IntegerField`. This specifies the default
|
:class:`DateTimeField`, or :class:`IntegerField`. This specifies the default
|
||||||
field to use in your model :class:`Manager`'s
|
field to use in your model :class:`Manager`'s
|
||||||
:meth:`~django.db.models.query.QuerySet.latest` method.
|
:meth:`~django.db.models.query.QuerySet.latest` and
|
||||||
|
:meth:`~django.db.models.query.QuerySet.earliest` methods.
|
||||||
|
|
||||||
Example::
|
Example::
|
||||||
|
|
||||||
|
|
|
@ -1477,14 +1477,23 @@ This example returns the latest ``Entry`` in the table, according to the
|
||||||
|
|
||||||
If your model's :ref:`Meta <meta-options>` specifies
|
If your model's :ref:`Meta <meta-options>` specifies
|
||||||
:attr:`~django.db.models.Options.get_latest_by`, you can leave off the
|
:attr:`~django.db.models.Options.get_latest_by`, you can leave off the
|
||||||
``field_name`` argument to ``latest()``. Django will use the field specified
|
``field_name`` argument to ``earliest()`` or ``latest()``. Django will use the
|
||||||
in :attr:`~django.db.models.Options.get_latest_by` by default.
|
field specified in :attr:`~django.db.models.Options.get_latest_by` by default.
|
||||||
|
|
||||||
Like :meth:`get()`, ``latest()`` raises
|
Like :meth:`get()`, ``earliest()`` and ``latest()`` raise
|
||||||
:exc:`~django.core.exceptions.DoesNotExist` if there is no object with the given
|
:exc:`~django.core.exceptions.DoesNotExist` if there is no object with the
|
||||||
parameters.
|
given parameters.
|
||||||
|
|
||||||
Note ``latest()`` exists purely for convenience and readability.
|
Note that ``earliest()`` and ``latest()`` exist purely for convenience and
|
||||||
|
readability.
|
||||||
|
|
||||||
|
earliest
|
||||||
|
~~~~~~~~
|
||||||
|
|
||||||
|
.. method:: earliest(field_name=None)
|
||||||
|
|
||||||
|
Works otherwise like :meth:`~django.db.models.query.QuerySet.latest` except
|
||||||
|
the direction is changed.
|
||||||
|
|
||||||
aggregate
|
aggregate
|
||||||
~~~~~~~~~
|
~~~~~~~~~
|
||||||
|
|
|
@ -28,6 +28,9 @@ Minor features
|
||||||
undefined if the given ``QuerySet`` isn't ordered and there are more than
|
undefined if the given ``QuerySet`` isn't ordered and there are more than
|
||||||
one ordered values to compare against.
|
one ordered values to compare against.
|
||||||
|
|
||||||
|
* Added :meth:`~django.db.models.query.QuerySet.earliest` for symmetry with
|
||||||
|
:meth:`~django.db.models.query.QuerySet.latest`.
|
||||||
|
|
||||||
Backwards incompatible changes in 1.6
|
Backwards incompatible changes in 1.6
|
||||||
=====================================
|
=====================================
|
||||||
|
|
||||||
|
|
|
@ -9,10 +9,8 @@ farthest into the future."
|
||||||
"""
|
"""
|
||||||
|
|
||||||
from django.db import models
|
from django.db import models
|
||||||
from django.utils.encoding import python_2_unicode_compatible
|
|
||||||
|
|
||||||
|
|
||||||
@python_2_unicode_compatible
|
|
||||||
class Article(models.Model):
|
class Article(models.Model):
|
||||||
headline = models.CharField(max_length=100)
|
headline = models.CharField(max_length=100)
|
||||||
pub_date = models.DateField()
|
pub_date = models.DateField()
|
||||||
|
@ -20,15 +18,15 @@ class Article(models.Model):
|
||||||
class Meta:
|
class Meta:
|
||||||
get_latest_by = 'pub_date'
|
get_latest_by = 'pub_date'
|
||||||
|
|
||||||
def __str__(self):
|
def __unicode__(self):
|
||||||
return self.headline
|
return self.headline
|
||||||
|
|
||||||
@python_2_unicode_compatible
|
|
||||||
class Person(models.Model):
|
class Person(models.Model):
|
||||||
name = models.CharField(max_length=30)
|
name = models.CharField(max_length=30)
|
||||||
birthday = models.DateField()
|
birthday = models.DateField()
|
||||||
|
|
||||||
# Note that this model doesn't have "get_latest_by" set.
|
# Note that this model doesn't have "get_latest_by" set.
|
||||||
|
|
||||||
def __str__(self):
|
def __unicode__(self):
|
||||||
return self.name
|
return self.name
|
|
@ -0,0 +1,123 @@
|
||||||
|
from __future__ import absolute_import
|
||||||
|
|
||||||
|
from datetime import datetime
|
||||||
|
|
||||||
|
from django.test import TestCase
|
||||||
|
|
||||||
|
from .models import Article, Person
|
||||||
|
|
||||||
|
|
||||||
|
class EarliestOrLatestTests(TestCase):
|
||||||
|
"""Tests for the earliest() and latest() objects methods"""
|
||||||
|
|
||||||
|
def tearDown(self):
|
||||||
|
"""Makes sure Article has a get_latest_by"""
|
||||||
|
if not Article._meta.get_latest_by:
|
||||||
|
Article._meta.get_latest_by = 'pub_date'
|
||||||
|
|
||||||
|
def test_earliest(self):
|
||||||
|
# Because no Articles exist yet, earliest() raises ArticleDoesNotExist.
|
||||||
|
self.assertRaises(Article.DoesNotExist, Article.objects.earliest)
|
||||||
|
|
||||||
|
a1 = Article.objects.create(
|
||||||
|
headline="Article 1", pub_date=datetime(2005, 7, 26),
|
||||||
|
expire_date=datetime(2005, 9, 1)
|
||||||
|
)
|
||||||
|
a2 = Article.objects.create(
|
||||||
|
headline="Article 2", pub_date=datetime(2005, 7, 27),
|
||||||
|
expire_date=datetime(2005, 7, 28)
|
||||||
|
)
|
||||||
|
a3 = Article.objects.create(
|
||||||
|
headline="Article 3", pub_date=datetime(2005, 7, 28),
|
||||||
|
expire_date=datetime(2005, 8, 27)
|
||||||
|
)
|
||||||
|
a4 = Article.objects.create(
|
||||||
|
headline="Article 4", pub_date=datetime(2005, 7, 28),
|
||||||
|
expire_date=datetime(2005, 7, 30)
|
||||||
|
)
|
||||||
|
|
||||||
|
# Get the earliest Article.
|
||||||
|
self.assertEqual(Article.objects.earliest(), a1)
|
||||||
|
# Get the earliest Article that matches certain filters.
|
||||||
|
self.assertEqual(
|
||||||
|
Article.objects.filter(pub_date__gt=datetime(2005, 7, 26)).earliest(),
|
||||||
|
a2
|
||||||
|
)
|
||||||
|
|
||||||
|
# Pass a custom field name to earliest() to change the field that's used
|
||||||
|
# to determine the earliest object.
|
||||||
|
self.assertEqual(Article.objects.earliest('expire_date'), a2)
|
||||||
|
self.assertEqual(Article.objects.filter(
|
||||||
|
pub_date__gt=datetime(2005, 7, 26)).earliest('expire_date'), a2)
|
||||||
|
|
||||||
|
# Ensure that earliest() overrides any other ordering specified on the
|
||||||
|
# query. Refs #11283.
|
||||||
|
self.assertEqual(Article.objects.order_by('id').earliest(), a1)
|
||||||
|
|
||||||
|
# Ensure that error is raised if the user forgot to add a get_latest_by
|
||||||
|
# in the Model.Meta
|
||||||
|
Article.objects.model._meta.get_latest_by = None
|
||||||
|
self.assertRaisesMessage(
|
||||||
|
AssertionError,
|
||||||
|
"earliest() and latest() require either a field_name parameter or "
|
||||||
|
"'get_latest_by' in the model",
|
||||||
|
lambda: Article.objects.earliest(),
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_latest(self):
|
||||||
|
# Because no Articles exist yet, latest() raises ArticleDoesNotExist.
|
||||||
|
self.assertRaises(Article.DoesNotExist, Article.objects.latest)
|
||||||
|
|
||||||
|
a1 = Article.objects.create(
|
||||||
|
headline="Article 1", pub_date=datetime(2005, 7, 26),
|
||||||
|
expire_date=datetime(2005, 9, 1)
|
||||||
|
)
|
||||||
|
a2 = Article.objects.create(
|
||||||
|
headline="Article 2", pub_date=datetime(2005, 7, 27),
|
||||||
|
expire_date=datetime(2005, 7, 28)
|
||||||
|
)
|
||||||
|
a3 = Article.objects.create(
|
||||||
|
headline="Article 3", pub_date=datetime(2005, 7, 27),
|
||||||
|
expire_date=datetime(2005, 8, 27)
|
||||||
|
)
|
||||||
|
a4 = Article.objects.create(
|
||||||
|
headline="Article 4", pub_date=datetime(2005, 7, 28),
|
||||||
|
expire_date=datetime(2005, 7, 30)
|
||||||
|
)
|
||||||
|
|
||||||
|
# Get the latest Article.
|
||||||
|
self.assertEqual(Article.objects.latest(), a4)
|
||||||
|
# Get the latest Article that matches certain filters.
|
||||||
|
self.assertEqual(
|
||||||
|
Article.objects.filter(pub_date__lt=datetime(2005, 7, 27)).latest(),
|
||||||
|
a1
|
||||||
|
)
|
||||||
|
|
||||||
|
# Pass a custom field name to latest() to change the field that's used
|
||||||
|
# to determine the latest object.
|
||||||
|
self.assertEqual(Article.objects.latest('expire_date'), a1)
|
||||||
|
self.assertEqual(
|
||||||
|
Article.objects.filter(pub_date__gt=datetime(2005, 7, 26)).latest('expire_date'),
|
||||||
|
a3,
|
||||||
|
)
|
||||||
|
|
||||||
|
# Ensure that latest() overrides any other ordering specified on the query. Refs #11283.
|
||||||
|
self.assertEqual(Article.objects.order_by('id').latest(), a4)
|
||||||
|
|
||||||
|
# Ensure that error is raised if the user forgot to add a get_latest_by
|
||||||
|
# in the Model.Meta
|
||||||
|
Article.objects.model._meta.get_latest_by = None
|
||||||
|
self.assertRaisesMessage(
|
||||||
|
AssertionError,
|
||||||
|
"earliest() and latest() require either a field_name parameter or "
|
||||||
|
"'get_latest_by' in the model",
|
||||||
|
lambda: Article.objects.latest(),
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_latest_manual(self):
|
||||||
|
# You can still use latest() with a model that doesn't have
|
||||||
|
# "get_latest_by" set -- just pass in the field name manually.
|
||||||
|
p1 = Person.objects.create(name="Ralph", birthday=datetime(1950, 1, 1))
|
||||||
|
p2 = Person.objects.create(name="Stephanie", birthday=datetime(1960, 2, 3))
|
||||||
|
self.assertRaises(AssertionError, Person.objects.latest)
|
||||||
|
self.assertEqual(Person.objects.latest("birthday"), p2)
|
|
@ -1,58 +0,0 @@
|
||||||
from __future__ import absolute_import
|
|
||||||
|
|
||||||
from datetime import datetime
|
|
||||||
|
|
||||||
from django.test import TestCase
|
|
||||||
|
|
||||||
from .models import Article, Person
|
|
||||||
|
|
||||||
|
|
||||||
class LatestTests(TestCase):
|
|
||||||
def test_latest(self):
|
|
||||||
# Because no Articles exist yet, latest() raises ArticleDoesNotExist.
|
|
||||||
self.assertRaises(Article.DoesNotExist, Article.objects.latest)
|
|
||||||
|
|
||||||
a1 = Article.objects.create(
|
|
||||||
headline="Article 1", pub_date=datetime(2005, 7, 26),
|
|
||||||
expire_date=datetime(2005, 9, 1)
|
|
||||||
)
|
|
||||||
a2 = Article.objects.create(
|
|
||||||
headline="Article 2", pub_date=datetime(2005, 7, 27),
|
|
||||||
expire_date=datetime(2005, 7, 28)
|
|
||||||
)
|
|
||||||
a3 = Article.objects.create(
|
|
||||||
headline="Article 3", pub_date=datetime(2005, 7, 27),
|
|
||||||
expire_date=datetime(2005, 8, 27)
|
|
||||||
)
|
|
||||||
a4 = Article.objects.create(
|
|
||||||
headline="Article 4", pub_date=datetime(2005, 7, 28),
|
|
||||||
expire_date=datetime(2005, 7, 30)
|
|
||||||
)
|
|
||||||
|
|
||||||
# Get the latest Article.
|
|
||||||
self.assertEqual(Article.objects.latest(), a4)
|
|
||||||
# Get the latest Article that matches certain filters.
|
|
||||||
self.assertEqual(
|
|
||||||
Article.objects.filter(pub_date__lt=datetime(2005, 7, 27)).latest(),
|
|
||||||
a1
|
|
||||||
)
|
|
||||||
|
|
||||||
# Pass a custom field name to latest() to change the field that's used
|
|
||||||
# to determine the latest object.
|
|
||||||
self.assertEqual(Article.objects.latest('expire_date'), a1)
|
|
||||||
self.assertEqual(
|
|
||||||
Article.objects.filter(pub_date__gt=datetime(2005, 7, 26)).latest('expire_date'),
|
|
||||||
a3,
|
|
||||||
)
|
|
||||||
|
|
||||||
# Ensure that latest() overrides any other ordering specified on the query. Refs #11283.
|
|
||||||
self.assertEqual(Article.objects.order_by('id').latest(), a4)
|
|
||||||
|
|
||||||
def test_latest_manual(self):
|
|
||||||
# You can still use latest() with a model that doesn't have
|
|
||||||
# "get_latest_by" set -- just pass in the field name manually.
|
|
||||||
p1 = Person.objects.create(name="Ralph", birthday=datetime(1950, 1, 1))
|
|
||||||
p2 = Person.objects.create(name="Stephanie", birthday=datetime(1960, 2, 3))
|
|
||||||
self.assertRaises(AssertionError, Person.objects.latest)
|
|
||||||
|
|
||||||
self.assertEqual(Person.objects.latest("birthday"), p2)
|
|
Loading…
Reference in New Issue