创建Django自定义用户模型
环境
Windows 11 家庭版
Python 3.10.2
Django 4.0.2
支持使用虚拟环境
Background (背景)
在Django中,根据理论来说应该创建自定义用户模型。虽然在本页中没有提及,但在以后的初次登录时,需要提前准备一个用于强制更改密码的列。
相关文章
Django 第1章:创建 Django 自定义用户模型
Django 第2章:在 Django 初次登录时强制进行密码更改
Django 第3章:强制要求在一定时间内未更改密码的用户进行密码修改
Django 第4章:生成随机且具有有效期限的 URL,并经由上级批准以发行账户
Django 第5章:通过密码尝试次数锁定和使用随机且具有有效期限的 URL 进行本人验证以解除锁定
状态
刚刚使用python startproject mysite构建了项目。
现在要创建自定义用户模型,并将认证方式更改为使用Email和密码验证。
步骤:mysite/settings.py
用下面的命令创建usersApp。
python manage.py startapp users。
接下来,在settings.py中使刚刚创建的users被识别,并定义AUTH_USER_MODEL。
INSTALLED_APPS = [
'django.contrib.admin',
'django.contrib.auth',
'django.contrib.contenttypes',
'django.contrib.sessions',
'django.contrib.messages',
'django.contrib.staticfiles',
# Local
'users', #追加
]
AUTH_USER_MODEL = 'users.User' # 追加
...
LANGUAGE_CODE = 'ja' # 変更
TIME_ZONE = 'Asia/Tokyo' # 変更
## 步骤: 用户/模型.py
from django.utils import timezone
from django.db import models
from django.contrib.auth.models import AbstractBaseUser, PermissionsMixin, UserManager
from django.core.mail import send_mail
import uuid as uuid_lib
class UserManager(UserManager):
def _create_user(self, email, password, **extra_fields):
email = self.normalize_email(email)
user = self.model(email=email, **extra_fields)
user.set_password(password)
user.save(using=self._db)
return user
def create_user(self, email, password=None, **extra_fields):
extra_fields.setdefault('is_staff', False)
extra_fields.setdefault('is_superuser', False)
return self._create_user(email, password, **extra_fields)
def create_superuser(self, email, password, **extra_fields):
extra_fields.setdefault('is_staff', True)
extra_fields.setdefault('is_superuser', True)
if extra_fields.get("is_staff") is not True:
raise ValueError('Superuser must have is_staff=True.')
if extra_fields.get("is_superuser") is not True:
raise ValueError('Superuser must have is_superuser=True.')
email = self.normalize_email(email)
user = self.model(email=email, **extra_fields)
user.username = email
user.set_password(password)
user.save(using=self._db)
return user
class User(AbstractBaseUser, PermissionsMixin):
"""Custom User"""
class Meta:
verbose_name = 'ユーザ'
verbose_name_plural = 'ユーザ'
uuid = models.UUIDField(default=uuid_lib.uuid4, primary_key=True, editable=False) # 管理ID
username = models.CharField(max_length=30, unique=False) # ユーザ氏名
email = models.EmailField(unique=True, blank=True, null=True) # メールアドレス = これで認証する
is_active = models.BooleanField(default=True) # アクティブ権限
is_staff = models.BooleanField(default=True) # スタッフ権限
is_superuser = models.BooleanField(default=False) # 管理者権限
date_joined = models.DateTimeField(default=timezone.now) # アカウント作成日時
password_changed = models.BooleanField(default=False) # パスワードを変更したかどうかのフラグ
password_changed_date = models.DateTimeField(blank=True, null=True) # 最終パスワード変更日時
objects = UserManager()
EMAIL_FIELD = 'email'
USERNAME_FIELD = 'email'
REQUIRED_FIELD = ''
def clean(self):
super().clean()
self.email = self.__class__.objects.normalize_email(self.email)
def email_user(self, subject, message, from_email=None, **kwargs):
send_mail(subject, message, from_email, [self.email], **kwargs)
def __str__(self):
return self.email
def get_full_name(self):
return self.username
def get_short_name(self):
return self.username
verbose_name的作用是在管理界面上以日语而不是英语显示Users。
在email = models.EmailField(unique=True, blank=True, null=True)中,如果没有blank=True, null=True,那么在添加用户时会在管理界面上出现IntegrityError: UNIQUE constraint failed: users_user.email的错误,请注意。
![image.png](https://cdn.silicloud.com/blog-img/blog/img/657d8ad8913a08637a6cb7ba/15-0.png)
![image.png](https://cdn.silicloud.com/blog-img/blog/img/657d8ad8913a08637a6cb7ba/16-0.png)
在 `admin.py` 文件中,由于仅仅使用 `models.py` 无法确认更改,因此需要在管理界面上显示自定义用户。
from django.contrib import admin
from django.contrib.auth.admin import UserAdmin
from django.utils.translation import gettext, gettext_lazy as _
from .models import User
@admin.register(User)
class UserAdmin(UserAdmin):
fieldsets = (
(None, {'fields': ('username', 'password')}),
(_('Personal Info',), {'fields': ('email',)}),
(_('Permissions',), {'fields': ('is_active', 'is_staff', 'is_superuser',)}),
(_('Password',), {'fields': ('password_changed', 'password_changed_date',)}),
(_('Important Dates',), {'fields': ('last_login', 'date_joined',)}),
)
list_display = ('username', 'email', 'is_active',)
![image.png](https://cdn.silicloud.com/blog-img/blog/img/657d8ad8913a08637a6cb7ba/19-0.png)
![image.png](https://cdn.silicloud.com/blog-img/blog/img/657d8ad8913a08637a6cb7ba/20-0.png)
![image.png](https://cdn.silicloud.com/blog-img/blog/img/657d8ad8913a08637a6cb7ba/21-0.png)
![image.png](https://cdn.silicloud.com/blog-img/blog/img/657d8ad8913a08637a6cb7ba/22-0.png)
![image.png](https://cdn.silicloud.com/blog-img/blog/img/657d8ad8913a08637a6cb7ba/23-0.png)
![image.png](https://cdn.silicloud.com/blog-img/blog/img/657d8ad8913a08637a6cb7ba/24-0.png)
![image.png](https://cdn.silicloud.com/blog-img/blog/img/657d8ad8913a08637a6cb7ba/25-0.png)
![image.png](https://cdn.silicloud.com/blog-img/blog/img/657d8ad8913a08637a6cb7ba/26-0.png)
![image.png](https://cdn.silicloud.com/blog-img/blog/img/657d8ad8913a08637a6cb7ba/27-0.png)
辛苦了。
请提供更多的上下文以便我进行准确的翻译。
Django 第1篇:创建自定义用户模型
Django 第2篇:在Django初次登录时强制修改密码
Django 第3篇:对于一段时间未修改密码的用户,强制要求其修改密码
Django 第4篇:生成随机且带有有效期的URL,并由上级人员批准后发放账户
Django 第5篇:通过密码尝试次数锁定和使用随机且带有效期的URL进行本人确认解锁