mirror of
				https://github.com/django/django.git
				synced 2025-10-26 07:06:08 +00:00 
			
		
		
		
	[1.8.x] Cleaned up schema tests
Thanks Tim Graham for the review.
Backport of 0204714b0b from master
			
			
This commit is contained in:
		| @@ -52,3 +52,7 @@ class CustomManyToManyField(RelatedField): | |||||||
|     _get_m2m_attr = ManyToManyField.__dict__['_get_m2m_attr'] |     _get_m2m_attr = ManyToManyField.__dict__['_get_m2m_attr'] | ||||||
|     _get_m2m_reverse_attr = ManyToManyField.__dict__['_get_m2m_reverse_attr'] |     _get_m2m_reverse_attr = ManyToManyField.__dict__['_get_m2m_reverse_attr'] | ||||||
|     _get_m2m_db_table = ManyToManyField.__dict__['_get_m2m_db_table'] |     _get_m2m_db_table = ManyToManyField.__dict__['_get_m2m_db_table'] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class InheritedManyToManyField(ManyToManyField): | ||||||
|  |     pass | ||||||
|   | |||||||
| @@ -25,24 +25,9 @@ class AuthorWithDefaultHeight(models.Model): | |||||||
|         apps = new_apps |         apps = new_apps | ||||||
|  |  | ||||||
|  |  | ||||||
| class AuthorWithM2M(models.Model): | class AuthorWithEvenLongerName(models.Model): | ||||||
|     name = models.CharField(max_length=255) |     name = models.CharField(max_length=255) | ||||||
|  |     height = models.PositiveIntegerField(null=True, blank=True) | ||||||
|     class Meta: |  | ||||||
|         apps = new_apps |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class AuthorWithM2MThrough(models.Model): |  | ||||||
|     name = models.CharField(max_length=255) |  | ||||||
|     tags = models.ManyToManyField("schema.TagM2MTest", related_name="authors", through="AuthorTag") |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|         apps = new_apps |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class AuthorTag(models.Model): |  | ||||||
|     author = models.ForeignKey("schema.AuthorWithM2MThrough") |  | ||||||
|     tag = models.ForeignKey("schema.TagM2MTest") |  | ||||||
|  |  | ||||||
|     class Meta: |     class Meta: | ||||||
|         apps = new_apps |         apps = new_apps | ||||||
| @@ -67,6 +52,13 @@ class BookWeak(models.Model): | |||||||
|         apps = new_apps |         apps = new_apps | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class BookWithLongName(models.Model): | ||||||
|  |     author_foreign_key_with_really_long_field_name = models.ForeignKey(AuthorWithEvenLongerName) | ||||||
|  |  | ||||||
|  |     class Meta: | ||||||
|  |         apps = new_apps | ||||||
|  |  | ||||||
|  |  | ||||||
| class BookWithO2O(models.Model): | class BookWithO2O(models.Model): | ||||||
|     author = models.OneToOneField(Author) |     author = models.OneToOneField(Author) | ||||||
|     title = models.CharField(max_length=100, db_index=True) |     title = models.CharField(max_length=100, db_index=True) | ||||||
| @@ -77,31 +69,6 @@ class BookWithO2O(models.Model): | |||||||
|         db_table = "schema_book" |         db_table = "schema_book" | ||||||
|  |  | ||||||
|  |  | ||||||
| class BookWithM2M(models.Model): |  | ||||||
|     author = models.ForeignKey(Author) |  | ||||||
|     title = models.CharField(max_length=100, db_index=True) |  | ||||||
|     pub_date = models.DateTimeField() |  | ||||||
|     tags = models.ManyToManyField("TagM2MTest", related_name="books") |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|         apps = new_apps |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class TagThrough(models.Model): |  | ||||||
|     book = models.ForeignKey("schema.BookWithM2MThrough") |  | ||||||
|     tag = models.ForeignKey("schema.TagM2MTest") |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|         apps = new_apps |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class BookWithM2MThrough(models.Model): |  | ||||||
|     tags = models.ManyToManyField("TagM2MTest", related_name="books", through=TagThrough) |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|         apps = new_apps |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class BookWithSlug(models.Model): | class BookWithSlug(models.Model): | ||||||
|     author = models.ForeignKey(Author) |     author = models.ForeignKey(Author) | ||||||
|     title = models.CharField(max_length=100, db_index=True) |     title = models.CharField(max_length=100, db_index=True) | ||||||
| @@ -113,6 +80,10 @@ class BookWithSlug(models.Model): | |||||||
|         db_table = "schema_book" |         db_table = "schema_book" | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class Note(models.Model): | ||||||
|  |     info = models.TextField() | ||||||
|  |  | ||||||
|  |  | ||||||
| class Tag(models.Model): | class Tag(models.Model): | ||||||
|     title = models.CharField(max_length=255) |     title = models.CharField(max_length=255) | ||||||
|     slug = models.SlugField(unique=True) |     slug = models.SlugField(unique=True) | ||||||
| @@ -121,14 +92,6 @@ class Tag(models.Model): | |||||||
|         apps = new_apps |         apps = new_apps | ||||||
|  |  | ||||||
|  |  | ||||||
| class TagM2MTest(models.Model): |  | ||||||
|     title = models.CharField(max_length=255) |  | ||||||
|     slug = models.SlugField(unique=True) |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|         apps = new_apps |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class TagIndexed(models.Model): | class TagIndexed(models.Model): | ||||||
|     title = models.CharField(max_length=255) |     title = models.CharField(max_length=255) | ||||||
|     slug = models.SlugField(unique=True) |     slug = models.SlugField(unique=True) | ||||||
| @@ -138,6 +101,14 @@ class TagIndexed(models.Model): | |||||||
|         index_together = [["slug", "title"]] |         index_together = [["slug", "title"]] | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class TagM2MTest(models.Model): | ||||||
|  |     title = models.CharField(max_length=255) | ||||||
|  |     slug = models.SlugField(unique=True) | ||||||
|  |  | ||||||
|  |     class Meta: | ||||||
|  |         apps = new_apps | ||||||
|  |  | ||||||
|  |  | ||||||
| class TagUniqueRename(models.Model): | class TagUniqueRename(models.Model): | ||||||
|     title = models.CharField(max_length=255) |     title = models.CharField(max_length=255) | ||||||
|     slug2 = models.SlugField(unique=True) |     slug2 = models.SlugField(unique=True) | ||||||
| @@ -147,30 +118,6 @@ class TagUniqueRename(models.Model): | |||||||
|         db_table = "schema_tag" |         db_table = "schema_tag" | ||||||
|  |  | ||||||
|  |  | ||||||
| class UniqueTest(models.Model): |  | ||||||
|     year = models.IntegerField() |  | ||||||
|     slug = models.SlugField(unique=False) |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|         apps = new_apps |  | ||||||
|         unique_together = ["year", "slug"] |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class AuthorWithEvenLongerName(models.Model): |  | ||||||
|     name = models.CharField(max_length=255) |  | ||||||
|     height = models.PositiveIntegerField(null=True, blank=True) |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|         apps = new_apps |  | ||||||
|  |  | ||||||
|  |  | ||||||
| class BookWithLongName(models.Model): |  | ||||||
|     author_foreign_key_with_really_long_field_name = models.ForeignKey(AuthorWithEvenLongerName) |  | ||||||
|  |  | ||||||
|     class Meta: |  | ||||||
|         apps = new_apps |  | ||||||
|  |  | ||||||
|  |  | ||||||
| # Based on tests/reserved_names/models.py | # Based on tests/reserved_names/models.py | ||||||
| @python_2_unicode_compatible | @python_2_unicode_compatible | ||||||
| class Thing(models.Model): | class Thing(models.Model): | ||||||
| @@ -183,5 +130,10 @@ class Thing(models.Model): | |||||||
|         return self.when |         return self.when | ||||||
|  |  | ||||||
|  |  | ||||||
| class Note(models.Model): | class UniqueTest(models.Model): | ||||||
|     info = models.TextField() |     year = models.IntegerField() | ||||||
|  |     slug = models.SlugField(unique=False) | ||||||
|  |  | ||||||
|  |     class Meta: | ||||||
|  |         apps = new_apps | ||||||
|  |         unique_together = ["year", "slug"] | ||||||
|   | |||||||
| @@ -1,17 +1,21 @@ | |||||||
| import datetime | import datetime | ||||||
|  | import itertools | ||||||
| import unittest | import unittest | ||||||
|  |  | ||||||
| from django.test import TransactionTestCase |  | ||||||
| from django.db import connection, DatabaseError, IntegrityError, OperationalError | from django.db import connection, DatabaseError, IntegrityError, OperationalError | ||||||
| from django.db.models.fields import (BinaryField, BooleanField, CharField, IntegerField, | from django.db.models import Model | ||||||
|     PositiveIntegerField, SlugField, TextField) | from django.db.models.fields import (BinaryField, BooleanField, CharField, DateTimeField, | ||||||
|  |     IntegerField, PositiveIntegerField, SlugField, TextField) | ||||||
| from django.db.models.fields.related import ForeignKey, ManyToManyField, OneToOneField | from django.db.models.fields.related import ForeignKey, ManyToManyField, OneToOneField | ||||||
| from django.db.transaction import atomic | from django.db.transaction import atomic | ||||||
| from .fields import CustomManyToManyField | from django.test import TransactionTestCase | ||||||
| from .models import (Author, AuthorWithDefaultHeight, AuthorWithM2M, Book, BookWithLongName, |  | ||||||
|     BookWithSlug, BookWithM2M, Tag, TagIndexed, TagM2MTest, TagUniqueRename, | from .fields import CustomManyToManyField, InheritedManyToManyField | ||||||
|     UniqueTest, Thing, TagThrough, BookWithM2MThrough, AuthorTag, AuthorWithM2MThrough, | from .models import ( | ||||||
|     AuthorWithEvenLongerName, BookWeak, Note, BookWithO2O) |     Author, AuthorWithDefaultHeight, AuthorWithEvenLongerName, Book, BookWeak, | ||||||
|  |     BookWithLongName, BookWithO2O, BookWithSlug, Note, Tag, TagIndexed, | ||||||
|  |     TagM2MTest, TagUniqueRename, Thing, UniqueTest, new_apps | ||||||
|  | ) | ||||||
|  |  | ||||||
|  |  | ||||||
| class SchemaTests(TransactionTestCase): | class SchemaTests(TransactionTestCase): | ||||||
| @@ -26,24 +30,34 @@ class SchemaTests(TransactionTestCase): | |||||||
|     available_apps = [] |     available_apps = [] | ||||||
|  |  | ||||||
|     models = [ |     models = [ | ||||||
|         Author, AuthorWithM2M, Book, BookWithLongName, BookWithSlug, |         Author, AuthorWithDefaultHeight, AuthorWithEvenLongerName, Book, | ||||||
|         BookWithM2M, Tag, TagIndexed, TagM2MTest, TagUniqueRename, UniqueTest, |         BookWeak, BookWithLongName, BookWithO2O, BookWithSlug, Note, Tag, | ||||||
|         Thing, TagThrough, BookWithM2MThrough, AuthorWithEvenLongerName, |         TagIndexed, TagM2MTest, TagUniqueRename, Thing, UniqueTest, | ||||||
|         BookWeak, BookWithO2O, |  | ||||||
|     ] |     ] | ||||||
|  |  | ||||||
|     # Utility functions |     # Utility functions | ||||||
|  |  | ||||||
|  |     def setUp(self): | ||||||
|  |         # local_models should contain test dependent model classes that will be | ||||||
|  |         # automatically removed from the app cache on test tear down. | ||||||
|  |         self.local_models = [] | ||||||
|  |  | ||||||
|     def tearDown(self): |     def tearDown(self): | ||||||
|         # Delete any tables made for our models |         # Delete any tables made for our models | ||||||
|         self.delete_tables() |         self.delete_tables() | ||||||
|  |         new_apps.clear_cache() | ||||||
|  |         for model in new_apps.get_models(): | ||||||
|  |             model._meta._expire_cache() | ||||||
|  |         if 'schema' in new_apps.all_models: | ||||||
|  |             for model in self.local_models: | ||||||
|  |                 del new_apps.all_models['schema'][model._meta.model_name] | ||||||
|  |  | ||||||
|     def delete_tables(self): |     def delete_tables(self): | ||||||
|         "Deletes all model tables for our models for a clean test environment" |         "Deletes all model tables for our models for a clean test environment" | ||||||
|         with connection.cursor() as cursor: |         with connection.cursor() as cursor: | ||||||
|             connection.disable_constraint_checking() |             connection.disable_constraint_checking() | ||||||
|             table_names = connection.introspection.table_names(cursor) |             table_names = connection.introspection.table_names(cursor) | ||||||
|             for model in self.models: |             for model in itertools.chain(SchemaTests.models, self.local_models): | ||||||
|                 # Remove any M2M tables first |                 # Remove any M2M tables first | ||||||
|                 for field in model._meta.local_many_to_many: |                 for field in model._meta.local_many_to_many: | ||||||
|                     with atomic(): |                     with atomic(): | ||||||
| @@ -134,15 +148,11 @@ class SchemaTests(TransactionTestCase): | |||||||
|                 pub_date=datetime.datetime.now(), |                 pub_date=datetime.datetime.now(), | ||||||
|             ) |             ) | ||||||
|         # Repoint the FK constraint |         # Repoint the FK constraint | ||||||
|  |         old_field = Book._meta.get_field("author") | ||||||
|         new_field = ForeignKey(Tag) |         new_field = ForeignKey(Tag) | ||||||
|         new_field.set_attributes_from_name("author") |         new_field.set_attributes_from_name("author") | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.alter_field( |             editor.alter_field(Book, old_field, new_field, strict=True) | ||||||
|                 Book, |  | ||||||
|                 Book._meta.get_field("author"), |  | ||||||
|                 new_field, |  | ||||||
|                 strict=True, |  | ||||||
|             ) |  | ||||||
|         # Make sure the new FK constraint is present |         # Make sure the new FK constraint is present | ||||||
|         constraints = self.get_constraints(Book._meta.db_table) |         constraints = self.get_constraints(Book._meta.db_table) | ||||||
|         for name, details in constraints.items(): |         for name, details in constraints.items(): | ||||||
| @@ -173,25 +183,17 @@ class SchemaTests(TransactionTestCase): | |||||||
|         new_field = ForeignKey(Tag, db_constraint=False) |         new_field = ForeignKey(Tag, db_constraint=False) | ||||||
|         new_field.set_attributes_from_name("tag") |         new_field.set_attributes_from_name("tag") | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.add_field( |             editor.add_field(Author, new_field) | ||||||
|                 Author, |  | ||||||
|                 new_field, |  | ||||||
|             ) |  | ||||||
|         # Make sure no FK constraint is present |         # Make sure no FK constraint is present | ||||||
|         constraints = self.get_constraints(Author._meta.db_table) |         constraints = self.get_constraints(Author._meta.db_table) | ||||||
|         for name, details in constraints.items(): |         for name, details in constraints.items(): | ||||||
|             if details['columns'] == ["tag_id"] and details['foreign_key']: |             if details['columns'] == ["tag_id"] and details['foreign_key']: | ||||||
|                 self.fail("FK constraint for tag_id found") |                 self.fail("FK constraint for tag_id found") | ||||||
|         # Alter to one with a constraint |         # Alter to one with a constraint | ||||||
|         new_field_2 = ForeignKey(Tag) |         new_field2 = ForeignKey(Tag) | ||||||
|         new_field_2.set_attributes_from_name("tag") |         new_field2.set_attributes_from_name("tag") | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.alter_field( |             editor.alter_field(Author, new_field, new_field2, strict=True) | ||||||
|                 Author, |  | ||||||
|                 new_field, |  | ||||||
|                 new_field_2, |  | ||||||
|                 strict=True, |  | ||||||
|             ) |  | ||||||
|         # Make sure the new FK constraint is present |         # Make sure the new FK constraint is present | ||||||
|         constraints = self.get_constraints(Author._meta.db_table) |         constraints = self.get_constraints(Author._meta.db_table) | ||||||
|         for name, details in constraints.items(): |         for name, details in constraints.items(): | ||||||
| @@ -201,45 +203,56 @@ class SchemaTests(TransactionTestCase): | |||||||
|         else: |         else: | ||||||
|             self.fail("No FK constraint for tag_id found") |             self.fail("No FK constraint for tag_id found") | ||||||
|         # Alter to one without a constraint again |         # Alter to one without a constraint again | ||||||
|         new_field_2 = ForeignKey(Tag) |         new_field2 = ForeignKey(Tag) | ||||||
|         new_field_2.set_attributes_from_name("tag") |         new_field2.set_attributes_from_name("tag") | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.alter_field( |             editor.alter_field(Author, new_field2, new_field, strict=True) | ||||||
|                 Author, |  | ||||||
|                 new_field_2, |  | ||||||
|                 new_field, |  | ||||||
|                 strict=True, |  | ||||||
|             ) |  | ||||||
|         # Make sure no FK constraint is present |         # Make sure no FK constraint is present | ||||||
|         constraints = self.get_constraints(Author._meta.db_table) |         constraints = self.get_constraints(Author._meta.db_table) | ||||||
|         for name, details in constraints.items(): |         for name, details in constraints.items(): | ||||||
|             if details['columns'] == ["tag_id"] and details['foreign_key']: |             if details['columns'] == ["tag_id"] and details['foreign_key']: | ||||||
|                 self.fail("FK constraint for tag_id found") |                 self.fail("FK constraint for tag_id found") | ||||||
|  |  | ||||||
|     @unittest.skipUnless(connection.features.supports_foreign_keys, "No FK support") |     def _test_m2m_db_constraint(self, M2MFieldClass): | ||||||
|     def test_m2m_db_constraint(self): |         class LocalAuthorWithM2M(Model): | ||||||
|  |             name = CharField(max_length=255) | ||||||
|  |  | ||||||
|  |             class Meta: | ||||||
|  |                 apps = new_apps | ||||||
|  |  | ||||||
|  |         self.local_models = [LocalAuthorWithM2M] | ||||||
|  |  | ||||||
|         # Create the table |         # Create the table | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.create_model(Tag) |             editor.create_model(Tag) | ||||||
|             editor.create_model(Author) |             editor.create_model(LocalAuthorWithM2M) | ||||||
|         # Check that initial tables are there |         # Check that initial tables are there | ||||||
|         list(Author.objects.all()) |         list(LocalAuthorWithM2M.objects.all()) | ||||||
|         list(Tag.objects.all()) |         list(Tag.objects.all()) | ||||||
|         # Make a db_constraint=False FK |         # Make a db_constraint=False FK | ||||||
|         new_field = ManyToManyField("schema.Tag", related_name="authors", db_constraint=False) |         new_field = M2MFieldClass(Tag, related_name="authors", db_constraint=False) | ||||||
|         new_field.contribute_to_class(Author, "tags") |         new_field.contribute_to_class(LocalAuthorWithM2M, "tags") | ||||||
|         # Add the field |         # Add the field | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.add_field( |             editor.add_field(LocalAuthorWithM2M, new_field) | ||||||
|                 Author, |  | ||||||
|                 new_field, |  | ||||||
|             ) |  | ||||||
|         # Make sure no FK constraint is present |         # Make sure no FK constraint is present | ||||||
|         constraints = self.get_constraints(new_field.rel.through._meta.db_table) |         constraints = self.get_constraints(new_field.rel.through._meta.db_table) | ||||||
|         for name, details in constraints.items(): |         for name, details in constraints.items(): | ||||||
|             if details['columns'] == ["tag_id"] and details['foreign_key']: |             if details['columns'] == ["tag_id"] and details['foreign_key']: | ||||||
|                 self.fail("FK constraint for tag_id found") |                 self.fail("FK constraint for tag_id found") | ||||||
|  |  | ||||||
|  |     @unittest.skipUnless(connection.features.supports_foreign_keys, "No FK support") | ||||||
|  |     def test_m2m_db_constraint(self): | ||||||
|  |         self._test_m2m_db_constraint(ManyToManyField) | ||||||
|  |  | ||||||
|  |     @unittest.skipUnless(connection.features.supports_foreign_keys, "No FK support") | ||||||
|  |     def test_m2m_db_constraint_custom(self): | ||||||
|  |         self._test_m2m_db_constraint(CustomManyToManyField) | ||||||
|  |  | ||||||
|  |     @unittest.skipUnless(connection.features.supports_foreign_keys, "No FK support") | ||||||
|  |     def test_m2m_db_constraint_inherited(self): | ||||||
|  |         self._test_m2m_db_constraint(InheritedManyToManyField) | ||||||
|  |  | ||||||
|     def test_add_field(self): |     def test_add_field(self): | ||||||
|         """ |         """ | ||||||
|         Tests adding fields to models |         Tests adding fields to models | ||||||
| @@ -254,10 +267,7 @@ class SchemaTests(TransactionTestCase): | |||||||
|         new_field = IntegerField(null=True) |         new_field = IntegerField(null=True) | ||||||
|         new_field.set_attributes_from_name("age") |         new_field.set_attributes_from_name("age") | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.add_field( |             editor.add_field(Author, new_field) | ||||||
|                 Author, |  | ||||||
|                 new_field, |  | ||||||
|             ) |  | ||||||
|         # Ensure the field is right afterwards |         # Ensure the field is right afterwards | ||||||
|         columns = self.column_classes(Author) |         columns = self.column_classes(Author) | ||||||
|         self.assertEqual(columns['age'][0], "IntegerField") |         self.assertEqual(columns['age'][0], "IntegerField") | ||||||
| @@ -280,10 +290,7 @@ class SchemaTests(TransactionTestCase): | |||||||
|         new_field = CharField(max_length=30, default="Godwin") |         new_field = CharField(max_length=30, default="Godwin") | ||||||
|         new_field.set_attributes_from_name("surname") |         new_field.set_attributes_from_name("surname") | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.add_field( |             editor.add_field(Author, new_field) | ||||||
|                 Author, |  | ||||||
|                 new_field, |  | ||||||
|             ) |  | ||||||
|         # Ensure the field is right afterwards |         # Ensure the field is right afterwards | ||||||
|         columns = self.column_classes(Author) |         columns = self.column_classes(Author) | ||||||
|         self.assertEqual(columns['surname'][0], "CharField") |         self.assertEqual(columns['surname'][0], "CharField") | ||||||
| @@ -308,10 +315,7 @@ class SchemaTests(TransactionTestCase): | |||||||
|         new_field = BooleanField(default=False) |         new_field = BooleanField(default=False) | ||||||
|         new_field.set_attributes_from_name("awesome") |         new_field.set_attributes_from_name("awesome") | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.add_field( |             editor.add_field(Author, new_field) | ||||||
|                 Author, |  | ||||||
|                 new_field, |  | ||||||
|             ) |  | ||||||
|         # Ensure the field is right afterwards |         # Ensure the field is right afterwards | ||||||
|         columns = self.column_classes(Author) |         columns = self.column_classes(Author) | ||||||
|         # BooleanField are stored as TINYINT(1) on MySQL. |         # BooleanField are stored as TINYINT(1) on MySQL. | ||||||
| @@ -345,10 +349,7 @@ class SchemaTests(TransactionTestCase): | |||||||
|         new_field = TestTransformField(default={1: 2}) |         new_field = TestTransformField(default={1: 2}) | ||||||
|         new_field.set_attributes_from_name("thing") |         new_field.set_attributes_from_name("thing") | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.add_field( |             editor.add_field(Author, new_field) | ||||||
|                 Author, |  | ||||||
|                 new_field, |  | ||||||
|             ) |  | ||||||
|         # Ensure the field is there |         # Ensure the field is there | ||||||
|         columns = self.column_classes(Author) |         columns = self.column_classes(Author) | ||||||
|         field_type, field_info = columns['thing'] |         field_type, field_info = columns['thing'] | ||||||
| @@ -367,10 +368,7 @@ class SchemaTests(TransactionTestCase): | |||||||
|         new_field = BinaryField(blank=True) |         new_field = BinaryField(blank=True) | ||||||
|         new_field.set_attributes_from_name("bits") |         new_field.set_attributes_from_name("bits") | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.add_field( |             editor.add_field(Author, new_field) | ||||||
|                 Author, |  | ||||||
|                 new_field, |  | ||||||
|             ) |  | ||||||
|         # Ensure the field is right afterwards |         # Ensure the field is right afterwards | ||||||
|         columns = self.column_classes(Author) |         columns = self.column_classes(Author) | ||||||
|         # MySQL annoyingly uses the same backend, so it'll come back as one of |         # MySQL annoyingly uses the same backend, so it'll come back as one of | ||||||
| @@ -389,15 +387,11 @@ class SchemaTests(TransactionTestCase): | |||||||
|         self.assertEqual(columns['name'][0], "CharField") |         self.assertEqual(columns['name'][0], "CharField") | ||||||
|         self.assertEqual(bool(columns['name'][1][6]), bool(connection.features.interprets_empty_strings_as_nulls)) |         self.assertEqual(bool(columns['name'][1][6]), bool(connection.features.interprets_empty_strings_as_nulls)) | ||||||
|         # Alter the name field to a TextField |         # Alter the name field to a TextField | ||||||
|  |         old_field = Author._meta.get_field("name") | ||||||
|         new_field = TextField(null=True) |         new_field = TextField(null=True) | ||||||
|         new_field.set_attributes_from_name("name") |         new_field.set_attributes_from_name("name") | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.alter_field( |             editor.alter_field(Author, old_field, new_field, strict=True) | ||||||
|                 Author, |  | ||||||
|                 Author._meta.get_field("name"), |  | ||||||
|                 new_field, |  | ||||||
|                 strict=True, |  | ||||||
|             ) |  | ||||||
|         # Ensure the field is right afterwards |         # Ensure the field is right afterwards | ||||||
|         columns = self.column_classes(Author) |         columns = self.column_classes(Author) | ||||||
|         self.assertEqual(columns['name'][0], "TextField") |         self.assertEqual(columns['name'][0], "TextField") | ||||||
| @@ -406,12 +400,7 @@ class SchemaTests(TransactionTestCase): | |||||||
|         new_field2 = TextField(null=False) |         new_field2 = TextField(null=False) | ||||||
|         new_field2.set_attributes_from_name("name") |         new_field2.set_attributes_from_name("name") | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.alter_field( |             editor.alter_field(Author, new_field, new_field2, strict=True) | ||||||
|                 Author, |  | ||||||
|                 new_field, |  | ||||||
|                 new_field2, |  | ||||||
|                 strict=True, |  | ||||||
|             ) |  | ||||||
|         # Ensure the field is right afterwards |         # Ensure the field is right afterwards | ||||||
|         columns = self.column_classes(Author) |         columns = self.column_classes(Author) | ||||||
|         self.assertEqual(columns['name'][0], "TextField") |         self.assertEqual(columns['name'][0], "TextField") | ||||||
| @@ -420,15 +409,14 @@ class SchemaTests(TransactionTestCase): | |||||||
|     def test_alter_text_field(self): |     def test_alter_text_field(self): | ||||||
|         # Regression for "BLOB/TEXT column 'info' can't have a default value") |         # Regression for "BLOB/TEXT column 'info' can't have a default value") | ||||||
|         # on MySQL. |         # on MySQL. | ||||||
|  |         # Create the table | ||||||
|  |         with connection.schema_editor() as editor: | ||||||
|  |             editor.create_model(Note) | ||||||
|  |         old_field = Note._meta.get_field("info") | ||||||
|         new_field = TextField(blank=True) |         new_field = TextField(blank=True) | ||||||
|         new_field.set_attributes_from_name("info") |         new_field.set_attributes_from_name("info") | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.alter_field( |             editor.alter_field(Note, old_field, new_field, strict=True) | ||||||
|                 Note, |  | ||||||
|                 Note._meta.get_field("info"), |  | ||||||
|                 new_field, |  | ||||||
|                 strict=True, |  | ||||||
|             ) |  | ||||||
|  |  | ||||||
|     def test_alter_null_to_not_null(self): |     def test_alter_null_to_not_null(self): | ||||||
|         """ |         """ | ||||||
| @@ -447,14 +435,11 @@ class SchemaTests(TransactionTestCase): | |||||||
|         self.assertEqual(Author.objects.get(name='Not null author').height, 12) |         self.assertEqual(Author.objects.get(name='Not null author').height, 12) | ||||||
|         self.assertIsNone(Author.objects.get(name='Null author').height) |         self.assertIsNone(Author.objects.get(name='Null author').height) | ||||||
|         # Alter the height field to NOT NULL with default |         # Alter the height field to NOT NULL with default | ||||||
|  |         old_field = Author._meta.get_field("height") | ||||||
|         new_field = PositiveIntegerField(default=42) |         new_field = PositiveIntegerField(default=42) | ||||||
|         new_field.set_attributes_from_name("height") |         new_field.set_attributes_from_name("height") | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.alter_field( |             editor.alter_field(Author, old_field, new_field) | ||||||
|                 Author, |  | ||||||
|                 Author._meta.get_field("height"), |  | ||||||
|                 new_field |  | ||||||
|             ) |  | ||||||
|         # Ensure the field is right afterwards |         # Ensure the field is right afterwards | ||||||
|         columns = self.column_classes(Author) |         columns = self.column_classes(Author) | ||||||
|         self.assertFalse(columns['height'][1][6]) |         self.assertFalse(columns['height'][1][6]) | ||||||
| @@ -475,14 +460,11 @@ class SchemaTests(TransactionTestCase): | |||||||
|         columns = self.column_classes(AuthorWithDefaultHeight) |         columns = self.column_classes(AuthorWithDefaultHeight) | ||||||
|         self.assertTrue(columns['height'][1][6]) |         self.assertTrue(columns['height'][1][6]) | ||||||
|         # Alter the height field to NOT NULL keeping the previous default |         # Alter the height field to NOT NULL keeping the previous default | ||||||
|  |         old_field = AuthorWithDefaultHeight._meta.get_field("height") | ||||||
|         new_field = PositiveIntegerField(default=42) |         new_field = PositiveIntegerField(default=42) | ||||||
|         new_field.set_attributes_from_name("height") |         new_field.set_attributes_from_name("height") | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.alter_field( |             editor.alter_field(AuthorWithDefaultHeight, old_field, new_field) | ||||||
|                 AuthorWithDefaultHeight, |  | ||||||
|                 AuthorWithDefaultHeight._meta.get_field("height"), |  | ||||||
|                 new_field, |  | ||||||
|             ) |  | ||||||
|         # Ensure the field is right afterwards |         # Ensure the field is right afterwards | ||||||
|         columns = self.column_classes(AuthorWithDefaultHeight) |         columns = self.column_classes(AuthorWithDefaultHeight) | ||||||
|         self.assertFalse(columns['height'][1][6]) |         self.assertFalse(columns['height'][1][6]) | ||||||
| @@ -508,15 +490,11 @@ class SchemaTests(TransactionTestCase): | |||||||
|         else: |         else: | ||||||
|             self.fail("No FK constraint for author_id found") |             self.fail("No FK constraint for author_id found") | ||||||
|         # Alter the FK |         # Alter the FK | ||||||
|  |         old_field = Book._meta.get_field("author") | ||||||
|         new_field = ForeignKey(Author, editable=False) |         new_field = ForeignKey(Author, editable=False) | ||||||
|         new_field.set_attributes_from_name("author") |         new_field.set_attributes_from_name("author") | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.alter_field( |             editor.alter_field(Book, old_field, new_field, strict=True) | ||||||
|                 Book, |  | ||||||
|                 Book._meta.get_field("author"), |  | ||||||
|                 new_field, |  | ||||||
|                 strict=True, |  | ||||||
|             ) |  | ||||||
|         # Ensure the field is right afterwards |         # Ensure the field is right afterwards | ||||||
|         columns = self.column_classes(Book) |         columns = self.column_classes(Book) | ||||||
|         self.assertEqual(columns['author_id'][0], "IntegerField") |         self.assertEqual(columns['author_id'][0], "IntegerField") | ||||||
| @@ -556,15 +534,11 @@ class SchemaTests(TransactionTestCase): | |||||||
|                     author_is_fk = True |                     author_is_fk = True | ||||||
|         self.assertTrue(author_is_fk, "No FK constraint for author_id found") |         self.assertTrue(author_is_fk, "No FK constraint for author_id found") | ||||||
|         # Alter the OneToOneField to ForeignKey |         # Alter the OneToOneField to ForeignKey | ||||||
|  |         old_field = BookWithO2O._meta.get_field("author") | ||||||
|         new_field = ForeignKey(Author) |         new_field = ForeignKey(Author) | ||||||
|         new_field.set_attributes_from_name("author") |         new_field.set_attributes_from_name("author") | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.alter_field( |             editor.alter_field(BookWithO2O, old_field, new_field, strict=True) | ||||||
|                 BookWithO2O, |  | ||||||
|                 BookWithO2O._meta.get_field("author"), |  | ||||||
|                 new_field, |  | ||||||
|                 strict=True, |  | ||||||
|             ) |  | ||||||
|         # Ensure the field is right afterwards |         # Ensure the field is right afterwards | ||||||
|         columns = self.column_classes(Book) |         columns = self.column_classes(Book) | ||||||
|         self.assertEqual(columns['author_id'][0], "IntegerField") |         self.assertEqual(columns['author_id'][0], "IntegerField") | ||||||
| @@ -606,15 +580,11 @@ class SchemaTests(TransactionTestCase): | |||||||
|                     author_is_fk = True |                     author_is_fk = True | ||||||
|         self.assertTrue(author_is_fk, "No FK constraint for author_id found") |         self.assertTrue(author_is_fk, "No FK constraint for author_id found") | ||||||
|         # Alter the ForeignKey to OneToOneField |         # Alter the ForeignKey to OneToOneField | ||||||
|  |         old_field = Book._meta.get_field("author") | ||||||
|         new_field = OneToOneField(Author) |         new_field = OneToOneField(Author) | ||||||
|         new_field.set_attributes_from_name("author") |         new_field.set_attributes_from_name("author") | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.alter_field( |             editor.alter_field(Book, old_field, new_field, strict=True) | ||||||
|                 Book, |  | ||||||
|                 Book._meta.get_field("author"), |  | ||||||
|                 new_field, |  | ||||||
|                 strict=True, |  | ||||||
|             ) |  | ||||||
|         # Ensure the field is right afterwards |         # Ensure the field is right afterwards | ||||||
|         columns = self.column_classes(BookWithO2O) |         columns = self.column_classes(BookWithO2O) | ||||||
|         self.assertEqual(columns['author_id'][0], "IntegerField") |         self.assertEqual(columns['author_id'][0], "IntegerField") | ||||||
| @@ -639,17 +609,12 @@ class SchemaTests(TransactionTestCase): | |||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.create_model(Author) |             editor.create_model(Author) | ||||||
|  |  | ||||||
|  |         old_field = Author._meta.get_field("id") | ||||||
|         new_field = IntegerField(primary_key=True) |         new_field = IntegerField(primary_key=True) | ||||||
|         new_field.set_attributes_from_name("id") |         new_field.set_attributes_from_name("id") | ||||||
|         new_field.model = Author |         new_field.model = Author | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.alter_field( |             editor.alter_field(Author, old_field, new_field, strict=True) | ||||||
|                 Author, |  | ||||||
|                 Author._meta.get_field("id"), |  | ||||||
|                 new_field, |  | ||||||
|                 strict=True, |  | ||||||
|             ) |  | ||||||
|  |  | ||||||
|         # This will fail if DROP DEFAULT is inadvertently executed on this |         # This will fail if DROP DEFAULT is inadvertently executed on this | ||||||
|         # field which drops the id sequence, at least on PostgreSQL. |         # field which drops the id sequence, at least on PostgreSQL. | ||||||
|         Author.objects.create(name='Foo') |         Author.objects.create(name='Foo') | ||||||
| @@ -666,127 +631,202 @@ class SchemaTests(TransactionTestCase): | |||||||
|         self.assertEqual(columns['name'][0], "CharField") |         self.assertEqual(columns['name'][0], "CharField") | ||||||
|         self.assertNotIn("display_name", columns) |         self.assertNotIn("display_name", columns) | ||||||
|         # Alter the name field's name |         # Alter the name field's name | ||||||
|  |         old_field = Author._meta.get_field("name") | ||||||
|         new_field = CharField(max_length=254) |         new_field = CharField(max_length=254) | ||||||
|         new_field.set_attributes_from_name("display_name") |         new_field.set_attributes_from_name("display_name") | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.alter_field( |             editor.alter_field(Author, old_field, new_field, strict=True) | ||||||
|                 Author, |  | ||||||
|                 Author._meta.get_field("name"), |  | ||||||
|                 new_field, |  | ||||||
|                 strict=True, |  | ||||||
|             ) |  | ||||||
|         # Ensure the field is right afterwards |         # Ensure the field is right afterwards | ||||||
|         columns = self.column_classes(Author) |         columns = self.column_classes(Author) | ||||||
|         self.assertEqual(columns['display_name'][0], "CharField") |         self.assertEqual(columns['display_name'][0], "CharField") | ||||||
|         self.assertNotIn("name", columns) |         self.assertNotIn("name", columns) | ||||||
|  |  | ||||||
|     def test_m2m_create(self): |     def _test_m2m_create(self, M2MFieldClass): | ||||||
|         """ |         """ | ||||||
|         Tests M2M fields on models during creation |         Tests M2M fields on models during creation | ||||||
|         """ |         """ | ||||||
|  |         class LocalBookWithM2M(Model): | ||||||
|  |             author = ForeignKey(Author) | ||||||
|  |             title = CharField(max_length=100, db_index=True) | ||||||
|  |             pub_date = DateTimeField() | ||||||
|  |             tags = M2MFieldClass("TagM2MTest", related_name="books") | ||||||
|  |  | ||||||
|  |             class Meta: | ||||||
|  |                 apps = new_apps | ||||||
|  |  | ||||||
|  |         self.local_models = [LocalBookWithM2M] | ||||||
|  |  | ||||||
|         # Create the tables |         # Create the tables | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.create_model(Author) |             editor.create_model(Author) | ||||||
|             editor.create_model(TagM2MTest) |             editor.create_model(TagM2MTest) | ||||||
|             editor.create_model(BookWithM2M) |             editor.create_model(LocalBookWithM2M) | ||||||
|         # Ensure there is now an m2m table there |         # Ensure there is now an m2m table there | ||||||
|         columns = self.column_classes(BookWithM2M._meta.get_field("tags").rel.through) |         columns = self.column_classes(LocalBookWithM2M._meta.get_field("tags").rel.through) | ||||||
|         self.assertEqual(columns['tagm2mtest_id'][0], "IntegerField") |         self.assertEqual(columns['tagm2mtest_id'][0], "IntegerField") | ||||||
|  |  | ||||||
|     def test_m2m_create_through(self): |     def test_m2m_create(self): | ||||||
|  |         self._test_m2m_create(ManyToManyField) | ||||||
|  |  | ||||||
|  |     def test_m2m_create_custom(self): | ||||||
|  |         self._test_m2m_create(CustomManyToManyField) | ||||||
|  |  | ||||||
|  |     def test_m2m_create_inherited(self): | ||||||
|  |         self._test_m2m_create(InheritedManyToManyField) | ||||||
|  |  | ||||||
|  |     def _test_m2m_create_through(self, M2MFieldClass): | ||||||
|         """ |         """ | ||||||
|         Tests M2M fields on models during creation with through models |         Tests M2M fields on models during creation with through models | ||||||
|         """ |         """ | ||||||
|  |         class LocalTagThrough(Model): | ||||||
|  |             book = ForeignKey("schema.LocalBookWithM2MThrough") | ||||||
|  |             tag = ForeignKey("schema.TagM2MTest") | ||||||
|  |  | ||||||
|  |             class Meta: | ||||||
|  |                 apps = new_apps | ||||||
|  |  | ||||||
|  |         class LocalBookWithM2MThrough(Model): | ||||||
|  |             tags = M2MFieldClass("TagM2MTest", related_name="books", through=LocalTagThrough) | ||||||
|  |  | ||||||
|  |             class Meta: | ||||||
|  |                 apps = new_apps | ||||||
|  |  | ||||||
|  |         self.local_models = [LocalTagThrough, LocalBookWithM2MThrough] | ||||||
|  |  | ||||||
|         # Create the tables |         # Create the tables | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.create_model(TagThrough) |             editor.create_model(LocalTagThrough) | ||||||
|             editor.create_model(TagM2MTest) |             editor.create_model(TagM2MTest) | ||||||
|             editor.create_model(BookWithM2MThrough) |             editor.create_model(LocalBookWithM2MThrough) | ||||||
|         # Ensure there is now an m2m table there |         # Ensure there is now an m2m table there | ||||||
|         columns = self.column_classes(TagThrough) |         columns = self.column_classes(LocalTagThrough) | ||||||
|         self.assertEqual(columns['book_id'][0], "IntegerField") |         self.assertEqual(columns['book_id'][0], "IntegerField") | ||||||
|         self.assertEqual(columns['tag_id'][0], "IntegerField") |         self.assertEqual(columns['tag_id'][0], "IntegerField") | ||||||
|  |  | ||||||
|     def test_m2m(self): |     def test_m2m_create_through(self): | ||||||
|  |         self._test_m2m_create_through(ManyToManyField) | ||||||
|  |  | ||||||
|  |     def test_m2m_create_through_custom(self): | ||||||
|  |         self._test_m2m_create_through(CustomManyToManyField) | ||||||
|  |  | ||||||
|  |     def test_m2m_create_through_inherited(self): | ||||||
|  |         self._test_m2m_create_through(InheritedManyToManyField) | ||||||
|  |  | ||||||
|  |     def _test_m2m(self, M2MFieldClass): | ||||||
|         """ |         """ | ||||||
|         Tests adding/removing M2M fields on models |         Tests adding/removing M2M fields on models | ||||||
|         """ |         """ | ||||||
|  |         class LocalAuthorWithM2M(Model): | ||||||
|  |             name = CharField(max_length=255) | ||||||
|  |  | ||||||
|  |             class Meta: | ||||||
|  |                 apps = new_apps | ||||||
|  |  | ||||||
|  |         self.local_models = [LocalAuthorWithM2M] | ||||||
|  |  | ||||||
|         # Create the tables |         # Create the tables | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.create_model(AuthorWithM2M) |             editor.create_model(LocalAuthorWithM2M) | ||||||
|             editor.create_model(TagM2MTest) |             editor.create_model(TagM2MTest) | ||||||
|         # Create an M2M field |         # Create an M2M field | ||||||
|         new_field = ManyToManyField("schema.TagM2MTest", related_name="authors") |         new_field = M2MFieldClass("schema.TagM2MTest", related_name="authors") | ||||||
|         new_field.contribute_to_class(AuthorWithM2M, "tags") |         new_field.contribute_to_class(LocalAuthorWithM2M, "tags") | ||||||
|         try: |  | ||||||
|         # Ensure there's no m2m table there |         # Ensure there's no m2m table there | ||||||
|         self.assertRaises(DatabaseError, self.column_classes, new_field.rel.through) |         self.assertRaises(DatabaseError, self.column_classes, new_field.rel.through) | ||||||
|         # Add the field |         # Add the field | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|                 editor.add_field( |             editor.add_field(LocalAuthorWithM2M, new_field) | ||||||
|                     Author, |  | ||||||
|                     new_field, |  | ||||||
|                 ) |  | ||||||
|         # Ensure there is now an m2m table there |         # Ensure there is now an m2m table there | ||||||
|         columns = self.column_classes(new_field.rel.through) |         columns = self.column_classes(new_field.rel.through) | ||||||
|         self.assertEqual(columns['tagm2mtest_id'][0], "IntegerField") |         self.assertEqual(columns['tagm2mtest_id'][0], "IntegerField") | ||||||
|  |  | ||||||
|         # "Alter" the field. This should not rename the DB table to itself. |         # "Alter" the field. This should not rename the DB table to itself. | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|                 editor.alter_field( |             editor.alter_field(LocalAuthorWithM2M, new_field, new_field) | ||||||
|                     Author, |  | ||||||
|                     new_field, |  | ||||||
|                     new_field, |  | ||||||
|                 ) |  | ||||||
|  |  | ||||||
|         # Remove the M2M table again |         # Remove the M2M table again | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|                 editor.remove_field( |             editor.remove_field(LocalAuthorWithM2M, new_field) | ||||||
|                     Author, |  | ||||||
|                     new_field, |  | ||||||
|                 ) |  | ||||||
|         # Ensure there's no m2m table there |         # Ensure there's no m2m table there | ||||||
|         self.assertRaises(DatabaseError, self.column_classes, new_field.rel.through) |         self.assertRaises(DatabaseError, self.column_classes, new_field.rel.through) | ||||||
|         finally: |  | ||||||
|             # Cleanup model states |  | ||||||
|             AuthorWithM2M._meta.local_many_to_many.remove(new_field) |  | ||||||
|  |  | ||||||
|     def test_m2m_through_alter(self): |     def test_m2m(self): | ||||||
|  |         self._test_m2m(ManyToManyField) | ||||||
|  |  | ||||||
|  |     def test_m2m_custom(self): | ||||||
|  |         self._test_m2m(CustomManyToManyField) | ||||||
|  |  | ||||||
|  |     def test_m2m_inherited(self): | ||||||
|  |         self._test_m2m(InheritedManyToManyField) | ||||||
|  |  | ||||||
|  |     def _test_m2m_through_alter(self, M2MFieldClass): | ||||||
|         """ |         """ | ||||||
|         Tests altering M2Ms with explicit through models (should no-op) |         Tests altering M2Ms with explicit through models (should no-op) | ||||||
|         """ |         """ | ||||||
|  |         class LocalAuthorTag(Model): | ||||||
|  |             author = ForeignKey("schema.LocalAuthorWithM2MThrough") | ||||||
|  |             tag = ForeignKey("schema.TagM2MTest") | ||||||
|  |  | ||||||
|  |             class Meta: | ||||||
|  |                 apps = new_apps | ||||||
|  |  | ||||||
|  |         class LocalAuthorWithM2MThrough(Model): | ||||||
|  |             name = CharField(max_length=255) | ||||||
|  |             tags = M2MFieldClass("schema.TagM2MTest", related_name="authors", through=LocalAuthorTag) | ||||||
|  |  | ||||||
|  |             class Meta: | ||||||
|  |                 apps = new_apps | ||||||
|  |  | ||||||
|  |         self.local_models = [LocalAuthorTag, LocalAuthorWithM2MThrough] | ||||||
|  |  | ||||||
|         # Create the tables |         # Create the tables | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.create_model(AuthorTag) |             editor.create_model(LocalAuthorTag) | ||||||
|             editor.create_model(AuthorWithM2MThrough) |             editor.create_model(LocalAuthorWithM2MThrough) | ||||||
|             editor.create_model(TagM2MTest) |             editor.create_model(TagM2MTest) | ||||||
|         # Ensure the m2m table is there |         # Ensure the m2m table is there | ||||||
|         self.assertEqual(len(self.column_classes(AuthorTag)), 3) |         self.assertEqual(len(self.column_classes(LocalAuthorTag)), 3) | ||||||
|         # "Alter" the field's blankness. This should not actually do anything. |         # "Alter" the field's blankness. This should not actually do anything. | ||||||
|  |         old_field = LocalAuthorWithM2MThrough._meta.get_field("tags") | ||||||
|  |         new_field = M2MFieldClass("schema.TagM2MTest", related_name="authors", through=LocalAuthorTag) | ||||||
|  |         new_field.contribute_to_class(LocalAuthorWithM2MThrough, "tags") | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             old_field = AuthorWithM2MThrough._meta.get_field("tags") |             editor.alter_field(LocalAuthorWithM2MThrough, old_field, new_field) | ||||||
|             new_field = ManyToManyField("schema.TagM2MTest", related_name="authors", through="AuthorTag") |  | ||||||
|             new_field.contribute_to_class(AuthorWithM2MThrough, "tags") |  | ||||||
|             editor.alter_field( |  | ||||||
|                 Author, |  | ||||||
|                 old_field, |  | ||||||
|                 new_field, |  | ||||||
|             ) |  | ||||||
|         # Ensure the m2m table is still there |         # Ensure the m2m table is still there | ||||||
|         self.assertEqual(len(self.column_classes(AuthorTag)), 3) |         self.assertEqual(len(self.column_classes(LocalAuthorTag)), 3) | ||||||
|  |  | ||||||
|     def test_m2m_repoint(self): |     def test_m2m_through_alter(self): | ||||||
|  |         self._test_m2m_through_alter(ManyToManyField) | ||||||
|  |  | ||||||
|  |     def test_m2m_through_alter_custom(self): | ||||||
|  |         self._test_m2m_through_alter(CustomManyToManyField) | ||||||
|  |  | ||||||
|  |     def test_m2m_through_alter_inherited(self): | ||||||
|  |         self._test_m2m_through_alter(InheritedManyToManyField) | ||||||
|  |  | ||||||
|  |     def _test_m2m_repoint(self, M2MFieldClass): | ||||||
|         """ |         """ | ||||||
|         Tests repointing M2M fields |         Tests repointing M2M fields | ||||||
|         """ |         """ | ||||||
|  |         class LocalBookWithM2M(Model): | ||||||
|  |             author = ForeignKey(Author) | ||||||
|  |             title = CharField(max_length=100, db_index=True) | ||||||
|  |             pub_date = DateTimeField() | ||||||
|  |             tags = M2MFieldClass("TagM2MTest", related_name="books") | ||||||
|  |  | ||||||
|  |             class Meta: | ||||||
|  |                 apps = new_apps | ||||||
|  |  | ||||||
|  |         self.local_models = [LocalBookWithM2M] | ||||||
|  |  | ||||||
|         # Create the tables |         # Create the tables | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.create_model(Author) |             editor.create_model(Author) | ||||||
|             editor.create_model(BookWithM2M) |             editor.create_model(LocalBookWithM2M) | ||||||
|             editor.create_model(TagM2MTest) |             editor.create_model(TagM2MTest) | ||||||
|             editor.create_model(UniqueTest) |             editor.create_model(UniqueTest) | ||||||
|         # Ensure the M2M exists and points to TagM2MTest |         # Ensure the M2M exists and points to TagM2MTest | ||||||
|         constraints = self.get_constraints(BookWithM2M._meta.get_field("tags").rel.through._meta.db_table) |         constraints = self.get_constraints(LocalBookWithM2M._meta.get_field("tags").rel.through._meta.db_table) | ||||||
|         if connection.features.supports_foreign_keys: |         if connection.features.supports_foreign_keys: | ||||||
|             for name, details in constraints.items(): |             for name, details in constraints.items(): | ||||||
|                 if details['columns'] == ["tagm2mtest_id"] and details['foreign_key']: |                 if details['columns'] == ["tagm2mtest_id"] and details['foreign_key']: | ||||||
| @@ -795,17 +835,13 @@ class SchemaTests(TransactionTestCase): | |||||||
|             else: |             else: | ||||||
|                 self.fail("No FK constraint for tagm2mtest_id found") |                 self.fail("No FK constraint for tagm2mtest_id found") | ||||||
|         # Repoint the M2M |         # Repoint the M2M | ||||||
|         new_field = ManyToManyField(UniqueTest) |         old_field = LocalBookWithM2M._meta.get_field("tags") | ||||||
|         new_field.contribute_to_class(BookWithM2M, "uniques") |         new_field = M2MFieldClass(UniqueTest) | ||||||
|         try: |         new_field.contribute_to_class(LocalBookWithM2M, "uniques") | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|                 editor.alter_field( |             editor.alter_field(LocalBookWithM2M, old_field, new_field) | ||||||
|                     Author, |  | ||||||
|                     BookWithM2M._meta.get_field("tags"), |  | ||||||
|                     new_field, |  | ||||||
|                 ) |  | ||||||
|         # Ensure old M2M is gone |         # Ensure old M2M is gone | ||||||
|             self.assertRaises(DatabaseError, self.column_classes, BookWithM2M._meta.get_field("tags").rel.through) |         self.assertRaises(DatabaseError, self.column_classes, LocalBookWithM2M._meta.get_field("tags").rel.through) | ||||||
|         # Ensure the new M2M exists and points to UniqueTest |         # Ensure the new M2M exists and points to UniqueTest | ||||||
|         constraints = self.get_constraints(new_field.rel.through._meta.db_table) |         constraints = self.get_constraints(new_field.rel.through._meta.db_table) | ||||||
|         if connection.features.supports_foreign_keys: |         if connection.features.supports_foreign_keys: | ||||||
| @@ -815,13 +851,15 @@ class SchemaTests(TransactionTestCase): | |||||||
|                     break |                     break | ||||||
|             else: |             else: | ||||||
|                 self.fail("No FK constraint for uniquetest_id found") |                 self.fail("No FK constraint for uniquetest_id found") | ||||||
|         finally: |  | ||||||
|             # Cleanup through table separately |     def test_m2m_repoint(self): | ||||||
|             with connection.schema_editor() as editor: |         self._test_m2m_repoint(ManyToManyField) | ||||||
|                 editor.remove_field(BookWithM2M, BookWithM2M._meta.get_field("uniques")) |  | ||||||
|             # Cleanup model states |     def test_m2m_repoint_custom(self): | ||||||
|             BookWithM2M._meta.local_many_to_many.remove(new_field) |         self._test_m2m_repoint(CustomManyToManyField) | ||||||
|             BookWithM2M._meta._expire_cache() |  | ||||||
|  |     def test_m2m_repoint_inherited(self): | ||||||
|  |         self._test_m2m_repoint(InheritedManyToManyField) | ||||||
|  |  | ||||||
|     @unittest.skipUnless(connection.features.supports_column_check_constraints, "No check constraints") |     @unittest.skipUnless(connection.features.supports_column_check_constraints, "No check constraints") | ||||||
|     def test_check_constraints(self): |     def test_check_constraints(self): | ||||||
| @@ -839,27 +877,19 @@ class SchemaTests(TransactionTestCase): | |||||||
|         else: |         else: | ||||||
|             self.fail("No check constraint for height found") |             self.fail("No check constraint for height found") | ||||||
|         # Alter the column to remove it |         # Alter the column to remove it | ||||||
|  |         old_field = Author._meta.get_field("height") | ||||||
|         new_field = IntegerField(null=True, blank=True) |         new_field = IntegerField(null=True, blank=True) | ||||||
|         new_field.set_attributes_from_name("height") |         new_field.set_attributes_from_name("height") | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.alter_field( |             editor.alter_field(Author, old_field, new_field, strict=True) | ||||||
|                 Author, |  | ||||||
|                 Author._meta.get_field("height"), |  | ||||||
|                 new_field, |  | ||||||
|                 strict=True, |  | ||||||
|             ) |  | ||||||
|         constraints = self.get_constraints(Author._meta.db_table) |         constraints = self.get_constraints(Author._meta.db_table) | ||||||
|         for name, details in constraints.items(): |         for name, details in constraints.items(): | ||||||
|             if details['columns'] == ["height"] and details['check']: |             if details['columns'] == ["height"] and details['check']: | ||||||
|                 self.fail("Check constraint for height found") |                 self.fail("Check constraint for height found") | ||||||
|         # Alter the column to re-add it |         # Alter the column to re-add it | ||||||
|  |         new_field2 = Author._meta.get_field("height") | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.alter_field( |             editor.alter_field(Author, new_field, new_field2, strict=True) | ||||||
|                 Author, |  | ||||||
|                 new_field, |  | ||||||
|                 Author._meta.get_field("height"), |  | ||||||
|                 strict=True, |  | ||||||
|             ) |  | ||||||
|         constraints = self.get_constraints(Author._meta.db_table) |         constraints = self.get_constraints(Author._meta.db_table) | ||||||
|         for name, details in constraints.items(): |         for name, details in constraints.items(): | ||||||
|             if details['columns'] == ["height"] and details['check']: |             if details['columns'] == ["height"] and details['check']: | ||||||
| @@ -879,43 +909,29 @@ class SchemaTests(TransactionTestCase): | |||||||
|         self.assertRaises(IntegrityError, Tag.objects.create, title="bar", slug="foo") |         self.assertRaises(IntegrityError, Tag.objects.create, title="bar", slug="foo") | ||||||
|         Tag.objects.all().delete() |         Tag.objects.all().delete() | ||||||
|         # Alter the slug field to be non-unique |         # Alter the slug field to be non-unique | ||||||
|  |         old_field = Tag._meta.get_field("slug") | ||||||
|         new_field = SlugField(unique=False) |         new_field = SlugField(unique=False) | ||||||
|         new_field.set_attributes_from_name("slug") |         new_field.set_attributes_from_name("slug") | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.alter_field( |             editor.alter_field(Tag, old_field, new_field, strict=True) | ||||||
|                 Tag, |  | ||||||
|                 Tag._meta.get_field("slug"), |  | ||||||
|                 new_field, |  | ||||||
|                 strict=True, |  | ||||||
|             ) |  | ||||||
|         # Ensure the field is no longer unique |         # Ensure the field is no longer unique | ||||||
|         Tag.objects.create(title="foo", slug="foo") |         Tag.objects.create(title="foo", slug="foo") | ||||||
|         Tag.objects.create(title="bar", slug="foo") |         Tag.objects.create(title="bar", slug="foo") | ||||||
|         Tag.objects.all().delete() |         Tag.objects.all().delete() | ||||||
|         # Alter the slug field to be unique |         # Alter the slug field to be unique | ||||||
|         new_new_field = SlugField(unique=True) |         new_field2 = SlugField(unique=True) | ||||||
|         new_new_field.set_attributes_from_name("slug") |         new_field2.set_attributes_from_name("slug") | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.alter_field( |             editor.alter_field(Tag, new_field, new_field2, strict=True) | ||||||
|                 Tag, |  | ||||||
|                 new_field, |  | ||||||
|                 new_new_field, |  | ||||||
|                 strict=True, |  | ||||||
|             ) |  | ||||||
|         # Ensure the field is unique again |         # Ensure the field is unique again | ||||||
|         Tag.objects.create(title="foo", slug="foo") |         Tag.objects.create(title="foo", slug="foo") | ||||||
|         self.assertRaises(IntegrityError, Tag.objects.create, title="bar", slug="foo") |         self.assertRaises(IntegrityError, Tag.objects.create, title="bar", slug="foo") | ||||||
|         Tag.objects.all().delete() |         Tag.objects.all().delete() | ||||||
|         # Rename the field |         # Rename the field | ||||||
|         new_field = SlugField(unique=False) |         new_field3 = SlugField(unique=True) | ||||||
|         new_field.set_attributes_from_name("slug2") |         new_field3.set_attributes_from_name("slug2") | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.alter_field( |             editor.alter_field(Tag, new_field2, new_field3, strict=True) | ||||||
|                 Tag, |  | ||||||
|                 Tag._meta.get_field("slug"), |  | ||||||
|                 TagUniqueRename._meta.get_field("slug2"), |  | ||||||
|                 strict=True, |  | ||||||
|             ) |  | ||||||
|         # Ensure the field is still unique |         # Ensure the field is still unique | ||||||
|         TagUniqueRename.objects.create(title="foo", slug2="foo") |         TagUniqueRename.objects.create(title="foo", slug2="foo") | ||||||
|         self.assertRaises(IntegrityError, TagUniqueRename.objects.create, title="bar", slug2="foo") |         self.assertRaises(IntegrityError, TagUniqueRename.objects.create, title="bar", slug2="foo") | ||||||
| @@ -936,24 +952,16 @@ class SchemaTests(TransactionTestCase): | |||||||
|         UniqueTest.objects.all().delete() |         UniqueTest.objects.all().delete() | ||||||
|         # Alter the model to its non-unique-together companion |         # Alter the model to its non-unique-together companion | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.alter_unique_together( |             editor.alter_unique_together(UniqueTest, UniqueTest._meta.unique_together, []) | ||||||
|                 UniqueTest, |  | ||||||
|                 UniqueTest._meta.unique_together, |  | ||||||
|                 [], |  | ||||||
|             ) |  | ||||||
|         # Ensure the fields are no longer unique |         # Ensure the fields are no longer unique | ||||||
|         UniqueTest.objects.create(year=2012, slug="foo") |         UniqueTest.objects.create(year=2012, slug="foo") | ||||||
|         UniqueTest.objects.create(year=2012, slug="foo") |         UniqueTest.objects.create(year=2012, slug="foo") | ||||||
|         UniqueTest.objects.all().delete() |         UniqueTest.objects.all().delete() | ||||||
|         # Alter it back |         # Alter it back | ||||||
|         new_new_field = SlugField(unique=True) |         new_field2 = SlugField(unique=True) | ||||||
|         new_new_field.set_attributes_from_name("slug") |         new_field2.set_attributes_from_name("slug") | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.alter_unique_together( |             editor.alter_unique_together(UniqueTest, [], UniqueTest._meta.unique_together) | ||||||
|                 UniqueTest, |  | ||||||
|                 [], |  | ||||||
|                 UniqueTest._meta.unique_together, |  | ||||||
|             ) |  | ||||||
|         # Ensure the fields are unique again |         # Ensure the fields are unique again | ||||||
|         UniqueTest.objects.create(year=2012, slug="foo") |         UniqueTest.objects.create(year=2012, slug="foo") | ||||||
|         self.assertRaises(IntegrityError, UniqueTest.objects.create, year=2012, slug="foo") |         self.assertRaises(IntegrityError, UniqueTest.objects.create, year=2012, slug="foo") | ||||||
| @@ -977,11 +985,7 @@ class SchemaTests(TransactionTestCase): | |||||||
|         ) |         ) | ||||||
|         # Alter the model to add an index |         # Alter the model to add an index | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.alter_index_together( |             editor.alter_index_together(Tag, [], [("slug", "title")]) | ||||||
|                 Tag, |  | ||||||
|                 [], |  | ||||||
|                 [("slug", "title")], |  | ||||||
|             ) |  | ||||||
|         # Ensure there is now an index |         # Ensure there is now an index | ||||||
|         self.assertEqual( |         self.assertEqual( | ||||||
|             True, |             True, | ||||||
| @@ -992,14 +996,10 @@ class SchemaTests(TransactionTestCase): | |||||||
|             ), |             ), | ||||||
|         ) |         ) | ||||||
|         # Alter it back |         # Alter it back | ||||||
|         new_new_field = SlugField(unique=True) |         new_field2 = SlugField(unique=True) | ||||||
|         new_new_field.set_attributes_from_name("slug") |         new_field2.set_attributes_from_name("slug") | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.alter_index_together( |             editor.alter_index_together(Tag, [("slug", "title")], []) | ||||||
|                 Tag, |  | ||||||
|                 [("slug", "title")], |  | ||||||
|                 [], |  | ||||||
|             ) |  | ||||||
|         # Ensure there's no index |         # Ensure there's no index | ||||||
|         self.assertEqual( |         self.assertEqual( | ||||||
|             False, |             False, | ||||||
| @@ -1039,22 +1039,14 @@ class SchemaTests(TransactionTestCase): | |||||||
|         self.assertEqual(columns['name'][0], "CharField") |         self.assertEqual(columns['name'][0], "CharField") | ||||||
|         # Alter the table |         # Alter the table | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.alter_db_table( |             editor.alter_db_table(Author, "schema_author", "schema_otherauthor") | ||||||
|                 Author, |  | ||||||
|                 "schema_author", |  | ||||||
|                 "schema_otherauthor", |  | ||||||
|             ) |  | ||||||
|         # Ensure the table is there afterwards |         # Ensure the table is there afterwards | ||||||
|         Author._meta.db_table = "schema_otherauthor" |         Author._meta.db_table = "schema_otherauthor" | ||||||
|         columns = self.column_classes(Author) |         columns = self.column_classes(Author) | ||||||
|         self.assertEqual(columns['name'][0], "CharField") |         self.assertEqual(columns['name'][0], "CharField") | ||||||
|         # Alter the table again |         # Alter the table again | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.alter_db_table( |             editor.alter_db_table(Author, "schema_otherauthor", "schema_author") | ||||||
|                 Author, |  | ||||||
|                 "schema_otherauthor", |  | ||||||
|                 "schema_author", |  | ||||||
|             ) |  | ||||||
|         # Ensure the table is still there |         # Ensure the table is still there | ||||||
|         Author._meta.db_table = "schema_author" |         Author._meta.db_table = "schema_author" | ||||||
|         columns = self.column_classes(Author) |         columns = self.column_classes(Author) | ||||||
| @@ -1074,53 +1066,38 @@ class SchemaTests(TransactionTestCase): | |||||||
|             self.get_indexes(Book._meta.db_table), |             self.get_indexes(Book._meta.db_table), | ||||||
|         ) |         ) | ||||||
|         # Alter to remove the index |         # Alter to remove the index | ||||||
|  |         old_field = Book._meta.get_field("title") | ||||||
|         new_field = CharField(max_length=100, db_index=False) |         new_field = CharField(max_length=100, db_index=False) | ||||||
|         new_field.set_attributes_from_name("title") |         new_field.set_attributes_from_name("title") | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.alter_field( |             editor.alter_field(Book, old_field, new_field, strict=True) | ||||||
|                 Book, |  | ||||||
|                 Book._meta.get_field("title"), |  | ||||||
|                 new_field, |  | ||||||
|                 strict=True, |  | ||||||
|             ) |  | ||||||
|         # Ensure the table is there and has no index |         # Ensure the table is there and has no index | ||||||
|         self.assertNotIn( |         self.assertNotIn( | ||||||
|             "title", |             "title", | ||||||
|             self.get_indexes(Book._meta.db_table), |             self.get_indexes(Book._meta.db_table), | ||||||
|         ) |         ) | ||||||
|         # Alter to re-add the index |         # Alter to re-add the index | ||||||
|  |         new_field2 = Book._meta.get_field("title") | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.alter_field( |             editor.alter_field(Book, new_field, new_field2, strict=True) | ||||||
|                 Book, |  | ||||||
|                 new_field, |  | ||||||
|                 Book._meta.get_field("title"), |  | ||||||
|                 strict=True, |  | ||||||
|             ) |  | ||||||
|         # Ensure the table is there and has the index again |         # Ensure the table is there and has the index again | ||||||
|         self.assertIn( |         self.assertIn( | ||||||
|             "title", |             "title", | ||||||
|             self.get_indexes(Book._meta.db_table), |             self.get_indexes(Book._meta.db_table), | ||||||
|         ) |         ) | ||||||
|         # Add a unique column, verify that creates an implicit index |         # Add a unique column, verify that creates an implicit index | ||||||
|  |         new_field3 = BookWithSlug._meta.get_field("slug") | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.add_field( |             editor.add_field(Book, new_field3) | ||||||
|                 Book, |  | ||||||
|                 BookWithSlug._meta.get_field("slug"), |  | ||||||
|             ) |  | ||||||
|         self.assertIn( |         self.assertIn( | ||||||
|             "slug", |             "slug", | ||||||
|             self.get_indexes(Book._meta.db_table), |             self.get_indexes(Book._meta.db_table), | ||||||
|         ) |         ) | ||||||
|         # Remove the unique, check the index goes with it |         # Remove the unique, check the index goes with it | ||||||
|         new_field2 = CharField(max_length=20, unique=False) |         new_field4 = CharField(max_length=20, unique=False) | ||||||
|         new_field2.set_attributes_from_name("slug") |         new_field4.set_attributes_from_name("slug") | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.alter_field( |             editor.alter_field(BookWithSlug, new_field3, new_field4, strict=True) | ||||||
|                 BookWithSlug, |  | ||||||
|                 BookWithSlug._meta.get_field("slug"), |  | ||||||
|                 new_field2, |  | ||||||
|                 strict=True, |  | ||||||
|             ) |  | ||||||
|         self.assertNotIn( |         self.assertNotIn( | ||||||
|             "slug", |             "slug", | ||||||
|             self.get_indexes(Book._meta.db_table), |             self.get_indexes(Book._meta.db_table), | ||||||
| @@ -1138,16 +1115,14 @@ class SchemaTests(TransactionTestCase): | |||||||
|             self.get_indexes(Tag._meta.db_table)['id']['primary_key'], |             self.get_indexes(Tag._meta.db_table)['id']['primary_key'], | ||||||
|         ) |         ) | ||||||
|         # Alter to change the PK |         # Alter to change the PK | ||||||
|  |         id_field = Tag._meta.get_field("id") | ||||||
|  |         old_field = Tag._meta.get_field("slug") | ||||||
|         new_field = SlugField(primary_key=True) |         new_field = SlugField(primary_key=True) | ||||||
|         new_field.set_attributes_from_name("slug") |         new_field.set_attributes_from_name("slug") | ||||||
|         new_field.model = Tag |         new_field.model = Tag | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.remove_field(Tag, Tag._meta.get_field("id")) |             editor.remove_field(Tag, id_field) | ||||||
|             editor.alter_field( |             editor.alter_field(Tag, old_field, new_field) | ||||||
|                 Tag, |  | ||||||
|                 Tag._meta.get_field("slug"), |  | ||||||
|                 new_field, |  | ||||||
|             ) |  | ||||||
|         # Ensure the PK changed |         # Ensure the PK changed | ||||||
|         self.assertNotIn( |         self.assertNotIn( | ||||||
|             'id', |             'id', | ||||||
| @@ -1203,10 +1178,7 @@ class SchemaTests(TransactionTestCase): | |||||||
|         new_field = ForeignKey(AuthorWithEvenLongerName, related_name="something") |         new_field = ForeignKey(AuthorWithEvenLongerName, related_name="something") | ||||||
|         new_field.set_attributes_from_name("author_other_really_long_named_i_mean_so_long_fk") |         new_field.set_attributes_from_name("author_other_really_long_named_i_mean_so_long_fk") | ||||||
|         with connection.schema_editor() as editor: |         with connection.schema_editor() as editor: | ||||||
|             editor.add_field( |             editor.add_field(BookWithLongName, new_field) | ||||||
|                 BookWithLongName, |  | ||||||
|                 new_field, |  | ||||||
|             ) |  | ||||||
|  |  | ||||||
|     def test_creation_deletion_reserved_names(self): |     def test_creation_deletion_reserved_names(self): | ||||||
|         """ |         """ | ||||||
| @@ -1304,47 +1276,3 @@ class SchemaTests(TransactionTestCase): | |||||||
|             cursor.execute("SELECT surname FROM schema_author;") |             cursor.execute("SELECT surname FROM schema_author;") | ||||||
|             item = cursor.fetchall()[0] |             item = cursor.fetchall()[0] | ||||||
|             self.assertEqual(item[0], None if connection.features.interprets_empty_strings_as_nulls else '') |             self.assertEqual(item[0], None if connection.features.interprets_empty_strings_as_nulls else '') | ||||||
|  |  | ||||||
|     def test_custom_manytomanyfield(self): |  | ||||||
|         """ |  | ||||||
|         #24104 - Schema editors should look for many_to_many |  | ||||||
|         """ |  | ||||||
|         # Create the tables |  | ||||||
|         with connection.schema_editor() as editor: |  | ||||||
|             editor.create_model(AuthorWithM2M) |  | ||||||
|             editor.create_model(TagM2MTest) |  | ||||||
|         # Create an M2M field |  | ||||||
|         new_field = CustomManyToManyField("schema.TagM2MTest", related_name="authors") |  | ||||||
|         new_field.contribute_to_class(AuthorWithM2M, "tags") |  | ||||||
|         # Ensure there's no m2m table there |  | ||||||
|         self.assertRaises(DatabaseError, self.column_classes, new_field.rel.through) |  | ||||||
|         try: |  | ||||||
|             # Add the field |  | ||||||
|             with connection.schema_editor() as editor: |  | ||||||
|                 editor.add_field( |  | ||||||
|                     AuthorWithM2M, |  | ||||||
|                     new_field, |  | ||||||
|                 ) |  | ||||||
|             # Ensure there is now an m2m table there |  | ||||||
|             columns = self.column_classes(new_field.rel.through) |  | ||||||
|             self.assertEqual(columns['tagm2mtest_id'][0], "IntegerField") |  | ||||||
|  |  | ||||||
|             # "Alter" the field. This should not rename the DB table to itself. |  | ||||||
|             with connection.schema_editor() as editor: |  | ||||||
|                 editor.alter_field( |  | ||||||
|                     AuthorWithM2M, |  | ||||||
|                     new_field, |  | ||||||
|                     new_field, |  | ||||||
|                 ) |  | ||||||
|  |  | ||||||
|             # Remove the M2M table again |  | ||||||
|             with connection.schema_editor() as editor: |  | ||||||
|                 editor.remove_field( |  | ||||||
|                     AuthorWithM2M, |  | ||||||
|                     new_field, |  | ||||||
|                 ) |  | ||||||
|             # Ensure there's no m2m table there |  | ||||||
|             self.assertRaises(DatabaseError, self.column_classes, new_field.rel.through) |  | ||||||
|         finally: |  | ||||||
|             # Cleanup model states |  | ||||||
|             AuthorWithM2M._meta.local_many_to_many.remove(new_field) |  | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user