Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Fix validation for ListSerializer #8979

Merged
merged 20 commits into from
May 29, 2023
Merged
Show file tree
Hide file tree
Changes from 19 commits
Commits
Show all changes
20 commits
Select commit Hold shift + click to select a range
a717d74
fix: Make the instance variable of child serializer point to the corr…
saadullahaleem May 10, 2023
4bbc689
fix formatting issues for list serializer validation fix
saadullahaleem May 10, 2023
516ba52
fix imports sorting for list serializer tests
saadullahaleem May 10, 2023
b9c8d19
remove django 2.2 from docs index (#8982)
deronnax May 14, 2023
e6655e3
Declared Django 4.2 support in README.md (#8985)
MehrazRumman May 15, 2023
01b99dd
Fix Links in Documentation to Django `reverse` and `reverse_lazy` (#8…
theomega May 17, 2023
8d7e250
fix URLPathVersioning reverse fallback (#7247)
jornvanwier May 18, 2023
b9b50bd
Make set_value a method within `Serializer` (#8001)
tienne-B May 24, 2023
869d46f
fix: Make the instance variable of child serializer point to the corr…
saadullahaleem May 10, 2023
f969143
Make set_value a method within `Serializer` (#8001)
tienne-B May 24, 2023
a1f03d5
fix: Make the instance variable of child serializer point to the corr…
saadullahaleem May 10, 2023
9ac6417
fix: Make the instance variable of child serializer point to the corr…
saadullahaleem May 10, 2023
76110bf
fix formatting issues for list serializer validation fix
saadullahaleem May 10, 2023
c1ee7e7
fix: Make the instance variable of child serializer point to the corr…
saadullahaleem May 10, 2023
74cb500
fix formatting issues for list serializer validation fix
saadullahaleem May 10, 2023
9577880
Merge branch 'master' into fix/list_serializer_validation
saadullahaleem May 27, 2023
7197a86
fix linting
saadullahaleem May 27, 2023
bb4102a
Update rest_framework/serializers.py
auvipy May 27, 2023
0dc6201
Update rest_framework/serializers.py
saadullahaleem May 27, 2023
d747449
fix: instance variable in list serializer, remove commented code
saadullahaleem May 28, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
15 changes: 14 additions & 1 deletion rest_framework/serializers.py
Original file line number Diff line number Diff line change
Expand Up @@ -609,6 +609,13 @@ def __init__(self, *args, **kwargs):
self.min_length = kwargs.pop('min_length', None)
assert self.child is not None, '`child` is a required argument.'
assert not inspect.isclass(self.child), '`child` has not been instantiated.'

# instance, data = kwargs.get('instance', []), kwargs.get('data', []) # if you prefer one line assignment support
instance = kwargs.get('instance', [])
data = kwargs.get('data', [])
if instance and data:
assert len(data) == len(instance), 'Data and instance should have same length'

super().__init__(*args, **kwargs)
self.child.bind(field_name='', parent=self)

Expand Down Expand Up @@ -683,7 +690,13 @@ def to_internal_value(self, data):
ret = []
errors = []

for item in data:
for idx, item in enumerate(data):
if (
hasattr(self, 'instance')
and self.instance
and len(self.instance) > idx
):
self.child.instance = self.instance[idx]
try:
validated = self.child.run_validation(item)
except ValidationError as exc:
Expand Down
61 changes: 61 additions & 0 deletions tests/test_serializer.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@
import pickle
import re
import sys
import unittest
from collections import ChainMap
from collections.abc import Mapping

Expand Down Expand Up @@ -783,3 +784,63 @@ def test_nested_key(self):
ret = {'a': 1}
self.s.set_value(ret, ['x', 'y'], 2)
assert ret == {'a': 1, 'x': {'y': 2}}


class MyClass(models.Model):
name = models.CharField(max_length=100)
value = models.CharField(max_length=100, blank=True)

app_label = "test"

@property
def is_valid(self):
return self.name == 'valid'


class MyClassSerializer(serializers.ModelSerializer):
class Meta:
model = MyClass
fields = ('id', 'name', 'value')

def validate_value(self, value):
if value and not self.instance.is_valid:
raise serializers.ValidationError(
'Status cannot be set for invalid instance')
return value


class TestMultipleObjectsValidation(unittest.TestCase):
def setUp(self):
self.objs = [
MyClass(name='valid'),
MyClass(name='invalid'),
MyClass(name='other'),
]

def test_multiple_objects_are_validated_separately(self):

serializer = MyClassSerializer(
data=[{'value': 'set', 'id': instance.id} for instance in
self.objs],
instance=self.objs,
many=True,
partial=True,
)

assert not serializer.is_valid()
assert serializer.errors == [
{},
{'value': ['Status cannot be set for invalid instance']},
{'value': ['Status cannot be set for invalid instance']}
]

def test_exception_raised_when_data_and_instance_length_different(self):

with self.assertRaises(AssertionError):
MyClassSerializer(
data=[{'value': 'set', 'id': instance.id} for instance in
self.objs],
instance=self.objs[:-1],
many=True,
partial=True,
)