{
  "Selected_candidate": {
    "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": "When executing a geometry lookup like\nmy_geo_model.objects.filter(my_geom__intersects=wkt_string)\n(i.e. using a WKT string as value)  django.contrib.gis emits the following errors:\n2020-01-13 10:16:07,145 - django.contrib.gis - ERROR - GDAL_ERROR 4: b'POLYGON ((1 1, 1 0, 0 0, 0 1, 1 1)): No such file or directory'\n2020-01-13 10:16:08,403 - django.contrib.gis - ERROR - GDAL_ERROR 10: b\"Pointer 'hObject' is NULL in 'GDALGetDescription'.\\n\"\nThe reason is that if passed a string, the string is first treated as a potential filename which GDAL tries to open and fails (\"no such file\"). Only then it is tried to open the string as WKT/GeoJSON etc. Older Django versions did not emit these errors. However, in commit 6f44f714c9 a check in django/contrib/gis/gdal/raster/source.py:69 whether the file exists was removed.\nSilencing all errors from django.contrib.gis is not really a feasable workaround, as it would mean silcencing interesting GIS errors too.\nAccording to\n​\nhttps://docs.djangoproject.com/en/2.2/ref/contrib/gis/db-api/#geometry-lookups\npassing WKT strings is allowed, but there's no mention of being able to pass a filename as parameter.\nIt's a bit unexpected that Django first tries to open a file before it checks whether the passed string is valid WKT/GeoJSON, and that using WKT results in error messages. Or am I misunderstanding the documentation and calling the API wrongly?\nRelevant parts of example source:\nmodels.py:\n\nclass Area(models.Model):\n    area = models.PolygonField()\n\nsettings.py:\n\nLOGGING = {\n    'version': 1,\n    'handlers': {\n        'console': {\n            'level': 'INFO',\n            'class': 'logging.StreamHandler',\n        },\n    },\n    'root': {\n        'handlers': ['console'],\n        'level': 'INFO',\n    }\n}\n\nmain.py:\n\nfrom app.models import Area\nArea.objects.filter(area__intersects='POLYGON ((1 1, 1 0, 0 0, 0 1, 1 1))')",
    "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)"
      }
    ]
  },
  "Justification": "Candidate E is the most helpful because it addresses a bug related to the handling of geometrical data types in Django, specifically pertaining to how the framework deals with WKT strings in filter queries. While the current bug is about a NotSupportedError when filtering a model with a specific attribute, both reports are connected through the database query system in Django. The fix related to handling WKT strings can provide insights into how attribute properties are evaluated in query filters and could guide adjustments to the current implementation that raises an error when attempting a similar operation with filterable attributes. The shared context of query handling makes this candidate particularly relevant in diagnosing and resolving the current issue.",
  "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"
}