{
  "instance_id": "django__django-13028",
  "repo": "django/django",
  "created_at": "2020-06-05T19:49:04Z",
  "problem_statement": "Queryset raises NotSupportedError when RHS has filterable=False attribute.\nDescription\n\t \n\t\t(last modified by Nicolas Baccelli)\n\t \nI'm migrating my app to django 3.0.7 and I hit a strange behavior using a model class with a field labeled filterable\nclass ProductMetaDataType(models.Model):\n\tlabel = models.CharField(max_length=255, unique=True, blank=False, null=False)\n\tfilterable = models.BooleanField(default=False, verbose_name=_(\"filterable\"))\n\tclass Meta:\n\t\tapp_label = \"adminpricing\"\n\t\tverbose_name = _(\"product meta data type\")\n\t\tverbose_name_plural = _(\"product meta data types\")\n\tdef __str__(self):\n\t\treturn self.label\nclass ProductMetaData(models.Model):\n\tid = models.BigAutoField(primary_key=True)\n\tproduct = models.ForeignKey(\n\t\tProduit, null=False, blank=False, on_delete=models.CASCADE\n\t)\n\tvalue = models.TextField(null=False, blank=False)\n\tmarketplace = models.ForeignKey(\n\t\tPlateforme, null=False, blank=False, on_delete=models.CASCADE\n\t)\n\tdate_created = models.DateTimeField(null=True, default=timezone.now)\n\tmetadata_type = models.ForeignKey(\n\t\tProductMetaDataType, null=False, blank=False, on_delete=models.CASCADE\n\t)\n\tclass Meta:\n\t\tapp_label = \"adminpricing\"\n\t\tverbose_name = _(\"product meta data\")\n\t\tverbose_name_plural = _(\"product meta datas\")\nError happened when filtering ProductMetaData with a metadata_type :\nProductMetaData.objects.filter(value=\"Dark Vador\", metadata_type=self.brand_metadata)\nError traceback :\nTraceback (most recent call last):\n File \"/backoffice/backoffice/adminpricing/tests/test_pw.py\", line 481, in test_checkpolicywarning_by_fields\n\tfor p in ProductMetaData.objects.filter(\n File \"/usr/local/lib/python3.8/site-packages/django/db/models/manager.py\", line 82, in manager_method\n\treturn getattr(self.get_queryset(), name)(*args, **kwargs)\n File \"/usr/local/lib/python3.8/site-packages/django/db/models/query.py\", line 904, in filter\n\treturn self._filter_or_exclude(False, *args, **kwargs)\n File \"/usr/local/lib/python3.8/site-packages/django/db/models/query.py\", line 923, in _filter_or_exclude\n\tclone.query.add_q(Q(*args, **kwargs))\n File \"/usr/local/lib/python3.8/site-packages/django/db/models/sql/query.py\", line 1351, in add_q\n\tclause, _ = self._add_q(q_object, self.used_aliases)\n File \"/usr/local/lib/python3.8/site-packages/django/db/models/sql/query.py\", line 1378, in _add_q\n\tchild_clause, needed_inner = self.build_filter(\n File \"/usr/local/lib/python3.8/site-packages/django/db/models/sql/query.py\", line 1264, in build_filter\n\tself.check_filterable(value)\n File \"/usr/local/lib/python3.8/site-packages/django/db/models/sql/query.py\", line 1131, in check_filterable\n\traise NotSupportedError(\ndjango.db.utils.NotSupportedError: ProductMetaDataType is disallowed in the filter clause.\nI changed label to filterable_test and it fixed this issue\nThis should be documented or fix.\n",
  "patch": "diff --git a/django/db/models/sql/query.py b/django/db/models/sql/query.py\n--- a/django/db/models/sql/query.py\n+++ b/django/db/models/sql/query.py\n@@ -1124,7 +1124,10 @@ def check_related_objects(self, field, value, opts):\n \n     def check_filterable(self, expression):\n         \"\"\"Raise an error if expression cannot be used in a WHERE clause.\"\"\"\n-        if not getattr(expression, 'filterable', True):\n+        if (\n+            hasattr(expression, 'resolve_expression') and\n+            not getattr(expression, 'filterable', True)\n+        ):\n             raise NotSupportedError(\n                 expression.__class__.__name__ + ' is disallowed in the filter '\n                 'clause.'\n",
  "similar_bug_items": [
    {
      "pr_number": 6178,
      "pr_title": "Fixed #26186 -- Documented how app relative relationships of abstract models behave.",
      "pr_body": "This partially reverts commit bc7d201bdbaeac14a49f51a9ef292d6312b4c45e.\n\nRefs #25858.\n",
      "issue_id": 26186,
      "issue_title": "When extending from abstract model, ForeignKey points to wrong application",
      "issue_body": "",
      "issue_closed_at": "2016-02-29T21:07:53",
      "base_commit": "eac1423f9ebcf432dc5be95d605d124a05ab2686",
      "changes": [
        {
          "file": "django/db/models/fields/related.py",
          "type": "line",
          "name": "line 35",
          "code": "RECURSIVE_RELATIONSHIP_CONSTANT = 'self'\n\n\ndef resolve_relation(scope_model, relation, resolve_recursive_relationship=True):\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, resolve_recursive_relationship=True):\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        if resolve_recursive_relationship:\n            relation = scope_model\n    # Look for an \"app.Model\" relation\n    elif isinstance(relation, six.string_types) and '.' 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": 9112,
      "pr_title": "Fixed #27846 -- clear all cached reverse relationships on refresh_from_db()",
      "pr_body": "https://code.djangoproject.com/ticket/27846",
      "issue_id": 27846,
      "issue_title": "refresh_from_db() doesn't clear reverse OneToOneFields",
      "issue_body": "",
      "issue_closed_at": "2017-10-12T16:25:22",
      "base_commit": "df0aebc893973c78d7d2cda712ba4133dbe29b6e",
      "changes": [
        {
          "file": "django/db/models/base.py",
          "type": "function",
          "name": "refresh_from_db",
          "class_name": "Model",
          "code": "def refresh_from_db(self, using=None, fields=None):\n        \"\"\"\n        Reload field values from the database.\n\n        By default, the reloading happens from the database this instance was\n        loaded from, or by the read router if this instance wasn't loaded from\n        any database. The using parameter will override the default.\n\n        Fields can be used to specify which fields to reload. The fields\n        should be an iterable of field attnames. If fields is None, then\n        all non-deferred fields are reloaded.\n\n        When accessing deferred fields of an instance, the deferred loading\n        of the field will call this method.\n        \"\"\"\n        if fields is not None:\n            if len(fields) == 0:\n                return\n            if any(LOOKUP_SEP in f for f in fields):\n                raise ValueError(\n                    'Found \"%s\" in fields argument. Relations and transforms '\n                    'are not allowed in fields.' % LOOKUP_SEP)\n\n        db = using if using is not None else self._state.db\n        db_instance_qs = self.__class__._default_manager.using(db).filter(pk=self.pk)\n\n        # Use provided fields, if not set then reload all non-deferred fields.\n        deferred_fields = self.get_deferred_fields()\n        if fields is not None:\n            fields = list(fields)\n            db_instance_qs = db_instance_qs.only(*fields)\n        elif deferred_fields:\n            fields = [f.attname for f in self._meta.concrete_fields\n                      if f.attname not in deferred_fields]\n            db_instance_qs = db_instance_qs.only(*fields)\n\n        db_instance = db_instance_qs.get()\n        non_loaded_fields = db_instance.get_deferred_fields()\n        for field in self._meta.concrete_fields:\n            if field.attname in non_loaded_fields:\n                # This field wasn't refreshed - skip ahead.\n                continue\n            setattr(self, field.attname, getattr(db_instance, field.attname))\n            # Throw away stale foreign key references.\n            if field.is_relation and field.is_cached(self):\n                rel_instance = field.get_cached_value(self)\n                local_val = getattr(db_instance, field.attname)\n                related_val = None if rel_instance is None else getattr(rel_instance, field.target_field.attname)\n                if local_val != related_val or (local_val is None and related_val is None):\n                    field.delete_cached_value(self)\n        self._state.db = db_instance._state.db"
        }
      ]
    },
    {
      "pr_number": 11886,
      "pr_title": "Fixed #30405 -- Fixed source code mismatch crash in ExceptionReporter. ",
      "pr_body": "[ticket 30405](https://code.djangoproject.com/ticket/30405)",
      "issue_id": 30405,
      "issue_title": "IndexError in _get_lines_from_file when module does not match file contents (via loader)",
      "issue_body": "",
      "issue_closed_at": "2019-11-12T04:53:04",
      "base_commit": "6e2f05b2e33a6c80c7a411ce76af7b5a08acb835",
      "changes": [
        {
          "file": "django/views/debug.py",
          "type": "function",
          "name": "get_traceback_text",
          "class_name": "ExceptionReporter",
          "code": "def get_traceback_text(self):\n        \"\"\"Return plain text version of debug 500 HTTP error page.\"\"\"\n        with Path(CURRENT_DIR, 'templates', 'technical_500.txt').open(encoding='utf-8') as fh:\n            t = DEBUG_ENGINE.from_string(fh.read())\n        c = Context(self.get_traceback_data(), autoescape=False, use_l10n=False)\n        return t.render(c)"
        },
        {
          "file": "django/views/debug.py",
          "type": "function",
          "name": "_get_lines_from_file",
          "class_name": "ExceptionReporter",
          "code": "def _get_lines_from_file(self, filename, lineno, context_lines, loader=None, module_name=None):\n        \"\"\"\n        Return context_lines before and after lineno from file.\n        Return (pre_context_lineno, pre_context, context_line, post_context).\n        \"\"\"\n        source = None\n        if hasattr(loader, 'get_source'):\n            try:\n                source = loader.get_source(module_name)\n            except ImportError:\n                pass\n            if source is not None:\n                source = source.splitlines()\n        if source is None:\n            try:\n                with open(filename, 'rb') as fp:\n                    source = fp.read().splitlines()\n            except OSError:\n                pass\n        if source is None:\n            return None, [], None, []\n\n        # If we just read the source from a file, or if the loader did not\n        # apply tokenize.detect_encoding to decode the source into a\n        # string, then we should do that ourselves.\n        if isinstance(source[0], bytes):\n            encoding = 'ascii'\n            for line in source[:2]:\n                # File coding may be specified. Match pattern from PEP-263\n                # (https://www.python.org/dev/peps/pep-0263/)\n                match = re.search(br'coding[:=]\\s*([-\\w.]+)', line)\n                if match:\n                    encoding = match.group(1).decode('ascii')\n                    break\n            source = [str(sline, encoding, 'replace') for sline in source]\n\n        lower_bound = max(0, lineno - context_lines)\n        upper_bound = lineno + context_lines\n\n        pre_context = source[lower_bound:lineno]\n        context_line = source[lineno]\n        post_context = source[lineno + 1:upper_bound]\n\n        return lower_bound, pre_context, context_line, post_context"
        },
        {
          "file": "django/views/debug.py",
          "type": "function",
          "name": "_get_lines_from_file",
          "class_name": "ExceptionReporter",
          "code": "def _get_lines_from_file(self, filename, lineno, context_lines, loader=None, module_name=None):\n        \"\"\"\n        Return context_lines before and after lineno from file.\n        Return (pre_context_lineno, pre_context, context_line, post_context).\n        \"\"\"\n        source = None\n        if hasattr(loader, 'get_source'):\n            try:\n                source = loader.get_source(module_name)\n            except ImportError:\n                pass\n            if source is not None:\n                source = source.splitlines()\n        if source is None:\n            try:\n                with open(filename, 'rb') as fp:\n                    source = fp.read().splitlines()\n            except OSError:\n                pass\n        if source is None:\n            return None, [], None, []\n\n        # If we just read the source from a file, or if the loader did not\n        # apply tokenize.detect_encoding to decode the source into a\n        # string, then we should do that ourselves.\n        if isinstance(source[0], bytes):\n            encoding = 'ascii'\n            for line in source[:2]:\n                # File coding may be specified. Match pattern from PEP-263\n                # (https://www.python.org/dev/peps/pep-0263/)\n                match = re.search(br'coding[:=]\\s*([-\\w.]+)', line)\n                if match:\n                    encoding = match.group(1).decode('ascii')\n                    break\n            source = [str(sline, encoding, 'replace') for sline in source]\n\n        lower_bound = max(0, lineno - context_lines)\n        upper_bound = lineno + context_lines\n\n        pre_context = source[lower_bound:lineno]\n        context_line = source[lineno]\n        post_context = source[lineno + 1:upper_bound]\n\n        return lower_bound, pre_context, context_line, post_context"
        }
      ]
    },
    {
      "pr_number": 7036,
      "pr_title": "Fixed #27024 -- Prevented logging error with empty string as geometry widget value",
      "pr_body": "Thanks Gavin Wahl for the report.\n",
      "issue_id": 27024,
      "issue_title": "BaseGeometryWidget logs a false positive: Error creating geometry from value ''",
      "issue_body": "",
      "issue_closed_at": "2016-08-08T09:25:25",
      "base_commit": "2a11d2d7a7d5c6609c85dbc631fad6b8a8645a64",
      "changes": [
        {
          "file": "django/contrib/gis/admin/widgets.py",
          "type": "function",
          "name": "render",
          "class_name": "OpenLayersWidget",
          "code": "def render(self, name, value, attrs=None):\n        # Update the template parameters with any attributes passed in.\n        if attrs:\n            self.params.update(attrs)\n            self.params['editable'] = self.params['modifiable']\n        else:\n            self.params['editable'] = True\n\n        # Defaulting the WKT value to a blank string -- this\n        # will be tested in the JavaScript and the appropriate\n        # interface will be constructed.\n        self.params['wkt'] = ''\n\n        # If a string reaches here (via a validation error on another\n        # field) then just reconstruct the Geometry.\n        if isinstance(value, six.string_types):\n            try:\n                value = GEOSGeometry(value)\n            except (GEOSException, ValueError) as err:\n                logger.error(\"Error creating geometry from value '%s' (%s)\", value, err)\n                value = None\n\n        if (value and value.geom_type.upper() != self.geom_type and\n                self.geom_type != 'GEOMETRY'):\n            value = None\n\n        # Constructing the dictionary of the map options.\n        self.params['map_options'] = self.map_options()\n\n        # Constructing the JavaScript module name using the name of\n        # the GeometryField (passed in via the `attrs` keyword).\n        # Use the 'name' attr for the field name (rather than 'field')\n        self.params['name'] = name\n        # note: we must switch out dashes for underscores since js\n        # functions are created using the module variable\n        js_safe_name = self.params['name'].replace('-', '_')\n        self.params['module'] = 'geodjango_%s' % js_safe_name\n\n        if value:\n            # Transforming the geometry to the projection used on the\n            # OpenLayers map.\n            srid = self.params['srid']\n            if value.srid != srid:\n                try:\n                    ogr = value.ogr\n                    ogr.transform(srid)\n                    wkt = ogr.wkt\n                except GDALException as err:\n                    logger.error(\n                        \"Error transforming geometry from srid '%s' to srid '%s' (%s)\",\n                        value.srid, srid, err\n                    )\n                    wkt = ''\n            else:\n                wkt = value.wkt\n\n            # Setting the parameter WKT with that of the transformed\n            # geometry.\n            self.params['wkt'] = wkt\n\n        self.params.update(geo_context)\n        return loader.render_to_string(self.template, self.params)"
        },
        {
          "file": "django/contrib/gis/forms/widgets.py",
          "type": "function",
          "name": "deserialize",
          "class_name": "BaseGeometryWidget",
          "code": "def deserialize(self, value):\n        try:\n            return GEOSGeometry(value, self.map_srid)\n        except (GEOSException, ValueError) as err:\n            logger.error(\"Error creating geometry from value '%s' (%s)\", value, err)\n        return None"
        }
      ]
    },
    {
      "pr_number": 12329,
      "pr_title": "Fixed #31162 -- Prevented error logs when using WKT strings in lookups.",
      "pr_body": "ticket-31162\r\n\r\nRegression in 6f44f714c92d2966dca390ebd3054e5fb0bb0c80.",
      "issue_id": 31162,
      "issue_title": "GIS error logging when using WKT string as input to filter() query.",
      "issue_body": "",
      "issue_closed_at": "2020-01-16T07:35:03",
      "base_commit": "1e0dcd6c8bfa4519c21014c73eb510620dd1a000",
      "changes": [
        {
          "file": "django/contrib/gis/gdal/raster/source.py",
          "type": "function",
          "name": "__init__",
          "class_name": "GDALRaster",
          "code": "def __init__(self, ds_input, write=False):\n        self._write = 1 if write else 0\n        Driver.ensure_registered()\n\n        # Preprocess json inputs. This converts json strings to dictionaries,\n        # which are parsed below the same way as direct dictionary inputs.\n        if isinstance(ds_input, str) and json_regex.match(ds_input):\n            ds_input = json.loads(ds_input)\n\n        # If input is a valid file path, try setting file as source.\n        if isinstance(ds_input, str):\n            try:\n                # GDALOpen will auto-detect the data source type.\n                self._ptr = capi.open_ds(force_bytes(ds_input), self._write)\n            except GDALException as err:\n                raise GDALException('Could not open the datasource at \"{}\" ({}).'.format(ds_input, err))\n        elif isinstance(ds_input, bytes):\n            # Create a new raster in write mode.\n            self._write = 1\n            # Get size of buffer.\n            size = sys.getsizeof(ds_input)\n            # Pass data to ctypes, keeping a reference to the ctypes object so\n            # that the vsimem file remains available until the GDALRaster is\n            # deleted.\n            self._ds_input = c_buffer(ds_input)\n            # Create random name to reference in vsimem filesystem.\n            vsi_path = os.path.join(VSI_FILESYSTEM_BASE_PATH, str(uuid.uuid4()))\n            # Create vsimem file from buffer.\n            capi.create_vsi_file_from_mem_buffer(\n                force_bytes(vsi_path),\n                byref(self._ds_input),\n                size,\n                VSI_TAKE_BUFFER_OWNERSHIP,\n            )\n            # Open the new vsimem file as a GDALRaster.\n            try:\n                self._ptr = capi.open_ds(force_bytes(vsi_path), self._write)\n            except GDALException:\n                # Remove the broken file from the VSI filesystem.\n                capi.unlink_vsi_file(force_bytes(vsi_path))\n                raise GDALException('Failed creating VSI raster from the input buffer.')\n        elif isinstance(ds_input, dict):\n            # A new raster needs to be created in write mode\n            self._write = 1\n\n            # Create driver (in memory by default)\n            driver = Driver(ds_input.get('driver', 'MEM'))\n\n            # For out of memory drivers, check filename argument\n            if driver.name != 'MEM' and 'name' not in ds_input:\n                raise GDALException('Specify name for creation of raster with driver \"{}\".'.format(driver.name))\n\n            # Check if width and height where specified\n            if 'width' not in ds_input or 'height' not in ds_input:\n                raise GDALException('Specify width and height attributes for JSON or dict input.')\n\n            # Check if srid was specified\n            if 'srid' not in ds_input:\n                raise GDALException('Specify srid for JSON or dict input.')\n\n            # Create null terminated gdal options array.\n            papsz_options = []\n            for key, val in ds_input.get('papsz_options', {}).items():\n                option = '{}={}'.format(key, val)\n                papsz_options.append(option.upper().encode())\n            papsz_options.append(None)\n\n            # Convert papszlist to ctypes array.\n            papsz_options = (c_char_p * len(papsz_options))(*papsz_options)\n\n            # Create GDAL Raster\n            self._ptr = capi.create_ds(\n                driver._ptr,\n                force_bytes(ds_input.get('name', '')),\n                ds_input['width'],\n                ds_input['height'],\n                ds_input.get('nr_of_bands', len(ds_input.get('bands', []))),\n                ds_input.get('datatype', 6),\n                byref(papsz_options),\n            )\n\n            # Set band data if provided\n            for i, band_input in enumerate(ds_input.get('bands', [])):\n                band = self.bands[i]\n                if 'nodata_value' in band_input:\n                    band.nodata_value = band_input['nodata_value']\n                    # Instantiate band filled with nodata values if only\n                    # partial input data has been provided.\n                    if band.nodata_value is not None and (\n                            'data' not in band_input or\n                            'size' in band_input or\n                            'shape' in band_input):\n                        band.data(data=(band.nodata_value,), shape=(1, 1))\n                # Set band data values from input.\n                band.data(\n                    data=band_input.get('data'),\n                    size=band_input.get('size'),\n                    shape=band_input.get('shape'),\n                    offset=band_input.get('offset'),\n                )\n\n            # Set SRID\n            self.srs = ds_input.get('srid')\n\n            # Set additional properties if provided\n            if 'origin' in ds_input:\n                self.origin.x, self.origin.y = ds_input['origin']\n\n            if 'scale' in ds_input:\n                self.scale.x, self.scale.y = ds_input['scale']\n\n            if 'skew' in ds_input:\n                self.skew.x, self.skew.y = ds_input['skew']\n        elif isinstance(ds_input, c_void_p):\n            # Instantiate the object using an existing pointer to a gdal raster.\n            self._ptr = ds_input\n        else:\n            raise GDALException('Invalid data source input type: \"{}\".'.format(type(ds_input)))"
        },
        {
          "file": "django/contrib/gis/gdal/raster/source.py",
          "type": "function",
          "name": "vsi_buffer",
          "class_name": "GDALRaster",
          "code": "def vsi_buffer(self):\n        if not self.is_vsi_based:\n            return None\n        # Prepare an integer that will contain the buffer length.\n        out_length = c_int()\n        # Get the data using the vsi file name.\n        dat = capi.get_mem_buffer_from_vsi_file(\n            force_bytes(self.name),\n            byref(out_length),\n            VSI_DELETE_BUFFER_ON_READ,\n        )\n        # Read the full buffer pointer.\n        return string_at(dat, out_length.value)"
        }
      ]
    }
  ]
}