{
  "instance_id": "django__django-14534",
  "repo": "django/django",
  "created_at": "2021-06-17T15:37:34Z",
  "problem_statement": "BoundWidget.id_for_label ignores id set by ChoiceWidget.options\nDescription\n\t\nIf you look at the implementation of BoundField.subwidgets\nclass BoundField:\n\t...\n\tdef subwidgets(self):\n\t\tid_ = self.field.widget.attrs.get('id') or self.auto_id\n\t\tattrs = {'id': id_} if id_ else {}\n\t\tattrs = self.build_widget_attrs(attrs)\n\t\treturn [\n\t\t\tBoundWidget(self.field.widget, widget, self.form.renderer)\n\t\t\tfor widget in self.field.widget.subwidgets(self.html_name, self.value(), attrs=attrs)\n\t\t]\none sees that self.field.widget.subwidgets(self.html_name, self.value(), attrs=attrs) returns a dict and assigns it to widget. Now widget['attrs']['id'] contains the \"id\" we would like to use when rendering the label of our CheckboxSelectMultiple.\nHowever BoundWidget.id_for_label() is implemented as\nclass BoundWidget:\n\t...\n\tdef id_for_label(self):\n\t\treturn 'id_%s_%s' % (self.data['name'], self.data['index'])\nignoring the id available through self.data['attrs']['id']. This re-implementation for rendering the \"id\" is confusing and presumably not intended. Nobody has probably realized that so far, because rarely the auto_id-argument is overridden when initializing a form. If however we do, one would assume that the method BoundWidget.id_for_label renders that string as specified through the auto_id format-string.\nBy changing the code from above to\nclass BoundWidget:\n\t...\n\tdef id_for_label(self):\n\t\treturn self.data['attrs']['id']\nthat function behaves as expected.\nPlease note that this error only occurs when rendering the subwidgets of a widget of type CheckboxSelectMultiple. This has nothing to do with the method BoundField.id_for_label().\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@@ -277,7 +277,7 @@ def template_name(self):\n \n     @property\n     def id_for_label(self):\n-        return 'id_%s_%s' % (self.data['name'], self.data['index'])\n+        return self.data['attrs'].get('id')\n \n     @property\n     def choice_label(self):\n",
  "similar_bug_items": [
    {
      "pr_number": 8362,
      "pr_title": "Fixed #28058 -- Restored empty BoundFields evaluating to True.",
      "pr_body": "https://code.djangoproject.com/ticket/28058",
      "issue_id": 28058,
      "issue_title": "Empty Select widget (no choices) evaluates to False",
      "issue_body": "Hey guys,\nI'm not sure I catched a bug, but I discovered a change between\nDjango==1.10.6\nand 1.11.\nWhen you initialize a form with a\nSelect\nwidget, and the select widget has no choices (basically an empty tuple), when evaluating the field with\nbool()\n, it will evaluate to\nFalse\n. This causes the\ndjango_widget_tweaks\nmodule not to render a widget when trying to modify it from within a template, as it uses a bool-like evaluation to check if anything is passed to the set_attr.\ncode to reproduce:\nfrom django import forms\n\nclass MyForm(forms.Form):\n    select = forms.ChoiceField(choices=())\n\nx = MyForm()\n\nfor item in x:\n    print(bool(item))\nIt will print\nTrue\nwith\nDjango==1.10.6\n, and\nFalse\nwith\nDjango==1.11\n.\nThe underlying problem is that\nlen()\nfor a Boundfield with no choices will return 0 in the newer version, whereas it returns 1 in the older version.\nthe template code that fails to render with\nDjango==1.11\n{{ widget|set_attr('style:width: 100%') }}\nLike I said, I'm not sure if this change is intended, nevertheless I reported it. Please let me know if this is intended, so I can let the creator of\ndjango_widget_tweaks\nknow.\nCheers",
      "issue_closed_at": "2017-04-17T07:49:12",
      "base_commit": "e5dce7b0fbd2965e524ce97114f501319ec2bb6f",
      "changes": [
        {
          "file": "django/forms/boundfield.py",
          "type": "function",
          "name": "subwidgets",
          "class_name": "BoundField",
          "code": "def subwidgets(self):\n        \"\"\"\n        Most widgets yield a single subwidget, but others like RadioSelect and\n        CheckboxSelectMultiple produce one subwidget for each choice.\n\n        This property is cached so that only one database query occurs when\n        rendering ModelChoiceFields.\n        \"\"\"\n        id_ = self.field.widget.attrs.get('id') or self.auto_id\n        attrs = {'id': id_} if id_ else {}\n        attrs = self.build_widget_attrs(attrs)\n        return list(\n            BoundWidget(self.field.widget, widget, self.form.renderer)\n            for widget in self.field.widget.subwidgets(self.html_name, self.value(), attrs=attrs)\n        )"
        }
      ]
    },
    {
      "pr_number": 3503,
      "pr_title": "Fixed #23795 -- Fixed a regression in custom form fields",
      "pr_body": "Custom form fields having a `queryset` attribute but no\n`limit_choices_to` could no longer be used in ModelForms.\n\nRefs #2445.\n\nThanks to @artscoop for the report.\n",
      "issue_id": 23795,
      "issue_title": "Django form fields : limit_choices_to should not be mandatory with queryset",
      "issue_body": "Hi,\nI've found something that might be a bug, and which didn't happen with Django 1.6.\nI have an admin definition which uses a custom form field (an ajax field for ForeignKey lookups which inherits from\nCharField\n), and changes the form as follows:\ndef get_form(self, request, obj=None, **kwargs):\n        form = super(ProfileAdmin, self).get_form(request, obj, **kwargs)\n        if obj is not None:\n            form.base_fields['picture'].queryset = obj.pictures.all()\n        return form\nThis works well with Django 1.6 (and the queryset filtering is ok), but clashes with Django 1.7 code (*django/forms/models.py, 333, in BaseModelForm.\ninit\n*)\nif hasattr(formfield, 'queryset'):\n                limit_choices_to = formfield.limit_choices_to  # this attribute might not be a member of the field\n                if limit_choices_to is not None:\n                    if callable(limit_choices_to):\n                        limit_choices_to = limit_choices_to()\n                    formfield.queryset = formfield.queryset.complex_filter(limit_choices_to)\nThis fails with an\nAttributeError\nbecause Django assumes, without checking, that when a formfield is altered to use a queryset, it also uses a\nlimit_choices_to\nattribute. This is not always true, it seems.\nI see no error when instead of the custom field, I use a\nModelChoiceField\n(this makes sense,\nModelChoiceField\nalways has a\nlimit_choices_to\nfield).\nlimit_choices_to = formfield.limit_choices_to\nshould be\nlimit_choices_to = getattr(formfield, 'limit_choices_to', None)",
      "issue_closed_at": "2014-11-12T15:39:19",
      "base_commit": "11b7680d0e640a7d4b1c942f74e9197b6ec924b1",
      "changes": [
        {
          "file": "django/forms/fields.py",
          "type": "function",
          "name": "widget_attrs",
          "class_name": "DecimalField",
          "code": "def widget_attrs(self, widget):\n        attrs = super(DecimalField, self).widget_attrs(widget)\n        if isinstance(widget, NumberInput) and 'step' not in widget.attrs:\n            if self.decimal_places is not None:\n                # Use exponential notation for small values since they might\n                # be parsed as 0 otherwise. ref #20765\n                step = str(Decimal('1') / 10 ** self.decimal_places).lower()\n            else:\n                step = 'any'\n            attrs.setdefault('step', step)\n        return attrs"
        },
        {
          "file": "django/forms/models.py",
          "type": "function",
          "name": "__init__",
          "class_name": "ModelMultipleChoiceField",
          "code": "def __init__(self, queryset, cache_choices=None, required=True,\n                 widget=None, label=None, initial=None,\n                 help_text='', *args, **kwargs):\n        super(ModelMultipleChoiceField, self).__init__(queryset, None,\n            cache_choices, required, widget, label, initial, help_text,\n            *args, **kwargs)"
        },
        {
          "file": "django/forms/models.py",
          "type": "function",
          "name": "__init__",
          "class_name": "ModelMultipleChoiceField",
          "code": "def __init__(self, queryset, cache_choices=None, required=True,\n                 widget=None, label=None, initial=None,\n                 help_text='', *args, **kwargs):\n        super(ModelMultipleChoiceField, self).__init__(queryset, None,\n            cache_choices, required, widget, label, initial, help_text,\n            *args, **kwargs)"
        }
      ]
    },
    {
      "pr_number": 12263,
      "pr_title": "Fixed #31166 -- Used \"raise from\" when raising ImproperlyConfigured exceptions in django.urls.resolvers.",
      "pr_body": "This will make it have a text of \"this exception is the direct result of\" instead of \"During handling of the above exception, another exception occurred\" \r\n\r\nThis is more accurate for the case of this exception.",
      "issue_id": 31166,
      "issue_title": "Provide context for ImproperlyConfigured exceptions in URL resolver.",
      "issue_body": "The patch is ready here:\n​\nhttps://github.com/django/django/pull/12263\nThis will make it have a text of \"this exception is the direct result of\" instead of \"During handling of the above exception, another exception occurred\"\nThis is more accurate for the case of this exception.\nIf this change will be merged, there are a couple more place in this module where it can be added.\nWhen it happened to me a few weeks ago, it was due to a circular import, which I then fixed. Now of course, when I tried to reproduce the circular import to show you, I couldn't. But I could achieve the same thing by renaming\nurlpatterns\nto\nxurlpatterns\n, so it wouldn't be available. This is the error I get:\nException in thread django-main-thread:\nTraceback (most recent call last):\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\urls\\resolvers.py\", line 590, in url_patterns\n    iter(patterns)\nTypeError: 'module' object is not iterable\n\nDuring handling of the above exception, another exception occurred:\n\nTraceback (most recent call last):\n  File \"C:\\Program Files\\Python38\\lib\\threading.py\", line 932, in _bootstrap_inner\n    self.run()\n  File \"C:\\Program Files\\Python38\\lib\\threading.py\", line 870, in run\n    self._target(*self._args, **self._kwargs)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\utils\\autoreload.py\", line 53, in wrapper\n    fn(*args, **kwargs)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\management\\commands\\runserver.py\", line 117, in inner_run\n    self.check(display_num_errors=True)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\management\\base.py\", line 392, in check\n    all_issues = self._run_checks(\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\management\\base.py\", line 382, in _run_checks\n    return checks.run_checks(**kwargs)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\checks\\registry.py\", line 72, in run_checks\n    new_errors = check(app_configs=app_configs)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\checks\\urls.py\", line 13, in check_url_config\n    return check_resolver(resolver)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\checks\\urls.py\", line 23, in check_resolver\n    return check_method()\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\urls\\resolvers.py\", line 408, in check\n    messages.extend(check_resolver(pattern))\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\checks\\urls.py\", line 23, in check_resolver\n    return check_method()\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\urls\\resolvers.py\", line 407, in check\n    for pattern in self.url_patterns:\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\utils\\functional.py\", line 48, in __get__\n    res = instance.__dict__[self.name] = self.func(instance)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\urls\\resolvers.py\", line 597, in url_patterns\n    raise ImproperlyConfigured(msg.format(name=self.urlconf_name))\ndjango.core.exceptions.ImproperlyConfigured: The included URLconf '<module 'barb.urls' from 'C:\\\\Users\\\\Administrator\\\\Desktop\\\\foof\\\\barb\\\\urls.py'>' does not appear to have any patterns in it. If you see valid patterns in the file then the issue is probably caused by a circular import.\nThe bad part is\nDuring handling of the above exception, another exception occurred\n.\nAfter my fix, it looks like this:\nException in thread django-main-thread:\nTraceback (most recent call last):\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\urls\\resolvers.py\", line 590, in url_patterns\n    iter(patterns)\nTypeError: 'module' object is not iterable\n\nThe above exception was the direct cause of the following exception:\n\nTraceback (most recent call last):\n  File \"C:\\Program Files\\Python38\\lib\\threading.py\", line 932, in _bootstrap_inner\n    self.run()\n  File \"C:\\Program Files\\Python38\\lib\\threading.py\", line 870, in run\n    self._target(*self._args, **self._kwargs)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\utils\\autoreload.py\", line 53, in wrapper\n    fn(*args, **kwargs)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\management\\commands\\runserver.py\", line 117, in inner_run\n    self.check(display_num_errors=True)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\management\\base.py\", line 392, in check\n    all_issues = self._run_checks(\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\management\\base.py\", line 382, in _run_checks\n    return checks.run_checks(**kwargs)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\checks\\registry.py\", line 72, in run_checks\n    new_errors = check(app_configs=app_configs)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\checks\\urls.py\", line 13, in check_url_config\n    return check_resolver(resolver)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\checks\\urls.py\", line 23, in check_resolver\n    return check_method()\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\urls\\resolvers.py\", line 408, in check\n    messages.extend(check_resolver(pattern))\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\checks\\urls.py\", line 23, in check_resolver\n    return check_method()\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\urls\\resolvers.py\", line 407, in check\n    for pattern in self.url_patterns:\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\utils\\functional.py\", line 48, in __get__\n    res = instance.__dict__[self.name] = self.func(instance)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\urls\\resolvers.py\", line 597, in url_patterns\n    raise ImproperlyConfigured(msg.format(name=self.urlconf_name)) from type_error # Ram hack, remove\ndjango.core.exceptions.ImproperlyConfigured: The included URLconf '<module 'barb.urls' from 'C:\\\\Users\\\\Administrator\\\\Desktop\\\\foof\\\\barb\\\\urls.py'>' does not appear to have any patterns in it. If you see valid patterns in the file then the issue is probably caused by a circular import.\nNow the text is corrected to\nThe above exception was the direct cause of the following exception",
      "issue_closed_at": "2020-01-17T05:21:04",
      "base_commit": "73563183c2ea92e9ef1d3a1f790a503acc14ade2",
      "changes": [
        {
          "file": "django/urls/resolvers.py",
          "type": "function",
          "name": "_compile",
          "class_name": "RoutePattern",
          "code": "def _compile(self, route):\n        return re.compile(_route_to_regex(route, self._is_endpoint)[0])"
        },
        {
          "file": "django/urls/resolvers.py",
          "type": "function",
          "name": "_route_to_regex",
          "class_name": null,
          "code": "def _route_to_regex(route, is_endpoint=False):\n    \"\"\"\n    Convert a path pattern into a regular expression. Return the regular\n    expression and a dictionary mapping the capture names to the converters.\n    For example, 'foo/<int:pk>' returns '^foo\\\\/(?P<pk>[0-9]+)'\n    and {'pk': <django.urls.converters.IntConverter>}.\n    \"\"\"\n    if not set(route).isdisjoint(string.whitespace):\n        raise ImproperlyConfigured(\"URL route '%s' cannot contain whitespace.\" % route)\n    original_route = route\n    parts = ['^']\n    converters = {}\n    while True:\n        match = _PATH_PARAMETER_COMPONENT_RE.search(route)\n        if not match:\n            parts.append(re.escape(route))\n            break\n        parts.append(re.escape(route[:match.start()]))\n        route = route[match.end():]\n        parameter = match.group('parameter')\n        if not parameter.isidentifier():\n            raise ImproperlyConfigured(\n                \"URL route '%s' uses parameter name %r which isn't a valid \"\n                \"Python identifier.\" % (original_route, parameter)\n            )\n        raw_converter = match.group('converter')\n        if raw_converter is None:\n            # If a converter isn't specified, the default is `str`.\n            raw_converter = 'str'\n        try:\n            converter = get_converter(raw_converter)\n        except KeyError as e:\n            raise ImproperlyConfigured(\n                \"URL route '%s' uses invalid converter %s.\" % (original_route, e)\n            )\n        converters[parameter] = converter\n        parts.append('(?P<' + parameter + '>' + converter.regex + ')')\n    if is_endpoint:\n        parts.append('$')\n    return ''.join(parts), converters"
        },
        {
          "file": "django/urls/resolvers.py",
          "type": "function",
          "name": "url_patterns",
          "class_name": "URLResolver",
          "code": "def url_patterns(self):\n        # urlconf_module might be a valid set of patterns, so we default to it\n        patterns = getattr(self.urlconf_module, \"urlpatterns\", self.urlconf_module)\n        try:\n            iter(patterns)\n        except TypeError:\n            msg = (\n                \"The included URLconf '{name}' does not appear to have any \"\n                \"patterns in it. If you see valid patterns in the file then \"\n                \"the issue is probably caused by a circular import.\"\n            )\n            raise ImproperlyConfigured(msg.format(name=self.urlconf_name))\n        return patterns"
        }
      ]
    },
    {
      "pr_number": 8401,
      "pr_title": "Fixed #28040 -- Updated SplitArrayWidget to use template-based widget rendering.",
      "pr_body": "https://code.djangoproject.com/ticket/28040",
      "issue_id": 28040,
      "issue_title": "Update SplitArrayWidget to use template-based widget rendering",
      "issue_body": "Error:\nAttributeError: 'SplitArrayWidget' object has no attribute 'template_name'\n.\nTraceback:\nTemplate error:\nIn template /home/vagrant/apps/pg/python/templates/shared/partials/structure.html, error at line 0\n   'SplitArrayWidget' object has no attribute 'template_name'   1 : {% load static_files %}\n   2 : {% spaceless %}\n   3 : <!DOCTYPE html>\n   4 : <html lang=\"pl\" itemscope itemtype=\"http://schema.org/Organization\" {% block html_attrs %}{% endblock html_attrs %} {% block tracking_scripts %}data-pg=\"google-tag-manager\" data-config-google-tag-manager=\"googleTagManager\"{% endblock tracking_scripts %}>\n   5 :     <head prefix=\"og: http://ogp.me/ns# fb: http://ogp.me/ns/fb#\">\n   6 :         <meta charset=\"utf-8\">\n   7 :         <meta http-equiv=\"X-UA-Compatible\" content=\"IE=edge\">\n   8 :         {% block head %}\n   9 : \n   10 :             {% block meta %}\n\n\nTraceback:\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/core/handlers/exception.py\" in inner\n  41.             response = get_response(request)\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/core/handlers/base.py\" in _get_response\n  217.                 response = self.process_exception_by_middleware(e, request)\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/core/handlers/base.py\" in _get_response\n  215.                 response = response.render()\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/template/response.py\" in render\n  107.             self.content = self.rendered_content\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/template/response.py\" in rendered_content\n  84.         content = template.render(context, self._request)\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/template/backends/django.py\" in render\n  66.             return self.template.render(context)\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/template/base.py\" in render\n  207.                     return self._render(context)\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/template/base.py\" in _render\n  199.         return self.nodelist.render(context)\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/template/base.py\" in render\n  990.                 bit = node.render_annotated(context)\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/template/base.py\" in render_annotated\n  957.             return self.render(context)\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/template/loader_tags.py\" in render\n  177.             return compiled_parent._render(context)\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/template/base.py\" in _render\n  199.         return self.nodelist.render(context)\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/template/base.py\" in render\n  990.                 bit = node.render_annotated(context)\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/template/base.py\" in render_annotated\n  957.             return self.render(context)\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/template/loader_tags.py\" in render\n  177.             return compiled_parent._render(context)\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/template/base.py\" in _render\n  199.         return self.nodelist.render(context)\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/template/base.py\" in render\n  990.                 bit = node.render_annotated(context)\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/template/base.py\" in render_annotated\n  957.             return self.render(context)\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/template/loader_tags.py\" in render\n  177.             return compiled_parent._render(context)\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/template/base.py\" in _render\n  199.         return self.nodelist.render(context)\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/template/base.py\" in render\n  990.                 bit = node.render_annotated(context)\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/template/base.py\" in render_annotated\n  957.             return self.render(context)\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/template/defaulttags.py\" in render\n  411.         return strip_spaces_between_tags(self.nodelist.render(context).strip())\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/template/base.py\" in render\n  990.                 bit = node.render_annotated(context)\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/template/base.py\" in render_annotated\n  957.             return self.render(context)\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/template/loader_tags.py\" in render\n  72.                 result = block.nodelist.render(context)\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/template/base.py\" in render\n  990.                 bit = node.render_annotated(context)\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/template/base.py\" in render_annotated\n  957.             return self.render(context)\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/template/loader_tags.py\" in render\n  72.                 result = block.nodelist.render(context)\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/template/base.py\" in render\n  990.                 bit = node.render_annotated(context)\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/template/base.py\" in render_annotated\n  957.             return self.render(context)\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/template/library.py\" in render\n  225.         _dict = self.func(*resolved_args, **resolved_kwargs)\n\nFile \"/home/vagrant/apps/pg/python/apps/shared/forms/templatetags/forms.py\" in render_field\n  85.     if not field:\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/forms/boundfield.py\" in __len__\n  64.         return len(self.subwidgets)\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/utils/functional.py\" in __get__\n  35.         res = instance.__dict__[self.name] = self.func(instance)\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/forms/boundfield.py\" in subwidgets\n  57.             for widget in self.field.widget.subwidgets(self.html_name, self.value(), attrs=attrs)\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/forms/boundfield.py\" in <genexpr>\n  56.             BoundWidget(self.field.widget, widget, self.form.renderer)\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/forms/widgets.py\" in subwidgets\n  191.         context = self.get_context(name, value, attrs)\n\nFile \"/usr/local/lib/python3.4/dist-packages/django/forms/widgets.py\" in get_context\n  212.             'template_name': self.template_name,\n\nException Type: AttributeError at /xxxx/\nException Value: 'SplitArrayWidget' object has no attribute 'template_name'\nI'm temporary fix this in my app by subclass django Widget and add template_name property\nfrom django.contrib.postgres import forms as postgres_forms\n\n\nclass SplitArrayWidget(postgres_forms.SplitArrayWidget):\n    template_name = ''",
      "issue_closed_at": "2017-04-29T18:00:32",
      "base_commit": "c920db1e57bf6bec65b075fd0baac72e13db9919",
      "changes": [
        {
          "file": "django/contrib/postgres/forms/array.py",
          "type": "line",
          "name": "line 6",
          "code": "    ArrayMaxLengthValidator, ArrayMinLengthValidator,\n)\nfrom django.core.exceptions import ValidationError\nfrom django.utils.safestring import mark_safe\nfrom django.utils.translation import gettext_lazy as _\n\nfrom ..utils import prefix_validation_error"
        },
        {
          "file": "django/contrib/postgres/forms/array.py",
          "type": "function",
          "name": "run_validators",
          "class_name": "SimpleArrayField",
          "code": "def run_validators(self, value):\n        super().run_validators(value)\n        errors = []\n        for index, item in enumerate(value):\n            try:\n                self.base_field.run_validators(item)\n            except ValidationError as error:\n                errors.append(prefix_validation_error(\n                    error,\n                    prefix=self.error_messages['item_invalid'],\n                    code='item_invalid',\n                    params={'nth': index},\n                ))\n        if errors:\n            raise ValidationError(errors)"
        },
        {
          "file": "django/contrib/postgres/forms/array.py",
          "type": "function",
          "name": "id_for_label",
          "class_name": "SplitArrayWidget",
          "code": "def id_for_label(self, id_):\n        # See the comment for RadioSelect.id_for_label()\n        if id_:\n            id_ += '_0'\n        return id_"
        },
        {
          "file": "django/contrib/postgres/forms/array.py",
          "type": "function",
          "name": "render",
          "class_name": "SplitArrayWidget",
          "code": "def render(self, name, value, attrs=None, renderer=None):\n        if self.is_localized:\n            self.widget.is_localized = self.is_localized\n        value = value or []\n        output = []\n        final_attrs = self.build_attrs(attrs)\n        id_ = final_attrs.get('id')\n        for i in range(max(len(value), self.size)):\n            try:\n                widget_value = value[i]\n            except IndexError:\n                widget_value = None\n            if id_:\n                final_attrs = dict(final_attrs, id='%s_%s' % (id_, i))\n            output.append(self.widget.render(name + '_%s' % i, widget_value, final_attrs, renderer))\n        return mark_safe(self.format_output(output))"
        }
      ]
    },
    {
      "pr_number": 11119,
      "pr_title": "Fixed #29791 -- Honor engine's autoescape attribute in render_to_string",
      "pr_body": "Passed down autoescape value from Engine to Context in Engine.render_to_string.",
      "issue_id": 29791,
      "issue_title": "Engine.render_to_string() should honor the autoescape attribute",
      "issue_body": "In\nEngine.render_to_string\n, a\nContext\nis created without specifying the engine\nautoescape\nattribute. So if you create en engine with\nautoescape=False\nand then call its\nrender_to_string()\nmethod, the result will always be autoescaped. It was probably overlooked in\n[19a5f6da329d58653bcda85]\n.",
      "issue_closed_at": "2019-03-25T17:26:24",
      "base_commit": "d4df5e1b0b1c643fe0fc521add0236764ec8e92a",
      "changes": [
        {
          "file": "django/template/engine.py",
          "type": "function",
          "name": "render_to_string",
          "class_name": "Engine",
          "code": "def render_to_string(self, template_name, context=None):\n        \"\"\"\n        Render the template specified by template_name with the given context.\n        For use in Django's test suite.\n        \"\"\"\n        if isinstance(template_name, (list, tuple)):\n            t = self.select_template(template_name)\n        else:\n            t = self.get_template(template_name)\n        # Django < 1.8 accepted a Context in `context` even though that's\n        # unintended. Preserve this ability but don't rewrap `context`.\n        if isinstance(context, Context):\n            return t.render(context)\n        else:\n            return t.render(Context(context))"
        }
      ]
    }
  ]
}