Django自定義插件實(shí)現(xiàn)網(wǎng)站登錄驗(yàn)證碼功能
前言
網(wǎng)站登錄的時(shí)候我們常常會(huì)看到隨機(jī)的驗(yàn)證碼需要輸入后臺(tái)驗(yàn)證,如圖:
現(xiàn)在我們來(lái)實(shí)現(xiàn)在Django中通過自定制插件來(lái)實(shí)現(xiàn)隨機(jī)驗(yàn)證
check_code.py
基于PIL生成一個(gè)帶驗(yàn)證碼的圖片和驗(yàn)證碼,生成驗(yàn)證碼圖片需要Monaco.ttf字體(重要),可按自己要求更改check_code中的字體和字體文件位置
#!/usr/bin/env python # -*- coding:utf-8 -*- import random from PIL import Image, ImageDraw, ImageFont, ImageFilter _letter_cases = "abcdefghjkmnpqrstuvwxy" # 小寫字母,去除可能干擾的i,l,o,z _upper_cases = _letter_cases.upper() # 大寫字母 _numbers = ''.join(map(str, range(3, 10))) # 數(shù)字 init_chars = ''.join((_letter_cases, _upper_cases, _numbers)) def create_validate_code(size=(120, 30), chars=init_chars, img_type="GIF", mode="RGB", bg_color=(255, 255, 255), fg_color=(0, 0, 255), font_size=18, font_type="Monaco.ttf", length=4, draw_lines=True, n_line=(1, 2), draw_points=True, point_chance=2): """ @todo: 生成驗(yàn)證碼圖片 @param size: 圖片的大小,格式(寬,高),默認(rèn)為(120, 30) @param chars: 允許的字符集合,格式字符串 @param img_type: 圖片保存的格式,默認(rèn)為GIF,可選的為GIF,JPEG,TIFF,PNG @param mode: 圖片模式,默認(rèn)為RGB @param bg_color: 背景顏色,默認(rèn)為白色 @param fg_color: 前景色,驗(yàn)證碼字符顏色,默認(rèn)為藍(lán)色#0000FF @param font_size: 驗(yàn)證碼字體大小 @param font_type: 驗(yàn)證碼字體,默認(rèn)為 ae_AlArabiya.ttf @param length: 驗(yàn)證碼字符個(gè)數(shù) @param draw_lines: 是否劃干擾線 @param n_lines: 干擾線的條數(shù)范圍,格式元組,默認(rèn)為(1, 2),只有draw_lines為True時(shí)有效 @param draw_points: 是否畫干擾點(diǎn) @param point_chance: 干擾點(diǎn)出現(xiàn)的概率,大小范圍[0, 100] @return: [0]: PIL Image實(shí)例 @return: [1]: 驗(yàn)證碼圖片中的字符串 """ width, height = size # 寬高 # 創(chuàng)建圖形 img = Image.new(mode, size, bg_color) draw = ImageDraw.Draw(img) # 創(chuàng)建畫筆 def get_chars(): """生成給定長(zhǎng)度的字符串,返回列表格式""" return random.sample(chars, length) def create_lines(): """繪制干擾線""" line_num = random.randint(*n_line) # 干擾線條數(shù) for i in range(line_num): # 起始點(diǎn) begin = (random.randint(0, size[0]), random.randint(0, size[1])) # 結(jié)束點(diǎn) end = (random.randint(0, size[0]), random.randint(0, size[1])) draw.line([begin, end], fill=(0, 0, 0)) def create_points(): """繪制干擾點(diǎn)""" chance = min(100, max(0, int(point_chance))) # 大小限制在[0, 100] for w in range(width): for h in range(height): tmp = random.randint(0, 100) if tmp > 100 - chance: draw.point((w, h), fill=(0, 0, 0)) def create_strs(): """繪制驗(yàn)證碼字符""" c_chars = get_chars() strs = ' %s ' % ' '.join(c_chars) # 每個(gè)字符前后以空格隔開 font = ImageFont.truetype(font_type, font_size) font_width, font_height = font.getsize(strs) draw.text(((width - font_width) / 3, (height - font_height) / 3), strs, font=font, fill=fg_color) return ''.join(c_chars) if draw_lines: create_lines() if draw_points: create_points() strs = create_strs() # 圖形扭曲參數(shù) params = [1 - float(random.randint(1, 2)) / 100, 0, 0, 0, 1 - float(random.randint(1, 10)) / 100, float(random.randint(1, 2)) / 500, 0.001, float(random.randint(1, 2)) / 500 ] img = img.transform(size, Image.PERSPECTIVE, params) # 創(chuàng)建扭曲 img = img.filter(ImageFilter.EDGE_ENHANCE_MORE) # 濾鏡,邊界加強(qiáng)(閾值更大) return img, strs
創(chuàng)建urls和views
views.py:
# 將check_code包放在合適的位置,導(dǎo)入即可,我是放在utils下面 from utils import check_code def create_code_img(request): f = BytesIO() #直接在內(nèi)存開辟一點(diǎn)空間存放臨時(shí)生成的圖片 img, code = check_code.create_validate_code() #調(diào)用check_code生成照片和驗(yàn)證碼 request.session['check_code'] = code #將驗(yàn)證碼存在服務(wù)器的session中,用于校驗(yàn) img.save(f,'PNG') #生成的圖片放置于開辟的內(nèi)存中 return HttpResponse(f.getvalue()) #將內(nèi)存的數(shù)據(jù)讀取出來(lái),并以HttpResponse返回
urls我的設(shè)置:url(r'^create_code_img/', views.create_code_img)
前端應(yīng)用驗(yàn)證碼和點(diǎn)擊自動(dòng)刷新
html:
<div class="row"> <div class="col-xs-7"> <input type="text" class="form-control" name="check_code" id="check_code" placeholder="請(qǐng)輸入驗(yàn)證碼"> </div> <div class="col-xs-5"> <img id="check_code_img" src="/create_code_img/" onclick="refresh_check_code(this)"> {# src是url路徑,可得到驗(yàn)證碼圖片,點(diǎn)擊時(shí)調(diào)用refresh_check_code#} </div><br></div>
javascript:
<script> function refresh_check_code(ths) { ths.src += '?'; {# src后面加問好會(huì)自動(dòng)刷新驗(yàn)證碼img的src#} } </script>
login的Views
login的Views進(jìn)行數(shù)據(jù)驗(yàn)證,然后做相應(yīng)的處理
post_check_code = request.POST.get('check_code') session_check_code = request.session['check_code'] if post_check_code.lower() == session_check_code.lower() : pass
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
python獲取代碼運(yùn)行時(shí)間的實(shí)例代碼
今天小編就為大家分享一篇python獲取代碼運(yùn)行時(shí)間的實(shí)例代碼,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來(lái)看看吧2018-06-06python編寫學(xué)生成績(jī)管理系統(tǒng)的邏輯結(jié)構(gòu)及功能實(shí)現(xiàn)
這篇文章主要為大家介紹了python編寫學(xué)生成績(jī)管理系統(tǒng)實(shí)現(xiàn)八個(gè)功能示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-04-04Python機(jī)器學(xué)習(xí)之Kmeans基礎(chǔ)算法
這篇文章主要介紹了Python機(jī)器學(xué)習(xí)之Kmeans基礎(chǔ)算法,文中有非常詳細(xì)的代碼示例,對(duì)正在學(xué)習(xí)python的小伙伴們有非常好的幫助,需要的朋友可以參考下2021-04-04python3.6之xlwt如何設(shè)置單元格對(duì)齊方式
這篇文章主要介紹了python3.6之xlwt如何設(shè)置單元格對(duì)齊方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-06-06Requests什么的通通爬不了的Python超強(qiáng)反爬蟲方案!
今天帶大家學(xué)習(xí)Requests什么的通通爬不了的Python超強(qiáng)反爬蟲方案,文中有非常詳細(xì)的圖文介紹及代碼示例,對(duì)正在學(xué)習(xí)python的小伙伴們有很好的幫助,需要的朋友可以參考下2021-05-05Python NumPy數(shù)組利器之np.zeros函數(shù)詳解與應(yīng)用實(shí)例
在Python的科學(xué)計(jì)算庫(kù)NumPy中,numpy.zeros()是一個(gè)非常重要的函數(shù),它用于創(chuàng)建一個(gè)指定形狀和數(shù)據(jù)類型的全零數(shù)組,這篇文章主要給大家介紹了關(guān)于Python NumPy數(shù)組利器之np.zeros函數(shù)詳解與應(yīng)用實(shí)例的相關(guān)資料,需要的朋友可以參考下2024-06-06Python對(duì)FTP交互封裝的實(shí)現(xiàn)
本文主要介紹了Python對(duì)FTP交互封裝的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-06-06詳解pyqt5的UI中嵌入matplotlib圖形并實(shí)時(shí)刷新(挖坑和填坑)
這篇文章主要介紹了詳解pyqt5的UI中嵌入matplotlib圖形并實(shí)時(shí)刷新(挖坑和填坑),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08