django自定义用户模型密码未被哈希处理

Ben*_*Max 10 django hash django-admin django-permissions django-users

我有自己的自定义用户模型,也有自己的Manger.

楷模:

class MyUser(AbstractBaseUser, PermissionsMixin):
    email = models.EmailField(max_length=255, unique=True)
    first_name = models.CharField(max_length=35)
    last_name = models.CharField(max_length=35)
    username = models.CharField(max_length=70, unique=True)
    date_of_birth = models.DateField()
    is_active = models.BooleanField(default=True)
    is_admin = models.BooleanField(default=False)

    @property
    def is_staff(self):
        return self.is_admin

    def get_full_name(self):
        return ('%s %s') % (self.first_name, self.last_name)

    def get_short_name(self):
        return self.username

    objects = MyUserManager()
    USERNAME_FIELD = 'email'
    REQUIRED_FIELDS = ['first_name', 'last_name', 'username', 'date_of_birth']
Run Code Online (Sandbox Code Playgroud)

经理:

class MyUserManager(BaseUserManager):
    def create_user(self, email, first_name, last_name, username, date_of_birth, password=None, **kwargs):
        if not email:
            raise ValueError('User must have an email address')

        user = self.model(
            email=self.normalize_email(email),
            first_name=first_name,
            last_name=last_name,
            username=username,
            date_of_birth=date_of_birth,
            **kwargs
        )
        user.set_password(self.cleaned_data["password"])
        user.save(using=self._db)
        return user

    def create_superuser(self, email, first_name, last_name, username, date_of_birth, password, **kwargs):
        user = self.create_user(
            email,
            first_name=first_name,
            last_name=last_name,
            username=username,
            date_of_birth=date_of_birth,
            password=password,
            is_superuser=True,
            **kwargs
        )
        user.is_admin = True
        user.save(using=self._db)
        return user
Run Code Online (Sandbox Code Playgroud)

创建新用户时没有任何错误,一切正常.但是当我尝试登录时我不能.所以我检查了用户的密码进行确认,密码显示为纯文本strongpassword,当更改管理表单以获取散列密码时,ReadOnlyPasswordHashField我在密码字段中出现错误,即使我set_password()create_user()函数内部使用了Manger .

无效的密码格式或未知的散列算法

但是,如果我手动set_password('strongpassword')为该用户执行操作,则会对其进行哈希处理.你能帮我解决这个问题吗?谢谢.

Ala*_*air 13

看起来您以不使用经理create_user方法的方式创建了用户,例如通过Django管理员.

如果您创建自定义用户,则需要定义正确处理密码的自定义模型表单和模型管理员.

否则,当通过Django管理员创建用户时,密码不会被散列.

用于创建自定义用户的文档中示例显示了如何创建模型表单和模型管理员.


小智 6

我知道现在已经太晚了,但我只是将其发布以供将来参考。如果您通过调用序列化器上的保存函数来创建新用户,则需要重写序列化器的创建函数,如下所示(这非常明显,但我在它上面停留了一会儿。 ...)

class SignUpView(views.APIView):
    authentication_classes = ()
    permission_classes = (permissions.AllowAny,)

    def post(self, request, format=None):
        serializer = UserSerializer(data=request.data)
        serializer.is_valid(raise_exception=True)
        serializer.save()
        return Response(serializer.data, status=status.HTTP_201_CREATED)
Run Code Online (Sandbox Code Playgroud)
class UserSerializer(serializers.ModelSerializer):

    password = serializers.CharField(
        min_length=6, write_only=True, required=True)

    class Meta:
        model = User
        fields = (
            'id', 'email', 'password', 'is_staff',
            'is_active', 'date_joined')

    def create(self, validated_data):
        return User.objects.create_user(**validated_data)
Run Code Online (Sandbox Code Playgroud)