{
    "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 E is the most relevant as it deals with issues involving references to model fields, specifically similar to how the CURRENT bug is about handling template filters and outputs where certain fields are affected by autoescaping behavior. Although they are not directly related in terms of error types, they both deal with the implications of how references/fields are managed, and Candidate E provides a clear context of how those references can lead to errors in handling user inputs and migrations, which is analogous to managing join outputs in templates. The failure to correctly apply field references can lead to misunderstood outputs, paralleling the autoescape bug. The impact and patch similarity further enhance its relevance for resolving the current issue."
}