123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288 |
- from __future__ import unicode_literals
- import datetime
- from decimal import Decimal
- from django.core.exceptions import FieldError
- from django.db.models import (
- Sum, Count,
- F, Value, Func,
- IntegerField, BooleanField, CharField)
- from django.db.models.fields import FieldDoesNotExist
- from django.test import TestCase
- from .models import Author, Book, Store, DepartmentStore, Company, Employee
- class NonAggregateAnnotationTestCase(TestCase):
- fixtures = ["annotations.json"]
- def test_basic_annotation(self):
- books = Book.objects.annotate(
- is_book=Value(1, output_field=IntegerField()))
- for book in books:
- self.assertEqual(book.is_book, 1)
- def test_basic_f_annotation(self):
- books = Book.objects.annotate(another_rating=F('rating'))
- for book in books:
- self.assertEqual(book.another_rating, book.rating)
- def test_joined_annotation(self):
- books = Book.objects.select_related('publisher').annotate(
- num_awards=F('publisher__num_awards'))
- for book in books:
- self.assertEqual(book.num_awards, book.publisher.num_awards)
- def test_annotate_with_aggregation(self):
- books = Book.objects.annotate(
- is_book=Value(1, output_field=IntegerField()),
- rating_count=Count('rating'))
- for book in books:
- self.assertEqual(book.is_book, 1)
- self.assertEqual(book.rating_count, 1)
- def test_aggregate_over_annotation(self):
- agg = Author.objects.annotate(other_age=F('age')).aggregate(otherage_sum=Sum('other_age'))
- other_agg = Author.objects.aggregate(age_sum=Sum('age'))
- self.assertEqual(agg['otherage_sum'], other_agg['age_sum'])
- def test_filter_annotation(self):
- books = Book.objects.annotate(
- is_book=Value(1, output_field=IntegerField())
- ).filter(is_book=1)
- for book in books:
- self.assertEqual(book.is_book, 1)
- def test_filter_annotation_with_f(self):
- books = Book.objects.annotate(
- other_rating=F('rating')
- ).filter(other_rating=3.5)
- for book in books:
- self.assertEqual(book.other_rating, 3.5)
- def test_filter_annotation_with_double_f(self):
- books = Book.objects.annotate(
- other_rating=F('rating')
- ).filter(other_rating=F('rating'))
- for book in books:
- self.assertEqual(book.other_rating, book.rating)
- def test_filter_agg_with_double_f(self):
- books = Book.objects.annotate(
- sum_rating=Sum('rating')
- ).filter(sum_rating=F('sum_rating'))
- for book in books:
- self.assertEqual(book.sum_rating, book.rating)
- def test_filter_wrong_annotation(self):
- with self.assertRaisesRegexp(FieldError, "Cannot resolve keyword .*"):
- list(Book.objects.annotate(
- sum_rating=Sum('rating')
- ).filter(sum_rating=F('nope')))
- def test_update_with_annotation(self):
- book_preupdate = Book.objects.get(pk=2)
- Book.objects.annotate(other_rating=F('rating') - 1).update(rating=F('other_rating'))
- book_postupdate = Book.objects.get(pk=2)
- self.assertEqual(book_preupdate.rating - 1, book_postupdate.rating)
- def test_annotation_with_m2m(self):
- books = Book.objects.annotate(author_age=F('authors__age')).filter(pk=1).order_by('author_age')
- self.assertEqual(books[0].author_age, 34)
- self.assertEqual(books[1].author_age, 35)
- def test_annotation_reverse_m2m(self):
- books = Book.objects.annotate(
- store_name=F('store__name')).filter(
- name='Practical Django Projects').order_by(
- 'store_name')
- self.assertQuerysetEqual(
- books, [
- 'Amazon.com',
- 'Books.com',
- 'Mamma and Pappa\'s Books'
- ],
- lambda b: b.store_name
- )
- def test_values_annotation(self):
- """
- Annotations can reference fields in a values clause,
- and contribute to an existing values clause.
- """
-
- qs = Book.objects.values('rating').annotate(other_rating=F('rating') - 1)
- book = qs.get(pk=1)
- self.assertEqual(book['rating'] - 1, book['other_rating'])
-
- book = qs.get(other_rating=4)
- self.assertEqual(book['other_rating'], 4)
-
- book = qs.annotate(other_isbn=F('isbn')).get(other_rating=4)
- self.assertEqual(book['other_rating'], 4)
- self.assertEqual(book['other_isbn'], '155860191')
- def test_defer_annotation(self):
- """
- Deferred attributes can be referenced by an annotation,
- but they are not themselves deferred, and cannot be deferred.
- """
- qs = Book.objects.defer('rating').annotate(other_rating=F('rating') - 1)
- with self.assertNumQueries(2):
- book = qs.get(other_rating=4)
- self.assertEqual(book.rating, 5)
- self.assertEqual(book.other_rating, 4)
- with self.assertRaisesRegexp(FieldDoesNotExist, "\w has no field named u?'other_rating'"):
- book = qs.defer('other_rating').get(other_rating=4)
- def test_mti_annotations(self):
- """
- Fields on an inherited model can be referenced by an
- annotated field.
- """
- d = DepartmentStore.objects.create(
- name='Angus & Robinson',
- original_opening=datetime.date(2014, 3, 8),
- friday_night_closing=datetime.time(21, 00, 00),
- chain='Westfield'
- )
- books = Book.objects.filter(rating__gt=4)
- for b in books:
- d.books.add(b)
- qs = DepartmentStore.objects.annotate(
- other_name=F('name'),
- other_chain=F('chain'),
- is_open=Value(True, BooleanField()),
- book_isbn=F('books__isbn')
- ).select_related('store').order_by('book_isbn').filter(chain='Westfield')
- self.assertQuerysetEqual(
- qs, [
- ('Angus & Robinson', 'Westfield', True, '155860191'),
- ('Angus & Robinson', 'Westfield', True, '159059725')
- ],
- lambda d: (d.other_name, d.other_chain, d.is_open, d.book_isbn)
- )
- def test_column_field_ordering(self):
- """
- Test that columns are aligned in the correct order for
- resolve_columns. This test will fail on mysql if column
- ordering is out. Column fields should be aligned as:
- 1. extra_select
- 2. model_fields
- 3. annotation_fields
- 4. model_related_fields
- """
- store = Store.objects.first()
- Employee.objects.create(id=1, first_name='Max', manager=True, last_name='Paine',
- store=store, age=23, salary=Decimal(50000.00))
- Employee.objects.create(id=2, first_name='Buffy', manager=False, last_name='Summers',
- store=store, age=18, salary=Decimal(40000.00))
- qs = Employee.objects.extra(
- select={'random_value': '42'}
- ).select_related('store').annotate(
- annotated_value=Value(17, output_field=IntegerField())
- )
- rows = [
- (1, 'Max', True, 42, 'Paine', 23, Decimal(50000.00), store.name, 17),
- (2, 'Buffy', False, 42, 'Summers', 18, Decimal(40000.00), store.name, 17)
- ]
- self.assertQuerysetEqual(
- qs.order_by('id'), rows,
- lambda e: (
- e.id, e.first_name, e.manager, e.random_value, e.last_name, e.age,
- e.salary, e.store.name, e.annotated_value))
- def test_column_field_ordering_with_deferred(self):
- store = Store.objects.first()
- Employee.objects.create(id=1, first_name='Max', manager=True, last_name='Paine',
- store=store, age=23, salary=Decimal(50000.00))
- Employee.objects.create(id=2, first_name='Buffy', manager=False, last_name='Summers',
- store=store, age=18, salary=Decimal(40000.00))
- qs = Employee.objects.extra(
- select={'random_value': '42'}
- ).select_related('store').annotate(
- annotated_value=Value(17, output_field=IntegerField())
- )
- rows = [
- (1, 'Max', True, 42, 'Paine', 23, Decimal(50000.00), store.name, 17),
- (2, 'Buffy', False, 42, 'Summers', 18, Decimal(40000.00), store.name, 17)
- ]
-
- self.assertQuerysetEqual(
- qs.defer('age').order_by('id'), rows,
- lambda e: (
- e.id, e.first_name, e.manager, e.random_value, e.last_name, e.age,
- e.salary, e.store.name, e.annotated_value))
- def test_custom_functions(self):
- Company(name='Apple', motto=None, ticker_name='APPL', description='Beautiful Devices').save()
- Company(name='Django Software Foundation', motto=None, ticker_name=None, description=None).save()
- Company(name='Google', motto='Do No Evil', ticker_name='GOOG', description='Internet Company').save()
- Company(name='Yahoo', motto=None, ticker_name=None, description='Internet Company').save()
- qs = Company.objects.annotate(
- tagline=Func(
- F('motto'),
- F('ticker_name'),
- F('description'),
- Value('No Tag'),
- function='COALESCE')
- ).order_by('name')
- self.assertQuerysetEqual(
- qs, [
- ('Apple', 'APPL'),
- ('Django Software Foundation', 'No Tag'),
- ('Google', 'Do No Evil'),
- ('Yahoo', 'Internet Company')
- ],
- lambda c: (c.name, c.tagline)
- )
- def test_custom_functions_can_ref_other_functions(self):
- Company(name='Apple', motto=None, ticker_name='APPL', description='Beautiful Devices').save()
- Company(name='Django Software Foundation', motto=None, ticker_name=None, description=None).save()
- Company(name='Google', motto='Do No Evil', ticker_name='GOOG', description='Internet Company').save()
- Company(name='Yahoo', motto=None, ticker_name=None, description='Internet Company').save()
- class Lower(Func):
- function = 'LOWER'
- qs = Company.objects.annotate(
- tagline=Func(
- F('motto'),
- F('ticker_name'),
- F('description'),
- Value('No Tag'),
- function='COALESCE')
- ).annotate(
- tagline_lower=Lower(F('tagline'), output_field=CharField())
- ).order_by('name')
-
-
- self.assertQuerysetEqual(
- qs, [
- ('Apple', 'APPL'.lower()),
- ('Django Software Foundation', 'No Tag'.lower()),
- ('Google', 'Do No Evil'.lower()),
- ('Yahoo', 'Internet Company'.lower())
- ],
- lambda c: (c.name, c.tagline_lower)
- )
|