{
  "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",
  "similar_bug_items": [
    {
      "pr_number": 8213,
      "pr_title": "Refs #27935 -- Fixed BrinIndex.max_name_length if a project's default database isn't PostgreSQL.",
      "pr_body": "",
      "issue_id": 27935,
      "issue_title": "BrinIndex crashes if name  > 30 characters",
      "issue_body": "",
      "issue_closed_at": "2017-03-20T10:30:05",
      "base_commit": "6cb0a3ac2836c044987f7a0cca0d1e99d52e002e",
      "changes": [
        {
          "file": "django/contrib/postgres/indexes.py",
          "type": "line",
          "name": "line 1",
          "code": "from django.db import connection\nfrom django.db.models import Index\nfrom django.utils.functional import cached_property\n\n__all__ = ['BrinIndex', 'GinIndex']\n\n\nclass BrinIndex(Index):\n    suffix = 'brin'\n\n    def __init__(self, fields=[], name=None, pages_per_range=None):\n        if pages_per_range is not None and pages_per_range <= 0:"
        },
        {
          "file": "django/contrib/postgres/indexes.py",
          "type": "function",
          "name": "get_sql_create_template_values",
          "class_name": "BrinIndex",
          "code": "def get_sql_create_template_values(self, model, schema_editor, using):\n        parameters = super().get_sql_create_template_values(model, schema_editor, using=' USING brin')\n        if self.pages_per_range is not None:\n            parameters['extra'] = ' WITH (pages_per_range={})'.format(\n                schema_editor.quote_value(self.pages_per_range)) + parameters['extra']\n        return parameters"
        }
      ]
    },
    {
      "pr_number": 8899,
      "pr_title": "[1.11.x] Fixed #28451 -- Fixed regression in Oracle sequence naming",
      "pr_body": "Backport of 99c7a18f4edf84d3b2bbddd3dfc7c3287bc8fa23 from master,\r\nwith additional migration script & release notes.",
      "issue_id": 28451,
      "issue_title": "Change in Oracle sequence name truncation causes regression when updating existing database",
      "issue_body": "",
      "issue_closed_at": "2017-08-22T15:09:28",
      "base_commit": "60f81118f412268f317abbcc7509e315a714315d",
      "changes": [
        {
          "file": "django/db/backends/oracle/operations.py",
          "type": "function",
          "name": "combine_expression",
          "class_name": "DatabaseOperations",
          "code": "def combine_expression(self, connector, sub_expressions):\n        lhs, rhs = sub_expressions\n        if connector == '%%':\n            return 'MOD(%s)' % ','.join(sub_expressions)\n        elif connector == '&':\n            return 'BITAND(%s)' % ','.join(sub_expressions)\n        elif connector == '|':\n            return 'BITAND(-%(lhs)s-1,%(rhs)s)+%(lhs)s' % {'lhs': lhs, 'rhs': rhs}\n        elif connector == '<<':\n            return '(%(lhs)s * POWER(2, %(rhs)s))' % {'lhs': lhs, 'rhs': rhs}\n        elif connector == '>>':\n            return 'FLOOR(%(lhs)s / POWER(2, %(rhs)s))' % {'lhs': lhs, 'rhs': rhs}\n        elif connector == '^':\n            return 'POWER(%s)' % ','.join(sub_expressions)\n        return super(DatabaseOperations, self).combine_expression(connector, sub_expressions)"
        }
      ]
    },
    {
      "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": "",
      "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"
        }
      ]
    },
    {
      "pr_number": 9412,
      "pr_title": "Fixed #28876 -- Fixed class-based model indexes and constraints names for models with quoted db_table.",
      "pr_body": "https://code.djangoproject.com/ticket/28876",
      "issue_id": 28876,
      "issue_title": "badly formed Table Index and constraint's name when  `Meta.db_table` is set with quoted string",
      "issue_body": "",
      "issue_closed_at": "2017-12-05T16:01:30",
      "base_commit": "f2ec89691236b59f54f9cf41bccd79a5853af36c",
      "changes": [
        {
          "file": "django/db/backends/base/schema.py",
          "type": "function",
          "name": "_rename_field_sql",
          "class_name": "BaseDatabaseSchemaEditor",
          "code": "def _rename_field_sql(self, table, old_field, new_field, new_type):\n        return self.sql_rename_column % {\n            \"table\": self.quote_name(table),\n            \"old_column\": self.quote_name(old_field.column),\n            \"new_column\": self.quote_name(new_field.column),\n            \"type\": new_type,\n        }"
        },
        {
          "file": "django/db/backends/base/schema.py",
          "type": "function",
          "name": "create_fk_name",
          "class_name": "BaseDatabaseSchemaEditor",
          "code": "def create_fk_name(*args, **kwargs):\n            return self.quote_name(self._create_index_name(*args, **kwargs))"
        },
        {
          "file": "django/db/models/indexes.py",
          "type": "line",
          "name": "line 1",
          "code": "import hashlib\n\nfrom django.utils.encoding import force_bytes\n\n__all__ = ['Index']"
        },
        {
          "file": "django/db/models/indexes.py",
          "type": "function",
          "name": "set_name_with_model",
          "class_name": "Index",
          "code": "def set_name_with_model(self, model):\n        \"\"\"\n        Generate a unique name for the index.\n\n        The name is divided into 3 parts - table name (12 chars), field name\n        (8 chars) and unique hash + suffix (10 chars). Each part is made to\n        fit its size by truncating the excess length.\n        \"\"\"\n        table_name = model._meta.db_table\n        column_names = [model._meta.get_field(field_name).column for field_name, order in self.fields_orders]\n        column_names_with_order = [\n            (('-%s' if order else '%s') % column_name)\n            for column_name, (field_name, order) in zip(column_names, self.fields_orders)\n        ]\n        # The length of the parts of the name is based on the default max\n        # length of 30 characters.\n        hash_data = [table_name] + column_names_with_order + [self.suffix]\n        self.name = '%s_%s_%s' % (\n            table_name[:11],\n            column_names[0][:7],\n            '%s_%s' % (self._hash_generator(*hash_data), self.suffix),\n        )\n        assert len(self.name) <= self.max_name_length, (\n            'Index too long for multiple database support. Is self.suffix '\n            'longer than 3 characters?'\n        )\n        self.check_name()"
        }
      ]
    },
    {
      "pr_number": 8713,
      "pr_title": "Fixed #28382 -- Prevented BaseExpression._output_field from being set if _resolve_output_field() fails.",
      "pr_body": "https://code.djangoproject.com/ticket/28382",
      "issue_id": 28382,
      "issue_title": "BaseExpression.output_field should not be set after BaseExpression._resolve_output_field() fails",
      "issue_body": "",
      "issue_closed_at": "2017-07-11T07:46:42",
      "base_commit": "306b961a4dc1fd308c6f298b406fb41906ebaf2d",
      "changes": [
        {
          "file": "django/db/models/aggregates.py",
          "type": "class",
          "name": "Avg",
          "code": "class Avg(Aggregate):\n    function = 'AVG'\n    name = 'Avg'\n\n    def _resolve_output_field(self):\n        source_field = self.get_source_fields()[0]\n        if isinstance(source_field, (IntegerField, DecimalField)):\n            self._output_field = FloatField()\n        super()._resolve_output_field()\n\n    def as_oracle(self, compiler, connection):\n        if self.output_field.get_internal_type() == 'DurationField':\n            expression = self.get_source_expressions()[0]\n            from django.db.backends.oracle.functions import IntervalToSeconds, SecondsToInterval\n            return compiler.compile(\n                SecondsToInterval(Avg(IntervalToSeconds(expression)))\n            )\n        return super().as_sql(compiler, connection)"
        },
        {
          "file": "django/db/models/expressions.py",
          "type": "function",
          "name": "_output_field_or_none",
          "class_name": "BaseExpression",
          "code": "def _output_field_or_none(self):\n        \"\"\"\n        Return the output field of this expression, or None if no output type\n        can be resolved. Note that the 'output_field' property will raise\n        FieldError if no type can be resolved, but this attribute allows for\n        None values.\n        \"\"\"\n        if self._output_field is None:\n            self._resolve_output_field()\n        return self._output_field"
        },
        {
          "file": "django/db/models/expressions.py",
          "type": "function",
          "name": "_resolve_output_field",
          "class_name": "BaseExpression",
          "code": "def _resolve_output_field(self):\n        \"\"\"\n        Attempt to infer the output type of the expression. If the output\n        fields of all source fields match then, simply infer the same type\n        here. This isn't always correct, but it makes sense most of the time.\n\n        Consider the difference between `2 + 2` and `2 / 3`. Inferring\n        the type here is a convenience for the common case. The user should\n        supply their own output_field with more complex computations.\n\n        If a source does not have an `_output_field` then we exclude it from\n        this check. If all sources are `None`, then an error will be thrown\n        higher up the stack in the `output_field` property.\n        \"\"\"\n        if self._output_field is None:\n            sources = self.get_source_fields()\n            num_sources = len(sources)\n            if num_sources == 0:\n                self._output_field = None\n            else:\n                for source in sources:\n                    if self._output_field is None:\n                        self._output_field = source\n                    if source is not None and not isinstance(self._output_field, source.__class__):\n                        raise FieldError(\n                            \"Expression contains mixed types. You must set output_field\")"
        }
      ]
    }
  ]
}