DRF:在序列化程序验证期间访问SerializerMethodField

时间:2015-01-18 14:10:24

标签: django django-rest-framework

我正在使用Django Rest Framework 3.0,我有一个模型:

class Vote(models.Model):
    name = ...
    token = models.CharField(max_length=50)

其中token是我从request IP信息生成的唯一标识符,以防止同一用户投票两次

我有一个序列化器:

class VoteSerializer(serializers.ModelSerializer):
    name = ...
    token = serializers.SerializerMethodField()

    class Meta:
        model = Vote
        fields = ("id", "name", "token")

    def validate(self, data):
        if Rating.objects.filter(token=data['token'], name=data['name']).exists():
            raise serializers.ValidationError("You have already voted for this")
        return data

    def get_token(self, request):
        s = ''.join((self.context['request'].META['REMOTE_ADDR'], self.context['request'].META.get('HTTP_USER_AGENT', '')))
        return md5(s).hexdigest()

CreateView

但我得到了一个

KeyError: 'token' 

当我尝试发布并创建新的Vote时。为什么验证时数据中不包含token字段?

The docs mention

  

它可用于将任何类型的数据添加到对象的序列化表示中。

所以我原以为在validate期间也可以使用它?

1 个答案:

答案 0 :(得分:2)

调查一下,似乎SerializerMethodField字段在验证发生后被调用(没有深入研究代码,我不知道为什么会这样 - 这似乎是反直觉的。)

我已经将相关代码移到了视图中(从概念上来说,这实际上更有道理)。

要使其正常运行,我需要执行以下操作:

class VoteCreateView(generics.CreateAPIView):
    serializer_class = VoteSerializer

    def get_serializer(self, *args, **kwargs):
        # kwarg.data is a request MergedDict which is immutable so we have
        # to copy the data to a dict first before inserting our token
        d = {}
        for k, v in kwargs['data'].iteritems():
            d[k] = v
        d['token'] = self.get_token()
        kwargs['data'] = d
        return super(RatingCreateView, self).get_serializer(*args, **kwargs)

    def get_token(self):
        s = ''.join((self.request.META['REMOTE_ADDR'], self.request.META.get('HTTP_USER_AGENT', '')))
        return md5(s).hexdigest()

我真的希望这不是正确的做法,因为对于看似非常简单的情况来说,这似乎完全令人费解。希望其他人可以发布更好的方法。