diff options
| -rw-r--r-- | docs/tutorial/6-viewsets-and-routers.md | 4 | ||||
| -rw-r--r-- | rest_framework/fields.py | 17 | ||||
| -rw-r--r-- | tests/test_fields.py | 46 | 
3 files changed, 46 insertions, 21 deletions
| diff --git a/docs/tutorial/6-viewsets-and-routers.md b/docs/tutorial/6-viewsets-and-routers.md index 816e9da6..d55a60de 100644 --- a/docs/tutorial/6-viewsets-and-routers.md +++ b/docs/tutorial/6-viewsets-and-routers.md @@ -44,8 +44,8 @@ Next we're going to replace the `SnippetList`, `SnippetDetail` and `SnippetHighl              snippet = self.get_object()              return Response(snippet.highlighted) -        def pre_save(self, obj): -            obj.owner = self.request.user +        def perform_create(self, serializer): +                serializer.save(owner=self.request.user)  This time we've used the `ModelViewSet` class in order to get the complete set of default read and write operations. diff --git a/rest_framework/fields.py b/rest_framework/fields.py index 5be2a21b..c40dc3fb 100644 --- a/rest_framework/fields.py +++ b/rest_framework/fields.py @@ -184,13 +184,11 @@ class Field(object):          self.style = {} if style is None else style          self.allow_null = allow_null -        if allow_null and self.default_empty_html is empty: -            # HTML input cannot represent `None` values, so we need to -            # forcibly coerce empty HTML values to `None` if `allow_null=True`. -            self.default_empty_html = None - -        if default is not empty: -            self.default_empty_html = default +        if self.default_empty_html is not empty: +            if not required: +                self.default_empty_html = empty +            elif default is not empty: +                self.default_empty_html = default          if validators is not None:              self.validators = validators[:] @@ -562,6 +560,11 @@ class CharField(Field):              message = self.error_messages['min_length'].format(min_length=min_length)              self.validators.append(MinLengthValidator(min_length, message=message)) +        if self.allow_null and (not self.allow_blank) and (self.default is empty): +            # HTML input cannot represent `None` values, so we need to +            # forcibly coerce empty HTML values to `None` if `allow_null=True`. +            self.default_empty_html = None +      def run_validation(self, data=empty):          # Test for the empty string here so that it does not get validated,          # and so that subclasses do not need to handle it explicitly diff --git a/tests/test_fields.py b/tests/test_fields.py index 7f7af5cc..04c721d3 100644 --- a/tests/test_fields.py +++ b/tests/test_fields.py @@ -215,25 +215,47 @@ class TestBooleanHTMLInput:          assert serializer.validated_data == {'archived': False} +class MockHTMLDict(dict): +    """ +    This class mocks up a dictionary like object, that behaves +    as if it was returned for multipart or urlencoded data. +    """ +    getlist = None + +  class TestCharHTMLInput: -    def setup(self): +    def test_empty_html_checkbox(self):          class TestSerializer(serializers.Serializer):              message = serializers.CharField(default='happy') -        self.Serializer = TestSerializer -    def test_empty_html_checkbox(self): -        """ -        HTML checkboxes do not send any value, but should be treated -        as `False` by BooleanField. -        """ -        # This class mocks up a dictionary like object, that behaves -        # as if it was returned for multipart or urlencoded data. -        class MockHTMLDict(dict): -            getlist = None -        serializer = self.Serializer(data=MockHTMLDict()) +        serializer = TestSerializer(data=MockHTMLDict())          assert serializer.is_valid()          assert serializer.validated_data == {'message': 'happy'} +    def test_empty_html_checkbox_allow_null(self): +        class TestSerializer(serializers.Serializer): +            message = serializers.CharField(allow_null=True) + +        serializer = TestSerializer(data=MockHTMLDict()) +        assert serializer.is_valid() +        assert serializer.validated_data == {'message': None} + +    def test_empty_html_checkbox_allow_null_allow_blank(self): +        class TestSerializer(serializers.Serializer): +            message = serializers.CharField(allow_null=True, allow_blank=True) + +        serializer = TestSerializer(data=MockHTMLDict({})) +        assert serializer.is_valid() +        assert serializer.validated_data == {'message': ''} + +    def test_empty_html_required_false(self): +        class TestSerializer(serializers.Serializer): +            message = serializers.CharField(required=False) + +        serializer = TestSerializer(data=MockHTMLDict()) +        assert serializer.is_valid() +        assert serializer.validated_data == {} +  class TestCreateOnlyDefault:      def setup(self): | 
