diff --git a/django/db/migrations/operations/models.py b/django/db/migrations/operations/models.py index 56cb8fe608a..69998331fb1 100644 --- a/django/db/migrations/operations/models.py +++ b/django/db/migrations/operations/models.py @@ -161,7 +161,7 @@ class RenameModel(Operation): # Get all of the related objects we need to repoint all_related_objects = ( f for f in model._meta.get_fields(include_hidden=True) - if f.auto_created and not f.concrete and not (f.hidden or f.many_to_many) + if f.auto_created and not f.concrete and (not f.hidden or f.many_to_many) ) # Rename the model state.models[app_label, self.new_name_lower] = state.models[app_label, self.old_name_lower] diff --git a/docs/releases/1.8.1.txt b/docs/releases/1.8.1.txt index d2b16d62da2..b0bcbbfeea8 100644 --- a/docs/releases/1.8.1.txt +++ b/docs/releases/1.8.1.txt @@ -72,6 +72,9 @@ Bugfixes * Restored the ability to use iterators as queryset filter arguments (:ticket:`24719`). +* Fixed a migration crash when renaming the target model of a many-to-many + relation (:ticket:`24725`). + Optimizations ============= diff --git a/tests/migrations/test_operations.py b/tests/migrations/test_operations.py index b8354f32d93..de5cf5813cf 100644 --- a/tests/migrations/test_operations.py +++ b/tests/migrations/test_operations.py @@ -614,6 +614,32 @@ class OperationTests(OperationTestBase): self.assertEqual(Rider.objects.count(), 2) self.assertEqual(Pony._meta.get_field('riders').remote_field.through.objects.count(), 2) + def test_rename_m2m_target_model(self): + app_label = "test_rename_m2m_target_model" + project_state = self.apply_operations(app_label, ProjectState(), operations=[ + migrations.CreateModel("Rider", fields=[]), + migrations.CreateModel("Pony", fields=[ + ("riders", models.ManyToManyField("Rider")), + ]), + ]) + Pony = project_state.apps.get_model(app_label, "Pony") + Rider = project_state.apps.get_model(app_label, "Rider") + pony = Pony.objects.create() + rider = Rider.objects.create() + pony.riders.add(rider) + + project_state = self.apply_operations(app_label, project_state, operations=[ + migrations.RenameModel("Rider", "Rider2"), + ]) + Pony = project_state.apps.get_model(app_label, "Pony") + Rider = project_state.apps.get_model(app_label, "Rider2") + pony = Pony.objects.create() + rider = Rider.objects.create() + pony.riders.add(rider) + self.assertEqual(Pony.objects.count(), 2) + self.assertEqual(Rider.objects.count(), 2) + self.assertEqual(Pony._meta.get_field('riders').remote_field.through.objects.count(), 2) + def test_add_field(self): """ Tests the AddField operation.