{
  "Selected_candidate": {
    "pr_number": 7097,
    "pr_title": "Fixed #27068 -- Unified form field initial data retrieval.",
    "pr_body": "https://code.djangoproject.com/ticket/27068\n",
    "issue_id": 27068,
    "issue_title": "Acquire form's initial data more consistently",
    "issue_body": "In\nBoundField\n, sometimes initial data\n​\nhandles callables\n. While other times,\n​\nit doesn't\n. This can lead to a theoretical bug when the initial data is a callable, but the field is disabled.\nOther times, the initial callable is handled, but microseconds are\n​\nnot chopped\nleading to theoretical false positives that data has changed.\nInitial data should be handled more consistently across all cases.\nMarking as a bug due to the theoretical edge cases that can produce unexpected results. Noticed these inconsistencies while working on\n#27037\n.",
    "issue_closed_at": "2016-08-18T19:51:32",
    "base_commit": "13857b45ca54a519a58361238442af84262c0d23",
    "changes": [
      {
        "file": "django/forms/boundfield.py",
        "type": "function",
        "name": "value",
        "class_name": "BoundField",
        "code": "def value(self):\n        \"\"\"\n        Returns the value for this BoundField, using the initial value if\n        the form is not bound or the data otherwise.\n        \"\"\"\n        if not self.form.is_bound:\n            data = self.initial\n        else:\n            data = self.field.bound_data(\n                self.data, self.form.initial.get(self.name, self.field.initial)\n            )\n        return self.field.prepare_value(data)"
      },
      {
        "file": "django/forms/boundfield.py",
        "type": "function",
        "name": "id_for_label",
        "class_name": "BoundField",
        "code": "def id_for_label(self):\n        \"\"\"\n        Wrapper around the field widget's `id_for_label` method.\n        Useful, for example, for focusing on this field regardless of whether\n        it has a single widget or a MultiWidget.\n        \"\"\"\n        widget = self.field.widget\n        id_ = widget.attrs.get('id') or self.auto_id\n        return widget.id_for_label(id_)"
      },
      {
        "file": "django/forms/forms.py",
        "type": "function",
        "name": "_clean_fields",
        "class_name": "BaseForm",
        "code": "def _clean_fields(self):\n        for name, field in self.fields.items():\n            # value_from_datadict() gets the data from the data dictionaries.\n            # Each widget type knows how to retrieve its own data, because some\n            # widgets split data over several HTML fields.\n            if field.disabled:\n                value = self.initial.get(name, field.initial)\n            else:\n                value = field.widget.value_from_datadict(self.data, self.files, self.add_prefix(name))\n            try:\n                if isinstance(field, FileField):\n                    initial = self.initial.get(name, field.initial)\n                    value = field.clean(value, initial)\n                else:\n                    value = field.clean(value)\n                self.cleaned_data[name] = value\n                if hasattr(self, 'clean_%s' % name):\n                    value = getattr(self, 'clean_%s' % name)()\n                    self.cleaned_data[name] = value\n            except ValidationError as e:\n                self.add_error(name, e)"
      },
      {
        "file": "django/forms/forms.py",
        "type": "function",
        "name": "changed_data",
        "class_name": "BaseForm",
        "code": "def changed_data(self):\n        data = []\n        for name, field in self.fields.items():\n            prefixed_name = self.add_prefix(name)\n            data_value = field.widget.value_from_datadict(self.data, self.files, prefixed_name)\n            if not field.show_hidden_initial:\n                initial_value = self.initial.get(name, field.initial)\n                if callable(initial_value):\n                    initial_value = initial_value()\n            else:\n                initial_prefixed_name = self.add_initial_prefix(name)\n                hidden_widget = field.hidden_widget()\n                try:\n                    initial_value = field.to_python(hidden_widget.value_from_datadict(\n                        self.data, self.files, initial_prefixed_name))\n                except ValidationError:\n                    # Always assume data has changed if validation fails.\n                    data.append(name)\n                    continue\n            if field.has_changed(initial_value, data_value):\n                data.append(name)\n        return data"
      },
      {
        "file": "django/forms/forms.py",
        "type": "function",
        "name": "visible_fields",
        "class_name": "BaseForm",
        "code": "def visible_fields(self):\n        \"\"\"\n        Returns a list of BoundField objects that aren't hidden fields.\n        The opposite of the hidden_fields() method.\n        \"\"\"\n        return [field for field in self if not field.is_hidden]"
      }
    ]
  },
  "Justification": "Candidate A is the most relevant as it directly addresses issues related to handling initial data in form fields, specifically with callable defaults. This aligns closely with the CURRENT bug, where the correct propagation of default values in the ArrayField is not occurring during form submissions in the admin. The structural similarity in dealing with initial values, as well as the fact that both bug reports involve admin forms, makes this candidate particularly useful in identifying potential solutions to the CURRENT bug.",
  "instance_id": "django__django-16229",
  "repo": "django/django",
  "created_at": "2022-10-26T11:42:55Z",
  "problem_statement": "ModelForm fields with callable defaults don't correctly propagate default values\nDescription\n\t\nWhen creating an object via the admin, if an inline contains an ArrayField in error, the validation will be bypassed (and the inline dismissed) if we submit the form a second time (without modification).\ngo to /admin/my_app/thing/add/\ntype anything in plop\nsubmit -> it shows an error on the inline\nsubmit again -> no errors, plop become unfilled\n# models.py\nclass Thing(models.Model):\n\tpass\nclass RelatedModel(models.Model):\n\tthing = models.ForeignKey(Thing, on_delete=models.CASCADE)\n\tplop = ArrayField(\n\t\tmodels.CharField(max_length=42),\n\t\tdefault=list,\n\t)\n# admin.py\nclass RelatedModelForm(forms.ModelForm):\n\tdef clean(self):\n\t\traise ValidationError(\"whatever\")\nclass RelatedModelInline(admin.TabularInline):\n\tform = RelatedModelForm\n\tmodel = RelatedModel\n\textra = 1\n@admin.register(Thing)\nclass ThingAdmin(admin.ModelAdmin):\n\tinlines = [\n\t\tRelatedModelInline\n\t]\nIt seems related to the hidden input containing the initial value:\n<input type=\"hidden\" name=\"initial-relatedmodel_set-0-plop\" value=\"test\" id=\"initial-relatedmodel_set-0-id_relatedmodel_set-0-plop\">\nI can fix the issue locally by forcing show_hidden_initial=False on the field (in the form init)\n",
  "patch": "diff --git a/django/forms/boundfield.py b/django/forms/boundfield.py\n--- a/django/forms/boundfield.py\n+++ b/django/forms/boundfield.py\n@@ -96,9 +96,17 @@ def as_widget(self, widget=None, attrs=None, only_initial=False):\n             attrs.setdefault(\n                 \"id\", self.html_initial_id if only_initial else self.auto_id\n             )\n+        if only_initial and self.html_initial_name in self.form.data:\n+            # Propagate the hidden initial value.\n+            value = self.form._widget_data_value(\n+                self.field.hidden_widget(),\n+                self.html_initial_name,\n+            )\n+        else:\n+            value = self.value()\n         return widget.render(\n             name=self.html_initial_name if only_initial else self.html_name,\n-            value=self.value(),\n+            value=value,\n             attrs=attrs,\n             renderer=self.form.renderer,\n         )\n"
}