{
  "Selected_candidate": {
    "pr_number": 11393,
    "pr_title": "Fixed #29396, #30494 -- Added indirect values support to __year lookups.",
    "pr_body": "https://code.djangoproject.com/ticket/30494\r\n\r\nThe previous heuristics were naively enabling the `BETWEEN` optimization on successful cast of the first rhs SQL params to an integer while it was not appropriate for a lot of database resolved expressions.\r\n\r\nThanks Alexey Chernov for the report.",
    "issue_id": 29396,
    "issue_title": "The __year lookp crashes with IndexError when passed a non-direct values/expression.",
    "issue_body": "With the following models:\nclass\nModelWithDate\n(\nmodels\n.\nModel\n):\nid\n=\nmodels\n.\nAutoField\n(\nprimary_key\n=\nTrue\n)\ndate\n=\nmodels\n.\nDateField\n()\nclass\nModelWithYear\n(\nmodels\n.\nModel\n):\nid\n=\nmodels\n.\nAutoField\n(\nprimary_key\n=\nTrue\n)\nyear\n=\nmodels\n.\nIntegerField\n()\ndate_ref\n=\nmodels\n.\nForeignKey\n(\nto\n=\nModelWithDate\n,\non_delete\n=\nmodels\n.\nCASCADE\n)\nthe following code:\n>>>\ndates\n=\nModelWithDate\n.\nobjects\n.\nfilter\n(\ndate__year__gte\n=\nOuterRef\n(\n\"year\"\n))\n>>>\ndates_subq\n=\nSubquery\n(\ndates\n.\nvalues\n(\n\"id\"\n))\n>>>\nModelWithYear\n.\nobjects\n.\nfilter\n(\ndate_ref__in\n=\ndates_subq\n)\ncauses an exception:\nTraceback (most recent call last):\n...\nFile\n\"/usr/lib/python3/dist-packages/django/db/models/lookups.py\"\n, line\n90\n, in\nprocess_rhs\nsql\n,\nparams\n=\ncompiler\n.\ncompile\n(\nvalue\n)\nFile\n\"/usr/lib/python3/dist-packages/django/db/models/sql/compiler.py\"\n, line\n391\n, in\ncompile\nsql\n,\nparams\n=\nnode\n.\nas_sql\n(\nself\n,\nself\n.\nconnection\n)\nFile\n\"/usr/lib/python3/dist-packages/django/db/models/expressions.py\"\n, line\n1041\n, in\nas_sql\ntemplate_params\n[\n'subquery'\n],\nsql_params\n=\nself\n.\nqueryset\n.\nquery\n.\nget_compiler\n(\nconnection\n=\nconnection\n)\n.\nas_sql\n()\nFile\n\"/usr/lib/python3/dist-packages/django/db/models/sql/compiler.py\"\n, line\n459\n, in\nas_sql\nwhere\n,\nw_params\n=\nself\n.\ncompile\n(\nself\n.\nwhere\n)\nif\nself\n.\nwhere\nis\nnot\nNone\nelse\n(\n\"\"\n,\n[])\nFile\n\"/usr/lib/python3/dist-packages/django/db/models/sql/compiler.py\"\n, line\n391\n, in\ncompile\nsql\n,\nparams\n=\nnode\n.\nas_sql\n(\nself\n,\nself\n.\nconnection\n)\nFile\n\"/usr/lib/python3/dist-packages/django/db/models/sql/where.py\"\n, line\n80\n, in\nas_sql\nsql\n,\nparams\n=\ncompiler\n.\ncompile\n(\nchild\n)\nFile\n\"/usr/lib/python3/dist-packages/django/db/models/sql/compiler.py\"\n, line\n391\n, in\ncompile\nsql\n,\nparams\n=\nnode\n.\nas_sql\n(\nself\n,\nself\n.\nconnection\n)\nFile\n\"/usr/lib/python3/dist-packages/django/db/models/lookups.py\"\n, line\n523\n, in\nas_sql\nstart\n,\nfinish\n=\nself\n.\nyear_lookup_bounds\n(\nconnection\n,\nrhs_params\n[\n0\n])\nIndexError\n:\nlist index out of range\nThe same happens if I create an annotation using ExtractYear and use it directly.\nA small test project is attached. The bug can be reproduced with “DJANGO_SETTINGS_MODULE=settings python3 ./test.py”. Tested with Django 2.0.5 and 1.11.13.",
    "issue_closed_at": "2019-05-21T00:53:19",
    "base_commit": "1d0bab0bfd77edcf1228d45bf654457a8ff1890d",
    "changes": [
      {
        "file": "django/db/models/lookups.py",
        "type": "function",
        "name": "year_lookup_bounds",
        "class_name": "YearLookup",
        "code": "def year_lookup_bounds(self, connection, year):\n        output_field = self.lhs.lhs.output_field\n        if isinstance(output_field, DateTimeField):\n            bounds = connection.ops.year_lookup_bounds_for_datetime_field(year)\n        else:\n            bounds = connection.ops.year_lookup_bounds_for_date_field(year)\n        return bounds"
      }
    ]
  },
  "Justification": "Candidate B discusses errors related to year lookups in Django, which involves underlying issues with handling year values that could provide essential insights into improving the year interpretation logic of the `parse_http_date`. The fact that it addresses a year-related bug, and the fix is directed at supporting value lookups, can shed light on how to enhance the CURRENT bug's two-digit year handling logic, making it highly relevant for debugging and fixing the new issue concerning RFC compliance.",
  "instance_id": "django__django-11848",
  "repo": "django/django",
  "created_at": "2019-09-28T04:28:22Z",
  "problem_statement": "django.utils.http.parse_http_date two digit year check is incorrect\nDescription\n\t \n\t\t(last modified by Ad Timmering)\n\t \nRFC 850 does not mention this, but in RFC 7231 (and there's something similar in RFC 2822), there's the following quote:\nRecipients of a timestamp value in rfc850-date format, which uses a\ntwo-digit year, MUST interpret a timestamp that appears to be more\nthan 50 years in the future as representing the most recent year in\nthe past that had the same last two digits.\nCurrent logic is hard coded to consider 0-69 to be in 2000-2069, and 70-99 to be 1970-1999, instead of comparing versus the current year.\n",
  "patch": "diff --git a/django/utils/http.py b/django/utils/http.py\n--- a/django/utils/http.py\n+++ b/django/utils/http.py\n@@ -176,10 +176,14 @@ def parse_http_date(date):\n     try:\n         year = int(m.group('year'))\n         if year < 100:\n-            if year < 70:\n-                year += 2000\n+            current_year = datetime.datetime.utcnow().year\n+            current_century = current_year - (current_year % 100)\n+            if year - (current_year % 100) > 50:\n+                # year that appears to be more than 50 years in the future are\n+                # interpreted as representing the past.\n+                year += current_century - 100\n             else:\n-                year += 1900\n+                year += current_century\n         month = MONTHS.index(m.group('mon').lower()) + 1\n         day = int(m.group('day'))\n         hour = int(m.group('hour'))\n"
}