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

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.1 注册容联云账号

3.1.1 注册账号
https://www.yuntongxun.com/user/login

 

3.1.2 重要数据

 

 

 

3.1.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]>

 

posted @ 2020-11-03 21:57  狐狸大大爱吃糖  阅读(263)  评论(0编辑  收藏  举报