2015-11-11 70 views
3

我正在尝试创建使用我的API创建新用户的视图。 我为我的用户使用自定义模型,并创建了一个名为Profile的模型来管理与认证无关的数据。Django Rest Framework创建用户和用户配置文件

我是Django世界的新手,可能相当困难。

这里是我的models.py

class UserManager(BaseUserManager): 
    def create_user(self, email, password=None): 
     if not email: 
      raise ValueError('User must have an email address') 

     user = self.model(
      email = self.normalize_email(email), 
     ) 
     user.set_password(password) 
     user.save() 
     return user 

    def create_superuser(self, email, password): 
     user = self.create_user(email, password=password) 
     user.is_admin = True 
     user.save() 
     return user 


class User(AbstractBaseUser): 
    objects = UserManager() 
    email = models.EmailField(unique=True, db_index=True) 
    created = models.DateTimeField('created', auto_now_add=True) 
    modified = models.DateTimeField(auto_now=True) 

    is_active = models.BooleanField('active', default=True) 
    is_admin = models.BooleanField('admin', default=False) 

    USERNAME_FIELD = 'email' 

    ordering = ('created',) 

    def is_staff(self): 
     return self.is_admin 

    def has_perm(self, perm, obj=None): 
     return True 

    def has_module_perms(self, app_label): 
     return True 

    def get_short_name(self): 
     return self.email 

    def get_full_name(self): 
     return self.email 

    def __unicode__(self): 
     return self.email 


class Profile(models.Model): 
    GENDER = (
     ('M', 'Homme'), 
     ('F', 'Femme'), 
    ) 

    user = models.OneToOneField(settings.AUTH_USER_MODEL) 
    first_name = models.CharField(max_length=120, blank=False) 
    last_name = models.CharField(max_length=120, blank=False) 
    gender = models.CharField(max_length=1, choices=GENDER) 
    zip_code = models.CharField(max_length=5, validators=[MinLengthValidator(5)], blank=False) 

    def __unicode__(self): 
     return u'Profile of user: {0}'.format(self.user.email) 


def create_profile(sender, instance, created, **kwargs): 
    if created: 
     Profile.objects.create(user=instance) 
post_save.connect(create_profile, sender=User) 


def delete_user(sender, instance=None, **kwargs): 
    try: 
     instance.user 
    except User.DoesNotExist: 
     pass 
    else: 
     instance.user.delete() 
post_delete.connect(delete_user, sender=Profile) 

这里是我的serializers.py

class ProfileSerializer(serializers.ModelSerializer): 
    class Meta: 
     model = Profile 
     fields = ('first_name', 'last_name', 'gender', 'zip_code',) 


class UserSerializer(serializers.ModelSerializer): 
    profile = ProfileSerializer(required=True) 
    class Meta: 
     model = User 
     fields = ('url', 'email', 'profile', 'created',) 

这里是views.py

class UserList(generics.ListCreateAPIView): 
    permission_classes = (IsAuthenticatedOrWriteOnly,) 
    serializer_class = UserSerializer 

    def get_queryset(self): 
     if self.request.user.is_staff: 
      return User.objects.all() 
     else: 
      return self.request.user 

    def post(self, request, format=None): 
     serializer = ProfileSerializer(data=request.data) 
     print serializer.__dict__ 
     if serializer.is_valid(): 
      print "valid" 
     return Response("placeholder", status=status.HTTP_201_CREATED) 

我的目标是能够同时创建用户和配置文件,同时能够验证所有内容,我如何实现这一目标?

回答

3

事实上,Django Rest Framework无法处理嵌套关系的工作,因此您必须自己实现这些方法。我会给你一些你的代码应该是什么样子的例子。

您的看法:

class UserList(generics.ListCreateAPIView): 
    permission_classes = (IsAuthenticatedOrWriteOnly,) 
    serializer_class = UserSerializer 

    def post(self, request, format=None): 
     serializer = UserSerializer(data=request.data) 
     if serializer.is_valid(): 
      serializer.save() 
      return Response(serializer.data, status=status.HTTP_201_CREATED) 
     return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST) 

现在,你的串行的save方法将调用一个方法create当你想要的时候,你要更新一个对象来创建一个对象,并update方法。因此,让我们实施您的UserSerializercreate方法,该方法将创建配置文件和用户。这里是你的UserSerializer应该是什么样子一个简单的例子:

class UserSerializer(serializers.ModelSerializer): 
    profile = ProfileSerializer(required=True) 
    class Meta: 
     model = User 
     fields = ('url', 'email', 'profile', 'created',) 

    def create(self, validated_data): 

     # create user 
     user = User.objects.create(
      url = validated_data['url'], 
      email = validated_data['email'], 
      # etc ... 
     ) 

     profile_data = validated_data.pop('profile') 
     # create profile 
     profile = Profile.objects.create(
      user = user 
      first_name = profile_data['first_name'], 
      last_name = profile_data['last_name'], 
      # etc... 
     ) 

     return user 

正如我所说的,这是一个例子,你必须完成它做你想做的事,但现在,你知道该怎么做:)要定义更新过程中的行为,请执行update方法。

+0

谢谢,这对我有点帮助,即使我已经开始了解这一点,创建方法很容易实现,但我必须说,更新不是。 因为例如在我的UserSerializer中我有一个用于密码和密码确认的CharField,它们都是必需的,但是每当我想更新使用时,我需要再次提供这两个密码。 – soueuls

+0

请更新您的问题,然后让您谈论创作。你能更清楚地知道你想在更新方法中做什么吗? –

相关问题