python全栈开发day74-普通验证码和滑块验证码
一、昨日内容回顾
django认证系统
1. 默认auth_user ```
# 创建用户
from django.contrib.auth.models import User
User.objects.create_user()
User.objects.create_superuser(username=,password=...)
# 去数据库校验用户名或密码是否正常,正确返回user对象,错误返回None
user = auth.authticate(username=...,password=...)
# auth.login(request,user)
# 1. 让当前用户登录,给cookie和session写入数据
# 2. 这步操作执行之后,用户的浏览器有了cookie,session表里有了对应数据
# session中根据cookie在session中找到user信息 ———》 user_obj/匿名用户
# request.user = user_obj/匿名用户
# auth.logout(request) # 退出登录删除session中的用户信息,利用request.session.flush()
# user = request.user
user.is_authticated()
# 给登录认证加装饰器
from django.contrib.auth.decorators import login_required
# from django.contrib.auth.models import User
user = User.objects.create_user(username='',password='',email='')
user = User.objects.create_superuser()
# user = request.user
user.check_password("旧密码")
user.set_password("新密码")
user.save()
``` 2. 扩展默认auth_user
from django.contrib.auth.models import AbstractUser
class UserInfo(AbstractUser):
....
....
AUTH_USER_MODEL = "app名.UserInfo"
二、今日内容总结
1、Django ORM的三种模式:
1)、使用默认的ManyToManyField创建第三张表:
优势:
可以使用ORM提供的快捷方法:
add(),all(),clear(),set(),remove()
劣势:不能扩展第三张表。
from django.db import models # Create your models here. class Book(models.Model):
title = models.CharField(max_length=32, verbose_name="书名") def __str__(self):
return self.title class Author(models.Model):
name = models.CharField(max_length=32, verbose_name="作者姓名")
books = models.ManyToManyField(to='Book') def __str__(self):
return self.name
2). 自己创建第三张关系表
1. 优势:
1. 可以自己扩展第三章关系表的字段(婚恋网站的男女用户的约会记录)
2. 劣势:
1. 不能使用ORM提供的快捷方法
from django.db import models # Create your models here. class Book(models.Model):
title = models.CharField(max_length=32, verbose_name="书名") def __str__(self):
return self.title class Author(models.Model):
name = models.CharField(max_length=32, verbose_name="作者姓名") def __str__(self):
return self.name # 自己创建多对多的第三张表
class Author2Book(models.Model):
book = models.ForeignKey(to='Book')
author = models.ForeignKey(to='Author') class Meta:
unique_together = (('book', 'author'), )
3). 自己创建第三张表,在ManyToManyField中通过through和through_fields指定表名和字段
1. 优势:
1. 可以使用ORM提供的部分快捷方法
1. all()
2. 可以扩展第三张关系表的字段
from django.db import models # Create your models here. class Book(models.Model):
title = models.CharField(max_length=32, verbose_name="书名") def __str__(self):
return self.title class Author(models.Model):
name = models.CharField(max_length=32, verbose_name="作者姓名")
books = models.ManyToManyField(
to='Book',
through='Author2Book',
through_fields=('author', 'book')
) def __str__(self):
return self.name # 自己创建多对多的第三张表
class Author2Book(models.Model):
book = models.ForeignKey(to='Book')
author = models.ForeignKey(to='Author') class Meta:
unique_together = (('book', 'author'), )
2.bss项目验证码登录
1. 登录页面实现及验证码布局
2. 生成验证码图片
1. pip install pillow
from PIL import Image, ImageDraw, ImageFont
3. 如何生成验证码?
4. 验证码保存在哪里?
5. 点击刷新验证码如何实现?
from django.shortcuts import render, HttpResponse
from django.http import JsonResponse
from django import views
from blog.forms import LoginForm
from django.contrib.auth import authenticate, login, logout
# Create your views here. V_CODE = "" # 登录
class Login(views.View): def get(self, request):
form_obj = LoginForm()
return render(request, "login.html", {"form_obj": form_obj}) def post(self, request):
res = {"code": 0}
print(request.POST)
username = request.POST.get('username')
pwd = request.POST.get('password')
v_code = request.POST.get('v_code')
print(v_code)
print(V_CODE)
# 先判断验证码是否正确
if v_code.upper() != request.session.get("v_code", ""):
res["code"] = 1
res["msg"] = "验证码错误"
else:
# 校验用户名密码是否正确
user = authenticate(username=username, password=pwd)
if user:
# 用户名密码正确
login(request, user)
else:
# 用户名或密码错误
res["code"] = 1
res["msg"] = "用户名或密码错误"
return JsonResponse(res) # 首页
class Index(views.View):
def get(self, request):
return render(request, "index.html") # 专门用来返回验证码图片的视图
def v_code(request):
# 随机生成图片
from PIL import Image, ImageDraw, ImageFont
import random
# 生成随机颜色的方法
def random_color():
return random.randint(0, 255), random.randint(0, 255), random.randint(0, 255)
# 生成图片对象
image_obj = Image.new(
"RGB", # 生成图片的模式
(250, 35), # 图片大小
random_color()
)
# 生成一个准备写字的画笔
draw_obj = ImageDraw.Draw(image_obj) # 在哪里写
font_obj = ImageFont.truetype('static/font/kumo.ttf', size=28) # 加载本地的字体文件 # 生成随机验证码
tmp = []
for i in range(5):
n = str(random.randint(0, 9))
l = chr(random.randint(65, 90))
u = chr(random.randint(97, 122))
r = random.choice([n, l, u])
tmp.append(r)
# 每一次取到要写的东西之后,往图片上写
draw_obj.text(
(i*45+25, 0), # 坐标
r, # 内容
fill=random_color(), # 颜色
font=font_obj # 字体
) # # 加干扰线
# width = 250 # 图片宽度(防止越界)
# height = 35
# for i in range(5):
# x1 = random.randint(0, width)
# x2 = random.randint(0, width)
# y1 = random.randint(0, height)
# y2 = random.randint(0, height)
# draw_obj.line((x1, y1, x2, y2), fill=random_color())
#
# # 加干扰点
# for i in range(40):
# draw_obj.point([random.randint(0, width), random.randint(0, height)], fill=random_color())
# x = random.randint(0, width)
# y = random.randint(0, height)
# draw_obj.arc((x, y, x+4, y+4), 0, 90, fill=random_color()) v_code = "".join(tmp) # 得到最终的验证码
# global V_CODE
# V_CODE = v_code # 保存在全局变量不行!!!
# 将该次请求生成的验证码保存在该请求对应的session数据中
request.session['v_code'] = v_code.upper() # 将上一步生成的图片保存在本地的static目录下
# 每一次 都在硬盘中保存再读取都涉及IO操作,会慢
# with open('static/oo.png', 'wb') as f:
# image_obj.save(f)
#
# with open('static/oo.png', "rb") as f:
# data = f.read()
# 直接将生成的图片保存在内存中
from io import BytesIO
f = BytesIO()
image_obj.save(f, "png")
# 从内存读取图片数据
data = f.getvalue()
return HttpResponse(data, content_type="image/png")
验证码校验
{% load static %}
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>登录</title>
<link type="text/css" rel='stylesheet' href="{% static 'bootstrap/dist/css/bootstrap.css' %}">
</head>
<body>
<div class="container">
<div class="row">
<div class="col-md-4 col-md-offset-4">
<form action="/login/" method="post" class="form-horizontal" id="login_form">
{% csrf_token %}
<div class="form-group">
<label for="{{ form_obj.username.id_for_label }}">{{ form_obj.username.label }}</label>
{{ form_obj.username }}
</div>
<div class="form-group">
<label for="{{ form_obj.password.id_for_label }}">{{ form_obj.password.label }}</label>
{{ form_obj.password }}
</div>
<div class="form-group">
<label for="v_code" style="display: block">验证码</label>
<input type="text" id="v_code" class="form-control" style="display: inline-block;width: 180px">
<img src="/v_code/" id='img_check' alt="" width="200px" height="35px" border="black solid 1px">
</div>
<p id="error-p" class="err-text"></p>
<button id="login-btn" type="button" class="btn btn-default">登录</button>
</form>
</div>
</div>
</div>
<script src="{% static 'jquery/jquery.js' %}"></script>
<script>
$('#login-btn').click(function () {
let password = $('#id_password').val();
let username = $('#id_username').val();
let csrf_token = $("[name='csrfmiddlewaretoken']").val();
let v_code = $("#v_code").val()
$.ajax({
url: '/login/',
type: 'post',
data: {
username: username,
password: password,
csrfmiddlewaretoken: csrf_token,
v_code:v_code,
},
success: function (res) {
if(res.code!==0){
$('#error-p').text(res.msg).css("color",'red')
}else {
location.href='/index/'
}
}
})
})
$('input').focus(function () {
$('#error-p').text('');
})
$("#img_check").click(function () {
this.src += '?';
})
</script>
</body>
</html>
login.html
三、预习和扩展
1. 用%s还是format
https://www.cnblogs.com/liwenzhou/p/8570701.html
python全栈开发day74-普通验证码和滑块验证码的更多相关文章
- Python全栈开发【模块】
Python全栈开发[模块] 本节内容: 模块介绍 time random os sys json & picle shelve XML hashlib ConfigParser loggin ...
- Python全栈开发【面向对象进阶】
Python全栈开发[面向对象进阶] 本节内容: isinstance(obj,cls)和issubclass(sub,super) 反射 __setattr__,__delattr__,__geta ...
- Python全栈开发【面向对象】
Python全栈开发[面向对象] 本节内容: 三大编程范式 面向对象设计与面向对象编程 类和对象 静态属性.类方法.静态方法 类组合 继承 多态 封装 三大编程范式 三大编程范式: 1.面向过程编程 ...
- Python全栈开发【基础四】
Python全栈开发[基础四] 本节内容: 匿名函数(lambda) 函数式编程(map,filter,reduce) 文件处理 迭代器 三元表达式 列表解析与生成器表达式 生成器 匿名函数 lamb ...
- Python全栈开发【基础三】
Python全栈开发[基础三] 本节内容: 函数(全局与局部变量) 递归 内置函数 函数 一.定义和使用 函数最重要的是减少代码的重用性和增强代码可读性 def 函数名(参数): ... 函数体 . ...
- Python全栈开发【基础二】
Python全栈开发[基础二] 本节内容: Python 运算符(算术运算.比较运算.赋值运算.逻辑运算.成员运算) 基本数据类型(数字.布尔值.字符串.列表.元组.字典) 其他(编码,range,f ...
- Python全栈开发【基础一】
Python全栈开发[第一篇] 本节内容: Python 的种类 Python 的环境 Python 入门(解释器.编码.变量.input输入.if流程控制与缩进.while循环) if流程控制与wh ...
- python 全栈开发之路 day1
python 全栈开发之路 day1 本节内容 计算机发展介绍 计算机硬件组成 计算机基本原理 计算机 计算机(computer)俗称电脑,是一种用于高速计算的电子计算机器,可以进行数值计算,又可 ...
- Python全栈开发
Python全栈开发 一文让你彻底明白Python装饰器原理,从此面试工作再也不怕了. 一.装饰器 装饰器可以使函数执行前和执行后分别执行其他的附加功能,这种在代码运行期间动态增加功能的方式,称之为“ ...
- 老男孩最新Python全栈开发视频教程(92天全)重点内容梳理笔记 看完就是全栈开发工程师
为什么要写这个系列博客呢? 说来讽刺,91年生人的我,同龄人大多有一份事业,或者有一个家庭了.而我,念了次985大学,年少轻狂,在大学期间迷信创业,觉得大学里的许多课程如同吃翔一样学了几乎一辈子都用不 ...
随机推荐
- Codeforces 1091E New Year and the Acquaintance Estimation Erdős–Gallai定理
题目链接:E - New Year and the Acquaintance Estimation 题解参考: Havel–Hakimi algorithm 和 Erdős–Gallai theore ...
- 通过URL传递中文参数的乱码处理
环境:web.xml中配置了 <filter> <filter-name>encodingFilter</filter-name> <filter-class ...
- ranlib 作用
ar 命令用于更新,维护管理静态库. ranlib 命令用于 更新库的符号索引表. 当只执行了ar命令(用于更新)时, ld连接时会仍然报错,查找不到更新的变量或函数,此时需要用ranlib来更新库的 ...
- page_cleaner: 1000ms intended loop took 4724ms. The settings might not be optimal. (flushed=1037, during the time.)
2018-07-09T14:28:56.853600Z 0 [Note] InnoDB: page_cleaner: 1000ms intended loop took 4724ms. The set ...
- SmartGit/HG
SmartGit/HG 是一款开放源代码的.跨平台的.支持 Git 和 Mercurial 的 SVN 图形客户端,可运行在Windows.Linux 和 MAC OS X 系统上.可用的最新版本 S ...
- tcp和udp协议的聊天 和udp协议的时间同步机制-----编码
tcp协议聊天 服务端:: 客户端 udp协议的聊天 ############ udp协议 ########### 服务器 import socket sk = socket.socket(type ...
- 43)django-用户认证,授权,自定义用户认证
##用户认证 django自带用户认证系统,包括认证和授权.用户认证系统由用户,权限,用户组,密码,cookie和session给组成. ###用户认证系统设置 #settings.py INSTAL ...
- Python-Numpy数组计算
一.NumPy:数组计算 1.NumPy是高性能科学计算和数据分析的基础包.它是pandas等其他各种工具的基础.2.NumPy的主要功能: ndarray,一个多维数组结构,高效且节省空间 无需循环 ...
- react native 打包Ignoring return value of function declared with warn_unused_result attribute
从 github上下载 项目 用于学习查看别人的代码, 当执行完npm install 用xcode 打开 发现俩个错误提示Ignoring return value of function dec ...
- MybatisPlus使用介绍
创建UserController测试类 package com.cppdy.controller; import org.apache.ibatis.session.RowBounds; import ...