{
  "Selected_candidate": {
    "pr_number": 5969,
    "pr_title": "[1.9.x] Fixed #25858 -- Bound abstract model application relative relationships.",
    "pr_body": "Thanks to Karl Hobley for the report and Markus, Shai, Aymeric for their input\nand Tim for the review.\n\nBackport of bc7d201bdbaeac14a49f51a9ef292d6312b4c45e from master\n",
    "issue_id": 25858,
    "issue_title": "Deriving from abstract model with foreign key to model in same app broken on Django 1.9",
    "issue_body": "As of Django 1.9, it is no longer possible to derive a new model from abstract model in another app if that abstract model has a local foreign key to a model in the same app.\nSee example project here:\n​\nhttps://github.com/kaedroho/djangobugtest\nFor example, I have two apps: app1 and app2\napp1 contains a concrete model and an abstract model. The abstract model has a foreign key to the concrete one (note the Foreign key is defined as a string and doesn't include the app label):\nclass ConcreteModel(models.Model):\n    pass\n\n\nclass AbstractModel(models.Model):\n    foreign_key = models.ForeignKey('ConcreteModel')\n\n    class Meta:\n        abstract = True\nIn app2, we have another model that derives from the abstract model in app 1\nclass DerivedModel(app1.AbstractModel):\n    pass\nWhen running makemigrations, the following system check error occurs:\napp2.DerivedModel.foreign_key: (fields.E300) Field defines a relation with model 'ConcreteModel', which is either not installed, or is abstract.\nFor apps upgrading to Django 1.9 and already have migrations, the following error occurs when running tests (after the database has been initialised but before the first test runs. It looks like it's happening while serialising the database):\nValueError: Related model u'ConcreteModel' cannot be resolved\nTraceback from an actual site:\n​\nhttps://gist.github.com/kaedroho/01b29332308018abfa29\nThis issue only occurs when the ForeignKey is defined with a string that doesn't contain the app label. So referencing the model class directly or using \"app_label.ModelName\" works.",
    "issue_closed_at": "2016-01-11T12:59:11",
    "base_commit": "76b4015ea9b02fdbc3acebe71ed79bcc902697da",
    "changes": [
      {
        "file": "django/db/models/fields/related.py",
        "type": "line",
        "name": "line 39",
        "code": "RECURSIVE_RELATIONSHIP_CONSTANT = 'self'\n\n\ndef resolve_relation(scope_model, relation):\n    \"\"\"\n    Transform relation into a model or fully-qualified model string of the form\n    \"app_label.ModelName\", relative to scope_model."
      },
      {
        "file": "django/db/models/fields/related.py",
        "type": "function",
        "name": "resolve_relation",
        "class_name": null,
        "code": "def resolve_relation(scope_model, relation):\n    \"\"\"\n    Transform relation into a model or fully-qualified model string of the form\n    \"app_label.ModelName\", relative to scope_model.\n\n    The relation argument can be:\n      * RECURSIVE_RELATIONSHIP_CONSTANT, i.e. the string \"self\", in which case\n        the model argument will be returned.\n      * A bare model name without an app_label, in which case scope_model's\n        app_label will be prepended.\n      * An \"app_label.ModelName\" string.\n      * A model class, which will be returned unchanged.\n    \"\"\"\n    # Check for recursive relations\n    if relation == RECURSIVE_RELATIONSHIP_CONSTANT:\n        relation = scope_model\n\n    # Look for an \"app.Model\" relation\n    if isinstance(relation, six.string_types):\n        if \".\" not in relation:\n            relation = \"%s.%s\" % (scope_model._meta.app_label, relation)\n\n    return relation"
      },
      {
        "file": "django/db/models/fields/related.py",
        "type": "function",
        "name": "resolve_related_class",
        "class_name": "RelatedField",
        "code": "def resolve_related_class(model, related, field):\n                field.remote_field.model = related\n                field.do_related_class(related, model)"
      },
      {
        "file": "django/db/models/fields/related.py",
        "type": "function",
        "name": "resolve_through_model",
        "class_name": "ManyToManyField",
        "code": "def resolve_through_model(_, model, field):\n                    field.remote_field.through = model"
      }
    ]
  },
  "Justification": "Candidate C addresses issues related to abstract models and their inheritance in Django, which is directly relevant to the CURRENT bug regarding the `get_FOO_display` method in an inherited model setup. The bug in Candidate C also involves relationships and constraints defined in abstract models, providing insight into how Django handles these structures. Since the CURRENT bug involves a similar concept of choices being overridden in an inherited model, the details around the fixes and the underlying structure will likely guide the developer towards understanding the current issue more effectively than the other reports.",
  "instance_id": "django__django-12284",
  "repo": "django/django",
  "created_at": "2020-01-07T11:06:31Z",
  "problem_statement": "Model.get_FOO_display() does not work correctly with inherited choices.\nDescription\n\t \n\t\t(last modified by Mariusz Felisiak)\n\t \nGiven a base model with choices A containing 3 tuples\nChild Model inherits the base model overrides the choices A and adds 2 more tuples\nget_foo_display does not work correctly for the new tuples added\nExample:\nclass A(models.Model):\n foo_choice = [(\"A\",\"output1\"),(\"B\",\"output2\")]\n field_foo = models.CharField(max_length=254,choices=foo_choice)\n class Meta:\n\t abstract = True\nclass B(A):\n foo_choice = [(\"A\",\"output1\"),(\"B\",\"output2\"),(\"C\",\"output3\")]\n field_foo = models.CharField(max_length=254,choices=foo_choice)\nUpon invoking get_field_foo_display() on instance of B , \nFor value \"A\" and \"B\" the output works correctly i.e. returns \"output1\" / \"output2\"\nbut for value \"C\" the method returns \"C\" and not \"output3\" which is the expected behaviour\n",
  "patch": "diff --git a/django/db/models/fields/__init__.py b/django/db/models/fields/__init__.py\n--- a/django/db/models/fields/__init__.py\n+++ b/django/db/models/fields/__init__.py\n@@ -764,7 +764,11 @@ def contribute_to_class(self, cls, name, private_only=False):\n             if not getattr(cls, self.attname, None):\n                 setattr(cls, self.attname, self.descriptor_class(self))\n         if self.choices is not None:\n-            if not hasattr(cls, 'get_%s_display' % self.name):\n+            # Don't override a get_FOO_display() method defined explicitly on\n+            # this class, but don't check methods derived from inheritance, to\n+            # allow overriding inherited choices. For more complex inheritance\n+            # structures users should override contribute_to_class().\n+            if 'get_%s_display' % self.name not in cls.__dict__:\n                 setattr(\n                     cls,\n                     'get_%s_display' % self.name,\n"
}