mirror of
				https://github.com/django/django.git
				synced 2025-10-25 06:36:07 +00:00 
			
		
		
		
	Began implementing a shared set of test models to speed up tests.
This commit is contained in:
		| @@ -5,6 +5,7 @@ from operator import attrgetter | |||||||
|  |  | ||||||
| from django.core.exceptions import FieldError | from django.core.exceptions import FieldError | ||||||
| from django.test import TestCase, skipUnlessDBFeature | from django.test import TestCase, skipUnlessDBFeature | ||||||
|  | from django.utils import six | ||||||
|  |  | ||||||
| from shared_models.models import Author, Book | from shared_models.models import Author, Book | ||||||
|  |  | ||||||
| @@ -20,19 +21,19 @@ class LookupTests(TestCase): | |||||||
|         self.au2 = Author(name='Author 2') |         self.au2 = Author(name='Author 2') | ||||||
|         self.au2.save() |         self.au2.save() | ||||||
|         # Create a couple of Books. |         # Create a couple of Books. | ||||||
|         self.b1 = Book(name='Book 1', pubdate=datetime(2005, 7, 26), author=self.au1) |         self.b1 = Book(title='Book 1', pubdate=datetime(2005, 7, 26), author=self.au1) | ||||||
|         self.b1.save() |         self.b1.save() | ||||||
|         self.b2 = Book(name='Book 2', pubdate=datetime(2005, 7, 27), author=self.au1) |         self.b2 = Book(title='Book 2', pubdate=datetime(2005, 7, 27), author=self.au1) | ||||||
|         self.b2.save() |         self.b2.save() | ||||||
|         self.b3 = Book(name='Book 3', pubdate=datetime(2005, 7, 27), author=self.au1) |         self.b3 = Book(title='Book 3', pubdate=datetime(2005, 7, 27), author=self.au1) | ||||||
|         self.b3.save() |         self.b3.save() | ||||||
|         self.b4 = Book(name='Book 4', pubdate=datetime(2005, 7, 28), author=self.au1) |         self.b4 = Book(title='Book 4', pubdate=datetime(2005, 7, 28), author=self.au1) | ||||||
|         self.b4.save() |         self.b4.save() | ||||||
|         self.b5 = Book(name='Book 5', pubdate=datetime(2005, 8, 1, 9, 0), author=self.au2) |         self.b5 = Book(title='Book 5', pubdate=datetime(2005, 8, 1, 9, 0), author=self.au2) | ||||||
|         self.b5.save() |         self.b5.save() | ||||||
|         self.b6 = Book(name='Book 6', pubdate=datetime(2005, 8, 1, 8, 0), author=self.au2) |         self.b6 = Book(title='Book 6', pubdate=datetime(2005, 8, 1, 8, 0), author=self.au2) | ||||||
|         self.b6.save() |         self.b6.save() | ||||||
|         self.b7 = Book(name='Book 7', pubdate=datetime(2005, 7, 27), author=self.au2) |         self.b7 = Book(title='Book 7', pubdate=datetime(2005, 7, 27), author=self.au2) | ||||||
|         self.b7.save() |         self.b7.save() | ||||||
|         # Create a few Tags. |         # Create a few Tags. | ||||||
|         self.t1 = Tag(name='Tag 1') |         self.t1 = Tag(name='Tag 1') | ||||||
| @@ -85,18 +86,18 @@ class LookupTests(TestCase): | |||||||
|                 'Book 7', |                 'Book 7', | ||||||
|                 'Book 1', |                 'Book 1', | ||||||
|             ], |             ], | ||||||
|             transform=attrgetter('name')) |             transform=attrgetter('title')) | ||||||
|         # iterator() can be used on any QuerySet. |         # iterator() can be used on any QuerySet. | ||||||
|         self.assertQuerysetEqual( |         self.assertQuerysetEqual( | ||||||
|             Book.objects.filter(name__endswith='4').iterator(), |             Book.objects.filter(title__endswith='4').iterator(), | ||||||
|             ['Book 4'], |             ['Book 4'], | ||||||
|             transform=attrgetter('name')) |             transform=attrgetter('title')) | ||||||
|  |  | ||||||
|     def test_count(self): |     def test_count(self): | ||||||
|         # count() returns the number of objects matching search criteria. |         # count() returns the number of objects matching search criteria. | ||||||
|         self.assertEqual(Book.objects.count(), 7) |         self.assertEqual(Book.objects.count(), 7) | ||||||
|         self.assertEqual(Book.objects.filter(pubdate__exact=datetime(2005, 7, 27)).count(), 3) |         self.assertEqual(Book.objects.filter(pubdate__exact=datetime(2005, 7, 27)).count(), 3) | ||||||
|         self.assertEqual(Book.objects.filter(name__startswith='Blah blah').count(), 0) |         self.assertEqual(Book.objects.filter(title__startswith='Blah blah').count(), 0) | ||||||
|  |  | ||||||
|         # count() should respect sliced query sets. |         # count() should respect sliced query sets. | ||||||
|         articles = Book.objects.all() |         articles = Book.objects.all() | ||||||
| @@ -128,43 +129,43 @@ class LookupTests(TestCase): | |||||||
|         # values() returns a list of dictionaries instead of object instances -- |         # values() returns a list of dictionaries instead of object instances -- | ||||||
|         # and you can specify which fields you want to retrieve. |         # and you can specify which fields you want to retrieve. | ||||||
|         identity = lambda x:x |         identity = lambda x:x | ||||||
|         self.assertQuerysetEqual(Book.objects.values('name'), |         self.assertQuerysetEqual(Book.objects.values('title'), | ||||||
|             [ |             [ | ||||||
|                 {'name': 'Book 5'}, |                 {'title': 'Book 5'}, | ||||||
|                 {'name': 'Book 6'}, |                 {'title': 'Book 6'}, | ||||||
|                 {'name': 'Book 4'}, |                 {'title': 'Book 4'}, | ||||||
|                 {'name': 'Book 2'}, |                 {'title': 'Book 2'}, | ||||||
|                 {'name': 'Book 3'}, |                 {'title': 'Book 3'}, | ||||||
|                 {'name': 'Book 7'}, |                 {'title': 'Book 7'}, | ||||||
|                 {'name': 'Book 1'}, |                 {'title': 'Book 1'}, | ||||||
|             ], |             ], | ||||||
|             transform=identity) |             transform=identity) | ||||||
|         self.assertQuerysetEqual( |         self.assertQuerysetEqual( | ||||||
|             Book.objects.filter(pubdate__exact=datetime(2005, 7, 27)).values('id'), |             Book.objects.filter(pubdate__exact=datetime(2005, 7, 27)).values('id'), | ||||||
|             [{'id': self.b2.id}, {'id': self.b3.id}, {'id': self.b7.id}], |             [{'id': self.b2.id}, {'id': self.b3.id}, {'id': self.b7.id}], | ||||||
|             transform=identity) |             transform=identity) | ||||||
|         self.assertQuerysetEqual(Book.objects.values('id', 'name'), |         self.assertQuerysetEqual(Book.objects.values('id', 'title'), | ||||||
|             [ |             [ | ||||||
|                 {'id': self.b5.id, 'name': 'Book 5'}, |                 {'id': self.b5.id, 'title': 'Book 5'}, | ||||||
|                 {'id': self.b6.id, 'name': 'Book 6'}, |                 {'id': self.b6.id, 'title': 'Book 6'}, | ||||||
|                 {'id': self.b4.id, 'name': 'Book 4'}, |                 {'id': self.b4.id, 'title': 'Book 4'}, | ||||||
|                 {'id': self.b2.id, 'name': 'Book 2'}, |                 {'id': self.b2.id, 'title': 'Book 2'}, | ||||||
|                 {'id': self.b3.id, 'name': 'Book 3'}, |                 {'id': self.b3.id, 'title': 'Book 3'}, | ||||||
|                 {'id': self.b7.id, 'name': 'Book 7'}, |                 {'id': self.b7.id, 'title': 'Book 7'}, | ||||||
|                 {'id': self.b1.id, 'name': 'Book 1'}, |                 {'id': self.b1.id, 'title': 'Book 1'}, | ||||||
|             ], |             ], | ||||||
|             transform=identity) |             transform=identity) | ||||||
|         # You can use values() with iterator() for memory savings, |         # You can use values() with iterator() for memory savings, | ||||||
|         # because iterator() uses database-level iteration. |         # because iterator() uses database-level iteration. | ||||||
|         self.assertQuerysetEqual(Book.objects.values('id', 'name').iterator(), |         self.assertQuerysetEqual(Book.objects.values('id', 'title').iterator(), | ||||||
|             [ |             [ | ||||||
|                 {'name': 'Book 5', 'id': self.b5.id}, |                 {'title': 'Book 5', 'id': self.b5.id}, | ||||||
|                 {'name': 'Book 6', 'id': self.b6.id}, |                 {'title': 'Book 6', 'id': self.b6.id}, | ||||||
|                 {'name': 'Book 4', 'id': self.b4.id}, |                 {'title': 'Book 4', 'id': self.b4.id}, | ||||||
|                 {'name': 'Book 2', 'id': self.b2.id}, |                 {'title': 'Book 2', 'id': self.b2.id}, | ||||||
|                 {'name': 'Book 3', 'id': self.b3.id}, |                 {'title': 'Book 3', 'id': self.b3.id}, | ||||||
|                 {'name': 'Book 7', 'id': self.b7.id}, |                 {'title': 'Book 7', 'id': self.b7.id}, | ||||||
|                 {'name': 'Book 1', 'id': self.b1.id}, |                 {'title': 'Book 1', 'id': self.b1.id}, | ||||||
|             ], |             ], | ||||||
|             transform=identity) |             transform=identity) | ||||||
|         # The values() method works with "extra" fields specified in extra(select). |         # The values() method works with "extra" fields specified in extra(select). | ||||||
| @@ -204,39 +205,39 @@ class LookupTests(TestCase): | |||||||
|             }], transform=identity) |             }], transform=identity) | ||||||
|         # You can specify fields from forward and reverse relations, just like filter(). |         # You can specify fields from forward and reverse relations, just like filter(). | ||||||
|         self.assertQuerysetEqual( |         self.assertQuerysetEqual( | ||||||
|             Book.objects.values('name', 'author__name'), |             Book.objects.values('title', 'author__name'), | ||||||
|             [ |             [ | ||||||
|                 {'name': self.b5.name, 'author__name': self.au2.name}, |                 {'title': self.b5.title, 'author__name': self.au2.name}, | ||||||
|                 {'name': self.b6.name, 'author__name': self.au2.name}, |                 {'title': self.b6.title, 'author__name': self.au2.name}, | ||||||
|                 {'name': self.b4.name, 'author__name': self.au1.name}, |                 {'title': self.b4.title, 'author__name': self.au1.name}, | ||||||
|                 {'name': self.b2.name, 'author__name': self.au1.name}, |                 {'title': self.b2.title, 'author__name': self.au1.name}, | ||||||
|                 {'name': self.b3.name, 'author__name': self.au1.name}, |                 {'title': self.b3.title, 'author__name': self.au1.name}, | ||||||
|                 {'name': self.b7.name, 'author__name': self.au2.name}, |                 {'title': self.b7.title, 'author__name': self.au2.name}, | ||||||
|                 {'name': self.b1.name, 'author__name': self.au1.name}, |                 {'title': self.b1.title, 'author__name': self.au1.name}, | ||||||
|             ], transform=identity) |             ], transform=identity) | ||||||
|         self.assertQuerysetEqual( |         self.assertQuerysetEqual( | ||||||
|             Author.objects.values('name', 'book__name').order_by('name', 'book__name'), |             Author.objects.values('name', 'book__title').order_by('name', 'book__title'), | ||||||
|             [ |             [ | ||||||
|                 {'name': self.au1.name, 'book__name': self.b1.name}, |                 {'name': self.au1.name, 'book__title': self.b1.title}, | ||||||
|                 {'name': self.au1.name, 'book__name': self.b2.name}, |                 {'name': self.au1.name, 'book__title': self.b2.title}, | ||||||
|                 {'name': self.au1.name, 'book__name': self.b3.name}, |                 {'name': self.au1.name, 'book__title': self.b3.title}, | ||||||
|                 {'name': self.au1.name, 'book__name': self.b4.name}, |                 {'name': self.au1.name, 'book__title': self.b4.title}, | ||||||
|                 {'name': self.au2.name, 'book__name': self.b5.name}, |                 {'name': self.au2.name, 'book__title': self.b5.title}, | ||||||
|                 {'name': self.au2.name, 'book__name': self.b6.name}, |                 {'name': self.au2.name, 'book__title': self.b6.title}, | ||||||
|                 {'name': self.au2.name, 'book__name': self.b7.name}, |                 {'name': self.au2.name, 'book__title': self.b7.title}, | ||||||
|             ], transform=identity) |             ], transform=identity) | ||||||
|         self.assertQuerysetEqual( |         self.assertQuerysetEqual( | ||||||
|             Author.objects.values('name', 'book__name', 'book__tag__name').order_by('name', 'book__name', 'book__tag__name'), |             Author.objects.values('name', 'book__title', 'book__tag__name').order_by('name', 'book__title', 'book__tag__name'), | ||||||
|             [ |             [ | ||||||
|                 {'name': self.au1.name, 'book__name': self.b1.name, 'book__tag__name': self.t1.name}, |                 {'name': self.au1.name, 'book__title': self.b1.title, 'book__tag__name': self.t1.name}, | ||||||
|                 {'name': self.au1.name, 'book__name': self.b2.name, 'book__tag__name': self.t1.name}, |                 {'name': self.au1.name, 'book__title': self.b2.title, 'book__tag__name': self.t1.name}, | ||||||
|                 {'name': self.au1.name, 'book__name': self.b3.name, 'book__tag__name': self.t1.name}, |                 {'name': self.au1.name, 'book__title': self.b3.title, 'book__tag__name': self.t1.name}, | ||||||
|                 {'name': self.au1.name, 'book__name': self.b3.name, 'book__tag__name': self.t2.name}, |                 {'name': self.au1.name, 'book__title': self.b3.title, 'book__tag__name': self.t2.name}, | ||||||
|                 {'name': self.au1.name, 'book__name': self.b4.name, 'book__tag__name': self.t2.name}, |                 {'name': self.au1.name, 'book__title': self.b4.title, 'book__tag__name': self.t2.name}, | ||||||
|                 {'name': self.au2.name, 'book__name': self.b5.name, 'book__tag__name': self.t2.name}, |                 {'name': self.au2.name, 'book__title': self.b5.title, 'book__tag__name': self.t2.name}, | ||||||
|                 {'name': self.au2.name, 'book__name': self.b5.name, 'book__tag__name': self.t3.name}, |                 {'name': self.au2.name, 'book__title': self.b5.title, 'book__tag__name': self.t3.name}, | ||||||
|                 {'name': self.au2.name, 'book__name': self.b6.name, 'book__tag__name': self.t3.name}, |                 {'name': self.au2.name, 'book__title': self.b6.title, 'book__tag__name': self.t3.name}, | ||||||
|                 {'name': self.au2.name, 'book__name': self.b7.name, 'book__tag__name': self.t3.name}, |                 {'name': self.au2.name, 'book__title': self.b7.title, 'book__tag__name': self.t3.name}, | ||||||
|             ], transform=identity) |             ], transform=identity) | ||||||
|         # However, an exception FieldDoesNotExist will be thrown if you specify |         # However, an exception FieldDoesNotExist will be thrown if you specify | ||||||
|         # a non-existent field name in values() (a field that is neither in the |         # a non-existent field name in values() (a field that is neither in the | ||||||
| @@ -249,7 +250,7 @@ class LookupTests(TestCase): | |||||||
|             [{ |             [{ | ||||||
|                 'id': self.b5.id, |                 'id': self.b5.id, | ||||||
|                 'author_id': self.au2.id, |                 'author_id': self.au2.id, | ||||||
|                 'name': 'Book 5', |                 'title': 'Book 5', | ||||||
|                 'pages': 0, |                 'pages': 0, | ||||||
|                 'pubdate': datetime(2005, 8, 1, 9, 0) |                 'pubdate': datetime(2005, 8, 1, 9, 0) | ||||||
|             }], transform=identity) |             }], transform=identity) | ||||||
| @@ -260,7 +261,7 @@ class LookupTests(TestCase): | |||||||
|         # Within each tuple, the order of the elements is the same as the order |         # Within each tuple, the order of the elements is the same as the order | ||||||
|         # of fields in the values_list() call. |         # of fields in the values_list() call. | ||||||
|         identity = lambda x:x |         identity = lambda x:x | ||||||
|         self.assertQuerysetEqual(Book.objects.values_list('name'), |         self.assertQuerysetEqual(Book.objects.values_list('title'), | ||||||
|             [ |             [ | ||||||
|                 ('Book 5',), |                 ('Book 5',), | ||||||
|                 ('Book 6',), |                 ('Book 6',), | ||||||
| @@ -309,19 +310,19 @@ class LookupTests(TestCase): | |||||||
|             ], |             ], | ||||||
|             transform=identity) |             transform=identity) | ||||||
|         self.assertQuerysetEqual( |         self.assertQuerysetEqual( | ||||||
|             Author.objects.values_list('name', 'book__name', 'book__tag__name').order_by('name', 'book__name', 'book__tag__name'), |             Author.objects.values_list('name', 'book__title', 'book__tag__name').order_by('name', 'book__title', 'book__tag__name'), | ||||||
|             [ |             [ | ||||||
|                 (self.au1.name, self.b1.name, self.t1.name), |                 (self.au1.name, self.b1.title, self.t1.name), | ||||||
|                 (self.au1.name, self.b2.name, self.t1.name), |                 (self.au1.name, self.b2.title, self.t1.name), | ||||||
|                 (self.au1.name, self.b3.name, self.t1.name), |                 (self.au1.name, self.b3.title, self.t1.name), | ||||||
|                 (self.au1.name, self.b3.name, self.t2.name), |                 (self.au1.name, self.b3.title, self.t2.name), | ||||||
|                 (self.au1.name, self.b4.name, self.t2.name), |                 (self.au1.name, self.b4.title, self.t2.name), | ||||||
|                 (self.au2.name, self.b5.name, self.t2.name), |                 (self.au2.name, self.b5.title, self.t2.name), | ||||||
|                 (self.au2.name, self.b5.name, self.t3.name), |                 (self.au2.name, self.b5.title, self.t3.name), | ||||||
|                 (self.au2.name, self.b6.name, self.t3.name), |                 (self.au2.name, self.b6.title, self.t3.name), | ||||||
|                 (self.au2.name, self.b7.name, self.t3.name), |                 (self.au2.name, self.b7.title, self.t3.name), | ||||||
|             ], transform=identity) |             ], transform=identity) | ||||||
|         self.assertRaises(TypeError, Book.objects.values_list, 'id', 'name', flat=True) |         self.assertRaises(TypeError, Book.objects.values_list, 'id', 'title', flat=True) | ||||||
|  |  | ||||||
|     def test_get_next_previous_by(self): |     def test_get_next_previous_by(self): | ||||||
|         # Every DateField and DateTimeField creates get_next_by_FOO() and |         # Every DateField and DateTimeField creates get_next_by_FOO() and | ||||||
| @@ -332,7 +333,7 @@ class LookupTests(TestCase): | |||||||
|                          '<Book: Book 2>') |                          '<Book: Book 2>') | ||||||
|         self.assertEqual(repr(self.b2.get_next_by_pubdate()), |         self.assertEqual(repr(self.b2.get_next_by_pubdate()), | ||||||
|                          '<Book: Book 3>') |                          '<Book: Book 3>') | ||||||
|         self.assertEqual(repr(self.b2.get_next_by_pubdate(name__endswith='6')), |         self.assertEqual(repr(self.b2.get_next_by_pubdate(title__endswith='6')), | ||||||
|                          '<Book: Book 6>') |                          '<Book: Book 6>') | ||||||
|         self.assertEqual(repr(self.b3.get_next_by_pubdate()), |         self.assertEqual(repr(self.b3.get_next_by_pubdate()), | ||||||
|                          '<Book: Book 7>') |                          '<Book: Book 7>') | ||||||
| @@ -360,9 +361,9 @@ class LookupTests(TestCase): | |||||||
|     def test_escaping(self): |     def test_escaping(self): | ||||||
|         # Underscores, percent signs and backslashes have special meaning in the |         # Underscores, percent signs and backslashes have special meaning in the | ||||||
|         # underlying SQL code, but Django handles the quoting of them automatically. |         # underlying SQL code, but Django handles the quoting of them automatically. | ||||||
|         b8 = Book(name='Book_ with underscore', pubdate=datetime(2005, 11, 20)) |         b8 = Book(title='Book_ with underscore', pubdate=datetime(2005, 11, 20)) | ||||||
|         b8.save() |         b8.save() | ||||||
|         self.assertQuerysetEqual(Book.objects.filter(name__startswith='Book'), |         self.assertQuerysetEqual(Book.objects.filter(title__startswith='Book'), | ||||||
|             [ |             [ | ||||||
|                 '<Book: Book_ with underscore>', |                 '<Book: Book_ with underscore>', | ||||||
|                 '<Book: Book 5>', |                 '<Book: Book 5>', | ||||||
| @@ -373,11 +374,11 @@ class LookupTests(TestCase): | |||||||
|                 '<Book: Book 7>', |                 '<Book: Book 7>', | ||||||
|                 '<Book: Book 1>', |                 '<Book: Book 1>', | ||||||
|             ]) |             ]) | ||||||
|         self.assertQuerysetEqual(Book.objects.filter(name__startswith='Book_'), |         self.assertQuerysetEqual(Book.objects.filter(title__startswith='Book_'), | ||||||
|                                  ['<Book: Book_ with underscore>']) |                                  ['<Book: Book_ with underscore>']) | ||||||
|         b9 = Book(name='Book% with percent sign', pubdate=datetime(2005, 11, 21)) |         b9 = Book(title='Book% with percent sign', pubdate=datetime(2005, 11, 21)) | ||||||
|         b9.save() |         b9.save() | ||||||
|         self.assertQuerysetEqual(Book.objects.filter(name__startswith='Book'), |         self.assertQuerysetEqual(Book.objects.filter(title__startswith='Book'), | ||||||
|             [ |             [ | ||||||
|                 '<Book: Book% with percent sign>', |                 '<Book: Book% with percent sign>', | ||||||
|                 '<Book: Book_ with underscore>', |                 '<Book: Book_ with underscore>', | ||||||
| @@ -389,21 +390,21 @@ class LookupTests(TestCase): | |||||||
|                 '<Book: Book 7>', |                 '<Book: Book 7>', | ||||||
|                 '<Book: Book 1>', |                 '<Book: Book 1>', | ||||||
|             ]) |             ]) | ||||||
|         self.assertQuerysetEqual(Book.objects.filter(name__startswith='Book%'), |         self.assertQuerysetEqual(Book.objects.filter(title__startswith='Book%'), | ||||||
|                                  ['<Book: Book% with percent sign>']) |                                  ['<Book: Book% with percent sign>']) | ||||||
|         b10 = Book(name='Book with \\ backslash', pubdate=datetime(2005, 11, 22)) |         b10 = Book(title='Book with \\ backslash', pubdate=datetime(2005, 11, 22)) | ||||||
|         b10.save() |         b10.save() | ||||||
|         self.assertQuerysetEqual(Book.objects.filter(name__contains='\\'), |         self.assertQuerysetEqual(Book.objects.filter(title__contains='\\'), | ||||||
|                                  ['<Book: Book with \ backslash>']) |                                  ['<Book: Book with \ backslash>']) | ||||||
|  |  | ||||||
|     def test_exclude(self): |     def test_exclude(self): | ||||||
|         b8 = Book.objects.create(name='Book_ with underscore', pubdate=datetime(2005, 11, 20)) |         b8 = Book.objects.create(title='Book_ with underscore', pubdate=datetime(2005, 11, 20)) | ||||||
|         b9 = Book.objects.create(name='Book% with percent sign', pubdate=datetime(2005, 11, 21)) |         b9 = Book.objects.create(title='Book% with percent sign', pubdate=datetime(2005, 11, 21)) | ||||||
|         b10 = Book.objects.create(name='Book with \\ backslash', pubdate=datetime(2005, 11, 22)) |         b10 = Book.objects.create(title='Book with \\ backslash', pubdate=datetime(2005, 11, 22)) | ||||||
|  |  | ||||||
|         # exclude() is the opposite of filter() when doing lookups: |         # exclude() is the opposite of filter() when doing lookups: | ||||||
|         self.assertQuerysetEqual( |         self.assertQuerysetEqual( | ||||||
|             Book.objects.filter(name__contains='Book').exclude(name__contains='with'), |             Book.objects.filter(title__contains='Book').exclude(title__contains='with'), | ||||||
|             [ |             [ | ||||||
|                 '<Book: Book 5>', |                 '<Book: Book 5>', | ||||||
|                 '<Book: Book 6>', |                 '<Book: Book 6>', | ||||||
| @@ -413,7 +414,7 @@ class LookupTests(TestCase): | |||||||
|                 '<Book: Book 7>', |                 '<Book: Book 7>', | ||||||
|                 '<Book: Book 1>', |                 '<Book: Book 1>', | ||||||
|             ]) |             ]) | ||||||
|         self.assertQuerysetEqual(Book.objects.exclude(name__startswith="Book_"), |         self.assertQuerysetEqual(Book.objects.exclude(title__startswith="Book_"), | ||||||
|             [ |             [ | ||||||
|                 '<Book: Book with \\ backslash>', |                 '<Book: Book with \\ backslash>', | ||||||
|                 '<Book: Book% with percent sign>', |                 '<Book: Book% with percent sign>', | ||||||
| @@ -425,7 +426,7 @@ class LookupTests(TestCase): | |||||||
|                 '<Book: Book 7>', |                 '<Book: Book 7>', | ||||||
|                 '<Book: Book 1>', |                 '<Book: Book 1>', | ||||||
|             ]) |             ]) | ||||||
|         self.assertQuerysetEqual(Book.objects.exclude(name="Book 7"), |         self.assertQuerysetEqual(Book.objects.exclude(title="Book 7"), | ||||||
|             [ |             [ | ||||||
|                 '<Book: Book with \\ backslash>', |                 '<Book: Book with \\ backslash>', | ||||||
|                 '<Book: Book% with percent sign>', |                 '<Book: Book% with percent sign>', | ||||||
| @@ -442,12 +443,12 @@ class LookupTests(TestCase): | |||||||
|        # none() returns a QuerySet that behaves like any other QuerySet object |        # none() returns a QuerySet that behaves like any other QuerySet object | ||||||
|         self.assertQuerysetEqual(Book.objects.none(), []) |         self.assertQuerysetEqual(Book.objects.none(), []) | ||||||
|         self.assertQuerysetEqual( |         self.assertQuerysetEqual( | ||||||
|             Book.objects.none().filter(name__startswith='Book'), []) |             Book.objects.none().filter(title__startswith='Book'), []) | ||||||
|         self.assertQuerysetEqual( |         self.assertQuerysetEqual( | ||||||
|             Book.objects.filter(name__startswith='Book').none(), []) |             Book.objects.filter(title__startswith='Book').none(), []) | ||||||
|         self.assertEqual(Book.objects.none().count(), 0) |         self.assertEqual(Book.objects.none().count(), 0) | ||||||
|         self.assertEqual( |         self.assertEqual( | ||||||
|             Book.objects.none().update(name="This should not take effect"), 0) |             Book.objects.none().update(title="This should not take effect"), 0) | ||||||
|         self.assertQuerysetEqual( |         self.assertQuerysetEqual( | ||||||
|             [article for article in Book.objects.none().iterator()], |             [article for article in Book.objects.none().iterator()], | ||||||
|             []) |             []) | ||||||
| @@ -468,46 +469,41 @@ class LookupTests(TestCase): | |||||||
|  |  | ||||||
|     def test_error_messages(self): |     def test_error_messages(self): | ||||||
|         # Programming errors are pointed out with nice error messages |         # Programming errors are pointed out with nice error messages | ||||||
|         try: |         with six.assertRaisesRegex(self, FieldError, "Cannot resolve keyword 'pubdate_year' " | ||||||
|             Book.objects.filter(pubdate_year='2005').count() |              "into field. Choices are: .+"): | ||||||
|             self.fail('FieldError not raised') |                 Book.objects.filter(pubdate_year='2005').count() | ||||||
|         except FieldError as ex: |  | ||||||
|             self.assertEqual(str(ex), "Cannot resolve keyword 'pubdate_year' " |         with self.assertRaises(FieldError, msg="Join on field 'title' not permitted. " | ||||||
|                              "into field. Choices are: author, id, name, pages, pubdate, tag, tags") |              "Did you misspell 'starts' for the lookup type?"): | ||||||
|         try: |                 Book.objects.filter(title__starts='Book') | ||||||
|             Book.objects.filter(name__starts='Book') |  | ||||||
|             self.fail('FieldError not raised') |  | ||||||
|         except FieldError as ex: |  | ||||||
|             self.assertEqual(str(ex), "Join on field 'name' not permitted. " |  | ||||||
|                              "Did you misspell 'starts' for the lookup type?") |  | ||||||
|  |  | ||||||
|     def test_regex(self): |     def test_regex(self): | ||||||
|         # Create some articles with a bit more interesting names for testing field lookups: |         # Create some articles with a bit more interesting names for testing field lookups: | ||||||
|         for a in Book.objects.all(): |         for a in Book.objects.all(): | ||||||
|             a.delete() |             a.delete() | ||||||
|         now = datetime.now() |         now = datetime.now() | ||||||
|         b1 = Book(pubdate=now, name='f') |         b1 = Book(pubdate=now, title='f') | ||||||
|         b1.save() |         b1.save() | ||||||
|         b2 = Book(pubdate=now, name='fo') |         b2 = Book(pubdate=now, title='fo') | ||||||
|         b2.save() |         b2.save() | ||||||
|         b3 = Book(pubdate=now, name='foo') |         b3 = Book(pubdate=now, title='foo') | ||||||
|         b3.save() |         b3.save() | ||||||
|         b4 = Book(pubdate=now, name='fooo') |         b4 = Book(pubdate=now, title='fooo') | ||||||
|         b4.save() |         b4.save() | ||||||
|         b5 = Book(pubdate=now, name='hey-Foo') |         b5 = Book(pubdate=now, title='hey-Foo') | ||||||
|         b5.save() |         b5.save() | ||||||
|         b6 = Book(pubdate=now, name='bar') |         b6 = Book(pubdate=now, title='bar') | ||||||
|         b6.save() |         b6.save() | ||||||
|         b7 = Book(pubdate=now, name='AbBa') |         b7 = Book(pubdate=now, title='AbBa') | ||||||
|         b7.save() |         b7.save() | ||||||
|         b8 = Book(pubdate=now, name='baz') |         b8 = Book(pubdate=now, title='baz') | ||||||
|         b8.save() |         b8.save() | ||||||
|         b9 = Book(pubdate=now, name='baxZ') |         b9 = Book(pubdate=now, title='baxZ') | ||||||
|         b9.save() |         b9.save() | ||||||
|         # zero-or-more |         # zero-or-more | ||||||
|         self.assertQuerysetEqual(Book.objects.filter(name__regex=r'fo*'), |         self.assertQuerysetEqual(Book.objects.filter(title__regex=r'fo*'), | ||||||
|             ['<Book: f>', '<Book: fo>', '<Book: foo>', '<Book: fooo>']) |             ['<Book: f>', '<Book: fo>', '<Book: foo>', '<Book: fooo>']) | ||||||
|         self.assertQuerysetEqual(Book.objects.filter(name__iregex=r'fo*'), |         self.assertQuerysetEqual(Book.objects.filter(title__iregex=r'fo*'), | ||||||
|             [ |             [ | ||||||
|                 '<Book: f>', |                 '<Book: f>', | ||||||
|                 '<Book: fo>', |                 '<Book: fo>', | ||||||
| @@ -516,54 +512,54 @@ class LookupTests(TestCase): | |||||||
|                 '<Book: hey-Foo>', |                 '<Book: hey-Foo>', | ||||||
|             ]) |             ]) | ||||||
|         # one-or-more |         # one-or-more | ||||||
|         self.assertQuerysetEqual(Book.objects.filter(name__regex=r'fo+'), |         self.assertQuerysetEqual(Book.objects.filter(title__regex=r'fo+'), | ||||||
|             ['<Book: fo>', '<Book: foo>', '<Book: fooo>']) |             ['<Book: fo>', '<Book: foo>', '<Book: fooo>']) | ||||||
|         # wildcard |         # wildcard | ||||||
|         self.assertQuerysetEqual(Book.objects.filter(name__regex=r'fooo?'), |         self.assertQuerysetEqual(Book.objects.filter(title__regex=r'fooo?'), | ||||||
|             ['<Book: foo>', '<Book: fooo>']) |             ['<Book: foo>', '<Book: fooo>']) | ||||||
|         # leading anchor |         # leading anchor | ||||||
|         self.assertQuerysetEqual(Book.objects.filter(name__regex=r'^b'), |         self.assertQuerysetEqual(Book.objects.filter(title__regex=r'^b'), | ||||||
|             ['<Book: bar>', '<Book: baxZ>', '<Book: baz>']) |             ['<Book: bar>', '<Book: baxZ>', '<Book: baz>']) | ||||||
|         self.assertQuerysetEqual(Book.objects.filter(name__iregex=r'^a'), |         self.assertQuerysetEqual(Book.objects.filter(title__iregex=r'^a'), | ||||||
|             ['<Book: AbBa>']) |             ['<Book: AbBa>']) | ||||||
|         # trailing anchor |         # trailing anchor | ||||||
|         self.assertQuerysetEqual(Book.objects.filter(name__regex=r'z$'), |         self.assertQuerysetEqual(Book.objects.filter(title__regex=r'z$'), | ||||||
|             ['<Book: baz>']) |             ['<Book: baz>']) | ||||||
|         self.assertQuerysetEqual(Book.objects.filter(name__iregex=r'z$'), |         self.assertQuerysetEqual(Book.objects.filter(title__iregex=r'z$'), | ||||||
|             ['<Book: baxZ>', '<Book: baz>']) |             ['<Book: baxZ>', '<Book: baz>']) | ||||||
|         # character sets |         # character sets | ||||||
|         self.assertQuerysetEqual(Book.objects.filter(name__regex=r'ba[rz]'), |         self.assertQuerysetEqual(Book.objects.filter(title__regex=r'ba[rz]'), | ||||||
|             ['<Book: bar>', '<Book: baz>']) |             ['<Book: bar>', '<Book: baz>']) | ||||||
|         self.assertQuerysetEqual(Book.objects.filter(name__regex=r'ba.[RxZ]'), |         self.assertQuerysetEqual(Book.objects.filter(title__regex=r'ba.[RxZ]'), | ||||||
|             ['<Book: baxZ>']) |             ['<Book: baxZ>']) | ||||||
|         self.assertQuerysetEqual(Book.objects.filter(name__iregex=r'ba[RxZ]'), |         self.assertQuerysetEqual(Book.objects.filter(title__iregex=r'ba[RxZ]'), | ||||||
|             ['<Book: bar>', '<Book: baxZ>', '<Book: baz>']) |             ['<Book: bar>', '<Book: baxZ>', '<Book: baz>']) | ||||||
|  |  | ||||||
|         # and more articles: |         # and more articles: | ||||||
|         b10 = Book(pubdate=now, name='foobar') |         b10 = Book(pubdate=now, title='foobar') | ||||||
|         b10.save() |         b10.save() | ||||||
|         b11 = Book(pubdate=now, name='foobaz') |         b11 = Book(pubdate=now, title='foobaz') | ||||||
|         b11.save() |         b11.save() | ||||||
|         b12 = Book(pubdate=now, name='ooF') |         b12 = Book(pubdate=now, title='ooF') | ||||||
|         b12.save() |         b12.save() | ||||||
|         b13 = Book(pubdate=now, name='foobarbaz') |         b13 = Book(pubdate=now, title='foobarbaz') | ||||||
|         b13.save() |         b13.save() | ||||||
|         b14 = Book(pubdate=now, name='zoocarfaz') |         b14 = Book(pubdate=now, title='zoocarfaz') | ||||||
|         b14.save() |         b14.save() | ||||||
|         b15 = Book(pubdate=now, name='barfoobaz') |         b15 = Book(pubdate=now, title='barfoobaz') | ||||||
|         b15.save() |         b15.save() | ||||||
|         b16 = Book(pubdate=now, name='bazbaRFOO') |         b16 = Book(pubdate=now, title='bazbaRFOO') | ||||||
|         b16.save() |         b16.save() | ||||||
|  |  | ||||||
|         # alternation |         # alternation | ||||||
|         self.assertQuerysetEqual(Book.objects.filter(name__regex=r'oo(f|b)'), |         self.assertQuerysetEqual(Book.objects.filter(title__regex=r'oo(f|b)'), | ||||||
|             [ |             [ | ||||||
|                 '<Book: barfoobaz>', |                 '<Book: barfoobaz>', | ||||||
|                 '<Book: foobar>', |                 '<Book: foobar>', | ||||||
|                 '<Book: foobarbaz>', |                 '<Book: foobarbaz>', | ||||||
|                 '<Book: foobaz>', |                 '<Book: foobaz>', | ||||||
|             ]) |             ]) | ||||||
|         self.assertQuerysetEqual(Book.objects.filter(name__iregex=r'oo(f|b)'), |         self.assertQuerysetEqual(Book.objects.filter(title__iregex=r'oo(f|b)'), | ||||||
|             [ |             [ | ||||||
|                 '<Book: barfoobaz>', |                 '<Book: barfoobaz>', | ||||||
|                 '<Book: foobar>', |                 '<Book: foobar>', | ||||||
| @@ -571,11 +567,11 @@ class LookupTests(TestCase): | |||||||
|                 '<Book: foobaz>', |                 '<Book: foobaz>', | ||||||
|                 '<Book: ooF>', |                 '<Book: ooF>', | ||||||
|             ]) |             ]) | ||||||
|         self.assertQuerysetEqual(Book.objects.filter(name__regex=r'^foo(f|b)'), |         self.assertQuerysetEqual(Book.objects.filter(title__regex=r'^foo(f|b)'), | ||||||
|             ['<Book: foobar>', '<Book: foobarbaz>', '<Book: foobaz>']) |             ['<Book: foobar>', '<Book: foobarbaz>', '<Book: foobaz>']) | ||||||
|  |  | ||||||
|         # greedy matching |         # greedy matching | ||||||
|         self.assertQuerysetEqual(Book.objects.filter(name__regex=r'b.*az'), |         self.assertQuerysetEqual(Book.objects.filter(title__regex=r'b.*az'), | ||||||
|             [ |             [ | ||||||
|                 '<Book: barfoobaz>', |                 '<Book: barfoobaz>', | ||||||
|                 '<Book: baz>', |                 '<Book: baz>', | ||||||
| @@ -583,7 +579,7 @@ class LookupTests(TestCase): | |||||||
|                 '<Book: foobarbaz>', |                 '<Book: foobarbaz>', | ||||||
|                 '<Book: foobaz>', |                 '<Book: foobaz>', | ||||||
|             ]) |             ]) | ||||||
|         self.assertQuerysetEqual(Book.objects.filter(name__iregex=r'b.*ar'), |         self.assertQuerysetEqual(Book.objects.filter(title__iregex=r'b.*ar'), | ||||||
|             [ |             [ | ||||||
|                 '<Book: bar>', |                 '<Book: bar>', | ||||||
|                 '<Book: barfoobaz>', |                 '<Book: barfoobaz>', | ||||||
| @@ -596,21 +592,21 @@ class LookupTests(TestCase): | |||||||
|     def test_regex_backreferencing(self): |     def test_regex_backreferencing(self): | ||||||
|         # grouping and backreferences |         # grouping and backreferences | ||||||
|         now = datetime.now() |         now = datetime.now() | ||||||
|         b10 = Book(pubdate=now, name='foobar') |         b10 = Book(pubdate=now, title='foobar') | ||||||
|         b10.save() |         b10.save() | ||||||
|         b11 = Book(pubdate=now, name='foobaz') |         b11 = Book(pubdate=now, title='foobaz') | ||||||
|         b11.save() |         b11.save() | ||||||
|         b12 = Book(pubdate=now, name='ooF') |         b12 = Book(pubdate=now, title='ooF') | ||||||
|         b12.save() |         b12.save() | ||||||
|         b13 = Book(pubdate=now, name='foobarbaz') |         b13 = Book(pubdate=now, title='foobarbaz') | ||||||
|         b13.save() |         b13.save() | ||||||
|         b14 = Book(pubdate=now, name='zoocarfaz') |         b14 = Book(pubdate=now, title='zoocarfaz') | ||||||
|         b14.save() |         b14.save() | ||||||
|         b15 = Book(pubdate=now, name='barfoobaz') |         b15 = Book(pubdate=now, title='barfoobaz') | ||||||
|         b15.save() |         b15.save() | ||||||
|         b16 = Book(pubdate=now, name='bazbaRFOO') |         b16 = Book(pubdate=now, title='bazbaRFOO') | ||||||
|         b16.save() |         b16.save() | ||||||
|         self.assertQuerysetEqual(Book.objects.filter(name__regex=r'b(.).*b\1'), |         self.assertQuerysetEqual(Book.objects.filter(title__regex=r'b(.).*b\1'), | ||||||
|             ['<Book: barfoobaz>', '<Book: bazbaRFOO>', '<Book: foobarbaz>']) |             ['<Book: barfoobaz>', '<Book: bazbaRFOO>', '<Book: foobarbaz>']) | ||||||
|  |  | ||||||
|     def test_nonfield_lookups(self): |     def test_nonfield_lookups(self): | ||||||
|   | |||||||
| @@ -16,6 +16,7 @@ from django.db import models | |||||||
| from django.utils import six | from django.utils import six | ||||||
| from django.utils.encoding import python_2_unicode_compatible | from django.utils.encoding import python_2_unicode_compatible | ||||||
|  |  | ||||||
|  | from shared_models.models import Author, Book | ||||||
|  |  | ||||||
| temp_storage_dir = tempfile.mkdtemp(dir=os.environ['DJANGO_TEST_TEMP_DIR']) | temp_storage_dir = tempfile.mkdtemp(dir=os.environ['DJANGO_TEST_TEMP_DIR']) | ||||||
| temp_storage = FileSystemStorage(temp_storage_dir) | temp_storage = FileSystemStorage(temp_storage_dir) | ||||||
| @@ -44,23 +45,13 @@ class Category(models.Model): | |||||||
|     def __repr__(self): |     def __repr__(self): | ||||||
|         return self.__str__() |         return self.__str__() | ||||||
|  |  | ||||||
| @python_2_unicode_compatible |  | ||||||
| class Writer(models.Model): |  | ||||||
|     name = models.CharField(max_length=50, help_text='Use both first and last names.') |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|         ordering = ('name',) |  | ||||||
|  |  | ||||||
|     def __str__(self): |  | ||||||
|         return self.name |  | ||||||
|  |  | ||||||
| @python_2_unicode_compatible | @python_2_unicode_compatible | ||||||
| class Article(models.Model): | class Article(models.Model): | ||||||
|     headline = models.CharField(max_length=50) |     headline = models.CharField(max_length=50) | ||||||
|     slug = models.SlugField() |     slug = models.SlugField() | ||||||
|     pub_date = models.DateField() |     pub_date = models.DateField() | ||||||
|     created = models.DateField(editable=False) |     created = models.DateField(editable=False) | ||||||
|     writer = models.ForeignKey(Writer) |     writer = models.ForeignKey(Author) | ||||||
|     article = models.TextField() |     article = models.TextField() | ||||||
|     categories = models.ManyToManyField(Category, blank=True) |     categories = models.ManyToManyField(Category, blank=True) | ||||||
|     status = models.PositiveIntegerField(choices=ARTICLE_STATUS, blank=True, null=True) |     status = models.PositiveIntegerField(choices=ARTICLE_STATUS, blank=True, null=True) | ||||||
| @@ -80,12 +71,12 @@ class ImprovedArticle(models.Model): | |||||||
| class ImprovedArticleWithParentLink(models.Model): | class ImprovedArticleWithParentLink(models.Model): | ||||||
|     article = models.OneToOneField(Article, parent_link=True) |     article = models.OneToOneField(Article, parent_link=True) | ||||||
|  |  | ||||||
| class BetterWriter(Writer): | class BetterAuthor(Author): | ||||||
|     score = models.IntegerField() |     score = models.IntegerField() | ||||||
|  |  | ||||||
| @python_2_unicode_compatible | @python_2_unicode_compatible | ||||||
| class WriterProfile(models.Model): | class AuthorProfile(models.Model): | ||||||
|     writer = models.OneToOneField(Writer, primary_key=True) |     writer = models.OneToOneField(Author, primary_key=True) | ||||||
|     age = models.PositiveIntegerField() |     age = models.PositiveIntegerField() | ||||||
|  |  | ||||||
|     def __str__(self): |     def __str__(self): | ||||||
| @@ -192,14 +183,6 @@ class Inventory(models.Model): | |||||||
|     def __repr__(self): |     def __repr__(self): | ||||||
|         return self.__str__() |         return self.__str__() | ||||||
|  |  | ||||||
| class Book(models.Model): |  | ||||||
|     title = models.CharField(max_length=40) |  | ||||||
|     author = models.ForeignKey(Writer, blank=True, null=True) |  | ||||||
|     special_id = models.IntegerField(blank=True, null=True, unique=True) |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|         unique_together = ('title', 'author') |  | ||||||
|  |  | ||||||
| class BookXtra(models.Model): | class BookXtra(models.Model): | ||||||
|     isbn = models.CharField(max_length=16, unique=True) |     isbn = models.CharField(max_length=16, unique=True) | ||||||
|     suffix1 = models.IntegerField(blank=True, default=0) |     suffix1 = models.IntegerField(blank=True, default=0) | ||||||
|   | |||||||
| @@ -15,11 +15,13 @@ from django.utils.unittest import skipUnless | |||||||
| from django.test import TestCase | from django.test import TestCase | ||||||
| from django.utils import six | from django.utils import six | ||||||
|  |  | ||||||
| from .models import (Article, ArticleStatus, BetterWriter, BigInt, Book, | from shared_models.models import Author, Book | ||||||
|  |  | ||||||
|  | from .models import (Article, ArticleStatus, BetterAuthor, BigInt, | ||||||
|     Category, CommaSeparatedInteger, CustomFieldForExclusionModel, DerivedBook, |     Category, CommaSeparatedInteger, CustomFieldForExclusionModel, DerivedBook, | ||||||
|     DerivedPost, ExplicitPK, FlexibleDatePost, ImprovedArticle, |     DerivedPost, ExplicitPK, FlexibleDatePost, ImprovedArticle, | ||||||
|     ImprovedArticleWithParentLink, Inventory, Post, Price, |     ImprovedArticleWithParentLink, Inventory, Post, Price, | ||||||
|     Product, TextFile, Writer, WriterProfile, Colour, ColourfulItem, |     Product, TextFile, AuthorProfile, Colour, ColourfulItem, | ||||||
|     test_images) |     test_images) | ||||||
|  |  | ||||||
| if test_images: | if test_images: | ||||||
| @@ -44,11 +46,13 @@ class PriceForm(forms.ModelForm): | |||||||
|  |  | ||||||
| class BookForm(forms.ModelForm): | class BookForm(forms.ModelForm): | ||||||
|     class Meta: |     class Meta: | ||||||
|        model = Book |         fields = ['title', 'author', 'pubdate'] | ||||||
|  |         model = Book | ||||||
|  |  | ||||||
|  |  | ||||||
| class DerivedBookForm(forms.ModelForm): | class DerivedBookForm(forms.ModelForm): | ||||||
|     class Meta: |     class Meta: | ||||||
|  |         fields = ['title', 'author', 'isbn', 'suffix1', 'suffix2'] | ||||||
|         model = DerivedBook |         model = DerivedBook | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -68,11 +72,11 @@ class DerivedPostForm(forms.ModelForm): | |||||||
|         model = DerivedPost |         model = DerivedPost | ||||||
|  |  | ||||||
|  |  | ||||||
| class CustomWriterForm(forms.ModelForm): | class CustomAuthorForm(forms.ModelForm): | ||||||
|    name = forms.CharField(required=False) |    name = forms.CharField(required=False) | ||||||
|  |  | ||||||
|    class Meta: |    class Meta: | ||||||
|        model = Writer |        model = Author | ||||||
|  |  | ||||||
|  |  | ||||||
| class FlexDatePostForm(forms.ModelForm): | class FlexDatePostForm(forms.ModelForm): | ||||||
| @@ -101,7 +105,7 @@ class PartialArticleForm(forms.ModelForm): | |||||||
|  |  | ||||||
| class RoykoForm(forms.ModelForm): | class RoykoForm(forms.ModelForm): | ||||||
|     class Meta: |     class Meta: | ||||||
|         model = Writer |         model = Author | ||||||
|  |  | ||||||
| class TestArticleForm(forms.ModelForm): | class TestArticleForm(forms.ModelForm): | ||||||
|     class Meta: |     class Meta: | ||||||
| @@ -144,13 +148,13 @@ class ImprovedArticleWithParentLinkForm(forms.ModelForm): | |||||||
|     class Meta: |     class Meta: | ||||||
|         model = ImprovedArticleWithParentLink |         model = ImprovedArticleWithParentLink | ||||||
|  |  | ||||||
| class BetterWriterForm(forms.ModelForm): | class BetterAuthorForm(forms.ModelForm): | ||||||
|     class Meta: |     class Meta: | ||||||
|         model = BetterWriter |         model = BetterAuthor | ||||||
|  |  | ||||||
| class WriterProfileForm(forms.ModelForm): | class AuthorProfileForm(forms.ModelForm): | ||||||
|     class Meta: |     class Meta: | ||||||
|         model = WriterProfile |         model = AuthorProfile | ||||||
|  |  | ||||||
| class TextFileForm(forms.ModelForm): | class TextFileForm(forms.ModelForm): | ||||||
|     class Meta: |     class Meta: | ||||||
| @@ -206,13 +210,13 @@ class ModelFormBaseTest(TestCase): | |||||||
|                                      forms.fields.BooleanField)) |                                      forms.fields.BooleanField)) | ||||||
|  |  | ||||||
|     def test_override_field(self): |     def test_override_field(self): | ||||||
|         class WriterForm(forms.ModelForm): |         class AuthorForm(forms.ModelForm): | ||||||
|             book = forms.CharField(required=False) |             book = forms.CharField(required=False) | ||||||
|  |  | ||||||
|             class Meta: |             class Meta: | ||||||
|                 model = Writer |                 model = Author | ||||||
|  |  | ||||||
|         wf = WriterForm({'name': 'Richard Lockridge'}) |         wf = AuthorForm({'name': 'Richard Lockridge'}) | ||||||
|         self.assertTrue(wf.is_valid()) |         self.assertTrue(wf.is_valid()) | ||||||
|  |  | ||||||
|     def test_limit_fields(self): |     def test_limit_fields(self): | ||||||
| @@ -411,7 +415,7 @@ class ValidationTest(TestCase): | |||||||
|         assert form.is_valid() |         assert form.is_valid() | ||||||
|  |  | ||||||
|     def test_notrequired_overrides_notblank(self): |     def test_notrequired_overrides_notblank(self): | ||||||
|         form = CustomWriterForm({}) |         form = CustomAuthorForm({}) | ||||||
|         assert form.is_valid() |         assert form.is_valid() | ||||||
|  |  | ||||||
|  |  | ||||||
| @@ -420,7 +424,7 @@ class ValidationTest(TestCase): | |||||||
| # unique/unique_together validation | # unique/unique_together validation | ||||||
| class UniqueTest(TestCase): | class UniqueTest(TestCase): | ||||||
|     def setUp(self): |     def setUp(self): | ||||||
|         self.writer = Writer.objects.create(name='Mike Royko') |         self.author = Author.objects.create(name='Mike Royko') | ||||||
|  |  | ||||||
|     def test_simple_unique(self): |     def test_simple_unique(self): | ||||||
|         form = ProductForm({'slug': 'teddy-bear-blue'}) |         form = ProductForm({'slug': 'teddy-bear-blue'}) | ||||||
| @@ -444,33 +448,31 @@ class UniqueTest(TestCase): | |||||||
|  |  | ||||||
|     def test_unique_null(self): |     def test_unique_null(self): | ||||||
|         title = 'I May Be Wrong But I Doubt It' |         title = 'I May Be Wrong But I Doubt It' | ||||||
|         form = BookForm({'title': title, 'author': self.writer.pk}) |         form = BookForm({'title': title, 'author': self.author.pk, 'pubdate': '2012-12-12 00:00:00'}) | ||||||
|         self.assertTrue(form.is_valid()) |         self.assertTrue(form.is_valid()) | ||||||
|         form.save() |         form.save() | ||||||
|         form = BookForm({'title': title, 'author': self.writer.pk}) |         form = BookForm({'title': title, 'author': self.author.pk, 'pubdate': '2012-12-12 00:00:00'}) | ||||||
|         self.assertFalse(form.is_valid()) |         self.assertFalse(form.is_valid()) | ||||||
|         self.assertEqual(len(form.errors), 1) |         self.assertEqual(len(form.errors), 1) | ||||||
|         self.assertEqual(form.errors['__all__'], ['Book with this Title and Author already exists.']) |         self.assertEqual(form.errors['__all__'], ['Book with this Title and Author already exists.']) | ||||||
|         form = BookForm({'title': title}) |         form = BookForm({'title': title, 'pubdate': '2012-12-12 00:00:00'}) | ||||||
|         self.assertTrue(form.is_valid()) |         self.assertTrue(form.is_valid()) | ||||||
|         form.save() |         form.save() | ||||||
|         form = BookForm({'title': title}) |         form = BookForm({'title': title, 'pubdate': '2012-12-12 00:00:00'}) | ||||||
|         self.assertTrue(form.is_valid()) |         self.assertTrue(form.is_valid()) | ||||||
|  |  | ||||||
|     def test_inherited_unique(self): |     def test_inherited_unique(self): | ||||||
|         title = 'Boss' |         form = BetterAuthorForm({'name': 'Mike Royko', 'score': 3}) | ||||||
|         Book.objects.create(title=title, author=self.writer, special_id=1) |  | ||||||
|         form = DerivedBookForm({'title': 'Other', 'author': self.writer.pk, 'special_id': '1', 'isbn': '12345'}) |  | ||||||
|         self.assertFalse(form.is_valid()) |         self.assertFalse(form.is_valid()) | ||||||
|         self.assertEqual(len(form.errors), 1) |         self.assertEqual(len(form.errors), 1) | ||||||
|         self.assertEqual(form.errors['special_id'], ['Book with this Special id already exists.']) |         self.assertEqual(form.errors['name'], ['Author with this Name already exists.']) | ||||||
|  |  | ||||||
|     def test_inherited_unique_together(self): |     def test_inherited_unique_together(self): | ||||||
|         title = 'Boss' |         title = 'Boss' | ||||||
|         form = BookForm({'title': title, 'author': self.writer.pk}) |         form = BookForm({'title': title, 'author': self.author.pk, 'pubdate': '2012-12-12 00:00:00'}) | ||||||
|         self.assertTrue(form.is_valid()) |         self.assertTrue(form.is_valid()) | ||||||
|         form.save() |         form.save() | ||||||
|         form = DerivedBookForm({'title': title, 'author': self.writer.pk, 'isbn': '12345'}) |         form = DerivedBookForm({'title': title, 'author': self.author.pk, 'isbn': '12345'}) | ||||||
|         self.assertFalse(form.is_valid()) |         self.assertFalse(form.is_valid()) | ||||||
|         self.assertEqual(len(form.errors), 1) |         self.assertEqual(len(form.errors), 1) | ||||||
|         self.assertEqual(form.errors['__all__'], ['Book with this Title and Author already exists.']) |         self.assertEqual(form.errors['__all__'], ['Book with this Title and Author already exists.']) | ||||||
| @@ -478,8 +480,9 @@ class UniqueTest(TestCase): | |||||||
|     def test_abstract_inherited_unique(self): |     def test_abstract_inherited_unique(self): | ||||||
|         title = 'Boss' |         title = 'Boss' | ||||||
|         isbn = '12345' |         isbn = '12345' | ||||||
|         dbook = DerivedBook.objects.create(title=title, author=self.writer, isbn=isbn) |         dbook = DerivedBook.objects.create(title=title, author=self.author, isbn=isbn, | ||||||
|         form = DerivedBookForm({'title': 'Other', 'author': self.writer.pk, 'isbn': isbn}) |             pubdate='2012-12-12 00:00') | ||||||
|  |         form = DerivedBookForm({'title': 'Other', 'author': self.author.pk, 'isbn': isbn}) | ||||||
|         self.assertFalse(form.is_valid()) |         self.assertFalse(form.is_valid()) | ||||||
|         self.assertEqual(len(form.errors), 1) |         self.assertEqual(len(form.errors), 1) | ||||||
|         self.assertEqual(form.errors['isbn'], ['Derived book with this Isbn already exists.']) |         self.assertEqual(form.errors['isbn'], ['Derived book with this Isbn already exists.']) | ||||||
| @@ -487,10 +490,11 @@ class UniqueTest(TestCase): | |||||||
|     def test_abstract_inherited_unique_together(self): |     def test_abstract_inherited_unique_together(self): | ||||||
|         title = 'Boss' |         title = 'Boss' | ||||||
|         isbn = '12345' |         isbn = '12345' | ||||||
|         dbook = DerivedBook.objects.create(title=title, author=self.writer, isbn=isbn) |         dbook = DerivedBook.objects.create(title=title, author=self.author, isbn=isbn, | ||||||
|  |             pubdate='2012-12-12 00:00') | ||||||
|         form = DerivedBookForm({ |         form = DerivedBookForm({ | ||||||
|                     'title': 'Other', |                     'title': 'Other', | ||||||
|                     'author': self.writer.pk, |                     'author': self.author.pk, | ||||||
|                     'isbn': '9876', |                     'isbn': '9876', | ||||||
|                     'suffix1': '0', |                     'suffix1': '0', | ||||||
|                     'suffix2': '0' |                     'suffix2': '0' | ||||||
| @@ -591,7 +595,7 @@ class ModelToDictTests(TestCase): | |||||||
|         ] |         ] | ||||||
|         for c in categories: |         for c in categories: | ||||||
|             c.save() |             c.save() | ||||||
|         writer = Writer(name='Test writer') |         writer = Author(name='Test writer') | ||||||
|         writer.save() |         writer.save() | ||||||
|  |  | ||||||
|         art = Article( |         art = Article( | ||||||
| @@ -700,10 +704,10 @@ class OldFormForXTests(TestCase): | |||||||
|         with self.assertRaises(ValueError): |         with self.assertRaises(ValueError): | ||||||
|             f.save() |             f.save() | ||||||
|  |  | ||||||
|         # Create a couple of Writers. |         # Create a couple of Authors. | ||||||
|         w_royko = Writer(name='Mike Royko') |         w_royko = Author(name='Mike Royko') | ||||||
|         w_royko.save() |         w_royko.save() | ||||||
|         w_woodward = Writer(name='Bob Woodward') |         w_woodward = Author(name='Bob Woodward') | ||||||
|         w_woodward.save() |         w_woodward.save() | ||||||
|         # ManyToManyFields are represented by a MultipleChoiceField, ForeignKeys and any |         # ManyToManyFields are represented by a MultipleChoiceField, ForeignKeys and any | ||||||
|         # fields with the 'choices' attribute are represented by a ChoiceField. |         # fields with the 'choices' attribute are represented by a ChoiceField. | ||||||
| @@ -741,9 +745,9 @@ class OldFormForXTests(TestCase): | |||||||
|  |  | ||||||
|         # When the ModelForm is passed an instance, that instance's current values are |         # When the ModelForm is passed an instance, that instance's current values are | ||||||
|         # inserted as 'initial' data in each Field. |         # inserted as 'initial' data in each Field. | ||||||
|         w = Writer.objects.get(name='Mike Royko') |         w = Author.objects.get(name='Mike Royko') | ||||||
|         f = RoykoForm(auto_id=False, instance=w) |         f = RoykoForm(auto_id=False, instance=w) | ||||||
|         self.assertHTMLEqual(six.text_type(f), '''<tr><th>Name:</th><td><input type="text" name="name" value="Mike Royko" maxlength="50" /><br /><span class="helptext">Use both first and last names.</span></td></tr>''') |         self.assertHTMLEqual(six.text_type(f), '''<tr><th>Name:</th><td><input type="text" name="name" value="Mike Royko" maxlength="100" /><br /><span class="helptext">Use both first and last names.</span></td></tr>''') | ||||||
|  |  | ||||||
|         art = Article( |         art = Article( | ||||||
|                     headline='Test article', |                     headline='Test article', | ||||||
| @@ -955,7 +959,7 @@ class OldFormForXTests(TestCase): | |||||||
|  |  | ||||||
|         c4 = Category.objects.create(name='Fourth', url='4th') |         c4 = Category.objects.create(name='Fourth', url='4th') | ||||||
|         self.assertEqual(c4.name, 'Fourth') |         self.assertEqual(c4.name, 'Fourth') | ||||||
|         w_bernstein = Writer.objects.create(name='Carl Bernstein') |         w_bernstein = Author.objects.create(name='Carl Bernstein') | ||||||
|         self.assertEqual(w_bernstein.name, 'Carl Bernstein') |         self.assertEqual(w_bernstein.name, 'Carl Bernstein') | ||||||
|         self.assertHTMLEqual(f.as_ul(), '''<li>Headline: <input type="text" name="headline" maxlength="50" /></li> |         self.assertHTMLEqual(f.as_ul(), '''<li>Headline: <input type="text" name="headline" maxlength="50" /></li> | ||||||
| <li>Slug: <input type="text" name="slug" maxlength="50" /></li> | <li>Slug: <input type="text" name="slug" maxlength="50" /></li> | ||||||
| @@ -1132,17 +1136,17 @@ class OldFormForXTests(TestCase): | |||||||
|  |  | ||||||
|         self.assertEqual(list(ImprovedArticleWithParentLinkForm.base_fields), []) |         self.assertEqual(list(ImprovedArticleWithParentLinkForm.base_fields), []) | ||||||
|  |  | ||||||
|         bw = BetterWriter(name='Joe Better', score=10) |         bw = BetterAuthor(name='Joe Better', score=10) | ||||||
|         bw.save() |         bw.save() | ||||||
|         self.assertEqual(sorted(model_to_dict(bw)), |         self.assertEqual(sorted(model_to_dict(bw)), | ||||||
|                          ['id', 'name', 'score', 'writer_ptr']) |                          ['author_ptr', 'id', 'name', 'score']) | ||||||
|  |  | ||||||
|         form = BetterWriterForm({'name': 'Some Name', 'score': 12}) |         form = BetterAuthorForm({'name': 'Some Name', 'score': 12}) | ||||||
|         self.assertEqual(form.is_valid(), True) |         self.assertEqual(form.is_valid(), True) | ||||||
|         bw2 = form.save() |         bw2 = form.save() | ||||||
|         bw2.delete() |         bw2.delete() | ||||||
|  |  | ||||||
|         form = WriterProfileForm() |         form = AuthorProfileForm() | ||||||
|         self.assertHTMLEqual(form.as_p(), '''<p><label for="id_writer">Writer:</label> <select name="writer" id="id_writer"> |         self.assertHTMLEqual(form.as_p(), '''<p><label for="id_writer">Writer:</label> <select name="writer" id="id_writer"> | ||||||
| <option value="" selected="selected">---------</option> | <option value="" selected="selected">---------</option> | ||||||
| <option value="%s">Bob Woodward</option> | <option value="%s">Bob Woodward</option> | ||||||
| @@ -1156,11 +1160,11 @@ class OldFormForXTests(TestCase): | |||||||
|             'writer': six.text_type(w_woodward.pk), |             'writer': six.text_type(w_woodward.pk), | ||||||
|             'age': '65', |             'age': '65', | ||||||
|         } |         } | ||||||
|         form = WriterProfileForm(data) |         form = AuthorProfileForm(data) | ||||||
|         instance = form.save() |         instance = form.save() | ||||||
|         self.assertEqual(six.text_type(instance), 'Bob Woodward is 65') |         self.assertEqual(six.text_type(instance), 'Bob Woodward is 65') | ||||||
|  |  | ||||||
|         form = WriterProfileForm(instance=instance) |         form = AuthorProfileForm(instance=instance) | ||||||
|         self.assertHTMLEqual(form.as_p(), '''<p><label for="id_writer">Writer:</label> <select name="writer" id="id_writer"> |         self.assertHTMLEqual(form.as_p(), '''<p><label for="id_writer">Writer:</label> <select name="writer" id="id_writer"> | ||||||
| <option value="">---------</option> | <option value="">---------</option> | ||||||
| <option value="%s" selected="selected">Bob Woodward</option> | <option value="%s" selected="selected">Bob Woodward</option> | ||||||
|   | |||||||
| @@ -1,3 +1,5 @@ | |||||||
|  | from __future__ import unicode_literals | ||||||
|  |  | ||||||
| from django.db import models | from django.db import models | ||||||
| from django.utils import timezone | from django.utils import timezone | ||||||
| from django.utils.encoding import python_2_unicode_compatible | from django.utils.encoding import python_2_unicode_compatible | ||||||
| @@ -9,7 +11,11 @@ class Tag(models.Model): | |||||||
|  |  | ||||||
| @python_2_unicode_compatible | @python_2_unicode_compatible | ||||||
| class Author(models.Model): | class Author(models.Model): | ||||||
|     name = models.CharField(max_length=100) |     name = models.CharField(max_length=100, help_text='Use both first and last names.', | ||||||
|  |         unique=True) | ||||||
|  |  | ||||||
|  |     class Meta: | ||||||
|  |         ordering = ['name'] | ||||||
|  |  | ||||||
|     def __str__(self): |     def __str__(self): | ||||||
|         return self.name |         return self.name | ||||||
| @@ -17,14 +23,15 @@ class Author(models.Model): | |||||||
|  |  | ||||||
| @python_2_unicode_compatible | @python_2_unicode_compatible | ||||||
| class Book(models.Model): | class Book(models.Model): | ||||||
|     name = models.CharField(max_length=200) |     title = models.CharField(max_length=200) | ||||||
|     pages = models.IntegerField(default=0) |     pages = models.IntegerField(default=0) | ||||||
|     author = models.ForeignKey(Author, null=True) |     author = models.ForeignKey(Author, null=True, blank=True) | ||||||
|     pubdate = models.DateTimeField() |     pubdate = models.DateTimeField() | ||||||
|     tags = models.ManyToManyField(Tag) |     tags = models.ManyToManyField(Tag) | ||||||
|  |  | ||||||
|     class Meta: |     class Meta: | ||||||
|         ordering = ['-pubdate', 'name'] |         ordering = ['-pubdate', 'title'] | ||||||
|  |         unique_together = ['title', 'author'] | ||||||
|  |  | ||||||
|     def __str__(self): |     def __str__(self): | ||||||
|         return self.name |         return self.title | ||||||
|   | |||||||
| @@ -77,7 +77,7 @@ class SignalsRegressTests(TestCase): | |||||||
|             "Is created" |             "Is created" | ||||||
|         ]) |         ]) | ||||||
|  |  | ||||||
|         b1 = Book(name='Snow Crash', pubdate='2012-02-02 12:00') |         b1 = Book(title='Snow Crash', pubdate='2012-02-02 12:00') | ||||||
|         self.assertEqual(self.get_signal_output(b1.save), [ |         self.assertEqual(self.get_signal_output(b1.save), [ | ||||||
|             "pre_save signal, Snow Crash", |             "pre_save signal, Snow Crash", | ||||||
|             "post_save signal, Snow Crash", |             "post_save signal, Snow Crash", | ||||||
| @@ -87,7 +87,7 @@ class SignalsRegressTests(TestCase): | |||||||
|     def test_m2m_signals(self): |     def test_m2m_signals(self): | ||||||
|         """ Assigning and removing to/from m2m shouldn't generate an m2m signal """ |         """ Assigning and removing to/from m2m shouldn't generate an m2m signal """ | ||||||
|  |  | ||||||
|         b1 = Book(name='Snow Crash', pubdate='2012-02-02 12:00') |         b1 = Book(title='Snow Crash', pubdate='2012-02-02 12:00') | ||||||
|         self.get_signal_output(b1.save) |         self.get_signal_output(b1.save) | ||||||
|         a1 = Author(name='Neal Stephenson') |         a1 = Author(name='Neal Stephenson') | ||||||
|         self.get_signal_output(a1.save) |         self.get_signal_output(a1.save) | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user