mirror of
https://github.com/django/django.git
synced 2025-10-24 06:06:09 +00:00
Fixed #26819 -- Fixed BaseModelFormSet.validate_unique() "unhashable type: list" crash.
This commit is contained in:
@@ -696,8 +696,12 @@ class BaseModelFormSet(BaseFormSet):
|
|||||||
for field in unique_check if field in form.cleaned_data
|
for field in unique_check if field in form.cleaned_data
|
||||||
)
|
)
|
||||||
# Reduce Model instances to their primary key values
|
# Reduce Model instances to their primary key values
|
||||||
row_data = tuple(d._get_pk_val() if hasattr(d, '_get_pk_val') else d
|
row_data = tuple(
|
||||||
for d in row_data)
|
d._get_pk_val() if hasattr(d, '_get_pk_val')
|
||||||
|
# Prevent "unhashable type: list" errors later on.
|
||||||
|
else tuple(d) if isinstance(d, list)
|
||||||
|
else d for d in row_data
|
||||||
|
)
|
||||||
if row_data and None not in row_data:
|
if row_data and None not in row_data:
|
||||||
# if we've already seen it then we have a uniqueness failure
|
# if we've already seen it then we have a uniqueness failure
|
||||||
if row_data in seen_data:
|
if row_data in seen_data:
|
||||||
|
@@ -1459,6 +1459,33 @@ class ModelFormsetTest(TestCase):
|
|||||||
self.assertEqual(player1.team, team)
|
self.assertEqual(player1.team, team)
|
||||||
self.assertEqual(player1.name, 'Bobby')
|
self.assertEqual(player1.name, 'Bobby')
|
||||||
|
|
||||||
|
def test_inlineformset_with_arrayfield(self):
|
||||||
|
class SimpleArrayField(forms.CharField):
|
||||||
|
"""A proxy for django.contrib.postgres.forms.SimpleArrayField."""
|
||||||
|
def to_python(self, value):
|
||||||
|
value = super().to_python(value)
|
||||||
|
return value.split(',') if value else []
|
||||||
|
|
||||||
|
class BookForm(forms.ModelForm):
|
||||||
|
title = SimpleArrayField()
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = Book
|
||||||
|
fields = ('title',)
|
||||||
|
|
||||||
|
BookFormSet = inlineformset_factory(Author, Book, form=BookForm)
|
||||||
|
data = {
|
||||||
|
'book_set-TOTAL_FORMS': '3',
|
||||||
|
'book_set-INITIAL_FORMS': '0',
|
||||||
|
'book_set-MAX_NUM_FORMS': '',
|
||||||
|
'book_set-0-title': 'test1,test2',
|
||||||
|
'book_set-1-title': 'test1,test2',
|
||||||
|
'book_set-2-title': 'test3,test4',
|
||||||
|
}
|
||||||
|
author = Author.objects.create(name='test')
|
||||||
|
formset = BookFormSet(data, instance=author)
|
||||||
|
self.assertEqual(formset.errors, [{}, {'__all__': ['Please correct the duplicate values below.']}, {}])
|
||||||
|
|
||||||
def test_model_formset_with_custom_pk(self):
|
def test_model_formset_with_custom_pk(self):
|
||||||
# a formset for a Model that has a custom primary key that still needs to be
|
# a formset for a Model that has a custom primary key that still needs to be
|
||||||
# added to the formset automatically
|
# added to the formset automatically
|
||||||
|
Reference in New Issue
Block a user