Django奇怪的SlugField验证,在clean()之前没有引发错误,返回了未清理的数据

时间:2018-02-19 09:40:02

标签: python django django-models django-forms django-2.0

django 2.0

我有一个django模型,有不同的slug字段:

from django.core.validators import validate_slug

class MyModel(models.Model):
     # with slug field
     slug_field = models.SlugField(max_length=200)

     # or charfield with slug validator (should be exactly the same)
     char_field = models.CharField(max_length=200, validators=[validate_slug])

我的第一个问题,在我的表单中,我有一个干净的方法,来验证多个字段的值,而不是单独的。理论上应该在clean_fields方法之后调用此方法,但即使clean_fields引发错误也会调用此方法。

我的forms.py:

class MyForm(forms.ModelForm):
    class Meta:
        model = MyModel
        fields = '__all__'

    def clean(self):
        cleaned_data = super().clean()
        print(cleaned_data.get('slug_field'))  # > None
        print(cleaned_data.get('char_field'))  # > ééé; uncleaned data
        print(self.errors)  # only from slug_field
        return cleaned_data

使用SlugField时,slug_field中未设置cleaned_data,如果它无效,并且在引发错误并由表单返回给用户之后。 (我不明白为什么clean()甚至到达,因为clean_fields()之前已经提出错误了)

问题在于,CharField包含任何自定义验证程序(validate_slug或自制验证程序),未清除的值将在cleaned_data中返回。但是,仍然会提出验证错误,但是之后。

这对我来说非常危险,因为我过去信任cleaned_data,修改未保存在模型中的数据。

1 个答案:

答案 0 :(得分:5)

在字段的验证程序之后调用clean()方法 。如果别名无效,则它不会出现在cleaned_data中。您的clean方法应该处理这种情况,例如:

def clean():
    cleaned_data = super().clean()
    print(self.errors)  # You should see the alias error here
    if 'alias' in cleaned_data:
        print(cleaned_data['alias'])
        # do any code that relies on cleaned_data['alias'] here
    return cleaned_data

有关详细信息,请参阅cleaning fields that depend on each other上的文档。