{
  "Selected_candidate": {
    "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": "I wish to have ability to write something like this:\nfrom django.utils.functional import lazy\n\nobj, created = model.objects.get_or_create(\n     key=jwt,\n     defaults=lazy(self.get_defaults_for_model, dict)(jwt) \n)\nBut at the moment\n_extract_model_params\nprepare defaults before it's realy needed.\nI think\n_extract_model_params\nshould be separated to\n_prepare_model_lookup\nand\n_prepare_model_params\n.\nSo it will be possible to call\n_prepare_model_params\nwhen\nmodel.DoesNotExist\nor even inside\n_create_object_from_params\n.",
    "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"
      }
    ]
  },
  "Justification": "Candidate A is the most helpful because it involves the same module (`django/db/models/query.py`) as the CURRENT bug report, which pertains to how query filters are resolved. Both bugs deal with errors arising during the processing of arguments within the Django ORM, particularly with how tuples or lazy defaults are handled during query construction. The structural similarity in how defaults and lookups are processed provides direct insights into resolving the construction error in named tuples outlined in the CURRENT bug. Additionally, both bugs address underlying issues related to argument handling in Django model queries, making this candidate's fix particularly relevant for debugging the CURRENT bug.",
  "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"
}