我正在使用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
字段?
它可用于将任何类型的数据添加到对象的序列化表示中。
所以我原以为在validate
期间也可以使用它?
答案 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()
我真的希望这不是正确的做法,因为对于看似非常简单的情况来说,这似乎完全令人费解。希望其他人可以发布更好的方法。