Django缓存+图片验证
1 django缓存
1.1 django缓存作用
由于Django是动态网站,所有每次请求均会去数据进行相应的操作,当程序访问量大时,耗时必然会更加明显,缓存将一个某个views的返回值保存至内存或者memcache中,5分钟内再有人来访问时,则不再去执行view中的操作,而是直接从内存或者Redis中之前缓存的内容拿到,并返回
1.2 django六种缓存
1.2.1 开发调试缓存
# 开发调试缓存(虽然配置上,但实际没有缓存,还是到数据库取)
CACHES = {
'default': {
'BACKEND': 'django.core.cache.backends.dummy.DummyCache', # 引擎
#注: 下面这些参数时公用的,五种缓存都可以使用
'TIMEOUT': 300, # 缓存超时时间(默认300,None表示永不过期,0表示立即过期)
'OPTIONS':{
'MAX_ENTRIES': 300, # 最大缓存个数(默认300)
'CULL_FREQUENCY': 3, # 缓存到达最大个数之后,剔除缓存个数的比例(3就是1/3)
},
}
}
1.2.2 内存缓存
# 注:内存缓存本质上就是在内存中维护一个字典,所存储的形式就是字典的键值对组合
CACHES = {
'default': {
'BACKEND': 'django.core.cache.backends.locmem.LocMemCache',
'LOCATION': 'unique-snowflake', #这个参数指定变量名必须唯一
}
}
1.2.3 文件缓存
CACHES = {
'default': {
'BACKEND': 'django.core.cache.backends.filebased.FileBasedCache',
'LOCATION': os.path.join(BASE_DIR,'cache'), #缓存内容存放的文件夹路径
}
}
1.2.4 数据库缓存
'''
memcache没有办法持久化,数据类型单一,集群方向很强(使用少)
'''
# 注:执行创建表命令 python manage.py createcachetable
CACHES = {
'default': {
'BACKEND': 'django.core.cache.backends.db.DatabaseCache',
'LOCATION': 'my_cache_table', # 数据库表名(名字是自己取的)
}
}
1.2.5 Memcache缓存(两种)
# 注:Memcache缓存有两个模块:python-memcached模块、 pylibmc模块
CACHES = {
'default': {
'BACKEND': 'django.core.cache.backends.memcached.MemcachedCache',
'LOCATION': '127.0.0.1:11211', #使用ip加端口连接memcached
}
}
CACHES = {
'default': {
'BACKEND': 'django.core.cache.backends.memcached.MemcachedCache',
'LOCATION': 'unix:/tmp/memcached.sock', #以文件的形式连本地memcached
}
}
CACHES = {
'default': {
'BACKEND': 'django.core.cache.backends.memcached.MemcachedCache',
'LOCATION': [ # memcached天生支持集群
#1 均衡分配
'172.19.26.240:11211',
'172.19.26.242:11211',
#2 调整权重(权重和请求比例成正比)
('172.19.26.240:11211',1),
('172.19.26.242:11211',10),
]
}
}
# 注: pylibmc模块只改变上面'BACKEND'配置为下面样式即可
# 'BACKEND': 'django.core.cache.backends.memcached.PyLibMCCache',
1.3 django中缓存三种应用
1.3.1 页面级别缓存
- 需要在views.py文件中引入cache_page
# 1、views.py文件中的处理函数
from django.views.decorators.cache import cache_page
@cache_page(6) #6秒后缓存失效
def cache(request):
import time
ctime = time.time()
return render(request,'cache.html',{'ctime':ctime})
# 2、cache.html文件中内容都会被缓存,5s后才会失效
<body>
<h1>{{ ctime }}</h1> #页面刷新时时间不会时刻变化,5s过后才会变一次
</body>
1.3.2 模板级别缓存
- 直接在cache.html模板文件中指定某个值放入缓存
{#1 在文件最顶部引入TemplateTag#}
{% load cache %}
<body>
{#2 使用缓存 c1是缓存的key值 #}
{% cache 5 c1 %} {# 将数据缓存5秒 #}
{{ ctime }}
{% endcache %}
</body>
1.3.3 全局缓存
- 只需要再settings.py中间件配置的首尾各加一条
MIDDLEWARE = [
'django.middleware.cache.UpdateCacheMiddleware',
# 其他中间件...
'django.middleware.cache.FetchFromCacheMiddleware',
]
1.4 安装django缓存模块
pip install django-redis==4.12.1
2 实现图片验证码
2.1 syl/settings.py 中配置缓存
# 缓存配置
CACHES = {
# django存缓默认位置,redis 0号库
# default: 连接名称
"default": {
"BACKEND": "django_redis.cache.RedisCache",
"LOCATION": "redis://127.0.0.1:6379/0",
"OPTIONS": {
"CLIENT_CLASS": "django_redis.client.DefaultClient",
}
},
# django session存 reidis 1 号库(现在基本不需要使用)
"session": {
"BACKEND": "django_redis.cache.RedisCache",
"LOCATION": "redis://127.0.0.1:6379/1",
"OPTIONS": {
"CLIENT_CLASS": "django_redis.client.DefaultClient",
}
},
# 图形验证码,存redis 2号库
"img_code": {
"BACKEND": "django_redis.cache.RedisCache",
"LOCATION": "redis://127.0.0.1:6379/2",
"OPTIONS": {
"CLIENT_CLASS": "django_redis.client.DefaultClient",
}
}
}
# 配置session使用redis存储
SESSION_ENGINE = "django.contrib.sessions.backends.cache"
# 配置session存储的位置: 使用cache中的 session配置
SESSION_CACHE_ALIAS = "session"
2.2 新建应用verifications
'''2.1 在apps文件夹下新建应用: verifications'''
python ../manage.py startapp verifications # 切换到apps文件夹下执行创建命令
'''2.2 在syl/settings.py中添加应用'''
INSTALLED_APPS = [ 'verifications.apps.VerificationsConfig', ]
'''2.3 在syl/urls.py主路由中添加'''
path('verify/', include('verifications.urls')),
'''2.4 添加子路由: verifications/urls.py'''
from django.urls import path
from . import views urlpatterns = [
# path('image_codes/', views.ImageCodeView.as_view())
]
2.3 图片验证码captcha使用
1.下载captcha压缩包captcha.zip,放到项目packages文件夹下
2.解压captcha.zip放到syl/libs文件夹下
3.解压文件中的syl/libs/captcha/captcha.py 右键运行即可生成图片验证码
unzip xxx.zip
2.4 在verifications/views.py中使用
from django.shortcuts import render
# Create your views here.
from django.http import HttpResponse, HttpResponseForbidden
from django.views import View
from django_redis import get_redis_connection
from libs.captcha.captcha import captcha
class ImageCodeView(View):
def get(self, request):
# 1.接收数据
uuid = request.GET.get('uuid')
# 2.校验数据
if not uuid:
return HttpResponseForbidden('uuid无效')
# 3.处理业务
# 获取图片文本内容和图片二进制代码
text, image = captcha.generate_captcha()
# 4.把uuid和图片文本存入redis
redis_client = get_redis_connection('img_code') # 获取redis客户端
# 5.写入redis(是字符串)
redis_client.setex(uuid, 60 * 5, text)
# 6.返回响应图片
return HttpResponse(image, content_type='image/jpg')
2.5 测试验证码接口
http://192.168.56.100:8888/verify/image_codes/?uuid=66ea64aa-fbe6-11ea-a3d3- 005056c00008
- linux终端
127.0.0.1:6379># select 2
OK
127.0.0.1:6379[2]># keys *
1) "66ea64aa-fbe6-11ea-a3d3-005056c00008"
127.0.0.1:6379[2]># get 66ea64aa-fbe6-11ea-a3d3-005056c00008
"JEZ6"
3 短信验证
3.2 使用容联云发送代码测试
3.2.1 安装容联云sdk
pip install ronglian_sms_sdk
3.2.2 utils/rl_sms.py
# -*- coding: utf-8 -*-
# -*- coding: utf-8 -*-
from ronglian_sms_sdk import SmsSDK
accId = '8a216da8757784cd01758d5dce960962'
accToken = '384f442ed2e54edab1bae52d5ab7702f'
appId = '8a216da8757784cd01758d5dcf630969'
def send_message(phone, datas):
sdk = SmsSDK(accId, accToken, appId)
tid = '1'
# 测试模板id为: 1. 内容为: 【云通讯】您的验证码是{1},请于{2}分钟内正确输 入。
# mobile = '13303479527'
# datas = ('666777', '5')
# 模板中的参数按照位置传递
resp = sdk.sendMessage(tid, phone, datas)
return resp
3.2.3 配置路由
1)子路由verifications/urls.py
from django.urls import path
from . import views
urlpatterns = [
path('image_codes/', views.ImageCodeView.as_view()),
path('sms_codes/', views.SmsCodeView.as_view()),
]
2) 主路由urls.py
from django.contrib import admin
from django.urls import path, include
urlpatterns = [
path('admin/', admin.site.urls),
path('user/', include('apps.user.urls')),
path('verify/', include('apps.verifications.urls')),
]
3.3.4 views.py
from django.shortcuts import render
# Create your views here.
from django.http import HttpResponse, HttpResponseForbidden
from django.views import View
from django_redis import get_redis_connection
# from libs.captcha.captcha import captcha
from captcha.captcha import captcha
class ImageCodeView(View):
def get(self, request):
# 1.接收数据
uuid = request.GET.get('uuid')
# 2.校验数据
if not uuid:
return HttpResponseForbidden('uuid无效')
# 3.处理业务
# 获取图片文本内容和图片二进制代码
text, image = captcha.generate_captcha()
# 4.把uuid和图片文本存入redis
redis_client = get_redis_connection('img_code') # 获取redis客户端
# 5.写入redis(是字符串)
redis_client.setex(uuid, 60 * 5, text)
# 6.返回响应图片
return HttpResponse(image, content_type='image/jpg')
from rest_framework.permissions import AllowAny
from rest_framework.views import APIView
from rest_framework.response import Response
import re
import random
from utils.rl_sms import send_message
class SmsCodeView(APIView):
# 1. 所有人可以访问
permission_classes = (AllowAny,)
def post(self, request):
# 1. 获取参数
phone = request.data.get('phone')
image_code = request.data.get('image_code')
image_code_uuid = request.data.get('image_code_uuid')
# 2. 检查参数
if not all([phone, image_code, image_code_uuid]):
return Response({"code": 999, "msg": "参数不全"})
if not re.match(r'^1[3456789]\d{9}$', phone):
return Response({"code": 999, "msg": "手机号码不正确"})
# 3. 检查是否发送
redis_client = get_redis_connection('img_code')
phone_exists = redis_client.get(phone)
if phone_exists:
return Response({"code": 999, "msg": "频繁发送, 请稍后再试"})
redis_image_code = redis_client.get(image_code_uuid)
if redis_image_code:
# bytes 转成 string
redis_image_code = redis_image_code.decode()
# 比较用户提供的图片内容是否和redis中保存的一致
if image_code.upper() != redis_image_code:
return Response({'code': 999, 'msg': '图片验证码不正确'})
# 4. 发送
code = '%06d' % random.randint(0, 999999) # 随机6位验证码
send_resp = send_message(phone, (code, "5"))
# 5.1 保存code 到 redis中
redis_client.setex(phone, 60 * 5, code) # phone:code, 5分钟有效期 #
# 5.2 从redis中删除这个图片验证码, 以防再次被使用
# redis_client.delete(image_code_uuid)
# 5.3 使用 pipeline 批量操作
pl = redis_client.pipeline()
pl.setex(phone, 60 * 5, code)
pl.delete(image_code_uuid)
pl.execute()
# 6. 返回结果
return Response({"code": 0, "msg": "短信发送成功"})
3.3 展示效果
3.3.1 vue页面效果
3.3.2 redis效果
root@dev:~# whereis redis
redis: /etc/redis
root@dev:redis# redis-cli
127.0.0.1:6379> keys *
(empty list or set)
127.0.0.1:6379> select 2
OK
127.0.0.1:6379[2]> keys *
1) "58da4b1e-e60d-4699-aca2-938153f248eb"
2) "13321181528"
127.0.0.1:6379[2]>