django-simple-captcha 验证码插件 和自己实现验证码的

来源:互联网 发布:一个c语言程序的编写 编辑:程序博客网 时间:2024/05/16 21:47

第三方验证码插件的使用

官方文档:http://django-simple-captcha.readthedocs.io/en/latest/usage.html#installationgithub:https://github.com/mbi/django-simple-captcha

django-simple-captcha 一个很简单的django验证组件,实现原理是数据库中生成key和随机字符,然后将key与sn存入数据库,用户提交时根据key去数据库中查询比对是否正确,没有使用常用的cookies session, 也挺简单,每个验证码都有过期时间,程序会自动清除过期的验证码

django 版本:1.9.1

安装pip install  django-simple-captcha==0.4.6
在setting 中配置INSTALLED_APPS = [    'django.contrib.admin',    'django.contrib.auth',    'django.contrib.contenttypes',    'django.contrib.sessions',    'django.contrib.messages',    'django.contrib.staticfiles',    'captcha',]
url中的配置加入url(r'^captcha/', include('captcha.urls')),urlpatterns = [    url(r'^admin/', admin.site.urls),    url(r'^captcha/', include('captcha.urls')),]
数据库的迁移 makemigrationsmigrate
app下自定义froms.py文件,创建一个注册formfrom django import formsfrom captcha.fields import CaptchaFieldclass RegisterForm(forms.Form):    email = forms.EmailField(required=True)    password = forms.CharField(required=True, min_length=5)    captcha = CaptchaField(error_messages={"invalid": u"验证码错误"})
视图函数中处理class RegisterView(View):    """注册视图"""    def get(self, request):        register_form = Registerform()  # 获取表单对象  然后返回前端        return render(request, 'register.html', {'register_form': register_form})    def post(self, request):        print(request.POST)  #打印下提交的数据        register_form = Registerform(request.POST)        if register_form.is_valid():            pass        else:            print(register_form.errors)
前端处理  在form表单中引用这个变量{{ register_form.captcha }}在用js实现点击验证码 就能进行切换支持大小写 在前端生成的内容 :               <label>验&nbsp;证&nbsp;码</label>                            <img src="/captcha/image/d8826580b3125692f6e2f55eef81d906cc0b5d8e/" alt="captcha" class="captcha" /> <input id="id_captcha_0" name="captcha_0" type="hidden" value="d8826580b3125692f6e2f55eef81d906cc0b5d8e" /> <input autocomplete="off" id="id_captcha_1" name="captcha_1" type="text" />

下面是自己实现的验证码

依赖pillow库生成验证码代码:# coding=utf-8import randomimport stringfrom PIL import Image, ImageDraw, ImageFilter, ImageFontimport io  # python3 中是io  pytho2 中是 stringIo# image 负责处理图片# ImageDraw 负责处理画笔# ImageFont 负责处理画笔# ImageFilter 负责处理 过滤镜def create_check():    # 1 创建画布    img = Image.new("RGB", (150, 50), (255, 255, 255))    """    第一个参数: 代表要采用RGB的颜色模式    第二个参数:代表图片的大小    第三个参数:具体的图片的颜色    """    # 2 创建画笔    draw = ImageDraw.Draw(img)    # 3 绘制线条和点    #  绘制线    for i in range(random.randint(1, 10)):        # for i in range(5):        draw.line(            # 在绘制线条的时候 是需要两个点确定一条直线的 每个点靠x y 两个值来确定位置的            [                (random.randint(1, 150), random.randint(1, 150)),  # 这个是一个 x 和 y 确定的坐标的位置                # 第二个x 和 要确定的 坐标的位置  两个坐标 确定了一条直线                (random.randint(1, 150), random.randint(1, 150))            ],            fill=(0, 0, 0)        )    # 开始绘制点    for j in range(1000):        draw.point(            [                random.randint(1, 150),                random.randint(1, 150)            ],            fill=(0, 0, 0)        )    # 4 绘制文字    str_temp = string.ascii_letters  # 生成大小写字母    digtial_temp = string.digits  # 生成数字  0-9    font_list = list(str_temp + digtial_temp)  # 组合成一个list列表    c_chars = "".join(random.sample(font_list, 5))  # 从指定的列表中随机的返回固定的长度的字符    # 5 绘制字体    font = ImageFont.truetype("simsun", 32)  # 这个字体是怎么设置的????? 需要使用什么样的 字体    draw.text((5, 5), c_chars, font=font, fill="green")    """    第一个参数:代表文字的位置  距离上和左的距离    第二个参数:代表文字的内容    第三个参数:代表文字的字体和大小    第四个参数:代表字体的颜色    """    # 5 定义扭曲的参数    params = [1 - float(random.randint(1, 2)) / 100,              0,              0,              0,              1 - float(random.randint(1, 2)) / 100,              float(random.randint(1, 2)) / 500,              0.001,  # 总体的比列              float(random.randint(1, 1)) / 500,              ]    # 6 使用滤镜 添加滤镜    img = img.transform((150, 50), Image.PERSPECTIVE, params)    # 扭曲的范围   扭曲的样式   扭曲的参数    # 进行扭曲    img = img.filter(ImageFilter.EDGE_ENHANCE_MORE)    # 展示画布    # img.save("demo.jpg")  # 保存文件   返回图片的内容 然后把内容保存在内存中 这里就不再在保存在本地了    # img.show()  # 这个是展示生成的图片的    return img, c_chars  # 给调用者返回这个img  也可以不用返回  直接生成在内存中

在视图函数中调用验证码

# 导入工具包中的生成图片的方法    from utils import check_codeimport io  # 可以把图片保存咋内存中# 调用生成验证码的方法 并把验证码存入内存中  返回给前端def tupian(request):    """获取验证码图片"""    img, c_chars = check_code.create_check()    # 上面调用了生成图片的函数 返回了生成图片的对象 和生成图片上面的文字    stream = io.BytesIO()  # 返回一个内存的字节流对象    img.save(stream, 'png')  # 把图片保存在内存中 然后 直接从内存中读取返回    # 把生成图片的字母保存在session    request.session['c_chars'] = c_chars    return HttpResponse(stream.getvalue())  # 直接去内存中 取出对象 然后进行返回# 验证提交上来的验证码    def yanzhengma(request):    """返回验证码表单"""    if request.method == 'GET':        return render(request, "验证码.html")    elif request.method == 'POST':        yanzhengma = request.POST.get("yanzhengma")        # 获取session 中保存的验证码的信息  然后和用户提交上来的进行对比         c_chars = request.session['c_chars']        # 把session中的 和获取到的字符串全部 变成 小写 来进行对比        yanzhengma = yanzhengma.lower()        c_chars = c_chars.lower()        if yanzhengma == c_chars:            return HttpResponse("登陆成功".encode())        else:            return HttpResponse("登陆失败".encode())

前端处理

<body>    <h3>验证码表单</h3>    <div class="container">        <form action="/login/yanzhengma/" method="POST">            <input type="text" name="yanzhengma" id="yanzhengma">            <input type="file" name="file" id="file">            <input type="submit" id="submit">        </form>    </div>    <div id="see">        <h5>验证码</h5>        {% comment %}在这里请求图片的时候就会返回这个视图函数读取的内容{% endcomment %}        <img src="/login/tupian/" alt="" onclick="change_yanzhengma(this)">    </div>    <script>        //点击图片换验证码        function change_yanzhengma(ths) {            //点击图片进行切换 给img加上一个点击事件  然后            // 然后去重新给这个src 来赋值  获取原来的再加上个?            //动态的去改变URL的地址            ths.src = ths.src + "?";        }    </script></body>