【发布时间】:2016-09-29 00:14:35
【问题描述】:
我很困惑如何在 DRF 中的序列化程序和视图中实现方法:
我有一个扩展 AbstractBaseUser 的帐户模型。视图集如下所示:
class AccountViewSet(viewsets.ModelViewSet):
lookup_field = 'username'
queryset = Account.objects.all()
serializer_class = AccountSerializer
def get_permissions(self):
if self.request.method in permissions.SAFE_METHODS:
return (permissions.AllowAny(), TokenHasReadWriteScope())
if self.request.method == 'POST':
return (permissions.AllowAny(), TokenHasReadWriteScope())
return (permissions.IsAuthenticated(), IsAccountOwner(), TokenHasReadWriteScope())
def create(self, request):
serializer = self.serializer_class(data=request.data)
if serializer.is_valid():
Account.objects.create_user(**serializer.validated_data)
return Response(serializer.validated_data, status=status.HTTP_201_CREATED)
return Response({
'status': 'Bad request',
'message': 'Account could not be created with received data.'
}, status=status.HTTP_400_BAD_REQUEST)
这样的序列化器:
class AccountSerializer(serializers.ModelSerializer):
password = serializers.CharField(write_only=True, required=False)
confirm_password = serializers.CharField(write_only=True, required=False)
class Meta:
model = Account
fields = ('id', 'email', 'username', 'created_at', 'updated_at',
'first_name', 'last_name', 'tagline', 'password',
'confirm_password',)
read_only_fields = ('created_at', 'updated_at',)
def create(self, validated_data):
return Account.objects.create(**validated_data)
def update(self, instance, validated_data):
instance.username = validated_data.get('username', instance.username)
instance.save()
password = validated_data.get('password', None)
confirm_password = validated_data.get('confirm_password', None)
if password and confirm_password:
instance.set_password(password)
instance.save()
update_session_auth_hash(self.context.get('request'), instance)
return instance
def validate(self, data):
if data['password'] and data['confirm_password'] and data['password'] == data['confirm_password']:
try:
validate_password(data['password'], user=data['username']):
return data
except ValidationError:
raise serializers.ValidationError("Password is not valid.")
raise serializers.ValidationError("Passwords do not match.")
在视图的 create 方法中,它会检查序列化程序是否有效,然后将其保存并根据结果返回响应。我的第一个问题是何时调用序列化程序 create() 方法?在我看来,通过在视图的 create() 方法中调用 create_user (模型方法)似乎完全绕过了该方法。它会被调用吗?拥有它有什么意义?
其次,我在从更新方法返回状态码时遇到问题,实例保存在序列化程序中。如果验证失败,序列化程序 update() 中的代码会起作用吗?
这是我目前所拥有的:
def update(self, request, pk=None):
serializer = self.serializer_class(data=request.data)
if serializer.is_valid():
<< what goes here??? >>
return Response(serializer.validated_data, status=status.HTTP_200_OK)
except serializers.ValidationError as e:
return Response({
'status': 'Bad request',
'message': str(e)
}, status=status.HTTP_400_BAD_REQUEST)
return Response({
'status': 'Bad request',
'message': 'Account could not be updated with received data.'
}, status=status.HTTP_400_BAD_REQUEST)
我迫切需要澄清一下。我不确定如何通过视图/序列化器方法请求流,也不确定如何将实例保存在序列化器中并决定同时在视图中返回哪个响应。
编辑:
我删除了create 和update 方法,并为AccountViewSet 修复了get_permissions,并按照您的建议将用户名验证添加到validate。我还更新了序列化程序的创建和更新方法,以下是新版本:
def create(self, validated_data):
instance = super(AccountSerializer, self).create(validated_data)
instance.set_password(validated_data['password'])
instance.save()
return instance
def update(self, instance, validated_data):
instance.username = validated_data.get('username', instance.username)
password = validated_data.get('password', None)
confirm_password = validated_data.get('confirm_password', None)
if password and confirm_password:
instance.set_password(password)
instance.save()
update_session_auth_hash(self.context.get('request'), instance)
else:
instance.save()
return instance
我唯一的问题是有必要在create 之后调用set_password 吗? create 不会给新用户设置密码吗? create 和 update 的视图中没有代码可以吗?在没有视图代码的情况下,serializer.save() 在哪里被调用,序列化程序 validate 什么时候在没有调用 serializer.is_valid() 的情况下运行?
【问题讨论】:
标签: django serialization django-rest-framework