Python Django Rest框架创建用户和用户配置文件

Python Django Rest框架创建用户和用户配置文件,python,django,django-rest-framework,Python,Django,Django Rest Framework,我正在尝试创建视图,该视图将使用我的API创建新用户。 我为我的用户使用了一个自定义模型,还创建了一个名为Profile的模型来管理与身份验证无关的数据 我是Django世界的新手,这可能很难 这是我的模特 class UserManager(BaseUserManager): def create_user(self, email, password=None): if not email: raise ValueError('User must

我正在尝试创建视图,该视图将使用我的API创建新用户。 我为我的用户使用了一个自定义模型,还创建了一个名为Profile的模型来管理与身份验证无关的数据

我是Django世界的新手,这可能很难

这是我的模特

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)

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

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

你的看法:

class用户列表(generics.ListCreateAPIView):
权限\u类=(IsAuthenticatedOrWriteOnly,)
serializer\u class=UserSerializer
def post(自我、请求、格式=无):
serializer=UserSerializer(data=request.data)
如果序列化程序.u有效():
serializer.save()
返回响应(serializer.data,status=status.HTTP\u 201\u已创建)
返回响应(serializer.errors,status=status.HTTP\u 400\u BAD\u请求)
现在,序列化程序的
save
方法将在创建对象时调用
create
方法,在更新对象时调用
update
方法。因此,让我们实现
UserSerializer
create
方法,该方法将创建概要文件和用户。下面是一个简单的示例,说明了
UserSerializer
应该是什么样子:

class UserSerializer(serializers.ModelSerializer):
profile=ProfileSerializer(必需=True)
类元:
模型=用户
字段=('url','email','profile','created',)
def创建(自我验证的_数据):
#创建用户
user=user.objects.create(
url=已验证的_数据['url'],
电子邮件=已验证的_数据['email'],
#等等。。。
)
profile\u data=validated\u data.pop(‘profile’)
#创建配置文件
profile=profile.objects.create(
用户=用户
first_name=配置文件数据['first_name'],
last_name=配置文件数据['last_name'],
#等等。。。
)
返回用户

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

这可能已经晚了,但为什么不完全绕过drf并使用如下信号将概要文件创建触发器降级到模型:

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)
class User(AbstractBaseUser):
      pass
class UserProfile(models.Model):
      user =  models.OneToOneField(User, on_delete=models.CASCADE, related_name='customer_profile')
      country = models.CharField(blank=True, max_length=250)
signals.py

@receiver(post_save, sender=User)
def create_user_profile(sender, instance, created, **kwargs):
    if created:
        UserProfile.objects.create(
            user = instance
        )
这基本上会在通过任何方式创建用户后触发相应的配置文件行创建;管理员、drf等

然后可以使用序列化程序更新数据。
希望这能帮助其他在这里绊倒的人

谢谢,这对我有点帮助,即使我已经开始明白这一点,create方法很容易实现,但我必须说update方法不是。因为例如在我的UserSerializer中,我有一个用于密码和密码确认的字符域,它们都是必需的,但每当我想更新使用时,我需要再次提供这两个密码。请更新您的问题,因为您谈论的是创建。你能更明确地告诉我你想在更新方法中做什么吗?太棒了,我在搜索这样的东西,它工作起来很有魅力,谢谢你@LouisBarranqueiro