Django之验证码的生成和使用
1、基于PIL生成一个带验证码的图片和验证码,生成验证码图片需要Monaco.ttf字体,可按自己要求更改check_code中的字体和字体文件位置,如下图
1 #!/usr/bin/env python 2 # -*- coding:utf-8 -*- 3 4 import random 5 from PIL import Image, ImageDraw, ImageFont, ImageFilter 6 7 _letter_cases = "abcdefghjkmnpqrstuvwxy" # 小写字母,去除可能干扰的i,l,o,z 8 _upper_cases = _letter_cases.upper() # 大写字母 9 _numbers = ''.join(map(str, range(3, 10))) # 数字 10 init_chars = ''.join((_letter_cases, _upper_cases, _numbers)) 11 12 13 def create_validate_code(size=(120, 30), 14 chars=init_chars, 15 img_type="GIF", 16 mode="RGB", 17 bg_color=(255, 255, 255), 18 fg_color=(0, 0, 255), 19 font_size=18, 20 font_type="Monaco.ttf", 21 length=4, 22 draw_lines=True, 23 n_line=(1, 2), 24 draw_points=True, 25 point_chance=2): 26 """ 27 @todo: 生成验证码图片 28 @param size: 图片的大小,格式(宽,高),默认为(120, 30) 29 @param chars: 允许的字符集合,格式字符串 30 @param img_type: 图片保存的格式,默认为GIF,可选的为GIF,JPEG,TIFF,PNG 31 @param mode: 图片模式,默认为RGB 32 @param bg_color: 背景颜色,默认为白色 33 @param fg_color: 前景色,验证码字符颜色,默认为蓝色#0000FF 34 @param font_size: 验证码字体大小 35 @param font_type: 验证码字体,默认为 ae_AlArabiya.ttf 36 @param length: 验证码字符个数 37 @param draw_lines: 是否划干扰线 38 @param n_lines: 干扰线的条数范围,格式元组,默认为(1, 2),只有draw_lines为True时有效 39 @param draw_points: 是否画干扰点 40 @param point_chance: 干扰点出现的概率,大小范围[0, 100] 41 @return: [0]: PIL Image实例 42 @return: [1]: 验证码图片中的字符串 43 """ 44 45 width, height = size # 宽高 46 # 创建图形 47 img = Image.new(mode, size, bg_color) 48 draw = ImageDraw.Draw(img) # 创建画笔 49 50 def get_chars(): 51 """生成给定长度的字符串,返回列表格式""" 52 return random.sample(chars, length) 53 54 def create_lines(): 55 """绘制干扰线""" 56 line_num = random.randint(*n_line) # 干扰线条数 57 58 for i in range(line_num): 59 # 起始点 60 begin = (random.randint(0, size[0]), random.randint(0, size[1])) 61 # 结束点 62 end = (random.randint(0, size[0]), random.randint(0, size[1])) 63 draw.line([begin, end], fill=(0, 0, 0)) 64 65 def create_points(): 66 """绘制干扰点""" 67 chance = min(100, max(0, int(point_chance))) # 大小限制在[0, 100] 68 69 for w in range(width): 70 for h in range(height): 71 tmp = random.randint(0, 100) 72 if tmp > 100 - chance: 73 draw.point((w, h), fill=(0, 0, 0)) 74 75 def create_strs(): 76 """绘制验证码字符""" 77 c_chars = get_chars() 78 strs = ' %s ' % ' '.join(c_chars) # 每个字符前后以空格隔开 79 80 font = ImageFont.truetype(font_type, font_size) 81 font_width, font_height = font.getsize(strs) 82 83 draw.text(((width - font_width) / 3, (height - font_height) / 3), 84 strs, font=font, fill=fg_color) 85 86 return ''.join(c_chars) 87 88 if draw_lines: 89 create_lines() 90 if draw_points: 91 create_points() 92 strs = create_strs() 93 94 # 图形扭曲参数 95 params = [1 - float(random.randint(1, 2)) / 100, 96 0, 97 0, 98 0, 99 1 - float(random.randint(1, 10)) / 100, 100 float(random.randint(1, 2)) / 500, 101 0.001, 102 float(random.randint(1, 2)) / 500 103 ] 104 img = img.transform(size, Image.PERSPECTIVE, params) # 创建扭曲 105 106 img = img.filter(ImageFilter.EDGE_ENHANCE_MORE) # 滤镜,边界加强(阈值更大) 107 108 return img, strs
2、创建urls和views,请按自己需求创建
1 # 将check_code包放在合适的位置,导入即可,我是放在utils下面 2 from utils import check_code 3 4 def create_code_img(request): 5 f = BytesIO() #直接在内存开辟一点空间存放临时生成的图片 6 7 img, code = check_code.create_validate_code() #调用check_code生成照片和验证码 8 request.session['check_code'] = code #将验证码存在服务器的session中,用于校验 9 img.save(f,'PNG') #生成的图片放置于开辟的内存中 10 return HttpResponse(f.getvalue()) #将内存的数据读取出来,并以HttpResponse返回
urls我的设置:url(r'^create_code_img/', views.create_code_img)
3、前端应用验证码和点击自动刷新
1 <div class="row"> 2 <div class="col-xs-7"> 3 <input type="text" class="form-control" name="check_code" id="check_code" placeholder="请输入验证码"> 4 </div> 5 <div class="col-xs-5"> 6 <img id="check_code_img" src="/create_code_img/" onclick="refresh_check_code(this)"> 7 {# src是url路径,可得到验证码图片,点击时调用refresh_check_code#} 8 </div> 9 </div>
1 <script> 2 function refresh_check_code(ths) { 3 ths.src += '?'; 4 {# src后面加问好会自动刷新验证码img的src#} 5 } 6 </script>
4、login的Views进行数据验证,然后做相应的处理
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
人,从刚出生来到这个世界,便开始探索这个世界。累了就歇会,精神了就继续探索,直至死亡。