{
  "Selected_candidate": {
    "pr_number": 9383,
    "pr_title": " Fixed #25817 -- Made RenameField repoint to_field/to_fields references. ",
    "pr_body": "https://code.djangoproject.com/ticket/25817\r\n\r\nThis is still missing tests for `ForeignObject` multiple `to_fields` and still causes SQLite to crash (probably because of the table rebuild).",
    "issue_id": 25817,
    "issue_title": "Unable to rename a field reference in foreign key 'to_field'",
    "issue_body": "Steps to reproduce:\nCreate a model with a foreign key referencing another model's field via the 'to_field' arg.\nGenerate the initial migration\nclass\nBar\n(\nmodels\n.\nModel\n):\nbar_id\n=\nmodels\n.\nCharField\n(\nmax_length\n=\n255\n,\ndb_index\n=\nTrue\n,\nunique\n=\nTrue\n)\nclass\nBazz\n(\nmodels\n.\nModel\n):\nbar\n=\nmodels\n.\nForeignKey\n(\nBar\n,\nto_field\n=\n'bar_id'\n)\nRename the field referenced in 'to_field' and create a migration for the change\nRename 'bar_id' to 'external_id':\nclass\nBar\n(\nmodels\n.\nModel\n):\nexternal_id\n=\nmodels\n.\nCharField\n(\nmax_length\n=\n255\n,\ndb_index\n=\nTrue\n,\nunique\n=\nTrue\n)\nclass\nBazz\n(\nmodels\n.\nModel\n):\nbar\n=\nmodels\n.\nForeignKey\n(\nBar\n,\nto_field\n=\n'external_id'\n)\nMigration:\noperations\n=\n[\nmigrations\n.\nRenameField\n(\nmodel_name\n=\n'bar'\n,\nold_name\n=\n'bar_id'\n,\nnew_name\n=\n'external_id'\n,\n),\nmigrations\n.\nAlterField\n(\nmodel_name\n=\n'bazz'\n,\nname\n=\n'bar'\n,\nfield\n=\nmodels\n.\nForeignKey\n(\nto\n=\n'form_processor.Bar'\n,\nto_field\n=\nb\n'external_id'\n),\npreserve_default\n=\nTrue\n,\n),\n]\nRun the migration\nError:\nTraceback (most recent call last):\n  File \"./manage.py\", line 73, in <module>\n    execute_from_command_line(sys.argv)\n  File \"/home/skelly/.virtualenvs/hq/local/lib/python2.7/site-packages/django/core/management/__init__.py\", line 385, in execute_from_command_line\n    utility.execute()\n  File \"/home/skelly/.virtualenvs/hq/local/lib/python2.7/site-packages/django/core/management/__init__.py\", line 377, in execute\n    self.fetch_command(subcommand).run_from_argv(self.argv)\n  File \"/home/skelly/.virtualenvs/hq/local/lib/python2.7/site-packages/django/core/management/base.py\", line 288, in run_from_argv\n    self.execute(*args, **options.__dict__)\n  File \"/home/skelly/.virtualenvs/hq/local/lib/python2.7/site-packages/django/core/management/commands/sqlmigrate.py\", line 30, in execute\n    return super(Command, self).execute(*args, **options)\n  File \"/home/skelly/.virtualenvs/hq/local/lib/python2.7/site-packages/django/core/management/base.py\", line 338, in execute\n    output = self.handle(*args, **options)\n  File \"/home/skelly/.virtualenvs/hq/local/lib/python2.7/site-packages/django/core/management/commands/sqlmigrate.py\", line 61, in handle\n    sql_statements = executor.collect_sql(plan)\n  File \"/home/skelly/.virtualenvs/hq/local/lib/python2.7/site-packages/django/db/migrations/executor.py\", line 82, in collect_sql\n    migration.apply(project_state, schema_editor, collect_sql=True)\n  File \"/home/skelly/.virtualenvs/hq/local/lib/python2.7/site-packages/django/db/migrations/migration.py\", line 108, in apply\n    operation.database_forwards(self.app_label, schema_editor, project_state, new_state)\n  File \"/home/skelly/.virtualenvs/hq/local/lib/python2.7/site-packages/django/db/migrations/operations/fields.py\", line 139, in database_forwards\n    schema_editor.alter_field(from_model, from_field, to_field)\n  File \"/home/skelly/.virtualenvs/hq/local/lib/python2.7/site-packages/django/db/backends/schema.py\", line 445, in alter_field\n    old_db_params = old_field.db_parameters(connection=self.connection)\n  File \"/home/skelly/.virtualenvs/hq/local/lib/python2.7/site-packages/django/db/models/fields/related.py\", line 1787, in db_parameters\n    return {\"type\": self.db_type(connection), \"check\": []}\n  File \"/home/skelly/.virtualenvs/hq/local/lib/python2.7/site-packages/django/db/models/fields/related.py\", line 1778, in db_type\n    rel_field = self.related_field\n  File \"/home/skelly/.virtualenvs/hq/local/lib/python2.7/site-packages/django/db/models/fields/related.py\", line 1684, in related_field\n    return self.foreign_related_fields[0]\n  File \"/home/skelly/.virtualenvs/hq/local/lib/python2.7/site-packages/django/db/models/fields/related.py\", line 1442, in foreign_related_fields\n    return tuple(rhs_field for lhs_field, rhs_field in self.related_fields)\n  File \"/home/skelly/.virtualenvs/hq/local/lib/python2.7/site-packages/django/db/models/fields/related.py\", line 1429, in related_fields\n    self._related_fields = self.resolve_related_fields()\n  File \"/home/skelly/.virtualenvs/hq/local/lib/python2.7/site-packages/django/db/models/fields/related.py\", line 1422, in resolve_related_fields\n    else self.rel.to._meta.get_field_by_name(to_field_name)[0])\n  File \"/home/skelly/.virtualenvs/hq/local/lib/python2.7/site-packages/django/db/models/options.py\", line 420, in get_field_by_name\n    % (self.object_name, name))\ndjango.db.models.fields.FieldDoesNotExist: Bar has no field named 'bar_id'",
    "issue_closed_at": "2017-12-30T14:17:55",
    "base_commit": "2faeb21d2f618d5bfe9f8f6c574730d3f9407b2a",
    "changes": [
      {
        "file": "django/db/migrations/autodetector.py",
        "type": "function",
        "name": "generate_altered_fields",
        "class_name": "MigrationAutodetector",
        "code": "def generate_altered_fields(self):\n        \"\"\"\n        Make AlterField operations, or possibly RemovedField/AddField if alter\n        isn's possible.\n        \"\"\"\n        for app_label, model_name, field_name in sorted(self.old_field_keys & self.new_field_keys):\n            # Did the field change?\n            old_model_name = self.renamed_models.get((app_label, model_name), model_name)\n            old_field_name = self.renamed_fields.get((app_label, model_name, field_name), field_name)\n            old_field = self.old_apps.get_model(app_label, old_model_name)._meta.get_field(old_field_name)\n            new_field = self.new_apps.get_model(app_label, model_name)._meta.get_field(field_name)\n            # Implement any model renames on relations; these are handled by RenameModel\n            # so we need to exclude them from the comparison\n            if hasattr(new_field, \"remote_field\") and getattr(new_field.remote_field, \"model\", None):\n                rename_key = (\n                    new_field.remote_field.model._meta.app_label,\n                    new_field.remote_field.model._meta.model_name,\n                )\n                if rename_key in self.renamed_models:\n                    new_field.remote_field.model = old_field.remote_field.model\n            if hasattr(new_field, \"remote_field\") and getattr(new_field.remote_field, \"through\", None):\n                rename_key = (\n                    new_field.remote_field.through._meta.app_label,\n                    new_field.remote_field.through._meta.model_name,\n                )\n                if rename_key in self.renamed_models:\n                    new_field.remote_field.through = old_field.remote_field.through\n            old_field_dec = self.deep_deconstruct(old_field)\n            new_field_dec = self.deep_deconstruct(new_field)\n            if old_field_dec != new_field_dec:\n                both_m2m = old_field.many_to_many and new_field.many_to_many\n                neither_m2m = not old_field.many_to_many and not new_field.many_to_many\n                if both_m2m or neither_m2m:\n                    # Either both fields are m2m or neither is\n                    preserve_default = True\n                    if (old_field.null and not new_field.null and not new_field.has_default() and\n                            not new_field.many_to_many):\n                        field = new_field.clone()\n                        new_default = self.questioner.ask_not_null_alteration(field_name, model_name)\n                        if new_default is not models.NOT_PROVIDED:\n                            field.default = new_default\n                            preserve_default = False\n                    else:\n                        field = new_field\n                    self.add_operation(\n                        app_label,\n                        operations.AlterField(\n                            model_name=model_name,\n                            name=field_name,\n                            field=field,\n                            preserve_default=preserve_default,\n                        )\n                    )\n                else:\n                    # We cannot alter between m2m and concrete fields\n                    self._generate_removed_field(app_label, model_name, field_name)\n                    self._generate_added_field(app_label, model_name, field_name)"
      },
      {
        "file": "django/db/migrations/operations/fields.py",
        "type": "function",
        "name": "state_forwards",
        "class_name": "RenameField",
        "code": "def state_forwards(self, app_label, state):\n        model_state = state.models[app_label, self.model_name_lower]\n        # Rename the field\n        fields = model_state.fields\n        for index, (name, field) in enumerate(fields):\n            if name == self.old_name:\n                fields[index] = (self.new_name, field)\n                # Delay rendering of relationships if it's not a relational\n                # field and not referenced by a foreign key.\n                delay = (\n                    not field.is_relation and\n                    not is_referenced_by_foreign_key(state, self.model_name_lower, field, self.name)\n                )\n                break\n        else:\n            raise FieldDoesNotExist(\n                \"%s.%s has no field named '%s'\" % (app_label, self.model_name, self.old_name)\n            )\n        # Fix index/unique_together to refer to the new field\n        options = model_state.options\n        for option in ('index_together', 'unique_together'):\n            if option in options:\n                options[option] = [\n                    [self.new_name if n == self.old_name else n for n in together]\n                    for together in options[option]\n                ]\n        state.reload_model(app_label, self.model_name_lower, delay=delay)"
      },
      {
        "file": "django/db/migrations/operations/fields.py",
        "type": "function",
        "name": "state_forwards",
        "class_name": "RenameField",
        "code": "def state_forwards(self, app_label, state):\n        model_state = state.models[app_label, self.model_name_lower]\n        # Rename the field\n        fields = model_state.fields\n        for index, (name, field) in enumerate(fields):\n            if name == self.old_name:\n                fields[index] = (self.new_name, field)\n                # Delay rendering of relationships if it's not a relational\n                # field and not referenced by a foreign key.\n                delay = (\n                    not field.is_relation and\n                    not is_referenced_by_foreign_key(state, self.model_name_lower, field, self.name)\n                )\n                break\n        else:\n            raise FieldDoesNotExist(\n                \"%s.%s has no field named '%s'\" % (app_label, self.model_name, self.old_name)\n            )\n        # Fix index/unique_together to refer to the new field\n        options = model_state.options\n        for option in ('index_together', 'unique_together'):\n            if option in options:\n                options[option] = [\n                    [self.new_name if n == self.old_name else n for n in together]\n                    for together in options[option]\n                ]\n        state.reload_model(app_label, self.model_name_lower, delay=delay)"
      },
      {
        "file": "django/db/migrations/operations/fields.py",
        "type": "function",
        "name": "state_forwards",
        "class_name": "RenameField",
        "code": "def state_forwards(self, app_label, state):\n        model_state = state.models[app_label, self.model_name_lower]\n        # Rename the field\n        fields = model_state.fields\n        for index, (name, field) in enumerate(fields):\n            if name == self.old_name:\n                fields[index] = (self.new_name, field)\n                # Delay rendering of relationships if it's not a relational\n                # field and not referenced by a foreign key.\n                delay = (\n                    not field.is_relation and\n                    not is_referenced_by_foreign_key(state, self.model_name_lower, field, self.name)\n                )\n                break\n        else:\n            raise FieldDoesNotExist(\n                \"%s.%s has no field named '%s'\" % (app_label, self.model_name, self.old_name)\n            )\n        # Fix index/unique_together to refer to the new field\n        options = model_state.options\n        for option in ('index_together', 'unique_together'):\n            if option in options:\n                options[option] = [\n                    [self.new_name if n == self.old_name else n for n in together]\n                    for together in options[option]\n                ]\n        state.reload_model(app_label, self.model_name_lower, delay=delay)"
      }
    ]
  },
  "Justification": "Candidate C is the most relevant because it directly addresses a similar issue with foreign keys and the renaming of fields that are referenced through the 'to_field' argument. Both the CURRENT bug and Candidate C's bug report revolve around difficulties encountered when renaming fields and ensuring that foreign key references correctly point to the new field name. The alignment in the problem domain and type of fix enhances its value in guiding the debugging process for the CURRENT bug, making it particularly useful for understanding the migration issues at play.",
  "instance_id": "django__django-11910",
  "repo": "django/django",
  "created_at": "2019-10-14T01:56:49Z",
  "problem_statement": "ForeignKey's to_field parameter gets the old field's name when renaming a PrimaryKey.\nDescription\n\t\nHaving these two models \nclass ModelA(models.Model):\n\tfield_wrong = models.CharField('field1', max_length=50, primary_key=True) # I'm a Primary key.\nclass ModelB(models.Model):\n\tfield_fk = models.ForeignKey(ModelA, blank=True, null=True, on_delete=models.CASCADE) \n... migrations applyed ...\nthe ModelA.field_wrong field has been renamed ... and Django recognizes the \"renaming\"\n# Primary key renamed\nclass ModelA(models.Model):\n\tfield_fixed = models.CharField('field1', max_length=50, primary_key=True) # I'm a Primary key.\nAttempts to to_field parameter. \nThe to_field points to the old_name (field_typo) and not to the new one (\"field_fixed\")\nclass Migration(migrations.Migration):\n\tdependencies = [\n\t\t('app1', '0001_initial'),\n\t]\n\toperations = [\n\t\tmigrations.RenameField(\n\t\t\tmodel_name='modela',\n\t\t\told_name='field_wrong',\n\t\t\tnew_name='field_fixed',\n\t\t),\n\t\tmigrations.AlterField(\n\t\t\tmodel_name='modelb',\n\t\t\tname='modela',\n\t\t\tfield=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.CASCADE, to='app1.ModelB', to_field='field_wrong'),\n\t\t),\n\t]\n",
  "patch": "diff --git a/django/db/migrations/autodetector.py b/django/db/migrations/autodetector.py\n--- a/django/db/migrations/autodetector.py\n+++ b/django/db/migrations/autodetector.py\n@@ -927,6 +927,10 @@ def generate_altered_fields(self):\n                 if remote_field_name:\n                     to_field_rename_key = rename_key + (remote_field_name,)\n                     if to_field_rename_key in self.renamed_fields:\n+                        # Repoint both model and field name because to_field\n+                        # inclusion in ForeignKey.deconstruct() is based on\n+                        # both.\n+                        new_field.remote_field.model = old_field.remote_field.model\n                         new_field.remote_field.field_name = old_field.remote_field.field_name\n                 # Handle ForeignObjects which can have multiple from_fields/to_fields.\n                 from_fields = getattr(new_field, 'from_fields', None)\n"
}