From 082abce81e48fe9397cce096e13d5199c99adfe9 Mon Sep 17 00:00:00 2001 From: Loic Bistuer Date: Tue, 7 Oct 2014 00:09:21 +0700 Subject: [PATCH] [1.7.x] Fixed #23594 -- Fixed deepcopy on ErrorList. Thanks Troy Grosfield for the report and Tim Graham for the tests. Backport of ec2fd02bb3 from master --- django/forms/utils.py | 9 +++++++++ docs/releases/1.7.1.txt | 2 ++ tests/forms_tests/tests/test_util.py | 23 +++++++++++++++++++++++ 3 files changed, 34 insertions(+) diff --git a/django/forms/utils.py b/django/forms/utils.py index 2147014a5de..aa14e6fe559 100644 --- a/django/forms/utils.py +++ b/django/forms/utils.py @@ -131,6 +131,15 @@ class ErrorList(UserList, list): return list(error)[0] return force_text(error) + def __reduce_ex__(self, *args, **kwargs): + # The `list` reduce function returns an iterator as the fourth element + # that is normally used for repopulating. Since we only inherit from + # `list` for `isinstance` backward compatibility (Refs #17413) we + # nullify this iterator as it would otherwise result in duplicate + # entries. (Refs #23594) + info = super(UserList, self).__reduce_ex__(*args, **kwargs) + return info[:3] + (None, None) + # Utilities for time zone support in DateTimeField et al. diff --git a/docs/releases/1.7.1.txt b/docs/releases/1.7.1.txt index c481a4b4c49..f743447d56e 100644 --- a/docs/releases/1.7.1.txt +++ b/docs/releases/1.7.1.txt @@ -89,3 +89,5 @@ Bugfixes * Fixed ``MigrationWriter`` to handle builtin types without imports (:ticket:`23560`). + +* Fixed ``deepcopy`` on ``ErrorList`` (:ticket:`23594`). diff --git a/tests/forms_tests/tests/test_util.py b/tests/forms_tests/tests/test_util.py index 5f0b549bdb8..c2dbc5ba8ae 100644 --- a/tests/forms_tests/tests/test_util.py +++ b/tests/forms_tests/tests/test_util.py @@ -1,6 +1,8 @@ # -*- coding: utf-8 -*- from __future__ import unicode_literals +import copy + from django.core.exceptions import ValidationError from django.forms.utils import flatatt, ErrorDict, ErrorList from django.test import TestCase @@ -66,3 +68,24 @@ class FormsUtilTestCase(TestCase): '') self.assertHTMLEqual(str(ErrorDict({'name': mark_safe(example)})), '') + + def test_error_dict_copy(self): + e = ErrorDict() + e['__all__'] = ErrorList([ + ValidationError( + message='message %(i)s', + params={'i': 1}, + ), + ValidationError( + message='message %(i)s', + params={'i': 2}, + ), + ]) + + e_copy = copy.copy(e) + self.assertEqual(e, e_copy) + self.assertEqual(e.as_data(), e_copy.as_data()) + + e_deepcopy = copy.deepcopy(e) + self.assertEqual(e, e_deepcopy) + self.assertEqual(e.as_data(), e_copy.as_data())