Django之Auth模块

Auth模块引入

  • 我们在创建一个Django项目之后,直接执行数据库迁移命令会自动生成很多表
    • django_session
    • auth_user
  • Django在启动之后就可以直接访问admin路由,需要输入用户名和密码,数据参考的就是auth_user表,并且必须是管理员用户才能进入

【1】创建超级用户(管理员)

在终端输入一下指令

python 复制代码
python36 manage.py createsuperuser

在终端中操作

python 复制代码
E:\Python39\python.exe "E:\Pycharm\PyCharm 2023.1.3\plugins\python\helpers\pycharm\django_manage.py" createsuperuser "E:/Old Boy/django_project/day14"
Tracking file by folder pattern:  migrations
Username (leave blank to use 'administrator'):  dream
Email address:  
Warning: Password input may be echoed.
Password:  521
Warning: Password input may be echoed.
Password (again):  521
This password is too short. It must contain at least 8 characters.
Bypass password validation and create user anyway? [y/N]: This password is entirely numeric.
 y
Superuser created successfully.

Process finished with exit code 0

【2】利用auth_user表完成登录注册功能

【2.0】基础登录

【2.0.1】配置路由

python 复制代码
path('login/', views.login),

【2.0.2】前端代码

html 复制代码
<form action="" method="post">
    {#  取消crsf校验  #}
    {% csrf_token %}
    <p>username:<input type="text" name="username"></p>
    <p>password:<input type="text" name="password"></p>
    <input type="submit" class="btn btn-success">
</form>

【2.0.3】后端代码

python 复制代码
from django.shortcuts import render
from django.contrib import auth


# Create your views here.
def login(request):
    if request.method == 'POST':
        # (1) 取到前端输入的用户名和密码
        username = request.POST.get('username')
        password = request.POST.get('password')
        # (2) 进行用户名和密码的校验
        # 从用户表中获取数据
        # --- 1、表如何获取
        # --- 2、表中的密码是密文,如何比对
        # 导入 auth 校验模块 : from django.contrib import auth
        user_obj = auth.authenticate(request, username=username, password=password)
        # [1] 用户名和密码正确的情况下
        print(user_obj)  # dream ---- 这是用户对象内部封装的一个方法 __str__ 方法
        print(user_obj.username)  # dream
        print(user_obj.password)  # pbkdf2_sha256$260000$011PbZAjKIWBfAUJ61Rcyn$vNUYq5L70/ljTLEeJ2dBJtDTEKFDTKzFioFPjZYMdU4=
        # [2] 用户名和密码不正确的情况下
        print(user_obj)  # None ---- 如果数据不符合则返回None

        '''
        【1】自动查找 auth_user 表
        【2】自动给密码加密进行比对
        注意事项:
            参数必须传入用户名和密码
            不能只传入一个用户名(一步就帮助我们筛选出用户数据)
        '''

    return render(request, 'login.html')

【2.1】保存用户状态

  • 如果使用auth模块,就使用其中所有封装好的方法
    • 如果不想使用,就单独封装方法,不要调用其中的方法
python 复制代码
auth.login(request, user_obj)
  • 类似于 request.session[key] = user_obj
  • 只要执行了上面的方法,就可以在任何地方通过 request.user获取当前用户的登录对象
python 复制代码
from django.shortcuts import render
from django.contrib import auth


# Create your views here.
def login(request):
    if request.method == 'POST':
        # (1) 取到前端输入的用户名和密码
        username = request.POST.get('username')
        password = request.POST.get('password')
        # (2) 进行用户名和密码的校验
        # 从用户表中获取数据
        # --- 1、表如何获取
        # --- 2、表中的密码是密文,如何比对
        # 导入 auth 校验模块 : from django.contrib import auth
        user_obj = auth.authenticate(request, username=username, password=password)

        # 判断当前用户是否存在 --- 存在则有值,不存在则返回None
        if user_obj:
            # 保存用户状态
            auth.login(request, user_obj)  # 类似于 request.session[key] = user_obj
             # 只要执行了上面的方法,就可以在任何地方通过 request.user 获取当前用户的登录对象


        '''
        【1】自动查找 auth_user 表
        【2】自动给密码加密进行比对
        注意事项:
            参数必须传入用户名和密码
            不能只传入一个用户名(一步就帮助我们筛选出用户数据)
        '''

    return render(request, 'login.html')

【2.2】登录后跳转

python 复制代码
from django.shortcuts import render, redirect, HttpResponse
from django.contrib import auth


# Create your views here.
def login(request):
    if request.method == 'POST':
        # (1) 取到前端输入的用户名和密码
        username = request.POST.get('username')
        password = request.POST.get('password')
        # (2) 进行用户名和密码的校验
        # 从用户表中获取数据
        # --- 1、表如何获取
        # --- 2、表中的密码是密文,如何比对
        # 导入 auth 校验模块 : from django.contrib import auth
        user_obj = auth.authenticate(request, username=username, password=password)

        # 判断当前用户是否存在 --- 存在则有值,不存在则返回None
        if user_obj:
            # 保存用户状态
            auth.login(request, user_obj)  # 类似于 request.session[key] = user_obj
            # 只要执行了上面的方法,就可以在任何地方通过 request.user 获取当前用户的登录对象
            # 登陆成功后跳转页面
            return redirect('/home/')

        '''
        【1】自动查找 auth_user 表
        【2】自动给密码加密进行比对
        注意事项:
            参数必须传入用户名和密码
            不能只传入一个用户名(一步就帮助我们筛选出用户数据)
        '''

    return render(request, 'login.html')

def home(request):
    print(request.user)  # dream ---- 这是用户对象内部封装的一个方法 __str__ 方法
    print(request.user)
    # 登陆成功: dream ---- 这是用户对象内部封装的一个方法 __str__ 方法
    # 未登录访问: AnonymousUser ---- 匿名用户
    # 本质上是自动去django_session里面查找到当前用户对象,然后封装到 request.user 中
    return HttpResponse("OK")
  • 判断用户是否登录
python 复制代码
def home(request):
    print(request.user)
    # 登陆成功: dream ---- 这是用户对象内部封装的一个方法 __str__ 方法
    # 未登录访问: AnonymousUser ---- 匿名用户
    # 本质上是自动去django_session里面查找到当前用户对象,然后封装到 request.user 中
    # 判断用户是否登陆
    print(request.user.is_authenticated()) # 匿名用户返回 False

    return HttpResponse("OK")

【2.2.1】登录后才能访问页面 -------- 局部配置

python 复制代码
from django.contrib.auth.decorators import login_required


# 添加装饰器 --- 指定未登录的跳转页面
@login_required(login_url='/login/') # 局部配置
def home(request):
    '''用户登录后才能访问的页面'''
    print(request.user)
    # 登陆成功: dream ---- 这是用户对象内部封装的一个方法 __str__ 方法
    # 未登录访问: AnonymousUser ---- 匿名用户
    # 本质上是自动去django_session里面查找到当前用户对象,然后封装到 request.user 中
    # 判断用户是否登陆
    print(request.user.is_authenticated())  # 匿名用户返回 False

    return HttpResponse("OK")

【2.2.2】登录后才能访问页面 ----- 全局配置

  • 在settings文件中添加
python 复制代码
LOGIN_URL = '/login/'
python 复制代码
from django.contrib.auth.decorators import login_required


# 添加装饰器 --- 指定未登录的跳转页面
@login_required
def home(request):
    '''用户登录后才能访问的页面'''
    print(request.user)
    # 登陆成功: dream ---- 这是用户对象内部封装的一个方法 __str__ 方法
    # 未登录访问: AnonymousUser ---- 匿名用户
    # 本质上是自动去django_session里面查找到当前用户对象,然后封装到 request.user 中
    # 判断用户是否登陆
    print(request.user.is_authenticated())  # 匿名用户返回 False

    return HttpResponse("OK")

@login_required
def index(request):
    return HttpResponse("index")

【2.3】小结

  • 局部/全局优先级
    • 局部大于全局
  • 各自的优点
    • 全局的优点
      • 无需书写重复的代码,但是页面的跳转很单一
    • 局部的优点
      • 在于不同的视图函数再用户没有登录的情况下可以跳转到不同的页面

【3】修改密码

【3.1】路由文件的配置

python 复制代码
# 修改密码
path('set_password/', views.set_password),

【3.2】后端代码

python 复制代码
@login_required
def set_password(request):
    if request.method == 'POST':
        old_password = request.POST.get('old_password')
        new_password = request.POST.get('new_password')
        confirm_password = request.POST.get('confirm_password')
        # 先校验两次密是否一致
        if new_password == confirm_password:
            # 校验旧密码是否相同
            is_right = request.user.check_password(old_password)  # 内部自己加密密码进行比对
            # 返回的结果为  True 或者 False
            if is_right:
                # 修改密码
                request.user.set_password(confirm_password) # 仅仅在修改对象的属性
                # 修改完密码后进行保存数据
                request.user.save()

        return redirect('/login/')

    return render(request, 'set_password.html', locals())

【3.3】前端代码

python 复制代码
<form action="" method="post">
    {#  取消crsf校验  #}
    {% csrf_token %}
    <p>username:<input type="text" name="username" disabled value="{{ request.user.username }}"></p>
    <p>old_password:<input type="text" name="old_password"></p>
    <p>new_password:<input type="text" name="new_password"></p>
    <p>confirm_password:<input type="text" name="confirm_password"></p>
    <input type="submit" class="btn btn-success">
</form>

【4】注销

【4.1】路由配置

python 复制代码
# 注销用户
path('login_out/', views.login_out),

【4.2】后端代码

python 复制代码
@login_required
def login_out(request):
    auth.logout(request)  # 清空当前登录用户的数据 ----- request.session.flush()
    return redirect('/login/')

【5】注册功能

【5.1】路由配置

python 复制代码
# 注册用户
path('register/', views.register),

【5.2】后盾

python 复制代码
def register(request):
    if request.method == 'POST':
        username = request.POST.get('username')
        password = request.POST.get('password')
        confirm_password = request.POST.get('confirm_password')
        # 操作auth_user表写入数据
        # User.objects.create(username=username, password=confirm_password)  # 会写入数据,但是保存的密码是明文的,没有加密
        # 创建普通用户
        User.objects.create_user(username=username, password=confirm_password)
        # 创建超级用户 - 了解 --- 使用代码创建超级用户,邮箱和密码是必填的,而用命令创建可以忽略邮箱
        # User.objects.create_superuser(username=username, password=password,email=email)
    return render(request, 'register.html')

【6】总结

【6.1】校验密码是否正确

python 复制代码
from django.contrib import auth
# (1) 取到前端输入的用户名和密码
username = request.POST.get('username')
password = request.POST.get('password')
# (2) 进行用户名和密码的校验 ---- 参数必须传用户名和密码
user_obj = auth.authenticate(request, username=username, password=password)


# [1] 用户名和密码正确的情况下
print(user_obj)  
# dream ---- 这是用户对象内部封装的一个方法 __str__ 方法
print(user_obj.username)  
# dream
print(user_obj.password)  
# pbkdf2_sha256$260000$011PbZAjKIWBfAUJ61Rcyn$vNUYq5L70/ljTLEeJ2dBJtDTEKFDTKzFioFPjZYMdU4=

# [2] 用户名和密码不正确的情况下
print(user_obj)  # None ---- 如果数据不符合则返回None

【6.2】保存用户状态

python 复制代码
# 判断当前用户是否存在 --- 存在则有值,不存在则返回None
if user_obj:
    # 保存用户状态
    auth.login(request, user_obj)  # 类似于 request.session[key] = user_obj
    # 只要执行了上面的方法,就可以在任何地方通过 request.user 获取当前用户的登录对象

【6.3】判断当前用户是否登录

python 复制代码
# 判断用户是否登陆
# 匿名用户返回 False  正常用户返回 True
print(request.user.is_authenticated)  

【6.4】获取当前登录的用户

python 复制代码
print(request.user)
# 登陆成功: dream ---- 这是用户对象内部封装的一个方法 __str__ 方法
# 未登录访问: AnonymousUser ---- 匿名用户
# 本质上是自动去django_session里面查找到当前用户对象,然后封装到 request.user 中

【6.5】检验用户是否登录装饰器

python 复制代码
from django.contrib.auth.decorators import login_required
  • 1、局部配置
  • 2、全局配置
  • 3、全局/局部的优缺点

【6.6】校验原密码

python 复制代码
# 校验旧密码是否相同
is_right = request.user.check_password(old_password)  # 内部自己加密密码进行比对
# 返回的结果为  True 或者 False

【6.7】修改密码

python 复制代码
# 修改密码
request.user.set_password(confirm_password)  # 仅仅在修改对象的属性
# 修改完密码后进行保存数据
request.user.save()

【6.8】注销登录用户

python 复制代码
auth.logout(request)  
# 清空当前登录用户的数据 ----- request.session.flush()

【6.9】注册

python 复制代码
# 操作auth_user表写入数据
# 会写入数据,但是保存的密码是明文的,没有加密
User.objects.create(username=username, password=confirm_password) 
python 复制代码
# 创建普通用户
User.objects.create_user(username=username, password=confirm_password)
python 复制代码
# 创建超级用户 - 了解 --- 使用代码创建超级用户,邮箱和密码是必填的,而用命令创建可以忽略邮箱
User.objects.create_superuser(username=username, password=password,email=email)

【7】扩展auth_user表

【7.1】方式一

python 复制代码
from django.db import models
from django.contrib.auth.models import User, AbstractUser


# Create your models here.
# 扩展 auth_user 表
# 第一种方式 : 一对一关系(不推荐)
class UserDetail(models.Model):
    phone = models.CharField(max_length=32)
    user = models.OneToOneField(to='User', on_delete=models.CASCADE)

【7.2】方式二

python 复制代码
from django.db import models
from django.contrib.auth.models import User, AbstractUser

# 第二种方式 : 面向对象的继承
class UserInfo(AbstractUser):
    '''
    如果继承了AbstractUser
    那么在执行数据库迁移命令的时候,auth_user表就不会被创建
    而 UserInfo 会在 auth_user表 的基础上添加自定义扩展的字段

    优点:
        直接通过自己定义的表快速完成操作及扩展

    前提
        (1)在执行之前没有执行过数据库迁移命令
            auth_user 表没有被创建
            如果当前库已经被创建,则需要更换新的库
        (2)继承的表里面不要覆盖 AbstractUser 里面的字段名
            表里面有的字段不要动,只扩展额外的字段即可
        (3)需要再配置文件中声明Django要使用 UserInfo 替代 auth_user
            AUTH_USER_MODEL = 'app01.UserInfo'  ---'应用名.表名'
    '''
    phone = models.CharField(max_length=32)
  • 需要再配置文件中声明Django要使用UserInfo代替auth_user
python 复制代码
 AUTH_USER_MODEL = 'app01.UserInfo'  ---'应用名.表名'
  • 如果自己写表代替了auth_user
  • auth模块功能正常使用,参考的表也由auth_user变成了UserInfo
相关推荐
wn5317 分钟前
【Go - 类型断言】
服务器·开发语言·后端·golang
Narutolxy16 分钟前
Python 单元测试:深入理解与实战应用20240919
python·单元测试·log4j
希冀12330 分钟前
【操作系统】1.2操作系统的发展与分类
后端
Amo Xiang39 分钟前
2024 Python3.10 系统入门+进阶(十五):文件及目录操作
开发语言·python
liangbm31 小时前
数学建模笔记——动态规划
笔记·python·算法·数学建模·动态规划·背包问题·优化问题
GoppViper1 小时前
golang学习笔记29——golang 中如何将 GitHub 最新提交的版本设置为 v1.0.0
笔记·git·后端·学习·golang·github·源代码管理
B站计算机毕业设计超人1 小时前
计算机毕业设计Python+Flask微博情感分析 微博舆情预测 微博爬虫 微博大数据 舆情分析系统 大数据毕业设计 NLP文本分类 机器学习 深度学习 AI
爬虫·python·深度学习·算法·机器学习·自然语言处理·数据可视化
羊小猪~~1 小时前
深度学习基础案例5--VGG16人脸识别(体验学习的痛苦与乐趣)
人工智能·python·深度学习·学习·算法·机器学习·cnn
爱上语文2 小时前
Springboot的三层架构
java·开发语言·spring boot·后端·spring
serve the people2 小时前
springboot 单独新建一个文件实时写数据,当文件大于100M时按照日期时间做文件名进行归档
java·spring boot·后端