upload project source code
This commit is contained in:
139
后端源码/yifan.action-ai.cn/app/utils/captcha_util.py
Normal file
139
后端源码/yifan.action-ai.cn/app/utils/captcha_util.py
Normal file
@@ -0,0 +1,139 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
|
||||
import base64
|
||||
import random
|
||||
import string
|
||||
from io import BytesIO
|
||||
from typing import Tuple
|
||||
from PIL import Image, ImageDraw, ImageFont
|
||||
|
||||
from app.config.setting import settings
|
||||
|
||||
|
||||
class CaptchaUtil:
|
||||
"""
|
||||
验证码工具类
|
||||
"""
|
||||
@classmethod
|
||||
def generate_captcha(cls) -> Tuple[str, str]:
|
||||
"""
|
||||
生成带有噪声和干扰的验证码图片(4位随机字符)。
|
||||
|
||||
返回:
|
||||
- Tuple[str, str]: [base64图片字符串, 验证码值]。
|
||||
"""
|
||||
# 生成4位随机验证码
|
||||
chars = string.digits + string.ascii_letters
|
||||
captcha_value = ''.join(random.sample(chars, 4))
|
||||
|
||||
# 创建一张随机颜色背景的图片
|
||||
width, height = 160, 60
|
||||
background_color = tuple(random.randint(230, 255) for _ in range(3))
|
||||
image = Image.new('RGB', (width, height), color=background_color)
|
||||
draw = ImageDraw.Draw(image)
|
||||
|
||||
# 使用指定字体
|
||||
font = ImageFont.truetype(font=settings.CAPTCHA_FONT_PATH, size=settings.CAPTCHA_FONT_SIZE)
|
||||
|
||||
# 计算文本总宽度和高度
|
||||
total_width = sum(draw.textbbox((0, 0), char, font=font)[2] for char in captcha_value)
|
||||
text_height = draw.textbbox((0, 0), captcha_value[0], font=font)[3]
|
||||
|
||||
# 计算起始位置,使文字居中
|
||||
x_start = (width - total_width) / 2
|
||||
y_start = (height - text_height) / 2 - draw.textbbox((0, 0), captcha_value[0], font=font)[1]
|
||||
|
||||
# 绘制字符
|
||||
x = x_start
|
||||
for char in captcha_value:
|
||||
# 使用深色文字,增加对比度
|
||||
text_color = tuple(random.randint(0, 80) for _ in range(3))
|
||||
|
||||
# 随机偏移,增加干扰
|
||||
x_offset = x + random.uniform(-2, 2)
|
||||
y_offset = y_start + random.uniform(-2, 2)
|
||||
|
||||
# 绘制字符
|
||||
draw.text((x_offset, y_offset), char, font=font, fill=text_color)
|
||||
|
||||
# 更新x坐标,增加字符间距的随机性
|
||||
x += draw.textbbox((0, 0), char, font=font)[2] + random.uniform(1, 5)
|
||||
|
||||
# 添加干扰线
|
||||
for _ in range(4):
|
||||
line_color = tuple(random.randint(150, 200) for _ in range(3))
|
||||
points = [(i, int(random.uniform(0, height))) for i in range(0, width, 20)]
|
||||
draw.line(points, fill=line_color, width=1)
|
||||
|
||||
# 添加随机噪点
|
||||
for _ in range(width * height // 60):
|
||||
point_color = tuple(random.randint(0, 255) for _ in range(3))
|
||||
draw.point(
|
||||
(random.randint(0, width), random.randint(0, height)),
|
||||
fill=point_color
|
||||
)
|
||||
|
||||
# 将图像数据保存到内存中并转换为base64
|
||||
buffer = BytesIO()
|
||||
image.save(buffer, format='PNG', optimize=True)
|
||||
base64_string = base64.b64encode(buffer.getvalue()).decode()
|
||||
|
||||
return base64_string, captcha_value
|
||||
|
||||
@classmethod
|
||||
def captcha_arithmetic(cls) -> Tuple[str, int]:
|
||||
"""
|
||||
创建验证码图片(加减乘运算)。
|
||||
|
||||
返回:
|
||||
- Tuple[str, int]: [base64图片字符串, 计算结果]。
|
||||
"""
|
||||
# 创建空白图像,使用随机浅色背景
|
||||
background_color = tuple(random.randint(230, 255) for _ in range(3))
|
||||
image = Image.new('RGB', (160, 60), color=background_color)
|
||||
draw = ImageDraw.Draw(image)
|
||||
|
||||
# 设置字体
|
||||
font = ImageFont.truetype(font=settings.CAPTCHA_FONT_PATH, size=settings.CAPTCHA_FONT_SIZE)
|
||||
|
||||
# 生成运算数字和运算符
|
||||
operators = ['+', '-', '*']
|
||||
operator = random.choice(operators)
|
||||
|
||||
# 对于减法,确保num1大于num2
|
||||
if operator == '-':
|
||||
num1 = random.randint(6, 10)
|
||||
num2 = random.randint(1, 5)
|
||||
else:
|
||||
num1 = random.randint(1, 9)
|
||||
num2 = random.randint(1, 9)
|
||||
|
||||
# 计算结果
|
||||
result_map = {
|
||||
'+': lambda x, y: x + y,
|
||||
'-': lambda x, y: x - y,
|
||||
'*': lambda x, y: x * y
|
||||
}
|
||||
captcha_value = result_map[operator](num1, num2)
|
||||
|
||||
# 绘制文本,使用深色增加对比度
|
||||
text = f'{num1} {operator} {num2} = ?'
|
||||
text_bbox = draw.textbbox((0, 0), text, font=font)
|
||||
text_width = text_bbox[2] - text_bbox[0]
|
||||
x = (160 - text_width) // 2
|
||||
draw.text((x, 15), text, fill=(0, 0, 139), font=font)
|
||||
|
||||
# 添加干扰线
|
||||
for _ in range(3):
|
||||
line_color = tuple(random.randint(150, 200) for _ in range(3))
|
||||
draw.line([
|
||||
(random.randint(0, 160), random.randint(0, 60)),
|
||||
(random.randint(0, 160), random.randint(0, 60))
|
||||
], fill=line_color, width=1)
|
||||
|
||||
# 将图像数据保存到内存中并转换为base64
|
||||
buffer = BytesIO()
|
||||
image.save(buffer, format='PNG', optimize=True)
|
||||
base64_string = base64.b64encode(buffer.getvalue()).decode()
|
||||
|
||||
return base64_string, captcha_value
|
||||
Reference in New Issue
Block a user