圖片驗證碼的生成
阿新 • • 發佈:2020-12-25
圖片驗證碼的生成
1.django 快取設定
1.1 安裝Django快取模組
pip install django-redis==4.12.1
1.2 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.新建應用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())
]
3.圖形驗證碼captcha使用
1.下載captcha壓縮包captcha.zip,放到專案packages資料夾下
2.解壓captcha.zip放到syl/libs資料夾下
3.解壓檔案中的syl/libs/captcha/captcha.py 右鍵執行即可生成圖片驗證碼
unzip xxx.zip
4.在verifications/views.py中使用
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)
uuid = request.GET.get('uuid')
# 2.校驗資料
if not uuid:
return HttpResponseForbidden('uuid無效')
# 3.處理業務
# 獲取圖片文字內容和圖片二進位制程式碼
text, image = captcha.generate_captcha() # text=GPMZ; image=圖片
# 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')
5.測試驗證碼介面
http://192.168.56.100:8888/verify/image_codes/?uuid=66ea64aa-fbe6-11ea-a3d3-005056c00008
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"
阿里雲簡訊服務的使用
申請簡訊服務,領取免費簡訊,申請簽名和模板
[外鏈圖片轉存失敗,源站可能有防盜鏈機制,建議將圖片儲存下來直接上傳(img-Ervjy6rw-1608810993222)(/Users/zhangpengpeng/jiyunedu/md文件彙總/小實訓課件/2002B授課筆記/pic/簡訊領取.png)]
[外鏈圖片轉存失敗,源站可能有防盜鏈機制,建議將圖片儲存下來直接上傳(img-Zoe4KFzG-1608810993226)(/Users/zhangpengpeng/jiyunedu/md文件彙總/小實訓課件/2002B授課筆記/pic/簽名和模板.png)]
[外鏈圖片轉存失敗,源站可能有防盜鏈機制,建議將圖片儲存下來直接上傳(img-sD7ZApwS-1608810993228)(/Users/zhangpengpeng/jiyunedu/md文件彙總/小實訓課件/2002B授課筆記/pic/添加簽名和模板.png)]
- 封裝發簡訊程式碼utils/MyBaseView.py
from aliyunsdkcore.client import AcsClient
from aliyunsdkcore.request import CommonRequest
from code2001B.settings import ALY_ACCESSKEY_ID,ALY_ACCESSKEY_SECRET,QINIU_ID,QINIU_SECRET,SHAZFB_APPID,SHA_RETURN_URL,SHA_URL
def send_sms(phone,data):
client = AcsClient(ALY_ACCESSKEY_ID, ALY_ACCESSKEY_SECRET, 'cn-hangzhou')
#data ={ "code" : 123456 }
request = CommonRequest()
request.set_accept_format('json')
request.set_domain('dysmsapi.aliyuncs.com')
request.set_method('POST')
request.set_protocol_type('https') # https | http
request.set_version('2017-05-25')
request.set_action_name('SendSms')
request.add_query_param('RegionId', "cn-hangzhou")
request.add_query_param('PhoneNumbers', phone)
request.add_query_param('SignName', "美多商城")
request.add_query_param('TemplateCode', "SMS_185212884")
request.add_query_param('TemplateParam', data)
response1 = client.do_action(request)
# python2: print(response)
res = json.loads(str(response1, encoding='utf-8'))
- 簡訊傳送和驗證的邏輯
from rest_framework.views import APIView
from rest_framework.response import Response
from libs.captcha.captcha import captcha
from django_redis import get_redis_connection
from django.http.response import HttpResponse
import random
from verificationsapp.tasks import send_sms
from django_redis import get_redis_connection
class SendSMSCode(APIView):
def post(self,request):
phone = request.data.get("phone")
image_code = request.data.get("image_code")
image_code_uuid = request.data.get("image_code_uuid")
print(phone)
print(image_code_uuid)
print(image_code)
if not all([phone,image_code,image_code_uuid]):
return Response({"code":4005,"msg":"引數不全"})
#先獲取redis 裡的圖片驗證碼來比對
redis_cli = get_redis_connection("img_code")
redis_img_code = redis_cli.get(image_code_uuid).decode()
print(redis_img_code)
print(image_code)
if image_code.lower() != redis_img_code.lower():
return Response({"code":4003,"msg":"引數錯誤"})
#傳送簡訊
num = random.randint(100000,999999)
print(num)
send_data = {"code":10086}
# send_sms.delay(phone,send_data)
# 刪除redis裡的image_code,儲存phone_code
#pipeline管道:作用就是把多個命令放在一起來執行
pl = redis_cli.pipeline()
pl.setex(phone,60*5,num)
pl.delete(image_code_uuid)
pl.execute()
return Response({"code":0,"msg":"傳送成功"})
pipeline原理
1.1 redis傳送資料原理
-
Redis是建立在TCP協議基礎上的CS架構,客戶端client對redis server採取請求響應的方式互動。
-
一般來說客戶端從提交請求到得到伺服器相應,需要傳送兩個tcp報文。
-
設想這樣的一個場景,你要批量的執行一系列redis命令,例如執行100次get key,這時你要向redis請求100次+獲取響應100次。如果能一次性將100個請求提交給redis server,執行完成之後批量的獲取相應,只需要向redis請求1次,然後批量執行完命令,一次性結果,效能是不是會好很多呢?
1.2 未使用pipeline執行N條命令
1.3 使用了pipeline執行N條命令