From a0ed3cfad1ac4873ecfd2baebaa2e978e483ba99 Mon Sep 17 00:00:00 2001 From: Baptiste Mispelon Date: Fri, 19 Nov 2021 21:45:33 +0100 Subject: [PATCH] Fixed #33305 -- Fixed autodetector crash for ForeignKey with hardcoded "to" attribute. Co-authored-by: Simon Charette --- django/db/migrations/autodetector.py | 2 +- tests/migrations/test_autodetector.py | 22 ++++++++++++++++++++++ 2 files changed, 23 insertions(+), 1 deletion(-) diff --git a/django/db/migrations/autodetector.py b/django/db/migrations/autodetector.py index 97977d72ef0..bf9c2acd260 100644 --- a/django/db/migrations/autodetector.py +++ b/django/db/migrations/autodetector.py @@ -96,7 +96,7 @@ class MigrationAutodetector: for name, field in sorted(fields.items()): deconstruction = self.deep_deconstruct(field) if field.remote_field and field.remote_field.model: - del deconstruction[2]['to'] + deconstruction[2].pop('to', None) fields_def.append(deconstruction) return fields_def diff --git a/tests/migrations/test_autodetector.py b/tests/migrations/test_autodetector.py index 3b466011456..18055f242b4 100644 --- a/tests/migrations/test_autodetector.py +++ b/tests/migrations/test_autodetector.py @@ -2834,6 +2834,28 @@ class AutodetectorTests(TestCase): expected_number, ) + def test_add_custom_fk_with_hardcoded_to(self): + class HardcodedForeignKey(models.ForeignKey): + def __init__(self, *args, **kwargs): + kwargs['to'] = 'testapp.Author' + super().__init__(*args, **kwargs) + + def deconstruct(self): + name, path, args, kwargs = super().deconstruct() + del kwargs['to'] + return name, path, args, kwargs + + book_hardcoded_fk_to = ModelState('testapp', 'Book', [ + ('author', HardcodedForeignKey(on_delete=models.CASCADE)), + ]) + changes = self.get_changes( + [self.author_empty], + [self.author_empty, book_hardcoded_fk_to], + ) + self.assertNumberMigrations(changes, 'testapp', 1) + self.assertOperationTypes(changes, 'testapp', 0, ['CreateModel']) + self.assertOperationAttributes(changes, 'testapp', 0, 0, name='Book') + class MigrationSuggestNameTests(SimpleTestCase): def test_no_operations(self):