OnlineJudge/utils/captcha/__init__.py

101 lines
3.6 KiB
Python
Raw Normal View History

2015-09-09 07:50:36 +00:00
"""
2017-08-20 12:41:48 +00:00
Copyright 2013 TY<tianyu0915@gmail.com>
2015-09-09 07:50:36 +00:00
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
"""
import os
2017-08-20 12:41:48 +00:00
import time
2015-09-09 07:50:36 +00:00
import random
2017-08-20 12:41:48 +00:00
from PIL import Image, ImageDraw, ImageFont
2015-09-09 07:50:36 +00:00
2017-08-20 00:35:59 +00:00
class Captcha(object):
2015-09-09 07:50:36 +00:00
def __init__(self, request):
2017-01-23 08:25:14 +00:00
"""
2017-08-20 12:41:48 +00:00
初始化,设置各种属性
"""
2015-09-09 07:50:36 +00:00
self.django_request = request
2017-08-20 00:35:59 +00:00
self.session_key = "_django_captcha_key"
2017-08-20 12:41:48 +00:00
self.captcha_expires_time = "_django_captcha_expires_time"
2015-09-09 07:50:36 +00:00
2017-08-20 12:41:48 +00:00
# 验证码图片尺寸
self.img_width = 90
2015-09-09 07:50:36 +00:00
self.img_height = 30
2017-08-20 12:41:48 +00:00
def _get_font_size(self, code):
"""
将图片高度的80%作为字体大小
"""
2015-09-09 07:50:36 +00:00
s1 = int(self.img_height * 0.8)
2017-08-20 12:41:48 +00:00
s2 = int(self.img_width / len(code))
2017-08-20 00:35:59 +00:00
return int(min((s1, s2)) + max((s1, s2)) * 0.05)
2015-09-09 07:50:36 +00:00
2017-08-18 22:10:48 +00:00
def _set_answer(self, answer):
2017-08-20 12:41:48 +00:00
"""
设置答案和过期时间
"""
2017-08-18 22:10:48 +00:00
self.django_request.session[self.session_key] = str(answer)
2017-08-20 12:41:48 +00:00
self.django_request.session[self.captcha_expires_time] = time.time() + 60
2015-09-09 07:50:36 +00:00
2017-08-20 12:41:48 +00:00
def _make_code(self):
"""
生成随机数或随机字符串
"""
string = random.sample("abcdefghkmnpqrstuvwxyzABCDEFGHGKMNOPQRSTUVWXYZ23456789", 4)
self._set_answer("".join(string))
return string
2017-08-18 22:10:48 +00:00
def get(self):
"""
2017-08-20 12:41:48 +00:00
生成验证码图片,返回值为图片的bytes
"""
background = (random.randrange(200, 255), random.randrange(200, 255), random.randrange(200, 255))
code_color = (random.randrange(0, 50), random.randrange(0, 50), random.randrange(0, 50), 255)
2015-09-09 07:50:36 +00:00
2017-08-20 12:41:48 +00:00
font_path = os.path.join(os.path.normpath(os.path.dirname(__file__)), "timesbi.ttf")
2017-08-18 22:10:48 +00:00
2017-08-20 12:41:48 +00:00
image = Image.new("RGB", (self.img_width, self.img_height), background)
code = self._make_code()
font_size = self._get_font_size(code)
draw = ImageDraw.Draw(image)
2017-08-18 22:10:48 +00:00
2017-08-20 12:41:48 +00:00
# x是第一个字母的x坐标
x = random.randrange(int(font_size * 0.3), int(font_size * 0.5))
2017-08-18 22:10:48 +00:00
2017-08-20 12:41:48 +00:00
for i in code:
# 字符y坐标
y = random.randrange(1, 7)
# 随机字符大小
font = ImageFont.truetype(font_path.replace("\\", "/"), font_size + random.randrange(-3, 7))
draw.text((x, y), i, font=font, fill=code_color)
# 随机化字符之间的距离 字符粘连可以降低识别率
x += font_size * random.randrange(6, 8) / 10
2017-08-18 22:10:48 +00:00
2017-08-20 12:41:48 +00:00
self.django_request.session[self.session_key] = "".join(code)
return image
2017-08-18 22:10:48 +00:00
2017-08-20 12:41:48 +00:00
def check(self, code):
"""
检查用户输入的验证码是否正确
2017-08-18 22:10:48 +00:00
"""
2017-08-20 00:35:59 +00:00
_code = self.django_request.session.get(self.session_key) or ""
2017-08-20 12:41:48 +00:00
if not _code:
return False
expires_time = self.django_request.session.get(self.captcha_expires_time) or 0
# 注意 如果验证之后不清除之前的验证码的话 可能会造成重复验证的现象
del self.django_request.session[self.session_key]
del self.django_request.session[self.captcha_expires_time]
if _code.lower() == str(code).lower() and time.time() < expires_time:
return True
else:
return False