{
  "instance_id": "django__django-13590",
  "repo": "django/django",
  "created_at": "2020-10-23T09:34:55Z",
  "problem_statement": "Upgrading 2.2>3.0 causes named tuples used as arguments to __range to error.\nDescription\n\t\nI noticed this while upgrading a project from 2.2 to 3.0.\nThis project passes named 2-tuples as arguments to range queryset filters. This works fine on 2.2. On 3.0 it causes the following error: TypeError: __new__() missing 1 required positional argument: 'far'.\nThis happens because django.db.models.sql.query.Query.resolve_lookup_value goes into the tuple elements to resolve lookups and then attempts to reconstitute the tuple with the resolved elements.\nWhen it attempts to construct the new tuple it preserves the type (the named tuple) but it passes a iterator to it's constructor.\nNamedTuples don't have the code path for copying an iterator, and so it errors on insufficient arguments.\nThe fix is to * expand the contents of the iterator into the constructor.\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@@ -1077,10 +1077,14 @@ def resolve_lookup_value(self, value, can_reuse, allow_joins):\n         elif isinstance(value, (list, tuple)):\n             # The items of the iterable may be expressions and therefore need\n             # to be resolved independently.\n-            return type(value)(\n+            values = (\n                 self.resolve_lookup_value(sub_value, can_reuse, allow_joins)\n                 for sub_value in value\n             )\n+            type_ = type(value)\n+            if hasattr(type_, '_make'):  # namedtuple\n+                return type_(*values)\n+            return type_(values)\n         return value\n \n     def solve_lookup_type(self, lookup):\n",
  "similar_bug_items": [
    {
      "pr_number": 9964,
      "pr_title": "Fixed #29413 -- Prevented evaluation of QuerySet.get_or_create()/update_or_create() defaults unless needed.",
      "pr_body": "https://code.djangoproject.com/ticket/29413",
      "issue_id": 29413,
      "issue_title": "QuerySet.get_or_create()/update_or_create() shouldn't evaluate lazy defaults unless they are needed",
      "issue_body": "",
      "issue_closed_at": "2018-07-16T21:09:12",
      "base_commit": "4d48ddd8f93800a80330ec1dee7b7d4afe6ea95a",
      "changes": [
        {
          "file": "django/db/models/query.py",
          "type": "function",
          "name": "get_or_create",
          "class_name": "QuerySet",
          "code": "def get_or_create(self, defaults=None, **kwargs):\n        \"\"\"\n        Look up an object with the given kwargs, creating one if necessary.\n        Return a tuple of (object, created), where created is a boolean\n        specifying whether an object was created.\n        \"\"\"\n        lookup, params = self._extract_model_params(defaults, **kwargs)\n        # The get() needs to be targeted at the write database in order\n        # to avoid potential transaction consistency problems.\n        self._for_write = True\n        try:\n            return self.get(**lookup), False\n        except self.model.DoesNotExist:\n            return self._create_object_from_params(lookup, params)"
        },
        {
          "file": "django/db/models/query.py",
          "type": "function",
          "name": "update_or_create",
          "class_name": "QuerySet",
          "code": "def update_or_create(self, defaults=None, **kwargs):\n        \"\"\"\n        Look up an object with the given kwargs, updating one with defaults\n        if it exists, otherwise create a new one.\n        Return a tuple (object, created), where created is a boolean\n        specifying whether an object was created.\n        \"\"\"\n        defaults = defaults or {}\n        lookup, params = self._extract_model_params(defaults, **kwargs)\n        self._for_write = True\n        with transaction.atomic(using=self.db):\n            try:\n                obj = self.select_for_update().get(**lookup)\n            except self.model.DoesNotExist:\n                obj, created = self._create_object_from_params(lookup, params)\n                if created:\n                    return obj, created\n            for k, v in defaults.items():\n                setattr(obj, k, v() if callable(v) else v)\n            obj.save(using=self.db)\n        return obj, False"
        },
        {
          "file": "django/db/models/query.py",
          "type": "function",
          "name": "_create_object_from_params",
          "class_name": "QuerySet",
          "code": "def _create_object_from_params(self, lookup, params):\n        \"\"\"\n        Try to create an object using passed params. Used by get_or_create()\n        and update_or_create().\n        \"\"\"\n        try:\n            with transaction.atomic(using=self.db):\n                params = {k: v() if callable(v) else v for k, v in params.items()}\n                obj = self.create(**params)\n            return obj, True\n        except IntegrityError as e:\n            try:\n                return self.get(**lookup), False\n            except self.model.DoesNotExist:\n                pass\n            raise e"
        },
        {
          "file": "django/db/models/query.py",
          "type": "function",
          "name": "_extract_model_params",
          "class_name": "QuerySet",
          "code": "def _extract_model_params(self, defaults, **kwargs):\n        \"\"\"\n        Prepare `lookup` (kwargs that are valid model attributes), `params`\n        (for creating a model instance) based on given kwargs; for use by\n        get_or_create() and update_or_create().\n        \"\"\"\n        defaults = defaults or {}\n        lookup = kwargs.copy()\n        for f in self.model._meta.fields:\n            if f.attname in lookup:\n                lookup[f.name] = lookup.pop(f.attname)\n        params = {k: v for k, v in kwargs.items() if LOOKUP_SEP not in k}\n        params.update(defaults)\n        property_names = self.model._meta._property_names\n        invalid_params = []\n        for param in params:\n            try:\n                self.model._meta.get_field(param)\n            except exceptions.FieldDoesNotExist:\n                # It's okay to use a model's property if it has a setter.\n                if not (param in property_names and getattr(self.model, param).fset):\n                    invalid_params.append(param)\n        if invalid_params:\n            raise exceptions.FieldError(\n                \"Invalid field name(s) for model %s: '%s'.\" % (\n                    self.model._meta.object_name,\n                    \"', '\".join(sorted(invalid_params)),\n                ))\n        return lookup, params"
        }
      ]
    },
    {
      "pr_number": 9396,
      "pr_title": "[1.11.x] Fixed #28856 -- Fixed an issue with caching of coerced gfk pointing to mti model.",
      "pr_body": "https://code.djangoproject.com/ticket/28856\r\n\r\nSee #9395 for the patch against master.",
      "issue_id": 28856,
      "issue_title": "GenericForeignKey attributes create new instances on every access",
      "issue_body": "",
      "issue_closed_at": "2017-12-08T13:15:18",
      "base_commit": "3545e844885608932a692d952c12cd863e2320b5",
      "changes": [
        {
          "file": "django/contrib/contenttypes/fields.py",
          "type": "function",
          "name": "__get__",
          "class_name": "GenericForeignKey",
          "code": "def __get__(self, instance, cls=None):\n        if instance is None:\n            return self\n\n        # Don't use getattr(instance, self.ct_field) here because that might\n        # reload the same ContentType over and over (#5570). Instead, get the\n        # content type ID here, and later when the actual instance is needed,\n        # use ContentType.objects.get_for_id(), which has a global cache.\n        f = self.model._meta.get_field(self.ct_field)\n        ct_id = getattr(instance, f.get_attname(), None)\n        pk_val = getattr(instance, self.fk_field)\n\n        try:\n            rel_obj = getattr(instance, self.cache_attr)\n        except AttributeError:\n            rel_obj = None\n        else:\n            if rel_obj and (ct_id != self.get_content_type(obj=rel_obj, using=instance._state.db).id or\n                            rel_obj._meta.pk.to_python(pk_val) != rel_obj._get_pk_val()):\n                rel_obj = None\n\n        if rel_obj is not None:\n            return rel_obj\n\n        if ct_id is not None:\n            ct = self.get_content_type(id=ct_id, using=instance._state.db)\n            try:\n                rel_obj = ct.get_object_for_this_type(pk=pk_val)\n            except ObjectDoesNotExist:\n                pass\n        setattr(instance, self.cache_attr, rel_obj)\n        return rel_obj"
        }
      ]
    },
    {
      "pr_number": 10027,
      "pr_title": "Fixed #24384 -- Allowed compilemessages to continue running after nonfatal errors.",
      "pr_body": "",
      "issue_id": 24384,
      "issue_title": "compilemessages shouldn't return with exit code 0 when it fails",
      "issue_body": "",
      "issue_closed_at": "2018-06-13T14:27:08",
      "base_commit": "b30f9b131c9489b9d9f21c311ecb46d0aea91381",
      "changes": [
        {
          "file": "django/core/management/commands/compilemessages.py",
          "type": "function",
          "name": "handle",
          "class_name": "Command",
          "code": "def handle(self, **options):\n        locale = options['locale']\n        exclude = options['exclude']\n        self.verbosity = options['verbosity']\n        if options['fuzzy']:\n            self.program_options = self.program_options + ['-f']\n\n        if find_command(self.program) is None:\n            raise CommandError(\"Can't find %s. Make sure you have GNU gettext \"\n                               \"tools 0.15 or newer installed.\" % self.program)\n\n        basedirs = [os.path.join('conf', 'locale'), 'locale']\n        if os.environ.get('DJANGO_SETTINGS_MODULE'):\n            from django.conf import settings\n            basedirs.extend(settings.LOCALE_PATHS)\n\n        # Walk entire tree, looking for locale directories\n        for dirpath, dirnames, filenames in os.walk('.', topdown=True):\n            for dirname in dirnames:\n                if dirname == 'locale':\n                    basedirs.append(os.path.join(dirpath, dirname))\n\n        # Gather existing directories.\n        basedirs = set(map(os.path.abspath, filter(os.path.isdir, basedirs)))\n\n        if not basedirs:\n            raise CommandError(\"This script should be run from the Django Git \"\n                               \"checkout or your project or app tree, or with \"\n                               \"the settings module specified.\")\n\n        # Build locale list\n        all_locales = []\n        for basedir in basedirs:\n            locale_dirs = filter(os.path.isdir, glob.glob('%s/*' % basedir))\n            all_locales.extend(map(os.path.basename, locale_dirs))\n\n        # Account for excluded locales\n        locales = locale or all_locales\n        locales = set(locales).difference(exclude)\n\n        for basedir in basedirs:\n            if locales:\n                dirs = [os.path.join(basedir, l, 'LC_MESSAGES') for l in locales]\n            else:\n                dirs = [basedir]\n            locations = []\n            for ldir in dirs:\n                for dirpath, dirnames, filenames in os.walk(ldir):\n                    locations.extend((dirpath, f) for f in filenames if f.endswith('.po'))\n            if locations:\n                self.compile_messages(locations)"
        },
        {
          "file": "django/core/management/commands/compilemessages.py",
          "type": "function",
          "name": "handle",
          "class_name": "Command",
          "code": "def handle(self, **options):\n        locale = options['locale']\n        exclude = options['exclude']\n        self.verbosity = options['verbosity']\n        if options['fuzzy']:\n            self.program_options = self.program_options + ['-f']\n\n        if find_command(self.program) is None:\n            raise CommandError(\"Can't find %s. Make sure you have GNU gettext \"\n                               \"tools 0.15 or newer installed.\" % self.program)\n\n        basedirs = [os.path.join('conf', 'locale'), 'locale']\n        if os.environ.get('DJANGO_SETTINGS_MODULE'):\n            from django.conf import settings\n            basedirs.extend(settings.LOCALE_PATHS)\n\n        # Walk entire tree, looking for locale directories\n        for dirpath, dirnames, filenames in os.walk('.', topdown=True):\n            for dirname in dirnames:\n                if dirname == 'locale':\n                    basedirs.append(os.path.join(dirpath, dirname))\n\n        # Gather existing directories.\n        basedirs = set(map(os.path.abspath, filter(os.path.isdir, basedirs)))\n\n        if not basedirs:\n            raise CommandError(\"This script should be run from the Django Git \"\n                               \"checkout or your project or app tree, or with \"\n                               \"the settings module specified.\")\n\n        # Build locale list\n        all_locales = []\n        for basedir in basedirs:\n            locale_dirs = filter(os.path.isdir, glob.glob('%s/*' % basedir))\n            all_locales.extend(map(os.path.basename, locale_dirs))\n\n        # Account for excluded locales\n        locales = locale or all_locales\n        locales = set(locales).difference(exclude)\n\n        for basedir in basedirs:\n            if locales:\n                dirs = [os.path.join(basedir, l, 'LC_MESSAGES') for l in locales]\n            else:\n                dirs = [basedir]\n            locations = []\n            for ldir in dirs:\n                for dirpath, dirnames, filenames in os.walk(ldir):\n                    locations.extend((dirpath, f) for f in filenames if f.endswith('.po'))\n            if locations:\n                self.compile_messages(locations)"
        },
        {
          "file": "django/core/management/commands/compilemessages.py",
          "type": "function",
          "name": "compile_messages",
          "class_name": "Command",
          "code": "def compile_messages(self, locations):\n        \"\"\"\n        Locations is a list of tuples: [(directory, file), ...]\n        \"\"\"\n        for i, (dirpath, f) in enumerate(locations):\n            if self.verbosity > 0:\n                self.stdout.write('processing file %s in %s\\n' % (f, dirpath))\n            po_path = os.path.join(dirpath, f)\n            if has_bom(po_path):\n                raise CommandError(\"The %s file has a BOM (Byte Order Mark). \"\n                                   \"Django only supports .po files encoded in \"\n                                   \"UTF-8 and without any BOM.\" % po_path)\n            base_path = os.path.splitext(po_path)[0]\n\n            # Check writability on first location\n            if i == 0 and not is_writable(base_path + '.mo'):\n                self.stderr.write(\"The po files under %s are in a seemingly not writable location. \"\n                                  \"mo files will not be updated/created.\" % dirpath)\n                return\n\n            args = [self.program] + self.program_options + [\n                '-o', base_path + '.mo', base_path + '.po'\n            ]\n            output, errors, status = popen_wrapper(args)\n            if status:\n                if errors:\n                    msg = \"Execution of %s failed: %s\" % (self.program, errors)\n                else:\n                    msg = \"Execution of %s failed\" % self.program\n                raise CommandError(msg)"
        },
        {
          "file": "django/core/management/commands/compilemessages.py",
          "type": "function",
          "name": "compile_messages",
          "class_name": "Command",
          "code": "def compile_messages(self, locations):\n        \"\"\"\n        Locations is a list of tuples: [(directory, file), ...]\n        \"\"\"\n        for i, (dirpath, f) in enumerate(locations):\n            if self.verbosity > 0:\n                self.stdout.write('processing file %s in %s\\n' % (f, dirpath))\n            po_path = os.path.join(dirpath, f)\n            if has_bom(po_path):\n                raise CommandError(\"The %s file has a BOM (Byte Order Mark). \"\n                                   \"Django only supports .po files encoded in \"\n                                   \"UTF-8 and without any BOM.\" % po_path)\n            base_path = os.path.splitext(po_path)[0]\n\n            # Check writability on first location\n            if i == 0 and not is_writable(base_path + '.mo'):\n                self.stderr.write(\"The po files under %s are in a seemingly not writable location. \"\n                                  \"mo files will not be updated/created.\" % dirpath)\n                return\n\n            args = [self.program] + self.program_options + [\n                '-o', base_path + '.mo', base_path + '.po'\n            ]\n            output, errors, status = popen_wrapper(args)\n            if status:\n                if errors:\n                    msg = \"Execution of %s failed: %s\" % (self.program, errors)\n                else:\n                    msg = \"Execution of %s failed\" % self.program\n                raise CommandError(msg)"
        }
      ]
    },
    {
      "pr_number": 9861,
      "pr_title": "Fixed #29296 -- Fix handling of callable object views in admindocs",
      "pr_body": "See https://code.djangoproject.com/ticket/29296\r\n\r\nI'm not sure how you normally handle backports, but a backported version of this commit can be found here:\r\nhttps://github.com/PaulSD/django/commit/b2128a2afa14c64b39a5fae1f9a63b2959c4f752",
      "issue_id": 29296,
      "issue_title": "admindocs ViewIndexView crashes if a syndication Feed (or something without __qualname__) is configured",
      "issue_body": "",
      "issue_closed_at": "2018-04-12T12:34:31",
      "base_commit": "ee17bb8a67a9e7e688da6e6f4b3be1b3a69c09b0",
      "changes": [
        {
          "file": "django/contrib/admindocs/middleware.py",
          "type": "line",
          "name": "line 2",
          "code": "from django.http import HttpResponse\nfrom django.utils.deprecation import MiddlewareMixin\n\n\nclass XViewMiddleware(MiddlewareMixin):\n    \"\"\""
        },
        {
          "file": "django/contrib/admindocs/middleware.py",
          "type": "function",
          "name": "process_view",
          "class_name": "XViewMiddleware",
          "code": "def process_view(self, request, view_func, view_args, view_kwargs):\n        \"\"\"\n        If the request method is HEAD and either the IP is internal or the\n        user is a logged-in staff member, return a responsewith an x-view\n        header indicating the view function. This is used to lookup the view\n        function for an arbitrary page.\n        \"\"\"\n        assert hasattr(request, 'user'), (\n            \"The XView middleware requires authentication middleware to be \"\n            \"installed. Edit your MIDDLEWARE%s setting to insert \"\n            \"'django.contrib.auth.middleware.AuthenticationMiddleware'.\" % (\n                \"_CLASSES\" if settings.MIDDLEWARE is None else \"\"\n            )\n        )\n        if request.method == 'HEAD' and (request.META.get('REMOTE_ADDR') in settings.INTERNAL_IPS or\n                                         (request.user.is_active and request.user.is_staff)):\n            response = HttpResponse()\n            response['X-View'] = \"%s.%s\" % (view_func.__module__, view_func.__name__)\n            return response"
        },
        {
          "file": "django/contrib/admindocs/utils.py",
          "type": "line",
          "name": "line 18",
          "code": "    docutils_is_available = True\n\n\ndef trim_docstring(docstring):\n    \"\"\"\n    Uniformly trim leading/trailing whitespace from docstrings."
        },
        {
          "file": "django/contrib/admindocs/views.py",
          "type": "line",
          "name": "line 23",
          "code": "from django.utils.translation import gettext as _\nfrom django.views.generic import TemplateView\n\n# Exclude methods starting with these strings from documentation\nMODEL_METHODS_EXCLUDE = ('_', 'add_', 'delete', 'save', 'set_')\n"
        },
        {
          "file": "django/contrib/admindocs/views.py",
          "type": "function",
          "name": "get_context_data",
          "class_name": "TemplateDetailView",
          "code": "def get_context_data(self, **kwargs):\n        template = self.kwargs['template']\n        templates = []\n        try:\n            default_engine = Engine.get_default()\n        except ImproperlyConfigured:\n            # Non-trivial TEMPLATES settings aren't supported (#24125).\n            pass\n        else:\n            # This doesn't account for template loaders (#24128).\n            for index, directory in enumerate(default_engine.dirs):\n                template_file = os.path.join(directory, template)\n                if os.path.exists(template_file):\n                    with open(template_file) as f:\n                        template_contents = f.read()\n                else:\n                    template_contents = ''\n                templates.append({\n                    'file': template_file,\n                    'exists': os.path.exists(template_file),\n                    'contents': template_contents,\n                    'order': index,\n                })\n        return super().get_context_data(**{\n            **kwargs,\n            'name': template,\n            'templates': templates,\n        })"
        }
      ]
    },
    {
      "pr_number": 13251,
      "pr_title": "Fixed #31783 -- Fixed crash when filtering againts \"negate\" field.",
      "pr_body": "https://code.djangoproject.com/ticket/31783\r\nI added a regression test for the @charettes patch.",
      "issue_id": 31783,
      "issue_title": "Filtering on a field named `negate` raises a TypeError",
      "issue_body": "",
      "issue_closed_at": "2020-07-29T02:03:33",
      "base_commit": "b6dfdaff33f19757b1cb9b3bf1d17f28b94859d4",
      "changes": [
        {
          "file": "django/db/models/query.py",
          "type": "function",
          "name": "__init__",
          "class_name": "RelatedPopulator",
          "code": "def __init__(self, klass_info, select, db):\n        self.db = db\n        # Pre-compute needed attributes. The attributes are:\n        #  - model_cls: the possibly deferred model class to instantiate\n        #  - either:\n        #    - cols_start, cols_end: usually the columns in the row are\n        #      in the same order model_cls.__init__ expects them, so we\n        #      can instantiate by model_cls(*row[cols_start:cols_end])\n        #    - reorder_for_init: When select_related descends to a child\n        #      class, then we want to reuse the already selected parent\n        #      data. However, in this case the parent data isn't necessarily\n        #      in the same order that Model.__init__ expects it to be, so\n        #      we have to reorder the parent data. The reorder_for_init\n        #      attribute contains a function used to reorder the field data\n        #      in the order __init__ expects it.\n        #  - pk_idx: the index of the primary key field in the reordered\n        #    model data. Used to check if a related object exists at all.\n        #  - init_list: the field attnames fetched from the database. For\n        #    deferred models this isn't the same as all attnames of the\n        #    model's fields.\n        #  - related_populators: a list of RelatedPopulator instances if\n        #    select_related() descends to related models from this model.\n        #  - local_setter, remote_setter: Methods to set cached values on\n        #    the object being populated and on the remote object. Usually\n        #    these are Field.set_cached_value() methods.\n        select_fields = klass_info['select_fields']\n        from_parent = klass_info['from_parent']\n        if not from_parent:\n            self.cols_start = select_fields[0]\n            self.cols_end = select_fields[-1] + 1\n            self.init_list = [\n                f[0].target.attname for f in select[self.cols_start:self.cols_end]\n            ]\n            self.reorder_for_init = None\n        else:\n            attname_indexes = {select[idx][0].target.attname: idx for idx in select_fields}\n            model_init_attnames = (f.attname for f in klass_info['model']._meta.concrete_fields)\n            self.init_list = [attname for attname in model_init_attnames if attname in attname_indexes]\n            self.reorder_for_init = operator.itemgetter(*[attname_indexes[attname] for attname in self.init_list])\n\n        self.model_cls = klass_info['model']\n        self.pk_idx = self.init_list.index(self.model_cls._meta.pk.attname)\n        self.related_populators = get_related_populators(klass_info, select, self.db)\n        self.local_setter = klass_info['local_setter']\n        self.remote_setter = klass_info['remote_setter']"
        },
        {
          "file": "django/db/models/query.py",
          "type": "function",
          "name": "filter",
          "class_name": "QuerySet",
          "code": "def filter(self, *args, **kwargs):\n        \"\"\"\n        Return a new QuerySet instance with the args ANDed to the existing\n        set.\n        \"\"\"\n        self._not_support_combined_queries('filter')\n        return self._filter_or_exclude(False, *args, **kwargs)"
        },
        {
          "file": "django/db/models/query.py",
          "type": "function",
          "name": "exclude",
          "class_name": "QuerySet",
          "code": "def exclude(self, *args, **kwargs):\n        \"\"\"\n        Return a new QuerySet instance with NOT (args) ANDed to the existing\n        set.\n        \"\"\"\n        self._not_support_combined_queries('exclude')\n        return self._filter_or_exclude(True, *args, **kwargs)"
        },
        {
          "file": "django/db/models/query.py",
          "type": "function",
          "name": "_filter_or_exclude",
          "class_name": "QuerySet",
          "code": "def _filter_or_exclude(self, negate, *args, **kwargs):\n        if args or kwargs:\n            assert not self.query.is_sliced, \\\n                \"Cannot filter a query once a slice has been taken.\"\n\n        clone = self._chain()\n        if self._defer_next_filter:\n            self._defer_next_filter = False\n            clone._deferred_filter = negate, args, kwargs\n        else:\n            clone._filter_or_exclude_inplace(negate, *args, **kwargs)\n        return clone"
        },
        {
          "file": "django/db/models/query.py",
          "type": "function",
          "name": "complex_filter",
          "class_name": "QuerySet",
          "code": "def complex_filter(self, filter_obj):\n        \"\"\"\n        Return a new QuerySet instance with filter_obj added to the filters.\n\n        filter_obj can be a Q object or a dictionary of keyword lookup\n        arguments.\n\n        This exists to support framework features such as 'limit_choices_to',\n        and usually it will be more natural to use other methods.\n        \"\"\"\n        if isinstance(filter_obj, Q):\n            clone = self._chain()\n            clone.query.add_q(filter_obj)\n            return clone\n        else:\n            return self._filter_or_exclude(False, **filter_obj)"
        }
      ]
    }
  ]
}