0

0

Django自定义用户模型:Admin登录失效与正确实现指南

DDD

DDD

发布时间:2025-11-07 12:41:01

|

675人浏览过

|

来源于php中文网

原创

Django自定义用户模型:Admin登录失效与正确实现指南

本文深入探讨了在django中实现自定义用户模型时,超级用户无法登录admin面板的常见问题。通过分析`abstractbaseuser`和`permissionsmixin`的内部机制,指出了因重复定义密码字段和验证方法导致的冲突。文章提供了正确的模型实现方式,强调了利用django内置认证功能的重要性,确保自定义用户模型能够安全、高效地与admin系统集成。

引言:自定义用户模型的必要性

Django提供了一个强大的内置用户模型,但在许多实际应用中,开发者可能需要根据业务需求扩展或替换它。例如,使用电子邮件而非用户名作为主要认证凭据,或者添加额外的用户属性。Django通过AUTH_USER_MODEL设置和AbstractBaseUser、PermissionsMixin等基类,为自定义用户模型提供了灵活的框架。然而,不正确的实现方式可能导致一系列问题,其中最常见的就是超级用户无法登录Django Admin面板。

Django Admin登录失效的常见问题

当开发者尝试构建一个基于电子邮件的自定义用户模型,并继承自AbstractBaseUser和PermissionsMixin时,可能会遇到超级用户在Django Admin登录时反复提示“请输入正确的电子邮件和密码”的错误,即使确认凭据无误。这个问题通常发生在以下场景:

  1. 自定义用户模型 (Customers):

    from django.utils import timezone
    from django.db import models
    from django.contrib.auth.models import AbstractBaseUser, PermissionsMixin, UserManager
    
    # 自定义管理器
    class CustomerManager(UserManager):
        def _create_user(self, email, password, **extra_fields):
            if not email:
                raise ValueError('Customers must have an email address')
            user = self.model(
                email=email,
                **extra_fields
            )
            user.set_password(password) # 使用Django内置方法设置密码
            user.save(using=self._db)
            return user
    
        def create_user(self, email=None, password=None, **extra_fields):
            extra_fields.setdefault('is_superuser', False)
            extra_fields.setdefault('is_staff', False)
            return self._create_user(email, password, **extra_fields)
    
        def create_superuser(self, name, last_name, email, phone, password, **kwargs):
            kwargs.setdefault('is_superuser', True)
            kwargs.setdefault('is_staff', True)
            return self._create_user(email, password, **kwargs)
    
    # 自定义用户模型
    class Customers (AbstractBaseUser, PermissionsMixin):
        name = models.CharField(max_length=20)
        last_name = models.CharField(max_length=20)
        email = models.EmailField(blank=False, unique=True)
        phone = models.CharField(max_length=15)
        password = models.CharField(max_length=20) # 错误:重复定义密码字段
    
        is_active = models.BooleanField(default=True)
        is_staff = models.BooleanField(default=False)
        is_superuser = models.BooleanField(default=False) # 错误:重复定义is_superuser
    
        date_joined = models.DateTimeField(default=timezone.now)
        last_login = models.DateTimeField(blank=True, null=True)
        objects = CustomerManager()
    
        USERNAME_FIELD = 'email'
        EMAIL_FIELD = 'email'
        REQUIRED_FIELDS = ['name', 'last_name', 'phone']
    
        class Meta:
            verbose_name = 'Customer'
            verbose_name_plural = 'Customers'
    
        def get_full_name(self):
            return self.name + ' ' + self.last_name
    
        def get_short_name(self):
            return self.name
    
        def check_password(self, password): # 错误:覆盖了AbstractBaseUser的密码验证方法
            return self.password == password
  2. settings.py配置:

    AUTH_USER_MODEL = 'customers.Customers'

尽管在自定义管理器中使用了user.set_password(password)来设置密码,但由于模型内部的错误定义,导致登录验证失败。

问题根源分析

上述问题的核心在于对AbstractBaseUser和PermissionsMixin这两个基类提供的核心功能的重复定义和不当覆盖。

  1. password 字段的重复定义:AbstractBaseUser基类已经包含了对用户密码的管理,它会在内部维护一个经过哈希处理的密码字段。当我们在Customers模型中再次定义password = models.CharField(max_length=20)时,实际上是创建了一个新的、未经Django哈希机制处理的明文字段。虽然CustomerManager中的set_password方法会调用AbstractBaseUser的set_password来正确哈希密码并存储,但当Django尝试验证密码时,它可能会错误地使用我们自定义的明文password字段,或者在后续的数据库操作中造成混淆。

  2. is_superuser 字段的重复定义:PermissionsMixin基类已经提供了is_superuser和is_staff等权限相关的布尔字段。重复定义is_superuser = models.BooleanField(default=False)同样会导致不必要的冗余和潜在的冲突。

    CodeBuddy
    CodeBuddy

    腾讯云AI代码助手

    下载
  3. check_password 方法的覆盖:AbstractBaseUser提供了一个健壮且安全的check_password方法,用于验证用户输入的密码与数据库中存储的哈希密码是否匹配。通过自定义def check_password(self, password): return self.password == password,我们完全覆盖了Django内置的安全验证逻辑。这个自定义方法直接比较了用户输入的密码与模型中自定义的明文password字段,这不仅不安全(因为密码以明文形式存储或被错误处理),也绕过了Django的哈希密码验证机制,从而导致登录失败。

解决方案:精简与信任Django内置功能

解决此问题的关键在于移除自定义用户模型中与AbstractBaseUser和PermissionsMixin功能重复的字段和方法,充分信任并利用Django提供的内置认证机制。

修正后的 Customers 模型:

from django.utils import timezone
from django.db import models
from django.contrib.auth.models import AbstractBaseUser, PermissionsMixin, UserManager

# CustomerManager 保持不变,因为它正确地使用了 set_password
class CustomerManager(UserManager):
    def _create_user(self, email, password, **extra_fields):
        if not email:
            raise ValueError('Customers must have an email address')
        user = self.model(
            email=email,
            **extra_fields
        )
        user.set_password(password)
        user.save(using=self._db)
        return user

    def create_user(self, email=None, password=None, **extra_fields):
        extra_fields.setdefault('is_superuser', False)
        extra_fields.setdefault('is_staff', False)
        return self._create_user(email, password, **extra_fields)

    def create_superuser(self, name, last_name, email, phone, password, **kwargs):
        kwargs.setdefault('is_superuser', True)
        kwargs.setdefault('is_staff', True)
        return self._create_user(email, password, **kwargs)

# 修正后的自定义用户模型
class Customers (AbstractBaseUser, PermissionsMixin):
    name = models.CharField(max_length=20)
    last_name = models.CharField(max_length=20)
    email = models.EmailField(blank=False, unique=True)
    phone = models.CharField(max_length=15)

    # 移除 password 字段,AbstractBaseUser 已提供

    is_active = models.BooleanField(default=True)
    is_staff = models.BooleanField(default=False)
    # 移除 is_superuser 字段,PermissionsMixin 已提供

    date_joined = models.DateTimeField(default=timezone.now)
    last_login = models.DateTimeField(blank=True, null=True)
    objects = CustomerManager()

    USERNAME_FIELD = 'email'
    EMAIL_FIELD = 'email'
    REQUIRED_FIELDS = ['name', 'last_name', 'phone']

    class Meta:
        verbose_name = 'Customer'
        verbose_name_plural = 'Customers'

    def get_full_name(self):
        return self.name + ' ' + self.last_name

    def get_short_name(self):
        return self.name

    # 移除 check_password 方法,AbstractBaseUser 已提供安全的验证逻辑

修改说明:

  1. 移除 password 字段: AbstractBaseUser 已经内置了处理密码的机制,包括存储哈希值。我们无需再显式定义一个 password 字段。
  2. 移除 is_superuser 字段: PermissionsMixin 已经提供了 is_superuser 和 is_staff 字段,用于管理用户权限和Admin访问。
  3. 移除 check_password 方法: AbstractBaseUser 提供了安全的 check_password 方法,它能够正确地验证用户输入的密码与数据库中存储的哈希密码。覆盖此方法会导致安全漏洞和认证失败。

通过这些修改,自定义用户模型将正确地继承和利用Django内置的认证和权限管理功能,从而使超级用户能够顺利登录Admin面板。

注意事项与最佳实践

  • 密码管理: 始终使用 user.set_password('your_password') 来设置用户密码,而不是直接赋值给 user.password。set_password 方法会负责对密码进行哈希处理。
  • 权限管理: AbstractBaseUser 和 PermissionsMixin 共同提供了完整的用户认证和权限管理框架。除非有非常特殊的需求,否则应尽量避免覆盖或重新实现它们的核心方法和字段。
  • 数据库迁移: 在修改模型后,务必运行 python manage.py makemigrations 和 python manage.py migrate 来更新数据库结构。如果现有数据与新模型不兼容,可能需要手动处理或清空数据库重新创建超级用户。
  • 自定义管理器: UserManager 是处理用户创建和管理的关键。在自定义管理器中,确保 create_user 和 create_superuser 方法正确地调用了 _create_user 并最终使用了 user.set_password()。

总结

在Django中构建自定义用户模型是一项常见的任务,但它要求开发者对AbstractBaseUser和PermissionsMixin的内部工作原理有清晰的理解。当遇到超级用户无法登录Admin面板的问题时,首先应检查自定义模型中是否存在对这些基类功能的重复定义或不当覆盖,特别是关于密码字段和验证方法。通过精简模型,移除冗余字段和方法,并充分利用Django内置的安全认证机制,可以确保自定义用户模型不仅功能完善,而且安全可靠地与Django Admin系统集成。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
default gateway怎么配置
default gateway怎么配置

配置default gateway的步骤:1、了解网络环境;2、获取路由器IP地址;3、登录路由器管理界面;4、找到并配置WAN口设置;5、配置默认网关;6、保存设置并退出;7、检查网络连接是否正常。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

223

2023.12.07

数据库三范式
数据库三范式

数据库三范式是一种设计规范,用于规范化关系型数据库中的数据结构,它通过消除冗余数据、提高数据库性能和数据一致性,提供了一种有效的数据库设计方法。本专题提供数据库三范式相关的文章、下载和课程。

358

2023.06.29

如何删除数据库
如何删除数据库

删除数据库是指在MySQL中完全移除一个数据库及其所包含的所有数据和结构,作用包括:1、释放存储空间;2、确保数据的安全性;3、提高数据库的整体性能,加速查询和操作的执行速度。尽管删除数据库具有一些好处,但在执行任何删除操作之前,务必谨慎操作,并备份重要的数据。删除数据库将永久性地删除所有相关数据和结构,无法回滚。

2082

2023.08.14

vb怎么连接数据库
vb怎么连接数据库

在VB中,连接数据库通常使用ADO(ActiveX 数据对象)或 DAO(Data Access Objects)这两个技术来实现:1、引入ADO库;2、创建ADO连接对象;3、配置连接字符串;4、打开连接;5、执行SQL语句;6、处理查询结果;7、关闭连接即可。

349

2023.08.31

MySQL恢复数据库
MySQL恢复数据库

MySQL恢复数据库的方法有使用物理备份恢复、使用逻辑备份恢复、使用二进制日志恢复和使用数据库复制进行恢复等。本专题为大家提供MySQL数据库相关的文章、下载、课程内容,供大家免费下载体验。

256

2023.09.05

vb中怎么连接access数据库
vb中怎么连接access数据库

vb中连接access数据库的步骤包括引用必要的命名空间、创建连接字符串、创建连接对象、打开连接、执行SQL语句和关闭连接。本专题为大家提供连接access数据库相关的文章、下载、课程内容,供大家免费下载体验。

326

2023.10.09

数据库对象名无效怎么解决
数据库对象名无效怎么解决

数据库对象名无效解决办法:1、检查使用的对象名是否正确,确保没有拼写错误;2、检查数据库中是否已存在具有相同名称的对象,如果是,请更改对象名为一个不同的名称,然后重新创建;3、确保在连接数据库时使用了正确的用户名、密码和数据库名称;4、尝试重启数据库服务,然后再次尝试创建或使用对象;5、尝试更新驱动程序,然后再次尝试创建或使用对象。

412

2023.10.16

vb连接access数据库的方法
vb连接access数据库的方法

vb连接access数据库方法:1、使用ADO连接,首先导入System.Data.OleDb模块,然后定义一个连接字符串,接着创建一个OleDbConnection对象并使用Open() 方法打开连接;2、使用DAO连接,首先导入 Microsoft.Jet.OLEDB模块,然后定义一个连接字符串,接着创建一个JetConnection对象并使用Open()方法打开连接即可。

411

2023.10.16

clawdbot ai使用教程 保姆级clawdbot部署安装手册
clawdbot ai使用教程 保姆级clawdbot部署安装手册

Clawdbot是一个“有灵魂”的AI助手,可以帮用户清空收件箱、发送电子邮件、管理日历、办理航班值机等等,并且可以接入用户常用的任何聊天APP,所有的操作均可通过WhatsApp、Telegram等平台完成,用户只需通过对话,就能操控设备自动执行各类任务。

14

2026.01.29

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 22.3万人学习

Django 教程
Django 教程

共28课时 | 3.6万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号